From 416376b36ebc0bb9449d98a38ddac47aedf2c502 Mon Sep 17 00:00:00 2001 From: Haesun Park Date: Wed, 17 Feb 2021 23:45:10 +0900 Subject: [PATCH] =?UTF-8?q?=EC=BD=94=EB=9E=A9,=20=ED=85=90=EC=84=9C?= =?UTF-8?q?=ED=94=8C=EB=A1=9C=202.4=EC=97=90=EC=84=9C=20=EC=8B=A4=ED=96=89?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- 11_training_deep_neural_networks.ipynb | 10667 +++++++++++++---------- 1 file changed, 5840 insertions(+), 4827 deletions(-) diff --git a/11_training_deep_neural_networks.ipynb b/11_training_deep_neural_networks.ipynb index 26ab2d5fc..470e5dc4f 100644 --- a/11_training_deep_neural_networks.ipynb +++ b/11_training_deep_neural_networks.ipynb @@ -1,4830 +1,5843 @@ { - "cells": [ - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "**11장 – 심층 신경망 훈련하기**" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "_이 노트북은 11장에 있는 모든 샘플 코드와 연습문제 해답을 가지고 있습니다._" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "\n", - " \n", - "
\n", - " 구글 코랩에서 실행하기\n", - "
" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "# 설정" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "먼저 몇 개의 모듈을 임포트합니다. 맷플롯립 그래프를 인라인으로 출력하도록 만들고 그림을 저장하는 함수를 준비합니다. 또한 파이썬 버전이 3.5 이상인지 확인합니다(파이썬 2.x에서도 동작하지만 곧 지원이 중단되므로 파이썬 3을 사용하는 것이 좋습니다). 사이킷런 버전이 0.20 이상인지와 텐서플로 버전이 2.0 이상인지 확인합니다." - ] - }, - { - "cell_type": "code", - "execution_count": 1, - "metadata": {}, - "outputs": [], - "source": [ - "# 파이썬 ≥3.5 필수\n", - "import sys\n", - "assert sys.version_info >= (3, 5)\n", - "\n", - "# 사이킷런 ≥0.20 필수\n", - "import sklearn\n", - "assert sklearn.__version__ >= \"0.20\"\n", - "\n", - "# 텐서플로 ≥2.0 필수\n", - "import tensorflow as tf\n", - "from tensorflow import keras\n", - "assert tf.__version__ >= \"2.0\"\n", - "\n", - "%load_ext tensorboard\n", - "\n", - "# 공통 모듈 임포트\n", - "import numpy as np\n", - "import os\n", - "\n", - "# 노트북 실행 결과를 동일하게 유지하기 위해\n", - "np.random.seed(42)\n", - "\n", - "# 깔끔한 그래프 출력을 위해\n", - "%matplotlib inline\n", - "import matplotlib as mpl\n", - "import matplotlib.pyplot as plt\n", - "mpl.rc('axes', labelsize=14)\n", - "mpl.rc('xtick', labelsize=12)\n", - "mpl.rc('ytick', labelsize=12)\n", - "\n", - "# 그림을 저장할 위치\n", - "PROJECT_ROOT_DIR = \".\"\n", - "CHAPTER_ID = \"deep\"\n", - "IMAGES_PATH = os.path.join(PROJECT_ROOT_DIR, \"images\", CHAPTER_ID)\n", - "os.makedirs(IMAGES_PATH, exist_ok=True)\n", - "\n", - "def save_fig(fig_id, tight_layout=True, fig_extension=\"png\", resolution=300):\n", - " path = os.path.join(IMAGES_PATH, fig_id + \".\" + fig_extension)\n", - " print(\"그림 저장:\", fig_id)\n", - " if tight_layout:\n", - " plt.tight_layout()\n", - " plt.savefig(path, format=fig_extension, dpi=resolution)" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "# 그레이디언트 소실과 폭주 문제" - ] - }, - { - "cell_type": "code", - "execution_count": 2, - "metadata": {}, - "outputs": [], - "source": [ - "def logit(z):\n", - " return 1 / (1 + np.exp(-z))" - ] - }, - { - "cell_type": "code", - "execution_count": 3, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "그림 저장: sigmoid_saturation_plot\n" - ] - }, - { - "data": { - "image/png": "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\n", - "text/plain": [ - "
" - ] - }, - "metadata": { - "needs_background": "light" - }, - "output_type": "display_data" - } - ], - "source": [ - "z = np.linspace(-5, 5, 200)\n", - "\n", - "plt.plot([-5, 5], [0, 0], 'k-')\n", - "plt.plot([-5, 5], [1, 1], 'k--')\n", - "plt.plot([0, 0], [-0.2, 1.2], 'k-')\n", - "plt.plot([-5, 5], [-3/4, 7/4], 'g--')\n", - "plt.plot(z, logit(z), \"b-\", linewidth=2)\n", - "props = dict(facecolor='black', shrink=0.1)\n", - "plt.annotate('Saturating', xytext=(3.5, 0.7), xy=(5, 1), arrowprops=props, fontsize=14, ha=\"center\")\n", - "plt.annotate('Saturating', xytext=(-3.5, 0.3), xy=(-5, 0), arrowprops=props, fontsize=14, ha=\"center\")\n", - "plt.annotate('Linear', xytext=(2, 0.2), xy=(0, 0.5), arrowprops=props, fontsize=14, ha=\"center\")\n", - "plt.grid(True)\n", - "plt.title(\"Sigmoid activation function\", fontsize=14)\n", - "plt.axis([-5, 5, -0.2, 1.2])\n", - "\n", - "save_fig(\"sigmoid_saturation_plot\")\n", - "plt.show()" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "## Xavier 초기화와 He 초기화" - ] - }, - { - "cell_type": "code", - "execution_count": 4, - "metadata": {}, - "outputs": [ - { - "data": { - "text/plain": [ - "['Constant',\n", - " 'GlorotNormal',\n", - " 'GlorotUniform',\n", - " 'HeNormal',\n", - " 'HeUniform',\n", - " 'Identity',\n", - " 'Initializer',\n", - " 'LecunNormal',\n", - " 'LecunUniform',\n", - " 'Ones',\n", - " 'Orthogonal',\n", - " 'RandomNormal',\n", - " 'RandomUniform',\n", - " 'TruncatedNormal',\n", - " 'VarianceScaling',\n", - " 'Zeros',\n", - " 'constant',\n", - " 'deserialize',\n", - " 'get',\n", - " 'glorot_normal',\n", - " 'glorot_uniform',\n", - " 'he_normal',\n", - " 'he_uniform',\n", - " 'identity',\n", - " 'lecun_normal',\n", - " 'lecun_uniform',\n", - " 'ones',\n", - " 'orthogonal',\n", - " 'random_normal',\n", - " 'random_uniform',\n", - " 'serialize',\n", - " 'truncated_normal',\n", - " 'variance_scaling',\n", - " 'zeros']" - ] - }, - "execution_count": 4, - "metadata": {}, - "output_type": "execute_result" - } - ], - "source": [ - "[name for name in dir(keras.initializers) if not name.startswith(\"_\")]" - ] - }, - { - "cell_type": "code", - "execution_count": 5, - "metadata": {}, - "outputs": [ - { - "data": { - "text/plain": [ - "" - ] - }, - "execution_count": 5, - "metadata": {}, - "output_type": "execute_result" - } - ], - "source": [ - "keras.layers.Dense(10, activation=\"relu\", kernel_initializer=\"he_normal\")" - ] - }, - { - "cell_type": "code", - "execution_count": 6, - "metadata": {}, - "outputs": [ - { - "data": { - "text/plain": [ - "" - ] - }, - "execution_count": 6, - "metadata": {}, - "output_type": "execute_result" - } - ], - "source": [ - "init = keras.initializers.VarianceScaling(scale=2., mode='fan_avg',\n", - " distribution='uniform')\n", - "keras.layers.Dense(10, activation=\"relu\", kernel_initializer=init)" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "## 수렴하지 않는 활성화 함수" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "### LeakyReLU" - ] - }, - { - "cell_type": "code", - "execution_count": 7, - "metadata": {}, - "outputs": [], - "source": [ - "def leaky_relu(z, alpha=0.01):\n", - " return np.maximum(alpha*z, z)" - ] - }, - { - "cell_type": "code", - "execution_count": 8, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "그림 저장: leaky_relu_plot\n" - ] - }, - { - "data": { - "image/png": "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EXLPOSfoRCKxUYEKGTJkCGeddVbQMUTqpbjYj+20aZPvMeK224JOJBI7FSiRFFFRAYMH+z2oHj1gyhQwCzqVSOxUoCJQXFxMfn4+e+21F61bt+b444+nsLBw5/M//PADF110ER06dKBly5Z0796dp59+utbXXLhwIW3atGHChAnxji9p4g9/8Oee2rXzjSKysoJOJFI/KlB1cM5x5plnsmbNGubPn88nn3xCv379OPHEE/n2228BKCsro0+fPsyfP58lS5Zw/fXXM2zYMBYuXFjta86aNYvzzz+fiRMncpW6kpYGMGOGv8apaVOYORMOOCDoRCL116Aj6qaiN998k8WLF7N+/XpatmwJwOjRo3nppZeYOnUqN910E/vttx8jRozY+Tv5+fn87W9/47nnnuOkk07a5fUmTpzIiBEjmDVrFqecckqjvhdJTYsXw29/66cfegiqbHIiSUsFqg4fffQRpaWl7Lnnnrs8XlZWxvLlywGoqKhgzJgxzJgxgzVr1rB161bKy8t3G2J57ty5PPHEE7z99tscffTRjfUWJIWtX+8bRWzZAkOGwLXXBp1IpOGoQNWhsrKS3Nxc3nnnnd2ey87OBmDs2LE8+OCDjBs3jh49epCVlcWtt97Kd999t8v8PXv25LPPPmPy5MkcddRRmM5gSz1s2wYXXABffw2/+IXvrVyblKQSFag69OnTh3Xr1tGkSRM6d+5c7TyLFi3i7LPP5pJLLgH8easvvviCNm3a7DLfAQccwCOPPEJeXh75+flMnDhRRUpi9vvf+17K99kH5szxw2iIpBI1kgizadMmFi9evMvtoIMO4thjj+Xcc8/llVde4auvvuL999/nzjvv3LlX1aVLFxYuXMiiRYtYunQpw4cP56uvvqp2GZ07d+bNN9/k1VdfZdiwYTjnGvMtSop48kl49FE/4OCcObDvvkEnEml4KlBh3nnnHXr37r3LbcSIEbz88suceOKJXHnllRxyyCFceOGFLFu2jH1Dnwq33347Rx55JKeffjr9+vWjVatWDB48uMblHHjggRQUFPDKK6+oSEnU3nsPrrnGT0+YAEcdFWwekXjRIb6QKVOmMGXKlBqfHzduHOPGjav2ubZt2zJnzpxaX7+goGCX+wceeCDffPNNtDElza1eDf37+/NP1133U+s9kVSkPSiRJLFlC5x/PqxbByeeCGPH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\n", - "text/plain": [ - "
" - ] - }, - "metadata": { - "needs_background": "light" - }, - "output_type": "display_data" - } - ], - "source": [ - "plt.plot(z, leaky_relu(z, 0.05), \"b-\", linewidth=2)\n", - "plt.plot([-5, 5], [0, 0], 'k-')\n", - "plt.plot([0, 0], [-0.5, 4.2], 'k-')\n", - "plt.grid(True)\n", - "props = dict(facecolor='black', shrink=0.1)\n", - "plt.annotate('Leak', xytext=(-3.5, 0.5), xy=(-5, -0.2), arrowprops=props, fontsize=14, ha=\"center\")\n", - "plt.title(\"Leaky ReLU activation function\", fontsize=14)\n", - "plt.axis([-5, 5, -0.5, 4.2])\n", - "\n", - "save_fig(\"leaky_relu_plot\")\n", - "plt.show()" - ] - }, - { - "cell_type": "code", - "execution_count": 9, - "metadata": {}, - "outputs": [ - { - "data": { - "text/plain": [ - "['deserialize',\n", - " 'elu',\n", - " 'exponential',\n", - " 'get',\n", - " 'hard_sigmoid',\n", - " 'linear',\n", - " 'relu',\n", - " 'selu',\n", - " 'serialize',\n", - " 'sigmoid',\n", - " 'softmax',\n", - " 'softplus',\n", - " 'softsign',\n", - " 'swish',\n", - " 'tanh']" - ] - }, - "execution_count": 9, - "metadata": {}, - "output_type": "execute_result" - } - ], - "source": [ - "[m for m in dir(keras.activations) if not m.startswith(\"_\")]" - ] - }, - { - "cell_type": "code", - "execution_count": 10, - "metadata": {}, - "outputs": [ - { - "data": { - "text/plain": [ - "['LeakyReLU', 'PReLU', 'ReLU', 'ThresholdedReLU']" - ] - }, - "execution_count": 10, - "metadata": {}, - "output_type": "execute_result" - } - ], - "source": [ - "[m for m in dir(keras.layers) if \"relu\" in m.lower()]" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "LeakyReLU를 사용해 패션 MNIST에서 신경망을 훈련해 보죠:" - ] - }, - { - "cell_type": "code", - "execution_count": 11, - "metadata": {}, - "outputs": [], - "source": [ - "(X_train_full, y_train_full), (X_test, y_test) = keras.datasets.fashion_mnist.load_data()\n", - "X_train_full = X_train_full / 255.0\n", - "X_test = X_test / 255.0\n", - "X_valid, X_train = X_train_full[:5000], X_train_full[5000:]\n", - "y_valid, y_train = y_train_full[:5000], y_train_full[5000:]" - ] - }, - { - "cell_type": "code", - "execution_count": 12, - "metadata": {}, - "outputs": [], - "source": [ - "tf.random.set_seed(42)\n", - "np.random.seed(42)\n", - "\n", - "model = keras.models.Sequential([\n", - " keras.layers.Flatten(input_shape=[28, 28]),\n", - " keras.layers.Dense(300, kernel_initializer=\"he_normal\"),\n", - " keras.layers.LeakyReLU(),\n", - " keras.layers.Dense(100, kernel_initializer=\"he_normal\"),\n", - " keras.layers.LeakyReLU(),\n", - " keras.layers.Dense(10, activation=\"softmax\")\n", - "])" - ] - }, - { - "cell_type": "code", - "execution_count": 13, - "metadata": {}, - "outputs": [], - "source": [ - "model.compile(loss=\"sparse_categorical_crossentropy\",\n", - " optimizer=keras.optimizers.SGD(lr=1e-3),\n", - " metrics=[\"accuracy\"])" - ] - }, - { - "cell_type": "code", - "execution_count": 14, - "metadata": { - "scrolled": true - }, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "Epoch 1/10\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 1.2819 - accuracy: 0.6229 - val_loss: 0.8886 - val_accuracy: 0.7160\n", - "Epoch 2/10\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.7955 - accuracy: 0.7361 - val_loss: 0.7130 - val_accuracy: 0.7656\n", - "Epoch 3/10\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.6816 - accuracy: 0.7721 - val_loss: 0.6427 - val_accuracy: 0.7898\n", - "Epoch 4/10\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.6217 - accuracy: 0.7943 - val_loss: 0.5900 - val_accuracy: 0.8066\n", - "Epoch 5/10\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.5832 - accuracy: 0.8075 - val_loss: 0.5582 - val_accuracy: 0.8202\n", - "Epoch 6/10\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.5553 - accuracy: 0.8157 - val_loss: 0.5350 - val_accuracy: 0.8238\n", - "Epoch 7/10\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.5338 - accuracy: 0.8224 - val_loss: 0.5157 - val_accuracy: 0.8304\n", - "Epoch 8/10\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.5172 - accuracy: 0.8273 - val_loss: 0.5079 - val_accuracy: 0.8282\n", - "Epoch 9/10\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.5040 - accuracy: 0.8289 - val_loss: 0.4895 - val_accuracy: 0.8386\n", - "Epoch 10/10\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.4924 - accuracy: 0.8321 - val_loss: 0.4817 - val_accuracy: 0.8396\n" - ] - } - ], - "source": [ - "history = model.fit(X_train, y_train, epochs=10,\n", - " validation_data=(X_valid, y_valid))" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "PReLU를 테스트해 보죠:" - ] - }, - { - "cell_type": "code", - "execution_count": 15, - "metadata": {}, - "outputs": [], - "source": [ - "tf.random.set_seed(42)\n", - "np.random.seed(42)\n", - "\n", - "model = keras.models.Sequential([\n", - " keras.layers.Flatten(input_shape=[28, 28]),\n", - " keras.layers.Dense(300, kernel_initializer=\"he_normal\"),\n", - " keras.layers.PReLU(),\n", - " keras.layers.Dense(100, kernel_initializer=\"he_normal\"),\n", - " keras.layers.PReLU(),\n", - " keras.layers.Dense(10, activation=\"softmax\")\n", - "])" - ] - }, - { - "cell_type": "code", - "execution_count": 16, - "metadata": {}, - "outputs": [], - "source": [ - "model.compile(loss=\"sparse_categorical_crossentropy\",\n", - " optimizer=keras.optimizers.SGD(lr=1e-3),\n", - " metrics=[\"accuracy\"])" - ] - }, - { - "cell_type": "code", - "execution_count": 17, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "Epoch 1/10\n", - "1719/1719 [==============================] - 6s 3ms/step - loss: 1.3461 - accuracy: 0.6209 - val_loss: 0.9255 - val_accuracy: 0.7184\n", - "Epoch 2/10\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.8197 - accuracy: 0.7355 - val_loss: 0.7305 - val_accuracy: 0.7628\n", - "Epoch 3/10\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.6966 - accuracy: 0.7694 - val_loss: 0.6565 - val_accuracy: 0.7880\n", - "Epoch 4/10\n", - "1719/1719 [==============================] - 6s 3ms/step - loss: 0.6331 - accuracy: 0.7909 - val_loss: 0.6003 - val_accuracy: 0.8048\n", - "Epoch 5/10\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.5917 - accuracy: 0.8057 - val_loss: 0.5656 - val_accuracy: 0.8184\n", - "Epoch 6/10\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.5618 - accuracy: 0.8134 - val_loss: 0.5406 - val_accuracy: 0.8238\n", - "Epoch 7/10\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.5390 - accuracy: 0.8206 - val_loss: 0.5196 - val_accuracy: 0.8312\n", - "Epoch 8/10\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.5213 - accuracy: 0.8257 - val_loss: 0.5113 - val_accuracy: 0.8320\n", - "Epoch 9/10\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.5070 - accuracy: 0.8288 - val_loss: 0.4916 - val_accuracy: 0.8380\n", - "Epoch 10/10\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.4945 - accuracy: 0.8315 - val_loss: 0.4826 - val_accuracy: 0.8396\n" - ] - } - ], - "source": [ - "history = model.fit(X_train, y_train, epochs=10,\n", - " validation_data=(X_valid, y_valid))" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "### ELU" - ] - }, - { - "cell_type": "code", - "execution_count": 18, - "metadata": {}, - "outputs": [], - "source": [ - "def elu(z, alpha=1):\n", - " return np.where(z < 0, alpha * (np.exp(z) - 1), z)" - ] - }, - { - "cell_type": "code", - "execution_count": 19, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "그림 저장: elu_plot\n" - ] - }, - { - "data": { - "image/png": "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\n", - "text/plain": [ - "
" - ] - }, - "metadata": { - "needs_background": "light" - }, - "output_type": "display_data" - } - ], - "source": [ - "plt.plot(z, elu(z), \"b-\", linewidth=2)\n", - "plt.plot([-5, 5], [0, 0], 'k-')\n", - "plt.plot([-5, 5], [-1, -1], 'k--')\n", - "plt.plot([0, 0], [-2.2, 3.2], 'k-')\n", - "plt.grid(True)\n", - "plt.title(r\"ELU activation function ($\\alpha=1$)\", fontsize=14)\n", - "plt.axis([-5, 5, -2.2, 3.2])\n", - "\n", - "save_fig(\"elu_plot\")\n", - "plt.show()" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "텐서플로에서 쉽게 ELU를 적용할 수 있습니다. 층을 만들 때 활성화 함수로 지정하면 됩니다:" - ] - }, - { - "cell_type": "code", - "execution_count": 20, - "metadata": {}, - "outputs": [ - { - "data": { - "text/plain": [ - "" - ] - }, - "execution_count": 20, - "metadata": {}, - "output_type": "execute_result" - } - ], - "source": [ - "keras.layers.Dense(10, activation=\"elu\")" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "### SELU" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "Günter Klambauer, Thomas Unterthiner, Andreas Mayr는 2017년 한 [훌륭한 논문](https://arxiv.org/pdf/1706.02515.pdf)에서 SELU 활성화 함수를 소개했습니다. 훈련하는 동안 완전 연결 층만 쌓아서 신경망을 만들고 SELU 활성화 함수와 LeCun 초기화를 사용한다면 자기 정규화됩니다. 각 층의 출력이 평균과\n", - "표준편차를 보존하는 경향이 있습니다. 이는 그레이디언트 소실과 폭주 문제를 막아줍니다. 그 결과로 SELU 활성화 함수는 이런 종류의 네트워크(특히 아주 깊은 네트워크)에서 다른 활성화 함수보다 뛰어난 성능을 종종 냅니다. 따라서 꼭 시도해 봐야 합니다. 하지만 SELU 활성화 함수의 자기 정규화 특징은 쉽게 깨집니다. ℓ1나 ℓ2 정규화, 드롭아웃, 맥스 노름, 스킵 연결이나 시퀀셜하지 않은 다른 토폴로지를 사용할 수 없습니다(즉 순환 신경망은 자기 정규화되지 않습니다). 하지만 실전에서 시퀀셜 CNN과 잘 동작합니다. 자기 정규화가 깨지면 SELU가 다른 활성화 함수보다 더 나은 성능을 내지 않을 것입니다." - ] - }, - { - "cell_type": "code", - "execution_count": 21, - "metadata": {}, - "outputs": [], - "source": [ - "from scipy.special import erfc\n", - "\n", - "# alpha와 scale은 평균 0과 표준 편차 1로 자기 정규화합니다\n", - "# (논문에 있는 식 14 참조):\n", - "alpha_0_1 = -np.sqrt(2 / np.pi) / (erfc(1/np.sqrt(2)) * np.exp(1/2) - 1)\n", - "scale_0_1 = (1 - erfc(1 / np.sqrt(2)) * np.sqrt(np.e)) * np.sqrt(2 * np.pi) * (2 * erfc(np.sqrt(2))*np.e**2 + np.pi*erfc(1/np.sqrt(2))**2*np.e - 2*(2+np.pi)*erfc(1/np.sqrt(2))*np.sqrt(np.e)+np.pi+2)**(-1/2)" - ] - }, - { - "cell_type": "code", - "execution_count": 22, - "metadata": {}, - "outputs": [], - "source": [ - "def selu(z, scale=scale_0_1, alpha=alpha_0_1):\n", - " return scale * elu(z, alpha)" - ] - }, - { - "cell_type": "code", - "execution_count": 23, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "그림 저장: selu_plot\n" - ] - }, - { - "data": { - "image/png": "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\n", - "text/plain": [ - "
" - ] - }, - "metadata": { - "needs_background": "light" - }, - "output_type": "display_data" - } - ], - "source": [ - "plt.plot(z, selu(z), \"b-\", linewidth=2)\n", - "plt.plot([-5, 5], [0, 0], 'k-')\n", - "plt.plot([-5, 5], [-1.758, -1.758], 'k--')\n", - "plt.plot([0, 0], [-2.2, 3.2], 'k-')\n", - "plt.grid(True)\n", - "plt.title(\"SELU activation function\", fontsize=14)\n", - "plt.axis([-5, 5, -2.2, 3.2])\n", - "\n", - "save_fig(\"selu_plot\")\n", - "plt.show()" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "기본적으로 SELU 하이퍼파라미터(`scale`과 `alpha`)는 각 뉴런의 평균 출력이 0에 가깝고 표준 편차는 1에 가깝도록 조정됩니다(입력은 평균이 0이고 표준 편차 1로 표준화되었다고 가정합니다). 이 활성화 함수를 사용하면 1,000개의 층이 있는 심층 신경망도 모든 층에 걸쳐 거의 평균이 0이고 표준 편차를 1로 유지합니다. 이를 통해 그레이디언트 폭주와 소실 문제를 피할 수 있습니다:" - ] - }, - { - "cell_type": "code", - "execution_count": 24, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "Layer 0: mean -0.00, std deviation 1.00\n", - "Layer 100: mean 0.02, std deviation 0.96\n", - "Layer 200: mean 0.01, std deviation 0.90\n", - "Layer 300: mean -0.02, std deviation 0.92\n", - "Layer 400: mean 0.05, std deviation 0.89\n", - "Layer 500: mean 0.01, std deviation 0.93\n", - "Layer 600: mean 0.02, std deviation 0.92\n", - "Layer 700: mean -0.02, std deviation 0.90\n", - "Layer 800: mean 0.05, std deviation 0.83\n", - "Layer 900: mean 0.02, std deviation 1.00\n" - ] - } - ], - "source": [ - "np.random.seed(42)\n", - "Z = np.random.normal(size=(500, 100)) # 표준화된 입력\n", - "for layer in range(1000):\n", - " W = np.random.normal(size=(100, 100), scale=np.sqrt(1 / 100)) # LeCun 초기화\n", - " Z = selu(np.dot(Z, W))\n", - " means = np.mean(Z, axis=0).mean()\n", - " stds = np.std(Z, axis=0).mean()\n", - " if layer % 100 == 0:\n", - " print(\"Layer {}: mean {:.2f}, std deviation {:.2f}\".format(layer, means, stds))" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "쉽게 SELU를 사용할 수 있습니다:" - ] - }, - { - "cell_type": "code", - "execution_count": 25, - "metadata": {}, - "outputs": [ - { - "data": { - "text/plain": [ - "" - ] - }, - "execution_count": 25, - "metadata": {}, - "output_type": "execute_result" - } - ], - "source": [ - "keras.layers.Dense(10, activation=\"selu\",\n", - " kernel_initializer=\"lecun_normal\")" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "100개의 은닉층과 SELU 활성화 함수를 사용한 패션 MNIST를 위한 신경망을 만들어 보죠:" - ] - }, - { - "cell_type": "code", - "execution_count": 26, - "metadata": {}, - "outputs": [], - "source": [ - "np.random.seed(42)\n", - "tf.random.set_seed(42)" - ] - }, - { - "cell_type": "code", - "execution_count": 27, - "metadata": {}, - "outputs": [], - "source": [ - "model = keras.models.Sequential()\n", - "model.add(keras.layers.Flatten(input_shape=[28, 28]))\n", - "model.add(keras.layers.Dense(300, activation=\"selu\",\n", - " kernel_initializer=\"lecun_normal\"))\n", - "for layer in range(99):\n", - " model.add(keras.layers.Dense(100, activation=\"selu\",\n", - " kernel_initializer=\"lecun_normal\"))\n", - "model.add(keras.layers.Dense(10, activation=\"softmax\"))" - ] - }, - { - "cell_type": "code", - "execution_count": 28, - "metadata": {}, - "outputs": [], - "source": [ - "model.compile(loss=\"sparse_categorical_crossentropy\",\n", - " optimizer=keras.optimizers.SGD(lr=1e-3),\n", - " metrics=[\"accuracy\"])" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "이제 훈련해 보죠. 입력을 평균 0과 표준 편차 1로 바꾸어야 한다는 것을 잊지 마세요:" - ] - }, - { - "cell_type": "code", - "execution_count": 29, - "metadata": {}, - "outputs": [], - "source": [ - "pixel_means = X_train.mean(axis=0, keepdims=True)\n", - "pixel_stds = X_train.std(axis=0, keepdims=True)\n", - "X_train_scaled = (X_train - pixel_means) / pixel_stds\n", - "X_valid_scaled = (X_valid - pixel_means) / pixel_stds\n", - "X_test_scaled = (X_test - pixel_means) / pixel_stds" - ] - }, - { - "cell_type": "code", - "execution_count": 30, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "Epoch 1/5\n", - "1719/1719 [==============================] - 32s 19ms/step - loss: 1.4254 - accuracy: 0.4457 - val_loss: 0.9036 - val_accuracy: 0.6758\n", - "Epoch 2/5\n", - "1719/1719 [==============================] - 32s 19ms/step - loss: 0.8673 - accuracy: 0.6903 - val_loss: 0.7675 - val_accuracy: 0.7316\n", - "Epoch 3/5\n", - "1719/1719 [==============================] - 32s 18ms/step - loss: 0.6920 - accuracy: 0.7525 - val_loss: 0.6481 - val_accuracy: 0.7694\n", - "Epoch 4/5\n", - "1719/1719 [==============================] - 32s 18ms/step - loss: 0.6801 - accuracy: 0.7533 - val_loss: 0.6137 - val_accuracy: 0.7852\n", - "Epoch 5/5\n", - "1719/1719 [==============================] - 32s 18ms/step - loss: 0.5883 - accuracy: 0.7845 - val_loss: 0.5503 - val_accuracy: 0.8036\n" - ] - } - ], - "source": [ - "history = model.fit(X_train_scaled, y_train, epochs=5,\n", - " validation_data=(X_valid_scaled, y_valid))" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "대신 ReLU 활성화 함수를 사용하면 어떤 일이 일어나는지 확인해 보죠:" - ] - }, - { - "cell_type": "code", - "execution_count": 31, - "metadata": {}, - "outputs": [], - "source": [ - "np.random.seed(42)\n", - "tf.random.set_seed(42)" - ] - }, - { - "cell_type": "code", - "execution_count": 32, - "metadata": {}, - "outputs": [], - "source": [ - "model = keras.models.Sequential()\n", - "model.add(keras.layers.Flatten(input_shape=[28, 28]))\n", - "model.add(keras.layers.Dense(300, activation=\"relu\", kernel_initializer=\"he_normal\"))\n", - "for layer in range(99):\n", - " model.add(keras.layers.Dense(100, activation=\"relu\", kernel_initializer=\"he_normal\"))\n", - "model.add(keras.layers.Dense(10, activation=\"softmax\"))" - ] - }, - { - "cell_type": "code", - "execution_count": 33, - "metadata": {}, - "outputs": [], - "source": [ - "model.compile(loss=\"sparse_categorical_crossentropy\",\n", - " optimizer=keras.optimizers.SGD(lr=1e-3),\n", - " metrics=[\"accuracy\"])" - ] - }, - { - "cell_type": "code", - "execution_count": 34, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "Epoch 1/5\n", - "1719/1719 [==============================] - 33s 19ms/step - loss: 1.8139 - accuracy: 0.2607 - val_loss: 1.4307 - val_accuracy: 0.3734\n", - "Epoch 2/5\n", - "1719/1719 [==============================] - 32s 19ms/step - loss: 1.1872 - accuracy: 0.4937 - val_loss: 1.0023 - val_accuracy: 0.5844\n", - "Epoch 3/5\n", - "1719/1719 [==============================] - 32s 19ms/step - loss: 0.9595 - accuracy: 0.6029 - val_loss: 0.8268 - val_accuracy: 0.6698\n", - "Epoch 4/5\n", - "1719/1719 [==============================] - 32s 19ms/step - loss: 0.9046 - accuracy: 0.6324 - val_loss: 0.8080 - val_accuracy: 0.6908\n", - "Epoch 5/5\n", - "1719/1719 [==============================] - 32s 19ms/step - loss: 0.8454 - accuracy: 0.6642 - val_loss: 0.7522 - val_accuracy: 0.7180\n" - ] - } - ], - "source": [ - "history = model.fit(X_train_scaled, y_train, epochs=5,\n", - " validation_data=(X_valid_scaled, y_valid))" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "좋지 않군요. 그레이디언트 폭주나 소실 문제가 발생한 것입니다." - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "# 배치 정규화" - ] - }, - { - "cell_type": "code", - "execution_count": 35, - "metadata": {}, - "outputs": [], - "source": [ - "model = keras.models.Sequential([\n", - " keras.layers.Flatten(input_shape=[28, 28]),\n", - " keras.layers.BatchNormalization(),\n", - " keras.layers.Dense(300, activation=\"relu\"),\n", - " keras.layers.BatchNormalization(),\n", - " keras.layers.Dense(100, activation=\"relu\"),\n", - " keras.layers.BatchNormalization(),\n", - " keras.layers.Dense(10, activation=\"softmax\")\n", - "])" - ] - }, - { - "cell_type": "code", - "execution_count": 36, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "Model: \"sequential_4\"\n", - "_________________________________________________________________\n", - "Layer (type) Output Shape Param # \n", - "=================================================================\n", - "flatten_4 (Flatten) (None, 784) 0 \n", - "_________________________________________________________________\n", - "batch_normalization (BatchNo (None, 784) 3136 \n", - "_________________________________________________________________\n", - "dense_212 (Dense) (None, 300) 235500 \n", - "_________________________________________________________________\n", - "batch_normalization_1 (Batch (None, 300) 1200 \n", - "_________________________________________________________________\n", - "dense_213 (Dense) (None, 100) 30100 \n", - "_________________________________________________________________\n", - "batch_normalization_2 (Batch (None, 100) 400 \n", - "_________________________________________________________________\n", - "dense_214 (Dense) (None, 10) 1010 \n", - "=================================================================\n", - "Total params: 271,346\n", - "Trainable params: 268,978\n", - "Non-trainable params: 2,368\n", - "_________________________________________________________________\n" - ] - } - ], - "source": [ - "model.summary()" - ] - }, - { - "cell_type": "code", - "execution_count": 37, - "metadata": {}, - "outputs": [ - { - "data": { - "text/plain": [ - "[('batch_normalization/gamma:0', True),\n", - " ('batch_normalization/beta:0', True),\n", - " ('batch_normalization/moving_mean:0', False),\n", - " ('batch_normalization/moving_variance:0', False)]" - ] - }, - "execution_count": 37, - "metadata": {}, - "output_type": "execute_result" - } - ], - "source": [ - "bn1 = model.layers[1]\n", - "[(var.name, var.trainable) for var in bn1.variables]" - ] - }, - { - "cell_type": "code", - "execution_count": 38, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "WARNING:tensorflow:From :1: Layer.updates (from tensorflow.python.keras.engine.base_layer) is deprecated and will be removed in a future version.\n", - "Instructions for updating:\n", - "This property should not be used in TensorFlow 2.0, as updates are applied automatically.\n" - ] - }, - { - "data": { - "text/plain": [ - "[]" - ] - }, - "execution_count": 38, - "metadata": {}, - "output_type": "execute_result" - } - ], - "source": [ - "bn1.updates" - ] - }, - { - "cell_type": "code", - "execution_count": 39, - "metadata": {}, - "outputs": [], - "source": [ - "model.compile(loss=\"sparse_categorical_crossentropy\",\n", - " optimizer=keras.optimizers.SGD(lr=1e-3),\n", - " metrics=[\"accuracy\"])" - ] - }, - { - "cell_type": "code", - "execution_count": 40, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "Epoch 1/10\n", - "1719/1719 [==============================] - 8s 5ms/step - loss: 0.8750 - accuracy: 0.7123 - val_loss: 0.5525 - val_accuracy: 0.8228\n", - "Epoch 2/10\n", - "1719/1719 [==============================] - 8s 5ms/step - loss: 0.5753 - accuracy: 0.8031 - val_loss: 0.4724 - val_accuracy: 0.8476\n", - "Epoch 3/10\n", - "1719/1719 [==============================] - 8s 5ms/step - loss: 0.5189 - accuracy: 0.8205 - val_loss: 0.4375 - val_accuracy: 0.8546\n", - "Epoch 4/10\n", - "1719/1719 [==============================] - 8s 5ms/step - loss: 0.4827 - accuracy: 0.8322 - val_loss: 0.4152 - val_accuracy: 0.8594\n", - "Epoch 5/10\n", - "1719/1719 [==============================] - 8s 5ms/step - loss: 0.4565 - accuracy: 0.8408 - val_loss: 0.3997 - val_accuracy: 0.8636\n", - "Epoch 6/10\n", - "1719/1719 [==============================] - 8s 5ms/step - loss: 0.4398 - accuracy: 0.8472 - val_loss: 0.3867 - val_accuracy: 0.8700\n", - "Epoch 7/10\n", - "1719/1719 [==============================] - 8s 5ms/step - loss: 0.4242 - accuracy: 0.8511 - val_loss: 0.3762 - val_accuracy: 0.8706\n", - "Epoch 8/10\n", - "1719/1719 [==============================] - 8s 5ms/step - loss: 0.4144 - accuracy: 0.8541 - val_loss: 0.3710 - val_accuracy: 0.8736\n", - "Epoch 9/10\n", - "1719/1719 [==============================] - 8s 5ms/step - loss: 0.4024 - accuracy: 0.8581 - val_loss: 0.3630 - val_accuracy: 0.8756\n", - "Epoch 10/10\n", - "1719/1719 [==============================] - 8s 5ms/step - loss: 0.3915 - accuracy: 0.8623 - val_loss: 0.3572 - val_accuracy: 0.8754\n" - ] - } - ], - "source": [ - "history = model.fit(X_train, y_train, epochs=10,\n", - " validation_data=(X_valid, y_valid))" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "이따금 활성화 함수전에 BN을 적용해도 잘 동작합니다(여기에는 논란의 여지가 있습니다). 또한 `BatchNormalization` 층 이전의 층은 편향을 위한 항이 필요 없습니다. `BatchNormalization` 층이 이를 무효화하기 때문입니다. 따라서 필요 없는 파라미터이므로 `use_bias=False`를 지정하여 층을 만들 수 있습니다:" - ] - }, - { - "cell_type": "code", - "execution_count": 41, - "metadata": {}, - "outputs": [], - "source": [ - "model = keras.models.Sequential([\n", - " keras.layers.Flatten(input_shape=[28, 28]),\n", - " keras.layers.BatchNormalization(),\n", - " keras.layers.Dense(300, use_bias=False),\n", - " keras.layers.BatchNormalization(),\n", - " keras.layers.Activation(\"relu\"),\n", - " keras.layers.Dense(100, use_bias=False),\n", - " keras.layers.BatchNormalization(),\n", - " keras.layers.Activation(\"relu\"),\n", - " keras.layers.Dense(10, activation=\"softmax\")\n", - "])" - ] - }, - { - "cell_type": "code", - "execution_count": 42, - "metadata": {}, - "outputs": [], - "source": [ - "model.compile(loss=\"sparse_categorical_crossentropy\",\n", - " optimizer=keras.optimizers.SGD(lr=1e-3),\n", - " metrics=[\"accuracy\"])" - ] - }, - { - "cell_type": "code", - "execution_count": 43, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "Epoch 1/10\n", - "1719/1719 [==============================] - 8s 5ms/step - loss: 1.0317 - accuracy: 0.6757 - val_loss: 0.6767 - val_accuracy: 0.7816\n", - "Epoch 2/10\n", - "1719/1719 [==============================] - 8s 5ms/step - loss: 0.6790 - accuracy: 0.7792 - val_loss: 0.5566 - val_accuracy: 0.8180\n", - "Epoch 3/10\n", - "1719/1719 [==============================] - 8s 5ms/step - loss: 0.5960 - accuracy: 0.8037 - val_loss: 0.5007 - val_accuracy: 0.8360\n", - "Epoch 4/10\n", - "1719/1719 [==============================] - 8s 5ms/step - loss: 0.5447 - accuracy: 0.8192 - val_loss: 0.4666 - val_accuracy: 0.8448\n", - "Epoch 5/10\n", - "1719/1719 [==============================] - 8s 5ms/step - loss: 0.5109 - accuracy: 0.8279 - val_loss: 0.4434 - val_accuracy: 0.8534\n", - "Epoch 6/10\n", - "1719/1719 [==============================] - 8s 5ms/step - loss: 0.4898 - accuracy: 0.8336 - val_loss: 0.4263 - val_accuracy: 0.8550\n", - "Epoch 7/10\n", - "1719/1719 [==============================] - 8s 5ms/step - loss: 0.4712 - accuracy: 0.8397 - val_loss: 0.4130 - val_accuracy: 0.8572\n", - "Epoch 8/10\n", - "1719/1719 [==============================] - 8s 5ms/step - loss: 0.4560 - accuracy: 0.8441 - val_loss: 0.4035 - val_accuracy: 0.8606\n", - "Epoch 9/10\n", - "1719/1719 [==============================] - 8s 5ms/step - loss: 0.4441 - accuracy: 0.8473 - val_loss: 0.3943 - val_accuracy: 0.8642\n", - "Epoch 10/10\n", - "1719/1719 [==============================] - 8s 5ms/step - loss: 0.4332 - accuracy: 0.8505 - val_loss: 0.3874 - val_accuracy: 0.8662\n" - ] - } - ], - "source": [ - "history = model.fit(X_train, y_train, epochs=10,\n", - " validation_data=(X_valid, y_valid))" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "## 그레이디언트 클리핑" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "모든 케라스 옵티마이저는 `clipnorm`이나 `clipvalue` 매개변수를 지원합니다:" - ] - }, - { - "cell_type": "code", - "execution_count": 44, - "metadata": {}, - "outputs": [], - "source": [ - "optimizer = keras.optimizers.SGD(clipvalue=1.0)" - ] - }, - { - "cell_type": "code", - "execution_count": 45, - "metadata": {}, - "outputs": [], - "source": [ - "optimizer = keras.optimizers.SGD(clipnorm=1.0)" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "## 사전 훈련된 층 재사용하기" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "### 케라스 모델 재사용하기" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "패션 MNIST 훈련 세트를 두 개로 나누어 보죠:\n", - "* `X_train_A`: 샌달과 셔츠(클래스 5와 6)을 제외한 모든 이미지\n", - "* `X_train_B`: 샌달과 셔츠 이미지 중 처음 200개만 가진 작은 훈련 세트\n", - "\n", - "검증 세트와 테스트 세트도 이렇게 나눕니다. 하지만 이미지 개수는 제한하지 않습니다.\n", - "\n", - "A 세트(8개의 클래스를 가진 분류 문제)에서 모델을 훈련하고 이를 재사용하여 B 세트(이진 분류)를 해결해 보겠습니다. A 작업에서 B 작업으로 약간의 지식이 전달되기를 기대합니다. 왜냐하면 A 세트의 클래스(스니커즈, 앵클 부츠, 코트, 티셔츠 등)가 B 세트에 있는 클래스(샌달과 셔츠)와 조금 비슷하기 때문입니다. 하지만 `Dense` 층을 사용하기 때문에 동일한 위치에 나타난 패턴만 재사용할 수 있습니다(반대로 합성곱 층은 훨씬 많은 정보를 전송합니다. 학습한 패턴을 이미지의 어느 위치에서나 감지할 수 있기 때문입니다. CNN 장에서 자세히 알아 보겠습니다)." - ] - }, - { - "cell_type": "code", - "execution_count": 46, - "metadata": {}, - "outputs": [], - "source": [ - "def split_dataset(X, y):\n", - " y_5_or_6 = (y == 5) | (y == 6) # sandals or shirts\n", - " y_A = y[~y_5_or_6]\n", - " y_A[y_A > 6] -= 2 # class indices 7, 8, 9 should be moved to 5, 6, 7\n", - " y_B = (y[y_5_or_6] == 6).astype(np.float32) # binary classification task: is it a shirt (class 6)?\n", - " return ((X[~y_5_or_6], y_A),\n", - " (X[y_5_or_6], y_B))\n", - "\n", - "(X_train_A, y_train_A), (X_train_B, y_train_B) = split_dataset(X_train, y_train)\n", - "(X_valid_A, y_valid_A), (X_valid_B, y_valid_B) = split_dataset(X_valid, y_valid)\n", - "(X_test_A, y_test_A), (X_test_B, y_test_B) = split_dataset(X_test, y_test)\n", - "X_train_B = X_train_B[:200]\n", - "y_train_B = y_train_B[:200]" - ] - }, - { - "cell_type": "code", - "execution_count": 47, - "metadata": {}, - "outputs": [ - { - "data": { - "text/plain": [ - "(43986, 28, 28)" - ] - }, - "execution_count": 47, - "metadata": {}, - "output_type": "execute_result" - } - ], - "source": [ - "X_train_A.shape" - ] - }, - { - "cell_type": "code", - "execution_count": 48, - "metadata": {}, - "outputs": [ - { - "data": { - "text/plain": [ - "(200, 28, 28)" - ] - }, - "execution_count": 48, - "metadata": {}, - "output_type": "execute_result" - } - ], - "source": [ - "X_train_B.shape" - ] - }, - { - "cell_type": "code", - "execution_count": 49, - "metadata": {}, - "outputs": [ - { - "data": { - "text/plain": [ - "array([4, 0, 5, 7, 7, 7, 4, 4, 3, 4, 0, 1, 6, 3, 4, 3, 2, 6, 5, 3, 4, 5,\n", - " 1, 3, 4, 2, 0, 6, 7, 1], dtype=uint8)" - ] - }, - "execution_count": 49, - "metadata": {}, - "output_type": "execute_result" - } - ], - "source": [ - "y_train_A[:30]" - ] - }, - { - "cell_type": "code", - "execution_count": 50, - "metadata": {}, - "outputs": [ - { - "data": { - "text/plain": [ - "array([1., 1., 0., 0., 0., 0., 1., 1., 1., 0., 0., 1., 1., 0., 0., 0., 0.,\n", - " 0., 0., 1., 1., 0., 0., 1., 1., 0., 1., 1., 1., 1.], dtype=float32)" - ] - }, - "execution_count": 50, - "metadata": {}, - "output_type": "execute_result" - } - ], - "source": [ - "y_train_B[:30]" - ] - }, - { - "cell_type": "code", - "execution_count": 51, - "metadata": {}, - "outputs": [], - "source": [ - "tf.random.set_seed(42)\n", - "np.random.seed(42)" - ] - }, - { - "cell_type": "code", - "execution_count": 52, - "metadata": {}, - "outputs": [], - "source": [ - "model_A = keras.models.Sequential()\n", - "model_A.add(keras.layers.Flatten(input_shape=[28, 28]))\n", - "for n_hidden in (300, 100, 50, 50, 50):\n", - " model_A.add(keras.layers.Dense(n_hidden, activation=\"selu\"))\n", - "model_A.add(keras.layers.Dense(8, activation=\"softmax\"))" - ] - }, - { - "cell_type": "code", - "execution_count": 53, - "metadata": {}, - "outputs": [], - "source": [ - "model_A.compile(loss=\"sparse_categorical_crossentropy\",\n", - " optimizer=keras.optimizers.SGD(lr=1e-3),\n", - " metrics=[\"accuracy\"])" - ] - }, - { - "cell_type": "code", - "execution_count": 54, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "Epoch 1/20\n", - "1375/1375 [==============================] - 5s 3ms/step - loss: 0.5927 - accuracy: 0.8103 - val_loss: 0.3894 - val_accuracy: 0.8662\n", - "Epoch 2/20\n", - "1375/1375 [==============================] - 4s 3ms/step - loss: 0.3523 - accuracy: 0.8785 - val_loss: 0.3286 - val_accuracy: 0.8834\n", - "Epoch 3/20\n", - "1375/1375 [==============================] - 4s 3ms/step - loss: 0.3170 - accuracy: 0.8896 - val_loss: 0.3011 - val_accuracy: 0.8984\n", - "Epoch 4/20\n", - "1375/1375 [==============================] - 4s 3ms/step - loss: 0.2974 - accuracy: 0.8975 - val_loss: 0.2895 - val_accuracy: 0.9018\n", - "Epoch 5/20\n", - "1375/1375 [==============================] - 4s 3ms/step - loss: 0.2835 - accuracy: 0.9020 - val_loss: 0.2774 - val_accuracy: 0.9071\n", - "Epoch 6/20\n", - "1375/1375 [==============================] - 5s 3ms/step - loss: 0.2730 - accuracy: 0.9063 - val_loss: 0.2736 - val_accuracy: 0.9061\n", - "Epoch 7/20\n", - "1375/1375 [==============================] - 4s 3ms/step - loss: 0.2642 - accuracy: 0.9092 - val_loss: 0.2717 - val_accuracy: 0.9083\n", - "Epoch 8/20\n", - "1375/1375 [==============================] - 4s 3ms/step - loss: 0.2573 - accuracy: 0.9126 - val_loss: 0.2590 - val_accuracy: 0.9141\n", - "Epoch 9/20\n", - "1375/1375 [==============================] - 4s 3ms/step - loss: 0.2519 - accuracy: 0.9137 - val_loss: 0.2562 - val_accuracy: 0.9145\n", - "Epoch 10/20\n", - "1375/1375 [==============================] - 4s 3ms/step - loss: 0.2469 - accuracy: 0.9155 - val_loss: 0.2542 - val_accuracy: 0.9155\n", - "Epoch 11/20\n", - "1375/1375 [==============================] - 4s 3ms/step - loss: 0.2423 - accuracy: 0.9178 - val_loss: 0.2494 - val_accuracy: 0.9163\n", - "Epoch 12/20\n", - "1375/1375 [==============================] - 4s 3ms/step - loss: 0.2383 - accuracy: 0.9188 - val_loss: 0.2512 - val_accuracy: 0.9128\n", - "Epoch 13/20\n", - "1375/1375 [==============================] - 4s 3ms/step - loss: 0.2351 - accuracy: 0.9198 - val_loss: 0.2448 - val_accuracy: 0.9158\n", - "Epoch 14/20\n", - "1375/1375 [==============================] - 4s 3ms/step - loss: 0.2316 - accuracy: 0.9210 - val_loss: 0.2416 - val_accuracy: 0.9175\n", - "Epoch 15/20\n", - "1375/1375 [==============================] - 4s 3ms/step - loss: 0.2288 - accuracy: 0.9213 - val_loss: 0.2451 - val_accuracy: 0.9200\n", - "Epoch 16/20\n", - "1375/1375 [==============================] - 4s 3ms/step - loss: 0.2255 - accuracy: 0.9223 - val_loss: 0.2386 - val_accuracy: 0.9200\n", - "Epoch 17/20\n", - "1375/1375 [==============================] - 4s 3ms/step - loss: 0.2231 - accuracy: 0.9232 - val_loss: 0.2411 - val_accuracy: 0.9178\n", - "Epoch 18/20\n", - "1375/1375 [==============================] - 4s 3ms/step - loss: 0.2201 - accuracy: 0.9245 - val_loss: 0.2428 - val_accuracy: 0.9150\n", - "Epoch 19/20\n", - "1375/1375 [==============================] - 4s 3ms/step - loss: 0.2178 - accuracy: 0.9249 - val_loss: 0.2329 - val_accuracy: 0.9205\n", - "Epoch 20/20\n", - "1375/1375 [==============================] - 4s 3ms/step - loss: 0.2156 - accuracy: 0.9261 - val_loss: 0.2331 - val_accuracy: 0.9208\n" - ] - } - ], - "source": [ - "history = model_A.fit(X_train_A, y_train_A, epochs=20,\n", - " validation_data=(X_valid_A, y_valid_A))" - ] - }, - { - "cell_type": "code", - "execution_count": 55, - "metadata": {}, - "outputs": [], - "source": [ - "model_A.save(\"my_model_A.h5\")" - ] - }, - { - "cell_type": "code", - "execution_count": 56, - "metadata": {}, - "outputs": [], - "source": [ - "model_B = keras.models.Sequential()\n", - "model_B.add(keras.layers.Flatten(input_shape=[28, 28]))\n", - "for n_hidden in (300, 100, 50, 50, 50):\n", - " model_B.add(keras.layers.Dense(n_hidden, activation=\"selu\"))\n", - "model_B.add(keras.layers.Dense(1, activation=\"sigmoid\"))" - ] - }, - { - "cell_type": "code", - "execution_count": 57, - "metadata": {}, - "outputs": [], - "source": [ - "model_B.compile(loss=\"binary_crossentropy\",\n", - " optimizer=keras.optimizers.SGD(lr=1e-3),\n", - " metrics=[\"accuracy\"])" - ] - }, - { - "cell_type": "code", - "execution_count": 58, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "Epoch 1/20\n", - "7/7 [==============================] - 0s 42ms/step - loss: 0.9573 - accuracy: 0.4650 - val_loss: 0.6314 - val_accuracy: 0.6004\n", - "Epoch 2/20\n", - "7/7 [==============================] - 0s 16ms/step - loss: 0.5692 - accuracy: 0.7450 - val_loss: 0.4784 - val_accuracy: 0.8529\n", - "Epoch 3/20\n", - "7/7 [==============================] - 0s 16ms/step - loss: 0.4503 - accuracy: 0.8650 - val_loss: 0.4102 - val_accuracy: 0.8945\n", - "Epoch 4/20\n", - "7/7 [==============================] - 0s 16ms/step - loss: 0.3879 - accuracy: 0.8950 - val_loss: 0.3647 - val_accuracy: 0.9178\n", - "Epoch 5/20\n", - "7/7 [==============================] - 0s 17ms/step - loss: 0.3435 - accuracy: 0.9250 - val_loss: 0.3300 - val_accuracy: 0.9320\n", - "Epoch 6/20\n", - "7/7 [==============================] - 0s 16ms/step - loss: 0.3081 - accuracy: 0.9300 - val_loss: 0.3019 - val_accuracy: 0.9402\n", - "Epoch 7/20\n", - "7/7 [==============================] - 0s 15ms/step - loss: 0.2800 - accuracy: 0.9350 - val_loss: 0.2804 - val_accuracy: 0.9422\n", - "Epoch 8/20\n", - "7/7 [==============================] - 0s 16ms/step - loss: 0.2564 - accuracy: 0.9450 - val_loss: 0.2606 - val_accuracy: 0.9473\n", - "Epoch 9/20\n", - "7/7 [==============================] - 0s 17ms/step - loss: 0.2362 - accuracy: 0.9550 - val_loss: 0.2428 - val_accuracy: 0.9523\n", - "Epoch 10/20\n", - "7/7 [==============================] - 0s 16ms/step - loss: 0.2188 - accuracy: 0.9600 - val_loss: 0.2281 - val_accuracy: 0.9544\n", - "Epoch 11/20\n", - "7/7 [==============================] - 0s 18ms/step - loss: 0.2036 - accuracy: 0.9700 - val_loss: 0.2150 - val_accuracy: 0.9584\n", - "Epoch 12/20\n", - "7/7 [==============================] - 0s 16ms/step - loss: 0.1898 - accuracy: 0.9700 - val_loss: 0.2036 - val_accuracy: 0.9584\n", - "Epoch 13/20\n", - "7/7 [==============================] - 0s 17ms/step - loss: 0.1773 - accuracy: 0.9750 - val_loss: 0.1931 - val_accuracy: 0.9615\n", - "Epoch 14/20\n", - "7/7 [==============================] - 0s 16ms/step - loss: 0.1668 - accuracy: 0.9800 - val_loss: 0.1838 - val_accuracy: 0.9635\n", - "Epoch 15/20\n", - "7/7 [==============================] - 0s 16ms/step - loss: 0.1570 - accuracy: 0.9900 - val_loss: 0.1746 - val_accuracy: 0.9686\n", - "Epoch 16/20\n", - "7/7 [==============================] - 0s 16ms/step - loss: 0.1481 - accuracy: 0.9900 - val_loss: 0.1674 - val_accuracy: 0.9686\n", - "Epoch 17/20\n", - "7/7 [==============================] - 0s 16ms/step - loss: 0.1406 - accuracy: 0.9900 - val_loss: 0.1604 - val_accuracy: 0.9706\n", - "Epoch 18/20\n", - "7/7 [==============================] - 0s 16ms/step - loss: 0.1334 - accuracy: 0.9900 - val_loss: 0.1539 - val_accuracy: 0.9706\n", - "Epoch 19/20\n", - "7/7 [==============================] - 0s 18ms/step - loss: 0.1268 - accuracy: 0.9900 - val_loss: 0.1482 - val_accuracy: 0.9716\n", - "Epoch 20/20\n", - "7/7 [==============================] - 0s 16ms/step - loss: 0.1208 - accuracy: 0.9900 - val_loss: 0.1431 - val_accuracy: 0.9716\n" - ] - } - ], - "source": [ - "history = model_B.fit(X_train_B, y_train_B, epochs=20,\n", - " validation_data=(X_valid_B, y_valid_B))" - ] - }, - { - "cell_type": "code", - "execution_count": 59, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "Model: \"sequential_5\"\n", - "_________________________________________________________________\n", - "Layer (type) Output Shape Param # \n", - "=================================================================\n", - "flatten_5 (Flatten) (None, 784) 0 \n", - "_________________________________________________________________\n", - "batch_normalization_3 (Batch (None, 784) 3136 \n", - "_________________________________________________________________\n", - "dense_215 (Dense) (None, 300) 235200 \n", - "_________________________________________________________________\n", - "batch_normalization_4 (Batch (None, 300) 1200 \n", - "_________________________________________________________________\n", - "activation (Activation) (None, 300) 0 \n", - "_________________________________________________________________\n", - "dense_216 (Dense) (None, 100) 30000 \n", - "_________________________________________________________________\n", - "batch_normalization_5 (Batch (None, 100) 400 \n", - "_________________________________________________________________\n", - "activation_1 (Activation) (None, 100) 0 \n", - "_________________________________________________________________\n", - "dense_217 (Dense) (None, 10) 1010 \n", - "=================================================================\n", - "Total params: 270,946\n", - "Trainable params: 268,578\n", - "Non-trainable params: 2,368\n", - "_________________________________________________________________\n" - ] - } - ], - "source": [ - "model.summary()" - ] - }, - { - "cell_type": "code", - "execution_count": 60, - "metadata": {}, - "outputs": [], - "source": [ - "model_A = keras.models.load_model(\"my_model_A.h5\")\n", - "model_B_on_A = keras.models.Sequential(model_A.layers[:-1])\n", - "model_B_on_A.add(keras.layers.Dense(1, activation=\"sigmoid\"))" - ] - }, - { - "cell_type": "code", - "execution_count": 61, - "metadata": {}, - "outputs": [], - "source": [ - "model_A_clone = keras.models.clone_model(model_A)\n", - "model_A_clone.set_weights(model_A.get_weights())" - ] - }, - { - "cell_type": "code", - "execution_count": 62, - "metadata": {}, - "outputs": [], - "source": [ - "for layer in model_B_on_A.layers[:-1]:\n", - " layer.trainable = False\n", - "\n", - "model_B_on_A.compile(loss=\"binary_crossentropy\",\n", - " optimizer=keras.optimizers.SGD(lr=1e-3),\n", - " metrics=[\"accuracy\"])" - ] - }, - { - "cell_type": "code", - "execution_count": 63, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "Epoch 1/4\n", - "7/7 [==============================] - 0s 39ms/step - loss: 0.5803 - accuracy: 0.6500 - val_loss: 0.5842 - val_accuracy: 0.6329\n", - "Epoch 2/4\n", - "7/7 [==============================] - 0s 16ms/step - loss: 0.5436 - accuracy: 0.6800 - val_loss: 0.5466 - val_accuracy: 0.6724\n", - "Epoch 3/4\n", - "7/7 [==============================] - 0s 16ms/step - loss: 0.5066 - accuracy: 0.7300 - val_loss: 0.5144 - val_accuracy: 0.7099\n", - "Epoch 4/4\n", - "7/7 [==============================] - 0s 16ms/step - loss: 0.4749 - accuracy: 0.7500 - val_loss: 0.4855 - val_accuracy: 0.7312\n", - "Epoch 1/16\n", - "7/7 [==============================] - 0s 41ms/step - loss: 0.3964 - accuracy: 0.8100 - val_loss: 0.3461 - val_accuracy: 0.8631\n", - "Epoch 2/16\n", - "7/7 [==============================] - 0s 15ms/step - loss: 0.2799 - accuracy: 0.9350 - val_loss: 0.2603 - val_accuracy: 0.9260\n", - "Epoch 3/16\n", - "7/7 [==============================] - 0s 16ms/step - loss: 0.2083 - accuracy: 0.9650 - val_loss: 0.2110 - val_accuracy: 0.9544\n", - "Epoch 4/16\n", - "7/7 [==============================] - 0s 16ms/step - loss: 0.1670 - accuracy: 0.9800 - val_loss: 0.1790 - val_accuracy: 0.9696\n", - "Epoch 5/16\n", - "7/7 [==============================] - 0s 18ms/step - loss: 0.1397 - accuracy: 0.9800 - val_loss: 0.1562 - val_accuracy: 0.9757\n", - "Epoch 6/16\n", - "7/7 [==============================] - 0s 16ms/step - loss: 0.1198 - accuracy: 0.9950 - val_loss: 0.1394 - val_accuracy: 0.9807\n", - "Epoch 7/16\n", - "7/7 [==============================] - 0s 16ms/step - loss: 0.1051 - accuracy: 0.9950 - val_loss: 0.1267 - val_accuracy: 0.9838\n", - "Epoch 8/16\n", - "7/7 [==============================] - 0s 16ms/step - loss: 0.0938 - accuracy: 0.9950 - val_loss: 0.1164 - val_accuracy: 0.9858\n", - "Epoch 9/16\n", - "7/7 [==============================] - 0s 15ms/step - loss: 0.0848 - accuracy: 1.0000 - val_loss: 0.1067 - val_accuracy: 0.9888\n", - "Epoch 10/16\n", - "7/7 [==============================] - 0s 16ms/step - loss: 0.0763 - accuracy: 1.0000 - val_loss: 0.1001 - val_accuracy: 0.9899\n", - "Epoch 11/16\n", - "7/7 [==============================] - 0s 15ms/step - loss: 0.0705 - accuracy: 1.0000 - val_loss: 0.0941 - val_accuracy: 0.9899\n", - "Epoch 12/16\n", - "7/7 [==============================] - 0s 15ms/step - loss: 0.0650 - accuracy: 1.0000 - val_loss: 0.0889 - val_accuracy: 0.9899\n", - "Epoch 13/16\n", - "7/7 [==============================] - 0s 17ms/step - loss: 0.0603 - accuracy: 1.0000 - val_loss: 0.0840 - val_accuracy: 0.9899\n", - "Epoch 14/16\n", - "7/7 [==============================] - 0s 18ms/step - loss: 0.0560 - accuracy: 1.0000 - val_loss: 0.0804 - val_accuracy: 0.9899\n", - "Epoch 15/16\n", - "7/7 [==============================] - 0s 18ms/step - loss: 0.0526 - accuracy: 1.0000 - val_loss: 0.0770 - val_accuracy: 0.9899\n", - "Epoch 16/16\n", - "7/7 [==============================] - 0s 18ms/step - loss: 0.0497 - accuracy: 1.0000 - val_loss: 0.0740 - val_accuracy: 0.9899\n" - ] - } - ], - "source": [ - "history = model_B_on_A.fit(X_train_B, y_train_B, epochs=4,\n", - " validation_data=(X_valid_B, y_valid_B))\n", - "\n", - "for layer in model_B_on_A.layers[:-1]:\n", - " layer.trainable = True\n", - "\n", - "model_B_on_A.compile(loss=\"binary_crossentropy\",\n", - " optimizer=keras.optimizers.SGD(lr=1e-3),\n", - " metrics=[\"accuracy\"])\n", - "history = model_B_on_A.fit(X_train_B, y_train_B, epochs=16,\n", - " validation_data=(X_valid_B, y_valid_B))" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "마지막 점수는 어떤가요?" - ] - }, - { - "cell_type": "code", - "execution_count": 64, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "63/63 [==============================] - 0s 2ms/step - loss: 0.1408 - accuracy: 0.9705\n" - ] - }, - { - "data": { - "text/plain": [ - "[0.1408407837152481, 0.9704999923706055]" - ] - }, - "execution_count": 64, - "metadata": {}, - "output_type": "execute_result" - } - ], - "source": [ - "model_B.evaluate(X_test_B, y_test_B)" - ] - }, - { - "cell_type": "code", - "execution_count": 65, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "63/63 [==============================] - 0s 2ms/step - loss: 0.0683 - accuracy: 0.9930\n" - ] - }, - { - "data": { - "text/plain": [ - "[0.0683005154132843, 0.9929999709129333]" - ] - }, - "execution_count": 65, - "metadata": {}, - "output_type": "execute_result" - } - ], - "source": [ - "model_B_on_A.evaluate(X_test_B, y_test_B)" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "훌륭하네요! 꽤 많은 정보를 전달했습니다: 오차율이 4배나 줄었네요!" - ] - }, - { - "cell_type": "code", - "execution_count": 66, - "metadata": {}, - "outputs": [ - { - "data": { - "text/plain": [ - "4.066666666666663" - ] - }, - "execution_count": 66, - "metadata": {}, - "output_type": "execute_result" - } - ], - "source": [ - "(100 - 96.95) / (100 - 99.25)" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "# 고속 옵티마이저" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "## 모멘텀 옵티마이저" - ] - }, - { - "cell_type": "code", - "execution_count": 67, - "metadata": {}, - "outputs": [], - "source": [ - "optimizer = keras.optimizers.SGD(lr=0.001, momentum=0.9)" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "## 네스테로프 가속 경사" - ] - }, - { - "cell_type": "code", - "execution_count": 68, - "metadata": {}, - "outputs": [], - "source": [ - "optimizer = keras.optimizers.SGD(lr=0.001, momentum=0.9, nesterov=True)" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "## AdaGrad" - ] - }, - { - "cell_type": "code", - "execution_count": 69, - "metadata": {}, - "outputs": [], - "source": [ - "optimizer = keras.optimizers.Adagrad(lr=0.001)" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "## RMSProp" - ] - }, - { - "cell_type": "code", - "execution_count": 70, - "metadata": {}, - "outputs": [], - "source": [ - "optimizer = keras.optimizers.RMSprop(lr=0.001, rho=0.9)" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "## Adam 옵티마이저" - ] - }, - { - "cell_type": "code", - "execution_count": 71, - "metadata": {}, - "outputs": [], - "source": [ - "optimizer = keras.optimizers.Adam(lr=0.001, beta_1=0.9, beta_2=0.999)" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "## Adamax 옵티마이저" - ] - }, - { - "cell_type": "code", - "execution_count": 72, - "metadata": {}, - "outputs": [], - "source": [ - "optimizer = keras.optimizers.Adamax(lr=0.001, beta_1=0.9, beta_2=0.999)" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "## Nadam 옵티마이저" - ] - }, - { - "cell_type": "code", - "execution_count": 73, - "metadata": {}, - "outputs": [], - "source": [ - "optimizer = keras.optimizers.Nadam(lr=0.001, beta_1=0.9, beta_2=0.999)" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "## 학습률 스케줄링" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "### 거듭제곱 스케줄링" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "```lr = lr0 / (1 + steps / s)**c```\n", - "* 케라스는 `c=1`과 `s = 1 / decay`을 사용합니다" - ] - }, - { - "cell_type": "code", - "execution_count": 74, - "metadata": {}, - "outputs": [], - "source": [ - "optimizer = keras.optimizers.SGD(lr=0.01, decay=1e-4)" - ] - }, - { - "cell_type": "code", - "execution_count": 75, - "metadata": {}, - "outputs": [], - "source": [ - "model = keras.models.Sequential([\n", - " keras.layers.Flatten(input_shape=[28, 28]),\n", - " keras.layers.Dense(300, activation=\"selu\", kernel_initializer=\"lecun_normal\"),\n", - " keras.layers.Dense(100, activation=\"selu\", kernel_initializer=\"lecun_normal\"),\n", - " keras.layers.Dense(10, activation=\"softmax\")\n", - "])\n", - "model.compile(loss=\"sparse_categorical_crossentropy\", optimizer=optimizer, metrics=[\"accuracy\"])" - ] - }, - { - "cell_type": "code", - "execution_count": 76, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "Epoch 1/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.4855 - accuracy: 0.8303 - val_loss: 0.4029 - val_accuracy: 0.8604\n", - "Epoch 2/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.3781 - accuracy: 0.8658 - val_loss: 0.3716 - val_accuracy: 0.8720\n", - "Epoch 3/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.3457 - accuracy: 0.8770 - val_loss: 0.3749 - val_accuracy: 0.8742\n", - "Epoch 4/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.3250 - accuracy: 0.8830 - val_loss: 0.3501 - val_accuracy: 0.8800\n", - "Epoch 5/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.3101 - accuracy: 0.8892 - val_loss: 0.3447 - val_accuracy: 0.8794\n", - "Epoch 6/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2966 - accuracy: 0.8935 - val_loss: 0.3412 - val_accuracy: 0.8828\n", - "Epoch 7/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2867 - accuracy: 0.8974 - val_loss: 0.3355 - val_accuracy: 0.8864\n", - "Epoch 8/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2777 - accuracy: 0.9010 - val_loss: 0.3408 - val_accuracy: 0.8834\n", - "Epoch 9/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2698 - accuracy: 0.9020 - val_loss: 0.3289 - val_accuracy: 0.8880\n", - "Epoch 10/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2628 - accuracy: 0.9048 - val_loss: 0.3259 - val_accuracy: 0.8880\n", - "Epoch 11/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2571 - accuracy: 0.9080 - val_loss: 0.3265 - val_accuracy: 0.8876\n", - "Epoch 12/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2512 - accuracy: 0.9098 - val_loss: 0.3331 - val_accuracy: 0.8830\n", - "Epoch 13/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2461 - accuracy: 0.9127 - val_loss: 0.3253 - val_accuracy: 0.8892\n", - "Epoch 14/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2422 - accuracy: 0.9135 - val_loss: 0.3286 - val_accuracy: 0.8900\n", - "Epoch 15/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2374 - accuracy: 0.9152 - val_loss: 0.3241 - val_accuracy: 0.8880\n", - "Epoch 16/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2335 - accuracy: 0.9170 - val_loss: 0.3202 - val_accuracy: 0.8904\n", - "Epoch 17/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2299 - accuracy: 0.9181 - val_loss: 0.3233 - val_accuracy: 0.8912\n", - "Epoch 18/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2262 - accuracy: 0.9200 - val_loss: 0.3188 - val_accuracy: 0.8932\n", - "Epoch 19/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2232 - accuracy: 0.9210 - val_loss: 0.3227 - val_accuracy: 0.8902\n", - "Epoch 20/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2199 - accuracy: 0.9221 - val_loss: 0.3207 - val_accuracy: 0.8912\n", - "Epoch 21/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2168 - accuracy: 0.9236 - val_loss: 0.3206 - val_accuracy: 0.8918\n", - "Epoch 22/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2142 - accuracy: 0.9245 - val_loss: 0.3179 - val_accuracy: 0.8942\n", - "Epoch 23/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2116 - accuracy: 0.9250 - val_loss: 0.3193 - val_accuracy: 0.8908\n", - "Epoch 24/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2088 - accuracy: 0.9266 - val_loss: 0.3212 - val_accuracy: 0.8886\n", - "Epoch 25/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2068 - accuracy: 0.9266 - val_loss: 0.3211 - val_accuracy: 0.8926\n" - ] - } - ], - "source": [ - "n_epochs = 25\n", - "history = model.fit(X_train_scaled, y_train, epochs=n_epochs,\n", - " validation_data=(X_valid_scaled, y_valid))" - ] - }, - { - "cell_type": "code", - "execution_count": 77, - "metadata": {}, - "outputs": [ - { - "data": { - "image/png": "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\n", - "text/plain": [ - "
" - ] - }, - "metadata": { - "needs_background": "light" - }, - "output_type": "display_data" - } - ], - "source": [ - "learning_rate = 0.01\n", - "decay = 1e-4\n", - "batch_size = 32\n", - "n_steps_per_epoch = len(X_train) // batch_size\n", - "epochs = np.arange(n_epochs)\n", - "lrs = learning_rate / (1 + decay * epochs * n_steps_per_epoch)\n", - "\n", - "plt.plot(epochs, lrs, \"o-\")\n", - "plt.axis([0, n_epochs - 1, 0, 0.01])\n", - "plt.xlabel(\"Epoch\")\n", - "plt.ylabel(\"Learning Rate\")\n", - "plt.title(\"Power Scheduling\", fontsize=14)\n", - "plt.grid(True)\n", - "plt.show()" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "### 지수 기반 스케줄링" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "```lr = lr0 * 0.1**(epoch / s)```" - ] - }, - { - "cell_type": "code", - "execution_count": 78, - "metadata": {}, - "outputs": [], - "source": [ - "def exponential_decay_fn(epoch):\n", - " return 0.01 * 0.1**(epoch / 20)" - ] - }, - { - "cell_type": "code", - "execution_count": 79, - "metadata": {}, - "outputs": [], - "source": [ - "def exponential_decay(lr0, s):\n", - " def exponential_decay_fn(epoch):\n", - " return lr0 * 0.1**(epoch / s)\n", - " return exponential_decay_fn\n", - "\n", - "exponential_decay_fn = exponential_decay(lr0=0.01, s=20)" - ] - }, - { - "cell_type": "code", - "execution_count": 80, - "metadata": {}, - "outputs": [], - "source": [ - "model = keras.models.Sequential([\n", - " keras.layers.Flatten(input_shape=[28, 28]),\n", - " keras.layers.Dense(300, activation=\"selu\", kernel_initializer=\"lecun_normal\"),\n", - " keras.layers.Dense(100, activation=\"selu\", kernel_initializer=\"lecun_normal\"),\n", - " keras.layers.Dense(10, activation=\"softmax\")\n", - "])\n", - "model.compile(loss=\"sparse_categorical_crossentropy\", optimizer=\"nadam\", metrics=[\"accuracy\"])\n", - "n_epochs = 25" - ] - }, - { - "cell_type": "code", - "execution_count": 81, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "Epoch 1/25\n", - "1719/1719 [==============================] - 8s 4ms/step - loss: 0.8717 - accuracy: 0.7511 - val_loss: 0.9296 - val_accuracy: 0.7502\n", - "Epoch 2/25\n", - "1719/1719 [==============================] - 7s 4ms/step - loss: 0.7778 - accuracy: 0.7756 - val_loss: 0.6632 - val_accuracy: 0.8124\n", - "Epoch 3/25\n", - "1719/1719 [==============================] - 8s 4ms/step - loss: 0.6531 - accuracy: 0.8051 - val_loss: 0.7064 - val_accuracy: 0.7834\n", - "Epoch 4/25\n", - "1719/1719 [==============================] - 7s 4ms/step - loss: 0.5903 - accuracy: 0.8228 - val_loss: 0.5971 - val_accuracy: 0.8258\n", - "Epoch 5/25\n", - "1719/1719 [==============================] - 7s 4ms/step - loss: 0.5431 - accuracy: 0.8379 - val_loss: 0.5389 - val_accuracy: 0.8488\n", - "Epoch 6/25\n", - "1719/1719 [==============================] - 7s 4ms/step - loss: 0.5012 - accuracy: 0.8489 - val_loss: 0.5297 - val_accuracy: 0.8562\n", - "Epoch 7/25\n", - "1719/1719 [==============================] - 7s 4ms/step - loss: 0.4695 - accuracy: 0.8578 - val_loss: 0.5339 - val_accuracy: 0.8446\n", - "Epoch 8/25\n", - "1719/1719 [==============================] - 7s 4ms/step - loss: 0.4328 - accuracy: 0.8667 - val_loss: 0.7335 - val_accuracy: 0.8280\n", - "Epoch 9/25\n", - "1719/1719 [==============================] - 7s 4ms/step - loss: 0.4036 - accuracy: 0.8750 - val_loss: 0.5740 - val_accuracy: 0.8618\n", - "Epoch 10/25\n", - "1719/1719 [==============================] - 7s 4ms/step - loss: 0.3857 - accuracy: 0.8807 - val_loss: 0.4759 - val_accuracy: 0.8672\n", - "Epoch 11/25\n", - "1719/1719 [==============================] - 7s 4ms/step - loss: 0.3601 - accuracy: 0.8864 - val_loss: 0.4779 - val_accuracy: 0.8630\n", - "Epoch 12/25\n", - "1719/1719 [==============================] - 7s 4ms/step - loss: 0.3322 - accuracy: 0.8924 - val_loss: 0.4897 - val_accuracy: 0.8636\n", - "Epoch 13/25\n", - "1719/1719 [==============================] - 7s 4ms/step - loss: 0.3204 - accuracy: 0.8987 - val_loss: 0.4892 - val_accuracy: 0.8690\n", - "Epoch 14/25\n", - "1719/1719 [==============================] - 7s 4ms/step - loss: 0.2968 - accuracy: 0.9039 - val_loss: 0.4638 - val_accuracy: 0.8734\n", - "Epoch 15/25\n", - "1719/1719 [==============================] - 7s 4ms/step - loss: 0.2779 - accuracy: 0.9104 - val_loss: 0.5087 - val_accuracy: 0.8758\n", - "Epoch 16/25\n", - "1719/1719 [==============================] - 7s 4ms/step - loss: 0.2632 - accuracy: 0.9150 - val_loss: 0.4718 - val_accuracy: 0.8770\n", - "Epoch 17/25\n", - "1719/1719 [==============================] - 7s 4ms/step - loss: 0.2479 - accuracy: 0.9191 - val_loss: 0.5167 - val_accuracy: 0.8774\n", - "Epoch 18/25\n", - "1719/1719 [==============================] - 7s 4ms/step - loss: 0.2369 - accuracy: 0.9243 - val_loss: 0.4961 - val_accuracy: 0.8786\n", - "Epoch 19/25\n", - "1719/1719 [==============================] - 7s 4ms/step - loss: 0.2202 - accuracy: 0.9292 - val_loss: 0.5280 - val_accuracy: 0.8834\n", - "Epoch 20/25\n", - "1719/1719 [==============================] - 7s 4ms/step - loss: 0.2113 - accuracy: 0.9312 - val_loss: 0.5491 - val_accuracy: 0.8758\n", - "Epoch 21/25\n", - "1719/1719 [==============================] - 7s 4ms/step - loss: 0.1985 - accuracy: 0.9363 - val_loss: 0.5469 - val_accuracy: 0.8790\n", - "Epoch 22/25\n", - "1719/1719 [==============================] - 7s 4ms/step - loss: 0.1878 - accuracy: 0.9393 - val_loss: 0.5512 - val_accuracy: 0.8822\n", - "Epoch 23/25\n", - "1719/1719 [==============================] - 7s 4ms/step - loss: 0.1781 - accuracy: 0.9425 - val_loss: 0.5817 - val_accuracy: 0.8848\n", - "Epoch 24/25\n", - "1719/1719 [==============================] - 8s 4ms/step - loss: 0.1709 - accuracy: 0.9450 - val_loss: 0.5742 - val_accuracy: 0.8818\n", - "Epoch 25/25\n", - "1719/1719 [==============================] - 7s 4ms/step - loss: 0.1638 - accuracy: 0.9484 - val_loss: 0.6042 - val_accuracy: 0.8842\n" - ] - } - ], - "source": [ - "lr_scheduler = keras.callbacks.LearningRateScheduler(exponential_decay_fn)\n", - "history = model.fit(X_train_scaled, y_train, epochs=n_epochs,\n", - " validation_data=(X_valid_scaled, y_valid),\n", - " callbacks=[lr_scheduler])" - ] - }, - { - "cell_type": "code", - "execution_count": 82, - "metadata": {}, - "outputs": [ - { - "data": { - "image/png": "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\n", - "text/plain": [ - "
" - ] - }, - "metadata": { - "needs_background": "light" - }, - "output_type": "display_data" - } - ], - "source": [ - "plt.plot(history.epoch, history.history[\"lr\"], \"o-\")\n", - "plt.axis([0, n_epochs - 1, 0, 0.011])\n", - "plt.xlabel(\"Epoch\")\n", - "plt.ylabel(\"Learning Rate\")\n", - "plt.title(\"Exponential Scheduling\", fontsize=14)\n", - "plt.grid(True)\n", - "plt.show()" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "이 스케줄 함수는 두 번째 매개변수로 현재 학습률을 받을 수 있습니다:" - ] - }, - { - "cell_type": "code", - "execution_count": 83, - "metadata": {}, - "outputs": [], - "source": [ - "def exponential_decay_fn(epoch, lr):\n", - " return lr * 0.1**(1 / 20)" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "에포크가 아니라 반복마다 학습률을 업데이트하려면 사용자 정의 콜백 클래스를 작성해야 합니다:" - ] - }, - { - "cell_type": "code", - "execution_count": 84, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "Epoch 1/25\n", - "1719/1719 [==============================] - 9s 5ms/step - loss: 0.7848 - accuracy: 0.7711 - val_loss: 0.8494 - val_accuracy: 0.7580\n", - "Epoch 2/25\n", - "1719/1719 [==============================] - 9s 5ms/step - loss: 0.6253 - accuracy: 0.8057 - val_loss: 0.7549 - val_accuracy: 0.7640\n", - "Epoch 3/25\n", - "1719/1719 [==============================] - 9s 5ms/step - loss: 0.5786 - accuracy: 0.8196 - val_loss: 0.6209 - val_accuracy: 0.8128\n", - "Epoch 4/25\n", - "1719/1719 [==============================] - 9s 5ms/step - loss: 0.5178 - accuracy: 0.8388 - val_loss: 0.5581 - val_accuracy: 0.8470\n", - "Epoch 5/25\n", - "1719/1719 [==============================] - 9s 5ms/step - loss: 0.4988 - accuracy: 0.8480 - val_loss: 0.5315 - val_accuracy: 0.8428\n", - "Epoch 6/25\n", - "1719/1719 [==============================] - 9s 5ms/step - loss: 0.4408 - accuracy: 0.8641 - val_loss: 0.4676 - val_accuracy: 0.8598\n", - "Epoch 7/25\n", - "1719/1719 [==============================] - 9s 5ms/step - loss: 0.4063 - accuracy: 0.8689 - val_loss: 0.6225 - val_accuracy: 0.8402\n", - "Epoch 8/25\n", - "1719/1719 [==============================] - 9s 5ms/step - loss: 0.3650 - accuracy: 0.8791 - val_loss: 0.4607 - val_accuracy: 0.8638\n", - "Epoch 9/25\n", - "1719/1719 [==============================] - 9s 5ms/step - loss: 0.3442 - accuracy: 0.8854 - val_loss: 0.4588 - val_accuracy: 0.8570\n", - "Epoch 10/25\n", - "1719/1719 [==============================] - 9s 5ms/step - loss: 0.3196 - accuracy: 0.8927 - val_loss: 0.4427 - val_accuracy: 0.8814\n", - "Epoch 11/25\n", - "1719/1719 [==============================] - 9s 5ms/step - loss: 0.2950 - accuracy: 0.8996 - val_loss: 0.4303 - val_accuracy: 0.8810\n", - "Epoch 12/25\n", - "1719/1719 [==============================] - 9s 5ms/step - loss: 0.2699 - accuracy: 0.9077 - val_loss: 0.4377 - val_accuracy: 0.8674\n", - "Epoch 13/25\n", - "1719/1719 [==============================] - 9s 5ms/step - loss: 0.2528 - accuracy: 0.9119 - val_loss: 0.4323 - val_accuracy: 0.8862\n", - "Epoch 14/25\n", - "1719/1719 [==============================] - 9s 5ms/step - loss: 0.2340 - accuracy: 0.9187 - val_loss: 0.4424 - val_accuracy: 0.8794\n", - "Epoch 15/25\n", - "1719/1719 [==============================] - 9s 5ms/step - loss: 0.2159 - accuracy: 0.9236 - val_loss: 0.4204 - val_accuracy: 0.8878\n", - "Epoch 16/25\n", - "1719/1719 [==============================] - 9s 5ms/step - loss: 0.2008 - accuracy: 0.9302 - val_loss: 0.4568 - val_accuracy: 0.8928\n", - "Epoch 17/25\n", - "1719/1719 [==============================] - 9s 5ms/step - loss: 0.1866 - accuracy: 0.9343 - val_loss: 0.4430 - val_accuracy: 0.8920\n", - "Epoch 18/25\n", - "1719/1719 [==============================] - 9s 5ms/step - loss: 0.1738 - accuracy: 0.9395 - val_loss: 0.4850 - val_accuracy: 0.8928\n", - "Epoch 19/25\n", - "1719/1719 [==============================] - 9s 5ms/step - loss: 0.1589 - accuracy: 0.9454 - val_loss: 0.4853 - val_accuracy: 0.8948\n", - "Epoch 20/25\n", - "1719/1719 [==============================] - 9s 5ms/step - loss: 0.1488 - accuracy: 0.9493 - val_loss: 0.4679 - val_accuracy: 0.8890\n", - "Epoch 21/25\n", - "1719/1719 [==============================] - 9s 5ms/step - loss: 0.1392 - accuracy: 0.9526 - val_loss: 0.5147 - val_accuracy: 0.8906\n", - "Epoch 22/25\n", - "1719/1719 [==============================] - 9s 5ms/step - loss: 0.1298 - accuracy: 0.9567 - val_loss: 0.5229 - val_accuracy: 0.8906\n", - "Epoch 23/25\n", - "1719/1719 [==============================] - 9s 5ms/step - loss: 0.1223 - accuracy: 0.9599 - val_loss: 0.5352 - val_accuracy: 0.8884\n", - "Epoch 24/25\n", - "1719/1719 [==============================] - 9s 5ms/step - loss: 0.1117 - accuracy: 0.9631 - val_loss: 0.5800 - val_accuracy: 0.8884\n", - "Epoch 25/25\n", - "1719/1719 [==============================] - 9s 5ms/step - loss: 0.1068 - accuracy: 0.9648 - val_loss: 0.5695 - val_accuracy: 0.8898\n" - ] - } - ], - "source": [ - "K = keras.backend\n", - "\n", - "class ExponentialDecay(keras.callbacks.Callback):\n", - " def __init__(self, s=40000):\n", - " super().__init__()\n", - " self.s = s\n", - "\n", - " def on_batch_begin(self, batch, logs=None):\n", - " # 노트: 에포크마다 `batch` 매개변수가 재설정됩니다\n", - " lr = K.get_value(self.model.optimizer.lr)\n", - " K.set_value(self.model.optimizer.lr, lr * 0.1**(1 / s))\n", - "\n", - " def on_epoch_end(self, epoch, logs=None):\n", - " logs = logs or {}\n", - " logs['lr'] = K.get_value(self.model.optimizer.lr)\n", - "\n", - "model = keras.models.Sequential([\n", - " keras.layers.Flatten(input_shape=[28, 28]),\n", - " keras.layers.Dense(300, activation=\"selu\", kernel_initializer=\"lecun_normal\"),\n", - " keras.layers.Dense(100, activation=\"selu\", kernel_initializer=\"lecun_normal\"),\n", - " keras.layers.Dense(10, activation=\"softmax\")\n", - "])\n", - "lr0 = 0.01\n", - "optimizer = keras.optimizers.Nadam(lr=lr0)\n", - "model.compile(loss=\"sparse_categorical_crossentropy\", optimizer=optimizer, metrics=[\"accuracy\"])\n", - "n_epochs = 25\n", - "\n", - "s = 20 * len(X_train) // 32 # 20 에포크 동안 스텝 횟수 (배치 크기 = 32)\n", - "exp_decay = ExponentialDecay(s)\n", - "history = model.fit(X_train_scaled, y_train, epochs=n_epochs,\n", - " validation_data=(X_valid_scaled, y_valid),\n", - " callbacks=[exp_decay])" - ] - }, - { - "cell_type": "code", - "execution_count": 85, - "metadata": {}, - "outputs": [], - "source": [ - "n_steps = n_epochs * len(X_train) // 32\n", - "steps = np.arange(n_steps)\n", - "lrs = lr0 * 0.1**(steps / s)" - ] - }, - { - "cell_type": "code", - "execution_count": 86, - "metadata": { - "scrolled": true - }, - "outputs": [ - { - "data": { - "image/png": "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\n", - "text/plain": [ - "
" - ] - }, - "metadata": { - "needs_background": "light" - }, - "output_type": "display_data" - } - ], - "source": [ - "plt.plot(steps, lrs, \"-\", linewidth=2)\n", - "plt.axis([0, n_steps - 1, 0, lr0 * 1.1])\n", - "plt.xlabel(\"Batch\")\n", - "plt.ylabel(\"Learning Rate\")\n", - "plt.title(\"Exponential Scheduling (per batch)\", fontsize=14)\n", - "plt.grid(True)\n", - "plt.show()" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "### 기간별 고정 스케줄링" - ] - }, - { - "cell_type": "code", - "execution_count": 87, - "metadata": {}, - "outputs": [], - "source": [ - "def piecewise_constant_fn(epoch):\n", - " if epoch < 5:\n", - " return 0.01\n", - " elif epoch < 15:\n", - " return 0.005\n", - " else:\n", - " return 0.001" - ] - }, - { - "cell_type": "code", - "execution_count": 88, - "metadata": {}, - "outputs": [], - "source": [ - "def piecewise_constant(boundaries, values):\n", - " boundaries = np.array([0] + boundaries)\n", - " values = np.array(values)\n", - " def piecewise_constant_fn(epoch):\n", - " return values[np.argmax(boundaries > epoch) - 1]\n", - " return piecewise_constant_fn\n", - "\n", - "piecewise_constant_fn = piecewise_constant([5, 15], [0.01, 0.005, 0.001])" - ] - }, - { - "cell_type": "code", - "execution_count": 89, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "Epoch 1/25\n", - "1719/1719 [==============================] - 8s 5ms/step - loss: 0.8830 - accuracy: 0.7497 - val_loss: 1.0209 - val_accuracy: 0.7004\n", - "Epoch 2/25\n", - "1719/1719 [==============================] - 8s 4ms/step - loss: 0.9927 - accuracy: 0.6931 - val_loss: 0.8880 - val_accuracy: 0.7178\n", - "Epoch 3/25\n", - "1719/1719 [==============================] - 8s 4ms/step - loss: 0.8996 - accuracy: 0.7185 - val_loss: 1.0695 - val_accuracy: 0.6856\n", - "Epoch 4/25\n", - "1719/1719 [==============================] - 8s 4ms/step - loss: 0.9428 - accuracy: 0.7076 - val_loss: 0.9724 - val_accuracy: 0.7206\n", - "Epoch 5/25\n", - "1719/1719 [==============================] - 8s 4ms/step - loss: 0.9195 - accuracy: 0.7078 - val_loss: 1.2058 - val_accuracy: 0.7076\n", - "Epoch 6/25\n", - "1719/1719 [==============================] - 8s 4ms/step - loss: 0.6494 - accuracy: 0.7722 - val_loss: 0.6743 - val_accuracy: 0.7360\n", - "Epoch 7/25\n", - "1719/1719 [==============================] - 8s 4ms/step - loss: 0.6002 - accuracy: 0.7929 - val_loss: 0.7192 - val_accuracy: 0.7718\n", - "Epoch 8/25\n", - "1719/1719 [==============================] - 8s 4ms/step - loss: 0.5850 - accuracy: 0.7999 - val_loss: 0.6331 - val_accuracy: 0.7328\n", - "Epoch 9/25\n", - "1719/1719 [==============================] - 8s 4ms/step - loss: 0.5570 - accuracy: 0.8213 - val_loss: 0.6222 - val_accuracy: 0.7668\n", - "Epoch 10/25\n", - "1719/1719 [==============================] - 8s 4ms/step - loss: 0.5511 - accuracy: 0.8165 - val_loss: 0.6102 - val_accuracy: 0.8204\n", - "Epoch 11/25\n", - "1719/1719 [==============================] - 8s 4ms/step - loss: 0.5360 - accuracy: 0.8182 - val_loss: 0.6345 - val_accuracy: 0.8292\n", - "Epoch 12/25\n", - "1719/1719 [==============================] - 7s 4ms/step - loss: 0.5291 - accuracy: 0.8363 - val_loss: 0.6106 - val_accuracy: 0.8334\n", - "Epoch 13/25\n", - "1719/1719 [==============================] - 8s 4ms/step - loss: 0.5104 - accuracy: 0.8400 - val_loss: 0.6339 - val_accuracy: 0.8244\n", - "Epoch 14/25\n", - "1719/1719 [==============================] - 8s 4ms/step - loss: 0.4812 - accuracy: 0.8557 - val_loss: 0.6163 - val_accuracy: 0.7798\n", - "Epoch 15/25\n", - "1719/1719 [==============================] - 8s 4ms/step - loss: 0.4852 - accuracy: 0.8543 - val_loss: 0.8202 - val_accuracy: 0.8402\n", - "Epoch 16/25\n", - "1719/1719 [==============================] - 8s 4ms/step - loss: 0.3420 - accuracy: 0.8901 - val_loss: 0.5218 - val_accuracy: 0.8708\n", - "Epoch 17/25\n", - "1719/1719 [==============================] - 8s 4ms/step - loss: 0.3108 - accuracy: 0.9007 - val_loss: 0.5677 - val_accuracy: 0.8652\n", - "Epoch 18/25\n", - "1719/1719 [==============================] - 8s 4ms/step - loss: 0.2922 - accuracy: 0.9061 - val_loss: 0.6053 - val_accuracy: 0.8768\n", - "Epoch 19/25\n", - "1719/1719 [==============================] - 8s 4ms/step - loss: 0.2756 - accuracy: 0.9134 - val_loss: 0.5898 - val_accuracy: 0.8774\n", - "Epoch 20/25\n", - "1719/1719 [==============================] - 8s 4ms/step - loss: 0.2668 - accuracy: 0.9170 - val_loss: 0.5603 - val_accuracy: 0.8854\n", - "Epoch 21/25\n", - "1719/1719 [==============================] - 8s 4ms/step - loss: 0.2574 - accuracy: 0.9201 - val_loss: 0.5782 - val_accuracy: 0.8792\n", - "Epoch 22/25\n", - "1719/1719 [==============================] - 8s 4ms/step - loss: 0.2518 - accuracy: 0.9230 - val_loss: 0.5958 - val_accuracy: 0.8758\n", - "Epoch 23/25\n", - "1719/1719 [==============================] - 8s 4ms/step - loss: 0.2396 - accuracy: 0.9254 - val_loss: 0.6070 - val_accuracy: 0.8762\n", - "Epoch 24/25\n", - "1719/1719 [==============================] - 8s 4ms/step - loss: 0.2405 - accuracy: 0.9264 - val_loss: 0.6612 - val_accuracy: 0.8788\n", - "Epoch 25/25\n", - "1719/1719 [==============================] - 8s 4ms/step - loss: 0.2244 - accuracy: 0.9291 - val_loss: 0.6447 - val_accuracy: 0.8786\n" - ] - } - ], - "source": [ - "lr_scheduler = keras.callbacks.LearningRateScheduler(piecewise_constant_fn)\n", - "\n", - "model = keras.models.Sequential([\n", - " keras.layers.Flatten(input_shape=[28, 28]),\n", - " keras.layers.Dense(300, activation=\"selu\", kernel_initializer=\"lecun_normal\"),\n", - " keras.layers.Dense(100, activation=\"selu\", kernel_initializer=\"lecun_normal\"),\n", - " keras.layers.Dense(10, activation=\"softmax\")\n", - "])\n", - "model.compile(loss=\"sparse_categorical_crossentropy\", optimizer=\"nadam\", metrics=[\"accuracy\"])\n", - "n_epochs = 25\n", - "history = model.fit(X_train_scaled, y_train, epochs=n_epochs,\n", - " validation_data=(X_valid_scaled, y_valid),\n", - " callbacks=[lr_scheduler])" - ] - }, - { - "cell_type": "code", - "execution_count": 90, - "metadata": {}, - "outputs": [ - { - "data": { - "image/png": "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\n", - "text/plain": [ - "
" - ] - }, - "metadata": { - "needs_background": "light" - }, - "output_type": "display_data" - } - ], - "source": [ - "plt.plot(history.epoch, [piecewise_constant_fn(epoch) for epoch in history.epoch], \"o-\")\n", - "plt.axis([0, n_epochs - 1, 0, 0.011])\n", - "plt.xlabel(\"Epoch\")\n", - "plt.ylabel(\"Learning Rate\")\n", - "plt.title(\"Piecewise Constant Scheduling\", fontsize=14)\n", - "plt.grid(True)\n", - "plt.show()" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "### 성능 기반 스케줄링" - ] - }, - { - "cell_type": "code", - "execution_count": 91, - "metadata": {}, - "outputs": [], - "source": [ - "tf.random.set_seed(42)\n", - "np.random.seed(42)" - ] - }, - { - "cell_type": "code", - "execution_count": 92, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "Epoch 1/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.5882 - accuracy: 0.8079 - val_loss: 0.4696 - val_accuracy: 0.8538\n", - "Epoch 2/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.4975 - accuracy: 0.8391 - val_loss: 0.5668 - val_accuracy: 0.8406\n", - "Epoch 3/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.5165 - accuracy: 0.8419 - val_loss: 0.5295 - val_accuracy: 0.8496\n", - "Epoch 4/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.5144 - accuracy: 0.8454 - val_loss: 0.5411 - val_accuracy: 0.8480\n", - "Epoch 5/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.5096 - accuracy: 0.8496 - val_loss: 0.4733 - val_accuracy: 0.8490\n", - "Epoch 6/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.5276 - accuracy: 0.8515 - val_loss: 0.7935 - val_accuracy: 0.8416\n", - "Epoch 7/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.3111 - accuracy: 0.8925 - val_loss: 0.4018 - val_accuracy: 0.8694\n", - "Epoch 8/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2627 - accuracy: 0.9045 - val_loss: 0.4398 - val_accuracy: 0.8712\n", - "Epoch 9/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2391 - accuracy: 0.9116 - val_loss: 0.3970 - val_accuracy: 0.8890\n", - "Epoch 10/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2252 - accuracy: 0.9175 - val_loss: 0.4090 - val_accuracy: 0.8892\n", - "Epoch 11/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2109 - accuracy: 0.9216 - val_loss: 0.4515 - val_accuracy: 0.8844\n", - "Epoch 12/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2034 - accuracy: 0.9247 - val_loss: 0.4781 - val_accuracy: 0.8808\n", - "Epoch 13/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.1892 - accuracy: 0.9294 - val_loss: 0.4578 - val_accuracy: 0.8904\n", - "Epoch 14/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.1850 - accuracy: 0.9307 - val_loss: 0.4853 - val_accuracy: 0.8808\n", - "Epoch 15/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.1252 - accuracy: 0.9504 - val_loss: 0.4423 - val_accuracy: 0.8902\n", - "Epoch 16/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.1088 - accuracy: 0.9579 - val_loss: 0.4663 - val_accuracy: 0.8946\n", - "Epoch 17/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.0992 - accuracy: 0.9620 - val_loss: 0.4872 - val_accuracy: 0.8932\n", - "Epoch 18/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.0938 - accuracy: 0.9641 - val_loss: 0.5198 - val_accuracy: 0.8862\n", - "Epoch 19/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.0883 - accuracy: 0.9660 - val_loss: 0.5091 - val_accuracy: 0.8922\n", - "Epoch 20/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.0668 - accuracy: 0.9752 - val_loss: 0.5149 - val_accuracy: 0.8946\n", - "Epoch 21/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.0608 - accuracy: 0.9779 - val_loss: 0.5289 - val_accuracy: 0.8946\n", - "Epoch 22/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.0575 - accuracy: 0.9786 - val_loss: 0.5383 - val_accuracy: 0.8928\n", - "Epoch 23/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.0538 - accuracy: 0.9801 - val_loss: 0.5474 - val_accuracy: 0.8928\n", - "Epoch 24/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.0506 - accuracy: 0.9818 - val_loss: 0.5690 - val_accuracy: 0.8908\n", - "Epoch 25/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.0425 - accuracy: 0.9858 - val_loss: 0.5667 - val_accuracy: 0.8904\n" - ] - } - ], - "source": [ - "lr_scheduler = keras.callbacks.ReduceLROnPlateau(factor=0.5, patience=5)\n", - "\n", - "model = keras.models.Sequential([\n", - " keras.layers.Flatten(input_shape=[28, 28]),\n", - " keras.layers.Dense(300, activation=\"selu\", kernel_initializer=\"lecun_normal\"),\n", - " keras.layers.Dense(100, activation=\"selu\", kernel_initializer=\"lecun_normal\"),\n", - " keras.layers.Dense(10, activation=\"softmax\")\n", - "])\n", - "optimizer = keras.optimizers.SGD(lr=0.02, momentum=0.9)\n", - "model.compile(loss=\"sparse_categorical_crossentropy\", optimizer=optimizer, metrics=[\"accuracy\"])\n", - "n_epochs = 25\n", - "history = model.fit(X_train_scaled, y_train, epochs=n_epochs,\n", - " validation_data=(X_valid_scaled, y_valid),\n", - " callbacks=[lr_scheduler])" - ] - }, - { - "cell_type": "code", - "execution_count": 93, - "metadata": {}, - "outputs": [ - { - "data": { - "image/png": "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\n", - "text/plain": [ - "
" - ] - }, - "metadata": { - "needs_background": "light" - }, - "output_type": "display_data" - } - ], - "source": [ - "plt.plot(history.epoch, history.history[\"lr\"], \"bo-\")\n", - "plt.xlabel(\"Epoch\")\n", - "plt.ylabel(\"Learning Rate\", color='b')\n", - "plt.tick_params('y', colors='b')\n", - "plt.gca().set_xlim(0, n_epochs - 1)\n", - "plt.grid(True)\n", - "\n", - "ax2 = plt.gca().twinx()\n", - "ax2.plot(history.epoch, history.history[\"val_loss\"], \"r^-\")\n", - "ax2.set_ylabel('Validation Loss', color='r')\n", - "ax2.tick_params('y', colors='r')\n", - "\n", - "plt.title(\"Reduce LR on Plateau\", fontsize=14)\n", - "plt.show()" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "### tf.keras 스케줄러" - ] - }, - { - "cell_type": "code", - "execution_count": 94, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "Epoch 1/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.4894 - accuracy: 0.8277 - val_loss: 0.4096 - val_accuracy: 0.8592\n", - "Epoch 2/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.3820 - accuracy: 0.8650 - val_loss: 0.3742 - val_accuracy: 0.8700\n", - "Epoch 3/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.3487 - accuracy: 0.8767 - val_loss: 0.3736 - val_accuracy: 0.8686\n", - "Epoch 4/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.3265 - accuracy: 0.8838 - val_loss: 0.3496 - val_accuracy: 0.8798\n", - "Epoch 5/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.3105 - accuracy: 0.8899 - val_loss: 0.3434 - val_accuracy: 0.8800\n", - "Epoch 6/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2959 - accuracy: 0.8950 - val_loss: 0.3415 - val_accuracy: 0.8808\n", - "Epoch 7/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2855 - accuracy: 0.8987 - val_loss: 0.3354 - val_accuracy: 0.8818\n", - "Epoch 8/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2761 - accuracy: 0.9016 - val_loss: 0.3366 - val_accuracy: 0.8810\n", - "Epoch 9/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2678 - accuracy: 0.9053 - val_loss: 0.3265 - val_accuracy: 0.8852\n", - "Epoch 10/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2608 - accuracy: 0.9069 - val_loss: 0.3240 - val_accuracy: 0.8848\n", - "Epoch 11/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2551 - accuracy: 0.9088 - val_loss: 0.3251 - val_accuracy: 0.8868\n", - "Epoch 12/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2497 - accuracy: 0.9126 - val_loss: 0.3302 - val_accuracy: 0.8810\n", - "Epoch 13/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2449 - accuracy: 0.9136 - val_loss: 0.3218 - val_accuracy: 0.8872\n", - "Epoch 14/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2415 - accuracy: 0.9147 - val_loss: 0.3222 - val_accuracy: 0.8860\n", - "Epoch 15/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2375 - accuracy: 0.9167 - val_loss: 0.3208 - val_accuracy: 0.8876\n", - "Epoch 16/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2343 - accuracy: 0.9179 - val_loss: 0.3185 - val_accuracy: 0.8882\n", - "Epoch 17/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2317 - accuracy: 0.9186 - val_loss: 0.3198 - val_accuracy: 0.8890\n", - "Epoch 18/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2291 - accuracy: 0.9199 - val_loss: 0.3169 - val_accuracy: 0.8904\n", - "Epoch 19/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2269 - accuracy: 0.9206 - val_loss: 0.3197 - val_accuracy: 0.8888\n", - "Epoch 20/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2250 - accuracy: 0.9220 - val_loss: 0.3169 - val_accuracy: 0.8902\n", - "Epoch 21/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2229 - accuracy: 0.9224 - val_loss: 0.3180 - val_accuracy: 0.8904\n", - "Epoch 22/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2216 - accuracy: 0.9225 - val_loss: 0.3163 - val_accuracy: 0.8912\n", - "Epoch 23/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2201 - accuracy: 0.9233 - val_loss: 0.3171 - val_accuracy: 0.8906\n", - "Epoch 24/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2188 - accuracy: 0.9243 - val_loss: 0.3166 - val_accuracy: 0.8908\n", - "Epoch 25/25\n", - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2179 - accuracy: 0.9243 - val_loss: 0.3165 - val_accuracy: 0.8904\n" - ] - } - ], - "source": [ - "model = keras.models.Sequential([\n", - " keras.layers.Flatten(input_shape=[28, 28]),\n", - " keras.layers.Dense(300, activation=\"selu\", kernel_initializer=\"lecun_normal\"),\n", - " keras.layers.Dense(100, activation=\"selu\", kernel_initializer=\"lecun_normal\"),\n", - " keras.layers.Dense(10, activation=\"softmax\")\n", - "])\n", - "s = 20 * len(X_train) // 32 # number of steps in 20 epochs (batch size = 32)\n", - "learning_rate = keras.optimizers.schedules.ExponentialDecay(0.01, s, 0.1)\n", - "optimizer = keras.optimizers.SGD(learning_rate)\n", - "model.compile(loss=\"sparse_categorical_crossentropy\", optimizer=optimizer, metrics=[\"accuracy\"])\n", - "n_epochs = 25\n", - "history = model.fit(X_train_scaled, y_train, epochs=n_epochs,\n", - " validation_data=(X_valid_scaled, y_valid))" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "구간별 고정 스케줄링은 다음을 사용하세요:" - ] - }, - { - "cell_type": "code", - "execution_count": 95, - "metadata": {}, - "outputs": [], - "source": [ - "learning_rate = keras.optimizers.schedules.PiecewiseConstantDecay(\n", - " boundaries=[5. * n_steps_per_epoch, 15. * n_steps_per_epoch],\n", - " values=[0.01, 0.005, 0.001])" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "### 1사이클 스케줄링" - ] - }, - { - "cell_type": "code", - "execution_count": 96, - "metadata": {}, - "outputs": [], - "source": [ - "K = keras.backend\n", - "\n", - "class ExponentialLearningRate(keras.callbacks.Callback):\n", - " def __init__(self, factor):\n", - " self.factor = factor\n", - " self.rates = []\n", - " self.losses = []\n", - " def on_batch_end(self, batch, logs):\n", - " self.rates.append(K.get_value(self.model.optimizer.lr))\n", - " self.losses.append(logs[\"loss\"])\n", - " K.set_value(self.model.optimizer.lr, self.model.optimizer.lr * self.factor)\n", - "\n", - "def find_learning_rate(model, X, y, epochs=1, batch_size=32, min_rate=10**-5, max_rate=10):\n", - " init_weights = model.get_weights()\n", - " iterations = len(X) // batch_size * epochs\n", - " factor = np.exp(np.log(max_rate / min_rate) / iterations)\n", - " init_lr = K.get_value(model.optimizer.lr)\n", - " K.set_value(model.optimizer.lr, min_rate)\n", - " exp_lr = ExponentialLearningRate(factor)\n", - " history = model.fit(X, y, epochs=epochs, batch_size=batch_size,\n", - " callbacks=[exp_lr])\n", - " K.set_value(model.optimizer.lr, init_lr)\n", - " model.set_weights(init_weights)\n", - " return exp_lr.rates, exp_lr.losses\n", - "\n", - "def plot_lr_vs_loss(rates, losses):\n", - " plt.plot(rates, losses)\n", - " plt.gca().set_xscale('log')\n", - " plt.hlines(min(losses), min(rates), max(rates))\n", - " plt.axis([min(rates), max(rates), min(losses), (losses[0] + min(losses)) / 2])\n", - " plt.xlabel(\"Learning rate\")\n", - " plt.ylabel(\"Loss\")" - ] - }, - { - "cell_type": "code", - "execution_count": 97, - "metadata": {}, - "outputs": [], - "source": [ - "tf.random.set_seed(42)\n", - "np.random.seed(42)\n", - "\n", - "model = keras.models.Sequential([\n", - " keras.layers.Flatten(input_shape=[28, 28]),\n", - " keras.layers.Dense(300, activation=\"selu\", kernel_initializer=\"lecun_normal\"),\n", - " keras.layers.Dense(100, activation=\"selu\", kernel_initializer=\"lecun_normal\"),\n", - " keras.layers.Dense(10, activation=\"softmax\")\n", - "])\n", - "model.compile(loss=\"sparse_categorical_crossentropy\",\n", - " optimizer=keras.optimizers.SGD(lr=1e-3),\n", - " metrics=[\"accuracy\"])" - ] - }, - { - "cell_type": "code", - "execution_count": 98, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "430/430 [==============================] - 2s 4ms/step - loss: nan - accuracy: 0.3859 \n" - ] - }, - { - "data": { - "image/png": "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\n", - "text/plain": [ - "
" - ] - }, - "metadata": { - "needs_background": "light" - }, - "output_type": "display_data" - } - ], - "source": [ - "batch_size = 128\n", - "rates, losses = find_learning_rate(model, X_train_scaled, y_train, epochs=1, batch_size=batch_size)\n", - "plot_lr_vs_loss(rates, losses)" - ] - }, - { - "cell_type": "code", - "execution_count": 99, - "metadata": {}, - "outputs": [], - "source": [ - "class OneCycleScheduler(keras.callbacks.Callback):\n", - " def __init__(self, iterations, max_rate, start_rate=None,\n", - " last_iterations=None, last_rate=None):\n", - " self.iterations = iterations\n", - " self.max_rate = max_rate\n", - " self.start_rate = start_rate or max_rate / 10\n", - " self.last_iterations = last_iterations or iterations // 10 + 1\n", - " self.half_iteration = (iterations - self.last_iterations) // 2\n", - " self.last_rate = last_rate or self.start_rate / 1000\n", - " self.iteration = 0\n", - " def _interpolate(self, iter1, iter2, rate1, rate2):\n", - " return ((rate2 - rate1) * (self.iteration - iter1)\n", - " / (iter2 - iter1) + rate1)\n", - " def on_batch_begin(self, batch, logs):\n", - " if self.iteration < self.half_iteration:\n", - " rate = self._interpolate(0, self.half_iteration, self.start_rate, self.max_rate)\n", - " elif self.iteration < 2 * self.half_iteration:\n", - " rate = self._interpolate(self.half_iteration, 2 * self.half_iteration,\n", - " self.max_rate, self.start_rate)\n", - " else:\n", - " rate = self._interpolate(2 * self.half_iteration, self.iterations,\n", - " self.start_rate, self.last_rate)\n", - " rate = max(rate, self.last_rate)\n", - " self.iteration += 1\n", - " K.set_value(self.model.optimizer.lr, rate)" - ] - }, - { - "cell_type": "code", - "execution_count": 100, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "Epoch 1/25\n", - "430/430 [==============================] - 2s 4ms/step - loss: 0.6572 - accuracy: 0.7740 - val_loss: 0.4872 - val_accuracy: 0.8338\n", - "Epoch 2/25\n", - "430/430 [==============================] - 1s 3ms/step - loss: 0.4581 - accuracy: 0.8397 - val_loss: 0.4274 - val_accuracy: 0.8524\n", - "Epoch 3/25\n", - "430/430 [==============================] - 1s 3ms/step - loss: 0.4121 - accuracy: 0.8545 - val_loss: 0.4116 - val_accuracy: 0.8588\n", - "Epoch 4/25\n", - "430/430 [==============================] - 1s 3ms/step - loss: 0.3837 - accuracy: 0.8641 - val_loss: 0.3870 - val_accuracy: 0.8686\n", - "Epoch 5/25\n", - "430/430 [==============================] - 1s 3ms/step - loss: 0.3639 - accuracy: 0.8717 - val_loss: 0.3765 - val_accuracy: 0.8676\n", - "Epoch 6/25\n", - "430/430 [==============================] - 1s 3ms/step - loss: 0.3457 - accuracy: 0.8774 - val_loss: 0.3742 - val_accuracy: 0.8708\n", - "Epoch 7/25\n", - "430/430 [==============================] - 1s 3ms/step - loss: 0.3330 - accuracy: 0.8811 - val_loss: 0.3634 - val_accuracy: 0.8704\n", - "Epoch 8/25\n", - "430/430 [==============================] - 1s 3ms/step - loss: 0.3185 - accuracy: 0.8862 - val_loss: 0.3958 - val_accuracy: 0.8608\n", - "Epoch 9/25\n", - "430/430 [==============================] - 1s 3ms/step - loss: 0.3065 - accuracy: 0.8893 - val_loss: 0.3483 - val_accuracy: 0.8762\n", - "Epoch 10/25\n", - "430/430 [==============================] - 1s 3ms/step - loss: 0.2945 - accuracy: 0.8924 - val_loss: 0.3396 - val_accuracy: 0.8812\n", - "Epoch 11/25\n", - "430/430 [==============================] - 2s 4ms/step - loss: 0.2838 - accuracy: 0.8963 - val_loss: 0.3460 - val_accuracy: 0.8796\n", - "Epoch 12/25\n", - "430/430 [==============================] - 1s 3ms/step - loss: 0.2709 - accuracy: 0.9023 - val_loss: 0.3644 - val_accuracy: 0.8696\n", - "Epoch 13/25\n", - "430/430 [==============================] - 1s 3ms/step - loss: 0.2536 - accuracy: 0.9081 - val_loss: 0.3350 - val_accuracy: 0.8838\n", - "Epoch 14/25\n", - "430/430 [==============================] - 1s 3ms/step - loss: 0.2405 - accuracy: 0.9134 - val_loss: 0.3466 - val_accuracy: 0.8812\n", - "Epoch 15/25\n", - "430/430 [==============================] - 1s 3ms/step - loss: 0.2280 - accuracy: 0.9183 - val_loss: 0.3260 - val_accuracy: 0.8840\n", - "Epoch 16/25\n", - "430/430 [==============================] - 1s 3ms/step - loss: 0.2160 - accuracy: 0.9234 - val_loss: 0.3292 - val_accuracy: 0.8834\n", - "Epoch 17/25\n", - "430/430 [==============================] - 1s 3ms/step - loss: 0.2062 - accuracy: 0.9264 - val_loss: 0.3354 - val_accuracy: 0.8862\n", - "Epoch 18/25\n", - "430/430 [==============================] - 1s 3ms/step - loss: 0.1978 - accuracy: 0.9305 - val_loss: 0.3236 - val_accuracy: 0.8906\n", - "Epoch 19/25\n", - "430/430 [==============================] - 1s 3ms/step - loss: 0.1892 - accuracy: 0.9337 - val_loss: 0.3233 - val_accuracy: 0.8904\n", - "Epoch 20/25\n", - "430/430 [==============================] - 2s 4ms/step - loss: 0.1821 - accuracy: 0.9369 - val_loss: 0.3221 - val_accuracy: 0.8926\n", - "Epoch 21/25\n", - "430/430 [==============================] - 1s 3ms/step - loss: 0.1752 - accuracy: 0.9401 - val_loss: 0.3215 - val_accuracy: 0.8904\n", - "Epoch 22/25\n", - "430/430 [==============================] - 1s 3ms/step - loss: 0.1701 - accuracy: 0.9418 - val_loss: 0.3180 - val_accuracy: 0.8956\n", - "Epoch 23/25\n", - "430/430 [==============================] - 1s 3ms/step - loss: 0.1655 - accuracy: 0.9438 - val_loss: 0.3186 - val_accuracy: 0.8942\n", - "Epoch 24/25\n", - "430/430 [==============================] - 2s 4ms/step - loss: 0.1628 - accuracy: 0.9458 - val_loss: 0.3176 - val_accuracy: 0.8924\n", - "Epoch 25/25\n", - "430/430 [==============================] - 1s 3ms/step - loss: 0.1611 - accuracy: 0.9460 - val_loss: 0.3169 - val_accuracy: 0.8930\n" - ] - } - ], - "source": [ - "n_epochs = 25\n", - "onecycle = OneCycleScheduler(len(X_train) // batch_size * n_epochs, max_rate=0.05)\n", - "history = model.fit(X_train_scaled, y_train, epochs=n_epochs, batch_size=batch_size,\n", - " validation_data=(X_valid_scaled, y_valid),\n", - " callbacks=[onecycle])" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "# 규제를 사용해 과대적합 피하기" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "## $\\ell_1$과 $\\ell_2$ 규제" - ] - }, - { - "cell_type": "code", - "execution_count": 101, - "metadata": {}, - "outputs": [], - "source": [ - "layer = keras.layers.Dense(100, activation=\"elu\",\n", - " kernel_initializer=\"he_normal\",\n", - " kernel_regularizer=keras.regularizers.l2(0.01))\n", - "# or l1(0.1) for ℓ1 regularization with a factor or 0.1\n", - "# or l1_l2(0.1, 0.01) for both ℓ1 and ℓ2 regularization, with factors 0.1 and 0.01 respectively" - ] - }, - { - "cell_type": "code", - "execution_count": 102, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "Epoch 1/2\n", - "1719/1719 [==============================] - 8s 5ms/step - loss: 1.5956 - accuracy: 0.8124 - val_loss: 0.7169 - val_accuracy: 0.8340\n", - "Epoch 2/2\n", - "1719/1719 [==============================] - 8s 5ms/step - loss: 0.7197 - accuracy: 0.8274 - val_loss: 0.6850 - val_accuracy: 0.8376\n" - ] - } - ], - "source": [ - "model = keras.models.Sequential([\n", - " keras.layers.Flatten(input_shape=[28, 28]),\n", - " keras.layers.Dense(300, activation=\"elu\",\n", - " kernel_initializer=\"he_normal\",\n", - " kernel_regularizer=keras.regularizers.l2(0.01)),\n", - " keras.layers.Dense(100, activation=\"elu\",\n", - " kernel_initializer=\"he_normal\",\n", - " kernel_regularizer=keras.regularizers.l2(0.01)),\n", - " keras.layers.Dense(10, activation=\"softmax\",\n", - " kernel_regularizer=keras.regularizers.l2(0.01))\n", - "])\n", - "model.compile(loss=\"sparse_categorical_crossentropy\", optimizer=\"nadam\", metrics=[\"accuracy\"])\n", - "n_epochs = 2\n", - "history = model.fit(X_train_scaled, y_train, epochs=n_epochs,\n", - " validation_data=(X_valid_scaled, y_valid))" - ] - }, - { - "cell_type": "code", - "execution_count": 103, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "Epoch 1/2\n", - "1719/1719 [==============================] - 8s 5ms/step - loss: 1.6313 - accuracy: 0.8113 - val_loss: 0.7218 - val_accuracy: 0.8310\n", - "Epoch 2/2\n", - "1719/1719 [==============================] - 8s 5ms/step - loss: 0.7187 - accuracy: 0.8273 - val_loss: 0.6826 - val_accuracy: 0.8382\n" - ] - } - ], - "source": [ - "from functools import partial\n", - "\n", - "RegularizedDense = partial(keras.layers.Dense,\n", - " activation=\"elu\",\n", - " kernel_initializer=\"he_normal\",\n", - " kernel_regularizer=keras.regularizers.l2(0.01))\n", - "\n", - "model = keras.models.Sequential([\n", - " keras.layers.Flatten(input_shape=[28, 28]),\n", - " RegularizedDense(300),\n", - " RegularizedDense(100),\n", - " RegularizedDense(10, activation=\"softmax\")\n", - "])\n", - "model.compile(loss=\"sparse_categorical_crossentropy\", optimizer=\"nadam\", metrics=[\"accuracy\"])\n", - "n_epochs = 2\n", - "history = model.fit(X_train_scaled, y_train, epochs=n_epochs,\n", - " validation_data=(X_valid_scaled, y_valid))" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "## 드롭아웃" - ] - }, - { - "cell_type": "code", - "execution_count": 104, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "Epoch 1/2\n", - "1719/1719 [==============================] - 8s 5ms/step - loss: 0.5838 - accuracy: 0.7998 - val_loss: 0.3730 - val_accuracy: 0.8644\n", - "Epoch 2/2\n", - "1719/1719 [==============================] - 8s 5ms/step - loss: 0.4209 - accuracy: 0.8443 - val_loss: 0.3406 - val_accuracy: 0.8724\n" - ] - } - ], - "source": [ - "model = keras.models.Sequential([\n", - " keras.layers.Flatten(input_shape=[28, 28]),\n", - " keras.layers.Dropout(rate=0.2),\n", - " keras.layers.Dense(300, activation=\"elu\", kernel_initializer=\"he_normal\"),\n", - " keras.layers.Dropout(rate=0.2),\n", - " keras.layers.Dense(100, activation=\"elu\", kernel_initializer=\"he_normal\"),\n", - " keras.layers.Dropout(rate=0.2),\n", - " keras.layers.Dense(10, activation=\"softmax\")\n", - "])\n", - "model.compile(loss=\"sparse_categorical_crossentropy\", optimizer=\"nadam\", metrics=[\"accuracy\"])\n", - "n_epochs = 2\n", - "history = model.fit(X_train_scaled, y_train, epochs=n_epochs,\n", - " validation_data=(X_valid_scaled, y_valid))" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "## 알파 드롭아웃" - ] - }, - { - "cell_type": "code", - "execution_count": 105, - "metadata": {}, - "outputs": [], - "source": [ - "tf.random.set_seed(42)\n", - "np.random.seed(42)" - ] - }, - { - "cell_type": "code", - "execution_count": 106, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "Epoch 1/20\n", - "1719/1719 [==============================] - 6s 3ms/step - loss: 0.6654 - accuracy: 0.7595 - val_loss: 0.5929 - val_accuracy: 0.8406\n", - "Epoch 2/20\n", - "1719/1719 [==============================] - 6s 3ms/step - loss: 0.5605 - accuracy: 0.7933 - val_loss: 0.5605 - val_accuracy: 0.8400\n", - "Epoch 3/20\n", - "1719/1719 [==============================] - 6s 3ms/step - loss: 0.5283 - accuracy: 0.8050 - val_loss: 0.4858 - val_accuracy: 0.8596\n", - "Epoch 4/20\n", - "1719/1719 [==============================] - 6s 3ms/step - loss: 0.5072 - accuracy: 0.8125 - val_loss: 0.4629 - val_accuracy: 0.8582\n", - "Epoch 5/20\n", - "1719/1719 [==============================] - 6s 3ms/step - loss: 0.4915 - accuracy: 0.8186 - val_loss: 0.4698 - val_accuracy: 0.8552\n", - "Epoch 6/20\n", - "1719/1719 [==============================] - 6s 3ms/step - loss: 0.4866 - accuracy: 0.8186 - val_loss: 0.4810 - val_accuracy: 0.8612\n", - "Epoch 7/20\n", - "1719/1719 [==============================] - 6s 3ms/step - loss: 0.4724 - accuracy: 0.8251 - val_loss: 0.4792 - val_accuracy: 0.8650\n", - "Epoch 8/20\n", - "1719/1719 [==============================] - 6s 3ms/step - loss: 0.4634 - accuracy: 0.8287 - val_loss: 0.4587 - val_accuracy: 0.8628\n", - "Epoch 9/20\n", - "1719/1719 [==============================] - 6s 3ms/step - loss: 0.4578 - accuracy: 0.8307 - val_loss: 0.4105 - val_accuracy: 0.8718\n", - "Epoch 10/20\n", - "1719/1719 [==============================] - 6s 3ms/step - loss: 0.4531 - accuracy: 0.8333 - val_loss: 0.4714 - val_accuracy: 0.8648\n", - "Epoch 11/20\n", - "1719/1719 [==============================] - 6s 3ms/step - loss: 0.4484 - accuracy: 0.8326 - val_loss: 0.4181 - val_accuracy: 0.8674\n", - "Epoch 12/20\n", - "1719/1719 [==============================] - 6s 3ms/step - loss: 0.4468 - accuracy: 0.8330 - val_loss: 0.5498 - val_accuracy: 0.8500\n", - "Epoch 13/20\n", - "1719/1719 [==============================] - 6s 3ms/step - loss: 0.4421 - accuracy: 0.8363 - val_loss: 0.4546 - val_accuracy: 0.8630\n", - "Epoch 14/20\n", - "1719/1719 [==============================] - 6s 3ms/step - loss: 0.4314 - accuracy: 0.8396 - val_loss: 0.4554 - val_accuracy: 0.8676\n", - "Epoch 15/20\n", - "1719/1719 [==============================] - 6s 3ms/step - loss: 0.4341 - accuracy: 0.8367 - val_loss: 0.4458 - val_accuracy: 0.8662\n", - "Epoch 16/20\n", - "1719/1719 [==============================] - 6s 3ms/step - loss: 0.4312 - accuracy: 0.8412 - val_loss: 0.4325 - val_accuracy: 0.8750\n", - "Epoch 17/20\n", - "1719/1719 [==============================] - 6s 3ms/step - loss: 0.4261 - accuracy: 0.8419 - val_loss: 0.5457 - val_accuracy: 0.8554\n", - "Epoch 18/20\n", - "1719/1719 [==============================] - 6s 3ms/step - loss: 0.4258 - accuracy: 0.8417 - val_loss: 0.5050 - val_accuracy: 0.8700\n", - "Epoch 19/20\n", - "1719/1719 [==============================] - 6s 3ms/step - loss: 0.4208 - accuracy: 0.8437 - val_loss: 0.4899 - val_accuracy: 0.8696\n", - "Epoch 20/20\n", - "1719/1719 [==============================] - 6s 3ms/step - loss: 0.4234 - accuracy: 0.8414 - val_loss: 0.4014 - val_accuracy: 0.8782\n" - ] - } - ], - "source": [ - "model = keras.models.Sequential([\n", - " keras.layers.Flatten(input_shape=[28, 28]),\n", - " keras.layers.AlphaDropout(rate=0.2),\n", - " keras.layers.Dense(300, activation=\"selu\", kernel_initializer=\"lecun_normal\"),\n", - " keras.layers.AlphaDropout(rate=0.2),\n", - " keras.layers.Dense(100, activation=\"selu\", kernel_initializer=\"lecun_normal\"),\n", - " keras.layers.AlphaDropout(rate=0.2),\n", - " keras.layers.Dense(10, activation=\"softmax\")\n", - "])\n", - "optimizer = keras.optimizers.SGD(lr=0.01, momentum=0.9, nesterov=True)\n", - "model.compile(loss=\"sparse_categorical_crossentropy\", optimizer=optimizer, metrics=[\"accuracy\"])\n", - "n_epochs = 20\n", - "history = model.fit(X_train_scaled, y_train, epochs=n_epochs,\n", - " validation_data=(X_valid_scaled, y_valid))" - ] - }, - { - "cell_type": "code", - "execution_count": 107, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "313/313 [==============================] - 1s 2ms/step - loss: 0.4338 - accuracy: 0.8697\n" - ] - }, - { - "data": { - "text/plain": [ - "[0.4337695240974426, 0.869700014591217]" - ] - }, - "execution_count": 107, - "metadata": {}, - "output_type": "execute_result" - } - ], - "source": [ - "model.evaluate(X_test_scaled, y_test)" - ] - }, - { - "cell_type": "code", - "execution_count": 108, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "1719/1719 [==============================] - 4s 2ms/step - loss: 0.3243 - accuracy: 0.8887\n" - ] - }, - { - "data": { - "text/plain": [ - "[0.32432350516319275, 0.8887272477149963]" - ] - }, - "execution_count": 108, - "metadata": {}, - "output_type": "execute_result" - } - ], - "source": [ - "model.evaluate(X_train_scaled, y_train)" - ] - }, - { - "cell_type": "code", - "execution_count": 109, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "1719/1719 [==============================] - 5s 3ms/step - loss: 0.4167 - accuracy: 0.8463\n" - ] - } - ], - "source": [ - "history = model.fit(X_train_scaled, y_train)" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "## MC 드롭아웃" - ] - }, - { - "cell_type": "code", - "execution_count": 110, - "metadata": {}, - "outputs": [], - "source": [ - "tf.random.set_seed(42)\n", - "np.random.seed(42)" - ] - }, - { - "cell_type": "code", - "execution_count": 111, - "metadata": { - "scrolled": true - }, - "outputs": [], - "source": [ - "y_probas = np.stack([model(X_test_scaled, training=True)\n", - " for sample in range(100)])\n", - "y_proba = y_probas.mean(axis=0)\n", - "y_std = y_probas.std(axis=0)" - ] - }, - { - "cell_type": "code", - "execution_count": 112, - "metadata": {}, - "outputs": [ - { - "data": { - "text/plain": [ - "array([[0., 0., 0., 0., 0., 0., 0., 0., 0., 1.]], dtype=float32)" - ] - }, - "execution_count": 112, - "metadata": {}, - "output_type": "execute_result" - } - ], - "source": [ - "np.round(model.predict(X_test_scaled[:1]), 2)" - ] - }, - { - "cell_type": "code", - "execution_count": 113, - "metadata": {}, - "outputs": [ - { - "data": { - "text/plain": [ - "array([[[0. , 0. , 0. , 0. , 0. , 0.43, 0. , 0.18, 0. , 0.39]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.75, 0. , 0.25]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.42, 0. , 0. , 0. , 0.58]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.26, 0. , 0.72]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.12, 0. , 0.07, 0. , 0.8 ]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0. , 0. , 0.3 , 0. , 0.7 ]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.36, 0. , 0.13, 0. , 0.51]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0. , 0. , 0.34, 0. , 0.66]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.75, 0. , 0.02, 0. , 0.23]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.02, 0. , 0.04, 0. , 0.94]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.02, 0. , 0.03, 0. , 0.95]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.64, 0. , 0. , 0. , 0.36]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.68, 0. , 0.05, 0. , 0.28]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.31, 0. , 0.04, 0. , 0.65]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.05, 0. , 0.02, 0. , 0.93]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.42, 0. , 0.02, 0. , 0.57]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.02, 0. , 0. , 0. , 0.98]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.03, 0. , 0.96]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0. , 0. , 0.52, 0. , 0.48]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.28, 0. , 0.12, 0. , 0.6 ]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.42, 0. , 0.01, 0. , 0.56]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.96, 0. , 0.01, 0. , 0.03]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0. , 0. , 0. , 0. , 1. ]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0. , 0. , 0.04, 0. , 0.96]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0. , 0. , 0.99]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.75, 0. , 0.11, 0. , 0.14]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.21, 0. , 0.18, 0. , 0.61]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.44, 0. , 0.08, 0. , 0.48]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.63, 0. , 0.01, 0. , 0.36]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.23, 0. , 0.55, 0. , 0.22]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.31, 0. , 0.01, 0. , 0.68]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.06, 0. , 0.01, 0. , 0.93]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.3 , 0. , 0.02, 0. , 0.68]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.91, 0. , 0.04, 0. , 0.05]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.06, 0. , 0.02, 0. , 0.93]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.33, 0. , 0.36, 0. , 0.31]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.99]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.6 , 0. , 0.03, 0. , 0.37]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.14, 0. , 0.08, 0. , 0.77]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.66, 0. , 0. , 0. , 0.34]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.11, 0. , 0.01, 0. , 0.88]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0. , 0. , 0.99]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.47, 0. , 0.08, 0. , 0.44]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.99]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.01, 0. , 0.98]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0. , 0. , 0. , 0. , 1. ]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.12, 0. , 0.11, 0. , 0.77]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.21, 0. , 0.78]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0. , 0. , 0.08, 0. , 0.92]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.2 , 0. , 0.79]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.49, 0. , 0.11, 0. , 0.41]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.69, 0. , 0.06, 0. , 0.25]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0. , 0. , 0. , 0. , 1. ]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.31, 0. , 0.01, 0. , 0.68]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.03, 0. , 0.01, 0. , 0.97]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.05, 0. , 0.25, 0. , 0.71]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.02, 0. , 0.47, 0. , 0.51]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.07, 0. , 0.72, 0. , 0.21]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.49, 0. , 0.5 ]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.06, 0. , 0.02, 0. , 0.92]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.6 , 0. , 0.02, 0. , 0.38]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.08, 0. , 0.02, 0. , 0.9 ]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.14, 0. , 0.02, 0. , 0.84]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.02, 0. , 0.97]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.44, 0. , 0.02, 0. , 0.54]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.18, 0. , 0.08, 0. , 0.74]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.49, 0. , 0.04, 0. , 0.47]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.99]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.11, 0. , 0.08, 0. , 0.81]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.07, 0. , 0.09, 0. , 0.84]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0. , 0. , 0.98]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.03, 0. , 0.04, 0. , 0.93]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.38, 0. , 0.03, 0. , 0.59]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.01, 0. , 0.98]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.1 , 0. , 0.02, 0. , 0.88]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.98]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.02, 0. , 0.01, 0. , 0.98]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.49, 0. , 0.22, 0. , 0.29]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.03, 0. , 0.96]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.03, 0. , 0.96]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.3 , 0. , 0.69]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.28, 0. , 0. , 0. , 0.72]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0. , 0. , 0. , 0. , 1. ]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.03, 0. , 0.03, 0. , 0.94]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.04, 0. , 0.09, 0. , 0.87]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.12, 0. , 0.03, 0. , 0.85]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.98, 0. , 0. , 0. , 0.02]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.17, 0. , 0. , 0. , 0.83]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.05, 0. , 0.01, 0. , 0.94]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0. , 0. , 0. , 0. , 1. ]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.02, 0. , 0. , 0. , 0.98]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.04, 0. , 0.13, 0. , 0.83]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.93, 0. , 0.03, 0. , 0.04]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.86, 0. , 0.01, 0. , 0.13]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.59, 0. , 0.02, 0. , 0.38]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.02, 0. , 0.01, 0. , 0.97]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.11, 0. , 0.01, 0. , 0.88]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.01, 0. , 0.98]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.78, 0. , 0.01, 0. , 0.21]],\n", - "\n", - " [[0. , 0. , 0. , 0. , 0. , 0.48, 0. , 0.01, 0. , 0.51]]],\n", - " dtype=float32)" - ] - }, - "execution_count": 113, - "metadata": {}, - "output_type": "execute_result" - } - ], - "source": [ - "np.round(y_probas[:, :1], 2)" - ] - }, - { - "cell_type": "code", - "execution_count": 114, - "metadata": {}, - "outputs": [ - { - "data": { - "text/plain": [ - "array([[0. , 0. , 0. , 0. , 0. , 0.23, 0. , 0.09, 0. , 0.68]],\n", - " dtype=float32)" - ] - }, - "execution_count": 114, - "metadata": {}, - "output_type": "execute_result" - } - ], - "source": [ - "np.round(y_proba[:1], 2)" - ] - }, - { - "cell_type": "code", - "execution_count": 115, - "metadata": {}, - "outputs": [ - { - "data": { - "text/plain": [ - "array([[0. , 0. , 0. , 0. , 0. , 0.28, 0. , 0.15, 0. , 0.29]],\n", - " dtype=float32)" - ] - }, - "execution_count": 115, - "metadata": {}, - "output_type": "execute_result" - } - ], - "source": [ - "y_std = y_probas.std(axis=0)\n", - "np.round(y_std[:1], 2)" - ] - }, - { - "cell_type": "code", - "execution_count": 116, - "metadata": {}, - "outputs": [], - "source": [ - "y_pred = np.argmax(y_proba, axis=1)" - ] - }, - { - "cell_type": "code", - "execution_count": 117, - "metadata": {}, - "outputs": [ - { - "data": { - "text/plain": [ - "0.8666" - ] - }, - "execution_count": 117, - "metadata": {}, - "output_type": "execute_result" - } - ], - "source": [ - "accuracy = np.sum(y_pred == y_test) / len(y_test)\n", - "accuracy" - ] - }, - { - "cell_type": "code", - "execution_count": 118, - "metadata": {}, - "outputs": [], - "source": [ - "class MCDropout(keras.layers.Dropout):\n", - " def call(self, inputs):\n", - " return super().call(inputs, training=True)\n", - "\n", - "class MCAlphaDropout(keras.layers.AlphaDropout):\n", - " def call(self, inputs):\n", - " return super().call(inputs, training=True)" - ] - }, - { - "cell_type": "code", - "execution_count": 119, - "metadata": {}, - "outputs": [], - "source": [ - "tf.random.set_seed(42)\n", - "np.random.seed(42)" - ] - }, - { - "cell_type": "code", - "execution_count": 120, - "metadata": {}, - "outputs": [], - "source": [ - "mc_model = keras.models.Sequential([\n", - " MCAlphaDropout(layer.rate) if isinstance(layer, keras.layers.AlphaDropout) else layer\n", - " for layer in model.layers\n", - "])" - ] - }, - { - "cell_type": "code", - "execution_count": 121, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "Model: \"sequential_20\"\n", - "_________________________________________________________________\n", - "Layer (type) Output Shape Param # \n", - "=================================================================\n", - "flatten_18 (Flatten) (None, 784) 0 \n", - "_________________________________________________________________\n", - "mc_alpha_dropout (MCAlphaDro (None, 784) 0 \n", - "_________________________________________________________________\n", - "dense_262 (Dense) (None, 300) 235500 \n", - "_________________________________________________________________\n", - "mc_alpha_dropout_1 (MCAlphaD (None, 300) 0 \n", - "_________________________________________________________________\n", - "dense_263 (Dense) (None, 100) 30100 \n", - "_________________________________________________________________\n", - "mc_alpha_dropout_2 (MCAlphaD (None, 100) 0 \n", - "_________________________________________________________________\n", - "dense_264 (Dense) (None, 10) 1010 \n", - "=================================================================\n", - "Total params: 266,610\n", - "Trainable params: 266,610\n", - "Non-trainable params: 0\n", - "_________________________________________________________________\n" - ] - } - ], - "source": [ - "mc_model.summary()" - ] - }, - { - "cell_type": "code", - "execution_count": 122, - "metadata": {}, - "outputs": [], - "source": [ - "optimizer = keras.optimizers.SGD(lr=0.01, momentum=0.9, nesterov=True)\n", - "mc_model.compile(loss=\"sparse_categorical_crossentropy\", optimizer=optimizer, metrics=[\"accuracy\"])" - ] - }, - { - "cell_type": "code", - "execution_count": 123, - "metadata": {}, - "outputs": [], - "source": [ - "mc_model.set_weights(model.get_weights())" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "이제 MC 드롭아웃을 모델에 사용할 수 있습니다:" - ] - }, - { - "cell_type": "code", - "execution_count": 124, - "metadata": {}, - "outputs": [ - { - "data": { - "text/plain": [ - "array([[0. , 0. , 0. , 0. , 0. , 0.22, 0. , 0.15, 0. , 0.63]],\n", - " dtype=float32)" - ] - }, - "execution_count": 124, - "metadata": {}, - "output_type": "execute_result" - } - ], - "source": [ - "np.round(np.mean([mc_model.predict(X_test_scaled[:1]) for sample in range(100)], axis=0), 2)" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "## 맥스 노름" - ] - }, - { - "cell_type": "code", - "execution_count": 125, - "metadata": {}, - "outputs": [], - "source": [ - "layer = keras.layers.Dense(100, activation=\"selu\", kernel_initializer=\"lecun_normal\",\n", - " kernel_constraint=keras.constraints.max_norm(1.))" - ] - }, - { - "cell_type": "code", - "execution_count": 126, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "Epoch 1/2\n", - "1719/1719 [==============================] - 8s 5ms/step - loss: 0.4749 - accuracy: 0.8337 - val_loss: 0.3665 - val_accuracy: 0.8676\n", - "Epoch 2/2\n", - "1719/1719 [==============================] - 8s 5ms/step - loss: 0.3539 - accuracy: 0.8703 - val_loss: 0.3700 - val_accuracy: 0.8672\n" - ] - } - ], - "source": [ - "MaxNormDense = partial(keras.layers.Dense,\n", - " activation=\"selu\", kernel_initializer=\"lecun_normal\",\n", - " kernel_constraint=keras.constraints.max_norm(1.))\n", - "\n", - "model = keras.models.Sequential([\n", - " keras.layers.Flatten(input_shape=[28, 28]),\n", - " MaxNormDense(300),\n", - " MaxNormDense(100),\n", - " keras.layers.Dense(10, activation=\"softmax\")\n", - "])\n", - "model.compile(loss=\"sparse_categorical_crossentropy\", optimizer=\"nadam\", metrics=[\"accuracy\"])\n", - "n_epochs = 2\n", - "history = model.fit(X_train_scaled, y_train, epochs=n_epochs,\n", - " validation_data=(X_valid_scaled, y_valid))" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "# 연습문제 해답" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "## 1. to 7." - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "부록 A 참조." - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "## 8. CIFAR10에서 딥러닝" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "### a.\n", - "*문제: 100개의 뉴런을 가진 은닉층 20개로 심층 신경망을 만들어보세요(너무 많은 것 같지만 이 연습문제의 핵심입니다). He 초기화와 ELU 활성화 함수를 사용하세요.*" - ] - }, - { - "cell_type": "code", - "execution_count": 127, - "metadata": {}, - "outputs": [], - "source": [ - "keras.backend.clear_session()\n", - "tf.random.set_seed(42)\n", - "np.random.seed(42)\n", - "\n", - "model = keras.models.Sequential()\n", - "model.add(keras.layers.Flatten(input_shape=[32, 32, 3]))\n", - "for _ in range(20):\n", - " model.add(keras.layers.Dense(100,\n", - " activation=\"elu\",\n", - " kernel_initializer=\"he_normal\"))" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "### b.\n", - "*문제: Nadam 옵티마이저와 조기 종료를 사용하여 CIFAR10 데이터셋에 이 네트워크를 훈련하세요. `keras.datasets.cifar10.load_ data()`를 사용하여 데이터를 적재할 수 있습니다. 이 데이터셋은 10개의 클래스와 32×32 크기의 컬러 이미지 60,000개로 구성됩니다(50,000개는 훈련, 10,000개는 테스트). 따라서 10개의 뉴런과 소프트맥스 활성화 함수를 사용하는 출력층이 필요합니다. 모델 구조와 하이퍼파라미터를 바꿀 때마다 적절한 학습률을 찾아야 한다는 것을 기억하세요.*" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "모델에 출력층을 추가합니다:" - ] - }, - { - "cell_type": "code", - "execution_count": 128, - "metadata": {}, - "outputs": [], - "source": [ - "model.add(keras.layers.Dense(10, activation=\"softmax\"))" - ] + "nbformat": 4, + "nbformat_minor": 0, + "metadata": { + "kernelspec": { + "display_name": "TensorFlow 2.3 on Python 3.6 (CUDA 10.1)", + "language": "python", + "name": "python3" + }, + "language_info": { + "codemirror_mode": { + "name": "ipython", + "version": 3 + }, + "file_extension": ".py", + "mimetype": "text/x-python", + "name": "python", + "nbconvert_exporter": "python", + "pygments_lexer": "ipython3", + "version": "3.6.9" + }, + "nav_menu": { + "height": "360px", + "width": "416px" + }, + "toc": { + "navigate_menu": true, + "number_sections": true, + "sideBar": true, + "threshold": 6, + "toc_cell": false, + "toc_section_display": "block", + "toc_window_display": false + }, + "colab": { + "name": "11_training_deep_neural_networks.ipynb", + "provenance": [] + }, + "accelerator": "GPU" }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "학습률 5e-5인 Nadam 옵티마이저를 사용해 보죠. 학습률 1e-5, 3e-5, 1e-4, 3e-4, 1e-3, 3e-3, 1e-2를 테스트하고 10번의 에포크 동안 (아래 텐서보드 콜백으로) 학습 곡선을 비교해 보았습니다. 학습률 3e-5와 1e-4가 꽤 좋았기 때문에 5e-5를 시도해 보았고 조금 더 나은 결과를 냈습니다." - ] - }, - { - "cell_type": "code", - "execution_count": 129, - "metadata": {}, - "outputs": [], - "source": [ - "optimizer = keras.optimizers.Nadam(lr=5e-5)\n", - "model.compile(loss=\"sparse_categorical_crossentropy\",\n", - " optimizer=optimizer,\n", - " metrics=[\"accuracy\"])" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "CIFAR10 데이터셋을 로드하죠. 조기 종료를 사용하기 때문에 검증 세트가 필요합니다. 원본 훈련 세트에서 처음 5,000개를 검증 세트로 사용하겠습니다:" - ] - }, - { - "cell_type": "code", - "execution_count": 130, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "Downloading data from https://www.cs.toronto.edu/~kriz/cifar-10-python.tar.gz\n", - "170500096/170498071 [==============================] - 18s 0us/step\n" - ] - } - ], - "source": [ - "(X_train_full, y_train_full), (X_test, y_test) = keras.datasets.cifar10.load_data()\n", - "\n", - "X_train = X_train_full[5000:]\n", - "y_train = y_train_full[5000:]\n", - "X_valid = X_train_full[:5000]\n", - "y_valid = y_train_full[:5000]" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "이제 콜백을 만들고 모델을 훈련합니다:" - ] - }, - { - "cell_type": "code", - "execution_count": 131, - "metadata": {}, - "outputs": [], - "source": [ - "early_stopping_cb = keras.callbacks.EarlyStopping(patience=20)\n", - "model_checkpoint_cb = keras.callbacks.ModelCheckpoint(\"my_cifar10_model.h5\", save_best_only=True)\n", - "run_index = 1 # 모델을 훈련할 때마다 증가시킴\n", - "run_logdir = os.path.join(os.curdir, \"my_cifar10_logs\", \"run_{:03d}\".format(run_index))\n", - "tensorboard_cb = keras.callbacks.TensorBoard(run_logdir)\n", - "callbacks = [early_stopping_cb, model_checkpoint_cb, tensorboard_cb]" - ] - }, - { - "cell_type": "code", - "execution_count": 132, - "metadata": {}, - "outputs": [ - { - "data": { - "text/plain": [ - "ERROR: Failed to launch TensorBoard (exited with 255).\n", - "Contents of stderr:\n", - "E0809 03:11:42.951561 139734898673472 program.py:312] TensorBoard could not bind to port 6006, it was already in use\n", - "ERROR: TensorBoard could not bind to port 6006, it was already in use" - ] - }, - "metadata": {}, - "output_type": "display_data" - } - ], - "source": [ - "%tensorboard --logdir=./my_cifar10_logs --port=6006" - ] - }, - { - "cell_type": "code", - "execution_count": 133, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "Epoch 1/100\n", - " 1/1407 [..............................] - ETA: 0s - loss: 165.9660 - accuracy: 0.0625WARNING:tensorflow:From /home/work/.local/lib/python3.6/site-packages/tensorflow/python/ops/summary_ops_v2.py:1277: stop (from tensorflow.python.eager.profiler) is deprecated and will be removed after 2020-07-01.\n", - "Instructions for updating:\n", - "use `tf.profiler.experimental.stop` instead.\n", - " 2/1407 [..............................] - ETA: 1:52 - loss: 133.4792 - accuracy: 0.1250WARNING:tensorflow:Callbacks method `on_train_batch_end` is slow compared to the batch time (batch time: 0.0164s vs `on_train_batch_end` time: 0.1398s). Check your callbacks.\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 4.0015 - accuracy: 0.1711 - val_loss: 2.0790 - val_accuracy: 0.2354\n", - "Epoch 2/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 2.0265 - accuracy: 0.2568 - val_loss: 2.0408 - val_accuracy: 0.2488\n", - "Epoch 3/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.9195 - accuracy: 0.2983 - val_loss: 1.9150 - val_accuracy: 0.2916\n", - "Epoch 4/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.8406 - accuracy: 0.3284 - val_loss: 1.9163 - val_accuracy: 0.3050\n", - "Epoch 5/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.7846 - accuracy: 0.3525 - val_loss: 1.7578 - val_accuracy: 0.3592\n", - "Epoch 6/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.7352 - accuracy: 0.3698 - val_loss: 1.7461 - val_accuracy: 0.3570\n", - "Epoch 7/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.6930 - accuracy: 0.3869 - val_loss: 1.7101 - val_accuracy: 0.3752\n", - "Epoch 8/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.6576 - accuracy: 0.4002 - val_loss: 1.6530 - val_accuracy: 0.3962\n", - "Epoch 9/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.6233 - accuracy: 0.4169 - val_loss: 1.6236 - val_accuracy: 0.4070\n", - "Epoch 10/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.5965 - accuracy: 0.4248 - val_loss: 1.6644 - val_accuracy: 0.4026\n", - "Epoch 11/100\n", - "1407/1407 [==============================] - 21s 15ms/step - loss: 1.5763 - accuracy: 0.4328 - val_loss: 1.6739 - val_accuracy: 0.3946\n", - "Epoch 12/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.5534 - accuracy: 0.4420 - val_loss: 1.6191 - val_accuracy: 0.4186\n", - "Epoch 13/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.5332 - accuracy: 0.4468 - val_loss: 1.6050 - val_accuracy: 0.4168\n", - "Epoch 14/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.5181 - accuracy: 0.4531 - val_loss: 1.6009 - val_accuracy: 0.4312\n", - "Epoch 15/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.4995 - accuracy: 0.4616 - val_loss: 1.5658 - val_accuracy: 0.4366\n", - "Epoch 16/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.4827 - accuracy: 0.4652 - val_loss: 1.5856 - val_accuracy: 0.4376\n", - "Epoch 17/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.4717 - accuracy: 0.4713 - val_loss: 1.5381 - val_accuracy: 0.4478\n", - "Epoch 18/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.4565 - accuracy: 0.4756 - val_loss: 1.5496 - val_accuracy: 0.4478\n", - "Epoch 19/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.4422 - accuracy: 0.4797 - val_loss: 1.5506 - val_accuracy: 0.4478\n", - "Epoch 20/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.4327 - accuracy: 0.4816 - val_loss: 1.5574 - val_accuracy: 0.4528\n", - "Epoch 21/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.4213 - accuracy: 0.4875 - val_loss: 1.5716 - val_accuracy: 0.4410\n", - "Epoch 22/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.4108 - accuracy: 0.4903 - val_loss: 1.5377 - val_accuracy: 0.4544\n", - "Epoch 23/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.3972 - accuracy: 0.4968 - val_loss: 1.5215 - val_accuracy: 0.4526\n", - "Epoch 24/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.3865 - accuracy: 0.4998 - val_loss: 1.5374 - val_accuracy: 0.4470\n", - "Epoch 25/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.3781 - accuracy: 0.5056 - val_loss: 1.5694 - val_accuracy: 0.4466\n", - "Epoch 26/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.3672 - accuracy: 0.5104 - val_loss: 1.5405 - val_accuracy: 0.4542\n", - "Epoch 27/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.3583 - accuracy: 0.5110 - val_loss: 1.5103 - val_accuracy: 0.4690\n", - "Epoch 28/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.3481 - accuracy: 0.5161 - val_loss: 1.5282 - val_accuracy: 0.4572\n", - "Epoch 29/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.3387 - accuracy: 0.5186 - val_loss: 1.5302 - val_accuracy: 0.4540\n", - "Epoch 30/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.3307 - accuracy: 0.5206 - val_loss: 1.5451 - val_accuracy: 0.4666\n", - "Epoch 31/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.3235 - accuracy: 0.5245 - val_loss: 1.5553 - val_accuracy: 0.4604\n", - "Epoch 32/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.3138 - accuracy: 0.5274 - val_loss: 1.5263 - val_accuracy: 0.4604\n", - "Epoch 33/100\n", - "1407/1407 [==============================] - 21s 15ms/step - loss: 1.3025 - accuracy: 0.5323 - val_loss: 1.5133 - val_accuracy: 0.4736\n", - "Epoch 34/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.2956 - accuracy: 0.5340 - val_loss: 1.5021 - val_accuracy: 0.4748\n", - "Epoch 35/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.2876 - accuracy: 0.5345 - val_loss: 1.5427 - val_accuracy: 0.4606\n", - "Epoch 36/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.2794 - accuracy: 0.5408 - val_loss: 1.5662 - val_accuracy: 0.4586\n", - "Epoch 37/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.2706 - accuracy: 0.5423 - val_loss: 1.5014 - val_accuracy: 0.4778\n", - "Epoch 38/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.2615 - accuracy: 0.5464 - val_loss: 1.5048 - val_accuracy: 0.4736\n", - "Epoch 39/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.2541 - accuracy: 0.5483 - val_loss: 1.5195 - val_accuracy: 0.4656\n", - "Epoch 40/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.2448 - accuracy: 0.5542 - val_loss: 1.5167 - val_accuracy: 0.4790\n", - "Epoch 41/100\n", - "1407/1407 [==============================] - 21s 15ms/step - loss: 1.2387 - accuracy: 0.5545 - val_loss: 1.5400 - val_accuracy: 0.4660\n", - "Epoch 42/100\n", - "1407/1407 [==============================] - 21s 15ms/step - loss: 1.2335 - accuracy: 0.5566 - val_loss: 1.5325 - val_accuracy: 0.4680\n", - "Epoch 43/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.2257 - accuracy: 0.5580 - val_loss: 1.5342 - val_accuracy: 0.4722\n", - "Epoch 44/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.2173 - accuracy: 0.5606 - val_loss: 1.5537 - val_accuracy: 0.4580\n", - "Epoch 45/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.2082 - accuracy: 0.5643 - val_loss: 1.5200 - val_accuracy: 0.4770\n", - "Epoch 46/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.2037 - accuracy: 0.5666 - val_loss: 1.5349 - val_accuracy: 0.4716\n", - "Epoch 47/100\n", - "1407/1407 [==============================] - 21s 15ms/step - loss: 1.1935 - accuracy: 0.5673 - val_loss: 1.5299 - val_accuracy: 0.4774\n", - "Epoch 48/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.1884 - accuracy: 0.5720 - val_loss: 1.5361 - val_accuracy: 0.4724\n", - "Epoch 49/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.1769 - accuracy: 0.5766 - val_loss: 1.5410 - val_accuracy: 0.4720\n", - "Epoch 50/100\n", - "1407/1407 [==============================] - 21s 15ms/step - loss: 1.1782 - accuracy: 0.5752 - val_loss: 1.5620 - val_accuracy: 0.4714\n", - "Epoch 51/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.1649 - accuracy: 0.5804 - val_loss: 1.5364 - val_accuracy: 0.4710\n", - "Epoch 52/100\n" - ] - }, - { - "name": "stdout", - "output_type": "stream", - "text": [ - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.1599 - accuracy: 0.5818 - val_loss: 1.5514 - val_accuracy: 0.4658\n", - "Epoch 53/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.1511 - accuracy: 0.5853 - val_loss: 1.5512 - val_accuracy: 0.4798\n", - "Epoch 54/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.1485 - accuracy: 0.5887 - val_loss: 1.5754 - val_accuracy: 0.4648\n", - "Epoch 55/100\n", - "1407/1407 [==============================] - 21s 15ms/step - loss: 1.1398 - accuracy: 0.5905 - val_loss: 1.5335 - val_accuracy: 0.4806\n", - "Epoch 56/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.1352 - accuracy: 0.5909 - val_loss: 1.5603 - val_accuracy: 0.4732\n", - "Epoch 57/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.1285 - accuracy: 0.5933 - val_loss: 1.5535 - val_accuracy: 0.4824\n" - ] - }, - { - "data": { - "text/plain": [ - "" - ] - }, - "execution_count": 133, - "metadata": {}, - "output_type": "execute_result" - } - ], - "source": [ - "model.fit(X_train, y_train, epochs=100,\n", - " validation_data=(X_valid, y_valid),\n", - " callbacks=callbacks)" - ] - }, - { - "cell_type": "code", - "execution_count": 134, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "157/157 [==============================] - 0s 3ms/step - loss: 1.5014 - accuracy: 0.0882\n" - ] - }, - { - "data": { - "text/plain": [ - "[1.5013599395751953, 0.08820000290870667]" - ] - }, - "execution_count": 134, - "metadata": {}, - "output_type": "execute_result" - } - ], - "source": [ - "model = keras.models.load_model(\"my_cifar10_model.h5\")\n", - "model.evaluate(X_valid, y_valid)" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "가장 낮은 검증 손실을 내는 모델은 검증 세트에서 약 47% 정확도를 얻었습니다. 이 검증 점수에 도달하는데 39번의 에포크가 걸렸습니다. (GPU가 없는) 제 노트북에서 에포크당 약 10초 정도 걸렸습니다. 배치 정규화를 사용해 성능을 올릴 수 있는지 확인해 보죠." - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "### c.\n", - "*문제: 배치 정규화를 추가하고 학습 곡선을 비교해보세요. 이전보다 빠르게 수렴하나요? 더 좋은 모델이 만들어지나요? 훈련 속도에는 어떤 영향을 미치나요?*" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "다음 코드는 위의 코드와 배우 비슷합니다. 몇 가지 다른 점은 아래와 같습니다:\n", - "\n", - "* 출력층을 제외하고 모든 `Dense` 층 다음에 (활성화 함수 전에) BN 층을 추가했습니다. 처음 은닉층 전에도 BN 층을 추가했습니다.\n", - "* 학습률을 5e-4로 바꾸었습니다. 1e-5, 3e-5, 5e-5, 1e-4, 3e-4, 5e-4, 1e-3, 3e-3를 시도해 보고 20번 에포크 후에 검증 세트 성능이 가장 좋은 것을 선택했습니다.\n", - "* run_logdir를 run_bn_* 으로 이름을 바꾸고 모델 파일 이름을 my_cifar10_bn_model.h5로 변경했습니다." - ] - }, - { - "cell_type": "code", - "execution_count": 135, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "Epoch 1/100\n", - " 2/1407 [..............................] - ETA: 9:29 - loss: 2.8693 - accuracy: 0.1094WARNING:tensorflow:Callbacks method `on_train_batch_end` is slow compared to the batch time (batch time: 0.0364s vs `on_train_batch_end` time: 0.7737s). Check your callbacks.\n", - "1407/1407 [==============================] - 51s 36ms/step - loss: 1.8431 - accuracy: 0.3390 - val_loss: 1.7148 - val_accuracy: 0.3886\n", - "Epoch 2/100\n", - "1407/1407 [==============================] - 49s 35ms/step - loss: 1.6690 - accuracy: 0.4046 - val_loss: 1.6174 - val_accuracy: 0.4144\n", - "Epoch 3/100\n", - "1407/1407 [==============================] - 49s 35ms/step - loss: 1.5972 - accuracy: 0.4320 - val_loss: 1.5171 - val_accuracy: 0.4478\n", - "Epoch 4/100\n", - "1407/1407 [==============================] - 50s 35ms/step - loss: 1.5463 - accuracy: 0.4495 - val_loss: 1.4883 - val_accuracy: 0.4688\n", - "Epoch 5/100\n", - "1407/1407 [==============================] - 50s 35ms/step - loss: 1.5051 - accuracy: 0.4641 - val_loss: 1.4369 - val_accuracy: 0.4892\n", - "Epoch 6/100\n", - "1407/1407 [==============================] - 49s 35ms/step - loss: 1.4684 - accuracy: 0.4793 - val_loss: 1.4056 - val_accuracy: 0.5018\n", - "Epoch 7/100\n", - "1407/1407 [==============================] - 49s 35ms/step - loss: 1.4350 - accuracy: 0.4895 - val_loss: 1.4292 - val_accuracy: 0.4888\n", - "Epoch 8/100\n", - "1407/1407 [==============================] - 49s 35ms/step - loss: 1.4087 - accuracy: 0.5006 - val_loss: 1.4021 - val_accuracy: 0.5088\n", - "Epoch 9/100\n", - "1407/1407 [==============================] - 49s 35ms/step - loss: 1.3834 - accuracy: 0.5095 - val_loss: 1.3738 - val_accuracy: 0.5110\n", - "Epoch 10/100\n", - "1407/1407 [==============================] - 49s 35ms/step - loss: 1.3645 - accuracy: 0.5167 - val_loss: 1.3432 - val_accuracy: 0.5252\n", - "Epoch 11/100\n", - "1407/1407 [==============================] - 49s 35ms/step - loss: 1.3428 - accuracy: 0.5258 - val_loss: 1.3583 - val_accuracy: 0.5132\n", - "Epoch 12/100\n", - "1407/1407 [==============================] - 49s 35ms/step - loss: 1.3227 - accuracy: 0.5316 - val_loss: 1.3820 - val_accuracy: 0.5052\n", - "Epoch 13/100\n", - "1407/1407 [==============================] - 48s 34ms/step - loss: 1.3010 - accuracy: 0.5371 - val_loss: 1.3794 - val_accuracy: 0.5094\n", - "Epoch 14/100\n", - "1407/1407 [==============================] - 49s 35ms/step - loss: 1.2838 - accuracy: 0.5446 - val_loss: 1.3531 - val_accuracy: 0.5260\n", - "Epoch 15/100\n", - "1407/1407 [==============================] - 49s 35ms/step - loss: 1.2621 - accuracy: 0.5548 - val_loss: 1.3641 - val_accuracy: 0.5256\n", - "Epoch 16/100\n", - "1407/1407 [==============================] - 49s 35ms/step - loss: 1.2535 - accuracy: 0.5572 - val_loss: 1.3720 - val_accuracy: 0.5276\n", - "Epoch 17/100\n", - "1407/1407 [==============================] - 49s 35ms/step - loss: 1.2355 - accuracy: 0.5609 - val_loss: 1.3184 - val_accuracy: 0.5348\n", - "Epoch 18/100\n", - "1407/1407 [==============================] - 49s 35ms/step - loss: 1.2164 - accuracy: 0.5685 - val_loss: 1.3487 - val_accuracy: 0.5296\n", - "Epoch 19/100\n", - "1407/1407 [==============================] - 49s 35ms/step - loss: 1.2037 - accuracy: 0.5770 - val_loss: 1.3278 - val_accuracy: 0.5366\n", - "Epoch 20/100\n", - "1407/1407 [==============================] - 49s 35ms/step - loss: 1.1916 - accuracy: 0.5789 - val_loss: 1.3592 - val_accuracy: 0.5260\n", - "Epoch 21/100\n", - "1407/1407 [==============================] - 49s 35ms/step - loss: 1.1782 - accuracy: 0.5848 - val_loss: 1.3478 - val_accuracy: 0.5302\n", - "Epoch 22/100\n", - "1407/1407 [==============================] - 49s 35ms/step - loss: 1.1587 - accuracy: 0.5913 - val_loss: 1.3477 - val_accuracy: 0.5308\n", - "Epoch 23/100\n", - "1407/1407 [==============================] - 49s 35ms/step - loss: 1.1481 - accuracy: 0.5933 - val_loss: 1.3285 - val_accuracy: 0.5378\n", - "Epoch 24/100\n", - "1407/1407 [==============================] - 49s 35ms/step - loss: 1.1395 - accuracy: 0.5989 - val_loss: 1.3393 - val_accuracy: 0.5388\n", - "Epoch 25/100\n", - "1407/1407 [==============================] - 49s 35ms/step - loss: 1.1285 - accuracy: 0.6044 - val_loss: 1.3436 - val_accuracy: 0.5354\n", - "Epoch 26/100\n", - "1407/1407 [==============================] - 49s 35ms/step - loss: 1.1080 - accuracy: 0.6085 - val_loss: 1.3496 - val_accuracy: 0.5258\n", - "Epoch 27/100\n", - "1407/1407 [==============================] - 49s 35ms/step - loss: 1.0971 - accuracy: 0.6143 - val_loss: 1.3484 - val_accuracy: 0.5350\n", - "Epoch 28/100\n", - "1407/1407 [==============================] - 49s 35ms/step - loss: 1.0978 - accuracy: 0.6121 - val_loss: 1.3698 - val_accuracy: 0.5274\n", - "Epoch 29/100\n", - "1407/1407 [==============================] - 49s 35ms/step - loss: 1.0825 - accuracy: 0.6198 - val_loss: 1.3416 - val_accuracy: 0.5348\n", - "Epoch 30/100\n", - "1407/1407 [==============================] - 49s 35ms/step - loss: 1.0698 - accuracy: 0.6219 - val_loss: 1.3363 - val_accuracy: 0.5366\n", - "Epoch 31/100\n", - "1407/1407 [==============================] - 49s 35ms/step - loss: 1.0569 - accuracy: 0.6262 - val_loss: 1.3536 - val_accuracy: 0.5356\n", - "Epoch 32/100\n", - "1407/1407 [==============================] - 49s 35ms/step - loss: 1.0489 - accuracy: 0.6306 - val_loss: 1.3822 - val_accuracy: 0.5220\n", - "Epoch 33/100\n", - "1407/1407 [==============================] - 49s 35ms/step - loss: 1.0387 - accuracy: 0.6338 - val_loss: 1.3633 - val_accuracy: 0.5404\n", - "Epoch 34/100\n", - "1407/1407 [==============================] - 49s 35ms/step - loss: 1.0342 - accuracy: 0.6344 - val_loss: 1.3611 - val_accuracy: 0.5364\n", - "Epoch 35/100\n", - "1407/1407 [==============================] - 49s 35ms/step - loss: 1.0163 - accuracy: 0.6422 - val_loss: 1.3904 - val_accuracy: 0.5356\n", - "Epoch 36/100\n", - "1407/1407 [==============================] - 49s 35ms/step - loss: 1.0137 - accuracy: 0.6421 - val_loss: 1.3795 - val_accuracy: 0.5408\n", - "Epoch 37/100\n", - "1407/1407 [==============================] - 49s 35ms/step - loss: 0.9991 - accuracy: 0.6491 - val_loss: 1.3334 - val_accuracy: 0.5444\n", - "157/157 [==============================] - 1s 5ms/step - loss: 1.3184 - accuracy: 0.1154\n" - ] - }, - { - "data": { - "text/plain": [ - "[1.3183687925338745, 0.11540000140666962]" - ] - }, - "execution_count": 135, - "metadata": {}, - "output_type": "execute_result" - } - ], - "source": [ - "keras.backend.clear_session()\n", - "tf.random.set_seed(42)\n", - "np.random.seed(42)\n", - "\n", - "model = keras.models.Sequential()\n", - "model.add(keras.layers.Flatten(input_shape=[32, 32, 3]))\n", - "model.add(keras.layers.BatchNormalization())\n", - "for _ in range(20):\n", - " model.add(keras.layers.Dense(100, kernel_initializer=\"he_normal\"))\n", - " model.add(keras.layers.BatchNormalization())\n", - " model.add(keras.layers.Activation(\"elu\"))\n", - "model.add(keras.layers.Dense(10, activation=\"softmax\"))\n", - "\n", - "optimizer = keras.optimizers.Nadam(lr=5e-4)\n", - "model.compile(loss=\"sparse_categorical_crossentropy\",\n", - " optimizer=optimizer,\n", - " metrics=[\"accuracy\"])\n", - "\n", - "early_stopping_cb = keras.callbacks.EarlyStopping(patience=20)\n", - "model_checkpoint_cb = keras.callbacks.ModelCheckpoint(\"my_cifar10_bn_model.h5\", save_best_only=True)\n", - "run_index = 1 # 모델을 훈련할 때마다 증가시킴\n", - "run_logdir = os.path.join(os.curdir, \"my_cifar10_logs\", \"run_bn_{:03d}\".format(run_index))\n", - "tensorboard_cb = keras.callbacks.TensorBoard(run_logdir)\n", - "callbacks = [early_stopping_cb, model_checkpoint_cb, tensorboard_cb]\n", - "\n", - "model.fit(X_train, y_train, epochs=100,\n", - " validation_data=(X_valid, y_valid),\n", - " callbacks=callbacks)\n", - "\n", - "model = keras.models.load_model(\"my_cifar10_bn_model.h5\")\n", - "model.evaluate(X_valid, y_valid)" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "* *이전보다 빠르게 수렴하나요?* 훨씬 빠릅니다! 이전 모델은 가장 낮은 검증 손실에 도달하기 위해 39 에포크가 걸렸지만 BN을 사용한 새 모델은 18 에포크가 걸렸습니다. 이전 모델보다 두 배 이상 빠릅니다. BN 층은 훈련을 안정적으로 수행하고 더 큰 학습률을 사용할 수 있기 때문에 수렴이 빨라졌습니다.\n", - "* *BN이 더 좋은 모델을 만드나요?* 네! 최종 모델의 성능이 47%가 아니라 55% 정확도로 더 좋습니다. 이는 아주 좋은 모델이 아니지만 적어도 이전보다는 낫습니다(합성곱 신경망이 더 낫겠지만 이는 다른 주제입니다. 14장을 참고하세요).\n", - "* *BN이 훈련 속도에 영향을 미치나요?* 모델이 두 배나 빠르게 수렴했지만 각 에포크는 10초가 아니라 16초가 걸렸습니다. BN 층에서 추가된 계산 때문입니다. 따라서 전체적으로 에포크 횟수가 50% 정도 줄었지만 훈련 시간(탁상 시계 시간)은 30% 정도 줄었습니다. 결국 크게 향상되었습니다!" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "### d.\n", - "*문제: 배치 정규화를 SELU로 바꾸어보세요. 네트워크가 자기 정규화하기 위해 필요한 변경 사항을 적용해보세요(즉, 입력 특성 표준화, 르쿤 정규분포 초기화, 완전 연결 층만 순차적으로 쌓은 심층 신경망 등).*" - ] - }, - { - "cell_type": "code", - "execution_count": 136, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "Epoch 1/100\n", - " 2/1407 [..............................] - ETA: 5:52 - loss: 3.0440 - accuracy: 0.1094WARNING:tensorflow:Callbacks method `on_train_batch_end` is slow compared to the batch time (batch time: 0.0169s vs `on_train_batch_end` time: 0.4852s). Check your callbacks.\n", - "1407/1407 [==============================] - 23s 16ms/step - loss: 1.9306 - accuracy: 0.3076 - val_loss: 1.8329 - val_accuracy: 0.3406\n", - "Epoch 2/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.7066 - accuracy: 0.3947 - val_loss: 1.7136 - val_accuracy: 0.3778\n", - "Epoch 3/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.6115 - accuracy: 0.4309 - val_loss: 1.6743 - val_accuracy: 0.3992\n", - "Epoch 4/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.5459 - accuracy: 0.4578 - val_loss: 1.6328 - val_accuracy: 0.4376\n", - "Epoch 5/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.4916 - accuracy: 0.4795 - val_loss: 1.6314 - val_accuracy: 0.4330\n", - "Epoch 6/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.4437 - accuracy: 0.4977 - val_loss: 1.5327 - val_accuracy: 0.4724\n", - "Epoch 7/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.4014 - accuracy: 0.5092 - val_loss: 1.5317 - val_accuracy: 0.4680\n", - "Epoch 8/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.3726 - accuracy: 0.5226 - val_loss: 1.4981 - val_accuracy: 0.4874\n", - "Epoch 9/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.3349 - accuracy: 0.5344 - val_loss: 1.5136 - val_accuracy: 0.4734\n", - "Epoch 10/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.2971 - accuracy: 0.5486 - val_loss: 1.5214 - val_accuracy: 0.4786\n", - "Epoch 11/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.2706 - accuracy: 0.5600 - val_loss: 1.5285 - val_accuracy: 0.4838\n", - "Epoch 12/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.2470 - accuracy: 0.5695 - val_loss: 1.4795 - val_accuracy: 0.4980\n", - "Epoch 13/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.2192 - accuracy: 0.5766 - val_loss: 1.4753 - val_accuracy: 0.4980\n", - "Epoch 14/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.1908 - accuracy: 0.5918 - val_loss: 1.4862 - val_accuracy: 0.4942\n", - "Epoch 15/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.1665 - accuracy: 0.5987 - val_loss: 1.5071 - val_accuracy: 0.5012\n", - "Epoch 16/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.1519 - accuracy: 0.6057 - val_loss: 1.5167 - val_accuracy: 0.5024\n", - "Epoch 17/100\n", - "1407/1407 [==============================] - 21s 15ms/step - loss: 1.1274 - accuracy: 0.6107 - val_loss: 1.5477 - val_accuracy: 0.4968\n", - "Epoch 18/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.1049 - accuracy: 0.6254 - val_loss: 1.5362 - val_accuracy: 0.5068\n", - "Epoch 19/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.0837 - accuracy: 0.6309 - val_loss: 1.5754 - val_accuracy: 0.5022\n", - "Epoch 20/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.0680 - accuracy: 0.6372 - val_loss: 1.5238 - val_accuracy: 0.5052\n", - "Epoch 21/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.0469 - accuracy: 0.6425 - val_loss: 1.5312 - val_accuracy: 0.5156\n", - "Epoch 22/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.0303 - accuracy: 0.6500 - val_loss: 1.5359 - val_accuracy: 0.5084\n", - "Epoch 23/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.0084 - accuracy: 0.6583 - val_loss: 1.5369 - val_accuracy: 0.5042\n", - "Epoch 24/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 0.9944 - accuracy: 0.6624 - val_loss: 1.6013 - val_accuracy: 0.5006\n", - "Epoch 25/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 0.9793 - accuracy: 0.6706 - val_loss: 1.5512 - val_accuracy: 0.5174\n", - "Epoch 26/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 0.9579 - accuracy: 0.6772 - val_loss: 1.6008 - val_accuracy: 0.4988\n", - "Epoch 27/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 0.9484 - accuracy: 0.6799 - val_loss: 1.5746 - val_accuracy: 0.5110\n", - "Epoch 28/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 48.1621 - accuracy: 0.6212 - val_loss: 1.6674 - val_accuracy: 0.4208\n", - "Epoch 29/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.2912 - accuracy: 0.5517 - val_loss: 1.5598 - val_accuracy: 0.4728\n", - "Epoch 30/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.2315 - accuracy: 0.5735 - val_loss: 1.5738 - val_accuracy: 0.4764\n", - "Epoch 31/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.1684 - accuracy: 0.5940 - val_loss: 1.5514 - val_accuracy: 0.4878\n", - "Epoch 32/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.1293 - accuracy: 0.6084 - val_loss: 1.5258 - val_accuracy: 0.4978\n", - "Epoch 33/100\n", - "1407/1407 [==============================] - 22s 15ms/step - loss: 1.1000 - accuracy: 0.6188 - val_loss: 1.5369 - val_accuracy: 0.4970\n", - "157/157 [==============================] - 1s 3ms/step - loss: 1.4753 - accuracy: 0.1256\n" - ] - }, - { - "data": { - "text/plain": [ - "[1.475338339805603, 0.12559999525547028]" - ] - }, - "execution_count": 136, - "metadata": {}, - "output_type": "execute_result" - } - ], - "source": [ - "keras.backend.clear_session()\n", - "tf.random.set_seed(42)\n", - "np.random.seed(42)\n", - "\n", - "model = keras.models.Sequential()\n", - "model.add(keras.layers.Flatten(input_shape=[32, 32, 3]))\n", - "for _ in range(20):\n", - " model.add(keras.layers.Dense(100,\n", - " kernel_initializer=\"lecun_normal\",\n", - " activation=\"selu\"))\n", - "model.add(keras.layers.Dense(10, activation=\"softmax\"))\n", - "\n", - "optimizer = keras.optimizers.Nadam(lr=7e-4)\n", - "model.compile(loss=\"sparse_categorical_crossentropy\",\n", - " optimizer=optimizer,\n", - " metrics=[\"accuracy\"])\n", - "\n", - "early_stopping_cb = keras.callbacks.EarlyStopping(patience=20)\n", - "model_checkpoint_cb = keras.callbacks.ModelCheckpoint(\"my_cifar10_selu_model.h5\", save_best_only=True)\n", - "run_index = 1 # 모델을 훈련할 때마다 증가시킴\n", - "run_logdir = os.path.join(os.curdir, \"my_cifar10_logs\", \"run_selu_{:03d}\".format(run_index))\n", - "tensorboard_cb = keras.callbacks.TensorBoard(run_logdir)\n", - "callbacks = [early_stopping_cb, model_checkpoint_cb, tensorboard_cb]\n", - "\n", - "X_means = X_train.mean(axis=0)\n", - "X_stds = X_train.std(axis=0)\n", - "X_train_scaled = (X_train - X_means) / X_stds\n", - "X_valid_scaled = (X_valid - X_means) / X_stds\n", - "X_test_scaled = (X_test - X_means) / X_stds\n", - "\n", - "model.fit(X_train_scaled, y_train, epochs=100,\n", - " validation_data=(X_valid_scaled, y_valid),\n", - " callbacks=callbacks)\n", - "\n", - "model = keras.models.load_model(\"my_cifar10_selu_model.h5\")\n", - "model.evaluate(X_valid_scaled, y_valid)" - ] - }, - { - "cell_type": "code", - "execution_count": 137, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "157/157 [==============================] - 1s 3ms/step - loss: 1.4753 - accuracy: 0.1256\n" - ] - }, - { - "data": { - "text/plain": [ - "[1.475338339805603, 0.12559999525547028]" - ] - }, - "execution_count": 137, - "metadata": {}, - "output_type": "execute_result" - } - ], - "source": [ - "model = keras.models.load_model(\"my_cifar10_selu_model.h5\")\n", - "model.evaluate(X_valid_scaled, y_valid)" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "51.4% 정확도를 얻었습니다. 원래 모델보다 더 좋습니다. 하지만 배치 정규화를 사용한 모델만큼 좋지는 않습니다. 최고의 모델에 도달하는데 13 에포크가 걸렸습니다. 이는 원본 모델이나 BN 모델보다 더 빠른 것입니다. 각 에포크는 원본 모델처럼 10초만 걸렸습니다. 따라서 이 모델이 지금까지 가장 빠른 모델입니다(에포크와 탁상 시계 기준으로)." - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "### e.\n", - "*문제: 알파 드롭아웃으로 모델에 규제를 적용해보세요. 그다음 모델을 다시 훈련하지 않고 MC 드롭아웃으로 더 높은 정확도를 얻을 수 있는지 확인해보세요.*" - ] - }, - { - "cell_type": "code", - "execution_count": 138, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "Epoch 1/100\n", - " 2/1407 [..............................] - ETA: 4:07 - loss: 2.9857 - accuracy: 0.0938WARNING:tensorflow:Callbacks method `on_train_batch_end` is slow compared to the batch time (batch time: 0.0168s vs `on_train_batch_end` time: 0.3359s). Check your callbacks.\n", - "1407/1407 [==============================] - 23s 17ms/step - loss: 1.8896 - accuracy: 0.3275 - val_loss: 1.7313 - val_accuracy: 0.3970\n", - "Epoch 2/100\n", - "1407/1407 [==============================] - 23s 16ms/step - loss: 1.6589 - accuracy: 0.4157 - val_loss: 1.7183 - val_accuracy: 0.3916\n", - "Epoch 3/100\n", - "1407/1407 [==============================] - 23s 16ms/step - loss: 1.5727 - accuracy: 0.4479 - val_loss: 1.6073 - val_accuracy: 0.4364\n", - "Epoch 4/100\n", - "1407/1407 [==============================] - 23s 16ms/step - loss: 1.5085 - accuracy: 0.4734 - val_loss: 1.5741 - val_accuracy: 0.4524\n", - "Epoch 5/100\n", - "1407/1407 [==============================] - 23s 16ms/step - loss: 1.4525 - accuracy: 0.4946 - val_loss: 1.5663 - val_accuracy: 0.4592\n", - "Epoch 6/100\n", - "1407/1407 [==============================] - 23s 16ms/step - loss: 1.4032 - accuracy: 0.5124 - val_loss: 1.5255 - val_accuracy: 0.4644\n", - "Epoch 7/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.3581 - accuracy: 0.5255 - val_loss: 1.6598 - val_accuracy: 0.4662\n", - "Epoch 8/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.3209 - accuracy: 0.5400 - val_loss: 1.5027 - val_accuracy: 0.5002\n", - "Epoch 9/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.2845 - accuracy: 0.5562 - val_loss: 1.5246 - val_accuracy: 0.4896\n", - "Epoch 10/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.2526 - accuracy: 0.5659 - val_loss: 1.5510 - val_accuracy: 0.4956\n", - "Epoch 11/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.2160 - accuracy: 0.5808 - val_loss: 1.5559 - val_accuracy: 0.5002\n", - "Epoch 12/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.1902 - accuracy: 0.5900 - val_loss: 1.5478 - val_accuracy: 0.4968\n", - "Epoch 13/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.1602 - accuracy: 0.6021 - val_loss: 1.5727 - val_accuracy: 0.5124\n", - "Epoch 14/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.1392 - accuracy: 0.6102 - val_loss: 1.5654 - val_accuracy: 0.4944\n", - "Epoch 15/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.1086 - accuracy: 0.6210 - val_loss: 1.5868 - val_accuracy: 0.5064\n", - "Epoch 16/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.0856 - accuracy: 0.6289 - val_loss: 1.6016 - val_accuracy: 0.5042\n", - "Epoch 17/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.0620 - accuracy: 0.6397 - val_loss: 1.6458 - val_accuracy: 0.4968\n", - "Epoch 18/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.0511 - accuracy: 0.6405 - val_loss: 1.6276 - val_accuracy: 0.5096\n", - "Epoch 19/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.0203 - accuracy: 0.6514 - val_loss: 1.7246 - val_accuracy: 0.5062\n", - "Epoch 20/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 1.0024 - accuracy: 0.6598 - val_loss: 1.6570 - val_accuracy: 0.5064\n", - "Epoch 21/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 0.9845 - accuracy: 0.6662 - val_loss: 1.6697 - val_accuracy: 0.4990\n", - "Epoch 22/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 0.9641 - accuracy: 0.6738 - val_loss: 1.7560 - val_accuracy: 0.5010\n", - "Epoch 23/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 0.9387 - accuracy: 0.6797 - val_loss: 1.7716 - val_accuracy: 0.5008\n", - "Epoch 24/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 0.9290 - accuracy: 0.6852 - val_loss: 1.7688 - val_accuracy: 0.5026\n", - "Epoch 25/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 0.9176 - accuracy: 0.6899 - val_loss: 1.8131 - val_accuracy: 0.5042\n", - "Epoch 26/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 0.8925 - accuracy: 0.6986 - val_loss: 1.8228 - val_accuracy: 0.4904\n", - "Epoch 27/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 0.8680 - accuracy: 0.7060 - val_loss: 1.8546 - val_accuracy: 0.5048\n", - "Epoch 28/100\n", - "1407/1407 [==============================] - 22s 16ms/step - loss: 0.8638 - accuracy: 0.7091 - val_loss: 1.8004 - val_accuracy: 0.4954\n", - "157/157 [==============================] - 1s 3ms/step - loss: 1.5027 - accuracy: 0.0914\n" - ] - }, - { - "data": { - "text/plain": [ - "[1.5026599168777466, 0.09139999747276306]" - ] - }, - "execution_count": 138, - "metadata": {}, - "output_type": "execute_result" - } - ], - "source": [ - "keras.backend.clear_session()\n", - "tf.random.set_seed(42)\n", - "np.random.seed(42)\n", - "\n", - "model = keras.models.Sequential()\n", - "model.add(keras.layers.Flatten(input_shape=[32, 32, 3]))\n", - "for _ in range(20):\n", - " model.add(keras.layers.Dense(100,\n", - " kernel_initializer=\"lecun_normal\",\n", - " activation=\"selu\"))\n", - "\n", - "model.add(keras.layers.AlphaDropout(rate=0.1))\n", - "model.add(keras.layers.Dense(10, activation=\"softmax\"))\n", - "\n", - "optimizer = keras.optimizers.Nadam(lr=5e-4)\n", - "model.compile(loss=\"sparse_categorical_crossentropy\",\n", - " optimizer=optimizer,\n", - " metrics=[\"accuracy\"])\n", - "\n", - "early_stopping_cb = keras.callbacks.EarlyStopping(patience=20)\n", - "model_checkpoint_cb = keras.callbacks.ModelCheckpoint(\"my_cifar10_alpha_dropout_model.h5\", save_best_only=True)\n", - "run_index = 1 # 모델을 훈련할 때마다 증가시킴\n", - "run_logdir = os.path.join(os.curdir, \"my_cifar10_logs\", \"run_alpha_dropout_{:03d}\".format(run_index))\n", - "tensorboard_cb = keras.callbacks.TensorBoard(run_logdir)\n", - "callbacks = [early_stopping_cb, model_checkpoint_cb, tensorboard_cb]\n", - "\n", - "X_means = X_train.mean(axis=0)\n", - "X_stds = X_train.std(axis=0)\n", - "X_train_scaled = (X_train - X_means) / X_stds\n", - "X_valid_scaled = (X_valid - X_means) / X_stds\n", - "X_test_scaled = (X_test - X_means) / X_stds\n", - "\n", - "model.fit(X_train_scaled, y_train, epochs=100,\n", - " validation_data=(X_valid_scaled, y_valid),\n", - " callbacks=callbacks)\n", - "\n", - "model = keras.models.load_model(\"my_cifar10_alpha_dropout_model.h5\")\n", - "model.evaluate(X_valid_scaled, y_valid)" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "이 모델은 검증 세트에서 50.8% 정확도에 도달합니다. 드롭아웃이 없을 때보다(51.4%) 조금 더 나쁩니다. 하이퍼파라미터 탐색을 좀 많이 수행해 보면 더 나아 질 수 있습니다(드롭아웃 비율 5%, 10%, 20%, 40%과 학습률 1e-4, 3e-4, 5e-4, 1e-3을 시도했습니다). 하지만 이 경우에는 크지 않을 것 같습니다." - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "이제 MC 드롭아웃을 사용해 보죠. 앞서 사용한 `MCAlphaDropout` 클래스를 복사해 사용하겠습니다:" - ] - }, - { - "cell_type": "code", - "execution_count": 139, - "metadata": {}, - "outputs": [], - "source": [ - "class MCAlphaDropout(keras.layers.AlphaDropout):\n", - " def call(self, inputs):\n", - " return super().call(inputs, training=True)" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "방금 훈련했던 모델과 (같은 가중치를 가진) 동일한 새로운 모델을 만들어 보죠. 하지만 `AlphaDropout` 층 대신 `MCAlphaDropout` 드롭아웃 층을 사용합니다:" - ] - }, - { - "cell_type": "code", - "execution_count": 140, - "metadata": {}, - "outputs": [], - "source": [ - "mc_model = keras.models.Sequential([\n", - " MCAlphaDropout(layer.rate) if isinstance(layer, keras.layers.AlphaDropout) else layer\n", - " for layer in model.layers\n", - "])" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "그다음 몇 가지 유틸리티 함수를 추가합니다. 첫 번째 함수는 모델을 여러 번 실행합니다(기본적으로 10번). 그다음 평균한 예측 클래스 확률을 반환합니다. 두 번째 함수는 이 평균 확률을 사용해 각 샘플의 클래스를 예측합니다:" - ] - }, - { - "cell_type": "code", - "execution_count": 141, - "metadata": {}, - "outputs": [], - "source": [ - "def mc_dropout_predict_probas(mc_model, X, n_samples=10):\n", - " Y_probas = [mc_model.predict(X) for sample in range(n_samples)]\n", - " return np.mean(Y_probas, axis=0)\n", - "\n", - "def mc_dropout_predict_classes(mc_model, X, n_samples=10):\n", - " Y_probas = mc_dropout_predict_probas(mc_model, X, n_samples)\n", - " return np.argmax(Y_probas, axis=1)" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "이제 검증 세트의 모든 샘플에 대해 예측을 만들고 정확도를 계산해 보죠:" - ] - }, - { - "cell_type": "code", - "execution_count": 142, - "metadata": {}, - "outputs": [ - { - "data": { - "text/plain": [ - "0.497" - ] - }, - "execution_count": 142, - "metadata": {}, - "output_type": "execute_result" - } - ], - "source": [ - "keras.backend.clear_session()\n", - "tf.random.set_seed(42)\n", - "np.random.seed(42)\n", - "\n", - "y_pred = mc_dropout_predict_classes(mc_model, X_valid_scaled)\n", - "accuracy = np.mean(y_pred == y_valid[:, 0])\n", - "accuracy" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "이 경우에는 실제적인 정확도 향상이 없습니다(50.8%에서 50.9%).\n", - "\n", - "따라서 이 연습문에서 얻은 최상의 모델은 배치 정규화 모델입니다." - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "### f.\n", - "*문제: 1사이클 스케줄링으로 모델을 다시 훈련하고 훈련 속도와 모델 정확도가 향상되는지 확인해보세요.*" - ] - }, - { - "cell_type": "code", - "execution_count": 143, - "metadata": {}, - "outputs": [], - "source": [ - "keras.backend.clear_session()\n", - "tf.random.set_seed(42)\n", - "np.random.seed(42)\n", - "\n", - "model = keras.models.Sequential()\n", - "model.add(keras.layers.Flatten(input_shape=[32, 32, 3]))\n", - "for _ in range(20):\n", - " model.add(keras.layers.Dense(100,\n", - " kernel_initializer=\"lecun_normal\",\n", - " activation=\"selu\"))\n", - "\n", - "model.add(keras.layers.AlphaDropout(rate=0.1))\n", - "model.add(keras.layers.Dense(10, activation=\"softmax\"))\n", - "\n", - "optimizer = keras.optimizers.SGD(lr=1e-3)\n", - "model.compile(loss=\"sparse_categorical_crossentropy\",\n", - " optimizer=optimizer,\n", - " metrics=[\"accuracy\"])" - ] - }, - { - "cell_type": "code", - "execution_count": 144, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "352/352 [==============================] - 2s 7ms/step - loss: nan - accuracy: 0.1399\n" - ] - }, - { - "data": { - "text/plain": [ - "[1e-05, 9.999868, 2.6130447387695312, 4.006446089063372]" - ] - }, - "execution_count": 144, - "metadata": {}, - "output_type": "execute_result" - }, - { - "data": { - "image/png": "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\n", - "text/plain": [ - "
" - ] - }, - "metadata": { - "needs_background": "light" - }, - "output_type": "display_data" - } - ], - "source": [ - "batch_size = 128\n", - "rates, losses = find_learning_rate(model, X_train_scaled, y_train, epochs=1, batch_size=batch_size)\n", - "plot_lr_vs_loss(rates, losses)\n", - "plt.axis([min(rates), max(rates), min(losses), (losses[0] + min(losses)) / 1.4])" - ] - }, - { - "cell_type": "code", - "execution_count": 145, - "metadata": {}, - "outputs": [], - "source": [ - "keras.backend.clear_session()\n", - "tf.random.set_seed(42)\n", - "np.random.seed(42)\n", - "\n", - "model = keras.models.Sequential()\n", - "model.add(keras.layers.Flatten(input_shape=[32, 32, 3]))\n", - "for _ in range(20):\n", - " model.add(keras.layers.Dense(100,\n", - " kernel_initializer=\"lecun_normal\",\n", - " activation=\"selu\"))\n", - "\n", - "model.add(keras.layers.AlphaDropout(rate=0.1))\n", - "model.add(keras.layers.Dense(10, activation=\"softmax\"))\n", - "\n", - "optimizer = keras.optimizers.SGD(lr=1e-2)\n", - "model.compile(loss=\"sparse_categorical_crossentropy\",\n", - " optimizer=optimizer,\n", - " metrics=[\"accuracy\"])" - ] - }, - { - "cell_type": "code", - "execution_count": 146, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "Epoch 1/15\n", - "352/352 [==============================] - 3s 9ms/step - loss: 2.0537 - accuracy: 0.2843 - val_loss: 1.7811 - val_accuracy: 0.3744\n", - "Epoch 2/15\n", - "352/352 [==============================] - 3s 7ms/step - loss: 1.7635 - accuracy: 0.3765 - val_loss: 1.6431 - val_accuracy: 0.4252\n", - "Epoch 3/15\n", - "352/352 [==============================] - 3s 7ms/step - loss: 1.6241 - accuracy: 0.4217 - val_loss: 1.6001 - val_accuracy: 0.4368\n", - "Epoch 4/15\n", - "352/352 [==============================] - 3s 7ms/step - loss: 1.5434 - accuracy: 0.4520 - val_loss: 1.6114 - val_accuracy: 0.4310\n", - "Epoch 5/15\n", - "352/352 [==============================] - 3s 7ms/step - loss: 1.4914 - accuracy: 0.4710 - val_loss: 1.5895 - val_accuracy: 0.4434\n", - "Epoch 6/15\n", - "352/352 [==============================] - 3s 7ms/step - loss: 1.4510 - accuracy: 0.4818 - val_loss: 1.5678 - val_accuracy: 0.4506\n", - "Epoch 7/15\n", - "352/352 [==============================] - 3s 7ms/step - loss: 1.4143 - accuracy: 0.4979 - val_loss: 1.6717 - val_accuracy: 0.4294\n", - "Epoch 8/15\n", - "352/352 [==============================] - 3s 7ms/step - loss: 1.3462 - accuracy: 0.5199 - val_loss: 1.4928 - val_accuracy: 0.4956\n", - "Epoch 9/15\n", - "352/352 [==============================] - 3s 7ms/step - loss: 1.2691 - accuracy: 0.5481 - val_loss: 1.5294 - val_accuracy: 0.4818\n", - "Epoch 10/15\n", - "352/352 [==============================] - 3s 7ms/step - loss: 1.1994 - accuracy: 0.5713 - val_loss: 1.5165 - val_accuracy: 0.4978\n", - "Epoch 11/15\n", - "352/352 [==============================] - 3s 7ms/step - loss: 1.1308 - accuracy: 0.5980 - val_loss: 1.5070 - val_accuracy: 0.5100\n", - "Epoch 12/15\n", - "352/352 [==============================] - 3s 7ms/step - loss: 1.0632 - accuracy: 0.6184 - val_loss: 1.4833 - val_accuracy: 0.5244\n", - "Epoch 13/15\n", - "352/352 [==============================] - 3s 7ms/step - loss: 0.9932 - accuracy: 0.6447 - val_loss: 1.5314 - val_accuracy: 0.5292\n", - "Epoch 14/15\n", - "352/352 [==============================] - 3s 7ms/step - loss: 0.9279 - accuracy: 0.6671 - val_loss: 1.5495 - val_accuracy: 0.5248\n", - "Epoch 15/15\n", - "352/352 [==============================] - 3s 7ms/step - loss: 0.8880 - accuracy: 0.6845 - val_loss: 1.5840 - val_accuracy: 0.5288\n" - ] + "cells": [ + { + "cell_type": "markdown", + "metadata": { + "id": "YT0En5VzLOo4" + }, + "source": [ + "**11장 – 심층 신경망 훈련하기**" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "fmWAAqyeLOo9" + }, + "source": [ + "_이 노트북은 11장에 있는 모든 샘플 코드와 연습문제 해답을 가지고 있습니다._" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "mYZM1mKhLOo9" + }, + "source": [ + "\n", + " \n", + "
\n", + " 구글 코랩에서 실행하기\n", + "
" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "4mgMAdZzLOo-" + }, + "source": [ + "# 설정" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "EnYYn3E9LOo-" + }, + "source": [ + "먼저 몇 개의 모듈을 임포트합니다. 맷플롯립 그래프를 인라인으로 출력하도록 만들고 그림을 저장하는 함수를 준비합니다. 또한 파이썬 버전이 3.5 이상인지 확인합니다(파이썬 2.x에서도 동작하지만 곧 지원이 중단되므로 파이썬 3을 사용하는 것이 좋습니다). 사이킷런 버전이 0.20 이상인지와 텐서플로 버전이 2.0 이상인지 확인합니다." + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "M1cwJnMALOo-" + }, + "source": [ + "# 파이썬 ≥3.5 필수\n", + "import sys\n", + "assert sys.version_info >= (3, 5)\n", + "\n", + "# 사이킷런 ≥0.20 필수\n", + "import sklearn\n", + "assert sklearn.__version__ >= \"0.20\"\n", + "\n", + "# 텐서플로 ≥2.0 필수\n", + "import tensorflow as tf\n", + "from tensorflow import keras\n", + "assert tf.__version__ >= \"2.0\"\n", + "\n", + "%load_ext tensorboard\n", + "\n", + "# 공통 모듈 임포트\n", + "import numpy as np\n", + "import os\n", + "\n", + "# 노트북 실행 결과를 동일하게 유지하기 위해\n", + "np.random.seed(42)\n", + "\n", + "# 깔끔한 그래프 출력을 위해\n", + "%matplotlib inline\n", + "import matplotlib as mpl\n", + "import matplotlib.pyplot as plt\n", + "mpl.rc('axes', labelsize=14)\n", + "mpl.rc('xtick', labelsize=12)\n", + "mpl.rc('ytick', labelsize=12)\n", + "\n", + "# 그림을 저장할 위치\n", + "PROJECT_ROOT_DIR = \".\"\n", + "CHAPTER_ID = \"deep\"\n", + "IMAGES_PATH = os.path.join(PROJECT_ROOT_DIR, \"images\", CHAPTER_ID)\n", + "os.makedirs(IMAGES_PATH, exist_ok=True)\n", + "\n", + "def save_fig(fig_id, tight_layout=True, fig_extension=\"png\", resolution=300):\n", + " path = os.path.join(IMAGES_PATH, fig_id + \".\" + fig_extension)\n", + " print(\"그림 저장:\", fig_id)\n", + " if tight_layout:\n", + " plt.tight_layout()\n", + " plt.savefig(path, format=fig_extension, dpi=resolution)" + ], + "execution_count": 1, + "outputs": [] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "Uts4Q0qDLOpA" + }, + "source": [ + "# 그레이디언트 소실과 폭주 문제" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "C27geIKhLOpA" + }, + "source": [ + "def logit(z):\n", + " return 1 / (1 + np.exp(-z))" + ], + "execution_count": 2, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "YnIN46vHLOpA", + "outputId": "58f5e1b1-9ff7-46fd-9779-c7eb28f29c1c", + "colab": { + "base_uri": "https://localhost:8080/", + "height": 315 + } + }, + "source": [ + "z = np.linspace(-5, 5, 200)\n", + "\n", + "plt.plot([-5, 5], [0, 0], 'k-')\n", + "plt.plot([-5, 5], [1, 1], 'k--')\n", + "plt.plot([0, 0], [-0.2, 1.2], 'k-')\n", + "plt.plot([-5, 5], [-3/4, 7/4], 'g--')\n", + "plt.plot(z, logit(z), \"b-\", linewidth=2)\n", + "props = dict(facecolor='black', shrink=0.1)\n", + "plt.annotate('Saturating', xytext=(3.5, 0.7), xy=(5, 1), arrowprops=props, fontsize=14, ha=\"center\")\n", + "plt.annotate('Saturating', xytext=(-3.5, 0.3), xy=(-5, 0), arrowprops=props, fontsize=14, ha=\"center\")\n", + "plt.annotate('Linear', xytext=(2, 0.2), xy=(0, 0.5), arrowprops=props, fontsize=14, ha=\"center\")\n", + "plt.grid(True)\n", + "plt.title(\"Sigmoid activation function\", fontsize=14)\n", + "plt.axis([-5, 5, -0.2, 1.2])\n", + "\n", + "save_fig(\"sigmoid_saturation_plot\")\n", + "plt.show()" + ], + "execution_count": 3, + "outputs": [ + { + "output_type": "stream", + "text": [ + "그림 저장: sigmoid_saturation_plot\n" + ], + "name": "stdout" + }, + { + "output_type": "display_data", + "data": { + "image/png": "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\n", + "text/plain": [ + "
" + ] + }, + "metadata": { + "tags": [], + "needs_background": "light" + } + } + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "z9my8UBcLOpB" + }, + "source": [ + "## Xavier 초기화와 He 초기화" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "P0IEVGXYLOpC", + "outputId": "d9bbf038-78f4-43bc-8b80-0e352849cea5", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "[name for name in dir(keras.initializers) if not name.startswith(\"_\")]" + ], + "execution_count": 4, + "outputs": [ + { + "output_type": "execute_result", + "data": { + "text/plain": [ + "['Constant',\n", + " 'GlorotNormal',\n", + " 'GlorotUniform',\n", + " 'HeNormal',\n", + " 'HeUniform',\n", + " 'Identity',\n", + " 'Initializer',\n", + " 'LecunNormal',\n", + " 'LecunUniform',\n", + " 'Ones',\n", + " 'Orthogonal',\n", + " 'RandomNormal',\n", + " 'RandomUniform',\n", + " 'TruncatedNormal',\n", + " 'VarianceScaling',\n", + " 'Zeros',\n", + " 'constant',\n", + " 'deserialize',\n", + " 'get',\n", + " 'glorot_normal',\n", + " 'glorot_uniform',\n", + " 'he_normal',\n", + " 'he_uniform',\n", + " 'identity',\n", + " 'lecun_normal',\n", + " 'lecun_uniform',\n", + " 'ones',\n", + " 'orthogonal',\n", + " 'random_normal',\n", + " 'random_uniform',\n", + " 'serialize',\n", + " 'truncated_normal',\n", + " 'variance_scaling',\n", + " 'zeros']" + ] + }, + "metadata": { + "tags": [] + }, + "execution_count": 4 + } + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "kBczZ42qLOpC", + "outputId": "72b19158-02f2-4689-b333-3cad5f5ec4bb", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "keras.layers.Dense(10, activation=\"relu\", kernel_initializer=\"he_normal\")" + ], + "execution_count": 5, + "outputs": [ + { + "output_type": "execute_result", + "data": { + "text/plain": [ + "" + ] + }, + "metadata": { + "tags": [] + }, + "execution_count": 5 + } + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "iSo-rN5kLOpC", + "outputId": "7fb5008f-e89b-48b9-d6f9-5f7bb5f76f8d", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "init = keras.initializers.VarianceScaling(scale=2., mode='fan_avg',\n", + " distribution='uniform')\n", + "keras.layers.Dense(10, activation=\"relu\", kernel_initializer=init)" + ], + "execution_count": 6, + "outputs": [ + { + "output_type": "execute_result", + "data": { + "text/plain": [ + "" + ] + }, + "metadata": { + "tags": [] + }, + "execution_count": 6 + } + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "ehWhAiMGLOpC" + }, + "source": [ + "## 수렴하지 않는 활성화 함수" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "Xm1OvmTKLOpD" + }, + "source": [ + "### LeakyReLU" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "WPfFeM26LOpD" + }, + "source": [ + "def leaky_relu(z, alpha=0.01):\n", + " return np.maximum(alpha*z, z)" + ], + "execution_count": 7, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "mmZMEcRNLOpD", + "outputId": "f26e717a-01f4-436f-d839-57520655d516", + "colab": { + "base_uri": "https://localhost:8080/", + "height": 315 + } + }, + "source": [ + "plt.plot(z, leaky_relu(z, 0.05), \"b-\", linewidth=2)\n", + "plt.plot([-5, 5], [0, 0], 'k-')\n", + "plt.plot([0, 0], [-0.5, 4.2], 'k-')\n", + "plt.grid(True)\n", + "props = dict(facecolor='black', shrink=0.1)\n", + "plt.annotate('Leak', xytext=(-3.5, 0.5), xy=(-5, -0.2), arrowprops=props, fontsize=14, ha=\"center\")\n", + "plt.title(\"Leaky ReLU activation function\", fontsize=14)\n", + "plt.axis([-5, 5, -0.5, 4.2])\n", + "\n", + "save_fig(\"leaky_relu_plot\")\n", + "plt.show()" + ], + "execution_count": 8, + "outputs": [ + { + "output_type": "stream", + "text": [ + "그림 저장: leaky_relu_plot\n" + ], + "name": "stdout" + }, + { + "output_type": "display_data", + "data": { + "image/png": "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/vKXQScSiY0KVMjo0aMZNGhQ0DFEGqW42I/ttGOH7zHippuCTiQSOxUokRRRXQ0jRvgjqCOPhJkzwSzoVCKxU4GKwMqVKxk4cCDt27enc+fOnH/++WzatOnbx5ctW8app55Kbm4u2dnZnHjiibz33nv7XOc999xDbm4uf//73+MdX9LELbf4a0+dOvlGEVlZQScSaRwVqAZs3LiRk08+mT59+vD++++zaNEiSkpKOOuss6ip8SOP7ty5k4suuoh33nmH999/n379+nHGGWfw9ddf77U+5xwTJkxg6tSpvPXWWxx33HHN/ZQkBc2Z43/j1LIlzJ0L//VfQScSabwmHVE3FT322GP07duXe+6559t5zzzzDJ06daKwsJD+/fvzs5/9bI+/mTp1KvPnz+cvf/kLF1544bfzq6urueSSS1iyZAlLliyhe/fuzfY8JHUtXw4XX+ynH3gABgwINo9IU1GBasAHH3zA22+/TVYd50vWrFlD//792bJlC7fccgt/+9vf2Lx5M9XV1ZSVlfH555/vsfyECRPIyMhg6dKldO7cubmegqSwrVt9o4iyMhg9GsaNCzqRSNNRgWpATU0NAwcOZMqUKXs91iXUHfSoUaPYvHkzDz74ID169GC//fZjwIABVFZW7rH8z3/+c5577jlee+01Ro8e3RzxJYXt2gXDhsHnn8OPfuR7K1ejCEklKlANOProo5k7dy7du3enVatWdS6zePFiHn74YQYOHAjA5s2b2bhx417LnXHGGZxzzjkMGzYMM2PUqFFxzS6p7Ve/8r2UH3ggLFjgh9EQSSVqJBFmx44dLF++fI/bwIEDKS4u5rzzzmPp0qV89tlnLFq0iLFjx7Jz504AevbsyaxZs1i5ciXLli1j+PDhtK5nJLhBgwbxwgsvcMUVV/DMM88059OTFPLUU/CHP/gBBxcsgK5dg04k0vR0BBXmnXfe4aijjtpj3pAhQ1iyZAkTJ07k9NNPp7y8nG7dunHqqaey3377ATBjxgzGjh3LMcccQ9euXZk0aRJbt26tdzuDBg1i7ty5nHvuuQCMHDkyfk9KUs6778JVV/npadNADUElValAhcycOZOZM2fW+/i+uinq27cvS5cu3WPeRRddtMd959we988880zKysqiDyppbf16OOccf/3pmmu+a70nkop0ik8kSZSVweDBsHkz/OxnUEe7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\n", + "text/plain": [ + "
" + ] + }, + "metadata": { + "tags": [], + "needs_background": "light" + } + } + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "c34zBGD7LOpD", + "outputId": "f4b860b4-3a2d-4a12-e458-de2efaa93051", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "[m for m in dir(keras.activations) if not m.startswith(\"_\")]" + ], + "execution_count": 9, + "outputs": [ + { + "output_type": "execute_result", + "data": { + "text/plain": [ + "['deserialize',\n", + " 'elu',\n", + " 'exponential',\n", + " 'gelu',\n", + " 'get',\n", + " 'hard_sigmoid',\n", + " 'linear',\n", + " 'relu',\n", + " 'selu',\n", + " 'serialize',\n", + " 'sigmoid',\n", + " 'softmax',\n", + " 'softplus',\n", + " 'softsign',\n", + " 'swish',\n", + " 'tanh']" + ] + }, + "metadata": { + "tags": [] + }, + "execution_count": 9 + } + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "eD78rel6LOpD", + "outputId": "777d81bf-2701-4898-96c1-e9f18627cf82", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "[m for m in dir(keras.layers) if \"relu\" in m.lower()]" + ], + "execution_count": 10, + "outputs": [ + { + "output_type": "execute_result", + "data": { + "text/plain": [ + "['LeakyReLU', 'PReLU', 'ReLU', 'ThresholdedReLU']" + ] + }, + "metadata": { + "tags": [] + }, + "execution_count": 10 + } + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "zT5BPUCELOpE" + }, + "source": [ + "LeakyReLU를 사용해 패션 MNIST에서 신경망을 훈련해 보죠:" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "aW_rOsgrLOpE", + "outputId": "43769acc-8742-4108-fe80-83acde336789", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "(X_train_full, y_train_full), (X_test, y_test) = keras.datasets.fashion_mnist.load_data()\n", + "X_train_full = X_train_full / 255.0\n", + "X_test = X_test / 255.0\n", + "X_valid, X_train = X_train_full[:5000], X_train_full[5000:]\n", + "y_valid, y_train = y_train_full[:5000], y_train_full[5000:]" + ], + "execution_count": 11, + "outputs": [ + { + "output_type": "stream", + "text": [ + "Downloading data from https://storage.googleapis.com/tensorflow/tf-keras-datasets/train-labels-idx1-ubyte.gz\n", + "32768/29515 [=================================] - 0s 0us/step\n", + "Downloading data from https://storage.googleapis.com/tensorflow/tf-keras-datasets/train-images-idx3-ubyte.gz\n", + "26427392/26421880 [==============================] - 0s 0us/step\n", + "Downloading data from https://storage.googleapis.com/tensorflow/tf-keras-datasets/t10k-labels-idx1-ubyte.gz\n", + "8192/5148 [===============================================] - 0s 0us/step\n", + "Downloading data from https://storage.googleapis.com/tensorflow/tf-keras-datasets/t10k-images-idx3-ubyte.gz\n", + "4423680/4422102 [==============================] - 0s 0us/step\n" + ], + "name": "stdout" + } + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "zvSaZ4qGLOpE" + }, + "source": [ + "tf.random.set_seed(42)\n", + "np.random.seed(42)\n", + "\n", + "model = keras.models.Sequential([\n", + " keras.layers.Flatten(input_shape=[28, 28]),\n", + " keras.layers.Dense(300, kernel_initializer=\"he_normal\"),\n", + " keras.layers.LeakyReLU(),\n", + " keras.layers.Dense(100, kernel_initializer=\"he_normal\"),\n", + " keras.layers.LeakyReLU(),\n", + " keras.layers.Dense(10, activation=\"softmax\")\n", + "])" + ], + "execution_count": 12, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "8iTzAFZ5LOpE" + }, + "source": [ + "model.compile(loss=\"sparse_categorical_crossentropy\",\n", + " optimizer=keras.optimizers.SGD(lr=1e-3),\n", + " metrics=[\"accuracy\"])" + ], + "execution_count": 13, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "scrolled": true, + "id": "JqVTPhfHLOpE", + "outputId": "9efa9999-4ff7-44b9-94b0-9e5bbd8c1a61", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "history = model.fit(X_train, y_train, epochs=10,\n", + " validation_data=(X_valid, y_valid))" + ], + "execution_count": 14, + "outputs": [ + { + "output_type": "stream", + "text": [ + "Epoch 1/10\n", + "1719/1719 [==============================] - 5s 2ms/step - loss: 1.6314 - accuracy: 0.5054 - val_loss: 0.8886 - val_accuracy: 0.7160\n", + "Epoch 2/10\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.8416 - accuracy: 0.7247 - val_loss: 0.7130 - val_accuracy: 0.7656\n", + "Epoch 3/10\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.7053 - accuracy: 0.7637 - val_loss: 0.6427 - val_accuracy: 0.7898\n", + "Epoch 4/10\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.6325 - accuracy: 0.7908 - val_loss: 0.5900 - val_accuracy: 0.8066\n", + "Epoch 5/10\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.5992 - accuracy: 0.8021 - val_loss: 0.5582 - val_accuracy: 0.8198\n", + "Epoch 6/10\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.5624 - accuracy: 0.8142 - val_loss: 0.5350 - val_accuracy: 0.8238\n", + "Epoch 7/10\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.5379 - accuracy: 0.8217 - val_loss: 0.5156 - val_accuracy: 0.8304\n", + "Epoch 8/10\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.5152 - accuracy: 0.8296 - val_loss: 0.5079 - val_accuracy: 0.8284\n", + "Epoch 9/10\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.5100 - accuracy: 0.8270 - val_loss: 0.4895 - val_accuracy: 0.8388\n", + "Epoch 10/10\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.4918 - accuracy: 0.8339 - val_loss: 0.4817 - val_accuracy: 0.8398\n" + ], + "name": "stdout" + } + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "O4AS1dmTLOpE" + }, + "source": [ + "PReLU를 테스트해 보죠:" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "eLqcJVxfLOpF" + }, + "source": [ + "tf.random.set_seed(42)\n", + "np.random.seed(42)\n", + "\n", + "model = keras.models.Sequential([\n", + " keras.layers.Flatten(input_shape=[28, 28]),\n", + " keras.layers.Dense(300, kernel_initializer=\"he_normal\"),\n", + " keras.layers.PReLU(),\n", + " keras.layers.Dense(100, kernel_initializer=\"he_normal\"),\n", + " keras.layers.PReLU(),\n", + " keras.layers.Dense(10, activation=\"softmax\")\n", + "])" + ], + "execution_count": 15, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "BMKER0vWLOpF" + }, + "source": [ + "model.compile(loss=\"sparse_categorical_crossentropy\",\n", + " optimizer=keras.optimizers.SGD(lr=1e-3),\n", + " metrics=[\"accuracy\"])" + ], + "execution_count": 16, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "eb_t9z8gLOpF", + "outputId": "20f6177f-04fa-4de4-905d-ab7e94014b71", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "history = model.fit(X_train, y_train, epochs=10,\n", + " validation_data=(X_valid, y_valid))" + ], + "execution_count": 17, + "outputs": [ + { + "output_type": "stream", + "text": [ + "Epoch 1/10\n", + "1719/1719 [==============================] - 4s 2ms/step - loss: 1.6969 - accuracy: 0.4974 - val_loss: 0.9255 - val_accuracy: 0.7186\n", + "Epoch 2/10\n", + "1719/1719 [==============================] - 4s 2ms/step - loss: 0.8706 - accuracy: 0.7246 - val_loss: 0.7305 - val_accuracy: 0.7630\n", + "Epoch 3/10\n", + "1719/1719 [==============================] - 4s 2ms/step - loss: 0.7211 - accuracy: 0.7621 - val_loss: 0.6565 - val_accuracy: 0.7882\n", + "Epoch 4/10\n", + "1719/1719 [==============================] - 4s 2ms/step - loss: 0.6448 - accuracy: 0.7880 - val_loss: 0.6003 - val_accuracy: 0.8046\n", + "Epoch 5/10\n", + "1719/1719 [==============================] - 4s 2ms/step - loss: 0.6078 - accuracy: 0.8003 - val_loss: 0.5656 - val_accuracy: 0.8184\n", + "Epoch 6/10\n", + "1719/1719 [==============================] - 4s 2ms/step - loss: 0.5693 - accuracy: 0.8119 - val_loss: 0.5406 - val_accuracy: 0.8238\n", + "Epoch 7/10\n", + "1719/1719 [==============================] - 4s 2ms/step - loss: 0.5428 - accuracy: 0.8193 - val_loss: 0.5196 - val_accuracy: 0.8314\n", + "Epoch 8/10\n", + "1719/1719 [==============================] - 4s 2ms/step - loss: 0.5193 - accuracy: 0.8283 - val_loss: 0.5113 - val_accuracy: 0.8318\n", + "Epoch 9/10\n", + "1719/1719 [==============================] - 4s 2ms/step - loss: 0.5129 - accuracy: 0.8273 - val_loss: 0.4917 - val_accuracy: 0.8380\n", + "Epoch 10/10\n", + "1719/1719 [==============================] - 4s 2ms/step - loss: 0.4941 - accuracy: 0.8313 - val_loss: 0.4826 - val_accuracy: 0.8396\n" + ], + "name": "stdout" + } + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "Vkr7NgOmLOpF" + }, + "source": [ + "### ELU" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "6znM2r0RLOpF" + }, + "source": [ + "def elu(z, alpha=1):\n", + " return np.where(z < 0, alpha * (np.exp(z) - 1), z)" + ], + "execution_count": 18, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "WpZo3sG7LOpF", + "outputId": "8f1f0b6e-6585-4c53-d651-159fe61b98e4", + "colab": { + "base_uri": "https://localhost:8080/", + "height": 315 + } + }, + "source": [ + "plt.plot(z, elu(z), \"b-\", linewidth=2)\n", + "plt.plot([-5, 5], [0, 0], 'k-')\n", + "plt.plot([-5, 5], [-1, -1], 'k--')\n", + "plt.plot([0, 0], [-2.2, 3.2], 'k-')\n", + "plt.grid(True)\n", + "plt.title(r\"ELU activation function ($\\alpha=1$)\", fontsize=14)\n", + "plt.axis([-5, 5, -2.2, 3.2])\n", + "\n", + "save_fig(\"elu_plot\")\n", + "plt.show()" + ], + "execution_count": 19, + "outputs": [ + { + "output_type": "stream", + "text": [ + "그림 저장: elu_plot\n" + ], + "name": "stdout" + }, + { + "output_type": "display_data", + "data": { + "image/png": "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\n", + "text/plain": [ + "
" + ] + }, + "metadata": { + "tags": [], + "needs_background": "light" + } + } + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "CFEUCydjLOpF" + }, + "source": [ + "텐서플로에서 쉽게 ELU를 적용할 수 있습니다. 층을 만들 때 활성화 함수로 지정하면 됩니다:" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "zFuLenUVLOpG", + "outputId": "e03fe963-ee36-47a5-faf8-243a0968f03e", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "keras.layers.Dense(10, activation=\"elu\")" + ], + "execution_count": 20, + "outputs": [ + { + "output_type": "execute_result", + "data": { + "text/plain": [ + "" + ] + }, + "metadata": { + "tags": [] + }, + "execution_count": 20 + } + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "mkd4JNYELOpG" + }, + "source": [ + "### SELU" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "mzoAhWWhLOpG" + }, + "source": [ + "Günter Klambauer, Thomas Unterthiner, Andreas Mayr는 2017년 한 [훌륭한 논문](https://arxiv.org/pdf/1706.02515.pdf)에서 SELU 활성화 함수를 소개했습니다. 훈련하는 동안 완전 연결 층만 쌓아서 신경망을 만들고 SELU 활성화 함수와 LeCun 초기화를 사용한다면 자기 정규화됩니다. 각 층의 출력이 평균과\n", + "표준편차를 보존하는 경향이 있습니다. 이는 그레이디언트 소실과 폭주 문제를 막아줍니다. 그 결과로 SELU 활성화 함수는 이런 종류의 네트워크(특히 아주 깊은 네트워크)에서 다른 활성화 함수보다 뛰어난 성능을 종종 냅니다. 따라서 꼭 시도해 봐야 합니다. 하지만 SELU 활성화 함수의 자기 정규화 특징은 쉽게 깨집니다. ℓ1나 ℓ2 정규화, 드롭아웃, 맥스 노름, 스킵 연결이나 시퀀셜하지 않은 다른 토폴로지를 사용할 수 없습니다(즉 순환 신경망은 자기 정규화되지 않습니다). 하지만 실전에서 시퀀셜 CNN과 잘 동작합니다. 자기 정규화가 깨지면 SELU가 다른 활성화 함수보다 더 나은 성능을 내지 않을 것입니다." + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "NYXUfjHCLOpG" + }, + "source": [ + "from scipy.special import erfc\n", + "\n", + "# alpha와 scale은 평균 0과 표준 편차 1로 자기 정규화합니다\n", + "# (논문에 있는 식 14 참조):\n", + "alpha_0_1 = -np.sqrt(2 / np.pi) / (erfc(1/np.sqrt(2)) * np.exp(1/2) - 1)\n", + "scale_0_1 = (1 - erfc(1 / np.sqrt(2)) * np.sqrt(np.e)) * np.sqrt(2 * np.pi) * (2 * erfc(np.sqrt(2))*np.e**2 + np.pi*erfc(1/np.sqrt(2))**2*np.e - 2*(2+np.pi)*erfc(1/np.sqrt(2))*np.sqrt(np.e)+np.pi+2)**(-1/2)" + ], + "execution_count": 21, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "fyup_FSkLOpG" + }, + "source": [ + "def selu(z, scale=scale_0_1, alpha=alpha_0_1):\n", + " return scale * elu(z, alpha)" + ], + "execution_count": 22, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "cNpfVGctLOpH", + "outputId": "d29c5855-b472-4dac-d118-465af3c5b7ae", + "colab": { + "base_uri": "https://localhost:8080/", + "height": 315 + } + }, + "source": [ + "plt.plot(z, selu(z), \"b-\", linewidth=2)\n", + "plt.plot([-5, 5], [0, 0], 'k-')\n", + "plt.plot([-5, 5], [-1.758, -1.758], 'k--')\n", + "plt.plot([0, 0], [-2.2, 3.2], 'k-')\n", + "plt.grid(True)\n", + "plt.title(\"SELU activation function\", fontsize=14)\n", + "plt.axis([-5, 5, -2.2, 3.2])\n", + "\n", + "save_fig(\"selu_plot\")\n", + "plt.show()" + ], + "execution_count": 23, + "outputs": [ + { + "output_type": "stream", + "text": [ + "그림 저장: selu_plot\n" + ], + "name": "stdout" + }, + { + "output_type": "display_data", + "data": { + "image/png": "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\n", + "text/plain": [ + "
" + ] + }, + "metadata": { + "tags": [], + "needs_background": "light" + } + } + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "ZFaak_jnLOpH" + }, + "source": [ + "기본적으로 SELU 하이퍼파라미터(`scale`과 `alpha`)는 각 뉴런의 평균 출력이 0에 가깝고 표준 편차는 1에 가깝도록 조정됩니다(입력은 평균이 0이고 표준 편차 1로 표준화되었다고 가정합니다). 이 활성화 함수를 사용하면 1,000개의 층이 있는 심층 신경망도 모든 층에 걸쳐 거의 평균이 0이고 표준 편차를 1로 유지합니다. 이를 통해 그레이디언트 폭주와 소실 문제를 피할 수 있습니다:" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "qWawJStoLOpH", + "outputId": "38cfdca8-2c4e-4f69-ce42-366f34c49ee6", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "np.random.seed(42)\n", + "Z = np.random.normal(size=(500, 100)) # 표준화된 입력\n", + "for layer in range(1000):\n", + " W = np.random.normal(size=(100, 100), scale=np.sqrt(1 / 100)) # LeCun 초기화\n", + " Z = selu(np.dot(Z, W))\n", + " means = np.mean(Z, axis=0).mean()\n", + " stds = np.std(Z, axis=0).mean()\n", + " if layer % 100 == 0:\n", + " print(\"Layer {}: mean {:.2f}, std deviation {:.2f}\".format(layer, means, stds))" + ], + "execution_count": 24, + "outputs": [ + { + "output_type": "stream", + "text": [ + "Layer 0: mean -0.00, std deviation 1.00\n", + "Layer 100: mean 0.02, std deviation 0.96\n", + "Layer 200: mean 0.01, std deviation 0.90\n", + "Layer 300: mean -0.02, std deviation 0.92\n", + "Layer 400: mean 0.05, std deviation 0.89\n", + "Layer 500: mean 0.01, std deviation 0.93\n", + "Layer 600: mean 0.02, std deviation 0.92\n", + "Layer 700: mean -0.02, std deviation 0.90\n", + "Layer 800: mean 0.05, std deviation 0.83\n", + "Layer 900: mean 0.02, std deviation 1.00\n" + ], + "name": "stdout" + } + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "j5dAdmg4LOpI" + }, + "source": [ + "쉽게 SELU를 사용할 수 있습니다:" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "qFGwIMj2LOpI", + "outputId": "f9d69899-51cc-4c31-ecdf-429787846e20", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "keras.layers.Dense(10, activation=\"selu\",\n", + " kernel_initializer=\"lecun_normal\")" + ], + "execution_count": 25, + "outputs": [ + { + "output_type": "execute_result", + "data": { + "text/plain": [ + "" + ] + }, + "metadata": { + "tags": [] + }, + "execution_count": 25 + } + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "1D-HvWzgLOpI" + }, + "source": [ + "100개의 은닉층과 SELU 활성화 함수를 사용한 패션 MNIST를 위한 신경망을 만들어 보죠:" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "YS-krPA1LOpI" + }, + "source": [ + "np.random.seed(42)\n", + "tf.random.set_seed(42)" + ], + "execution_count": 26, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "fXLkoFMELOpI" + }, + "source": [ + "model = keras.models.Sequential()\n", + "model.add(keras.layers.Flatten(input_shape=[28, 28]))\n", + "model.add(keras.layers.Dense(300, activation=\"selu\",\n", + " kernel_initializer=\"lecun_normal\"))\n", + "for layer in range(99):\n", + " model.add(keras.layers.Dense(100, activation=\"selu\",\n", + " kernel_initializer=\"lecun_normal\"))\n", + "model.add(keras.layers.Dense(10, activation=\"softmax\"))" + ], + "execution_count": 27, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "hxrqewr6LOpI" + }, + "source": [ + "model.compile(loss=\"sparse_categorical_crossentropy\",\n", + " optimizer=keras.optimizers.SGD(lr=1e-3),\n", + " metrics=[\"accuracy\"])" + ], + "execution_count": 28, + "outputs": [] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "s2oGm-VSLOpJ" + }, + "source": [ + "이제 훈련해 보죠. 입력을 평균 0과 표준 편차 1로 바꾸어야 한다는 것을 잊지 마세요:" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "FBS7ewjeLOpJ" + }, + "source": [ + "pixel_means = X_train.mean(axis=0, keepdims=True)\n", + "pixel_stds = X_train.std(axis=0, keepdims=True)\n", + "X_train_scaled = (X_train - pixel_means) / pixel_stds\n", + "X_valid_scaled = (X_valid - pixel_means) / pixel_stds\n", + "X_test_scaled = (X_test - pixel_means) / pixel_stds" + ], + "execution_count": 29, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "DVONLiENLOpJ", + "outputId": "fa5c2881-0f6e-41e1-dbb4-b628f3e84538", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "history = model.fit(X_train_scaled, y_train, epochs=5,\n", + " validation_data=(X_valid_scaled, y_valid))" + ], + "execution_count": 30, + "outputs": [ + { + "output_type": "stream", + "text": [ + "Epoch 1/5\n", + "1719/1719 [==============================] - 17s 8ms/step - loss: 1.5863 - accuracy: 0.3956 - val_loss: 0.9179 - val_accuracy: 0.6592\n", + "Epoch 2/5\n", + "1719/1719 [==============================] - 14s 8ms/step - loss: 0.8182 - accuracy: 0.6961 - val_loss: 0.6438 - val_accuracy: 0.7696\n", + "Epoch 3/5\n", + "1719/1719 [==============================] - 14s 8ms/step - loss: 0.6735 - accuracy: 0.7521 - val_loss: 0.6299 - val_accuracy: 0.7524\n", + "Epoch 4/5\n", + "1719/1719 [==============================] - 14s 8ms/step - loss: 0.5877 - accuracy: 0.7837 - val_loss: 0.5827 - val_accuracy: 0.7988\n", + "Epoch 5/5\n", + "1719/1719 [==============================] - 14s 8ms/step - loss: 0.5487 - accuracy: 0.8016 - val_loss: 0.5130 - val_accuracy: 0.8224\n" + ], + "name": "stdout" + } + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "5k-Ud-OSLOpJ" + }, + "source": [ + "대신 ReLU 활성화 함수를 사용하면 어떤 일이 일어나는지 확인해 보죠:" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "EFFeHw5pLOpK" + }, + "source": [ + "np.random.seed(42)\n", + "tf.random.set_seed(42)" + ], + "execution_count": 31, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "EIUDPcNJLOpK" + }, + "source": [ + "model = keras.models.Sequential()\n", + "model.add(keras.layers.Flatten(input_shape=[28, 28]))\n", + "model.add(keras.layers.Dense(300, activation=\"relu\", kernel_initializer=\"he_normal\"))\n", + "for layer in range(99):\n", + " model.add(keras.layers.Dense(100, activation=\"relu\", kernel_initializer=\"he_normal\"))\n", + "model.add(keras.layers.Dense(10, activation=\"softmax\"))" + ], + "execution_count": 32, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "xZxJ2w38LOpK" + }, + "source": [ + "model.compile(loss=\"sparse_categorical_crossentropy\",\n", + " optimizer=keras.optimizers.SGD(lr=1e-3),\n", + " metrics=[\"accuracy\"])" + ], + "execution_count": 33, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "R94X2ls5LOpL", + "outputId": "3e1ad881-0171-4bd1-9ff5-7d969926d285", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "history = model.fit(X_train_scaled, y_train, epochs=5,\n", + " validation_data=(X_valid_scaled, y_valid))" + ], + "execution_count": 34, + "outputs": [ + { + "output_type": "stream", + "text": [ + "Epoch 1/5\n", + "1719/1719 [==============================] - 17s 8ms/step - loss: 2.0576 - accuracy: 0.2020 - val_loss: 1.3588 - val_accuracy: 0.3958\n", + "Epoch 2/5\n", + "1719/1719 [==============================] - 14s 8ms/step - loss: 1.2621 - accuracy: 0.4541 - val_loss: 0.9904 - val_accuracy: 0.5662\n", + "Epoch 3/5\n", + "1719/1719 [==============================] - 14s 8ms/step - loss: 0.9950 - accuracy: 0.5893 - val_loss: 0.8148 - val_accuracy: 0.6720\n", + "Epoch 4/5\n", + "1719/1719 [==============================] - 14s 8ms/step - loss: 0.8142 - accuracy: 0.6783 - val_loss: 0.7579 - val_accuracy: 0.7082\n", + "Epoch 5/5\n", + "1719/1719 [==============================] - 14s 8ms/step - loss: 0.8113 - accuracy: 0.6770 - val_loss: 0.8736 - val_accuracy: 0.6720\n" + ], + "name": "stdout" + } + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "pBPfwJ4WLOpL" + }, + "source": [ + "좋지 않군요. 그레이디언트 폭주나 소실 문제가 발생한 것입니다." + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "PVu2Q063LOpL" + }, + "source": [ + "# 배치 정규화" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "RLBH8PjPLOpL" + }, + "source": [ + "model = keras.models.Sequential([\n", + " keras.layers.Flatten(input_shape=[28, 28]),\n", + " keras.layers.BatchNormalization(),\n", + " keras.layers.Dense(300, activation=\"relu\"),\n", + " keras.layers.BatchNormalization(),\n", + " keras.layers.Dense(100, activation=\"relu\"),\n", + " keras.layers.BatchNormalization(),\n", + " keras.layers.Dense(10, activation=\"softmax\")\n", + "])" + ], + "execution_count": 35, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "d5L_RucvLOpL", + "outputId": "b4b0c837-c48a-4ac0-da2d-a946764b8d58", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "model.summary()" + ], + "execution_count": 36, + "outputs": [ + { + "output_type": "stream", + "text": [ + "Model: \"sequential_4\"\n", + "_________________________________________________________________\n", + "Layer (type) Output Shape Param # \n", + "=================================================================\n", + "flatten_4 (Flatten) (None, 784) 0 \n", + "_________________________________________________________________\n", + "batch_normalization (BatchNo (None, 784) 3136 \n", + "_________________________________________________________________\n", + "dense_212 (Dense) (None, 300) 235500 \n", + "_________________________________________________________________\n", + "batch_normalization_1 (Batch (None, 300) 1200 \n", + "_________________________________________________________________\n", + "dense_213 (Dense) (None, 100) 30100 \n", + "_________________________________________________________________\n", + "batch_normalization_2 (Batch (None, 100) 400 \n", + "_________________________________________________________________\n", + "dense_214 (Dense) (None, 10) 1010 \n", + "=================================================================\n", + "Total params: 271,346\n", + "Trainable params: 268,978\n", + "Non-trainable params: 2,368\n", + "_________________________________________________________________\n" + ], + "name": "stdout" + } + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "S4YmgIVtLOpL", + "outputId": "1830a1e3-5a00-4dab-b6c1-df4a11b7c0b4", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "bn1 = model.layers[1]\n", + "[(var.name, var.trainable) for var in bn1.variables]" + ], + "execution_count": 37, + "outputs": [ + { + "output_type": "execute_result", + "data": { + "text/plain": [ + "[('batch_normalization/gamma:0', True),\n", + " ('batch_normalization/beta:0', True),\n", + " ('batch_normalization/moving_mean:0', False),\n", + " ('batch_normalization/moving_variance:0', False)]" + ] + }, + "metadata": { + "tags": [] + }, + "execution_count": 37 + } + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "ya2gRrxPLOpM", + "outputId": "90f2b7c4-9fd7-4aa1-b182-9508cc95ddd1", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "bn1.updates" + ], + "execution_count": 38, + "outputs": [ + { + "output_type": "stream", + "text": [ + "/usr/local/lib/python3.6/dist-packages/tensorflow/python/keras/engine/base_layer.py:1402: UserWarning: `layer.updates` will be removed in a future version. This property should not be used in TensorFlow 2.0, as `updates` are applied automatically.\n", + " warnings.warn('`layer.updates` will be removed in a future version. '\n" + ], + "name": "stderr" + }, + { + "output_type": "execute_result", + "data": { + "text/plain": [ + "[]" + ] + }, + "metadata": { + "tags": [] + }, + "execution_count": 38 + } + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "TqJdxp-vLOpN" + }, + "source": [ + "model.compile(loss=\"sparse_categorical_crossentropy\",\n", + " optimizer=keras.optimizers.SGD(lr=1e-3),\n", + " metrics=[\"accuracy\"])" + ], + "execution_count": 39, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "yNutyZRELOpN", + "outputId": "ef471abc-7bc4-44be-cc40-5068a8b7b2c9", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "history = model.fit(X_train, y_train, epochs=10,\n", + " validation_data=(X_valid, y_valid))" + ], + "execution_count": 40, + "outputs": [ + { + "output_type": "stream", + "text": [ + "Epoch 1/10\n", + "1719/1719 [==============================] - 6s 3ms/step - loss: 1.2287 - accuracy: 0.5994 - val_loss: 0.5525 - val_accuracy: 0.8230\n", + "Epoch 2/10\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.5995 - accuracy: 0.7958 - val_loss: 0.4725 - val_accuracy: 0.8470\n", + "Epoch 3/10\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.5312 - accuracy: 0.8172 - val_loss: 0.4375 - val_accuracy: 0.8550\n", + "Epoch 4/10\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.4885 - accuracy: 0.8294 - val_loss: 0.4152 - val_accuracy: 0.8604\n", + "Epoch 5/10\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.4718 - accuracy: 0.8349 - val_loss: 0.3997 - val_accuracy: 0.8644\n", + "Epoch 6/10\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.4420 - accuracy: 0.8461 - val_loss: 0.3867 - val_accuracy: 0.8690\n", + "Epoch 7/10\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.4286 - accuracy: 0.8495 - val_loss: 0.3763 - val_accuracy: 0.8702\n", + "Epoch 8/10\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.4087 - accuracy: 0.8551 - val_loss: 0.3713 - val_accuracy: 0.8740\n", + "Epoch 9/10\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.4080 - accuracy: 0.8564 - val_loss: 0.3631 - val_accuracy: 0.8750\n", + "Epoch 10/10\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.3903 - accuracy: 0.8616 - val_loss: 0.3571 - val_accuracy: 0.8754\n" + ], + "name": "stdout" + } + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "IeYDw1BwLOpN" + }, + "source": [ + "이따금 활성화 함수전에 BN을 적용해도 잘 동작합니다(여기에는 논란의 여지가 있습니다). 또한 `BatchNormalization` 층 이전의 층은 편향을 위한 항이 필요 없습니다. `BatchNormalization` 층이 이를 무효화하기 때문입니다. 따라서 필요 없는 파라미터이므로 `use_bias=False`를 지정하여 층을 만들 수 있습니다:" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "-QU5Mn0fLOpN" + }, + "source": [ + "model = keras.models.Sequential([\n", + " keras.layers.Flatten(input_shape=[28, 28]),\n", + " keras.layers.BatchNormalization(),\n", + " keras.layers.Dense(300, use_bias=False),\n", + " keras.layers.BatchNormalization(),\n", + " keras.layers.Activation(\"relu\"),\n", + " keras.layers.Dense(100, use_bias=False),\n", + " keras.layers.BatchNormalization(),\n", + " keras.layers.Activation(\"relu\"),\n", + " keras.layers.Dense(10, activation=\"softmax\")\n", + "])" + ], + "execution_count": 41, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "1Y-7t8DsLOpN" + }, + "source": [ + "model.compile(loss=\"sparse_categorical_crossentropy\",\n", + " optimizer=keras.optimizers.SGD(lr=1e-3),\n", + " metrics=[\"accuracy\"])" + ], + "execution_count": 42, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "fhatQCD6LOpN", + "outputId": "804ad977-910d-4ed8-a241-8e914ed0ac0c", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "history = model.fit(X_train, y_train, epochs=10,\n", + " validation_data=(X_valid, y_valid))" + ], + "execution_count": 43, + "outputs": [ + { + "output_type": "stream", + "text": [ + "Epoch 1/10\n", + "1719/1719 [==============================] - 6s 3ms/step - loss: 1.3677 - accuracy: 0.5605 - val_loss: 0.6767 - val_accuracy: 0.7812\n", + "Epoch 2/10\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.7136 - accuracy: 0.7702 - val_loss: 0.5566 - val_accuracy: 0.8182\n", + "Epoch 3/10\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.6123 - accuracy: 0.7990 - val_loss: 0.5007 - val_accuracy: 0.8362\n", + "Epoch 4/10\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.5547 - accuracy: 0.8148 - val_loss: 0.4666 - val_accuracy: 0.8448\n", + "Epoch 5/10\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.5254 - accuracy: 0.8232 - val_loss: 0.4433 - val_accuracy: 0.8532\n", + "Epoch 6/10\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.4947 - accuracy: 0.8326 - val_loss: 0.4262 - val_accuracy: 0.8550\n", + "Epoch 7/10\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.4736 - accuracy: 0.8388 - val_loss: 0.4130 - val_accuracy: 0.8564\n", + "Epoch 8/10\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.4550 - accuracy: 0.8443 - val_loss: 0.4034 - val_accuracy: 0.8612\n", + "Epoch 9/10\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.4495 - accuracy: 0.8439 - val_loss: 0.3943 - val_accuracy: 0.8638\n", + "Epoch 10/10\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.4333 - accuracy: 0.8495 - val_loss: 0.3874 - val_accuracy: 0.8660\n" + ], + "name": "stdout" + } + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "CWhcWzXCLOpO" + }, + "source": [ + "## 그레이디언트 클리핑" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "L73OupHpLOpO" + }, + "source": [ + "모든 케라스 옵티마이저는 `clipnorm`이나 `clipvalue` 매개변수를 지원합니다:" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "PN-aVXL-LOpO" + }, + "source": [ + "optimizer = keras.optimizers.SGD(clipvalue=1.0)" + ], + "execution_count": 44, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "g9Wj01r7LOpO" + }, + "source": [ + "optimizer = keras.optimizers.SGD(clipnorm=1.0)" + ], + "execution_count": 45, + "outputs": [] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "rxEJ6pNyLOpO" + }, + "source": [ + "## 사전 훈련된 층 재사용하기" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "u-g_PGcmLOpO" + }, + "source": [ + "### 케라스 모델 재사용하기" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "rdiVjfkOLOpO" + }, + "source": [ + "패션 MNIST 훈련 세트를 두 개로 나누어 보죠:\n", + "* `X_train_A`: 샌달과 셔츠(클래스 5와 6)을 제외한 모든 이미지\n", + "* `X_train_B`: 샌달과 셔츠 이미지 중 처음 200개만 가진 작은 훈련 세트\n", + "\n", + "검증 세트와 테스트 세트도 이렇게 나눕니다. 하지만 이미지 개수는 제한하지 않습니다.\n", + "\n", + "A 세트(8개의 클래스를 가진 분류 문제)에서 모델을 훈련하고 이를 재사용하여 B 세트(이진 분류)를 해결해 보겠습니다. A 작업에서 B 작업으로 약간의 지식이 전달되기를 기대합니다. 왜냐하면 A 세트의 클래스(스니커즈, 앵클 부츠, 코트, 티셔츠 등)가 B 세트에 있는 클래스(샌달과 셔츠)와 조금 비슷하기 때문입니다. 하지만 `Dense` 층을 사용하기 때문에 동일한 위치에 나타난 패턴만 재사용할 수 있습니다(반대로 합성곱 층은 훨씬 많은 정보를 전송합니다. 학습한 패턴을 이미지의 어느 위치에서나 감지할 수 있기 때문입니다. CNN 장에서 자세히 알아 보겠습니다)." + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "QKHsibRsLOpO" + }, + "source": [ + "def split_dataset(X, y):\n", + " y_5_or_6 = (y == 5) | (y == 6) # sandals or shirts\n", + " y_A = y[~y_5_or_6]\n", + " y_A[y_A > 6] -= 2 # class indices 7, 8, 9 should be moved to 5, 6, 7\n", + " y_B = (y[y_5_or_6] == 6).astype(np.float32) # binary classification task: is it a shirt (class 6)?\n", + " return ((X[~y_5_or_6], y_A),\n", + " (X[y_5_or_6], y_B))\n", + "\n", + "(X_train_A, y_train_A), (X_train_B, y_train_B) = split_dataset(X_train, y_train)\n", + "(X_valid_A, y_valid_A), (X_valid_B, y_valid_B) = split_dataset(X_valid, y_valid)\n", + "(X_test_A, y_test_A), (X_test_B, y_test_B) = split_dataset(X_test, y_test)\n", + "X_train_B = X_train_B[:200]\n", + "y_train_B = y_train_B[:200]" + ], + "execution_count": 46, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "2s04GUdELOpO", + "outputId": "e7e01e53-fa88-485a-b086-c2269d7e2c61", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "X_train_A.shape" + ], + "execution_count": 47, + "outputs": [ + { + "output_type": "execute_result", + "data": { + "text/plain": [ + "(43986, 28, 28)" + ] + }, + "metadata": { + "tags": [] + }, + "execution_count": 47 + } + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "kcEEehIGLOpP", + "outputId": "65d1002c-3fba-49fb-cd77-bea76dddc1c6", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "X_train_B.shape" + ], + "execution_count": 48, + "outputs": [ + { + "output_type": "execute_result", + "data": { + "text/plain": [ + "(200, 28, 28)" + ] + }, + "metadata": { + "tags": [] + }, + "execution_count": 48 + } + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "smwubTT0LOpP", + "outputId": "c801b7f2-5397-483e-e840-d3141878d4ae", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "y_train_A[:30]" + ], + "execution_count": 49, + "outputs": [ + { + "output_type": "execute_result", + "data": { + "text/plain": [ + "array([4, 0, 5, 7, 7, 7, 4, 4, 3, 4, 0, 1, 6, 3, 4, 3, 2, 6, 5, 3, 4, 5,\n", + " 1, 3, 4, 2, 0, 6, 7, 1], dtype=uint8)" + ] + }, + "metadata": { + "tags": [] + }, + "execution_count": 49 + } + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "hNJqisASLOpP", + "outputId": "9551ff38-fe97-4a93-94d2-f84b19ab4efb", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "y_train_B[:30]" + ], + "execution_count": 50, + "outputs": [ + { + "output_type": "execute_result", + "data": { + "text/plain": [ + "array([1., 1., 0., 0., 0., 0., 1., 1., 1., 0., 0., 1., 1., 0., 0., 0., 0.,\n", + " 0., 0., 1., 1., 0., 0., 1., 1., 0., 1., 1., 1., 1.], dtype=float32)" + ] + }, + "metadata": { + "tags": [] + }, + "execution_count": 50 + } + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "2qkd71BzLOpP" + }, + "source": [ + "tf.random.set_seed(42)\n", + "np.random.seed(42)" + ], + "execution_count": 51, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "_dou1YzGLOpP" + }, + "source": [ + "model_A = keras.models.Sequential()\n", + "model_A.add(keras.layers.Flatten(input_shape=[28, 28]))\n", + "for n_hidden in (300, 100, 50, 50, 50):\n", + " model_A.add(keras.layers.Dense(n_hidden, activation=\"selu\"))\n", + "model_A.add(keras.layers.Dense(8, activation=\"softmax\"))" + ], + "execution_count": 52, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "C6TIeUe-LOpQ" + }, + "source": [ + "model_A.compile(loss=\"sparse_categorical_crossentropy\",\n", + " optimizer=keras.optimizers.SGD(lr=1e-3),\n", + " metrics=[\"accuracy\"])" + ], + "execution_count": 53, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "4xR6ixj8LOpQ", + "outputId": "ec8f3003-4b1d-4c68-f8fc-b154cc26a402", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "history = model_A.fit(X_train_A, y_train_A, epochs=20,\n", + " validation_data=(X_valid_A, y_valid_A))" + ], + "execution_count": 54, + "outputs": [ + { + "output_type": "stream", + "text": [ + "Epoch 1/20\n", + "1375/1375 [==============================] - 4s 2ms/step - loss: 0.9248 - accuracy: 0.6994 - val_loss: 0.3894 - val_accuracy: 0.8665\n", + "Epoch 2/20\n", + "1375/1375 [==============================] - 3s 2ms/step - loss: 0.3651 - accuracy: 0.8748 - val_loss: 0.3288 - val_accuracy: 0.8829\n", + "Epoch 3/20\n", + "1375/1375 [==============================] - 3s 2ms/step - loss: 0.3182 - accuracy: 0.8894 - val_loss: 0.3012 - val_accuracy: 0.8996\n", + "Epoch 4/20\n", + "1375/1375 [==============================] - 3s 2ms/step - loss: 0.3049 - accuracy: 0.8955 - val_loss: 0.2895 - val_accuracy: 0.9013\n", + "Epoch 5/20\n", + "1375/1375 [==============================] - 3s 2ms/step - loss: 0.2804 - accuracy: 0.9027 - val_loss: 0.2774 - val_accuracy: 0.9063\n", + "Epoch 6/20\n", + "1375/1375 [==============================] - 3s 2ms/step - loss: 0.2702 - accuracy: 0.9079 - val_loss: 0.2734 - val_accuracy: 0.9066\n", + "Epoch 7/20\n", + "1375/1375 [==============================] - 3s 2ms/step - loss: 0.2626 - accuracy: 0.9088 - val_loss: 0.2720 - val_accuracy: 0.9083\n", + "Epoch 8/20\n", + "1375/1375 [==============================] - 3s 2ms/step - loss: 0.2609 - accuracy: 0.9119 - val_loss: 0.2591 - val_accuracy: 0.9138\n", + "Epoch 9/20\n", + "1375/1375 [==============================] - 3s 2ms/step - loss: 0.2558 - accuracy: 0.9110 - val_loss: 0.2563 - val_accuracy: 0.9143\n", + "Epoch 10/20\n", + "1375/1375 [==============================] - 3s 2ms/step - loss: 0.2511 - accuracy: 0.9139 - val_loss: 0.2543 - val_accuracy: 0.9158\n", + "Epoch 11/20\n", + "1375/1375 [==============================] - 3s 2ms/step - loss: 0.2431 - accuracy: 0.9173 - val_loss: 0.2496 - val_accuracy: 0.9153\n", + "Epoch 12/20\n", + "1375/1375 [==============================] - 3s 2ms/step - loss: 0.2422 - accuracy: 0.9170 - val_loss: 0.2512 - val_accuracy: 0.9126\n", + "Epoch 13/20\n", + "1375/1375 [==============================] - 3s 2ms/step - loss: 0.2360 - accuracy: 0.9179 - val_loss: 0.2446 - val_accuracy: 0.9158\n", + "Epoch 14/20\n", + "1375/1375 [==============================] - 3s 2ms/step - loss: 0.2267 - accuracy: 0.9230 - val_loss: 0.2416 - val_accuracy: 0.9175\n", + "Epoch 15/20\n", + "1375/1375 [==============================] - 3s 2ms/step - loss: 0.2225 - accuracy: 0.9241 - val_loss: 0.2450 - val_accuracy: 0.9188\n", + "Epoch 16/20\n", + "1375/1375 [==============================] - 3s 2ms/step - loss: 0.2262 - accuracy: 0.9214 - val_loss: 0.2386 - val_accuracy: 0.9193\n", + "Epoch 17/20\n", + "1375/1375 [==============================] - 3s 2ms/step - loss: 0.2191 - accuracy: 0.9252 - val_loss: 0.2405 - val_accuracy: 0.9178\n", + "Epoch 18/20\n", + "1375/1375 [==============================] - 3s 2ms/step - loss: 0.2172 - accuracy: 0.9253 - val_loss: 0.2426 - val_accuracy: 0.9158\n", + "Epoch 19/20\n", + "1375/1375 [==============================] - 3s 2ms/step - loss: 0.2181 - accuracy: 0.9246 - val_loss: 0.2331 - val_accuracy: 0.9213\n", + "Epoch 20/20\n", + "1375/1375 [==============================] - 3s 2ms/step - loss: 0.2113 - accuracy: 0.9271 - val_loss: 0.2332 - val_accuracy: 0.9203\n" + ], + "name": "stdout" + } + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "8UMBGi07LOpQ" + }, + "source": [ + "model_A.save(\"my_model_A.h5\")" + ], + "execution_count": 55, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "qb4XQRc2LOpQ" + }, + "source": [ + "model_B = keras.models.Sequential()\n", + "model_B.add(keras.layers.Flatten(input_shape=[28, 28]))\n", + "for n_hidden in (300, 100, 50, 50, 50):\n", + " model_B.add(keras.layers.Dense(n_hidden, activation=\"selu\"))\n", + "model_B.add(keras.layers.Dense(1, activation=\"sigmoid\"))" + ], + "execution_count": 56, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "bTfx9aSGLOpQ" + }, + "source": [ + "model_B.compile(loss=\"binary_crossentropy\",\n", + " optimizer=keras.optimizers.SGD(lr=1e-3),\n", + " metrics=[\"accuracy\"])" + ], + "execution_count": 57, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "dd6o7rDmLOpQ", + "outputId": "b1022791-e04c-438b-db1d-7da924db831c", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "history = model_B.fit(X_train_B, y_train_B, epochs=20,\n", + " validation_data=(X_valid_B, y_valid_B))" + ], + "execution_count": 58, + "outputs": [ + { + "output_type": "stream", + "text": [ + "Epoch 1/20\n", + "7/7 [==============================] - 1s 35ms/step - loss: 1.0360 - accuracy: 0.4975 - val_loss: 0.6314 - val_accuracy: 0.6004\n", + "Epoch 2/20\n", + "7/7 [==============================] - 0s 16ms/step - loss: 0.5883 - accuracy: 0.6971 - val_loss: 0.4784 - val_accuracy: 0.8529\n", + "Epoch 3/20\n", + "7/7 [==============================] - 0s 14ms/step - loss: 0.4380 - accuracy: 0.8854 - val_loss: 0.4102 - val_accuracy: 0.8945\n", + "Epoch 4/20\n", + "7/7 [==============================] - 0s 13ms/step - loss: 0.4021 - accuracy: 0.8712 - val_loss: 0.3647 - val_accuracy: 0.9178\n", + "Epoch 5/20\n", + "7/7 [==============================] - 0s 14ms/step - loss: 0.3361 - accuracy: 0.9348 - val_loss: 0.3300 - val_accuracy: 0.9320\n", + "Epoch 6/20\n", + "7/7 [==============================] - 0s 14ms/step - loss: 0.3113 - accuracy: 0.9233 - val_loss: 0.3019 - val_accuracy: 0.9402\n", + "Epoch 7/20\n", + "7/7 [==============================] - 0s 14ms/step - loss: 0.2817 - accuracy: 0.9299 - val_loss: 0.2804 - val_accuracy: 0.9422\n", + "Epoch 8/20\n", + "7/7 [==============================] - 0s 14ms/step - loss: 0.2632 - accuracy: 0.9379 - val_loss: 0.2606 - val_accuracy: 0.9473\n", + "Epoch 9/20\n", + "7/7 [==============================] - 0s 16ms/step - loss: 0.2373 - accuracy: 0.9481 - val_loss: 0.2428 - val_accuracy: 0.9523\n", + "Epoch 10/20\n", + "7/7 [==============================] - 0s 14ms/step - loss: 0.2229 - accuracy: 0.9657 - val_loss: 0.2281 - val_accuracy: 0.9544\n", + "Epoch 11/20\n", + "7/7 [==============================] - 0s 14ms/step - loss: 0.2155 - accuracy: 0.9590 - val_loss: 0.2150 - val_accuracy: 0.9584\n", + "Epoch 12/20\n", + "7/7 [==============================] - 0s 14ms/step - loss: 0.1834 - accuracy: 0.9738 - val_loss: 0.2036 - val_accuracy: 0.9584\n", + "Epoch 13/20\n", + "7/7 [==============================] - 0s 14ms/step - loss: 0.1671 - accuracy: 0.9828 - val_loss: 0.1931 - val_accuracy: 0.9615\n", + "Epoch 14/20\n", + "7/7 [==============================] - 0s 14ms/step - loss: 0.1527 - accuracy: 0.9915 - val_loss: 0.1838 - val_accuracy: 0.9635\n", + "Epoch 15/20\n", + "7/7 [==============================] - 0s 14ms/step - loss: 0.1595 - accuracy: 0.9904 - val_loss: 0.1746 - val_accuracy: 0.9686\n", + "Epoch 16/20\n", + "7/7 [==============================] - 0s 14ms/step - loss: 0.1473 - accuracy: 0.9937 - val_loss: 0.1674 - val_accuracy: 0.9686\n", + "Epoch 17/20\n", + "7/7 [==============================] - 0s 14ms/step - loss: 0.1412 - accuracy: 0.9944 - val_loss: 0.1604 - val_accuracy: 0.9706\n", + "Epoch 18/20\n", + "7/7 [==============================] - 0s 14ms/step - loss: 0.1242 - accuracy: 0.9931 - val_loss: 0.1539 - val_accuracy: 0.9706\n", + "Epoch 19/20\n", + "7/7 [==============================] - 0s 14ms/step - loss: 0.1224 - accuracy: 0.9931 - val_loss: 0.1482 - val_accuracy: 0.9716\n", + "Epoch 20/20\n", + "7/7 [==============================] - 0s 15ms/step - loss: 0.1096 - accuracy: 0.9912 - val_loss: 0.1431 - val_accuracy: 0.9716\n" + ], + "name": "stdout" + } + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "Bh48HMITLOpQ", + "outputId": "8996eafe-9ad5-4013-8f80-733cd9f94a78", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "model.summary()" + ], + "execution_count": 59, + "outputs": [ + { + "output_type": "stream", + "text": [ + "Model: \"sequential_5\"\n", + "_________________________________________________________________\n", + "Layer (type) Output Shape Param # \n", + "=================================================================\n", + "flatten_5 (Flatten) (None, 784) 0 \n", + "_________________________________________________________________\n", + "batch_normalization_3 (Batch (None, 784) 3136 \n", + "_________________________________________________________________\n", + "dense_215 (Dense) (None, 300) 235200 \n", + "_________________________________________________________________\n", + "batch_normalization_4 (Batch (None, 300) 1200 \n", + "_________________________________________________________________\n", + "activation (Activation) (None, 300) 0 \n", + "_________________________________________________________________\n", + "dense_216 (Dense) (None, 100) 30000 \n", + "_________________________________________________________________\n", + "batch_normalization_5 (Batch (None, 100) 400 \n", + "_________________________________________________________________\n", + "activation_1 (Activation) (None, 100) 0 \n", + "_________________________________________________________________\n", + "dense_217 (Dense) (None, 10) 1010 \n", + "=================================================================\n", + "Total params: 270,946\n", + "Trainable params: 268,578\n", + "Non-trainable params: 2,368\n", + "_________________________________________________________________\n" + ], + "name": "stdout" + } + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "xpurR4NDLOpQ" + }, + "source": [ + "model_A = keras.models.load_model(\"my_model_A.h5\")\n", + "model_B_on_A = keras.models.Sequential(model_A.layers[:-1])\n", + "model_B_on_A.add(keras.layers.Dense(1, activation=\"sigmoid\"))" + ], + "execution_count": 60, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "XPRZk6DqLOpR" + }, + "source": [ + "model_A_clone = keras.models.clone_model(model_A)\n", + "model_A_clone.set_weights(model_A.get_weights())" + ], + "execution_count": 61, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "kNxjby1PLOpR" + }, + "source": [ + "for layer in model_B_on_A.layers[:-1]:\n", + " layer.trainable = False\n", + "\n", + "model_B_on_A.compile(loss=\"binary_crossentropy\",\n", + " optimizer=keras.optimizers.SGD(lr=1e-3),\n", + " metrics=[\"accuracy\"])" + ], + "execution_count": 62, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "yvlj3S1jLOpR", + "outputId": "0a930150-7ddf-4eff-bd36-ac3bd55bb243", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "history = model_B_on_A.fit(X_train_B, y_train_B, epochs=4,\n", + " validation_data=(X_valid_B, y_valid_B))\n", + "\n", + "for layer in model_B_on_A.layers[:-1]:\n", + " layer.trainable = True\n", + "\n", + "model_B_on_A.compile(loss=\"binary_crossentropy\",\n", + " optimizer=keras.optimizers.SGD(lr=1e-3),\n", + " metrics=[\"accuracy\"])\n", + "history = model_B_on_A.fit(X_train_B, y_train_B, epochs=16,\n", + " validation_data=(X_valid_B, y_valid_B))" + ], + "execution_count": 63, + "outputs": [ + { + "output_type": "stream", + "text": [ + "Epoch 1/4\n", + "7/7 [==============================] - 1s 56ms/step - loss: 0.6098 - accuracy: 0.6233 - val_loss: 0.5795 - val_accuracy: 0.6389\n", + "Epoch 2/4\n", + "7/7 [==============================] - 0s 16ms/step - loss: 0.5498 - accuracy: 0.6707 - val_loss: 0.5424 - val_accuracy: 0.6815\n", + "Epoch 3/4\n", + "7/7 [==============================] - 0s 14ms/step - loss: 0.4847 - accuracy: 0.7509 - val_loss: 0.5107 - val_accuracy: 0.7110\n", + "Epoch 4/4\n", + "7/7 [==============================] - 0s 14ms/step - loss: 0.4849 - accuracy: 0.7405 - val_loss: 0.4823 - val_accuracy: 0.7343\n", + "Epoch 1/16\n", + "7/7 [==============================] - 1s 36ms/step - loss: 0.4345 - accuracy: 0.7823 - val_loss: 0.3440 - val_accuracy: 0.8671\n", + "Epoch 2/16\n", + "7/7 [==============================] - 0s 14ms/step - loss: 0.2951 - accuracy: 0.9143 - val_loss: 0.2593 - val_accuracy: 0.9290\n", + "Epoch 3/16\n", + "7/7 [==============================] - 0s 14ms/step - loss: 0.2023 - accuracy: 0.9777 - val_loss: 0.2104 - val_accuracy: 0.9554\n", + "Epoch 4/16\n", + "7/7 [==============================] - 0s 15ms/step - loss: 0.1745 - accuracy: 0.9789 - val_loss: 0.1786 - val_accuracy: 0.9696\n", + "Epoch 5/16\n", + "7/7 [==============================] - 0s 14ms/step - loss: 0.1341 - accuracy: 0.9809 - val_loss: 0.1559 - val_accuracy: 0.9767\n", + "Epoch 6/16\n", + "7/7 [==============================] - 0s 14ms/step - loss: 0.1169 - accuracy: 0.9973 - val_loss: 0.1391 - val_accuracy: 0.9807\n", + "Epoch 7/16\n", + "7/7 [==============================] - 0s 14ms/step - loss: 0.1134 - accuracy: 0.9931 - val_loss: 0.1266 - val_accuracy: 0.9838\n", + "Epoch 8/16\n", + "7/7 [==============================] - 0s 14ms/step - loss: 0.0998 - accuracy: 0.9931 - val_loss: 0.1163 - val_accuracy: 0.9858\n", + "Epoch 9/16\n", + "7/7 [==============================] - 0s 14ms/step - loss: 0.0832 - accuracy: 1.0000 - val_loss: 0.1066 - val_accuracy: 0.9888\n", + "Epoch 10/16\n", + "7/7 [==============================] - 0s 14ms/step - loss: 0.0774 - accuracy: 1.0000 - val_loss: 0.1000 - val_accuracy: 0.9899\n", + "Epoch 11/16\n", + "7/7 [==============================] - 0s 15ms/step - loss: 0.0689 - accuracy: 1.0000 - val_loss: 0.0940 - val_accuracy: 0.9899\n", + "Epoch 12/16\n", + "7/7 [==============================] - 0s 14ms/step - loss: 0.0719 - accuracy: 1.0000 - val_loss: 0.0889 - val_accuracy: 0.9899\n", + "Epoch 13/16\n", + "7/7 [==============================] - 0s 14ms/step - loss: 0.0566 - accuracy: 1.0000 - val_loss: 0.0840 - val_accuracy: 0.9899\n", + "Epoch 14/16\n", + "7/7 [==============================] - 0s 14ms/step - loss: 0.0494 - accuracy: 1.0000 - val_loss: 0.0803 - val_accuracy: 0.9899\n", + "Epoch 15/16\n", + "7/7 [==============================] - 0s 15ms/step - loss: 0.0545 - accuracy: 1.0000 - val_loss: 0.0770 - val_accuracy: 0.9899\n", + "Epoch 16/16\n", + "7/7 [==============================] - 0s 14ms/step - loss: 0.0472 - accuracy: 1.0000 - val_loss: 0.0740 - val_accuracy: 0.9899\n" + ], + "name": "stdout" + } + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "ceskFW97LOpR" + }, + "source": [ + "마지막 점수는 어떤가요?" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "fmdxUJ9FLOpR", + "outputId": "6d812b28-28b8-46e3-eebc-84aa1f6bbb7f", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "model_B.evaluate(X_test_B, y_test_B)" + ], + "execution_count": 64, + "outputs": [ + { + "output_type": "stream", + "text": [ + "63/63 [==============================] - 0s 2ms/step - loss: 0.1408 - accuracy: 0.9705\n" + ], + "name": "stdout" + }, + { + "output_type": "execute_result", + "data": { + "text/plain": [ + "[0.1408407986164093, 0.9704999923706055]" + ] + }, + "metadata": { + "tags": [] + }, + "execution_count": 64 + } + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "804aagCRLOpR", + "outputId": "bfa379e0-7c67-45bd-ff9c-177f0ceb1645", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "model_B_on_A.evaluate(X_test_B, y_test_B)" + ], + "execution_count": 65, + "outputs": [ + { + "output_type": "stream", + "text": [ + "63/63 [==============================] - 0s 2ms/step - loss: 0.0682 - accuracy: 0.9935\n" + ], + "name": "stdout" + }, + { + "output_type": "execute_result", + "data": { + "text/plain": [ + "[0.06819812208414078, 0.9934999942779541]" + ] + }, + "metadata": { + "tags": [] + }, + "execution_count": 65 + } + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "eoEbJs2sLOpR" + }, + "source": [ + "훌륭하네요! 꽤 많은 정보를 전달했습니다: 오차율이 4배나 줄었네요!" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "2Oge57HRLOpS", + "outputId": "213f9022-8f47-46c8-bb2f-e3d375e8b215", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "(100 - 96.95) / (100 - 99.25)" + ], + "execution_count": 66, + "outputs": [ + { + "output_type": "execute_result", + "data": { + "text/plain": [ + "4.066666666666663" + ] + }, + "metadata": { + "tags": [] + }, + "execution_count": 66 + } + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "t1Y9NJBjLOpS" + }, + "source": [ + "# 고속 옵티마이저" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "Ie9qzCm6LOpS" + }, + "source": [ + "## 모멘텀 옵티마이저" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "-T_eBS6iLOpS" + }, + "source": [ + "optimizer = keras.optimizers.SGD(lr=0.001, momentum=0.9)" + ], + "execution_count": 67, + "outputs": [] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "gvoYzuDRLOpT" + }, + "source": [ + "## 네스테로프 가속 경사" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "DbUw8WL2LOpT" + }, + "source": [ + "optimizer = keras.optimizers.SGD(lr=0.001, momentum=0.9, nesterov=True)" + ], + "execution_count": 68, + "outputs": [] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "MzY29JPMLOpT" + }, + "source": [ + "## AdaGrad" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "dR9NgMEoLOpT" + }, + "source": [ + "optimizer = keras.optimizers.Adagrad(lr=0.001)" + ], + "execution_count": 69, + "outputs": [] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "RdzjwRZOLOpT" + }, + "source": [ + "## RMSProp" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "TkjnInhCLOpT" + }, + "source": [ + "optimizer = keras.optimizers.RMSprop(lr=0.001, rho=0.9)" + ], + "execution_count": 70, + "outputs": [] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "Aqd3XjSlLOpT" + }, + "source": [ + "## Adam 옵티마이저" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "ibdJe_cILOpU" + }, + "source": [ + "optimizer = keras.optimizers.Adam(lr=0.001, beta_1=0.9, beta_2=0.999)" + ], + "execution_count": 71, + "outputs": [] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "vOuhwhjQLOpU" + }, + "source": [ + "## Adamax 옵티마이저" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "_pwlFvReLOpU" + }, + "source": [ + "optimizer = keras.optimizers.Adamax(lr=0.001, beta_1=0.9, beta_2=0.999)" + ], + "execution_count": 72, + "outputs": [] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "-UL-CtdMLOpU" + }, + "source": [ + "## Nadam 옵티마이저" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "L2jGWQ8vLOpU" + }, + "source": [ + "optimizer = keras.optimizers.Nadam(lr=0.001, beta_1=0.9, beta_2=0.999)" + ], + "execution_count": 73, + "outputs": [] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "Ow2z1pbiLOpU" + }, + "source": [ + "## 학습률 스케줄링" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "DziEJzDxLOpU" + }, + "source": [ + "### 거듭제곱 스케줄링" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "_QKqYIBSLOpU" + }, + "source": [ + "```lr = lr0 / (1 + steps / s)**c```\n", + "* 케라스는 `c=1`과 `s = 1 / decay`을 사용합니다" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "mEhuxik_LOpU" + }, + "source": [ + "optimizer = keras.optimizers.SGD(lr=0.01, decay=1e-4)" + ], + "execution_count": 74, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "3ZRa8D0PLOpV" + }, + "source": [ + "model = keras.models.Sequential([\n", + " keras.layers.Flatten(input_shape=[28, 28]),\n", + " keras.layers.Dense(300, activation=\"selu\", kernel_initializer=\"lecun_normal\"),\n", + " keras.layers.Dense(100, activation=\"selu\", kernel_initializer=\"lecun_normal\"),\n", + " keras.layers.Dense(10, activation=\"softmax\")\n", + "])\n", + "model.compile(loss=\"sparse_categorical_crossentropy\", optimizer=optimizer, metrics=[\"accuracy\"])" + ], + "execution_count": 75, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "VpPb6h-cLOpV", + "outputId": "4ef44500-2026-4c95-a8d1-e5ad2bcc4757", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "n_epochs = 25\n", + "history = model.fit(X_train_scaled, y_train, epochs=n_epochs,\n", + " validation_data=(X_valid_scaled, y_valid))" + ], + "execution_count": 76, + "outputs": [ + { + "output_type": "stream", + "text": [ + "Epoch 1/25\n", + "1719/1719 [==============================] - 4s 2ms/step - loss: 0.5980 - accuracy: 0.7933 - val_loss: 0.4029 - val_accuracy: 0.8604\n", + "Epoch 2/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.3830 - accuracy: 0.8637 - val_loss: 0.3716 - val_accuracy: 0.8720\n", + "Epoch 3/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.3492 - accuracy: 0.8773 - val_loss: 0.3749 - val_accuracy: 0.8732\n", + "Epoch 4/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.3276 - accuracy: 0.8812 - val_loss: 0.3499 - val_accuracy: 0.8800\n", + "Epoch 5/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.3172 - accuracy: 0.8858 - val_loss: 0.3448 - val_accuracy: 0.8788\n", + "Epoch 6/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2923 - accuracy: 0.8938 - val_loss: 0.3410 - val_accuracy: 0.8832\n", + "Epoch 7/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2870 - accuracy: 0.8972 - val_loss: 0.3354 - val_accuracy: 0.8870\n", + "Epoch 8/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2721 - accuracy: 0.9032 - val_loss: 0.3405 - val_accuracy: 0.8836\n", + "Epoch 9/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2729 - accuracy: 0.9004 - val_loss: 0.3288 - val_accuracy: 0.8888\n", + "Epoch 10/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2584 - accuracy: 0.9071 - val_loss: 0.3258 - val_accuracy: 0.8884\n", + "Epoch 11/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2529 - accuracy: 0.9096 - val_loss: 0.3265 - val_accuracy: 0.8880\n", + "Epoch 12/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2484 - accuracy: 0.9100 - val_loss: 0.3331 - val_accuracy: 0.8826\n", + "Epoch 13/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2420 - accuracy: 0.9145 - val_loss: 0.3252 - val_accuracy: 0.8892\n", + "Epoch 14/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2371 - accuracy: 0.9147 - val_loss: 0.3286 - val_accuracy: 0.8898\n", + "Epoch 15/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2363 - accuracy: 0.9158 - val_loss: 0.3240 - val_accuracy: 0.8876\n", + "Epoch 16/25\n", + "1719/1719 [==============================] - 4s 2ms/step - loss: 0.2311 - accuracy: 0.9176 - val_loss: 0.3201 - val_accuracy: 0.8900\n", + "Epoch 17/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2234 - accuracy: 0.9207 - val_loss: 0.3233 - val_accuracy: 0.8906\n", + "Epoch 18/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2248 - accuracy: 0.9194 - val_loss: 0.3187 - val_accuracy: 0.8938\n", + "Epoch 19/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2236 - accuracy: 0.9213 - val_loss: 0.3227 - val_accuracy: 0.8904\n", + "Epoch 20/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2229 - accuracy: 0.9221 - val_loss: 0.3208 - val_accuracy: 0.8916\n", + "Epoch 21/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2194 - accuracy: 0.9233 - val_loss: 0.3204 - val_accuracy: 0.8914\n", + "Epoch 22/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2163 - accuracy: 0.9231 - val_loss: 0.3177 - val_accuracy: 0.8942\n", + "Epoch 23/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2129 - accuracy: 0.9243 - val_loss: 0.3192 - val_accuracy: 0.8902\n", + "Epoch 24/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2077 - accuracy: 0.9276 - val_loss: 0.3210 - val_accuracy: 0.8894\n", + "Epoch 25/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2102 - accuracy: 0.9257 - val_loss: 0.3210 - val_accuracy: 0.8926\n" + ], + "name": "stdout" + } + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "We15bby8LOpV", + "outputId": "9c836093-b473-4b24-8eab-49e38af1f30e", + "colab": { + "base_uri": "https://localhost:8080/", + "height": 303 + } + }, + "source": [ + "learning_rate = 0.01\n", + "decay = 1e-4\n", + "batch_size = 32\n", + "n_steps_per_epoch = len(X_train) // batch_size\n", + "epochs = np.arange(n_epochs)\n", + "lrs = learning_rate / (1 + decay * epochs * n_steps_per_epoch)\n", + "\n", + "plt.plot(epochs, lrs, \"o-\")\n", + "plt.axis([0, n_epochs - 1, 0, 0.01])\n", + "plt.xlabel(\"Epoch\")\n", + "plt.ylabel(\"Learning Rate\")\n", + "plt.title(\"Power Scheduling\", fontsize=14)\n", + "plt.grid(True)\n", + "plt.show()" + ], + "execution_count": 77, + "outputs": [ + { + "output_type": "display_data", + "data": { + "image/png": "iVBORw0KGgoAAAANSUhEUgAAAZgAAAEeCAYAAAC30gOQAAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAALEgAACxIB0t1+/AAAADh0RVh0U29mdHdhcmUAbWF0cGxvdGxpYiB2ZXJzaW9uMy4yLjIsIGh0dHA6Ly9tYXRwbG90bGliLm9yZy+WH4yJAAAgAElEQVR4nO3deXxU5fn//9eVBQiBsBsgsqksArKIogWtVKpoqxW1tr/W3bZYbT/tV62KtrZWba1baxfrUrW4tmoFUVGpilRBRVAEZBVZlH0zQEISQrh+f5wTHCYzyQQzM1nez8djHpm5z33OueYYc3Gf+z73be6OiIhIXctIdwAiItI4KcGIiEhSKMGIiEhSKMGIiEhSKMGIiEhSKMGIiEhSKMGINBBmdpGZFSXp2B+Z2Y213GeVmf0i3mcRJRhpUMxsgpl5+Co3sxVmdqeZ5aY7tpqYWS8ze9zM1phZmZmtM7MpZjY03bHVkaOBv6c7CKk/stIdgMgBeA04H8gGjgceBHKBy9IZVCUzy3b38ugy4FXgE+A7wFqgK3Ay0D7lQSaBu29OdwxSv6gFIw1RmbtvcPfP3P1J4AlgLICZNTezu81so5mVmtm7ZnZc5Y7h5/ERnx8PW0Odw88tw9bFceFnM7NrzOwTMysxswVmdl7E/j3D/b9nZtPMrAS4NEbMA4BDgZ+4+9vuvtrd33H337r76xHHa2Nm95rZ+jD+xWb23cgDmdno8JZWsZm9YWa9orafbmbvh/uvNLPfmVmziO0Hmdnk8PusNrNLooMNv9O3o8qqvQUW45aZm9k4M3smjHVF5LUL6xxjZh+Esc41s2+E+42Kdx5pOJRgpDEoIWjNANwOfBe4BBgKLABeMbMu4fbpwKiIfU8AtkSUjQD2AO+Fn28BfgD8BOgP3Arcb2bfjIrhVoLbQ/2B52LEuBnYC5xtZjHvHJiZAS+FMV0cHutKYHdEtebAdeH3+wrQFrgv4hhjCBLu3wiS2iXAt4HfRxxjAnAY8HWCxHwB0DNWTHXg18BkYDDwFPCwmXUPY20FvAgsAYYB1wB3JCkOSQd310uvBvMi+OP4YsTn4QQJ4imC22S7gQsitmcS3Ja6Jfx8ClBEcHv4MGAHQRK5P9x+C/Ba+D6XIHkdHxXD3cBL4fuegANXJRD7T4Di8Pz/A24GBkRsP4kgCR0eZ/+LwnP1jSg7FygDLPz8JnBD1H5jw3Ma0Cc8xsiI7T2ACuDGiDIHvh11nFXAL2rx2YFbIz5nAbuA88LPlwLbgJyIOt8P9xuV7t81vb78Sy0YaYhOMbMiMysF3iH4o/p/BLegsoGZlRXdvSKs0z8smkHQCjiaoNUyg6BPZ1S4fRRBK4dwnxYELaCiyhdBX8+hUTHNqSlod78H6EzwR3QGcAbwoZmdH1YZCqx398XVHKbM3ZdGfF4HNAPahZ+HAb+MivdJgmTZGTicIIlVttBw99XhcZJhfsR59hC05A4Ki/oBH7l7SUT9WUmKQ9JAnfzSEL0JjAPKgXUedqhX9qPEEfyT2r3IzN4HvkaQQN4A3gW6m9lhBImnso+m8h9gpwOfRh2vPOpzcSKBu/tO4HngeTP7FTCVoCXzWCL7E9y+2++QUbFmAL8Fnomxb2QnfE3TqDtBiydSdqyKNYi+To5uzTcZSjDSEO1y9+Uxyj8huEU2MnyPmWUS9FU8GVFvOkGC6Qf82d1LzWwW8Ev2739ZRHD7qYe7T6vrL+HubmZLgCPDorlAFzM7vIZWTHU+APrFuT6E58sguLX4dljWnWBEW6TNQJeI/fIjP9eRJcCFZpYT0YoZXsfnkDRSgpFGw92Lzexe4DYz2wKsBK4A8tn/+YzpwFUE/QEfRJT9Evifu+8Oj7fTzO4E7gw74N8EWgHHAnvd/YFEYzOzIQQti8cIEtdugs78S4B/hdVeJ7hF9KyZXQEsI+gnynX3WAMHYrkJeNHMVgNPEyTMgcBwd7/G3Zea2SsEAxXGEfQx/TH8GWka8BMze5ugf+b3QGmi3zdBTxL0ef3DzH5PkOSuD7dpoapGQE1VaWyuJejw/yfwITAIOMXd10fUmRH+fCvso4EgwWTxRf9LpRuAG4FfAAsJnmU5myB51cYaYAXBqKp3w9iuAu4k6D/C3fcCpxL0IT0OLAb+TNDHkhB3nwp8k6CF9l74Gs/+t/guCuOfBrxA8Id+VdShrgrjnQ78h+BZo02JxpFgrDsJbj8OIGi93UFwraHuk5mkQeXIExGRtDOzM4BJwEHuviXd8ciXo1tkIpI2ZnYhQUvpM4JbeXcDLyi5NA4pvUVmZu3NbFL4VO9qM/t+nHpmZreZ2dbwdVt4D7xy+wNmttTM9prZRTH2v8LMNpjZDjN72MyaJ/FriciByyfol1oK3AO8DJxX7R7SYKS6D+Yegs7NfIIHxO41swEx6o0jeDhsMME99NPZf/qNecDlfNFBu0/4JPN4YDTBA2SHEHSuikg94+63u3tPd2/u7j3c/fKwb0YagZT1wVgw2+3nwEB3XxaWPQasdffxUXXfBiZUjtIxsx8AP3L3Y6PqzQAedPcJEWVPAqvc/frw82jgCXev7hkJERGpY6nsg+kD7KlMLqF5BEM1ow0It0XWi9XSiWUAwdxHkfvmm1kHd98aWTEcpjkOICMnb1hWm4P2beuZpwF2AHv37iUjQ9cimq5LVbomsTX267Js2bIt7t4p1rZUJphWBPM+RdoOtI5Td3tUvVZmZl5zkyvWvoTn2S/BhC2kBwCad+ntXS68G4CCtjnMHH9iDadpGqZPn86oUaPSHUa9o+tSla5JbI39uoTPXMWUyrRaBORFleUBse63RtfNA4oSSC7x9iXOeapokZ3B1WP6JlJVRESqkcoEswzIMrPeEWWDCR5ei7Yw3FZTvVhi7bsx+vZYPGcM7srYoQUJnkpEROJJWYJx92JgInCTmeWa2UiC2WRjTfL3KHClmRWYWVeCp4onVG40s2Zm1oJgMr5sM2thZhkR+/7AzPqbWVvgV5H7xtMzL4P+XfKY+1khevhUROTLS3XP0+VADsGUE/8CLnP3hWZ2fDiteKX7CaawWAB8BEwJyyr9l2DupBEEfSglwFcB3P0VgkWn3iCYHmM18JtEgrvkuF4s21jEzOUJNXZERKQaKX2S3923ES5tG1X+FkHnfOVnJ1jd7po4xxlVw3n+SDCBX62cPrgLf3h5MQ/PXMlxvTvWdncREYnQeMfOHYDmWZmcd2wPpi3ZxIrNRTXvICIicSnBRDn3mB40y8xgwtur0h2KiEiDpgQTpVPr5nxrSFeembOG7buiF+MTEZFEKcHEcPHInpSUV/DUnOhVckVEJFFKMDEM6NqGYw9pzyNvr2ZPxd50hyMi0iApwcRxycherC0s4b+LNqY7FBGRBkkJJo7Rh+fTvX1LHp5R25VxRUQElGDiyswwLhrRkzmrP2feZ4XpDkdEpMFRgqnGOUcdTKvmWfxzploxIiK1pQRTjdYtsvnOUd14cf56Nu4oTXc4IiINihJMDS4a0ZMKdx57J+6SByIiEoMSTA26d2jJyf3zeWLWakrLK9IdjohIg6EEk4BLRvbi813lPDd3bbpDERFpMJRgEjC8V3sGdM3j4ZkrtVaMiEiClGASYGZcMlJrxYiI1IYSTIJOG9yFjq2a87CGLIuIJEQJJkHNszI5P1wr5hOtFSMiUiMlmFo499juwVoxM1elOxQRkXpPCaYWOrZqzhlDuvKf97VWjIhITZRgaunikb0oKa/g37O1VoyISHWUYGqpf9c8vnJIBx55e5XWihERqYYSzAG45LherNteytSFWitGRCQeJZgDcGK/g+iQm80VT31Ir/FTGPmHaXrKX0QkSla6A2iIXpi3jh2leyivCJ7qX1tYwnUTFwAwdmhBOkMTEak31II5AHdMXbovuVQqKa/gjqlL0xSRiEj9owRzANYVltSqXESkKVKCOQBd2+bUqlxEpClSgjkAV4/pS0525n5l2ZnG1WP6pikiEZH6R538B6CyI/+OqUtZV1hCVqaRk53BSf3z0xyZiEj9oQRzgMYOLdiXaN5f/Tln3/s297yxnGtO6ZfmyERE6gfdIqsDw3q046wjC3jwrZWs2lKc7nBEROoFJZg6Mv6UfjTLyuDmFxelOxQRkXpBCaaOHJTXgp+NPozXl2zijSWb0h2OiEjaKcHUoYtG9OKQTrnc9OIiyvZUpDscEZG0SmmCMbP2ZjbJzIrNbLWZfT9OPTOz28xsa/i6zcwsYvsQM3vfzHaFP4dEbGtuZveZ2UYz22ZmL5hZSuZvaZaVwa9P68/KLcU8PGNVKk4pIlJvpboFcw+wG8gHzgXuNbMBMeqNA8YCg4FBwOnApQBm1gyYDDwOtAMeASaH5QA/B74S7tcV+Bz4a5K+TxWj+h7E1w/P56/TPmbjjtJUnVZEpN5JWYIxs1zgbOAGdy9y9xnA88D5MapfCNzl7mvcfS1wF3BRuG0UwfDqu929zN3/AhhwYri9FzDV3Te6eynwFBAriSXNr0/rz569zq0vLU7laUVE6pVUPgfTB9jj7ssiyuYBJ8SoOyDcFllvQMS2+e4eOdvk/LD8FeAh4M9m1hUoJGgpvRwrIDMbR9BaolOnTkyfPr2WXym+Md0zee7DdQxovo3e7TJr3qGeKioqqtPr0ljoulSlaxJbU74uqUwwrYAdUWXbgdZx6m6Pqtcq7IeJ3hZ9nI+Bz4C1QAWwAPhprIDc/QHgAYC+ffv6qFGjEvwqNRs+Yg9z7vofkz5txgtnHEdmhtW8Uz00ffp06vK6NBa6LlXpmsTWlK9LKvtgioC8qLI8YGcCdfOAorDVUtNx7gGaAx2AXGAicVowydSyWRbXf+NwFq3fwb9nf5rq04uIpF0qE8wyIMvMekeUDQYWxqi7MNwWq95CYFDkqDKCDv3K7UOACe6+zd3LCDr4h5tZxzr4DrVy2qAuHNOrPXdOXUrhrt2pPr2ISFqlLMG4ezFBa+ImM8s1s5HAGcBjMao/ClxpZgVhX8pVwIRw23SCW18/C4ckV97+mhb+nA1cYGZtzCwbuBxY5+5bkvG9qmNm3PitAWwvKeePry6reQcRkUYk1cOULwdygE3Av4DL3H2hmR1vZkUR9e4HXiDoP/kImBKW4e67CYYwX0DQiX8JMDYsB/gFUErQF7MZ+AZwZpK/V1yHd8nj/GN78Pi7q1m0LroLSkSk8UrpbMruvo0gOUSXv0XQeV/52YFrwles48wFhsXZtpVg5Fi9ccVJfXh+3jpufGEhT407lv3v7omINE6aKiYF2rZsxtVj+vHeym28MH99usMREUkJJZgU+e7R3RhYkMfvpyxm1+496Q5HRCTplGBSJDPDuPH0AWzYUcrw371Or/FTGPmHaTw3d226QxMRSQqtaJlCaz4vIdOMorKgBbO2sITrJi4AvliGWUSksVALJoXumLqUiv1muIGS8grumLo0TRGJiCSPEkwKrSssqVW5iEhDpgSTQl3b5tSqXESkIVOCSaGrx/QlJ7vqzMqXjTokDdGIiCSXEkwKjR1awK1nHUFB2xwM6NS6OZkGUxduZO9er3F/EZGGRKPIUmzs0IL9Row9OetTrp+0gL+9sZyfje5dzZ4iIg2LWjBp9r3h3ThzaAF/em0ZM5enfD5OEZGkUYJJMzPjlrEDObRTK37+77ls3FGa7pBEROpEwgnGzE41sxfNbJGZdQvLfmhmo5MXXtOQ2zyLe889kuKyCv7vX3PZU7E33SGJiHxpCSUYMzsXeJpgCvxeQHa4KZM4Mx5L7fTOb83vzxrIeyu3ced/tXaMiDR8ibZgrgF+5O5XAJEzNb5LsIKk1IEzhx7M94Z3577/fcLrizemOxwRkS8l0QTTG3gnRnkRkFd34chvTu/PgK55XPn0PD7btivd4YiIHLBEE8w6oE+M8q8Cn9RdONIiO5O/n3ske/c6P33yA8r2VKQ7JBGRA5JognkA+IuZjQw/dzOzC4HbgXuTElkT1qNDLnecM4h5a7Zz60tL0h2OiMgBSehBS3e/3czaAK8CLYA3gDLgTne/J4nxNVmnDOzCD47rxUMzVnJ0z/Z8c1CXdIckIlIrCQ9TdvdfAh2B4cCxQCd3vyFZgQmMP7UfR3Zvy7XPzmfF5qJ0hyMiUisJtWDM7GHg5+6+E5gTUZ4L/NXdL0lSfE1admYGf/v+kXzzL29x7j/excxYv72Urm1zuHpMXy1SJiL1WqItmAuBWHPK5wAX1F04Eq1r2xzOOepg1u8oY932UpwvVsLUcssiUp9Vm2DMrL2ZdQAMaBd+rnx1Ak4D9MBGkk2Zv6FKmVbCFJH6rqZbZFsAD1+LYmx34Dd1HZTsTythikhDVFOC+RpB62UacDawLWLbbmC1u69LUmwS6to2h7UxkolWwhSR+qzaBOPu/wMws17AZ+6uWRjT4Ooxfblu4gJKyvd/6PLbw9TJLyL1V6LPwawGMLOuQHegWdT2N+s+NKlUOVrsjqlLWVdYQn5eC/ZUVPDwzFWc1L8zAwvapDlCEZGqEh2m3BV4kmBqGCe4bRa5xm/VhealTkWvhLm2sITv3PcO5z80i6cu/Qp98lunMToRkaoSHaZ8N1AB9Ad2AccD5wCLgVOSE5pUp6BtDk/88BiyMzM478FZrNpSnO6QRET2k2iCOQG41t2XELRcNrv7ROBa4OZkBSfV69kxlyd+eAzlFXs598FZMQcCiIikS6IJJodgyDIEI8kOCt8vAgbVdVCSuN75rXnsB8ewo7Scc//xLpu05LKI1BOJJpglQL/w/YfAj82sB/ATQI+Tp9nAgjZMuHg4m3aWcd5Ds9hWvDvdIYmIJJxg/gx0Dt/fBJwMrAAuB65PQlxSS8N6tOPBC49i9dZdXPDwLHaUlqc7JBFp4hJKMO7+hLtPCN9/APQEjga6u/sziZ4snGJmkpkVm9lqM/t+nHpmZreZ2dbwdZuZWcT2IWb2vpntCn8Oidr/SDN708yKzGyjmf080RgbshGHduS+84axdMNOLv7nbIrL9tS8k4hIkiQ8XX8kd98VJppiMxtfi13vIZgBIB84F7jXzAbEqDcOGAsMJujjOR24FMDMmgGTgceBdsAjwOSwHDPrCLwC3A90AA4D/lvb79hQfa3fQfzl/xvK3E8/50ePzqG0XCtiikh61JhgzKyjmX3TzE42s8ywLNvM/h+wCvhFIicKp/Y/G7jB3YvcfQbwPHB+jOoXAne5+xp3XwvcBVwUbhtF8PzO3e5e5u5/IXgu58Rw+5XA1LDVVebuO919cSIxNhanHtGFO88ZzDsrtnLW32cy4tbX6TV+CiP/ME0zMItIylT7oKWZjQCmAG0IhifPNrOLgElANsEQ5YcTPFcfYI+7L4som0cwBDragHBbZL0BEdvmu3vkg57zw/JXCBZDW2BmbxO0XmYBP3H3T2N8v3EErSU6derE9OnTE/wq9V974Lgumby1bue+srWFJVzzzIcsWryIEV2zEzpOUVFRo7oudUXXpSpdk9ia8nWp6Un+m4GpwC3AxcAVwIsEHf2PRf2Rr0krYEdU2XYg1iPorcJtkfVahf0w0duij3MwcCRwErAAuB34FzAy+iTu/gDwAEDfvn191KhRiX+bBuCX704D9n82ZvdemPJpJtd/f1RCx5g+fTqN7brUBV2XqnRNYmvK16WmW2SDgZvd/SPgBoJWzHXu/mgtkwtAEZAXVZYH7Eygbh5QFJ6zpuOUAJPcfba7lwK/BUaYWZObsEvT/ItIOtWUYNoDmyHo2CeYJmbuAZ5rGZBlZr0jygYDC2PUXRhui1VvITAoclQZwUCAyu3z2X+etNomwkYj3nT+nVo3T3EkItIUJTKKrHIlyw4Ef6zzola2bJ/Iidy9GJgI3GRmuWY2EjgDeCxG9UeBK82sIJxo8ypgQrhtOsG8aD8zs+Zm9tOwfFr485/AmeFQ5myCltcMd4++rdboXT2mLznZVech3VlazrsrtqYhIhFpShJJMIsIWjGbCPo/ZoefNxNMH7O5Fue7nGDamU0E/SKXuftCMzvezIoi6t0PvEDQh/IRwUCD+wHcfTfBEOYLgELgEmBsWI67TyN4+HNKeJ7DgJjP2zR2Y4cWcOtZR1DQNgcjmCDzhm8eTte2OZz/0CyefX9NukMUkUYskRUt64y7byNIDtHlbxEkr8rPDlwTvmIdZy4wrJrz3Avc+2XjbQyip/kH+PZR3bj8ife56pl5rNpazBVf70NGhsU5gojIgUloRUtpXNrkZDPh4uH8atJH/HXaclZuKebOcwbTIsbtNBGRA5XQgmPS+GRnZvCHs4/gkE653PryEtYVlvDABUfRsZUGAIhI3TigqWKkcTAzLj3hUO4770gWrd/BmX+fyfJNsUaNi4jUnhKMcMrALjw17iuUlu/lzL+/zczlW2reSUSkBrpFJgAM7taW534ykh9MmM2FD7/H2UcWMGP5VtYWllDw7jSuHtO3ymABEZHqqAUj+xS0zeGZH3+Fww5qxVNz1uxbgnltYQnXTVygiTJFpFYSasGYWbwJLR0oBZYDT7n7uroKTNKjdYvsmIuVlZRXcMfUpWrFiEjCEr1F1gk4HthL8OAjwECCafLfB84ieEL/eHf/sM6jlJRaX1gas1xzmIlIbSR6i2wm8DJwsLt/1d2/SjBr8UsEi3n1IHhy/q6kRCkpFW8Os5bNM7WAmYgkLNEE83PgpnDCS2Df5Je/A64Ip2m5DRgSZ39pQGLNYZaZYRSXVfCtv81g8froVRdERKpKNMG0ArrEKO/MF1O87ECj0hqFyDnMIOj8v+ucwTxyyXA+31XOGX+byUMzVrJ3b5OdqFpEEpBoQpgEPGRm1xBMdglwNMFiXhPDz8MJpuSXRqByDrPoxZJe+fnxXPvsAm5+cRHTl27iznMGk5/XIn2Biki9lWgL5scEK1s+DnwSvh4nWKL48rDOYuBHdR2g1C8dWjXnHxcM43dnDmT2qm2ccvebTF24Id1hiUg9lFCCcfdd7v5jggXIhoav9u5+WbjOC+7+oUaQNQ1mxrnH9ODF/zuegnY5XPrY+1w3cT67du9Jd2giUo/Uqs8kTCbzkxSLNDCHHdSKiZeN5I+vLuP+Nz9h1optjB3aladmr2FdYQld2+ZoBgCRJizRBy1bEIwkGw0cRFTLx90H1X1o0hA0y8pg/Kn9OKFPJ378+Bz++OrH+7ZVzgAAKMmINEGJtmD+DpwJPAO8TRNe515i+8qhHWjZLIvtJfvfJtMMACJNV6IJZixwjru/lsxgpGHbsF0zAIjIFxIdRbYL+CyZgUjDF28GADP493uf6rkZkSYm0QRzO3ClmWnhdokr1gwAzbMy6NGhJeMnLuDMe99m/prCNEUnIqmW6C2ykwgmuzzFzBYB+0236+7fquvApOGp7Ge5Y+rS/UaRnTGkK899uJbfTVnCGffM5HvDu3P1yX1pl9sszRGLSDIlmmC2EDzNL1KtyhkAop059GBGH57P3a9+zCPvrOKlBeu5Zkw/vnt0NzIz1DAWaYwSSjDufnGyA5HGL69FNr8+vT/fOfpgfj15IddPWsC/Z3/KTWcMZNWW4iotH408E2nYNDmlpFy/znk8Ne5Ynp+3jlumLGbsPTPJzDAqwkEAen5GpHGI28lvZvPNrF34fkH4OeYrdeFKY2FmnDGkgGlXnUCr5pn7kkulyudnRKThqq4F8yxQFr7/TwpikSaodYtsistiL2Km52dEGra4CcbdfxvrvUhd69o2h7UxkklGhvH07M8468gCsjITHVEvIvWF/q+VtIv1/EyzzAy6tmnBNc/O5+t//B+T5q6pchtNROq3hBKMmbU3s3vNbJmZFZrZjshXsoOUxi1yBU0jWEHz9m8P4s1rvsY/LjiKnGZZXPHUPMbc/SZT5q/XjAAiDUSio8geIlgD5gFgHZrsUupYvOdnTuqfz+h+B/HKwg386dVl/OTJD+jXuTVXntSHk/rnM/nDdRreLFJPJZpgRgMnufusZAYjEktGhvGNI7owZkBnXpy/jrtf+5hxj71Pt3Y5bNxRxu6KvYCGN4vUN4n2wWwCipIZiEhNMjOCoc2vXvFV7vj2INZtL92XXCppeLNI/ZFogvklcJOZtUpmMCKJyMrM4JyjusXti9HwZpH6IdEE8yvgZGCTmS3Wg5ZSH1S3PMCdU5eyaUfs9WlEJDUSTTD/Ae4EbgP+TfAQZuQrIeFotElmVmxmq83s+3HqmZndZmZbw9dtkUsFmNkQM3vfzHaFP4fEOEazMBmuSTQ+aVhiDm/OymBA1zzumb6ckbdN48qnP2Thuu1pilCkaauxk9/MsoFc4B53X/0lz3cPsBvIB4YAU8xsnrsvjKo3jmAVzcEEI9ZeBVYC95lZM2AycDfBUs6XApPNrLe77444xtXAZqD1l4xZ6ql4ywOMHVrA6q3F/HPmKp6Z8xkTP1jLsYe054fHHcKJ/Q4iI8N4bu5ajT4TSbIaE4y7l5vZZQR/zA+YmeUCZwMD3b0ImGFmzwPnA+Ojql8I3OXua8J97wJ+BNwHjArjvtvdHfiLmf0COBF4JazfCzgPuBL4x5eJW+q3eMObe3TI5cZvDeCKk/rw1OxPmTBzFT98dA69OuZyZPe2TFmwntJyjT4TSaZEhyn/l+AP+MNf4lx9gD3uviyibB5wQoy6A8JtkfUGRGybHyaXSvPD8lfCz38Frgeq7e01s3EErSU6derE9OnTE/oiTUlRUVGDvy59gJuPyWDOxuZMXVXCsx8UV6lTUl7BzZPn0Xb7xwkdszFcl7qmaxJbU74uiSaY14Hfm9kg4H1gv/9D3X1iAsdoBUQ/9b+d2LewWoXbIuu1CvthorftdxwzOxPIdPdJZjaquoDc/QGCh0fp27evjxpVbfUmafr06TSW6zIauMadQ657KeaTwttKPeHv2piuS13RNYmtKV+XRBPM38KfP4uxzYHMGOXRioC8qLI8YGcCdfOAInd3M4t7nPA23O3ANxKIR5ogM4s7uSYGNz6/kO8e3Y3Du0T/iolIbSU0iszdM6p5JZJcAJYBWWbWO6JsMBDdwU9YNjhOvYXAoMhRZcCgsLw30BN4y8w2ABOBLma2wcx6JhinNHLxJtccfHAbnuEIjOUAABV3SURBVJz1Kaf++S2+9bcZPP7uanaUlqcpSpGGL2UrWrp7sZlNJHhg84cEo8jOAEbEqP4ocKWZvUTQQrqKoF8FYDpQAfzMzO4j6PwHmAbsBbpFHGcEQevrSIIRZSLVjj77vHg3z324lqdmf8avnvuIW6Ys4hsDu/Cdo7txTK/2++Y+W1tYQsG70zT6TKQaCSeYcHXLU4HuQLPIbe5+U4KHuZxgoMAmYCtwmbsvNLPjgZfdvXKmgPuBQ4AF4ecHwzLcfbeZjQ3L/gAsBsZGDFHeEBHzNmCvu+8rE4H4o8/a5Tbj4pG9uGhETxas3c5Tsz/j+Q/XMXHuWjrmZlNYsoc9WtpZJCEJJRgzOxaYQrDCZSdgLdAl/LwKSCjBuPs2gudbosvfIui8r/zswDXhK9Zx5gLDEjjfdODgRGITiWRmDDq4LYMObsuvvtmflxas57qJC/Yll0qVc58pwYhUleiT/HcATwAFQCnBkOXuwByCp/tFGq2cZpmcPexgyqMm1qy0trCESXPXsFP9NSL7SfQW2SDgB+EorgqgubuvMLNrgScJko9IoxZv9FmmwRVPzaNZVgYn9j2I0wZ3YXS/fHKaJTr+RaRxSjTBRE7BshHoQdD3UQR0reugROqjq8f05bqJCygpr9hXlpOdye/HDqR7x5a8MG89Uxas55WFG2jZLJPRh+dz2qAunNCnE698tEFT00iTk2iC+QA4mmCo8XTgFjPLJ5iORbMpS5MQOfpsbWEJBVGJYliP9txwWn/eW7mNF+ev4+WPNvDCvHU0zzT27IUK1+AAaVoSTTC/5Isn7n9FMIz4rwQJ5+IkxCVSL1WOPov3dHZmhvGVQzvwlUM78NtvDeDtT7by48ffp6yiYr96JeUV3PbKEiUYadQSSjDuPifi/WaC4coiUo2szAy+2qcTJbsrYm5fv72U79z3DqMPP4iv98/n0E5az08al1o9aGlmRwGHAi+GD07mAmXuvicp0Yk0AvEGB7RukUXx7j3c+vISbn15Cb065jK6X5BsjurRjqzMDC0rIA1aos/B5BOswTKc4Mn63sAK4I8Ew5Z/nqwARRq6eIMDbj5jIGOHFrCusITXl2zitUUbefSd1Tw4YyVtcrI5rFMu89dup7xCfTfSMCXagvkTweixDsCnEeXP8MUULiISQ3VT00DQwjn/2B6cf2wPisr2MOPjzby6aBOT5q4h6rlOSsoruH2q+m6kYUg0wYwGRrv75/vPMcknBA9cikg14k1NE61V8yxOGdiFUwZ2YeIHsVf7XldYyo8fe5/jenfk+N4d6dEht67DFakTiSaYHPZ/FqZSJ4JbZCJSx+L13bRslsmCtdt5ZWEwxV739i2DZHNYR0Yc2pE2LbPVdyP1QqIJ5k3gIoJVIgHczDKBawkWIxOROhb3wc4zj+CMIV1ZsaWYGR9v4a2Pt/D8h+t4ctanZBgUtMthfWGpJuWUtEs0wVwD/M/MjgaaA3cRLFHcBhiZpNhEmrSa+m4O7dSKQzu14sIRPSmv2MuHnxXy1sdbuHf68piTct4yZREnD8inZbOUrdIhTVyiz8EsMrMjgMsIZlBuQdDBf4+7r09ifCJNWqJ9N9mZGRzdsz1H92zPX1//OGadLUW7GXTjfxl0cBuG9+rAMb3aM6xnO/JaZO+ro1trUpcS/qdMuKbKbyLLzKyHmT3t7t+p88hE5IDE67vpkNuM7xzdjfdWbuOhGSu473+fkGHQv2sew3t2wHH+9d6nlJYHs0br1pp8WV+2rdwWOLsuAhGRuhGv7+aG0/rvSxQluyuY++nnzFq5jfdWbuOJWasp21N1OQINi5YvQzdjRRqZmvpuIFjjZsRhHRlxWEcAyvZU0O9Xr+AxjreusJTzH5rF0O7tGNq9LUO7taVtyy8Wta28raZlpCWaEoxII5Ro302l5lmZ1Q6L3lK0m79N+3jfg5+HdMxlaPd2mMEL89bta/3otppEUoIREaD6YdFjhxZQXLaH+Wu2M/ezz/lgdSH/W7aJLUVVH48rKa/g9y8t5luDu5KRYVW2S9NRbYIxs+dr2D+vDmMRkTSq6dZabvOsfUsRALg7h1z3Uszbapt2ljH4t/9lQEEeRxS0YWBBG44oaEPPDrn7ko5GrDV+NbVgtiawfWUdxSIiaVabW2tmFve2WtucbE4b3IUFa3fwyDur2R3eQmvdPIsBBXnkZGcwY/lWTeTZyFWbYNxdi4mJSFzxbqvd+K0B+xJFecVePt5YxIK1hSxYu50Fa3fw7optVY5VUl7Bjc8vpHuHlvTr3DruA6Fq+TQc6oMRkQNW0zLSEDwE2r9rHv275vHdo4OyXuOnxLy1VlhSzll/fxsz6NG+JYd3yaNf5zwO79Kaw7vkMWfVNq6f9NG+hKaWT/2mBCMiX0pNy0jHEu/WWn5ec24+YyBLNuxk8fodLNmwk1cWbsDDbGRQJTGVlFdwx9SlSjD1kBKMiKRcvFtr1516OCcP6MzJAzrvKy8u28PSjTtZsn4n109aEPN4awtLOO/BWfTOb0Xvg1rTJ78VvfNb0yZH0+CkkxKMiKRcIg+DVsptnsWR3dtxZPd23PPG8pgtn5zsTHaUlvPv9z7bL2kd1Lo5ffJbk2HwzgoNKkg1JRgRSYvaPgwK8Vs+t54VPKuzd6+ztrCEjzft5OONRSzbWMTHm3ayYM32mLfWrp+0gM07yzikUy6HdGpFt3Y5ZGVm7FdPLZ8DpwQjIg1GTS2fjAyjW/uWdGvfkhP75e/br9f4KTGPt2t3Bb97afG+z9mZRo8OuRzSMUg420t2M/GDtZqp4AApwYhIg3IgLZ94gwoK2uYw5WfH8cnmYlZsLtr3c8WWYt5YumnfLbVIJeUV3DD5I8ygZ4dcenbIpU3L7Cr1NEebEoyINAHxbq1dPaYvbVs2Y1iPZgzr0W6/ffZU7KX3L1+OOZx6Z+kefv7vD/d9btsymx4dcunVoSU9OuSypbiU/8xRy0cJRkQavdoMKqiUlZkRt+XTtU0LJlwynFVbilm9dRcrtxazemsxs1d9zuR56/YNq45UUl7Br577iJ2l5ftu4x3cLofmWZn71WtMfT5KMCLSJNTloIJrTulHn/zW9MlvXWWf6pY+KCrbww2TF+77bAad81rQrV2QcIrLynl9yaYDGu1WHxOTEoyISBwH0vKpbumDrm1bMOnykXy6bRefbdvFp+FrzbYSZi7fwoYdpVX2KSmv4Npn5zNz+RYK2uVQ0DaHgnY5HNy2JZ3btKBZVgbPzV27XyKsL7fkUppgzKw98BBwMrAFuM7dn4xRz4A/AD8Mix4ExrsHDU8zGxIe53BgMfADd/8w3HY1cCHQIzzH3939jmR+LxFpvOq05TOmH/l5LcjPa8HRPdtX2S/eFDple/by5seb2bSzbL/bb2aQ37oF24p3s7ti/xVJS8oruO2VJZwxpCvBn9TYktnySXUL5h5gN5APDAGmmNk8d18YVW8cMBYYTDAzxKsEszbfZ2bNgMnA3cDfgUuByWbW2913E8wmcQEwHzgU+K+Zfebu/076txMRIbE52mKpbrTbzPEnUrangg3bS1nzeQlrPy9hTWHw89kP1sQ83vrtpQz4zVS6tGlB17Y5dGnTgi5tcujaNvi5eP12/vTax5SW134wQmViatb5sGHx6qQswZhZLnA2MNDdi4AZ4Xoz5wPjo6pfCNzl7mvCfe8CfgTcB4wK4747bNH8xcx+AZwIvOLut0ccZ6mZTQZGAkowIpIyBzJHW3Wj3SC4/dajQy49OuTut9+7K7bGTExtcrI4+8hurN9ewrrtpSzdsJnNRWUxByFUKimv4IbnPqJ49x46h62tzm1a0L5ls/3W8omOMxbz6s5Uh8xsKDDT3VtGlP0COMHdT4+qux042d1nhZ+PAt5w99ZmdkW47dSI+i+G2++KOo4BHwD3u/t9MWIaR9BaolOnTsOefvrpOvq2jUdRURGtWrVKdxj1jq5LVbomsdX2ury9rpxnl5WztdTp0MI4u082I7pWfc4mep8JH+1md8RdsmYZcNHAZlX23bPX+bzU2Vbq3Ppe1T6feDIN2jY32rUwPt2xd9+51j/y/yhb/3HMe3CpvEXWCtgRVbYdqDoMI6i7PapeqzBhRG+r7jg3AhnAP2MF5O4PAA8A9O3b1xP9V0ZTUpt/fTUlui5V6ZrEVtvrMgq4vpbnGAX0P4C+lEeXTYs7GOHZy0awYXspG3eUsmF7KRt2lO17v7ywprUoA6lMMEVUXWI5D9iZQN08oMjd3cwSOo6Z/ZSgL+Z4dy/7MoGLiNR3dT0YoUubHLq0yYm538g/xE5M0TJqrFF3lgFZZtY7omwwEN3BT1g2OE69hcAg239YxKDI45jZJQT9OqMr+3FERGR/Y4cWcOtZR1DQNgcjGExQOXFoda4e05ec7Mxq60AKWzDuXmxmE4GbzOyHBKPIzgBGxKj+KHClmb1EMIrsKuCv4bbpQAXwMzO7j6DzH2AagJmdC/we+Jq7r0jS1xERaRQOpOUTOUpufTX1UtmCAbgcyAE2Af8CLnP3hWZ2fHjrq9L9wAvAAuAjYEpYRjgUeSzB7a9C4BJgbFgOcAvQAZhtZkXhq0oHv4iIHLixQwuYOf5Edm9Y/n68Oil9DsbdtxEkh+jytwg67ys/O3BN+Ip1nLlAzLHX7t6rToIVEZEvJdUtGBERaSKUYEREJCmUYEREJCmUYEREJCmUYEREJCmUYEREJCmUYEREJCmUYEREJCmUYEREJCmUYEREJCmUYEREJCmUYEREJCmUYEREJCmUYEREJCmUYEREJCmUYEREJCmUYEREJCmUYEREJCmUYEREJCmUYEREJCmUYEREJCmUYEREJCmUYEREJCmUYEREJCmUYEREJCmUYEREJCmUYEREJCmUYEREJCmUYEREJCmUYEREJCmUYEREJCmUYEREJCmUYEREJCmUYEREJClSmmDMrL2ZTTKzYjNbbWbfj1PPzOw2M9savm4zM4vYPsTM3jezXeHPIYnuKyIiqZHqFsw9wG4gHzgXuNfMBsSoNw4YCwwGBgGnA5cCmFkzYDLwONAOeASYHJZXu6+IiKROyhKMmeUCZwM3uHuRu88AngfOj1H9QuAud1/j7muBu4CLwm2jgCzgbncvc/e/AAacmMC+IiKSIlkpPFcfYI+7L4somwecEKPugHBbZL0BEdvmu7tHbJ8flr9Sw777MbNxBC0egDIz+yixr9KkdAS2pDuIekjXpSpdk9ga+3XpEW9DKhNMK2BHVNl2oHWcutuj6rUK+1Kit0UfJ+6+UUkJd38AeADAzOa4+1GJf52mQdclNl2XqnRNYmvK1yWVfTBFQF5UWR6wM4G6eUBRmCBqOk51+4qISIqkMsEsA7LMrHdE2WBgYYy6C8NtseotBAZFjQwbFLU93r4iIpIiKUsw7l4MTARuMrNcMxsJnAE8FqP6o8CVZlZgZl2Bq4AJ4bbpQAXwMzNrbmY/DcunJbBvdR6o/bdqEnRdYtN1qUrXJLYme10slXeOzKw98DBwErAVGO/uT5rZ8cDL7t4qrGfAbcAPw10fBK6tvM1lZkPDsv7AYuAH7j43kX1FRCQ1UppgRESk6dBUMSIikhRKMCIikhRNPsEkOj9aU2Nm082s1MyKwtfSdMeUamb2UzObY2ZlZjYhattoM1sSzof3hpnFfdissYl3Xcysp5l5xO9MkZndkMZQUyocdPRQ+Hdkp5l9aGanRmxvcr8zTT7BkPj8aE3RT929Vfjqm+5g0mAdcAvBwJR9zKwjwYjIG4D2wBzgqZRHlz4xr0uEthG/NzenMK50ywI+I5idpA3wK+DpMPE2yd+ZVD7JX+9EzI820N2LgBlmVjk/2vi0Bidp5+4TAczsKODgiE1nAQvd/Zlw+43AFjPr5+5LUh5oilVzXZq08FGMGyOKXjSzlcAwoANN8Hemqbdg4s2PphZM4FYz22JmM81sVLqDqUf2m+8u/MPyCfq9qbTazNaY2T/Df7k3SWaWT/A3ZiFN9HemqSeY2syP1tRcCxwCFBA8KPaCmR2a3pDqjZrmw2uqtgBHE0x+OIzgejyR1ojSxMyyCb77I2ELpUn+zjT1BFOb+dGaFHef5e47wyURHgFmAt9Id1z1hH5vYgiX4Zjj7nvcfSPwU+BkM2vUf0SjmVkGwQwluwmuATTR35mmnmBqMz9aU+cE6+5I1Hx3YV/eoej3JlrlU9xN5u9MOJPIQwSDhs529/JwU5P8nWky/+FjqeX8aE2GmbU1szFm1sLMsszsXOCrBOvtNBnhd28BZAKZldcDmAQMNLOzw+2/JlijqNF21kaKd13M7Bgz62tmGWbWAfgLMN3do28NNWb3AocDp7t7SUR50/ydcfcm/SIYMvgcUAx8Cnw/3TGl+wV0AmYTNN8LgXeBk9IdVxquw40E/wqPfN0Ybvs6sAQoIZiAtWe64033dQG+B6wM/19aTzDxbOd0x5vC69IjvBalBLfEKl/nNtXfGc1FJiIiSdGkb5GJiEjyKMGIiEhSKMGIiEhSKMGIiEhSKMGIiEhSKMGIiEhSKMGINFLh2izfTncc0nQpwYgkgZlNCP/AR7/eTXdsIqnSpNeDEUmy1wjWFoq0Ox2BiKSDWjAiyVPm7huiXttg3+2rn5rZlHAJ3dVmdl7kzmZ2hJm9ZmYlZrYtbBW1iapzoZktCJcv3mhmj0TF0N7MngmXBF8RfQ6RZFKCEUmf3wLPA0MI1tx5NFwlsnK23akEc1kNB84ERhCxTLGZXQrcD/wTGESwnMJHUef4NTCZYCbfp4CHzax78r6SyBc0F5lIEpjZBOA8gokPI93j7teamQMPuvuPIvZ5Ddjg7ueZ2Y+AO4GD3X1nuH0U8AbQ292Xm9ka4HF3j7m8d3iOP7j7deHnLIIF9sa5++N1+HVFYlIfjEjyvAmMiyorjHj/TtS2d4Bvhu8PJ5jOPXJBqreBvUB/M9tBsNro6zXEML/yjbvvMbPNwEGJhS/y5SjBiCTPLndfnoTj1ua2Q3nUZ0e3xiVF9Ismkj7Hxvi8OHy/GDgiarnhEQT/zy52903AWmB00qMUOUBqwYgkT3Mz6xxVVuHum8P3Z5nZbILFp75NkCyOCbc9QTAI4FEz+zXQjqBDf2JEq+h3wJ/MbCMwBWgJjHb3u5L1hURqQwlGJHm+TrCyY6S1wMHh+xuBswmWFt4MXOzuswHcfZeZjQHuBt4jGCwwGfh55YHc/V4z2w1cBdwGbANeStaXEaktjSITSYNwhNc57v6fdMcikizqgxERkaRQghERkaTQLTIREUkKtWBERCQplGBERCQplGBERCQplGBERCQplGBERCQp/n8M9qwIio552wAAAABJRU5ErkJggg==\n", + "text/plain": [ + "
" + ] + }, + "metadata": { + "tags": [], + "needs_background": "light" + } + } + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "txkhYc0BLOpV" + }, + "source": [ + "### 지수 기반 스케줄링" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "m1fGE70_LOpV" + }, + "source": [ + "```lr = lr0 * 0.1**(epoch / s)```" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "MR9yQ-PhLOpV" + }, + "source": [ + "def exponential_decay_fn(epoch):\n", + " return 0.01 * 0.1**(epoch / 20)" + ], + "execution_count": 78, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "rkHd1gawLOpW" + }, + "source": [ + "def exponential_decay(lr0, s):\n", + " def exponential_decay_fn(epoch):\n", + " return lr0 * 0.1**(epoch / s)\n", + " return exponential_decay_fn\n", + "\n", + "exponential_decay_fn = exponential_decay(lr0=0.01, s=20)" + ], + "execution_count": 79, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "ZpgT0jO3LOpW" + }, + "source": [ + "model = keras.models.Sequential([\n", + " keras.layers.Flatten(input_shape=[28, 28]),\n", + " keras.layers.Dense(300, activation=\"selu\", kernel_initializer=\"lecun_normal\"),\n", + " keras.layers.Dense(100, activation=\"selu\", kernel_initializer=\"lecun_normal\"),\n", + " keras.layers.Dense(10, activation=\"softmax\")\n", + "])\n", + "model.compile(loss=\"sparse_categorical_crossentropy\", optimizer=\"nadam\", metrics=[\"accuracy\"])\n", + "n_epochs = 25" + ], + "execution_count": 80, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "cQu05HxbLOpW", + "outputId": "2145961f-e67b-4e92-b9b3-1a28482bccf3", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "lr_scheduler = keras.callbacks.LearningRateScheduler(exponential_decay_fn)\n", + "history = model.fit(X_train_scaled, y_train, epochs=n_epochs,\n", + " validation_data=(X_valid_scaled, y_valid),\n", + " callbacks=[lr_scheduler])" + ], + "execution_count": 81, + "outputs": [ + { + "output_type": "stream", + "text": [ + "Epoch 1/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 1.1106 - accuracy: 0.7344 - val_loss: 0.8633 - val_accuracy: 0.7328\n", + "Epoch 2/25\n", + "1719/1719 [==============================] - 4s 3ms/step - loss: 0.7067 - accuracy: 0.7824 - val_loss: 0.7695 - val_accuracy: 0.7576\n", + "Epoch 3/25\n", + "1719/1719 [==============================] - 4s 3ms/step - loss: 0.6652 - accuracy: 0.8040 - val_loss: 0.6658 - val_accuracy: 0.8108\n", + "Epoch 4/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.5463 - accuracy: 0.8319 - val_loss: 0.5714 - val_accuracy: 0.8504\n", + "Epoch 5/25\n", + "1719/1719 [==============================] - 4s 3ms/step - loss: 0.5098 - accuracy: 0.8400 - val_loss: 0.5460 - val_accuracy: 0.8506\n", + "Epoch 6/25\n", + "1719/1719 [==============================] - 4s 3ms/step - loss: 0.4484 - accuracy: 0.8567 - val_loss: 0.5718 - val_accuracy: 0.8620\n", + "Epoch 7/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.4328 - accuracy: 0.8703 - val_loss: 0.4861 - val_accuracy: 0.8628\n", + "Epoch 8/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.3765 - accuracy: 0.8795 - val_loss: 0.5380 - val_accuracy: 0.8580\n", + "Epoch 9/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.3632 - accuracy: 0.8844 - val_loss: 0.4584 - val_accuracy: 0.8704\n", + "Epoch 10/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.3122 - accuracy: 0.8944 - val_loss: 0.4571 - val_accuracy: 0.8796\n", + "Epoch 11/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2846 - accuracy: 0.9025 - val_loss: 0.4326 - val_accuracy: 0.8786\n", + "Epoch 12/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2725 - accuracy: 0.9079 - val_loss: 0.4818 - val_accuracy: 0.8758\n", + "Epoch 13/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2495 - accuracy: 0.9137 - val_loss: 0.4428 - val_accuracy: 0.8822\n", + "Epoch 14/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2275 - accuracy: 0.9207 - val_loss: 0.4235 - val_accuracy: 0.8786\n", + "Epoch 15/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2211 - accuracy: 0.9237 - val_loss: 0.4484 - val_accuracy: 0.8858\n", + "Epoch 16/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2007 - accuracy: 0.9297 - val_loss: 0.4862 - val_accuracy: 0.8778\n", + "Epoch 17/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.1796 - accuracy: 0.9362 - val_loss: 0.4971 - val_accuracy: 0.8846\n", + "Epoch 18/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.1682 - accuracy: 0.9414 - val_loss: 0.5270 - val_accuracy: 0.8832\n", + "Epoch 19/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.1587 - accuracy: 0.9450 - val_loss: 0.5135 - val_accuracy: 0.8902\n", + "Epoch 20/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.1439 - accuracy: 0.9486 - val_loss: 0.4861 - val_accuracy: 0.8860\n", + "Epoch 21/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.1377 - accuracy: 0.9510 - val_loss: 0.5442 - val_accuracy: 0.8870\n", + "Epoch 22/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.1350 - accuracy: 0.9546 - val_loss: 0.5270 - val_accuracy: 0.8890\n", + "Epoch 23/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.1185 - accuracy: 0.9589 - val_loss: 0.5708 - val_accuracy: 0.8870\n", + "Epoch 24/25\n", + "1719/1719 [==============================] - 4s 3ms/step - loss: 0.1078 - accuracy: 0.9637 - val_loss: 0.6175 - val_accuracy: 0.8864\n", + "Epoch 25/25\n", + "1719/1719 [==============================] - 4s 3ms/step - loss: 0.1110 - accuracy: 0.9623 - val_loss: 0.6164 - val_accuracy: 0.8856\n" + ], + "name": "stdout" + } + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "o06AUsZjLOpW", + "outputId": "d9bf73fb-2b6d-4c02-eabd-61d2384be458", + "colab": { + "base_uri": "https://localhost:8080/", + "height": 303 + } + }, + "source": [ + "plt.plot(history.epoch, history.history[\"lr\"], \"o-\")\n", + "plt.axis([0, n_epochs - 1, 0, 0.011])\n", + "plt.xlabel(\"Epoch\")\n", + "plt.ylabel(\"Learning Rate\")\n", + "plt.title(\"Exponential Scheduling\", fontsize=14)\n", + "plt.grid(True)\n", + "plt.show()" + ], + "execution_count": 82, + "outputs": [ + { + "output_type": "display_data", + "data": { + "image/png": "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\n", + "text/plain": [ + "
" + ] + }, + "metadata": { + "tags": [], + "needs_background": "light" + } + } + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "y4leAQjZLOpW" + }, + "source": [ + "이 스케줄 함수는 두 번째 매개변수로 현재 학습률을 받을 수 있습니다:" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "S5QPkNZmLOpW" + }, + "source": [ + "def exponential_decay_fn(epoch, lr):\n", + " return lr * 0.1**(1 / 20)" + ], + "execution_count": 83, + "outputs": [] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "2MB6gpmtLOpX" + }, + "source": [ + "에포크가 아니라 반복마다 학습률을 업데이트하려면 사용자 정의 콜백 클래스를 작성해야 합니다:" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "qpWM_UXaLOpX", + "outputId": "332f517a-6770-4ead-e4f8-ac358bcb3adc", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "K = keras.backend\n", + "\n", + "class ExponentialDecay(keras.callbacks.Callback):\n", + " def __init__(self, s=40000):\n", + " super().__init__()\n", + " self.s = s\n", + "\n", + " def on_batch_begin(self, batch, logs=None):\n", + " # 노트: 에포크마다 `batch` 매개변수가 재설정됩니다\n", + " lr = K.get_value(self.model.optimizer.lr)\n", + " K.set_value(self.model.optimizer.lr, lr * 0.1**(1 / s))\n", + "\n", + " def on_epoch_end(self, epoch, logs=None):\n", + " logs = logs or {}\n", + " logs['lr'] = K.get_value(self.model.optimizer.lr)\n", + "\n", + "model = keras.models.Sequential([\n", + " keras.layers.Flatten(input_shape=[28, 28]),\n", + " keras.layers.Dense(300, activation=\"selu\", kernel_initializer=\"lecun_normal\"),\n", + " keras.layers.Dense(100, activation=\"selu\", kernel_initializer=\"lecun_normal\"),\n", + " keras.layers.Dense(10, activation=\"softmax\")\n", + "])\n", + "lr0 = 0.01\n", + "optimizer = keras.optimizers.Nadam(lr=lr0)\n", + "model.compile(loss=\"sparse_categorical_crossentropy\", optimizer=optimizer, metrics=[\"accuracy\"])\n", + "n_epochs = 25\n", + "\n", + "s = 20 * len(X_train) // 32 # 20 에포크 동안 스텝 횟수 (배치 크기 = 32)\n", + "exp_decay = ExponentialDecay(s)\n", + "history = model.fit(X_train_scaled, y_train, epochs=n_epochs,\n", + " validation_data=(X_valid_scaled, y_valid),\n", + " callbacks=[exp_decay])" + ], + "execution_count": 84, + "outputs": [ + { + "output_type": "stream", + "text": [ + "Epoch 1/25\n", + "1719/1719 [==============================] - 6s 3ms/step - loss: 1.1115 - accuracy: 0.7381 - val_loss: 0.7662 - val_accuracy: 0.7760\n", + "Epoch 2/25\n", + "1719/1719 [==============================] - 6s 3ms/step - loss: 0.6374 - accuracy: 0.7985 - val_loss: 0.5625 - val_accuracy: 0.8226\n", + "Epoch 3/25\n", + "1719/1719 [==============================] - 6s 3ms/step - loss: 0.5862 - accuracy: 0.8178 - val_loss: 0.8932 - val_accuracy: 0.7694\n", + "Epoch 4/25\n", + "1719/1719 [==============================] - 6s 3ms/step - loss: 0.5377 - accuracy: 0.8325 - val_loss: 0.4869 - val_accuracy: 0.8520\n", + "Epoch 5/25\n", + "1719/1719 [==============================] - 6s 3ms/step - loss: 0.4675 - accuracy: 0.8487 - val_loss: 0.4377 - val_accuracy: 0.8692\n", + "Epoch 6/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.4113 - accuracy: 0.8651 - val_loss: 0.4454 - val_accuracy: 0.8648\n", + "Epoch 7/25\n", + "1719/1719 [==============================] - 6s 3ms/step - loss: 0.3786 - accuracy: 0.8751 - val_loss: 0.5048 - val_accuracy: 0.8596\n", + "Epoch 8/25\n", + "1719/1719 [==============================] - 6s 3ms/step - loss: 0.3625 - accuracy: 0.8837 - val_loss: 0.4919 - val_accuracy: 0.8526\n", + "Epoch 9/25\n", + "1719/1719 [==============================] - 6s 3ms/step - loss: 0.3259 - accuracy: 0.8895 - val_loss: 0.4738 - val_accuracy: 0.8604\n", + "Epoch 10/25\n", + "1719/1719 [==============================] - 6s 3ms/step - loss: 0.3111 - accuracy: 0.8963 - val_loss: 0.4203 - val_accuracy: 0.8876\n", + "Epoch 11/25\n", + "1719/1719 [==============================] - 6s 3ms/step - loss: 0.2740 - accuracy: 0.9079 - val_loss: 0.4345 - val_accuracy: 0.8852\n", + "Epoch 12/25\n", + "1719/1719 [==============================] - 6s 3ms/step - loss: 0.2592 - accuracy: 0.9117 - val_loss: 0.4351 - val_accuracy: 0.8820\n", + "Epoch 13/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2383 - accuracy: 0.9200 - val_loss: 0.4287 - val_accuracy: 0.8882\n", + "Epoch 14/25\n", + "1719/1719 [==============================] - 6s 3ms/step - loss: 0.2309 - accuracy: 0.9236 - val_loss: 0.4414 - val_accuracy: 0.8838\n", + "Epoch 15/25\n", + "1719/1719 [==============================] - 6s 3ms/step - loss: 0.2098 - accuracy: 0.9290 - val_loss: 0.4232 - val_accuracy: 0.8916\n", + "Epoch 16/25\n", + "1719/1719 [==============================] - 6s 3ms/step - loss: 0.1936 - accuracy: 0.9342 - val_loss: 0.4336 - val_accuracy: 0.8896\n", + "Epoch 17/25\n", + "1719/1719 [==============================] - 6s 3ms/step - loss: 0.1782 - accuracy: 0.9392 - val_loss: 0.4625 - val_accuracy: 0.8890\n", + "Epoch 18/25\n", + "1719/1719 [==============================] - 6s 3ms/step - loss: 0.1644 - accuracy: 0.9456 - val_loss: 0.4693 - val_accuracy: 0.8904\n", + "Epoch 19/25\n", + "1719/1719 [==============================] - 6s 3ms/step - loss: 0.1554 - accuracy: 0.9496 - val_loss: 0.4725 - val_accuracy: 0.8982\n", + "Epoch 20/25\n", + "1719/1719 [==============================] - 6s 3ms/step - loss: 0.1481 - accuracy: 0.9515 - val_loss: 0.5076 - val_accuracy: 0.8902\n", + "Epoch 21/25\n", + "1719/1719 [==============================] - 6s 3ms/step - loss: 0.1368 - accuracy: 0.9555 - val_loss: 0.5094 - val_accuracy: 0.8940\n", + "Epoch 22/25\n", + "1719/1719 [==============================] - 6s 3ms/step - loss: 0.1278 - accuracy: 0.9575 - val_loss: 0.5544 - val_accuracy: 0.8934\n", + "Epoch 23/25\n", + "1719/1719 [==============================] - 6s 3ms/step - loss: 0.1194 - accuracy: 0.9619 - val_loss: 0.5734 - val_accuracy: 0.8902\n", + "Epoch 24/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.1135 - accuracy: 0.9647 - val_loss: 0.5960 - val_accuracy: 0.8928\n", + "Epoch 25/25\n", + "1719/1719 [==============================] - 6s 3ms/step - loss: 0.1080 - accuracy: 0.9659 - val_loss: 0.6002 - val_accuracy: 0.8896\n" + ], + "name": "stdout" + } + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "xsajfS10LOpX" + }, + "source": [ + "n_steps = n_epochs * len(X_train) // 32\n", + "steps = np.arange(n_steps)\n", + "lrs = lr0 * 0.1**(steps / s)" + ], + "execution_count": 85, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "scrolled": true, + "id": "nn7VunvILOpX", + "outputId": "a0f14eed-c080-434a-9f35-20e29c6d8745", + "colab": { + "base_uri": "https://localhost:8080/", + "height": 303 + } + }, + "source": [ + "plt.plot(steps, lrs, \"-\", linewidth=2)\n", + "plt.axis([0, n_steps - 1, 0, lr0 * 1.1])\n", + "plt.xlabel(\"Batch\")\n", + "plt.ylabel(\"Learning Rate\")\n", + "plt.title(\"Exponential Scheduling (per batch)\", fontsize=14)\n", + "plt.grid(True)\n", + "plt.show()" + ], + "execution_count": 86, + "outputs": [ + { + "output_type": "display_data", + "data": { + "image/png": "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\n", + "text/plain": [ + "
" + ] + }, + "metadata": { + "tags": [], + "needs_background": "light" + } + } + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "t_Zq20ztLOpX" + }, + "source": [ + "### 기간별 고정 스케줄링" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "up9WMAqWLOpX" + }, + "source": [ + "def piecewise_constant_fn(epoch):\n", + " if epoch < 5:\n", + " return 0.01\n", + " elif epoch < 15:\n", + " return 0.005\n", + " else:\n", + " return 0.001" + ], + "execution_count": 87, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "8isT8EI0LOpY" + }, + "source": [ + "def piecewise_constant(boundaries, values):\n", + " boundaries = np.array([0] + boundaries)\n", + " values = np.array(values)\n", + " def piecewise_constant_fn(epoch):\n", + " return values[np.argmax(boundaries > epoch) - 1]\n", + " return piecewise_constant_fn\n", + "\n", + "piecewise_constant_fn = piecewise_constant([5, 15], [0.01, 0.005, 0.001])" + ], + "execution_count": 88, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "-1w63apOLOpY", + "outputId": "d2623190-28dd-4400-a52d-4ad907c3d199", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "lr_scheduler = keras.callbacks.LearningRateScheduler(piecewise_constant_fn)\n", + "\n", + "model = keras.models.Sequential([\n", + " keras.layers.Flatten(input_shape=[28, 28]),\n", + " keras.layers.Dense(300, activation=\"selu\", kernel_initializer=\"lecun_normal\"),\n", + " keras.layers.Dense(100, activation=\"selu\", kernel_initializer=\"lecun_normal\"),\n", + " keras.layers.Dense(10, activation=\"softmax\")\n", + "])\n", + "model.compile(loss=\"sparse_categorical_crossentropy\", optimizer=\"nadam\", metrics=[\"accuracy\"])\n", + "n_epochs = 25\n", + "history = model.fit(X_train_scaled, y_train, epochs=n_epochs,\n", + " validation_data=(X_valid_scaled, y_valid),\n", + " callbacks=[lr_scheduler])" + ], + "execution_count": 89, + "outputs": [ + { + "output_type": "stream", + "text": [ + "Epoch 1/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 1.2318 - accuracy: 0.7142 - val_loss: 0.9157 - val_accuracy: 0.7232\n", + "Epoch 2/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.7203 - accuracy: 0.7776 - val_loss: 0.6123 - val_accuracy: 0.8198\n", + "Epoch 3/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.8270 - accuracy: 0.7619 - val_loss: 1.4205 - val_accuracy: 0.6206\n", + "Epoch 4/25\n", + "1719/1719 [==============================] - 4s 3ms/step - loss: 0.8446 - accuracy: 0.7524 - val_loss: 0.9360 - val_accuracy: 0.7016\n", + "Epoch 5/25\n", + "1719/1719 [==============================] - 4s 3ms/step - loss: 0.8689 - accuracy: 0.7226 - val_loss: 0.8600 - val_accuracy: 0.7660\n", + "Epoch 6/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.6233 - accuracy: 0.7984 - val_loss: 0.6819 - val_accuracy: 0.8138\n", + "Epoch 7/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.5443 - accuracy: 0.8315 - val_loss: 0.5935 - val_accuracy: 0.8408\n", + "Epoch 8/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.5136 - accuracy: 0.8388 - val_loss: 0.7235 - val_accuracy: 0.7882\n", + "Epoch 9/25\n", + "1719/1719 [==============================] - 4s 3ms/step - loss: 0.5320 - accuracy: 0.8341 - val_loss: 0.6815 - val_accuracy: 0.7836\n", + "Epoch 10/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.4771 - accuracy: 0.8513 - val_loss: 0.6984 - val_accuracy: 0.8196\n", + "Epoch 11/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.4934 - accuracy: 0.8493 - val_loss: 0.6040 - val_accuracy: 0.8466\n", + "Epoch 12/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.4599 - accuracy: 0.8600 - val_loss: 0.6230 - val_accuracy: 0.8420\n", + "Epoch 13/25\n", + "1719/1719 [==============================] - 4s 3ms/step - loss: 0.4432 - accuracy: 0.8611 - val_loss: 0.5845 - val_accuracy: 0.8510\n", + "Epoch 14/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.4706 - accuracy: 0.8597 - val_loss: 0.6285 - val_accuracy: 0.8418\n", + "Epoch 15/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.4539 - accuracy: 0.8629 - val_loss: 0.6618 - val_accuracy: 0.8432\n", + "Epoch 16/25\n", + "1719/1719 [==============================] - 4s 3ms/step - loss: 0.3486 - accuracy: 0.8865 - val_loss: 0.4722 - val_accuracy: 0.8680\n", + "Epoch 17/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2977 - accuracy: 0.9014 - val_loss: 0.4846 - val_accuracy: 0.8630\n", + "Epoch 18/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2881 - accuracy: 0.9052 - val_loss: 0.5084 - val_accuracy: 0.8698\n", + "Epoch 19/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2758 - accuracy: 0.9102 - val_loss: 0.4696 - val_accuracy: 0.8700\n", + "Epoch 20/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2721 - accuracy: 0.9082 - val_loss: 0.4798 - val_accuracy: 0.8736\n", + "Epoch 21/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2666 - accuracy: 0.9129 - val_loss: 0.5165 - val_accuracy: 0.8674\n", + "Epoch 22/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2562 - accuracy: 0.9160 - val_loss: 0.5196 - val_accuracy: 0.8728\n", + "Epoch 23/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2504 - accuracy: 0.9179 - val_loss: 0.5501 - val_accuracy: 0.8680\n", + "Epoch 24/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2419 - accuracy: 0.9202 - val_loss: 0.6129 - val_accuracy: 0.8692\n", + "Epoch 25/25\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.2459 - accuracy: 0.9209 - val_loss: 0.5537 - val_accuracy: 0.8682\n" + ], + "name": "stdout" + } + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "iP46umHULOpY", + "outputId": "039e04fb-2e51-4fc0-c056-db18aa65606b", + "colab": { + "base_uri": "https://localhost:8080/", + "height": 303 + } + }, + "source": [ + "plt.plot(history.epoch, [piecewise_constant_fn(epoch) for epoch in history.epoch], \"o-\")\n", + "plt.axis([0, n_epochs - 1, 0, 0.011])\n", + "plt.xlabel(\"Epoch\")\n", + "plt.ylabel(\"Learning Rate\")\n", + "plt.title(\"Piecewise Constant Scheduling\", fontsize=14)\n", + "plt.grid(True)\n", + "plt.show()" + ], + "execution_count": 90, + "outputs": [ + { + "output_type": "display_data", + "data": { + "image/png": "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\n", + "text/plain": [ + "
" + ] + }, + "metadata": { + "tags": [], + "needs_background": "light" + } + } + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "PcBjd75fLOpY" + }, + "source": [ + "### 성능 기반 스케줄링" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "JBL86sCNLOpY" + }, + "source": [ + "tf.random.set_seed(42)\n", + "np.random.seed(42)" + ], + "execution_count": 91, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "e_Bc-HBcLOpY", + "outputId": "2867d0a7-dc95-4878-e655-7e0934e01bc7", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "lr_scheduler = keras.callbacks.ReduceLROnPlateau(factor=0.5, patience=5)\n", + "\n", + "model = keras.models.Sequential([\n", + " keras.layers.Flatten(input_shape=[28, 28]),\n", + " keras.layers.Dense(300, activation=\"selu\", kernel_initializer=\"lecun_normal\"),\n", + " keras.layers.Dense(100, activation=\"selu\", kernel_initializer=\"lecun_normal\"),\n", + " keras.layers.Dense(10, activation=\"softmax\")\n", + "])\n", + "optimizer = keras.optimizers.SGD(lr=0.02, momentum=0.9)\n", + "model.compile(loss=\"sparse_categorical_crossentropy\", optimizer=optimizer, metrics=[\"accuracy\"])\n", + "n_epochs = 25\n", + "history = model.fit(X_train_scaled, y_train, epochs=n_epochs,\n", + " validation_data=(X_valid_scaled, y_valid),\n", + " callbacks=[lr_scheduler])" + ], + "execution_count": 92, + "outputs": [ + { + "output_type": "stream", + "text": [ + "Epoch 1/25\n", + "1719/1719 [==============================] - 4s 2ms/step - loss: 0.7100 - accuracy: 0.7765 - val_loss: 0.4821 - val_accuracy: 0.8490\n", + "Epoch 2/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.4904 - accuracy: 0.8387 - val_loss: 0.5661 - val_accuracy: 0.8390\n", + "Epoch 3/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.4964 - accuracy: 0.8431 - val_loss: 0.5486 - val_accuracy: 0.8502\n", + "Epoch 4/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.5018 - accuracy: 0.8470 - val_loss: 0.4818 - val_accuracy: 0.8566\n", + "Epoch 5/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.5284 - accuracy: 0.8433 - val_loss: 0.5047 - val_accuracy: 0.8502\n", + "Epoch 6/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.5061 - accuracy: 0.8549 - val_loss: 0.5721 - val_accuracy: 0.8520\n", + "Epoch 7/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.5315 - accuracy: 0.8530 - val_loss: 0.5131 - val_accuracy: 0.8626\n", + "Epoch 8/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.5037 - accuracy: 0.8578 - val_loss: 0.6451 - val_accuracy: 0.8064\n", + "Epoch 9/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.4976 - accuracy: 0.8588 - val_loss: 0.5515 - val_accuracy: 0.8620\n", + "Epoch 10/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.3127 - accuracy: 0.8938 - val_loss: 0.4293 - val_accuracy: 0.8776\n", + "Epoch 11/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2376 - accuracy: 0.9126 - val_loss: 0.4315 - val_accuracy: 0.8856\n", + "Epoch 12/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2253 - accuracy: 0.9182 - val_loss: 0.4620 - val_accuracy: 0.8760\n", + "Epoch 13/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2032 - accuracy: 0.9240 - val_loss: 0.4370 - val_accuracy: 0.8890\n", + "Epoch 14/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.1862 - accuracy: 0.9286 - val_loss: 0.5031 - val_accuracy: 0.8702\n", + "Epoch 15/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.1882 - accuracy: 0.9294 - val_loss: 0.4409 - val_accuracy: 0.8862\n", + "Epoch 16/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.1423 - accuracy: 0.9449 - val_loss: 0.4379 - val_accuracy: 0.8930\n", + "Epoch 17/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.1136 - accuracy: 0.9557 - val_loss: 0.4688 - val_accuracy: 0.8924\n", + "Epoch 18/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.1068 - accuracy: 0.9582 - val_loss: 0.4734 - val_accuracy: 0.8876\n", + "Epoch 19/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.1023 - accuracy: 0.9604 - val_loss: 0.4930 - val_accuracy: 0.8938\n", + "Epoch 20/25\n", + "1719/1719 [==============================] - 4s 2ms/step - loss: 0.0986 - accuracy: 0.9611 - val_loss: 0.5015 - val_accuracy: 0.8932\n", + "Epoch 21/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.0803 - accuracy: 0.9684 - val_loss: 0.5080 - val_accuracy: 0.8966\n", + "Epoch 22/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.0751 - accuracy: 0.9712 - val_loss: 0.5266 - val_accuracy: 0.8964\n", + "Epoch 23/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.0700 - accuracy: 0.9742 - val_loss: 0.5284 - val_accuracy: 0.8946\n", + "Epoch 24/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.0654 - accuracy: 0.9759 - val_loss: 0.5423 - val_accuracy: 0.8932\n", + "Epoch 25/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.0632 - accuracy: 0.9770 - val_loss: 0.5530 - val_accuracy: 0.8970\n" + ], + "name": "stdout" + } + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "8pgrsYWWLOpZ", + "outputId": "011acd77-c574-43e2-f8a3-bfa865f7f146", + "colab": { + "base_uri": "https://localhost:8080/", + "height": 303 + } + }, + "source": [ + "plt.plot(history.epoch, history.history[\"lr\"], \"bo-\")\n", + "plt.xlabel(\"Epoch\")\n", + "plt.ylabel(\"Learning Rate\", color='b')\n", + "plt.tick_params('y', colors='b')\n", + "plt.gca().set_xlim(0, n_epochs - 1)\n", + "plt.grid(True)\n", + "\n", + "ax2 = plt.gca().twinx()\n", + "ax2.plot(history.epoch, history.history[\"val_loss\"], \"r^-\")\n", + "ax2.set_ylabel('Validation Loss', color='r')\n", + "ax2.tick_params('y', colors='r')\n", + "\n", + "plt.title(\"Reduce LR on Plateau\", fontsize=14)\n", + "plt.show()" + ], + "execution_count": 93, + "outputs": [ + { + "output_type": "display_data", + "data": { + "image/png": "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\n", + "text/plain": [ + "
" + ] + }, + "metadata": { + "tags": [], + "needs_background": "light" + } + } + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "HezOhTgrLOpZ" + }, + "source": [ + "### tf.keras 스케줄러" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "mM6gXVz4LOpZ", + "outputId": "4af8e052-d12e-464a-9c5a-922ae9842c13", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "model = keras.models.Sequential([\n", + " keras.layers.Flatten(input_shape=[28, 28]),\n", + " keras.layers.Dense(300, activation=\"selu\", kernel_initializer=\"lecun_normal\"),\n", + " keras.layers.Dense(100, activation=\"selu\", kernel_initializer=\"lecun_normal\"),\n", + " keras.layers.Dense(10, activation=\"softmax\")\n", + "])\n", + "s = 20 * len(X_train) // 32 # number of steps in 20 epochs (batch size = 32)\n", + "learning_rate = keras.optimizers.schedules.ExponentialDecay(0.01, s, 0.1)\n", + "optimizer = keras.optimizers.SGD(learning_rate)\n", + "model.compile(loss=\"sparse_categorical_crossentropy\", optimizer=optimizer, metrics=[\"accuracy\"])\n", + "n_epochs = 25\n", + "history = model.fit(X_train_scaled, y_train, epochs=n_epochs,\n", + " validation_data=(X_valid_scaled, y_valid))" + ], + "execution_count": 94, + "outputs": [ + { + "output_type": "stream", + "text": [ + "Epoch 1/25\n", + "1719/1719 [==============================] - 4s 2ms/step - loss: 0.5995 - accuracy: 0.7923 - val_loss: 0.4092 - val_accuracy: 0.8604\n", + "Epoch 2/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.3890 - accuracy: 0.8613 - val_loss: 0.3737 - val_accuracy: 0.8694\n", + "Epoch 3/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.3531 - accuracy: 0.8774 - val_loss: 0.3731 - val_accuracy: 0.8692\n", + "Epoch 4/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.3296 - accuracy: 0.8813 - val_loss: 0.3492 - val_accuracy: 0.8800\n", + "Epoch 5/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.3176 - accuracy: 0.8866 - val_loss: 0.3430 - val_accuracy: 0.8792\n", + "Epoch 6/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2928 - accuracy: 0.8954 - val_loss: 0.3414 - val_accuracy: 0.8812\n", + "Epoch 7/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2852 - accuracy: 0.8985 - val_loss: 0.3356 - val_accuracy: 0.8816\n", + "Epoch 8/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2713 - accuracy: 0.9039 - val_loss: 0.3365 - val_accuracy: 0.8814\n", + "Epoch 9/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2713 - accuracy: 0.9044 - val_loss: 0.3266 - val_accuracy: 0.8860\n", + "Epoch 10/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2569 - accuracy: 0.9084 - val_loss: 0.3240 - val_accuracy: 0.8848\n", + "Epoch 11/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2500 - accuracy: 0.9115 - val_loss: 0.3252 - val_accuracy: 0.8866\n", + "Epoch 12/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2452 - accuracy: 0.9147 - val_loss: 0.3302 - val_accuracy: 0.8812\n", + "Epoch 13/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2408 - accuracy: 0.9155 - val_loss: 0.3219 - val_accuracy: 0.8858\n", + "Epoch 14/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2377 - accuracy: 0.9159 - val_loss: 0.3223 - val_accuracy: 0.8864\n", + "Epoch 15/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2377 - accuracy: 0.9171 - val_loss: 0.3209 - val_accuracy: 0.8878\n", + "Epoch 16/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2316 - accuracy: 0.9192 - val_loss: 0.3185 - val_accuracy: 0.8896\n", + "Epoch 17/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2264 - accuracy: 0.9213 - val_loss: 0.3198 - val_accuracy: 0.8886\n", + "Epoch 18/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2283 - accuracy: 0.9188 - val_loss: 0.3169 - val_accuracy: 0.8902\n", + "Epoch 19/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2283 - accuracy: 0.9203 - val_loss: 0.3198 - val_accuracy: 0.8894\n", + "Epoch 20/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2287 - accuracy: 0.9218 - val_loss: 0.3170 - val_accuracy: 0.8902\n", + "Epoch 21/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2264 - accuracy: 0.9209 - val_loss: 0.3180 - val_accuracy: 0.8904\n", + "Epoch 22/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2256 - accuracy: 0.9200 - val_loss: 0.3164 - val_accuracy: 0.8914\n", + "Epoch 23/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2222 - accuracy: 0.9233 - val_loss: 0.3171 - val_accuracy: 0.8902\n", + "Epoch 24/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2180 - accuracy: 0.9241 - val_loss: 0.3166 - val_accuracy: 0.8898\n", + "Epoch 25/25\n", + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.2221 - accuracy: 0.9234 - val_loss: 0.3165 - val_accuracy: 0.8914\n" + ], + "name": "stdout" + } + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "e7s1E4aILOpZ" + }, + "source": [ + "구간별 고정 스케줄링은 다음을 사용하세요:" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "mB1XJs34LOpa" + }, + "source": [ + "learning_rate = keras.optimizers.schedules.PiecewiseConstantDecay(\n", + " boundaries=[5. * n_steps_per_epoch, 15. * n_steps_per_epoch],\n", + " values=[0.01, 0.005, 0.001])" + ], + "execution_count": 95, + "outputs": [] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "-QgJ_wlTLOpa" + }, + "source": [ + "### 1사이클 스케줄링" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "v2ZxFRyHLOpa" + }, + "source": [ + "K = keras.backend\n", + "\n", + "class ExponentialLearningRate(keras.callbacks.Callback):\n", + " def __init__(self, factor):\n", + " self.factor = factor\n", + " self.rates = []\n", + " self.losses = []\n", + " def on_batch_end(self, batch, logs):\n", + " self.rates.append(K.get_value(self.model.optimizer.lr))\n", + " self.losses.append(logs[\"loss\"])\n", + " K.set_value(self.model.optimizer.lr, self.model.optimizer.lr * self.factor)\n", + "\n", + "def find_learning_rate(model, X, y, epochs=1, batch_size=32, min_rate=10**-5, max_rate=10):\n", + " init_weights = model.get_weights()\n", + " iterations = len(X) // batch_size * epochs\n", + " factor = np.exp(np.log(max_rate / min_rate) / iterations)\n", + " init_lr = K.get_value(model.optimizer.lr)\n", + " K.set_value(model.optimizer.lr, min_rate)\n", + " exp_lr = ExponentialLearningRate(factor)\n", + " history = model.fit(X, y, epochs=epochs, batch_size=batch_size,\n", + " callbacks=[exp_lr])\n", + " K.set_value(model.optimizer.lr, init_lr)\n", + " model.set_weights(init_weights)\n", + " return exp_lr.rates, exp_lr.losses\n", + "\n", + "def plot_lr_vs_loss(rates, losses):\n", + " plt.plot(rates, losses)\n", + " plt.gca().set_xscale('log')\n", + " plt.hlines(min(losses), min(rates), max(rates))\n", + " plt.axis([min(rates), max(rates), min(losses), (losses[0] + min(losses)) / 2])\n", + " plt.xlabel(\"Learning rate\")\n", + " plt.ylabel(\"Loss\")" + ], + "execution_count": 96, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "tmWjq-nILOpa" + }, + "source": [ + "tf.random.set_seed(42)\n", + "np.random.seed(42)\n", + "\n", + "model = keras.models.Sequential([\n", + " keras.layers.Flatten(input_shape=[28, 28]),\n", + " keras.layers.Dense(300, activation=\"selu\", kernel_initializer=\"lecun_normal\"),\n", + " keras.layers.Dense(100, activation=\"selu\", kernel_initializer=\"lecun_normal\"),\n", + " keras.layers.Dense(10, activation=\"softmax\")\n", + "])\n", + "model.compile(loss=\"sparse_categorical_crossentropy\",\n", + " optimizer=keras.optimizers.SGD(lr=1e-3),\n", + " metrics=[\"accuracy\"])" + ], + "execution_count": 97, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "N7dpX40rLOpa", + "outputId": "fcea5e32-513b-4269-960e-939e0f398353", + "colab": { + "base_uri": "https://localhost:8080/", + "height": 345 + } + }, + "source": [ + "batch_size = 128\n", + "rates, losses = find_learning_rate(model, X_train_scaled, y_train, epochs=1, batch_size=batch_size)\n", + "plot_lr_vs_loss(rates, losses)" + ], + "execution_count": 98, + "outputs": [ + { + "output_type": "stream", + "text": [ + " 1/430 [..............................] - ETA: 1:35 - loss: 2.6624 - accuracy: 0.1406WARNING:tensorflow:Callback method `on_train_batch_end` is slow compared to the batch time (batch time: 0.0016s vs `on_train_batch_end` time: 0.0017s). Check your callbacks.\n", + "430/430 [==============================] - 1s 3ms/step - loss: nan - accuracy: 0.3135\n" + ], + "name": "stdout" + }, + { + "output_type": "display_data", + "data": { + "image/png": "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\n", + "text/plain": [ + "
" + ] + }, + "metadata": { + "tags": [], + "needs_background": "light" + } + } + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "20u-WHSyLOpc" + }, + "source": [ + "class OneCycleScheduler(keras.callbacks.Callback):\n", + " def __init__(self, iterations, max_rate, start_rate=None,\n", + " last_iterations=None, last_rate=None):\n", + " self.iterations = iterations\n", + " self.max_rate = max_rate\n", + " self.start_rate = start_rate or max_rate / 10\n", + " self.last_iterations = last_iterations or iterations // 10 + 1\n", + " self.half_iteration = (iterations - self.last_iterations) // 2\n", + " self.last_rate = last_rate or self.start_rate / 1000\n", + " self.iteration = 0\n", + " def _interpolate(self, iter1, iter2, rate1, rate2):\n", + " return ((rate2 - rate1) * (self.iteration - iter1)\n", + " / (iter2 - iter1) + rate1)\n", + " def on_batch_begin(self, batch, logs):\n", + " if self.iteration < self.half_iteration:\n", + " rate = self._interpolate(0, self.half_iteration, self.start_rate, self.max_rate)\n", + " elif self.iteration < 2 * self.half_iteration:\n", + " rate = self._interpolate(self.half_iteration, 2 * self.half_iteration,\n", + " self.max_rate, self.start_rate)\n", + " else:\n", + " rate = self._interpolate(2 * self.half_iteration, self.iterations,\n", + " self.start_rate, self.last_rate)\n", + " rate = max(rate, self.last_rate)\n", + " self.iteration += 1\n", + " K.set_value(self.model.optimizer.lr, rate)" + ], + "execution_count": 99, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "3T9DaJx8LOpc", + "outputId": "8bc95003-5f2e-43bc-f8f0-3b729625f0a0", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "n_epochs = 25\n", + "onecycle = OneCycleScheduler(len(X_train) // batch_size * n_epochs, max_rate=0.05)\n", + "history = model.fit(X_train_scaled, y_train, epochs=n_epochs, batch_size=batch_size,\n", + " validation_data=(X_valid_scaled, y_valid),\n", + " callbacks=[onecycle])" + ], + "execution_count": 100, + "outputs": [ + { + "output_type": "stream", + "text": [ + "Epoch 1/25\n", + "430/430 [==============================] - 1s 3ms/step - loss: 0.6572 - accuracy: 0.7739 - val_loss: 0.4871 - val_accuracy: 0.8336\n", + "Epoch 2/25\n", + "430/430 [==============================] - 1s 3ms/step - loss: 0.4581 - accuracy: 0.8396 - val_loss: 0.4274 - val_accuracy: 0.8526\n", + "Epoch 3/25\n", + "430/430 [==============================] - 1s 2ms/step - loss: 0.4121 - accuracy: 0.8546 - val_loss: 0.4114 - val_accuracy: 0.8584\n", + "Epoch 4/25\n", + "430/430 [==============================] - 1s 3ms/step - loss: 0.3837 - accuracy: 0.8641 - val_loss: 0.3871 - val_accuracy: 0.8688\n", + "Epoch 5/25\n", + "430/430 [==============================] - 1s 3ms/step - loss: 0.3639 - accuracy: 0.8717 - val_loss: 0.3765 - val_accuracy: 0.8680\n", + "Epoch 6/25\n", + "430/430 [==============================] - 1s 2ms/step - loss: 0.3456 - accuracy: 0.8774 - val_loss: 0.3744 - val_accuracy: 0.8706\n", + "Epoch 7/25\n", + "430/430 [==============================] - 1s 3ms/step - loss: 0.3329 - accuracy: 0.8809 - val_loss: 0.3634 - val_accuracy: 0.8706\n", + "Epoch 8/25\n", + "430/430 [==============================] - 1s 3ms/step - loss: 0.3184 - accuracy: 0.8858 - val_loss: 0.3949 - val_accuracy: 0.8612\n", + "Epoch 9/25\n", + "430/430 [==============================] - 1s 3ms/step - loss: 0.3065 - accuracy: 0.8891 - val_loss: 0.3487 - val_accuracy: 0.8772\n", + "Epoch 10/25\n", + "430/430 [==============================] - 1s 2ms/step - loss: 0.2944 - accuracy: 0.8922 - val_loss: 0.3398 - val_accuracy: 0.8808\n", + "Epoch 11/25\n", + "430/430 [==============================] - 1s 3ms/step - loss: 0.2839 - accuracy: 0.8960 - val_loss: 0.3456 - val_accuracy: 0.8820\n", + "Epoch 12/25\n", + "430/430 [==============================] - 1s 3ms/step - loss: 0.2707 - accuracy: 0.9026 - val_loss: 0.3652 - val_accuracy: 0.8694\n", + "Epoch 13/25\n", + "430/430 [==============================] - 1s 2ms/step - loss: 0.2537 - accuracy: 0.9081 - val_loss: 0.3364 - val_accuracy: 0.8830\n", + "Epoch 14/25\n", + "430/430 [==============================] - 1s 2ms/step - loss: 0.2403 - accuracy: 0.9137 - val_loss: 0.3464 - val_accuracy: 0.8808\n", + "Epoch 15/25\n", + "430/430 [==============================] - 1s 3ms/step - loss: 0.2279 - accuracy: 0.9181 - val_loss: 0.3261 - val_accuracy: 0.8848\n", + "Epoch 16/25\n", + "430/430 [==============================] - 1s 3ms/step - loss: 0.2160 - accuracy: 0.9232 - val_loss: 0.3297 - val_accuracy: 0.8844\n", + "Epoch 17/25\n", + "430/430 [==============================] - 1s 3ms/step - loss: 0.2062 - accuracy: 0.9269 - val_loss: 0.3357 - val_accuracy: 0.8862\n", + "Epoch 18/25\n", + "430/430 [==============================] - 1s 2ms/step - loss: 0.1979 - accuracy: 0.9306 - val_loss: 0.3251 - val_accuracy: 0.8894\n", + "Epoch 19/25\n", + "430/430 [==============================] - 1s 3ms/step - loss: 0.1893 - accuracy: 0.9340 - val_loss: 0.3234 - val_accuracy: 0.8908\n", + "Epoch 20/25\n", + "430/430 [==============================] - 1s 3ms/step - loss: 0.1821 - accuracy: 0.9365 - val_loss: 0.3227 - val_accuracy: 0.8932\n", + "Epoch 21/25\n", + "430/430 [==============================] - 1s 3ms/step - loss: 0.1753 - accuracy: 0.9403 - val_loss: 0.3223 - val_accuracy: 0.8918\n", + "Epoch 22/25\n", + "430/430 [==============================] - 1s 3ms/step - loss: 0.1701 - accuracy: 0.9417 - val_loss: 0.3187 - val_accuracy: 0.8946\n", + "Epoch 23/25\n", + "430/430 [==============================] - 1s 3ms/step - loss: 0.1656 - accuracy: 0.9438 - val_loss: 0.3191 - val_accuracy: 0.8940\n", + "Epoch 24/25\n", + "430/430 [==============================] - 1s 3ms/step - loss: 0.1628 - accuracy: 0.9457 - val_loss: 0.3181 - val_accuracy: 0.8934\n", + "Epoch 25/25\n", + "430/430 [==============================] - 1s 3ms/step - loss: 0.1611 - accuracy: 0.9461 - val_loss: 0.3174 - val_accuracy: 0.8942\n" + ], + "name": "stdout" + } + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "i8jPCyOvLOpc" + }, + "source": [ + "# 규제를 사용해 과대적합 피하기" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "bNDKKZZxLOpc" + }, + "source": [ + "## $\\ell_1$과 $\\ell_2$ 규제" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "58lwkbU-LOpc" + }, + "source": [ + "layer = keras.layers.Dense(100, activation=\"elu\",\n", + " kernel_initializer=\"he_normal\",\n", + " kernel_regularizer=keras.regularizers.l2(0.01))\n", + "# or l1(0.1) for ℓ1 regularization with a factor or 0.1\n", + "# or l1_l2(0.1, 0.01) for both ℓ1 and ℓ2 regularization, with factors 0.1 and 0.01 respectively" + ], + "execution_count": 101, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "vcZsRVXwLOpd", + "outputId": "acd092e5-9485-48bc-824d-dac90b171290", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "model = keras.models.Sequential([\n", + " keras.layers.Flatten(input_shape=[28, 28]),\n", + " keras.layers.Dense(300, activation=\"elu\",\n", + " kernel_initializer=\"he_normal\",\n", + " kernel_regularizer=keras.regularizers.l2(0.01)),\n", + " keras.layers.Dense(100, activation=\"elu\",\n", + " kernel_initializer=\"he_normal\",\n", + " kernel_regularizer=keras.regularizers.l2(0.01)),\n", + " keras.layers.Dense(10, activation=\"softmax\",\n", + " kernel_regularizer=keras.regularizers.l2(0.01))\n", + "])\n", + "model.compile(loss=\"sparse_categorical_crossentropy\", optimizer=\"nadam\", metrics=[\"accuracy\"])\n", + "n_epochs = 2\n", + "history = model.fit(X_train_scaled, y_train, epochs=n_epochs,\n", + " validation_data=(X_valid_scaled, y_valid))" + ], + "execution_count": 102, + "outputs": [ + { + "output_type": "stream", + "text": [ + "Epoch 1/2\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 3.2189 - accuracy: 0.7967 - val_loss: 0.7169 - val_accuracy: 0.8340\n", + "Epoch 2/2\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.7280 - accuracy: 0.8247 - val_loss: 0.6850 - val_accuracy: 0.8376\n" + ], + "name": "stdout" + } + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "XuNxc8x3LOpd", + "outputId": "baae0e04-7717-402f-eb20-9c6bc906d8a7", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "from functools import partial\n", + "\n", + "RegularizedDense = partial(keras.layers.Dense,\n", + " activation=\"elu\",\n", + " kernel_initializer=\"he_normal\",\n", + " kernel_regularizer=keras.regularizers.l2(0.01))\n", + "\n", + "model = keras.models.Sequential([\n", + " keras.layers.Flatten(input_shape=[28, 28]),\n", + " RegularizedDense(300),\n", + " RegularizedDense(100),\n", + " RegularizedDense(10, activation=\"softmax\")\n", + "])\n", + "model.compile(loss=\"sparse_categorical_crossentropy\", optimizer=\"nadam\", metrics=[\"accuracy\"])\n", + "n_epochs = 2\n", + "history = model.fit(X_train_scaled, y_train, epochs=n_epochs,\n", + " validation_data=(X_valid_scaled, y_valid))" + ], + "execution_count": 103, + "outputs": [ + { + "output_type": "stream", + "text": [ + "Epoch 1/2\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 3.2911 - accuracy: 0.7924 - val_loss: 0.7218 - val_accuracy: 0.8310\n", + "Epoch 2/2\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.7282 - accuracy: 0.8245 - val_loss: 0.6826 - val_accuracy: 0.8382\n" + ], + "name": "stdout" + } + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "xdTO1zZbLOpd" + }, + "source": [ + "## 드롭아웃" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "ppZX7nrcLOpd", + "outputId": "78c2ea5f-c378-48ab-d877-9bad958ec21b", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "model = keras.models.Sequential([\n", + " keras.layers.Flatten(input_shape=[28, 28]),\n", + " keras.layers.Dropout(rate=0.2),\n", + " keras.layers.Dense(300, activation=\"elu\", kernel_initializer=\"he_normal\"),\n", + " keras.layers.Dropout(rate=0.2),\n", + " keras.layers.Dense(100, activation=\"elu\", kernel_initializer=\"he_normal\"),\n", + " keras.layers.Dropout(rate=0.2),\n", + " keras.layers.Dense(10, activation=\"softmax\")\n", + "])\n", + "model.compile(loss=\"sparse_categorical_crossentropy\", optimizer=\"nadam\", metrics=[\"accuracy\"])\n", + "n_epochs = 2\n", + "history = model.fit(X_train_scaled, y_train, epochs=n_epochs,\n", + " validation_data=(X_valid_scaled, y_valid))" + ], + "execution_count": 104, + "outputs": [ + { + "output_type": "stream", + "text": [ + "Epoch 1/2\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.7611 - accuracy: 0.7576 - val_loss: 0.3730 - val_accuracy: 0.8644\n", + "Epoch 2/2\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.4306 - accuracy: 0.8403 - val_loss: 0.3408 - val_accuracy: 0.8726\n" + ], + "name": "stdout" + } + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "FT_W0H3rLOpd" + }, + "source": [ + "## 알파 드롭아웃" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "BPvHOOYcLOpd" + }, + "source": [ + "tf.random.set_seed(42)\n", + "np.random.seed(42)" + ], + "execution_count": 105, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "CB_Oe24YLOpe", + "outputId": "15e5a530-ca81-4bca-ac0a-cc2184ea6bd3", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "model = keras.models.Sequential([\n", + " keras.layers.Flatten(input_shape=[28, 28]),\n", + " keras.layers.AlphaDropout(rate=0.2),\n", + " keras.layers.Dense(300, activation=\"selu\", kernel_initializer=\"lecun_normal\"),\n", + " keras.layers.AlphaDropout(rate=0.2),\n", + " keras.layers.Dense(100, activation=\"selu\", kernel_initializer=\"lecun_normal\"),\n", + " keras.layers.AlphaDropout(rate=0.2),\n", + " keras.layers.Dense(10, activation=\"softmax\")\n", + "])\n", + "optimizer = keras.optimizers.SGD(lr=0.01, momentum=0.9, nesterov=True)\n", + "model.compile(loss=\"sparse_categorical_crossentropy\", optimizer=optimizer, metrics=[\"accuracy\"])\n", + "n_epochs = 20\n", + "history = model.fit(X_train_scaled, y_train, epochs=n_epochs,\n", + " validation_data=(X_valid_scaled, y_valid))" + ], + "execution_count": 106, + "outputs": [ + { + "output_type": "stream", + "text": [ + "Epoch 1/20\n", + "1719/1719 [==============================] - 4s 2ms/step - loss: 0.8023 - accuracy: 0.7146 - val_loss: 0.5778 - val_accuracy: 0.8446\n", + "Epoch 2/20\n", + "1719/1719 [==============================] - 4s 2ms/step - loss: 0.5662 - accuracy: 0.7904 - val_loss: 0.5146 - val_accuracy: 0.8536\n", + "Epoch 3/20\n", + "1719/1719 [==============================] - 4s 2ms/step - loss: 0.5259 - accuracy: 0.8057 - val_loss: 0.4904 - val_accuracy: 0.8598\n", + "Epoch 4/20\n", + "1719/1719 [==============================] - 4s 2ms/step - loss: 0.5127 - accuracy: 0.8093 - val_loss: 0.4837 - val_accuracy: 0.8596\n", + "Epoch 5/20\n", + "1719/1719 [==============================] - 4s 2ms/step - loss: 0.5074 - accuracy: 0.8123 - val_loss: 0.4244 - val_accuracy: 0.8692\n", + "Epoch 6/20\n", + "1719/1719 [==============================] - 4s 2ms/step - loss: 0.4787 - accuracy: 0.8205 - val_loss: 0.4594 - val_accuracy: 0.8640\n", + "Epoch 7/20\n", + "1719/1719 [==============================] - 4s 2ms/step - loss: 0.4721 - accuracy: 0.8267 - val_loss: 0.4696 - val_accuracy: 0.8608\n", + "Epoch 8/20\n", + "1719/1719 [==============================] - 4s 2ms/step - loss: 0.4575 - accuracy: 0.8306 - val_loss: 0.4156 - val_accuracy: 0.8708\n", + "Epoch 9/20\n", + "1719/1719 [==============================] - 4s 2ms/step - loss: 0.4626 - accuracy: 0.8275 - val_loss: 0.4342 - val_accuracy: 0.8734\n", + "Epoch 10/20\n", + "1719/1719 [==============================] - 4s 2ms/step - loss: 0.4550 - accuracy: 0.8323 - val_loss: 0.4300 - val_accuracy: 0.8656\n", + "Epoch 11/20\n", + "1719/1719 [==============================] - 4s 2ms/step - loss: 0.4453 - accuracy: 0.8340 - val_loss: 0.4266 - val_accuracy: 0.8746\n", + "Epoch 12/20\n", + "1719/1719 [==============================] - 4s 2ms/step - loss: 0.4420 - accuracy: 0.8349 - val_loss: 0.5200 - val_accuracy: 0.8574\n", + "Epoch 13/20\n", + "1719/1719 [==============================] - 4s 2ms/step - loss: 0.4344 - accuracy: 0.8393 - val_loss: 0.4301 - val_accuracy: 0.8740\n", + "Epoch 14/20\n", + "1719/1719 [==============================] - 4s 2ms/step - loss: 0.4307 - accuracy: 0.8389 - val_loss: 0.4462 - val_accuracy: 0.8656\n", + "Epoch 15/20\n", + "1719/1719 [==============================] - 4s 2ms/step - loss: 0.4313 - accuracy: 0.8374 - val_loss: 0.4415 - val_accuracy: 0.8658\n", + "Epoch 16/20\n", + "1719/1719 [==============================] - 4s 2ms/step - loss: 0.4270 - accuracy: 0.8391 - val_loss: 0.4181 - val_accuracy: 0.8776\n", + "Epoch 17/20\n", + "1719/1719 [==============================] - 4s 2ms/step - loss: 0.4207 - accuracy: 0.8426 - val_loss: 0.5417 - val_accuracy: 0.8564\n", + "Epoch 18/20\n", + "1719/1719 [==============================] - 4s 2ms/step - loss: 0.4362 - accuracy: 0.8378 - val_loss: 0.4669 - val_accuracy: 0.8718\n", + "Epoch 19/20\n", + "1719/1719 [==============================] - 4s 2ms/step - loss: 0.4271 - accuracy: 0.8418 - val_loss: 0.4745 - val_accuracy: 0.8700\n", + "Epoch 20/20\n", + "1719/1719 [==============================] - 4s 2ms/step - loss: 0.4188 - accuracy: 0.8415 - val_loss: 0.4486 - val_accuracy: 0.8714\n" + ], + "name": "stdout" + } + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "fisM8Wc8LOpe", + "outputId": "8783b0b2-0a9c-427d-a745-517443a53217", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "model.evaluate(X_test_scaled, y_test)" + ], + "execution_count": 107, + "outputs": [ + { + "output_type": "stream", + "text": [ + "313/313 [==============================] - 0s 2ms/step - loss: 0.4861 - accuracy: 0.8584\n" + ], + "name": "stdout" + }, + { + "output_type": "execute_result", + "data": { + "text/plain": [ + "[0.48614442348480225, 0.8583999872207642]" + ] + }, + "metadata": { + "tags": [] + }, + "execution_count": 107 + } + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "C3UO369ULOpe", + "outputId": "bb158abb-93c6-4eef-acee-b55d70e8419e", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "model.evaluate(X_train_scaled, y_train)" + ], + "execution_count": 108, + "outputs": [ + { + "output_type": "stream", + "text": [ + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.3614 - accuracy: 0.8815\n" + ], + "name": "stdout" + }, + { + "output_type": "execute_result", + "data": { + "text/plain": [ + "[0.3614204227924347, 0.8815454840660095]" + ] + }, + "metadata": { + "tags": [] + }, + "execution_count": 108 + } + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "TLppurUFLOpe", + "outputId": "4cf53ffb-1221-410a-9652-833b0a85e6c6", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "history = model.fit(X_train_scaled, y_train)" + ], + "execution_count": 109, + "outputs": [ + { + "output_type": "stream", + "text": [ + "1719/1719 [==============================] - 3s 2ms/step - loss: 0.4217 - accuracy: 0.8434\n" + ], + "name": "stdout" + } + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "TP8X2xCnLOpe" + }, + "source": [ + "## MC 드롭아웃" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "Uy_uwuIRLOpe" + }, + "source": [ + "tf.random.set_seed(42)\n", + "np.random.seed(42)" + ], + "execution_count": 110, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "scrolled": true, + "id": "GPDfhAMBLOpf" + }, + "source": [ + "y_probas = np.stack([model(X_test_scaled, training=True)\n", + " for sample in range(100)])\n", + "y_proba = y_probas.mean(axis=0)\n", + "y_std = y_probas.std(axis=0)" + ], + "execution_count": 111, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "nzrxJDyrLOpf", + "outputId": "3b272a1c-75a1-4d81-cfbb-e53079e2bc12", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "np.round(model.predict(X_test_scaled[:1]), 2)" + ], + "execution_count": 112, + "outputs": [ + { + "output_type": "execute_result", + "data": { + "text/plain": [ + "array([[0. , 0. , 0. , 0. , 0. , 0. , 0. , 0.02, 0. , 0.98]],\n", + " dtype=float32)" + ] + }, + "metadata": { + "tags": [] + }, + "execution_count": 112 + } + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "08HJBSaVLOpf", + "outputId": "12a1a91f-04da-42bd-8b09-8c7713f29918", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "np.round(y_probas[:, :1], 2)" + ], + "execution_count": 113, + "outputs": [ + { + "output_type": "execute_result", + "data": { + "text/plain": [ + "array([[[0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.73, 0. , 0.26]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.05, 0. , 0.92, 0. , 0.04]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.03, 0. , 0. , 0. , 0.97]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.1 , 0. , 0.89]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0. , 0. , 0.53, 0. , 0.46]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0. , 0. , 0.71, 0. , 0.29]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.09, 0. , 0.47, 0. , 0.44]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.18, 0. , 0.26, 0. , 0.56]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.55, 0. , 0.07, 0. , 0.37]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.99]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.02, 0. , 0.14, 0. , 0.84]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.21, 0. , 0.22, 0. , 0.57]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.3 , 0. , 0.68]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.29, 0. , 0.28, 0. , 0.44]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.04, 0. , 0.14, 0. , 0.82]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.21, 0. , 0.12, 0. , 0.68]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0. , 0. , 0.99]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.02, 0. , 0.77, 0. , 0.21]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.34, 0. , 0.65]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.02, 0. , 0.02, 0. , 0.96]],\n", + "\n", + " [[0.01, 0. , 0. , 0. , 0. , 0.86, 0.02, 0.01, 0. , 0.1 ]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.12, 0. , 0.07, 0. , 0.81]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0. , 0. , 0. , 0. , 1. ]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0. , 0. , 0.11, 0. , 0.88]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.08, 0. , 0.06, 0. , 0.86]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.04, 0. , 0.72, 0. , 0.23]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.03, 0. , 0.2 , 0. , 0.78]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.19, 0. , 0.31, 0. , 0.5 ]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.03, 0. , 0.05, 0. , 0.92]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.2 , 0. , 0.64, 0. , 0.16]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0. , 0. , 0. , 0. , 0.99]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0. , 0. , 0. , 0. , 1. ]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.68, 0. , 0.02, 0. , 0.3 ]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.15, 0. , 0.74, 0. , 0.11]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.02, 0. , 0.19, 0. , 0.79]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.09, 0. , 0.63, 0. , 0.29]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0. , 0. , 0.04, 0. , 0.95]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.02, 0. , 0.09, 0. , 0.89]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.07, 0. , 0.38, 0. , 0.56]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.78, 0. , 0.02, 0. , 0.2 ]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.16, 0. , 0.82]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.99]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.33, 0. , 0.37, 0. , 0.3 ]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.99]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.01, 0. , 0.98]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.99]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.17, 0. , 0.12, 0. , 0.71]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0. , 0. , 0.03, 0. , 0.96]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0. , 0. , 0.29, 0. , 0.71]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.2 , 0. , 0.79]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.27, 0. , 0.29, 0. , 0.45]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.89, 0. , 0.1 ]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.99]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.2 , 0. , 0.04, 0. , 0.76]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.13, 0. , 0.02, 0. , 0.85]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.1 , 0. , 0.89]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0. , 0. , 0.16, 0. , 0.84]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.04, 0. , 0.16, 0. , 0.8 ]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.07, 0. , 0.25, 0. , 0.68]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.14, 0. , 0.2 , 0. , 0.66]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.22, 0. , 0.03, 0. , 0.75]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0. , 0. , 0.13, 0. , 0.87]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.04, 0. , 0.05, 0. , 0.91]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.04, 0. , 0.35, 0. , 0.61]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.49, 0. , 0.46, 0. , 0.05]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.51, 0. , 0.45, 0. , 0.04]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.1 , 0. , 0.47, 0. , 0.43]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0. , 0. , 0.05, 0. , 0.95]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.05, 0. , 0.37, 0. , 0.57]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.15, 0. , 0.61, 0.02, 0.22]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.99]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.08, 0. , 0.28, 0. , 0.64]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.02, 0. , 0.1 , 0. , 0.88]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.02, 0. , 0.37, 0. , 0.61]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.1 , 0. , 0.52, 0. , 0.37]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.12, 0. , 0.03, 0. , 0.85]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.05, 0. , 0.15, 0. , 0.8 ]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.38, 0. , 0.61]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.39, 0. , 0.6 ]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.02, 0. , 0.24, 0. , 0.74]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0. , 0. , 0.06, 0. , 0.94]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.01, 0. , 0.98]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0. , 0. , 0. , 0. , 1. ]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.02, 0. , 0.97]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.04, 0. , 0.09, 0. , 0.87]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.03, 0. , 0.55, 0. , 0.41]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.92, 0. , 0.04, 0. , 0.04]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.4 , 0. , 0.02, 0. , 0.58]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0. , 0. , 0.02, 0. , 0.98]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0. , 0. , 0.03, 0. , 0.97]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.02, 0. , 0.04, 0. , 0.93]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.03, 0. , 0.12, 0. , 0.84]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.28, 0. , 0.38, 0. , 0.35]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.39, 0. , 0.17, 0. , 0.44]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.02, 0. , 0.16, 0. , 0.82]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.01, 0. , 0.98]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.02, 0. , 0.35, 0. , 0.64]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.01, 0. , 0.04, 0. , 0.96]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.2 , 0. , 0.33, 0. , 0.47]],\n", + "\n", + " [[0. , 0. , 0. , 0. , 0. , 0.11, 0. , 0.11, 0. , 0.78]]],\n", + " dtype=float32)" + ] + }, + "metadata": { + "tags": [] + }, + "execution_count": 113 + } + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "XMqmf4UmLOpf", + "outputId": "69d29b09-945c-441e-c2f7-9c3b8e3c564c", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "np.round(y_proba[:1], 2)" + ], + "execution_count": 114, + "outputs": [ + { + "output_type": "execute_result", + "data": { + "text/plain": [ + "array([[0. , 0. , 0. , 0. , 0. , 0.11, 0. , 0.22, 0. , 0.67]],\n", + " dtype=float32)" + ] + }, + "metadata": { + "tags": [] + }, + "execution_count": 114 + } + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "lISHnMdnLOpf", + "outputId": "9d1137d4-bbd6-4f9e-a527-66cf9daef495", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "y_std = y_probas.std(axis=0)\n", + "np.round(y_std[:1], 2)" + ], + "execution_count": 115, + "outputs": [ + { + "output_type": "execute_result", + "data": { + "text/plain": [ + "array([[0. , 0. , 0. , 0. , 0. , 0.19, 0. , 0.23, 0. , 0.29]],\n", + " dtype=float32)" + ] + }, + "metadata": { + "tags": [] + }, + "execution_count": 115 + } + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "iW-062pHLOpf" + }, + "source": [ + "y_pred = np.argmax(y_proba, axis=1)" + ], + "execution_count": 116, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "DL6bPgKCLOpg", + "outputId": "cf362713-825c-4238-c802-cac884209a40", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "accuracy = np.sum(y_pred == y_test) / len(y_test)\n", + "accuracy" + ], + "execution_count": 117, + "outputs": [ + { + "output_type": "execute_result", + "data": { + "text/plain": [ + "0.8656" + ] + }, + "metadata": { + "tags": [] + }, + "execution_count": 117 + } + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "yxvL3LWZLOpg" + }, + "source": [ + "class MCDropout(keras.layers.Dropout):\n", + " def call(self, inputs):\n", + " return super().call(inputs, training=True)\n", + "\n", + "class MCAlphaDropout(keras.layers.AlphaDropout):\n", + " def call(self, inputs):\n", + " return super().call(inputs, training=True)" + ], + "execution_count": 118, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "Zzd5DCm7LOpg" + }, + "source": [ + "tf.random.set_seed(42)\n", + "np.random.seed(42)" + ], + "execution_count": 119, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "WvapJbsLLOpg" + }, + "source": [ + "mc_model = keras.models.Sequential([\n", + " MCAlphaDropout(layer.rate) if isinstance(layer, keras.layers.AlphaDropout) else layer\n", + " for layer in model.layers\n", + "])" + ], + "execution_count": 120, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "tkkDpmTaLOpg", + "outputId": "7378308e-f78f-446b-83fe-bb6e9a998a62", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "mc_model.summary()" + ], + "execution_count": 121, + "outputs": [ + { + "output_type": "stream", + "text": [ + "Model: \"sequential_20\"\n", + "_________________________________________________________________\n", + "Layer (type) Output Shape Param # \n", + "=================================================================\n", + "flatten_18 (Flatten) (None, 784) 0 \n", + "_________________________________________________________________\n", + "mc_alpha_dropout (MCAlphaDro (None, 784) 0 \n", + "_________________________________________________________________\n", + "dense_262 (Dense) (None, 300) 235500 \n", + "_________________________________________________________________\n", + "mc_alpha_dropout_1 (MCAlphaD (None, 300) 0 \n", + "_________________________________________________________________\n", + "dense_263 (Dense) (None, 100) 30100 \n", + "_________________________________________________________________\n", + "mc_alpha_dropout_2 (MCAlphaD (None, 100) 0 \n", + "_________________________________________________________________\n", + "dense_264 (Dense) (None, 10) 1010 \n", + "=================================================================\n", + "Total params: 266,610\n", + "Trainable params: 266,610\n", + "Non-trainable params: 0\n", + "_________________________________________________________________\n" + ], + "name": "stdout" + } + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "dXbWp13vLOpg" + }, + "source": [ + "optimizer = keras.optimizers.SGD(lr=0.01, momentum=0.9, nesterov=True)\n", + "mc_model.compile(loss=\"sparse_categorical_crossentropy\", optimizer=optimizer, metrics=[\"accuracy\"])" + ], + "execution_count": 122, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "9ywhSDViLOpg" + }, + "source": [ + "mc_model.set_weights(model.get_weights())" + ], + "execution_count": 123, + "outputs": [] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "JZd8UdsALOph" + }, + "source": [ + "이제 MC 드롭아웃을 모델에 사용할 수 있습니다:" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "IVSdXF7TLOph", + "outputId": "a580e136-de40-46e7-e200-c730000cff26", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "np.round(np.mean([mc_model.predict(X_test_scaled[:1]) for sample in range(100)], axis=0), 2)" + ], + "execution_count": 124, + "outputs": [ + { + "output_type": "execute_result", + "data": { + "text/plain": [ + "array([[0. , 0. , 0. , 0. , 0. , 0.14, 0. , 0.25, 0. , 0.61]],\n", + " dtype=float32)" + ] + }, + "metadata": { + "tags": [] + }, + "execution_count": 124 + } + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "MafcuHfYLOph" + }, + "source": [ + "## 맥스 노름" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "Rm22pePYLOph" + }, + "source": [ + "layer = keras.layers.Dense(100, activation=\"selu\", kernel_initializer=\"lecun_normal\",\n", + " kernel_constraint=keras.constraints.max_norm(1.))" + ], + "execution_count": 125, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "gWqSW1bvLOpi", + "outputId": "d2224c69-2eb1-4a62-e14d-34dfb1360660", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "MaxNormDense = partial(keras.layers.Dense,\n", + " activation=\"selu\", kernel_initializer=\"lecun_normal\",\n", + " kernel_constraint=keras.constraints.max_norm(1.))\n", + "\n", + "model = keras.models.Sequential([\n", + " keras.layers.Flatten(input_shape=[28, 28]),\n", + " MaxNormDense(300),\n", + " MaxNormDense(100),\n", + " keras.layers.Dense(10, activation=\"softmax\")\n", + "])\n", + "model.compile(loss=\"sparse_categorical_crossentropy\", optimizer=\"nadam\", metrics=[\"accuracy\"])\n", + "n_epochs = 2\n", + "history = model.fit(X_train_scaled, y_train, epochs=n_epochs,\n", + " validation_data=(X_valid_scaled, y_valid))" + ], + "execution_count": 126, + "outputs": [ + { + "output_type": "stream", + "text": [ + "Epoch 1/2\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.5766 - accuracy: 0.8018 - val_loss: 0.3723 - val_accuracy: 0.8638\n", + "Epoch 2/2\n", + "1719/1719 [==============================] - 5s 3ms/step - loss: 0.3530 - accuracy: 0.8692 - val_loss: 0.3769 - val_accuracy: 0.8684\n" + ], + "name": "stdout" + } + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "A53qrK-PLOpi" + }, + "source": [ + "# 연습문제 해답" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "IOsQc6lpLOpi" + }, + "source": [ + "## 1. to 7." + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "ebf2enC_LOpi" + }, + "source": [ + "부록 A 참조." + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "h1PVxctOLOpi" + }, + "source": [ + "## 8. CIFAR10에서 딥러닝" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "qGwtSO0fLOpi" + }, + "source": [ + "### a.\n", + "*문제: 100개의 뉴런을 가진 은닉층 20개로 심층 신경망을 만들어보세요(너무 많은 것 같지만 이 연습문제의 핵심입니다). He 초기화와 ELU 활성화 함수를 사용하세요.*" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "p1Mix75VLOpj" + }, + "source": [ + "keras.backend.clear_session()\n", + "tf.random.set_seed(42)\n", + "np.random.seed(42)\n", + "\n", + "model = keras.models.Sequential()\n", + "model.add(keras.layers.Flatten(input_shape=[32, 32, 3]))\n", + "for _ in range(20):\n", + " model.add(keras.layers.Dense(100,\n", + " activation=\"elu\",\n", + " kernel_initializer=\"he_normal\"))" + ], + "execution_count": 127, + "outputs": [] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "2glrTwRcLOpj" + }, + "source": [ + "### b.\n", + "*문제: Nadam 옵티마이저와 조기 종료를 사용하여 CIFAR10 데이터셋에 이 네트워크를 훈련하세요. `keras.datasets.cifar10.load_ data()`를 사용하여 데이터를 적재할 수 있습니다. 이 데이터셋은 10개의 클래스와 32×32 크기의 컬러 이미지 60,000개로 구성됩니다(50,000개는 훈련, 10,000개는 테스트). 따라서 10개의 뉴런과 소프트맥스 활성화 함수를 사용하는 출력층이 필요합니다. 모델 구조와 하이퍼파라미터를 바꿀 때마다 적절한 학습률을 찾아야 한다는 것을 기억하세요.*" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "8fkKGlD0LOpj" + }, + "source": [ + "모델에 출력층을 추가합니다:" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "pvTdbgiCLOpj" + }, + "source": [ + "model.add(keras.layers.Dense(10, activation=\"softmax\"))" + ], + "execution_count": 128, + "outputs": [] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "ux0ROP0tLOpk" + }, + "source": [ + "학습률 5e-5인 Nadam 옵티마이저를 사용해 보죠. 학습률 1e-5, 3e-5, 1e-4, 3e-4, 1e-3, 3e-3, 1e-2를 테스트하고 10번의 에포크 동안 (아래 텐서보드 콜백으로) 학습 곡선을 비교해 보았습니다. 학습률 3e-5와 1e-4가 꽤 좋았기 때문에 5e-5를 시도해 보았고 조금 더 나은 결과를 냈습니다." + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "1oReiGeCLOpk" + }, + "source": [ + "optimizer = keras.optimizers.Nadam(lr=5e-5)\n", + "model.compile(loss=\"sparse_categorical_crossentropy\",\n", + " optimizer=optimizer,\n", + " metrics=[\"accuracy\"])" + ], + "execution_count": 129, + "outputs": [] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "24PcKfxeLOpk" + }, + "source": [ + "CIFAR10 데이터셋을 로드하죠. 조기 종료를 사용하기 때문에 검증 세트가 필요합니다. 원본 훈련 세트에서 처음 5,000개를 검증 세트로 사용하겠습니다:" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "cLWnS6RKLOpk", + "outputId": "8b69d659-ad10-4a5d-a6be-6484e343b258", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "(X_train_full, y_train_full), (X_test, y_test) = keras.datasets.cifar10.load_data()\n", + "\n", + "X_train = X_train_full[5000:]\n", + "y_train = y_train_full[5000:]\n", + "X_valid = X_train_full[:5000]\n", + "y_valid = y_train_full[:5000]" + ], + "execution_count": 130, + "outputs": [ + { + "output_type": "stream", + "text": [ + "Downloading data from https://www.cs.toronto.edu/~kriz/cifar-10-python.tar.gz\n", + "170500096/170498071 [==============================] - 3s 0us/step\n" + ], + "name": "stdout" + } + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "H76ZTjNxLOpk" + }, + "source": [ + "이제 콜백을 만들고 모델을 훈련합니다:" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "FuQ8FelNLOpk" + }, + "source": [ + "early_stopping_cb = keras.callbacks.EarlyStopping(patience=20)\n", + "model_checkpoint_cb = keras.callbacks.ModelCheckpoint(\"my_cifar10_model.h5\", save_best_only=True)\n", + "run_index = 1 # 모델을 훈련할 때마다 증가시킴\n", + "run_logdir = os.path.join(os.curdir, \"my_cifar10_logs\", \"run_{:03d}\".format(run_index))\n", + "tensorboard_cb = keras.callbacks.TensorBoard(run_logdir)\n", + "callbacks = [early_stopping_cb, model_checkpoint_cb, tensorboard_cb]" + ], + "execution_count": 131, + "outputs": [] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "7VuUEbNacg7v" + }, + "source": [ + "코랩에서 아래 셀에 있는 `%tensorboard` 명령을 실행하면 다음과 같은 텐서보드 화면을 볼 수 있습니다." + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "x_NtuiwVcQiV" + }, + "source": [ + "![스크린샷 2021-02-17 오후 11.42.41.png](data:image/png;base64,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EEhuAQJDyX19ODoEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBOIqQGAorpxsDAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQACB5BYgMJTc14ejQwABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAgrgIEhuLKycYQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEhuAQJDyX19ODoEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBOIqQGAorpxsDAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQACB5BYgMJTc14ejQwABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAgrgIEhuLKycYQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEhuAQJDyX19ODoEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBOIqQGAorpxsDAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQACB5BYgMJTc14ejQwABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAgrgIEhuLKycYQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEhuAQJDyX19ODoEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBOIqQGAorpxsDAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQACB5BYgMJTc14ejQwABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAgrgIEhuLKycYQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEhuAQJDyX19ODoEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBOIqQGAorpxsDAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQACB5BYgMJTc14ejQwABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAgrgIEhuLKycYQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEhuAQJDyX19ODoEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBOIqQGAorpxsDAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQACB5BYgMJTc14ejQwABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAgrgIEhuLKycYQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEhuAQJDyX19ODoEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBOIqQGAorpxsDAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQACB5BYgMJTc14ejQwABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAgrgIEhuLKycYQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEhuAQJDyX19ODoEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBOIqQGAorpxsDAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQACB5BYgMJTc14ejQwABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAgrgIEhuLKycYQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEhuAQJDyX19ODoEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBOIqQGAorpxsDAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQACB5BYgMJTc14ejQwABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAgrgIEhuLKycYQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQ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AQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAIFUFSAwlKpXnvNGAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQSEkBAkMpedk5aQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAIFUFSAwlKpXnvNGAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQSEkBAkMpedk5aQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAIFUFSAwlKpXnvNGAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQSEkBAkMpedk5aQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAIFUFSAwlKpXnvNGAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQSEkBAkMpedk5aQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAIFUFSAwlKpXnvNGAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQSEkBAkMpedk5aQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAIFUFSAwlKpXnvNGAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQSEkBAkMpedk5aQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAIFUFSAwlKpXnvNGAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQSEkBAkMpedk5aQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAIFUFSAwlKpXnvNGAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQSEkBAkMpedk5aQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAIFUFSAwlKpXnvNGAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEAAAQQQSEkBAkMpedk5aQQQQAABBBBAAAEEEEAAAQQQQAABBBBA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AgggMF0FCAxN1yvLeSGAAAIIIIAAAggggAACCCCAAAIIIIAAAggggAACU0OAwNDUuE4cJQIIIIAAAghMIwECQ9PoYnIqCCCAAAIIIIAAAggggMD/s3ceYFJUWRs+rgi6gCL6g4IiKC5pF0HBgAFzTgRzds1iRDEgYsCIgXXNIoI550UUE2IkGBABA6ASFFcFRUGMP+/dPb13iuqe7pkemPCd55mp7gq3br1Vfavqnu+eIwIiIAIiIAIiIAIiIAIiIAIiUAUJSDBUBU+aqiwCIiACIiACIlC1CUgwVLXPn2ovAiIgAiIgAiIgAiIgAiIgAiIgAiIgAiIgAiIgAiIgAiIgAlWdgARDVf0Mqv4iIAIiIAIiIAJVjoAEQ1XulKnCIiACIiACIiACIiACIiACIiACIiACIiACIiACIiACIiACIlCtCEgwVK1Opw5GBERABERABESgKhCQYKgqnCXVUQREQAREQAREQAREQAREQAREQAREQAREQAREQAREQAREQASqLwEJhqrvudWRiYAIiIAIiIAIVFICEgxV0hOjaomACIiACIiACIiACIiACIiACIiACIiACIiACIiACIiACIhADSEgwVANOdE6TBEQAREQAREQgcpDQIKhynMuVBMREAEREAEREAEREAEREAEREAEREAEREAEREAEREAEREAERqIkEJBiqiWddxywCIiACIiACIrBMCUgwtEzxa+ciIAIiIAIiIAIiIAIiIAIiIAIiIAIiIAIiIAIiIAIiIAIiUOMJSDBU4y8BARABERABERABEVjaBCQYWtrEtT8REAEREAEREAEREAEREAEREAEREAEREAEREAEREAEREAEREIGYgARDMQ19FgEREAEREAEREIGlQECCoaUAWbsQAREQAREQAREQAREQAREQAREQAREQAREQAREQAREQARGoNgR+/PFHGz16tG222Wa2yiqrZI7rhx9+sFdffdW23357q1WrVmZ+rg+LFi2yd99912bPnm0dOnSwFi1a5Fq92i6TYKjanlodmAiIgAiIgAiIQGUlIMFQZT0zqpcIiIAIiIAIiIAIiIAIiIAIiIAIiIAIiIAIiIAIiIAIiEBlJPD555/b3//+d9t6662tb9++mSpOnz7djjnmGHviiScsH//LrFmz7KSTTrL111/fGjdubG+//ba1adPGzj33XFtuueUy5daEDxIM1YSzrGMUAREQAREQARGoVATyeWCtVBVWZURABERABERABERABERABERABERABERABERABERABERABERgGRJwwdDyyy9v559/vnXp0iXUJptg6LfffrPff//dVlhhhUytf/311yAu6tq1qx122GFhPpGLTjjhBNtjjz2sZ8+emXX5QCSi2rVrV1shkQRDJU63voiACIiACIiACIhAxROQYKjiGWsPIiACIiACIiACIiACIiACIiACIiACIiACIiACIiACIiAC1YeAC4YOOOAAe/bZZ+3222+3evXqWVIw9Msvv9jVV19tb731lvG5U6dO1rt3b6tfv75NnTrVjjvuOBs2bJg1adIkA+fGG2+0jz/+2K699lobNWqU3XfffbbaaquFtGUIlI499ljbbbfdMutXlw8SDFWXM6njEAEREAEREAERqDIEJBiqMqdKFRUBERABERABERABERABERABERABERABERABERABERABEagEBFww9OCDD9oFF1xgzZo1C0KgpGAIsdDs2bPDMqo9aNAgW2WVVUIas2eeecbuv//+IBiKD+mNN96wyy67LKQ1e/nll+2qq66yAQMGWLt27cI8BEaPPvpoiDYUb1fVP0swVNXPoOovAiIgAiIgAiJQ5QhIMFTlTpkqLAIiIAIiIAIiIAIiIAIiIAIiIAIiIAIiIAIiIAIiIAIisAwJuGDo4Ycftu+++86OP/54u+iii6xhw4YhzdgTTzxhK620ku29995BLLTVVluF2o4ZM8bOO+88e+qpp4Loh+9EEopt2rRpIYoQ6yAeuuuuu2zIkCGZVfbbbz/r06ePbbTRRpl51eGDBEPV4SzqGERABERABERABKoUAQmGqtTpUmVFQAREQAREQAREQAREQAREQAREQAREQAREQAREQAREQASWMYFYMETEINKGDR8+3M455xw75ZRTQiSg+fPn28EHHxyWrb766qHG33//vfXo0cOGDh0a0pddccUV9thjj1mtWrUyR0TkIURC9957r7300ktLCIb69u1rbdu2tYMOOiizTXX4IMFQdTiLOgYREAEREAEREIEqRUCCoSp1ulRZERABERABERABERABERABERABERABERABERABERABERCBZUwgKRj67bffrFevXla3bl177733gmBohRVWsN13390uvPBC23TTTUONiSjEdyIQzZ071w488EAjbVn79u0zR3TJJZfYokWLQsSiNMHQvvvua2eeeaZ17tw5s011+CDBUHU4izoGERABERABERCBKkVAgqEqdbpUWREQAREQAREQAREQAREQAREQAREQAREQAREQAREQAREQgWVMICkYojqffPJJEA0hHkIQhP/l0ksvDSnLevfubcstt5xdc801QVREWjJs0KBB9tlnn9m5555rRCF66623bODAgTZgwABr06ZNiDDENvy1bNnSRowYYddff31IZ1anTp1QRnX5J8FQdTmTOg4REAEREAEREIEqQ0CCoSpzqlRRERABERABERABERABERABERABERABERABERABERABERCBSkAgTTBEtYYMGRJSkLlgiBRkV111lY0fP96IONShQwc7++yzbcUVVwxHgbjoxhtvtFdffdV+/PFHa9asWUg1tvnmm4flRBi6+eabrWnTpvbpp5/azz//bMcee6ztscceYXl1+ifBUHU6mzoWERABERABERCBKkFAgqEqcZpUSREQAREQAREQAREQAREQAREQAREQAREQAREQAREQAREQgSpKgBRjRBiqXbt26hH88ccftnDhwhCVKF4hTkn2ww8/BKFRrVq14lWqzWcJhqrNqdSBiIAIiIAIiIAIVBUCEgxVlTOleoqACIiACIiACIiACIiACIiACIiACIiACIiACIiACIiACNQkArFgqLoftwRD1f0M6/hEQAREQAREQAQqHQEJhirdKVGFREAEREAEREAEREAEREAEREAEREAEREAEREAEREAEREAERMAmTpxoL7/8svXq1ava05BgqNqfYh2gCIiACIiACIhAZSMgwVBlOyOqjwiIgAiIgAiIgAiIgAiIgAiIgAiIgAiIgAiIgAiIgAiIgAjULAISDNWs862jFQEREAEREAERqAQEJBiqBCdBVRABERABERABERABERABERABERABERABERABERABERABERCBGkxAgqEafPJ16CIgAiIgAiIgAsuGgARDy4a79lp1CYwdO9batWtnyd/Oxx9/bHXr1rUmTZoslYMbNWqUde3adYl9TZo0yf7v//4v/C2xsAJmzJk93ebMnmaNm6y7+K9F2MOEcS+EaftO24Vpch3/zkJfx7fxclgH8zLDF/0TAREQAREQAREQAREQAREQAREQAREQAREQAREQgSpA4Jtvvgm1/Oijj0rU1ufHM1dbbbXMV//M1D9nFlbzDxIMVfMTrMMTAREQAREQARGofASSoofKV0PVqKYQOOyww2z77be3Qw45pEyHfP/994d8zgMGDCjT9mz05ptv2plnnhlyQi+//PKp5fz1r3+1hx56yNq0aVNiOTmkERLVq1fPJkyYYAMHDswsnz59ejguck3XqlUrMz/bh3yO5U9/+pP9/vvvSxSx33772T777GM9e/ZcYpnPWLBggfXt29eee+45a9q0qR177LHWo0cPX5zXFEFPnd9m2uRJE8L6DVetbw0b1Ldv5823b+fOD/NatviPeOqT6bNLfPd1fBsWJtfh+9RPZ1v7jbbNiIpCIfonAiIgAiIgAiIgAiIgAiIgAiIgAiIgAiIgAiIgApWQAGIgFwilCYPKUuW//OUvYTOflqWMqrKNBENV5UypniIgAiIgAiIgAtWGgARD1eZUVukD+eyzz2zLLbe0xo0bGxF8ymKIYBYuXFiuURdHHnlkENHccssttttuu6VWozTBENu1b9/evvzyS1txxRVDGZdeeql98sknNmTIkNQykzPzOZbyCIYQFNWvX98uuOACe++996xPnz52ww032LbbbpusSup3xEIjnxpsO23TKXV5sWYiLKq7Wmtba72Ni1WkyhEBERABERABERABERABERABERABERABERABERCBohJAJORCobhgjxCUnMbr+OdYYMTn+Luvg2ioOguHJBjyM62pCIiACIiACIiACCwlAhIMLSXQ2k1OAhdeeGFI8fXoo4/arbfean/729/C+kTm6devn02ZMiWk37rsssusdu3adtNNN9k999xjq6yyivXu3TsIXYj6M3XqVDv77LODWOfoo4+2f//733bMMcfYXXfdZS+99JIxr0WLFvbII49Yo0aN7PLLL7cNNtgg7OuHH36wtm3b2vXXX2/Dhg0L67Dg119/Dft48cUXrVu3bqGs4cOHhwhDgwcPtmuuuca22GILmzVrlu2+++52/PHH23bbbWcnnniide/ePZSNgOif//xnOIabb77Z7rjjjnC8rLvvvvva+PHjQzmrr756EDwRvciPZfTo0XbeeefZTz/9ZLvuuqv1798/lIlgiLJuu+22ECWIurB9HGEIIdZpp50WeBx66KHh+L/66qvA4Ntvv82kVbv66qttzJgx9sADD4R1YPL0008HARf7Xn/99cM+/R+CoWnvDzePIOTzK2KKYOjPq5WM5lQR+1GZIiACIiACIiACIiACIiACIiACIiACIiACIiACIlAIgTShEOIgF/W4UKiQMuN1XYTkU19W2YVDLnYq9PglGPIzrKkIiIAIiIAIiIAILCUCEgwtJdDaTVYCf/zxh7Vs2dJeeeUVe+KJJ+zjjz+2a6+9NqxPmrKuXbuGdF4IcHbaaacg8Nljjz3snXfeCaM2mM+2iGc++OCDIPg566yzbP78+SEt2OOPP26nnHKKff3117bNNtuE6D9XXnllWP/dd98N4h12hojnrbfeCpF21l13XRs3bpz93//9n40YMcJIc/bggw/a3LlzbfPNN7fXX3/d1lprLWvdurU99dRTtt5664V0akQoOuGEE+zOO+8Mghu2oU4IiaZNm2YzZswIAqaHH3441K9z58724Ycfhn0hLkKIhFiHuvixIBKivjAiAtBVV11lXbp0seWWW84QWO2www7h+BBCIe6JBUMIlRBj7bjjjiFNGVwQRiFyeuGFFzLnBJZsx4snjGB+7rnnBpbPPvus8Ze0eTNG2y8Lv07OLvr3z+f8bBttuV/Ry1WBIiACIiACIiACIiACIiACIiACIiACIiACIiACIlBWArFYyEVChQpkCtl3vD+2y1c0NGnSpNDPTV93RRkCIeq32WabhV288cYbIUJSvnX0ekkw5CQ0FQEREAEREAEREIGlRECCoaUEWrvJSgDhCpF+Ro4cGV4iELl8+umntsIKK9gll1wSBDmIgvbcc88QUYioQZtsskmIzEN0ng033DCUTdQhF9kQoei+++4z0ocRSYcXExcMIaDZaqutbOLEiXbQQQeFlFwUQEo06oEgiKhFCIKIztOrV68QeQghEOYpydgXgiD+MNZr165diDD0448/hig+iISuuOIKIxoQ+8WoP9GOJkyYEKIDPf/886GOLCeKERYfC+nJXnvtNeMlC3ESkYJOOumkUObvv/8e1ieN2+mnn25EI3LBUKdOnaxDhw4h7RgrIYbimGB3++232zPPPBO25R8ire23396ISIRgyBn9/PPP1qRJkyB0WmmllTLr82FpCYaefWmcddziwMC2RAX0RQREQAREQAREQAREQAREQAREQAREQAREQAREQASWAYFYvFOoKKa81S10395fTD/6cccdZ88991x5q7DE9vRdw6G8gikJhpZAqxkiIAIiIAIiIAIiULEEJBiqWL4qvXQCiHYQC6266qphZaLw3H333Zl0Xm+//XYQ5ZAua8iQIUHQ8v3339uTTz4ZIvkgLPrXv/5VQmRDCi1EOQhksgmGEPwccMABQbiDYIY0YEQWInIPIh3qg6jn8MMPD5GNWBdzwRACHF6EbrnlljA/Fgwxg+hIRPa54IILQpQiohAR9ahjx45BkMP+KJNoQ9Qxm2CIMogstPXWW4eIPwh+koIh6nnIIYcE8ZO/ACK8QhhFGje35s2bW7NmzYKQiH3CDiOtGTwRJMWCIQRJjDyBT8OGDb0YmzDuBZs5dax17tgqM6+iPowcPcVWWLllOP6KHAVTUfVXuSIgAiIgAiIgAiIgAiIgAiIgAiIgAiIgAiIgAtWLQFkj6BSLgouGEOh4VJ9sZXt/cY8ePWz27NnWtGnTbKvmPd/3T2T9YpoEQ8WkqbJEQAREQAREQAREIA8CEgzlAUmrVBiB7777LqTzIqJQvXr1wn4eeeSRkJLr6aefDqKhSy+9NKT+QlDz66+/hnRcROW5+uqr7aeffrI111zTZs6cGcRDHmEIQQ0CnVNPPTWkDyM6UTLCUCwYIv0W4hn24UaUIlKDTZ8+3R566CG79957g+CHNGQvv/xyENCQGmz8+PEh8tHOO+9se++9d4gwRBlECzrmmGNC/Yj8gyFsuv7660N0nzlz5oT0aKT7gkOaYIhjJ9XYF198Ycsvv3wQIJGOjeMiatF7771n1POiiy4yohoRzchfAHv27BmiLyEG2mijjUJKNPgx0oNUaCwnghJlH3jggXbssceG6EUIhnbbbTc744wzwjH07ds3CKPCAfz335zZ02386AeWimBowe+r2Q+/raYIQ/EJ0GcREAEREAEREAEREAEREAEREAEREAEREAEREIFlQoDo8GuvvXbYN2Kd8kbVKctBkAIM0RLmdfjll1+sT58+IQp927ZtQ38zfcveX0wf8ZlnnhkGsLIdfc+PPvqoMciU7TbeeGMj4jyR7F9//fXQR0w/dNrxIRhiftoyyi6rSTBUVnLaTgREQAREQAREQATKSECCoTKC02ZFIXDzzTfbq6++GiIKeYGLFi0KkYEQw4wZMyakByMtFoIZRDsrr7xySCX2+eefW506dYKoiBeaOI0XeZl32WUXa9SoUYim8/jjj4dUYHH0HBcMvfPOO7bOOuuEiEREJnIbOHBgEAsxZaQE6bpI24VYCTETL128PA0bNiwIeb788kvbZ599zFOX/fHHH+Flq1+/fnbUUUeFYnlpQ4zDSI4NNtjAiJ5EBCCiGaUJhhAXnX322YaIqlWrVsb2RCYaNGiQ1a1bN0Qa8pRmCI+ov78AIghCqHTxxRcHoRPRgqgrgidEQgcffHCIHEQ9qV///v1DHWG0yiqrhPRwRC666667Qjo45+LT6e8Pt3p1FvnXCpl+O2++rb1eZ/vzam1C+aRzU5ShCkGtQkVABERABERABERABERABERABERABERABERABEoh8MknnxgDOgcPHmxz587NiHVK2azoiz3CDwV7lB8Gv44aNSpE6b/99tttypQpYdCt9xe3a9cu9F9PnDjRhg4dGvrk77vvPqOfnf5h+qpPOeUUq127tl1yySV2+eWXh3p7v3HRDyKlQAmGUqBolgiIgAiIgAiIgAhUJAEJhiqSrsouFgGi5yCQiW3hwoVBMESknaR99dVXttJKK1mtWrXslVdeMYRJjz32WHK1gr6zP8pMWrb5yfXi74iiEDvla7nWz2f/pFhL+62z7YorrhjSsHldXFTFyBTSs8EwzX5Z8LX9svDf9vLzT4TFbdq2t1Ub1LfJkybY3MVCH//OZ+ZhXbpsHqbJdfw727Md9vrrrxmCoR32OMoaN2lhCLwQgm29ODWbREMBkf6JgAiIgAiIgAiIgAiIgAiIgAiIgAiIgAiIgAgsRQIMaNxkk01CdPZmzZqFPRPRnb+lZXF0oXjfL730kh199NFhQCuDSb1+aYKh/fffPwy4Peyww0K1ERe1bNky/DHQdPXVVw8DSkeMGGFjx44tcWjsn7+KOGYJhkqg1hcREAEREAEREAERqHgCaSKCit+r9iACFUuAkRFE/9lpp53s3XffDRF5EJrISifggqGtttqq9JUXrzFh3AthvfadtgtT/964ybpB6EP6sjmzp6Wu49v4Or4NKzMPQyyE8TJOKjgiOzEaRiYCIiACIiACIiACIiACIiACIiACIiACIiACIiACS5vA999/b08++aT98MMPIVI++4+FOxVVH0Q6RBZiipEOjEGfsc2YMcMeeuihEGXo0EMPDanG0gRDhxxyiHXr1i1E74+3J9I/acsaNGgQZjPo9cADD4xXCZ9Jh8YxKyXZEmg0QwREQAREQAREQASqFgEJhqrW+VJt8yfw9ddf27Rp04zczKQzk+VHgJC0jD4h9VtlM6Ukq2xnRPURAREQAREQAREQAREQAREQAREQAREQAREQgZpD4JlnnrHnn38+pPr66aefQjSf7t27ZwB41B2fZhaU8YOLg2KhEEWliYUGDBhgLVq0sIMOOiikJjvvvPNs9OjRliYYIho/f6Qmmz9/fkizNnLkSDvnnHOsadOm1rt3byNq/bhx4yzbwFLqRj28jsUQDynCUBkvFG0mAiIgAiIgAiIgAmUlIMFQWclpOxEQgWVFAOEQptRky+oMaL8iIAIiIAIiIAIiIAIiIAIiIAIiIAIiIAIiUPMIEFUIQc7nn39uRN9BLLT33nsbUYe+/PLLJYC4cMjFND5dYsXFM1x4wzIEQlg8j++kDjviiCNSI/uw7IADDrAVV1yRVe3CCy+0HXfcMVUwhNgJQdErr7xi8+bNC5+JSMRx9e3b1z788ENjnf79+1uPHj1Cedn+UUePOOTH6/X379m2Tc6XYChJRN9FQAREQAREQAREoIIJSDBUwYBVvAiIQFEJeGoyxEJKM1dUtCpMBERABERABERABERABERABERABERABERABEQgDwILFy4MgqE//elPmbUnT55sv//+u02fPj0zrxgfEBlRbuvWrYNQ6L333rPDDz+8RNH0k1577bVhHqKmevXqlVie7cuiRYtshRVWsPg4WJfjW2mllbJtlnM+AiIXOkkwlBOVFoqACIiACIiACIjAsicgwdCyPweqgQiIQGEEPvjgA0M4JMFQYdy0tgiIgAiIgAiIgAiIgAiIgAiIgAiIgAiIgAiIwNIh4FF2YgFNPnv2KEQutvHv+Wxb1ddRhKGqfgZVfxEQAREQAREQgSpHQIKhKnfKVGEREAEREAEREAEREAEREAEREAEREAEREAEREAEREAEREIEqSMCj78RVr0mioPi4k58lGEoS0XcREAEREAEREAERqGACEgxVMGAVXyUIjB071tq1a2fJ38PHH39sdevWtSZNmiyV4xg1apR17dp1iX1NmjTJSMHFX2WxObOnW8MG9W3kU4Pt23nzrU3b9uGP+a+9/lqoZveeB4fpy88/EdbZevu9wjaTJ00w/tZo0sK6dNk8rPPow3eHqa/DNrX/vLr9baPtrPHi9dKMKENEG1KkoTQ6micCIiACIiACIiACIiACIiACIiACIiACIiACIiACVYeABENV51yppiIgAiIgAiIgAtWEQFIgUU0OS4dRCQgsWLAgiF/uvPNOa9OmTZlqdMopp9hOO+1ku+66a5m2Z6N//vOfhvDnuuuuy1rGX//6V3vooYeWqGevXr2CkIj805tvvrntt99+mTIefPBBe+WVV+z666/PzMv1IZ9jIVc0+aiTxn732Wcf69mzZ3JR5ju8+/bta88995w1bdrUjj32WOvRo0dmeSEf3n77bRs6dOgSxxyXcfctfa1zx1ZBABTPL+bnT6bPtrqrtbb2nbZLLfbll1/OpCZDTPXrr7/a888/b1tssUVqnu5//etfNn78eDv//PNTyyvrzDfffNPWWGMNa968eVmLWGK7L774wsaMGWNt27a19ddff4nloyqknAAAQABJREFUmiECIiACIiACIiACIiACIiACIiACIiACIiACIiAC1YmABEPV6WzqWERABERABERABKoEAQmGqsRpqpKVRCh07rnn2gEHHGADBw4s0zF89dVXtsoqq1idOnXKtP0ff/wRREALFy60d955xxo2bJhaTmmCofr16xvHgxjHbccdd7SDDz7YDj30UJ+Vc5rPsZRHMISgiHpecMEF9t5771mfPn3shhtusG233TZnvZIL+/XrF0Q13333nR100EF2wgknJFexCeNesJlTxwbB0BILizwDwdCfV8suOEM05BGG5s+fb927d7fBgwfbOuuss0RNHn74YUPcc9VVVy2xrJAZlEOY4G222SZshmgMYQ+siXp0yy23hGuf81kWu+yyy+zDDz+0TTfd1CZOnBiiXF1xxRVlKUrbiIAIiIAIiIAIiIAIiIAIiIAIiIAIiIAIiIAIiECVICDBUJU4TaqkCIiACIiACIhAdSIgwVB1OpuV61gQUyAYOvroo+2TTz6xWrVqhQq++OKLQUD0ww8/2IEHHmjHH398mH/TTTfZPffcEwRCvXv3DuKLs846y3bYYQfbfvvt7Y033rCTTz45RHJBIEIEFoQflN+iRQt75JFHrFGjRnb55ZfbBhtsEMokxdeVV15p7du3D2nFTjrppDCfSDTsg7p069bN7rrrLhs+fHgQFyE2ueaaa0KUmlmzZtnuu+9uhx12WIjag4iDfSD++ctf/mIsX3755e2cc84x9kWEmfPOO8823HDDIBqZPn16EHwQXYjIN34s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pBa27rrrWsOGDaveQSyjGn/22WfB6cLucbQQkkwmAiIgAiJQ/QhIMFT9zqmOSARqIoHKElGoJrLXMYuACIiACIiACIiACIiACIiACIiACIiACIiACIhAeQlIMPRfgoSuIlTT3LlzrXbt2rbyyiuH8E21atUqL+Mat/0///nPjOiF/Ht//etfaxyDsh7wE088YePHjw+bb7LJJrbbbruVtagqud1vv/2Ws96EjEsLBZdzIy0UAREQgUpIQIKhSnhSVKWlToAc0kSmSf4ePv744xD+tUmTJkulTuSu7tq16xL7IrUWUXKWZaScJSr13xkTxr1gX30x3Vq3bW+N11zX5nwxzaZMmhCWdt1u7zCdPOm9Euv490ZrtrA2bTcI64x64fEw9XK+nTff5syeZu07bRfmF/rPU5UxRUzkkYfI1w3HrbfeutAiS6zP+xohdmUiIAIiIAIiIAIiIAIiIAIiIAIiIAIiIAIiIAIiIALlJ1CjBUM///yzvfHGG/bWW28ZeemSRu67Vq1a2aabbmprrbVWcrG+ZyEgwVAWMHnMrsmCIdJGXHXVVTkpkb+yQYMG4Y/IVe3bt7d11lkn5zZaKAIiIAKVkUBSIFEZ66g6VU0C5FNG/HLnnXdamzZtynQQp5xyiu2000626667lml7NuJ5EOHPddddl7UMROUISZL17NWrVxCaTJ482TbffHPbb7/9MmU8+OCD9sorr9j111+fmZfrQz7HgiD5999/X6IY9rvPPvtYz549l1jmM+Ddt29fe+6556xp06Z27LHHWo8ePXxx3tOZM2faoEGD7MUXX7SOHTvaueeea+utt94S248f/YA1a1zbPpk+OyxruGp9a9igviH0+Xbu/DCvZYv/CK18Hf/u6/g2rJxch+9zF5e11nqdCxYNJaMNcW5jy0cwhCAII/+2m8/z78mpC4h8+pe//CW5ir6LgAiIgAiIgAiIgAiIgAiIgAiIgAiIgAiIgAiIgAikEKixgqH333/fnnnmmVShUJITEU0222wz23777U0Rh5J0lvwuwdCSTPKdI8FQbsFQGsf111/fdthhB1tjjTXSFmueCIiACFRKAhIMVcrTUi0qhVAIsckBBxxgAwcOLNMxffXVV7bKKqsY4vmy2B9//BFEQAsXLrR33nkna3ra0gRD9evXD8InxDhuO+64ox188MF26KGH+qyc03yO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IIAAAggggAACCBROgMBQjl/5vndY2XodCidZQntnr59n+965JtfZCAzl4uABAggggAACCS9AYCjhh4gOIlBogfXr19vnn39uffv2jXisf/zjH9akSRM7/fTTI64v6MKJEyda7969rXLlyu4QVBcqqCT7IYAAAggggAACCCCAAAIIIIAAAggggAACqS1AYChnfAkMpfaTnKtDAAEEEEAg0QQIDCXaiNAfBIpe4L333rO7777bJkyYEPHgN910kx111FF2zjnnRFwf68I777zThg0b5sJH+/bts9/97nd2+eWX25FHHmnTp0+3adOm2ejRoy0zMzPWQ7IdAggggAACCCCAAAIIIIAAAggggAACCCCAQBoIEBjKGWQCQ2nwTOcSEUAAAQQQSCABAkMJNBh0BYFiEogWGNq1a5dVqlTJIgWGDhw4YHv27LFDDz00Yq9279590Lrhw4fb73//e+vcufNB+9x33302e/Zsu+uuu6xGjRoHrWcBAggggAACCCCAAAIIIIAAAggggAACCCCAQPoKEBjKGXsCQ6nxD0BvrqhVqFAhNS6oAFexd+9e27Ztm3sjKSMjw8qUKVOAo7ALAggggEBxCxAYKm5hjo9A6QuEB4YWLFhgf/vb32zLli3WuHFjUzioZ8+eoQpD48ePd9WIduzYYa1atXIhoOrVq9u//vUv2759u3377bf21VdfWdOmTW3MmDHuNe+1117rXvspgDR06FA7//zzbdCgQXbPPffYk08+aXPnzrX9+/dblSpV7OabbzZtP3bsWKtdu7YDuuiii+yUU06xwYMHlz4YPUAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBAoUQECQznckQJDe9/5TVwDUa7TZVa2etO49inIxtnr59m+d67JtWu5tqOsXJvRuZYl04MNGzZEnaohr+vIyspy0zhom9WrV9tLL73kNtebJU2aNMlr15Rat379evvggw9s3bp1pjeYfCtbtqx7M6pt27Z29NFH+8XcIoAAAggkgACBoQQYBLqAQDELhAeGLrnkEmvXrp2de+65Lvxz3XXXuZCPpiTTtGGPPvqo3XrrrVazZk277bbbrHXr1nbmmWe68M/MmTPt9ttvt0aNGpmmIDvkkEPsmmv+9ztBeIWhE0880e6//35btWqVLVmyxIWMNDWa2oUXXmgnn3yym8Js7dq1dsEFF9jzzz9vCibREEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBNJLgMBQznhHCgztGdcvrmdCpGPEdYAYN07FwJA+Lf3000/HKPDzZvokdMuWLd2CGTNm2Mcff+zuazqGvn37/rxhzr19+/aZgklq+gR2Qd8U0SfCNY2EWt26da1cuXLufml806fFJ02aZIsXL8739F27drXevXvnux0bIIAAAgiUjACBoZJx5iwIlKZAMDCk16Fnn322qYpQ1apVXbeCU5L98Y9/dK8tTz/9dLfu008/tTfffNNNJaZqQWpXXnmlu1V4SOEiVQpSixQY0hRlev2r0JAqG/nAkMJBH374oXv8wgsvmM7z97//3R2HbwgggAACCCCAAAIIIIAAAggggAACCCCAAALpJUBgKGe8I4V9CAyV3D+EYGCoVq1abgqGWM5+1FFHmbZX27hxo73xxhtuaoeTTjrJ6tWrl+sQmzZtsscff9wt06e19cnqgrSJEyeGAjqawqFatWoFOUyR7KM3kebPn++OpWnYOnbsaA0aNLDKlSu7SkPff/+9zZkzJ1R1qEWLFu66SzPkVCQXzkEQQACBFBAgMJQCg8glIJCPQDAwpJDPI488kiskHwwMqcqQptdVsN03VYvU69d7773XLfKBIQV+NE3Zc88955aHB4b69+/vqhgdc8wxLiyk6kU+MOSDS88++6z95S9/ca8NVZGIhgACCCCAAAIIIIAAAggggAACCCCAAAIIIJB+AgSGcsacwFDpPvGDgSFNnaUpxYq6pVpg6KuvvrKXX37ZMelT6nqTKdKbz3v37rVXXnnFVq5c6bbt1q2bnXDCCUXNy/EQQAABBOIUiPQzO85DsDkCCCS4QDAwpHC7Kgy9+OKLlpGR4XoeDAwpDNS+fXs7//zzD7qq8ApDsQaGRo4caa+++qqb7sySWYn7AABAAElEQVQHhnTwa6+91lXp1HS+qjLEz6ODyFmAAAIIIIAAAggggAACCCCAAAIIIIAAAgikhQCBoZxhJjBUus91AkPx+7/99ts2d+5ct6MCVgpaRWs7duywxx57zH1qXZWILrvsMitfvny0zVmOAAIIIFACArxBXwLInAKBUhYIBobUlUsuucS6dOnigt56/XvNNde4oLyC3wqCK7yjqclUFXLRokX22Wef2YgRIyy/wJBCRgoH9evXz5YvX24XX3yx3XDDDW46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Qr56UPht8LLCw0V6HL7Mb6/X11Qe8hrcIoAAAggggAACCCCAAAIIIIAAAggggAACCHgBAkM5EgSG/NOhdG5TKTCkqj4vvviig1Q1H1X1yasFp8/Ka5qsvI6R37rVq1fbCy+8YAcOHLBq1arZRRdd5HZRoEjTf6llZmba8OHD3f14vsUbGCpTpow1b948plOUK1fOTj75ZLdtMDCk6dD8G2j5HUhVjzS1mRqBofy0WI8AAiUpQGCoJLU5FwKJJaDqksFwj39do1CPv1/QHvvgkL/1x/HHLuzx/fG4RQABBBBAAAEEEEAAAQQQQAABBBBAAAEEEEh+AQJDOWNIYKh0n8ipFBgKBmjatWtnAwYMyBN32rRprgKQNjr99NMPmmot0s4bNmywjRs3ulWamqt69eqRNsu1TIGhVatWuWUXX3yxVa1a1d1/4IEHbMeOHaYQzrnnnptrn1geBK83rynJXn/9dVuyZIk75JgxY6xSpUqxHD60zaZNm+zxxx93jzt16mT9+h08fV9o4yh3CAxFgWExAgiUigCBoVJh56QIlLpAMCyk8I4P8hRXx1TFSF++HXPMMYUOJfljcYsAAggggAACCCCAAAIIIIAAAggggAACCCCQ3AIEhnLGj8BQ6T6JUykwpCo+d955p2VnZ1vDhg1t5MiReeI+//zzpgpAaqr8owpA+bW5c+fa22+/7TZTBSNVMsqvvfXWWzZv3jy3maYe0xRkauPGjbO1a9eapv+64oorTBWAorWtW7e6VYceeqhVrFjR3Y81MPT+++/bBx984PY544wz7Igjjoh2mojL5XnXXXc510aNGtmIESMibpfXQgJDeemwDgEESlqAwFBJi3M+BEpfIBjeKengTjCoVNLnLn15eoAAAggggAACCCCAAAIIIIAAAggggAACCCAQSYDAUI4KgaFIT42SW5ZKgSGpBUNAZ511lh122GERMb/77jt76qmn3Lr69evbOeecE3G78IXffPONPfvss26xKgONGjXKypYtG75Zrsc+GKSFF154YagqUbDC0eDBg61ly5a59vMPgucMTrUWa2Doq6++spdfftkdLr/pzxS6UkBIU5IFm6xkpnb22WebqitFa/v27bPy5cvnWk1gKBcHDxBAoJQFCAyV8gBwegRKQWDixInurKoqpK+Sbj40RGCopOU5HwIIIIAAAggggAACCCCA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btw4t0+jRo1CQkLhB1LAx6sG5H8e3i/acqyQkH8fLzDUoUMH05jeffddV2no4YcftlmzZtn7779vt956q1WrVs1uu+022759u/3444/Wrl27qIGhq6++2tq2bWtXXHFFyJRkgwYNstGjR9ubb77phrB69WpX4UyBqRkzZviHFXy+ZcsWN6558+a50NDdd9/tKqJpmjQaAggggAACCCCAAAIIIIAAAggggAACeUGAwFDgLm/evNkmT55snTt3zvQ914cjmzZtsubNm2f6WBwAAQQQQAABBFJTgMBQat5XruqggKYRWr58uZtCSO+Jy5YtC00cAlQZigMphbv4w0IZmY4sIzQK/qjikEJ9XoWgX375xR3KW/bCQVqpdVr2+mgasljNHxgaNmyYqzSmdaomtGfPHhs6dKgtW/b/7N0HmFRF1sbxQ4aVLFlyWEGSIhkRgSFIUEGCSBAESYtgTuvqmlY/dQVcRYIKKKCAGECUDIIBJIMSREUBhyRIzsjXp6SaOz09M909ocP91/MM3Tff+tVFh5m3T+0QDepoZbJTp07Jk08+KbfeemuSgaGZM2fKsGHD5NlnnzWBoM6dO8t3330nu3btMkGijh07yn//+19zWzfeeKM0adJEHnnkkSRv880335QxY8ZItmzZTHBJl/Ply5fk/mxAAAEEEEAAAQQQQAABBBBAAAEEEEAglgRcHxjas2eP6A8v9YeTAwcOlDx58qRqfBcsWGA+wVilShVp27atZMqUKVXn42AEEEAAAQQQiD0BAkOxN6b0KKHAH3/8IatXrzYhegJDCW2SWnJWGapQtoRULFciqV1ZH4MCdiqyjAoLOQk1BKQVh5xhIA0H6ToNB+l6u+w8Ltj3GhLSfx9nzZo10aE6NViuXLkSrfe3QisR6ZSHvtMc6vm1ypmeR6dHrF+/vqliVLZsWRk0aJAsX748wenef/99ufLKv6YA1J8HaFU0GgIIIIAAAggggAACCCCAAAIIIIAAAm4ScHVgSEuQa+ny48ePS926dc2nD1MafP1B4vbt20UDQf7a6dOnRX/wuG/fPqlXr54pp+5vP9YhgAACCCCAgHsFCAy5d+xjvedaXUin/9Ff/uv3zVphSINDcXFxsd71NOnfhlULZcPqReZchIbShDTiT3Lw0FH5aXu86Ku2UKYii/hOZvANbtmyRR566CEzJeLo0aMz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+0u+fPkSrHNWDSpYsKD06NFDcuTIkWCfAwcOiN6fDfnoxtQEhiZMmCBapUubTvem077REEAAgVgVIDAUqyNLv9JaQKdKXb16dbKnJTSULA8bEUAgAgQIDEXAIHALCCCAAAIIIIAAAggggAACCCCAAAKJBAgMJSIJ3woNbmgAo3Tp0vLBBx+Yach872bNmjXStWtXE2bRbX369JHXXnvNdzeWkxFwBob0F7bOqd60opCGVk6dOpXoDLVr15YmTZpI5syZE21zrpg6dars2LHDu6pYsWImDFSkSBFTtUer8uh2W4lIpy/r2bNnoinO9ASBBoZ0+jrdV6evs61kyZJSqlQp0W0aEnJus/voa69evaRo0aLOVbJnzx559913vevy5s0rV111lblH9dEwlb5qlSRnS01gSINWGrjSpiGqihUrmiCchq18w0/Oa/IeAQQQiEYBAkPROGrcczgEnIGhAgUKiC77BoR0nW6jIYAAApEqQGAoUkeG+0IAAQQQQAABBBBAAAEEEEAAAQTcLUBgKELGX4Mq7du3ly+//NLckU4RVb9+fVOdpnr16vLbb7+JhoWmTZtmAiC6k049pZ+41jAHLXABZ2AokKMKFSokjRo1Eq0uFEjT6kFLly5N8dPwei795ZZWLEqqolSggSE916FDh2TSpEmilYSSatmzZxetPjRnzhzvc9S6dWvRZ8y36Q+1165d67s6wbI+g1p5yE6Pl5rA0O+//y5aZcg3hJTcFGsJboYFBBBAIIoECAxF0WBxq2EVsIEh/Z7p2muvNd9f6bq4uLiw3hcXRwABBIIRIDAUjBb7IoAAAggggAACCCCAAAIIIIAAAghklACBoYySDuA6R48elX79+pmpqlLaXau9jB07Vq655pqUdmW7j0BygaEsWbKYAFaePHnk8ssvN1V1NLASStu5c6csW7bMVOLRQJiz6RRgOoY69ZZW00mqBRMY0nNoZSQNna1fv17+/PNP72n1GlptSMNCGjCbMmWKCaHpDlWqVJF27dp593W+2bx5syxatMhbDclu06pINWvWNIG2JUuWeMNRqQkM6bl1ej0NMzmnOSMwZNV5RQCBWBIgMBRLo0lf0lNAp13VplWFtNkAkYaHqCpkSPgDAQSiQIDAUBQMEreIAAIIIIAAAggggAACCCCAAAIIuFCAwFAEDroGWsaNGycLFy5MEJzQ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ceOMeAz4DRo0yC1XSyiyY+zYsfbpp59a3bp17cQTTwyLon0iIAIiIAIiIAIlRkCCoRK74SpuTgl4K0PU6RVEQAREIBcEJBjKBXVdUwREQAREQAREQAREQAREQAREQAREQAREIBGBkhMMLV261O666y5nGWjHHXe07t27hzKaNm2aPfPMM7Zy5Up3vHr16s4q0c8//1y+r2bNmrb33ntbmzZtQtNgJ9finK5du1q/fv1ixtMBERABERABERCB0iEgwVDp3GuVNPcEvJUhuSXL/b1QDkSgVAlIMFSqd17lFgEREAEREAEREAEREAEREAEREAEREIH8JlBygiFuxxNPPGEzZsywdddd1w4//PCYd2jx4sX2+uuv25dffmm///57ebwaNWo4i0O4IVtrrbXK90ev/PTTT3b//fe73QcffHCoa7Poc7QtAiIgAiIgAiJQ/AQkGCr+e6wS5hcBDdbn1/1QbkSg1AjoHVRqd1zlFQEREAEREAEREAEREAEREAEREAEREIHCIFCSgqGvvvrKRo8e7e7Q4MGDrXHjxnHvFm7IEA8tWbLEWRlq0KCBc08W96TIQToFJ0+ebE2aNLGjjjoqUXQdFwEREAEREAERKBECEgyVyI1WMfOGgHdLJitDeXNLlBERKCkCEgyV1O1WYUVABERABERABERABERABERABERABESgYAiUpGCIu/PII4/YnDlzbKuttrKddtop4zcMcdG9997rLBPtu+++1q5du4xfQwmKgAiIgAiIgAgUJgEJX6otlwAAQABJREFUhgrzvinXhUvAuyVjogCiIQUREAERqAoCuDhfuHChffjhh26ikt4/VUFd1xABERABERABERABERABERABERABERABEUiWQMkKhubNm2cjRoywatWq2WGHHWZNmzZNlllS8caNG+c6Bdu2bWv77bdfUucokgiIgAiIgAiIQGkQkGCoNO6zSplfBLyVod69e+dXxpQbERCBoiWAYGj48OG2wQYbSDBUtHdZBRMBERABERABERABERABERABERABERCBwiVQsoIhbtmkSZPsrbfesmbNmtnAgQOtZs2aGbmTn332mT377LNWt25dJ0bChZmCCIiACIiACIiACHgCEgx5ElqKQNUR8FaG5Jas6pjrSiJQ6gSWL19u99xzj82fP9/1NzRs2NBOOOGEUsei8ouACIiACIiACIiACIiACIiACIiACIiACOQJgZIWDHEPxowZY19++aWtv/76huuwGjVqVOrWzJgxw5566ilbY4017MADD7RWrVpVKj2dLAIiIAIiIAIiUHwEJBgqvnuqEhUGgZdffllWPgrjVimXIlA0BHBJ9sQTTxjffiwby8pZ0dxaFUQEREAEREAEREAEREAEREAEREAEREAECp5AyQuGVqxYYS+++KJ98cUXbqZfnTp1KnVTX331Vfv4449t9913N9yRKYiACIiACIiACIhANAEJhqKJaFsEqoaAd0smK0NVw1tXEYFSJ/DHH3/Y0KFDDQtnHTp0sMWLF9uQIUNKHYvKLwIiIAIiIAIiIAIiIAIiIAIiIAIiIAIikCcESl4w5O/DN99846wM+e10l3QEVq9e3eSGLF2COk8EREAEREAEip+ABEPFf49Vwvwk4N2SNW7c2BANKYiACIhANgksW7bMxo4da0uWLLHWrVvbjz/+6CwRV6tWLZuXVdoiIAIiIAIiIAIiIAIiIAIiIAIiIAIiIAIikBQBCYaSwqRIIiACIiACIiACIpA5AhIMZY6lUhKBVAjMmTPHuSP+/fffbfPNNzcsDvXt2zeVJBRXBERABFIigCvEX375xb766ivbaKONbNddd03pfEUWAREQAREQAREQAREQAREQAREQAREQAREQgWwRkGAoW2SVrgiIgAiIgAiIgAjEICDBUAww2i0CWSaAtY8bb7zREAy1aNHCevXqZe3bt8/yVZW8CIhAKRJYuXKl/fnnn67oEyZMsEaNGtkWW2xhNWvWLEUcKrMIiIAIiIAIiIAIiIAIiIAIiIAIiIAIiEAeEpBgKA9virIkAiIgAiIgAiJQ3AQkGCru+6vS5TeByZMn23fffWc8h/Pnz3euhBEOKYiACIhAJgmMGTPGvv/+e5fkvHnzrFatWoY7xCFDhmTyMkorAwT8ffrkk08ykJqSyAWBdddd112WpV/PRT50TREQAREQAREQAREQAREQAREQAREoNAISDBXaHVN+RUAEREAEREAECp6ABEMFfwtVgAInMHToUJs2bZrts88+hpuyPffc02rXrl3gpVL2RUAE8pXA008/bQ0bNrS2bdva+uuvn6/ZLKl8IRJCIOTFQiVV+BIorBcOdenSpQRKqyKKgAiIgAiIgAiIgAiIgAiIgAiIQPoEJBhKn53OFAEREAEREAEREIG0CEgwlBY2nSQCGSPwwAMPWN26dZ1rMpbdu3e31q1bZyx9JSQCIiACnsC9995rM2fOtGbNmhluEc8991x/SMscEUAoFG1NqEmTJi43WIFSKEwCCxYscJYDg7lHMCTRUJCI1kVABERABERABERABERABERABESgIgEJhiry0JYIiIAIiIAIiIAIZJ2ABENZR6wLiEBcAqNHj3YD97/99pu1adPGevbsGTe+DoqACIhAOgR+/vlne/XVV23u3LnWqVMn++GHH2zAgAFWrVq1dJLTOZUkEG1VCJFQ+/btzYuFKpm8Ts8TArgb5W/69OnlOZJwqByFVkRABERABERABERABERABERABESgAgEJhirg0IYIiIAIiIAIiIAIZJ+ABEPZZ6wriEAiAt988419/vnnVqdOHWdpYuDAgSbLEomo6bgIiECqBLBohjuyL7/80tZZZx0bPHhwqkkofoYITJgwodwFGUKhDh06ZChlJZOPBHA9GhQN7bzzzoarMgUREAEREAEREAEREAEREAEREAEREIG/CUgw9DcLrYmACIiACIiACIhAlRCQYKhKMOsiIhCTAFYmJk6caB999JF17NjR9t13X5s9e7ZtuOGGMc/RAREQARFIh8Aff/xhL774ojVo0MC6detmtWrVSicZnVNJAkE3ZNwHWRWqJNACOv3dd991FocQCyEaUhABERABERABERABERABERABERABEfibgARDf7PQmgiIgAiIgAiIgAhUCQEJhqoEsy4iAqEEvvvuO3v55Zdtu+22s3nz5tnbb79ty5cvt7p169qJJ55oa6yxRuh52ikCIiAC6RAYNmyYrb322s6CGRZPZGEoHYqVOycoFpJlocqxLMSzcU+GaIgg12SFeAeVZxEQAREQAREQAREQAREQAREQgWwSkGAom3SVtgiIgAiIgAiIgAiEEJBgKASKdolADgiMGDHClixZYi1atHCDiD/99JNts802OciJLikCIlCMBJYtW2Zjx461Ro0aOcEQ7pH2228/q1atWjEWN2/L9J///MflTWKhvL1FWc9Y0D2ZXJNlHbcuIAIiIAIiIAIiIAIiIAIiIAIiUEAEJBgqoJulrIqACIiACIiACBQHAQmGiuM+qhSFT+DJJ590lj8mT55s1atXt969e9vGG29c+AVTCURABPKGwKhRo2zGjBlWp04d69q1q/Xp0ydv8lYKGcEF5YQJE1xR+/XrVwpFVhljEPCuyWRlKAYg7RYBERABERABERABERABERABEShJAhIMleRtV6FFQAREQAREQARySUCCoVzS17VF4G8CS5cutdGjR9u6665rW2yxhf3www/WqVMnuSX7G5HWREAEMkAAK0ONGzeWBbMMsEw1Ce+OrEmTJtatW7dUT1f8IiLgBUN887EypCACIiACIiACIiACIiACIiACIiACImAmwZB+BSIgAiIgAiIgAiJQxQQkGKpi4LqcCMQh8Oeff9p9991nm2++ua1cudLmzJlj++yzT5wzdEgEREAEkicwfPhwa9CggU2ZMsVwiXXIIYckf7JiVpqAFwzJHVmlURZ8AvPnzzdEQxIMFfytVAFEQAREQAREQAREQAREQAREQAQySECCoQzCVFIiIAIiIAIiIAIikAwBCYaSoaQ4IlA1BObOnWsvvviitW7d2rbccksbN26cHXDAAVVzcV1FBESgqAksW7bMxo8fb9999521bNnS2B4wYIDVqFGjqMudT4XDHRluySQYyqe7kpu8eMEQVz/ooINykwldVQREQAREQAREQAREQAREQAREQATyjIAEQ3l2Q5QdERABERABERCB4icgwVDx32OVsHAI/Prrr3bPPfdYvXr1bMGCBdazZ0/baqutCqcAyqkIiEBeE3j44Ydt2rRp1qNHD5s1a5YNHjw4r/NbbJn7z3/+44qEOzLckimUNgEEwgRckmFpSEEEREAEREAEREAEREAEREAEREAESp2ABEOl/gtQ+UVABERABERABKqcgARDVY5cFxSBuAQWLVrkrAzVrl3bOnXq5OJutNFGcc/RQREQARFIlsDLL7/s3JIhWlljjTWSPU3xMkBAgqEMQCyiJCQYKqKbqaKIgAiIgAiIgAiIgAiIgAiIgAhkhIAEQxnBqEREQAREQAREQAREIHkCEgwlz0oxRaCqCHzwwQc2e/ZsZwlkww03tE022cTatm1bVZfXdURABIqYAIKhxo0bO7eHRVzMvCyaBEN5eVtylikJhnKGXhcWAREQAREQAREQAREQAREQARHIUwISDOXpjVG28oPAn3/+mR8ZUS5EQAREoMgIlPrsegmGiuwHreIUBYFRo0bZ2muvbVOnTrVdd93VysrKnHuyoiicCiECIpBTAhIM5Q6/BEO5Y5+PV5ZgKB/v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x/3+srIy69mzpw0fPryChdizzz7bnUf6yYREZXnttdfskksusQkTJlRI7rPPPrP999/fPvnkkwr7ozdeeOEFu+2221w7k/xeddVV1qhRo+hoMbcR86z45Udbv1ktmzazzMXr0LalW/rtJo3rW5NG9W3+wiU2f8GS0Dj+HB/Hn0Nk0pk+q8wOOeYKd67+iYAIiIAIiIAIFC8BCYaK996WZMmCAiEAsO3/cEWGaIi/evXqOWsVJQlJhRYBERCBAiHAe3vp0qWuswqxUDwrQxIMFchNVTZFoAAJeLdkjRs3lpWhArx/yrII5AMBCYZydxckGMod+3y8sgRD+XhXlCcREIF8IHDkkUfaSy+9ZHfffbfttttuaWVp9uzZ1rp167TO5aSvv/7att9+e2vWrJmzIhSWkBcGxRIMsX/w4MFOLPTvf//bJYH16ubNm9uHH35obdq0CUt2tX2JylIZwdB7771nAwcOtCuvvNJ69OhhV199tSE04htFv1cy4eVnhlm9WsvNC36SOSedOAiJOnU/LJ1TdY4IiIAIiIAIiEABEZBgqIBulrIan4AXCxHLr4eJhRAMNWzYMH5iOioCIiACIpAXBDDzjFjI/wVFQ2QwKBQKrudF5uNkQhaG4sDRIRHIQwK4JUM4JCtDeXhzlCURKAACEgzl7iZJMJQ79vl4ZQmG8vGuKE8iIAK5JvDzzz9b586d7fbbb3eWeZ544onyLN1111328MMPu770M844w3beeWebOXOmXXDBBfb555/bjjvu6KzjYM2/V69exrlYyrn33nvtpptucsfXWmst14466KCDbJtttrFdd93VHn30Uee2izjesg4We+grefLJJ+2ee+6xTTfd1OXDi4C+//5722effezGG290FoZi7Z84caKdcMIJNmXKFHc+6WHN55VXXjGEQKeccoqz7IPbMCz7rL322jZkyBDr1KmTUW8g3t57711elqFDh9oDDzzg8nbcccfZAQccYAiGTj/9dCeuIv2+ffvaddddZ19++WUFC0Ncd+TIkdalSxc788wzHedjjjnGWWG67LLLXP4YqyAPb731lm288ca23Xbb2SGHHOLuRZ8+fdx1PCN/Y7AwVPuPOc6CkN+XrWXTDQdkK2mlKwIiIAIiIAIikCcEJBjKkxuhbFSegBcJkZIXCvll0LoQLm6aNGlS+QsqBREQAREQgawTmD9/vnMhGSYYQiAUFAkF17OesUpeQIKhSgLU6SJQxQS8laF27dpVMG1fxdnQ5URABAqUgARDubtxEgzljn0+XlmCoXy8K8qTCIhArgkghvnf//5nd9xxh2vrYAGnadOmNm3aNNtjjz1s8uTJTgiDCOf111+3I444wgmBcBvGvn79+tm+++7rLAN9+umnhngI8c0zzzxjHTp0cC7GOOfEE090fTh8mwcMGOBEMVgzwk0YffjEJf0xY8bYV1995QRHsDnnnHMMUdO1115rTz31lJ122mmGeCjWfp8W10cIhYswrkMezj//fNtwww2dyzMEPAh1sES00047ubgXXnihE/NgkYiyLFu2zI4++mh7/PHHbcmSJU7k9MUXXxiM9ttvP3vjjTesbt26TsiE+In0vEsyhES33nqryzPWjS699FInRkKohOgJa0o+7LXXXnbggQfawQcf7BiNGzfOXQuBEsKpc88910d1yxXLfrSFsydW2JeNDSwMLa/Wyrpu1SsbyStNERABERABERCBPCEgwVCe3Ahlo/IEaAz44IVCfolgaOXKle7vt99+c40eH1dLERABERCB/CXwww8/uM6mGjVqGH9BC0MSDOXvfVPORKAYCWBliICVIQUREAERSIWABEOp0MpsXAmGMsuz0FOTYKjQ76DyLwIikA0CCFdwi4V7LKwI4VYMUQ79Mf/3f//nLOpgVWeLLbZwl7/iiiucGAix0J577lluyR9XYohssL4zatQoJ7LhhJNPPtmJdBAM0adDPz0Bi0YIg2655RYbP368ywNCmZ9++sm6du1qs2bNchPIsM7z2GOPOUFP0CVZrP2kffHFFxuWe84++2wngsIqEpaOuDbtOvL49ttvO6ESVpEQDGHhaIcdduD0cvHTOuus4zhgdQiLRVhOol7H5DbiT5gwwcXHutDixYud4McLhhAo4Q5t6623dnFwQUZ54Xzfffc5tu5A5N8///lPJ6waNGhQBUZvvvmmEzS9+uqrPqpbVpVgaNrMMps5e7H12XOwxlMq3AFtiIAIiIAIiEBxEZBgqLjuZ0mXxguGgkvWg9aFEA0tX77cVfpLGpYKLwIiIAIFQgAT07Vr13ZioTArQxTDWxbyy0IomiwMFcJdUh5FoCKBGTNmONP1cktWkYu2REAEEhOQYCgxo2zFkGAoW2QLM10JhgrzvinXIiAC2SOA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kiVL3LjArrvumtb5YSf98ccfbqIr4wb8/ho2bBgWTftEQAREQAREoGgISDAkwVBe/JglGMqL21CQmcC6DS7FqMhvuOGGtttuu1mNGjVWK8v48ePtgw8+cBZtDj30UENYRJBgaDVUKe3wgiEaervvvntK58aKLMFQLDKluV+CodK87yq1COQzgREjRtgzzzzjBK/UQ3CFethhh9n++++fz9lW3kRABHJIIBOCoZUrV9q4ceMMwQODM8ccc4ztt99+OSxVYVxagqHCuE9VlUsJhqqKtK4jAiKQDwSYYHnWWWfZq6++atWrV08rS4glRo8ebVgNTycgttlxxx2NNlQskcjjjz9uuLh69NFHK1yCthZilLKyMuf+iv7foFga4QnHvTXzCidHbcyYMcOOPfZYe+mll6KO/L15ySWXuH7jCy+88O+dkTVEMLT1Pvnkkwr7ozdeeOEFu+2224xr9ezZ06666ipr1KhRdLSE2whg4PHcc8/ZE088ERp/ynvjbelPn1uHti1Dj2dq57SZZbZNvxNCk0MwxG+radOmrrxEuuOOO9zk3QsuuCD0nL322ssuu+yySk244bp33323/fvf/7Zq1ao53kOHDrWrr77abfN7WnvttW2nnXYKzUOindxv8ti2bVuXzuuvv26nnHJK2uklup6Oi4AIiIAIiEA+EJBgKIOCoSuuuMLmzZvn7uuVV15pa621llGhoHPwww8/tM8//9xVNLbccktXiWI2crzwf//3f+UV0fvuu88OOuigeNFd5WjYsGFO/czARZ06dWyzzTZzf1SQYim7YyU6adIkw80TIgtcBxGolCPKOProo61Dhw6xTl1tPw2LN99809555x3HCIsi5O2AAw5w6UkwtBoy7UiSwNNPP+2sB2G9hoYfVoXCArNS7rzzTvvtt9+cBSIaq4RYgiHi00BjBkGyjeo///zTsDjAuTQIg67PwvIU3FeZcykTeV1zzTWtbt26wWTjrvvzGjRoYDVr1owbN9bBXAqGMpH/WOXS/vwhIMFQ/twL5UQERMDc93bfffe1Sy+91FnWo65MPf+xxx5zf6nMPPzll18qfLexFhKrHiP2IiACFQn89NNPbgdLBgT4y3RggIv6tQ/xnlHq8gh6wurUnEe/QJMmTYy+gOjw66+/urZ79P7obQbLGAzp3bu3GzzafvvtJRiKhhSyLcFQCJQS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16lTJ/fVV1+5xo0bJzh8+eWX7rvvvvP7U6dOdSeccIJ78sknXZcuXdwee+zht9133z1R3hKff/65W2211fxmeXWJ1T7VZ7HqUlpB+QrhMZ+Rw59Ro0YlSq+++uq+f4oVVO/YsWMTx9u3b++0lSKIozaFUl63FH3jGhCAQHYEHnjgAadNz74KxxxzjGvQoIG79NJL/XPvDTfc4Hr06OHOPfdc98EHH/hyW2+9tf8/sOWWW7qPPvrI7b333m7ixIlum222ceecc45bfvnl3cEHH+yff//whz/UaIh+4xVS/ebot9B+ewcOHFjjvGx2wt/a5PJDhgzxv7VWf6rrJ59TqH1+awtFsnT1TJo0yWnTPI3mf7IN+gxq0/dg/fXXT5w2YcIE/5068cQTXa9evfx36aKLLvL3Pb17906U033TSy+95Mu2bds2kX/QQQf5e4e//vWv7rfffnP77LOP69y5s/9eJgotTeje7ffff3dXXHGFz9b398033/T3eMo488wz3cyZM92tt97qVlxxRV8m+Y/u71T/dttt5/7yl78kDv/rX/9yp59+ur/v69+/v8/X9/mee+5xQ4cO9feTicK1JFKd9/rrr7vrr7/eHXDAAW6nnXZK1KC+fPzxx+7OO+/0ebpffeSRR3wfwnvKd99911122WXu5JNP9r9Z2fQ1cZFaEl988YWbMmWKa968uf+dq6V4nQ7/+9//dupLruGPf/yj00aAAASqh0C57i90L6Vr2z2V/q/ZM3P10KenEIAABCAAAQhAoPIJLIdgqPZB1CTP3LlzaxTUhEzfvn3dcsstVyM/2x3ptF599VU/2RGeowUjTWASIAABCEAAAhCIL4EoCYZ0T3L22We7YcOGOS1ItWnTxl144YVe1PPpp5+6Pffc00nU3L17d3fzzTf7e6JXXnnFL8K0bNnSrbDCCm748OFO6ccff9yLhHT+euutlxAM/fLLL34RWYse33//va//8MMPTzvAn332mdtll12crq+QSTAUVmKTlVr8zTRRqQlNO26Tq9rXBKeOmWjHJjytjO2H18w1rfvKZKFQWEd4feWrL2Hsdwr4R9czbnYdu2YBL0NVEIBABRBIFgzNnj3bCx302y0R6N/+9jcvBFh77bW9KOj99993tQmGttpqKy8wfeuttxJCJEOR6vdOedos5Cvm0W9tqqDneAlRU11b5Yv1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TTfd5EUqdlhiGPmaL2aQKEbXtaDr6brFDHUVDL3xxhtOdayyyireWpBch1kIBUPfffedu+GGG7xlIR1v1qyZLzZv3jwfSzi0xx57JCajv//+e3fttdf6YzpXbOrVq+ckolKQwEgT2O+//74XKSlPloiyFQxJkHTjjTe6b7/91ouQZM1pyZIl3tKQ6lpnnXXcoYce6q+pfQIEIAABCECgkAQQDBWSZrzrSmdliIndeI87vYNAFAjkIhi67bbbnCwS3XLLLVk3PRQLyaqQxEF6qUPBREShaEhCIbMCvNVWWzk9wxEgUBcCtnCq/6kKqQRECIbqQjh+5yIYit+Y0iMIVAsB+5+X3F8s1yYTYR8CEIAABCAAAQhUHwEEQ2nGfNiwYTWOHHXUUTX2i7VT6utmKxiSYGfo0KH+zdGNN97Y7bPPPh6BCYa0I+s/W265pZ/o1cOGREAS+fz0009eCPXNN9+4Fi1auP3339+7K9M5cgV2zz33uBkzZngx0GmnnbaMW7BMLsnyEQzJYtHNN9/spk+f7jp06OD22muvxDUnT57s7rvvPidrT7m6QFN/CBCAAAQgAIFsCCAYyoYSZYxAqsndbbfd1g4TQwACECgIAblqliXZl156yQ0ePNj985//dE8//bR/qePFF190F154obe6e9xxx3mLrGZh6KyzznJnn32200sZspB78cUXu9dff90pX1Z7d9xxR3fuuecu08Ynn3zS54WiICsUiol69OjhXZ9JLCSRkMRCBAgUisALL7ywTFUmIFKMYGgZPFWdgWCoqoefzkOgYgmkep60ziAYMhLEEIAABCAAAQhAoHoJIBhKM/alFu5YM0p93WwEQxILyXWXJokVZAlos8028+lQMDRo0CDXu3dvnx/+kcuwt956y1shOv30030cHpdlH7kf0wSzJoN322238LC3HLRw4cKUAp58BENyofbyyy/7twdPOumkZawIjRs3zj388MNewHTGGWcs094ajWMHAhCAAAQgkAcBBEN5QKvSU8zCkBYtJbTWpoBgqEo/EHQbAkUk8Mwzz3hR0MiRI/1vzeabb+7kgrpBgwauT58+/nlQlmX79u3r3UxPnTo14ZIs2cKQREJ6xltvvfXcNtts46688soaz4qhIGjgwIEpe2WCIj0v/v7777ggS0mJzLoS0P9VLaSmC3qZSFvTpk1dt27d0hUj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aW2hYChM64LaVwiFQmHaH+QPBCAAAQhAAAIVQwDBULSGSgtK2nDhEK1xoTUQiDoBCX7mzJnjZs+e7bp06eK0gCPLQRIGtW7d2n3wwQduiy228C+VvPfee35R5/PPP/eCRL18omfMJUuW+N+f/v37R727tA8CEIBAJAnYfVzYuG233daLiCQmyiXovKgEBENRGQnaAYHoEHjttdfc+++/7y0KyarQVlttFZ3G0RIIQAACEIAABCAAgYolEEvBkAmENCqWVqwtFAqZm7FQICQLQSYUatCggWvWrJkr1KLejz/+6ObPn+/01qkJh+T2IhQQSRykY6FoSOIgEwhZrL6Fae0TIAABCEAAAhCoDAKFureojN5Gu5W2yIRYKNrjROsgUAoC3333nZs0aZLr3bt34nIPPfSQ69mzp1trrbX84rOskek5bOutt3ZDhw51PXr0cF999ZV/uUSWZufOnetWW20116JFC/fxxx97cZCsC912223ulFNOcddff7079thjvchIdTVs2ND169fP6SUVAgQgAAEI5E4gWVgT3tPZfV62tUbJLVlyv7LtQ9j/bM+hHAQgUBkE7rnnHi84nzZtmn/ZRdZxsYpbGWNHKyEAAQhAAAIQgECUCVSFYCiTWMisCik2sZAEO2uuuaY3C1+MwdNE9DfffOMtGUksZKKh0NpQOtFQKBIK08VoJ3VCAAIQgAAEIFAcAgiGisM111ptISZqLihy7QflIQCBuhOQlaCXXnrJDR482LugVo1ajBk5cqR/Xtt8883dyy+/7LQwM336dP+Sx8yZM1337t29G+qvv/7aDRgwwJfV896YMWP8/nPPPeefK+0tcLkiW2+99ereYGqAAAQgAIEEAXMtq4xkK0HhscQJaRJRuie0+9Q0TU2bjWAoLRoOQKDiCdx3333eMuW+++7rxo0b53bccce8PCJUPAg6AAEIQAACEIAABCBQUAKxEwyZRSFRMqFQOstCoUhIYiFtMgu/9tprFxRyuso0Aa3JZAmGTDRkcSZLQ6FQKEynuw75EIAABCAAAQhEiwCCoWiMh9xUaDEmSm+TR4MMrYBA9RF4/fXX/Usdsgir58H69et7CNqX27FOnTo5uYGQ9SBZBdK+hEPdunVz66yzjhs2bJjbZ5993P333+8Xbjp27Og23XRT/0wqS7NYEKq+zxQ9hgAESkfA7ulSiWVy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/PPPvWu22bNn+/yNN97Yi4tqFFq6E1oN0vdEln/at29fo5iYH3jgge7jjz9O5KcTDH333XdOn3cLalu4b/nz5s1zsrql3xMFuaSTFapk0ZB+m+TO7pFHHvHlJABTGwkQgEB1E0AwVLfxtzcvu3bt6kVDdauNsyuFQCgGCtPp2h+KgcJ0uvLkQwACEIAABCAQfwKaA8km6N5B95rlCrmIm8I2IjYIaZCGQOUQsGdcvsOVM2a0FAIQgAAEIAABCMSFQKwEQ1qYN7GQxWZhSIv6ZmVIloa0aC+rIFEMckl19dVXJ4RCJhYyC0MmFrJYfQhFQmG6EP1r3rx5wl1WIerLpY5iC4aee+45bznI2iRLN7L8I9drEpeMGzfOLVy40A4n4r59+7obb7wxsW+JGTNmuMGDB3tRkPIkPtppp51c586dvRWdyZMnuwceeMBJbKKw9tpru5tvvtm1adPG74d/lH/DDTcksuQuTa7KVJc+6++8845799133ZdffunLSKhy0003pRS3JCpJkSi3YEhN0nexV69erlu3bl6kNXHiRPfkk096N4I6rs+0eKSyACUxkMbMmMpyz6677uoFYLLsIytFEoXp+x+GdIIhlenXr58zoZIskmkMxb9t27auT58+iWrkLu7SSy/1bVamHuo1/q1atXLff/+9t04kC0XffPONP0f90Hcb08IJhCQgULUEEAzlP/RMpObPrlLODMVAYTpd+00MpP/DFizP9omjRWDRokVePK97uFThiiuu8Pddu+++e6rDafN0XywroLrfzid88skn7rLLLnN33XWXq62N+dTPORCAQHYEZA1Yz6lbbLFFWmu02dQky75yB62Xg9q1a7fMKV988YXTVtfrLFMxGZEmgGAo0sND4yBQtQTsOZc5w6r9CNBxCEAAAhCAAAQgUDYCsREMiaBZGFKsCSaJhiQY0hYKhiQwkDUTCRSiGGRt5sMPP/RWhZKtC8nKkAmFQgtDYT8KKRhSXbKms9pqq4WXKFm62IIhdeTyyy9fxsJPcge33HJL17p164SFGwlcnn/++eRifl8LFbI09Omnn6Y8bpmymHP99ddntIzw0EMP+fZpojNTkNDp3HPPdQMGDMhULOWxcgqGZLnn22+/zchKAiC5/tJnIV14+umn/eKQvvPpgoRvW221lRs5cqQvkkkwpEWmCy+8cJmqDjnkEHfi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961q1aqRbVZefPFFw4UZgXywZDRr1iybPHmyzZkzx37//Xfr06ePYQY0HDhv3rx5Nn/+fPfH8erVqzuLNh07dszywz58Luc888wzbneDBg3s7LPPNlxsPf/880ad+du8ebNzs3XEEUfYWWedla3LpGiCoQ0bNjhRzccff2z84fII6yx16tRxro9S1UUb1oWw6vLbb79Zr169Ylp3Wb58QsrjzQAAQABJREFUud1333129NFHW4cOHcJNoO0UI5DugqG///7bPWvWrFljN95443bPqhRrLhVXBNKOgARDadekqpAIiIAIRNw8yEqCLgYREAEREAERiJ+At3gi4UL8zBRTBOIlgCjIC4PC5+jeCxPRtgiIgAiIgAiIgAhkJoGMEww999xzdu2117rWrlixohPo7LjjjnG3/pQpU2zr1q0uPpZmSCMYgkIb8kFQ1L1794j7M+Lef//9dsEFF0ROQ6iCi7PsLB0VK1bMLr74YidsYT0cXnnlFbv00kvd7nPOOceuvvpqu+SSS2zZsmXhqG577733NgQThx12WNTjwXqQzmWXXeZEMp9//nnU+HAYNGiQHXfccVGPJ/NOb4mFsrdt2zZS1P/7v/+z6dOnW/369W3fffe177//3rAyhSCrXbt2htAIoRbtEYtjJDGtJB2BdBcMAZzn1YQJE+zQQw+NPB+SriFUIBHIUAISDGVow6vaIiACaU+AWdwEREMKIiACIiACIiAC8RGQa7L4OCmWCOSFAIIhrAxFC1405K0QxYoX7VztEwEREAEREAEREAERSA8CGScYuvfee+2BBx5wrXf88cfbCy+8kNCWDApt9t9/fzfLFEsfwRAUDCHAQZCDuysC4iUG9xEarVu3zhYtWuSW/vxrrrnGbr31Vr8ZWQYFQ5UqVXLnYFGIUK5cOStfvrx99dVX9tdff0XOKVGihGFh6aSTTors8yvBeiCGQXj0yy+/OHEMZStevLgtXbrUNm7c6E+xChUq2Ny5c23XXXeN7Ev2FXjceeedhjgIYRcWnXx46623bPz48W4TQdRBBx3k3NDhjg6mWFmCMe18/fXX+9OKdInVJ9pkr732ctdSQRcGsRtWs/bcc89cZfXzzz/bli1bHMfcCPbIhDb78ccfrUyZMsY1HE9A5Ld27VpXzp133tmdkhvBUDzl/e6776xfv34u7Z49e1rlypXdOpa5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Jl6rHyT121Y1EAEREAEREAERSE8CEgyF2vXHH3907ouw4LHbbrs5KyiVKlXK0ZJHKJk8beK6CNdElAFxEFZc4g3RBEP+XNJlNivWZ5jJWrp0aX9Iy38JILAaO3as43711VeLS4iABEMhICmw6a/punXrWufOnVOgxCqiCGQWAXWUZVZ7q7YiIAIiwASOuXPnGhYgeQfwx3cZboVxISvRkK4RERABERABEYiPgHd9hGCId+myZcuc9b74zlYsERCBgibA+MaCBQtsyZIlrq/90EMPtWHDhlmbNm3c2ERB56/0RUAEREAEREAEREAEckdAgqHc8Ura2NkJhpK20ElUMDrwhw4dagsXLrQWLVpYs2bNkqh0RV8UCYaKvg1yUwIEgkOGDHGiR6yFSSSYG3qKKwKFQ0CCocLhrFxEQAREIFkI4OoWwVCDBg0MC5D8VitWrJidfvrphrWhU089NVmKqnKIgAiIgAiIQFITmDNnjhMj8G5t2bKlK2u1atWSuswqnAhkEoE///zTeVJYs2aN65OcNWuWdevWzRYvXmxY9kdApCACIiACIiACIiACIpA8BCQYSp62yFdJJBjKFz538ubNm23gwIH2/fffO4ssWGZR+IeABEOpcyXgF7x3795ulvp1112XK0tlqVNLlVQEUp+ABEOp34aqgQiIgAjklQCzrefPn28MojRs2NBZHcJltIIIiIAIiIAIiED2BHh/IrStVauWvf/++4bLo3r16mV/ko6KgAgUGYERI0YYE3Xpr2SJezLEfvvss4+VLFmyyMqljEVABERABERABERABP4jIMHQfyxSek2CocQ0Hx8sY8aMscaNG5tmJ/3HdOPGjbZo0SK3o3bt2vqg+w9N0q1t3brVHn30UWfmt0KFCklXPhVIBETgHwISDOlKEAEREIHMJYCr6OHDh1u5cuWM325YHapSpYrtvPPOmQtFNRcBERABERCBOAisXLnSZs6caY0aNXLCoZ9++slwTbbHHnvEcbaiiIAIFDaBL774wlnZ/Ouvv6xSpUpWpkwZt92hQwdjgurBBx9c2EVSfiIgAiIgAiIgAiIgAiECEgyFgKTqpgRDqdpyKrcIiIAIiEAmEpBgKBNbXXUWAREQgf8IMFHh3XfftQ0bNhjvhI8//tjOOeccK1++/H+RtCYCIiACIiACIrAdgRUrVtiWLVucSOijjz5yS0RDCiIgAslJYO3atYZgfv369fbcc8/Z+eef7yyE4Z6XCbtY3FQQAREQAREQAREQAREoOgISDBUd+4TmLMFQQnEqMREQAREQAREoUAISDBUoXiUuAiIgAilBYMKECYZb5P3339+5Zfj666/tuOOOS4myq5AiIAIiIAIikAwEvvnmG+Nvv/32c3/JUCaVQQREYHsCWBgaMmSIHX744TZv3jzDOliXLl1s2rRpduqpp8pK2PbItEcEREAEREAEREAECo2ABEOFhrpgM5JgqGD5KnUREAEREAERSCQBCYYSSVNpiYAIiEBqEnjrrbfcTOu///7bVq1aZaeddprtu+++qVkZlVoEREAEREAEiogAVobkmqyI4CtbEYiTwPPPP281atSwI444wh544AG3vnTpUitVqpSzMPTll1864bzcC8YJVNFEQAREQAREQAREIIEEJBhKIMyiTOq3336z1atXuyIUL17cKleuXJTFUd4iIAIiIAIiIALZEJBgKBs4OiQCIiACGUKAmdavvfaaLV682OrWrevEQnzXHX300RlCQNUUAREQAREQgfwTQCwk12T556gURKAgCaxbt87KlSvnsliwYIHNmjXLSpYs6X7/VqhQwSpVqmSTJ0+29u3bF2QxlLYIiIAIiIAIiIAIiEAUAhIMRYGiXSIgAiIgAiIgAiJQkAQkGCpIukpbBERABFKLwKeffurcqSxbtszq169vO+ywg1umVi1UWhEQAREQAREoOgJyTVZ07JWzCOSFABY2CYMHD7YDDzzQuRScPn26derUyf0WzkuaOkcEREAEREAEREAERCBvBCQYyhs3nSUCIiACIiACIiACeSYgwVCe0elEERABEUg7ApMmTbJNmzbZihUr7MILL7Rp06bZeeedl3b1VIVEQAREQAREoCAJeNdkTZs2LchslLYIiEACCeCqjP6RefPmOQtEuOjF0lC1atXshBNOSGBOSkoEREAEREAEREAERCAWAQmGYpHRfhEQAREQAREQAREoIAISDBUQWCUrAiIgAilIYPny5YZoaNWqVda5c2crVaqU7bHHHilYExVZBERABERABIqOgFyTFR175SwCeSXw+++/29y5c+2PP/5wSWBxE4tDWNzk93CdOnXymrTOEwEREAEREAEREAERiJOABENxglI0ERABERABERABEUgUAQmGEkVS6YiACIhAehD4+eefbfbs2VaxYkU78sgj06NSqoUIiIAIiIAIFDIBb2WId6nEt4UMX9mJQD4JjB8/3ooVK2a4623WrJntueeetv/+++czVZ0uAiIgAiIgAiIgAiKQEwEJhnIipOMiIAIiIAIiIAIikGACEgwlGKiSEwEREIE0IPDNN98YfxrkTIPGVBVEQAREQASKjMDUqVOdWEgC3CJrAmUsAnkisHTpUnv77beNZd26de2QQw6x/fbbL09p6SQREAEREAEREAEREIH4CUgwFD8rxRQBERABERABERCBhBCQYCghGJWICIiACKQdAQ1ypl2TqkIiIAIiIAKFTMALcBEaSGxQyPCVnQjkkwCuBTds2GCrV6821nUf5xOoThcBERABERABERCBOAhIMBQHJEURAREQAREQAREQgUQSkGAokTSVlgiIgAikDwE/yCkrQ+nTpqqJCIiACIhA4ROQa7LCZ64cRSDRBPzvYtwLymJYoukqPREQAREQAREQARH4j4AEQ/+x0JoIiIAIiIAIiIAIFAoBCYYKBbMyEQEREIGUJMAgJ0EDIynZfCq0CIiACIhAEhDAMgnvUwkNkqAxVAQRyAcBLxoiCQnq8wFSp4qACIiACIiACIhANgQkGMoGjg6JgAiIgAiIgAiIQEEQkGCoIKgqTREQARFIDwJ+kFMuGNKjPVULERABERCBoiHghQZ6nxYNf+UqAoki4O9l0tP9nCiqSkcEREAEREAEREAE/iMgwdB/LLQmAiIgAiIgAiIgAoVCQIKhQsGsTERABEQgZQl4VypNmzZN2Tqo4CIgAiIgAiJQ1AT8+1SWSYq6JZS/COSPAIJ6hEMsJRrKH0udLQIiIAIiIAIiIAJhAhIMhYloWwREQAREQAREQAQKmIAEQwUMWMmLgAiIQIoTkJWhFG9AFV8EREAERCApCPj3qVyTJUVzqBAikG8C3tqQREP5RqkEREAEREAEREAERCBCQIKhCAqtiIAIiIAIiIAIiEDhEJBgqHA4KxcREAERSGYCDHisXbvW6tWrF7WY3iqCrAxFxaOdIpARBF599VU78cQTbbfddkvp+q5fv96WLl1qhx12WErXQ4VPTgJbtmyx+fPn27Jly+zAAw+0OnXq2A477BAprH+fyspQBIlWRCClCXjREEJAhEMsY4XVq1fb3Llzbffdd7dDDz3U9txzz1hRC2T/X3/9ZfPmzbM1a9ZYgwYNrEyZMgWSjxJNbgJffvmle09VrlzZ/RbaddddXYFz+h5M7lqpdCIgAiIgAulEQIKhdGpN1UUEREAEREAERCAlCEgwlBLNpEKKgAiIQIESmDJlin399dd2xRVXuHweeOABO/vss6169epu21tFyO8M6h9//NGGDh1q//vf/2zHHXcs0DopcREQgcQRQATRs2dPu+aaa6xq1ar24Ycf2qJFi6xDhw6Jy+TflMaPH+8GURs3bpzQtN9//32bPHmyvfvuu3b00Udb7969E5q+EhMBrrH+/fs7odBee+3lhAG1a9d2906xYsUigKZOnepEBYiGFERABFKfgBcNUZNov5V5h95+++22YsUKO+qoo+y7775zv7tvu+02q1u3bqEAWLJkifXp08f9tkeoNGvWLLv22mst0e/aQqmMMskTAYRiN954oxOs1axZ0xYsWGCIyO655x7jnRX+HsxTJjpJBERABERABBJAQIKhBEBUEiIgAiIgAiIgAiKQGwISDOWGluKKgAiIQGYQOOecc5yoJziYiVUEOppPPfXULNYSPJHff//ddtlll8ix8DbxsLjQpUsXmzRpku28887+VLfcvHmzlShRIss+vxEtLX9MSxEQgcIn8Morr9iMGTPswQcfzJL533//bX/88YcVL148y36/sWnTJvMz2dnHvR2Oi5CHWe8dO3b0p7nl1q1b3cBWUHgRjLBx40YrWbJkcFeWdQZmGaj99ttvDSsPEgxlwaONBBC47LLL7IgjjjCWBK7J888/32666SZr2LBhJIcffvjBCe6iCQsikWKsZPc+zO5YjOQMIQMWkHbaaadYUbRfBEQgDgJeXE/U8L09e/Zs69u3r40ePdrKli3rUhsxYoSx/4knnogj9X+i8B7kj9/bPkR7j/pjwWWPHj1s7733tuuuu87tfvHFF917/JFHHglG03oaExg0aJAtXLjQWPpw/fXX2/77729XX3213xX3Mi/vnLgTV0QREAEREIGMJiDBUEY3vyovAiIgAiIgAiJQFAQkGCoK6spTBERABJKLwJgxYwzz9BdddJGbeYrLHgb1W7Zs6QbtEQrdddddbjY0g4pXXXWVnXLKKbZu3Tpr3769nXTSSYbFBAZDKlWq5CwsIA4iDdyYMfCPO6Nhw4bZb7/9ZqVLl3YDqPXr17exY8faSy+95PZXqVLF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qMzTqVMnU7WedJuPT7DtogJDCuGoQo9CVDfddFPalbP+/e9/22uvveY+5zfffHPssx/sU4+ZBob0d0HTDqqpUlCVKlXc8+CXzrOmntNnf5999rHTTz/dvaUKaXfccYd7fsUVV7hgX9CnuEffwJCmAdM0evobdsopp1iXLl2K25V73/cY09o4KyGQxwIEhvL45DF0BBDISCCsX6QzQmBlBBBAAAEEEEAAgRIL+N5X+vYr8YDZAAIIIIAAAggggECoBMJ6X0lgKO4yVcWL3377zVVDUfUevY5Cq1ixopvaSgEoBUz0OirtwQcftIkTJ1rNmjVdMEOPiW3z5s126623umpUClOpykpQ0aUkgaGPP/7YXnzxRdOUUwr1JFZy0n4VTFmwYIELFClYpPb888/bp59+6gIvCi8lhkx03V5//fXuOt5jjz3sz3/+c+yQyiMwNHr0aHvppZecmcJOySrTqHLSvHnzrH379rGx+vpoA3L79ddfXahF4Zb4pmpROp/yTVXdSYbffPONmyouONcKbilgpH49e/a0Aw88MH6zLsw1YsQIN6WY+txzzz0F3k/2QtOOaeo4Td0WX0Eoft3HHnvMxo0bV6AikkI7Ovdyk9nAgQPju8Se6xj0frVq1WLLfAND2sAjjzxi3377rbv29DmI326wA1VG2nrrrV3ITct8jzHYHo8IhFWAwFBYzyzHhQACiQJh/SKdeJy8RgABBBBAAAEEEChdAd/7St9+pXs0bB0BBBBAAAEEEEAg3wTCel9JYKiIK1GhIf0EFYJUiSgMTZWDgopICqkkBlXCcIzpHsOiRYtMU3apkouqKvXu3dtatWoV666qUwr1TJ482S07//zzY9PCaUFJAkPxgaAWLVq46jv16tVz+1FI6O233zZVNtL5+tvf/har+KRqMwqLrFmzxrbffnvr169fbMoxXasKdfzwww9Wt25dN3WUHoNWHoEhHedtt93mrHStnXHGGTFDVa1RhR0FbX7//XfTdG8777yzG66vjzoHVY00pZ4CMkElsMBBASYFmRTWOv744+2QQw6JhcAUGtS0Zaru1LVrVzvxxBODbvbkk0/aV1995banaj+77bab66dzovOlbaqlGxjSug888IBNmjTJhfZOO+00V0lKoT0dv86jzqdCN0cffbT7UR81BXf0ntq+++5rPXr0iIXHdG1oPB988IG7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YGeFBd59911744033Buauuqwww6zVq1aWfXq1W3BggX2wQcfmEIbakcffbT7cS/4lZaAgnpXXnmlW7esAy9hCAytX7/e3n77bVdh6Mgjj7RKlSql5V6aK3366aemz80JJ5xge+yxR0a7uumJm+3rKV+7KkJH7HOEde7YeUtQb5N9MO5DGz1+tG3ctNEO37ubDTxpYEbbZWUE8k2AwFC+nTHGiwACvgJh/SLt60E/BBBAAAEEEEAAAT8B3/tK335+o6QXAggggAACCCCAQFgFwnpfGZnAkC7MmTNn2sKFC901qrCQqp1st912VtrBIQWF5syZYzNmzDBVGFLTPwq1b1r5CsybN89uvfVW27RpkwuRnXHGGUmvh5deeslNV6bqQpdccom1aNGifAeeR3snMJRHJyvFUKdOnerCc7vuuqvVqVPHxowZY2+99ZaruLXLLrvYypUr7bvvvrM2bdrYNttsk2IrZqvXrrY/3XCam4LsqP2OsnO69y+w7s1P/M3GThlrNbdUF3rm+mcKvMcLBMImQGAobGeU40EAgVQCYf0inep4WY4AAggggAACCCBQOgK+95W+/UrnKNgqAggggAACCCCAQL4KhPW+MjKBIQVCVCVGj4mtQYMG1rBhQ9OjfrLRlixZYvpZvHixe0zcZtWqVa1jx46uakjie7wuOwFN9aTrQufjxhtvtBo1kk+DpCovV1xxha1bt866detm3bt3TzlIrfvbb7+5YIWqFGXaNC3aokWLXP9q1apl1F3BjbVr17rrWNOopdO0/rJly6x+/fpWpUqVdLq4ddasWeP61a1b11VjStUx08DQH3/84fwUzsqkApc+2/q86TMcVOEpaYUhjUU2mlZNPhUrVkx1mAWW+x6DNqLrR/vU36RMp7/zOZfpHOOzzz5rmtZR15SCjrquVS1tzz33tBUrVtiPP/7opvPr0qWLde3atYBF/AtNQ/bCBy+6Racecqo13TL9WHz78JuP7K4X73KLRlz5qNWvXT/+bZ4jECoBAkOhOp0cDAIIFCEQ1i/SRRwybyGAAAIIIIAAAgiUgoDvfaVvv1I4BDaJAAIIIIAAAgggkMcCYb2vjExgaPr06S5MoGpCe+21l3uuikNBxZ/g2tT7ChyoioZafICodu3aseoz6qcgRNAUDlL7/fffY1OOBe/pUdvVP9oVAvj888/dW3reunXr+NV4XoYCOoeXXXaZC2go5HDiiScWufcHH3zQJk6c6KYrU5WhxKYqLJo6Stda0HT9dOrUyY4//vjYtRO8p8errrrKhZD69u3rqk69+OKL9vPPP7sx6f0mTZpY7969XTUsvQ7a6NGj3TRqquZy8cUX2zvvvGNffPFFLJym6+3AAw+04447Lul+tR1N0Tdq1ChX+UqvFQbRP7A17ZU+I6naV199Zf/617/s119/ja2i0JCCVNpnEHJ57rnnbNy4ca6ajIJWahqXQjf6fF1zzTWx/nqidXQcmu5KYSQ1BVN23HFH69Gjh9WqVcstS/ylaeOef/5556Zzqv2rAlSvXr3cdHJjx441Tf83aNCgxK4pXyuwowo66hv8jdB2VZVM51KPyVqmxxCcR3kMHjzYRowYYT/99JML3wTTt+nv1H333ed2d/nll7u/IYn79jmXmRyjqqNNnjzZpk2b5sJMiftX0G6HHXZwVbpUZci3jZn4hd369K1WcauK9uwNz1jVylV9N0U/BHJegMBQzp8iBogAAlkSCOsX6SzxsBkEEEAAAQQQQACBNAV87yt9+6U5LFZDAAEEEEAAAQQQiIhAWO8rIxMY+vbbb10gQf/o79Chg7tsFQRQ0Gf+/PkuQBSEFLJ1TSvsoMCHQiP6UVhCTZU59A94VbXRFD+08hFQVZS77vpPNZOBAwda+/btixyIqv6ocpCCIwpHxDeFhR544AFXwUrvN27c2AXKVPFHrWnTptavXz9r1KhRfDcX9lFVGFVqURhJz7feemv3qKo2atqegkXxUz198MEH9uqrr7rgTatWrez77793gR/1VcWXoO29996mMFJi07ErhKJ96DrUP671WdD0eWoK/vTs2TOxmwu7qdqMmq5njUkhuWCfCvece+65biwvv/yyq96kCjZBuK5mzZqu+o8CMpdeemls+6qq9NBDD7nPhhYqTKcqQZpCUO9pfZ2jZs2axfroiY5Df5yDz64+c/JatWqV66+Q0dKlSzMKDGn92267LXZMMtWxannQzjrrLNt9992Dl+7R5xiC86hxK3QVH8IKAkMKoN15551uH9ddd12ha8jnXPoeowYxYcIEU7BNTS6nnXaabbvtts7dLSzBr7teuts+HPehtWnexu447/YSbImuCOS+AIGh3D9HjBABBLIjENYv0tnRYSsIIIAAAggggAAC6Qr43lf6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7du3t379+rmqQv5S69evt3HjxpmWklN1p6FDh1qVKqVf7efb1d/a5bdd4S+b1s8hFwyxow46Mq1jOQgBBDIXIDCUuRktEEAAAQQQQAABBBBAAAEEEEAAAQQQQAABBBBAAAEESk+AwFDpWWbFmSptXb27slDDrOgLnUAAAXNL1I0fP95WrFhh9erVs3bt2tmee+5pqjr00UcfmZaaq1y5sg0ZMsRatWpVJmQEhsqElZMiUCIBAkMl4qMxAggggAACCCCAAAIIIIAAAggggAACCCCAAAIIIIBACQUIDJUQkOYIIIBAcQIKBT311FP2ySef2M6dOwsc3rJlS+vdu7fpJxsCCOSPAIGh/JlrRooAAggggAACCCCAAAIIIIAAAggggAACCCCAAAIIZKMAgaFsnBX6hAACsRTYunWrLVmyxC1BVr9+fWvWrJmrNqQKQ2wIIJBfAgSG8mu+GS0CCCCAAAIIIIAAAggggAACCCCAAAIIIIAAAgggkG0CBIaybUboDwIIIIAAAgjEXoDAUOynmAEigAACCCCAAAIIIIAAAggggAACCCCAAAIIIIAAAlktQGAoq6eHziGAAAIIIIBAHAUIDMVxVhkTAggggAACCCCAAAIIIIAAAggggAACCCCAAAIIIJA7AgSGcmeu6CkCCCCAAAIIxESAwFBMJpJhIIAAAggggAACCCCAAAIIIIAAAggggAACCCCAAAI5KkBgKEcnjm4jgAACCCCAQO4KEBjK3bmj5wgggAACCCCAAAIIIIAAAggggAACCCCAAAIIIIBAHAQIDMVhFhkDAggggAACCOSUAIGhnJouOosAAggggAACCCCAAAIIIIAAAggggAACCCCAAAIIxE6AwFDsppQBIYAAAggggEC2CxAYyvYZon8IIIAAAggggAACCCCAAAIIIIAAAggggAACCCCAQLwFCAzFe34ZHQIIIIAAAghkoQCBoSycFLqEAAIIIIAAAggggAACCCCAAAIIIIAAAggggAACCOSRAIGhPJpshooAAggggAAC2SFAYCg75oFeIIAAAggggAACCCCAAAIIIIAAAggggAACCCCAAAL5KkBgKF9nnnEjgAACCCCAQIUJEBiqMHoujAACCCCAAAIIIIAAAggggAACCCCAAAIIIIAAAgggsFuAwBC3AQIIIIAAAgggUM4CBIbKGZzLIYAAAggggAAC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IIAAAggggAACCCCAAAIIIIAAAgikFiAwlNqGPQgggAACCCCAAAIIIIAAAggggAACCCCAAAIIIIAAAggggAACCCCAAAIIxE6AwFDsppQBIYAAAggggAACCCCAAAIIIIAAAggggAACCCCAAAIIIIAAAggggAACCCCQWoDAUGob9iCAAAIIIIAAAggggAACCCCAAAIIIIAAAggggAACCCCAAAIIIIAAAgggEDsBAkOxm1IGhAACCCCAAAIIIIAAAggggAACCCCAAAIIIIAAAggggAACCCCAAAIIIIBAagECQ6lt2IMAAggggAACCCCAAAIIIIAAAggggAACCCCAAAIIIIAAAggggAACCCCAQOwECAzFbkoZEAIIIIAAAggggAACCCCAAAIIIIAAAggggAACCCCAAAIIIIAAAggggAACqQUIDKW2YQ8CCCCAAAIIIIAAAggggAACCCCAAAIIIIAAAggggAACCCCAAAIIIIAAArETIDAUuyllQAgggAACCCCAAAIIIIAAAggggAACCCCAAAIIIIAAAggggAACCCCAAAIIpBYgMJTahj0IIIAAAggggAACCCCAAAIIIIAAAggggAACCCCAAAIIIIAAAggggAACCMROgMBQ7KaUASGAAAIIIIAAAggggAACCCCAAAIIIIAAAggggAACCCCAAAIIIIAAAgggkFqAwFBqG/YggAACCCCAAAIIIIAAAggggAACCCCAAAIIIIAAAggggAACCCCAAAIIIBA7AQJDsZtSBoQAAggggAACCCCAAAIIIIAAAggggAACCCCAAAIIIIAAAggggAACCCCAQGoBAkOpbdiDAAIIIIAAAggggAACCCCAAAIIIIAAAggggAACCCCAAAIIIIAAAggggEDsBP4/oInSneD1PzUAAAAASUVORK5CYII=)" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "GAv7c6NtLOpl", + "outputId": "94e3f040-b384-4fb9-a530-d86e2ee4e664", + "colab": { + "resources": { + "https://localhost:6006/?tensorboardColab=true": { + "data": "<!doctype html><meta name="tb-relative-root" content="./"><!doctype html><!--
@license
Copyright 2019 The TensorFlow Authors. All Rights Reserved.

Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at

    http://www.apache.org/licenses/LICENSE-2.0

Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
--><html><head><meta charset="utf-8">
<title>TensorBoard</title>
<link rel="shortcut icon" href="data:image/png;base64,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">
<link rel="apple-touch-icon" href="data:image/png;base64,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">

<style>
@font-face {
  font-family: 'Roboto';
  font-style: normal;
  font-weight: 400;
  src: local('Roboto'), local('Roboto-Regular'), url(/font-roboto/uYECMKoHcO9x1wdmbyHIm3-_kf6ByYO6CLYdB4HQE-Y.woff2) format('woff2');
  unicode-range: U+0400-045F, U+0490-0491, U+04B0-04B1, U+2116;
}
@font-face {
  font-family: 'Roboto';
  font-style: normal;
  font-weight: 400;
  src: local('Roboto'), local('Roboto-Regular'), url(/font-roboto/sTdaA6j0Psb920Vjv-mrzH-_kf6ByYO6CLYdB4HQE-Y.woff2) format('woff2');
  unicode-range: U+0460-052F, U+20B4, U+2DE0-2DFF, U+A640-A69F;
}
@font-face {
  font-family: 'Roboto';
  font-style: normal;
  font-weight: 400;
  src: local('Roboto'), local('Roboto-Regular'), url(/font-roboto/_VYFx-s824kXq_Ul2BHqYH-_kf6ByYO6CLYdB4HQE-Y.woff2) format('woff2');
  unicode-range: U+0370-03FF;
}
@font-face {
  font-family: 'Roboto';
  font-style: normal;
  font-weight: 400;
  src: local('Roboto'), local('Roboto-Regular'), url(/font-roboto/tnj4SB6DNbdaQnsM8CFqBX-_kf6ByYO6CLYdB4HQE-Y.woff2) format('woff2');
  unicode-range: U+1F00-1FFF;
}
@font-face {
  font-family: 'Roboto';
  font-style: normal;
  font-weight: 400;
  src: local('Roboto'), local('Roboto-Regular'), url(/font-roboto/oMMgfZMQthOryQo9n22dcuvvDin1pK8aKteLpeZ5c0A.woff2) format('woff2');
  unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2212, U+2215;
}
@font-face {
  font-family: 'Roboto';
  font-style: normal;
  font-weight: 400;
  src: local('Roboto'), local('Roboto-Regular'), url(/font-roboto/Ks_cVxiCiwUWVsFWFA3Bjn-_kf6ByYO6CLYdB4HQE-Y.woff2) format('woff2');
  unicode-range: U+0100-024F, U+1E00-1EFF, U+20A0-20AB, U+20AD-20CF, U+2C60-2C7F, U+A720-A7FF;
}
@font-face {
  font-family: 'Roboto';
  font-style: normal;
  font-weight: 400;
  src: local('Roboto'), local('Roboto-Regular'), url(/font-roboto/NJ4vxlgWwWbEsv18dAhqnn-_kf6ByYO6CLYdB4HQE-Y.woff2) format('woff2');
  unicode-range: U+0102-0103, U+1EA0-1EF9, U+20AB;
}
@font-face {
  font-family: 'Roboto';
  font-style: normal;
  font-weight: 700;
  src: local('Roboto Bold'), local('Roboto-Bold'), url(/font-roboto/isZ-wbCXNKAbnjo6_TwHToX0hVgzZQUfRDuZrPvH3D8.woff2) format('woff2');
  unicode-range: U+0400-045F, U+0490-0491, U+04B0-04B1, U+2116;
}
@font-face {
  font-family: 'Roboto';
  font-style: normal;
  font-weight: 700;
  src: local('Roboto Bold'), local('Roboto-Bold'), url(/font-roboto/77FXFjRbGzN4aCrSFhlh3oX0hVgzZQUfRDuZrPvH3D8.woff2) format('woff2');
  unicode-range: U+0460-052F, U+20B4, U+2DE0-2DFF, U+A640-A69F;
}
@font-face {
  font-family: 'Roboto';
  font-style: normal;
  font-weight: 700;
  src: local('Roboto Bold'), local('Roboto-Bold'), url(/font-roboto/jSN2CGVDbcVyCnfJfjSdfIX0hVgzZQUfRDuZrPvH3D8.woff2) format('woff2');
  unicode-range: U+0370-03FF;
}
@font-face {
  font-family: 'Roboto';
  font-style: normal;
  font-weight: 700;
  src: local('Roboto Bold'), local('Roboto-Bold'), url(/font-roboto/UX6i4JxQDm3fVTc1CPuwqoX0hVgzZQUfRDuZrPvH3D8.woff2) format('woff2');
  unicode-range: U+1F00-1FFF;
}
@font-face {
  font-family: 'Roboto';
  font-style: normal;
  font-weight: 700;
  src: local('Roboto Bold'), local('Roboto-Bold'), url(/font-roboto/d-6IYplOFocCacKzxwXSOJBw1xU1rKptJj_0jans920.woff2) format('woff2');
  unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2212, U+2215;
}
@font-face {
  font-family: 'Roboto';
  font-style: normal;
  font-weight: 700;
  src: local('Roboto Bold'), local('Roboto-Bold'), url(/font-roboto/97uahxiqZRoncBaCEI3aW4X0hVgzZQUfRDuZrPvH3D8.woff2) format('woff2');
  unicode-range: U+0100-024F, U+1E00-1EFF, U+20A0-20AB, U+20AD-20CF, U+2C60-2C7F, U+A720-A7FF;
}
@font-face {
  font-family: 'Roboto';
  font-style: normal;
  font-weight: 700;
  src: local('Roboto Bold'), local('Roboto-Bold'), url(/font-roboto/PwZc-YbIL414wB9rB1IAPYX0hVgzZQUfRDuZrPvH3D8.woff2) format('woff2');
  unicode-range: U+0102-0103, U+1EA0-1EF9, U+20AB;
}
@font-face {
  font-family: 'Roboto';
  font-style: italic;
  font-weight: 700;
  src: local('Roboto Bold Italic'), local('Roboto-BoldItalic'), url(/font-roboto/t6Nd4cfPRhZP44Q5QAjcC14sYYdJg5dU2qzJEVSuta0.woff2) format('woff2');
  unicode-range: U+0400-045F, U+0490-0491, U+04B0-04B1, U+2116;
}
@font-face {
  font-family: 'Roboto';
  font-style: italic;
  font-weight: 700;
  src: local('Roboto Bold Italic'), local('Roboto-BoldItalic'), url(/font-roboto/t6Nd4cfPRhZP44Q5QAjcC_ZraR2Tg8w2lzm7kLNL0-w.woff2) format('woff2');
  unicode-range: U+0460-052F, U+20B4, U+2DE0-2DFF, U+A640-A69F;
}
@font-face {
  font-family: 'Roboto';
  font-style: italic;
  font-weight: 700;
  src: local('Roboto Bold Italic'), local('Roboto-BoldItalic'), url(/font-roboto/t6Nd4cfPRhZP44Q5QAjcCwt_Rm691LTebKfY2ZkKSmI.woff2) format('woff2');
  unicode-range: U+0370-03FF;
}
@font-face {
  font-family: 'Roboto';
  font-style: italic;
  font-weight: 700;
  src: local('Roboto Bold Italic'), local('Roboto-BoldItalic'), url(/font-roboto/t6Nd4cfPRhZP44Q5QAjcC1BW26QxpSj-_ZKm_xT4hWw.woff2) format('woff2');
  unicode-range: U+1F00-1FFF;
}
@font-face {
  font-family: 'Roboto';
  font-style: italic;
  font-weight: 700;
  src: local('Roboto Bold Italic'), local('Roboto-BoldItalic'), url(/font-roboto/t6Nd4cfPRhZP44Q5QAjcC4gp9Q8gbYrhqGlRav_IXfk.woff2) format('woff2');
  unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2212, U+2215;
}
@font-face {
  font-family: 'Roboto';
  font-style: italic;
  font-weight: 700;
  src: local('Roboto Bold Italic'), local('Roboto-BoldItalic'), url(/font-roboto/t6Nd4cfPRhZP44Q5QAjcC6E8kM4xWR1_1bYURRojRGc.woff2) format('woff2');
  unicode-range: U+0100-024F, U+1E00-1EFF, U+20A0-20AB, U+20AD-20CF, U+2C60-2C7F, U+A720-A7FF;
}
@font-face {
  font-family: 'Roboto';
  font-style: italic;
  font-weight: 700;
  src: local('Roboto Bold Italic'), local('Roboto-BoldItalic'), url(/font-roboto/t6Nd4cfPRhZP44Q5QAjcC9DiNsR5a-9Oe_Ivpu8XWlY.woff2) format('woff2');
  unicode-range: U+0102-0103, U+1EA0-1EF9, U+20AB;
}
@font-face {
  font-family: 'Roboto';
  font-style: italic;
  font-weight: 400;
  src: local('Roboto Italic'), local('Roboto-Italic'), url(/font-roboto/OpXUqTo0UgQQhGj_SFdLWBkAz4rYn47Zy2rvigWQf6w.woff2) format('woff2');
  unicode-range: U+0400-045F, U+0490-0491, U+04B0-04B1, U+2116;
}
@font-face {
  font-family: 'Roboto';
  font-style: italic;
  font-weight: 400;
  src: local('Roboto Italic'), local('Roboto-Italic'), url(/font-roboto/WxrXJa0C3KdtC7lMafG4dRkAz4rYn47Zy2rvigWQf6w.woff2) format('woff2');
  unicode-range: U+0460-052F, U+20B4, U+2DE0-2DFF, U+A640-A69F;
}
@font-face {
  font-family: 'Roboto';
  font-style: italic;
  font-weight: 400;
  src: local('Roboto Italic'), local('Roboto-Italic'), url(/font-roboto/cDKhRaXnQTOVbaoxwdOr9xkAz4rYn47Zy2rvigWQf6w.woff2) format('woff2');
  unicode-range: U+0370-03FF;
}
@font-face {
  font-family: 'Roboto';
  font-style: italic;
  font-weight: 400;
  src: local('Roboto Italic'), local('Roboto-Italic'), url(/font-roboto/1hZf02POANh32k2VkgEoUBkAz4rYn47Zy2rvigWQf6w.woff2) format('woff2');
  unicode-range: U+1F00-1FFF;
}
@font-face {
  font-family: 'Roboto';
  font-style: italic;
  font-weight: 400;
  src: local('Roboto Italic'), local('Roboto-Italic'), url(/font-roboto/vPcynSL0qHq_6dX7lKVByXYhjbSpvc47ee6xR_80Hnw.woff2) format('woff2');
  unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2212, U+2215;
}
@font-face {
  font-family: 'Roboto';
  font-style: italic;
  font-weight: 400;
  src: local('Roboto Italic'), local('Roboto-Italic'), url(/font-roboto/vSzulfKSK0LLjjfeaxcREhkAz4rYn47Zy2rvigWQf6w.woff2) format('woff2');
  unicode-range: U+0100-024F, U+1E00-1EFF, U+20A0-20AB, U+20AD-20CF, U+2C60-2C7F, U+A720-A7FF;
}
@font-face {
  font-family: 'Roboto';
  font-style: italic;
  font-weight: 400;
  src: local('Roboto Italic'), local('Roboto-Italic'), url(/font-roboto/K23cxWVTrIFD6DJsEVi07RkAz4rYn47Zy2rvigWQf6w.woff2) format('woff2');
  unicode-range: U+0102-0103, U+1EA0-1EF9, U+20AB;
}
@font-face {
  font-family: 'Roboto';
  font-style: normal;
  font-weight: 300;
  src: local('Roboto Light'), local('Roboto-Light'), url(/font-roboto/Fl4y0QdOxyyTHEGMXX8kcYX0hVgzZQUfRDuZrPvH3D8.woff2) format('woff2');
  unicode-range: U+0400-045F, U+0490-0491, U+04B0-04B1, U+2116;
}
@font-face {
  font-family: 'Roboto';
  font-style: normal;
  font-weight: 300;
  src: local('Roboto Light'), local('Roboto-Light'), url(/font-roboto/0eC6fl06luXEYWpBSJvXCIX0hVgzZQUfRDuZrPvH3D8.woff2) format('woff2');
  unicode-range: U+0460-052F, U+20B4, U+2DE0-2DFF, U+A640-A69F;
}
@font-face {
  font-family: 'Roboto';
  font-style: normal;
  font-weight: 300;
  src: local('Roboto Light'), local('Roboto-Light'), url(/font-roboto/I3S1wsgSg9YCurV6PUkTOYX0hVgzZQUfRDuZrPvH3D8.woff2) format('woff2');
  unicode-range: U+0370-03FF;
}
@font-face {
  font-family: 'Roboto';
  font-style: normal;
  font-weight: 300;
  src: local('Roboto Light'), local('Roboto-Light'), url(/font-roboto/-L14Jk06m6pUHB-5mXQQnYX0hVgzZQUfRDuZrPvH3D8.woff2) format('woff2');
  unicode-range: U+1F00-1FFF;
}
@font-face {
  font-family: 'Roboto';
  font-style: normal;
  font-weight: 300;
  src: local('Roboto Light'), local('Roboto-Light'), url(/font-roboto/Hgo13k-tfSpn0qi1SFdUfZBw1xU1rKptJj_0jans920.woff2) format('woff2');
  unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2212, U+2215;
}
@font-face {
  font-family: 'Roboto';
  font-style: normal;
  font-weight: 300;
  src: local('Roboto Light'), local('Roboto-Light'), url(/font-roboto/Pru33qjShpZSmG3z6VYwnYX0hVgzZQUfRDuZrPvH3D8.woff2) format('woff2');
  unicode-range: U+0100-024F, U+1E00-1EFF, U+20A0-20AB, U+20AD-20CF, U+2C60-2C7F, U+A720-A7FF;
}
@font-face {
  font-family: 'Roboto';
  font-style: normal;
  font-weight: 300;
  src: local('Roboto Light'), local('Roboto-Light'), url(/font-roboto/NYDWBdD4gIq26G5XYbHsFIX0hVgzZQUfRDuZrPvH3D8.woff2) format('woff2');
  unicode-range: U+0102-0103, U+1EA0-1EF9, U+20AB;
}
@font-face {
  font-family: 'Roboto';
  font-style: italic;
  font-weight: 300;
  src: local('Roboto Light Italic'), local('Roboto-LightItalic'), url(/font-roboto/7m8l7TlFO-S3VkhHuR0at14sYYdJg5dU2qzJEVSuta0.woff2) format('woff2');
  unicode-range: U+0400-045F, U+0490-0491, U+04B0-04B1, U+2116;
}
@font-face {
  font-family: 'Roboto';
  font-style: italic;
  font-weight: 300;
  src: local('Roboto Light Italic'), local('Roboto-LightItalic'), url(/font-roboto/7m8l7TlFO-S3VkhHuR0at_ZraR2Tg8w2lzm7kLNL0-w.woff2) format('woff2');
  unicode-range: U+0460-052F, U+20B4, U+2DE0-2DFF, U+A640-A69F;
}
@font-face {
  font-family: 'Roboto';
  font-style: italic;
  font-weight: 300;
  src: local('Roboto Light Italic'), local('Roboto-LightItalic'), url(/font-roboto/7m8l7TlFO-S3VkhHuR0atwt_Rm691LTebKfY2ZkKSmI.woff2) format('woff2');
  unicode-range: U+0370-03FF;
}
@font-face {
  font-family: 'Roboto';
  font-style: italic;
  font-weight: 300;
  src: local('Roboto Light Italic'), local('Roboto-LightItalic'), url(/font-roboto/7m8l7TlFO-S3VkhHuR0at1BW26QxpSj-_ZKm_xT4hWw.woff2) format('woff2');
  unicode-range: U+1F00-1FFF;
}
@font-face {
  font-family: 'Roboto';
  font-style: italic;
  font-weight: 300;
  src: local('Roboto Light Italic'), local('Roboto-LightItalic'), url(/font-roboto/7m8l7TlFO-S3VkhHuR0at4gp9Q8gbYrhqGlRav_IXfk.woff2) format('woff2');
  unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2212, U+2215;
}
@font-face {
  font-family: 'Roboto';
  font-style: italic;
  font-weight: 300;
  src: local('Roboto Light Italic'), local('Roboto-LightItalic'), url(/font-roboto/7m8l7TlFO-S3VkhHuR0at6E8kM4xWR1_1bYURRojRGc.woff2) format('woff2');
  unicode-range: U+0100-024F, U+1E00-1EFF, U+20A0-20AB, U+20AD-20CF, U+2C60-2C7F, U+A720-A7FF;
}
@font-face {
  font-family: 'Roboto';
  font-style: italic;
  font-weight: 300;
  src: local('Roboto Light Italic'), local('Roboto-LightItalic'), url(/font-roboto/7m8l7TlFO-S3VkhHuR0at9DiNsR5a-9Oe_Ivpu8XWlY.woff2) format('woff2');
  unicode-range: U+0102-0103, U+1EA0-1EF9, U+20AB;
}
@font-face {
  font-family: 'Roboto';
  font-style: normal;
  font-weight: 500;
  src: local('Roboto Medium'), local('Roboto-Medium'), url(/font-roboto/oHi30kwQWvpCWqAhzHcCSIX0hVgzZQUfRDuZrPvH3D8.woff2) format('woff2');
  unicode-range: U+0400-045F, U+0490-0491, U+04B0-04B1, U+2116;
}
@font-face {
  font-family: 'Roboto';
  font-style: normal;
  font-weight: 500;
  src: local('Roboto Medium'), local('Roboto-Medium'), url(/font-roboto/ZLqKeelYbATG60EpZBSDy4X0hVgzZQUfRDuZrPvH3D8.woff2) format('woff2');
  unicode-range: U+0460-052F, U+20B4, U+2DE0-2DFF, U+A640-A69F;
}
@font-face {
  font-family: 'Roboto';
  font-style: normal;
  font-weight: 500;
  src: local('Roboto Medium'), local('Roboto-Medium'), url(/font-roboto/mx9Uck6uB63VIKFYnEMXrYX0hVgzZQUfRDuZrPvH3D8.woff2) format('woff2');
  unicode-range: U+0370-03FF;
}
@font-face {
  font-family: 'Roboto';
  font-style: normal;
  font-weight: 500;
  src: local('Roboto Medium'), local('Roboto-Medium'), url(/font-roboto/rGvHdJnr2l75qb0YND9NyIX0hVgzZQUfRDuZrPvH3D8.woff2) format('woff2');
  unicode-range: U+1F00-1FFF;
}
@font-face {
  font-family: 'Roboto';
  font-style: normal;
  font-weight: 500;
  src: local('Roboto Medium'), local('Roboto-Medium'), url(/font-roboto/RxZJdnzeo3R5zSexge8UUZBw1xU1rKptJj_0jans920.woff2) format('woff2');
  unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2212, U+2215;
}
@font-face {
  font-family: 'Roboto';
  font-style: normal;
  font-weight: 500;
  src: local('Roboto Medium'), local('Roboto-Medium'), url(/font-roboto/oOeFwZNlrTefzLYmlVV1UIX0hVgzZQUfRDuZrPvH3D8.woff2) format('woff2');
  unicode-range: U+0100-024F, U+1E00-1EFF, U+20A0-20AB, U+20AD-20CF, U+2C60-2C7F, U+A720-A7FF;
}
@font-face {
  font-family: 'Roboto';
  font-style: normal;
  font-weight: 500;
  src: local('Roboto Medium'), local('Roboto-Medium'), url(/font-roboto/mbmhprMH69Zi6eEPBYVFhYX0hVgzZQUfRDuZrPvH3D8.woff2) format('woff2');
  unicode-range: U+0102-0103, U+1EA0-1EF9, U+20AB;
}
@font-face {
  font-family: 'Roboto';
  font-style: italic;
  font-weight: 500;
  src: local('Roboto Medium Italic'), local('Roboto-MediumItalic'), url(/font-roboto/OLffGBTaF0XFOW1gnuHF0V4sYYdJg5dU2qzJEVSuta0.woff2) format('woff2');
  unicode-range: U+0400-045F, U+0490-0491, U+04B0-04B1, U+2116;
}
@font-face {
  font-family: 'Roboto';
  font-style: italic;
  font-weight: 500;
  src: local('Roboto Medium Italic'), local('Roboto-MediumItalic'), url(/font-roboto/OLffGBTaF0XFOW1gnuHF0fZraR2Tg8w2lzm7kLNL0-w.woff2) format('woff2');
  unicode-range: U+0460-052F, U+20B4, U+2DE0-2DFF, U+A640-A69F;
}
@font-face {
  font-family: 'Roboto';
  font-style: italic;
  font-weight: 500;
  src: local('Roboto Medium Italic'), local('Roboto-MediumItalic'), url(/font-roboto/OLffGBTaF0XFOW1gnuHF0Qt_Rm691LTebKfY2ZkKSmI.woff2) format('woff2');
  unicode-range: U+0370-03FF;
}
@font-face {
  font-family: 'Roboto';
  font-style: italic;
  font-weight: 500;
  src: local('Roboto Medium Italic'), local('Roboto-MediumItalic'), url(/font-roboto/OLffGBTaF0XFOW1gnuHF0VBW26QxpSj-_ZKm_xT4hWw.woff2) format('woff2');
  unicode-range: U+1F00-1FFF;
}
@font-face {
  font-family: 'Roboto';
  font-style: italic;
  font-weight: 500;
  src: local('Roboto Medium Italic'), local('Roboto-MediumItalic'), url(/font-roboto/OLffGBTaF0XFOW1gnuHF0Ygp9Q8gbYrhqGlRav_IXfk.woff2) format('woff2');
  unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2212, U+2215;
}
@font-face {
  font-family: 'Roboto';
  font-style: italic;
  font-weight: 500;
  src: local('Roboto Medium Italic'), local('Roboto-MediumItalic'), url(/font-roboto/OLffGBTaF0XFOW1gnuHF0aE8kM4xWR1_1bYURRojRGc.woff2) format('woff2');
  unicode-range: U+0100-024F, U+1E00-1EFF, U+20A0-20AB, U+20AD-20CF, U+2C60-2C7F, U+A720-A7FF;
}
@font-face {
  font-family: 'Roboto';
  font-style: italic;
  font-weight: 500;
  src: local('Roboto Medium Italic'), local('Roboto-MediumItalic'), url(/font-roboto/OLffGBTaF0XFOW1gnuHF0dDiNsR5a-9Oe_Ivpu8XWlY.woff2) format('woff2');
  unicode-range: U+0102-0103, U+1EA0-1EF9, U+20AB;
}
@font-face {
  font-family: 'Roboto Mono';
  font-style: normal;
  font-weight: 400;
  src: local('Roboto Mono'), local('RobotoMono-Regular'), url(/font-roboto/hMqPNLsu_dywMa4C_DEpY14sYYdJg5dU2qzJEVSuta0.woff2) format('woff2');
  unicode-range: U+0400-045F, U+0490-0491, U+04B0-04B1, U+2116;
}
@font-face {
  font-family: 'Roboto Mono';
  font-style: normal;
  font-weight: 400;
  src: local('Roboto Mono'), local('RobotoMono-Regular'), url(/font-roboto/hMqPNLsu_dywMa4C_DEpY_ZraR2Tg8w2lzm7kLNL0-w.woff2) format('woff2');
  unicode-range: U+0460-052F, U+20B4, U+2DE0-2DFF, U+A640-A69F;
}
@font-face {
  font-family: 'Roboto Mono';
  font-style: normal;
  font-weight: 400;
  src: local('Roboto Mono'), local('RobotoMono-Regular'), url(/font-roboto/hMqPNLsu_dywMa4C_DEpYwt_Rm691LTebKfY2ZkKSmI.woff2) format('woff2');
  unicode-range: U+0370-03FF;
}
@font-face {
  font-family: 'Roboto Mono';
  font-style: normal;
  font-weight: 400;
  src: local('Roboto Mono'), local('RobotoMono-Regular'), url(/font-roboto/hMqPNLsu_dywMa4C_DEpY1BW26QxpSj-_ZKm_xT4hWw.woff2) format('woff2');
  unicode-range: U+1F00-1FFF;
}
@font-face {
  font-family: 'Roboto Mono';
  font-style: normal;
  font-weight: 400;
  src: local('Roboto Mono'), local('RobotoMono-Regular'), url(/font-roboto/hMqPNLsu_dywMa4C_DEpY4gp9Q8gbYrhqGlRav_IXfk.woff2) format('woff2');
  unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2212, U+2215;
}
@font-face {
  font-family: 'Roboto Mono';
  font-style: normal;
  font-weight: 400;
  src: local('Roboto Mono'), local('RobotoMono-Regular'), url(/font-roboto/hMqPNLsu_dywMa4C_DEpY6E8kM4xWR1_1bYURRojRGc.woff2) format('woff2');
  unicode-range: U+0100-024F, U+1E00-1EFF, U+20A0-20AB, U+20AD-20CF, U+2C60-2C7F, U+A720-A7FF;
}
@font-face {
  font-family: 'Roboto Mono';
  font-style: normal;
  font-weight: 400;
  src: local('Roboto Mono'), local('RobotoMono-Regular'), url(/font-roboto/hMqPNLsu_dywMa4C_DEpY9DiNsR5a-9Oe_Ivpu8XWlY.woff2) format('woff2');
  unicode-range: U+0102-0103, U+1EA0-1EF9, U+20AB;
}
@font-face {
  font-family: 'Roboto Mono';
  font-style: normal;
  font-weight: 700;
  src: local('Roboto Mono Bold'), local('RobotoMono-Bold'), url(/font-roboto/N4duVc9C58uwPiY8_59Fz1x-M1I1w5OMiqnVF8xBLhU.woff2) format('woff2');
  unicode-range: U+0400-045F, U+0490-0491, U+04B0-04B1, U+2116;
}
@font-face {
  font-family: 'Roboto Mono';
  font-style: normal;
  font-weight: 700;
  src: local('Roboto Mono Bold'), local('RobotoMono-Bold'), url(/font-roboto/N4duVc9C58uwPiY8_59FzwXaAXup5mZlfK6xRLrhsco.woff2) format('woff2');
  unicode-range: U+0460-052F, U+20B4, U+2DE0-2DFF, U+A640-A69F;
}
@font-face {
  font-family: 'Roboto Mono';
  font-style: normal;
  font-weight: 700;
  src: local('Roboto Mono Bold'), local('RobotoMono-Bold'), url(/font-roboto/N4duVc9C58uwPiY8_59Fzwn6Wqxo-xwxilDXPU8chVU.woff2) format('woff2');
  unicode-range: U+0370-03FF;
}
@font-face {
  font-family: 'Roboto Mono';
  font-style: normal;
  font-weight: 700;
  src: local('Roboto Mono Bold'), local('RobotoMono-Bold'), url(/font-roboto/N4duVc9C58uwPiY8_59Fz1T7aJLK6nKpn36IMwTcMMc.woff2) format('woff2');
  unicode-range: U+1F00-1FFF;
}
@font-face {
  font-family: 'Roboto Mono';
  font-style: normal;
  font-weight: 700;
  src: local('Roboto Mono Bold'), local('RobotoMono-Bold'), url(/font-roboto/N4duVc9C58uwPiY8_59Fz_79_ZuUxCigM2DespTnFaw.woff2) format('woff2');
  unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2212, U+2215;
}
@font-face {
  font-family: 'Roboto Mono';
  font-style: normal;
  font-weight: 700;
  src: local('Roboto Mono Bold'), local('RobotoMono-Bold'), url(/font-roboto/N4duVc9C58uwPiY8_59Fz4gd9OEPUCN3AdYW0e8tat4.woff2) format('woff2');
  unicode-range: U+0100-024F, U+1E00-1EFF, U+20A0-20AB, U+20AD-20CF, U+2C60-2C7F, U+A720-A7FF;
}
@font-face {
  font-family: 'Roboto Mono';
  font-style: normal;
  font-weight: 700;
  src: local('Roboto Mono Bold'), local('RobotoMono-Bold'), url(/font-roboto/N4duVc9C58uwPiY8_59Fz8bIQSYZnWLaWC9QNCpTK_U.woff2) format('woff2');
  unicode-range: U+0102-0103, U+1EA0-1EF9, U+20AB;
}
</style>



<style>.mat-badge-content{font-weight:600;font-size:12px;font-family:Roboto, "Helvetica Neue", sans-serif}.mat-badge-small .mat-badge-content{font-size:9px}.mat-badge-large .mat-badge-content{font-size:24px}.mat-h1,.mat-headline,.mat-typography h1{font:400 24px/32px Roboto, "Helvetica Neue", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h2,.mat-title,.mat-typography h2{font:500 20px/32px Roboto, "Helvetica Neue", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h3,.mat-subheading-2,.mat-typography h3{font:400 16px/28px Roboto, "Helvetica Neue", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h4,.mat-subheading-1,.mat-typography h4{font:400 15px/24px Roboto, "Helvetica Neue", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h5,.mat-typography h5{font:400 calc(14px * 0.83)/20px Roboto, "Helvetica Neue", sans-serif;margin:0 0 12px}.mat-h6,.mat-typography h6{font:400 calc(14px * 0.67)/20px Roboto, "Helvetica Neue", sans-serif;margin:0 0 12px}.mat-body-strong,.mat-body-2{font:500 14px/24px Roboto, "Helvetica Neue", sans-serif;letter-spacing:normal}.mat-body,.mat-body-1,.mat-typography{font:400 14px/20px Roboto, "Helvetica Neue", sans-serif;letter-spacing:normal}.mat-body p,.mat-body-1 p,.mat-typography p{margin:0 0 12px}.mat-small,.mat-caption{font:400 12px/20px Roboto, "Helvetica Neue", sans-serif;letter-spacing:normal}.mat-display-4,.mat-typography .mat-display-4{font:300 112px/112px Roboto, "Helvetica Neue", sans-serif;letter-spacing:-0.05em;margin:0 0 56px}.mat-display-3,.mat-typography .mat-display-3{font:400 56px/56px Roboto, "Helvetica Neue", sans-serif;letter-spacing:-0.02em;margin:0 0 64px}.mat-display-2,.mat-typography .mat-display-2{font:400 45px/48px Roboto, "Helvetica Neue", sans-serif;letter-spacing:-0.005em;margin:0 0 64px}.mat-display-1,.mat-typography .mat-display-1{font:400 34px/40px Roboto, "Helvetica Neue", sans-serif;letter-spacing:normal;margin:0 0 64px}.mat-bottom-sheet-container{font:400 14px/20px Roboto, "Helvetica Neue", sans-serif;letter-spacing:normal}.mat-button,.mat-raised-button,.mat-icon-button,.mat-stroked-button,.mat-flat-button,.mat-fab,.mat-mini-fab{font-family:Roboto, "Helvetica Neue", sans-serif;font-size:14px;font-weight:500}.mat-button-toggle{font-family:Roboto, "Helvetica Neue", sans-serif}.mat-card{font-family:Roboto, "Helvetica Neue", sans-serif}.mat-card-title{font-size:24px;font-weight:500}.mat-card-header .mat-card-title{font-size:20px}.mat-card-subtitle,.mat-card-content{font-size:14px}.mat-checkbox{font-family:Roboto, "Helvetica Neue", sans-serif}.mat-checkbox-layout .mat-checkbox-label{line-height:24px}.mat-chip{font-size:14px;font-weight:500}.mat-chip .mat-chip-trailing-icon.mat-icon,.mat-chip .mat-chip-remove.mat-icon{font-size:18px}.mat-table{font-family:Roboto, "Helvetica Neue", sans-serif}.mat-header-cell{font-size:12px;font-weight:500}.mat-cell,.mat-footer-cell{font-size:14px}.mat-calendar{font-family:Roboto, "Helvetica Neue", sans-serif}.mat-calendar-body{font-size:13px}.mat-calendar-body-label,.mat-calendar-period-button{font-size:14px;font-weight:500}.mat-calendar-table-header th{font-size:11px;font-weight:400}.mat-dialog-title{font:500 20px/32px Roboto, "Helvetica Neue", sans-serif;letter-spacing:normal}.mat-expansion-panel-header{font-family:Roboto, "Helvetica Neue", sans-serif;font-size:15px;font-weight:400}.mat-expansion-panel-content{font:400 14px/20px Roboto, "Helvetica Neue", sans-serif;letter-spacing:normal}.mat-form-field{font-size:inherit;font-weight:400;line-height:1.125;font-family:Roboto, "Helvetica Neue", sans-serif;letter-spacing:normal}.mat-form-field-wrapper{padding-bottom:1.34375em}.mat-form-field-prefix .mat-icon,.mat-form-field-suffix .mat-icon{font-size:150%;line-height:1.125}.mat-form-field-prefix .mat-icon-button,.mat-form-field-suffix .mat-icon-button{height:1.5em;width:1.5em}.mat-form-field-prefix .mat-icon-button .mat-icon,.mat-form-field-suffix .mat-icon-button .mat-icon{height:1.125em;line-height:1.125}.mat-form-field-infix{padding:.5em 0;border-top:.84375em solid transparent}.mat-form-field-can-float.mat-form-field-should-float .mat-form-field-label,.mat-form-field-can-float .mat-input-server:focus+.mat-form-field-label-wrapper .mat-form-field-label{transform:translateY(-1.34375em) scale(0.75);width:133.3333333333%}.mat-form-field-can-float .mat-input-server[label]:not(:label-shown)+.mat-form-field-label-wrapper .mat-form-field-label{transform:translateY(-1.34374em) scale(0.75);width:133.3333433333%}.mat-form-field-label-wrapper{top:-0.84375em;padding-top:.84375em}.mat-form-field-label{top:1.34375em}.mat-form-field-underline{bottom:1.34375em}.mat-form-field-subscript-wrapper{font-size:75%;margin-top:.6666666667em;top:calc(100% - 1.7916666667em)}.mat-form-field-appearance-legacy .mat-form-field-wrapper{padding-bottom:1.25em}.mat-form-field-appearance-legacy .mat-form-field-infix{padding:.4375em 0}.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float .mat-form-field-label,.mat-form-field-appearance-legacy.mat-form-field-can-float .mat-input-server:focus+.mat-form-field-label-wrapper .mat-form-field-label{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.001px);-ms-transform:translateY(-1.28125em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-legacy.mat-form-field-can-float .mat-form-field-autofill-control:-webkit-autofill+.mat-form-field-label-wrapper .mat-form-field-label{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00101px);-ms-transform:translateY(-1.28124em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-legacy.mat-form-field-can-float .mat-input-server[label]:not(:label-shown)+.mat-form-field-label-wrapper .mat-form-field-label{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00102px);-ms-transform:translateY(-1.28123em) scale(0.75);width:133.3333533333%}.mat-form-field-appearance-legacy .mat-form-field-label{top:1.28125em}.mat-form-field-appearance-legacy .mat-form-field-underline{bottom:1.25em}.mat-form-field-appearance-legacy .mat-form-field-subscript-wrapper{margin-top:.5416666667em;top:calc(100% - 1.6666666667em)}@media print{.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float .mat-form-field-label,.mat-form-field-appearance-legacy.mat-form-field-can-float .mat-input-server:focus+.mat-form-field-label-wrapper .mat-form-field-label{transform:translateY(-1.28122em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float .mat-form-field-autofill-control:-webkit-autofill+.mat-form-field-label-wrapper .mat-form-field-label{transform:translateY(-1.28121em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float .mat-input-server[label]:not(:label-shown)+.mat-form-field-label-wrapper .mat-form-field-label{transform:translateY(-1.2812em) scale(0.75)}}.mat-form-field-appearance-fill .mat-form-field-infix{padding:.25em 0 .75em 0}.mat-form-field-appearance-fill .mat-form-field-label{top:1.09375em;margin-top:-0.5em}.mat-form-field-appearance-fill.mat-form-field-can-float.mat-form-field-should-float .mat-form-field-label,.mat-form-field-appearance-fill.mat-form-field-can-float .mat-input-server:focus+.mat-form-field-label-wrapper .mat-form-field-label{transform:translateY(-0.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-fill.mat-form-field-can-float .mat-input-server[label]:not(:label-shown)+.mat-form-field-label-wrapper .mat-form-field-label{transform:translateY(-0.59374em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-outline .mat-form-field-infix{padding:1em 0 1em 0}.mat-form-field-appearance-outline .mat-form-field-label{top:1.84375em;margin-top:-0.25em}.mat-form-field-appearance-outline.mat-form-field-can-float.mat-form-field-should-float .mat-form-field-label,.mat-form-field-appearance-outline.mat-form-field-can-float .mat-input-server:focus+.mat-form-field-label-wrapper .mat-form-field-label{transform:translateY(-1.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-outline.mat-form-field-can-float .mat-input-server[label]:not(:label-shown)+.mat-form-field-label-wrapper .mat-form-field-label{transform:translateY(-1.59374em) scale(0.75);width:133.3333433333%}.mat-grid-tile-header,.mat-grid-tile-footer{font-size:14px}.mat-grid-tile-header .mat-line,.mat-grid-tile-footer .mat-line{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-grid-tile-header .mat-line:nth-child(n+2),.mat-grid-tile-footer .mat-line:nth-child(n+2){font-size:12px}input.mat-input-element{margin-top:-0.0625em}.mat-menu-item{font-family:Roboto, "Helvetica Neue", sans-serif;font-size:14px;font-weight:400}.mat-paginator,.mat-paginator-page-size .mat-select-trigger{font-family:Roboto, "Helvetica Neue", sans-serif;font-size:12px}.mat-radio-button{font-family:Roboto, "Helvetica Neue", sans-serif}.mat-select{font-family:Roboto, "Helvetica Neue", sans-serif}.mat-select-trigger{height:1.125em}.mat-slide-toggle-content{font-family:Roboto, "Helvetica Neue", sans-serif}.mat-slider-thumb-label-text{font-family:Roboto, "Helvetica Neue", sans-serif;font-size:12px;font-weight:500}.mat-stepper-vertical,.mat-stepper-horizontal{font-family:Roboto, "Helvetica Neue", sans-serif}.mat-step-label{font-size:14px;font-weight:400}.mat-step-sub-label-error{font-weight:normal}.mat-step-label-error{font-size:14px}.mat-step-label-selected{font-size:14px;font-weight:500}.mat-tab-group{font-family:Roboto, "Helvetica Neue", sans-serif}.mat-tab-label,.mat-tab-link{font-family:Roboto, "Helvetica Neue", sans-serif;font-size:14px;font-weight:500}.mat-toolbar,.mat-toolbar h1,.mat-toolbar h2,.mat-toolbar h3,.mat-toolbar h4,.mat-toolbar h5,.mat-toolbar h6{font:500 20px/32px Roboto, "Helvetica Neue", sans-serif;letter-spacing:normal;margin:0}.mat-tooltip{font-family:Roboto, "Helvetica Neue", sans-serif;font-size:10px;padding-top:6px;padding-bottom:6px}.mat-tooltip-handset{font-size:14px;padding-top:8px;padding-bottom:8px}.mat-list-item{font-family:Roboto, "Helvetica Neue", sans-serif}.mat-list-option{font-family:Roboto, "Helvetica Neue", sans-serif}.mat-list-base .mat-list-item{font-size:16px}.mat-list-base .mat-list-item .mat-line{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base .mat-list-item .mat-line:nth-child(n+2){font-size:14px}.mat-list-base .mat-list-option{font-size:16px}.mat-list-base .mat-list-option .mat-line{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base .mat-list-option .mat-line:nth-child(n+2){font-size:14px}.mat-list-base .mat-subheader{font-family:Roboto, "Helvetica Neue", sans-serif;font-size:14px;font-weight:500}.mat-list-base[dense] .mat-list-item{font-size:12px}.mat-list-base[dense] .mat-list-item .mat-line{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense] .mat-list-item .mat-line:nth-child(n+2){font-size:12px}.mat-list-base[dense] .mat-list-option{font-size:12px}.mat-list-base[dense] .mat-list-option .mat-line{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense] .mat-list-option .mat-line:nth-child(n+2){font-size:12px}.mat-list-base[dense] .mat-subheader{font-family:Roboto, "Helvetica Neue", sans-serif;font-size:12px;font-weight:500}.mat-option{font-family:Roboto, "Helvetica Neue", sans-serif;font-size:16px}.mat-optgroup-label{font:500 14px/24px Roboto, "Helvetica Neue", sans-serif;letter-spacing:normal}.mat-simple-snackbar{font-family:Roboto, "Helvetica Neue", sans-serif;font-size:14px}.mat-simple-snackbar-action{line-height:1;font-family:inherit;font-size:inherit;font-weight:500}.mat-tree{font-family:Roboto, "Helvetica Neue", sans-serif}.mat-tree-node,.mat-nested-tree-node{font-weight:400;font-size:14px}.mat-ripple{overflow:hidden;position:relative}.mat-ripple:not(:empty){transform:translateZ(0)}.mat-ripple.mat-ripple-unbounded{overflow:visible}.mat-ripple-element{position:absolute;border-radius:50%;pointer-events:none;transition:opacity,transform 0ms cubic-bezier(0, 0, 0.2, 1);transform:scale(0)}.cdk-high-contrast-active .mat-ripple-element{display:none}.cdk-visually-hidden{border:0;clip:rect(0 0 0 0);height:1px;margin:-1px;overflow:hidden;padding:0;position:absolute;width:1px;outline:0;-webkit-appearance:none;-moz-appearance:none}.cdk-overlay-container,.cdk-global-overlay-wrapper{pointer-events:none;top:0;left:0;height:100%;width:100%}.cdk-overlay-container{position:fixed;z-index:1000}.cdk-overlay-container:empty{display:none}.cdk-global-overlay-wrapper{display:flex;position:absolute;z-index:1000}.cdk-overlay-pane{position:absolute;pointer-events:auto;box-sizing:border-box;z-index:1000;display:flex;max-width:100%;max-height:100%}.cdk-overlay-backdrop{position:absolute;top:0;bottom:0;left:0;right:0;z-index:1000;pointer-events:auto;-webkit-tap-highlight-color:transparent;transition:opacity 400ms cubic-bezier(0.25, 0.8, 0.25, 1);opacity:0}.cdk-overlay-backdrop.cdk-overlay-backdrop-showing{opacity:1}@media screen and (-ms-high-contrast: active){.cdk-overlay-backdrop.cdk-overlay-backdrop-showing{opacity:.6}}.cdk-overlay-dark-backdrop{background:rgba(0,0,0,.32)}.cdk-overlay-transparent-backdrop,.cdk-overlay-transparent-backdrop.cdk-overlay-backdrop-showing{opacity:0}.cdk-overlay-connected-position-bounding-box{position:absolute;z-index:1000;display:flex;flex-direction:column;min-width:1px;min-height:1px}.cdk-global-scrollblock{position:fixed;width:100%;overflow-y:scroll}@keyframes cdk-text-field-autofill-start{/*!*/}@keyframes cdk-text-field-autofill-end{/*!*/}.cdk-text-field-autofill-monitored:-webkit-autofill{animation:cdk-text-field-autofill-start 0s 1ms}.cdk-text-field-autofill-monitored:not(:-webkit-autofill){animation:cdk-text-field-autofill-end 0s 1ms}textarea.cdk-textarea-autosize{resize:none}textarea.cdk-textarea-autosize-measuring{padding:2px 0 !important;box-sizing:content-box !important;height:auto !important;overflow:hidden !important}textarea.cdk-textarea-autosize-measuring-firefox{padding:2px 0 !important;box-sizing:content-box !important;height:0 !important}.mat-ripple-element{background-color:rgba(0,0,0,.1)}.mat-option{color:#212121}.mat-option:hover:not(.mat-option-disabled),.mat-option:focus:not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-selected:not(.mat-option-multiple):not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-active{background:rgba(0,0,0,.04);color:#212121}.mat-option.mat-option-disabled{color:rgba(0,0,0,.38)}.mat-primary .mat-option.mat-selected:not(.mat-option-disabled){color:#f57c00}.mat-accent .mat-option.mat-selected:not(.mat-option-disabled){color:#ff9800}.mat-warn .mat-option.mat-selected:not(.mat-option-disabled){color:#f44336}.mat-optgroup-label{color:#616161}.mat-optgroup-disabled .mat-optgroup-label{color:rgba(0,0,0,.38)}.mat-pseudo-checkbox{color:#616161}.mat-pseudo-checkbox::after{color:#fafafa}.mat-pseudo-checkbox-disabled{color:#b0b0b0}.mat-primary .mat-pseudo-checkbox-checked,.mat-primary .mat-pseudo-checkbox-indeterminate{background:#f57c00}.mat-pseudo-checkbox-checked,.mat-pseudo-checkbox-indeterminate,.mat-accent .mat-pseudo-checkbox-checked,.mat-accent .mat-pseudo-checkbox-indeterminate{background:#ff9800}.mat-warn .mat-pseudo-checkbox-checked,.mat-warn .mat-pseudo-checkbox-indeterminate{background:#f44336}.mat-pseudo-checkbox-checked.mat-pseudo-checkbox-disabled,.mat-pseudo-checkbox-indeterminate.mat-pseudo-checkbox-disabled{background:#b0b0b0}.mat-elevation-z0{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z1{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z2{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z3{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z4{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z5{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 5px 8px 0px rgba(0, 0, 0, 0.14),0px 1px 14px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z6{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z7{box-shadow:0px 4px 5px -2px rgba(0, 0, 0, 0.2),0px 7px 10px 1px rgba(0, 0, 0, 0.14),0px 2px 16px 1px rgba(0, 0, 0, 0.12)}.mat-elevation-z8{box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z9{box-shadow:0px 5px 6px -3px rgba(0, 0, 0, 0.2),0px 9px 12px 1px rgba(0, 0, 0, 0.14),0px 3px 16px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z10{box-shadow:0px 6px 6px -3px rgba(0, 0, 0, 0.2),0px 10px 14px 1px rgba(0, 0, 0, 0.14),0px 4px 18px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z11{box-shadow:0px 6px 7px -4px rgba(0, 0, 0, 0.2),0px 11px 15px 1px rgba(0, 0, 0, 0.14),0px 4px 20px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z12{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z13{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 13px 19px 2px rgba(0, 0, 0, 0.14),0px 5px 24px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z14{box-shadow:0px 7px 9px -4px rgba(0, 0, 0, 0.2),0px 14px 21px 2px rgba(0, 0, 0, 0.14),0px 5px 26px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z15{box-shadow:0px 8px 9px -5px rgba(0, 0, 0, 0.2),0px 15px 22px 2px rgba(0, 0, 0, 0.14),0px 6px 28px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z16{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z17{box-shadow:0px 8px 11px -5px rgba(0, 0, 0, 0.2),0px 17px 26px 2px rgba(0, 0, 0, 0.14),0px 6px 32px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z18{box-shadow:0px 9px 11px -5px rgba(0, 0, 0, 0.2),0px 18px 28px 2px rgba(0, 0, 0, 0.14),0px 7px 34px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z19{box-shadow:0px 9px 12px -6px rgba(0, 0, 0, 0.2),0px 19px 29px 2px rgba(0, 0, 0, 0.14),0px 7px 36px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z20{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 20px 31px 3px rgba(0, 0, 0, 0.14),0px 8px 38px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z21{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 21px 33px 3px rgba(0, 0, 0, 0.14),0px 8px 40px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z22{box-shadow:0px 10px 14px -6px rgba(0, 0, 0, 0.2),0px 22px 35px 3px rgba(0, 0, 0, 0.14),0px 8px 42px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z23{box-shadow:0px 11px 14px -7px rgba(0, 0, 0, 0.2),0px 23px 36px 3px rgba(0, 0, 0, 0.14),0px 9px 44px 8px rgba(0, 0, 0, 0.12)}.mat-elevation-z24{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12)}.mat-app-background{background-color:#fafafa;color:#212121}.mat-theme-loaded-marker{display:none}.mat-autocomplete-panel{background:#fff;color:#212121}.mat-autocomplete-panel:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-autocomplete-panel .mat-option.mat-selected:not(.mat-active):not(:hover){background:#fff}.mat-autocomplete-panel .mat-option.mat-selected:not(.mat-active):not(:hover):not(.mat-option-disabled){color:#212121}.mat-badge-content{color:#fff;background:#f57c00}.cdk-high-contrast-active .mat-badge-content{outline:solid 1px;border-radius:0}.mat-badge-accent .mat-badge-content{background:#ff9800;color:#fff}.mat-badge-warn .mat-badge-content{color:#fff;background:#f44336}.mat-badge{position:relative}.mat-badge-hidden .mat-badge-content{display:none}.mat-badge-disabled .mat-badge-content{background:#b9b9b9;color:#757575}.mat-badge-content{position:absolute;text-align:center;display:inline-block;border-radius:50%;transition:transform 200ms ease-in-out;transform:scale(0.6);overflow:hidden;white-space:nowrap;text-overflow:ellipsis;pointer-events:none}.ng-animate-disabled .mat-badge-content,.mat-badge-content._mat-animation-noopable{transition:none}.mat-badge-content.mat-badge-active{transform:none}.mat-badge-small .mat-badge-content{width:16px;height:16px;line-height:16px}.mat-badge-small.mat-badge-above .mat-badge-content{top:-8px}.mat-badge-small.mat-badge-below .mat-badge-content{bottom:-8px}.mat-badge-small.mat-badge-before .mat-badge-content{left:-16px}[dir=rtl] .mat-badge-small.mat-badge-before .mat-badge-content{left:auto;right:-16px}.mat-badge-small.mat-badge-after .mat-badge-content{right:-16px}[dir=rtl] .mat-badge-small.mat-badge-after .mat-badge-content{right:auto;left:-16px}.mat-badge-small.mat-badge-overlap.mat-badge-before .mat-badge-content{left:-8px}[dir=rtl] .mat-badge-small.mat-badge-overlap.mat-badge-before .mat-badge-content{left:auto;right:-8px}.mat-badge-small.mat-badge-overlap.mat-badge-after .mat-badge-content{right:-8px}[dir=rtl] .mat-badge-small.mat-badge-overlap.mat-badge-after .mat-badge-content{right:auto;left:-8px}.mat-badge-medium .mat-badge-content{width:22px;height:22px;line-height:22px}.mat-badge-medium.mat-badge-above .mat-badge-content{top:-11px}.mat-badge-medium.mat-badge-below .mat-badge-content{bottom:-11px}.mat-badge-medium.mat-badge-before .mat-badge-content{left:-22px}[dir=rtl] .mat-badge-medium.mat-badge-before .mat-badge-content{left:auto;right:-22px}.mat-badge-medium.mat-badge-after .mat-badge-content{right:-22px}[dir=rtl] .mat-badge-medium.mat-badge-after .mat-badge-content{right:auto;left:-22px}.mat-badge-medium.mat-badge-overlap.mat-badge-before .mat-badge-content{left:-11px}[dir=rtl] .mat-badge-medium.mat-badge-overlap.mat-badge-before .mat-badge-content{left:auto;right:-11px}.mat-badge-medium.mat-badge-overlap.mat-badge-after .mat-badge-content{right:-11px}[dir=rtl] .mat-badge-medium.mat-badge-overlap.mat-badge-after .mat-badge-content{right:auto;left:-11px}.mat-badge-large .mat-badge-content{width:28px;height:28px;line-height:28px}.mat-badge-large.mat-badge-above .mat-badge-content{top:-14px}.mat-badge-large.mat-badge-below .mat-badge-content{bottom:-14px}.mat-badge-large.mat-badge-before .mat-badge-content{left:-28px}[dir=rtl] .mat-badge-large.mat-badge-before .mat-badge-content{left:auto;right:-28px}.mat-badge-large.mat-badge-after .mat-badge-content{right:-28px}[dir=rtl] .mat-badge-large.mat-badge-after .mat-badge-content{right:auto;left:-28px}.mat-badge-large.mat-badge-overlap.mat-badge-before .mat-badge-content{left:-14px}[dir=rtl] .mat-badge-large.mat-badge-overlap.mat-badge-before .mat-badge-content{left:auto;right:-14px}.mat-badge-large.mat-badge-overlap.mat-badge-after .mat-badge-content{right:-14px}[dir=rtl] .mat-badge-large.mat-badge-overlap.mat-badge-after .mat-badge-content{right:auto;left:-14px}.mat-bottom-sheet-container{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-button,.mat-icon-button,.mat-stroked-button{color:inherit;background:transparent}.mat-button.mat-primary,.mat-icon-button.mat-primary,.mat-stroked-button.mat-primary{color:#f57c00}.mat-button.mat-accent,.mat-icon-button.mat-accent,.mat-stroked-button.mat-accent{color:#ff9800}.mat-button.mat-warn,.mat-icon-button.mat-warn,.mat-stroked-button.mat-warn{color:#f44336}.mat-button.mat-primary[disabled],.mat-button.mat-accent[disabled],.mat-button.mat-warn[disabled],.mat-button[disabled][disabled],.mat-icon-button.mat-primary[disabled],.mat-icon-button.mat-accent[disabled],.mat-icon-button.mat-warn[disabled],.mat-icon-button[disabled][disabled],.mat-stroked-button.mat-primary[disabled],.mat-stroked-button.mat-accent[disabled],.mat-stroked-button.mat-warn[disabled],.mat-stroked-button[disabled][disabled]{color:rgba(0,0,0,.26)}.mat-button.mat-primary .mat-button-focus-overlay,.mat-icon-button.mat-primary .mat-button-focus-overlay,.mat-stroked-button.mat-primary .mat-button-focus-overlay{background-color:#f57c00}.mat-button.mat-accent .mat-button-focus-overlay,.mat-icon-button.mat-accent .mat-button-focus-overlay,.mat-stroked-button.mat-accent .mat-button-focus-overlay{background-color:#ff9800}.mat-button.mat-warn .mat-button-focus-overlay,.mat-icon-button.mat-warn .mat-button-focus-overlay,.mat-stroked-button.mat-warn .mat-button-focus-overlay{background-color:#f44336}.mat-button[disabled] .mat-button-focus-overlay,.mat-icon-button[disabled] .mat-button-focus-overlay,.mat-stroked-button[disabled] .mat-button-focus-overlay{background-color:transparent}.mat-button .mat-ripple-element,.mat-icon-button .mat-ripple-element,.mat-stroked-button .mat-ripple-element{opacity:.1;background-color:currentColor}.mat-button-focus-overlay{background:#000}.mat-stroked-button:not([disabled]){border-color:rgba(0,0,0,.12)}.mat-flat-button,.mat-raised-button,.mat-fab,.mat-mini-fab{color:#212121;background-color:#fff}.mat-flat-button.mat-primary,.mat-raised-button.mat-primary,.mat-fab.mat-primary,.mat-mini-fab.mat-primary{color:#fff}.mat-flat-button.mat-accent,.mat-raised-button.mat-accent,.mat-fab.mat-accent,.mat-mini-fab.mat-accent{color:#fff}.mat-flat-button.mat-warn,.mat-raised-button.mat-warn,.mat-fab.mat-warn,.mat-mini-fab.mat-warn{color:#fff}.mat-flat-button.mat-primary[disabled],.mat-flat-button.mat-accent[disabled],.mat-flat-button.mat-warn[disabled],.mat-flat-button[disabled][disabled],.mat-raised-button.mat-primary[disabled],.mat-raised-button.mat-accent[disabled],.mat-raised-button.mat-warn[disabled],.mat-raised-button[disabled][disabled],.mat-fab.mat-primary[disabled],.mat-fab.mat-accent[disabled],.mat-fab.mat-warn[disabled],.mat-fab[disabled][disabled],.mat-mini-fab.mat-primary[disabled],.mat-mini-fab.mat-accent[disabled],.mat-mini-fab.mat-warn[disabled],.mat-mini-fab[disabled][disabled]{color:rgba(0,0,0,.26)}.mat-flat-button.mat-primary,.mat-raised-button.mat-primary,.mat-fab.mat-primary,.mat-mini-fab.mat-primary{background-color:#f57c00}.mat-flat-button.mat-accent,.mat-raised-button.mat-accent,.mat-fab.mat-accent,.mat-mini-fab.mat-accent{background-color:#ff9800}.mat-flat-button.mat-warn,.mat-raised-button.mat-warn,.mat-fab.mat-warn,.mat-mini-fab.mat-warn{background-color:#f44336}.mat-flat-button.mat-primary[disabled],.mat-flat-button.mat-accent[disabled],.mat-flat-button.mat-warn[disabled],.mat-flat-button[disabled][disabled],.mat-raised-button.mat-primary[disabled],.mat-raised-button.mat-accent[disabled],.mat-raised-button.mat-warn[disabled],.mat-raised-button[disabled][disabled],.mat-fab.mat-primary[disabled],.mat-fab.mat-accent[disabled],.mat-fab.mat-warn[disabled],.mat-fab[disabled][disabled],.mat-mini-fab.mat-primary[disabled],.mat-mini-fab.mat-accent[disabled],.mat-mini-fab.mat-warn[disabled],.mat-mini-fab[disabled][disabled]{background-color:rgba(0,0,0,.12)}.mat-flat-button.mat-primary .mat-ripple-element,.mat-raised-button.mat-primary .mat-ripple-element,.mat-fab.mat-primary .mat-ripple-element,.mat-mini-fab.mat-primary .mat-ripple-element{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-accent .mat-ripple-element,.mat-raised-button.mat-accent .mat-ripple-element,.mat-fab.mat-accent .mat-ripple-element,.mat-mini-fab.mat-accent .mat-ripple-element{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-warn .mat-ripple-element,.mat-raised-button.mat-warn .mat-ripple-element,.mat-fab.mat-warn .mat-ripple-element,.mat-mini-fab.mat-warn .mat-ripple-element{background-color:rgba(255,255,255,.1)}.mat-stroked-button:not([class*=mat-elevation-z]),.mat-flat-button:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-raised-button[disabled]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-fab:not([class*=mat-elevation-z]),.mat-mini-fab:not([class*=mat-elevation-z]){box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-fab:not([disabled]):active:not([class*=mat-elevation-z]),.mat-mini-fab:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-fab[disabled]:not([class*=mat-elevation-z]),.mat-mini-fab[disabled]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone,.mat-button-toggle-group{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard,.mat-button-toggle-group-appearance-standard{box-shadow:none}.mat-button-toggle{color:rgba(0,0,0,.38)}.mat-button-toggle .mat-button-toggle-focus-overlay{background-color:rgba(0,0,0,.12)}.mat-button-toggle-appearance-standard{color:#212121;background:#fff}.mat-button-toggle-appearance-standard .mat-button-toggle-focus-overlay{background-color:#000}.mat-button-toggle-group-appearance-standard .mat-button-toggle+.mat-button-toggle{border-left:solid 1px rgba(0,0,0,.12)}[dir=rtl] .mat-button-toggle-group-appearance-standard .mat-button-toggle+.mat-button-toggle{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-group-appearance-standard.mat-button-toggle-vertical .mat-button-toggle+.mat-button-toggle{border-left:none;border-right:none;border-top:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-checked{background-color:#e0e0e0;color:#616161}.mat-button-toggle-checked.mat-button-toggle-appearance-standard{color:#212121}.mat-button-toggle-disabled{color:rgba(0,0,0,.26);background-color:#eee}.mat-button-toggle-disabled.mat-button-toggle-appearance-standard{background:#fff}.mat-button-toggle-disabled.mat-button-toggle-checked{background-color:#bdbdbd}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard,.mat-button-toggle-group-appearance-standard{border:solid 1px rgba(0,0,0,.12)}.mat-card{background:#fff;color:#212121}.mat-card:not([class*=mat-elevation-z]){box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-card.mat-card-flat:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-card-subtitle{color:#616161}.mat-checkbox-frame{border-color:#616161}.mat-checkbox-checkmark{fill:#fafafa}.mat-checkbox-checkmark-path{stroke:#fafafa !important}.mat-checkbox-mixedmark{background-color:#fafafa}.mat-checkbox-indeterminate.mat-primary .mat-checkbox-background,.mat-checkbox-checked.mat-primary .mat-checkbox-background{background-color:#f57c00}.mat-checkbox-indeterminate.mat-accent .mat-checkbox-background,.mat-checkbox-checked.mat-accent .mat-checkbox-background{background-color:#ff9800}.mat-checkbox-indeterminate.mat-warn .mat-checkbox-background,.mat-checkbox-checked.mat-warn .mat-checkbox-background{background-color:#f44336}.mat-checkbox-disabled.mat-checkbox-checked .mat-checkbox-background,.mat-checkbox-disabled.mat-checkbox-indeterminate .mat-checkbox-background{background-color:#b0b0b0}.mat-checkbox-disabled:not(.mat-checkbox-checked) .mat-checkbox-frame{border-color:#b0b0b0}.mat-checkbox-disabled .mat-checkbox-label{color:#616161}.mat-checkbox .mat-ripple-element{background-color:#000}.mat-checkbox-checked:not(.mat-checkbox-disabled).mat-primary .mat-ripple-element,.mat-checkbox:active:not(.mat-checkbox-disabled).mat-primary .mat-ripple-element{background:#f57c00}.mat-checkbox-checked:not(.mat-checkbox-disabled).mat-accent .mat-ripple-element,.mat-checkbox:active:not(.mat-checkbox-disabled).mat-accent .mat-ripple-element{background:#ff9800}.mat-checkbox-checked:not(.mat-checkbox-disabled).mat-warn .mat-ripple-element,.mat-checkbox:active:not(.mat-checkbox-disabled).mat-warn .mat-ripple-element{background:#f44336}.mat-chip.mat-standard-chip{background-color:#e0e0e0;color:#212121}.mat-chip.mat-standard-chip .mat-chip-remove{color:#212121;opacity:.4}.mat-chip.mat-standard-chip:not(.mat-chip-disabled):active{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-chip.mat-standard-chip:not(.mat-chip-disabled) .mat-chip-remove:hover{opacity:.54}.mat-chip.mat-standard-chip.mat-chip-disabled{opacity:.4}.mat-chip.mat-standard-chip::after{background:#000}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary{background-color:#f57c00;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary .mat-chip-remove{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary .mat-ripple-element{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn{background-color:#f44336;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn .mat-chip-remove{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn .mat-ripple-element{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent{background-color:#ff9800;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent .mat-chip-remove{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent .mat-ripple-element{background-color:rgba(255,255,255,.1)}.mat-table{background:#fff}.mat-table thead,.mat-table tbody,.mat-table tfoot,mat-header-row,mat-row,mat-footer-row,[mat-header-row],[mat-row],[mat-footer-row],.mat-table-sticky{background:inherit}mat-row,mat-header-row,mat-footer-row,th.mat-header-cell,td.mat-cell,td.mat-footer-cell{border-bottom-color:rgba(0,0,0,.12)}.mat-header-cell{color:#616161}.mat-cell,.mat-footer-cell{color:#212121}.mat-calendar-arrow{border-top-color:rgba(0,0,0,.54)}.mat-datepicker-toggle,.mat-datepicker-content .mat-calendar-next-button,.mat-datepicker-content .mat-calendar-previous-button{color:rgba(0,0,0,.54)}.mat-calendar-table-header{color:rgba(0,0,0,.38)}.mat-calendar-table-header-divider::after{background:rgba(0,0,0,.12)}.mat-calendar-body-label{color:#616161}.mat-calendar-body-cell-content{color:#212121;border-color:transparent}.mat-calendar-body-disabled>.mat-calendar-body-cell-content:not(.mat-calendar-body-selected){color:#757575}.mat-calendar-body-cell:not(.mat-calendar-body-disabled):hover>.mat-calendar-body-cell-content:not(.mat-calendar-body-selected),.cdk-keyboard-focused .mat-calendar-body-active>.mat-calendar-body-cell-content:not(.mat-calendar-body-selected),.cdk-program-focused .mat-calendar-body-active>.mat-calendar-body-cell-content:not(.mat-calendar-body-selected){background-color:rgba(0,0,0,.04)}.mat-calendar-body-today:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.38)}.mat-calendar-body-disabled>.mat-calendar-body-today:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.18)}.mat-calendar-body-selected{background-color:#f57c00;color:#fff}.mat-calendar-body-disabled>.mat-calendar-body-selected{background-color:rgba(245,124,0,.4)}.mat-calendar-body-today.mat-calendar-body-selected{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12);background-color:#fff;color:#212121}.mat-datepicker-content.mat-accent .mat-calendar-body-selected{background-color:#ff9800;color:#fff}.mat-datepicker-content.mat-accent .mat-calendar-body-disabled>.mat-calendar-body-selected{background-color:rgba(255,152,0,.4)}.mat-datepicker-content.mat-accent .mat-calendar-body-today.mat-calendar-body-selected{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content.mat-warn .mat-calendar-body-selected{background-color:#f44336;color:#fff}.mat-datepicker-content.mat-warn .mat-calendar-body-disabled>.mat-calendar-body-selected{background-color:rgba(244,67,54,.4)}.mat-datepicker-content.mat-warn .mat-calendar-body-today.mat-calendar-body-selected{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content-touch{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-datepicker-toggle-active{color:#f57c00}.mat-datepicker-toggle-active.mat-accent{color:#ff9800}.mat-datepicker-toggle-active.mat-warn{color:#f44336}.mat-dialog-container{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-divider{border-top-color:rgba(0,0,0,.12)}.mat-divider-vertical{border-right-color:rgba(0,0,0,.12)}.mat-expansion-panel{background:#fff;color:#212121}.mat-expansion-panel:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-action-row{border-top-color:rgba(0,0,0,.12)}.mat-expansion-panel .mat-expansion-panel-header.cdk-keyboard-focused:not([aria-disabled=true]),.mat-expansion-panel .mat-expansion-panel-header.cdk-program-focused:not([aria-disabled=true]),.mat-expansion-panel:not(.mat-expanded) .mat-expansion-panel-header:hover:not([aria-disabled=true]){background:rgba(0,0,0,.04)}@media(hover: none){.mat-expansion-panel:not(.mat-expanded):not([aria-disabled=true]) .mat-expansion-panel-header:hover{background:#fff}}.mat-expansion-panel-header-title{color:#212121}.mat-expansion-panel-header-description,.mat-expansion-indicator::after{color:#616161}.mat-expansion-panel-header[aria-disabled=true]{color:rgba(0,0,0,.26)}.mat-expansion-panel-header[aria-disabled=true] .mat-expansion-panel-header-title,.mat-expansion-panel-header[aria-disabled=true] .mat-expansion-panel-header-description{color:inherit}.mat-form-field-label{color:rgba(97,97,97,.6)}.mat-hint{color:rgba(97,97,97,.6)}.mat-form-field.mat-focused .mat-form-field-label{color:#f57c00}.mat-form-field.mat-focused .mat-form-field-label.mat-accent{color:#ff9800}.mat-form-field.mat-focused .mat-form-field-label.mat-warn{color:#f44336}.mat-focused .mat-form-field-required-marker{color:#ff9800}.mat-form-field-ripple{background-color:rgba(0,0,0,.87)}.mat-form-field.mat-focused .mat-form-field-ripple{background-color:#f57c00}.mat-form-field.mat-focused .mat-form-field-ripple.mat-accent{background-color:#ff9800}.mat-form-field.mat-focused .mat-form-field-ripple.mat-warn{background-color:#f44336}.mat-form-field-type-mat-native-select.mat-focused:not(.mat-form-field-invalid) .mat-form-field-infix::after{color:#f57c00}.mat-form-field-type-mat-native-select.mat-focused:not(.mat-form-field-invalid).mat-accent .mat-form-field-infix::after{color:#ff9800}.mat-form-field-type-mat-native-select.mat-focused:not(.mat-form-field-invalid).mat-warn .mat-form-field-infix::after{color:#f44336}.mat-form-field.mat-form-field-invalid .mat-form-field-label{color:#f44336}.mat-form-field.mat-form-field-invalid .mat-form-field-label.mat-accent,.mat-form-field.mat-form-field-invalid .mat-form-field-label .mat-form-field-required-marker{color:#f44336}.mat-form-field.mat-form-field-invalid .mat-form-field-ripple,.mat-form-field.mat-form-field-invalid .mat-form-field-ripple.mat-accent{background-color:#f44336}.mat-error{color:#f44336}.mat-form-field-appearance-legacy .mat-form-field-label{color:#616161}.mat-form-field-appearance-legacy .mat-hint{color:#616161}.mat-form-field-appearance-legacy .mat-form-field-underline{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-legacy.mat-form-field-disabled .mat-form-field-underline{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-standard .mat-form-field-underline{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-standard.mat-form-field-disabled .mat-form-field-underline{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-fill .mat-form-field-flex{background-color:rgba(0,0,0,.04)}.mat-form-field-appearance-fill.mat-form-field-disabled .mat-form-field-flex{background-color:rgba(0,0,0,.02)}.mat-form-field-appearance-fill .mat-form-field-underline::before{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-fill.mat-form-field-disabled .mat-form-field-label{color:#757575}.mat-form-field-appearance-fill.mat-form-field-disabled .mat-form-field-underline::before{background-color:transparent}.mat-form-field-appearance-outline .mat-form-field-outline{color:rgba(0,0,0,.12)}.mat-form-field-appearance-outline .mat-form-field-outline-thick{color:rgba(0,0,0,.87)}.mat-form-field-appearance-outline.mat-focused .mat-form-field-outline-thick{color:#f57c00}.mat-form-field-appearance-outline.mat-focused.mat-accent .mat-form-field-outline-thick{color:#ff9800}.mat-form-field-appearance-outline.mat-focused.mat-warn .mat-form-field-outline-thick{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-invalid.mat-form-field-invalid .mat-form-field-outline-thick{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-disabled .mat-form-field-label{color:#757575}.mat-form-field-appearance-outline.mat-form-field-disabled .mat-form-field-outline{color:rgba(0,0,0,.06)}.mat-icon.mat-primary{color:#f57c00}.mat-icon.mat-accent{color:#ff9800}.mat-icon.mat-warn{color:#f44336}.mat-form-field-type-mat-native-select .mat-form-field-infix::after{color:#616161}.mat-input-element:disabled,.mat-form-field-type-mat-native-select.mat-form-field-disabled .mat-form-field-infix::after{color:#757575}.mat-input-element{caret-color:#f57c00}.mat-input-element::placeholder{color:rgba(97,97,97,.42)}.mat-input-element::-moz-placeholder{color:rgba(97,97,97,.42)}.mat-input-element::-webkit-input-placeholder{color:rgba(97,97,97,.42)}.mat-input-element:-ms-input-placeholder{color:rgba(97,97,97,.42)}.mat-accent .mat-input-element{caret-color:#ff9800}.mat-warn .mat-input-element,.mat-form-field-invalid .mat-input-element{caret-color:#f44336}.mat-form-field-type-mat-native-select.mat-form-field-invalid .mat-form-field-infix::after{color:#f44336}.mat-list-base .mat-list-item{color:#212121}.mat-list-base .mat-list-option{color:#212121}.mat-list-base .mat-subheader{color:#616161}.mat-list-item-disabled{background-color:#eee}.mat-list-option:hover,.mat-list-option:focus,.mat-nav-list .mat-list-item:hover,.mat-nav-list .mat-list-item:focus,.mat-action-list .mat-list-item:hover,.mat-action-list .mat-list-item:focus{background:rgba(0,0,0,.04)}.mat-list-single-selected-option,.mat-list-single-selected-option:hover,.mat-list-single-selected-option:focus{background:rgba(0,0,0,.12)}.mat-menu-panel{background:#fff}.mat-menu-panel:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-menu-item{background:transparent;color:#212121}.mat-menu-item[disabled],.mat-menu-item[disabled]::after{color:rgba(0,0,0,.38)}.mat-menu-item .mat-icon-no-color,.mat-menu-item-submenu-trigger::after{color:rgba(0,0,0,.54)}.mat-menu-item:hover:not([disabled]),.mat-menu-item.cdk-program-focused:not([disabled]),.mat-menu-item.cdk-keyboard-focused:not([disabled]),.mat-menu-item-highlighted:not([disabled]){background:rgba(0,0,0,.04)}.mat-paginator{background:#fff}.mat-paginator,.mat-paginator-page-size .mat-select-trigger{color:#616161}.mat-paginator-decrement,.mat-paginator-increment{border-top:2px solid rgba(0,0,0,.54);border-right:2px solid rgba(0,0,0,.54)}.mat-paginator-first,.mat-paginator-last{border-top:2px solid rgba(0,0,0,.54)}.mat-icon-button[disabled] .mat-paginator-decrement,.mat-icon-button[disabled] .mat-paginator-increment,.mat-icon-button[disabled] .mat-paginator-first,.mat-icon-button[disabled] .mat-paginator-last{border-color:rgba(0,0,0,.38)}.mat-progress-bar-background{fill:#ff9800}.mat-progress-bar-buffer{background-color:#ff9800}.mat-progress-bar-fill::after{background-color:#f57c00}.mat-progress-bar.mat-accent .mat-progress-bar-background{fill:#ff9800}.mat-progress-bar.mat-accent .mat-progress-bar-buffer{background-color:#ff9800}.mat-progress-bar.mat-accent .mat-progress-bar-fill::after{background-color:#ff9800}.mat-progress-bar.mat-warn .mat-progress-bar-background{fill:#ffcdd2}.mat-progress-bar.mat-warn .mat-progress-bar-buffer{background-color:#ffcdd2}.mat-progress-bar.mat-warn .mat-progress-bar-fill::after{background-color:#f44336}.mat-progress-spinner circle,.mat-spinner circle{stroke:#f57c00}.mat-progress-spinner.mat-accent circle,.mat-spinner.mat-accent circle{stroke:#ff9800}.mat-progress-spinner.mat-warn circle,.mat-spinner.mat-warn circle{stroke:#f44336}.mat-radio-outer-circle{border-color:#616161}.mat-radio-button.mat-primary.mat-radio-checked .mat-radio-outer-circle{border-color:#f57c00}.mat-radio-button.mat-primary .mat-radio-inner-circle,.mat-radio-button.mat-primary .mat-radio-ripple .mat-ripple-element:not(.mat-radio-persistent-ripple),.mat-radio-button.mat-primary.mat-radio-checked .mat-radio-persistent-ripple,.mat-radio-button.mat-primary:active .mat-radio-persistent-ripple{background-color:#f57c00}.mat-radio-button.mat-accent.mat-radio-checked .mat-radio-outer-circle{border-color:#ff9800}.mat-radio-button.mat-accent .mat-radio-inner-circle,.mat-radio-button.mat-accent .mat-radio-ripple .mat-ripple-element:not(.mat-radio-persistent-ripple),.mat-radio-button.mat-accent.mat-radio-checked .mat-radio-persistent-ripple,.mat-radio-button.mat-accent:active .mat-radio-persistent-ripple{background-color:#ff9800}.mat-radio-button.mat-warn.mat-radio-checked .mat-radio-outer-circle{border-color:#f44336}.mat-radio-button.mat-warn .mat-radio-inner-circle,.mat-radio-button.mat-warn .mat-radio-ripple .mat-ripple-element:not(.mat-radio-persistent-ripple),.mat-radio-button.mat-warn.mat-radio-checked .mat-radio-persistent-ripple,.mat-radio-button.mat-warn:active .mat-radio-persistent-ripple{background-color:#f44336}.mat-radio-button.mat-radio-disabled.mat-radio-checked .mat-radio-outer-circle,.mat-radio-button.mat-radio-disabled .mat-radio-outer-circle{border-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled .mat-radio-ripple .mat-ripple-element,.mat-radio-button.mat-radio-disabled .mat-radio-inner-circle{background-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled .mat-radio-label-content{color:rgba(0,0,0,.38)}.mat-radio-button .mat-ripple-element{background-color:#000}.mat-select-value{color:#212121}.mat-select-placeholder{color:rgba(97,97,97,.42)}.mat-select-disabled .mat-select-value{color:#757575}.mat-select-arrow{color:#616161}.mat-select-panel{background:#fff}.mat-select-panel:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-select-panel .mat-option.mat-selected:not(.mat-option-multiple){background:rgba(0,0,0,.12)}.mat-form-field.mat-focused.mat-primary .mat-select-arrow{color:#f57c00}.mat-form-field.mat-focused.mat-accent .mat-select-arrow{color:#ff9800}.mat-form-field.mat-focused.mat-warn .mat-select-arrow{color:#f44336}.mat-form-field .mat-select.mat-select-invalid .mat-select-arrow{color:#f44336}.mat-form-field .mat-select.mat-select-disabled .mat-select-arrow{color:#757575}.mat-drawer-container{background-color:#fafafa;color:#212121}.mat-drawer{background-color:#fff;color:#212121}.mat-drawer.mat-drawer-push{background-color:#fff}.mat-drawer:not(.mat-drawer-side){box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-drawer-side{border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-side.mat-drawer-end{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl] .mat-drawer-side{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl] .mat-drawer-side.mat-drawer-end{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-backdrop.mat-drawer-shown{background-color:rgba(0,0,0,.6)}.mat-slide-toggle.mat-checked .mat-slide-toggle-thumb{background-color:#ff9800}.mat-slide-toggle.mat-checked .mat-slide-toggle-bar{background-color:rgba(255,152,0,.54)}.mat-slide-toggle.mat-checked .mat-ripple-element{background-color:#ff9800}.mat-slide-toggle.mat-primary.mat-checked .mat-slide-toggle-thumb{background-color:#f57c00}.mat-slide-toggle.mat-primary.mat-checked .mat-slide-toggle-bar{background-color:rgba(245,124,0,.54)}.mat-slide-toggle.mat-primary.mat-checked .mat-ripple-element{background-color:#f57c00}.mat-slide-toggle.mat-warn.mat-checked .mat-slide-toggle-thumb{background-color:#f44336}.mat-slide-toggle.mat-warn.mat-checked .mat-slide-toggle-bar{background-color:rgba(244,67,54,.54)}.mat-slide-toggle.mat-warn.mat-checked .mat-ripple-element{background-color:#f44336}.mat-slide-toggle:not(.mat-checked) .mat-ripple-element{background-color:#000}.mat-slide-toggle-thumb{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12);background-color:#fafafa}.mat-slide-toggle-bar{background-color:rgba(0,0,0,.38)}.mat-slider-track-background{background-color:rgba(0,0,0,.26)}.mat-primary .mat-slider-track-fill,.mat-primary .mat-slider-thumb,.mat-primary .mat-slider-thumb-label{background-color:#f57c00}.mat-primary .mat-slider-thumb-label-text{color:#fff}.mat-accent .mat-slider-track-fill,.mat-accent .mat-slider-thumb,.mat-accent .mat-slider-thumb-label{background-color:#ff9800}.mat-accent .mat-slider-thumb-label-text{color:#fff}.mat-warn .mat-slider-track-fill,.mat-warn .mat-slider-thumb,.mat-warn .mat-slider-thumb-label{background-color:#f44336}.mat-warn .mat-slider-thumb-label-text{color:#fff}.mat-slider-focus-ring{background-color:rgba(255,152,0,.2)}.mat-slider:hover .mat-slider-track-background,.cdk-focused .mat-slider-track-background{background-color:rgba(0,0,0,.38)}.mat-slider-disabled .mat-slider-track-background,.mat-slider-disabled .mat-slider-track-fill,.mat-slider-disabled .mat-slider-thumb{background-color:rgba(0,0,0,.26)}.mat-slider-disabled:hover .mat-slider-track-background{background-color:rgba(0,0,0,.26)}.mat-slider-min-value .mat-slider-focus-ring{background-color:rgba(0,0,0,.12)}.mat-slider-min-value.mat-slider-thumb-label-showing .mat-slider-thumb,.mat-slider-min-value.mat-slider-thumb-label-showing .mat-slider-thumb-label{background-color:rgba(0,0,0,.87)}.mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused .mat-slider-thumb,.mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused .mat-slider-thumb-label{background-color:rgba(0,0,0,.26)}.mat-slider-min-value:not(.mat-slider-thumb-label-showing) .mat-slider-thumb{border-color:rgba(0,0,0,.26);background-color:transparent}.mat-slider-min-value:not(.mat-slider-thumb-label-showing):hover .mat-slider-thumb,.mat-slider-min-value:not(.mat-slider-thumb-label-showing).cdk-focused .mat-slider-thumb{border-color:rgba(0,0,0,.38)}.mat-slider-min-value:not(.mat-slider-thumb-label-showing):hover.mat-slider-disabled .mat-slider-thumb,.mat-slider-min-value:not(.mat-slider-thumb-label-showing).cdk-focused.mat-slider-disabled .mat-slider-thumb{border-color:rgba(0,0,0,.26)}.mat-slider-has-ticks .mat-slider-wrapper::after{border-color:rgba(0,0,0,.7)}.mat-slider-horizontal .mat-slider-ticks{background-image:repeating-linear-gradient(to right, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent);background-image:-moz-repeating-linear-gradient(0.0001deg, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-slider-vertical .mat-slider-ticks{background-image:repeating-linear-gradient(to bottom, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-step-header.cdk-keyboard-focused,.mat-step-header.cdk-program-focused,.mat-step-header:hover{background-color:rgba(0,0,0,.04)}@media(hover: none){.mat-step-header:hover{background:none}}.mat-step-header .mat-step-label,.mat-step-header .mat-step-optional{color:#616161}.mat-step-header .mat-step-icon{background-color:#616161;color:#fff}.mat-step-header .mat-step-icon-selected,.mat-step-header .mat-step-icon-state-done,.mat-step-header .mat-step-icon-state-edit{background-color:#f57c00;color:#fff}.mat-step-header .mat-step-icon-state-error{background-color:transparent;color:#f44336}.mat-step-header .mat-step-label.mat-step-label-active{color:#212121}.mat-step-header .mat-step-label.mat-step-label-error{color:#f44336}.mat-stepper-horizontal,.mat-stepper-vertical{background-color:#fff}.mat-stepper-vertical-line::before{border-left-color:rgba(0,0,0,.12)}.mat-horizontal-stepper-header::before,.mat-horizontal-stepper-header::after,.mat-stepper-horizontal-line{border-top-color:rgba(0,0,0,.12)}.mat-sort-header-arrow{color:#616161}.mat-tab-nav-bar,.mat-tab-header{border-bottom:1px solid rgba(0,0,0,.12)}.mat-tab-group-inverted-header .mat-tab-nav-bar,.mat-tab-group-inverted-header .mat-tab-header{border-top:1px solid rgba(0,0,0,.12);border-bottom:none}.mat-tab-label,.mat-tab-link{color:#212121}.mat-tab-label.mat-tab-disabled,.mat-tab-link.mat-tab-disabled{color:#757575}.mat-tab-header-pagination-chevron{border-color:#212121}.mat-tab-header-pagination-disabled .mat-tab-header-pagination-chevron{border-color:#757575}.mat-tab-group[class*=mat-background-] .mat-tab-header,.mat-tab-nav-bar[class*=mat-background-]{border-bottom:none;border-top:none}.mat-tab-group.mat-primary .mat-tab-label.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-group.mat-primary .mat-tab-label.cdk-program-focused:not(.mat-tab-disabled),.mat-tab-group.mat-primary .mat-tab-link.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-group.mat-primary .mat-tab-link.cdk-program-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-primary .mat-tab-label.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-primary .mat-tab-label.cdk-program-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-primary .mat-tab-link.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-primary .mat-tab-link.cdk-program-focused:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-primary .mat-ink-bar,.mat-tab-nav-bar.mat-primary .mat-ink-bar{background-color:#f57c00}.mat-tab-group.mat-primary.mat-background-primary .mat-ink-bar,.mat-tab-nav-bar.mat-primary.mat-background-primary .mat-ink-bar{background-color:#fff}.mat-tab-group.mat-accent .mat-tab-label.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-group.mat-accent .mat-tab-label.cdk-program-focused:not(.mat-tab-disabled),.mat-tab-group.mat-accent .mat-tab-link.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-group.mat-accent .mat-tab-link.cdk-program-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-accent .mat-tab-label.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-accent .mat-tab-label.cdk-program-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-accent .mat-tab-link.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-accent .mat-tab-link.cdk-program-focused:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-accent .mat-ink-bar,.mat-tab-nav-bar.mat-accent .mat-ink-bar{background-color:#ff9800}.mat-tab-group.mat-accent.mat-background-accent .mat-ink-bar,.mat-tab-nav-bar.mat-accent.mat-background-accent .mat-ink-bar{background-color:#fff}.mat-tab-group.mat-warn .mat-tab-label.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-group.mat-warn .mat-tab-label.cdk-program-focused:not(.mat-tab-disabled),.mat-tab-group.mat-warn .mat-tab-link.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-group.mat-warn .mat-tab-link.cdk-program-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-warn .mat-tab-label.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-warn .mat-tab-label.cdk-program-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-warn .mat-tab-link.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-warn .mat-tab-link.cdk-program-focused:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-warn .mat-ink-bar,.mat-tab-nav-bar.mat-warn .mat-ink-bar{background-color:#f44336}.mat-tab-group.mat-warn.mat-background-warn .mat-ink-bar,.mat-tab-nav-bar.mat-warn.mat-background-warn .mat-ink-bar{background-color:#fff}.mat-tab-group.mat-background-primary .mat-tab-label.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-group.mat-background-primary .mat-tab-label.cdk-program-focused:not(.mat-tab-disabled),.mat-tab-group.mat-background-primary .mat-tab-link.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-group.mat-background-primary .mat-tab-link.cdk-program-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-background-primary .mat-tab-label.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-background-primary .mat-tab-label.cdk-program-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-background-primary .mat-tab-link.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-background-primary .mat-tab-link.cdk-program-focused:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-primary .mat-tab-header,.mat-tab-group.mat-background-primary .mat-tab-links,.mat-tab-group.mat-background-primary .mat-tab-header-pagination,.mat-tab-nav-bar.mat-background-primary .mat-tab-header,.mat-tab-nav-bar.mat-background-primary .mat-tab-links,.mat-tab-nav-bar.mat-background-primary .mat-tab-header-pagination{background-color:#f57c00}.mat-tab-group.mat-background-primary .mat-tab-label,.mat-tab-group.mat-background-primary .mat-tab-link,.mat-tab-nav-bar.mat-background-primary .mat-tab-label,.mat-tab-nav-bar.mat-background-primary .mat-tab-link{color:#fff}.mat-tab-group.mat-background-primary .mat-tab-label.mat-tab-disabled,.mat-tab-group.mat-background-primary .mat-tab-link.mat-tab-disabled,.mat-tab-nav-bar.mat-background-primary .mat-tab-label.mat-tab-disabled,.mat-tab-nav-bar.mat-background-primary .mat-tab-link.mat-tab-disabled{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary .mat-tab-header-pagination-chevron,.mat-tab-nav-bar.mat-background-primary .mat-tab-header-pagination-chevron{border-color:#fff}.mat-tab-group.mat-background-primary .mat-tab-header-pagination-disabled .mat-tab-header-pagination-chevron,.mat-tab-nav-bar.mat-background-primary .mat-tab-header-pagination-disabled .mat-tab-header-pagination-chevron{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary .mat-ripple-element,.mat-tab-nav-bar.mat-background-primary .mat-ripple-element{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-accent .mat-tab-label.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-group.mat-background-accent .mat-tab-label.cdk-program-focused:not(.mat-tab-disabled),.mat-tab-group.mat-background-accent .mat-tab-link.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-group.mat-background-accent .mat-tab-link.cdk-program-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-background-accent .mat-tab-label.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-background-accent .mat-tab-label.cdk-program-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-background-accent .mat-tab-link.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-background-accent .mat-tab-link.cdk-program-focused:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-accent .mat-tab-header,.mat-tab-group.mat-background-accent .mat-tab-links,.mat-tab-group.mat-background-accent .mat-tab-header-pagination,.mat-tab-nav-bar.mat-background-accent .mat-tab-header,.mat-tab-nav-bar.mat-background-accent .mat-tab-links,.mat-tab-nav-bar.mat-background-accent .mat-tab-header-pagination{background-color:#ff9800}.mat-tab-group.mat-background-accent .mat-tab-label,.mat-tab-group.mat-background-accent .mat-tab-link,.mat-tab-nav-bar.mat-background-accent .mat-tab-label,.mat-tab-nav-bar.mat-background-accent .mat-tab-link{color:#fff}.mat-tab-group.mat-background-accent .mat-tab-label.mat-tab-disabled,.mat-tab-group.mat-background-accent .mat-tab-link.mat-tab-disabled,.mat-tab-nav-bar.mat-background-accent .mat-tab-label.mat-tab-disabled,.mat-tab-nav-bar.mat-background-accent .mat-tab-link.mat-tab-disabled{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent .mat-tab-header-pagination-chevron,.mat-tab-nav-bar.mat-background-accent .mat-tab-header-pagination-chevron{border-color:#fff}.mat-tab-group.mat-background-accent .mat-tab-header-pagination-disabled .mat-tab-header-pagination-chevron,.mat-tab-nav-bar.mat-background-accent .mat-tab-header-pagination-disabled .mat-tab-header-pagination-chevron{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent .mat-ripple-element,.mat-tab-nav-bar.mat-background-accent .mat-ripple-element{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-warn .mat-tab-label.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-group.mat-background-warn .mat-tab-label.cdk-program-focused:not(.mat-tab-disabled),.mat-tab-group.mat-background-warn .mat-tab-link.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-group.mat-background-warn .mat-tab-link.cdk-program-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-background-warn .mat-tab-label.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-background-warn .mat-tab-label.cdk-program-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-background-warn .mat-tab-link.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-background-warn .mat-tab-link.cdk-program-focused:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-background-warn .mat-tab-header,.mat-tab-group.mat-background-warn .mat-tab-links,.mat-tab-group.mat-background-warn .mat-tab-header-pagination,.mat-tab-nav-bar.mat-background-warn .mat-tab-header,.mat-tab-nav-bar.mat-background-warn .mat-tab-links,.mat-tab-nav-bar.mat-background-warn .mat-tab-header-pagination{background-color:#f44336}.mat-tab-group.mat-background-warn .mat-tab-label,.mat-tab-group.mat-background-warn .mat-tab-link,.mat-tab-nav-bar.mat-background-warn .mat-tab-label,.mat-tab-nav-bar.mat-background-warn .mat-tab-link{color:#fff}.mat-tab-group.mat-background-warn .mat-tab-label.mat-tab-disabled,.mat-tab-group.mat-background-warn .mat-tab-link.mat-tab-disabled,.mat-tab-nav-bar.mat-background-warn .mat-tab-label.mat-tab-disabled,.mat-tab-nav-bar.mat-background-warn .mat-tab-link.mat-tab-disabled{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn .mat-tab-header-pagination-chevron,.mat-tab-nav-bar.mat-background-warn .mat-tab-header-pagination-chevron{border-color:#fff}.mat-tab-group.mat-background-warn .mat-tab-header-pagination-disabled .mat-tab-header-pagination-chevron,.mat-tab-nav-bar.mat-background-warn .mat-tab-header-pagination-disabled .mat-tab-header-pagination-chevron{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn .mat-ripple-element,.mat-tab-nav-bar.mat-background-warn .mat-ripple-element{background-color:rgba(255,255,255,.12)}.mat-toolbar{background:#f5f5f5;color:#212121}.mat-toolbar.mat-primary{background:#f57c00;color:#fff}.mat-toolbar.mat-accent{background:#ff9800;color:#fff}.mat-toolbar.mat-warn{background:#f44336;color:#fff}.mat-toolbar .mat-form-field-underline,.mat-toolbar .mat-form-field-ripple,.mat-toolbar .mat-focused .mat-form-field-ripple{background-color:currentColor}.mat-toolbar .mat-form-field-label,.mat-toolbar .mat-focused .mat-form-field-label,.mat-toolbar .mat-select-value,.mat-toolbar .mat-select-arrow,.mat-toolbar .mat-form-field.mat-focused .mat-select-arrow{color:inherit}.mat-toolbar .mat-input-element{caret-color:currentColor}.mat-tooltip{background:rgba(97,97,97,.9)}.mat-tree{background:#fff}.mat-tree-node,.mat-nested-tree-node{color:#212121}.mat-snack-bar-container{color:rgba(255,255,255,.7);background:#323232;box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-simple-snackbar-action{color:#ff9800}
</style>

<style>
  html,
  body {
    margin: 0;
    padding: 0;
    height: 100%;
    font-family: Roboto, sans-serif;
    color: var(--primary-text-color);

    /* Legacy mechanism to avoid issues with subpixel anti-aliasing on macOS.
     *
     * In the past [1], macOS subpixel AA caused excessive bolding for light-on-dark text; this rule
     * avoids that by requesting non-subpixel AA always, rather than the default behavior, which is
     * to use subpixel AA when available. The original issue was "fixed" by removing subpixel AA in
     * macOS 14 (Mojave), but for legacy reasons they preserved the bolding effect as an option.
     * Chrome then in turn updated its font rendering to apply that bolding effect [2], which means
     * that even though the `-webkit-font-smoothing` docs [3] suggest that setting `antialiased`
     * would have no effect for recent versions of macOS, it still is needed to avoid the bolding.
     *
     * [1]: http://www.lighterra.com/articles/macosxtextaabug/
     * [2]: https://bugs.chromium.org/p/chromium/issues/detail?id=858861
     * [3]: https://developer.mozilla.org/en-US/docs/Web/CSS/font-smooth
     *
     */

    -webkit-font-smoothing: antialiased;
  }
</style>

</head><body><tb-webapp></tb-webapp><script src="index.js"></script></body></html>", + "ok": true, + "headers": [ + [ + "content-type", + "text/html; charset=utf-8" + ] + ], + "status": 200, + "status_text": "" + }, + "https://localhost:6006/index.js": { + "data": "var CLOSURE_NO_DEPS = true;
// Copyright 2006 The Closure Library Authors. All Rights Reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
//      http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS-IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

/**
 * @fileoverview Bootstrap for the Google JS Library (Closure).
 *
 * In uncompiled mode base.js will attempt to load Closure's deps file, unless
 * the global <code>CLOSURE_NO_DEPS</code> is set to true.  This allows projects
 * to include their own deps file(s) from different locations.
 *
 * Avoid including base.js more than once. This is strictly discouraged and not
 * supported. goog.require(...) won't work properly in that case.
 *
 * @provideGoog
 */


/**
 * @define {boolean} Overridden to true by the compiler.
 */
var COMPILED = false;


/**
 * Base namespace for the Closure library.  Checks to see goog is already
 * defined in the current scope before assigning to prevent clobbering if
 * base.js is loaded more than once.
 *
 * @const
 */
var goog = goog || {};

/**
 * Reference to the global object.
 * https://www.ecma-international.org/ecma-262/9.0/index.html#sec-global-object
 *
 * More info on this implementation here:
 * https://docs.google.com/document/d/1NAeW4Wk7I7FV0Y2tcUFvQdGMc89k2vdgSXInw8_nvCI/edit
 *
 * @const
 * @suppress {undefinedVars} self won't be referenced unless `this` is falsy.
 * @type {!Global}
 */
goog.global =
    // Check `this` first for backwards compatibility.
    // Valid unless running as an ES module or in a function wrapper called
    //   without setting `this` properly.
    // Note that base.js can't usefully be imported as an ES module, but it may
    // be compiled into bundles that are loadable as ES modules.
    this ||
    // https://developer.mozilla.org/en-US/docs/Web/API/Window/self
    // For in-page browser environments and workers.
    self;


/**
 * A hook for overriding the define values in uncompiled mode.
 *
 * In uncompiled mode, `CLOSURE_UNCOMPILED_DEFINES` may be defined before
 * loading base.js.  If a key is defined in `CLOSURE_UNCOMPILED_DEFINES`,
 * `goog.define` will use the value instead of the default value.  This
 * allows flags to be overwritten without compilation (this is normally
 * accomplished with the compiler's "define" flag).
 *
 * Example:
 * <pre>
 *   var CLOSURE_UNCOMPILED_DEFINES = {'goog.DEBUG': false};
 * </pre>
 *
 * @type {Object<string, (string|number|boolean)>|undefined}
 */
goog.global.CLOSURE_UNCOMPILED_DEFINES;


/**
 * A hook for overriding the define values in uncompiled or compiled mode,
 * like CLOSURE_UNCOMPILED_DEFINES but effective in compiled code.  In
 * uncompiled code CLOSURE_UNCOMPILED_DEFINES takes precedence.
 *
 * Also unlike CLOSURE_UNCOMPILED_DEFINES the values must be number, boolean or
 * string literals or the compiler will emit an error.
 *
 * While any @define value may be set, only those set with goog.define will be
 * effective for uncompiled code.
 *
 * Example:
 * <pre>
 *   var CLOSURE_DEFINES = {'goog.DEBUG': false} ;
 * </pre>
 *
 * @type {Object<string, (string|number|boolean)>|undefined}
 */
goog.global.CLOSURE_DEFINES;


/**
 * Returns true if the specified value is not undefined.
 *
 * @param {?} val Variable to test.
 * @return {boolean} Whether variable is defined.
 * @deprecated Use `val !== undefined` instead.
 */
goog.isDef = function(val) {
  // void 0 always evaluates to undefined and hence we do not need to depend on
  // the definition of the global variable named 'undefined'.
  return val !== void 0;
};

/**
 * Returns true if the specified value is a string.
 * @param {?} val Variable to test.
 * @return {boolean} Whether variable is a string.
 * @deprecated Use `typeof val === 'string'` instead.
 */
goog.isString = function(val) {
  return typeof val == 'string';
};


/**
 * Returns true if the specified value is a boolean.
 * @param {?} val Variable to test.
 * @return {boolean} Whether variable is boolean.
 * @deprecated Use `typeof val === 'boolean'` instead.
 */
goog.isBoolean = function(val) {
  return typeof val == 'boolean';
};


/**
 * Returns true if the specified value is a number.
 * @param {?} val Variable to test.
 * @return {boolean} Whether variable is a number.
 * @deprecated Use `typeof val === 'number'` instead.
 */
goog.isNumber = function(val) {
  return typeof val == 'number';
};


/**
 * Builds an object structure for the provided namespace path, ensuring that
 * names that already exist are not overwritten. For example:
 * "a.b.c" -> a = {};a.b={};a.b.c={};
 * Used by goog.provide and goog.exportSymbol.
 * @param {string} name name of the object that this file defines.
 * @param {*=} opt_object the object to expose at the end of the path.
 * @param {Object=} opt_objectToExportTo The object to add the path to; default
 *     is `goog.global`.
 * @private
 */
goog.exportPath_ = function(name, opt_object, opt_objectToExportTo) {
  var parts = name.split('.');
  var cur = opt_objectToExportTo || goog.global;

  // Internet Explorer exhibits strange behavior when throwing errors from
  // methods externed in this manner.  See the testExportSymbolExceptions in
  // base_test.html for an example.
  if (!(parts[0] in cur) && typeof cur.execScript != 'undefined') {
    cur.execScript('var ' + parts[0]);
  }

  for (var part; parts.length && (part = parts.shift());) {
    if (!parts.length && opt_object !== undefined) {
      // last part and we have an object; use it
      cur[part] = opt_object;
    } else if (cur[part] && cur[part] !== Object.prototype[part]) {
      cur = cur[part];
    } else {
      cur = cur[part] = {};
    }
  }
};


/**
 * Defines a named value. In uncompiled mode, the value is retrieved from
 * CLOSURE_DEFINES or CLOSURE_UNCOMPILED_DEFINES if the object is defined and
 * has the property specified, and otherwise used the defined defaultValue.
 * When compiled the default can be overridden using the compiler options or the
 * value set in the CLOSURE_DEFINES object. Returns the defined value so that it
 * can be used safely in modules. Note that the value type MUST be either
 * boolean, number, or string.
 *
 * @param {string} name The distinguished name to provide.
 * @param {T} defaultValue
 * @return {T} The defined value.
 * @template T
 */
goog.define = function(name, defaultValue) {
  var value = defaultValue;
  if (!COMPILED) {
    var uncompiledDefines = goog.global.CLOSURE_UNCOMPILED_DEFINES;
    var defines = goog.global.CLOSURE_DEFINES;
    if (uncompiledDefines &&
        // Anti DOM-clobbering runtime check (b/37736576).
        /** @type {?} */ (uncompiledDefines).nodeType === undefined &&
        Object.prototype.hasOwnProperty.call(uncompiledDefines, name)) {
      value = uncompiledDefines[name];
    } else if (
        defines &&
        // Anti DOM-clobbering runtime check (b/37736576).
        /** @type {?} */ (defines).nodeType === undefined &&
        Object.prototype.hasOwnProperty.call(defines, name)) {
      value = defines[name];
    }
  }
  return value;
};


/**
 * @define {number} Integer year indicating the set of browser features that are
 * guaranteed to be present.  This is defined to include exactly features that
 * work correctly on all "modern" browsers that are stable on January 1 of the
 * specified year.  For example,
 * ```js
 * if (goog.FEATURESET_YEAR >= 2019) {
 *   // use APIs known to be available on all major stable browsers Jan 1, 2019
 * } else {
 *   // polyfill for older browsers
 * }
 * ```
 * This is intended to be the primary define for removing
 * unnecessary browser compatibility code (such as ponyfills and workarounds),
 * and should inform the default value for most other defines:
 * ```js
 * const ASSUME_NATIVE_PROMISE =
 *     goog.define('ASSUME_NATIVE_PROMISE', goog.FEATURESET_YEAR >= 2016);
 * ```
 *
 * The default assumption is that IE9 is the lowest supported browser, which was
 * first available Jan 1, 2012.
 *
 * TODO(user): Reference more thorough documentation when it's available.
 */
goog.FEATURESET_YEAR = goog.define('goog.FEATURESET_YEAR', 2012);


/**
 * @define {boolean} DEBUG is provided as a convenience so that debugging code
 * that should not be included in a production. It can be easily stripped
 * by specifying --define goog.DEBUG=false to the Closure Compiler aka
 * JSCompiler. For example, most toString() methods should be declared inside an
 * "if (goog.DEBUG)" conditional because they are generally used for debugging
 * purposes and it is difficult for the JSCompiler to statically determine
 * whether they are used.
 */
goog.DEBUG = goog.define('goog.DEBUG', true);


/**
 * @define {string} LOCALE defines the locale being used for compilation. It is
 * used to select locale specific data to be compiled in js binary. BUILD rule
 * can specify this value by "--define goog.LOCALE=<locale_name>" as a compiler
 * option.
 *
 * Take into account that the locale code format is important. You should use
 * the canonical Unicode format with hyphen as a delimiter. Language must be
 * lowercase, Language Script - Capitalized, Region - UPPERCASE.
 * There are few examples: pt-BR, en, en-US, sr-Latin-BO, zh-Hans-CN.
 *
 * See more info about locale codes here:
 * http://www.unicode.org/reports/tr35/#Unicode_Language_and_Locale_Identifiers
 *
 * For language codes you should use values defined by ISO 693-1. See it here
 * http://www.w3.org/WAI/ER/IG/ert/iso639.htm. There is only one exception from
 * this rule: the Hebrew language. For legacy reasons the old code (iw) should
 * be used instead of the new code (he).
 *
 */
goog.LOCALE = goog.define('goog.LOCALE', 'en');  // default to en


/**
 * @define {boolean} Whether this code is running on trusted sites.
 *
 * On untrusted sites, several native functions can be defined or overridden by
 * external libraries like Prototype, Datejs, and JQuery and setting this flag
 * to false forces closure to use its own implementations when possible.
 *
 * If your JavaScript can be loaded by a third party site and you are wary about
 * relying on non-standard implementations, specify
 * "--define goog.TRUSTED_SITE=false" to the compiler.
 */
goog.TRUSTED_SITE = goog.define('goog.TRUSTED_SITE', true);


/**
 * @define {boolean} Whether a project is expected to be running in strict mode.
 *
 * This define can be used to trigger alternate implementations compatible with
 * running in EcmaScript Strict mode or warn about unavailable functionality.
 * @see https://goo.gl/PudQ4y
 *
 */
goog.STRICT_MODE_COMPATIBLE = goog.define('goog.STRICT_MODE_COMPATIBLE', false);


/**
 * @define {boolean} Whether code that calls {@link goog.setTestOnly} should
 *     be disallowed in the compilation unit.
 */
goog.DISALLOW_TEST_ONLY_CODE =
    goog.define('goog.DISALLOW_TEST_ONLY_CODE', COMPILED && !goog.DEBUG);


/**
 * @define {boolean} Whether to use a Chrome app CSP-compliant method for
 *     loading scripts via goog.require. @see appendScriptSrcNode_.
 */
goog.ENABLE_CHROME_APP_SAFE_SCRIPT_LOADING =
    goog.define('goog.ENABLE_CHROME_APP_SAFE_SCRIPT_LOADING', false);


/**
 * Defines a namespace in Closure.
 *
 * A namespace may only be defined once in a codebase. It may be defined using
 * goog.provide() or goog.module().
 *
 * The presence of one or more goog.provide() calls in a file indicates
 * that the file defines the given objects/namespaces.
 * Provided symbols must not be null or undefined.
 *
 * In addition, goog.provide() creates the object stubs for a namespace
 * (for example, goog.provide("goog.foo.bar") will create the object
 * goog.foo.bar if it does not already exist).
 *
 * Build tools also scan for provide/require/module statements
 * to discern dependencies, build dependency files (see deps.js), etc.
 *
 * @see goog.require
 * @see goog.module
 * @param {string} name Namespace provided by this file in the form
 *     "goog.package.part".
 */
goog.provide = function(name) {
  if (goog.isInModuleLoader_()) {
    throw new Error('goog.provide cannot be used within a module.');
  }
  if (!COMPILED) {
    // Ensure that the same namespace isn't provided twice.
    // A goog.module/goog.provide maps a goog.require to a specific file
    if (goog.isProvided_(name)) {
      throw new Error('Namespace "' + name + '" already declared.');
    }
  }

  goog.constructNamespace_(name);
};


/**
 * @param {string} name Namespace provided by this file in the form
 *     "goog.package.part".
 * @param {Object=} opt_obj The object to embed in the namespace.
 * @private
 */
goog.constructNamespace_ = function(name, opt_obj) {
  if (!COMPILED) {
    delete goog.implicitNamespaces_[name];

    var namespace = name;
    while ((namespace = namespace.substring(0, namespace.lastIndexOf('.')))) {
      if (goog.getObjectByName(namespace)) {
        break;
      }
      goog.implicitNamespaces_[namespace] = true;
    }
  }

  goog.exportPath_(name, opt_obj);
};


/**
 * Returns CSP nonce, if set for any script tag.
 * @param {?Window=} opt_window The window context used to retrieve the nonce.
 *     Defaults to global context.
 * @return {string} CSP nonce or empty string if no nonce is present.
 */
goog.getScriptNonce = function(opt_window) {
  if (opt_window && opt_window != goog.global) {
    return goog.getScriptNonce_(opt_window.document);
  }
  if (goog.cspNonce_ === null) {
    goog.cspNonce_ = goog.getScriptNonce_(goog.global.document);
  }
  return goog.cspNonce_;
};


/**
 * According to the CSP3 spec a nonce must be a valid base64 string.
 * @see https://www.w3.org/TR/CSP3/#grammardef-base64-value
 * @private @const
 */
goog.NONCE_PATTERN_ = /^[\w+/_-]+[=]{0,2}$/;


/**
 * @private {?string}
 */
goog.cspNonce_ = null;


/**
 * Returns CSP nonce, if set for any script tag.
 * @param {!Document} doc
 * @return {string} CSP nonce or empty string if no nonce is present.
 * @private
 */
goog.getScriptNonce_ = function(doc) {
  var script = doc.querySelector && doc.querySelector('script[nonce]');
  if (script) {
    // Try to get the nonce from the IDL property first, because browsers that
    // implement additional nonce protection features (currently only Chrome) to
    // prevent nonce stealing via CSS do not expose the nonce via attributes.
    // See https://github.com/whatwg/html/issues/2369
    var nonce = script['nonce'] || script.getAttribute('nonce');
    if (nonce && goog.NONCE_PATTERN_.test(nonce)) {
      return nonce;
    }
  }
  return '';
};


/**
 * Module identifier validation regexp.
 * Note: This is a conservative check, it is very possible to be more lenient,
 *   the primary exclusion here is "/" and "\" and a leading ".", these
 *   restrictions are intended to leave the door open for using goog.require
 *   with relative file paths rather than module identifiers.
 * @private
 */
goog.VALID_MODULE_RE_ = /^[a-zA-Z_$][a-zA-Z0-9._$]*$/;


/**
 * Defines a module in Closure.
 *
 * Marks that this file must be loaded as a module and claims the namespace.
 *
 * A namespace may only be defined once in a codebase. It may be defined using
 * goog.provide() or goog.module().
 *
 * goog.module() has three requirements:
 * - goog.module may not be used in the same file as goog.provide.
 * - goog.module must be the first statement in the file.
 * - only one goog.module is allowed per file.
 *
 * When a goog.module annotated file is loaded, it is enclosed in
 * a strict function closure. This means that:
 * - any variables declared in a goog.module file are private to the file
 * (not global), though the compiler is expected to inline the module.
 * - The code must obey all the rules of "strict" JavaScript.
 * - the file will be marked as "use strict"
 *
 * NOTE: unlike goog.provide, goog.module does not declare any symbols by
 * itself. If declared symbols are desired, use
 * goog.module.declareLegacyNamespace().
 *
 *
 * See the public goog.module proposal: http://goo.gl/Va1hin
 *
 * @param {string} name Namespace provided by this file in the form
 *     "goog.package.part", is expected but not required.
 * @return {void}
 */
goog.module = function(name) {
  if (typeof name !== 'string' || !name ||
      name.search(goog.VALID_MODULE_RE_) == -1) {
    throw new Error('Invalid module identifier');
  }
  if (!goog.isInGoogModuleLoader_()) {
    throw new Error(
        'Module ' + name + ' has been loaded incorrectly. Note, ' +
        'modules cannot be loaded as normal scripts. They require some kind of ' +
        'pre-processing step. You\'re likely trying to load a module via a ' +
        'script tag or as a part of a concatenated bundle without rewriting the ' +
        'module. For more info see: ' +
        'https://github.com/google/closure-library/wiki/goog.module:-an-ES6-module-like-alternative-to-goog.provide.');
  }
  if (goog.moduleLoaderState_.moduleName) {
    throw new Error('goog.module may only be called once per module.');
  }

  // Store the module name for the loader.
  goog.moduleLoaderState_.moduleName = name;
  if (!COMPILED) {
    // Ensure that the same namespace isn't provided twice.
    // A goog.module/goog.provide maps a goog.require to a specific file
    if (goog.isProvided_(name)) {
      throw new Error('Namespace "' + name + '" already declared.');
    }
    delete goog.implicitNamespaces_[name];
  }
};


/**
 * @param {string} name The module identifier.
 * @return {?} The module exports for an already loaded module or null.
 *
 * Note: This is not an alternative to goog.require, it does not
 * indicate a hard dependency, instead it is used to indicate
 * an optional dependency or to access the exports of a module
 * that has already been loaded.
 * @suppress {missingProvide}
 */
goog.module.get = function(name) {
  return goog.module.getInternal_(name);
};


/**
 * @param {string} name The module identifier.
 * @return {?} The module exports for an already loaded module or null.
 * @private
 */
goog.module.getInternal_ = function(name) {
  if (!COMPILED) {
    if (name in goog.loadedModules_) {
      return goog.loadedModules_[name].exports;
    } else if (!goog.implicitNamespaces_[name]) {
      var ns = goog.getObjectByName(name);
      return ns != null ? ns : null;
    }
  }
  return null;
};


/**
 * Types of modules the debug loader can load.
 * @enum {string}
 */
goog.ModuleType = {
  ES6: 'es6',
  GOOG: 'goog'
};


/**
 * @private {?{
 *   moduleName: (string|undefined),
 *   declareLegacyNamespace:boolean,
 *   type: ?goog.ModuleType
 * }}
 */
goog.moduleLoaderState_ = null;


/**
 * @private
 * @return {boolean} Whether a goog.module or an es6 module is currently being
 *     initialized.
 */
goog.isInModuleLoader_ = function() {
  return goog.isInGoogModuleLoader_() || goog.isInEs6ModuleLoader_();
};


/**
 * @private
 * @return {boolean} Whether a goog.module is currently being initialized.
 */
goog.isInGoogModuleLoader_ = function() {
  return !!goog.moduleLoaderState_ &&
      goog.moduleLoaderState_.type == goog.ModuleType.GOOG;
};


/**
 * @private
 * @return {boolean} Whether an es6 module is currently being initialized.
 */
goog.isInEs6ModuleLoader_ = function() {
  var inLoader = !!goog.moduleLoaderState_ &&
      goog.moduleLoaderState_.type == goog.ModuleType.ES6;

  if (inLoader) {
    return true;
  }

  var jscomp = goog.global['$jscomp'];

  if (jscomp) {
    // jscomp may not have getCurrentModulePath if this is a compiled bundle
    // that has some of the runtime, but not all of it. This can happen if
    // optimizations are turned on so the unused runtime is removed but renaming
    // and Closure pass are off (so $jscomp is still named $jscomp and the
    // goog.provide/require calls still exist).
    if (typeof jscomp.getCurrentModulePath != 'function') {
      return false;
    }

    // Bundled ES6 module.
    return !!jscomp.getCurrentModulePath();
  }

  return false;
};


/**
 * Provide the module's exports as a globally accessible object under the
 * module's declared name.  This is intended to ease migration to goog.module
 * for files that have existing usages.
 * @suppress {missingProvide}
 */
goog.module.declareLegacyNamespace = function() {
  if (!COMPILED && !goog.isInGoogModuleLoader_()) {
    throw new Error(
        'goog.module.declareLegacyNamespace must be called from ' +
        'within a goog.module');
  }
  if (!COMPILED && !goog.moduleLoaderState_.moduleName) {
    throw new Error(
        'goog.module must be called prior to ' +
        'goog.module.declareLegacyNamespace.');
  }
  goog.moduleLoaderState_.declareLegacyNamespace = true;
};


/**
 * Associates an ES6 module with a Closure module ID so that is available via
 * goog.require. The associated ID  acts like a goog.module ID - it does not
 * create any global names, it is merely available via goog.require /
 * goog.module.get / goog.forwardDeclare / goog.requireType. goog.require and
 * goog.module.get will return the entire module as if it was import *'d. This
 * allows Closure files to reference ES6 modules for the sake of migration.
 *
 * @param {string} namespace
 * @suppress {missingProvide}
 */
goog.declareModuleId = function(namespace) {
  if (!COMPILED) {
    if (!goog.isInEs6ModuleLoader_()) {
      throw new Error(
          'goog.declareModuleId may only be called from ' +
          'within an ES6 module');
    }
    if (goog.moduleLoaderState_ && goog.moduleLoaderState_.moduleName) {
      throw new Error(
          'goog.declareModuleId may only be called once per module.');
    }
    if (namespace in goog.loadedModules_) {
      throw new Error(
          'Module with namespace "' + namespace + '" already exists.');
    }
  }
  if (goog.moduleLoaderState_) {
    // Not bundled - debug loading.
    goog.moduleLoaderState_.moduleName = namespace;
  } else {
    // Bundled - not debug loading, no module loader state.
    var jscomp = goog.global['$jscomp'];
    if (!jscomp || typeof jscomp.getCurrentModulePath != 'function') {
      throw new Error(
          'Module with namespace "' + namespace +
          '" has been loaded incorrectly.');
    }
    var exports = jscomp.require(jscomp.getCurrentModulePath());
    goog.loadedModules_[namespace] = {
      exports: exports,
      type: goog.ModuleType.ES6,
      moduleId: namespace
    };
  }
};


/**
 * Marks that the current file should only be used for testing, and never for
 * live code in production.
 *
 * In the case of unit tests, the message may optionally be an exact namespace
 * for the test (e.g. 'goog.stringTest'). The linter will then ignore the extra
 * provide (if not explicitly defined in the code).
 *
 * @param {string=} opt_message Optional message to add to the error that's
 *     raised when used in production code.
 */
goog.setTestOnly = function(opt_message) {
  if (goog.DISALLOW_TEST_ONLY_CODE) {
    opt_message = opt_message || '';
    throw new Error(
        'Importing test-only code into non-debug environment' +
        (opt_message ? ': ' + opt_message : '.'));
  }
};


/**
 * Forward declares a symbol. This is an indication to the compiler that the
 * symbol may be used in the source yet is not required and may not be provided
 * in compilation.
 *
 * The most common usage of forward declaration is code that takes a type as a
 * function parameter but does not need to require it. By forward declaring
 * instead of requiring, no hard dependency is made, and (if not required
 * elsewhere) the namespace may never be required and thus, not be pulled
 * into the JavaScript binary. If it is required elsewhere, it will be type
 * checked as normal.
 *
 * Before using goog.forwardDeclare, please read the documentation at
 * https://github.com/google/closure-compiler/wiki/Bad-Type-Annotation to
 * understand the options and tradeoffs when working with forward declarations.
 *
 * @param {string} name The namespace to forward declare in the form of
 *     "goog.package.part".
 */
goog.forwardDeclare = function(name) {};


/**
 * Forward declare type information. Used to assign types to goog.global
 * referenced object that would otherwise result in unknown type references
 * and thus block property disambiguation.
 */
goog.forwardDeclare('Document');
goog.forwardDeclare('HTMLScriptElement');
goog.forwardDeclare('XMLHttpRequest');


if (!COMPILED) {
  /**
   * Check if the given name has been goog.provided. This will return false for
   * names that are available only as implicit namespaces.
   * @param {string} name name of the object to look for.
   * @return {boolean} Whether the name has been provided.
   * @private
   */
  goog.isProvided_ = function(name) {
    return (name in goog.loadedModules_) ||
        (!goog.implicitNamespaces_[name] && goog.getObjectByName(name) != null);
  };

  /**
   * Namespaces implicitly defined by goog.provide. For example,
   * goog.provide('goog.events.Event') implicitly declares that 'goog' and
   * 'goog.events' must be namespaces.
   *
   * @type {!Object<string, (boolean|undefined)>}
   * @private
   */
  goog.implicitNamespaces_ = {'goog.module': true};

  // NOTE: We add goog.module as an implicit namespace as goog.module is defined
  // here and because the existing module package has not been moved yet out of
  // the goog.module namespace. This satisifies both the debug loader and
  // ahead-of-time dependency management.
}


/**
 * Returns an object based on its fully qualified external name.  The object
 * is not found if null or undefined.  If you are using a compilation pass that
 * renames property names beware that using this function will not find renamed
 * properties.
 *
 * @param {string} name The fully qualified name.
 * @param {Object=} opt_obj The object within which to look; default is
 *     |goog.global|.
 * @return {?} The value (object or primitive) or, if not found, null.
 */
goog.getObjectByName = function(name, opt_obj) {
  var parts = name.split('.');
  var cur = opt_obj || goog.global;
  for (var i = 0; i < parts.length; i++) {
    cur = cur[parts[i]];
    if (cur == null) {
      return null;
    }
  }
  return cur;
};


/**
 * Globalizes a whole namespace, such as goog or goog.lang.
 *
 * @param {!Object} obj The namespace to globalize.
 * @param {Object=} opt_global The object to add the properties to.
 * @deprecated Properties may be explicitly exported to the global scope, but
 *     this should no longer be done in bulk.
 */
goog.globalize = function(obj, opt_global) {
  var global = opt_global || goog.global;
  for (var x in obj) {
    global[x] = obj[x];
  }
};


/**
 * Adds a dependency from a file to the files it requires.
 * @param {string} relPath The path to the js file.
 * @param {!Array<string>} provides An array of strings with
 *     the names of the objects this file provides.
 * @param {!Array<string>} requires An array of strings with
 *     the names of the objects this file requires.
 * @param {boolean|!Object<string>=} opt_loadFlags Parameters indicating
 *     how the file must be loaded.  The boolean 'true' is equivalent
 *     to {'module': 'goog'} for backwards-compatibility.  Valid properties
 *     and values include {'module': 'goog'} and {'lang': 'es6'}.
 */
goog.addDependency = function(relPath, provides, requires, opt_loadFlags) {
  if (!COMPILED && goog.DEPENDENCIES_ENABLED) {
    goog.debugLoader_.addDependency(relPath, provides, requires, opt_loadFlags);
  }
};




// NOTE(nnaze): The debug DOM loader was included in base.js as an original way
// to do "debug-mode" development.  The dependency system can sometimes be
// confusing, as can the debug DOM loader's asynchronous nature.
//
// With the DOM loader, a call to goog.require() is not blocking -- the script
// will not load until some point after the current script.  If a namespace is
// needed at runtime, it needs to be defined in a previous script, or loaded via
// require() with its registered dependencies.
//
// User-defined namespaces may need their own deps file. For a reference on
// creating a deps file, see:
// Externally: https://developers.google.com/closure/library/docs/depswriter
//
// Because of legacy clients, the DOM loader can't be easily removed from
// base.js.  Work was done to make it disableable or replaceable for
// different environments (DOM-less JavaScript interpreters like Rhino or V8,
// for example). See bootstrap/ for more information.


/**
 * @define {boolean} Whether to enable the debug loader.
 *
 * If enabled, a call to goog.require() will attempt to load the namespace by
 * appending a script tag to the DOM (if the namespace has been registered).
 *
 * If disabled, goog.require() will simply assert that the namespace has been
 * provided (and depend on the fact that some outside tool correctly ordered
 * the script).
 */
goog.ENABLE_DEBUG_LOADER = goog.define('goog.ENABLE_DEBUG_LOADER', true);


/**
 * @param {string} msg
 * @private
 */
goog.logToConsole_ = function(msg) {
  if (goog.global.console) {
    goog.global.console['error'](msg);
  }
};


/**
 * Implements a system for the dynamic resolution of dependencies that works in
 * parallel with the BUILD system.
 *
 * Note that all calls to goog.require will be stripped by the compiler.
 *
 * @see goog.provide
 * @param {string} namespace Namespace (as was given in goog.provide,
 *     goog.module, or goog.declareModuleId) in the form
 *     "goog.package.part".
 * @return {?} If called within a goog.module or ES6 module file, the associated
 *     namespace or module otherwise null.
 */
goog.require = function(namespace) {
  if (!COMPILED) {
    // Might need to lazy load on old IE.
    if (goog.ENABLE_DEBUG_LOADER) {
      goog.debugLoader_.requested(namespace);
    }

    // If the object already exists we do not need to do anything.
    if (goog.isProvided_(namespace)) {
      if (goog.isInModuleLoader_()) {
        return goog.module.getInternal_(namespace);
      }
    } else if (goog.ENABLE_DEBUG_LOADER) {
      var moduleLoaderState = goog.moduleLoaderState_;
      goog.moduleLoaderState_ = null;
      try {
        goog.debugLoader_.load_(namespace);
      } finally {
        goog.moduleLoaderState_ = moduleLoaderState;
      }
    }

    return null;
  }
};


/**
 * Requires a symbol for its type information. This is an indication to the
 * compiler that the symbol may appear in type annotations, yet it is not
 * referenced at runtime.
 *
 * When called within a goog.module or ES6 module file, the return value may be
 * assigned to or destructured into a variable, but it may not be otherwise used
 * in code outside of a type annotation.
 *
 * Note that all calls to goog.requireType will be stripped by the compiler.
 *
 * @param {string} namespace Namespace (as was given in goog.provide,
 *     goog.module, or goog.declareModuleId) in the form
 *     "goog.package.part".
 * @return {?}
 */
goog.requireType = function(namespace) {
  // Return an empty object so that single-level destructuring of the return
  // value doesn't crash at runtime when using the debug loader. Multi-level
  // destructuring isn't supported.
  return {};
};


/**
 * Path for included scripts.
 * @type {string}
 */
goog.basePath = '';


/**
 * A hook for overriding the base path.
 * @type {string|undefined}
 */
goog.global.CLOSURE_BASE_PATH;


/**
 * Whether to attempt to load Closure's deps file. By default, when uncompiled,
 * deps files will attempt to be loaded.
 * @type {boolean|undefined}
 */
goog.global.CLOSURE_NO_DEPS;


/**
 * A function to import a single script. This is meant to be overridden when
 * Closure is being run in non-HTML contexts, such as web workers. It's defined
 * in the global scope so that it can be set before base.js is loaded, which
 * allows deps.js to be imported properly.
 *
 * The first parameter the script source, which is a relative URI. The second,
 * optional parameter is the script contents, in the event the script needed
 * transformation. It should return true if the script was imported, false
 * otherwise.
 * @type {(function(string, string=): boolean)|undefined}
 */
goog.global.CLOSURE_IMPORT_SCRIPT;


/**
 * Null function used for default values of callbacks, etc.
 * @return {void} Nothing.
 */
goog.nullFunction = function() {};


/**
 * When defining a class Foo with an abstract method bar(), you can do:
 * Foo.prototype.bar = goog.abstractMethod
 *
 * Now if a subclass of Foo fails to override bar(), an error will be thrown
 * when bar() is invoked.
 *
 * @type {!Function}
 * @throws {Error} when invoked to indicate the method should be overridden.
 * @deprecated Use "@abstract" annotation instead of goog.abstractMethod in new
 *     code. See
 *     https://github.com/google/closure-compiler/wiki/@abstract-classes-and-methods
 */
goog.abstractMethod = function() {
  throw new Error('unimplemented abstract method');
};


/**
 * Adds a `getInstance` static method that always returns the same
 * instance object.
 * @param {!Function} ctor The constructor for the class to add the static
 *     method to.
 * @suppress {missingProperties} 'instance_' isn't a property on 'Function'
 *     but we don't have a better type to use here.
 */
goog.addSingletonGetter = function(ctor) {
  // instance_ is immediately set to prevent issues with sealed constructors
  // such as are encountered when a constructor is returned as the export object
  // of a goog.module in unoptimized code.
  // Delcare type to avoid conformance violations that ctor.instance_ is unknown
  /** @type {undefined|!Object} @suppress {underscore} */
  ctor.instance_ = undefined;
  ctor.getInstance = function() {
    if (ctor.instance_) {
      return ctor.instance_;
    }
    if (goog.DEBUG) {
      // NOTE: JSCompiler can't optimize away Array#push.
      goog.instantiatedSingletons_[goog.instantiatedSingletons_.length] = ctor;
    }
    // Cast to avoid conformance violations that ctor.instance_ is unknown
    return /** @type {!Object|undefined} */ (ctor.instance_) = new ctor;
  };
};


/**
 * All singleton classes that have been instantiated, for testing. Don't read
 * it directly, use the `goog.testing.singleton` module. The compiler
 * removes this variable if unused.
 * @type {!Array<!Function>}
 * @private
 */
goog.instantiatedSingletons_ = [];


/**
 * @define {boolean} Whether to load goog.modules using `eval` when using
 * the debug loader.  This provides a better debugging experience as the
 * source is unmodified and can be edited using Chrome Workspaces or similar.
 * However in some environments the use of `eval` is banned
 * so we provide an alternative.
 */
goog.LOAD_MODULE_USING_EVAL = goog.define('goog.LOAD_MODULE_USING_EVAL', true);


/**
 * @define {boolean} Whether the exports of goog.modules should be sealed when
 * possible.
 */
goog.SEAL_MODULE_EXPORTS = goog.define('goog.SEAL_MODULE_EXPORTS', goog.DEBUG);


/**
 * The registry of initialized modules:
 * The module identifier or path to module exports map.
 * @private @const {!Object<string, {exports:?,type:string,moduleId:string}>}
 */
goog.loadedModules_ = {};


/**
 * True if the debug loader enabled and used.
 * @const {boolean}
 */
goog.DEPENDENCIES_ENABLED = !COMPILED && goog.ENABLE_DEBUG_LOADER;


/**
 * @define {string} How to decide whether to transpile.  Valid values
 * are 'always', 'never', and 'detect'.  The default ('detect') is to
 * use feature detection to determine which language levels need
 * transpilation.
 */
// NOTE(sdh): we could expand this to accept a language level to bypass
// detection: e.g. goog.TRANSPILE == 'es5' would transpile ES6 files but
// would leave ES3 and ES5 files alone.
goog.TRANSPILE = goog.define('goog.TRANSPILE', 'detect');

/**
 * @define {boolean} If true assume that ES modules have already been
 * transpiled by the jscompiler (in the same way that transpile.js would
 * transpile them - to jscomp modules). Useful only for servers that wish to use
 * the debug loader and transpile server side. Thus this is only respected if
 * goog.TRANSPILE is "never".
 */
goog.ASSUME_ES_MODULES_TRANSPILED =
    goog.define('goog.ASSUME_ES_MODULES_TRANSPILED', false);


/**
 * @define {string} If a file needs to be transpiled what the output language
 * should be. By default this is the highest language level this file detects
 * the current environment supports. Generally this flag should not be set, but
 * it could be useful to override. Example: If the current environment supports
 * ES6 then by default ES7+ files will be transpiled to ES6, unless this is
 * overridden.
 *
 * Valid values include: es3, es5, es6, es7, and es8. Anything not recognized
 * is treated as es3.
 *
 * Note that setting this value does not force transpilation. Just if
 * transpilation occurs this will be the output. So this is most useful when
 * goog.TRANSPILE is set to 'always' and then forcing the language level to be
 * something lower than what the environment detects.
 */
goog.TRANSPILE_TO_LANGUAGE = goog.define('goog.TRANSPILE_TO_LANGUAGE', '');


/**
 * @define {string} Path to the transpiler.  Executing the script at this
 * path (relative to base.js) should define a function $jscomp.transpile.
 */
goog.TRANSPILER = goog.define('goog.TRANSPILER', 'transpile.js');


/**
 * @package {?boolean}
 * Visible for testing.
 */
goog.hasBadLetScoping = null;


/**
 * @return {boolean}
 * @package Visible for testing.
 */
goog.useSafari10Workaround = function() {
  if (goog.hasBadLetScoping == null) {
    var hasBadLetScoping;
    try {
      hasBadLetScoping = !eval(
          '"use strict";' +
          'let x = 1; function f() { return typeof x; };' +
          'f() == "number";');
    } catch (e) {
      // Assume that ES6 syntax isn't supported.
      hasBadLetScoping = false;
    }
    goog.hasBadLetScoping = hasBadLetScoping;
  }
  return goog.hasBadLetScoping;
};


/**
 * @param {string} moduleDef
 * @return {string}
 * @package Visible for testing.
 */
goog.workaroundSafari10EvalBug = function(moduleDef) {
  return '(function(){' + moduleDef +
      '\n' +  // Terminate any trailing single line comment.
      ';' +   // Terminate any trailing expression.
      '})();\n';
};


/**
 * @param {function(?):?|string} moduleDef The module definition.
 */
goog.loadModule = function(moduleDef) {
  // NOTE: we allow function definitions to be either in the from
  // of a string to eval (which keeps the original source intact) or
  // in a eval forbidden environment (CSP) we allow a function definition
  // which in its body must call `goog.module`, and return the exports
  // of the module.
  var previousState = goog.moduleLoaderState_;
  try {
    goog.moduleLoaderState_ = {
      moduleName: '',
      declareLegacyNamespace: false,
      type: goog.ModuleType.GOOG
    };
    var exports;
    if (goog.isFunction(moduleDef)) {
      exports = moduleDef.call(undefined, {});
    } else if (typeof moduleDef === 'string') {
      if (goog.useSafari10Workaround()) {
        moduleDef = goog.workaroundSafari10EvalBug(moduleDef);
      }

      exports = goog.loadModuleFromSource_.call(undefined, moduleDef);
    } else {
      throw new Error('Invalid module definition');
    }

    var moduleName = goog.moduleLoaderState_.moduleName;
    if (typeof moduleName === 'string' && moduleName) {
      // Don't seal legacy namespaces as they may be used as a parent of
      // another namespace
      if (goog.moduleLoaderState_.declareLegacyNamespace) {
        goog.constructNamespace_(moduleName, exports);
      } else if (
          goog.SEAL_MODULE_EXPORTS && Object.seal &&
          typeof exports == 'object' && exports != null) {
        Object.seal(exports);
      }

      var data = {
        exports: exports,
        type: goog.ModuleType.GOOG,
        moduleId: goog.moduleLoaderState_.moduleName
      };
      goog.loadedModules_[moduleName] = data;
    } else {
      throw new Error('Invalid module name \"' + moduleName + '\"');
    }
  } finally {
    goog.moduleLoaderState_ = previousState;
  }
};


/**
 * @private @const
 */
goog.loadModuleFromSource_ = /** @type {function(string):?} */ (function() {
  // NOTE: we avoid declaring parameters or local variables here to avoid
  // masking globals or leaking values into the module definition.
  'use strict';
  var exports = {};
  eval(arguments[0]);
  return exports;
});


/**
 * Normalize a file path by removing redundant ".." and extraneous "." file
 * path components.
 * @param {string} path
 * @return {string}
 * @private
 */
goog.normalizePath_ = function(path) {
  var components = path.split('/');
  var i = 0;
  while (i < components.length) {
    if (components[i] == '.') {
      components.splice(i, 1);
    } else if (
        i && components[i] == '..' && components[i - 1] &&
        components[i - 1] != '..') {
      components.splice(--i, 2);
    } else {
      i++;
    }
  }
  return components.join('/');
};


/**
 * Provides a hook for loading a file when using Closure's goog.require() API
 * with goog.modules.  In particular this hook is provided to support Node.js.
 *
 * @type {(function(string):string)|undefined}
 */
goog.global.CLOSURE_LOAD_FILE_SYNC;


/**
 * Loads file by synchronous XHR. Should not be used in production environments.
 * @param {string} src Source URL.
 * @return {?string} File contents, or null if load failed.
 * @private
 */
goog.loadFileSync_ = function(src) {
  if (goog.global.CLOSURE_LOAD_FILE_SYNC) {
    return goog.global.CLOSURE_LOAD_FILE_SYNC(src);
  } else {
    try {
      /** @type {XMLHttpRequest} */
      var xhr = new goog.global['XMLHttpRequest']();
      xhr.open('get', src, false);
      xhr.send();
      // NOTE: Successful http: requests have a status of 200, but successful
      // file: requests may have a status of zero.  Any other status, or a
      // thrown exception (particularly in case of file: requests) indicates
      // some sort of error, which we treat as a missing or unavailable file.
      return xhr.status == 0 || xhr.status == 200 ? xhr.responseText : null;
    } catch (err) {
      // No need to rethrow or log, since errors should show up on their own.
      return null;
    }
  }
};


/**
 * Lazily retrieves the transpiler and applies it to the source.
 * @param {string} code JS code.
 * @param {string} path Path to the code.
 * @param {string} target Language level output.
 * @return {string} The transpiled code.
 * @private
 */
goog.transpile_ = function(code, path, target) {
  var jscomp = goog.global['$jscomp'];
  if (!jscomp) {
    goog.global['$jscomp'] = jscomp = {};
  }
  var transpile = jscomp.transpile;
  if (!transpile) {
    var transpilerPath = goog.basePath + goog.TRANSPILER;
    var transpilerCode = goog.loadFileSync_(transpilerPath);
    if (transpilerCode) {
      // This must be executed synchronously, since by the time we know we
      // need it, we're about to load and write the ES6 code synchronously,
      // so a normal script-tag load will be too slow. Wrapped in a function
      // so that code is eval'd in the global scope.
      (function() {
        (0, eval)(transpilerCode + '\n//# sourceURL=' + transpilerPath);
      }).call(goog.global);
      // Even though the transpiler is optional, if $gwtExport is found, it's
      // a sign the transpiler was loaded and the $jscomp.transpile *should*
      // be there.
      if (goog.global['$gwtExport'] && goog.global['$gwtExport']['$jscomp'] &&
          !goog.global['$gwtExport']['$jscomp']['transpile']) {
        throw new Error(
            'The transpiler did not properly export the "transpile" ' +
            'method. $gwtExport: ' + JSON.stringify(goog.global['$gwtExport']));
      }
      // transpile.js only exports a single $jscomp function, transpile. We
      // grab just that and add it to the existing definition of $jscomp which
      // contains the polyfills.
      goog.global['$jscomp'].transpile =
          goog.global['$gwtExport']['$jscomp']['transpile'];
      jscomp = goog.global['$jscomp'];
      transpile = jscomp.transpile;
    }
  }
  if (!transpile) {
    // The transpiler is an optional component.  If it's not available then
    // replace it with a pass-through function that simply logs.
    var suffix = ' requires transpilation but no transpiler was found.';
    transpile = jscomp.transpile = function(code, path) {
      // TODO(sdh): figure out some way to get this error to show up
      // in test results, noting that the failure may occur in many
      // different ways, including in loadModule() before the test
      // runner even comes up.
      goog.logToConsole_(path + suffix);
      return code;
    };
  }
  // Note: any transpilation errors/warnings will be logged to the console.
  return transpile(code, path, target);
};

//==============================================================================
// Language Enhancements
//==============================================================================


/**
 * This is a "fixed" version of the typeof operator.  It differs from the typeof
 * operator in such a way that null returns 'null' and arrays return 'array'.
 * @param {?} value The value to get the type of.
 * @return {string} The name of the type.
 */
goog.typeOf = function(value) {
  var s = typeof value;
  if (s == 'object') {
    if (value) {
      // Check these first, so we can avoid calling Object.prototype.toString if
      // possible.
      //
      // IE improperly marshals typeof across execution contexts, but a
      // cross-context object will still return false for "instanceof Object".
      if (value instanceof Array) {
        return 'array';
      } else if (value instanceof Object) {
        return s;
      }

      // HACK: In order to use an Object prototype method on the arbitrary
      //   value, the compiler requires the value be cast to type Object,
      //   even though the ECMA spec explicitly allows it.
      var className = Object.prototype.toString.call(
          /** @type {!Object} */ (value));
      // In Firefox 3.6, attempting to access iframe window objects' length
      // property throws an NS_ERROR_FAILURE, so we need to special-case it
      // here.
      if (className == '[object Window]') {
        return 'object';
      }

      // We cannot always use constructor == Array or instanceof Array because
      // different frames have different Array objects. In IE6, if the iframe
      // where the array was created is destroyed, the array loses its
      // prototype. Then dereferencing val.splice here throws an exception, so
      // we can't use goog.isFunction. Calling typeof directly returns 'unknown'
      // so that will work. In this case, this function will return false and
      // most array functions will still work because the array is still
      // array-like (supports length and []) even though it has lost its
      // prototype.
      // Mark Miller noticed that Object.prototype.toString
      // allows access to the unforgeable [[Class]] property.
      //  15.2.4.2 Object.prototype.toString ( )
      //  When the toString method is called, the following steps are taken:
      //      1. Get the [[Class]] property of this object.
      //      2. Compute a string value by concatenating the three strings
      //         "[object ", Result(1), and "]".
      //      3. Return Result(2).
      // and this behavior survives the destruction of the execution context.
      if ((className == '[object Array]' ||
           // In IE all non value types are wrapped as objects across window
           // boundaries (not iframe though) so we have to do object detection
           // for this edge case.
           typeof value.length == 'number' &&
               typeof value.splice != 'undefined' &&
               typeof value.propertyIsEnumerable != 'undefined' &&
               !value.propertyIsEnumerable('splice')

               )) {
        return 'array';
      }
      // HACK: There is still an array case that fails.
      //     function ArrayImpostor() {}
      //     ArrayImpostor.prototype = [];
      //     var impostor = new ArrayImpostor;
      // this can be fixed by getting rid of the fast path
      // (value instanceof Array) and solely relying on
      // (value && Object.prototype.toString.vall(value) === '[object Array]')
      // but that would require many more function calls and is not warranted
      // unless closure code is receiving objects from untrusted sources.

      // IE in cross-window calls does not correctly marshal the function type
      // (it appears just as an object) so we cannot use just typeof val ==
      // 'function'. However, if the object has a call property, it is a
      // function.
      if ((className == '[object Function]' ||
           typeof value.call != 'undefined' &&
               typeof value.propertyIsEnumerable != 'undefined' &&
               !value.propertyIsEnumerable('call'))) {
        return 'function';
      }

    } else {
      return 'null';
    }

  } else if (s == 'function' && typeof value.call == 'undefined') {
    // In Safari typeof nodeList returns 'function', and on Firefox typeof
    // behaves similarly for HTML{Applet,Embed,Object}, Elements and RegExps. We
    // would like to return object for those and we can detect an invalid
    // function by making sure that the function object has a call method.
    return 'object';
  }
  return s;
};


/**
 * Returns true if the specified value is null.
 * @param {?} val Variable to test.
 * @return {boolean} Whether variable is null.
 * @deprecated Use `val === null` instead.
 */
goog.isNull = function(val) {
  return val === null;
};


/**
 * Returns true if the specified value is defined and not null.
 * @param {?} val Variable to test.
 * @return {boolean} Whether variable is defined and not null.
 * @deprecated Use `val != null` instead.
 */
goog.isDefAndNotNull = function(val) {
  // Note that undefined == null.
  return val != null;
};


/**
 * Returns true if the specified value is an array.
 * @param {?} val Variable to test.
 * @return {boolean} Whether variable is an array.
 */
goog.isArray = function(val) {
  return goog.typeOf(val) == 'array';
};


/**
 * Returns true if the object looks like an array. To qualify as array like
 * the value needs to be either a NodeList or an object with a Number length
 * property. Note that for this function neither strings nor functions are
 * considered "array-like".
 *
 * @param {?} val Variable to test.
 * @return {boolean} Whether variable is an array.
 */
goog.isArrayLike = function(val) {
  var type = goog.typeOf(val);
  // We do not use goog.isObject here in order to exclude function values.
  return type == 'array' || type == 'object' && typeof val.length == 'number';
};


/**
 * Returns true if the object looks like a Date. To qualify as Date-like the
 * value needs to be an object and have a getFullYear() function.
 * @param {?} val Variable to test.
 * @return {boolean} Whether variable is a like a Date.
 */
goog.isDateLike = function(val) {
  return goog.isObject(val) && typeof val.getFullYear == 'function';
};


/**
 * Returns true if the specified value is a function.
 * @param {?} val Variable to test.
 * @return {boolean} Whether variable is a function.
 */
goog.isFunction = function(val) {
  return goog.typeOf(val) == 'function';
};


/**
 * Returns true if the specified value is an object.  This includes arrays and
 * functions.
 * @param {?} val Variable to test.
 * @return {boolean} Whether variable is an object.
 */
goog.isObject = function(val) {
  var type = typeof val;
  return type == 'object' && val != null || type == 'function';
  // return Object(val) === val also works, but is slower, especially if val is
  // not an object.
};


/**
 * Gets a unique ID for an object. This mutates the object so that further calls
 * with the same object as a parameter returns the same value. The unique ID is
 * guaranteed to be unique across the current session amongst objects that are
 * passed into `getUid`. There is no guarantee that the ID is unique or
 * consistent across sessions. It is unsafe to generate unique ID for function
 * prototypes.
 *
 * @param {Object} obj The object to get the unique ID for.
 * @return {number} The unique ID for the object.
 */
goog.getUid = function(obj) {
  // TODO(arv): Make the type stricter, do not accept null.

  // In Opera window.hasOwnProperty exists but always returns false so we avoid
  // using it. As a consequence the unique ID generated for BaseClass.prototype
  // and SubClass.prototype will be the same.
  // TODO(b/141512323): UUIDs are broken for ctors with class-side inheritance.
  return obj[goog.UID_PROPERTY_] ||
      (obj[goog.UID_PROPERTY_] = ++goog.uidCounter_);
};


/**
 * Whether the given object is already assigned a unique ID.
 *
 * This does not modify the object.
 *
 * @param {!Object} obj The object to check.
 * @return {boolean} Whether there is an assigned unique id for the object.
 */
goog.hasUid = function(obj) {
  return !!obj[goog.UID_PROPERTY_];
};


/**
 * Removes the unique ID from an object. This is useful if the object was
 * previously mutated using `goog.getUid` in which case the mutation is
 * undone.
 * @param {Object} obj The object to remove the unique ID field from.
 */
goog.removeUid = function(obj) {
  // TODO(arv): Make the type stricter, do not accept null.

  // In IE, DOM nodes are not instances of Object and throw an exception if we
  // try to delete.  Instead we try to use removeAttribute.
  if (obj !== null && 'removeAttribute' in obj) {
    obj.removeAttribute(goog.UID_PROPERTY_);
  }

  try {
    delete obj[goog.UID_PROPERTY_];
  } catch (ex) {
  }
};


/**
 * Name for unique ID property. Initialized in a way to help avoid collisions
 * with other closure JavaScript on the same page.
 * @type {string}
 * @private
 */
goog.UID_PROPERTY_ = 'closure_uid_' + ((Math.random() * 1e9) >>> 0);


/**
 * Counter for UID.
 * @type {number}
 * @private
 */
goog.uidCounter_ = 0;


/**
 * Adds a hash code field to an object. The hash code is unique for the
 * given object.
 * @param {Object} obj The object to get the hash code for.
 * @return {number} The hash code for the object.
 * @deprecated Use goog.getUid instead.
 */
goog.getHashCode = goog.getUid;


/**
 * Removes the hash code field from an object.
 * @param {Object} obj The object to remove the field from.
 * @deprecated Use goog.removeUid instead.
 */
goog.removeHashCode = goog.removeUid;


/**
 * Clones a value. The input may be an Object, Array, or basic type. Objects and
 * arrays will be cloned recursively.
 *
 * WARNINGS:
 * <code>goog.cloneObject</code> does not detect reference loops. Objects that
 * refer to themselves will cause infinite recursion.
 *
 * <code>goog.cloneObject</code> is unaware of unique identifiers, and copies
 * UIDs created by <code>getUid</code> into cloned results.
 *
 * @param {*} obj The value to clone.
 * @return {*} A clone of the input value.
 * @deprecated goog.cloneObject is unsafe. Prefer the goog.object methods.
 */
goog.cloneObject = function(obj) {
  var type = goog.typeOf(obj);
  if (type == 'object' || type == 'array') {
    if (typeof obj.clone === 'function') {
      return obj.clone();
    }
    var clone = type == 'array' ? [] : {};
    for (var key in obj) {
      clone[key] = goog.cloneObject(obj[key]);
    }
    return clone;
  }

  return obj;
};


/**
 * A native implementation of goog.bind.
 * @param {?function(this:T, ...)} fn A function to partially apply.
 * @param {T} selfObj Specifies the object which this should point to when the
 *     function is run.
 * @param {...*} var_args Additional arguments that are partially applied to the
 *     function.
 * @return {!Function} A partially-applied form of the function goog.bind() was
 *     invoked as a method of.
 * @template T
 * @private
 */
goog.bindNative_ = function(fn, selfObj, var_args) {
  return /** @type {!Function} */ (fn.call.apply(fn.bind, arguments));
};


/**
 * A pure-JS implementation of goog.bind.
 * @param {?function(this:T, ...)} fn A function to partially apply.
 * @param {T} selfObj Specifies the object which this should point to when the
 *     function is run.
 * @param {...*} var_args Additional arguments that are partially applied to the
 *     function.
 * @return {!Function} A partially-applied form of the function goog.bind() was
 *     invoked as a method of.
 * @template T
 * @private
 */
goog.bindJs_ = function(fn, selfObj, var_args) {
  if (!fn) {
    throw new Error();
  }

  if (arguments.length > 2) {
    var boundArgs = Array.prototype.slice.call(arguments, 2);
    return function() {
      // Prepend the bound arguments to the current arguments.
      var newArgs = Array.prototype.slice.call(arguments);
      Array.prototype.unshift.apply(newArgs, boundArgs);
      return fn.apply(selfObj, newArgs);
    };

  } else {
    return function() {
      return fn.apply(selfObj, arguments);
    };
  }
};


/**
 * Partially applies this function to a particular 'this object' and zero or
 * more arguments. The result is a new function with some arguments of the first
 * function pre-filled and the value of this 'pre-specified'.
 *
 * Remaining arguments specified at call-time are appended to the pre-specified
 * ones.
 *
 * Also see: {@link #partial}.
 *
 * Usage:
 * <pre>var barMethBound = goog.bind(myFunction, myObj, 'arg1', 'arg2');
 * barMethBound('arg3', 'arg4');</pre>
 *
 * @param {?function(this:T, ...)} fn A function to partially apply.
 * @param {T} selfObj Specifies the object which this should point to when the
 *     function is run.
 * @param {...*} var_args Additional arguments that are partially applied to the
 *     function.
 * @return {!Function} A partially-applied form of the function goog.bind() was
 *     invoked as a method of.
 * @template T
 * @suppress {deprecated} See above.
 */
goog.bind = function(fn, selfObj, var_args) {
  // TODO(nicksantos): narrow the type signature.
  if (Function.prototype.bind &&
      // NOTE(nicksantos): Somebody pulled base.js into the default Chrome
      // extension environment. This means that for Chrome extensions, they get
      // the implementation of Function.prototype.bind that calls goog.bind
      // instead of the native one. Even worse, we don't want to introduce a
      // circular dependency between goog.bind and Function.prototype.bind, so
      // we have to hack this to make sure it works correctly.
      Function.prototype.bind.toString().indexOf('native code') != -1) {
    goog.bind = goog.bindNative_;
  } else {
    goog.bind = goog.bindJs_;
  }
  return goog.bind.apply(null, arguments);
};


/**
 * Like goog.bind(), except that a 'this object' is not required. Useful when
 * the target function is already bound.
 *
 * Usage:
 * var g = goog.partial(f, arg1, arg2);
 * g(arg3, arg4);
 *
 * @param {Function} fn A function to partially apply.
 * @param {...*} var_args Additional arguments that are partially applied to fn.
 * @return {!Function} A partially-applied form of the function goog.partial()
 *     was invoked as a method of.
 */
goog.partial = function(fn, var_args) {
  var args = Array.prototype.slice.call(arguments, 1);
  return function() {
    // Clone the array (with slice()) and append additional arguments
    // to the existing arguments.
    var newArgs = args.slice();
    newArgs.push.apply(newArgs, arguments);
    return fn.apply(/** @type {?} */ (this), newArgs);
  };
};


/**
 * Copies all the members of a source object to a target object. This method
 * does not work on all browsers for all objects that contain keys such as
 * toString or hasOwnProperty. Use goog.object.extend for this purpose.
 *
 * NOTE: Some have advocated for the use of goog.mixin to setup classes
 * with multiple inheritence (traits, mixins, etc).  However, as it simply
 * uses "for in", this is not compatible with ES6 classes whose methods are
 * non-enumerable.  Changing this, would break cases where non-enumerable
 * properties are not expected.
 *
 * @param {Object} target Target.
 * @param {Object} source Source.
 * @deprecated Prefer Object.assign
 */
goog.mixin = function(target, source) {
  for (var x in source) {
    target[x] = source[x];
  }

  // For IE7 or lower, the for-in-loop does not contain any properties that are
  // not enumerable on the prototype object (for example, isPrototypeOf from
  // Object.prototype) but also it will not include 'replace' on objects that
  // extend String and change 'replace' (not that it is common for anyone to
  // extend anything except Object).
};


/**
 * @return {number} An integer value representing the number of milliseconds
 *     between midnight, January 1, 1970 and the current time.
 * @deprecated Use Date.now
 */
goog.now = (goog.TRUSTED_SITE && Date.now) || (function() {
             // Unary plus operator converts its operand to a number which in
             // the case of
             // a date is done by calling getTime().
             return +new Date();
           });


/**
 * Evals JavaScript in the global scope.  In IE this uses execScript, other
 * browsers use goog.global.eval. If goog.global.eval does not evaluate in the
 * global scope (for example, in Safari), appends a script tag instead.
 * Throws an exception if neither execScript or eval is defined.
 * @param {string} script JavaScript string.
 */
goog.globalEval = function(script) {
  if (goog.global.execScript) {
    goog.global.execScript(script, 'JavaScript');
  } else if (goog.global.eval) {
    // Test to see if eval works
    if (goog.evalWorksForGlobals_ == null) {
      try {
        goog.global.eval('var _evalTest_ = 1;');
      } catch (ignore) {
      }
      if (typeof goog.global['_evalTest_'] != 'undefined') {
        try {
          delete goog.global['_evalTest_'];
        } catch (ignore) {
          // Microsoft edge fails the deletion above in strict mode.
        }
        goog.evalWorksForGlobals_ = true;
      } else {
        goog.evalWorksForGlobals_ = false;
      }
    }

    if (goog.evalWorksForGlobals_) {
      goog.global.eval(script);
    } else {
      /** @type {!Document} */
      var doc = goog.global.document;
      var scriptElt =
          /** @type {!HTMLScriptElement} */ (doc.createElement('script'));
      scriptElt.type = 'text/javascript';
      scriptElt.defer = false;
      // Note(user): can't use .innerHTML since "t('<test>')" will fail and
      // .text doesn't work in Safari 2.  Therefore we append a text node.
      scriptElt.appendChild(doc.createTextNode(script));
      doc.head.appendChild(scriptElt);
      doc.head.removeChild(scriptElt);
    }
  } else {
    throw new Error('goog.globalEval not available');
  }
};


/**
 * Indicates whether or not we can call 'eval' directly to eval code in the
 * global scope. Set to a Boolean by the first call to goog.globalEval (which
 * empirically tests whether eval works for globals). @see goog.globalEval
 * @type {?boolean}
 * @private
 */
goog.evalWorksForGlobals_ = null;


/**
 * Optional map of CSS class names to obfuscated names used with
 * goog.getCssName().
 * @private {!Object<string, string>|undefined}
 * @see goog.setCssNameMapping
 */
goog.cssNameMapping_;


/**
 * Optional obfuscation style for CSS class names. Should be set to either
 * 'BY_WHOLE' or 'BY_PART' if defined.
 * @type {string|undefined}
 * @private
 * @see goog.setCssNameMapping
 */
goog.cssNameMappingStyle_;



/**
 * A hook for modifying the default behavior goog.getCssName. The function
 * if present, will receive the standard output of the goog.getCssName as
 * its input.
 *
 * @type {(function(string):string)|undefined}
 */
goog.global.CLOSURE_CSS_NAME_MAP_FN;


/**
 * Handles strings that are intended to be used as CSS class names.
 *
 * This function works in tandem with @see goog.setCssNameMapping.
 *
 * Without any mapping set, the arguments are simple joined with a hyphen and
 * passed through unaltered.
 *
 * When there is a mapping, there are two possible styles in which these
 * mappings are used. In the BY_PART style, each part (i.e. in between hyphens)
 * of the passed in css name is rewritten according to the map. In the BY_WHOLE
 * style, the full css name is looked up in the map directly. If a rewrite is
 * not specified by the map, the compiler will output a warning.
 *
 * When the mapping is passed to the compiler, it will replace calls to
 * goog.getCssName with the strings from the mapping, e.g.
 *     var x = goog.getCssName('foo');
 *     var y = goog.getCssName(this.baseClass, 'active');
 *  becomes:
 *     var x = 'foo';
 *     var y = this.baseClass + '-active';
 *
 * If one argument is passed it will be processed, if two are passed only the
 * modifier will be processed, as it is assumed the first argument was generated
 * as a result of calling goog.getCssName.
 *
 * @param {string} className The class name.
 * @param {string=} opt_modifier A modifier to be appended to the class name.
 * @return {string} The class name or the concatenation of the class name and
 *     the modifier.
 */
goog.getCssName = function(className, opt_modifier) {
  // String() is used for compatibility with compiled soy where the passed
  // className can be non-string objects.
  if (String(className).charAt(0) == '.') {
    throw new Error(
        'className passed in goog.getCssName must not start with ".".' +
        ' You passed: ' + className);
  }

  var getMapping = function(cssName) {
    return goog.cssNameMapping_[cssName] || cssName;
  };

  var renameByParts = function(cssName) {
    // Remap all the parts individually.
    var parts = cssName.split('-');
    var mapped = [];
    for (var i = 0; i < parts.length; i++) {
      mapped.push(getMapping(parts[i]));
    }
    return mapped.join('-');
  };

  var rename;
  if (goog.cssNameMapping_) {
    rename =
        goog.cssNameMappingStyle_ == 'BY_WHOLE' ? getMapping : renameByParts;
  } else {
    rename = function(a) {
      return a;
    };
  }

  var result =
      opt_modifier ? className + '-' + rename(opt_modifier) : rename(className);

  // The special CLOSURE_CSS_NAME_MAP_FN allows users to specify further
  // processing of the class name.
  if (goog.global.CLOSURE_CSS_NAME_MAP_FN) {
    return goog.global.CLOSURE_CSS_NAME_MAP_FN(result);
  }

  return result;
};


/**
 * Sets the map to check when returning a value from goog.getCssName(). Example:
 * <pre>
 * goog.setCssNameMapping({
 *   "goog": "a",
 *   "disabled": "b",
 * });
 *
 * var x = goog.getCssName('goog');
 * // The following evaluates to: "a a-b".
 * goog.getCssName('goog') + ' ' + goog.getCssName(x, 'disabled')
 * </pre>
 * When declared as a map of string literals to string literals, the JSCompiler
 * will replace all calls to goog.getCssName() using the supplied map if the
 * --process_closure_primitives flag is set.
 *
 * @param {!Object} mapping A map of strings to strings where keys are possible
 *     arguments to goog.getCssName() and values are the corresponding values
 *     that should be returned.
 * @param {string=} opt_style The style of css name mapping. There are two valid
 *     options: 'BY_PART', and 'BY_WHOLE'.
 * @see goog.getCssName for a description.
 */
goog.setCssNameMapping = function(mapping, opt_style) {
  goog.cssNameMapping_ = mapping;
  goog.cssNameMappingStyle_ = opt_style;
};


/**
 * To use CSS renaming in compiled mode, one of the input files should have a
 * call to goog.setCssNameMapping() with an object literal that the JSCompiler
 * can extract and use to replace all calls to goog.getCssName(). In uncompiled
 * mode, JavaScript code should be loaded before this base.js file that declares
 * a global variable, CLOSURE_CSS_NAME_MAPPING, which is used below. This is
 * to ensure that the mapping is loaded before any calls to goog.getCssName()
 * are made in uncompiled mode.
 *
 * A hook for overriding the CSS name mapping.
 * @type {!Object<string, string>|undefined}
 */
goog.global.CLOSURE_CSS_NAME_MAPPING;


if (!COMPILED && goog.global.CLOSURE_CSS_NAME_MAPPING) {
  // This does not call goog.setCssNameMapping() because the JSCompiler
  // requires that goog.setCssNameMapping() be called with an object literal.
  goog.cssNameMapping_ = goog.global.CLOSURE_CSS_NAME_MAPPING;
}


/**
 * Gets a localized message.
 *
 * This function is a compiler primitive. If you give the compiler a localized
 * message bundle, it will replace the string at compile-time with a localized
 * version, and expand goog.getMsg call to a concatenated string.
 *
 * Messages must be initialized in the form:
 * <code>
 * var MSG_NAME = goog.getMsg('Hello {$placeholder}', {'placeholder': 'world'});
 * </code>
 *
 * This function produces a string which should be treated as plain text. Use
 * {@link goog.html.SafeHtmlFormatter} in conjunction with goog.getMsg to
 * produce SafeHtml.
 *
 * @param {string} str Translatable string, places holders in the form {$foo}.
 * @param {Object<string, string>=} opt_values Maps place holder name to value.
 * @param {{html: boolean}=} opt_options Options:
 *     html: Escape '<' in str to '&lt;'. Used by Closure Templates where the
 *     generated code size and performance is critical which is why {@link
 *     goog.html.SafeHtmlFormatter} is not used. The value must be literal true
 *     or false.
 * @return {string} message with placeholders filled.
 */
goog.getMsg = function(str, opt_values, opt_options) {
  if (opt_options && opt_options.html) {
    // Note that '&' is not replaced because the translation can contain HTML
    // entities.
    str = str.replace(/</g, '&lt;');
  }
  if (opt_values) {
    str = str.replace(/\{\$([^}]+)}/g, function(match, key) {
      return (opt_values != null && key in opt_values) ? opt_values[key] :
                                                         match;
    });
  }
  return str;
};


/**
 * Gets a localized message. If the message does not have a translation, gives a
 * fallback message.
 *
 * This is useful when introducing a new message that has not yet been
 * translated into all languages.
 *
 * This function is a compiler primitive. Must be used in the form:
 * <code>var x = goog.getMsgWithFallback(MSG_A, MSG_B);</code>
 * where MSG_A and MSG_B were initialized with goog.getMsg.
 *
 * @param {string} a The preferred message.
 * @param {string} b The fallback message.
 * @return {string} The best translated message.
 */
goog.getMsgWithFallback = function(a, b) {
  return a;
};


/**
 * Exposes an unobfuscated global namespace path for the given object.
 * Note that fields of the exported object *will* be obfuscated, unless they are
 * exported in turn via this function or goog.exportProperty.
 *
 * Also handy for making public items that are defined in anonymous closures.
 *
 * ex. goog.exportSymbol('public.path.Foo', Foo);
 *
 * ex. goog.exportSymbol('public.path.Foo.staticFunction', Foo.staticFunction);
 *     public.path.Foo.staticFunction();
 *
 * ex. goog.exportSymbol('public.path.Foo.prototype.myMethod',
 *                       Foo.prototype.myMethod);
 *     new public.path.Foo().myMethod();
 *
 * @param {string} publicPath Unobfuscated name to export.
 * @param {*} object Object the name should point to.
 * @param {Object=} opt_objectToExportTo The object to add the path to; default
 *     is goog.global.
 */
goog.exportSymbol = function(publicPath, object, opt_objectToExportTo) {
  goog.exportPath_(publicPath, object, opt_objectToExportTo);
};


/**
 * Exports a property unobfuscated into the object's namespace.
 * ex. goog.exportProperty(Foo, 'staticFunction', Foo.staticFunction);
 * ex. goog.exportProperty(Foo.prototype, 'myMethod', Foo.prototype.myMethod);
 * @param {Object} object Object whose static property is being exported.
 * @param {string} publicName Unobfuscated name to export.
 * @param {*} symbol Object the name should point to.
 */
goog.exportProperty = function(object, publicName, symbol) {
  object[publicName] = symbol;
};


/**
 * Inherit the prototype methods from one constructor into another.
 *
 * Usage:
 * <pre>
 * function ParentClass(a, b) { }
 * ParentClass.prototype.foo = function(a) { };
 *
 * function ChildClass(a, b, c) {
 *   ChildClass.base(this, 'constructor', a, b);
 * }
 * goog.inherits(ChildClass, ParentClass);
 *
 * var child = new ChildClass('a', 'b', 'see');
 * child.foo(); // This works.
 * </pre>
 *
 * @param {!Function} childCtor Child class.
 * @param {!Function} parentCtor Parent class.
 * @suppress {strictMissingProperties} superClass_ and base is not defined on
 *    Function.
 */
goog.inherits = function(childCtor, parentCtor) {
  /** @constructor */
  function tempCtor() {}
  tempCtor.prototype = parentCtor.prototype;
  childCtor.superClass_ = parentCtor.prototype;
  childCtor.prototype = new tempCtor();
  /** @override */
  childCtor.prototype.constructor = childCtor;

  /**
   * Calls superclass constructor/method.
   *
   * This function is only available if you use goog.inherits to
   * express inheritance relationships between classes.
   *
   * NOTE: This is a replacement for goog.base and for superClass_
   * property defined in childCtor.
   *
   * @param {!Object} me Should always be "this".
   * @param {string} methodName The method name to call. Calling
   *     superclass constructor can be done with the special string
   *     'constructor'.
   * @param {...*} var_args The arguments to pass to superclass
   *     method/constructor.
   * @return {*} The return value of the superclass method/constructor.
   */
  childCtor.base = function(me, methodName, var_args) {
    // Copying using loop to avoid deop due to passing arguments object to
    // function. This is faster in many JS engines as of late 2014.
    var args = new Array(arguments.length - 2);
    for (var i = 2; i < arguments.length; i++) {
      args[i - 2] = arguments[i];
    }
    return parentCtor.prototype[methodName].apply(me, args);
  };
};


/**
 * Call up to the superclass.
 *
 * If this is called from a constructor, then this calls the superclass
 * constructor with arguments 1-N.
 *
 * If this is called from a prototype method, then you must pass the name of the
 * method as the second argument to this function. If you do not, you will get a
 * runtime error. This calls the superclass' method with arguments 2-N.
 *
 * This function only works if you use goog.inherits to express inheritance
 * relationships between your classes.
 *
 * This function is a compiler primitive. At compile-time, the compiler will do
 * macro expansion to remove a lot of the extra overhead that this function
 * introduces. The compiler will also enforce a lot of the assumptions that this
 * function makes, and treat it as a compiler error if you break them.
 *
 * @param {!Object} me Should always be "this".
 * @param {*=} opt_methodName The method name if calling a super method.
 * @param {...*} var_args The rest of the arguments.
 * @return {*} The return value of the superclass method.
 * @suppress {es5Strict} This method can not be used in strict mode, but
 *     all Closure Library consumers must depend on this file.
 * @deprecated goog.base is not strict mode compatible.  Prefer the static
 *     "base" method added to the constructor by goog.inherits
 *     or ES6 classes and the "super" keyword.
 */
goog.base = function(me, opt_methodName, var_args) {
  var caller = arguments.callee.caller;

  if (goog.STRICT_MODE_COMPATIBLE || (goog.DEBUG && !caller)) {
    throw new Error(
        'arguments.caller not defined.  goog.base() cannot be used ' +
        'with strict mode code. See ' +
        'http://www.ecma-international.org/ecma-262/5.1/#sec-C');
  }

  if (typeof caller.superClass_ !== 'undefined') {
    // Copying using loop to avoid deop due to passing arguments object to
    // function. This is faster in many JS engines as of late 2014.
    var ctorArgs = new Array(arguments.length - 1);
    for (var i = 1; i < arguments.length; i++) {
      ctorArgs[i - 1] = arguments[i];
    }
    // This is a constructor. Call the superclass constructor.
    return /** @type {!Function} */ (caller.superClass_)
        .constructor.apply(me, ctorArgs);
  }

  if (typeof opt_methodName != 'string' && typeof opt_methodName != 'symbol') {
    throw new Error(
        'method names provided to goog.base must be a string or a symbol');
  }

  // Copying using loop to avoid deop due to passing arguments object to
  // function. This is faster in many JS engines as of late 2014.
  var args = new Array(arguments.length - 2);
  for (var i = 2; i < arguments.length; i++) {
    args[i - 2] = arguments[i];
  }
  var foundCaller = false;
  for (var proto = me.constructor.prototype; proto;
       proto = Object.getPrototypeOf(proto)) {
    if (proto[opt_methodName] === caller) {
      foundCaller = true;
    } else if (foundCaller) {
      return proto[opt_methodName].apply(me, args);
    }
  }

  // If we did not find the caller in the prototype chain, then one of two
  // things happened:
  // 1) The caller is an instance method.
  // 2) This method was not called by the right caller.
  if (me[opt_methodName] === caller) {
    return me.constructor.prototype[opt_methodName].apply(me, args);
  } else {
    throw new Error(
        'goog.base called from a method of one name ' +
        'to a method of a different name');
  }
};


/**
 * Allow for aliasing within scope functions.  This function exists for
 * uncompiled code - in compiled code the calls will be inlined and the aliases
 * applied.  In uncompiled code the function is simply run since the aliases as
 * written are valid JavaScript.
 *
 *
 * @param {function()} fn Function to call.  This function can contain aliases
 *     to namespaces (e.g. "var dom = goog.dom") or classes
 *     (e.g. "var Timer = goog.Timer").
 */
goog.scope = function(fn) {
  if (goog.isInModuleLoader_()) {
    throw new Error('goog.scope is not supported within a module.');
  }
  fn.call(goog.global);
};


/*
 * To support uncompiled, strict mode bundles that use eval to divide source
 * like so:
 *    eval('someSource;//# sourceUrl sourcefile.js');
 * We need to export the globally defined symbols "goog" and "COMPILED".
 * Exporting "goog" breaks the compiler optimizations, so we required that
 * be defined externally.
 * NOTE: We don't use goog.exportSymbol here because we don't want to trigger
 * extern generation when that compiler option is enabled.
 */
if (!COMPILED) {
  goog.global['COMPILED'] = COMPILED;
}


//==============================================================================
// goog.defineClass implementation
//==============================================================================


/**
 * Creates a restricted form of a Closure "class":
 *   - from the compiler's perspective, the instance returned from the
 *     constructor is sealed (no new properties may be added).  This enables
 *     better checks.
 *   - the compiler will rewrite this definition to a form that is optimal
 *     for type checking and optimization (initially this will be a more
 *     traditional form).
 *
 * @param {Function} superClass The superclass, Object or null.
 * @param {goog.defineClass.ClassDescriptor} def
 *     An object literal describing
 *     the class.  It may have the following properties:
 *     "constructor": the constructor function
 *     "statics": an object literal containing methods to add to the constructor
 *        as "static" methods or a function that will receive the constructor
 *        function as its only parameter to which static properties can
 *        be added.
 *     all other properties are added to the prototype.
 * @return {!Function} The class constructor.
 * @deprecated Use ES6 class syntax instead.
 */
goog.defineClass = function(superClass, def) {
  // TODO(johnlenz): consider making the superClass an optional parameter.
  var constructor = def.constructor;
  var statics = def.statics;
  // Wrap the constructor prior to setting up the prototype and static methods.
  if (!constructor || constructor == Object.prototype.constructor) {
    constructor = function() {
      throw new Error(
          'cannot instantiate an interface (no constructor defined).');
    };
  }

  var cls = goog.defineClass.createSealingConstructor_(constructor, superClass);
  if (superClass) {
    goog.inherits(cls, superClass);
  }

  // Remove all the properties that should not be copied to the prototype.
  delete def.constructor;
  delete def.statics;

  goog.defineClass.applyProperties_(cls.prototype, def);
  if (statics != null) {
    if (statics instanceof Function) {
      statics(cls);
    } else {
      goog.defineClass.applyProperties_(cls, statics);
    }
  }

  return cls;
};


/**
 * @typedef {{
 *   constructor: (!Function|undefined),
 *   statics: (Object|undefined|function(Function):void)
 * }}
 */
goog.defineClass.ClassDescriptor;


/**
 * @define {boolean} Whether the instances returned by goog.defineClass should
 *     be sealed when possible.
 *
 * When sealing is disabled the constructor function will not be wrapped by
 * goog.defineClass, making it incompatible with ES6 class methods.
 */
goog.defineClass.SEAL_CLASS_INSTANCES =
    goog.define('goog.defineClass.SEAL_CLASS_INSTANCES', goog.DEBUG);


/**
 * If goog.defineClass.SEAL_CLASS_INSTANCES is enabled and Object.seal is
 * defined, this function will wrap the constructor in a function that seals the
 * results of the provided constructor function.
 *
 * @param {!Function} ctr The constructor whose results maybe be sealed.
 * @param {Function} superClass The superclass constructor.
 * @return {!Function} The replacement constructor.
 * @private
 */
goog.defineClass.createSealingConstructor_ = function(ctr, superClass) {
  if (!goog.defineClass.SEAL_CLASS_INSTANCES) {
    // Do now wrap the constructor when sealing is disabled. Angular code
    // depends on this for injection to work properly.
    return ctr;
  }

  // Compute whether the constructor is sealable at definition time, rather
  // than when the instance is being constructed.
  var superclassSealable = !goog.defineClass.isUnsealable_(superClass);

  /**
   * @this {Object}
   * @return {?}
   */
  var wrappedCtr = function() {
    // Don't seal an instance of a subclass when it calls the constructor of
    // its super class as there is most likely still setup to do.
    var instance = ctr.apply(this, arguments) || this;
    instance[goog.UID_PROPERTY_] = instance[goog.UID_PROPERTY_];

    if (this.constructor === wrappedCtr && superclassSealable &&
        Object.seal instanceof Function) {
      Object.seal(instance);
    }
    return instance;
  };

  return wrappedCtr;
};


/**
 * @param {Function} ctr The constructor to test.
 * @return {boolean} Whether the constructor has been tagged as unsealable
 *     using goog.tagUnsealableClass.
 * @private
 */
goog.defineClass.isUnsealable_ = function(ctr) {
  return ctr && ctr.prototype &&
      ctr.prototype[goog.UNSEALABLE_CONSTRUCTOR_PROPERTY_];
};


// TODO(johnlenz): share these values with the goog.object
/**
 * The names of the fields that are defined on Object.prototype.
 * @type {!Array<string>}
 * @private
 * @const
 */
goog.defineClass.OBJECT_PROTOTYPE_FIELDS_ = [
  'constructor', 'hasOwnProperty', 'isPrototypeOf', 'propertyIsEnumerable',
  'toLocaleString', 'toString', 'valueOf'
];


// TODO(johnlenz): share this function with the goog.object
/**
 * @param {!Object} target The object to add properties to.
 * @param {!Object} source The object to copy properties from.
 * @private
 */
goog.defineClass.applyProperties_ = function(target, source) {
  // TODO(johnlenz): update this to support ES5 getters/setters

  var key;
  for (key in source) {
    if (Object.prototype.hasOwnProperty.call(source, key)) {
      target[key] = source[key];
    }
  }

  // For IE the for-in-loop does not contain any properties that are not
  // enumerable on the prototype object (for example isPrototypeOf from
  // Object.prototype) and it will also not include 'replace' on objects that
  // extend String and change 'replace' (not that it is common for anyone to
  // extend anything except Object).
  for (var i = 0; i < goog.defineClass.OBJECT_PROTOTYPE_FIELDS_.length; i++) {
    key = goog.defineClass.OBJECT_PROTOTYPE_FIELDS_[i];
    if (Object.prototype.hasOwnProperty.call(source, key)) {
      target[key] = source[key];
    }
  }
};


/**
 * Sealing classes breaks the older idiom of assigning properties on the
 * prototype rather than in the constructor. As such, goog.defineClass
 * must not seal subclasses of these old-style classes until they are fixed.
 * Until then, this marks a class as "broken", instructing defineClass
 * not to seal subclasses.
 * @param {!Function} ctr The legacy constructor to tag as unsealable.
 */
goog.tagUnsealableClass = function(ctr) {
  if (!COMPILED && goog.defineClass.SEAL_CLASS_INSTANCES) {
    ctr.prototype[goog.UNSEALABLE_CONSTRUCTOR_PROPERTY_] = true;
  }
};


/**
 * Name for unsealable tag property.
 * @const @private {string}
 */
goog.UNSEALABLE_CONSTRUCTOR_PROPERTY_ = 'goog_defineClass_legacy_unsealable';


// There's a bug in the compiler where without collapse properties the
// Closure namespace defines do not guard code correctly. To help reduce code
// size also check for !COMPILED even though it redundant until this is fixed.
if (!COMPILED && goog.DEPENDENCIES_ENABLED) {

  /**
   * Tries to detect whether is in the context of an HTML document.
   * @return {boolean} True if it looks like HTML document.
   * @private
   */
  goog.inHtmlDocument_ = function() {
    /** @type {!Document} */
    var doc = goog.global.document;
    return doc != null && 'write' in doc;  // XULDocument misses write.
  };


  /**
   * We'd like to check for if the document readyState is 'loading'; however
   * there are bugs on IE 10 and below where the readyState being anything other
   * than 'complete' is not reliable.
   * @return {boolean}
   * @private
   */
  goog.isDocumentLoading_ = function() {
    // attachEvent is available on IE 6 thru 10 only, and thus can be used to
    // detect those browsers.
    /** @type {!HTMLDocument} */
    var doc = goog.global.document;
    return doc.attachEvent ? doc.readyState != 'complete' :
                             doc.readyState == 'loading';
  };


  /**
   * Tries to detect the base path of base.js script that bootstraps Closure.
   * @private
   */
  goog.findBasePath_ = function() {
    if (goog.global.CLOSURE_BASE_PATH != undefined &&
        // Anti DOM-clobbering runtime check (b/37736576).
        typeof goog.global.CLOSURE_BASE_PATH === 'string') {
      goog.basePath = goog.global.CLOSURE_BASE_PATH;
      return;
    } else if (!goog.inHtmlDocument_()) {
      return;
    }
    /** @type {!Document} */
    var doc = goog.global.document;
    // If we have a currentScript available, use it exclusively.
    var currentScript = doc.currentScript;
    if (currentScript) {
      var scripts = [currentScript];
    } else {
      var scripts = doc.getElementsByTagName('SCRIPT');
    }
    // Search backwards since the current script is in almost all cases the one
    // that has base.js.
    for (var i = scripts.length - 1; i >= 0; --i) {
      var script = /** @type {!HTMLScriptElement} */ (scripts[i]);
      var src = script.src;
      var qmark = src.lastIndexOf('?');
      var l = qmark == -1 ? src.length : qmark;
      if (src.substr(l - 7, 7) == 'base.js') {
        goog.basePath = src.substr(0, l - 7);
        return;
      }
    }
  };

  goog.findBasePath_();

  /** @struct @constructor @final */
  goog.Transpiler = function() {
    /** @private {?Object<string, boolean>} */
    this.requiresTranspilation_ = null;
    /** @private {string} */
    this.transpilationTarget_ = goog.TRANSPILE_TO_LANGUAGE;
  };


  /**
   * Returns a newly created map from language mode string to a boolean
   * indicating whether transpilation should be done for that mode as well as
   * the highest level language that this environment supports.
   *
   * Guaranteed invariant:
   * For any two modes, l1 and l2 where l2 is a newer mode than l1,
   * `map[l1] == true` implies that `map[l2] == true`.
   *
   * Note this method is extracted and used elsewhere, so it cannot rely on
   * anything external (it should easily be able to be transformed into a
   * standalone, top level function).
   *
   * @private
   * @return {{
   *   target: string,
   *   map: !Object<string, boolean>
   * }}
   */
  goog.Transpiler.prototype.createRequiresTranspilation_ = function() {
    var transpilationTarget = 'es3';
    var /** !Object<string, boolean> */ requiresTranspilation = {'es3': false};
    var transpilationRequiredForAllLaterModes = false;

    /**
     * Adds an entry to requiresTranspliation for the given language mode.
     *
     * IMPORTANT: Calls must be made in order from oldest to newest language
     * mode.
     * @param {string} modeName
     * @param {function(): boolean} isSupported Returns true if the JS engine
     *     supports the given mode.
     */
    function addNewerLanguageTranspilationCheck(modeName, isSupported) {
      if (transpilationRequiredForAllLaterModes) {
        requiresTranspilation[modeName] = true;
      } else if (isSupported()) {
        transpilationTarget = modeName;
        requiresTranspilation[modeName] = false;
      } else {
        requiresTranspilation[modeName] = true;
        transpilationRequiredForAllLaterModes = true;
      }
    }

    /**
     * Does the given code evaluate without syntax errors and return a truthy
     * result?
     */
    function /** boolean */ evalCheck(/** string */ code) {
      try {
        return !!eval(code);
      } catch (ignored) {
        return false;
      }
    }

    var userAgent = goog.global.navigator && goog.global.navigator.userAgent ?
        goog.global.navigator.userAgent :
        '';

    // Identify ES3-only browsers by their incorrect treatment of commas.
    addNewerLanguageTranspilationCheck('es5', function() {
      return evalCheck('[1,].length==1');
    });
    addNewerLanguageTranspilationCheck('es6', function() {
      // Edge has a non-deterministic (i.e., not reproducible) bug with ES6:
      // https://github.com/Microsoft/ChakraCore/issues/1496.
      var re = /Edge\/(\d+)(\.\d)*/i;
      var edgeUserAgent = userAgent.match(re);
      if (edgeUserAgent) {
        // The Reflect.construct test below is flaky on Edge. It can sometimes
        // pass or fail on 40 15.15063, so just exit early for Edge and treat
        // it as ES5. Until we're on a more up to date version just always use
        // ES5. See https://github.com/Microsoft/ChakraCore/issues/3217.
        return false;
      }
      // Test es6: [FF50 (?), Edge 14 (?), Chrome 50]
      //   (a) default params (specifically shadowing locals),
      //   (b) destructuring, (c) block-scoped functions,
      //   (d) for-of (const), (e) new.target/Reflect.construct
      var es6fullTest =
          'class X{constructor(){if(new.target!=String)throw 1;this.x=42}}' +
          'let q=Reflect.construct(X,[],String);if(q.x!=42||!(q instanceof ' +
          'String))throw 1;for(const a of[2,3]){if(a==2)continue;function ' +
          'f(z={a}){let a=0;return z.a}{function f(){return 0;}}return f()' +
          '==3}';

      return evalCheck('(()=>{"use strict";' + es6fullTest + '})()');
    });
    // ** and **= are the only new features in 'es7'
    addNewerLanguageTranspilationCheck('es7', function() {
      return evalCheck('2 ** 2 == 4');
    });
    // async functions are the only new features in 'es8'
    addNewerLanguageTranspilationCheck('es8', function() {
      return evalCheck('async () => 1, true');
    });
    addNewerLanguageTranspilationCheck('es9', function() {
      return evalCheck('({...rest} = {}), true');
    });
    addNewerLanguageTranspilationCheck('es_next', function() {
      return false;  // assume it always need to transpile
    });
    return {target: transpilationTarget, map: requiresTranspilation};
  };


  /**
   * Determines whether the given language needs to be transpiled.
   * @param {string} lang
   * @param {string|undefined} module
   * @return {boolean}
   */
  goog.Transpiler.prototype.needsTranspile = function(lang, module) {
    if (goog.TRANSPILE == 'always') {
      return true;
    } else if (goog.TRANSPILE == 'never') {
      return false;
    } else if (!this.requiresTranspilation_) {
      var obj = this.createRequiresTranspilation_();
      this.requiresTranspilation_ = obj.map;
      this.transpilationTarget_ = this.transpilationTarget_ || obj.target;
    }
    if (lang in this.requiresTranspilation_) {
      if (this.requiresTranspilation_[lang]) {
        return true;
      } else if (
          goog.inHtmlDocument_() && module == 'es6' &&
          !('noModule' in goog.global.document.createElement('script'))) {
        return true;
      } else {
        return false;
      }
    } else {
      throw new Error('Unknown language mode: ' + lang);
    }
  };


  /**
   * Lazily retrieves the transpiler and applies it to the source.
   * @param {string} code JS code.
   * @param {string} path Path to the code.
   * @return {string} The transpiled code.
   */
  goog.Transpiler.prototype.transpile = function(code, path) {
    // TODO(johnplaisted): We should delete goog.transpile_ and just have this
    // function. But there's some compile error atm where goog.global is being
    // stripped incorrectly without this.
    return goog.transpile_(code, path, this.transpilationTarget_);
  };


  /** @private @final {!goog.Transpiler} */
  goog.transpiler_ = new goog.Transpiler();

  /**
   * Rewrites closing script tags in input to avoid ending an enclosing script
   * tag.
   *
   * @param {string} str
   * @return {string}
   * @private
   */
  goog.protectScriptTag_ = function(str) {
    return str.replace(/<\/(SCRIPT)/ig, '\\x3c/$1');
  };


  /**
   * A debug loader is responsible for downloading and executing javascript
   * files in an unbundled, uncompiled environment.
   *
   * This can be custimized via the setDependencyFactory method, or by
   * CLOSURE_IMPORT_SCRIPT/CLOSURE_LOAD_FILE_SYNC.
   *
   * @struct @constructor @final @private
   */
  goog.DebugLoader_ = function() {
    /** @private @const {!Object<string, !goog.Dependency>} */
    this.dependencies_ = {};
    /** @private @const {!Object<string, string>} */
    this.idToPath_ = {};
    /** @private @const {!Object<string, boolean>} */
    this.written_ = {};
    /** @private @const {!Array<!goog.Dependency>} */
    this.loadingDeps_ = [];
    /** @private {!Array<!goog.Dependency>} */
    this.depsToLoad_ = [];
    /** @private {boolean} */
    this.paused_ = false;
    /** @private {!goog.DependencyFactory} */
    this.factory_ = new goog.DependencyFactory(goog.transpiler_);
    /** @private @const {!Object<string, !Function>} */
    this.deferredCallbacks_ = {};
    /** @private @const {!Array<string>} */
    this.deferredQueue_ = [];
  };

  /**
   * @param {!Array<string>} namespaces
   * @param {function(): undefined} callback Function to call once all the
   *     namespaces have loaded.
   */
  goog.DebugLoader_.prototype.bootstrap = function(namespaces, callback) {
    var cb = callback;
    function resolve() {
      if (cb) {
        goog.global.setTimeout(cb, 0);
        cb = null;
      }
    }

    if (!namespaces.length) {
      resolve();
      return;
    }

    var deps = [];
    for (var i = 0; i < namespaces.length; i++) {
      var path = this.getPathFromDeps_(namespaces[i]);
      if (!path) {
        throw new Error('Unregonized namespace: ' + namespaces[i]);
      }
      deps.push(this.dependencies_[path]);
    }

    var require = goog.require;
    var loaded = 0;
    for (var i = 0; i < namespaces.length; i++) {
      require(namespaces[i]);
      deps[i].onLoad(function() {
        if (++loaded == namespaces.length) {
          resolve();
        }
      });
    }
  };


  /**
   * Loads the Closure Dependency file.
   *
   * Exposed a public function so CLOSURE_NO_DEPS can be set to false, base
   * loaded, setDependencyFactory called, and then this called. i.e. allows
   * custom loading of the deps file.
   */
  goog.DebugLoader_.prototype.loadClosureDeps = function() {
    // Circumvent addDependency, which would try to transpile deps.js if
    // transpile is set to always.
    var relPath = 'deps.js';
    this.depsToLoad_.push(this.factory_.createDependency(
        goog.normalizePath_(goog.basePath + relPath), relPath, [], [], {},
        false));
    this.loadDeps_();
  };


  /**
   * Notifies the debug loader when a dependency has been requested.
   *
   * @param {string} absPathOrId Path of the dependency or goog id.
   * @param {boolean=} opt_force
   */
  goog.DebugLoader_.prototype.requested = function(absPathOrId, opt_force) {
    var path = this.getPathFromDeps_(absPathOrId);
    if (path &&
        (opt_force || this.areDepsLoaded_(this.dependencies_[path].requires))) {
      var callback = this.deferredCallbacks_[path];
      if (callback) {
        delete this.deferredCallbacks_[path];
        callback();
      }
    }
  };


  /**
   * Sets the dependency factory, which can be used to create custom
   * goog.Dependency implementations to control how dependencies are loaded.
   *
   * @param {!goog.DependencyFactory} factory
   */
  goog.DebugLoader_.prototype.setDependencyFactory = function(factory) {
    this.factory_ = factory;
  };


  /**
   * Travserses the dependency graph and queues the given dependency, and all of
   * its transitive dependencies, for loading and then starts loading if not
   * paused.
   *
   * @param {string} namespace
   * @private
   */
  goog.DebugLoader_.prototype.load_ = function(namespace) {
    if (!this.getPathFromDeps_(namespace)) {
      var errorMessage = 'goog.require could not find: ' + namespace;

      goog.logToConsole_(errorMessage);
      throw Error(errorMessage);
    } else {
      var loader = this;

      var deps = [];

      /** @param {string} namespace */
      var visit = function(namespace) {
        var path = loader.getPathFromDeps_(namespace);

        if (!path) {
          throw new Error('Bad dependency path or symbol: ' + namespace);
        }

        if (loader.written_[path]) {
          return;
        }

        loader.written_[path] = true;

        var dep = loader.dependencies_[path];
        for (var i = 0; i < dep.requires.length; i++) {
          if (!goog.isProvided_(dep.requires[i])) {
            visit(dep.requires[i]);
          }
        }

        deps.push(dep);
      };

      visit(namespace);

      var wasLoading = !!this.depsToLoad_.length;
      this.depsToLoad_ = this.depsToLoad_.concat(deps);

      if (!this.paused_ && !wasLoading) {
        this.loadDeps_();
      }
    }
  };


  /**
   * Loads any queued dependencies until they are all loaded or paused.
   *
   * @private
   */
  goog.DebugLoader_.prototype.loadDeps_ = function() {
    var loader = this;
    var paused = this.paused_;

    while (this.depsToLoad_.length && !paused) {
      (function() {
        var loadCallDone = false;
        var dep = loader.depsToLoad_.shift();

        var loaded = false;
        loader.loading_(dep);

        var controller = {
          pause: function() {
            if (loadCallDone) {
              throw new Error('Cannot call pause after the call to load.');
            } else {
              paused = true;
            }
          },
          resume: function() {
            if (loadCallDone) {
              loader.resume_();
            } else {
              // Some dep called pause and then resume in the same load call.
              // Just keep running this same loop.
              paused = false;
            }
          },
          loaded: function() {
            if (loaded) {
              throw new Error('Double call to loaded.');
            }

            loaded = true;
            loader.loaded_(dep);
          },
          pending: function() {
            // Defensive copy.
            var pending = [];
            for (var i = 0; i < loader.loadingDeps_.length; i++) {
              pending.push(loader.loadingDeps_[i]);
            }
            return pending;
          },
          /**
           * @param {goog.ModuleType} type
           */
          setModuleState: function(type) {
            goog.moduleLoaderState_ = {
              type: type,
              moduleName: '',
              declareLegacyNamespace: false
            };
          },
          /** @type {function(string, string, string=)} */
          registerEs6ModuleExports: function(
              path, exports, opt_closureNamespace) {
            if (opt_closureNamespace) {
              goog.loadedModules_[opt_closureNamespace] = {
                exports: exports,
                type: goog.ModuleType.ES6,
                moduleId: opt_closureNamespace || ''
              };
            }
          },
          /** @type {function(string, ?)} */
          registerGoogModuleExports: function(moduleId, exports) {
            goog.loadedModules_[moduleId] = {
              exports: exports,
              type: goog.ModuleType.GOOG,
              moduleId: moduleId
            };
          },
          clearModuleState: function() {
            goog.moduleLoaderState_ = null;
          },
          defer: function(callback) {
            if (loadCallDone) {
              throw new Error(
                  'Cannot register with defer after the call to load.');
            }
            loader.defer_(dep, callback);
          },
          areDepsLoaded: function() {
            return loader.areDepsLoaded_(dep.requires);
          }
        };

        try {
          dep.load(controller);
        } finally {
          loadCallDone = true;
        }
      })();
    }

    if (paused) {
      this.pause_();
    }
  };


  /** @private */
  goog.DebugLoader_.prototype.pause_ = function() {
    this.paused_ = true;
  };


  /** @private */
  goog.DebugLoader_.prototype.resume_ = function() {
    if (this.paused_) {
      this.paused_ = false;
      this.loadDeps_();
    }
  };


  /**
   * Marks the given dependency as loading (load has been called but it has not
   * yet marked itself as finished). Useful for dependencies that want to know
   * what else is loading. Example: goog.modules cannot eval if there are
   * loading dependencies.
   *
   * @param {!goog.Dependency} dep
   * @private
   */
  goog.DebugLoader_.prototype.loading_ = function(dep) {
    this.loadingDeps_.push(dep);
  };


  /**
   * Marks the given dependency as having finished loading and being available
   * for require.
   *
   * @param {!goog.Dependency} dep
   * @private
   */
  goog.DebugLoader_.prototype.loaded_ = function(dep) {
    for (var i = 0; i < this.loadingDeps_.length; i++) {
      if (this.loadingDeps_[i] == dep) {
        this.loadingDeps_.splice(i, 1);
        break;
      }
    }

    for (var i = 0; i < this.deferredQueue_.length; i++) {
      if (this.deferredQueue_[i] == dep.path) {
        this.deferredQueue_.splice(i, 1);
        break;
      }
    }

    if (this.loadingDeps_.length == this.deferredQueue_.length &&
        !this.depsToLoad_.length) {
      // Something has asked to load these, but they may not be directly
      // required again later, so load them now that we know we're done loading
      // everything else. e.g. a goog module entry point.
      while (this.deferredQueue_.length) {
        this.requested(this.deferredQueue_.shift(), true);
      }
    }

    dep.loaded();
  };


  /**
   * @param {!Array<string>} pathsOrIds
   * @return {boolean}
   * @private
   */
  goog.DebugLoader_.prototype.areDepsLoaded_ = function(pathsOrIds) {
    for (var i = 0; i < pathsOrIds.length; i++) {
      var path = this.getPathFromDeps_(pathsOrIds[i]);
      if (!path ||
          (!(path in this.deferredCallbacks_) &&
           !goog.isProvided_(pathsOrIds[i]))) {
        return false;
      }
    }

    return true;
  };


  /**
   * @param {string} absPathOrId
   * @return {?string}
   * @private
   */
  goog.DebugLoader_.prototype.getPathFromDeps_ = function(absPathOrId) {
    if (absPathOrId in this.idToPath_) {
      return this.idToPath_[absPathOrId];
    } else if (absPathOrId in this.dependencies_) {
      return absPathOrId;
    } else {
      return null;
    }
  };


  /**
   * @param {!goog.Dependency} dependency
   * @param {!Function} callback
   * @private
   */
  goog.DebugLoader_.prototype.defer_ = function(dependency, callback) {
    this.deferredCallbacks_[dependency.path] = callback;
    this.deferredQueue_.push(dependency.path);
  };


  /**
   * Interface for goog.Dependency implementations to have some control over
   * loading of dependencies.
   *
   * @record
   */
  goog.LoadController = function() {};


  /**
   * Tells the controller to halt loading of more dependencies.
   */
  goog.LoadController.prototype.pause = function() {};


  /**
   * Tells the controller to resume loading of more dependencies if paused.
   */
  goog.LoadController.prototype.resume = function() {};


  /**
   * Tells the controller that this dependency has finished loading.
   *
   * This causes this to be removed from pending() and any load callbacks to
   * fire.
   */
  goog.LoadController.prototype.loaded = function() {};


  /**
   * List of dependencies on which load has been called but which have not
   * called loaded on their controller. This includes the current dependency.
   *
   * @return {!Array<!goog.Dependency>}
   */
  goog.LoadController.prototype.pending = function() {};


  /**
   * Registers an object as an ES6 module's exports so that goog.modules may
   * require it by path.
   *
   * @param {string} path Full path of the module.
   * @param {?} exports
   * @param {string=} opt_closureNamespace Closure namespace to associate with
   *     this module.
   */
  goog.LoadController.prototype.registerEs6ModuleExports = function(
      path, exports, opt_closureNamespace) {};


  /**
   * Sets the current module state.
   *
   * @param {goog.ModuleType} type Type of module.
   */
  goog.LoadController.prototype.setModuleState = function(type) {};


  /**
   * Clears the current module state.
   */
  goog.LoadController.prototype.clearModuleState = function() {};


  /**
   * Registers a callback to call once the dependency is actually requested
   * via goog.require + all of the immediate dependencies have been loaded or
   * all other files have been loaded. Allows for lazy loading until
   * require'd without pausing dependency loading, which is needed on old IE.
   *
   * @param {!Function} callback
   */
  goog.LoadController.prototype.defer = function(callback) {};


  /**
   * @return {boolean}
   */
  goog.LoadController.prototype.areDepsLoaded = function() {};


  /**
   * Basic super class for all dependencies Closure Library can load.
   *
   * This default implementation is designed to load untranspiled, non-module
   * scripts in a web broswer.
   *
   * For transpiled non-goog.module files {@see goog.TranspiledDependency}.
   * For goog.modules see {@see goog.GoogModuleDependency}.
   * For untranspiled ES6 modules {@see goog.Es6ModuleDependency}.
   *
   * @param {string} path Absolute path of this script.
   * @param {string} relativePath Path of this script relative to goog.basePath.
   * @param {!Array<string>} provides goog.provided or goog.module symbols
   *     in this file.
   * @param {!Array<string>} requires goog symbols or relative paths to Closure
   *     this depends on.
   * @param {!Object<string, string>} loadFlags
   * @struct @constructor
   */
  goog.Dependency = function(
      path, relativePath, provides, requires, loadFlags) {
    /** @const */
    this.path = path;
    /** @const */
    this.relativePath = relativePath;
    /** @const */
    this.provides = provides;
    /** @const */
    this.requires = requires;
    /** @const */
    this.loadFlags = loadFlags;
    /** @private {boolean} */
    this.loaded_ = false;
    /** @private {!Array<function()>} */
    this.loadCallbacks_ = [];
  };


  /**
   * @return {string} The pathname part of this dependency's path if it is a
   *     URI.
   */
  goog.Dependency.prototype.getPathName = function() {
    var pathName = this.path;
    var protocolIndex = pathName.indexOf('://');
    if (protocolIndex >= 0) {
      pathName = pathName.substring(protocolIndex + 3);
      var slashIndex = pathName.indexOf('/');
      if (slashIndex >= 0) {
        pathName = pathName.substring(slashIndex + 1);
      }
    }
    return pathName;
  };


  /**
   * @param {function()} callback Callback to fire as soon as this has loaded.
   * @final
   */
  goog.Dependency.prototype.onLoad = function(callback) {
    if (this.loaded_) {
      callback();
    } else {
      this.loadCallbacks_.push(callback);
    }
  };


  /**
   * Marks this dependency as loaded and fires any callbacks registered with
   * onLoad.
   * @final
   */
  goog.Dependency.prototype.loaded = function() {
    this.loaded_ = true;
    var callbacks = this.loadCallbacks_;
    this.loadCallbacks_ = [];
    for (var i = 0; i < callbacks.length; i++) {
      callbacks[i]();
    }
  };


  /**
   * Whether or not document.written / appended script tags should be deferred.
   *
   * @private {boolean}
   */
  goog.Dependency.defer_ = false;


  /**
   * Map of script ready / state change callbacks. Old IE cannot handle putting
   * these properties on goog.global.
   *
   * @private @const {!Object<string, function(?):undefined>}
   */
  goog.Dependency.callbackMap_ = {};


  /**
   * @param {function(...?):?} callback
   * @return {string}
   * @private
   */
  goog.Dependency.registerCallback_ = function(callback) {
    var key = Math.random().toString(32);
    goog.Dependency.callbackMap_[key] = callback;
    return key;
  };


  /**
   * @param {string} key
   * @private
   */
  goog.Dependency.unregisterCallback_ = function(key) {
    delete goog.Dependency.callbackMap_[key];
  };


  /**
   * @param {string} key
   * @param {...?} var_args
   * @private
   * @suppress {unusedPrivateMembers}
   */
  goog.Dependency.callback_ = function(key, var_args) {
    if (key in goog.Dependency.callbackMap_) {
      var callback = goog.Dependency.callbackMap_[key];
      var args = [];
      for (var i = 1; i < arguments.length; i++) {
        args.push(arguments[i]);
      }
      callback.apply(undefined, args);
    } else {
      var errorMessage = 'Callback key ' + key +
          ' does not exist (was base.js loaded more than once?).';
      throw Error(errorMessage);
    }
  };


  /**
   * Starts loading this dependency. This dependency can pause loading if it
   * needs to and resume it later via the controller interface.
   *
   * When this is loaded it should call controller.loaded(). Note that this will
   * end up calling the loaded method of this dependency; there is no need to
   * call it explicitly.
   *
   * @param {!goog.LoadController} controller
   */
  goog.Dependency.prototype.load = function(controller) {
    if (goog.global.CLOSURE_IMPORT_SCRIPT) {
      if (goog.global.CLOSURE_IMPORT_SCRIPT(this.path)) {
        controller.loaded();
      } else {
        controller.pause();
      }
      return;
    }

    if (!goog.inHtmlDocument_()) {
      goog.logToConsole_(
          'Cannot use default debug loader outside of HTML documents.');
      if (this.relativePath == 'deps.js') {
        // Some old code is relying on base.js auto loading deps.js failing with
        // no error before later setting CLOSURE_IMPORT_SCRIPT.
        // CLOSURE_IMPORT_SCRIPT should be set *before* base.js is loaded, or
        // CLOSURE_NO_DEPS set to true.
        goog.logToConsole_(
            'Consider setting CLOSURE_IMPORT_SCRIPT before loading base.js, ' +
            'or setting CLOSURE_NO_DEPS to true.');
        controller.loaded();
      } else {
        controller.pause();
      }
      return;
    }

    /** @type {!HTMLDocument} */
    var doc = goog.global.document;

    // If the user tries to require a new symbol after document load,
    // something has gone terribly wrong. Doing a document.write would
    // wipe out the page. This does not apply to the CSP-compliant method
    // of writing script tags.
    if (doc.readyState == 'complete' &&
        !goog.ENABLE_CHROME_APP_SAFE_SCRIPT_LOADING) {
      // Certain test frameworks load base.js multiple times, which tries
      // to write deps.js each time. If that happens, just fail silently.
      // These frameworks wipe the page between each load of base.js, so this
      // is OK.
      var isDeps = /\bdeps.js$/.test(this.path);
      if (isDeps) {
        controller.loaded();
        return;
      } else {
        throw Error('Cannot write "' + this.path + '" after document load');
      }
    }

    if (!goog.ENABLE_CHROME_APP_SAFE_SCRIPT_LOADING &&
        goog.isDocumentLoading_()) {
      var key = goog.Dependency.registerCallback_(function(script) {
        if (!goog.DebugLoader_.IS_OLD_IE_ || script.readyState == 'complete') {
          goog.Dependency.unregisterCallback_(key);
          controller.loaded();
        }
      });
      var nonceAttr = !goog.DebugLoader_.IS_OLD_IE_ && goog.getScriptNonce() ?
          ' nonce="' + goog.getScriptNonce() + '"' :
          '';
      var event =
          goog.DebugLoader_.IS_OLD_IE_ ? 'onreadystatechange' : 'onload';
      var defer = goog.Dependency.defer_ ? 'defer' : '';
      var script = '<script src="' + this.path + '" ' + event +
          '="goog.Dependency.callback_(\'' + key +
          '\', this)" type="text/javascript" ' + defer + nonceAttr + '><' +
          '/script>';
      doc.write(
          goog.TRUSTED_TYPES_POLICY_ ?
              goog.TRUSTED_TYPES_POLICY_.createHTML(script) :
              script);
    } else {
      var scriptEl =
          /** @type {!HTMLScriptElement} */ (doc.createElement('script'));
      scriptEl.defer = goog.Dependency.defer_;
      scriptEl.async = false;
      scriptEl.type = 'text/javascript';

      // If CSP nonces are used, propagate them to dynamically created scripts.
      // This is necessary to allow nonce-based CSPs without 'strict-dynamic'.
      var nonce = goog.getScriptNonce();
      if (nonce) {
        scriptEl.setAttribute('nonce', nonce);
      }

      if (goog.DebugLoader_.IS_OLD_IE_) {
        // Execution order is not guaranteed on old IE, halt loading and write
        // these scripts one at a time, after each loads.
        controller.pause();
        scriptEl.onreadystatechange = function() {
          if (scriptEl.readyState == 'loaded' ||
              scriptEl.readyState == 'complete') {
            controller.loaded();
            controller.resume();
          }
        };
      } else {
        scriptEl.onload = function() {
          scriptEl.onload = null;
          controller.loaded();
        };
      }

      scriptEl.src = goog.TRUSTED_TYPES_POLICY_ ?
          goog.TRUSTED_TYPES_POLICY_.createScriptURL(this.path) :
          this.path;
      doc.head.appendChild(scriptEl);
    }
  };


  /**
   * @param {string} path Absolute path of this script.
   * @param {string} relativePath Path of this script relative to goog.basePath.
   * @param {!Array<string>} provides Should be an empty array.
   *     TODO(johnplaisted) add support for adding closure namespaces to ES6
   *     modules for interop purposes.
   * @param {!Array<string>} requires goog symbols or relative paths to Closure
   *     this depends on.
   * @param {!Object<string, string>} loadFlags
   * @struct @constructor
   * @extends {goog.Dependency}
   */
  goog.Es6ModuleDependency = function(
      path, relativePath, provides, requires, loadFlags) {
    goog.Es6ModuleDependency.base(
        this, 'constructor', path, relativePath, provides, requires, loadFlags);
  };
  goog.inherits(goog.Es6ModuleDependency, goog.Dependency);


  /** @override */
  goog.Es6ModuleDependency.prototype.load = function(controller) {
    if (goog.global.CLOSURE_IMPORT_SCRIPT) {
      if (goog.global.CLOSURE_IMPORT_SCRIPT(this.path)) {
        controller.loaded();
      } else {
        controller.pause();
      }
      return;
    }

    if (!goog.inHtmlDocument_()) {
      goog.logToConsole_(
          'Cannot use default debug loader outside of HTML documents.');
      controller.pause();
      return;
    }

    /** @type {!HTMLDocument} */
    var doc = goog.global.document;

    var dep = this;

    // TODO(johnplaisted): Does document.writing really speed up anything? Any
    // difference between this and just waiting for interactive mode and then
    // appending?
    function write(src, contents) {
      if (contents) {
        var script = '<script type="module" crossorigin>' + contents + '</' +
            'script>';
        doc.write(
            goog.TRUSTED_TYPES_POLICY_ ?
                goog.TRUSTED_TYPES_POLICY_.createHTML(script) :
                script);
      } else {
        var script = '<script type="module" crossorigin src="' + src + '"></' +
            'script>';
        doc.write(
            goog.TRUSTED_TYPES_POLICY_ ?
                goog.TRUSTED_TYPES_POLICY_.createHTML(script) :
                script);
      }
    }

    function append(src, contents) {
      var scriptEl =
          /** @type {!HTMLScriptElement} */ (doc.createElement('script'));
      scriptEl.defer = true;
      scriptEl.async = false;
      scriptEl.type = 'module';
      scriptEl.setAttribute('crossorigin', true);

      // If CSP nonces are used, propagate them to dynamically created scripts.
      // This is necessary to allow nonce-based CSPs without 'strict-dynamic'.
      var nonce = goog.getScriptNonce();
      if (nonce) {
        scriptEl.setAttribute('nonce', nonce);
      }

      if (contents) {
        scriptEl.textContent = goog.TRUSTED_TYPES_POLICY_ ?
            goog.TRUSTED_TYPES_POLICY_.createScript(contents) :
            contents;
      } else {
        scriptEl.src = goog.TRUSTED_TYPES_POLICY_ ?
            goog.TRUSTED_TYPES_POLICY_.createScriptURL(src) :
            src;
      }

      doc.head.appendChild(scriptEl);
    }

    var create;

    if (goog.isDocumentLoading_()) {
      create = write;
      // We can ONLY call document.write if we are guaranteed that any
      // non-module script tags document.written after this are deferred.
      // Small optimization, in theory document.writing is faster.
      goog.Dependency.defer_ = true;
    } else {
      create = append;
    }

    // Write 4 separate tags here:
    // 1) Sets the module state at the correct time (just before execution).
    // 2) A src node for this, which just hopefully lets the browser load it a
    //    little early (no need to parse #3).
    // 3) Import the module and register it.
    // 4) Clear the module state at the correct time. Guaranteed to run even
    //    if there is an error in the module (#3 will not run if there is an
    //    error in the module).
    var beforeKey = goog.Dependency.registerCallback_(function() {
      goog.Dependency.unregisterCallback_(beforeKey);
      controller.setModuleState(goog.ModuleType.ES6);
    });
    create(undefined, 'goog.Dependency.callback_("' + beforeKey + '")');

    // TODO(johnplaisted): Does this really speed up anything?
    create(this.path, undefined);

    var registerKey = goog.Dependency.registerCallback_(function(exports) {
      goog.Dependency.unregisterCallback_(registerKey);
      controller.registerEs6ModuleExports(
          dep.path, exports, goog.moduleLoaderState_.moduleName);
    });
    create(
        undefined,
        'import * as m from "' + this.path + '"; goog.Dependency.callback_("' +
            registerKey + '", m)');

    var afterKey = goog.Dependency.registerCallback_(function() {
      goog.Dependency.unregisterCallback_(afterKey);
      controller.clearModuleState();
      controller.loaded();
    });
    create(undefined, 'goog.Dependency.callback_("' + afterKey + '")');
  };


  /**
   * Superclass of any dependency that needs to be loaded into memory,
   * transformed, and then eval'd (goog.modules and transpiled files).
   *
   * @param {string} path Absolute path of this script.
   * @param {string} relativePath Path of this script relative to goog.basePath.
   * @param {!Array<string>} provides goog.provided or goog.module symbols
   *     in this file.
   * @param {!Array<string>} requires goog symbols or relative paths to Closure
   *     this depends on.
   * @param {!Object<string, string>} loadFlags
   * @struct @constructor @abstract
   * @extends {goog.Dependency}
   */
  goog.TransformedDependency = function(
      path, relativePath, provides, requires, loadFlags) {
    goog.TransformedDependency.base(
        this, 'constructor', path, relativePath, provides, requires, loadFlags);
    /** @private {?string} */
    this.contents_ = null;

    /**
     * Whether to lazily make the synchronous XHR (when goog.require'd) or make
     * the synchronous XHR when initially loading. On FireFox 61 there is a bug
     * where an ES6 module cannot make a synchronous XHR (rather, it can, but if
     * it does then no other ES6 modules will load after).
     *
     * tl;dr we lazy load due to bugs on older browsers and eager load due to
     * bugs on newer ones.
     *
     * https://bugzilla.mozilla.org/show_bug.cgi?id=1477090
     *
     * @private @const {boolean}
     */
    this.lazyFetch_ = !goog.inHtmlDocument_() ||
        !('noModule' in goog.global.document.createElement('script'));
  };
  goog.inherits(goog.TransformedDependency, goog.Dependency);


  /** @override */
  goog.TransformedDependency.prototype.load = function(controller) {
    var dep = this;

    function fetch() {
      dep.contents_ = goog.loadFileSync_(dep.path);

      if (dep.contents_) {
        dep.contents_ = dep.transform(dep.contents_);
        if (dep.contents_) {
          dep.contents_ += '\n//# sourceURL=' + dep.path;
        }
      }
    }

    if (goog.global.CLOSURE_IMPORT_SCRIPT) {
      fetch();
      if (this.contents_ &&
          goog.global.CLOSURE_IMPORT_SCRIPT('', this.contents_)) {
        this.contents_ = null;
        controller.loaded();
      } else {
        controller.pause();
      }
      return;
    }


    var isEs6 = this.loadFlags['module'] == goog.ModuleType.ES6;

    if (!this.lazyFetch_) {
      fetch();
    }

    function load() {
      if (dep.lazyFetch_) {
        fetch();
      }

      if (!dep.contents_) {
        // loadFileSync_ or transform are responsible. Assume they logged an
        // error.
        return;
      }

      if (isEs6) {
        controller.setModuleState(goog.ModuleType.ES6);
      }

      var namespace;

      try {
        var contents = dep.contents_;
        dep.contents_ = null;
        goog.globalEval(contents);
        if (isEs6) {
          namespace = goog.moduleLoaderState_.moduleName;
        }
      } finally {
        if (isEs6) {
          controller.clearModuleState();
        }
      }

      if (isEs6) {
        // Due to circular dependencies this may not be available for require
        // right now.
        goog.global['$jscomp']['require']['ensure'](
            [dep.getPathName()], function() {
              controller.registerEs6ModuleExports(
                  dep.path,
                  goog.global['$jscomp']['require'](dep.getPathName()),
                  namespace);
            });
      }

      controller.loaded();
    }

    // Do not fetch now; in FireFox 47 the synchronous XHR doesn't block all
    // events. If we fetched now and then document.write'd the contents the
    // document.write would be an eval and would execute too soon! Instead write
    // a script tag to fetch and eval synchronously at the correct time.
    function fetchInOwnScriptThenLoad() {
      /** @type {!HTMLDocument} */
      var doc = goog.global.document;

      var key = goog.Dependency.registerCallback_(function() {
        goog.Dependency.unregisterCallback_(key);
        load();
      });

      var script = '<script type="text/javascript">' +
          goog.protectScriptTag_('goog.Dependency.callback_("' + key + '");') +
          '</' +
          'script>';
      doc.write(
          goog.TRUSTED_TYPES_POLICY_ ?
              goog.TRUSTED_TYPES_POLICY_.createHTML(script) :
              script);
    }

    // If one thing is pending it is this.
    var anythingElsePending = controller.pending().length > 1;

    // If anything else is loading we need to lazy load due to bugs in old IE.
    // Specifically script tags with src and script tags with contents could
    // execute out of order if document.write is used, so we cannot use
    // document.write. Do not pause here; it breaks old IE as well.
    var useOldIeWorkAround =
        anythingElsePending && goog.DebugLoader_.IS_OLD_IE_;

    // Additionally if we are meant to defer scripts but the page is still
    // loading (e.g. an ES6 module is loading) then also defer. Or if we are
    // meant to defer and anything else is pending then defer (those may be
    // scripts that did not need transformation and are just script tags with
    // defer set to true, and we need to evaluate after that deferred script).
    var needsAsyncLoading = goog.Dependency.defer_ &&
        (anythingElsePending || goog.isDocumentLoading_());

    if (useOldIeWorkAround || needsAsyncLoading) {
      // Note that we only defer when we have to rather than 100% of the time.
      // Always defering would work, but then in theory the order of
      // goog.require calls would then matter. We want to enforce that most of
      // the time the order of the require calls does not matter.
      controller.defer(function() {
        load();
      });
      return;
    }
    // TODO(johnplaisted): Externs are missing onreadystatechange for
    // HTMLDocument.
    /** @type {?} */
    var doc = goog.global.document;

    var isInternetExplorer =
        goog.inHtmlDocument_() && 'ActiveXObject' in goog.global;

    // Don't delay in any version of IE. There's bug around this that will
    // cause out of order script execution. This means that on older IE ES6
    // modules will load too early (while the document is still loading + the
    // dom is not available). The other option is to load too late (when the
    // document is complete and the onload even will never fire). This seems
    // to be the lesser of two evils as scripts already act like the former.
    if (isEs6 && goog.inHtmlDocument_() && goog.isDocumentLoading_() &&
        !isInternetExplorer) {
      goog.Dependency.defer_ = true;
      // Transpiled ES6 modules still need to load like regular ES6 modules,
      // aka only after the document is interactive.
      controller.pause();
      var oldCallback = doc.onreadystatechange;
      doc.onreadystatechange = function() {
        if (doc.readyState == 'interactive') {
          doc.onreadystatechange = oldCallback;
          load();
          controller.resume();
        }
        if (goog.isFunction(oldCallback)) {
          oldCallback.apply(undefined, arguments);
        }
      };
    } else {
      // Always eval on old IE.
      if (goog.DebugLoader_.IS_OLD_IE_ || !goog.inHtmlDocument_() ||
          !goog.isDocumentLoading_()) {
        load();
      } else {
        fetchInOwnScriptThenLoad();
      }
    }
  };


  /**
   * @param {string} contents
   * @return {string}
   * @abstract
   */
  goog.TransformedDependency.prototype.transform = function(contents) {};


  /**
   * Any non-goog.module dependency which needs to be transpiled before eval.
   *
   * @param {string} path Absolute path of this script.
   * @param {string} relativePath Path of this script relative to goog.basePath.
   * @param {!Array<string>} provides goog.provided or goog.module symbols
   *     in this file.
   * @param {!Array<string>} requires goog symbols or relative paths to Closure
   *     this depends on.
   * @param {!Object<string, string>} loadFlags
   * @param {!goog.Transpiler} transpiler
   * @struct @constructor
   * @extends {goog.TransformedDependency}
   */
  goog.TranspiledDependency = function(
      path, relativePath, provides, requires, loadFlags, transpiler) {
    goog.TranspiledDependency.base(
        this, 'constructor', path, relativePath, provides, requires, loadFlags);
    /** @protected @const*/
    this.transpiler = transpiler;
  };
  goog.inherits(goog.TranspiledDependency, goog.TransformedDependency);


  /** @override */
  goog.TranspiledDependency.prototype.transform = function(contents) {
    // Transpile with the pathname so that ES6 modules are domain agnostic.
    return this.transpiler.transpile(contents, this.getPathName());
  };


  /**
   * An ES6 module dependency that was transpiled to a jscomp module outside
   * of the debug loader, e.g. server side.
   *
   * @param {string} path Absolute path of this script.
   * @param {string} relativePath Path of this script relative to goog.basePath.
   * @param {!Array<string>} provides goog.provided or goog.module symbols
   *     in this file.
   * @param {!Array<string>} requires goog symbols or relative paths to Closure
   *     this depends on.
   * @param {!Object<string, string>} loadFlags
   * @struct @constructor
   * @extends {goog.TransformedDependency}
   */
  goog.PreTranspiledEs6ModuleDependency = function(
      path, relativePath, provides, requires, loadFlags) {
    goog.PreTranspiledEs6ModuleDependency.base(
        this, 'constructor', path, relativePath, provides, requires, loadFlags);
  };
  goog.inherits(
      goog.PreTranspiledEs6ModuleDependency, goog.TransformedDependency);


  /** @override */
  goog.PreTranspiledEs6ModuleDependency.prototype.transform = function(
      contents) {
    return contents;
  };


  /**
   * A goog.module, transpiled or not. Will always perform some minimal
   * transformation even when not transpiled to wrap in a goog.loadModule
   * statement.
   *
   * @param {string} path Absolute path of this script.
   * @param {string} relativePath Path of this script relative to goog.basePath.
   * @param {!Array<string>} provides goog.provided or goog.module symbols
   *     in this file.
   * @param {!Array<string>} requires goog symbols or relative paths to Closure
   *     this depends on.
   * @param {!Object<string, string>} loadFlags
   * @param {boolean} needsTranspile
   * @param {!goog.Transpiler} transpiler
   * @struct @constructor
   * @extends {goog.TransformedDependency}
   */
  goog.GoogModuleDependency = function(
      path, relativePath, provides, requires, loadFlags, needsTranspile,
      transpiler) {
    goog.GoogModuleDependency.base(
        this, 'constructor', path, relativePath, provides, requires, loadFlags);
    /** @private @const */
    this.needsTranspile_ = needsTranspile;
    /** @private @const */
    this.transpiler_ = transpiler;
  };
  goog.inherits(goog.GoogModuleDependency, goog.TransformedDependency);


  /** @override */
  goog.GoogModuleDependency.prototype.transform = function(contents) {
    if (this.needsTranspile_) {
      contents = this.transpiler_.transpile(contents, this.getPathName());
    }

    if (!goog.LOAD_MODULE_USING_EVAL || goog.global.JSON === undefined) {
      return '' +
          'goog.loadModule(function(exports) {' +
          '"use strict";' + contents +
          '\n' +  // terminate any trailing single line comment.
          ';return exports' +
          '});' +
          '\n//# sourceURL=' + this.path + '\n';
    } else {
      return '' +
          'goog.loadModule(' +
          goog.global.JSON.stringify(
              contents + '\n//# sourceURL=' + this.path + '\n') +
          ');';
    }
  };


  /**
   * Whether the browser is IE9 or earlier, which needs special handling
   * for deferred modules.
   * @const @private {boolean}
   */
  goog.DebugLoader_.IS_OLD_IE_ = !!(
      !goog.global.atob && goog.global.document && goog.global.document['all']);


  /**
   * @param {string} relPath
   * @param {!Array<string>|undefined} provides
   * @param {!Array<string>} requires
   * @param {boolean|!Object<string>=} opt_loadFlags
   * @see goog.addDependency
   */
  goog.DebugLoader_.prototype.addDependency = function(
      relPath, provides, requires, opt_loadFlags) {
    provides = provides || [];
    relPath = relPath.replace(/\\/g, '/');
    var path = goog.normalizePath_(goog.basePath + relPath);
    if (!opt_loadFlags || typeof opt_loadFlags === 'boolean') {
      opt_loadFlags = opt_loadFlags ? {'module': goog.ModuleType.GOOG} : {};
    }
    var dep = this.factory_.createDependency(
        path, relPath, provides, requires, opt_loadFlags,
        goog.transpiler_.needsTranspile(
            opt_loadFlags['lang'] || 'es3', opt_loadFlags['module']));
    this.dependencies_[path] = dep;
    for (var i = 0; i < provides.length; i++) {
      this.idToPath_[provides[i]] = path;
    }
    this.idToPath_[relPath] = path;
  };


  /**
   * Creates goog.Dependency instances for the debug loader to load.
   *
   * Should be overridden to have the debug loader use custom subclasses of
   * goog.Dependency.
   *
   * @param {!goog.Transpiler} transpiler
   * @struct @constructor
   */
  goog.DependencyFactory = function(transpiler) {
    /** @protected @const */
    this.transpiler = transpiler;
  };


  /**
   * @param {string} path Absolute path of the file.
   * @param {string} relativePath Path relative to closure’s base.js.
   * @param {!Array<string>} provides Array of provided goog.provide/module ids.
   * @param {!Array<string>} requires Array of required goog.provide/module /
   *     relative ES6 module paths.
   * @param {!Object<string, string>} loadFlags
   * @param {boolean} needsTranspile True if the file needs to be transpiled
   *     per the goog.Transpiler.
   * @return {!goog.Dependency}
   */
  goog.DependencyFactory.prototype.createDependency = function(
      path, relativePath, provides, requires, loadFlags, needsTranspile) {

    if (loadFlags['module'] == goog.ModuleType.GOOG) {
      return new goog.GoogModuleDependency(
          path, relativePath, provides, requires, loadFlags, needsTranspile,
          this.transpiler);
    } else if (needsTranspile) {
      return new goog.TranspiledDependency(
          path, relativePath, provides, requires, loadFlags, this.transpiler);
    } else {
      if (loadFlags['module'] == goog.ModuleType.ES6) {
        if (goog.TRANSPILE == 'never' && goog.ASSUME_ES_MODULES_TRANSPILED) {
          return new goog.PreTranspiledEs6ModuleDependency(
              path, relativePath, provides, requires, loadFlags);
        } else {
          return new goog.Es6ModuleDependency(
              path, relativePath, provides, requires, loadFlags);
        }
      } else {
        return new goog.Dependency(
            path, relativePath, provides, requires, loadFlags);
      }
    }
  };


  /** @private @const */
  goog.debugLoader_ = new goog.DebugLoader_();


  /**
   * Loads the Closure Dependency file.
   *
   * Exposed a public function so CLOSURE_NO_DEPS can be set to false, base
   * loaded, setDependencyFactory called, and then this called. i.e. allows
   * custom loading of the deps file.
   */
  goog.loadClosureDeps = function() {
    goog.debugLoader_.loadClosureDeps();
  };


  /**
   * Sets the dependency factory, which can be used to create custom
   * goog.Dependency implementations to control how dependencies are loaded.
   *
   * Note: if you wish to call this function and provide your own implemnetation
   * it is a wise idea to set CLOSURE_NO_DEPS to true, otherwise the dependency
   * file and all of its goog.addDependency calls will use the default factory.
   * You can call goog.loadClosureDeps to load the Closure dependency file
   * later, after your factory is injected.
   *
   * @param {!goog.DependencyFactory} factory
   */
  goog.setDependencyFactory = function(factory) {
    goog.debugLoader_.setDependencyFactory(factory);
  };


  if (!goog.global.CLOSURE_NO_DEPS) {
    goog.debugLoader_.loadClosureDeps();
  }


  /**
   * Bootstraps the given namespaces and calls the callback once they are
   * available either via goog.require. This is a replacement for using
   * `goog.require` to bootstrap Closure JavaScript. Previously a `goog.require`
   * in an HTML file would guarantee that the require'd namespace was available
   * in the next immediate script tag. With ES6 modules this no longer a
   * guarantee.
   *
   * @param {!Array<string>} namespaces
   * @param {function(): ?} callback Function to call once all the namespaces
   *     have loaded. Always called asynchronously.
   */
  goog.bootstrap = function(namespaces, callback) {
    goog.debugLoader_.bootstrap(namespaces, callback);
  };
}


/**
 * @define {string} Trusted Types policy name. If non-empty then Closure will
 * use Trusted Types.
 */
goog.TRUSTED_TYPES_POLICY_NAME =
    goog.define('goog.TRUSTED_TYPES_POLICY_NAME', '');


/**
 * Returns the parameter.
 * @param {string} s
 * @return {string}
 * @private
 */
goog.identity_ = function(s) {
  return s;
};


/**
 * Creates Trusted Types policy if Trusted Types are supported by the browser.
 * The policy just blesses any string as a Trusted Type. It is not visibility
 * restricted because anyone can also call TrustedTypes.createPolicy directly.
 * However, the allowed names should be restricted by a HTTP header and the
 * reference to the created policy should be visibility restricted.
 * @param {string} name
 * @return {?TrustedTypePolicy}
 */
goog.createTrustedTypesPolicy = function(name) {
  var policy = null;
  // TODO(koto): Remove window.TrustedTypes variant when the newer API ships.
  var policyFactory = goog.global.trustedTypes || goog.global.TrustedTypes;
  if (!policyFactory || !policyFactory.createPolicy) {
    return policy;
  }
  // TrustedTypes.createPolicy throws if called with a name that is already
  // registered, even in report-only mode. Until the API changes, catch the
  // error not to break the applications functionally. In such case, the code
  // will fall back to using regular Safe Types.
  // TODO(koto): Remove catching once createPolicy API stops throwing.
  try {
    policy = policyFactory.createPolicy(name, {
      createHTML: goog.identity_,
      createScript: goog.identity_,
      createScriptURL: goog.identity_,
      createURL: goog.identity_
    });
  } catch (e) {
    goog.logToConsole_(e.message);
  }
  return policy;
};


/** @private @const {?TrustedTypePolicy} */
goog.TRUSTED_TYPES_POLICY_ = goog.TRUSTED_TYPES_POLICY_NAME ?
    goog.createTrustedTypesPolicy(goog.TRUSTED_TYPES_POLICY_NAME + '#base') :
    null;

// Copyright 2019 The Closure Library Authors. All Rights Reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
//      http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS-IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//
// This file has been auto-generated by GenJsDeps, please do not edit.

// Disable Clang formatter for this file.
// See http://goo.gl/SdiwZH
// clang-format off

goog.addDependency('collections/sets.js', ['goog.collections.sets'], ['goog.labs.collections.iterables'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('collections/sets_test.js', ['goog.collections.setsTest'], ['goog.collections.sets', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('../../third_party/closure/goog/mochikit/async/deferred.js', ['goog.async.Deferred', 'goog.async.Deferred.AlreadyCalledError', 'goog.async.Deferred.CanceledError'], ['goog.Promise', 'goog.Thenable', 'goog.array', 'goog.asserts', 'goog.debug.Error'], {});
goog.addDependency('../../third_party/closure/goog/mochikit/async/deferred_async_test.js', ['goog.async.deferredAsyncTest'], ['goog.async.Deferred', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('../../third_party/closure/goog/mochikit/async/deferred_test.js', ['goog.async.deferredTest'], ['goog.Promise', 'goog.Thenable', 'goog.async.Deferred', 'goog.testing.MockClock', 'goog.testing.PropertyReplacer', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('../../third_party/closure/goog/mochikit/async/deferredlist.js', ['goog.async.DeferredList'], ['goog.async.Deferred'], {});
goog.addDependency('../../third_party/closure/goog/mochikit/async/deferredlist_test.js', ['goog.async.deferredListTest'], ['goog.array', 'goog.async.Deferred', 'goog.async.DeferredList', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('proto/proto.js', ['goog.proto'], ['goog.proto.Serializer'], {});
goog.addDependency('proto/serializer.js', ['goog.proto.Serializer'], ['goog.json.Serializer', 'goog.string'], {});
goog.addDependency('proto/serializer_test.js', ['goog.protoTest'], ['goog.proto', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('a11y/aria/announcer.js', ['goog.a11y.aria.Announcer'], ['goog.Disposable', 'goog.Timer', 'goog.a11y.aria', 'goog.a11y.aria.LivePriority', 'goog.a11y.aria.State', 'goog.dom', 'goog.dom.TagName', 'goog.object'], {});
goog.addDependency('a11y/aria/announcer_test.js', ['goog.a11y.aria.AnnouncerTest'], ['goog.a11y.aria', 'goog.a11y.aria.Announcer', 'goog.a11y.aria.LivePriority', 'goog.a11y.aria.State', 'goog.array', 'goog.dom', 'goog.dom.TagName', 'goog.dom.iframe', 'goog.testing.MockClock', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('a11y/aria/aria.js', ['goog.a11y.aria'], ['goog.a11y.aria.Role', 'goog.a11y.aria.State', 'goog.a11y.aria.datatables', 'goog.array', 'goog.asserts', 'goog.dom', 'goog.dom.TagName', 'goog.object', 'goog.string'], {});
goog.addDependency('a11y/aria/aria_test.js', ['goog.a11y.ariaTest'], ['goog.a11y.aria', 'goog.a11y.aria.Role', 'goog.a11y.aria.State', 'goog.dom', 'goog.dom.TagName', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('a11y/aria/attributes.js', ['goog.a11y.aria.AutoCompleteValues', 'goog.a11y.aria.CheckedValues', 'goog.a11y.aria.DropEffectValues', 'goog.a11y.aria.ExpandedValues', 'goog.a11y.aria.GrabbedValues', 'goog.a11y.aria.InvalidValues', 'goog.a11y.aria.LivePriority', 'goog.a11y.aria.OrientationValues', 'goog.a11y.aria.PressedValues', 'goog.a11y.aria.RelevantValues', 'goog.a11y.aria.SelectedValues', 'goog.a11y.aria.SortValues', 'goog.a11y.aria.State'], [], {});
goog.addDependency('a11y/aria/datatables.js', ['goog.a11y.aria.datatables'], ['goog.a11y.aria.State', 'goog.object'], {});
goog.addDependency('a11y/aria/roles.js', ['goog.a11y.aria.Role'], [], {});
goog.addDependency('array/array.js', ['goog.array'], ['goog.asserts'], {});
goog.addDependency('array/array_test.js', ['goog.arrayTest'], ['goog.array', 'goog.dom', 'goog.dom.TagName', 'goog.testing.PropertyReplacer', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es7', 'module': 'goog'});
goog.addDependency('asserts/asserts.js', ['goog.asserts', 'goog.asserts.AssertionError'], ['goog.debug.Error', 'goog.dom.NodeType'], {});
goog.addDependency('asserts/asserts_test.js', ['goog.assertsTest'], ['goog.asserts', 'goog.asserts.AssertionError', 'goog.dom', 'goog.dom.TagName', 'goog.reflect', 'goog.string', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('async/animationdelay.js', ['goog.async.AnimationDelay'], ['goog.Disposable', 'goog.events', 'goog.functions'], {});
goog.addDependency('async/animationdelay_test.js', ['goog.async.AnimationDelayTest'], ['goog.Promise', 'goog.Timer', 'goog.async.AnimationDelay', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('async/conditionaldelay.js', ['goog.async.ConditionalDelay'], ['goog.Disposable', 'goog.async.Delay'], {});
goog.addDependency('async/conditionaldelay_test.js', ['goog.async.ConditionalDelayTest'], ['goog.async.ConditionalDelay', 'goog.testing.MockClock', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('async/debouncer.js', ['goog.async.Debouncer'], ['goog.Disposable', 'goog.Timer'], {});
goog.addDependency('async/debouncer_test.js', ['goog.async.DebouncerTest'], ['goog.array', 'goog.async.Debouncer', 'goog.testing.MockClock', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('async/delay.js', ['goog.Delay', 'goog.async.Delay'], ['goog.Disposable', 'goog.Timer'], {});
goog.addDependency('async/delay_test.js', ['goog.async.DelayTest'], ['goog.async.Delay', 'goog.testing.MockClock', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('async/freelist.js', ['goog.async.FreeList'], [], {'lang': 'es6'});
goog.addDependency('async/freelist_test.js', ['goog.async.FreeListTest'], ['goog.async.FreeList', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('async/nexttick.js', ['goog.async.nextTick', 'goog.async.throwException'], ['goog.debug.entryPointRegistry', 'goog.dom', 'goog.dom.TagName', 'goog.dom.safe', 'goog.functions', 'goog.html.SafeHtml', 'goog.html.TrustedResourceUrl', 'goog.labs.userAgent.browser', 'goog.labs.userAgent.engine', 'goog.string.Const'], {});
goog.addDependency('async/nexttick_test.js', ['goog.async.nextTickTest'], ['goog.Promise', 'goog.async.nextTick', 'goog.debug.ErrorHandler', 'goog.debug.entryPointRegistry', 'goog.dom', 'goog.dom.TagName', 'goog.labs.userAgent.browser', 'goog.testing.MockClock', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('async/run.js', ['goog.async.run'], ['goog.async.WorkQueue', 'goog.async.nextTick', 'goog.async.throwException'], {});
goog.addDependency('async/run_next_tick_test.js', ['goog.async.runNextTickTest'], ['goog.async.run', 'goog.testing.MockClock', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('async/run_test.js', ['goog.async.runTest'], ['goog.async.run', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('async/throttle.js', ['goog.Throttle', 'goog.async.Throttle'], ['goog.Disposable', 'goog.Timer'], {});
goog.addDependency('async/throttle_test.js', ['goog.async.ThrottleTest'], ['goog.async.Throttle', 'goog.testing.MockClock', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('async/workqueue.js', ['goog.async.WorkItem', 'goog.async.WorkQueue'], ['goog.asserts', 'goog.async.FreeList'], {});
goog.addDependency('async/workqueue_test.js', ['goog.async.WorkQueueTest'], ['goog.async.WorkQueue', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('base.js', ['goog'], [], {});
goog.addDependency('base_module_test.js', ['goog.baseModuleTest'], ['goog.Timer', 'goog.test_module', 'goog.testing.PropertyReplacer', 'goog.testing.jsunit', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('base_test.js', ['goog.baseTest'], ['goog.Promise', 'goog.Timer', 'goog.Uri', 'goog.dom', 'goog.dom.TagName', 'goog.object', 'goog.test_module', 'goog.testing.PropertyReplacer', 'goog.testing.jsunit', 'goog.testing.recordFunction', 'goog.userAgent'], {'lang': 'es6'});
goog.addDependency('color/alpha.js', ['goog.color.alpha'], ['goog.color'], {});
goog.addDependency('color/alpha_test.js', ['goog.color.alphaTest'], ['goog.array', 'goog.color.alpha', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('color/color.js', ['goog.color', 'goog.color.Hsl', 'goog.color.Hsv', 'goog.color.Rgb'], ['goog.color.names', 'goog.math'], {});
goog.addDependency('color/color_test.js', ['goog.colorTest'], ['goog.array', 'goog.color', 'goog.color.names', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('color/names.js', ['goog.color.names'], [], {});
goog.addDependency('crypt/aes.js', ['goog.crypt.Aes'], ['goog.asserts', 'goog.crypt.BlockCipher'], {});
goog.addDependency('crypt/aes_test.js', ['goog.crypt.AesTest'], ['goog.crypt', 'goog.crypt.Aes', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('crypt/arc4.js', ['goog.crypt.Arc4'], ['goog.asserts'], {});
goog.addDependency('crypt/arc4_test.js', ['goog.crypt.Arc4Test'], ['goog.array', 'goog.crypt.Arc4', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('crypt/base64.js', ['goog.crypt.base64'], ['goog.asserts', 'goog.crypt', 'goog.string', 'goog.userAgent', 'goog.userAgent.product'], {'lang': 'es5'});
goog.addDependency('crypt/base64_test.js', ['goog.crypt.base64Test'], ['goog.crypt', 'goog.crypt.base64', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('crypt/basen.js', ['goog.crypt.baseN'], [], {'lang': 'es6'});
goog.addDependency('crypt/basen_test.js', ['goog.crypt.baseNTest'], ['goog.crypt.baseN', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('crypt/blobhasher.js', ['goog.crypt.BlobHasher', 'goog.crypt.BlobHasher.EventType'], ['goog.asserts', 'goog.events.EventTarget', 'goog.fs', 'goog.log'], {});
goog.addDependency('crypt/blobhasher_test.js', ['goog.crypt.BlobHasherTest'], ['goog.crypt', 'goog.crypt.BlobHasher', 'goog.crypt.Md5', 'goog.events', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('crypt/blockcipher.js', ['goog.crypt.BlockCipher'], [], {});
goog.addDependency('crypt/bytestring_perf.js', ['goog.crypt.byteArrayToStringPerf'], ['goog.array', 'goog.dom', 'goog.testing.PerformanceTable'], {});
goog.addDependency('crypt/cbc.js', ['goog.crypt.Cbc'], ['goog.array', 'goog.asserts', 'goog.crypt', 'goog.crypt.BlockCipher'], {});
goog.addDependency('crypt/cbc_test.js', ['goog.crypt.CbcTest'], ['goog.crypt', 'goog.crypt.Aes', 'goog.crypt.Cbc', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('crypt/crypt.js', ['goog.crypt'], ['goog.array', 'goog.asserts'], {});
goog.addDependency('crypt/crypt_test.js', ['goog.cryptTest'], ['goog.crypt', 'goog.string', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('crypt/ctr.js', ['goog.crypt.Ctr'], ['goog.array', 'goog.asserts', 'goog.crypt'], {});
goog.addDependency('crypt/ctr_test.js', ['goog.crypt.CtrTest'], ['goog.crypt', 'goog.crypt.Aes', 'goog.crypt.Ctr', 'goog.testing.jsunit'], {'lang': 'es6'});
goog.addDependency('crypt/hash.js', ['goog.crypt.Hash'], [], {});
goog.addDependency('crypt/hash32.js', ['goog.crypt.hash32'], ['goog.crypt'], {});
goog.addDependency('crypt/hash32_test.js', ['goog.crypt.hash32Test'], ['goog.crypt.hash32', 'goog.testing.TestCase', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('crypt/hashtester.js', ['goog.crypt.hashTester'], ['goog.array', 'goog.crypt', 'goog.dom', 'goog.dom.TagName', 'goog.reflect', 'goog.testing.PerformanceTable', 'goog.testing.PseudoRandom', 'goog.testing.asserts'], {});
goog.addDependency('crypt/hmac.js', ['goog.crypt.Hmac'], ['goog.crypt.Hash'], {});
goog.addDependency('crypt/hmac_test.js', ['goog.crypt.HmacTest'], ['goog.crypt.Hmac', 'goog.crypt.Sha1', 'goog.crypt.hashTester', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('crypt/md5.js', ['goog.crypt.Md5'], ['goog.crypt.Hash'], {});
goog.addDependency('crypt/md5_test.js', ['goog.crypt.Md5Test'], ['goog.crypt', 'goog.crypt.Md5', 'goog.crypt.hashTester', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('crypt/pbkdf2.js', ['goog.crypt.pbkdf2'], ['goog.array', 'goog.asserts', 'goog.crypt', 'goog.crypt.Hmac', 'goog.crypt.Sha1'], {});
goog.addDependency('crypt/pbkdf2_test.js', ['goog.crypt.pbkdf2Test'], ['goog.crypt', 'goog.crypt.pbkdf2', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('crypt/sha1.js', ['goog.crypt.Sha1'], ['goog.crypt.Hash'], {});
goog.addDependency('crypt/sha1_test.js', ['goog.crypt.Sha1Test'], ['goog.crypt', 'goog.crypt.Sha1', 'goog.crypt.hashTester', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('crypt/sha2.js', ['goog.crypt.Sha2'], ['goog.array', 'goog.asserts', 'goog.crypt.Hash'], {});
goog.addDependency('crypt/sha224.js', ['goog.crypt.Sha224'], ['goog.crypt.Sha2'], {});
goog.addDependency('crypt/sha224_test.js', ['goog.crypt.Sha224Test'], ['goog.crypt', 'goog.crypt.Sha224', 'goog.crypt.hashTester', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('crypt/sha256.js', ['goog.crypt.Sha256'], ['goog.crypt.Sha2'], {});
goog.addDependency('crypt/sha256_test.js', ['goog.crypt.Sha256Test'], ['goog.crypt', 'goog.crypt.Sha256', 'goog.crypt.hashTester', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('crypt/sha2_64bit.js', ['goog.crypt.Sha2_64bit'], ['goog.array', 'goog.asserts', 'goog.crypt.Hash', 'goog.math.Long'], {});
goog.addDependency('crypt/sha2_64bit_test.js', ['goog.crypt.Sha2_64bit_test'], ['goog.array', 'goog.crypt', 'goog.crypt.Sha384', 'goog.crypt.Sha512', 'goog.crypt.Sha512_256', 'goog.crypt.hashTester', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('crypt/sha384.js', ['goog.crypt.Sha384'], ['goog.crypt.Sha2_64bit'], {});
goog.addDependency('crypt/sha512.js', ['goog.crypt.Sha512'], ['goog.crypt.Sha2_64bit'], {});
goog.addDependency('crypt/sha512_256.js', ['goog.crypt.Sha512_256'], ['goog.crypt.Sha2_64bit'], {});
goog.addDependency('cssom/cssom.js', ['goog.cssom', 'goog.cssom.CssRuleType'], ['goog.array', 'goog.dom', 'goog.dom.TagName'], {});
goog.addDependency('cssom/cssom_test.js', ['goog.cssomTest'], ['goog.array', 'goog.cssom', 'goog.cssom.CssRuleType', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('cssom/iframe/style.js', ['goog.cssom.iframe.style'], ['goog.asserts', 'goog.cssom', 'goog.dom', 'goog.dom.NodeType', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.string', 'goog.style', 'goog.userAgent'], {});
goog.addDependency('cssom/iframe/style_test.js', ['goog.cssom.iframe.styleTest'], ['goog.cssom', 'goog.cssom.iframe.style', 'goog.dom', 'goog.dom.DomHelper', 'goog.dom.TagName', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('datasource/datamanager.js', ['goog.ds.DataManager'], ['goog.ds.BasicNodeList', 'goog.ds.DataNode', 'goog.ds.Expr', 'goog.object', 'goog.string', 'goog.structs', 'goog.structs.Map'], {});
goog.addDependency('datasource/datasource.js', ['goog.ds.BaseDataNode', 'goog.ds.BasicNodeList', 'goog.ds.DataNode', 'goog.ds.DataNodeList', 'goog.ds.EmptyNodeList', 'goog.ds.LoadState', 'goog.ds.SortedNodeList', 'goog.ds.Util', 'goog.ds.logger'], ['goog.array', 'goog.log'], {});
goog.addDependency('datasource/datasource_test.js', ['goog.ds.JsDataSourceTest'], ['goog.dom.xml', 'goog.ds.DataManager', 'goog.ds.JsDataSource', 'goog.ds.SortedNodeList', 'goog.ds.XmlDataSource', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('datasource/expr.js', ['goog.ds.Expr'], ['goog.ds.BasicNodeList', 'goog.ds.EmptyNodeList', 'goog.string'], {});
goog.addDependency('datasource/expr_test.js', ['goog.ds.ExprTest'], ['goog.ds.DataManager', 'goog.ds.Expr', 'goog.ds.JsDataSource', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('datasource/fastdatanode.js', ['goog.ds.AbstractFastDataNode', 'goog.ds.FastDataNode', 'goog.ds.FastListNode', 'goog.ds.PrimitiveFastDataNode'], ['goog.ds.DataManager', 'goog.ds.DataNodeList', 'goog.ds.EmptyNodeList', 'goog.string'], {});
goog.addDependency('datasource/fastdatanode_test.js', ['goog.ds.FastDataNodeTest'], ['goog.array', 'goog.ds.DataManager', 'goog.ds.Expr', 'goog.ds.FastDataNode', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('datasource/jsdatasource.js', ['goog.ds.JsDataSource', 'goog.ds.JsPropertyDataSource'], ['goog.ds.BaseDataNode', 'goog.ds.BasicNodeList', 'goog.ds.DataManager', 'goog.ds.DataNode', 'goog.ds.EmptyNodeList', 'goog.ds.LoadState'], {});
goog.addDependency('datasource/jsondatasource.js', ['goog.ds.JsonDataSource'], ['goog.dom', 'goog.dom.TagName', 'goog.dom.safe', 'goog.ds.DataManager', 'goog.ds.JsDataSource', 'goog.ds.LoadState', 'goog.ds.logger', 'goog.log'], {});
goog.addDependency('datasource/jsxmlhttpdatasource.js', ['goog.ds.JsXmlHttpDataSource'], ['goog.Uri', 'goog.ds.DataManager', 'goog.ds.FastDataNode', 'goog.ds.LoadState', 'goog.ds.logger', 'goog.events', 'goog.log', 'goog.net.EventType', 'goog.net.XhrIo'], {});
goog.addDependency('datasource/jsxmlhttpdatasource_test.js', ['goog.ds.JsXmlHttpDataSourceTest'], ['goog.ds.JsXmlHttpDataSource', 'goog.testing.TestQueue', 'goog.testing.net.XhrIo', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('datasource/xmldatasource.js', ['goog.ds.XmlDataSource', 'goog.ds.XmlHttpDataSource'], ['goog.Uri', 'goog.dom.NodeType', 'goog.dom.xml', 'goog.ds.BasicNodeList', 'goog.ds.DataManager', 'goog.ds.DataNode', 'goog.ds.LoadState', 'goog.ds.logger', 'goog.log', 'goog.net.XhrIo', 'goog.string'], {});
goog.addDependency('date/date.js', ['goog.date', 'goog.date.Date', 'goog.date.DateTime', 'goog.date.Interval', 'goog.date.month', 'goog.date.weekDay'], ['goog.asserts', 'goog.date.DateLike', 'goog.i18n.DateTimeSymbols', 'goog.string'], {});
goog.addDependency('date/date_test.js', ['goog.dateTest'], ['goog.array', 'goog.date', 'goog.date.Date', 'goog.date.DateTime', 'goog.date.Interval', 'goog.date.month', 'goog.date.weekDay', 'goog.i18n.DateTimeSymbols', 'goog.testing.ExpectedFailures', 'goog.testing.testSuite', 'goog.userAgent', 'goog.userAgent.platform', 'goog.userAgent.product', 'goog.userAgent.product.isVersion'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('date/datelike.js', ['goog.date.DateLike'], [], {});
goog.addDependency('date/daterange.js', ['goog.date.DateRange', 'goog.date.DateRange.Iterator', 'goog.date.DateRange.StandardDateRangeKeys'], ['goog.date.Date', 'goog.date.Interval', 'goog.iter.Iterator', 'goog.iter.StopIteration'], {});
goog.addDependency('date/daterange_test.js', ['goog.date.DateRangeTest'], ['goog.date.Date', 'goog.date.DateRange', 'goog.date.Interval', 'goog.i18n.DateTimeSymbols', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('date/duration.js', ['goog.date.duration'], ['goog.i18n.DateTimeFormat', 'goog.i18n.MessageFormat'], {});
goog.addDependency('date/duration_test.js', ['goog.date.durationTest'], ['goog.date.duration', 'goog.i18n.DateTimeFormat', 'goog.i18n.DateTimeSymbols', 'goog.i18n.DateTimeSymbols_bn', 'goog.i18n.DateTimeSymbols_en', 'goog.i18n.DateTimeSymbols_fa', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('date/relative.js', ['goog.date.relative', 'goog.date.relative.TimeDeltaFormatter', 'goog.date.relative.Unit'], ['goog.i18n.DateTimeFormat', 'goog.i18n.DateTimePatterns', 'goog.i18n.RelativeDateTimeFormat'], {});
goog.addDependency('date/relative_test.js', ['goog.date.relativeTest'], ['goog.date.relativeCommonTests'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('date/relativecommontests.js', ['goog.date.relativeCommonTests'], ['goog.date.DateTime', 'goog.date.relative', 'goog.i18n.DateTimeFormat', 'goog.i18n.DateTimePatterns_ar', 'goog.i18n.DateTimePatterns_bn', 'goog.i18n.DateTimePatterns_es', 'goog.i18n.DateTimePatterns_fa', 'goog.i18n.DateTimePatterns_fr', 'goog.i18n.DateTimePatterns_no', 'goog.i18n.DateTimeSymbols_ar', 'goog.i18n.DateTimeSymbols_bn', 'goog.i18n.DateTimeSymbols_es', 'goog.i18n.DateTimeSymbols_fa', 'goog.i18n.DateTimeSymbols_fr', 'goog.i18n.DateTimeSymbols_no', 'goog.i18n.NumberFormatSymbols_bn', 'goog.i18n.NumberFormatSymbols_en', 'goog.i18n.NumberFormatSymbols_fa', 'goog.i18n.NumberFormatSymbols_no', 'goog.i18n.relativeDateTimeSymbols', 'goog.testing.PropertyReplacer', 'goog.testing.jsunit'], {'lang': 'es6'});
goog.addDependency('date/utcdatetime.js', ['goog.date.UtcDateTime'], ['goog.date', 'goog.date.Date', 'goog.date.DateTime', 'goog.date.Interval'], {});
goog.addDependency('date/utcdatetime_test.js', ['goog.date.UtcDateTimeTest'], ['goog.date.Interval', 'goog.date.UtcDateTime', 'goog.date.month', 'goog.date.weekDay', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('db/cursor.js', ['goog.db.Cursor'], ['goog.async.Deferred', 'goog.db.Error', 'goog.db.KeyRange', 'goog.debug', 'goog.events.EventTarget'], {});
goog.addDependency('db/db.js', ['goog.db', 'goog.db.BlockedCallback', 'goog.db.UpgradeNeededCallback'], ['goog.asserts', 'goog.async.Deferred', 'goog.db.Error', 'goog.db.IndexedDb', 'goog.db.Transaction'], {});
goog.addDependency('db/db_test.js', ['goog.dbTest'], ['goog.Promise', 'goog.array', 'goog.db', 'goog.db.Cursor', 'goog.db.Error', 'goog.db.IndexedDb', 'goog.db.KeyRange', 'goog.db.Transaction', 'goog.events', 'goog.testing.PropertyReplacer', 'goog.testing.TestCase', 'goog.testing.asserts', 'goog.testing.testSuite', 'goog.userAgent.product'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('db/error.js', ['goog.db.DomErrorLike', 'goog.db.Error', 'goog.db.Error.ErrorCode', 'goog.db.Error.ErrorName', 'goog.db.Error.VersionChangeBlockedError'], ['goog.asserts', 'goog.debug.Error'], {});
goog.addDependency('db/index.js', ['goog.db.Index'], ['goog.async.Deferred', 'goog.db.Cursor', 'goog.db.Error', 'goog.db.KeyRange', 'goog.debug'], {});
goog.addDependency('db/indexeddb.js', ['goog.db.IndexedDb'], ['goog.db.Error', 'goog.db.ObjectStore', 'goog.db.Transaction', 'goog.events.Event', 'goog.events.EventHandler', 'goog.events.EventTarget'], {'lang': 'es6'});
goog.addDependency('db/keyrange.js', ['goog.db.KeyRange'], [], {});
goog.addDependency('db/objectstore.js', ['goog.db.ObjectStore'], ['goog.async.Deferred', 'goog.db.Cursor', 'goog.db.Error', 'goog.db.Index', 'goog.db.KeyRange', 'goog.debug'], {});
goog.addDependency('db/transaction.js', ['goog.db.Transaction', 'goog.db.Transaction.TransactionMode'], ['goog.async.Deferred', 'goog.db.Error', 'goog.db.ObjectStore', 'goog.events', 'goog.events.EventHandler', 'goog.events.EventTarget'], {});
goog.addDependency('debug/console.js', ['goog.debug.Console'], ['goog.debug.LogManager', 'goog.debug.Logger', 'goog.debug.TextFormatter'], {});
goog.addDependency('debug/console_test.js', ['goog.debug.ConsoleTest'], ['goog.debug.Console', 'goog.debug.LogRecord', 'goog.debug.Logger', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('debug/debug.js', ['goog.debug'], ['goog.array', 'goog.debug.errorcontext', 'goog.userAgent'], {});
goog.addDependency('debug/debug_test.js', ['goog.debugTest'], ['goog.debug', 'goog.debug.errorcontext', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('debug/debugwindow.js', ['goog.debug.DebugWindow'], ['goog.debug.HtmlFormatter', 'goog.debug.LogManager', 'goog.debug.Logger', 'goog.dom.safe', 'goog.html.SafeHtml', 'goog.html.SafeStyleSheet', 'goog.string.Const', 'goog.structs.CircularBuffer', 'goog.userAgent'], {});
goog.addDependency('debug/debugwindow_test.js', ['goog.debug.DebugWindowTest'], ['goog.debug.DebugWindow', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('debug/devcss/devcss.js', ['goog.debug.DevCss', 'goog.debug.DevCss.UserAgent'], ['goog.asserts', 'goog.cssom', 'goog.dom.classlist', 'goog.events', 'goog.events.EventType', 'goog.string', 'goog.userAgent'], {});
goog.addDependency('debug/devcss/devcss_test.js', ['goog.debug.DevCssTest'], ['goog.debug.DevCss', 'goog.style', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('debug/devcss/devcssrunner.js', ['goog.debug.devCssRunner'], ['goog.debug.DevCss'], {});
goog.addDependency('debug/divconsole.js', ['goog.debug.DivConsole'], ['goog.debug.HtmlFormatter', 'goog.debug.LogManager', 'goog.dom.DomHelper', 'goog.dom.TagName', 'goog.dom.safe', 'goog.html.SafeHtml', 'goog.html.SafeStyleSheet', 'goog.string.Const', 'goog.style'], {});
goog.addDependency('debug/enhanceerror_test.js', ['goog.debugEnhanceErrorTest'], ['goog.debug', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('debug/entrypointregistry.js', ['goog.debug.EntryPointMonitor', 'goog.debug.entryPointRegistry'], ['goog.asserts'], {});
goog.addDependency('debug/entrypointregistry_test.js', ['goog.debug.entryPointRegistryTest'], ['goog.debug.ErrorHandler', 'goog.debug.entryPointRegistry', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('debug/error.js', ['goog.debug.Error'], [], {'lang': 'es6'});
goog.addDependency('debug/error_test.js', ['goog.debug.ErrorTest'], ['goog.debug.Error', 'goog.testing.ExpectedFailures', 'goog.testing.testSuite', 'goog.userAgent', 'goog.userAgent.product'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('debug/errorcontext.js', ['goog.debug.errorcontext'], [], {});
goog.addDependency('debug/errorcontext_test.js', ['goog.debug.errorcontextTest'], ['goog.debug.errorcontext', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('debug/errorhandler.js', ['goog.debug.ErrorHandler', 'goog.debug.ErrorHandler.ProtectedFunctionError'], ['goog.Disposable', 'goog.asserts', 'goog.debug', 'goog.debug.EntryPointMonitor', 'goog.debug.Error', 'goog.debug.Trace'], {'lang': 'es6'});
goog.addDependency('debug/errorhandler_async_test.js', ['goog.debug.ErrorHandlerAsyncTest'], ['goog.Promise', 'goog.debug.ErrorHandler', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es8', 'module': 'goog'});
goog.addDependency('debug/errorhandler_test.js', ['goog.debug.ErrorHandlerTest'], ['goog.debug.ErrorHandler', 'goog.testing.MockControl', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('debug/errorhandlerweakdep.js', ['goog.debug.errorHandlerWeakDep'], [], {});
goog.addDependency('debug/errorreporter.js', ['goog.debug.ErrorReporter', 'goog.debug.ErrorReporter.ExceptionEvent'], ['goog.asserts', 'goog.debug', 'goog.debug.Error', 'goog.debug.ErrorHandler', 'goog.debug.entryPointRegistry', 'goog.debug.errorcontext', 'goog.events', 'goog.events.Event', 'goog.events.EventTarget', 'goog.log', 'goog.net.XhrIo', 'goog.object', 'goog.string', 'goog.uri.utils', 'goog.userAgent'], {});
goog.addDependency('debug/errorreporter_test.js', ['goog.debug.ErrorReporterTest'], ['goog.debug.Error', 'goog.debug.ErrorReporter', 'goog.debug.errorcontext', 'goog.events', 'goog.functions', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite', 'goog.userAgent', 'goog.userAgent.product'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('debug/fancywindow.js', ['goog.debug.FancyWindow'], ['goog.array', 'goog.asserts', 'goog.debug.DebugWindow', 'goog.debug.LogManager', 'goog.debug.Logger', 'goog.dom.DomHelper', 'goog.dom.TagName', 'goog.dom.safe', 'goog.html.SafeHtml', 'goog.html.SafeStyleSheet', 'goog.object', 'goog.string', 'goog.string.Const', 'goog.userAgent'], {});
goog.addDependency('debug/formatter.js', ['goog.debug.Formatter', 'goog.debug.HtmlFormatter', 'goog.debug.TextFormatter'], ['goog.debug', 'goog.debug.Logger', 'goog.debug.RelativeTimeProvider', 'goog.html.SafeHtml', 'goog.html.SafeUrl', 'goog.html.uncheckedconversions', 'goog.string.Const'], {});
goog.addDependency('debug/formatter_test.js', ['goog.debug.FormatterTest'], ['goog.debug.HtmlFormatter', 'goog.debug.LogRecord', 'goog.debug.Logger', 'goog.html.SafeHtml', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('debug/fpsdisplay.js', ['goog.debug.FpsDisplay'], ['goog.asserts', 'goog.async.AnimationDelay', 'goog.dom', 'goog.dom.TagName', 'goog.ui.Component'], {});
goog.addDependency('debug/fpsdisplay_test.js', ['goog.debug.FpsDisplayTest'], ['goog.Timer', 'goog.debug.FpsDisplay', 'goog.testing.TestCase', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('debug/logbuffer.js', ['goog.debug.LogBuffer'], ['goog.asserts', 'goog.debug.LogRecord'], {});
goog.addDependency('debug/logbuffer_test.js', ['goog.debug.LogBufferTest'], ['goog.debug.LogBuffer', 'goog.debug.Logger', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('debug/logger.js', ['goog.debug.LogManager', 'goog.debug.Loggable', 'goog.debug.Logger', 'goog.debug.Logger.Level'], ['goog.array', 'goog.asserts', 'goog.debug', 'goog.debug.LogBuffer', 'goog.debug.LogRecord'], {});
goog.addDependency('debug/logger_test.js', ['goog.debug.LoggerTest'], ['goog.debug.LogManager', 'goog.debug.Logger', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('debug/logrecord.js', ['goog.debug.LogRecord'], [], {});
goog.addDependency('debug/logrecordserializer.js', ['goog.debug.logRecordSerializer'], ['goog.debug.LogRecord', 'goog.debug.Logger', 'goog.json', 'goog.object'], {});
goog.addDependency('debug/logrecordserializer_test.js', ['goog.debug.logRecordSerializerTest'], ['goog.debug.LogRecord', 'goog.debug.Logger', 'goog.debug.logRecordSerializer', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('debug/relativetimeprovider.js', ['goog.debug.RelativeTimeProvider'], [], {});
goog.addDependency('debug/tracer.js', ['goog.debug.StopTraceDetail', 'goog.debug.Trace'], ['goog.array', 'goog.asserts', 'goog.debug.Logger', 'goog.iter', 'goog.log', 'goog.structs.Map', 'goog.structs.SimplePool'], {});
goog.addDependency('debug/tracer_test.js', ['goog.debug.TraceTest'], ['goog.array', 'goog.debug.Trace', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('defineclass_test.js', ['goog.defineClassTest'], ['goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('delegate/delegateregistry.js', ['goog.delegate.DelegateRegistry'], ['goog.array', 'goog.asserts', 'goog.debug'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('delegate/delegateregistry_test.js', ['goog.delegate.DelegateRegistryTest'], ['goog.array', 'goog.delegate.DelegateRegistry', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('delegate/delegates.js', ['goog.delegate.delegates'], [], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('delegate/delegates_test.js', ['goog.delegate.delegatesTest'], ['goog.delegate.delegates', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('disposable/disposable.js', ['goog.Disposable', 'goog.dispose', 'goog.disposeAll'], ['goog.disposable.IDisposable'], {});
goog.addDependency('disposable/disposable_test.js', ['goog.DisposableTest'], ['goog.Disposable', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('disposable/idisposable.js', ['goog.disposable.IDisposable'], [], {});
goog.addDependency('dom/abstractmultirange.js', ['goog.dom.AbstractMultiRange'], ['goog.array', 'goog.dom', 'goog.dom.AbstractRange', 'goog.dom.TextRange'], {});
goog.addDependency('dom/abstractrange.js', ['goog.dom.AbstractRange', 'goog.dom.RangeIterator', 'goog.dom.RangeType'], ['goog.dom', 'goog.dom.NodeType', 'goog.dom.SavedCaretRange', 'goog.dom.TagIterator', 'goog.userAgent'], {});
goog.addDependency('dom/abstractrange_test.js', ['goog.dom.AbstractRangeTest'], ['goog.dom', 'goog.dom.AbstractRange', 'goog.dom.Range', 'goog.dom.TagName', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/animationframe/animationframe.js', ['goog.dom.animationFrame', 'goog.dom.animationFrame.Spec', 'goog.dom.animationFrame.State'], ['goog.dom.animationFrame.polyfill'], {});
goog.addDependency('dom/animationframe/animationframe_test.js', ['goog.dom.AnimationFrameTest'], ['goog.dom.animationFrame', 'goog.testing.MockClock', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/animationframe/polyfill.js', ['goog.dom.animationFrame.polyfill'], [], {'lang': 'es6'});
goog.addDependency('dom/annotate.js', ['goog.dom.annotate', 'goog.dom.annotate.AnnotateFn'], ['goog.array', 'goog.asserts', 'goog.dom', 'goog.dom.NodeType', 'goog.dom.TagName', 'goog.dom.safe', 'goog.html.SafeHtml', 'goog.object'], {});
goog.addDependency('dom/annotate_test.js', ['goog.dom.annotateTest'], ['goog.dom', 'goog.dom.TagName', 'goog.dom.annotate', 'goog.html.SafeHtml', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/asserts.js', ['goog.dom.asserts'], ['goog.asserts'], {});
goog.addDependency('dom/asserts_test.js', ['goog.dom.assertsTest'], ['goog.dom.asserts', 'goog.testing.PropertyReplacer', 'goog.testing.StrictMock', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/attr.js', ['goog.dom.Attr'], [], {});
goog.addDependency('dom/browserfeature.js', ['goog.dom.BrowserFeature'], ['goog.userAgent'], {});
goog.addDependency('dom/browserfeature_test.js', ['goog.dom.BrowserFeatureTest'], ['goog.dom', 'goog.dom.BrowserFeature', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/browserrange/abstractrange.js', ['goog.dom.browserrange.AbstractRange'], ['goog.array', 'goog.asserts', 'goog.dom', 'goog.dom.NodeType', 'goog.dom.RangeEndpoint', 'goog.dom.TagName', 'goog.dom.TextRangeIterator', 'goog.iter', 'goog.math.Coordinate', 'goog.string', 'goog.string.StringBuffer', 'goog.userAgent'], {});
goog.addDependency('dom/browserrange/browserrange.js', ['goog.dom.browserrange', 'goog.dom.browserrange.Error'], ['goog.dom', 'goog.dom.BrowserFeature', 'goog.dom.NodeType', 'goog.dom.browserrange.GeckoRange', 'goog.dom.browserrange.IeRange', 'goog.dom.browserrange.OperaRange', 'goog.dom.browserrange.W3cRange', 'goog.dom.browserrange.WebKitRange', 'goog.userAgent'], {});
goog.addDependency('dom/browserrange/browserrange_test.js', ['goog.dom.browserrangeTest'], ['goog.dom', 'goog.dom.NodeType', 'goog.dom.Range', 'goog.dom.RangeEndpoint', 'goog.dom.TagName', 'goog.dom.browserrange', 'goog.html.testing', 'goog.testing.dom', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/browserrange/geckorange.js', ['goog.dom.browserrange.GeckoRange'], ['goog.dom.browserrange.W3cRange'], {});
goog.addDependency('dom/browserrange/ierange.js', ['goog.dom.browserrange.IeRange'], ['goog.array', 'goog.dom', 'goog.dom.NodeType', 'goog.dom.RangeEndpoint', 'goog.dom.TagName', 'goog.dom.browserrange.AbstractRange', 'goog.dom.safe', 'goog.html.uncheckedconversions', 'goog.log', 'goog.string'], {});
goog.addDependency('dom/browserrange/operarange.js', ['goog.dom.browserrange.OperaRange'], ['goog.dom.browserrange.W3cRange'], {});
goog.addDependency('dom/browserrange/w3crange.js', ['goog.dom.browserrange.W3cRange'], ['goog.array', 'goog.dom', 'goog.dom.NodeType', 'goog.dom.RangeEndpoint', 'goog.dom.TagName', 'goog.dom.browserrange.AbstractRange', 'goog.string', 'goog.userAgent'], {});
goog.addDependency('dom/browserrange/webkitrange.js', ['goog.dom.browserrange.WebKitRange'], ['goog.dom.RangeEndpoint', 'goog.dom.browserrange.W3cRange', 'goog.userAgent'], {});
goog.addDependency('dom/bufferedviewportsizemonitor.js', ['goog.dom.BufferedViewportSizeMonitor'], ['goog.asserts', 'goog.async.Delay', 'goog.events', 'goog.events.EventTarget', 'goog.events.EventType'], {});
goog.addDependency('dom/bufferedviewportsizemonitor_test.js', ['goog.dom.BufferedViewportSizeMonitorTest'], ['goog.dom.BufferedViewportSizeMonitor', 'goog.dom.ViewportSizeMonitor', 'goog.events', 'goog.events.EventType', 'goog.math.Size', 'goog.testing.MockClock', 'goog.testing.events', 'goog.testing.events.Event', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/classes.js', ['goog.dom.classes'], ['goog.array'], {});
goog.addDependency('dom/classes_test.js', ['goog.dom.classes_test'], ['goog.dom', 'goog.dom.TagName', 'goog.dom.classes', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/classlist.js', ['goog.dom.classlist'], ['goog.array'], {});
goog.addDependency('dom/classlist_test.js', ['goog.dom.classlist_test'], ['goog.dom', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.testing.ExpectedFailures', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/controlrange.js', ['goog.dom.ControlRange', 'goog.dom.ControlRangeIterator'], ['goog.array', 'goog.dom', 'goog.dom.AbstractMultiRange', 'goog.dom.AbstractRange', 'goog.dom.RangeIterator', 'goog.dom.RangeType', 'goog.dom.SavedRange', 'goog.dom.TagWalkType', 'goog.dom.TextRange', 'goog.iter.StopIteration', 'goog.userAgent'], {});
goog.addDependency('dom/controlrange_test.js', ['goog.dom.ControlRangeTest'], ['goog.dom', 'goog.dom.ControlRange', 'goog.dom.RangeType', 'goog.dom.TagName', 'goog.dom.TextRange', 'goog.testing.dom', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/dataset.js', ['goog.dom.dataset'], ['goog.labs.userAgent.browser', 'goog.string', 'goog.userAgent.product'], {});
goog.addDependency('dom/dataset_test.js', ['goog.dom.datasetTest'], ['goog.dom', 'goog.dom.dataset', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/dom.js', ['goog.dom', 'goog.dom.Appendable', 'goog.dom.DomHelper'], ['goog.array', 'goog.asserts', 'goog.dom.BrowserFeature', 'goog.dom.NodeType', 'goog.dom.TagName', 'goog.dom.safe', 'goog.html.SafeHtml', 'goog.html.uncheckedconversions', 'goog.math.Coordinate', 'goog.math.Size', 'goog.object', 'goog.string', 'goog.string.Unicode', 'goog.userAgent'], {});
goog.addDependency('dom/dom_compile_test.js', ['goog.dom.DomCompileTest'], ['goog.dom', 'goog.dom.TagName', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/dom_test.js', ['goog.dom.dom_test'], ['goog.array', 'goog.asserts', 'goog.dom', 'goog.dom.BrowserFeature', 'goog.dom.DomHelper', 'goog.dom.InputType', 'goog.dom.NodeType', 'goog.dom.TagName', 'goog.functions', 'goog.html.SafeUrl', 'goog.html.testing', 'goog.object', 'goog.string.Const', 'goog.string.Unicode', 'goog.testing.PropertyReplacer', 'goog.testing.asserts', 'goog.testing.testSuite', 'goog.userAgent', 'goog.userAgent.product.isVersion'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/fontsizemonitor.js', ['goog.dom.FontSizeMonitor', 'goog.dom.FontSizeMonitor.EventType'], ['goog.dom', 'goog.dom.TagName', 'goog.events', 'goog.events.EventTarget', 'goog.events.EventType', 'goog.userAgent'], {});
goog.addDependency('dom/fontsizemonitor_test.js', ['goog.dom.FontSizeMonitorTest'], ['goog.dom', 'goog.dom.FontSizeMonitor', 'goog.dom.TagName', 'goog.events', 'goog.events.Event', 'goog.testing.PropertyReplacer', 'goog.testing.events', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/forms.js', ['goog.dom.forms'], ['goog.dom.InputType', 'goog.dom.TagName', 'goog.dom.safe', 'goog.structs.Map', 'goog.window'], {});
goog.addDependency('dom/forms_test.js', ['goog.dom.formsTest'], ['goog.dom', 'goog.dom.forms', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/fullscreen.js', ['goog.dom.fullscreen', 'goog.dom.fullscreen.EventType'], ['goog.dom'], {});
goog.addDependency('dom/fullscreen_test.js', ['goog.dom.fullscreen_test'], ['goog.dom.DomHelper', 'goog.dom.fullscreen', 'goog.testing.PropertyReplacer', 'goog.testing.asserts', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/htmlelement.js', ['goog.dom.HtmlElement'], [], {});
goog.addDependency('dom/iframe.js', ['goog.dom.iframe'], ['goog.dom', 'goog.dom.TagName', 'goog.dom.safe', 'goog.html.SafeHtml', 'goog.html.SafeStyle', 'goog.html.TrustedResourceUrl', 'goog.string.Const', 'goog.userAgent'], {});
goog.addDependency('dom/iframe_test.js', ['goog.dom.iframeTest'], ['goog.dom', 'goog.dom.iframe', 'goog.html.SafeHtml', 'goog.html.SafeStyle', 'goog.string.Const', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/inputtype.js', ['goog.dom.InputType'], [], {});
goog.addDependency('dom/inputtype_test.js', ['goog.dom.InputTypeTest'], ['goog.dom.InputType', 'goog.object', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/iter.js', ['goog.dom.iter.AncestorIterator', 'goog.dom.iter.ChildIterator', 'goog.dom.iter.SiblingIterator'], ['goog.iter.Iterator', 'goog.iter.StopIteration'], {});
goog.addDependency('dom/iter_test.js', ['goog.dom.iterTest'], ['goog.dom', 'goog.dom.NodeType', 'goog.dom.iter.AncestorIterator', 'goog.dom.iter.ChildIterator', 'goog.dom.iter.SiblingIterator', 'goog.testing.dom', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/multirange.js', ['goog.dom.MultiRange', 'goog.dom.MultiRangeIterator'], ['goog.array', 'goog.dom', 'goog.dom.AbstractMultiRange', 'goog.dom.AbstractRange', 'goog.dom.RangeIterator', 'goog.dom.RangeType', 'goog.dom.SavedRange', 'goog.dom.TextRange', 'goog.iter', 'goog.iter.StopIteration', 'goog.log'], {});
goog.addDependency('dom/multirange_test.js', ['goog.dom.MultiRangeTest'], ['goog.dom', 'goog.dom.MultiRange', 'goog.dom.Range', 'goog.iter', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/nodeiterator.js', ['goog.dom.NodeIterator'], ['goog.dom.TagIterator'], {});
goog.addDependency('dom/nodeiterator_test.js', ['goog.dom.NodeIteratorTest'], ['goog.dom', 'goog.dom.NodeIterator', 'goog.testing.dom', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/nodeoffset.js', ['goog.dom.NodeOffset'], ['goog.Disposable', 'goog.dom.TagName'], {});
goog.addDependency('dom/nodeoffset_test.js', ['goog.dom.NodeOffsetTest'], ['goog.dom', 'goog.dom.NodeOffset', 'goog.dom.NodeType', 'goog.dom.TagName', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/nodetype.js', ['goog.dom.NodeType'], [], {});
goog.addDependency('dom/pattern/abstractpattern.js', ['goog.dom.pattern.AbstractPattern'], ['goog.dom.TagWalkType', 'goog.dom.pattern.MatchType'], {});
goog.addDependency('dom/pattern/allchildren.js', ['goog.dom.pattern.AllChildren'], ['goog.dom.pattern.AbstractPattern', 'goog.dom.pattern.MatchType'], {});
goog.addDependency('dom/pattern/callback/callback.js', ['goog.dom.pattern.callback'], ['goog.dom', 'goog.dom.TagWalkType', 'goog.iter'], {});
goog.addDependency('dom/pattern/callback/counter.js', ['goog.dom.pattern.callback.Counter'], [], {});
goog.addDependency('dom/pattern/callback/test.js', ['goog.dom.pattern.callback.Test'], ['goog.iter.StopIteration'], {});
goog.addDependency('dom/pattern/childmatches.js', ['goog.dom.pattern.ChildMatches'], ['goog.dom.pattern.AllChildren', 'goog.dom.pattern.MatchType'], {});
goog.addDependency('dom/pattern/endtag.js', ['goog.dom.pattern.EndTag'], ['goog.dom.TagWalkType', 'goog.dom.pattern.Tag'], {});
goog.addDependency('dom/pattern/fulltag.js', ['goog.dom.pattern.FullTag'], ['goog.dom.pattern.MatchType', 'goog.dom.pattern.StartTag', 'goog.dom.pattern.Tag'], {});
goog.addDependency('dom/pattern/matcher.js', ['goog.dom.pattern.Matcher'], ['goog.dom.TagIterator', 'goog.dom.pattern.MatchType', 'goog.iter'], {});
goog.addDependency('dom/pattern/matcher_test.js', ['goog.dom.pattern.matcherTest'], ['goog.dom', 'goog.dom.TagName', 'goog.dom.pattern.EndTag', 'goog.dom.pattern.FullTag', 'goog.dom.pattern.Matcher', 'goog.dom.pattern.Repeat', 'goog.dom.pattern.Sequence', 'goog.dom.pattern.StartTag', 'goog.dom.pattern.callback.Counter', 'goog.dom.pattern.callback.Test', 'goog.iter.StopIteration', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/pattern/nodetype.js', ['goog.dom.pattern.NodeType'], ['goog.dom.pattern.AbstractPattern', 'goog.dom.pattern.MatchType'], {});
goog.addDependency('dom/pattern/pattern.js', ['goog.dom.pattern', 'goog.dom.pattern.MatchType'], [], {});
goog.addDependency('dom/pattern/pattern_test.js', ['goog.dom.patternTest'], ['goog.dom', 'goog.dom.NodeType', 'goog.dom.TagWalkType', 'goog.dom.pattern.AllChildren', 'goog.dom.pattern.ChildMatches', 'goog.dom.pattern.EndTag', 'goog.dom.pattern.FullTag', 'goog.dom.pattern.MatchType', 'goog.dom.pattern.NodeType', 'goog.dom.pattern.Repeat', 'goog.dom.pattern.Sequence', 'goog.dom.pattern.StartTag', 'goog.dom.pattern.Text', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/pattern/repeat.js', ['goog.dom.pattern.Repeat'], ['goog.dom.NodeType', 'goog.dom.pattern.AbstractPattern', 'goog.dom.pattern.MatchType'], {});
goog.addDependency('dom/pattern/sequence.js', ['goog.dom.pattern.Sequence'], ['goog.dom.NodeType', 'goog.dom.pattern', 'goog.dom.pattern.AbstractPattern', 'goog.dom.pattern.MatchType'], {});
goog.addDependency('dom/pattern/starttag.js', ['goog.dom.pattern.StartTag'], ['goog.dom.TagWalkType', 'goog.dom.pattern.Tag'], {});
goog.addDependency('dom/pattern/tag.js', ['goog.dom.pattern.Tag'], ['goog.dom.pattern', 'goog.dom.pattern.AbstractPattern', 'goog.dom.pattern.MatchType', 'goog.object'], {});
goog.addDependency('dom/pattern/text.js', ['goog.dom.pattern.Text'], ['goog.dom.NodeType', 'goog.dom.pattern', 'goog.dom.pattern.AbstractPattern', 'goog.dom.pattern.MatchType'], {});
goog.addDependency('dom/range.js', ['goog.dom.Range'], ['goog.dom', 'goog.dom.AbstractRange', 'goog.dom.BrowserFeature', 'goog.dom.ControlRange', 'goog.dom.MultiRange', 'goog.dom.NodeType', 'goog.dom.TextRange'], {});
goog.addDependency('dom/range_test.js', ['goog.dom.RangeTest'], ['goog.dom', 'goog.dom.NodeType', 'goog.dom.Range', 'goog.dom.RangeType', 'goog.dom.TagName', 'goog.dom.TextRange', 'goog.dom.browserrange', 'goog.testing.dom', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/rangeendpoint.js', ['goog.dom.RangeEndpoint'], [], {});
goog.addDependency('dom/safe.js', ['goog.dom.safe', 'goog.dom.safe.InsertAdjacentHtmlPosition'], ['goog.asserts', 'goog.dom.asserts', 'goog.functions', 'goog.html.SafeHtml', 'goog.html.SafeScript', 'goog.html.SafeStyle', 'goog.html.SafeUrl', 'goog.html.TrustedResourceUrl', 'goog.html.uncheckedconversions', 'goog.string.Const', 'goog.string.internal'], {});
goog.addDependency('dom/safe_test.js', ['goog.dom.safeTest'], ['goog.asserts', 'goog.dom', 'goog.dom.TagName', 'goog.dom.safe', 'goog.dom.safe.InsertAdjacentHtmlPosition', 'goog.html.SafeHtml', 'goog.html.SafeScript', 'goog.html.SafeStyle', 'goog.html.SafeUrl', 'goog.html.TrustedResourceUrl', 'goog.html.testing', 'goog.string', 'goog.string.Const', 'goog.testing', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/savedcaretrange.js', ['goog.dom.SavedCaretRange'], ['goog.array', 'goog.dom', 'goog.dom.SavedRange', 'goog.dom.TagName', 'goog.string'], {});
goog.addDependency('dom/savedcaretrange_test.js', ['goog.dom.SavedCaretRangeTest'], ['goog.dom', 'goog.dom.Range', 'goog.dom.SavedCaretRange', 'goog.testing.dom', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/savedrange.js', ['goog.dom.SavedRange'], ['goog.Disposable', 'goog.log'], {});
goog.addDependency('dom/savedrange_test.js', ['goog.dom.SavedRangeTest'], ['goog.dom', 'goog.dom.Range', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/selection.js', ['goog.dom.selection'], ['goog.dom.InputType', 'goog.string', 'goog.userAgent'], {});
goog.addDependency('dom/selection_test.js', ['goog.dom.selectionTest'], ['goog.dom', 'goog.dom.InputType', 'goog.dom.TagName', 'goog.dom.selection', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/tagiterator.js', ['goog.dom.TagIterator', 'goog.dom.TagWalkType'], ['goog.dom', 'goog.dom.NodeType', 'goog.iter.Iterator', 'goog.iter.StopIteration'], {});
goog.addDependency('dom/tagiterator_test.js', ['goog.dom.TagIteratorTest'], ['goog.dom', 'goog.dom.TagIterator', 'goog.dom.TagName', 'goog.dom.TagWalkType', 'goog.iter', 'goog.iter.StopIteration', 'goog.testing.dom', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/tagname.js', ['goog.dom.TagName'], ['goog.dom.HtmlElement'], {});
goog.addDependency('dom/tagname_test.js', ['goog.dom.TagNameTest'], ['goog.dom.TagName', 'goog.object', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/tags.js', ['goog.dom.tags'], ['goog.object'], {});
goog.addDependency('dom/tags_test.js', ['goog.dom.tagsTest'], ['goog.dom.tags', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/textassert.js', ['goog.dom.textAssert'], ['goog.asserts', 'goog.dom', 'goog.dom.TagName'], {});
goog.addDependency('dom/textassert_test.js', ['goog.dom.textassert_test'], ['goog.dom.textAssert', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/textrange.js', ['goog.dom.TextRange'], ['goog.array', 'goog.dom', 'goog.dom.AbstractRange', 'goog.dom.RangeType', 'goog.dom.SavedRange', 'goog.dom.TagName', 'goog.dom.TextRangeIterator', 'goog.dom.browserrange', 'goog.string', 'goog.userAgent'], {});
goog.addDependency('dom/textrange_test.js', ['goog.dom.TextRangeTest'], ['goog.dom', 'goog.dom.ControlRange', 'goog.dom.Range', 'goog.dom.TextRange', 'goog.math.Coordinate', 'goog.style', 'goog.testing.ExpectedFailures', 'goog.testing.testSuite', 'goog.userAgent', 'goog.userAgent.product'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/textrangeiterator.js', ['goog.dom.TextRangeIterator'], ['goog.array', 'goog.dom', 'goog.dom.NodeType', 'goog.dom.RangeIterator', 'goog.dom.TagName', 'goog.iter.StopIteration'], {});
goog.addDependency('dom/textrangeiterator_test.js', ['goog.dom.TextRangeIteratorTest'], ['goog.dom', 'goog.dom.TagName', 'goog.dom.TextRangeIterator', 'goog.iter.StopIteration', 'goog.testing.dom', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/uri.js', ['goog.dom.uri'], ['goog.dom', 'goog.dom.TagName', 'goog.dom.safe', 'goog.html.uncheckedconversions', 'goog.string.Const'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/uri_test.js', ['goog.dom.uriTest'], ['goog.dom.uri', 'goog.testing.testSuite', 'goog.userAgent.product'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/vendor.js', ['goog.dom.vendor'], ['goog.string', 'goog.userAgent'], {});
goog.addDependency('dom/vendor_test.js', ['goog.dom.vendorTest'], ['goog.array', 'goog.dom.vendor', 'goog.labs.userAgent.util', 'goog.testing.MockUserAgent', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite', 'goog.userAgent', 'goog.userAgentTestUtil'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/viewportsizemonitor.js', ['goog.dom.ViewportSizeMonitor'], ['goog.dom', 'goog.events', 'goog.events.EventTarget', 'goog.events.EventType', 'goog.math.Size'], {});
goog.addDependency('dom/viewportsizemonitor_test.js', ['goog.dom.ViewportSizeMonitorTest'], ['goog.dom.ViewportSizeMonitor', 'goog.events', 'goog.events.Event', 'goog.events.EventTarget', 'goog.events.EventType', 'goog.math.Size', 'goog.testing.MockClock', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('dom/xml.js', ['goog.dom.xml'], ['goog.dom', 'goog.dom.NodeType', 'goog.dom.safe', 'goog.html.legacyconversions', 'goog.userAgent'], {});
goog.addDependency('dom/xml_test.js', ['goog.dom.xmlTest'], ['goog.dom', 'goog.dom.TagName', 'goog.dom.xml', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('editor/browserfeature.js', ['goog.editor.BrowserFeature'], ['goog.editor.defines', 'goog.labs.userAgent.browser', 'goog.userAgent', 'goog.userAgent.product', 'goog.userAgent.product.isVersion'], {});
goog.addDependency('editor/browserfeature_test.js', ['goog.editor.BrowserFeatureTest'], ['goog.dom', 'goog.dom.Range', 'goog.dom.TagName', 'goog.editor.BrowserFeature', 'goog.testing.ExpectedFailures', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('editor/clicktoeditwrapper.js', ['goog.editor.ClickToEditWrapper'], ['goog.Disposable', 'goog.dom', 'goog.dom.Range', 'goog.dom.TagName', 'goog.editor.BrowserFeature', 'goog.editor.Command', 'goog.editor.Field', 'goog.editor.range', 'goog.events.BrowserEvent', 'goog.events.EventHandler', 'goog.events.EventType'], {});
goog.addDependency('editor/clicktoeditwrapper_test.js', ['goog.editor.ClickToEditWrapperTest'], ['goog.dom', 'goog.dom.Range', 'goog.dom.TagName', 'goog.editor.ClickToEditWrapper', 'goog.editor.SeamlessField', 'goog.testing.MockClock', 'goog.testing.events', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('editor/command.js', ['goog.editor.Command'], [], {});
goog.addDependency('editor/contenteditablefield.js', ['goog.editor.ContentEditableField'], ['goog.asserts', 'goog.editor.Field', 'goog.log'], {});
goog.addDependency('editor/contenteditablefield_test.js', ['goog.editor.ContentEditableFieldTest'], ['goog.dom', 'goog.editor.ContentEditableField', 'goog.html.SafeHtml', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('editor/defines.js', ['goog.editor.defines'], [], {});
goog.addDependency('editor/field.js', ['goog.editor.Field', 'goog.editor.Field.EventType'], ['goog.a11y.aria', 'goog.a11y.aria.Role', 'goog.array', 'goog.asserts', 'goog.async.Delay', 'goog.dom', 'goog.dom.Range', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.dom.safe', 'goog.editor.BrowserFeature', 'goog.editor.Command', 'goog.editor.PluginImpl', 'goog.editor.icontent', 'goog.editor.icontent.FieldFormatInfo', 'goog.editor.icontent.FieldStyleInfo', 'goog.editor.node', 'goog.editor.range', 'goog.events', 'goog.events.EventHandler', 'goog.events.EventTarget', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.functions', 'goog.html.SafeHtml', 'goog.html.SafeStyleSheet', 'goog.log', 'goog.log.Level', 'goog.string', 'goog.string.Unicode', 'goog.style', 'goog.userAgent', 'goog.userAgent.product'], {});
goog.addDependency('editor/field_test.js', ['goog.editor.field_test'], ['goog.array', 'goog.dom', 'goog.dom.Range', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.editor.BrowserFeature', 'goog.editor.Field', 'goog.editor.Plugin', 'goog.editor.range', 'goog.events', 'goog.events.BrowserEvent', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.functions', 'goog.html.SafeHtml', 'goog.testing.LooseMock', 'goog.testing.MockClock', 'goog.testing.dom', 'goog.testing.events', 'goog.testing.events.Event', 'goog.testing.recordFunction', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('editor/focus.js', ['goog.editor.focus'], ['goog.dom.selection'], {});
goog.addDependency('editor/focus_test.js', ['goog.editor.focusTest'], ['goog.dom.selection', 'goog.editor.BrowserFeature', 'goog.editor.focus', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('editor/icontent.js', ['goog.editor.icontent', 'goog.editor.icontent.FieldFormatInfo', 'goog.editor.icontent.FieldStyleInfo'], ['goog.dom', 'goog.editor.BrowserFeature', 'goog.style', 'goog.userAgent'], {});
goog.addDependency('editor/icontent_test.js', ['goog.editor.icontentTest'], ['goog.dom', 'goog.dom.TagName', 'goog.editor.BrowserFeature', 'goog.editor.icontent', 'goog.editor.icontent.FieldFormatInfo', 'goog.editor.icontent.FieldStyleInfo', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('editor/link.js', ['goog.editor.Link'], ['goog.array', 'goog.dom', 'goog.dom.NodeType', 'goog.dom.Range', 'goog.dom.TagName', 'goog.editor.BrowserFeature', 'goog.editor.Command', 'goog.editor.Field', 'goog.editor.node', 'goog.editor.range', 'goog.string', 'goog.string.Unicode', 'goog.uri.utils', 'goog.uri.utils.ComponentIndex'], {});
goog.addDependency('editor/link_test.js', ['goog.editor.LinkTest'], ['goog.dom', 'goog.dom.NodeType', 'goog.dom.Range', 'goog.dom.TagName', 'goog.editor.Link', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('editor/node.js', ['goog.editor.node'], ['goog.dom', 'goog.dom.NodeType', 'goog.dom.TagName', 'goog.dom.iter.ChildIterator', 'goog.dom.iter.SiblingIterator', 'goog.iter', 'goog.object', 'goog.string', 'goog.string.Unicode', 'goog.userAgent'], {});
goog.addDependency('editor/node_test.js', ['goog.editor.nodeTest'], ['goog.array', 'goog.dom', 'goog.dom.NodeType', 'goog.dom.TagName', 'goog.editor.node', 'goog.style', 'goog.testing.ExpectedFailures', 'goog.testing.dom', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('editor/plugin.js', ['goog.editor.Plugin'], ['goog.editor.Field', 'goog.editor.PluginImpl'], {});
goog.addDependency('editor/plugin_impl.js', ['goog.editor.PluginImpl'], ['goog.events.EventTarget', 'goog.functions', 'goog.log', 'goog.object', 'goog.reflect', 'goog.userAgent'], {});
goog.addDependency('editor/plugin_test.js', ['goog.editor.PluginTest'], ['goog.editor.Field', 'goog.editor.Plugin', 'goog.functions', 'goog.testing.StrictMock', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('editor/plugins/abstractbubbleplugin.js', ['goog.editor.plugins.AbstractBubblePlugin'], ['goog.array', 'goog.dom', 'goog.dom.NodeType', 'goog.dom.Range', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.editor.Plugin', 'goog.editor.style', 'goog.events', 'goog.events.EventHandler', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.events.actionEventWrapper', 'goog.functions', 'goog.string.Unicode', 'goog.ui.Component', 'goog.ui.editor.Bubble', 'goog.userAgent'], {});
goog.addDependency('editor/plugins/abstractbubbleplugin_test.js', ['goog.editor.plugins.AbstractBubblePluginTest'], ['goog.dom', 'goog.dom.TagName', 'goog.editor.plugins.AbstractBubblePlugin', 'goog.events.BrowserEvent', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.functions', 'goog.style', 'goog.testing.editor.FieldMock', 'goog.testing.editor.TestHelper', 'goog.testing.events', 'goog.testing.events.Event', 'goog.testing.testSuite', 'goog.ui.editor.Bubble', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('editor/plugins/abstractdialogplugin.js', ['goog.editor.plugins.AbstractDialogPlugin', 'goog.editor.plugins.AbstractDialogPlugin.EventType'], ['goog.dom', 'goog.dom.Range', 'goog.editor.Field', 'goog.editor.Plugin', 'goog.editor.range', 'goog.events', 'goog.ui.editor.AbstractDialog'], {});
goog.addDependency('editor/plugins/abstractdialogplugin_test.js', ['goog.editor.plugins.AbstractDialogPluginTest'], ['goog.dom', 'goog.dom.SavedRange', 'goog.dom.TagName', 'goog.editor.Field', 'goog.editor.plugins.AbstractDialogPlugin', 'goog.events.Event', 'goog.events.EventHandler', 'goog.functions', 'goog.html.SafeHtml', 'goog.testing.MockClock', 'goog.testing.MockControl', 'goog.testing.PropertyReplacer', 'goog.testing.editor.FieldMock', 'goog.testing.editor.TestHelper', 'goog.testing.events', 'goog.testing.mockmatchers.ArgumentMatcher', 'goog.testing.testSuite', 'goog.ui.editor.AbstractDialog', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('editor/plugins/abstracttabhandler.js', ['goog.editor.plugins.AbstractTabHandler'], ['goog.editor.Plugin', 'goog.events.KeyCodes', 'goog.userAgent'], {});
goog.addDependency('editor/plugins/abstracttabhandler_test.js', ['goog.editor.plugins.AbstractTabHandlerTest'], ['goog.editor.Field', 'goog.editor.plugins.AbstractTabHandler', 'goog.events.BrowserEvent', 'goog.events.KeyCodes', 'goog.testing.StrictMock', 'goog.testing.editor.FieldMock', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('editor/plugins/basictextformatter.js', ['goog.editor.plugins.BasicTextFormatter', 'goog.editor.plugins.BasicTextFormatter.COMMAND'], ['goog.array', 'goog.dom', 'goog.dom.NodeType', 'goog.dom.Range', 'goog.dom.TagName', 'goog.editor.BrowserFeature', 'goog.editor.Command', 'goog.editor.Link', 'goog.editor.Plugin', 'goog.editor.node', 'goog.editor.range', 'goog.editor.style', 'goog.iter', 'goog.iter.StopIteration', 'goog.log', 'goog.object', 'goog.string', 'goog.string.Unicode', 'goog.style', 'goog.ui.editor.messages', 'goog.userAgent'], {});
goog.addDependency('editor/plugins/basictextformatter_test.js', ['goog.editor.plugins.BasicTextFormatterTest'], ['goog.array', 'goog.dom', 'goog.dom.Range', 'goog.dom.TagName', 'goog.editor.BrowserFeature', 'goog.editor.Command', 'goog.editor.Field', 'goog.editor.Plugin', 'goog.editor.plugins.BasicTextFormatter', 'goog.html.SafeHtml', 'goog.object', 'goog.style', 'goog.testing.ExpectedFailures', 'goog.testing.LooseMock', 'goog.testing.PropertyReplacer', 'goog.testing.editor.FieldMock', 'goog.testing.editor.TestHelper', 'goog.testing.mockmatchers', 'goog.testing.testSuite', 'goog.userAgent', 'goog.userAgent.product'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('editor/plugins/blockquote.js', ['goog.editor.plugins.Blockquote'], ['goog.dom', 'goog.dom.NodeType', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.editor.BrowserFeature', 'goog.editor.Command', 'goog.editor.Plugin', 'goog.editor.node', 'goog.functions', 'goog.log'], {});
goog.addDependency('editor/plugins/blockquote_test.js', ['goog.editor.plugins.BlockquoteTest'], ['goog.dom', 'goog.dom.Range', 'goog.dom.TagName', 'goog.editor.BrowserFeature', 'goog.editor.plugins.Blockquote', 'goog.testing.editor.FieldMock', 'goog.testing.editor.TestHelper', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('editor/plugins/emoticons.js', ['goog.editor.plugins.Emoticons'], ['goog.dom.TagName', 'goog.editor.Plugin', 'goog.editor.range', 'goog.functions', 'goog.ui.emoji.Emoji', 'goog.userAgent'], {});
goog.addDependency('editor/plugins/emoticons_test.js', ['goog.editor.plugins.EmoticonsTest'], ['goog.Uri', 'goog.array', 'goog.dom', 'goog.dom.TagName', 'goog.editor.Field', 'goog.editor.plugins.Emoticons', 'goog.testing.testSuite', 'goog.ui.emoji.Emoji', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('editor/plugins/enterhandler.js', ['goog.editor.plugins.EnterHandler'], ['goog.dom', 'goog.dom.NodeOffset', 'goog.dom.NodeType', 'goog.dom.Range', 'goog.dom.TagName', 'goog.editor.BrowserFeature', 'goog.editor.Plugin', 'goog.editor.node', 'goog.editor.plugins.Blockquote', 'goog.editor.range', 'goog.editor.style', 'goog.events.KeyCodes', 'goog.functions', 'goog.object', 'goog.string', 'goog.userAgent'], {});
goog.addDependency('editor/plugins/enterhandler_test.js', ['goog.editor.plugins.EnterHandlerTest'], ['goog.dom', 'goog.dom.NodeType', 'goog.dom.Range', 'goog.dom.TagName', 'goog.editor.BrowserFeature', 'goog.editor.Field', 'goog.editor.Plugin', 'goog.editor.plugins.Blockquote', 'goog.editor.plugins.EnterHandler', 'goog.editor.range', 'goog.events', 'goog.events.KeyCodes', 'goog.html.testing', 'goog.testing.ExpectedFailures', 'goog.testing.MockClock', 'goog.testing.dom', 'goog.testing.editor.TestHelper', 'goog.testing.events', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('editor/plugins/firststrong.js', ['goog.editor.plugins.FirstStrong'], ['goog.dom.NodeType', 'goog.dom.TagIterator', 'goog.dom.TagName', 'goog.editor.Command', 'goog.editor.Field', 'goog.editor.Plugin', 'goog.editor.node', 'goog.editor.range', 'goog.i18n.bidi', 'goog.i18n.uChar', 'goog.iter', 'goog.userAgent'], {});
goog.addDependency('editor/plugins/firststrong_test.js', ['goog.editor.plugins.FirstStrongTest'], ['goog.dom.Range', 'goog.editor.Command', 'goog.editor.Field', 'goog.editor.plugins.FirstStrong', 'goog.editor.range', 'goog.events.KeyCodes', 'goog.html.testing', 'goog.testing.MockClock', 'goog.testing.editor.TestHelper', 'goog.testing.events', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('editor/plugins/headerformatter.js', ['goog.editor.plugins.HeaderFormatter'], ['goog.editor.Command', 'goog.editor.Plugin', 'goog.userAgent'], {});
goog.addDependency('editor/plugins/headerformatter_test.js', ['goog.editor.plugins.HeaderFormatterTest'], ['goog.dom', 'goog.editor.Command', 'goog.editor.plugins.BasicTextFormatter', 'goog.editor.plugins.HeaderFormatter', 'goog.events.BrowserEvent', 'goog.testing.LooseMock', 'goog.testing.editor.FieldMock', 'goog.testing.editor.TestHelper', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('editor/plugins/linkbubble.js', ['goog.editor.plugins.LinkBubble', 'goog.editor.plugins.LinkBubble.Action'], ['goog.array', 'goog.dom', 'goog.dom.Range', 'goog.dom.TagName', 'goog.editor.Command', 'goog.editor.Link', 'goog.editor.plugins.AbstractBubblePlugin', 'goog.functions', 'goog.string', 'goog.style', 'goog.ui.editor.messages', 'goog.uri.utils', 'goog.window'], {});
goog.addDependency('editor/plugins/linkbubble_test.js', ['goog.editor.plugins.LinkBubbleTest'], ['goog.dom', 'goog.dom.Range', 'goog.dom.TagName', 'goog.editor.Command', 'goog.editor.Link', 'goog.editor.plugins.LinkBubble', 'goog.events.BrowserEvent', 'goog.events.Event', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.string', 'goog.style', 'goog.testing.FunctionMock', 'goog.testing.PropertyReplacer', 'goog.testing.editor.FieldMock', 'goog.testing.editor.TestHelper', 'goog.testing.events', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('editor/plugins/linkdialogplugin.js', ['goog.editor.plugins.LinkDialogPlugin'], ['goog.array', 'goog.dom', 'goog.editor.Command', 'goog.editor.plugins.AbstractDialogPlugin', 'goog.events.EventHandler', 'goog.functions', 'goog.ui.editor.AbstractDialog', 'goog.ui.editor.LinkDialog', 'goog.uri.utils'], {});
goog.addDependency('editor/plugins/linkdialogplugin_test.js', ['goog.ui.editor.plugins.LinkDialogTest'], ['goog.dom', 'goog.dom.DomHelper', 'goog.dom.NodeType', 'goog.dom.TagName', 'goog.editor.BrowserFeature', 'goog.editor.Command', 'goog.editor.Field', 'goog.editor.Link', 'goog.editor.plugins.LinkDialogPlugin', 'goog.html.SafeHtml', 'goog.string', 'goog.string.Unicode', 'goog.testing.MockControl', 'goog.testing.editor.FieldMock', 'goog.testing.editor.TestHelper', 'goog.testing.editor.dom', 'goog.testing.events', 'goog.testing.mockmatchers', 'goog.testing.testSuite', 'goog.ui.editor.AbstractDialog', 'goog.ui.editor.LinkDialog', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('editor/plugins/linkshortcutplugin.js', ['goog.editor.plugins.LinkShortcutPlugin'], ['goog.editor.Command', 'goog.editor.Plugin'], {});
goog.addDependency('editor/plugins/linkshortcutplugin_test.js', ['goog.editor.plugins.LinkShortcutPluginTest'], ['goog.dom', 'goog.dom.TagName', 'goog.editor.Field', 'goog.editor.plugins.BasicTextFormatter', 'goog.editor.plugins.LinkBubble', 'goog.editor.plugins.LinkShortcutPlugin', 'goog.events.KeyCodes', 'goog.testing.PropertyReplacer', 'goog.testing.dom', 'goog.testing.events', 'goog.testing.testSuite', 'goog.userAgent.product'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('editor/plugins/listtabhandler.js', ['goog.editor.plugins.ListTabHandler'], ['goog.dom', 'goog.dom.TagName', 'goog.editor.Command', 'goog.editor.plugins.AbstractTabHandler', 'goog.iter'], {});
goog.addDependency('editor/plugins/listtabhandler_test.js', ['goog.editor.plugins.ListTabHandlerTest'], ['goog.dom', 'goog.editor.Command', 'goog.editor.plugins.ListTabHandler', 'goog.events.BrowserEvent', 'goog.events.KeyCodes', 'goog.functions', 'goog.testing.StrictMock', 'goog.testing.editor.FieldMock', 'goog.testing.editor.TestHelper', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('editor/plugins/loremipsum.js', ['goog.editor.plugins.LoremIpsum'], ['goog.asserts', 'goog.dom', 'goog.editor.Command', 'goog.editor.Field', 'goog.editor.Plugin', 'goog.editor.node', 'goog.functions', 'goog.html.SafeHtml', 'goog.userAgent'], {});
goog.addDependency('editor/plugins/loremipsum_test.js', ['goog.editor.plugins.LoremIpsumTest'], ['goog.dom', 'goog.editor.Command', 'goog.editor.Field', 'goog.editor.plugins.LoremIpsum', 'goog.html.SafeHtml', 'goog.string.Unicode', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('editor/plugins/removeformatting.js', ['goog.editor.plugins.RemoveFormatting'], ['goog.dom', 'goog.dom.NodeType', 'goog.dom.Range', 'goog.dom.TagName', 'goog.editor.BrowserFeature', 'goog.editor.Plugin', 'goog.editor.node', 'goog.editor.range', 'goog.string', 'goog.userAgent'], {});
goog.addDependency('editor/plugins/removeformatting_test.js', ['goog.editor.plugins.RemoveFormattingTest'], ['goog.dom', 'goog.dom.Range', 'goog.dom.TagName', 'goog.editor.BrowserFeature', 'goog.editor.plugins.RemoveFormatting', 'goog.string', 'goog.testing.ExpectedFailures', 'goog.testing.dom', 'goog.testing.editor.FieldMock', 'goog.testing.editor.TestHelper', 'goog.testing.testSuite', 'goog.userAgent', 'goog.userAgent.product'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('editor/plugins/spacestabhandler.js', ['goog.editor.plugins.SpacesTabHandler'], ['goog.dom.TagName', 'goog.editor.plugins.AbstractTabHandler', 'goog.editor.range'], {});
goog.addDependency('editor/plugins/spacestabhandler_test.js', ['goog.editor.plugins.SpacesTabHandlerTest'], ['goog.dom', 'goog.dom.Range', 'goog.editor.plugins.SpacesTabHandler', 'goog.events.BrowserEvent', 'goog.events.KeyCodes', 'goog.functions', 'goog.testing.StrictMock', 'goog.testing.editor.FieldMock', 'goog.testing.editor.TestHelper', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('editor/plugins/tableeditor.js', ['goog.editor.plugins.TableEditor'], ['goog.array', 'goog.dom', 'goog.dom.Range', 'goog.dom.TagName', 'goog.editor.Plugin', 'goog.editor.Table', 'goog.editor.node', 'goog.editor.range', 'goog.object', 'goog.userAgent'], {});
goog.addDependency('editor/plugins/tableeditor_test.js', ['goog.editor.plugins.TableEditorTest'], ['goog.dom', 'goog.dom.Range', 'goog.dom.TagName', 'goog.editor.plugins.TableEditor', 'goog.object', 'goog.string', 'goog.testing.ExpectedFailures', 'goog.testing.TestCase', 'goog.testing.editor.FieldMock', 'goog.testing.editor.TestHelper', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('editor/plugins/tagonenterhandler.js', ['goog.editor.plugins.TagOnEnterHandler'], ['goog.dom', 'goog.dom.NodeType', 'goog.dom.Range', 'goog.dom.TagName', 'goog.editor.Command', 'goog.editor.node', 'goog.editor.plugins.EnterHandler', 'goog.editor.range', 'goog.editor.style', 'goog.events.KeyCodes', 'goog.functions', 'goog.string.Unicode', 'goog.style', 'goog.userAgent'], {});
goog.addDependency('editor/plugins/tagonenterhandler_test.js', ['goog.editor.plugins.TagOnEnterHandlerTest'], ['goog.dom', 'goog.dom.NodeType', 'goog.dom.Range', 'goog.dom.TagName', 'goog.editor.BrowserFeature', 'goog.editor.Field', 'goog.editor.Plugin', 'goog.editor.plugins.TagOnEnterHandler', 'goog.events.KeyCodes', 'goog.html.SafeHtml', 'goog.string.Unicode', 'goog.testing.dom', 'goog.testing.editor.TestHelper', 'goog.testing.events', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('editor/plugins/undoredo.js', ['goog.editor.plugins.UndoRedo'], ['goog.dom', 'goog.dom.NodeOffset', 'goog.dom.Range', 'goog.editor.BrowserFeature', 'goog.editor.Command', 'goog.editor.Field', 'goog.editor.Plugin', 'goog.editor.node', 'goog.editor.plugins.UndoRedoManager', 'goog.editor.plugins.UndoRedoState', 'goog.events', 'goog.events.EventHandler', 'goog.log', 'goog.object'], {});
goog.addDependency('editor/plugins/undoredo_test.js', ['goog.editor.plugins.UndoRedoTest'], ['goog.array', 'goog.dom', 'goog.dom.browserrange', 'goog.editor.Field', 'goog.editor.plugins.LoremIpsum', 'goog.editor.plugins.UndoRedo', 'goog.events', 'goog.functions', 'goog.html.SafeHtml', 'goog.testing.MockClock', 'goog.testing.PropertyReplacer', 'goog.testing.StrictMock', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('editor/plugins/undoredomanager.js', ['goog.editor.plugins.UndoRedoManager', 'goog.editor.plugins.UndoRedoManager.EventType'], ['goog.editor.plugins.UndoRedoState', 'goog.events', 'goog.events.EventTarget'], {});
goog.addDependency('editor/plugins/undoredomanager_test.js', ['goog.editor.plugins.UndoRedoManagerTest'], ['goog.editor.plugins.UndoRedoManager', 'goog.editor.plugins.UndoRedoState', 'goog.events', 'goog.testing.StrictMock', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('editor/plugins/undoredostate.js', ['goog.editor.plugins.UndoRedoState'], ['goog.events.EventTarget'], {});
goog.addDependency('editor/plugins/undoredostate_test.js', ['goog.editor.plugins.UndoRedoStateTest'], ['goog.editor.plugins.UndoRedoState', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('editor/range.js', ['goog.editor.range', 'goog.editor.range.Point'], ['goog.array', 'goog.dom', 'goog.dom.NodeType', 'goog.dom.Range', 'goog.dom.RangeEndpoint', 'goog.dom.SavedCaretRange', 'goog.editor.node', 'goog.editor.style', 'goog.iter', 'goog.userAgent'], {});
goog.addDependency('editor/range_test.js', ['goog.editor.rangeTest'], ['goog.dom', 'goog.dom.Range', 'goog.dom.TagName', 'goog.editor.range', 'goog.editor.range.Point', 'goog.string', 'goog.testing.dom', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('editor/seamlessfield.js', ['goog.editor.SeamlessField'], ['goog.cssom.iframe.style', 'goog.dom', 'goog.dom.Range', 'goog.dom.TagName', 'goog.dom.safe', 'goog.editor.BrowserFeature', 'goog.editor.Field', 'goog.editor.icontent', 'goog.editor.icontent.FieldFormatInfo', 'goog.editor.icontent.FieldStyleInfo', 'goog.editor.node', 'goog.events', 'goog.events.EventType', 'goog.html.SafeHtml', 'goog.log', 'goog.style'], {});
goog.addDependency('editor/seamlessfield_test.js', ['goog.editor.seamlessfield_test'], ['goog.dom', 'goog.dom.DomHelper', 'goog.dom.Range', 'goog.dom.TagName', 'goog.editor.BrowserFeature', 'goog.editor.Field', 'goog.editor.SeamlessField', 'goog.events', 'goog.functions', 'goog.html.SafeHtml', 'goog.style', 'goog.testing.MockClock', 'goog.testing.MockRange', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('editor/style.js', ['goog.editor.style'], ['goog.array', 'goog.asserts', 'goog.dom', 'goog.dom.NodeType', 'goog.dom.TagName', 'goog.editor.BrowserFeature', 'goog.events.EventHandler', 'goog.events.EventType', 'goog.object', 'goog.style', 'goog.userAgent'], {});
goog.addDependency('editor/style_test.js', ['goog.editor.styleTest'], ['goog.dom', 'goog.dom.TagName', 'goog.editor.BrowserFeature', 'goog.editor.style', 'goog.events.EventHandler', 'goog.events.EventType', 'goog.style', 'goog.testing.LooseMock', 'goog.testing.mockmatchers', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('editor/table.js', ['goog.editor.Table', 'goog.editor.TableCell', 'goog.editor.TableRow'], ['goog.asserts', 'goog.dom', 'goog.dom.DomHelper', 'goog.dom.NodeType', 'goog.dom.TagName', 'goog.log', 'goog.string.Unicode', 'goog.style'], {});
goog.addDependency('editor/table_test.js', ['goog.editor.TableTest'], ['goog.dom', 'goog.dom.TagName', 'goog.editor.Table', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('events/actioneventwrapper.js', ['goog.events.actionEventWrapper'], ['goog.a11y.aria', 'goog.a11y.aria.Role', 'goog.dom', 'goog.events', 'goog.events.EventHandler', 'goog.events.EventType', 'goog.events.EventWrapper', 'goog.events.KeyCodes'], {});
goog.addDependency('events/actioneventwrapper_test.js', ['goog.events.actionEventWrapperTest'], ['goog.a11y.aria', 'goog.a11y.aria.Role', 'goog.events', 'goog.events.EventHandler', 'goog.events.KeyCodes', 'goog.events.actionEventWrapper', 'goog.testing.events', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('events/actionhandler.js', ['goog.events.ActionEvent', 'goog.events.ActionHandler', 'goog.events.ActionHandler.EventType', 'goog.events.BeforeActionEvent'], ['goog.events', 'goog.events.BrowserEvent', 'goog.events.EventTarget', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.userAgent'], {});
goog.addDependency('events/actionhandler_test.js', ['goog.events.ActionHandlerTest'], ['goog.dom', 'goog.events', 'goog.events.ActionHandler', 'goog.testing.events', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('events/browserevent.js', ['goog.events.BrowserEvent', 'goog.events.BrowserEvent.MouseButton', 'goog.events.BrowserEvent.PointerType'], ['goog.debug', 'goog.events.BrowserFeature', 'goog.events.Event', 'goog.events.EventType', 'goog.reflect', 'goog.userAgent'], {});
goog.addDependency('events/browserevent_test.js', ['goog.events.BrowserEventTest'], ['goog.events.BrowserEvent', 'goog.events.BrowserFeature', 'goog.math.Coordinate', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('events/browserfeature.js', ['goog.events.BrowserFeature'], ['goog.userAgent'], {});
goog.addDependency('events/event.js', ['goog.events.Event', 'goog.events.EventLike'], ['goog.Disposable', 'goog.events.EventId'], {});
goog.addDependency('events/event_test.js', ['goog.events.EventTest'], ['goog.events.Event', 'goog.events.EventId', 'goog.events.EventTarget', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('events/eventhandler.js', ['goog.events.EventHandler'], ['goog.Disposable', 'goog.events', 'goog.object'], {});
goog.addDependency('events/eventhandler_test.js', ['goog.events.EventHandlerTest'], ['goog.events', 'goog.events.EventHandler', 'goog.events.EventTarget', 'goog.testing.PropertyReplacer', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('events/eventid.js', ['goog.events.EventId'], [], {});
goog.addDependency('events/events.js', ['goog.events', 'goog.events.CaptureSimulationMode', 'goog.events.Key', 'goog.events.ListenableType'], ['goog.asserts', 'goog.debug.entryPointRegistry', 'goog.events.BrowserEvent', 'goog.events.BrowserFeature', 'goog.events.Listenable', 'goog.events.ListenerMap'], {});
goog.addDependency('events/events_test.js', ['goog.eventsTest'], ['goog.asserts.AssertionError', 'goog.debug.EntryPointMonitor', 'goog.debug.ErrorHandler', 'goog.debug.entryPointRegistry', 'goog.dom', 'goog.dom.TagName', 'goog.events', 'goog.events.BrowserFeature', 'goog.events.CaptureSimulationMode', 'goog.events.Event', 'goog.events.EventTarget', 'goog.events.EventType', 'goog.events.Listener', 'goog.functions', 'goog.testing.PropertyReplacer', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('events/eventtarget.js', ['goog.events.EventTarget'], ['goog.Disposable', 'goog.asserts', 'goog.events', 'goog.events.Event', 'goog.events.Listenable', 'goog.events.ListenerMap', 'goog.object'], {});
goog.addDependency('events/eventtarget_test.js', ['goog.events.EventTargetTest'], ['goog.events.EventTarget', 'goog.events.Listenable', 'goog.events.eventTargetTester', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('events/eventtarget_via_googevents_test.js', ['goog.events.EventTargetGoogEventsTest'], ['goog.events', 'goog.events.EventTarget', 'goog.events.eventTargetTester', 'goog.testing', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('events/eventtarget_via_w3cinterface_test.js', ['goog.events.EventTargetW3CTest'], ['goog.events.EventTarget', 'goog.events.eventTargetTester', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('events/eventtargettester.js', ['goog.events.eventTargetTester'], ['goog.array', 'goog.events.Event', 'goog.events.EventTarget', 'goog.events.Listenable', 'goog.testing.asserts', 'goog.testing.recordFunction'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('events/eventtype.js', ['goog.events.EventType', 'goog.events.MouseAsMouseEventType', 'goog.events.MouseEvents', 'goog.events.PointerAsMouseEventType', 'goog.events.PointerAsTouchEventType', 'goog.events.PointerFallbackEventType', 'goog.events.PointerTouchFallbackEventType'], ['goog.events.BrowserFeature', 'goog.userAgent'], {});
goog.addDependency('events/eventtype_test.js', ['goog.events.EventTypeTest'], ['goog.events.BrowserFeature', 'goog.events.EventType', 'goog.events.PointerFallbackEventType', 'goog.events.PointerTouchFallbackEventType', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('events/eventwrapper.js', ['goog.events.EventWrapper'], [], {});
goog.addDependency('events/filedrophandler.js', ['goog.events.FileDropHandler', 'goog.events.FileDropHandler.EventType'], ['goog.array', 'goog.dom', 'goog.events.BrowserEvent', 'goog.events.EventHandler', 'goog.events.EventTarget', 'goog.events.EventType', 'goog.log', 'goog.log.Level'], {});
goog.addDependency('events/filedrophandler_test.js', ['goog.events.FileDropHandlerTest'], ['goog.events', 'goog.events.BrowserEvent', 'goog.events.EventTarget', 'goog.events.EventType', 'goog.events.FileDropHandler', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('events/focushandler.js', ['goog.events.FocusHandler', 'goog.events.FocusHandler.EventType'], ['goog.events', 'goog.events.BrowserEvent', 'goog.events.EventTarget', 'goog.userAgent'], {});
goog.addDependency('events/imehandler.js', ['goog.events.ImeHandler', 'goog.events.ImeHandler.Event', 'goog.events.ImeHandler.EventType'], ['goog.events.Event', 'goog.events.EventHandler', 'goog.events.EventTarget', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.userAgent'], {});
goog.addDependency('events/imehandler_test.js', ['goog.events.ImeHandlerTest'], ['goog.array', 'goog.dom', 'goog.events', 'goog.events.ImeHandler', 'goog.events.KeyCodes', 'goog.object', 'goog.string', 'goog.testing.PropertyReplacer', 'goog.testing.events', 'goog.testing.events.Event', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('events/inputhandler.js', ['goog.events.InputHandler', 'goog.events.InputHandler.EventType'], ['goog.Timer', 'goog.dom.TagName', 'goog.events.BrowserEvent', 'goog.events.EventHandler', 'goog.events.EventTarget', 'goog.events.KeyCodes', 'goog.userAgent'], {});
goog.addDependency('events/inputhandler_test.js', ['goog.events.InputHandlerTest'], ['goog.dom', 'goog.events.EventHandler', 'goog.events.EventType', 'goog.events.InputHandler', 'goog.events.KeyCodes', 'goog.testing.events', 'goog.testing.events.Event', 'goog.testing.recordFunction', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('events/keycodes.js', ['goog.events.KeyCodes'], ['goog.userAgent'], {});
goog.addDependency('events/keycodes_test.js', ['goog.events.KeyCodesTest'], ['goog.events.BrowserEvent', 'goog.events.KeyCodes', 'goog.object', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('events/keyhandler.js', ['goog.events.KeyEvent', 'goog.events.KeyHandler', 'goog.events.KeyHandler.EventType'], ['goog.events', 'goog.events.BrowserEvent', 'goog.events.EventTarget', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.userAgent'], {});
goog.addDependency('events/keyhandler_test.js', ['goog.events.KeyEventTest'], ['goog.dom', 'goog.dom.TagName', 'goog.events', 'goog.events.BrowserEvent', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.events.KeyHandler', 'goog.testing.events', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('events/keynames.js', ['goog.events.KeyNames'], [], {});
goog.addDependency('events/keys.js', ['goog.events.Keys'], [], {'lang': 'es5'});
goog.addDependency('events/listenable.js', ['goog.events.Listenable', 'goog.events.ListenableKey'], ['goog.events.EventId'], {});
goog.addDependency('events/listenable_test.js', ['goog.events.ListenableTest'], ['goog.events.Listenable', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('events/listener.js', ['goog.events.Listener'], ['goog.events.ListenableKey'], {});
goog.addDependency('events/listenermap.js', ['goog.events.ListenerMap'], ['goog.array', 'goog.events.Listener', 'goog.object'], {});
goog.addDependency('events/listenermap_test.js', ['goog.events.ListenerMapTest'], ['goog.dispose', 'goog.events', 'goog.events.EventId', 'goog.events.EventTarget', 'goog.events.ListenerMap', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('events/mousewheelhandler.js', ['goog.events.MouseWheelEvent', 'goog.events.MouseWheelHandler', 'goog.events.MouseWheelHandler.EventType'], ['goog.dom', 'goog.events', 'goog.events.BrowserEvent', 'goog.events.EventTarget', 'goog.math', 'goog.style', 'goog.userAgent'], {});
goog.addDependency('events/mousewheelhandler_test.js', ['goog.events.MouseWheelHandlerTest'], ['goog.dom', 'goog.events', 'goog.events.BrowserEvent', 'goog.events.MouseWheelEvent', 'goog.events.MouseWheelHandler', 'goog.functions', 'goog.string', 'goog.testing.PropertyReplacer', 'goog.testing.events', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('events/onlinehandler.js', ['goog.events.OnlineHandler', 'goog.events.OnlineHandler.EventType'], ['goog.Timer', 'goog.events.BrowserFeature', 'goog.events.EventHandler', 'goog.events.EventTarget', 'goog.events.EventType', 'goog.net.NetworkStatusMonitor'], {});
goog.addDependency('events/onlinelistener_test.js', ['goog.events.OnlineHandlerTest'], ['goog.events', 'goog.events.BrowserFeature', 'goog.events.Event', 'goog.events.EventHandler', 'goog.events.OnlineHandler', 'goog.net.NetworkStatusMonitor', 'goog.testing.MockClock', 'goog.testing.PropertyReplacer', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('events/pastehandler.js', ['goog.events.PasteHandler', 'goog.events.PasteHandler.EventType', 'goog.events.PasteHandler.State'], ['goog.Timer', 'goog.async.ConditionalDelay', 'goog.events.BrowserEvent', 'goog.events.EventHandler', 'goog.events.EventTarget', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.log', 'goog.userAgent'], {});
goog.addDependency('events/pastehandler_test.js', ['goog.events.PasteHandlerTest'], ['goog.dom', 'goog.events', 'goog.events.BrowserEvent', 'goog.events.EventTarget', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.events.PasteHandler', 'goog.testing.MockClock', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('events/wheelevent.js', ['goog.events.WheelEvent'], ['goog.asserts', 'goog.events.BrowserEvent'], {});
goog.addDependency('events/wheelhandler.js', ['goog.events.WheelHandler'], ['goog.dom', 'goog.events', 'goog.events.EventTarget', 'goog.events.WheelEvent', 'goog.style', 'goog.userAgent', 'goog.userAgent.product', 'goog.userAgent.product.isVersion'], {});
goog.addDependency('events/wheelhandler_test.js', ['goog.events.WheelHandlerTest'], ['goog.dom', 'goog.events', 'goog.events.BrowserEvent', 'goog.events.WheelEvent', 'goog.events.WheelHandler', 'goog.string', 'goog.testing.PropertyReplacer', 'goog.testing.events', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('format/emailaddress.js', ['goog.format.EmailAddress'], ['goog.string'], {});
goog.addDependency('format/emailaddress_test.js', ['goog.format.EmailAddressTest'], ['goog.array', 'goog.format.EmailAddress', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('format/format.js', ['goog.format'], ['goog.i18n.GraphemeBreak', 'goog.string', 'goog.userAgent'], {});
goog.addDependency('format/format_test.js', ['goog.formatTest'], ['goog.dom', 'goog.dom.TagName', 'goog.format', 'goog.string', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('format/htmlprettyprinter.js', ['goog.format.HtmlPrettyPrinter', 'goog.format.HtmlPrettyPrinter.Buffer'], ['goog.dom.TagName', 'goog.object', 'goog.string.StringBuffer'], {});
goog.addDependency('format/htmlprettyprinter_test.js', ['goog.format.HtmlPrettyPrinterTest'], ['goog.format.HtmlPrettyPrinter', 'goog.testing.MockClock', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('format/internationalizedemailaddress.js', ['goog.format.InternationalizedEmailAddress'], ['goog.format.EmailAddress', 'goog.string'], {});
goog.addDependency('format/internationalizedemailaddress_test.js', ['goog.format.InternationalizedEmailAddressTest'], ['goog.array', 'goog.format.InternationalizedEmailAddress', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('format/jsonprettyprinter.js', ['goog.format.JsonPrettyPrinter', 'goog.format.JsonPrettyPrinter.SafeHtmlDelimiters', 'goog.format.JsonPrettyPrinter.TextDelimiters'], ['goog.html.SafeHtml', 'goog.json', 'goog.json.Serializer', 'goog.string', 'goog.string.format'], {});
goog.addDependency('format/jsonprettyprinter_test.js', ['goog.format.JsonPrettyPrinterTest'], ['goog.format.JsonPrettyPrinter', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('fs/entry.js', ['goog.fs.DirectoryEntry', 'goog.fs.DirectoryEntry.Behavior', 'goog.fs.Entry', 'goog.fs.FileEntry'], [], {});
goog.addDependency('fs/entryimpl.js', ['goog.fs.DirectoryEntryImpl', 'goog.fs.EntryImpl', 'goog.fs.FileEntryImpl'], ['goog.array', 'goog.async.Deferred', 'goog.fs.DirectoryEntry', 'goog.fs.Entry', 'goog.fs.Error', 'goog.fs.FileEntry', 'goog.fs.FileWriter', 'goog.functions', 'goog.string'], {});
goog.addDependency('fs/error.js', ['goog.fs.DOMErrorLike', 'goog.fs.Error', 'goog.fs.Error.ErrorCode'], ['goog.asserts', 'goog.debug.Error', 'goog.object', 'goog.string'], {});
goog.addDependency('fs/filereader.js', ['goog.fs.FileReader', 'goog.fs.FileReader.EventType', 'goog.fs.FileReader.ReadyState'], ['goog.async.Deferred', 'goog.events.EventTarget', 'goog.fs.Error', 'goog.fs.ProgressEvent'], {});
goog.addDependency('fs/filesaver.js', ['goog.fs.FileSaver', 'goog.fs.FileSaver.EventType', 'goog.fs.FileSaver.ReadyState'], ['goog.events.EventTarget', 'goog.fs.Error', 'goog.fs.ProgressEvent'], {});
goog.addDependency('fs/filesystem.js', ['goog.fs.FileSystem'], [], {});
goog.addDependency('fs/filesystemimpl.js', ['goog.fs.FileSystemImpl'], ['goog.fs.DirectoryEntryImpl', 'goog.fs.FileSystem'], {});
goog.addDependency('fs/filewriter.js', ['goog.fs.FileWriter'], ['goog.fs.Error', 'goog.fs.FileSaver'], {});
goog.addDependency('fs/fs.js', ['goog.fs'], ['goog.array', 'goog.async.Deferred', 'goog.fs.Error', 'goog.fs.FileReader', 'goog.fs.FileSystemImpl', 'goog.fs.url', 'goog.userAgent'], {});
goog.addDependency('fs/fs_test.js', ['goog.fsTest'], ['goog.Promise', 'goog.array', 'goog.dom', 'goog.dom.TagName', 'goog.events', 'goog.fs', 'goog.fs.DirectoryEntry', 'goog.fs.Error', 'goog.fs.FileReader', 'goog.fs.FileSaver', 'goog.string', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('fs/progressevent.js', ['goog.fs.ProgressEvent'], ['goog.events.Event'], {});
goog.addDependency('fs/url.js', ['goog.fs.url'], [], {'lang': 'es6'});
goog.addDependency('fs/url_test.js', ['goog.urlTest'], ['goog.fs.url', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('functions/functions.js', ['goog.functions'], [], {'lang': 'es6'});
goog.addDependency('functions/functions_test.js', ['goog.functionsTest'], ['goog.array', 'goog.functions', 'goog.testing.MockClock', 'goog.testing.PropertyReplacer', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('fx/abstractdragdrop.js', ['goog.fx.AbstractDragDrop', 'goog.fx.AbstractDragDrop.EventType', 'goog.fx.DragDropEvent', 'goog.fx.DragDropItem'], ['goog.array', 'goog.asserts', 'goog.dom', 'goog.dom.classlist', 'goog.events', 'goog.events.Event', 'goog.events.EventHandler', 'goog.events.EventTarget', 'goog.events.EventType', 'goog.fx.Dragger', 'goog.math.Box', 'goog.math.Coordinate', 'goog.style'], {});
goog.addDependency('fx/abstractdragdrop_test.js', ['goog.fx.AbstractDragDropTest'], ['goog.array', 'goog.dom', 'goog.dom.TagName', 'goog.events', 'goog.events.EventType', 'goog.functions', 'goog.fx.AbstractDragDrop', 'goog.fx.DragDropItem', 'goog.math.Box', 'goog.math.Coordinate', 'goog.style', 'goog.testing.events', 'goog.testing.events.Event', 'goog.testing.jsunit'], {'lang': 'es6'});
goog.addDependency('fx/anim/anim.js', ['goog.fx.anim', 'goog.fx.anim.Animated'], ['goog.async.AnimationDelay', 'goog.async.Delay', 'goog.object'], {});
goog.addDependency('fx/anim/anim_test.js', ['goog.fx.animTest'], ['goog.async.AnimationDelay', 'goog.async.Delay', 'goog.events', 'goog.functions', 'goog.fx.Animation', 'goog.fx.anim', 'goog.object', 'goog.testing.MockClock', 'goog.testing.PropertyReplacer', 'goog.testing.recordFunction', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('fx/animation.js', ['goog.fx.Animation', 'goog.fx.Animation.EventType', 'goog.fx.Animation.State', 'goog.fx.AnimationEvent'], ['goog.array', 'goog.asserts', 'goog.events.Event', 'goog.fx.Transition', 'goog.fx.TransitionBase', 'goog.fx.anim', 'goog.fx.anim.Animated'], {});
goog.addDependency('fx/animation_test.js', ['goog.fx.AnimationTest'], ['goog.events', 'goog.fx.Animation', 'goog.testing.MockClock', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('fx/animationqueue.js', ['goog.fx.AnimationParallelQueue', 'goog.fx.AnimationQueue', 'goog.fx.AnimationSerialQueue'], ['goog.array', 'goog.asserts', 'goog.events', 'goog.fx.Animation', 'goog.fx.Transition', 'goog.fx.TransitionBase'], {});
goog.addDependency('fx/animationqueue_test.js', ['goog.fx.AnimationQueueTest'], ['goog.events', 'goog.fx.Animation', 'goog.fx.AnimationParallelQueue', 'goog.fx.AnimationSerialQueue', 'goog.fx.Transition', 'goog.fx.anim', 'goog.testing.MockClock', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('fx/css3/fx.js', ['goog.fx.css3'], ['goog.fx.css3.Transition'], {});
goog.addDependency('fx/css3/transition.js', ['goog.fx.css3.Transition'], ['goog.Timer', 'goog.asserts', 'goog.fx.TransitionBase', 'goog.style', 'goog.style.transition'], {});
goog.addDependency('fx/css3/transition_test.js', ['goog.fx.css3.TransitionTest'], ['goog.dispose', 'goog.dom', 'goog.dom.TagName', 'goog.events', 'goog.fx.Transition', 'goog.fx.css3.Transition', 'goog.style.transition', 'goog.testing.MockClock', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('fx/cssspriteanimation.js', ['goog.fx.CssSpriteAnimation'], ['goog.fx.Animation'], {});
goog.addDependency('fx/cssspriteanimation_test.js', ['goog.fx.CssSpriteAnimationTest'], ['goog.fx.CssSpriteAnimation', 'goog.math.Box', 'goog.math.Size', 'goog.testing.MockClock', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('fx/dom.js', ['goog.fx.dom', 'goog.fx.dom.BgColorTransform', 'goog.fx.dom.ColorTransform', 'goog.fx.dom.Fade', 'goog.fx.dom.FadeIn', 'goog.fx.dom.FadeInAndShow', 'goog.fx.dom.FadeOut', 'goog.fx.dom.FadeOutAndHide', 'goog.fx.dom.PredefinedEffect', 'goog.fx.dom.Resize', 'goog.fx.dom.ResizeHeight', 'goog.fx.dom.ResizeWidth', 'goog.fx.dom.Scroll', 'goog.fx.dom.Slide', 'goog.fx.dom.SlideFrom', 'goog.fx.dom.Swipe'], ['goog.color', 'goog.events', 'goog.fx.Animation', 'goog.fx.Transition', 'goog.style', 'goog.style.bidi'], {});
goog.addDependency('fx/dragdrop.js', ['goog.fx.DragDrop'], ['goog.fx.AbstractDragDrop', 'goog.fx.DragDropItem'], {});
goog.addDependency('fx/dragdropgroup.js', ['goog.fx.DragDropGroup'], ['goog.dom', 'goog.fx.AbstractDragDrop', 'goog.fx.DragDropItem'], {});
goog.addDependency('fx/dragdropgroup_test.js', ['goog.fx.DragDropGroupTest'], ['goog.events', 'goog.fx.DragDropGroup', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('fx/dragger.js', ['goog.fx.DragEvent', 'goog.fx.Dragger', 'goog.fx.Dragger.EventType'], ['goog.dom', 'goog.dom.TagName', 'goog.events', 'goog.events.Event', 'goog.events.EventHandler', 'goog.events.EventTarget', 'goog.events.EventType', 'goog.math.Coordinate', 'goog.math.Rect', 'goog.style', 'goog.style.bidi', 'goog.userAgent'], {});
goog.addDependency('fx/dragger_test.js', ['goog.fx.DraggerTest'], ['goog.dom', 'goog.dom.TagName', 'goog.events', 'goog.events.BrowserEvent', 'goog.events.Event', 'goog.events.EventType', 'goog.fx.Dragger', 'goog.math.Rect', 'goog.style.bidi', 'goog.testing.StrictMock', 'goog.testing.events', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('fx/draglistgroup.js', ['goog.fx.DragListDirection', 'goog.fx.DragListGroup', 'goog.fx.DragListGroup.EventType', 'goog.fx.DragListGroupEvent', 'goog.fx.DragListPermission'], ['goog.array', 'goog.asserts', 'goog.dom', 'goog.dom.classlist', 'goog.events', 'goog.events.Event', 'goog.events.EventHandler', 'goog.events.EventId', 'goog.events.EventTarget', 'goog.events.EventType', 'goog.fx.Dragger', 'goog.math.Coordinate', 'goog.string', 'goog.style'], {});
goog.addDependency('fx/draglistgroup_test.js', ['goog.fx.DragListGroupTest'], ['goog.array', 'goog.dom', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.events', 'goog.events.BrowserEvent', 'goog.events.BrowserFeature', 'goog.events.Event', 'goog.events.EventType', 'goog.fx.DragEvent', 'goog.fx.DragListDirection', 'goog.fx.DragListGroup', 'goog.fx.DragListPermission', 'goog.fx.Dragger', 'goog.math.Coordinate', 'goog.object', 'goog.style', 'goog.testing.events', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('fx/dragscrollsupport.js', ['goog.fx.DragScrollSupport'], ['goog.Disposable', 'goog.Timer', 'goog.dom', 'goog.events.EventHandler', 'goog.events.EventType', 'goog.math.Coordinate', 'goog.style'], {});
goog.addDependency('fx/dragscrollsupport_test.js', ['goog.fx.DragScrollSupportTest'], ['goog.fx.DragScrollSupport', 'goog.math.Coordinate', 'goog.math.Rect', 'goog.testing.MockClock', 'goog.testing.events', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('fx/easing.js', ['goog.fx.easing'], [], {});
goog.addDependency('fx/easing_test.js', ['goog.fx.easingTest'], ['goog.fx.easing', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('fx/fx.js', ['goog.fx'], ['goog.asserts', 'goog.fx.Animation', 'goog.fx.Animation.EventType', 'goog.fx.Animation.State', 'goog.fx.AnimationEvent', 'goog.fx.Transition.EventType', 'goog.fx.easing'], {});
goog.addDependency('fx/fx_test.js', ['goog.fxTest'], ['goog.fx.Animation', 'goog.object', 'goog.testing.MockClock', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('fx/transition.js', ['goog.fx.Transition', 'goog.fx.Transition.EventType'], [], {});
goog.addDependency('fx/transitionbase.js', ['goog.fx.TransitionBase', 'goog.fx.TransitionBase.State'], ['goog.events.EventTarget', 'goog.fx.Transition'], {});
goog.addDependency('goog.js', [], [], {'lang': 'es6', 'module': 'es6'});
goog.addDependency('graphics/abstractgraphics.js', ['goog.graphics.AbstractGraphics'], ['goog.dom', 'goog.graphics.AffineTransform', 'goog.graphics.Element', 'goog.graphics.EllipseElement', 'goog.graphics.Fill', 'goog.graphics.Font', 'goog.graphics.GroupElement', 'goog.graphics.Path', 'goog.graphics.PathElement', 'goog.graphics.RectElement', 'goog.graphics.Stroke', 'goog.graphics.StrokeAndFillElement', 'goog.graphics.TextElement', 'goog.math.Coordinate', 'goog.math.Size', 'goog.style', 'goog.ui.Component'], {});
goog.addDependency('graphics/affinetransform.js', ['goog.graphics.AffineTransform'], [], {'lang': 'es6'});
goog.addDependency('graphics/affinetransform_test.js', ['goog.graphics.AffineTransformTest'], ['goog.array', 'goog.graphics', 'goog.graphics.AffineTransform', 'goog.math', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('graphics/canvaselement.js', ['goog.graphics.CanvasEllipseElement', 'goog.graphics.CanvasGroupElement', 'goog.graphics.CanvasImageElement', 'goog.graphics.CanvasPathElement', 'goog.graphics.CanvasRectElement', 'goog.graphics.CanvasTextElement'], ['goog.array', 'goog.dom', 'goog.dom.TagName', 'goog.dom.safe', 'goog.graphics.EllipseElement', 'goog.graphics.Font', 'goog.graphics.GroupElement', 'goog.graphics.ImageElement', 'goog.graphics.Path', 'goog.graphics.PathElement', 'goog.graphics.RectElement', 'goog.graphics.TextElement', 'goog.html.SafeHtml', 'goog.html.uncheckedconversions', 'goog.math', 'goog.string', 'goog.string.Const'], {});
goog.addDependency('graphics/canvasgraphics.js', ['goog.graphics.CanvasGraphics'], ['goog.dom.TagName', 'goog.events.EventType', 'goog.graphics.AbstractGraphics', 'goog.graphics.CanvasEllipseElement', 'goog.graphics.CanvasGroupElement', 'goog.graphics.CanvasImageElement', 'goog.graphics.CanvasPathElement', 'goog.graphics.CanvasRectElement', 'goog.graphics.CanvasTextElement', 'goog.graphics.Font', 'goog.graphics.SolidFill', 'goog.math.Size', 'goog.style'], {});
goog.addDependency('graphics/canvasgraphics_test.js', ['goog.graphics.CanvasGraphicsTest'], ['goog.dom', 'goog.graphics.CanvasGraphics', 'goog.graphics.SolidFill', 'goog.graphics.Stroke', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('graphics/element.js', ['goog.graphics.Element'], ['goog.asserts', 'goog.events', 'goog.events.EventTarget', 'goog.events.Listenable', 'goog.graphics.AffineTransform', 'goog.math'], {});
goog.addDependency('graphics/ellipseelement.js', ['goog.graphics.EllipseElement'], ['goog.graphics.StrokeAndFillElement'], {});
goog.addDependency('graphics/ext/coordinates.js', ['goog.graphics.ext.coordinates'], ['goog.string'], {});
goog.addDependency('graphics/ext/coordinates_test.js', ['goog.graphics.ext.coordinatesTest'], ['goog.graphics', 'goog.graphics.ext.coordinates', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('graphics/ext/element.js', ['goog.graphics.ext.Element'], ['goog.events.EventTarget', 'goog.functions', 'goog.graphics.ext.coordinates'], {});
goog.addDependency('graphics/ext/element_test.js', ['goog.graphics.ext.ElementTest'], ['goog.graphics', 'goog.graphics.ext', 'goog.testing.StrictMock', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('graphics/ext/ellipse.js', ['goog.graphics.ext.Ellipse'], ['goog.graphics.ext.StrokeAndFillElement'], {});
goog.addDependency('graphics/ext/ext.js', ['goog.graphics.ext'], ['goog.graphics.ext.Ellipse', 'goog.graphics.ext.Graphics', 'goog.graphics.ext.Group', 'goog.graphics.ext.Image', 'goog.graphics.ext.Rectangle', 'goog.graphics.ext.Shape', 'goog.graphics.ext.coordinates'], {});
goog.addDependency('graphics/ext/graphics.js', ['goog.graphics.ext.Graphics'], ['goog.events', 'goog.events.EventType', 'goog.graphics', 'goog.graphics.ext.Group'], {});
goog.addDependency('graphics/ext/group.js', ['goog.graphics.ext.Group'], ['goog.array', 'goog.graphics.ext.Element'], {});
goog.addDependency('graphics/ext/image.js', ['goog.graphics.ext.Image'], ['goog.graphics.ext.Element'], {});
goog.addDependency('graphics/ext/path.js', ['goog.graphics.ext.Path'], ['goog.graphics.AffineTransform', 'goog.graphics.Path', 'goog.math.Rect'], {});
goog.addDependency('graphics/ext/path_test.js', ['goog.graphics.ext.PathTest'], ['goog.graphics', 'goog.graphics.ext.Path', 'goog.math.Rect', 'goog.testing.graphics', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('graphics/ext/rectangle.js', ['goog.graphics.ext.Rectangle'], ['goog.graphics.ext.StrokeAndFillElement'], {});
goog.addDependency('graphics/ext/shape.js', ['goog.graphics.ext.Shape'], ['goog.graphics.ext.StrokeAndFillElement'], {});
goog.addDependency('graphics/ext/strokeandfillelement.js', ['goog.graphics.ext.StrokeAndFillElement'], ['goog.graphics.ext.Element'], {});
goog.addDependency('graphics/fill.js', ['goog.graphics.Fill'], [], {});
goog.addDependency('graphics/font.js', ['goog.graphics.Font'], [], {});
goog.addDependency('graphics/graphics.js', ['goog.graphics'], ['goog.dom', 'goog.graphics.CanvasGraphics', 'goog.graphics.SvgGraphics', 'goog.graphics.VmlGraphics', 'goog.userAgent'], {});
goog.addDependency('graphics/groupelement.js', ['goog.graphics.GroupElement'], ['goog.graphics.Element'], {});
goog.addDependency('graphics/imageelement.js', ['goog.graphics.ImageElement'], ['goog.graphics.Element'], {});
goog.addDependency('graphics/lineargradient.js', ['goog.graphics.LinearGradient'], ['goog.asserts', 'goog.graphics.Fill'], {});
goog.addDependency('graphics/path.js', ['goog.graphics.Path', 'goog.graphics.Path.Segment'], ['goog.array', 'goog.graphics.AffineTransform', 'goog.math'], {});
goog.addDependency('graphics/path_test.js', ['goog.graphics.PathTest'], ['goog.array', 'goog.graphics.AffineTransform', 'goog.graphics.Path', 'goog.testing.graphics', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('graphics/pathelement.js', ['goog.graphics.PathElement'], ['goog.graphics.StrokeAndFillElement'], {});
goog.addDependency('graphics/paths.js', ['goog.graphics.paths'], ['goog.graphics.Path', 'goog.math.Coordinate'], {});
goog.addDependency('graphics/paths_test.js', ['goog.graphics.pathsTest'], ['goog.dom', 'goog.graphics', 'goog.graphics.paths', 'goog.math.Coordinate', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('graphics/rectelement.js', ['goog.graphics.RectElement'], ['goog.graphics.StrokeAndFillElement'], {});
goog.addDependency('graphics/solidfill.js', ['goog.graphics.SolidFill'], ['goog.graphics.Fill'], {});
goog.addDependency('graphics/solidfill_test.js', ['goog.graphics.SolidFillTest'], ['goog.graphics.SolidFill', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('graphics/stroke.js', ['goog.graphics.Stroke'], [], {});
goog.addDependency('graphics/strokeandfillelement.js', ['goog.graphics.StrokeAndFillElement'], ['goog.graphics.Element'], {});
goog.addDependency('graphics/svgelement.js', ['goog.graphics.SvgEllipseElement', 'goog.graphics.SvgGroupElement', 'goog.graphics.SvgImageElement', 'goog.graphics.SvgPathElement', 'goog.graphics.SvgRectElement', 'goog.graphics.SvgTextElement'], ['goog.dom', 'goog.graphics.EllipseElement', 'goog.graphics.GroupElement', 'goog.graphics.ImageElement', 'goog.graphics.PathElement', 'goog.graphics.RectElement', 'goog.graphics.TextElement'], {});
goog.addDependency('graphics/svggraphics.js', ['goog.graphics.SvgGraphics'], ['goog.Timer', 'goog.dom', 'goog.events.EventHandler', 'goog.events.EventType', 'goog.graphics.AbstractGraphics', 'goog.graphics.Font', 'goog.graphics.LinearGradient', 'goog.graphics.Path', 'goog.graphics.SolidFill', 'goog.graphics.Stroke', 'goog.graphics.SvgEllipseElement', 'goog.graphics.SvgGroupElement', 'goog.graphics.SvgImageElement', 'goog.graphics.SvgPathElement', 'goog.graphics.SvgRectElement', 'goog.graphics.SvgTextElement', 'goog.math', 'goog.math.Size', 'goog.style', 'goog.userAgent'], {});
goog.addDependency('graphics/svggraphics_test.js', ['goog.graphics.SvgGraphicsTest'], ['goog.dom', 'goog.graphics.AffineTransform', 'goog.graphics.SolidFill', 'goog.graphics.SvgGraphics', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('graphics/textelement.js', ['goog.graphics.TextElement'], ['goog.graphics.StrokeAndFillElement'], {});
goog.addDependency('graphics/vmlelement.js', ['goog.graphics.VmlEllipseElement', 'goog.graphics.VmlGroupElement', 'goog.graphics.VmlImageElement', 'goog.graphics.VmlPathElement', 'goog.graphics.VmlRectElement', 'goog.graphics.VmlTextElement'], ['goog.dom', 'goog.graphics.EllipseElement', 'goog.graphics.GroupElement', 'goog.graphics.ImageElement', 'goog.graphics.PathElement', 'goog.graphics.RectElement', 'goog.graphics.TextElement'], {});
goog.addDependency('graphics/vmlgraphics.js', ['goog.graphics.VmlGraphics'], ['goog.array', 'goog.dom.TagName', 'goog.dom.safe', 'goog.events', 'goog.events.EventHandler', 'goog.events.EventType', 'goog.graphics.AbstractGraphics', 'goog.graphics.Font', 'goog.graphics.LinearGradient', 'goog.graphics.Path', 'goog.graphics.SolidFill', 'goog.graphics.VmlEllipseElement', 'goog.graphics.VmlGroupElement', 'goog.graphics.VmlImageElement', 'goog.graphics.VmlPathElement', 'goog.graphics.VmlRectElement', 'goog.graphics.VmlTextElement', 'goog.html.uncheckedconversions', 'goog.math', 'goog.math.Size', 'goog.reflect', 'goog.string', 'goog.string.Const', 'goog.style', 'goog.userAgent'], {});
goog.addDependency('history/event.js', ['goog.history.Event'], ['goog.events.Event', 'goog.history.EventType'], {});
goog.addDependency('history/eventtype.js', ['goog.history.EventType'], [], {});
goog.addDependency('history/history.js', ['goog.History', 'goog.History.Event', 'goog.History.EventType'], ['goog.Timer', 'goog.asserts', 'goog.dom', 'goog.dom.InputType', 'goog.dom.safe', 'goog.events.Event', 'goog.events.EventHandler', 'goog.events.EventTarget', 'goog.events.EventType', 'goog.history.Event', 'goog.history.EventType', 'goog.html.SafeHtml', 'goog.html.TrustedResourceUrl', 'goog.html.uncheckedconversions', 'goog.labs.userAgent.device', 'goog.memoize', 'goog.string', 'goog.string.Const', 'goog.userAgent'], {});
goog.addDependency('history/history_test.js', ['goog.HistoryTest'], ['goog.History', 'goog.dispose', 'goog.dom', 'goog.html.TrustedResourceUrl', 'goog.string.Const', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('history/html5history.js', ['goog.history.Html5History', 'goog.history.Html5History.TokenTransformer'], ['goog.asserts', 'goog.events', 'goog.events.EventTarget', 'goog.events.EventType', 'goog.history.Event'], {});
goog.addDependency('history/html5history_test.js', ['goog.history.Html5HistoryTest'], ['goog.Timer', 'goog.events', 'goog.events.EventType', 'goog.history.EventType', 'goog.history.Html5History', 'goog.testing.MockControl', 'goog.testing.mockmatchers', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('html/cssspecificity.js', ['goog.html.CssSpecificity'], ['goog.userAgent', 'goog.userAgent.product'], {'module': 'goog'});
goog.addDependency('html/cssspecificity_test.js', ['goog.html.CssSpecificityTest'], ['goog.html.CssSpecificity', 'goog.testing.testSuite', 'goog.userAgent', 'goog.userAgent.product'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('html/flash.js', ['goog.html.flash'], ['goog.asserts', 'goog.html.SafeHtml'], {});
goog.addDependency('html/flash_test.js', ['goog.html.flashTest'], ['goog.html.SafeHtml', 'goog.html.TrustedResourceUrl', 'goog.html.flash', 'goog.string.Const', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('html/legacyconversions.js', ['goog.html.legacyconversions'], ['goog.html.SafeHtml', 'goog.html.SafeScript', 'goog.html.SafeStyle', 'goog.html.SafeStyleSheet', 'goog.html.SafeUrl', 'goog.html.TrustedResourceUrl'], {});
goog.addDependency('html/legacyconversions_test.js', ['goog.html.legacyconversionsTest'], ['goog.html.SafeHtml', 'goog.html.SafeScript', 'goog.html.SafeStyle', 'goog.html.SafeStyleSheet', 'goog.html.SafeUrl', 'goog.html.TrustedResourceUrl', 'goog.html.legacyconversions', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('html/safehtml.js', ['goog.html.SafeHtml'], ['goog.array', 'goog.asserts', 'goog.dom.TagName', 'goog.dom.tags', 'goog.html.SafeScript', 'goog.html.SafeStyle', 'goog.html.SafeStyleSheet', 'goog.html.SafeUrl', 'goog.html.TrustedResourceUrl', 'goog.html.trustedtypes', 'goog.i18n.bidi.Dir', 'goog.i18n.bidi.DirectionalString', 'goog.labs.userAgent.browser', 'goog.object', 'goog.string.Const', 'goog.string.TypedString', 'goog.string.internal'], {});
goog.addDependency('html/safehtml_test.js', ['goog.html.safeHtmlTest'], ['goog.html.SafeHtml', 'goog.html.SafeScript', 'goog.html.SafeStyle', 'goog.html.SafeStyleSheet', 'goog.html.SafeUrl', 'goog.html.TrustedResourceUrl', 'goog.html.testing', 'goog.html.trustedtypes', 'goog.i18n.bidi.Dir', 'goog.labs.userAgent.browser', 'goog.object', 'goog.string.Const', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('html/safehtmlformatter.js', ['goog.html.SafeHtmlFormatter'], ['goog.asserts', 'goog.dom.tags', 'goog.html.SafeHtml', 'goog.string'], {});
goog.addDependency('html/safehtmlformatter_test.js', ['goog.html.safeHtmlFormatterTest'], ['goog.html.SafeHtml', 'goog.html.SafeHtmlFormatter', 'goog.html.SafeUrl', 'goog.string', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('html/safescript.js', ['goog.html.SafeScript'], ['goog.asserts', 'goog.html.trustedtypes', 'goog.string.Const', 'goog.string.TypedString'], {});
goog.addDependency('html/safescript_test.js', ['goog.html.safeScriptTest'], ['goog.html.SafeScript', 'goog.html.trustedtypes', 'goog.object', 'goog.string.Const', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('html/safestyle.js', ['goog.html.SafeStyle'], ['goog.array', 'goog.asserts', 'goog.html.SafeUrl', 'goog.string.Const', 'goog.string.TypedString', 'goog.string.internal'], {'lang': 'es5'});
goog.addDependency('html/safestyle_test.js', ['goog.html.safeStyleTest'], ['goog.html.SafeStyle', 'goog.html.SafeUrl', 'goog.object', 'goog.string.Const', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('html/safestylesheet.js', ['goog.html.SafeStyleSheet'], ['goog.array', 'goog.asserts', 'goog.html.SafeStyle', 'goog.object', 'goog.string.Const', 'goog.string.TypedString', 'goog.string.internal'], {});
goog.addDependency('html/safestylesheet_test.js', ['goog.html.safeStyleSheetTest'], ['goog.html.SafeStyle', 'goog.html.SafeStyleSheet', 'goog.object', 'goog.string.Const', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('html/safeurl.js', ['goog.html.SafeUrl'], ['goog.asserts', 'goog.fs.url', 'goog.html.TrustedResourceUrl', 'goog.i18n.bidi.Dir', 'goog.i18n.bidi.DirectionalString', 'goog.string.Const', 'goog.string.TypedString', 'goog.string.internal'], {});
goog.addDependency('html/safeurl_test.js', ['goog.html.safeUrlTest'], ['goog.html.SafeUrl', 'goog.html.TrustedResourceUrl', 'goog.html.safeUrlTestVectors', 'goog.i18n.bidi.Dir', 'goog.object', 'goog.string.Const', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('html/safeurl_test_vectors.js', ['goog.html.safeUrlTestVectors'], [], {});
goog.addDependency('html/sanitizer/attributewhitelist.js', ['goog.html.sanitizer.AttributeSanitizedWhitelist', 'goog.html.sanitizer.AttributeWhitelist'], [], {});
goog.addDependency('html/sanitizer/csspropertysanitizer.js', ['goog.html.sanitizer.CssPropertySanitizer'], ['goog.asserts', 'goog.html.SafeUrl', 'goog.object', 'goog.string'], {'module': 'goog'});
goog.addDependency('html/sanitizer/csspropertysanitizer_test.js', ['goog.html.sanitizer.CssPropertySanitizerTest'], ['goog.functions', 'goog.html.SafeUrl', 'goog.html.sanitizer.CssPropertySanitizer', 'goog.html.sanitizer.noclobber', 'goog.testing.testSuite', 'goog.userAgent.product'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('html/sanitizer/csssanitizer.js', ['goog.html.sanitizer.CssSanitizer'], ['goog.array', 'goog.dom', 'goog.dom.TagName', 'goog.dom.safe', 'goog.html.CssSpecificity', 'goog.html.SafeStyle', 'goog.html.SafeStyleSheet', 'goog.html.SafeUrl', 'goog.html.sanitizer.CssPropertySanitizer', 'goog.html.sanitizer.noclobber', 'goog.html.uncheckedconversions', 'goog.object', 'goog.string', 'goog.string.Const', 'goog.userAgent', 'goog.userAgent.product'], {});
goog.addDependency('html/sanitizer/csssanitizer_test.js', ['goog.html.CssSanitizerTest'], ['goog.array', 'goog.html.SafeStyle', 'goog.html.SafeStyleSheet', 'goog.html.SafeUrl', 'goog.html.sanitizer.CssSanitizer', 'goog.html.testing', 'goog.string', 'goog.testing.dom', 'goog.testing.testSuite', 'goog.userAgent', 'goog.userAgent.product', 'goog.userAgent.product.isVersion'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('html/sanitizer/elementweakmap.js', ['goog.html.sanitizer.ElementWeakMap'], ['goog.html.sanitizer.noclobber'], {'module': 'goog'});
goog.addDependency('html/sanitizer/elementweakmap_test.js', ['goog.html.sanitizer.ElementWeakMapTest'], ['goog.html.sanitizer.ElementWeakMap', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('html/sanitizer/html_test_vectors.js', ['goog.html.htmlTestVectors'], [], {'lang': 'es5'});
goog.addDependency('html/sanitizer/htmlsanitizer.js', ['goog.html.sanitizer.HtmlSanitizer', 'goog.html.sanitizer.HtmlSanitizer.Builder', 'goog.html.sanitizer.HtmlSanitizerAttributePolicy', 'goog.html.sanitizer.HtmlSanitizerPolicy', 'goog.html.sanitizer.HtmlSanitizerPolicyContext', 'goog.html.sanitizer.HtmlSanitizerPolicyHints', 'goog.html.sanitizer.HtmlSanitizerUrlPolicy'], ['goog.array', 'goog.asserts', 'goog.dom', 'goog.dom.TagName', 'goog.functions', 'goog.html.SafeHtml', 'goog.html.SafeStyle', 'goog.html.SafeStyleSheet', 'goog.html.SafeUrl', 'goog.html.sanitizer.AttributeSanitizedWhitelist', 'goog.html.sanitizer.AttributeWhitelist', 'goog.html.sanitizer.CssSanitizer', 'goog.html.sanitizer.SafeDomTreeProcessor', 'goog.html.sanitizer.TagBlacklist', 'goog.html.sanitizer.TagWhitelist', 'goog.html.sanitizer.noclobber', 'goog.html.uncheckedconversions', 'goog.object', 'goog.string', 'goog.string.Const'], {'lang': 'es5'});
goog.addDependency('html/sanitizer/htmlsanitizer_test.js', ['goog.html.HtmlSanitizerTest'], ['goog.array', 'goog.dom', 'goog.functions', 'goog.html.SafeHtml', 'goog.html.SafeUrl', 'goog.html.sanitizer.HtmlSanitizer', 'goog.html.sanitizer.HtmlSanitizer.Builder', 'goog.html.sanitizer.TagWhitelist', 'goog.html.testing', 'goog.object', 'goog.string.Const', 'goog.testing.dom', 'goog.testing.testSuite', 'goog.userAgent', 'goog.userAgent.product'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('html/sanitizer/htmlsanitizer_unified_test.js', ['goog.html.HtmlSanitizerUnifiedTest'], ['goog.html.SafeHtml', 'goog.html.htmlTestVectors', 'goog.html.sanitizer.HtmlSanitizer.Builder', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('html/sanitizer/noclobber.js', ['goog.html.sanitizer.noclobber'], ['goog.asserts', 'goog.dom.NodeType', 'goog.userAgent.product'], {'lang': 'es5', 'module': 'goog'});
goog.addDependency('html/sanitizer/noclobber_test.js', ['goog.html.sanitizer.noclobberTest'], ['goog.dom.NodeType', 'goog.html.sanitizer.noclobber', 'goog.testing.PropertyReplacer', 'goog.testing.dom', 'goog.testing.testSuite', 'goog.userAgent.product'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('html/sanitizer/safedomtreeprocessor.js', ['goog.html.sanitizer.SafeDomTreeProcessor'], ['goog.dom', 'goog.dom.NodeType', 'goog.dom.TagName', 'goog.dom.safe', 'goog.html.sanitizer.ElementWeakMap', 'goog.html.sanitizer.noclobber', 'goog.html.uncheckedconversions', 'goog.log', 'goog.string.Const', 'goog.userAgent'], {'module': 'goog'});
goog.addDependency('html/sanitizer/safedomtreeprocessor_test.js', ['goog.html.sanitizer.SafeDomTreeProcessorTest'], ['goog.html.sanitizer.SafeDomTreeProcessor', 'goog.html.sanitizer.noclobber', 'goog.testing.dom', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('html/sanitizer/tagblacklist.js', ['goog.html.sanitizer.TagBlacklist'], [], {});
goog.addDependency('html/sanitizer/tagwhitelist.js', ['goog.html.sanitizer.TagWhitelist'], [], {});
goog.addDependency('html/sanitizer/unsafe.js', ['goog.html.sanitizer.unsafe'], ['goog.asserts', 'goog.html.sanitizer.HtmlSanitizer.Builder', 'goog.string', 'goog.string.Const'], {});
goog.addDependency('html/sanitizer/unsafe_test.js', ['goog.html.UnsafeTest'], ['goog.html.SafeHtml', 'goog.html.sanitizer.AttributeWhitelist', 'goog.html.sanitizer.HtmlSanitizer', 'goog.html.sanitizer.TagWhitelist', 'goog.html.sanitizer.unsafe', 'goog.string.Const', 'goog.testing.dom', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('html/silverlight.js', ['goog.html.silverlight'], ['goog.html.SafeHtml', 'goog.html.TrustedResourceUrl', 'goog.html.flash', 'goog.string.Const'], {});
goog.addDependency('html/silverlight_test.js', ['goog.html.silverlightTest'], ['goog.html.SafeHtml', 'goog.html.TrustedResourceUrl', 'goog.html.silverlight', 'goog.string.Const', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('html/testing.js', ['goog.html.testing'], ['goog.html.SafeHtml', 'goog.html.SafeScript', 'goog.html.SafeStyle', 'goog.html.SafeStyleSheet', 'goog.html.SafeUrl', 'goog.html.TrustedResourceUrl', 'goog.testing.mockmatchers.ArgumentMatcher'], {});
goog.addDependency('html/textextractor.js', ['goog.html.textExtractor'], ['goog.array', 'goog.dom.TagName', 'goog.html.sanitizer.HtmlSanitizer', 'goog.object', 'goog.userAgent'], {});
goog.addDependency('html/textextractor_test.js', ['goog.html.textExtractorTest'], ['goog.html.textExtractor', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('html/trustedresourceurl.js', ['goog.html.TrustedResourceUrl'], ['goog.asserts', 'goog.html.trustedtypes', 'goog.i18n.bidi.Dir', 'goog.i18n.bidi.DirectionalString', 'goog.string.Const', 'goog.string.TypedString'], {});
goog.addDependency('html/trustedresourceurl_test.js', ['goog.html.trustedResourceUrlTest'], ['goog.html.TrustedResourceUrl', 'goog.html.trustedtypes', 'goog.i18n.bidi.Dir', 'goog.object', 'goog.string.Const', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('html/trustedtypes.js', ['goog.html.trustedtypes'], [], {});
goog.addDependency('html/uncheckedconversions.js', ['goog.html.uncheckedconversions'], ['goog.asserts', 'goog.html.SafeHtml', 'goog.html.SafeScript', 'goog.html.SafeStyle', 'goog.html.SafeStyleSheet', 'goog.html.SafeUrl', 'goog.html.TrustedResourceUrl', 'goog.string.Const', 'goog.string.internal'], {});
goog.addDependency('html/uncheckedconversions_test.js', ['goog.html.uncheckedconversionsTest'], ['goog.html.SafeHtml', 'goog.html.SafeScript', 'goog.html.SafeStyle', 'goog.html.SafeStyleSheet', 'goog.html.SafeUrl', 'goog.html.TrustedResourceUrl', 'goog.html.uncheckedconversions', 'goog.i18n.bidi.Dir', 'goog.string.Const', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('html/utils.js', ['goog.html.utils'], ['goog.string'], {});
goog.addDependency('html/utils_test.js', ['goog.html.UtilsTest'], ['goog.array', 'goog.dom.TagName', 'goog.html.utils', 'goog.object', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('i18n/bidi.js', ['goog.i18n.bidi', 'goog.i18n.bidi.Dir', 'goog.i18n.bidi.DirectionalString', 'goog.i18n.bidi.Format'], [], {'lang': 'es6'});
goog.addDependency('i18n/bidi_test.js', ['goog.i18n.bidiTest'], ['goog.i18n.bidi', 'goog.i18n.bidi.Dir', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('i18n/bidiformatter.js', ['goog.i18n.BidiFormatter'], ['goog.html.SafeHtml', 'goog.i18n.bidi', 'goog.i18n.bidi.Dir', 'goog.i18n.bidi.Format'], {});
goog.addDependency('i18n/bidiformatter_test.js', ['goog.i18n.BidiFormatterTest'], ['goog.html.SafeHtml', 'goog.html.testing', 'goog.i18n.BidiFormatter', 'goog.i18n.bidi.Dir', 'goog.i18n.bidi.Format', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('i18n/charlistdecompressor.js', ['goog.i18n.CharListDecompressor'], ['goog.array', 'goog.i18n.uChar'], {});
goog.addDependency('i18n/charlistdecompressor_test.js', ['goog.i18n.CharListDecompressorTest'], ['goog.i18n.CharListDecompressor', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('i18n/charpickerdata.js', ['goog.i18n.CharPickerData'], [], {});
goog.addDependency('i18n/collation.js', ['goog.i18n.collation'], [], {'lang': 'es6'});
goog.addDependency('i18n/collation_test.js', ['goog.i18n.collationTest'], ['goog.i18n.collation', 'goog.testing.ExpectedFailures', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('i18n/compactnumberformatsymbols.js', ['goog.i18n.CompactNumberFormatSymbols', 'goog.i18n.CompactNumberFormatSymbols_af', 'goog.i18n.CompactNumberFormatSymbols_am', 'goog.i18n.CompactNumberFormatSymbols_ar', 'goog.i18n.CompactNumberFormatSymbols_ar_DZ', 'goog.i18n.CompactNumberFormatSymbols_ar_EG', 'goog.i18n.CompactNumberFormatSymbols_az', 'goog.i18n.CompactNumberFormatSymbols_be', 'goog.i18n.CompactNumberFormatSymbols_bg', 'goog.i18n.CompactNumberFormatSymbols_bn', 'goog.i18n.CompactNumberFormatSymbols_br', 'goog.i18n.CompactNumberFormatSymbols_bs', 'goog.i18n.CompactNumberFormatSymbols_ca', 'goog.i18n.CompactNumberFormatSymbols_chr', 'goog.i18n.CompactNumberFormatSymbols_cs', 'goog.i18n.CompactNumberFormatSymbols_cy', 'goog.i18n.CompactNumberFormatSymbols_da', 'goog.i18n.CompactNumberFormatSymbols_de', 'goog.i18n.CompactNumberFormatSymbols_de_AT', 'goog.i18n.CompactNumberFormatSymbols_de_CH', 'goog.i18n.CompactNumberFormatSymbols_el', 'goog.i18n.CompactNumberFormatSymbols_en', 'goog.i18n.CompactNumberFormatSymbols_en_AU', 'goog.i18n.CompactNumberFormatSymbols_en_CA', 'goog.i18n.CompactNumberFormatSymbols_en_GB', 'goog.i18n.CompactNumberFormatSymbols_en_IE', 'goog.i18n.CompactNumberFormatSymbols_en_IN', 'goog.i18n.CompactNumberFormatSymbols_en_SG', 'goog.i18n.CompactNumberFormatSymbols_en_US', 'goog.i18n.CompactNumberFormatSymbols_en_ZA', 'goog.i18n.CompactNumberFormatSymbols_es', 'goog.i18n.CompactNumberFormatSymbols_es_419', 'goog.i18n.CompactNumberFormatSymbols_es_ES', 'goog.i18n.CompactNumberFormatSymbols_es_MX', 'goog.i18n.CompactNumberFormatSymbols_es_US', 'goog.i18n.CompactNumberFormatSymbols_et', 'goog.i18n.CompactNumberFormatSymbols_eu', 'goog.i18n.CompactNumberFormatSymbols_fa', 'goog.i18n.CompactNumberFormatSymbols_fi', 'goog.i18n.CompactNumberFormatSymbols_fil', 'goog.i18n.CompactNumberFormatSymbols_fr', 'goog.i18n.CompactNumberFormatSymbols_fr_CA', 'goog.i18n.CompactNumberFormatSymbols_ga', 'goog.i18n.CompactNumberFormatSymbols_gl', 'goog.i18n.CompactNumberFormatSymbols_gsw', 'goog.i18n.CompactNumberFormatSymbols_gu', 'goog.i18n.CompactNumberFormatSymbols_haw', 'goog.i18n.CompactNumberFormatSymbols_he', 'goog.i18n.CompactNumberFormatSymbols_hi', 'goog.i18n.CompactNumberFormatSymbols_hr', 'goog.i18n.CompactNumberFormatSymbols_hu', 'goog.i18n.CompactNumberFormatSymbols_hy', 'goog.i18n.CompactNumberFormatSymbols_id', 'goog.i18n.CompactNumberFormatSymbols_in', 'goog.i18n.CompactNumberFormatSymbols_is', 'goog.i18n.CompactNumberFormatSymbols_it', 'goog.i18n.CompactNumberFormatSymbols_iw', 'goog.i18n.CompactNumberFormatSymbols_ja', 'goog.i18n.CompactNumberFormatSymbols_ka', 'goog.i18n.CompactNumberFormatSymbols_kk', 'goog.i18n.CompactNumberFormatSymbols_km', 'goog.i18n.CompactNumberFormatSymbols_kn', 'goog.i18n.CompactNumberFormatSymbols_ko', 'goog.i18n.CompactNumberFormatSymbols_ky', 'goog.i18n.CompactNumberFormatSymbols_ln', 'goog.i18n.CompactNumberFormatSymbols_lo', 'goog.i18n.CompactNumberFormatSymbols_lt', 'goog.i18n.CompactNumberFormatSymbols_lv', 'goog.i18n.CompactNumberFormatSymbols_mk', 'goog.i18n.CompactNumberFormatSymbols_ml', 'goog.i18n.CompactNumberFormatSymbols_mn', 'goog.i18n.CompactNumberFormatSymbols_mo', 'goog.i18n.CompactNumberFormatSymbols_mr', 'goog.i18n.CompactNumberFormatSymbols_ms', 'goog.i18n.CompactNumberFormatSymbols_mt', 'goog.i18n.CompactNumberFormatSymbols_my', 'goog.i18n.CompactNumberFormatSymbols_nb', 'goog.i18n.CompactNumberFormatSymbols_ne', 'goog.i18n.CompactNumberFormatSymbols_nl', 'goog.i18n.CompactNumberFormatSymbols_no', 'goog.i18n.CompactNumberFormatSymbols_no_NO', 'goog.i18n.CompactNumberFormatSymbols_or', 'goog.i18n.CompactNumberFormatSymbols_pa', 'goog.i18n.CompactNumberFormatSymbols_pl', 'goog.i18n.CompactNumberFormatSymbols_pt', 'goog.i18n.CompactNumberFormatSymbols_pt_BR', 'goog.i18n.CompactNumberFormatSymbols_pt_PT', 'goog.i18n.CompactNumberFormatSymbols_ro', 'goog.i18n.CompactNumberFormatSymbols_ru', 'goog.i18n.CompactNumberFormatSymbols_sh', 'goog.i18n.CompactNumberFormatSymbols_si', 'goog.i18n.CompactNumberFormatSymbols_sk', 'goog.i18n.CompactNumberFormatSymbols_sl', 'goog.i18n.CompactNumberFormatSymbols_sq', 'goog.i18n.CompactNumberFormatSymbols_sr', 'goog.i18n.CompactNumberFormatSymbols_sr_Latn', 'goog.i18n.CompactNumberFormatSymbols_sv', 'goog.i18n.CompactNumberFormatSymbols_sw', 'goog.i18n.CompactNumberFormatSymbols_ta', 'goog.i18n.CompactNumberFormatSymbols_te', 'goog.i18n.CompactNumberFormatSymbols_th', 'goog.i18n.CompactNumberFormatSymbols_tl', 'goog.i18n.CompactNumberFormatSymbols_tr', 'goog.i18n.CompactNumberFormatSymbols_uk', 'goog.i18n.CompactNumberFormatSymbols_ur', 'goog.i18n.CompactNumberFormatSymbols_uz', 'goog.i18n.CompactNumberFormatSymbols_vi', 'goog.i18n.CompactNumberFormatSymbols_zh', 'goog.i18n.CompactNumberFormatSymbols_zh_CN', 'goog.i18n.CompactNumberFormatSymbols_zh_HK', 'goog.i18n.CompactNumberFormatSymbols_zh_TW', 'goog.i18n.CompactNumberFormatSymbols_zu'], [], {});
goog.addDependency('i18n/compactnumberformatsymbolsext.js', ['goog.i18n.CompactNumberFormatSymbolsExt', 'goog.i18n.CompactNumberFormatSymbols_af_NA', 'goog.i18n.CompactNumberFormatSymbols_af_ZA', 'goog.i18n.CompactNumberFormatSymbols_agq', 'goog.i18n.CompactNumberFormatSymbols_agq_CM', 'goog.i18n.CompactNumberFormatSymbols_ak', 'goog.i18n.CompactNumberFormatSymbols_ak_GH', 'goog.i18n.CompactNumberFormatSymbols_am_ET', 'goog.i18n.CompactNumberFormatSymbols_ar_001', 'goog.i18n.CompactNumberFormatSymbols_ar_AE', 'goog.i18n.CompactNumberFormatSymbols_ar_BH', 'goog.i18n.CompactNumberFormatSymbols_ar_DJ', 'goog.i18n.CompactNumberFormatSymbols_ar_EH', 'goog.i18n.CompactNumberFormatSymbols_ar_ER', 'goog.i18n.CompactNumberFormatSymbols_ar_IL', 'goog.i18n.CompactNumberFormatSymbols_ar_IQ', 'goog.i18n.CompactNumberFormatSymbols_ar_JO', 'goog.i18n.CompactNumberFormatSymbols_ar_KM', 'goog.i18n.CompactNumberFormatSymbols_ar_KW', 'goog.i18n.CompactNumberFormatSymbols_ar_LB', 'goog.i18n.CompactNumberFormatSymbols_ar_LY', 'goog.i18n.CompactNumberFormatSymbols_ar_MA', 'goog.i18n.CompactNumberFormatSymbols_ar_MR', 'goog.i18n.CompactNumberFormatSymbols_ar_OM', 'goog.i18n.CompactNumberFormatSymbols_ar_PS', 'goog.i18n.CompactNumberFormatSymbols_ar_QA', 'goog.i18n.CompactNumberFormatSymbols_ar_SA', 'goog.i18n.CompactNumberFormatSymbols_ar_SD', 'goog.i18n.CompactNumberFormatSymbols_ar_SO', 'goog.i18n.CompactNumberFormatSymbols_ar_SS', 'goog.i18n.CompactNumberFormatSymbols_ar_SY', 'goog.i18n.CompactNumberFormatSymbols_ar_TD', 'goog.i18n.CompactNumberFormatSymbols_ar_TN', 'goog.i18n.CompactNumberFormatSymbols_ar_XB', 'goog.i18n.CompactNumberFormatSymbols_ar_YE', 'goog.i18n.CompactNumberFormatSymbols_as', 'goog.i18n.CompactNumberFormatSymbols_as_IN', 'goog.i18n.CompactNumberFormatSymbols_asa', 'goog.i18n.CompactNumberFormatSymbols_asa_TZ', 'goog.i18n.CompactNumberFormatSymbols_ast', 'goog.i18n.CompactNumberFormatSymbols_ast_ES', 'goog.i18n.CompactNumberFormatSymbols_az_Cyrl', 'goog.i18n.CompactNumberFormatSymbols_az_Cyrl_AZ', 'goog.i18n.CompactNumberFormatSymbols_az_Latn', 'goog.i18n.CompactNumberFormatSymbols_az_Latn_AZ', 'goog.i18n.CompactNumberFormatSymbols_bas', 'goog.i18n.CompactNumberFormatSymbols_bas_CM', 'goog.i18n.CompactNumberFormatSymbols_be_BY', 'goog.i18n.CompactNumberFormatSymbols_bem', 'goog.i18n.CompactNumberFormatSymbols_bem_ZM', 'goog.i18n.CompactNumberFormatSymbols_bez', 'goog.i18n.CompactNumberFormatSymbols_bez_TZ', 'goog.i18n.CompactNumberFormatSymbols_bg_BG', 'goog.i18n.CompactNumberFormatSymbols_bm', 'goog.i18n.CompactNumberFormatSymbols_bm_ML', 'goog.i18n.CompactNumberFormatSymbols_bn_BD', 'goog.i18n.CompactNumberFormatSymbols_bn_IN', 'goog.i18n.CompactNumberFormatSymbols_bo', 'goog.i18n.CompactNumberFormatSymbols_bo_CN', 'goog.i18n.CompactNumberFormatSymbols_bo_IN', 'goog.i18n.CompactNumberFormatSymbols_br_FR', 'goog.i18n.CompactNumberFormatSymbols_brx', 'goog.i18n.CompactNumberFormatSymbols_brx_IN', 'goog.i18n.CompactNumberFormatSymbols_bs_Cyrl', 'goog.i18n.CompactNumberFormatSymbols_bs_Cyrl_BA', 'goog.i18n.CompactNumberFormatSymbols_bs_Latn', 'goog.i18n.CompactNumberFormatSymbols_bs_Latn_BA', 'goog.i18n.CompactNumberFormatSymbols_ca_AD', 'goog.i18n.CompactNumberFormatSymbols_ca_ES', 'goog.i18n.CompactNumberFormatSymbols_ca_FR', 'goog.i18n.CompactNumberFormatSymbols_ca_IT', 'goog.i18n.CompactNumberFormatSymbols_ccp', 'goog.i18n.CompactNumberFormatSymbols_ccp_BD', 'goog.i18n.CompactNumberFormatSymbols_ccp_IN', 'goog.i18n.CompactNumberFormatSymbols_ce', 'goog.i18n.CompactNumberFormatSymbols_ce_RU', 'goog.i18n.CompactNumberFormatSymbols_ceb', 'goog.i18n.CompactNumberFormatSymbols_ceb_PH', 'goog.i18n.CompactNumberFormatSymbols_cgg', 'goog.i18n.CompactNumberFormatSymbols_cgg_UG', 'goog.i18n.CompactNumberFormatSymbols_chr_US', 'goog.i18n.CompactNumberFormatSymbols_ckb', 'goog.i18n.CompactNumberFormatSymbols_ckb_IQ', 'goog.i18n.CompactNumberFormatSymbols_ckb_IR', 'goog.i18n.CompactNumberFormatSymbols_cs_CZ', 'goog.i18n.CompactNumberFormatSymbols_cy_GB', 'goog.i18n.CompactNumberFormatSymbols_da_DK', 'goog.i18n.CompactNumberFormatSymbols_da_GL', 'goog.i18n.CompactNumberFormatSymbols_dav', 'goog.i18n.CompactNumberFormatSymbols_dav_KE', 'goog.i18n.CompactNumberFormatSymbols_de_BE', 'goog.i18n.CompactNumberFormatSymbols_de_DE', 'goog.i18n.CompactNumberFormatSymbols_de_IT', 'goog.i18n.CompactNumberFormatSymbols_de_LI', 'goog.i18n.CompactNumberFormatSymbols_de_LU', 'goog.i18n.CompactNumberFormatSymbols_dje', 'goog.i18n.CompactNumberFormatSymbols_dje_NE', 'goog.i18n.CompactNumberFormatSymbols_dsb', 'goog.i18n.CompactNumberFormatSymbols_dsb_DE', 'goog.i18n.CompactNumberFormatSymbols_dua', 'goog.i18n.CompactNumberFormatSymbols_dua_CM', 'goog.i18n.CompactNumberFormatSymbols_dyo', 'goog.i18n.CompactNumberFormatSymbols_dyo_SN', 'goog.i18n.CompactNumberFormatSymbols_dz', 'goog.i18n.CompactNumberFormatSymbols_dz_BT', 'goog.i18n.CompactNumberFormatSymbols_ebu', 'goog.i18n.CompactNumberFormatSymbols_ebu_KE', 'goog.i18n.CompactNumberFormatSymbols_ee', 'goog.i18n.CompactNumberFormatSymbols_ee_GH', 'goog.i18n.CompactNumberFormatSymbols_ee_TG', 'goog.i18n.CompactNumberFormatSymbols_el_CY', 'goog.i18n.CompactNumberFormatSymbols_el_GR', 'goog.i18n.CompactNumberFormatSymbols_en_001', 'goog.i18n.CompactNumberFormatSymbols_en_150', 'goog.i18n.CompactNumberFormatSymbols_en_AE', 'goog.i18n.CompactNumberFormatSymbols_en_AG', 'goog.i18n.CompactNumberFormatSymbols_en_AI', 'goog.i18n.CompactNumberFormatSymbols_en_AS', 'goog.i18n.CompactNumberFormatSymbols_en_AT', 'goog.i18n.CompactNumberFormatSymbols_en_BB', 'goog.i18n.CompactNumberFormatSymbols_en_BE', 'goog.i18n.CompactNumberFormatSymbols_en_BI', 'goog.i18n.CompactNumberFormatSymbols_en_BM', 'goog.i18n.CompactNumberFormatSymbols_en_BS', 'goog.i18n.CompactNumberFormatSymbols_en_BW', 'goog.i18n.CompactNumberFormatSymbols_en_BZ', 'goog.i18n.CompactNumberFormatSymbols_en_CC', 'goog.i18n.CompactNumberFormatSymbols_en_CH', 'goog.i18n.CompactNumberFormatSymbols_en_CK', 'goog.i18n.CompactNumberFormatSymbols_en_CM', 'goog.i18n.CompactNumberFormatSymbols_en_CX', 'goog.i18n.CompactNumberFormatSymbols_en_CY', 'goog.i18n.CompactNumberFormatSymbols_en_DE', 'goog.i18n.CompactNumberFormatSymbols_en_DG', 'goog.i18n.CompactNumberFormatSymbols_en_DK', 'goog.i18n.CompactNumberFormatSymbols_en_DM', 'goog.i18n.CompactNumberFormatSymbols_en_ER', 'goog.i18n.CompactNumberFormatSymbols_en_FI', 'goog.i18n.CompactNumberFormatSymbols_en_FJ', 'goog.i18n.CompactNumberFormatSymbols_en_FK', 'goog.i18n.CompactNumberFormatSymbols_en_FM', 'goog.i18n.CompactNumberFormatSymbols_en_GD', 'goog.i18n.CompactNumberFormatSymbols_en_GG', 'goog.i18n.CompactNumberFormatSymbols_en_GH', 'goog.i18n.CompactNumberFormatSymbols_en_GI', 'goog.i18n.CompactNumberFormatSymbols_en_GM', 'goog.i18n.CompactNumberFormatSymbols_en_GU', 'goog.i18n.CompactNumberFormatSymbols_en_GY', 'goog.i18n.CompactNumberFormatSymbols_en_HK', 'goog.i18n.CompactNumberFormatSymbols_en_IL', 'goog.i18n.CompactNumberFormatSymbols_en_IM', 'goog.i18n.CompactNumberFormatSymbols_en_IO', 'goog.i18n.CompactNumberFormatSymbols_en_JE', 'goog.i18n.CompactNumberFormatSymbols_en_JM', 'goog.i18n.CompactNumberFormatSymbols_en_KE', 'goog.i18n.CompactNumberFormatSymbols_en_KI', 'goog.i18n.CompactNumberFormatSymbols_en_KN', 'goog.i18n.CompactNumberFormatSymbols_en_KY', 'goog.i18n.CompactNumberFormatSymbols_en_LC', 'goog.i18n.CompactNumberFormatSymbols_en_LR', 'goog.i18n.CompactNumberFormatSymbols_en_LS', 'goog.i18n.CompactNumberFormatSymbols_en_MG', 'goog.i18n.CompactNumberFormatSymbols_en_MH', 'goog.i18n.CompactNumberFormatSymbols_en_MO', 'goog.i18n.CompactNumberFormatSymbols_en_MP', 'goog.i18n.CompactNumberFormatSymbols_en_MS', 'goog.i18n.CompactNumberFormatSymbols_en_MT', 'goog.i18n.CompactNumberFormatSymbols_en_MU', 'goog.i18n.CompactNumberFormatSymbols_en_MW', 'goog.i18n.CompactNumberFormatSymbols_en_MY', 'goog.i18n.CompactNumberFormatSymbols_en_NA', 'goog.i18n.CompactNumberFormatSymbols_en_NF', 'goog.i18n.CompactNumberFormatSymbols_en_NG', 'goog.i18n.CompactNumberFormatSymbols_en_NL', 'goog.i18n.CompactNumberFormatSymbols_en_NR', 'goog.i18n.CompactNumberFormatSymbols_en_NU', 'goog.i18n.CompactNumberFormatSymbols_en_NZ', 'goog.i18n.CompactNumberFormatSymbols_en_PG', 'goog.i18n.CompactNumberFormatSymbols_en_PH', 'goog.i18n.CompactNumberFormatSymbols_en_PK', 'goog.i18n.CompactNumberFormatSymbols_en_PN', 'goog.i18n.CompactNumberFormatSymbols_en_PR', 'goog.i18n.CompactNumberFormatSymbols_en_PW', 'goog.i18n.CompactNumberFormatSymbols_en_RW', 'goog.i18n.CompactNumberFormatSymbols_en_SB', 'goog.i18n.CompactNumberFormatSymbols_en_SC', 'goog.i18n.CompactNumberFormatSymbols_en_SD', 'goog.i18n.CompactNumberFormatSymbols_en_SE', 'goog.i18n.CompactNumberFormatSymbols_en_SH', 'goog.i18n.CompactNumberFormatSymbols_en_SI', 'goog.i18n.CompactNumberFormatSymbols_en_SL', 'goog.i18n.CompactNumberFormatSymbols_en_SS', 'goog.i18n.CompactNumberFormatSymbols_en_SX', 'goog.i18n.CompactNumberFormatSymbols_en_SZ', 'goog.i18n.CompactNumberFormatSymbols_en_TC', 'goog.i18n.CompactNumberFormatSymbols_en_TK', 'goog.i18n.CompactNumberFormatSymbols_en_TO', 'goog.i18n.CompactNumberFormatSymbols_en_TT', 'goog.i18n.CompactNumberFormatSymbols_en_TV', 'goog.i18n.CompactNumberFormatSymbols_en_TZ', 'goog.i18n.CompactNumberFormatSymbols_en_UG', 'goog.i18n.CompactNumberFormatSymbols_en_UM', 'goog.i18n.CompactNumberFormatSymbols_en_US_POSIX', 'goog.i18n.CompactNumberFormatSymbols_en_VC', 'goog.i18n.CompactNumberFormatSymbols_en_VG', 'goog.i18n.CompactNumberFormatSymbols_en_VI', 'goog.i18n.CompactNumberFormatSymbols_en_VU', 'goog.i18n.CompactNumberFormatSymbols_en_WS', 'goog.i18n.CompactNumberFormatSymbols_en_XA', 'goog.i18n.CompactNumberFormatSymbols_en_ZM', 'goog.i18n.CompactNumberFormatSymbols_en_ZW', 'goog.i18n.CompactNumberFormatSymbols_eo', 'goog.i18n.CompactNumberFormatSymbols_eo_001', 'goog.i18n.CompactNumberFormatSymbols_es_AR', 'goog.i18n.CompactNumberFormatSymbols_es_BO', 'goog.i18n.CompactNumberFormatSymbols_es_BR', 'goog.i18n.CompactNumberFormatSymbols_es_BZ', 'goog.i18n.CompactNumberFormatSymbols_es_CL', 'goog.i18n.CompactNumberFormatSymbols_es_CO', 'goog.i18n.CompactNumberFormatSymbols_es_CR', 'goog.i18n.CompactNumberFormatSymbols_es_CU', 'goog.i18n.CompactNumberFormatSymbols_es_DO', 'goog.i18n.CompactNumberFormatSymbols_es_EA', 'goog.i18n.CompactNumberFormatSymbols_es_EC', 'goog.i18n.CompactNumberFormatSymbols_es_GQ', 'goog.i18n.CompactNumberFormatSymbols_es_GT', 'goog.i18n.CompactNumberFormatSymbols_es_HN', 'goog.i18n.CompactNumberFormatSymbols_es_IC', 'goog.i18n.CompactNumberFormatSymbols_es_NI', 'goog.i18n.CompactNumberFormatSymbols_es_PA', 'goog.i18n.CompactNumberFormatSymbols_es_PE', 'goog.i18n.CompactNumberFormatSymbols_es_PH', 'goog.i18n.CompactNumberFormatSymbols_es_PR', 'goog.i18n.CompactNumberFormatSymbols_es_PY', 'goog.i18n.CompactNumberFormatSymbols_es_SV', 'goog.i18n.CompactNumberFormatSymbols_es_UY', 'goog.i18n.CompactNumberFormatSymbols_es_VE', 'goog.i18n.CompactNumberFormatSymbols_et_EE', 'goog.i18n.CompactNumberFormatSymbols_eu_ES', 'goog.i18n.CompactNumberFormatSymbols_ewo', 'goog.i18n.CompactNumberFormatSymbols_ewo_CM', 'goog.i18n.CompactNumberFormatSymbols_fa_AF', 'goog.i18n.CompactNumberFormatSymbols_fa_IR', 'goog.i18n.CompactNumberFormatSymbols_ff', 'goog.i18n.CompactNumberFormatSymbols_ff_Latn', 'goog.i18n.CompactNumberFormatSymbols_ff_Latn_BF', 'goog.i18n.CompactNumberFormatSymbols_ff_Latn_CM', 'goog.i18n.CompactNumberFormatSymbols_ff_Latn_GH', 'goog.i18n.CompactNumberFormatSymbols_ff_Latn_GM', 'goog.i18n.CompactNumberFormatSymbols_ff_Latn_GN', 'goog.i18n.CompactNumberFormatSymbols_ff_Latn_GW', 'goog.i18n.CompactNumberFormatSymbols_ff_Latn_LR', 'goog.i18n.CompactNumberFormatSymbols_ff_Latn_MR', 'goog.i18n.CompactNumberFormatSymbols_ff_Latn_NE', 'goog.i18n.CompactNumberFormatSymbols_ff_Latn_NG', 'goog.i18n.CompactNumberFormatSymbols_ff_Latn_SL', 'goog.i18n.CompactNumberFormatSymbols_ff_Latn_SN', 'goog.i18n.CompactNumberFormatSymbols_fi_FI', 'goog.i18n.CompactNumberFormatSymbols_fil_PH', 'goog.i18n.CompactNumberFormatSymbols_fo', 'goog.i18n.CompactNumberFormatSymbols_fo_DK', 'goog.i18n.CompactNumberFormatSymbols_fo_FO', 'goog.i18n.CompactNumberFormatSymbols_fr_BE', 'goog.i18n.CompactNumberFormatSymbols_fr_BF', 'goog.i18n.CompactNumberFormatSymbols_fr_BI', 'goog.i18n.CompactNumberFormatSymbols_fr_BJ', 'goog.i18n.CompactNumberFormatSymbols_fr_BL', 'goog.i18n.CompactNumberFormatSymbols_fr_CD', 'goog.i18n.CompactNumberFormatSymbols_fr_CF', 'goog.i18n.CompactNumberFormatSymbols_fr_CG', 'goog.i18n.CompactNumberFormatSymbols_fr_CH', 'goog.i18n.CompactNumberFormatSymbols_fr_CI', 'goog.i18n.CompactNumberFormatSymbols_fr_CM', 'goog.i18n.CompactNumberFormatSymbols_fr_DJ', 'goog.i18n.CompactNumberFormatSymbols_fr_DZ', 'goog.i18n.CompactNumberFormatSymbols_fr_FR', 'goog.i18n.CompactNumberFormatSymbols_fr_GA', 'goog.i18n.CompactNumberFormatSymbols_fr_GF', 'goog.i18n.CompactNumberFormatSymbols_fr_GN', 'goog.i18n.CompactNumberFormatSymbols_fr_GP', 'goog.i18n.CompactNumberFormatSymbols_fr_GQ', 'goog.i18n.CompactNumberFormatSymbols_fr_HT', 'goog.i18n.CompactNumberFormatSymbols_fr_KM', 'goog.i18n.CompactNumberFormatSymbols_fr_LU', 'goog.i18n.CompactNumberFormatSymbols_fr_MA', 'goog.i18n.CompactNumberFormatSymbols_fr_MC', 'goog.i18n.CompactNumberFormatSymbols_fr_MF', 'goog.i18n.CompactNumberFormatSymbols_fr_MG', 'goog.i18n.CompactNumberFormatSymbols_fr_ML', 'goog.i18n.CompactNumberFormatSymbols_fr_MQ', 'goog.i18n.CompactNumberFormatSymbols_fr_MR', 'goog.i18n.CompactNumberFormatSymbols_fr_MU', 'goog.i18n.CompactNumberFormatSymbols_fr_NC', 'goog.i18n.CompactNumberFormatSymbols_fr_NE', 'goog.i18n.CompactNumberFormatSymbols_fr_PF', 'goog.i18n.CompactNumberFormatSymbols_fr_PM', 'goog.i18n.CompactNumberFormatSymbols_fr_RE', 'goog.i18n.CompactNumberFormatSymbols_fr_RW', 'goog.i18n.CompactNumberFormatSymbols_fr_SC', 'goog.i18n.CompactNumberFormatSymbols_fr_SN', 'goog.i18n.CompactNumberFormatSymbols_fr_SY', 'goog.i18n.CompactNumberFormatSymbols_fr_TD', 'goog.i18n.CompactNumberFormatSymbols_fr_TG', 'goog.i18n.CompactNumberFormatSymbols_fr_TN', 'goog.i18n.CompactNumberFormatSymbols_fr_VU', 'goog.i18n.CompactNumberFormatSymbols_fr_WF', 'goog.i18n.CompactNumberFormatSymbols_fr_YT', 'goog.i18n.CompactNumberFormatSymbols_fur', 'goog.i18n.CompactNumberFormatSymbols_fur_IT', 'goog.i18n.CompactNumberFormatSymbols_fy', 'goog.i18n.CompactNumberFormatSymbols_fy_NL', 'goog.i18n.CompactNumberFormatSymbols_ga_IE', 'goog.i18n.CompactNumberFormatSymbols_gd', 'goog.i18n.CompactNumberFormatSymbols_gd_GB', 'goog.i18n.CompactNumberFormatSymbols_gl_ES', 'goog.i18n.CompactNumberFormatSymbols_gsw_CH', 'goog.i18n.CompactNumberFormatSymbols_gsw_FR', 'goog.i18n.CompactNumberFormatSymbols_gsw_LI', 'goog.i18n.CompactNumberFormatSymbols_gu_IN', 'goog.i18n.CompactNumberFormatSymbols_guz', 'goog.i18n.CompactNumberFormatSymbols_guz_KE', 'goog.i18n.CompactNumberFormatSymbols_gv', 'goog.i18n.CompactNumberFormatSymbols_gv_IM', 'goog.i18n.CompactNumberFormatSymbols_ha', 'goog.i18n.CompactNumberFormatSymbols_ha_GH', 'goog.i18n.CompactNumberFormatSymbols_ha_NE', 'goog.i18n.CompactNumberFormatSymbols_ha_NG', 'goog.i18n.CompactNumberFormatSymbols_haw_US', 'goog.i18n.CompactNumberFormatSymbols_he_IL', 'goog.i18n.CompactNumberFormatSymbols_hi_IN', 'goog.i18n.CompactNumberFormatSymbols_hr_BA', 'goog.i18n.CompactNumberFormatSymbols_hr_HR', 'goog.i18n.CompactNumberFormatSymbols_hsb', 'goog.i18n.CompactNumberFormatSymbols_hsb_DE', 'goog.i18n.CompactNumberFormatSymbols_hu_HU', 'goog.i18n.CompactNumberFormatSymbols_hy_AM', 'goog.i18n.CompactNumberFormatSymbols_ia', 'goog.i18n.CompactNumberFormatSymbols_ia_001', 'goog.i18n.CompactNumberFormatSymbols_id_ID', 'goog.i18n.CompactNumberFormatSymbols_ig', 'goog.i18n.CompactNumberFormatSymbols_ig_NG', 'goog.i18n.CompactNumberFormatSymbols_ii', 'goog.i18n.CompactNumberFormatSymbols_ii_CN', 'goog.i18n.CompactNumberFormatSymbols_is_IS', 'goog.i18n.CompactNumberFormatSymbols_it_CH', 'goog.i18n.CompactNumberFormatSymbols_it_IT', 'goog.i18n.CompactNumberFormatSymbols_it_SM', 'goog.i18n.CompactNumberFormatSymbols_it_VA', 'goog.i18n.CompactNumberFormatSymbols_ja_JP', 'goog.i18n.CompactNumberFormatSymbols_jgo', 'goog.i18n.CompactNumberFormatSymbols_jgo_CM', 'goog.i18n.CompactNumberFormatSymbols_jmc', 'goog.i18n.CompactNumberFormatSymbols_jmc_TZ', 'goog.i18n.CompactNumberFormatSymbols_jv', 'goog.i18n.CompactNumberFormatSymbols_jv_ID', 'goog.i18n.CompactNumberFormatSymbols_ka_GE', 'goog.i18n.CompactNumberFormatSymbols_kab', 'goog.i18n.CompactNumberFormatSymbols_kab_DZ', 'goog.i18n.CompactNumberFormatSymbols_kam', 'goog.i18n.CompactNumberFormatSymbols_kam_KE', 'goog.i18n.CompactNumberFormatSymbols_kde', 'goog.i18n.CompactNumberFormatSymbols_kde_TZ', 'goog.i18n.CompactNumberFormatSymbols_kea', 'goog.i18n.CompactNumberFormatSymbols_kea_CV', 'goog.i18n.CompactNumberFormatSymbols_khq', 'goog.i18n.CompactNumberFormatSymbols_khq_ML', 'goog.i18n.CompactNumberFormatSymbols_ki', 'goog.i18n.CompactNumberFormatSymbols_ki_KE', 'goog.i18n.CompactNumberFormatSymbols_kk_KZ', 'goog.i18n.CompactNumberFormatSymbols_kkj', 'goog.i18n.CompactNumberFormatSymbols_kkj_CM', 'goog.i18n.CompactNumberFormatSymbols_kl', 'goog.i18n.CompactNumberFormatSymbols_kl_GL', 'goog.i18n.CompactNumberFormatSymbols_kln', 'goog.i18n.CompactNumberFormatSymbols_kln_KE', 'goog.i18n.CompactNumberFormatSymbols_km_KH', 'goog.i18n.CompactNumberFormatSymbols_kn_IN', 'goog.i18n.CompactNumberFormatSymbols_ko_KP', 'goog.i18n.CompactNumberFormatSymbols_ko_KR', 'goog.i18n.CompactNumberFormatSymbols_kok', 'goog.i18n.CompactNumberFormatSymbols_kok_IN', 'goog.i18n.CompactNumberFormatSymbols_ks', 'goog.i18n.CompactNumberFormatSymbols_ks_IN', 'goog.i18n.CompactNumberFormatSymbols_ksb', 'goog.i18n.CompactNumberFormatSymbols_ksb_TZ', 'goog.i18n.CompactNumberFormatSymbols_ksf', 'goog.i18n.CompactNumberFormatSymbols_ksf_CM', 'goog.i18n.CompactNumberFormatSymbols_ksh', 'goog.i18n.CompactNumberFormatSymbols_ksh_DE', 'goog.i18n.CompactNumberFormatSymbols_ku', 'goog.i18n.CompactNumberFormatSymbols_ku_TR', 'goog.i18n.CompactNumberFormatSymbols_kw', 'goog.i18n.CompactNumberFormatSymbols_kw_GB', 'goog.i18n.CompactNumberFormatSymbols_ky_KG', 'goog.i18n.CompactNumberFormatSymbols_lag', 'goog.i18n.CompactNumberFormatSymbols_lag_TZ', 'goog.i18n.CompactNumberFormatSymbols_lb', 'goog.i18n.CompactNumberFormatSymbols_lb_LU', 'goog.i18n.CompactNumberFormatSymbols_lg', 'goog.i18n.CompactNumberFormatSymbols_lg_UG', 'goog.i18n.CompactNumberFormatSymbols_lkt', 'goog.i18n.CompactNumberFormatSymbols_lkt_US', 'goog.i18n.CompactNumberFormatSymbols_ln_AO', 'goog.i18n.CompactNumberFormatSymbols_ln_CD', 'goog.i18n.CompactNumberFormatSymbols_ln_CF', 'goog.i18n.CompactNumberFormatSymbols_ln_CG', 'goog.i18n.CompactNumberFormatSymbols_lo_LA', 'goog.i18n.CompactNumberFormatSymbols_lrc', 'goog.i18n.CompactNumberFormatSymbols_lrc_IQ', 'goog.i18n.CompactNumberFormatSymbols_lrc_IR', 'goog.i18n.CompactNumberFormatSymbols_lt_LT', 'goog.i18n.CompactNumberFormatSymbols_lu', 'goog.i18n.CompactNumberFormatSymbols_lu_CD', 'goog.i18n.CompactNumberFormatSymbols_luo', 'goog.i18n.CompactNumberFormatSymbols_luo_KE', 'goog.i18n.CompactNumberFormatSymbols_luy', 'goog.i18n.CompactNumberFormatSymbols_luy_KE', 'goog.i18n.CompactNumberFormatSymbols_lv_LV', 'goog.i18n.CompactNumberFormatSymbols_mas', 'goog.i18n.CompactNumberFormatSymbols_mas_KE', 'goog.i18n.CompactNumberFormatSymbols_mas_TZ', 'goog.i18n.CompactNumberFormatSymbols_mer', 'goog.i18n.CompactNumberFormatSymbols_mer_KE', 'goog.i18n.CompactNumberFormatSymbols_mfe', 'goog.i18n.CompactNumberFormatSymbols_mfe_MU', 'goog.i18n.CompactNumberFormatSymbols_mg', 'goog.i18n.CompactNumberFormatSymbols_mg_MG', 'goog.i18n.CompactNumberFormatSymbols_mgh', 'goog.i18n.CompactNumberFormatSymbols_mgh_MZ', 'goog.i18n.CompactNumberFormatSymbols_mgo', 'goog.i18n.CompactNumberFormatSymbols_mgo_CM', 'goog.i18n.CompactNumberFormatSymbols_mi', 'goog.i18n.CompactNumberFormatSymbols_mi_NZ', 'goog.i18n.CompactNumberFormatSymbols_mk_MK', 'goog.i18n.CompactNumberFormatSymbols_ml_IN', 'goog.i18n.CompactNumberFormatSymbols_mn_MN', 'goog.i18n.CompactNumberFormatSymbols_mr_IN', 'goog.i18n.CompactNumberFormatSymbols_ms_BN', 'goog.i18n.CompactNumberFormatSymbols_ms_MY', 'goog.i18n.CompactNumberFormatSymbols_ms_SG', 'goog.i18n.CompactNumberFormatSymbols_mt_MT', 'goog.i18n.CompactNumberFormatSymbols_mua', 'goog.i18n.CompactNumberFormatSymbols_mua_CM', 'goog.i18n.CompactNumberFormatSymbols_my_MM', 'goog.i18n.CompactNumberFormatSymbols_mzn', 'goog.i18n.CompactNumberFormatSymbols_mzn_IR', 'goog.i18n.CompactNumberFormatSymbols_naq', 'goog.i18n.CompactNumberFormatSymbols_naq_NA', 'goog.i18n.CompactNumberFormatSymbols_nb_NO', 'goog.i18n.CompactNumberFormatSymbols_nb_SJ', 'goog.i18n.CompactNumberFormatSymbols_nd', 'goog.i18n.CompactNumberFormatSymbols_nd_ZW', 'goog.i18n.CompactNumberFormatSymbols_nds', 'goog.i18n.CompactNumberFormatSymbols_nds_DE', 'goog.i18n.CompactNumberFormatSymbols_nds_NL', 'goog.i18n.CompactNumberFormatSymbols_ne_IN', 'goog.i18n.CompactNumberFormatSymbols_ne_NP', 'goog.i18n.CompactNumberFormatSymbols_nl_AW', 'goog.i18n.CompactNumberFormatSymbols_nl_BE', 'goog.i18n.CompactNumberFormatSymbols_nl_BQ', 'goog.i18n.CompactNumberFormatSymbols_nl_CW', 'goog.i18n.CompactNumberFormatSymbols_nl_NL', 'goog.i18n.CompactNumberFormatSymbols_nl_SR', 'goog.i18n.CompactNumberFormatSymbols_nl_SX', 'goog.i18n.CompactNumberFormatSymbols_nmg', 'goog.i18n.CompactNumberFormatSymbols_nmg_CM', 'goog.i18n.CompactNumberFormatSymbols_nn', 'goog.i18n.CompactNumberFormatSymbols_nn_NO', 'goog.i18n.CompactNumberFormatSymbols_nnh', 'goog.i18n.CompactNumberFormatSymbols_nnh_CM', 'goog.i18n.CompactNumberFormatSymbols_nus', 'goog.i18n.CompactNumberFormatSymbols_nus_SS', 'goog.i18n.CompactNumberFormatSymbols_nyn', 'goog.i18n.CompactNumberFormatSymbols_nyn_UG', 'goog.i18n.CompactNumberFormatSymbols_om', 'goog.i18n.CompactNumberFormatSymbols_om_ET', 'goog.i18n.CompactNumberFormatSymbols_om_KE', 'goog.i18n.CompactNumberFormatSymbols_or_IN', 'goog.i18n.CompactNumberFormatSymbols_os', 'goog.i18n.CompactNumberFormatSymbols_os_GE', 'goog.i18n.CompactNumberFormatSymbols_os_RU', 'goog.i18n.CompactNumberFormatSymbols_pa_Arab', 'goog.i18n.CompactNumberFormatSymbols_pa_Arab_PK', 'goog.i18n.CompactNumberFormatSymbols_pa_Guru', 'goog.i18n.CompactNumberFormatSymbols_pa_Guru_IN', 'goog.i18n.CompactNumberFormatSymbols_pl_PL', 'goog.i18n.CompactNumberFormatSymbols_ps', 'goog.i18n.CompactNumberFormatSymbols_ps_AF', 'goog.i18n.CompactNumberFormatSymbols_ps_PK', 'goog.i18n.CompactNumberFormatSymbols_pt_AO', 'goog.i18n.CompactNumberFormatSymbols_pt_CH', 'goog.i18n.CompactNumberFormatSymbols_pt_CV', 'goog.i18n.CompactNumberFormatSymbols_pt_GQ', 'goog.i18n.CompactNumberFormatSymbols_pt_GW', 'goog.i18n.CompactNumberFormatSymbols_pt_LU', 'goog.i18n.CompactNumberFormatSymbols_pt_MO', 'goog.i18n.CompactNumberFormatSymbols_pt_MZ', 'goog.i18n.CompactNumberFormatSymbols_pt_ST', 'goog.i18n.CompactNumberFormatSymbols_pt_TL', 'goog.i18n.CompactNumberFormatSymbols_qu', 'goog.i18n.CompactNumberFormatSymbols_qu_BO', 'goog.i18n.CompactNumberFormatSymbols_qu_EC', 'goog.i18n.CompactNumberFormatSymbols_qu_PE', 'goog.i18n.CompactNumberFormatSymbols_rm', 'goog.i18n.CompactNumberFormatSymbols_rm_CH', 'goog.i18n.CompactNumberFormatSymbols_rn', 'goog.i18n.CompactNumberFormatSymbols_rn_BI', 'goog.i18n.CompactNumberFormatSymbols_ro_MD', 'goog.i18n.CompactNumberFormatSymbols_ro_RO', 'goog.i18n.CompactNumberFormatSymbols_rof', 'goog.i18n.CompactNumberFormatSymbols_rof_TZ', 'goog.i18n.CompactNumberFormatSymbols_ru_BY', 'goog.i18n.CompactNumberFormatSymbols_ru_KG', 'goog.i18n.CompactNumberFormatSymbols_ru_KZ', 'goog.i18n.CompactNumberFormatSymbols_ru_MD', 'goog.i18n.CompactNumberFormatSymbols_ru_RU', 'goog.i18n.CompactNumberFormatSymbols_ru_UA', 'goog.i18n.CompactNumberFormatSymbols_rw', 'goog.i18n.CompactNumberFormatSymbols_rw_RW', 'goog.i18n.CompactNumberFormatSymbols_rwk', 'goog.i18n.CompactNumberFormatSymbols_rwk_TZ', 'goog.i18n.CompactNumberFormatSymbols_sah', 'goog.i18n.CompactNumberFormatSymbols_sah_RU', 'goog.i18n.CompactNumberFormatSymbols_saq', 'goog.i18n.CompactNumberFormatSymbols_saq_KE', 'goog.i18n.CompactNumberFormatSymbols_sbp', 'goog.i18n.CompactNumberFormatSymbols_sbp_TZ', 'goog.i18n.CompactNumberFormatSymbols_sd', 'goog.i18n.CompactNumberFormatSymbols_sd_PK', 'goog.i18n.CompactNumberFormatSymbols_se', 'goog.i18n.CompactNumberFormatSymbols_se_FI', 'goog.i18n.CompactNumberFormatSymbols_se_NO', 'goog.i18n.CompactNumberFormatSymbols_se_SE', 'goog.i18n.CompactNumberFormatSymbols_seh', 'goog.i18n.CompactNumberFormatSymbols_seh_MZ', 'goog.i18n.CompactNumberFormatSymbols_ses', 'goog.i18n.CompactNumberFormatSymbols_ses_ML', 'goog.i18n.CompactNumberFormatSymbols_sg', 'goog.i18n.CompactNumberFormatSymbols_sg_CF', 'goog.i18n.CompactNumberFormatSymbols_shi', 'goog.i18n.CompactNumberFormatSymbols_shi_Latn', 'goog.i18n.CompactNumberFormatSymbols_shi_Latn_MA', 'goog.i18n.CompactNumberFormatSymbols_shi_Tfng', 'goog.i18n.CompactNumberFormatSymbols_shi_Tfng_MA', 'goog.i18n.CompactNumberFormatSymbols_si_LK', 'goog.i18n.CompactNumberFormatSymbols_sk_SK', 'goog.i18n.CompactNumberFormatSymbols_sl_SI', 'goog.i18n.CompactNumberFormatSymbols_smn', 'goog.i18n.CompactNumberFormatSymbols_smn_FI', 'goog.i18n.CompactNumberFormatSymbols_sn', 'goog.i18n.CompactNumberFormatSymbols_sn_ZW', 'goog.i18n.CompactNumberFormatSymbols_so', 'goog.i18n.CompactNumberFormatSymbols_so_DJ', 'goog.i18n.CompactNumberFormatSymbols_so_ET', 'goog.i18n.CompactNumberFormatSymbols_so_KE', 'goog.i18n.CompactNumberFormatSymbols_so_SO', 'goog.i18n.CompactNumberFormatSymbols_sq_AL', 'goog.i18n.CompactNumberFormatSymbols_sq_MK', 'goog.i18n.CompactNumberFormatSymbols_sq_XK', 'goog.i18n.CompactNumberFormatSymbols_sr_Cyrl', 'goog.i18n.CompactNumberFormatSymbols_sr_Cyrl_BA', 'goog.i18n.CompactNumberFormatSymbols_sr_Cyrl_ME', 'goog.i18n.CompactNumberFormatSymbols_sr_Cyrl_RS', 'goog.i18n.CompactNumberFormatSymbols_sr_Cyrl_XK', 'goog.i18n.CompactNumberFormatSymbols_sr_Latn_BA', 'goog.i18n.CompactNumberFormatSymbols_sr_Latn_ME', 'goog.i18n.CompactNumberFormatSymbols_sr_Latn_RS', 'goog.i18n.CompactNumberFormatSymbols_sr_Latn_XK', 'goog.i18n.CompactNumberFormatSymbols_sv_AX', 'goog.i18n.CompactNumberFormatSymbols_sv_FI', 'goog.i18n.CompactNumberFormatSymbols_sv_SE', 'goog.i18n.CompactNumberFormatSymbols_sw_CD', 'goog.i18n.CompactNumberFormatSymbols_sw_KE', 'goog.i18n.CompactNumberFormatSymbols_sw_TZ', 'goog.i18n.CompactNumberFormatSymbols_sw_UG', 'goog.i18n.CompactNumberFormatSymbols_ta_IN', 'goog.i18n.CompactNumberFormatSymbols_ta_LK', 'goog.i18n.CompactNumberFormatSymbols_ta_MY', 'goog.i18n.CompactNumberFormatSymbols_ta_SG', 'goog.i18n.CompactNumberFormatSymbols_te_IN', 'goog.i18n.CompactNumberFormatSymbols_teo', 'goog.i18n.CompactNumberFormatSymbols_teo_KE', 'goog.i18n.CompactNumberFormatSymbols_teo_UG', 'goog.i18n.CompactNumberFormatSymbols_tg', 'goog.i18n.CompactNumberFormatSymbols_tg_TJ', 'goog.i18n.CompactNumberFormatSymbols_th_TH', 'goog.i18n.CompactNumberFormatSymbols_ti', 'goog.i18n.CompactNumberFormatSymbols_ti_ER', 'goog.i18n.CompactNumberFormatSymbols_ti_ET', 'goog.i18n.CompactNumberFormatSymbols_tk', 'goog.i18n.CompactNumberFormatSymbols_tk_TM', 'goog.i18n.CompactNumberFormatSymbols_to', 'goog.i18n.CompactNumberFormatSymbols_to_TO', 'goog.i18n.CompactNumberFormatSymbols_tr_CY', 'goog.i18n.CompactNumberFormatSymbols_tr_TR', 'goog.i18n.CompactNumberFormatSymbols_tt', 'goog.i18n.CompactNumberFormatSymbols_tt_RU', 'goog.i18n.CompactNumberFormatSymbols_twq', 'goog.i18n.CompactNumberFormatSymbols_twq_NE', 'goog.i18n.CompactNumberFormatSymbols_tzm', 'goog.i18n.CompactNumberFormatSymbols_tzm_MA', 'goog.i18n.CompactNumberFormatSymbols_ug', 'goog.i18n.CompactNumberFormatSymbols_ug_CN', 'goog.i18n.CompactNumberFormatSymbols_uk_UA', 'goog.i18n.CompactNumberFormatSymbols_ur_IN', 'goog.i18n.CompactNumberFormatSymbols_ur_PK', 'goog.i18n.CompactNumberFormatSymbols_uz_Arab', 'goog.i18n.CompactNumberFormatSymbols_uz_Arab_AF', 'goog.i18n.CompactNumberFormatSymbols_uz_Cyrl', 'goog.i18n.CompactNumberFormatSymbols_uz_Cyrl_UZ', 'goog.i18n.CompactNumberFormatSymbols_uz_Latn', 'goog.i18n.CompactNumberFormatSymbols_uz_Latn_UZ', 'goog.i18n.CompactNumberFormatSymbols_vai', 'goog.i18n.CompactNumberFormatSymbols_vai_Latn', 'goog.i18n.CompactNumberFormatSymbols_vai_Latn_LR', 'goog.i18n.CompactNumberFormatSymbols_vai_Vaii', 'goog.i18n.CompactNumberFormatSymbols_vai_Vaii_LR', 'goog.i18n.CompactNumberFormatSymbols_vi_VN', 'goog.i18n.CompactNumberFormatSymbols_vun', 'goog.i18n.CompactNumberFormatSymbols_vun_TZ', 'goog.i18n.CompactNumberFormatSymbols_wae', 'goog.i18n.CompactNumberFormatSymbols_wae_CH', 'goog.i18n.CompactNumberFormatSymbols_wo', 'goog.i18n.CompactNumberFormatSymbols_wo_SN', 'goog.i18n.CompactNumberFormatSymbols_xh', 'goog.i18n.CompactNumberFormatSymbols_xh_ZA', 'goog.i18n.CompactNumberFormatSymbols_xog', 'goog.i18n.CompactNumberFormatSymbols_xog_UG', 'goog.i18n.CompactNumberFormatSymbols_yav', 'goog.i18n.CompactNumberFormatSymbols_yav_CM', 'goog.i18n.CompactNumberFormatSymbols_yi', 'goog.i18n.CompactNumberFormatSymbols_yi_001', 'goog.i18n.CompactNumberFormatSymbols_yo', 'goog.i18n.CompactNumberFormatSymbols_yo_BJ', 'goog.i18n.CompactNumberFormatSymbols_yo_NG', 'goog.i18n.CompactNumberFormatSymbols_yue', 'goog.i18n.CompactNumberFormatSymbols_yue_Hans', 'goog.i18n.CompactNumberFormatSymbols_yue_Hans_CN', 'goog.i18n.CompactNumberFormatSymbols_yue_Hant', 'goog.i18n.CompactNumberFormatSymbols_yue_Hant_HK', 'goog.i18n.CompactNumberFormatSymbols_zgh', 'goog.i18n.CompactNumberFormatSymbols_zgh_MA', 'goog.i18n.CompactNumberFormatSymbols_zh_Hans', 'goog.i18n.CompactNumberFormatSymbols_zh_Hans_CN', 'goog.i18n.CompactNumberFormatSymbols_zh_Hans_HK', 'goog.i18n.CompactNumberFormatSymbols_zh_Hans_MO', 'goog.i18n.CompactNumberFormatSymbols_zh_Hans_SG', 'goog.i18n.CompactNumberFormatSymbols_zh_Hant', 'goog.i18n.CompactNumberFormatSymbols_zh_Hant_HK', 'goog.i18n.CompactNumberFormatSymbols_zh_Hant_MO', 'goog.i18n.CompactNumberFormatSymbols_zh_Hant_TW', 'goog.i18n.CompactNumberFormatSymbols_zu_ZA'], ['goog.i18n.CompactNumberFormatSymbols'], {});
goog.addDependency('i18n/currency.js', ['goog.i18n.currency', 'goog.i18n.currency.CurrencyInfo', 'goog.i18n.currency.CurrencyInfoTier2'], [], {'lang': 'es6'});
goog.addDependency('i18n/currency_test.js', ['goog.i18n.currencyTest'], ['goog.i18n.NumberFormat', 'goog.i18n.currency', 'goog.i18n.currency.CurrencyInfo', 'goog.object', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('i18n/currencycodemap.js', ['goog.i18n.currencyCodeMap', 'goog.i18n.currencyCodeMapTier2'], [], {});
goog.addDependency('i18n/dateintervalformat.js', ['goog.i18n.DateIntervalFormat'], ['goog.array', 'goog.asserts', 'goog.date.DateLike', 'goog.date.DateRange', 'goog.date.DateTime', 'goog.date.Interval', 'goog.i18n.DateTimeFormat', 'goog.i18n.DateTimeSymbols', 'goog.i18n.DateTimeSymbolsType', 'goog.i18n.TimeZone', 'goog.i18n.dateIntervalSymbols', 'goog.object'], {'lang': 'es5', 'module': 'goog'});
goog.addDependency('i18n/dateintervalformat_test.js', ['goog.i18n.DateIntervalFormatTest'], ['goog.date.Date', 'goog.date.DateRange', 'goog.date.DateTime', 'goog.date.Interval', 'goog.i18n.DateIntervalFormat', 'goog.i18n.DateTimeFormat', 'goog.i18n.DateTimeSymbols_ar_EG', 'goog.i18n.DateTimeSymbols_en', 'goog.i18n.DateTimeSymbols_fr_CA', 'goog.i18n.DateTimeSymbols_gl', 'goog.i18n.DateTimeSymbols_hi', 'goog.i18n.DateTimeSymbols_zh', 'goog.i18n.TimeZone', 'goog.i18n.dateIntervalPatterns', 'goog.i18n.dateIntervalSymbols', 'goog.object', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('i18n/dateintervalpatterns.js', ['goog.i18n.dateIntervalPatterns'], ['goog.i18n.dateIntervalSymbols'], {'module': 'goog'});
goog.addDependency('i18n/dateintervalpatternsext.js', ['goog.i18n.dateIntervalPatternsExt'], ['goog.i18n.dateIntervalPatterns'], {'module': 'goog'});
goog.addDependency('i18n/dateintervalsymbols.js', ['goog.i18n.dateIntervalSymbols'], [], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('i18n/dateintervalsymbolsext.js', ['goog.i18n.dateIntervalSymbolsExt'], ['goog.i18n.dateIntervalSymbols'], {'module': 'goog'});
goog.addDependency('i18n/datetimeformat.js', ['goog.i18n.DateTimeFormat', 'goog.i18n.DateTimeFormat.Format'], ['goog.asserts', 'goog.date', 'goog.i18n.DateTimeSymbols', 'goog.i18n.TimeZone', 'goog.string'], {});
goog.addDependency('i18n/datetimeformat_test.js', ['goog.i18n.DateTimeFormatTest'], ['goog.date.Date', 'goog.date.DateTime', 'goog.i18n.DateTimeFormat', 'goog.i18n.DateTimePatterns', 'goog.i18n.DateTimePatterns_ar_EG', 'goog.i18n.DateTimePatterns_bg', 'goog.i18n.DateTimePatterns_de', 'goog.i18n.DateTimePatterns_en', 'goog.i18n.DateTimePatterns_en_XA', 'goog.i18n.DateTimePatterns_fa', 'goog.i18n.DateTimePatterns_fr', 'goog.i18n.DateTimePatterns_ja', 'goog.i18n.DateTimePatterns_sv', 'goog.i18n.DateTimePatterns_zh_HK', 'goog.i18n.DateTimePatterns_zh_Hant_TW', 'goog.i18n.DateTimeSymbols', 'goog.i18n.DateTimeSymbols_ar_AE', 'goog.i18n.DateTimeSymbols_ar_EG', 'goog.i18n.DateTimeSymbols_ar_SA', 'goog.i18n.DateTimeSymbols_bn_BD', 'goog.i18n.DateTimeSymbols_de', 'goog.i18n.DateTimeSymbols_en', 'goog.i18n.DateTimeSymbols_en_GB', 'goog.i18n.DateTimeSymbols_en_IE', 'goog.i18n.DateTimeSymbols_en_IN', 'goog.i18n.DateTimeSymbols_en_US', 'goog.i18n.DateTimeSymbols_fa', 'goog.i18n.DateTimeSymbols_fr', 'goog.i18n.DateTimeSymbols_fr_DJ', 'goog.i18n.DateTimeSymbols_he_IL', 'goog.i18n.DateTimeSymbols_ja', 'goog.i18n.DateTimeSymbols_ro_RO', 'goog.i18n.DateTimeSymbols_sv', 'goog.i18n.TimeZone', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('i18n/datetimeparse.js', ['goog.i18n.DateTimeParse'], ['goog.asserts', 'goog.date', 'goog.i18n.DateTimeFormat', 'goog.i18n.DateTimeSymbols'], {});
goog.addDependency('i18n/datetimeparse_test.js', ['goog.i18n.DateTimeParseTest'], ['goog.date.Date', 'goog.i18n.DateTimeFormat', 'goog.i18n.DateTimeParse', 'goog.i18n.DateTimeSymbols', 'goog.i18n.DateTimeSymbols_en', 'goog.i18n.DateTimeSymbols_fa', 'goog.i18n.DateTimeSymbols_fr', 'goog.i18n.DateTimeSymbols_pl', 'goog.i18n.DateTimeSymbols_zh', 'goog.testing.ExpectedFailures', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('i18n/datetimepatterns.js', ['goog.i18n.DateTimePatterns', 'goog.i18n.DateTimePatterns_af', 'goog.i18n.DateTimePatterns_am', 'goog.i18n.DateTimePatterns_ar', 'goog.i18n.DateTimePatterns_ar_DZ', 'goog.i18n.DateTimePatterns_ar_EG', 'goog.i18n.DateTimePatterns_az', 'goog.i18n.DateTimePatterns_be', 'goog.i18n.DateTimePatterns_bg', 'goog.i18n.DateTimePatterns_bn', 'goog.i18n.DateTimePatterns_br', 'goog.i18n.DateTimePatterns_bs', 'goog.i18n.DateTimePatterns_ca', 'goog.i18n.DateTimePatterns_chr', 'goog.i18n.DateTimePatterns_cs', 'goog.i18n.DateTimePatterns_cy', 'goog.i18n.DateTimePatterns_da', 'goog.i18n.DateTimePatterns_de', 'goog.i18n.DateTimePatterns_de_AT', 'goog.i18n.DateTimePatterns_de_CH', 'goog.i18n.DateTimePatterns_el', 'goog.i18n.DateTimePatterns_en', 'goog.i18n.DateTimePatterns_en_AU', 'goog.i18n.DateTimePatterns_en_CA', 'goog.i18n.DateTimePatterns_en_GB', 'goog.i18n.DateTimePatterns_en_IE', 'goog.i18n.DateTimePatterns_en_IN', 'goog.i18n.DateTimePatterns_en_SG', 'goog.i18n.DateTimePatterns_en_US', 'goog.i18n.DateTimePatterns_en_ZA', 'goog.i18n.DateTimePatterns_es', 'goog.i18n.DateTimePatterns_es_419', 'goog.i18n.DateTimePatterns_es_ES', 'goog.i18n.DateTimePatterns_es_MX', 'goog.i18n.DateTimePatterns_es_US', 'goog.i18n.DateTimePatterns_et', 'goog.i18n.DateTimePatterns_eu', 'goog.i18n.DateTimePatterns_fa', 'goog.i18n.DateTimePatterns_fi', 'goog.i18n.DateTimePatterns_fil', 'goog.i18n.DateTimePatterns_fr', 'goog.i18n.DateTimePatterns_fr_CA', 'goog.i18n.DateTimePatterns_ga', 'goog.i18n.DateTimePatterns_gl', 'goog.i18n.DateTimePatterns_gsw', 'goog.i18n.DateTimePatterns_gu', 'goog.i18n.DateTimePatterns_haw', 'goog.i18n.DateTimePatterns_he', 'goog.i18n.DateTimePatterns_hi', 'goog.i18n.DateTimePatterns_hr', 'goog.i18n.DateTimePatterns_hu', 'goog.i18n.DateTimePatterns_hy', 'goog.i18n.DateTimePatterns_id', 'goog.i18n.DateTimePatterns_in', 'goog.i18n.DateTimePatterns_is', 'goog.i18n.DateTimePatterns_it', 'goog.i18n.DateTimePatterns_iw', 'goog.i18n.DateTimePatterns_ja', 'goog.i18n.DateTimePatterns_ka', 'goog.i18n.DateTimePatterns_kk', 'goog.i18n.DateTimePatterns_km', 'goog.i18n.DateTimePatterns_kn', 'goog.i18n.DateTimePatterns_ko', 'goog.i18n.DateTimePatterns_ky', 'goog.i18n.DateTimePatterns_ln', 'goog.i18n.DateTimePatterns_lo', 'goog.i18n.DateTimePatterns_lt', 'goog.i18n.DateTimePatterns_lv', 'goog.i18n.DateTimePatterns_mk', 'goog.i18n.DateTimePatterns_ml', 'goog.i18n.DateTimePatterns_mn', 'goog.i18n.DateTimePatterns_mo', 'goog.i18n.DateTimePatterns_mr', 'goog.i18n.DateTimePatterns_ms', 'goog.i18n.DateTimePatterns_mt', 'goog.i18n.DateTimePatterns_my', 'goog.i18n.DateTimePatterns_nb', 'goog.i18n.DateTimePatterns_ne', 'goog.i18n.DateTimePatterns_nl', 'goog.i18n.DateTimePatterns_no', 'goog.i18n.DateTimePatterns_no_NO', 'goog.i18n.DateTimePatterns_or', 'goog.i18n.DateTimePatterns_pa', 'goog.i18n.DateTimePatterns_pl', 'goog.i18n.DateTimePatterns_pt', 'goog.i18n.DateTimePatterns_pt_BR', 'goog.i18n.DateTimePatterns_pt_PT', 'goog.i18n.DateTimePatterns_ro', 'goog.i18n.DateTimePatterns_ru', 'goog.i18n.DateTimePatterns_sh', 'goog.i18n.DateTimePatterns_si', 'goog.i18n.DateTimePatterns_sk', 'goog.i18n.DateTimePatterns_sl', 'goog.i18n.DateTimePatterns_sq', 'goog.i18n.DateTimePatterns_sr', 'goog.i18n.DateTimePatterns_sr_Latn', 'goog.i18n.DateTimePatterns_sv', 'goog.i18n.DateTimePatterns_sw', 'goog.i18n.DateTimePatterns_ta', 'goog.i18n.DateTimePatterns_te', 'goog.i18n.DateTimePatterns_th', 'goog.i18n.DateTimePatterns_tl', 'goog.i18n.DateTimePatterns_tr', 'goog.i18n.DateTimePatterns_uk', 'goog.i18n.DateTimePatterns_ur', 'goog.i18n.DateTimePatterns_uz', 'goog.i18n.DateTimePatterns_vi', 'goog.i18n.DateTimePatterns_zh', 'goog.i18n.DateTimePatterns_zh_CN', 'goog.i18n.DateTimePatterns_zh_HK', 'goog.i18n.DateTimePatterns_zh_TW', 'goog.i18n.DateTimePatterns_zu'], [], {});
goog.addDependency('i18n/datetimepatternsext.js', ['goog.i18n.DateTimePatternsExt', 'goog.i18n.DateTimePatterns_af_NA', 'goog.i18n.DateTimePatterns_af_ZA', 'goog.i18n.DateTimePatterns_agq', 'goog.i18n.DateTimePatterns_agq_CM', 'goog.i18n.DateTimePatterns_ak', 'goog.i18n.DateTimePatterns_ak_GH', 'goog.i18n.DateTimePatterns_am_ET', 'goog.i18n.DateTimePatterns_ar_001', 'goog.i18n.DateTimePatterns_ar_AE', 'goog.i18n.DateTimePatterns_ar_BH', 'goog.i18n.DateTimePatterns_ar_DJ', 'goog.i18n.DateTimePatterns_ar_EH', 'goog.i18n.DateTimePatterns_ar_ER', 'goog.i18n.DateTimePatterns_ar_IL', 'goog.i18n.DateTimePatterns_ar_IQ', 'goog.i18n.DateTimePatterns_ar_JO', 'goog.i18n.DateTimePatterns_ar_KM', 'goog.i18n.DateTimePatterns_ar_KW', 'goog.i18n.DateTimePatterns_ar_LB', 'goog.i18n.DateTimePatterns_ar_LY', 'goog.i18n.DateTimePatterns_ar_MA', 'goog.i18n.DateTimePatterns_ar_MR', 'goog.i18n.DateTimePatterns_ar_OM', 'goog.i18n.DateTimePatterns_ar_PS', 'goog.i18n.DateTimePatterns_ar_QA', 'goog.i18n.DateTimePatterns_ar_SA', 'goog.i18n.DateTimePatterns_ar_SD', 'goog.i18n.DateTimePatterns_ar_SO', 'goog.i18n.DateTimePatterns_ar_SS', 'goog.i18n.DateTimePatterns_ar_SY', 'goog.i18n.DateTimePatterns_ar_TD', 'goog.i18n.DateTimePatterns_ar_TN', 'goog.i18n.DateTimePatterns_ar_XB', 'goog.i18n.DateTimePatterns_ar_YE', 'goog.i18n.DateTimePatterns_as', 'goog.i18n.DateTimePatterns_as_IN', 'goog.i18n.DateTimePatterns_asa', 'goog.i18n.DateTimePatterns_asa_TZ', 'goog.i18n.DateTimePatterns_ast', 'goog.i18n.DateTimePatterns_ast_ES', 'goog.i18n.DateTimePatterns_az_Cyrl', 'goog.i18n.DateTimePatterns_az_Cyrl_AZ', 'goog.i18n.DateTimePatterns_az_Latn', 'goog.i18n.DateTimePatterns_az_Latn_AZ', 'goog.i18n.DateTimePatterns_bas', 'goog.i18n.DateTimePatterns_bas_CM', 'goog.i18n.DateTimePatterns_be_BY', 'goog.i18n.DateTimePatterns_bem', 'goog.i18n.DateTimePatterns_bem_ZM', 'goog.i18n.DateTimePatterns_bez', 'goog.i18n.DateTimePatterns_bez_TZ', 'goog.i18n.DateTimePatterns_bg_BG', 'goog.i18n.DateTimePatterns_bm', 'goog.i18n.DateTimePatterns_bm_ML', 'goog.i18n.DateTimePatterns_bn_BD', 'goog.i18n.DateTimePatterns_bn_IN', 'goog.i18n.DateTimePatterns_bo', 'goog.i18n.DateTimePatterns_bo_CN', 'goog.i18n.DateTimePatterns_bo_IN', 'goog.i18n.DateTimePatterns_br_FR', 'goog.i18n.DateTimePatterns_brx', 'goog.i18n.DateTimePatterns_brx_IN', 'goog.i18n.DateTimePatterns_bs_Cyrl', 'goog.i18n.DateTimePatterns_bs_Cyrl_BA', 'goog.i18n.DateTimePatterns_bs_Latn', 'goog.i18n.DateTimePatterns_bs_Latn_BA', 'goog.i18n.DateTimePatterns_ca_AD', 'goog.i18n.DateTimePatterns_ca_ES', 'goog.i18n.DateTimePatterns_ca_FR', 'goog.i18n.DateTimePatterns_ca_IT', 'goog.i18n.DateTimePatterns_ccp', 'goog.i18n.DateTimePatterns_ccp_BD', 'goog.i18n.DateTimePatterns_ccp_IN', 'goog.i18n.DateTimePatterns_ce', 'goog.i18n.DateTimePatterns_ce_RU', 'goog.i18n.DateTimePatterns_ceb', 'goog.i18n.DateTimePatterns_ceb_PH', 'goog.i18n.DateTimePatterns_cgg', 'goog.i18n.DateTimePatterns_cgg_UG', 'goog.i18n.DateTimePatterns_chr_US', 'goog.i18n.DateTimePatterns_ckb', 'goog.i18n.DateTimePatterns_ckb_IQ', 'goog.i18n.DateTimePatterns_ckb_IR', 'goog.i18n.DateTimePatterns_cs_CZ', 'goog.i18n.DateTimePatterns_cy_GB', 'goog.i18n.DateTimePatterns_da_DK', 'goog.i18n.DateTimePatterns_da_GL', 'goog.i18n.DateTimePatterns_dav', 'goog.i18n.DateTimePatterns_dav_KE', 'goog.i18n.DateTimePatterns_de_BE', 'goog.i18n.DateTimePatterns_de_DE', 'goog.i18n.DateTimePatterns_de_IT', 'goog.i18n.DateTimePatterns_de_LI', 'goog.i18n.DateTimePatterns_de_LU', 'goog.i18n.DateTimePatterns_dje', 'goog.i18n.DateTimePatterns_dje_NE', 'goog.i18n.DateTimePatterns_dsb', 'goog.i18n.DateTimePatterns_dsb_DE', 'goog.i18n.DateTimePatterns_dua', 'goog.i18n.DateTimePatterns_dua_CM', 'goog.i18n.DateTimePatterns_dyo', 'goog.i18n.DateTimePatterns_dyo_SN', 'goog.i18n.DateTimePatterns_dz', 'goog.i18n.DateTimePatterns_dz_BT', 'goog.i18n.DateTimePatterns_ebu', 'goog.i18n.DateTimePatterns_ebu_KE', 'goog.i18n.DateTimePatterns_ee', 'goog.i18n.DateTimePatterns_ee_GH', 'goog.i18n.DateTimePatterns_ee_TG', 'goog.i18n.DateTimePatterns_el_CY', 'goog.i18n.DateTimePatterns_el_GR', 'goog.i18n.DateTimePatterns_en_001', 'goog.i18n.DateTimePatterns_en_150', 'goog.i18n.DateTimePatterns_en_AE', 'goog.i18n.DateTimePatterns_en_AG', 'goog.i18n.DateTimePatterns_en_AI', 'goog.i18n.DateTimePatterns_en_AS', 'goog.i18n.DateTimePatterns_en_AT', 'goog.i18n.DateTimePatterns_en_BB', 'goog.i18n.DateTimePatterns_en_BE', 'goog.i18n.DateTimePatterns_en_BI', 'goog.i18n.DateTimePatterns_en_BM', 'goog.i18n.DateTimePatterns_en_BS', 'goog.i18n.DateTimePatterns_en_BW', 'goog.i18n.DateTimePatterns_en_BZ', 'goog.i18n.DateTimePatterns_en_CC', 'goog.i18n.DateTimePatterns_en_CH', 'goog.i18n.DateTimePatterns_en_CK', 'goog.i18n.DateTimePatterns_en_CM', 'goog.i18n.DateTimePatterns_en_CX', 'goog.i18n.DateTimePatterns_en_CY', 'goog.i18n.DateTimePatterns_en_DE', 'goog.i18n.DateTimePatterns_en_DG', 'goog.i18n.DateTimePatterns_en_DK', 'goog.i18n.DateTimePatterns_en_DM', 'goog.i18n.DateTimePatterns_en_ER', 'goog.i18n.DateTimePatterns_en_FI', 'goog.i18n.DateTimePatterns_en_FJ', 'goog.i18n.DateTimePatterns_en_FK', 'goog.i18n.DateTimePatterns_en_FM', 'goog.i18n.DateTimePatterns_en_GD', 'goog.i18n.DateTimePatterns_en_GG', 'goog.i18n.DateTimePatterns_en_GH', 'goog.i18n.DateTimePatterns_en_GI', 'goog.i18n.DateTimePatterns_en_GM', 'goog.i18n.DateTimePatterns_en_GU', 'goog.i18n.DateTimePatterns_en_GY', 'goog.i18n.DateTimePatterns_en_HK', 'goog.i18n.DateTimePatterns_en_IL', 'goog.i18n.DateTimePatterns_en_IM', 'goog.i18n.DateTimePatterns_en_IO', 'goog.i18n.DateTimePatterns_en_JE', 'goog.i18n.DateTimePatterns_en_JM', 'goog.i18n.DateTimePatterns_en_KE', 'goog.i18n.DateTimePatterns_en_KI', 'goog.i18n.DateTimePatterns_en_KN', 'goog.i18n.DateTimePatterns_en_KY', 'goog.i18n.DateTimePatterns_en_LC', 'goog.i18n.DateTimePatterns_en_LR', 'goog.i18n.DateTimePatterns_en_LS', 'goog.i18n.DateTimePatterns_en_MG', 'goog.i18n.DateTimePatterns_en_MH', 'goog.i18n.DateTimePatterns_en_MO', 'goog.i18n.DateTimePatterns_en_MP', 'goog.i18n.DateTimePatterns_en_MS', 'goog.i18n.DateTimePatterns_en_MT', 'goog.i18n.DateTimePatterns_en_MU', 'goog.i18n.DateTimePatterns_en_MW', 'goog.i18n.DateTimePatterns_en_MY', 'goog.i18n.DateTimePatterns_en_NA', 'goog.i18n.DateTimePatterns_en_NF', 'goog.i18n.DateTimePatterns_en_NG', 'goog.i18n.DateTimePatterns_en_NL', 'goog.i18n.DateTimePatterns_en_NR', 'goog.i18n.DateTimePatterns_en_NU', 'goog.i18n.DateTimePatterns_en_NZ', 'goog.i18n.DateTimePatterns_en_PG', 'goog.i18n.DateTimePatterns_en_PH', 'goog.i18n.DateTimePatterns_en_PK', 'goog.i18n.DateTimePatterns_en_PN', 'goog.i18n.DateTimePatterns_en_PR', 'goog.i18n.DateTimePatterns_en_PW', 'goog.i18n.DateTimePatterns_en_RW', 'goog.i18n.DateTimePatterns_en_SB', 'goog.i18n.DateTimePatterns_en_SC', 'goog.i18n.DateTimePatterns_en_SD', 'goog.i18n.DateTimePatterns_en_SE', 'goog.i18n.DateTimePatterns_en_SH', 'goog.i18n.DateTimePatterns_en_SI', 'goog.i18n.DateTimePatterns_en_SL', 'goog.i18n.DateTimePatterns_en_SS', 'goog.i18n.DateTimePatterns_en_SX', 'goog.i18n.DateTimePatterns_en_SZ', 'goog.i18n.DateTimePatterns_en_TC', 'goog.i18n.DateTimePatterns_en_TK', 'goog.i18n.DateTimePatterns_en_TO', 'goog.i18n.DateTimePatterns_en_TT', 'goog.i18n.DateTimePatterns_en_TV', 'goog.i18n.DateTimePatterns_en_TZ', 'goog.i18n.DateTimePatterns_en_UG', 'goog.i18n.DateTimePatterns_en_UM', 'goog.i18n.DateTimePatterns_en_US_POSIX', 'goog.i18n.DateTimePatterns_en_VC', 'goog.i18n.DateTimePatterns_en_VG', 'goog.i18n.DateTimePatterns_en_VI', 'goog.i18n.DateTimePatterns_en_VU', 'goog.i18n.DateTimePatterns_en_WS', 'goog.i18n.DateTimePatterns_en_XA', 'goog.i18n.DateTimePatterns_en_ZM', 'goog.i18n.DateTimePatterns_en_ZW', 'goog.i18n.DateTimePatterns_eo', 'goog.i18n.DateTimePatterns_eo_001', 'goog.i18n.DateTimePatterns_es_AR', 'goog.i18n.DateTimePatterns_es_BO', 'goog.i18n.DateTimePatterns_es_BR', 'goog.i18n.DateTimePatterns_es_BZ', 'goog.i18n.DateTimePatterns_es_CL', 'goog.i18n.DateTimePatterns_es_CO', 'goog.i18n.DateTimePatterns_es_CR', 'goog.i18n.DateTimePatterns_es_CU', 'goog.i18n.DateTimePatterns_es_DO', 'goog.i18n.DateTimePatterns_es_EA', 'goog.i18n.DateTimePatterns_es_EC', 'goog.i18n.DateTimePatterns_es_GQ', 'goog.i18n.DateTimePatterns_es_GT', 'goog.i18n.DateTimePatterns_es_HN', 'goog.i18n.DateTimePatterns_es_IC', 'goog.i18n.DateTimePatterns_es_NI', 'goog.i18n.DateTimePatterns_es_PA', 'goog.i18n.DateTimePatterns_es_PE', 'goog.i18n.DateTimePatterns_es_PH', 'goog.i18n.DateTimePatterns_es_PR', 'goog.i18n.DateTimePatterns_es_PY', 'goog.i18n.DateTimePatterns_es_SV', 'goog.i18n.DateTimePatterns_es_UY', 'goog.i18n.DateTimePatterns_es_VE', 'goog.i18n.DateTimePatterns_et_EE', 'goog.i18n.DateTimePatterns_eu_ES', 'goog.i18n.DateTimePatterns_ewo', 'goog.i18n.DateTimePatterns_ewo_CM', 'goog.i18n.DateTimePatterns_fa_AF', 'goog.i18n.DateTimePatterns_fa_IR', 'goog.i18n.DateTimePatterns_ff', 'goog.i18n.DateTimePatterns_ff_Latn', 'goog.i18n.DateTimePatterns_ff_Latn_BF', 'goog.i18n.DateTimePatterns_ff_Latn_CM', 'goog.i18n.DateTimePatterns_ff_Latn_GH', 'goog.i18n.DateTimePatterns_ff_Latn_GM', 'goog.i18n.DateTimePatterns_ff_Latn_GN', 'goog.i18n.DateTimePatterns_ff_Latn_GW', 'goog.i18n.DateTimePatterns_ff_Latn_LR', 'goog.i18n.DateTimePatterns_ff_Latn_MR', 'goog.i18n.DateTimePatterns_ff_Latn_NE', 'goog.i18n.DateTimePatterns_ff_Latn_NG', 'goog.i18n.DateTimePatterns_ff_Latn_SL', 'goog.i18n.DateTimePatterns_ff_Latn_SN', 'goog.i18n.DateTimePatterns_fi_FI', 'goog.i18n.DateTimePatterns_fil_PH', 'goog.i18n.DateTimePatterns_fo', 'goog.i18n.DateTimePatterns_fo_DK', 'goog.i18n.DateTimePatterns_fo_FO', 'goog.i18n.DateTimePatterns_fr_BE', 'goog.i18n.DateTimePatterns_fr_BF', 'goog.i18n.DateTimePatterns_fr_BI', 'goog.i18n.DateTimePatterns_fr_BJ', 'goog.i18n.DateTimePatterns_fr_BL', 'goog.i18n.DateTimePatterns_fr_CD', 'goog.i18n.DateTimePatterns_fr_CF', 'goog.i18n.DateTimePatterns_fr_CG', 'goog.i18n.DateTimePatterns_fr_CH', 'goog.i18n.DateTimePatterns_fr_CI', 'goog.i18n.DateTimePatterns_fr_CM', 'goog.i18n.DateTimePatterns_fr_DJ', 'goog.i18n.DateTimePatterns_fr_DZ', 'goog.i18n.DateTimePatterns_fr_FR', 'goog.i18n.DateTimePatterns_fr_GA', 'goog.i18n.DateTimePatterns_fr_GF', 'goog.i18n.DateTimePatterns_fr_GN', 'goog.i18n.DateTimePatterns_fr_GP', 'goog.i18n.DateTimePatterns_fr_GQ', 'goog.i18n.DateTimePatterns_fr_HT', 'goog.i18n.DateTimePatterns_fr_KM', 'goog.i18n.DateTimePatterns_fr_LU', 'goog.i18n.DateTimePatterns_fr_MA', 'goog.i18n.DateTimePatterns_fr_MC', 'goog.i18n.DateTimePatterns_fr_MF', 'goog.i18n.DateTimePatterns_fr_MG', 'goog.i18n.DateTimePatterns_fr_ML', 'goog.i18n.DateTimePatterns_fr_MQ', 'goog.i18n.DateTimePatterns_fr_MR', 'goog.i18n.DateTimePatterns_fr_MU', 'goog.i18n.DateTimePatterns_fr_NC', 'goog.i18n.DateTimePatterns_fr_NE', 'goog.i18n.DateTimePatterns_fr_PF', 'goog.i18n.DateTimePatterns_fr_PM', 'goog.i18n.DateTimePatterns_fr_RE', 'goog.i18n.DateTimePatterns_fr_RW', 'goog.i18n.DateTimePatterns_fr_SC', 'goog.i18n.DateTimePatterns_fr_SN', 'goog.i18n.DateTimePatterns_fr_SY', 'goog.i18n.DateTimePatterns_fr_TD', 'goog.i18n.DateTimePatterns_fr_TG', 'goog.i18n.DateTimePatterns_fr_TN', 'goog.i18n.DateTimePatterns_fr_VU', 'goog.i18n.DateTimePatterns_fr_WF', 'goog.i18n.DateTimePatterns_fr_YT', 'goog.i18n.DateTimePatterns_fur', 'goog.i18n.DateTimePatterns_fur_IT', 'goog.i18n.DateTimePatterns_fy', 'goog.i18n.DateTimePatterns_fy_NL', 'goog.i18n.DateTimePatterns_ga_IE', 'goog.i18n.DateTimePatterns_gd', 'goog.i18n.DateTimePatterns_gd_GB', 'goog.i18n.DateTimePatterns_gl_ES', 'goog.i18n.DateTimePatterns_gsw_CH', 'goog.i18n.DateTimePatterns_gsw_FR', 'goog.i18n.DateTimePatterns_gsw_LI', 'goog.i18n.DateTimePatterns_gu_IN', 'goog.i18n.DateTimePatterns_guz', 'goog.i18n.DateTimePatterns_guz_KE', 'goog.i18n.DateTimePatterns_gv', 'goog.i18n.DateTimePatterns_gv_IM', 'goog.i18n.DateTimePatterns_ha', 'goog.i18n.DateTimePatterns_ha_GH', 'goog.i18n.DateTimePatterns_ha_NE', 'goog.i18n.DateTimePatterns_ha_NG', 'goog.i18n.DateTimePatterns_haw_US', 'goog.i18n.DateTimePatterns_he_IL', 'goog.i18n.DateTimePatterns_hi_IN', 'goog.i18n.DateTimePatterns_hr_BA', 'goog.i18n.DateTimePatterns_hr_HR', 'goog.i18n.DateTimePatterns_hsb', 'goog.i18n.DateTimePatterns_hsb_DE', 'goog.i18n.DateTimePatterns_hu_HU', 'goog.i18n.DateTimePatterns_hy_AM', 'goog.i18n.DateTimePatterns_ia', 'goog.i18n.DateTimePatterns_ia_001', 'goog.i18n.DateTimePatterns_id_ID', 'goog.i18n.DateTimePatterns_ig', 'goog.i18n.DateTimePatterns_ig_NG', 'goog.i18n.DateTimePatterns_ii', 'goog.i18n.DateTimePatterns_ii_CN', 'goog.i18n.DateTimePatterns_is_IS', 'goog.i18n.DateTimePatterns_it_CH', 'goog.i18n.DateTimePatterns_it_IT', 'goog.i18n.DateTimePatterns_it_SM', 'goog.i18n.DateTimePatterns_it_VA', 'goog.i18n.DateTimePatterns_ja_JP', 'goog.i18n.DateTimePatterns_jgo', 'goog.i18n.DateTimePatterns_jgo_CM', 'goog.i18n.DateTimePatterns_jmc', 'goog.i18n.DateTimePatterns_jmc_TZ', 'goog.i18n.DateTimePatterns_jv', 'goog.i18n.DateTimePatterns_jv_ID', 'goog.i18n.DateTimePatterns_ka_GE', 'goog.i18n.DateTimePatterns_kab', 'goog.i18n.DateTimePatterns_kab_DZ', 'goog.i18n.DateTimePatterns_kam', 'goog.i18n.DateTimePatterns_kam_KE', 'goog.i18n.DateTimePatterns_kde', 'goog.i18n.DateTimePatterns_kde_TZ', 'goog.i18n.DateTimePatterns_kea', 'goog.i18n.DateTimePatterns_kea_CV', 'goog.i18n.DateTimePatterns_khq', 'goog.i18n.DateTimePatterns_khq_ML', 'goog.i18n.DateTimePatterns_ki', 'goog.i18n.DateTimePatterns_ki_KE', 'goog.i18n.DateTimePatterns_kk_KZ', 'goog.i18n.DateTimePatterns_kkj', 'goog.i18n.DateTimePatterns_kkj_CM', 'goog.i18n.DateTimePatterns_kl', 'goog.i18n.DateTimePatterns_kl_GL', 'goog.i18n.DateTimePatterns_kln', 'goog.i18n.DateTimePatterns_kln_KE', 'goog.i18n.DateTimePatterns_km_KH', 'goog.i18n.DateTimePatterns_kn_IN', 'goog.i18n.DateTimePatterns_ko_KP', 'goog.i18n.DateTimePatterns_ko_KR', 'goog.i18n.DateTimePatterns_kok', 'goog.i18n.DateTimePatterns_kok_IN', 'goog.i18n.DateTimePatterns_ks', 'goog.i18n.DateTimePatterns_ks_IN', 'goog.i18n.DateTimePatterns_ksb', 'goog.i18n.DateTimePatterns_ksb_TZ', 'goog.i18n.DateTimePatterns_ksf', 'goog.i18n.DateTimePatterns_ksf_CM', 'goog.i18n.DateTimePatterns_ksh', 'goog.i18n.DateTimePatterns_ksh_DE', 'goog.i18n.DateTimePatterns_ku', 'goog.i18n.DateTimePatterns_ku_TR', 'goog.i18n.DateTimePatterns_kw', 'goog.i18n.DateTimePatterns_kw_GB', 'goog.i18n.DateTimePatterns_ky_KG', 'goog.i18n.DateTimePatterns_lag', 'goog.i18n.DateTimePatterns_lag_TZ', 'goog.i18n.DateTimePatterns_lb', 'goog.i18n.DateTimePatterns_lb_LU', 'goog.i18n.DateTimePatterns_lg', 'goog.i18n.DateTimePatterns_lg_UG', 'goog.i18n.DateTimePatterns_lkt', 'goog.i18n.DateTimePatterns_lkt_US', 'goog.i18n.DateTimePatterns_ln_AO', 'goog.i18n.DateTimePatterns_ln_CD', 'goog.i18n.DateTimePatterns_ln_CF', 'goog.i18n.DateTimePatterns_ln_CG', 'goog.i18n.DateTimePatterns_lo_LA', 'goog.i18n.DateTimePatterns_lrc', 'goog.i18n.DateTimePatterns_lrc_IQ', 'goog.i18n.DateTimePatterns_lrc_IR', 'goog.i18n.DateTimePatterns_lt_LT', 'goog.i18n.DateTimePatterns_lu', 'goog.i18n.DateTimePatterns_lu_CD', 'goog.i18n.DateTimePatterns_luo', 'goog.i18n.DateTimePatterns_luo_KE', 'goog.i18n.DateTimePatterns_luy', 'goog.i18n.DateTimePatterns_luy_KE', 'goog.i18n.DateTimePatterns_lv_LV', 'goog.i18n.DateTimePatterns_mas', 'goog.i18n.DateTimePatterns_mas_KE', 'goog.i18n.DateTimePatterns_mas_TZ', 'goog.i18n.DateTimePatterns_mer', 'goog.i18n.DateTimePatterns_mer_KE', 'goog.i18n.DateTimePatterns_mfe', 'goog.i18n.DateTimePatterns_mfe_MU', 'goog.i18n.DateTimePatterns_mg', 'goog.i18n.DateTimePatterns_mg_MG', 'goog.i18n.DateTimePatterns_mgh', 'goog.i18n.DateTimePatterns_mgh_MZ', 'goog.i18n.DateTimePatterns_mgo', 'goog.i18n.DateTimePatterns_mgo_CM', 'goog.i18n.DateTimePatterns_mi', 'goog.i18n.DateTimePatterns_mi_NZ', 'goog.i18n.DateTimePatterns_mk_MK', 'goog.i18n.DateTimePatterns_ml_IN', 'goog.i18n.DateTimePatterns_mn_MN', 'goog.i18n.DateTimePatterns_mr_IN', 'goog.i18n.DateTimePatterns_ms_BN', 'goog.i18n.DateTimePatterns_ms_MY', 'goog.i18n.DateTimePatterns_ms_SG', 'goog.i18n.DateTimePatterns_mt_MT', 'goog.i18n.DateTimePatterns_mua', 'goog.i18n.DateTimePatterns_mua_CM', 'goog.i18n.DateTimePatterns_my_MM', 'goog.i18n.DateTimePatterns_mzn', 'goog.i18n.DateTimePatterns_mzn_IR', 'goog.i18n.DateTimePatterns_naq', 'goog.i18n.DateTimePatterns_naq_NA', 'goog.i18n.DateTimePatterns_nb_NO', 'goog.i18n.DateTimePatterns_nb_SJ', 'goog.i18n.DateTimePatterns_nd', 'goog.i18n.DateTimePatterns_nd_ZW', 'goog.i18n.DateTimePatterns_nds', 'goog.i18n.DateTimePatterns_nds_DE', 'goog.i18n.DateTimePatterns_nds_NL', 'goog.i18n.DateTimePatterns_ne_IN', 'goog.i18n.DateTimePatterns_ne_NP', 'goog.i18n.DateTimePatterns_nl_AW', 'goog.i18n.DateTimePatterns_nl_BE', 'goog.i18n.DateTimePatterns_nl_BQ', 'goog.i18n.DateTimePatterns_nl_CW', 'goog.i18n.DateTimePatterns_nl_NL', 'goog.i18n.DateTimePatterns_nl_SR', 'goog.i18n.DateTimePatterns_nl_SX', 'goog.i18n.DateTimePatterns_nmg', 'goog.i18n.DateTimePatterns_nmg_CM', 'goog.i18n.DateTimePatterns_nn', 'goog.i18n.DateTimePatterns_nn_NO', 'goog.i18n.DateTimePatterns_nnh', 'goog.i18n.DateTimePatterns_nnh_CM', 'goog.i18n.DateTimePatterns_nus', 'goog.i18n.DateTimePatterns_nus_SS', 'goog.i18n.DateTimePatterns_nyn', 'goog.i18n.DateTimePatterns_nyn_UG', 'goog.i18n.DateTimePatterns_om', 'goog.i18n.DateTimePatterns_om_ET', 'goog.i18n.DateTimePatterns_om_KE', 'goog.i18n.DateTimePatterns_or_IN', 'goog.i18n.DateTimePatterns_os', 'goog.i18n.DateTimePatterns_os_GE', 'goog.i18n.DateTimePatterns_os_RU', 'goog.i18n.DateTimePatterns_pa_Arab', 'goog.i18n.DateTimePatterns_pa_Arab_PK', 'goog.i18n.DateTimePatterns_pa_Guru', 'goog.i18n.DateTimePatterns_pa_Guru_IN', 'goog.i18n.DateTimePatterns_pl_PL', 'goog.i18n.DateTimePatterns_ps', 'goog.i18n.DateTimePatterns_ps_AF', 'goog.i18n.DateTimePatterns_ps_PK', 'goog.i18n.DateTimePatterns_pt_AO', 'goog.i18n.DateTimePatterns_pt_CH', 'goog.i18n.DateTimePatterns_pt_CV', 'goog.i18n.DateTimePatterns_pt_GQ', 'goog.i18n.DateTimePatterns_pt_GW', 'goog.i18n.DateTimePatterns_pt_LU', 'goog.i18n.DateTimePatterns_pt_MO', 'goog.i18n.DateTimePatterns_pt_MZ', 'goog.i18n.DateTimePatterns_pt_ST', 'goog.i18n.DateTimePatterns_pt_TL', 'goog.i18n.DateTimePatterns_qu', 'goog.i18n.DateTimePatterns_qu_BO', 'goog.i18n.DateTimePatterns_qu_EC', 'goog.i18n.DateTimePatterns_qu_PE', 'goog.i18n.DateTimePatterns_rm', 'goog.i18n.DateTimePatterns_rm_CH', 'goog.i18n.DateTimePatterns_rn', 'goog.i18n.DateTimePatterns_rn_BI', 'goog.i18n.DateTimePatterns_ro_MD', 'goog.i18n.DateTimePatterns_ro_RO', 'goog.i18n.DateTimePatterns_rof', 'goog.i18n.DateTimePatterns_rof_TZ', 'goog.i18n.DateTimePatterns_ru_BY', 'goog.i18n.DateTimePatterns_ru_KG', 'goog.i18n.DateTimePatterns_ru_KZ', 'goog.i18n.DateTimePatterns_ru_MD', 'goog.i18n.DateTimePatterns_ru_RU', 'goog.i18n.DateTimePatterns_ru_UA', 'goog.i18n.DateTimePatterns_rw', 'goog.i18n.DateTimePatterns_rw_RW', 'goog.i18n.DateTimePatterns_rwk', 'goog.i18n.DateTimePatterns_rwk_TZ', 'goog.i18n.DateTimePatterns_sah', 'goog.i18n.DateTimePatterns_sah_RU', 'goog.i18n.DateTimePatterns_saq', 'goog.i18n.DateTimePatterns_saq_KE', 'goog.i18n.DateTimePatterns_sbp', 'goog.i18n.DateTimePatterns_sbp_TZ', 'goog.i18n.DateTimePatterns_sd', 'goog.i18n.DateTimePatterns_sd_PK', 'goog.i18n.DateTimePatterns_se', 'goog.i18n.DateTimePatterns_se_FI', 'goog.i18n.DateTimePatterns_se_NO', 'goog.i18n.DateTimePatterns_se_SE', 'goog.i18n.DateTimePatterns_seh', 'goog.i18n.DateTimePatterns_seh_MZ', 'goog.i18n.DateTimePatterns_ses', 'goog.i18n.DateTimePatterns_ses_ML', 'goog.i18n.DateTimePatterns_sg', 'goog.i18n.DateTimePatterns_sg_CF', 'goog.i18n.DateTimePatterns_shi', 'goog.i18n.DateTimePatterns_shi_Latn', 'goog.i18n.DateTimePatterns_shi_Latn_MA', 'goog.i18n.DateTimePatterns_shi_Tfng', 'goog.i18n.DateTimePatterns_shi_Tfng_MA', 'goog.i18n.DateTimePatterns_si_LK', 'goog.i18n.DateTimePatterns_sk_SK', 'goog.i18n.DateTimePatterns_sl_SI', 'goog.i18n.DateTimePatterns_smn', 'goog.i18n.DateTimePatterns_smn_FI', 'goog.i18n.DateTimePatterns_sn', 'goog.i18n.DateTimePatterns_sn_ZW', 'goog.i18n.DateTimePatterns_so', 'goog.i18n.DateTimePatterns_so_DJ', 'goog.i18n.DateTimePatterns_so_ET', 'goog.i18n.DateTimePatterns_so_KE', 'goog.i18n.DateTimePatterns_so_SO', 'goog.i18n.DateTimePatterns_sq_AL', 'goog.i18n.DateTimePatterns_sq_MK', 'goog.i18n.DateTimePatterns_sq_XK', 'goog.i18n.DateTimePatterns_sr_Cyrl', 'goog.i18n.DateTimePatterns_sr_Cyrl_BA', 'goog.i18n.DateTimePatterns_sr_Cyrl_ME', 'goog.i18n.DateTimePatterns_sr_Cyrl_RS', 'goog.i18n.DateTimePatterns_sr_Cyrl_XK', 'goog.i18n.DateTimePatterns_sr_Latn_BA', 'goog.i18n.DateTimePatterns_sr_Latn_ME', 'goog.i18n.DateTimePatterns_sr_Latn_RS', 'goog.i18n.DateTimePatterns_sr_Latn_XK', 'goog.i18n.DateTimePatterns_sv_AX', 'goog.i18n.DateTimePatterns_sv_FI', 'goog.i18n.DateTimePatterns_sv_SE', 'goog.i18n.DateTimePatterns_sw_CD', 'goog.i18n.DateTimePatterns_sw_KE', 'goog.i18n.DateTimePatterns_sw_TZ', 'goog.i18n.DateTimePatterns_sw_UG', 'goog.i18n.DateTimePatterns_ta_IN', 'goog.i18n.DateTimePatterns_ta_LK', 'goog.i18n.DateTimePatterns_ta_MY', 'goog.i18n.DateTimePatterns_ta_SG', 'goog.i18n.DateTimePatterns_te_IN', 'goog.i18n.DateTimePatterns_teo', 'goog.i18n.DateTimePatterns_teo_KE', 'goog.i18n.DateTimePatterns_teo_UG', 'goog.i18n.DateTimePatterns_tg', 'goog.i18n.DateTimePatterns_tg_TJ', 'goog.i18n.DateTimePatterns_th_TH', 'goog.i18n.DateTimePatterns_ti', 'goog.i18n.DateTimePatterns_ti_ER', 'goog.i18n.DateTimePatterns_ti_ET', 'goog.i18n.DateTimePatterns_tk', 'goog.i18n.DateTimePatterns_tk_TM', 'goog.i18n.DateTimePatterns_to', 'goog.i18n.DateTimePatterns_to_TO', 'goog.i18n.DateTimePatterns_tr_CY', 'goog.i18n.DateTimePatterns_tr_TR', 'goog.i18n.DateTimePatterns_tt', 'goog.i18n.DateTimePatterns_tt_RU', 'goog.i18n.DateTimePatterns_twq', 'goog.i18n.DateTimePatterns_twq_NE', 'goog.i18n.DateTimePatterns_tzm', 'goog.i18n.DateTimePatterns_tzm_MA', 'goog.i18n.DateTimePatterns_ug', 'goog.i18n.DateTimePatterns_ug_CN', 'goog.i18n.DateTimePatterns_uk_UA', 'goog.i18n.DateTimePatterns_ur_IN', 'goog.i18n.DateTimePatterns_ur_PK', 'goog.i18n.DateTimePatterns_uz_Arab', 'goog.i18n.DateTimePatterns_uz_Arab_AF', 'goog.i18n.DateTimePatterns_uz_Cyrl', 'goog.i18n.DateTimePatterns_uz_Cyrl_UZ', 'goog.i18n.DateTimePatterns_uz_Latn', 'goog.i18n.DateTimePatterns_uz_Latn_UZ', 'goog.i18n.DateTimePatterns_vai', 'goog.i18n.DateTimePatterns_vai_Latn', 'goog.i18n.DateTimePatterns_vai_Latn_LR', 'goog.i18n.DateTimePatterns_vai_Vaii', 'goog.i18n.DateTimePatterns_vai_Vaii_LR', 'goog.i18n.DateTimePatterns_vi_VN', 'goog.i18n.DateTimePatterns_vun', 'goog.i18n.DateTimePatterns_vun_TZ', 'goog.i18n.DateTimePatterns_wae', 'goog.i18n.DateTimePatterns_wae_CH', 'goog.i18n.DateTimePatterns_wo', 'goog.i18n.DateTimePatterns_wo_SN', 'goog.i18n.DateTimePatterns_xh', 'goog.i18n.DateTimePatterns_xh_ZA', 'goog.i18n.DateTimePatterns_xog', 'goog.i18n.DateTimePatterns_xog_UG', 'goog.i18n.DateTimePatterns_yav', 'goog.i18n.DateTimePatterns_yav_CM', 'goog.i18n.DateTimePatterns_yi', 'goog.i18n.DateTimePatterns_yi_001', 'goog.i18n.DateTimePatterns_yo', 'goog.i18n.DateTimePatterns_yo_BJ', 'goog.i18n.DateTimePatterns_yo_NG', 'goog.i18n.DateTimePatterns_yue', 'goog.i18n.DateTimePatterns_yue_Hans', 'goog.i18n.DateTimePatterns_yue_Hans_CN', 'goog.i18n.DateTimePatterns_yue_Hant', 'goog.i18n.DateTimePatterns_yue_Hant_HK', 'goog.i18n.DateTimePatterns_zgh', 'goog.i18n.DateTimePatterns_zgh_MA', 'goog.i18n.DateTimePatterns_zh_Hans', 'goog.i18n.DateTimePatterns_zh_Hans_CN', 'goog.i18n.DateTimePatterns_zh_Hans_HK', 'goog.i18n.DateTimePatterns_zh_Hans_MO', 'goog.i18n.DateTimePatterns_zh_Hans_SG', 'goog.i18n.DateTimePatterns_zh_Hant', 'goog.i18n.DateTimePatterns_zh_Hant_HK', 'goog.i18n.DateTimePatterns_zh_Hant_MO', 'goog.i18n.DateTimePatterns_zh_Hant_TW', 'goog.i18n.DateTimePatterns_zu_ZA'], ['goog.i18n.DateTimePatterns'], {});
goog.addDependency('i18n/datetimesymbols.js', ['goog.i18n.DateTimeSymbols', 'goog.i18n.DateTimeSymbolsType', 'goog.i18n.DateTimeSymbols_af', 'goog.i18n.DateTimeSymbols_am', 'goog.i18n.DateTimeSymbols_ar', 'goog.i18n.DateTimeSymbols_ar_DZ', 'goog.i18n.DateTimeSymbols_ar_EG', 'goog.i18n.DateTimeSymbols_az', 'goog.i18n.DateTimeSymbols_be', 'goog.i18n.DateTimeSymbols_bg', 'goog.i18n.DateTimeSymbols_bn', 'goog.i18n.DateTimeSymbols_br', 'goog.i18n.DateTimeSymbols_bs', 'goog.i18n.DateTimeSymbols_ca', 'goog.i18n.DateTimeSymbols_chr', 'goog.i18n.DateTimeSymbols_cs', 'goog.i18n.DateTimeSymbols_cy', 'goog.i18n.DateTimeSymbols_da', 'goog.i18n.DateTimeSymbols_de', 'goog.i18n.DateTimeSymbols_de_AT', 'goog.i18n.DateTimeSymbols_de_CH', 'goog.i18n.DateTimeSymbols_el', 'goog.i18n.DateTimeSymbols_en', 'goog.i18n.DateTimeSymbols_en_AU', 'goog.i18n.DateTimeSymbols_en_CA', 'goog.i18n.DateTimeSymbols_en_GB', 'goog.i18n.DateTimeSymbols_en_IE', 'goog.i18n.DateTimeSymbols_en_IN', 'goog.i18n.DateTimeSymbols_en_ISO', 'goog.i18n.DateTimeSymbols_en_SG', 'goog.i18n.DateTimeSymbols_en_US', 'goog.i18n.DateTimeSymbols_en_ZA', 'goog.i18n.DateTimeSymbols_es', 'goog.i18n.DateTimeSymbols_es_419', 'goog.i18n.DateTimeSymbols_es_ES', 'goog.i18n.DateTimeSymbols_es_MX', 'goog.i18n.DateTimeSymbols_es_US', 'goog.i18n.DateTimeSymbols_et', 'goog.i18n.DateTimeSymbols_eu', 'goog.i18n.DateTimeSymbols_fa', 'goog.i18n.DateTimeSymbols_fi', 'goog.i18n.DateTimeSymbols_fil', 'goog.i18n.DateTimeSymbols_fr', 'goog.i18n.DateTimeSymbols_fr_CA', 'goog.i18n.DateTimeSymbols_ga', 'goog.i18n.DateTimeSymbols_gl', 'goog.i18n.DateTimeSymbols_gsw', 'goog.i18n.DateTimeSymbols_gu', 'goog.i18n.DateTimeSymbols_haw', 'goog.i18n.DateTimeSymbols_he', 'goog.i18n.DateTimeSymbols_hi', 'goog.i18n.DateTimeSymbols_hr', 'goog.i18n.DateTimeSymbols_hu', 'goog.i18n.DateTimeSymbols_hy', 'goog.i18n.DateTimeSymbols_id', 'goog.i18n.DateTimeSymbols_in', 'goog.i18n.DateTimeSymbols_is', 'goog.i18n.DateTimeSymbols_it', 'goog.i18n.DateTimeSymbols_iw', 'goog.i18n.DateTimeSymbols_ja', 'goog.i18n.DateTimeSymbols_ka', 'goog.i18n.DateTimeSymbols_kk', 'goog.i18n.DateTimeSymbols_km', 'goog.i18n.DateTimeSymbols_kn', 'goog.i18n.DateTimeSymbols_ko', 'goog.i18n.DateTimeSymbols_ky', 'goog.i18n.DateTimeSymbols_ln', 'goog.i18n.DateTimeSymbols_lo', 'goog.i18n.DateTimeSymbols_lt', 'goog.i18n.DateTimeSymbols_lv', 'goog.i18n.DateTimeSymbols_mk', 'goog.i18n.DateTimeSymbols_ml', 'goog.i18n.DateTimeSymbols_mn', 'goog.i18n.DateTimeSymbols_mo', 'goog.i18n.DateTimeSymbols_mr', 'goog.i18n.DateTimeSymbols_ms', 'goog.i18n.DateTimeSymbols_mt', 'goog.i18n.DateTimeSymbols_my', 'goog.i18n.DateTimeSymbols_nb', 'goog.i18n.DateTimeSymbols_ne', 'goog.i18n.DateTimeSymbols_nl', 'goog.i18n.DateTimeSymbols_no', 'goog.i18n.DateTimeSymbols_no_NO', 'goog.i18n.DateTimeSymbols_or', 'goog.i18n.DateTimeSymbols_pa', 'goog.i18n.DateTimeSymbols_pl', 'goog.i18n.DateTimeSymbols_pt', 'goog.i18n.DateTimeSymbols_pt_BR', 'goog.i18n.DateTimeSymbols_pt_PT', 'goog.i18n.DateTimeSymbols_ro', 'goog.i18n.DateTimeSymbols_ru', 'goog.i18n.DateTimeSymbols_sh', 'goog.i18n.DateTimeSymbols_si', 'goog.i18n.DateTimeSymbols_sk', 'goog.i18n.DateTimeSymbols_sl', 'goog.i18n.DateTimeSymbols_sq', 'goog.i18n.DateTimeSymbols_sr', 'goog.i18n.DateTimeSymbols_sr_Latn', 'goog.i18n.DateTimeSymbols_sv', 'goog.i18n.DateTimeSymbols_sw', 'goog.i18n.DateTimeSymbols_ta', 'goog.i18n.DateTimeSymbols_te', 'goog.i18n.DateTimeSymbols_th', 'goog.i18n.DateTimeSymbols_tl', 'goog.i18n.DateTimeSymbols_tr', 'goog.i18n.DateTimeSymbols_uk', 'goog.i18n.DateTimeSymbols_ur', 'goog.i18n.DateTimeSymbols_uz', 'goog.i18n.DateTimeSymbols_vi', 'goog.i18n.DateTimeSymbols_zh', 'goog.i18n.DateTimeSymbols_zh_CN', 'goog.i18n.DateTimeSymbols_zh_HK', 'goog.i18n.DateTimeSymbols_zh_TW', 'goog.i18n.DateTimeSymbols_zu'], [], {});
goog.addDependency('i18n/datetimesymbolsext.js', ['goog.i18n.DateTimeSymbolsExt', 'goog.i18n.DateTimeSymbols_af_NA', 'goog.i18n.DateTimeSymbols_af_ZA', 'goog.i18n.DateTimeSymbols_agq', 'goog.i18n.DateTimeSymbols_agq_CM', 'goog.i18n.DateTimeSymbols_ak', 'goog.i18n.DateTimeSymbols_ak_GH', 'goog.i18n.DateTimeSymbols_am_ET', 'goog.i18n.DateTimeSymbols_ar_001', 'goog.i18n.DateTimeSymbols_ar_AE', 'goog.i18n.DateTimeSymbols_ar_BH', 'goog.i18n.DateTimeSymbols_ar_DJ', 'goog.i18n.DateTimeSymbols_ar_EH', 'goog.i18n.DateTimeSymbols_ar_ER', 'goog.i18n.DateTimeSymbols_ar_IL', 'goog.i18n.DateTimeSymbols_ar_IQ', 'goog.i18n.DateTimeSymbols_ar_JO', 'goog.i18n.DateTimeSymbols_ar_KM', 'goog.i18n.DateTimeSymbols_ar_KW', 'goog.i18n.DateTimeSymbols_ar_LB', 'goog.i18n.DateTimeSymbols_ar_LY', 'goog.i18n.DateTimeSymbols_ar_MA', 'goog.i18n.DateTimeSymbols_ar_MR', 'goog.i18n.DateTimeSymbols_ar_OM', 'goog.i18n.DateTimeSymbols_ar_PS', 'goog.i18n.DateTimeSymbols_ar_QA', 'goog.i18n.DateTimeSymbols_ar_SA', 'goog.i18n.DateTimeSymbols_ar_SD', 'goog.i18n.DateTimeSymbols_ar_SO', 'goog.i18n.DateTimeSymbols_ar_SS', 'goog.i18n.DateTimeSymbols_ar_SY', 'goog.i18n.DateTimeSymbols_ar_TD', 'goog.i18n.DateTimeSymbols_ar_TN', 'goog.i18n.DateTimeSymbols_ar_XB', 'goog.i18n.DateTimeSymbols_ar_YE', 'goog.i18n.DateTimeSymbols_as', 'goog.i18n.DateTimeSymbols_as_IN', 'goog.i18n.DateTimeSymbols_asa', 'goog.i18n.DateTimeSymbols_asa_TZ', 'goog.i18n.DateTimeSymbols_ast', 'goog.i18n.DateTimeSymbols_ast_ES', 'goog.i18n.DateTimeSymbols_az_Cyrl', 'goog.i18n.DateTimeSymbols_az_Cyrl_AZ', 'goog.i18n.DateTimeSymbols_az_Latn', 'goog.i18n.DateTimeSymbols_az_Latn_AZ', 'goog.i18n.DateTimeSymbols_bas', 'goog.i18n.DateTimeSymbols_bas_CM', 'goog.i18n.DateTimeSymbols_be_BY', 'goog.i18n.DateTimeSymbols_bem', 'goog.i18n.DateTimeSymbols_bem_ZM', 'goog.i18n.DateTimeSymbols_bez', 'goog.i18n.DateTimeSymbols_bez_TZ', 'goog.i18n.DateTimeSymbols_bg_BG', 'goog.i18n.DateTimeSymbols_bm', 'goog.i18n.DateTimeSymbols_bm_ML', 'goog.i18n.DateTimeSymbols_bn_BD', 'goog.i18n.DateTimeSymbols_bn_IN', 'goog.i18n.DateTimeSymbols_bo', 'goog.i18n.DateTimeSymbols_bo_CN', 'goog.i18n.DateTimeSymbols_bo_IN', 'goog.i18n.DateTimeSymbols_br_FR', 'goog.i18n.DateTimeSymbols_brx', 'goog.i18n.DateTimeSymbols_brx_IN', 'goog.i18n.DateTimeSymbols_bs_Cyrl', 'goog.i18n.DateTimeSymbols_bs_Cyrl_BA', 'goog.i18n.DateTimeSymbols_bs_Latn', 'goog.i18n.DateTimeSymbols_bs_Latn_BA', 'goog.i18n.DateTimeSymbols_ca_AD', 'goog.i18n.DateTimeSymbols_ca_ES', 'goog.i18n.DateTimeSymbols_ca_FR', 'goog.i18n.DateTimeSymbols_ca_IT', 'goog.i18n.DateTimeSymbols_ccp', 'goog.i18n.DateTimeSymbols_ccp_BD', 'goog.i18n.DateTimeSymbols_ccp_IN', 'goog.i18n.DateTimeSymbols_ce', 'goog.i18n.DateTimeSymbols_ce_RU', 'goog.i18n.DateTimeSymbols_ceb', 'goog.i18n.DateTimeSymbols_ceb_PH', 'goog.i18n.DateTimeSymbols_cgg', 'goog.i18n.DateTimeSymbols_cgg_UG', 'goog.i18n.DateTimeSymbols_chr_US', 'goog.i18n.DateTimeSymbols_ckb', 'goog.i18n.DateTimeSymbols_ckb_IQ', 'goog.i18n.DateTimeSymbols_ckb_IR', 'goog.i18n.DateTimeSymbols_cs_CZ', 'goog.i18n.DateTimeSymbols_cy_GB', 'goog.i18n.DateTimeSymbols_da_DK', 'goog.i18n.DateTimeSymbols_da_GL', 'goog.i18n.DateTimeSymbols_dav', 'goog.i18n.DateTimeSymbols_dav_KE', 'goog.i18n.DateTimeSymbols_de_BE', 'goog.i18n.DateTimeSymbols_de_DE', 'goog.i18n.DateTimeSymbols_de_IT', 'goog.i18n.DateTimeSymbols_de_LI', 'goog.i18n.DateTimeSymbols_de_LU', 'goog.i18n.DateTimeSymbols_dje', 'goog.i18n.DateTimeSymbols_dje_NE', 'goog.i18n.DateTimeSymbols_dsb', 'goog.i18n.DateTimeSymbols_dsb_DE', 'goog.i18n.DateTimeSymbols_dua', 'goog.i18n.DateTimeSymbols_dua_CM', 'goog.i18n.DateTimeSymbols_dyo', 'goog.i18n.DateTimeSymbols_dyo_SN', 'goog.i18n.DateTimeSymbols_dz', 'goog.i18n.DateTimeSymbols_dz_BT', 'goog.i18n.DateTimeSymbols_ebu', 'goog.i18n.DateTimeSymbols_ebu_KE', 'goog.i18n.DateTimeSymbols_ee', 'goog.i18n.DateTimeSymbols_ee_GH', 'goog.i18n.DateTimeSymbols_ee_TG', 'goog.i18n.DateTimeSymbols_el_CY', 'goog.i18n.DateTimeSymbols_el_GR', 'goog.i18n.DateTimeSymbols_en_001', 'goog.i18n.DateTimeSymbols_en_150', 'goog.i18n.DateTimeSymbols_en_AE', 'goog.i18n.DateTimeSymbols_en_AG', 'goog.i18n.DateTimeSymbols_en_AI', 'goog.i18n.DateTimeSymbols_en_AS', 'goog.i18n.DateTimeSymbols_en_AT', 'goog.i18n.DateTimeSymbols_en_BB', 'goog.i18n.DateTimeSymbols_en_BE', 'goog.i18n.DateTimeSymbols_en_BI', 'goog.i18n.DateTimeSymbols_en_BM', 'goog.i18n.DateTimeSymbols_en_BS', 'goog.i18n.DateTimeSymbols_en_BW', 'goog.i18n.DateTimeSymbols_en_BZ', 'goog.i18n.DateTimeSymbols_en_CC', 'goog.i18n.DateTimeSymbols_en_CH', 'goog.i18n.DateTimeSymbols_en_CK', 'goog.i18n.DateTimeSymbols_en_CM', 'goog.i18n.DateTimeSymbols_en_CX', 'goog.i18n.DateTimeSymbols_en_CY', 'goog.i18n.DateTimeSymbols_en_DE', 'goog.i18n.DateTimeSymbols_en_DG', 'goog.i18n.DateTimeSymbols_en_DK', 'goog.i18n.DateTimeSymbols_en_DM', 'goog.i18n.DateTimeSymbols_en_ER', 'goog.i18n.DateTimeSymbols_en_FI', 'goog.i18n.DateTimeSymbols_en_FJ', 'goog.i18n.DateTimeSymbols_en_FK', 'goog.i18n.DateTimeSymbols_en_FM', 'goog.i18n.DateTimeSymbols_en_GD', 'goog.i18n.DateTimeSymbols_en_GG', 'goog.i18n.DateTimeSymbols_en_GH', 'goog.i18n.DateTimeSymbols_en_GI', 'goog.i18n.DateTimeSymbols_en_GM', 'goog.i18n.DateTimeSymbols_en_GU', 'goog.i18n.DateTimeSymbols_en_GY', 'goog.i18n.DateTimeSymbols_en_HK', 'goog.i18n.DateTimeSymbols_en_IL', 'goog.i18n.DateTimeSymbols_en_IM', 'goog.i18n.DateTimeSymbols_en_IO', 'goog.i18n.DateTimeSymbols_en_JE', 'goog.i18n.DateTimeSymbols_en_JM', 'goog.i18n.DateTimeSymbols_en_KE', 'goog.i18n.DateTimeSymbols_en_KI', 'goog.i18n.DateTimeSymbols_en_KN', 'goog.i18n.DateTimeSymbols_en_KY', 'goog.i18n.DateTimeSymbols_en_LC', 'goog.i18n.DateTimeSymbols_en_LR', 'goog.i18n.DateTimeSymbols_en_LS', 'goog.i18n.DateTimeSymbols_en_MG', 'goog.i18n.DateTimeSymbols_en_MH', 'goog.i18n.DateTimeSymbols_en_MO', 'goog.i18n.DateTimeSymbols_en_MP', 'goog.i18n.DateTimeSymbols_en_MS', 'goog.i18n.DateTimeSymbols_en_MT', 'goog.i18n.DateTimeSymbols_en_MU', 'goog.i18n.DateTimeSymbols_en_MW', 'goog.i18n.DateTimeSymbols_en_MY', 'goog.i18n.DateTimeSymbols_en_NA', 'goog.i18n.DateTimeSymbols_en_NF', 'goog.i18n.DateTimeSymbols_en_NG', 'goog.i18n.DateTimeSymbols_en_NL', 'goog.i18n.DateTimeSymbols_en_NR', 'goog.i18n.DateTimeSymbols_en_NU', 'goog.i18n.DateTimeSymbols_en_NZ', 'goog.i18n.DateTimeSymbols_en_PG', 'goog.i18n.DateTimeSymbols_en_PH', 'goog.i18n.DateTimeSymbols_en_PK', 'goog.i18n.DateTimeSymbols_en_PN', 'goog.i18n.DateTimeSymbols_en_PR', 'goog.i18n.DateTimeSymbols_en_PW', 'goog.i18n.DateTimeSymbols_en_RW', 'goog.i18n.DateTimeSymbols_en_SB', 'goog.i18n.DateTimeSymbols_en_SC', 'goog.i18n.DateTimeSymbols_en_SD', 'goog.i18n.DateTimeSymbols_en_SE', 'goog.i18n.DateTimeSymbols_en_SH', 'goog.i18n.DateTimeSymbols_en_SI', 'goog.i18n.DateTimeSymbols_en_SL', 'goog.i18n.DateTimeSymbols_en_SS', 'goog.i18n.DateTimeSymbols_en_SX', 'goog.i18n.DateTimeSymbols_en_SZ', 'goog.i18n.DateTimeSymbols_en_TC', 'goog.i18n.DateTimeSymbols_en_TK', 'goog.i18n.DateTimeSymbols_en_TO', 'goog.i18n.DateTimeSymbols_en_TT', 'goog.i18n.DateTimeSymbols_en_TV', 'goog.i18n.DateTimeSymbols_en_TZ', 'goog.i18n.DateTimeSymbols_en_UG', 'goog.i18n.DateTimeSymbols_en_UM', 'goog.i18n.DateTimeSymbols_en_US_POSIX', 'goog.i18n.DateTimeSymbols_en_VC', 'goog.i18n.DateTimeSymbols_en_VG', 'goog.i18n.DateTimeSymbols_en_VI', 'goog.i18n.DateTimeSymbols_en_VU', 'goog.i18n.DateTimeSymbols_en_WS', 'goog.i18n.DateTimeSymbols_en_XA', 'goog.i18n.DateTimeSymbols_en_ZM', 'goog.i18n.DateTimeSymbols_en_ZW', 'goog.i18n.DateTimeSymbols_eo', 'goog.i18n.DateTimeSymbols_eo_001', 'goog.i18n.DateTimeSymbols_es_AR', 'goog.i18n.DateTimeSymbols_es_BO', 'goog.i18n.DateTimeSymbols_es_BR', 'goog.i18n.DateTimeSymbols_es_BZ', 'goog.i18n.DateTimeSymbols_es_CL', 'goog.i18n.DateTimeSymbols_es_CO', 'goog.i18n.DateTimeSymbols_es_CR', 'goog.i18n.DateTimeSymbols_es_CU', 'goog.i18n.DateTimeSymbols_es_DO', 'goog.i18n.DateTimeSymbols_es_EA', 'goog.i18n.DateTimeSymbols_es_EC', 'goog.i18n.DateTimeSymbols_es_GQ', 'goog.i18n.DateTimeSymbols_es_GT', 'goog.i18n.DateTimeSymbols_es_HN', 'goog.i18n.DateTimeSymbols_es_IC', 'goog.i18n.DateTimeSymbols_es_NI', 'goog.i18n.DateTimeSymbols_es_PA', 'goog.i18n.DateTimeSymbols_es_PE', 'goog.i18n.DateTimeSymbols_es_PH', 'goog.i18n.DateTimeSymbols_es_PR', 'goog.i18n.DateTimeSymbols_es_PY', 'goog.i18n.DateTimeSymbols_es_SV', 'goog.i18n.DateTimeSymbols_es_UY', 'goog.i18n.DateTimeSymbols_es_VE', 'goog.i18n.DateTimeSymbols_et_EE', 'goog.i18n.DateTimeSymbols_eu_ES', 'goog.i18n.DateTimeSymbols_ewo', 'goog.i18n.DateTimeSymbols_ewo_CM', 'goog.i18n.DateTimeSymbols_fa_AF', 'goog.i18n.DateTimeSymbols_fa_IR', 'goog.i18n.DateTimeSymbols_ff', 'goog.i18n.DateTimeSymbols_ff_Latn', 'goog.i18n.DateTimeSymbols_ff_Latn_BF', 'goog.i18n.DateTimeSymbols_ff_Latn_CM', 'goog.i18n.DateTimeSymbols_ff_Latn_GH', 'goog.i18n.DateTimeSymbols_ff_Latn_GM', 'goog.i18n.DateTimeSymbols_ff_Latn_GN', 'goog.i18n.DateTimeSymbols_ff_Latn_GW', 'goog.i18n.DateTimeSymbols_ff_Latn_LR', 'goog.i18n.DateTimeSymbols_ff_Latn_MR', 'goog.i18n.DateTimeSymbols_ff_Latn_NE', 'goog.i18n.DateTimeSymbols_ff_Latn_NG', 'goog.i18n.DateTimeSymbols_ff_Latn_SL', 'goog.i18n.DateTimeSymbols_ff_Latn_SN', 'goog.i18n.DateTimeSymbols_fi_FI', 'goog.i18n.DateTimeSymbols_fil_PH', 'goog.i18n.DateTimeSymbols_fo', 'goog.i18n.DateTimeSymbols_fo_DK', 'goog.i18n.DateTimeSymbols_fo_FO', 'goog.i18n.DateTimeSymbols_fr_BE', 'goog.i18n.DateTimeSymbols_fr_BF', 'goog.i18n.DateTimeSymbols_fr_BI', 'goog.i18n.DateTimeSymbols_fr_BJ', 'goog.i18n.DateTimeSymbols_fr_BL', 'goog.i18n.DateTimeSymbols_fr_CD', 'goog.i18n.DateTimeSymbols_fr_CF', 'goog.i18n.DateTimeSymbols_fr_CG', 'goog.i18n.DateTimeSymbols_fr_CH', 'goog.i18n.DateTimeSymbols_fr_CI', 'goog.i18n.DateTimeSymbols_fr_CM', 'goog.i18n.DateTimeSymbols_fr_DJ', 'goog.i18n.DateTimeSymbols_fr_DZ', 'goog.i18n.DateTimeSymbols_fr_FR', 'goog.i18n.DateTimeSymbols_fr_GA', 'goog.i18n.DateTimeSymbols_fr_GF', 'goog.i18n.DateTimeSymbols_fr_GN', 'goog.i18n.DateTimeSymbols_fr_GP', 'goog.i18n.DateTimeSymbols_fr_GQ', 'goog.i18n.DateTimeSymbols_fr_HT', 'goog.i18n.DateTimeSymbols_fr_KM', 'goog.i18n.DateTimeSymbols_fr_LU', 'goog.i18n.DateTimeSymbols_fr_MA', 'goog.i18n.DateTimeSymbols_fr_MC', 'goog.i18n.DateTimeSymbols_fr_MF', 'goog.i18n.DateTimeSymbols_fr_MG', 'goog.i18n.DateTimeSymbols_fr_ML', 'goog.i18n.DateTimeSymbols_fr_MQ', 'goog.i18n.DateTimeSymbols_fr_MR', 'goog.i18n.DateTimeSymbols_fr_MU', 'goog.i18n.DateTimeSymbols_fr_NC', 'goog.i18n.DateTimeSymbols_fr_NE', 'goog.i18n.DateTimeSymbols_fr_PF', 'goog.i18n.DateTimeSymbols_fr_PM', 'goog.i18n.DateTimeSymbols_fr_RE', 'goog.i18n.DateTimeSymbols_fr_RW', 'goog.i18n.DateTimeSymbols_fr_SC', 'goog.i18n.DateTimeSymbols_fr_SN', 'goog.i18n.DateTimeSymbols_fr_SY', 'goog.i18n.DateTimeSymbols_fr_TD', 'goog.i18n.DateTimeSymbols_fr_TG', 'goog.i18n.DateTimeSymbols_fr_TN', 'goog.i18n.DateTimeSymbols_fr_VU', 'goog.i18n.DateTimeSymbols_fr_WF', 'goog.i18n.DateTimeSymbols_fr_YT', 'goog.i18n.DateTimeSymbols_fur', 'goog.i18n.DateTimeSymbols_fur_IT', 'goog.i18n.DateTimeSymbols_fy', 'goog.i18n.DateTimeSymbols_fy_NL', 'goog.i18n.DateTimeSymbols_ga_IE', 'goog.i18n.DateTimeSymbols_gd', 'goog.i18n.DateTimeSymbols_gd_GB', 'goog.i18n.DateTimeSymbols_gl_ES', 'goog.i18n.DateTimeSymbols_gsw_CH', 'goog.i18n.DateTimeSymbols_gsw_FR', 'goog.i18n.DateTimeSymbols_gsw_LI', 'goog.i18n.DateTimeSymbols_gu_IN', 'goog.i18n.DateTimeSymbols_guz', 'goog.i18n.DateTimeSymbols_guz_KE', 'goog.i18n.DateTimeSymbols_gv', 'goog.i18n.DateTimeSymbols_gv_IM', 'goog.i18n.DateTimeSymbols_ha', 'goog.i18n.DateTimeSymbols_ha_GH', 'goog.i18n.DateTimeSymbols_ha_NE', 'goog.i18n.DateTimeSymbols_ha_NG', 'goog.i18n.DateTimeSymbols_haw_US', 'goog.i18n.DateTimeSymbols_he_IL', 'goog.i18n.DateTimeSymbols_hi_IN', 'goog.i18n.DateTimeSymbols_hr_BA', 'goog.i18n.DateTimeSymbols_hr_HR', 'goog.i18n.DateTimeSymbols_hsb', 'goog.i18n.DateTimeSymbols_hsb_DE', 'goog.i18n.DateTimeSymbols_hu_HU', 'goog.i18n.DateTimeSymbols_hy_AM', 'goog.i18n.DateTimeSymbols_ia', 'goog.i18n.DateTimeSymbols_ia_001', 'goog.i18n.DateTimeSymbols_id_ID', 'goog.i18n.DateTimeSymbols_ig', 'goog.i18n.DateTimeSymbols_ig_NG', 'goog.i18n.DateTimeSymbols_ii', 'goog.i18n.DateTimeSymbols_ii_CN', 'goog.i18n.DateTimeSymbols_is_IS', 'goog.i18n.DateTimeSymbols_it_CH', 'goog.i18n.DateTimeSymbols_it_IT', 'goog.i18n.DateTimeSymbols_it_SM', 'goog.i18n.DateTimeSymbols_it_VA', 'goog.i18n.DateTimeSymbols_ja_JP', 'goog.i18n.DateTimeSymbols_jgo', 'goog.i18n.DateTimeSymbols_jgo_CM', 'goog.i18n.DateTimeSymbols_jmc', 'goog.i18n.DateTimeSymbols_jmc_TZ', 'goog.i18n.DateTimeSymbols_jv', 'goog.i18n.DateTimeSymbols_jv_ID', 'goog.i18n.DateTimeSymbols_ka_GE', 'goog.i18n.DateTimeSymbols_kab', 'goog.i18n.DateTimeSymbols_kab_DZ', 'goog.i18n.DateTimeSymbols_kam', 'goog.i18n.DateTimeSymbols_kam_KE', 'goog.i18n.DateTimeSymbols_kde', 'goog.i18n.DateTimeSymbols_kde_TZ', 'goog.i18n.DateTimeSymbols_kea', 'goog.i18n.DateTimeSymbols_kea_CV', 'goog.i18n.DateTimeSymbols_khq', 'goog.i18n.DateTimeSymbols_khq_ML', 'goog.i18n.DateTimeSymbols_ki', 'goog.i18n.DateTimeSymbols_ki_KE', 'goog.i18n.DateTimeSymbols_kk_KZ', 'goog.i18n.DateTimeSymbols_kkj', 'goog.i18n.DateTimeSymbols_kkj_CM', 'goog.i18n.DateTimeSymbols_kl', 'goog.i18n.DateTimeSymbols_kl_GL', 'goog.i18n.DateTimeSymbols_kln', 'goog.i18n.DateTimeSymbols_kln_KE', 'goog.i18n.DateTimeSymbols_km_KH', 'goog.i18n.DateTimeSymbols_kn_IN', 'goog.i18n.DateTimeSymbols_ko_KP', 'goog.i18n.DateTimeSymbols_ko_KR', 'goog.i18n.DateTimeSymbols_kok', 'goog.i18n.DateTimeSymbols_kok_IN', 'goog.i18n.DateTimeSymbols_ks', 'goog.i18n.DateTimeSymbols_ks_IN', 'goog.i18n.DateTimeSymbols_ksb', 'goog.i18n.DateTimeSymbols_ksb_TZ', 'goog.i18n.DateTimeSymbols_ksf', 'goog.i18n.DateTimeSymbols_ksf_CM', 'goog.i18n.DateTimeSymbols_ksh', 'goog.i18n.DateTimeSymbols_ksh_DE', 'goog.i18n.DateTimeSymbols_ku', 'goog.i18n.DateTimeSymbols_ku_TR', 'goog.i18n.DateTimeSymbols_kw', 'goog.i18n.DateTimeSymbols_kw_GB', 'goog.i18n.DateTimeSymbols_ky_KG', 'goog.i18n.DateTimeSymbols_lag', 'goog.i18n.DateTimeSymbols_lag_TZ', 'goog.i18n.DateTimeSymbols_lb', 'goog.i18n.DateTimeSymbols_lb_LU', 'goog.i18n.DateTimeSymbols_lg', 'goog.i18n.DateTimeSymbols_lg_UG', 'goog.i18n.DateTimeSymbols_lkt', 'goog.i18n.DateTimeSymbols_lkt_US', 'goog.i18n.DateTimeSymbols_ln_AO', 'goog.i18n.DateTimeSymbols_ln_CD', 'goog.i18n.DateTimeSymbols_ln_CF', 'goog.i18n.DateTimeSymbols_ln_CG', 'goog.i18n.DateTimeSymbols_lo_LA', 'goog.i18n.DateTimeSymbols_lrc', 'goog.i18n.DateTimeSymbols_lrc_IQ', 'goog.i18n.DateTimeSymbols_lrc_IR', 'goog.i18n.DateTimeSymbols_lt_LT', 'goog.i18n.DateTimeSymbols_lu', 'goog.i18n.DateTimeSymbols_lu_CD', 'goog.i18n.DateTimeSymbols_luo', 'goog.i18n.DateTimeSymbols_luo_KE', 'goog.i18n.DateTimeSymbols_luy', 'goog.i18n.DateTimeSymbols_luy_KE', 'goog.i18n.DateTimeSymbols_lv_LV', 'goog.i18n.DateTimeSymbols_mas', 'goog.i18n.DateTimeSymbols_mas_KE', 'goog.i18n.DateTimeSymbols_mas_TZ', 'goog.i18n.DateTimeSymbols_mer', 'goog.i18n.DateTimeSymbols_mer_KE', 'goog.i18n.DateTimeSymbols_mfe', 'goog.i18n.DateTimeSymbols_mfe_MU', 'goog.i18n.DateTimeSymbols_mg', 'goog.i18n.DateTimeSymbols_mg_MG', 'goog.i18n.DateTimeSymbols_mgh', 'goog.i18n.DateTimeSymbols_mgh_MZ', 'goog.i18n.DateTimeSymbols_mgo', 'goog.i18n.DateTimeSymbols_mgo_CM', 'goog.i18n.DateTimeSymbols_mi', 'goog.i18n.DateTimeSymbols_mi_NZ', 'goog.i18n.DateTimeSymbols_mk_MK', 'goog.i18n.DateTimeSymbols_ml_IN', 'goog.i18n.DateTimeSymbols_mn_MN', 'goog.i18n.DateTimeSymbols_mr_IN', 'goog.i18n.DateTimeSymbols_ms_BN', 'goog.i18n.DateTimeSymbols_ms_MY', 'goog.i18n.DateTimeSymbols_ms_SG', 'goog.i18n.DateTimeSymbols_mt_MT', 'goog.i18n.DateTimeSymbols_mua', 'goog.i18n.DateTimeSymbols_mua_CM', 'goog.i18n.DateTimeSymbols_my_MM', 'goog.i18n.DateTimeSymbols_mzn', 'goog.i18n.DateTimeSymbols_mzn_IR', 'goog.i18n.DateTimeSymbols_naq', 'goog.i18n.DateTimeSymbols_naq_NA', 'goog.i18n.DateTimeSymbols_nb_NO', 'goog.i18n.DateTimeSymbols_nb_SJ', 'goog.i18n.DateTimeSymbols_nd', 'goog.i18n.DateTimeSymbols_nd_ZW', 'goog.i18n.DateTimeSymbols_nds', 'goog.i18n.DateTimeSymbols_nds_DE', 'goog.i18n.DateTimeSymbols_nds_NL', 'goog.i18n.DateTimeSymbols_ne_IN', 'goog.i18n.DateTimeSymbols_ne_NP', 'goog.i18n.DateTimeSymbols_nl_AW', 'goog.i18n.DateTimeSymbols_nl_BE', 'goog.i18n.DateTimeSymbols_nl_BQ', 'goog.i18n.DateTimeSymbols_nl_CW', 'goog.i18n.DateTimeSymbols_nl_NL', 'goog.i18n.DateTimeSymbols_nl_SR', 'goog.i18n.DateTimeSymbols_nl_SX', 'goog.i18n.DateTimeSymbols_nmg', 'goog.i18n.DateTimeSymbols_nmg_CM', 'goog.i18n.DateTimeSymbols_nn', 'goog.i18n.DateTimeSymbols_nn_NO', 'goog.i18n.DateTimeSymbols_nnh', 'goog.i18n.DateTimeSymbols_nnh_CM', 'goog.i18n.DateTimeSymbols_nus', 'goog.i18n.DateTimeSymbols_nus_SS', 'goog.i18n.DateTimeSymbols_nyn', 'goog.i18n.DateTimeSymbols_nyn_UG', 'goog.i18n.DateTimeSymbols_om', 'goog.i18n.DateTimeSymbols_om_ET', 'goog.i18n.DateTimeSymbols_om_KE', 'goog.i18n.DateTimeSymbols_or_IN', 'goog.i18n.DateTimeSymbols_os', 'goog.i18n.DateTimeSymbols_os_GE', 'goog.i18n.DateTimeSymbols_os_RU', 'goog.i18n.DateTimeSymbols_pa_Arab', 'goog.i18n.DateTimeSymbols_pa_Arab_PK', 'goog.i18n.DateTimeSymbols_pa_Guru', 'goog.i18n.DateTimeSymbols_pa_Guru_IN', 'goog.i18n.DateTimeSymbols_pl_PL', 'goog.i18n.DateTimeSymbols_ps', 'goog.i18n.DateTimeSymbols_ps_AF', 'goog.i18n.DateTimeSymbols_ps_PK', 'goog.i18n.DateTimeSymbols_pt_AO', 'goog.i18n.DateTimeSymbols_pt_CH', 'goog.i18n.DateTimeSymbols_pt_CV', 'goog.i18n.DateTimeSymbols_pt_GQ', 'goog.i18n.DateTimeSymbols_pt_GW', 'goog.i18n.DateTimeSymbols_pt_LU', 'goog.i18n.DateTimeSymbols_pt_MO', 'goog.i18n.DateTimeSymbols_pt_MZ', 'goog.i18n.DateTimeSymbols_pt_ST', 'goog.i18n.DateTimeSymbols_pt_TL', 'goog.i18n.DateTimeSymbols_qu', 'goog.i18n.DateTimeSymbols_qu_BO', 'goog.i18n.DateTimeSymbols_qu_EC', 'goog.i18n.DateTimeSymbols_qu_PE', 'goog.i18n.DateTimeSymbols_rm', 'goog.i18n.DateTimeSymbols_rm_CH', 'goog.i18n.DateTimeSymbols_rn', 'goog.i18n.DateTimeSymbols_rn_BI', 'goog.i18n.DateTimeSymbols_ro_MD', 'goog.i18n.DateTimeSymbols_ro_RO', 'goog.i18n.DateTimeSymbols_rof', 'goog.i18n.DateTimeSymbols_rof_TZ', 'goog.i18n.DateTimeSymbols_ru_BY', 'goog.i18n.DateTimeSymbols_ru_KG', 'goog.i18n.DateTimeSymbols_ru_KZ', 'goog.i18n.DateTimeSymbols_ru_MD', 'goog.i18n.DateTimeSymbols_ru_RU', 'goog.i18n.DateTimeSymbols_ru_UA', 'goog.i18n.DateTimeSymbols_rw', 'goog.i18n.DateTimeSymbols_rw_RW', 'goog.i18n.DateTimeSymbols_rwk', 'goog.i18n.DateTimeSymbols_rwk_TZ', 'goog.i18n.DateTimeSymbols_sah', 'goog.i18n.DateTimeSymbols_sah_RU', 'goog.i18n.DateTimeSymbols_saq', 'goog.i18n.DateTimeSymbols_saq_KE', 'goog.i18n.DateTimeSymbols_sbp', 'goog.i18n.DateTimeSymbols_sbp_TZ', 'goog.i18n.DateTimeSymbols_sd', 'goog.i18n.DateTimeSymbols_sd_PK', 'goog.i18n.DateTimeSymbols_se', 'goog.i18n.DateTimeSymbols_se_FI', 'goog.i18n.DateTimeSymbols_se_NO', 'goog.i18n.DateTimeSymbols_se_SE', 'goog.i18n.DateTimeSymbols_seh', 'goog.i18n.DateTimeSymbols_seh_MZ', 'goog.i18n.DateTimeSymbols_ses', 'goog.i18n.DateTimeSymbols_ses_ML', 'goog.i18n.DateTimeSymbols_sg', 'goog.i18n.DateTimeSymbols_sg_CF', 'goog.i18n.DateTimeSymbols_shi', 'goog.i18n.DateTimeSymbols_shi_Latn', 'goog.i18n.DateTimeSymbols_shi_Latn_MA', 'goog.i18n.DateTimeSymbols_shi_Tfng', 'goog.i18n.DateTimeSymbols_shi_Tfng_MA', 'goog.i18n.DateTimeSymbols_si_LK', 'goog.i18n.DateTimeSymbols_sk_SK', 'goog.i18n.DateTimeSymbols_sl_SI', 'goog.i18n.DateTimeSymbols_smn', 'goog.i18n.DateTimeSymbols_smn_FI', 'goog.i18n.DateTimeSymbols_sn', 'goog.i18n.DateTimeSymbols_sn_ZW', 'goog.i18n.DateTimeSymbols_so', 'goog.i18n.DateTimeSymbols_so_DJ', 'goog.i18n.DateTimeSymbols_so_ET', 'goog.i18n.DateTimeSymbols_so_KE', 'goog.i18n.DateTimeSymbols_so_SO', 'goog.i18n.DateTimeSymbols_sq_AL', 'goog.i18n.DateTimeSymbols_sq_MK', 'goog.i18n.DateTimeSymbols_sq_XK', 'goog.i18n.DateTimeSymbols_sr_Cyrl', 'goog.i18n.DateTimeSymbols_sr_Cyrl_BA', 'goog.i18n.DateTimeSymbols_sr_Cyrl_ME', 'goog.i18n.DateTimeSymbols_sr_Cyrl_RS', 'goog.i18n.DateTimeSymbols_sr_Cyrl_XK', 'goog.i18n.DateTimeSymbols_sr_Latn_BA', 'goog.i18n.DateTimeSymbols_sr_Latn_ME', 'goog.i18n.DateTimeSymbols_sr_Latn_RS', 'goog.i18n.DateTimeSymbols_sr_Latn_XK', 'goog.i18n.DateTimeSymbols_sv_AX', 'goog.i18n.DateTimeSymbols_sv_FI', 'goog.i18n.DateTimeSymbols_sv_SE', 'goog.i18n.DateTimeSymbols_sw_CD', 'goog.i18n.DateTimeSymbols_sw_KE', 'goog.i18n.DateTimeSymbols_sw_TZ', 'goog.i18n.DateTimeSymbols_sw_UG', 'goog.i18n.DateTimeSymbols_ta_IN', 'goog.i18n.DateTimeSymbols_ta_LK', 'goog.i18n.DateTimeSymbols_ta_MY', 'goog.i18n.DateTimeSymbols_ta_SG', 'goog.i18n.DateTimeSymbols_te_IN', 'goog.i18n.DateTimeSymbols_teo', 'goog.i18n.DateTimeSymbols_teo_KE', 'goog.i18n.DateTimeSymbols_teo_UG', 'goog.i18n.DateTimeSymbols_tg', 'goog.i18n.DateTimeSymbols_tg_TJ', 'goog.i18n.DateTimeSymbols_th_TH', 'goog.i18n.DateTimeSymbols_ti', 'goog.i18n.DateTimeSymbols_ti_ER', 'goog.i18n.DateTimeSymbols_ti_ET', 'goog.i18n.DateTimeSymbols_tk', 'goog.i18n.DateTimeSymbols_tk_TM', 'goog.i18n.DateTimeSymbols_to', 'goog.i18n.DateTimeSymbols_to_TO', 'goog.i18n.DateTimeSymbols_tr_CY', 'goog.i18n.DateTimeSymbols_tr_TR', 'goog.i18n.DateTimeSymbols_tt', 'goog.i18n.DateTimeSymbols_tt_RU', 'goog.i18n.DateTimeSymbols_twq', 'goog.i18n.DateTimeSymbols_twq_NE', 'goog.i18n.DateTimeSymbols_tzm', 'goog.i18n.DateTimeSymbols_tzm_MA', 'goog.i18n.DateTimeSymbols_ug', 'goog.i18n.DateTimeSymbols_ug_CN', 'goog.i18n.DateTimeSymbols_uk_UA', 'goog.i18n.DateTimeSymbols_ur_IN', 'goog.i18n.DateTimeSymbols_ur_PK', 'goog.i18n.DateTimeSymbols_uz_Arab', 'goog.i18n.DateTimeSymbols_uz_Arab_AF', 'goog.i18n.DateTimeSymbols_uz_Cyrl', 'goog.i18n.DateTimeSymbols_uz_Cyrl_UZ', 'goog.i18n.DateTimeSymbols_uz_Latn', 'goog.i18n.DateTimeSymbols_uz_Latn_UZ', 'goog.i18n.DateTimeSymbols_vai', 'goog.i18n.DateTimeSymbols_vai_Latn', 'goog.i18n.DateTimeSymbols_vai_Latn_LR', 'goog.i18n.DateTimeSymbols_vai_Vaii', 'goog.i18n.DateTimeSymbols_vai_Vaii_LR', 'goog.i18n.DateTimeSymbols_vi_VN', 'goog.i18n.DateTimeSymbols_vun', 'goog.i18n.DateTimeSymbols_vun_TZ', 'goog.i18n.DateTimeSymbols_wae', 'goog.i18n.DateTimeSymbols_wae_CH', 'goog.i18n.DateTimeSymbols_wo', 'goog.i18n.DateTimeSymbols_wo_SN', 'goog.i18n.DateTimeSymbols_xh', 'goog.i18n.DateTimeSymbols_xh_ZA', 'goog.i18n.DateTimeSymbols_xog', 'goog.i18n.DateTimeSymbols_xog_UG', 'goog.i18n.DateTimeSymbols_yav', 'goog.i18n.DateTimeSymbols_yav_CM', 'goog.i18n.DateTimeSymbols_yi', 'goog.i18n.DateTimeSymbols_yi_001', 'goog.i18n.DateTimeSymbols_yo', 'goog.i18n.DateTimeSymbols_yo_BJ', 'goog.i18n.DateTimeSymbols_yo_NG', 'goog.i18n.DateTimeSymbols_yue', 'goog.i18n.DateTimeSymbols_yue_Hans', 'goog.i18n.DateTimeSymbols_yue_Hans_CN', 'goog.i18n.DateTimeSymbols_yue_Hant', 'goog.i18n.DateTimeSymbols_yue_Hant_HK', 'goog.i18n.DateTimeSymbols_zgh', 'goog.i18n.DateTimeSymbols_zgh_MA', 'goog.i18n.DateTimeSymbols_zh_Hans', 'goog.i18n.DateTimeSymbols_zh_Hans_CN', 'goog.i18n.DateTimeSymbols_zh_Hans_HK', 'goog.i18n.DateTimeSymbols_zh_Hans_MO', 'goog.i18n.DateTimeSymbols_zh_Hans_SG', 'goog.i18n.DateTimeSymbols_zh_Hant', 'goog.i18n.DateTimeSymbols_zh_Hant_HK', 'goog.i18n.DateTimeSymbols_zh_Hant_MO', 'goog.i18n.DateTimeSymbols_zh_Hant_TW', 'goog.i18n.DateTimeSymbols_zu_ZA'], ['goog.i18n.DateTimeSymbols'], {});
goog.addDependency('i18n/graphemebreak.js', ['goog.i18n.GraphemeBreak'], ['goog.asserts', 'goog.i18n.uChar', 'goog.structs.InversionMap'], {});
goog.addDependency('i18n/graphemebreak_test.js', ['goog.i18n.GraphemeBreakTest'], ['goog.i18n.GraphemeBreak', 'goog.i18n.uChar', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('i18n/localefeature.js', ['goog.i18n.LocaleFeature'], [], {'module': 'goog'});
goog.addDependency('i18n/localefeature_test.js', ['goog.i18n.LocaleFeatureTest'], ['goog.i18n.LocaleFeature', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('i18n/messageformat.js', ['goog.i18n.MessageFormat'], ['goog.array', 'goog.asserts', 'goog.i18n.CompactNumberFormatSymbols', 'goog.i18n.NumberFormat', 'goog.i18n.NumberFormatSymbols', 'goog.i18n.ordinalRules', 'goog.i18n.pluralRules'], {});
goog.addDependency('i18n/messageformat_test.js', ['goog.i18n.MessageFormatTest'], ['goog.i18n.MessageFormat', 'goog.i18n.NumberFormatSymbols_hr', 'goog.i18n.pluralRules', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('i18n/mime.js', ['goog.i18n.mime', 'goog.i18n.mime.encode'], ['goog.array', 'goog.i18n.uChar'], {});
goog.addDependency('i18n/mime_test.js', ['goog.i18n.mime.encodeTest'], ['goog.i18n.mime.encode', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('i18n/numberformat.js', ['goog.i18n.NumberFormat', 'goog.i18n.NumberFormat.CurrencyStyle', 'goog.i18n.NumberFormat.Format'], ['goog.asserts', 'goog.i18n.CompactNumberFormatSymbols', 'goog.i18n.NumberFormatSymbols', 'goog.i18n.NumberFormatSymbols_u_nu_latn', 'goog.i18n.currency', 'goog.math', 'goog.string'], {});
goog.addDependency('i18n/numberformat_test.js', ['goog.i18n.NumberFormatTest'], ['goog.i18n.CompactNumberFormatSymbols', 'goog.i18n.CompactNumberFormatSymbols_de', 'goog.i18n.CompactNumberFormatSymbols_en', 'goog.i18n.CompactNumberFormatSymbols_fr', 'goog.i18n.NumberFormat', 'goog.i18n.NumberFormatSymbols', 'goog.i18n.NumberFormatSymbols_ar_EG', 'goog.i18n.NumberFormatSymbols_ar_EG_u_nu_latn', 'goog.i18n.NumberFormatSymbols_de', 'goog.i18n.NumberFormatSymbols_en', 'goog.i18n.NumberFormatSymbols_en_AU', 'goog.i18n.NumberFormatSymbols_en_US', 'goog.i18n.NumberFormatSymbols_fi', 'goog.i18n.NumberFormatSymbols_fr', 'goog.i18n.NumberFormatSymbols_pl', 'goog.i18n.NumberFormatSymbols_ro', 'goog.i18n.NumberFormatSymbols_u_nu_latn', 'goog.string', 'goog.testing.ExpectedFailures', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite', 'goog.userAgent', 'goog.userAgent.product', 'goog.userAgent.product.isVersion'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('i18n/numberformatsymbols.js', ['goog.i18n.NumberFormatSymbols', 'goog.i18n.NumberFormatSymbols_af', 'goog.i18n.NumberFormatSymbols_am', 'goog.i18n.NumberFormatSymbols_ar', 'goog.i18n.NumberFormatSymbols_ar_DZ', 'goog.i18n.NumberFormatSymbols_ar_EG', 'goog.i18n.NumberFormatSymbols_ar_EG_u_nu_latn', 'goog.i18n.NumberFormatSymbols_az', 'goog.i18n.NumberFormatSymbols_be', 'goog.i18n.NumberFormatSymbols_bg', 'goog.i18n.NumberFormatSymbols_bn', 'goog.i18n.NumberFormatSymbols_bn_u_nu_latn', 'goog.i18n.NumberFormatSymbols_br', 'goog.i18n.NumberFormatSymbols_bs', 'goog.i18n.NumberFormatSymbols_ca', 'goog.i18n.NumberFormatSymbols_chr', 'goog.i18n.NumberFormatSymbols_cs', 'goog.i18n.NumberFormatSymbols_cy', 'goog.i18n.NumberFormatSymbols_da', 'goog.i18n.NumberFormatSymbols_de', 'goog.i18n.NumberFormatSymbols_de_AT', 'goog.i18n.NumberFormatSymbols_de_CH', 'goog.i18n.NumberFormatSymbols_el', 'goog.i18n.NumberFormatSymbols_en', 'goog.i18n.NumberFormatSymbols_en_AU', 'goog.i18n.NumberFormatSymbols_en_CA', 'goog.i18n.NumberFormatSymbols_en_GB', 'goog.i18n.NumberFormatSymbols_en_IE', 'goog.i18n.NumberFormatSymbols_en_IN', 'goog.i18n.NumberFormatSymbols_en_SG', 'goog.i18n.NumberFormatSymbols_en_US', 'goog.i18n.NumberFormatSymbols_en_ZA', 'goog.i18n.NumberFormatSymbols_es', 'goog.i18n.NumberFormatSymbols_es_419', 'goog.i18n.NumberFormatSymbols_es_ES', 'goog.i18n.NumberFormatSymbols_es_MX', 'goog.i18n.NumberFormatSymbols_es_US', 'goog.i18n.NumberFormatSymbols_et', 'goog.i18n.NumberFormatSymbols_eu', 'goog.i18n.NumberFormatSymbols_fa', 'goog.i18n.NumberFormatSymbols_fa_u_nu_latn', 'goog.i18n.NumberFormatSymbols_fi', 'goog.i18n.NumberFormatSymbols_fil', 'goog.i18n.NumberFormatSymbols_fr', 'goog.i18n.NumberFormatSymbols_fr_CA', 'goog.i18n.NumberFormatSymbols_ga', 'goog.i18n.NumberFormatSymbols_gl', 'goog.i18n.NumberFormatSymbols_gsw', 'goog.i18n.NumberFormatSymbols_gu', 'goog.i18n.NumberFormatSymbols_haw', 'goog.i18n.NumberFormatSymbols_he', 'goog.i18n.NumberFormatSymbols_hi', 'goog.i18n.NumberFormatSymbols_hr', 'goog.i18n.NumberFormatSymbols_hu', 'goog.i18n.NumberFormatSymbols_hy', 'goog.i18n.NumberFormatSymbols_id', 'goog.i18n.NumberFormatSymbols_in', 'goog.i18n.NumberFormatSymbols_is', 'goog.i18n.NumberFormatSymbols_it', 'goog.i18n.NumberFormatSymbols_iw', 'goog.i18n.NumberFormatSymbols_ja', 'goog.i18n.NumberFormatSymbols_ka', 'goog.i18n.NumberFormatSymbols_kk', 'goog.i18n.NumberFormatSymbols_km', 'goog.i18n.NumberFormatSymbols_kn', 'goog.i18n.NumberFormatSymbols_ko', 'goog.i18n.NumberFormatSymbols_ky', 'goog.i18n.NumberFormatSymbols_ln', 'goog.i18n.NumberFormatSymbols_lo', 'goog.i18n.NumberFormatSymbols_lt', 'goog.i18n.NumberFormatSymbols_lv', 'goog.i18n.NumberFormatSymbols_mk', 'goog.i18n.NumberFormatSymbols_ml', 'goog.i18n.NumberFormatSymbols_mn', 'goog.i18n.NumberFormatSymbols_mo', 'goog.i18n.NumberFormatSymbols_mr', 'goog.i18n.NumberFormatSymbols_mr_u_nu_latn', 'goog.i18n.NumberFormatSymbols_ms', 'goog.i18n.NumberFormatSymbols_mt', 'goog.i18n.NumberFormatSymbols_my', 'goog.i18n.NumberFormatSymbols_my_u_nu_latn', 'goog.i18n.NumberFormatSymbols_nb', 'goog.i18n.NumberFormatSymbols_ne', 'goog.i18n.NumberFormatSymbols_ne_u_nu_latn', 'goog.i18n.NumberFormatSymbols_nl', 'goog.i18n.NumberFormatSymbols_no', 'goog.i18n.NumberFormatSymbols_no_NO', 'goog.i18n.NumberFormatSymbols_or', 'goog.i18n.NumberFormatSymbols_pa', 'goog.i18n.NumberFormatSymbols_pl', 'goog.i18n.NumberFormatSymbols_pt', 'goog.i18n.NumberFormatSymbols_pt_BR', 'goog.i18n.NumberFormatSymbols_pt_PT', 'goog.i18n.NumberFormatSymbols_ro', 'goog.i18n.NumberFormatSymbols_ru', 'goog.i18n.NumberFormatSymbols_sh', 'goog.i18n.NumberFormatSymbols_si', 'goog.i18n.NumberFormatSymbols_sk', 'goog.i18n.NumberFormatSymbols_sl', 'goog.i18n.NumberFormatSymbols_sq', 'goog.i18n.NumberFormatSymbols_sr', 'goog.i18n.NumberFormatSymbols_sr_Latn', 'goog.i18n.NumberFormatSymbols_sv', 'goog.i18n.NumberFormatSymbols_sw', 'goog.i18n.NumberFormatSymbols_ta', 'goog.i18n.NumberFormatSymbols_te', 'goog.i18n.NumberFormatSymbols_th', 'goog.i18n.NumberFormatSymbols_tl', 'goog.i18n.NumberFormatSymbols_tr', 'goog.i18n.NumberFormatSymbols_u_nu_latn', 'goog.i18n.NumberFormatSymbols_uk', 'goog.i18n.NumberFormatSymbols_ur', 'goog.i18n.NumberFormatSymbols_uz', 'goog.i18n.NumberFormatSymbols_vi', 'goog.i18n.NumberFormatSymbols_zh', 'goog.i18n.NumberFormatSymbols_zh_CN', 'goog.i18n.NumberFormatSymbols_zh_HK', 'goog.i18n.NumberFormatSymbols_zh_TW', 'goog.i18n.NumberFormatSymbols_zu'], [], {});
goog.addDependency('i18n/numberformatsymbolsext.js', ['goog.i18n.NumberFormatSymbolsExt', 'goog.i18n.NumberFormatSymbols_af_NA', 'goog.i18n.NumberFormatSymbols_af_ZA', 'goog.i18n.NumberFormatSymbols_agq', 'goog.i18n.NumberFormatSymbols_agq_CM', 'goog.i18n.NumberFormatSymbols_ak', 'goog.i18n.NumberFormatSymbols_ak_GH', 'goog.i18n.NumberFormatSymbols_am_ET', 'goog.i18n.NumberFormatSymbols_ar_001', 'goog.i18n.NumberFormatSymbols_ar_AE', 'goog.i18n.NumberFormatSymbols_ar_AE_u_nu_latn', 'goog.i18n.NumberFormatSymbols_ar_BH', 'goog.i18n.NumberFormatSymbols_ar_BH_u_nu_latn', 'goog.i18n.NumberFormatSymbols_ar_DJ', 'goog.i18n.NumberFormatSymbols_ar_DJ_u_nu_latn', 'goog.i18n.NumberFormatSymbols_ar_EH', 'goog.i18n.NumberFormatSymbols_ar_ER', 'goog.i18n.NumberFormatSymbols_ar_ER_u_nu_latn', 'goog.i18n.NumberFormatSymbols_ar_IL', 'goog.i18n.NumberFormatSymbols_ar_IL_u_nu_latn', 'goog.i18n.NumberFormatSymbols_ar_IQ', 'goog.i18n.NumberFormatSymbols_ar_IQ_u_nu_latn', 'goog.i18n.NumberFormatSymbols_ar_JO', 'goog.i18n.NumberFormatSymbols_ar_JO_u_nu_latn', 'goog.i18n.NumberFormatSymbols_ar_KM', 'goog.i18n.NumberFormatSymbols_ar_KM_u_nu_latn', 'goog.i18n.NumberFormatSymbols_ar_KW', 'goog.i18n.NumberFormatSymbols_ar_KW_u_nu_latn', 'goog.i18n.NumberFormatSymbols_ar_LB', 'goog.i18n.NumberFormatSymbols_ar_LB_u_nu_latn', 'goog.i18n.NumberFormatSymbols_ar_LY', 'goog.i18n.NumberFormatSymbols_ar_MA', 'goog.i18n.NumberFormatSymbols_ar_MR', 'goog.i18n.NumberFormatSymbols_ar_MR_u_nu_latn', 'goog.i18n.NumberFormatSymbols_ar_OM', 'goog.i18n.NumberFormatSymbols_ar_OM_u_nu_latn', 'goog.i18n.NumberFormatSymbols_ar_PS', 'goog.i18n.NumberFormatSymbols_ar_PS_u_nu_latn', 'goog.i18n.NumberFormatSymbols_ar_QA', 'goog.i18n.NumberFormatSymbols_ar_QA_u_nu_latn', 'goog.i18n.NumberFormatSymbols_ar_SA', 'goog.i18n.NumberFormatSymbols_ar_SA_u_nu_latn', 'goog.i18n.NumberFormatSymbols_ar_SD', 'goog.i18n.NumberFormatSymbols_ar_SD_u_nu_latn', 'goog.i18n.NumberFormatSymbols_ar_SO', 'goog.i18n.NumberFormatSymbols_ar_SO_u_nu_latn', 'goog.i18n.NumberFormatSymbols_ar_SS', 'goog.i18n.NumberFormatSymbols_ar_SS_u_nu_latn', 'goog.i18n.NumberFormatSymbols_ar_SY', 'goog.i18n.NumberFormatSymbols_ar_SY_u_nu_latn', 'goog.i18n.NumberFormatSymbols_ar_TD', 'goog.i18n.NumberFormatSymbols_ar_TD_u_nu_latn', 'goog.i18n.NumberFormatSymbols_ar_TN', 'goog.i18n.NumberFormatSymbols_ar_XB', 'goog.i18n.NumberFormatSymbols_ar_YE', 'goog.i18n.NumberFormatSymbols_ar_YE_u_nu_latn', 'goog.i18n.NumberFormatSymbols_as', 'goog.i18n.NumberFormatSymbols_as_IN', 'goog.i18n.NumberFormatSymbols_as_IN_u_nu_latn', 'goog.i18n.NumberFormatSymbols_as_u_nu_latn', 'goog.i18n.NumberFormatSymbols_asa', 'goog.i18n.NumberFormatSymbols_asa_TZ', 'goog.i18n.NumberFormatSymbols_ast', 'goog.i18n.NumberFormatSymbols_ast_ES', 'goog.i18n.NumberFormatSymbols_az_Cyrl', 'goog.i18n.NumberFormatSymbols_az_Cyrl_AZ', 'goog.i18n.NumberFormatSymbols_az_Latn', 'goog.i18n.NumberFormatSymbols_az_Latn_AZ', 'goog.i18n.NumberFormatSymbols_bas', 'goog.i18n.NumberFormatSymbols_bas_CM', 'goog.i18n.NumberFormatSymbols_be_BY', 'goog.i18n.NumberFormatSymbols_bem', 'goog.i18n.NumberFormatSymbols_bem_ZM', 'goog.i18n.NumberFormatSymbols_bez', 'goog.i18n.NumberFormatSymbols_bez_TZ', 'goog.i18n.NumberFormatSymbols_bg_BG', 'goog.i18n.NumberFormatSymbols_bm', 'goog.i18n.NumberFormatSymbols_bm_ML', 'goog.i18n.NumberFormatSymbols_bn_BD', 'goog.i18n.NumberFormatSymbols_bn_BD_u_nu_latn', 'goog.i18n.NumberFormatSymbols_bn_IN', 'goog.i18n.NumberFormatSymbols_bn_IN_u_nu_latn', 'goog.i18n.NumberFormatSymbols_bo', 'goog.i18n.NumberFormatSymbols_bo_CN', 'goog.i18n.NumberFormatSymbols_bo_IN', 'goog.i18n.NumberFormatSymbols_br_FR', 'goog.i18n.NumberFormatSymbols_brx', 'goog.i18n.NumberFormatSymbols_brx_IN', 'goog.i18n.NumberFormatSymbols_bs_Cyrl', 'goog.i18n.NumberFormatSymbols_bs_Cyrl_BA', 'goog.i18n.NumberFormatSymbols_bs_Latn', 'goog.i18n.NumberFormatSymbols_bs_Latn_BA', 'goog.i18n.NumberFormatSymbols_ca_AD', 'goog.i18n.NumberFormatSymbols_ca_ES', 'goog.i18n.NumberFormatSymbols_ca_FR', 'goog.i18n.NumberFormatSymbols_ca_IT', 'goog.i18n.NumberFormatSymbols_ccp', 'goog.i18n.NumberFormatSymbols_ccp_BD', 'goog.i18n.NumberFormatSymbols_ccp_BD_u_nu_latn', 'goog.i18n.NumberFormatSymbols_ccp_IN', 'goog.i18n.NumberFormatSymbols_ccp_IN_u_nu_latn', 'goog.i18n.NumberFormatSymbols_ccp_u_nu_latn', 'goog.i18n.NumberFormatSymbols_ce', 'goog.i18n.NumberFormatSymbols_ce_RU', 'goog.i18n.NumberFormatSymbols_ceb', 'goog.i18n.NumberFormatSymbols_ceb_PH', 'goog.i18n.NumberFormatSymbols_cgg', 'goog.i18n.NumberFormatSymbols_cgg_UG', 'goog.i18n.NumberFormatSymbols_chr_US', 'goog.i18n.NumberFormatSymbols_ckb', 'goog.i18n.NumberFormatSymbols_ckb_IQ', 'goog.i18n.NumberFormatSymbols_ckb_IQ_u_nu_latn', 'goog.i18n.NumberFormatSymbols_ckb_IR', 'goog.i18n.NumberFormatSymbols_ckb_IR_u_nu_latn', 'goog.i18n.NumberFormatSymbols_ckb_u_nu_latn', 'goog.i18n.NumberFormatSymbols_cs_CZ', 'goog.i18n.NumberFormatSymbols_cy_GB', 'goog.i18n.NumberFormatSymbols_da_DK', 'goog.i18n.NumberFormatSymbols_da_GL', 'goog.i18n.NumberFormatSymbols_dav', 'goog.i18n.NumberFormatSymbols_dav_KE', 'goog.i18n.NumberFormatSymbols_de_BE', 'goog.i18n.NumberFormatSymbols_de_DE', 'goog.i18n.NumberFormatSymbols_de_IT', 'goog.i18n.NumberFormatSymbols_de_LI', 'goog.i18n.NumberFormatSymbols_de_LU', 'goog.i18n.NumberFormatSymbols_dje', 'goog.i18n.NumberFormatSymbols_dje_NE', 'goog.i18n.NumberFormatSymbols_dsb', 'goog.i18n.NumberFormatSymbols_dsb_DE', 'goog.i18n.NumberFormatSymbols_dua', 'goog.i18n.NumberFormatSymbols_dua_CM', 'goog.i18n.NumberFormatSymbols_dyo', 'goog.i18n.NumberFormatSymbols_dyo_SN', 'goog.i18n.NumberFormatSymbols_dz', 'goog.i18n.NumberFormatSymbols_dz_BT', 'goog.i18n.NumberFormatSymbols_dz_BT_u_nu_latn', 'goog.i18n.NumberFormatSymbols_dz_u_nu_latn', 'goog.i18n.NumberFormatSymbols_ebu', 'goog.i18n.NumberFormatSymbols_ebu_KE', 'goog.i18n.NumberFormatSymbols_ee', 'goog.i18n.NumberFormatSymbols_ee_GH', 'goog.i18n.NumberFormatSymbols_ee_TG', 'goog.i18n.NumberFormatSymbols_el_CY', 'goog.i18n.NumberFormatSymbols_el_GR', 'goog.i18n.NumberFormatSymbols_en_001', 'goog.i18n.NumberFormatSymbols_en_150', 'goog.i18n.NumberFormatSymbols_en_AE', 'goog.i18n.NumberFormatSymbols_en_AG', 'goog.i18n.NumberFormatSymbols_en_AI', 'goog.i18n.NumberFormatSymbols_en_AS', 'goog.i18n.NumberFormatSymbols_en_AT', 'goog.i18n.NumberFormatSymbols_en_BB', 'goog.i18n.NumberFormatSymbols_en_BE', 'goog.i18n.NumberFormatSymbols_en_BI', 'goog.i18n.NumberFormatSymbols_en_BM', 'goog.i18n.NumberFormatSymbols_en_BS', 'goog.i18n.NumberFormatSymbols_en_BW', 'goog.i18n.NumberFormatSymbols_en_BZ', 'goog.i18n.NumberFormatSymbols_en_CC', 'goog.i18n.NumberFormatSymbols_en_CH', 'goog.i18n.NumberFormatSymbols_en_CK', 'goog.i18n.NumberFormatSymbols_en_CM', 'goog.i18n.NumberFormatSymbols_en_CX', 'goog.i18n.NumberFormatSymbols_en_CY', 'goog.i18n.NumberFormatSymbols_en_DE', 'goog.i18n.NumberFormatSymbols_en_DG', 'goog.i18n.NumberFormatSymbols_en_DK', 'goog.i18n.NumberFormatSymbols_en_DM', 'goog.i18n.NumberFormatSymbols_en_ER', 'goog.i18n.NumberFormatSymbols_en_FI', 'goog.i18n.NumberFormatSymbols_en_FJ', 'goog.i18n.NumberFormatSymbols_en_FK', 'goog.i18n.NumberFormatSymbols_en_FM', 'goog.i18n.NumberFormatSymbols_en_GD', 'goog.i18n.NumberFormatSymbols_en_GG', 'goog.i18n.NumberFormatSymbols_en_GH', 'goog.i18n.NumberFormatSymbols_en_GI', 'goog.i18n.NumberFormatSymbols_en_GM', 'goog.i18n.NumberFormatSymbols_en_GU', 'goog.i18n.NumberFormatSymbols_en_GY', 'goog.i18n.NumberFormatSymbols_en_HK', 'goog.i18n.NumberFormatSymbols_en_IL', 'goog.i18n.NumberFormatSymbols_en_IM', 'goog.i18n.NumberFormatSymbols_en_IO', 'goog.i18n.NumberFormatSymbols_en_JE', 'goog.i18n.NumberFormatSymbols_en_JM', 'goog.i18n.NumberFormatSymbols_en_KE', 'goog.i18n.NumberFormatSymbols_en_KI', 'goog.i18n.NumberFormatSymbols_en_KN', 'goog.i18n.NumberFormatSymbols_en_KY', 'goog.i18n.NumberFormatSymbols_en_LC', 'goog.i18n.NumberFormatSymbols_en_LR', 'goog.i18n.NumberFormatSymbols_en_LS', 'goog.i18n.NumberFormatSymbols_en_MG', 'goog.i18n.NumberFormatSymbols_en_MH', 'goog.i18n.NumberFormatSymbols_en_MO', 'goog.i18n.NumberFormatSymbols_en_MP', 'goog.i18n.NumberFormatSymbols_en_MS', 'goog.i18n.NumberFormatSymbols_en_MT', 'goog.i18n.NumberFormatSymbols_en_MU', 'goog.i18n.NumberFormatSymbols_en_MW', 'goog.i18n.NumberFormatSymbols_en_MY', 'goog.i18n.NumberFormatSymbols_en_NA', 'goog.i18n.NumberFormatSymbols_en_NF', 'goog.i18n.NumberFormatSymbols_en_NG', 'goog.i18n.NumberFormatSymbols_en_NL', 'goog.i18n.NumberFormatSymbols_en_NR', 'goog.i18n.NumberFormatSymbols_en_NU', 'goog.i18n.NumberFormatSymbols_en_NZ', 'goog.i18n.NumberFormatSymbols_en_PG', 'goog.i18n.NumberFormatSymbols_en_PH', 'goog.i18n.NumberFormatSymbols_en_PK', 'goog.i18n.NumberFormatSymbols_en_PN', 'goog.i18n.NumberFormatSymbols_en_PR', 'goog.i18n.NumberFormatSymbols_en_PW', 'goog.i18n.NumberFormatSymbols_en_RW', 'goog.i18n.NumberFormatSymbols_en_SB', 'goog.i18n.NumberFormatSymbols_en_SC', 'goog.i18n.NumberFormatSymbols_en_SD', 'goog.i18n.NumberFormatSymbols_en_SE', 'goog.i18n.NumberFormatSymbols_en_SH', 'goog.i18n.NumberFormatSymbols_en_SI', 'goog.i18n.NumberFormatSymbols_en_SL', 'goog.i18n.NumberFormatSymbols_en_SS', 'goog.i18n.NumberFormatSymbols_en_SX', 'goog.i18n.NumberFormatSymbols_en_SZ', 'goog.i18n.NumberFormatSymbols_en_TC', 'goog.i18n.NumberFormatSymbols_en_TK', 'goog.i18n.NumberFormatSymbols_en_TO', 'goog.i18n.NumberFormatSymbols_en_TT', 'goog.i18n.NumberFormatSymbols_en_TV', 'goog.i18n.NumberFormatSymbols_en_TZ', 'goog.i18n.NumberFormatSymbols_en_UG', 'goog.i18n.NumberFormatSymbols_en_UM', 'goog.i18n.NumberFormatSymbols_en_US_POSIX', 'goog.i18n.NumberFormatSymbols_en_VC', 'goog.i18n.NumberFormatSymbols_en_VG', 'goog.i18n.NumberFormatSymbols_en_VI', 'goog.i18n.NumberFormatSymbols_en_VU', 'goog.i18n.NumberFormatSymbols_en_WS', 'goog.i18n.NumberFormatSymbols_en_XA', 'goog.i18n.NumberFormatSymbols_en_ZM', 'goog.i18n.NumberFormatSymbols_en_ZW', 'goog.i18n.NumberFormatSymbols_eo', 'goog.i18n.NumberFormatSymbols_eo_001', 'goog.i18n.NumberFormatSymbols_es_AR', 'goog.i18n.NumberFormatSymbols_es_BO', 'goog.i18n.NumberFormatSymbols_es_BR', 'goog.i18n.NumberFormatSymbols_es_BZ', 'goog.i18n.NumberFormatSymbols_es_CL', 'goog.i18n.NumberFormatSymbols_es_CO', 'goog.i18n.NumberFormatSymbols_es_CR', 'goog.i18n.NumberFormatSymbols_es_CU', 'goog.i18n.NumberFormatSymbols_es_DO', 'goog.i18n.NumberFormatSymbols_es_EA', 'goog.i18n.NumberFormatSymbols_es_EC', 'goog.i18n.NumberFormatSymbols_es_GQ', 'goog.i18n.NumberFormatSymbols_es_GT', 'goog.i18n.NumberFormatSymbols_es_HN', 'goog.i18n.NumberFormatSymbols_es_IC', 'goog.i18n.NumberFormatSymbols_es_NI', 'goog.i18n.NumberFormatSymbols_es_PA', 'goog.i18n.NumberFormatSymbols_es_PE', 'goog.i18n.NumberFormatSymbols_es_PH', 'goog.i18n.NumberFormatSymbols_es_PR', 'goog.i18n.NumberFormatSymbols_es_PY', 'goog.i18n.NumberFormatSymbols_es_SV', 'goog.i18n.NumberFormatSymbols_es_UY', 'goog.i18n.NumberFormatSymbols_es_VE', 'goog.i18n.NumberFormatSymbols_et_EE', 'goog.i18n.NumberFormatSymbols_eu_ES', 'goog.i18n.NumberFormatSymbols_ewo', 'goog.i18n.NumberFormatSymbols_ewo_CM', 'goog.i18n.NumberFormatSymbols_fa_AF', 'goog.i18n.NumberFormatSymbols_fa_AF_u_nu_latn', 'goog.i18n.NumberFormatSymbols_fa_IR', 'goog.i18n.NumberFormatSymbols_fa_IR_u_nu_latn', 'goog.i18n.NumberFormatSymbols_ff', 'goog.i18n.NumberFormatSymbols_ff_Latn', 'goog.i18n.NumberFormatSymbols_ff_Latn_BF', 'goog.i18n.NumberFormatSymbols_ff_Latn_CM', 'goog.i18n.NumberFormatSymbols_ff_Latn_GH', 'goog.i18n.NumberFormatSymbols_ff_Latn_GM', 'goog.i18n.NumberFormatSymbols_ff_Latn_GN', 'goog.i18n.NumberFormatSymbols_ff_Latn_GW', 'goog.i18n.NumberFormatSymbols_ff_Latn_LR', 'goog.i18n.NumberFormatSymbols_ff_Latn_MR', 'goog.i18n.NumberFormatSymbols_ff_Latn_NE', 'goog.i18n.NumberFormatSymbols_ff_Latn_NG', 'goog.i18n.NumberFormatSymbols_ff_Latn_SL', 'goog.i18n.NumberFormatSymbols_ff_Latn_SN', 'goog.i18n.NumberFormatSymbols_fi_FI', 'goog.i18n.NumberFormatSymbols_fil_PH', 'goog.i18n.NumberFormatSymbols_fo', 'goog.i18n.NumberFormatSymbols_fo_DK', 'goog.i18n.NumberFormatSymbols_fo_FO', 'goog.i18n.NumberFormatSymbols_fr_BE', 'goog.i18n.NumberFormatSymbols_fr_BF', 'goog.i18n.NumberFormatSymbols_fr_BI', 'goog.i18n.NumberFormatSymbols_fr_BJ', 'goog.i18n.NumberFormatSymbols_fr_BL', 'goog.i18n.NumberFormatSymbols_fr_CD', 'goog.i18n.NumberFormatSymbols_fr_CF', 'goog.i18n.NumberFormatSymbols_fr_CG', 'goog.i18n.NumberFormatSymbols_fr_CH', 'goog.i18n.NumberFormatSymbols_fr_CI', 'goog.i18n.NumberFormatSymbols_fr_CM', 'goog.i18n.NumberFormatSymbols_fr_DJ', 'goog.i18n.NumberFormatSymbols_fr_DZ', 'goog.i18n.NumberFormatSymbols_fr_FR', 'goog.i18n.NumberFormatSymbols_fr_GA', 'goog.i18n.NumberFormatSymbols_fr_GF', 'goog.i18n.NumberFormatSymbols_fr_GN', 'goog.i18n.NumberFormatSymbols_fr_GP', 'goog.i18n.NumberFormatSymbols_fr_GQ', 'goog.i18n.NumberFormatSymbols_fr_HT', 'goog.i18n.NumberFormatSymbols_fr_KM', 'goog.i18n.NumberFormatSymbols_fr_LU', 'goog.i18n.NumberFormatSymbols_fr_MA', 'goog.i18n.NumberFormatSymbols_fr_MC', 'goog.i18n.NumberFormatSymbols_fr_MF', 'goog.i18n.NumberFormatSymbols_fr_MG', 'goog.i18n.NumberFormatSymbols_fr_ML', 'goog.i18n.NumberFormatSymbols_fr_MQ', 'goog.i18n.NumberFormatSymbols_fr_MR', 'goog.i18n.NumberFormatSymbols_fr_MU', 'goog.i18n.NumberFormatSymbols_fr_NC', 'goog.i18n.NumberFormatSymbols_fr_NE', 'goog.i18n.NumberFormatSymbols_fr_PF', 'goog.i18n.NumberFormatSymbols_fr_PM', 'goog.i18n.NumberFormatSymbols_fr_RE', 'goog.i18n.NumberFormatSymbols_fr_RW', 'goog.i18n.NumberFormatSymbols_fr_SC', 'goog.i18n.NumberFormatSymbols_fr_SN', 'goog.i18n.NumberFormatSymbols_fr_SY', 'goog.i18n.NumberFormatSymbols_fr_TD', 'goog.i18n.NumberFormatSymbols_fr_TG', 'goog.i18n.NumberFormatSymbols_fr_TN', 'goog.i18n.NumberFormatSymbols_fr_VU', 'goog.i18n.NumberFormatSymbols_fr_WF', 'goog.i18n.NumberFormatSymbols_fr_YT', 'goog.i18n.NumberFormatSymbols_fur', 'goog.i18n.NumberFormatSymbols_fur_IT', 'goog.i18n.NumberFormatSymbols_fy', 'goog.i18n.NumberFormatSymbols_fy_NL', 'goog.i18n.NumberFormatSymbols_ga_IE', 'goog.i18n.NumberFormatSymbols_gd', 'goog.i18n.NumberFormatSymbols_gd_GB', 'goog.i18n.NumberFormatSymbols_gl_ES', 'goog.i18n.NumberFormatSymbols_gsw_CH', 'goog.i18n.NumberFormatSymbols_gsw_FR', 'goog.i18n.NumberFormatSymbols_gsw_LI', 'goog.i18n.NumberFormatSymbols_gu_IN', 'goog.i18n.NumberFormatSymbols_guz', 'goog.i18n.NumberFormatSymbols_guz_KE', 'goog.i18n.NumberFormatSymbols_gv', 'goog.i18n.NumberFormatSymbols_gv_IM', 'goog.i18n.NumberFormatSymbols_ha', 'goog.i18n.NumberFormatSymbols_ha_GH', 'goog.i18n.NumberFormatSymbols_ha_NE', 'goog.i18n.NumberFormatSymbols_ha_NG', 'goog.i18n.NumberFormatSymbols_haw_US', 'goog.i18n.NumberFormatSymbols_he_IL', 'goog.i18n.NumberFormatSymbols_hi_IN', 'goog.i18n.NumberFormatSymbols_hr_BA', 'goog.i18n.NumberFormatSymbols_hr_HR', 'goog.i18n.NumberFormatSymbols_hsb', 'goog.i18n.NumberFormatSymbols_hsb_DE', 'goog.i18n.NumberFormatSymbols_hu_HU', 'goog.i18n.NumberFormatSymbols_hy_AM', 'goog.i18n.NumberFormatSymbols_ia', 'goog.i18n.NumberFormatSymbols_ia_001', 'goog.i18n.NumberFormatSymbols_id_ID', 'goog.i18n.NumberFormatSymbols_ig', 'goog.i18n.NumberFormatSymbols_ig_NG', 'goog.i18n.NumberFormatSymbols_ii', 'goog.i18n.NumberFormatSymbols_ii_CN', 'goog.i18n.NumberFormatSymbols_is_IS', 'goog.i18n.NumberFormatSymbols_it_CH', 'goog.i18n.NumberFormatSymbols_it_IT', 'goog.i18n.NumberFormatSymbols_it_SM', 'goog.i18n.NumberFormatSymbols_it_VA', 'goog.i18n.NumberFormatSymbols_ja_JP', 'goog.i18n.NumberFormatSymbols_jgo', 'goog.i18n.NumberFormatSymbols_jgo_CM', 'goog.i18n.NumberFormatSymbols_jmc', 'goog.i18n.NumberFormatSymbols_jmc_TZ', 'goog.i18n.NumberFormatSymbols_jv', 'goog.i18n.NumberFormatSymbols_jv_ID', 'goog.i18n.NumberFormatSymbols_ka_GE', 'goog.i18n.NumberFormatSymbols_kab', 'goog.i18n.NumberFormatSymbols_kab_DZ', 'goog.i18n.NumberFormatSymbols_kam', 'goog.i18n.NumberFormatSymbols_kam_KE', 'goog.i18n.NumberFormatSymbols_kde', 'goog.i18n.NumberFormatSymbols_kde_TZ', 'goog.i18n.NumberFormatSymbols_kea', 'goog.i18n.NumberFormatSymbols_kea_CV', 'goog.i18n.NumberFormatSymbols_khq', 'goog.i18n.NumberFormatSymbols_khq_ML', 'goog.i18n.NumberFormatSymbols_ki', 'goog.i18n.NumberFormatSymbols_ki_KE', 'goog.i18n.NumberFormatSymbols_kk_KZ', 'goog.i18n.NumberFormatSymbols_kkj', 'goog.i18n.NumberFormatSymbols_kkj_CM', 'goog.i18n.NumberFormatSymbols_kl', 'goog.i18n.NumberFormatSymbols_kl_GL', 'goog.i18n.NumberFormatSymbols_kln', 'goog.i18n.NumberFormatSymbols_kln_KE', 'goog.i18n.NumberFormatSymbols_km_KH', 'goog.i18n.NumberFormatSymbols_kn_IN', 'goog.i18n.NumberFormatSymbols_ko_KP', 'goog.i18n.NumberFormatSymbols_ko_KR', 'goog.i18n.NumberFormatSymbols_kok', 'goog.i18n.NumberFormatSymbols_kok_IN', 'goog.i18n.NumberFormatSymbols_ks', 'goog.i18n.NumberFormatSymbols_ks_IN', 'goog.i18n.NumberFormatSymbols_ks_IN_u_nu_latn', 'goog.i18n.NumberFormatSymbols_ks_u_nu_latn', 'goog.i18n.NumberFormatSymbols_ksb', 'goog.i18n.NumberFormatSymbols_ksb_TZ', 'goog.i18n.NumberFormatSymbols_ksf', 'goog.i18n.NumberFormatSymbols_ksf_CM', 'goog.i18n.NumberFormatSymbols_ksh', 'goog.i18n.NumberFormatSymbols_ksh_DE', 'goog.i18n.NumberFormatSymbols_ku', 'goog.i18n.NumberFormatSymbols_ku_TR', 'goog.i18n.NumberFormatSymbols_kw', 'goog.i18n.NumberFormatSymbols_kw_GB', 'goog.i18n.NumberFormatSymbols_ky_KG', 'goog.i18n.NumberFormatSymbols_lag', 'goog.i18n.NumberFormatSymbols_lag_TZ', 'goog.i18n.NumberFormatSymbols_lb', 'goog.i18n.NumberFormatSymbols_lb_LU', 'goog.i18n.NumberFormatSymbols_lg', 'goog.i18n.NumberFormatSymbols_lg_UG', 'goog.i18n.NumberFormatSymbols_lkt', 'goog.i18n.NumberFormatSymbols_lkt_US', 'goog.i18n.NumberFormatSymbols_ln_AO', 'goog.i18n.NumberFormatSymbols_ln_CD', 'goog.i18n.NumberFormatSymbols_ln_CF', 'goog.i18n.NumberFormatSymbols_ln_CG', 'goog.i18n.NumberFormatSymbols_lo_LA', 'goog.i18n.NumberFormatSymbols_lrc', 'goog.i18n.NumberFormatSymbols_lrc_IQ', 'goog.i18n.NumberFormatSymbols_lrc_IQ_u_nu_latn', 'goog.i18n.NumberFormatSymbols_lrc_IR', 'goog.i18n.NumberFormatSymbols_lrc_IR_u_nu_latn', 'goog.i18n.NumberFormatSymbols_lrc_u_nu_latn', 'goog.i18n.NumberFormatSymbols_lt_LT', 'goog.i18n.NumberFormatSymbols_lu', 'goog.i18n.NumberFormatSymbols_lu_CD', 'goog.i18n.NumberFormatSymbols_luo', 'goog.i18n.NumberFormatSymbols_luo_KE', 'goog.i18n.NumberFormatSymbols_luy', 'goog.i18n.NumberFormatSymbols_luy_KE', 'goog.i18n.NumberFormatSymbols_lv_LV', 'goog.i18n.NumberFormatSymbols_mas', 'goog.i18n.NumberFormatSymbols_mas_KE', 'goog.i18n.NumberFormatSymbols_mas_TZ', 'goog.i18n.NumberFormatSymbols_mer', 'goog.i18n.NumberFormatSymbols_mer_KE', 'goog.i18n.NumberFormatSymbols_mfe', 'goog.i18n.NumberFormatSymbols_mfe_MU', 'goog.i18n.NumberFormatSymbols_mg', 'goog.i18n.NumberFormatSymbols_mg_MG', 'goog.i18n.NumberFormatSymbols_mgh', 'goog.i18n.NumberFormatSymbols_mgh_MZ', 'goog.i18n.NumberFormatSymbols_mgo', 'goog.i18n.NumberFormatSymbols_mgo_CM', 'goog.i18n.NumberFormatSymbols_mi', 'goog.i18n.NumberFormatSymbols_mi_NZ', 'goog.i18n.NumberFormatSymbols_mk_MK', 'goog.i18n.NumberFormatSymbols_ml_IN', 'goog.i18n.NumberFormatSymbols_mn_MN', 'goog.i18n.NumberFormatSymbols_mr_IN', 'goog.i18n.NumberFormatSymbols_mr_IN_u_nu_latn', 'goog.i18n.NumberFormatSymbols_ms_BN', 'goog.i18n.NumberFormatSymbols_ms_MY', 'goog.i18n.NumberFormatSymbols_ms_SG', 'goog.i18n.NumberFormatSymbols_mt_MT', 'goog.i18n.NumberFormatSymbols_mua', 'goog.i18n.NumberFormatSymbols_mua_CM', 'goog.i18n.NumberFormatSymbols_my_MM', 'goog.i18n.NumberFormatSymbols_my_MM_u_nu_latn', 'goog.i18n.NumberFormatSymbols_mzn', 'goog.i18n.NumberFormatSymbols_mzn_IR', 'goog.i18n.NumberFormatSymbols_mzn_IR_u_nu_latn', 'goog.i18n.NumberFormatSymbols_mzn_u_nu_latn', 'goog.i18n.NumberFormatSymbols_naq', 'goog.i18n.NumberFormatSymbols_naq_NA', 'goog.i18n.NumberFormatSymbols_nb_NO', 'goog.i18n.NumberFormatSymbols_nb_SJ', 'goog.i18n.NumberFormatSymbols_nd', 'goog.i18n.NumberFormatSymbols_nd_ZW', 'goog.i18n.NumberFormatSymbols_nds', 'goog.i18n.NumberFormatSymbols_nds_DE', 'goog.i18n.NumberFormatSymbols_nds_NL', 'goog.i18n.NumberFormatSymbols_ne_IN', 'goog.i18n.NumberFormatSymbols_ne_IN_u_nu_latn', 'goog.i18n.NumberFormatSymbols_ne_NP', 'goog.i18n.NumberFormatSymbols_ne_NP_u_nu_latn', 'goog.i18n.NumberFormatSymbols_nl_AW', 'goog.i18n.NumberFormatSymbols_nl_BE', 'goog.i18n.NumberFormatSymbols_nl_BQ', 'goog.i18n.NumberFormatSymbols_nl_CW', 'goog.i18n.NumberFormatSymbols_nl_NL', 'goog.i18n.NumberFormatSymbols_nl_SR', 'goog.i18n.NumberFormatSymbols_nl_SX', 'goog.i18n.NumberFormatSymbols_nmg', 'goog.i18n.NumberFormatSymbols_nmg_CM', 'goog.i18n.NumberFormatSymbols_nn', 'goog.i18n.NumberFormatSymbols_nn_NO', 'goog.i18n.NumberFormatSymbols_nnh', 'goog.i18n.NumberFormatSymbols_nnh_CM', 'goog.i18n.NumberFormatSymbols_nus', 'goog.i18n.NumberFormatSymbols_nus_SS', 'goog.i18n.NumberFormatSymbols_nyn', 'goog.i18n.NumberFormatSymbols_nyn_UG', 'goog.i18n.NumberFormatSymbols_om', 'goog.i18n.NumberFormatSymbols_om_ET', 'goog.i18n.NumberFormatSymbols_om_KE', 'goog.i18n.NumberFormatSymbols_or_IN', 'goog.i18n.NumberFormatSymbols_os', 'goog.i18n.NumberFormatSymbols_os_GE', 'goog.i18n.NumberFormatSymbols_os_RU', 'goog.i18n.NumberFormatSymbols_pa_Arab', 'goog.i18n.NumberFormatSymbols_pa_Arab_PK', 'goog.i18n.NumberFormatSymbols_pa_Arab_PK_u_nu_latn', 'goog.i18n.NumberFormatSymbols_pa_Arab_u_nu_latn', 'goog.i18n.NumberFormatSymbols_pa_Guru', 'goog.i18n.NumberFormatSymbols_pa_Guru_IN', 'goog.i18n.NumberFormatSymbols_pl_PL', 'goog.i18n.NumberFormatSymbols_ps', 'goog.i18n.NumberFormatSymbols_ps_AF', 'goog.i18n.NumberFormatSymbols_ps_AF_u_nu_latn', 'goog.i18n.NumberFormatSymbols_ps_PK', 'goog.i18n.NumberFormatSymbols_ps_PK_u_nu_latn', 'goog.i18n.NumberFormatSymbols_ps_u_nu_latn', 'goog.i18n.NumberFormatSymbols_pt_AO', 'goog.i18n.NumberFormatSymbols_pt_CH', 'goog.i18n.NumberFormatSymbols_pt_CV', 'goog.i18n.NumberFormatSymbols_pt_GQ', 'goog.i18n.NumberFormatSymbols_pt_GW', 'goog.i18n.NumberFormatSymbols_pt_LU', 'goog.i18n.NumberFormatSymbols_pt_MO', 'goog.i18n.NumberFormatSymbols_pt_MZ', 'goog.i18n.NumberFormatSymbols_pt_ST', 'goog.i18n.NumberFormatSymbols_pt_TL', 'goog.i18n.NumberFormatSymbols_qu', 'goog.i18n.NumberFormatSymbols_qu_BO', 'goog.i18n.NumberFormatSymbols_qu_EC', 'goog.i18n.NumberFormatSymbols_qu_PE', 'goog.i18n.NumberFormatSymbols_rm', 'goog.i18n.NumberFormatSymbols_rm_CH', 'goog.i18n.NumberFormatSymbols_rn', 'goog.i18n.NumberFormatSymbols_rn_BI', 'goog.i18n.NumberFormatSymbols_ro_MD', 'goog.i18n.NumberFormatSymbols_ro_RO', 'goog.i18n.NumberFormatSymbols_rof', 'goog.i18n.NumberFormatSymbols_rof_TZ', 'goog.i18n.NumberFormatSymbols_ru_BY', 'goog.i18n.NumberFormatSymbols_ru_KG', 'goog.i18n.NumberFormatSymbols_ru_KZ', 'goog.i18n.NumberFormatSymbols_ru_MD', 'goog.i18n.NumberFormatSymbols_ru_RU', 'goog.i18n.NumberFormatSymbols_ru_UA', 'goog.i18n.NumberFormatSymbols_rw', 'goog.i18n.NumberFormatSymbols_rw_RW', 'goog.i18n.NumberFormatSymbols_rwk', 'goog.i18n.NumberFormatSymbols_rwk_TZ', 'goog.i18n.NumberFormatSymbols_sah', 'goog.i18n.NumberFormatSymbols_sah_RU', 'goog.i18n.NumberFormatSymbols_saq', 'goog.i18n.NumberFormatSymbols_saq_KE', 'goog.i18n.NumberFormatSymbols_sbp', 'goog.i18n.NumberFormatSymbols_sbp_TZ', 'goog.i18n.NumberFormatSymbols_sd', 'goog.i18n.NumberFormatSymbols_sd_PK', 'goog.i18n.NumberFormatSymbols_sd_PK_u_nu_latn', 'goog.i18n.NumberFormatSymbols_sd_u_nu_latn', 'goog.i18n.NumberFormatSymbols_se', 'goog.i18n.NumberFormatSymbols_se_FI', 'goog.i18n.NumberFormatSymbols_se_NO', 'goog.i18n.NumberFormatSymbols_se_SE', 'goog.i18n.NumberFormatSymbols_seh', 'goog.i18n.NumberFormatSymbols_seh_MZ', 'goog.i18n.NumberFormatSymbols_ses', 'goog.i18n.NumberFormatSymbols_ses_ML', 'goog.i18n.NumberFormatSymbols_sg', 'goog.i18n.NumberFormatSymbols_sg_CF', 'goog.i18n.NumberFormatSymbols_shi', 'goog.i18n.NumberFormatSymbols_shi_Latn', 'goog.i18n.NumberFormatSymbols_shi_Latn_MA', 'goog.i18n.NumberFormatSymbols_shi_Tfng', 'goog.i18n.NumberFormatSymbols_shi_Tfng_MA', 'goog.i18n.NumberFormatSymbols_si_LK', 'goog.i18n.NumberFormatSymbols_sk_SK', 'goog.i18n.NumberFormatSymbols_sl_SI', 'goog.i18n.NumberFormatSymbols_smn', 'goog.i18n.NumberFormatSymbols_smn_FI', 'goog.i18n.NumberFormatSymbols_sn', 'goog.i18n.NumberFormatSymbols_sn_ZW', 'goog.i18n.NumberFormatSymbols_so', 'goog.i18n.NumberFormatSymbols_so_DJ', 'goog.i18n.NumberFormatSymbols_so_ET', 'goog.i18n.NumberFormatSymbols_so_KE', 'goog.i18n.NumberFormatSymbols_so_SO', 'goog.i18n.NumberFormatSymbols_sq_AL', 'goog.i18n.NumberFormatSymbols_sq_MK', 'goog.i18n.NumberFormatSymbols_sq_XK', 'goog.i18n.NumberFormatSymbols_sr_Cyrl', 'goog.i18n.NumberFormatSymbols_sr_Cyrl_BA', 'goog.i18n.NumberFormatSymbols_sr_Cyrl_ME', 'goog.i18n.NumberFormatSymbols_sr_Cyrl_RS', 'goog.i18n.NumberFormatSymbols_sr_Cyrl_XK', 'goog.i18n.NumberFormatSymbols_sr_Latn_BA', 'goog.i18n.NumberFormatSymbols_sr_Latn_ME', 'goog.i18n.NumberFormatSymbols_sr_Latn_RS', 'goog.i18n.NumberFormatSymbols_sr_Latn_XK', 'goog.i18n.NumberFormatSymbols_sv_AX', 'goog.i18n.NumberFormatSymbols_sv_FI', 'goog.i18n.NumberFormatSymbols_sv_SE', 'goog.i18n.NumberFormatSymbols_sw_CD', 'goog.i18n.NumberFormatSymbols_sw_KE', 'goog.i18n.NumberFormatSymbols_sw_TZ', 'goog.i18n.NumberFormatSymbols_sw_UG', 'goog.i18n.NumberFormatSymbols_ta_IN', 'goog.i18n.NumberFormatSymbols_ta_LK', 'goog.i18n.NumberFormatSymbols_ta_MY', 'goog.i18n.NumberFormatSymbols_ta_SG', 'goog.i18n.NumberFormatSymbols_te_IN', 'goog.i18n.NumberFormatSymbols_teo', 'goog.i18n.NumberFormatSymbols_teo_KE', 'goog.i18n.NumberFormatSymbols_teo_UG', 'goog.i18n.NumberFormatSymbols_tg', 'goog.i18n.NumberFormatSymbols_tg_TJ', 'goog.i18n.NumberFormatSymbols_th_TH', 'goog.i18n.NumberFormatSymbols_ti', 'goog.i18n.NumberFormatSymbols_ti_ER', 'goog.i18n.NumberFormatSymbols_ti_ET', 'goog.i18n.NumberFormatSymbols_tk', 'goog.i18n.NumberFormatSymbols_tk_TM', 'goog.i18n.NumberFormatSymbols_to', 'goog.i18n.NumberFormatSymbols_to_TO', 'goog.i18n.NumberFormatSymbols_tr_CY', 'goog.i18n.NumberFormatSymbols_tr_TR', 'goog.i18n.NumberFormatSymbols_tt', 'goog.i18n.NumberFormatSymbols_tt_RU', 'goog.i18n.NumberFormatSymbols_twq', 'goog.i18n.NumberFormatSymbols_twq_NE', 'goog.i18n.NumberFormatSymbols_tzm', 'goog.i18n.NumberFormatSymbols_tzm_MA', 'goog.i18n.NumberFormatSymbols_ug', 'goog.i18n.NumberFormatSymbols_ug_CN', 'goog.i18n.NumberFormatSymbols_uk_UA', 'goog.i18n.NumberFormatSymbols_ur_IN', 'goog.i18n.NumberFormatSymbols_ur_IN_u_nu_latn', 'goog.i18n.NumberFormatSymbols_ur_PK', 'goog.i18n.NumberFormatSymbols_uz_Arab', 'goog.i18n.NumberFormatSymbols_uz_Arab_AF', 'goog.i18n.NumberFormatSymbols_uz_Arab_AF_u_nu_latn', 'goog.i18n.NumberFormatSymbols_uz_Arab_u_nu_latn', 'goog.i18n.NumberFormatSymbols_uz_Cyrl', 'goog.i18n.NumberFormatSymbols_uz_Cyrl_UZ', 'goog.i18n.NumberFormatSymbols_uz_Latn', 'goog.i18n.NumberFormatSymbols_uz_Latn_UZ', 'goog.i18n.NumberFormatSymbols_vai', 'goog.i18n.NumberFormatSymbols_vai_Latn', 'goog.i18n.NumberFormatSymbols_vai_Latn_LR', 'goog.i18n.NumberFormatSymbols_vai_Vaii', 'goog.i18n.NumberFormatSymbols_vai_Vaii_LR', 'goog.i18n.NumberFormatSymbols_vi_VN', 'goog.i18n.NumberFormatSymbols_vun', 'goog.i18n.NumberFormatSymbols_vun_TZ', 'goog.i18n.NumberFormatSymbols_wae', 'goog.i18n.NumberFormatSymbols_wae_CH', 'goog.i18n.NumberFormatSymbols_wo', 'goog.i18n.NumberFormatSymbols_wo_SN', 'goog.i18n.NumberFormatSymbols_xh', 'goog.i18n.NumberFormatSymbols_xh_ZA', 'goog.i18n.NumberFormatSymbols_xog', 'goog.i18n.NumberFormatSymbols_xog_UG', 'goog.i18n.NumberFormatSymbols_yav', 'goog.i18n.NumberFormatSymbols_yav_CM', 'goog.i18n.NumberFormatSymbols_yi', 'goog.i18n.NumberFormatSymbols_yi_001', 'goog.i18n.NumberFormatSymbols_yo', 'goog.i18n.NumberFormatSymbols_yo_BJ', 'goog.i18n.NumberFormatSymbols_yo_NG', 'goog.i18n.NumberFormatSymbols_yue', 'goog.i18n.NumberFormatSymbols_yue_Hans', 'goog.i18n.NumberFormatSymbols_yue_Hans_CN', 'goog.i18n.NumberFormatSymbols_yue_Hant', 'goog.i18n.NumberFormatSymbols_yue_Hant_HK', 'goog.i18n.NumberFormatSymbols_zgh', 'goog.i18n.NumberFormatSymbols_zgh_MA', 'goog.i18n.NumberFormatSymbols_zh_Hans', 'goog.i18n.NumberFormatSymbols_zh_Hans_CN', 'goog.i18n.NumberFormatSymbols_zh_Hans_HK', 'goog.i18n.NumberFormatSymbols_zh_Hans_MO', 'goog.i18n.NumberFormatSymbols_zh_Hans_SG', 'goog.i18n.NumberFormatSymbols_zh_Hant', 'goog.i18n.NumberFormatSymbols_zh_Hant_HK', 'goog.i18n.NumberFormatSymbols_zh_Hant_MO', 'goog.i18n.NumberFormatSymbols_zh_Hant_TW', 'goog.i18n.NumberFormatSymbols_zu_ZA'], ['goog.i18n.NumberFormatSymbols', 'goog.i18n.NumberFormatSymbols_u_nu_latn'], {});
goog.addDependency('i18n/ordinalrules.js', ['goog.i18n.ordinalRules'], [], {'lang': 'es6'});
goog.addDependency('i18n/pluralrules.js', ['goog.i18n.pluralRules'], [], {'lang': 'es6'});
goog.addDependency('i18n/pluralrules_test.js', ['goog.i18n.pluralRulesTest'], ['goog.i18n.pluralRules', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('i18n/relativedatetimeformat.js', ['goog.i18n.RelativeDateTimeFormat'], ['goog.asserts', 'goog.i18n.LocaleFeature', 'goog.i18n.MessageFormat', 'goog.i18n.relativeDateTimeSymbols'], {'lang': 'es5', 'module': 'goog'});
goog.addDependency('i18n/relativedatetimeformat_test.js', ['goog.i18n.RelativeDateTimeFormatTest'], ['goog.i18n.LocaleFeature', 'goog.i18n.NumberFormatSymbols_ar_EG', 'goog.i18n.NumberFormatSymbols_en', 'goog.i18n.NumberFormatSymbols_es', 'goog.i18n.NumberFormatSymbols_fa', 'goog.i18n.RelativeDateTimeFormat', 'goog.i18n.relativeDateTimeSymbols', 'goog.i18n.relativeDateTimeSymbolsExt', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('i18n/relativedatetimesymbols.js', ['goog.i18n.relativeDateTimeSymbols'], [], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('i18n/relativedatetimesymbolsext.js', ['goog.i18n.relativeDateTimeSymbolsExt'], ['goog.i18n.relativeDateTimeSymbols'], {'lang': 'es5', 'module': 'goog'});
goog.addDependency('i18n/timezone.js', ['goog.i18n.TimeZone'], ['goog.array', 'goog.date.DateLike', 'goog.object', 'goog.string'], {});
goog.addDependency('i18n/timezone_test.js', ['goog.i18n.TimeZoneTest'], ['goog.i18n.TimeZone', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('i18n/uchar.js', ['goog.i18n.uChar'], [], {'lang': 'es6'});
goog.addDependency('i18n/uchar/localnamefetcher.js', ['goog.i18n.uChar.LocalNameFetcher'], ['goog.i18n.uChar.NameFetcher', 'goog.i18n.uCharNames', 'goog.log'], {});
goog.addDependency('i18n/uchar/localnamefetcher_test.js', ['goog.i18n.uChar.LocalNameFetcherTest'], ['goog.i18n.uChar.LocalNameFetcher', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('i18n/uchar/namefetcher.js', ['goog.i18n.uChar.NameFetcher'], [], {});
goog.addDependency('i18n/uchar/remotenamefetcher.js', ['goog.i18n.uChar.RemoteNameFetcher'], ['goog.Disposable', 'goog.Uri', 'goog.events', 'goog.i18n.uChar', 'goog.i18n.uChar.NameFetcher', 'goog.log', 'goog.net.EventType', 'goog.net.XhrIo'], {});
goog.addDependency('i18n/uchar/remotenamefetcher_test.js', ['goog.i18n.uChar.RemoteNameFetcherTest'], ['goog.i18n.uChar.RemoteNameFetcher', 'goog.net.XhrIo', 'goog.testing.net.XhrIo', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('i18n/uchar_test.js', ['goog.i18n.uCharTest'], ['goog.i18n.uChar', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('i18n/ucharnames.js', ['goog.i18n.uCharNames'], ['goog.i18n.uChar'], {});
goog.addDependency('i18n/ucharnames_test.js', ['goog.i18n.uCharNamesTest'], ['goog.i18n.uCharNames', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('iter/es6.js', ['goog.iter.es6'], ['goog.iter.Iterable', 'goog.iter.Iterator', 'goog.iter.StopIteration'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('iter/es6_test.js', ['goog.iter.es6Test'], ['goog.iter', 'goog.iter.es6', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('iter/iter.js', ['goog.iter', 'goog.iter.Iterable', 'goog.iter.Iterator', 'goog.iter.StopIteration'], ['goog.array', 'goog.asserts', 'goog.functions', 'goog.math'], {});
goog.addDependency('iter/iter_test.js', ['goog.iterTest'], ['goog.iter', 'goog.iter.Iterator', 'goog.iter.StopIteration', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('json/hybrid.js', ['goog.json.hybrid'], ['goog.asserts', 'goog.json'], {});
goog.addDependency('json/hybrid_test.js', ['goog.json.hybridTest'], ['goog.json', 'goog.json.hybrid', 'goog.testing.PropertyReplacer', 'goog.testing.recordFunction', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('json/json.js', ['goog.json', 'goog.json.Replacer', 'goog.json.Reviver', 'goog.json.Serializer'], [], {'lang': 'es6'});
goog.addDependency('json/json_perf.js', ['goog.jsonPerf'], ['goog.dom', 'goog.json', 'goog.math', 'goog.string', 'goog.testing.PerformanceTable', 'goog.testing.PropertyReplacer', 'goog.testing.jsunit'], {});
goog.addDependency('json/json_test.js', ['goog.jsonTest'], ['goog.functions', 'goog.json', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('json/jsonable.js', ['goog.json.Jsonable'], [], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('json/nativejsonprocessor.js', ['goog.json.NativeJsonProcessor'], ['goog.asserts', 'goog.json.Processor'], {});
goog.addDependency('json/processor.js', ['goog.json.Processor'], ['goog.string.Parser', 'goog.string.Stringifier'], {});
goog.addDependency('json/processor_test.js', ['goog.json.processorTest'], ['goog.json.NativeJsonProcessor', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/collections/iterables.js', ['goog.labs.collections.iterables'], [], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/collections/iterables_test.js', ['goog.labs.iterableTest'], ['goog.labs.collections.iterables', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/dom/pagevisibilitymonitor.js', ['goog.labs.dom.PageVisibilityEvent', 'goog.labs.dom.PageVisibilityMonitor', 'goog.labs.dom.PageVisibilityState'], ['goog.dom', 'goog.dom.vendor', 'goog.events', 'goog.events.Event', 'goog.events.EventTarget', 'goog.events.EventType', 'goog.memoize'], {});
goog.addDependency('labs/dom/pagevisibilitymonitor_test.js', ['goog.labs.dom.PageVisibilityMonitorTest'], ['goog.events', 'goog.functions', 'goog.labs.dom.PageVisibilityMonitor', 'goog.testing.PropertyReplacer', 'goog.testing.events', 'goog.testing.events.Event', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/events/nondisposableeventtarget.js', ['goog.labs.events.NonDisposableEventTarget'], ['goog.array', 'goog.asserts', 'goog.events.Event', 'goog.events.Listenable', 'goog.events.ListenerMap', 'goog.object'], {});
goog.addDependency('labs/events/nondisposableeventtarget_test.js', ['goog.labs.events.NonDisposableEventTargetTest'], ['goog.events.Listenable', 'goog.events.eventTargetTester', 'goog.labs.events.NonDisposableEventTarget', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/events/nondisposableeventtarget_via_googevents_test.js', ['goog.labs.events.NonDisposableEventTargetGoogEventsTest'], ['goog.events', 'goog.events.eventTargetTester', 'goog.labs.events.NonDisposableEventTarget', 'goog.testing', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/events/touch.js', ['goog.labs.events.touch', 'goog.labs.events.touch.TouchData'], ['goog.array', 'goog.asserts', 'goog.events.EventType', 'goog.string'], {});
goog.addDependency('labs/events/touch_test.js', ['goog.labs.events.touchTest'], ['goog.labs.events.touch', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/format/csv.js', ['goog.labs.format.csv', 'goog.labs.format.csv.ParseError', 'goog.labs.format.csv.Token'], ['goog.array', 'goog.asserts', 'goog.debug.Error', 'goog.object', 'goog.string', 'goog.string.newlines'], {});
goog.addDependency('labs/format/csv_test.js', ['goog.labs.format.csvTest'], ['goog.labs.format.csv', 'goog.labs.format.csv.ParseError', 'goog.object', 'goog.testing.asserts', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/i18n/listformat.js', ['goog.labs.i18n.GenderInfo', 'goog.labs.i18n.GenderInfo.Gender', 'goog.labs.i18n.ListFormat'], ['goog.asserts', 'goog.labs.i18n.ListFormatSymbols'], {});
goog.addDependency('labs/i18n/listformat_test.js', ['goog.labs.i18n.ListFormatTest'], ['goog.labs.i18n.GenderInfo', 'goog.labs.i18n.ListFormat', 'goog.labs.i18n.ListFormatSymbols', 'goog.labs.i18n.ListFormatSymbols_el', 'goog.labs.i18n.ListFormatSymbols_en', 'goog.labs.i18n.ListFormatSymbols_fr', 'goog.labs.i18n.ListFormatSymbols_ml', 'goog.labs.i18n.ListFormatSymbols_zu', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/i18n/listsymbols.js', ['goog.labs.i18n.ListFormatSymbols', 'goog.labs.i18n.ListFormatSymbols_af', 'goog.labs.i18n.ListFormatSymbols_am', 'goog.labs.i18n.ListFormatSymbols_ar', 'goog.labs.i18n.ListFormatSymbols_ar_DZ', 'goog.labs.i18n.ListFormatSymbols_ar_EG', 'goog.labs.i18n.ListFormatSymbols_az', 'goog.labs.i18n.ListFormatSymbols_be', 'goog.labs.i18n.ListFormatSymbols_bg', 'goog.labs.i18n.ListFormatSymbols_bn', 'goog.labs.i18n.ListFormatSymbols_br', 'goog.labs.i18n.ListFormatSymbols_bs', 'goog.labs.i18n.ListFormatSymbols_ca', 'goog.labs.i18n.ListFormatSymbols_chr', 'goog.labs.i18n.ListFormatSymbols_cs', 'goog.labs.i18n.ListFormatSymbols_cy', 'goog.labs.i18n.ListFormatSymbols_da', 'goog.labs.i18n.ListFormatSymbols_de', 'goog.labs.i18n.ListFormatSymbols_de_AT', 'goog.labs.i18n.ListFormatSymbols_de_CH', 'goog.labs.i18n.ListFormatSymbols_el', 'goog.labs.i18n.ListFormatSymbols_en', 'goog.labs.i18n.ListFormatSymbols_en_AU', 'goog.labs.i18n.ListFormatSymbols_en_CA', 'goog.labs.i18n.ListFormatSymbols_en_GB', 'goog.labs.i18n.ListFormatSymbols_en_IE', 'goog.labs.i18n.ListFormatSymbols_en_IN', 'goog.labs.i18n.ListFormatSymbols_en_SG', 'goog.labs.i18n.ListFormatSymbols_en_US', 'goog.labs.i18n.ListFormatSymbols_en_ZA', 'goog.labs.i18n.ListFormatSymbols_es', 'goog.labs.i18n.ListFormatSymbols_es_419', 'goog.labs.i18n.ListFormatSymbols_es_ES', 'goog.labs.i18n.ListFormatSymbols_es_MX', 'goog.labs.i18n.ListFormatSymbols_es_US', 'goog.labs.i18n.ListFormatSymbols_et', 'goog.labs.i18n.ListFormatSymbols_eu', 'goog.labs.i18n.ListFormatSymbols_fa', 'goog.labs.i18n.ListFormatSymbols_fi', 'goog.labs.i18n.ListFormatSymbols_fil', 'goog.labs.i18n.ListFormatSymbols_fr', 'goog.labs.i18n.ListFormatSymbols_fr_CA', 'goog.labs.i18n.ListFormatSymbols_ga', 'goog.labs.i18n.ListFormatSymbols_gl', 'goog.labs.i18n.ListFormatSymbols_gsw', 'goog.labs.i18n.ListFormatSymbols_gu', 'goog.labs.i18n.ListFormatSymbols_haw', 'goog.labs.i18n.ListFormatSymbols_he', 'goog.labs.i18n.ListFormatSymbols_hi', 'goog.labs.i18n.ListFormatSymbols_hr', 'goog.labs.i18n.ListFormatSymbols_hu', 'goog.labs.i18n.ListFormatSymbols_hy', 'goog.labs.i18n.ListFormatSymbols_id', 'goog.labs.i18n.ListFormatSymbols_in', 'goog.labs.i18n.ListFormatSymbols_is', 'goog.labs.i18n.ListFormatSymbols_it', 'goog.labs.i18n.ListFormatSymbols_iw', 'goog.labs.i18n.ListFormatSymbols_ja', 'goog.labs.i18n.ListFormatSymbols_ka', 'goog.labs.i18n.ListFormatSymbols_kk', 'goog.labs.i18n.ListFormatSymbols_km', 'goog.labs.i18n.ListFormatSymbols_kn', 'goog.labs.i18n.ListFormatSymbols_ko', 'goog.labs.i18n.ListFormatSymbols_ky', 'goog.labs.i18n.ListFormatSymbols_ln', 'goog.labs.i18n.ListFormatSymbols_lo', 'goog.labs.i18n.ListFormatSymbols_lt', 'goog.labs.i18n.ListFormatSymbols_lv', 'goog.labs.i18n.ListFormatSymbols_mk', 'goog.labs.i18n.ListFormatSymbols_ml', 'goog.labs.i18n.ListFormatSymbols_mn', 'goog.labs.i18n.ListFormatSymbols_mo', 'goog.labs.i18n.ListFormatSymbols_mr', 'goog.labs.i18n.ListFormatSymbols_ms', 'goog.labs.i18n.ListFormatSymbols_mt', 'goog.labs.i18n.ListFormatSymbols_my', 'goog.labs.i18n.ListFormatSymbols_nb', 'goog.labs.i18n.ListFormatSymbols_ne', 'goog.labs.i18n.ListFormatSymbols_nl', 'goog.labs.i18n.ListFormatSymbols_no', 'goog.labs.i18n.ListFormatSymbols_no_NO', 'goog.labs.i18n.ListFormatSymbols_or', 'goog.labs.i18n.ListFormatSymbols_pa', 'goog.labs.i18n.ListFormatSymbols_pl', 'goog.labs.i18n.ListFormatSymbols_pt', 'goog.labs.i18n.ListFormatSymbols_pt_BR', 'goog.labs.i18n.ListFormatSymbols_pt_PT', 'goog.labs.i18n.ListFormatSymbols_ro', 'goog.labs.i18n.ListFormatSymbols_ru', 'goog.labs.i18n.ListFormatSymbols_sh', 'goog.labs.i18n.ListFormatSymbols_si', 'goog.labs.i18n.ListFormatSymbols_sk', 'goog.labs.i18n.ListFormatSymbols_sl', 'goog.labs.i18n.ListFormatSymbols_sq', 'goog.labs.i18n.ListFormatSymbols_sr', 'goog.labs.i18n.ListFormatSymbols_sr_Latn', 'goog.labs.i18n.ListFormatSymbols_sv', 'goog.labs.i18n.ListFormatSymbols_sw', 'goog.labs.i18n.ListFormatSymbols_ta', 'goog.labs.i18n.ListFormatSymbols_te', 'goog.labs.i18n.ListFormatSymbols_th', 'goog.labs.i18n.ListFormatSymbols_tl', 'goog.labs.i18n.ListFormatSymbols_tr', 'goog.labs.i18n.ListFormatSymbols_uk', 'goog.labs.i18n.ListFormatSymbols_ur', 'goog.labs.i18n.ListFormatSymbols_uz', 'goog.labs.i18n.ListFormatSymbols_vi', 'goog.labs.i18n.ListFormatSymbols_zh', 'goog.labs.i18n.ListFormatSymbols_zh_CN', 'goog.labs.i18n.ListFormatSymbols_zh_HK', 'goog.labs.i18n.ListFormatSymbols_zh_TW', 'goog.labs.i18n.ListFormatSymbols_zu'], [], {});
goog.addDependency('labs/i18n/listsymbolsext.js', ['goog.labs.i18n.ListFormatSymbolsExt', 'goog.labs.i18n.ListFormatSymbols_af_NA', 'goog.labs.i18n.ListFormatSymbols_af_ZA', 'goog.labs.i18n.ListFormatSymbols_agq', 'goog.labs.i18n.ListFormatSymbols_agq_CM', 'goog.labs.i18n.ListFormatSymbols_ak', 'goog.labs.i18n.ListFormatSymbols_ak_GH', 'goog.labs.i18n.ListFormatSymbols_am_ET', 'goog.labs.i18n.ListFormatSymbols_ar_001', 'goog.labs.i18n.ListFormatSymbols_ar_AE', 'goog.labs.i18n.ListFormatSymbols_ar_BH', 'goog.labs.i18n.ListFormatSymbols_ar_DJ', 'goog.labs.i18n.ListFormatSymbols_ar_EH', 'goog.labs.i18n.ListFormatSymbols_ar_ER', 'goog.labs.i18n.ListFormatSymbols_ar_IL', 'goog.labs.i18n.ListFormatSymbols_ar_IQ', 'goog.labs.i18n.ListFormatSymbols_ar_JO', 'goog.labs.i18n.ListFormatSymbols_ar_KM', 'goog.labs.i18n.ListFormatSymbols_ar_KW', 'goog.labs.i18n.ListFormatSymbols_ar_LB', 'goog.labs.i18n.ListFormatSymbols_ar_LY', 'goog.labs.i18n.ListFormatSymbols_ar_MA', 'goog.labs.i18n.ListFormatSymbols_ar_MR', 'goog.labs.i18n.ListFormatSymbols_ar_OM', 'goog.labs.i18n.ListFormatSymbols_ar_PS', 'goog.labs.i18n.ListFormatSymbols_ar_QA', 'goog.labs.i18n.ListFormatSymbols_ar_SA', 'goog.labs.i18n.ListFormatSymbols_ar_SD', 'goog.labs.i18n.ListFormatSymbols_ar_SO', 'goog.labs.i18n.ListFormatSymbols_ar_SS', 'goog.labs.i18n.ListFormatSymbols_ar_SY', 'goog.labs.i18n.ListFormatSymbols_ar_TD', 'goog.labs.i18n.ListFormatSymbols_ar_TN', 'goog.labs.i18n.ListFormatSymbols_ar_XB', 'goog.labs.i18n.ListFormatSymbols_ar_YE', 'goog.labs.i18n.ListFormatSymbols_as', 'goog.labs.i18n.ListFormatSymbols_as_IN', 'goog.labs.i18n.ListFormatSymbols_asa', 'goog.labs.i18n.ListFormatSymbols_asa_TZ', 'goog.labs.i18n.ListFormatSymbols_ast', 'goog.labs.i18n.ListFormatSymbols_ast_ES', 'goog.labs.i18n.ListFormatSymbols_az_Cyrl', 'goog.labs.i18n.ListFormatSymbols_az_Cyrl_AZ', 'goog.labs.i18n.ListFormatSymbols_az_Latn', 'goog.labs.i18n.ListFormatSymbols_az_Latn_AZ', 'goog.labs.i18n.ListFormatSymbols_bas', 'goog.labs.i18n.ListFormatSymbols_bas_CM', 'goog.labs.i18n.ListFormatSymbols_be_BY', 'goog.labs.i18n.ListFormatSymbols_bem', 'goog.labs.i18n.ListFormatSymbols_bem_ZM', 'goog.labs.i18n.ListFormatSymbols_bez', 'goog.labs.i18n.ListFormatSymbols_bez_TZ', 'goog.labs.i18n.ListFormatSymbols_bg_BG', 'goog.labs.i18n.ListFormatSymbols_bm', 'goog.labs.i18n.ListFormatSymbols_bm_ML', 'goog.labs.i18n.ListFormatSymbols_bn_BD', 'goog.labs.i18n.ListFormatSymbols_bn_IN', 'goog.labs.i18n.ListFormatSymbols_bo', 'goog.labs.i18n.ListFormatSymbols_bo_CN', 'goog.labs.i18n.ListFormatSymbols_bo_IN', 'goog.labs.i18n.ListFormatSymbols_br_FR', 'goog.labs.i18n.ListFormatSymbols_brx', 'goog.labs.i18n.ListFormatSymbols_brx_IN', 'goog.labs.i18n.ListFormatSymbols_bs_Cyrl', 'goog.labs.i18n.ListFormatSymbols_bs_Cyrl_BA', 'goog.labs.i18n.ListFormatSymbols_bs_Latn', 'goog.labs.i18n.ListFormatSymbols_bs_Latn_BA', 'goog.labs.i18n.ListFormatSymbols_ca_AD', 'goog.labs.i18n.ListFormatSymbols_ca_ES', 'goog.labs.i18n.ListFormatSymbols_ca_FR', 'goog.labs.i18n.ListFormatSymbols_ca_IT', 'goog.labs.i18n.ListFormatSymbols_ccp', 'goog.labs.i18n.ListFormatSymbols_ccp_BD', 'goog.labs.i18n.ListFormatSymbols_ccp_IN', 'goog.labs.i18n.ListFormatSymbols_ce', 'goog.labs.i18n.ListFormatSymbols_ce_RU', 'goog.labs.i18n.ListFormatSymbols_ceb', 'goog.labs.i18n.ListFormatSymbols_ceb_PH', 'goog.labs.i18n.ListFormatSymbols_cgg', 'goog.labs.i18n.ListFormatSymbols_cgg_UG', 'goog.labs.i18n.ListFormatSymbols_chr_US', 'goog.labs.i18n.ListFormatSymbols_ckb', 'goog.labs.i18n.ListFormatSymbols_ckb_IQ', 'goog.labs.i18n.ListFormatSymbols_ckb_IR', 'goog.labs.i18n.ListFormatSymbols_cs_CZ', 'goog.labs.i18n.ListFormatSymbols_cy_GB', 'goog.labs.i18n.ListFormatSymbols_da_DK', 'goog.labs.i18n.ListFormatSymbols_da_GL', 'goog.labs.i18n.ListFormatSymbols_dav', 'goog.labs.i18n.ListFormatSymbols_dav_KE', 'goog.labs.i18n.ListFormatSymbols_de_BE', 'goog.labs.i18n.ListFormatSymbols_de_DE', 'goog.labs.i18n.ListFormatSymbols_de_IT', 'goog.labs.i18n.ListFormatSymbols_de_LI', 'goog.labs.i18n.ListFormatSymbols_de_LU', 'goog.labs.i18n.ListFormatSymbols_dje', 'goog.labs.i18n.ListFormatSymbols_dje_NE', 'goog.labs.i18n.ListFormatSymbols_dsb', 'goog.labs.i18n.ListFormatSymbols_dsb_DE', 'goog.labs.i18n.ListFormatSymbols_dua', 'goog.labs.i18n.ListFormatSymbols_dua_CM', 'goog.labs.i18n.ListFormatSymbols_dyo', 'goog.labs.i18n.ListFormatSymbols_dyo_SN', 'goog.labs.i18n.ListFormatSymbols_dz', 'goog.labs.i18n.ListFormatSymbols_dz_BT', 'goog.labs.i18n.ListFormatSymbols_ebu', 'goog.labs.i18n.ListFormatSymbols_ebu_KE', 'goog.labs.i18n.ListFormatSymbols_ee', 'goog.labs.i18n.ListFormatSymbols_ee_GH', 'goog.labs.i18n.ListFormatSymbols_ee_TG', 'goog.labs.i18n.ListFormatSymbols_el_CY', 'goog.labs.i18n.ListFormatSymbols_el_GR', 'goog.labs.i18n.ListFormatSymbols_en_001', 'goog.labs.i18n.ListFormatSymbols_en_150', 'goog.labs.i18n.ListFormatSymbols_en_AE', 'goog.labs.i18n.ListFormatSymbols_en_AG', 'goog.labs.i18n.ListFormatSymbols_en_AI', 'goog.labs.i18n.ListFormatSymbols_en_AS', 'goog.labs.i18n.ListFormatSymbols_en_AT', 'goog.labs.i18n.ListFormatSymbols_en_BB', 'goog.labs.i18n.ListFormatSymbols_en_BE', 'goog.labs.i18n.ListFormatSymbols_en_BI', 'goog.labs.i18n.ListFormatSymbols_en_BM', 'goog.labs.i18n.ListFormatSymbols_en_BS', 'goog.labs.i18n.ListFormatSymbols_en_BW', 'goog.labs.i18n.ListFormatSymbols_en_BZ', 'goog.labs.i18n.ListFormatSymbols_en_CC', 'goog.labs.i18n.ListFormatSymbols_en_CH', 'goog.labs.i18n.ListFormatSymbols_en_CK', 'goog.labs.i18n.ListFormatSymbols_en_CM', 'goog.labs.i18n.ListFormatSymbols_en_CX', 'goog.labs.i18n.ListFormatSymbols_en_CY', 'goog.labs.i18n.ListFormatSymbols_en_DE', 'goog.labs.i18n.ListFormatSymbols_en_DG', 'goog.labs.i18n.ListFormatSymbols_en_DK', 'goog.labs.i18n.ListFormatSymbols_en_DM', 'goog.labs.i18n.ListFormatSymbols_en_ER', 'goog.labs.i18n.ListFormatSymbols_en_FI', 'goog.labs.i18n.ListFormatSymbols_en_FJ', 'goog.labs.i18n.ListFormatSymbols_en_FK', 'goog.labs.i18n.ListFormatSymbols_en_FM', 'goog.labs.i18n.ListFormatSymbols_en_GD', 'goog.labs.i18n.ListFormatSymbols_en_GG', 'goog.labs.i18n.ListFormatSymbols_en_GH', 'goog.labs.i18n.ListFormatSymbols_en_GI', 'goog.labs.i18n.ListFormatSymbols_en_GM', 'goog.labs.i18n.ListFormatSymbols_en_GU', 'goog.labs.i18n.ListFormatSymbols_en_GY', 'goog.labs.i18n.ListFormatSymbols_en_HK', 'goog.labs.i18n.ListFormatSymbols_en_IL', 'goog.labs.i18n.ListFormatSymbols_en_IM', 'goog.labs.i18n.ListFormatSymbols_en_IO', 'goog.labs.i18n.ListFormatSymbols_en_JE', 'goog.labs.i18n.ListFormatSymbols_en_JM', 'goog.labs.i18n.ListFormatSymbols_en_KE', 'goog.labs.i18n.ListFormatSymbols_en_KI', 'goog.labs.i18n.ListFormatSymbols_en_KN', 'goog.labs.i18n.ListFormatSymbols_en_KY', 'goog.labs.i18n.ListFormatSymbols_en_LC', 'goog.labs.i18n.ListFormatSymbols_en_LR', 'goog.labs.i18n.ListFormatSymbols_en_LS', 'goog.labs.i18n.ListFormatSymbols_en_MG', 'goog.labs.i18n.ListFormatSymbols_en_MH', 'goog.labs.i18n.ListFormatSymbols_en_MO', 'goog.labs.i18n.ListFormatSymbols_en_MP', 'goog.labs.i18n.ListFormatSymbols_en_MS', 'goog.labs.i18n.ListFormatSymbols_en_MT', 'goog.labs.i18n.ListFormatSymbols_en_MU', 'goog.labs.i18n.ListFormatSymbols_en_MW', 'goog.labs.i18n.ListFormatSymbols_en_MY', 'goog.labs.i18n.ListFormatSymbols_en_NA', 'goog.labs.i18n.ListFormatSymbols_en_NF', 'goog.labs.i18n.ListFormatSymbols_en_NG', 'goog.labs.i18n.ListFormatSymbols_en_NL', 'goog.labs.i18n.ListFormatSymbols_en_NR', 'goog.labs.i18n.ListFormatSymbols_en_NU', 'goog.labs.i18n.ListFormatSymbols_en_NZ', 'goog.labs.i18n.ListFormatSymbols_en_PG', 'goog.labs.i18n.ListFormatSymbols_en_PH', 'goog.labs.i18n.ListFormatSymbols_en_PK', 'goog.labs.i18n.ListFormatSymbols_en_PN', 'goog.labs.i18n.ListFormatSymbols_en_PR', 'goog.labs.i18n.ListFormatSymbols_en_PW', 'goog.labs.i18n.ListFormatSymbols_en_RW', 'goog.labs.i18n.ListFormatSymbols_en_SB', 'goog.labs.i18n.ListFormatSymbols_en_SC', 'goog.labs.i18n.ListFormatSymbols_en_SD', 'goog.labs.i18n.ListFormatSymbols_en_SE', 'goog.labs.i18n.ListFormatSymbols_en_SH', 'goog.labs.i18n.ListFormatSymbols_en_SI', 'goog.labs.i18n.ListFormatSymbols_en_SL', 'goog.labs.i18n.ListFormatSymbols_en_SS', 'goog.labs.i18n.ListFormatSymbols_en_SX', 'goog.labs.i18n.ListFormatSymbols_en_SZ', 'goog.labs.i18n.ListFormatSymbols_en_TC', 'goog.labs.i18n.ListFormatSymbols_en_TK', 'goog.labs.i18n.ListFormatSymbols_en_TO', 'goog.labs.i18n.ListFormatSymbols_en_TT', 'goog.labs.i18n.ListFormatSymbols_en_TV', 'goog.labs.i18n.ListFormatSymbols_en_TZ', 'goog.labs.i18n.ListFormatSymbols_en_UG', 'goog.labs.i18n.ListFormatSymbols_en_UM', 'goog.labs.i18n.ListFormatSymbols_en_US_POSIX', 'goog.labs.i18n.ListFormatSymbols_en_VC', 'goog.labs.i18n.ListFormatSymbols_en_VG', 'goog.labs.i18n.ListFormatSymbols_en_VI', 'goog.labs.i18n.ListFormatSymbols_en_VU', 'goog.labs.i18n.ListFormatSymbols_en_WS', 'goog.labs.i18n.ListFormatSymbols_en_XA', 'goog.labs.i18n.ListFormatSymbols_en_ZM', 'goog.labs.i18n.ListFormatSymbols_en_ZW', 'goog.labs.i18n.ListFormatSymbols_eo', 'goog.labs.i18n.ListFormatSymbols_eo_001', 'goog.labs.i18n.ListFormatSymbols_es_AR', 'goog.labs.i18n.ListFormatSymbols_es_BO', 'goog.labs.i18n.ListFormatSymbols_es_BR', 'goog.labs.i18n.ListFormatSymbols_es_BZ', 'goog.labs.i18n.ListFormatSymbols_es_CL', 'goog.labs.i18n.ListFormatSymbols_es_CO', 'goog.labs.i18n.ListFormatSymbols_es_CR', 'goog.labs.i18n.ListFormatSymbols_es_CU', 'goog.labs.i18n.ListFormatSymbols_es_DO', 'goog.labs.i18n.ListFormatSymbols_es_EA', 'goog.labs.i18n.ListFormatSymbols_es_EC', 'goog.labs.i18n.ListFormatSymbols_es_GQ', 'goog.labs.i18n.ListFormatSymbols_es_GT', 'goog.labs.i18n.ListFormatSymbols_es_HN', 'goog.labs.i18n.ListFormatSymbols_es_IC', 'goog.labs.i18n.ListFormatSymbols_es_NI', 'goog.labs.i18n.ListFormatSymbols_es_PA', 'goog.labs.i18n.ListFormatSymbols_es_PE', 'goog.labs.i18n.ListFormatSymbols_es_PH', 'goog.labs.i18n.ListFormatSymbols_es_PR', 'goog.labs.i18n.ListFormatSymbols_es_PY', 'goog.labs.i18n.ListFormatSymbols_es_SV', 'goog.labs.i18n.ListFormatSymbols_es_UY', 'goog.labs.i18n.ListFormatSymbols_es_VE', 'goog.labs.i18n.ListFormatSymbols_et_EE', 'goog.labs.i18n.ListFormatSymbols_eu_ES', 'goog.labs.i18n.ListFormatSymbols_ewo', 'goog.labs.i18n.ListFormatSymbols_ewo_CM', 'goog.labs.i18n.ListFormatSymbols_fa_AF', 'goog.labs.i18n.ListFormatSymbols_fa_IR', 'goog.labs.i18n.ListFormatSymbols_ff', 'goog.labs.i18n.ListFormatSymbols_ff_Latn', 'goog.labs.i18n.ListFormatSymbols_ff_Latn_BF', 'goog.labs.i18n.ListFormatSymbols_ff_Latn_CM', 'goog.labs.i18n.ListFormatSymbols_ff_Latn_GH', 'goog.labs.i18n.ListFormatSymbols_ff_Latn_GM', 'goog.labs.i18n.ListFormatSymbols_ff_Latn_GN', 'goog.labs.i18n.ListFormatSymbols_ff_Latn_GW', 'goog.labs.i18n.ListFormatSymbols_ff_Latn_LR', 'goog.labs.i18n.ListFormatSymbols_ff_Latn_MR', 'goog.labs.i18n.ListFormatSymbols_ff_Latn_NE', 'goog.labs.i18n.ListFormatSymbols_ff_Latn_NG', 'goog.labs.i18n.ListFormatSymbols_ff_Latn_SL', 'goog.labs.i18n.ListFormatSymbols_ff_Latn_SN', 'goog.labs.i18n.ListFormatSymbols_fi_FI', 'goog.labs.i18n.ListFormatSymbols_fil_PH', 'goog.labs.i18n.ListFormatSymbols_fo', 'goog.labs.i18n.ListFormatSymbols_fo_DK', 'goog.labs.i18n.ListFormatSymbols_fo_FO', 'goog.labs.i18n.ListFormatSymbols_fr_BE', 'goog.labs.i18n.ListFormatSymbols_fr_BF', 'goog.labs.i18n.ListFormatSymbols_fr_BI', 'goog.labs.i18n.ListFormatSymbols_fr_BJ', 'goog.labs.i18n.ListFormatSymbols_fr_BL', 'goog.labs.i18n.ListFormatSymbols_fr_CD', 'goog.labs.i18n.ListFormatSymbols_fr_CF', 'goog.labs.i18n.ListFormatSymbols_fr_CG', 'goog.labs.i18n.ListFormatSymbols_fr_CH', 'goog.labs.i18n.ListFormatSymbols_fr_CI', 'goog.labs.i18n.ListFormatSymbols_fr_CM', 'goog.labs.i18n.ListFormatSymbols_fr_DJ', 'goog.labs.i18n.ListFormatSymbols_fr_DZ', 'goog.labs.i18n.ListFormatSymbols_fr_FR', 'goog.labs.i18n.ListFormatSymbols_fr_GA', 'goog.labs.i18n.ListFormatSymbols_fr_GF', 'goog.labs.i18n.ListFormatSymbols_fr_GN', 'goog.labs.i18n.ListFormatSymbols_fr_GP', 'goog.labs.i18n.ListFormatSymbols_fr_GQ', 'goog.labs.i18n.ListFormatSymbols_fr_HT', 'goog.labs.i18n.ListFormatSymbols_fr_KM', 'goog.labs.i18n.ListFormatSymbols_fr_LU', 'goog.labs.i18n.ListFormatSymbols_fr_MA', 'goog.labs.i18n.ListFormatSymbols_fr_MC', 'goog.labs.i18n.ListFormatSymbols_fr_MF', 'goog.labs.i18n.ListFormatSymbols_fr_MG', 'goog.labs.i18n.ListFormatSymbols_fr_ML', 'goog.labs.i18n.ListFormatSymbols_fr_MQ', 'goog.labs.i18n.ListFormatSymbols_fr_MR', 'goog.labs.i18n.ListFormatSymbols_fr_MU', 'goog.labs.i18n.ListFormatSymbols_fr_NC', 'goog.labs.i18n.ListFormatSymbols_fr_NE', 'goog.labs.i18n.ListFormatSymbols_fr_PF', 'goog.labs.i18n.ListFormatSymbols_fr_PM', 'goog.labs.i18n.ListFormatSymbols_fr_RE', 'goog.labs.i18n.ListFormatSymbols_fr_RW', 'goog.labs.i18n.ListFormatSymbols_fr_SC', 'goog.labs.i18n.ListFormatSymbols_fr_SN', 'goog.labs.i18n.ListFormatSymbols_fr_SY', 'goog.labs.i18n.ListFormatSymbols_fr_TD', 'goog.labs.i18n.ListFormatSymbols_fr_TG', 'goog.labs.i18n.ListFormatSymbols_fr_TN', 'goog.labs.i18n.ListFormatSymbols_fr_VU', 'goog.labs.i18n.ListFormatSymbols_fr_WF', 'goog.labs.i18n.ListFormatSymbols_fr_YT', 'goog.labs.i18n.ListFormatSymbols_fur', 'goog.labs.i18n.ListFormatSymbols_fur_IT', 'goog.labs.i18n.ListFormatSymbols_fy', 'goog.labs.i18n.ListFormatSymbols_fy_NL', 'goog.labs.i18n.ListFormatSymbols_ga_IE', 'goog.labs.i18n.ListFormatSymbols_gd', 'goog.labs.i18n.ListFormatSymbols_gd_GB', 'goog.labs.i18n.ListFormatSymbols_gl_ES', 'goog.labs.i18n.ListFormatSymbols_gsw_CH', 'goog.labs.i18n.ListFormatSymbols_gsw_FR', 'goog.labs.i18n.ListFormatSymbols_gsw_LI', 'goog.labs.i18n.ListFormatSymbols_gu_IN', 'goog.labs.i18n.ListFormatSymbols_guz', 'goog.labs.i18n.ListFormatSymbols_guz_KE', 'goog.labs.i18n.ListFormatSymbols_gv', 'goog.labs.i18n.ListFormatSymbols_gv_IM', 'goog.labs.i18n.ListFormatSymbols_ha', 'goog.labs.i18n.ListFormatSymbols_ha_GH', 'goog.labs.i18n.ListFormatSymbols_ha_NE', 'goog.labs.i18n.ListFormatSymbols_ha_NG', 'goog.labs.i18n.ListFormatSymbols_haw_US', 'goog.labs.i18n.ListFormatSymbols_he_IL', 'goog.labs.i18n.ListFormatSymbols_hi_IN', 'goog.labs.i18n.ListFormatSymbols_hr_BA', 'goog.labs.i18n.ListFormatSymbols_hr_HR', 'goog.labs.i18n.ListFormatSymbols_hsb', 'goog.labs.i18n.ListFormatSymbols_hsb_DE', 'goog.labs.i18n.ListFormatSymbols_hu_HU', 'goog.labs.i18n.ListFormatSymbols_hy_AM', 'goog.labs.i18n.ListFormatSymbols_ia', 'goog.labs.i18n.ListFormatSymbols_ia_001', 'goog.labs.i18n.ListFormatSymbols_id_ID', 'goog.labs.i18n.ListFormatSymbols_ig', 'goog.labs.i18n.ListFormatSymbols_ig_NG', 'goog.labs.i18n.ListFormatSymbols_ii', 'goog.labs.i18n.ListFormatSymbols_ii_CN', 'goog.labs.i18n.ListFormatSymbols_is_IS', 'goog.labs.i18n.ListFormatSymbols_it_CH', 'goog.labs.i18n.ListFormatSymbols_it_IT', 'goog.labs.i18n.ListFormatSymbols_it_SM', 'goog.labs.i18n.ListFormatSymbols_it_VA', 'goog.labs.i18n.ListFormatSymbols_ja_JP', 'goog.labs.i18n.ListFormatSymbols_jgo', 'goog.labs.i18n.ListFormatSymbols_jgo_CM', 'goog.labs.i18n.ListFormatSymbols_jmc', 'goog.labs.i18n.ListFormatSymbols_jmc_TZ', 'goog.labs.i18n.ListFormatSymbols_jv', 'goog.labs.i18n.ListFormatSymbols_jv_ID', 'goog.labs.i18n.ListFormatSymbols_ka_GE', 'goog.labs.i18n.ListFormatSymbols_kab', 'goog.labs.i18n.ListFormatSymbols_kab_DZ', 'goog.labs.i18n.ListFormatSymbols_kam', 'goog.labs.i18n.ListFormatSymbols_kam_KE', 'goog.labs.i18n.ListFormatSymbols_kde', 'goog.labs.i18n.ListFormatSymbols_kde_TZ', 'goog.labs.i18n.ListFormatSymbols_kea', 'goog.labs.i18n.ListFormatSymbols_kea_CV', 'goog.labs.i18n.ListFormatSymbols_khq', 'goog.labs.i18n.ListFormatSymbols_khq_ML', 'goog.labs.i18n.ListFormatSymbols_ki', 'goog.labs.i18n.ListFormatSymbols_ki_KE', 'goog.labs.i18n.ListFormatSymbols_kk_KZ', 'goog.labs.i18n.ListFormatSymbols_kkj', 'goog.labs.i18n.ListFormatSymbols_kkj_CM', 'goog.labs.i18n.ListFormatSymbols_kl', 'goog.labs.i18n.ListFormatSymbols_kl_GL', 'goog.labs.i18n.ListFormatSymbols_kln', 'goog.labs.i18n.ListFormatSymbols_kln_KE', 'goog.labs.i18n.ListFormatSymbols_km_KH', 'goog.labs.i18n.ListFormatSymbols_kn_IN', 'goog.labs.i18n.ListFormatSymbols_ko_KP', 'goog.labs.i18n.ListFormatSymbols_ko_KR', 'goog.labs.i18n.ListFormatSymbols_kok', 'goog.labs.i18n.ListFormatSymbols_kok_IN', 'goog.labs.i18n.ListFormatSymbols_ks', 'goog.labs.i18n.ListFormatSymbols_ks_IN', 'goog.labs.i18n.ListFormatSymbols_ksb', 'goog.labs.i18n.ListFormatSymbols_ksb_TZ', 'goog.labs.i18n.ListFormatSymbols_ksf', 'goog.labs.i18n.ListFormatSymbols_ksf_CM', 'goog.labs.i18n.ListFormatSymbols_ksh', 'goog.labs.i18n.ListFormatSymbols_ksh_DE', 'goog.labs.i18n.ListFormatSymbols_ku', 'goog.labs.i18n.ListFormatSymbols_ku_TR', 'goog.labs.i18n.ListFormatSymbols_kw', 'goog.labs.i18n.ListFormatSymbols_kw_GB', 'goog.labs.i18n.ListFormatSymbols_ky_KG', 'goog.labs.i18n.ListFormatSymbols_lag', 'goog.labs.i18n.ListFormatSymbols_lag_TZ', 'goog.labs.i18n.ListFormatSymbols_lb', 'goog.labs.i18n.ListFormatSymbols_lb_LU', 'goog.labs.i18n.ListFormatSymbols_lg', 'goog.labs.i18n.ListFormatSymbols_lg_UG', 'goog.labs.i18n.ListFormatSymbols_lkt', 'goog.labs.i18n.ListFormatSymbols_lkt_US', 'goog.labs.i18n.ListFormatSymbols_ln_AO', 'goog.labs.i18n.ListFormatSymbols_ln_CD', 'goog.labs.i18n.ListFormatSymbols_ln_CF', 'goog.labs.i18n.ListFormatSymbols_ln_CG', 'goog.labs.i18n.ListFormatSymbols_lo_LA', 'goog.labs.i18n.ListFormatSymbols_lrc', 'goog.labs.i18n.ListFormatSymbols_lrc_IQ', 'goog.labs.i18n.ListFormatSymbols_lrc_IR', 'goog.labs.i18n.ListFormatSymbols_lt_LT', 'goog.labs.i18n.ListFormatSymbols_lu', 'goog.labs.i18n.ListFormatSymbols_lu_CD', 'goog.labs.i18n.ListFormatSymbols_luo', 'goog.labs.i18n.ListFormatSymbols_luo_KE', 'goog.labs.i18n.ListFormatSymbols_luy', 'goog.labs.i18n.ListFormatSymbols_luy_KE', 'goog.labs.i18n.ListFormatSymbols_lv_LV', 'goog.labs.i18n.ListFormatSymbols_mas', 'goog.labs.i18n.ListFormatSymbols_mas_KE', 'goog.labs.i18n.ListFormatSymbols_mas_TZ', 'goog.labs.i18n.ListFormatSymbols_mer', 'goog.labs.i18n.ListFormatSymbols_mer_KE', 'goog.labs.i18n.ListFormatSymbols_mfe', 'goog.labs.i18n.ListFormatSymbols_mfe_MU', 'goog.labs.i18n.ListFormatSymbols_mg', 'goog.labs.i18n.ListFormatSymbols_mg_MG', 'goog.labs.i18n.ListFormatSymbols_mgh', 'goog.labs.i18n.ListFormatSymbols_mgh_MZ', 'goog.labs.i18n.ListFormatSymbols_mgo', 'goog.labs.i18n.ListFormatSymbols_mgo_CM', 'goog.labs.i18n.ListFormatSymbols_mi', 'goog.labs.i18n.ListFormatSymbols_mi_NZ', 'goog.labs.i18n.ListFormatSymbols_mk_MK', 'goog.labs.i18n.ListFormatSymbols_ml_IN', 'goog.labs.i18n.ListFormatSymbols_mn_MN', 'goog.labs.i18n.ListFormatSymbols_mr_IN', 'goog.labs.i18n.ListFormatSymbols_ms_BN', 'goog.labs.i18n.ListFormatSymbols_ms_MY', 'goog.labs.i18n.ListFormatSymbols_ms_SG', 'goog.labs.i18n.ListFormatSymbols_mt_MT', 'goog.labs.i18n.ListFormatSymbols_mua', 'goog.labs.i18n.ListFormatSymbols_mua_CM', 'goog.labs.i18n.ListFormatSymbols_my_MM', 'goog.labs.i18n.ListFormatSymbols_mzn', 'goog.labs.i18n.ListFormatSymbols_mzn_IR', 'goog.labs.i18n.ListFormatSymbols_naq', 'goog.labs.i18n.ListFormatSymbols_naq_NA', 'goog.labs.i18n.ListFormatSymbols_nb_NO', 'goog.labs.i18n.ListFormatSymbols_nb_SJ', 'goog.labs.i18n.ListFormatSymbols_nd', 'goog.labs.i18n.ListFormatSymbols_nd_ZW', 'goog.labs.i18n.ListFormatSymbols_nds', 'goog.labs.i18n.ListFormatSymbols_nds_DE', 'goog.labs.i18n.ListFormatSymbols_nds_NL', 'goog.labs.i18n.ListFormatSymbols_ne_IN', 'goog.labs.i18n.ListFormatSymbols_ne_NP', 'goog.labs.i18n.ListFormatSymbols_nl_AW', 'goog.labs.i18n.ListFormatSymbols_nl_BE', 'goog.labs.i18n.ListFormatSymbols_nl_BQ', 'goog.labs.i18n.ListFormatSymbols_nl_CW', 'goog.labs.i18n.ListFormatSymbols_nl_NL', 'goog.labs.i18n.ListFormatSymbols_nl_SR', 'goog.labs.i18n.ListFormatSymbols_nl_SX', 'goog.labs.i18n.ListFormatSymbols_nmg', 'goog.labs.i18n.ListFormatSymbols_nmg_CM', 'goog.labs.i18n.ListFormatSymbols_nn', 'goog.labs.i18n.ListFormatSymbols_nn_NO', 'goog.labs.i18n.ListFormatSymbols_nnh', 'goog.labs.i18n.ListFormatSymbols_nnh_CM', 'goog.labs.i18n.ListFormatSymbols_nus', 'goog.labs.i18n.ListFormatSymbols_nus_SS', 'goog.labs.i18n.ListFormatSymbols_nyn', 'goog.labs.i18n.ListFormatSymbols_nyn_UG', 'goog.labs.i18n.ListFormatSymbols_om', 'goog.labs.i18n.ListFormatSymbols_om_ET', 'goog.labs.i18n.ListFormatSymbols_om_KE', 'goog.labs.i18n.ListFormatSymbols_or_IN', 'goog.labs.i18n.ListFormatSymbols_os', 'goog.labs.i18n.ListFormatSymbols_os_GE', 'goog.labs.i18n.ListFormatSymbols_os_RU', 'goog.labs.i18n.ListFormatSymbols_pa_Arab', 'goog.labs.i18n.ListFormatSymbols_pa_Arab_PK', 'goog.labs.i18n.ListFormatSymbols_pa_Guru', 'goog.labs.i18n.ListFormatSymbols_pa_Guru_IN', 'goog.labs.i18n.ListFormatSymbols_pl_PL', 'goog.labs.i18n.ListFormatSymbols_ps', 'goog.labs.i18n.ListFormatSymbols_ps_AF', 'goog.labs.i18n.ListFormatSymbols_ps_PK', 'goog.labs.i18n.ListFormatSymbols_pt_AO', 'goog.labs.i18n.ListFormatSymbols_pt_CH', 'goog.labs.i18n.ListFormatSymbols_pt_CV', 'goog.labs.i18n.ListFormatSymbols_pt_GQ', 'goog.labs.i18n.ListFormatSymbols_pt_GW', 'goog.labs.i18n.ListFormatSymbols_pt_LU', 'goog.labs.i18n.ListFormatSymbols_pt_MO', 'goog.labs.i18n.ListFormatSymbols_pt_MZ', 'goog.labs.i18n.ListFormatSymbols_pt_ST', 'goog.labs.i18n.ListFormatSymbols_pt_TL', 'goog.labs.i18n.ListFormatSymbols_qu', 'goog.labs.i18n.ListFormatSymbols_qu_BO', 'goog.labs.i18n.ListFormatSymbols_qu_EC', 'goog.labs.i18n.ListFormatSymbols_qu_PE', 'goog.labs.i18n.ListFormatSymbols_rm', 'goog.labs.i18n.ListFormatSymbols_rm_CH', 'goog.labs.i18n.ListFormatSymbols_rn', 'goog.labs.i18n.ListFormatSymbols_rn_BI', 'goog.labs.i18n.ListFormatSymbols_ro_MD', 'goog.labs.i18n.ListFormatSymbols_ro_RO', 'goog.labs.i18n.ListFormatSymbols_rof', 'goog.labs.i18n.ListFormatSymbols_rof_TZ', 'goog.labs.i18n.ListFormatSymbols_ru_BY', 'goog.labs.i18n.ListFormatSymbols_ru_KG', 'goog.labs.i18n.ListFormatSymbols_ru_KZ', 'goog.labs.i18n.ListFormatSymbols_ru_MD', 'goog.labs.i18n.ListFormatSymbols_ru_RU', 'goog.labs.i18n.ListFormatSymbols_ru_UA', 'goog.labs.i18n.ListFormatSymbols_rw', 'goog.labs.i18n.ListFormatSymbols_rw_RW', 'goog.labs.i18n.ListFormatSymbols_rwk', 'goog.labs.i18n.ListFormatSymbols_rwk_TZ', 'goog.labs.i18n.ListFormatSymbols_sah', 'goog.labs.i18n.ListFormatSymbols_sah_RU', 'goog.labs.i18n.ListFormatSymbols_saq', 'goog.labs.i18n.ListFormatSymbols_saq_KE', 'goog.labs.i18n.ListFormatSymbols_sbp', 'goog.labs.i18n.ListFormatSymbols_sbp_TZ', 'goog.labs.i18n.ListFormatSymbols_sd', 'goog.labs.i18n.ListFormatSymbols_sd_PK', 'goog.labs.i18n.ListFormatSymbols_se', 'goog.labs.i18n.ListFormatSymbols_se_FI', 'goog.labs.i18n.ListFormatSymbols_se_NO', 'goog.labs.i18n.ListFormatSymbols_se_SE', 'goog.labs.i18n.ListFormatSymbols_seh', 'goog.labs.i18n.ListFormatSymbols_seh_MZ', 'goog.labs.i18n.ListFormatSymbols_ses', 'goog.labs.i18n.ListFormatSymbols_ses_ML', 'goog.labs.i18n.ListFormatSymbols_sg', 'goog.labs.i18n.ListFormatSymbols_sg_CF', 'goog.labs.i18n.ListFormatSymbols_shi', 'goog.labs.i18n.ListFormatSymbols_shi_Latn', 'goog.labs.i18n.ListFormatSymbols_shi_Latn_MA', 'goog.labs.i18n.ListFormatSymbols_shi_Tfng', 'goog.labs.i18n.ListFormatSymbols_shi_Tfng_MA', 'goog.labs.i18n.ListFormatSymbols_si_LK', 'goog.labs.i18n.ListFormatSymbols_sk_SK', 'goog.labs.i18n.ListFormatSymbols_sl_SI', 'goog.labs.i18n.ListFormatSymbols_smn', 'goog.labs.i18n.ListFormatSymbols_smn_FI', 'goog.labs.i18n.ListFormatSymbols_sn', 'goog.labs.i18n.ListFormatSymbols_sn_ZW', 'goog.labs.i18n.ListFormatSymbols_so', 'goog.labs.i18n.ListFormatSymbols_so_DJ', 'goog.labs.i18n.ListFormatSymbols_so_ET', 'goog.labs.i18n.ListFormatSymbols_so_KE', 'goog.labs.i18n.ListFormatSymbols_so_SO', 'goog.labs.i18n.ListFormatSymbols_sq_AL', 'goog.labs.i18n.ListFormatSymbols_sq_MK', 'goog.labs.i18n.ListFormatSymbols_sq_XK', 'goog.labs.i18n.ListFormatSymbols_sr_Cyrl', 'goog.labs.i18n.ListFormatSymbols_sr_Cyrl_BA', 'goog.labs.i18n.ListFormatSymbols_sr_Cyrl_ME', 'goog.labs.i18n.ListFormatSymbols_sr_Cyrl_RS', 'goog.labs.i18n.ListFormatSymbols_sr_Cyrl_XK', 'goog.labs.i18n.ListFormatSymbols_sr_Latn_BA', 'goog.labs.i18n.ListFormatSymbols_sr_Latn_ME', 'goog.labs.i18n.ListFormatSymbols_sr_Latn_RS', 'goog.labs.i18n.ListFormatSymbols_sr_Latn_XK', 'goog.labs.i18n.ListFormatSymbols_sv_AX', 'goog.labs.i18n.ListFormatSymbols_sv_FI', 'goog.labs.i18n.ListFormatSymbols_sv_SE', 'goog.labs.i18n.ListFormatSymbols_sw_CD', 'goog.labs.i18n.ListFormatSymbols_sw_KE', 'goog.labs.i18n.ListFormatSymbols_sw_TZ', 'goog.labs.i18n.ListFormatSymbols_sw_UG', 'goog.labs.i18n.ListFormatSymbols_ta_IN', 'goog.labs.i18n.ListFormatSymbols_ta_LK', 'goog.labs.i18n.ListFormatSymbols_ta_MY', 'goog.labs.i18n.ListFormatSymbols_ta_SG', 'goog.labs.i18n.ListFormatSymbols_te_IN', 'goog.labs.i18n.ListFormatSymbols_teo', 'goog.labs.i18n.ListFormatSymbols_teo_KE', 'goog.labs.i18n.ListFormatSymbols_teo_UG', 'goog.labs.i18n.ListFormatSymbols_tg', 'goog.labs.i18n.ListFormatSymbols_tg_TJ', 'goog.labs.i18n.ListFormatSymbols_th_TH', 'goog.labs.i18n.ListFormatSymbols_ti', 'goog.labs.i18n.ListFormatSymbols_ti_ER', 'goog.labs.i18n.ListFormatSymbols_ti_ET', 'goog.labs.i18n.ListFormatSymbols_tk', 'goog.labs.i18n.ListFormatSymbols_tk_TM', 'goog.labs.i18n.ListFormatSymbols_to', 'goog.labs.i18n.ListFormatSymbols_to_TO', 'goog.labs.i18n.ListFormatSymbols_tr_CY', 'goog.labs.i18n.ListFormatSymbols_tr_TR', 'goog.labs.i18n.ListFormatSymbols_tt', 'goog.labs.i18n.ListFormatSymbols_tt_RU', 'goog.labs.i18n.ListFormatSymbols_twq', 'goog.labs.i18n.ListFormatSymbols_twq_NE', 'goog.labs.i18n.ListFormatSymbols_tzm', 'goog.labs.i18n.ListFormatSymbols_tzm_MA', 'goog.labs.i18n.ListFormatSymbols_ug', 'goog.labs.i18n.ListFormatSymbols_ug_CN', 'goog.labs.i18n.ListFormatSymbols_uk_UA', 'goog.labs.i18n.ListFormatSymbols_ur_IN', 'goog.labs.i18n.ListFormatSymbols_ur_PK', 'goog.labs.i18n.ListFormatSymbols_uz_Arab', 'goog.labs.i18n.ListFormatSymbols_uz_Arab_AF', 'goog.labs.i18n.ListFormatSymbols_uz_Cyrl', 'goog.labs.i18n.ListFormatSymbols_uz_Cyrl_UZ', 'goog.labs.i18n.ListFormatSymbols_uz_Latn', 'goog.labs.i18n.ListFormatSymbols_uz_Latn_UZ', 'goog.labs.i18n.ListFormatSymbols_vai', 'goog.labs.i18n.ListFormatSymbols_vai_Latn', 'goog.labs.i18n.ListFormatSymbols_vai_Latn_LR', 'goog.labs.i18n.ListFormatSymbols_vai_Vaii', 'goog.labs.i18n.ListFormatSymbols_vai_Vaii_LR', 'goog.labs.i18n.ListFormatSymbols_vi_VN', 'goog.labs.i18n.ListFormatSymbols_vun', 'goog.labs.i18n.ListFormatSymbols_vun_TZ', 'goog.labs.i18n.ListFormatSymbols_wae', 'goog.labs.i18n.ListFormatSymbols_wae_CH', 'goog.labs.i18n.ListFormatSymbols_wo', 'goog.labs.i18n.ListFormatSymbols_wo_SN', 'goog.labs.i18n.ListFormatSymbols_xh', 'goog.labs.i18n.ListFormatSymbols_xh_ZA', 'goog.labs.i18n.ListFormatSymbols_xog', 'goog.labs.i18n.ListFormatSymbols_xog_UG', 'goog.labs.i18n.ListFormatSymbols_yav', 'goog.labs.i18n.ListFormatSymbols_yav_CM', 'goog.labs.i18n.ListFormatSymbols_yi', 'goog.labs.i18n.ListFormatSymbols_yi_001', 'goog.labs.i18n.ListFormatSymbols_yo', 'goog.labs.i18n.ListFormatSymbols_yo_BJ', 'goog.labs.i18n.ListFormatSymbols_yo_NG', 'goog.labs.i18n.ListFormatSymbols_yue', 'goog.labs.i18n.ListFormatSymbols_yue_Hans', 'goog.labs.i18n.ListFormatSymbols_yue_Hans_CN', 'goog.labs.i18n.ListFormatSymbols_yue_Hant', 'goog.labs.i18n.ListFormatSymbols_yue_Hant_HK', 'goog.labs.i18n.ListFormatSymbols_zgh', 'goog.labs.i18n.ListFormatSymbols_zgh_MA', 'goog.labs.i18n.ListFormatSymbols_zh_Hans', 'goog.labs.i18n.ListFormatSymbols_zh_Hans_CN', 'goog.labs.i18n.ListFormatSymbols_zh_Hans_HK', 'goog.labs.i18n.ListFormatSymbols_zh_Hans_MO', 'goog.labs.i18n.ListFormatSymbols_zh_Hans_SG', 'goog.labs.i18n.ListFormatSymbols_zh_Hant', 'goog.labs.i18n.ListFormatSymbols_zh_Hant_HK', 'goog.labs.i18n.ListFormatSymbols_zh_Hant_MO', 'goog.labs.i18n.ListFormatSymbols_zh_Hant_TW', 'goog.labs.i18n.ListFormatSymbols_zu_ZA'], ['goog.labs.i18n.ListFormatSymbols'], {});
goog.addDependency('labs/mock/mock.js', ['goog.labs.mock', 'goog.labs.mock.TimeoutError', 'goog.labs.mock.VerificationError'], ['goog.array', 'goog.asserts', 'goog.debug', 'goog.debug.Error', 'goog.functions', 'goog.labs.mock.timeout', 'goog.labs.mock.timeout.TimeoutMode', 'goog.labs.mock.verification', 'goog.labs.mock.verification.BaseVerificationMode', 'goog.labs.mock.verification.VerificationMode', 'goog.object'], {'lang': 'es6'});
goog.addDependency('labs/mock/mock_test.js', ['goog.labs.mockTest'], ['goog.array', 'goog.labs.mock', 'goog.labs.mock.TimeoutError', 'goog.labs.mock.VerificationError', 'goog.labs.mock.timeout', 'goog.labs.mock.verification', 'goog.labs.testing.AnythingMatcher', 'goog.labs.testing.GreaterThanMatcher', 'goog.string', 'goog.testing.jsunit'], {'lang': 'es8'});
goog.addDependency('labs/mock/timeoutmode.js', ['goog.labs.mock.timeout', 'goog.labs.mock.timeout.TimeoutMode'], [], {'lang': 'es6'});
goog.addDependency('labs/mock/verificationmode.js', ['goog.labs.mock.verification', 'goog.labs.mock.verification.BaseVerificationMode', 'goog.labs.mock.verification.VerificationMode'], [], {'lang': 'es6'});
goog.addDependency('labs/mock/verificationmode_test.js', ['goog.labs.mock.VerificationModeTest'], ['goog.labs.mock.verification', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/net/image.js', ['goog.labs.net.image'], ['goog.Promise', 'goog.dom.safe', 'goog.events.EventHandler', 'goog.events.EventType', 'goog.html.SafeUrl', 'goog.net.EventType', 'goog.userAgent'], {});
goog.addDependency('labs/net/image_test.js', ['goog.labs.net.imageTest'], ['goog.labs.net.image', 'goog.string', 'goog.testing.TestCase', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/net/webchannel.js', ['goog.net.WebChannel'], ['goog.events', 'goog.events.Event', 'goog.events.Listenable', 'goog.net.XmlHttpFactory'], {});
goog.addDependency('labs/net/webchannel/basetestchannel.js', ['goog.labs.net.webChannel.BaseTestChannel'], ['goog.labs.net.webChannel.Channel', 'goog.labs.net.webChannel.ChannelRequest', 'goog.labs.net.webChannel.WebChannelDebug', 'goog.labs.net.webChannel.requestStats', 'goog.net.WebChannel'], {});
goog.addDependency('labs/net/webchannel/channel.js', ['goog.labs.net.webChannel.Channel'], [], {'lang': 'es6'});
goog.addDependency('labs/net/webchannel/channelrequest.js', ['goog.labs.net.webChannel.ChannelRequest'], ['goog.Timer', 'goog.async.Throttle', 'goog.events.EventHandler', 'goog.labs.net.webChannel.Channel', 'goog.labs.net.webChannel.WebChannelDebug', 'goog.labs.net.webChannel.environment', 'goog.labs.net.webChannel.requestStats', 'goog.net.ErrorCode', 'goog.net.EventType', 'goog.net.WebChannel', 'goog.net.XmlHttp', 'goog.object', 'goog.string', 'goog.userAgent'], {});
goog.addDependency('labs/net/webchannel/channelrequest_test.js', ['goog.labs.net.webChannel.channelRequestTest'], ['goog.Uri', 'goog.functions', 'goog.labs.net.webChannel.ChannelRequest', 'goog.labs.net.webChannel.WebChannelDebug', 'goog.labs.net.webChannel.requestStats', 'goog.labs.net.webChannel.requestStats.ServerReachability', 'goog.testing.MockClock', 'goog.testing.PropertyReplacer', 'goog.testing.net.XhrIo', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/net/webchannel/connectionstate.js', ['goog.labs.net.webChannel.ConnectionState'], [], {});
goog.addDependency('labs/net/webchannel/environment.js', ['goog.labs.net.webChannel.environment'], ['goog.userAgent'], {'module': 'goog'});
goog.addDependency('labs/net/webchannel/environment_test.js', ['goog.labs.net.webChannel.EnvironmentTest'], ['goog.labs.net.webChannel.environment', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/net/webchannel/forwardchannelrequestpool.js', ['goog.labs.net.webChannel.ForwardChannelRequestPool'], ['goog.array', 'goog.labs.net.webChannel.ChannelRequest', 'goog.labs.net.webChannel.Wire', 'goog.string', 'goog.structs.Set'], {'module': 'goog'});
goog.addDependency('labs/net/webchannel/forwardchannelrequestpool_test.js', ['goog.labs.net.webChannel.ForwardChannelRequestPoolTest'], ['goog.labs.net.webChannel.ChannelRequest', 'goog.labs.net.webChannel.ForwardChannelRequestPool', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/net/webchannel/netutils.js', ['goog.labs.net.webChannel.netUtils'], ['goog.Uri', 'goog.labs.net.webChannel.WebChannelDebug'], {});
goog.addDependency('labs/net/webchannel/requeststats.js', ['goog.labs.net.webChannel.requestStats', 'goog.labs.net.webChannel.requestStats.Event', 'goog.labs.net.webChannel.requestStats.ServerReachability', 'goog.labs.net.webChannel.requestStats.ServerReachabilityEvent', 'goog.labs.net.webChannel.requestStats.Stat', 'goog.labs.net.webChannel.requestStats.StatEvent', 'goog.labs.net.webChannel.requestStats.TimingEvent'], ['goog.events.Event', 'goog.events.EventTarget'], {});
goog.addDependency('labs/net/webchannel/webchannelbase.js', ['goog.labs.net.webChannel.WebChannelBase'], ['goog.Uri', 'goog.array', 'goog.asserts', 'goog.async.run', 'goog.json', 'goog.labs.net.webChannel.BaseTestChannel', 'goog.labs.net.webChannel.Channel', 'goog.labs.net.webChannel.ChannelRequest', 'goog.labs.net.webChannel.ConnectionState', 'goog.labs.net.webChannel.ForwardChannelRequestPool', 'goog.labs.net.webChannel.WebChannelDebug', 'goog.labs.net.webChannel.Wire', 'goog.labs.net.webChannel.WireV8', 'goog.labs.net.webChannel.netUtils', 'goog.labs.net.webChannel.requestStats', 'goog.net.WebChannel', 'goog.net.XhrIo', 'goog.net.XmlHttpFactory', 'goog.net.rpc.HttpCors', 'goog.object', 'goog.string', 'goog.structs'], {});
goog.addDependency('labs/net/webchannel/webchannelbase_test.js', ['goog.labs.net.webChannel.webChannelBaseTest'], ['goog.Timer', 'goog.array', 'goog.dom', 'goog.functions', 'goog.json', 'goog.labs.net.webChannel.ChannelRequest', 'goog.labs.net.webChannel.ForwardChannelRequestPool', 'goog.labs.net.webChannel.WebChannelBase', 'goog.labs.net.webChannel.WebChannelBaseTransport', 'goog.labs.net.webChannel.WebChannelDebug', 'goog.labs.net.webChannel.Wire', 'goog.labs.net.webChannel.netUtils', 'goog.labs.net.webChannel.requestStats', 'goog.labs.net.webChannel.requestStats.Stat', 'goog.structs.Map', 'goog.testing.MockClock', 'goog.testing.PropertyReplacer', 'goog.testing.asserts', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/net/webchannel/webchannelbasetransport.js', ['goog.labs.net.webChannel.WebChannelBaseTransport'], ['goog.asserts', 'goog.events.EventTarget', 'goog.json', 'goog.labs.net.webChannel.ChannelRequest', 'goog.labs.net.webChannel.WebChannelBase', 'goog.labs.net.webChannel.Wire', 'goog.log', 'goog.net.WebChannel', 'goog.net.WebChannelTransport', 'goog.object', 'goog.string', 'goog.string.path'], {});
goog.addDependency('labs/net/webchannel/webchannelbasetransport_test.js', ['goog.labs.net.webChannel.webChannelBaseTransportTest'], ['goog.events', 'goog.functions', 'goog.json', 'goog.labs.net.webChannel.ChannelRequest', 'goog.labs.net.webChannel.WebChannelBase', 'goog.labs.net.webChannel.WebChannelBaseTransport', 'goog.labs.net.webChannel.Wire', 'goog.net.WebChannel', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/net/webchannel/webchanneldebug.js', ['goog.labs.net.webChannel.WebChannelDebug'], ['goog.json', 'goog.log'], {});
goog.addDependency('labs/net/webchannel/wire.js', ['goog.labs.net.webChannel.Wire'], [], {'lang': 'es6'});
goog.addDependency('labs/net/webchannel/wirev8.js', ['goog.labs.net.webChannel.WireV8'], ['goog.asserts', 'goog.json', 'goog.json.NativeJsonProcessor', 'goog.labs.net.webChannel.Wire', 'goog.structs'], {});
goog.addDependency('labs/net/webchannel/wirev8_test.js', ['goog.labs.net.webChannel.WireV8Test'], ['goog.labs.net.webChannel.WireV8', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/net/webchanneltransport.js', ['goog.net.WebChannelTransport'], [], {});
goog.addDependency('labs/net/webchanneltransportfactory.js', ['goog.net.createWebChannelTransport'], ['goog.functions', 'goog.labs.net.webChannel.WebChannelBaseTransport'], {});
goog.addDependency('labs/net/xhr.js', ['goog.labs.net.xhr', 'goog.labs.net.xhr.Error', 'goog.labs.net.xhr.HttpError', 'goog.labs.net.xhr.Options', 'goog.labs.net.xhr.PostData', 'goog.labs.net.xhr.ResponseType', 'goog.labs.net.xhr.TimeoutError'], ['goog.Promise', 'goog.asserts', 'goog.debug.Error', 'goog.net.HttpStatus', 'goog.net.XmlHttp', 'goog.object', 'goog.string', 'goog.uri.utils', 'goog.userAgent'], {});
goog.addDependency('labs/net/xhr_test.js', ['goog.labs.net.xhrTest'], ['goog.Promise', 'goog.events', 'goog.events.EventType', 'goog.labs.net.xhr', 'goog.net.WrapperXmlHttpFactory', 'goog.net.XhrLike', 'goog.net.XmlHttp', 'goog.testing.MockClock', 'goog.testing.TestCase', 'goog.testing.jsunit', 'goog.userAgent'], {'lang': 'es6'});
goog.addDependency('labs/pubsub/broadcastpubsub.js', ['goog.labs.pubsub.BroadcastPubSub'], ['goog.Disposable', 'goog.Timer', 'goog.array', 'goog.async.run', 'goog.events.EventHandler', 'goog.events.EventType', 'goog.log', 'goog.math', 'goog.pubsub.PubSub', 'goog.storage.Storage', 'goog.storage.mechanism.HTML5LocalStorage', 'goog.string', 'goog.userAgent'], {});
goog.addDependency('labs/pubsub/broadcastpubsub_test.js', ['goog.labs.pubsub.BroadcastPubSubTest'], ['goog.array', 'goog.debug.Logger', 'goog.json', 'goog.labs.pubsub.BroadcastPubSub', 'goog.storage.Storage', 'goog.structs.Map', 'goog.testing.MockClock', 'goog.testing.MockControl', 'goog.testing.events', 'goog.testing.events.Event', 'goog.testing.mockmatchers', 'goog.testing.mockmatchers.ArgumentMatcher', 'goog.testing.recordFunction', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/storage/boundedcollectablestorage.js', ['goog.labs.storage.BoundedCollectableStorage'], ['goog.array', 'goog.asserts', 'goog.iter', 'goog.storage.CollectableStorage', 'goog.storage.ErrorCode', 'goog.storage.ExpiringStorage'], {});
goog.addDependency('labs/storage/boundedcollectablestorage_test.js', ['goog.labs.storage.BoundedCollectableStorageTest'], ['goog.labs.storage.BoundedCollectableStorage', 'goog.storage.collectableStorageTester', 'goog.storage.storageTester', 'goog.testing.MockClock', 'goog.testing.storage.FakeMechanism', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/structs/multimap.js', ['goog.labs.structs.Multimap'], ['goog.array', 'goog.object'], {'lang': 'es6'});
goog.addDependency('labs/structs/multimap_test.js', ['goog.labs.structs.MultimapTest'], ['goog.labs.structs.Multimap', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/style/pixeldensitymonitor.js', ['goog.labs.style.PixelDensityMonitor', 'goog.labs.style.PixelDensityMonitor.Density', 'goog.labs.style.PixelDensityMonitor.EventType'], ['goog.events', 'goog.events.EventTarget'], {});
goog.addDependency('labs/style/pixeldensitymonitor_test.js', ['goog.labs.style.PixelDensityMonitorTest'], ['goog.array', 'goog.dom.DomHelper', 'goog.events', 'goog.labs.style.PixelDensityMonitor', 'goog.testing.MockControl', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/testing/assertthat.js', ['goog.labs.testing.MatcherError', 'goog.labs.testing.assertThat'], ['goog.debug.Error'], {});
goog.addDependency('labs/testing/assertthat_test.js', ['goog.labs.testing.assertThatTest'], ['goog.labs.testing.MatcherError', 'goog.labs.testing.assertThat', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/testing/decoratormatcher.js', ['goog.labs.testing.AnythingMatcher'], ['goog.labs.testing.Matcher'], {});
goog.addDependency('labs/testing/decoratormatcher_test.js', ['goog.labs.testing.decoratorMatcherTest'], ['goog.labs.testing.AnythingMatcher', 'goog.labs.testing.GreaterThanMatcher', 'goog.labs.testing.MatcherError', 'goog.labs.testing.assertThat', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/testing/dictionarymatcher.js', ['goog.labs.testing.HasEntriesMatcher', 'goog.labs.testing.HasEntryMatcher', 'goog.labs.testing.HasKeyMatcher', 'goog.labs.testing.HasValueMatcher'], ['goog.asserts', 'goog.labs.testing.Matcher', 'goog.object'], {});
goog.addDependency('labs/testing/dictionarymatcher_test.js', ['goog.labs.testing.dictionaryMatcherTest'], ['goog.labs.testing.HasEntryMatcher', 'goog.labs.testing.MatcherError', 'goog.labs.testing.assertThat', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/testing/environment.js', ['goog.labs.testing.Environment'], ['goog.Thenable', 'goog.array', 'goog.asserts', 'goog.debug.Console', 'goog.testing.MockClock', 'goog.testing.MockControl', 'goog.testing.PropertyReplacer', 'goog.testing.TestCase', 'goog.testing.jsunit'], {'lang': 'es6'});
goog.addDependency('labs/testing/environment_test.js', ['goog.labs.testing.environmentTest'], ['goog.asserts', 'goog.labs.testing.Environment', 'goog.testing.MockControl', 'goog.testing.PropertyReplacer', 'goog.testing.TestCase', 'goog.testing.testSuite'], {'lang': 'es8', 'module': 'goog'});
goog.addDependency('labs/testing/environment_usage_test.js', ['goog.labs.testing.environmentUsageTest'], ['goog.labs.testing.Environment', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/testing/json_fuzzing.js', ['goog.labs.testing.JsonFuzzing'], ['goog.string', 'goog.testing.PseudoRandom'], {});
goog.addDependency('labs/testing/json_fuzzing_test.js', ['goog.labs.testing.JsonFuzzingTest'], ['goog.json', 'goog.labs.testing.JsonFuzzing', 'goog.testing.asserts', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/testing/logicmatcher.js', ['goog.labs.testing.AllOfMatcher', 'goog.labs.testing.AnyOfMatcher', 'goog.labs.testing.IsNotMatcher', 'goog.labs.testing.logicMatchers'], ['goog.array', 'goog.labs.testing.Matcher'], {});
goog.addDependency('labs/testing/logicmatcher_test.js', ['goog.labs.testing.logicMatcherTest'], ['goog.labs.testing.AllOfMatcher', 'goog.labs.testing.GreaterThanMatcher', 'goog.labs.testing.MatcherError', 'goog.labs.testing.assertThat', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/testing/matcher.js', ['goog.labs.testing.Matcher'], [], {'lang': 'es6'});
goog.addDependency('labs/testing/numbermatcher.js', ['goog.labs.testing.AnyNumberMatcher', 'goog.labs.testing.CloseToMatcher', 'goog.labs.testing.EqualToMatcher', 'goog.labs.testing.GreaterThanEqualToMatcher', 'goog.labs.testing.GreaterThanMatcher', 'goog.labs.testing.LessThanEqualToMatcher', 'goog.labs.testing.LessThanMatcher'], ['goog.asserts', 'goog.labs.testing.Matcher'], {});
goog.addDependency('labs/testing/numbermatcher_test.js', ['goog.labs.testing.numberMatcherTest'], ['goog.labs.testing.LessThanMatcher', 'goog.labs.testing.MatcherError', 'goog.labs.testing.assertThat', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/testing/objectmatcher.js', ['goog.labs.testing.AnyObjectMatcher', 'goog.labs.testing.HasPropertyMatcher', 'goog.labs.testing.InstanceOfMatcher', 'goog.labs.testing.IsNullMatcher', 'goog.labs.testing.IsNullOrUndefinedMatcher', 'goog.labs.testing.IsUndefinedMatcher', 'goog.labs.testing.ObjectEqualsMatcher'], ['goog.labs.testing.Matcher'], {});
goog.addDependency('labs/testing/objectmatcher_test.js', ['goog.labs.testing.objectMatcherTest'], ['goog.labs.testing.MatcherError', 'goog.labs.testing.ObjectEqualsMatcher', 'goog.labs.testing.assertThat', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/testing/stringmatcher.js', ['goog.labs.testing.AnyStringMatcher', 'goog.labs.testing.ContainsStringMatcher', 'goog.labs.testing.EndsWithMatcher', 'goog.labs.testing.EqualToIgnoringWhitespaceMatcher', 'goog.labs.testing.EqualsMatcher', 'goog.labs.testing.RegexMatcher', 'goog.labs.testing.StartsWithMatcher', 'goog.labs.testing.StringContainsInOrderMatcher'], ['goog.asserts', 'goog.labs.testing.Matcher', 'goog.string'], {});
goog.addDependency('labs/testing/stringmatcher_test.js', ['goog.labs.testing.stringMatcherTest'], ['goog.labs.testing.MatcherError', 'goog.labs.testing.StringContainsInOrderMatcher', 'goog.labs.testing.assertThat', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/useragent/browser.js', ['goog.labs.userAgent.browser'], ['goog.array', 'goog.labs.userAgent.util', 'goog.object', 'goog.string.internal'], {});
goog.addDependency('labs/useragent/browser_test.js', ['goog.labs.userAgent.browserTest'], ['goog.labs.userAgent.browser', 'goog.labs.userAgent.testAgents', 'goog.labs.userAgent.util', 'goog.object', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/useragent/device.js', ['goog.labs.userAgent.device'], ['goog.labs.userAgent.util'], {});
goog.addDependency('labs/useragent/device_test.js', ['goog.labs.userAgent.deviceTest'], ['goog.labs.userAgent.device', 'goog.labs.userAgent.testAgents', 'goog.labs.userAgent.util', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/useragent/engine.js', ['goog.labs.userAgent.engine'], ['goog.array', 'goog.labs.userAgent.util', 'goog.string'], {});
goog.addDependency('labs/useragent/engine_test.js', ['goog.labs.userAgent.engineTest'], ['goog.labs.userAgent.engine', 'goog.labs.userAgent.testAgents', 'goog.labs.userAgent.util', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/useragent/extra.js', ['goog.labs.userAgent.extra'], ['goog.labs.userAgent.browser', 'goog.labs.userAgent.platform'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/useragent/extra_test.js', ['goog.labs.userAgent.extraTest'], ['goog.labs.userAgent.browser', 'goog.labs.userAgent.extra', 'goog.labs.userAgent.testAgents', 'goog.labs.userAgent.util', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/useragent/platform.js', ['goog.labs.userAgent.platform'], ['goog.labs.userAgent.util', 'goog.string'], {});
goog.addDependency('labs/useragent/platform_test.js', ['goog.labs.userAgent.platformTest'], ['goog.labs.userAgent.platform', 'goog.labs.userAgent.testAgents', 'goog.labs.userAgent.util', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/useragent/test_agents.js', ['goog.labs.userAgent.testAgents'], [], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/useragent/util.js', ['goog.labs.userAgent.util'], ['goog.string.internal'], {});
goog.addDependency('labs/useragent/util_test.js', ['goog.labs.userAgent.utilTest'], ['goog.functions', 'goog.labs.userAgent.testAgents', 'goog.labs.userAgent.util', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('labs/useragent/verifier.js', ['goog.labs.useragent.verifier'], [], {'lang': 'es6'});
goog.addDependency('labs/useragent/verifier_test.js', ['goog.labs.useragent.verifierTest'], ['goog.labs.userAgent.browser', 'goog.labs.useragent.verifier', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('loader/abstractmodulemanager.js', ['goog.loader.AbstractModuleManager', 'goog.loader.AbstractModuleManager.CallbackType', 'goog.loader.AbstractModuleManager.FailureType'], ['goog.module.AbstractModuleLoader', 'goog.module.ModuleInfo', 'goog.module.ModuleLoadCallback'], {});
goog.addDependency('loader/activemodulemanager.js', ['goog.loader.activeModuleManager'], ['goog.asserts', 'goog.loader.AbstractModuleManager'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('locale/countries.js', ['goog.locale.countries'], [], {});
goog.addDependency('locale/countrylanguagenames_test.js', ['goog.locale.countryLanguageNamesTest'], ['goog.locale', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('locale/defaultlocalenameconstants.js', ['goog.locale.defaultLocaleNameConstants'], [], {});
goog.addDependency('locale/genericfontnames.js', ['goog.locale.genericFontNames'], [], {});
goog.addDependency('locale/genericfontnames_test.js', ['goog.locale.genericFontNamesTest'], ['goog.locale.genericFontNames', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('locale/genericfontnamesdata.js', ['goog.locale.genericFontNamesData'], [], {});
goog.addDependency('locale/locale.js', ['goog.locale'], ['goog.locale.nativeNameConstants'], {});
goog.addDependency('locale/nativenameconstants.js', ['goog.locale.nativeNameConstants'], [], {});
goog.addDependency('locale/scriptToLanguages.js', ['goog.locale.scriptToLanguages'], ['goog.locale'], {});
goog.addDependency('locale/timezonedetection.js', ['goog.locale.timeZoneDetection'], ['goog.locale.TimeZoneFingerprint'], {});
goog.addDependency('locale/timezonedetection_test.js', ['goog.locale.timeZoneDetectionTest'], ['goog.locale.timeZoneDetection', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('locale/timezonefingerprint.js', ['goog.locale.TimeZoneFingerprint'], [], {});
goog.addDependency('locale/timezonelist.js', ['goog.locale.TimeZoneList', 'goog.locale.getTimeZoneAllLongNames', 'goog.locale.getTimeZoneSelectedLongNames', 'goog.locale.getTimeZoneSelectedShortNames'], ['goog.locale'], {});
goog.addDependency('locale/timezonelist_test.js', ['goog.locale.TimeZoneListTest'], ['goog.locale', 'goog.locale.TimeZoneList', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('log/log.js', ['goog.log', 'goog.log.Level', 'goog.log.LogRecord', 'goog.log.Logger'], ['goog.debug', 'goog.debug.LogManager', 'goog.debug.LogRecord', 'goog.debug.Logger'], {});
goog.addDependency('log/log_test.js', ['goog.logTest'], ['goog.debug.LogManager', 'goog.log', 'goog.log.Level', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('math/affinetransform.js', ['goog.math.AffineTransform'], [], {'lang': 'es6'});
goog.addDependency('math/affinetransform_test.js', ['goog.math.AffineTransformTest'], ['goog.array', 'goog.math', 'goog.math.AffineTransform', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('math/bezier.js', ['goog.math.Bezier'], ['goog.math', 'goog.math.Coordinate'], {});
goog.addDependency('math/bezier_test.js', ['goog.math.BezierTest'], ['goog.math', 'goog.math.Bezier', 'goog.math.Coordinate', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('math/box.js', ['goog.math.Box'], ['goog.asserts', 'goog.math.Coordinate'], {});
goog.addDependency('math/box_test.js', ['goog.math.BoxTest'], ['goog.math.Box', 'goog.math.Coordinate', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('math/coordinate.js', ['goog.math.Coordinate'], ['goog.math'], {});
goog.addDependency('math/coordinate3.js', ['goog.math.Coordinate3'], [], {'lang': 'es6'});
goog.addDependency('math/coordinate3_test.js', ['goog.math.Coordinate3Test'], ['goog.math.Coordinate3', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('math/coordinate_test.js', ['goog.math.CoordinateTest'], ['goog.math.Coordinate', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('math/exponentialbackoff.js', ['goog.math.ExponentialBackoff'], ['goog.asserts'], {});
goog.addDependency('math/exponentialbackoff_test.js', ['goog.math.ExponentialBackoffTest'], ['goog.math.ExponentialBackoff', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('math/integer.js', ['goog.math.Integer'], ['goog.reflect'], {});
goog.addDependency('math/integer_test.js', ['goog.math.IntegerTest'], ['goog.math.Integer', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('math/interpolator/interpolator1.js', ['goog.math.interpolator.Interpolator1'], [], {});
goog.addDependency('math/interpolator/linear1.js', ['goog.math.interpolator.Linear1'], ['goog.array', 'goog.asserts', 'goog.math', 'goog.math.interpolator.Interpolator1'], {});
goog.addDependency('math/interpolator/linear1_test.js', ['goog.math.interpolator.Linear1Test'], ['goog.math.interpolator.Linear1', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('math/interpolator/pchip1.js', ['goog.math.interpolator.Pchip1'], ['goog.math', 'goog.math.interpolator.Spline1'], {});
goog.addDependency('math/interpolator/pchip1_test.js', ['goog.math.interpolator.Pchip1Test'], ['goog.math.interpolator.Pchip1', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('math/interpolator/spline1.js', ['goog.math.interpolator.Spline1'], ['goog.array', 'goog.asserts', 'goog.math', 'goog.math.interpolator.Interpolator1', 'goog.math.tdma'], {});
goog.addDependency('math/interpolator/spline1_test.js', ['goog.math.interpolator.Spline1Test'], ['goog.math.interpolator.Spline1', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('math/irect.js', ['goog.math.IRect'], [], {});
goog.addDependency('math/line.js', ['goog.math.Line'], ['goog.math', 'goog.math.Coordinate'], {});
goog.addDependency('math/line_test.js', ['goog.math.LineTest'], ['goog.math.Coordinate', 'goog.math.Line', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('math/long.js', ['goog.math.Long'], ['goog.asserts', 'goog.reflect'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('math/long_test.js', ['goog.math.LongTest'], ['goog.asserts', 'goog.math.Long', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('math/math.js', ['goog.math'], ['goog.array', 'goog.asserts'], {});
goog.addDependency('math/math_test.js', ['goog.mathTest'], ['goog.math', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('math/matrix.js', ['goog.math.Matrix'], ['goog.array', 'goog.asserts', 'goog.math', 'goog.math.Size', 'goog.string'], {});
goog.addDependency('math/matrix_test.js', ['goog.math.MatrixTest'], ['goog.math.Matrix', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('math/path.js', ['goog.math.Path', 'goog.math.Path.Segment'], ['goog.array', 'goog.math', 'goog.math.AffineTransform'], {});
goog.addDependency('math/path_test.js', ['goog.math.PathTest'], ['goog.array', 'goog.math.AffineTransform', 'goog.math.Path', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('math/paths.js', ['goog.math.paths'], ['goog.math.Coordinate', 'goog.math.Path'], {});
goog.addDependency('math/paths_test.js', ['goog.math.pathsTest'], ['goog.math.Coordinate', 'goog.math.paths', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('math/range.js', ['goog.math.Range'], ['goog.asserts'], {});
goog.addDependency('math/range_test.js', ['goog.math.RangeTest'], ['goog.math.Range', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('math/rangeset.js', ['goog.math.RangeSet'], ['goog.array', 'goog.iter.Iterator', 'goog.iter.StopIteration', 'goog.math.Range'], {});
goog.addDependency('math/rangeset_test.js', ['goog.math.RangeSetTest'], ['goog.iter', 'goog.math.Range', 'goog.math.RangeSet', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('math/rect.js', ['goog.math.Rect'], ['goog.asserts', 'goog.math.Box', 'goog.math.Coordinate', 'goog.math.IRect', 'goog.math.Size'], {});
goog.addDependency('math/rect_test.js', ['goog.math.RectTest'], ['goog.math.Box', 'goog.math.Coordinate', 'goog.math.Rect', 'goog.math.Size', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('math/size.js', ['goog.math.Size'], [], {'lang': 'es6'});
goog.addDependency('math/size_test.js', ['goog.math.SizeTest'], ['goog.math.Size', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('math/tdma.js', ['goog.math.tdma'], [], {'lang': 'es6'});
goog.addDependency('math/tdma_test.js', ['goog.math.tdmaTest'], ['goog.math.tdma', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('math/vec2.js', ['goog.math.Vec2'], ['goog.math', 'goog.math.Coordinate'], {'lang': 'es6'});
goog.addDependency('math/vec2_test.js', ['goog.math.Vec2Test'], ['goog.math.Vec2', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('math/vec3.js', ['goog.math.Vec3'], ['goog.math', 'goog.math.Coordinate3'], {});
goog.addDependency('math/vec3_test.js', ['goog.math.Vec3Test'], ['goog.math.Coordinate3', 'goog.math.Vec3', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('memoize/memoize.js', ['goog.memoize'], [], {'lang': 'es6'});
goog.addDependency('memoize/memoize_test.js', ['goog.memoizeTest'], ['goog.memoize', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('messaging/abstractchannel.js', ['goog.messaging.AbstractChannel'], ['goog.Disposable', 'goog.json', 'goog.log', 'goog.messaging.MessageChannel'], {});
goog.addDependency('messaging/abstractchannel_test.js', ['goog.messaging.AbstractChannelTest'], ['goog.messaging.AbstractChannel', 'goog.testing.MockControl', 'goog.testing.async.MockControl', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('messaging/bufferedchannel.js', ['goog.messaging.BufferedChannel'], ['goog.Disposable', 'goog.Timer', 'goog.events', 'goog.log', 'goog.messaging.MessageChannel', 'goog.messaging.MultiChannel'], {});
goog.addDependency('messaging/bufferedchannel_test.js', ['goog.messaging.BufferedChannelTest'], ['goog.debug.Console', 'goog.dom', 'goog.dom.TagName', 'goog.log', 'goog.log.Level', 'goog.messaging.BufferedChannel', 'goog.testing.MockClock', 'goog.testing.MockControl', 'goog.testing.async.MockControl', 'goog.testing.messaging.MockMessageChannel', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('messaging/deferredchannel.js', ['goog.messaging.DeferredChannel'], ['goog.Disposable', 'goog.messaging.MessageChannel'], {});
goog.addDependency('messaging/deferredchannel_test.js', ['goog.messaging.DeferredChannelTest'], ['goog.async.Deferred', 'goog.messaging.DeferredChannel', 'goog.testing.MockControl', 'goog.testing.async.MockControl', 'goog.testing.messaging.MockMessageChannel', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('messaging/loggerclient.js', ['goog.messaging.LoggerClient'], ['goog.Disposable', 'goog.debug', 'goog.debug.LogManager', 'goog.debug.Logger'], {});
goog.addDependency('messaging/loggerclient_test.js', ['goog.messaging.LoggerClientTest'], ['goog.debug', 'goog.debug.Logger', 'goog.messaging.LoggerClient', 'goog.testing.MockControl', 'goog.testing.messaging.MockMessageChannel', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('messaging/loggerserver.js', ['goog.messaging.LoggerServer'], ['goog.Disposable', 'goog.log', 'goog.log.Level'], {});
goog.addDependency('messaging/loggerserver_test.js', ['goog.messaging.LoggerServerTest'], ['goog.debug.LogManager', 'goog.debug.Logger', 'goog.log', 'goog.log.Level', 'goog.messaging.LoggerServer', 'goog.testing.MockControl', 'goog.testing.PropertyReplacer', 'goog.testing.messaging.MockMessageChannel', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('messaging/messagechannel.js', ['goog.messaging.MessageChannel'], [], {});
goog.addDependency('messaging/messaging.js', ['goog.messaging'], [], {});
goog.addDependency('messaging/messaging_test.js', ['goog.testing.messaging.MockMessageChannelTest'], ['goog.messaging', 'goog.testing.MockControl', 'goog.testing.messaging.MockMessageChannel', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('messaging/multichannel.js', ['goog.messaging.MultiChannel', 'goog.messaging.MultiChannel.VirtualChannel'], ['goog.Disposable', 'goog.log', 'goog.messaging.MessageChannel', 'goog.object'], {});
goog.addDependency('messaging/multichannel_test.js', ['goog.messaging.MultiChannelTest'], ['goog.messaging.MultiChannel', 'goog.testing.MockControl', 'goog.testing.messaging.MockMessageChannel', 'goog.testing.mockmatchers.IgnoreArgument', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('messaging/portcaller.js', ['goog.messaging.PortCaller'], ['goog.Disposable', 'goog.async.Deferred', 'goog.messaging.DeferredChannel', 'goog.messaging.PortChannel', 'goog.messaging.PortNetwork', 'goog.object'], {});
goog.addDependency('messaging/portcaller_test.js', ['goog.messaging.PortCallerTest'], ['goog.events.EventTarget', 'goog.messaging.PortCaller', 'goog.messaging.PortNetwork', 'goog.testing.MockControl', 'goog.testing.messaging.MockMessageChannel', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('messaging/portchannel.js', ['goog.messaging.PortChannel'], ['goog.Timer', 'goog.array', 'goog.async.Deferred', 'goog.debug', 'goog.events', 'goog.events.EventType', 'goog.json', 'goog.log', 'goog.messaging.AbstractChannel', 'goog.messaging.DeferredChannel', 'goog.object', 'goog.string', 'goog.userAgent'], {});
goog.addDependency('messaging/portchannel_test.js', ['goog.messaging.PortChannelTest'], ['goog.Promise', 'goog.Timer', 'goog.dom', 'goog.dom.TagName', 'goog.events', 'goog.events.EventTarget', 'goog.events.EventType', 'goog.json', 'goog.messaging.PortChannel', 'goog.testing.MockControl', 'goog.testing.TestCase', 'goog.testing.messaging.MockMessageEvent', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('messaging/portnetwork.js', ['goog.messaging.PortNetwork'], [], {});
goog.addDependency('messaging/portnetwork_test.js', ['goog.messaging.PortNetworkTest'], ['goog.Promise', 'goog.Timer', 'goog.labs.userAgent.browser', 'goog.messaging.PortChannel', 'goog.messaging.PortOperator', 'goog.testing.TestCase', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('messaging/portoperator.js', ['goog.messaging.PortOperator'], ['goog.Disposable', 'goog.asserts', 'goog.log', 'goog.messaging.PortChannel', 'goog.messaging.PortNetwork', 'goog.object'], {});
goog.addDependency('messaging/portoperator_test.js', ['goog.messaging.PortOperatorTest'], ['goog.messaging.PortNetwork', 'goog.messaging.PortOperator', 'goog.testing.MockControl', 'goog.testing.PropertyReplacer', 'goog.testing.messaging.MockMessageChannel', 'goog.testing.messaging.MockMessagePort', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('messaging/respondingchannel.js', ['goog.messaging.RespondingChannel'], ['goog.Disposable', 'goog.Promise', 'goog.log', 'goog.messaging.MultiChannel'], {});
goog.addDependency('messaging/respondingchannel_test.js', ['goog.messaging.RespondingChannelTest'], ['goog.Promise', 'goog.messaging.RespondingChannel', 'goog.testing.MockControl', 'goog.testing.PropertyReplacer', 'goog.testing.messaging.MockMessageChannel', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('messaging/testdata/portchannel_worker.js', ['goog.messaging.testdata.portchannel_worker'], ['goog.messaging.PortChannel'], {});
goog.addDependency('messaging/testdata/portnetwork_worker1.js', ['goog.messaging.testdata.portnetwork_worker1'], ['goog.messaging.PortCaller', 'goog.messaging.PortChannel'], {});
goog.addDependency('messaging/testdata/portnetwork_worker2.js', ['goog.messaging.testdata.portnetwork_worker2'], ['goog.messaging.PortCaller', 'goog.messaging.PortChannel'], {});
goog.addDependency('module/abstractmoduleloader.js', ['goog.module.AbstractModuleLoader'], ['goog.module', 'goog.module.ModuleInfo'], {});
goog.addDependency('module/basemodule.js', ['goog.module.BaseModule'], ['goog.Disposable', 'goog.module'], {});
goog.addDependency('module/loader.js', ['goog.module.Loader'], ['goog.Timer', 'goog.array', 'goog.asserts', 'goog.dom', 'goog.dom.TagName', 'goog.dom.safe', 'goog.html.legacyconversions', 'goog.module', 'goog.object'], {});
goog.addDependency('module/module.js', ['goog.module'], [], {});
goog.addDependency('module/moduleinfo.js', ['goog.module.ModuleInfo'], ['goog.Disposable', 'goog.async.throwException', 'goog.functions', 'goog.html.TrustedResourceUrl', 'goog.module', 'goog.module.BaseModule', 'goog.module.ModuleLoadCallback'], {});
goog.addDependency('module/moduleinfo_test.js', ['goog.module.ModuleInfoTest'], ['goog.module.BaseModule', 'goog.module.ModuleInfo', 'goog.testing.MockClock', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('module/moduleloadcallback.js', ['goog.module.ModuleLoadCallback'], ['goog.debug.entryPointRegistry', 'goog.module'], {});
goog.addDependency('module/moduleloadcallback_test.js', ['goog.module.ModuleLoadCallbackTest'], ['goog.debug.ErrorHandler', 'goog.debug.entryPointRegistry', 'goog.functions', 'goog.module.ModuleLoadCallback', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('module/moduleloader.js', ['goog.module.ModuleLoader'], ['goog.Timer', 'goog.array', 'goog.asserts', 'goog.dom', 'goog.dom.safe', 'goog.events', 'goog.events.Event', 'goog.events.EventHandler', 'goog.events.EventId', 'goog.events.EventTarget', 'goog.functions', 'goog.html.TrustedResourceUrl', 'goog.labs.userAgent.browser', 'goog.log', 'goog.module.AbstractModuleLoader', 'goog.net.BulkLoader', 'goog.net.EventType', 'goog.net.jsloader', 'goog.userAgent', 'goog.userAgent.product'], {'lang': 'es6'});
goog.addDependency('module/moduleloader_test.js', ['goog.module.ModuleLoaderTest'], ['goog.Promise', 'goog.array', 'goog.dom', 'goog.dom.TagName', 'goog.events', 'goog.functions', 'goog.html.TrustedResourceUrl', 'goog.loader.activeModuleManager', 'goog.module.ModuleLoader', 'goog.module.ModuleManager', 'goog.net.BulkLoader', 'goog.net.XmlHttp', 'goog.object', 'goog.string.Const', 'goog.testing.PropertyReplacer', 'goog.testing.TestCase', 'goog.testing.events.EventObserver', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('module/modulemanager.js', ['goog.module.ModuleManager', 'goog.module.ModuleManager.CallbackType', 'goog.module.ModuleManager.FailureType'], ['goog.array', 'goog.asserts', 'goog.async.Deferred', 'goog.debug.Trace', 'goog.disposable.IDisposable', 'goog.disposeAll', 'goog.loader.AbstractModuleManager', 'goog.loader.activeModuleManager', 'goog.log', 'goog.module', 'goog.module.ModuleInfo', 'goog.module.ModuleLoadCallback', 'goog.object'], {'lang': 'es6'});
goog.addDependency('module/modulemanager_test.js', ['goog.module.ModuleManagerTest'], ['goog.array', 'goog.functions', 'goog.module.BaseModule', 'goog.module.ModuleManager', 'goog.testing', 'goog.testing.MockClock', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('module/testdata/modA_1.js', ['goog.module.testdata.modA_1'], [], {});
goog.addDependency('module/testdata/modA_2.js', ['goog.module.testdata.modA_2'], ['goog.module.ModuleManager'], {});
goog.addDependency('module/testdata/modB_1.js', ['goog.module.testdata.modB_1'], ['goog.module.ModuleManager'], {});
goog.addDependency('net/browserchannel.js', ['goog.net.BrowserChannel', 'goog.net.BrowserChannel.Error', 'goog.net.BrowserChannel.Event', 'goog.net.BrowserChannel.Handler', 'goog.net.BrowserChannel.LogSaver', 'goog.net.BrowserChannel.QueuedMap', 'goog.net.BrowserChannel.ServerReachability', 'goog.net.BrowserChannel.ServerReachabilityEvent', 'goog.net.BrowserChannel.Stat', 'goog.net.BrowserChannel.StatEvent', 'goog.net.BrowserChannel.State', 'goog.net.BrowserChannel.TimingEvent'], ['goog.Uri', 'goog.array', 'goog.asserts', 'goog.debug.TextFormatter', 'goog.events.Event', 'goog.events.EventTarget', 'goog.json', 'goog.json.NativeJsonProcessor', 'goog.log', 'goog.net.BrowserTestChannel', 'goog.net.ChannelDebug', 'goog.net.ChannelRequest', 'goog.net.XhrIo', 'goog.net.tmpnetwork', 'goog.object', 'goog.string', 'goog.structs', 'goog.structs.CircularBuffer'], {});
goog.addDependency('net/browserchannel_test.js', ['goog.net.BrowserChannelTest'], ['goog.Timer', 'goog.array', 'goog.dom', 'goog.functions', 'goog.json', 'goog.net.BrowserChannel', 'goog.net.ChannelDebug', 'goog.net.ChannelRequest', 'goog.net.tmpnetwork', 'goog.structs.Map', 'goog.testing.MockClock', 'goog.testing.PropertyReplacer', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('net/browsertestchannel.js', ['goog.net.BrowserTestChannel'], ['goog.json.NativeJsonProcessor', 'goog.net.ChannelRequest', 'goog.net.ChannelRequest.Error', 'goog.net.tmpnetwork', 'goog.string.Parser'], {});
goog.addDependency('net/bulkloader.js', ['goog.net.BulkLoader'], ['goog.events.Event', 'goog.events.EventHandler', 'goog.events.EventTarget', 'goog.log', 'goog.net.BulkLoaderHelper', 'goog.net.EventType', 'goog.net.XhrIo'], {});
goog.addDependency('net/bulkloader_test.js', ['goog.net.BulkLoaderTest'], ['goog.events.Event', 'goog.events.EventHandler', 'goog.net.BulkLoader', 'goog.net.EventType', 'goog.testing.MockClock', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('net/bulkloaderhelper.js', ['goog.net.BulkLoaderHelper'], ['goog.Disposable'], {});
goog.addDependency('net/channeldebug.js', ['goog.net.ChannelDebug'], ['goog.json', 'goog.log'], {'lang': 'es6'});
goog.addDependency('net/channelrequest.js', ['goog.net.ChannelRequest', 'goog.net.ChannelRequest.Error'], ['goog.Timer', 'goog.async.Throttle', 'goog.dom.TagName', 'goog.dom.safe', 'goog.events.EventHandler', 'goog.html.SafeUrl', 'goog.html.uncheckedconversions', 'goog.net.ErrorCode', 'goog.net.EventType', 'goog.net.XmlHttp', 'goog.object', 'goog.string', 'goog.string.Const', 'goog.userAgent'], {});
goog.addDependency('net/channelrequest_test.js', ['goog.net.ChannelRequestTest'], ['goog.Uri', 'goog.functions', 'goog.net.BrowserChannel', 'goog.net.ChannelDebug', 'goog.net.ChannelRequest', 'goog.testing.MockClock', 'goog.testing.PropertyReplacer', 'goog.testing.net.XhrIo', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('net/cookies.js', ['goog.net.Cookies', 'goog.net.cookies'], ['goog.asserts', 'goog.string'], {'lang': 'es5'});
goog.addDependency('net/cookies_test.js', ['goog.net.cookiesTest'], ['goog.array', 'goog.net.Cookies', 'goog.net.cookies', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('net/corsxmlhttpfactory.js', ['goog.net.CorsXmlHttpFactory', 'goog.net.IeCorsXhrAdapter'], ['goog.net.HttpStatus', 'goog.net.XhrLike', 'goog.net.XmlHttp', 'goog.net.XmlHttpFactory'], {});
goog.addDependency('net/corsxmlhttpfactory_test.js', ['goog.net.CorsXmlHttpFactoryTest'], ['goog.net.CorsXmlHttpFactory', 'goog.net.IeCorsXhrAdapter', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('net/crossdomainrpc.js', ['goog.net.CrossDomainRpc'], ['goog.Uri', 'goog.dom', 'goog.dom.TagName', 'goog.dom.safe', 'goog.events', 'goog.events.EventTarget', 'goog.events.EventType', 'goog.html.SafeHtml', 'goog.log', 'goog.net.EventType', 'goog.net.HttpStatus', 'goog.string', 'goog.userAgent'], {});
goog.addDependency('net/crossdomainrpc_test.js', ['goog.net.CrossDomainRpcTest'], ['goog.Promise', 'goog.log', 'goog.net.CrossDomainRpc', 'goog.testing.TestCase', 'goog.testing.testSuite', 'goog.userAgent', 'goog.userAgent.product'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('net/errorcode.js', ['goog.net.ErrorCode'], [], {});
goog.addDependency('net/eventtype.js', ['goog.net.EventType'], [], {});
goog.addDependency('net/fetchxmlhttpfactory.js', ['goog.net.FetchXmlHttp', 'goog.net.FetchXmlHttpFactory'], ['goog.asserts', 'goog.events.EventTarget', 'goog.functions', 'goog.log', 'goog.net.XhrLike', 'goog.net.XmlHttpFactory'], {'lang': 'es5'});
goog.addDependency('net/fetchxmlhttpfactory_test.js', ['goog.net.FetchXmlHttpFactoryTest'], ['goog.net.FetchXmlHttp', 'goog.net.FetchXmlHttpFactory', 'goog.testing.MockControl', 'goog.testing.recordFunction', 'goog.testing.testSuite', 'goog.userAgent.product', 'goog.userAgent.product.isVersion'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('net/filedownloader.js', ['goog.net.FileDownloader', 'goog.net.FileDownloader.Error'], ['goog.Disposable', 'goog.asserts', 'goog.async.Deferred', 'goog.crypt.hash32', 'goog.debug.Error', 'goog.events', 'goog.events.EventHandler', 'goog.fs', 'goog.fs.DirectoryEntry', 'goog.fs.Error', 'goog.fs.FileSaver', 'goog.net.EventType', 'goog.net.XhrIo', 'goog.net.XhrIoPool', 'goog.object'], {});
goog.addDependency('net/filedownloader_test.js', ['goog.net.FileDownloaderTest'], ['goog.fs.Error', 'goog.net.ErrorCode', 'goog.net.FileDownloader', 'goog.net.XhrIo', 'goog.testing.PropertyReplacer', 'goog.testing.TestCase', 'goog.testing.fs', 'goog.testing.fs.FileSystem', 'goog.testing.net.XhrIoPool', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('net/httpstatus.js', ['goog.net.HttpStatus'], [], {});
goog.addDependency('net/httpstatusname.js', ['goog.net.HttpStatusName'], [], {});
goog.addDependency('net/iframeio.js', ['goog.net.IframeIo', 'goog.net.IframeIo.IncrementalDataEvent'], ['goog.Timer', 'goog.Uri', 'goog.array', 'goog.asserts', 'goog.debug.HtmlFormatter', 'goog.dom', 'goog.dom.InputType', 'goog.dom.TagName', 'goog.dom.safe', 'goog.events', 'goog.events.Event', 'goog.events.EventTarget', 'goog.events.EventType', 'goog.html.SafeUrl', 'goog.html.legacyconversions', 'goog.html.uncheckedconversions', 'goog.json', 'goog.log', 'goog.log.Level', 'goog.net.ErrorCode', 'goog.net.EventType', 'goog.reflect', 'goog.string', 'goog.string.Const', 'goog.structs', 'goog.userAgent'], {});
goog.addDependency('net/iframeio_test.js', ['goog.net.IframeIoTest'], ['goog.debug', 'goog.debug.DivConsole', 'goog.debug.LogManager', 'goog.dom', 'goog.dom.TagName', 'goog.events', 'goog.events.EventType', 'goog.log', 'goog.log.Level', 'goog.net.IframeIo', 'goog.testing.events', 'goog.testing.events.Event', 'goog.testing.jsunit', 'goog.userAgent'], {'lang': 'es6'});
goog.addDependency('net/iframeloadmonitor.js', ['goog.net.IframeLoadMonitor'], ['goog.dom', 'goog.events', 'goog.events.EventTarget', 'goog.events.EventType', 'goog.userAgent'], {});
goog.addDependency('net/iframeloadmonitor_test.js', ['goog.net.IframeLoadMonitorTest'], ['goog.Promise', 'goog.Timer', 'goog.dom', 'goog.dom.TagName', 'goog.events', 'goog.net.IframeLoadMonitor', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('net/imageloader.js', ['goog.net.ImageLoader'], ['goog.array', 'goog.dom', 'goog.dom.TagName', 'goog.events.EventHandler', 'goog.events.EventTarget', 'goog.events.EventType', 'goog.net.EventType', 'goog.object', 'goog.userAgent'], {});
goog.addDependency('net/imageloader_test.js', ['goog.net.ImageLoaderTest'], ['goog.Promise', 'goog.Timer', 'goog.array', 'goog.dispose', 'goog.events', 'goog.events.Event', 'goog.events.EventType', 'goog.net.EventType', 'goog.net.ImageLoader', 'goog.object', 'goog.string', 'goog.testing.TestCase', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('net/ipaddress.js', ['goog.net.IpAddress', 'goog.net.Ipv4Address', 'goog.net.Ipv6Address'], ['goog.array', 'goog.math.Integer', 'goog.object', 'goog.string'], {});
goog.addDependency('net/ipaddress_test.js', ['goog.net.IpAddressTest'], ['goog.array', 'goog.math.Integer', 'goog.net.IpAddress', 'goog.net.Ipv4Address', 'goog.net.Ipv6Address', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('net/jsloader.js', ['goog.net.jsloader', 'goog.net.jsloader.Error', 'goog.net.jsloader.ErrorCode', 'goog.net.jsloader.Options'], ['goog.array', 'goog.async.Deferred', 'goog.debug.Error', 'goog.dom', 'goog.dom.TagName', 'goog.dom.safe', 'goog.html.TrustedResourceUrl', 'goog.object'], {});
goog.addDependency('net/jsloader_test.js', ['goog.net.jsloaderTest'], ['goog.array', 'goog.dom', 'goog.dom.TagName', 'goog.html.TrustedResourceUrl', 'goog.net.jsloader', 'goog.net.jsloader.ErrorCode', 'goog.string.Const', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('net/jsonp.js', ['goog.net.Jsonp'], ['goog.html.TrustedResourceUrl', 'goog.net.jsloader', 'goog.object'], {});
goog.addDependency('net/jsonp_test.js', ['goog.net.JsonpTest'], ['goog.html.TrustedResourceUrl', 'goog.net.Jsonp', 'goog.string.Const', 'goog.testing.PropertyReplacer', 'goog.testing.recordFunction', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('net/mockiframeio.js', ['goog.net.MockIFrameIo'], ['goog.events.EventTarget', 'goog.net.ErrorCode', 'goog.net.EventType', 'goog.net.IframeIo'], {});
goog.addDependency('net/multiiframeloadmonitor.js', ['goog.net.MultiIframeLoadMonitor'], ['goog.events', 'goog.net.IframeLoadMonitor'], {});
goog.addDependency('net/multiiframeloadmonitor_test.js', ['goog.net.MultiIframeLoadMonitorTest'], ['goog.Promise', 'goog.Timer', 'goog.dom', 'goog.dom.TagName', 'goog.net.MultiIframeLoadMonitor', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('net/networkstatusmonitor.js', ['goog.net.NetworkStatusMonitor'], ['goog.events.Listenable'], {});
goog.addDependency('net/networktester.js', ['goog.net.NetworkTester'], ['goog.Timer', 'goog.Uri', 'goog.dom.safe', 'goog.log'], {});
goog.addDependency('net/networktester_test.js', ['goog.net.NetworkTesterTest'], ['goog.Uri', 'goog.net.NetworkTester', 'goog.testing.MockClock', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('net/rpc/httpcors.js', ['goog.net.rpc.HttpCors'], ['goog.Uri', 'goog.object', 'goog.string', 'goog.uri.utils'], {'module': 'goog'});
goog.addDependency('net/rpc/httpcors_test.js', ['goog.net.rpc.HttpCorsTest'], ['goog.Uri', 'goog.net.rpc.HttpCors', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('net/streams/base64pbstreamparser.js', ['goog.net.streams.Base64PbStreamParser'], ['goog.asserts', 'goog.net.streams.Base64StreamDecoder', 'goog.net.streams.PbStreamParser', 'goog.net.streams.StreamParser'], {'module': 'goog'});
goog.addDependency('net/streams/base64pbstreamparser_test.js', ['goog.net.streams.Base64PbStreamParserTest'], ['goog.crypt.base64', 'goog.net.streams.Base64PbStreamParser', 'goog.object', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('net/streams/base64streamdecoder.js', ['goog.net.streams.Base64StreamDecoder'], ['goog.asserts', 'goog.crypt.base64'], {});
goog.addDependency('net/streams/base64streamdecoder_test.js', ['goog.net.streams.Base64StreamDecoderTest'], ['goog.net.streams.Base64StreamDecoder', 'goog.testing.asserts', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('net/streams/jsonstreamparser.js', ['goog.net.streams.JsonStreamParser', 'goog.net.streams.JsonStreamParser.Options'], ['goog.asserts', 'goog.net.streams.StreamParser', 'goog.net.streams.utils'], {});
goog.addDependency('net/streams/jsonstreamparser_test.js', ['goog.net.streams.JsonStreamParserTest'], ['goog.array', 'goog.json', 'goog.labs.testing.JsonFuzzing', 'goog.net.streams.JsonStreamParser', 'goog.testing.asserts', 'goog.testing.testSuite', 'goog.uri.utils'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('net/streams/nodereadablestream.js', ['goog.net.streams.NodeReadableStream'], [], {});
goog.addDependency('net/streams/pbjsonstreamparser.js', ['goog.net.streams.PbJsonStreamParser'], ['goog.asserts', 'goog.net.streams.JsonStreamParser', 'goog.net.streams.StreamParser', 'goog.net.streams.utils'], {'module': 'goog'});
goog.addDependency('net/streams/pbjsonstreamparser_test.js', ['goog.net.streams.PbJsonStreamParserTest'], ['goog.net.streams.PbJsonStreamParser', 'goog.object', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('net/streams/pbstreamparser.js', ['goog.net.streams.PbStreamParser'], ['goog.asserts', 'goog.net.streams.StreamParser'], {});
goog.addDependency('net/streams/pbstreamparser_test.js', ['goog.net.streams.PbStreamParserTest'], ['goog.net.streams.PbStreamParser', 'goog.object', 'goog.testing.asserts', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('net/streams/streamfactory.js', ['goog.net.streams.createXhrNodeReadableStream'], ['goog.asserts', 'goog.net.streams.XhrNodeReadableStream', 'goog.net.streams.XhrStreamReader'], {});
goog.addDependency('net/streams/streamparser.js', ['goog.net.streams.StreamParser'], [], {});
goog.addDependency('net/streams/utils.js', ['goog.net.streams.utils'], [], {'module': 'goog'});
goog.addDependency('net/streams/xhrnodereadablestream.js', ['goog.net.streams.XhrNodeReadableStream'], ['goog.array', 'goog.log', 'goog.net.streams.NodeReadableStream', 'goog.net.streams.XhrStreamReader'], {});
goog.addDependency('net/streams/xhrnodereadablestream_test.js', ['goog.net.streams.XhrNodeReadableStreamTest'], ['goog.net.streams.NodeReadableStream', 'goog.net.streams.XhrNodeReadableStream', 'goog.net.streams.XhrStreamReader', 'goog.testing.PropertyReplacer', 'goog.testing.asserts', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('net/streams/xhrstreamreader.js', ['goog.net.streams.XhrStreamReader'], ['goog.events.EventHandler', 'goog.log', 'goog.net.ErrorCode', 'goog.net.EventType', 'goog.net.HttpStatus', 'goog.net.XhrIo', 'goog.net.XmlHttp', 'goog.net.streams.Base64PbStreamParser', 'goog.net.streams.JsonStreamParser', 'goog.net.streams.PbJsonStreamParser', 'goog.net.streams.PbStreamParser', 'goog.string', 'goog.userAgent'], {});
goog.addDependency('net/streams/xhrstreamreader_test.js', ['goog.net.streams.XhrStreamReaderTest'], ['goog.net.ErrorCode', 'goog.net.HttpStatus', 'goog.net.XhrIo', 'goog.net.XmlHttp', 'goog.net.streams.Base64PbStreamParser', 'goog.net.streams.JsonStreamParser', 'goog.net.streams.PbJsonStreamParser', 'goog.net.streams.PbStreamParser', 'goog.net.streams.XhrStreamReader', 'goog.object', 'goog.testing.net.XhrIo', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('net/testdata/jsloader_test1.js', ['goog.net.testdata.jsloader_test1'], [], {});
goog.addDependency('net/testdata/jsloader_test2.js', ['goog.net.testdata.jsloader_test2'], [], {});
goog.addDependency('net/testdata/jsloader_test3.js', ['goog.net.testdata.jsloader_test3'], [], {});
goog.addDependency('net/testdata/jsloader_test4.js', ['goog.net.testdata.jsloader_test4'], [], {});
goog.addDependency('net/tmpnetwork.js', ['goog.net.tmpnetwork'], ['goog.Uri', 'goog.dom.safe', 'goog.net.ChannelDebug'], {});
goog.addDependency('net/websocket.js', ['goog.net.WebSocket', 'goog.net.WebSocket.ErrorEvent', 'goog.net.WebSocket.EventType', 'goog.net.WebSocket.MessageEvent'], ['goog.Timer', 'goog.asserts', 'goog.debug.entryPointRegistry', 'goog.events', 'goog.events.Event', 'goog.events.EventTarget', 'goog.log'], {'lang': 'es5'});
goog.addDependency('net/websocket_test.js', ['goog.net.WebSocketTest'], ['goog.debug.EntryPointMonitor', 'goog.debug.ErrorHandler', 'goog.debug.entryPointRegistry', 'goog.events', 'goog.functions', 'goog.net.WebSocket', 'goog.testing.MockClock', 'goog.testing.PropertyReplacer', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('net/wrapperxmlhttpfactory.js', ['goog.net.WrapperXmlHttpFactory'], ['goog.net.XhrLike', 'goog.net.XmlHttpFactory'], {});
goog.addDependency('net/xhrio.js', ['goog.net.XhrIo', 'goog.net.XhrIo.ResponseType'], ['goog.Timer', 'goog.array', 'goog.asserts', 'goog.debug.entryPointRegistry', 'goog.events.EventTarget', 'goog.json.hybrid', 'goog.log', 'goog.net.ErrorCode', 'goog.net.EventType', 'goog.net.HttpStatus', 'goog.net.XmlHttp', 'goog.object', 'goog.string', 'goog.structs', 'goog.structs.Map', 'goog.uri.utils', 'goog.userAgent'], {});
goog.addDependency('net/xhrio_test.js', ['goog.net.XhrIoTest'], ['goog.Uri', 'goog.debug.EntryPointMonitor', 'goog.debug.ErrorHandler', 'goog.debug.entryPointRegistry', 'goog.events', 'goog.functions', 'goog.net.EventType', 'goog.net.WrapperXmlHttpFactory', 'goog.net.XhrIo', 'goog.net.XmlHttp', 'goog.object', 'goog.string', 'goog.testing.MockClock', 'goog.testing.PropertyReplacer', 'goog.testing.jsunit', 'goog.testing.net.XhrIo', 'goog.testing.recordFunction', 'goog.userAgent.product'], {'lang': 'es6'});
goog.addDependency('net/xhriopool.js', ['goog.net.XhrIoPool'], ['goog.net.XhrIo', 'goog.structs.PriorityPool'], {});
goog.addDependency('net/xhriopool_test.js', ['goog.net.XhrIoPoolTest'], ['goog.net.XhrIoPool', 'goog.structs.Map', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('net/xhrlike.js', ['goog.net.XhrLike'], [], {});
goog.addDependency('net/xhrmanager.js', ['goog.net.XhrManager', 'goog.net.XhrManager.Event', 'goog.net.XhrManager.Request'], ['goog.events', 'goog.events.Event', 'goog.events.EventHandler', 'goog.events.EventTarget', 'goog.net.ErrorCode', 'goog.net.EventType', 'goog.net.XhrIo', 'goog.net.XhrIoPool', 'goog.structs.Map'], {});
goog.addDependency('net/xhrmanager_test.js', ['goog.net.XhrManagerTest'], ['goog.events', 'goog.net.EventType', 'goog.net.XhrIo', 'goog.net.XhrManager', 'goog.testing.net.XhrIoPool', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('net/xmlhttp.js', ['goog.net.DefaultXmlHttpFactory', 'goog.net.XmlHttp', 'goog.net.XmlHttp.OptionType', 'goog.net.XmlHttp.ReadyState', 'goog.net.XmlHttpDefines'], ['goog.asserts', 'goog.net.WrapperXmlHttpFactory', 'goog.net.XmlHttpFactory'], {});
goog.addDependency('net/xmlhttpfactory.js', ['goog.net.XmlHttpFactory'], ['goog.net.XhrLike'], {});
goog.addDependency('net/xpc/crosspagechannel.js', ['goog.net.xpc.CrossPageChannel'], ['goog.Uri', 'goog.async.Deferred', 'goog.async.Delay', 'goog.dispose', 'goog.dom', 'goog.dom.TagName', 'goog.dom.safe', 'goog.events', 'goog.events.EventHandler', 'goog.events.EventType', 'goog.html.legacyconversions', 'goog.json', 'goog.log', 'goog.messaging.AbstractChannel', 'goog.net.xpc', 'goog.net.xpc.CfgFields', 'goog.net.xpc.ChannelStates', 'goog.net.xpc.CrossPageChannelRole', 'goog.net.xpc.DirectTransport', 'goog.net.xpc.NativeMessagingTransport', 'goog.net.xpc.TransportTypes', 'goog.net.xpc.UriCfgFields', 'goog.string', 'goog.uri.utils', 'goog.userAgent'], {});
goog.addDependency('net/xpc/crosspagechannel_test.js', ['goog.net.xpc.CrossPageChannelTest'], ['goog.Disposable', 'goog.Promise', 'goog.Timer', 'goog.Uri', 'goog.dom', 'goog.dom.TagName', 'goog.labs.userAgent.browser', 'goog.log', 'goog.log.Level', 'goog.net.xpc', 'goog.net.xpc.CfgFields', 'goog.net.xpc.CrossPageChannel', 'goog.net.xpc.CrossPageChannelRole', 'goog.net.xpc.TransportTypes', 'goog.object', 'goog.testing.PropertyReplacer', 'goog.testing.TestCase', 'goog.testing.jsunit'], {'lang': 'es8'});
goog.addDependency('net/xpc/crosspagechannelrole.js', ['goog.net.xpc.CrossPageChannelRole'], [], {});
goog.addDependency('net/xpc/directtransport.js', ['goog.net.xpc.DirectTransport'], ['goog.Timer', 'goog.async.Deferred', 'goog.events.EventHandler', 'goog.log', 'goog.net.xpc', 'goog.net.xpc.CfgFields', 'goog.net.xpc.CrossPageChannelRole', 'goog.net.xpc.Transport', 'goog.net.xpc.TransportTypes', 'goog.object'], {});
goog.addDependency('net/xpc/directtransport_test.js', ['goog.net.xpc.DirectTransportTest'], ['goog.Promise', 'goog.dom', 'goog.dom.TagName', 'goog.labs.userAgent.browser', 'goog.log', 'goog.log.Level', 'goog.net.xpc', 'goog.net.xpc.CfgFields', 'goog.net.xpc.CrossPageChannel', 'goog.net.xpc.CrossPageChannelRole', 'goog.net.xpc.TransportTypes', 'goog.testing.TestCase', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('net/xpc/iframepollingtransport.js', ['goog.net.xpc.IframePollingTransport', 'goog.net.xpc.IframePollingTransport.Receiver', 'goog.net.xpc.IframePollingTransport.Sender'], ['goog.array', 'goog.dom', 'goog.dom.TagName', 'goog.dom.safe', 'goog.log', 'goog.log.Level', 'goog.net.xpc', 'goog.net.xpc.CfgFields', 'goog.net.xpc.CrossPageChannelRole', 'goog.net.xpc.Transport', 'goog.net.xpc.TransportTypes', 'goog.userAgent'], {});
goog.addDependency('net/xpc/iframepollingtransport_test.js', ['goog.net.xpc.IframePollingTransportTest'], ['goog.Timer', 'goog.dom', 'goog.dom.TagName', 'goog.functions', 'goog.net.xpc.CfgFields', 'goog.net.xpc.CrossPageChannel', 'goog.net.xpc.CrossPageChannelRole', 'goog.net.xpc.IframePollingTransport', 'goog.object', 'goog.testing.MockClock', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('net/xpc/nativemessagingtransport.js', ['goog.net.xpc.NativeMessagingTransport'], ['goog.Timer', 'goog.asserts', 'goog.async.Deferred', 'goog.events', 'goog.events.EventHandler', 'goog.log', 'goog.net.xpc', 'goog.net.xpc.CrossPageChannelRole', 'goog.net.xpc.Transport', 'goog.net.xpc.TransportTypes'], {});
goog.addDependency('net/xpc/nativemessagingtransport_test.js', ['goog.net.xpc.NativeMessagingTransportTest'], ['goog.dom', 'goog.events', 'goog.net.xpc', 'goog.net.xpc.CfgFields', 'goog.net.xpc.CrossPageChannel', 'goog.net.xpc.CrossPageChannelRole', 'goog.net.xpc.NativeMessagingTransport', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('net/xpc/relay.js', ['goog.net.xpc.relay'], [], {'lang': 'es6'});
goog.addDependency('net/xpc/transport.js', ['goog.net.xpc.Transport'], ['goog.Disposable', 'goog.dom', 'goog.net.xpc.TransportNames'], {});
goog.addDependency('net/xpc/xpc.js', ['goog.net.xpc', 'goog.net.xpc.CfgFields', 'goog.net.xpc.ChannelStates', 'goog.net.xpc.TransportNames', 'goog.net.xpc.TransportTypes', 'goog.net.xpc.UriCfgFields'], ['goog.log'], {});
goog.addDependency('object/object.js', ['goog.object'], [], {'lang': 'es6'});
goog.addDependency('object/object_test.js', ['goog.objectTest'], ['goog.array', 'goog.functions', 'goog.object', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('positioning/absoluteposition.js', ['goog.positioning.AbsolutePosition'], ['goog.math.Coordinate', 'goog.positioning', 'goog.positioning.AbstractPosition'], {});
goog.addDependency('positioning/abstractposition.js', ['goog.positioning.AbstractPosition'], [], {});
goog.addDependency('positioning/anchoredposition.js', ['goog.positioning.AnchoredPosition'], ['goog.positioning', 'goog.positioning.AbstractPosition'], {});
goog.addDependency('positioning/anchoredposition_test.js', ['goog.positioning.AnchoredPositionTest'], ['goog.dom', 'goog.positioning.AnchoredPosition', 'goog.positioning.Corner', 'goog.positioning.Overflow', 'goog.style', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('positioning/anchoredviewportposition.js', ['goog.positioning.AnchoredViewportPosition'], ['goog.positioning', 'goog.positioning.AnchoredPosition', 'goog.positioning.Overflow', 'goog.positioning.OverflowStatus'], {});
goog.addDependency('positioning/anchoredviewportposition_test.js', ['goog.positioning.AnchoredViewportPositionTest'], ['goog.dom', 'goog.math.Box', 'goog.positioning.AnchoredViewportPosition', 'goog.positioning.Corner', 'goog.positioning.OverflowStatus', 'goog.style', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('positioning/clientposition.js', ['goog.positioning.ClientPosition'], ['goog.asserts', 'goog.dom', 'goog.math.Coordinate', 'goog.positioning', 'goog.positioning.AbstractPosition', 'goog.style'], {});
goog.addDependency('positioning/clientposition_test.js', ['goog.positioning.clientPositionTest'], ['goog.dom', 'goog.dom.TagName', 'goog.positioning.ClientPosition', 'goog.positioning.Corner', 'goog.style', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('positioning/menuanchoredposition.js', ['goog.positioning.MenuAnchoredPosition'], ['goog.positioning.AnchoredViewportPosition', 'goog.positioning.Overflow'], {});
goog.addDependency('positioning/menuanchoredposition_test.js', ['goog.positioning.MenuAnchoredPositionTest'], ['goog.dom', 'goog.dom.TagName', 'goog.positioning.Corner', 'goog.positioning.MenuAnchoredPosition', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('positioning/positioning.js', ['goog.positioning', 'goog.positioning.Corner', 'goog.positioning.CornerBit', 'goog.positioning.Overflow', 'goog.positioning.OverflowStatus'], ['goog.asserts', 'goog.dom', 'goog.dom.TagName', 'goog.math.Coordinate', 'goog.math.Rect', 'goog.math.Size', 'goog.style', 'goog.style.bidi'], {});
goog.addDependency('positioning/positioning_test.js', ['goog.positioningTest'], ['goog.dom', 'goog.dom.DomHelper', 'goog.dom.TagName', 'goog.labs.userAgent.browser', 'goog.math.Box', 'goog.math.Coordinate', 'goog.math.Size', 'goog.positioning', 'goog.positioning.Corner', 'goog.positioning.Overflow', 'goog.positioning.OverflowStatus', 'goog.style', 'goog.testing.ExpectedFailures', 'goog.testing.testSuite', 'goog.userAgent', 'goog.userAgent.product'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('positioning/viewportclientposition.js', ['goog.positioning.ViewportClientPosition'], ['goog.dom', 'goog.math.Coordinate', 'goog.positioning', 'goog.positioning.ClientPosition', 'goog.positioning.Overflow', 'goog.positioning.OverflowStatus', 'goog.style'], {});
goog.addDependency('positioning/viewportclientposition_test.js', ['goog.positioning.ViewportClientPositionTest'], ['goog.dom', 'goog.positioning.Corner', 'goog.positioning.Overflow', 'goog.positioning.ViewportClientPosition', 'goog.style', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('positioning/viewportposition.js', ['goog.positioning.ViewportPosition'], ['goog.math.Coordinate', 'goog.positioning', 'goog.positioning.AbstractPosition', 'goog.positioning.Corner', 'goog.style'], {});
goog.addDependency('promise/nativeresolver.js', ['goog.promise.NativeResolver'], [], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('promise/nativeresolver_test.js', ['goog.promise.nativeResolverTest'], ['goog.promise.NativeResolver', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('promise/promise.js', ['goog.Promise'], ['goog.Thenable', 'goog.asserts', 'goog.async.FreeList', 'goog.async.run', 'goog.async.throwException', 'goog.debug.Error', 'goog.promise.Resolver'], {});
goog.addDependency('promise/promise_test.js', ['goog.PromiseTest'], ['goog.Promise', 'goog.Thenable', 'goog.Timer', 'goog.functions', 'goog.testing.MockClock', 'goog.testing.PropertyReplacer', 'goog.testing.TestCase', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es8', 'module': 'goog'});
goog.addDependency('promise/resolver.js', ['goog.promise.Resolver'], [], {});
goog.addDependency('promise/testsuiteadapter.js', ['goog.promise.testSuiteAdapter'], ['goog.Promise'], {});
goog.addDependency('promise/thenable.js', ['goog.Thenable'], [], {});
goog.addDependency('proto2/descriptor.js', ['goog.proto2.Descriptor', 'goog.proto2.Metadata'], ['goog.array', 'goog.asserts', 'goog.object', 'goog.string'], {});
goog.addDependency('proto2/descriptor_test.js', ['goog.proto2.DescriptorTest'], ['goog.proto2.Descriptor', 'goog.proto2.Message', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('proto2/fielddescriptor.js', ['goog.proto2.FieldDescriptor'], ['goog.asserts', 'goog.string'], {});
goog.addDependency('proto2/fielddescriptor_test.js', ['goog.proto2.FieldDescriptorTest'], ['goog.proto2.FieldDescriptor', 'goog.proto2.Message', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('proto2/lazydeserializer.js', ['goog.proto2.LazyDeserializer'], ['goog.asserts', 'goog.proto2.Message', 'goog.proto2.Serializer'], {});
goog.addDependency('proto2/message.js', ['goog.proto2.Message'], ['goog.asserts', 'goog.proto2.Descriptor', 'goog.proto2.FieldDescriptor'], {});
goog.addDependency('proto2/message_test.js', ['goog.proto2.MessageTest'], ['goog.testing.testSuite', 'proto2.TestAllTypes', 'proto2.TestAllTypes.NestedEnum', 'proto2.TestAllTypes.NestedMessage', 'proto2.TestAllTypes.OptionalGroup', 'proto2.TestAllTypes.RepeatedGroup'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('proto2/objectserializer.js', ['goog.proto2.ObjectSerializer'], ['goog.asserts', 'goog.proto2.FieldDescriptor', 'goog.proto2.Serializer', 'goog.string'], {});
goog.addDependency('proto2/objectserializer_test.js', ['goog.proto2.ObjectSerializerTest'], ['goog.proto2.ObjectSerializer', 'goog.proto2.Serializer', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite', 'proto2.TestAllTypes'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('proto2/package_test.pb.js', ['someprotopackage.TestPackageTypes'], ['goog.proto2.Message', 'proto2.TestAllTypes'], {'lang': 'es6'});
goog.addDependency('proto2/pbliteserializer.js', ['goog.proto2.PbLiteSerializer'], ['goog.asserts', 'goog.proto2.FieldDescriptor', 'goog.proto2.LazyDeserializer', 'goog.proto2.Serializer'], {});
goog.addDependency('proto2/pbliteserializer_test.js', ['goog.proto2.PbLiteSerializerTest'], ['goog.proto2.PbLiteSerializer', 'goog.testing.testSuite', 'proto2.TestAllTypes'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('proto2/proto_test.js', ['goog.proto2.messageTest'], ['goog.proto2.FieldDescriptor', 'goog.testing.testSuite', 'proto2.TestAllTypes', 'proto2.TestDefaultParent', 'someprotopackage.TestPackageTypes'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('proto2/serializer.js', ['goog.proto2.Serializer'], ['goog.asserts', 'goog.proto2.FieldDescriptor', 'goog.proto2.Message'], {});
goog.addDependency('proto2/test.pb.js', ['proto2.TestAllTypes', 'proto2.TestAllTypes.NestedEnum', 'proto2.TestAllTypes.NestedMessage', 'proto2.TestAllTypes.OptionalGroup', 'proto2.TestAllTypes.RepeatedGroup', 'proto2.TestDefaultChild', 'proto2.TestDefaultParent'], ['goog.proto2.Message'], {});
goog.addDependency('proto2/textformatserializer.js', ['goog.proto2.TextFormatSerializer'], ['goog.array', 'goog.asserts', 'goog.math', 'goog.object', 'goog.proto2.FieldDescriptor', 'goog.proto2.Message', 'goog.proto2.Serializer', 'goog.string'], {});
goog.addDependency('proto2/textformatserializer_test.js', ['goog.proto2.TextFormatSerializerTest'], ['goog.proto2.ObjectSerializer', 'goog.proto2.TextFormatSerializer', 'goog.testing.testSuite', 'proto2.TestAllTypes'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('proto2/util.js', ['goog.proto2.Util'], ['goog.asserts'], {});
goog.addDependency('pubsub/pubsub.js', ['goog.pubsub.PubSub'], ['goog.Disposable', 'goog.array', 'goog.async.run'], {});
goog.addDependency('pubsub/pubsub_test.js', ['goog.pubsub.PubSubTest'], ['goog.array', 'goog.pubsub.PubSub', 'goog.testing.MockClock', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('pubsub/topicid.js', ['goog.pubsub.TopicId'], [], {});
goog.addDependency('pubsub/typedpubsub.js', ['goog.pubsub.TypedPubSub'], ['goog.Disposable', 'goog.pubsub.PubSub'], {});
goog.addDependency('pubsub/typedpubsub_test.js', ['goog.pubsub.TypedPubSubTest'], ['goog.array', 'goog.pubsub.TopicId', 'goog.pubsub.TypedPubSub', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('reflect/reflect.js', ['goog.reflect'], [], {'lang': 'es6'});
goog.addDependency('reflect/reflect_test.js', ['goog.reflectTest'], ['goog.object', 'goog.reflect', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('result/chain_test.js', ['goog.result.chainTest'], ['goog.Timer', 'goog.result', 'goog.testing.MockClock', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('result/combine_test.js', ['goog.result.combineTest'], ['goog.Timer', 'goog.array', 'goog.result', 'goog.testing.MockClock', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('result/deferredadaptor.js', ['goog.result.DeferredAdaptor'], ['goog.async.Deferred', 'goog.result', 'goog.result.Result'], {});
goog.addDependency('result/deferredadaptor_test.js', ['goog.result.DeferredAdaptorTest'], ['goog.async.Deferred', 'goog.result', 'goog.result.DeferredAdaptor', 'goog.result.SimpleResult', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('result/dependentresult.js', ['goog.result.DependentResult'], ['goog.result.Result'], {});
goog.addDependency('result/result_interface.js', ['goog.result.Result'], ['goog.Thenable'], {});
goog.addDependency('result/resultutil.js', ['goog.result'], ['goog.array', 'goog.result.DependentResult', 'goog.result.Result', 'goog.result.SimpleResult'], {});
goog.addDependency('result/resultutil_test.js', ['goog.resultTest'], ['goog.result', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('result/simpleresult.js', ['goog.result.SimpleResult', 'goog.result.SimpleResult.StateError'], ['goog.Promise', 'goog.Thenable', 'goog.debug.Error', 'goog.result.Result'], {});
goog.addDependency('result/simpleresult_test.js', ['goog.result.SimpleResultTest'], ['goog.Promise', 'goog.Thenable', 'goog.Timer', 'goog.result', 'goog.testing.MockClock', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('result/transform_test.js', ['goog.result.transformTest'], ['goog.Timer', 'goog.result', 'goog.result.SimpleResult', 'goog.testing.MockClock', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('result/wait_test.js', ['goog.result.waitTest'], ['goog.Timer', 'goog.result', 'goog.result.SimpleResult', 'goog.testing.MockClock', 'goog.testing.PropertyReplacer', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('soy/data.js', ['goog.soy.data.SanitizedContent', 'goog.soy.data.SanitizedContentKind', 'goog.soy.data.SanitizedCss', 'goog.soy.data.SanitizedHtml', 'goog.soy.data.SanitizedHtmlAttribute', 'goog.soy.data.SanitizedJs', 'goog.soy.data.SanitizedTrustedResourceUri', 'goog.soy.data.SanitizedUri'], ['goog.Uri', 'goog.asserts', 'goog.html.SafeHtml', 'goog.html.SafeScript', 'goog.html.SafeStyle', 'goog.html.SafeStyleSheet', 'goog.html.SafeUrl', 'goog.html.TrustedResourceUrl', 'goog.html.uncheckedconversions', 'goog.i18n.bidi.Dir', 'goog.string.Const'], {});
goog.addDependency('soy/data_test.js', ['goog.soy.dataTest'], ['goog.html.SafeHtml', 'goog.html.SafeStyleSheet', 'goog.html.SafeUrl', 'goog.html.TrustedResourceUrl', 'goog.soy.testHelper', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('soy/renderer.js', ['goog.soy.InjectedDataSupplier', 'goog.soy.Renderer'], ['goog.asserts', 'goog.dom', 'goog.soy', 'goog.soy.data.SanitizedContent', 'goog.soy.data.SanitizedContentKind'], {});
goog.addDependency('soy/renderer_test.js', ['goog.soy.RendererTest'], ['goog.dom', 'goog.dom.NodeType', 'goog.dom.TagName', 'goog.html.SafeHtml', 'goog.i18n.bidi.Dir', 'goog.soy.Renderer', 'goog.soy.data.SanitizedContentKind', 'goog.soy.testHelper', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('soy/soy.js', ['goog.soy'], ['goog.asserts', 'goog.dom', 'goog.dom.NodeType', 'goog.dom.TagName', 'goog.dom.safe', 'goog.html.SafeHtml', 'goog.soy.data.SanitizedContent'], {});
goog.addDependency('soy/soy_test.js', ['goog.soyTest'], ['goog.dom', 'goog.dom.NodeType', 'goog.dom.TagName', 'goog.functions', 'goog.soy', 'goog.soy.testHelper', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('soy/soy_testhelper.js', ['goog.soy.testHelper'], ['goog.dom', 'goog.dom.TagName', 'goog.i18n.bidi.Dir', 'goog.soy.data.SanitizedContent', 'goog.soy.data.SanitizedContentKind', 'goog.soy.data.SanitizedCss', 'goog.soy.data.SanitizedTrustedResourceUri', 'goog.string', 'goog.userAgent'], {'lang': 'es6'});
goog.addDependency('spell/spellcheck.js', ['goog.spell.SpellCheck', 'goog.spell.SpellCheck.WordChangedEvent'], ['goog.Timer', 'goog.events.Event', 'goog.events.EventTarget', 'goog.structs.Set'], {});
goog.addDependency('spell/spellcheck_test.js', ['goog.spell.SpellCheckTest'], ['goog.spell.SpellCheck', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('stats/basicstat.js', ['goog.stats.BasicStat'], ['goog.asserts', 'goog.log', 'goog.string.format', 'goog.structs.CircularBuffer'], {});
goog.addDependency('stats/basicstat_test.js', ['goog.stats.BasicStatTest'], ['goog.array', 'goog.stats.BasicStat', 'goog.string.format', 'goog.testing.PseudoRandom', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('storage/collectablestorage.js', ['goog.storage.CollectableStorage'], ['goog.array', 'goog.iter', 'goog.storage.ErrorCode', 'goog.storage.ExpiringStorage', 'goog.storage.RichStorage'], {});
goog.addDependency('storage/collectablestorage_test.js', ['goog.storage.CollectableStorageTest'], ['goog.storage.CollectableStorage', 'goog.storage.collectableStorageTester', 'goog.storage.storageTester', 'goog.testing.MockClock', 'goog.testing.storage.FakeMechanism', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('storage/collectablestoragetester.js', ['goog.storage.collectableStorageTester'], ['goog.testing.asserts'], {});
goog.addDependency('storage/encryptedstorage.js', ['goog.storage.EncryptedStorage'], ['goog.crypt', 'goog.crypt.Arc4', 'goog.crypt.Sha1', 'goog.crypt.base64', 'goog.json', 'goog.json.Serializer', 'goog.storage.CollectableStorage', 'goog.storage.ErrorCode', 'goog.storage.RichStorage'], {});
goog.addDependency('storage/encryptedstorage_test.js', ['goog.storage.EncryptedStorageTest'], ['goog.json', 'goog.storage.EncryptedStorage', 'goog.storage.ErrorCode', 'goog.storage.RichStorage', 'goog.storage.collectableStorageTester', 'goog.storage.storageTester', 'goog.testing.MockClock', 'goog.testing.PseudoRandom', 'goog.testing.storage.FakeMechanism', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('storage/errorcode.js', ['goog.storage.ErrorCode'], [], {});
goog.addDependency('storage/expiringstorage.js', ['goog.storage.ExpiringStorage'], ['goog.storage.RichStorage'], {});
goog.addDependency('storage/expiringstorage_test.js', ['goog.storage.ExpiringStorageTest'], ['goog.storage.ExpiringStorage', 'goog.storage.storageTester', 'goog.testing.MockClock', 'goog.testing.storage.FakeMechanism', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('storage/mechanism/errorcode.js', ['goog.storage.mechanism.ErrorCode'], [], {});
goog.addDependency('storage/mechanism/errorhandlingmechanism.js', ['goog.storage.mechanism.ErrorHandlingMechanism'], ['goog.storage.mechanism.Mechanism'], {});
goog.addDependency('storage/mechanism/errorhandlingmechanism_test.js', ['goog.storage.mechanism.ErrorHandlingMechanismTest'], ['goog.storage.mechanism.ErrorHandlingMechanism', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('storage/mechanism/html5localstorage.js', ['goog.storage.mechanism.HTML5LocalStorage'], ['goog.storage.mechanism.HTML5WebStorage'], {});
goog.addDependency('storage/mechanism/html5localstorage_test.js', ['goog.storage.mechanism.HTML5LocalStorageTest'], ['goog.storage.mechanism.HTML5LocalStorage', 'goog.storage.mechanism.mechanismSeparationTester', 'goog.storage.mechanism.mechanismSharingTester', 'goog.storage.mechanism.mechanismTestDefinition', 'goog.testing.testSuite', 'goog.userAgent', 'goog.userAgent.product'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('storage/mechanism/html5sessionstorage.js', ['goog.storage.mechanism.HTML5SessionStorage'], ['goog.storage.mechanism.HTML5WebStorage'], {});
goog.addDependency('storage/mechanism/html5sessionstorage_test.js', ['goog.storage.mechanism.HTML5SessionStorageTest'], ['goog.storage.mechanism.HTML5SessionStorage', 'goog.storage.mechanism.mechanismSeparationTester', 'goog.storage.mechanism.mechanismSharingTester', 'goog.storage.mechanism.mechanismTestDefinition', 'goog.testing.testSuite', 'goog.userAgent', 'goog.userAgent.product'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('storage/mechanism/html5webstorage.js', ['goog.storage.mechanism.HTML5WebStorage'], ['goog.asserts', 'goog.iter.Iterator', 'goog.iter.StopIteration', 'goog.storage.mechanism.ErrorCode', 'goog.storage.mechanism.IterableMechanism'], {});
goog.addDependency('storage/mechanism/html5webstorage_test.js', ['goog.storage.mechanism.HTML5WebStorageTest'], ['goog.storage.mechanism.ErrorCode', 'goog.storage.mechanism.HTML5WebStorage', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('storage/mechanism/ieuserdata.js', ['goog.storage.mechanism.IEUserData'], ['goog.asserts', 'goog.iter.Iterator', 'goog.iter.StopIteration', 'goog.storage.mechanism.ErrorCode', 'goog.storage.mechanism.IterableMechanism', 'goog.structs.Map', 'goog.userAgent'], {});
goog.addDependency('storage/mechanism/ieuserdata_test.js', ['goog.storage.mechanism.IEUserDataTest'], ['goog.storage.mechanism.IEUserData', 'goog.storage.mechanism.mechanismSeparationTester', 'goog.storage.mechanism.mechanismSharingTester', 'goog.storage.mechanism.mechanismTestDefinition', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('storage/mechanism/iterablemechanism.js', ['goog.storage.mechanism.IterableMechanism'], ['goog.array', 'goog.asserts', 'goog.iter', 'goog.storage.mechanism.Mechanism'], {});
goog.addDependency('storage/mechanism/iterablemechanismtester.js', ['goog.storage.mechanism.iterableMechanismTester'], ['goog.iter', 'goog.iter.StopIteration', 'goog.testing.asserts'], {});
goog.addDependency('storage/mechanism/mechanism.js', ['goog.storage.mechanism.Mechanism'], [], {});
goog.addDependency('storage/mechanism/mechanismfactory.js', ['goog.storage.mechanism.mechanismfactory'], ['goog.storage.mechanism.HTML5LocalStorage', 'goog.storage.mechanism.HTML5SessionStorage', 'goog.storage.mechanism.IEUserData', 'goog.storage.mechanism.PrefixedMechanism'], {});
goog.addDependency('storage/mechanism/mechanismfactory_test.js', ['goog.storage.mechanism.mechanismfactoryTest'], ['goog.storage.mechanism.mechanismfactory', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('storage/mechanism/mechanismseparationtester.js', ['goog.storage.mechanism.mechanismSeparationTester'], ['goog.iter.StopIteration', 'goog.storage.mechanism.mechanismTestDefinition', 'goog.testing.asserts'], {});
goog.addDependency('storage/mechanism/mechanismsharingtester.js', ['goog.storage.mechanism.mechanismSharingTester'], ['goog.iter.StopIteration', 'goog.storage.mechanism.mechanismTestDefinition', 'goog.testing.asserts'], {});
goog.addDependency('storage/mechanism/mechanismtestdefinition.js', ['goog.storage.mechanism.mechanismTestDefinition'], [], {});
goog.addDependency('storage/mechanism/mechanismtester.js', ['goog.storage.mechanism.mechanismTester'], ['goog.storage.mechanism.ErrorCode', 'goog.testing.asserts', 'goog.userAgent', 'goog.userAgent.product', 'goog.userAgent.product.isVersion'], {});
goog.addDependency('storage/mechanism/prefixedmechanism.js', ['goog.storage.mechanism.PrefixedMechanism'], ['goog.iter.Iterator', 'goog.storage.mechanism.IterableMechanism'], {});
goog.addDependency('storage/mechanism/prefixedmechanism_test.js', ['goog.storage.mechanism.PrefixedMechanismTest'], ['goog.storage.mechanism.HTML5LocalStorage', 'goog.storage.mechanism.PrefixedMechanism', 'goog.storage.mechanism.mechanismSeparationTester', 'goog.storage.mechanism.mechanismSharingTester', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('storage/richstorage.js', ['goog.storage.RichStorage', 'goog.storage.RichStorage.Wrapper'], ['goog.storage.ErrorCode', 'goog.storage.Storage'], {});
goog.addDependency('storage/richstorage_test.js', ['goog.storage.RichStorageTest'], ['goog.storage.ErrorCode', 'goog.storage.RichStorage', 'goog.storage.storageTester', 'goog.testing.storage.FakeMechanism', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('storage/storage.js', ['goog.storage.Storage'], ['goog.json', 'goog.storage.ErrorCode'], {});
goog.addDependency('storage/storage_test.js', ['goog.storage.storage_test'], ['goog.functions', 'goog.storage.ErrorCode', 'goog.storage.Storage', 'goog.storage.storageTester', 'goog.testing.asserts', 'goog.testing.storage.FakeMechanism', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('storage/storagetester.js', ['goog.storage.storageTester'], ['goog.storage.Storage', 'goog.structs.Map', 'goog.testing.asserts'], {});
goog.addDependency('streams/defines.js', ['goog.streams.defines'], [], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('streams/full.js', ['goog.streams.full'], ['goog.streams.defines', 'goog.streams.fullImpl', 'goog.streams.fullNativeImpl', 'goog.streams.fullTypes'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('streams/full_impl.js', ['goog.streams.fullImpl'], ['goog.asserts', 'goog.promise.NativeResolver', 'goog.streams.fullTypes', 'goog.streams.liteImpl'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('streams/full_impl_test.js', ['goog.streams.fullImplTest'], ['goog.streams.fullImpl', 'goog.streams.fullTestCases', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('streams/full_native_impl.js', ['goog.streams.fullNativeImpl'], ['goog.streams.fullTypes', 'goog.streams.liteNativeImpl'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('streams/full_native_impl_test.js', ['goog.streams.fullNativeImplTest'], ['goog.streams.fullNativeImpl', 'goog.streams.fullTestCases', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('streams/full_test_cases.js', ['goog.streams.fullTestCases'], ['goog.streams.fullTypes', 'goog.streams.liteTestCases', 'goog.testing.recordFunction'], {'lang': 'es9', 'module': 'goog'});
goog.addDependency('streams/full_types.js', ['goog.streams.fullTypes'], ['goog.streams.liteTypes'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('streams/lite.js', ['goog.streams.lite'], ['goog.streams.defines', 'goog.streams.liteImpl', 'goog.streams.liteNativeImpl', 'goog.streams.liteTypes'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('streams/lite_impl.js', ['goog.streams.liteImpl'], ['goog.asserts', 'goog.promise.NativeResolver', 'goog.streams.liteTypes'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('streams/lite_impl_test.js', ['goog.streams.liteImplTest'], ['goog.streams.liteImpl', 'goog.streams.liteTestCases', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('streams/lite_native_impl.js', ['goog.streams.liteNativeImpl'], ['goog.streams.liteTypes'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('streams/lite_native_impl_test.js', ['goog.streams.liteNativeImplTest'], ['goog.streams.liteNativeImpl', 'goog.streams.liteTestCases', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('streams/lite_test_cases.js', ['goog.streams.liteTestCases'], ['goog.streams.liteTypes', 'goog.testing.jsunit'], {'lang': 'es8', 'module': 'goog'});
goog.addDependency('streams/lite_types.js', ['goog.streams.liteTypes'], [], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('string/const.js', ['goog.string.Const'], ['goog.asserts', 'goog.string.TypedString'], {});
goog.addDependency('string/const_test.js', ['goog.string.constTest'], ['goog.string.Const', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('string/internal.js', ['goog.string.internal'], [], {'lang': 'es6'});
goog.addDependency('string/linkify.js', ['goog.string.linkify'], ['goog.html.SafeHtml', 'goog.string'], {});
goog.addDependency('string/linkify_test.js', ['goog.string.linkifyTest'], ['goog.dom', 'goog.dom.TagName', 'goog.dom.safe', 'goog.html.SafeHtml', 'goog.string', 'goog.string.linkify', 'goog.testing.dom', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('string/newlines.js', ['goog.string.newlines', 'goog.string.newlines.Line'], ['goog.array'], {});
goog.addDependency('string/newlines_test.js', ['goog.string.newlinesTest'], ['goog.string.newlines', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('string/parser.js', ['goog.string.Parser'], [], {});
goog.addDependency('string/path.js', ['goog.string.path'], ['goog.array', 'goog.string'], {});
goog.addDependency('string/path_test.js', ['goog.string.pathTest'], ['goog.string.path', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('string/string.js', ['goog.string', 'goog.string.Unicode'], ['goog.dom.safe', 'goog.html.uncheckedconversions', 'goog.string.Const', 'goog.string.internal'], {});
goog.addDependency('string/string_test.js', ['goog.stringTest'], ['goog.dom', 'goog.dom.TagName', 'goog.functions', 'goog.object', 'goog.string', 'goog.string.Unicode', 'goog.testing.MockControl', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('string/stringbuffer.js', ['goog.string.StringBuffer'], [], {'lang': 'es6'});
goog.addDependency('string/stringbuffer_test.js', ['goog.string.StringBufferTest'], ['goog.string.StringBuffer', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('string/stringformat.js', ['goog.string.format'], ['goog.string'], {});
goog.addDependency('string/stringformat_test.js', ['goog.string.formatTest'], ['goog.string.format', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('string/stringifier.js', ['goog.string.Stringifier'], [], {});
goog.addDependency('string/typedstring.js', ['goog.string.TypedString'], [], {});
goog.addDependency('structs/avltree.js', ['goog.structs.AvlTree'], ['goog.asserts', 'goog.structs.Collection'], {'module': 'goog'});
goog.addDependency('structs/avltree_test.js', ['goog.structs.AvlTreeTest'], ['goog.array', 'goog.structs.AvlTree', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('structs/circularbuffer.js', ['goog.structs.CircularBuffer'], [], {'lang': 'es6'});
goog.addDependency('structs/circularbuffer_test.js', ['goog.structs.CircularBufferTest'], ['goog.structs.CircularBuffer', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('structs/collection.js', ['goog.structs.Collection'], [], {});
goog.addDependency('structs/collection_test.js', ['goog.structs.CollectionTest'], ['goog.structs.AvlTree', 'goog.structs.Set', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('structs/heap.js', ['goog.structs.Heap'], ['goog.array', 'goog.object', 'goog.structs.Node'], {});
goog.addDependency('structs/heap_test.js', ['goog.structs.HeapTest'], ['goog.structs', 'goog.structs.Heap', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('structs/inversionmap.js', ['goog.structs.InversionMap'], ['goog.array', 'goog.asserts'], {});
goog.addDependency('structs/inversionmap_test.js', ['goog.structs.InversionMapTest'], ['goog.structs.InversionMap', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('structs/linkedmap.js', ['goog.structs.LinkedMap'], ['goog.structs.Map'], {});
goog.addDependency('structs/linkedmap_test.js', ['goog.structs.LinkedMapTest'], ['goog.structs.LinkedMap', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('structs/map.js', ['goog.structs.Map'], ['goog.iter.Iterator', 'goog.iter.StopIteration'], {});
goog.addDependency('structs/map_test.js', ['goog.structs.MapTest'], ['goog.iter', 'goog.structs', 'goog.structs.Map', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('structs/node.js', ['goog.structs.Node'], [], {});
goog.addDependency('structs/pool.js', ['goog.structs.Pool'], ['goog.Disposable', 'goog.structs.Queue', 'goog.structs.Set'], {});
goog.addDependency('structs/pool_test.js', ['goog.structs.PoolTest'], ['goog.structs.Pool', 'goog.testing.MockClock', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('structs/prioritypool.js', ['goog.structs.PriorityPool'], ['goog.structs.Pool', 'goog.structs.PriorityQueue'], {});
goog.addDependency('structs/prioritypool_test.js', ['goog.structs.PriorityPoolTest'], ['goog.structs.PriorityPool', 'goog.testing.MockClock', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('structs/priorityqueue.js', ['goog.structs.PriorityQueue'], ['goog.structs.Heap'], {});
goog.addDependency('structs/priorityqueue_test.js', ['goog.structs.PriorityQueueTest'], ['goog.structs', 'goog.structs.PriorityQueue', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('structs/quadtree.js', ['goog.structs.QuadTree', 'goog.structs.QuadTree.Node', 'goog.structs.QuadTree.Point'], ['goog.math.Coordinate'], {});
goog.addDependency('structs/quadtree_test.js', ['goog.structs.QuadTreeTest'], ['goog.structs', 'goog.structs.QuadTree', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('structs/queue.js', ['goog.structs.Queue'], ['goog.array'], {});
goog.addDependency('structs/queue_test.js', ['goog.structs.QueueTest'], ['goog.structs.Queue', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('structs/set.js', ['goog.structs.Set'], ['goog.structs', 'goog.structs.Collection', 'goog.structs.Map'], {});
goog.addDependency('structs/set_test.js', ['goog.structs.SetTest'], ['goog.iter', 'goog.structs', 'goog.structs.Set', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('structs/simplepool.js', ['goog.structs.SimplePool'], ['goog.Disposable'], {});
goog.addDependency('structs/stringset.js', ['goog.structs.StringSet'], ['goog.asserts', 'goog.iter'], {});
goog.addDependency('structs/stringset_test.js', ['goog.structs.StringSetTest'], ['goog.array', 'goog.iter', 'goog.structs.StringSet', 'goog.testing.asserts', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('structs/structs.js', ['goog.structs'], ['goog.array', 'goog.object'], {});
goog.addDependency('structs/structs_test.js', ['goog.structsTest'], ['goog.array', 'goog.dom', 'goog.dom.TagName', 'goog.structs', 'goog.structs.Map', 'goog.structs.Set', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('structs/treenode.js', ['goog.structs.TreeNode'], ['goog.array', 'goog.asserts', 'goog.structs.Node'], {});
goog.addDependency('structs/treenode_test.js', ['goog.structs.TreeNodeTest'], ['goog.structs.TreeNode', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('structs/trie.js', ['goog.structs.Trie'], ['goog.object', 'goog.structs'], {});
goog.addDependency('structs/trie_test.js', ['goog.structs.TrieTest'], ['goog.object', 'goog.structs', 'goog.structs.Trie', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('style/bidi.js', ['goog.style.bidi'], ['goog.dom', 'goog.style', 'goog.userAgent', 'goog.userAgent.platform', 'goog.userAgent.product', 'goog.userAgent.product.isVersion'], {});
goog.addDependency('style/bidi_test.js', ['goog.style.bidiTest'], ['goog.dom', 'goog.style', 'goog.style.bidi', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('style/cursor.js', ['goog.style.cursor'], ['goog.userAgent'], {});
goog.addDependency('style/cursor_test.js', ['goog.style.cursorTest'], ['goog.style.cursor', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('style/style.js', ['goog.style'], ['goog.array', 'goog.asserts', 'goog.dom', 'goog.dom.NodeType', 'goog.dom.TagName', 'goog.dom.vendor', 'goog.html.SafeStyleSheet', 'goog.math.Box', 'goog.math.Coordinate', 'goog.math.Rect', 'goog.math.Size', 'goog.object', 'goog.reflect', 'goog.string', 'goog.userAgent'], {});
goog.addDependency('style/style_document_scroll_test.js', ['goog.style.style_document_scroll_test'], ['goog.dom', 'goog.style', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('style/style_test.js', ['goog.style_test'], ['goog.array', 'goog.color', 'goog.dom', 'goog.dom.TagName', 'goog.events.BrowserEvent', 'goog.html.testing', 'goog.labs.userAgent.util', 'goog.math.Box', 'goog.math.Coordinate', 'goog.math.Rect', 'goog.math.Size', 'goog.object', 'goog.style', 'goog.testing.ExpectedFailures', 'goog.testing.MockUserAgent', 'goog.testing.asserts', 'goog.testing.testSuite', 'goog.userAgent', 'goog.userAgentTestUtil', 'goog.userAgentTestUtil.UserAgents'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('style/style_webkit_scrollbars_test.js', ['goog.style.webkitScrollbarsTest'], ['goog.asserts', 'goog.style', 'goog.styleScrollbarTester', 'goog.testing.ExpectedFailures', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('style/stylescrollbartester.js', ['goog.styleScrollbarTester'], ['goog.dom', 'goog.dom.TagName', 'goog.style', 'goog.testing.asserts'], {});
goog.addDependency('style/transform.js', ['goog.style.transform'], ['goog.functions', 'goog.math.Coordinate', 'goog.math.Coordinate3', 'goog.style', 'goog.userAgent', 'goog.userAgent.product.isVersion'], {});
goog.addDependency('style/transform_test.js', ['goog.style.transformTest'], ['goog.dom', 'goog.dom.TagName', 'goog.style', 'goog.style.transform', 'goog.testing.testSuite', 'goog.userAgent', 'goog.userAgent.product.isVersion'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('style/transition.js', ['goog.style.transition', 'goog.style.transition.Css3Property'], ['goog.array', 'goog.asserts', 'goog.dom', 'goog.dom.TagName', 'goog.dom.safe', 'goog.dom.vendor', 'goog.functions', 'goog.html.SafeHtml', 'goog.style', 'goog.userAgent'], {});
goog.addDependency('style/transition_test.js', ['goog.style.transitionTest'], ['goog.style', 'goog.style.transition', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('test_module.js', ['goog.test_module'], ['goog.test_module_dep'], {'module': 'goog'});
goog.addDependency('test_module_dep.js', ['goog.test_module_dep'], [], {'module': 'goog'});
goog.addDependency('testing/assertionfailure.js', ['goog.testing.safe.assertionFailure'], ['goog.asserts', 'goog.testing.asserts'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/asserts.js', ['goog.testing.asserts'], ['goog.testing.JsUnitException'], {});
goog.addDependency('testing/asserts_test.js', ['goog.testing.assertsTest'], ['goog.Promise', 'goog.array', 'goog.async.Deferred', 'goog.dom', 'goog.iter.Iterator', 'goog.iter.StopIteration', 'goog.structs.Map', 'goog.structs.Set', 'goog.testing.TestCase', 'goog.testing.asserts', 'goog.testing.testSuite', 'goog.userAgent', 'goog.userAgent.product'], {'lang': 'es8', 'module': 'goog'});
goog.addDependency('testing/async/mockcontrol.js', ['goog.testing.async.MockControl'], ['goog.asserts', 'goog.async.Deferred', 'goog.debug', 'goog.testing.MockControl', 'goog.testing.asserts', 'goog.testing.mockmatchers.IgnoreArgument'], {});
goog.addDependency('testing/async/mockcontrol_test.js', ['goog.testing.async.MockControlTest'], ['goog.async.Deferred', 'goog.testing.MockControl', 'goog.testing.asserts', 'goog.testing.async.MockControl', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/asynctestcase.js', ['goog.testing.AsyncTestCase', 'goog.testing.AsyncTestCase.ControlBreakingException'], ['goog.asserts', 'goog.testing.TestCase', 'goog.testing.asserts'], {});
goog.addDependency('testing/asynctestcase_async_test.js', ['goog.testing.AsyncTestCaseAsyncTest'], ['goog.testing.AsyncTestCase', 'goog.testing.TestCase', 'goog.testing.jsunit'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/asynctestcase_noasync_test.js', ['goog.testing.AsyncTestCaseSyncTest'], ['goog.testing.AsyncTestCase', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/asynctestcase_test.js', ['goog.testing.AsyncTestCaseTest'], ['goog.debug.Error', 'goog.testing.AsyncTestCase', 'goog.testing.asserts', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/benchmark.js', ['goog.testing.benchmark'], ['goog.dom', 'goog.dom.TagName', 'goog.testing.PerformanceTable', 'goog.testing.PerformanceTimer', 'goog.testing.TestCase'], {});
goog.addDependency('testing/continuationtestcase.js', ['goog.testing.ContinuationTestCase', 'goog.testing.ContinuationTestCase.ContinuationTest', 'goog.testing.ContinuationTestCase.Step'], ['goog.array', 'goog.events.EventHandler', 'goog.events.EventTarget', 'goog.testing.TestCase', 'goog.testing.asserts'], {});
goog.addDependency('testing/continuationtestcase_test.js', ['goog.testing.ContinuationTestCaseTest'], ['goog.events', 'goog.events.EventTarget', 'goog.testing.ContinuationTestCase', 'goog.testing.MockClock', 'goog.testing.PropertyReplacer', 'goog.testing.TestCase', 'goog.testing.jsunit'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/deferredtestcase.js', ['goog.testing.DeferredTestCase'], ['goog.async.Deferred', 'goog.testing.AsyncTestCase', 'goog.testing.TestCase'], {});
goog.addDependency('testing/deferredtestcase_test.js', ['goog.testing.DeferredTestCaseTest'], ['goog.async.Deferred', 'goog.testing.DeferredTestCase', 'goog.testing.TestCase', 'goog.testing.TestRunner', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/dom.js', ['goog.testing.dom'], ['goog.array', 'goog.asserts', 'goog.dom', 'goog.dom.AbstractRange', 'goog.dom.InputType', 'goog.dom.NodeIterator', 'goog.dom.NodeType', 'goog.dom.TagIterator', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.iter', 'goog.object', 'goog.string', 'goog.style', 'goog.testing.asserts', 'goog.userAgent'], {});
goog.addDependency('testing/dom_test.js', ['goog.testing.domTest'], ['goog.dom', 'goog.dom.TagName', 'goog.testing.dom', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/editor/dom.js', ['goog.testing.editor.dom'], ['goog.dom.AbstractRange', 'goog.dom.NodeType', 'goog.dom.TagIterator', 'goog.dom.TagWalkType', 'goog.iter', 'goog.string', 'goog.testing.asserts'], {});
goog.addDependency('testing/editor/dom_test.js', ['goog.testing.editor.domTest'], ['goog.dom', 'goog.dom.TagName', 'goog.functions', 'goog.testing.editor.dom', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/editor/fieldmock.js', ['goog.testing.editor.FieldMock'], ['goog.dom', 'goog.dom.Range', 'goog.editor.Field', 'goog.testing.LooseMock', 'goog.testing.mockmatchers'], {});
goog.addDependency('testing/editor/testhelper.js', ['goog.testing.editor.TestHelper'], ['goog.Disposable', 'goog.dom', 'goog.dom.Range', 'goog.editor.BrowserFeature', 'goog.editor.node', 'goog.editor.plugins.AbstractBubblePlugin', 'goog.testing.dom'], {});
goog.addDependency('testing/editor/testhelper_test.js', ['goog.testing.editor.TestHelperTest'], ['goog.dom', 'goog.dom.TagName', 'goog.editor.node', 'goog.testing.editor.TestHelper', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/events/eventobserver.js', ['goog.testing.events.EventObserver'], ['goog.array', 'goog.events.Event'], {});
goog.addDependency('testing/events/eventobserver_test.js', ['goog.testing.events.EventObserverTest'], ['goog.array', 'goog.events', 'goog.events.Event', 'goog.events.EventTarget', 'goog.testing.events.EventObserver', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/events/events.js', ['goog.testing.events', 'goog.testing.events.Event'], ['goog.Disposable', 'goog.asserts', 'goog.dom.NodeType', 'goog.events', 'goog.events.BrowserEvent', 'goog.events.BrowserFeature', 'goog.events.EventTarget', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.object', 'goog.style', 'goog.userAgent'], {});
goog.addDependency('testing/events/events_test.js', ['goog.testing.eventsTest'], ['goog.array', 'goog.dom', 'goog.dom.InputType', 'goog.dom.TagName', 'goog.events', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.math.Coordinate', 'goog.string', 'goog.style', 'goog.testing.PropertyReplacer', 'goog.testing.events', 'goog.testing.recordFunction', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/events/matchers.js', ['goog.testing.events.EventMatcher'], ['goog.events.Event', 'goog.testing.mockmatchers.ArgumentMatcher'], {});
goog.addDependency('testing/events/matchers_test.js', ['goog.testing.events.EventMatcherTest'], ['goog.events.Event', 'goog.testing.events.EventMatcher', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/events/onlinehandler.js', ['goog.testing.events.OnlineHandler'], ['goog.events.EventTarget', 'goog.net.NetworkStatusMonitor'], {});
goog.addDependency('testing/events/onlinehandler_test.js', ['goog.testing.events.OnlineHandlerTest'], ['goog.events', 'goog.net.NetworkStatusMonitor', 'goog.testing.events.EventObserver', 'goog.testing.events.OnlineHandler', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/expectedfailures.js', ['goog.testing.ExpectedFailures'], ['goog.asserts', 'goog.debug.DivConsole', 'goog.dom', 'goog.dom.TagName', 'goog.events', 'goog.events.EventType', 'goog.log', 'goog.style', 'goog.testing.JsUnitException', 'goog.testing.TestCase', 'goog.testing.asserts'], {});
goog.addDependency('testing/expectedfailures_test.js', ['goog.testing.ExpectedFailuresTest'], ['goog.debug.Logger', 'goog.testing.ExpectedFailures', 'goog.testing.JsUnitException', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/fs/blob.js', ['goog.testing.fs.Blob'], ['goog.crypt', 'goog.crypt.base64'], {});
goog.addDependency('testing/fs/blob_test.js', ['goog.testing.fs.BlobTest'], ['goog.dom', 'goog.testing.fs.Blob', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/fs/directoryentry_test.js', ['goog.testing.fs.DirectoryEntryTest'], ['goog.array', 'goog.fs.DirectoryEntry', 'goog.fs.Error', 'goog.testing.MockClock', 'goog.testing.TestCase', 'goog.testing.fs.FileSystem', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/fs/entry.js', ['goog.testing.fs.DirectoryEntry', 'goog.testing.fs.Entry', 'goog.testing.fs.FileEntry'], ['goog.Timer', 'goog.array', 'goog.asserts', 'goog.async.Deferred', 'goog.fs.DirectoryEntry', 'goog.fs.DirectoryEntryImpl', 'goog.fs.Entry', 'goog.fs.Error', 'goog.fs.FileEntry', 'goog.functions', 'goog.object', 'goog.string', 'goog.testing.fs.File', 'goog.testing.fs.FileWriter'], {});
goog.addDependency('testing/fs/entry_test.js', ['goog.testing.fs.EntryTest'], ['goog.fs.DirectoryEntry', 'goog.fs.Error', 'goog.testing.MockClock', 'goog.testing.TestCase', 'goog.testing.fs.FileSystem', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/fs/file.js', ['goog.testing.fs.File'], ['goog.testing.fs.Blob'], {});
goog.addDependency('testing/fs/fileentry_test.js', ['goog.testing.fs.FileEntryTest'], ['goog.testing.MockClock', 'goog.testing.fs.FileEntry', 'goog.testing.fs.FileSystem', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/fs/filereader.js', ['goog.testing.fs.FileReader'], ['goog.Timer', 'goog.events.EventTarget', 'goog.fs.Error', 'goog.fs.FileReader', 'goog.testing.fs.Blob', 'goog.testing.fs.ProgressEvent'], {});
goog.addDependency('testing/fs/filereader_test.js', ['goog.testing.fs.FileReaderTest'], ['goog.Promise', 'goog.array', 'goog.events', 'goog.fs.Error', 'goog.fs.FileReader', 'goog.object', 'goog.testing.events.EventObserver', 'goog.testing.fs.FileReader', 'goog.testing.fs.FileSystem', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/fs/filesystem.js', ['goog.testing.fs.FileSystem'], ['goog.fs.FileSystem', 'goog.testing.fs.DirectoryEntry'], {});
goog.addDependency('testing/fs/filewriter.js', ['goog.testing.fs.FileWriter'], ['goog.Timer', 'goog.events.EventTarget', 'goog.fs.Error', 'goog.fs.FileSaver', 'goog.string', 'goog.testing.fs.Blob', 'goog.testing.fs.File', 'goog.testing.fs.ProgressEvent'], {});
goog.addDependency('testing/fs/filewriter_test.js', ['goog.testing.fs.FileWriterTest'], ['goog.Promise', 'goog.array', 'goog.events', 'goog.fs.Error', 'goog.fs.FileSaver', 'goog.object', 'goog.testing.MockClock', 'goog.testing.events.EventObserver', 'goog.testing.fs.Blob', 'goog.testing.fs.FileSystem', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/fs/fs.js', ['goog.testing.fs'], ['goog.Timer', 'goog.array', 'goog.async.Deferred', 'goog.fs', 'goog.testing.PropertyReplacer', 'goog.testing.fs.Blob', 'goog.testing.fs.FileSystem'], {});
goog.addDependency('testing/fs/fs_test.js', ['goog.testing.fsTest'], ['goog.testing.fs', 'goog.testing.fs.Blob', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/fs/integration_test.js', ['goog.testing.fs.integrationTest'], ['goog.Promise', 'goog.events', 'goog.fs', 'goog.fs.DirectoryEntry', 'goog.fs.Error', 'goog.fs.FileSaver', 'goog.testing.PropertyReplacer', 'goog.testing.fs', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/fs/progressevent.js', ['goog.testing.fs.ProgressEvent'], ['goog.events.Event'], {});
goog.addDependency('testing/functionmock.js', ['goog.testing', 'goog.testing.FunctionMock', 'goog.testing.GlobalFunctionMock', 'goog.testing.MethodMock'], ['goog.object', 'goog.testing.LooseMock', 'goog.testing.Mock', 'goog.testing.PropertyReplacer', 'goog.testing.StrictMock'], {});
goog.addDependency('testing/functionmock_test.js', ['goog.testing.FunctionMockTest'], ['goog.array', 'goog.string', 'goog.testing', 'goog.testing.FunctionMock', 'goog.testing.Mock', 'goog.testing.StrictMock', 'goog.testing.asserts', 'goog.testing.mockmatchers', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/graphics.js', ['goog.testing.graphics'], ['goog.graphics.Path', 'goog.testing.asserts'], {});
goog.addDependency('testing/i18n/asserts.js', ['goog.testing.i18n.asserts'], ['goog.testing.jsunit'], {'lang': 'es6'});
goog.addDependency('testing/i18n/asserts_test.js', ['goog.testing.i18n.assertsTest'], ['goog.testing.ExpectedFailures', 'goog.testing.i18n.asserts', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/jstdasyncwrapper.js', ['goog.testing.JsTdAsyncWrapper'], ['goog.Promise'], {});
goog.addDependency('testing/jstdtestcaseadapter.js', ['goog.testing.JsTdTestCaseAdapter'], ['goog.async.run', 'goog.functions', 'goog.testing.JsTdAsyncWrapper', 'goog.testing.TestCase', 'goog.testing.jsunit'], {});
goog.addDependency('testing/jsunit.js', ['goog.testing.jsunit'], ['goog.dom.TagName', 'goog.testing.TestCase', 'goog.testing.TestRunner', 'goog.userAgent'], {});
goog.addDependency('testing/jsunitexception.js', ['goog.testing.JsUnitException'], ['goog.testing.stacktrace'], {});
goog.addDependency('testing/loosemock.js', ['goog.testing.LooseExpectationCollection', 'goog.testing.LooseMock'], ['goog.array', 'goog.asserts', 'goog.structs.Map', 'goog.structs.Set', 'goog.testing.Mock'], {});
goog.addDependency('testing/loosemock_test.js', ['goog.testing.LooseMockTest'], ['goog.testing.LooseMock', 'goog.testing.mockmatchers', 'goog.testing.testSuite'], {'lang': 'es8', 'module': 'goog'});
goog.addDependency('testing/messaging/mockmessagechannel.js', ['goog.testing.messaging.MockMessageChannel'], ['goog.messaging.AbstractChannel', 'goog.testing.MockControl', 'goog.testing.asserts'], {});
goog.addDependency('testing/messaging/mockmessageevent.js', ['goog.testing.messaging.MockMessageEvent'], ['goog.events.BrowserEvent', 'goog.events.EventType', 'goog.testing.events.Event'], {});
goog.addDependency('testing/messaging/mockmessageport.js', ['goog.testing.messaging.MockMessagePort'], ['goog.events.EventTarget', 'goog.testing.MockControl'], {});
goog.addDependency('testing/messaging/mockportnetwork.js', ['goog.testing.messaging.MockPortNetwork'], ['goog.messaging.PortNetwork', 'goog.testing.messaging.MockMessageChannel'], {});
goog.addDependency('testing/mock.js', ['goog.testing.Mock', 'goog.testing.MockExpectation'], ['goog.Promise', 'goog.array', 'goog.asserts', 'goog.object', 'goog.promise.Resolver', 'goog.testing.JsUnitException', 'goog.testing.MockInterface', 'goog.testing.mockmatchers'], {});
goog.addDependency('testing/mock_test.js', ['goog.testing.MockTest'], ['goog.array', 'goog.testing', 'goog.testing.Mock', 'goog.testing.MockControl', 'goog.testing.MockExpectation', 'goog.testing.testSuite'], {'lang': 'es8', 'module': 'goog'});
goog.addDependency('testing/mockclassfactory.js', ['goog.testing.MockClassFactory', 'goog.testing.MockClassRecord'], ['goog.array', 'goog.object', 'goog.testing.LooseMock', 'goog.testing.StrictMock', 'goog.testing.TestCase', 'goog.testing.mockmatchers'], {});
goog.addDependency('testing/mockclassfactory_test.js', ['fake.BaseClass', 'fake.ChildClass', 'goog.testing.MockClassFactoryTest'], ['goog.testing', 'goog.testing.MockClassFactory', 'goog.testing.jsunit'], {'lang': 'es6'});
goog.addDependency('testing/mockclock.js', ['goog.testing.MockClock'], ['goog.Disposable', 'goog.Promise', 'goog.Thenable', 'goog.async.run', 'goog.testing.PropertyReplacer', 'goog.testing.events', 'goog.testing.events.Event'], {});
goog.addDependency('testing/mockclock_test.js', ['goog.testing.MockClockTest'], ['goog.Promise', 'goog.Timer', 'goog.events', 'goog.functions', 'goog.testing.MockClock', 'goog.testing.PropertyReplacer', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/mockcontrol.js', ['goog.testing.MockControl'], ['goog.Promise', 'goog.array', 'goog.testing', 'goog.testing.LooseMock', 'goog.testing.StrictMock'], {});
goog.addDependency('testing/mockcontrol_test.js', ['goog.testing.MockControlTest'], ['goog.testing.Mock', 'goog.testing.MockControl', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/mockinterface.js', ['goog.testing.MockInterface'], ['goog.Promise'], {});
goog.addDependency('testing/mockmatchers.js', ['goog.testing.mockmatchers', 'goog.testing.mockmatchers.ArgumentMatcher', 'goog.testing.mockmatchers.IgnoreArgument', 'goog.testing.mockmatchers.InstanceOf', 'goog.testing.mockmatchers.ObjectEquals', 'goog.testing.mockmatchers.RegexpMatch', 'goog.testing.mockmatchers.SaveArgument', 'goog.testing.mockmatchers.TypeOf'], ['goog.array', 'goog.dom', 'goog.testing.asserts'], {});
goog.addDependency('testing/mockmatchers_test.js', ['goog.testing.mockmatchersTest'], ['goog.dom', 'goog.dom.TagName', 'goog.testing.jsunit', 'goog.testing.mockmatchers', 'goog.testing.mockmatchers.ArgumentMatcher'], {'lang': 'es6'});
goog.addDependency('testing/mockrandom.js', ['goog.testing.MockRandom'], ['goog.Disposable'], {});
goog.addDependency('testing/mockrandom_test.js', ['goog.testing.MockRandomTest'], ['goog.testing.MockRandom', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/mockrange.js', ['goog.testing.MockRange'], ['goog.dom.AbstractRange', 'goog.testing.LooseMock'], {});
goog.addDependency('testing/mockrange_test.js', ['goog.testing.MockRangeTest'], ['goog.testing.MockRange', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/mockstorage.js', ['goog.testing.MockStorage'], ['goog.structs.Map'], {});
goog.addDependency('testing/mockstorage_test.js', ['goog.testing.MockStorageTest'], ['goog.testing.MockStorage', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/mockuseragent.js', ['goog.testing.MockUserAgent'], ['goog.Disposable', 'goog.labs.userAgent.util', 'goog.testing.PropertyReplacer', 'goog.userAgent'], {});
goog.addDependency('testing/mockuseragent_test.js', ['goog.testing.MockUserAgentTest'], ['goog.dispose', 'goog.testing.MockUserAgent', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/multitestrunner.js', ['goog.testing.MultiTestRunner', 'goog.testing.MultiTestRunner.TestFrame'], ['goog.Timer', 'goog.array', 'goog.asserts', 'goog.dom', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.events.EventHandler', 'goog.functions', 'goog.object', 'goog.string', 'goog.testing.TestCase', 'goog.ui.Component', 'goog.ui.ServerChart', 'goog.ui.TableSorter'], {});
goog.addDependency('testing/multitestrunner_test.js', ['goog.testing.MultiTestRunnerTest'], ['goog.Promise', 'goog.array', 'goog.events', 'goog.testing.MockControl', 'goog.testing.MultiTestRunner', 'goog.testing.PropertyReplacer', 'goog.testing.TestCase', 'goog.testing.asserts', 'goog.testing.events', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/net/mockiframeio.js', ['goog.testing.net.MockIFrameIo'], ['goog.events.EventTarget', 'goog.net.ErrorCode', 'goog.net.EventType', 'goog.net.IframeIo', 'goog.testing.TestQueue'], {});
goog.addDependency('testing/net/xhrio.js', ['goog.testing.net.XhrIo'], ['goog.Uri', 'goog.array', 'goog.dom.xml', 'goog.events', 'goog.net.ErrorCode', 'goog.net.EventType', 'goog.net.HttpStatus', 'goog.net.XhrIo', 'goog.net.XmlHttp', 'goog.object', 'goog.structs', 'goog.structs.Map', 'goog.testing.TestQueue', 'goog.uri.utils'], {});
goog.addDependency('testing/net/xhrio_test.js', ['goog.testing.net.XhrIoTest'], ['goog.dom.xml', 'goog.events', 'goog.events.Event', 'goog.net.ErrorCode', 'goog.net.EventType', 'goog.net.XmlHttp', 'goog.object', 'goog.testing.MockControl', 'goog.testing.asserts', 'goog.testing.mockmatchers.InstanceOf', 'goog.testing.net.XhrIo', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/net/xhriopool.js', ['goog.testing.net.XhrIoPool'], ['goog.net.XhrIoPool', 'goog.testing.net.XhrIo'], {});
goog.addDependency('testing/objectpropertystring.js', ['goog.testing.ObjectPropertyString'], [], {});
goog.addDependency('testing/parallel_closure_test_suite.js', ['goog.testing.parallelClosureTestSuite'], ['goog.Promise', 'goog.asserts', 'goog.events', 'goog.json', 'goog.testing.MultiTestRunner', 'goog.testing.TestCase', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/parallel_closure_test_suite_test.js', ['goog.testing.parallelClosureTestSuiteTest'], ['goog.dom', 'goog.testing.MockControl', 'goog.testing.MultiTestRunner', 'goog.testing.PropertyReplacer', 'goog.testing.TestCase', 'goog.testing.mockmatchers', 'goog.testing.mockmatchers.ArgumentMatcher', 'goog.testing.parallelClosureTestSuite', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/performancetable.js', ['goog.testing.PerformanceTable'], ['goog.asserts', 'goog.dom', 'goog.dom.TagName', 'goog.testing.PerformanceTimer'], {});
goog.addDependency('testing/performancetimer.js', ['goog.testing.PerformanceTimer', 'goog.testing.PerformanceTimer.Task'], ['goog.array', 'goog.async.Deferred', 'goog.math'], {'lang': 'es6'});
goog.addDependency('testing/performancetimer_test.js', ['goog.testing.PerformanceTimerTest'], ['goog.async.Deferred', 'goog.dom', 'goog.math', 'goog.testing.MockClock', 'goog.testing.PerformanceTimer', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/propertyreplacer.js', ['goog.testing.PropertyReplacer'], ['goog.asserts', 'goog.userAgent'], {});
goog.addDependency('testing/propertyreplacer_test.js', ['goog.testing.PropertyReplacerTest'], ['goog.dom', 'goog.dom.TagName', 'goog.testing.PropertyReplacer', 'goog.testing.asserts', 'goog.testing.testSuite', 'goog.userAgent.product', 'goog.userAgent.product.isVersion'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/proto2/proto2.js', ['goog.testing.proto2'], ['goog.proto2.Message', 'goog.proto2.ObjectSerializer', 'goog.testing.asserts'], {});
goog.addDependency('testing/proto2/proto2_test.js', ['goog.testing.proto2Test'], ['goog.testing.proto2', 'goog.testing.testSuite', 'proto2.TestAllTypes'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/pseudorandom.js', ['goog.testing.PseudoRandom'], ['goog.Disposable'], {});
goog.addDependency('testing/pseudorandom_test.js', ['goog.testing.PseudoRandomTest'], ['goog.testing.PseudoRandom', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/recordfunction.js', ['goog.testing.FunctionCall', 'goog.testing.recordConstructor', 'goog.testing.recordFunction'], ['goog.Promise', 'goog.promise.Resolver', 'goog.testing.asserts'], {});
goog.addDependency('testing/recordfunction_test.js', ['goog.testing.recordFunctionTest'], ['goog.functions', 'goog.testing.PropertyReplacer', 'goog.testing.recordConstructor', 'goog.testing.recordFunction', 'goog.testing.testSuite'], {'lang': 'es8', 'module': 'goog'});
goog.addDependency('testing/shardingtestcase.js', ['goog.testing.ShardingTestCase'], ['goog.asserts', 'goog.testing.TestCase'], {});
goog.addDependency('testing/shardingtestcase_test.js', ['goog.testing.ShardingTestCaseTest'], ['goog.testing.ShardingTestCase', 'goog.testing.TestCase', 'goog.testing.asserts', 'goog.testing.jsunit'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/singleton.js', ['goog.testing.singleton'], [], {'lang': 'es6'});
goog.addDependency('testing/singleton_test.js', ['goog.testing.singletonTest'], ['goog.testing.asserts', 'goog.testing.singleton', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/stacktrace.js', ['goog.testing.stacktrace', 'goog.testing.stacktrace.Frame'], [], {'lang': 'es6'});
goog.addDependency('testing/stacktrace_test.js', ['goog.testing.stacktraceTest'], ['goog.functions', 'goog.string', 'goog.testing.ExpectedFailures', 'goog.testing.PropertyReplacer', 'goog.testing.StrictMock', 'goog.testing.asserts', 'goog.testing.stacktrace', 'goog.testing.stacktrace.Frame', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/storage/fakemechanism.js', ['goog.testing.storage.FakeMechanism'], ['goog.storage.mechanism.IterableMechanism', 'goog.structs.Map'], {});
goog.addDependency('testing/strictmock.js', ['goog.testing.StrictMock'], ['goog.array', 'goog.asserts', 'goog.structs.Set', 'goog.testing.Mock'], {});
goog.addDependency('testing/strictmock_test.js', ['goog.testing.StrictMockTest'], ['goog.testing.StrictMock', 'goog.testing.testSuite'], {'lang': 'es8', 'module': 'goog'});
goog.addDependency('testing/style/layoutasserts.js', ['goog.testing.style.layoutasserts'], ['goog.style', 'goog.testing.asserts', 'goog.testing.style'], {});
goog.addDependency('testing/style/layoutasserts_test.js', ['goog.testing.style.layoutassertsTest'], ['goog.dom', 'goog.dom.TagName', 'goog.style', 'goog.testing.style.layoutasserts', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/style/style.js', ['goog.testing.style'], ['goog.dom', 'goog.math.Rect', 'goog.style'], {});
goog.addDependency('testing/style/style_test.js', ['goog.testing.styleTest'], ['goog.dom', 'goog.dom.TagName', 'goog.style', 'goog.testing.style', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/testcase.js', ['goog.testing.TestCase', 'goog.testing.TestCase.Error', 'goog.testing.TestCase.Order', 'goog.testing.TestCase.Result', 'goog.testing.TestCase.Test'], ['goog.Promise', 'goog.Thenable', 'goog.array', 'goog.asserts', 'goog.dom', 'goog.dom.TagName', 'goog.object', 'goog.testing.JsUnitException', 'goog.testing.asserts'], {});
goog.addDependency('testing/testcase_test.js', ['goog.testing.TestCaseTest'], ['goog.Promise', 'goog.Timer', 'goog.functions', 'goog.string', 'goog.testing.ExpectedFailures', 'goog.testing.FunctionMock', 'goog.testing.JsUnitException', 'goog.testing.MethodMock', 'goog.testing.MockRandom', 'goog.testing.PropertyReplacer', 'goog.testing.TestCase', 'goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es8', 'module': 'goog'});
goog.addDependency('testing/testqueue.js', ['goog.testing.TestQueue'], [], {});
goog.addDependency('testing/testrunner.js', ['goog.testing.TestRunner'], ['goog.dom', 'goog.dom.TagName', 'goog.dom.safe', 'goog.json', 'goog.testing.TestCase', 'goog.userAgent'], {});
goog.addDependency('testing/testrunner_test.js', ['goog.testing.TestRunnerTest'], ['goog.testing.TestCase', 'goog.testing.TestRunner', 'goog.testing.asserts', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/testsuite.js', ['goog.testing.testSuite'], ['goog.labs.testing.Environment', 'goog.testing.TestCase'], {});
goog.addDependency('testing/testsuite_test.js', ['goog.testing.testSuiteTest'], ['goog.testing.TestCase', 'goog.testing.asserts', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/ui/rendererasserts.js', ['goog.testing.ui.rendererasserts'], ['goog.testing.asserts', 'goog.ui.ControlRenderer'], {});
goog.addDependency('testing/ui/rendererasserts_test.js', ['goog.testing.ui.rendererassertsTest'], ['goog.testing.asserts', 'goog.testing.testSuite', 'goog.testing.ui.rendererasserts', 'goog.ui.ControlRenderer'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('testing/ui/rendererharness.js', ['goog.testing.ui.RendererHarness'], ['goog.Disposable', 'goog.dom.NodeType', 'goog.testing.asserts', 'goog.testing.dom', 'goog.ui.Control', 'goog.ui.ControlRenderer'], {});
goog.addDependency('testing/ui/style.js', ['goog.testing.ui.style'], ['goog.array', 'goog.asserts', 'goog.dom', 'goog.dom.classlist', 'goog.testing.asserts'], {});
goog.addDependency('testing/ui/style_test.js', ['goog.testing.ui.styleTest'], ['goog.dom', 'goog.testing.testSuite', 'goog.testing.ui.style'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('timer/timer.js', ['goog.Timer'], ['goog.Promise', 'goog.events.EventTarget'], {});
goog.addDependency('timer/timer_test.js', ['goog.TimerTest'], ['goog.Promise', 'goog.Timer', 'goog.events', 'goog.testing.MockClock', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('tweak/entries.js', ['goog.tweak.BaseEntry', 'goog.tweak.BasePrimitiveSetting', 'goog.tweak.BaseSetting', 'goog.tweak.BooleanGroup', 'goog.tweak.BooleanInGroupSetting', 'goog.tweak.BooleanSetting', 'goog.tweak.ButtonAction', 'goog.tweak.NumericSetting', 'goog.tweak.StringSetting'], ['goog.array', 'goog.asserts', 'goog.log', 'goog.object'], {});
goog.addDependency('tweak/entries_test.js', ['goog.tweak.BaseEntryTest'], ['goog.testing.MockControl', 'goog.testing.testSuite', 'goog.tweak.testhelpers'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('tweak/registry.js', ['goog.tweak.Registry'], ['goog.array', 'goog.asserts', 'goog.log', 'goog.string', 'goog.tweak.BasePrimitiveSetting', 'goog.tweak.BaseSetting', 'goog.tweak.BooleanSetting', 'goog.tweak.NumericSetting', 'goog.tweak.StringSetting', 'goog.uri.utils'], {});
goog.addDependency('tweak/registry_test.js', ['goog.tweak.RegistryTest'], ['goog.asserts.AssertionError', 'goog.testing.testSuite', 'goog.tweak', 'goog.tweak.testhelpers'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('tweak/testhelpers.js', ['goog.tweak.testhelpers'], ['goog.tweak', 'goog.tweak.BooleanGroup', 'goog.tweak.BooleanInGroupSetting', 'goog.tweak.BooleanSetting', 'goog.tweak.ButtonAction', 'goog.tweak.NumericSetting', 'goog.tweak.Registry', 'goog.tweak.StringSetting'], {});
goog.addDependency('tweak/tweak.js', ['goog.tweak', 'goog.tweak.ConfigParams'], ['goog.asserts', 'goog.tweak.BaseSetting', 'goog.tweak.BooleanGroup', 'goog.tweak.BooleanInGroupSetting', 'goog.tweak.BooleanSetting', 'goog.tweak.ButtonAction', 'goog.tweak.NumericSetting', 'goog.tweak.Registry', 'goog.tweak.StringSetting'], {});
goog.addDependency('tweak/tweakui.js', ['goog.tweak.EntriesPanel', 'goog.tweak.TweakUi'], ['goog.array', 'goog.asserts', 'goog.dom', 'goog.dom.TagName', 'goog.dom.safe', 'goog.html.SafeHtml', 'goog.html.SafeStyleSheet', 'goog.object', 'goog.string.Const', 'goog.style', 'goog.tweak', 'goog.tweak.BaseEntry', 'goog.tweak.BooleanGroup', 'goog.tweak.BooleanInGroupSetting', 'goog.tweak.BooleanSetting', 'goog.tweak.ButtonAction', 'goog.tweak.NumericSetting', 'goog.tweak.StringSetting', 'goog.ui.Zippy', 'goog.userAgent'], {});
goog.addDependency('tweak/tweakui_test.js', ['goog.tweak.TweakUiTest'], ['goog.dom', 'goog.dom.TagName', 'goog.string', 'goog.testing.testSuite', 'goog.tweak', 'goog.tweak.TweakUi', 'goog.tweak.testhelpers'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/abstractspellchecker.js', ['goog.ui.AbstractSpellChecker', 'goog.ui.AbstractSpellChecker.AsyncResult'], ['goog.a11y.aria', 'goog.array', 'goog.asserts', 'goog.dom', 'goog.dom.InputType', 'goog.dom.NodeType', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.dom.selection', 'goog.events', 'goog.events.Event', 'goog.events.EventType', 'goog.math.Coordinate', 'goog.spell.SpellCheck', 'goog.structs.Set', 'goog.style', 'goog.ui.Component', 'goog.ui.MenuItem', 'goog.ui.MenuSeparator', 'goog.ui.PopupMenu'], {});
goog.addDependency('ui/ac/ac.js', ['goog.ui.ac'], ['goog.ui.ac.ArrayMatcher', 'goog.ui.ac.AutoComplete', 'goog.ui.ac.InputHandler', 'goog.ui.ac.Renderer'], {});
goog.addDependency('ui/ac/ac_test.js', ['goog.ui.acTest'], ['goog.asserts', 'goog.dom', 'goog.dom.NodeType', 'goog.dom.classlist', 'goog.dom.selection', 'goog.events', 'goog.events.BrowserEvent', 'goog.events.Event', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.style', 'goog.testing.MockClock', 'goog.testing.events', 'goog.testing.testSuite', 'goog.ui.ac', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/ac/arraymatcher.js', ['goog.ui.ac.ArrayMatcher'], ['goog.string'], {});
goog.addDependency('ui/ac/arraymatcher_test.js', ['goog.ui.ac.ArrayMatcherTest'], ['goog.testing.testSuite', 'goog.ui.ac.ArrayMatcher'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/ac/autocomplete.js', ['goog.ui.ac.AutoComplete', 'goog.ui.ac.AutoComplete.EventType'], ['goog.array', 'goog.asserts', 'goog.events', 'goog.events.EventTarget', 'goog.object', 'goog.ui.ac.RenderOptions'], {});
goog.addDependency('ui/ac/autocomplete_test.js', ['goog.ui.ac.AutoCompleteTest'], ['goog.a11y.aria', 'goog.a11y.aria.Role', 'goog.dom', 'goog.dom.InputType', 'goog.dom.TagName', 'goog.events.EventHandler', 'goog.events.EventTarget', 'goog.string', 'goog.testing.MockControl', 'goog.testing.events', 'goog.testing.mockmatchers', 'goog.testing.testSuite', 'goog.ui.ac.AutoComplete', 'goog.ui.ac.InputHandler', 'goog.ui.ac.RenderOptions', 'goog.ui.ac.Renderer'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/ac/cachingmatcher.js', ['goog.ui.ac.CachingMatcher'], ['goog.array', 'goog.async.Throttle', 'goog.ui.ac.ArrayMatcher', 'goog.ui.ac.RenderOptions'], {});
goog.addDependency('ui/ac/cachingmatcher_test.js', ['goog.ui.ac.CachingMatcherTest'], ['goog.testing.MockControl', 'goog.testing.mockmatchers', 'goog.testing.testSuite', 'goog.ui.ac.CachingMatcher'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/ac/inputhandler.js', ['goog.ui.ac.InputHandler'], ['goog.Disposable', 'goog.Timer', 'goog.a11y.aria', 'goog.a11y.aria.Role', 'goog.a11y.aria.State', 'goog.dom', 'goog.dom.selection', 'goog.events.EventHandler', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.events.KeyHandler', 'goog.string', 'goog.userAgent', 'goog.userAgent.product'], {});
goog.addDependency('ui/ac/inputhandler_test.js', ['goog.ui.ac.InputHandlerTest'], ['goog.a11y.aria', 'goog.a11y.aria.Role', 'goog.a11y.aria.State', 'goog.dom', 'goog.dom.TagName', 'goog.dom.selection', 'goog.events.BrowserEvent', 'goog.events.Event', 'goog.events.EventTarget', 'goog.events.KeyCodes', 'goog.events.KeyHandler', 'goog.functions', 'goog.object', 'goog.testing.MockClock', 'goog.testing.testSuite', 'goog.ui.ac.InputHandler', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/ac/remote.js', ['goog.ui.ac.Remote'], ['goog.ui.ac.AutoComplete', 'goog.ui.ac.InputHandler', 'goog.ui.ac.RemoteArrayMatcher', 'goog.ui.ac.Renderer'], {});
goog.addDependency('ui/ac/remotearraymatcher.js', ['goog.ui.ac.RemoteArrayMatcher'], ['goog.Disposable', 'goog.Uri', 'goog.events', 'goog.net.EventType', 'goog.net.XhrIo'], {});
goog.addDependency('ui/ac/remotearraymatcher_test.js', ['goog.ui.ac.RemoteArrayMatcherTest'], ['goog.net.XhrIo', 'goog.testing.MockControl', 'goog.testing.net.XhrIo', 'goog.testing.testSuite', 'goog.ui.ac.RemoteArrayMatcher'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/ac/renderer.js', ['goog.ui.ac.Renderer', 'goog.ui.ac.Renderer.CustomRenderer'], ['goog.a11y.aria', 'goog.a11y.aria.Role', 'goog.a11y.aria.State', 'goog.array', 'goog.asserts', 'goog.dispose', 'goog.dom', 'goog.dom.NodeType', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.events', 'goog.events.EventTarget', 'goog.events.EventType', 'goog.fx.dom.FadeInAndShow', 'goog.fx.dom.FadeOutAndHide', 'goog.positioning', 'goog.positioning.Corner', 'goog.positioning.Overflow', 'goog.string', 'goog.style', 'goog.ui.IdGenerator', 'goog.ui.ac.AutoComplete'], {'lang': 'es6'});
goog.addDependency('ui/ac/renderer_test.js', ['goog.ui.ac.RendererTest'], ['goog.a11y.aria', 'goog.a11y.aria.State', 'goog.dom', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.events', 'goog.fx.dom.FadeInAndShow', 'goog.fx.dom.FadeOutAndHide', 'goog.string', 'goog.style', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite', 'goog.ui.ac.AutoComplete', 'goog.ui.ac.Renderer'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/ac/renderoptions.js', ['goog.ui.ac.RenderOptions'], [], {});
goog.addDependency('ui/ac/richinputhandler.js', ['goog.ui.ac.RichInputHandler'], ['goog.ui.ac.InputHandler'], {});
goog.addDependency('ui/ac/richremote.js', ['goog.ui.ac.RichRemote'], ['goog.ui.ac.AutoComplete', 'goog.ui.ac.Remote', 'goog.ui.ac.Renderer', 'goog.ui.ac.RichInputHandler', 'goog.ui.ac.RichRemoteArrayMatcher'], {});
goog.addDependency('ui/ac/richremotearraymatcher.js', ['goog.ui.ac.RichRemoteArrayMatcher'], ['goog.dom', 'goog.ui.ac.RemoteArrayMatcher'], {});
goog.addDependency('ui/ac/richremotearraymatcher_test.js', ['goog.ui.ac.RichRemoteArrayMatcherTest'], ['goog.net.XhrIo', 'goog.testing.MockControl', 'goog.testing.net.XhrIo', 'goog.testing.testSuite', 'goog.ui.ac.RichRemoteArrayMatcher'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/activitymonitor.js', ['goog.ui.ActivityMonitor'], ['goog.array', 'goog.asserts', 'goog.dom', 'goog.events.EventHandler', 'goog.events.EventTarget', 'goog.events.EventType'], {});
goog.addDependency('ui/activitymonitor_test.js', ['goog.ui.ActivityMonitorTest'], ['goog.dom', 'goog.events', 'goog.events.Event', 'goog.events.EventType', 'goog.testing.MockClock', 'goog.testing.PropertyReplacer', 'goog.testing.events', 'goog.testing.recordFunction', 'goog.testing.testSuite', 'goog.ui.ActivityMonitor'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/advancedtooltip.js', ['goog.ui.AdvancedTooltip'], ['goog.events', 'goog.events.EventType', 'goog.math.Box', 'goog.math.Coordinate', 'goog.style', 'goog.ui.Tooltip', 'goog.userAgent'], {});
goog.addDependency('ui/advancedtooltip_test.js', ['goog.ui.AdvancedTooltipTest'], ['goog.dom', 'goog.dom.TagName', 'goog.events.Event', 'goog.events.EventType', 'goog.math.Box', 'goog.math.Coordinate', 'goog.style', 'goog.testing.MockClock', 'goog.testing.events', 'goog.testing.testSuite', 'goog.ui.AdvancedTooltip', 'goog.ui.Tooltip', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/animatedzippy.js', ['goog.ui.AnimatedZippy'], ['goog.a11y.aria.Role', 'goog.dom', 'goog.dom.TagName', 'goog.events', 'goog.fx.Animation', 'goog.fx.Transition', 'goog.fx.easing', 'goog.ui.Zippy', 'goog.ui.ZippyEvent'], {});
goog.addDependency('ui/animatedzippy_test.js', ['goog.ui.AnimatedZippyTest'], ['goog.a11y.aria', 'goog.a11y.aria.Role', 'goog.a11y.aria.State', 'goog.asserts', 'goog.dom', 'goog.events', 'goog.functions', 'goog.fx.Animation', 'goog.fx.Transition', 'goog.testing.PropertyReplacer', 'goog.testing.asserts', 'goog.testing.testSuite', 'goog.ui.AnimatedZippy', 'goog.ui.Zippy'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/attachablemenu.js', ['goog.ui.AttachableMenu'], ['goog.a11y.aria', 'goog.a11y.aria.State', 'goog.array', 'goog.asserts', 'goog.dom', 'goog.dom.classlist', 'goog.events.Event', 'goog.events.KeyCodes', 'goog.string', 'goog.style', 'goog.ui.ItemEvent', 'goog.ui.MenuBase', 'goog.ui.PopupBase', 'goog.userAgent'], {});
goog.addDependency('ui/bidiinput.js', ['goog.ui.BidiInput'], ['goog.dom', 'goog.dom.InputType', 'goog.dom.TagName', 'goog.events', 'goog.events.InputHandler', 'goog.i18n.bidi', 'goog.ui.Component'], {});
goog.addDependency('ui/bidiinput_test.js', ['goog.ui.BidiInputTest'], ['goog.dom', 'goog.testing.testSuite', 'goog.ui.BidiInput'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/bubble.js', ['goog.ui.Bubble'], ['goog.Timer', 'goog.dom.safe', 'goog.events', 'goog.events.EventType', 'goog.html.SafeHtml', 'goog.math.Box', 'goog.positioning', 'goog.positioning.AbsolutePosition', 'goog.positioning.AnchoredPosition', 'goog.positioning.Corner', 'goog.positioning.CornerBit', 'goog.string.Const', 'goog.style', 'goog.ui.Component', 'goog.ui.Popup'], {});
goog.addDependency('ui/button.js', ['goog.ui.Button', 'goog.ui.Button.Side'], ['goog.events.EventType', 'goog.events.KeyCodes', 'goog.events.KeyHandler', 'goog.ui.ButtonRenderer', 'goog.ui.ButtonSide', 'goog.ui.Component', 'goog.ui.Control', 'goog.ui.NativeButtonRenderer', 'goog.ui.registry'], {});
goog.addDependency('ui/button_test.js', ['goog.ui.ButtonTest'], ['goog.dom', 'goog.dom.classlist', 'goog.events', 'goog.events.Event', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.events.KeyHandler', 'goog.testing.events', 'goog.testing.testSuite', 'goog.ui.Button', 'goog.ui.ButtonRenderer', 'goog.ui.ButtonSide', 'goog.ui.Component', 'goog.ui.NativeButtonRenderer'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/buttonrenderer.js', ['goog.ui.ButtonRenderer'], ['goog.a11y.aria', 'goog.a11y.aria.Role', 'goog.a11y.aria.State', 'goog.asserts', 'goog.ui.ButtonSide', 'goog.ui.Component', 'goog.ui.ControlRenderer'], {});
goog.addDependency('ui/buttonrenderer_test.js', ['goog.ui.ButtonRendererTest'], ['goog.a11y.aria', 'goog.a11y.aria.Role', 'goog.a11y.aria.State', 'goog.dom', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.testing.ExpectedFailures', 'goog.testing.testSuite', 'goog.testing.ui.rendererasserts', 'goog.ui.Button', 'goog.ui.ButtonRenderer', 'goog.ui.ButtonSide', 'goog.ui.Component', 'goog.ui.ControlRenderer'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/buttonside.js', ['goog.ui.ButtonSide'], [], {});
goog.addDependency('ui/charcounter.js', ['goog.ui.CharCounter', 'goog.ui.CharCounter.Display'], ['goog.dom', 'goog.events', 'goog.events.EventTarget', 'goog.events.InputHandler'], {});
goog.addDependency('ui/charcounter_test.js', ['goog.ui.CharCounterTest'], ['goog.dom', 'goog.testing.asserts', 'goog.testing.testSuite', 'goog.ui.CharCounter', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/charpicker.js', ['goog.ui.CharPicker'], ['goog.a11y.aria', 'goog.a11y.aria.State', 'goog.array', 'goog.asserts', 'goog.dom', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.events', 'goog.events.Event', 'goog.events.EventHandler', 'goog.events.EventType', 'goog.events.InputHandler', 'goog.events.KeyCodes', 'goog.events.KeyHandler', 'goog.i18n.CharListDecompressor', 'goog.i18n.CharPickerData', 'goog.i18n.uChar', 'goog.i18n.uChar.NameFetcher', 'goog.structs.Set', 'goog.style', 'goog.ui.Button', 'goog.ui.Component', 'goog.ui.ContainerScroller', 'goog.ui.FlatButtonRenderer', 'goog.ui.HoverCard', 'goog.ui.LabelInput', 'goog.ui.Menu', 'goog.ui.MenuButton', 'goog.ui.MenuItem', 'goog.ui.Tooltip'], {});
goog.addDependency('ui/charpicker_test.js', ['goog.ui.CharPickerTest'], ['goog.a11y.aria', 'goog.a11y.aria.State', 'goog.dispose', 'goog.dom', 'goog.events.Event', 'goog.events.EventType', 'goog.i18n.CharPickerData', 'goog.i18n.uChar.NameFetcher', 'goog.testing.MockControl', 'goog.testing.events', 'goog.testing.mockmatchers', 'goog.testing.testSuite', 'goog.ui.CharPicker', 'goog.ui.FlatButtonRenderer'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/checkbox.js', ['goog.ui.Checkbox', 'goog.ui.Checkbox.State'], ['goog.a11y.aria', 'goog.a11y.aria.State', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.string', 'goog.ui.CheckboxRenderer', 'goog.ui.Component', 'goog.ui.Control', 'goog.ui.registry'], {});
goog.addDependency('ui/checkbox_test.js', ['goog.ui.CheckboxTest'], ['goog.a11y.aria', 'goog.a11y.aria.Role', 'goog.a11y.aria.State', 'goog.dom', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.events', 'goog.events.KeyCodes', 'goog.testing.events', 'goog.testing.testSuite', 'goog.ui.Checkbox', 'goog.ui.CheckboxRenderer', 'goog.ui.Component', 'goog.ui.ControlRenderer', 'goog.ui.decorate'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/checkboxmenuitem.js', ['goog.ui.CheckBoxMenuItem'], ['goog.ui.MenuItem', 'goog.ui.registry'], {});
goog.addDependency('ui/checkboxrenderer.js', ['goog.ui.CheckboxRenderer'], ['goog.a11y.aria', 'goog.a11y.aria.Role', 'goog.a11y.aria.State', 'goog.array', 'goog.asserts', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.object', 'goog.ui.ControlRenderer'], {});
goog.addDependency('ui/colormenubutton.js', ['goog.ui.ColorMenuButton'], ['goog.array', 'goog.object', 'goog.ui.ColorMenuButtonRenderer', 'goog.ui.ColorPalette', 'goog.ui.Component', 'goog.ui.Menu', 'goog.ui.MenuButton', 'goog.ui.registry'], {});
goog.addDependency('ui/colormenubuttonrenderer.js', ['goog.ui.ColorMenuButtonRenderer'], ['goog.asserts', 'goog.color', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.ui.MenuButtonRenderer', 'goog.userAgent'], {});
goog.addDependency('ui/colormenubuttonrenderer_test.js', ['goog.ui.ColorMenuButtonTest'], ['goog.dom', 'goog.dom.TagName', 'goog.testing.testSuite', 'goog.testing.ui.RendererHarness', 'goog.testing.ui.rendererasserts', 'goog.ui.ColorMenuButton', 'goog.ui.ColorMenuButtonRenderer', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/colorpalette.js', ['goog.ui.ColorPalette'], ['goog.array', 'goog.color', 'goog.dom.TagName', 'goog.style', 'goog.ui.Palette', 'goog.ui.PaletteRenderer'], {});
goog.addDependency('ui/colorpalette_test.js', ['goog.ui.ColorPaletteTest'], ['goog.color', 'goog.dom.TagName', 'goog.testing.testSuite', 'goog.ui.ColorPalette'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/colorpicker.js', ['goog.ui.ColorPicker', 'goog.ui.ColorPicker.EventType'], ['goog.ui.ColorPalette', 'goog.ui.Component'], {});
goog.addDependency('ui/combobox.js', ['goog.ui.ComboBox', 'goog.ui.ComboBoxItem'], ['goog.Timer', 'goog.asserts', 'goog.dom', 'goog.dom.InputType', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.events.EventType', 'goog.events.InputHandler', 'goog.events.KeyCodes', 'goog.events.KeyHandler', 'goog.log', 'goog.positioning.Corner', 'goog.positioning.MenuAnchoredPosition', 'goog.string', 'goog.style', 'goog.ui.Component', 'goog.ui.ItemEvent', 'goog.ui.LabelInput', 'goog.ui.Menu', 'goog.ui.MenuItem', 'goog.ui.MenuSeparator', 'goog.ui.registry', 'goog.userAgent'], {});
goog.addDependency('ui/combobox_test.js', ['goog.ui.ComboBoxTest'], ['goog.dom', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.events.KeyCodes', 'goog.testing.MockClock', 'goog.testing.events', 'goog.testing.testSuite', 'goog.ui.ComboBox', 'goog.ui.ComboBoxItem', 'goog.ui.Component', 'goog.ui.ControlRenderer', 'goog.ui.LabelInput', 'goog.ui.Menu', 'goog.ui.MenuItem'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/component.js', ['goog.ui.Component', 'goog.ui.Component.Error', 'goog.ui.Component.EventType', 'goog.ui.Component.State'], ['goog.array', 'goog.asserts', 'goog.dom', 'goog.dom.NodeType', 'goog.dom.TagName', 'goog.events.EventHandler', 'goog.events.EventTarget', 'goog.object', 'goog.style', 'goog.ui.IdGenerator'], {});
goog.addDependency('ui/component_test.js', ['goog.ui.ComponentTest'], ['goog.dom', 'goog.dom.DomHelper', 'goog.dom.NodeType', 'goog.dom.TagName', 'goog.events.EventTarget', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite', 'goog.ui.Component'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/componentutil.js', ['goog.ui.ComponentUtil'], ['goog.events.MouseAsMouseEventType', 'goog.events.MouseEvents', 'goog.events.PointerAsMouseEventType'], {});
goog.addDependency('ui/componentutil_test.js', ['goog.ui.ComponentUtilTest'], ['goog.events.MouseAsMouseEventType', 'goog.events.PointerAsMouseEventType', 'goog.testing.testSuite', 'goog.ui.Component', 'goog.ui.ComponentUtil'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/container.js', ['goog.ui.Container', 'goog.ui.Container.EventType', 'goog.ui.Container.Orientation'], ['goog.a11y.aria', 'goog.a11y.aria.State', 'goog.asserts', 'goog.dom', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.events.KeyHandler', 'goog.object', 'goog.style', 'goog.ui.Component', 'goog.ui.ComponentUtil', 'goog.ui.ContainerRenderer', 'goog.ui.Control'], {});
goog.addDependency('ui/container_test.js', ['goog.ui.ContainerTest'], ['goog.a11y.aria', 'goog.dom', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.events', 'goog.events.BrowserEvent', 'goog.events.Event', 'goog.events.KeyCodes', 'goog.events.KeyEvent', 'goog.events.PointerFallbackEventType', 'goog.testing.events', 'goog.testing.events.Event', 'goog.testing.testSuite', 'goog.ui.Component', 'goog.ui.Container', 'goog.ui.Control'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/containerrenderer.js', ['goog.ui.ContainerRenderer'], ['goog.a11y.aria', 'goog.array', 'goog.asserts', 'goog.dom.NodeType', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.string', 'goog.style', 'goog.ui.registry', 'goog.userAgent'], {});
goog.addDependency('ui/containerrenderer_test.js', ['goog.ui.ContainerRendererTest'], ['goog.dom', 'goog.dom.TagName', 'goog.style', 'goog.testing.ExpectedFailures', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite', 'goog.testing.ui.rendererasserts', 'goog.ui.Container', 'goog.ui.ContainerRenderer', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/containerscroller.js', ['goog.ui.ContainerScroller'], ['goog.Disposable', 'goog.Timer', 'goog.events.EventHandler', 'goog.style', 'goog.ui.Component', 'goog.ui.Container'], {});
goog.addDependency('ui/containerscroller_test.js', ['goog.ui.ContainerScrollerTest'], ['goog.dom', 'goog.testing.MockClock', 'goog.testing.events', 'goog.testing.testSuite', 'goog.ui.Container', 'goog.ui.ContainerScroller'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/control.js', ['goog.ui.Control'], ['goog.Disposable', 'goog.array', 'goog.dom', 'goog.events.BrowserEvent', 'goog.events.Event', 'goog.events.EventHandler', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.events.KeyHandler', 'goog.string', 'goog.ui.Component', 'goog.ui.ComponentUtil', 'goog.ui.ControlContent', 'goog.ui.ControlRenderer', 'goog.ui.registry', 'goog.userAgent'], {});
goog.addDependency('ui/control_test.js', ['goog.ui.ControlTest'], ['goog.a11y.aria', 'goog.a11y.aria.State', 'goog.array', 'goog.dom', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.events', 'goog.events.BrowserEvent', 'goog.events.KeyCodes', 'goog.events.PointerFallbackEventType', 'goog.html.testing', 'goog.object', 'goog.string', 'goog.style', 'goog.testing.ExpectedFailures', 'goog.testing.events', 'goog.testing.events.Event', 'goog.testing.testSuite', 'goog.ui.Component', 'goog.ui.Control', 'goog.ui.ControlRenderer', 'goog.ui.registry', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/controlcontent.js', ['goog.ui.ControlContent'], [], {});
goog.addDependency('ui/controlrenderer.js', ['goog.ui.ControlRenderer'], ['goog.a11y.aria', 'goog.a11y.aria.Role', 'goog.a11y.aria.State', 'goog.array', 'goog.asserts', 'goog.dom', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.object', 'goog.string', 'goog.style', 'goog.ui.Component', 'goog.ui.ControlContent', 'goog.userAgent'], {});
goog.addDependency('ui/controlrenderer_test.js', ['goog.ui.ControlRendererTest'], ['goog.a11y.aria', 'goog.a11y.aria.Role', 'goog.a11y.aria.State', 'goog.dom', 'goog.dom.NodeType', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.object', 'goog.style', 'goog.testing.ExpectedFailures', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite', 'goog.ui.Component', 'goog.ui.Control', 'goog.ui.ControlRenderer', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/cookieeditor.js', ['goog.ui.CookieEditor'], ['goog.asserts', 'goog.dom', 'goog.dom.TagName', 'goog.events.EventType', 'goog.net.cookies', 'goog.string', 'goog.style', 'goog.ui.Component'], {});
goog.addDependency('ui/cookieeditor_test.js', ['goog.ui.CookieEditorTest'], ['goog.dom', 'goog.events.Event', 'goog.events.EventType', 'goog.net.cookies', 'goog.testing.events', 'goog.testing.testSuite', 'goog.ui.CookieEditor'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/css3buttonrenderer.js', ['goog.ui.Css3ButtonRenderer'], ['goog.asserts', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.ui.Button', 'goog.ui.ButtonRenderer', 'goog.ui.Component', 'goog.ui.INLINE_BLOCK_CLASSNAME', 'goog.ui.registry'], {});
goog.addDependency('ui/css3menubuttonrenderer.js', ['goog.ui.Css3MenuButtonRenderer'], ['goog.dom', 'goog.dom.TagName', 'goog.ui.INLINE_BLOCK_CLASSNAME', 'goog.ui.MenuButton', 'goog.ui.MenuButtonRenderer', 'goog.ui.registry'], {});
goog.addDependency('ui/cssnames.js', ['goog.ui.INLINE_BLOCK_CLASSNAME'], [], {});
goog.addDependency('ui/custombutton.js', ['goog.ui.CustomButton'], ['goog.ui.Button', 'goog.ui.CustomButtonRenderer', 'goog.ui.registry'], {});
goog.addDependency('ui/custombuttonrenderer.js', ['goog.ui.CustomButtonRenderer'], ['goog.a11y.aria.Role', 'goog.asserts', 'goog.dom.NodeType', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.string', 'goog.ui.ButtonRenderer', 'goog.ui.INLINE_BLOCK_CLASSNAME'], {});
goog.addDependency('ui/customcolorpalette.js', ['goog.ui.CustomColorPalette'], ['goog.color', 'goog.dom', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.ui.ColorPalette', 'goog.ui.Component'], {});
goog.addDependency('ui/customcolorpalette_test.js', ['goog.ui.CustomColorPaletteTest'], ['goog.dom.TagName', 'goog.dom.classlist', 'goog.testing.testSuite', 'goog.ui.CustomColorPalette'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/datepicker.js', ['goog.ui.DatePicker', 'goog.ui.DatePicker.Events', 'goog.ui.DatePickerEvent'], ['goog.a11y.aria', 'goog.asserts', 'goog.date.Date', 'goog.date.DateRange', 'goog.date.Interval', 'goog.dom', 'goog.dom.NodeType', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.events.Event', 'goog.events.EventType', 'goog.events.KeyHandler', 'goog.i18n.DateTimeFormat', 'goog.i18n.DateTimePatterns', 'goog.i18n.DateTimeSymbols', 'goog.style', 'goog.ui.Component', 'goog.ui.DefaultDatePickerRenderer', 'goog.ui.IdGenerator'], {});
goog.addDependency('ui/datepicker_test.js', ['goog.ui.DatePickerTest'], ['goog.a11y.aria', 'goog.a11y.aria.Role', 'goog.date.Date', 'goog.date.DateRange', 'goog.dom', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.events', 'goog.events.KeyCodes', 'goog.i18n.DateTimeSymbols', 'goog.i18n.DateTimeSymbols_en_US', 'goog.i18n.DateTimeSymbols_zh_HK', 'goog.style', 'goog.testing.events', 'goog.testing.recordFunction', 'goog.testing.testSuite', 'goog.ui.DatePicker'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/datepickerrenderer.js', ['goog.ui.DatePickerRenderer'], [], {});
goog.addDependency('ui/decorate.js', ['goog.ui.decorate'], ['goog.ui.registry'], {});
goog.addDependency('ui/decorate_test.js', ['goog.ui.decorateTest'], ['goog.testing.testSuite', 'goog.ui.decorate', 'goog.ui.registry'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/defaultdatepickerrenderer.js', ['goog.ui.DefaultDatePickerRenderer'], ['goog.dom', 'goog.dom.TagName', 'goog.ui.DatePickerRenderer'], {});
goog.addDependency('ui/dialog.js', ['goog.ui.Dialog', 'goog.ui.Dialog.ButtonSet', 'goog.ui.Dialog.ButtonSet.DefaultButtons', 'goog.ui.Dialog.DefaultButtonCaptions', 'goog.ui.Dialog.DefaultButtonKeys', 'goog.ui.Dialog.Event', 'goog.ui.Dialog.EventType'], ['goog.a11y.aria', 'goog.a11y.aria.Role', 'goog.a11y.aria.State', 'goog.asserts', 'goog.dom', 'goog.dom.NodeType', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.dom.safe', 'goog.events', 'goog.events.Event', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.events.Keys', 'goog.fx.Dragger', 'goog.html.SafeHtml', 'goog.math.Rect', 'goog.string', 'goog.structs.Map', 'goog.style', 'goog.ui.ModalPopup'], {});
goog.addDependency('ui/dialog_test.js', ['goog.ui.DialogTest'], ['goog.a11y.aria', 'goog.a11y.aria.Role', 'goog.a11y.aria.State', 'goog.dom', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.events', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.fx.css3', 'goog.html.SafeHtml', 'goog.html.testing', 'goog.style', 'goog.testing.MockClock', 'goog.testing.events', 'goog.testing.recordFunction', 'goog.testing.testSuite', 'goog.ui.Dialog', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/dimensionpicker.js', ['goog.ui.DimensionPicker'], ['goog.events.BrowserEvent.PointerType', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.math.Size', 'goog.ui.Component', 'goog.ui.ComponentUtil', 'goog.ui.Control', 'goog.ui.DimensionPickerRenderer', 'goog.ui.registry'], {});
goog.addDependency('ui/dimensionpicker_test.js', ['goog.ui.DimensionPickerTest'], ['goog.dom', 'goog.dom.TagName', 'goog.events.BrowserEvent', 'goog.events.KeyCodes', 'goog.math.Size', 'goog.testing.testSuite', 'goog.testing.ui.rendererasserts', 'goog.ui.DimensionPicker', 'goog.ui.DimensionPickerRenderer'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/dimensionpickerrenderer.js', ['goog.ui.DimensionPickerRenderer'], ['goog.a11y.aria.Announcer', 'goog.a11y.aria.LivePriority', 'goog.dom', 'goog.dom.TagName', 'goog.i18n.bidi', 'goog.style', 'goog.ui.ControlRenderer', 'goog.userAgent'], {});
goog.addDependency('ui/dimensionpickerrenderer_test.js', ['goog.ui.DimensionPickerRendererTest'], ['goog.a11y.aria.LivePriority', 'goog.array', 'goog.testing.recordFunction', 'goog.testing.testSuite', 'goog.ui.DimensionPicker', 'goog.ui.DimensionPickerRenderer'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/dragdropdetector.js', ['goog.ui.DragDropDetector', 'goog.ui.DragDropDetector.EventType', 'goog.ui.DragDropDetector.ImageDropEvent', 'goog.ui.DragDropDetector.LinkDropEvent'], ['goog.dom', 'goog.dom.InputType', 'goog.dom.TagName', 'goog.events.Event', 'goog.events.EventHandler', 'goog.events.EventTarget', 'goog.events.EventType', 'goog.math.Coordinate', 'goog.string', 'goog.style', 'goog.userAgent'], {});
goog.addDependency('ui/drilldownrow.js', ['goog.ui.DrilldownRow'], ['goog.asserts', 'goog.dom', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.dom.safe', 'goog.html.SafeHtml', 'goog.string.Unicode', 'goog.ui.Component'], {});
goog.addDependency('ui/drilldownrow_test.js', ['goog.ui.DrilldownRowTest'], ['goog.dom', 'goog.dom.TagName', 'goog.html.SafeHtml', 'goog.testing.testSuite', 'goog.ui.DrilldownRow'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/editor/abstractdialog.js', ['goog.ui.editor.AbstractDialog', 'goog.ui.editor.AbstractDialog.Builder', 'goog.ui.editor.AbstractDialog.EventType'], ['goog.asserts', 'goog.dom', 'goog.dom.classlist', 'goog.events.EventTarget', 'goog.string', 'goog.ui.Dialog', 'goog.ui.PopupBase'], {});
goog.addDependency('ui/editor/abstractdialog_test.js', ['goog.ui.editor.AbstractDialogTest'], ['goog.dom', 'goog.dom.DomHelper', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.events.Event', 'goog.events.EventHandler', 'goog.events.KeyCodes', 'goog.testing.MockControl', 'goog.testing.events', 'goog.testing.mockmatchers.ArgumentMatcher', 'goog.testing.testSuite', 'goog.ui.editor.AbstractDialog', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/editor/bubble.js', ['goog.ui.editor.Bubble'], ['goog.asserts', 'goog.dom', 'goog.dom.TagName', 'goog.dom.ViewportSizeMonitor', 'goog.dom.classlist', 'goog.editor.style', 'goog.events.EventHandler', 'goog.events.EventTarget', 'goog.events.EventType', 'goog.functions', 'goog.log', 'goog.math.Box', 'goog.object', 'goog.positioning', 'goog.positioning.Corner', 'goog.positioning.Overflow', 'goog.positioning.OverflowStatus', 'goog.string', 'goog.style', 'goog.ui.Component', 'goog.ui.PopupBase', 'goog.userAgent'], {});
goog.addDependency('ui/editor/bubble_test.js', ['goog.ui.editor.BubbleTest'], ['goog.dom', 'goog.dom.TagName', 'goog.events', 'goog.events.EventType', 'goog.positioning.Corner', 'goog.positioning.OverflowStatus', 'goog.string', 'goog.style', 'goog.testing.editor.TestHelper', 'goog.testing.events', 'goog.testing.testSuite', 'goog.ui.Component', 'goog.ui.editor.Bubble', 'goog.userAgent.product'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/editor/defaulttoolbar.js', ['goog.ui.editor.ButtonDescriptor', 'goog.ui.editor.DefaultToolbar'], ['goog.asserts', 'goog.dom', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.editor.Command', 'goog.style', 'goog.ui.editor.ToolbarFactory', 'goog.ui.editor.messages', 'goog.userAgent'], {});
goog.addDependency('ui/editor/linkdialog.js', ['goog.ui.editor.LinkDialog', 'goog.ui.editor.LinkDialog.BeforeTestLinkEvent', 'goog.ui.editor.LinkDialog.EventType', 'goog.ui.editor.LinkDialog.OkEvent'], ['goog.a11y.aria', 'goog.a11y.aria.State', 'goog.dom', 'goog.dom.InputType', 'goog.dom.TagName', 'goog.dom.safe', 'goog.editor.BrowserFeature', 'goog.editor.Link', 'goog.editor.focus', 'goog.editor.node', 'goog.events.Event', 'goog.events.EventHandler', 'goog.events.InputHandler', 'goog.html.SafeHtml', 'goog.html.SafeHtmlFormatter', 'goog.string', 'goog.string.Unicode', 'goog.style', 'goog.ui.Button', 'goog.ui.Component', 'goog.ui.LinkButtonRenderer', 'goog.ui.editor.AbstractDialog', 'goog.ui.editor.TabPane', 'goog.ui.editor.messages', 'goog.userAgent', 'goog.window'], {});
goog.addDependency('ui/editor/linkdialog_test.js', ['goog.ui.editor.LinkDialogTest'], ['goog.dom', 'goog.dom.DomHelper', 'goog.dom.TagName', 'goog.editor.BrowserFeature', 'goog.editor.Link', 'goog.events', 'goog.events.EventHandler', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.style', 'goog.testing.MockControl', 'goog.testing.PropertyReplacer', 'goog.testing.dom', 'goog.testing.events', 'goog.testing.events.Event', 'goog.testing.mockmatchers', 'goog.testing.mockmatchers.ArgumentMatcher', 'goog.testing.testSuite', 'goog.ui.editor.AbstractDialog', 'goog.ui.editor.LinkDialog', 'goog.ui.editor.messages', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/editor/messages.js', ['goog.ui.editor.messages'], ['goog.html.SafeHtmlFormatter'], {});
goog.addDependency('ui/editor/tabpane.js', ['goog.ui.editor.TabPane'], ['goog.asserts', 'goog.dom', 'goog.dom.InputType', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.events.EventHandler', 'goog.events.EventType', 'goog.style', 'goog.ui.Component', 'goog.ui.Control', 'goog.ui.Tab', 'goog.ui.TabBar'], {});
goog.addDependency('ui/editor/toolbarcontroller.js', ['goog.ui.editor.ToolbarController'], ['goog.editor.Field', 'goog.events.EventHandler', 'goog.events.EventTarget', 'goog.ui.Component'], {});
goog.addDependency('ui/editor/toolbarfactory.js', ['goog.ui.editor.ToolbarFactory'], ['goog.array', 'goog.dom', 'goog.dom.TagName', 'goog.string', 'goog.string.Unicode', 'goog.style', 'goog.ui.Component', 'goog.ui.Container', 'goog.ui.Option', 'goog.ui.Toolbar', 'goog.ui.ToolbarButton', 'goog.ui.ToolbarColorMenuButton', 'goog.ui.ToolbarMenuButton', 'goog.ui.ToolbarRenderer', 'goog.ui.ToolbarSelect', 'goog.userAgent'], {});
goog.addDependency('ui/editor/toolbarfactory_test.js', ['goog.ui.editor.ToolbarFactoryTest'], ['goog.dom', 'goog.testing.ExpectedFailures', 'goog.testing.editor.TestHelper', 'goog.testing.testSuite', 'goog.ui.editor.ToolbarFactory', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/emoji/emoji.js', ['goog.ui.emoji.Emoji'], [], {});
goog.addDependency('ui/emoji/emojipalette.js', ['goog.ui.emoji.EmojiPalette'], ['goog.events.EventType', 'goog.net.ImageLoader', 'goog.ui.Palette', 'goog.ui.emoji.Emoji', 'goog.ui.emoji.EmojiPaletteRenderer'], {});
goog.addDependency('ui/emoji/emojipaletterenderer.js', ['goog.ui.emoji.EmojiPaletteRenderer'], ['goog.a11y.aria', 'goog.asserts', 'goog.dom.NodeType', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.style', 'goog.ui.PaletteRenderer', 'goog.ui.emoji.Emoji'], {});
goog.addDependency('ui/emoji/emojipicker.js', ['goog.ui.emoji.EmojiPicker'], ['goog.dom.TagName', 'goog.style', 'goog.ui.Component', 'goog.ui.TabPane', 'goog.ui.emoji.Emoji', 'goog.ui.emoji.EmojiPalette', 'goog.ui.emoji.EmojiPaletteRenderer', 'goog.ui.emoji.ProgressiveEmojiPaletteRenderer'], {});
goog.addDependency('ui/emoji/emojipicker_test.js', ['goog.ui.emoji.EmojiPickerTest'], ['goog.dom.TagName', 'goog.dom.classlist', 'goog.events.EventHandler', 'goog.style', 'goog.testing.events', 'goog.testing.testSuite', 'goog.ui.Component', 'goog.ui.emoji.Emoji', 'goog.ui.emoji.EmojiPicker', 'goog.ui.emoji.SpriteInfo'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/emoji/fast_nonprogressive_emojipicker_test.js', ['goog.ui.emoji.FastNonProgressiveEmojiPickerTest'], ['goog.Promise', 'goog.dom.classlist', 'goog.events', 'goog.events.EventType', 'goog.net.EventType', 'goog.style', 'goog.testing.TestCase', 'goog.testing.testSuite', 'goog.ui.emoji.Emoji', 'goog.ui.emoji.EmojiPicker', 'goog.ui.emoji.SpriteInfo'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/emoji/fast_progressive_emojipicker_test.js', ['goog.ui.emoji.FastProgressiveEmojiPickerTest'], ['goog.Promise', 'goog.dom.classlist', 'goog.events', 'goog.events.EventType', 'goog.net.EventType', 'goog.style', 'goog.testing.testSuite', 'goog.ui.emoji.Emoji', 'goog.ui.emoji.EmojiPicker', 'goog.ui.emoji.SpriteInfo'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/emoji/popupemojipicker.js', ['goog.ui.emoji.PopupEmojiPicker'], ['goog.events.EventType', 'goog.positioning.AnchoredPosition', 'goog.positioning.Corner', 'goog.ui.Component', 'goog.ui.Popup', 'goog.ui.emoji.EmojiPicker'], {});
goog.addDependency('ui/emoji/popupemojipicker_test.js', ['goog.ui.emoji.PopupEmojiPickerTest'], ['goog.dom', 'goog.testing.testSuite', 'goog.ui.emoji.PopupEmojiPicker'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/emoji/progressiveemojipaletterenderer.js', ['goog.ui.emoji.ProgressiveEmojiPaletteRenderer'], ['goog.dom.TagName', 'goog.style', 'goog.ui.emoji.EmojiPaletteRenderer'], {});
goog.addDependency('ui/emoji/spriteinfo.js', ['goog.ui.emoji.SpriteInfo'], [], {'lang': 'es6'});
goog.addDependency('ui/emoji/spriteinfo_test.js', ['goog.ui.emoji.SpriteInfoTest'], ['goog.testing.testSuite', 'goog.ui.emoji.SpriteInfo'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/filteredmenu.js', ['goog.ui.FilteredMenu'], ['goog.a11y.aria', 'goog.a11y.aria.AutoCompleteValues', 'goog.a11y.aria.State', 'goog.dom', 'goog.dom.InputType', 'goog.dom.TagName', 'goog.events', 'goog.events.EventType', 'goog.events.InputHandler', 'goog.events.KeyCodes', 'goog.string', 'goog.style', 'goog.ui.Component', 'goog.ui.FilterObservingMenuItem', 'goog.ui.Menu', 'goog.ui.MenuItem', 'goog.userAgent'], {});
goog.addDependency('ui/filteredmenu_test.js', ['goog.ui.FilteredMenuTest'], ['goog.a11y.aria', 'goog.a11y.aria.AutoCompleteValues', 'goog.a11y.aria.State', 'goog.dom', 'goog.dom.TagName', 'goog.events', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.math.Rect', 'goog.style', 'goog.testing.events', 'goog.testing.testSuite', 'goog.ui.FilteredMenu', 'goog.ui.MenuItem'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/filterobservingmenuitem.js', ['goog.ui.FilterObservingMenuItem'], ['goog.ui.FilterObservingMenuItemRenderer', 'goog.ui.MenuItem', 'goog.ui.registry'], {});
goog.addDependency('ui/filterobservingmenuitemrenderer.js', ['goog.ui.FilterObservingMenuItemRenderer'], ['goog.ui.MenuItemRenderer'], {});
goog.addDependency('ui/flatbuttonrenderer.js', ['goog.ui.FlatButtonRenderer'], ['goog.a11y.aria.Role', 'goog.asserts', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.ui.Button', 'goog.ui.ButtonRenderer', 'goog.ui.INLINE_BLOCK_CLASSNAME', 'goog.ui.registry'], {});
goog.addDependency('ui/flatmenubuttonrenderer.js', ['goog.ui.FlatMenuButtonRenderer'], ['goog.dom', 'goog.dom.TagName', 'goog.style', 'goog.ui.FlatButtonRenderer', 'goog.ui.INLINE_BLOCK_CLASSNAME', 'goog.ui.Menu', 'goog.ui.MenuButton', 'goog.ui.MenuRenderer', 'goog.ui.registry'], {});
goog.addDependency('ui/formpost.js', ['goog.ui.FormPost'], ['goog.array', 'goog.dom.InputType', 'goog.dom.TagName', 'goog.dom.safe', 'goog.html.SafeHtml', 'goog.ui.Component'], {});
goog.addDependency('ui/formpost_test.js', ['goog.ui.FormPostTest'], ['goog.array', 'goog.dom', 'goog.dom.TagName', 'goog.object', 'goog.testing.testSuite', 'goog.ui.FormPost', 'goog.userAgent.product', 'goog.userAgent.product.isVersion'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/gauge.js', ['goog.ui.Gauge', 'goog.ui.GaugeColoredRange'], ['goog.a11y.aria', 'goog.asserts', 'goog.dom.TagName', 'goog.events', 'goog.fx.Animation', 'goog.fx.Transition', 'goog.fx.easing', 'goog.graphics', 'goog.graphics.Font', 'goog.graphics.Path', 'goog.graphics.SolidFill', 'goog.math', 'goog.ui.Component', 'goog.ui.GaugeTheme'], {});
goog.addDependency('ui/gaugetheme.js', ['goog.ui.GaugeTheme'], ['goog.graphics.LinearGradient', 'goog.graphics.SolidFill', 'goog.graphics.Stroke'], {});
goog.addDependency('ui/hovercard.js', ['goog.ui.HoverCard', 'goog.ui.HoverCard.EventType', 'goog.ui.HoverCard.TriggerEvent'], ['goog.array', 'goog.dom', 'goog.events', 'goog.events.Event', 'goog.events.EventType', 'goog.ui.AdvancedTooltip', 'goog.ui.PopupBase', 'goog.ui.Tooltip'], {});
goog.addDependency('ui/hovercard_test.js', ['goog.ui.HoverCardTest'], ['goog.dom', 'goog.events', 'goog.math.Coordinate', 'goog.style', 'goog.testing.MockClock', 'goog.testing.events', 'goog.testing.events.Event', 'goog.testing.testSuite', 'goog.ui.HoverCard'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/hsvapalette.js', ['goog.ui.HsvaPalette'], ['goog.array', 'goog.color.alpha', 'goog.dom.TagName', 'goog.events', 'goog.events.EventType', 'goog.style', 'goog.ui.Component', 'goog.ui.HsvPalette'], {});
goog.addDependency('ui/hsvapalette_test.js', ['goog.ui.HsvaPaletteTest'], ['goog.color.alpha', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.events.Event', 'goog.math.Coordinate', 'goog.style', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite', 'goog.ui.HsvaPalette', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/hsvpalette.js', ['goog.ui.HsvPalette'], ['goog.color', 'goog.dom.InputType', 'goog.dom.TagName', 'goog.events', 'goog.events.EventType', 'goog.events.InputHandler', 'goog.style', 'goog.style.bidi', 'goog.ui.Component', 'goog.userAgent'], {});
goog.addDependency('ui/hsvpalette_test.js', ['goog.ui.HsvPaletteTest'], ['goog.color', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.events', 'goog.events.Event', 'goog.math.Coordinate', 'goog.style', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite', 'goog.ui.Component', 'goog.ui.HsvPalette', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/idgenerator.js', ['goog.ui.IdGenerator'], [], {});
goog.addDependency('ui/idletimer.js', ['goog.ui.IdleTimer'], ['goog.Timer', 'goog.events', 'goog.events.EventTarget', 'goog.structs.Set', 'goog.ui.ActivityMonitor'], {});
goog.addDependency('ui/idletimer_test.js', ['goog.ui.IdleTimerTest'], ['goog.events', 'goog.testing.MockClock', 'goog.testing.testSuite', 'goog.ui.IdleTimer', 'goog.ui.MockActivityMonitor'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/iframemask.js', ['goog.ui.IframeMask'], ['goog.Disposable', 'goog.Timer', 'goog.dom', 'goog.dom.iframe', 'goog.events.EventHandler', 'goog.structs.Pool', 'goog.style'], {});
goog.addDependency('ui/iframemask_test.js', ['goog.ui.IframeMaskTest'], ['goog.dom', 'goog.dom.TagName', 'goog.dom.iframe', 'goog.structs.Pool', 'goog.style', 'goog.testing.MockClock', 'goog.testing.StrictMock', 'goog.testing.testSuite', 'goog.ui.IframeMask', 'goog.ui.Popup', 'goog.ui.PopupBase', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/imagelessbuttonrenderer.js', ['goog.ui.ImagelessButtonRenderer'], ['goog.dom.TagName', 'goog.dom.classlist', 'goog.ui.Button', 'goog.ui.Component', 'goog.ui.CustomButtonRenderer', 'goog.ui.INLINE_BLOCK_CLASSNAME', 'goog.ui.registry'], {});
goog.addDependency('ui/imagelessmenubuttonrenderer.js', ['goog.ui.ImagelessMenuButtonRenderer'], ['goog.dom', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.ui.INLINE_BLOCK_CLASSNAME', 'goog.ui.MenuButton', 'goog.ui.MenuButtonRenderer', 'goog.ui.registry'], {});
goog.addDependency('ui/inputdatepicker.js', ['goog.ui.InputDatePicker'], ['goog.date.DateTime', 'goog.dom', 'goog.dom.InputType', 'goog.dom.TagName', 'goog.i18n.DateTimeParse', 'goog.string', 'goog.ui.Component', 'goog.ui.DatePicker', 'goog.ui.LabelInput', 'goog.ui.PopupBase', 'goog.ui.PopupDatePicker'], {});
goog.addDependency('ui/inputdatepicker_test.js', ['goog.ui.InputDatePickerTest'], ['goog.dom', 'goog.i18n.DateTimeFormat', 'goog.i18n.DateTimeParse', 'goog.testing.testSuite', 'goog.ui.InputDatePicker'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/itemevent.js', ['goog.ui.ItemEvent'], ['goog.events.Event'], {});
goog.addDependency('ui/keyboardeventdata.js', ['goog.ui.KeyboardEventData'], ['goog.asserts', 'goog.events.BrowserEvent'], {'lang': 'es6'});
goog.addDependency('ui/keyboardshortcuthandler.js', ['goog.ui.KeyboardShortcutEvent', 'goog.ui.KeyboardShortcutHandler', 'goog.ui.KeyboardShortcutHandler.EventType', 'goog.ui.KeyboardShortcutHandler.Modifiers'], ['goog.array', 'goog.asserts', 'goog.dom.TagName', 'goog.events', 'goog.events.Event', 'goog.events.EventTarget', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.events.KeyNames', 'goog.events.Keys', 'goog.object', 'goog.ui.KeyboardEventData', 'goog.ui.SyntheticKeyboardEvent', 'goog.userAgent'], {});
goog.addDependency('ui/keyboardshortcuthandler_test.js', ['goog.ui.KeyboardShortcutHandlerTest'], ['goog.dom', 'goog.events', 'goog.events.BrowserEvent', 'goog.events.KeyCodes', 'goog.testing.MockClock', 'goog.testing.PropertyReplacer', 'goog.testing.StrictMock', 'goog.testing.events', 'goog.testing.testSuite', 'goog.ui.KeyboardShortcutHandler', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/labelinput.js', ['goog.ui.LabelInput'], ['goog.Timer', 'goog.a11y.aria', 'goog.a11y.aria.State', 'goog.asserts', 'goog.dom', 'goog.dom.InputType', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.events.EventHandler', 'goog.events.EventType', 'goog.ui.Component', 'goog.userAgent'], {});
goog.addDependency('ui/labelinput_test.js', ['goog.ui.LabelInputTest'], ['goog.a11y.aria', 'goog.a11y.aria.State', 'goog.dom', 'goog.dom.classlist', 'goog.events.EventType', 'goog.testing.MockClock', 'goog.testing.events', 'goog.testing.events.Event', 'goog.testing.testSuite', 'goog.ui.LabelInput', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/linkbuttonrenderer.js', ['goog.ui.LinkButtonRenderer'], ['goog.ui.Button', 'goog.ui.FlatButtonRenderer', 'goog.ui.registry'], {});
goog.addDependency('ui/media/flashobject.js', ['goog.ui.media.FlashObject', 'goog.ui.media.FlashObject.ScriptAccessLevel', 'goog.ui.media.FlashObject.Wmodes'], ['goog.asserts', 'goog.dom.TagName', 'goog.dom.safe', 'goog.events.Event', 'goog.events.EventHandler', 'goog.events.EventType', 'goog.html.TrustedResourceUrl', 'goog.html.flash', 'goog.log', 'goog.object', 'goog.string', 'goog.structs.Map', 'goog.style', 'goog.ui.Component', 'goog.userAgent', 'goog.userAgent.flash'], {});
goog.addDependency('ui/media/flashobject_test.js', ['goog.ui.media.FlashObjectTest'], ['goog.dom', 'goog.dom.DomHelper', 'goog.dom.TagName', 'goog.events', 'goog.events.Event', 'goog.events.EventType', 'goog.html.testing', 'goog.testing.MockControl', 'goog.testing.events', 'goog.testing.testSuite', 'goog.ui.media.FlashObject', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/media/flickr.js', ['goog.ui.media.FlickrSet', 'goog.ui.media.FlickrSetModel'], ['goog.html.TrustedResourceUrl', 'goog.string.Const', 'goog.ui.media.FlashObject', 'goog.ui.media.Media', 'goog.ui.media.MediaModel', 'goog.ui.media.MediaRenderer'], {});
goog.addDependency('ui/media/flickr_test.js', ['goog.ui.media.FlickrSetTest'], ['goog.dom', 'goog.dom.TagName', 'goog.html.testing', 'goog.testing.testSuite', 'goog.ui.media.FlashObject', 'goog.ui.media.FlickrSet', 'goog.ui.media.FlickrSetModel', 'goog.ui.media.Media'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/media/googlevideo.js', ['goog.ui.media.GoogleVideo', 'goog.ui.media.GoogleVideoModel'], ['goog.html.TrustedResourceUrl', 'goog.string', 'goog.string.Const', 'goog.ui.media.FlashObject', 'goog.ui.media.Media', 'goog.ui.media.MediaModel', 'goog.ui.media.MediaRenderer'], {});
goog.addDependency('ui/media/googlevideo_test.js', ['goog.ui.media.GoogleVideoTest'], ['goog.dom', 'goog.dom.TagName', 'goog.testing.testSuite', 'goog.ui.media.FlashObject', 'goog.ui.media.GoogleVideo', 'goog.ui.media.GoogleVideoModel', 'goog.ui.media.Media'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/media/media.js', ['goog.ui.media.Media', 'goog.ui.media.MediaRenderer'], ['goog.asserts', 'goog.dom.TagName', 'goog.style', 'goog.ui.Component', 'goog.ui.Control', 'goog.ui.ControlRenderer'], {});
goog.addDependency('ui/media/media_test.js', ['goog.ui.media.MediaTest'], ['goog.dom', 'goog.dom.TagName', 'goog.html.testing', 'goog.math.Size', 'goog.testing.testSuite', 'goog.ui.ControlRenderer', 'goog.ui.media.Media', 'goog.ui.media.MediaModel', 'goog.ui.media.MediaRenderer'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/media/mediamodel.js', ['goog.ui.media.MediaModel', 'goog.ui.media.MediaModel.Category', 'goog.ui.media.MediaModel.Credit', 'goog.ui.media.MediaModel.Credit.Role', 'goog.ui.media.MediaModel.Credit.Scheme', 'goog.ui.media.MediaModel.Medium', 'goog.ui.media.MediaModel.MimeType', 'goog.ui.media.MediaModel.Player', 'goog.ui.media.MediaModel.SubTitle', 'goog.ui.media.MediaModel.Thumbnail'], ['goog.array', 'goog.html.TrustedResourceUrl'], {});
goog.addDependency('ui/media/mediamodel_test.js', ['goog.ui.media.MediaModelTest'], ['goog.testing.testSuite', 'goog.ui.media.MediaModel'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/media/mp3.js', ['goog.ui.media.Mp3'], ['goog.string', 'goog.ui.media.FlashObject', 'goog.ui.media.Media', 'goog.ui.media.MediaRenderer'], {});
goog.addDependency('ui/media/mp3_test.js', ['goog.ui.media.Mp3Test'], ['goog.dom', 'goog.dom.TagName', 'goog.html.testing', 'goog.testing.testSuite', 'goog.ui.media.FlashObject', 'goog.ui.media.Media', 'goog.ui.media.MediaModel', 'goog.ui.media.Mp3'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/media/photo.js', ['goog.ui.media.Photo'], ['goog.dom.TagName', 'goog.ui.media.Media', 'goog.ui.media.MediaRenderer'], {});
goog.addDependency('ui/media/photo_test.js', ['goog.ui.media.PhotoTest'], ['goog.dom', 'goog.dom.TagName', 'goog.html.testing', 'goog.testing.testSuite', 'goog.ui.media.MediaModel', 'goog.ui.media.Photo'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/media/picasa.js', ['goog.ui.media.PicasaAlbum', 'goog.ui.media.PicasaAlbumModel'], ['goog.html.TrustedResourceUrl', 'goog.string.Const', 'goog.ui.media.FlashObject', 'goog.ui.media.Media', 'goog.ui.media.MediaModel', 'goog.ui.media.MediaRenderer'], {});
goog.addDependency('ui/media/picasa_test.js', ['goog.ui.media.PicasaTest'], ['goog.dom', 'goog.dom.TagName', 'goog.testing.testSuite', 'goog.ui.media.FlashObject', 'goog.ui.media.Media', 'goog.ui.media.PicasaAlbum', 'goog.ui.media.PicasaAlbumModel'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/media/vimeo.js', ['goog.ui.media.Vimeo', 'goog.ui.media.VimeoModel'], ['goog.html.TrustedResourceUrl', 'goog.string', 'goog.string.Const', 'goog.ui.media.FlashObject', 'goog.ui.media.Media', 'goog.ui.media.MediaModel', 'goog.ui.media.MediaRenderer'], {});
goog.addDependency('ui/media/vimeo_test.js', ['goog.ui.media.VimeoTest'], ['goog.dom', 'goog.dom.TagName', 'goog.testing.testSuite', 'goog.ui.media.FlashObject', 'goog.ui.media.Media', 'goog.ui.media.Vimeo', 'goog.ui.media.VimeoModel'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/media/youtube.js', ['goog.ui.media.Youtube', 'goog.ui.media.YoutubeModel'], ['goog.dom.TagName', 'goog.html.TrustedResourceUrl', 'goog.string', 'goog.string.Const', 'goog.ui.Component', 'goog.ui.media.FlashObject', 'goog.ui.media.Media', 'goog.ui.media.MediaModel', 'goog.ui.media.MediaRenderer'], {});
goog.addDependency('ui/media/youtube_test.js', ['goog.ui.media.YoutubeTest'], ['goog.dom', 'goog.dom.TagName', 'goog.testing.testSuite', 'goog.ui.media.FlashObject', 'goog.ui.media.Youtube', 'goog.ui.media.YoutubeModel'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/menu.js', ['goog.ui.Menu', 'goog.ui.Menu.EventType'], ['goog.dom.TagName', 'goog.math.Coordinate', 'goog.string', 'goog.style', 'goog.ui.Component.EventType', 'goog.ui.Component.State', 'goog.ui.Container', 'goog.ui.Container.Orientation', 'goog.ui.MenuHeader', 'goog.ui.MenuItem', 'goog.ui.MenuRenderer', 'goog.ui.MenuSeparator'], {});
goog.addDependency('ui/menu_test.js', ['goog.ui.MenuTest'], ['goog.dom', 'goog.events', 'goog.math.Coordinate', 'goog.testing.events', 'goog.testing.testSuite', 'goog.ui.Component', 'goog.ui.Menu'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/menubar.js', ['goog.ui.menuBar'], ['goog.ui.Container', 'goog.ui.MenuBarRenderer'], {});
goog.addDependency('ui/menubardecorator.js', ['goog.ui.menuBarDecorator'], ['goog.ui.MenuBarRenderer', 'goog.ui.menuBar', 'goog.ui.registry'], {});
goog.addDependency('ui/menubarrenderer.js', ['goog.ui.MenuBarRenderer'], ['goog.a11y.aria.Role', 'goog.ui.Container', 'goog.ui.ContainerRenderer'], {});
goog.addDependency('ui/menubase.js', ['goog.ui.MenuBase'], ['goog.events.EventHandler', 'goog.events.EventType', 'goog.events.KeyHandler', 'goog.ui.Popup'], {});
goog.addDependency('ui/menubutton.js', ['goog.ui.MenuButton'], ['goog.Timer', 'goog.a11y.aria', 'goog.a11y.aria.State', 'goog.asserts', 'goog.dom', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.events.KeyHandler', 'goog.math.Box', 'goog.math.Coordinate', 'goog.math.Rect', 'goog.positioning', 'goog.positioning.Corner', 'goog.positioning.MenuAnchoredPosition', 'goog.positioning.Overflow', 'goog.style', 'goog.ui.Button', 'goog.ui.Component', 'goog.ui.IdGenerator', 'goog.ui.Menu', 'goog.ui.MenuButtonRenderer', 'goog.ui.MenuItem', 'goog.ui.MenuRenderer', 'goog.ui.SubMenu', 'goog.ui.registry', 'goog.userAgent', 'goog.userAgent.product'], {'lang': 'es6'});
goog.addDependency('ui/menubutton_test.js', ['goog.ui.MenuButtonTest'], ['goog.Timer', 'goog.a11y.aria', 'goog.a11y.aria.State', 'goog.dom', 'goog.dom.TagName', 'goog.events', 'goog.events.Event', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.events.KeyHandler', 'goog.positioning', 'goog.positioning.Corner', 'goog.positioning.MenuAnchoredPosition', 'goog.positioning.Overflow', 'goog.style', 'goog.testing.ExpectedFailures', 'goog.testing.PropertyReplacer', 'goog.testing.events', 'goog.testing.recordFunction', 'goog.testing.testSuite', 'goog.ui.Component', 'goog.ui.Menu', 'goog.ui.MenuButton', 'goog.ui.MenuItem', 'goog.ui.SubMenu', 'goog.userAgent', 'goog.userAgent.product', 'goog.userAgent.product.isVersion'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/menubuttonrenderer.js', ['goog.ui.MenuButtonRenderer'], ['goog.dom', 'goog.dom.TagName', 'goog.style', 'goog.ui.CustomButtonRenderer', 'goog.ui.INLINE_BLOCK_CLASSNAME', 'goog.ui.Menu', 'goog.ui.MenuRenderer'], {});
goog.addDependency('ui/menubuttonrenderer_test.js', ['goog.ui.MenuButtonRendererTest'], ['goog.a11y.aria', 'goog.a11y.aria.State', 'goog.dom', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.testing.testSuite', 'goog.testing.ui.rendererasserts', 'goog.ui.MenuButton', 'goog.ui.MenuButtonRenderer', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/menuheader.js', ['goog.ui.MenuHeader'], ['goog.ui.Component', 'goog.ui.Control', 'goog.ui.MenuHeaderRenderer', 'goog.ui.registry'], {});
goog.addDependency('ui/menuheaderrenderer.js', ['goog.ui.MenuHeaderRenderer'], ['goog.ui.ControlRenderer'], {});
goog.addDependency('ui/menuitem.js', ['goog.ui.MenuItem'], ['goog.a11y.aria.Role', 'goog.array', 'goog.dom', 'goog.dom.classlist', 'goog.math.Coordinate', 'goog.string', 'goog.ui.Component', 'goog.ui.Control', 'goog.ui.MenuItemRenderer', 'goog.ui.registry'], {});
goog.addDependency('ui/menuitem_test.js', ['goog.ui.MenuItemTest'], ['goog.a11y.aria', 'goog.a11y.aria.Role', 'goog.array', 'goog.dom', 'goog.dom.NodeType', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.events.KeyCodes', 'goog.html.testing', 'goog.math.Coordinate', 'goog.testing.events', 'goog.testing.recordFunction', 'goog.testing.testSuite', 'goog.ui.Component', 'goog.ui.MenuItem', 'goog.ui.MenuItemRenderer'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/menuitemrenderer.js', ['goog.ui.MenuItemRenderer'], ['goog.a11y.aria.Role', 'goog.asserts', 'goog.dom', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.ui.Component', 'goog.ui.ControlRenderer'], {});
goog.addDependency('ui/menuitemrenderer_test.js', ['goog.ui.MenuItemRendererTest'], ['goog.a11y.aria', 'goog.a11y.aria.Role', 'goog.a11y.aria.State', 'goog.dom', 'goog.dom.classlist', 'goog.testing.testSuite', 'goog.testing.ui.rendererasserts', 'goog.ui.Component', 'goog.ui.MenuItem', 'goog.ui.MenuItemRenderer'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/menurenderer.js', ['goog.ui.MenuRenderer'], ['goog.a11y.aria', 'goog.a11y.aria.Role', 'goog.a11y.aria.State', 'goog.asserts', 'goog.dom', 'goog.dom.TagName', 'goog.ui.ContainerRenderer', 'goog.ui.Separator'], {});
goog.addDependency('ui/menuseparator.js', ['goog.ui.MenuSeparator'], ['goog.ui.MenuSeparatorRenderer', 'goog.ui.Separator', 'goog.ui.registry'], {});
goog.addDependency('ui/menuseparatorrenderer.js', ['goog.ui.MenuSeparatorRenderer'], ['goog.dom', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.ui.ControlRenderer'], {});
goog.addDependency('ui/menuseparatorrenderer_test.js', ['goog.ui.MenuSeparatorRendererTest'], ['goog.dom', 'goog.testing.testSuite', 'goog.ui.MenuSeparator', 'goog.ui.MenuSeparatorRenderer'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/mockactivitymonitor.js', ['goog.ui.MockActivityMonitor'], ['goog.events.EventType', 'goog.ui.ActivityMonitor'], {});
goog.addDependency('ui/mockactivitymonitor_test.js', ['goog.ui.MockActivityMonitorTest'], ['goog.events', 'goog.functions', 'goog.testing.recordFunction', 'goog.testing.testSuite', 'goog.ui.ActivityMonitor', 'goog.ui.MockActivityMonitor'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/modalariavisibilityhelper.js', ['goog.ui.ModalAriaVisibilityHelper'], ['goog.a11y.aria', 'goog.a11y.aria.State'], {});
goog.addDependency('ui/modalariavisibilityhelper_test.js', ['goog.ui.ModalAriaVisibilityHelperTest'], ['goog.a11y.aria', 'goog.a11y.aria.State', 'goog.dom', 'goog.string', 'goog.testing.testSuite', 'goog.ui.ModalAriaVisibilityHelper'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/modalpopup.js', ['goog.ui.ModalPopup'], ['goog.Timer', 'goog.asserts', 'goog.dom', 'goog.dom.TagName', 'goog.dom.animationFrame', 'goog.dom.classlist', 'goog.dom.iframe', 'goog.events', 'goog.events.EventType', 'goog.events.FocusHandler', 'goog.fx.Transition', 'goog.string', 'goog.style', 'goog.ui.Component', 'goog.ui.ModalAriaVisibilityHelper', 'goog.ui.PopupBase', 'goog.userAgent'], {});
goog.addDependency('ui/modalpopup_test.js', ['goog.ui.ModalPopupTest'], ['goog.a11y.aria', 'goog.a11y.aria.State', 'goog.dispose', 'goog.dom', 'goog.dom.TagName', 'goog.events', 'goog.events.EventTarget', 'goog.events.EventType', 'goog.fx.Transition', 'goog.fx.css3', 'goog.string', 'goog.style', 'goog.testing.MockClock', 'goog.testing.events', 'goog.testing.testSuite', 'goog.ui.ModalPopup', 'goog.ui.PopupBase'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/nativebuttonrenderer.js', ['goog.ui.NativeButtonRenderer'], ['goog.asserts', 'goog.dom.InputType', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.events.EventType', 'goog.ui.ButtonRenderer', 'goog.ui.Component'], {});
goog.addDependency('ui/nativebuttonrenderer_test.js', ['goog.ui.NativeButtonRendererTest'], ['goog.dom', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.events', 'goog.testing.ExpectedFailures', 'goog.testing.events', 'goog.testing.testSuite', 'goog.testing.ui.rendererasserts', 'goog.ui.Button', 'goog.ui.Component', 'goog.ui.NativeButtonRenderer', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/option.js', ['goog.ui.Option'], ['goog.ui.Component', 'goog.ui.MenuItem', 'goog.ui.registry'], {});
goog.addDependency('ui/palette.js', ['goog.ui.Palette'], ['goog.array', 'goog.dom', 'goog.events', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.math.Size', 'goog.ui.Component', 'goog.ui.Control', 'goog.ui.PaletteRenderer', 'goog.ui.SelectionModel'], {});
goog.addDependency('ui/palette_test.js', ['goog.ui.PaletteTest'], ['goog.a11y.aria', 'goog.dom', 'goog.events', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.events.KeyEvent', 'goog.testing.events.Event', 'goog.testing.recordFunction', 'goog.testing.testSuite', 'goog.ui.Component', 'goog.ui.Container', 'goog.ui.Palette'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/paletterenderer.js', ['goog.ui.PaletteRenderer'], ['goog.a11y.aria', 'goog.a11y.aria.Role', 'goog.a11y.aria.State', 'goog.array', 'goog.asserts', 'goog.dom', 'goog.dom.NodeIterator', 'goog.dom.NodeType', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.dom.dataset', 'goog.iter', 'goog.style', 'goog.ui.ControlRenderer', 'goog.userAgent'], {});
goog.addDependency('ui/paletterenderer_test.js', ['goog.ui.PaletteRendererTest'], ['goog.a11y.aria', 'goog.a11y.aria.Role', 'goog.a11y.aria.State', 'goog.array', 'goog.dom', 'goog.dom.TagName', 'goog.html.testing', 'goog.testing.testSuite', 'goog.ui.Palette', 'goog.ui.PaletteRenderer'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/plaintextspellchecker.js', ['goog.ui.PlainTextSpellChecker'], ['goog.Timer', 'goog.a11y.aria', 'goog.asserts', 'goog.dom', 'goog.dom.TagName', 'goog.events.EventHandler', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.events.KeyHandler', 'goog.spell.SpellCheck', 'goog.style', 'goog.ui.AbstractSpellChecker', 'goog.ui.Component', 'goog.userAgent'], {});
goog.addDependency('ui/plaintextspellchecker_test.js', ['goog.ui.PlainTextSpellCheckerTest'], ['goog.Timer', 'goog.dom', 'goog.events.KeyCodes', 'goog.spell.SpellCheck', 'goog.testing.events', 'goog.testing.testSuite', 'goog.ui.AbstractSpellChecker', 'goog.ui.PlainTextSpellChecker'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/popup.js', ['goog.ui.Popup'], ['goog.math.Box', 'goog.positioning.AbstractPosition', 'goog.positioning.Corner', 'goog.style', 'goog.ui.PopupBase'], {});
goog.addDependency('ui/popup_test.js', ['goog.ui.PopupTest'], ['goog.positioning.AnchoredPosition', 'goog.positioning.Corner', 'goog.style', 'goog.testing.testSuite', 'goog.ui.Popup', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/popupbase.js', ['goog.ui.PopupBase', 'goog.ui.PopupBase.EventType', 'goog.ui.PopupBase.Type'], ['goog.Timer', 'goog.array', 'goog.dom', 'goog.dom.TagName', 'goog.events', 'goog.events.EventHandler', 'goog.events.EventTarget', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.fx.Transition', 'goog.style', 'goog.userAgent'], {});
goog.addDependency('ui/popupbase_test.js', ['goog.ui.PopupBaseTest'], ['goog.dom', 'goog.dom.TagName', 'goog.events', 'goog.events.EventTarget', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.fx.Transition', 'goog.fx.css3', 'goog.testing.MockClock', 'goog.testing.events', 'goog.testing.events.Event', 'goog.testing.testSuite', 'goog.ui.PopupBase'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/popupcolorpicker.js', ['goog.ui.PopupColorPicker'], ['goog.asserts', 'goog.dom.classlist', 'goog.events.EventType', 'goog.positioning.AnchoredPosition', 'goog.positioning.Corner', 'goog.ui.ColorPicker', 'goog.ui.Component', 'goog.ui.Popup'], {});
goog.addDependency('ui/popupcolorpicker_test.js', ['goog.ui.PopupColorPickerTest'], ['goog.dom', 'goog.events', 'goog.testing.events', 'goog.testing.testSuite', 'goog.ui.ColorPicker', 'goog.ui.PopupColorPicker'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/popupdatepicker.js', ['goog.ui.PopupDatePicker'], ['goog.events.EventType', 'goog.positioning.AnchoredViewportPosition', 'goog.positioning.Corner', 'goog.style', 'goog.ui.Component', 'goog.ui.DatePicker', 'goog.ui.Popup', 'goog.ui.PopupBase'], {});
goog.addDependency('ui/popupdatepicker_test.js', ['goog.ui.PopupDatePickerTest'], ['goog.date.Date', 'goog.dom', 'goog.dom.TagName', 'goog.events', 'goog.style', 'goog.testing.MockControl', 'goog.testing.recordFunction', 'goog.testing.testSuite', 'goog.ui.DatePicker', 'goog.ui.PopupBase', 'goog.ui.PopupDatePicker'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/popupmenu.js', ['goog.ui.PopupMenu'], ['goog.events', 'goog.events.BrowserEvent', 'goog.events.BrowserEvent.MouseButton', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.positioning.AnchoredViewportPosition', 'goog.positioning.Corner', 'goog.positioning.MenuAnchoredPosition', 'goog.positioning.Overflow', 'goog.positioning.ViewportClientPosition', 'goog.structs.Map', 'goog.style', 'goog.ui.Component', 'goog.ui.Menu', 'goog.ui.PopupBase'], {});
goog.addDependency('ui/popupmenu_test.js', ['goog.ui.PopupMenuTest'], ['goog.dom', 'goog.events.BrowserEvent', 'goog.events.EventHandler', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.math.Box', 'goog.math.Coordinate', 'goog.positioning.Corner', 'goog.style', 'goog.testing.events', 'goog.testing.testSuite', 'goog.ui.Menu', 'goog.ui.MenuItem', 'goog.ui.PopupMenu'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/progressbar.js', ['goog.ui.ProgressBar', 'goog.ui.ProgressBar.Orientation'], ['goog.a11y.aria', 'goog.asserts', 'goog.dom', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.events', 'goog.events.EventType', 'goog.ui.Component', 'goog.ui.RangeModel', 'goog.userAgent'], {});
goog.addDependency('ui/prompt.js', ['goog.ui.Prompt'], ['goog.Timer', 'goog.dom', 'goog.dom.InputType', 'goog.dom.TagName', 'goog.events', 'goog.events.EventType', 'goog.functions', 'goog.html.SafeHtml', 'goog.ui.Component', 'goog.ui.Dialog', 'goog.userAgent'], {});
goog.addDependency('ui/prompt_test.js', ['goog.ui.PromptTest'], ['goog.dom.selection', 'goog.events.InputHandler', 'goog.events.KeyCodes', 'goog.functions', 'goog.string', 'goog.testing.events', 'goog.testing.testSuite', 'goog.ui.BidiInput', 'goog.ui.Dialog', 'goog.ui.Prompt', 'goog.userAgent', 'goog.userAgent.product'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/rangemodel.js', ['goog.ui.RangeModel'], ['goog.events.EventTarget', 'goog.ui.Component'], {});
goog.addDependency('ui/rangemodel_test.js', ['goog.ui.RangeModelTest'], ['goog.testing.testSuite', 'goog.ui.RangeModel'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/ratings.js', ['goog.ui.Ratings', 'goog.ui.Ratings.EventType'], ['goog.a11y.aria', 'goog.a11y.aria.Role', 'goog.a11y.aria.State', 'goog.asserts', 'goog.dom', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.events.EventType', 'goog.ui.Component'], {});
goog.addDependency('ui/registry.js', ['goog.ui.registry'], ['goog.asserts', 'goog.dom.classlist'], {});
goog.addDependency('ui/registry_test.js', ['goog.ui.registryTest'], ['goog.object', 'goog.testing.testSuite', 'goog.ui.registry'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/richtextspellchecker.js', ['goog.ui.RichTextSpellChecker'], ['goog.Timer', 'goog.asserts', 'goog.dom', 'goog.dom.NodeType', 'goog.dom.Range', 'goog.events.EventHandler', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.events.KeyHandler', 'goog.math.Coordinate', 'goog.spell.SpellCheck', 'goog.string.StringBuffer', 'goog.style', 'goog.ui.AbstractSpellChecker', 'goog.ui.Component', 'goog.ui.PopupMenu'], {});
goog.addDependency('ui/richtextspellchecker_test.js', ['goog.ui.RichTextSpellCheckerTest'], ['goog.dom.Range', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.events.KeyCodes', 'goog.object', 'goog.spell.SpellCheck', 'goog.testing.MockClock', 'goog.testing.events', 'goog.testing.testSuite', 'goog.ui.RichTextSpellChecker'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/roundedpanel.js', ['goog.ui.BaseRoundedPanel', 'goog.ui.CssRoundedPanel', 'goog.ui.GraphicsRoundedPanel', 'goog.ui.RoundedPanel', 'goog.ui.RoundedPanel.Corner'], ['goog.asserts', 'goog.dom', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.graphics', 'goog.graphics.Path', 'goog.graphics.SolidFill', 'goog.graphics.Stroke', 'goog.math', 'goog.math.Coordinate', 'goog.style', 'goog.ui.Component', 'goog.userAgent'], {});
goog.addDependency('ui/roundedpanel_test.js', ['goog.ui.RoundedPanelTest'], ['goog.testing.testSuite', 'goog.ui.CssRoundedPanel', 'goog.ui.GraphicsRoundedPanel', 'goog.ui.RoundedPanel', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/roundedtabrenderer.js', ['goog.ui.RoundedTabRenderer'], ['goog.dom', 'goog.dom.TagName', 'goog.ui.Tab', 'goog.ui.TabBar', 'goog.ui.TabRenderer', 'goog.ui.registry'], {});
goog.addDependency('ui/scrollfloater.js', ['goog.ui.ScrollFloater', 'goog.ui.ScrollFloater.EventType'], ['goog.array', 'goog.asserts', 'goog.dom', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.events.EventType', 'goog.style', 'goog.ui.Component', 'goog.userAgent'], {});
goog.addDependency('ui/scrollfloater_test.js', ['goog.ui.ScrollFloaterTest'], ['goog.dom', 'goog.events', 'goog.style', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite', 'goog.ui.ScrollFloater'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/select.js', ['goog.ui.Select'], ['goog.a11y.aria', 'goog.a11y.aria.Role', 'goog.a11y.aria.State', 'goog.array', 'goog.events.EventType', 'goog.ui.Component', 'goog.ui.IdGenerator', 'goog.ui.MenuButton', 'goog.ui.MenuItem', 'goog.ui.MenuRenderer', 'goog.ui.SelectionModel', 'goog.ui.registry'], {});
goog.addDependency('ui/select_test.js', ['goog.ui.SelectTest'], ['goog.a11y.aria', 'goog.a11y.aria.Role', 'goog.a11y.aria.State', 'goog.dom', 'goog.events', 'goog.testing.recordFunction', 'goog.testing.testSuite', 'goog.ui.Component', 'goog.ui.CustomButtonRenderer', 'goog.ui.Menu', 'goog.ui.MenuItem', 'goog.ui.Select', 'goog.ui.Separator'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/selectionmenubutton.js', ['goog.ui.SelectionMenuButton', 'goog.ui.SelectionMenuButton.SelectionState'], ['goog.dom.InputType', 'goog.dom.TagName', 'goog.events.EventType', 'goog.style', 'goog.ui.Component', 'goog.ui.MenuButton', 'goog.ui.MenuItem', 'goog.ui.registry'], {});
goog.addDependency('ui/selectionmenubutton_test.js', ['goog.ui.SelectionMenuButtonTest'], ['goog.dom', 'goog.events', 'goog.testing.events', 'goog.testing.testSuite', 'goog.ui.Component', 'goog.ui.SelectionMenuButton'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/selectionmodel.js', ['goog.ui.SelectionModel'], ['goog.array', 'goog.events.EventTarget', 'goog.events.EventType'], {});
goog.addDependency('ui/selectionmodel_test.js', ['goog.ui.SelectionModelTest'], ['goog.array', 'goog.testing.recordFunction', 'goog.testing.testSuite', 'goog.ui.SelectionModel'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/separator.js', ['goog.ui.Separator'], ['goog.a11y.aria', 'goog.asserts', 'goog.ui.Component', 'goog.ui.Control', 'goog.ui.MenuSeparatorRenderer', 'goog.ui.registry'], {});
goog.addDependency('ui/serverchart.js', ['goog.ui.ServerChart', 'goog.ui.ServerChart.AxisDisplayType', 'goog.ui.ServerChart.ChartType', 'goog.ui.ServerChart.EncodingType', 'goog.ui.ServerChart.Event', 'goog.ui.ServerChart.LegendPosition', 'goog.ui.ServerChart.MaximumValue', 'goog.ui.ServerChart.MultiAxisAlignment', 'goog.ui.ServerChart.MultiAxisType', 'goog.ui.ServerChart.UriParam', 'goog.ui.ServerChart.UriTooLongEvent'], ['goog.Uri', 'goog.array', 'goog.asserts', 'goog.dom.TagName', 'goog.dom.safe', 'goog.events.Event', 'goog.string', 'goog.ui.Component'], {});
goog.addDependency('ui/serverchart_test.js', ['goog.ui.ServerChartTest'], ['goog.Uri', 'goog.events', 'goog.testing.testSuite', 'goog.ui.ServerChart'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/slider.js', ['goog.ui.Slider', 'goog.ui.Slider.Orientation'], ['goog.a11y.aria', 'goog.a11y.aria.Role', 'goog.dom', 'goog.dom.TagName', 'goog.ui.SliderBase'], {});
goog.addDependency('ui/sliderbase.js', ['goog.ui.SliderBase', 'goog.ui.SliderBase.AnimationFactory', 'goog.ui.SliderBase.Orientation'], ['goog.Timer', 'goog.a11y.aria', 'goog.a11y.aria.Role', 'goog.a11y.aria.State', 'goog.array', 'goog.asserts', 'goog.dom', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.events', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.events.KeyHandler', 'goog.events.MouseWheelHandler', 'goog.functions', 'goog.fx.AnimationParallelQueue', 'goog.fx.Dragger', 'goog.fx.Transition', 'goog.fx.dom.ResizeHeight', 'goog.fx.dom.ResizeWidth', 'goog.fx.dom.Slide', 'goog.math', 'goog.math.Coordinate', 'goog.style', 'goog.style.bidi', 'goog.ui.Component', 'goog.ui.RangeModel'], {});
goog.addDependency('ui/sliderbase_test.js', ['goog.ui.SliderBaseTest'], ['goog.a11y.aria', 'goog.a11y.aria.State', 'goog.dom', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.events', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.fx.Animation', 'goog.math.Coordinate', 'goog.style', 'goog.style.bidi', 'goog.testing.MockClock', 'goog.testing.MockControl', 'goog.testing.events', 'goog.testing.mockmatchers', 'goog.testing.recordFunction', 'goog.testing.testSuite', 'goog.ui.Component', 'goog.ui.SliderBase', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/splitpane.js', ['goog.ui.SplitPane', 'goog.ui.SplitPane.Orientation'], ['goog.asserts', 'goog.dom', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.events.EventType', 'goog.fx.Dragger', 'goog.math.Rect', 'goog.math.Size', 'goog.style', 'goog.ui.Component', 'goog.userAgent'], {});
goog.addDependency('ui/splitpane_test.js', ['goog.ui.SplitPaneTest'], ['goog.dom', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.events', 'goog.math.Size', 'goog.style', 'goog.testing.events', 'goog.testing.recordFunction', 'goog.testing.testSuite', 'goog.ui.Component', 'goog.ui.SplitPane'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/style/app/buttonrenderer.js', ['goog.ui.style.app.ButtonRenderer'], ['goog.dom.TagName', 'goog.dom.classlist', 'goog.ui.Button', 'goog.ui.CustomButtonRenderer', 'goog.ui.INLINE_BLOCK_CLASSNAME', 'goog.ui.registry'], {});
goog.addDependency('ui/style/app/buttonrenderer_test.js', ['goog.ui.style.app.ButtonRendererTest'], ['goog.dom', 'goog.testing.testSuite', 'goog.testing.ui.style', 'goog.ui.Button', 'goog.ui.Component', 'goog.ui.style.app.ButtonRenderer', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/style/app/menubuttonrenderer.js', ['goog.ui.style.app.MenuButtonRenderer'], ['goog.a11y.aria.Role', 'goog.array', 'goog.dom', 'goog.dom.TagName', 'goog.style', 'goog.ui.Menu', 'goog.ui.MenuRenderer', 'goog.ui.style.app.ButtonRenderer'], {});
goog.addDependency('ui/style/app/menubuttonrenderer_test.js', ['goog.ui.style.app.MenuButtonRendererTest'], ['goog.dom', 'goog.testing.testSuite', 'goog.testing.ui.style', 'goog.ui.Component', 'goog.ui.MenuButton', 'goog.ui.style.app.MenuButtonRenderer'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/style/app/primaryactionbuttonrenderer.js', ['goog.ui.style.app.PrimaryActionButtonRenderer'], ['goog.ui.Button', 'goog.ui.registry', 'goog.ui.style.app.ButtonRenderer'], {});
goog.addDependency('ui/style/app/primaryactionbuttonrenderer_test.js', ['goog.ui.style.app.PrimaryActionButtonRendererTest'], ['goog.dom', 'goog.testing.testSuite', 'goog.testing.ui.style', 'goog.ui.Button', 'goog.ui.Component', 'goog.ui.style.app.PrimaryActionButtonRenderer'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/submenu.js', ['goog.ui.SubMenu'], ['goog.Timer', 'goog.asserts', 'goog.dom', 'goog.dom.classlist', 'goog.events.KeyCodes', 'goog.positioning.AnchoredViewportPosition', 'goog.positioning.Corner', 'goog.style', 'goog.ui.Component', 'goog.ui.Menu', 'goog.ui.MenuItem', 'goog.ui.SubMenuRenderer', 'goog.ui.registry'], {});
goog.addDependency('ui/submenu_test.js', ['goog.ui.SubMenuTest'], ['goog.a11y.aria', 'goog.a11y.aria.State', 'goog.dom', 'goog.dom.classlist', 'goog.events', 'goog.events.Event', 'goog.events.KeyCodes', 'goog.events.KeyHandler', 'goog.functions', 'goog.positioning', 'goog.positioning.Overflow', 'goog.style', 'goog.testing.MockClock', 'goog.testing.events', 'goog.testing.testSuite', 'goog.ui.Component', 'goog.ui.Menu', 'goog.ui.MenuItem', 'goog.ui.SubMenu', 'goog.ui.SubMenuRenderer'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/submenurenderer.js', ['goog.ui.SubMenuRenderer'], ['goog.a11y.aria', 'goog.a11y.aria.State', 'goog.asserts', 'goog.dom', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.style', 'goog.ui.Menu', 'goog.ui.MenuItemRenderer'], {});
goog.addDependency('ui/synthetickeyboardevent.js', ['goog.ui.SyntheticKeyboardEvent'], ['goog.events.Event', 'goog.ui.KeyboardEventData'], {});
goog.addDependency('ui/tab.js', ['goog.ui.Tab'], ['goog.ui.Component', 'goog.ui.Control', 'goog.ui.TabRenderer', 'goog.ui.registry'], {});
goog.addDependency('ui/tab_test.js', ['goog.ui.TabTest'], ['goog.dom', 'goog.testing.testSuite', 'goog.ui.Component', 'goog.ui.Tab', 'goog.ui.TabRenderer'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/tabbar.js', ['goog.ui.TabBar', 'goog.ui.TabBar.Location'], ['goog.ui.Component.EventType', 'goog.ui.Container', 'goog.ui.Container.Orientation', 'goog.ui.Tab', 'goog.ui.TabBarRenderer', 'goog.ui.registry'], {});
goog.addDependency('ui/tabbar_test.js', ['goog.ui.TabBarTest'], ['goog.dom', 'goog.events', 'goog.events.Event', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.testing.testSuite', 'goog.ui.Component', 'goog.ui.Container', 'goog.ui.Tab', 'goog.ui.TabBar', 'goog.ui.TabBarRenderer'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/tabbarrenderer.js', ['goog.ui.TabBarRenderer'], ['goog.a11y.aria.Role', 'goog.object', 'goog.ui.ContainerRenderer'], {});
goog.addDependency('ui/tabbarrenderer_test.js', ['goog.ui.TabBarRendererTest'], ['goog.a11y.aria.Role', 'goog.dom', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.testing.testSuite', 'goog.testing.ui.rendererasserts', 'goog.ui.Container', 'goog.ui.TabBar', 'goog.ui.TabBarRenderer'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/tablesorter.js', ['goog.ui.TableSorter', 'goog.ui.TableSorter.EventType'], ['goog.array', 'goog.dom', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.events.EventType', 'goog.functions', 'goog.ui.Component'], {});
goog.addDependency('ui/tablesorter_test.js', ['goog.ui.TableSorterTest'], ['goog.array', 'goog.dom', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.testing.events', 'goog.testing.testSuite', 'goog.ui.TableSorter'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/tabpane.js', ['goog.ui.TabPane', 'goog.ui.TabPane.Events', 'goog.ui.TabPane.TabLocation', 'goog.ui.TabPane.TabPage', 'goog.ui.TabPaneEvent'], ['goog.asserts', 'goog.dom', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.events', 'goog.events.Event', 'goog.events.EventTarget', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.html.SafeStyleSheet', 'goog.style'], {});
goog.addDependency('ui/tabpane_test.js', ['goog.ui.TabPaneTest'], ['goog.dom', 'goog.testing.testSuite', 'goog.ui.TabPane'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/tabrenderer.js', ['goog.ui.TabRenderer'], ['goog.a11y.aria.Role', 'goog.ui.Component', 'goog.ui.ControlRenderer'], {});
goog.addDependency('ui/tabrenderer_test.js', ['goog.ui.TabRendererTest'], ['goog.a11y.aria.Role', 'goog.dom', 'goog.dom.classlist', 'goog.testing.dom', 'goog.testing.testSuite', 'goog.testing.ui.rendererasserts', 'goog.ui.Tab', 'goog.ui.TabRenderer'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/textarea.js', ['goog.ui.Textarea', 'goog.ui.Textarea.EventType'], ['goog.asserts', 'goog.dom', 'goog.dom.classlist', 'goog.events.EventType', 'goog.style', 'goog.ui.Control', 'goog.ui.TextareaRenderer', 'goog.userAgent'], {});
goog.addDependency('ui/textarea_test.js', ['goog.ui.TextareaTest'], ['goog.dom', 'goog.dom.classlist', 'goog.events', 'goog.style', 'goog.testing.ExpectedFailures', 'goog.testing.events.EventObserver', 'goog.testing.testSuite', 'goog.ui.Textarea', 'goog.ui.TextareaRenderer', 'goog.userAgent', 'goog.userAgent.product'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/textarearenderer.js', ['goog.ui.TextareaRenderer'], ['goog.dom.TagName', 'goog.ui.Component', 'goog.ui.ControlRenderer'], {});
goog.addDependency('ui/togglebutton.js', ['goog.ui.ToggleButton'], ['goog.ui.Button', 'goog.ui.Component', 'goog.ui.CustomButtonRenderer', 'goog.ui.registry'], {});
goog.addDependency('ui/toolbar.js', ['goog.ui.Toolbar'], ['goog.ui.Container', 'goog.ui.ToolbarRenderer'], {});
goog.addDependency('ui/toolbar_test.js', ['goog.ui.ToolbarTest'], ['goog.a11y.aria', 'goog.dom', 'goog.events.EventType', 'goog.testing.events', 'goog.testing.events.Event', 'goog.testing.testSuite', 'goog.ui.Toolbar', 'goog.ui.ToolbarMenuButton'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/toolbarbutton.js', ['goog.ui.ToolbarButton'], ['goog.ui.Button', 'goog.ui.ToolbarButtonRenderer', 'goog.ui.registry'], {});
goog.addDependency('ui/toolbarbuttonrenderer.js', ['goog.ui.ToolbarButtonRenderer'], ['goog.ui.CustomButtonRenderer'], {});
goog.addDependency('ui/toolbarcolormenubutton.js', ['goog.ui.ToolbarColorMenuButton'], ['goog.ui.ColorMenuButton', 'goog.ui.ToolbarColorMenuButtonRenderer', 'goog.ui.registry'], {});
goog.addDependency('ui/toolbarcolormenubuttonrenderer.js', ['goog.ui.ToolbarColorMenuButtonRenderer'], ['goog.asserts', 'goog.dom.classlist', 'goog.ui.ColorMenuButtonRenderer', 'goog.ui.MenuButtonRenderer', 'goog.ui.ToolbarMenuButtonRenderer'], {});
goog.addDependency('ui/toolbarcolormenubuttonrenderer_test.js', ['goog.ui.ToolbarColorMenuButtonRendererTest'], ['goog.dom', 'goog.testing.testSuite', 'goog.testing.ui.RendererHarness', 'goog.testing.ui.rendererasserts', 'goog.ui.ToolbarColorMenuButton', 'goog.ui.ToolbarColorMenuButtonRenderer'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/toolbarmenubutton.js', ['goog.ui.ToolbarMenuButton'], ['goog.ui.MenuButton', 'goog.ui.ToolbarMenuButtonRenderer', 'goog.ui.registry'], {});
goog.addDependency('ui/toolbarmenubuttonrenderer.js', ['goog.ui.ToolbarMenuButtonRenderer'], ['goog.ui.MenuButtonRenderer'], {});
goog.addDependency('ui/toolbarrenderer.js', ['goog.ui.ToolbarRenderer'], ['goog.a11y.aria.Role', 'goog.dom.TagName', 'goog.ui.Container', 'goog.ui.ContainerRenderer', 'goog.ui.Separator', 'goog.ui.ToolbarSeparatorRenderer'], {});
goog.addDependency('ui/toolbarselect.js', ['goog.ui.ToolbarSelect'], ['goog.ui.Select', 'goog.ui.ToolbarMenuButtonRenderer', 'goog.ui.registry'], {});
goog.addDependency('ui/toolbarseparator.js', ['goog.ui.ToolbarSeparator'], ['goog.ui.Separator', 'goog.ui.ToolbarSeparatorRenderer', 'goog.ui.registry'], {});
goog.addDependency('ui/toolbarseparatorrenderer.js', ['goog.ui.ToolbarSeparatorRenderer'], ['goog.asserts', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.ui.INLINE_BLOCK_CLASSNAME', 'goog.ui.MenuSeparatorRenderer'], {});
goog.addDependency('ui/toolbarseparatorrenderer_test.js', ['goog.ui.ToolbarSeparatorRendererTest'], ['goog.dom', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.testing.testSuite', 'goog.ui.Component', 'goog.ui.INLINE_BLOCK_CLASSNAME', 'goog.ui.ToolbarSeparator', 'goog.ui.ToolbarSeparatorRenderer'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/toolbartogglebutton.js', ['goog.ui.ToolbarToggleButton'], ['goog.ui.ToggleButton', 'goog.ui.ToolbarButtonRenderer', 'goog.ui.registry'], {});
goog.addDependency('ui/tooltip.js', ['goog.ui.Tooltip', 'goog.ui.Tooltip.CursorTooltipPosition', 'goog.ui.Tooltip.ElementTooltipPosition', 'goog.ui.Tooltip.State'], ['goog.Timer', 'goog.array', 'goog.asserts', 'goog.dom', 'goog.dom.TagName', 'goog.dom.safe', 'goog.events', 'goog.events.EventType', 'goog.events.FocusHandler', 'goog.math.Box', 'goog.math.Coordinate', 'goog.positioning', 'goog.positioning.AnchoredPosition', 'goog.positioning.Corner', 'goog.positioning.Overflow', 'goog.positioning.OverflowStatus', 'goog.positioning.ViewportPosition', 'goog.structs.Set', 'goog.style', 'goog.ui.Popup', 'goog.ui.PopupBase'], {});
goog.addDependency('ui/tooltip_test.js', ['goog.ui.TooltipTest'], ['goog.dom', 'goog.dom.TagName', 'goog.events.Event', 'goog.events.EventHandler', 'goog.events.EventType', 'goog.events.FocusHandler', 'goog.html.testing', 'goog.math.Coordinate', 'goog.positioning.AbsolutePosition', 'goog.style', 'goog.testing.MockClock', 'goog.testing.TestQueue', 'goog.testing.events', 'goog.testing.testSuite', 'goog.ui.PopupBase', 'goog.ui.Tooltip', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/tree/basenode.js', ['goog.ui.tree.BaseNode', 'goog.ui.tree.BaseNode.EventType'], ['goog.Timer', 'goog.a11y.aria', 'goog.a11y.aria.State', 'goog.asserts', 'goog.dom.safe', 'goog.events.Event', 'goog.events.KeyCodes', 'goog.html.SafeHtml', 'goog.html.SafeStyle', 'goog.string', 'goog.string.StringBuffer', 'goog.style', 'goog.ui.Component'], {});
goog.addDependency('ui/tree/basenode_test.js', ['goog.ui.tree.BaseNodeTest'], ['goog.a11y.aria', 'goog.dom', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.html.testing', 'goog.testing.testSuite', 'goog.ui.Component', 'goog.ui.tree.BaseNode', 'goog.ui.tree.TreeControl', 'goog.ui.tree.TreeNode'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/tree/treecontrol.js', ['goog.ui.tree.TreeControl'], ['goog.a11y.aria', 'goog.asserts', 'goog.dom.classlist', 'goog.events.EventType', 'goog.events.FocusHandler', 'goog.events.KeyHandler', 'goog.html.SafeHtml', 'goog.log', 'goog.ui.tree.BaseNode', 'goog.ui.tree.TreeNode', 'goog.ui.tree.TypeAhead', 'goog.userAgent'], {});
goog.addDependency('ui/tree/treecontrol_test.js', ['goog.ui.tree.TreeControlTest'], ['goog.dom', 'goog.testing.testSuite', 'goog.ui.tree.TreeControl'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/tree/treenode.js', ['goog.ui.tree.TreeNode'], ['goog.ui.tree.BaseNode'], {});
goog.addDependency('ui/tree/typeahead.js', ['goog.ui.tree.TypeAhead', 'goog.ui.tree.TypeAhead.Offset'], ['goog.array', 'goog.events.KeyCodes', 'goog.string', 'goog.structs.Trie'], {});
goog.addDependency('ui/tree/typeahead_test.js', ['goog.ui.tree.TypeAheadTest'], ['goog.dom', 'goog.events.KeyCodes', 'goog.testing.testSuite', 'goog.ui.tree.TreeControl', 'goog.ui.tree.TypeAhead'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/tristatemenuitem.js', ['goog.ui.TriStateMenuItem', 'goog.ui.TriStateMenuItem.State'], ['goog.dom.classlist', 'goog.ui.Component', 'goog.ui.MenuItem', 'goog.ui.TriStateMenuItemRenderer', 'goog.ui.registry'], {});
goog.addDependency('ui/tristatemenuitemrenderer.js', ['goog.ui.TriStateMenuItemRenderer'], ['goog.asserts', 'goog.dom.classlist', 'goog.ui.MenuItemRenderer'], {});
goog.addDependency('ui/twothumbslider.js', ['goog.ui.TwoThumbSlider'], ['goog.a11y.aria', 'goog.a11y.aria.Role', 'goog.dom', 'goog.dom.TagName', 'goog.ui.SliderBase'], {});
goog.addDependency('ui/twothumbslider_test.js', ['goog.ui.TwoThumbSliderTest'], ['goog.testing.testSuite', 'goog.ui.SliderBase', 'goog.ui.TwoThumbSlider'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('ui/zippy.js', ['goog.ui.Zippy', 'goog.ui.Zippy.Events', 'goog.ui.ZippyEvent'], ['goog.a11y.aria', 'goog.a11y.aria.Role', 'goog.a11y.aria.State', 'goog.dom', 'goog.dom.classlist', 'goog.events.Event', 'goog.events.EventHandler', 'goog.events.EventTarget', 'goog.events.EventType', 'goog.events.KeyCodes', 'goog.events.KeyHandler', 'goog.style'], {});
goog.addDependency('ui/zippy_test.js', ['goog.ui.ZippyTest'], ['goog.a11y.aria', 'goog.dom', 'goog.dom.TagName', 'goog.dom.classlist', 'goog.events', 'goog.events.KeyCodes', 'goog.object', 'goog.testing.events', 'goog.testing.testSuite', 'goog.ui.Zippy'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('uri/uri.js', ['goog.Uri', 'goog.Uri.QueryData'], ['goog.array', 'goog.asserts', 'goog.string', 'goog.structs', 'goog.structs.Map', 'goog.uri.utils', 'goog.uri.utils.ComponentIndex', 'goog.uri.utils.StandardQueryParam'], {});
goog.addDependency('uri/uri_test.js', ['goog.UriTest'], ['goog.Uri', 'goog.testing.testSuite'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('uri/utils.js', ['goog.uri.utils', 'goog.uri.utils.ComponentIndex', 'goog.uri.utils.QueryArray', 'goog.uri.utils.QueryValue', 'goog.uri.utils.StandardQueryParam'], ['goog.array', 'goog.asserts', 'goog.string'], {});
goog.addDependency('uri/utils_test.js', ['goog.uri.utilsTest'], ['goog.functions', 'goog.string', 'goog.testing.testSuite', 'goog.uri.utils'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('useragent/adobereader.js', ['goog.userAgent.adobeReader'], ['goog.string', 'goog.userAgent'], {'module': 'goog'});
goog.addDependency('useragent/adobereader_test.js', ['goog.userAgent.adobeReaderTest'], ['goog.testing.testSuite', 'goog.userAgent.adobeReader'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('useragent/flash.js', ['goog.userAgent.flash'], ['goog.string'], {});
goog.addDependency('useragent/flash_test.js', ['goog.userAgent.flashTest'], ['goog.testing.testSuite', 'goog.userAgent.flash'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('useragent/iphoto.js', ['goog.userAgent.iphoto'], ['goog.string', 'goog.userAgent'], {});
goog.addDependency('useragent/jscript.js', ['goog.userAgent.jscript'], ['goog.string'], {});
goog.addDependency('useragent/jscript_test.js', ['goog.userAgent.jscriptTest'], ['goog.testing.testSuite', 'goog.userAgent.jscript'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('useragent/keyboard.js', ['goog.userAgent.keyboard'], ['goog.labs.userAgent.platform'], {});
goog.addDependency('useragent/keyboard_test.js', ['goog.userAgent.keyboardTest'], ['goog.labs.userAgent.testAgents', 'goog.labs.userAgent.util', 'goog.testing.MockUserAgent', 'goog.testing.testSuite', 'goog.userAgent.keyboard', 'goog.userAgentTestUtil'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('useragent/platform.js', ['goog.userAgent.platform'], ['goog.string', 'goog.userAgent'], {});
goog.addDependency('useragent/platform_test.js', ['goog.userAgent.platformTest'], ['goog.testing.MockUserAgent', 'goog.testing.testSuite', 'goog.userAgent', 'goog.userAgent.platform', 'goog.userAgentTestUtil'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('useragent/product.js', ['goog.userAgent.product'], ['goog.labs.userAgent.browser', 'goog.labs.userAgent.platform', 'goog.userAgent'], {});
goog.addDependency('useragent/product_isversion.js', ['goog.userAgent.product.isVersion'], ['goog.labs.userAgent.platform', 'goog.string', 'goog.userAgent', 'goog.userAgent.product'], {});
goog.addDependency('useragent/product_test.js', ['goog.userAgent.productTest'], ['goog.array', 'goog.labs.userAgent.testAgents', 'goog.labs.userAgent.util', 'goog.testing.MockUserAgent', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite', 'goog.userAgent', 'goog.userAgent.product', 'goog.userAgent.product.isVersion', 'goog.userAgentTestUtil'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('useragent/useragent.js', ['goog.userAgent'], ['goog.labs.userAgent.browser', 'goog.labs.userAgent.engine', 'goog.labs.userAgent.platform', 'goog.labs.userAgent.util', 'goog.reflect', 'goog.string'], {});
goog.addDependency('useragent/useragent_quirks_test.js', ['goog.userAgentQuirksTest'], ['goog.testing.testSuite', 'goog.userAgent'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('useragent/useragent_test.js', ['goog.userAgentTest'], ['goog.array', 'goog.labs.userAgent.platform', 'goog.labs.userAgent.testAgents', 'goog.labs.userAgent.util', 'goog.testing.PropertyReplacer', 'goog.testing.testSuite', 'goog.userAgent', 'goog.userAgentTestUtil'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('useragent/useragenttestutil.js', ['goog.userAgentTestUtil', 'goog.userAgentTestUtil.UserAgents'], ['goog.labs.userAgent.browser', 'goog.labs.userAgent.engine', 'goog.labs.userAgent.platform', 'goog.object', 'goog.userAgent', 'goog.userAgent.keyboard', 'goog.userAgent.platform', 'goog.userAgent.product', 'goog.userAgent.product.isVersion'], {});
goog.addDependency('vec/float32array.js', ['goog.vec.Float32Array'], [], {'lang': 'es6'});
goog.addDependency('vec/float32array_test.js', ['goog.vec.Float32ArrayTest'], ['goog.testing.testSuite', 'goog.vec.Float32Array'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('vec/float64array.js', ['goog.vec.Float64Array'], [], {'lang': 'es6'});
goog.addDependency('vec/float64array_test.js', ['goog.vec.Float64ArrayTest'], ['goog.testing.testSuite', 'goog.vec.Float64Array'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('vec/mat3.js', ['goog.vec.Mat3'], ['goog.vec'], {});
goog.addDependency('vec/mat3_test.js', ['goog.vec.Mat3Test'], ['goog.testing.testSuite', 'goog.vec.Mat3'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('vec/mat3d.js', ['goog.vec.mat3d', 'goog.vec.mat3d.Type'], ['goog.vec', 'goog.vec.vec3d.Type'], {});
goog.addDependency('vec/mat3d_test.js', ['goog.vec.mat3dTest'], ['goog.testing.testSuite', 'goog.vec.mat3d'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('vec/mat3f.js', ['goog.vec.mat3f', 'goog.vec.mat3f.Type'], ['goog.vec', 'goog.vec.vec3f.Type'], {});
goog.addDependency('vec/mat3f_test.js', ['goog.vec.mat3fTest'], ['goog.testing.testSuite', 'goog.vec.mat3f'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('vec/mat4.js', ['goog.vec.Mat4'], ['goog.vec', 'goog.vec.Vec3', 'goog.vec.Vec4'], {});
goog.addDependency('vec/mat4_test.js', ['goog.vec.Mat4Test'], ['goog.testing.testSuite', 'goog.vec.Mat4', 'goog.vec.Vec3', 'goog.vec.Vec4'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('vec/mat4d.js', ['goog.vec.mat4d', 'goog.vec.mat4d.Type'], ['goog.vec', 'goog.vec.Quaternion', 'goog.vec.vec3d', 'goog.vec.vec4d'], {});
goog.addDependency('vec/mat4d_test.js', ['goog.vec.mat4dTest'], ['goog.testing.testSuite', 'goog.vec.Quaternion', 'goog.vec.mat4d', 'goog.vec.vec3d', 'goog.vec.vec4d'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('vec/mat4f.js', ['goog.vec.mat4f', 'goog.vec.mat4f.Type'], ['goog.vec', 'goog.vec.Quaternion', 'goog.vec.vec3f', 'goog.vec.vec4f'], {});
goog.addDependency('vec/mat4f_test.js', ['goog.vec.mat4fTest'], ['goog.testing.testSuite', 'goog.vec.Quaternion', 'goog.vec.mat4f', 'goog.vec.vec3f', 'goog.vec.vec4f'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('vec/quaternion.js', ['goog.vec.Quaternion', 'goog.vec.Quaternion.AnyType'], ['goog.vec', 'goog.vec.Vec3', 'goog.vec.Vec4'], {});
goog.addDependency('vec/quaternion_test.js', ['goog.vec.QuaternionTest'], ['goog.testing.testSuite', 'goog.vec.Mat3', 'goog.vec.Mat4', 'goog.vec.Quaternion', 'goog.vec.Vec3', 'goog.vec.vec3f'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('vec/ray.js', ['goog.vec.Ray'], ['goog.vec.Vec3'], {});
goog.addDependency('vec/ray_test.js', ['goog.vec.RayTest'], ['goog.testing.testSuite', 'goog.vec.Ray'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('vec/vec.js', ['goog.vec', 'goog.vec.AnyType', 'goog.vec.ArrayType', 'goog.vec.Float32', 'goog.vec.Float64', 'goog.vec.Number'], ['goog.vec.Float32Array', 'goog.vec.Float64Array'], {});
goog.addDependency('vec/vec2.js', ['goog.vec.Vec2'], ['goog.vec'], {});
goog.addDependency('vec/vec2_test.js', ['goog.vec.Vec2Test'], ['goog.testing.testSuite', 'goog.vec.Vec2'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('vec/vec2d.js', ['goog.vec.vec2d', 'goog.vec.vec2d.Type'], ['goog.vec'], {});
goog.addDependency('vec/vec2d_test.js', ['goog.vec.vec2dTest'], ['goog.testing.testSuite', 'goog.vec.vec2d'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('vec/vec2f.js', ['goog.vec.vec2f', 'goog.vec.vec2f.Type'], ['goog.vec'], {});
goog.addDependency('vec/vec2f_test.js', ['goog.vec.vec2fTest'], ['goog.testing.testSuite', 'goog.vec.vec2f'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('vec/vec3.js', ['goog.vec.Vec3'], ['goog.vec'], {});
goog.addDependency('vec/vec3_test.js', ['goog.vec.Vec3Test'], ['goog.testing.testSuite', 'goog.vec.Vec3'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('vec/vec3d.js', ['goog.vec.vec3d', 'goog.vec.vec3d.Type'], ['goog.vec'], {});
goog.addDependency('vec/vec3d_test.js', ['goog.vec.vec3dTest'], ['goog.testing.testSuite', 'goog.vec.vec3d'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('vec/vec3f.js', ['goog.vec.vec3f', 'goog.vec.vec3f.Type'], ['goog.vec'], {});
goog.addDependency('vec/vec3f_test.js', ['goog.vec.vec3fTest'], ['goog.testing.testSuite', 'goog.vec.vec3f'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('vec/vec4.js', ['goog.vec.Vec4'], ['goog.vec'], {});
goog.addDependency('vec/vec4_test.js', ['goog.vec.Vec4Test'], ['goog.testing.testSuite', 'goog.vec.Vec4'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('vec/vec4d.js', ['goog.vec.vec4d', 'goog.vec.vec4d.Type'], ['goog.vec'], {});
goog.addDependency('vec/vec4d_test.js', ['goog.vec.vec4dTest'], ['goog.testing.testSuite', 'goog.vec.vec4d'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('vec/vec4f.js', ['goog.vec.vec4f', 'goog.vec.vec4f.Type'], ['goog.vec'], {});
goog.addDependency('vec/vec4f_test.js', ['goog.vec.vec4fTest'], ['goog.testing.testSuite', 'goog.vec.vec4f'], {'lang': 'es6', 'module': 'goog'});
goog.addDependency('webgl/webgl.js', ['goog.webgl'], [], {});
goog.addDependency('window/window.js', ['goog.window'], ['goog.dom', 'goog.dom.TagName', 'goog.dom.safe', 'goog.html.SafeUrl', 'goog.html.uncheckedconversions', 'goog.labs.userAgent.platform', 'goog.string', 'goog.string.Const', 'goog.userAgent'], {});
goog.addDependency('window/window_test.js', ['goog.windowTest'], ['goog.Promise', 'goog.dom', 'goog.dom.TagName', 'goog.events', 'goog.functions', 'goog.html.SafeUrl', 'goog.labs.userAgent.browser', 'goog.labs.userAgent.engine', 'goog.labs.userAgent.platform', 'goog.string', 'goog.testing.PropertyReplacer', 'goog.testing.TestCase', 'goog.testing.testSuite', 'goog.window'], {'lang': 'es6', 'module': 'goog'});

window.polymerSkipLoadingFontRoboto = true;
/** vim: et:ts=4:sw=4:sts=4
 * @license RequireJS 2.3.6 Copyright jQuery Foundation and other contributors.
 * Released under MIT license, https://github.com/requirejs/requirejs/blob/master/LICENSE
 */
//Not using strict: uneven strict support in browsers, #392, and causes
//problems with requirejs.exec()/transpiler plugins that may not be strict.
/*jslint regexp: true, nomen: true, sloppy: true */
/*global window, navigator, document, importScripts, setTimeout, opera */

var requirejs, require, define;
(function (global, setTimeout) {
    var req, s, head, baseElement, dataMain, src,
        interactiveScript, currentlyAddingScript, mainScript, subPath,
        version = '2.3.6',
        commentRegExp = /\/\*[\s\S]*?\*\/|([^:"'=]|^)\/\/.*$/mg,
        cjsRequireRegExp = /[^.]\s*require\s*\(\s*["']([^'"\s]+)["']\s*\)/g,
        jsSuffixRegExp = /\.js$/,
        currDirRegExp = /^\.\//,
        op = Object.prototype,
        ostring = op.toString,
        hasOwn = op.hasOwnProperty,
        isBrowser = !!(typeof window !== 'undefined' && typeof navigator !== 'undefined' && window.document),
        isWebWorker = !isBrowser && typeof importScripts !== 'undefined',
        //PS3 indicates loaded and complete, but need to wait for complete
        //specifically. Sequence is 'loading', 'loaded', execution,
        // then 'complete'. The UA check is unfortunate, but not sure how
        //to feature test w/o causing perf issues.
        readyRegExp = isBrowser && navigator.platform === 'PLAYSTATION 3' ?
                      /^complete$/ : /^(complete|loaded)$/,
        defContextName = '_',
        //Oh the tragedy, detecting opera. See the usage of isOpera for reason.
        isOpera = typeof opera !== 'undefined' && opera.toString() === '[object Opera]',
        contexts = {},
        cfg = {},
        globalDefQueue = [],
        useInteractive = false;

    //Could match something like ')//comment', do not lose the prefix to comment.
    function commentReplace(match, singlePrefix) {
        return singlePrefix || '';
    }

    function isFunction(it) {
        return ostring.call(it) === '[object Function]';
    }

    function isArray(it) {
        return ostring.call(it) === '[object Array]';
    }

    /**
     * Helper function for iterating over an array. If the func returns
     * a true value, it will break out of the loop.
     */
    function each(ary, func) {
        if (ary) {
            var i;
            for (i = 0; i < ary.length; i += 1) {
                if (ary[i] && func(ary[i], i, ary)) {
                    break;
                }
            }
        }
    }

    /**
     * Helper function for iterating over an array backwards. If the func
     * returns a true value, it will break out of the loop.
     */
    function eachReverse(ary, func) {
        if (ary) {
            var i;
            for (i = ary.length - 1; i > -1; i -= 1) {
                if (ary[i] && func(ary[i], i, ary)) {
                    break;
                }
            }
        }
    }

    function hasProp(obj, prop) {
        return hasOwn.call(obj, prop);
    }

    function getOwn(obj, prop) {
        return hasProp(obj, prop) && obj[prop];
    }

    /**
     * Cycles over properties in an object and calls a function for each
     * property value. If the function returns a truthy value, then the
     * iteration is stopped.
     */
    function eachProp(obj, func) {
        var prop;
        for (prop in obj) {
            if (hasProp(obj, prop)) {
                if (func(obj[prop], prop)) {
                    break;
                }
            }
        }
    }

    /**
     * Simple function to mix in properties from source into target,
     * but only if target does not already have a property of the same name.
     */
    function mixin(target, source, force, deepStringMixin) {
        if (source) {
            eachProp(source, function (value, prop) {
                if (force || !hasProp(target, prop)) {
                    if (deepStringMixin && typeof value === 'object' && value &&
                        !isArray(value) && !isFunction(value) &&
                        !(value instanceof RegExp)) {

                        if (!target[prop]) {
                            target[prop] = {};
                        }
                        mixin(target[prop], value, force, deepStringMixin);
                    } else {
                        target[prop] = value;
                    }
                }
            });
        }
        return target;
    }

    //Similar to Function.prototype.bind, but the 'this' object is specified
    //first, since it is easier to read/figure out what 'this' will be.
    function bind(obj, fn) {
        return function () {
            return fn.apply(obj, arguments);
        };
    }

    function scripts() {
        return document.getElementsByTagName('script');
    }

    function defaultOnError(err) {
        throw err;
    }

    //Allow getting a global that is expressed in
    //dot notation, like 'a.b.c'.
    function getGlobal(value) {
        if (!value) {
            return value;
        }
        var g = global;
        each(value.split('.'), function (part) {
            g = g[part];
        });
        return g;
    }

    /**
     * Constructs an error with a pointer to an URL with more information.
     * @param {String} id the error ID that maps to an ID on a web page.
     * @param {String} message human readable error.
     * @param {Error} [err] the original error, if there is one.
     *
     * @returns {Error}
     */
    function makeError(id, msg, err, requireModules) {
        var e = new Error(msg + '\nhttps://requirejs.org/docs/errors.html#' + id);
        e.requireType = id;
        e.requireModules = requireModules;
        if (err) {
            e.originalError = err;
        }
        return e;
    }

    if (typeof define !== 'undefined') {
        //If a define is already in play via another AMD loader,
        //do not overwrite.
        return;
    }

    if (typeof requirejs !== 'undefined') {
        if (isFunction(requirejs)) {
            //Do not overwrite an existing requirejs instance.
            return;
        }
        cfg = requirejs;
        requirejs = undefined;
    }

    //Allow for a require config object
    if (typeof require !== 'undefined' && !isFunction(require)) {
        //assume it is a config object.
        cfg = require;
        require = undefined;
    }

    function newContext(contextName) {
        var inCheckLoaded, Module, context, handlers,
            checkLoadedTimeoutId,
            config = {
                //Defaults. Do not set a default for map
                //config to speed up normalize(), which
                //will run faster if there is no default.
                waitSeconds: 7,
                baseUrl: './',
                paths: {},
                bundles: {},
                pkgs: {},
                shim: {},
                config: {}
            },
            registry = {},
            //registry of just enabled modules, to speed
            //cycle breaking code when lots of modules
            //are registered, but not activated.
            enabledRegistry = {},
            undefEvents = {},
            defQueue = [],
            defined = {},
            urlFetched = {},
            bundlesMap = {},
            requireCounter = 1,
            unnormalizedCounter = 1;

        /**
         * Trims the . and .. from an array of path segments.
         * It will keep a leading path segment if a .. will become
         * the first path segment, to help with module name lookups,
         * which act like paths, but can be remapped. But the end result,
         * all paths that use this function should look normalized.
         * NOTE: this method MODIFIES the input array.
         * @param {Array} ary the array of path segments.
         */
        function trimDots(ary) {
            var i, part;
            for (i = 0; i < ary.length; i++) {
                part = ary[i];
                if (part === '.') {
                    ary.splice(i, 1);
                    i -= 1;
                } else if (part === '..') {
                    // If at the start, or previous value is still ..,
                    // keep them so that when converted to a path it may
                    // still work when converted to a path, even though
                    // as an ID it is less than ideal. In larger point
                    // releases, may be better to just kick out an error.
                    if (i === 0 || (i === 1 && ary[2] === '..') || ary[i - 1] === '..') {
                        continue;
                    } else if (i > 0) {
                        ary.splice(i - 1, 2);
                        i -= 2;
                    }
                }
            }
        }

        /**
         * Given a relative module name, like ./something, normalize it to
         * a real name that can be mapped to a path.
         * @param {String} name the relative name
         * @param {String} baseName a real name that the name arg is relative
         * to.
         * @param {Boolean} applyMap apply the map config to the value. Should
         * only be done if this normalization is for a dependency ID.
         * @returns {String} normalized name
         */
        function normalize(name, baseName, applyMap) {
            var pkgMain, mapValue, nameParts, i, j, nameSegment, lastIndex,
                foundMap, foundI, foundStarMap, starI, normalizedBaseParts,
                baseParts = (baseName && baseName.split('/')),
                map = config.map,
                starMap = map && map['*'];

            //Adjust any relative paths.
            if (name) {
                name = name.split('/');
                lastIndex = name.length - 1;

                // If wanting node ID compatibility, strip .js from end
                // of IDs. Have to do this here, and not in nameToUrl
                // because node allows either .js or non .js to map
                // to same file.
                if (config.nodeIdCompat && jsSuffixRegExp.test(name[lastIndex])) {
                    name[lastIndex] = name[lastIndex].replace(jsSuffixRegExp, '');
                }

                // Starts with a '.' so need the baseName
                if (name[0].charAt(0) === '.' && baseParts) {
                    //Convert baseName to array, and lop off the last part,
                    //so that . matches that 'directory' and not name of the baseName's
                    //module. For instance, baseName of 'one/two/three', maps to
                    //'one/two/three.js', but we want the directory, 'one/two' for
                    //this normalization.
                    normalizedBaseParts = baseParts.slice(0, baseParts.length - 1);
                    name = normalizedBaseParts.concat(name);
                }

                trimDots(name);
                name = name.join('/');
            }

            //Apply map config if available.
            if (applyMap && map && (baseParts || starMap)) {
                nameParts = name.split('/');

                outerLoop: for (i = nameParts.length; i > 0; i -= 1) {
                    nameSegment = nameParts.slice(0, i).join('/');

                    if (baseParts) {
                        //Find the longest baseName segment match in the config.
                        //So, do joins on the biggest to smallest lengths of baseParts.
                        for (j = baseParts.length; j > 0; j -= 1) {
                            mapValue = getOwn(map, baseParts.slice(0, j).join('/'));

                            //baseName segment has config, find if it has one for
                            //this name.
                            if (mapValue) {
                                mapValue = getOwn(mapValue, nameSegment);
                                if (mapValue) {
                                    //Match, update name to the new value.
                                    foundMap = mapValue;
                                    foundI = i;
                                    break outerLoop;
                                }
                            }
                        }
                    }

                    //Check for a star map match, but just hold on to it,
                    //if there is a shorter segment match later in a matching
                    //config, then favor over this star map.
                    if (!foundStarMap && starMap && getOwn(starMap, nameSegment)) {
                        foundStarMap = getOwn(starMap, nameSegment);
                        starI = i;
                    }
                }

                if (!foundMap && foundStarMap) {
                    foundMap = foundStarMap;
                    foundI = starI;
                }

                if (foundMap) {
                    nameParts.splice(0, foundI, foundMap);
                    name = nameParts.join('/');
                }
            }

            // If the name points to a package's name, use
            // the package main instead.
            pkgMain = getOwn(config.pkgs, name);

            return pkgMain ? pkgMain : name;
        }

        function removeScript(name) {
            if (isBrowser) {
                each(scripts(), function (scriptNode) {
                    if (scriptNode.getAttribute('data-requiremodule') === name &&
                            scriptNode.getAttribute('data-requirecontext') === context.contextName) {
                        scriptNode.parentNode.removeChild(scriptNode);
                        return true;
                    }
                });
            }
        }

        function hasPathFallback(id) {
            var pathConfig = getOwn(config.paths, id);
            if (pathConfig && isArray(pathConfig) && pathConfig.length > 1) {
                //Pop off the first array value, since it failed, and
                //retry
                pathConfig.shift();
                context.require.undef(id);

                //Custom require that does not do map translation, since
                //ID is "absolute", already mapped/resolved.
                context.makeRequire(null, {
                    skipMap: true
                })([id]);

                return true;
            }
        }

        //Turns a plugin!resource to [plugin, resource]
        //with the plugin being undefined if the name
        //did not have a plugin prefix.
        function splitPrefix(name) {
            var prefix,
                index = name ? name.indexOf('!') : -1;
            if (index > -1) {
                prefix = name.substring(0, index);
                name = name.substring(index + 1, name.length);
            }
            return [prefix, name];
        }

        /**
         * Creates a module mapping that includes plugin prefix, module
         * name, and path. If parentModuleMap is provided it will
         * also normalize the name via require.normalize()
         *
         * @param {String} name the module name
         * @param {String} [parentModuleMap] parent module map
         * for the module name, used to resolve relative names.
         * @param {Boolean} isNormalized: is the ID already normalized.
         * This is true if this call is done for a define() module ID.
         * @param {Boolean} applyMap: apply the map config to the ID.
         * Should only be true if this map is for a dependency.
         *
         * @returns {Object}
         */
        function makeModuleMap(name, parentModuleMap, isNormalized, applyMap) {
            var url, pluginModule, suffix, nameParts,
                prefix = null,
                parentName = parentModuleMap ? parentModuleMap.name : null,
                originalName = name,
                isDefine = true,
                normalizedName = '';

            //If no name, then it means it is a require call, generate an
            //internal name.
            if (!name) {
                isDefine = false;
                name = '_@r' + (requireCounter += 1);
            }

            nameParts = splitPrefix(name);
            prefix = nameParts[0];
            name = nameParts[1];

            if (prefix) {
                prefix = normalize(prefix, parentName, applyMap);
                pluginModule = getOwn(defined, prefix);
            }

            //Account for relative paths if there is a base name.
            if (name) {
                if (prefix) {
                    if (isNormalized) {
                        normalizedName = name;
                    } else if (pluginModule && pluginModule.normalize) {
                        //Plugin is loaded, use its normalize method.
                        normalizedName = pluginModule.normalize(name, function (name) {
                            return normalize(name, parentName, applyMap);
                        });
                    } else {
                        // If nested plugin references, then do not try to
                        // normalize, as it will not normalize correctly. This
                        // places a restriction on resourceIds, and the longer
                        // term solution is not to normalize until plugins are
                        // loaded and all normalizations to allow for async
                        // loading of a loader plugin. But for now, fixes the
                        // common uses. Details in #1131
                        normalizedName = name.indexOf('!') === -1 ?
                                         normalize(name, parentName, applyMap) :
                                         name;
                    }
                } else {
                    //A regular module.
                    normalizedName = normalize(name, parentName, applyMap);

                    //Normalized name may be a plugin ID due to map config
                    //application in normalize. The map config values must
                    //already be normalized, so do not need to redo that part.
                    nameParts = splitPrefix(normalizedName);
                    prefix = nameParts[0];
                    normalizedName = nameParts[1];
                    isNormalized = true;

                    url = context.nameToUrl(normalizedName);
                }
            }

            //If the id is a plugin id that cannot be determined if it needs
            //normalization, stamp it with a unique ID so two matching relative
            //ids that may conflict can be separate.
            suffix = prefix && !pluginModule && !isNormalized ?
                     '_unnormalized' + (unnormalizedCounter += 1) :
                     '';

            return {
                prefix: prefix,
                name: normalizedName,
                parentMap: parentModuleMap,
                unnormalized: !!suffix,
                url: url,
                originalName: originalName,
                isDefine: isDefine,
                id: (prefix ?
                        prefix + '!' + normalizedName :
                        normalizedName) + suffix
            };
        }

        function getModule(depMap) {
            var id = depMap.id,
                mod = getOwn(registry, id);

            if (!mod) {
                mod = registry[id] = new context.Module(depMap);
            }

            return mod;
        }

        function on(depMap, name, fn) {
            var id = depMap.id,
                mod = getOwn(registry, id);

            if (hasProp(defined, id) &&
                    (!mod || mod.defineEmitComplete)) {
                if (name === 'defined') {
                    fn(defined[id]);
                }
            } else {
                mod = getModule(depMap);
                if (mod.error && name === 'error') {
                    fn(mod.error);
                } else {
                    mod.on(name, fn);
                }
            }
        }

        function onError(err, errback) {
            var ids = err.requireModules,
                notified = false;

            if (errback) {
                errback(err);
            } else {
                each(ids, function (id) {
                    var mod = getOwn(registry, id);
                    if (mod) {
                        //Set error on module, so it skips timeout checks.
                        mod.error = err;
                        if (mod.events.error) {
                            notified = true;
                            mod.emit('error', err);
                        }
                    }
                });

                if (!notified) {
                    req.onError(err);
                }
            }
        }

        /**
         * Internal method to transfer globalQueue items to this context's
         * defQueue.
         */
        function takeGlobalQueue() {
            //Push all the globalDefQueue items into the context's defQueue
            if (globalDefQueue.length) {
                each(globalDefQueue, function(queueItem) {
                    var id = queueItem[0];
                    if (typeof id === 'string') {
                        context.defQueueMap[id] = true;
                    }
                    defQueue.push(queueItem);
                });
                globalDefQueue = [];
            }
        }

        handlers = {
            'require': function (mod) {
                if (mod.require) {
                    return mod.require;
                } else {
                    return (mod.require = context.makeRequire(mod.map));
                }
            },
            'exports': function (mod) {
                mod.usingExports = true;
                if (mod.map.isDefine) {
                    if (mod.exports) {
                        return (defined[mod.map.id] = mod.exports);
                    } else {
                        return (mod.exports = defined[mod.map.id] = {});
                    }
                }
            },
            'module': function (mod) {
                if (mod.module) {
                    return mod.module;
                } else {
                    return (mod.module = {
                        id: mod.map.id,
                        uri: mod.map.url,
                        config: function () {
                            return getOwn(config.config, mod.map.id) || {};
                        },
                        exports: mod.exports || (mod.exports = {})
                    });
                }
            }
        };

        function cleanRegistry(id) {
            //Clean up machinery used for waiting modules.
            delete registry[id];
            delete enabledRegistry[id];
        }

        function breakCycle(mod, traced, processed) {
            var id = mod.map.id;

            if (mod.error) {
                mod.emit('error', mod.error);
            } else {
                traced[id] = true;
                each(mod.depMaps, function (depMap, i) {
                    var depId = depMap.id,
                        dep = getOwn(registry, depId);

                    //Only force things that have not completed
                    //being defined, so still in the registry,
                    //and only if it has not been matched up
                    //in the module already.
                    if (dep && !mod.depMatched[i] && !processed[depId]) {
                        if (getOwn(traced, depId)) {
                            mod.defineDep(i, defined[depId]);
                            mod.check(); //pass false?
                        } else {
                            breakCycle(dep, traced, processed);
                        }
                    }
                });
                processed[id] = true;
            }
        }

        function checkLoaded() {
            var err, usingPathFallback,
                waitInterval = config.waitSeconds * 1000,
                //It is possible to disable the wait interval by using waitSeconds of 0.
                expired = waitInterval && (context.startTime + waitInterval) < new Date().getTime(),
                noLoads = [],
                reqCalls = [],
                stillLoading = false,
                needCycleCheck = true;

            //Do not bother if this call was a result of a cycle break.
            if (inCheckLoaded) {
                return;
            }

            inCheckLoaded = true;

            //Figure out the state of all the modules.
            eachProp(enabledRegistry, function (mod) {
                var map = mod.map,
                    modId = map.id;

                //Skip things that are not enabled or in error state.
                if (!mod.enabled) {
                    return;
                }

                if (!map.isDefine) {
                    reqCalls.push(mod);
                }

                if (!mod.error) {
                    //If the module should be executed, and it has not
                    //been inited and time is up, remember it.
                    if (!mod.inited && expired) {
                        if (hasPathFallback(modId)) {
                            usingPathFallback = true;
                            stillLoading = true;
                        } else {
                            noLoads.push(modId);
                            removeScript(modId);
                        }
                    } else if (!mod.inited && mod.fetched && map.isDefine) {
                        stillLoading = true;
                        if (!map.prefix) {
                            //No reason to keep looking for unfinished
                            //loading. If the only stillLoading is a
                            //plugin resource though, keep going,
                            //because it may be that a plugin resource
                            //is waiting on a non-plugin cycle.
                            return (needCycleCheck = false);
                        }
                    }
                }
            });

            if (expired && noLoads.length) {
                //If wait time expired, throw error of unloaded modules.
                err = makeError('timeout', 'Load timeout for modules: ' + noLoads, null, noLoads);
                err.contextName = context.contextName;
                return onError(err);
            }

            //Not expired, check for a cycle.
            if (needCycleCheck) {
                each(reqCalls, function (mod) {
                    breakCycle(mod, {}, {});
                });
            }

            //If still waiting on loads, and the waiting load is something
            //other than a plugin resource, or there are still outstanding
            //scripts, then just try back later.
            if ((!expired || usingPathFallback) && stillLoading) {
                //Something is still waiting to load. Wait for it, but only
                //if a timeout is not already in effect.
                if ((isBrowser || isWebWorker) && !checkLoadedTimeoutId) {
                    checkLoadedTimeoutId = setTimeout(function () {
                        checkLoadedTimeoutId = 0;
                        checkLoaded();
                    }, 50);
                }
            }

            inCheckLoaded = false;
        }

        Module = function (map) {
            this.events = getOwn(undefEvents, map.id) || {};
            this.map = map;
            this.shim = getOwn(config.shim, map.id);
            this.depExports = [];
            this.depMaps = [];
            this.depMatched = [];
            this.pluginMaps = {};
            this.depCount = 0;

            /* this.exports this.factory
               this.depMaps = [],
               this.enabled, this.fetched
            */
        };

        Module.prototype = {
            init: function (depMaps, factory, errback, options) {
                options = options || {};

                //Do not do more inits if already done. Can happen if there
                //are multiple define calls for the same module. That is not
                //a normal, common case, but it is also not unexpected.
                if (this.inited) {
                    return;
                }

                this.factory = factory;

                if (errback) {
                    //Register for errors on this module.
                    this.on('error', errback);
                } else if (this.events.error) {
                    //If no errback already, but there are error listeners
                    //on this module, set up an errback to pass to the deps.
                    errback = bind(this, function (err) {
                        this.emit('error', err);
                    });
                }

                //Do a copy of the dependency array, so that
                //source inputs are not modified. For example
                //"shim" deps are passed in here directly, and
                //doing a direct modification of the depMaps array
                //would affect that config.
                this.depMaps = depMaps && depMaps.slice(0);

                this.errback = errback;

                //Indicate this module has be initialized
                this.inited = true;

                this.ignore = options.ignore;

                //Could have option to init this module in enabled mode,
                //or could have been previously marked as enabled. However,
                //the dependencies are not known until init is called. So
                //if enabled previously, now trigger dependencies as enabled.
                if (options.enabled || this.enabled) {
                    //Enable this module and dependencies.
                    //Will call this.check()
                    this.enable();
                } else {
                    this.check();
                }
            },

            defineDep: function (i, depExports) {
                //Because of cycles, defined callback for a given
                //export can be called more than once.
                if (!this.depMatched[i]) {
                    this.depMatched[i] = true;
                    this.depCount -= 1;
                    this.depExports[i] = depExports;
                }
            },

            fetch: function () {
                if (this.fetched) {
                    return;
                }
                this.fetched = true;

                context.startTime = (new Date()).getTime();

                var map = this.map;

                //If the manager is for a plugin managed resource,
                //ask the plugin to load it now.
                if (this.shim) {
                    context.makeRequire(this.map, {
                        enableBuildCallback: true
                    })(this.shim.deps || [], bind(this, function () {
                        return map.prefix ? this.callPlugin() : this.load();
                    }));
                } else {
                    //Regular dependency.
                    return map.prefix ? this.callPlugin() : this.load();
                }
            },

            load: function () {
                var url = this.map.url;

                //Regular dependency.
                if (!urlFetched[url]) {
                    urlFetched[url] = true;
                    context.load(this.map.id, url);
                }
            },

            /**
             * Checks if the module is ready to define itself, and if so,
             * define it.
             */
            check: function () {
                if (!this.enabled || this.enabling) {
                    return;
                }

                var err, cjsModule,
                    id = this.map.id,
                    depExports = this.depExports,
                    exports = this.exports,
                    factory = this.factory;

                if (!this.inited) {
                    // Only fetch if not already in the defQueue.
                    if (!hasProp(context.defQueueMap, id)) {
                        this.fetch();
                    }
                } else if (this.error) {
                    this.emit('error', this.error);
                } else if (!this.defining) {
                    //The factory could trigger another require call
                    //that would result in checking this module to
                    //define itself again. If already in the process
                    //of doing that, skip this work.
                    this.defining = true;

                    if (this.depCount < 1 && !this.defined) {
                        if (isFunction(factory)) {
                            //If there is an error listener, favor passing
                            //to that instead of throwing an error. However,
                            //only do it for define()'d  modules. require
                            //errbacks should not be called for failures in
                            //their callbacks (#699). However if a global
                            //onError is set, use that.
                            if ((this.events.error && this.map.isDefine) ||
                                req.onError !== defaultOnError) {
                                try {
                                    exports = context.execCb(id, factory, depExports, exports);
                                } catch (e) {
                                    err = e;
                                }
                            } else {
                                exports = context.execCb(id, factory, depExports, exports);
                            }

                            // Favor return value over exports. If node/cjs in play,
                            // then will not have a return value anyway. Favor
                            // module.exports assignment over exports object.
                            if (this.map.isDefine && exports === undefined) {
                                cjsModule = this.module;
                                if (cjsModule) {
                                    exports = cjsModule.exports;
                                } else if (this.usingExports) {
                                    //exports already set the defined value.
                                    exports = this.exports;
                                }
                            }

                            if (err) {
                                err.requireMap = this.map;
                                err.requireModules = this.map.isDefine ? [this.map.id] : null;
                                err.requireType = this.map.isDefine ? 'define' : 'require';
                                return onError((this.error = err));
                            }

                        } else {
                            //Just a literal value
                            exports = factory;
                        }

                        this.exports = exports;

                        if (this.map.isDefine && !this.ignore) {
                            defined[id] = exports;

                            if (req.onResourceLoad) {
                                var resLoadMaps = [];
                                each(this.depMaps, function (depMap) {
                                    resLoadMaps.push(depMap.normalizedMap || depMap);
                                });
                                req.onResourceLoad(context, this.map, resLoadMaps);
                            }
                        }

                        //Clean up
                        cleanRegistry(id);

                        this.defined = true;
                    }

                    //Finished the define stage. Allow calling check again
                    //to allow define notifications below in the case of a
                    //cycle.
                    this.defining = false;

                    if (this.defined && !this.defineEmitted) {
                        this.defineEmitted = true;
                        this.emit('defined', this.exports);
                        this.defineEmitComplete = true;
                    }

                }
            },

            callPlugin: function () {
                var map = this.map,
                    id = map.id,
                    //Map already normalized the prefix.
                    pluginMap = makeModuleMap(map.prefix);

                //Mark this as a dependency for this plugin, so it
                //can be traced for cycles.
                this.depMaps.push(pluginMap);

                on(pluginMap, 'defined', bind(this, function (plugin) {
                    var load, normalizedMap, normalizedMod,
                        bundleId = getOwn(bundlesMap, this.map.id),
                        name = this.map.name,
                        parentName = this.map.parentMap ? this.map.parentMap.name : null,
                        localRequire = context.makeRequire(map.parentMap, {
                            enableBuildCallback: true
                        });

                    //If current map is not normalized, wait for that
                    //normalized name to load instead of continuing.
                    if (this.map.unnormalized) {
                        //Normalize the ID if the plugin allows it.
                        if (plugin.normalize) {
                            name = plugin.normalize(name, function (name) {
                                return normalize(name, parentName, true);
                            }) || '';
                        }

                        //prefix and name should already be normalized, no need
                        //for applying map config again either.
                        normalizedMap = makeModuleMap(map.prefix + '!' + name,
                                                      this.map.parentMap,
                                                      true);
                        on(normalizedMap,
                            'defined', bind(this, function (value) {
                                this.map.normalizedMap = normalizedMap;
                                this.init([], function () { return value; }, null, {
                                    enabled: true,
                                    ignore: true
                                });
                            }));

                        normalizedMod = getOwn(registry, normalizedMap.id);
                        if (normalizedMod) {
                            //Mark this as a dependency for this plugin, so it
                            //can be traced for cycles.
                            this.depMaps.push(normalizedMap);

                            if (this.events.error) {
                                normalizedMod.on('error', bind(this, function (err) {
                                    this.emit('error', err);
                                }));
                            }
                            normalizedMod.enable();
                        }

                        return;
                    }

                    //If a paths config, then just load that file instead to
                    //resolve the plugin, as it is built into that paths layer.
                    if (bundleId) {
                        this.map.url = context.nameToUrl(bundleId);
                        this.load();
                        return;
                    }

                    load = bind(this, function (value) {
                        this.init([], function () { return value; }, null, {
                            enabled: true
                        });
                    });

                    load.error = bind(this, function (err) {
                        this.inited = true;
                        this.error = err;
                        err.requireModules = [id];

                        //Remove temp unnormalized modules for this module,
                        //since they will never be resolved otherwise now.
                        eachProp(registry, function (mod) {
                            if (mod.map.id.indexOf(id + '_unnormalized') === 0) {
                                cleanRegistry(mod.map.id);
                            }
                        });

                        onError(err);
                    });

                    //Allow plugins to load other code without having to know the
                    //context or how to 'complete' the load.
                    load.fromText = bind(this, function (text, textAlt) {
                        /*jslint evil: true */
                        var moduleName = map.name,
                            moduleMap = makeModuleMap(moduleName),
                            hasInteractive = useInteractive;

                        //As of 2.1.0, support just passing the text, to reinforce
                        //fromText only being called once per resource. Still
                        //support old style of passing moduleName but discard
                        //that moduleName in favor of the internal ref.
                        if (textAlt) {
                            text = textAlt;
                        }

                        //Turn off interactive script matching for IE for any define
                        //calls in the text, then turn it back on at the end.
                        if (hasInteractive) {
                            useInteractive = false;
                        }

                        //Prime the system by creating a module instance for
                        //it.
                        getModule(moduleMap);

                        //Transfer any config to this other module.
                        if (hasProp(config.config, id)) {
                            config.config[moduleName] = config.config[id];
                        }

                        try {
                            req.exec(text);
                        } catch (e) {
                            return onError(makeError('fromtexteval',
                                             'fromText eval for ' + id +
                                            ' failed: ' + e,
                                             e,
                                             [id]));
                        }

                        if (hasInteractive) {
                            useInteractive = true;
                        }

                        //Mark this as a dependency for the plugin
                        //resource
                        this.depMaps.push(moduleMap);

                        //Support anonymous modules.
                        context.completeLoad(moduleName);

                        //Bind the value of that module to the value for this
                        //resource ID.
                        localRequire([moduleName], load);
                    });

                    //Use parentName here since the plugin's name is not reliable,
                    //could be some weird string with no path that actually wants to
                    //reference the parentName's path.
                    plugin.load(map.name, localRequire, load, config);
                }));

                context.enable(pluginMap, this);
                this.pluginMaps[pluginMap.id] = pluginMap;
            },

            enable: function () {
                enabledRegistry[this.map.id] = this;
                this.enabled = true;

                //Set flag mentioning that the module is enabling,
                //so that immediate calls to the defined callbacks
                //for dependencies do not trigger inadvertent load
                //with the depCount still being zero.
                this.enabling = true;

                //Enable each dependency
                each(this.depMaps, bind(this, function (depMap, i) {
                    var id, mod, handler;

                    if (typeof depMap === 'string') {
                        //Dependency needs to be converted to a depMap
                        //and wired up to this module.
                        depMap = makeModuleMap(depMap,
                                               (this.map.isDefine ? this.map : this.map.parentMap),
                                               false,
                                               !this.skipMap);
                        this.depMaps[i] = depMap;

                        handler = getOwn(handlers, depMap.id);

                        if (handler) {
                            this.depExports[i] = handler(this);
                            return;
                        }

                        this.depCount += 1;

                        on(depMap, 'defined', bind(this, function (depExports) {
                            if (this.undefed) {
                                return;
                            }
                            this.defineDep(i, depExports);
                            this.check();
                        }));

                        if (this.errback) {
                            on(depMap, 'error', bind(this, this.errback));
                        } else if (this.events.error) {
                            // No direct errback on this module, but something
                            // else is listening for errors, so be sure to
                            // propagate the error correctly.
                            on(depMap, 'error', bind(this, function(err) {
                                this.emit('error', err);
                            }));
                        }
                    }

                    id = depMap.id;
                    mod = registry[id];

                    //Skip special modules like 'require', 'exports', 'module'
                    //Also, don't call enable if it is already enabled,
                    //important in circular dependency cases.
                    if (!hasProp(handlers, id) && mod && !mod.enabled) {
                        context.enable(depMap, this);
                    }
                }));

                //Enable each plugin that is used in
                //a dependency
                eachProp(this.pluginMaps, bind(this, function (pluginMap) {
                    var mod = getOwn(registry, pluginMap.id);
                    if (mod && !mod.enabled) {
                        context.enable(pluginMap, this);
                    }
                }));

                this.enabling = false;

                this.check();
            },

            on: function (name, cb) {
                var cbs = this.events[name];
                if (!cbs) {
                    cbs = this.events[name] = [];
                }
                cbs.push(cb);
            },

            emit: function (name, evt) {
                each(this.events[name], function (cb) {
                    cb(evt);
                });
                if (name === 'error') {
                    //Now that the error handler was triggered, remove
                    //the listeners, since this broken Module instance
                    //can stay around for a while in the registry.
                    delete this.events[name];
                }
            }
        };

        function callGetModule(args) {
            //Skip modules already defined.
            if (!hasProp(defined, args[0])) {
                getModule(makeModuleMap(args[0], null, true)).init(args[1], args[2]);
            }
        }

        function removeListener(node, func, name, ieName) {
            //Favor detachEvent because of IE9
            //issue, see attachEvent/addEventListener comment elsewhere
            //in this file.
            if (node.detachEvent && !isOpera) {
                //Probably IE. If not it will throw an error, which will be
                //useful to know.
                if (ieName) {
                    node.detachEvent(ieName, func);
                }
            } else {
                node.removeEventListener(name, func, false);
            }
        }

        /**
         * Given an event from a script node, get the requirejs info from it,
         * and then removes the event listeners on the node.
         * @param {Event} evt
         * @returns {Object}
         */
        function getScriptData(evt) {
            //Using currentTarget instead of target for Firefox 2.0's sake. Not
            //all old browsers will be supported, but this one was easy enough
            //to support and still makes sense.
            var node = evt.currentTarget || evt.srcElement;

            //Remove the listeners once here.
            removeListener(node, context.onScriptLoad, 'load', 'onreadystatechange');
            removeListener(node, context.onScriptError, 'error');

            return {
                node: node,
                id: node && node.getAttribute('data-requiremodule')
            };
        }

        function intakeDefines() {
            var args;

            //Any defined modules in the global queue, intake them now.
            takeGlobalQueue();

            //Make sure any remaining defQueue items get properly processed.
            while (defQueue.length) {
                args = defQueue.shift();
                if (args[0] === null) {
                    return onError(makeError('mismatch', 'Mismatched anonymous define() module: ' +
                        args[args.length - 1]));
                } else {
                    //args are id, deps, factory. Should be normalized by the
                    //define() function.
                    callGetModule(args);
                }
            }
            context.defQueueMap = {};
        }

        context = {
            config: config,
            contextName: contextName,
            registry: registry,
            defined: defined,
            urlFetched: urlFetched,
            defQueue: defQueue,
            defQueueMap: {},
            Module: Module,
            makeModuleMap: makeModuleMap,
            nextTick: req.nextTick,
            onError: onError,

            /**
             * Set a configuration for the context.
             * @param {Object} cfg config object to integrate.
             */
            configure: function (cfg) {
                //Make sure the baseUrl ends in a slash.
                if (cfg.baseUrl) {
                    if (cfg.baseUrl.charAt(cfg.baseUrl.length - 1) !== '/') {
                        cfg.baseUrl += '/';
                    }
                }

                // Convert old style urlArgs string to a function.
                if (typeof cfg.urlArgs === 'string') {
                    var urlArgs = cfg.urlArgs;
                    cfg.urlArgs = function(id, url) {
                        return (url.indexOf('?') === -1 ? '?' : '&') + urlArgs;
                    };
                }

                //Save off the paths since they require special processing,
                //they are additive.
                var shim = config.shim,
                    objs = {
                        paths: true,
                        bundles: true,
                        config: true,
                        map: true
                    };

                eachProp(cfg, function (value, prop) {
                    if (objs[prop]) {
                        if (!config[prop]) {
                            config[prop] = {};
                        }
                        mixin(config[prop], value, true, true);
                    } else {
                        config[prop] = value;
                    }
                });

                //Reverse map the bundles
                if (cfg.bundles) {
                    eachProp(cfg.bundles, function (value, prop) {
                        each(value, function (v) {
                            if (v !== prop) {
                                bundlesMap[v] = prop;
                            }
                        });
                    });
                }

                //Merge shim
                if (cfg.shim) {
                    eachProp(cfg.shim, function (value, id) {
                        //Normalize the structure
                        if (isArray(value)) {
                            value = {
                                deps: value
                            };
                        }
                        if ((value.exports || value.init) && !value.exportsFn) {
                            value.exportsFn = context.makeShimExports(value);
                        }
                        shim[id] = value;
                    });
                    config.shim = shim;
                }

                //Adjust packages if necessary.
                if (cfg.packages) {
                    each(cfg.packages, function (pkgObj) {
                        var location, name;

                        pkgObj = typeof pkgObj === 'string' ? {name: pkgObj} : pkgObj;

                        name = pkgObj.name;
                        location = pkgObj.location;
                        if (location) {
                            config.paths[name] = pkgObj.location;
                        }

                        //Save pointer to main module ID for pkg name.
                        //Remove leading dot in main, so main paths are normalized,
                        //and remove any trailing .js, since different package
                        //envs have different conventions: some use a module name,
                        //some use a file name.
                        config.pkgs[name] = pkgObj.name + '/' + (pkgObj.main || 'main')
                                     .replace(currDirRegExp, '')
                                     .replace(jsSuffixRegExp, '');
                    });
                }

                //If there are any "waiting to execute" modules in the registry,
                //update the maps for them, since their info, like URLs to load,
                //may have changed.
                eachProp(registry, function (mod, id) {
                    //If module already has init called, since it is too
                    //late to modify them, and ignore unnormalized ones
                    //since they are transient.
                    if (!mod.inited && !mod.map.unnormalized) {
                        mod.map = makeModuleMap(id, null, true);
                    }
                });

                //If a deps array or a config callback is specified, then call
                //require with those args. This is useful when require is defined as a
                //config object before require.js is loaded.
                if (cfg.deps || cfg.callback) {
                    context.require(cfg.deps || [], cfg.callback);
                }
            },

            makeShimExports: function (value) {
                function fn() {
                    var ret;
                    if (value.init) {
                        ret = value.init.apply(global, arguments);
                    }
                    return ret || (value.exports && getGlobal(value.exports));
                }
                return fn;
            },

            makeRequire: function (relMap, options) {
                options = options || {};

                function localRequire(deps, callback, errback) {
                    var id, map, requireMod;

                    if (options.enableBuildCallback && callback && isFunction(callback)) {
                        callback.__requireJsBuild = true;
                    }

                    if (typeof deps === 'string') {
                        if (isFunction(callback)) {
                            //Invalid call
                            return onError(makeError('requireargs', 'Invalid require call'), errback);
                        }

                        //If require|exports|module are requested, get the
                        //value for them from the special handlers. Caveat:
                        //this only works while module is being defined.
                        if (relMap && hasProp(handlers, deps)) {
                            return handlers[deps](registry[relMap.id]);
                        }

                        //Synchronous access to one module. If require.get is
                        //available (as in the Node adapter), prefer that.
                        if (req.get) {
                            return req.get(context, deps, relMap, localRequire);
                        }

                        //Normalize module name, if it contains . or ..
                        map = makeModuleMap(deps, relMap, false, true);
                        id = map.id;

                        if (!hasProp(defined, id)) {
                            return onError(makeError('notloaded', 'Module name "' +
                                        id +
                                        '" has not been loaded yet for context: ' +
                                        contextName +
                                        (relMap ? '' : '. Use require([])')));
                        }
                        return defined[id];
                    }

                    //Grab defines waiting in the global queue.
                    intakeDefines();

                    //Mark all the dependencies as needing to be loaded.
                    context.nextTick(function () {
                        //Some defines could have been added since the
                        //require call, collect them.
                        intakeDefines();

                        requireMod = getModule(makeModuleMap(null, relMap));

                        //Store if map config should be applied to this require
                        //call for dependencies.
                        requireMod.skipMap = options.skipMap;

                        requireMod.init(deps, callback, errback, {
                            enabled: true
                        });

                        checkLoaded();
                    });

                    return localRequire;
                }

                mixin(localRequire, {
                    isBrowser: isBrowser,

                    /**
                     * Converts a module name + .extension into an URL path.
                     * *Requires* the use of a module name. It does not support using
                     * plain URLs like nameToUrl.
                     */
                    toUrl: function (moduleNamePlusExt) {
                        var ext,
                            index = moduleNamePlusExt.lastIndexOf('.'),
                            segment = moduleNamePlusExt.split('/')[0],
                            isRelative = segment === '.' || segment === '..';

                        //Have a file extension alias, and it is not the
                        //dots from a relative path.
                        if (index !== -1 && (!isRelative || index > 1)) {
                            ext = moduleNamePlusExt.substring(index, moduleNamePlusExt.length);
                            moduleNamePlusExt = moduleNamePlusExt.substring(0, index);
                        }

                        return context.nameToUrl(normalize(moduleNamePlusExt,
                                                relMap && relMap.id, true), ext,  true);
                    },

                    defined: function (id) {
                        return hasProp(defined, makeModuleMap(id, relMap, false, true).id);
                    },

                    specified: function (id) {
                        id = makeModuleMap(id, relMap, false, true).id;
                        return hasProp(defined, id) || hasProp(registry, id);
                    }
                });

                //Only allow undef on top level require calls
                if (!relMap) {
                    localRequire.undef = function (id) {
                        //Bind any waiting define() calls to this context,
                        //fix for #408
                        takeGlobalQueue();

                        var map = makeModuleMap(id, relMap, true),
                            mod = getOwn(registry, id);

                        mod.undefed = true;
                        removeScript(id);

                        delete defined[id];
                        delete urlFetched[map.url];
                        delete undefEvents[id];

                        //Clean queued defines too. Go backwards
                        //in array so that the splices do not
                        //mess up the iteration.
                        eachReverse(defQueue, function(args, i) {
                            if (args[0] === id) {
                                defQueue.splice(i, 1);
                            }
                        });
                        delete context.defQueueMap[id];

                        if (mod) {
                            //Hold on to listeners in case the
                            //module will be attempted to be reloaded
                            //using a different config.
                            if (mod.events.defined) {
                                undefEvents[id] = mod.events;
                            }

                            cleanRegistry(id);
                        }
                    };
                }

                return localRequire;
            },

            /**
             * Called to enable a module if it is still in the registry
             * awaiting enablement. A second arg, parent, the parent module,
             * is passed in for context, when this method is overridden by
             * the optimizer. Not shown here to keep code compact.
             */
            enable: function (depMap) {
                var mod = getOwn(registry, depMap.id);
                if (mod) {
                    getModule(depMap).enable();
                }
            },

            /**
             * Internal method used by environment adapters to complete a load event.
             * A load event could be a script load or just a load pass from a synchronous
             * load call.
             * @param {String} moduleName the name of the module to potentially complete.
             */
            completeLoad: function (moduleName) {
                var found, args, mod,
                    shim = getOwn(config.shim, moduleName) || {},
                    shExports = shim.exports;

                takeGlobalQueue();

                while (defQueue.length) {
                    args = defQueue.shift();
                    if (args[0] === null) {
                        args[0] = moduleName;
                        //If already found an anonymous module and bound it
                        //to this name, then this is some other anon module
                        //waiting for its completeLoad to fire.
                        if (found) {
                            break;
                        }
                        found = true;
                    } else if (args[0] === moduleName) {
                        //Found matching define call for this script!
                        found = true;
                    }

                    callGetModule(args);
                }
                context.defQueueMap = {};

                //Do this after the cycle of callGetModule in case the result
                //of those calls/init calls changes the registry.
                mod = getOwn(registry, moduleName);

                if (!found && !hasProp(defined, moduleName) && mod && !mod.inited) {
                    if (config.enforceDefine && (!shExports || !getGlobal(shExports))) {
                        if (hasPathFallback(moduleName)) {
                            return;
                        } else {
                            return onError(makeError('nodefine',
                                             'No define call for ' + moduleName,
                                             null,
                                             [moduleName]));
                        }
                    } else {
                        //A script that does not call define(), so just simulate
                        //the call for it.
                        callGetModule([moduleName, (shim.deps || []), shim.exportsFn]);
                    }
                }

                checkLoaded();
            },

            /**
             * Converts a module name to a file path. Supports cases where
             * moduleName may actually be just an URL.
             * Note that it **does not** call normalize on the moduleName,
             * it is assumed to have already been normalized. This is an
             * internal API, not a public one. Use toUrl for the public API.
             */
            nameToUrl: function (moduleName, ext, skipExt) {
                var paths, syms, i, parentModule, url,
                    parentPath, bundleId,
                    pkgMain = getOwn(config.pkgs, moduleName);

                if (pkgMain) {
                    moduleName = pkgMain;
                }

                bundleId = getOwn(bundlesMap, moduleName);

                if (bundleId) {
                    return context.nameToUrl(bundleId, ext, skipExt);
                }

                //If a colon is in the URL, it indicates a protocol is used and it is just
                //an URL to a file, or if it starts with a slash, contains a query arg (i.e. ?)
                //or ends with .js, then assume the user meant to use an url and not a module id.
                //The slash is important for protocol-less URLs as well as full paths.
                if (req.jsExtRegExp.test(moduleName)) {
                    //Just a plain path, not module name lookup, so just return it.
                    //Add extension if it is included. This is a bit wonky, only non-.js things pass
                    //an extension, this method probably needs to be reworked.
                    url = moduleName + (ext || '');
                } else {
                    //A module that needs to be converted to a path.
                    paths = config.paths;

                    syms = moduleName.split('/');
                    //For each module name segment, see if there is a path
                    //registered for it. Start with most specific name
                    //and work up from it.
                    for (i = syms.length; i > 0; i -= 1) {
                        parentModule = syms.slice(0, i).join('/');

                        parentPath = getOwn(paths, parentModule);
                        if (parentPath) {
                            //If an array, it means there are a few choices,
                            //Choose the one that is desired
                            if (isArray(parentPath)) {
                                parentPath = parentPath[0];
                            }
                            syms.splice(0, i, parentPath);
                            break;
                        }
                    }

                    //Join the path parts together, then figure out if baseUrl is needed.
                    url = syms.join('/');
                    url += (ext || (/^data\:|^blob\:|\?/.test(url) || skipExt ? '' : '.js'));
                    url = (url.charAt(0) === '/' || url.match(/^[\w\+\.\-]+:/) ? '' : config.baseUrl) + url;
                }

                return config.urlArgs && !/^blob\:/.test(url) ?
                       url + config.urlArgs(moduleName, url) : url;
            },

            //Delegates to req.load. Broken out as a separate function to
            //allow overriding in the optimizer.
            load: function (id, url) {
                req.load(context, id, url);
            },

            /**
             * Executes a module callback function. Broken out as a separate function
             * solely to allow the build system to sequence the files in the built
             * layer in the right sequence.
             *
             * @private
             */
            execCb: function (name, callback, args, exports) {
                return callback.apply(exports, args);
            },

            /**
             * callback for script loads, used to check status of loading.
             *
             * @param {Event} evt the event from the browser for the script
             * that was loaded.
             */
            onScriptLoad: function (evt) {
                //Using currentTarget instead of target for Firefox 2.0's sake. Not
                //all old browsers will be supported, but this one was easy enough
                //to support and still makes sense.
                if (evt.type === 'load' ||
                        (readyRegExp.test((evt.currentTarget || evt.srcElement).readyState))) {
                    //Reset interactive script so a script node is not held onto for
                    //to long.
                    interactiveScript = null;

                    //Pull out the name of the module and the context.
                    var data = getScriptData(evt);
                    context.completeLoad(data.id);
                }
            },

            /**
             * Callback for script errors.
             */
            onScriptError: function (evt) {
                var data = getScriptData(evt);
                if (!hasPathFallback(data.id)) {
                    var parents = [];
                    eachProp(registry, function(value, key) {
                        if (key.indexOf('_@r') !== 0) {
                            each(value.depMaps, function(depMap) {
                                if (depMap.id === data.id) {
                                    parents.push(key);
                                    return true;
                                }
                            });
                        }
                    });
                    return onError(makeError('scripterror', 'Script error for "' + data.id +
                                             (parents.length ?
                                             '", needed by: ' + parents.join(', ') :
                                             '"'), evt, [data.id]));
                }
            }
        };

        context.require = context.makeRequire();
        return context;
    }

    /**
     * Main entry point.
     *
     * If the only argument to require is a string, then the module that
     * is represented by that string is fetched for the appropriate context.
     *
     * If the first argument is an array, then it will be treated as an array
     * of dependency string names to fetch. An optional function callback can
     * be specified to execute when all of those dependencies are available.
     *
     * Make a local req variable to help Caja compliance (it assumes things
     * on a require that are not standardized), and to give a short
     * name for minification/local scope use.
     */
    req = requirejs = function (deps, callback, errback, optional) {

        //Find the right context, use default
        var context, config,
            contextName = defContextName;

        // Determine if have config object in the call.
        if (!isArray(deps) && typeof deps !== 'string') {
            // deps is a config object
            config = deps;
            if (isArray(callback)) {
                // Adjust args if there are dependencies
                deps = callback;
                callback = errback;
                errback = optional;
            } else {
                deps = [];
            }
        }

        if (config && config.context) {
            contextName = config.context;
        }

        context = getOwn(contexts, contextName);
        if (!context) {
            context = contexts[contextName] = req.s.newContext(contextName);
        }

        if (config) {
            context.configure(config);
        }

        return context.require(deps, callback, errback);
    };

    /**
     * Support require.config() to make it easier to cooperate with other
     * AMD loaders on globally agreed names.
     */
    req.config = function (config) {
        return req(config);
    };

    /**
     * Execute something after the current tick
     * of the event loop. Override for other envs
     * that have a better solution than setTimeout.
     * @param  {Function} fn function to execute later.
     */
    req.nextTick = typeof setTimeout !== 'undefined' ? function (fn) {
        setTimeout(fn, 4);
    } : function (fn) { fn(); };

    /**
     * Export require as a global, but only if it does not already exist.
     */
    if (!require) {
        require = req;
    }

    req.version = version;

    //Used to filter out dependencies that are already paths.
    req.jsExtRegExp = /^\/|:|\?|\.js$/;
    req.isBrowser = isBrowser;
    s = req.s = {
        contexts: contexts,
        newContext: newContext
    };

    //Create default context.
    req({});

    //Exports some context-sensitive methods on global require.
    each([
        'toUrl',
        'undef',
        'defined',
        'specified'
    ], function (prop) {
        //Reference from contexts instead of early binding to default context,
        //so that during builds, the latest instance of the default context
        //with its config gets used.
        req[prop] = function () {
            var ctx = contexts[defContextName];
            return ctx.require[prop].apply(ctx, arguments);
        };
    });

    if (isBrowser) {
        head = s.head = document.getElementsByTagName('head')[0];
        //If BASE tag is in play, using appendChild is a problem for IE6.
        //When that browser dies, this can be removed. Details in this jQuery bug:
        //http://dev.jquery.com/ticket/2709
        baseElement = document.getElementsByTagName('base')[0];
        if (baseElement) {
            head = s.head = baseElement.parentNode;
        }
    }

    /**
     * Any errors that require explicitly generates will be passed to this
     * function. Intercept/override it if you want custom error handling.
     * @param {Error} err the error object.
     */
    req.onError = defaultOnError;

    /**
     * Creates the node for the load command. Only used in browser envs.
     */
    req.createNode = function (config, moduleName, url) {
        var node = config.xhtml ?
                document.createElementNS('http://www.w3.org/1999/xhtml', 'html:script') :
                document.createElement('script');
        node.type = config.scriptType || 'text/javascript';
        node.charset = 'utf-8';
        node.async = true;
        return node;
    };

    /**
     * Does the request to load a module for the browser case.
     * Make this a separate function to allow other environments
     * to override it.
     *
     * @param {Object} context the require context to find state.
     * @param {String} moduleName the name of the module.
     * @param {Object} url the URL to the module.
     */
    req.load = function (context, moduleName, url) {
        var config = (context && context.config) || {},
            node;
        if (isBrowser) {
            //In the browser so use a script tag
            node = req.createNode(config, moduleName, url);

            node.setAttribute('data-requirecontext', context.contextName);
            node.setAttribute('data-requiremodule', moduleName);

            //Set up load listener. Test attachEvent first because IE9 has
            //a subtle issue in its addEventListener and script onload firings
            //that do not match the behavior of all other browsers with
            //addEventListener support, which fire the onload event for a
            //script right after the script execution. See:
            //https://connect.microsoft.com/IE/feedback/details/648057/script-onload-event-is-not-fired-immediately-after-script-execution
            //UNFORTUNATELY Opera implements attachEvent but does not follow the script
            //script execution mode.
            if (node.attachEvent &&
                    //Check if node.attachEvent is artificially added by custom script or
                    //natively supported by browser
                    //read https://github.com/requirejs/requirejs/issues/187
                    //if we can NOT find [native code] then it must NOT natively supported.
                    //in IE8, node.attachEvent does not have toString()
                    //Note the test for "[native code" with no closing brace, see:
                    //https://github.com/requirejs/requirejs/issues/273
                    !(node.attachEvent.toString && node.attachEvent.toString().indexOf('[native code') < 0) &&
                    !isOpera) {
                //Probably IE. IE (at least 6-8) do not fire
                //script onload right after executing the script, so
                //we cannot tie the anonymous define call to a name.
                //However, IE reports the script as being in 'interactive'
                //readyState at the time of the define call.
                useInteractive = true;

                node.attachEvent('onreadystatechange', context.onScriptLoad);
                //It would be great to add an error handler here to catch
                //404s in IE9+. However, onreadystatechange will fire before
                //the error handler, so that does not help. If addEventListener
                //is used, then IE will fire error before load, but we cannot
                //use that pathway given the connect.microsoft.com issue
                //mentioned above about not doing the 'script execute,
                //then fire the script load event listener before execute
                //next script' that other browsers do.
                //Best hope: IE10 fixes the issues,
                //and then destroys all installs of IE 6-9.
                //node.attachEvent('onerror', context.onScriptError);
            } else {
                node.addEventListener('load', context.onScriptLoad, false);
                node.addEventListener('error', context.onScriptError, false);
            }
            node.src = url;

            //Calling onNodeCreated after all properties on the node have been
            //set, but before it is placed in the DOM.
            if (config.onNodeCreated) {
                config.onNodeCreated(node, config, moduleName, url);
            }

            //For some cache cases in IE 6-8, the script executes before the end
            //of the appendChild execution, so to tie an anonymous define
            //call to the module name (which is stored on the node), hold on
            //to a reference to this node, but clear after the DOM insertion.
            currentlyAddingScript = node;
            if (baseElement) {
                head.insertBefore(node, baseElement);
            } else {
                head.appendChild(node);
            }
            currentlyAddingScript = null;

            return node;
        } else if (isWebWorker) {
            try {
                //In a web worker, use importScripts. This is not a very
                //efficient use of importScripts, importScripts will block until
                //its script is downloaded and evaluated. However, if web workers
                //are in play, the expectation is that a build has been done so
                //that only one script needs to be loaded anyway. This may need
                //to be reevaluated if other use cases become common.

                // Post a task to the event loop to work around a bug in WebKit
                // where the worker gets garbage-collected after calling
                // importScripts(): https://webkit.org/b/153317
                setTimeout(function() {}, 0);
                importScripts(url);

                //Account for anonymous modules
                context.completeLoad(moduleName);
            } catch (e) {
                context.onError(makeError('importscripts',
                                'importScripts failed for ' +
                                    moduleName + ' at ' + url,
                                e,
                                [moduleName]));
            }
        }
    };

    function getInteractiveScript() {
        if (interactiveScript && interactiveScript.readyState === 'interactive') {
            return interactiveScript;
        }

        eachReverse(scripts(), function (script) {
            if (script.readyState === 'interactive') {
                return (interactiveScript = script);
            }
        });
        return interactiveScript;
    }

    //Look for a data-main script attribute, which could also adjust the baseUrl.
    if (isBrowser && !cfg.skipDataMain) {
        //Figure out baseUrl. Get it from the script tag with require.js in it.
        eachReverse(scripts(), function (script) {
            //Set the 'head' where we can append children by
            //using the script's parent.
            if (!head) {
                head = script.parentNode;
            }

            //Look for a data-main attribute to set main script for the page
            //to load. If it is there, the path to data main becomes the
            //baseUrl, if it is not already set.
            dataMain = script.getAttribute('data-main');
            if (dataMain) {
                //Preserve dataMain in case it is a path (i.e. contains '?')
                mainScript = dataMain;

                //Set final baseUrl if there is not already an explicit one,
                //but only do so if the data-main value is not a loader plugin
                //module ID.
                if (!cfg.baseUrl && mainScript.indexOf('!') === -1) {
                    //Pull off the directory of data-main for use as the
                    //baseUrl.
                    src = mainScript.split('/');
                    mainScript = src.pop();
                    subPath = src.length ? src.join('/')  + '/' : './';

                    cfg.baseUrl = subPath;
                }

                //Strip off any trailing .js since mainScript is now
                //like a module name.
                mainScript = mainScript.replace(jsSuffixRegExp, '');

                //If mainScript is still a path, fall back to dataMain
                if (req.jsExtRegExp.test(mainScript)) {
                    mainScript = dataMain;
                }

                //Put the data-main script in the files to load.
                cfg.deps = cfg.deps ? cfg.deps.concat(mainScript) : [mainScript];

                return true;
            }
        });
    }

    /**
     * The function that handles definitions of modules. Differs from
     * require() in that a string for the module should be the first argument,
     * and the function to execute after dependencies are loaded should
     * return a value to define the module corresponding to the first argument's
     * name.
     */
    define = function (name, deps, callback) {
        var node, context;

        //Allow for anonymous modules
        if (typeof name !== 'string') {
            //Adjust args appropriately
            callback = deps;
            deps = name;
            name = null;
        }

        //This module may not have dependencies
        if (!isArray(deps)) {
            callback = deps;
            deps = null;
        }

        //If no name, and callback is a function, then figure out if it a
        //CommonJS thing with dependencies.
        if (!deps && isFunction(callback)) {
            deps = [];
            //Remove comments from the callback string,
            //look for require calls, and pull them into the dependencies,
            //but only if there are function args.
            if (callback.length) {
                callback
                    .toString()
                    .replace(commentRegExp, commentReplace)
                    .replace(cjsRequireRegExp, function (match, dep) {
                        deps.push(dep);
                    });

                //May be a CommonJS thing even without require calls, but still
                //could use exports, and module. Avoid doing exports and module
                //work though if it just needs require.
                //REQUIRES the function to expect the CommonJS variables in the
                //order listed below.
                deps = (callback.length === 1 ? ['require'] : ['require', 'exports', 'module']).concat(deps);
            }
        }

        //If in IE 6-8 and hit an anonymous define() call, do the interactive
        //work.
        if (useInteractive) {
            node = currentlyAddingScript || getInteractiveScript();
            if (node) {
                if (!name) {
                    name = node.getAttribute('data-requiremodule');
                }
                context = contexts[node.getAttribute('data-requirecontext')];
            }
        }

        //Always save off evaluating the def call until the script onload handler.
        //This allows multiple modules to be in a file without prematurely
        //tracing dependencies, and allows for anonymous module support,
        //where the module name is not known until the script onload event
        //occurs. If no context, use the global queue, and get it processed
        //in the onscript load callback.
        if (context) {
            context.defQueue.push([name, deps, callback]);
            context.defQueueMap[name] = true;
        } else {
            globalDefQueue.push([name, deps, callback]);
        }
    };

    define.amd = {
        jQuery: true
    };

    /**
     * Executes the text. Normally just uses eval, but can be modified
     * to use a better, environment-specific call. Only used for transpiling
     * loader plugins, not for plain JS modules.
     * @param {String} text the text to execute/evaluate.
     */
    req.exec = function (text) {
        /*jslint evil: true */
        return eval(text);
    };

    //Set up with config info.
    req(cfg);
}(this, (typeof setTimeout === 'undefined' ? undefined : setTimeout)));

// Copyright 2014 Google Inc. All rights reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
//     You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
//     See the License for the specific language governing permissions and
// limitations under the License.

!function(){var a={},b={},c={};!function(a,b){function c(a){if("number"==typeof a)return a;var b={};for(var c in a)b[c]=a[c];return b}function d(){this._delay=0,this._endDelay=0,this._fill="none",this._iterationStart=0,this._iterations=1,this._duration=0,this._playbackRate=1,this._direction="normal",this._easing="linear",this._easingFunction=x}function e(){return a.isDeprecated("Invalid timing inputs","2016-03-02","TypeError exceptions will be thrown instead.",!0)}function f(b,c,e){var f=new d;return c&&(f.fill="both",f.duration="auto"),"number"!=typeof b||isNaN(b)?void 0!==b&&Object.getOwnPropertyNames(b).forEach(function(c){if("auto"!=b[c]){if(("number"==typeof f[c]||"duration"==c)&&("number"!=typeof b[c]||isNaN(b[c])))return;if("fill"==c&&-1==v.indexOf(b[c]))return;if("direction"==c&&-1==w.indexOf(b[c]))return;if("playbackRate"==c&&1!==b[c]&&a.isDeprecated("AnimationEffectTiming.playbackRate","2014-11-28","Use Animation.playbackRate instead."))return;f[c]=b[c]}}):f.duration=b,f}function g(a){return"number"==typeof a&&(a=isNaN(a)?{duration:0}:{duration:a}),a}function h(b,c){return b=a.numericTimingToObject(b),f(b,c)}function i(a,b,c,d){return a<0||a>1||c<0||c>1?x:function(e){function f(a,b,c){return 3*a*(1-c)*(1-c)*c+3*b*(1-c)*c*c+c*c*c}if(e<=0){var g=0;return a>0?g=b/a:!b&&c>0&&(g=d/c),g*e}if(e>=1){var h=0;return c<1?h=(d-1)/(c-1):1==c&&a<1&&(h=(b-1)/(a-1)),1+h*(e-1)}for(var i=0,j=1;i<j;){var k=(i+j)/2,l=f(a,c,k);if(Math.abs(e-l)<1e-5)return f(b,d,k);l<e?i=k:j=k}return f(b,d,k)}}function j(a,b){return function(c){if(c>=1)return 1;var d=1/a;return(c+=b*d)-c%d}}function k(a){C||(C=document.createElement("div").style),C.animationTimingFunction="",C.animationTimingFunction=a;var b=C.animationTimingFunction;if(""==b&&e())throw new TypeError(a+" is not a valid value for easing");return b}function l(a){if("linear"==a)return x;var b=E.exec(a);if(b)return i.apply(this,b.slice(1).map(Number));var c=F.exec(a);if(c)return j(Number(c[1]),A);var d=G.exec(a);return d?j(Number(d[1]),{start:y,middle:z,end:A}[d[2]]):B[a]||x}function m(a){return Math.abs(n(a)/a.playbackRate)}function n(a){return 0===a.duration||0===a.iterations?0:a.duration*a.iterations}function o(a,b,c){if(null==b)return H;var d=c.delay+a+c.endDelay;return b<Math.min(c.delay,d)?I:b>=Math.min(c.delay+a,d)?J:K}function p(a,b,c,d,e){switch(d){case I:return"backwards"==b||"both"==b?0:null;case K:return c-e;case J:return"forwards"==b||"both"==b?a:null;case H:return null}}function q(a,b,c,d,e){var f=e;return 0===a?b!==I&&(f+=c):f+=d/a,f}function r(a,b,c,d,e,f){var g=a===1/0?b%1:a%1;return 0!==g||c!==J||0===d||0===e&&0!==f||(g=1),g}function s(a,b,c,d){return a===J&&b===1/0?1/0:1===c?Math.floor(d)-1:Math.floor(d)}function t(a,b,c){var d=a;if("normal"!==a&&"reverse"!==a){var e=b;"alternate-reverse"===a&&(e+=1),d="normal",e!==1/0&&e%2!=0&&(d="reverse")}return"normal"===d?c:1-c}function u(a,b,c){var d=o(a,b,c),e=p(a,c.fill,b,d,c.delay);if(null===e)return null;var f=q(c.duration,d,c.iterations,e,c.iterationStart),g=r(f,c.iterationStart,d,c.iterations,e,c.duration),h=s(d,c.iterations,g,f),i=t(c.direction,h,g);return c._easingFunction(i)}var v="backwards|forwards|both|none".split("|"),w="reverse|alternate|alternate-reverse".split("|"),x=function(a){return a};d.prototype={_setMember:function(b,c){this["_"+b]=c,this._effect&&(this._effect._timingInput[b]=c,this._effect._timing=a.normalizeTimingInput(this._effect._timingInput),this._effect.activeDuration=a.calculateActiveDuration(this._effect._timing),this._effect._animation&&this._effect._animation._rebuildUnderlyingAnimation())},get playbackRate(){return this._playbackRate},set delay(a){this._setMember("delay",a)},get delay(){return this._delay},set endDelay(a){this._setMember("endDelay",a)},get endDelay(){return this._endDelay},set fill(a){this._setMember("fill",a)},get fill(){return this._fill},set iterationStart(a){if((isNaN(a)||a<0)&&e())throw new TypeError("iterationStart must be a non-negative number, received: "+a);this._setMember("iterationStart",a)},get iterationStart(){return this._iterationStart},set duration(a){if("auto"!=a&&(isNaN(a)||a<0)&&e())throw new TypeError("duration must be non-negative or auto, received: "+a);this._setMember("duration",a)},get duration(){return this._duration},set direction(a){this._setMember("direction",a)},get direction(){return this._direction},set easing(a){this._easingFunction=l(k(a)),this._setMember("easing",a)},get easing(){return this._easing},set iterations(a){if((isNaN(a)||a<0)&&e())throw new TypeError("iterations must be non-negative, received: "+a);this._setMember("iterations",a)},get iterations(){return this._iterations}};var y=1,z=.5,A=0,B={ease:i(.25,.1,.25,1),"ease-in":i(.42,0,1,1),"ease-out":i(0,0,.58,1),"ease-in-out":i(.42,0,.58,1),"step-start":j(1,y),"step-middle":j(1,z),"step-end":j(1,A)},C=null,D="\\s*(-?\\d+\\.?\\d*|-?\\.\\d+)\\s*",E=new RegExp("cubic-bezier\\("+D+","+D+","+D+","+D+"\\)"),F=/steps\(\s*(\d+)\s*\)/,G=/steps\(\s*(\d+)\s*,\s*(start|middle|end)\s*\)/,H=0,I=1,J=2,K=3;a.cloneTimingInput=c,a.makeTiming=f,a.numericTimingToObject=g,a.normalizeTimingInput=h,a.calculateActiveDuration=m,a.calculateIterationProgress=u,a.calculatePhase=o,a.normalizeEasing=k,a.parseEasingFunction=l}(a),function(a,b){function c(a,b){return a in k?k[a][b]||b:b}function d(a){return"display"===a||0===a.lastIndexOf("animation",0)||0===a.lastIndexOf("transition",0)}function e(a,b,e){if(!d(a)){var f=h[a];if(f){i.style[a]=b;for(var g in f){var j=f[g],k=i.style[j];e[j]=c(j,k)}}else e[a]=c(a,b)}}function f(a){var b=[];for(var c in a)if(!(c in["easing","offset","composite"])){var d=a[c];Array.isArray(d)||(d=[d]);for(var e,f=d.length,g=0;g<f;g++)e={},e.offset="offset"in a?a.offset:1==f?1:g/(f-1),"easing"in a&&(e.easing=a.easing),"composite"in a&&(e.composite=a.composite),e[c]=d[g],b.push(e)}return b.sort(function(a,b){return a.offset-b.offset}),b}function g(b){function c(){var a=d.length;null==d[a-1].offset&&(d[a-1].offset=1),a>1&&null==d[0].offset&&(d[0].offset=0);for(var b=0,c=d[0].offset,e=1;e<a;e++){var f=d[e].offset;if(null!=f){for(var g=1;g<e-b;g++)d[b+g].offset=c+(f-c)*g/(e-b);b=e,c=f}}}if(null==b)return[];window.Symbol&&Symbol.iterator&&Array.prototype.from&&b[Symbol.iterator]&&(b=Array.from(b)),Array.isArray(b)||(b=f(b));for(var d=b.map(function(b){var c={};for(var d in b){var f=b[d];if("offset"==d){if(null!=f){if(f=Number(f),!isFinite(f))throw new TypeError("Keyframe offsets must be numbers.");if(f<0||f>1)throw new TypeError("Keyframe offsets must be between 0 and 1.")}}else if("composite"==d){if("add"==f||"accumulate"==f)throw{type:DOMException.NOT_SUPPORTED_ERR,name:"NotSupportedError",message:"add compositing is not supported"};if("replace"!=f)throw new TypeError("Invalid composite mode "+f+".")}else f="easing"==d?a.normalizeEasing(f):""+f;e(d,f,c)}return void 0==c.offset&&(c.offset=null),void 0==c.easing&&(c.easing="linear"),c}),g=!0,h=-1/0,i=0;i<d.length;i++){var j=d[i].offset;if(null!=j){if(j<h)throw new TypeError("Keyframes are not loosely sorted by offset. Sort or specify offsets.");h=j}else g=!1}return d=d.filter(function(a){return a.offset>=0&&a.offset<=1}),g||c(),d}var h={background:["backgroundImage","backgroundPosition","backgroundSize","backgroundRepeat","backgroundAttachment","backgroundOrigin","backgroundClip","backgroundColor"],border:["borderTopColor","borderTopStyle","borderTopWidth","borderRightColor","borderRightStyle","borderRightWidth","borderBottomColor","borderBottomStyle","borderBottomWidth","borderLeftColor","borderLeftStyle","borderLeftWidth"],borderBottom:["borderBottomWidth","borderBottomStyle","borderBottomColor"],borderColor:["borderTopColor","borderRightColor","borderBottomColor","borderLeftColor"],borderLeft:["borderLeftWidth","borderLeftStyle","borderLeftColor"],borderRadius:["borderTopLeftRadius","borderTopRightRadius","borderBottomRightRadius","borderBottomLeftRadius"],borderRight:["borderRightWidth","borderRightStyle","borderRightColor"],borderTop:["borderTopWidth","borderTopStyle","borderTopColor"],borderWidth:["borderTopWidth","borderRightWidth","borderBottomWidth","borderLeftWidth"],flex:["flexGrow","flexShrink","flexBasis"],font:["fontFamily","fontSize","fontStyle","fontVariant","fontWeight","lineHeight"],margin:["marginTop","marginRight","marginBottom","marginLeft"],outline:["outlineColor","outlineStyle","outlineWidth"],padding:["paddingTop","paddingRight","paddingBottom","paddingLeft"]},i=document.createElementNS("http://www.w3.org/1999/xhtml","div"),j={thin:"1px",medium:"3px",thick:"5px"},k={borderBottomWidth:j,borderLeftWidth:j,borderRightWidth:j,borderTopWidth:j,fontSize:{"xx-small":"60%","x-small":"75%",small:"89%",medium:"100%",large:"120%","x-large":"150%","xx-large":"200%"},fontWeight:{normal:"400",bold:"700"},outlineWidth:j,textShadow:{none:"0px 0px 0px transparent"},boxShadow:{none:"0px 0px 0px 0px transparent"}};a.convertToArrayForm=f,a.normalizeKeyframes=g}(a),function(a){var b={};a.isDeprecated=function(a,c,d,e){var f=e?"are":"is",g=new Date,h=new Date(c);return h.setMonth(h.getMonth()+3),!(g<h&&(a in b||console.warn("Web Animations: "+a+" "+f+" deprecated and will stop working on "+h.toDateString()+". "+d),b[a]=!0,1))},a.deprecated=function(b,c,d,e){var f=e?"are":"is";if(a.isDeprecated(b,c,d,e))throw new Error(b+" "+f+" no longer supported. "+d)}}(a),function(){if(document.documentElement.animate){var c=document.documentElement.animate([],0),d=!0;if(c&&(d=!1,"play|currentTime|pause|reverse|playbackRate|cancel|finish|startTime|playState".split("|").forEach(function(a){void 0===c[a]&&(d=!0)})),!d)return}!function(a,b,c){function d(a){for(var b={},c=0;c<a.length;c++)for(var d in a[c])if("offset"!=d&&"easing"!=d&&"composite"!=d){var e={offset:a[c].offset,easing:a[c].easing,value:a[c][d]};b[d]=b[d]||[],b[d].push(e)}for(var f in b){var g=b[f];if(0!=g[0].offset||1!=g[g.length-1].offset)throw{type:DOMException.NOT_SUPPORTED_ERR,name:"NotSupportedError",message:"Partial keyframes are not supported"}}return b}function e(c){var d=[];for(var e in c)for(var f=c[e],g=0;g<f.length-1;g++){var h=g,i=g+1,j=f[h].offset,k=f[i].offset,l=j,m=k;0==g&&(l=-1/0,0==k&&(i=h)),g==f.length-2&&(m=1/0,1==j&&(h=i)),d.push({applyFrom:l,applyTo:m,startOffset:f[h].offset,endOffset:f[i].offset,easingFunction:a.parseEasingFunction(f[h].easing),property:e,interpolation:b.propertyInterpolation(e,f[h].value,f[i].value)})}return d.sort(function(a,b){return a.startOffset-b.startOffset}),d}b.convertEffectInput=function(c){var f=a.normalizeKeyframes(c),g=d(f),h=e(g);return function(a,c){if(null!=c)h.filter(function(a){return c>=a.applyFrom&&c<a.applyTo}).forEach(function(d){var e=c-d.startOffset,f=d.endOffset-d.startOffset,g=0==f?0:d.easingFunction(e/f);b.apply(a,d.property,d.interpolation(g))});else for(var d in g)"offset"!=d&&"easing"!=d&&"composite"!=d&&b.clear(a,d)}}}(a,b),function(a,b,c){function d(a){return a.replace(/-(.)/g,function(a,b){return b.toUpperCase()})}function e(a,b,c){h[c]=h[c]||[],h[c].push([a,b])}function f(a,b,c){for(var f=0;f<c.length;f++){e(a,b,d(c[f]))}}function g(c,e,f){var g=c;/-/.test(c)&&!a.isDeprecated("Hyphenated property names","2016-03-22","Use camelCase instead.",!0)&&(g=d(c)),"initial"!=e&&"initial"!=f||("initial"==e&&(e=i[g]),"initial"==f&&(f=i[g]));for(var j=e==f?[]:h[g],k=0;j&&k<j.length;k++){var l=j[k][0](e),m=j[k][0](f);if(void 0!==l&&void 0!==m){var n=j[k][1](l,m);if(n){var o=b.Interpolation.apply(null,n);return function(a){return 0==a?e:1==a?f:o(a)}}}}return b.Interpolation(!1,!0,function(a){return a?f:e})}var h={};b.addPropertiesHandler=f;var i={backgroundColor:"transparent",backgroundPosition:"0% 0%",borderBottomColor:"currentColor",borderBottomLeftRadius:"0px",borderBottomRightRadius:"0px",borderBottomWidth:"3px",borderLeftColor:"currentColor",borderLeftWidth:"3px",borderRightColor:"currentColor",borderRightWidth:"3px",borderSpacing:"2px",borderTopColor:"currentColor",borderTopLeftRadius:"0px",borderTopRightRadius:"0px",borderTopWidth:"3px",bottom:"auto",clip:"rect(0px, 0px, 0px, 0px)",color:"black",fontSize:"100%",fontWeight:"400",height:"auto",left:"auto",letterSpacing:"normal",lineHeight:"120%",marginBottom:"0px",marginLeft:"0px",marginRight:"0px",marginTop:"0px",maxHeight:"none",maxWidth:"none",minHeight:"0px",minWidth:"0px",opacity:"1.0",outlineColor:"invert",outlineOffset:"0px",outlineWidth:"3px",paddingBottom:"0px",paddingLeft:"0px",paddingRight:"0px",paddingTop:"0px",right:"auto",strokeDasharray:"none",strokeDashoffset:"0px",textIndent:"0px",textShadow:"0px 0px 0px transparent",top:"auto",transform:"",verticalAlign:"0px",visibility:"visible",width:"auto",wordSpacing:"normal",zIndex:"auto"};b.propertyInterpolation=g}(a,b),function(a,b,c){function d(b){var c=a.calculateActiveDuration(b),d=function(d){return a.calculateIterationProgress(c,d,b)};return d._totalDuration=b.delay+c+b.endDelay,d}b.KeyframeEffect=function(c,e,f,g){var h,i=d(a.normalizeTimingInput(f)),j=b.convertEffectInput(e),k=function(){j(c,h)};return k._update=function(a){return null!==(h=i(a))},k._clear=function(){j(c,null)},k._hasSameTarget=function(a){return c===a},k._target=c,k._totalDuration=i._totalDuration,k._id=g,k}}(a,b),function(a,b){a.apply=function(b,c,d){b.style[a.propertyName(c)]=d},a.clear=function(b,c){b.style[a.propertyName(c)]=""}}(b),function(a){window.Element.prototype.animate=function(b,c){var d="";return c&&c.id&&(d=c.id),a.timeline._play(a.KeyframeEffect(this,b,c,d))}}(b),function(a,b){function c(a,b,d){if("number"==typeof a&&"number"==typeof b)return a*(1-d)+b*d;if("boolean"==typeof a&&"boolean"==typeof b)return d<.5?a:b;if(a.length==b.length){for(var e=[],f=0;f<a.length;f++)e.push(c(a[f],b[f],d));return e}throw"Mismatched interpolation arguments "+a+":"+b}a.Interpolation=function(a,b,d){return function(e){return d(c(a,b,e))}}}(b),function(a,b,c){a.sequenceNumber=0;var d=function(a,b,c){this.target=a,this.currentTime=b,this.timelineTime=c,this.type="finish",this.bubbles=!1,this.cancelable=!1,this.currentTarget=a,this.defaultPrevented=!1,this.eventPhase=Event.AT_TARGET,this.timeStamp=Date.now()};b.Animation=function(b){this.id="",b&&b._id&&(this.id=b._id),this._sequenceNumber=a.sequenceNumber++,this._currentTime=0,this._startTime=null,this._paused=!1,this._playbackRate=1,this._inTimeline=!0,this._finishedFlag=!0,this.onfinish=null,this._finishHandlers=[],this._effect=b,this._inEffect=this._effect._update(0),this._idle=!0,this._currentTimePending=!1},b.Animation.prototype={_ensureAlive:function(){this.playbackRate<0&&0===this.currentTime?this._inEffect=this._effect._update(-1):this._inEffect=this._effect._update(this.currentTime),this._inTimeline||!this._inEffect&&this._finishedFlag||(this._inTimeline=!0,b.timeline._animations.push(this))},_tickCurrentTime:function(a,b){a!=this._currentTime&&(this._currentTime=a,this._isFinished&&!b&&(this._currentTime=this._playbackRate>0?this._totalDuration:0),this._ensureAlive())},get currentTime(){return this._idle||this._currentTimePending?null:this._currentTime},set currentTime(a){a=+a,isNaN(a)||(b.restart(),this._paused||null==this._startTime||(this._startTime=this._timeline.currentTime-a/this._playbackRate),this._currentTimePending=!1,this._currentTime!=a&&(this._idle&&(this._idle=!1,this._paused=!0),this._tickCurrentTime(a,!0),b.applyDirtiedAnimation(this)))},get startTime(){return this._startTime},set startTime(a){a=+a,isNaN(a)||this._paused||this._idle||(this._startTime=a,this._tickCurrentTime((this._timeline.currentTime-this._startTime)*this.playbackRate),b.applyDirtiedAnimation(this))},get playbackRate(){return this._playbackRate},set playbackRate(a){if(a!=this._playbackRate){var c=this.currentTime;this._playbackRate=a,this._startTime=null,"paused"!=this.playState&&"idle"!=this.playState&&(this._finishedFlag=!1,this._idle=!1,this._ensureAlive(),b.applyDirtiedAnimation(this)),null!=c&&(this.currentTime=c)}},get _isFinished(){return!this._idle&&(this._playbackRate>0&&this._currentTime>=this._totalDuration||this._playbackRate<0&&this._currentTime<=0)},get _totalDuration(){return this._effect._totalDuration},get playState(){return this._idle?"idle":null==this._startTime&&!this._paused&&0!=this.playbackRate||this._currentTimePending?"pending":this._paused?"paused":this._isFinished?"finished":"running"},_rewind:function(){if(this._playbackRate>=0)this._currentTime=0;else{if(!(this._totalDuration<1/0))throw new DOMException("Unable to rewind negative playback rate animation with infinite duration","InvalidStateError");this._currentTime=this._totalDuration}},play:function(){this._paused=!1,(this._isFinished||this._idle)&&(this._rewind(),this._startTime=null),this._finishedFlag=!1,this._idle=!1,this._ensureAlive(),b.applyDirtiedAnimation(this)},pause:function(){this._isFinished||this._paused||this._idle?this._idle&&(this._rewind(),this._idle=!1):this._currentTimePending=!0,this._startTime=null,this._paused=!0},finish:function(){this._idle||(this.currentTime=this._playbackRate>0?this._totalDuration:0,this._startTime=this._totalDuration-this.currentTime,this._currentTimePending=!1,b.applyDirtiedAnimation(this))},cancel:function(){this._inEffect&&(this._inEffect=!1,this._idle=!0,this._paused=!1,this._finishedFlag=!0,this._currentTime=0,this._startTime=null,this._effect._update(null),b.applyDirtiedAnimation(this))},reverse:function(){this.playbackRate*=-1,this.play()},addEventListener:function(a,b){"function"==typeof b&&"finish"==a&&this._finishHandlers.push(b)},removeEventListener:function(a,b){if("finish"==a){var c=this._finishHandlers.indexOf(b);c>=0&&this._finishHandlers.splice(c,1)}},_fireEvents:function(a){if(this._isFinished){if(!this._finishedFlag){var b=new d(this,this._currentTime,a),c=this._finishHandlers.concat(this.onfinish?[this.onfinish]:[]);setTimeout(function(){c.forEach(function(a){a.call(b.target,b)})},0),this._finishedFlag=!0}}else this._finishedFlag=!1},_tick:function(a,b){this._idle||this._paused||(null==this._startTime?b&&(this.startTime=a-this._currentTime/this.playbackRate):this._isFinished||this._tickCurrentTime((a-this._startTime)*this.playbackRate)),b&&(this._currentTimePending=!1,this._fireEvents(a))},get _needsTick(){return this.playState in{pending:1,running:1}||!this._finishedFlag},_targetAnimations:function(){var a=this._effect._target;return a._activeAnimations||(a._activeAnimations=[]),a._activeAnimations},_markTarget:function(){var a=this._targetAnimations();-1===a.indexOf(this)&&a.push(this)},_unmarkTarget:function(){var a=this._targetAnimations(),b=a.indexOf(this);-1!==b&&a.splice(b,1)}}}(a,b),function(a,b,c){function d(a){var b=j;j=[],a<q.currentTime&&(a=q.currentTime),q._animations.sort(e),q._animations=h(a,!0,q._animations)[0],b.forEach(function(b){b[1](a)}),g(),l=void 0}function e(a,b){return a._sequenceNumber-b._sequenceNumber}function f(){this._animations=[],this.currentTime=window.performance&&performance.now?performance.now():0}function g(){o.forEach(function(a){a()}),o.length=0}function h(a,c,d){p=!0,n=!1,b.timeline.currentTime=a,m=!1;var e=[],f=[],g=[],h=[];return d.forEach(function(b){b._tick(a,c),b._inEffect?(f.push(b._effect),b._markTarget()):(e.push(b._effect),b._unmarkTarget()),b._needsTick&&(m=!0);var d=b._inEffect||b._needsTick;b._inTimeline=d,d?g.push(b):h.push(b)}),o.push.apply(o,e),o.push.apply(o,f),m&&requestAnimationFrame(function(){}),p=!1,[g,h]}var i=window.requestAnimationFrame,j=[],k=0;window.requestAnimationFrame=function(a){var b=k++;return 0==j.length&&i(d),j.push([b,a]),b},window.cancelAnimationFrame=function(a){j.forEach(function(b){b[0]==a&&(b[1]=function(){})})},f.prototype={_play:function(c){c._timing=a.normalizeTimingInput(c.timing);var d=new b.Animation(c);return d._idle=!1,d._timeline=this,this._animations.push(d),b.restart(),b.applyDirtiedAnimation(d),d}};var l=void 0,m=!1,n=!1;b.restart=function(){return m||(m=!0,requestAnimationFrame(function(){}),n=!0),n},b.applyDirtiedAnimation=function(a){if(!p){a._markTarget();var c=a._targetAnimations();c.sort(e),h(b.timeline.currentTime,!1,c.slice())[1].forEach(function(a){var b=q._animations.indexOf(a);-1!==b&&q._animations.splice(b,1)}),g()}};var o=[],p=!1,q=new f;b.timeline=q}(a,b),function(a){function b(a,b){var c=a.exec(b);if(c)return c=a.ignoreCase?c[0].toLowerCase():c[0],[c,b.substr(c.length)]}function c(a,b){b=b.replace(/^\s*/,"");var c=a(b);if(c)return[c[0],c[1].replace(/^\s*/,"")]}function d(a,d,e){a=c.bind(null,a);for(var f=[];;){var g=a(e);if(!g)return[f,e];if(f.push(g[0]),e=g[1],!(g=b(d,e))||""==g[1])return[f,e];e=g[1]}}function e(a,b){for(var c=0,d=0;d<b.length&&(!/\s|,/.test(b[d])||0!=c);d++)if("("==b[d])c++;else if(")"==b[d]&&(c--,0==c&&d++,c<=0))break;var e=a(b.substr(0,d));return void 0==e?void 0:[e,b.substr(d)]}function f(a,b){for(var c=a,d=b;c&&d;)c>d?c%=d:d%=c;return c=a*b/(c+d)}function g(a){return function(b){var c=a(b);return c&&(c[0]=void 0),c}}function h(a,b){return function(c){return a(c)||[b,c]}}function i(b,c){for(var d=[],e=0;e<b.length;e++){var f=a.consumeTrimmed(b[e],c);if(!f||""==f[0])return;void 0!==f[0]&&d.push(f[0]),c=f[1]}if(""==c)return d}function j(a,b,c,d,e){for(var g=[],h=[],i=[],j=f(d.length,e.length),k=0;k<j;k++){var l=b(d[k%d.length],e[k%e.length]);if(!l)return;g.push(l[0]),h.push(l[1]),i.push(l[2])}return[g,h,function(b){var d=b.map(function(a,b){return i[b](a)}).join(c);return a?a(d):d}]}function k(a,b,c){for(var d=[],e=[],f=[],g=0,h=0;h<c.length;h++)if("function"==typeof c[h]){var i=c[h](a[g],b[g++]);d.push(i[0]),e.push(i[1]),f.push(i[2])}else!function(a){d.push(!1),e.push(!1),f.push(function(){return c[a]})}(h);return[d,e,function(a){for(var b="",c=0;c<a.length;c++)b+=f[c](a[c]);return b}]}a.consumeToken=b,a.consumeTrimmed=c,a.consumeRepeated=d,a.consumeParenthesised=e,a.ignore=g,a.optional=h,a.consumeList=i,a.mergeNestedRepeated=j.bind(null,null),a.mergeWrappedNestedRepeated=j,a.mergeList=k}(b),function(a){function b(b){function c(b){var c=a.consumeToken(/^inset/i,b);return c?(d.inset=!0,c):(c=a.consumeLengthOrPercent(b))?(d.lengths.push(c[0]),c):(c=a.consumeColor(b),c?(d.color=c[0],c):void 0)}var d={inset:!1,lengths:[],color:null},e=a.consumeRepeated(c,/^/,b);if(e&&e[0].length)return[d,e[1]]}function c(c){var d=a.consumeRepeated(b,/^,/,c);if(d&&""==d[1])return d[0]}function d(b,c){for(;b.lengths.length<Math.max(b.lengths.length,c.lengths.length);)b.lengths.push({px:0});for(;c.lengths.length<Math.max(b.lengths.length,c.lengths.length);)c.lengths.push({px:0});if(b.inset==c.inset&&!!b.color==!!c.color){for(var d,e=[],f=[[],0],g=[[],0],h=0;h<b.lengths.length;h++){var i=a.mergeDimensions(b.lengths[h],c.lengths[h],2==h);f[0].push(i[0]),g[0].push(i[1]),e.push(i[2])}if(b.color&&c.color){var j=a.mergeColors(b.color,c.color);f[1]=j[0],g[1]=j[1],d=j[2]}return[f,g,function(a){for(var c=b.inset?"inset ":" ",f=0;f<e.length;f++)c+=e[f](a[0][f])+" ";return d&&(c+=d(a[1])),c}]}}function e(b,c,d,e){function f(a){return{inset:a,color:[0,0,0,0],lengths:[{px:0},{px:0},{px:0},{px:0}]}}for(var g=[],h=[],i=0;i<d.length||i<e.length;i++){var j=d[i]||f(e[i].inset),k=e[i]||f(d[i].inset);g.push(j),h.push(k)}return a.mergeNestedRepeated(b,c,g,h)}var f=e.bind(null,d,", ");a.addPropertiesHandler(c,f,["box-shadow","text-shadow"])}(b),function(a,b){function c(a){return a.toFixed(3).replace(/0+$/,"").replace(/\.$/,"")}function d(a,b,c){return Math.min(b,Math.max(a,c))}function e(a){if(/^\s*[-+]?(\d*\.)?\d+\s*$/.test(a))return Number(a)}function f(a,b){return[a,b,c]}function g(a,b){if(0!=a)return i(0,1/0)(a,b)}function h(a,b){return[a,b,function(a){return Math.round(d(1,1/0,a))}]}function i(a,b){return function(e,f){return[e,f,function(e){return c(d(a,b,e))}]}}function j(a){var b=a.trim().split(/\s*[\s,]\s*/);if(0!==b.length){for(var c=[],d=0;d<b.length;d++){var f=e(b[d]);if(void 0===f)return;c.push(f)}return c}}function k(a,b){if(a.length==b.length)return[a,b,function(a){return a.map(c).join(" ")}]}function l(a,b){return[a,b,Math.round]}a.clamp=d,a.addPropertiesHandler(j,k,["stroke-dasharray"]),a.addPropertiesHandler(e,i(0,1/0),["border-image-width","line-height"]),a.addPropertiesHandler(e,i(0,1),["opacity","shape-image-threshold"]),a.addPropertiesHandler(e,g,["flex-grow","flex-shrink"]),a.addPropertiesHandler(e,h,["orphans","widows"]),a.addPropertiesHandler(e,l,["z-index"]),a.parseNumber=e,a.parseNumberList=j,a.mergeNumbers=f,a.numberToString=c}(b),function(a,b){function c(a,b){if("visible"==a||"visible"==b)return[0,1,function(c){return c<=0?a:c>=1?b:"visible"}]}a.addPropertiesHandler(String,c,["visibility"])}(b),function(a,b){function c(a){a=a.trim(),f.fillStyle="#000",f.fillStyle=a;var b=f.fillStyle;if(f.fillStyle="#fff",f.fillStyle=a,b==f.fillStyle){f.fillRect(0,0,1,1);var c=f.getImageData(0,0,1,1).data;f.clearRect(0,0,1,1);var d=c[3]/255;return[c[0]*d,c[1]*d,c[2]*d,d]}}function d(b,c){return[b,c,function(b){function c(a){return Math.max(0,Math.min(255,a))}if(b[3])for(var d=0;d<3;d++)b[d]=Math.round(c(b[d]/b[3]));return b[3]=a.numberToString(a.clamp(0,1,b[3])),"rgba("+b.join(",")+")"}]}var e=document.createElementNS("http://www.w3.org/1999/xhtml","canvas");e.width=e.height=1;var f=e.getContext("2d");a.addPropertiesHandler(c,d,["background-color","border-bottom-color","border-left-color","border-right-color","border-top-color","color","fill","flood-color","lighting-color","outline-color","stop-color","stroke","text-decoration-color"]),a.consumeColor=a.consumeParenthesised.bind(null,c),a.mergeColors=d}(b),function(a,b){function c(a){function b(){var b=h.exec(a);g=b?b[0]:void 0}function c(){var a=Number(g);return b(),a}function d(){if("("!==g)return c();b();var a=f();return")"!==g?NaN:(b(),a)}function e(){for(var a=d();"*"===g||"/"===g;){var c=g;b();var e=d();"*"===c?a*=e:a/=e}return a}function f(){for(var a=e();"+"===g||"-"===g;){var c=g;b();var d=e();"+"===c?a+=d:a-=d}return a}var g,h=/([\+\-\w\.]+|[\(\)\*\/])/g;return b(),f()}function d(a,b){if("0"==(b=b.trim().toLowerCase())&&"px".search(a)>=0)return{px:0};if(/^[^(]*$|^calc/.test(b)){b=b.replace(/calc\(/g,"(");var d={};b=b.replace(a,function(a){return d[a]=null,"U"+a});for(var e="U("+a.source+")",f=b.replace(/[-+]?(\d*\.)?\d+([Ee][-+]?\d+)?/g,"N").replace(new RegExp("N"+e,"g"),"D").replace(/\s[+-]\s/g,"O").replace(/\s/g,""),g=[/N\*(D)/g,/(N|D)[*\/]N/g,/(N|D)O\1/g,/\((N|D)\)/g],h=0;h<g.length;)g[h].test(f)?(f=f.replace(g[h],"$1"),h=0):h++;if("D"==f){for(var i in d){var j=c(b.replace(new RegExp("U"+i,"g"),"").replace(new RegExp(e,"g"),"*0"));if(!isFinite(j))return;d[i]=j}return d}}}function e(a,b){return f(a,b,!0)}function f(b,c,d){var e,f=[];for(e in b)f.push(e);for(e in c)f.indexOf(e)<0&&f.push(e);return b=f.map(function(a){return b[a]||0}),c=f.map(function(a){return c[a]||0}),[b,c,function(b){var c=b.map(function(c,e){return 1==b.length&&d&&(c=Math.max(c,0)),a.numberToString(c)+f[e]}).join(" + ");return b.length>1?"calc("+c+")":c}]}var g="px|em|ex|ch|rem|vw|vh|vmin|vmax|cm|mm|in|pt|pc",h=d.bind(null,new RegExp(g,"g")),i=d.bind(null,new RegExp(g+"|%","g")),j=d.bind(null,/deg|rad|grad|turn/g);a.parseLength=h,a.parseLengthOrPercent=i,a.consumeLengthOrPercent=a.consumeParenthesised.bind(null,i),a.parseAngle=j,a.mergeDimensions=f;var k=a.consumeParenthesised.bind(null,h),l=a.consumeRepeated.bind(void 0,k,/^/),m=a.consumeRepeated.bind(void 0,l,/^,/);a.consumeSizePairList=m;var n=function(a){var b=m(a);if(b&&""==b[1])return b[0]},o=a.mergeNestedRepeated.bind(void 0,e," "),p=a.mergeNestedRepeated.bind(void 0,o,",");a.mergeNonNegativeSizePair=o,a.addPropertiesHandler(n,p,["background-size"]),a.addPropertiesHandler(i,e,["border-bottom-width","border-image-width","border-left-width","border-right-width","border-top-width","flex-basis","font-size","height","line-height","max-height","max-width","outline-width","width"]),a.addPropertiesHandler(i,f,["border-bottom-left-radius","border-bottom-right-radius","border-top-left-radius","border-top-right-radius","bottom","left","letter-spacing","margin-bottom","margin-left","margin-right","margin-top","min-height","min-width","outline-offset","padding-bottom","padding-left","padding-right","padding-top","perspective","right","shape-margin","stroke-dashoffset","text-indent","top","vertical-align","word-spacing"])}(b),function(a,b){function c(b){return a.consumeLengthOrPercent(b)||a.consumeToken(/^auto/,b)}function d(b){var d=a.consumeList([a.ignore(a.consumeToken.bind(null,/^rect/)),a.ignore(a.consumeToken.bind(null,/^\(/)),a.consumeRepeated.bind(null,c,/^,/),a.ignore(a.consumeToken.bind(null,/^\)/))],b);if(d&&4==d[0].length)return d[0]}function e(b,c){return"auto"==b||"auto"==c?[!0,!1,function(d){var e=d?b:c;if("auto"==e)return"auto";var f=a.mergeDimensions(e,e);return f[2](f[0])}]:a.mergeDimensions(b,c)}function f(a){return"rect("+a+")"}var g=a.mergeWrappedNestedRepeated.bind(null,f,e,", ");a.parseBox=d,a.mergeBoxes=g,a.addPropertiesHandler(d,g,["clip"])}(b),function(a,b){function c(a){return function(b){var c=0;return a.map(function(a){return a===k?b[c++]:a})}}function d(a){return a}function e(b){if("none"==(b=b.toLowerCase().trim()))return[];for(var c,d=/\s*(\w+)\(([^)]*)\)/g,e=[],f=0;c=d.exec(b);){if(c.index!=f)return;f=c.index+c[0].length;var g=c[1],h=n[g];if(!h)return;var i=c[2].split(","),j=h[0];if(j.length<i.length)return;for(var k=[],o=0;o<j.length;o++){var p,q=i[o],r=j[o];if(void 0===(p=q?{A:function(b){return"0"==b.trim()?m:a.parseAngle(b)},N:a.parseNumber,T:a.parseLengthOrPercent,L:a.parseLength}[r.toUpperCase()](q):{a:m,n:k[0],t:l}[r]))return;k.push(p)}if(e.push({t:g,d:k}),d.lastIndex==b.length)return e}}function f(a){return a.toFixed(6).replace(".000000","")}function g(b,c){if(b.decompositionPair!==c){b.decompositionPair=c;var d=a.makeMatrixDecomposition(b)}if(c.decompositionPair!==b){c.decompositionPair=b;var e=a.makeMatrixDecomposition(c)}return null==d[0]||null==e[0]?[[!1],[!0],function(a){return a?c[0].d:b[0].d}]:(d[0].push(0),e[0].push(1),[d,e,function(b){var c=a.quat(d[0][3],e[0][3],b[5]);return a.composeMatrix(b[0],b[1],b[2],c,b[4]).map(f).join(",")}])}function h(a){return a.replace(/[xy]/,"")}function i(a){return a.replace(/(x|y|z|3d)?$/,"3d")}function j(b,c){var d=a.makeMatrixDecomposition&&!0,e=!1;if(!b.length||!c.length){b.length||(e=!0,b=c,c=[]);for(var f=0;f<b.length;f++){var j=b[f].t,k=b[f].d,l="scale"==j.substr(0,5)?1:0;c.push({t:j,d:k.map(function(a){if("number"==typeof a)return l;var b={};for(var c in a)b[c]=l;return b})})}}var m=function(a,b){return"perspective"==a&&"perspective"==b||("matrix"==a||"matrix3d"==a)&&("matrix"==b||"matrix3d"==b)},o=[],p=[],q=[];if(b.length!=c.length){if(!d)return;var r=g(b,c);o=[r[0]],p=[r[1]],q=[["matrix",[r[2]]]]}else for(var f=0;f<b.length;f++){var j,s=b[f].t,t=c[f].t,u=b[f].d,v=c[f].d,w=n[s],x=n[t];if(m(s,t)){if(!d)return;var r=g([b[f]],[c[f]]);o.push(r[0]),p.push(r[1]),q.push(["matrix",[r[2]]])}else{if(s==t)j=s;else if(w[2]&&x[2]&&h(s)==h(t))j=h(s),u=w[2](u),v=x[2](v);else{if(!w[1]||!x[1]||i(s)!=i(t)){if(!d)return;var r=g(b,c);o=[r[0]],p=[r[1]],q=[["matrix",[r[2]]]];break}j=i(s),u=w[1](u),v=x[1](v)}for(var y=[],z=[],A=[],B=0;B<u.length;B++){var C="number"==typeof u[B]?a.mergeNumbers:a.mergeDimensions,r=C(u[B],v[B]);y[B]=r[0],z[B]=r[1],A.push(r[2])}o.push(y),p.push(z),q.push([j,A])}}if(e){var D=o;o=p,p=D}return[o,p,function(a){return a.map(function(a,b){var c=a.map(function(a,c){return q[b][1][c](a)}).join(",");return"matrix"==q[b][0]&&16==c.split(",").length&&(q[b][0]="matrix3d"),q[b][0]+"("+c+")"}).join(" ")}]}var k=null,l={px:0},m={deg:0},n={matrix:["NNNNNN",[k,k,0,0,k,k,0,0,0,0,1,0,k,k,0,1],d],matrix3d:["NNNNNNNNNNNNNNNN",d],rotate:["A"],rotatex:["A"],rotatey:["A"],rotatez:["A"],rotate3d:["NNNA"],perspective:["L"],scale:["Nn",c([k,k,1]),d],scalex:["N",c([k,1,1]),c([k,1])],scaley:["N",c([1,k,1]),c([1,k])],scalez:["N",c([1,1,k])],scale3d:["NNN",d],skew:["Aa",null,d],skewx:["A",null,c([k,m])],skewy:["A",null,c([m,k])],translate:["Tt",c([k,k,l]),d],translatex:["T",c([k,l,l]),c([k,l])],translatey:["T",c([l,k,l]),c([l,k])],translatez:["L",c([l,l,k])],translate3d:["TTL",d]};a.addPropertiesHandler(e,j,["transform"]),a.transformToSvgMatrix=function(b){var c=a.transformListToMatrix(e(b));return"matrix("+f(c[0])+" "+f(c[1])+" "+f(c[4])+" "+f(c[5])+" "+f(c[12])+" "+f(c[13])+")"}}(b),function(a,b){function c(a,b){b.concat([a]).forEach(function(b){b in document.documentElement.style&&(d[a]=b),e[b]=a})}var d={},e={};c("transform",["webkitTransform","msTransform"]),c("transformOrigin",["webkitTransformOrigin"]),c("perspective",["webkitPerspective"]),c("perspectiveOrigin",["webkitPerspectiveOrigin"]),a.propertyName=function(a){return d[a]||a},a.unprefixedPropertyName=function(a){return e[a]||a}}(b)}(),function(){if(void 0===document.createElement("div").animate([]).oncancel){var a;if(window.performance&&performance.now)var a=function(){return performance.now()};else var a=function(){return Date.now()};var b=function(a,b,c){this.target=a,this.currentTime=b,this.timelineTime=c,this.type="cancel",this.bubbles=!1,this.cancelable=!1,this.currentTarget=a,this.defaultPrevented=!1,this.eventPhase=Event.AT_TARGET,this.timeStamp=Date.now()},c=window.Element.prototype.animate;window.Element.prototype.animate=function(d,e){var f=c.call(this,d,e);f._cancelHandlers=[],f.oncancel=null;var g=f.cancel;f.cancel=function(){g.call(this);var c=new b(this,null,a()),d=this._cancelHandlers.concat(this.oncancel?[this.oncancel]:[]);setTimeout(function(){d.forEach(function(a){a.call(c.target,c)})},0)};var h=f.addEventListener;f.addEventListener=function(a,b){"function"==typeof b&&"cancel"==a?this._cancelHandlers.push(b):h.call(this,a,b)};var i=f.removeEventListener;return f.removeEventListener=function(a,b){if("cancel"==a){var c=this._cancelHandlers.indexOf(b);c>=0&&this._cancelHandlers.splice(c,1)}else i.call(this,a,b)},f}}}(),function(a){var b=document.documentElement,c=null,d=!1;try{var e=getComputedStyle(b).getPropertyValue("opacity"),f="0"==e?"1":"0";c=b.animate({opacity:[f,f]},{duration:1}),c.currentTime=0,d=getComputedStyle(b).getPropertyValue("opacity")==f}catch(a){}finally{c&&c.cancel()}if(!d){var g=window.Element.prototype.animate;window.Element.prototype.animate=function(b,c){return window.Symbol&&Symbol.iterator&&Array.prototype.from&&b[Symbol.iterator]&&(b=Array.from(b)),Array.isArray(b)||null===b||(b=a.convertToArrayForm(b)),g.call(this,b,c)}}}(a),function(a,b,c){function d(a){var c=b.timeline;c.currentTime=a,c._discardAnimations(),0==c._animations.length?f=!1:requestAnimationFrame(d)}var e=window.requestAnimationFrame;window.requestAnimationFrame=function(a){return e(function(c){b.timeline._updateAnimationsPromises(),a(c),b.timeline._updateAnimationsPromises()})},b.AnimationTimeline=function(){this._animations=[],this.currentTime=void 0},b.AnimationTimeline.prototype={getAnimations:function(){return this._discardAnimations(),this._animations.slice()},_updateAnimationsPromises:function(){b.animationsWithPromises=b.animationsWithPromises.filter(function(a){return a._updatePromises()})},_discardAnimations:function(){this._updateAnimationsPromises(),this._animations=this._animations.filter(function(a){return"finished"!=a.playState&&"idle"!=a.playState})},_play:function(a){var c=new b.Animation(a,this);return this._animations.push(c),b.restartWebAnimationsNextTick(),c._updatePromises(),c._animation.play(),c._updatePromises(),c},play:function(a){return a&&a.remove(),this._play(a)}};var f=!1;b.restartWebAnimationsNextTick=function(){f||(f=!0,requestAnimationFrame(d))};var g=new b.AnimationTimeline;b.timeline=g;try{Object.defineProperty(window.document,"timeline",{configurable:!0,get:function(){return g}})}catch(a){}try{window.document.timeline=g}catch(a){}}(0,c),function(a,b,c){b.animationsWithPromises=[],b.Animation=function(b,c){if(this.id="",b&&b._id&&(this.id=b._id),this.effect=b,b&&(b._animation=this),!c)throw new Error("Animation with null timeline is not supported");this._timeline=c,this._sequenceNumber=a.sequenceNumber++,this._holdTime=0,this._paused=!1,this._isGroup=!1,this._animation=null,this._childAnimations=[],this._callback=null,this._oldPlayState="idle",this._rebuildUnderlyingAnimation(),this._animation.cancel(),this._updatePromises()},b.Animation.prototype={_updatePromises:function(){var a=this._oldPlayState,b=this.playState;return this._readyPromise&&b!==a&&("idle"==b?(this._rejectReadyPromise(),this._readyPromise=void 0):"pending"==a?this._resolveReadyPromise():"pending"==b&&(this._readyPromise=void 0)),this._finishedPromise&&b!==a&&("idle"==b?(this._rejectFinishedPromise(),this._finishedPromise=void 0):"finished"==b?this._resolveFinishedPromise():"finished"==a&&(this._finishedPromise=void 0)),this._oldPlayState=this.playState,this._readyPromise||this._finishedPromise},_rebuildUnderlyingAnimation:function(){this._updatePromises();var a,c,d,e,f=!!this._animation;f&&(a=this.playbackRate,c=this._paused,d=this.startTime,e=this.currentTime,this._animation.cancel(),this._animation._wrapper=null,this._animation=null),(!this.effect||this.effect instanceof window.KeyframeEffect)&&(this._animation=b.newUnderlyingAnimationForKeyframeEffect(this.effect),b.bindAnimationForKeyframeEffect(this)),(this.effect instanceof window.SequenceEffect||this.effect instanceof window.GroupEffect)&&(this._animation=b.newUnderlyingAnimationForGroup(this.effect),b.bindAnimationForGroup(this)),this.effect&&this.effect._onsample&&b.bindAnimationForCustomEffect(this),f&&(1!=a&&(this.playbackRate=a),null!==d?this.startTime=d:null!==e?this.currentTime=e:null!==this._holdTime&&(this.currentTime=this._holdTime),c&&this.pause()),this._updatePromises()},_updateChildren:function(){if(this.effect&&"idle"!=this.playState){var a=this.effect._timing.delay;this._childAnimations.forEach(function(c){this._arrangeChildren(c,a),this.effect instanceof window.SequenceEffect&&(a+=b.groupChildDuration(c.effect))}.bind(this))}},_setExternalAnimation:function(a){if(this.effect&&this._isGroup)for(var b=0;b<this.effect.children.length;b++)this.effect.children[b]._animation=a,this._childAnimations[b]._setExternalAnimation(a)},_constructChildAnimations:function(){if(this.effect&&this._isGroup){var a=this.effect._timing.delay;this._removeChildAnimations(),this.effect.children.forEach(function(c){var d=b.timeline._play(c);this._childAnimations.push(d),d.playbackRate=this.playbackRate,this._paused&&d.pause(),c._animation=this.effect._animation,this._arrangeChildren(d,a),this.effect instanceof window.SequenceEffect&&(a+=b.groupChildDuration(c))}.bind(this))}},_arrangeChildren:function(a,b){null===this.startTime?a.currentTime=this.currentTime-b/this.playbackRate:a.startTime!==this.startTime+b/this.playbackRate&&(a.startTime=this.startTime+b/this.playbackRate)},get timeline(){return this._timeline},get playState(){return this._animation?this._animation.playState:"idle"},get finished(){return window.Promise?(this._finishedPromise||(-1==b.animationsWithPromises.indexOf(this)&&b.animationsWithPromises.push(this),this._finishedPromise=new Promise(function(a,b){this._resolveFinishedPromise=function(){a(this)},this._rejectFinishedPromise=function(){b({type:DOMException.ABORT_ERR,name:"AbortError"})}}.bind(this)),"finished"==this.playState&&this._resolveFinishedPromise()),this._finishedPromise):(console.warn("Animation Promises require JavaScript Promise constructor"),null)},get ready(){return window.Promise?(this._readyPromise||(-1==b.animationsWithPromises.indexOf(this)&&b.animationsWithPromises.push(this),this._readyPromise=new Promise(function(a,b){this._resolveReadyPromise=function(){a(this)},this._rejectReadyPromise=function(){b({type:DOMException.ABORT_ERR,name:"AbortError"})}}.bind(this)),"pending"!==this.playState&&this._resolveReadyPromise()),this._readyPromise):(console.warn("Animation Promises require JavaScript Promise constructor"),null)},get onfinish(){return this._animation.onfinish},set onfinish(a){this._animation.onfinish="function"==typeof a?function(b){b.target=this,a.call(this,b)}.bind(this):a},get oncancel(){return this._animation.oncancel},set oncancel(a){this._animation.oncancel="function"==typeof a?function(b){b.target=this,a.call(this,b)}.bind(this):a},get currentTime(){this._updatePromises();var a=this._animation.currentTime;return this._updatePromises(),a},set currentTime(a){this._updatePromises(),this._animation.currentTime=isFinite(a)?a:Math.sign(a)*Number.MAX_VALUE,this._register(),this._forEachChild(function(b,c){b.currentTime=a-c}),this._updatePromises()},get startTime(){return this._animation.startTime},set startTime(a){this._updatePromises(),this._animation.startTime=isFinite(a)?a:Math.sign(a)*Number.MAX_VALUE,this._register(),this._forEachChild(function(b,c){b.startTime=a+c}),this._updatePromises()},get playbackRate(){return this._animation.playbackRate},set playbackRate(a){this._updatePromises();var b=this.currentTime;this._animation.playbackRate=a,this._forEachChild(function(b){b.playbackRate=a}),null!==b&&(this.currentTime=b),this._updatePromises()},play:function(){this._updatePromises(),this._paused=!1,this._animation.play(),-1==this._timeline._animations.indexOf(this)&&this._timeline._animations.push(this),this._register(),b.awaitStartTime(this),this._forEachChild(function(a){var b=a.currentTime;a.play(),a.currentTime=b}),this._updatePromises()},pause:function(){this._updatePromises(),this.currentTime&&(this._holdTime=this.currentTime),this._animation.pause(),this._register(),this._forEachChild(function(a){a.pause()}),this._paused=!0,this._updatePromises()},finish:function(){this._updatePromises(),this._animation.finish(),this._register(),this._updatePromises()},cancel:function(){this._updatePromises(),this._animation.cancel(),this._register(),this._removeChildAnimations(),this._updatePromises()},reverse:function(){this._updatePromises();var a=this.currentTime;this._animation.reverse(),this._forEachChild(function(a){a.reverse()}),null!==a&&(this.currentTime=a),this._updatePromises()},addEventListener:function(a,b){var c=b;"function"==typeof b&&(c=function(a){a.target=this,b.call(this,a)}.bind(this),b._wrapper=c),this._animation.addEventListener(a,c)},removeEventListener:function(a,b){this._animation.removeEventListener(a,b&&b._wrapper||b)},_removeChildAnimations:function(){for(;this._childAnimations.length;)this._childAnimations.pop().cancel()},_forEachChild:function(b){var c=0;if(this.effect.children&&this._childAnimations.length<this.effect.children.length&&this._constructChildAnimations(),this._childAnimations.forEach(function(a){b.call(this,a,c),this.effect instanceof window.SequenceEffect&&(c+=a.effect.activeDuration)}.bind(this)),"pending"!=this.playState){var d=this.effect._timing,e=this.currentTime;null!==e&&(e=a.calculateIterationProgress(a.calculateActiveDuration(d),e,d)),(null==e||isNaN(e))&&this._removeChildAnimations()}}},window.Animation=b.Animation}(a,c),function(a,b,c){function d(b){this._frames=a.normalizeKeyframes(b)}function e(){for(var a=!1;i.length;)i.shift()._updateChildren(),a=!0;return a}var f=function(a){if(a._animation=void 0,a instanceof window.SequenceEffect||a instanceof window.GroupEffect)for(var b=0;b<a.children.length;b++)f(a.children[b])};b.removeMulti=function(a){for(var b=[],c=0;c<a.length;c++){var d=a[c];d._parent?(-1==b.indexOf(d._parent)&&b.push(d._parent),d._parent.children.splice(d._parent.children.indexOf(d),1),d._parent=null,f(d)):d._animation&&d._animation.effect==d&&(d._animation.cancel(),d._animation.effect=new KeyframeEffect(null,[]),d._animation._callback&&(d._animation._callback._animation=null),d._animation._rebuildUnderlyingAnimation(),f(d))}for(c=0;c<b.length;c++)b[c]._rebuild()},b.KeyframeEffect=function(b,c,e,f){return this.target=b,this._parent=null,e=a.numericTimingToObject(e),this._timingInput=a.cloneTimingInput(e),this._timing=a.normalizeTimingInput(e),this.timing=a.makeTiming(e,!1,this),this.timing._effect=this,"function"==typeof c?(a.deprecated("Custom KeyframeEffect","2015-06-22","Use KeyframeEffect.onsample instead."),this._normalizedKeyframes=c):this._normalizedKeyframes=new d(c),this._keyframes=c,this.activeDuration=a.calculateActiveDuration(this._timing),this._id=f,this},b.KeyframeEffect.prototype={getFrames:function(){return"function"==typeof this._normalizedKeyframes?this._normalizedKeyframes:this._normalizedKeyframes._frames},set onsample(a){if("function"==typeof this.getFrames())throw new Error("Setting onsample on custom effect KeyframeEffect is not supported.");this._onsample=a,this._animation&&this._animation._rebuildUnderlyingAnimation()},get parent(){return this._parent},clone:function(){if("function"==typeof this.getFrames())throw new Error("Cloning custom effects is not supported.");var b=new KeyframeEffect(this.target,[],a.cloneTimingInput(this._timingInput),this._id);return b._normalizedKeyframes=this._normalizedKeyframes,b._keyframes=this._keyframes,b},remove:function(){b.removeMulti([this])}};var g=Element.prototype.animate;Element.prototype.animate=function(a,c){var d="";return c&&c.id&&(d=c.id),b.timeline._play(new b.KeyframeEffect(this,a,c,d))};var h=document.createElementNS("http://www.w3.org/1999/xhtml","div");b.newUnderlyingAnimationForKeyframeEffect=function(a){if(a){var b=a.target||h,c=a._keyframes;"function"==typeof c&&(c=[]);var d=a._timingInput;d.id=a._id}else var b=h,c=[],d=0;return g.apply(b,[c,d])},b.bindAnimationForKeyframeEffect=function(a){a.effect&&"function"==typeof a.effect._normalizedKeyframes&&b.bindAnimationForCustomEffect(a)};var i=[];b.awaitStartTime=function(a){null===a.startTime&&a._isGroup&&(0==i.length&&requestAnimationFrame(e),i.push(a))};var j=window.getComputedStyle;Object.defineProperty(window,"getComputedStyle",{configurable:!0,enumerable:!0,value:function(){b.timeline._updateAnimationsPromises();var a=j.apply(this,arguments);return e()&&(a=j.apply(this,arguments)),b.timeline._updateAnimationsPromises(),a}}),window.KeyframeEffect=b.KeyframeEffect,window.Element.prototype.getAnimations=function(){return document.timeline.getAnimations().filter(function(a){return null!==a.effect&&a.effect.target==this}.bind(this))}}(a,c),function(a,b,c){function d(a){a._registered||(a._registered=!0,g.push(a),h||(h=!0,requestAnimationFrame(e)))}function e(a){var b=g;g=[],b.sort(function(a,b){return a._sequenceNumber-b._sequenceNumber}),b=b.filter(function(a){a();var b=a._animation?a._animation.playState:"idle";return"running"!=b&&"pending"!=b&&(a._registered=!1),a._registered}),g.push.apply(g,b),g.length?(h=!0,requestAnimationFrame(e)):h=!1}var f=(document.createElementNS("http://www.w3.org/1999/xhtml","div"),0);b.bindAnimationForCustomEffect=function(b){var c,e=b.effect.target,g="function"==typeof b.effect.getFrames();c=g?b.effect.getFrames():b.effect._onsample;var h=b.effect.timing,i=null;h=a.normalizeTimingInput(h);var j=function(){var d=j._animation?j._animation.currentTime:null;null!==d&&(d=a.calculateIterationProgress(a.calculateActiveDuration(h),d,h),isNaN(d)&&(d=null)),d!==i&&(g?c(d,e,b.effect):c(d,b.effect,b.effect._animation)),i=d};j._animation=b,j._registered=!1,j._sequenceNumber=f++,b._callback=j,d(j)};var g=[],h=!1;b.Animation.prototype._register=function(){this._callback&&d(this._callback)}}(a,c),function(a,b,c){function d(a){return a._timing.delay+a.activeDuration+a._timing.endDelay}function e(b,c,d){this._id=d,this._parent=null,this.children=b||[],this._reparent(this.children),c=a.numericTimingToObject(c),this._timingInput=a.cloneTimingInput(c),this._timing=a.normalizeTimingInput(c,!0),this.timing=a.makeTiming(c,!0,this),this.timing._effect=this,"auto"===this._timing.duration&&(this._timing.duration=this.activeDuration)}window.SequenceEffect=function(){e.apply(this,arguments)},window.GroupEffect=function(){e.apply(this,arguments)},e.prototype={_isAncestor:function(a){for(var b=this;null!==b;){if(b==a)return!0;b=b._parent}return!1},_rebuild:function(){for(var a=this;a;)"auto"===a.timing.duration&&(a._timing.duration=a.activeDuration),a=a._parent;this._animation&&this._animation._rebuildUnderlyingAnimation()},_reparent:function(a){b.removeMulti(a);for(var c=0;c<a.length;c++)a[c]._parent=this},_putChild:function(a,b){for(var c=b?"Cannot append an ancestor or self":"Cannot prepend an ancestor or self",d=0;d<a.length;d++)if(this._isAncestor(a[d]))throw{type:DOMException.HIERARCHY_REQUEST_ERR,name:"HierarchyRequestError",message:c};for(var d=0;d<a.length;d++)b?this.children.push(a[d]):this.children.unshift(a[d]);this._reparent(a),this._rebuild()},append:function(){this._putChild(arguments,!0)},prepend:function(){this._putChild(arguments,!1)},get parent(){return this._parent},get firstChild(){return this.children.length?this.children[0]:null},get lastChild(){return this.children.length?this.children[this.children.length-1]:null},clone:function(){for(var b=a.cloneTimingInput(this._timingInput),c=[],d=0;d<this.children.length;d++)c.push(this.children[d].clone());return this instanceof GroupEffect?new GroupEffect(c,b):new SequenceEffect(c,b)},remove:function(){b.removeMulti([this])}},window.SequenceEffect.prototype=Object.create(e.prototype),Object.defineProperty(window.SequenceEffect.prototype,"activeDuration",{get:function(){var a=0;return this.children.forEach(function(b){a+=d(b)}),Math.max(a,0)}}),window.GroupEffect.prototype=Object.create(e.prototype),Object.defineProperty(window.GroupEffect.prototype,"activeDuration",{get:function(){var a=0;return this.children.forEach(function(b){a=Math.max(a,d(b))}),a}}),b.newUnderlyingAnimationForGroup=function(c){var d,e=null,f=function(b){var c=d._wrapper;if(c&&"pending"!=c.playState&&c.effect)return null==b?void c._removeChildAnimations():0==b&&c.playbackRate<0&&(e||(e=a.normalizeTimingInput(c.effect.timing)),b=a.calculateIterationProgress(a.calculateActiveDuration(e),-1,e),isNaN(b)||null==b)?(c._forEachChild(function(a){a.currentTime=-1}),void c._removeChildAnimations()):void 0},g=new KeyframeEffect(null,[],c._timing,c._id);return g.onsample=f,d=b.timeline._play(g)},b.bindAnimationForGroup=function(a){a._animation._wrapper=a,a._isGroup=!0,b.awaitStartTime(a),a._constructChildAnimations(),a._setExternalAnimation(a)},b.groupChildDuration=d}(a,c)}();

(function () {

    /*! *****************************************************************************
    Copyright (c) Microsoft Corporation.

    Permission to use, copy, modify, and/or distribute this software for any
    purpose with or without fee is hereby granted.

    THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH
    REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
    AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,
    INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
    LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR
    OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
    PERFORMANCE OF THIS SOFTWARE.
    ***************************************************************************** */

    function __decorate(decorators, target, key, desc) {
        var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
        if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
        else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
        return c > 3 && r && Object.defineProperty(target, key, r), r;
    }

    function __metadata(metadataKey, metadataValue) {
        if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(metadataKey, metadataValue);
    }

    function __awaiter(thisArg, _arguments, P, generator) {
        function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
        return new (P || (P = Promise))(function (resolve, reject) {
            function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
            function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
            function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
            step((generator = generator.apply(thisArg, _arguments || [])).next());
        });
    }

    /**
     * @license
     * Copyright (c) 2016 The Polymer Project Authors. All rights reserved.
     * This code may only be used under the BSD style license found at
     * http://polymer.github.io/LICENSE.txt The complete set of authors may be found
     * at http://polymer.github.io/AUTHORS.txt The complete set of contributors may
     * be found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by
     * Google as part of the polymer project is also subject to an additional IP
     * rights grant found at http://polymer.github.io/PATENTS.txt
     */
    /**
     * A TypeScript class decorator factory that registers the class as a custom
     * element.
     *
     * If `tagname` is provided, it will be used as the custom element name, and
     * will be assigned to the class static `is` property. If `tagname` is omitted,
     * the static `is` property of the class will be used instead. If neither exist,
     * or if both exist but have different values (except in the case that the `is`
     * property is not an own-property of the class), an exception is thrown.
     */
    function customElement(tagname) {
        return (class_) => {
            if (tagname) {
                // Only check that tag names match when `is` is our own property. It might
                // be inherited from a superclass, in which case it's ok if they're
                // different, and we'll override it with our own property below.
                if (class_.hasOwnProperty('is')) {
                    if (tagname !== class_.is) {
                        throw new Error(`custom element tag names do not match: ` +
                            `(${tagname} !== ${class_.is})`);
                    }
                }
                else {
                    Object.defineProperty(class_, 'is', { value: tagname });
                }
            }
            // Throws if tag name is missing or invalid.
            window.customElements.define(class_.is, class_);
        };
    }
    function createProperty(proto, name, options) {
        if (!proto.constructor.hasOwnProperty('properties')) {
            Object.defineProperty(proto.constructor, 'properties', { value: {} });
        }
        proto.constructor.properties[name] = Object.assign({}, proto.constructor.properties[name], options);
    }
    /**
     * A TypeScript property decorator factory that defines this as a Polymer
     * property.
     *
     * This function must be invoked to return a decorator.
     *
     * @ExportDecoratedItems
     */
    function property(options) {
        return (proto, propName) => {
            createProperty(proto, propName, options);
        };
    }
    /**
     * A TypeScript property decorator factory that causes the decorated method to
     * be called when a property changes.
     *
     * This function must be invoked to return a decorator.
     *
     * @ExportDecoratedItems
     */
    function observe(...targets) {
        return (proto, propName) => {
            if (!proto.constructor.hasOwnProperty('observers')) {
                Object.defineProperty(proto.constructor, 'observers', { value: [] });
            }
            proto.constructor.observers.push(`${propName}(${targets.join(',')})`);
        };
    }
    /**
     * A TypeScript accessor decorator factory that causes the decorated getter to
     * be called when a set of dependencies change. The arguments of this decorator
     * should be paths of the data dependencies as described
     * [here](https://www.polymer-project.org/2.0/docs/devguide/observers#define-a-computed-property)
     * The decorated getter should not have an associated setter.
     *
     * This function must be invoked to return a decorator.
     *
     * @ExportDecoratedItems
     */
    function computed(firstTarget, ...moreTargets) {
        return (proto, propName, descriptor) => {
            const fnName = `__compute${propName}`;
            Object.defineProperty(proto, fnName, { value: descriptor.get });
            descriptor.get = undefined;
            const targets = [firstTarget, ...moreTargets].join(',');
            createProperty(proto, propName, { computed: `${fnName}(${targets})` });
        };
    }

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    /* eslint-disable no-unused-vars */
    /**
     * When using Closure Compiler, JSCompiler_renameProperty(property, object) is replaced by the munged name for object[property]
     * We cannot alias this function, so we have to use a small shim that has the same behavior when not compiling.
     *
     * @param {?} prop Property name
     * @param {*} obj Reference object
     * @return {string} Potentially renamed property name
     */
    window.JSCompiler_renameProperty = function(prop, obj) {
      return prop;
    };

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    let CSS_URL_RX = /(url\()([^)]*)(\))/g;
    let ABS_URL = /(^\/[^\/])|(^#)|(^[\w-\d]*:)/;
    let workingURL;
    let resolveDoc;
    /**
     * Resolves the given URL against the provided `baseUri'.
     *
     * Note that this function performs no resolution for URLs that start
     * with `/` (absolute URLs) or `#` (hash identifiers).  For general purpose
     * URL resolution, use `window.URL`.
     *
     * @param {string} url Input URL to resolve
     * @param {?string=} baseURI Base URI to resolve the URL against
     * @return {string} resolved URL
     */
    function resolveUrl(url, baseURI) {
      if (url && ABS_URL.test(url)) {
        return url;
      }
      if (url === '//') {
        return url;
      }
      // Lazy feature detection.
      if (workingURL === undefined) {
        workingURL = false;
        try {
          const u = new URL('b', 'http://a');
          u.pathname = 'c%20d';
          workingURL = (u.href === 'http://a/c%20d');
        } catch (e) {
          // silently fail
        }
      }
      if (!baseURI) {
        baseURI = document.baseURI || window.location.href;
      }
      if (workingURL) {
        try {
          return (new URL(url, baseURI)).href;
        } catch (e) {
          // Bad url or baseURI structure. Do not attempt to resolve.
          return url;
        }
      }
      // Fallback to creating an anchor into a disconnected document.
      if (!resolveDoc) {
        resolveDoc = document.implementation.createHTMLDocument('temp');
        resolveDoc.base = resolveDoc.createElement('base');
        resolveDoc.head.appendChild(resolveDoc.base);
        resolveDoc.anchor = resolveDoc.createElement('a');
        resolveDoc.body.appendChild(resolveDoc.anchor);
      }
      resolveDoc.base.href = baseURI;
      resolveDoc.anchor.href = url;
      return resolveDoc.anchor.href || url;

    }

    /**
     * Resolves any relative URL's in the given CSS text against the provided
     * `ownerDocument`'s `baseURI`.
     *
     * @param {string} cssText CSS text to process
     * @param {string} baseURI Base URI to resolve the URL against
     * @return {string} Processed CSS text with resolved URL's
     */
    function resolveCss(cssText, baseURI) {
      return cssText.replace(CSS_URL_RX, function(m, pre, url, post) {
        return pre + '\'' +
          resolveUrl(url.replace(/["']/g, ''), baseURI) +
          '\'' + post;
      });
    }

    /**
     * Returns a path from a given `url`. The path includes the trailing
     * `/` from the url.
     *
     * @param {string} url Input URL to transform
     * @return {string} resolved path
     */
    function pathFromUrl(url) {
      return url.substring(0, url.lastIndexOf('/') + 1);
    }

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */
    const useShadow = !(window.ShadyDOM) || !(window.ShadyDOM.inUse);
    const useNativeCSSProperties = Boolean(!window.ShadyCSS || window.ShadyCSS.nativeCss);
    const supportsAdoptingStyleSheets = useShadow &&
        ('adoptedStyleSheets' in Document.prototype) &&
        ('replaceSync' in CSSStyleSheet.prototype) &&
        // Since spec may change, feature detect exact API we need
        (() => {
          try {
            const sheet = new CSSStyleSheet();
            sheet.replaceSync('');
            const host = document.createElement('div');
            host.attachShadow({mode: 'open'});
            host.shadowRoot.adoptedStyleSheets = [sheet];
            return (host.shadowRoot.adoptedStyleSheets[0] === sheet);
          } catch(e) {
            return false;
          }
        })();

    /**
     * Globally settable property that is automatically assigned to
     * `ElementMixin` instances, useful for binding in templates to
     * make URL's relative to an application's root.  Defaults to the main
     * document URL, but can be overridden by users.  It may be useful to set
     * `rootPath` to provide a stable application mount path when
     * using client side routing.
     */
    let rootPath = window.Polymer && window.Polymer.rootPath ||
      pathFromUrl(document.baseURI || window.location.href);

    /**
     * A global callback used to sanitize any value before inserting it into the DOM.
     * The callback signature is:
     *
     *  function sanitizeDOMValue(value, name, type, node) { ... }
     *
     * Where:
     *
     * `value` is the value to sanitize.
     * `name` is the name of an attribute or property (for example, href).
     * `type` indicates where the value is being inserted: one of property, attribute, or text.
     * `node` is the node where the value is being inserted.
     *
     * @type {(function(*,string,string,?Node):*)|undefined}
     */
    let sanitizeDOMValue =
      window.Polymer && window.Polymer.sanitizeDOMValue || undefined;

    /**
     * Globally settable property to make Polymer Gestures use passive TouchEvent listeners when recognizing gestures.
     * When set to `true`, gestures made from touch will not be able to prevent scrolling, allowing for smoother
     * scrolling performance.
     * Defaults to `false` for backwards compatibility.
     */
    let passiveTouchGestures =
      window.Polymer && window.Polymer.setPassiveTouchGestures || false;

    /**
     * Setting to ensure Polymer template evaluation only occurs based on tempates
     * defined in trusted script.  When true, `<dom-module>` re-registration is
     * disallowed, `<dom-bind>` is disabled, and `<dom-if>`/`<dom-repeat>`
     * templates will only evaluate in the context of a trusted element template.
     */
    let strictTemplatePolicy =
      window.Polymer && window.Polymer.strictTemplatePolicy || false;

    /**
     * Setting to enable dom-module lookup from Polymer.Element.  By default,
     * templates must be defined in script using the `static get template()`
     * getter and the `html` tag function.  To enable legacy loading of templates
     * via dom-module, set this flag to true.
     */
    let allowTemplateFromDomModule =
      window.Polymer && window.Polymer.allowTemplateFromDomModule || false;

    /**
     * Setting to skip processing style includes and re-writing urls in css styles.
     * Normally "included" styles are pulled into the element and all urls in styles
     * are re-written to be relative to the containing script url.
     * If no includes or relative urls are used in styles, these steps can be
     * skipped as an optimization.
     */
    let legacyOptimizations =
      window.Polymer && window.Polymer.legacyOptimizations || false;

    /**
     * Setting to add warnings useful when migrating from Polymer 1.x to 2.x.
     */
    let legacyWarnings =
      window.Polymer && window.Polymer.legacyWarnings || false;

    /**
     * Setting to perform initial rendering synchronously when running under ShadyDOM.
     * This matches the behavior of Polymer 1.
     */
    let syncInitialRender =
      window.Polymer && window.Polymer.syncInitialRender || false;

    /**
     * Setting to retain the legacy Polymer 1 behavior for multi-property
     * observers around undefined values. Observers and computed property methods
     * are not called until no argument is undefined.
     */
    let legacyUndefined =
      window.Polymer && window.Polymer.legacyUndefined || false;

    /**
     * Setting to ensure computed properties are computed in order to ensure
     * re-computation never occurs in a given turn.
     */
    let orderedComputed =
      window.Polymer && window.Polymer.orderedComputed || false;

    /**
     * Setting to remove nested templates inside `dom-if` and `dom-repeat` as
     * part of element template parsing.  This is a performance optimization that
     * eliminates most of the tax of needing two elements due to the loss of
     * type-extended templates as a result of the V1 specification changes.
     */
    let removeNestedTemplates =
      window.Polymer && window.Polymer.removeNestedTemplates || false;

    /**
     * Setting to place `dom-if` elements in a performance-optimized mode that takes
     * advantage of lighter-weight host runtime template stamping to eliminate the
     * need for an intermediate Templatizer `TemplateInstance` to mange the nodes
     * stamped by `dom-if`.  Under this setting, any Templatizer-provided API's
     * such as `modelForElement` will not be available for nodes stamped by
     * `dom-if`.
     */
    let fastDomIf = window.Polymer && window.Polymer.fastDomIf || false;

    /**
     * Setting to disable `dom-change` and `rendered-item-count` events from
     * `dom-if` and `dom-repeat`. Users can opt back into `dom-change` events by
     * setting the `notify-dom-change` attribute (`notifyDomChange: true` property)
     * to `dom-if`/`don-repeat` instances.
     */
    let suppressTemplateNotifications =
      window.Polymer && window.Polymer.suppressTemplateNotifications || false;

    /**
     * Setting to disable use of dynamic attributes. This is an optimization
     * to avoid setting `observedAttributes`. Instead attributes are read
     * once at create time and set/removeAttribute are patched.
     */
    let legacyNoObservedAttributes =
      window.Polymer && window.Polymer.legacyNoObservedAttributes || false;

    /**
     * Setting to enable use of `adoptedStyleSheets` for sharing style sheets
     * between component instances' shadow roots, if the app uses built Shady CSS
     * styles.
     */
    let useAdoptedStyleSheetsWithBuiltCSS =
      window.Polymer && window.Polymer.useAdoptedStyleSheetsWithBuiltCSS || false;

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    // unique global id for deduping mixins.
    let dedupeId = 0;

    /* eslint-disable valid-jsdoc */
    /**
     * Wraps an ES6 class expression mixin such that the mixin is only applied
     * if it has not already been applied its base argument. Also memoizes mixin
     * applications.
     *
     * @template T
     * @param {T} mixin ES6 class expression mixin to wrap
     * @return {T}
     * @suppress {invalidCasts}
     */
    const dedupingMixin = function(mixin) {
      let mixinApplications = /** @type {!MixinFunction} */(mixin).__mixinApplications;
      if (!mixinApplications) {
        mixinApplications = new WeakMap();
        /** @type {!MixinFunction} */(mixin).__mixinApplications = mixinApplications;
      }
      // maintain a unique id for each mixin
      let mixinDedupeId = dedupeId++;
      function dedupingMixin(base) {
        let baseSet = /** @type {!MixinFunction} */(base).__mixinSet;
        if (baseSet && baseSet[mixinDedupeId]) {
          return base;
        }
        let map = mixinApplications;
        let extended = map.get(base);
        if (!extended) {
          extended = /** @type {!Function} */(mixin)(base);
          map.set(base, extended);
          // copy inherited mixin set from the extended class, or the base class
          // NOTE: we avoid use of Set here because some browser (IE11)
          // cannot extend a base Set via the constructor.
          let mixinSet = Object.create(/** @type {!MixinFunction} */(extended).__mixinSet || baseSet || null);
          mixinSet[mixinDedupeId] = true;
          /** @type {!MixinFunction} */(extended).__mixinSet = mixinSet;
        }
        return extended;
      }

      return dedupingMixin;
    };
    /* eslint-enable valid-jsdoc */

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    let modules = {};
    let lcModules = {};
    /**
     * Sets a dom-module into the global registry by id.
     *
     * @param {string} id dom-module id
     * @param {DomModule} module dom-module instance
     * @return {void}
     */
    function setModule(id, module) {
      // store id separate from lowercased id so that
      // in all cases mixedCase id will stored distinctly
      // and lowercase version is a fallback
      modules[id] = lcModules[id.toLowerCase()] = module;
    }
    /**
     * Retrieves a dom-module from the global registry by id.
     *
     * @param {string} id dom-module id
     * @return {DomModule!} dom-module instance
     */
    function findModule(id) {
      return modules[id] || lcModules[id.toLowerCase()];
    }

    function styleOutsideTemplateCheck(inst) {
      if (inst.querySelector('style')) {
        console.warn('dom-module %s has style outside template', inst.id);
      }
    }

    /**
     * The `dom-module` element registers the dom it contains to the name given
     * by the module's id attribute. It provides a unified database of dom
     * accessible via its static `import` API.
     *
     * A key use case of `dom-module` is for providing custom element `<template>`s
     * via HTML imports that are parsed by the native HTML parser, that can be
     * relocated during a bundling pass and still looked up by `id`.
     *
     * Example:
     *
     *     <dom-module id="foo">
     *       <img src="stuff.png">
     *     </dom-module>
     *
     * Then in code in some other location that cannot access the dom-module above
     *
     *     let img = customElements.get('dom-module').import('foo', 'img');
     *
     * @customElement
     * @extends HTMLElement
     * @summary Custom element that provides a registry of relocatable DOM content
     *   by `id` that is agnostic to bundling.
     * @unrestricted
     */
    class DomModule extends HTMLElement {

      /** @override */
      static get observedAttributes() { return ['id']; }

      /**
       * Retrieves the element specified by the css `selector` in the module
       * registered by `id`. For example, this.import('foo', 'img');
       * @param {string} id The id of the dom-module in which to search.
       * @param {string=} selector The css selector by which to find the element.
       * @return {Element} Returns the element which matches `selector` in the
       * module registered at the specified `id`.
       *
       * @export
       * @nocollapse Referred to indirectly in style-gather.js
       */
      static import(id, selector) {
        if (id) {
          let m = findModule(id);
          if (m && selector) {
            return m.querySelector(selector);
          }
          return m;
        }
        return null;
      }

      /* eslint-disable no-unused-vars */
      /**
       * @param {string} name Name of attribute.
       * @param {?string} old Old value of attribute.
       * @param {?string} value Current value of attribute.
       * @param {?string} namespace Attribute namespace.
       * @return {void}
       * @override
       */
      attributeChangedCallback(name, old, value, namespace) {
        if (old !== value) {
          this.register();
        }
      }
      /* eslint-enable no-unused-args */

      /**
       * The absolute URL of the original location of this `dom-module`.
       *
       * This value will differ from this element's `ownerDocument` in the
       * following ways:
       * - Takes into account any `assetpath` attribute added during bundling
       *   to indicate the original location relative to the bundled location
       * - Uses the HTMLImports polyfill's `importForElement` API to ensure
       *   the path is relative to the import document's location since
       *   `ownerDocument` is not currently polyfilled
       */
      get assetpath() {
        // Don't override existing assetpath.
        if (!this.__assetpath) {
          // note: assetpath set via an attribute must be relative to this
          // element's location; accomodate polyfilled HTMLImports
          const owner = window.HTMLImports && HTMLImports.importForElement ?
            HTMLImports.importForElement(this) || document : this.ownerDocument;
          const url = resolveUrl(
            this.getAttribute('assetpath') || '', owner.baseURI);
          this.__assetpath = pathFromUrl(url);
        }
        return this.__assetpath;
      }

      /**
       * Registers the dom-module at a given id. This method should only be called
       * when a dom-module is imperatively created. For
       * example, `document.createElement('dom-module').register('foo')`.
       * @param {string=} id The id at which to register the dom-module.
       * @return {void}
       */
      register(id) {
        id = id || this.id;
        if (id) {
          // Under strictTemplatePolicy, reject and null out any re-registered
          // dom-module since it is ambiguous whether first-in or last-in is trusted
          if (strictTemplatePolicy && findModule(id) !== undefined) {
            setModule(id, null);
            throw new Error(`strictTemplatePolicy: dom-module ${id} re-registered`);
          }
          this.id = id;
          setModule(id, this);
          styleOutsideTemplateCheck(this);
        }
      }
    }

    DomModule.prototype['modules'] = modules;

    customElements.define('dom-module', DomModule);

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    const MODULE_STYLE_LINK_SELECTOR = 'link[rel=import][type~=css]';
    const INCLUDE_ATTR = 'include';
    const SHADY_UNSCOPED_ATTR = 'shady-unscoped';

    /**
     * @param {string} moduleId .
     * @return {?DomModule} .
     */
    function importModule(moduleId) {
      return /** @type {?DomModule} */(DomModule.import(moduleId));
    }

    function styleForImport(importDoc) {
      // NOTE: polyfill affordance.
      // under the HTMLImports polyfill, there will be no 'body',
      // but the import pseudo-doc can be used directly.
      let container = importDoc.body ? importDoc.body : importDoc;
      const importCss = resolveCss(container.textContent,
        importDoc.baseURI);
      const style = document.createElement('style');
      style.textContent = importCss;
      return style;
    }


    /**
     * Returns a list of <style> elements in a space-separated list of `dom-module`s.
     *
     * @function
     * @param {string} moduleIds List of dom-module id's within which to
     * search for css.
     * @return {!Array<!HTMLStyleElement>} Array of contained <style> elements
     */
    function stylesFromModules(moduleIds) {
     const modules = moduleIds.trim().split(/\s+/);
     const styles = [];
     for (let i=0; i < modules.length; i++) {
       styles.push(...stylesFromModule(modules[i]));
     }
     return styles;
    }

    /**
     * Returns a list of <style> elements in a given `dom-module`.
     * Styles in a `dom-module` can come either from `<style>`s within the
     * first `<template>`, or else from one or more
     * `<link rel="import" type="css">` links outside the template.
     *
     * @param {string} moduleId dom-module id to gather styles from
     * @return {!Array<!HTMLStyleElement>} Array of contained styles.
     */
    function stylesFromModule(moduleId) {
      const m = importModule(moduleId);

      if (!m) {
        console.warn('Could not find style data in module named', moduleId);
        return [];
      }

      if (m._styles === undefined) {
        const styles = [];
        // module imports: <link rel="import" type="css">
        styles.push(..._stylesFromModuleImports(m));
        // include css from the first template in the module
        const template = /** @type {?HTMLTemplateElement} */(
            m.querySelector('template'));
        if (template) {
          styles.push(...stylesFromTemplate(template,
            /** @type {templateWithAssetPath} */(m).assetpath));
        }

        m._styles = styles;
      }

      return m._styles;
    }

    /**
     * Returns the `<style>` elements within a given template.
     *
     * @param {!HTMLTemplateElement} template Template to gather styles from
     * @param {string=} baseURI baseURI for style content
     * @return {!Array<!HTMLStyleElement>} Array of styles
     */
    function stylesFromTemplate(template, baseURI) {
      if (!template._styles) {
        const styles = [];
        // if element is a template, get content from its .content
        const e$ = template.content.querySelectorAll('style');
        for (let i=0; i < e$.length; i++) {
          let e = e$[i];
          // support style sharing by allowing styles to "include"
          // other dom-modules that contain styling
          let include = e.getAttribute(INCLUDE_ATTR);
          if (include) {
            styles.push(...stylesFromModules(include).filter(function(item, index, self) {
              return self.indexOf(item) === index;
            }));
          }
          if (baseURI) {
            e.textContent =
                resolveCss(e.textContent, /** @type {string} */ (baseURI));
          }
          styles.push(e);
        }
        template._styles = styles;
      }
      return template._styles;
    }

    /**
     * Returns a list of <style> elements  from stylesheets loaded via `<link rel="import" type="css">` links within the specified `dom-module`.
     *
     * @param {string} moduleId Id of `dom-module` to gather CSS from
     * @return {!Array<!HTMLStyleElement>} Array of contained styles.
     */
    function stylesFromModuleImports(moduleId) {
     let m = importModule(moduleId);
     return m ? _stylesFromModuleImports(m) : [];
    }

    /**
     * @param {!HTMLElement} module dom-module element that could contain `<link rel="import" type="css">` styles
     * @return {!Array<!HTMLStyleElement>} Array of contained styles
     */
    function _stylesFromModuleImports(module) {
      const styles = [];
      const p$ = module.querySelectorAll(MODULE_STYLE_LINK_SELECTOR);
      for (let i=0; i < p$.length; i++) {
        let p = p$[i];
        if (p.import) {
          const importDoc = p.import;
          const unscoped = p.hasAttribute(SHADY_UNSCOPED_ATTR);
          if (unscoped && !importDoc._unscopedStyle) {
            const style = styleForImport(importDoc);
            style.setAttribute(SHADY_UNSCOPED_ATTR, '');
            importDoc._unscopedStyle = style;
          } else if (!importDoc._style) {
            importDoc._style = styleForImport(importDoc);
          }
          styles.push(unscoped ? importDoc._unscopedStyle : importDoc._style);
        }
      }
      return styles;
    }

    /**
     *
     * Returns CSS text of styles in a space-separated list of `dom-module`s.
     * Note: This method is deprecated, use `stylesFromModules` instead.
     *
     * @deprecated
     * @param {string} moduleIds List of dom-module id's within which to
     * search for css.
     * @return {string} Concatenated CSS content from specified `dom-module`s
     */
    function cssFromModules(moduleIds) {
     let modules = moduleIds.trim().split(/\s+/);
     let cssText = '';
     for (let i=0; i < modules.length; i++) {
       cssText += cssFromModule(modules[i]);
     }
     return cssText;
    }

    /**
     * Returns CSS text of styles in a given `dom-module`.  CSS in a `dom-module`
     * can come either from `<style>`s within the first `<template>`, or else
     * from one or more `<link rel="import" type="css">` links outside the
     * template.
     *
     * Any `<styles>` processed are removed from their original location.
     * Note: This method is deprecated, use `styleFromModule` instead.
     *
     * @deprecated
     * @param {string} moduleId dom-module id to gather styles from
     * @return {string} Concatenated CSS content from specified `dom-module`
     */
    function cssFromModule(moduleId) {
      let m = importModule(moduleId);
      if (m && m._cssText === undefined) {
        // module imports: <link rel="import" type="css">
        let cssText = _cssFromModuleImports(m);
        // include css from the first template in the module
        let t = /** @type {?HTMLTemplateElement} */(m.querySelector('template'));
        if (t) {
          cssText += cssFromTemplate(t,
            /** @type {templateWithAssetPath} */(m).assetpath);
        }
        m._cssText = cssText || null;
      }
      if (!m) {
        console.warn('Could not find style data in module named', moduleId);
      }
      return m && m._cssText || '';
    }

    /**
     * Returns CSS text of `<styles>` within a given template.
     *
     * Any `<styles>` processed are removed from their original location.
     * Note: This method is deprecated, use `styleFromTemplate` instead.
     *
     * @deprecated
     * @param {!HTMLTemplateElement} template Template to gather styles from
     * @param {string} baseURI Base URI to resolve the URL against
     * @return {string} Concatenated CSS content from specified template
     */
    function cssFromTemplate(template, baseURI) {
      let cssText = '';
      const e$ = stylesFromTemplate(template, baseURI);
      // if element is a template, get content from its .content
      for (let i=0; i < e$.length; i++) {
        let e = e$[i];
        if (e.parentNode) {
          e.parentNode.removeChild(e);
        }
        cssText += e.textContent;
      }
      return cssText;
    }

    /**
     * @deprecated
     * @param {!HTMLElement} module dom-module element that could contain `<link rel="import" type="css">` styles
     * @return {string} Concatenated CSS content from links in the dom-module
     */
    function _cssFromModuleImports(module) {
      let cssText = '';
      let styles = _stylesFromModuleImports(module);
      for (let i=0; i < styles.length; i++) {
        cssText += styles[i].textContent;
      }
      return cssText;
    }

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    /* eslint-disable valid-jsdoc */
    /**
     * Node wrapper to ensure ShadowDOM safe operation regardless of polyfill
     * presence or mode. Note that with the introduction of `ShadyDOM.noPatch`,
     * a node wrapper must be used to access ShadowDOM API.
     * This is similar to using `Polymer.dom` but relies exclusively
     * on the presence of the ShadyDOM polyfill rather than requiring the loading
     * of legacy (Polymer.dom) API.
     * @type {function(Node):Node}
     */
    const wrap = (window['ShadyDOM'] && window['ShadyDOM']['noPatch'] && window['ShadyDOM']['wrap']) ?
      window['ShadyDOM']['wrap'] :
      (window['ShadyDOM'] ? (n) => ShadyDOM['patch'](n) : (n) => n);

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    /**
     * Module with utilities for manipulating structured data path strings.
     *
     * @summary Module with utilities for manipulating structured data path strings.
     */

    /**
     * Returns true if the given string is a structured data path (has dots).
     *
     * Example:
     *
     * ```
     * isPath('foo.bar.baz') // true
     * isPath('foo')         // false
     * ```
     *
     * @param {string} path Path string
     * @return {boolean} True if the string contained one or more dots
     */
    function isPath(path) {
      return path.indexOf('.') >= 0;
    }

    /**
     * Returns the root property name for the given path.
     *
     * Example:
     *
     * ```
     * root('foo.bar.baz') // 'foo'
     * root('foo')         // 'foo'
     * ```
     *
     * @param {string} path Path string
     * @return {string} Root property name
     */
    function root(path) {
      let dotIndex = path.indexOf('.');
      if (dotIndex === -1) {
        return path;
      }
      return path.slice(0, dotIndex);
    }

    /**
     * Given `base` is `foo.bar`, `foo` is an ancestor, `foo.bar` is not
     * Returns true if the given path is an ancestor of the base path.
     *
     * Example:
     *
     * ```
     * isAncestor('foo.bar', 'foo')         // true
     * isAncestor('foo.bar', 'foo.bar')     // false
     * isAncestor('foo.bar', 'foo.bar.baz') // false
     * ```
     *
     * @param {string} base Path string to test against.
     * @param {string} path Path string to test.
     * @return {boolean} True if `path` is an ancestor of `base`.
     */
    function isAncestor(base, path) {
      //     base.startsWith(path + '.');
      return base.indexOf(path + '.') === 0;
    }

    /**
     * Given `base` is `foo.bar`, `foo.bar.baz` is an descendant
     *
     * Example:
     *
     * ```
     * isDescendant('foo.bar', 'foo.bar.baz') // true
     * isDescendant('foo.bar', 'foo.bar')     // false
     * isDescendant('foo.bar', 'foo')         // false
     * ```
     *
     * @param {string} base Path string to test against.
     * @param {string} path Path string to test.
     * @return {boolean} True if `path` is a descendant of `base`.
     */
    function isDescendant(base, path) {
      //     path.startsWith(base + '.');
      return path.indexOf(base + '.') === 0;
    }

    /**
     * Replaces a previous base path with a new base path, preserving the
     * remainder of the path.
     *
     * User must ensure `path` has a prefix of `base`.
     *
     * Example:
     *
     * ```
     * translate('foo.bar', 'zot', 'foo.bar.baz') // 'zot.baz'
     * ```
     *
     * @param {string} base Current base string to remove
     * @param {string} newBase New base string to replace with
     * @param {string} path Path to translate
     * @return {string} Translated string
     */
    function translate(base, newBase, path) {
      return newBase + path.slice(base.length);
    }

    /**
     * @param {string} base Path string to test against
     * @param {string} path Path string to test
     * @return {boolean} True if `path` is equal to `base`
     */
    function matches(base, path) {
      return (base === path) ||
             isAncestor(base, path) ||
             isDescendant(base, path);
    }

    /**
     * Converts array-based paths to flattened path.  String-based paths
     * are returned as-is.
     *
     * Example:
     *
     * ```
     * normalize(['foo.bar', 0, 'baz'])  // 'foo.bar.0.baz'
     * normalize('foo.bar.0.baz')        // 'foo.bar.0.baz'
     * ```
     *
     * @param {string | !Array<string|number>} path Input path
     * @return {string} Flattened path
     */
    function normalize(path) {
      if (Array.isArray(path)) {
        let parts = [];
        for (let i=0; i<path.length; i++) {
          let args = path[i].toString().split('.');
          for (let j=0; j<args.length; j++) {
            parts.push(args[j]);
          }
        }
        return parts.join('.');
      } else {
        return path;
      }
    }

    /**
     * Splits a path into an array of property names. Accepts either arrays
     * of path parts or strings.
     *
     * Example:
     *
     * ```
     * split(['foo.bar', 0, 'baz'])  // ['foo', 'bar', '0', 'baz']
     * split('foo.bar.0.baz')        // ['foo', 'bar', '0', 'baz']
     * ```
     *
     * @param {string | !Array<string|number>} path Input path
     * @return {!Array<string>} Array of path parts
     * @suppress {checkTypes}
     */
    function split(path) {
      if (Array.isArray(path)) {
        return normalize(path).split('.');
      }
      return path.toString().split('.');
    }

    /**
     * Reads a value from a path.  If any sub-property in the path is `undefined`,
     * this method returns `undefined` (will never throw.
     *
     * @param {Object} root Object from which to dereference path from
     * @param {string | !Array<string|number>} path Path to read
     * @param {Object=} info If an object is provided to `info`, the normalized
     *  (flattened) path will be set to `info.path`.
     * @return {*} Value at path, or `undefined` if the path could not be
     *  fully dereferenced.
     */
    function get(root, path, info) {
      let prop = root;
      let parts = split(path);
      // Loop over path parts[0..n-1] and dereference
      for (let i=0; i<parts.length; i++) {
        if (!prop) {
          return;
        }
        let part = parts[i];
        prop = prop[part];
      }
      if (info) {
        info.path = parts.join('.');
      }
      return prop;
    }

    /**
     * Sets a value to a path.  If any sub-property in the path is `undefined`,
     * this method will no-op.
     *
     * @param {Object} root Object from which to dereference path from
     * @param {string | !Array<string|number>} path Path to set
     * @param {*} value Value to set to path
     * @return {string | undefined} The normalized version of the input path
     */
    function set(root, path, value) {
      let prop = root;
      let parts = split(path);
      let last = parts[parts.length-1];
      if (parts.length > 1) {
        // Loop over path parts[0..n-2] and dereference
        for (let i=0; i<parts.length-1; i++) {
          let part = parts[i];
          prop = prop[part];
          if (!prop) {
            return;
          }
        }
        // Set value to object at end of path
        prop[last] = value;
      } else {
        // Simple property set
        prop[path] = value;
      }
      return parts.join('.');
    }

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    const caseMap = {};
    const DASH_TO_CAMEL = /-[a-z]/g;
    const CAMEL_TO_DASH = /([A-Z])/g;

    /**
     * @fileoverview Module with utilities for converting between "dash-case" and
     * "camelCase" identifiers.
     */

    /**
     * Converts "dash-case" identifier (e.g. `foo-bar-baz`) to "camelCase"
     * (e.g. `fooBarBaz`).
     *
     * @param {string} dash Dash-case identifier
     * @return {string} Camel-case representation of the identifier
     */
    function dashToCamelCase(dash) {
      return caseMap[dash] || (
        caseMap[dash] = dash.indexOf('-') < 0 ? dash : dash.replace(DASH_TO_CAMEL,
          (m) => m[1].toUpperCase()
        )
      );
    }

    /**
     * Converts "camelCase" identifier (e.g. `fooBarBaz`) to "dash-case"
     * (e.g. `foo-bar-baz`).
     *
     * @param {string} camel Camel-case identifier
     * @return {string} Dash-case representation of the identifier
     */
    function camelToDashCase(camel) {
      return caseMap[camel] || (
        caseMap[camel] = camel.replace(CAMEL_TO_DASH, '-$1').toLowerCase()
      );
    }

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    // Microtask implemented using Mutation Observer
    let microtaskCurrHandle = 0;
    let microtaskLastHandle = 0;
    let microtaskCallbacks = [];
    let microtaskNodeContent = 0;
    let microtaskScheduled = false;
    let microtaskNode = document.createTextNode('');
    new window.MutationObserver(microtaskFlush).observe(microtaskNode, {characterData: true});

    function microtaskFlush() {
      microtaskScheduled = false;
      const len = microtaskCallbacks.length;
      for (let i = 0; i < len; i++) {
        let cb = microtaskCallbacks[i];
        if (cb) {
          try {
            cb();
          } catch (e) {
            setTimeout(() => { throw e; });
          }
        }
      }
      microtaskCallbacks.splice(0, len);
      microtaskLastHandle += len;
    }

    /**
     * Async interface wrapper around `setTimeout`.
     *
     * @namespace
     * @summary Async interface wrapper around `setTimeout`.
     */
    const timeOut = {
      /**
       * Returns a sub-module with the async interface providing the provided
       * delay.
       *
       * @memberof timeOut
       * @param {number=} delay Time to wait before calling callbacks in ms
       * @return {!AsyncInterface} An async timeout interface
       */
      after(delay) {
        return {
          run(fn) { return window.setTimeout(fn, delay); },
          cancel(handle) {
            window.clearTimeout(handle);
          }
        };
      },
      /**
       * Enqueues a function called in the next task.
       *
       * @memberof timeOut
       * @param {!Function} fn Callback to run
       * @param {number=} delay Delay in milliseconds
       * @return {number} Handle used for canceling task
       */
      run(fn, delay) {
        return window.setTimeout(fn, delay);
      },
      /**
       * Cancels a previously enqueued `timeOut` callback.
       *
       * @memberof timeOut
       * @param {number} handle Handle returned from `run` of callback to cancel
       * @return {void}
       */
      cancel(handle) {
        window.clearTimeout(handle);
      }
    };

    /**
     * Async interface wrapper around `requestAnimationFrame`.
     *
     * @namespace
     * @summary Async interface wrapper around `requestAnimationFrame`.
     */
    const animationFrame = {
      /**
       * Enqueues a function called at `requestAnimationFrame` timing.
       *
       * @memberof animationFrame
       * @param {function(number):void} fn Callback to run
       * @return {number} Handle used for canceling task
       */
      run(fn) {
        return window.requestAnimationFrame(fn);
      },
      /**
       * Cancels a previously enqueued `animationFrame` callback.
       *
       * @memberof animationFrame
       * @param {number} handle Handle returned from `run` of callback to cancel
       * @return {void}
       */
      cancel(handle) {
        window.cancelAnimationFrame(handle);
      }
    };

    /**
     * Async interface wrapper around `requestIdleCallback`.  Falls back to
     * `setTimeout` on browsers that do not support `requestIdleCallback`.
     *
     * @namespace
     * @summary Async interface wrapper around `requestIdleCallback`.
     */
    const idlePeriod = {
      /**
       * Enqueues a function called at `requestIdleCallback` timing.
       *
       * @memberof idlePeriod
       * @param {function(!IdleDeadline):void} fn Callback to run
       * @return {number} Handle used for canceling task
       */
      run(fn) {
        return window.requestIdleCallback ?
          window.requestIdleCallback(fn) :
          window.setTimeout(fn, 16);
      },
      /**
       * Cancels a previously enqueued `idlePeriod` callback.
       *
       * @memberof idlePeriod
       * @param {number} handle Handle returned from `run` of callback to cancel
       * @return {void}
       */
      cancel(handle) {
        window.cancelIdleCallback ?
          window.cancelIdleCallback(handle) :
          window.clearTimeout(handle);
      }
    };

    /**
     * Async interface for enqueuing callbacks that run at microtask timing.
     *
     * Note that microtask timing is achieved via a single `MutationObserver`,
     * and thus callbacks enqueued with this API will all run in a single
     * batch, and not interleaved with other microtasks such as promises.
     * Promises are avoided as an implementation choice for the time being
     * due to Safari bugs that cause Promises to lack microtask guarantees.
     *
     * @namespace
     * @summary Async interface for enqueuing callbacks that run at microtask
     *   timing.
     */
    const microTask = {

      /**
       * Enqueues a function called at microtask timing.
       *
       * @memberof microTask
       * @param {!Function=} callback Callback to run
       * @return {number} Handle used for canceling task
       */
      run(callback) {
        if (!microtaskScheduled) {
          microtaskScheduled = true;
          microtaskNode.textContent = microtaskNodeContent++;
        }
        microtaskCallbacks.push(callback);
        return microtaskCurrHandle++;
      },

      /**
       * Cancels a previously enqueued `microTask` callback.
       *
       * @memberof microTask
       * @param {number} handle Handle returned from `run` of callback to cancel
       * @return {void}
       */
      cancel(handle) {
        const idx = handle - microtaskLastHandle;
        if (idx >= 0) {
          if (!microtaskCallbacks[idx]) {
            throw new Error('invalid async handle: ' + handle);
          }
          microtaskCallbacks[idx] = null;
        }
      }

    };

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    /** @const {!AsyncInterface} */
    const microtask = microTask;

    /**
     * Element class mixin that provides basic meta-programming for creating one
     * or more property accessors (getter/setter pair) that enqueue an async
     * (batched) `_propertiesChanged` callback.
     *
     * For basic usage of this mixin, call `MyClass.createProperties(props)`
     * once at class definition time to create property accessors for properties
     * named in props, implement `_propertiesChanged` to react as desired to
     * property changes, and implement `static get observedAttributes()` and
     * include lowercase versions of any property names that should be set from
     * attributes. Last, call `this._enableProperties()` in the element's
     * `connectedCallback` to enable the accessors.
     *
     * @mixinFunction
     * @polymer
     * @summary Element class mixin for reacting to property changes from
     *   generated property accessors.
     * @template T
     * @param {function(new:T)} superClass Class to apply mixin to.
     * @return {function(new:T)} superClass with mixin applied.
     */
    const PropertiesChanged = dedupingMixin(
        /**
         * @template T
         * @param {function(new:T)} superClass Class to apply mixin to.
         * @return {function(new:T)} superClass with mixin applied.
         */
        (superClass) => {

      /**
       * @polymer
       * @mixinClass
       * @implements {Polymer_PropertiesChanged}
       * @unrestricted
       */
      class PropertiesChanged extends superClass {

        /**
         * Creates property accessors for the given property names.
         * @param {!Object} props Object whose keys are names of accessors.
         * @return {void}
         * @protected
         * @nocollapse
         */
        static createProperties(props) {
          const proto = this.prototype;
          for (let prop in props) {
            // don't stomp an existing accessor
            if (!(prop in proto)) {
              proto._createPropertyAccessor(prop);
            }
          }
        }

        /**
         * Returns an attribute name that corresponds to the given property.
         * The attribute name is the lowercased property name. Override to
         * customize this mapping.
         * @param {string} property Property to convert
         * @return {string} Attribute name corresponding to the given property.
         *
         * @protected
         * @nocollapse
         */
        static attributeNameForProperty(property) {
          return property.toLowerCase();
        }

        /**
         * Override point to provide a type to which to deserialize a value to
         * a given property.
         * @param {string} name Name of property
         *
         * @protected
         * @nocollapse
         */
        static typeForProperty(name) { } //eslint-disable-line no-unused-vars

        /**
         * Creates a setter/getter pair for the named property with its own
         * local storage.  The getter returns the value in the local storage,
         * and the setter calls `_setProperty`, which updates the local storage
         * for the property and enqueues a `_propertiesChanged` callback.
         *
         * This method may be called on a prototype or an instance.  Calling
         * this method may overwrite a property value that already exists on
         * the prototype/instance by creating the accessor.
         *
         * @param {string} property Name of the property
         * @param {boolean=} readOnly When true, no setter is created; the
         *   protected `_setProperty` function must be used to set the property
         * @return {void}
         * @protected
         * @override
         */
        _createPropertyAccessor(property, readOnly) {
          this._addPropertyToAttributeMap(property);
          if (!this.hasOwnProperty(JSCompiler_renameProperty('__dataHasAccessor', this))) {
            this.__dataHasAccessor = Object.assign({}, this.__dataHasAccessor);
          }
          if (!this.__dataHasAccessor[property]) {
            this.__dataHasAccessor[property] = true;
            this._definePropertyAccessor(property, readOnly);
          }
        }

        /**
         * Adds the given `property` to a map matching attribute names
         * to property names, using `attributeNameForProperty`. This map is
         * used when deserializing attribute values to properties.
         *
         * @param {string} property Name of the property
         * @override
         */
        _addPropertyToAttributeMap(property) {
          if (!this.hasOwnProperty(JSCompiler_renameProperty('__dataAttributes', this))) {
            this.__dataAttributes = Object.assign({}, this.__dataAttributes);
          }
          // This check is technically not correct; it's an optimization that
          // assumes that if a _property_ name is already in the map (note this is
          // an attr->property map), the property mapped directly to the attribute
          // and it has already been mapped.  This would fail if
          // `attributeNameForProperty` were overridden such that this was not the
          // case.
          let attr = this.__dataAttributes[property];
          if (!attr) {
            attr = this.constructor.attributeNameForProperty(property);
            this.__dataAttributes[attr] = property;
          }
          return attr;
        }

        /**
         * Defines a property accessor for the given property.
         * @param {string} property Name of the property
         * @param {boolean=} readOnly When true, no setter is created
         * @return {void}
         * @override
         */
         _definePropertyAccessor(property, readOnly) {
          Object.defineProperty(this, property, {
            /* eslint-disable valid-jsdoc */
            /** @this {PropertiesChanged} */
            get() {
              // Inline for perf instead of using `_getProperty`
              return this.__data[property];
            },
            /** @this {PropertiesChanged} */
            set: readOnly ? function () {} : function (value) {
              // Inline for perf instead of using `_setProperty`
              if (this._setPendingProperty(property, value, true)) {
                this._invalidateProperties();
              }
            }
            /* eslint-enable */
          });
        }

        constructor() {
          super();
          /** @type {boolean} */
          this.__dataEnabled = false;
          this.__dataReady = false;
          this.__dataInvalid = false;
          this.__data = {};
          this.__dataPending = null;
          this.__dataOld = null;
          this.__dataInstanceProps = null;
          /** @type {number} */
          // NOTE: used to track re-entrant calls to `_flushProperties`
          this.__dataCounter = 0;
          this.__serializing = false;
          this._initializeProperties();
        }

        /**
         * Lifecycle callback called when properties are enabled via
         * `_enableProperties`.
         *
         * Users may override this function to implement behavior that is
         * dependent on the element having its property data initialized, e.g.
         * from defaults (initialized from `constructor`, `_initializeProperties`),
         * `attributeChangedCallback`, or values propagated from host e.g. via
         * bindings.  `super.ready()` must be called to ensure the data system
         * becomes enabled.
         *
         * @return {void}
         * @public
         * @override
         */
        ready() {
          this.__dataReady = true;
          this._flushProperties();
        }

        /**
         * Initializes the local storage for property accessors.
         *
         * Provided as an override point for performing any setup work prior
         * to initializing the property accessor system.
         *
         * @return {void}
         * @protected
         * @override
         */
        _initializeProperties() {
          // Capture instance properties; these will be set into accessors
          // during first flush. Don't set them here, since we want
          // these to overwrite defaults/constructor assignments
          for (let p in this.__dataHasAccessor) {
            if (this.hasOwnProperty(p)) {
              this.__dataInstanceProps = this.__dataInstanceProps || {};
              this.__dataInstanceProps[p] = this[p];
              delete this[p];
            }
          }
        }

        /**
         * Called at ready time with bag of instance properties that overwrote
         * accessors when the element upgraded.
         *
         * The default implementation sets these properties back into the
         * setter at ready time.  This method is provided as an override
         * point for customizing or providing more efficient initialization.
         *
         * @param {Object} props Bag of property values that were overwritten
         *   when creating property accessors.
         * @return {void}
         * @protected
         * @override
         */
        _initializeInstanceProperties(props) {
          Object.assign(this, props);
        }

        /**
         * Updates the local storage for a property (via `_setPendingProperty`)
         * and enqueues a `_proeprtiesChanged` callback.
         *
         * @param {string} property Name of the property
         * @param {*} value Value to set
         * @return {void}
         * @protected
         * @override
         */
        _setProperty(property, value) {
          if (this._setPendingProperty(property, value)) {
            this._invalidateProperties();
          }
        }

        /**
         * Returns the value for the given property.
         * @param {string} property Name of property
         * @return {*} Value for the given property
         * @protected
         * @override
         */
        _getProperty(property) {
          return this.__data[property];
        }

        /* eslint-disable no-unused-vars */
        /**
         * Updates the local storage for a property, records the previous value,
         * and adds it to the set of "pending changes" that will be passed to the
         * `_propertiesChanged` callback.  This method does not enqueue the
         * `_propertiesChanged` callback.
         *
         * @param {string} property Name of the property
         * @param {*} value Value to set
         * @param {boolean=} ext Not used here; affordance for closure
         * @return {boolean} Returns true if the property changed
         * @protected
         * @override
         */
        _setPendingProperty(property, value, ext) {
          let old = this.__data[property];
          let changed = this._shouldPropertyChange(property, value, old);
          if (changed) {
            if (!this.__dataPending) {
              this.__dataPending = {};
              this.__dataOld = {};
            }
            // Ensure old is captured from the last turn
            if (this.__dataOld && !(property in this.__dataOld)) {
              this.__dataOld[property] = old;
            }
            this.__data[property] = value;
            this.__dataPending[property] = value;
          }
          return changed;
        }
        /* eslint-enable */

        /**
         * @param {string} property Name of the property
         * @return {boolean} Returns true if the property is pending.
         */
        _isPropertyPending(property) {
          return !!(this.__dataPending && this.__dataPending.hasOwnProperty(property));
        }

        /**
         * Marks the properties as invalid, and enqueues an async
         * `_propertiesChanged` callback.
         *
         * @return {void}
         * @protected
         * @override
         */
        _invalidateProperties() {
          if (!this.__dataInvalid && this.__dataReady) {
            this.__dataInvalid = true;
            microtask.run(() => {
              if (this.__dataInvalid) {
                this.__dataInvalid = false;
                this._flushProperties();
              }
            });
          }
        }

        /**
         * Call to enable property accessor processing. Before this method is
         * called accessor values will be set but side effects are
         * queued. When called, any pending side effects occur immediately.
         * For elements, generally `connectedCallback` is a normal spot to do so.
         * It is safe to call this method multiple times as it only turns on
         * property accessors once.
         *
         * @return {void}
         * @protected
         * @override
         */
        _enableProperties() {
          if (!this.__dataEnabled) {
            this.__dataEnabled = true;
            if (this.__dataInstanceProps) {
              this._initializeInstanceProperties(this.__dataInstanceProps);
              this.__dataInstanceProps = null;
            }
            this.ready();
          }
        }

        /**
         * Calls the `_propertiesChanged` callback with the current set of
         * pending changes (and old values recorded when pending changes were
         * set), and resets the pending set of changes. Generally, this method
         * should not be called in user code.
         *
         * @return {void}
         * @protected
         * @override
         */
        _flushProperties() {
          this.__dataCounter++;
          const props = this.__data;
          const changedProps = this.__dataPending;
          const old = this.__dataOld;
          if (this._shouldPropertiesChange(props, changedProps, old)) {
            this.__dataPending = null;
            this.__dataOld = null;
            this._propertiesChanged(props, changedProps, old);
          }
          this.__dataCounter--;
        }

        /**
         * Called in `_flushProperties` to determine if `_propertiesChanged`
         * should be called. The default implementation returns true if
         * properties are pending. Override to customize when
         * `_propertiesChanged` is called.
         * @param {!Object} currentProps Bag of all current accessor values
         * @param {?Object} changedProps Bag of properties changed since the last
         *   call to `_propertiesChanged`
         * @param {?Object} oldProps Bag of previous values for each property
         *   in `changedProps`
         * @return {boolean} true if changedProps is truthy
         * @override
         */
        _shouldPropertiesChange(currentProps, changedProps, oldProps) { // eslint-disable-line no-unused-vars
          return Boolean(changedProps);
        }

        /**
         * Callback called when any properties with accessors created via
         * `_createPropertyAccessor` have been set.
         *
         * @param {!Object} currentProps Bag of all current accessor values
         * @param {?Object} changedProps Bag of properties changed since the last
         *   call to `_propertiesChanged`
         * @param {?Object} oldProps Bag of previous values for each property
         *   in `changedProps`
         * @return {void}
         * @protected
         * @override
         */
        _propertiesChanged(currentProps, changedProps, oldProps) { // eslint-disable-line no-unused-vars
        }

        /**
         * Method called to determine whether a property value should be
         * considered as a change and cause the `_propertiesChanged` callback
         * to be enqueued.
         *
         * The default implementation returns `true` if a strict equality
         * check fails. The method always returns false for `NaN`.
         *
         * Override this method to e.g. provide stricter checking for
         * Objects/Arrays when using immutable patterns.
         *
         * @param {string} property Property name
         * @param {*} value New property value
         * @param {*} old Previous property value
         * @return {boolean} Whether the property should be considered a change
         *   and enqueue a `_proeprtiesChanged` callback
         * @protected
         * @override
         */
        _shouldPropertyChange(property, value, old) {
          return (
            // Strict equality check
            (old !== value &&
              // This ensures (old==NaN, value==NaN) always returns false
              (old === old || value === value))
          );
        }

        /**
         * Implements native Custom Elements `attributeChangedCallback` to
         * set an attribute value to a property via `_attributeToProperty`.
         *
         * @param {string} name Name of attribute that changed
         * @param {?string} old Old attribute value
         * @param {?string} value New attribute value
         * @param {?string} namespace Attribute namespace.
         * @return {void}
         * @suppress {missingProperties} Super may or may not implement the callback
         * @override
         */
        attributeChangedCallback(name, old, value, namespace) {
          if (old !== value) {
            this._attributeToProperty(name, value);
          }
          if (super.attributeChangedCallback) {
            super.attributeChangedCallback(name, old, value, namespace);
          }
        }

        /**
         * Deserializes an attribute to its associated property.
         *
         * This method calls the `_deserializeValue` method to convert the string to
         * a typed value.
         *
         * @param {string} attribute Name of attribute to deserialize.
         * @param {?string} value of the attribute.
         * @param {*=} type type to deserialize to, defaults to the value
         * returned from `typeForProperty`
         * @return {void}
         * @override
         */
        _attributeToProperty(attribute, value, type) {
          if (!this.__serializing) {
            const map = this.__dataAttributes;
            const property = map && map[attribute] || attribute;
            this[property] = this._deserializeValue(value, type ||
              this.constructor.typeForProperty(property));
          }
        }

        /**
         * Serializes a property to its associated attribute.
         *
         * @suppress {invalidCasts} Closure can't figure out `this` is an element.
         *
         * @param {string} property Property name to reflect.
         * @param {string=} attribute Attribute name to reflect to.
         * @param {*=} value Property value to refect.
         * @return {void}
         * @override
         */
        _propertyToAttribute(property, attribute, value) {
          this.__serializing = true;
          value = (arguments.length < 3) ? this[property] : value;
          this._valueToNodeAttribute(/** @type {!HTMLElement} */(this), value,
            attribute || this.constructor.attributeNameForProperty(property));
          this.__serializing = false;
        }

        /**
         * Sets a typed value to an HTML attribute on a node.
         *
         * This method calls the `_serializeValue` method to convert the typed
         * value to a string.  If the `_serializeValue` method returns `undefined`,
         * the attribute will be removed (this is the default for boolean
         * type `false`).
         *
         * @param {Element} node Element to set attribute to.
         * @param {*} value Value to serialize.
         * @param {string} attribute Attribute name to serialize to.
         * @return {void}
         * @override
         */
        _valueToNodeAttribute(node, value, attribute) {
          const str = this._serializeValue(value);
          if (attribute === 'class' || attribute === 'name' || attribute === 'slot') {
            node = /** @type {?Element} */(wrap(node));
          }
          if (str === undefined) {
            node.removeAttribute(attribute);
          } else {
            node.setAttribute(attribute, str);
          }
        }

        /**
         * Converts a typed JavaScript value to a string.
         *
         * This method is called when setting JS property values to
         * HTML attributes.  Users may override this method to provide
         * serialization for custom types.
         *
         * @param {*} value Property value to serialize.
         * @return {string | undefined} String serialized from the provided
         * property  value.
         * @override
         */
        _serializeValue(value) {
          switch (typeof value) {
            case 'boolean':
              return value ? '' : undefined;
            default:
              return value != null ? value.toString() : undefined;
          }
        }

        /**
         * Converts a string to a typed JavaScript value.
         *
         * This method is called when reading HTML attribute values to
         * JS properties.  Users may override this method to provide
         * deserialization for custom `type`s. Types for `Boolean`, `String`,
         * and `Number` convert attributes to the expected types.
         *
         * @param {?string} value Value to deserialize.
         * @param {*=} type Type to deserialize the string to.
         * @return {*} Typed value deserialized from the provided string.
         * @override
         */
        _deserializeValue(value, type) {
          switch (type) {
            case Boolean:
              return (value !== null);
            case Number:
              return Number(value);
            default:
              return value;
          }
        }

      }

      return PropertiesChanged;
    });

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    // Save map of native properties; this forms a blacklist or properties
    // that won't have their values "saved" by `saveAccessorValue`, since
    // reading from an HTMLElement accessor from the context of a prototype throws
    const nativeProperties = {};
    let proto = HTMLElement.prototype;
    while (proto) {
      let props = Object.getOwnPropertyNames(proto);
      for (let i=0; i<props.length; i++) {
        nativeProperties[props[i]] = true;
      }
      proto = Object.getPrototypeOf(proto);
    }

    /**
     * Used to save the value of a property that will be overridden with
     * an accessor. If the `model` is a prototype, the values will be saved
     * in `__dataProto`, and it's up to the user (or downstream mixin) to
     * decide how/when to set these values back into the accessors.
     * If `model` is already an instance (it has a `__data` property), then
     * the value will be set as a pending property, meaning the user should
     * call `_invalidateProperties` or `_flushProperties` to take effect
     *
     * @param {Object} model Prototype or instance
     * @param {string} property Name of property
     * @return {void}
     * @private
     */
    function saveAccessorValue(model, property) {
      // Don't read/store value for any native properties since they could throw
      if (!nativeProperties[property]) {
        let value = model[property];
        if (value !== undefined) {
          if (model.__data) {
            // Adding accessor to instance; update the property
            // It is the user's responsibility to call _flushProperties
            model._setPendingProperty(property, value);
          } else {
            // Adding accessor to proto; save proto's value for instance-time use
            if (!model.__dataProto) {
              model.__dataProto = {};
            } else if (!model.hasOwnProperty(JSCompiler_renameProperty('__dataProto', model))) {
              model.__dataProto = Object.create(model.__dataProto);
            }
            model.__dataProto[property] = value;
          }
        }
      }
    }

    /**
     * Element class mixin that provides basic meta-programming for creating one
     * or more property accessors (getter/setter pair) that enqueue an async
     * (batched) `_propertiesChanged` callback.
     *
     * For basic usage of this mixin:
     *
     * -   Declare attributes to observe via the standard `static get
     *     observedAttributes()`. Use `dash-case` attribute names to represent
     *     `camelCase` property names.
     * -   Implement the `_propertiesChanged` callback on the class.
     * -   Call `MyClass.createPropertiesForAttributes()` **once** on the class to
     *     generate property accessors for each observed attribute. This must be
     *     called before the first instance is created, for example, by calling it
     *     before calling `customElements.define`. It can also be called lazily from
     *     the element's `constructor`, as long as it's guarded so that the call is
     *     only made once, when the first instance is created.
     * -   Call `this._enableProperties()` in the element's `connectedCallback` to
     *     enable the accessors.
     *
     * Any `observedAttributes` will automatically be
     * deserialized via `attributeChangedCallback` and set to the associated
     * property using `dash-case`-to-`camelCase` convention.
     *
     * @mixinFunction
     * @polymer
     * @appliesMixin PropertiesChanged
     * @summary Element class mixin for reacting to property changes from
     *   generated property accessors.
     * @template T
     * @param {function(new:T)} superClass Class to apply mixin to.
     * @return {function(new:T)} superClass with mixin applied.
     */
    const PropertyAccessors = dedupingMixin(superClass => {

      /**
       * @constructor
       * @implements {Polymer_PropertiesChanged}
       * @unrestricted
       * @private
       */
       const base = PropertiesChanged(superClass);

      /**
       * @polymer
       * @mixinClass
       * @implements {Polymer_PropertyAccessors}
       * @extends {base}
       * @unrestricted
       */
      class PropertyAccessors extends base {

        /**
         * Generates property accessors for all attributes in the standard
         * static `observedAttributes` array.
         *
         * Attribute names are mapped to property names using the `dash-case` to
         * `camelCase` convention
         *
         * @return {void}
         * @nocollapse
         */
        static createPropertiesForAttributes() {
          let a$ =  /** @type {?} */ (this).observedAttributes;
          for (let i=0; i < a$.length; i++) {
            this.prototype._createPropertyAccessor(dashToCamelCase(a$[i]));
          }
        }

        /**
         * Returns an attribute name that corresponds to the given property.
         * By default, converts camel to dash case, e.g. `fooBar` to `foo-bar`.
         * @param {string} property Property to convert
         * @return {string} Attribute name corresponding to the given property.
         *
         * @protected
         * @nocollapse
         */
        static attributeNameForProperty(property) {
          return camelToDashCase(property);
        }

        /**
         * Overrides PropertiesChanged implementation to initialize values for
         * accessors created for values that already existed on the element
         * prototype.
         *
         * @return {void}
         * @protected
         * @override
         */
        _initializeProperties() {
          if (this.__dataProto) {
            this._initializeProtoProperties(this.__dataProto);
            this.__dataProto = null;
          }
          super._initializeProperties();
        }

        /**
         * Called at instance time with bag of properties that were overwritten
         * by accessors on the prototype when accessors were created.
         *
         * The default implementation sets these properties back into the
         * setter at instance time.  This method is provided as an override
         * point for customizing or providing more efficient initialization.
         *
         * @param {Object} props Bag of property values that were overwritten
         *   when creating property accessors.
         * @return {void}
         * @protected
         * @override
         */
        _initializeProtoProperties(props) {
          for (let p in props) {
            this._setProperty(p, props[p]);
          }
        }

        /**
         * Ensures the element has the given attribute. If it does not,
         * assigns the given value to the attribute.
         *
         * @suppress {invalidCasts} Closure can't figure out `this` is infact an
         *     element
         *
         * @param {string} attribute Name of attribute to ensure is set.
         * @param {string} value of the attribute.
         * @return {void}
         * @override
         */
        _ensureAttribute(attribute, value) {
          const el = /** @type {!HTMLElement} */(this);
          if (!el.hasAttribute(attribute)) {
            this._valueToNodeAttribute(el, value, attribute);
          }
        }

        /**
         * Overrides PropertiesChanged implemention to serialize objects as JSON.
         *
         * @param {*} value Property value to serialize.
         * @return {string | undefined} String serialized from the provided property
         *     value.
         * @override
         */
        _serializeValue(value) {
          /* eslint-disable no-fallthrough */
          switch (typeof value) {
            case 'object':
              if (value instanceof Date) {
                return value.toString();
              } else if (value) {
                try {
                  return JSON.stringify(value);
                } catch(x) {
                  return '';
                }
              }

            default:
              return super._serializeValue(value);
          }
        }

        /**
         * Converts a string to a typed JavaScript value.
         *
         * This method is called by Polymer when reading HTML attribute values to
         * JS properties.  Users may override this method on Polymer element
         * prototypes to provide deserialization for custom `type`s.  Note,
         * the `type` argument is the value of the `type` field provided in the
         * `properties` configuration object for a given property, and is
         * by convention the constructor for the type to deserialize.
         *
         *
         * @param {?string} value Attribute value to deserialize.
         * @param {*=} type Type to deserialize the string to.
         * @return {*} Typed value deserialized from the provided string.
         * @override
         */
        _deserializeValue(value, type) {
          /**
           * @type {*}
           */
          let outValue;
          switch (type) {
            case Object:
              try {
                outValue = JSON.parse(/** @type {string} */(value));
              } catch(x) {
                // allow non-JSON literals like Strings and Numbers
                outValue = value;
              }
              break;
            case Array:
              try {
                outValue = JSON.parse(/** @type {string} */(value));
              } catch(x) {
                outValue = null;
                console.warn(`Polymer::Attributes: couldn't decode Array as JSON: ${value}`);
              }
              break;
            case Date:
              outValue = isNaN(value) ? String(value) : Number(value);
              outValue = new Date(outValue);
              break;
            default:
              outValue = super._deserializeValue(value, type);
              break;
          }
          return outValue;
        }
        /* eslint-enable no-fallthrough */

        /**
         * Overrides PropertiesChanged implementation to save existing prototype
         * property value so that it can be reset.
         * @param {string} property Name of the property
         * @param {boolean=} readOnly When true, no setter is created
         *
         * When calling on a prototype, any overwritten values are saved in
         * `__dataProto`, and it is up to the subclasser to decide how/when
         * to set those properties back into the accessor.  When calling on an
         * instance, the overwritten value is set via `_setPendingProperty`,
         * and the user should call `_invalidateProperties` or `_flushProperties`
         * for the values to take effect.
         * @protected
         * @return {void}
         * @override
         */
        _definePropertyAccessor(property, readOnly) {
          saveAccessorValue(this, property);
          super._definePropertyAccessor(property, readOnly);
        }

        /**
         * Returns true if this library created an accessor for the given property.
         *
         * @param {string} property Property name
         * @return {boolean} True if an accessor was created
         * @override
         */
        _hasAccessor(property) {
          return this.__dataHasAccessor && this.__dataHasAccessor[property];
        }

        /**
         * Returns true if the specified property has a pending change.
         *
         * @param {string} prop Property name
         * @return {boolean} True if property has a pending change
         * @protected
         * @override
         */
        _isPropertyPending(prop) {
          return Boolean(this.__dataPending && (prop in this.__dataPending));
        }

      }

      return PropertyAccessors;

    });

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    // 1.x backwards-compatible auto-wrapper for template type extensions
    // This is a clear layering violation and gives favored-nation status to
    // dom-if and dom-repeat templates.  This is a conceit we're choosing to keep
    // a.) to ease 1.x backwards-compatibility due to loss of `is`, and
    // b.) to maintain if/repeat capability in parser-constrained elements
    //     (e.g. table, select) in lieu of native CE type extensions without
    //     massive new invention in this space (e.g. directive system)
    const templateExtensions = {
      'dom-if': true,
      'dom-repeat': true
    };

    let placeholderBugDetect = false;
    let placeholderBug = false;

    function hasPlaceholderBug() {
      if (!placeholderBugDetect) {
        placeholderBugDetect = true;
        const t = document.createElement('textarea');
        t.placeholder = 'a';
        placeholderBug = t.placeholder === t.textContent;
      }
      return placeholderBug;
    }

    /**
     * Some browsers have a bug with textarea, where placeholder text is copied as
     * a textnode child of the textarea.
     *
     * If the placeholder is a binding, this can break template stamping in two
     * ways.
     *
     * One issue is that when the `placeholder` attribute is removed when the
     * binding is processed, the textnode child of the textarea is deleted, and the
     * template info tries to bind into that node.
     *
     * With `legacyOptimizations` in use, when the template is stamped and the
     * `textarea.textContent` binding is processed, no corresponding node is found
     * because it was removed during parsing. An exception is generated when this
     * binding is updated.
     *
     * With `legacyOptimizations` not in use, the template is cloned before
     * processing and this changes the above behavior. The cloned template also has
     * a value property set to the placeholder and textContent. This prevents the
     * removal of the textContent when the placeholder attribute is removed.
     * Therefore the exception does not occur. However, there is an extra
     * unnecessary binding.
     *
     * @param {!Node} node Check node for placeholder bug
     * @return {void}
     */
    function fixPlaceholder(node) {
      if (hasPlaceholderBug() && node.localName === 'textarea' && node.placeholder
            && node.placeholder === node.textContent) {
        node.textContent = null;
      }
    }

    function wrapTemplateExtension(node) {
      let is = node.getAttribute('is');
      if (is && templateExtensions[is]) {
        let t = node;
        t.removeAttribute('is');
        node = t.ownerDocument.createElement(is);
        t.parentNode.replaceChild(node, t);
        node.appendChild(t);
        while(t.attributes.length) {
          node.setAttribute(t.attributes[0].name, t.attributes[0].value);
          t.removeAttribute(t.attributes[0].name);
        }
      }
      return node;
    }

    function findTemplateNode(root, nodeInfo) {
      // recursively ascend tree until we hit root
      let parent = nodeInfo.parentInfo && findTemplateNode(root, nodeInfo.parentInfo);
      // unwind the stack, returning the indexed node at each level
      if (parent) {
        // note: marginally faster than indexing via childNodes
        // (http://jsperf.com/childnodes-lookup)
        for (let n=parent.firstChild, i=0; n; n=n.nextSibling) {
          if (nodeInfo.parentIndex === i++) {
            return n;
          }
        }
      } else {
        return root;
      }
    }

    // construct `$` map (from id annotations)
    function applyIdToMap(inst, map, node, nodeInfo) {
      if (nodeInfo.id) {
        map[nodeInfo.id] = node;
      }
    }

    // install event listeners (from event annotations)
    function applyEventListener(inst, node, nodeInfo) {
      if (nodeInfo.events && nodeInfo.events.length) {
        for (let j=0, e$=nodeInfo.events, e; (j<e$.length) && (e=e$[j]); j++) {
          inst._addMethodEventListenerToNode(node, e.name, e.value, inst);
        }
      }
    }

    // push configuration references at configure time
    function applyTemplateInfo(inst, node, nodeInfo, parentTemplateInfo) {
      if (nodeInfo.templateInfo) {
        // Give the node an instance of this templateInfo and set its parent
        node._templateInfo = nodeInfo.templateInfo;
        node._parentTemplateInfo = parentTemplateInfo;
      }
    }

    function createNodeEventHandler(context, eventName, methodName) {
      // Instances can optionally have a _methodHost which allows redirecting where
      // to find methods. Currently used by `templatize`.
      context = context._methodHost || context;
      let handler = function(e) {
        if (context[methodName]) {
          context[methodName](e, e.detail);
        } else {
          console.warn('listener method `' + methodName + '` not defined');
        }
      };
      return handler;
    }

    /**
     * Element mixin that provides basic template parsing and stamping, including
     * the following template-related features for stamped templates:
     *
     * - Declarative event listeners (`on-eventname="listener"`)
     * - Map of node id's to stamped node instances (`this.$.id`)
     * - Nested template content caching/removal and re-installation (performance
     *   optimization)
     *
     * @mixinFunction
     * @polymer
     * @summary Element class mixin that provides basic template parsing and stamping
     */
    const TemplateStamp = dedupingMixin(
        /**
         * @template T
         * @param {function(new:T)} superClass Class to apply mixin to.
         * @return {function(new:T)} superClass with mixin applied.
         */
        (superClass) => {

      /**
       * @polymer
       * @mixinClass
       * @implements {Polymer_TemplateStamp}
       */
      class TemplateStamp extends superClass {

        /**
         * Scans a template to produce template metadata.
         *
         * Template-specific metadata are stored in the object returned, and node-
         * specific metadata are stored in objects in its flattened `nodeInfoList`
         * array.  Only nodes in the template that were parsed as nodes of
         * interest contain an object in `nodeInfoList`.  Each `nodeInfo` object
         * contains an `index` (`childNodes` index in parent) and optionally
         * `parent`, which points to node info of its parent (including its index).
         *
         * The template metadata object returned from this method has the following
         * structure (many fields optional):
         *
         * ```js
         *   {
         *     // Flattened list of node metadata (for nodes that generated metadata)
         *     nodeInfoList: [
         *       {
         *         // `id` attribute for any nodes with id's for generating `$` map
         *         id: {string},
         *         // `on-event="handler"` metadata
         *         events: [
         *           {
         *             name: {string},   // event name
         *             value: {string},  // handler method name
         *           }, ...
         *         ],
         *         // Notes when the template contained a `<slot>` for shady DOM
         *         // optimization purposes
         *         hasInsertionPoint: {boolean},
         *         // For nested `<template>`` nodes, nested template metadata
         *         templateInfo: {object}, // nested template metadata
         *         // Metadata to allow efficient retrieval of instanced node
         *         // corresponding to this metadata
         *         parentInfo: {number},   // reference to parent nodeInfo>
         *         parentIndex: {number},  // index in parent's `childNodes` collection
         *         infoIndex: {number},    // index of this `nodeInfo` in `templateInfo.nodeInfoList`
         *       },
         *       ...
         *     ],
         *     // When true, the template had the `strip-whitespace` attribute
         *     // or was nested in a template with that setting
         *     stripWhitespace: {boolean},
         *     // For nested templates, nested template content is moved into
         *     // a document fragment stored here; this is an optimization to
         *     // avoid the cost of nested template cloning
         *     content: {DocumentFragment}
         *   }
         * ```
         *
         * This method kicks off a recursive treewalk as follows:
         *
         * ```
         *    _parseTemplate <---------------------+
         *      _parseTemplateContent              |
         *        _parseTemplateNode  <------------|--+
         *          _parseTemplateNestedTemplate --+  |
         *          _parseTemplateChildNodes ---------+
         *          _parseTemplateNodeAttributes
         *            _parseTemplateNodeAttribute
         *
         * ```
         *
         * These methods may be overridden to add custom metadata about templates
         * to either `templateInfo` or `nodeInfo`.
         *
         * Note that this method may be destructive to the template, in that
         * e.g. event annotations may be removed after being noted in the
         * template metadata.
         *
         * @param {!HTMLTemplateElement} template Template to parse
         * @param {TemplateInfo=} outerTemplateInfo Template metadata from the outer
         *   template, for parsing nested templates
         * @return {!TemplateInfo} Parsed template metadata
         * @nocollapse
         */
        static _parseTemplate(template, outerTemplateInfo) {
          // since a template may be re-used, memo-ize metadata
          if (!template._templateInfo) {
            // TODO(rictic): fix typing
            let /** ? */ templateInfo = template._templateInfo = {};
            templateInfo.nodeInfoList = [];
            templateInfo.nestedTemplate = Boolean(outerTemplateInfo);
            templateInfo.stripWhiteSpace =
              (outerTemplateInfo && outerTemplateInfo.stripWhiteSpace) ||
              template.hasAttribute('strip-whitespace');
             // TODO(rictic): fix typing
             this._parseTemplateContent(
                 template, templateInfo, /** @type {?} */ ({parent: null}));
          }
          return template._templateInfo;
        }

        /**
         * See docs for _parseTemplateNode.
         *
         * @param {!HTMLTemplateElement} template .
         * @param {!TemplateInfo} templateInfo .
         * @param {!NodeInfo} nodeInfo .
         * @return {boolean} .
         * @nocollapse
         */
        static _parseTemplateContent(template, templateInfo, nodeInfo) {
          return this._parseTemplateNode(template.content, templateInfo, nodeInfo);
        }

        /**
         * Parses template node and adds template and node metadata based on
         * the current node, and its `childNodes` and `attributes`.
         *
         * This method may be overridden to add custom node or template specific
         * metadata based on this node.
         *
         * @param {Node} node Node to parse
         * @param {!TemplateInfo} templateInfo Template metadata for current template
         * @param {!NodeInfo} nodeInfo Node metadata for current template.
         * @return {boolean} `true` if the visited node added node-specific
         *   metadata to `nodeInfo`
         * @nocollapse
         */
        static _parseTemplateNode(node, templateInfo, nodeInfo) {
          let noted = false;
          let element = /** @type {!HTMLTemplateElement} */ (node);
          if (element.localName == 'template' && !element.hasAttribute('preserve-content')) {
            noted = this._parseTemplateNestedTemplate(element, templateInfo, nodeInfo) || noted;
          } else if (element.localName === 'slot') {
            // For ShadyDom optimization, indicating there is an insertion point
            templateInfo.hasInsertionPoint = true;
          }
          fixPlaceholder(element);
          if (element.firstChild) {
            this._parseTemplateChildNodes(element, templateInfo, nodeInfo);
          }
          if (element.hasAttributes && element.hasAttributes()) {
            noted = this._parseTemplateNodeAttributes(element, templateInfo, nodeInfo) || noted;
          }
          // Checking `nodeInfo.noted` allows a child node of this node (who gets
          // access to `parentInfo`) to cause the parent to be noted, which
          // otherwise has no return path via `_parseTemplateChildNodes` (used by
          // some optimizations)
          return noted || nodeInfo.noted;
        }

        /**
         * Parses template child nodes for the given root node.
         *
         * This method also wraps whitelisted legacy template extensions
         * (`is="dom-if"` and `is="dom-repeat"`) with their equivalent element
         * wrappers, collapses text nodes, and strips whitespace from the template
         * if the `templateInfo.stripWhitespace` setting was provided.
         *
         * @param {Node} root Root node whose `childNodes` will be parsed
         * @param {!TemplateInfo} templateInfo Template metadata for current template
         * @param {!NodeInfo} nodeInfo Node metadata for current template.
         * @return {void}
         */
        static _parseTemplateChildNodes(root, templateInfo, nodeInfo) {
          if (root.localName === 'script' || root.localName === 'style') {
            return;
          }
          for (let node=root.firstChild, parentIndex=0, next; node; node=next) {
            // Wrap templates
            if (node.localName == 'template') {
              node = wrapTemplateExtension(node);
            }
            // collapse adjacent textNodes: fixes an IE issue that can cause
            // text nodes to be inexplicably split =(
            // note that root.normalize() should work but does not so we do this
            // manually.
            next = node.nextSibling;
            if (node.nodeType === Node.TEXT_NODE) {
              let /** Node */ n = next;
              while (n && (n.nodeType === Node.TEXT_NODE)) {
                node.textContent += n.textContent;
                next = n.nextSibling;
                root.removeChild(n);
                n = next;
              }
              // optionally strip whitespace
              if (templateInfo.stripWhiteSpace && !node.textContent.trim()) {
                root.removeChild(node);
                continue;
              }
            }
            let childInfo =
                /** @type {!NodeInfo} */ ({parentIndex, parentInfo: nodeInfo});
            if (this._parseTemplateNode(node, templateInfo, childInfo)) {
              childInfo.infoIndex = templateInfo.nodeInfoList.push(childInfo) - 1;
            }
            // Increment if not removed
            if (node.parentNode) {
              parentIndex++;
            }
          }
        }

        /**
         * Parses template content for the given nested `<template>`.
         *
         * Nested template info is stored as `templateInfo` in the current node's
         * `nodeInfo`. `template.content` is removed and stored in `templateInfo`.
         * It will then be the responsibility of the host to set it back to the
         * template and for users stamping nested templates to use the
         * `_contentForTemplate` method to retrieve the content for this template
         * (an optimization to avoid the cost of cloning nested template content).
         *
         * @param {HTMLTemplateElement} node Node to parse (a <template>)
         * @param {TemplateInfo} outerTemplateInfo Template metadata for current template
         *   that includes the template `node`
         * @param {!NodeInfo} nodeInfo Node metadata for current template.
         * @return {boolean} `true` if the visited node added node-specific
         *   metadata to `nodeInfo`
         * @nocollapse
         */
        static _parseTemplateNestedTemplate(node, outerTemplateInfo, nodeInfo) {
          // TODO(rictic): the type of node should be non-null
          let element = /** @type {!HTMLTemplateElement} */ (node);
          let templateInfo = this._parseTemplate(element, outerTemplateInfo);
          let content = templateInfo.content =
              element.content.ownerDocument.createDocumentFragment();
          content.appendChild(element.content);
          nodeInfo.templateInfo = templateInfo;
          return true;
        }

        /**
         * Parses template node attributes and adds node metadata to `nodeInfo`
         * for nodes of interest.
         *
         * @param {Element} node Node to parse
         * @param {!TemplateInfo} templateInfo Template metadata for current
         *     template
         * @param {!NodeInfo} nodeInfo Node metadata for current template.
         * @return {boolean} `true` if the visited node added node-specific
         *   metadata to `nodeInfo`
         * @nocollapse
         */
        static _parseTemplateNodeAttributes(node, templateInfo, nodeInfo) {
          // Make copy of original attribute list, since the order may change
          // as attributes are added and removed
          let noted = false;
          let attrs = Array.from(node.attributes);
          for (let i=attrs.length-1, a; (a=attrs[i]); i--) {
            noted = this._parseTemplateNodeAttribute(node, templateInfo, nodeInfo, a.name, a.value) || noted;
          }
          return noted;
        }

        /**
         * Parses a single template node attribute and adds node metadata to
         * `nodeInfo` for attributes of interest.
         *
         * This implementation adds metadata for `on-event="handler"` attributes
         * and `id` attributes.
         *
         * @param {Element} node Node to parse
         * @param {!TemplateInfo} templateInfo Template metadata for current template
         * @param {!NodeInfo} nodeInfo Node metadata for current template.
         * @param {string} name Attribute name
         * @param {string} value Attribute value
         * @return {boolean} `true` if the visited node added node-specific
         *   metadata to `nodeInfo`
         * @nocollapse
         */
        static _parseTemplateNodeAttribute(node, templateInfo, nodeInfo, name, value) {
          // events (on-*)
          if (name.slice(0, 3) === 'on-') {
            node.removeAttribute(name);
            nodeInfo.events = nodeInfo.events || [];
            nodeInfo.events.push({
              name: name.slice(3),
              value
            });
            return true;
          }
          // static id
          else if (name === 'id') {
            nodeInfo.id = value;
            return true;
          }
          return false;
        }

        /**
         * Returns the `content` document fragment for a given template.
         *
         * For nested templates, Polymer performs an optimization to cache nested
         * template content to avoid the cost of cloning deeply nested templates.
         * This method retrieves the cached content for a given template.
         *
         * @param {HTMLTemplateElement} template Template to retrieve `content` for
         * @return {DocumentFragment} Content fragment
         * @nocollapse
         */
        static _contentForTemplate(template) {
          let templateInfo = /** @type {HTMLTemplateElementWithInfo} */ (template)._templateInfo;
          return (templateInfo && templateInfo.content) || template.content;
        }

        /**
         * Clones the provided template content and returns a document fragment
         * containing the cloned dom.
         *
         * The template is parsed (once and memoized) using this library's
         * template parsing features, and provides the following value-added
         * features:
         * * Adds declarative event listeners for `on-event="handler"` attributes
         * * Generates an "id map" for all nodes with id's under `$` on returned
         *   document fragment
         * * Passes template info including `content` back to templates as
         *   `_templateInfo` (a performance optimization to avoid deep template
         *   cloning)
         *
         * Note that the memoized template parsing process is destructive to the
         * template: attributes for bindings and declarative event listeners are
         * removed after being noted in notes, and any nested `<template>.content`
         * is removed and stored in notes as well.
         *
         * @param {!HTMLTemplateElement} template Template to stamp
         * @param {TemplateInfo=} templateInfo Optional template info associated
         *   with the template to be stamped; if omitted the template will be
         *   automatically parsed.
         * @return {!StampedTemplate} Cloned template content
         * @override
         */
        _stampTemplate(template, templateInfo) {
          // Polyfill support: bootstrap the template if it has not already been
          if (template && !template.content &&
              window.HTMLTemplateElement && HTMLTemplateElement.decorate) {
            HTMLTemplateElement.decorate(template);
          }
          // Accepting the `templateInfo` via an argument allows for creating
          // instances of the `templateInfo` by the caller, useful for adding
          // instance-time information to the prototypical data
          templateInfo = templateInfo || this.constructor._parseTemplate(template);
          let nodeInfo = templateInfo.nodeInfoList;
          let content = templateInfo.content || template.content;
          let dom = /** @type {DocumentFragment} */ (document.importNode(content, true));
          // NOTE: ShadyDom optimization indicating there is an insertion point
          dom.__noInsertionPoint = !templateInfo.hasInsertionPoint;
          let nodes = dom.nodeList = new Array(nodeInfo.length);
          dom.$ = {};
          for (let i=0, l=nodeInfo.length, info; (i<l) && (info=nodeInfo[i]); i++) {
            let node = nodes[i] = findTemplateNode(dom, info);
            applyIdToMap(this, dom.$, node, info);
            applyTemplateInfo(this, node, info, templateInfo);
            applyEventListener(this, node, info);
          }
          dom = /** @type {!StampedTemplate} */(dom); // eslint-disable-line no-self-assign
          return dom;
        }

        /**
         * Adds an event listener by method name for the event provided.
         *
         * This method generates a handler function that looks up the method
         * name at handling time.
         *
         * @param {!EventTarget} node Node to add listener on
         * @param {string} eventName Name of event
         * @param {string} methodName Name of method
         * @param {*=} context Context the method will be called on (defaults
         *   to `node`)
         * @return {Function} Generated handler function
         * @override
         */
        _addMethodEventListenerToNode(node, eventName, methodName, context) {
          context = context || node;
          let handler = createNodeEventHandler(context, eventName, methodName);
          this._addEventListenerToNode(node, eventName, handler);
          return handler;
        }

        /**
         * Override point for adding custom or simulated event handling.
         *
         * @param {!EventTarget} node Node to add event listener to
         * @param {string} eventName Name of event
         * @param {function(!Event):void} handler Listener function to add
         * @return {void}
         * @override
         */
        _addEventListenerToNode(node, eventName, handler) {
          node.addEventListener(eventName, handler);
        }

        /**
         * Override point for adding custom or simulated event handling.
         *
         * @param {!EventTarget} node Node to remove event listener from
         * @param {string} eventName Name of event
         * @param {function(!Event):void} handler Listener function to remove
         * @return {void}
         * @override
         */
        _removeEventListenerFromNode(node, eventName, handler) {
          node.removeEventListener(eventName, handler);
        }

      }

      return TemplateStamp;

    });

    /**
     * @fileoverview
     * @suppress {checkPrototypalTypes}
     * @license Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
     * This code may only be used under the BSD style license found at
     * http://polymer.github.io/LICENSE.txt The complete set of authors may be found
     * at http://polymer.github.io/AUTHORS.txt The complete set of contributors may
     * be found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by
     * Google as part of the polymer project is also subject to an additional IP
     * rights grant found at http://polymer.github.io/PATENTS.txt
     */

    // Monotonically increasing unique ID used for de-duping effects triggered
    // from multiple properties in the same turn
    let dedupeId$1 = 0;

    const NOOP = [];

    /**
     * Property effect types; effects are stored on the prototype using these keys
     * @enum {string}
     */
    const TYPES = {
      COMPUTE: '__computeEffects',
      REFLECT: '__reflectEffects',
      NOTIFY: '__notifyEffects',
      PROPAGATE: '__propagateEffects',
      OBSERVE: '__observeEffects',
      READ_ONLY: '__readOnly'
    };

    const COMPUTE_INFO = '__computeInfo';

    /** @const {!RegExp} */
    const capitalAttributeRegex = /[A-Z]/;

    /**
     * Ensures that the model has an own-property map of effects for the given type.
     * The model may be a prototype or an instance.
     *
     * Property effects are stored as arrays of effects by property in a map,
     * by named type on the model. e.g.
     *
     *   __computeEffects: {
     *     foo: [ ... ],
     *     bar: [ ... ]
     *   }
     *
     * If the model does not yet have an effect map for the type, one is created
     * and returned.  If it does, but it is not an own property (i.e. the
     * prototype had effects), the the map is deeply cloned and the copy is
     * set on the model and returned, ready for new effects to be added.
     *
     * @param {Object} model Prototype or instance
     * @param {string} type Property effect type
     * @param {boolean=} cloneArrays Clone any arrays assigned to the map when
     *   extending a superclass map onto this subclass
     * @return {Object} The own-property map of effects for the given type
     * @private
     */
    function ensureOwnEffectMap(model, type, cloneArrays) {
      let effects = model[type];
      if (!effects) {
        effects = model[type] = {};
      } else if (!model.hasOwnProperty(type)) {
        effects = model[type] = Object.create(model[type]);
        if (cloneArrays) {
          for (let p in effects) {
            let protoFx = effects[p];
            // Perf optimization over Array.slice
            let instFx = effects[p] = Array(protoFx.length);
            for (let i=0; i<protoFx.length; i++) {
              instFx[i] = protoFx[i];
            }
          }
        }
      }
      return effects;
    }

    // -- effects ----------------------------------------------

    /**
     * Runs all effects of a given type for the given set of property changes
     * on an instance.
     *
     * @param {!Polymer_PropertyEffects} inst The instance with effects to run
     * @param {?Object} effects Object map of property-to-Array of effects
     * @param {?Object} props Bag of current property changes
     * @param {?Object=} oldProps Bag of previous values for changed properties
     * @param {boolean=} hasPaths True with `props` contains one or more paths
     * @param {*=} extraArgs Additional metadata to pass to effect function
     * @return {boolean} True if an effect ran for this property
     * @private
     */
    function runEffects(inst, effects, props, oldProps, hasPaths, extraArgs) {
      if (effects) {
        let ran = false;
        const id = dedupeId$1++;
        for (let prop in props) {
          // Inline `runEffectsForProperty` for perf.
          let rootProperty = hasPaths ? root(prop) : prop;
          let fxs = effects[rootProperty];
          if (fxs) {
            for (let i=0, l=fxs.length, fx; (i<l) && (fx=fxs[i]); i++) {
              if ((!fx.info || fx.info.lastRun !== id) &&
                  (!hasPaths || pathMatchesTrigger(prop, fx.trigger))) {
                if (fx.info) {
                  fx.info.lastRun = id;
                }
                fx.fn(inst, prop, props, oldProps, fx.info, hasPaths, extraArgs);
                ran = true;
              }
            }
          }
        }
        return ran;
      }
      return false;
    }

    /**
     * Runs a list of effects for a given property.
     *
     * @param {!Polymer_PropertyEffects} inst The instance with effects to run
     * @param {!Object} effects Object map of property-to-Array of effects
     * @param {number} dedupeId Counter used for de-duping effects
     * @param {string} prop Name of changed property
     * @param {*} props Changed properties
     * @param {*} oldProps Old properties
     * @param {boolean=} hasPaths True with `props` contains one or more paths
     * @param {*=} extraArgs Additional metadata to pass to effect function
     * @return {boolean} True if an effect ran for this property
     * @private
     */
    function runEffectsForProperty(inst, effects, dedupeId, prop, props, oldProps, hasPaths, extraArgs) {
      let ran = false;
      let rootProperty = hasPaths ? root(prop) : prop;
      let fxs = effects[rootProperty];
      if (fxs) {
        for (let i=0, l=fxs.length, fx; (i<l) && (fx=fxs[i]); i++) {
          if ((!fx.info || fx.info.lastRun !== dedupeId) &&
              (!hasPaths || pathMatchesTrigger(prop, fx.trigger))) {
            if (fx.info) {
              fx.info.lastRun = dedupeId;
            }
            fx.fn(inst, prop, props, oldProps, fx.info, hasPaths, extraArgs);
            ran = true;
          }
        }
      }
      return ran;
    }

    /**
     * Determines whether a property/path that has changed matches the trigger
     * criteria for an effect.  A trigger is a descriptor with the following
     * structure, which matches the descriptors returned from `parseArg`.
     * e.g. for `foo.bar.*`:
     * ```
     * trigger: {
     *   name: 'a.b',
     *   structured: true,
     *   wildcard: true
     * }
     * ```
     * If no trigger is given, the path is deemed to match.
     *
     * @param {string} path Path or property that changed
     * @param {?DataTrigger} trigger Descriptor
     * @return {boolean} Whether the path matched the trigger
     */
    function pathMatchesTrigger(path, trigger) {
      if (trigger) {
        let triggerPath = /** @type {string} */ (trigger.name);
        return (triggerPath == path) ||
            !!(trigger.structured && isAncestor(triggerPath, path)) ||
            !!(trigger.wildcard && isDescendant(triggerPath, path));
      } else {
        return true;
      }
    }

    /**
     * Implements the "observer" effect.
     *
     * Calls the method with `info.methodName` on the instance, passing the
     * new and old values.
     *
     * @param {!Polymer_PropertyEffects} inst The instance the effect will be run on
     * @param {string} property Name of property
     * @param {Object} props Bag of current property changes
     * @param {Object} oldProps Bag of previous values for changed properties
     * @param {?} info Effect metadata
     * @return {void}
     * @private
     */
    function runObserverEffect(inst, property, props, oldProps, info) {
      let fn = typeof info.method === "string" ? inst[info.method] : info.method;
      let changedProp = info.property;
      if (fn) {
        fn.call(inst, inst.__data[changedProp], oldProps[changedProp]);
      } else if (!info.dynamicFn) {
        console.warn('observer method `' + info.method + '` not defined');
      }
    }

    /**
     * Runs "notify" effects for a set of changed properties.
     *
     * This method differs from the generic `runEffects` method in that it
     * will dispatch path notification events in the case that the property
     * changed was a path and the root property for that path didn't have a
     * "notify" effect.  This is to maintain 1.0 behavior that did not require
     * `notify: true` to ensure object sub-property notifications were
     * sent.
     *
     * @param {!Polymer_PropertyEffects} inst The instance with effects to run
     * @param {Object} notifyProps Bag of properties to notify
     * @param {Object} props Bag of current property changes
     * @param {Object} oldProps Bag of previous values for changed properties
     * @param {boolean} hasPaths True with `props` contains one or more paths
     * @return {void}
     * @private
     */
    function runNotifyEffects(inst, notifyProps, props, oldProps, hasPaths) {
      // Notify
      let fxs = inst[TYPES.NOTIFY];
      let notified;
      let id = dedupeId$1++;
      // Try normal notify effects; if none, fall back to try path notification
      for (let prop in notifyProps) {
        if (notifyProps[prop]) {
          if (fxs && runEffectsForProperty(inst, fxs, id, prop, props, oldProps, hasPaths)) {
            notified = true;
          } else if (hasPaths && notifyPath(inst, prop, props)) {
            notified = true;
          }
        }
      }
      // Flush host if we actually notified and host was batching
      // And the host has already initialized clients; this prevents
      // an issue with a host observing data changes before clients are ready.
      let host;
      if (notified && (host = inst.__dataHost) && host._invalidateProperties) {
        host._invalidateProperties();
      }
    }

    /**
     * Dispatches {property}-changed events with path information in the detail
     * object to indicate a sub-path of the property was changed.
     *
     * @param {!Polymer_PropertyEffects} inst The element from which to fire the
     *     event
     * @param {string} path The path that was changed
     * @param {Object} props Bag of current property changes
     * @return {boolean} Returns true if the path was notified
     * @private
     */
    function notifyPath(inst, path, props) {
      let rootProperty = root(path);
      if (rootProperty !== path) {
        let eventName = camelToDashCase(rootProperty) + '-changed';
        dispatchNotifyEvent(inst, eventName, props[path], path);
        return true;
      }
      return false;
    }

    /**
     * Dispatches {property}-changed events to indicate a property (or path)
     * changed.
     *
     * @param {!Polymer_PropertyEffects} inst The element from which to fire the
     *     event
     * @param {string} eventName The name of the event to send
     *     ('{property}-changed')
     * @param {*} value The value of the changed property
     * @param {string | null | undefined} path If a sub-path of this property
     *     changed, the path that changed (optional).
     * @return {void}
     * @private
     * @suppress {invalidCasts}
     */
    function dispatchNotifyEvent(inst, eventName, value, path) {
      let detail = {
        value: value,
        queueProperty: true
      };
      if (path) {
        detail.path = path;
      }
      // As a performance optimization, we could elide the wrap here since notifying
      // events are non-bubbling and shouldn't need retargeting. However, a very
      // small number of internal tests failed in obscure ways, which may indicate
      // user code relied on timing differences resulting from ShadyDOM flushing
      // as a result of the wrapped `dispatchEvent`.
      wrap(/** @type {!HTMLElement} */(inst)).dispatchEvent(new CustomEvent(eventName, { detail }));
    }

    /**
     * Implements the "notify" effect.
     *
     * Dispatches a non-bubbling event named `info.eventName` on the instance
     * with a detail object containing the new `value`.
     *
     * @param {!Polymer_PropertyEffects} inst The instance the effect will be run on
     * @param {string} property Name of property
     * @param {Object} props Bag of current property changes
     * @param {Object} oldProps Bag of previous values for changed properties
     * @param {?} info Effect metadata
     * @param {boolean} hasPaths True with `props` contains one or more paths
     * @return {void}
     * @private
     */
    function runNotifyEffect(inst, property, props, oldProps, info, hasPaths) {
      let rootProperty = hasPaths ? root(property) : property;
      let path = rootProperty != property ? property : null;
      let value = path ? get(inst, path) : inst.__data[property];
      if (path && value === undefined) {
        value = props[property];  // specifically for .splices
      }
      dispatchNotifyEvent(inst, info.eventName, value, path);
    }

    /**
     * Handler function for 2-way notification events. Receives context
     * information captured in the `addNotifyListener` closure from the
     * `__notifyListeners` metadata.
     *
     * Sets the value of the notified property to the host property or path.  If
     * the event contained path information, translate that path to the host
     * scope's name for that path first.
     *
     * @param {CustomEvent} event Notification event (e.g. '<property>-changed')
     * @param {!Polymer_PropertyEffects} inst Host element instance handling the
     *     notification event
     * @param {string} fromProp Child element property that was bound
     * @param {string} toPath Host property/path that was bound
     * @param {boolean} negate Whether the binding was negated
     * @return {void}
     * @private
     */
    function handleNotification(event, inst, fromProp, toPath, negate) {
      let value;
      let detail = /** @type {Object} */(event.detail);
      let fromPath = detail && detail.path;
      if (fromPath) {
        toPath = translate(fromProp, toPath, fromPath);
        value = detail && detail.value;
      } else {
        value = event.currentTarget[fromProp];
      }
      value = negate ? !value : value;
      if (!inst[TYPES.READ_ONLY] || !inst[TYPES.READ_ONLY][toPath]) {
        if (inst._setPendingPropertyOrPath(toPath, value, true, Boolean(fromPath))
          && (!detail || !detail.queueProperty)) {
          inst._invalidateProperties();
        }
      }
    }

    /**
     * Implements the "reflect" effect.
     *
     * Sets the attribute named `info.attrName` to the given property value.
     *
     * @param {!Polymer_PropertyEffects} inst The instance the effect will be run on
     * @param {string} property Name of property
     * @param {Object} props Bag of current property changes
     * @param {Object} oldProps Bag of previous values for changed properties
     * @param {?} info Effect metadata
     * @return {void}
     * @private
     */
    function runReflectEffect(inst, property, props, oldProps, info) {
      let value = inst.__data[property];
      if (sanitizeDOMValue) {
        value = sanitizeDOMValue(value, info.attrName, 'attribute', /** @type {Node} */(inst));
      }
      inst._propertyToAttribute(property, info.attrName, value);
    }

    /**
     * Runs "computed" effects for a set of changed properties.
     *
     * This method differs from the generic `runEffects` method in that it
     * continues to run computed effects based on the output of each pass until
     * there are no more newly computed properties.  This ensures that all
     * properties that will be computed by the initial set of changes are
     * computed before other effects (binding propagation, observers, and notify)
     * run.
     *
     * @param {!Polymer_PropertyEffects} inst The instance the effect will be run on
     * @param {?Object} changedProps Bag of changed properties
     * @param {?Object} oldProps Bag of previous values for changed properties
     * @param {boolean} hasPaths True with `props` contains one or more paths
     * @return {void}
     * @private
     */
    function runComputedEffects(inst, changedProps, oldProps, hasPaths) {
      let computeEffects = inst[TYPES.COMPUTE];
      if (computeEffects) {
        if (orderedComputed) {
          // Runs computed effects in efficient order by keeping a topologically-
          // sorted queue of compute effects to run, and inserting subsequently
          // invalidated effects as they are run
          dedupeId$1++;
          const order = getComputedOrder(inst);
          const queue = [];
          for (let p in changedProps) {
            enqueueEffectsFor(p, computeEffects, queue, order, hasPaths);
          }
          let info;
          while ((info = queue.shift())) {
            if (runComputedEffect(inst, '', changedProps, oldProps, info)) {
              enqueueEffectsFor(info.methodInfo, computeEffects, queue, order, hasPaths);
            }
          }
          Object.assign(/** @type {!Object} */ (oldProps), inst.__dataOld);
          Object.assign(/** @type {!Object} */ (changedProps), inst.__dataPending);
          inst.__dataPending = null;
        } else {
          // Original Polymer 2.x computed effects order, which continues running
          // effects until no further computed properties have been invalidated
          let inputProps = changedProps;
          while (runEffects(inst, computeEffects, inputProps, oldProps, hasPaths)) {
            Object.assign(/** @type {!Object} */ (oldProps), inst.__dataOld);
            Object.assign(/** @type {!Object} */ (changedProps), inst.__dataPending);
            inputProps = inst.__dataPending;
            inst.__dataPending = null;
          }
        }
      }
    }

    /**
     * Inserts a computed effect into a queue, given the specified order. Performs
     * the insert using a binary search.
     *
     * Used by `orderedComputed: true` computed property algorithm.
     *
     * @param {Object} info Property effects metadata
     * @param {Array<Object>} queue Ordered queue of effects
     * @param {Map<string,number>} order Map of computed property name->topological
     *   sort order
     */
    const insertEffect = (info, queue, order) => {
      let start = 0;
      let end = queue.length - 1;
      let idx = -1;
      while (start <= end) {
        const mid = (start + end) >> 1;
        // Note `methodInfo` is where the computed property name is stored in
        // the effect metadata
        const cmp = order.get(queue[mid].methodInfo) - order.get(info.methodInfo);
        if (cmp < 0) {
          start = mid + 1;
        } else if (cmp > 0) {
          end = mid - 1;
        } else {
          idx = mid;
          break;
        }
      }
      if (idx < 0) {
        idx = end + 1;
      }
      queue.splice(idx, 0, info);
    };

    /**
     * Inserts all downstream computed effects invalidated by the specified property
     * into the topologically-sorted queue of effects to be run.
     *
     * Used by `orderedComputed: true` computed property algorithm.
     *
     * @param {string} prop Property name
     * @param {Object} computeEffects Computed effects for this element
     * @param {Array<Object>} queue Topologically-sorted queue of computed effects
     *   to be run
     * @param {Map<string,number>} order Map of computed property name->topological
     *   sort order
     * @param {boolean} hasPaths True with `changedProps` contains one or more paths
     */
    const enqueueEffectsFor = (prop, computeEffects, queue, order, hasPaths) => {
      const rootProperty = hasPaths ? root(prop) : prop;
      const fxs = computeEffects[rootProperty];
      if (fxs) {
        for (let i=0; i<fxs.length; i++) {
          const fx = fxs[i];
          if ((fx.info.lastRun !== dedupeId$1) &&
              (!hasPaths || pathMatchesTrigger(prop, fx.trigger))) {
            fx.info.lastRun = dedupeId$1;
            insertEffect(fx.info, queue, order);
          }
        }
      }
    };

    /**
     * Generates and retrieves a memoized map of computed property name to its
     * topologically-sorted order.
     *
     * The map is generated by first assigning a "dependency count" to each property
     * (defined as number properties it depends on, including its method for
     * "dynamic functions"). Any properties that have no dependencies are added to
     * the `ready` queue, which are properties whose order can be added to the final
     * order map. Properties are popped off the `ready` queue one by one and a.) added as
     * the next property in the order map, and b.) each property that it is a
     * dependency for has its dep count decremented (and if that property's dep
     * count goes to zero, it is added to the `ready` queue), until all properties
     * have been visited and ordered.
     *
     * Used by `orderedComputed: true` computed property algorithm.
     *
     * @param {!Polymer_PropertyEffects} inst The instance to retrieve the computed
     *   effect order for.
     * @return {Map<string,number>} Map of computed property name->topological sort
     *   order
     */
    function getComputedOrder(inst) {
      let ordered = inst.constructor.__orderedComputedDeps;
      if (!ordered) {
        ordered = new Map();
        const effects = inst[TYPES.COMPUTE];
        let {counts, ready, total} = dependencyCounts(inst);
        let curr;
        while ((curr = ready.shift())) {
          ordered.set(curr, ordered.size);
          const computedByCurr = effects[curr];
          if (computedByCurr) {
            computedByCurr.forEach(fx => {
              // Note `methodInfo` is where the computed property name is stored
              const computedProp = fx.info.methodInfo;
              --total;
              if (--counts[computedProp] === 0) {
                ready.push(computedProp);
              }
            });
          }
        }
        if (total !== 0) {
          const el = /** @type {HTMLElement} */ (inst);
          console.warn(`Computed graph for ${el.localName} incomplete; circular?`);
        }
        inst.constructor.__orderedComputedDeps = ordered;
      }
      return ordered;
    }

    /**
     * Generates a map of property-to-dependency count (`counts`, where "dependency
     * count" is the number of dependencies a given property has assuming it is a
     * computed property, otherwise 0).  It also returns a pre-populated list of
     * `ready` properties that have no dependencies and a `total` count, which is
     * used for error-checking the graph.
     *
     * Used by `orderedComputed: true` computed property algorithm.
     *
     * @param {!Polymer_PropertyEffects} inst The instance to generate dependency
     *   counts for.
     * @return {!Object} Object containing `counts` map (property-to-dependency
     *   count) and pre-populated `ready` array of properties that had zero
     *   dependencies.
     */
    function dependencyCounts(inst) {
      const infoForComputed = inst[COMPUTE_INFO];
      const counts = {};
      const computedDeps = inst[TYPES.COMPUTE];
      const ready = [];
      let total = 0;
      // Count dependencies for each computed property
      for (let p in infoForComputed) {
        const info = infoForComputed[p];
        // Be sure to add the method name itself in case of "dynamic functions"
        total += counts[p] =
          info.args.filter(a => !a.literal).length + (info.dynamicFn ? 1 : 0);
      }
      // Build list of ready properties (that aren't themselves computed)
      for (let p in computedDeps) {
        if (!infoForComputed[p]) {
          ready.push(p);
        }
      }
      return {counts, ready, total};
    }

    /**
     * Implements the "computed property" effect by running the method with the
     * values of the arguments specified in the `info` object and setting the
     * return value to the computed property specified.
     *
     * @param {!Polymer_PropertyEffects} inst The instance the effect will be run on
     * @param {string} property Name of property
     * @param {?Object} changedProps Bag of current property changes
     * @param {?Object} oldProps Bag of previous values for changed properties
     * @param {?} info Effect metadata
     * @return {boolean} True when the property being computed changed
     * @private
     */
    function runComputedEffect(inst, property, changedProps, oldProps, info) {
      // Dirty check dependencies and run if any invalid
      let result = runMethodEffect(inst, property, changedProps, oldProps, info);
      // Abort if method returns a no-op result
      if (result === NOOP) {
        return false;
      }
      let computedProp = info.methodInfo;
      if (inst.__dataHasAccessor && inst.__dataHasAccessor[computedProp]) {
        return inst._setPendingProperty(computedProp, result, true);
      } else {
        inst[computedProp] = result;
        return false;
      }
    }

    /**
     * Computes path changes based on path links set up using the `linkPaths`
     * API.
     *
     * @param {!Polymer_PropertyEffects} inst The instance whose props are changing
     * @param {string} path Path that has changed
     * @param {*} value Value of changed path
     * @return {void}
     * @private
     */
    function computeLinkedPaths(inst, path, value) {
      let links = inst.__dataLinkedPaths;
      if (links) {
        let link;
        for (let a in links) {
          let b = links[a];
          if (isDescendant(a, path)) {
            link = translate(a, b, path);
            inst._setPendingPropertyOrPath(link, value, true, true);
          } else if (isDescendant(b, path)) {
            link = translate(b, a, path);
            inst._setPendingPropertyOrPath(link, value, true, true);
          }
        }
      }
    }

    // -- bindings ----------------------------------------------

    /**
     * Adds binding metadata to the current `nodeInfo`, and binding effects
     * for all part dependencies to `templateInfo`.
     *
     * @param {Function} constructor Class that `_parseTemplate` is currently
     *   running on
     * @param {TemplateInfo} templateInfo Template metadata for current template
     * @param {NodeInfo} nodeInfo Node metadata for current template node
     * @param {string} kind Binding kind, either 'property', 'attribute', or 'text'
     * @param {string} target Target property name
     * @param {!Array<!BindingPart>} parts Array of binding part metadata
     * @param {string=} literal Literal text surrounding binding parts (specified
     *   only for 'property' bindings, since these must be initialized as part
     *   of boot-up)
     * @return {void}
     * @private
     */
    function addBinding(constructor, templateInfo, nodeInfo, kind, target, parts, literal) {
      // Create binding metadata and add to nodeInfo
      nodeInfo.bindings = nodeInfo.bindings || [];
      let /** Binding */ binding = { kind, target, parts, literal, isCompound: (parts.length !== 1) };
      nodeInfo.bindings.push(binding);
      // Add listener info to binding metadata
      if (shouldAddListener(binding)) {
        let {event, negate} = binding.parts[0];
        binding.listenerEvent = event || (camelToDashCase(target) + '-changed');
        binding.listenerNegate = negate;
      }
      // Add "propagate" property effects to templateInfo
      let index = templateInfo.nodeInfoList.length;
      for (let i=0; i<binding.parts.length; i++) {
        let part = binding.parts[i];
        part.compoundIndex = i;
        addEffectForBindingPart(constructor, templateInfo, binding, part, index);
      }
    }

    /**
     * Adds property effects to the given `templateInfo` for the given binding
     * part.
     *
     * @param {Function} constructor Class that `_parseTemplate` is currently
     *   running on
     * @param {TemplateInfo} templateInfo Template metadata for current template
     * @param {!Binding} binding Binding metadata
     * @param {!BindingPart} part Binding part metadata
     * @param {number} index Index into `nodeInfoList` for this node
     * @return {void}
     */
    function addEffectForBindingPart(constructor, templateInfo, binding, part, index) {
      if (!part.literal) {
        if (binding.kind === 'attribute' && binding.target[0] === '-') {
          console.warn('Cannot set attribute ' + binding.target +
            ' because "-" is not a valid attribute starting character');
        } else {
          let dependencies = part.dependencies;
          let info = { index, binding, part, evaluator: constructor };
          for (let j=0; j<dependencies.length; j++) {
            let trigger = dependencies[j];
            if (typeof trigger == 'string') {
              trigger = parseArg(trigger);
              trigger.wildcard = true;
            }
            constructor._addTemplatePropertyEffect(templateInfo, trigger.rootProperty, {
              fn: runBindingEffect,
              info, trigger
            });
          }
        }
      }
    }

    /**
     * Implements the "binding" (property/path binding) effect.
     *
     * Note that binding syntax is overridable via `_parseBindings` and
     * `_evaluateBinding`.  This method will call `_evaluateBinding` for any
     * non-literal parts returned from `_parseBindings`.  However,
     * there is no support for _path_ bindings via custom binding parts,
     * as this is specific to Polymer's path binding syntax.
     *
     * @param {!Polymer_PropertyEffects} inst The instance the effect will be run on
     * @param {string} path Name of property
     * @param {Object} props Bag of current property changes
     * @param {Object} oldProps Bag of previous values for changed properties
     * @param {?} info Effect metadata
     * @param {boolean} hasPaths True with `props` contains one or more paths
     * @param {Array} nodeList List of nodes associated with `nodeInfoList` template
     *   metadata
     * @return {void}
     * @private
     */
    function runBindingEffect(inst, path, props, oldProps, info, hasPaths, nodeList) {
      let node = nodeList[info.index];
      let binding = info.binding;
      let part = info.part;
      // Subpath notification: transform path and set to client
      // e.g.: foo="{{obj.sub}}", path: 'obj.sub.prop', set 'foo.prop'=obj.sub.prop
      if (hasPaths && part.source && (path.length > part.source.length) &&
          (binding.kind == 'property') && !binding.isCompound &&
          node.__isPropertyEffectsClient &&
          node.__dataHasAccessor && node.__dataHasAccessor[binding.target]) {
        let value = props[path];
        path = translate(part.source, binding.target, path);
        if (node._setPendingPropertyOrPath(path, value, false, true)) {
          inst._enqueueClient(node);
        }
      } else {
        let value = info.evaluator._evaluateBinding(inst, part, path, props, oldProps, hasPaths);
        // Propagate value to child
        // Abort if value is a no-op result
        if (value !== NOOP) {
          applyBindingValue(inst, node, binding, part, value);
        }
      }
    }

    /**
     * Sets the value for an "binding" (binding) effect to a node,
     * either as a property or attribute.
     *
     * @param {!Polymer_PropertyEffects} inst The instance owning the binding effect
     * @param {Node} node Target node for binding
     * @param {!Binding} binding Binding metadata
     * @param {!BindingPart} part Binding part metadata
     * @param {*} value Value to set
     * @return {void}
     * @private
     */
    function applyBindingValue(inst, node, binding, part, value) {
      value = computeBindingValue(node, value, binding, part);
      if (sanitizeDOMValue) {
        value = sanitizeDOMValue(value, binding.target, binding.kind, node);
      }
      if (binding.kind == 'attribute') {
        // Attribute binding
        inst._valueToNodeAttribute(/** @type {Element} */(node), value, binding.target);
      } else {
        // Property binding
        let prop = binding.target;
        if (node.__isPropertyEffectsClient &&
            node.__dataHasAccessor && node.__dataHasAccessor[prop]) {
          if (!node[TYPES.READ_ONLY] || !node[TYPES.READ_ONLY][prop]) {
            if (node._setPendingProperty(prop, value)) {
              inst._enqueueClient(node);
            }
          }
        } else {
          // In legacy no-batching mode, bindings applied before dataReady are
          // equivalent to the "apply config" phase, which only set managed props
          inst._setUnmanagedPropertyToNode(node, prop, value);
        }
      }
    }

    /**
     * Transforms an "binding" effect value based on compound & negation
     * effect metadata, as well as handling for special-case properties
     *
     * @param {Node} node Node the value will be set to
     * @param {*} value Value to set
     * @param {!Binding} binding Binding metadata
     * @param {!BindingPart} part Binding part metadata
     * @return {*} Transformed value to set
     * @private
     */
    function computeBindingValue(node, value, binding, part) {
      if (binding.isCompound) {
        let storage = node.__dataCompoundStorage[binding.target];
        storage[part.compoundIndex] = value;
        value = storage.join('');
      }
      if (binding.kind !== 'attribute') {
        // Some browsers serialize `undefined` to `"undefined"`
        if (binding.target === 'textContent' ||
            (binding.target === 'value' &&
              (node.localName === 'input' || node.localName === 'textarea'))) {
          value = value == undefined ? '' : value;
        }
      }
      return value;
    }

    /**
     * Returns true if a binding's metadata meets all the requirements to allow
     * 2-way binding, and therefore a `<property>-changed` event listener should be
     * added:
     * - used curly braces
     * - is a property (not attribute) binding
     * - is not a textContent binding
     * - is not compound
     *
     * @param {!Binding} binding Binding metadata
     * @return {boolean} True if 2-way listener should be added
     * @private
     */
    function shouldAddListener(binding) {
      return Boolean(binding.target) &&
             binding.kind != 'attribute' &&
             binding.kind != 'text' &&
             !binding.isCompound &&
             binding.parts[0].mode === '{';
    }

    /**
     * Setup compound binding storage structures, notify listeners, and dataHost
     * references onto the bound nodeList.
     *
     * @param {!Polymer_PropertyEffects} inst Instance that bas been previously
     *     bound
     * @param {TemplateInfo} templateInfo Template metadata
     * @return {void}
     * @private
     */
    function setupBindings(inst, templateInfo) {
      // Setup compound storage, dataHost, and notify listeners
      let {nodeList, nodeInfoList} = templateInfo;
      if (nodeInfoList.length) {
        for (let i=0; i < nodeInfoList.length; i++) {
          let info = nodeInfoList[i];
          let node = nodeList[i];
          let bindings = info.bindings;
          if (bindings) {
            for (let i=0; i<bindings.length; i++) {
              let binding = bindings[i];
              setupCompoundStorage(node, binding);
              addNotifyListener(node, inst, binding);
            }
          }
          // This ensures all bound elements have a host set, regardless
          // of whether they upgrade synchronous to creation
          node.__dataHost = inst;
        }
      }
    }

    /**
     * Initializes `__dataCompoundStorage` local storage on a bound node with
     * initial literal data for compound bindings, and sets the joined
     * literal parts to the bound property.
     *
     * When changes to compound parts occur, they are first set into the compound
     * storage array for that property, and then the array is joined to result in
     * the final value set to the property/attribute.
     *
     * @param {Node} node Bound node to initialize
     * @param {Binding} binding Binding metadata
     * @return {void}
     * @private
     */
    function setupCompoundStorage(node, binding) {
      if (binding.isCompound) {
        // Create compound storage map
        let storage = node.__dataCompoundStorage ||
          (node.__dataCompoundStorage = {});
        let parts = binding.parts;
        // Copy literals from parts into storage for this binding
        let literals = new Array(parts.length);
        for (let j=0; j<parts.length; j++) {
          literals[j] = parts[j].literal;
        }
        let target = binding.target;
        storage[target] = literals;
        // Configure properties with their literal parts
        if (binding.literal && binding.kind == 'property') {
          // Note, className needs style scoping so this needs wrapping.
          // We may also want to consider doing this for `textContent` and
          // `innerHTML`.
          if (target === 'className') {
            node = wrap(node);
          }
          node[target] = binding.literal;
        }
      }
    }

    /**
     * Adds a 2-way binding notification event listener to the node specified
     *
     * @param {Object} node Child element to add listener to
     * @param {!Polymer_PropertyEffects} inst Host element instance to handle
     *     notification event
     * @param {Binding} binding Binding metadata
     * @return {void}
     * @private
     */
    function addNotifyListener(node, inst, binding) {
      if (binding.listenerEvent) {
        let part = binding.parts[0];
        node.addEventListener(binding.listenerEvent, function(e) {
          handleNotification(e, inst, binding.target, part.source, part.negate);
        });
      }
    }

    // -- for method-based effects (complexObserver & computed) --------------

    /**
     * Adds property effects for each argument in the method signature (and
     * optionally, for the method name if `dynamic` is true) that calls the
     * provided effect function.
     *
     * @param {Element | Object} model Prototype or instance
     * @param {!MethodSignature} sig Method signature metadata
     * @param {string} type Type of property effect to add
     * @param {Function} effectFn Function to run when arguments change
     * @param {*=} methodInfo Effect-specific information to be included in
     *   method effect metadata
     * @param {boolean|Object=} dynamicFn Boolean or object map indicating whether
     *   method names should be included as a dependency to the effect. Note,
     *   defaults to true if the signature is static (sig.static is true).
     * @return {!Object} Effect metadata for this method effect
     * @private
     */
    function createMethodEffect(model, sig, type, effectFn, methodInfo, dynamicFn) {
      dynamicFn = sig.static || (dynamicFn &&
        (typeof dynamicFn !== 'object' || dynamicFn[sig.methodName]));
      let info = {
        methodName: sig.methodName,
        args: sig.args,
        methodInfo,
        dynamicFn
      };
      for (let i=0, arg; (i<sig.args.length) && (arg=sig.args[i]); i++) {
        if (!arg.literal) {
          model._addPropertyEffect(arg.rootProperty, type, {
            fn: effectFn, info: info, trigger: arg
          });
        }
      }
      if (dynamicFn) {
        model._addPropertyEffect(sig.methodName, type, {
          fn: effectFn, info: info
        });
      }
      return info;
    }

    /**
     * Calls a method with arguments marshaled from properties on the instance
     * based on the method signature contained in the effect metadata.
     *
     * Multi-property observers, computed properties, and inline computing
     * functions call this function to invoke the method, then use the return
     * value accordingly.
     *
     * @param {!Polymer_PropertyEffects} inst The instance the effect will be run on
     * @param {string} property Name of property
     * @param {Object} props Bag of current property changes
     * @param {Object} oldProps Bag of previous values for changed properties
     * @param {?} info Effect metadata
     * @return {*} Returns the return value from the method invocation
     * @private
     */
    function runMethodEffect(inst, property, props, oldProps, info) {
      // Instances can optionally have a _methodHost which allows redirecting where
      // to find methods. Currently used by `templatize`.
      let context = inst._methodHost || inst;
      let fn = context[info.methodName];
      if (fn) {
        let args = inst._marshalArgs(info.args, property, props);
        return args === NOOP ? NOOP : fn.apply(context, args);
      } else if (!info.dynamicFn) {
        console.warn('method `' + info.methodName + '` not defined');
      }
    }

    const emptyArray = [];

    // Regular expressions used for binding
    const IDENT  = '(?:' + '[a-zA-Z_$][\\w.:$\\-*]*' + ')';
    const NUMBER = '(?:' + '[-+]?[0-9]*\\.?[0-9]+(?:[eE][-+]?[0-9]+)?' + ')';
    const SQUOTE_STRING = '(?:' + '\'(?:[^\'\\\\]|\\\\.)*\'' + ')';
    const DQUOTE_STRING = '(?:' + '"(?:[^"\\\\]|\\\\.)*"' + ')';
    const STRING = '(?:' + SQUOTE_STRING + '|' + DQUOTE_STRING + ')';
    const ARGUMENT = '(?:(' + IDENT + '|' + NUMBER + '|' +  STRING + ')\\s*' + ')';
    const ARGUMENTS = '(?:' + ARGUMENT + '(?:,\\s*' + ARGUMENT + ')*' + ')';
    const ARGUMENT_LIST = '(?:' + '\\(\\s*' +
                                  '(?:' + ARGUMENTS + '?' + ')' +
                                '\\)\\s*' + ')';
    const BINDING = '(' + IDENT + '\\s*' + ARGUMENT_LIST + '?' + ')'; // Group 3
    const OPEN_BRACKET = '(\\[\\[|{{)' + '\\s*';
    const CLOSE_BRACKET = '(?:]]|}})';
    const NEGATE = '(?:(!)\\s*)?'; // Group 2
    const EXPRESSION = OPEN_BRACKET + NEGATE + BINDING + CLOSE_BRACKET;
    const bindingRegex = new RegExp(EXPRESSION, "g");

    /**
     * Create a string from binding parts of all the literal parts
     *
     * @param {!Array<BindingPart>} parts All parts to stringify
     * @return {string} String made from the literal parts
     */
    function literalFromParts(parts) {
      let s = '';
      for (let i=0; i<parts.length; i++) {
        let literal = parts[i].literal;
        s += literal || '';
      }
      return s;
    }

    /**
     * Parses an expression string for a method signature, and returns a metadata
     * describing the method in terms of `methodName`, `static` (whether all the
     * arguments are literals), and an array of `args`
     *
     * @param {string} expression The expression to parse
     * @return {?MethodSignature} The method metadata object if a method expression was
     *   found, otherwise `undefined`
     * @private
     */
    function parseMethod(expression) {
      // tries to match valid javascript property names
      let m = expression.match(/([^\s]+?)\(([\s\S]*)\)/);
      if (m) {
        let methodName = m[1];
        let sig = { methodName, static: true, args: emptyArray };
        if (m[2].trim()) {
          // replace escaped commas with comma entity, split on un-escaped commas
          let args = m[2].replace(/\\,/g, '&comma;').split(',');
          return parseArgs(args, sig);
        } else {
          return sig;
        }
      }
      return null;
    }

    /**
     * Parses an array of arguments and sets the `args` property of the supplied
     * signature metadata object. Sets the `static` property to false if any
     * argument is a non-literal.
     *
     * @param {!Array<string>} argList Array of argument names
     * @param {!MethodSignature} sig Method signature metadata object
     * @return {!MethodSignature} The updated signature metadata object
     * @private
     */
    function parseArgs(argList, sig) {
      sig.args = argList.map(function(rawArg) {
        let arg = parseArg(rawArg);
        if (!arg.literal) {
          sig.static = false;
        }
        return arg;
      }, this);
      return sig;
    }

    /**
     * Parses an individual argument, and returns an argument metadata object
     * with the following fields:
     *
     *   {
     *     value: 'prop',        // property/path or literal value
     *     literal: false,       // whether argument is a literal
     *     structured: false,    // whether the property is a path
     *     rootProperty: 'prop', // the root property of the path
     *     wildcard: false       // whether the argument was a wildcard '.*' path
     *   }
     *
     * @param {string} rawArg The string value of the argument
     * @return {!MethodArg} Argument metadata object
     * @private
     */
    function parseArg(rawArg) {
      // clean up whitespace
      let arg = rawArg.trim()
        // replace comma entity with comma
        .replace(/&comma;/g, ',')
        // repair extra escape sequences; note only commas strictly need
        // escaping, but we allow any other char to be escaped since its
        // likely users will do this
        .replace(/\\(.)/g, '\$1')
        ;
      // basic argument descriptor
      let a = {
        name: arg,
        value: '',
        literal: false
      };
      // detect literal value (must be String or Number)
      let fc = arg[0];
      if (fc === '-') {
        fc = arg[1];
      }
      if (fc >= '0' && fc <= '9') {
        fc = '#';
      }
      switch(fc) {
        case "'":
        case '"':
          a.value = arg.slice(1, -1);
          a.literal = true;
          break;
        case '#':
          a.value = Number(arg);
          a.literal = true;
          break;
      }
      // if not literal, look for structured path
      if (!a.literal) {
        a.rootProperty = root(arg);
        // detect structured path (has dots)
        a.structured = isPath(arg);
        if (a.structured) {
          a.wildcard = (arg.slice(-2) == '.*');
          if (a.wildcard) {
            a.name = arg.slice(0, -2);
          }
        }
      }
      return a;
    }

    function getArgValue(data, props, path) {
      let value = get(data, path);
      // when data is not stored e.g. `splices`, get the value from changedProps
      // TODO(kschaaf): Note, this can cause a rare issue where the wildcard
      // info.value could pull a stale value out of changedProps during a reentrant
      // change that sets the value back to undefined.
      // https://github.com/Polymer/polymer/issues/5479
      if (value === undefined) {
        value = props[path];
      }
      return value;
    }

    // data api

    /**
     * Sends array splice notifications (`.splices` and `.length`)
     *
     * Note: this implementation only accepts normalized paths
     *
     * @param {!Polymer_PropertyEffects} inst Instance to send notifications to
     * @param {Array} array The array the mutations occurred on
     * @param {string} path The path to the array that was mutated
     * @param {Array} splices Array of splice records
     * @return {void}
     * @private
     */
    function notifySplices(inst, array, path, splices) {
      const splicesData = { indexSplices: splices };
      // Legacy behavior stored splices in `__data__` so it was *not* ephemeral.
      // To match this behavior, we store splices directly on the array.
      if (legacyUndefined && !inst._overrideLegacyUndefined) {
        array.splices = splicesData;
      }
      inst.notifyPath(path + '.splices', splicesData);
      inst.notifyPath(path + '.length', array.length);
      // Clear splice data only when it's stored on the array.
      if (legacyUndefined && !inst._overrideLegacyUndefined) {
        splicesData.indexSplices = [];
      }
    }

    /**
     * Creates a splice record and sends an array splice notification for
     * the described mutation
     *
     * Note: this implementation only accepts normalized paths
     *
     * @param {!Polymer_PropertyEffects} inst Instance to send notifications to
     * @param {Array} array The array the mutations occurred on
     * @param {string} path The path to the array that was mutated
     * @param {number} index Index at which the array mutation occurred
     * @param {number} addedCount Number of added items
     * @param {Array} removed Array of removed items
     * @return {void}
     * @private
     */
    function notifySplice(inst, array, path, index, addedCount, removed) {
      notifySplices(inst, array, path, [{
        index: index,
        addedCount: addedCount,
        removed: removed,
        object: array,
        type: 'splice'
      }]);
    }

    /**
     * Returns an upper-cased version of the string.
     *
     * @param {string} name String to uppercase
     * @return {string} Uppercased string
     * @private
     */
    function upper(name) {
      return name[0].toUpperCase() + name.substring(1);
    }

    /**
     * Element class mixin that provides meta-programming for Polymer's template
     * binding and data observation (collectively, "property effects") system.
     *
     * This mixin uses provides the following key static methods for adding
     * property effects to an element class:
     * - `addPropertyEffect`
     * - `createPropertyObserver`
     * - `createMethodObserver`
     * - `createNotifyingProperty`
     * - `createReadOnlyProperty`
     * - `createReflectedProperty`
     * - `createComputedProperty`
     * - `bindTemplate`
     *
     * Each method creates one or more property accessors, along with metadata
     * used by this mixin's implementation of `_propertiesChanged` to perform
     * the property effects.
     *
     * Underscored versions of the above methods also exist on the element
     * prototype for adding property effects on instances at runtime.
     *
     * Note that this mixin overrides several `PropertyAccessors` methods, in
     * many cases to maintain guarantees provided by the Polymer 1.x features;
     * notably it changes property accessors to be synchronous by default
     * whereas the default when using `PropertyAccessors` standalone is to be
     * async by default.
     *
     * @mixinFunction
     * @polymer
     * @appliesMixin TemplateStamp
     * @appliesMixin PropertyAccessors
     * @summary Element class mixin that provides meta-programming for Polymer's
     * template binding and data observation system.
     */
    const PropertyEffects = dedupingMixin(superClass => {

      /**
       * @constructor
       * @implements {Polymer_PropertyAccessors}
       * @implements {Polymer_TemplateStamp}
       * @unrestricted
       * @private
       */
      const propertyEffectsBase = TemplateStamp(PropertyAccessors(superClass));

      /**
       * @polymer
       * @mixinClass
       * @implements {Polymer_PropertyEffects}
       * @extends {propertyEffectsBase}
       * @unrestricted
       */
      class PropertyEffects extends propertyEffectsBase {

        constructor() {
          super();
          /** @type {boolean} */
          // Used to identify users of this mixin, ala instanceof
          this.__isPropertyEffectsClient = true;
          /** @type {boolean} */
          this.__dataClientsReady;
          /** @type {Array} */
          this.__dataPendingClients;
          /** @type {Object} */
          this.__dataToNotify;
          /** @type {Object} */
          this.__dataLinkedPaths;
          /** @type {boolean} */
          this.__dataHasPaths;
          /** @type {Object} */
          this.__dataCompoundStorage;
          /** @type {Polymer_PropertyEffects} */
          this.__dataHost;
          /** @type {!Object} */
          this.__dataTemp;
          /** @type {boolean} */
          this.__dataClientsInitialized;
          /** @type {!Object} */
          this.__data;
          /** @type {!Object|null} */
          this.__dataPending;
          /** @type {!Object} */
          this.__dataOld;
          /** @type {Object} */
          this.__computeEffects;
          /** @type {Object} */
          this.__computeInfo;
          /** @type {Object} */
          this.__reflectEffects;
          /** @type {Object} */
          this.__notifyEffects;
          /** @type {Object} */
          this.__propagateEffects;
          /** @type {Object} */
          this.__observeEffects;
          /** @type {Object} */
          this.__readOnly;
          /** @type {!TemplateInfo} */
          this.__templateInfo;
          /** @type {boolean} */
          this._overrideLegacyUndefined;
        }

        get PROPERTY_EFFECT_TYPES() {
          return TYPES;
        }

        /**
         * @override
         * @return {void}
         */
        _initializeProperties() {
          super._initializeProperties();
          this._registerHost();
          this.__dataClientsReady = false;
          this.__dataPendingClients = null;
          this.__dataToNotify = null;
          this.__dataLinkedPaths = null;
          this.__dataHasPaths = false;
          // May be set on instance prior to upgrade
          this.__dataCompoundStorage = this.__dataCompoundStorage || null;
          this.__dataHost = this.__dataHost || null;
          this.__dataTemp = {};
          this.__dataClientsInitialized = false;
        }

        _registerHost() {
          if (hostStack.length) {
            let host = hostStack[hostStack.length-1];
            host._enqueueClient(this);
            // This ensures even non-bound elements have a host set, as
            // long as they upgrade synchronously
            this.__dataHost = host;
          }
        }

        /**
         * Overrides `PropertyAccessors` implementation to provide a
         * more efficient implementation of initializing properties from
         * the prototype on the instance.
         *
         * @override
         * @param {Object} props Properties to initialize on the prototype
         * @return {void}
         */
        _initializeProtoProperties(props) {
          this.__data = Object.create(props);
          this.__dataPending = Object.create(props);
          this.__dataOld = {};
        }

        /**
         * Overrides `PropertyAccessors` implementation to avoid setting
         * `_setProperty`'s `shouldNotify: true`.
         *
         * @override
         * @param {Object} props Properties to initialize on the instance
         * @return {void}
         */
        _initializeInstanceProperties(props) {
          let readOnly = this[TYPES.READ_ONLY];
          for (let prop in props) {
            if (!readOnly || !readOnly[prop]) {
              this.__dataPending = this.__dataPending || {};
              this.__dataOld = this.__dataOld || {};
              this.__data[prop] = this.__dataPending[prop] = props[prop];
            }
          }
        }

        // Prototype setup ----------------------------------------

        /**
         * Equivalent to static `addPropertyEffect` API but can be called on
         * an instance to add effects at runtime.  See that method for
         * full API docs.
         *
         * @override
         * @param {string} property Property that should trigger the effect
         * @param {string} type Effect type, from this.PROPERTY_EFFECT_TYPES
         * @param {Object=} effect Effect metadata object
         * @return {void}
         * @protected
         */
        _addPropertyEffect(property, type, effect) {
          this._createPropertyAccessor(property, type == TYPES.READ_ONLY);
          // effects are accumulated into arrays per property based on type
          let effects = ensureOwnEffectMap(this, type, true)[property];
          if (!effects) {
            effects = this[type][property] = [];
          }
          effects.push(effect);
        }

        /**
         * Removes the given property effect.
         *
         * @override
         * @param {string} property Property the effect was associated with
         * @param {string} type Effect type, from this.PROPERTY_EFFECT_TYPES
         * @param {Object=} effect Effect metadata object to remove
         * @return {void}
         */
        _removePropertyEffect(property, type, effect) {
          let effects = ensureOwnEffectMap(this, type, true)[property];
          let idx = effects.indexOf(effect);
          if (idx >= 0) {
            effects.splice(idx, 1);
          }
        }

        /**
         * Returns whether the current prototype/instance has a property effect
         * of a certain type.
         *
         * @override
         * @param {string} property Property name
         * @param {string=} type Effect type, from this.PROPERTY_EFFECT_TYPES
         * @return {boolean} True if the prototype/instance has an effect of this
         *     type
         * @protected
         */
        _hasPropertyEffect(property, type) {
          let effects = this[type];
          return Boolean(effects && effects[property]);
        }

        /**
         * Returns whether the current prototype/instance has a "read only"
         * accessor for the given property.
         *
         * @override
         * @param {string} property Property name
         * @return {boolean} True if the prototype/instance has an effect of this
         *     type
         * @protected
         */
        _hasReadOnlyEffect(property) {
          return this._hasPropertyEffect(property, TYPES.READ_ONLY);
        }

        /**
         * Returns whether the current prototype/instance has a "notify"
         * property effect for the given property.
         *
         * @override
         * @param {string} property Property name
         * @return {boolean} True if the prototype/instance has an effect of this
         *     type
         * @protected
         */
        _hasNotifyEffect(property) {
          return this._hasPropertyEffect(property, TYPES.NOTIFY);
        }

        /**
         * Returns whether the current prototype/instance has a "reflect to
         * attribute" property effect for the given property.
         *
         * @override
         * @param {string} property Property name
         * @return {boolean} True if the prototype/instance has an effect of this
         *     type
         * @protected
         */
        _hasReflectEffect(property) {
          return this._hasPropertyEffect(property, TYPES.REFLECT);
        }

        /**
         * Returns whether the current prototype/instance has a "computed"
         * property effect for the given property.
         *
         * @override
         * @param {string} property Property name
         * @return {boolean} True if the prototype/instance has an effect of this
         *     type
         * @protected
         */
        _hasComputedEffect(property) {
          return this._hasPropertyEffect(property, TYPES.COMPUTE);
        }

        // Runtime ----------------------------------------

        /**
         * Sets a pending property or path.  If the root property of the path in
         * question had no accessor, the path is set, otherwise it is enqueued
         * via `_setPendingProperty`.
         *
         * This function isolates relatively expensive functionality necessary
         * for the public API (`set`, `setProperties`, `notifyPath`, and property
         * change listeners via {{...}} bindings), such that it is only done
         * when paths enter the system, and not at every propagation step.  It
         * also sets a `__dataHasPaths` flag on the instance which is used to
         * fast-path slower path-matching code in the property effects host paths.
         *
         * `path` can be a path string or array of path parts as accepted by the
         * public API.
         *
         * @override
         * @param {string | !Array<number|string>} path Path to set
         * @param {*} value Value to set
         * @param {boolean=} shouldNotify Set to true if this change should
         *  cause a property notification event dispatch
         * @param {boolean=} isPathNotification If the path being set is a path
         *   notification of an already changed value, as opposed to a request
         *   to set and notify the change.  In the latter `false` case, a dirty
         *   check is performed and then the value is set to the path before
         *   enqueuing the pending property change.
         * @return {boolean} Returns true if the property/path was enqueued in
         *   the pending changes bag.
         * @protected
         */
        _setPendingPropertyOrPath(path, value, shouldNotify, isPathNotification) {
          if (isPathNotification ||
              root(Array.isArray(path) ? path[0] : path) !== path) {
            // Dirty check changes being set to a path against the actual object,
            // since this is the entry point for paths into the system; from here
            // the only dirty checks are against the `__dataTemp` cache to prevent
            // duplicate work in the same turn only. Note, if this was a notification
            // of a change already set to a path (isPathNotification: true),
            // we always let the change through and skip the `set` since it was
            // already dirty checked at the point of entry and the underlying
            // object has already been updated
            if (!isPathNotification) {
              let old = get(this, path);
              path = /** @type {string} */ (set(this, path, value));
              // Use property-accessor's simpler dirty check
              if (!path || !super._shouldPropertyChange(path, value, old)) {
                return false;
              }
            }
            this.__dataHasPaths = true;
            if (this._setPendingProperty(/**@type{string}*/(path), value, shouldNotify)) {
              computeLinkedPaths(this, /**@type{string}*/ (path), value);
              return true;
            }
          } else {
            if (this.__dataHasAccessor && this.__dataHasAccessor[path]) {
              return this._setPendingProperty(/**@type{string}*/(path), value, shouldNotify);
            } else {
              this[path] = value;
            }
          }
          return false;
        }

        /**
         * Applies a value to a non-Polymer element/node's property.
         *
         * The implementation makes a best-effort at binding interop:
         * Some native element properties have side-effects when
         * re-setting the same value (e.g. setting `<input>.value` resets the
         * cursor position), so we do a dirty-check before setting the value.
         * However, for better interop with non-Polymer custom elements that
         * accept objects, we explicitly re-set object changes coming from the
         * Polymer world (which may include deep object changes without the
         * top reference changing), erring on the side of providing more
         * information.
         *
         * Users may override this method to provide alternate approaches.
         *
         * @override
         * @param {!Node} node The node to set a property on
         * @param {string} prop The property to set
         * @param {*} value The value to set
         * @return {void}
         * @protected
         */
        _setUnmanagedPropertyToNode(node, prop, value) {
          // It is a judgment call that resetting primitives is
          // "bad" and resettings objects is also "good"; alternatively we could
          // implement a whitelist of tag & property values that should never
          // be reset (e.g. <input>.value && <select>.value)
          if (value !== node[prop] || typeof value == 'object') {
            // Note, className needs style scoping so this needs wrapping.
            if (prop === 'className') {
              node = /** @type {!Node} */(wrap(node));
            }
            node[prop] = value;
          }
        }

        /**
         * Overrides the `PropertiesChanged` implementation to introduce special
         * dirty check logic depending on the property & value being set:
         *
         * 1. Any value set to a path (e.g. 'obj.prop': 42 or 'obj.prop': {...})
         *    Stored in `__dataTemp`, dirty checked against `__dataTemp`
         * 2. Object set to simple property (e.g. 'prop': {...})
         *    Stored in `__dataTemp` and `__data`, dirty checked against
         *    `__dataTemp` by default implementation of `_shouldPropertyChange`
         * 3. Primitive value set to simple property (e.g. 'prop': 42)
         *    Stored in `__data`, dirty checked against `__data`
         *
         * The dirty-check is important to prevent cycles due to two-way
         * notification, but paths and objects are only dirty checked against any
         * previous value set during this turn via a "temporary cache" that is
         * cleared when the last `_propertiesChanged` exits. This is so:
         * a. any cached array paths (e.g. 'array.3.prop') may be invalidated
         *    due to array mutations like shift/unshift/splice; this is fine
         *    since path changes are dirty-checked at user entry points like `set`
         * b. dirty-checking for objects only lasts one turn to allow the user
         *    to mutate the object in-place and re-set it with the same identity
         *    and have all sub-properties re-propagated in a subsequent turn.
         *
         * The temp cache is not necessarily sufficient to prevent invalid array
         * paths, since a splice can happen during the same turn (with pathological
         * user code); we could introduce a "fixup" for temporarily cached array
         * paths if needed: https://github.com/Polymer/polymer/issues/4227
         *
         * @override
         * @param {string} property Name of the property
         * @param {*} value Value to set
         * @param {boolean=} shouldNotify True if property should fire notification
         *   event (applies only for `notify: true` properties)
         * @return {boolean} Returns true if the property changed
         */
        _setPendingProperty(property, value, shouldNotify) {
          let propIsPath = this.__dataHasPaths && isPath(property);
          let prevProps = propIsPath ? this.__dataTemp : this.__data;
          if (this._shouldPropertyChange(property, value, prevProps[property])) {
            if (!this.__dataPending) {
              this.__dataPending = {};
              this.__dataOld = {};
            }
            // Ensure old is captured from the last turn
            if (!(property in this.__dataOld)) {
              this.__dataOld[property] = this.__data[property];
            }
            // Paths are stored in temporary cache (cleared at end of turn),
            // which is used for dirty-checking, all others stored in __data
            if (propIsPath) {
              this.__dataTemp[property] = value;
            } else {
              this.__data[property] = value;
            }
            // All changes go into pending property bag, passed to _propertiesChanged
            this.__dataPending[property] = value;
            // Track properties that should notify separately
            if (propIsPath || (this[TYPES.NOTIFY] && this[TYPES.NOTIFY][property])) {
              this.__dataToNotify = this.__dataToNotify || {};
              this.__dataToNotify[property] = shouldNotify;
            }
            return true;
          }
          return false;
        }

        /**
         * Overrides base implementation to ensure all accessors set `shouldNotify`
         * to true, for per-property notification tracking.
         *
         * @override
         * @param {string} property Name of the property
         * @param {*} value Value to set
         * @return {void}
         */
        _setProperty(property, value) {
          if (this._setPendingProperty(property, value, true)) {
            this._invalidateProperties();
          }
        }

        /**
         * Overrides `PropertyAccessor`'s default async queuing of
         * `_propertiesChanged`: if `__dataReady` is false (has not yet been
         * manually flushed), the function no-ops; otherwise flushes
         * `_propertiesChanged` synchronously.
         *
         * @override
         * @return {void}
         */
        _invalidateProperties() {
          if (this.__dataReady) {
            this._flushProperties();
          }
        }

        /**
         * Enqueues the given client on a list of pending clients, whose
         * pending property changes can later be flushed via a call to
         * `_flushClients`.
         *
         * @override
         * @param {Object} client PropertyEffects client to enqueue
         * @return {void}
         * @protected
         */
        _enqueueClient(client) {
          this.__dataPendingClients = this.__dataPendingClients || [];
          if (client !== this) {
            this.__dataPendingClients.push(client);
          }
        }

        /**
         * Flushes any clients previously enqueued via `_enqueueClient`, causing
         * their `_flushProperties` method to run.
         *
         * @override
         * @return {void}
         * @protected
         */
        _flushClients() {
          if (!this.__dataClientsReady) {
            this.__dataClientsReady = true;
            this._readyClients();
            // Override point where accessors are turned on; importantly,
            // this is after clients have fully readied, providing a guarantee
            // that any property effects occur only after all clients are ready.
            this.__dataReady = true;
          } else {
            this.__enableOrFlushClients();
          }
        }

        // NOTE: We ensure clients either enable or flush as appropriate. This
        // handles two corner cases:
        // (1) clients flush properly when connected/enabled before the host
        // enables; e.g.
        //   (a) Templatize stamps with no properties and does not flush and
        //   (b) the instance is inserted into dom and
        //   (c) then the instance flushes.
        // (2) clients enable properly when not connected/enabled when the host
        // flushes; e.g.
        //   (a) a template is runtime stamped and not yet connected/enabled
        //   (b) a host sets a property, causing stamped dom to flush
        //   (c) the stamped dom enables.
        __enableOrFlushClients() {
          let clients = this.__dataPendingClients;
          if (clients) {
            this.__dataPendingClients = null;
            for (let i=0; i < clients.length; i++) {
              let client = clients[i];
              if (!client.__dataEnabled) {
                client._enableProperties();
              } else if (client.__dataPending) {
                client._flushProperties();
              }
            }
          }
        }

        /**
         * Perform any initial setup on client dom. Called before the first
         * `_flushProperties` call on client dom and before any element
         * observers are called.
         *
         * @override
         * @return {void}
         * @protected
         */
        _readyClients() {
          this.__enableOrFlushClients();
        }

        /**
         * Sets a bag of property changes to this instance, and
         * synchronously processes all effects of the properties as a batch.
         *
         * Property names must be simple properties, not paths.  Batched
         * path propagation is not supported.
         *
         * @override
         * @param {Object} props Bag of one or more key-value pairs whose key is
         *   a property and value is the new value to set for that property.
         * @param {boolean=} setReadOnly When true, any private values set in
         *   `props` will be set. By default, `setProperties` will not set
         *   `readOnly: true` root properties.
         * @return {void}
         * @public
         */
        setProperties(props, setReadOnly) {
          for (let path in props) {
            if (setReadOnly || !this[TYPES.READ_ONLY] || !this[TYPES.READ_ONLY][path]) {
              //TODO(kschaaf): explicitly disallow paths in setProperty?
              // wildcard observers currently only pass the first changed path
              // in the `info` object, and you could do some odd things batching
              // paths, e.g. {'foo.bar': {...}, 'foo': null}
              this._setPendingPropertyOrPath(path, props[path], true);
            }
          }
          this._invalidateProperties();
        }

        /**
         * Overrides `PropertyAccessors` so that property accessor
         * side effects are not enabled until after client dom is fully ready.
         * Also calls `_flushClients` callback to ensure client dom is enabled
         * that was not enabled as a result of flushing properties.
         *
         * @override
         * @return {void}
         */
        ready() {
          // It is important that `super.ready()` is not called here as it
          // immediately turns on accessors. Instead, we wait until `readyClients`
          // to enable accessors to provide a guarantee that clients are ready
          // before processing any accessors side effects.
          this._flushProperties();
          // If no data was pending, `_flushProperties` will not `flushClients`
          // so ensure this is done.
          if (!this.__dataClientsReady) {
            this._flushClients();
          }
          // Before ready, client notifications do not trigger _flushProperties.
          // Therefore a flush is necessary here if data has been set.
          if (this.__dataPending) {
            this._flushProperties();
          }
        }

        /**
         * Implements `PropertyAccessors`'s properties changed callback.
         *
         * Runs each class of effects for the batch of changed properties in
         * a specific order (compute, propagate, reflect, observe, notify).
         *
         * @override
         * @param {!Object} currentProps Bag of all current accessor values
         * @param {?Object} changedProps Bag of properties changed since the last
         *   call to `_propertiesChanged`
         * @param {?Object} oldProps Bag of previous values for each property
         *   in `changedProps`
         * @return {void}
         */
        _propertiesChanged(currentProps, changedProps, oldProps) {
          // ----------------------------
          // let c = Object.getOwnPropertyNames(changedProps || {});
          // window.debug && console.group(this.localName + '#' + this.id + ': ' + c);
          // if (window.debug) { debugger; }
          // ----------------------------
          let hasPaths = this.__dataHasPaths;
          this.__dataHasPaths = false;
          let notifyProps;
          // Compute properties
          runComputedEffects(this, changedProps, oldProps, hasPaths);
          // Clear notify properties prior to possible reentry (propagate, observe),
          // but after computing effects have a chance to add to them
          notifyProps = this.__dataToNotify;
          this.__dataToNotify = null;
          // Propagate properties to clients
          this._propagatePropertyChanges(changedProps, oldProps, hasPaths);
          // Flush clients
          this._flushClients();
          // Reflect properties
          runEffects(this, this[TYPES.REFLECT], changedProps, oldProps, hasPaths);
          // Observe properties
          runEffects(this, this[TYPES.OBSERVE], changedProps, oldProps, hasPaths);
          // Notify properties to host
          if (notifyProps) {
            runNotifyEffects(this, notifyProps, changedProps, oldProps, hasPaths);
          }
          // Clear temporary cache at end of turn
          if (this.__dataCounter == 1) {
            this.__dataTemp = {};
          }
          // ----------------------------
          // window.debug && console.groupEnd(this.localName + '#' + this.id + ': ' + c);
          // ----------------------------
        }

        /**
         * Called to propagate any property changes to stamped template nodes
         * managed by this element.
         *
         * @override
         * @param {Object} changedProps Bag of changed properties
         * @param {Object} oldProps Bag of previous values for changed properties
         * @param {boolean} hasPaths True with `props` contains one or more paths
         * @return {void}
         * @protected
         */
        _propagatePropertyChanges(changedProps, oldProps, hasPaths) {
          if (this[TYPES.PROPAGATE]) {
            runEffects(this, this[TYPES.PROPAGATE], changedProps, oldProps, hasPaths);
          }
          if (this.__templateInfo) {
            this._runEffectsForTemplate(this.__templateInfo, changedProps, oldProps, hasPaths);
          }
        }

        _runEffectsForTemplate(templateInfo, changedProps, oldProps, hasPaths) {
          const baseRunEffects = (changedProps, hasPaths) => {
            runEffects(this, templateInfo.propertyEffects, changedProps, oldProps,
              hasPaths, templateInfo.nodeList);
            for (let info=templateInfo.firstChild; info; info=info.nextSibling) {
              this._runEffectsForTemplate(info, changedProps, oldProps, hasPaths);
            }
          };
          if (templateInfo.runEffects) {
            templateInfo.runEffects(baseRunEffects, changedProps, hasPaths);
          } else {
            baseRunEffects(changedProps, hasPaths);
          }
        }

        /**
         * Aliases one data path as another, such that path notifications from one
         * are routed to the other.
         *
         * @override
         * @param {string | !Array<string|number>} to Target path to link.
         * @param {string | !Array<string|number>} from Source path to link.
         * @return {void}
         * @public
         */
        linkPaths(to, from) {
          to = normalize(to);
          from = normalize(from);
          this.__dataLinkedPaths = this.__dataLinkedPaths || {};
          this.__dataLinkedPaths[to] = from;
        }

        /**
         * Removes a data path alias previously established with `_linkPaths`.
         *
         * Note, the path to unlink should be the target (`to`) used when
         * linking the paths.
         *
         * @override
         * @param {string | !Array<string|number>} path Target path to unlink.
         * @return {void}
         * @public
         */
        unlinkPaths(path) {
          path = normalize(path);
          if (this.__dataLinkedPaths) {
            delete this.__dataLinkedPaths[path];
          }
        }

        /**
         * Notify that an array has changed.
         *
         * Example:
         *
         *     this.items = [ {name: 'Jim'}, {name: 'Todd'}, {name: 'Bill'} ];
         *     ...
         *     this.items.splice(1, 1, {name: 'Sam'});
         *     this.items.push({name: 'Bob'});
         *     this.notifySplices('items', [
         *       { index: 1, removed: [{name: 'Todd'}], addedCount: 1,
         *         object: this.items, type: 'splice' },
         *       { index: 3, removed: [], addedCount: 1,
         *         object: this.items, type: 'splice'}
         *     ]);
         *
         * @param {string} path Path that should be notified.
         * @param {Array} splices Array of splice records indicating ordered
         *   changes that occurred to the array. Each record should have the
         *   following fields:
         *    * index: index at which the change occurred
         *    * removed: array of items that were removed from this index
         *    * addedCount: number of new items added at this index
         *    * object: a reference to the array in question
         *    * type: the string literal 'splice'
         *
         *   Note that splice records _must_ be normalized such that they are
         *   reported in index order (raw results from `Object.observe` are not
         *   ordered and must be normalized/merged before notifying).
         *
         * @override
         * @return {void}
         * @public
         */
        notifySplices(path, splices) {
          let info = {path: ''};
          let array = /** @type {Array} */(get(this, path, info));
          notifySplices(this, array, info.path, splices);
        }

        /**
         * Convenience method for reading a value from a path.
         *
         * Note, if any part in the path is undefined, this method returns
         * `undefined` (this method does not throw when dereferencing undefined
         * paths).
         *
         * @override
         * @param {(string|!Array<(string|number)>)} path Path to the value
         *   to read.  The path may be specified as a string (e.g. `foo.bar.baz`)
         *   or an array of path parts (e.g. `['foo.bar', 'baz']`).  Note that
         *   bracketed expressions are not supported; string-based path parts
         *   *must* be separated by dots.  Note that when dereferencing array
         *   indices, the index may be used as a dotted part directly
         *   (e.g. `users.12.name` or `['users', 12, 'name']`).
         * @param {Object=} root Root object from which the path is evaluated.
         * @return {*} Value at the path, or `undefined` if any part of the path
         *   is undefined.
         * @public
         */
        get(path, root) {
          return get(root || this, path);
        }

        /**
         * Convenience method for setting a value to a path and notifying any
         * elements bound to the same path.
         *
         * Note, if any part in the path except for the last is undefined,
         * this method does nothing (this method does not throw when
         * dereferencing undefined paths).
         *
         * @override
         * @param {(string|!Array<(string|number)>)} path Path to the value
         *   to write.  The path may be specified as a string (e.g. `'foo.bar.baz'`)
         *   or an array of path parts (e.g. `['foo.bar', 'baz']`).  Note that
         *   bracketed expressions are not supported; string-based path parts
         *   *must* be separated by dots.  Note that when dereferencing array
         *   indices, the index may be used as a dotted part directly
         *   (e.g. `'users.12.name'` or `['users', 12, 'name']`).
         * @param {*} value Value to set at the specified path.
         * @param {Object=} root Root object from which the path is evaluated.
         *   When specified, no notification will occur.
         * @return {void}
         * @public
         */
        set(path, value, root) {
          if (root) {
            set(root, path, value);
          } else {
            if (!this[TYPES.READ_ONLY] || !this[TYPES.READ_ONLY][/** @type {string} */(path)]) {
              if (this._setPendingPropertyOrPath(path, value, true)) {
                this._invalidateProperties();
              }
            }
          }
        }

        /**
         * Adds items onto the end of the array at the path specified.
         *
         * The arguments after `path` and return value match that of
         * `Array.prototype.push`.
         *
         * This method notifies other paths to the same array that a
         * splice occurred to the array.
         *
         * @override
         * @param {string | !Array<string|number>} path Path to array.
         * @param {...*} items Items to push onto array
         * @return {number} New length of the array.
         * @public
         */
        push(path, ...items) {
          let info = {path: ''};
          let array = /** @type {Array}*/(get(this, path, info));
          let len = array.length;
          let ret = array.push(...items);
          if (items.length) {
            notifySplice(this, array, info.path, len, items.length, []);
          }
          return ret;
        }

        /**
         * Removes an item from the end of array at the path specified.
         *
         * The arguments after `path` and return value match that of
         * `Array.prototype.pop`.
         *
         * This method notifies other paths to the same array that a
         * splice occurred to the array.
         *
         * @override
         * @param {string | !Array<string|number>} path Path to array.
         * @return {*} Item that was removed.
         * @public
         */
        pop(path) {
          let info = {path: ''};
          let array = /** @type {Array} */(get(this, path, info));
          let hadLength = Boolean(array.length);
          let ret = array.pop();
          if (hadLength) {
            notifySplice(this, array, info.path, array.length, 0, [ret]);
          }
          return ret;
        }

        /**
         * Starting from the start index specified, removes 0 or more items
         * from the array and inserts 0 or more new items in their place.
         *
         * The arguments after `path` and return value match that of
         * `Array.prototype.splice`.
         *
         * This method notifies other paths to the same array that a
         * splice occurred to the array.
         *
         * @override
         * @param {string | !Array<string|number>} path Path to array.
         * @param {number} start Index from which to start removing/inserting.
         * @param {number=} deleteCount Number of items to remove.
         * @param {...*} items Items to insert into array.
         * @return {!Array} Array of removed items.
         * @public
         */
        splice(path, start, deleteCount, ...items) {
          let info = {path : ''};
          let array = /** @type {Array} */(get(this, path, info));
          // Normalize fancy native splice handling of crazy start values
          if (start < 0) {
            start = array.length - Math.floor(-start);
          } else if (start) {
            start = Math.floor(start);
          }
          // array.splice does different things based on the number of arguments
          // you pass in. Therefore, array.splice(0) and array.splice(0, undefined)
          // do different things. In the former, the whole array is cleared. In the
          // latter, no items are removed.
          // This means that we need to detect whether 1. one of the arguments
          // is actually passed in and then 2. determine how many arguments
          // we should pass on to the native array.splice
          //
          let ret;
          // Omit any additional arguments if they were not passed in
          if (arguments.length === 2) {
            ret = array.splice(start);
          // Either start was undefined and the others were defined, but in this
          // case we can safely pass on all arguments
          //
          // Note: this includes the case where none of the arguments were passed in,
          // e.g. this.splice('array'). However, if both start and deleteCount
          // are undefined, array.splice will not modify the array (as expected)
          } else {
            ret = array.splice(start, deleteCount, ...items);
          }
          // At the end, check whether any items were passed in (e.g. insertions)
          // or if the return array contains items (e.g. deletions).
          // Only notify if items were added or deleted.
          if (items.length || ret.length) {
            notifySplice(this, array, info.path, start, items.length, ret);
          }
          return ret;
        }

        /**
         * Removes an item from the beginning of array at the path specified.
         *
         * The arguments after `path` and return value match that of
         * `Array.prototype.pop`.
         *
         * This method notifies other paths to the same array that a
         * splice occurred to the array.
         *
         * @override
         * @param {string | !Array<string|number>} path Path to array.
         * @return {*} Item that was removed.
         * @public
         */
        shift(path) {
          let info = {path: ''};
          let array = /** @type {Array} */(get(this, path, info));
          let hadLength = Boolean(array.length);
          let ret = array.shift();
          if (hadLength) {
            notifySplice(this, array, info.path, 0, 0, [ret]);
          }
          return ret;
        }

        /**
         * Adds items onto the beginning of the array at the path specified.
         *
         * The arguments after `path` and return value match that of
         * `Array.prototype.push`.
         *
         * This method notifies other paths to the same array that a
         * splice occurred to the array.
         *
         * @override
         * @param {string | !Array<string|number>} path Path to array.
         * @param {...*} items Items to insert info array
         * @return {number} New length of the array.
         * @public
         */
        unshift(path, ...items) {
          let info = {path: ''};
          let array = /** @type {Array} */(get(this, path, info));
          let ret = array.unshift(...items);
          if (items.length) {
            notifySplice(this, array, info.path, 0, items.length, []);
          }
          return ret;
        }

        /**
         * Notify that a path has changed.
         *
         * Example:
         *
         *     this.item.user.name = 'Bob';
         *     this.notifyPath('item.user.name');
         *
         * @override
         * @param {string} path Path that should be notified.
         * @param {*=} value Value at the path (optional).
         * @return {void}
         * @public
         */
        notifyPath(path, value) {
          /** @type {string} */
          let propPath;
          if (arguments.length == 1) {
            // Get value if not supplied
            let info = {path: ''};
            value = get(this, path, info);
            propPath = info.path;
          } else if (Array.isArray(path)) {
            // Normalize path if needed
            propPath = normalize(path);
          } else {
            propPath = /** @type{string} */(path);
          }
          if (this._setPendingPropertyOrPath(propPath, value, true, true)) {
            this._invalidateProperties();
          }
        }

        /**
         * Equivalent to static `createReadOnlyProperty` API but can be called on
         * an instance to add effects at runtime.  See that method for
         * full API docs.
         *
         * @override
         * @param {string} property Property name
         * @param {boolean=} protectedSetter Creates a custom protected setter
         *   when `true`.
         * @return {void}
         * @protected
         */
        _createReadOnlyProperty(property, protectedSetter) {
          this._addPropertyEffect(property, TYPES.READ_ONLY);
          if (protectedSetter) {
            this['_set' + upper(property)] = /** @this {PropertyEffects} */function(value) {
              this._setProperty(property, value);
            };
          }
        }

        /**
         * Equivalent to static `createPropertyObserver` API but can be called on
         * an instance to add effects at runtime.  See that method for
         * full API docs.
         *
         * @override
         * @param {string} property Property name
         * @param {string|function(*,*)} method Function or name of observer method
         *     to call
         * @param {boolean=} dynamicFn Whether the method name should be included as
         *   a dependency to the effect.
         * @return {void}
         * @protected
         */
        _createPropertyObserver(property, method, dynamicFn) {
          let info = { property, method, dynamicFn: Boolean(dynamicFn) };
          this._addPropertyEffect(property, TYPES.OBSERVE, {
            fn: runObserverEffect, info, trigger: {name: property}
          });
          if (dynamicFn) {
            this._addPropertyEffect(/** @type {string} */(method), TYPES.OBSERVE, {
              fn: runObserverEffect, info, trigger: {name: method}
            });
          }
        }

        /**
         * Equivalent to static `createMethodObserver` API but can be called on
         * an instance to add effects at runtime.  See that method for
         * full API docs.
         *
         * @override
         * @param {string} expression Method expression
         * @param {boolean|Object=} dynamicFn Boolean or object map indicating
         *   whether method names should be included as a dependency to the effect.
         * @return {void}
         * @protected
         */
        _createMethodObserver(expression, dynamicFn) {
          let sig = parseMethod(expression);
          if (!sig) {
            throw new Error("Malformed observer expression '" + expression + "'");
          }
          createMethodEffect(this, sig, TYPES.OBSERVE, runMethodEffect, null, dynamicFn);
        }

        /**
         * Equivalent to static `createNotifyingProperty` API but can be called on
         * an instance to add effects at runtime.  See that method for
         * full API docs.
         *
         * @override
         * @param {string} property Property name
         * @return {void}
         * @protected
         */
        _createNotifyingProperty(property) {
          this._addPropertyEffect(property, TYPES.NOTIFY, {
            fn: runNotifyEffect,
            info: {
              eventName: camelToDashCase(property) + '-changed',
              property: property
            }
          });
        }

        /**
         * Equivalent to static `createReflectedProperty` API but can be called on
         * an instance to add effects at runtime.  See that method for
         * full API docs.
         *
         * @override
         * @param {string} property Property name
         * @return {void}
         * @protected
         * @suppress {missingProperties} go/missingfnprops
         */
        _createReflectedProperty(property) {
          let attr = this.constructor.attributeNameForProperty(property);
          if (attr[0] === '-') {
            console.warn('Property ' + property + ' cannot be reflected to attribute ' +
              attr + ' because "-" is not a valid starting attribute name. Use a lowercase first letter for the property instead.');
          } else {
            this._addPropertyEffect(property, TYPES.REFLECT, {
              fn: runReflectEffect,
              info: {
                attrName: attr
              }
            });
          }
        }

        /**
         * Equivalent to static `createComputedProperty` API but can be called on
         * an instance to add effects at runtime.  See that method for
         * full API docs.
         *
         * @override
         * @param {string} property Name of computed property to set
         * @param {string} expression Method expression
         * @param {boolean|Object=} dynamicFn Boolean or object map indicating
         *   whether method names should be included as a dependency to the effect.
         * @return {void}
         * @protected
         */
        _createComputedProperty(property, expression, dynamicFn) {
          let sig = parseMethod(expression);
          if (!sig) {
            throw new Error("Malformed computed expression '" + expression + "'");
          }
          const info = createMethodEffect(this, sig, TYPES.COMPUTE, runComputedEffect, property, dynamicFn);
          // Effects are normally stored as map of dependency->effect, but for
          // ordered computation, we also need tree of computedProp->dependencies
          ensureOwnEffectMap(this, COMPUTE_INFO)[property] = info;
        }

        /**
         * Gather the argument values for a method specified in the provided array
         * of argument metadata.
         *
         * The `path` and `value` arguments are used to fill in wildcard descriptor
         * when the method is being called as a result of a path notification.
         *
         * @param {!Array<!MethodArg>} args Array of argument metadata
         * @param {string} path Property/path name that triggered the method effect
         * @param {Object} props Bag of current property changes
         * @return {!Array<*>} Array of argument values
         * @private
         */
        _marshalArgs(args, path, props) {
          const data = this.__data;
          const values = [];
          for (let i=0, l=args.length; i<l; i++) {
            let {name, structured, wildcard, value, literal} = args[i];
            if (!literal) {
              if (wildcard) {
                const matches = isDescendant(name, path);
                const pathValue = getArgValue(data, props, matches ? path : name);
                value = {
                  path: matches ? path : name,
                  value: pathValue,
                  base: matches ? get(data, name) : pathValue
                };
              } else {
                value = structured ? getArgValue(data, props, name) : data[name];
              }
            }
            // When the `legacyUndefined` flag is enabled, pass a no-op value
            // so that the observer, computed property, or compound binding is aborted.
            if (legacyUndefined && !this._overrideLegacyUndefined && value === undefined && args.length > 1) {
              return NOOP;
            }
            values[i] = value;
          }
          return values;
        }

        // -- static class methods ------------

        /**
         * Ensures an accessor exists for the specified property, and adds
         * to a list of "property effects" that will run when the accessor for
         * the specified property is set.  Effects are grouped by "type", which
         * roughly corresponds to a phase in effect processing.  The effect
         * metadata should be in the following form:
         *
         *     {
         *       fn: effectFunction, // Reference to function to call to perform effect
         *       info: { ... }       // Effect metadata passed to function
         *       trigger: {          // Optional triggering metadata; if not provided
         *         name: string      // the property is treated as a wildcard
         *         structured: boolean
         *         wildcard: boolean
         *       }
         *     }
         *
         * Effects are called from `_propertiesChanged` in the following order by
         * type:
         *
         * 1. COMPUTE
         * 2. PROPAGATE
         * 3. REFLECT
         * 4. OBSERVE
         * 5. NOTIFY
         *
         * Effect functions are called with the following signature:
         *
         *     effectFunction(inst, path, props, oldProps, info, hasPaths)
         *
         * @param {string} property Property that should trigger the effect
         * @param {string} type Effect type, from this.PROPERTY_EFFECT_TYPES
         * @param {Object=} effect Effect metadata object
         * @return {void}
         * @protected
         * @nocollapse
         */
        static addPropertyEffect(property, type, effect) {
          this.prototype._addPropertyEffect(property, type, effect);
        }

        /**
         * Creates a single-property observer for the given property.
         *
         * @param {string} property Property name
         * @param {string|function(*,*)} method Function or name of observer method to call
         * @param {boolean=} dynamicFn Whether the method name should be included as
         *   a dependency to the effect.
         * @return {void}
         * @protected
         * @nocollapse
         */
        static createPropertyObserver(property, method, dynamicFn) {
          this.prototype._createPropertyObserver(property, method, dynamicFn);
        }

        /**
         * Creates a multi-property "method observer" based on the provided
         * expression, which should be a string in the form of a normal JavaScript
         * function signature: `'methodName(arg1, [..., argn])'`.  Each argument
         * should correspond to a property or path in the context of this
         * prototype (or instance), or may be a literal string or number.
         *
         * @param {string} expression Method expression
         * @param {boolean|Object=} dynamicFn Boolean or object map indicating
         * @return {void}
         *   whether method names should be included as a dependency to the effect.
         * @protected
         * @nocollapse
         */
        static createMethodObserver(expression, dynamicFn) {
          this.prototype._createMethodObserver(expression, dynamicFn);
        }

        /**
         * Causes the setter for the given property to dispatch `<property>-changed`
         * events to notify of changes to the property.
         *
         * @param {string} property Property name
         * @return {void}
         * @protected
         * @nocollapse
         */
        static createNotifyingProperty(property) {
          this.prototype._createNotifyingProperty(property);
        }

        /**
         * Creates a read-only accessor for the given property.
         *
         * To set the property, use the protected `_setProperty` API.
         * To create a custom protected setter (e.g. `_setMyProp()` for
         * property `myProp`), pass `true` for `protectedSetter`.
         *
         * Note, if the property will have other property effects, this method
         * should be called first, before adding other effects.
         *
         * @param {string} property Property name
         * @param {boolean=} protectedSetter Creates a custom protected setter
         *   when `true`.
         * @return {void}
         * @protected
         * @nocollapse
         */
        static createReadOnlyProperty(property, protectedSetter) {
          this.prototype._createReadOnlyProperty(property, protectedSetter);
        }

        /**
         * Causes the setter for the given property to reflect the property value
         * to a (dash-cased) attribute of the same name.
         *
         * @param {string} property Property name
         * @return {void}
         * @protected
         * @nocollapse
         */
        static createReflectedProperty(property) {
          this.prototype._createReflectedProperty(property);
        }

        /**
         * Creates a computed property whose value is set to the result of the
         * method described by the given `expression` each time one or more
         * arguments to the method changes.  The expression should be a string
         * in the form of a normal JavaScript function signature:
         * `'methodName(arg1, [..., argn])'`
         *
         * @param {string} property Name of computed property to set
         * @param {string} expression Method expression
         * @param {boolean|Object=} dynamicFn Boolean or object map indicating whether
         *   method names should be included as a dependency to the effect.
         * @return {void}
         * @protected
         * @nocollapse
         */
        static createComputedProperty(property, expression, dynamicFn) {
          this.prototype._createComputedProperty(property, expression, dynamicFn);
        }

        /**
         * Parses the provided template to ensure binding effects are created
         * for them, and then ensures property accessors are created for any
         * dependent properties in the template.  Binding effects for bound
         * templates are stored in a linked list on the instance so that
         * templates can be efficiently stamped and unstamped.
         *
         * @param {!HTMLTemplateElement} template Template containing binding
         *   bindings
         * @return {!TemplateInfo} Template metadata object
         * @protected
         * @nocollapse
         */
        static bindTemplate(template) {
          return this.prototype._bindTemplate(template);
        }

        // -- binding ----------------------------------------------

        /*
         * Overview of binding flow:
         *
         * During finalization (`instanceBinding==false`, `wasPreBound==false`):
         *  `_bindTemplate(t, false)` called directly during finalization - parses
         *  the template (for the first time), and then assigns that _prototypical_
         *  template info to `__preboundTemplateInfo` _on the prototype_; note in
         *  this case `wasPreBound` is false; this is the first time we're binding
         *  it, thus we create accessors.
         *
         * During first stamping (`instanceBinding==true`, `wasPreBound==true`):
         *   `_stampTemplate` calls `_bindTemplate(t, true)`: the `templateInfo`
         *   returned matches the prebound one, and so this is `wasPreBound == true`
         *   state; thus we _skip_ creating accessors, but _do_ create an instance
         *   of the template info to serve as the start of our linked list (needs to
         *   be an instance, not the prototypical one, so that we can add `nodeList`
         *   to it to contain the `nodeInfo`-ordered list of instance nodes for
         *   bindings, and so we can chain runtime-stamped template infos off of
         *   it). At this point, the call to `_stampTemplate` calls
         *   `applyTemplateInfo` for each nested `<template>` found during parsing
         *   to hand prototypical `_templateInfo` to them; we also pass the _parent_
         *   `templateInfo` to the `<template>` so that we have the instance-time
         *   parent to link the `templateInfo` under in the case it was
         *   runtime-stamped.
         *
         * During subsequent runtime stamping (`instanceBinding==true`,
         *   `wasPreBound==false`): `_stampTemplate` calls `_bindTemplate(t, true)`
         *   - here `templateInfo` is guaranteed to _not_ match the prebound one,
         *   because it was either a different template altogether, or even if it
         *   was the same template, the step above created a instance of the info;
         *   in this case `wasPreBound == false`, so we _do_ create accessors, _and_
         *   link a instance into the linked list.
         */

        /**
         * Equivalent to static `bindTemplate` API but can be called on an instance
         * to add effects at runtime.  See that method for full API docs.
         *
         * This method may be called on the prototype (for prototypical template
         * binding, to avoid creating accessors every instance) once per prototype,
         * and will be called with `runtimeBinding: true` by `_stampTemplate` to
         * create and link an instance of the template metadata associated with a
         * particular stamping.
         *
         * @override
         * @param {!HTMLTemplateElement} template Template containing binding
         * bindings
         * @param {boolean=} instanceBinding When false (default), performs
         * "prototypical" binding of the template and overwrites any previously
         * bound template for the class. When true (as passed from
         * `_stampTemplate`), the template info is instanced and linked into the
         * list of bound templates.
         * @return {!TemplateInfo} Template metadata object; for `runtimeBinding`,
         * this is an instance of the prototypical template info
         * @protected
         * @suppress {missingProperties} go/missingfnprops
         */
        _bindTemplate(template, instanceBinding) {
          let templateInfo = this.constructor._parseTemplate(template);
          let wasPreBound = this.__preBoundTemplateInfo == templateInfo;
          // Optimization: since this is called twice for proto-bound templates,
          // don't attempt to recreate accessors if this template was pre-bound
          if (!wasPreBound) {
            for (let prop in templateInfo.propertyEffects) {
              this._createPropertyAccessor(prop);
            }
          }
          if (instanceBinding) {
            // For instance-time binding, create instance of template metadata
            // and link into tree of templates if necessary
            templateInfo = /** @type {!TemplateInfo} */(Object.create(templateInfo));
            templateInfo.wasPreBound = wasPreBound;
            if (!this.__templateInfo) {
              // Set the info to the root of the tree
              this.__templateInfo = templateInfo;
            } else {
              // Append this template info onto the end of its parent template's
              // list, which will determine the tree structure via which property
              // effects are run; if this template was not nested in another
              // template, use the root template (the first stamped one) as the
              // parent. Note, `parent` is the `templateInfo` instance for this
              // template's parent (containing) template, which was set up in
              // `applyTemplateInfo`.  While a given template's `parent` is set
              // apriori, it is only added to the parent's child list at the point
              // that it is being bound, since a template may or may not ever be
              // stamped, and may be stamped more than once (in which case instances
              // of the template info will be in the tree under its parent more than
              // once).
              const parent = template._parentTemplateInfo || this.__templateInfo;
              const previous = parent.lastChild;
              templateInfo.parent = parent;
              parent.lastChild = templateInfo;
              templateInfo.previousSibling = previous;
              if (previous) {
                previous.nextSibling = templateInfo;
              } else {
                parent.firstChild = templateInfo;
              }
            }
          } else {
            this.__preBoundTemplateInfo = templateInfo;
          }
          return templateInfo;
        }

        /**
         * Adds a property effect to the given template metadata, which is run
         * at the "propagate" stage of `_propertiesChanged` when the template
         * has been bound to the element via `_bindTemplate`.
         *
         * The `effect` object should match the format in `_addPropertyEffect`.
         *
         * @param {Object} templateInfo Template metadata to add effect to
         * @param {string} prop Property that should trigger the effect
         * @param {Object=} effect Effect metadata object
         * @return {void}
         * @protected
         * @nocollapse
         */
        static _addTemplatePropertyEffect(templateInfo, prop, effect) {
          let hostProps = templateInfo.hostProps = templateInfo.hostProps || {};
          hostProps[prop] = true;
          let effects = templateInfo.propertyEffects = templateInfo.propertyEffects || {};
          let propEffects = effects[prop] = effects[prop] || [];
          propEffects.push(effect);
        }

        /**
         * Stamps the provided template and performs instance-time setup for
         * Polymer template features, including data bindings, declarative event
         * listeners, and the `this.$` map of `id`'s to nodes.  A document fragment
         * is returned containing the stamped DOM, ready for insertion into the
         * DOM.
         *
         * This method may be called more than once; however note that due to
         * `shadycss` polyfill limitations, only styles from templates prepared
         * using `ShadyCSS.prepareTemplate` will be correctly polyfilled (scoped
         * to the shadow root and support CSS custom properties), and note that
         * `ShadyCSS.prepareTemplate` may only be called once per element. As such,
         * any styles required by in runtime-stamped templates must be included
         * in the main element template.
         *
         * @param {!HTMLTemplateElement} template Template to stamp
         * @param {TemplateInfo=} templateInfo Optional bound template info associated
         *   with the template to be stamped; if omitted the template will be
         *   automatically bound.
         * @return {!StampedTemplate} Cloned template content
         * @override
         * @protected
         */
        _stampTemplate(template, templateInfo) {
          templateInfo =  templateInfo || /** @type {!TemplateInfo} */(this._bindTemplate(template, true));
          // Ensures that created dom is `_enqueueClient`'d to this element so
          // that it can be flushed on next call to `_flushProperties`
          hostStack.push(this);
          let dom = super._stampTemplate(template, templateInfo);
          hostStack.pop();
          // Add template-instance-specific data to instanced templateInfo
          templateInfo.nodeList = dom.nodeList;
          // Capture child nodes to allow unstamping of non-prototypical templates
          if (!templateInfo.wasPreBound) {
            let nodes = templateInfo.childNodes = [];
            for (let n=dom.firstChild; n; n=n.nextSibling) {
              nodes.push(n);
            }
          }
          dom.templateInfo = templateInfo;
          // Setup compound storage, 2-way listeners, and dataHost for bindings
          setupBindings(this, templateInfo);
          // Flush properties into template nodes; the check on `__dataClientsReady`
          // ensures we don't needlessly run effects for an element's initial
          // prototypical template stamping since they will happen as a part of the
          // first call to `_propertiesChanged`. This flag is set to true
          // after running the initial propagate effects, and immediately before
          // flushing clients. Since downstream clients could cause stamping on
          // this host (e.g. a fastDomIf `dom-if` being forced to render
          // synchronously), this flag ensures effects for runtime-stamped templates
          // are run at this point during the initial element boot-up.
          if (this.__dataClientsReady) {
            this._runEffectsForTemplate(templateInfo, this.__data, null, false);
            this._flushClients();
          }
          return dom;
        }

        /**
         * Removes and unbinds the nodes previously contained in the provided
         * DocumentFragment returned from `_stampTemplate`.
         *
         * @override
         * @param {!StampedTemplate} dom DocumentFragment previously returned
         *   from `_stampTemplate` associated with the nodes to be removed
         * @return {void}
         * @protected
         */
        _removeBoundDom(dom) {
          // Unlink template info; Note that while the child is unlinked from its
          // parent list, a template's `parent` reference is never removed, since
          // this is is determined by the tree structure and applied at
          // `applyTemplateInfo` time.
          const templateInfo = dom.templateInfo;
          const {previousSibling, nextSibling, parent} = templateInfo;
          if (previousSibling) {
            previousSibling.nextSibling = nextSibling;
          } else if (parent) {
            parent.firstChild = nextSibling;
          }
          if (nextSibling) {
            nextSibling.previousSibling = previousSibling;
          } else if (parent) {
            parent.lastChild = previousSibling;
          }
          templateInfo.nextSibling = templateInfo.previousSibling = null;
          // Remove stamped nodes
          let nodes = templateInfo.childNodes;
          for (let i=0; i<nodes.length; i++) {
            let node = nodes[i];
            wrap(wrap(node).parentNode).removeChild(node);
          }
        }

        /**
         * Overrides default `TemplateStamp` implementation to add support for
         * parsing bindings from `TextNode`'s' `textContent`.  A `bindings`
         * array is added to `nodeInfo` and populated with binding metadata
         * with information capturing the binding target, and a `parts` array
         * with one or more metadata objects capturing the source(s) of the
         * binding.
         *
         * @param {Node} node Node to parse
         * @param {TemplateInfo} templateInfo Template metadata for current template
         * @param {NodeInfo} nodeInfo Node metadata for current template node
         * @return {boolean} `true` if the visited node added node-specific
         *   metadata to `nodeInfo`
         * @protected
         * @suppress {missingProperties} Interfaces in closure do not inherit statics, but classes do
         * @nocollapse
         */
        static _parseTemplateNode(node, templateInfo, nodeInfo) {
          // TODO(https://github.com/google/closure-compiler/issues/3240):
          //     Change back to just super.methodCall()
          let noted = propertyEffectsBase._parseTemplateNode.call(
            this, node, templateInfo, nodeInfo);
          if (node.nodeType === Node.TEXT_NODE) {
            let parts = this._parseBindings(node.textContent, templateInfo);
            if (parts) {
              // Initialize the textContent with any literal parts
              // NOTE: default to a space here so the textNode remains; some browsers
              // (IE) omit an empty textNode following cloneNode/importNode.
              node.textContent = literalFromParts(parts) || ' ';
              addBinding(this, templateInfo, nodeInfo, 'text', 'textContent', parts);
              noted = true;
            }
          }
          return noted;
        }

        /**
         * Overrides default `TemplateStamp` implementation to add support for
         * parsing bindings from attributes.  A `bindings`
         * array is added to `nodeInfo` and populated with binding metadata
         * with information capturing the binding target, and a `parts` array
         * with one or more metadata objects capturing the source(s) of the
         * binding.
         *
         * @param {Element} node Node to parse
         * @param {TemplateInfo} templateInfo Template metadata for current template
         * @param {NodeInfo} nodeInfo Node metadata for current template node
         * @param {string} name Attribute name
         * @param {string} value Attribute value
         * @return {boolean} `true` if the visited node added node-specific
         *   metadata to `nodeInfo`
         * @protected
         * @suppress {missingProperties} Interfaces in closure do not inherit statics, but classes do
         * @nocollapse
         */
        static _parseTemplateNodeAttribute(node, templateInfo, nodeInfo, name, value) {
          let parts = this._parseBindings(value, templateInfo);
          if (parts) {
            // Attribute or property
            let origName = name;
            let kind = 'property';
            // The only way we see a capital letter here is if the attr has
            // a capital letter in it per spec. In this case, to make sure
            // this binding works, we go ahead and make the binding to the attribute.
            if (capitalAttributeRegex.test(name)) {
              kind = 'attribute';
            } else if (name[name.length-1] == '$') {
              name = name.slice(0, -1);
              kind = 'attribute';
            }
            // Initialize attribute bindings with any literal parts
            let literal = literalFromParts(parts);
            if (literal && kind == 'attribute') {
              // Ensure a ShadyCSS template scoped style is not removed
              // when a class$ binding's initial literal value is set.
              if (name == 'class' && node.hasAttribute('class')) {
                literal += ' ' + node.getAttribute(name);
              }
              node.setAttribute(name, literal);
            }
            // support disable-upgrade
            if (kind == 'attribute' && origName == 'disable-upgrade$') {
              node.setAttribute(name, '');
            }
            // Clear attribute before removing, since IE won't allow removing
            // `value` attribute if it previously had a value (can't
            // unconditionally set '' before removing since attributes with `$`
            // can't be set using setAttribute)
            if (node.localName === 'input' && origName === 'value') {
              node.setAttribute(origName, '');
            }
            // Remove annotation
            node.removeAttribute(origName);
            // Case hackery: attributes are lower-case, but bind targets
            // (properties) are case sensitive. Gambit is to map dash-case to
            // camel-case: `foo-bar` becomes `fooBar`.
            // Attribute bindings are excepted.
            if (kind === 'property') {
              name = dashToCamelCase(name);
            }
            addBinding(this, templateInfo, nodeInfo, kind, name, parts, literal);
            return true;
          } else {
            // TODO(https://github.com/google/closure-compiler/issues/3240):
            //     Change back to just super.methodCall()
            return propertyEffectsBase._parseTemplateNodeAttribute.call(
              this, node, templateInfo, nodeInfo, name, value);
          }
        }

        /**
         * Overrides default `TemplateStamp` implementation to add support for
         * binding the properties that a nested template depends on to the template
         * as `_host_<property>`.
         *
         * @param {Node} node Node to parse
         * @param {TemplateInfo} templateInfo Template metadata for current template
         * @param {NodeInfo} nodeInfo Node metadata for current template node
         * @return {boolean} `true` if the visited node added node-specific
         *   metadata to `nodeInfo`
         * @protected
         * @suppress {missingProperties} Interfaces in closure do not inherit statics, but classes do
         * @nocollapse
         */
        static _parseTemplateNestedTemplate(node, templateInfo, nodeInfo) {
          // TODO(https://github.com/google/closure-compiler/issues/3240):
          //     Change back to just super.methodCall()
          let noted = propertyEffectsBase._parseTemplateNestedTemplate.call(
            this, node, templateInfo, nodeInfo);
          const parent = node.parentNode;
          const nestedTemplateInfo = nodeInfo.templateInfo;
          const isDomIf = parent.localName === 'dom-if';
          const isDomRepeat = parent.localName === 'dom-repeat';
          // Remove nested template and redirect its host bindings & templateInfo
          // onto the parent (dom-if/repeat element)'s nodeInfo
          if (removeNestedTemplates && (isDomIf || isDomRepeat)) {
            parent.removeChild(node);
            // Use the parent's nodeInfo (for the dom-if/repeat) to record the
            // templateInfo, and use that for any host property bindings below
            nodeInfo = nodeInfo.parentInfo;
            nodeInfo.templateInfo = nestedTemplateInfo;
            // Ensure the parent dom-if/repeat is noted since it now may have host
            // bindings; it may not have been if it did not have its own bindings
            nodeInfo.noted = true;
            noted = false;
          }
          // Merge host props into outer template and add bindings
          let hostProps = nestedTemplateInfo.hostProps;
          if (fastDomIf && isDomIf) {
            // `fastDomIf` mode uses runtime-template stamping to add accessors/
            // effects to properties used in its template; as such we don't need to
            // tax the host element with `_host_` bindings for the `dom-if`.
            // However, in the event it is nested in a `dom-repeat`, it is still
            // important that its host properties are added to the
            // TemplateInstance's `hostProps` so that they are forwarded to the
            // TemplateInstance.
            if (hostProps) {
              templateInfo.hostProps =
                Object.assign(templateInfo.hostProps || {}, hostProps);
              // Ensure the dom-if is noted so that it has a __dataHost, since
              // `fastDomIf` uses the host for runtime template stamping; note this
              // was already ensured above in the `removeNestedTemplates` case
              if (!removeNestedTemplates) {
                nodeInfo.parentInfo.noted = true;
              }
            }
          } else {
            let mode = '{';
            for (let source in hostProps) {
              let parts = [{ mode, source, dependencies: [source], hostProp: true }];
              addBinding(this, templateInfo, nodeInfo, 'property', '_host_' + source, parts);
            }
          }
          return noted;
        }

        /**
         * Called to parse text in a template (either attribute values or
         * textContent) into binding metadata.
         *
         * Any overrides of this method should return an array of binding part
         * metadata  representing one or more bindings found in the provided text
         * and any "literal" text in between.  Any non-literal parts will be passed
         * to `_evaluateBinding` when any dependencies change.  The only required
         * fields of each "part" in the returned array are as follows:
         *
         * - `dependencies` - Array containing trigger metadata for each property
         *   that should trigger the binding to update
         * - `literal` - String containing text if the part represents a literal;
         *   in this case no `dependencies` are needed
         *
         * Additional metadata for use by `_evaluateBinding` may be provided in
         * each part object as needed.
         *
         * The default implementation handles the following types of bindings
         * (one or more may be intermixed with literal strings):
         * - Property binding: `[[prop]]`
         * - Path binding: `[[object.prop]]`
         * - Negated property or path bindings: `[[!prop]]` or `[[!object.prop]]`
         * - Two-way property or path bindings (supports negation):
         *   `{{prop}}`, `{{object.prop}}`, `{{!prop}}` or `{{!object.prop}}`
         * - Inline computed method (supports negation):
         *   `[[compute(a, 'literal', b)]]`, `[[!compute(a, 'literal', b)]]`
         *
         * The default implementation uses a regular expression for best
         * performance. However, the regular expression uses a white-list of
         * allowed characters in a data-binding, which causes problems for
         * data-bindings that do use characters not in this white-list.
         *
         * Instead of updating the white-list with all allowed characters,
         * there is a StrictBindingParser (see lib/mixins/strict-binding-parser)
         * that uses a state machine instead. This state machine is able to handle
         * all characters. However, it is slightly less performant, therefore we
         * extracted it into a separate optional mixin.
         *
         * @param {string} text Text to parse from attribute or textContent
         * @param {Object} templateInfo Current template metadata
         * @return {Array<!BindingPart>} Array of binding part metadata
         * @protected
         * @nocollapse
         */
        static _parseBindings(text, templateInfo) {
          let parts = [];
          let lastIndex = 0;
          let m;
          // Example: "literal1{{prop}}literal2[[!compute(foo,bar)]]final"
          // Regex matches:
          //        Iteration 1:  Iteration 2:
          // m[1]: '{{'          '[['
          // m[2]: ''            '!'
          // m[3]: 'prop'        'compute(foo,bar)'
          while ((m = bindingRegex.exec(text)) !== null) {
            // Add literal part
            if (m.index > lastIndex) {
              parts.push({literal: text.slice(lastIndex, m.index)});
            }
            // Add binding part
            let mode = m[1][0];
            let negate = Boolean(m[2]);
            let source = m[3].trim();
            let customEvent = false, notifyEvent = '', colon = -1;
            if (mode == '{' && (colon = source.indexOf('::')) > 0) {
              notifyEvent = source.substring(colon + 2);
              source = source.substring(0, colon);
              customEvent = true;
            }
            let signature = parseMethod(source);
            let dependencies = [];
            if (signature) {
              // Inline computed function
              let {args, methodName} = signature;
              for (let i=0; i<args.length; i++) {
                let arg = args[i];
                if (!arg.literal) {
                  dependencies.push(arg);
                }
              }
              let dynamicFns = templateInfo.dynamicFns;
              if (dynamicFns && dynamicFns[methodName] || signature.static) {
                dependencies.push(methodName);
                signature.dynamicFn = true;
              }
            } else {
              // Property or path
              dependencies.push(source);
            }
            parts.push({
              source, mode, negate, customEvent, signature, dependencies,
              event: notifyEvent
            });
            lastIndex = bindingRegex.lastIndex;
          }
          // Add a final literal part
          if (lastIndex && lastIndex < text.length) {
            let literal = text.substring(lastIndex);
            if (literal) {
              parts.push({
                literal: literal
              });
            }
          }
          if (parts.length) {
            return parts;
          } else {
            return null;
          }
        }

        /**
         * Called to evaluate a previously parsed binding part based on a set of
         * one or more changed dependencies.
         *
         * @param {!Polymer_PropertyEffects} inst Element that should be used as
         *     scope for binding dependencies
         * @param {BindingPart} part Binding part metadata
         * @param {string} path Property/path that triggered this effect
         * @param {Object} props Bag of current property changes
         * @param {Object} oldProps Bag of previous values for changed properties
         * @param {boolean} hasPaths True with `props` contains one or more paths
         * @return {*} Value the binding part evaluated to
         * @protected
         * @nocollapse
         */
        static _evaluateBinding(inst, part, path, props, oldProps, hasPaths) {
          let value;
          if (part.signature) {
            value = runMethodEffect(inst, path, props, oldProps, part.signature);
          } else if (path != part.source) {
            value = get(inst, part.source);
          } else {
            if (hasPaths && isPath(path)) {
              value = get(inst, path);
            } else {
              value = inst.__data[path];
            }
          }
          if (part.negate) {
            value = !value;
          }
          return value;
        }

      }

      return PropertyEffects;
    });

    /**
     * Stack for enqueuing client dom created by a host element.
     *
     * By default elements are flushed via `_flushProperties` when
     * `connectedCallback` is called. Elements attach their client dom to
     * themselves at `ready` time which results from this first flush.
     * This provides an ordering guarantee that the client dom an element
     * creates is flushed before the element itself (i.e. client `ready`
     * fires before host `ready`).
     *
     * However, if `_flushProperties` is called *before* an element is connected,
     * as for example `Templatize` does, this ordering guarantee cannot be
     * satisfied because no elements are connected. (Note: Bound elements that
     * receive data do become enqueued clients and are properly ordered but
     * unbound elements are not.)
     *
     * To maintain the desired "client before host" ordering guarantee for this
     * case we rely on the "host stack. Client nodes registers themselves with
     * the creating host element when created. This ensures that all client dom
     * is readied in the proper order, maintaining the desired guarantee.
     *
     * @private
     */
    const hostStack = [];

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    /**
     * Registers a class prototype for telemetry purposes.
     * @param {!PolymerElementConstructor} prototype Element prototype to register
     * @protected
     */
    function register(prototype) {
    }

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    /**
     * Creates a copy of `props` with each property normalized such that
     * upgraded it is an object with at least a type property { type: Type}.
     *
     * @param {!Object} props Properties to normalize
     * @return {!Object} Copy of input `props` with normalized properties that
     * are in the form {type: Type}
     * @private
     */
    function normalizeProperties(props) {
      const output = {};
      for (let p in props) {
        const o = props[p];
        output[p] = (typeof o === 'function') ? {type: o} : o;
      }
      return output;
    }

    /**
     * Mixin that provides a minimal starting point to using the PropertiesChanged
     * mixin by providing a mechanism to declare properties in a static
     * getter (e.g. static get properties() { return { foo: String } }). Changes
     * are reported via the `_propertiesChanged` method.
     *
     * This mixin provides no specific support for rendering. Users are expected
     * to create a ShadowRoot and put content into it and update it in whatever
     * way makes sense. This can be done in reaction to properties changing by
     * implementing `_propertiesChanged`.
     *
     * @mixinFunction
     * @polymer
     * @appliesMixin PropertiesChanged
     * @summary Mixin that provides a minimal starting point for using
     * the PropertiesChanged mixin by providing a declarative `properties` object.
     * @template T
     * @param {function(new:T)} superClass Class to apply mixin to.
     * @return {function(new:T)} superClass with mixin applied.
     */
    const PropertiesMixin = dedupingMixin(superClass => {

     /**
      * @constructor
      * @implements {Polymer_PropertiesChanged}
      * @private
      */
     const base = PropertiesChanged(superClass);

     /**
      * Returns the super class constructor for the given class, if it is an
      * instance of the PropertiesMixin.
      *
      * @param {!PropertiesMixinConstructor} constructor PropertiesMixin constructor
      * @return {?PropertiesMixinConstructor} Super class constructor
      */
     function superPropertiesClass(constructor) {
       const superCtor = Object.getPrototypeOf(constructor);

       // Note, the `PropertiesMixin` class below only refers to the class
       // generated by this call to the mixin; the instanceof test only works
       // because the mixin is deduped and guaranteed only to apply once, hence
       // all constructors in a proto chain will see the same `PropertiesMixin`
       return (superCtor.prototype instanceof PropertiesMixin) ?
         /** @type {!PropertiesMixinConstructor} */ (superCtor) : null;
     }

     /**
      * Returns a memoized version of the `properties` object for the
      * given class. Properties not in object format are converted to at
      * least {type}.
      *
      * @param {PropertiesMixinConstructor} constructor PropertiesMixin constructor
      * @return {Object} Memoized properties object
      */
     function ownProperties(constructor) {
       if (!constructor.hasOwnProperty(JSCompiler_renameProperty('__ownProperties', constructor))) {
         let props = null;

         if (constructor.hasOwnProperty(JSCompiler_renameProperty('properties', constructor))) {
           const properties = constructor.properties;

           if (properties) {
            props = normalizeProperties(properties);
           }
         }

         constructor.__ownProperties = props;
       }
       return constructor.__ownProperties;
     }

     /**
      * @polymer
      * @mixinClass
      * @extends {base}
      * @implements {Polymer_PropertiesMixin}
      * @unrestricted
      */
     class PropertiesMixin extends base {

       /**
        * Implements standard custom elements getter to observes the attributes
        * listed in `properties`.
        * @suppress {missingProperties} Interfaces in closure do not inherit statics, but classes do
        * @nocollapse
        */
       static get observedAttributes() {
         if (!this.hasOwnProperty(JSCompiler_renameProperty('__observedAttributes', this))) {
           register(this.prototype);
           const props = this._properties;
           this.__observedAttributes = props ? Object.keys(props).map(p => this.prototype._addPropertyToAttributeMap(p)) : [];
         }
         return this.__observedAttributes;
       }

       /**
        * Finalizes an element definition, including ensuring any super classes
        * are also finalized. This includes ensuring property
        * accessors exist on the element prototype. This method calls
        * `_finalizeClass` to finalize each constructor in the prototype chain.
        * @return {void}
        * @nocollapse
        */
       static finalize() {
         if (!this.hasOwnProperty(JSCompiler_renameProperty('__finalized', this))) {
           const superCtor = superPropertiesClass(/** @type {!PropertiesMixinConstructor} */(this));
           if (superCtor) {
             superCtor.finalize();
           }
           this.__finalized = true;
           this._finalizeClass();
         }
       }

       /**
        * Finalize an element class. This includes ensuring property
        * accessors exist on the element prototype. This method is called by
        * `finalize` and finalizes the class constructor.
        *
        * @protected
        * @nocollapse
        */
       static _finalizeClass() {
         const props = ownProperties(/** @type {!PropertiesMixinConstructor} */(this));
         if (props) {
           /** @type {?} */ (this).createProperties(props);
         }
       }

       /**
        * Returns a memoized version of all properties, including those inherited
        * from super classes. Properties not in object format are converted to
        * at least {type}.
        *
        * @return {Object} Object containing properties for this class
        * @protected
        * @nocollapse
        */
       static get _properties() {
         if (!this.hasOwnProperty(
           JSCompiler_renameProperty('__properties', this))) {
           const superCtor = superPropertiesClass(/** @type {!PropertiesMixinConstructor} */(this));
           this.__properties = Object.assign({},
             superCtor && superCtor._properties,
             ownProperties(/** @type {PropertiesMixinConstructor} */(this)));
         }
         return this.__properties;
       }

       /**
        * Overrides `PropertiesChanged` method to return type specified in the
        * static `properties` object for the given property.
        * @param {string} name Name of property
        * @return {*} Type to which to deserialize attribute
        *
        * @protected
        * @nocollapse
        */
       static typeForProperty(name) {
         const info = this._properties[name];
         return info && info.type;
       }

       /**
        * Overrides `PropertiesChanged` method and adds a call to
        * `finalize` which lazily configures the element's property accessors.
        * @override
        * @return {void}
        */
       _initializeProperties() {
         this.constructor.finalize();
         super._initializeProperties();
       }

       /**
        * Called when the element is added to a document.
        * Calls `_enableProperties` to turn on property system from
        * `PropertiesChanged`.
        * @suppress {missingProperties} Super may or may not implement the callback
        * @return {void}
        * @override
        */
       connectedCallback() {
         if (super.connectedCallback) {
           super.connectedCallback();
         }
         this._enableProperties();
       }

       /**
        * Called when the element is removed from a document
        * @suppress {missingProperties} Super may or may not implement the callback
        * @return {void}
        * @override
        */
       disconnectedCallback() {
         if (super.disconnectedCallback) {
           super.disconnectedCallback();
         }
       }

     }

     return PropertiesMixin;

    });

    /**
     * @fileoverview
     * @suppress {checkPrototypalTypes}
     * @license Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
     * This code may only be used under the BSD style license found at
     * http://polymer.github.io/LICENSE.txt The complete set of authors may be found
     * at http://polymer.github.io/AUTHORS.txt The complete set of contributors may
     * be found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by
     * Google as part of the polymer project is also subject to an additional IP
     * rights grant found at http://polymer.github.io/PATENTS.txt
     */

    /**
     * Current Polymer version in Semver notation.
     * @type {string} Semver notation of the current version of Polymer.
     */
    const version = '3.4.1';

    const builtCSS = window.ShadyCSS && window.ShadyCSS['cssBuild'];

    /**
     * Element class mixin that provides the core API for Polymer's meta-programming
     * features including template stamping, data-binding, attribute deserialization,
     * and property change observation.
     *
     * Subclassers may provide the following static getters to return metadata
     * used to configure Polymer's features for the class:
     *
     * - `static get is()`: When the template is provided via a `dom-module`,
     *   users should return the `dom-module` id from a static `is` getter.  If
     *   no template is needed or the template is provided directly via the
     *   `template` getter, there is no need to define `is` for the element.
     *
     * - `static get template()`: Users may provide the template directly (as
     *   opposed to via `dom-module`) by implementing a static `template` getter.
     *   The getter must return an `HTMLTemplateElement`.
     *
     * - `static get properties()`: Should return an object describing
     *   property-related metadata used by Polymer features (key: property name
     *   value: object containing property metadata). Valid keys in per-property
     *   metadata include:
     *   - `type` (String|Number|Object|Array|...): Used by
     *     `attributeChangedCallback` to determine how string-based attributes
     *     are deserialized to JavaScript property values.
     *   - `notify` (boolean): Causes a change in the property to fire a
     *     non-bubbling event called `<property>-changed`. Elements that have
     *     enabled two-way binding to the property use this event to observe changes.
     *   - `readOnly` (boolean): Creates a getter for the property, but no setter.
     *     To set a read-only property, use the private setter method
     *     `_setProperty(property, value)`.
     *   - `observer` (string): Observer method name that will be called when
     *     the property changes. The arguments of the method are
     *     `(value, previousValue)`.
     *   - `computed` (string): String describing method and dependent properties
     *     for computing the value of this property (e.g. `'computeFoo(bar, zot)'`).
     *     Computed properties are read-only by default and can only be changed
     *     via the return value of the computing method.
     *
     * - `static get observers()`: Array of strings describing multi-property
     *   observer methods and their dependent properties (e.g.
     *   `'observeABC(a, b, c)'`).
     *
     * The base class provides default implementations for the following standard
     * custom element lifecycle callbacks; users may override these, but should
     * call the super method to ensure
     * - `constructor`: Run when the element is created or upgraded
     * - `connectedCallback`: Run each time the element is connected to the
     *   document
     * - `disconnectedCallback`: Run each time the element is disconnected from
     *   the document
     * - `attributeChangedCallback`: Run each time an attribute in
     *   `observedAttributes` is set or removed (note: this element's default
     *   `observedAttributes` implementation will automatically return an array
     *   of dash-cased attributes based on `properties`)
     *
     * @mixinFunction
     * @polymer
     * @appliesMixin PropertyEffects
     * @appliesMixin PropertiesMixin
     * @property rootPath {string} Set to the value of `rootPath`,
     *   which defaults to the main document path
     * @property importPath {string} Set to the value of the class's static
     *   `importPath` property, which defaults to the path of this element's
     *   `dom-module` (when `is` is used), but can be overridden for other
     *   import strategies.
     * @summary Element class mixin that provides the core API for Polymer's
     * meta-programming features.
     * @template T
     * @param {function(new:T)} superClass Class to apply mixin to.
     * @return {function(new:T)} superClass with mixin applied.
     */
    const ElementMixin = dedupingMixin(base => {
      /**
       * @constructor
       * @implements {Polymer_PropertyEffects}
       * @implements {Polymer_PropertiesMixin}
       * @extends {HTMLElement}
       * @private
       */
      const polymerElementBase = PropertiesMixin(PropertyEffects(base));

      /**
       * Returns a list of properties with default values.
       * This list is created as an optimization since it is a subset of
       * the list returned from `_properties`.
       * This list is used in `_initializeProperties` to set property defaults.
       *
       * @param {PolymerElementConstructor} constructor Element class
       * @return {PolymerElementProperties} Flattened properties for this class
       *   that have default values
       * @private
       */
      function propertyDefaults(constructor) {
        if (!constructor.hasOwnProperty(
          JSCompiler_renameProperty('__propertyDefaults', constructor))) {
          constructor.__propertyDefaults = null;
          let props = constructor._properties;
          for (let p in props) {
            let info = props[p];
            if ('value' in info) {
              constructor.__propertyDefaults = constructor.__propertyDefaults || {};
              constructor.__propertyDefaults[p] = info;
            }
          }
        }
        return constructor.__propertyDefaults;
      }

      /**
       * Returns a memoized version of the `observers` array.
       * @param {PolymerElementConstructor} constructor Element class
       * @return {Array} Array containing own observers for the given class
       * @protected
       */
      function ownObservers(constructor) {
        if (!constructor.hasOwnProperty(
          JSCompiler_renameProperty('__ownObservers', constructor))) {
          constructor.__ownObservers =
              constructor.hasOwnProperty(
                  JSCompiler_renameProperty('observers', constructor)) ?
              /** @type {PolymerElementConstructor} */ (constructor).observers :
              null;
        }
        return constructor.__ownObservers;
      }

      /**
       * Creates effects for a property.
       *
       * Note, once a property has been set to
       * `readOnly`, `computed`, `reflectToAttribute`, or `notify`
       * these values may not be changed. For example, a subclass cannot
       * alter these settings. However, additional `observers` may be added
       * by subclasses.
       *
       * The info object should contain property metadata as follows:
       *
       * * `type`: {function} type to which an attribute matching the property
       * is deserialized. Note the property is camel-cased from a dash-cased
       * attribute. For example, 'foo-bar' attribute is deserialized to a
       * property named 'fooBar'.
       *
       * * `readOnly`: {boolean} creates a readOnly property and
       * makes a private setter for the private of the form '_setFoo' for a
       * property 'foo',
       *
       * * `computed`: {string} creates a computed property. A computed property
       * is also automatically set to `readOnly: true`. The value is calculated
       * by running a method and arguments parsed from the given string. For
       * example 'compute(foo)' will compute a given property when the
       * 'foo' property changes by executing the 'compute' method. This method
       * must return the computed value.
       *
       * * `reflectToAttribute`: {boolean} If true, the property value is reflected
       * to an attribute of the same name. Note, the attribute is dash-cased
       * so a property named 'fooBar' is reflected as 'foo-bar'.
       *
       * * `notify`: {boolean} sends a non-bubbling notification event when
       * the property changes. For example, a property named 'foo' sends an
       * event named 'foo-changed' with `event.detail` set to the value of
       * the property.
       *
       * * observer: {string} name of a method that runs when the property
       * changes. The arguments of the method are (value, previousValue).
       *
       * Note: Users may want control over modifying property
       * effects via subclassing. For example, a user might want to make a
       * reflectToAttribute property not do so in a subclass. We've chosen to
       * disable this because it leads to additional complication.
       * For example, a readOnly effect generates a special setter. If a subclass
       * disables the effect, the setter would fail unexpectedly.
       * Based on feedback, we may want to try to make effects more malleable
       * and/or provide an advanced api for manipulating them.
       *
       * @param {!PolymerElement} proto Element class prototype to add accessors
       *   and effects to
       * @param {string} name Name of the property.
       * @param {Object} info Info object from which to create property effects.
       * Supported keys:
       * @param {Object} allProps Flattened map of all properties defined in this
       *   element (including inherited properties)
       * @return {void}
       * @private
       */
      function createPropertyFromConfig(proto, name, info, allProps) {
        // computed forces readOnly...
        if (info.computed) {
          info.readOnly = true;
        }
        // Note, since all computed properties are readOnly, this prevents
        // adding additional computed property effects (which leads to a confusing
        // setup where multiple triggers for setting a property)
        // While we do have `hasComputedEffect` this is set on the property's
        // dependencies rather than itself.
        if (info.computed) {
          if (proto._hasReadOnlyEffect(name)) {
            console.warn(`Cannot redefine computed property '${name}'.`);
          } else {
            proto._createComputedProperty(name, info.computed, allProps);
          }
        }
        if (info.readOnly && !proto._hasReadOnlyEffect(name)) {
          proto._createReadOnlyProperty(name, !info.computed);
        } else if (info.readOnly === false && proto._hasReadOnlyEffect(name)) {
          console.warn(`Cannot make readOnly property '${name}' non-readOnly.`);
        }
        if (info.reflectToAttribute && !proto._hasReflectEffect(name)) {
          proto._createReflectedProperty(name);
        } else if (info.reflectToAttribute === false && proto._hasReflectEffect(name)) {
          console.warn(`Cannot make reflected property '${name}' non-reflected.`);
        }
        if (info.notify && !proto._hasNotifyEffect(name)) {
          proto._createNotifyingProperty(name);
        } else if (info.notify === false && proto._hasNotifyEffect(name)) {
          console.warn(`Cannot make notify property '${name}' non-notify.`);
        }
        // always add observer
        if (info.observer) {
          proto._createPropertyObserver(name, info.observer, allProps[info.observer]);
        }
        // always create the mapping from attribute back to property for deserialization.
        proto._addPropertyToAttributeMap(name);
      }

      /**
       * Process all style elements in the element template. Styles with the
       * `include` attribute are processed such that any styles in
       * the associated "style modules" are included in the element template.
       * @param {PolymerElementConstructor} klass Element class
       * @param {!HTMLTemplateElement} template Template to process
       * @param {string} is Name of element
       * @param {string} baseURI Base URI for element
       * @private
       */
      function processElementStyles(klass, template, is, baseURI) {
        if (!builtCSS) {
          const templateStyles = template.content.querySelectorAll('style');
          const stylesWithImports = stylesFromTemplate(template);
          // insert styles from <link rel="import" type="css"> at the top of the template
          const linkedStyles = stylesFromModuleImports(is);
          const firstTemplateChild = template.content.firstElementChild;
          for (let idx = 0; idx < linkedStyles.length; idx++) {
            let s = linkedStyles[idx];
            s.textContent = klass._processStyleText(s.textContent, baseURI);
            template.content.insertBefore(s, firstTemplateChild);
          }
          // keep track of the last "concrete" style in the template we have encountered
          let templateStyleIndex = 0;
          // ensure all gathered styles are actually in this template.
          for (let i = 0; i < stylesWithImports.length; i++) {
            let s = stylesWithImports[i];
            let templateStyle = templateStyles[templateStyleIndex];
            // if the style is not in this template, it's been "included" and
            // we put a clone of it in the template before the style that included it
            if (templateStyle !== s) {
              s = s.cloneNode(true);
              templateStyle.parentNode.insertBefore(s, templateStyle);
            } else {
              templateStyleIndex++;
            }
            s.textContent = klass._processStyleText(s.textContent, baseURI);
          }
        }
        if (window.ShadyCSS) {
          window.ShadyCSS.prepareTemplate(template, is);
        }
        // Support for `adoptedStylesheets` relies on using native Shadow DOM
        // and built CSS. Built CSS is required because runtime transformation of
        // `@apply` is not supported. This is because ShadyCSS relies on being able
        // to update a `style` element in the element template and this is
        // removed when using `adoptedStyleSheets`.
        // Note, it would be more efficient to allow style includes to become
        // separate stylesheets; however, because of `@apply` these are
        // potentially not shareable and sharing the ones that could be shared
        // would require some coordination. To keep it simple, all the includes
        // and styles are collapsed into a single shareable stylesheet.
        if (useAdoptedStyleSheetsWithBuiltCSS && builtCSS &&
            supportsAdoptingStyleSheets) {
          // Remove styles in template and make a shareable stylesheet
          const styles = template.content.querySelectorAll('style');
          if (styles) {
            let css = '';
            Array.from(styles).forEach(s => {
              css += s.textContent;
              s.parentNode.removeChild(s);
            });
            klass._styleSheet = new CSSStyleSheet();
            klass._styleSheet.replaceSync(css);
          }
        }
      }

      /**
       * Look up template from dom-module for element
       *
       * @param {string} is Element name to look up
       * @return {?HTMLTemplateElement|undefined} Template found in dom module, or
       *   undefined if not found
       * @protected
       */
      function getTemplateFromDomModule(is) {
        let template = null;
        // Under strictTemplatePolicy in 3.x+, dom-module lookup is only allowed
        // when opted-in via allowTemplateFromDomModule
        if (is && (!strictTemplatePolicy || allowTemplateFromDomModule)) {
          template = /** @type {?HTMLTemplateElement} */ (
              DomModule.import(is, 'template'));
          // Under strictTemplatePolicy, require any element with an `is`
          // specified to have a dom-module
          if (strictTemplatePolicy && !template) {
            throw new Error(`strictTemplatePolicy: expecting dom-module or null template for ${is}`);
          }
        }
        return template;
      }

      /**
       * @polymer
       * @mixinClass
       * @unrestricted
       * @implements {Polymer_ElementMixin}
       * @extends {polymerElementBase}
       */
      class PolymerElement extends polymerElementBase {

        /**
         * Current Polymer version in Semver notation.
         * @type {string} Semver notation of the current version of Polymer.
         * @nocollapse
         */
        static get polymerElementVersion() {
          return version;
        }

        /**
         * Override of PropertiesMixin _finalizeClass to create observers and
         * find the template.
         * @return {void}
         * @protected
         * @suppress {missingProperties} Interfaces in closure do not inherit statics, but classes do
         * @nocollapse
         */
        static _finalizeClass() {
          // TODO(https://github.com/google/closure-compiler/issues/3240):
          //     Change back to just super.methodCall()
          polymerElementBase._finalizeClass.call(this);
          const observers = ownObservers(this);
          if (observers) {
            this.createObservers(observers, this._properties);
          }
          this._prepareTemplate();
        }

        /** @nocollapse */
        static _prepareTemplate() {
          // note: create "working" template that is finalized at instance time
          let template = /** @type {PolymerElementConstructor} */ (this).template;
          if (template) {
            if (typeof template === 'string') {
              console.error('template getter must return HTMLTemplateElement');
              template = null;
            } else if (!legacyOptimizations) {
              template = template.cloneNode(true);
            }
          }

          /** @override */
          this.prototype._template = template;
        }

        /**
         * Override of PropertiesChanged createProperties to create accessors
         * and property effects for all of the properties.
         * @param {!Object} props .
         * @return {void}
         * @protected
         * @nocollapse
         */
        static createProperties(props) {
          for (let p in props) {
            createPropertyFromConfig(
                /** @type {?} */ (this.prototype), p, props[p], props);
          }
        }

        /**
         * Creates observers for the given `observers` array.
         * Leverages `PropertyEffects` to create observers.
         * @param {Object} observers Array of observer descriptors for
         *   this class
         * @param {Object} dynamicFns Object containing keys for any properties
         *   that are functions and should trigger the effect when the function
         *   reference is changed
         * @return {void}
         * @protected
         * @nocollapse
         */
        static createObservers(observers, dynamicFns) {
          const proto = this.prototype;
          for (let i=0; i < observers.length; i++) {
            proto._createMethodObserver(observers[i], dynamicFns);
          }
        }

        /**
         * Returns the template that will be stamped into this element's shadow root.
         *
         * If a `static get is()` getter is defined, the default implementation will
         * return the first `<template>` in a `dom-module` whose `id` matches this
         * element's `is` (note that a `_template` property on the class prototype
         * takes precedence over the `dom-module` template, to maintain legacy
         * element semantics; a subclass will subsequently fall back to its super
         * class template if neither a `prototype._template` or a `dom-module` for
         * the class's `is` was found).
         *
         * Users may override this getter to return an arbitrary template
         * (in which case the `is` getter is unnecessary). The template returned
         * must be an `HTMLTemplateElement`.
         *
         * Note that when subclassing, if the super class overrode the default
         * implementation and the subclass would like to provide an alternate
         * template via a `dom-module`, it should override this getter and
         * return `DomModule.import(this.is, 'template')`.
         *
         * If a subclass would like to modify the super class template, it should
         * clone it rather than modify it in place.  If the getter does expensive
         * work such as cloning/modifying a template, it should memoize the
         * template for maximum performance:
         *
         *   let memoizedTemplate;
         *   class MySubClass extends MySuperClass {
         *     static get template() {
         *       if (!memoizedTemplate) {
         *         memoizedTemplate = super.template.cloneNode(true);
         *         let subContent = document.createElement('div');
         *         subContent.textContent = 'This came from MySubClass';
         *         memoizedTemplate.content.appendChild(subContent);
         *       }
         *       return memoizedTemplate;
         *     }
         *   }
         *
         * @return {!HTMLTemplateElement|string} Template to be stamped
         * @nocollapse
         */
        static get template() {
          // Explanation of template-related properties:
          // - constructor.template (this getter): the template for the class.
          //     This can come from the prototype (for legacy elements), from a
          //     dom-module, or from the super class's template (or can be overridden
          //     altogether by the user)
          // - constructor._template: memoized version of constructor.template
          // - prototype._template: working template for the element, which will be
          //     parsed and modified in place. It is a cloned version of
          //     constructor.template, saved in _finalizeClass(). Note that before
          //     this getter is called, for legacy elements this could be from a
          //     _template field on the info object passed to Polymer(), a behavior,
          //     or set in registered(); once the static getter runs, a clone of it
          //     will overwrite it on the prototype as the working template.
          if (!this.hasOwnProperty(JSCompiler_renameProperty('_template', this))) {
            const protoTemplate = this.prototype.hasOwnProperty(
              JSCompiler_renameProperty('_template', this.prototype)) ?
              this.prototype._template : undefined;
            this._template =
              // If user has put template on prototype (e.g. in legacy via registered
              // callback or info object), prefer that first. Note that `null` is
              // used as a sentinel to indicate "no template" and can be used to
              // override a super template, whereas `undefined` is used as a
              // sentinel to mean "fall-back to default template lookup" via
              // dom-module and/or super.template.
              protoTemplate !== undefined ? protoTemplate :
              // Look in dom-module associated with this element's is
              ((this.hasOwnProperty(JSCompiler_renameProperty('is', this)) &&
              (getTemplateFromDomModule(/** @type {PolymerElementConstructor}*/ (this).is))) ||
              // Next look for superclass template (call the super impl this
              // way so that `this` points to the superclass)
              Object.getPrototypeOf(/** @type {PolymerElementConstructor}*/ (this).prototype).constructor.template);
          }
          return this._template;
        }

        /**
         * Set the template.
         *
         * @param {!HTMLTemplateElement|string} value Template to set.
         * @nocollapse
         */
        static set template(value) {
          this._template = value;
        }

        /**
         * Path matching the url from which the element was imported.
         *
         * This path is used to resolve url's in template style cssText.
         * The `importPath` property is also set on element instances and can be
         * used to create bindings relative to the import path.
         *
         * For elements defined in ES modules, users should implement
         * `static get importMeta() { return import.meta; }`, and the default
         * implementation of `importPath` will  return `import.meta.url`'s path.
         * For elements defined in HTML imports, this getter will return the path
         * to the document containing a `dom-module` element matching this
         * element's static `is` property.
         *
         * Note, this path should contain a trailing `/`.
         *
         * @return {string} The import path for this element class
         * @suppress {missingProperties}
         * @nocollapse
         */
        static get importPath() {
          if (!this.hasOwnProperty(JSCompiler_renameProperty('_importPath', this))) {
            const meta = this.importMeta;
            if (meta) {
              this._importPath = pathFromUrl(meta.url);
            } else {
              const module = DomModule.import(/** @type {PolymerElementConstructor} */ (this).is);
              this._importPath = (module && module.assetpath) ||
                Object.getPrototypeOf(/** @type {PolymerElementConstructor}*/ (this).prototype).constructor.importPath;
            }
          }
          return this._importPath;
        }

        constructor() {
          super();
          /** @type {HTMLTemplateElement} */
          this._template;
          /** @type {string} */
          this._importPath;
          /** @type {string} */
          this.rootPath;
          /** @type {string} */
          this.importPath;
          /** @type {StampedTemplate | HTMLElement | ShadowRoot} */
          this.root;
          /** @type {!Object<string, !Element>} */
          this.$;
        }

        /**
         * Overrides the default `PropertyAccessors` to ensure class
         * metaprogramming related to property accessors and effects has
         * completed (calls `finalize`).
         *
         * It also initializes any property defaults provided via `value` in
         * `properties` metadata.
         *
         * @return {void}
         * @override
         * @suppress {invalidCasts,missingProperties} go/missingfnprops
         */
        _initializeProperties() {
          this.constructor.finalize();
          // note: finalize template when we have access to `localName` to
          // avoid dependence on `is` for polyfilling styling.
          this.constructor._finalizeTemplate(/** @type {!HTMLElement} */(this).localName);
          super._initializeProperties();
          // set path defaults
          this.rootPath = rootPath;
          this.importPath = this.constructor.importPath;
          // apply property defaults...
          let p$ = propertyDefaults(this.constructor);
          if (!p$) {
            return;
          }
          for (let p in p$) {
            let info = p$[p];
            if (this._canApplyPropertyDefault(p)) {
              let value = typeof info.value == 'function' ?
                info.value.call(this) :
                info.value;
              // Set via `_setProperty` if there is an accessor, to enable
              // initializing readOnly property defaults
              if (this._hasAccessor(p)) {
                this._setPendingProperty(p, value, true);
              } else {
                this[p] = value;
              }
            }
          }
        }

        /**
         * Determines if a property dfeault can be applied. For example, this
         * prevents a default from being applied when a property that has no
         * accessor is overridden by its host before upgrade (e.g. via a binding).
         * @override
         * @param {string} property Name of the property
         * @return {boolean} Returns true if the property default can be applied.
         */
        _canApplyPropertyDefault(property) {
          return !this.hasOwnProperty(property);
        }

        /**
         * Gather style text for a style element in the template.
         *
         * @param {string} cssText Text containing styling to process
         * @param {string} baseURI Base URI to rebase CSS paths against
         * @return {string} The processed CSS text
         * @protected
         * @nocollapse
         */
        static _processStyleText(cssText, baseURI) {
          return resolveCss(cssText, baseURI);
        }

        /**
        * Configures an element `proto` to function with a given `template`.
        * The element name `is` and extends `ext` must be specified for ShadyCSS
        * style scoping.
        *
        * @param {string} is Tag name (or type extension name) for this element
        * @return {void}
        * @protected
        * @nocollapse
        */
        static _finalizeTemplate(is) {
          /** @const {HTMLTemplateElement} */
          const template = this.prototype._template;
          if (template && !template.__polymerFinalized) {
            template.__polymerFinalized = true;
            const importPath = this.importPath;
            const baseURI = importPath ? resolveUrl(importPath) : '';
            // e.g. support `include="module-name"`, and ShadyCSS
            processElementStyles(this, template, is, baseURI);
            this.prototype._bindTemplate(template);
          }
        }

        /**
         * Provides a default implementation of the standard Custom Elements
         * `connectedCallback`.
         *
         * The default implementation enables the property effects system and
         * flushes any pending properties, and updates shimmed CSS properties
         * when using the ShadyCSS scoping/custom properties polyfill.
         *
         * @override
         * @suppress {missingProperties, invalidCasts} Super may or may not
         *     implement the callback
         * @return {void}
         */
        connectedCallback() {
          if (window.ShadyCSS && this._template) {
            window.ShadyCSS.styleElement(/** @type {!HTMLElement} */(this));
          }
          super.connectedCallback();
        }

        /**
         * Stamps the element template.
         *
         * @return {void}
         * @override
         */
        ready() {
          if (this._template) {
            this.root = this._stampTemplate(this._template);
            this.$ = this.root.$;
          }
          super.ready();
        }

        /**
         * Implements `PropertyEffects`'s `_readyClients` call. Attaches
         * element dom by calling `_attachDom` with the dom stamped from the
         * element's template via `_stampTemplate`. Note that this allows
         * client dom to be attached to the element prior to any observers
         * running.
         *
         * @return {void}
         * @override
         */
        _readyClients() {
          if (this._template) {
            this.root = this._attachDom(/** @type {StampedTemplate} */(this.root));
          }
          // The super._readyClients here sets the clients initialized flag.
          // We must wait to do this until after client dom is created/attached
          // so that this flag can be checked to prevent notifications fired
          // during this process from being handled before clients are ready.
          super._readyClients();
        }


        /**
         * Attaches an element's stamped dom to itself. By default,
         * this method creates a `shadowRoot` and adds the dom to it.
         * However, this method may be overridden to allow an element
         * to put its dom in another location.
         *
         * @override
         * @throws {Error}
         * @suppress {missingReturn}
         * @param {StampedTemplate} dom to attach to the element.
         * @return {ShadowRoot} node to which the dom has been attached.
         */
        _attachDom(dom) {
          const n = wrap(this);
          if (n.attachShadow) {
            if (dom) {
              if (!n.shadowRoot) {
                n.attachShadow({mode: 'open', shadyUpgradeFragment: dom});
                n.shadowRoot.appendChild(dom);
                // When `adoptedStyleSheets` is supported a stylesheet is made
                // available on the element constructor.
                if (this.constructor._styleSheet) {
                  n.shadowRoot.adoptedStyleSheets = [this.constructor._styleSheet];
                }
              }
              if (syncInitialRender && window.ShadyDOM) {
                window.ShadyDOM.flushInitial(n.shadowRoot);
              }
              return n.shadowRoot;
            }
            return null;
          } else {
            throw new Error('ShadowDOM not available. ' +
              // TODO(sorvell): move to compile-time conditional when supported
            'PolymerElement can create dom as children instead of in ' +
            'ShadowDOM by setting `this.root = this;\` before \`ready\`.');
          }
        }

        /**
         * When using the ShadyCSS scoping and custom property shim, causes all
         * shimmed styles in this element (and its subtree) to be updated
         * based on current custom property values.
         *
         * The optional parameter overrides inline custom property styles with an
         * object of properties where the keys are CSS properties, and the values
         * are strings.
         *
         * Example: `this.updateStyles({'--color': 'blue'})`
         *
         * These properties are retained unless a value of `null` is set.
         *
         * Note: This function does not support updating CSS mixins.
         * You can not dynamically change the value of an `@apply`.
         *
         * @override
         * @param {Object=} properties Bag of custom property key/values to
         *   apply to this element.
         * @return {void}
         * @suppress {invalidCasts}
         */
        updateStyles(properties) {
          if (window.ShadyCSS) {
            window.ShadyCSS.styleSubtree(/** @type {!HTMLElement} */(this), properties);
          }
        }

        /**
         * Rewrites a given URL relative to a base URL. The base URL defaults to
         * the original location of the document containing the `dom-module` for
         * this element. This method will return the same URL before and after
         * bundling.
         *
         * Note that this function performs no resolution for URLs that start
         * with `/` (absolute URLs) or `#` (hash identifiers).  For general purpose
         * URL resolution, use `window.URL`.
         *
         * @override
         * @param {string} url URL to resolve.
         * @param {string=} base Optional base URL to resolve against, defaults
         * to the element's `importPath`
         * @return {string} Rewritten URL relative to base
         */
        resolveUrl(url, base) {
          if (!base && this.importPath) {
            base = resolveUrl(this.importPath);
          }
          return resolveUrl(url, base);
        }

        /**
         * Overrides `PropertyEffects` to add map of dynamic functions on
         * template info, for consumption by `PropertyEffects` template binding
         * code. This map determines which method templates should have accessors
         * created for them.
         *
         * @param {!HTMLTemplateElement} template Template
         * @param {!TemplateInfo} templateInfo Template metadata for current template
         * @param {!NodeInfo} nodeInfo Node metadata for current template.
         * @return {boolean} .
         * @suppress {missingProperties} Interfaces in closure do not inherit statics, but classes do
         * @nocollapse
         */
        static _parseTemplateContent(template, templateInfo, nodeInfo) {
          templateInfo.dynamicFns = templateInfo.dynamicFns || this._properties;
          // TODO(https://github.com/google/closure-compiler/issues/3240):
          //     Change back to just super.methodCall()
          return polymerElementBase._parseTemplateContent.call(
            this, template, templateInfo, nodeInfo);
        }

        /**
         * Overrides `PropertyEffects` to warn on use of undeclared properties in
         * template.
         *
         * @param {Object} templateInfo Template metadata to add effect to
         * @param {string} prop Property that should trigger the effect
         * @param {Object=} effect Effect metadata object
         * @return {void}
         * @protected
         * @suppress {missingProperties} Interfaces in closure do not inherit statics, but classes do
         * @nocollapse
         */
        static _addTemplatePropertyEffect(templateInfo, prop, effect) {
          // Warn if properties are used in template without being declared.
          // Properties must be listed in `properties` to be included in
          // `observedAttributes` since CE V1 reads that at registration time, and
          // since we want to keep template parsing lazy, we can't automatically
          // add undeclared properties used in templates to `observedAttributes`.
          // The warning is only enabled in `legacyOptimizations` mode, since
          // we don't want to spam existing users who might have adopted the
          // shorthand when attribute deserialization is not important.
          if (legacyWarnings && !(prop in this._properties) &&
              // Methods used in templates with no dependencies (or only literal
              // dependencies) become accessors with template effects; ignore these
              !(effect.info.part.signature && effect.info.part.signature.static) &&
              // Warnings for bindings added to nested templates are handled by
              // templatizer so ignore both the host-to-template bindings
              // (`hostProp`) and TemplateInstance-to-child bindings
              // (`nestedTemplate`)
              !effect.info.part.hostProp && !templateInfo.nestedTemplate) {
            console.warn(`Property '${prop}' used in template but not declared in 'properties'; ` +
              `attribute will not be observed.`);
          }
          // TODO(https://github.com/google/closure-compiler/issues/3240):
          //     Change back to just super.methodCall()
          return polymerElementBase._addTemplatePropertyEffect.call(
            this, templateInfo, prop, effect);
        }

      }

      return PolymerElement;
    });

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    /**
     * Class representing a static string value which can be used to filter
     * strings by asseting that they have been created via this class. The
     * `value` property returns the string passed to the constructor.
     */
    class LiteralString {
      constructor(string) {
        /** @type {string} */
        this.value = string.toString();
      }
      /**
       * @return {string} LiteralString string value
       * @override
       */
      toString() {
        return this.value;
      }
    }

    /**
     * @param {*} value Object to stringify into HTML
     * @return {string} HTML stringified form of `obj`
     */
    function literalValue(value) {
      if (value instanceof LiteralString) {
        return /** @type {!LiteralString} */(value).value;
      } else {
        throw new Error(
            `non-literal value passed to Polymer's htmlLiteral function: ${value}`
        );
      }
    }

    /**
     * @param {*} value Object to stringify into HTML
     * @return {string} HTML stringified form of `obj`
     */
    function htmlValue(value) {
      if (value instanceof HTMLTemplateElement) {
        return /** @type {!HTMLTemplateElement } */(value).innerHTML;
      } else if (value instanceof LiteralString) {
        return literalValue(value);
      } else {
        throw new Error(
            `non-template value passed to Polymer's html function: ${value}`);
      }
    }

    /**
     * A template literal tag that creates an HTML <template> element from the
     * contents of the string.
     *
     * This allows you to write a Polymer Template in JavaScript.
     *
     * Templates can be composed by interpolating `HTMLTemplateElement`s in
     * expressions in the JavaScript template literal. The nested template's
     * `innerHTML` is included in the containing template.  The only other
     * values allowed in expressions are those returned from `htmlLiteral`
     * which ensures only literal values from JS source ever reach the HTML, to
     * guard against XSS risks.
     *
     * All other values are disallowed in expressions to help prevent XSS
     * attacks; however, `htmlLiteral` can be used to compose static
     * string values into templates. This is useful to compose strings into
     * places that do not accept html, like the css text of a `style`
     * element.
     *
     * Example:
     *
     *     static get template() {
     *       return html`
     *         <style>:host{ content:"..." }</style>
     *         <div class="shadowed">${this.partialTemplate}</div>
     *         ${super.template}
     *       `;
     *     }
     *     static get partialTemplate() { return html`<span>Partial!</span>`; }
     *
     * @param {!ITemplateArray} strings Constant parts of tagged template literal
     * @param {...*} values Variable parts of tagged template literal
     * @return {!HTMLTemplateElement} Constructed HTMLTemplateElement
     */
    const html = function html(strings, ...values) {
      const template = /** @type {!HTMLTemplateElement} */(document.createElement('template'));
      template.innerHTML = values.reduce((acc, v, idx) =>
          acc + htmlValue(v) + strings[idx + 1], strings[0]);
      return template;
    };

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    /**
     * Base class that provides the core API for Polymer's meta-programming
     * features including template stamping, data-binding, attribute deserialization,
     * and property change observation.
     *
     * @customElement
     * @polymer
     * @constructor
     * @implements {Polymer_ElementMixin}
     * @extends HTMLElement
     * @appliesMixin ElementMixin
     * @summary Custom element base class that provides the core API for Polymer's
     *   key meta-programming features including template stamping, data-binding,
     *   attribute deserialization, and property change observation
     */
    const PolymerElement = ElementMixin(HTMLElement);

    var commonjsGlobal = typeof globalThis !== 'undefined' ? globalThis : typeof window !== 'undefined' ? window : typeof global !== 'undefined' ? global : typeof self !== 'undefined' ? self : {};

    function createCommonjsModule(fn, basedir, module) {
    	return module = {
    	  path: basedir,
    	  exports: {},
    	  require: function (path, base) {
          return commonjsRequire(path, (base === undefined || base === null) ? module.path : base);
        }
    	}, fn(module, module.exports), module.exports;
    }

    function commonjsRequire () {
    	throw new Error('Dynamic requires are not currently supported by @rollup/plugin-commonjs');
    }

    var lodash = createCommonjsModule(function (module, exports) {
    (function() {

      /** Used as a safe reference for `undefined` in pre-ES5 environments. */
      var undefined$1;

      /** Used as the semantic version number. */
      var VERSION = '4.17.19';

      /** Used as the size to enable large array optimizations. */
      var LARGE_ARRAY_SIZE = 200;

      /** Error message constants. */
      var CORE_ERROR_TEXT = 'Unsupported core-js use. Try https://npms.io/search?q=ponyfill.',
          FUNC_ERROR_TEXT = 'Expected a function';

      /** Used to stand-in for `undefined` hash values. */
      var HASH_UNDEFINED = '__lodash_hash_undefined__';

      /** Used as the maximum memoize cache size. */
      var MAX_MEMOIZE_SIZE = 500;

      /** Used as the internal argument placeholder. */
      var PLACEHOLDER = '__lodash_placeholder__';

      /** Used to compose bitmasks for cloning. */
      var CLONE_DEEP_FLAG = 1,
          CLONE_FLAT_FLAG = 2,
          CLONE_SYMBOLS_FLAG = 4;

      /** Used to compose bitmasks for value comparisons. */
      var COMPARE_PARTIAL_FLAG = 1,
          COMPARE_UNORDERED_FLAG = 2;

      /** Used to compose bitmasks for function metadata. */
      var WRAP_BIND_FLAG = 1,
          WRAP_BIND_KEY_FLAG = 2,
          WRAP_CURRY_BOUND_FLAG = 4,
          WRAP_CURRY_FLAG = 8,
          WRAP_CURRY_RIGHT_FLAG = 16,
          WRAP_PARTIAL_FLAG = 32,
          WRAP_PARTIAL_RIGHT_FLAG = 64,
          WRAP_ARY_FLAG = 128,
          WRAP_REARG_FLAG = 256,
          WRAP_FLIP_FLAG = 512;

      /** Used as default options for `_.truncate`. */
      var DEFAULT_TRUNC_LENGTH = 30,
          DEFAULT_TRUNC_OMISSION = '...';

      /** Used to detect hot functions by number of calls within a span of milliseconds. */
      var HOT_COUNT = 800,
          HOT_SPAN = 16;

      /** Used to indicate the type of lazy iteratees. */
      var LAZY_FILTER_FLAG = 1,
          LAZY_MAP_FLAG = 2,
          LAZY_WHILE_FLAG = 3;

      /** Used as references for various `Number` constants. */
      var INFINITY = 1 / 0,
          MAX_SAFE_INTEGER = 9007199254740991,
          MAX_INTEGER = 1.7976931348623157e+308,
          NAN = 0 / 0;

      /** Used as references for the maximum length and index of an array. */
      var MAX_ARRAY_LENGTH = 4294967295,
          MAX_ARRAY_INDEX = MAX_ARRAY_LENGTH - 1,
          HALF_MAX_ARRAY_LENGTH = MAX_ARRAY_LENGTH >>> 1;

      /** Used to associate wrap methods with their bit flags. */
      var wrapFlags = [
        ['ary', WRAP_ARY_FLAG],
        ['bind', WRAP_BIND_FLAG],
        ['bindKey', WRAP_BIND_KEY_FLAG],
        ['curry', WRAP_CURRY_FLAG],
        ['curryRight', WRAP_CURRY_RIGHT_FLAG],
        ['flip', WRAP_FLIP_FLAG],
        ['partial', WRAP_PARTIAL_FLAG],
        ['partialRight', WRAP_PARTIAL_RIGHT_FLAG],
        ['rearg', WRAP_REARG_FLAG]
      ];

      /** `Object#toString` result references. */
      var argsTag = '[object Arguments]',
          arrayTag = '[object Array]',
          asyncTag = '[object AsyncFunction]',
          boolTag = '[object Boolean]',
          dateTag = '[object Date]',
          domExcTag = '[object DOMException]',
          errorTag = '[object Error]',
          funcTag = '[object Function]',
          genTag = '[object GeneratorFunction]',
          mapTag = '[object Map]',
          numberTag = '[object Number]',
          nullTag = '[object Null]',
          objectTag = '[object Object]',
          promiseTag = '[object Promise]',
          proxyTag = '[object Proxy]',
          regexpTag = '[object RegExp]',
          setTag = '[object Set]',
          stringTag = '[object String]',
          symbolTag = '[object Symbol]',
          undefinedTag = '[object Undefined]',
          weakMapTag = '[object WeakMap]',
          weakSetTag = '[object WeakSet]';

      var arrayBufferTag = '[object ArrayBuffer]',
          dataViewTag = '[object DataView]',
          float32Tag = '[object Float32Array]',
          float64Tag = '[object Float64Array]',
          int8Tag = '[object Int8Array]',
          int16Tag = '[object Int16Array]',
          int32Tag = '[object Int32Array]',
          uint8Tag = '[object Uint8Array]',
          uint8ClampedTag = '[object Uint8ClampedArray]',
          uint16Tag = '[object Uint16Array]',
          uint32Tag = '[object Uint32Array]';

      /** Used to match empty string literals in compiled template source. */
      var reEmptyStringLeading = /\b__p \+= '';/g,
          reEmptyStringMiddle = /\b(__p \+=) '' \+/g,
          reEmptyStringTrailing = /(__e\(.*?\)|\b__t\)) \+\n'';/g;

      /** Used to match HTML entities and HTML characters. */
      var reEscapedHtml = /&(?:amp|lt|gt|quot|#39);/g,
          reUnescapedHtml = /[&<>"']/g,
          reHasEscapedHtml = RegExp(reEscapedHtml.source),
          reHasUnescapedHtml = RegExp(reUnescapedHtml.source);

      /** Used to match template delimiters. */
      var reEscape = /<%-([\s\S]+?)%>/g,
          reEvaluate = /<%([\s\S]+?)%>/g,
          reInterpolate = /<%=([\s\S]+?)%>/g;

      /** Used to match property names within property paths. */
      var reIsDeepProp = /\.|\[(?:[^[\]]*|(["'])(?:(?!\1)[^\\]|\\.)*?\1)\]/,
          reIsPlainProp = /^\w*$/,
          rePropName = /[^.[\]]+|\[(?:(-?\d+(?:\.\d+)?)|(["'])((?:(?!\2)[^\\]|\\.)*?)\2)\]|(?=(?:\.|\[\])(?:\.|\[\]|$))/g;

      /**
       * Used to match `RegExp`
       * [syntax characters](http://ecma-international.org/ecma-262/7.0/#sec-patterns).
       */
      var reRegExpChar = /[\\^$.*+?()[\]{}|]/g,
          reHasRegExpChar = RegExp(reRegExpChar.source);

      /** Used to match leading and trailing whitespace. */
      var reTrim = /^\s+|\s+$/g,
          reTrimStart = /^\s+/,
          reTrimEnd = /\s+$/;

      /** Used to match wrap detail comments. */
      var reWrapComment = /\{(?:\n\/\* \[wrapped with .+\] \*\/)?\n?/,
          reWrapDetails = /\{\n\/\* \[wrapped with (.+)\] \*/,
          reSplitDetails = /,? & /;

      /** Used to match words composed of alphanumeric characters. */
      var reAsciiWord = /[^\x00-\x2f\x3a-\x40\x5b-\x60\x7b-\x7f]+/g;

      /** Used to match backslashes in property paths. */
      var reEscapeChar = /\\(\\)?/g;

      /**
       * Used to match
       * [ES template delimiters](http://ecma-international.org/ecma-262/7.0/#sec-template-literal-lexical-components).
       */
      var reEsTemplate = /\$\{([^\\}]*(?:\\.[^\\}]*)*)\}/g;

      /** Used to match `RegExp` flags from their coerced string values. */
      var reFlags = /\w*$/;

      /** Used to detect bad signed hexadecimal string values. */
      var reIsBadHex = /^[-+]0x[0-9a-f]+$/i;

      /** Used to detect binary string values. */
      var reIsBinary = /^0b[01]+$/i;

      /** Used to detect host constructors (Safari). */
      var reIsHostCtor = /^\[object .+?Constructor\]$/;

      /** Used to detect octal string values. */
      var reIsOctal = /^0o[0-7]+$/i;

      /** Used to detect unsigned integer values. */
      var reIsUint = /^(?:0|[1-9]\d*)$/;

      /** Used to match Latin Unicode letters (excluding mathematical operators). */
      var reLatin = /[\xc0-\xd6\xd8-\xf6\xf8-\xff\u0100-\u017f]/g;

      /** Used to ensure capturing order of template delimiters. */
      var reNoMatch = /($^)/;

      /** Used to match unescaped characters in compiled string literals. */
      var reUnescapedString = /['\n\r\u2028\u2029\\]/g;

      /** Used to compose unicode character classes. */
      var rsAstralRange = '\\ud800-\\udfff',
          rsComboMarksRange = '\\u0300-\\u036f',
          reComboHalfMarksRange = '\\ufe20-\\ufe2f',
          rsComboSymbolsRange = '\\u20d0-\\u20ff',
          rsComboRange = rsComboMarksRange + reComboHalfMarksRange + rsComboSymbolsRange,
          rsDingbatRange = '\\u2700-\\u27bf',
          rsLowerRange = 'a-z\\xdf-\\xf6\\xf8-\\xff',
          rsMathOpRange = '\\xac\\xb1\\xd7\\xf7',
          rsNonCharRange = '\\x00-\\x2f\\x3a-\\x40\\x5b-\\x60\\x7b-\\xbf',
          rsPunctuationRange = '\\u2000-\\u206f',
          rsSpaceRange = ' \\t\\x0b\\f\\xa0\\ufeff\\n\\r\\u2028\\u2029\\u1680\\u180e\\u2000\\u2001\\u2002\\u2003\\u2004\\u2005\\u2006\\u2007\\u2008\\u2009\\u200a\\u202f\\u205f\\u3000',
          rsUpperRange = 'A-Z\\xc0-\\xd6\\xd8-\\xde',
          rsVarRange = '\\ufe0e\\ufe0f',
          rsBreakRange = rsMathOpRange + rsNonCharRange + rsPunctuationRange + rsSpaceRange;

      /** Used to compose unicode capture groups. */
      var rsApos = "['\u2019]",
          rsAstral = '[' + rsAstralRange + ']',
          rsBreak = '[' + rsBreakRange + ']',
          rsCombo = '[' + rsComboRange + ']',
          rsDigits = '\\d+',
          rsDingbat = '[' + rsDingbatRange + ']',
          rsLower = '[' + rsLowerRange + ']',
          rsMisc = '[^' + rsAstralRange + rsBreakRange + rsDigits + rsDingbatRange + rsLowerRange + rsUpperRange + ']',
          rsFitz = '\\ud83c[\\udffb-\\udfff]',
          rsModifier = '(?:' + rsCombo + '|' + rsFitz + ')',
          rsNonAstral = '[^' + rsAstralRange + ']',
          rsRegional = '(?:\\ud83c[\\udde6-\\uddff]){2}',
          rsSurrPair = '[\\ud800-\\udbff][\\udc00-\\udfff]',
          rsUpper = '[' + rsUpperRange + ']',
          rsZWJ = '\\u200d';

      /** Used to compose unicode regexes. */
      var rsMiscLower = '(?:' + rsLower + '|' + rsMisc + ')',
          rsMiscUpper = '(?:' + rsUpper + '|' + rsMisc + ')',
          rsOptContrLower = '(?:' + rsApos + '(?:d|ll|m|re|s|t|ve))?',
          rsOptContrUpper = '(?:' + rsApos + '(?:D|LL|M|RE|S|T|VE))?',
          reOptMod = rsModifier + '?',
          rsOptVar = '[' + rsVarRange + ']?',
          rsOptJoin = '(?:' + rsZWJ + '(?:' + [rsNonAstral, rsRegional, rsSurrPair].join('|') + ')' + rsOptVar + reOptMod + ')*',
          rsOrdLower = '\\d*(?:1st|2nd|3rd|(?![123])\\dth)(?=\\b|[A-Z_])',
          rsOrdUpper = '\\d*(?:1ST|2ND|3RD|(?![123])\\dTH)(?=\\b|[a-z_])',
          rsSeq = rsOptVar + reOptMod + rsOptJoin,
          rsEmoji = '(?:' + [rsDingbat, rsRegional, rsSurrPair].join('|') + ')' + rsSeq,
          rsSymbol = '(?:' + [rsNonAstral + rsCombo + '?', rsCombo, rsRegional, rsSurrPair, rsAstral].join('|') + ')';

      /** Used to match apostrophes. */
      var reApos = RegExp(rsApos, 'g');

      /**
       * Used to match [combining diacritical marks](https://en.wikipedia.org/wiki/Combining_Diacritical_Marks) and
       * [combining diacritical marks for symbols](https://en.wikipedia.org/wiki/Combining_Diacritical_Marks_for_Symbols).
       */
      var reComboMark = RegExp(rsCombo, 'g');

      /** Used to match [string symbols](https://mathiasbynens.be/notes/javascript-unicode). */
      var reUnicode = RegExp(rsFitz + '(?=' + rsFitz + ')|' + rsSymbol + rsSeq, 'g');

      /** Used to match complex or compound words. */
      var reUnicodeWord = RegExp([
        rsUpper + '?' + rsLower + '+' + rsOptContrLower + '(?=' + [rsBreak, rsUpper, '$'].join('|') + ')',
        rsMiscUpper + '+' + rsOptContrUpper + '(?=' + [rsBreak, rsUpper + rsMiscLower, '$'].join('|') + ')',
        rsUpper + '?' + rsMiscLower + '+' + rsOptContrLower,
        rsUpper + '+' + rsOptContrUpper,
        rsOrdUpper,
        rsOrdLower,
        rsDigits,
        rsEmoji
      ].join('|'), 'g');

      /** Used to detect strings with [zero-width joiners or code points from the astral planes](http://eev.ee/blog/2015/09/12/dark-corners-of-unicode/). */
      var reHasUnicode = RegExp('[' + rsZWJ + rsAstralRange  + rsComboRange + rsVarRange + ']');

      /** Used to detect strings that need a more robust regexp to match words. */
      var reHasUnicodeWord = /[a-z][A-Z]|[A-Z]{2}[a-z]|[0-9][a-zA-Z]|[a-zA-Z][0-9]|[^a-zA-Z0-9 ]/;

      /** Used to assign default `context` object properties. */
      var contextProps = [
        'Array', 'Buffer', 'DataView', 'Date', 'Error', 'Float32Array', 'Float64Array',
        'Function', 'Int8Array', 'Int16Array', 'Int32Array', 'Map', 'Math', 'Object',
        'Promise', 'RegExp', 'Set', 'String', 'Symbol', 'TypeError', 'Uint8Array',
        'Uint8ClampedArray', 'Uint16Array', 'Uint32Array', 'WeakMap',
        '_', 'clearTimeout', 'isFinite', 'parseInt', 'setTimeout'
      ];

      /** Used to make template sourceURLs easier to identify. */
      var templateCounter = -1;

      /** Used to identify `toStringTag` values of typed arrays. */
      var typedArrayTags = {};
      typedArrayTags[float32Tag] = typedArrayTags[float64Tag] =
      typedArrayTags[int8Tag] = typedArrayTags[int16Tag] =
      typedArrayTags[int32Tag] = typedArrayTags[uint8Tag] =
      typedArrayTags[uint8ClampedTag] = typedArrayTags[uint16Tag] =
      typedArrayTags[uint32Tag] = true;
      typedArrayTags[argsTag] = typedArrayTags[arrayTag] =
      typedArrayTags[arrayBufferTag] = typedArrayTags[boolTag] =
      typedArrayTags[dataViewTag] = typedArrayTags[dateTag] =
      typedArrayTags[errorTag] = typedArrayTags[funcTag] =
      typedArrayTags[mapTag] = typedArrayTags[numberTag] =
      typedArrayTags[objectTag] = typedArrayTags[regexpTag] =
      typedArrayTags[setTag] = typedArrayTags[stringTag] =
      typedArrayTags[weakMapTag] = false;

      /** Used to identify `toStringTag` values supported by `_.clone`. */
      var cloneableTags = {};
      cloneableTags[argsTag] = cloneableTags[arrayTag] =
      cloneableTags[arrayBufferTag] = cloneableTags[dataViewTag] =
      cloneableTags[boolTag] = cloneableTags[dateTag] =
      cloneableTags[float32Tag] = cloneableTags[float64Tag] =
      cloneableTags[int8Tag] = cloneableTags[int16Tag] =
      cloneableTags[int32Tag] = cloneableTags[mapTag] =
      cloneableTags[numberTag] = cloneableTags[objectTag] =
      cloneableTags[regexpTag] = cloneableTags[setTag] =
      cloneableTags[stringTag] = cloneableTags[symbolTag] =
      cloneableTags[uint8Tag] = cloneableTags[uint8ClampedTag] =
      cloneableTags[uint16Tag] = cloneableTags[uint32Tag] = true;
      cloneableTags[errorTag] = cloneableTags[funcTag] =
      cloneableTags[weakMapTag] = false;

      /** Used to map Latin Unicode letters to basic Latin letters. */
      var deburredLetters = {
        // Latin-1 Supplement block.
        '\xc0': 'A',  '\xc1': 'A', '\xc2': 'A', '\xc3': 'A', '\xc4': 'A', '\xc5': 'A',
        '\xe0': 'a',  '\xe1': 'a', '\xe2': 'a', '\xe3': 'a', '\xe4': 'a', '\xe5': 'a',
        '\xc7': 'C',  '\xe7': 'c',
        '\xd0': 'D',  '\xf0': 'd',
        '\xc8': 'E',  '\xc9': 'E', '\xca': 'E', '\xcb': 'E',
        '\xe8': 'e',  '\xe9': 'e', '\xea': 'e', '\xeb': 'e',
        '\xcc': 'I',  '\xcd': 'I', '\xce': 'I', '\xcf': 'I',
        '\xec': 'i',  '\xed': 'i', '\xee': 'i', '\xef': 'i',
        '\xd1': 'N',  '\xf1': 'n',
        '\xd2': 'O',  '\xd3': 'O', '\xd4': 'O', '\xd5': 'O', '\xd6': 'O', '\xd8': 'O',
        '\xf2': 'o',  '\xf3': 'o', '\xf4': 'o', '\xf5': 'o', '\xf6': 'o', '\xf8': 'o',
        '\xd9': 'U',  '\xda': 'U', '\xdb': 'U', '\xdc': 'U',
        '\xf9': 'u',  '\xfa': 'u', '\xfb': 'u', '\xfc': 'u',
        '\xdd': 'Y',  '\xfd': 'y', '\xff': 'y',
        '\xc6': 'Ae', '\xe6': 'ae',
        '\xde': 'Th', '\xfe': 'th',
        '\xdf': 'ss',
        // Latin Extended-A block.
        '\u0100': 'A',  '\u0102': 'A', '\u0104': 'A',
        '\u0101': 'a',  '\u0103': 'a', '\u0105': 'a',
        '\u0106': 'C',  '\u0108': 'C', '\u010a': 'C', '\u010c': 'C',
        '\u0107': 'c',  '\u0109': 'c', '\u010b': 'c', '\u010d': 'c',
        '\u010e': 'D',  '\u0110': 'D', '\u010f': 'd', '\u0111': 'd',
        '\u0112': 'E',  '\u0114': 'E', '\u0116': 'E', '\u0118': 'E', '\u011a': 'E',
        '\u0113': 'e',  '\u0115': 'e', '\u0117': 'e', '\u0119': 'e', '\u011b': 'e',
        '\u011c': 'G',  '\u011e': 'G', '\u0120': 'G', '\u0122': 'G',
        '\u011d': 'g',  '\u011f': 'g', '\u0121': 'g', '\u0123': 'g',
        '\u0124': 'H',  '\u0126': 'H', '\u0125': 'h', '\u0127': 'h',
        '\u0128': 'I',  '\u012a': 'I', '\u012c': 'I', '\u012e': 'I', '\u0130': 'I',
        '\u0129': 'i',  '\u012b': 'i', '\u012d': 'i', '\u012f': 'i', '\u0131': 'i',
        '\u0134': 'J',  '\u0135': 'j',
        '\u0136': 'K',  '\u0137': 'k', '\u0138': 'k',
        '\u0139': 'L',  '\u013b': 'L', '\u013d': 'L', '\u013f': 'L', '\u0141': 'L',
        '\u013a': 'l',  '\u013c': 'l', '\u013e': 'l', '\u0140': 'l', '\u0142': 'l',
        '\u0143': 'N',  '\u0145': 'N', '\u0147': 'N', '\u014a': 'N',
        '\u0144': 'n',  '\u0146': 'n', '\u0148': 'n', '\u014b': 'n',
        '\u014c': 'O',  '\u014e': 'O', '\u0150': 'O',
        '\u014d': 'o',  '\u014f': 'o', '\u0151': 'o',
        '\u0154': 'R',  '\u0156': 'R', '\u0158': 'R',
        '\u0155': 'r',  '\u0157': 'r', '\u0159': 'r',
        '\u015a': 'S',  '\u015c': 'S', '\u015e': 'S', '\u0160': 'S',
        '\u015b': 's',  '\u015d': 's', '\u015f': 's', '\u0161': 's',
        '\u0162': 'T',  '\u0164': 'T', '\u0166': 'T',
        '\u0163': 't',  '\u0165': 't', '\u0167': 't',
        '\u0168': 'U',  '\u016a': 'U', '\u016c': 'U', '\u016e': 'U', '\u0170': 'U', '\u0172': 'U',
        '\u0169': 'u',  '\u016b': 'u', '\u016d': 'u', '\u016f': 'u', '\u0171': 'u', '\u0173': 'u',
        '\u0174': 'W',  '\u0175': 'w',
        '\u0176': 'Y',  '\u0177': 'y', '\u0178': 'Y',
        '\u0179': 'Z',  '\u017b': 'Z', '\u017d': 'Z',
        '\u017a': 'z',  '\u017c': 'z', '\u017e': 'z',
        '\u0132': 'IJ', '\u0133': 'ij',
        '\u0152': 'Oe', '\u0153': 'oe',
        '\u0149': "'n", '\u017f': 's'
      };

      /** Used to map characters to HTML entities. */
      var htmlEscapes = {
        '&': '&amp;',
        '<': '&lt;',
        '>': '&gt;',
        '"': '&quot;',
        "'": '&#39;'
      };

      /** Used to map HTML entities to characters. */
      var htmlUnescapes = {
        '&amp;': '&',
        '&lt;': '<',
        '&gt;': '>',
        '&quot;': '"',
        '&#39;': "'"
      };

      /** Used to escape characters for inclusion in compiled string literals. */
      var stringEscapes = {
        '\\': '\\',
        "'": "'",
        '\n': 'n',
        '\r': 'r',
        '\u2028': 'u2028',
        '\u2029': 'u2029'
      };

      /** Built-in method references without a dependency on `root`. */
      var freeParseFloat = parseFloat,
          freeParseInt = parseInt;

      /** Detect free variable `global` from Node.js. */
      var freeGlobal = typeof commonjsGlobal == 'object' && commonjsGlobal && commonjsGlobal.Object === Object && commonjsGlobal;

      /** Detect free variable `self`. */
      var freeSelf = typeof self == 'object' && self && self.Object === Object && self;

      /** Used as a reference to the global object. */
      var root = freeGlobal || freeSelf || Function('return this')();

      /** Detect free variable `exports`. */
      var freeExports =  exports && !exports.nodeType && exports;

      /** Detect free variable `module`. */
      var freeModule = freeExports && 'object' == 'object' && module && !module.nodeType && module;

      /** Detect the popular CommonJS extension `module.exports`. */
      var moduleExports = freeModule && freeModule.exports === freeExports;

      /** Detect free variable `process` from Node.js. */
      var freeProcess = moduleExports && freeGlobal.process;

      /** Used to access faster Node.js helpers. */
      var nodeUtil = (function() {
        try {
          // Use `util.types` for Node.js 10+.
          var types = freeModule && freeModule.require && freeModule.require('util').types;

          if (types) {
            return types;
          }

          // Legacy `process.binding('util')` for Node.js < 10.
          return freeProcess && freeProcess.binding && freeProcess.binding('util');
        } catch (e) {}
      }());

      /* Node.js helper references. */
      var nodeIsArrayBuffer = nodeUtil && nodeUtil.isArrayBuffer,
          nodeIsDate = nodeUtil && nodeUtil.isDate,
          nodeIsMap = nodeUtil && nodeUtil.isMap,
          nodeIsRegExp = nodeUtil && nodeUtil.isRegExp,
          nodeIsSet = nodeUtil && nodeUtil.isSet,
          nodeIsTypedArray = nodeUtil && nodeUtil.isTypedArray;

      /*--------------------------------------------------------------------------*/

      /**
       * A faster alternative to `Function#apply`, this function invokes `func`
       * with the `this` binding of `thisArg` and the arguments of `args`.
       *
       * @private
       * @param {Function} func The function to invoke.
       * @param {*} thisArg The `this` binding of `func`.
       * @param {Array} args The arguments to invoke `func` with.
       * @returns {*} Returns the result of `func`.
       */
      function apply(func, thisArg, args) {
        switch (args.length) {
          case 0: return func.call(thisArg);
          case 1: return func.call(thisArg, args[0]);
          case 2: return func.call(thisArg, args[0], args[1]);
          case 3: return func.call(thisArg, args[0], args[1], args[2]);
        }
        return func.apply(thisArg, args);
      }

      /**
       * A specialized version of `baseAggregator` for arrays.
       *
       * @private
       * @param {Array} [array] The array to iterate over.
       * @param {Function} setter The function to set `accumulator` values.
       * @param {Function} iteratee The iteratee to transform keys.
       * @param {Object} accumulator The initial aggregated object.
       * @returns {Function} Returns `accumulator`.
       */
      function arrayAggregator(array, setter, iteratee, accumulator) {
        var index = -1,
            length = array == null ? 0 : array.length;

        while (++index < length) {
          var value = array[index];
          setter(accumulator, value, iteratee(value), array);
        }
        return accumulator;
      }

      /**
       * A specialized version of `_.forEach` for arrays without support for
       * iteratee shorthands.
       *
       * @private
       * @param {Array} [array] The array to iterate over.
       * @param {Function} iteratee The function invoked per iteration.
       * @returns {Array} Returns `array`.
       */
      function arrayEach(array, iteratee) {
        var index = -1,
            length = array == null ? 0 : array.length;

        while (++index < length) {
          if (iteratee(array[index], index, array) === false) {
            break;
          }
        }
        return array;
      }

      /**
       * A specialized version of `_.forEachRight` for arrays without support for
       * iteratee shorthands.
       *
       * @private
       * @param {Array} [array] The array to iterate over.
       * @param {Function} iteratee The function invoked per iteration.
       * @returns {Array} Returns `array`.
       */
      function arrayEachRight(array, iteratee) {
        var length = array == null ? 0 : array.length;

        while (length--) {
          if (iteratee(array[length], length, array) === false) {
            break;
          }
        }
        return array;
      }

      /**
       * A specialized version of `_.every` for arrays without support for
       * iteratee shorthands.
       *
       * @private
       * @param {Array} [array] The array to iterate over.
       * @param {Function} predicate The function invoked per iteration.
       * @returns {boolean} Returns `true` if all elements pass the predicate check,
       *  else `false`.
       */
      function arrayEvery(array, predicate) {
        var index = -1,
            length = array == null ? 0 : array.length;

        while (++index < length) {
          if (!predicate(array[index], index, array)) {
            return false;
          }
        }
        return true;
      }

      /**
       * A specialized version of `_.filter` for arrays without support for
       * iteratee shorthands.
       *
       * @private
       * @param {Array} [array] The array to iterate over.
       * @param {Function} predicate The function invoked per iteration.
       * @returns {Array} Returns the new filtered array.
       */
      function arrayFilter(array, predicate) {
        var index = -1,
            length = array == null ? 0 : array.length,
            resIndex = 0,
            result = [];

        while (++index < length) {
          var value = array[index];
          if (predicate(value, index, array)) {
            result[resIndex++] = value;
          }
        }
        return result;
      }

      /**
       * A specialized version of `_.includes` for arrays without support for
       * specifying an index to search from.
       *
       * @private
       * @param {Array} [array] The array to inspect.
       * @param {*} target The value to search for.
       * @returns {boolean} Returns `true` if `target` is found, else `false`.
       */
      function arrayIncludes(array, value) {
        var length = array == null ? 0 : array.length;
        return !!length && baseIndexOf(array, value, 0) > -1;
      }

      /**
       * This function is like `arrayIncludes` except that it accepts a comparator.
       *
       * @private
       * @param {Array} [array] The array to inspect.
       * @param {*} target The value to search for.
       * @param {Function} comparator The comparator invoked per element.
       * @returns {boolean} Returns `true` if `target` is found, else `false`.
       */
      function arrayIncludesWith(array, value, comparator) {
        var index = -1,
            length = array == null ? 0 : array.length;

        while (++index < length) {
          if (comparator(value, array[index])) {
            return true;
          }
        }
        return false;
      }

      /**
       * A specialized version of `_.map` for arrays without support for iteratee
       * shorthands.
       *
       * @private
       * @param {Array} [array] The array to iterate over.
       * @param {Function} iteratee The function invoked per iteration.
       * @returns {Array} Returns the new mapped array.
       */
      function arrayMap(array, iteratee) {
        var index = -1,
            length = array == null ? 0 : array.length,
            result = Array(length);

        while (++index < length) {
          result[index] = iteratee(array[index], index, array);
        }
        return result;
      }

      /**
       * Appends the elements of `values` to `array`.
       *
       * @private
       * @param {Array} array The array to modify.
       * @param {Array} values The values to append.
       * @returns {Array} Returns `array`.
       */
      function arrayPush(array, values) {
        var index = -1,
            length = values.length,
            offset = array.length;

        while (++index < length) {
          array[offset + index] = values[index];
        }
        return array;
      }

      /**
       * A specialized version of `_.reduce` for arrays without support for
       * iteratee shorthands.
       *
       * @private
       * @param {Array} [array] The array to iterate over.
       * @param {Function} iteratee The function invoked per iteration.
       * @param {*} [accumulator] The initial value.
       * @param {boolean} [initAccum] Specify using the first element of `array` as
       *  the initial value.
       * @returns {*} Returns the accumulated value.
       */
      function arrayReduce(array, iteratee, accumulator, initAccum) {
        var index = -1,
            length = array == null ? 0 : array.length;

        if (initAccum && length) {
          accumulator = array[++index];
        }
        while (++index < length) {
          accumulator = iteratee(accumulator, array[index], index, array);
        }
        return accumulator;
      }

      /**
       * A specialized version of `_.reduceRight` for arrays without support for
       * iteratee shorthands.
       *
       * @private
       * @param {Array} [array] The array to iterate over.
       * @param {Function} iteratee The function invoked per iteration.
       * @param {*} [accumulator] The initial value.
       * @param {boolean} [initAccum] Specify using the last element of `array` as
       *  the initial value.
       * @returns {*} Returns the accumulated value.
       */
      function arrayReduceRight(array, iteratee, accumulator, initAccum) {
        var length = array == null ? 0 : array.length;
        if (initAccum && length) {
          accumulator = array[--length];
        }
        while (length--) {
          accumulator = iteratee(accumulator, array[length], length, array);
        }
        return accumulator;
      }

      /**
       * A specialized version of `_.some` for arrays without support for iteratee
       * shorthands.
       *
       * @private
       * @param {Array} [array] The array to iterate over.
       * @param {Function} predicate The function invoked per iteration.
       * @returns {boolean} Returns `true` if any element passes the predicate check,
       *  else `false`.
       */
      function arraySome(array, predicate) {
        var index = -1,
            length = array == null ? 0 : array.length;

        while (++index < length) {
          if (predicate(array[index], index, array)) {
            return true;
          }
        }
        return false;
      }

      /**
       * Gets the size of an ASCII `string`.
       *
       * @private
       * @param {string} string The string inspect.
       * @returns {number} Returns the string size.
       */
      var asciiSize = baseProperty('length');

      /**
       * Converts an ASCII `string` to an array.
       *
       * @private
       * @param {string} string The string to convert.
       * @returns {Array} Returns the converted array.
       */
      function asciiToArray(string) {
        return string.split('');
      }

      /**
       * Splits an ASCII `string` into an array of its words.
       *
       * @private
       * @param {string} The string to inspect.
       * @returns {Array} Returns the words of `string`.
       */
      function asciiWords(string) {
        return string.match(reAsciiWord) || [];
      }

      /**
       * The base implementation of methods like `_.findKey` and `_.findLastKey`,
       * without support for iteratee shorthands, which iterates over `collection`
       * using `eachFunc`.
       *
       * @private
       * @param {Array|Object} collection The collection to inspect.
       * @param {Function} predicate The function invoked per iteration.
       * @param {Function} eachFunc The function to iterate over `collection`.
       * @returns {*} Returns the found element or its key, else `undefined`.
       */
      function baseFindKey(collection, predicate, eachFunc) {
        var result;
        eachFunc(collection, function(value, key, collection) {
          if (predicate(value, key, collection)) {
            result = key;
            return false;
          }
        });
        return result;
      }

      /**
       * The base implementation of `_.findIndex` and `_.findLastIndex` without
       * support for iteratee shorthands.
       *
       * @private
       * @param {Array} array The array to inspect.
       * @param {Function} predicate The function invoked per iteration.
       * @param {number} fromIndex The index to search from.
       * @param {boolean} [fromRight] Specify iterating from right to left.
       * @returns {number} Returns the index of the matched value, else `-1`.
       */
      function baseFindIndex(array, predicate, fromIndex, fromRight) {
        var length = array.length,
            index = fromIndex + (fromRight ? 1 : -1);

        while ((fromRight ? index-- : ++index < length)) {
          if (predicate(array[index], index, array)) {
            return index;
          }
        }
        return -1;
      }

      /**
       * The base implementation of `_.indexOf` without `fromIndex` bounds checks.
       *
       * @private
       * @param {Array} array The array to inspect.
       * @param {*} value The value to search for.
       * @param {number} fromIndex The index to search from.
       * @returns {number} Returns the index of the matched value, else `-1`.
       */
      function baseIndexOf(array, value, fromIndex) {
        return value === value
          ? strictIndexOf(array, value, fromIndex)
          : baseFindIndex(array, baseIsNaN, fromIndex);
      }

      /**
       * This function is like `baseIndexOf` except that it accepts a comparator.
       *
       * @private
       * @param {Array} array The array to inspect.
       * @param {*} value The value to search for.
       * @param {number} fromIndex The index to search from.
       * @param {Function} comparator The comparator invoked per element.
       * @returns {number} Returns the index of the matched value, else `-1`.
       */
      function baseIndexOfWith(array, value, fromIndex, comparator) {
        var index = fromIndex - 1,
            length = array.length;

        while (++index < length) {
          if (comparator(array[index], value)) {
            return index;
          }
        }
        return -1;
      }

      /**
       * The base implementation of `_.isNaN` without support for number objects.
       *
       * @private
       * @param {*} value The value to check.
       * @returns {boolean} Returns `true` if `value` is `NaN`, else `false`.
       */
      function baseIsNaN(value) {
        return value !== value;
      }

      /**
       * The base implementation of `_.mean` and `_.meanBy` without support for
       * iteratee shorthands.
       *
       * @private
       * @param {Array} array The array to iterate over.
       * @param {Function} iteratee The function invoked per iteration.
       * @returns {number} Returns the mean.
       */
      function baseMean(array, iteratee) {
        var length = array == null ? 0 : array.length;
        return length ? (baseSum(array, iteratee) / length) : NAN;
      }

      /**
       * The base implementation of `_.property` without support for deep paths.
       *
       * @private
       * @param {string} key The key of the property to get.
       * @returns {Function} Returns the new accessor function.
       */
      function baseProperty(key) {
        return function(object) {
          return object == null ? undefined$1 : object[key];
        };
      }

      /**
       * The base implementation of `_.propertyOf` without support for deep paths.
       *
       * @private
       * @param {Object} object The object to query.
       * @returns {Function} Returns the new accessor function.
       */
      function basePropertyOf(object) {
        return function(key) {
          return object == null ? undefined$1 : object[key];
        };
      }

      /**
       * The base implementation of `_.reduce` and `_.reduceRight`, without support
       * for iteratee shorthands, which iterates over `collection` using `eachFunc`.
       *
       * @private
       * @param {Array|Object} collection The collection to iterate over.
       * @param {Function} iteratee The function invoked per iteration.
       * @param {*} accumulator The initial value.
       * @param {boolean} initAccum Specify using the first or last element of
       *  `collection` as the initial value.
       * @param {Function} eachFunc The function to iterate over `collection`.
       * @returns {*} Returns the accumulated value.
       */
      function baseReduce(collection, iteratee, accumulator, initAccum, eachFunc) {
        eachFunc(collection, function(value, index, collection) {
          accumulator = initAccum
            ? (initAccum = false, value)
            : iteratee(accumulator, value, index, collection);
        });
        return accumulator;
      }

      /**
       * The base implementation of `_.sortBy` which uses `comparer` to define the
       * sort order of `array` and replaces criteria objects with their corresponding
       * values.
       *
       * @private
       * @param {Array} array The array to sort.
       * @param {Function} comparer The function to define sort order.
       * @returns {Array} Returns `array`.
       */
      function baseSortBy(array, comparer) {
        var length = array.length;

        array.sort(comparer);
        while (length--) {
          array[length] = array[length].value;
        }
        return array;
      }

      /**
       * The base implementation of `_.sum` and `_.sumBy` without support for
       * iteratee shorthands.
       *
       * @private
       * @param {Array} array The array to iterate over.
       * @param {Function} iteratee The function invoked per iteration.
       * @returns {number} Returns the sum.
       */
      function baseSum(array, iteratee) {
        var result,
            index = -1,
            length = array.length;

        while (++index < length) {
          var current = iteratee(array[index]);
          if (current !== undefined$1) {
            result = result === undefined$1 ? current : (result + current);
          }
        }
        return result;
      }

      /**
       * The base implementation of `_.times` without support for iteratee shorthands
       * or max array length checks.
       *
       * @private
       * @param {number} n The number of times to invoke `iteratee`.
       * @param {Function} iteratee The function invoked per iteration.
       * @returns {Array} Returns the array of results.
       */
      function baseTimes(n, iteratee) {
        var index = -1,
            result = Array(n);

        while (++index < n) {
          result[index] = iteratee(index);
        }
        return result;
      }

      /**
       * The base implementation of `_.toPairs` and `_.toPairsIn` which creates an array
       * of key-value pairs for `object` corresponding to the property names of `props`.
       *
       * @private
       * @param {Object} object The object to query.
       * @param {Array} props The property names to get values for.
       * @returns {Object} Returns the key-value pairs.
       */
      function baseToPairs(object, props) {
        return arrayMap(props, function(key) {
          return [key, object[key]];
        });
      }

      /**
       * The base implementation of `_.unary` without support for storing metadata.
       *
       * @private
       * @param {Function} func The function to cap arguments for.
       * @returns {Function} Returns the new capped function.
       */
      function baseUnary(func) {
        return function(value) {
          return func(value);
        };
      }

      /**
       * The base implementation of `_.values` and `_.valuesIn` which creates an
       * array of `object` property values corresponding to the property names
       * of `props`.
       *
       * @private
       * @param {Object} object The object to query.
       * @param {Array} props The property names to get values for.
       * @returns {Object} Returns the array of property values.
       */
      function baseValues(object, props) {
        return arrayMap(props, function(key) {
          return object[key];
        });
      }

      /**
       * Checks if a `cache` value for `key` exists.
       *
       * @private
       * @param {Object} cache The cache to query.
       * @param {string} key The key of the entry to check.
       * @returns {boolean} Returns `true` if an entry for `key` exists, else `false`.
       */
      function cacheHas(cache, key) {
        return cache.has(key);
      }

      /**
       * Used by `_.trim` and `_.trimStart` to get the index of the first string symbol
       * that is not found in the character symbols.
       *
       * @private
       * @param {Array} strSymbols The string symbols to inspect.
       * @param {Array} chrSymbols The character symbols to find.
       * @returns {number} Returns the index of the first unmatched string symbol.
       */
      function charsStartIndex(strSymbols, chrSymbols) {
        var index = -1,
            length = strSymbols.length;

        while (++index < length && baseIndexOf(chrSymbols, strSymbols[index], 0) > -1) {}
        return index;
      }

      /**
       * Used by `_.trim` and `_.trimEnd` to get the index of the last string symbol
       * that is not found in the character symbols.
       *
       * @private
       * @param {Array} strSymbols The string symbols to inspect.
       * @param {Array} chrSymbols The character symbols to find.
       * @returns {number} Returns the index of the last unmatched string symbol.
       */
      function charsEndIndex(strSymbols, chrSymbols) {
        var index = strSymbols.length;

        while (index-- && baseIndexOf(chrSymbols, strSymbols[index], 0) > -1) {}
        return index;
      }

      /**
       * Gets the number of `placeholder` occurrences in `array`.
       *
       * @private
       * @param {Array} array The array to inspect.
       * @param {*} placeholder The placeholder to search for.
       * @returns {number} Returns the placeholder count.
       */
      function countHolders(array, placeholder) {
        var length = array.length,
            result = 0;

        while (length--) {
          if (array[length] === placeholder) {
            ++result;
          }
        }
        return result;
      }

      /**
       * Used by `_.deburr` to convert Latin-1 Supplement and Latin Extended-A
       * letters to basic Latin letters.
       *
       * @private
       * @param {string} letter The matched letter to deburr.
       * @returns {string} Returns the deburred letter.
       */
      var deburrLetter = basePropertyOf(deburredLetters);

      /**
       * Used by `_.escape` to convert characters to HTML entities.
       *
       * @private
       * @param {string} chr The matched character to escape.
       * @returns {string} Returns the escaped character.
       */
      var escapeHtmlChar = basePropertyOf(htmlEscapes);

      /**
       * Used by `_.template` to escape characters for inclusion in compiled string literals.
       *
       * @private
       * @param {string} chr The matched character to escape.
       * @returns {string} Returns the escaped character.
       */
      function escapeStringChar(chr) {
        return '\\' + stringEscapes[chr];
      }

      /**
       * Gets the value at `key` of `object`.
       *
       * @private
       * @param {Object} [object] The object to query.
       * @param {string} key The key of the property to get.
       * @returns {*} Returns the property value.
       */
      function getValue(object, key) {
        return object == null ? undefined$1 : object[key];
      }

      /**
       * Checks if `string` contains Unicode symbols.
       *
       * @private
       * @param {string} string The string to inspect.
       * @returns {boolean} Returns `true` if a symbol is found, else `false`.
       */
      function hasUnicode(string) {
        return reHasUnicode.test(string);
      }

      /**
       * Checks if `string` contains a word composed of Unicode symbols.
       *
       * @private
       * @param {string} string The string to inspect.
       * @returns {boolean} Returns `true` if a word is found, else `false`.
       */
      function hasUnicodeWord(string) {
        return reHasUnicodeWord.test(string);
      }

      /**
       * Converts `iterator` to an array.
       *
       * @private
       * @param {Object} iterator The iterator to convert.
       * @returns {Array} Returns the converted array.
       */
      function iteratorToArray(iterator) {
        var data,
            result = [];

        while (!(data = iterator.next()).done) {
          result.push(data.value);
        }
        return result;
      }

      /**
       * Converts `map` to its key-value pairs.
       *
       * @private
       * @param {Object} map The map to convert.
       * @returns {Array} Returns the key-value pairs.
       */
      function mapToArray(map) {
        var index = -1,
            result = Array(map.size);

        map.forEach(function(value, key) {
          result[++index] = [key, value];
        });
        return result;
      }

      /**
       * Creates a unary function that invokes `func` with its argument transformed.
       *
       * @private
       * @param {Function} func The function to wrap.
       * @param {Function} transform The argument transform.
       * @returns {Function} Returns the new function.
       */
      function overArg(func, transform) {
        return function(arg) {
          return func(transform(arg));
        };
      }

      /**
       * Replaces all `placeholder` elements in `array` with an internal placeholder
       * and returns an array of their indexes.
       *
       * @private
       * @param {Array} array The array to modify.
       * @param {*} placeholder The placeholder to replace.
       * @returns {Array} Returns the new array of placeholder indexes.
       */
      function replaceHolders(array, placeholder) {
        var index = -1,
            length = array.length,
            resIndex = 0,
            result = [];

        while (++index < length) {
          var value = array[index];
          if (value === placeholder || value === PLACEHOLDER) {
            array[index] = PLACEHOLDER;
            result[resIndex++] = index;
          }
        }
        return result;
      }

      /**
       * Converts `set` to an array of its values.
       *
       * @private
       * @param {Object} set The set to convert.
       * @returns {Array} Returns the values.
       */
      function setToArray(set) {
        var index = -1,
            result = Array(set.size);

        set.forEach(function(value) {
          result[++index] = value;
        });
        return result;
      }

      /**
       * Converts `set` to its value-value pairs.
       *
       * @private
       * @param {Object} set The set to convert.
       * @returns {Array} Returns the value-value pairs.
       */
      function setToPairs(set) {
        var index = -1,
            result = Array(set.size);

        set.forEach(function(value) {
          result[++index] = [value, value];
        });
        return result;
      }

      /**
       * A specialized version of `_.indexOf` which performs strict equality
       * comparisons of values, i.e. `===`.
       *
       * @private
       * @param {Array} array The array to inspect.
       * @param {*} value The value to search for.
       * @param {number} fromIndex The index to search from.
       * @returns {number} Returns the index of the matched value, else `-1`.
       */
      function strictIndexOf(array, value, fromIndex) {
        var index = fromIndex - 1,
            length = array.length;

        while (++index < length) {
          if (array[index] === value) {
            return index;
          }
        }
        return -1;
      }

      /**
       * A specialized version of `_.lastIndexOf` which performs strict equality
       * comparisons of values, i.e. `===`.
       *
       * @private
       * @param {Array} array The array to inspect.
       * @param {*} value The value to search for.
       * @param {number} fromIndex The index to search from.
       * @returns {number} Returns the index of the matched value, else `-1`.
       */
      function strictLastIndexOf(array, value, fromIndex) {
        var index = fromIndex + 1;
        while (index--) {
          if (array[index] === value) {
            return index;
          }
        }
        return index;
      }

      /**
       * Gets the number of symbols in `string`.
       *
       * @private
       * @param {string} string The string to inspect.
       * @returns {number} Returns the string size.
       */
      function stringSize(string) {
        return hasUnicode(string)
          ? unicodeSize(string)
          : asciiSize(string);
      }

      /**
       * Converts `string` to an array.
       *
       * @private
       * @param {string} string The string to convert.
       * @returns {Array} Returns the converted array.
       */
      function stringToArray(string) {
        return hasUnicode(string)
          ? unicodeToArray(string)
          : asciiToArray(string);
      }

      /**
       * Used by `_.unescape` to convert HTML entities to characters.
       *
       * @private
       * @param {string} chr The matched character to unescape.
       * @returns {string} Returns the unescaped character.
       */
      var unescapeHtmlChar = basePropertyOf(htmlUnescapes);

      /**
       * Gets the size of a Unicode `string`.
       *
       * @private
       * @param {string} string The string inspect.
       * @returns {number} Returns the string size.
       */
      function unicodeSize(string) {
        var result = reUnicode.lastIndex = 0;
        while (reUnicode.test(string)) {
          ++result;
        }
        return result;
      }

      /**
       * Converts a Unicode `string` to an array.
       *
       * @private
       * @param {string} string The string to convert.
       * @returns {Array} Returns the converted array.
       */
      function unicodeToArray(string) {
        return string.match(reUnicode) || [];
      }

      /**
       * Splits a Unicode `string` into an array of its words.
       *
       * @private
       * @param {string} The string to inspect.
       * @returns {Array} Returns the words of `string`.
       */
      function unicodeWords(string) {
        return string.match(reUnicodeWord) || [];
      }

      /*--------------------------------------------------------------------------*/

      /**
       * Create a new pristine `lodash` function using the `context` object.
       *
       * @static
       * @memberOf _
       * @since 1.1.0
       * @category Util
       * @param {Object} [context=root] The context object.
       * @returns {Function} Returns a new `lodash` function.
       * @example
       *
       * _.mixin({ 'foo': _.constant('foo') });
       *
       * var lodash = _.runInContext();
       * lodash.mixin({ 'bar': lodash.constant('bar') });
       *
       * _.isFunction(_.foo);
       * // => true
       * _.isFunction(_.bar);
       * // => false
       *
       * lodash.isFunction(lodash.foo);
       * // => false
       * lodash.isFunction(lodash.bar);
       * // => true
       *
       * // Create a suped-up `defer` in Node.js.
       * var defer = _.runInContext({ 'setTimeout': setImmediate }).defer;
       */
      var runInContext = (function runInContext(context) {
        context = context == null ? root : _.defaults(root.Object(), context, _.pick(root, contextProps));

        /** Built-in constructor references. */
        var Array = context.Array,
            Date = context.Date,
            Error = context.Error,
            Function = context.Function,
            Math = context.Math,
            Object = context.Object,
            RegExp = context.RegExp,
            String = context.String,
            TypeError = context.TypeError;

        /** Used for built-in method references. */
        var arrayProto = Array.prototype,
            funcProto = Function.prototype,
            objectProto = Object.prototype;

        /** Used to detect overreaching core-js shims. */
        var coreJsData = context['__core-js_shared__'];

        /** Used to resolve the decompiled source of functions. */
        var funcToString = funcProto.toString;

        /** Used to check objects for own properties. */
        var hasOwnProperty = objectProto.hasOwnProperty;

        /** Used to generate unique IDs. */
        var idCounter = 0;

        /** Used to detect methods masquerading as native. */
        var maskSrcKey = (function() {
          var uid = /[^.]+$/.exec(coreJsData && coreJsData.keys && coreJsData.keys.IE_PROTO || '');
          return uid ? ('Symbol(src)_1.' + uid) : '';
        }());

        /**
         * Used to resolve the
         * [`toStringTag`](http://ecma-international.org/ecma-262/7.0/#sec-object.prototype.tostring)
         * of values.
         */
        var nativeObjectToString = objectProto.toString;

        /** Used to infer the `Object` constructor. */
        var objectCtorString = funcToString.call(Object);

        /** Used to restore the original `_` reference in `_.noConflict`. */
        var oldDash = root._;

        /** Used to detect if a method is native. */
        var reIsNative = RegExp('^' +
          funcToString.call(hasOwnProperty).replace(reRegExpChar, '\\$&')
          .replace(/hasOwnProperty|(function).*?(?=\\\()| for .+?(?=\\\])/g, '$1.*?') + '$'
        );

        /** Built-in value references. */
        var Buffer = moduleExports ? context.Buffer : undefined$1,
            Symbol = context.Symbol,
            Uint8Array = context.Uint8Array,
            allocUnsafe = Buffer ? Buffer.allocUnsafe : undefined$1,
            getPrototype = overArg(Object.getPrototypeOf, Object),
            objectCreate = Object.create,
            propertyIsEnumerable = objectProto.propertyIsEnumerable,
            splice = arrayProto.splice,
            spreadableSymbol = Symbol ? Symbol.isConcatSpreadable : undefined$1,
            symIterator = Symbol ? Symbol.iterator : undefined$1,
            symToStringTag = Symbol ? Symbol.toStringTag : undefined$1;

        var defineProperty = (function() {
          try {
            var func = getNative(Object, 'defineProperty');
            func({}, '', {});
            return func;
          } catch (e) {}
        }());

        /** Mocked built-ins. */
        var ctxClearTimeout = context.clearTimeout !== root.clearTimeout && context.clearTimeout,
            ctxNow = Date && Date.now !== root.Date.now && Date.now,
            ctxSetTimeout = context.setTimeout !== root.setTimeout && context.setTimeout;

        /* Built-in method references for those with the same name as other `lodash` methods. */
        var nativeCeil = Math.ceil,
            nativeFloor = Math.floor,
            nativeGetSymbols = Object.getOwnPropertySymbols,
            nativeIsBuffer = Buffer ? Buffer.isBuffer : undefined$1,
            nativeIsFinite = context.isFinite,
            nativeJoin = arrayProto.join,
            nativeKeys = overArg(Object.keys, Object),
            nativeMax = Math.max,
            nativeMin = Math.min,
            nativeNow = Date.now,
            nativeParseInt = context.parseInt,
            nativeRandom = Math.random,
            nativeReverse = arrayProto.reverse;

        /* Built-in method references that are verified to be native. */
        var DataView = getNative(context, 'DataView'),
            Map = getNative(context, 'Map'),
            Promise = getNative(context, 'Promise'),
            Set = getNative(context, 'Set'),
            WeakMap = getNative(context, 'WeakMap'),
            nativeCreate = getNative(Object, 'create');

        /** Used to store function metadata. */
        var metaMap = WeakMap && new WeakMap;

        /** Used to lookup unminified function names. */
        var realNames = {};

        /** Used to detect maps, sets, and weakmaps. */
        var dataViewCtorString = toSource(DataView),
            mapCtorString = toSource(Map),
            promiseCtorString = toSource(Promise),
            setCtorString = toSource(Set),
            weakMapCtorString = toSource(WeakMap);

        /** Used to convert symbols to primitives and strings. */
        var symbolProto = Symbol ? Symbol.prototype : undefined$1,
            symbolValueOf = symbolProto ? symbolProto.valueOf : undefined$1,
            symbolToString = symbolProto ? symbolProto.toString : undefined$1;

        /*------------------------------------------------------------------------*/

        /**
         * Creates a `lodash` object which wraps `value` to enable implicit method
         * chain sequences. Methods that operate on and return arrays, collections,
         * and functions can be chained together. Methods that retrieve a single value
         * or may return a primitive value will automatically end the chain sequence
         * and return the unwrapped value. Otherwise, the value must be unwrapped
         * with `_#value`.
         *
         * Explicit chain sequences, which must be unwrapped with `_#value`, may be
         * enabled using `_.chain`.
         *
         * The execution of chained methods is lazy, that is, it's deferred until
         * `_#value` is implicitly or explicitly called.
         *
         * Lazy evaluation allows several methods to support shortcut fusion.
         * Shortcut fusion is an optimization to merge iteratee calls; this avoids
         * the creation of intermediate arrays and can greatly reduce the number of
         * iteratee executions. Sections of a chain sequence qualify for shortcut
         * fusion if the section is applied to an array and iteratees accept only
         * one argument. The heuristic for whether a section qualifies for shortcut
         * fusion is subject to change.
         *
         * Chaining is supported in custom builds as long as the `_#value` method is
         * directly or indirectly included in the build.
         *
         * In addition to lodash methods, wrappers have `Array` and `String` methods.
         *
         * The wrapper `Array` methods are:
         * `concat`, `join`, `pop`, `push`, `shift`, `sort`, `splice`, and `unshift`
         *
         * The wrapper `String` methods are:
         * `replace` and `split`
         *
         * The wrapper methods that support shortcut fusion are:
         * `at`, `compact`, `drop`, `dropRight`, `dropWhile`, `filter`, `find`,
         * `findLast`, `head`, `initial`, `last`, `map`, `reject`, `reverse`, `slice`,
         * `tail`, `take`, `takeRight`, `takeRightWhile`, `takeWhile`, and `toArray`
         *
         * The chainable wrapper methods are:
         * `after`, `ary`, `assign`, `assignIn`, `assignInWith`, `assignWith`, `at`,
         * `before`, `bind`, `bindAll`, `bindKey`, `castArray`, `chain`, `chunk`,
         * `commit`, `compact`, `concat`, `conforms`, `constant`, `countBy`, `create`,
         * `curry`, `debounce`, `defaults`, `defaultsDeep`, `defer`, `delay`,
         * `difference`, `differenceBy`, `differenceWith`, `drop`, `dropRight`,
         * `dropRightWhile`, `dropWhile`, `extend`, `extendWith`, `fill`, `filter`,
         * `flatMap`, `flatMapDeep`, `flatMapDepth`, `flatten`, `flattenDeep`,
         * `flattenDepth`, `flip`, `flow`, `flowRight`, `fromPairs`, `functions`,
         * `functionsIn`, `groupBy`, `initial`, `intersection`, `intersectionBy`,
         * `intersectionWith`, `invert`, `invertBy`, `invokeMap`, `iteratee`, `keyBy`,
         * `keys`, `keysIn`, `map`, `mapKeys`, `mapValues`, `matches`, `matchesProperty`,
         * `memoize`, `merge`, `mergeWith`, `method`, `methodOf`, `mixin`, `negate`,
         * `nthArg`, `omit`, `omitBy`, `once`, `orderBy`, `over`, `overArgs`,
         * `overEvery`, `overSome`, `partial`, `partialRight`, `partition`, `pick`,
         * `pickBy`, `plant`, `property`, `propertyOf`, `pull`, `pullAll`, `pullAllBy`,
         * `pullAllWith`, `pullAt`, `push`, `range`, `rangeRight`, `rearg`, `reject`,
         * `remove`, `rest`, `reverse`, `sampleSize`, `set`, `setWith`, `shuffle`,
         * `slice`, `sort`, `sortBy`, `splice`, `spread`, `tail`, `take`, `takeRight`,
         * `takeRightWhile`, `takeWhile`, `tap`, `throttle`, `thru`, `toArray`,
         * `toPairs`, `toPairsIn`, `toPath`, `toPlainObject`, `transform`, `unary`,
         * `union`, `unionBy`, `unionWith`, `uniq`, `uniqBy`, `uniqWith`, `unset`,
         * `unshift`, `unzip`, `unzipWith`, `update`, `updateWith`, `values`,
         * `valuesIn`, `without`, `wrap`, `xor`, `xorBy`, `xorWith`, `zip`,
         * `zipObject`, `zipObjectDeep`, and `zipWith`
         *
         * The wrapper methods that are **not** chainable by default are:
         * `add`, `attempt`, `camelCase`, `capitalize`, `ceil`, `clamp`, `clone`,
         * `cloneDeep`, `cloneDeepWith`, `cloneWith`, `conformsTo`, `deburr`,
         * `defaultTo`, `divide`, `each`, `eachRight`, `endsWith`, `eq`, `escape`,
         * `escapeRegExp`, `every`, `find`, `findIndex`, `findKey`, `findLast`,
         * `findLastIndex`, `findLastKey`, `first`, `floor`, `forEach`, `forEachRight`,
         * `forIn`, `forInRight`, `forOwn`, `forOwnRight`, `get`, `gt`, `gte`, `has`,
         * `hasIn`, `head`, `identity`, `includes`, `indexOf`, `inRange`, `invoke`,
         * `isArguments`, `isArray`, `isArrayBuffer`, `isArrayLike`, `isArrayLikeObject`,
         * `isBoolean`, `isBuffer`, `isDate`, `isElement`, `isEmpty`, `isEqual`,
         * `isEqualWith`, `isError`, `isFinite`, `isFunction`, `isInteger`, `isLength`,
         * `isMap`, `isMatch`, `isMatchWith`, `isNaN`, `isNative`, `isNil`, `isNull`,
         * `isNumber`, `isObject`, `isObjectLike`, `isPlainObject`, `isRegExp`,
         * `isSafeInteger`, `isSet`, `isString`, `isUndefined`, `isTypedArray`,
         * `isWeakMap`, `isWeakSet`, `join`, `kebabCase`, `last`, `lastIndexOf`,
         * `lowerCase`, `lowerFirst`, `lt`, `lte`, `max`, `maxBy`, `mean`, `meanBy`,
         * `min`, `minBy`, `multiply`, `noConflict`, `noop`, `now`, `nth`, `pad`,
         * `padEnd`, `padStart`, `parseInt`, `pop`, `random`, `reduce`, `reduceRight`,
         * `repeat`, `result`, `round`, `runInContext`, `sample`, `shift`, `size`,
         * `snakeCase`, `some`, `sortedIndex`, `sortedIndexBy`, `sortedLastIndex`,
         * `sortedLastIndexBy`, `startCase`, `startsWith`, `stubArray`, `stubFalse`,
         * `stubObject`, `stubString`, `stubTrue`, `subtract`, `sum`, `sumBy`,
         * `template`, `times`, `toFinite`, `toInteger`, `toJSON`, `toLength`,
         * `toLower`, `toNumber`, `toSafeInteger`, `toString`, `toUpper`, `trim`,
         * `trimEnd`, `trimStart`, `truncate`, `unescape`, `uniqueId`, `upperCase`,
         * `upperFirst`, `value`, and `words`
         *
         * @name _
         * @constructor
         * @category Seq
         * @param {*} value The value to wrap in a `lodash` instance.
         * @returns {Object} Returns the new `lodash` wrapper instance.
         * @example
         *
         * function square(n) {
         *   return n * n;
         * }
         *
         * var wrapped = _([1, 2, 3]);
         *
         * // Returns an unwrapped value.
         * wrapped.reduce(_.add);
         * // => 6
         *
         * // Returns a wrapped value.
         * var squares = wrapped.map(square);
         *
         * _.isArray(squares);
         * // => false
         *
         * _.isArray(squares.value());
         * // => true
         */
        function lodash(value) {
          if (isObjectLike(value) && !isArray(value) && !(value instanceof LazyWrapper)) {
            if (value instanceof LodashWrapper) {
              return value;
            }
            if (hasOwnProperty.call(value, '__wrapped__')) {
              return wrapperClone(value);
            }
          }
          return new LodashWrapper(value);
        }

        /**
         * The base implementation of `_.create` without support for assigning
         * properties to the created object.
         *
         * @private
         * @param {Object} proto The object to inherit from.
         * @returns {Object} Returns the new object.
         */
        var baseCreate = (function() {
          function object() {}
          return function(proto) {
            if (!isObject(proto)) {
              return {};
            }
            if (objectCreate) {
              return objectCreate(proto);
            }
            object.prototype = proto;
            var result = new object;
            object.prototype = undefined$1;
            return result;
          };
        }());

        /**
         * The function whose prototype chain sequence wrappers inherit from.
         *
         * @private
         */
        function baseLodash() {
          // No operation performed.
        }

        /**
         * The base constructor for creating `lodash` wrapper objects.
         *
         * @private
         * @param {*} value The value to wrap.
         * @param {boolean} [chainAll] Enable explicit method chain sequences.
         */
        function LodashWrapper(value, chainAll) {
          this.__wrapped__ = value;
          this.__actions__ = [];
          this.__chain__ = !!chainAll;
          this.__index__ = 0;
          this.__values__ = undefined$1;
        }

        /**
         * By default, the template delimiters used by lodash are like those in
         * embedded Ruby (ERB) as well as ES2015 template strings. Change the
         * following template settings to use alternative delimiters.
         *
         * @static
         * @memberOf _
         * @type {Object}
         */
        lodash.templateSettings = {

          /**
           * Used to detect `data` property values to be HTML-escaped.
           *
           * @memberOf _.templateSettings
           * @type {RegExp}
           */
          'escape': reEscape,

          /**
           * Used to detect code to be evaluated.
           *
           * @memberOf _.templateSettings
           * @type {RegExp}
           */
          'evaluate': reEvaluate,

          /**
           * Used to detect `data` property values to inject.
           *
           * @memberOf _.templateSettings
           * @type {RegExp}
           */
          'interpolate': reInterpolate,

          /**
           * Used to reference the data object in the template text.
           *
           * @memberOf _.templateSettings
           * @type {string}
           */
          'variable': '',

          /**
           * Used to import variables into the compiled template.
           *
           * @memberOf _.templateSettings
           * @type {Object}
           */
          'imports': {

            /**
             * A reference to the `lodash` function.
             *
             * @memberOf _.templateSettings.imports
             * @type {Function}
             */
            '_': lodash
          }
        };

        // Ensure wrappers are instances of `baseLodash`.
        lodash.prototype = baseLodash.prototype;
        lodash.prototype.constructor = lodash;

        LodashWrapper.prototype = baseCreate(baseLodash.prototype);
        LodashWrapper.prototype.constructor = LodashWrapper;

        /*------------------------------------------------------------------------*/

        /**
         * Creates a lazy wrapper object which wraps `value` to enable lazy evaluation.
         *
         * @private
         * @constructor
         * @param {*} value The value to wrap.
         */
        function LazyWrapper(value) {
          this.__wrapped__ = value;
          this.__actions__ = [];
          this.__dir__ = 1;
          this.__filtered__ = false;
          this.__iteratees__ = [];
          this.__takeCount__ = MAX_ARRAY_LENGTH;
          this.__views__ = [];
        }

        /**
         * Creates a clone of the lazy wrapper object.
         *
         * @private
         * @name clone
         * @memberOf LazyWrapper
         * @returns {Object} Returns the cloned `LazyWrapper` object.
         */
        function lazyClone() {
          var result = new LazyWrapper(this.__wrapped__);
          result.__actions__ = copyArray(this.__actions__);
          result.__dir__ = this.__dir__;
          result.__filtered__ = this.__filtered__;
          result.__iteratees__ = copyArray(this.__iteratees__);
          result.__takeCount__ = this.__takeCount__;
          result.__views__ = copyArray(this.__views__);
          return result;
        }

        /**
         * Reverses the direction of lazy iteration.
         *
         * @private
         * @name reverse
         * @memberOf LazyWrapper
         * @returns {Object} Returns the new reversed `LazyWrapper` object.
         */
        function lazyReverse() {
          if (this.__filtered__) {
            var result = new LazyWrapper(this);
            result.__dir__ = -1;
            result.__filtered__ = true;
          } else {
            result = this.clone();
            result.__dir__ *= -1;
          }
          return result;
        }

        /**
         * Extracts the unwrapped value from its lazy wrapper.
         *
         * @private
         * @name value
         * @memberOf LazyWrapper
         * @returns {*} Returns the unwrapped value.
         */
        function lazyValue() {
          var array = this.__wrapped__.value(),
              dir = this.__dir__,
              isArr = isArray(array),
              isRight = dir < 0,
              arrLength = isArr ? array.length : 0,
              view = getView(0, arrLength, this.__views__),
              start = view.start,
              end = view.end,
              length = end - start,
              index = isRight ? end : (start - 1),
              iteratees = this.__iteratees__,
              iterLength = iteratees.length,
              resIndex = 0,
              takeCount = nativeMin(length, this.__takeCount__);

          if (!isArr || (!isRight && arrLength == length && takeCount == length)) {
            return baseWrapperValue(array, this.__actions__);
          }
          var result = [];

          outer:
          while (length-- && resIndex < takeCount) {
            index += dir;

            var iterIndex = -1,
                value = array[index];

            while (++iterIndex < iterLength) {
              var data = iteratees[iterIndex],
                  iteratee = data.iteratee,
                  type = data.type,
                  computed = iteratee(value);

              if (type == LAZY_MAP_FLAG) {
                value = computed;
              } else if (!computed) {
                if (type == LAZY_FILTER_FLAG) {
                  continue outer;
                } else {
                  break outer;
                }
              }
            }
            result[resIndex++] = value;
          }
          return result;
        }

        // Ensure `LazyWrapper` is an instance of `baseLodash`.
        LazyWrapper.prototype = baseCreate(baseLodash.prototype);
        LazyWrapper.prototype.constructor = LazyWrapper;

        /*------------------------------------------------------------------------*/

        /**
         * Creates a hash object.
         *
         * @private
         * @constructor
         * @param {Array} [entries] The key-value pairs to cache.
         */
        function Hash(entries) {
          var index = -1,
              length = entries == null ? 0 : entries.length;

          this.clear();
          while (++index < length) {
            var entry = entries[index];
            this.set(entry[0], entry[1]);
          }
        }

        /**
         * Removes all key-value entries from the hash.
         *
         * @private
         * @name clear
         * @memberOf Hash
         */
        function hashClear() {
          this.__data__ = nativeCreate ? nativeCreate(null) : {};
          this.size = 0;
        }

        /**
         * Removes `key` and its value from the hash.
         *
         * @private
         * @name delete
         * @memberOf Hash
         * @param {Object} hash The hash to modify.
         * @param {string} key The key of the value to remove.
         * @returns {boolean} Returns `true` if the entry was removed, else `false`.
         */
        function hashDelete(key) {
          var result = this.has(key) && delete this.__data__[key];
          this.size -= result ? 1 : 0;
          return result;
        }

        /**
         * Gets the hash value for `key`.
         *
         * @private
         * @name get
         * @memberOf Hash
         * @param {string} key The key of the value to get.
         * @returns {*} Returns the entry value.
         */
        function hashGet(key) {
          var data = this.__data__;
          if (nativeCreate) {
            var result = data[key];
            return result === HASH_UNDEFINED ? undefined$1 : result;
          }
          return hasOwnProperty.call(data, key) ? data[key] : undefined$1;
        }

        /**
         * Checks if a hash value for `key` exists.
         *
         * @private
         * @name has
         * @memberOf Hash
         * @param {string} key The key of the entry to check.
         * @returns {boolean} Returns `true` if an entry for `key` exists, else `false`.
         */
        function hashHas(key) {
          var data = this.__data__;
          return nativeCreate ? (data[key] !== undefined$1) : hasOwnProperty.call(data, key);
        }

        /**
         * Sets the hash `key` to `value`.
         *
         * @private
         * @name set
         * @memberOf Hash
         * @param {string} key The key of the value to set.
         * @param {*} value The value to set.
         * @returns {Object} Returns the hash instance.
         */
        function hashSet(key, value) {
          var data = this.__data__;
          this.size += this.has(key) ? 0 : 1;
          data[key] = (nativeCreate && value === undefined$1) ? HASH_UNDEFINED : value;
          return this;
        }

        // Add methods to `Hash`.
        Hash.prototype.clear = hashClear;
        Hash.prototype['delete'] = hashDelete;
        Hash.prototype.get = hashGet;
        Hash.prototype.has = hashHas;
        Hash.prototype.set = hashSet;

        /*------------------------------------------------------------------------*/

        /**
         * Creates an list cache object.
         *
         * @private
         * @constructor
         * @param {Array} [entries] The key-value pairs to cache.
         */
        function ListCache(entries) {
          var index = -1,
              length = entries == null ? 0 : entries.length;

          this.clear();
          while (++index < length) {
            var entry = entries[index];
            this.set(entry[0], entry[1]);
          }
        }

        /**
         * Removes all key-value entries from the list cache.
         *
         * @private
         * @name clear
         * @memberOf ListCache
         */
        function listCacheClear() {
          this.__data__ = [];
          this.size = 0;
        }

        /**
         * Removes `key` and its value from the list cache.
         *
         * @private
         * @name delete
         * @memberOf ListCache
         * @param {string} key The key of the value to remove.
         * @returns {boolean} Returns `true` if the entry was removed, else `false`.
         */
        function listCacheDelete(key) {
          var data = this.__data__,
              index = assocIndexOf(data, key);

          if (index < 0) {
            return false;
          }
          var lastIndex = data.length - 1;
          if (index == lastIndex) {
            data.pop();
          } else {
            splice.call(data, index, 1);
          }
          --this.size;
          return true;
        }

        /**
         * Gets the list cache value for `key`.
         *
         * @private
         * @name get
         * @memberOf ListCache
         * @param {string} key The key of the value to get.
         * @returns {*} Returns the entry value.
         */
        function listCacheGet(key) {
          var data = this.__data__,
              index = assocIndexOf(data, key);

          return index < 0 ? undefined$1 : data[index][1];
        }

        /**
         * Checks if a list cache value for `key` exists.
         *
         * @private
         * @name has
         * @memberOf ListCache
         * @param {string} key The key of the entry to check.
         * @returns {boolean} Returns `true` if an entry for `key` exists, else `false`.
         */
        function listCacheHas(key) {
          return assocIndexOf(this.__data__, key) > -1;
        }

        /**
         * Sets the list cache `key` to `value`.
         *
         * @private
         * @name set
         * @memberOf ListCache
         * @param {string} key The key of the value to set.
         * @param {*} value The value to set.
         * @returns {Object} Returns the list cache instance.
         */
        function listCacheSet(key, value) {
          var data = this.__data__,
              index = assocIndexOf(data, key);

          if (index < 0) {
            ++this.size;
            data.push([key, value]);
          } else {
            data[index][1] = value;
          }
          return this;
        }

        // Add methods to `ListCache`.
        ListCache.prototype.clear = listCacheClear;
        ListCache.prototype['delete'] = listCacheDelete;
        ListCache.prototype.get = listCacheGet;
        ListCache.prototype.has = listCacheHas;
        ListCache.prototype.set = listCacheSet;

        /*------------------------------------------------------------------------*/

        /**
         * Creates a map cache object to store key-value pairs.
         *
         * @private
         * @constructor
         * @param {Array} [entries] The key-value pairs to cache.
         */
        function MapCache(entries) {
          var index = -1,
              length = entries == null ? 0 : entries.length;

          this.clear();
          while (++index < length) {
            var entry = entries[index];
            this.set(entry[0], entry[1]);
          }
        }

        /**
         * Removes all key-value entries from the map.
         *
         * @private
         * @name clear
         * @memberOf MapCache
         */
        function mapCacheClear() {
          this.size = 0;
          this.__data__ = {
            'hash': new Hash,
            'map': new (Map || ListCache),
            'string': new Hash
          };
        }

        /**
         * Removes `key` and its value from the map.
         *
         * @private
         * @name delete
         * @memberOf MapCache
         * @param {string} key The key of the value to remove.
         * @returns {boolean} Returns `true` if the entry was removed, else `false`.
         */
        function mapCacheDelete(key) {
          var result = getMapData(this, key)['delete'](key);
          this.size -= result ? 1 : 0;
          return result;
        }

        /**
         * Gets the map value for `key`.
         *
         * @private
         * @name get
         * @memberOf MapCache
         * @param {string} key The key of the value to get.
         * @returns {*} Returns the entry value.
         */
        function mapCacheGet(key) {
          return getMapData(this, key).get(key);
        }

        /**
         * Checks if a map value for `key` exists.
         *
         * @private
         * @name has
         * @memberOf MapCache
         * @param {string} key The key of the entry to check.
         * @returns {boolean} Returns `true` if an entry for `key` exists, else `false`.
         */
        function mapCacheHas(key) {
          return getMapData(this, key).has(key);
        }

        /**
         * Sets the map `key` to `value`.
         *
         * @private
         * @name set
         * @memberOf MapCache
         * @param {string} key The key of the value to set.
         * @param {*} value The value to set.
         * @returns {Object} Returns the map cache instance.
         */
        function mapCacheSet(key, value) {
          var data = getMapData(this, key),
              size = data.size;

          data.set(key, value);
          this.size += data.size == size ? 0 : 1;
          return this;
        }

        // Add methods to `MapCache`.
        MapCache.prototype.clear = mapCacheClear;
        MapCache.prototype['delete'] = mapCacheDelete;
        MapCache.prototype.get = mapCacheGet;
        MapCache.prototype.has = mapCacheHas;
        MapCache.prototype.set = mapCacheSet;

        /*------------------------------------------------------------------------*/

        /**
         *
         * Creates an array cache object to store unique values.
         *
         * @private
         * @constructor
         * @param {Array} [values] The values to cache.
         */
        function SetCache(values) {
          var index = -1,
              length = values == null ? 0 : values.length;

          this.__data__ = new MapCache;
          while (++index < length) {
            this.add(values[index]);
          }
        }

        /**
         * Adds `value` to the array cache.
         *
         * @private
         * @name add
         * @memberOf SetCache
         * @alias push
         * @param {*} value The value to cache.
         * @returns {Object} Returns the cache instance.
         */
        function setCacheAdd(value) {
          this.__data__.set(value, HASH_UNDEFINED);
          return this;
        }

        /**
         * Checks if `value` is in the array cache.
         *
         * @private
         * @name has
         * @memberOf SetCache
         * @param {*} value The value to search for.
         * @returns {number} Returns `true` if `value` is found, else `false`.
         */
        function setCacheHas(value) {
          return this.__data__.has(value);
        }

        // Add methods to `SetCache`.
        SetCache.prototype.add = SetCache.prototype.push = setCacheAdd;
        SetCache.prototype.has = setCacheHas;

        /*------------------------------------------------------------------------*/

        /**
         * Creates a stack cache object to store key-value pairs.
         *
         * @private
         * @constructor
         * @param {Array} [entries] The key-value pairs to cache.
         */
        function Stack(entries) {
          var data = this.__data__ = new ListCache(entries);
          this.size = data.size;
        }

        /**
         * Removes all key-value entries from the stack.
         *
         * @private
         * @name clear
         * @memberOf Stack
         */
        function stackClear() {
          this.__data__ = new ListCache;
          this.size = 0;
        }

        /**
         * Removes `key` and its value from the stack.
         *
         * @private
         * @name delete
         * @memberOf Stack
         * @param {string} key The key of the value to remove.
         * @returns {boolean} Returns `true` if the entry was removed, else `false`.
         */
        function stackDelete(key) {
          var data = this.__data__,
              result = data['delete'](key);

          this.size = data.size;
          return result;
        }

        /**
         * Gets the stack value for `key`.
         *
         * @private
         * @name get
         * @memberOf Stack
         * @param {string} key The key of the value to get.
         * @returns {*} Returns the entry value.
         */
        function stackGet(key) {
          return this.__data__.get(key);
        }

        /**
         * Checks if a stack value for `key` exists.
         *
         * @private
         * @name has
         * @memberOf Stack
         * @param {string} key The key of the entry to check.
         * @returns {boolean} Returns `true` if an entry for `key` exists, else `false`.
         */
        function stackHas(key) {
          return this.__data__.has(key);
        }

        /**
         * Sets the stack `key` to `value`.
         *
         * @private
         * @name set
         * @memberOf Stack
         * @param {string} key The key of the value to set.
         * @param {*} value The value to set.
         * @returns {Object} Returns the stack cache instance.
         */
        function stackSet(key, value) {
          var data = this.__data__;
          if (data instanceof ListCache) {
            var pairs = data.__data__;
            if (!Map || (pairs.length < LARGE_ARRAY_SIZE - 1)) {
              pairs.push([key, value]);
              this.size = ++data.size;
              return this;
            }
            data = this.__data__ = new MapCache(pairs);
          }
          data.set(key, value);
          this.size = data.size;
          return this;
        }

        // Add methods to `Stack`.
        Stack.prototype.clear = stackClear;
        Stack.prototype['delete'] = stackDelete;
        Stack.prototype.get = stackGet;
        Stack.prototype.has = stackHas;
        Stack.prototype.set = stackSet;

        /*------------------------------------------------------------------------*/

        /**
         * Creates an array of the enumerable property names of the array-like `value`.
         *
         * @private
         * @param {*} value The value to query.
         * @param {boolean} inherited Specify returning inherited property names.
         * @returns {Array} Returns the array of property names.
         */
        function arrayLikeKeys(value, inherited) {
          var isArr = isArray(value),
              isArg = !isArr && isArguments(value),
              isBuff = !isArr && !isArg && isBuffer(value),
              isType = !isArr && !isArg && !isBuff && isTypedArray(value),
              skipIndexes = isArr || isArg || isBuff || isType,
              result = skipIndexes ? baseTimes(value.length, String) : [],
              length = result.length;

          for (var key in value) {
            if ((inherited || hasOwnProperty.call(value, key)) &&
                !(skipIndexes && (
                   // Safari 9 has enumerable `arguments.length` in strict mode.
                   key == 'length' ||
                   // Node.js 0.10 has enumerable non-index properties on buffers.
                   (isBuff && (key == 'offset' || key == 'parent')) ||
                   // PhantomJS 2 has enumerable non-index properties on typed arrays.
                   (isType && (key == 'buffer' || key == 'byteLength' || key == 'byteOffset')) ||
                   // Skip index properties.
                   isIndex(key, length)
                ))) {
              result.push(key);
            }
          }
          return result;
        }

        /**
         * A specialized version of `_.sample` for arrays.
         *
         * @private
         * @param {Array} array The array to sample.
         * @returns {*} Returns the random element.
         */
        function arraySample(array) {
          var length = array.length;
          return length ? array[baseRandom(0, length - 1)] : undefined$1;
        }

        /**
         * A specialized version of `_.sampleSize` for arrays.
         *
         * @private
         * @param {Array} array The array to sample.
         * @param {number} n The number of elements to sample.
         * @returns {Array} Returns the random elements.
         */
        function arraySampleSize(array, n) {
          return shuffleSelf(copyArray(array), baseClamp(n, 0, array.length));
        }

        /**
         * A specialized version of `_.shuffle` for arrays.
         *
         * @private
         * @param {Array} array The array to shuffle.
         * @returns {Array} Returns the new shuffled array.
         */
        function arrayShuffle(array) {
          return shuffleSelf(copyArray(array));
        }

        /**
         * This function is like `assignValue` except that it doesn't assign
         * `undefined` values.
         *
         * @private
         * @param {Object} object The object to modify.
         * @param {string} key The key of the property to assign.
         * @param {*} value The value to assign.
         */
        function assignMergeValue(object, key, value) {
          if ((value !== undefined$1 && !eq(object[key], value)) ||
              (value === undefined$1 && !(key in object))) {
            baseAssignValue(object, key, value);
          }
        }

        /**
         * Assigns `value` to `key` of `object` if the existing value is not equivalent
         * using [`SameValueZero`](http://ecma-international.org/ecma-262/7.0/#sec-samevaluezero)
         * for equality comparisons.
         *
         * @private
         * @param {Object} object The object to modify.
         * @param {string} key The key of the property to assign.
         * @param {*} value The value to assign.
         */
        function assignValue(object, key, value) {
          var objValue = object[key];
          if (!(hasOwnProperty.call(object, key) && eq(objValue, value)) ||
              (value === undefined$1 && !(key in object))) {
            baseAssignValue(object, key, value);
          }
        }

        /**
         * Gets the index at which the `key` is found in `array` of key-value pairs.
         *
         * @private
         * @param {Array} array The array to inspect.
         * @param {*} key The key to search for.
         * @returns {number} Returns the index of the matched value, else `-1`.
         */
        function assocIndexOf(array, key) {
          var length = array.length;
          while (length--) {
            if (eq(array[length][0], key)) {
              return length;
            }
          }
          return -1;
        }

        /**
         * Aggregates elements of `collection` on `accumulator` with keys transformed
         * by `iteratee` and values set by `setter`.
         *
         * @private
         * @param {Array|Object} collection The collection to iterate over.
         * @param {Function} setter The function to set `accumulator` values.
         * @param {Function} iteratee The iteratee to transform keys.
         * @param {Object} accumulator The initial aggregated object.
         * @returns {Function} Returns `accumulator`.
         */
        function baseAggregator(collection, setter, iteratee, accumulator) {
          baseEach(collection, function(value, key, collection) {
            setter(accumulator, value, iteratee(value), collection);
          });
          return accumulator;
        }

        /**
         * The base implementation of `_.assign` without support for multiple sources
         * or `customizer` functions.
         *
         * @private
         * @param {Object} object The destination object.
         * @param {Object} source The source object.
         * @returns {Object} Returns `object`.
         */
        function baseAssign(object, source) {
          return object && copyObject(source, keys(source), object);
        }

        /**
         * The base implementation of `_.assignIn` without support for multiple sources
         * or `customizer` functions.
         *
         * @private
         * @param {Object} object The destination object.
         * @param {Object} source The source object.
         * @returns {Object} Returns `object`.
         */
        function baseAssignIn(object, source) {
          return object && copyObject(source, keysIn(source), object);
        }

        /**
         * The base implementation of `assignValue` and `assignMergeValue` without
         * value checks.
         *
         * @private
         * @param {Object} object The object to modify.
         * @param {string} key The key of the property to assign.
         * @param {*} value The value to assign.
         */
        function baseAssignValue(object, key, value) {
          if (key == '__proto__' && defineProperty) {
            defineProperty(object, key, {
              'configurable': true,
              'enumerable': true,
              'value': value,
              'writable': true
            });
          } else {
            object[key] = value;
          }
        }

        /**
         * The base implementation of `_.at` without support for individual paths.
         *
         * @private
         * @param {Object} object The object to iterate over.
         * @param {string[]} paths The property paths to pick.
         * @returns {Array} Returns the picked elements.
         */
        function baseAt(object, paths) {
          var index = -1,
              length = paths.length,
              result = Array(length),
              skip = object == null;

          while (++index < length) {
            result[index] = skip ? undefined$1 : get(object, paths[index]);
          }
          return result;
        }

        /**
         * The base implementation of `_.clamp` which doesn't coerce arguments.
         *
         * @private
         * @param {number} number The number to clamp.
         * @param {number} [lower] The lower bound.
         * @param {number} upper The upper bound.
         * @returns {number} Returns the clamped number.
         */
        function baseClamp(number, lower, upper) {
          if (number === number) {
            if (upper !== undefined$1) {
              number = number <= upper ? number : upper;
            }
            if (lower !== undefined$1) {
              number = number >= lower ? number : lower;
            }
          }
          return number;
        }

        /**
         * The base implementation of `_.clone` and `_.cloneDeep` which tracks
         * traversed objects.
         *
         * @private
         * @param {*} value The value to clone.
         * @param {boolean} bitmask The bitmask flags.
         *  1 - Deep clone
         *  2 - Flatten inherited properties
         *  4 - Clone symbols
         * @param {Function} [customizer] The function to customize cloning.
         * @param {string} [key] The key of `value`.
         * @param {Object} [object] The parent object of `value`.
         * @param {Object} [stack] Tracks traversed objects and their clone counterparts.
         * @returns {*} Returns the cloned value.
         */
        function baseClone(value, bitmask, customizer, key, object, stack) {
          var result,
              isDeep = bitmask & CLONE_DEEP_FLAG,
              isFlat = bitmask & CLONE_FLAT_FLAG,
              isFull = bitmask & CLONE_SYMBOLS_FLAG;

          if (customizer) {
            result = object ? customizer(value, key, object, stack) : customizer(value);
          }
          if (result !== undefined$1) {
            return result;
          }
          if (!isObject(value)) {
            return value;
          }
          var isArr = isArray(value);
          if (isArr) {
            result = initCloneArray(value);
            if (!isDeep) {
              return copyArray(value, result);
            }
          } else {
            var tag = getTag(value),
                isFunc = tag == funcTag || tag == genTag;

            if (isBuffer(value)) {
              return cloneBuffer(value, isDeep);
            }
            if (tag == objectTag || tag == argsTag || (isFunc && !object)) {
              result = (isFlat || isFunc) ? {} : initCloneObject(value);
              if (!isDeep) {
                return isFlat
                  ? copySymbolsIn(value, baseAssignIn(result, value))
                  : copySymbols(value, baseAssign(result, value));
              }
            } else {
              if (!cloneableTags[tag]) {
                return object ? value : {};
              }
              result = initCloneByTag(value, tag, isDeep);
            }
          }
          // Check for circular references and return its corresponding clone.
          stack || (stack = new Stack);
          var stacked = stack.get(value);
          if (stacked) {
            return stacked;
          }
          stack.set(value, result);

          if (isSet(value)) {
            value.forEach(function(subValue) {
              result.add(baseClone(subValue, bitmask, customizer, subValue, value, stack));
            });
          } else if (isMap(value)) {
            value.forEach(function(subValue, key) {
              result.set(key, baseClone(subValue, bitmask, customizer, key, value, stack));
            });
          }

          var keysFunc = isFull
            ? (isFlat ? getAllKeysIn : getAllKeys)
            : (isFlat ? keysIn : keys);

          var props = isArr ? undefined$1 : keysFunc(value);
          arrayEach(props || value, function(subValue, key) {
            if (props) {
              key = subValue;
              subValue = value[key];
            }
            // Recursively populate clone (susceptible to call stack limits).
            assignValue(result, key, baseClone(subValue, bitmask, customizer, key, value, stack));
          });
          return result;
        }

        /**
         * The base implementation of `_.conforms` which doesn't clone `source`.
         *
         * @private
         * @param {Object} source The object of property predicates to conform to.
         * @returns {Function} Returns the new spec function.
         */
        function baseConforms(source) {
          var props = keys(source);
          return function(object) {
            return baseConformsTo(object, source, props);
          };
        }

        /**
         * The base implementation of `_.conformsTo` which accepts `props` to check.
         *
         * @private
         * @param {Object} object The object to inspect.
         * @param {Object} source The object of property predicates to conform to.
         * @returns {boolean} Returns `true` if `object` conforms, else `false`.
         */
        function baseConformsTo(object, source, props) {
          var length = props.length;
          if (object == null) {
            return !length;
          }
          object = Object(object);
          while (length--) {
            var key = props[length],
                predicate = source[key],
                value = object[key];

            if ((value === undefined$1 && !(key in object)) || !predicate(value)) {
              return false;
            }
          }
          return true;
        }

        /**
         * The base implementation of `_.delay` and `_.defer` which accepts `args`
         * to provide to `func`.
         *
         * @private
         * @param {Function} func The function to delay.
         * @param {number} wait The number of milliseconds to delay invocation.
         * @param {Array} args The arguments to provide to `func`.
         * @returns {number|Object} Returns the timer id or timeout object.
         */
        function baseDelay(func, wait, args) {
          if (typeof func != 'function') {
            throw new TypeError(FUNC_ERROR_TEXT);
          }
          return setTimeout(function() { func.apply(undefined$1, args); }, wait);
        }

        /**
         * The base implementation of methods like `_.difference` without support
         * for excluding multiple arrays or iteratee shorthands.
         *
         * @private
         * @param {Array} array The array to inspect.
         * @param {Array} values The values to exclude.
         * @param {Function} [iteratee] The iteratee invoked per element.
         * @param {Function} [comparator] The comparator invoked per element.
         * @returns {Array} Returns the new array of filtered values.
         */
        function baseDifference(array, values, iteratee, comparator) {
          var index = -1,
              includes = arrayIncludes,
              isCommon = true,
              length = array.length,
              result = [],
              valuesLength = values.length;

          if (!length) {
            return result;
          }
          if (iteratee) {
            values = arrayMap(values, baseUnary(iteratee));
          }
          if (comparator) {
            includes = arrayIncludesWith;
            isCommon = false;
          }
          else if (values.length >= LARGE_ARRAY_SIZE) {
            includes = cacheHas;
            isCommon = false;
            values = new SetCache(values);
          }
          outer:
          while (++index < length) {
            var value = array[index],
                computed = iteratee == null ? value : iteratee(value);

            value = (comparator || value !== 0) ? value : 0;
            if (isCommon && computed === computed) {
              var valuesIndex = valuesLength;
              while (valuesIndex--) {
                if (values[valuesIndex] === computed) {
                  continue outer;
                }
              }
              result.push(value);
            }
            else if (!includes(values, computed, comparator)) {
              result.push(value);
            }
          }
          return result;
        }

        /**
         * The base implementation of `_.forEach` without support for iteratee shorthands.
         *
         * @private
         * @param {Array|Object} collection The collection to iterate over.
         * @param {Function} iteratee The function invoked per iteration.
         * @returns {Array|Object} Returns `collection`.
         */
        var baseEach = createBaseEach(baseForOwn);

        /**
         * The base implementation of `_.forEachRight` without support for iteratee shorthands.
         *
         * @private
         * @param {Array|Object} collection The collection to iterate over.
         * @param {Function} iteratee The function invoked per iteration.
         * @returns {Array|Object} Returns `collection`.
         */
        var baseEachRight = createBaseEach(baseForOwnRight, true);

        /**
         * The base implementation of `_.every` without support for iteratee shorthands.
         *
         * @private
         * @param {Array|Object} collection The collection to iterate over.
         * @param {Function} predicate The function invoked per iteration.
         * @returns {boolean} Returns `true` if all elements pass the predicate check,
         *  else `false`
         */
        function baseEvery(collection, predicate) {
          var result = true;
          baseEach(collection, function(value, index, collection) {
            result = !!predicate(value, index, collection);
            return result;
          });
          return result;
        }

        /**
         * The base implementation of methods like `_.max` and `_.min` which accepts a
         * `comparator` to determine the extremum value.
         *
         * @private
         * @param {Array} array The array to iterate over.
         * @param {Function} iteratee The iteratee invoked per iteration.
         * @param {Function} comparator The comparator used to compare values.
         * @returns {*} Returns the extremum value.
         */
        function baseExtremum(array, iteratee, comparator) {
          var index = -1,
              length = array.length;

          while (++index < length) {
            var value = array[index],
                current = iteratee(value);

            if (current != null && (computed === undefined$1
                  ? (current === current && !isSymbol(current))
                  : comparator(current, computed)
                )) {
              var computed = current,
                  result = value;
            }
          }
          return result;
        }

        /**
         * The base implementation of `_.fill` without an iteratee call guard.
         *
         * @private
         * @param {Array} array The array to fill.
         * @param {*} value The value to fill `array` with.
         * @param {number} [start=0] The start position.
         * @param {number} [end=array.length] The end position.
         * @returns {Array} Returns `array`.
         */
        function baseFill(array, value, start, end) {
          var length = array.length;

          start = toInteger(start);
          if (start < 0) {
            start = -start > length ? 0 : (length + start);
          }
          end = (end === undefined$1 || end > length) ? length : toInteger(end);
          if (end < 0) {
            end += length;
          }
          end = start > end ? 0 : toLength(end);
          while (start < end) {
            array[start++] = value;
          }
          return array;
        }

        /**
         * The base implementation of `_.filter` without support for iteratee shorthands.
         *
         * @private
         * @param {Array|Object} collection The collection to iterate over.
         * @param {Function} predicate The function invoked per iteration.
         * @returns {Array} Returns the new filtered array.
         */
        function baseFilter(collection, predicate) {
          var result = [];
          baseEach(collection, function(value, index, collection) {
            if (predicate(value, index, collection)) {
              result.push(value);
            }
          });
          return result;
        }

        /**
         * The base implementation of `_.flatten` with support for restricting flattening.
         *
         * @private
         * @param {Array} array The array to flatten.
         * @param {number} depth The maximum recursion depth.
         * @param {boolean} [predicate=isFlattenable] The function invoked per iteration.
         * @param {boolean} [isStrict] Restrict to values that pass `predicate` checks.
         * @param {Array} [result=[]] The initial result value.
         * @returns {Array} Returns the new flattened array.
         */
        function baseFlatten(array, depth, predicate, isStrict, result) {
          var index = -1,
              length = array.length;

          predicate || (predicate = isFlattenable);
          result || (result = []);

          while (++index < length) {
            var value = array[index];
            if (depth > 0 && predicate(value)) {
              if (depth > 1) {
                // Recursively flatten arrays (susceptible to call stack limits).
                baseFlatten(value, depth - 1, predicate, isStrict, result);
              } else {
                arrayPush(result, value);
              }
            } else if (!isStrict) {
              result[result.length] = value;
            }
          }
          return result;
        }

        /**
         * The base implementation of `baseForOwn` which iterates over `object`
         * properties returned by `keysFunc` and invokes `iteratee` for each property.
         * Iteratee functions may exit iteration early by explicitly returning `false`.
         *
         * @private
         * @param {Object} object The object to iterate over.
         * @param {Function} iteratee The function invoked per iteration.
         * @param {Function} keysFunc The function to get the keys of `object`.
         * @returns {Object} Returns `object`.
         */
        var baseFor = createBaseFor();

        /**
         * This function is like `baseFor` except that it iterates over properties
         * in the opposite order.
         *
         * @private
         * @param {Object} object The object to iterate over.
         * @param {Function} iteratee The function invoked per iteration.
         * @param {Function} keysFunc The function to get the keys of `object`.
         * @returns {Object} Returns `object`.
         */
        var baseForRight = createBaseFor(true);

        /**
         * The base implementation of `_.forOwn` without support for iteratee shorthands.
         *
         * @private
         * @param {Object} object The object to iterate over.
         * @param {Function} iteratee The function invoked per iteration.
         * @returns {Object} Returns `object`.
         */
        function baseForOwn(object, iteratee) {
          return object && baseFor(object, iteratee, keys);
        }

        /**
         * The base implementation of `_.forOwnRight` without support for iteratee shorthands.
         *
         * @private
         * @param {Object} object The object to iterate over.
         * @param {Function} iteratee The function invoked per iteration.
         * @returns {Object} Returns `object`.
         */
        function baseForOwnRight(object, iteratee) {
          return object && baseForRight(object, iteratee, keys);
        }

        /**
         * The base implementation of `_.functions` which creates an array of
         * `object` function property names filtered from `props`.
         *
         * @private
         * @param {Object} object The object to inspect.
         * @param {Array} props The property names to filter.
         * @returns {Array} Returns the function names.
         */
        function baseFunctions(object, props) {
          return arrayFilter(props, function(key) {
            return isFunction(object[key]);
          });
        }

        /**
         * The base implementation of `_.get` without support for default values.
         *
         * @private
         * @param {Object} object The object to query.
         * @param {Array|string} path The path of the property to get.
         * @returns {*} Returns the resolved value.
         */
        function baseGet(object, path) {
          path = castPath(path, object);

          var index = 0,
              length = path.length;

          while (object != null && index < length) {
            object = object[toKey(path[index++])];
          }
          return (index && index == length) ? object : undefined$1;
        }

        /**
         * The base implementation of `getAllKeys` and `getAllKeysIn` which uses
         * `keysFunc` and `symbolsFunc` to get the enumerable property names and
         * symbols of `object`.
         *
         * @private
         * @param {Object} object The object to query.
         * @param {Function} keysFunc The function to get the keys of `object`.
         * @param {Function} symbolsFunc The function to get the symbols of `object`.
         * @returns {Array} Returns the array of property names and symbols.
         */
        function baseGetAllKeys(object, keysFunc, symbolsFunc) {
          var result = keysFunc(object);
          return isArray(object) ? result : arrayPush(result, symbolsFunc(object));
        }

        /**
         * The base implementation of `getTag` without fallbacks for buggy environments.
         *
         * @private
         * @param {*} value The value to query.
         * @returns {string} Returns the `toStringTag`.
         */
        function baseGetTag(value) {
          if (value == null) {
            return value === undefined$1 ? undefinedTag : nullTag;
          }
          return (symToStringTag && symToStringTag in Object(value))
            ? getRawTag(value)
            : objectToString(value);
        }

        /**
         * The base implementation of `_.gt` which doesn't coerce arguments.
         *
         * @private
         * @param {*} value The value to compare.
         * @param {*} other The other value to compare.
         * @returns {boolean} Returns `true` if `value` is greater than `other`,
         *  else `false`.
         */
        function baseGt(value, other) {
          return value > other;
        }

        /**
         * The base implementation of `_.has` without support for deep paths.
         *
         * @private
         * @param {Object} [object] The object to query.
         * @param {Array|string} key The key to check.
         * @returns {boolean} Returns `true` if `key` exists, else `false`.
         */
        function baseHas(object, key) {
          return object != null && hasOwnProperty.call(object, key);
        }

        /**
         * The base implementation of `_.hasIn` without support for deep paths.
         *
         * @private
         * @param {Object} [object] The object to query.
         * @param {Array|string} key The key to check.
         * @returns {boolean} Returns `true` if `key` exists, else `false`.
         */
        function baseHasIn(object, key) {
          return object != null && key in Object(object);
        }

        /**
         * The base implementation of `_.inRange` which doesn't coerce arguments.
         *
         * @private
         * @param {number} number The number to check.
         * @param {number} start The start of the range.
         * @param {number} end The end of the range.
         * @returns {boolean} Returns `true` if `number` is in the range, else `false`.
         */
        function baseInRange(number, start, end) {
          return number >= nativeMin(start, end) && number < nativeMax(start, end);
        }

        /**
         * The base implementation of methods like `_.intersection`, without support
         * for iteratee shorthands, that accepts an array of arrays to inspect.
         *
         * @private
         * @param {Array} arrays The arrays to inspect.
         * @param {Function} [iteratee] The iteratee invoked per element.
         * @param {Function} [comparator] The comparator invoked per element.
         * @returns {Array} Returns the new array of shared values.
         */
        function baseIntersection(arrays, iteratee, comparator) {
          var includes = comparator ? arrayIncludesWith : arrayIncludes,
              length = arrays[0].length,
              othLength = arrays.length,
              othIndex = othLength,
              caches = Array(othLength),
              maxLength = Infinity,
              result = [];

          while (othIndex--) {
            var array = arrays[othIndex];
            if (othIndex && iteratee) {
              array = arrayMap(array, baseUnary(iteratee));
            }
            maxLength = nativeMin(array.length, maxLength);
            caches[othIndex] = !comparator && (iteratee || (length >= 120 && array.length >= 120))
              ? new SetCache(othIndex && array)
              : undefined$1;
          }
          array = arrays[0];

          var index = -1,
              seen = caches[0];

          outer:
          while (++index < length && result.length < maxLength) {
            var value = array[index],
                computed = iteratee ? iteratee(value) : value;

            value = (comparator || value !== 0) ? value : 0;
            if (!(seen
                  ? cacheHas(seen, computed)
                  : includes(result, computed, comparator)
                )) {
              othIndex = othLength;
              while (--othIndex) {
                var cache = caches[othIndex];
                if (!(cache
                      ? cacheHas(cache, computed)
                      : includes(arrays[othIndex], computed, comparator))
                    ) {
                  continue outer;
                }
              }
              if (seen) {
                seen.push(computed);
              }
              result.push(value);
            }
          }
          return result;
        }

        /**
         * The base implementation of `_.invert` and `_.invertBy` which inverts
         * `object` with values transformed by `iteratee` and set by `setter`.
         *
         * @private
         * @param {Object} object The object to iterate over.
         * @param {Function} setter The function to set `accumulator` values.
         * @param {Function} iteratee The iteratee to transform values.
         * @param {Object} accumulator The initial inverted object.
         * @returns {Function} Returns `accumulator`.
         */
        function baseInverter(object, setter, iteratee, accumulator) {
          baseForOwn(object, function(value, key, object) {
            setter(accumulator, iteratee(value), key, object);
          });
          return accumulator;
        }

        /**
         * The base implementation of `_.invoke` without support for individual
         * method arguments.
         *
         * @private
         * @param {Object} object The object to query.
         * @param {Array|string} path The path of the method to invoke.
         * @param {Array} args The arguments to invoke the method with.
         * @returns {*} Returns the result of the invoked method.
         */
        function baseInvoke(object, path, args) {
          path = castPath(path, object);
          object = parent(object, path);
          var func = object == null ? object : object[toKey(last(path))];
          return func == null ? undefined$1 : apply(func, object, args);
        }

        /**
         * The base implementation of `_.isArguments`.
         *
         * @private
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is an `arguments` object,
         */
        function baseIsArguments(value) {
          return isObjectLike(value) && baseGetTag(value) == argsTag;
        }

        /**
         * The base implementation of `_.isArrayBuffer` without Node.js optimizations.
         *
         * @private
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is an array buffer, else `false`.
         */
        function baseIsArrayBuffer(value) {
          return isObjectLike(value) && baseGetTag(value) == arrayBufferTag;
        }

        /**
         * The base implementation of `_.isDate` without Node.js optimizations.
         *
         * @private
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is a date object, else `false`.
         */
        function baseIsDate(value) {
          return isObjectLike(value) && baseGetTag(value) == dateTag;
        }

        /**
         * The base implementation of `_.isEqual` which supports partial comparisons
         * and tracks traversed objects.
         *
         * @private
         * @param {*} value The value to compare.
         * @param {*} other The other value to compare.
         * @param {boolean} bitmask The bitmask flags.
         *  1 - Unordered comparison
         *  2 - Partial comparison
         * @param {Function} [customizer] The function to customize comparisons.
         * @param {Object} [stack] Tracks traversed `value` and `other` objects.
         * @returns {boolean} Returns `true` if the values are equivalent, else `false`.
         */
        function baseIsEqual(value, other, bitmask, customizer, stack) {
          if (value === other) {
            return true;
          }
          if (value == null || other == null || (!isObjectLike(value) && !isObjectLike(other))) {
            return value !== value && other !== other;
          }
          return baseIsEqualDeep(value, other, bitmask, customizer, baseIsEqual, stack);
        }

        /**
         * A specialized version of `baseIsEqual` for arrays and objects which performs
         * deep comparisons and tracks traversed objects enabling objects with circular
         * references to be compared.
         *
         * @private
         * @param {Object} object The object to compare.
         * @param {Object} other The other object to compare.
         * @param {number} bitmask The bitmask flags. See `baseIsEqual` for more details.
         * @param {Function} customizer The function to customize comparisons.
         * @param {Function} equalFunc The function to determine equivalents of values.
         * @param {Object} [stack] Tracks traversed `object` and `other` objects.
         * @returns {boolean} Returns `true` if the objects are equivalent, else `false`.
         */
        function baseIsEqualDeep(object, other, bitmask, customizer, equalFunc, stack) {
          var objIsArr = isArray(object),
              othIsArr = isArray(other),
              objTag = objIsArr ? arrayTag : getTag(object),
              othTag = othIsArr ? arrayTag : getTag(other);

          objTag = objTag == argsTag ? objectTag : objTag;
          othTag = othTag == argsTag ? objectTag : othTag;

          var objIsObj = objTag == objectTag,
              othIsObj = othTag == objectTag,
              isSameTag = objTag == othTag;

          if (isSameTag && isBuffer(object)) {
            if (!isBuffer(other)) {
              return false;
            }
            objIsArr = true;
            objIsObj = false;
          }
          if (isSameTag && !objIsObj) {
            stack || (stack = new Stack);
            return (objIsArr || isTypedArray(object))
              ? equalArrays(object, other, bitmask, customizer, equalFunc, stack)
              : equalByTag(object, other, objTag, bitmask, customizer, equalFunc, stack);
          }
          if (!(bitmask & COMPARE_PARTIAL_FLAG)) {
            var objIsWrapped = objIsObj && hasOwnProperty.call(object, '__wrapped__'),
                othIsWrapped = othIsObj && hasOwnProperty.call(other, '__wrapped__');

            if (objIsWrapped || othIsWrapped) {
              var objUnwrapped = objIsWrapped ? object.value() : object,
                  othUnwrapped = othIsWrapped ? other.value() : other;

              stack || (stack = new Stack);
              return equalFunc(objUnwrapped, othUnwrapped, bitmask, customizer, stack);
            }
          }
          if (!isSameTag) {
            return false;
          }
          stack || (stack = new Stack);
          return equalObjects(object, other, bitmask, customizer, equalFunc, stack);
        }

        /**
         * The base implementation of `_.isMap` without Node.js optimizations.
         *
         * @private
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is a map, else `false`.
         */
        function baseIsMap(value) {
          return isObjectLike(value) && getTag(value) == mapTag;
        }

        /**
         * The base implementation of `_.isMatch` without support for iteratee shorthands.
         *
         * @private
         * @param {Object} object The object to inspect.
         * @param {Object} source The object of property values to match.
         * @param {Array} matchData The property names, values, and compare flags to match.
         * @param {Function} [customizer] The function to customize comparisons.
         * @returns {boolean} Returns `true` if `object` is a match, else `false`.
         */
        function baseIsMatch(object, source, matchData, customizer) {
          var index = matchData.length,
              length = index,
              noCustomizer = !customizer;

          if (object == null) {
            return !length;
          }
          object = Object(object);
          while (index--) {
            var data = matchData[index];
            if ((noCustomizer && data[2])
                  ? data[1] !== object[data[0]]
                  : !(data[0] in object)
                ) {
              return false;
            }
          }
          while (++index < length) {
            data = matchData[index];
            var key = data[0],
                objValue = object[key],
                srcValue = data[1];

            if (noCustomizer && data[2]) {
              if (objValue === undefined$1 && !(key in object)) {
                return false;
              }
            } else {
              var stack = new Stack;
              if (customizer) {
                var result = customizer(objValue, srcValue, key, object, source, stack);
              }
              if (!(result === undefined$1
                    ? baseIsEqual(srcValue, objValue, COMPARE_PARTIAL_FLAG | COMPARE_UNORDERED_FLAG, customizer, stack)
                    : result
                  )) {
                return false;
              }
            }
          }
          return true;
        }

        /**
         * The base implementation of `_.isNative` without bad shim checks.
         *
         * @private
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is a native function,
         *  else `false`.
         */
        function baseIsNative(value) {
          if (!isObject(value) || isMasked(value)) {
            return false;
          }
          var pattern = isFunction(value) ? reIsNative : reIsHostCtor;
          return pattern.test(toSource(value));
        }

        /**
         * The base implementation of `_.isRegExp` without Node.js optimizations.
         *
         * @private
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is a regexp, else `false`.
         */
        function baseIsRegExp(value) {
          return isObjectLike(value) && baseGetTag(value) == regexpTag;
        }

        /**
         * The base implementation of `_.isSet` without Node.js optimizations.
         *
         * @private
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is a set, else `false`.
         */
        function baseIsSet(value) {
          return isObjectLike(value) && getTag(value) == setTag;
        }

        /**
         * The base implementation of `_.isTypedArray` without Node.js optimizations.
         *
         * @private
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is a typed array, else `false`.
         */
        function baseIsTypedArray(value) {
          return isObjectLike(value) &&
            isLength(value.length) && !!typedArrayTags[baseGetTag(value)];
        }

        /**
         * The base implementation of `_.iteratee`.
         *
         * @private
         * @param {*} [value=_.identity] The value to convert to an iteratee.
         * @returns {Function} Returns the iteratee.
         */
        function baseIteratee(value) {
          // Don't store the `typeof` result in a variable to avoid a JIT bug in Safari 9.
          // See https://bugs.webkit.org/show_bug.cgi?id=156034 for more details.
          if (typeof value == 'function') {
            return value;
          }
          if (value == null) {
            return identity;
          }
          if (typeof value == 'object') {
            return isArray(value)
              ? baseMatchesProperty(value[0], value[1])
              : baseMatches(value);
          }
          return property(value);
        }

        /**
         * The base implementation of `_.keys` which doesn't treat sparse arrays as dense.
         *
         * @private
         * @param {Object} object The object to query.
         * @returns {Array} Returns the array of property names.
         */
        function baseKeys(object) {
          if (!isPrototype(object)) {
            return nativeKeys(object);
          }
          var result = [];
          for (var key in Object(object)) {
            if (hasOwnProperty.call(object, key) && key != 'constructor') {
              result.push(key);
            }
          }
          return result;
        }

        /**
         * The base implementation of `_.keysIn` which doesn't treat sparse arrays as dense.
         *
         * @private
         * @param {Object} object The object to query.
         * @returns {Array} Returns the array of property names.
         */
        function baseKeysIn(object) {
          if (!isObject(object)) {
            return nativeKeysIn(object);
          }
          var isProto = isPrototype(object),
              result = [];

          for (var key in object) {
            if (!(key == 'constructor' && (isProto || !hasOwnProperty.call(object, key)))) {
              result.push(key);
            }
          }
          return result;
        }

        /**
         * The base implementation of `_.lt` which doesn't coerce arguments.
         *
         * @private
         * @param {*} value The value to compare.
         * @param {*} other The other value to compare.
         * @returns {boolean} Returns `true` if `value` is less than `other`,
         *  else `false`.
         */
        function baseLt(value, other) {
          return value < other;
        }

        /**
         * The base implementation of `_.map` without support for iteratee shorthands.
         *
         * @private
         * @param {Array|Object} collection The collection to iterate over.
         * @param {Function} iteratee The function invoked per iteration.
         * @returns {Array} Returns the new mapped array.
         */
        function baseMap(collection, iteratee) {
          var index = -1,
              result = isArrayLike(collection) ? Array(collection.length) : [];

          baseEach(collection, function(value, key, collection) {
            result[++index] = iteratee(value, key, collection);
          });
          return result;
        }

        /**
         * The base implementation of `_.matches` which doesn't clone `source`.
         *
         * @private
         * @param {Object} source The object of property values to match.
         * @returns {Function} Returns the new spec function.
         */
        function baseMatches(source) {
          var matchData = getMatchData(source);
          if (matchData.length == 1 && matchData[0][2]) {
            return matchesStrictComparable(matchData[0][0], matchData[0][1]);
          }
          return function(object) {
            return object === source || baseIsMatch(object, source, matchData);
          };
        }

        /**
         * The base implementation of `_.matchesProperty` which doesn't clone `srcValue`.
         *
         * @private
         * @param {string} path The path of the property to get.
         * @param {*} srcValue The value to match.
         * @returns {Function} Returns the new spec function.
         */
        function baseMatchesProperty(path, srcValue) {
          if (isKey(path) && isStrictComparable(srcValue)) {
            return matchesStrictComparable(toKey(path), srcValue);
          }
          return function(object) {
            var objValue = get(object, path);
            return (objValue === undefined$1 && objValue === srcValue)
              ? hasIn(object, path)
              : baseIsEqual(srcValue, objValue, COMPARE_PARTIAL_FLAG | COMPARE_UNORDERED_FLAG);
          };
        }

        /**
         * The base implementation of `_.merge` without support for multiple sources.
         *
         * @private
         * @param {Object} object The destination object.
         * @param {Object} source The source object.
         * @param {number} srcIndex The index of `source`.
         * @param {Function} [customizer] The function to customize merged values.
         * @param {Object} [stack] Tracks traversed source values and their merged
         *  counterparts.
         */
        function baseMerge(object, source, srcIndex, customizer, stack) {
          if (object === source) {
            return;
          }
          baseFor(source, function(srcValue, key) {
            stack || (stack = new Stack);
            if (isObject(srcValue)) {
              baseMergeDeep(object, source, key, srcIndex, baseMerge, customizer, stack);
            }
            else {
              var newValue = customizer
                ? customizer(safeGet(object, key), srcValue, (key + ''), object, source, stack)
                : undefined$1;

              if (newValue === undefined$1) {
                newValue = srcValue;
              }
              assignMergeValue(object, key, newValue);
            }
          }, keysIn);
        }

        /**
         * A specialized version of `baseMerge` for arrays and objects which performs
         * deep merges and tracks traversed objects enabling objects with circular
         * references to be merged.
         *
         * @private
         * @param {Object} object The destination object.
         * @param {Object} source The source object.
         * @param {string} key The key of the value to merge.
         * @param {number} srcIndex The index of `source`.
         * @param {Function} mergeFunc The function to merge values.
         * @param {Function} [customizer] The function to customize assigned values.
         * @param {Object} [stack] Tracks traversed source values and their merged
         *  counterparts.
         */
        function baseMergeDeep(object, source, key, srcIndex, mergeFunc, customizer, stack) {
          var objValue = safeGet(object, key),
              srcValue = safeGet(source, key),
              stacked = stack.get(srcValue);

          if (stacked) {
            assignMergeValue(object, key, stacked);
            return;
          }
          var newValue = customizer
            ? customizer(objValue, srcValue, (key + ''), object, source, stack)
            : undefined$1;

          var isCommon = newValue === undefined$1;

          if (isCommon) {
            var isArr = isArray(srcValue),
                isBuff = !isArr && isBuffer(srcValue),
                isTyped = !isArr && !isBuff && isTypedArray(srcValue);

            newValue = srcValue;
            if (isArr || isBuff || isTyped) {
              if (isArray(objValue)) {
                newValue = objValue;
              }
              else if (isArrayLikeObject(objValue)) {
                newValue = copyArray(objValue);
              }
              else if (isBuff) {
                isCommon = false;
                newValue = cloneBuffer(srcValue, true);
              }
              else if (isTyped) {
                isCommon = false;
                newValue = cloneTypedArray(srcValue, true);
              }
              else {
                newValue = [];
              }
            }
            else if (isPlainObject(srcValue) || isArguments(srcValue)) {
              newValue = objValue;
              if (isArguments(objValue)) {
                newValue = toPlainObject(objValue);
              }
              else if (!isObject(objValue) || isFunction(objValue)) {
                newValue = initCloneObject(srcValue);
              }
            }
            else {
              isCommon = false;
            }
          }
          if (isCommon) {
            // Recursively merge objects and arrays (susceptible to call stack limits).
            stack.set(srcValue, newValue);
            mergeFunc(newValue, srcValue, srcIndex, customizer, stack);
            stack['delete'](srcValue);
          }
          assignMergeValue(object, key, newValue);
        }

        /**
         * The base implementation of `_.nth` which doesn't coerce arguments.
         *
         * @private
         * @param {Array} array The array to query.
         * @param {number} n The index of the element to return.
         * @returns {*} Returns the nth element of `array`.
         */
        function baseNth(array, n) {
          var length = array.length;
          if (!length) {
            return;
          }
          n += n < 0 ? length : 0;
          return isIndex(n, length) ? array[n] : undefined$1;
        }

        /**
         * The base implementation of `_.orderBy` without param guards.
         *
         * @private
         * @param {Array|Object} collection The collection to iterate over.
         * @param {Function[]|Object[]|string[]} iteratees The iteratees to sort by.
         * @param {string[]} orders The sort orders of `iteratees`.
         * @returns {Array} Returns the new sorted array.
         */
        function baseOrderBy(collection, iteratees, orders) {
          if (iteratees.length) {
            iteratees = arrayMap(iteratees, function(iteratee) {
              if (isArray(iteratee)) {
                return function(value) {
                  return baseGet(value, iteratee.length === 1 ? iteratee[0] : iteratee);
                }
              }
              return iteratee;
            });
          } else {
            iteratees = [identity];
          }

          var index = -1;
          iteratees = arrayMap(iteratees, baseUnary(getIteratee()));

          var result = baseMap(collection, function(value, key, collection) {
            var criteria = arrayMap(iteratees, function(iteratee) {
              return iteratee(value);
            });
            return { 'criteria': criteria, 'index': ++index, 'value': value };
          });

          return baseSortBy(result, function(object, other) {
            return compareMultiple(object, other, orders);
          });
        }

        /**
         * The base implementation of `_.pick` without support for individual
         * property identifiers.
         *
         * @private
         * @param {Object} object The source object.
         * @param {string[]} paths The property paths to pick.
         * @returns {Object} Returns the new object.
         */
        function basePick(object, paths) {
          return basePickBy(object, paths, function(value, path) {
            return hasIn(object, path);
          });
        }

        /**
         * The base implementation of  `_.pickBy` without support for iteratee shorthands.
         *
         * @private
         * @param {Object} object The source object.
         * @param {string[]} paths The property paths to pick.
         * @param {Function} predicate The function invoked per property.
         * @returns {Object} Returns the new object.
         */
        function basePickBy(object, paths, predicate) {
          var index = -1,
              length = paths.length,
              result = {};

          while (++index < length) {
            var path = paths[index],
                value = baseGet(object, path);

            if (predicate(value, path)) {
              baseSet(result, castPath(path, object), value);
            }
          }
          return result;
        }

        /**
         * A specialized version of `baseProperty` which supports deep paths.
         *
         * @private
         * @param {Array|string} path The path of the property to get.
         * @returns {Function} Returns the new accessor function.
         */
        function basePropertyDeep(path) {
          return function(object) {
            return baseGet(object, path);
          };
        }

        /**
         * The base implementation of `_.pullAllBy` without support for iteratee
         * shorthands.
         *
         * @private
         * @param {Array} array The array to modify.
         * @param {Array} values The values to remove.
         * @param {Function} [iteratee] The iteratee invoked per element.
         * @param {Function} [comparator] The comparator invoked per element.
         * @returns {Array} Returns `array`.
         */
        function basePullAll(array, values, iteratee, comparator) {
          var indexOf = comparator ? baseIndexOfWith : baseIndexOf,
              index = -1,
              length = values.length,
              seen = array;

          if (array === values) {
            values = copyArray(values);
          }
          if (iteratee) {
            seen = arrayMap(array, baseUnary(iteratee));
          }
          while (++index < length) {
            var fromIndex = 0,
                value = values[index],
                computed = iteratee ? iteratee(value) : value;

            while ((fromIndex = indexOf(seen, computed, fromIndex, comparator)) > -1) {
              if (seen !== array) {
                splice.call(seen, fromIndex, 1);
              }
              splice.call(array, fromIndex, 1);
            }
          }
          return array;
        }

        /**
         * The base implementation of `_.pullAt` without support for individual
         * indexes or capturing the removed elements.
         *
         * @private
         * @param {Array} array The array to modify.
         * @param {number[]} indexes The indexes of elements to remove.
         * @returns {Array} Returns `array`.
         */
        function basePullAt(array, indexes) {
          var length = array ? indexes.length : 0,
              lastIndex = length - 1;

          while (length--) {
            var index = indexes[length];
            if (length == lastIndex || index !== previous) {
              var previous = index;
              if (isIndex(index)) {
                splice.call(array, index, 1);
              } else {
                baseUnset(array, index);
              }
            }
          }
          return array;
        }

        /**
         * The base implementation of `_.random` without support for returning
         * floating-point numbers.
         *
         * @private
         * @param {number} lower The lower bound.
         * @param {number} upper The upper bound.
         * @returns {number} Returns the random number.
         */
        function baseRandom(lower, upper) {
          return lower + nativeFloor(nativeRandom() * (upper - lower + 1));
        }

        /**
         * The base implementation of `_.range` and `_.rangeRight` which doesn't
         * coerce arguments.
         *
         * @private
         * @param {number} start The start of the range.
         * @param {number} end The end of the range.
         * @param {number} step The value to increment or decrement by.
         * @param {boolean} [fromRight] Specify iterating from right to left.
         * @returns {Array} Returns the range of numbers.
         */
        function baseRange(start, end, step, fromRight) {
          var index = -1,
              length = nativeMax(nativeCeil((end - start) / (step || 1)), 0),
              result = Array(length);

          while (length--) {
            result[fromRight ? length : ++index] = start;
            start += step;
          }
          return result;
        }

        /**
         * The base implementation of `_.repeat` which doesn't coerce arguments.
         *
         * @private
         * @param {string} string The string to repeat.
         * @param {number} n The number of times to repeat the string.
         * @returns {string} Returns the repeated string.
         */
        function baseRepeat(string, n) {
          var result = '';
          if (!string || n < 1 || n > MAX_SAFE_INTEGER) {
            return result;
          }
          // Leverage the exponentiation by squaring algorithm for a faster repeat.
          // See https://en.wikipedia.org/wiki/Exponentiation_by_squaring for more details.
          do {
            if (n % 2) {
              result += string;
            }
            n = nativeFloor(n / 2);
            if (n) {
              string += string;
            }
          } while (n);

          return result;
        }

        /**
         * The base implementation of `_.rest` which doesn't validate or coerce arguments.
         *
         * @private
         * @param {Function} func The function to apply a rest parameter to.
         * @param {number} [start=func.length-1] The start position of the rest parameter.
         * @returns {Function} Returns the new function.
         */
        function baseRest(func, start) {
          return setToString(overRest(func, start, identity), func + '');
        }

        /**
         * The base implementation of `_.sample`.
         *
         * @private
         * @param {Array|Object} collection The collection to sample.
         * @returns {*} Returns the random element.
         */
        function baseSample(collection) {
          return arraySample(values(collection));
        }

        /**
         * The base implementation of `_.sampleSize` without param guards.
         *
         * @private
         * @param {Array|Object} collection The collection to sample.
         * @param {number} n The number of elements to sample.
         * @returns {Array} Returns the random elements.
         */
        function baseSampleSize(collection, n) {
          var array = values(collection);
          return shuffleSelf(array, baseClamp(n, 0, array.length));
        }

        /**
         * The base implementation of `_.set`.
         *
         * @private
         * @param {Object} object The object to modify.
         * @param {Array|string} path The path of the property to set.
         * @param {*} value The value to set.
         * @param {Function} [customizer] The function to customize path creation.
         * @returns {Object} Returns `object`.
         */
        function baseSet(object, path, value, customizer) {
          if (!isObject(object)) {
            return object;
          }
          path = castPath(path, object);

          var index = -1,
              length = path.length,
              lastIndex = length - 1,
              nested = object;

          while (nested != null && ++index < length) {
            var key = toKey(path[index]),
                newValue = value;

            if (key === '__proto__' || key === 'constructor' || key === 'prototype') {
              return object;
            }

            if (index != lastIndex) {
              var objValue = nested[key];
              newValue = customizer ? customizer(objValue, key, nested) : undefined$1;
              if (newValue === undefined$1) {
                newValue = isObject(objValue)
                  ? objValue
                  : (isIndex(path[index + 1]) ? [] : {});
              }
            }
            assignValue(nested, key, newValue);
            nested = nested[key];
          }
          return object;
        }

        /**
         * The base implementation of `setData` without support for hot loop shorting.
         *
         * @private
         * @param {Function} func The function to associate metadata with.
         * @param {*} data The metadata.
         * @returns {Function} Returns `func`.
         */
        var baseSetData = !metaMap ? identity : function(func, data) {
          metaMap.set(func, data);
          return func;
        };

        /**
         * The base implementation of `setToString` without support for hot loop shorting.
         *
         * @private
         * @param {Function} func The function to modify.
         * @param {Function} string The `toString` result.
         * @returns {Function} Returns `func`.
         */
        var baseSetToString = !defineProperty ? identity : function(func, string) {
          return defineProperty(func, 'toString', {
            'configurable': true,
            'enumerable': false,
            'value': constant(string),
            'writable': true
          });
        };

        /**
         * The base implementation of `_.shuffle`.
         *
         * @private
         * @param {Array|Object} collection The collection to shuffle.
         * @returns {Array} Returns the new shuffled array.
         */
        function baseShuffle(collection) {
          return shuffleSelf(values(collection));
        }

        /**
         * The base implementation of `_.slice` without an iteratee call guard.
         *
         * @private
         * @param {Array} array The array to slice.
         * @param {number} [start=0] The start position.
         * @param {number} [end=array.length] The end position.
         * @returns {Array} Returns the slice of `array`.
         */
        function baseSlice(array, start, end) {
          var index = -1,
              length = array.length;

          if (start < 0) {
            start = -start > length ? 0 : (length + start);
          }
          end = end > length ? length : end;
          if (end < 0) {
            end += length;
          }
          length = start > end ? 0 : ((end - start) >>> 0);
          start >>>= 0;

          var result = Array(length);
          while (++index < length) {
            result[index] = array[index + start];
          }
          return result;
        }

        /**
         * The base implementation of `_.some` without support for iteratee shorthands.
         *
         * @private
         * @param {Array|Object} collection The collection to iterate over.
         * @param {Function} predicate The function invoked per iteration.
         * @returns {boolean} Returns `true` if any element passes the predicate check,
         *  else `false`.
         */
        function baseSome(collection, predicate) {
          var result;

          baseEach(collection, function(value, index, collection) {
            result = predicate(value, index, collection);
            return !result;
          });
          return !!result;
        }

        /**
         * The base implementation of `_.sortedIndex` and `_.sortedLastIndex` which
         * performs a binary search of `array` to determine the index at which `value`
         * should be inserted into `array` in order to maintain its sort order.
         *
         * @private
         * @param {Array} array The sorted array to inspect.
         * @param {*} value The value to evaluate.
         * @param {boolean} [retHighest] Specify returning the highest qualified index.
         * @returns {number} Returns the index at which `value` should be inserted
         *  into `array`.
         */
        function baseSortedIndex(array, value, retHighest) {
          var low = 0,
              high = array == null ? low : array.length;

          if (typeof value == 'number' && value === value && high <= HALF_MAX_ARRAY_LENGTH) {
            while (low < high) {
              var mid = (low + high) >>> 1,
                  computed = array[mid];

              if (computed !== null && !isSymbol(computed) &&
                  (retHighest ? (computed <= value) : (computed < value))) {
                low = mid + 1;
              } else {
                high = mid;
              }
            }
            return high;
          }
          return baseSortedIndexBy(array, value, identity, retHighest);
        }

        /**
         * The base implementation of `_.sortedIndexBy` and `_.sortedLastIndexBy`
         * which invokes `iteratee` for `value` and each element of `array` to compute
         * their sort ranking. The iteratee is invoked with one argument; (value).
         *
         * @private
         * @param {Array} array The sorted array to inspect.
         * @param {*} value The value to evaluate.
         * @param {Function} iteratee The iteratee invoked per element.
         * @param {boolean} [retHighest] Specify returning the highest qualified index.
         * @returns {number} Returns the index at which `value` should be inserted
         *  into `array`.
         */
        function baseSortedIndexBy(array, value, iteratee, retHighest) {
          var low = 0,
              high = array == null ? 0 : array.length;
          if (high === 0) {
            return 0;
          }

          value = iteratee(value);
          var valIsNaN = value !== value,
              valIsNull = value === null,
              valIsSymbol = isSymbol(value),
              valIsUndefined = value === undefined$1;

          while (low < high) {
            var mid = nativeFloor((low + high) / 2),
                computed = iteratee(array[mid]),
                othIsDefined = computed !== undefined$1,
                othIsNull = computed === null,
                othIsReflexive = computed === computed,
                othIsSymbol = isSymbol(computed);

            if (valIsNaN) {
              var setLow = retHighest || othIsReflexive;
            } else if (valIsUndefined) {
              setLow = othIsReflexive && (retHighest || othIsDefined);
            } else if (valIsNull) {
              setLow = othIsReflexive && othIsDefined && (retHighest || !othIsNull);
            } else if (valIsSymbol) {
              setLow = othIsReflexive && othIsDefined && !othIsNull && (retHighest || !othIsSymbol);
            } else if (othIsNull || othIsSymbol) {
              setLow = false;
            } else {
              setLow = retHighest ? (computed <= value) : (computed < value);
            }
            if (setLow) {
              low = mid + 1;
            } else {
              high = mid;
            }
          }
          return nativeMin(high, MAX_ARRAY_INDEX);
        }

        /**
         * The base implementation of `_.sortedUniq` and `_.sortedUniqBy` without
         * support for iteratee shorthands.
         *
         * @private
         * @param {Array} array The array to inspect.
         * @param {Function} [iteratee] The iteratee invoked per element.
         * @returns {Array} Returns the new duplicate free array.
         */
        function baseSortedUniq(array, iteratee) {
          var index = -1,
              length = array.length,
              resIndex = 0,
              result = [];

          while (++index < length) {
            var value = array[index],
                computed = iteratee ? iteratee(value) : value;

            if (!index || !eq(computed, seen)) {
              var seen = computed;
              result[resIndex++] = value === 0 ? 0 : value;
            }
          }
          return result;
        }

        /**
         * The base implementation of `_.toNumber` which doesn't ensure correct
         * conversions of binary, hexadecimal, or octal string values.
         *
         * @private
         * @param {*} value The value to process.
         * @returns {number} Returns the number.
         */
        function baseToNumber(value) {
          if (typeof value == 'number') {
            return value;
          }
          if (isSymbol(value)) {
            return NAN;
          }
          return +value;
        }

        /**
         * The base implementation of `_.toString` which doesn't convert nullish
         * values to empty strings.
         *
         * @private
         * @param {*} value The value to process.
         * @returns {string} Returns the string.
         */
        function baseToString(value) {
          // Exit early for strings to avoid a performance hit in some environments.
          if (typeof value == 'string') {
            return value;
          }
          if (isArray(value)) {
            // Recursively convert values (susceptible to call stack limits).
            return arrayMap(value, baseToString) + '';
          }
          if (isSymbol(value)) {
            return symbolToString ? symbolToString.call(value) : '';
          }
          var result = (value + '');
          return (result == '0' && (1 / value) == -INFINITY) ? '-0' : result;
        }

        /**
         * The base implementation of `_.uniqBy` without support for iteratee shorthands.
         *
         * @private
         * @param {Array} array The array to inspect.
         * @param {Function} [iteratee] The iteratee invoked per element.
         * @param {Function} [comparator] The comparator invoked per element.
         * @returns {Array} Returns the new duplicate free array.
         */
        function baseUniq(array, iteratee, comparator) {
          var index = -1,
              includes = arrayIncludes,
              length = array.length,
              isCommon = true,
              result = [],
              seen = result;

          if (comparator) {
            isCommon = false;
            includes = arrayIncludesWith;
          }
          else if (length >= LARGE_ARRAY_SIZE) {
            var set = iteratee ? null : createSet(array);
            if (set) {
              return setToArray(set);
            }
            isCommon = false;
            includes = cacheHas;
            seen = new SetCache;
          }
          else {
            seen = iteratee ? [] : result;
          }
          outer:
          while (++index < length) {
            var value = array[index],
                computed = iteratee ? iteratee(value) : value;

            value = (comparator || value !== 0) ? value : 0;
            if (isCommon && computed === computed) {
              var seenIndex = seen.length;
              while (seenIndex--) {
                if (seen[seenIndex] === computed) {
                  continue outer;
                }
              }
              if (iteratee) {
                seen.push(computed);
              }
              result.push(value);
            }
            else if (!includes(seen, computed, comparator)) {
              if (seen !== result) {
                seen.push(computed);
              }
              result.push(value);
            }
          }
          return result;
        }

        /**
         * The base implementation of `_.unset`.
         *
         * @private
         * @param {Object} object The object to modify.
         * @param {Array|string} path The property path to unset.
         * @returns {boolean} Returns `true` if the property is deleted, else `false`.
         */
        function baseUnset(object, path) {
          path = castPath(path, object);
          object = parent(object, path);
          return object == null || delete object[toKey(last(path))];
        }

        /**
         * The base implementation of `_.update`.
         *
         * @private
         * @param {Object} object The object to modify.
         * @param {Array|string} path The path of the property to update.
         * @param {Function} updater The function to produce the updated value.
         * @param {Function} [customizer] The function to customize path creation.
         * @returns {Object} Returns `object`.
         */
        function baseUpdate(object, path, updater, customizer) {
          return baseSet(object, path, updater(baseGet(object, path)), customizer);
        }

        /**
         * The base implementation of methods like `_.dropWhile` and `_.takeWhile`
         * without support for iteratee shorthands.
         *
         * @private
         * @param {Array} array The array to query.
         * @param {Function} predicate The function invoked per iteration.
         * @param {boolean} [isDrop] Specify dropping elements instead of taking them.
         * @param {boolean} [fromRight] Specify iterating from right to left.
         * @returns {Array} Returns the slice of `array`.
         */
        function baseWhile(array, predicate, isDrop, fromRight) {
          var length = array.length,
              index = fromRight ? length : -1;

          while ((fromRight ? index-- : ++index < length) &&
            predicate(array[index], index, array)) {}

          return isDrop
            ? baseSlice(array, (fromRight ? 0 : index), (fromRight ? index + 1 : length))
            : baseSlice(array, (fromRight ? index + 1 : 0), (fromRight ? length : index));
        }

        /**
         * The base implementation of `wrapperValue` which returns the result of
         * performing a sequence of actions on the unwrapped `value`, where each
         * successive action is supplied the return value of the previous.
         *
         * @private
         * @param {*} value The unwrapped value.
         * @param {Array} actions Actions to perform to resolve the unwrapped value.
         * @returns {*} Returns the resolved value.
         */
        function baseWrapperValue(value, actions) {
          var result = value;
          if (result instanceof LazyWrapper) {
            result = result.value();
          }
          return arrayReduce(actions, function(result, action) {
            return action.func.apply(action.thisArg, arrayPush([result], action.args));
          }, result);
        }

        /**
         * The base implementation of methods like `_.xor`, without support for
         * iteratee shorthands, that accepts an array of arrays to inspect.
         *
         * @private
         * @param {Array} arrays The arrays to inspect.
         * @param {Function} [iteratee] The iteratee invoked per element.
         * @param {Function} [comparator] The comparator invoked per element.
         * @returns {Array} Returns the new array of values.
         */
        function baseXor(arrays, iteratee, comparator) {
          var length = arrays.length;
          if (length < 2) {
            return length ? baseUniq(arrays[0]) : [];
          }
          var index = -1,
              result = Array(length);

          while (++index < length) {
            var array = arrays[index],
                othIndex = -1;

            while (++othIndex < length) {
              if (othIndex != index) {
                result[index] = baseDifference(result[index] || array, arrays[othIndex], iteratee, comparator);
              }
            }
          }
          return baseUniq(baseFlatten(result, 1), iteratee, comparator);
        }

        /**
         * This base implementation of `_.zipObject` which assigns values using `assignFunc`.
         *
         * @private
         * @param {Array} props The property identifiers.
         * @param {Array} values The property values.
         * @param {Function} assignFunc The function to assign values.
         * @returns {Object} Returns the new object.
         */
        function baseZipObject(props, values, assignFunc) {
          var index = -1,
              length = props.length,
              valsLength = values.length,
              result = {};

          while (++index < length) {
            var value = index < valsLength ? values[index] : undefined$1;
            assignFunc(result, props[index], value);
          }
          return result;
        }

        /**
         * Casts `value` to an empty array if it's not an array like object.
         *
         * @private
         * @param {*} value The value to inspect.
         * @returns {Array|Object} Returns the cast array-like object.
         */
        function castArrayLikeObject(value) {
          return isArrayLikeObject(value) ? value : [];
        }

        /**
         * Casts `value` to `identity` if it's not a function.
         *
         * @private
         * @param {*} value The value to inspect.
         * @returns {Function} Returns cast function.
         */
        function castFunction(value) {
          return typeof value == 'function' ? value : identity;
        }

        /**
         * Casts `value` to a path array if it's not one.
         *
         * @private
         * @param {*} value The value to inspect.
         * @param {Object} [object] The object to query keys on.
         * @returns {Array} Returns the cast property path array.
         */
        function castPath(value, object) {
          if (isArray(value)) {
            return value;
          }
          return isKey(value, object) ? [value] : stringToPath(toString(value));
        }

        /**
         * A `baseRest` alias which can be replaced with `identity` by module
         * replacement plugins.
         *
         * @private
         * @type {Function}
         * @param {Function} func The function to apply a rest parameter to.
         * @returns {Function} Returns the new function.
         */
        var castRest = baseRest;

        /**
         * Casts `array` to a slice if it's needed.
         *
         * @private
         * @param {Array} array The array to inspect.
         * @param {number} start The start position.
         * @param {number} [end=array.length] The end position.
         * @returns {Array} Returns the cast slice.
         */
        function castSlice(array, start, end) {
          var length = array.length;
          end = end === undefined$1 ? length : end;
          return (!start && end >= length) ? array : baseSlice(array, start, end);
        }

        /**
         * A simple wrapper around the global [`clearTimeout`](https://mdn.io/clearTimeout).
         *
         * @private
         * @param {number|Object} id The timer id or timeout object of the timer to clear.
         */
        var clearTimeout = ctxClearTimeout || function(id) {
          return root.clearTimeout(id);
        };

        /**
         * Creates a clone of  `buffer`.
         *
         * @private
         * @param {Buffer} buffer The buffer to clone.
         * @param {boolean} [isDeep] Specify a deep clone.
         * @returns {Buffer} Returns the cloned buffer.
         */
        function cloneBuffer(buffer, isDeep) {
          if (isDeep) {
            return buffer.slice();
          }
          var length = buffer.length,
              result = allocUnsafe ? allocUnsafe(length) : new buffer.constructor(length);

          buffer.copy(result);
          return result;
        }

        /**
         * Creates a clone of `arrayBuffer`.
         *
         * @private
         * @param {ArrayBuffer} arrayBuffer The array buffer to clone.
         * @returns {ArrayBuffer} Returns the cloned array buffer.
         */
        function cloneArrayBuffer(arrayBuffer) {
          var result = new arrayBuffer.constructor(arrayBuffer.byteLength);
          new Uint8Array(result).set(new Uint8Array(arrayBuffer));
          return result;
        }

        /**
         * Creates a clone of `dataView`.
         *
         * @private
         * @param {Object} dataView The data view to clone.
         * @param {boolean} [isDeep] Specify a deep clone.
         * @returns {Object} Returns the cloned data view.
         */
        function cloneDataView(dataView, isDeep) {
          var buffer = isDeep ? cloneArrayBuffer(dataView.buffer) : dataView.buffer;
          return new dataView.constructor(buffer, dataView.byteOffset, dataView.byteLength);
        }

        /**
         * Creates a clone of `regexp`.
         *
         * @private
         * @param {Object} regexp The regexp to clone.
         * @returns {Object} Returns the cloned regexp.
         */
        function cloneRegExp(regexp) {
          var result = new regexp.constructor(regexp.source, reFlags.exec(regexp));
          result.lastIndex = regexp.lastIndex;
          return result;
        }

        /**
         * Creates a clone of the `symbol` object.
         *
         * @private
         * @param {Object} symbol The symbol object to clone.
         * @returns {Object} Returns the cloned symbol object.
         */
        function cloneSymbol(symbol) {
          return symbolValueOf ? Object(symbolValueOf.call(symbol)) : {};
        }

        /**
         * Creates a clone of `typedArray`.
         *
         * @private
         * @param {Object} typedArray The typed array to clone.
         * @param {boolean} [isDeep] Specify a deep clone.
         * @returns {Object} Returns the cloned typed array.
         */
        function cloneTypedArray(typedArray, isDeep) {
          var buffer = isDeep ? cloneArrayBuffer(typedArray.buffer) : typedArray.buffer;
          return new typedArray.constructor(buffer, typedArray.byteOffset, typedArray.length);
        }

        /**
         * Compares values to sort them in ascending order.
         *
         * @private
         * @param {*} value The value to compare.
         * @param {*} other The other value to compare.
         * @returns {number} Returns the sort order indicator for `value`.
         */
        function compareAscending(value, other) {
          if (value !== other) {
            var valIsDefined = value !== undefined$1,
                valIsNull = value === null,
                valIsReflexive = value === value,
                valIsSymbol = isSymbol(value);

            var othIsDefined = other !== undefined$1,
                othIsNull = other === null,
                othIsReflexive = other === other,
                othIsSymbol = isSymbol(other);

            if ((!othIsNull && !othIsSymbol && !valIsSymbol && value > other) ||
                (valIsSymbol && othIsDefined && othIsReflexive && !othIsNull && !othIsSymbol) ||
                (valIsNull && othIsDefined && othIsReflexive) ||
                (!valIsDefined && othIsReflexive) ||
                !valIsReflexive) {
              return 1;
            }
            if ((!valIsNull && !valIsSymbol && !othIsSymbol && value < other) ||
                (othIsSymbol && valIsDefined && valIsReflexive && !valIsNull && !valIsSymbol) ||
                (othIsNull && valIsDefined && valIsReflexive) ||
                (!othIsDefined && valIsReflexive) ||
                !othIsReflexive) {
              return -1;
            }
          }
          return 0;
        }

        /**
         * Used by `_.orderBy` to compare multiple properties of a value to another
         * and stable sort them.
         *
         * If `orders` is unspecified, all values are sorted in ascending order. Otherwise,
         * specify an order of "desc" for descending or "asc" for ascending sort order
         * of corresponding values.
         *
         * @private
         * @param {Object} object The object to compare.
         * @param {Object} other The other object to compare.
         * @param {boolean[]|string[]} orders The order to sort by for each property.
         * @returns {number} Returns the sort order indicator for `object`.
         */
        function compareMultiple(object, other, orders) {
          var index = -1,
              objCriteria = object.criteria,
              othCriteria = other.criteria,
              length = objCriteria.length,
              ordersLength = orders.length;

          while (++index < length) {
            var result = compareAscending(objCriteria[index], othCriteria[index]);
            if (result) {
              if (index >= ordersLength) {
                return result;
              }
              var order = orders[index];
              return result * (order == 'desc' ? -1 : 1);
            }
          }
          // Fixes an `Array#sort` bug in the JS engine embedded in Adobe applications
          // that causes it, under certain circumstances, to provide the same value for
          // `object` and `other`. See https://github.com/jashkenas/underscore/pull/1247
          // for more details.
          //
          // This also ensures a stable sort in V8 and other engines.
          // See https://bugs.chromium.org/p/v8/issues/detail?id=90 for more details.
          return object.index - other.index;
        }

        /**
         * Creates an array that is the composition of partially applied arguments,
         * placeholders, and provided arguments into a single array of arguments.
         *
         * @private
         * @param {Array} args The provided arguments.
         * @param {Array} partials The arguments to prepend to those provided.
         * @param {Array} holders The `partials` placeholder indexes.
         * @params {boolean} [isCurried] Specify composing for a curried function.
         * @returns {Array} Returns the new array of composed arguments.
         */
        function composeArgs(args, partials, holders, isCurried) {
          var argsIndex = -1,
              argsLength = args.length,
              holdersLength = holders.length,
              leftIndex = -1,
              leftLength = partials.length,
              rangeLength = nativeMax(argsLength - holdersLength, 0),
              result = Array(leftLength + rangeLength),
              isUncurried = !isCurried;

          while (++leftIndex < leftLength) {
            result[leftIndex] = partials[leftIndex];
          }
          while (++argsIndex < holdersLength) {
            if (isUncurried || argsIndex < argsLength) {
              result[holders[argsIndex]] = args[argsIndex];
            }
          }
          while (rangeLength--) {
            result[leftIndex++] = args[argsIndex++];
          }
          return result;
        }

        /**
         * This function is like `composeArgs` except that the arguments composition
         * is tailored for `_.partialRight`.
         *
         * @private
         * @param {Array} args The provided arguments.
         * @param {Array} partials The arguments to append to those provided.
         * @param {Array} holders The `partials` placeholder indexes.
         * @params {boolean} [isCurried] Specify composing for a curried function.
         * @returns {Array} Returns the new array of composed arguments.
         */
        function composeArgsRight(args, partials, holders, isCurried) {
          var argsIndex = -1,
              argsLength = args.length,
              holdersIndex = -1,
              holdersLength = holders.length,
              rightIndex = -1,
              rightLength = partials.length,
              rangeLength = nativeMax(argsLength - holdersLength, 0),
              result = Array(rangeLength + rightLength),
              isUncurried = !isCurried;

          while (++argsIndex < rangeLength) {
            result[argsIndex] = args[argsIndex];
          }
          var offset = argsIndex;
          while (++rightIndex < rightLength) {
            result[offset + rightIndex] = partials[rightIndex];
          }
          while (++holdersIndex < holdersLength) {
            if (isUncurried || argsIndex < argsLength) {
              result[offset + holders[holdersIndex]] = args[argsIndex++];
            }
          }
          return result;
        }

        /**
         * Copies the values of `source` to `array`.
         *
         * @private
         * @param {Array} source The array to copy values from.
         * @param {Array} [array=[]] The array to copy values to.
         * @returns {Array} Returns `array`.
         */
        function copyArray(source, array) {
          var index = -1,
              length = source.length;

          array || (array = Array(length));
          while (++index < length) {
            array[index] = source[index];
          }
          return array;
        }

        /**
         * Copies properties of `source` to `object`.
         *
         * @private
         * @param {Object} source The object to copy properties from.
         * @param {Array} props The property identifiers to copy.
         * @param {Object} [object={}] The object to copy properties to.
         * @param {Function} [customizer] The function to customize copied values.
         * @returns {Object} Returns `object`.
         */
        function copyObject(source, props, object, customizer) {
          var isNew = !object;
          object || (object = {});

          var index = -1,
              length = props.length;

          while (++index < length) {
            var key = props[index];

            var newValue = customizer
              ? customizer(object[key], source[key], key, object, source)
              : undefined$1;

            if (newValue === undefined$1) {
              newValue = source[key];
            }
            if (isNew) {
              baseAssignValue(object, key, newValue);
            } else {
              assignValue(object, key, newValue);
            }
          }
          return object;
        }

        /**
         * Copies own symbols of `source` to `object`.
         *
         * @private
         * @param {Object} source The object to copy symbols from.
         * @param {Object} [object={}] The object to copy symbols to.
         * @returns {Object} Returns `object`.
         */
        function copySymbols(source, object) {
          return copyObject(source, getSymbols(source), object);
        }

        /**
         * Copies own and inherited symbols of `source` to `object`.
         *
         * @private
         * @param {Object} source The object to copy symbols from.
         * @param {Object} [object={}] The object to copy symbols to.
         * @returns {Object} Returns `object`.
         */
        function copySymbolsIn(source, object) {
          return copyObject(source, getSymbolsIn(source), object);
        }

        /**
         * Creates a function like `_.groupBy`.
         *
         * @private
         * @param {Function} setter The function to set accumulator values.
         * @param {Function} [initializer] The accumulator object initializer.
         * @returns {Function} Returns the new aggregator function.
         */
        function createAggregator(setter, initializer) {
          return function(collection, iteratee) {
            var func = isArray(collection) ? arrayAggregator : baseAggregator,
                accumulator = initializer ? initializer() : {};

            return func(collection, setter, getIteratee(iteratee, 2), accumulator);
          };
        }

        /**
         * Creates a function like `_.assign`.
         *
         * @private
         * @param {Function} assigner The function to assign values.
         * @returns {Function} Returns the new assigner function.
         */
        function createAssigner(assigner) {
          return baseRest(function(object, sources) {
            var index = -1,
                length = sources.length,
                customizer = length > 1 ? sources[length - 1] : undefined$1,
                guard = length > 2 ? sources[2] : undefined$1;

            customizer = (assigner.length > 3 && typeof customizer == 'function')
              ? (length--, customizer)
              : undefined$1;

            if (guard && isIterateeCall(sources[0], sources[1], guard)) {
              customizer = length < 3 ? undefined$1 : customizer;
              length = 1;
            }
            object = Object(object);
            while (++index < length) {
              var source = sources[index];
              if (source) {
                assigner(object, source, index, customizer);
              }
            }
            return object;
          });
        }

        /**
         * Creates a `baseEach` or `baseEachRight` function.
         *
         * @private
         * @param {Function} eachFunc The function to iterate over a collection.
         * @param {boolean} [fromRight] Specify iterating from right to left.
         * @returns {Function} Returns the new base function.
         */
        function createBaseEach(eachFunc, fromRight) {
          return function(collection, iteratee) {
            if (collection == null) {
              return collection;
            }
            if (!isArrayLike(collection)) {
              return eachFunc(collection, iteratee);
            }
            var length = collection.length,
                index = fromRight ? length : -1,
                iterable = Object(collection);

            while ((fromRight ? index-- : ++index < length)) {
              if (iteratee(iterable[index], index, iterable) === false) {
                break;
              }
            }
            return collection;
          };
        }

        /**
         * Creates a base function for methods like `_.forIn` and `_.forOwn`.
         *
         * @private
         * @param {boolean} [fromRight] Specify iterating from right to left.
         * @returns {Function} Returns the new base function.
         */
        function createBaseFor(fromRight) {
          return function(object, iteratee, keysFunc) {
            var index = -1,
                iterable = Object(object),
                props = keysFunc(object),
                length = props.length;

            while (length--) {
              var key = props[fromRight ? length : ++index];
              if (iteratee(iterable[key], key, iterable) === false) {
                break;
              }
            }
            return object;
          };
        }

        /**
         * Creates a function that wraps `func` to invoke it with the optional `this`
         * binding of `thisArg`.
         *
         * @private
         * @param {Function} func The function to wrap.
         * @param {number} bitmask The bitmask flags. See `createWrap` for more details.
         * @param {*} [thisArg] The `this` binding of `func`.
         * @returns {Function} Returns the new wrapped function.
         */
        function createBind(func, bitmask, thisArg) {
          var isBind = bitmask & WRAP_BIND_FLAG,
              Ctor = createCtor(func);

          function wrapper() {
            var fn = (this && this !== root && this instanceof wrapper) ? Ctor : func;
            return fn.apply(isBind ? thisArg : this, arguments);
          }
          return wrapper;
        }

        /**
         * Creates a function like `_.lowerFirst`.
         *
         * @private
         * @param {string} methodName The name of the `String` case method to use.
         * @returns {Function} Returns the new case function.
         */
        function createCaseFirst(methodName) {
          return function(string) {
            string = toString(string);

            var strSymbols = hasUnicode(string)
              ? stringToArray(string)
              : undefined$1;

            var chr = strSymbols
              ? strSymbols[0]
              : string.charAt(0);

            var trailing = strSymbols
              ? castSlice(strSymbols, 1).join('')
              : string.slice(1);

            return chr[methodName]() + trailing;
          };
        }

        /**
         * Creates a function like `_.camelCase`.
         *
         * @private
         * @param {Function} callback The function to combine each word.
         * @returns {Function} Returns the new compounder function.
         */
        function createCompounder(callback) {
          return function(string) {
            return arrayReduce(words(deburr(string).replace(reApos, '')), callback, '');
          };
        }

        /**
         * Creates a function that produces an instance of `Ctor` regardless of
         * whether it was invoked as part of a `new` expression or by `call` or `apply`.
         *
         * @private
         * @param {Function} Ctor The constructor to wrap.
         * @returns {Function} Returns the new wrapped function.
         */
        function createCtor(Ctor) {
          return function() {
            // Use a `switch` statement to work with class constructors. See
            // http://ecma-international.org/ecma-262/7.0/#sec-ecmascript-function-objects-call-thisargument-argumentslist
            // for more details.
            var args = arguments;
            switch (args.length) {
              case 0: return new Ctor;
              case 1: return new Ctor(args[0]);
              case 2: return new Ctor(args[0], args[1]);
              case 3: return new Ctor(args[0], args[1], args[2]);
              case 4: return new Ctor(args[0], args[1], args[2], args[3]);
              case 5: return new Ctor(args[0], args[1], args[2], args[3], args[4]);
              case 6: return new Ctor(args[0], args[1], args[2], args[3], args[4], args[5]);
              case 7: return new Ctor(args[0], args[1], args[2], args[3], args[4], args[5], args[6]);
            }
            var thisBinding = baseCreate(Ctor.prototype),
                result = Ctor.apply(thisBinding, args);

            // Mimic the constructor's `return` behavior.
            // See https://es5.github.io/#x13.2.2 for more details.
            return isObject(result) ? result : thisBinding;
          };
        }

        /**
         * Creates a function that wraps `func` to enable currying.
         *
         * @private
         * @param {Function} func The function to wrap.
         * @param {number} bitmask The bitmask flags. See `createWrap` for more details.
         * @param {number} arity The arity of `func`.
         * @returns {Function} Returns the new wrapped function.
         */
        function createCurry(func, bitmask, arity) {
          var Ctor = createCtor(func);

          function wrapper() {
            var length = arguments.length,
                args = Array(length),
                index = length,
                placeholder = getHolder(wrapper);

            while (index--) {
              args[index] = arguments[index];
            }
            var holders = (length < 3 && args[0] !== placeholder && args[length - 1] !== placeholder)
              ? []
              : replaceHolders(args, placeholder);

            length -= holders.length;
            if (length < arity) {
              return createRecurry(
                func, bitmask, createHybrid, wrapper.placeholder, undefined$1,
                args, holders, undefined$1, undefined$1, arity - length);
            }
            var fn = (this && this !== root && this instanceof wrapper) ? Ctor : func;
            return apply(fn, this, args);
          }
          return wrapper;
        }

        /**
         * Creates a `_.find` or `_.findLast` function.
         *
         * @private
         * @param {Function} findIndexFunc The function to find the collection index.
         * @returns {Function} Returns the new find function.
         */
        function createFind(findIndexFunc) {
          return function(collection, predicate, fromIndex) {
            var iterable = Object(collection);
            if (!isArrayLike(collection)) {
              var iteratee = getIteratee(predicate, 3);
              collection = keys(collection);
              predicate = function(key) { return iteratee(iterable[key], key, iterable); };
            }
            var index = findIndexFunc(collection, predicate, fromIndex);
            return index > -1 ? iterable[iteratee ? collection[index] : index] : undefined$1;
          };
        }

        /**
         * Creates a `_.flow` or `_.flowRight` function.
         *
         * @private
         * @param {boolean} [fromRight] Specify iterating from right to left.
         * @returns {Function} Returns the new flow function.
         */
        function createFlow(fromRight) {
          return flatRest(function(funcs) {
            var length = funcs.length,
                index = length,
                prereq = LodashWrapper.prototype.thru;

            if (fromRight) {
              funcs.reverse();
            }
            while (index--) {
              var func = funcs[index];
              if (typeof func != 'function') {
                throw new TypeError(FUNC_ERROR_TEXT);
              }
              if (prereq && !wrapper && getFuncName(func) == 'wrapper') {
                var wrapper = new LodashWrapper([], true);
              }
            }
            index = wrapper ? index : length;
            while (++index < length) {
              func = funcs[index];

              var funcName = getFuncName(func),
                  data = funcName == 'wrapper' ? getData(func) : undefined$1;

              if (data && isLaziable(data[0]) &&
                    data[1] == (WRAP_ARY_FLAG | WRAP_CURRY_FLAG | WRAP_PARTIAL_FLAG | WRAP_REARG_FLAG) &&
                    !data[4].length && data[9] == 1
                  ) {
                wrapper = wrapper[getFuncName(data[0])].apply(wrapper, data[3]);
              } else {
                wrapper = (func.length == 1 && isLaziable(func))
                  ? wrapper[funcName]()
                  : wrapper.thru(func);
              }
            }
            return function() {
              var args = arguments,
                  value = args[0];

              if (wrapper && args.length == 1 && isArray(value)) {
                return wrapper.plant(value).value();
              }
              var index = 0,
                  result = length ? funcs[index].apply(this, args) : value;

              while (++index < length) {
                result = funcs[index].call(this, result);
              }
              return result;
            };
          });
        }

        /**
         * Creates a function that wraps `func` to invoke it with optional `this`
         * binding of `thisArg`, partial application, and currying.
         *
         * @private
         * @param {Function|string} func The function or method name to wrap.
         * @param {number} bitmask The bitmask flags. See `createWrap` for more details.
         * @param {*} [thisArg] The `this` binding of `func`.
         * @param {Array} [partials] The arguments to prepend to those provided to
         *  the new function.
         * @param {Array} [holders] The `partials` placeholder indexes.
         * @param {Array} [partialsRight] The arguments to append to those provided
         *  to the new function.
         * @param {Array} [holdersRight] The `partialsRight` placeholder indexes.
         * @param {Array} [argPos] The argument positions of the new function.
         * @param {number} [ary] The arity cap of `func`.
         * @param {number} [arity] The arity of `func`.
         * @returns {Function} Returns the new wrapped function.
         */
        function createHybrid(func, bitmask, thisArg, partials, holders, partialsRight, holdersRight, argPos, ary, arity) {
          var isAry = bitmask & WRAP_ARY_FLAG,
              isBind = bitmask & WRAP_BIND_FLAG,
              isBindKey = bitmask & WRAP_BIND_KEY_FLAG,
              isCurried = bitmask & (WRAP_CURRY_FLAG | WRAP_CURRY_RIGHT_FLAG),
              isFlip = bitmask & WRAP_FLIP_FLAG,
              Ctor = isBindKey ? undefined$1 : createCtor(func);

          function wrapper() {
            var length = arguments.length,
                args = Array(length),
                index = length;

            while (index--) {
              args[index] = arguments[index];
            }
            if (isCurried) {
              var placeholder = getHolder(wrapper),
                  holdersCount = countHolders(args, placeholder);
            }
            if (partials) {
              args = composeArgs(args, partials, holders, isCurried);
            }
            if (partialsRight) {
              args = composeArgsRight(args, partialsRight, holdersRight, isCurried);
            }
            length -= holdersCount;
            if (isCurried && length < arity) {
              var newHolders = replaceHolders(args, placeholder);
              return createRecurry(
                func, bitmask, createHybrid, wrapper.placeholder, thisArg,
                args, newHolders, argPos, ary, arity - length
              );
            }
            var thisBinding = isBind ? thisArg : this,
                fn = isBindKey ? thisBinding[func] : func;

            length = args.length;
            if (argPos) {
              args = reorder(args, argPos);
            } else if (isFlip && length > 1) {
              args.reverse();
            }
            if (isAry && ary < length) {
              args.length = ary;
            }
            if (this && this !== root && this instanceof wrapper) {
              fn = Ctor || createCtor(fn);
            }
            return fn.apply(thisBinding, args);
          }
          return wrapper;
        }

        /**
         * Creates a function like `_.invertBy`.
         *
         * @private
         * @param {Function} setter The function to set accumulator values.
         * @param {Function} toIteratee The function to resolve iteratees.
         * @returns {Function} Returns the new inverter function.
         */
        function createInverter(setter, toIteratee) {
          return function(object, iteratee) {
            return baseInverter(object, setter, toIteratee(iteratee), {});
          };
        }

        /**
         * Creates a function that performs a mathematical operation on two values.
         *
         * @private
         * @param {Function} operator The function to perform the operation.
         * @param {number} [defaultValue] The value used for `undefined` arguments.
         * @returns {Function} Returns the new mathematical operation function.
         */
        function createMathOperation(operator, defaultValue) {
          return function(value, other) {
            var result;
            if (value === undefined$1 && other === undefined$1) {
              return defaultValue;
            }
            if (value !== undefined$1) {
              result = value;
            }
            if (other !== undefined$1) {
              if (result === undefined$1) {
                return other;
              }
              if (typeof value == 'string' || typeof other == 'string') {
                value = baseToString(value);
                other = baseToString(other);
              } else {
                value = baseToNumber(value);
                other = baseToNumber(other);
              }
              result = operator(value, other);
            }
            return result;
          };
        }

        /**
         * Creates a function like `_.over`.
         *
         * @private
         * @param {Function} arrayFunc The function to iterate over iteratees.
         * @returns {Function} Returns the new over function.
         */
        function createOver(arrayFunc) {
          return flatRest(function(iteratees) {
            iteratees = arrayMap(iteratees, baseUnary(getIteratee()));
            return baseRest(function(args) {
              var thisArg = this;
              return arrayFunc(iteratees, function(iteratee) {
                return apply(iteratee, thisArg, args);
              });
            });
          });
        }

        /**
         * Creates the padding for `string` based on `length`. The `chars` string
         * is truncated if the number of characters exceeds `length`.
         *
         * @private
         * @param {number} length The padding length.
         * @param {string} [chars=' '] The string used as padding.
         * @returns {string} Returns the padding for `string`.
         */
        function createPadding(length, chars) {
          chars = chars === undefined$1 ? ' ' : baseToString(chars);

          var charsLength = chars.length;
          if (charsLength < 2) {
            return charsLength ? baseRepeat(chars, length) : chars;
          }
          var result = baseRepeat(chars, nativeCeil(length / stringSize(chars)));
          return hasUnicode(chars)
            ? castSlice(stringToArray(result), 0, length).join('')
            : result.slice(0, length);
        }

        /**
         * Creates a function that wraps `func` to invoke it with the `this` binding
         * of `thisArg` and `partials` prepended to the arguments it receives.
         *
         * @private
         * @param {Function} func The function to wrap.
         * @param {number} bitmask The bitmask flags. See `createWrap` for more details.
         * @param {*} thisArg The `this` binding of `func`.
         * @param {Array} partials The arguments to prepend to those provided to
         *  the new function.
         * @returns {Function} Returns the new wrapped function.
         */
        function createPartial(func, bitmask, thisArg, partials) {
          var isBind = bitmask & WRAP_BIND_FLAG,
              Ctor = createCtor(func);

          function wrapper() {
            var argsIndex = -1,
                argsLength = arguments.length,
                leftIndex = -1,
                leftLength = partials.length,
                args = Array(leftLength + argsLength),
                fn = (this && this !== root && this instanceof wrapper) ? Ctor : func;

            while (++leftIndex < leftLength) {
              args[leftIndex] = partials[leftIndex];
            }
            while (argsLength--) {
              args[leftIndex++] = arguments[++argsIndex];
            }
            return apply(fn, isBind ? thisArg : this, args);
          }
          return wrapper;
        }

        /**
         * Creates a `_.range` or `_.rangeRight` function.
         *
         * @private
         * @param {boolean} [fromRight] Specify iterating from right to left.
         * @returns {Function} Returns the new range function.
         */
        function createRange(fromRight) {
          return function(start, end, step) {
            if (step && typeof step != 'number' && isIterateeCall(start, end, step)) {
              end = step = undefined$1;
            }
            // Ensure the sign of `-0` is preserved.
            start = toFinite(start);
            if (end === undefined$1) {
              end = start;
              start = 0;
            } else {
              end = toFinite(end);
            }
            step = step === undefined$1 ? (start < end ? 1 : -1) : toFinite(step);
            return baseRange(start, end, step, fromRight);
          };
        }

        /**
         * Creates a function that performs a relational operation on two values.
         *
         * @private
         * @param {Function} operator The function to perform the operation.
         * @returns {Function} Returns the new relational operation function.
         */
        function createRelationalOperation(operator) {
          return function(value, other) {
            if (!(typeof value == 'string' && typeof other == 'string')) {
              value = toNumber(value);
              other = toNumber(other);
            }
            return operator(value, other);
          };
        }

        /**
         * Creates a function that wraps `func` to continue currying.
         *
         * @private
         * @param {Function} func The function to wrap.
         * @param {number} bitmask The bitmask flags. See `createWrap` for more details.
         * @param {Function} wrapFunc The function to create the `func` wrapper.
         * @param {*} placeholder The placeholder value.
         * @param {*} [thisArg] The `this` binding of `func`.
         * @param {Array} [partials] The arguments to prepend to those provided to
         *  the new function.
         * @param {Array} [holders] The `partials` placeholder indexes.
         * @param {Array} [argPos] The argument positions of the new function.
         * @param {number} [ary] The arity cap of `func`.
         * @param {number} [arity] The arity of `func`.
         * @returns {Function} Returns the new wrapped function.
         */
        function createRecurry(func, bitmask, wrapFunc, placeholder, thisArg, partials, holders, argPos, ary, arity) {
          var isCurry = bitmask & WRAP_CURRY_FLAG,
              newHolders = isCurry ? holders : undefined$1,
              newHoldersRight = isCurry ? undefined$1 : holders,
              newPartials = isCurry ? partials : undefined$1,
              newPartialsRight = isCurry ? undefined$1 : partials;

          bitmask |= (isCurry ? WRAP_PARTIAL_FLAG : WRAP_PARTIAL_RIGHT_FLAG);
          bitmask &= ~(isCurry ? WRAP_PARTIAL_RIGHT_FLAG : WRAP_PARTIAL_FLAG);

          if (!(bitmask & WRAP_CURRY_BOUND_FLAG)) {
            bitmask &= ~(WRAP_BIND_FLAG | WRAP_BIND_KEY_FLAG);
          }
          var newData = [
            func, bitmask, thisArg, newPartials, newHolders, newPartialsRight,
            newHoldersRight, argPos, ary, arity
          ];

          var result = wrapFunc.apply(undefined$1, newData);
          if (isLaziable(func)) {
            setData(result, newData);
          }
          result.placeholder = placeholder;
          return setWrapToString(result, func, bitmask);
        }

        /**
         * Creates a function like `_.round`.
         *
         * @private
         * @param {string} methodName The name of the `Math` method to use when rounding.
         * @returns {Function} Returns the new round function.
         */
        function createRound(methodName) {
          var func = Math[methodName];
          return function(number, precision) {
            number = toNumber(number);
            precision = precision == null ? 0 : nativeMin(toInteger(precision), 292);
            if (precision && nativeIsFinite(number)) {
              // Shift with exponential notation to avoid floating-point issues.
              // See [MDN](https://mdn.io/round#Examples) for more details.
              var pair = (toString(number) + 'e').split('e'),
                  value = func(pair[0] + 'e' + (+pair[1] + precision));

              pair = (toString(value) + 'e').split('e');
              return +(pair[0] + 'e' + (+pair[1] - precision));
            }
            return func(number);
          };
        }

        /**
         * Creates a set object of `values`.
         *
         * @private
         * @param {Array} values The values to add to the set.
         * @returns {Object} Returns the new set.
         */
        var createSet = !(Set && (1 / setToArray(new Set([,-0]))[1]) == INFINITY) ? noop : function(values) {
          return new Set(values);
        };

        /**
         * Creates a `_.toPairs` or `_.toPairsIn` function.
         *
         * @private
         * @param {Function} keysFunc The function to get the keys of a given object.
         * @returns {Function} Returns the new pairs function.
         */
        function createToPairs(keysFunc) {
          return function(object) {
            var tag = getTag(object);
            if (tag == mapTag) {
              return mapToArray(object);
            }
            if (tag == setTag) {
              return setToPairs(object);
            }
            return baseToPairs(object, keysFunc(object));
          };
        }

        /**
         * Creates a function that either curries or invokes `func` with optional
         * `this` binding and partially applied arguments.
         *
         * @private
         * @param {Function|string} func The function or method name to wrap.
         * @param {number} bitmask The bitmask flags.
         *    1 - `_.bind`
         *    2 - `_.bindKey`
         *    4 - `_.curry` or `_.curryRight` of a bound function
         *    8 - `_.curry`
         *   16 - `_.curryRight`
         *   32 - `_.partial`
         *   64 - `_.partialRight`
         *  128 - `_.rearg`
         *  256 - `_.ary`
         *  512 - `_.flip`
         * @param {*} [thisArg] The `this` binding of `func`.
         * @param {Array} [partials] The arguments to be partially applied.
         * @param {Array} [holders] The `partials` placeholder indexes.
         * @param {Array} [argPos] The argument positions of the new function.
         * @param {number} [ary] The arity cap of `func`.
         * @param {number} [arity] The arity of `func`.
         * @returns {Function} Returns the new wrapped function.
         */
        function createWrap(func, bitmask, thisArg, partials, holders, argPos, ary, arity) {
          var isBindKey = bitmask & WRAP_BIND_KEY_FLAG;
          if (!isBindKey && typeof func != 'function') {
            throw new TypeError(FUNC_ERROR_TEXT);
          }
          var length = partials ? partials.length : 0;
          if (!length) {
            bitmask &= ~(WRAP_PARTIAL_FLAG | WRAP_PARTIAL_RIGHT_FLAG);
            partials = holders = undefined$1;
          }
          ary = ary === undefined$1 ? ary : nativeMax(toInteger(ary), 0);
          arity = arity === undefined$1 ? arity : toInteger(arity);
          length -= holders ? holders.length : 0;

          if (bitmask & WRAP_PARTIAL_RIGHT_FLAG) {
            var partialsRight = partials,
                holdersRight = holders;

            partials = holders = undefined$1;
          }
          var data = isBindKey ? undefined$1 : getData(func);

          var newData = [
            func, bitmask, thisArg, partials, holders, partialsRight, holdersRight,
            argPos, ary, arity
          ];

          if (data) {
            mergeData(newData, data);
          }
          func = newData[0];
          bitmask = newData[1];
          thisArg = newData[2];
          partials = newData[3];
          holders = newData[4];
          arity = newData[9] = newData[9] === undefined$1
            ? (isBindKey ? 0 : func.length)
            : nativeMax(newData[9] - length, 0);

          if (!arity && bitmask & (WRAP_CURRY_FLAG | WRAP_CURRY_RIGHT_FLAG)) {
            bitmask &= ~(WRAP_CURRY_FLAG | WRAP_CURRY_RIGHT_FLAG);
          }
          if (!bitmask || bitmask == WRAP_BIND_FLAG) {
            var result = createBind(func, bitmask, thisArg);
          } else if (bitmask == WRAP_CURRY_FLAG || bitmask == WRAP_CURRY_RIGHT_FLAG) {
            result = createCurry(func, bitmask, arity);
          } else if ((bitmask == WRAP_PARTIAL_FLAG || bitmask == (WRAP_BIND_FLAG | WRAP_PARTIAL_FLAG)) && !holders.length) {
            result = createPartial(func, bitmask, thisArg, partials);
          } else {
            result = createHybrid.apply(undefined$1, newData);
          }
          var setter = data ? baseSetData : setData;
          return setWrapToString(setter(result, newData), func, bitmask);
        }

        /**
         * Used by `_.defaults` to customize its `_.assignIn` use to assign properties
         * of source objects to the destination object for all destination properties
         * that resolve to `undefined`.
         *
         * @private
         * @param {*} objValue The destination value.
         * @param {*} srcValue The source value.
         * @param {string} key The key of the property to assign.
         * @param {Object} object The parent object of `objValue`.
         * @returns {*} Returns the value to assign.
         */
        function customDefaultsAssignIn(objValue, srcValue, key, object) {
          if (objValue === undefined$1 ||
              (eq(objValue, objectProto[key]) && !hasOwnProperty.call(object, key))) {
            return srcValue;
          }
          return objValue;
        }

        /**
         * Used by `_.defaultsDeep` to customize its `_.merge` use to merge source
         * objects into destination objects that are passed thru.
         *
         * @private
         * @param {*} objValue The destination value.
         * @param {*} srcValue The source value.
         * @param {string} key The key of the property to merge.
         * @param {Object} object The parent object of `objValue`.
         * @param {Object} source The parent object of `srcValue`.
         * @param {Object} [stack] Tracks traversed source values and their merged
         *  counterparts.
         * @returns {*} Returns the value to assign.
         */
        function customDefaultsMerge(objValue, srcValue, key, object, source, stack) {
          if (isObject(objValue) && isObject(srcValue)) {
            // Recursively merge objects and arrays (susceptible to call stack limits).
            stack.set(srcValue, objValue);
            baseMerge(objValue, srcValue, undefined$1, customDefaultsMerge, stack);
            stack['delete'](srcValue);
          }
          return objValue;
        }

        /**
         * Used by `_.omit` to customize its `_.cloneDeep` use to only clone plain
         * objects.
         *
         * @private
         * @param {*} value The value to inspect.
         * @param {string} key The key of the property to inspect.
         * @returns {*} Returns the uncloned value or `undefined` to defer cloning to `_.cloneDeep`.
         */
        function customOmitClone(value) {
          return isPlainObject(value) ? undefined$1 : value;
        }

        /**
         * A specialized version of `baseIsEqualDeep` for arrays with support for
         * partial deep comparisons.
         *
         * @private
         * @param {Array} array The array to compare.
         * @param {Array} other The other array to compare.
         * @param {number} bitmask The bitmask flags. See `baseIsEqual` for more details.
         * @param {Function} customizer The function to customize comparisons.
         * @param {Function} equalFunc The function to determine equivalents of values.
         * @param {Object} stack Tracks traversed `array` and `other` objects.
         * @returns {boolean} Returns `true` if the arrays are equivalent, else `false`.
         */
        function equalArrays(array, other, bitmask, customizer, equalFunc, stack) {
          var isPartial = bitmask & COMPARE_PARTIAL_FLAG,
              arrLength = array.length,
              othLength = other.length;

          if (arrLength != othLength && !(isPartial && othLength > arrLength)) {
            return false;
          }
          // Check that cyclic values are equal.
          var arrStacked = stack.get(array);
          var othStacked = stack.get(other);
          if (arrStacked && othStacked) {
            return arrStacked == other && othStacked == array;
          }
          var index = -1,
              result = true,
              seen = (bitmask & COMPARE_UNORDERED_FLAG) ? new SetCache : undefined$1;

          stack.set(array, other);
          stack.set(other, array);

          // Ignore non-index properties.
          while (++index < arrLength) {
            var arrValue = array[index],
                othValue = other[index];

            if (customizer) {
              var compared = isPartial
                ? customizer(othValue, arrValue, index, other, array, stack)
                : customizer(arrValue, othValue, index, array, other, stack);
            }
            if (compared !== undefined$1) {
              if (compared) {
                continue;
              }
              result = false;
              break;
            }
            // Recursively compare arrays (susceptible to call stack limits).
            if (seen) {
              if (!arraySome(other, function(othValue, othIndex) {
                    if (!cacheHas(seen, othIndex) &&
                        (arrValue === othValue || equalFunc(arrValue, othValue, bitmask, customizer, stack))) {
                      return seen.push(othIndex);
                    }
                  })) {
                result = false;
                break;
              }
            } else if (!(
                  arrValue === othValue ||
                    equalFunc(arrValue, othValue, bitmask, customizer, stack)
                )) {
              result = false;
              break;
            }
          }
          stack['delete'](array);
          stack['delete'](other);
          return result;
        }

        /**
         * A specialized version of `baseIsEqualDeep` for comparing objects of
         * the same `toStringTag`.
         *
         * **Note:** This function only supports comparing values with tags of
         * `Boolean`, `Date`, `Error`, `Number`, `RegExp`, or `String`.
         *
         * @private
         * @param {Object} object The object to compare.
         * @param {Object} other The other object to compare.
         * @param {string} tag The `toStringTag` of the objects to compare.
         * @param {number} bitmask The bitmask flags. See `baseIsEqual` for more details.
         * @param {Function} customizer The function to customize comparisons.
         * @param {Function} equalFunc The function to determine equivalents of values.
         * @param {Object} stack Tracks traversed `object` and `other` objects.
         * @returns {boolean} Returns `true` if the objects are equivalent, else `false`.
         */
        function equalByTag(object, other, tag, bitmask, customizer, equalFunc, stack) {
          switch (tag) {
            case dataViewTag:
              if ((object.byteLength != other.byteLength) ||
                  (object.byteOffset != other.byteOffset)) {
                return false;
              }
              object = object.buffer;
              other = other.buffer;

            case arrayBufferTag:
              if ((object.byteLength != other.byteLength) ||
                  !equalFunc(new Uint8Array(object), new Uint8Array(other))) {
                return false;
              }
              return true;

            case boolTag:
            case dateTag:
            case numberTag:
              // Coerce booleans to `1` or `0` and dates to milliseconds.
              // Invalid dates are coerced to `NaN`.
              return eq(+object, +other);

            case errorTag:
              return object.name == other.name && object.message == other.message;

            case regexpTag:
            case stringTag:
              // Coerce regexes to strings and treat strings, primitives and objects,
              // as equal. See http://www.ecma-international.org/ecma-262/7.0/#sec-regexp.prototype.tostring
              // for more details.
              return object == (other + '');

            case mapTag:
              var convert = mapToArray;

            case setTag:
              var isPartial = bitmask & COMPARE_PARTIAL_FLAG;
              convert || (convert = setToArray);

              if (object.size != other.size && !isPartial) {
                return false;
              }
              // Assume cyclic values are equal.
              var stacked = stack.get(object);
              if (stacked) {
                return stacked == other;
              }
              bitmask |= COMPARE_UNORDERED_FLAG;

              // Recursively compare objects (susceptible to call stack limits).
              stack.set(object, other);
              var result = equalArrays(convert(object), convert(other), bitmask, customizer, equalFunc, stack);
              stack['delete'](object);
              return result;

            case symbolTag:
              if (symbolValueOf) {
                return symbolValueOf.call(object) == symbolValueOf.call(other);
              }
          }
          return false;
        }

        /**
         * A specialized version of `baseIsEqualDeep` for objects with support for
         * partial deep comparisons.
         *
         * @private
         * @param {Object} object The object to compare.
         * @param {Object} other The other object to compare.
         * @param {number} bitmask The bitmask flags. See `baseIsEqual` for more details.
         * @param {Function} customizer The function to customize comparisons.
         * @param {Function} equalFunc The function to determine equivalents of values.
         * @param {Object} stack Tracks traversed `object` and `other` objects.
         * @returns {boolean} Returns `true` if the objects are equivalent, else `false`.
         */
        function equalObjects(object, other, bitmask, customizer, equalFunc, stack) {
          var isPartial = bitmask & COMPARE_PARTIAL_FLAG,
              objProps = getAllKeys(object),
              objLength = objProps.length,
              othProps = getAllKeys(other),
              othLength = othProps.length;

          if (objLength != othLength && !isPartial) {
            return false;
          }
          var index = objLength;
          while (index--) {
            var key = objProps[index];
            if (!(isPartial ? key in other : hasOwnProperty.call(other, key))) {
              return false;
            }
          }
          // Check that cyclic values are equal.
          var objStacked = stack.get(object);
          var othStacked = stack.get(other);
          if (objStacked && othStacked) {
            return objStacked == other && othStacked == object;
          }
          var result = true;
          stack.set(object, other);
          stack.set(other, object);

          var skipCtor = isPartial;
          while (++index < objLength) {
            key = objProps[index];
            var objValue = object[key],
                othValue = other[key];

            if (customizer) {
              var compared = isPartial
                ? customizer(othValue, objValue, key, other, object, stack)
                : customizer(objValue, othValue, key, object, other, stack);
            }
            // Recursively compare objects (susceptible to call stack limits).
            if (!(compared === undefined$1
                  ? (objValue === othValue || equalFunc(objValue, othValue, bitmask, customizer, stack))
                  : compared
                )) {
              result = false;
              break;
            }
            skipCtor || (skipCtor = key == 'constructor');
          }
          if (result && !skipCtor) {
            var objCtor = object.constructor,
                othCtor = other.constructor;

            // Non `Object` object instances with different constructors are not equal.
            if (objCtor != othCtor &&
                ('constructor' in object && 'constructor' in other) &&
                !(typeof objCtor == 'function' && objCtor instanceof objCtor &&
                  typeof othCtor == 'function' && othCtor instanceof othCtor)) {
              result = false;
            }
          }
          stack['delete'](object);
          stack['delete'](other);
          return result;
        }

        /**
         * A specialized version of `baseRest` which flattens the rest array.
         *
         * @private
         * @param {Function} func The function to apply a rest parameter to.
         * @returns {Function} Returns the new function.
         */
        function flatRest(func) {
          return setToString(overRest(func, undefined$1, flatten), func + '');
        }

        /**
         * Creates an array of own enumerable property names and symbols of `object`.
         *
         * @private
         * @param {Object} object The object to query.
         * @returns {Array} Returns the array of property names and symbols.
         */
        function getAllKeys(object) {
          return baseGetAllKeys(object, keys, getSymbols);
        }

        /**
         * Creates an array of own and inherited enumerable property names and
         * symbols of `object`.
         *
         * @private
         * @param {Object} object The object to query.
         * @returns {Array} Returns the array of property names and symbols.
         */
        function getAllKeysIn(object) {
          return baseGetAllKeys(object, keysIn, getSymbolsIn);
        }

        /**
         * Gets metadata for `func`.
         *
         * @private
         * @param {Function} func The function to query.
         * @returns {*} Returns the metadata for `func`.
         */
        var getData = !metaMap ? noop : function(func) {
          return metaMap.get(func);
        };

        /**
         * Gets the name of `func`.
         *
         * @private
         * @param {Function} func The function to query.
         * @returns {string} Returns the function name.
         */
        function getFuncName(func) {
          var result = (func.name + ''),
              array = realNames[result],
              length = hasOwnProperty.call(realNames, result) ? array.length : 0;

          while (length--) {
            var data = array[length],
                otherFunc = data.func;
            if (otherFunc == null || otherFunc == func) {
              return data.name;
            }
          }
          return result;
        }

        /**
         * Gets the argument placeholder value for `func`.
         *
         * @private
         * @param {Function} func The function to inspect.
         * @returns {*} Returns the placeholder value.
         */
        function getHolder(func) {
          var object = hasOwnProperty.call(lodash, 'placeholder') ? lodash : func;
          return object.placeholder;
        }

        /**
         * Gets the appropriate "iteratee" function. If `_.iteratee` is customized,
         * this function returns the custom method, otherwise it returns `baseIteratee`.
         * If arguments are provided, the chosen function is invoked with them and
         * its result is returned.
         *
         * @private
         * @param {*} [value] The value to convert to an iteratee.
         * @param {number} [arity] The arity of the created iteratee.
         * @returns {Function} Returns the chosen function or its result.
         */
        function getIteratee() {
          var result = lodash.iteratee || iteratee;
          result = result === iteratee ? baseIteratee : result;
          return arguments.length ? result(arguments[0], arguments[1]) : result;
        }

        /**
         * Gets the data for `map`.
         *
         * @private
         * @param {Object} map The map to query.
         * @param {string} key The reference key.
         * @returns {*} Returns the map data.
         */
        function getMapData(map, key) {
          var data = map.__data__;
          return isKeyable(key)
            ? data[typeof key == 'string' ? 'string' : 'hash']
            : data.map;
        }

        /**
         * Gets the property names, values, and compare flags of `object`.
         *
         * @private
         * @param {Object} object The object to query.
         * @returns {Array} Returns the match data of `object`.
         */
        function getMatchData(object) {
          var result = keys(object),
              length = result.length;

          while (length--) {
            var key = result[length],
                value = object[key];

            result[length] = [key, value, isStrictComparable(value)];
          }
          return result;
        }

        /**
         * Gets the native function at `key` of `object`.
         *
         * @private
         * @param {Object} object The object to query.
         * @param {string} key The key of the method to get.
         * @returns {*} Returns the function if it's native, else `undefined`.
         */
        function getNative(object, key) {
          var value = getValue(object, key);
          return baseIsNative(value) ? value : undefined$1;
        }

        /**
         * A specialized version of `baseGetTag` which ignores `Symbol.toStringTag` values.
         *
         * @private
         * @param {*} value The value to query.
         * @returns {string} Returns the raw `toStringTag`.
         */
        function getRawTag(value) {
          var isOwn = hasOwnProperty.call(value, symToStringTag),
              tag = value[symToStringTag];

          try {
            value[symToStringTag] = undefined$1;
            var unmasked = true;
          } catch (e) {}

          var result = nativeObjectToString.call(value);
          if (unmasked) {
            if (isOwn) {
              value[symToStringTag] = tag;
            } else {
              delete value[symToStringTag];
            }
          }
          return result;
        }

        /**
         * Creates an array of the own enumerable symbols of `object`.
         *
         * @private
         * @param {Object} object The object to query.
         * @returns {Array} Returns the array of symbols.
         */
        var getSymbols = !nativeGetSymbols ? stubArray : function(object) {
          if (object == null) {
            return [];
          }
          object = Object(object);
          return arrayFilter(nativeGetSymbols(object), function(symbol) {
            return propertyIsEnumerable.call(object, symbol);
          });
        };

        /**
         * Creates an array of the own and inherited enumerable symbols of `object`.
         *
         * @private
         * @param {Object} object The object to query.
         * @returns {Array} Returns the array of symbols.
         */
        var getSymbolsIn = !nativeGetSymbols ? stubArray : function(object) {
          var result = [];
          while (object) {
            arrayPush(result, getSymbols(object));
            object = getPrototype(object);
          }
          return result;
        };

        /**
         * Gets the `toStringTag` of `value`.
         *
         * @private
         * @param {*} value The value to query.
         * @returns {string} Returns the `toStringTag`.
         */
        var getTag = baseGetTag;

        // Fallback for data views, maps, sets, and weak maps in IE 11 and promises in Node.js < 6.
        if ((DataView && getTag(new DataView(new ArrayBuffer(1))) != dataViewTag) ||
            (Map && getTag(new Map) != mapTag) ||
            (Promise && getTag(Promise.resolve()) != promiseTag) ||
            (Set && getTag(new Set) != setTag) ||
            (WeakMap && getTag(new WeakMap) != weakMapTag)) {
          getTag = function(value) {
            var result = baseGetTag(value),
                Ctor = result == objectTag ? value.constructor : undefined$1,
                ctorString = Ctor ? toSource(Ctor) : '';

            if (ctorString) {
              switch (ctorString) {
                case dataViewCtorString: return dataViewTag;
                case mapCtorString: return mapTag;
                case promiseCtorString: return promiseTag;
                case setCtorString: return setTag;
                case weakMapCtorString: return weakMapTag;
              }
            }
            return result;
          };
        }

        /**
         * Gets the view, applying any `transforms` to the `start` and `end` positions.
         *
         * @private
         * @param {number} start The start of the view.
         * @param {number} end The end of the view.
         * @param {Array} transforms The transformations to apply to the view.
         * @returns {Object} Returns an object containing the `start` and `end`
         *  positions of the view.
         */
        function getView(start, end, transforms) {
          var index = -1,
              length = transforms.length;

          while (++index < length) {
            var data = transforms[index],
                size = data.size;

            switch (data.type) {
              case 'drop':      start += size; break;
              case 'dropRight': end -= size; break;
              case 'take':      end = nativeMin(end, start + size); break;
              case 'takeRight': start = nativeMax(start, end - size); break;
            }
          }
          return { 'start': start, 'end': end };
        }

        /**
         * Extracts wrapper details from the `source` body comment.
         *
         * @private
         * @param {string} source The source to inspect.
         * @returns {Array} Returns the wrapper details.
         */
        function getWrapDetails(source) {
          var match = source.match(reWrapDetails);
          return match ? match[1].split(reSplitDetails) : [];
        }

        /**
         * Checks if `path` exists on `object`.
         *
         * @private
         * @param {Object} object The object to query.
         * @param {Array|string} path The path to check.
         * @param {Function} hasFunc The function to check properties.
         * @returns {boolean} Returns `true` if `path` exists, else `false`.
         */
        function hasPath(object, path, hasFunc) {
          path = castPath(path, object);

          var index = -1,
              length = path.length,
              result = false;

          while (++index < length) {
            var key = toKey(path[index]);
            if (!(result = object != null && hasFunc(object, key))) {
              break;
            }
            object = object[key];
          }
          if (result || ++index != length) {
            return result;
          }
          length = object == null ? 0 : object.length;
          return !!length && isLength(length) && isIndex(key, length) &&
            (isArray(object) || isArguments(object));
        }

        /**
         * Initializes an array clone.
         *
         * @private
         * @param {Array} array The array to clone.
         * @returns {Array} Returns the initialized clone.
         */
        function initCloneArray(array) {
          var length = array.length,
              result = new array.constructor(length);

          // Add properties assigned by `RegExp#exec`.
          if (length && typeof array[0] == 'string' && hasOwnProperty.call(array, 'index')) {
            result.index = array.index;
            result.input = array.input;
          }
          return result;
        }

        /**
         * Initializes an object clone.
         *
         * @private
         * @param {Object} object The object to clone.
         * @returns {Object} Returns the initialized clone.
         */
        function initCloneObject(object) {
          return (typeof object.constructor == 'function' && !isPrototype(object))
            ? baseCreate(getPrototype(object))
            : {};
        }

        /**
         * Initializes an object clone based on its `toStringTag`.
         *
         * **Note:** This function only supports cloning values with tags of
         * `Boolean`, `Date`, `Error`, `Map`, `Number`, `RegExp`, `Set`, or `String`.
         *
         * @private
         * @param {Object} object The object to clone.
         * @param {string} tag The `toStringTag` of the object to clone.
         * @param {boolean} [isDeep] Specify a deep clone.
         * @returns {Object} Returns the initialized clone.
         */
        function initCloneByTag(object, tag, isDeep) {
          var Ctor = object.constructor;
          switch (tag) {
            case arrayBufferTag:
              return cloneArrayBuffer(object);

            case boolTag:
            case dateTag:
              return new Ctor(+object);

            case dataViewTag:
              return cloneDataView(object, isDeep);

            case float32Tag: case float64Tag:
            case int8Tag: case int16Tag: case int32Tag:
            case uint8Tag: case uint8ClampedTag: case uint16Tag: case uint32Tag:
              return cloneTypedArray(object, isDeep);

            case mapTag:
              return new Ctor;

            case numberTag:
            case stringTag:
              return new Ctor(object);

            case regexpTag:
              return cloneRegExp(object);

            case setTag:
              return new Ctor;

            case symbolTag:
              return cloneSymbol(object);
          }
        }

        /**
         * Inserts wrapper `details` in a comment at the top of the `source` body.
         *
         * @private
         * @param {string} source The source to modify.
         * @returns {Array} details The details to insert.
         * @returns {string} Returns the modified source.
         */
        function insertWrapDetails(source, details) {
          var length = details.length;
          if (!length) {
            return source;
          }
          var lastIndex = length - 1;
          details[lastIndex] = (length > 1 ? '& ' : '') + details[lastIndex];
          details = details.join(length > 2 ? ', ' : ' ');
          return source.replace(reWrapComment, '{\n/* [wrapped with ' + details + '] */\n');
        }

        /**
         * Checks if `value` is a flattenable `arguments` object or array.
         *
         * @private
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is flattenable, else `false`.
         */
        function isFlattenable(value) {
          return isArray(value) || isArguments(value) ||
            !!(spreadableSymbol && value && value[spreadableSymbol]);
        }

        /**
         * Checks if `value` is a valid array-like index.
         *
         * @private
         * @param {*} value The value to check.
         * @param {number} [length=MAX_SAFE_INTEGER] The upper bounds of a valid index.
         * @returns {boolean} Returns `true` if `value` is a valid index, else `false`.
         */
        function isIndex(value, length) {
          var type = typeof value;
          length = length == null ? MAX_SAFE_INTEGER : length;

          return !!length &&
            (type == 'number' ||
              (type != 'symbol' && reIsUint.test(value))) &&
                (value > -1 && value % 1 == 0 && value < length);
        }

        /**
         * Checks if the given arguments are from an iteratee call.
         *
         * @private
         * @param {*} value The potential iteratee value argument.
         * @param {*} index The potential iteratee index or key argument.
         * @param {*} object The potential iteratee object argument.
         * @returns {boolean} Returns `true` if the arguments are from an iteratee call,
         *  else `false`.
         */
        function isIterateeCall(value, index, object) {
          if (!isObject(object)) {
            return false;
          }
          var type = typeof index;
          if (type == 'number'
                ? (isArrayLike(object) && isIndex(index, object.length))
                : (type == 'string' && index in object)
              ) {
            return eq(object[index], value);
          }
          return false;
        }

        /**
         * Checks if `value` is a property name and not a property path.
         *
         * @private
         * @param {*} value The value to check.
         * @param {Object} [object] The object to query keys on.
         * @returns {boolean} Returns `true` if `value` is a property name, else `false`.
         */
        function isKey(value, object) {
          if (isArray(value)) {
            return false;
          }
          var type = typeof value;
          if (type == 'number' || type == 'symbol' || type == 'boolean' ||
              value == null || isSymbol(value)) {
            return true;
          }
          return reIsPlainProp.test(value) || !reIsDeepProp.test(value) ||
            (object != null && value in Object(object));
        }

        /**
         * Checks if `value` is suitable for use as unique object key.
         *
         * @private
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is suitable, else `false`.
         */
        function isKeyable(value) {
          var type = typeof value;
          return (type == 'string' || type == 'number' || type == 'symbol' || type == 'boolean')
            ? (value !== '__proto__')
            : (value === null);
        }

        /**
         * Checks if `func` has a lazy counterpart.
         *
         * @private
         * @param {Function} func The function to check.
         * @returns {boolean} Returns `true` if `func` has a lazy counterpart,
         *  else `false`.
         */
        function isLaziable(func) {
          var funcName = getFuncName(func),
              other = lodash[funcName];

          if (typeof other != 'function' || !(funcName in LazyWrapper.prototype)) {
            return false;
          }
          if (func === other) {
            return true;
          }
          var data = getData(other);
          return !!data && func === data[0];
        }

        /**
         * Checks if `func` has its source masked.
         *
         * @private
         * @param {Function} func The function to check.
         * @returns {boolean} Returns `true` if `func` is masked, else `false`.
         */
        function isMasked(func) {
          return !!maskSrcKey && (maskSrcKey in func);
        }

        /**
         * Checks if `func` is capable of being masked.
         *
         * @private
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `func` is maskable, else `false`.
         */
        var isMaskable = coreJsData ? isFunction : stubFalse;

        /**
         * Checks if `value` is likely a prototype object.
         *
         * @private
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is a prototype, else `false`.
         */
        function isPrototype(value) {
          var Ctor = value && value.constructor,
              proto = (typeof Ctor == 'function' && Ctor.prototype) || objectProto;

          return value === proto;
        }

        /**
         * Checks if `value` is suitable for strict equality comparisons, i.e. `===`.
         *
         * @private
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` if suitable for strict
         *  equality comparisons, else `false`.
         */
        function isStrictComparable(value) {
          return value === value && !isObject(value);
        }

        /**
         * A specialized version of `matchesProperty` for source values suitable
         * for strict equality comparisons, i.e. `===`.
         *
         * @private
         * @param {string} key The key of the property to get.
         * @param {*} srcValue The value to match.
         * @returns {Function} Returns the new spec function.
         */
        function matchesStrictComparable(key, srcValue) {
          return function(object) {
            if (object == null) {
              return false;
            }
            return object[key] === srcValue &&
              (srcValue !== undefined$1 || (key in Object(object)));
          };
        }

        /**
         * A specialized version of `_.memoize` which clears the memoized function's
         * cache when it exceeds `MAX_MEMOIZE_SIZE`.
         *
         * @private
         * @param {Function} func The function to have its output memoized.
         * @returns {Function} Returns the new memoized function.
         */
        function memoizeCapped(func) {
          var result = memoize(func, function(key) {
            if (cache.size === MAX_MEMOIZE_SIZE) {
              cache.clear();
            }
            return key;
          });

          var cache = result.cache;
          return result;
        }

        /**
         * Merges the function metadata of `source` into `data`.
         *
         * Merging metadata reduces the number of wrappers used to invoke a function.
         * This is possible because methods like `_.bind`, `_.curry`, and `_.partial`
         * may be applied regardless of execution order. Methods like `_.ary` and
         * `_.rearg` modify function arguments, making the order in which they are
         * executed important, preventing the merging of metadata. However, we make
         * an exception for a safe combined case where curried functions have `_.ary`
         * and or `_.rearg` applied.
         *
         * @private
         * @param {Array} data The destination metadata.
         * @param {Array} source The source metadata.
         * @returns {Array} Returns `data`.
         */
        function mergeData(data, source) {
          var bitmask = data[1],
              srcBitmask = source[1],
              newBitmask = bitmask | srcBitmask,
              isCommon = newBitmask < (WRAP_BIND_FLAG | WRAP_BIND_KEY_FLAG | WRAP_ARY_FLAG);

          var isCombo =
            ((srcBitmask == WRAP_ARY_FLAG) && (bitmask == WRAP_CURRY_FLAG)) ||
            ((srcBitmask == WRAP_ARY_FLAG) && (bitmask == WRAP_REARG_FLAG) && (data[7].length <= source[8])) ||
            ((srcBitmask == (WRAP_ARY_FLAG | WRAP_REARG_FLAG)) && (source[7].length <= source[8]) && (bitmask == WRAP_CURRY_FLAG));

          // Exit early if metadata can't be merged.
          if (!(isCommon || isCombo)) {
            return data;
          }
          // Use source `thisArg` if available.
          if (srcBitmask & WRAP_BIND_FLAG) {
            data[2] = source[2];
            // Set when currying a bound function.
            newBitmask |= bitmask & WRAP_BIND_FLAG ? 0 : WRAP_CURRY_BOUND_FLAG;
          }
          // Compose partial arguments.
          var value = source[3];
          if (value) {
            var partials = data[3];
            data[3] = partials ? composeArgs(partials, value, source[4]) : value;
            data[4] = partials ? replaceHolders(data[3], PLACEHOLDER) : source[4];
          }
          // Compose partial right arguments.
          value = source[5];
          if (value) {
            partials = data[5];
            data[5] = partials ? composeArgsRight(partials, value, source[6]) : value;
            data[6] = partials ? replaceHolders(data[5], PLACEHOLDER) : source[6];
          }
          // Use source `argPos` if available.
          value = source[7];
          if (value) {
            data[7] = value;
          }
          // Use source `ary` if it's smaller.
          if (srcBitmask & WRAP_ARY_FLAG) {
            data[8] = data[8] == null ? source[8] : nativeMin(data[8], source[8]);
          }
          // Use source `arity` if one is not provided.
          if (data[9] == null) {
            data[9] = source[9];
          }
          // Use source `func` and merge bitmasks.
          data[0] = source[0];
          data[1] = newBitmask;

          return data;
        }

        /**
         * This function is like
         * [`Object.keys`](http://ecma-international.org/ecma-262/7.0/#sec-object.keys)
         * except that it includes inherited enumerable properties.
         *
         * @private
         * @param {Object} object The object to query.
         * @returns {Array} Returns the array of property names.
         */
        function nativeKeysIn(object) {
          var result = [];
          if (object != null) {
            for (var key in Object(object)) {
              result.push(key);
            }
          }
          return result;
        }

        /**
         * Converts `value` to a string using `Object.prototype.toString`.
         *
         * @private
         * @param {*} value The value to convert.
         * @returns {string} Returns the converted string.
         */
        function objectToString(value) {
          return nativeObjectToString.call(value);
        }

        /**
         * A specialized version of `baseRest` which transforms the rest array.
         *
         * @private
         * @param {Function} func The function to apply a rest parameter to.
         * @param {number} [start=func.length-1] The start position of the rest parameter.
         * @param {Function} transform The rest array transform.
         * @returns {Function} Returns the new function.
         */
        function overRest(func, start, transform) {
          start = nativeMax(start === undefined$1 ? (func.length - 1) : start, 0);
          return function() {
            var args = arguments,
                index = -1,
                length = nativeMax(args.length - start, 0),
                array = Array(length);

            while (++index < length) {
              array[index] = args[start + index];
            }
            index = -1;
            var otherArgs = Array(start + 1);
            while (++index < start) {
              otherArgs[index] = args[index];
            }
            otherArgs[start] = transform(array);
            return apply(func, this, otherArgs);
          };
        }

        /**
         * Gets the parent value at `path` of `object`.
         *
         * @private
         * @param {Object} object The object to query.
         * @param {Array} path The path to get the parent value of.
         * @returns {*} Returns the parent value.
         */
        function parent(object, path) {
          return path.length < 2 ? object : baseGet(object, baseSlice(path, 0, -1));
        }

        /**
         * Reorder `array` according to the specified indexes where the element at
         * the first index is assigned as the first element, the element at
         * the second index is assigned as the second element, and so on.
         *
         * @private
         * @param {Array} array The array to reorder.
         * @param {Array} indexes The arranged array indexes.
         * @returns {Array} Returns `array`.
         */
        function reorder(array, indexes) {
          var arrLength = array.length,
              length = nativeMin(indexes.length, arrLength),
              oldArray = copyArray(array);

          while (length--) {
            var index = indexes[length];
            array[length] = isIndex(index, arrLength) ? oldArray[index] : undefined$1;
          }
          return array;
        }

        /**
         * Gets the value at `key`, unless `key` is "__proto__" or "constructor".
         *
         * @private
         * @param {Object} object The object to query.
         * @param {string} key The key of the property to get.
         * @returns {*} Returns the property value.
         */
        function safeGet(object, key) {
          if (key === 'constructor' && typeof object[key] === 'function') {
            return;
          }

          if (key == '__proto__') {
            return;
          }

          return object[key];
        }

        /**
         * Sets metadata for `func`.
         *
         * **Note:** If this function becomes hot, i.e. is invoked a lot in a short
         * period of time, it will trip its breaker and transition to an identity
         * function to avoid garbage collection pauses in V8. See
         * [V8 issue 2070](https://bugs.chromium.org/p/v8/issues/detail?id=2070)
         * for more details.
         *
         * @private
         * @param {Function} func The function to associate metadata with.
         * @param {*} data The metadata.
         * @returns {Function} Returns `func`.
         */
        var setData = shortOut(baseSetData);

        /**
         * A simple wrapper around the global [`setTimeout`](https://mdn.io/setTimeout).
         *
         * @private
         * @param {Function} func The function to delay.
         * @param {number} wait The number of milliseconds to delay invocation.
         * @returns {number|Object} Returns the timer id or timeout object.
         */
        var setTimeout = ctxSetTimeout || function(func, wait) {
          return root.setTimeout(func, wait);
        };

        /**
         * Sets the `toString` method of `func` to return `string`.
         *
         * @private
         * @param {Function} func The function to modify.
         * @param {Function} string The `toString` result.
         * @returns {Function} Returns `func`.
         */
        var setToString = shortOut(baseSetToString);

        /**
         * Sets the `toString` method of `wrapper` to mimic the source of `reference`
         * with wrapper details in a comment at the top of the source body.
         *
         * @private
         * @param {Function} wrapper The function to modify.
         * @param {Function} reference The reference function.
         * @param {number} bitmask The bitmask flags. See `createWrap` for more details.
         * @returns {Function} Returns `wrapper`.
         */
        function setWrapToString(wrapper, reference, bitmask) {
          var source = (reference + '');
          return setToString(wrapper, insertWrapDetails(source, updateWrapDetails(getWrapDetails(source), bitmask)));
        }

        /**
         * Creates a function that'll short out and invoke `identity` instead
         * of `func` when it's called `HOT_COUNT` or more times in `HOT_SPAN`
         * milliseconds.
         *
         * @private
         * @param {Function} func The function to restrict.
         * @returns {Function} Returns the new shortable function.
         */
        function shortOut(func) {
          var count = 0,
              lastCalled = 0;

          return function() {
            var stamp = nativeNow(),
                remaining = HOT_SPAN - (stamp - lastCalled);

            lastCalled = stamp;
            if (remaining > 0) {
              if (++count >= HOT_COUNT) {
                return arguments[0];
              }
            } else {
              count = 0;
            }
            return func.apply(undefined$1, arguments);
          };
        }

        /**
         * A specialized version of `_.shuffle` which mutates and sets the size of `array`.
         *
         * @private
         * @param {Array} array The array to shuffle.
         * @param {number} [size=array.length] The size of `array`.
         * @returns {Array} Returns `array`.
         */
        function shuffleSelf(array, size) {
          var index = -1,
              length = array.length,
              lastIndex = length - 1;

          size = size === undefined$1 ? length : size;
          while (++index < size) {
            var rand = baseRandom(index, lastIndex),
                value = array[rand];

            array[rand] = array[index];
            array[index] = value;
          }
          array.length = size;
          return array;
        }

        /**
         * Converts `string` to a property path array.
         *
         * @private
         * @param {string} string The string to convert.
         * @returns {Array} Returns the property path array.
         */
        var stringToPath = memoizeCapped(function(string) {
          var result = [];
          if (string.charCodeAt(0) === 46 /* . */) {
            result.push('');
          }
          string.replace(rePropName, function(match, number, quote, subString) {
            result.push(quote ? subString.replace(reEscapeChar, '$1') : (number || match));
          });
          return result;
        });

        /**
         * Converts `value` to a string key if it's not a string or symbol.
         *
         * @private
         * @param {*} value The value to inspect.
         * @returns {string|symbol} Returns the key.
         */
        function toKey(value) {
          if (typeof value == 'string' || isSymbol(value)) {
            return value;
          }
          var result = (value + '');
          return (result == '0' && (1 / value) == -INFINITY) ? '-0' : result;
        }

        /**
         * Converts `func` to its source code.
         *
         * @private
         * @param {Function} func The function to convert.
         * @returns {string} Returns the source code.
         */
        function toSource(func) {
          if (func != null) {
            try {
              return funcToString.call(func);
            } catch (e) {}
            try {
              return (func + '');
            } catch (e) {}
          }
          return '';
        }

        /**
         * Updates wrapper `details` based on `bitmask` flags.
         *
         * @private
         * @returns {Array} details The details to modify.
         * @param {number} bitmask The bitmask flags. See `createWrap` for more details.
         * @returns {Array} Returns `details`.
         */
        function updateWrapDetails(details, bitmask) {
          arrayEach(wrapFlags, function(pair) {
            var value = '_.' + pair[0];
            if ((bitmask & pair[1]) && !arrayIncludes(details, value)) {
              details.push(value);
            }
          });
          return details.sort();
        }

        /**
         * Creates a clone of `wrapper`.
         *
         * @private
         * @param {Object} wrapper The wrapper to clone.
         * @returns {Object} Returns the cloned wrapper.
         */
        function wrapperClone(wrapper) {
          if (wrapper instanceof LazyWrapper) {
            return wrapper.clone();
          }
          var result = new LodashWrapper(wrapper.__wrapped__, wrapper.__chain__);
          result.__actions__ = copyArray(wrapper.__actions__);
          result.__index__  = wrapper.__index__;
          result.__values__ = wrapper.__values__;
          return result;
        }

        /*------------------------------------------------------------------------*/

        /**
         * Creates an array of elements split into groups the length of `size`.
         * If `array` can't be split evenly, the final chunk will be the remaining
         * elements.
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category Array
         * @param {Array} array The array to process.
         * @param {number} [size=1] The length of each chunk
         * @param- {Object} [guard] Enables use as an iteratee for methods like `_.map`.
         * @returns {Array} Returns the new array of chunks.
         * @example
         *
         * _.chunk(['a', 'b', 'c', 'd'], 2);
         * // => [['a', 'b'], ['c', 'd']]
         *
         * _.chunk(['a', 'b', 'c', 'd'], 3);
         * // => [['a', 'b', 'c'], ['d']]
         */
        function chunk(array, size, guard) {
          if ((guard ? isIterateeCall(array, size, guard) : size === undefined$1)) {
            size = 1;
          } else {
            size = nativeMax(toInteger(size), 0);
          }
          var length = array == null ? 0 : array.length;
          if (!length || size < 1) {
            return [];
          }
          var index = 0,
              resIndex = 0,
              result = Array(nativeCeil(length / size));

          while (index < length) {
            result[resIndex++] = baseSlice(array, index, (index += size));
          }
          return result;
        }

        /**
         * Creates an array with all falsey values removed. The values `false`, `null`,
         * `0`, `""`, `undefined`, and `NaN` are falsey.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Array
         * @param {Array} array The array to compact.
         * @returns {Array} Returns the new array of filtered values.
         * @example
         *
         * _.compact([0, 1, false, 2, '', 3]);
         * // => [1, 2, 3]
         */
        function compact(array) {
          var index = -1,
              length = array == null ? 0 : array.length,
              resIndex = 0,
              result = [];

          while (++index < length) {
            var value = array[index];
            if (value) {
              result[resIndex++] = value;
            }
          }
          return result;
        }

        /**
         * Creates a new array concatenating `array` with any additional arrays
         * and/or values.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Array
         * @param {Array} array The array to concatenate.
         * @param {...*} [values] The values to concatenate.
         * @returns {Array} Returns the new concatenated array.
         * @example
         *
         * var array = [1];
         * var other = _.concat(array, 2, [3], [[4]]);
         *
         * console.log(other);
         * // => [1, 2, 3, [4]]
         *
         * console.log(array);
         * // => [1]
         */
        function concat() {
          var length = arguments.length;
          if (!length) {
            return [];
          }
          var args = Array(length - 1),
              array = arguments[0],
              index = length;

          while (index--) {
            args[index - 1] = arguments[index];
          }
          return arrayPush(isArray(array) ? copyArray(array) : [array], baseFlatten(args, 1));
        }

        /**
         * Creates an array of `array` values not included in the other given arrays
         * using [`SameValueZero`](http://ecma-international.org/ecma-262/7.0/#sec-samevaluezero)
         * for equality comparisons. The order and references of result values are
         * determined by the first array.
         *
         * **Note:** Unlike `_.pullAll`, this method returns a new array.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Array
         * @param {Array} array The array to inspect.
         * @param {...Array} [values] The values to exclude.
         * @returns {Array} Returns the new array of filtered values.
         * @see _.without, _.xor
         * @example
         *
         * _.difference([2, 1], [2, 3]);
         * // => [1]
         */
        var difference = baseRest(function(array, values) {
          return isArrayLikeObject(array)
            ? baseDifference(array, baseFlatten(values, 1, isArrayLikeObject, true))
            : [];
        });

        /**
         * This method is like `_.difference` except that it accepts `iteratee` which
         * is invoked for each element of `array` and `values` to generate the criterion
         * by which they're compared. The order and references of result values are
         * determined by the first array. The iteratee is invoked with one argument:
         * (value).
         *
         * **Note:** Unlike `_.pullAllBy`, this method returns a new array.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Array
         * @param {Array} array The array to inspect.
         * @param {...Array} [values] The values to exclude.
         * @param {Function} [iteratee=_.identity] The iteratee invoked per element.
         * @returns {Array} Returns the new array of filtered values.
         * @example
         *
         * _.differenceBy([2.1, 1.2], [2.3, 3.4], Math.floor);
         * // => [1.2]
         *
         * // The `_.property` iteratee shorthand.
         * _.differenceBy([{ 'x': 2 }, { 'x': 1 }], [{ 'x': 1 }], 'x');
         * // => [{ 'x': 2 }]
         */
        var differenceBy = baseRest(function(array, values) {
          var iteratee = last(values);
          if (isArrayLikeObject(iteratee)) {
            iteratee = undefined$1;
          }
          return isArrayLikeObject(array)
            ? baseDifference(array, baseFlatten(values, 1, isArrayLikeObject, true), getIteratee(iteratee, 2))
            : [];
        });

        /**
         * This method is like `_.difference` except that it accepts `comparator`
         * which is invoked to compare elements of `array` to `values`. The order and
         * references of result values are determined by the first array. The comparator
         * is invoked with two arguments: (arrVal, othVal).
         *
         * **Note:** Unlike `_.pullAllWith`, this method returns a new array.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Array
         * @param {Array} array The array to inspect.
         * @param {...Array} [values] The values to exclude.
         * @param {Function} [comparator] The comparator invoked per element.
         * @returns {Array} Returns the new array of filtered values.
         * @example
         *
         * var objects = [{ 'x': 1, 'y': 2 }, { 'x': 2, 'y': 1 }];
         *
         * _.differenceWith(objects, [{ 'x': 1, 'y': 2 }], _.isEqual);
         * // => [{ 'x': 2, 'y': 1 }]
         */
        var differenceWith = baseRest(function(array, values) {
          var comparator = last(values);
          if (isArrayLikeObject(comparator)) {
            comparator = undefined$1;
          }
          return isArrayLikeObject(array)
            ? baseDifference(array, baseFlatten(values, 1, isArrayLikeObject, true), undefined$1, comparator)
            : [];
        });

        /**
         * Creates a slice of `array` with `n` elements dropped from the beginning.
         *
         * @static
         * @memberOf _
         * @since 0.5.0
         * @category Array
         * @param {Array} array The array to query.
         * @param {number} [n=1] The number of elements to drop.
         * @param- {Object} [guard] Enables use as an iteratee for methods like `_.map`.
         * @returns {Array} Returns the slice of `array`.
         * @example
         *
         * _.drop([1, 2, 3]);
         * // => [2, 3]
         *
         * _.drop([1, 2, 3], 2);
         * // => [3]
         *
         * _.drop([1, 2, 3], 5);
         * // => []
         *
         * _.drop([1, 2, 3], 0);
         * // => [1, 2, 3]
         */
        function drop(array, n, guard) {
          var length = array == null ? 0 : array.length;
          if (!length) {
            return [];
          }
          n = (guard || n === undefined$1) ? 1 : toInteger(n);
          return baseSlice(array, n < 0 ? 0 : n, length);
        }

        /**
         * Creates a slice of `array` with `n` elements dropped from the end.
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category Array
         * @param {Array} array The array to query.
         * @param {number} [n=1] The number of elements to drop.
         * @param- {Object} [guard] Enables use as an iteratee for methods like `_.map`.
         * @returns {Array} Returns the slice of `array`.
         * @example
         *
         * _.dropRight([1, 2, 3]);
         * // => [1, 2]
         *
         * _.dropRight([1, 2, 3], 2);
         * // => [1]
         *
         * _.dropRight([1, 2, 3], 5);
         * // => []
         *
         * _.dropRight([1, 2, 3], 0);
         * // => [1, 2, 3]
         */
        function dropRight(array, n, guard) {
          var length = array == null ? 0 : array.length;
          if (!length) {
            return [];
          }
          n = (guard || n === undefined$1) ? 1 : toInteger(n);
          n = length - n;
          return baseSlice(array, 0, n < 0 ? 0 : n);
        }

        /**
         * Creates a slice of `array` excluding elements dropped from the end.
         * Elements are dropped until `predicate` returns falsey. The predicate is
         * invoked with three arguments: (value, index, array).
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category Array
         * @param {Array} array The array to query.
         * @param {Function} [predicate=_.identity] The function invoked per iteration.
         * @returns {Array} Returns the slice of `array`.
         * @example
         *
         * var users = [
         *   { 'user': 'barney',  'active': true },
         *   { 'user': 'fred',    'active': false },
         *   { 'user': 'pebbles', 'active': false }
         * ];
         *
         * _.dropRightWhile(users, function(o) { return !o.active; });
         * // => objects for ['barney']
         *
         * // The `_.matches` iteratee shorthand.
         * _.dropRightWhile(users, { 'user': 'pebbles', 'active': false });
         * // => objects for ['barney', 'fred']
         *
         * // The `_.matchesProperty` iteratee shorthand.
         * _.dropRightWhile(users, ['active', false]);
         * // => objects for ['barney']
         *
         * // The `_.property` iteratee shorthand.
         * _.dropRightWhile(users, 'active');
         * // => objects for ['barney', 'fred', 'pebbles']
         */
        function dropRightWhile(array, predicate) {
          return (array && array.length)
            ? baseWhile(array, getIteratee(predicate, 3), true, true)
            : [];
        }

        /**
         * Creates a slice of `array` excluding elements dropped from the beginning.
         * Elements are dropped until `predicate` returns falsey. The predicate is
         * invoked with three arguments: (value, index, array).
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category Array
         * @param {Array} array The array to query.
         * @param {Function} [predicate=_.identity] The function invoked per iteration.
         * @returns {Array} Returns the slice of `array`.
         * @example
         *
         * var users = [
         *   { 'user': 'barney',  'active': false },
         *   { 'user': 'fred',    'active': false },
         *   { 'user': 'pebbles', 'active': true }
         * ];
         *
         * _.dropWhile(users, function(o) { return !o.active; });
         * // => objects for ['pebbles']
         *
         * // The `_.matches` iteratee shorthand.
         * _.dropWhile(users, { 'user': 'barney', 'active': false });
         * // => objects for ['fred', 'pebbles']
         *
         * // The `_.matchesProperty` iteratee shorthand.
         * _.dropWhile(users, ['active', false]);
         * // => objects for ['pebbles']
         *
         * // The `_.property` iteratee shorthand.
         * _.dropWhile(users, 'active');
         * // => objects for ['barney', 'fred', 'pebbles']
         */
        function dropWhile(array, predicate) {
          return (array && array.length)
            ? baseWhile(array, getIteratee(predicate, 3), true)
            : [];
        }

        /**
         * Fills elements of `array` with `value` from `start` up to, but not
         * including, `end`.
         *
         * **Note:** This method mutates `array`.
         *
         * @static
         * @memberOf _
         * @since 3.2.0
         * @category Array
         * @param {Array} array The array to fill.
         * @param {*} value The value to fill `array` with.
         * @param {number} [start=0] The start position.
         * @param {number} [end=array.length] The end position.
         * @returns {Array} Returns `array`.
         * @example
         *
         * var array = [1, 2, 3];
         *
         * _.fill(array, 'a');
         * console.log(array);
         * // => ['a', 'a', 'a']
         *
         * _.fill(Array(3), 2);
         * // => [2, 2, 2]
         *
         * _.fill([4, 6, 8, 10], '*', 1, 3);
         * // => [4, '*', '*', 10]
         */
        function fill(array, value, start, end) {
          var length = array == null ? 0 : array.length;
          if (!length) {
            return [];
          }
          if (start && typeof start != 'number' && isIterateeCall(array, value, start)) {
            start = 0;
            end = length;
          }
          return baseFill(array, value, start, end);
        }

        /**
         * This method is like `_.find` except that it returns the index of the first
         * element `predicate` returns truthy for instead of the element itself.
         *
         * @static
         * @memberOf _
         * @since 1.1.0
         * @category Array
         * @param {Array} array The array to inspect.
         * @param {Function} [predicate=_.identity] The function invoked per iteration.
         * @param {number} [fromIndex=0] The index to search from.
         * @returns {number} Returns the index of the found element, else `-1`.
         * @example
         *
         * var users = [
         *   { 'user': 'barney',  'active': false },
         *   { 'user': 'fred',    'active': false },
         *   { 'user': 'pebbles', 'active': true }
         * ];
         *
         * _.findIndex(users, function(o) { return o.user == 'barney'; });
         * // => 0
         *
         * // The `_.matches` iteratee shorthand.
         * _.findIndex(users, { 'user': 'fred', 'active': false });
         * // => 1
         *
         * // The `_.matchesProperty` iteratee shorthand.
         * _.findIndex(users, ['active', false]);
         * // => 0
         *
         * // The `_.property` iteratee shorthand.
         * _.findIndex(users, 'active');
         * // => 2
         */
        function findIndex(array, predicate, fromIndex) {
          var length = array == null ? 0 : array.length;
          if (!length) {
            return -1;
          }
          var index = fromIndex == null ? 0 : toInteger(fromIndex);
          if (index < 0) {
            index = nativeMax(length + index, 0);
          }
          return baseFindIndex(array, getIteratee(predicate, 3), index);
        }

        /**
         * This method is like `_.findIndex` except that it iterates over elements
         * of `collection` from right to left.
         *
         * @static
         * @memberOf _
         * @since 2.0.0
         * @category Array
         * @param {Array} array The array to inspect.
         * @param {Function} [predicate=_.identity] The function invoked per iteration.
         * @param {number} [fromIndex=array.length-1] The index to search from.
         * @returns {number} Returns the index of the found element, else `-1`.
         * @example
         *
         * var users = [
         *   { 'user': 'barney',  'active': true },
         *   { 'user': 'fred',    'active': false },
         *   { 'user': 'pebbles', 'active': false }
         * ];
         *
         * _.findLastIndex(users, function(o) { return o.user == 'pebbles'; });
         * // => 2
         *
         * // The `_.matches` iteratee shorthand.
         * _.findLastIndex(users, { 'user': 'barney', 'active': true });
         * // => 0
         *
         * // The `_.matchesProperty` iteratee shorthand.
         * _.findLastIndex(users, ['active', false]);
         * // => 2
         *
         * // The `_.property` iteratee shorthand.
         * _.findLastIndex(users, 'active');
         * // => 0
         */
        function findLastIndex(array, predicate, fromIndex) {
          var length = array == null ? 0 : array.length;
          if (!length) {
            return -1;
          }
          var index = length - 1;
          if (fromIndex !== undefined$1) {
            index = toInteger(fromIndex);
            index = fromIndex < 0
              ? nativeMax(length + index, 0)
              : nativeMin(index, length - 1);
          }
          return baseFindIndex(array, getIteratee(predicate, 3), index, true);
        }

        /**
         * Flattens `array` a single level deep.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Array
         * @param {Array} array The array to flatten.
         * @returns {Array} Returns the new flattened array.
         * @example
         *
         * _.flatten([1, [2, [3, [4]], 5]]);
         * // => [1, 2, [3, [4]], 5]
         */
        function flatten(array) {
          var length = array == null ? 0 : array.length;
          return length ? baseFlatten(array, 1) : [];
        }

        /**
         * Recursively flattens `array`.
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category Array
         * @param {Array} array The array to flatten.
         * @returns {Array} Returns the new flattened array.
         * @example
         *
         * _.flattenDeep([1, [2, [3, [4]], 5]]);
         * // => [1, 2, 3, 4, 5]
         */
        function flattenDeep(array) {
          var length = array == null ? 0 : array.length;
          return length ? baseFlatten(array, INFINITY) : [];
        }

        /**
         * Recursively flatten `array` up to `depth` times.
         *
         * @static
         * @memberOf _
         * @since 4.4.0
         * @category Array
         * @param {Array} array The array to flatten.
         * @param {number} [depth=1] The maximum recursion depth.
         * @returns {Array} Returns the new flattened array.
         * @example
         *
         * var array = [1, [2, [3, [4]], 5]];
         *
         * _.flattenDepth(array, 1);
         * // => [1, 2, [3, [4]], 5]
         *
         * _.flattenDepth(array, 2);
         * // => [1, 2, 3, [4], 5]
         */
        function flattenDepth(array, depth) {
          var length = array == null ? 0 : array.length;
          if (!length) {
            return [];
          }
          depth = depth === undefined$1 ? 1 : toInteger(depth);
          return baseFlatten(array, depth);
        }

        /**
         * The inverse of `_.toPairs`; this method returns an object composed
         * from key-value `pairs`.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Array
         * @param {Array} pairs The key-value pairs.
         * @returns {Object} Returns the new object.
         * @example
         *
         * _.fromPairs([['a', 1], ['b', 2]]);
         * // => { 'a': 1, 'b': 2 }
         */
        function fromPairs(pairs) {
          var index = -1,
              length = pairs == null ? 0 : pairs.length,
              result = {};

          while (++index < length) {
            var pair = pairs[index];
            result[pair[0]] = pair[1];
          }
          return result;
        }

        /**
         * Gets the first element of `array`.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @alias first
         * @category Array
         * @param {Array} array The array to query.
         * @returns {*} Returns the first element of `array`.
         * @example
         *
         * _.head([1, 2, 3]);
         * // => 1
         *
         * _.head([]);
         * // => undefined
         */
        function head(array) {
          return (array && array.length) ? array[0] : undefined$1;
        }

        /**
         * Gets the index at which the first occurrence of `value` is found in `array`
         * using [`SameValueZero`](http://ecma-international.org/ecma-262/7.0/#sec-samevaluezero)
         * for equality comparisons. If `fromIndex` is negative, it's used as the
         * offset from the end of `array`.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Array
         * @param {Array} array The array to inspect.
         * @param {*} value The value to search for.
         * @param {number} [fromIndex=0] The index to search from.
         * @returns {number} Returns the index of the matched value, else `-1`.
         * @example
         *
         * _.indexOf([1, 2, 1, 2], 2);
         * // => 1
         *
         * // Search from the `fromIndex`.
         * _.indexOf([1, 2, 1, 2], 2, 2);
         * // => 3
         */
        function indexOf(array, value, fromIndex) {
          var length = array == null ? 0 : array.length;
          if (!length) {
            return -1;
          }
          var index = fromIndex == null ? 0 : toInteger(fromIndex);
          if (index < 0) {
            index = nativeMax(length + index, 0);
          }
          return baseIndexOf(array, value, index);
        }

        /**
         * Gets all but the last element of `array`.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Array
         * @param {Array} array The array to query.
         * @returns {Array} Returns the slice of `array`.
         * @example
         *
         * _.initial([1, 2, 3]);
         * // => [1, 2]
         */
        function initial(array) {
          var length = array == null ? 0 : array.length;
          return length ? baseSlice(array, 0, -1) : [];
        }

        /**
         * Creates an array of unique values that are included in all given arrays
         * using [`SameValueZero`](http://ecma-international.org/ecma-262/7.0/#sec-samevaluezero)
         * for equality comparisons. The order and references of result values are
         * determined by the first array.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Array
         * @param {...Array} [arrays] The arrays to inspect.
         * @returns {Array} Returns the new array of intersecting values.
         * @example
         *
         * _.intersection([2, 1], [2, 3]);
         * // => [2]
         */
        var intersection = baseRest(function(arrays) {
          var mapped = arrayMap(arrays, castArrayLikeObject);
          return (mapped.length && mapped[0] === arrays[0])
            ? baseIntersection(mapped)
            : [];
        });

        /**
         * This method is like `_.intersection` except that it accepts `iteratee`
         * which is invoked for each element of each `arrays` to generate the criterion
         * by which they're compared. The order and references of result values are
         * determined by the first array. The iteratee is invoked with one argument:
         * (value).
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Array
         * @param {...Array} [arrays] The arrays to inspect.
         * @param {Function} [iteratee=_.identity] The iteratee invoked per element.
         * @returns {Array} Returns the new array of intersecting values.
         * @example
         *
         * _.intersectionBy([2.1, 1.2], [2.3, 3.4], Math.floor);
         * // => [2.1]
         *
         * // The `_.property` iteratee shorthand.
         * _.intersectionBy([{ 'x': 1 }], [{ 'x': 2 }, { 'x': 1 }], 'x');
         * // => [{ 'x': 1 }]
         */
        var intersectionBy = baseRest(function(arrays) {
          var iteratee = last(arrays),
              mapped = arrayMap(arrays, castArrayLikeObject);

          if (iteratee === last(mapped)) {
            iteratee = undefined$1;
          } else {
            mapped.pop();
          }
          return (mapped.length && mapped[0] === arrays[0])
            ? baseIntersection(mapped, getIteratee(iteratee, 2))
            : [];
        });

        /**
         * This method is like `_.intersection` except that it accepts `comparator`
         * which is invoked to compare elements of `arrays`. The order and references
         * of result values are determined by the first array. The comparator is
         * invoked with two arguments: (arrVal, othVal).
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Array
         * @param {...Array} [arrays] The arrays to inspect.
         * @param {Function} [comparator] The comparator invoked per element.
         * @returns {Array} Returns the new array of intersecting values.
         * @example
         *
         * var objects = [{ 'x': 1, 'y': 2 }, { 'x': 2, 'y': 1 }];
         * var others = [{ 'x': 1, 'y': 1 }, { 'x': 1, 'y': 2 }];
         *
         * _.intersectionWith(objects, others, _.isEqual);
         * // => [{ 'x': 1, 'y': 2 }]
         */
        var intersectionWith = baseRest(function(arrays) {
          var comparator = last(arrays),
              mapped = arrayMap(arrays, castArrayLikeObject);

          comparator = typeof comparator == 'function' ? comparator : undefined$1;
          if (comparator) {
            mapped.pop();
          }
          return (mapped.length && mapped[0] === arrays[0])
            ? baseIntersection(mapped, undefined$1, comparator)
            : [];
        });

        /**
         * Converts all elements in `array` into a string separated by `separator`.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Array
         * @param {Array} array The array to convert.
         * @param {string} [separator=','] The element separator.
         * @returns {string} Returns the joined string.
         * @example
         *
         * _.join(['a', 'b', 'c'], '~');
         * // => 'a~b~c'
         */
        function join(array, separator) {
          return array == null ? '' : nativeJoin.call(array, separator);
        }

        /**
         * Gets the last element of `array`.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Array
         * @param {Array} array The array to query.
         * @returns {*} Returns the last element of `array`.
         * @example
         *
         * _.last([1, 2, 3]);
         * // => 3
         */
        function last(array) {
          var length = array == null ? 0 : array.length;
          return length ? array[length - 1] : undefined$1;
        }

        /**
         * This method is like `_.indexOf` except that it iterates over elements of
         * `array` from right to left.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Array
         * @param {Array} array The array to inspect.
         * @param {*} value The value to search for.
         * @param {number} [fromIndex=array.length-1] The index to search from.
         * @returns {number} Returns the index of the matched value, else `-1`.
         * @example
         *
         * _.lastIndexOf([1, 2, 1, 2], 2);
         * // => 3
         *
         * // Search from the `fromIndex`.
         * _.lastIndexOf([1, 2, 1, 2], 2, 2);
         * // => 1
         */
        function lastIndexOf(array, value, fromIndex) {
          var length = array == null ? 0 : array.length;
          if (!length) {
            return -1;
          }
          var index = length;
          if (fromIndex !== undefined$1) {
            index = toInteger(fromIndex);
            index = index < 0 ? nativeMax(length + index, 0) : nativeMin(index, length - 1);
          }
          return value === value
            ? strictLastIndexOf(array, value, index)
            : baseFindIndex(array, baseIsNaN, index, true);
        }

        /**
         * Gets the element at index `n` of `array`. If `n` is negative, the nth
         * element from the end is returned.
         *
         * @static
         * @memberOf _
         * @since 4.11.0
         * @category Array
         * @param {Array} array The array to query.
         * @param {number} [n=0] The index of the element to return.
         * @returns {*} Returns the nth element of `array`.
         * @example
         *
         * var array = ['a', 'b', 'c', 'd'];
         *
         * _.nth(array, 1);
         * // => 'b'
         *
         * _.nth(array, -2);
         * // => 'c';
         */
        function nth(array, n) {
          return (array && array.length) ? baseNth(array, toInteger(n)) : undefined$1;
        }

        /**
         * Removes all given values from `array` using
         * [`SameValueZero`](http://ecma-international.org/ecma-262/7.0/#sec-samevaluezero)
         * for equality comparisons.
         *
         * **Note:** Unlike `_.without`, this method mutates `array`. Use `_.remove`
         * to remove elements from an array by predicate.
         *
         * @static
         * @memberOf _
         * @since 2.0.0
         * @category Array
         * @param {Array} array The array to modify.
         * @param {...*} [values] The values to remove.
         * @returns {Array} Returns `array`.
         * @example
         *
         * var array = ['a', 'b', 'c', 'a', 'b', 'c'];
         *
         * _.pull(array, 'a', 'c');
         * console.log(array);
         * // => ['b', 'b']
         */
        var pull = baseRest(pullAll);

        /**
         * This method is like `_.pull` except that it accepts an array of values to remove.
         *
         * **Note:** Unlike `_.difference`, this method mutates `array`.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Array
         * @param {Array} array The array to modify.
         * @param {Array} values The values to remove.
         * @returns {Array} Returns `array`.
         * @example
         *
         * var array = ['a', 'b', 'c', 'a', 'b', 'c'];
         *
         * _.pullAll(array, ['a', 'c']);
         * console.log(array);
         * // => ['b', 'b']
         */
        function pullAll(array, values) {
          return (array && array.length && values && values.length)
            ? basePullAll(array, values)
            : array;
        }

        /**
         * This method is like `_.pullAll` except that it accepts `iteratee` which is
         * invoked for each element of `array` and `values` to generate the criterion
         * by which they're compared. The iteratee is invoked with one argument: (value).
         *
         * **Note:** Unlike `_.differenceBy`, this method mutates `array`.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Array
         * @param {Array} array The array to modify.
         * @param {Array} values The values to remove.
         * @param {Function} [iteratee=_.identity] The iteratee invoked per element.
         * @returns {Array} Returns `array`.
         * @example
         *
         * var array = [{ 'x': 1 }, { 'x': 2 }, { 'x': 3 }, { 'x': 1 }];
         *
         * _.pullAllBy(array, [{ 'x': 1 }, { 'x': 3 }], 'x');
         * console.log(array);
         * // => [{ 'x': 2 }]
         */
        function pullAllBy(array, values, iteratee) {
          return (array && array.length && values && values.length)
            ? basePullAll(array, values, getIteratee(iteratee, 2))
            : array;
        }

        /**
         * This method is like `_.pullAll` except that it accepts `comparator` which
         * is invoked to compare elements of `array` to `values`. The comparator is
         * invoked with two arguments: (arrVal, othVal).
         *
         * **Note:** Unlike `_.differenceWith`, this method mutates `array`.
         *
         * @static
         * @memberOf _
         * @since 4.6.0
         * @category Array
         * @param {Array} array The array to modify.
         * @param {Array} values The values to remove.
         * @param {Function} [comparator] The comparator invoked per element.
         * @returns {Array} Returns `array`.
         * @example
         *
         * var array = [{ 'x': 1, 'y': 2 }, { 'x': 3, 'y': 4 }, { 'x': 5, 'y': 6 }];
         *
         * _.pullAllWith(array, [{ 'x': 3, 'y': 4 }], _.isEqual);
         * console.log(array);
         * // => [{ 'x': 1, 'y': 2 }, { 'x': 5, 'y': 6 }]
         */
        function pullAllWith(array, values, comparator) {
          return (array && array.length && values && values.length)
            ? basePullAll(array, values, undefined$1, comparator)
            : array;
        }

        /**
         * Removes elements from `array` corresponding to `indexes` and returns an
         * array of removed elements.
         *
         * **Note:** Unlike `_.at`, this method mutates `array`.
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category Array
         * @param {Array} array The array to modify.
         * @param {...(number|number[])} [indexes] The indexes of elements to remove.
         * @returns {Array} Returns the new array of removed elements.
         * @example
         *
         * var array = ['a', 'b', 'c', 'd'];
         * var pulled = _.pullAt(array, [1, 3]);
         *
         * console.log(array);
         * // => ['a', 'c']
         *
         * console.log(pulled);
         * // => ['b', 'd']
         */
        var pullAt = flatRest(function(array, indexes) {
          var length = array == null ? 0 : array.length,
              result = baseAt(array, indexes);

          basePullAt(array, arrayMap(indexes, function(index) {
            return isIndex(index, length) ? +index : index;
          }).sort(compareAscending));

          return result;
        });

        /**
         * Removes all elements from `array` that `predicate` returns truthy for
         * and returns an array of the removed elements. The predicate is invoked
         * with three arguments: (value, index, array).
         *
         * **Note:** Unlike `_.filter`, this method mutates `array`. Use `_.pull`
         * to pull elements from an array by value.
         *
         * @static
         * @memberOf _
         * @since 2.0.0
         * @category Array
         * @param {Array} array The array to modify.
         * @param {Function} [predicate=_.identity] The function invoked per iteration.
         * @returns {Array} Returns the new array of removed elements.
         * @example
         *
         * var array = [1, 2, 3, 4];
         * var evens = _.remove(array, function(n) {
         *   return n % 2 == 0;
         * });
         *
         * console.log(array);
         * // => [1, 3]
         *
         * console.log(evens);
         * // => [2, 4]
         */
        function remove(array, predicate) {
          var result = [];
          if (!(array && array.length)) {
            return result;
          }
          var index = -1,
              indexes = [],
              length = array.length;

          predicate = getIteratee(predicate, 3);
          while (++index < length) {
            var value = array[index];
            if (predicate(value, index, array)) {
              result.push(value);
              indexes.push(index);
            }
          }
          basePullAt(array, indexes);
          return result;
        }

        /**
         * Reverses `array` so that the first element becomes the last, the second
         * element becomes the second to last, and so on.
         *
         * **Note:** This method mutates `array` and is based on
         * [`Array#reverse`](https://mdn.io/Array/reverse).
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Array
         * @param {Array} array The array to modify.
         * @returns {Array} Returns `array`.
         * @example
         *
         * var array = [1, 2, 3];
         *
         * _.reverse(array);
         * // => [3, 2, 1]
         *
         * console.log(array);
         * // => [3, 2, 1]
         */
        function reverse(array) {
          return array == null ? array : nativeReverse.call(array);
        }

        /**
         * Creates a slice of `array` from `start` up to, but not including, `end`.
         *
         * **Note:** This method is used instead of
         * [`Array#slice`](https://mdn.io/Array/slice) to ensure dense arrays are
         * returned.
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category Array
         * @param {Array} array The array to slice.
         * @param {number} [start=0] The start position.
         * @param {number} [end=array.length] The end position.
         * @returns {Array} Returns the slice of `array`.
         */
        function slice(array, start, end) {
          var length = array == null ? 0 : array.length;
          if (!length) {
            return [];
          }
          if (end && typeof end != 'number' && isIterateeCall(array, start, end)) {
            start = 0;
            end = length;
          }
          else {
            start = start == null ? 0 : toInteger(start);
            end = end === undefined$1 ? length : toInteger(end);
          }
          return baseSlice(array, start, end);
        }

        /**
         * Uses a binary search to determine the lowest index at which `value`
         * should be inserted into `array` in order to maintain its sort order.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Array
         * @param {Array} array The sorted array to inspect.
         * @param {*} value The value to evaluate.
         * @returns {number} Returns the index at which `value` should be inserted
         *  into `array`.
         * @example
         *
         * _.sortedIndex([30, 50], 40);
         * // => 1
         */
        function sortedIndex(array, value) {
          return baseSortedIndex(array, value);
        }

        /**
         * This method is like `_.sortedIndex` except that it accepts `iteratee`
         * which is invoked for `value` and each element of `array` to compute their
         * sort ranking. The iteratee is invoked with one argument: (value).
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Array
         * @param {Array} array The sorted array to inspect.
         * @param {*} value The value to evaluate.
         * @param {Function} [iteratee=_.identity] The iteratee invoked per element.
         * @returns {number} Returns the index at which `value` should be inserted
         *  into `array`.
         * @example
         *
         * var objects = [{ 'x': 4 }, { 'x': 5 }];
         *
         * _.sortedIndexBy(objects, { 'x': 4 }, function(o) { return o.x; });
         * // => 0
         *
         * // The `_.property` iteratee shorthand.
         * _.sortedIndexBy(objects, { 'x': 4 }, 'x');
         * // => 0
         */
        function sortedIndexBy(array, value, iteratee) {
          return baseSortedIndexBy(array, value, getIteratee(iteratee, 2));
        }

        /**
         * This method is like `_.indexOf` except that it performs a binary
         * search on a sorted `array`.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Array
         * @param {Array} array The array to inspect.
         * @param {*} value The value to search for.
         * @returns {number} Returns the index of the matched value, else `-1`.
         * @example
         *
         * _.sortedIndexOf([4, 5, 5, 5, 6], 5);
         * // => 1
         */
        function sortedIndexOf(array, value) {
          var length = array == null ? 0 : array.length;
          if (length) {
            var index = baseSortedIndex(array, value);
            if (index < length && eq(array[index], value)) {
              return index;
            }
          }
          return -1;
        }

        /**
         * This method is like `_.sortedIndex` except that it returns the highest
         * index at which `value` should be inserted into `array` in order to
         * maintain its sort order.
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category Array
         * @param {Array} array The sorted array to inspect.
         * @param {*} value The value to evaluate.
         * @returns {number} Returns the index at which `value` should be inserted
         *  into `array`.
         * @example
         *
         * _.sortedLastIndex([4, 5, 5, 5, 6], 5);
         * // => 4
         */
        function sortedLastIndex(array, value) {
          return baseSortedIndex(array, value, true);
        }

        /**
         * This method is like `_.sortedLastIndex` except that it accepts `iteratee`
         * which is invoked for `value` and each element of `array` to compute their
         * sort ranking. The iteratee is invoked with one argument: (value).
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Array
         * @param {Array} array The sorted array to inspect.
         * @param {*} value The value to evaluate.
         * @param {Function} [iteratee=_.identity] The iteratee invoked per element.
         * @returns {number} Returns the index at which `value` should be inserted
         *  into `array`.
         * @example
         *
         * var objects = [{ 'x': 4 }, { 'x': 5 }];
         *
         * _.sortedLastIndexBy(objects, { 'x': 4 }, function(o) { return o.x; });
         * // => 1
         *
         * // The `_.property` iteratee shorthand.
         * _.sortedLastIndexBy(objects, { 'x': 4 }, 'x');
         * // => 1
         */
        function sortedLastIndexBy(array, value, iteratee) {
          return baseSortedIndexBy(array, value, getIteratee(iteratee, 2), true);
        }

        /**
         * This method is like `_.lastIndexOf` except that it performs a binary
         * search on a sorted `array`.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Array
         * @param {Array} array The array to inspect.
         * @param {*} value The value to search for.
         * @returns {number} Returns the index of the matched value, else `-1`.
         * @example
         *
         * _.sortedLastIndexOf([4, 5, 5, 5, 6], 5);
         * // => 3
         */
        function sortedLastIndexOf(array, value) {
          var length = array == null ? 0 : array.length;
          if (length) {
            var index = baseSortedIndex(array, value, true) - 1;
            if (eq(array[index], value)) {
              return index;
            }
          }
          return -1;
        }

        /**
         * This method is like `_.uniq` except that it's designed and optimized
         * for sorted arrays.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Array
         * @param {Array} array The array to inspect.
         * @returns {Array} Returns the new duplicate free array.
         * @example
         *
         * _.sortedUniq([1, 1, 2]);
         * // => [1, 2]
         */
        function sortedUniq(array) {
          return (array && array.length)
            ? baseSortedUniq(array)
            : [];
        }

        /**
         * This method is like `_.uniqBy` except that it's designed and optimized
         * for sorted arrays.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Array
         * @param {Array} array The array to inspect.
         * @param {Function} [iteratee] The iteratee invoked per element.
         * @returns {Array} Returns the new duplicate free array.
         * @example
         *
         * _.sortedUniqBy([1.1, 1.2, 2.3, 2.4], Math.floor);
         * // => [1.1, 2.3]
         */
        function sortedUniqBy(array, iteratee) {
          return (array && array.length)
            ? baseSortedUniq(array, getIteratee(iteratee, 2))
            : [];
        }

        /**
         * Gets all but the first element of `array`.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Array
         * @param {Array} array The array to query.
         * @returns {Array} Returns the slice of `array`.
         * @example
         *
         * _.tail([1, 2, 3]);
         * // => [2, 3]
         */
        function tail(array) {
          var length = array == null ? 0 : array.length;
          return length ? baseSlice(array, 1, length) : [];
        }

        /**
         * Creates a slice of `array` with `n` elements taken from the beginning.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Array
         * @param {Array} array The array to query.
         * @param {number} [n=1] The number of elements to take.
         * @param- {Object} [guard] Enables use as an iteratee for methods like `_.map`.
         * @returns {Array} Returns the slice of `array`.
         * @example
         *
         * _.take([1, 2, 3]);
         * // => [1]
         *
         * _.take([1, 2, 3], 2);
         * // => [1, 2]
         *
         * _.take([1, 2, 3], 5);
         * // => [1, 2, 3]
         *
         * _.take([1, 2, 3], 0);
         * // => []
         */
        function take(array, n, guard) {
          if (!(array && array.length)) {
            return [];
          }
          n = (guard || n === undefined$1) ? 1 : toInteger(n);
          return baseSlice(array, 0, n < 0 ? 0 : n);
        }

        /**
         * Creates a slice of `array` with `n` elements taken from the end.
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category Array
         * @param {Array} array The array to query.
         * @param {number} [n=1] The number of elements to take.
         * @param- {Object} [guard] Enables use as an iteratee for methods like `_.map`.
         * @returns {Array} Returns the slice of `array`.
         * @example
         *
         * _.takeRight([1, 2, 3]);
         * // => [3]
         *
         * _.takeRight([1, 2, 3], 2);
         * // => [2, 3]
         *
         * _.takeRight([1, 2, 3], 5);
         * // => [1, 2, 3]
         *
         * _.takeRight([1, 2, 3], 0);
         * // => []
         */
        function takeRight(array, n, guard) {
          var length = array == null ? 0 : array.length;
          if (!length) {
            return [];
          }
          n = (guard || n === undefined$1) ? 1 : toInteger(n);
          n = length - n;
          return baseSlice(array, n < 0 ? 0 : n, length);
        }

        /**
         * Creates a slice of `array` with elements taken from the end. Elements are
         * taken until `predicate` returns falsey. The predicate is invoked with
         * three arguments: (value, index, array).
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category Array
         * @param {Array} array The array to query.
         * @param {Function} [predicate=_.identity] The function invoked per iteration.
         * @returns {Array} Returns the slice of `array`.
         * @example
         *
         * var users = [
         *   { 'user': 'barney',  'active': true },
         *   { 'user': 'fred',    'active': false },
         *   { 'user': 'pebbles', 'active': false }
         * ];
         *
         * _.takeRightWhile(users, function(o) { return !o.active; });
         * // => objects for ['fred', 'pebbles']
         *
         * // The `_.matches` iteratee shorthand.
         * _.takeRightWhile(users, { 'user': 'pebbles', 'active': false });
         * // => objects for ['pebbles']
         *
         * // The `_.matchesProperty` iteratee shorthand.
         * _.takeRightWhile(users, ['active', false]);
         * // => objects for ['fred', 'pebbles']
         *
         * // The `_.property` iteratee shorthand.
         * _.takeRightWhile(users, 'active');
         * // => []
         */
        function takeRightWhile(array, predicate) {
          return (array && array.length)
            ? baseWhile(array, getIteratee(predicate, 3), false, true)
            : [];
        }

        /**
         * Creates a slice of `array` with elements taken from the beginning. Elements
         * are taken until `predicate` returns falsey. The predicate is invoked with
         * three arguments: (value, index, array).
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category Array
         * @param {Array} array The array to query.
         * @param {Function} [predicate=_.identity] The function invoked per iteration.
         * @returns {Array} Returns the slice of `array`.
         * @example
         *
         * var users = [
         *   { 'user': 'barney',  'active': false },
         *   { 'user': 'fred',    'active': false },
         *   { 'user': 'pebbles', 'active': true }
         * ];
         *
         * _.takeWhile(users, function(o) { return !o.active; });
         * // => objects for ['barney', 'fred']
         *
         * // The `_.matches` iteratee shorthand.
         * _.takeWhile(users, { 'user': 'barney', 'active': false });
         * // => objects for ['barney']
         *
         * // The `_.matchesProperty` iteratee shorthand.
         * _.takeWhile(users, ['active', false]);
         * // => objects for ['barney', 'fred']
         *
         * // The `_.property` iteratee shorthand.
         * _.takeWhile(users, 'active');
         * // => []
         */
        function takeWhile(array, predicate) {
          return (array && array.length)
            ? baseWhile(array, getIteratee(predicate, 3))
            : [];
        }

        /**
         * Creates an array of unique values, in order, from all given arrays using
         * [`SameValueZero`](http://ecma-international.org/ecma-262/7.0/#sec-samevaluezero)
         * for equality comparisons.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Array
         * @param {...Array} [arrays] The arrays to inspect.
         * @returns {Array} Returns the new array of combined values.
         * @example
         *
         * _.union([2], [1, 2]);
         * // => [2, 1]
         */
        var union = baseRest(function(arrays) {
          return baseUniq(baseFlatten(arrays, 1, isArrayLikeObject, true));
        });

        /**
         * This method is like `_.union` except that it accepts `iteratee` which is
         * invoked for each element of each `arrays` to generate the criterion by
         * which uniqueness is computed. Result values are chosen from the first
         * array in which the value occurs. The iteratee is invoked with one argument:
         * (value).
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Array
         * @param {...Array} [arrays] The arrays to inspect.
         * @param {Function} [iteratee=_.identity] The iteratee invoked per element.
         * @returns {Array} Returns the new array of combined values.
         * @example
         *
         * _.unionBy([2.1], [1.2, 2.3], Math.floor);
         * // => [2.1, 1.2]
         *
         * // The `_.property` iteratee shorthand.
         * _.unionBy([{ 'x': 1 }], [{ 'x': 2 }, { 'x': 1 }], 'x');
         * // => [{ 'x': 1 }, { 'x': 2 }]
         */
        var unionBy = baseRest(function(arrays) {
          var iteratee = last(arrays);
          if (isArrayLikeObject(iteratee)) {
            iteratee = undefined$1;
          }
          return baseUniq(baseFlatten(arrays, 1, isArrayLikeObject, true), getIteratee(iteratee, 2));
        });

        /**
         * This method is like `_.union` except that it accepts `comparator` which
         * is invoked to compare elements of `arrays`. Result values are chosen from
         * the first array in which the value occurs. The comparator is invoked
         * with two arguments: (arrVal, othVal).
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Array
         * @param {...Array} [arrays] The arrays to inspect.
         * @param {Function} [comparator] The comparator invoked per element.
         * @returns {Array} Returns the new array of combined values.
         * @example
         *
         * var objects = [{ 'x': 1, 'y': 2 }, { 'x': 2, 'y': 1 }];
         * var others = [{ 'x': 1, 'y': 1 }, { 'x': 1, 'y': 2 }];
         *
         * _.unionWith(objects, others, _.isEqual);
         * // => [{ 'x': 1, 'y': 2 }, { 'x': 2, 'y': 1 }, { 'x': 1, 'y': 1 }]
         */
        var unionWith = baseRest(function(arrays) {
          var comparator = last(arrays);
          comparator = typeof comparator == 'function' ? comparator : undefined$1;
          return baseUniq(baseFlatten(arrays, 1, isArrayLikeObject, true), undefined$1, comparator);
        });

        /**
         * Creates a duplicate-free version of an array, using
         * [`SameValueZero`](http://ecma-international.org/ecma-262/7.0/#sec-samevaluezero)
         * for equality comparisons, in which only the first occurrence of each element
         * is kept. The order of result values is determined by the order they occur
         * in the array.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Array
         * @param {Array} array The array to inspect.
         * @returns {Array} Returns the new duplicate free array.
         * @example
         *
         * _.uniq([2, 1, 2]);
         * // => [2, 1]
         */
        function uniq(array) {
          return (array && array.length) ? baseUniq(array) : [];
        }

        /**
         * This method is like `_.uniq` except that it accepts `iteratee` which is
         * invoked for each element in `array` to generate the criterion by which
         * uniqueness is computed. The order of result values is determined by the
         * order they occur in the array. The iteratee is invoked with one argument:
         * (value).
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Array
         * @param {Array} array The array to inspect.
         * @param {Function} [iteratee=_.identity] The iteratee invoked per element.
         * @returns {Array} Returns the new duplicate free array.
         * @example
         *
         * _.uniqBy([2.1, 1.2, 2.3], Math.floor);
         * // => [2.1, 1.2]
         *
         * // The `_.property` iteratee shorthand.
         * _.uniqBy([{ 'x': 1 }, { 'x': 2 }, { 'x': 1 }], 'x');
         * // => [{ 'x': 1 }, { 'x': 2 }]
         */
        function uniqBy(array, iteratee) {
          return (array && array.length) ? baseUniq(array, getIteratee(iteratee, 2)) : [];
        }

        /**
         * This method is like `_.uniq` except that it accepts `comparator` which
         * is invoked to compare elements of `array`. The order of result values is
         * determined by the order they occur in the array.The comparator is invoked
         * with two arguments: (arrVal, othVal).
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Array
         * @param {Array} array The array to inspect.
         * @param {Function} [comparator] The comparator invoked per element.
         * @returns {Array} Returns the new duplicate free array.
         * @example
         *
         * var objects = [{ 'x': 1, 'y': 2 }, { 'x': 2, 'y': 1 }, { 'x': 1, 'y': 2 }];
         *
         * _.uniqWith(objects, _.isEqual);
         * // => [{ 'x': 1, 'y': 2 }, { 'x': 2, 'y': 1 }]
         */
        function uniqWith(array, comparator) {
          comparator = typeof comparator == 'function' ? comparator : undefined$1;
          return (array && array.length) ? baseUniq(array, undefined$1, comparator) : [];
        }

        /**
         * This method is like `_.zip` except that it accepts an array of grouped
         * elements and creates an array regrouping the elements to their pre-zip
         * configuration.
         *
         * @static
         * @memberOf _
         * @since 1.2.0
         * @category Array
         * @param {Array} array The array of grouped elements to process.
         * @returns {Array} Returns the new array of regrouped elements.
         * @example
         *
         * var zipped = _.zip(['a', 'b'], [1, 2], [true, false]);
         * // => [['a', 1, true], ['b', 2, false]]
         *
         * _.unzip(zipped);
         * // => [['a', 'b'], [1, 2], [true, false]]
         */
        function unzip(array) {
          if (!(array && array.length)) {
            return [];
          }
          var length = 0;
          array = arrayFilter(array, function(group) {
            if (isArrayLikeObject(group)) {
              length = nativeMax(group.length, length);
              return true;
            }
          });
          return baseTimes(length, function(index) {
            return arrayMap(array, baseProperty(index));
          });
        }

        /**
         * This method is like `_.unzip` except that it accepts `iteratee` to specify
         * how regrouped values should be combined. The iteratee is invoked with the
         * elements of each group: (...group).
         *
         * @static
         * @memberOf _
         * @since 3.8.0
         * @category Array
         * @param {Array} array The array of grouped elements to process.
         * @param {Function} [iteratee=_.identity] The function to combine
         *  regrouped values.
         * @returns {Array} Returns the new array of regrouped elements.
         * @example
         *
         * var zipped = _.zip([1, 2], [10, 20], [100, 200]);
         * // => [[1, 10, 100], [2, 20, 200]]
         *
         * _.unzipWith(zipped, _.add);
         * // => [3, 30, 300]
         */
        function unzipWith(array, iteratee) {
          if (!(array && array.length)) {
            return [];
          }
          var result = unzip(array);
          if (iteratee == null) {
            return result;
          }
          return arrayMap(result, function(group) {
            return apply(iteratee, undefined$1, group);
          });
        }

        /**
         * Creates an array excluding all given values using
         * [`SameValueZero`](http://ecma-international.org/ecma-262/7.0/#sec-samevaluezero)
         * for equality comparisons.
         *
         * **Note:** Unlike `_.pull`, this method returns a new array.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Array
         * @param {Array} array The array to inspect.
         * @param {...*} [values] The values to exclude.
         * @returns {Array} Returns the new array of filtered values.
         * @see _.difference, _.xor
         * @example
         *
         * _.without([2, 1, 2, 3], 1, 2);
         * // => [3]
         */
        var without = baseRest(function(array, values) {
          return isArrayLikeObject(array)
            ? baseDifference(array, values)
            : [];
        });

        /**
         * Creates an array of unique values that is the
         * [symmetric difference](https://en.wikipedia.org/wiki/Symmetric_difference)
         * of the given arrays. The order of result values is determined by the order
         * they occur in the arrays.
         *
         * @static
         * @memberOf _
         * @since 2.4.0
         * @category Array
         * @param {...Array} [arrays] The arrays to inspect.
         * @returns {Array} Returns the new array of filtered values.
         * @see _.difference, _.without
         * @example
         *
         * _.xor([2, 1], [2, 3]);
         * // => [1, 3]
         */
        var xor = baseRest(function(arrays) {
          return baseXor(arrayFilter(arrays, isArrayLikeObject));
        });

        /**
         * This method is like `_.xor` except that it accepts `iteratee` which is
         * invoked for each element of each `arrays` to generate the criterion by
         * which by which they're compared. The order of result values is determined
         * by the order they occur in the arrays. The iteratee is invoked with one
         * argument: (value).
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Array
         * @param {...Array} [arrays] The arrays to inspect.
         * @param {Function} [iteratee=_.identity] The iteratee invoked per element.
         * @returns {Array} Returns the new array of filtered values.
         * @example
         *
         * _.xorBy([2.1, 1.2], [2.3, 3.4], Math.floor);
         * // => [1.2, 3.4]
         *
         * // The `_.property` iteratee shorthand.
         * _.xorBy([{ 'x': 1 }], [{ 'x': 2 }, { 'x': 1 }], 'x');
         * // => [{ 'x': 2 }]
         */
        var xorBy = baseRest(function(arrays) {
          var iteratee = last(arrays);
          if (isArrayLikeObject(iteratee)) {
            iteratee = undefined$1;
          }
          return baseXor(arrayFilter(arrays, isArrayLikeObject), getIteratee(iteratee, 2));
        });

        /**
         * This method is like `_.xor` except that it accepts `comparator` which is
         * invoked to compare elements of `arrays`. The order of result values is
         * determined by the order they occur in the arrays. The comparator is invoked
         * with two arguments: (arrVal, othVal).
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Array
         * @param {...Array} [arrays] The arrays to inspect.
         * @param {Function} [comparator] The comparator invoked per element.
         * @returns {Array} Returns the new array of filtered values.
         * @example
         *
         * var objects = [{ 'x': 1, 'y': 2 }, { 'x': 2, 'y': 1 }];
         * var others = [{ 'x': 1, 'y': 1 }, { 'x': 1, 'y': 2 }];
         *
         * _.xorWith(objects, others, _.isEqual);
         * // => [{ 'x': 2, 'y': 1 }, { 'x': 1, 'y': 1 }]
         */
        var xorWith = baseRest(function(arrays) {
          var comparator = last(arrays);
          comparator = typeof comparator == 'function' ? comparator : undefined$1;
          return baseXor(arrayFilter(arrays, isArrayLikeObject), undefined$1, comparator);
        });

        /**
         * Creates an array of grouped elements, the first of which contains the
         * first elements of the given arrays, the second of which contains the
         * second elements of the given arrays, and so on.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Array
         * @param {...Array} [arrays] The arrays to process.
         * @returns {Array} Returns the new array of grouped elements.
         * @example
         *
         * _.zip(['a', 'b'], [1, 2], [true, false]);
         * // => [['a', 1, true], ['b', 2, false]]
         */
        var zip = baseRest(unzip);

        /**
         * This method is like `_.fromPairs` except that it accepts two arrays,
         * one of property identifiers and one of corresponding values.
         *
         * @static
         * @memberOf _
         * @since 0.4.0
         * @category Array
         * @param {Array} [props=[]] The property identifiers.
         * @param {Array} [values=[]] The property values.
         * @returns {Object} Returns the new object.
         * @example
         *
         * _.zipObject(['a', 'b'], [1, 2]);
         * // => { 'a': 1, 'b': 2 }
         */
        function zipObject(props, values) {
          return baseZipObject(props || [], values || [], assignValue);
        }

        /**
         * This method is like `_.zipObject` except that it supports property paths.
         *
         * @static
         * @memberOf _
         * @since 4.1.0
         * @category Array
         * @param {Array} [props=[]] The property identifiers.
         * @param {Array} [values=[]] The property values.
         * @returns {Object} Returns the new object.
         * @example
         *
         * _.zipObjectDeep(['a.b[0].c', 'a.b[1].d'], [1, 2]);
         * // => { 'a': { 'b': [{ 'c': 1 }, { 'd': 2 }] } }
         */
        function zipObjectDeep(props, values) {
          return baseZipObject(props || [], values || [], baseSet);
        }

        /**
         * This method is like `_.zip` except that it accepts `iteratee` to specify
         * how grouped values should be combined. The iteratee is invoked with the
         * elements of each group: (...group).
         *
         * @static
         * @memberOf _
         * @since 3.8.0
         * @category Array
         * @param {...Array} [arrays] The arrays to process.
         * @param {Function} [iteratee=_.identity] The function to combine
         *  grouped values.
         * @returns {Array} Returns the new array of grouped elements.
         * @example
         *
         * _.zipWith([1, 2], [10, 20], [100, 200], function(a, b, c) {
         *   return a + b + c;
         * });
         * // => [111, 222]
         */
        var zipWith = baseRest(function(arrays) {
          var length = arrays.length,
              iteratee = length > 1 ? arrays[length - 1] : undefined$1;

          iteratee = typeof iteratee == 'function' ? (arrays.pop(), iteratee) : undefined$1;
          return unzipWith(arrays, iteratee);
        });

        /*------------------------------------------------------------------------*/

        /**
         * Creates a `lodash` wrapper instance that wraps `value` with explicit method
         * chain sequences enabled. The result of such sequences must be unwrapped
         * with `_#value`.
         *
         * @static
         * @memberOf _
         * @since 1.3.0
         * @category Seq
         * @param {*} value The value to wrap.
         * @returns {Object} Returns the new `lodash` wrapper instance.
         * @example
         *
         * var users = [
         *   { 'user': 'barney',  'age': 36 },
         *   { 'user': 'fred',    'age': 40 },
         *   { 'user': 'pebbles', 'age': 1 }
         * ];
         *
         * var youngest = _
         *   .chain(users)
         *   .sortBy('age')
         *   .map(function(o) {
         *     return o.user + ' is ' + o.age;
         *   })
         *   .head()
         *   .value();
         * // => 'pebbles is 1'
         */
        function chain(value) {
          var result = lodash(value);
          result.__chain__ = true;
          return result;
        }

        /**
         * This method invokes `interceptor` and returns `value`. The interceptor
         * is invoked with one argument; (value). The purpose of this method is to
         * "tap into" a method chain sequence in order to modify intermediate results.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Seq
         * @param {*} value The value to provide to `interceptor`.
         * @param {Function} interceptor The function to invoke.
         * @returns {*} Returns `value`.
         * @example
         *
         * _([1, 2, 3])
         *  .tap(function(array) {
         *    // Mutate input array.
         *    array.pop();
         *  })
         *  .reverse()
         *  .value();
         * // => [2, 1]
         */
        function tap(value, interceptor) {
          interceptor(value);
          return value;
        }

        /**
         * This method is like `_.tap` except that it returns the result of `interceptor`.
         * The purpose of this method is to "pass thru" values replacing intermediate
         * results in a method chain sequence.
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category Seq
         * @param {*} value The value to provide to `interceptor`.
         * @param {Function} interceptor The function to invoke.
         * @returns {*} Returns the result of `interceptor`.
         * @example
         *
         * _('  abc  ')
         *  .chain()
         *  .trim()
         *  .thru(function(value) {
         *    return [value];
         *  })
         *  .value();
         * // => ['abc']
         */
        function thru(value, interceptor) {
          return interceptor(value);
        }

        /**
         * This method is the wrapper version of `_.at`.
         *
         * @name at
         * @memberOf _
         * @since 1.0.0
         * @category Seq
         * @param {...(string|string[])} [paths] The property paths to pick.
         * @returns {Object} Returns the new `lodash` wrapper instance.
         * @example
         *
         * var object = { 'a': [{ 'b': { 'c': 3 } }, 4] };
         *
         * _(object).at(['a[0].b.c', 'a[1]']).value();
         * // => [3, 4]
         */
        var wrapperAt = flatRest(function(paths) {
          var length = paths.length,
              start = length ? paths[0] : 0,
              value = this.__wrapped__,
              interceptor = function(object) { return baseAt(object, paths); };

          if (length > 1 || this.__actions__.length ||
              !(value instanceof LazyWrapper) || !isIndex(start)) {
            return this.thru(interceptor);
          }
          value = value.slice(start, +start + (length ? 1 : 0));
          value.__actions__.push({
            'func': thru,
            'args': [interceptor],
            'thisArg': undefined$1
          });
          return new LodashWrapper(value, this.__chain__).thru(function(array) {
            if (length && !array.length) {
              array.push(undefined$1);
            }
            return array;
          });
        });

        /**
         * Creates a `lodash` wrapper instance with explicit method chain sequences enabled.
         *
         * @name chain
         * @memberOf _
         * @since 0.1.0
         * @category Seq
         * @returns {Object} Returns the new `lodash` wrapper instance.
         * @example
         *
         * var users = [
         *   { 'user': 'barney', 'age': 36 },
         *   { 'user': 'fred',   'age': 40 }
         * ];
         *
         * // A sequence without explicit chaining.
         * _(users).head();
         * // => { 'user': 'barney', 'age': 36 }
         *
         * // A sequence with explicit chaining.
         * _(users)
         *   .chain()
         *   .head()
         *   .pick('user')
         *   .value();
         * // => { 'user': 'barney' }
         */
        function wrapperChain() {
          return chain(this);
        }

        /**
         * Executes the chain sequence and returns the wrapped result.
         *
         * @name commit
         * @memberOf _
         * @since 3.2.0
         * @category Seq
         * @returns {Object} Returns the new `lodash` wrapper instance.
         * @example
         *
         * var array = [1, 2];
         * var wrapped = _(array).push(3);
         *
         * console.log(array);
         * // => [1, 2]
         *
         * wrapped = wrapped.commit();
         * console.log(array);
         * // => [1, 2, 3]
         *
         * wrapped.last();
         * // => 3
         *
         * console.log(array);
         * // => [1, 2, 3]
         */
        function wrapperCommit() {
          return new LodashWrapper(this.value(), this.__chain__);
        }

        /**
         * Gets the next value on a wrapped object following the
         * [iterator protocol](https://mdn.io/iteration_protocols#iterator).
         *
         * @name next
         * @memberOf _
         * @since 4.0.0
         * @category Seq
         * @returns {Object} Returns the next iterator value.
         * @example
         *
         * var wrapped = _([1, 2]);
         *
         * wrapped.next();
         * // => { 'done': false, 'value': 1 }
         *
         * wrapped.next();
         * // => { 'done': false, 'value': 2 }
         *
         * wrapped.next();
         * // => { 'done': true, 'value': undefined }
         */
        function wrapperNext() {
          if (this.__values__ === undefined$1) {
            this.__values__ = toArray(this.value());
          }
          var done = this.__index__ >= this.__values__.length,
              value = done ? undefined$1 : this.__values__[this.__index__++];

          return { 'done': done, 'value': value };
        }

        /**
         * Enables the wrapper to be iterable.
         *
         * @name Symbol.iterator
         * @memberOf _
         * @since 4.0.0
         * @category Seq
         * @returns {Object} Returns the wrapper object.
         * @example
         *
         * var wrapped = _([1, 2]);
         *
         * wrapped[Symbol.iterator]() === wrapped;
         * // => true
         *
         * Array.from(wrapped);
         * // => [1, 2]
         */
        function wrapperToIterator() {
          return this;
        }

        /**
         * Creates a clone of the chain sequence planting `value` as the wrapped value.
         *
         * @name plant
         * @memberOf _
         * @since 3.2.0
         * @category Seq
         * @param {*} value The value to plant.
         * @returns {Object} Returns the new `lodash` wrapper instance.
         * @example
         *
         * function square(n) {
         *   return n * n;
         * }
         *
         * var wrapped = _([1, 2]).map(square);
         * var other = wrapped.plant([3, 4]);
         *
         * other.value();
         * // => [9, 16]
         *
         * wrapped.value();
         * // => [1, 4]
         */
        function wrapperPlant(value) {
          var result,
              parent = this;

          while (parent instanceof baseLodash) {
            var clone = wrapperClone(parent);
            clone.__index__ = 0;
            clone.__values__ = undefined$1;
            if (result) {
              previous.__wrapped__ = clone;
            } else {
              result = clone;
            }
            var previous = clone;
            parent = parent.__wrapped__;
          }
          previous.__wrapped__ = value;
          return result;
        }

        /**
         * This method is the wrapper version of `_.reverse`.
         *
         * **Note:** This method mutates the wrapped array.
         *
         * @name reverse
         * @memberOf _
         * @since 0.1.0
         * @category Seq
         * @returns {Object} Returns the new `lodash` wrapper instance.
         * @example
         *
         * var array = [1, 2, 3];
         *
         * _(array).reverse().value()
         * // => [3, 2, 1]
         *
         * console.log(array);
         * // => [3, 2, 1]
         */
        function wrapperReverse() {
          var value = this.__wrapped__;
          if (value instanceof LazyWrapper) {
            var wrapped = value;
            if (this.__actions__.length) {
              wrapped = new LazyWrapper(this);
            }
            wrapped = wrapped.reverse();
            wrapped.__actions__.push({
              'func': thru,
              'args': [reverse],
              'thisArg': undefined$1
            });
            return new LodashWrapper(wrapped, this.__chain__);
          }
          return this.thru(reverse);
        }

        /**
         * Executes the chain sequence to resolve the unwrapped value.
         *
         * @name value
         * @memberOf _
         * @since 0.1.0
         * @alias toJSON, valueOf
         * @category Seq
         * @returns {*} Returns the resolved unwrapped value.
         * @example
         *
         * _([1, 2, 3]).value();
         * // => [1, 2, 3]
         */
        function wrapperValue() {
          return baseWrapperValue(this.__wrapped__, this.__actions__);
        }

        /*------------------------------------------------------------------------*/

        /**
         * Creates an object composed of keys generated from the results of running
         * each element of `collection` thru `iteratee`. The corresponding value of
         * each key is the number of times the key was returned by `iteratee`. The
         * iteratee is invoked with one argument: (value).
         *
         * @static
         * @memberOf _
         * @since 0.5.0
         * @category Collection
         * @param {Array|Object} collection The collection to iterate over.
         * @param {Function} [iteratee=_.identity] The iteratee to transform keys.
         * @returns {Object} Returns the composed aggregate object.
         * @example
         *
         * _.countBy([6.1, 4.2, 6.3], Math.floor);
         * // => { '4': 1, '6': 2 }
         *
         * // The `_.property` iteratee shorthand.
         * _.countBy(['one', 'two', 'three'], 'length');
         * // => { '3': 2, '5': 1 }
         */
        var countBy = createAggregator(function(result, value, key) {
          if (hasOwnProperty.call(result, key)) {
            ++result[key];
          } else {
            baseAssignValue(result, key, 1);
          }
        });

        /**
         * Checks if `predicate` returns truthy for **all** elements of `collection`.
         * Iteration is stopped once `predicate` returns falsey. The predicate is
         * invoked with three arguments: (value, index|key, collection).
         *
         * **Note:** This method returns `true` for
         * [empty collections](https://en.wikipedia.org/wiki/Empty_set) because
         * [everything is true](https://en.wikipedia.org/wiki/Vacuous_truth) of
         * elements of empty collections.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Collection
         * @param {Array|Object} collection The collection to iterate over.
         * @param {Function} [predicate=_.identity] The function invoked per iteration.
         * @param- {Object} [guard] Enables use as an iteratee for methods like `_.map`.
         * @returns {boolean} Returns `true` if all elements pass the predicate check,
         *  else `false`.
         * @example
         *
         * _.every([true, 1, null, 'yes'], Boolean);
         * // => false
         *
         * var users = [
         *   { 'user': 'barney', 'age': 36, 'active': false },
         *   { 'user': 'fred',   'age': 40, 'active': false }
         * ];
         *
         * // The `_.matches` iteratee shorthand.
         * _.every(users, { 'user': 'barney', 'active': false });
         * // => false
         *
         * // The `_.matchesProperty` iteratee shorthand.
         * _.every(users, ['active', false]);
         * // => true
         *
         * // The `_.property` iteratee shorthand.
         * _.every(users, 'active');
         * // => false
         */
        function every(collection, predicate, guard) {
          var func = isArray(collection) ? arrayEvery : baseEvery;
          if (guard && isIterateeCall(collection, predicate, guard)) {
            predicate = undefined$1;
          }
          return func(collection, getIteratee(predicate, 3));
        }

        /**
         * Iterates over elements of `collection`, returning an array of all elements
         * `predicate` returns truthy for. The predicate is invoked with three
         * arguments: (value, index|key, collection).
         *
         * **Note:** Unlike `_.remove`, this method returns a new array.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Collection
         * @param {Array|Object} collection The collection to iterate over.
         * @param {Function} [predicate=_.identity] The function invoked per iteration.
         * @returns {Array} Returns the new filtered array.
         * @see _.reject
         * @example
         *
         * var users = [
         *   { 'user': 'barney', 'age': 36, 'active': true },
         *   { 'user': 'fred',   'age': 40, 'active': false }
         * ];
         *
         * _.filter(users, function(o) { return !o.active; });
         * // => objects for ['fred']
         *
         * // The `_.matches` iteratee shorthand.
         * _.filter(users, { 'age': 36, 'active': true });
         * // => objects for ['barney']
         *
         * // The `_.matchesProperty` iteratee shorthand.
         * _.filter(users, ['active', false]);
         * // => objects for ['fred']
         *
         * // The `_.property` iteratee shorthand.
         * _.filter(users, 'active');
         * // => objects for ['barney']
         *
         * // Combining several predicates using `_.overEvery` or `_.overSome`.
         * _.filter(users, _.overSome([{ 'age': 36 }, ['age', 40]]));
         * // => objects for ['fred', 'barney']
         */
        function filter(collection, predicate) {
          var func = isArray(collection) ? arrayFilter : baseFilter;
          return func(collection, getIteratee(predicate, 3));
        }

        /**
         * Iterates over elements of `collection`, returning the first element
         * `predicate` returns truthy for. The predicate is invoked with three
         * arguments: (value, index|key, collection).
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Collection
         * @param {Array|Object} collection The collection to inspect.
         * @param {Function} [predicate=_.identity] The function invoked per iteration.
         * @param {number} [fromIndex=0] The index to search from.
         * @returns {*} Returns the matched element, else `undefined`.
         * @example
         *
         * var users = [
         *   { 'user': 'barney',  'age': 36, 'active': true },
         *   { 'user': 'fred',    'age': 40, 'active': false },
         *   { 'user': 'pebbles', 'age': 1,  'active': true }
         * ];
         *
         * _.find(users, function(o) { return o.age < 40; });
         * // => object for 'barney'
         *
         * // The `_.matches` iteratee shorthand.
         * _.find(users, { 'age': 1, 'active': true });
         * // => object for 'pebbles'
         *
         * // The `_.matchesProperty` iteratee shorthand.
         * _.find(users, ['active', false]);
         * // => object for 'fred'
         *
         * // The `_.property` iteratee shorthand.
         * _.find(users, 'active');
         * // => object for 'barney'
         */
        var find = createFind(findIndex);

        /**
         * This method is like `_.find` except that it iterates over elements of
         * `collection` from right to left.
         *
         * @static
         * @memberOf _
         * @since 2.0.0
         * @category Collection
         * @param {Array|Object} collection The collection to inspect.
         * @param {Function} [predicate=_.identity] The function invoked per iteration.
         * @param {number} [fromIndex=collection.length-1] The index to search from.
         * @returns {*} Returns the matched element, else `undefined`.
         * @example
         *
         * _.findLast([1, 2, 3, 4], function(n) {
         *   return n % 2 == 1;
         * });
         * // => 3
         */
        var findLast = createFind(findLastIndex);

        /**
         * Creates a flattened array of values by running each element in `collection`
         * thru `iteratee` and flattening the mapped results. The iteratee is invoked
         * with three arguments: (value, index|key, collection).
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Collection
         * @param {Array|Object} collection The collection to iterate over.
         * @param {Function} [iteratee=_.identity] The function invoked per iteration.
         * @returns {Array} Returns the new flattened array.
         * @example
         *
         * function duplicate(n) {
         *   return [n, n];
         * }
         *
         * _.flatMap([1, 2], duplicate);
         * // => [1, 1, 2, 2]
         */
        function flatMap(collection, iteratee) {
          return baseFlatten(map(collection, iteratee), 1);
        }

        /**
         * This method is like `_.flatMap` except that it recursively flattens the
         * mapped results.
         *
         * @static
         * @memberOf _
         * @since 4.7.0
         * @category Collection
         * @param {Array|Object} collection The collection to iterate over.
         * @param {Function} [iteratee=_.identity] The function invoked per iteration.
         * @returns {Array} Returns the new flattened array.
         * @example
         *
         * function duplicate(n) {
         *   return [[[n, n]]];
         * }
         *
         * _.flatMapDeep([1, 2], duplicate);
         * // => [1, 1, 2, 2]
         */
        function flatMapDeep(collection, iteratee) {
          return baseFlatten(map(collection, iteratee), INFINITY);
        }

        /**
         * This method is like `_.flatMap` except that it recursively flattens the
         * mapped results up to `depth` times.
         *
         * @static
         * @memberOf _
         * @since 4.7.0
         * @category Collection
         * @param {Array|Object} collection The collection to iterate over.
         * @param {Function} [iteratee=_.identity] The function invoked per iteration.
         * @param {number} [depth=1] The maximum recursion depth.
         * @returns {Array} Returns the new flattened array.
         * @example
         *
         * function duplicate(n) {
         *   return [[[n, n]]];
         * }
         *
         * _.flatMapDepth([1, 2], duplicate, 2);
         * // => [[1, 1], [2, 2]]
         */
        function flatMapDepth(collection, iteratee, depth) {
          depth = depth === undefined$1 ? 1 : toInteger(depth);
          return baseFlatten(map(collection, iteratee), depth);
        }

        /**
         * Iterates over elements of `collection` and invokes `iteratee` for each element.
         * The iteratee is invoked with three arguments: (value, index|key, collection).
         * Iteratee functions may exit iteration early by explicitly returning `false`.
         *
         * **Note:** As with other "Collections" methods, objects with a "length"
         * property are iterated like arrays. To avoid this behavior use `_.forIn`
         * or `_.forOwn` for object iteration.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @alias each
         * @category Collection
         * @param {Array|Object} collection The collection to iterate over.
         * @param {Function} [iteratee=_.identity] The function invoked per iteration.
         * @returns {Array|Object} Returns `collection`.
         * @see _.forEachRight
         * @example
         *
         * _.forEach([1, 2], function(value) {
         *   console.log(value);
         * });
         * // => Logs `1` then `2`.
         *
         * _.forEach({ 'a': 1, 'b': 2 }, function(value, key) {
         *   console.log(key);
         * });
         * // => Logs 'a' then 'b' (iteration order is not guaranteed).
         */
        function forEach(collection, iteratee) {
          var func = isArray(collection) ? arrayEach : baseEach;
          return func(collection, getIteratee(iteratee, 3));
        }

        /**
         * This method is like `_.forEach` except that it iterates over elements of
         * `collection` from right to left.
         *
         * @static
         * @memberOf _
         * @since 2.0.0
         * @alias eachRight
         * @category Collection
         * @param {Array|Object} collection The collection to iterate over.
         * @param {Function} [iteratee=_.identity] The function invoked per iteration.
         * @returns {Array|Object} Returns `collection`.
         * @see _.forEach
         * @example
         *
         * _.forEachRight([1, 2], function(value) {
         *   console.log(value);
         * });
         * // => Logs `2` then `1`.
         */
        function forEachRight(collection, iteratee) {
          var func = isArray(collection) ? arrayEachRight : baseEachRight;
          return func(collection, getIteratee(iteratee, 3));
        }

        /**
         * Creates an object composed of keys generated from the results of running
         * each element of `collection` thru `iteratee`. The order of grouped values
         * is determined by the order they occur in `collection`. The corresponding
         * value of each key is an array of elements responsible for generating the
         * key. The iteratee is invoked with one argument: (value).
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Collection
         * @param {Array|Object} collection The collection to iterate over.
         * @param {Function} [iteratee=_.identity] The iteratee to transform keys.
         * @returns {Object} Returns the composed aggregate object.
         * @example
         *
         * _.groupBy([6.1, 4.2, 6.3], Math.floor);
         * // => { '4': [4.2], '6': [6.1, 6.3] }
         *
         * // The `_.property` iteratee shorthand.
         * _.groupBy(['one', 'two', 'three'], 'length');
         * // => { '3': ['one', 'two'], '5': ['three'] }
         */
        var groupBy = createAggregator(function(result, value, key) {
          if (hasOwnProperty.call(result, key)) {
            result[key].push(value);
          } else {
            baseAssignValue(result, key, [value]);
          }
        });

        /**
         * Checks if `value` is in `collection`. If `collection` is a string, it's
         * checked for a substring of `value`, otherwise
         * [`SameValueZero`](http://ecma-international.org/ecma-262/7.0/#sec-samevaluezero)
         * is used for equality comparisons. If `fromIndex` is negative, it's used as
         * the offset from the end of `collection`.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Collection
         * @param {Array|Object|string} collection The collection to inspect.
         * @param {*} value The value to search for.
         * @param {number} [fromIndex=0] The index to search from.
         * @param- {Object} [guard] Enables use as an iteratee for methods like `_.reduce`.
         * @returns {boolean} Returns `true` if `value` is found, else `false`.
         * @example
         *
         * _.includes([1, 2, 3], 1);
         * // => true
         *
         * _.includes([1, 2, 3], 1, 2);
         * // => false
         *
         * _.includes({ 'a': 1, 'b': 2 }, 1);
         * // => true
         *
         * _.includes('abcd', 'bc');
         * // => true
         */
        function includes(collection, value, fromIndex, guard) {
          collection = isArrayLike(collection) ? collection : values(collection);
          fromIndex = (fromIndex && !guard) ? toInteger(fromIndex) : 0;

          var length = collection.length;
          if (fromIndex < 0) {
            fromIndex = nativeMax(length + fromIndex, 0);
          }
          return isString(collection)
            ? (fromIndex <= length && collection.indexOf(value, fromIndex) > -1)
            : (!!length && baseIndexOf(collection, value, fromIndex) > -1);
        }

        /**
         * Invokes the method at `path` of each element in `collection`, returning
         * an array of the results of each invoked method. Any additional arguments
         * are provided to each invoked method. If `path` is a function, it's invoked
         * for, and `this` bound to, each element in `collection`.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Collection
         * @param {Array|Object} collection The collection to iterate over.
         * @param {Array|Function|string} path The path of the method to invoke or
         *  the function invoked per iteration.
         * @param {...*} [args] The arguments to invoke each method with.
         * @returns {Array} Returns the array of results.
         * @example
         *
         * _.invokeMap([[5, 1, 7], [3, 2, 1]], 'sort');
         * // => [[1, 5, 7], [1, 2, 3]]
         *
         * _.invokeMap([123, 456], String.prototype.split, '');
         * // => [['1', '2', '3'], ['4', '5', '6']]
         */
        var invokeMap = baseRest(function(collection, path, args) {
          var index = -1,
              isFunc = typeof path == 'function',
              result = isArrayLike(collection) ? Array(collection.length) : [];

          baseEach(collection, function(value) {
            result[++index] = isFunc ? apply(path, value, args) : baseInvoke(value, path, args);
          });
          return result;
        });

        /**
         * Creates an object composed of keys generated from the results of running
         * each element of `collection` thru `iteratee`. The corresponding value of
         * each key is the last element responsible for generating the key. The
         * iteratee is invoked with one argument: (value).
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Collection
         * @param {Array|Object} collection The collection to iterate over.
         * @param {Function} [iteratee=_.identity] The iteratee to transform keys.
         * @returns {Object} Returns the composed aggregate object.
         * @example
         *
         * var array = [
         *   { 'dir': 'left', 'code': 97 },
         *   { 'dir': 'right', 'code': 100 }
         * ];
         *
         * _.keyBy(array, function(o) {
         *   return String.fromCharCode(o.code);
         * });
         * // => { 'a': { 'dir': 'left', 'code': 97 }, 'd': { 'dir': 'right', 'code': 100 } }
         *
         * _.keyBy(array, 'dir');
         * // => { 'left': { 'dir': 'left', 'code': 97 }, 'right': { 'dir': 'right', 'code': 100 } }
         */
        var keyBy = createAggregator(function(result, value, key) {
          baseAssignValue(result, key, value);
        });

        /**
         * Creates an array of values by running each element in `collection` thru
         * `iteratee`. The iteratee is invoked with three arguments:
         * (value, index|key, collection).
         *
         * Many lodash methods are guarded to work as iteratees for methods like
         * `_.every`, `_.filter`, `_.map`, `_.mapValues`, `_.reject`, and `_.some`.
         *
         * The guarded methods are:
         * `ary`, `chunk`, `curry`, `curryRight`, `drop`, `dropRight`, `every`,
         * `fill`, `invert`, `parseInt`, `random`, `range`, `rangeRight`, `repeat`,
         * `sampleSize`, `slice`, `some`, `sortBy`, `split`, `take`, `takeRight`,
         * `template`, `trim`, `trimEnd`, `trimStart`, and `words`
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Collection
         * @param {Array|Object} collection The collection to iterate over.
         * @param {Function} [iteratee=_.identity] The function invoked per iteration.
         * @returns {Array} Returns the new mapped array.
         * @example
         *
         * function square(n) {
         *   return n * n;
         * }
         *
         * _.map([4, 8], square);
         * // => [16, 64]
         *
         * _.map({ 'a': 4, 'b': 8 }, square);
         * // => [16, 64] (iteration order is not guaranteed)
         *
         * var users = [
         *   { 'user': 'barney' },
         *   { 'user': 'fred' }
         * ];
         *
         * // The `_.property` iteratee shorthand.
         * _.map(users, 'user');
         * // => ['barney', 'fred']
         */
        function map(collection, iteratee) {
          var func = isArray(collection) ? arrayMap : baseMap;
          return func(collection, getIteratee(iteratee, 3));
        }

        /**
         * This method is like `_.sortBy` except that it allows specifying the sort
         * orders of the iteratees to sort by. If `orders` is unspecified, all values
         * are sorted in ascending order. Otherwise, specify an order of "desc" for
         * descending or "asc" for ascending sort order of corresponding values.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Collection
         * @param {Array|Object} collection The collection to iterate over.
         * @param {Array[]|Function[]|Object[]|string[]} [iteratees=[_.identity]]
         *  The iteratees to sort by.
         * @param {string[]} [orders] The sort orders of `iteratees`.
         * @param- {Object} [guard] Enables use as an iteratee for methods like `_.reduce`.
         * @returns {Array} Returns the new sorted array.
         * @example
         *
         * var users = [
         *   { 'user': 'fred',   'age': 48 },
         *   { 'user': 'barney', 'age': 34 },
         *   { 'user': 'fred',   'age': 40 },
         *   { 'user': 'barney', 'age': 36 }
         * ];
         *
         * // Sort by `user` in ascending order and by `age` in descending order.
         * _.orderBy(users, ['user', 'age'], ['asc', 'desc']);
         * // => objects for [['barney', 36], ['barney', 34], ['fred', 48], ['fred', 40]]
         */
        function orderBy(collection, iteratees, orders, guard) {
          if (collection == null) {
            return [];
          }
          if (!isArray(iteratees)) {
            iteratees = iteratees == null ? [] : [iteratees];
          }
          orders = guard ? undefined$1 : orders;
          if (!isArray(orders)) {
            orders = orders == null ? [] : [orders];
          }
          return baseOrderBy(collection, iteratees, orders);
        }

        /**
         * Creates an array of elements split into two groups, the first of which
         * contains elements `predicate` returns truthy for, the second of which
         * contains elements `predicate` returns falsey for. The predicate is
         * invoked with one argument: (value).
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category Collection
         * @param {Array|Object} collection The collection to iterate over.
         * @param {Function} [predicate=_.identity] The function invoked per iteration.
         * @returns {Array} Returns the array of grouped elements.
         * @example
         *
         * var users = [
         *   { 'user': 'barney',  'age': 36, 'active': false },
         *   { 'user': 'fred',    'age': 40, 'active': true },
         *   { 'user': 'pebbles', 'age': 1,  'active': false }
         * ];
         *
         * _.partition(users, function(o) { return o.active; });
         * // => objects for [['fred'], ['barney', 'pebbles']]
         *
         * // The `_.matches` iteratee shorthand.
         * _.partition(users, { 'age': 1, 'active': false });
         * // => objects for [['pebbles'], ['barney', 'fred']]
         *
         * // The `_.matchesProperty` iteratee shorthand.
         * _.partition(users, ['active', false]);
         * // => objects for [['barney', 'pebbles'], ['fred']]
         *
         * // The `_.property` iteratee shorthand.
         * _.partition(users, 'active');
         * // => objects for [['fred'], ['barney', 'pebbles']]
         */
        var partition = createAggregator(function(result, value, key) {
          result[key ? 0 : 1].push(value);
        }, function() { return [[], []]; });

        /**
         * Reduces `collection` to a value which is the accumulated result of running
         * each element in `collection` thru `iteratee`, where each successive
         * invocation is supplied the return value of the previous. If `accumulator`
         * is not given, the first element of `collection` is used as the initial
         * value. The iteratee is invoked with four arguments:
         * (accumulator, value, index|key, collection).
         *
         * Many lodash methods are guarded to work as iteratees for methods like
         * `_.reduce`, `_.reduceRight`, and `_.transform`.
         *
         * The guarded methods are:
         * `assign`, `defaults`, `defaultsDeep`, `includes`, `merge`, `orderBy`,
         * and `sortBy`
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Collection
         * @param {Array|Object} collection The collection to iterate over.
         * @param {Function} [iteratee=_.identity] The function invoked per iteration.
         * @param {*} [accumulator] The initial value.
         * @returns {*} Returns the accumulated value.
         * @see _.reduceRight
         * @example
         *
         * _.reduce([1, 2], function(sum, n) {
         *   return sum + n;
         * }, 0);
         * // => 3
         *
         * _.reduce({ 'a': 1, 'b': 2, 'c': 1 }, function(result, value, key) {
         *   (result[value] || (result[value] = [])).push(key);
         *   return result;
         * }, {});
         * // => { '1': ['a', 'c'], '2': ['b'] } (iteration order is not guaranteed)
         */
        function reduce(collection, iteratee, accumulator) {
          var func = isArray(collection) ? arrayReduce : baseReduce,
              initAccum = arguments.length < 3;

          return func(collection, getIteratee(iteratee, 4), accumulator, initAccum, baseEach);
        }

        /**
         * This method is like `_.reduce` except that it iterates over elements of
         * `collection` from right to left.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Collection
         * @param {Array|Object} collection The collection to iterate over.
         * @param {Function} [iteratee=_.identity] The function invoked per iteration.
         * @param {*} [accumulator] The initial value.
         * @returns {*} Returns the accumulated value.
         * @see _.reduce
         * @example
         *
         * var array = [[0, 1], [2, 3], [4, 5]];
         *
         * _.reduceRight(array, function(flattened, other) {
         *   return flattened.concat(other);
         * }, []);
         * // => [4, 5, 2, 3, 0, 1]
         */
        function reduceRight(collection, iteratee, accumulator) {
          var func = isArray(collection) ? arrayReduceRight : baseReduce,
              initAccum = arguments.length < 3;

          return func(collection, getIteratee(iteratee, 4), accumulator, initAccum, baseEachRight);
        }

        /**
         * The opposite of `_.filter`; this method returns the elements of `collection`
         * that `predicate` does **not** return truthy for.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Collection
         * @param {Array|Object} collection The collection to iterate over.
         * @param {Function} [predicate=_.identity] The function invoked per iteration.
         * @returns {Array} Returns the new filtered array.
         * @see _.filter
         * @example
         *
         * var users = [
         *   { 'user': 'barney', 'age': 36, 'active': false },
         *   { 'user': 'fred',   'age': 40, 'active': true }
         * ];
         *
         * _.reject(users, function(o) { return !o.active; });
         * // => objects for ['fred']
         *
         * // The `_.matches` iteratee shorthand.
         * _.reject(users, { 'age': 40, 'active': true });
         * // => objects for ['barney']
         *
         * // The `_.matchesProperty` iteratee shorthand.
         * _.reject(users, ['active', false]);
         * // => objects for ['fred']
         *
         * // The `_.property` iteratee shorthand.
         * _.reject(users, 'active');
         * // => objects for ['barney']
         */
        function reject(collection, predicate) {
          var func = isArray(collection) ? arrayFilter : baseFilter;
          return func(collection, negate(getIteratee(predicate, 3)));
        }

        /**
         * Gets a random element from `collection`.
         *
         * @static
         * @memberOf _
         * @since 2.0.0
         * @category Collection
         * @param {Array|Object} collection The collection to sample.
         * @returns {*} Returns the random element.
         * @example
         *
         * _.sample([1, 2, 3, 4]);
         * // => 2
         */
        function sample(collection) {
          var func = isArray(collection) ? arraySample : baseSample;
          return func(collection);
        }

        /**
         * Gets `n` random elements at unique keys from `collection` up to the
         * size of `collection`.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Collection
         * @param {Array|Object} collection The collection to sample.
         * @param {number} [n=1] The number of elements to sample.
         * @param- {Object} [guard] Enables use as an iteratee for methods like `_.map`.
         * @returns {Array} Returns the random elements.
         * @example
         *
         * _.sampleSize([1, 2, 3], 2);
         * // => [3, 1]
         *
         * _.sampleSize([1, 2, 3], 4);
         * // => [2, 3, 1]
         */
        function sampleSize(collection, n, guard) {
          if ((guard ? isIterateeCall(collection, n, guard) : n === undefined$1)) {
            n = 1;
          } else {
            n = toInteger(n);
          }
          var func = isArray(collection) ? arraySampleSize : baseSampleSize;
          return func(collection, n);
        }

        /**
         * Creates an array of shuffled values, using a version of the
         * [Fisher-Yates shuffle](https://en.wikipedia.org/wiki/Fisher-Yates_shuffle).
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Collection
         * @param {Array|Object} collection The collection to shuffle.
         * @returns {Array} Returns the new shuffled array.
         * @example
         *
         * _.shuffle([1, 2, 3, 4]);
         * // => [4, 1, 3, 2]
         */
        function shuffle(collection) {
          var func = isArray(collection) ? arrayShuffle : baseShuffle;
          return func(collection);
        }

        /**
         * Gets the size of `collection` by returning its length for array-like
         * values or the number of own enumerable string keyed properties for objects.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Collection
         * @param {Array|Object|string} collection The collection to inspect.
         * @returns {number} Returns the collection size.
         * @example
         *
         * _.size([1, 2, 3]);
         * // => 3
         *
         * _.size({ 'a': 1, 'b': 2 });
         * // => 2
         *
         * _.size('pebbles');
         * // => 7
         */
        function size(collection) {
          if (collection == null) {
            return 0;
          }
          if (isArrayLike(collection)) {
            return isString(collection) ? stringSize(collection) : collection.length;
          }
          var tag = getTag(collection);
          if (tag == mapTag || tag == setTag) {
            return collection.size;
          }
          return baseKeys(collection).length;
        }

        /**
         * Checks if `predicate` returns truthy for **any** element of `collection`.
         * Iteration is stopped once `predicate` returns truthy. The predicate is
         * invoked with three arguments: (value, index|key, collection).
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Collection
         * @param {Array|Object} collection The collection to iterate over.
         * @param {Function} [predicate=_.identity] The function invoked per iteration.
         * @param- {Object} [guard] Enables use as an iteratee for methods like `_.map`.
         * @returns {boolean} Returns `true` if any element passes the predicate check,
         *  else `false`.
         * @example
         *
         * _.some([null, 0, 'yes', false], Boolean);
         * // => true
         *
         * var users = [
         *   { 'user': 'barney', 'active': true },
         *   { 'user': 'fred',   'active': false }
         * ];
         *
         * // The `_.matches` iteratee shorthand.
         * _.some(users, { 'user': 'barney', 'active': false });
         * // => false
         *
         * // The `_.matchesProperty` iteratee shorthand.
         * _.some(users, ['active', false]);
         * // => true
         *
         * // The `_.property` iteratee shorthand.
         * _.some(users, 'active');
         * // => true
         */
        function some(collection, predicate, guard) {
          var func = isArray(collection) ? arraySome : baseSome;
          if (guard && isIterateeCall(collection, predicate, guard)) {
            predicate = undefined$1;
          }
          return func(collection, getIteratee(predicate, 3));
        }

        /**
         * Creates an array of elements, sorted in ascending order by the results of
         * running each element in a collection thru each iteratee. This method
         * performs a stable sort, that is, it preserves the original sort order of
         * equal elements. The iteratees are invoked with one argument: (value).
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Collection
         * @param {Array|Object} collection The collection to iterate over.
         * @param {...(Function|Function[])} [iteratees=[_.identity]]
         *  The iteratees to sort by.
         * @returns {Array} Returns the new sorted array.
         * @example
         *
         * var users = [
         *   { 'user': 'fred',   'age': 48 },
         *   { 'user': 'barney', 'age': 36 },
         *   { 'user': 'fred',   'age': 30 },
         *   { 'user': 'barney', 'age': 34 }
         * ];
         *
         * _.sortBy(users, [function(o) { return o.user; }]);
         * // => objects for [['barney', 36], ['barney', 34], ['fred', 48], ['fred', 30]]
         *
         * _.sortBy(users, ['user', 'age']);
         * // => objects for [['barney', 34], ['barney', 36], ['fred', 30], ['fred', 48]]
         */
        var sortBy = baseRest(function(collection, iteratees) {
          if (collection == null) {
            return [];
          }
          var length = iteratees.length;
          if (length > 1 && isIterateeCall(collection, iteratees[0], iteratees[1])) {
            iteratees = [];
          } else if (length > 2 && isIterateeCall(iteratees[0], iteratees[1], iteratees[2])) {
            iteratees = [iteratees[0]];
          }
          return baseOrderBy(collection, baseFlatten(iteratees, 1), []);
        });

        /*------------------------------------------------------------------------*/

        /**
         * Gets the timestamp of the number of milliseconds that have elapsed since
         * the Unix epoch (1 January 1970 00:00:00 UTC).
         *
         * @static
         * @memberOf _
         * @since 2.4.0
         * @category Date
         * @returns {number} Returns the timestamp.
         * @example
         *
         * _.defer(function(stamp) {
         *   console.log(_.now() - stamp);
         * }, _.now());
         * // => Logs the number of milliseconds it took for the deferred invocation.
         */
        var now = ctxNow || function() {
          return root.Date.now();
        };

        /*------------------------------------------------------------------------*/

        /**
         * The opposite of `_.before`; this method creates a function that invokes
         * `func` once it's called `n` or more times.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Function
         * @param {number} n The number of calls before `func` is invoked.
         * @param {Function} func The function to restrict.
         * @returns {Function} Returns the new restricted function.
         * @example
         *
         * var saves = ['profile', 'settings'];
         *
         * var done = _.after(saves.length, function() {
         *   console.log('done saving!');
         * });
         *
         * _.forEach(saves, function(type) {
         *   asyncSave({ 'type': type, 'complete': done });
         * });
         * // => Logs 'done saving!' after the two async saves have completed.
         */
        function after(n, func) {
          if (typeof func != 'function') {
            throw new TypeError(FUNC_ERROR_TEXT);
          }
          n = toInteger(n);
          return function() {
            if (--n < 1) {
              return func.apply(this, arguments);
            }
          };
        }

        /**
         * Creates a function that invokes `func`, with up to `n` arguments,
         * ignoring any additional arguments.
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category Function
         * @param {Function} func The function to cap arguments for.
         * @param {number} [n=func.length] The arity cap.
         * @param- {Object} [guard] Enables use as an iteratee for methods like `_.map`.
         * @returns {Function} Returns the new capped function.
         * @example
         *
         * _.map(['6', '8', '10'], _.ary(parseInt, 1));
         * // => [6, 8, 10]
         */
        function ary(func, n, guard) {
          n = guard ? undefined$1 : n;
          n = (func && n == null) ? func.length : n;
          return createWrap(func, WRAP_ARY_FLAG, undefined$1, undefined$1, undefined$1, undefined$1, n);
        }

        /**
         * Creates a function that invokes `func`, with the `this` binding and arguments
         * of the created function, while it's called less than `n` times. Subsequent
         * calls to the created function return the result of the last `func` invocation.
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category Function
         * @param {number} n The number of calls at which `func` is no longer invoked.
         * @param {Function} func The function to restrict.
         * @returns {Function} Returns the new restricted function.
         * @example
         *
         * jQuery(element).on('click', _.before(5, addContactToList));
         * // => Allows adding up to 4 contacts to the list.
         */
        function before(n, func) {
          var result;
          if (typeof func != 'function') {
            throw new TypeError(FUNC_ERROR_TEXT);
          }
          n = toInteger(n);
          return function() {
            if (--n > 0) {
              result = func.apply(this, arguments);
            }
            if (n <= 1) {
              func = undefined$1;
            }
            return result;
          };
        }

        /**
         * Creates a function that invokes `func` with the `this` binding of `thisArg`
         * and `partials` prepended to the arguments it receives.
         *
         * The `_.bind.placeholder` value, which defaults to `_` in monolithic builds,
         * may be used as a placeholder for partially applied arguments.
         *
         * **Note:** Unlike native `Function#bind`, this method doesn't set the "length"
         * property of bound functions.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Function
         * @param {Function} func The function to bind.
         * @param {*} thisArg The `this` binding of `func`.
         * @param {...*} [partials] The arguments to be partially applied.
         * @returns {Function} Returns the new bound function.
         * @example
         *
         * function greet(greeting, punctuation) {
         *   return greeting + ' ' + this.user + punctuation;
         * }
         *
         * var object = { 'user': 'fred' };
         *
         * var bound = _.bind(greet, object, 'hi');
         * bound('!');
         * // => 'hi fred!'
         *
         * // Bound with placeholders.
         * var bound = _.bind(greet, object, _, '!');
         * bound('hi');
         * // => 'hi fred!'
         */
        var bind = baseRest(function(func, thisArg, partials) {
          var bitmask = WRAP_BIND_FLAG;
          if (partials.length) {
            var holders = replaceHolders(partials, getHolder(bind));
            bitmask |= WRAP_PARTIAL_FLAG;
          }
          return createWrap(func, bitmask, thisArg, partials, holders);
        });

        /**
         * Creates a function that invokes the method at `object[key]` with `partials`
         * prepended to the arguments it receives.
         *
         * This method differs from `_.bind` by allowing bound functions to reference
         * methods that may be redefined or don't yet exist. See
         * [Peter Michaux's article](http://peter.michaux.ca/articles/lazy-function-definition-pattern)
         * for more details.
         *
         * The `_.bindKey.placeholder` value, which defaults to `_` in monolithic
         * builds, may be used as a placeholder for partially applied arguments.
         *
         * @static
         * @memberOf _
         * @since 0.10.0
         * @category Function
         * @param {Object} object The object to invoke the method on.
         * @param {string} key The key of the method.
         * @param {...*} [partials] The arguments to be partially applied.
         * @returns {Function} Returns the new bound function.
         * @example
         *
         * var object = {
         *   'user': 'fred',
         *   'greet': function(greeting, punctuation) {
         *     return greeting + ' ' + this.user + punctuation;
         *   }
         * };
         *
         * var bound = _.bindKey(object, 'greet', 'hi');
         * bound('!');
         * // => 'hi fred!'
         *
         * object.greet = function(greeting, punctuation) {
         *   return greeting + 'ya ' + this.user + punctuation;
         * };
         *
         * bound('!');
         * // => 'hiya fred!'
         *
         * // Bound with placeholders.
         * var bound = _.bindKey(object, 'greet', _, '!');
         * bound('hi');
         * // => 'hiya fred!'
         */
        var bindKey = baseRest(function(object, key, partials) {
          var bitmask = WRAP_BIND_FLAG | WRAP_BIND_KEY_FLAG;
          if (partials.length) {
            var holders = replaceHolders(partials, getHolder(bindKey));
            bitmask |= WRAP_PARTIAL_FLAG;
          }
          return createWrap(key, bitmask, object, partials, holders);
        });

        /**
         * Creates a function that accepts arguments of `func` and either invokes
         * `func` returning its result, if at least `arity` number of arguments have
         * been provided, or returns a function that accepts the remaining `func`
         * arguments, and so on. The arity of `func` may be specified if `func.length`
         * is not sufficient.
         *
         * The `_.curry.placeholder` value, which defaults to `_` in monolithic builds,
         * may be used as a placeholder for provided arguments.
         *
         * **Note:** This method doesn't set the "length" property of curried functions.
         *
         * @static
         * @memberOf _
         * @since 2.0.0
         * @category Function
         * @param {Function} func The function to curry.
         * @param {number} [arity=func.length] The arity of `func`.
         * @param- {Object} [guard] Enables use as an iteratee for methods like `_.map`.
         * @returns {Function} Returns the new curried function.
         * @example
         *
         * var abc = function(a, b, c) {
         *   return [a, b, c];
         * };
         *
         * var curried = _.curry(abc);
         *
         * curried(1)(2)(3);
         * // => [1, 2, 3]
         *
         * curried(1, 2)(3);
         * // => [1, 2, 3]
         *
         * curried(1, 2, 3);
         * // => [1, 2, 3]
         *
         * // Curried with placeholders.
         * curried(1)(_, 3)(2);
         * // => [1, 2, 3]
         */
        function curry(func, arity, guard) {
          arity = guard ? undefined$1 : arity;
          var result = createWrap(func, WRAP_CURRY_FLAG, undefined$1, undefined$1, undefined$1, undefined$1, undefined$1, arity);
          result.placeholder = curry.placeholder;
          return result;
        }

        /**
         * This method is like `_.curry` except that arguments are applied to `func`
         * in the manner of `_.partialRight` instead of `_.partial`.
         *
         * The `_.curryRight.placeholder` value, which defaults to `_` in monolithic
         * builds, may be used as a placeholder for provided arguments.
         *
         * **Note:** This method doesn't set the "length" property of curried functions.
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category Function
         * @param {Function} func The function to curry.
         * @param {number} [arity=func.length] The arity of `func`.
         * @param- {Object} [guard] Enables use as an iteratee for methods like `_.map`.
         * @returns {Function} Returns the new curried function.
         * @example
         *
         * var abc = function(a, b, c) {
         *   return [a, b, c];
         * };
         *
         * var curried = _.curryRight(abc);
         *
         * curried(3)(2)(1);
         * // => [1, 2, 3]
         *
         * curried(2, 3)(1);
         * // => [1, 2, 3]
         *
         * curried(1, 2, 3);
         * // => [1, 2, 3]
         *
         * // Curried with placeholders.
         * curried(3)(1, _)(2);
         * // => [1, 2, 3]
         */
        function curryRight(func, arity, guard) {
          arity = guard ? undefined$1 : arity;
          var result = createWrap(func, WRAP_CURRY_RIGHT_FLAG, undefined$1, undefined$1, undefined$1, undefined$1, undefined$1, arity);
          result.placeholder = curryRight.placeholder;
          return result;
        }

        /**
         * Creates a debounced function that delays invoking `func` until after `wait`
         * milliseconds have elapsed since the last time the debounced function was
         * invoked. The debounced function comes with a `cancel` method to cancel
         * delayed `func` invocations and a `flush` method to immediately invoke them.
         * Provide `options` to indicate whether `func` should be invoked on the
         * leading and/or trailing edge of the `wait` timeout. The `func` is invoked
         * with the last arguments provided to the debounced function. Subsequent
         * calls to the debounced function return the result of the last `func`
         * invocation.
         *
         * **Note:** If `leading` and `trailing` options are `true`, `func` is
         * invoked on the trailing edge of the timeout only if the debounced function
         * is invoked more than once during the `wait` timeout.
         *
         * If `wait` is `0` and `leading` is `false`, `func` invocation is deferred
         * until to the next tick, similar to `setTimeout` with a timeout of `0`.
         *
         * See [David Corbacho's article](https://css-tricks.com/debouncing-throttling-explained-examples/)
         * for details over the differences between `_.debounce` and `_.throttle`.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Function
         * @param {Function} func The function to debounce.
         * @param {number} [wait=0] The number of milliseconds to delay.
         * @param {Object} [options={}] The options object.
         * @param {boolean} [options.leading=false]
         *  Specify invoking on the leading edge of the timeout.
         * @param {number} [options.maxWait]
         *  The maximum time `func` is allowed to be delayed before it's invoked.
         * @param {boolean} [options.trailing=true]
         *  Specify invoking on the trailing edge of the timeout.
         * @returns {Function} Returns the new debounced function.
         * @example
         *
         * // Avoid costly calculations while the window size is in flux.
         * jQuery(window).on('resize', _.debounce(calculateLayout, 150));
         *
         * // Invoke `sendMail` when clicked, debouncing subsequent calls.
         * jQuery(element).on('click', _.debounce(sendMail, 300, {
         *   'leading': true,
         *   'trailing': false
         * }));
         *
         * // Ensure `batchLog` is invoked once after 1 second of debounced calls.
         * var debounced = _.debounce(batchLog, 250, { 'maxWait': 1000 });
         * var source = new EventSource('/stream');
         * jQuery(source).on('message', debounced);
         *
         * // Cancel the trailing debounced invocation.
         * jQuery(window).on('popstate', debounced.cancel);
         */
        function debounce(func, wait, options) {
          var lastArgs,
              lastThis,
              maxWait,
              result,
              timerId,
              lastCallTime,
              lastInvokeTime = 0,
              leading = false,
              maxing = false,
              trailing = true;

          if (typeof func != 'function') {
            throw new TypeError(FUNC_ERROR_TEXT);
          }
          wait = toNumber(wait) || 0;
          if (isObject(options)) {
            leading = !!options.leading;
            maxing = 'maxWait' in options;
            maxWait = maxing ? nativeMax(toNumber(options.maxWait) || 0, wait) : maxWait;
            trailing = 'trailing' in options ? !!options.trailing : trailing;
          }

          function invokeFunc(time) {
            var args = lastArgs,
                thisArg = lastThis;

            lastArgs = lastThis = undefined$1;
            lastInvokeTime = time;
            result = func.apply(thisArg, args);
            return result;
          }

          function leadingEdge(time) {
            // Reset any `maxWait` timer.
            lastInvokeTime = time;
            // Start the timer for the trailing edge.
            timerId = setTimeout(timerExpired, wait);
            // Invoke the leading edge.
            return leading ? invokeFunc(time) : result;
          }

          function remainingWait(time) {
            var timeSinceLastCall = time - lastCallTime,
                timeSinceLastInvoke = time - lastInvokeTime,
                timeWaiting = wait - timeSinceLastCall;

            return maxing
              ? nativeMin(timeWaiting, maxWait - timeSinceLastInvoke)
              : timeWaiting;
          }

          function shouldInvoke(time) {
            var timeSinceLastCall = time - lastCallTime,
                timeSinceLastInvoke = time - lastInvokeTime;

            // Either this is the first call, activity has stopped and we're at the
            // trailing edge, the system time has gone backwards and we're treating
            // it as the trailing edge, or we've hit the `maxWait` limit.
            return (lastCallTime === undefined$1 || (timeSinceLastCall >= wait) ||
              (timeSinceLastCall < 0) || (maxing && timeSinceLastInvoke >= maxWait));
          }

          function timerExpired() {
            var time = now();
            if (shouldInvoke(time)) {
              return trailingEdge(time);
            }
            // Restart the timer.
            timerId = setTimeout(timerExpired, remainingWait(time));
          }

          function trailingEdge(time) {
            timerId = undefined$1;

            // Only invoke if we have `lastArgs` which means `func` has been
            // debounced at least once.
            if (trailing && lastArgs) {
              return invokeFunc(time);
            }
            lastArgs = lastThis = undefined$1;
            return result;
          }

          function cancel() {
            if (timerId !== undefined$1) {
              clearTimeout(timerId);
            }
            lastInvokeTime = 0;
            lastArgs = lastCallTime = lastThis = timerId = undefined$1;
          }

          function flush() {
            return timerId === undefined$1 ? result : trailingEdge(now());
          }

          function debounced() {
            var time = now(),
                isInvoking = shouldInvoke(time);

            lastArgs = arguments;
            lastThis = this;
            lastCallTime = time;

            if (isInvoking) {
              if (timerId === undefined$1) {
                return leadingEdge(lastCallTime);
              }
              if (maxing) {
                // Handle invocations in a tight loop.
                clearTimeout(timerId);
                timerId = setTimeout(timerExpired, wait);
                return invokeFunc(lastCallTime);
              }
            }
            if (timerId === undefined$1) {
              timerId = setTimeout(timerExpired, wait);
            }
            return result;
          }
          debounced.cancel = cancel;
          debounced.flush = flush;
          return debounced;
        }

        /**
         * Defers invoking the `func` until the current call stack has cleared. Any
         * additional arguments are provided to `func` when it's invoked.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Function
         * @param {Function} func The function to defer.
         * @param {...*} [args] The arguments to invoke `func` with.
         * @returns {number} Returns the timer id.
         * @example
         *
         * _.defer(function(text) {
         *   console.log(text);
         * }, 'deferred');
         * // => Logs 'deferred' after one millisecond.
         */
        var defer = baseRest(function(func, args) {
          return baseDelay(func, 1, args);
        });

        /**
         * Invokes `func` after `wait` milliseconds. Any additional arguments are
         * provided to `func` when it's invoked.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Function
         * @param {Function} func The function to delay.
         * @param {number} wait The number of milliseconds to delay invocation.
         * @param {...*} [args] The arguments to invoke `func` with.
         * @returns {number} Returns the timer id.
         * @example
         *
         * _.delay(function(text) {
         *   console.log(text);
         * }, 1000, 'later');
         * // => Logs 'later' after one second.
         */
        var delay = baseRest(function(func, wait, args) {
          return baseDelay(func, toNumber(wait) || 0, args);
        });

        /**
         * Creates a function that invokes `func` with arguments reversed.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Function
         * @param {Function} func The function to flip arguments for.
         * @returns {Function} Returns the new flipped function.
         * @example
         *
         * var flipped = _.flip(function() {
         *   return _.toArray(arguments);
         * });
         *
         * flipped('a', 'b', 'c', 'd');
         * // => ['d', 'c', 'b', 'a']
         */
        function flip(func) {
          return createWrap(func, WRAP_FLIP_FLAG);
        }

        /**
         * Creates a function that memoizes the result of `func`. If `resolver` is
         * provided, it determines the cache key for storing the result based on the
         * arguments provided to the memoized function. By default, the first argument
         * provided to the memoized function is used as the map cache key. The `func`
         * is invoked with the `this` binding of the memoized function.
         *
         * **Note:** The cache is exposed as the `cache` property on the memoized
         * function. Its creation may be customized by replacing the `_.memoize.Cache`
         * constructor with one whose instances implement the
         * [`Map`](http://ecma-international.org/ecma-262/7.0/#sec-properties-of-the-map-prototype-object)
         * method interface of `clear`, `delete`, `get`, `has`, and `set`.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Function
         * @param {Function} func The function to have its output memoized.
         * @param {Function} [resolver] The function to resolve the cache key.
         * @returns {Function} Returns the new memoized function.
         * @example
         *
         * var object = { 'a': 1, 'b': 2 };
         * var other = { 'c': 3, 'd': 4 };
         *
         * var values = _.memoize(_.values);
         * values(object);
         * // => [1, 2]
         *
         * values(other);
         * // => [3, 4]
         *
         * object.a = 2;
         * values(object);
         * // => [1, 2]
         *
         * // Modify the result cache.
         * values.cache.set(object, ['a', 'b']);
         * values(object);
         * // => ['a', 'b']
         *
         * // Replace `_.memoize.Cache`.
         * _.memoize.Cache = WeakMap;
         */
        function memoize(func, resolver) {
          if (typeof func != 'function' || (resolver != null && typeof resolver != 'function')) {
            throw new TypeError(FUNC_ERROR_TEXT);
          }
          var memoized = function() {
            var args = arguments,
                key = resolver ? resolver.apply(this, args) : args[0],
                cache = memoized.cache;

            if (cache.has(key)) {
              return cache.get(key);
            }
            var result = func.apply(this, args);
            memoized.cache = cache.set(key, result) || cache;
            return result;
          };
          memoized.cache = new (memoize.Cache || MapCache);
          return memoized;
        }

        // Expose `MapCache`.
        memoize.Cache = MapCache;

        /**
         * Creates a function that negates the result of the predicate `func`. The
         * `func` predicate is invoked with the `this` binding and arguments of the
         * created function.
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category Function
         * @param {Function} predicate The predicate to negate.
         * @returns {Function} Returns the new negated function.
         * @example
         *
         * function isEven(n) {
         *   return n % 2 == 0;
         * }
         *
         * _.filter([1, 2, 3, 4, 5, 6], _.negate(isEven));
         * // => [1, 3, 5]
         */
        function negate(predicate) {
          if (typeof predicate != 'function') {
            throw new TypeError(FUNC_ERROR_TEXT);
          }
          return function() {
            var args = arguments;
            switch (args.length) {
              case 0: return !predicate.call(this);
              case 1: return !predicate.call(this, args[0]);
              case 2: return !predicate.call(this, args[0], args[1]);
              case 3: return !predicate.call(this, args[0], args[1], args[2]);
            }
            return !predicate.apply(this, args);
          };
        }

        /**
         * Creates a function that is restricted to invoking `func` once. Repeat calls
         * to the function return the value of the first invocation. The `func` is
         * invoked with the `this` binding and arguments of the created function.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Function
         * @param {Function} func The function to restrict.
         * @returns {Function} Returns the new restricted function.
         * @example
         *
         * var initialize = _.once(createApplication);
         * initialize();
         * initialize();
         * // => `createApplication` is invoked once
         */
        function once(func) {
          return before(2, func);
        }

        /**
         * Creates a function that invokes `func` with its arguments transformed.
         *
         * @static
         * @since 4.0.0
         * @memberOf _
         * @category Function
         * @param {Function} func The function to wrap.
         * @param {...(Function|Function[])} [transforms=[_.identity]]
         *  The argument transforms.
         * @returns {Function} Returns the new function.
         * @example
         *
         * function doubled(n) {
         *   return n * 2;
         * }
         *
         * function square(n) {
         *   return n * n;
         * }
         *
         * var func = _.overArgs(function(x, y) {
         *   return [x, y];
         * }, [square, doubled]);
         *
         * func(9, 3);
         * // => [81, 6]
         *
         * func(10, 5);
         * // => [100, 10]
         */
        var overArgs = castRest(function(func, transforms) {
          transforms = (transforms.length == 1 && isArray(transforms[0]))
            ? arrayMap(transforms[0], baseUnary(getIteratee()))
            : arrayMap(baseFlatten(transforms, 1), baseUnary(getIteratee()));

          var funcsLength = transforms.length;
          return baseRest(function(args) {
            var index = -1,
                length = nativeMin(args.length, funcsLength);

            while (++index < length) {
              args[index] = transforms[index].call(this, args[index]);
            }
            return apply(func, this, args);
          });
        });

        /**
         * Creates a function that invokes `func` with `partials` prepended to the
         * arguments it receives. This method is like `_.bind` except it does **not**
         * alter the `this` binding.
         *
         * The `_.partial.placeholder` value, which defaults to `_` in monolithic
         * builds, may be used as a placeholder for partially applied arguments.
         *
         * **Note:** This method doesn't set the "length" property of partially
         * applied functions.
         *
         * @static
         * @memberOf _
         * @since 0.2.0
         * @category Function
         * @param {Function} func The function to partially apply arguments to.
         * @param {...*} [partials] The arguments to be partially applied.
         * @returns {Function} Returns the new partially applied function.
         * @example
         *
         * function greet(greeting, name) {
         *   return greeting + ' ' + name;
         * }
         *
         * var sayHelloTo = _.partial(greet, 'hello');
         * sayHelloTo('fred');
         * // => 'hello fred'
         *
         * // Partially applied with placeholders.
         * var greetFred = _.partial(greet, _, 'fred');
         * greetFred('hi');
         * // => 'hi fred'
         */
        var partial = baseRest(function(func, partials) {
          var holders = replaceHolders(partials, getHolder(partial));
          return createWrap(func, WRAP_PARTIAL_FLAG, undefined$1, partials, holders);
        });

        /**
         * This method is like `_.partial` except that partially applied arguments
         * are appended to the arguments it receives.
         *
         * The `_.partialRight.placeholder` value, which defaults to `_` in monolithic
         * builds, may be used as a placeholder for partially applied arguments.
         *
         * **Note:** This method doesn't set the "length" property of partially
         * applied functions.
         *
         * @static
         * @memberOf _
         * @since 1.0.0
         * @category Function
         * @param {Function} func The function to partially apply arguments to.
         * @param {...*} [partials] The arguments to be partially applied.
         * @returns {Function} Returns the new partially applied function.
         * @example
         *
         * function greet(greeting, name) {
         *   return greeting + ' ' + name;
         * }
         *
         * var greetFred = _.partialRight(greet, 'fred');
         * greetFred('hi');
         * // => 'hi fred'
         *
         * // Partially applied with placeholders.
         * var sayHelloTo = _.partialRight(greet, 'hello', _);
         * sayHelloTo('fred');
         * // => 'hello fred'
         */
        var partialRight = baseRest(function(func, partials) {
          var holders = replaceHolders(partials, getHolder(partialRight));
          return createWrap(func, WRAP_PARTIAL_RIGHT_FLAG, undefined$1, partials, holders);
        });

        /**
         * Creates a function that invokes `func` with arguments arranged according
         * to the specified `indexes` where the argument value at the first index is
         * provided as the first argument, the argument value at the second index is
         * provided as the second argument, and so on.
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category Function
         * @param {Function} func The function to rearrange arguments for.
         * @param {...(number|number[])} indexes The arranged argument indexes.
         * @returns {Function} Returns the new function.
         * @example
         *
         * var rearged = _.rearg(function(a, b, c) {
         *   return [a, b, c];
         * }, [2, 0, 1]);
         *
         * rearged('b', 'c', 'a')
         * // => ['a', 'b', 'c']
         */
        var rearg = flatRest(function(func, indexes) {
          return createWrap(func, WRAP_REARG_FLAG, undefined$1, undefined$1, undefined$1, indexes);
        });

        /**
         * Creates a function that invokes `func` with the `this` binding of the
         * created function and arguments from `start` and beyond provided as
         * an array.
         *
         * **Note:** This method is based on the
         * [rest parameter](https://mdn.io/rest_parameters).
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Function
         * @param {Function} func The function to apply a rest parameter to.
         * @param {number} [start=func.length-1] The start position of the rest parameter.
         * @returns {Function} Returns the new function.
         * @example
         *
         * var say = _.rest(function(what, names) {
         *   return what + ' ' + _.initial(names).join(', ') +
         *     (_.size(names) > 1 ? ', & ' : '') + _.last(names);
         * });
         *
         * say('hello', 'fred', 'barney', 'pebbles');
         * // => 'hello fred, barney, & pebbles'
         */
        function rest(func, start) {
          if (typeof func != 'function') {
            throw new TypeError(FUNC_ERROR_TEXT);
          }
          start = start === undefined$1 ? start : toInteger(start);
          return baseRest(func, start);
        }

        /**
         * Creates a function that invokes `func` with the `this` binding of the
         * create function and an array of arguments much like
         * [`Function#apply`](http://www.ecma-international.org/ecma-262/7.0/#sec-function.prototype.apply).
         *
         * **Note:** This method is based on the
         * [spread operator](https://mdn.io/spread_operator).
         *
         * @static
         * @memberOf _
         * @since 3.2.0
         * @category Function
         * @param {Function} func The function to spread arguments over.
         * @param {number} [start=0] The start position of the spread.
         * @returns {Function} Returns the new function.
         * @example
         *
         * var say = _.spread(function(who, what) {
         *   return who + ' says ' + what;
         * });
         *
         * say(['fred', 'hello']);
         * // => 'fred says hello'
         *
         * var numbers = Promise.all([
         *   Promise.resolve(40),
         *   Promise.resolve(36)
         * ]);
         *
         * numbers.then(_.spread(function(x, y) {
         *   return x + y;
         * }));
         * // => a Promise of 76
         */
        function spread(func, start) {
          if (typeof func != 'function') {
            throw new TypeError(FUNC_ERROR_TEXT);
          }
          start = start == null ? 0 : nativeMax(toInteger(start), 0);
          return baseRest(function(args) {
            var array = args[start],
                otherArgs = castSlice(args, 0, start);

            if (array) {
              arrayPush(otherArgs, array);
            }
            return apply(func, this, otherArgs);
          });
        }

        /**
         * Creates a throttled function that only invokes `func` at most once per
         * every `wait` milliseconds. The throttled function comes with a `cancel`
         * method to cancel delayed `func` invocations and a `flush` method to
         * immediately invoke them. Provide `options` to indicate whether `func`
         * should be invoked on the leading and/or trailing edge of the `wait`
         * timeout. The `func` is invoked with the last arguments provided to the
         * throttled function. Subsequent calls to the throttled function return the
         * result of the last `func` invocation.
         *
         * **Note:** If `leading` and `trailing` options are `true`, `func` is
         * invoked on the trailing edge of the timeout only if the throttled function
         * is invoked more than once during the `wait` timeout.
         *
         * If `wait` is `0` and `leading` is `false`, `func` invocation is deferred
         * until to the next tick, similar to `setTimeout` with a timeout of `0`.
         *
         * See [David Corbacho's article](https://css-tricks.com/debouncing-throttling-explained-examples/)
         * for details over the differences between `_.throttle` and `_.debounce`.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Function
         * @param {Function} func The function to throttle.
         * @param {number} [wait=0] The number of milliseconds to throttle invocations to.
         * @param {Object} [options={}] The options object.
         * @param {boolean} [options.leading=true]
         *  Specify invoking on the leading edge of the timeout.
         * @param {boolean} [options.trailing=true]
         *  Specify invoking on the trailing edge of the timeout.
         * @returns {Function} Returns the new throttled function.
         * @example
         *
         * // Avoid excessively updating the position while scrolling.
         * jQuery(window).on('scroll', _.throttle(updatePosition, 100));
         *
         * // Invoke `renewToken` when the click event is fired, but not more than once every 5 minutes.
         * var throttled = _.throttle(renewToken, 300000, { 'trailing': false });
         * jQuery(element).on('click', throttled);
         *
         * // Cancel the trailing throttled invocation.
         * jQuery(window).on('popstate', throttled.cancel);
         */
        function throttle(func, wait, options) {
          var leading = true,
              trailing = true;

          if (typeof func != 'function') {
            throw new TypeError(FUNC_ERROR_TEXT);
          }
          if (isObject(options)) {
            leading = 'leading' in options ? !!options.leading : leading;
            trailing = 'trailing' in options ? !!options.trailing : trailing;
          }
          return debounce(func, wait, {
            'leading': leading,
            'maxWait': wait,
            'trailing': trailing
          });
        }

        /**
         * Creates a function that accepts up to one argument, ignoring any
         * additional arguments.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Function
         * @param {Function} func The function to cap arguments for.
         * @returns {Function} Returns the new capped function.
         * @example
         *
         * _.map(['6', '8', '10'], _.unary(parseInt));
         * // => [6, 8, 10]
         */
        function unary(func) {
          return ary(func, 1);
        }

        /**
         * Creates a function that provides `value` to `wrapper` as its first
         * argument. Any additional arguments provided to the function are appended
         * to those provided to the `wrapper`. The wrapper is invoked with the `this`
         * binding of the created function.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Function
         * @param {*} value The value to wrap.
         * @param {Function} [wrapper=identity] The wrapper function.
         * @returns {Function} Returns the new function.
         * @example
         *
         * var p = _.wrap(_.escape, function(func, text) {
         *   return '<p>' + func(text) + '</p>';
         * });
         *
         * p('fred, barney, & pebbles');
         * // => '<p>fred, barney, &amp; pebbles</p>'
         */
        function wrap(value, wrapper) {
          return partial(castFunction(wrapper), value);
        }

        /*------------------------------------------------------------------------*/

        /**
         * Casts `value` as an array if it's not one.
         *
         * @static
         * @memberOf _
         * @since 4.4.0
         * @category Lang
         * @param {*} value The value to inspect.
         * @returns {Array} Returns the cast array.
         * @example
         *
         * _.castArray(1);
         * // => [1]
         *
         * _.castArray({ 'a': 1 });
         * // => [{ 'a': 1 }]
         *
         * _.castArray('abc');
         * // => ['abc']
         *
         * _.castArray(null);
         * // => [null]
         *
         * _.castArray(undefined);
         * // => [undefined]
         *
         * _.castArray();
         * // => []
         *
         * var array = [1, 2, 3];
         * console.log(_.castArray(array) === array);
         * // => true
         */
        function castArray() {
          if (!arguments.length) {
            return [];
          }
          var value = arguments[0];
          return isArray(value) ? value : [value];
        }

        /**
         * Creates a shallow clone of `value`.
         *
         * **Note:** This method is loosely based on the
         * [structured clone algorithm](https://mdn.io/Structured_clone_algorithm)
         * and supports cloning arrays, array buffers, booleans, date objects, maps,
         * numbers, `Object` objects, regexes, sets, strings, symbols, and typed
         * arrays. The own enumerable properties of `arguments` objects are cloned
         * as plain objects. An empty object is returned for uncloneable values such
         * as error objects, functions, DOM nodes, and WeakMaps.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Lang
         * @param {*} value The value to clone.
         * @returns {*} Returns the cloned value.
         * @see _.cloneDeep
         * @example
         *
         * var objects = [{ 'a': 1 }, { 'b': 2 }];
         *
         * var shallow = _.clone(objects);
         * console.log(shallow[0] === objects[0]);
         * // => true
         */
        function clone(value) {
          return baseClone(value, CLONE_SYMBOLS_FLAG);
        }

        /**
         * This method is like `_.clone` except that it accepts `customizer` which
         * is invoked to produce the cloned value. If `customizer` returns `undefined`,
         * cloning is handled by the method instead. The `customizer` is invoked with
         * up to four arguments; (value [, index|key, object, stack]).
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Lang
         * @param {*} value The value to clone.
         * @param {Function} [customizer] The function to customize cloning.
         * @returns {*} Returns the cloned value.
         * @see _.cloneDeepWith
         * @example
         *
         * function customizer(value) {
         *   if (_.isElement(value)) {
         *     return value.cloneNode(false);
         *   }
         * }
         *
         * var el = _.cloneWith(document.body, customizer);
         *
         * console.log(el === document.body);
         * // => false
         * console.log(el.nodeName);
         * // => 'BODY'
         * console.log(el.childNodes.length);
         * // => 0
         */
        function cloneWith(value, customizer) {
          customizer = typeof customizer == 'function' ? customizer : undefined$1;
          return baseClone(value, CLONE_SYMBOLS_FLAG, customizer);
        }

        /**
         * This method is like `_.clone` except that it recursively clones `value`.
         *
         * @static
         * @memberOf _
         * @since 1.0.0
         * @category Lang
         * @param {*} value The value to recursively clone.
         * @returns {*} Returns the deep cloned value.
         * @see _.clone
         * @example
         *
         * var objects = [{ 'a': 1 }, { 'b': 2 }];
         *
         * var deep = _.cloneDeep(objects);
         * console.log(deep[0] === objects[0]);
         * // => false
         */
        function cloneDeep(value) {
          return baseClone(value, CLONE_DEEP_FLAG | CLONE_SYMBOLS_FLAG);
        }

        /**
         * This method is like `_.cloneWith` except that it recursively clones `value`.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Lang
         * @param {*} value The value to recursively clone.
         * @param {Function} [customizer] The function to customize cloning.
         * @returns {*} Returns the deep cloned value.
         * @see _.cloneWith
         * @example
         *
         * function customizer(value) {
         *   if (_.isElement(value)) {
         *     return value.cloneNode(true);
         *   }
         * }
         *
         * var el = _.cloneDeepWith(document.body, customizer);
         *
         * console.log(el === document.body);
         * // => false
         * console.log(el.nodeName);
         * // => 'BODY'
         * console.log(el.childNodes.length);
         * // => 20
         */
        function cloneDeepWith(value, customizer) {
          customizer = typeof customizer == 'function' ? customizer : undefined$1;
          return baseClone(value, CLONE_DEEP_FLAG | CLONE_SYMBOLS_FLAG, customizer);
        }

        /**
         * Checks if `object` conforms to `source` by invoking the predicate
         * properties of `source` with the corresponding property values of `object`.
         *
         * **Note:** This method is equivalent to `_.conforms` when `source` is
         * partially applied.
         *
         * @static
         * @memberOf _
         * @since 4.14.0
         * @category Lang
         * @param {Object} object The object to inspect.
         * @param {Object} source The object of property predicates to conform to.
         * @returns {boolean} Returns `true` if `object` conforms, else `false`.
         * @example
         *
         * var object = { 'a': 1, 'b': 2 };
         *
         * _.conformsTo(object, { 'b': function(n) { return n > 1; } });
         * // => true
         *
         * _.conformsTo(object, { 'b': function(n) { return n > 2; } });
         * // => false
         */
        function conformsTo(object, source) {
          return source == null || baseConformsTo(object, source, keys(source));
        }

        /**
         * Performs a
         * [`SameValueZero`](http://ecma-international.org/ecma-262/7.0/#sec-samevaluezero)
         * comparison between two values to determine if they are equivalent.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Lang
         * @param {*} value The value to compare.
         * @param {*} other The other value to compare.
         * @returns {boolean} Returns `true` if the values are equivalent, else `false`.
         * @example
         *
         * var object = { 'a': 1 };
         * var other = { 'a': 1 };
         *
         * _.eq(object, object);
         * // => true
         *
         * _.eq(object, other);
         * // => false
         *
         * _.eq('a', 'a');
         * // => true
         *
         * _.eq('a', Object('a'));
         * // => false
         *
         * _.eq(NaN, NaN);
         * // => true
         */
        function eq(value, other) {
          return value === other || (value !== value && other !== other);
        }

        /**
         * Checks if `value` is greater than `other`.
         *
         * @static
         * @memberOf _
         * @since 3.9.0
         * @category Lang
         * @param {*} value The value to compare.
         * @param {*} other The other value to compare.
         * @returns {boolean} Returns `true` if `value` is greater than `other`,
         *  else `false`.
         * @see _.lt
         * @example
         *
         * _.gt(3, 1);
         * // => true
         *
         * _.gt(3, 3);
         * // => false
         *
         * _.gt(1, 3);
         * // => false
         */
        var gt = createRelationalOperation(baseGt);

        /**
         * Checks if `value` is greater than or equal to `other`.
         *
         * @static
         * @memberOf _
         * @since 3.9.0
         * @category Lang
         * @param {*} value The value to compare.
         * @param {*} other The other value to compare.
         * @returns {boolean} Returns `true` if `value` is greater than or equal to
         *  `other`, else `false`.
         * @see _.lte
         * @example
         *
         * _.gte(3, 1);
         * // => true
         *
         * _.gte(3, 3);
         * // => true
         *
         * _.gte(1, 3);
         * // => false
         */
        var gte = createRelationalOperation(function(value, other) {
          return value >= other;
        });

        /**
         * Checks if `value` is likely an `arguments` object.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Lang
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is an `arguments` object,
         *  else `false`.
         * @example
         *
         * _.isArguments(function() { return arguments; }());
         * // => true
         *
         * _.isArguments([1, 2, 3]);
         * // => false
         */
        var isArguments = baseIsArguments(function() { return arguments; }()) ? baseIsArguments : function(value) {
          return isObjectLike(value) && hasOwnProperty.call(value, 'callee') &&
            !propertyIsEnumerable.call(value, 'callee');
        };

        /**
         * Checks if `value` is classified as an `Array` object.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Lang
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is an array, else `false`.
         * @example
         *
         * _.isArray([1, 2, 3]);
         * // => true
         *
         * _.isArray(document.body.children);
         * // => false
         *
         * _.isArray('abc');
         * // => false
         *
         * _.isArray(_.noop);
         * // => false
         */
        var isArray = Array.isArray;

        /**
         * Checks if `value` is classified as an `ArrayBuffer` object.
         *
         * @static
         * @memberOf _
         * @since 4.3.0
         * @category Lang
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is an array buffer, else `false`.
         * @example
         *
         * _.isArrayBuffer(new ArrayBuffer(2));
         * // => true
         *
         * _.isArrayBuffer(new Array(2));
         * // => false
         */
        var isArrayBuffer = nodeIsArrayBuffer ? baseUnary(nodeIsArrayBuffer) : baseIsArrayBuffer;

        /**
         * Checks if `value` is array-like. A value is considered array-like if it's
         * not a function and has a `value.length` that's an integer greater than or
         * equal to `0` and less than or equal to `Number.MAX_SAFE_INTEGER`.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Lang
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is array-like, else `false`.
         * @example
         *
         * _.isArrayLike([1, 2, 3]);
         * // => true
         *
         * _.isArrayLike(document.body.children);
         * // => true
         *
         * _.isArrayLike('abc');
         * // => true
         *
         * _.isArrayLike(_.noop);
         * // => false
         */
        function isArrayLike(value) {
          return value != null && isLength(value.length) && !isFunction(value);
        }

        /**
         * This method is like `_.isArrayLike` except that it also checks if `value`
         * is an object.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Lang
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is an array-like object,
         *  else `false`.
         * @example
         *
         * _.isArrayLikeObject([1, 2, 3]);
         * // => true
         *
         * _.isArrayLikeObject(document.body.children);
         * // => true
         *
         * _.isArrayLikeObject('abc');
         * // => false
         *
         * _.isArrayLikeObject(_.noop);
         * // => false
         */
        function isArrayLikeObject(value) {
          return isObjectLike(value) && isArrayLike(value);
        }

        /**
         * Checks if `value` is classified as a boolean primitive or object.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Lang
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is a boolean, else `false`.
         * @example
         *
         * _.isBoolean(false);
         * // => true
         *
         * _.isBoolean(null);
         * // => false
         */
        function isBoolean(value) {
          return value === true || value === false ||
            (isObjectLike(value) && baseGetTag(value) == boolTag);
        }

        /**
         * Checks if `value` is a buffer.
         *
         * @static
         * @memberOf _
         * @since 4.3.0
         * @category Lang
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is a buffer, else `false`.
         * @example
         *
         * _.isBuffer(new Buffer(2));
         * // => true
         *
         * _.isBuffer(new Uint8Array(2));
         * // => false
         */
        var isBuffer = nativeIsBuffer || stubFalse;

        /**
         * Checks if `value` is classified as a `Date` object.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Lang
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is a date object, else `false`.
         * @example
         *
         * _.isDate(new Date);
         * // => true
         *
         * _.isDate('Mon April 23 2012');
         * // => false
         */
        var isDate = nodeIsDate ? baseUnary(nodeIsDate) : baseIsDate;

        /**
         * Checks if `value` is likely a DOM element.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Lang
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is a DOM element, else `false`.
         * @example
         *
         * _.isElement(document.body);
         * // => true
         *
         * _.isElement('<body>');
         * // => false
         */
        function isElement(value) {
          return isObjectLike(value) && value.nodeType === 1 && !isPlainObject(value);
        }

        /**
         * Checks if `value` is an empty object, collection, map, or set.
         *
         * Objects are considered empty if they have no own enumerable string keyed
         * properties.
         *
         * Array-like values such as `arguments` objects, arrays, buffers, strings, or
         * jQuery-like collections are considered empty if they have a `length` of `0`.
         * Similarly, maps and sets are considered empty if they have a `size` of `0`.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Lang
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is empty, else `false`.
         * @example
         *
         * _.isEmpty(null);
         * // => true
         *
         * _.isEmpty(true);
         * // => true
         *
         * _.isEmpty(1);
         * // => true
         *
         * _.isEmpty([1, 2, 3]);
         * // => false
         *
         * _.isEmpty({ 'a': 1 });
         * // => false
         */
        function isEmpty(value) {
          if (value == null) {
            return true;
          }
          if (isArrayLike(value) &&
              (isArray(value) || typeof value == 'string' || typeof value.splice == 'function' ||
                isBuffer(value) || isTypedArray(value) || isArguments(value))) {
            return !value.length;
          }
          var tag = getTag(value);
          if (tag == mapTag || tag == setTag) {
            return !value.size;
          }
          if (isPrototype(value)) {
            return !baseKeys(value).length;
          }
          for (var key in value) {
            if (hasOwnProperty.call(value, key)) {
              return false;
            }
          }
          return true;
        }

        /**
         * Performs a deep comparison between two values to determine if they are
         * equivalent.
         *
         * **Note:** This method supports comparing arrays, array buffers, booleans,
         * date objects, error objects, maps, numbers, `Object` objects, regexes,
         * sets, strings, symbols, and typed arrays. `Object` objects are compared
         * by their own, not inherited, enumerable properties. Functions and DOM
         * nodes are compared by strict equality, i.e. `===`.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Lang
         * @param {*} value The value to compare.
         * @param {*} other The other value to compare.
         * @returns {boolean} Returns `true` if the values are equivalent, else `false`.
         * @example
         *
         * var object = { 'a': 1 };
         * var other = { 'a': 1 };
         *
         * _.isEqual(object, other);
         * // => true
         *
         * object === other;
         * // => false
         */
        function isEqual(value, other) {
          return baseIsEqual(value, other);
        }

        /**
         * This method is like `_.isEqual` except that it accepts `customizer` which
         * is invoked to compare values. If `customizer` returns `undefined`, comparisons
         * are handled by the method instead. The `customizer` is invoked with up to
         * six arguments: (objValue, othValue [, index|key, object, other, stack]).
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Lang
         * @param {*} value The value to compare.
         * @param {*} other The other value to compare.
         * @param {Function} [customizer] The function to customize comparisons.
         * @returns {boolean} Returns `true` if the values are equivalent, else `false`.
         * @example
         *
         * function isGreeting(value) {
         *   return /^h(?:i|ello)$/.test(value);
         * }
         *
         * function customizer(objValue, othValue) {
         *   if (isGreeting(objValue) && isGreeting(othValue)) {
         *     return true;
         *   }
         * }
         *
         * var array = ['hello', 'goodbye'];
         * var other = ['hi', 'goodbye'];
         *
         * _.isEqualWith(array, other, customizer);
         * // => true
         */
        function isEqualWith(value, other, customizer) {
          customizer = typeof customizer == 'function' ? customizer : undefined$1;
          var result = customizer ? customizer(value, other) : undefined$1;
          return result === undefined$1 ? baseIsEqual(value, other, undefined$1, customizer) : !!result;
        }

        /**
         * Checks if `value` is an `Error`, `EvalError`, `RangeError`, `ReferenceError`,
         * `SyntaxError`, `TypeError`, or `URIError` object.
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category Lang
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is an error object, else `false`.
         * @example
         *
         * _.isError(new Error);
         * // => true
         *
         * _.isError(Error);
         * // => false
         */
        function isError(value) {
          if (!isObjectLike(value)) {
            return false;
          }
          var tag = baseGetTag(value);
          return tag == errorTag || tag == domExcTag ||
            (typeof value.message == 'string' && typeof value.name == 'string' && !isPlainObject(value));
        }

        /**
         * Checks if `value` is a finite primitive number.
         *
         * **Note:** This method is based on
         * [`Number.isFinite`](https://mdn.io/Number/isFinite).
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Lang
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is a finite number, else `false`.
         * @example
         *
         * _.isFinite(3);
         * // => true
         *
         * _.isFinite(Number.MIN_VALUE);
         * // => true
         *
         * _.isFinite(Infinity);
         * // => false
         *
         * _.isFinite('3');
         * // => false
         */
        function isFinite(value) {
          return typeof value == 'number' && nativeIsFinite(value);
        }

        /**
         * Checks if `value` is classified as a `Function` object.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Lang
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is a function, else `false`.
         * @example
         *
         * _.isFunction(_);
         * // => true
         *
         * _.isFunction(/abc/);
         * // => false
         */
        function isFunction(value) {
          if (!isObject(value)) {
            return false;
          }
          // The use of `Object#toString` avoids issues with the `typeof` operator
          // in Safari 9 which returns 'object' for typed arrays and other constructors.
          var tag = baseGetTag(value);
          return tag == funcTag || tag == genTag || tag == asyncTag || tag == proxyTag;
        }

        /**
         * Checks if `value` is an integer.
         *
         * **Note:** This method is based on
         * [`Number.isInteger`](https://mdn.io/Number/isInteger).
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Lang
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is an integer, else `false`.
         * @example
         *
         * _.isInteger(3);
         * // => true
         *
         * _.isInteger(Number.MIN_VALUE);
         * // => false
         *
         * _.isInteger(Infinity);
         * // => false
         *
         * _.isInteger('3');
         * // => false
         */
        function isInteger(value) {
          return typeof value == 'number' && value == toInteger(value);
        }

        /**
         * Checks if `value` is a valid array-like length.
         *
         * **Note:** This method is loosely based on
         * [`ToLength`](http://ecma-international.org/ecma-262/7.0/#sec-tolength).
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Lang
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is a valid length, else `false`.
         * @example
         *
         * _.isLength(3);
         * // => true
         *
         * _.isLength(Number.MIN_VALUE);
         * // => false
         *
         * _.isLength(Infinity);
         * // => false
         *
         * _.isLength('3');
         * // => false
         */
        function isLength(value) {
          return typeof value == 'number' &&
            value > -1 && value % 1 == 0 && value <= MAX_SAFE_INTEGER;
        }

        /**
         * Checks if `value` is the
         * [language type](http://www.ecma-international.org/ecma-262/7.0/#sec-ecmascript-language-types)
         * of `Object`. (e.g. arrays, functions, objects, regexes, `new Number(0)`, and `new String('')`)
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Lang
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is an object, else `false`.
         * @example
         *
         * _.isObject({});
         * // => true
         *
         * _.isObject([1, 2, 3]);
         * // => true
         *
         * _.isObject(_.noop);
         * // => true
         *
         * _.isObject(null);
         * // => false
         */
        function isObject(value) {
          var type = typeof value;
          return value != null && (type == 'object' || type == 'function');
        }

        /**
         * Checks if `value` is object-like. A value is object-like if it's not `null`
         * and has a `typeof` result of "object".
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Lang
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is object-like, else `false`.
         * @example
         *
         * _.isObjectLike({});
         * // => true
         *
         * _.isObjectLike([1, 2, 3]);
         * // => true
         *
         * _.isObjectLike(_.noop);
         * // => false
         *
         * _.isObjectLike(null);
         * // => false
         */
        function isObjectLike(value) {
          return value != null && typeof value == 'object';
        }

        /**
         * Checks if `value` is classified as a `Map` object.
         *
         * @static
         * @memberOf _
         * @since 4.3.0
         * @category Lang
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is a map, else `false`.
         * @example
         *
         * _.isMap(new Map);
         * // => true
         *
         * _.isMap(new WeakMap);
         * // => false
         */
        var isMap = nodeIsMap ? baseUnary(nodeIsMap) : baseIsMap;

        /**
         * Performs a partial deep comparison between `object` and `source` to
         * determine if `object` contains equivalent property values.
         *
         * **Note:** This method is equivalent to `_.matches` when `source` is
         * partially applied.
         *
         * Partial comparisons will match empty array and empty object `source`
         * values against any array or object value, respectively. See `_.isEqual`
         * for a list of supported value comparisons.
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category Lang
         * @param {Object} object The object to inspect.
         * @param {Object} source The object of property values to match.
         * @returns {boolean} Returns `true` if `object` is a match, else `false`.
         * @example
         *
         * var object = { 'a': 1, 'b': 2 };
         *
         * _.isMatch(object, { 'b': 2 });
         * // => true
         *
         * _.isMatch(object, { 'b': 1 });
         * // => false
         */
        function isMatch(object, source) {
          return object === source || baseIsMatch(object, source, getMatchData(source));
        }

        /**
         * This method is like `_.isMatch` except that it accepts `customizer` which
         * is invoked to compare values. If `customizer` returns `undefined`, comparisons
         * are handled by the method instead. The `customizer` is invoked with five
         * arguments: (objValue, srcValue, index|key, object, source).
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Lang
         * @param {Object} object The object to inspect.
         * @param {Object} source The object of property values to match.
         * @param {Function} [customizer] The function to customize comparisons.
         * @returns {boolean} Returns `true` if `object` is a match, else `false`.
         * @example
         *
         * function isGreeting(value) {
         *   return /^h(?:i|ello)$/.test(value);
         * }
         *
         * function customizer(objValue, srcValue) {
         *   if (isGreeting(objValue) && isGreeting(srcValue)) {
         *     return true;
         *   }
         * }
         *
         * var object = { 'greeting': 'hello' };
         * var source = { 'greeting': 'hi' };
         *
         * _.isMatchWith(object, source, customizer);
         * // => true
         */
        function isMatchWith(object, source, customizer) {
          customizer = typeof customizer == 'function' ? customizer : undefined$1;
          return baseIsMatch(object, source, getMatchData(source), customizer);
        }

        /**
         * Checks if `value` is `NaN`.
         *
         * **Note:** This method is based on
         * [`Number.isNaN`](https://mdn.io/Number/isNaN) and is not the same as
         * global [`isNaN`](https://mdn.io/isNaN) which returns `true` for
         * `undefined` and other non-number values.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Lang
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is `NaN`, else `false`.
         * @example
         *
         * _.isNaN(NaN);
         * // => true
         *
         * _.isNaN(new Number(NaN));
         * // => true
         *
         * isNaN(undefined);
         * // => true
         *
         * _.isNaN(undefined);
         * // => false
         */
        function isNaN(value) {
          // An `NaN` primitive is the only value that is not equal to itself.
          // Perform the `toStringTag` check first to avoid errors with some
          // ActiveX objects in IE.
          return isNumber(value) && value != +value;
        }

        /**
         * Checks if `value` is a pristine native function.
         *
         * **Note:** This method can't reliably detect native functions in the presence
         * of the core-js package because core-js circumvents this kind of detection.
         * Despite multiple requests, the core-js maintainer has made it clear: any
         * attempt to fix the detection will be obstructed. As a result, we're left
         * with little choice but to throw an error. Unfortunately, this also affects
         * packages, like [babel-polyfill](https://www.npmjs.com/package/babel-polyfill),
         * which rely on core-js.
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category Lang
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is a native function,
         *  else `false`.
         * @example
         *
         * _.isNative(Array.prototype.push);
         * // => true
         *
         * _.isNative(_);
         * // => false
         */
        function isNative(value) {
          if (isMaskable(value)) {
            throw new Error(CORE_ERROR_TEXT);
          }
          return baseIsNative(value);
        }

        /**
         * Checks if `value` is `null`.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Lang
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is `null`, else `false`.
         * @example
         *
         * _.isNull(null);
         * // => true
         *
         * _.isNull(void 0);
         * // => false
         */
        function isNull(value) {
          return value === null;
        }

        /**
         * Checks if `value` is `null` or `undefined`.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Lang
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is nullish, else `false`.
         * @example
         *
         * _.isNil(null);
         * // => true
         *
         * _.isNil(void 0);
         * // => true
         *
         * _.isNil(NaN);
         * // => false
         */
        function isNil(value) {
          return value == null;
        }

        /**
         * Checks if `value` is classified as a `Number` primitive or object.
         *
         * **Note:** To exclude `Infinity`, `-Infinity`, and `NaN`, which are
         * classified as numbers, use the `_.isFinite` method.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Lang
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is a number, else `false`.
         * @example
         *
         * _.isNumber(3);
         * // => true
         *
         * _.isNumber(Number.MIN_VALUE);
         * // => true
         *
         * _.isNumber(Infinity);
         * // => true
         *
         * _.isNumber('3');
         * // => false
         */
        function isNumber(value) {
          return typeof value == 'number' ||
            (isObjectLike(value) && baseGetTag(value) == numberTag);
        }

        /**
         * Checks if `value` is a plain object, that is, an object created by the
         * `Object` constructor or one with a `[[Prototype]]` of `null`.
         *
         * @static
         * @memberOf _
         * @since 0.8.0
         * @category Lang
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is a plain object, else `false`.
         * @example
         *
         * function Foo() {
         *   this.a = 1;
         * }
         *
         * _.isPlainObject(new Foo);
         * // => false
         *
         * _.isPlainObject([1, 2, 3]);
         * // => false
         *
         * _.isPlainObject({ 'x': 0, 'y': 0 });
         * // => true
         *
         * _.isPlainObject(Object.create(null));
         * // => true
         */
        function isPlainObject(value) {
          if (!isObjectLike(value) || baseGetTag(value) != objectTag) {
            return false;
          }
          var proto = getPrototype(value);
          if (proto === null) {
            return true;
          }
          var Ctor = hasOwnProperty.call(proto, 'constructor') && proto.constructor;
          return typeof Ctor == 'function' && Ctor instanceof Ctor &&
            funcToString.call(Ctor) == objectCtorString;
        }

        /**
         * Checks if `value` is classified as a `RegExp` object.
         *
         * @static
         * @memberOf _
         * @since 0.1.0
         * @category Lang
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is a regexp, else `false`.
         * @example
         *
         * _.isRegExp(/abc/);
         * // => true
         *
         * _.isRegExp('/abc/');
         * // => false
         */
        var isRegExp = nodeIsRegExp ? baseUnary(nodeIsRegExp) : baseIsRegExp;

        /**
         * Checks if `value` is a safe integer. An integer is safe if it's an IEEE-754
         * double precision number which isn't the result of a rounded unsafe integer.
         *
         * **Note:** This method is based on
         * [`Number.isSafeInteger`](https://mdn.io/Number/isSafeInteger).
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Lang
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is a safe integer, else `false`.
         * @example
         *
         * _.isSafeInteger(3);
         * // => true
         *
         * _.isSafeInteger(Number.MIN_VALUE);
         * // => false
         *
         * _.isSafeInteger(Infinity);
         * // => false
         *
         * _.isSafeInteger('3');
         * // => false
         */
        function isSafeInteger(value) {
          return isInteger(value) && value >= -MAX_SAFE_INTEGER && value <= MAX_SAFE_INTEGER;
        }

        /**
         * Checks if `value` is classified as a `Set` object.
         *
         * @static
         * @memberOf _
         * @since 4.3.0
         * @category Lang
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is a set, else `false`.
         * @example
         *
         * _.isSet(new Set);
         * // => true
         *
         * _.isSet(new WeakSet);
         * // => false
         */
        var isSet = nodeIsSet ? baseUnary(nodeIsSet) : baseIsSet;

        /**
         * Checks if `value` is classified as a `String` primitive or object.
         *
         * @static
         * @since 0.1.0
         * @memberOf _
         * @category Lang
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is a string, else `false`.
         * @example
         *
         * _.isString('abc');
         * // => true
         *
         * _.isString(1);
         * // => false
         */
        function isString(value) {
          return typeof value == 'string' ||
            (!isArray(value) && isObjectLike(value) && baseGetTag(value) == stringTag);
        }

        /**
         * Checks if `value` is classified as a `Symbol` primitive or object.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Lang
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is a symbol, else `false`.
         * @example
         *
         * _.isSymbol(Symbol.iterator);
         * // => true
         *
         * _.isSymbol('abc');
         * // => false
         */
        function isSymbol(value) {
          return typeof value == 'symbol' ||
            (isObjectLike(value) && baseGetTag(value) == symbolTag);
        }

        /**
         * Checks if `value` is classified as a typed array.
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category Lang
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is a typed array, else `false`.
         * @example
         *
         * _.isTypedArray(new Uint8Array);
         * // => true
         *
         * _.isTypedArray([]);
         * // => false
         */
        var isTypedArray = nodeIsTypedArray ? baseUnary(nodeIsTypedArray) : baseIsTypedArray;

        /**
         * Checks if `value` is `undefined`.
         *
         * @static
         * @since 0.1.0
         * @memberOf _
         * @category Lang
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is `undefined`, else `false`.
         * @example
         *
         * _.isUndefined(void 0);
         * // => true
         *
         * _.isUndefined(null);
         * // => false
         */
        function isUndefined(value) {
          return value === undefined$1;
        }

        /**
         * Checks if `value` is classified as a `WeakMap` object.
         *
         * @static
         * @memberOf _
         * @since 4.3.0
         * @category Lang
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is a weak map, else `false`.
         * @example
         *
         * _.isWeakMap(new WeakMap);
         * // => true
         *
         * _.isWeakMap(new Map);
         * // => false
         */
        function isWeakMap(value) {
          return isObjectLike(value) && getTag(value) == weakMapTag;
        }

        /**
         * Checks if `value` is classified as a `WeakSet` object.
         *
         * @static
         * @memberOf _
         * @since 4.3.0
         * @category Lang
         * @param {*} value The value to check.
         * @returns {boolean} Returns `true` if `value` is a weak set, else `false`.
         * @example
         *
         * _.isWeakSet(new WeakSet);
         * // => true
         *
         * _.isWeakSet(new Set);
         * // => false
         */
        function isWeakSet(value) {
          return isObjectLike(value) && baseGetTag(value) == weakSetTag;
        }

        /**
         * Checks if `value` is less than `other`.
         *
         * @static
         * @memberOf _
         * @since 3.9.0
         * @category Lang
         * @param {*} value The value to compare.
         * @param {*} other The other value to compare.
         * @returns {boolean} Returns `true` if `value` is less than `other`,
         *  else `false`.
         * @see _.gt
         * @example
         *
         * _.lt(1, 3);
         * // => true
         *
         * _.lt(3, 3);
         * // => false
         *
         * _.lt(3, 1);
         * // => false
         */
        var lt = createRelationalOperation(baseLt);

        /**
         * Checks if `value` is less than or equal to `other`.
         *
         * @static
         * @memberOf _
         * @since 3.9.0
         * @category Lang
         * @param {*} value The value to compare.
         * @param {*} other The other value to compare.
         * @returns {boolean} Returns `true` if `value` is less than or equal to
         *  `other`, else `false`.
         * @see _.gte
         * @example
         *
         * _.lte(1, 3);
         * // => true
         *
         * _.lte(3, 3);
         * // => true
         *
         * _.lte(3, 1);
         * // => false
         */
        var lte = createRelationalOperation(function(value, other) {
          return value <= other;
        });

        /**
         * Converts `value` to an array.
         *
         * @static
         * @since 0.1.0
         * @memberOf _
         * @category Lang
         * @param {*} value The value to convert.
         * @returns {Array} Returns the converted array.
         * @example
         *
         * _.toArray({ 'a': 1, 'b': 2 });
         * // => [1, 2]
         *
         * _.toArray('abc');
         * // => ['a', 'b', 'c']
         *
         * _.toArray(1);
         * // => []
         *
         * _.toArray(null);
         * // => []
         */
        function toArray(value) {
          if (!value) {
            return [];
          }
          if (isArrayLike(value)) {
            return isString(value) ? stringToArray(value) : copyArray(value);
          }
          if (symIterator && value[symIterator]) {
            return iteratorToArray(value[symIterator]());
          }
          var tag = getTag(value),
              func = tag == mapTag ? mapToArray : (tag == setTag ? setToArray : values);

          return func(value);
        }

        /**
         * Converts `value` to a finite number.
         *
         * @static
         * @memberOf _
         * @since 4.12.0
         * @category Lang
         * @param {*} value The value to convert.
         * @returns {number} Returns the converted number.
         * @example
         *
         * _.toFinite(3.2);
         * // => 3.2
         *
         * _.toFinite(Number.MIN_VALUE);
         * // => 5e-324
         *
         * _.toFinite(Infinity);
         * // => 1.7976931348623157e+308
         *
         * _.toFinite('3.2');
         * // => 3.2
         */
        function toFinite(value) {
          if (!value) {
            return value === 0 ? value : 0;
          }
          value = toNumber(value);
          if (value === INFINITY || value === -INFINITY) {
            var sign = (value < 0 ? -1 : 1);
            return sign * MAX_INTEGER;
          }
          return value === value ? value : 0;
        }

        /**
         * Converts `value` to an integer.
         *
         * **Note:** This method is loosely based on
         * [`ToInteger`](http://www.ecma-international.org/ecma-262/7.0/#sec-tointeger).
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Lang
         * @param {*} value The value to convert.
         * @returns {number} Returns the converted integer.
         * @example
         *
         * _.toInteger(3.2);
         * // => 3
         *
         * _.toInteger(Number.MIN_VALUE);
         * // => 0
         *
         * _.toInteger(Infinity);
         * // => 1.7976931348623157e+308
         *
         * _.toInteger('3.2');
         * // => 3
         */
        function toInteger(value) {
          var result = toFinite(value),
              remainder = result % 1;

          return result === result ? (remainder ? result - remainder : result) : 0;
        }

        /**
         * Converts `value` to an integer suitable for use as the length of an
         * array-like object.
         *
         * **Note:** This method is based on
         * [`ToLength`](http://ecma-international.org/ecma-262/7.0/#sec-tolength).
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Lang
         * @param {*} value The value to convert.
         * @returns {number} Returns the converted integer.
         * @example
         *
         * _.toLength(3.2);
         * // => 3
         *
         * _.toLength(Number.MIN_VALUE);
         * // => 0
         *
         * _.toLength(Infinity);
         * // => 4294967295
         *
         * _.toLength('3.2');
         * // => 3
         */
        function toLength(value) {
          return value ? baseClamp(toInteger(value), 0, MAX_ARRAY_LENGTH) : 0;
        }

        /**
         * Converts `value` to a number.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Lang
         * @param {*} value The value to process.
         * @returns {number} Returns the number.
         * @example
         *
         * _.toNumber(3.2);
         * // => 3.2
         *
         * _.toNumber(Number.MIN_VALUE);
         * // => 5e-324
         *
         * _.toNumber(Infinity);
         * // => Infinity
         *
         * _.toNumber('3.2');
         * // => 3.2
         */
        function toNumber(value) {
          if (typeof value == 'number') {
            return value;
          }
          if (isSymbol(value)) {
            return NAN;
          }
          if (isObject(value)) {
            var other = typeof value.valueOf == 'function' ? value.valueOf() : value;
            value = isObject(other) ? (other + '') : other;
          }
          if (typeof value != 'string') {
            return value === 0 ? value : +value;
          }
          value = value.replace(reTrim, '');
          var isBinary = reIsBinary.test(value);
          return (isBinary || reIsOctal.test(value))
            ? freeParseInt(value.slice(2), isBinary ? 2 : 8)
            : (reIsBadHex.test(value) ? NAN : +value);
        }

        /**
         * Converts `value` to a plain object flattening inherited enumerable string
         * keyed properties of `value` to own properties of the plain object.
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category Lang
         * @param {*} value The value to convert.
         * @returns {Object} Returns the converted plain object.
         * @example
         *
         * function Foo() {
         *   this.b = 2;
         * }
         *
         * Foo.prototype.c = 3;
         *
         * _.assign({ 'a': 1 }, new Foo);
         * // => { 'a': 1, 'b': 2 }
         *
         * _.assign({ 'a': 1 }, _.toPlainObject(new Foo));
         * // => { 'a': 1, 'b': 2, 'c': 3 }
         */
        function toPlainObject(value) {
          return copyObject(value, keysIn(value));
        }

        /**
         * Converts `value` to a safe integer. A safe integer can be compared and
         * represented correctly.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Lang
         * @param {*} value The value to convert.
         * @returns {number} Returns the converted integer.
         * @example
         *
         * _.toSafeInteger(3.2);
         * // => 3
         *
         * _.toSafeInteger(Number.MIN_VALUE);
         * // => 0
         *
         * _.toSafeInteger(Infinity);
         * // => 9007199254740991
         *
         * _.toSafeInteger('3.2');
         * // => 3
         */
        function toSafeInteger(value) {
          return value
            ? baseClamp(toInteger(value), -MAX_SAFE_INTEGER, MAX_SAFE_INTEGER)
            : (value === 0 ? value : 0);
        }

        /**
         * Converts `value` to a string. An empty string is returned for `null`
         * and `undefined` values. The sign of `-0` is preserved.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Lang
         * @param {*} value The value to convert.
         * @returns {string} Returns the converted string.
         * @example
         *
         * _.toString(null);
         * // => ''
         *
         * _.toString(-0);
         * // => '-0'
         *
         * _.toString([1, 2, 3]);
         * // => '1,2,3'
         */
        function toString(value) {
          return value == null ? '' : baseToString(value);
        }

        /*------------------------------------------------------------------------*/

        /**
         * Assigns own enumerable string keyed properties of source objects to the
         * destination object. Source objects are applied from left to right.
         * Subsequent sources overwrite property assignments of previous sources.
         *
         * **Note:** This method mutates `object` and is loosely based on
         * [`Object.assign`](https://mdn.io/Object/assign).
         *
         * @static
         * @memberOf _
         * @since 0.10.0
         * @category Object
         * @param {Object} object The destination object.
         * @param {...Object} [sources] The source objects.
         * @returns {Object} Returns `object`.
         * @see _.assignIn
         * @example
         *
         * function Foo() {
         *   this.a = 1;
         * }
         *
         * function Bar() {
         *   this.c = 3;
         * }
         *
         * Foo.prototype.b = 2;
         * Bar.prototype.d = 4;
         *
         * _.assign({ 'a': 0 }, new Foo, new Bar);
         * // => { 'a': 1, 'c': 3 }
         */
        var assign = createAssigner(function(object, source) {
          if (isPrototype(source) || isArrayLike(source)) {
            copyObject(source, keys(source), object);
            return;
          }
          for (var key in source) {
            if (hasOwnProperty.call(source, key)) {
              assignValue(object, key, source[key]);
            }
          }
        });

        /**
         * This method is like `_.assign` except that it iterates over own and
         * inherited source properties.
         *
         * **Note:** This method mutates `object`.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @alias extend
         * @category Object
         * @param {Object} object The destination object.
         * @param {...Object} [sources] The source objects.
         * @returns {Object} Returns `object`.
         * @see _.assign
         * @example
         *
         * function Foo() {
         *   this.a = 1;
         * }
         *
         * function Bar() {
         *   this.c = 3;
         * }
         *
         * Foo.prototype.b = 2;
         * Bar.prototype.d = 4;
         *
         * _.assignIn({ 'a': 0 }, new Foo, new Bar);
         * // => { 'a': 1, 'b': 2, 'c': 3, 'd': 4 }
         */
        var assignIn = createAssigner(function(object, source) {
          copyObject(source, keysIn(source), object);
        });

        /**
         * This method is like `_.assignIn` except that it accepts `customizer`
         * which is invoked to produce the assigned values. If `customizer` returns
         * `undefined`, assignment is handled by the method instead. The `customizer`
         * is invoked with five arguments: (objValue, srcValue, key, object, source).
         *
         * **Note:** This method mutates `object`.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @alias extendWith
         * @category Object
         * @param {Object} object The destination object.
         * @param {...Object} sources The source objects.
         * @param {Function} [customizer] The function to customize assigned values.
         * @returns {Object} Returns `object`.
         * @see _.assignWith
         * @example
         *
         * function customizer(objValue, srcValue) {
         *   return _.isUndefined(objValue) ? srcValue : objValue;
         * }
         *
         * var defaults = _.partialRight(_.assignInWith, customizer);
         *
         * defaults({ 'a': 1 }, { 'b': 2 }, { 'a': 3 });
         * // => { 'a': 1, 'b': 2 }
         */
        var assignInWith = createAssigner(function(object, source, srcIndex, customizer) {
          copyObject(source, keysIn(source), object, customizer);
        });

        /**
         * This method is like `_.assign` except that it accepts `customizer`
         * which is invoked to produce the assigned values. If `customizer` returns
         * `undefined`, assignment is handled by the method instead. The `customizer`
         * is invoked with five arguments: (objValue, srcValue, key, object, source).
         *
         * **Note:** This method mutates `object`.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Object
         * @param {Object} object The destination object.
         * @param {...Object} sources The source objects.
         * @param {Function} [customizer] The function to customize assigned values.
         * @returns {Object} Returns `object`.
         * @see _.assignInWith
         * @example
         *
         * function customizer(objValue, srcValue) {
         *   return _.isUndefined(objValue) ? srcValue : objValue;
         * }
         *
         * var defaults = _.partialRight(_.assignWith, customizer);
         *
         * defaults({ 'a': 1 }, { 'b': 2 }, { 'a': 3 });
         * // => { 'a': 1, 'b': 2 }
         */
        var assignWith = createAssigner(function(object, source, srcIndex, customizer) {
          copyObject(source, keys(source), object, customizer);
        });

        /**
         * Creates an array of values corresponding to `paths` of `object`.
         *
         * @static
         * @memberOf _
         * @since 1.0.0
         * @category Object
         * @param {Object} object The object to iterate over.
         * @param {...(string|string[])} [paths] The property paths to pick.
         * @returns {Array} Returns the picked values.
         * @example
         *
         * var object = { 'a': [{ 'b': { 'c': 3 } }, 4] };
         *
         * _.at(object, ['a[0].b.c', 'a[1]']);
         * // => [3, 4]
         */
        var at = flatRest(baseAt);

        /**
         * Creates an object that inherits from the `prototype` object. If a
         * `properties` object is given, its own enumerable string keyed properties
         * are assigned to the created object.
         *
         * @static
         * @memberOf _
         * @since 2.3.0
         * @category Object
         * @param {Object} prototype The object to inherit from.
         * @param {Object} [properties] The properties to assign to the object.
         * @returns {Object} Returns the new object.
         * @example
         *
         * function Shape() {
         *   this.x = 0;
         *   this.y = 0;
         * }
         *
         * function Circle() {
         *   Shape.call(this);
         * }
         *
         * Circle.prototype = _.create(Shape.prototype, {
         *   'constructor': Circle
         * });
         *
         * var circle = new Circle;
         * circle instanceof Circle;
         * // => true
         *
         * circle instanceof Shape;
         * // => true
         */
        function create(prototype, properties) {
          var result = baseCreate(prototype);
          return properties == null ? result : baseAssign(result, properties);
        }

        /**
         * Assigns own and inherited enumerable string keyed properties of source
         * objects to the destination object for all destination properties that
         * resolve to `undefined`. Source objects are applied from left to right.
         * Once a property is set, additional values of the same property are ignored.
         *
         * **Note:** This method mutates `object`.
         *
         * @static
         * @since 0.1.0
         * @memberOf _
         * @category Object
         * @param {Object} object The destination object.
         * @param {...Object} [sources] The source objects.
         * @returns {Object} Returns `object`.
         * @see _.defaultsDeep
         * @example
         *
         * _.defaults({ 'a': 1 }, { 'b': 2 }, { 'a': 3 });
         * // => { 'a': 1, 'b': 2 }
         */
        var defaults = baseRest(function(object, sources) {
          object = Object(object);

          var index = -1;
          var length = sources.length;
          var guard = length > 2 ? sources[2] : undefined$1;

          if (guard && isIterateeCall(sources[0], sources[1], guard)) {
            length = 1;
          }

          while (++index < length) {
            var source = sources[index];
            var props = keysIn(source);
            var propsIndex = -1;
            var propsLength = props.length;

            while (++propsIndex < propsLength) {
              var key = props[propsIndex];
              var value = object[key];

              if (value === undefined$1 ||
                  (eq(value, objectProto[key]) && !hasOwnProperty.call(object, key))) {
                object[key] = source[key];
              }
            }
          }

          return object;
        });

        /**
         * This method is like `_.defaults` except that it recursively assigns
         * default properties.
         *
         * **Note:** This method mutates `object`.
         *
         * @static
         * @memberOf _
         * @since 3.10.0
         * @category Object
         * @param {Object} object The destination object.
         * @param {...Object} [sources] The source objects.
         * @returns {Object} Returns `object`.
         * @see _.defaults
         * @example
         *
         * _.defaultsDeep({ 'a': { 'b': 2 } }, { 'a': { 'b': 1, 'c': 3 } });
         * // => { 'a': { 'b': 2, 'c': 3 } }
         */
        var defaultsDeep = baseRest(function(args) {
          args.push(undefined$1, customDefaultsMerge);
          return apply(mergeWith, undefined$1, args);
        });

        /**
         * This method is like `_.find` except that it returns the key of the first
         * element `predicate` returns truthy for instead of the element itself.
         *
         * @static
         * @memberOf _
         * @since 1.1.0
         * @category Object
         * @param {Object} object The object to inspect.
         * @param {Function} [predicate=_.identity] The function invoked per iteration.
         * @returns {string|undefined} Returns the key of the matched element,
         *  else `undefined`.
         * @example
         *
         * var users = {
         *   'barney':  { 'age': 36, 'active': true },
         *   'fred':    { 'age': 40, 'active': false },
         *   'pebbles': { 'age': 1,  'active': true }
         * };
         *
         * _.findKey(users, function(o) { return o.age < 40; });
         * // => 'barney' (iteration order is not guaranteed)
         *
         * // The `_.matches` iteratee shorthand.
         * _.findKey(users, { 'age': 1, 'active': true });
         * // => 'pebbles'
         *
         * // The `_.matchesProperty` iteratee shorthand.
         * _.findKey(users, ['active', false]);
         * // => 'fred'
         *
         * // The `_.property` iteratee shorthand.
         * _.findKey(users, 'active');
         * // => 'barney'
         */
        function findKey(object, predicate) {
          return baseFindKey(object, getIteratee(predicate, 3), baseForOwn);
        }

        /**
         * This method is like `_.findKey` except that it iterates over elements of
         * a collection in the opposite order.
         *
         * @static
         * @memberOf _
         * @since 2.0.0
         * @category Object
         * @param {Object} object The object to inspect.
         * @param {Function} [predicate=_.identity] The function invoked per iteration.
         * @returns {string|undefined} Returns the key of the matched element,
         *  else `undefined`.
         * @example
         *
         * var users = {
         *   'barney':  { 'age': 36, 'active': true },
         *   'fred':    { 'age': 40, 'active': false },
         *   'pebbles': { 'age': 1,  'active': true }
         * };
         *
         * _.findLastKey(users, function(o) { return o.age < 40; });
         * // => returns 'pebbles' assuming `_.findKey` returns 'barney'
         *
         * // The `_.matches` iteratee shorthand.
         * _.findLastKey(users, { 'age': 36, 'active': true });
         * // => 'barney'
         *
         * // The `_.matchesProperty` iteratee shorthand.
         * _.findLastKey(users, ['active', false]);
         * // => 'fred'
         *
         * // The `_.property` iteratee shorthand.
         * _.findLastKey(users, 'active');
         * // => 'pebbles'
         */
        function findLastKey(object, predicate) {
          return baseFindKey(object, getIteratee(predicate, 3), baseForOwnRight);
        }

        /**
         * Iterates over own and inherited enumerable string keyed properties of an
         * object and invokes `iteratee` for each property. The iteratee is invoked
         * with three arguments: (value, key, object). Iteratee functions may exit
         * iteration early by explicitly returning `false`.
         *
         * @static
         * @memberOf _
         * @since 0.3.0
         * @category Object
         * @param {Object} object The object to iterate over.
         * @param {Function} [iteratee=_.identity] The function invoked per iteration.
         * @returns {Object} Returns `object`.
         * @see _.forInRight
         * @example
         *
         * function Foo() {
         *   this.a = 1;
         *   this.b = 2;
         * }
         *
         * Foo.prototype.c = 3;
         *
         * _.forIn(new Foo, function(value, key) {
         *   console.log(key);
         * });
         * // => Logs 'a', 'b', then 'c' (iteration order is not guaranteed).
         */
        function forIn(object, iteratee) {
          return object == null
            ? object
            : baseFor(object, getIteratee(iteratee, 3), keysIn);
        }

        /**
         * This method is like `_.forIn` except that it iterates over properties of
         * `object` in the opposite order.
         *
         * @static
         * @memberOf _
         * @since 2.0.0
         * @category Object
         * @param {Object} object The object to iterate over.
         * @param {Function} [iteratee=_.identity] The function invoked per iteration.
         * @returns {Object} Returns `object`.
         * @see _.forIn
         * @example
         *
         * function Foo() {
         *   this.a = 1;
         *   this.b = 2;
         * }
         *
         * Foo.prototype.c = 3;
         *
         * _.forInRight(new Foo, function(value, key) {
         *   console.log(key);
         * });
         * // => Logs 'c', 'b', then 'a' assuming `_.forIn` logs 'a', 'b', then 'c'.
         */
        function forInRight(object, iteratee) {
          return object == null
            ? object
            : baseForRight(object, getIteratee(iteratee, 3), keysIn);
        }

        /**
         * Iterates over own enumerable string keyed properties of an object and
         * invokes `iteratee` for each property. The iteratee is invoked with three
         * arguments: (value, key, object). Iteratee functions may exit iteration
         * early by explicitly returning `false`.
         *
         * @static
         * @memberOf _
         * @since 0.3.0
         * @category Object
         * @param {Object} object The object to iterate over.
         * @param {Function} [iteratee=_.identity] The function invoked per iteration.
         * @returns {Object} Returns `object`.
         * @see _.forOwnRight
         * @example
         *
         * function Foo() {
         *   this.a = 1;
         *   this.b = 2;
         * }
         *
         * Foo.prototype.c = 3;
         *
         * _.forOwn(new Foo, function(value, key) {
         *   console.log(key);
         * });
         * // => Logs 'a' then 'b' (iteration order is not guaranteed).
         */
        function forOwn(object, iteratee) {
          return object && baseForOwn(object, getIteratee(iteratee, 3));
        }

        /**
         * This method is like `_.forOwn` except that it iterates over properties of
         * `object` in the opposite order.
         *
         * @static
         * @memberOf _
         * @since 2.0.0
         * @category Object
         * @param {Object} object The object to iterate over.
         * @param {Function} [iteratee=_.identity] The function invoked per iteration.
         * @returns {Object} Returns `object`.
         * @see _.forOwn
         * @example
         *
         * function Foo() {
         *   this.a = 1;
         *   this.b = 2;
         * }
         *
         * Foo.prototype.c = 3;
         *
         * _.forOwnRight(new Foo, function(value, key) {
         *   console.log(key);
         * });
         * // => Logs 'b' then 'a' assuming `_.forOwn` logs 'a' then 'b'.
         */
        function forOwnRight(object, iteratee) {
          return object && baseForOwnRight(object, getIteratee(iteratee, 3));
        }

        /**
         * Creates an array of function property names from own enumerable properties
         * of `object`.
         *
         * @static
         * @since 0.1.0
         * @memberOf _
         * @category Object
         * @param {Object} object The object to inspect.
         * @returns {Array} Returns the function names.
         * @see _.functionsIn
         * @example
         *
         * function Foo() {
         *   this.a = _.constant('a');
         *   this.b = _.constant('b');
         * }
         *
         * Foo.prototype.c = _.constant('c');
         *
         * _.functions(new Foo);
         * // => ['a', 'b']
         */
        function functions(object) {
          return object == null ? [] : baseFunctions(object, keys(object));
        }

        /**
         * Creates an array of function property names from own and inherited
         * enumerable properties of `object`.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Object
         * @param {Object} object The object to inspect.
         * @returns {Array} Returns the function names.
         * @see _.functions
         * @example
         *
         * function Foo() {
         *   this.a = _.constant('a');
         *   this.b = _.constant('b');
         * }
         *
         * Foo.prototype.c = _.constant('c');
         *
         * _.functionsIn(new Foo);
         * // => ['a', 'b', 'c']
         */
        function functionsIn(object) {
          return object == null ? [] : baseFunctions(object, keysIn(object));
        }

        /**
         * Gets the value at `path` of `object`. If the resolved value is
         * `undefined`, the `defaultValue` is returned in its place.
         *
         * @static
         * @memberOf _
         * @since 3.7.0
         * @category Object
         * @param {Object} object The object to query.
         * @param {Array|string} path The path of the property to get.
         * @param {*} [defaultValue] The value returned for `undefined` resolved values.
         * @returns {*} Returns the resolved value.
         * @example
         *
         * var object = { 'a': [{ 'b': { 'c': 3 } }] };
         *
         * _.get(object, 'a[0].b.c');
         * // => 3
         *
         * _.get(object, ['a', '0', 'b', 'c']);
         * // => 3
         *
         * _.get(object, 'a.b.c', 'default');
         * // => 'default'
         */
        function get(object, path, defaultValue) {
          var result = object == null ? undefined$1 : baseGet(object, path);
          return result === undefined$1 ? defaultValue : result;
        }

        /**
         * Checks if `path` is a direct property of `object`.
         *
         * @static
         * @since 0.1.0
         * @memberOf _
         * @category Object
         * @param {Object} object The object to query.
         * @param {Array|string} path The path to check.
         * @returns {boolean} Returns `true` if `path` exists, else `false`.
         * @example
         *
         * var object = { 'a': { 'b': 2 } };
         * var other = _.create({ 'a': _.create({ 'b': 2 }) });
         *
         * _.has(object, 'a');
         * // => true
         *
         * _.has(object, 'a.b');
         * // => true
         *
         * _.has(object, ['a', 'b']);
         * // => true
         *
         * _.has(other, 'a');
         * // => false
         */
        function has(object, path) {
          return object != null && hasPath(object, path, baseHas);
        }

        /**
         * Checks if `path` is a direct or inherited property of `object`.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Object
         * @param {Object} object The object to query.
         * @param {Array|string} path The path to check.
         * @returns {boolean} Returns `true` if `path` exists, else `false`.
         * @example
         *
         * var object = _.create({ 'a': _.create({ 'b': 2 }) });
         *
         * _.hasIn(object, 'a');
         * // => true
         *
         * _.hasIn(object, 'a.b');
         * // => true
         *
         * _.hasIn(object, ['a', 'b']);
         * // => true
         *
         * _.hasIn(object, 'b');
         * // => false
         */
        function hasIn(object, path) {
          return object != null && hasPath(object, path, baseHasIn);
        }

        /**
         * Creates an object composed of the inverted keys and values of `object`.
         * If `object` contains duplicate values, subsequent values overwrite
         * property assignments of previous values.
         *
         * @static
         * @memberOf _
         * @since 0.7.0
         * @category Object
         * @param {Object} object The object to invert.
         * @returns {Object} Returns the new inverted object.
         * @example
         *
         * var object = { 'a': 1, 'b': 2, 'c': 1 };
         *
         * _.invert(object);
         * // => { '1': 'c', '2': 'b' }
         */
        var invert = createInverter(function(result, value, key) {
          if (value != null &&
              typeof value.toString != 'function') {
            value = nativeObjectToString.call(value);
          }

          result[value] = key;
        }, constant(identity));

        /**
         * This method is like `_.invert` except that the inverted object is generated
         * from the results of running each element of `object` thru `iteratee`. The
         * corresponding inverted value of each inverted key is an array of keys
         * responsible for generating the inverted value. The iteratee is invoked
         * with one argument: (value).
         *
         * @static
         * @memberOf _
         * @since 4.1.0
         * @category Object
         * @param {Object} object The object to invert.
         * @param {Function} [iteratee=_.identity] The iteratee invoked per element.
         * @returns {Object} Returns the new inverted object.
         * @example
         *
         * var object = { 'a': 1, 'b': 2, 'c': 1 };
         *
         * _.invertBy(object);
         * // => { '1': ['a', 'c'], '2': ['b'] }
         *
         * _.invertBy(object, function(value) {
         *   return 'group' + value;
         * });
         * // => { 'group1': ['a', 'c'], 'group2': ['b'] }
         */
        var invertBy = createInverter(function(result, value, key) {
          if (value != null &&
              typeof value.toString != 'function') {
            value = nativeObjectToString.call(value);
          }

          if (hasOwnProperty.call(result, value)) {
            result[value].push(key);
          } else {
            result[value] = [key];
          }
        }, getIteratee);

        /**
         * Invokes the method at `path` of `object`.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Object
         * @param {Object} object The object to query.
         * @param {Array|string} path The path of the method to invoke.
         * @param {...*} [args] The arguments to invoke the method with.
         * @returns {*} Returns the result of the invoked method.
         * @example
         *
         * var object = { 'a': [{ 'b': { 'c': [1, 2, 3, 4] } }] };
         *
         * _.invoke(object, 'a[0].b.c.slice', 1, 3);
         * // => [2, 3]
         */
        var invoke = baseRest(baseInvoke);

        /**
         * Creates an array of the own enumerable property names of `object`.
         *
         * **Note:** Non-object values are coerced to objects. See the
         * [ES spec](http://ecma-international.org/ecma-262/7.0/#sec-object.keys)
         * for more details.
         *
         * @static
         * @since 0.1.0
         * @memberOf _
         * @category Object
         * @param {Object} object The object to query.
         * @returns {Array} Returns the array of property names.
         * @example
         *
         * function Foo() {
         *   this.a = 1;
         *   this.b = 2;
         * }
         *
         * Foo.prototype.c = 3;
         *
         * _.keys(new Foo);
         * // => ['a', 'b'] (iteration order is not guaranteed)
         *
         * _.keys('hi');
         * // => ['0', '1']
         */
        function keys(object) {
          return isArrayLike(object) ? arrayLikeKeys(object) : baseKeys(object);
        }

        /**
         * Creates an array of the own and inherited enumerable property names of `object`.
         *
         * **Note:** Non-object values are coerced to objects.
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category Object
         * @param {Object} object The object to query.
         * @returns {Array} Returns the array of property names.
         * @example
         *
         * function Foo() {
         *   this.a = 1;
         *   this.b = 2;
         * }
         *
         * Foo.prototype.c = 3;
         *
         * _.keysIn(new Foo);
         * // => ['a', 'b', 'c'] (iteration order is not guaranteed)
         */
        function keysIn(object) {
          return isArrayLike(object) ? arrayLikeKeys(object, true) : baseKeysIn(object);
        }

        /**
         * The opposite of `_.mapValues`; this method creates an object with the
         * same values as `object` and keys generated by running each own enumerable
         * string keyed property of `object` thru `iteratee`. The iteratee is invoked
         * with three arguments: (value, key, object).
         *
         * @static
         * @memberOf _
         * @since 3.8.0
         * @category Object
         * @param {Object} object The object to iterate over.
         * @param {Function} [iteratee=_.identity] The function invoked per iteration.
         * @returns {Object} Returns the new mapped object.
         * @see _.mapValues
         * @example
         *
         * _.mapKeys({ 'a': 1, 'b': 2 }, function(value, key) {
         *   return key + value;
         * });
         * // => { 'a1': 1, 'b2': 2 }
         */
        function mapKeys(object, iteratee) {
          var result = {};
          iteratee = getIteratee(iteratee, 3);

          baseForOwn(object, function(value, key, object) {
            baseAssignValue(result, iteratee(value, key, object), value);
          });
          return result;
        }

        /**
         * Creates an object with the same keys as `object` and values generated
         * by running each own enumerable string keyed property of `object` thru
         * `iteratee`. The iteratee is invoked with three arguments:
         * (value, key, object).
         *
         * @static
         * @memberOf _
         * @since 2.4.0
         * @category Object
         * @param {Object} object The object to iterate over.
         * @param {Function} [iteratee=_.identity] The function invoked per iteration.
         * @returns {Object} Returns the new mapped object.
         * @see _.mapKeys
         * @example
         *
         * var users = {
         *   'fred':    { 'user': 'fred',    'age': 40 },
         *   'pebbles': { 'user': 'pebbles', 'age': 1 }
         * };
         *
         * _.mapValues(users, function(o) { return o.age; });
         * // => { 'fred': 40, 'pebbles': 1 } (iteration order is not guaranteed)
         *
         * // The `_.property` iteratee shorthand.
         * _.mapValues(users, 'age');
         * // => { 'fred': 40, 'pebbles': 1 } (iteration order is not guaranteed)
         */
        function mapValues(object, iteratee) {
          var result = {};
          iteratee = getIteratee(iteratee, 3);

          baseForOwn(object, function(value, key, object) {
            baseAssignValue(result, key, iteratee(value, key, object));
          });
          return result;
        }

        /**
         * This method is like `_.assign` except that it recursively merges own and
         * inherited enumerable string keyed properties of source objects into the
         * destination object. Source properties that resolve to `undefined` are
         * skipped if a destination value exists. Array and plain object properties
         * are merged recursively. Other objects and value types are overridden by
         * assignment. Source objects are applied from left to right. Subsequent
         * sources overwrite property assignments of previous sources.
         *
         * **Note:** This method mutates `object`.
         *
         * @static
         * @memberOf _
         * @since 0.5.0
         * @category Object
         * @param {Object} object The destination object.
         * @param {...Object} [sources] The source objects.
         * @returns {Object} Returns `object`.
         * @example
         *
         * var object = {
         *   'a': [{ 'b': 2 }, { 'd': 4 }]
         * };
         *
         * var other = {
         *   'a': [{ 'c': 3 }, { 'e': 5 }]
         * };
         *
         * _.merge(object, other);
         * // => { 'a': [{ 'b': 2, 'c': 3 }, { 'd': 4, 'e': 5 }] }
         */
        var merge = createAssigner(function(object, source, srcIndex) {
          baseMerge(object, source, srcIndex);
        });

        /**
         * This method is like `_.merge` except that it accepts `customizer` which
         * is invoked to produce the merged values of the destination and source
         * properties. If `customizer` returns `undefined`, merging is handled by the
         * method instead. The `customizer` is invoked with six arguments:
         * (objValue, srcValue, key, object, source, stack).
         *
         * **Note:** This method mutates `object`.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Object
         * @param {Object} object The destination object.
         * @param {...Object} sources The source objects.
         * @param {Function} customizer The function to customize assigned values.
         * @returns {Object} Returns `object`.
         * @example
         *
         * function customizer(objValue, srcValue) {
         *   if (_.isArray(objValue)) {
         *     return objValue.concat(srcValue);
         *   }
         * }
         *
         * var object = { 'a': [1], 'b': [2] };
         * var other = { 'a': [3], 'b': [4] };
         *
         * _.mergeWith(object, other, customizer);
         * // => { 'a': [1, 3], 'b': [2, 4] }
         */
        var mergeWith = createAssigner(function(object, source, srcIndex, customizer) {
          baseMerge(object, source, srcIndex, customizer);
        });

        /**
         * The opposite of `_.pick`; this method creates an object composed of the
         * own and inherited enumerable property paths of `object` that are not omitted.
         *
         * **Note:** This method is considerably slower than `_.pick`.
         *
         * @static
         * @since 0.1.0
         * @memberOf _
         * @category Object
         * @param {Object} object The source object.
         * @param {...(string|string[])} [paths] The property paths to omit.
         * @returns {Object} Returns the new object.
         * @example
         *
         * var object = { 'a': 1, 'b': '2', 'c': 3 };
         *
         * _.omit(object, ['a', 'c']);
         * // => { 'b': '2' }
         */
        var omit = flatRest(function(object, paths) {
          var result = {};
          if (object == null) {
            return result;
          }
          var isDeep = false;
          paths = arrayMap(paths, function(path) {
            path = castPath(path, object);
            isDeep || (isDeep = path.length > 1);
            return path;
          });
          copyObject(object, getAllKeysIn(object), result);
          if (isDeep) {
            result = baseClone(result, CLONE_DEEP_FLAG | CLONE_FLAT_FLAG | CLONE_SYMBOLS_FLAG, customOmitClone);
          }
          var length = paths.length;
          while (length--) {
            baseUnset(result, paths[length]);
          }
          return result;
        });

        /**
         * The opposite of `_.pickBy`; this method creates an object composed of
         * the own and inherited enumerable string keyed properties of `object` that
         * `predicate` doesn't return truthy for. The predicate is invoked with two
         * arguments: (value, key).
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Object
         * @param {Object} object The source object.
         * @param {Function} [predicate=_.identity] The function invoked per property.
         * @returns {Object} Returns the new object.
         * @example
         *
         * var object = { 'a': 1, 'b': '2', 'c': 3 };
         *
         * _.omitBy(object, _.isNumber);
         * // => { 'b': '2' }
         */
        function omitBy(object, predicate) {
          return pickBy(object, negate(getIteratee(predicate)));
        }

        /**
         * Creates an object composed of the picked `object` properties.
         *
         * @static
         * @since 0.1.0
         * @memberOf _
         * @category Object
         * @param {Object} object The source object.
         * @param {...(string|string[])} [paths] The property paths to pick.
         * @returns {Object} Returns the new object.
         * @example
         *
         * var object = { 'a': 1, 'b': '2', 'c': 3 };
         *
         * _.pick(object, ['a', 'c']);
         * // => { 'a': 1, 'c': 3 }
         */
        var pick = flatRest(function(object, paths) {
          return object == null ? {} : basePick(object, paths);
        });

        /**
         * Creates an object composed of the `object` properties `predicate` returns
         * truthy for. The predicate is invoked with two arguments: (value, key).
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Object
         * @param {Object} object The source object.
         * @param {Function} [predicate=_.identity] The function invoked per property.
         * @returns {Object} Returns the new object.
         * @example
         *
         * var object = { 'a': 1, 'b': '2', 'c': 3 };
         *
         * _.pickBy(object, _.isNumber);
         * // => { 'a': 1, 'c': 3 }
         */
        function pickBy(object, predicate) {
          if (object == null) {
            return {};
          }
          var props = arrayMap(getAllKeysIn(object), function(prop) {
            return [prop];
          });
          predicate = getIteratee(predicate);
          return basePickBy(object, props, function(value, path) {
            return predicate(value, path[0]);
          });
        }

        /**
         * This method is like `_.get` except that if the resolved value is a
         * function it's invoked with the `this` binding of its parent object and
         * its result is returned.
         *
         * @static
         * @since 0.1.0
         * @memberOf _
         * @category Object
         * @param {Object} object The object to query.
         * @param {Array|string} path The path of the property to resolve.
         * @param {*} [defaultValue] The value returned for `undefined` resolved values.
         * @returns {*} Returns the resolved value.
         * @example
         *
         * var object = { 'a': [{ 'b': { 'c1': 3, 'c2': _.constant(4) } }] };
         *
         * _.result(object, 'a[0].b.c1');
         * // => 3
         *
         * _.result(object, 'a[0].b.c2');
         * // => 4
         *
         * _.result(object, 'a[0].b.c3', 'default');
         * // => 'default'
         *
         * _.result(object, 'a[0].b.c3', _.constant('default'));
         * // => 'default'
         */
        function result(object, path, defaultValue) {
          path = castPath(path, object);

          var index = -1,
              length = path.length;

          // Ensure the loop is entered when path is empty.
          if (!length) {
            length = 1;
            object = undefined$1;
          }
          while (++index < length) {
            var value = object == null ? undefined$1 : object[toKey(path[index])];
            if (value === undefined$1) {
              index = length;
              value = defaultValue;
            }
            object = isFunction(value) ? value.call(object) : value;
          }
          return object;
        }

        /**
         * Sets the value at `path` of `object`. If a portion of `path` doesn't exist,
         * it's created. Arrays are created for missing index properties while objects
         * are created for all other missing properties. Use `_.setWith` to customize
         * `path` creation.
         *
         * **Note:** This method mutates `object`.
         *
         * @static
         * @memberOf _
         * @since 3.7.0
         * @category Object
         * @param {Object} object The object to modify.
         * @param {Array|string} path The path of the property to set.
         * @param {*} value The value to set.
         * @returns {Object} Returns `object`.
         * @example
         *
         * var object = { 'a': [{ 'b': { 'c': 3 } }] };
         *
         * _.set(object, 'a[0].b.c', 4);
         * console.log(object.a[0].b.c);
         * // => 4
         *
         * _.set(object, ['x', '0', 'y', 'z'], 5);
         * console.log(object.x[0].y.z);
         * // => 5
         */
        function set(object, path, value) {
          return object == null ? object : baseSet(object, path, value);
        }

        /**
         * This method is like `_.set` except that it accepts `customizer` which is
         * invoked to produce the objects of `path`.  If `customizer` returns `undefined`
         * path creation is handled by the method instead. The `customizer` is invoked
         * with three arguments: (nsValue, key, nsObject).
         *
         * **Note:** This method mutates `object`.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Object
         * @param {Object} object The object to modify.
         * @param {Array|string} path The path of the property to set.
         * @param {*} value The value to set.
         * @param {Function} [customizer] The function to customize assigned values.
         * @returns {Object} Returns `object`.
         * @example
         *
         * var object = {};
         *
         * _.setWith(object, '[0][1]', 'a', Object);
         * // => { '0': { '1': 'a' } }
         */
        function setWith(object, path, value, customizer) {
          customizer = typeof customizer == 'function' ? customizer : undefined$1;
          return object == null ? object : baseSet(object, path, value, customizer);
        }

        /**
         * Creates an array of own enumerable string keyed-value pairs for `object`
         * which can be consumed by `_.fromPairs`. If `object` is a map or set, its
         * entries are returned.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @alias entries
         * @category Object
         * @param {Object} object The object to query.
         * @returns {Array} Returns the key-value pairs.
         * @example
         *
         * function Foo() {
         *   this.a = 1;
         *   this.b = 2;
         * }
         *
         * Foo.prototype.c = 3;
         *
         * _.toPairs(new Foo);
         * // => [['a', 1], ['b', 2]] (iteration order is not guaranteed)
         */
        var toPairs = createToPairs(keys);

        /**
         * Creates an array of own and inherited enumerable string keyed-value pairs
         * for `object` which can be consumed by `_.fromPairs`. If `object` is a map
         * or set, its entries are returned.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @alias entriesIn
         * @category Object
         * @param {Object} object The object to query.
         * @returns {Array} Returns the key-value pairs.
         * @example
         *
         * function Foo() {
         *   this.a = 1;
         *   this.b = 2;
         * }
         *
         * Foo.prototype.c = 3;
         *
         * _.toPairsIn(new Foo);
         * // => [['a', 1], ['b', 2], ['c', 3]] (iteration order is not guaranteed)
         */
        var toPairsIn = createToPairs(keysIn);

        /**
         * An alternative to `_.reduce`; this method transforms `object` to a new
         * `accumulator` object which is the result of running each of its own
         * enumerable string keyed properties thru `iteratee`, with each invocation
         * potentially mutating the `accumulator` object. If `accumulator` is not
         * provided, a new object with the same `[[Prototype]]` will be used. The
         * iteratee is invoked with four arguments: (accumulator, value, key, object).
         * Iteratee functions may exit iteration early by explicitly returning `false`.
         *
         * @static
         * @memberOf _
         * @since 1.3.0
         * @category Object
         * @param {Object} object The object to iterate over.
         * @param {Function} [iteratee=_.identity] The function invoked per iteration.
         * @param {*} [accumulator] The custom accumulator value.
         * @returns {*} Returns the accumulated value.
         * @example
         *
         * _.transform([2, 3, 4], function(result, n) {
         *   result.push(n *= n);
         *   return n % 2 == 0;
         * }, []);
         * // => [4, 9]
         *
         * _.transform({ 'a': 1, 'b': 2, 'c': 1 }, function(result, value, key) {
         *   (result[value] || (result[value] = [])).push(key);
         * }, {});
         * // => { '1': ['a', 'c'], '2': ['b'] }
         */
        function transform(object, iteratee, accumulator) {
          var isArr = isArray(object),
              isArrLike = isArr || isBuffer(object) || isTypedArray(object);

          iteratee = getIteratee(iteratee, 4);
          if (accumulator == null) {
            var Ctor = object && object.constructor;
            if (isArrLike) {
              accumulator = isArr ? new Ctor : [];
            }
            else if (isObject(object)) {
              accumulator = isFunction(Ctor) ? baseCreate(getPrototype(object)) : {};
            }
            else {
              accumulator = {};
            }
          }
          (isArrLike ? arrayEach : baseForOwn)(object, function(value, index, object) {
            return iteratee(accumulator, value, index, object);
          });
          return accumulator;
        }

        /**
         * Removes the property at `path` of `object`.
         *
         * **Note:** This method mutates `object`.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Object
         * @param {Object} object The object to modify.
         * @param {Array|string} path The path of the property to unset.
         * @returns {boolean} Returns `true` if the property is deleted, else `false`.
         * @example
         *
         * var object = { 'a': [{ 'b': { 'c': 7 } }] };
         * _.unset(object, 'a[0].b.c');
         * // => true
         *
         * console.log(object);
         * // => { 'a': [{ 'b': {} }] };
         *
         * _.unset(object, ['a', '0', 'b', 'c']);
         * // => true
         *
         * console.log(object);
         * // => { 'a': [{ 'b': {} }] };
         */
        function unset(object, path) {
          return object == null ? true : baseUnset(object, path);
        }

        /**
         * This method is like `_.set` except that accepts `updater` to produce the
         * value to set. Use `_.updateWith` to customize `path` creation. The `updater`
         * is invoked with one argument: (value).
         *
         * **Note:** This method mutates `object`.
         *
         * @static
         * @memberOf _
         * @since 4.6.0
         * @category Object
         * @param {Object} object The object to modify.
         * @param {Array|string} path The path of the property to set.
         * @param {Function} updater The function to produce the updated value.
         * @returns {Object} Returns `object`.
         * @example
         *
         * var object = { 'a': [{ 'b': { 'c': 3 } }] };
         *
         * _.update(object, 'a[0].b.c', function(n) { return n * n; });
         * console.log(object.a[0].b.c);
         * // => 9
         *
         * _.update(object, 'x[0].y.z', function(n) { return n ? n + 1 : 0; });
         * console.log(object.x[0].y.z);
         * // => 0
         */
        function update(object, path, updater) {
          return object == null ? object : baseUpdate(object, path, castFunction(updater));
        }

        /**
         * This method is like `_.update` except that it accepts `customizer` which is
         * invoked to produce the objects of `path`.  If `customizer` returns `undefined`
         * path creation is handled by the method instead. The `customizer` is invoked
         * with three arguments: (nsValue, key, nsObject).
         *
         * **Note:** This method mutates `object`.
         *
         * @static
         * @memberOf _
         * @since 4.6.0
         * @category Object
         * @param {Object} object The object to modify.
         * @param {Array|string} path The path of the property to set.
         * @param {Function} updater The function to produce the updated value.
         * @param {Function} [customizer] The function to customize assigned values.
         * @returns {Object} Returns `object`.
         * @example
         *
         * var object = {};
         *
         * _.updateWith(object, '[0][1]', _.constant('a'), Object);
         * // => { '0': { '1': 'a' } }
         */
        function updateWith(object, path, updater, customizer) {
          customizer = typeof customizer == 'function' ? customizer : undefined$1;
          return object == null ? object : baseUpdate(object, path, castFunction(updater), customizer);
        }

        /**
         * Creates an array of the own enumerable string keyed property values of `object`.
         *
         * **Note:** Non-object values are coerced to objects.
         *
         * @static
         * @since 0.1.0
         * @memberOf _
         * @category Object
         * @param {Object} object The object to query.
         * @returns {Array} Returns the array of property values.
         * @example
         *
         * function Foo() {
         *   this.a = 1;
         *   this.b = 2;
         * }
         *
         * Foo.prototype.c = 3;
         *
         * _.values(new Foo);
         * // => [1, 2] (iteration order is not guaranteed)
         *
         * _.values('hi');
         * // => ['h', 'i']
         */
        function values(object) {
          return object == null ? [] : baseValues(object, keys(object));
        }

        /**
         * Creates an array of the own and inherited enumerable string keyed property
         * values of `object`.
         *
         * **Note:** Non-object values are coerced to objects.
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category Object
         * @param {Object} object The object to query.
         * @returns {Array} Returns the array of property values.
         * @example
         *
         * function Foo() {
         *   this.a = 1;
         *   this.b = 2;
         * }
         *
         * Foo.prototype.c = 3;
         *
         * _.valuesIn(new Foo);
         * // => [1, 2, 3] (iteration order is not guaranteed)
         */
        function valuesIn(object) {
          return object == null ? [] : baseValues(object, keysIn(object));
        }

        /*------------------------------------------------------------------------*/

        /**
         * Clamps `number` within the inclusive `lower` and `upper` bounds.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Number
         * @param {number} number The number to clamp.
         * @param {number} [lower] The lower bound.
         * @param {number} upper The upper bound.
         * @returns {number} Returns the clamped number.
         * @example
         *
         * _.clamp(-10, -5, 5);
         * // => -5
         *
         * _.clamp(10, -5, 5);
         * // => 5
         */
        function clamp(number, lower, upper) {
          if (upper === undefined$1) {
            upper = lower;
            lower = undefined$1;
          }
          if (upper !== undefined$1) {
            upper = toNumber(upper);
            upper = upper === upper ? upper : 0;
          }
          if (lower !== undefined$1) {
            lower = toNumber(lower);
            lower = lower === lower ? lower : 0;
          }
          return baseClamp(toNumber(number), lower, upper);
        }

        /**
         * Checks if `n` is between `start` and up to, but not including, `end`. If
         * `end` is not specified, it's set to `start` with `start` then set to `0`.
         * If `start` is greater than `end` the params are swapped to support
         * negative ranges.
         *
         * @static
         * @memberOf _
         * @since 3.3.0
         * @category Number
         * @param {number} number The number to check.
         * @param {number} [start=0] The start of the range.
         * @param {number} end The end of the range.
         * @returns {boolean} Returns `true` if `number` is in the range, else `false`.
         * @see _.range, _.rangeRight
         * @example
         *
         * _.inRange(3, 2, 4);
         * // => true
         *
         * _.inRange(4, 8);
         * // => true
         *
         * _.inRange(4, 2);
         * // => false
         *
         * _.inRange(2, 2);
         * // => false
         *
         * _.inRange(1.2, 2);
         * // => true
         *
         * _.inRange(5.2, 4);
         * // => false
         *
         * _.inRange(-3, -2, -6);
         * // => true
         */
        function inRange(number, start, end) {
          start = toFinite(start);
          if (end === undefined$1) {
            end = start;
            start = 0;
          } else {
            end = toFinite(end);
          }
          number = toNumber(number);
          return baseInRange(number, start, end);
        }

        /**
         * Produces a random number between the inclusive `lower` and `upper` bounds.
         * If only one argument is provided a number between `0` and the given number
         * is returned. If `floating` is `true`, or either `lower` or `upper` are
         * floats, a floating-point number is returned instead of an integer.
         *
         * **Note:** JavaScript follows the IEEE-754 standard for resolving
         * floating-point values which can produce unexpected results.
         *
         * @static
         * @memberOf _
         * @since 0.7.0
         * @category Number
         * @param {number} [lower=0] The lower bound.
         * @param {number} [upper=1] The upper bound.
         * @param {boolean} [floating] Specify returning a floating-point number.
         * @returns {number} Returns the random number.
         * @example
         *
         * _.random(0, 5);
         * // => an integer between 0 and 5
         *
         * _.random(5);
         * // => also an integer between 0 and 5
         *
         * _.random(5, true);
         * // => a floating-point number between 0 and 5
         *
         * _.random(1.2, 5.2);
         * // => a floating-point number between 1.2 and 5.2
         */
        function random(lower, upper, floating) {
          if (floating && typeof floating != 'boolean' && isIterateeCall(lower, upper, floating)) {
            upper = floating = undefined$1;
          }
          if (floating === undefined$1) {
            if (typeof upper == 'boolean') {
              floating = upper;
              upper = undefined$1;
            }
            else if (typeof lower == 'boolean') {
              floating = lower;
              lower = undefined$1;
            }
          }
          if (lower === undefined$1 && upper === undefined$1) {
            lower = 0;
            upper = 1;
          }
          else {
            lower = toFinite(lower);
            if (upper === undefined$1) {
              upper = lower;
              lower = 0;
            } else {
              upper = toFinite(upper);
            }
          }
          if (lower > upper) {
            var temp = lower;
            lower = upper;
            upper = temp;
          }
          if (floating || lower % 1 || upper % 1) {
            var rand = nativeRandom();
            return nativeMin(lower + (rand * (upper - lower + freeParseFloat('1e-' + ((rand + '').length - 1)))), upper);
          }
          return baseRandom(lower, upper);
        }

        /*------------------------------------------------------------------------*/

        /**
         * Converts `string` to [camel case](https://en.wikipedia.org/wiki/CamelCase).
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category String
         * @param {string} [string=''] The string to convert.
         * @returns {string} Returns the camel cased string.
         * @example
         *
         * _.camelCase('Foo Bar');
         * // => 'fooBar'
         *
         * _.camelCase('--foo-bar--');
         * // => 'fooBar'
         *
         * _.camelCase('__FOO_BAR__');
         * // => 'fooBar'
         */
        var camelCase = createCompounder(function(result, word, index) {
          word = word.toLowerCase();
          return result + (index ? capitalize(word) : word);
        });

        /**
         * Converts the first character of `string` to upper case and the remaining
         * to lower case.
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category String
         * @param {string} [string=''] The string to capitalize.
         * @returns {string} Returns the capitalized string.
         * @example
         *
         * _.capitalize('FRED');
         * // => 'Fred'
         */
        function capitalize(string) {
          return upperFirst(toString(string).toLowerCase());
        }

        /**
         * Deburrs `string` by converting
         * [Latin-1 Supplement](https://en.wikipedia.org/wiki/Latin-1_Supplement_(Unicode_block)#Character_table)
         * and [Latin Extended-A](https://en.wikipedia.org/wiki/Latin_Extended-A)
         * letters to basic Latin letters and removing
         * [combining diacritical marks](https://en.wikipedia.org/wiki/Combining_Diacritical_Marks).
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category String
         * @param {string} [string=''] The string to deburr.
         * @returns {string} Returns the deburred string.
         * @example
         *
         * _.deburr('déjà vu');
         * // => 'deja vu'
         */
        function deburr(string) {
          string = toString(string);
          return string && string.replace(reLatin, deburrLetter).replace(reComboMark, '');
        }

        /**
         * Checks if `string` ends with the given target string.
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category String
         * @param {string} [string=''] The string to inspect.
         * @param {string} [target] The string to search for.
         * @param {number} [position=string.length] The position to search up to.
         * @returns {boolean} Returns `true` if `string` ends with `target`,
         *  else `false`.
         * @example
         *
         * _.endsWith('abc', 'c');
         * // => true
         *
         * _.endsWith('abc', 'b');
         * // => false
         *
         * _.endsWith('abc', 'b', 2);
         * // => true
         */
        function endsWith(string, target, position) {
          string = toString(string);
          target = baseToString(target);

          var length = string.length;
          position = position === undefined$1
            ? length
            : baseClamp(toInteger(position), 0, length);

          var end = position;
          position -= target.length;
          return position >= 0 && string.slice(position, end) == target;
        }

        /**
         * Converts the characters "&", "<", ">", '"', and "'" in `string` to their
         * corresponding HTML entities.
         *
         * **Note:** No other characters are escaped. To escape additional
         * characters use a third-party library like [_he_](https://mths.be/he).
         *
         * Though the ">" character is escaped for symmetry, characters like
         * ">" and "/" don't need escaping in HTML and have no special meaning
         * unless they're part of a tag or unquoted attribute value. See
         * [Mathias Bynens's article](https://mathiasbynens.be/notes/ambiguous-ampersands)
         * (under "semi-related fun fact") for more details.
         *
         * When working with HTML you should always
         * [quote attribute values](http://wonko.com/post/html-escaping) to reduce
         * XSS vectors.
         *
         * @static
         * @since 0.1.0
         * @memberOf _
         * @category String
         * @param {string} [string=''] The string to escape.
         * @returns {string} Returns the escaped string.
         * @example
         *
         * _.escape('fred, barney, & pebbles');
         * // => 'fred, barney, &amp; pebbles'
         */
        function escape(string) {
          string = toString(string);
          return (string && reHasUnescapedHtml.test(string))
            ? string.replace(reUnescapedHtml, escapeHtmlChar)
            : string;
        }

        /**
         * Escapes the `RegExp` special characters "^", "$", "\", ".", "*", "+",
         * "?", "(", ")", "[", "]", "{", "}", and "|" in `string`.
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category String
         * @param {string} [string=''] The string to escape.
         * @returns {string} Returns the escaped string.
         * @example
         *
         * _.escapeRegExp('[lodash](https://lodash.com/)');
         * // => '\[lodash\]\(https://lodash\.com/\)'
         */
        function escapeRegExp(string) {
          string = toString(string);
          return (string && reHasRegExpChar.test(string))
            ? string.replace(reRegExpChar, '\\$&')
            : string;
        }

        /**
         * Converts `string` to
         * [kebab case](https://en.wikipedia.org/wiki/Letter_case#Special_case_styles).
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category String
         * @param {string} [string=''] The string to convert.
         * @returns {string} Returns the kebab cased string.
         * @example
         *
         * _.kebabCase('Foo Bar');
         * // => 'foo-bar'
         *
         * _.kebabCase('fooBar');
         * // => 'foo-bar'
         *
         * _.kebabCase('__FOO_BAR__');
         * // => 'foo-bar'
         */
        var kebabCase = createCompounder(function(result, word, index) {
          return result + (index ? '-' : '') + word.toLowerCase();
        });

        /**
         * Converts `string`, as space separated words, to lower case.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category String
         * @param {string} [string=''] The string to convert.
         * @returns {string} Returns the lower cased string.
         * @example
         *
         * _.lowerCase('--Foo-Bar--');
         * // => 'foo bar'
         *
         * _.lowerCase('fooBar');
         * // => 'foo bar'
         *
         * _.lowerCase('__FOO_BAR__');
         * // => 'foo bar'
         */
        var lowerCase = createCompounder(function(result, word, index) {
          return result + (index ? ' ' : '') + word.toLowerCase();
        });

        /**
         * Converts the first character of `string` to lower case.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category String
         * @param {string} [string=''] The string to convert.
         * @returns {string} Returns the converted string.
         * @example
         *
         * _.lowerFirst('Fred');
         * // => 'fred'
         *
         * _.lowerFirst('FRED');
         * // => 'fRED'
         */
        var lowerFirst = createCaseFirst('toLowerCase');

        /**
         * Pads `string` on the left and right sides if it's shorter than `length`.
         * Padding characters are truncated if they can't be evenly divided by `length`.
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category String
         * @param {string} [string=''] The string to pad.
         * @param {number} [length=0] The padding length.
         * @param {string} [chars=' '] The string used as padding.
         * @returns {string} Returns the padded string.
         * @example
         *
         * _.pad('abc', 8);
         * // => '  abc   '
         *
         * _.pad('abc', 8, '_-');
         * // => '_-abc_-_'
         *
         * _.pad('abc', 3);
         * // => 'abc'
         */
        function pad(string, length, chars) {
          string = toString(string);
          length = toInteger(length);

          var strLength = length ? stringSize(string) : 0;
          if (!length || strLength >= length) {
            return string;
          }
          var mid = (length - strLength) / 2;
          return (
            createPadding(nativeFloor(mid), chars) +
            string +
            createPadding(nativeCeil(mid), chars)
          );
        }

        /**
         * Pads `string` on the right side if it's shorter than `length`. Padding
         * characters are truncated if they exceed `length`.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category String
         * @param {string} [string=''] The string to pad.
         * @param {number} [length=0] The padding length.
         * @param {string} [chars=' '] The string used as padding.
         * @returns {string} Returns the padded string.
         * @example
         *
         * _.padEnd('abc', 6);
         * // => 'abc   '
         *
         * _.padEnd('abc', 6, '_-');
         * // => 'abc_-_'
         *
         * _.padEnd('abc', 3);
         * // => 'abc'
         */
        function padEnd(string, length, chars) {
          string = toString(string);
          length = toInteger(length);

          var strLength = length ? stringSize(string) : 0;
          return (length && strLength < length)
            ? (string + createPadding(length - strLength, chars))
            : string;
        }

        /**
         * Pads `string` on the left side if it's shorter than `length`. Padding
         * characters are truncated if they exceed `length`.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category String
         * @param {string} [string=''] The string to pad.
         * @param {number} [length=0] The padding length.
         * @param {string} [chars=' '] The string used as padding.
         * @returns {string} Returns the padded string.
         * @example
         *
         * _.padStart('abc', 6);
         * // => '   abc'
         *
         * _.padStart('abc', 6, '_-');
         * // => '_-_abc'
         *
         * _.padStart('abc', 3);
         * // => 'abc'
         */
        function padStart(string, length, chars) {
          string = toString(string);
          length = toInteger(length);

          var strLength = length ? stringSize(string) : 0;
          return (length && strLength < length)
            ? (createPadding(length - strLength, chars) + string)
            : string;
        }

        /**
         * Converts `string` to an integer of the specified radix. If `radix` is
         * `undefined` or `0`, a `radix` of `10` is used unless `value` is a
         * hexadecimal, in which case a `radix` of `16` is used.
         *
         * **Note:** This method aligns with the
         * [ES5 implementation](https://es5.github.io/#x15.1.2.2) of `parseInt`.
         *
         * @static
         * @memberOf _
         * @since 1.1.0
         * @category String
         * @param {string} string The string to convert.
         * @param {number} [radix=10] The radix to interpret `value` by.
         * @param- {Object} [guard] Enables use as an iteratee for methods like `_.map`.
         * @returns {number} Returns the converted integer.
         * @example
         *
         * _.parseInt('08');
         * // => 8
         *
         * _.map(['6', '08', '10'], _.parseInt);
         * // => [6, 8, 10]
         */
        function parseInt(string, radix, guard) {
          if (guard || radix == null) {
            radix = 0;
          } else if (radix) {
            radix = +radix;
          }
          return nativeParseInt(toString(string).replace(reTrimStart, ''), radix || 0);
        }

        /**
         * Repeats the given string `n` times.
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category String
         * @param {string} [string=''] The string to repeat.
         * @param {number} [n=1] The number of times to repeat the string.
         * @param- {Object} [guard] Enables use as an iteratee for methods like `_.map`.
         * @returns {string} Returns the repeated string.
         * @example
         *
         * _.repeat('*', 3);
         * // => '***'
         *
         * _.repeat('abc', 2);
         * // => 'abcabc'
         *
         * _.repeat('abc', 0);
         * // => ''
         */
        function repeat(string, n, guard) {
          if ((guard ? isIterateeCall(string, n, guard) : n === undefined$1)) {
            n = 1;
          } else {
            n = toInteger(n);
          }
          return baseRepeat(toString(string), n);
        }

        /**
         * Replaces matches for `pattern` in `string` with `replacement`.
         *
         * **Note:** This method is based on
         * [`String#replace`](https://mdn.io/String/replace).
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category String
         * @param {string} [string=''] The string to modify.
         * @param {RegExp|string} pattern The pattern to replace.
         * @param {Function|string} replacement The match replacement.
         * @returns {string} Returns the modified string.
         * @example
         *
         * _.replace('Hi Fred', 'Fred', 'Barney');
         * // => 'Hi Barney'
         */
        function replace() {
          var args = arguments,
              string = toString(args[0]);

          return args.length < 3 ? string : string.replace(args[1], args[2]);
        }

        /**
         * Converts `string` to
         * [snake case](https://en.wikipedia.org/wiki/Snake_case).
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category String
         * @param {string} [string=''] The string to convert.
         * @returns {string} Returns the snake cased string.
         * @example
         *
         * _.snakeCase('Foo Bar');
         * // => 'foo_bar'
         *
         * _.snakeCase('fooBar');
         * // => 'foo_bar'
         *
         * _.snakeCase('--FOO-BAR--');
         * // => 'foo_bar'
         */
        var snakeCase = createCompounder(function(result, word, index) {
          return result + (index ? '_' : '') + word.toLowerCase();
        });

        /**
         * Splits `string` by `separator`.
         *
         * **Note:** This method is based on
         * [`String#split`](https://mdn.io/String/split).
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category String
         * @param {string} [string=''] The string to split.
         * @param {RegExp|string} separator The separator pattern to split by.
         * @param {number} [limit] The length to truncate results to.
         * @returns {Array} Returns the string segments.
         * @example
         *
         * _.split('a-b-c', '-', 2);
         * // => ['a', 'b']
         */
        function split(string, separator, limit) {
          if (limit && typeof limit != 'number' && isIterateeCall(string, separator, limit)) {
            separator = limit = undefined$1;
          }
          limit = limit === undefined$1 ? MAX_ARRAY_LENGTH : limit >>> 0;
          if (!limit) {
            return [];
          }
          string = toString(string);
          if (string && (
                typeof separator == 'string' ||
                (separator != null && !isRegExp(separator))
              )) {
            separator = baseToString(separator);
            if (!separator && hasUnicode(string)) {
              return castSlice(stringToArray(string), 0, limit);
            }
          }
          return string.split(separator, limit);
        }

        /**
         * Converts `string` to
         * [start case](https://en.wikipedia.org/wiki/Letter_case#Stylistic_or_specialised_usage).
         *
         * @static
         * @memberOf _
         * @since 3.1.0
         * @category String
         * @param {string} [string=''] The string to convert.
         * @returns {string} Returns the start cased string.
         * @example
         *
         * _.startCase('--foo-bar--');
         * // => 'Foo Bar'
         *
         * _.startCase('fooBar');
         * // => 'Foo Bar'
         *
         * _.startCase('__FOO_BAR__');
         * // => 'FOO BAR'
         */
        var startCase = createCompounder(function(result, word, index) {
          return result + (index ? ' ' : '') + upperFirst(word);
        });

        /**
         * Checks if `string` starts with the given target string.
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category String
         * @param {string} [string=''] The string to inspect.
         * @param {string} [target] The string to search for.
         * @param {number} [position=0] The position to search from.
         * @returns {boolean} Returns `true` if `string` starts with `target`,
         *  else `false`.
         * @example
         *
         * _.startsWith('abc', 'a');
         * // => true
         *
         * _.startsWith('abc', 'b');
         * // => false
         *
         * _.startsWith('abc', 'b', 1);
         * // => true
         */
        function startsWith(string, target, position) {
          string = toString(string);
          position = position == null
            ? 0
            : baseClamp(toInteger(position), 0, string.length);

          target = baseToString(target);
          return string.slice(position, position + target.length) == target;
        }

        /**
         * Creates a compiled template function that can interpolate data properties
         * in "interpolate" delimiters, HTML-escape interpolated data properties in
         * "escape" delimiters, and execute JavaScript in "evaluate" delimiters. Data
         * properties may be accessed as free variables in the template. If a setting
         * object is given, it takes precedence over `_.templateSettings` values.
         *
         * **Note:** In the development build `_.template` utilizes
         * [sourceURLs](http://www.html5rocks.com/en/tutorials/developertools/sourcemaps/#toc-sourceurl)
         * for easier debugging.
         *
         * For more information on precompiling templates see
         * [lodash's custom builds documentation](https://lodash.com/custom-builds).
         *
         * For more information on Chrome extension sandboxes see
         * [Chrome's extensions documentation](https://developer.chrome.com/extensions/sandboxingEval).
         *
         * @static
         * @since 0.1.0
         * @memberOf _
         * @category String
         * @param {string} [string=''] The template string.
         * @param {Object} [options={}] The options object.
         * @param {RegExp} [options.escape=_.templateSettings.escape]
         *  The HTML "escape" delimiter.
         * @param {RegExp} [options.evaluate=_.templateSettings.evaluate]
         *  The "evaluate" delimiter.
         * @param {Object} [options.imports=_.templateSettings.imports]
         *  An object to import into the template as free variables.
         * @param {RegExp} [options.interpolate=_.templateSettings.interpolate]
         *  The "interpolate" delimiter.
         * @param {string} [options.sourceURL='lodash.templateSources[n]']
         *  The sourceURL of the compiled template.
         * @param {string} [options.variable='obj']
         *  The data object variable name.
         * @param- {Object} [guard] Enables use as an iteratee for methods like `_.map`.
         * @returns {Function} Returns the compiled template function.
         * @example
         *
         * // Use the "interpolate" delimiter to create a compiled template.
         * var compiled = _.template('hello <%= user %>!');
         * compiled({ 'user': 'fred' });
         * // => 'hello fred!'
         *
         * // Use the HTML "escape" delimiter to escape data property values.
         * var compiled = _.template('<b><%- value %></b>');
         * compiled({ 'value': '<script>' });
         * // => '<b>&lt;script&gt;</b>'
         *
         * // Use the "evaluate" delimiter to execute JavaScript and generate HTML.
         * var compiled = _.template('<% _.forEach(users, function(user) { %><li><%- user %></li><% }); %>');
         * compiled({ 'users': ['fred', 'barney'] });
         * // => '<li>fred</li><li>barney</li>'
         *
         * // Use the internal `print` function in "evaluate" delimiters.
         * var compiled = _.template('<% print("hello " + user); %>!');
         * compiled({ 'user': 'barney' });
         * // => 'hello barney!'
         *
         * // Use the ES template literal delimiter as an "interpolate" delimiter.
         * // Disable support by replacing the "interpolate" delimiter.
         * var compiled = _.template('hello ${ user }!');
         * compiled({ 'user': 'pebbles' });
         * // => 'hello pebbles!'
         *
         * // Use backslashes to treat delimiters as plain text.
         * var compiled = _.template('<%= "\\<%- value %\\>" %>');
         * compiled({ 'value': 'ignored' });
         * // => '<%- value %>'
         *
         * // Use the `imports` option to import `jQuery` as `jq`.
         * var text = '<% jq.each(users, function(user) { %><li><%- user %></li><% }); %>';
         * var compiled = _.template(text, { 'imports': { 'jq': jQuery } });
         * compiled({ 'users': ['fred', 'barney'] });
         * // => '<li>fred</li><li>barney</li>'
         *
         * // Use the `sourceURL` option to specify a custom sourceURL for the template.
         * var compiled = _.template('hello <%= user %>!', { 'sourceURL': '/basic/greeting.jst' });
         * compiled(data);
         * // => Find the source of "greeting.jst" under the Sources tab or Resources panel of the web inspector.
         *
         * // Use the `variable` option to ensure a with-statement isn't used in the compiled template.
         * var compiled = _.template('hi <%= data.user %>!', { 'variable': 'data' });
         * compiled.source;
         * // => function(data) {
         * //   var __t, __p = '';
         * //   __p += 'hi ' + ((__t = ( data.user )) == null ? '' : __t) + '!';
         * //   return __p;
         * // }
         *
         * // Use custom template delimiters.
         * _.templateSettings.interpolate = /{{([\s\S]+?)}}/g;
         * var compiled = _.template('hello {{ user }}!');
         * compiled({ 'user': 'mustache' });
         * // => 'hello mustache!'
         *
         * // Use the `source` property to inline compiled templates for meaningful
         * // line numbers in error messages and stack traces.
         * fs.writeFileSync(path.join(process.cwd(), 'jst.js'), '\
         *   var JST = {\
         *     "main": ' + _.template(mainText).source + '\
         *   };\
         * ');
         */
        function template(string, options, guard) {
          // Based on John Resig's `tmpl` implementation
          // (http://ejohn.org/blog/javascript-micro-templating/)
          // and Laura Doktorova's doT.js (https://github.com/olado/doT).
          var settings = lodash.templateSettings;

          if (guard && isIterateeCall(string, options, guard)) {
            options = undefined$1;
          }
          string = toString(string);
          options = assignInWith({}, options, settings, customDefaultsAssignIn);

          var imports = assignInWith({}, options.imports, settings.imports, customDefaultsAssignIn),
              importsKeys = keys(imports),
              importsValues = baseValues(imports, importsKeys);

          var isEscaping,
              isEvaluating,
              index = 0,
              interpolate = options.interpolate || reNoMatch,
              source = "__p += '";

          // Compile the regexp to match each delimiter.
          var reDelimiters = RegExp(
            (options.escape || reNoMatch).source + '|' +
            interpolate.source + '|' +
            (interpolate === reInterpolate ? reEsTemplate : reNoMatch).source + '|' +
            (options.evaluate || reNoMatch).source + '|$'
          , 'g');

          // Use a sourceURL for easier debugging.
          // The sourceURL gets injected into the source that's eval-ed, so be careful
          // to normalize all kinds of whitespace, so e.g. newlines (and unicode versions of it) can't sneak in
          // and escape the comment, thus injecting code that gets evaled.
          var sourceURL = '//# sourceURL=' +
            (hasOwnProperty.call(options, 'sourceURL')
              ? (options.sourceURL + '').replace(/\s/g, ' ')
              : ('lodash.templateSources[' + (++templateCounter) + ']')
            ) + '\n';

          string.replace(reDelimiters, function(match, escapeValue, interpolateValue, esTemplateValue, evaluateValue, offset) {
            interpolateValue || (interpolateValue = esTemplateValue);

            // Escape characters that can't be included in string literals.
            source += string.slice(index, offset).replace(reUnescapedString, escapeStringChar);

            // Replace delimiters with snippets.
            if (escapeValue) {
              isEscaping = true;
              source += "' +\n__e(" + escapeValue + ") +\n'";
            }
            if (evaluateValue) {
              isEvaluating = true;
              source += "';\n" + evaluateValue + ";\n__p += '";
            }
            if (interpolateValue) {
              source += "' +\n((__t = (" + interpolateValue + ")) == null ? '' : __t) +\n'";
            }
            index = offset + match.length;

            // The JS engine embedded in Adobe products needs `match` returned in
            // order to produce the correct `offset` value.
            return match;
          });

          source += "';\n";

          // If `variable` is not specified wrap a with-statement around the generated
          // code to add the data object to the top of the scope chain.
          var variable = hasOwnProperty.call(options, 'variable') && options.variable;
          if (!variable) {
            source = 'with (obj) {\n' + source + '\n}\n';
          }
          // Cleanup code by stripping empty strings.
          source = (isEvaluating ? source.replace(reEmptyStringLeading, '') : source)
            .replace(reEmptyStringMiddle, '$1')
            .replace(reEmptyStringTrailing, '$1;');

          // Frame code as the function body.
          source = 'function(' + (variable || 'obj') + ') {\n' +
            (variable
              ? ''
              : 'obj || (obj = {});\n'
            ) +
            "var __t, __p = ''" +
            (isEscaping
               ? ', __e = _.escape'
               : ''
            ) +
            (isEvaluating
              ? ', __j = Array.prototype.join;\n' +
                "function print() { __p += __j.call(arguments, '') }\n"
              : ';\n'
            ) +
            source +
            'return __p\n}';

          var result = attempt(function() {
            return Function(importsKeys, sourceURL + 'return ' + source)
              .apply(undefined$1, importsValues);
          });

          // Provide the compiled function's source by its `toString` method or
          // the `source` property as a convenience for inlining compiled templates.
          result.source = source;
          if (isError(result)) {
            throw result;
          }
          return result;
        }

        /**
         * Converts `string`, as a whole, to lower case just like
         * [String#toLowerCase](https://mdn.io/toLowerCase).
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category String
         * @param {string} [string=''] The string to convert.
         * @returns {string} Returns the lower cased string.
         * @example
         *
         * _.toLower('--Foo-Bar--');
         * // => '--foo-bar--'
         *
         * _.toLower('fooBar');
         * // => 'foobar'
         *
         * _.toLower('__FOO_BAR__');
         * // => '__foo_bar__'
         */
        function toLower(value) {
          return toString(value).toLowerCase();
        }

        /**
         * Converts `string`, as a whole, to upper case just like
         * [String#toUpperCase](https://mdn.io/toUpperCase).
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category String
         * @param {string} [string=''] The string to convert.
         * @returns {string} Returns the upper cased string.
         * @example
         *
         * _.toUpper('--foo-bar--');
         * // => '--FOO-BAR--'
         *
         * _.toUpper('fooBar');
         * // => 'FOOBAR'
         *
         * _.toUpper('__foo_bar__');
         * // => '__FOO_BAR__'
         */
        function toUpper(value) {
          return toString(value).toUpperCase();
        }

        /**
         * Removes leading and trailing whitespace or specified characters from `string`.
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category String
         * @param {string} [string=''] The string to trim.
         * @param {string} [chars=whitespace] The characters to trim.
         * @param- {Object} [guard] Enables use as an iteratee for methods like `_.map`.
         * @returns {string} Returns the trimmed string.
         * @example
         *
         * _.trim('  abc  ');
         * // => 'abc'
         *
         * _.trim('-_-abc-_-', '_-');
         * // => 'abc'
         *
         * _.map(['  foo  ', '  bar  '], _.trim);
         * // => ['foo', 'bar']
         */
        function trim(string, chars, guard) {
          string = toString(string);
          if (string && (guard || chars === undefined$1)) {
            return string.replace(reTrim, '');
          }
          if (!string || !(chars = baseToString(chars))) {
            return string;
          }
          var strSymbols = stringToArray(string),
              chrSymbols = stringToArray(chars),
              start = charsStartIndex(strSymbols, chrSymbols),
              end = charsEndIndex(strSymbols, chrSymbols) + 1;

          return castSlice(strSymbols, start, end).join('');
        }

        /**
         * Removes trailing whitespace or specified characters from `string`.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category String
         * @param {string} [string=''] The string to trim.
         * @param {string} [chars=whitespace] The characters to trim.
         * @param- {Object} [guard] Enables use as an iteratee for methods like `_.map`.
         * @returns {string} Returns the trimmed string.
         * @example
         *
         * _.trimEnd('  abc  ');
         * // => '  abc'
         *
         * _.trimEnd('-_-abc-_-', '_-');
         * // => '-_-abc'
         */
        function trimEnd(string, chars, guard) {
          string = toString(string);
          if (string && (guard || chars === undefined$1)) {
            return string.replace(reTrimEnd, '');
          }
          if (!string || !(chars = baseToString(chars))) {
            return string;
          }
          var strSymbols = stringToArray(string),
              end = charsEndIndex(strSymbols, stringToArray(chars)) + 1;

          return castSlice(strSymbols, 0, end).join('');
        }

        /**
         * Removes leading whitespace or specified characters from `string`.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category String
         * @param {string} [string=''] The string to trim.
         * @param {string} [chars=whitespace] The characters to trim.
         * @param- {Object} [guard] Enables use as an iteratee for methods like `_.map`.
         * @returns {string} Returns the trimmed string.
         * @example
         *
         * _.trimStart('  abc  ');
         * // => 'abc  '
         *
         * _.trimStart('-_-abc-_-', '_-');
         * // => 'abc-_-'
         */
        function trimStart(string, chars, guard) {
          string = toString(string);
          if (string && (guard || chars === undefined$1)) {
            return string.replace(reTrimStart, '');
          }
          if (!string || !(chars = baseToString(chars))) {
            return string;
          }
          var strSymbols = stringToArray(string),
              start = charsStartIndex(strSymbols, stringToArray(chars));

          return castSlice(strSymbols, start).join('');
        }

        /**
         * Truncates `string` if it's longer than the given maximum string length.
         * The last characters of the truncated string are replaced with the omission
         * string which defaults to "...".
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category String
         * @param {string} [string=''] The string to truncate.
         * @param {Object} [options={}] The options object.
         * @param {number} [options.length=30] The maximum string length.
         * @param {string} [options.omission='...'] The string to indicate text is omitted.
         * @param {RegExp|string} [options.separator] The separator pattern to truncate to.
         * @returns {string} Returns the truncated string.
         * @example
         *
         * _.truncate('hi-diddly-ho there, neighborino');
         * // => 'hi-diddly-ho there, neighbo...'
         *
         * _.truncate('hi-diddly-ho there, neighborino', {
         *   'length': 24,
         *   'separator': ' '
         * });
         * // => 'hi-diddly-ho there,...'
         *
         * _.truncate('hi-diddly-ho there, neighborino', {
         *   'length': 24,
         *   'separator': /,? +/
         * });
         * // => 'hi-diddly-ho there...'
         *
         * _.truncate('hi-diddly-ho there, neighborino', {
         *   'omission': ' [...]'
         * });
         * // => 'hi-diddly-ho there, neig [...]'
         */
        function truncate(string, options) {
          var length = DEFAULT_TRUNC_LENGTH,
              omission = DEFAULT_TRUNC_OMISSION;

          if (isObject(options)) {
            var separator = 'separator' in options ? options.separator : separator;
            length = 'length' in options ? toInteger(options.length) : length;
            omission = 'omission' in options ? baseToString(options.omission) : omission;
          }
          string = toString(string);

          var strLength = string.length;
          if (hasUnicode(string)) {
            var strSymbols = stringToArray(string);
            strLength = strSymbols.length;
          }
          if (length >= strLength) {
            return string;
          }
          var end = length - stringSize(omission);
          if (end < 1) {
            return omission;
          }
          var result = strSymbols
            ? castSlice(strSymbols, 0, end).join('')
            : string.slice(0, end);

          if (separator === undefined$1) {
            return result + omission;
          }
          if (strSymbols) {
            end += (result.length - end);
          }
          if (isRegExp(separator)) {
            if (string.slice(end).search(separator)) {
              var match,
                  substring = result;

              if (!separator.global) {
                separator = RegExp(separator.source, toString(reFlags.exec(separator)) + 'g');
              }
              separator.lastIndex = 0;
              while ((match = separator.exec(substring))) {
                var newEnd = match.index;
              }
              result = result.slice(0, newEnd === undefined$1 ? end : newEnd);
            }
          } else if (string.indexOf(baseToString(separator), end) != end) {
            var index = result.lastIndexOf(separator);
            if (index > -1) {
              result = result.slice(0, index);
            }
          }
          return result + omission;
        }

        /**
         * The inverse of `_.escape`; this method converts the HTML entities
         * `&amp;`, `&lt;`, `&gt;`, `&quot;`, and `&#39;` in `string` to
         * their corresponding characters.
         *
         * **Note:** No other HTML entities are unescaped. To unescape additional
         * HTML entities use a third-party library like [_he_](https://mths.be/he).
         *
         * @static
         * @memberOf _
         * @since 0.6.0
         * @category String
         * @param {string} [string=''] The string to unescape.
         * @returns {string} Returns the unescaped string.
         * @example
         *
         * _.unescape('fred, barney, &amp; pebbles');
         * // => 'fred, barney, & pebbles'
         */
        function unescape(string) {
          string = toString(string);
          return (string && reHasEscapedHtml.test(string))
            ? string.replace(reEscapedHtml, unescapeHtmlChar)
            : string;
        }

        /**
         * Converts `string`, as space separated words, to upper case.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category String
         * @param {string} [string=''] The string to convert.
         * @returns {string} Returns the upper cased string.
         * @example
         *
         * _.upperCase('--foo-bar');
         * // => 'FOO BAR'
         *
         * _.upperCase('fooBar');
         * // => 'FOO BAR'
         *
         * _.upperCase('__foo_bar__');
         * // => 'FOO BAR'
         */
        var upperCase = createCompounder(function(result, word, index) {
          return result + (index ? ' ' : '') + word.toUpperCase();
        });

        /**
         * Converts the first character of `string` to upper case.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category String
         * @param {string} [string=''] The string to convert.
         * @returns {string} Returns the converted string.
         * @example
         *
         * _.upperFirst('fred');
         * // => 'Fred'
         *
         * _.upperFirst('FRED');
         * // => 'FRED'
         */
        var upperFirst = createCaseFirst('toUpperCase');

        /**
         * Splits `string` into an array of its words.
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category String
         * @param {string} [string=''] The string to inspect.
         * @param {RegExp|string} [pattern] The pattern to match words.
         * @param- {Object} [guard] Enables use as an iteratee for methods like `_.map`.
         * @returns {Array} Returns the words of `string`.
         * @example
         *
         * _.words('fred, barney, & pebbles');
         * // => ['fred', 'barney', 'pebbles']
         *
         * _.words('fred, barney, & pebbles', /[^, ]+/g);
         * // => ['fred', 'barney', '&', 'pebbles']
         */
        function words(string, pattern, guard) {
          string = toString(string);
          pattern = guard ? undefined$1 : pattern;

          if (pattern === undefined$1) {
            return hasUnicodeWord(string) ? unicodeWords(string) : asciiWords(string);
          }
          return string.match(pattern) || [];
        }

        /*------------------------------------------------------------------------*/

        /**
         * Attempts to invoke `func`, returning either the result or the caught error
         * object. Any additional arguments are provided to `func` when it's invoked.
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category Util
         * @param {Function} func The function to attempt.
         * @param {...*} [args] The arguments to invoke `func` with.
         * @returns {*} Returns the `func` result or error object.
         * @example
         *
         * // Avoid throwing errors for invalid selectors.
         * var elements = _.attempt(function(selector) {
         *   return document.querySelectorAll(selector);
         * }, '>_>');
         *
         * if (_.isError(elements)) {
         *   elements = [];
         * }
         */
        var attempt = baseRest(function(func, args) {
          try {
            return apply(func, undefined$1, args);
          } catch (e) {
            return isError(e) ? e : new Error(e);
          }
        });

        /**
         * Binds methods of an object to the object itself, overwriting the existing
         * method.
         *
         * **Note:** This method doesn't set the "length" property of bound functions.
         *
         * @static
         * @since 0.1.0
         * @memberOf _
         * @category Util
         * @param {Object} object The object to bind and assign the bound methods to.
         * @param {...(string|string[])} methodNames The object method names to bind.
         * @returns {Object} Returns `object`.
         * @example
         *
         * var view = {
         *   'label': 'docs',
         *   'click': function() {
         *     console.log('clicked ' + this.label);
         *   }
         * };
         *
         * _.bindAll(view, ['click']);
         * jQuery(element).on('click', view.click);
         * // => Logs 'clicked docs' when clicked.
         */
        var bindAll = flatRest(function(object, methodNames) {
          arrayEach(methodNames, function(key) {
            key = toKey(key);
            baseAssignValue(object, key, bind(object[key], object));
          });
          return object;
        });

        /**
         * Creates a function that iterates over `pairs` and invokes the corresponding
         * function of the first predicate to return truthy. The predicate-function
         * pairs are invoked with the `this` binding and arguments of the created
         * function.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Util
         * @param {Array} pairs The predicate-function pairs.
         * @returns {Function} Returns the new composite function.
         * @example
         *
         * var func = _.cond([
         *   [_.matches({ 'a': 1 }),           _.constant('matches A')],
         *   [_.conforms({ 'b': _.isNumber }), _.constant('matches B')],
         *   [_.stubTrue,                      _.constant('no match')]
         * ]);
         *
         * func({ 'a': 1, 'b': 2 });
         * // => 'matches A'
         *
         * func({ 'a': 0, 'b': 1 });
         * // => 'matches B'
         *
         * func({ 'a': '1', 'b': '2' });
         * // => 'no match'
         */
        function cond(pairs) {
          var length = pairs == null ? 0 : pairs.length,
              toIteratee = getIteratee();

          pairs = !length ? [] : arrayMap(pairs, function(pair) {
            if (typeof pair[1] != 'function') {
              throw new TypeError(FUNC_ERROR_TEXT);
            }
            return [toIteratee(pair[0]), pair[1]];
          });

          return baseRest(function(args) {
            var index = -1;
            while (++index < length) {
              var pair = pairs[index];
              if (apply(pair[0], this, args)) {
                return apply(pair[1], this, args);
              }
            }
          });
        }

        /**
         * Creates a function that invokes the predicate properties of `source` with
         * the corresponding property values of a given object, returning `true` if
         * all predicates return truthy, else `false`.
         *
         * **Note:** The created function is equivalent to `_.conformsTo` with
         * `source` partially applied.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Util
         * @param {Object} source The object of property predicates to conform to.
         * @returns {Function} Returns the new spec function.
         * @example
         *
         * var objects = [
         *   { 'a': 2, 'b': 1 },
         *   { 'a': 1, 'b': 2 }
         * ];
         *
         * _.filter(objects, _.conforms({ 'b': function(n) { return n > 1; } }));
         * // => [{ 'a': 1, 'b': 2 }]
         */
        function conforms(source) {
          return baseConforms(baseClone(source, CLONE_DEEP_FLAG));
        }

        /**
         * Creates a function that returns `value`.
         *
         * @static
         * @memberOf _
         * @since 2.4.0
         * @category Util
         * @param {*} value The value to return from the new function.
         * @returns {Function} Returns the new constant function.
         * @example
         *
         * var objects = _.times(2, _.constant({ 'a': 1 }));
         *
         * console.log(objects);
         * // => [{ 'a': 1 }, { 'a': 1 }]
         *
         * console.log(objects[0] === objects[1]);
         * // => true
         */
        function constant(value) {
          return function() {
            return value;
          };
        }

        /**
         * Checks `value` to determine whether a default value should be returned in
         * its place. The `defaultValue` is returned if `value` is `NaN`, `null`,
         * or `undefined`.
         *
         * @static
         * @memberOf _
         * @since 4.14.0
         * @category Util
         * @param {*} value The value to check.
         * @param {*} defaultValue The default value.
         * @returns {*} Returns the resolved value.
         * @example
         *
         * _.defaultTo(1, 10);
         * // => 1
         *
         * _.defaultTo(undefined, 10);
         * // => 10
         */
        function defaultTo(value, defaultValue) {
          return (value == null || value !== value) ? defaultValue : value;
        }

        /**
         * Creates a function that returns the result of invoking the given functions
         * with the `this` binding of the created function, where each successive
         * invocation is supplied the return value of the previous.
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category Util
         * @param {...(Function|Function[])} [funcs] The functions to invoke.
         * @returns {Function} Returns the new composite function.
         * @see _.flowRight
         * @example
         *
         * function square(n) {
         *   return n * n;
         * }
         *
         * var addSquare = _.flow([_.add, square]);
         * addSquare(1, 2);
         * // => 9
         */
        var flow = createFlow();

        /**
         * This method is like `_.flow` except that it creates a function that
         * invokes the given functions from right to left.
         *
         * @static
         * @since 3.0.0
         * @memberOf _
         * @category Util
         * @param {...(Function|Function[])} [funcs] The functions to invoke.
         * @returns {Function} Returns the new composite function.
         * @see _.flow
         * @example
         *
         * function square(n) {
         *   return n * n;
         * }
         *
         * var addSquare = _.flowRight([square, _.add]);
         * addSquare(1, 2);
         * // => 9
         */
        var flowRight = createFlow(true);

        /**
         * This method returns the first argument it receives.
         *
         * @static
         * @since 0.1.0
         * @memberOf _
         * @category Util
         * @param {*} value Any value.
         * @returns {*} Returns `value`.
         * @example
         *
         * var object = { 'a': 1 };
         *
         * console.log(_.identity(object) === object);
         * // => true
         */
        function identity(value) {
          return value;
        }

        /**
         * Creates a function that invokes `func` with the arguments of the created
         * function. If `func` is a property name, the created function returns the
         * property value for a given element. If `func` is an array or object, the
         * created function returns `true` for elements that contain the equivalent
         * source properties, otherwise it returns `false`.
         *
         * @static
         * @since 4.0.0
         * @memberOf _
         * @category Util
         * @param {*} [func=_.identity] The value to convert to a callback.
         * @returns {Function} Returns the callback.
         * @example
         *
         * var users = [
         *   { 'user': 'barney', 'age': 36, 'active': true },
         *   { 'user': 'fred',   'age': 40, 'active': false }
         * ];
         *
         * // The `_.matches` iteratee shorthand.
         * _.filter(users, _.iteratee({ 'user': 'barney', 'active': true }));
         * // => [{ 'user': 'barney', 'age': 36, 'active': true }]
         *
         * // The `_.matchesProperty` iteratee shorthand.
         * _.filter(users, _.iteratee(['user', 'fred']));
         * // => [{ 'user': 'fred', 'age': 40 }]
         *
         * // The `_.property` iteratee shorthand.
         * _.map(users, _.iteratee('user'));
         * // => ['barney', 'fred']
         *
         * // Create custom iteratee shorthands.
         * _.iteratee = _.wrap(_.iteratee, function(iteratee, func) {
         *   return !_.isRegExp(func) ? iteratee(func) : function(string) {
         *     return func.test(string);
         *   };
         * });
         *
         * _.filter(['abc', 'def'], /ef/);
         * // => ['def']
         */
        function iteratee(func) {
          return baseIteratee(typeof func == 'function' ? func : baseClone(func, CLONE_DEEP_FLAG));
        }

        /**
         * Creates a function that performs a partial deep comparison between a given
         * object and `source`, returning `true` if the given object has equivalent
         * property values, else `false`.
         *
         * **Note:** The created function is equivalent to `_.isMatch` with `source`
         * partially applied.
         *
         * Partial comparisons will match empty array and empty object `source`
         * values against any array or object value, respectively. See `_.isEqual`
         * for a list of supported value comparisons.
         *
         * **Note:** Multiple values can be checked by combining several matchers
         * using `_.overSome`
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category Util
         * @param {Object} source The object of property values to match.
         * @returns {Function} Returns the new spec function.
         * @example
         *
         * var objects = [
         *   { 'a': 1, 'b': 2, 'c': 3 },
         *   { 'a': 4, 'b': 5, 'c': 6 }
         * ];
         *
         * _.filter(objects, _.matches({ 'a': 4, 'c': 6 }));
         * // => [{ 'a': 4, 'b': 5, 'c': 6 }]
         *
         * // Checking for several possible values
         * _.filter(users, _.overSome([_.matches({ 'a': 1 }), _.matches({ 'a': 4 })]));
         * // => [{ 'a': 1, 'b': 2, 'c': 3 }, { 'a': 4, 'b': 5, 'c': 6 }]
         */
        function matches(source) {
          return baseMatches(baseClone(source, CLONE_DEEP_FLAG));
        }

        /**
         * Creates a function that performs a partial deep comparison between the
         * value at `path` of a given object to `srcValue`, returning `true` if the
         * object value is equivalent, else `false`.
         *
         * **Note:** Partial comparisons will match empty array and empty object
         * `srcValue` values against any array or object value, respectively. See
         * `_.isEqual` for a list of supported value comparisons.
         *
         * **Note:** Multiple values can be checked by combining several matchers
         * using `_.overSome`
         *
         * @static
         * @memberOf _
         * @since 3.2.0
         * @category Util
         * @param {Array|string} path The path of the property to get.
         * @param {*} srcValue The value to match.
         * @returns {Function} Returns the new spec function.
         * @example
         *
         * var objects = [
         *   { 'a': 1, 'b': 2, 'c': 3 },
         *   { 'a': 4, 'b': 5, 'c': 6 }
         * ];
         *
         * _.find(objects, _.matchesProperty('a', 4));
         * // => { 'a': 4, 'b': 5, 'c': 6 }
         *
         * // Checking for several possible values
         * _.filter(users, _.overSome([_.matchesProperty('a', 1), _.matchesProperty('a', 4)]));
         * // => [{ 'a': 1, 'b': 2, 'c': 3 }, { 'a': 4, 'b': 5, 'c': 6 }]
         */
        function matchesProperty(path, srcValue) {
          return baseMatchesProperty(path, baseClone(srcValue, CLONE_DEEP_FLAG));
        }

        /**
         * Creates a function that invokes the method at `path` of a given object.
         * Any additional arguments are provided to the invoked method.
         *
         * @static
         * @memberOf _
         * @since 3.7.0
         * @category Util
         * @param {Array|string} path The path of the method to invoke.
         * @param {...*} [args] The arguments to invoke the method with.
         * @returns {Function} Returns the new invoker function.
         * @example
         *
         * var objects = [
         *   { 'a': { 'b': _.constant(2) } },
         *   { 'a': { 'b': _.constant(1) } }
         * ];
         *
         * _.map(objects, _.method('a.b'));
         * // => [2, 1]
         *
         * _.map(objects, _.method(['a', 'b']));
         * // => [2, 1]
         */
        var method = baseRest(function(path, args) {
          return function(object) {
            return baseInvoke(object, path, args);
          };
        });

        /**
         * The opposite of `_.method`; this method creates a function that invokes
         * the method at a given path of `object`. Any additional arguments are
         * provided to the invoked method.
         *
         * @static
         * @memberOf _
         * @since 3.7.0
         * @category Util
         * @param {Object} object The object to query.
         * @param {...*} [args] The arguments to invoke the method with.
         * @returns {Function} Returns the new invoker function.
         * @example
         *
         * var array = _.times(3, _.constant),
         *     object = { 'a': array, 'b': array, 'c': array };
         *
         * _.map(['a[2]', 'c[0]'], _.methodOf(object));
         * // => [2, 0]
         *
         * _.map([['a', '2'], ['c', '0']], _.methodOf(object));
         * // => [2, 0]
         */
        var methodOf = baseRest(function(object, args) {
          return function(path) {
            return baseInvoke(object, path, args);
          };
        });

        /**
         * Adds all own enumerable string keyed function properties of a source
         * object to the destination object. If `object` is a function, then methods
         * are added to its prototype as well.
         *
         * **Note:** Use `_.runInContext` to create a pristine `lodash` function to
         * avoid conflicts caused by modifying the original.
         *
         * @static
         * @since 0.1.0
         * @memberOf _
         * @category Util
         * @param {Function|Object} [object=lodash] The destination object.
         * @param {Object} source The object of functions to add.
         * @param {Object} [options={}] The options object.
         * @param {boolean} [options.chain=true] Specify whether mixins are chainable.
         * @returns {Function|Object} Returns `object`.
         * @example
         *
         * function vowels(string) {
         *   return _.filter(string, function(v) {
         *     return /[aeiou]/i.test(v);
         *   });
         * }
         *
         * _.mixin({ 'vowels': vowels });
         * _.vowels('fred');
         * // => ['e']
         *
         * _('fred').vowels().value();
         * // => ['e']
         *
         * _.mixin({ 'vowels': vowels }, { 'chain': false });
         * _('fred').vowels();
         * // => ['e']
         */
        function mixin(object, source, options) {
          var props = keys(source),
              methodNames = baseFunctions(source, props);

          if (options == null &&
              !(isObject(source) && (methodNames.length || !props.length))) {
            options = source;
            source = object;
            object = this;
            methodNames = baseFunctions(source, keys(source));
          }
          var chain = !(isObject(options) && 'chain' in options) || !!options.chain,
              isFunc = isFunction(object);

          arrayEach(methodNames, function(methodName) {
            var func = source[methodName];
            object[methodName] = func;
            if (isFunc) {
              object.prototype[methodName] = function() {
                var chainAll = this.__chain__;
                if (chain || chainAll) {
                  var result = object(this.__wrapped__),
                      actions = result.__actions__ = copyArray(this.__actions__);

                  actions.push({ 'func': func, 'args': arguments, 'thisArg': object });
                  result.__chain__ = chainAll;
                  return result;
                }
                return func.apply(object, arrayPush([this.value()], arguments));
              };
            }
          });

          return object;
        }

        /**
         * Reverts the `_` variable to its previous value and returns a reference to
         * the `lodash` function.
         *
         * @static
         * @since 0.1.0
         * @memberOf _
         * @category Util
         * @returns {Function} Returns the `lodash` function.
         * @example
         *
         * var lodash = _.noConflict();
         */
        function noConflict() {
          if (root._ === this) {
            root._ = oldDash;
          }
          return this;
        }

        /**
         * This method returns `undefined`.
         *
         * @static
         * @memberOf _
         * @since 2.3.0
         * @category Util
         * @example
         *
         * _.times(2, _.noop);
         * // => [undefined, undefined]
         */
        function noop() {
          // No operation performed.
        }

        /**
         * Creates a function that gets the argument at index `n`. If `n` is negative,
         * the nth argument from the end is returned.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Util
         * @param {number} [n=0] The index of the argument to return.
         * @returns {Function} Returns the new pass-thru function.
         * @example
         *
         * var func = _.nthArg(1);
         * func('a', 'b', 'c', 'd');
         * // => 'b'
         *
         * var func = _.nthArg(-2);
         * func('a', 'b', 'c', 'd');
         * // => 'c'
         */
        function nthArg(n) {
          n = toInteger(n);
          return baseRest(function(args) {
            return baseNth(args, n);
          });
        }

        /**
         * Creates a function that invokes `iteratees` with the arguments it receives
         * and returns their results.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Util
         * @param {...(Function|Function[])} [iteratees=[_.identity]]
         *  The iteratees to invoke.
         * @returns {Function} Returns the new function.
         * @example
         *
         * var func = _.over([Math.max, Math.min]);
         *
         * func(1, 2, 3, 4);
         * // => [4, 1]
         */
        var over = createOver(arrayMap);

        /**
         * Creates a function that checks if **all** of the `predicates` return
         * truthy when invoked with the arguments it receives.
         *
         * Following shorthands are possible for providing predicates.
         * Pass an `Object` and it will be used as an parameter for `_.matches` to create the predicate.
         * Pass an `Array` of parameters for `_.matchesProperty` and the predicate will be created using them.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Util
         * @param {...(Function|Function[])} [predicates=[_.identity]]
         *  The predicates to check.
         * @returns {Function} Returns the new function.
         * @example
         *
         * var func = _.overEvery([Boolean, isFinite]);
         *
         * func('1');
         * // => true
         *
         * func(null);
         * // => false
         *
         * func(NaN);
         * // => false
         */
        var overEvery = createOver(arrayEvery);

        /**
         * Creates a function that checks if **any** of the `predicates` return
         * truthy when invoked with the arguments it receives.
         *
         * Following shorthands are possible for providing predicates.
         * Pass an `Object` and it will be used as an parameter for `_.matches` to create the predicate.
         * Pass an `Array` of parameters for `_.matchesProperty` and the predicate will be created using them.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Util
         * @param {...(Function|Function[])} [predicates=[_.identity]]
         *  The predicates to check.
         * @returns {Function} Returns the new function.
         * @example
         *
         * var func = _.overSome([Boolean, isFinite]);
         *
         * func('1');
         * // => true
         *
         * func(null);
         * // => true
         *
         * func(NaN);
         * // => false
         *
         * var matchesFunc = _.overSome([{ 'a': 1 }, { 'a': 2 }])
         * var matchesPropertyFunc = _.overSome([['a', 1], ['a', 2]])
         */
        var overSome = createOver(arraySome);

        /**
         * Creates a function that returns the value at `path` of a given object.
         *
         * @static
         * @memberOf _
         * @since 2.4.0
         * @category Util
         * @param {Array|string} path The path of the property to get.
         * @returns {Function} Returns the new accessor function.
         * @example
         *
         * var objects = [
         *   { 'a': { 'b': 2 } },
         *   { 'a': { 'b': 1 } }
         * ];
         *
         * _.map(objects, _.property('a.b'));
         * // => [2, 1]
         *
         * _.map(_.sortBy(objects, _.property(['a', 'b'])), 'a.b');
         * // => [1, 2]
         */
        function property(path) {
          return isKey(path) ? baseProperty(toKey(path)) : basePropertyDeep(path);
        }

        /**
         * The opposite of `_.property`; this method creates a function that returns
         * the value at a given path of `object`.
         *
         * @static
         * @memberOf _
         * @since 3.0.0
         * @category Util
         * @param {Object} object The object to query.
         * @returns {Function} Returns the new accessor function.
         * @example
         *
         * var array = [0, 1, 2],
         *     object = { 'a': array, 'b': array, 'c': array };
         *
         * _.map(['a[2]', 'c[0]'], _.propertyOf(object));
         * // => [2, 0]
         *
         * _.map([['a', '2'], ['c', '0']], _.propertyOf(object));
         * // => [2, 0]
         */
        function propertyOf(object) {
          return function(path) {
            return object == null ? undefined$1 : baseGet(object, path);
          };
        }

        /**
         * Creates an array of numbers (positive and/or negative) progressing from
         * `start` up to, but not including, `end`. A step of `-1` is used if a negative
         * `start` is specified without an `end` or `step`. If `end` is not specified,
         * it's set to `start` with `start` then set to `0`.
         *
         * **Note:** JavaScript follows the IEEE-754 standard for resolving
         * floating-point values which can produce unexpected results.
         *
         * @static
         * @since 0.1.0
         * @memberOf _
         * @category Util
         * @param {number} [start=0] The start of the range.
         * @param {number} end The end of the range.
         * @param {number} [step=1] The value to increment or decrement by.
         * @returns {Array} Returns the range of numbers.
         * @see _.inRange, _.rangeRight
         * @example
         *
         * _.range(4);
         * // => [0, 1, 2, 3]
         *
         * _.range(-4);
         * // => [0, -1, -2, -3]
         *
         * _.range(1, 5);
         * // => [1, 2, 3, 4]
         *
         * _.range(0, 20, 5);
         * // => [0, 5, 10, 15]
         *
         * _.range(0, -4, -1);
         * // => [0, -1, -2, -3]
         *
         * _.range(1, 4, 0);
         * // => [1, 1, 1]
         *
         * _.range(0);
         * // => []
         */
        var range = createRange();

        /**
         * This method is like `_.range` except that it populates values in
         * descending order.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Util
         * @param {number} [start=0] The start of the range.
         * @param {number} end The end of the range.
         * @param {number} [step=1] The value to increment or decrement by.
         * @returns {Array} Returns the range of numbers.
         * @see _.inRange, _.range
         * @example
         *
         * _.rangeRight(4);
         * // => [3, 2, 1, 0]
         *
         * _.rangeRight(-4);
         * // => [-3, -2, -1, 0]
         *
         * _.rangeRight(1, 5);
         * // => [4, 3, 2, 1]
         *
         * _.rangeRight(0, 20, 5);
         * // => [15, 10, 5, 0]
         *
         * _.rangeRight(0, -4, -1);
         * // => [-3, -2, -1, 0]
         *
         * _.rangeRight(1, 4, 0);
         * // => [1, 1, 1]
         *
         * _.rangeRight(0);
         * // => []
         */
        var rangeRight = createRange(true);

        /**
         * This method returns a new empty array.
         *
         * @static
         * @memberOf _
         * @since 4.13.0
         * @category Util
         * @returns {Array} Returns the new empty array.
         * @example
         *
         * var arrays = _.times(2, _.stubArray);
         *
         * console.log(arrays);
         * // => [[], []]
         *
         * console.log(arrays[0] === arrays[1]);
         * // => false
         */
        function stubArray() {
          return [];
        }

        /**
         * This method returns `false`.
         *
         * @static
         * @memberOf _
         * @since 4.13.0
         * @category Util
         * @returns {boolean} Returns `false`.
         * @example
         *
         * _.times(2, _.stubFalse);
         * // => [false, false]
         */
        function stubFalse() {
          return false;
        }

        /**
         * This method returns a new empty object.
         *
         * @static
         * @memberOf _
         * @since 4.13.0
         * @category Util
         * @returns {Object} Returns the new empty object.
         * @example
         *
         * var objects = _.times(2, _.stubObject);
         *
         * console.log(objects);
         * // => [{}, {}]
         *
         * console.log(objects[0] === objects[1]);
         * // => false
         */
        function stubObject() {
          return {};
        }

        /**
         * This method returns an empty string.
         *
         * @static
         * @memberOf _
         * @since 4.13.0
         * @category Util
         * @returns {string} Returns the empty string.
         * @example
         *
         * _.times(2, _.stubString);
         * // => ['', '']
         */
        function stubString() {
          return '';
        }

        /**
         * This method returns `true`.
         *
         * @static
         * @memberOf _
         * @since 4.13.0
         * @category Util
         * @returns {boolean} Returns `true`.
         * @example
         *
         * _.times(2, _.stubTrue);
         * // => [true, true]
         */
        function stubTrue() {
          return true;
        }

        /**
         * Invokes the iteratee `n` times, returning an array of the results of
         * each invocation. The iteratee is invoked with one argument; (index).
         *
         * @static
         * @since 0.1.0
         * @memberOf _
         * @category Util
         * @param {number} n The number of times to invoke `iteratee`.
         * @param {Function} [iteratee=_.identity] The function invoked per iteration.
         * @returns {Array} Returns the array of results.
         * @example
         *
         * _.times(3, String);
         * // => ['0', '1', '2']
         *
         *  _.times(4, _.constant(0));
         * // => [0, 0, 0, 0]
         */
        function times(n, iteratee) {
          n = toInteger(n);
          if (n < 1 || n > MAX_SAFE_INTEGER) {
            return [];
          }
          var index = MAX_ARRAY_LENGTH,
              length = nativeMin(n, MAX_ARRAY_LENGTH);

          iteratee = getIteratee(iteratee);
          n -= MAX_ARRAY_LENGTH;

          var result = baseTimes(length, iteratee);
          while (++index < n) {
            iteratee(index);
          }
          return result;
        }

        /**
         * Converts `value` to a property path array.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Util
         * @param {*} value The value to convert.
         * @returns {Array} Returns the new property path array.
         * @example
         *
         * _.toPath('a.b.c');
         * // => ['a', 'b', 'c']
         *
         * _.toPath('a[0].b.c');
         * // => ['a', '0', 'b', 'c']
         */
        function toPath(value) {
          if (isArray(value)) {
            return arrayMap(value, toKey);
          }
          return isSymbol(value) ? [value] : copyArray(stringToPath(toString(value)));
        }

        /**
         * Generates a unique ID. If `prefix` is given, the ID is appended to it.
         *
         * @static
         * @since 0.1.0
         * @memberOf _
         * @category Util
         * @param {string} [prefix=''] The value to prefix the ID with.
         * @returns {string} Returns the unique ID.
         * @example
         *
         * _.uniqueId('contact_');
         * // => 'contact_104'
         *
         * _.uniqueId();
         * // => '105'
         */
        function uniqueId(prefix) {
          var id = ++idCounter;
          return toString(prefix) + id;
        }

        /*------------------------------------------------------------------------*/

        /**
         * Adds two numbers.
         *
         * @static
         * @memberOf _
         * @since 3.4.0
         * @category Math
         * @param {number} augend The first number in an addition.
         * @param {number} addend The second number in an addition.
         * @returns {number} Returns the total.
         * @example
         *
         * _.add(6, 4);
         * // => 10
         */
        var add = createMathOperation(function(augend, addend) {
          return augend + addend;
        }, 0);

        /**
         * Computes `number` rounded up to `precision`.
         *
         * @static
         * @memberOf _
         * @since 3.10.0
         * @category Math
         * @param {number} number The number to round up.
         * @param {number} [precision=0] The precision to round up to.
         * @returns {number} Returns the rounded up number.
         * @example
         *
         * _.ceil(4.006);
         * // => 5
         *
         * _.ceil(6.004, 2);
         * // => 6.01
         *
         * _.ceil(6040, -2);
         * // => 6100
         */
        var ceil = createRound('ceil');

        /**
         * Divide two numbers.
         *
         * @static
         * @memberOf _
         * @since 4.7.0
         * @category Math
         * @param {number} dividend The first number in a division.
         * @param {number} divisor The second number in a division.
         * @returns {number} Returns the quotient.
         * @example
         *
         * _.divide(6, 4);
         * // => 1.5
         */
        var divide = createMathOperation(function(dividend, divisor) {
          return dividend / divisor;
        }, 1);

        /**
         * Computes `number` rounded down to `precision`.
         *
         * @static
         * @memberOf _
         * @since 3.10.0
         * @category Math
         * @param {number} number The number to round down.
         * @param {number} [precision=0] The precision to round down to.
         * @returns {number} Returns the rounded down number.
         * @example
         *
         * _.floor(4.006);
         * // => 4
         *
         * _.floor(0.046, 2);
         * // => 0.04
         *
         * _.floor(4060, -2);
         * // => 4000
         */
        var floor = createRound('floor');

        /**
         * Computes the maximum value of `array`. If `array` is empty or falsey,
         * `undefined` is returned.
         *
         * @static
         * @since 0.1.0
         * @memberOf _
         * @category Math
         * @param {Array} array The array to iterate over.
         * @returns {*} Returns the maximum value.
         * @example
         *
         * _.max([4, 2, 8, 6]);
         * // => 8
         *
         * _.max([]);
         * // => undefined
         */
        function max(array) {
          return (array && array.length)
            ? baseExtremum(array, identity, baseGt)
            : undefined$1;
        }

        /**
         * This method is like `_.max` except that it accepts `iteratee` which is
         * invoked for each element in `array` to generate the criterion by which
         * the value is ranked. The iteratee is invoked with one argument: (value).
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Math
         * @param {Array} array The array to iterate over.
         * @param {Function} [iteratee=_.identity] The iteratee invoked per element.
         * @returns {*} Returns the maximum value.
         * @example
         *
         * var objects = [{ 'n': 1 }, { 'n': 2 }];
         *
         * _.maxBy(objects, function(o) { return o.n; });
         * // => { 'n': 2 }
         *
         * // The `_.property` iteratee shorthand.
         * _.maxBy(objects, 'n');
         * // => { 'n': 2 }
         */
        function maxBy(array, iteratee) {
          return (array && array.length)
            ? baseExtremum(array, getIteratee(iteratee, 2), baseGt)
            : undefined$1;
        }

        /**
         * Computes the mean of the values in `array`.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Math
         * @param {Array} array The array to iterate over.
         * @returns {number} Returns the mean.
         * @example
         *
         * _.mean([4, 2, 8, 6]);
         * // => 5
         */
        function mean(array) {
          return baseMean(array, identity);
        }

        /**
         * This method is like `_.mean` except that it accepts `iteratee` which is
         * invoked for each element in `array` to generate the value to be averaged.
         * The iteratee is invoked with one argument: (value).
         *
         * @static
         * @memberOf _
         * @since 4.7.0
         * @category Math
         * @param {Array} array The array to iterate over.
         * @param {Function} [iteratee=_.identity] The iteratee invoked per element.
         * @returns {number} Returns the mean.
         * @example
         *
         * var objects = [{ 'n': 4 }, { 'n': 2 }, { 'n': 8 }, { 'n': 6 }];
         *
         * _.meanBy(objects, function(o) { return o.n; });
         * // => 5
         *
         * // The `_.property` iteratee shorthand.
         * _.meanBy(objects, 'n');
         * // => 5
         */
        function meanBy(array, iteratee) {
          return baseMean(array, getIteratee(iteratee, 2));
        }

        /**
         * Computes the minimum value of `array`. If `array` is empty or falsey,
         * `undefined` is returned.
         *
         * @static
         * @since 0.1.0
         * @memberOf _
         * @category Math
         * @param {Array} array The array to iterate over.
         * @returns {*} Returns the minimum value.
         * @example
         *
         * _.min([4, 2, 8, 6]);
         * // => 2
         *
         * _.min([]);
         * // => undefined
         */
        function min(array) {
          return (array && array.length)
            ? baseExtremum(array, identity, baseLt)
            : undefined$1;
        }

        /**
         * This method is like `_.min` except that it accepts `iteratee` which is
         * invoked for each element in `array` to generate the criterion by which
         * the value is ranked. The iteratee is invoked with one argument: (value).
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Math
         * @param {Array} array The array to iterate over.
         * @param {Function} [iteratee=_.identity] The iteratee invoked per element.
         * @returns {*} Returns the minimum value.
         * @example
         *
         * var objects = [{ 'n': 1 }, { 'n': 2 }];
         *
         * _.minBy(objects, function(o) { return o.n; });
         * // => { 'n': 1 }
         *
         * // The `_.property` iteratee shorthand.
         * _.minBy(objects, 'n');
         * // => { 'n': 1 }
         */
        function minBy(array, iteratee) {
          return (array && array.length)
            ? baseExtremum(array, getIteratee(iteratee, 2), baseLt)
            : undefined$1;
        }

        /**
         * Multiply two numbers.
         *
         * @static
         * @memberOf _
         * @since 4.7.0
         * @category Math
         * @param {number} multiplier The first number in a multiplication.
         * @param {number} multiplicand The second number in a multiplication.
         * @returns {number} Returns the product.
         * @example
         *
         * _.multiply(6, 4);
         * // => 24
         */
        var multiply = createMathOperation(function(multiplier, multiplicand) {
          return multiplier * multiplicand;
        }, 1);

        /**
         * Computes `number` rounded to `precision`.
         *
         * @static
         * @memberOf _
         * @since 3.10.0
         * @category Math
         * @param {number} number The number to round.
         * @param {number} [precision=0] The precision to round to.
         * @returns {number} Returns the rounded number.
         * @example
         *
         * _.round(4.006);
         * // => 4
         *
         * _.round(4.006, 2);
         * // => 4.01
         *
         * _.round(4060, -2);
         * // => 4100
         */
        var round = createRound('round');

        /**
         * Subtract two numbers.
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Math
         * @param {number} minuend The first number in a subtraction.
         * @param {number} subtrahend The second number in a subtraction.
         * @returns {number} Returns the difference.
         * @example
         *
         * _.subtract(6, 4);
         * // => 2
         */
        var subtract = createMathOperation(function(minuend, subtrahend) {
          return minuend - subtrahend;
        }, 0);

        /**
         * Computes the sum of the values in `array`.
         *
         * @static
         * @memberOf _
         * @since 3.4.0
         * @category Math
         * @param {Array} array The array to iterate over.
         * @returns {number} Returns the sum.
         * @example
         *
         * _.sum([4, 2, 8, 6]);
         * // => 20
         */
        function sum(array) {
          return (array && array.length)
            ? baseSum(array, identity)
            : 0;
        }

        /**
         * This method is like `_.sum` except that it accepts `iteratee` which is
         * invoked for each element in `array` to generate the value to be summed.
         * The iteratee is invoked with one argument: (value).
         *
         * @static
         * @memberOf _
         * @since 4.0.0
         * @category Math
         * @param {Array} array The array to iterate over.
         * @param {Function} [iteratee=_.identity] The iteratee invoked per element.
         * @returns {number} Returns the sum.
         * @example
         *
         * var objects = [{ 'n': 4 }, { 'n': 2 }, { 'n': 8 }, { 'n': 6 }];
         *
         * _.sumBy(objects, function(o) { return o.n; });
         * // => 20
         *
         * // The `_.property` iteratee shorthand.
         * _.sumBy(objects, 'n');
         * // => 20
         */
        function sumBy(array, iteratee) {
          return (array && array.length)
            ? baseSum(array, getIteratee(iteratee, 2))
            : 0;
        }

        /*------------------------------------------------------------------------*/

        // Add methods that return wrapped values in chain sequences.
        lodash.after = after;
        lodash.ary = ary;
        lodash.assign = assign;
        lodash.assignIn = assignIn;
        lodash.assignInWith = assignInWith;
        lodash.assignWith = assignWith;
        lodash.at = at;
        lodash.before = before;
        lodash.bind = bind;
        lodash.bindAll = bindAll;
        lodash.bindKey = bindKey;
        lodash.castArray = castArray;
        lodash.chain = chain;
        lodash.chunk = chunk;
        lodash.compact = compact;
        lodash.concat = concat;
        lodash.cond = cond;
        lodash.conforms = conforms;
        lodash.constant = constant;
        lodash.countBy = countBy;
        lodash.create = create;
        lodash.curry = curry;
        lodash.curryRight = curryRight;
        lodash.debounce = debounce;
        lodash.defaults = defaults;
        lodash.defaultsDeep = defaultsDeep;
        lodash.defer = defer;
        lodash.delay = delay;
        lodash.difference = difference;
        lodash.differenceBy = differenceBy;
        lodash.differenceWith = differenceWith;
        lodash.drop = drop;
        lodash.dropRight = dropRight;
        lodash.dropRightWhile = dropRightWhile;
        lodash.dropWhile = dropWhile;
        lodash.fill = fill;
        lodash.filter = filter;
        lodash.flatMap = flatMap;
        lodash.flatMapDeep = flatMapDeep;
        lodash.flatMapDepth = flatMapDepth;
        lodash.flatten = flatten;
        lodash.flattenDeep = flattenDeep;
        lodash.flattenDepth = flattenDepth;
        lodash.flip = flip;
        lodash.flow = flow;
        lodash.flowRight = flowRight;
        lodash.fromPairs = fromPairs;
        lodash.functions = functions;
        lodash.functionsIn = functionsIn;
        lodash.groupBy = groupBy;
        lodash.initial = initial;
        lodash.intersection = intersection;
        lodash.intersectionBy = intersectionBy;
        lodash.intersectionWith = intersectionWith;
        lodash.invert = invert;
        lodash.invertBy = invertBy;
        lodash.invokeMap = invokeMap;
        lodash.iteratee = iteratee;
        lodash.keyBy = keyBy;
        lodash.keys = keys;
        lodash.keysIn = keysIn;
        lodash.map = map;
        lodash.mapKeys = mapKeys;
        lodash.mapValues = mapValues;
        lodash.matches = matches;
        lodash.matchesProperty = matchesProperty;
        lodash.memoize = memoize;
        lodash.merge = merge;
        lodash.mergeWith = mergeWith;
        lodash.method = method;
        lodash.methodOf = methodOf;
        lodash.mixin = mixin;
        lodash.negate = negate;
        lodash.nthArg = nthArg;
        lodash.omit = omit;
        lodash.omitBy = omitBy;
        lodash.once = once;
        lodash.orderBy = orderBy;
        lodash.over = over;
        lodash.overArgs = overArgs;
        lodash.overEvery = overEvery;
        lodash.overSome = overSome;
        lodash.partial = partial;
        lodash.partialRight = partialRight;
        lodash.partition = partition;
        lodash.pick = pick;
        lodash.pickBy = pickBy;
        lodash.property = property;
        lodash.propertyOf = propertyOf;
        lodash.pull = pull;
        lodash.pullAll = pullAll;
        lodash.pullAllBy = pullAllBy;
        lodash.pullAllWith = pullAllWith;
        lodash.pullAt = pullAt;
        lodash.range = range;
        lodash.rangeRight = rangeRight;
        lodash.rearg = rearg;
        lodash.reject = reject;
        lodash.remove = remove;
        lodash.rest = rest;
        lodash.reverse = reverse;
        lodash.sampleSize = sampleSize;
        lodash.set = set;
        lodash.setWith = setWith;
        lodash.shuffle = shuffle;
        lodash.slice = slice;
        lodash.sortBy = sortBy;
        lodash.sortedUniq = sortedUniq;
        lodash.sortedUniqBy = sortedUniqBy;
        lodash.split = split;
        lodash.spread = spread;
        lodash.tail = tail;
        lodash.take = take;
        lodash.takeRight = takeRight;
        lodash.takeRightWhile = takeRightWhile;
        lodash.takeWhile = takeWhile;
        lodash.tap = tap;
        lodash.throttle = throttle;
        lodash.thru = thru;
        lodash.toArray = toArray;
        lodash.toPairs = toPairs;
        lodash.toPairsIn = toPairsIn;
        lodash.toPath = toPath;
        lodash.toPlainObject = toPlainObject;
        lodash.transform = transform;
        lodash.unary = unary;
        lodash.union = union;
        lodash.unionBy = unionBy;
        lodash.unionWith = unionWith;
        lodash.uniq = uniq;
        lodash.uniqBy = uniqBy;
        lodash.uniqWith = uniqWith;
        lodash.unset = unset;
        lodash.unzip = unzip;
        lodash.unzipWith = unzipWith;
        lodash.update = update;
        lodash.updateWith = updateWith;
        lodash.values = values;
        lodash.valuesIn = valuesIn;
        lodash.without = without;
        lodash.words = words;
        lodash.wrap = wrap;
        lodash.xor = xor;
        lodash.xorBy = xorBy;
        lodash.xorWith = xorWith;
        lodash.zip = zip;
        lodash.zipObject = zipObject;
        lodash.zipObjectDeep = zipObjectDeep;
        lodash.zipWith = zipWith;

        // Add aliases.
        lodash.entries = toPairs;
        lodash.entriesIn = toPairsIn;
        lodash.extend = assignIn;
        lodash.extendWith = assignInWith;

        // Add methods to `lodash.prototype`.
        mixin(lodash, lodash);

        /*------------------------------------------------------------------------*/

        // Add methods that return unwrapped values in chain sequences.
        lodash.add = add;
        lodash.attempt = attempt;
        lodash.camelCase = camelCase;
        lodash.capitalize = capitalize;
        lodash.ceil = ceil;
        lodash.clamp = clamp;
        lodash.clone = clone;
        lodash.cloneDeep = cloneDeep;
        lodash.cloneDeepWith = cloneDeepWith;
        lodash.cloneWith = cloneWith;
        lodash.conformsTo = conformsTo;
        lodash.deburr = deburr;
        lodash.defaultTo = defaultTo;
        lodash.divide = divide;
        lodash.endsWith = endsWith;
        lodash.eq = eq;
        lodash.escape = escape;
        lodash.escapeRegExp = escapeRegExp;
        lodash.every = every;
        lodash.find = find;
        lodash.findIndex = findIndex;
        lodash.findKey = findKey;
        lodash.findLast = findLast;
        lodash.findLastIndex = findLastIndex;
        lodash.findLastKey = findLastKey;
        lodash.floor = floor;
        lodash.forEach = forEach;
        lodash.forEachRight = forEachRight;
        lodash.forIn = forIn;
        lodash.forInRight = forInRight;
        lodash.forOwn = forOwn;
        lodash.forOwnRight = forOwnRight;
        lodash.get = get;
        lodash.gt = gt;
        lodash.gte = gte;
        lodash.has = has;
        lodash.hasIn = hasIn;
        lodash.head = head;
        lodash.identity = identity;
        lodash.includes = includes;
        lodash.indexOf = indexOf;
        lodash.inRange = inRange;
        lodash.invoke = invoke;
        lodash.isArguments = isArguments;
        lodash.isArray = isArray;
        lodash.isArrayBuffer = isArrayBuffer;
        lodash.isArrayLike = isArrayLike;
        lodash.isArrayLikeObject = isArrayLikeObject;
        lodash.isBoolean = isBoolean;
        lodash.isBuffer = isBuffer;
        lodash.isDate = isDate;
        lodash.isElement = isElement;
        lodash.isEmpty = isEmpty;
        lodash.isEqual = isEqual;
        lodash.isEqualWith = isEqualWith;
        lodash.isError = isError;
        lodash.isFinite = isFinite;
        lodash.isFunction = isFunction;
        lodash.isInteger = isInteger;
        lodash.isLength = isLength;
        lodash.isMap = isMap;
        lodash.isMatch = isMatch;
        lodash.isMatchWith = isMatchWith;
        lodash.isNaN = isNaN;
        lodash.isNative = isNative;
        lodash.isNil = isNil;
        lodash.isNull = isNull;
        lodash.isNumber = isNumber;
        lodash.isObject = isObject;
        lodash.isObjectLike = isObjectLike;
        lodash.isPlainObject = isPlainObject;
        lodash.isRegExp = isRegExp;
        lodash.isSafeInteger = isSafeInteger;
        lodash.isSet = isSet;
        lodash.isString = isString;
        lodash.isSymbol = isSymbol;
        lodash.isTypedArray = isTypedArray;
        lodash.isUndefined = isUndefined;
        lodash.isWeakMap = isWeakMap;
        lodash.isWeakSet = isWeakSet;
        lodash.join = join;
        lodash.kebabCase = kebabCase;
        lodash.last = last;
        lodash.lastIndexOf = lastIndexOf;
        lodash.lowerCase = lowerCase;
        lodash.lowerFirst = lowerFirst;
        lodash.lt = lt;
        lodash.lte = lte;
        lodash.max = max;
        lodash.maxBy = maxBy;
        lodash.mean = mean;
        lodash.meanBy = meanBy;
        lodash.min = min;
        lodash.minBy = minBy;
        lodash.stubArray = stubArray;
        lodash.stubFalse = stubFalse;
        lodash.stubObject = stubObject;
        lodash.stubString = stubString;
        lodash.stubTrue = stubTrue;
        lodash.multiply = multiply;
        lodash.nth = nth;
        lodash.noConflict = noConflict;
        lodash.noop = noop;
        lodash.now = now;
        lodash.pad = pad;
        lodash.padEnd = padEnd;
        lodash.padStart = padStart;
        lodash.parseInt = parseInt;
        lodash.random = random;
        lodash.reduce = reduce;
        lodash.reduceRight = reduceRight;
        lodash.repeat = repeat;
        lodash.replace = replace;
        lodash.result = result;
        lodash.round = round;
        lodash.runInContext = runInContext;
        lodash.sample = sample;
        lodash.size = size;
        lodash.snakeCase = snakeCase;
        lodash.some = some;
        lodash.sortedIndex = sortedIndex;
        lodash.sortedIndexBy = sortedIndexBy;
        lodash.sortedIndexOf = sortedIndexOf;
        lodash.sortedLastIndex = sortedLastIndex;
        lodash.sortedLastIndexBy = sortedLastIndexBy;
        lodash.sortedLastIndexOf = sortedLastIndexOf;
        lodash.startCase = startCase;
        lodash.startsWith = startsWith;
        lodash.subtract = subtract;
        lodash.sum = sum;
        lodash.sumBy = sumBy;
        lodash.template = template;
        lodash.times = times;
        lodash.toFinite = toFinite;
        lodash.toInteger = toInteger;
        lodash.toLength = toLength;
        lodash.toLower = toLower;
        lodash.toNumber = toNumber;
        lodash.toSafeInteger = toSafeInteger;
        lodash.toString = toString;
        lodash.toUpper = toUpper;
        lodash.trim = trim;
        lodash.trimEnd = trimEnd;
        lodash.trimStart = trimStart;
        lodash.truncate = truncate;
        lodash.unescape = unescape;
        lodash.uniqueId = uniqueId;
        lodash.upperCase = upperCase;
        lodash.upperFirst = upperFirst;

        // Add aliases.
        lodash.each = forEach;
        lodash.eachRight = forEachRight;
        lodash.first = head;

        mixin(lodash, (function() {
          var source = {};
          baseForOwn(lodash, function(func, methodName) {
            if (!hasOwnProperty.call(lodash.prototype, methodName)) {
              source[methodName] = func;
            }
          });
          return source;
        }()), { 'chain': false });

        /*------------------------------------------------------------------------*/

        /**
         * The semantic version number.
         *
         * @static
         * @memberOf _
         * @type {string}
         */
        lodash.VERSION = VERSION;

        // Assign default placeholders.
        arrayEach(['bind', 'bindKey', 'curry', 'curryRight', 'partial', 'partialRight'], function(methodName) {
          lodash[methodName].placeholder = lodash;
        });

        // Add `LazyWrapper` methods for `_.drop` and `_.take` variants.
        arrayEach(['drop', 'take'], function(methodName, index) {
          LazyWrapper.prototype[methodName] = function(n) {
            n = n === undefined$1 ? 1 : nativeMax(toInteger(n), 0);

            var result = (this.__filtered__ && !index)
              ? new LazyWrapper(this)
              : this.clone();

            if (result.__filtered__) {
              result.__takeCount__ = nativeMin(n, result.__takeCount__);
            } else {
              result.__views__.push({
                'size': nativeMin(n, MAX_ARRAY_LENGTH),
                'type': methodName + (result.__dir__ < 0 ? 'Right' : '')
              });
            }
            return result;
          };

          LazyWrapper.prototype[methodName + 'Right'] = function(n) {
            return this.reverse()[methodName](n).reverse();
          };
        });

        // Add `LazyWrapper` methods that accept an `iteratee` value.
        arrayEach(['filter', 'map', 'takeWhile'], function(methodName, index) {
          var type = index + 1,
              isFilter = type == LAZY_FILTER_FLAG || type == LAZY_WHILE_FLAG;

          LazyWrapper.prototype[methodName] = function(iteratee) {
            var result = this.clone();
            result.__iteratees__.push({
              'iteratee': getIteratee(iteratee, 3),
              'type': type
            });
            result.__filtered__ = result.__filtered__ || isFilter;
            return result;
          };
        });

        // Add `LazyWrapper` methods for `_.head` and `_.last`.
        arrayEach(['head', 'last'], function(methodName, index) {
          var takeName = 'take' + (index ? 'Right' : '');

          LazyWrapper.prototype[methodName] = function() {
            return this[takeName](1).value()[0];
          };
        });

        // Add `LazyWrapper` methods for `_.initial` and `_.tail`.
        arrayEach(['initial', 'tail'], function(methodName, index) {
          var dropName = 'drop' + (index ? '' : 'Right');

          LazyWrapper.prototype[methodName] = function() {
            return this.__filtered__ ? new LazyWrapper(this) : this[dropName](1);
          };
        });

        LazyWrapper.prototype.compact = function() {
          return this.filter(identity);
        };

        LazyWrapper.prototype.find = function(predicate) {
          return this.filter(predicate).head();
        };

        LazyWrapper.prototype.findLast = function(predicate) {
          return this.reverse().find(predicate);
        };

        LazyWrapper.prototype.invokeMap = baseRest(function(path, args) {
          if (typeof path == 'function') {
            return new LazyWrapper(this);
          }
          return this.map(function(value) {
            return baseInvoke(value, path, args);
          });
        });

        LazyWrapper.prototype.reject = function(predicate) {
          return this.filter(negate(getIteratee(predicate)));
        };

        LazyWrapper.prototype.slice = function(start, end) {
          start = toInteger(start);

          var result = this;
          if (result.__filtered__ && (start > 0 || end < 0)) {
            return new LazyWrapper(result);
          }
          if (start < 0) {
            result = result.takeRight(-start);
          } else if (start) {
            result = result.drop(start);
          }
          if (end !== undefined$1) {
            end = toInteger(end);
            result = end < 0 ? result.dropRight(-end) : result.take(end - start);
          }
          return result;
        };

        LazyWrapper.prototype.takeRightWhile = function(predicate) {
          return this.reverse().takeWhile(predicate).reverse();
        };

        LazyWrapper.prototype.toArray = function() {
          return this.take(MAX_ARRAY_LENGTH);
        };

        // Add `LazyWrapper` methods to `lodash.prototype`.
        baseForOwn(LazyWrapper.prototype, function(func, methodName) {
          var checkIteratee = /^(?:filter|find|map|reject)|While$/.test(methodName),
              isTaker = /^(?:head|last)$/.test(methodName),
              lodashFunc = lodash[isTaker ? ('take' + (methodName == 'last' ? 'Right' : '')) : methodName],
              retUnwrapped = isTaker || /^find/.test(methodName);

          if (!lodashFunc) {
            return;
          }
          lodash.prototype[methodName] = function() {
            var value = this.__wrapped__,
                args = isTaker ? [1] : arguments,
                isLazy = value instanceof LazyWrapper,
                iteratee = args[0],
                useLazy = isLazy || isArray(value);

            var interceptor = function(value) {
              var result = lodashFunc.apply(lodash, arrayPush([value], args));
              return (isTaker && chainAll) ? result[0] : result;
            };

            if (useLazy && checkIteratee && typeof iteratee == 'function' && iteratee.length != 1) {
              // Avoid lazy use if the iteratee has a "length" value other than `1`.
              isLazy = useLazy = false;
            }
            var chainAll = this.__chain__,
                isHybrid = !!this.__actions__.length,
                isUnwrapped = retUnwrapped && !chainAll,
                onlyLazy = isLazy && !isHybrid;

            if (!retUnwrapped && useLazy) {
              value = onlyLazy ? value : new LazyWrapper(this);
              var result = func.apply(value, args);
              result.__actions__.push({ 'func': thru, 'args': [interceptor], 'thisArg': undefined$1 });
              return new LodashWrapper(result, chainAll);
            }
            if (isUnwrapped && onlyLazy) {
              return func.apply(this, args);
            }
            result = this.thru(interceptor);
            return isUnwrapped ? (isTaker ? result.value()[0] : result.value()) : result;
          };
        });

        // Add `Array` methods to `lodash.prototype`.
        arrayEach(['pop', 'push', 'shift', 'sort', 'splice', 'unshift'], function(methodName) {
          var func = arrayProto[methodName],
              chainName = /^(?:push|sort|unshift)$/.test(methodName) ? 'tap' : 'thru',
              retUnwrapped = /^(?:pop|shift)$/.test(methodName);

          lodash.prototype[methodName] = function() {
            var args = arguments;
            if (retUnwrapped && !this.__chain__) {
              var value = this.value();
              return func.apply(isArray(value) ? value : [], args);
            }
            return this[chainName](function(value) {
              return func.apply(isArray(value) ? value : [], args);
            });
          };
        });

        // Map minified method names to their real names.
        baseForOwn(LazyWrapper.prototype, function(func, methodName) {
          var lodashFunc = lodash[methodName];
          if (lodashFunc) {
            var key = lodashFunc.name + '';
            if (!hasOwnProperty.call(realNames, key)) {
              realNames[key] = [];
            }
            realNames[key].push({ 'name': methodName, 'func': lodashFunc });
          }
        });

        realNames[createHybrid(undefined$1, WRAP_BIND_KEY_FLAG).name] = [{
          'name': 'wrapper',
          'func': undefined$1
        }];

        // Add methods to `LazyWrapper`.
        LazyWrapper.prototype.clone = lazyClone;
        LazyWrapper.prototype.reverse = lazyReverse;
        LazyWrapper.prototype.value = lazyValue;

        // Add chain sequence methods to the `lodash` wrapper.
        lodash.prototype.at = wrapperAt;
        lodash.prototype.chain = wrapperChain;
        lodash.prototype.commit = wrapperCommit;
        lodash.prototype.next = wrapperNext;
        lodash.prototype.plant = wrapperPlant;
        lodash.prototype.reverse = wrapperReverse;
        lodash.prototype.toJSON = lodash.prototype.valueOf = lodash.prototype.value = wrapperValue;

        // Add lazy aliases.
        lodash.prototype.first = lodash.prototype.head;

        if (symIterator) {
          lodash.prototype[symIterator] = wrapperToIterator;
        }
        return lodash;
      });

      /*--------------------------------------------------------------------------*/

      // Export lodash.
      var _ = runInContext();

      // Some AMD build optimizers, like r.js, check for condition patterns like:
      if (freeModule) {
        // Export for Node.js.
        (freeModule.exports = _)._ = _;
        // Export for CommonJS support.
        freeExports._ = _;
      }
      else {
        // Export to the global object.
        root._ = _;
      }
    }.call(commonjsGlobal));
    });

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    const nativeShadow = !(window['ShadyDOM'] && window['ShadyDOM']['inUse']);
    let nativeCssVariables_;

    /**
     * @param {(ShadyCSSOptions | ShadyCSSInterface)=} settings
     */
    function calcCssVariables(settings) {
      if (settings && settings['shimcssproperties']) {
        nativeCssVariables_ = false;
      } else {
        // chrome 49 has semi-working css vars, check if box-shadow works
        // safari 9.1 has a recalc bug: https://bugs.webkit.org/show_bug.cgi?id=155782
        // However, shim css custom properties are only supported with ShadyDOM enabled,
        // so fall back on native if we do not detect ShadyDOM
        // Edge 15: custom properties used in ::before and ::after will also be used in the parent element
        // https://developer.microsoft.com/en-us/microsoft-edge/platform/issues/12414257/
        nativeCssVariables_ = nativeShadow || Boolean(!navigator.userAgent.match(/AppleWebKit\/601|Edge\/15/) &&
          window.CSS && CSS.supports && CSS.supports('box-shadow', '0 0 0 var(--foo)'));
      }
    }

    /** @type {string | undefined} */
    let cssBuild;
    if (window.ShadyCSS && window.ShadyCSS.cssBuild !== undefined) {
      cssBuild = window.ShadyCSS.cssBuild;
    }

    /** @type {boolean} */
    const disableRuntime = Boolean(window.ShadyCSS && window.ShadyCSS.disableRuntime);

    if (window.ShadyCSS && window.ShadyCSS.nativeCss !== undefined) {
      nativeCssVariables_ = window.ShadyCSS.nativeCss;
    } else if (window.ShadyCSS) {
      calcCssVariables(window.ShadyCSS);
      // reset window variable to let ShadyCSS API take its place
      window.ShadyCSS = undefined;
    } else {
      calcCssVariables(window['WebComponents'] && window['WebComponents']['flags']);
    }

    // Hack for type error under new type inference which doesn't like that
    // nativeCssVariables is updated in a function and assigns the type
    // `function(): ?` instead of `boolean`.
    const nativeCssVariables = /** @type {boolean} */(nativeCssVariables_);

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    /** @unrestricted */
    class StyleNode {
      constructor() {
        /** @type {number} */
        this['start'] = 0;
        /** @type {number} */
        this['end'] = 0;
        /** @type {StyleNode} */
        this['previous'] = null;
        /** @type {StyleNode} */
        this['parent'] = null;
        /** @type {Array<StyleNode>} */
        this['rules'] = null;
        /** @type {string} */
        this['parsedCssText'] = '';
        /** @type {string} */
        this['cssText'] = '';
        /** @type {boolean} */
        this['atRule'] = false;
        /** @type {number} */
        this['type'] = 0;
        /** @type {string} */
        this['keyframesName'] = '';
        /** @type {string} */
        this['selector'] = '';
        /** @type {string} */
        this['parsedSelector'] = '';
      }
    }

    // given a string of css, return a simple rule tree
    /**
     * @param {string} text
     * @return {StyleNode}
     */
    function parse(text) {
      text = clean(text);
      return parseCss(lex(text), text);
    }

    // remove stuff we don't care about that may hinder parsing
    /**
     * @param {string} cssText
     * @return {string}
     */
    function clean(cssText) {
      return cssText.replace(RX.comments, '').replace(RX.port, '');
    }

    // super simple {...} lexer that returns a node tree
    /**
     * @param {string} text
     * @return {!StyleNode}
     */
    function lex(text) {
      let root = new StyleNode();
      root['start'] = 0;
      root['end'] = text.length;
      let n = root;
      for (let i = 0, l = text.length; i < l; i++) {
        if (text[i] === OPEN_BRACE) {
          if (!n['rules']) {
            n['rules'] = [];
          }
          let p = n;
          let previous = p['rules'][p['rules'].length - 1] || null;
          n = new StyleNode();
          n['start'] = i + 1;
          n['parent'] = p;
          n['previous'] = previous;
          p['rules'].push(n);
        } else if (text[i] === CLOSE_BRACE) {
          n['end'] = i + 1;
          n = n['parent'] || root;
        }
      }
      return root;
    }

    // add selectors/cssText to node tree
    /**
     * @param {StyleNode} node
     * @param {string} text
     * @return {!StyleNode}
     */
    function parseCss(node, text) {
      let t = text.substring(node['start'], node['end'] - 1);
      node['parsedCssText'] = node['cssText'] = t.trim();
      if (node['parent']) {
        let ss = node['previous'] ? node['previous']['end'] : node['parent']['start'];
        t = text.substring(ss, node['start'] - 1);
        t = _expandUnicodeEscapes(t);
        t = t.replace(RX.multipleSpaces, ' ');
        // TODO(sorvell): ad hoc; make selector include only after last ;
        // helps with mixin syntax
        t = t.substring(t.lastIndexOf(';') + 1);
        let s = node['parsedSelector'] = node['selector'] = t.trim();
        node['atRule'] = (s.indexOf(AT_START) === 0);
        // note, support a subset of rule types...
        if (node['atRule']) {
          if (s.indexOf(MEDIA_START) === 0) {
            node['type'] = types.MEDIA_RULE;
          } else if (s.match(RX.keyframesRule)) {
            node['type'] = types.KEYFRAMES_RULE;
            node['keyframesName'] =
              node['selector'].split(RX.multipleSpaces).pop();
          }
        } else {
          if (s.indexOf(VAR_START) === 0) {
            node['type'] = types.MIXIN_RULE;
          } else {
            node['type'] = types.STYLE_RULE;
          }
        }
      }
      let r$ = node['rules'];
      if (r$) {
        for (let i = 0, l = r$.length, r;
          (i < l) && (r = r$[i]); i++) {
          parseCss(r, text);
        }
      }
      return node;
    }

    /**
     * conversion of sort unicode escapes with spaces like `\33 ` (and longer) into
     * expanded form that doesn't require trailing space `\000033`
     * @param {string} s
     * @return {string}
     */
    function _expandUnicodeEscapes(s) {
      return s.replace(/\\([0-9a-f]{1,6})\s/gi, function() {
        let code = arguments[1],
          repeat = 6 - code.length;
        while (repeat--) {
          code = '0' + code;
        }
        return '\\' + code;
      });
    }

    /**
     * stringify parsed css.
     * @param {StyleNode} node
     * @param {boolean=} preserveProperties
     * @param {string=} text
     * @return {string}
     */
    function stringify(node, preserveProperties, text = '') {
      // calc rule cssText
      let cssText = '';
      if (node['cssText'] || node['rules']) {
        let r$ = node['rules'];
        if (r$ && !_hasMixinRules(r$)) {
          for (let i = 0, l = r$.length, r;
            (i < l) && (r = r$[i]); i++) {
            cssText = stringify(r, preserveProperties, cssText);
          }
        } else {
          cssText = preserveProperties ? node['cssText'] :
            removeCustomProps(node['cssText']);
          cssText = cssText.trim();
          if (cssText) {
            cssText = '  ' + cssText + '\n';
          }
        }
      }
      // emit rule if there is cssText
      if (cssText) {
        if (node['selector']) {
          text += node['selector'] + ' ' + OPEN_BRACE + '\n';
        }
        text += cssText;
        if (node['selector']) {
          text += CLOSE_BRACE + '\n\n';
        }
      }
      return text;
    }

    /**
     * @param {Array<StyleNode>} rules
     * @return {boolean}
     */
    function _hasMixinRules(rules) {
      let r = rules[0];
      return Boolean(r) && Boolean(r['selector']) && r['selector'].indexOf(VAR_START) === 0;
    }

    /**
     * @param {string} cssText
     * @return {string}
     */
    function removeCustomProps(cssText) {
      cssText = removeCustomPropAssignment(cssText);
      return removeCustomPropApply(cssText);
    }

    /**
     * @param {string} cssText
     * @return {string}
     */
    function removeCustomPropAssignment(cssText) {
      return cssText
        .replace(RX.customProp, '')
        .replace(RX.mixinProp, '');
    }

    /**
     * @param {string} cssText
     * @return {string}
     */
    function removeCustomPropApply(cssText) {
      return cssText
        .replace(RX.mixinApply, '')
        .replace(RX.varApply, '');
    }

    /** @enum {number} */
    const types = {
      STYLE_RULE: 1,
      KEYFRAMES_RULE: 7,
      MEDIA_RULE: 4,
      MIXIN_RULE: 1000
    };

    const OPEN_BRACE = '{';
    const CLOSE_BRACE = '}';

    // helper regexp's
    const RX = {
      comments: /\/\*[^*]*\*+([^/*][^*]*\*+)*\//gim,
      port: /@import[^;]*;/gim,
      customProp: /(?:^[^;\-\s}]+)?--[^;{}]*?:[^{};]*?(?:[;\n]|$)/gim,
      mixinProp: /(?:^[^;\-\s}]+)?--[^;{}]*?:[^{};]*?{[^}]*?}(?:[;\n]|$)?/gim,
      mixinApply: /@apply\s*\(?[^);]*\)?\s*(?:[;\n]|$)?/gim,
      varApply: /[^;:]*?:[^;]*?var\([^;]*\)(?:[;\n]|$)?/gim,
      keyframesRule: /^@[^\s]*keyframes/,
      multipleSpaces: /\s+/g
    };

    const VAR_START = '--';
    const MEDIA_START = '@media';
    const AT_START = '@';

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    const VAR_ASSIGN = /(?:^|[;\s{]\s*)(--[\w-]*?)\s*:\s*(?:((?:'(?:\\'|.)*?'|"(?:\\"|.)*?"|\([^)]*?\)|[^};{])+)|\{([^}]*)\}(?:(?=[;\s}])|$))/gi;
    const MIXIN_MATCH = /(?:^|\W+)@apply\s*\(?([^);\n]*)\)?/gi;
    const MEDIA_MATCH = /@media\s(.*)/;

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    /** @type {!Set<string>} */
    const styleTextSet = new Set();

    const scopingAttribute = 'shady-unscoped';

    /**
     * Add a specifically-marked style to the document directly, and only one copy of that style.
     *
     * @param {!HTMLStyleElement} style
     * @return {undefined}
     */
    function processUnscopedStyle(style) {
      const text = style.textContent;
      if (!styleTextSet.has(text)) {
        styleTextSet.add(text);
        const newStyle = document.createElement('style');
        newStyle.setAttribute('shady-unscoped', '');
        newStyle.textContent = text;
        document.head.appendChild(newStyle);
      }
    }

    /**
     * Check if a style is supposed to be unscoped
     * @param {!HTMLStyleElement} style
     * @return {boolean} true if the style has the unscoping attribute
     */
    function isUnscopedStyle(style) {
      return style.hasAttribute(scopingAttribute);
    }

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    /**
     * @param {string|StyleNode} rules
     * @param {function(StyleNode)=} callback
     * @return {string}
     */
    function toCssText (rules, callback) {
      if (!rules) {
        return '';
      }
      if (typeof rules === 'string') {
        rules = parse(rules);
      }
      if (callback) {
        forEachRule(rules, callback);
      }
      return stringify(rules, nativeCssVariables);
    }

    /**
     * @param {HTMLStyleElement} style
     * @return {StyleNode}
     */
    function rulesForStyle(style) {
      if (!style['__cssRules'] && style.textContent) {
        style['__cssRules'] = parse(style.textContent);
      }
      return style['__cssRules'] || null;
    }

    /**
     * @param {StyleNode} node
     * @param {Function=} styleRuleCallback
     * @param {Function=} keyframesRuleCallback
     * @param {boolean=} onlyActiveRules
     */
    function forEachRule(node, styleRuleCallback, keyframesRuleCallback, onlyActiveRules) {
      if (!node) {
        return;
      }
      let skipRules = false;
      let type = node['type'];
      if (onlyActiveRules) {
        if (type === types.MEDIA_RULE) {
          let matchMedia = node['selector'].match(MEDIA_MATCH);
          if (matchMedia) {
            // if rule is a non matching @media rule, skip subrules
            if (!window.matchMedia(matchMedia[1]).matches) {
              skipRules = true;
            }
          }
        }
      }
      if (type === types.STYLE_RULE) {
        styleRuleCallback(node);
      } else if (keyframesRuleCallback &&
        type === types.KEYFRAMES_RULE) {
        keyframesRuleCallback(node);
      } else if (type === types.MIXIN_RULE) {
        skipRules = true;
      }
      let r$ = node['rules'];
      if (r$ && !skipRules) {
        for (let i=0, l=r$.length, r; (i<l) && (r=r$[i]); i++) {
          forEachRule(r, styleRuleCallback, keyframesRuleCallback, onlyActiveRules);
        }
      }
    }

    /**
     * Walk from text[start] matching parens and
     * returns position of the outer end paren
     * @param {string} text
     * @param {number} start
     * @return {number}
     */
    function findMatchingParen(text, start) {
      let level = 0;
      for (let i=start, l=text.length; i < l; i++) {
        if (text[i] === '(') {
          level++;
        } else if (text[i] === ')') {
          if (--level === 0) {
            return i;
          }
        }
      }
      return -1;
    }

    /**
     * @param {string} str
     * @param {function(string, string, string, string)} callback
     */
    function processVariableAndFallback(str, callback) {
      // find 'var('
      let start = str.indexOf('var(');
      if (start === -1) {
        // no var?, everything is prefix
        return callback(str, '', '', '');
      }
      //${prefix}var(${inner})${suffix}
      let end = findMatchingParen(str, start + 3);
      let inner = str.substring(start + 4, end);
      let prefix = str.substring(0, start);
      // suffix may have other variables
      let suffix = processVariableAndFallback(str.substring(end + 1), callback);
      let comma = inner.indexOf(',');
      // value and fallback args should be trimmed to match in property lookup
      if (comma === -1) {
        // variable, no fallback
        return callback(prefix, inner.trim(), '', suffix);
      }
      // var(${value},${fallback})
      let value = inner.substring(0, comma).trim();
      let fallback = inner.substring(comma + 1).trim();
      return callback(prefix, value, fallback, suffix);
    }

    /**
     * @type {function(*):*}
     */
    const wrap$1 = window['ShadyDOM'] && window['ShadyDOM']['wrap'] || ((node) => node);

    /**
     * @param {Element | {is: string, extends: string}} element
     * @return {{is: string, typeExtension: string}}
     */
    function getIsExtends(element) {
      let localName = element['localName'];
      let is = '', typeExtension = '';
      /*
      NOTE: technically, this can be wrong for certain svg elements
      with `-` in the name like `<font-face>`
      */
      if (localName) {
        if (localName.indexOf('-') > -1) {
          is = localName;
        } else {
          typeExtension = localName;
          is = (element.getAttribute && element.getAttribute('is')) || '';
        }
      } else {
        is = /** @type {?} */(element).is;
        typeExtension = /** @type {?} */(element).extends;
      }
      return {is, typeExtension};
    }

    /**
     * @param {Element|DocumentFragment} element
     * @return {string}
     */
    function gatherStyleText(element) {
      /** @type {!Array<string>} */
      const styleTextParts = [];
      const styles = /** @type {!NodeList<!HTMLStyleElement>} */(element.querySelectorAll('style'));
      for (let i = 0; i < styles.length; i++) {
        const style = styles[i];
        if (isUnscopedStyle(style)) {
          if (!nativeShadow) {
            processUnscopedStyle(style);
            style.parentNode.removeChild(style);
          }
        } else {
          styleTextParts.push(style.textContent);
          style.parentNode.removeChild(style);
        }
      }
      return styleTextParts.join('').trim();
    }

    const CSS_BUILD_ATTR = 'css-build';

    /**
     * Return the polymer-css-build "build type" applied to this element
     *
     * @param {!HTMLElement} element
     * @return {string} Can be "", "shady", or "shadow"
     */
    function getCssBuild(element) {
      if (cssBuild !== undefined) {
        return /** @type {string} */(cssBuild);
      }
      if (element.__cssBuild === undefined) {
        // try attribute first, as it is the common case
        const attrValue = element.getAttribute(CSS_BUILD_ATTR);
        if (attrValue) {
          element.__cssBuild = attrValue;
        } else {
          const buildComment = getBuildComment(element);
          if (buildComment !== '') {
            // remove build comment so it is not needlessly copied into every element instance
            removeBuildComment(element);
          }
          element.__cssBuild = buildComment;
        }
      }
      return element.__cssBuild || '';
    }

    /**
     * Check if the given element, either a <template> or <style>, has been processed
     * by polymer-css-build.
     *
     * If so, then we can make a number of optimizations:
     * - polymer-css-build will decompose mixins into individual CSS Custom Properties,
     * so the ApplyShim can be skipped entirely.
     * - Under native ShadowDOM, the style text can just be copied into each instance
     * without modification
     * - If the build is "shady" and ShadyDOM is in use, the styling does not need
     * scoping beyond the shimming of CSS Custom Properties
     *
     * @param {!HTMLElement} element
     * @return {boolean}
     */
    function elementHasBuiltCss(element) {
      return getCssBuild(element) !== '';
    }

    /**
     * For templates made with tagged template literals, polymer-css-build will
     * insert a comment of the form `<!--css-build:shadow-->`
     *
     * @param {!HTMLElement} element
     * @return {string}
     */
    function getBuildComment(element) {
      const buildComment = element.localName === 'template' ?
          /** @type {!HTMLTemplateElement} */ (element).content.firstChild :
          element.firstChild;
      if (buildComment instanceof Comment) {
        const commentParts = buildComment.textContent.trim().split(':');
        if (commentParts[0] === CSS_BUILD_ATTR) {
          return commentParts[1];
        }
      }
      return '';
    }

    /**
     * @param {!HTMLElement} element
     */
    function removeBuildComment(element) {
      const buildComment = element.localName === 'template' ?
          /** @type {!HTMLTemplateElement} */ (element).content.firstChild :
          element.firstChild;
      buildComment.parentNode.removeChild(buildComment);
    }

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    /**
     * @param {Element} element
     * @param {Object=} properties
     */
    function updateNativeProperties(element, properties) {
      // remove previous properties
      for (let p in properties) {
        // NOTE: for bc with shim, don't apply null values.
        if (p === null) {
          element.style.removeProperty(p);
        } else {
          element.style.setProperty(p, properties[p]);
        }
      }
    }

    /**
     * @param {Element} element
     * @param {string} property
     * @return {string}
     */
    function getComputedStyleValue(element, property) {
      /**
       * @const {string}
       */
      const value = window.getComputedStyle(element).getPropertyValue(property);
      if (!value) {
        return '';
      } else {
        return value.trim();
      }
    }

    /**
     * return true if `cssText` contains a mixin definition or consumption
     * @param {string} cssText
     * @return {boolean}
     */
    function detectMixin(cssText) {
      const has = MIXIN_MATCH.test(cssText) || VAR_ASSIGN.test(cssText);
      // reset state of the regexes
      MIXIN_MATCH.lastIndex = 0;
      VAR_ASSIGN.lastIndex = 0;
      return has;
    }

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    const APPLY_NAME_CLEAN = /;\s*/m;
    const INITIAL_INHERIT = /^\s*(initial)|(inherit)\s*$/;
    const IMPORTANT = /\s*!important/;

    // separator used between mixin-name and mixin-property-name when producing properties
    // NOTE: plain '-' may cause collisions in user styles
    const MIXIN_VAR_SEP = '_-_';

    // map of mixin to property names
    // --foo: {border: 2px} -> {properties: {(--foo, ['border'])}, dependants: {'element-name': proto}}
    class MixinMap {
      constructor() {
        /** @type {!Object<string, !MixinMapEntry>} */
        this._map = {};
      }
      /**
       * @param {string} name
       * @param {!PropertyEntry} props
       */
      set(name, props) {
        name = name.trim();
        this._map[name] = {
          properties: props,
          dependants: {}
        };
      }
      /**
       * @param {string} name
       * @return {MixinMapEntry}
       */
      get(name) {
        name = name.trim();
        return this._map[name] || null;
      }
    }

    /**
     * Callback for when an element is marked invalid
     * @type {?function(string)}
     */
    let invalidCallback = null;

    /** @unrestricted */
    class ApplyShim {
      constructor() {
        /** @type {?string} */
        this._currentElement = null;
        /** @type {HTMLMetaElement} */
        this._measureElement = null;
        this._map = new MixinMap();
      }
      /**
       * return true if `cssText` contains a mixin definition or consumption
       * @param {string} cssText
       * @return {boolean}
       */
      detectMixin(cssText) {
        return detectMixin(cssText);
      }

      /**
       * Gather styles into one style for easier processing
       * @param {!HTMLTemplateElement} template
       * @return {HTMLStyleElement}
       */
      gatherStyles(template) {
        const styleText = gatherStyleText(template.content);
        if (styleText) {
          const style = /** @type {!HTMLStyleElement} */(document.createElement('style'));
          style.textContent = styleText;
          template.content.insertBefore(style, template.content.firstChild);
          return style;
        }
        return null;
      }
      /**
       * @param {!HTMLTemplateElement} template
       * @param {string} elementName
       * @return {StyleNode}
       */
      transformTemplate(template, elementName) {
        if (template._gatheredStyle === undefined) {
          template._gatheredStyle = this.gatherStyles(template);
        }
        /** @type {HTMLStyleElement} */
        const style = template._gatheredStyle;
        return style ? this.transformStyle(style, elementName) : null;
      }
      /**
       * @param {!HTMLStyleElement} style
       * @param {string} elementName
       * @return {StyleNode}
       */
      transformStyle(style, elementName = '') {
        let ast = rulesForStyle(style);
        this.transformRules(ast, elementName);
        style.textContent = toCssText(ast);
        return ast;
      }
      /**
       * @param {!HTMLStyleElement} style
       * @return {StyleNode}
       */
      transformCustomStyle(style) {
        let ast = rulesForStyle(style);
        forEachRule(ast, (rule) => {
          if (rule['selector'] === ':root') {
            rule['selector'] = 'html';
          }
          this.transformRule(rule);
        });
        style.textContent = toCssText(ast);
        return ast;
      }
      /**
       * @param {StyleNode} rules
       * @param {string} elementName
       */
      transformRules(rules, elementName) {
        this._currentElement = elementName;
        forEachRule(rules, (r) => {
          this.transformRule(r);
        });
        this._currentElement = null;
      }
      /**
       * @param {!StyleNode} rule
       */
      transformRule(rule) {
        rule['cssText'] = this.transformCssText(rule['parsedCssText'], rule);
        // :root was only used for variable assignment in property shim,
        // but generates invalid selectors with real properties.
        // replace with `:host > *`, which serves the same effect
        if (rule['selector'] === ':root') {
          rule['selector'] = ':host > *';
        }
      }
      /**
       * @param {string} cssText
       * @param {!StyleNode} rule
       * @return {string}
       */
      transformCssText(cssText, rule) {
        // produce variables
        cssText = cssText.replace(VAR_ASSIGN, (matchText, propertyName, valueProperty, valueMixin) =>
          this._produceCssProperties(matchText, propertyName, valueProperty, valueMixin, rule));
        // consume mixins
        return this._consumeCssProperties(cssText, rule);
      }
      /**
       * @param {string} property
       * @return {string}
       */
      _getInitialValueForProperty(property) {
        if (!this._measureElement) {
          this._measureElement = /** @type {HTMLMetaElement} */(document.createElement('meta'));
          this._measureElement.setAttribute('apply-shim-measure', '');
          this._measureElement.style.all = 'initial';
          document.head.appendChild(this._measureElement);
        }
        return window.getComputedStyle(this._measureElement).getPropertyValue(property);
      }
      /**
       * Walk over all rules before this rule to find fallbacks for mixins
       *
       * @param {!StyleNode} startRule
       * @return {!Object}
       */
      _fallbacksFromPreviousRules(startRule) {
        // find the "top" rule
        let topRule = startRule;
        while (topRule['parent']) {
          topRule = topRule['parent'];
        }
        const fallbacks = {};
        let seenStartRule = false;
        forEachRule(topRule, (r) => {
          // stop when we hit the input rule
          seenStartRule = seenStartRule || r === startRule;
          if (seenStartRule) {
            return;
          }
          // NOTE: Only matching selectors are "safe" for this fallback processing
          // It would be prohibitive to run `matchesSelector()` on each selector,
          // so we cheat and only check if the same selector string is used, which
          // guarantees things like specificity matching
          if (r['selector'] === startRule['selector']) {
            Object.assign(fallbacks, this._cssTextToMap(r['parsedCssText']));
          }
        });
        return fallbacks;
      }
      /**
       * replace mixin consumption with variable consumption
       * @param {string} text
       * @param {!StyleNode=} rule
       * @return {string}
       */
      _consumeCssProperties(text, rule) {
        /** @type {Array} */
        let m = null;
        // loop over text until all mixins with defintions have been applied
        while((m = MIXIN_MATCH.exec(text))) {
          let matchText = m[0];
          let mixinName = m[1];
          let idx = m.index;
          // collect properties before apply to be "defaults" if mixin might override them
          // match includes a "prefix", so find the start and end positions of @apply
          let applyPos = idx + matchText.indexOf('@apply');
          let afterApplyPos = idx + matchText.length;
          // find props defined before this @apply
          let textBeforeApply = text.slice(0, applyPos);
          let textAfterApply = text.slice(afterApplyPos);
          let defaults = rule ? this._fallbacksFromPreviousRules(rule) : {};
          Object.assign(defaults, this._cssTextToMap(textBeforeApply));
          let replacement = this._atApplyToCssProperties(mixinName, defaults);
          // use regex match position to replace mixin, keep linear processing time
          text = `${textBeforeApply}${replacement}${textAfterApply}`;
          // move regex search to _after_ replacement
          MIXIN_MATCH.lastIndex = idx + replacement.length;
        }
        return text;
      }
      /**
       * produce variable consumption at the site of mixin consumption
       * `@apply` --foo; -> for all props (${propname}: var(--foo_-_${propname}, ${fallback[propname]}}))
       * Example:
       *  border: var(--foo_-_border); padding: var(--foo_-_padding, 2px)
       *
       * @param {string} mixinName
       * @param {Object} fallbacks
       * @return {string}
       */
      _atApplyToCssProperties(mixinName, fallbacks) {
        mixinName = mixinName.replace(APPLY_NAME_CLEAN, '');
        let vars = [];
        let mixinEntry = this._map.get(mixinName);
        // if we depend on a mixin before it is created
        // make a sentinel entry in the map to add this element as a dependency for when it is defined.
        if (!mixinEntry) {
          this._map.set(mixinName, {});
          mixinEntry = this._map.get(mixinName);
        }
        if (mixinEntry) {
          if (this._currentElement) {
            mixinEntry.dependants[this._currentElement] = true;
          }
          let p, parts, f;
          const properties = mixinEntry.properties;
          for (p in properties) {
            f = fallbacks && fallbacks[p];
            parts = [p, ': var(', mixinName, MIXIN_VAR_SEP, p];
            if (f) {
              parts.push(',', f.replace(IMPORTANT, ''));
            }
            parts.push(')');
            if (IMPORTANT.test(properties[p])) {
              parts.push(' !important');
            }
            vars.push(parts.join(''));
          }
        }
        return vars.join('; ');
      }

      /**
       * @param {string} property
       * @param {string} value
       * @return {string}
       */
      _replaceInitialOrInherit(property, value) {
        let match = INITIAL_INHERIT.exec(value);
        if (match) {
          if (match[1]) {
            // initial
            // replace `initial` with the concrete initial value for this property
            value = this._getInitialValueForProperty(property);
          } else {
            // inherit
            // with this purposfully illegal value, the variable will be invalid at
            // compute time (https://www.w3.org/TR/css-variables/#invalid-at-computed-value-time)
            // and for inheriting values, will behave similarly
            // we cannot support the same behavior for non inheriting values like 'border'
            value = 'apply-shim-inherit';
          }
        }
        return value;
      }

      /**
       * "parse" a mixin definition into a map of properties and values
       * cssTextToMap('border: 2px solid black') -> ('border', '2px solid black')
       * @param {string} text
       * @param {boolean=} replaceInitialOrInherit
       * @return {!Object<string, string>}
       */
      _cssTextToMap(text, replaceInitialOrInherit = false) {
        let props = text.split(';');
        let property, value;
        let out = {};
        for (let i = 0, p, sp; i < props.length; i++) {
          p = props[i];
          if (p) {
            sp = p.split(':');
            // ignore lines that aren't definitions like @media
            if (sp.length > 1) {
              property = sp[0].trim();
              // some properties may have ':' in the value, like data urls
              value = sp.slice(1).join(':');
              if (replaceInitialOrInherit) {
                value = this._replaceInitialOrInherit(property, value);
              }
              out[property] = value;
            }
          }
        }
        return out;
      }

      /**
       * @param {MixinMapEntry} mixinEntry
       */
      _invalidateMixinEntry(mixinEntry) {
        if (!invalidCallback) {
          return;
        }
        for (let elementName in mixinEntry.dependants) {
          if (elementName !== this._currentElement) {
            invalidCallback(elementName);
          }
        }
      }

      /**
       * @param {string} matchText
       * @param {string} propertyName
       * @param {?string} valueProperty
       * @param {?string} valueMixin
       * @param {!StyleNode} rule
       * @return {string}
       */
      _produceCssProperties(matchText, propertyName, valueProperty, valueMixin, rule) {
        // handle case where property value is a mixin
        if (valueProperty) {
          // form: --mixin2: var(--mixin1), where --mixin1 is in the map
          processVariableAndFallback(valueProperty, (prefix, value) => {
            if (value && this._map.get(value)) {
              valueMixin = `@apply ${value};`;
            }
          });
        }
        if (!valueMixin) {
          return matchText;
        }
        let mixinAsProperties = this._consumeCssProperties('' + valueMixin, rule);
        let prefix = matchText.slice(0, matchText.indexOf('--'));
        // `initial` and `inherit` as properties in a map should be replaced because
        // these keywords are eagerly evaluated when the mixin becomes CSS Custom Properties,
        // and would set the variable value, rather than carry the keyword to the `var()` usage.
        let mixinValues = this._cssTextToMap(mixinAsProperties, true);
        let combinedProps = mixinValues;
        let mixinEntry = this._map.get(propertyName);
        let oldProps = mixinEntry && mixinEntry.properties;
        if (oldProps) {
          // NOTE: since we use mixin, the map of properties is updated here
          // and this is what we want.
          combinedProps = Object.assign(Object.create(oldProps), mixinValues);
        } else {
          this._map.set(propertyName, combinedProps);
        }
        let out = [];
        let p, v;
        // set variables defined by current mixin
        let needToInvalidate = false;
        for (p in combinedProps) {
          v = mixinValues[p];
          // if property not defined by current mixin, set initial
          if (v === undefined) {
            v = 'initial';
          }
          if (oldProps && !(p in oldProps)) {
            needToInvalidate = true;
          }
          out.push(`${propertyName}${MIXIN_VAR_SEP}${p}: ${v}`);
        }
        if (needToInvalidate) {
          this._invalidateMixinEntry(mixinEntry);
        }
        if (mixinEntry) {
          mixinEntry.properties = combinedProps;
        }
        // because the mixinMap is global, the mixin might conflict with
        // a different scope's simple variable definition:
        // Example:
        // some style somewhere:
        // --mixin1:{ ... }
        // --mixin2: var(--mixin1);
        // some other element:
        // --mixin1: 10px solid red;
        // --foo: var(--mixin1);
        // In this case, we leave the original variable definition in place.
        if (valueProperty) {
          prefix = `${matchText};${prefix}`;
        }
        return `${prefix}${out.join('; ')};`;
      }
    }

    /* exports */
    /* eslint-disable no-self-assign */
    ApplyShim.prototype['detectMixin'] = ApplyShim.prototype.detectMixin;
    ApplyShim.prototype['transformStyle'] = ApplyShim.prototype.transformStyle;
    ApplyShim.prototype['transformCustomStyle'] = ApplyShim.prototype.transformCustomStyle;
    ApplyShim.prototype['transformRules'] = ApplyShim.prototype.transformRules;
    ApplyShim.prototype['transformRule'] = ApplyShim.prototype.transformRule;
    ApplyShim.prototype['transformTemplate'] = ApplyShim.prototype.transformTemplate;
    ApplyShim.prototype['_separator'] = MIXIN_VAR_SEP;
    /* eslint-enable no-self-assign */
    Object.defineProperty(ApplyShim.prototype, 'invalidCallback', {
      /** @return {?function(string)} */
      get() {
        return invalidCallback;
      },
      /** @param {?function(string)} cb */
      set(cb) {
        invalidCallback = cb;
      }
    });

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    /**
     * @const {!Object<string, !HTMLTemplateElement>}
     */
    const templateMap = {};

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    /*
     * Utilities for handling invalidating apply-shim mixins for a given template.
     *
     * The invalidation strategy involves keeping track of the "current" version of a template's mixins, and updating that count when a mixin is invalidated.
     * The template
     */

    /** @const {string} */
    const CURRENT_VERSION = '_applyShimCurrentVersion';

    /** @const {string} */
    const NEXT_VERSION = '_applyShimNextVersion';

    /** @const {string} */
    const VALIDATING_VERSION = '_applyShimValidatingVersion';

    /**
     * @const {Promise<void>}
     */
    const promise = Promise.resolve();

    /**
     * @param {string} elementName
     */
    function invalidate(elementName){
      let template = templateMap[elementName];
      if (template) {
        invalidateTemplate(template);
      }
    }

    /**
     * This function can be called multiple times to mark a template invalid
     * and signal that the style inside must be regenerated.
     *
     * Use `startValidatingTemplate` to begin an asynchronous validation cycle.
     * During that cycle, call `templateIsValidating` to see if the template must
     * be revalidated
     * @param {HTMLTemplateElement} template
     */
    function invalidateTemplate(template) {
      // default the current version to 0
      template[CURRENT_VERSION] = template[CURRENT_VERSION] || 0;
      // ensure the "validating for" flag exists
      template[VALIDATING_VERSION] = template[VALIDATING_VERSION] || 0;
      // increment the next version
      template[NEXT_VERSION] = (template[NEXT_VERSION] || 0) + 1;
    }

    /**
     * @param {HTMLTemplateElement} template
     * @return {boolean}
     */
    function templateIsValid(template) {
      return template[CURRENT_VERSION] === template[NEXT_VERSION];
    }

    /**
     * Returns true if the template is currently invalid and `startValidating` has been called since the last invalidation.
     * If false, the template must be validated.
     * @param {HTMLTemplateElement} template
     * @return {boolean}
     */
    function templateIsValidating(template) {
      return !templateIsValid(template) && template[VALIDATING_VERSION] === template[NEXT_VERSION];
    }

    /**
     * Begin an asynchronous invalidation cycle.
     * This should be called after every validation of a template
     *
     * After one microtask, the template will be marked as valid until the next call to `invalidateTemplate`
     * @param {HTMLTemplateElement} template
     */
    function startValidatingTemplate(template) {
      // remember that the current "next version" is the reason for this validation cycle
      template[VALIDATING_VERSION] = template[NEXT_VERSION];
      // however, there only needs to be one async task to clear the counters
      if (!template._validating) {
        template._validating = true;
        promise.then(function() {
          // sync the current version to let future invalidations cause a refresh cycle
          template[CURRENT_VERSION] = template[NEXT_VERSION];
          template._validating = false;
        });
      }
    }

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    /** @type {Promise<void>} */
    let readyPromise = null;

    /** @type {?function(?function())} */
    let whenReady = window['HTMLImports'] && window['HTMLImports']['whenReady'] || null;

    /** @type {function()} */
    let resolveFn;

    /**
     * @param {?function()} callback
     */
    function documentWait(callback) {
      requestAnimationFrame(function() {
        if (whenReady) {
          whenReady(callback);
        } else {
          if (!readyPromise) {
            readyPromise = new Promise((resolve) => {resolveFn = resolve;});
            if (document.readyState === 'complete') {
              resolveFn();
            } else {
              document.addEventListener('readystatechange', () => {
                if (document.readyState === 'complete') {
                  resolveFn();
                }
              });
            }
          }
          readyPromise.then(function(){ callback && callback(); });
        }
      });
    }

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    const SEEN_MARKER = '__seenByShadyCSS';
    const CACHED_STYLE = '__shadyCSSCachedStyle';

    /** @type {?function(!HTMLStyleElement)} */
    let transformFn = null;

    /** @type {?function()} */
    let validateFn = null;

    /**
    This interface is provided to add document-level <style> elements to ShadyCSS for processing.
    These styles must be processed by ShadyCSS to simulate ShadowRoot upper-bound encapsulation from outside styles
    In addition, these styles may also need to be processed for @apply rules and CSS Custom Properties

    To add document-level styles to ShadyCSS, one can call `ShadyCSS.addDocumentStyle(styleElement)` or `ShadyCSS.addDocumentStyle({getStyle: () => styleElement})`

    In addition, if the process used to discover document-level styles can be synchronously flushed, one should set `ShadyCSS.documentStyleFlush`.
    This function will be called when calculating styles.

    An example usage of the document-level styling api can be found in `examples/document-style-lib.js`

    @unrestricted
    */
    class CustomStyleInterface {
      constructor() {
        /** @type {!Array<!CustomStyleProvider>} */
        this['customStyles'] = [];
        this['enqueued'] = false;
        // NOTE(dfreedm): use quotes here to prevent closure inlining to `function(){}`;
        documentWait(() => {
          if (window['ShadyCSS']['flushCustomStyles']) {
            window['ShadyCSS']['flushCustomStyles']();
          }
        });
      }
      /**
       * Queue a validation for new custom styles to batch style recalculations
       */
      enqueueDocumentValidation() {
        if (this['enqueued'] || !validateFn) {
          return;
        }
        this['enqueued'] = true;
        documentWait(validateFn);
      }
      /**
       * @param {!HTMLStyleElement} style
       */
      addCustomStyle(style) {
        if (!style[SEEN_MARKER]) {
          style[SEEN_MARKER] = true;
          this['customStyles'].push(style);
          this.enqueueDocumentValidation();
        }
      }
      /**
       * @param {!CustomStyleProvider} customStyle
       * @return {HTMLStyleElement}
       */
      getStyleForCustomStyle(customStyle) {
        if (customStyle[CACHED_STYLE]) {
          return customStyle[CACHED_STYLE];
        }
        let style;
        if (customStyle['getStyle']) {
          style = customStyle['getStyle']();
        } else {
          style = customStyle;
        }
        return style;
      }
      /**
       * @return {!Array<!CustomStyleProvider>}
       */
      processStyles() {
        const cs = this['customStyles'];
        for (let i = 0; i < cs.length; i++) {
          const customStyle = cs[i];
          if (customStyle[CACHED_STYLE]) {
            continue;
          }
          const style = this.getStyleForCustomStyle(customStyle);
          if (style) {
            // HTMLImports polyfill may have cloned the style into the main document,
            // which is referenced with __appliedElement.
            const styleToTransform = /** @type {!HTMLStyleElement} */(style['__appliedElement'] || style);
            if (transformFn) {
              transformFn(styleToTransform);
            }
            customStyle[CACHED_STYLE] = styleToTransform;
          }
        }
        return cs;
      }
    }

    /* eslint-disable no-self-assign */
    CustomStyleInterface.prototype['addCustomStyle'] = CustomStyleInterface.prototype.addCustomStyle;
    CustomStyleInterface.prototype['getStyleForCustomStyle'] = CustomStyleInterface.prototype.getStyleForCustomStyle;
    CustomStyleInterface.prototype['processStyles'] = CustomStyleInterface.prototype.processStyles;
    /* eslint-enable no-self-assign */

    Object.defineProperties(CustomStyleInterface.prototype, {
      'transformCallback': {
        /** @return {?function(!HTMLStyleElement)} */
        get() {
          return transformFn;
        },
        /** @param {?function(!HTMLStyleElement)} fn */
        set(fn) {
          transformFn = fn;
        }
      },
      'validateCallback': {
        /** @return {?function()} */
        get() {
          return validateFn;
        },
        /**
         * @param {?function()} fn
         * @this {CustomStyleInterface}
         */
        set(fn) {
          let needsEnqueue = false;
          if (!validateFn) {
            needsEnqueue = true;
          }
          validateFn = fn;
          if (needsEnqueue) {
            this.enqueueDocumentValidation();
          }
        },
      }
    });

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    /** @const {ApplyShim} */
    const applyShim = new ApplyShim();

    class ApplyShimInterface {
      constructor() {
        /** @type {?CustomStyleInterfaceInterface} */
        this.customStyleInterface = null;
        applyShim['invalidCallback'] = invalidate;
      }
      ensure() {
        if (this.customStyleInterface) {
          return;
        }
        if (window.ShadyCSS.CustomStyleInterface) {
          this.customStyleInterface =
              /** @type {!CustomStyleInterfaceInterface} */ (
                  window.ShadyCSS.CustomStyleInterface);
          this.customStyleInterface['transformCallback'] = (style) => {
            applyShim.transformCustomStyle(style);
          };
          this.customStyleInterface['validateCallback'] = () => {
            requestAnimationFrame(() => {
              if (this.customStyleInterface['enqueued']) {
                this.flushCustomStyles();
              }
            });
          };
        }
      }
      /**
       * @param {!HTMLTemplateElement} template
       * @param {string} elementName
       */
      prepareTemplate(template, elementName) {
        this.ensure();
        if (elementHasBuiltCss(template)) {
          return;
        }
        templateMap[elementName] = template;
        let ast = applyShim.transformTemplate(template, elementName);
        // save original style ast to use for revalidating instances
        template['_styleAst'] = ast;
      }
      flushCustomStyles() {
        this.ensure();
        if (!this.customStyleInterface) {
          return;
        }
        let styles = this.customStyleInterface['processStyles']();
        if (!this.customStyleInterface['enqueued']) {
          return;
        }
        for (let i = 0; i < styles.length; i++ ) {
          let cs = styles[i];
          let style = this.customStyleInterface['getStyleForCustomStyle'](cs);
          if (style) {
            applyShim.transformCustomStyle(style);
          }
        }
        this.customStyleInterface['enqueued'] = false;
      }
      /**
       * @param {HTMLElement} element
       * @param {Object=} properties
       */
      styleSubtree(element, properties) {
        this.ensure();
        if (properties) {
          updateNativeProperties(element, properties);
        }
        if (element.shadowRoot) {
          this.styleElement(element);
          let shadowChildren =
              /** @type {!ParentNode} */ (element.shadowRoot).children ||
              element.shadowRoot.childNodes;
          for (let i = 0; i < shadowChildren.length; i++) {
            this.styleSubtree(/** @type {HTMLElement} */(shadowChildren[i]));
          }
        } else {
          let children = element.children || element.childNodes;
          for (let i = 0; i < children.length; i++) {
            this.styleSubtree(/** @type {HTMLElement} */(children[i]));
          }
        }
      }
      /**
       * @param {HTMLElement} element
       */
      styleElement(element) {
        this.ensure();
        let {is} = getIsExtends(element);
        let template = templateMap[is];
        if (template && elementHasBuiltCss(template)) {
          return;
        }
        if (template && !templateIsValid(template)) {
          // only revalidate template once
          if (!templateIsValidating(template)) {
            this.prepareTemplate(template, is);
            startValidatingTemplate(template);
          }
          // update this element instance
          let root = element.shadowRoot;
          if (root) {
            let style = /** @type {HTMLStyleElement} */(root.querySelector('style'));
            if (style) {
              // reuse the template's style ast, it has all the original css text
              style['__cssRules'] = template['_styleAst'];
              style.textContent = toCssText(template['_styleAst']);
            }
          }
        }
      }
      /**
       * @param {Object=} properties
       */
      styleDocument(properties) {
        this.ensure();
        this.styleSubtree(document.body, properties);
      }
    }

    if (!window.ShadyCSS || !window.ShadyCSS.ScopingShim) {
      const applyShimInterface = new ApplyShimInterface();
      let CustomStyleInterface = window.ShadyCSS && window.ShadyCSS.CustomStyleInterface;

      /** @suppress {duplicate} */
      window.ShadyCSS = {
        /**
         * @param {!HTMLTemplateElement} template
         * @param {string} elementName
         * @param {string=} elementExtends
         */
        prepareTemplate(template, elementName, elementExtends) { // eslint-disable-line no-unused-vars
          applyShimInterface.flushCustomStyles();
          applyShimInterface.prepareTemplate(template, elementName);
        },

        /**
         * @param {!HTMLTemplateElement} template
         * @param {string} elementName
         * @param {string=} elementExtends
         */
        prepareTemplateStyles(template, elementName, elementExtends) {
          window.ShadyCSS.prepareTemplate(template, elementName, elementExtends);
        },

        /**
         * @param {!HTMLTemplateElement} template
         * @param {string} elementName
         */
        prepareTemplateDom(template, elementName) {}, // eslint-disable-line no-unused-vars

        /**
         * @param {!HTMLElement} element
         * @param {Object=} properties
         */
        styleSubtree(element, properties) {
          applyShimInterface.flushCustomStyles();
          applyShimInterface.styleSubtree(element, properties);
        },

        /**
         * @param {!HTMLElement} element
         */
        styleElement(element) {
          applyShimInterface.flushCustomStyles();
          applyShimInterface.styleElement(element);
        },

        /**
         * @param {Object=} properties
         */
        styleDocument(properties) {
          applyShimInterface.flushCustomStyles();
          applyShimInterface.styleDocument(properties);
        },

        /**
         * @param {Element} element
         * @param {string} property
         * @return {string}
         */
        getComputedStyleValue(element, property) {
          return getComputedStyleValue(element, property);
        },

        flushCustomStyles() {
          applyShimInterface.flushCustomStyles();
        },

        nativeCss: nativeCssVariables,
        nativeShadow: nativeShadow,
        cssBuild: cssBuild,
        disableRuntime: disableRuntime,
      };

      if (CustomStyleInterface) {
        window.ShadyCSS.CustomStyleInterface = CustomStyleInterface;
      }
    }

    window.ShadyCSS.ApplyShim = applyShim;

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    /**
     * @summary Collapse multiple callbacks into one invocation after a timer.
     */
    class Debouncer {
      constructor() {
        this._asyncModule = null;
        this._callback = null;
        this._timer = null;
      }
      /**
       * Sets the scheduler; that is, a module with the Async interface,
       * a callback and optional arguments to be passed to the run function
       * from the async module.
       *
       * @param {!AsyncInterface} asyncModule Object with Async interface.
       * @param {function()} callback Callback to run.
       * @return {void}
       */
      setConfig(asyncModule, callback) {
        this._asyncModule = asyncModule;
        this._callback = callback;
        this._timer = this._asyncModule.run(() => {
          this._timer = null;
          debouncerQueue.delete(this);
          this._callback();
        });
      }
      /**
       * Cancels an active debouncer and returns a reference to itself.
       *
       * @return {void}
       */
      cancel() {
        if (this.isActive()) {
          this._cancelAsync();
          // Canceling a debouncer removes its spot from the flush queue,
          // so if a debouncer is manually canceled and re-debounced, it
          // will reset its flush order (this is a very minor difference from 1.x)
          // Re-debouncing via the `debounce` API retains the 1.x FIFO flush order
          debouncerQueue.delete(this);
        }
      }
      /**
       * Cancels a debouncer's async callback.
       *
       * @return {void}
       */
      _cancelAsync() {
        if (this.isActive()) {
          this._asyncModule.cancel(/** @type {number} */(this._timer));
          this._timer = null;
        }
      }
      /**
       * Flushes an active debouncer and returns a reference to itself.
       *
       * @return {void}
       */
      flush() {
        if (this.isActive()) {
          this.cancel();
          this._callback();
        }
      }
      /**
       * Returns true if the debouncer is active.
       *
       * @return {boolean} True if active.
       */
      isActive() {
        return this._timer != null;
      }
      /**
       * Creates a debouncer if no debouncer is passed as a parameter
       * or it cancels an active debouncer otherwise. The following
       * example shows how a debouncer can be called multiple times within a
       * microtask and "debounced" such that the provided callback function is
       * called once. Add this method to a custom element:
       *
       * ```js
       * import {microTask} from '@polymer/polymer/lib/utils/async.js';
       * import {Debouncer} from '@polymer/polymer/lib/utils/debounce.js';
       * // ...
       *
       * _debounceWork() {
       *   this._debounceJob = Debouncer.debounce(this._debounceJob,
       *       microTask, () => this._doWork());
       * }
       * ```
       *
       * If the `_debounceWork` method is called multiple times within the same
       * microtask, the `_doWork` function will be called only once at the next
       * microtask checkpoint.
       *
       * Note: In testing it is often convenient to avoid asynchrony. To accomplish
       * this with a debouncer, you can use `enqueueDebouncer` and
       * `flush`. For example, extend the above example by adding
       * `enqueueDebouncer(this._debounceJob)` at the end of the
       * `_debounceWork` method. Then in a test, call `flush` to ensure
       * the debouncer has completed.
       *
       * @param {Debouncer?} debouncer Debouncer object.
       * @param {!AsyncInterface} asyncModule Object with Async interface
       * @param {function()} callback Callback to run.
       * @return {!Debouncer} Returns a debouncer object.
       */
      static debounce(debouncer, asyncModule, callback) {
        if (debouncer instanceof Debouncer) {
          // Cancel the async callback, but leave in debouncerQueue if it was
          // enqueued, to maintain 1.x flush order
          debouncer._cancelAsync();
        } else {
          debouncer = new Debouncer();
        }
        debouncer.setConfig(asyncModule, callback);
        return debouncer;
      }
    }

    let debouncerQueue = new Set();

    /**
     * Adds a `Debouncer` to a list of globally flushable tasks.
     *
     * @param {!Debouncer} debouncer Debouncer to enqueue
     * @return {void}
     */
    const enqueueDebouncer = function(debouncer) {
      debouncerQueue.add(debouncer);
    };

    /**
     * Flushes any enqueued debouncers
     *
     * @return {boolean} Returns whether any debouncers were flushed
     */
    const flushDebouncers = function() {
      const didFlush = Boolean(debouncerQueue.size);
      // If new debouncers are added while flushing, Set.forEach will ensure
      // newly added ones are also flushed
      debouncerQueue.forEach(debouncer => {
        try {
          debouncer.flush();
        } catch(e) {
          setTimeout(() => {
            throw e;
          });
        }
      });
      return didFlush;
    };

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    // detect native touch action support
    let HAS_NATIVE_TA = typeof document.head.style.touchAction === 'string';
    let GESTURE_KEY = '__polymerGestures';
    let HANDLED_OBJ = '__polymerGesturesHandled';
    let TOUCH_ACTION = '__polymerGesturesTouchAction';
    // radius for tap and track
    let TAP_DISTANCE = 25;
    let TRACK_DISTANCE = 5;
    // number of last N track positions to keep
    let TRACK_LENGTH = 2;

    // Disabling "mouse" handlers for 2500ms is enough
    let MOUSE_TIMEOUT = 2500;
    let MOUSE_EVENTS = ['mousedown', 'mousemove', 'mouseup', 'click'];
    // an array of bitmask values for mapping MouseEvent.which to MouseEvent.buttons
    let MOUSE_WHICH_TO_BUTTONS = [0, 1, 4, 2];
    let MOUSE_HAS_BUTTONS = (function() {
      try {
        return new MouseEvent('test', {buttons: 1}).buttons === 1;
      } catch (e) {
        return false;
      }
    })();

    /**
     * @param {string} name Possible mouse event name
     * @return {boolean} true if mouse event, false if not
     */
    function isMouseEvent(name) {
      return MOUSE_EVENTS.indexOf(name) > -1;
    }

    /* eslint no-empty: ["error", { "allowEmptyCatch": true }] */
    // check for passive event listeners
    let supportsPassive = false;
    (function() {
      try {
        let opts = Object.defineProperty({}, 'passive', {get() {supportsPassive = true;}});
        window.addEventListener('test', null, opts);
        window.removeEventListener('test', null, opts);
      } catch(e) {}
    })();

    /**
     * Generate settings for event listeners, dependant on `passiveTouchGestures`
     *
     * @param {string} eventName Event name to determine if `{passive}` option is
     *   needed
     * @return {{passive: boolean} | undefined} Options to use for addEventListener
     *   and removeEventListener
     */
    function PASSIVE_TOUCH(eventName) {
      if (isMouseEvent(eventName) || eventName === 'touchend') {
        return;
      }
      if (HAS_NATIVE_TA && supportsPassive && passiveTouchGestures) {
        return {passive: true};
      } else {
        return;
      }
    }

    // Check for touch-only devices
    let IS_TOUCH_ONLY = navigator.userAgent.match(/iP(?:[oa]d|hone)|Android/);

    // keep track of any labels hit by the mouseCanceller
    /** @type {!Array<!HTMLLabelElement>} */
    const clickedLabels = [];

    /** @type {!Object<boolean>} */
    const labellable = {
      'button': true,
      'input': true,
      'keygen': true,
      'meter': true,
      'output': true,
      'textarea': true,
      'progress': true,
      'select': true
    };

    // Defined at https://html.spec.whatwg.org/multipage/form-control-infrastructure.html#enabling-and-disabling-form-controls:-the-disabled-attribute
    /** @type {!Object<boolean>} */
    const canBeDisabled = {
      'button': true,
      'command': true,
      'fieldset': true,
      'input': true,
      'keygen': true,
      'optgroup': true,
      'option': true,
      'select': true,
      'textarea': true
    };

    /**
     * @param {HTMLElement} el Element to check labelling status
     * @return {boolean} element can have labels
     */
    function canBeLabelled(el) {
      return labellable[el.localName] || false;
    }

    /**
     * @param {HTMLElement} el Element that may be labelled.
     * @return {!Array<!HTMLLabelElement>} Relevant label for `el`
     */
    function matchingLabels(el) {
      let labels = Array.prototype.slice.call(/** @type {HTMLInputElement} */(el).labels || []);
      // IE doesn't have `labels` and Safari doesn't populate `labels`
      // if element is in a shadowroot.
      // In this instance, finding the non-ancestor labels is enough,
      // as the mouseCancellor code will handle ancstor labels
      if (!labels.length) {
        labels = [];
        let root = el.getRootNode();
        // if there is an id on `el`, check for all labels with a matching `for` attribute
        if (el.id) {
          let matching = root.querySelectorAll(`label[for = ${el.id}]`);
          for (let i = 0; i < matching.length; i++) {
            labels.push(/** @type {!HTMLLabelElement} */(matching[i]));
          }
        }
      }
      return labels;
    }

    // touch will make synthetic mouse events
    // `preventDefault` on touchend will cancel them,
    // but this breaks `<input>` focus and link clicks
    // disable mouse handlers for MOUSE_TIMEOUT ms after
    // a touchend to ignore synthetic mouse events
    let mouseCanceller = function(mouseEvent) {
      // Check for sourceCapabilities, used to distinguish synthetic events
      // if mouseEvent did not come from a device that fires touch events,
      // it was made by a real mouse and should be counted
      // http://wicg.github.io/InputDeviceCapabilities/#dom-inputdevicecapabilities-firestouchevents
      let sc = mouseEvent.sourceCapabilities;
      if (sc && !sc.firesTouchEvents) {
        return;
      }
      // skip synthetic mouse events
      mouseEvent[HANDLED_OBJ] = {skip: true};
      // disable "ghost clicks"
      if (mouseEvent.type === 'click') {
        let clickFromLabel = false;
        let path = getComposedPath(mouseEvent);
        for (let i = 0; i < path.length; i++) {
          if (path[i].nodeType === Node.ELEMENT_NODE) {
            if (path[i].localName === 'label') {
              clickedLabels.push(/** @type {!HTMLLabelElement} */ (path[i]));
            } else if (canBeLabelled(/** @type {!HTMLElement} */ (path[i]))) {
              let ownerLabels =
                  matchingLabels(/** @type {!HTMLElement} */ (path[i]));
              // check if one of the clicked labels is labelling this element
              for (let j = 0; j < ownerLabels.length; j++) {
                clickFromLabel = clickFromLabel || clickedLabels.indexOf(ownerLabels[j]) > -1;
              }
            }
          }
          if (path[i] === POINTERSTATE.mouse.target) {
            return;
          }
        }
        // if one of the clicked labels was labelling the target element,
        // this is not a ghost click
        if (clickFromLabel) {
          return;
        }
        mouseEvent.preventDefault();
        mouseEvent.stopPropagation();
      }
    };

    /**
     * @param {boolean=} setup True to add, false to remove.
     * @return {void}
     */
    function setupTeardownMouseCanceller(setup) {
      let events = IS_TOUCH_ONLY ? ['click'] : MOUSE_EVENTS;
      for (let i = 0, en; i < events.length; i++) {
        en = events[i];
        if (setup) {
          // reset clickLabels array
          clickedLabels.length = 0;
          document.addEventListener(en, mouseCanceller, true);
        } else {
          document.removeEventListener(en, mouseCanceller, true);
        }
      }
    }

    function ignoreMouse(e) {
      if (!POINTERSTATE.mouse.mouseIgnoreJob) {
        setupTeardownMouseCanceller(true);
      }
      let unset = function() {
        setupTeardownMouseCanceller();
        POINTERSTATE.mouse.target = null;
        POINTERSTATE.mouse.mouseIgnoreJob = null;
      };
      POINTERSTATE.mouse.target = getComposedPath(e)[0];
      POINTERSTATE.mouse.mouseIgnoreJob = Debouncer.debounce(
            POINTERSTATE.mouse.mouseIgnoreJob
          , timeOut.after(MOUSE_TIMEOUT)
          , unset);
    }

    /**
     * @param {MouseEvent} ev event to test for left mouse button down
     * @return {boolean} has left mouse button down
     */
    function hasLeftMouseButton(ev) {
      let type = ev.type;
      // exit early if the event is not a mouse event
      if (!isMouseEvent(type)) {
        return false;
      }
      // ev.button is not reliable for mousemove (0 is overloaded as both left button and no buttons)
      // instead we use ev.buttons (bitmask of buttons) or fall back to ev.which (deprecated, 0 for no buttons, 1 for left button)
      if (type === 'mousemove') {
        // allow undefined for testing events
        let buttons = ev.buttons === undefined ? 1 : ev.buttons;
        if ((ev instanceof window.MouseEvent) && !MOUSE_HAS_BUTTONS) {
          buttons = MOUSE_WHICH_TO_BUTTONS[ev.which] || 0;
        }
        // buttons is a bitmask, check that the left button bit is set (1)
        return Boolean(buttons & 1);
      } else {
        // allow undefined for testing events
        let button = ev.button === undefined ? 0 : ev.button;
        // ev.button is 0 in mousedown/mouseup/click for left button activation
        return button === 0;
      }
    }

    function isSyntheticClick(ev) {
      if (ev.type === 'click') {
        // ev.detail is 0 for HTMLElement.click in most browsers
        if (ev.detail === 0) {
          return true;
        }
        // in the worst case, check that the x/y position of the click is within
        // the bounding box of the target of the event
        // Thanks IE 10 >:(
        let t = _findOriginalTarget(ev);
        // make sure the target of the event is an element so we can use getBoundingClientRect,
        // if not, just assume it is a synthetic click
        if (!t.nodeType || /** @type {Element} */(t).nodeType !== Node.ELEMENT_NODE) {
          return true;
        }
        let bcr = /** @type {Element} */(t).getBoundingClientRect();
        // use page x/y to account for scrolling
        let x = ev.pageX, y = ev.pageY;
        // ev is a synthetic click if the position is outside the bounding box of the target
        return !((x >= bcr.left && x <= bcr.right) && (y >= bcr.top && y <= bcr.bottom));
      }
      return false;
    }

    let POINTERSTATE = {
      mouse: {
        target: null,
        mouseIgnoreJob: null
      },
      touch: {
        x: 0,
        y: 0,
        id: -1,
        scrollDecided: false
      }
    };

    function firstTouchAction(ev) {
      let ta = 'auto';
      let path = getComposedPath(ev);
      for (let i = 0, n; i < path.length; i++) {
        n = path[i];
        if (n[TOUCH_ACTION]) {
          ta = n[TOUCH_ACTION];
          break;
        }
      }
      return ta;
    }

    function trackDocument(stateObj, movefn, upfn) {
      stateObj.movefn = movefn;
      stateObj.upfn = upfn;
      document.addEventListener('mousemove', movefn);
      document.addEventListener('mouseup', upfn);
    }

    function untrackDocument(stateObj) {
      document.removeEventListener('mousemove', stateObj.movefn);
      document.removeEventListener('mouseup', stateObj.upfn);
      stateObj.movefn = null;
      stateObj.upfn = null;
    }

    {
      // use a document-wide touchend listener to start the ghost-click prevention mechanism
      // Use passive event listeners, if supported, to not affect scrolling performance
      document.addEventListener('touchend', ignoreMouse, supportsPassive ? {passive: true} : false);
    }

    /**
     * Returns the composedPath for the given event.
     * @param {Event} event to process
     * @return {!Array<!EventTarget>} Path of the event
     */
    const getComposedPath = window.ShadyDOM && window.ShadyDOM.noPatch ?
      window.ShadyDOM.composedPath :
      (event) => event.composedPath && event.composedPath() || [];

    /** @type {!Object<string, !GestureRecognizer>} */
    const gestures = {};

    /** @type {!Array<!GestureRecognizer>} */
    const recognizers = [];

    /**
     * Finds the element rendered on the screen at the provided coordinates.
     *
     * Similar to `document.elementFromPoint`, but pierces through
     * shadow roots.
     *
     * @param {number} x Horizontal pixel coordinate
     * @param {number} y Vertical pixel coordinate
     * @return {Element} Returns the deepest shadowRoot inclusive element
     * found at the screen position given.
     */
    function deepTargetFind(x, y) {
      let node = document.elementFromPoint(x, y);
      let next = node;
      // this code path is only taken when native ShadowDOM is used
      // if there is a shadowroot, it may have a node at x/y
      // if there is not a shadowroot, exit the loop
      while (next && next.shadowRoot && !window.ShadyDOM) {
        // if there is a node at x/y in the shadowroot, look deeper
        let oldNext = next;
        next = next.shadowRoot.elementFromPoint(x, y);
        // on Safari, elementFromPoint may return the shadowRoot host
        if (oldNext === next) {
          break;
        }
        if (next) {
          node = next;
        }
      }
      return node;
    }

    /**
     * a cheaper check than ev.composedPath()[0];
     *
     * @private
     * @param {Event|Touch} ev Event.
     * @return {EventTarget} Returns the event target.
     */
    function _findOriginalTarget(ev) {
      const path = getComposedPath(/** @type {?Event} */ (ev));
      // It shouldn't be, but sometimes path is empty (window on Safari).
      return path.length > 0 ? path[0] : ev.target;
    }

    /**
     * @private
     * @param {Event} ev Event.
     * @return {void}
     */
    function _handleNative(ev) {
      let handled;
      let type = ev.type;
      let node = ev.currentTarget;
      let gobj = node[GESTURE_KEY];
      if (!gobj) {
        return;
      }
      let gs = gobj[type];
      if (!gs) {
        return;
      }
      if (!ev[HANDLED_OBJ]) {
        ev[HANDLED_OBJ] = {};
        if (type.slice(0, 5) === 'touch') {
          ev = /** @type {TouchEvent} */(ev); // eslint-disable-line no-self-assign
          let t = ev.changedTouches[0];
          if (type === 'touchstart') {
            // only handle the first finger
            if (ev.touches.length === 1) {
              POINTERSTATE.touch.id = t.identifier;
            }
          }
          if (POINTERSTATE.touch.id !== t.identifier) {
            return;
          }
          if (!HAS_NATIVE_TA) {
            if (type === 'touchstart' || type === 'touchmove') {
              _handleTouchAction(ev);
            }
          }
        }
      }
      handled = ev[HANDLED_OBJ];
      // used to ignore synthetic mouse events
      if (handled.skip) {
        return;
      }
      // reset recognizer state
      for (let i = 0, r; i < recognizers.length; i++) {
        r = recognizers[i];
        if (gs[r.name] && !handled[r.name]) {
          if (r.flow && r.flow.start.indexOf(ev.type) > -1 && r.reset) {
            r.reset();
          }
        }
      }
      // enforce gesture recognizer order
      for (let i = 0, r; i < recognizers.length; i++) {
        r = recognizers[i];
        if (gs[r.name] && !handled[r.name]) {
          handled[r.name] = true;
          r[type](ev);
        }
      }
    }

    /**
     * @private
     * @param {TouchEvent} ev Event.
     * @return {void}
     */
    function _handleTouchAction(ev) {
      let t = ev.changedTouches[0];
      let type = ev.type;
      if (type === 'touchstart') {
        POINTERSTATE.touch.x = t.clientX;
        POINTERSTATE.touch.y = t.clientY;
        POINTERSTATE.touch.scrollDecided = false;
      } else if (type === 'touchmove') {
        if (POINTERSTATE.touch.scrollDecided) {
          return;
        }
        POINTERSTATE.touch.scrollDecided = true;
        let ta = firstTouchAction(ev);
        let shouldPrevent = false;
        let dx = Math.abs(POINTERSTATE.touch.x - t.clientX);
        let dy = Math.abs(POINTERSTATE.touch.y - t.clientY);
        if (!ev.cancelable) ; else if (ta === 'none') {
          shouldPrevent = true;
        } else if (ta === 'pan-x') {
          shouldPrevent = dy > dx;
        } else if (ta === 'pan-y') {
          shouldPrevent = dx > dy;
        }
        if (shouldPrevent) {
          ev.preventDefault();
        } else {
          prevent('track');
        }
      }
    }

    /**
     * Adds an event listener to a node for the given gesture type.
     *
     * @param {!EventTarget} node Node to add listener on
     * @param {string} evType Gesture type: `down`, `up`, `track`, or `tap`
     * @param {!function(!Event):void} handler Event listener function to call
     * @return {boolean} Returns true if a gesture event listener was added.
     */
    function addListener(node, evType, handler) {
      if (gestures[evType]) {
        _add(node, evType, handler);
        return true;
      }
      return false;
    }

    /**
     * Removes an event listener from a node for the given gesture type.
     *
     * @param {!EventTarget} node Node to remove listener from
     * @param {string} evType Gesture type: `down`, `up`, `track`, or `tap`
     * @param {!function(!Event):void} handler Event listener function previously passed to
     *  `addListener`.
     * @return {boolean} Returns true if a gesture event listener was removed.
     */
    function removeListener(node, evType, handler) {
      if (gestures[evType]) {
        _remove(node, evType, handler);
        return true;
      }
      return false;
    }

    /**
     * automate the event listeners for the native events
     *
     * @private
     * @param {!EventTarget} node Node on which to add the event.
     * @param {string} evType Event type to add.
     * @param {function(!Event)} handler Event handler function.
     * @return {void}
     */
    function _add(node, evType, handler) {
      let recognizer = gestures[evType];
      let deps = recognizer.deps;
      let name = recognizer.name;
      let gobj = node[GESTURE_KEY];
      if (!gobj) {
        node[GESTURE_KEY] = gobj = {};
      }
      for (let i = 0, dep, gd; i < deps.length; i++) {
        dep = deps[i];
        // don't add mouse handlers on iOS because they cause gray selection overlays
        if (IS_TOUCH_ONLY && isMouseEvent(dep) && dep !== 'click') {
          continue;
        }
        gd = gobj[dep];
        if (!gd) {
          gobj[dep] = gd = {_count: 0};
        }
        if (gd._count === 0) {
          node.addEventListener(dep, _handleNative, PASSIVE_TOUCH(dep));
        }
        gd[name] = (gd[name] || 0) + 1;
        gd._count = (gd._count || 0) + 1;
      }
      node.addEventListener(evType, handler);
      if (recognizer.touchAction) {
        setTouchAction(node, recognizer.touchAction);
      }
    }

    /**
     * automate event listener removal for native events
     *
     * @private
     * @param {!EventTarget} node Node on which to remove the event.
     * @param {string} evType Event type to remove.
     * @param {function(!Event): void} handler Event handler function.
     * @return {void}
     */
    function _remove(node, evType, handler) {
      let recognizer = gestures[evType];
      let deps = recognizer.deps;
      let name = recognizer.name;
      let gobj = node[GESTURE_KEY];
      if (gobj) {
        for (let i = 0, dep, gd; i < deps.length; i++) {
          dep = deps[i];
          gd = gobj[dep];
          if (gd && gd[name]) {
            gd[name] = (gd[name] || 1) - 1;
            gd._count = (gd._count || 1) - 1;
            if (gd._count === 0) {
              node.removeEventListener(dep, _handleNative, PASSIVE_TOUCH(dep));
            }
          }
        }
      }
      node.removeEventListener(evType, handler);
    }

    /**
     * Registers a new gesture event recognizer for adding new custom
     * gesture event types.
     *
     * @param {!GestureRecognizer} recog Gesture recognizer descriptor
     * @return {void}
     */
    function register$1(recog) {
      recognizers.push(recog);
      for (let i = 0; i < recog.emits.length; i++) {
        gestures[recog.emits[i]] = recog;
      }
    }

    /**
     * @private
     * @param {string} evName Event name.
     * @return {Object} Returns the gesture for the given event name.
     */
    function _findRecognizerByEvent(evName) {
      for (let i = 0, r; i < recognizers.length; i++) {
        r = recognizers[i];
        for (let j = 0, n; j < r.emits.length; j++) {
          n = r.emits[j];
          if (n === evName) {
            return r;
          }
        }
      }
      return null;
    }

    /**
     * Sets scrolling direction on node.
     *
     * This value is checked on first move, thus it should be called prior to
     * adding event listeners.
     *
     * @param {!EventTarget} node Node to set touch action setting on
     * @param {string} value Touch action value
     * @return {void}
     */
    function setTouchAction(node, value) {
      if (HAS_NATIVE_TA && node instanceof HTMLElement) {
        // NOTE: add touchAction async so that events can be added in
        // custom element constructors. Otherwise we run afoul of custom
        // elements restriction against settings attributes (style) in the
        // constructor.
        microTask.run(() => {
          node.style.touchAction = value;
        });
      }
      node[TOUCH_ACTION] = value;
    }

    /**
     * Dispatches an event on the `target` element of `type` with the given
     * `detail`.
     * @private
     * @param {!EventTarget} target The element on which to fire an event.
     * @param {string} type The type of event to fire.
     * @param {!Object=} detail The detail object to populate on the event.
     * @return {void}
     */
    function _fire(target, type, detail) {
      let ev = new Event(type, { bubbles: true, cancelable: true, composed: true });
      ev.detail = detail;
      wrap(/** @type {!Node} */(target)).dispatchEvent(ev);
      // forward `preventDefault` in a clean way
      if (ev.defaultPrevented) {
        let preventer = detail.preventer || detail.sourceEvent;
        if (preventer && preventer.preventDefault) {
          preventer.preventDefault();
        }
      }
    }

    /**
     * Prevents the dispatch and default action of the given event name.
     *
     * @param {string} evName Event name.
     * @return {void}
     */
    function prevent(evName) {
      let recognizer = _findRecognizerByEvent(evName);
      if (recognizer.info) {
        recognizer.info.prevent = true;
      }
    }

    /* eslint-disable valid-jsdoc */

    register$1({
      name: 'downup',
      deps: ['mousedown', 'touchstart', 'touchend'],
      flow: {
        start: ['mousedown', 'touchstart'],
        end: ['mouseup', 'touchend']
      },
      emits: ['down', 'up'],

      info: {
        movefn: null,
        upfn: null
      },

      /**
       * @this {GestureRecognizer}
       * @return {void}
       */
      reset: function() {
        untrackDocument(this.info);
      },

      /**
       * @this {GestureRecognizer}
       * @param {MouseEvent} e
       * @return {void}
       */
      mousedown: function(e) {
        if (!hasLeftMouseButton(e)) {
          return;
        }
        let t = _findOriginalTarget(e);
        let self = this;
        let movefn = function movefn(e) {
          if (!hasLeftMouseButton(e)) {
            downupFire('up', t, e);
            untrackDocument(self.info);
          }
        };
        let upfn = function upfn(e) {
          if (hasLeftMouseButton(e)) {
            downupFire('up', t, e);
          }
          untrackDocument(self.info);
        };
        trackDocument(this.info, movefn, upfn);
        downupFire('down', t, e);
      },
      /**
       * @this {GestureRecognizer}
       * @param {TouchEvent} e
       * @return {void}
       */
      touchstart: function(e) {
        downupFire('down', _findOriginalTarget(e), e.changedTouches[0], e);
      },
      /**
       * @this {GestureRecognizer}
       * @param {TouchEvent} e
       * @return {void}
       */
      touchend: function(e) {
        downupFire('up', _findOriginalTarget(e), e.changedTouches[0], e);
      }
    });

    /**
     * @param {string} type
     * @param {EventTarget} target
     * @param {Event|Touch} event
     * @param {Event=} preventer
     * @return {void}
     */
    function downupFire(type, target, event, preventer) {
      if (!target) {
        return;
      }
      _fire(target, type, {
        x: event.clientX,
        y: event.clientY,
        sourceEvent: event,
        preventer: preventer,
        prevent: function(e) {
          return prevent(e);
        }
      });
    }

    register$1({
      name: 'track',
      touchAction: 'none',
      deps: ['mousedown', 'touchstart', 'touchmove', 'touchend'],
      flow: {
        start: ['mousedown', 'touchstart'],
        end: ['mouseup', 'touchend']
      },
      emits: ['track'],

      info: {
        x: 0,
        y: 0,
        state: 'start',
        started: false,
        moves: [],
        /** @this {GestureInfo} */
        addMove: function(move) {
          if (this.moves.length > TRACK_LENGTH) {
            this.moves.shift();
          }
          this.moves.push(move);
        },
        movefn: null,
        upfn: null,
        prevent: false
      },

      /**
       * @this {GestureRecognizer}
       * @return {void}
       */
      reset: function() {
        this.info.state = 'start';
        this.info.started = false;
        this.info.moves = [];
        this.info.x = 0;
        this.info.y = 0;
        this.info.prevent = false;
        untrackDocument(this.info);
      },

      /**
       * @this {GestureRecognizer}
       * @param {MouseEvent} e
       * @return {void}
       */
      mousedown: function(e) {
        if (!hasLeftMouseButton(e)) {
          return;
        }
        let t = _findOriginalTarget(e);
        let self = this;
        let movefn = function movefn(e) {
          let x = e.clientX, y = e.clientY;
          if (trackHasMovedEnough(self.info, x, y)) {
            // first move is 'start', subsequent moves are 'move', mouseup is 'end'
            self.info.state = self.info.started ? (e.type === 'mouseup' ? 'end' : 'track') : 'start';
            if (self.info.state === 'start') {
              // if and only if tracking, always prevent tap
              prevent('tap');
            }
            self.info.addMove({x: x, y: y});
            if (!hasLeftMouseButton(e)) {
              // always fire "end"
              self.info.state = 'end';
              untrackDocument(self.info);
            }
            if (t) {
              trackFire(self.info, t, e);
            }
            self.info.started = true;
          }
        };
        let upfn = function upfn(e) {
          if (self.info.started) {
            movefn(e);
          }

          // remove the temporary listeners
          untrackDocument(self.info);
        };
        // add temporary document listeners as mouse retargets
        trackDocument(this.info, movefn, upfn);
        this.info.x = e.clientX;
        this.info.y = e.clientY;
      },
      /**
       * @this {GestureRecognizer}
       * @param {TouchEvent} e
       * @return {void}
       */
      touchstart: function(e) {
        let ct = e.changedTouches[0];
        this.info.x = ct.clientX;
        this.info.y = ct.clientY;
      },
      /**
       * @this {GestureRecognizer}
       * @param {TouchEvent} e
       * @return {void}
       */
      touchmove: function(e) {
        let t = _findOriginalTarget(e);
        let ct = e.changedTouches[0];
        let x = ct.clientX, y = ct.clientY;
        if (trackHasMovedEnough(this.info, x, y)) {
          if (this.info.state === 'start') {
            // if and only if tracking, always prevent tap
            prevent('tap');
          }
          this.info.addMove({x: x, y: y});
          trackFire(this.info, t, ct);
          this.info.state = 'track';
          this.info.started = true;
        }
      },
      /**
       * @this {GestureRecognizer}
       * @param {TouchEvent} e
       * @return {void}
       */
      touchend: function(e) {
        let t = _findOriginalTarget(e);
        let ct = e.changedTouches[0];
        // only trackend if track was started and not aborted
        if (this.info.started) {
          // reset started state on up
          this.info.state = 'end';
          this.info.addMove({x: ct.clientX, y: ct.clientY});
          trackFire(this.info, t, ct);
        }
      }
    });

    /**
     * @param {!GestureInfo} info
     * @param {number} x
     * @param {number} y
     * @return {boolean}
     */
    function trackHasMovedEnough(info, x, y) {
      if (info.prevent) {
        return false;
      }
      if (info.started) {
        return true;
      }
      let dx = Math.abs(info.x - x);
      let dy = Math.abs(info.y - y);
      return (dx >= TRACK_DISTANCE || dy >= TRACK_DISTANCE);
    }

    /**
     * @param {!GestureInfo} info
     * @param {?EventTarget} target
     * @param {Touch} touch
     * @return {void}
     */
    function trackFire(info, target, touch) {
      if (!target) {
        return;
      }
      let secondlast = info.moves[info.moves.length - 2];
      let lastmove = info.moves[info.moves.length - 1];
      let dx = lastmove.x - info.x;
      let dy = lastmove.y - info.y;
      let ddx, ddy = 0;
      if (secondlast) {
        ddx = lastmove.x - secondlast.x;
        ddy = lastmove.y - secondlast.y;
      }
      _fire(target, 'track', {
        state: info.state,
        x: touch.clientX,
        y: touch.clientY,
        dx: dx,
        dy: dy,
        ddx: ddx,
        ddy: ddy,
        sourceEvent: touch,
        hover: function() {
          return deepTargetFind(touch.clientX, touch.clientY);
        }
      });
    }

    register$1({
      name: 'tap',
      deps: ['mousedown', 'click', 'touchstart', 'touchend'],
      flow: {
        start: ['mousedown', 'touchstart'],
        end: ['click', 'touchend']
      },
      emits: ['tap'],
      info: {
        x: NaN,
        y: NaN,
        prevent: false
      },
      /**
       * @this {GestureRecognizer}
       * @return {void}
       */
      reset: function() {
        this.info.x = NaN;
        this.info.y = NaN;
        this.info.prevent = false;
      },
      /**
       * @this {GestureRecognizer}
       * @param {MouseEvent} e
       * @return {void}
       */
      mousedown: function(e) {
        if (hasLeftMouseButton(e)) {
          this.info.x = e.clientX;
          this.info.y = e.clientY;
        }
      },
      /**
       * @this {GestureRecognizer}
       * @param {MouseEvent} e
       * @return {void}
       */
      click: function(e) {
        if (hasLeftMouseButton(e)) {
          trackForward(this.info, e);
        }
      },
      /**
       * @this {GestureRecognizer}
       * @param {TouchEvent} e
       * @return {void}
       */
      touchstart: function(e) {
        const touch = e.changedTouches[0];
        this.info.x = touch.clientX;
        this.info.y = touch.clientY;
      },
      /**
       * @this {GestureRecognizer}
       * @param {TouchEvent} e
       * @return {void}
       */
      touchend: function(e) {
        trackForward(this.info, e.changedTouches[0], e);
      }
    });

    /**
     * @param {!GestureInfo} info
     * @param {Event | Touch} e
     * @param {Event=} preventer
     * @return {void}
     */
    function trackForward(info, e, preventer) {
      let dx = Math.abs(e.clientX - info.x);
      let dy = Math.abs(e.clientY - info.y);
      // find original target from `preventer` for TouchEvents, or `e` for MouseEvents
      let t = _findOriginalTarget((preventer || e));
      if (!t || (canBeDisabled[/** @type {!HTMLElement} */(t).localName] && t.hasAttribute('disabled'))) {
        return;
      }
      // dx,dy can be NaN if `click` has been simulated and there was no `down` for `start`
      if (isNaN(dx) || isNaN(dy) || (dx <= TAP_DISTANCE && dy <= TAP_DISTANCE) || isSyntheticClick(e)) {
        // prevent taps from being generated if an event has canceled them
        if (!info.prevent) {
          _fire(t, 'tap', {
            x: e.clientX,
            y: e.clientY,
            sourceEvent: e,
            preventer: preventer
          });
        }
      }
    }

    /* eslint-enable valid-jsdoc */

    /** @deprecated */
    const findOriginalTarget = _findOriginalTarget;

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    /**
     * Element class mixin that provides API for adding Polymer's cross-platform
     * gesture events to nodes.
     *
     * The API is designed to be compatible with override points implemented
     * in `TemplateStamp` such that declarative event listeners in
     * templates will support gesture events when this mixin is applied along with
     * `TemplateStamp`.
     *
     * @mixinFunction
     * @polymer
     * @summary Element class mixin that provides API for adding Polymer's
     *   cross-platform gesture events to nodes
     * @template T
     * @param {function(new:T)} superClass Class to apply mixin to.
     * @return {function(new:T)} superClass with mixin applied.
     */
    const GestureEventListeners = dedupingMixin((superClass) => {
      /**
       * @polymer
       * @mixinClass
       * @implements {Polymer_GestureEventListeners}
       */
      class GestureEventListeners extends superClass {
        /**
         * Add the event listener to the node if it is a gestures event.
         *
         * @param {!EventTarget} node Node to add event listener to
         * @param {string} eventName Name of event
         * @param {function(!Event):void} handler Listener function to add
         * @return {void}
         * @override
         */
        _addEventListenerToNode(node, eventName, handler) {
          if (!addListener(node, eventName, handler)) {
            super._addEventListenerToNode(node, eventName, handler);
          }
        }

        /**
         * Remove the event listener to the node if it is a gestures event.
         *
         * @param {!EventTarget} node Node to remove event listener from
         * @param {string} eventName Name of event
         * @param {function(!Event):void} handler Listener function to remove
         * @return {void}
         * @override
         */
        _removeEventListenerFromNode(node, eventName, handler) {
          if (!removeListener(node, eventName, handler)) {
            super._removeEventListenerFromNode(node, eventName, handler);
          }
        }
      }

      return GestureEventListeners;
    });

    /**
     * @fileoverview
     * @suppress {checkPrototypalTypes}
     * @license Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
     * This code may only be used under the BSD style license found at
     * http://polymer.github.io/LICENSE.txt The complete set of authors may be found
     * at http://polymer.github.io/AUTHORS.txt The complete set of contributors may
     * be found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by
     * Google as part of the polymer project is also subject to an additional IP
     * rights grant found at http://polymer.github.io/PATENTS.txt
     */

    const HOST_DIR = /:host\(:dir\((ltr|rtl)\)\)/g;
    const HOST_DIR_REPLACMENT = ':host([dir="$1"])';

    const EL_DIR = /([\s\w-#\.\[\]\*]*):dir\((ltr|rtl)\)/g;
    const EL_DIR_REPLACMENT = ':host([dir="$2"]) $1';

    const DIR_CHECK = /:dir\((?:ltr|rtl)\)/;

    const SHIM_SHADOW = Boolean(window['ShadyDOM'] && window['ShadyDOM']['inUse']);

    /**
     * @type {!Array<!Polymer_DirMixin>}
     */
    const DIR_INSTANCES = [];

    /** @type {?MutationObserver} */
    let observer = null;

    let documentDir = '';

    function getRTL() {
      documentDir = document.documentElement.getAttribute('dir');
    }

    /**
     * @param {!Polymer_DirMixin} instance Instance to set RTL status on
     */
    function setRTL(instance) {
      if (!instance.__autoDirOptOut) {
        const el = /** @type {!HTMLElement} */(instance);
        el.setAttribute('dir', documentDir);
      }
    }

    function updateDirection() {
      getRTL();
      documentDir = document.documentElement.getAttribute('dir');
      for (let i = 0; i < DIR_INSTANCES.length; i++) {
        setRTL(DIR_INSTANCES[i]);
      }
    }

    function takeRecords() {
      if (observer && observer.takeRecords().length) {
        updateDirection();
      }
    }

    /**
     * Element class mixin that allows elements to use the `:dir` CSS Selector to
     * have text direction specific styling.
     *
     * With this mixin, any stylesheet provided in the template will transform
     * `:dir` into `:host([dir])` and sync direction with the page via the
     * element's `dir` attribute.
     *
     * Elements can opt out of the global page text direction by setting the `dir`
     * attribute directly in `ready()` or in HTML.
     *
     * Caveats:
     * - Applications must set `<html dir="ltr">` or `<html dir="rtl">` to sync
     *   direction
     * - Automatic left-to-right or right-to-left styling is sync'd with the
     *   `<html>` element only.
     * - Changing `dir` at runtime is supported.
     * - Opting out of the global direction styling is permanent
     *
     * @mixinFunction
     * @polymer
     * @appliesMixin PropertyAccessors
     * @template T
     * @param {function(new:T)} superClass Class to apply mixin to.
     * @return {function(new:T)} superClass with mixin applied.
     */
    const DirMixin = dedupingMixin((base) => {

      if (!SHIM_SHADOW) {
        if (!observer) {
          getRTL();
          observer = new MutationObserver(updateDirection);
          observer.observe(document.documentElement, {attributes: true, attributeFilter: ['dir']});
        }
      }

      /**
       * @constructor
       * @implements {Polymer_PropertyAccessors}
       * @private
       */
      const elementBase = PropertyAccessors(base);

      /**
       * @polymer
       * @mixinClass
       * @implements {Polymer_DirMixin}
       */
      class Dir extends elementBase {

        /**
         * @param {string} cssText .
         * @param {string} baseURI .
         * @return {string} .
         * @suppress {missingProperties} Interfaces in closure do not inherit statics, but classes do
         * @nocollapse
         */
        static _processStyleText(cssText, baseURI) {
          // TODO(https://github.com/google/closure-compiler/issues/3240):
          //     Change back to just super.methodCall()
          cssText = elementBase._processStyleText.call(this, cssText, baseURI);
          if (!SHIM_SHADOW && DIR_CHECK.test(cssText)) {
            cssText = this._replaceDirInCssText(cssText);
            this.__activateDir = true;
          }
          return cssText;
        }

        /**
         * Replace `:dir` in the given CSS text
         *
         * @param {string} text CSS text to replace DIR
         * @return {string} Modified CSS
         * @nocollapse
         */
        static _replaceDirInCssText(text) {
          let replacedText = text;
          replacedText = replacedText.replace(HOST_DIR, HOST_DIR_REPLACMENT);
          replacedText = replacedText.replace(EL_DIR, EL_DIR_REPLACMENT);
          return replacedText;
        }

        constructor() {
          super();
          /** @type {boolean} */
          this.__autoDirOptOut = false;
        }

        /**
         * @override
         * @suppress {invalidCasts} Closure doesn't understand that `this` is an
         *     HTMLElement
         * @return {void}
         */
        ready() {
          super.ready();
          this.__autoDirOptOut = /** @type {!HTMLElement} */(this).hasAttribute('dir');
        }

        /**
         * @override
         * @suppress {missingProperties} If it exists on elementBase, it can be
         *   super'd
         * @return {void}
         */
        connectedCallback() {
          if (elementBase.prototype.connectedCallback) {
            super.connectedCallback();
          }
          if (this.constructor.__activateDir) {
            takeRecords();
            DIR_INSTANCES.push(this);
            setRTL(this);
          }
        }

        /**
         * @override
         * @suppress {missingProperties} If it exists on elementBase, it can be
         *   super'd
         * @return {void}
         */
        disconnectedCallback() {
          if (elementBase.prototype.disconnectedCallback) {
            super.disconnectedCallback();
          }
          if (this.constructor.__activateDir) {
            const idx = DIR_INSTANCES.indexOf(this);
            if (idx > -1) {
              DIR_INSTANCES.splice(idx, 1);
            }
          }
        }
      }

      Dir.__activateDir = false;

      return Dir;
    });

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    let scheduled = false;
    let beforeRenderQueue = [];
    let afterRenderQueue = [];

    function schedule() {
      scheduled = true;
      // before next render
      requestAnimationFrame(function() {
        scheduled = false;
        flushQueue(beforeRenderQueue);
        // after the render
        setTimeout(function() {
          runQueue(afterRenderQueue);
        });
      });
    }

    function flushQueue(queue) {
      while (queue.length) {
        callMethod(queue.shift());
      }
    }

    function runQueue(queue) {
      for (let i=0, l=queue.length; i < l; i++) {
        callMethod(queue.shift());
      }
    }

    function callMethod(info) {
      const context = info[0];
      const callback = info[1];
      const args = info[2];
      try {
        callback.apply(context, args);
      } catch(e) {
        setTimeout(() => {
          throw e;
        });
      }
    }


    /**
     * Enqueues a callback which will be run before the next render, at
     * `requestAnimationFrame` timing.
     *
     * This method is useful for enqueuing work that requires DOM measurement,
     * since measurement may not be reliable in custom element callbacks before
     * the first render, as well as for batching measurement tasks in general.
     *
     * Tasks in this queue may be flushed by calling `flush()`.
     *
     * @param {*} context Context object the callback function will be bound to
     * @param {function(...*):void} callback Callback function
     * @param {!Array=} args An array of arguments to call the callback function with
     * @return {void}
     */
    function beforeNextRender(context, callback, args) {
      if (!scheduled) {
        schedule();
      }
      beforeRenderQueue.push([context, callback, args]);
    }

    /**
     * Enqueues a callback which will be run after the next render, equivalent
     * to one task (`setTimeout`) after the next `requestAnimationFrame`.
     *
     * This method is useful for tuning the first-render performance of an
     * element or application by deferring non-critical work until after the
     * first paint.  Typical non-render-critical work may include adding UI
     * event listeners and aria attributes.
     *
     * @param {*} context Context object the callback function will be bound to
     * @param {function(...*):void} callback Callback function
     * @param {!Array=} args An array of arguments to call the callback function with
     * @return {void}
     */
    function afterNextRender(context, callback, args) {
      if (!scheduled) {
        schedule();
      }
      afterRenderQueue.push([context, callback, args]);
    }

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    function resolve() {
      document.body.removeAttribute('unresolved');
    }

    if (document.readyState === 'interactive' || document.readyState === 'complete') {
      resolve();
    } else {
      window.addEventListener('DOMContentLoaded', resolve);
    }

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    function newSplice(index, removed, addedCount) {
      return {
        index: index,
        removed: removed,
        addedCount: addedCount
      };
    }

    const EDIT_LEAVE = 0;
    const EDIT_UPDATE = 1;
    const EDIT_ADD = 2;
    const EDIT_DELETE = 3;

    // Note: This function is *based* on the computation of the Levenshtein
    // "edit" distance. The one change is that "updates" are treated as two
    // edits - not one. With Array splices, an update is really a delete
    // followed by an add. By retaining this, we optimize for "keeping" the
    // maximum array items in the original array. For example:
    //
    //   'xxxx123' -> '123yyyy'
    //
    // With 1-edit updates, the shortest path would be just to update all seven
    // characters. With 2-edit updates, we delete 4, leave 3, and add 4. This
    // leaves the substring '123' intact.
    function calcEditDistances(current, currentStart, currentEnd,
                                old, oldStart, oldEnd) {
      // "Deletion" columns
      let rowCount = oldEnd - oldStart + 1;
      let columnCount = currentEnd - currentStart + 1;
      let distances = new Array(rowCount);

      // "Addition" rows. Initialize null column.
      for (let i = 0; i < rowCount; i++) {
        distances[i] = new Array(columnCount);
        distances[i][0] = i;
      }

      // Initialize null row
      for (let j = 0; j < columnCount; j++)
        distances[0][j] = j;

      for (let i = 1; i < rowCount; i++) {
        for (let j = 1; j < columnCount; j++) {
          if (equals(current[currentStart + j - 1], old[oldStart + i - 1]))
            distances[i][j] = distances[i - 1][j - 1];
          else {
            let north = distances[i - 1][j] + 1;
            let west = distances[i][j - 1] + 1;
            distances[i][j] = north < west ? north : west;
          }
        }
      }

      return distances;
    }

    // This starts at the final weight, and walks "backward" by finding
    // the minimum previous weight recursively until the origin of the weight
    // matrix.
    function spliceOperationsFromEditDistances(distances) {
      let i = distances.length - 1;
      let j = distances[0].length - 1;
      let current = distances[i][j];
      let edits = [];
      while (i > 0 || j > 0) {
        if (i == 0) {
          edits.push(EDIT_ADD);
          j--;
          continue;
        }
        if (j == 0) {
          edits.push(EDIT_DELETE);
          i--;
          continue;
        }
        let northWest = distances[i - 1][j - 1];
        let west = distances[i - 1][j];
        let north = distances[i][j - 1];

        let min;
        if (west < north)
          min = west < northWest ? west : northWest;
        else
          min = north < northWest ? north : northWest;

        if (min == northWest) {
          if (northWest == current) {
            edits.push(EDIT_LEAVE);
          } else {
            edits.push(EDIT_UPDATE);
            current = northWest;
          }
          i--;
          j--;
        } else if (min == west) {
          edits.push(EDIT_DELETE);
          i--;
          current = west;
        } else {
          edits.push(EDIT_ADD);
          j--;
          current = north;
        }
      }

      edits.reverse();
      return edits;
    }

    /**
     * Splice Projection functions:
     *
     * A splice map is a representation of how a previous array of items
     * was transformed into a new array of items. Conceptually it is a list of
     * tuples of
     *
     *   <index, removed, addedCount>
     *
     * which are kept in ascending index order of. The tuple represents that at
     * the |index|, |removed| sequence of items were removed, and counting forward
     * from |index|, |addedCount| items were added.
     */

    /**
     * Lacking individual splice mutation information, the minimal set of
     * splices can be synthesized given the previous state and final state of an
     * array. The basic approach is to calculate the edit distance matrix and
     * choose the shortest path through it.
     *
     * Complexity: O(l * p)
     *   l: The length of the current array
     *   p: The length of the old array
     *
     * @param {!Array} current The current "changed" array for which to
     * calculate splices.
     * @param {number} currentStart Starting index in the `current` array for
     * which splices are calculated.
     * @param {number} currentEnd Ending index in the `current` array for
     * which splices are calculated.
     * @param {!Array} old The original "unchanged" array to compare `current`
     * against to determine splices.
     * @param {number} oldStart Starting index in the `old` array for
     * which splices are calculated.
     * @param {number} oldEnd Ending index in the `old` array for
     * which splices are calculated.
     * @return {!Array} Returns an array of splice record objects. Each of these
     * contains: `index` the location where the splice occurred; `removed`
     * the array of removed items from this location; `addedCount` the number
     * of items added at this location.
     */
    function calcSplices(current, currentStart, currentEnd,
                          old, oldStart, oldEnd) {
      let prefixCount = 0;
      let suffixCount = 0;
      let splice;

      let minLength = Math.min(currentEnd - currentStart, oldEnd - oldStart);
      if (currentStart == 0 && oldStart == 0)
        prefixCount = sharedPrefix(current, old, minLength);

      if (currentEnd == current.length && oldEnd == old.length)
        suffixCount = sharedSuffix(current, old, minLength - prefixCount);

      currentStart += prefixCount;
      oldStart += prefixCount;
      currentEnd -= suffixCount;
      oldEnd -= suffixCount;

      if (currentEnd - currentStart == 0 && oldEnd - oldStart == 0)
        return [];

      if (currentStart == currentEnd) {
        splice = newSplice(currentStart, [], 0);
        while (oldStart < oldEnd)
          splice.removed.push(old[oldStart++]);

        return [ splice ];
      } else if (oldStart == oldEnd)
        return [ newSplice(currentStart, [], currentEnd - currentStart) ];

      let ops = spliceOperationsFromEditDistances(
          calcEditDistances(current, currentStart, currentEnd,
                                 old, oldStart, oldEnd));

      splice = undefined;
      let splices = [];
      let index = currentStart;
      let oldIndex = oldStart;
      for (let i = 0; i < ops.length; i++) {
        switch(ops[i]) {
          case EDIT_LEAVE:
            if (splice) {
              splices.push(splice);
              splice = undefined;
            }

            index++;
            oldIndex++;
            break;
          case EDIT_UPDATE:
            if (!splice)
              splice = newSplice(index, [], 0);

            splice.addedCount++;
            index++;

            splice.removed.push(old[oldIndex]);
            oldIndex++;
            break;
          case EDIT_ADD:
            if (!splice)
              splice = newSplice(index, [], 0);

            splice.addedCount++;
            index++;
            break;
          case EDIT_DELETE:
            if (!splice)
              splice = newSplice(index, [], 0);

            splice.removed.push(old[oldIndex]);
            oldIndex++;
            break;
        }
      }

      if (splice) {
        splices.push(splice);
      }
      return splices;
    }

    function sharedPrefix(current, old, searchLength) {
      for (let i = 0; i < searchLength; i++)
        if (!equals(current[i], old[i]))
          return i;
      return searchLength;
    }

    function sharedSuffix(current, old, searchLength) {
      let index1 = current.length;
      let index2 = old.length;
      let count = 0;
      while (count < searchLength && equals(current[--index1], old[--index2]))
        count++;

      return count;
    }

    /**
     * Returns an array of splice records indicating the minimum edits required
     * to transform the `previous` array into the `current` array.
     *
     * Splice records are ordered by index and contain the following fields:
     * - `index`: index where edit started
     * - `removed`: array of removed items from this index
     * - `addedCount`: number of items added at this index
     *
     * This function is based on the Levenshtein "minimum edit distance"
     * algorithm. Note that updates are treated as removal followed by addition.
     *
     * The worst-case time complexity of this algorithm is `O(l * p)`
     *   l: The length of the current array
     *   p: The length of the previous array
     *
     * However, the worst-case complexity is reduced by an `O(n)` optimization
     * to detect any shared prefix & suffix between the two arrays and only
     * perform the more expensive minimum edit distance calculation over the
     * non-shared portions of the arrays.
     *
     * @function
     * @param {!Array} current The "changed" array for which splices will be
     * calculated.
     * @param {!Array} previous The "unchanged" original array to compare
     * `current` against to determine the splices.
     * @return {!Array} Returns an array of splice record objects. Each of these
     * contains: `index` the location where the splice occurred; `removed`
     * the array of removed items from this location; `addedCount` the number
     * of items added at this location.
     */
    function calculateSplices(current, previous) {
      return calcSplices(current, 0, current.length, previous, 0,
                              previous.length);
    }

    function equals(currentValue, previousValue) {
      return currentValue === previousValue;
    }

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    /**
     * Returns true if `node` is a slot element
     * @param {!Node} node Node to test.
     * @return {boolean} Returns true if the given `node` is a slot
     * @private
     */
    function isSlot(node) {
      return (node.localName === 'slot');
    }

    /**
     * Class that listens for changes (additions or removals) to
     * "flattened nodes" on a given `node`. The list of flattened nodes consists
     * of a node's children and, for any children that are `<slot>` elements,
     * the expanded flattened list of `assignedNodes`.
     * For example, if the observed node has children `<a></a><slot></slot><b></b>`
     * and the `<slot>` has one `<div>` assigned to it, then the flattened
     * nodes list is `<a></a><div></div><b></b>`. If the `<slot>` has other
     * `<slot>` elements assigned to it, these are flattened as well.
     *
     * The provided `callback` is called whenever any change to this list
     * of flattened nodes occurs, where an addition or removal of a node is
     * considered a change. The `callback` is called with one argument, an object
     * containing an array of any `addedNodes` and `removedNodes`.
     *
     * Note: the callback is called asynchronous to any changes
     * at a microtask checkpoint. This is because observation is performed using
     * `MutationObserver` and the `<slot>` element's `slotchange` event which
     * are asynchronous.
     *
     * An example:
     * ```js
     * class TestSelfObserve extends PolymerElement {
     *   static get is() { return 'test-self-observe';}
     *   connectedCallback() {
     *     super.connectedCallback();
     *     this._observer = new FlattenedNodesObserver(this, (info) => {
     *       this.info = info;
     *     });
     *   }
     *   disconnectedCallback() {
     *     super.disconnectedCallback();
     *     this._observer.disconnect();
     *   }
     * }
     * customElements.define(TestSelfObserve.is, TestSelfObserve);
     * ```
     *
     * @summary Class that listens for changes (additions or removals) to
     * "flattened nodes" on a given `node`.
     * @implements {PolymerDomApi.ObserveHandle}
     */
    let FlattenedNodesObserver = class {

      /**
       * Returns the list of flattened nodes for the given `node`.
       * This list consists of a node's children and, for any children
       * that are `<slot>` elements, the expanded flattened list of `assignedNodes`.
       * For example, if the observed node has children `<a></a><slot></slot><b></b>`
       * and the `<slot>` has one `<div>` assigned to it, then the flattened
       * nodes list is `<a></a><div></div><b></b>`. If the `<slot>` has other
       * `<slot>` elements assigned to it, these are flattened as well.
       *
       * @param {!HTMLElement|!HTMLSlotElement} node The node for which to
       *      return the list of flattened nodes.
       * @return {!Array<!Node>} The list of flattened nodes for the given `node`.
       * @nocollapse See https://github.com/google/closure-compiler/issues/2763
       */
      // eslint-disable-next-line
      static getFlattenedNodes(node) {
        const wrapped = wrap(node);
        if (isSlot(node)) {
          node = /** @type {!HTMLSlotElement} */(node); // eslint-disable-line no-self-assign
          return wrapped.assignedNodes({flatten: true});
        } else {
          return Array.from(wrapped.childNodes).map((node) => {
            if (isSlot(node)) {
              node = /** @type {!HTMLSlotElement} */(node); // eslint-disable-line no-self-assign
              return wrap(node).assignedNodes({flatten: true});
            } else {
              return [node];
            }
          }).reduce((a, b) => a.concat(b), []);
        }
      }

      /**
       * @param {!HTMLElement} target Node on which to listen for changes.
       * @param {?function(this: Element, { target: !HTMLElement, addedNodes: !Array<!Element>, removedNodes: !Array<!Element> }):void} callback Function called when there are additions
       * or removals from the target's list of flattened nodes.
       */
      // eslint-disable-next-line
      constructor(target, callback) {
        /**
         * @type {MutationObserver}
         * @private
         */
        this._shadyChildrenObserver = null;
        /**
         * @type {MutationObserver}
         * @private
         */
        this._nativeChildrenObserver = null;
        this._connected = false;
        /**
         * @type {!HTMLElement}
         * @private
         */
        this._target = target;
        this.callback = callback;
        this._effectiveNodes = [];
        this._observer = null;
        this._scheduled = false;
        /**
         * @type {function()}
         * @private
         */
        this._boundSchedule = () => {
          this._schedule();
        };
        this.connect();
        this._schedule();
      }

      /**
       * Activates an observer. This method is automatically called when
       * a `FlattenedNodesObserver` is created. It should only be called to
       * re-activate an observer that has been deactivated via the `disconnect` method.
       *
       * @return {void}
       */
      connect() {
        if (isSlot(this._target)) {
          this._listenSlots([this._target]);
        } else if (wrap(this._target).children) {
          this._listenSlots(
              /** @type {!NodeList<!Node>} */ (wrap(this._target).children));
          if (window.ShadyDOM) {
            this._shadyChildrenObserver =
              window.ShadyDOM.observeChildren(this._target, (mutations) => {
                this._processMutations(mutations);
              });
          } else {
            this._nativeChildrenObserver =
              new MutationObserver((mutations) => {
                this._processMutations(mutations);
              });
            this._nativeChildrenObserver.observe(this._target, {childList: true});
          }
        }
        this._connected = true;
      }

      /**
       * Deactivates the flattened nodes observer. After calling this method
       * the observer callback will not be called when changes to flattened nodes
       * occur. The `connect` method may be subsequently called to reactivate
       * the observer.
       *
       * @return {void}
       * @override
       */
      disconnect() {
        if (isSlot(this._target)) {
          this._unlistenSlots([this._target]);
        } else if (wrap(this._target).children) {
          this._unlistenSlots(
              /** @type {!NodeList<!Node>} */ (wrap(this._target).children));
          if (window.ShadyDOM && this._shadyChildrenObserver) {
            window.ShadyDOM.unobserveChildren(this._shadyChildrenObserver);
            this._shadyChildrenObserver = null;
          } else if (this._nativeChildrenObserver) {
            this._nativeChildrenObserver.disconnect();
            this._nativeChildrenObserver = null;
          }
        }
        this._connected = false;
      }

      /**
       * @return {void}
       * @private
       */
      _schedule() {
        if (!this._scheduled) {
          this._scheduled = true;
          microTask.run(() => this.flush());
        }
      }

      /**
       * @param {Array<MutationRecord>} mutations Mutations signaled by the mutation observer
       * @return {void}
       * @private
       */
      _processMutations(mutations) {
        this._processSlotMutations(mutations);
        this.flush();
      }

      /**
       * @param {Array<MutationRecord>} mutations Mutations signaled by the mutation observer
       * @return {void}
       * @private
       */
      _processSlotMutations(mutations) {
        if (mutations) {
          for (let i=0; i < mutations.length; i++) {
            let mutation = mutations[i];
            if (mutation.addedNodes) {
              this._listenSlots(mutation.addedNodes);
            }
            if (mutation.removedNodes) {
              this._unlistenSlots(mutation.removedNodes);
            }
          }
        }
      }

      /**
       * Flushes the observer causing any pending changes to be immediately
       * delivered the observer callback. By default these changes are delivered
       * asynchronously at the next microtask checkpoint.
       *
       * @return {boolean} Returns true if any pending changes caused the observer
       * callback to run.
       */
      flush() {
        if (!this._connected) {
          return false;
        }
        if (window.ShadyDOM) {
          ShadyDOM.flush();
        }
        if (this._nativeChildrenObserver) {
          this._processSlotMutations(this._nativeChildrenObserver.takeRecords());
        } else if (this._shadyChildrenObserver) {
          this._processSlotMutations(this._shadyChildrenObserver.takeRecords());
        }
        this._scheduled = false;
        let info = {
          target: this._target,
          addedNodes: [],
          removedNodes: []
        };
        let newNodes = this.constructor.getFlattenedNodes(this._target);
        let splices = calculateSplices(newNodes,
          this._effectiveNodes);
        // process removals
        for (let i=0, s; (i<splices.length) && (s=splices[i]); i++) {
          for (let j=0, n; (j < s.removed.length) && (n=s.removed[j]); j++) {
            info.removedNodes.push(n);
          }
        }
        // process adds
        for (let i=0, s; (i<splices.length) && (s=splices[i]); i++) {
          for (let j=s.index; j < s.index + s.addedCount; j++) {
            info.addedNodes.push(newNodes[j]);
          }
        }
        // update cache
        this._effectiveNodes = newNodes;
        let didFlush = false;
        if (info.addedNodes.length || info.removedNodes.length) {
          didFlush = true;
          this.callback.call(this._target, info);
        }
        return didFlush;
      }

      /**
       * @param {!Array<!Node>|!NodeList<!Node>} nodeList Nodes that could change
       * @return {void}
       * @private
       */
      _listenSlots(nodeList) {
        for (let i=0; i < nodeList.length; i++) {
          let n = nodeList[i];
          if (isSlot(n)) {
            n.addEventListener('slotchange', this._boundSchedule);
          }
        }
      }

      /**
       * @param {!Array<!Node>|!NodeList<!Node>} nodeList Nodes that could change
       * @return {void}
       * @private
       */
      _unlistenSlots(nodeList) {
        for (let i=0; i < nodeList.length; i++) {
          let n = nodeList[i];
          if (isSlot(n)) {
            n.removeEventListener('slotchange', this._boundSchedule);
          }
        }
      }

    };

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    /**
     * Forces several classes of asynchronously queued tasks to flush:
     * - Debouncers added via `enqueueDebouncer`
     * - ShadyDOM distribution
     *
     * @return {void}
     */
    const flush = function() {
      let shadyDOM, debouncers;
      do {
        shadyDOM = window.ShadyDOM && ShadyDOM.flush();
        if (window.ShadyCSS && window.ShadyCSS.ScopingShim) {
          window.ShadyCSS.ScopingShim.flush();
        }
        debouncers = flushDebouncers();
      } while (shadyDOM || debouncers);
    };

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */
    /* eslint-enable no-unused-vars */

    const p = Element.prototype;
    /**
     * @const {function(this:Node, string): boolean}
     */
    const normalizedMatchesSelector = p.matches || p.matchesSelector ||
      p.mozMatchesSelector || p.msMatchesSelector ||
      p.oMatchesSelector || p.webkitMatchesSelector;

    /**
     * Cross-platform `element.matches` shim.
     *
     * @function matchesSelector
     * @param {!Node} node Node to check selector against
     * @param {string} selector Selector to match
     * @return {boolean} True if node matched selector
     */
    const matchesSelector = function(node, selector) {
      return normalizedMatchesSelector.call(node, selector);
    };

    /**
     * Node API wrapper class returned from `Polymer.dom.(target)` when
     * `target` is a `Node`.
     * @implements {PolymerDomApi}
     * @unrestricted
     */
    class DomApiNative {

      /**
       * @param {!Node} node Node for which to create a Polymer.dom helper object.
       */
      constructor(node) {
        if (window['ShadyDOM'] && window['ShadyDOM']['inUse']) {
          window['ShadyDOM']['patch'](node);
        }
        this.node = node;
      }

      /**
       * Returns an instance of `FlattenedNodesObserver` that
       * listens for node changes on this element.
       *
       * @param {function(this:HTMLElement, { target: !HTMLElement, addedNodes: !Array<!Element>, removedNodes: !Array<!Element> }):void} callback Called when direct or distributed children
       *   of this element changes
       * @return {!PolymerDomApi.ObserveHandle} Observer instance
       * @override
       */
      observeNodes(callback) {
        return new FlattenedNodesObserver(
            /** @type {!HTMLElement} */(this.node), callback);
      }

      /**
       * Disconnects an observer previously created via `observeNodes`
       *
       * @param {!PolymerDomApi.ObserveHandle} observerHandle Observer instance
       *   to disconnect.
       * @return {void}
       * @override
       */
      unobserveNodes(observerHandle) {
        observerHandle.disconnect();
      }

      /**
       * Provided as a backwards-compatible API only.  This method does nothing.
       * @return {void}
       */
      notifyObserver() {}

      /**
       * Returns true if the provided node is contained with this element's
       * light-DOM children or shadow root, including any nested shadow roots
       * of children therein.
       *
       * @param {Node} node Node to test
       * @return {boolean} Returns true if the given `node` is contained within
       *   this element's light or shadow DOM.
       * @override
       */
      deepContains(node) {
        if (wrap(this.node).contains(node)) {
          return true;
        }
        let n = node;
        let doc = node.ownerDocument;
        // walk from node to `this` or `document`
        while (n && n !== doc && n !== this.node) {
          // use logical parentnode, or native ShadowRoot host
          n = wrap(n).parentNode || wrap(n).host;
        }
        return n === this.node;
      }

      /**
       * Returns the root node of this node.  Equivalent to `getRootNode()`.
       *
       * @return {!Node} Top most element in the dom tree in which the node
       * exists. If the node is connected to a document this is either a
       * shadowRoot or the document; otherwise, it may be the node
       * itself or a node or document fragment containing it.
       * @override
       */
      getOwnerRoot() {
        return wrap(this.node).getRootNode();
      }

      /**
       * For slot elements, returns the nodes assigned to the slot; otherwise
       * an empty array. It is equivalent to `<slot>.addignedNodes({flatten:true})`.
       *
       * @return {!Array<!Node>} Array of assigned nodes
       * @override
       */
      getDistributedNodes() {
        return (this.node.localName === 'slot') ?
          wrap(this.node).assignedNodes({flatten: true}) :
          [];
      }

      /**
       * Returns an array of all slots this element was distributed to.
       *
       * @return {!Array<!HTMLSlotElement>} Description
       * @override
       */
      getDestinationInsertionPoints() {
        let ip$ = [];
        let n = wrap(this.node).assignedSlot;
        while (n) {
          ip$.push(n);
          n = wrap(n).assignedSlot;
        }
        return ip$;
      }

      /**
       * Calls `importNode` on the `ownerDocument` for this node.
       *
       * @param {!Node} node Node to import
       * @param {boolean} deep True if the node should be cloned deeply during
       *   import
       * @return {Node} Clone of given node imported to this owner document
       */
      importNode(node, deep) {
        let doc = this.node instanceof Document ? this.node :
          this.node.ownerDocument;
        return wrap(doc).importNode(node, deep);
      }

      /**
       * @return {!Array<!Node>} Returns a flattened list of all child nodes and
       * nodes assigned to child slots.
       * @override
       */
      getEffectiveChildNodes() {
        return FlattenedNodesObserver.getFlattenedNodes(
            /** @type {!HTMLElement} */ (this.node));
      }

      /**
       * Returns a filtered list of flattened child elements for this element based
       * on the given selector.
       *
       * @param {string} selector Selector to filter nodes against
       * @return {!Array<!HTMLElement>} List of flattened child elements
       * @override
       */
      queryDistributedElements(selector) {
        let c$ = this.getEffectiveChildNodes();
        let list = [];
        for (let i=0, l=c$.length, c; (i<l) && (c=c$[i]); i++) {
          if ((c.nodeType === Node.ELEMENT_NODE) &&
              matchesSelector(c, selector)) {
            list.push(c);
          }
        }
        return list;
      }

      /**
       * For shadow roots, returns the currently focused element within this
       * shadow root.
       *
       * return {Node|undefined} Currently focused element
       * @override
       */
      get activeElement() {
        let node = this.node;
        return node._activeElement !== undefined ? node._activeElement : node.activeElement;
      }
    }

    function forwardMethods(proto, methods) {
      for (let i=0; i < methods.length; i++) {
        let method = methods[i];
        /* eslint-disable valid-jsdoc */
        proto[method] = /** @this {DomApiNative} */ function() {
          return this.node[method].apply(this.node, arguments);
        };
        /* eslint-enable */
      }
    }

    function forwardReadOnlyProperties(proto, properties) {
      for (let i=0; i < properties.length; i++) {
        let name = properties[i];
        Object.defineProperty(proto, name, {
          get: function() {
            const domApi = /** @type {DomApiNative} */(this);
            return domApi.node[name];
          },
          configurable: true
        });
      }
    }

    function forwardProperties(proto, properties) {
      for (let i=0; i < properties.length; i++) {
        let name = properties[i];
        Object.defineProperty(proto, name, {
          /**
           * @this {DomApiNative}
           * @return {*} .
           */
          get: function() {
            return this.node[name];
          },
          /**
           * @this {DomApiNative}
           * @param {*} value .
           */
          set: function(value) {
            this.node[name] = value;
          },
          configurable: true
        });
      }
    }


    /**
     * Event API wrapper class returned from `dom.(target)` when
     * `target` is an `Event`.
     */
    class EventApi {
      constructor(event) {
        this.event = event;
      }

      /**
       * Returns the first node on the `composedPath` of this event.
       *
       * @return {!EventTarget} The node this event was dispatched to
       */
      get rootTarget() {
        return this.path[0];
      }

      /**
       * Returns the local (re-targeted) target for this event.
       *
       * @return {!EventTarget} The local (re-targeted) target for this event.
       */
      get localTarget() {
        return this.event.target;
      }

      /**
       * Returns the `composedPath` for this event.
       * @return {!Array<!EventTarget>} The nodes this event propagated through
       */
      get path() {
        return this.event.composedPath();
      }
    }

    let DomApiImpl = DomApiNative;

    if (window['ShadyDOM'] && window['ShadyDOM']['inUse'] && window['ShadyDOM']['noPatch'] && window['ShadyDOM']['Wrapper']) {

      /**
       * @private
       * @extends {HTMLElement}
       */
      class Wrapper extends window['ShadyDOM']['Wrapper'] {}

      // copy bespoke API onto wrapper
      Object.getOwnPropertyNames(DomApiNative.prototype).forEach((prop) => {
        if (prop != 'activeElement') {
          Wrapper.prototype[prop] = DomApiNative.prototype[prop];
        }
      });

      // Note, `classList` is here only for legacy compatibility since it does not
      // trigger distribution in v1 Shadow DOM.
      forwardReadOnlyProperties(Wrapper.prototype, [
        'classList'
      ]);

      DomApiImpl = Wrapper;

      Object.defineProperties(EventApi.prototype, {

        // Returns the "lowest" node in the same root as the event's currentTarget.
        // When in `noPatch` mode, this must be calculated by walking the event's
        // path.
        localTarget: {
          get() {
            const current = this.event.currentTarget;
            const currentRoot = current && dom(current).getOwnerRoot();
            const p$ = this.path;
            for (let i = 0; i < p$.length; i++) {
              const e = p$[i];
              if (dom(e).getOwnerRoot() === currentRoot) {
                return e;
              }
            }
          },
          configurable: true
        },

        path: {
          get() {
            return window['ShadyDOM']['composedPath'](this.event);
          },
          configurable: true
        }
      });

    } else {

      // Methods that can provoke distribution or must return the logical, not
      // composed tree.
      forwardMethods(DomApiNative.prototype, [
        'cloneNode', 'appendChild', 'insertBefore', 'removeChild',
        'replaceChild', 'setAttribute', 'removeAttribute',
        'querySelector', 'querySelectorAll'
      ]);

      // Properties that should return the logical, not composed tree. Note, `classList`
      // is here only for legacy compatibility since it does not trigger distribution
      // in v1 Shadow DOM.
      forwardReadOnlyProperties(DomApiNative.prototype, [
        'parentNode', 'firstChild', 'lastChild',
        'nextSibling', 'previousSibling', 'firstElementChild',
        'lastElementChild', 'nextElementSibling', 'previousElementSibling',
        'childNodes', 'children', 'classList'
      ]);

      forwardProperties(DomApiNative.prototype, [
        'textContent', 'innerHTML', 'className'
      ]);
    }

    /**
     * Legacy DOM and Event manipulation API wrapper factory used to abstract
     * differences between native Shadow DOM and "Shady DOM" when polyfilling on
     * older browsers.
     *
     * Note that in Polymer 2.x use of `Polymer.dom` is no longer required and
     * in the majority of cases simply facades directly to the standard native
     * API.
     *
     * @summary Legacy DOM and Event manipulation API wrapper factory used to
     * abstract differences between native Shadow DOM and "Shady DOM."
     * @param {(Node|Event|DomApiNative|EventApi)=} obj Node or event to operate on
     * @return {!DomApiNative|!EventApi} Wrapper providing either node API or event API
     */
    const dom = function(obj) {
      obj = obj || document;
      if (obj instanceof DomApiImpl) {
        return /** @type {!DomApi} */(obj);
      }
      if (obj instanceof EventApi) {
        return /** @type {!EventApi} */(obj);
      }
      let helper = obj['__domApi'];
      if (!helper) {
        if (obj instanceof Event) {
          helper = new EventApi(obj);
        } else {
          helper = new DomApiImpl(/** @type {Node} */(obj));
        }
        obj['__domApi'] = helper;
      }
      return helper;
    };

    /**
    @license
    Copyright (c) 2019 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    const ShadyDOM$1 = window.ShadyDOM;
    const ShadyCSS = window.ShadyCSS;

    /**
     * Return true if node scope is correct.
     *
     * @param {!Element} node Node to check scope
     * @param {!Node} scope Scope reference
     * @return {boolean} True if node is in scope
     */
    function sameScope(node, scope) {
      return wrap(node).getRootNode() === scope;
    }

    /**
     * Ensure that elements in a ShadowDOM container are scoped correctly.
     * This function is only needed when ShadyDOM is used and unpatched DOM APIs are used in third party code.
     * This can happen in noPatch mode or when specialized APIs like ranges or tables are used to mutate DOM.
     *
     * @param  {!Element} container Container element to scope
     * @param  {boolean=} shouldObserve if true, start a mutation observer for added nodes to the container
     * @return {?MutationObserver} Returns a new MutationObserver on `container` if `shouldObserve` is true.
     */
    function scopeSubtree(container, shouldObserve = false) {
      // If using native ShadowDOM, abort
      if (!ShadyDOM$1 || !ShadyCSS) {
        return null;
      }
      // ShadyCSS handles DOM mutations when ShadyDOM does not handle scoping itself
      if (!ShadyDOM$1['handlesDynamicScoping']) {
        return null;
      }
      const ScopingShim = ShadyCSS['ScopingShim'];
      // if ScopingShim is not available, abort
      if (!ScopingShim) {
        return null;
      }
      // capture correct scope for container
      const containerScope = ScopingShim['scopeForNode'](container);
      const root = wrap(container).getRootNode();

      const scopify = (node) => {
        if (!sameScope(node, root)) {
          return;
        }
        // NOTE: native qSA does not honor scoped DOM, but it is faster, and the same behavior as Polymer v1
        const elements = Array.from(ShadyDOM$1['nativeMethods']['querySelectorAll'].call(node, '*'));
        elements.push(node);
        for (let i = 0; i < elements.length; i++) {
          const el = elements[i];
          if (!sameScope(el, root)) {
            continue;
          }
          const currentScope = ScopingShim['currentScopeForNode'](el);
          if (currentScope !== containerScope) {
            if (currentScope !== '') {
              ScopingShim['unscopeNode'](el, currentScope);
            }
            ScopingShim['scopeNode'](el, containerScope);
          }
        }
      };

      // scope everything in container
      scopify(container);

      if (shouldObserve) {
        const mo = new MutationObserver((mxns) => {
          for (let i = 0; i < mxns.length; i++) {
            const mxn = mxns[i];
            for (let j = 0; j < mxn.addedNodes.length; j++) {
              const addedNode = mxn.addedNodes[j];
              if (addedNode.nodeType === Node.ELEMENT_NODE) {
                scopify(addedNode);
              }
            }
          }
        });
        mo.observe(container, {childList: true, subtree: true});
        return mo;
      } else {
        return null;
      }
    }

    /**
     * @fileoverview
     * @suppress {checkPrototypalTypes}
     * @license Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
     * This code may only be used under the BSD style license found at
     * http://polymer.github.io/LICENSE.txt The complete set of authors may be found
     * at http://polymer.github.io/AUTHORS.txt The complete set of contributors may
     * be found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by
     * Google as part of the polymer project is also subject to an additional IP
     * rights grant found at http://polymer.github.io/PATENTS.txt
     */

    const DISABLED_ATTR = 'disable-upgrade';

    const findObservedAttributesGetter = (ctor) => {
      while (ctor) {
        const desc = Object.getOwnPropertyDescriptor(ctor, 'observedAttributes');
        if (desc) {
          return desc.get;
        }
        ctor = Object.getPrototypeOf(ctor.prototype).constructor;
      }
      return () => [];
    };

    /**
     * Element class mixin that allows the element to boot up in a non-enabled
     * state when the `disable-upgrade` attribute is present. This mixin is
     * designed to be used with element classes like PolymerElement that perform
     * initial startup work when they are first connected. When the
     * `disable-upgrade` attribute is removed, if the element is connected, it
     * boots up and "enables" as it otherwise would; if it is not connected, the
     * element boots up when it is next connected.
     *
     * Using `disable-upgrade` with PolymerElement prevents any data propagation
     * to the element, any element DOM from stamping, or any work done in
     * connected/disconnctedCallback from occuring, but it does not prevent work
     * done in the element constructor.
     *
     * Note, this mixin must be applied on top of any element class that
     * itself implements a `connectedCallback` so that it can control the work
     * done in `connectedCallback`. For example,
     *
     *     MyClass = DisableUpgradeMixin(class extends BaseClass {...});
     *
     * @mixinFunction
     * @polymer
     * @appliesMixin ElementMixin
     * @template T
     * @param {function(new:T)} superClass Class to apply mixin to.
     * @return {function(new:T)} superClass with mixin applied.
     */
    const DisableUpgradeMixin = dedupingMixin((base) => {
      /**
       * @constructor
       * @implements {Polymer_ElementMixin}
       * @extends {HTMLElement}
       * @private
       */
      const superClass = ElementMixin(base);

      // Work around for closure bug #126934458. Using `super` in a property
      // getter does not work so instead we search the Base prototype for an
      // implementation of observedAttributes so that we can override and call
      // the `super` getter. Note, this is done one time ever because we assume
      // that `Base` is always comes from `Polymer.LegacyElementMixn`.
      let observedAttributesGetter = findObservedAttributesGetter(superClass);

      /**
       * @polymer
       * @mixinClass
       * @implements {Polymer_DisableUpgradeMixin}
       */
      class DisableUpgradeClass extends superClass {

        constructor() {
          super();
          /** @type {boolean|undefined} */
          this.__isUpgradeDisabled;
        }

        static get observedAttributes() {
          return observedAttributesGetter.call(this).concat(DISABLED_ATTR);
        }

        // Prevent element from initializing properties when it's upgrade disabled.
        /** @override */
        _initializeProperties() {
          if (this.hasAttribute(DISABLED_ATTR)) {
            this.__isUpgradeDisabled = true;
          } else {
            super._initializeProperties();
          }
        }

        // Prevent element from enabling properties when it's upgrade disabled.
        // Normally overriding connectedCallback would be enough, but dom-* elements
        /** @override */
        _enableProperties() {
          if (!this.__isUpgradeDisabled) {
            super._enableProperties();
          }
        }

        // If the element starts upgrade-disabled and a property is set for
        // which an accessor exists, the default should not be applied.
        // This additional check is needed because defaults are applied via
        // `_initializeProperties` which is called after initial properties
        // have been set when the element starts upgrade-disabled.
        /** @override */
        _canApplyPropertyDefault(property) {
          return super._canApplyPropertyDefault(property) &&
            !(this.__isUpgradeDisabled && this._isPropertyPending(property));
        }

        /**
         * @override
         * @param {string} name Attribute name.
         * @param {?string} old The previous value for the attribute.
         * @param {?string} value The new value for the attribute.
         * @param {?string} namespace The XML namespace for the attribute.
         * @return {void}
         */
        attributeChangedCallback(name, old, value, namespace) {
          if (name == DISABLED_ATTR) {
            // When disable-upgrade is removed, intialize properties and
            // provoke connectedCallback if the element is already connected.
            if (this.__isUpgradeDisabled && value == null) {
              super._initializeProperties();
              this.__isUpgradeDisabled = false;
              if (wrap(this).isConnected) {
                super.connectedCallback();
              }
            }
          } else {
            super.attributeChangedCallback(
                name, old, value, /** @type {null|string} */ (namespace));
          }
        }

        // Prevent element from connecting when it's upgrade disabled.
        // This prevents user code in `attached` from being called.
        /** @override */
        connectedCallback() {
          if (!this.__isUpgradeDisabled) {
            super.connectedCallback();
          }
        }

        // Prevent element from disconnecting when it's upgrade disabled.
        // This avoids allowing user code `detached` from being called without a
        // paired call to `attached`.
        /** @override */
        disconnectedCallback() {
          if (!this.__isUpgradeDisabled) {
            super.disconnectedCallback();
          }
        }

      }

      return DisableUpgradeClass;
    });

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    const DISABLED_ATTR$1 = 'disable-upgrade';

    let styleInterface = window.ShadyCSS;

    /**
     * Element class mixin that provides Polymer's "legacy" API intended to be
     * backward-compatible to the greatest extent possible with the API
     * found on the Polymer 1.x `Polymer.Base` prototype applied to all elements
     * defined using the `Polymer({...})` function.
     *
     * @mixinFunction
     * @polymer
     * @appliesMixin ElementMixin
     * @appliesMixin GestureEventListeners
     * @appliesMixin DirMixin
     * @property isAttached {boolean} Set to `true` in this element's
     *   `connectedCallback` and `false` in `disconnectedCallback`
     * @summary Element class mixin that provides Polymer's "legacy" API
     */
    const LegacyElementMixin = dedupingMixin((base) => {

      // TODO(kschaaf): Note, the `@implements {Polymer_DirMixin}` is required here
      // (rather than on legacyElementBase) for unknown reasons.
      /**
       * @constructor
       * @implements {Polymer_ElementMixin}
       * @implements {Polymer_GestureEventListeners}
       * @implements {Polymer_DirMixin}
       * @extends {HTMLElement}
       * @private
       */
      const GesturesElement = GestureEventListeners(ElementMixin(base));

      // Note, the DirMixin does nothing if css is built so avoid including it
      // in that case.

      /**
       * @constructor
       * @extends {GesturesElement}
       * @private
       */
      const legacyElementBase = builtCSS ? GesturesElement :
        DirMixin(GesturesElement);

      const observedAttributesGetter = findObservedAttributesGetter(legacyElementBase);

      /**
       * Map of simple names to touch action names
       * @dict
       */
      const DIRECTION_MAP = {
        'x': 'pan-x',
        'y': 'pan-y',
        'none': 'none',
        'all': 'auto'
      };

      /**
       * @polymer
       * @mixinClass
       * @extends {legacyElementBase}
       * @implements {Polymer_LegacyElementMixin}
       * @unrestricted
       */
      class LegacyElement extends legacyElementBase {

        constructor() {
          super();
          /** @type {boolean} */
          this.isAttached;
          /** @type {?WeakMap<!Element, !Object<string, !Function>>} */
          this.__boundListeners;
          /** @type {?Object<string, ?Function>} */
          this._debouncers;
          // NOTE: Inlined for perf from version of DisableUpgradeMixin.
          /** @type {boolean|undefined} */
          this.__isUpgradeDisabled;
          /** @type {boolean|undefined} */
          this.__needsAttributesAtConnected;
          /** @type {boolean|undefined} */
          this._legacyForceObservedAttributes;
        }

        /**
         * Forwards `importMeta` from the prototype (i.e. from the info object
         * passed to `Polymer({...})`) to the static API.
         *
         * @return {!Object} The `import.meta` object set on the prototype
         * @suppress {missingProperties} `this` is always in the instance in
         *  closure for some reason even in a static method, rather than the class
         * @nocollapse
         */
        static get importMeta() {
          return this.prototype.importMeta;
        }

        /**
         * Legacy callback called during the `constructor`, for overriding
         * by the user.
         * @override
         * @return {void}
         */
        created() {}

        /**
         * Processes an attribute reaction when the `legacyNoObservedAttributes`
         * setting is in use.
         * @param {string} name Name of attribute that changed
         * @param {?string} old Old attribute value
         * @param {?string} value New attribute value
         * @return {void}
         */
        __attributeReaction(name, old, value) {
          if ((this.__dataAttributes && this.__dataAttributes[name]) || name === DISABLED_ATTR$1) {
            this.attributeChangedCallback(name, old, value, null);
          }
        }

        /**
         * Sets the value of an attribute.
         * @override
         * @param {string} name The name of the attribute to change.
         * @param {string|number|boolean|!TrustedHTML|!TrustedScriptURL|!TrustedURL} value The new attribute value.
         */
        setAttribute(name, value) {
          if (legacyNoObservedAttributes && !this._legacyForceObservedAttributes) {
            const oldValue = this.getAttribute(name);
            super.setAttribute(name, value);
            // value coerced to String for closure's benefit
            this.__attributeReaction(name, oldValue, String(value));
          } else {
            super.setAttribute(name, value);
          }
        }

        /**
         * Removes an attribute.
         * @override
         * @param {string} name The name of the attribute to remove.
         */
        removeAttribute(name) {
          if (legacyNoObservedAttributes && !this._legacyForceObservedAttributes) {
            const oldValue = this.getAttribute(name);
            super.removeAttribute(name);
            this.__attributeReaction(name, oldValue, null);
          } else {
            super.removeAttribute(name);
          }
        }

        // NOTE: Inlined for perf from version of DisableUpgradeMixin.
        static get observedAttributes() {
          if (legacyNoObservedAttributes && !this.prototype._legacyForceObservedAttributes) {
            // Ensure this element is property registered with the telemetry system.
            if (!this.hasOwnProperty(JSCompiler_renameProperty('__observedAttributes', this))) {
              this.__observedAttributes = [];
              register(this.prototype);
            }
            return this.__observedAttributes;
          } else {
            return observedAttributesGetter.call(this).concat(DISABLED_ATTR$1);
          }
        }

        // NOTE: Inlined for perf from version of DisableUpgradeMixin.
        // Prevent element from enabling properties when it's upgrade disabled.
        // Normally overriding connectedCallback would be enough, but dom-* elements
        /** @override */
        _enableProperties() {
          if (!this.__isUpgradeDisabled) {
            super._enableProperties();
          }
        }

        // NOTE: Inlined for perf from version of DisableUpgradeMixin.
        // If the element starts upgrade-disabled and a property is set for
        // which an accessor exists, the default should not be applied.
        // This additional check is needed because defaults are applied via
        // `_initializeProperties` which is called after initial properties
        // have been set when the element starts upgrade-disabled.
        /** @override */
        _canApplyPropertyDefault(property) {
          return super._canApplyPropertyDefault(property) &&
            !(this.__isUpgradeDisabled && this._isPropertyPending(property));
        }

        /**
         * Provides an implementation of `connectedCallback`
         * which adds Polymer legacy API's `attached` method.
         * @return {void}
         * @override
         */
        connectedCallback() {
          if (this.__needsAttributesAtConnected) {
            this._takeAttributes();
          }
          // NOTE: Inlined for perf from version of DisableUpgradeMixin.
          if (!this.__isUpgradeDisabled) {
            super.connectedCallback();
            this.isAttached = true;
            this.attached();
          }
        }

        /**
         * Legacy callback called during `connectedCallback`, for overriding
         * by the user.
         * @override
         * @return {void}
         */
        attached() {}

        /**
         * Provides an implementation of `disconnectedCallback`
         * which adds Polymer legacy API's `detached` method.
         * @return {void}
         * @override
         */
        disconnectedCallback() {
          // NOTE: Inlined for perf from version of DisableUpgradeMixin.
          if (!this.__isUpgradeDisabled) {
            super.disconnectedCallback();
            this.isAttached = false;
            this.detached();
          }
        }

        /**
         * Legacy callback called during `disconnectedCallback`, for overriding
         * by the user.
         * @override
         * @return {void}
         */
        detached() {}

        /**
         * Provides an override implementation of `attributeChangedCallback`
         * which adds the Polymer legacy API's `attributeChanged` method.
         * @param {string} name Name of attribute.
         * @param {?string} old Old value of attribute.
         * @param {?string} value Current value of attribute.
         * @param {?string} namespace Attribute namespace.
         * @return {void}
         * @override
         */
        attributeChangedCallback(name, old, value, namespace) {
          if (old !== value) {
            // NOTE: Inlined for perf from version of DisableUpgradeMixin.
            if (name == DISABLED_ATTR$1) {
              // When disable-upgrade is removed, intialize properties and
              // provoke connectedCallback if the element is already connected.
              if (this.__isUpgradeDisabled && value == null) {
                this._initializeProperties();
                this.__isUpgradeDisabled = false;
                if (wrap(this).isConnected) {
                  this.connectedCallback();
                }
              }
            } else {
              super.attributeChangedCallback(name, old, value, namespace);
              this.attributeChanged(name, old, value);
            }
          }
        }

        /**
         * Legacy callback called during `attributeChangedChallback`, for overriding
         * by the user.
         * @param {string} name Name of attribute.
         * @param {?string} old Old value of attribute.
         * @param {?string} value Current value of attribute.
         * @return {void}
         * @override
         */
        attributeChanged(name, old, value) {} // eslint-disable-line no-unused-vars

        /**
         * Overrides the default `Polymer.PropertyEffects` implementation to
         * add support for class initialization via the `_registered` callback.
         * This is called only when the first instance of the element is created.
         *
         * @return {void}
         * @override
         * @suppress {invalidCasts}
         */
        _initializeProperties() {
          // NOTE: Inlined for perf from version of DisableUpgradeMixin.
          // Only auto-use disable-upgrade if legacyOptimizations is set.
          if (legacyOptimizations && this.hasAttribute(DISABLED_ATTR$1)) {
            this.__isUpgradeDisabled = true;
          } else {
            let proto = Object.getPrototypeOf(this);
            if (!proto.hasOwnProperty(JSCompiler_renameProperty('__hasRegisterFinished', proto))) {
              this._registered();
              // backstop in case the `_registered` implementation does not set this
              proto.__hasRegisterFinished = true;
            }
            super._initializeProperties();
            this.root = /** @type {HTMLElement} */(this);
            this.created();
            // Pull all attribute values 1x if `legacyNoObservedAttributes` is set.
            if (legacyNoObservedAttributes && !this._legacyForceObservedAttributes) {
              if (this.hasAttributes()) {
                this._takeAttributes();
              // Element created from scratch or parser generated
              } else if (!this.parentNode) {
                this.__needsAttributesAtConnected = true;
              }
            }
            // Ensure listeners are applied immediately so that they are
            // added before declarative event listeners. This allows an element to
            // decorate itself via an event prior to any declarative listeners
            // seeing the event. Note, this ensures compatibility with 1.x ordering.
            this._applyListeners();
          }
        }

        _takeAttributes() {
          const a = this.attributes;
          for (let i=0, l=a.length; i < l; i++) {
            const attr = a[i];
            this.__attributeReaction(attr.name, null, attr.value);
          }
        }

        /**
         * Called automatically when an element is initializing.
         * Users may override this method to perform class registration time
         * work. The implementation should ensure the work is performed
         * only once for the class.
         * @protected
         * @return {void}
         * @override
         */
        _registered() {}

        /**
         * Overrides the default `Polymer.PropertyEffects` implementation to
         * add support for installing `hostAttributes` and `listeners`.
         *
         * @return {void}
         * @override
         */
        ready() {
          this._ensureAttributes();
          super.ready();
        }

        /**
         * Ensures an element has required attributes. Called when the element
         * is being readied via `ready`. Users should override to set the
         * element's required attributes. The implementation should be sure
         * to check and not override existing attributes added by
         * the user of the element. Typically, setting attributes should be left
         * to the element user and not done here; reasonable exceptions include
         * setting aria roles and focusability.
         * @protected
         * @return {void}
         * @override
         */
        _ensureAttributes() {}

        /**
         * Adds element event listeners. Called when the element
         * is being readied via `ready`. Users should override to
         * add any required element event listeners.
         * In performance critical elements, the work done here should be kept
         * to a minimum since it is done before the element is rendered. In
         * these elements, consider adding listeners asynchronously so as not to
         * block render.
         * @protected
         * @return {void}
         * @override
         */
        _applyListeners() {}

        /**
         * Converts a typed JavaScript value to a string.
         *
         * Note this method is provided as backward-compatible legacy API
         * only.  It is not directly called by any Polymer features. To customize
         * how properties are serialized to attributes for attribute bindings and
         * `reflectToAttribute: true` properties as well as this method, override
         * the `_serializeValue` method provided by `Polymer.PropertyAccessors`.
         *
         * @param {*} value Value to deserialize
         * @return {string | undefined} Serialized value
         * @override
         */
        serialize(value) {
          return this._serializeValue(value);
        }

        /**
         * Converts a string to a typed JavaScript value.
         *
         * Note this method is provided as backward-compatible legacy API
         * only.  It is not directly called by any Polymer features.  To customize
         * how attributes are deserialized to properties for in
         * `attributeChangedCallback`, override `_deserializeValue` method
         * provided by `Polymer.PropertyAccessors`.
         *
         * @param {string} value String to deserialize
         * @param {*} type Type to deserialize the string to
         * @return {*} Returns the deserialized value in the `type` given.
         * @override
         */
        deserialize(value, type) {
          return this._deserializeValue(value, type);
        }

        /**
         * Serializes a property to its associated attribute.
         *
         * Note this method is provided as backward-compatible legacy API
         * only.  It is not directly called by any Polymer features.
         *
         * @param {string} property Property name to reflect.
         * @param {string=} attribute Attribute name to reflect.
         * @param {*=} value Property value to reflect.
         * @return {void}
         * @override
         */
        reflectPropertyToAttribute(property, attribute, value) {
          this._propertyToAttribute(property, attribute, value);
        }

        /**
         * Sets a typed value to an HTML attribute on a node.
         *
         * Note this method is provided as backward-compatible legacy API
         * only.  It is not directly called by any Polymer features.
         *
         * @param {*} value Value to serialize.
         * @param {string} attribute Attribute name to serialize to.
         * @param {Element} node Element to set attribute to.
         * @return {void}
         * @override
         */
        serializeValueToAttribute(value, attribute, node) {
          this._valueToNodeAttribute(/** @type {Element} */ (node || this), value, attribute);
        }

        /**
         * Copies own properties (including accessor descriptors) from a source
         * object to a target object.
         *
         * @param {Object} prototype Target object to copy properties to.
         * @param {Object} api Source object to copy properties from.
         * @return {Object} prototype object that was passed as first argument.
         * @override
         */
        extend(prototype, api) {
          if (!(prototype && api)) {
            return prototype || api;
          }
          let n$ = Object.getOwnPropertyNames(api);
          for (let i=0, n; (i<n$.length) && (n=n$[i]); i++) {
            let pd = Object.getOwnPropertyDescriptor(api, n);
            if (pd) {
              Object.defineProperty(prototype, n, pd);
            }
          }
          return prototype;
        }

        /**
         * Copies props from a source object to a target object.
         *
         * Note, this method uses a simple `for...in` strategy for enumerating
         * properties.  To ensure only `ownProperties` are copied from source
         * to target and that accessor implementations are copied, use `extend`.
         *
         * @param {!Object} target Target object to copy properties to.
         * @param {!Object} source Source object to copy properties from.
         * @return {!Object} Target object that was passed as first argument.
         * @override
         */
        mixin(target, source) {
          for (let i in source) {
            target[i] = source[i];
          }
          return target;
        }

        /**
         * Sets the prototype of an object.
         *
         * Note this method is provided as backward-compatible legacy API
         * only.  It is not directly called by any Polymer features.
         * @param {Object} object The object on which to set the prototype.
         * @param {Object} prototype The prototype that will be set on the given
         * `object`.
         * @return {Object} Returns the given `object` with its prototype set
         * to the given `prototype` object.
         * @override
         */
        chainObject(object, prototype) {
          if (object && prototype && object !== prototype) {
            object.__proto__ = prototype;
          }
          return object;
        }

        /* **** Begin Template **** */

        /**
         * Calls `importNode` on the `content` of the `template` specified and
         * returns a document fragment containing the imported content.
         *
         * @param {HTMLTemplateElement} template HTML template element to instance.
         * @return {!DocumentFragment} Document fragment containing the imported
         *   template content.
         * @override
         * @suppress {missingProperties} go/missingfnprops
         */
        instanceTemplate(template) {
          let content = this.constructor._contentForTemplate(template);
          let dom = /** @type {!DocumentFragment} */
            (document.importNode(content, true));
          return dom;
        }

        /* **** Begin Events **** */



        /**
         * Dispatches a custom event with an optional detail value.
         *
         * @param {string} type Name of event type.
         * @param {*=} detail Detail value containing event-specific
         *   payload.
         * @param {{ bubbles: (boolean|undefined), cancelable: (boolean|undefined),
         *     composed: (boolean|undefined) }=}
         *  options Object specifying options.  These may include:
         *  `bubbles` (boolean, defaults to `true`),
         *  `cancelable` (boolean, defaults to false), and
         *  `node` on which to fire the event (HTMLElement, defaults to `this`).
         * @return {!Event} The new event that was fired.
         * @override
         */
        fire(type, detail, options) {
          options = options || {};
          detail = (detail === null || detail === undefined) ? {} : detail;
          let event = new Event(type, {
            bubbles: options.bubbles === undefined ? true : options.bubbles,
            cancelable: Boolean(options.cancelable),
            composed: options.composed === undefined ? true: options.composed
          });
          event.detail = detail;
          let node = options.node || this;
          wrap(node).dispatchEvent(event);
          return event;
        }

        /**
         * Convenience method to add an event listener on a given element,
         * late bound to a named method on this element.
         *
         * @param {?EventTarget} node Element to add event listener to.
         * @param {string} eventName Name of event to listen for.
         * @param {string} methodName Name of handler method on `this` to call.
         * @return {void}
         * @override
         */
        listen(node, eventName, methodName) {
          node = /** @type {!EventTarget} */ (node || this);
          let hbl = this.__boundListeners ||
            (this.__boundListeners = new WeakMap());
          let bl = hbl.get(node);
          if (!bl) {
            bl = {};
            hbl.set(node, bl);
          }
          let key = eventName + methodName;
          if (!bl[key]) {
            bl[key] = this._addMethodEventListenerToNode(
                /** @type {!Node} */ (node), eventName, methodName, this);
          }
        }

        /**
         * Convenience method to remove an event listener from a given element,
         * late bound to a named method on this element.
         *
         * @param {?EventTarget} node Element to remove event listener from.
         * @param {string} eventName Name of event to stop listening to.
         * @param {string} methodName Name of handler method on `this` to not call
         anymore.
         * @return {void}
         * @override
         */
        unlisten(node, eventName, methodName) {
          node = /** @type {!EventTarget} */ (node || this);
          let bl = this.__boundListeners &&
              this.__boundListeners.get(/** @type {!Element} */ (node));
          let key = eventName + methodName;
          let handler = bl && bl[key];
          if (handler) {
            this._removeEventListenerFromNode(
                /** @type {!Node} */ (node), eventName, handler);
            bl[key] = /** @type {?} */ (null);
          }
        }

        /**
         * Override scrolling behavior to all direction, one direction, or none.
         *
         * Valid scroll directions:
         *   - 'all': scroll in any direction
         *   - 'x': scroll only in the 'x' direction
         *   - 'y': scroll only in the 'y' direction
         *   - 'none': disable scrolling for this node
         *
         * @param {string=} direction Direction to allow scrolling
         * Defaults to `all`.
         * @param {Element=} node Element to apply scroll direction setting.
         * Defaults to `this`.
         * @return {void}
         * @override
         */
        setScrollDirection(direction, node) {
          setTouchAction(
              /** @type {!Element} */ (node || this),
              DIRECTION_MAP[direction] || 'auto');
        }
        /* **** End Events **** */

        /**
         * Convenience method to run `querySelector` on this local DOM scope.
         *
         * This function calls `Polymer.dom(this.root).querySelector(slctr)`.
         *
         * @param {string} slctr Selector to run on this local DOM scope
         * @return {Element} Element found by the selector, or null if not found.
         * @override
         */
        $$(slctr) {
          // Note, no need to `wrap` this because root is always patched
          return this.root.querySelector(slctr);
        }

        /**
         * Return the element whose local dom within which this element
         * is contained. This is a shorthand for
         * `this.getRootNode().host`.
         * @this {Element}
         * @return {?Node} The element whose local dom within which this element is
         * contained.
         * @override
         */
        get domHost() {
          let root = wrap(this).getRootNode();
          return (root instanceof DocumentFragment) ? /** @type {ShadowRoot} */ (root).host : root;
        }

        /**
         * Force this element to distribute its children to its local dom.
         * This should not be necessary as of Polymer 2.0.2 and is provided only
         * for backwards compatibility.
         * @return {void}
         * @override
         */
        distributeContent() {
          const thisEl = /** @type {Element} */ (this);
          const domApi = /** @type {PolymerDomApi} */(dom(thisEl));
          if (window.ShadyDOM && domApi.shadowRoot) {
            ShadyDOM.flush();
          }
        }

        /**
         * Returns a list of nodes that are the effective childNodes. The effective
         * childNodes list is the same as the element's childNodes except that
         * any `<content>` elements are replaced with the list of nodes distributed
         * to the `<content>`, the result of its `getDistributedNodes` method.
         * @return {!Array<!Node>} List of effective child nodes.
         * @suppress {invalidCasts} LegacyElementMixin must be applied to an
         *     HTMLElement
         * @override
         */
        getEffectiveChildNodes() {
          const thisEl = /** @type {Element} */ (this);
          const domApi = /** @type {PolymerDomApi} */ (dom(thisEl));
          return domApi.getEffectiveChildNodes();
        }

        /**
         * Returns a list of nodes distributed within this element that match
         * `selector`. These can be dom children or elements distributed to
         * children that are insertion points.
         * @param {string} selector Selector to run.
         * @return {!Array<!Node>} List of distributed elements that match selector.
         * @suppress {invalidCasts} LegacyElementMixin must be applied to an
         * HTMLElement
         * @override
         */
        queryDistributedElements(selector) {
          const thisEl = /** @type {Element} */ (this);
          const domApi = /** @type {PolymerDomApi} */ (dom(thisEl));
          return domApi.queryDistributedElements(selector);
        }

        /**
         * Returns a list of elements that are the effective children. The effective
         * children list is the same as the element's children except that
         * any `<content>` elements are replaced with the list of elements
         * distributed to the `<content>`.
         *
         * @return {!Array<!Node>} List of effective children.
         * @override
         */
        getEffectiveChildren() {
          let list = this.getEffectiveChildNodes();
          return list.filter(function(/** @type {!Node} */ n) {
            return (n.nodeType === Node.ELEMENT_NODE);
          });
        }

        /**
         * Returns a string of text content that is the concatenation of the
         * text content's of the element's effective childNodes (the elements
         * returned by <a href="#getEffectiveChildNodes>getEffectiveChildNodes</a>.
         *
         * @return {string} List of effective children.
         * @override
         */
        getEffectiveTextContent() {
          let cn = this.getEffectiveChildNodes();
          let tc = [];
          for (let i=0, c; (c = cn[i]); i++) {
            if (c.nodeType !== Node.COMMENT_NODE) {
              tc.push(c.textContent);
            }
          }
          return tc.join('');
        }

        /**
         * Returns the first effective childNode within this element that
         * match `selector`. These can be dom child nodes or elements distributed
         * to children that are insertion points.
         * @param {string} selector Selector to run.
         * @return {Node} First effective child node that matches selector.
         * @override
         */
        queryEffectiveChildren(selector) {
          let e$ = this.queryDistributedElements(selector);
          return e$ && e$[0];
        }

        /**
         * Returns a list of effective childNodes within this element that
         * match `selector`. These can be dom child nodes or elements distributed
         * to children that are insertion points.
         * @param {string} selector Selector to run.
         * @return {!Array<!Node>} List of effective child nodes that match
         *     selector.
         * @override
         */
        queryAllEffectiveChildren(selector) {
          return this.queryDistributedElements(selector);
        }

        /**
         * Returns a list of nodes distributed to this element's `<slot>`.
         *
         * If this element contains more than one `<slot>` in its local DOM,
         * an optional selector may be passed to choose the desired content.
         *
         * @param {string=} slctr CSS selector to choose the desired
         *   `<slot>`.  Defaults to `content`.
         * @return {!Array<!Node>} List of distributed nodes for the `<slot>`.
         * @override
         */
        getContentChildNodes(slctr) {
          // Note, no need to `wrap` this because root is always
          let content = this.root.querySelector(slctr || 'slot');
          return content ?
              /** @type {PolymerDomApi} */ (dom(content)).getDistributedNodes() :
              [];
        }

        /**
         * Returns a list of element children distributed to this element's
         * `<slot>`.
         *
         * If this element contains more than one `<slot>` in its
         * local DOM, an optional selector may be passed to choose the desired
         * content.  This method differs from `getContentChildNodes` in that only
         * elements are returned.
         *
         * @param {string=} slctr CSS selector to choose the desired
         *   `<content>`.  Defaults to `content`.
         * @return {!Array<!HTMLElement>} List of distributed nodes for the
         *   `<slot>`.
         * @suppress {invalidCasts}
         * @override
         */
        getContentChildren(slctr) {
          let children = /** @type {!Array<!HTMLElement>} */(this.getContentChildNodes(slctr).filter(function(n) {
            return (n.nodeType === Node.ELEMENT_NODE);
          }));
          return children;
        }

        /**
         * Checks whether an element is in this element's light DOM tree.
         *
         * @param {?Node} node The element to be checked.
         * @return {boolean} true if node is in this element's light DOM tree.
         * @suppress {invalidCasts} LegacyElementMixin must be applied to an
         * HTMLElement
         * @override
         */
        isLightDescendant(node) {
          const thisNode = /** @type {Node} */ (this);
          return thisNode !== node && wrap(thisNode).contains(node) &&
            wrap(thisNode).getRootNode() === wrap(node).getRootNode();
        }

        /**
         * Checks whether an element is in this element's local DOM tree.
         *
         * @param {!Element} node The element to be checked.
         * @return {boolean} true if node is in this element's local DOM tree.
         * @override
         */
        isLocalDescendant(node) {
          return this.root === wrap(node).getRootNode();
        }

        /**
         * No-op for backwards compatibility. This should now be handled by
         * ShadyCss library.
         * @param  {!Element} container Container element to scope
         * @param  {boolean=} shouldObserve if true, start a mutation observer for added nodes to the container
         * @return {?MutationObserver} Returns a new MutationObserver on `container` if `shouldObserve` is true.
         * @override
         */
        scopeSubtree(container, shouldObserve = false) {
          return scopeSubtree(container, shouldObserve);
        }

        /**
         * Returns the computed style value for the given property.
         * @param {string} property The css property name.
         * @return {string} Returns the computed css property value for the given
         * `property`.
         * @suppress {invalidCasts} LegacyElementMixin must be applied to an
         *     HTMLElement
         * @override
         */
        getComputedStyleValue(property) {
          return styleInterface.getComputedStyleValue(/** @type {!Element} */(this), property);
        }

        // debounce

        /**
         * Call `debounce` to collapse multiple requests for a named task into
         * one invocation which is made after the wait time has elapsed with
         * no new request.  If no wait time is given, the callback will be called
         * at microtask timing (guaranteed before paint).
         *
         *     debouncedClickAction(e) {
         *       // will not call `processClick` more than once per 100ms
         *       this.debounce('click', function() {
         *        this.processClick();
         *       } 100);
         *     }
         *
         * @param {string} jobName String to identify the debounce job.
         * @param {function():void} callback Function that is called (with `this`
         *   context) when the wait time elapses.
         * @param {number=} wait Optional wait time in milliseconds (ms) after the
         *   last signal that must elapse before invoking `callback`
         * @return {!Object} Returns a debouncer object on which exists the
         * following methods: `isActive()` returns true if the debouncer is
         * active; `cancel()` cancels the debouncer if it is active;
         * `flush()` immediately invokes the debounced callback if the debouncer
         * is active.
         * @override
         */
        debounce(jobName, callback, wait) {
          this._debouncers = this._debouncers || {};
          return this._debouncers[jobName] = Debouncer.debounce(
                this._debouncers[jobName]
              , wait > 0 ? timeOut.after(wait) : microTask
              , callback.bind(this));
        }

        /**
         * Returns whether a named debouncer is active.
         *
         * @param {string} jobName The name of the debouncer started with `debounce`
         * @return {boolean} Whether the debouncer is active (has not yet fired).
         * @override
         */
        isDebouncerActive(jobName) {
          this._debouncers = this._debouncers || {};
          let debouncer = this._debouncers[jobName];
          return !!(debouncer && debouncer.isActive());
        }

        /**
         * Immediately calls the debouncer `callback` and inactivates it.
         *
         * @param {string} jobName The name of the debouncer started with `debounce`
         * @return {void}
         * @override
         */
        flushDebouncer(jobName) {
          this._debouncers = this._debouncers || {};
          let debouncer = this._debouncers[jobName];
          if (debouncer) {
            debouncer.flush();
          }
        }

        /**
         * Cancels an active debouncer.  The `callback` will not be called.
         *
         * @param {string} jobName The name of the debouncer started with `debounce`
         * @return {void}
         * @override
         */
        cancelDebouncer(jobName) {
          this._debouncers = this._debouncers || {};
          let debouncer = this._debouncers[jobName];
          if (debouncer) {
            debouncer.cancel();
          }
        }

        /**
         * Runs a callback function asynchronously.
         *
         * By default (if no waitTime is specified), async callbacks are run at
         * microtask timing, which will occur before paint.
         *
         * @param {!Function} callback The callback function to run, bound to
         *     `this`.
         * @param {number=} waitTime Time to wait before calling the
         *   `callback`.  If unspecified or 0, the callback will be run at microtask
         *   timing (before paint).
         * @return {number} Handle that may be used to cancel the async job.
         * @override
         */
        async(callback, waitTime) {
          return waitTime > 0 ? timeOut.run(callback.bind(this), waitTime) :
              ~microTask.run(callback.bind(this));
        }

        /**
         * Cancels an async operation started with `async`.
         *
         * @param {number} handle Handle returned from original `async` call to
         *   cancel.
         * @return {void}
         * @override
         */
        cancelAsync(handle) {
          handle < 0 ? microTask.cancel(~handle) :
              timeOut.cancel(handle);
        }

        // other

        /**
         * Convenience method for creating an element and configuring it.
         *
         * @param {string} tag HTML element tag to create.
         * @param {Object=} props Object of properties to configure on the
         *    instance.
         * @return {!Element} Newly created and configured element.
         * @override
         */
        create(tag, props) {
          let elt = document.createElement(tag);
          if (props) {
            if (elt.setProperties) {
              elt.setProperties(props);
            } else {
              for (let n in props) {
                elt[n] = props[n];
              }
            }
          }
          return elt;
        }

        /**
         * Polyfill for Element.prototype.matches, which is sometimes still
         * prefixed.
         *
         * @param {string} selector Selector to test.
         * @param {!Element=} node Element to test the selector against.
         * @return {boolean} Whether the element matches the selector.
         * @override
         */
        elementMatches(selector, node) {
          return matchesSelector( (node || this), selector);
        }

        /**
         * Toggles an HTML attribute on or off.
         *
         * @param {string} name HTML attribute name
         * @param {boolean=} bool Boolean to force the attribute on or off.
         *    When unspecified, the state of the attribute will be reversed.
         * @return {boolean} true if the attribute now exists
         * @override
         */
        toggleAttribute(name, bool) {
          let node = /** @type {Element} */(this);
          if (arguments.length === 3) {
            node = /** @type {Element} */(arguments[2]);
          }
          if (arguments.length == 1) {
            bool = !node.hasAttribute(name);
          }
          if (bool) {
            wrap(node).setAttribute(name, '');
            return true;
          } else {
            wrap(node).removeAttribute(name);
            return false;
          }
        }


        /**
         * Toggles a CSS class on or off.
         *
         * @param {string} name CSS class name
         * @param {boolean=} bool Boolean to force the class on or off.
         *    When unspecified, the state of the class will be reversed.
         * @param {Element=} node Node to target.  Defaults to `this`.
         * @return {void}
         * @override
         */
        toggleClass(name, bool, node) {
          node = /** @type {Element} */ (node || this);
          if (arguments.length == 1) {
            bool = !node.classList.contains(name);
          }
          if (bool) {
            node.classList.add(name);
          } else {
            node.classList.remove(name);
          }
        }

        /**
         * Cross-platform helper for setting an element's CSS `transform` property.
         *
         * @param {string} transformText Transform setting.
         * @param {Element=} node Element to apply the transform to.
         * Defaults to `this`
         * @return {void}
         * @override
         */
        transform(transformText, node) {
          node = /** @type {Element} */ (node || this);
          node.style.webkitTransform = transformText;
          node.style.transform = transformText;
        }

        /**
         * Cross-platform helper for setting an element's CSS `translate3d`
         * property.
         *
         * @param {number|string} x X offset.
         * @param {number|string} y Y offset.
         * @param {number|string} z Z offset.
         * @param {Element=} node Element to apply the transform to.
         * Defaults to `this`.
         * @return {void}
         * @override
         */
        translate3d(x, y, z, node) {
          node = /** @type {Element} */ (node || this);
          this.transform('translate3d(' + x + ',' + y + ',' + z + ')', node);
        }

        /**
         * Removes an item from an array, if it exists.
         *
         * If the array is specified by path, a change notification is
         * generated, so that observers, data bindings and computed
         * properties watching that path can update.
         *
         * If the array is passed directly, **no change
         * notification is generated**.
         *
         * @param {string | !Array<number|string>} arrayOrPath Path to array from
         *     which to remove the item
         *   (or the array itself).
         * @param {*} item Item to remove.
         * @return {Array} Array containing item removed.
         * @override
         */
        arrayDelete(arrayOrPath, item) {
          let index;
          if (Array.isArray(arrayOrPath)) {
            index = arrayOrPath.indexOf(item);
            if (index >= 0) {
              return arrayOrPath.splice(index, 1);
            }
          } else {
            let arr = get(this, arrayOrPath);
            index = arr.indexOf(item);
            if (index >= 0) {
              return this.splice(arrayOrPath, index, 1);
            }
          }
          return null;
        }

        // logging

        /**
         * Facades `console.log`/`warn`/`error` as override point.
         *
         * @param {string} level One of 'log', 'warn', 'error'
         * @param {Array} args Array of strings or objects to log
         * @return {void}
         * @override
         */
        _logger(level, args) {
          // accept ['foo', 'bar'] and [['foo', 'bar']]
          if (Array.isArray(args) && args.length === 1 && Array.isArray(args[0])) {
            args = args[0];
          }
          switch(level) {
            case 'log':
            case 'warn':
            case 'error':
              console[level](...args);
          }
        }

        /**
         * Facades `console.log` as an override point.
         *
         * @param {...*} args Array of strings or objects to log
         * @return {void}
         * @override
         */
        _log(...args) {
          this._logger('log', args);
        }

        /**
         * Facades `console.warn` as an override point.
         *
         * @param {...*} args Array of strings or objects to log
         * @return {void}
         * @override
         */
        _warn(...args) {
          this._logger('warn', args);
        }

        /**
         * Facades `console.error` as an override point.
         *
         * @param {...*} args Array of strings or objects to log
         * @return {void}
         * @override
         */
        _error(...args) {
          this._logger('error', args);
        }

        /**
         * Formats a message using the element type an a method name.
         *
         * @param {string} methodName Method name to associate with message
         * @param {...*} args Array of strings or objects to log
         * @return {!Array} Array with formatting information for `console`
         *   logging.
         * @override
         */
        _logf(methodName, ...args) {
          return ['[%s::%s]', this.is, methodName, ...args];
        }

      }

      LegacyElement.prototype.is = '';

      return LegacyElement;
    });

    /* Copyright 2016 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * Compares tag names asciinumerically broken into components.
     *
     * <p>This is the comparison function used for sorting most string values in
     * TensorBoard. Unlike the standard asciibetical comparator, this function
     * knows that 'a10b' > 'a2b'. Fixed point and engineering notation are
     * supported. This function also splits the input by slash and underscore to
     * perform array comparison. Therefore it knows that 'a/a' < 'a+/a' even
     * though '+' < '/' in the ASCII table.
     */
    function compareTagNames(a, b) {
        let ai = 0;
        let bi = 0;
        while (true) {
            if (ai === a.length) {
                return bi === b.length ? 0 : -1;
            }
            if (bi === b.length) {
                return 1;
            }
            if (isDigit(a[ai]) && isDigit(b[bi])) {
                const ais = ai;
                const bis = bi;
                ai = consumeNumber(a, ai + 1);
                bi = consumeNumber(b, bi + 1);
                const an = parseFloat(a.slice(ais, ai));
                const bn = parseFloat(b.slice(bis, bi));
                if (an < bn) {
                    return -1;
                }
                if (an > bn) {
                    return 1;
                }
                continue;
            }
            if (isBreak(a[ai])) {
                if (!isBreak(b[bi])) {
                    return -1;
                }
            }
            else if (isBreak(b[bi])) {
                return 1;
            }
            else if (a[ai] < b[bi]) {
                return -1;
            }
            else if (a[ai] > b[bi]) {
                return 1;
            }
            ai++;
            bi++;
        }
    }
    function consumeNumber(s, i) {
        let State;
        (function (State) {
            State[State["NATURAL"] = 0] = "NATURAL";
            State[State["REAL"] = 1] = "REAL";
            State[State["EXPONENT_SIGN"] = 2] = "EXPONENT_SIGN";
            State[State["EXPONENT"] = 3] = "EXPONENT";
        })(State || (State = {}));
        let state = State.NATURAL;
        for (; i < s.length; i++) {
            if (state === State.NATURAL) {
                if (s[i] === '.') {
                    state = State.REAL;
                }
                else if (s[i] === 'e' || s[i] === 'E') {
                    state = State.EXPONENT_SIGN;
                }
                else if (!isDigit(s[i])) {
                    break;
                }
            }
            else if (state === State.REAL) {
                if (s[i] === 'e' || s[i] === 'E') {
                    state = State.EXPONENT_SIGN;
                }
                else if (!isDigit(s[i])) {
                    break;
                }
            }
            else if (state === State.EXPONENT_SIGN) {
                if (isDigit(s[i]) || s[i] === '+' || s[i] === '-') {
                    state = State.EXPONENT;
                }
                else {
                    break;
                }
            }
            else if (state === State.EXPONENT) {
                if (!isDigit(s[i])) {
                    break;
                }
            }
        }
        return i;
    }
    function isDigit(c) {
        return '0' <= c && c <= '9';
    }
    function isBreak(c) {
        // TODO(@jart): Remove underscore when people stop using it like a slash.
        return c === '/' || c === '_' || isDigit(c);
    }

    /* Copyright 2015 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    const TYPES$1 = [];
    /** Given a RunToTag, return sorted array of all runs */
    function getRunsNamed(r) {
        return lodash.keys(r).sort(compareTagNames);
    }
    /** Given a RunToTag, return array of all tags (sorted + dedup'd) */
    function getTags(r) {
        return lodash.union.apply(null, lodash.values(r)).sort(compareTagNames);
    }
    /**
     * Given a RunToTag and an array of runs, return every tag that appears for
     * at least one run.
     * Sorted, deduplicated.
     */
    function filterTags(r, runs) {
        let result = [];
        runs.forEach((x) => (result = result.concat(r[x])));
        return lodash.uniq(result).sort(compareTagNames);
    }

    /**
     * Manages many fetch requests. Launches up to nSimultaneousRequests
     * simultaneously, and maintains a LIFO queue of requests to process when
     * more urls are requested than can be handled at once. The queue can be
     * cleared.
     *
     * When a request is made, a Promise is returned which resolves with the
     * parsed JSON result from the request.
     */
    class RequestCancellationError extends Error {
        constructor() {
            super(...arguments);
            this.name = 'RequestCancellationError';
        }
    }
    class InvalidRequestOptionsError extends Error {
        constructor(msg) {
            super(msg);
            this.name = 'InvalidRequestOptionsError';
            // The following is needed due to a limitation of TypeScript when
            // extending 'Error'. See: https://github.com/Microsoft/TypeScript/wiki/Breaking-Changes#extending-built-ins-like-error-array-and-map-may-no-longer-work
            Object.setPrototypeOf(this, InvalidRequestOptionsError.prototype);
        }
    }
    class RequestNetworkError extends Error {
        constructor(req, url) {
            super();
            this.message = `RequestNetworkError: ${req.status} at ${url}`;
            this.name = 'RequestNetworkError';
            this.req = req;
            this.url = url;
        }
    }
    /** The HTTP method-type to use. Currently only 'GET' and 'POST' are
     * supported.
     */
    var HttpMethodType;
    (function (HttpMethodType) {
        HttpMethodType["GET"] = "GET";
        HttpMethodType["POST"] = "POST";
    })(HttpMethodType || (HttpMethodType = {}));
    /**
     * Holds options that can be used to configure the HTTP request.
     */
    class RequestOptions {
        // Validates this object. Throws InvalidRequestOptionsError on error.
        validate() {
            if (this.methodType === HttpMethodType.GET) {
                // We don't allow a body for a GET.
                if (this.body) {
                    throw new InvalidRequestOptionsError('body must be missing for a GET request.');
                }
            }
            // We allow body-less or contentType-less POSTs even if they don't
            // make much sense.
        }
    }
    class RequestManager {
        constructor(nSimultaneousRequests = 1000, maxRetries = 3) {
            this._queue = [];
            this._nActiveRequests = 0;
            this._nSimultaneousRequests = nSimultaneousRequests;
            this._maxRetries = maxRetries;
        }
        /**
         * Gives a promise that loads assets from given url (respects queuing). If
         * postData is provided, this request will use POST, not GET. This is an
         * object mapping POST keys to string values.
         */
        request(url, postData) {
            const requestOptions = requestOptionsFromPostData(postData);
            return this.requestWithOptions(url, requestOptions);
        }
        requestWithOptions(url, requestOptions) {
            requestOptions.validate();
            const promise = new Promise((resolve, reject) => {
                const resolver = { resolve: resolve, reject: reject };
                this._queue.push(resolver);
                this.launchRequests();
            })
                .then(() => {
                return this.promiseWithRetries(url, this._maxRetries, requestOptions);
            })
                .then((response) => {
                // Success - Let's free space for another active
                // request, and launch it
                this._nActiveRequests--;
                this.launchRequests();
                return response;
            }, (rejection) => {
                if (rejection.name === 'RequestNetworkError') {
                    // If we failed due to network error, we should
                    // decrement
                    // _nActiveRequests because this request was
                    // active
                    this._nActiveRequests--;
                    this.launchRequests();
                }
                return Promise.reject(rejection);
            });
            return promise;
        }
        fetch(url, fetchOptions) {
            return new Promise((resolve, reject) => {
                const resolver = { resolve: resolve, reject: reject };
                this._queue.push(resolver);
                this.launchRequests();
            }).then(() => {
                let numTries = 1;
                return new Promise((resolve) => {
                    const retryFetch = () => {
                        fetch(url, fetchOptions).then((response) => {
                            if (!response.ok && this._maxRetries > numTries) {
                                numTries++;
                                retryFetch();
                                return;
                            }
                            resolve(response);
                            this._nActiveRequests--;
                            this.launchRequests();
                        });
                    };
                    retryFetch();
                });
            });
        }
        clearQueue() {
            while (this._queue.length > 0) {
                this._queue
                    .pop()
                    .reject(new RequestCancellationError('Request cancelled by clearQueue'));
            }
        }
        /* Return number of currently pending requests */
        activeRequests() {
            return this._nActiveRequests;
        }
        /* Return total number of outstanding requests (includes queue) */
        outstandingRequests() {
            return this._nActiveRequests + this._queue.length;
        }
        launchRequests() {
            while (this._nActiveRequests < this._nSimultaneousRequests &&
                this._queue.length > 0) {
                this._nActiveRequests++;
                this._queue.pop().resolve();
            }
        }
        /**
         * Try to request a given URL using overwritable _promiseFromUrl method.
         * If the request fails for any reason, we will retry up to maxRetries
         * times. In practice, this will help us paper over transient network issues
         * like '502 Bad Gateway'.
         * By default, Chrome displays network errors in console, so
         * the user will be able to tell when the requests are failing. I think this
         * is a feature, if the request failures and retries are causing any
         * pain to users, they can see it and file issues.
         */
        promiseWithRetries(url, maxRetries, requestOptions) {
            var success = (x) => x;
            var failure = (x) => {
                if (maxRetries > 0) {
                    return this.promiseWithRetries(url, maxRetries - 1, requestOptions);
                }
                else {
                    return Promise.reject(x);
                }
            };
            return this._promiseFromUrl(url, requestOptions).then(success, failure);
        }
        /* Actually get promise from url using XMLHttpRequest */
        _promiseFromUrl(url, requestOptions) {
            return new Promise((resolve, reject) => {
                const req = buildXMLHttpRequest(requestOptions.methodType, url, requestOptions.withCredentials, requestOptions.contentType);
                req.onload = function () {
                    if (req.status === 200) {
                        resolve(JSON.parse(req.responseText));
                    }
                    else {
                        reject(new RequestNetworkError(req, url));
                    }
                };
                req.onerror = function () {
                    reject(new RequestNetworkError(req, url));
                };
                if (requestOptions.body) {
                    req.send(requestOptions.body);
                }
                else {
                    req.send();
                }
            });
        }
    }
    function buildXMLHttpRequest(methodType, url, withCredentials, contentType) {
        const req = new XMLHttpRequest();
        req.open(methodType, url);
        if (withCredentials) {
            req.withCredentials = withCredentials;
        }
        if (contentType) {
            req.setRequestHeader('Content-Type', contentType);
        }
        return req;
    }
    function requestOptionsFromPostData(postData) {
        const result = new RequestOptions();
        if (!postData) {
            result.methodType = HttpMethodType.GET;
            return result;
        }
        result.methodType = HttpMethodType.POST;
        result.body = formDataFromDictionary(postData);
        return result;
    }
    function formDataFromDictionary(postData) {
        const formData = new FormData();
        for (const [key, maybeValues] of Object.entries(postData)) {
            const values = Array.isArray(maybeValues) ? maybeValues : [maybeValues];
            for (const value of values) {
                formData.append(key, value);
            }
        }
        return formData;
    }

    const EXPERIMENTAL_PLUGINS_QUERY_PARAM = 'experimentalPlugin';
    /**
     * Save the initial URL query params, before the AppRoutingEffects initialize,
     * and before creating the router.
     */
    const initialURLSearchParams = new URLSearchParams(window.location.search);
    let _router = createRouter();
    /**
     * Create a router for communicating with the TensorBoard backend. You
     * can pass this to `setRouter` to make it the global router.
     *
     * @param dataDir {string=} The base prefix for data endpoints.
     */
    function createRouter(dataDir = 'data', urlSearchParams = initialURLSearchParams) {
        if (dataDir[dataDir.length - 1] === '/') {
            dataDir = dataDir.slice(0, dataDir.length - 1);
        }
        return {
            environment: () => createDataPath(dataDir, '/environment'),
            experiments: () => createDataPath(dataDir, '/experiments'),
            pluginRoute: (pluginName, route, params) => {
                return createDataPath(dataDir + '/plugin', `/${pluginName}${route}`, params);
            },
            pluginsListing: () => createDataPath(dataDir, '/plugins_listing', createSearchParam({
                [EXPERIMENTAL_PLUGINS_QUERY_PARAM]: urlSearchParams.getAll(EXPERIMENTAL_PLUGINS_QUERY_PARAM),
            })),
            runs: () => createDataPath(dataDir, '/runs'),
            runsForExperiment: (id) => {
                return createDataPath(dataDir, '/experiment_runs', createSearchParam({ experiment: String(id) }));
            },
        };
    }
    /**
     * @return {Router} the global router
     */
    function getRouter() {
        return _router;
    }
    /**
     * Set the global router, to be returned by future calls to `getRouter`.
     * You may wish to invoke this if you are running a demo server with a
     * custom path prefix, or if you have customized the TensorBoard backend
     * to use a different path.
     *
     * @param {Router} router the new global router
     */
    function setRouter(router) {
        if (router == null) {
            throw new Error('Router required, but got: ' + router);
        }
        _router = router;
    }
    function createDataPath(dataDir, route, params = new URLSearchParams()) {
        let relativePath = dataDir + route;
        if (String(params)) {
            const delimiter = route.includes('?') ? '&' : '?';
            relativePath += delimiter + String(params);
        }
        return relativePath;
    }
    function createSearchParam(params = {}) {
        const keys = Object.keys(params)
            .sort()
            .filter((k) => params[k]);
        const searchParams = new URLSearchParams();
        keys.forEach((key) => {
            const values = params[key];
            const array = Array.isArray(values) ? values : [values];
            array.forEach((val) => searchParams.append(key, val));
        });
        return searchParams;
    }

    /* Copyright 2017 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    var CategoryType;
    (function (CategoryType) {
        CategoryType[CategoryType["SEARCH_RESULTS"] = 0] = "SEARCH_RESULTS";
        CategoryType[CategoryType["PREFIX_GROUP"] = 1] = "PREFIX_GROUP";
    })(CategoryType || (CategoryType = {}));
    /**
     * Compute a category containing the search results for the given query.
     */
    function categorizeBySearchQuery(xs, query) {
        const re = (() => {
            try {
                return new RegExp(query);
            }
            catch (e) {
                return null;
            }
        })();
        return {
            name: query,
            metadata: {
                type: CategoryType.SEARCH_RESULTS,
                validRegex: !!re,
                universalRegex: query === '.*',
            },
            items: re ? xs.filter((x) => x.match(re)) : [],
        };
    }
    /**
     * Compute the quotient set $X/{\sim}$, where $a \sim b$ if $a$ and $b$
     * share a common `separator`-prefix. Order is preserved.
     */
    function categorizeByPrefix(xs, separator = '/') {
        const categories = [];
        const categoriesByName = {};
        xs.forEach((x) => {
            const index = x.indexOf(separator);
            const name = index >= 0 ? x.slice(0, index) : x;
            if (!categoriesByName[name]) {
                const category = {
                    name,
                    metadata: { type: CategoryType.PREFIX_GROUP },
                    items: [],
                };
                categoriesByName[name] = category;
                categories.push(category);
            }
            categoriesByName[name].items.push(x);
        });
        return categories;
    }
    /*
     * Compute the standard categorization of the given input, including
     * both search categories and prefix categories.
     */
    function categorize(xs, query = '') {
        const byFilter = [categorizeBySearchQuery(xs, query)];
        const byPrefix = categorizeByPrefix(xs);
        return [].concat(byFilter, byPrefix);
    }
    function categorizeTags(runToTag, selectedRuns, query) {
        const tags = getTags(runToTag);
        const categories = categorize(tags, query);
        const tagToRuns = createTagToRuns(lodash.pick(runToTag, selectedRuns));
        return categories.map(({ name, metadata, items }) => ({
            name,
            metadata,
            items: items.map((tag) => ({
                tag,
                runs: (tagToRuns.get(tag) || []).slice(),
            })),
        }));
    }
    function createTagToRuns(runToTag) {
        const tagToRun = new Map();
        Object.keys(runToTag).forEach((run) => {
            runToTag[run].forEach((tag) => {
                const runs = tagToRun.get(tag) || [];
                runs.push(run);
                tagToRun.set(tag, runs);
            });
        });
        return tagToRun;
    }
    function compareTagRun(a, b) {
        const c = compareTagNames(a.tag, b.tag);
        if (c != 0) {
            return c;
        }
        return compareTagNames(a.run, b.run);
    }
    function categorizeRunTagCombinations(runToTag, selectedRuns, query) {
        const tagCategories = categorizeTags(runToTag, selectedRuns, query);
        function explodeCategory(tagCategory) {
            const items = lodash.flatten(tagCategory.items.map(({ tag, runs }) => runs.map((run) => ({ tag, run }))));
            items.sort(compareTagRun);
            return {
                name: tagCategory.name,
                metadata: tagCategory.metadata,
                items,
            };
        }
        return tagCategories.map(explodeCategory);
    }

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    const lifecycleProps = {
      attached: true,
      detached: true,
      ready: true,
      created: true,
      beforeRegister: true,
      registered: true,
      attributeChanged: true,
      listeners: true,
      hostAttributes: true
    };

    const excludeOnInfo = {
      attached: true,
      detached: true,
      ready: true,
      created: true,
      beforeRegister: true,
      registered: true,
      attributeChanged: true,
      behaviors: true,
      _noAccessors: true
    };

    const excludeOnBehaviors = Object.assign({
      listeners: true,
      hostAttributes: true,
      properties: true,
      observers: true,
    }, excludeOnInfo);

    function copyProperties(source, target, excludeProps) {
      const noAccessors = source._noAccessors;
      const propertyNames = Object.getOwnPropertyNames(source);
      for (let i = 0; i < propertyNames.length; i++) {
        let p = propertyNames[i];
        if (p in excludeProps) {
          continue;
        }
        if (noAccessors) {
          target[p] = source[p];
        } else {
          let pd = Object.getOwnPropertyDescriptor(source, p);
          if (pd) {
            // ensure property is configurable so that a later behavior can
            // re-configure it.
            pd.configurable = true;
            Object.defineProperty(target, p, pd);
          }
        }
      }
    }

    /**
     * Applies a "legacy" behavior or array of behaviors to the provided class.
     *
     * Note: this method will automatically also apply the `LegacyElementMixin`
     * to ensure that any legacy behaviors can rely on legacy Polymer API on
     * the underlying element.
     *
     * @function
     * @template T
     * @param {!Object|!Array<!Object>} behaviors Behavior object or array of behaviors.
     * @param {function(new:T)} klass Element class.
     * @return {?} Returns a new Element class extended by the
     * passed in `behaviors` and also by `LegacyElementMixin`.
     * @suppress {invalidCasts, checkTypes}
     */
    function mixinBehaviors(behaviors, klass) {
      return GenerateClassFromInfo({}, LegacyElementMixin(klass), behaviors);
    }

    // NOTE:
    // 1.x
    // Behaviors were mixed in *in reverse order* and de-duped on the fly.
    // The rule was that behavior properties were copied onto the element
    // prototype if and only if the property did not already exist.
    // Given: Polymer{ behaviors: [A, B, C, A, B]}, property copy order was:
    // (1), B, (2), A, (3) C. This means prototype properties win over
    // B properties win over A win over C. This mirrors what would happen
    // with inheritance if element extended B extended A extended C.
    //
    // Again given, Polymer{ behaviors: [A, B, C, A, B]}, the resulting
    // `behaviors` array was [C, A, B].
    // Behavior lifecycle methods were called in behavior array order
    // followed by the element, e.g. (1) C.created, (2) A.created,
    // (3) B.created, (4) element.created. There was no support for
    // super, and "super-behavior" methods were callable only by name).
    //
    // 2.x
    // Behaviors are made into proper mixins which live in the
    // element's prototype chain. Behaviors are placed in the element prototype
    // eldest to youngest and de-duped youngest to oldest:
    // So, first [A, B, C, A, B] becomes [C, A, B] then,
    // the element prototype becomes (oldest) (1) PolymerElement, (2) class(C),
    // (3) class(A), (4) class(B), (5) class(Polymer({...})).
    // Result:
    // This means element properties win over B properties win over A win
    // over C. (same as 1.x)
    // If lifecycle is called (super then me), order is
    // (1) C.created, (2) A.created, (3) B.created, (4) element.created
    // (again same as 1.x)
    function applyBehaviors(proto, behaviors, lifecycle) {
      for (let i=0; i<behaviors.length; i++) {
        applyInfo(proto, behaviors[i], lifecycle, excludeOnBehaviors);
      }
    }

    function applyInfo(proto, info, lifecycle, excludeProps) {
      copyProperties(info, proto, excludeProps);
      for (let p in lifecycleProps) {
        if (info[p]) {
          lifecycle[p] = lifecycle[p] || [];
          lifecycle[p].push(info[p]);
        }
      }
    }

    /**
     * @param {Array} behaviors List of behaviors to flatten.
     * @param {Array=} list Target list to flatten behaviors into.
     * @param {Array=} exclude List of behaviors to exclude from the list.
     * @return {!Array} Returns the list of flattened behaviors.
     */
    function flattenBehaviors(behaviors, list, exclude) {
      list = list || [];
      for (let i=behaviors.length-1; i >= 0; i--) {
        let b = behaviors[i];
        if (b) {
          if (Array.isArray(b)) {
            flattenBehaviors(b, list);
          } else {
            // dedup
            if (list.indexOf(b) < 0 && (!exclude || exclude.indexOf(b) < 0)) {
              list.unshift(b);
            }
          }
        } else {
          console.warn('behavior is null, check for missing or 404 import');
        }
      }
      return list;
    }

    /**
     * Copies property descriptors from source to target, overwriting all fields
     * of any previous descriptor for a property *except* for `value`, which is
     * merged in from the target if it does not exist on the source.
     *
     * @param {*} target Target properties object
     * @param {*} source Source properties object
     */
    function mergeProperties(target, source) {
      for (const p in source) {
        const targetInfo = target[p];
        const sourceInfo = source[p];
        if (!('value' in sourceInfo) && targetInfo && ('value' in targetInfo)) {
          target[p] = Object.assign({value: targetInfo.value}, sourceInfo);
        } else {
          target[p] = sourceInfo;
        }
      }
    }

    const LegacyElement = LegacyElementMixin(HTMLElement);

    /* Note about construction and extension of legacy classes.
      [Changed in Q4 2018 to optimize performance.]

      When calling `Polymer` or `mixinBehaviors`, the generated class below is
      made. The list of behaviors was previously made into one generated class per
      behavior, but this is no longer the case as behaviors are now called
      manually. Note, there may *still* be multiple generated classes in the
      element's prototype chain if extension is used with `mixinBehaviors`.

      The generated class is directly tied to the info object and behaviors
      used to create it. That list of behaviors is filtered so it's only the
      behaviors not active on the superclass. In order to call through to the
      entire list of lifecycle methods, it's important to call `super`.

      The element's `properties` and `observers` are controlled via the finalization
      mechanism provided by `PropertiesMixin`. `Properties` and `observers` are
      collected by manually traversing the prototype chain and merging.

      To limit changes, the `_registered` method is called via `_initializeProperties`
      and not `_finalizeClass`.

    */
    /**
     * @param {!PolymerInit} info Polymer info object
     * @param {function(new:HTMLElement)} Base base class to extend with info object
     * @param {Object=} behaviors behaviors to copy into the element
     * @return {function(new:HTMLElement)} Generated class
     * @suppress {checkTypes}
     * @private
     */
    function GenerateClassFromInfo(info, Base, behaviors) {

      // manages behavior and lifecycle processing (filled in after class definition)
      let behaviorList;
      const lifecycle = {};

      /** @private */
      class PolymerGenerated extends Base {

        // explicitly not calling super._finalizeClass
        /** @nocollapse */
        static _finalizeClass() {
          // if calling via a subclass that hasn't been generated, pass through to super
          if (!this.hasOwnProperty(JSCompiler_renameProperty('generatedFrom', this))) {
            // TODO(https://github.com/google/closure-compiler/issues/3240):
            //     Change back to just super.methodCall()
            Base._finalizeClass.call(this);
          } else {
            // interleave properties and observers per behavior and `info`
            if (behaviorList) {
              for (let i=0, b; i < behaviorList.length; i++) {
                b = behaviorList[i];
                if (b.properties) {
                  this.createProperties(b.properties);
                }
                if (b.observers) {
                  this.createObservers(b.observers, b.properties);
                }
              }
            }
            if (info.properties) {
              this.createProperties(info.properties);
            }
            if (info.observers) {
              this.createObservers(info.observers, info.properties);
            }
            // make sure to prepare the element template
            this._prepareTemplate();
          }
        }

        /** @nocollapse */
        static get properties() {
          const properties = {};
          if (behaviorList) {
            for (let i=0; i < behaviorList.length; i++) {
              mergeProperties(properties, behaviorList[i].properties);
            }
          }
          mergeProperties(properties, info.properties);
          return properties;
        }

        /** @nocollapse */
        static get observers() {
          let observers = [];
          if (behaviorList) {
            for (let i=0, b; i < behaviorList.length; i++) {
              b = behaviorList[i];
              if (b.observers) {
                observers = observers.concat(b.observers);
              }
            }
          }
          if (info.observers) {
            observers = observers.concat(info.observers);
          }
          return observers;
        }

        /**
         * @return {void}
         */
        created() {
          super.created();
          const list = lifecycle.created;
          if (list) {
            for (let i=0; i < list.length; i++) {
              list[i].call(this);
            }
          }
        }

        /**
         * @return {void}
         */
        _registered() {
          /* NOTE: `beforeRegister` is called here for bc, but the behavior
            is different than in 1.x. In 1.0, the method was called *after*
            mixing prototypes together but *before* processing of meta-objects.
            However, dynamic effects can still be set here and can be done either
            in `beforeRegister` or `registered`. It is no longer possible to set
            `is` in `beforeRegister` as you could in 1.x.
          */
          // only proceed if the generated class' prototype has not been registered.
          const generatedProto = PolymerGenerated.prototype;
          if (!generatedProto.hasOwnProperty(JSCompiler_renameProperty('__hasRegisterFinished', generatedProto))) {
            generatedProto.__hasRegisterFinished = true;
            // ensure superclass is registered first.
            super._registered();
            // copy properties onto the generated class lazily if we're optimizing,
            if (legacyOptimizations) {
              copyPropertiesToProto(generatedProto);
            }
            // make sure legacy lifecycle is called on the *element*'s prototype
            // and not the generated class prototype; if the element has been
            // extended, these are *not* the same.
            const proto = Object.getPrototypeOf(this);
            let list = lifecycle.beforeRegister;
            if (list) {
              for (let i=0; i < list.length; i++) {
                list[i].call(proto);
              }
            }
            list = lifecycle.registered;
            if (list) {
              for (let i=0; i < list.length; i++) {
                list[i].call(proto);
              }
            }
          }
        }

        /**
         * @return {void}
         */
        _applyListeners() {
          super._applyListeners();
          const list = lifecycle.listeners;
          if (list) {
            for (let i=0; i < list.length; i++) {
              const listeners = list[i];
              if (listeners) {
                for (let l in listeners) {
                  this._addMethodEventListenerToNode(this, l, listeners[l]);
                }
              }
            }
          }
        }

        // note: exception to "super then me" rule;
        // do work before calling super so that super attributes
        // only apply if not already set.
        /**
         * @return {void}
         */
        _ensureAttributes() {
          const list = lifecycle.hostAttributes;
          if (list) {
            for (let i=list.length-1; i >= 0; i--) {
              const hostAttributes = list[i];
              for (let a in hostAttributes) {
                  this._ensureAttribute(a, hostAttributes[a]);
                }
            }
          }
          super._ensureAttributes();
        }

        /**
         * @return {void}
         */
        ready() {
          super.ready();
          let list = lifecycle.ready;
          if (list) {
            for (let i=0; i < list.length; i++) {
              list[i].call(this);
            }
          }
        }

        /**
         * @return {void}
         */
        attached() {
          super.attached();
          let list = lifecycle.attached;
          if (list) {
            for (let i=0; i < list.length; i++) {
              list[i].call(this);
            }
          }
        }

        /**
         * @return {void}
         */
        detached() {
          super.detached();
          let list = lifecycle.detached;
          if (list) {
            for (let i=0; i < list.length; i++) {
              list[i].call(this);
            }
          }
        }

        /**
         * Implements native Custom Elements `attributeChangedCallback` to
         * set an attribute value to a property via `_attributeToProperty`.
         *
         * @param {string} name Name of attribute that changed
         * @param {?string} old Old attribute value
         * @param {?string} value New attribute value
         * @return {void}
         */
        attributeChanged(name, old, value) {
          super.attributeChanged();
          let list = lifecycle.attributeChanged;
          if (list) {
            for (let i=0; i < list.length; i++) {
              list[i].call(this, name, old, value);
            }
          }
        }
      }

      // apply behaviors, note actual copying is done lazily at first instance creation
      if (behaviors) {
        // NOTE: ensure the behavior is extending a class with
        // legacy element api. This is necessary since behaviors expect to be able
        // to access 1.x legacy api.
        if (!Array.isArray(behaviors)) {
          behaviors = [behaviors];
        }
        let superBehaviors = Base.prototype.behaviors;
        // get flattened, deduped list of behaviors *not* already on super class
        behaviorList = flattenBehaviors(behaviors, null, superBehaviors);
        PolymerGenerated.prototype.behaviors = superBehaviors ?
          superBehaviors.concat(behaviors) : behaviorList;
      }

      const copyPropertiesToProto = (proto) => {
        if (behaviorList) {
          applyBehaviors(proto, behaviorList, lifecycle);
        }
        applyInfo(proto, info, lifecycle, excludeOnInfo);
      };

      // copy properties if we're not optimizing
      if (!legacyOptimizations) {
        copyPropertiesToProto(PolymerGenerated.prototype);
      }

      PolymerGenerated.generatedFrom = info;

      return PolymerGenerated;
    }

    /**
     * Generates a class that extends `LegacyElement` based on the
     * provided info object.  Metadata objects on the `info` object
     * (`properties`, `observers`, `listeners`, `behaviors`, `is`) are used
     * for Polymer's meta-programming systems, and any functions are copied
     * to the generated class.
     *
     * Valid "metadata" values are as follows:
     *
     * `is`: String providing the tag name to register the element under. In
     * addition, if a `dom-module` with the same id exists, the first template
     * in that `dom-module` will be stamped into the shadow root of this element,
     * with support for declarative event listeners (`on-...`), Polymer data
     * bindings (`[[...]]` and `{{...}}`), and id-based node finding into
     * `this.$`.
     *
     * `properties`: Object describing property-related metadata used by Polymer
     * features (key: property names, value: object containing property metadata).
     * Valid keys in per-property metadata include:
     * - `type` (String|Number|Object|Array|...): Used by
     *   `attributeChangedCallback` to determine how string-based attributes
     *   are deserialized to JavaScript property values.
     * - `notify` (boolean): Causes a change in the property to fire a
     *   non-bubbling event called `<property>-changed`. Elements that have
     *   enabled two-way binding to the property use this event to observe changes.
     * - `readOnly` (boolean): Creates a getter for the property, but no setter.
     *   To set a read-only property, use the private setter method
     *   `_setProperty(property, value)`.
     * - `observer` (string): Observer method name that will be called when
     *   the property changes. The arguments of the method are
     *   `(value, previousValue)`.
     * - `computed` (string): String describing method and dependent properties
     *   for computing the value of this property (e.g. `'computeFoo(bar, zot)'`).
     *   Computed properties are read-only by default and can only be changed
     *   via the return value of the computing method.
     *
     * `observers`: Array of strings describing multi-property observer methods
     *  and their dependent properties (e.g. `'observeABC(a, b, c)'`).
     *
     * `listeners`: Object describing event listeners to be added to each
     *  instance of this element (key: event name, value: method name).
     *
     * `behaviors`: Array of additional `info` objects containing metadata
     * and callbacks in the same format as the `info` object here which are
     * merged into this element.
     *
     * `hostAttributes`: Object listing attributes to be applied to the host
     *  once created (key: attribute name, value: attribute value).  Values
     *  are serialized based on the type of the value.  Host attributes should
     *  generally be limited to attributes such as `tabIndex` and `aria-...`.
     *  Attributes in `hostAttributes` are only applied if a user-supplied
     *  attribute is not already present (attributes in markup override
     *  `hostAttributes`).
     *
     * In addition, the following Polymer-specific callbacks may be provided:
     * - `registered`: called after first instance of this element,
     * - `created`: called during `constructor`
     * - `attached`: called during `connectedCallback`
     * - `detached`: called during `disconnectedCallback`
     * - `ready`: called before first `attached`, after all properties of
     *   this element have been propagated to its template and all observers
     *   have run
     *
     * @param {!PolymerInit} info Object containing Polymer metadata and functions
     *   to become class methods.
     * @template T
     * @param {function(T):T} mixin Optional mixin to apply to legacy base class
     *   before extending with Polymer metaprogramming.
     * @return {function(new:HTMLElement)} Generated class
     */
    const Class = function(info, mixin) {
      if (!info) {
        console.warn('Polymer.Class requires `info` argument');
      }
      let klass = mixin ? mixin(LegacyElement) :
          LegacyElement;
      klass = GenerateClassFromInfo(info, klass, info.behaviors);
      // decorate klass with registration info
      klass.is = klass.prototype.is = info.is;
      return klass;
    };

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    /**
     * Legacy class factory and registration helper for defining Polymer
     * elements.
     *
     * This method is equivalent to
     *
     *     import {Class} from '@polymer/polymer/lib/legacy/class.js';
     *     customElements.define(info.is, Class(info));
     *
     * See `Class` for details on valid legacy metadata format for `info`.
     *
     * @global
     * @override
     * @function
     * @param {!PolymerInit} info Object containing Polymer metadata and functions
     *   to become class methods.
     * @return {function(new: HTMLElement)} Generated class
     * @suppress {duplicate, invalidCasts, checkTypes}
     */
    const Polymer = function(info) {
      // if input is a `class` (aka a function with a prototype), use the prototype
      // remember that the `constructor` will never be called
      let klass;
      if (typeof info === 'function') {
        klass = info;
      } else {
        klass = Polymer.Class(info);
      }
      // Copy opt out for `legacyNoObservedAttributes` from info object to class.
      if (info._legacyForceObservedAttributes) {
        klass.prototype._legacyForceObservedAttributes = info._legacyForceObservedAttributes;
      }
      customElements.define(klass.is, /** @type {!HTMLElement} */(klass));
      return klass;
    };

    Polymer.Class = Class;

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    // Common implementation for mixin & behavior
    function mutablePropertyChange(inst, property, value, old, mutableData) {
      let isObject;
      if (mutableData) {
        isObject = (typeof value === 'object' && value !== null);
        // Pull `old` for Objects from temp cache, but treat `null` as a primitive
        if (isObject) {
          old = inst.__dataTemp[property];
        }
      }
      // Strict equality check, but return false for NaN===NaN
      let shouldChange = (old !== value && (old === old || value === value));
      // Objects are stored in temporary cache (cleared at end of
      // turn), which is used for dirty-checking
      if (isObject && shouldChange) {
        inst.__dataTemp[property] = value;
      }
      return shouldChange;
    }

    /**
     * Element class mixin to skip strict dirty-checking for objects and arrays
     * (always consider them to be "dirty"), for use on elements utilizing
     * `PropertyEffects`
     *
     * By default, `PropertyEffects` performs strict dirty checking on
     * objects, which means that any deep modifications to an object or array will
     * not be propagated unless "immutable" data patterns are used (i.e. all object
     * references from the root to the mutation were changed).
     *
     * Polymer also provides a proprietary data mutation and path notification API
     * (e.g. `notifyPath`, `set`, and array mutation API's) that allow efficient
     * mutation and notification of deep changes in an object graph to all elements
     * bound to the same object graph.
     *
     * In cases where neither immutable patterns nor the data mutation API can be
     * used, applying this mixin will cause Polymer to skip dirty checking for
     * objects and arrays (always consider them to be "dirty").  This allows a
     * user to make a deep modification to a bound object graph, and then either
     * simply re-set the object (e.g. `this.items = this.items`) or call `notifyPath`
     * (e.g. `this.notifyPath('items')`) to update the tree.  Note that all
     * elements that wish to be updated based on deep mutations must apply this
     * mixin or otherwise skip strict dirty checking for objects/arrays.
     * Specifically, any elements in the binding tree between the source of a
     * mutation and the consumption of it must apply this mixin or enable the
     * `OptionalMutableData` mixin.
     *
     * In order to make the dirty check strategy configurable, see
     * `OptionalMutableData`.
     *
     * Note, the performance characteristics of propagating large object graphs
     * will be worse as opposed to using strict dirty checking with immutable
     * patterns or Polymer's path notification API.
     *
     * @mixinFunction
     * @polymer
     * @summary Element class mixin to skip strict dirty-checking for objects
     *   and arrays
     * @template T
     * @param {function(new:T)} superClass Class to apply mixin to.
     * @return {function(new:T)} superClass with mixin applied.
     */
    const MutableData = dedupingMixin(superClass => {

      /**
       * @polymer
       * @mixinClass
       * @implements {Polymer_MutableData}
       */
      class MutableData extends superClass {
        /**
         * Overrides `PropertyEffects` to provide option for skipping
         * strict equality checking for Objects and Arrays.
         *
         * This method pulls the value to dirty check against from the `__dataTemp`
         * cache (rather than the normal `__data` cache) for Objects.  Since the temp
         * cache is cleared at the end of a turn, this implementation allows
         * side-effects of deep object changes to be processed by re-setting the
         * same object (using the temp cache as an in-turn backstop to prevent
         * cycles due to 2-way notification).
         *
         * @param {string} property Property name
         * @param {*} value New property value
         * @param {*} old Previous property value
         * @return {boolean} Whether the property should be considered a change
         * @protected
         */
        _shouldPropertyChange(property, value, old) {
          return mutablePropertyChange(this, property, value, old, true);
        }

      }

      return MutableData;

    });

    /**
     * Element class mixin to add the optional ability to skip strict
     * dirty-checking for objects and arrays (always consider them to be
     * "dirty") by setting a `mutable-data` attribute on an element instance.
     *
     * By default, `PropertyEffects` performs strict dirty checking on
     * objects, which means that any deep modifications to an object or array will
     * not be propagated unless "immutable" data patterns are used (i.e. all object
     * references from the root to the mutation were changed).
     *
     * Polymer also provides a proprietary data mutation and path notification API
     * (e.g. `notifyPath`, `set`, and array mutation API's) that allow efficient
     * mutation and notification of deep changes in an object graph to all elements
     * bound to the same object graph.
     *
     * In cases where neither immutable patterns nor the data mutation API can be
     * used, applying this mixin will allow Polymer to skip dirty checking for
     * objects and arrays (always consider them to be "dirty").  This allows a
     * user to make a deep modification to a bound object graph, and then either
     * simply re-set the object (e.g. `this.items = this.items`) or call `notifyPath`
     * (e.g. `this.notifyPath('items')`) to update the tree.  Note that all
     * elements that wish to be updated based on deep mutations must apply this
     * mixin or otherwise skip strict dirty checking for objects/arrays.
     * Specifically, any elements in the binding tree between the source of a
     * mutation and the consumption of it must enable this mixin or apply the
     * `MutableData` mixin.
     *
     * While this mixin adds the ability to forgo Object/Array dirty checking,
     * the `mutableData` flag defaults to false and must be set on the instance.
     *
     * Note, the performance characteristics of propagating large object graphs
     * will be worse by relying on `mutableData: true` as opposed to using
     * strict dirty checking with immutable patterns or Polymer's path notification
     * API.
     *
     * @mixinFunction
     * @polymer
     * @summary Element class mixin to optionally skip strict dirty-checking
     *   for objects and arrays
     */
    const OptionalMutableData = dedupingMixin(superClass => {

      /**
       * @mixinClass
       * @polymer
       * @implements {Polymer_OptionalMutableData}
       */
      class OptionalMutableData extends superClass {

        /** @nocollapse */
        static get properties() {
          return {
            /**
             * Instance-level flag for configuring the dirty-checking strategy
             * for this element.  When true, Objects and Arrays will skip dirty
             * checking, otherwise strict equality checking will be used.
             */
            mutableData: Boolean
          };
        }

        /**
         * Overrides `PropertyEffects` to provide option for skipping
         * strict equality checking for Objects and Arrays.
         *
         * When `this.mutableData` is true on this instance, this method
         * pulls the value to dirty check against from the `__dataTemp` cache
         * (rather than the normal `__data` cache) for Objects.  Since the temp
         * cache is cleared at the end of a turn, this implementation allows
         * side-effects of deep object changes to be processed by re-setting the
         * same object (using the temp cache as an in-turn backstop to prevent
         * cycles due to 2-way notification).
         *
         * @param {string} property Property name
         * @param {*} value New property value
         * @param {*} old Previous property value
         * @return {boolean} Whether the property should be considered a change
         * @protected
         */
        _shouldPropertyChange(property, value, old) {
          return mutablePropertyChange(this, property, value, old, this.mutableData);
        }
      }

      return OptionalMutableData;

    });

    // Export for use by legacy behavior
    MutableData._mutablePropertyChange = mutablePropertyChange;

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    // Base class for HTMLTemplateElement extension that has property effects
    // machinery for propagating host properties to children. This is an ES5
    // class only because Babel (incorrectly) requires super() in the class
    // constructor even though no `this` is used and it returns an instance.
    let newInstance = null;

    /**
     * @constructor
     * @extends {HTMLTemplateElement}
     * @private
     */
    function HTMLTemplateElementExtension() { return newInstance; }
    HTMLTemplateElementExtension.prototype = Object.create(HTMLTemplateElement.prototype, {
      constructor: {
        value: HTMLTemplateElementExtension,
        writable: true
      }
    });

    /**
     * @constructor
     * @implements {Polymer_PropertyEffects}
     * @extends {HTMLTemplateElementExtension}
     * @private
     */
    const DataTemplate = PropertyEffects(HTMLTemplateElementExtension);

    /**
     * @constructor
     * @implements {Polymer_MutableData}
     * @extends {DataTemplate}
     * @private
     */
    const MutableDataTemplate = MutableData(DataTemplate);

    // Applies a DataTemplate subclass to a <template> instance
    function upgradeTemplate(template, constructor) {
      newInstance = template;
      Object.setPrototypeOf(template, constructor.prototype);
      new constructor();
      newInstance = null;
    }

    /**
     * Base class for TemplateInstance.
     * @constructor
     * @extends {HTMLElement}
     * @implements {Polymer_PropertyEffects}
     * @private
     */
    const templateInstanceBase = PropertyEffects(class {});

    function showHideChildren(hide, children) {
      for (let i=0; i<children.length; i++) {
        let n = children[i];
        // Ignore non-changes
        if (Boolean(hide) != Boolean(n.__hideTemplateChildren__)) {
          // clear and restore text
          if (n.nodeType === Node.TEXT_NODE) {
            if (hide) {
              n.__polymerTextContent__ = n.textContent;
              n.textContent = '';
            } else {
              n.textContent = n.__polymerTextContent__;
            }
          // remove and replace slot
          } else if (n.localName === 'slot') {
            if (hide) {
              n.__polymerReplaced__ = document.createComment('hidden-slot');
              wrap(wrap(n).parentNode).replaceChild(n.__polymerReplaced__, n);
            } else {
              const replace = n.__polymerReplaced__;
              if (replace) {
                wrap(wrap(replace).parentNode).replaceChild(n, replace);
              }
            }
          }
          // hide and show nodes
          else if (n.style) {
            if (hide) {
              n.__polymerDisplay__ = n.style.display;
              n.style.display = 'none';
            } else {
              n.style.display = n.__polymerDisplay__;
            }
          }
        }
        n.__hideTemplateChildren__ = hide;
        if (n._showHideChildren) {
          n._showHideChildren(hide);
        }
      }
    }

    /**
     * @polymer
     * @customElement
     * @appliesMixin PropertyEffects
     * @unrestricted
     */
    class TemplateInstanceBase extends templateInstanceBase {
      constructor(props) {
        super();
        this._configureProperties(props);
        /** @type {!StampedTemplate} */
        this.root = this._stampTemplate(this.__dataHost);
        // Save list of stamped children
        let children = [];
        /** @suppress {invalidCasts} */
        this.children = /** @type {!NodeList} */ (children);
        // Polymer 1.x did not use `Polymer.dom` here so not bothering.
        for (let n = this.root.firstChild; n; n=n.nextSibling) {
          children.push(n);
          n.__templatizeInstance = this;
        }
        if (this.__templatizeOwner &&
          this.__templatizeOwner.__hideTemplateChildren__) {
          this._showHideChildren(true);
        }
        // Flush props only when props are passed if instance props exist
        // or when there isn't instance props.
        let options = this.__templatizeOptions;
        if ((props && options.instanceProps) || !options.instanceProps) {
          this._enableProperties();
        }
      }
      /**
       * Configure the given `props` by calling `_setPendingProperty`. Also
       * sets any properties stored in `__hostProps`.
       * @private
       * @param {Object} props Object of property name-value pairs to set.
       * @return {void}
       */
      _configureProperties(props) {
        let options = this.__templatizeOptions;
        if (options.forwardHostProp) {
          for (let hprop in this.__hostProps) {
            this._setPendingProperty(hprop, this.__dataHost['_host_' + hprop]);
          }
        }
        // Any instance props passed in the constructor will overwrite host props;
        // normally this would be a user error but we don't specifically filter them
        for (let iprop in props) {
          this._setPendingProperty(iprop, props[iprop]);
        }
      }
      /**
       * Forwards a host property to this instance.  This method should be
       * called on instances from the `options.forwardHostProp` callback
       * to propagate changes of host properties to each instance.
       *
       * Note this method enqueues the change, which are flushed as a batch.
       *
       * @param {string} prop Property or path name
       * @param {*} value Value of the property to forward
       * @return {void}
       */
      forwardHostProp(prop, value) {
        if (this._setPendingPropertyOrPath(prop, value, false, true)) {
          this.__dataHost._enqueueClient(this);
        }
      }

      /**
       * Override point for adding custom or simulated event handling.
       *
       * @override
       * @param {!Node} node Node to add event listener to
       * @param {string} eventName Name of event
       * @param {function(!Event):void} handler Listener function to add
       * @return {void}
       */
      _addEventListenerToNode(node, eventName, handler) {
        if (this._methodHost && this.__templatizeOptions.parentModel) {
          // If this instance should be considered a parent model, decorate
          // events this template instance as `model`
          this._methodHost._addEventListenerToNode(node, eventName, (e) => {
            e.model = this;
            handler(e);
          });
        } else {
          // Otherwise delegate to the template's host (which could be)
          // another template instance
          let templateHost = this.__dataHost.__dataHost;
          if (templateHost) {
            templateHost._addEventListenerToNode(node, eventName, handler);
          }
        }
      }
      /**
       * Shows or hides the template instance top level child elements. For
       * text nodes, `textContent` is removed while "hidden" and replaced when
       * "shown."
       * @param {boolean} hide Set to true to hide the children;
       * set to false to show them.
       * @return {void}
       * @protected
       */
      _showHideChildren(hide) {
        showHideChildren(hide, this.children);
      }
      /**
       * Overrides default property-effects implementation to intercept
       * textContent bindings while children are "hidden" and cache in
       * private storage for later retrieval.
       *
       * @override
       * @param {!Node} node The node to set a property on
       * @param {string} prop The property to set
       * @param {*} value The value to set
       * @return {void}
       * @protected
       */
      _setUnmanagedPropertyToNode(node, prop, value) {
        if (node.__hideTemplateChildren__ &&
            node.nodeType == Node.TEXT_NODE && prop == 'textContent') {
          node.__polymerTextContent__ = value;
        } else {
          super._setUnmanagedPropertyToNode(node, prop, value);
        }
      }
      /**
       * Find the parent model of this template instance.  The parent model
       * is either another templatize instance that had option `parentModel: true`,
       * or else the host element.
       *
       * @return {!Polymer_PropertyEffects} The parent model of this instance
       */
      get parentModel() {
        let model = this.__parentModel;
        if (!model) {
          let options;
          model = this;
          do {
            // A template instance's `__dataHost` is a <template>
            // `model.__dataHost.__dataHost` is the template's host
            model = model.__dataHost.__dataHost;
          } while ((options = model.__templatizeOptions) && !options.parentModel);
          this.__parentModel = model;
        }
        return model;
      }

      /**
       * Stub of HTMLElement's `dispatchEvent`, so that effects that may
       * dispatch events safely no-op.
       *
       * @param {Event} event Event to dispatch
       * @return {boolean} Always true.
       * @override
       */
       dispatchEvent(event) { // eslint-disable-line no-unused-vars
         return true;
      }
    }

    /**
     * @constructor
     * @extends {TemplateInstanceBase}
     * @implements {Polymer_MutableData}
     * @private
     */
    const MutableTemplateInstanceBase = MutableData(
        // This cast shouldn't be neccessary, but Closure doesn't understand that
        // TemplateInstanceBase is a constructor function.
        /** @type {function(new:TemplateInstanceBase)} */ (TemplateInstanceBase));

    function findMethodHost(template) {
      // Technically this should be the owner of the outermost template.
      // In shadow dom, this is always getRootNode().host, but we can
      // approximate this via cooperation with our dataHost always setting
      // `_methodHost` as long as there were bindings (or id's) on this
      // instance causing it to get a dataHost.
      let templateHost = template.__dataHost;
      return templateHost && templateHost._methodHost || templateHost;
    }

    /* eslint-disable valid-jsdoc */
    /**
     * @suppress {missingProperties} class.prototype is not defined for some reason
     */
    function createTemplatizerClass(template, templateInfo, options) {
      /**
       * @constructor
       * @extends {TemplateInstanceBase}
       */
      let templatizerBase = options.mutableData ?
        MutableTemplateInstanceBase : TemplateInstanceBase;

      // Affordance for global mixins onto TemplatizeInstance
      if (templatize.mixin) {
        templatizerBase = templatize.mixin(templatizerBase);
      }

      /**
       * Anonymous class created by the templatize
       * @constructor
       * @private
       */
      let klass = class extends templatizerBase { };
      /** @override */
      klass.prototype.__templatizeOptions = options;
      klass.prototype._bindTemplate(template);
      addNotifyEffects(klass, template, templateInfo, options);
      return klass;
    }

    /**
     * Adds propagate effects from the template to the template instance for
     * properties that the host binds to the template using the `_host_` prefix.
     *
     * @suppress {missingProperties} class.prototype is not defined for some reason
     */
    function addPropagateEffects(target, templateInfo, options, methodHost) {
      let userForwardHostProp = options.forwardHostProp;
      if (userForwardHostProp && templateInfo.hasHostProps) {
        // Under the `removeNestedTemplates` optimization, a custom element like
        // `dom-if` or `dom-repeat` can itself be treated as the "template"; this
        // flag is used to switch between upgrading a `<template>` to be a property
        // effects client vs. adding the effects directly to the custom element
        const isTemplate = target.localName == 'template';
        // Provide data API and property effects on memoized template class
        let klass = templateInfo.templatizeTemplateClass;
        if (!klass) {
          if (isTemplate) {
            /**
             * @constructor
             * @extends {DataTemplate}
             */
            let templatizedBase =
                options.mutableData ? MutableDataTemplate : DataTemplate;

            // NOTE: due to https://github.com/google/closure-compiler/issues/2928,
            // combining the next two lines into one assignment causes a spurious
            // type error.
            /** @private */
            class TemplatizedTemplate extends templatizedBase {}
            klass = templateInfo.templatizeTemplateClass = TemplatizedTemplate;
          } else {
            /**
             * @constructor
             * @extends {PolymerElement}
             */
            const templatizedBase = target.constructor;

            // Create a cached subclass of the base custom element class onto which
            // to put the template-specific propagate effects
            // NOTE: due to https://github.com/google/closure-compiler/issues/2928,
            // combining the next two lines into one assignment causes a spurious
            // type error.
            /** @private */
            class TemplatizedTemplateExtension extends templatizedBase {}
            klass = templateInfo.templatizeTemplateClass =
                TemplatizedTemplateExtension;
          }
          // Add template - >instances effects
          // and host <- template effects
          let hostProps = templateInfo.hostProps;
          for (let prop in hostProps) {
            klass.prototype._addPropertyEffect('_host_' + prop,
              klass.prototype.PROPERTY_EFFECT_TYPES.PROPAGATE,
              {fn: createForwardHostPropEffect(prop, userForwardHostProp)});
            klass.prototype._createNotifyingProperty('_host_' + prop);
          }
          if (legacyWarnings && methodHost) {
            warnOnUndeclaredProperties(templateInfo, options, methodHost);
          }
        }
        // Mix any pre-bound data into __data; no need to flush this to
        // instances since they pull from the template at instance-time
        if (target.__dataProto) {
          // Note, generally `__dataProto` could be chained, but it's guaranteed
          // to not be since this is a vanilla template we just added effects to
          Object.assign(target.__data, target.__dataProto);
        }
        if (isTemplate) {
          upgradeTemplate(target, klass);
          // Clear any pending data for performance
          target.__dataTemp = {};
          target.__dataPending = null;
          target.__dataOld = null;
          target._enableProperties();
        } else {
          // Swizzle the cached subclass prototype onto the custom element
          Object.setPrototypeOf(target, klass.prototype);
          // Check for any pre-bound instance host properties, and do the
          // instance property delete/assign dance for those (directly into data;
          // not need to go through accessor since they are pulled at instance time)
          const hostProps = templateInfo.hostProps;
          for (let prop in hostProps) {
            prop = '_host_' + prop;
            if (prop in target) {
              const val = target[prop];
              delete target[prop];
              target.__data[prop] = val;
            }
          }
        }
      }
    }
    /* eslint-enable valid-jsdoc */

    function createForwardHostPropEffect(hostProp, userForwardHostProp) {
      return function forwardHostProp(template, prop, props) {
        userForwardHostProp.call(template.__templatizeOwner,
          prop.substring('_host_'.length), props[prop]);
      };
    }

    function addNotifyEffects(klass, template, templateInfo, options) {
      let hostProps = templateInfo.hostProps || {};
      for (let iprop in options.instanceProps) {
        delete hostProps[iprop];
        let userNotifyInstanceProp = options.notifyInstanceProp;
        if (userNotifyInstanceProp) {
          klass.prototype._addPropertyEffect(iprop,
            klass.prototype.PROPERTY_EFFECT_TYPES.NOTIFY,
            {fn: createNotifyInstancePropEffect(iprop, userNotifyInstanceProp)});
        }
      }
      if (options.forwardHostProp && template.__dataHost) {
        for (let hprop in hostProps) {
          // As we're iterating hostProps in this function, note whether
          // there were any, for an optimization in addPropagateEffects
          if (!templateInfo.hasHostProps) {
            templateInfo.hasHostProps = true;
          }
          klass.prototype._addPropertyEffect(hprop,
            klass.prototype.PROPERTY_EFFECT_TYPES.NOTIFY,
            {fn: createNotifyHostPropEffect()});
        }
      }
    }

    function createNotifyInstancePropEffect(instProp, userNotifyInstanceProp) {
      return function notifyInstanceProp(inst, prop, props) {
        userNotifyInstanceProp.call(inst.__templatizeOwner,
          inst, prop, props[prop]);
      };
    }

    function createNotifyHostPropEffect() {
      return function notifyHostProp(inst, prop, props) {
        inst.__dataHost._setPendingPropertyOrPath('_host_' + prop, props[prop], true, true);
      };
    }


    /**
     * Returns an anonymous `PropertyEffects` class bound to the
     * `<template>` provided.  Instancing the class will result in the
     * template being stamped into a document fragment stored as the instance's
     * `root` property, after which it can be appended to the DOM.
     *
     * Templates may utilize all Polymer data-binding features as well as
     * declarative event listeners.  Event listeners and inline computing
     * functions in the template will be called on the host of the template.
     *
     * The constructor returned takes a single argument dictionary of initial
     * property values to propagate into template bindings.  Additionally
     * host properties can be forwarded in, and instance properties can be
     * notified out by providing optional callbacks in the `options` dictionary.
     *
     * Valid configuration in `options` are as follows:
     *
     * - `forwardHostProp(property, value)`: Called when a property referenced
     *   in the template changed on the template's host. As this library does
     *   not retain references to templates instanced by the user, it is the
     *   templatize owner's responsibility to forward host property changes into
     *   user-stamped instances.  The `instance.forwardHostProp(property, value)`
     *    method on the generated class should be called to forward host
     *   properties into the template to prevent unnecessary property-changed
     *   notifications. Any properties referenced in the template that are not
     *   defined in `instanceProps` will be notified up to the template's host
     *   automatically.
     * - `instanceProps`: Dictionary of property names that will be added
     *   to the instance by the templatize owner.  These properties shadow any
     *   host properties, and changes within the template to these properties
     *   will result in `notifyInstanceProp` being called.
     * - `mutableData`: When `true`, the generated class will skip strict
     *   dirty-checking for objects and arrays (always consider them to be
     *   "dirty").
     * - `notifyInstanceProp(instance, property, value)`: Called when
     *   an instance property changes.  Users may choose to call `notifyPath`
     *   on e.g. the owner to notify the change.
     * - `parentModel`: When `true`, events handled by declarative event listeners
     *   (`on-event="handler"`) will be decorated with a `model` property pointing
     *   to the template instance that stamped it.  It will also be returned
     *   from `instance.parentModel` in cases where template instance nesting
     *   causes an inner model to shadow an outer model.
     *
     * All callbacks are called bound to the `owner`. Any context
     * needed for the callbacks (such as references to `instances` stamped)
     * should be stored on the `owner` such that they can be retrieved via
     * `this`.
     *
     * When `options.forwardHostProp` is declared as an option, any properties
     * referenced in the template will be automatically forwarded from the host of
     * the `<template>` to instances, with the exception of any properties listed in
     * the `options.instanceProps` object.  `instanceProps` are assumed to be
     * managed by the owner of the instances, either passed into the constructor
     * or set after the fact.  Note, any properties passed into the constructor will
     * always be set to the instance (regardless of whether they would normally
     * be forwarded from the host).
     *
     * Note that `templatize()` can be run only once for a given `<template>`.
     * Further calls will result in an error. Also, there is a special
     * behavior if the template was duplicated through a mechanism such as
     * `<dom-repeat>` or `<test-fixture>`. In this case, all calls to
     * `templatize()` return the same class for all duplicates of a template.
     * The class returned from `templatize()` is generated only once using
     * the `options` from the first call. This means that any `options`
     * provided to subsequent calls will be ignored. Therefore, it is very
     * important not to close over any variables inside the callbacks. Also,
     * arrow functions must be avoided because they bind the outer `this`.
     * Inside the callbacks, any contextual information can be accessed
     * through `this`, which points to the `owner`.
     *
     * @param {!HTMLTemplateElement} template Template to templatize
     * @param {Polymer_PropertyEffects=} owner Owner of the template instances;
     *   any optional callbacks will be bound to this owner.
     * @param {Object=} options Options dictionary (see summary for details)
     * @return {function(new:TemplateInstanceBase, Object=)} Generated class bound
     *   to the template provided
     * @suppress {invalidCasts}
     */
    function templatize(template, owner, options) {
      // Under strictTemplatePolicy, the templatized element must be owned
      // by a (trusted) Polymer element, indicated by existence of _methodHost;
      // e.g. for dom-if & dom-repeat in main document, _methodHost is null
      if (strictTemplatePolicy && !findMethodHost(template)) {
        throw new Error('strictTemplatePolicy: template owner not trusted');
      }
      options = /** @type {!TemplatizeOptions} */(options || {});
      if (template.__templatizeOwner) {
        throw new Error('A <template> can only be templatized once');
      }
      template.__templatizeOwner = owner;
      const ctor = owner ? owner.constructor : TemplateInstanceBase;
      let templateInfo = ctor._parseTemplate(template);
      // Get memoized base class for the prototypical template, which
      // includes property effects for binding template & forwarding
      /**
       * @constructor
       * @extends {TemplateInstanceBase}
       */
      let baseClass = templateInfo.templatizeInstanceClass;
      if (!baseClass) {
        baseClass = createTemplatizerClass(template, templateInfo, options);
        templateInfo.templatizeInstanceClass = baseClass;
      }
      const methodHost = findMethodHost(template);
      // Host property forwarding must be installed onto template instance
      addPropagateEffects(template, templateInfo, options, methodHost);
      // Subclass base class and add reference for this specific template
      /** @private */
      let klass = class TemplateInstance extends baseClass {};
      /** @override */
      klass.prototype._methodHost = methodHost;
      /** @override */
      klass.prototype.__dataHost = /** @type {!DataTemplate} */ (template);
      /** @override */
      klass.prototype.__templatizeOwner = /** @type {!Object} */ (owner);
      /** @override */
      klass.prototype.__hostProps = templateInfo.hostProps;
      klass = /** @type {function(new:TemplateInstanceBase)} */(klass); //eslint-disable-line no-self-assign
      return klass;
    }

    function warnOnUndeclaredProperties(templateInfo, options, methodHost) {
      const declaredProps = methodHost.constructor._properties;
      const {propertyEffects} = templateInfo;
      const {instanceProps} = options;
      for (let prop in propertyEffects) {
        // Ensure properties with template effects are declared on the outermost
        // host (`methodHost`), unless they are instance props or static functions
        if (!declaredProps[prop] && !(instanceProps && instanceProps[prop])) {
          const effects = propertyEffects[prop];
          for (let i=0; i<effects.length; i++) {
            const {part} = effects[i].info;
            if (!(part.signature && part.signature.static)) {
              console.warn(`Property '${prop}' used in template but not ` +
                `declared in 'properties'; attribute will not be observed.`);
              break;
            }
          }
        }
      }
    }

    /**
     * Returns the template "model" associated with a given element, which
     * serves as the binding scope for the template instance the element is
     * contained in. A template model is an instance of
     * `TemplateInstanceBase`, and should be used to manipulate data
     * associated with this template instance.
     *
     * Example:
     *
     *   let model = modelForElement(el);
     *   if (model.index < 10) {
     *     model.set('item.checked', true);
     *   }
     *
     * @param {HTMLElement} template The model will be returned for
     *   elements stamped from this template (accepts either an HTMLTemplateElement)
     *   or a `<dom-if>`/`<dom-repeat>` element when using `removeNestedTemplates`
     *   optimization.
     * @param {Node=} node Node for which to return a template model.
     * @return {TemplateInstanceBase} Template instance representing the
     *   binding scope for the element
     */
    function modelForElement(template, node) {
      let model;
      while (node) {
        // An element with a __templatizeInstance marks the top boundary
        // of a scope; walk up until we find one, and then ensure that
        // its __dataHost matches `this`, meaning this dom-repeat stamped it
        if ((model = node.__dataHost ? node : node.__templatizeInstance)) {
          // Found an element stamped by another template; keep walking up
          // from its __dataHost
          if (model.__dataHost != template) {
            node = model.__dataHost;
          } else {
            return model;
          }
        } else {
          // Still in a template scope, keep going up until
          // a __templatizeInstance is found
          node = wrap(node).parentNode;
        }
      }
      return null;
    }

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    /**
     * The `Templatizer` behavior adds methods to generate instances of
     * templates that are each managed by an anonymous `PropertyEffects`
     * instance where data-bindings in the stamped template content are bound to
     * accessors on itself.
     *
     * This behavior is provided in Polymer 2.x-3.x as a hybrid-element convenience
     * only.  For non-hybrid usage, the `Templatize` library
     * should be used instead.
     *
     * Example:
     *
     *     import {dom} from '@polymer/polymer/lib/legacy/polymer.dom.js';
     *     // Get a template from somewhere, e.g. light DOM
     *     let template = this.querySelector('template');
     *     // Prepare the template
     *     this.templatize(template);
     *     // Instance the template with an initial data model
     *     let instance = this.stamp({myProp: 'initial'});
     *     // Insert the instance's DOM somewhere, e.g. light DOM
     *     dom(this).appendChild(instance.root);
     *     // Changing a property on the instance will propagate to bindings
     *     // in the template
     *     instance.myProp = 'new value';
     *
     * Users of `Templatizer` may need to implement the following abstract
     * API's to determine how properties and paths from the host should be
     * forwarded into to instances:
     *
     *     _forwardHostPropV2: function(prop, value)
     *
     * Likewise, users may implement these additional abstract API's to determine
     * how instance-specific properties that change on the instance should be
     * forwarded out to the host, if necessary.
     *
     *     _notifyInstancePropV2: function(inst, prop, value)
     *
     * In order to determine which properties are instance-specific and require
     * custom notification via `_notifyInstanceProp`, define an `_instanceProps`
     * object containing keys for each instance prop, for example:
     *
     *     _instanceProps: {
     *       item: true,
     *       index: true
     *     }
     *
     * Any properties used in the template that are not defined in _instanceProp
     * will be forwarded out to the Templatize `owner` automatically.
     *
     * Users may also implement the following abstract function to show or
     * hide any DOM generated using `stamp`:
     *
     *     _showHideChildren: function(shouldHide)
     *
     * Note that some callbacks are suffixed with `V2` in the Polymer 2.x behavior
     * as the implementations will need to differ from the callbacks required
     * by the 1.x Templatizer API due to changes in the `TemplateInstance` API
     * between versions 1.x and 2.x.
     *
     * @polymerBehavior
     */
    const Templatizer = {

      /**
       * Generates an anonymous `TemplateInstance` class (stored as `this.ctor`)
       * for the provided template.  This method should be called once per
       * template to prepare an element for stamping the template, followed
       * by `stamp` to create new instances of the template.
       *
       * @param {!HTMLTemplateElement} template Template to prepare
       * @param {boolean=} mutableData When `true`, the generated class will skip
       *   strict dirty-checking for objects and arrays (always consider them to
       *   be "dirty"). Defaults to false.
       * @return {void}
       * @this {TemplatizerUser}
       */
      templatize(template, mutableData) {
        this._templatizerTemplate = template;
        this.ctor =
            templatize(template, /** @type {!Polymer_PropertyEffects} */ (this), {
              mutableData: Boolean(mutableData),
              parentModel: this._parentModel,
              instanceProps: this._instanceProps,
              forwardHostProp: this._forwardHostPropV2,
              notifyInstanceProp: this._notifyInstancePropV2
            });
      },

      /**
       * Creates an instance of the template prepared by `templatize`.  The object
       * returned is an instance of the anonymous class generated by `templatize`
       * whose `root` property is a document fragment containing newly cloned
       * template content, and which has property accessors corresponding to
       * properties referenced in template bindings.
       *
       * @param {Object=} model Object containing initial property values to
       *   populate into the template bindings.
       * @return {TemplateInstanceBase} Returns the created instance of
       * the template prepared by `templatize`.
       * @this {TemplatizerUser}
       */
      stamp(model) {
        return new this.ctor(model);
      },

      /**
       * Returns the template "model" (`TemplateInstance`) associated with
       * a given element, which serves as the binding scope for the template
       * instance the element is contained in.  A template model should be used
       * to manipulate data associated with this template instance.
       *
       * @param {HTMLElement} el Element for which to return a template model.
       * @return {TemplateInstanceBase} Model representing the binding scope for
       *   the element.
       * @this {TemplatizerUser}
       */
      modelForElement(el) {
        return modelForElement(this._templatizerTemplate, el);
      }
    };

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    let elementsHidden = false;

    /**
     * @return {boolean} True if elements will be hidden globally
     */
    function hideElementsGlobally() {
      if (legacyOptimizations && !useShadow) {
        if (!elementsHidden) {
          elementsHidden = true;
          const style = document.createElement('style');
          style.textContent = 'dom-bind,dom-if,dom-repeat{display:none;}';
          document.head.appendChild(style);
        }
        return true;
      }
      return false;
    }

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    /**
     * @constructor
     * @extends {HTMLElement}
     * @implements {Polymer_PropertyEffects}
     * @implements {Polymer_OptionalMutableData}
     * @implements {Polymer_GestureEventListeners}
     * @private
     */
    const domBindBase =
      GestureEventListeners(
        OptionalMutableData(
          PropertyEffects(HTMLElement)));

    /**
     * Custom element to allow using Polymer's template features (data binding,
     * declarative event listeners, etc.) in the main document without defining
     * a new custom element.
     *
     * `<template>` tags utilizing bindings may be wrapped with the `<dom-bind>`
     * element, which will immediately stamp the wrapped template into the main
     * document and bind elements to the `dom-bind` element itself as the
     * binding scope.
     *
     * @polymer
     * @customElement
     * @appliesMixin PropertyEffects
     * @appliesMixin OptionalMutableData
     * @appliesMixin GestureEventListeners
     * @extends {domBindBase}
     * @summary Custom element to allow using Polymer's template features (data
     *   binding, declarative event listeners, etc.) in the main document.
     */
    class DomBind extends domBindBase {

      static get observedAttributes() { return ['mutable-data']; }

      constructor() {
        super();
        if (strictTemplatePolicy) {
          throw new Error(`strictTemplatePolicy: dom-bind not allowed`);
        }
        this.root = null;
        this.$ = null;
        this.__children = null;
      }

      /* eslint-disable no-unused-vars */
      /**
       * @override
       * @param {string} name Name of attribute that changed
       * @param {?string} old Old attribute value
       * @param {?string} value New attribute value
       * @param {?string} namespace Attribute namespace.
       * @return {void}
       */
      attributeChangedCallback(name, old, value, namespace) {
        // assumes only one observed attribute
        this.mutableData = true;
      }

      /**
       * @override
       * @return {void}
       */
      connectedCallback() {
        if (!hideElementsGlobally()) {
          this.style.display = 'none';
        }
        this.render();
      }

      /**
       * @override
       * @return {void}
       */
      disconnectedCallback() {
        this.__removeChildren();
      }

      __insertChildren() {
        wrap(wrap(this).parentNode).insertBefore(this.root, this);
      }

      __removeChildren() {
        if (this.__children) {
          for (let i=0; i<this.__children.length; i++) {
            this.root.appendChild(this.__children[i]);
          }
        }
      }

      /**
       * Forces the element to render its content. This is typically only
       * necessary to call if HTMLImports with the async attribute are used.
       * @return {void}
       */
      render() {
        let template;
        if (!this.__children) {
          template = /** @type {?HTMLTemplateElement} */(template || this.querySelector('template'));
          if (!template) {
            // Wait until childList changes and template should be there by then
            let observer = new MutationObserver(() => {
              template = /** @type {HTMLTemplateElement} */(this.querySelector('template'));
              if (template) {
                observer.disconnect();
                this.render();
              } else {
                throw new Error('dom-bind requires a <template> child');
              }
            });
            observer.observe(this, {childList: true});
            return;
          }
          this.root = this._stampTemplate(
            /** @type {!HTMLTemplateElement} */(template));
          this.$ = this.root.$;
          this.__children = [];
          for (let n=this.root.firstChild; n; n=n.nextSibling) {
            this.__children[this.__children.length] = n;
          }
          this._enableProperties();
        }
        this.__insertChildren();
        this.dispatchEvent(new CustomEvent('dom-change', {
          bubbles: true,
          composed: true
        }));
      }

    }

    customElements.define('dom-bind', DomBind);

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    /**
     * @constructor
     * @implements {Polymer_OptionalMutableData}
     * @extends {PolymerElement}
     * @private
     */
    const domRepeatBase = OptionalMutableData(PolymerElement);

    /**
     * The `<dom-repeat>` element will automatically stamp and binds one instance
     * of template content to each object in a user-provided array.
     * `dom-repeat` accepts an `items` property, and one instance of the template
     * is stamped for each item into the DOM at the location of the `dom-repeat`
     * element.  The `item` property will be set on each instance's binding
     * scope, thus templates should bind to sub-properties of `item`.
     *
     * Example:
     *
     * ```html
     * <dom-module id="employee-list">
     *
     *   <template>
     *
     *     <div> Employee list: </div>
     *     <dom-repeat items="{{employees}}">
     *       <template>
     *         <div>First name: <span>{{item.first}}</span></div>
     *         <div>Last name: <span>{{item.last}}</span></div>
     *       </template>
     *     </dom-repeat>
     *
     *   </template>
     *
     * </dom-module>
     * ```
     *
     * With the following custom element definition:
     *
     * ```js
     * class EmployeeList extends PolymerElement {
     *   static get is() { return 'employee-list'; }
     *   static get properties() {
     *     return {
     *       employees: {
     *         value() {
     *           return [
     *             {first: 'Bob', last: 'Smith'},
     *             {first: 'Sally', last: 'Johnson'},
     *             ...
     *           ];
     *         }
     *       }
     *     };
     *   }
     * }
     * ```
     *
     * Notifications for changes to items sub-properties will be forwarded to template
     * instances, which will update via the normal structured data notification system.
     *
     * Mutations to the `items` array itself should be made using the Array
     * mutation API's on the PropertyEffects mixin (`push`, `pop`, `splice`,
     * `shift`, `unshift`), and template instances will be kept in sync with the
     * data in the array.
     *
     * Events caught by event handlers within the `dom-repeat` template will be
     * decorated with a `model` property, which represents the binding scope for
     * each template instance.  The model should be used to manipulate data on the
     * instance, for example `event.model.set('item.checked', true);`.
     *
     * Alternatively, the model for a template instance for an element stamped by
     * a `dom-repeat` can be obtained using the `modelForElement` API on the
     * `dom-repeat` that stamped it, for example
     * `this.$.domRepeat.modelForElement(event.target).set('item.checked', true);`.
     * This may be useful for manipulating instance data of event targets obtained
     * by event handlers on parents of the `dom-repeat` (event delegation).
     *
     * A view-specific filter/sort may be applied to each `dom-repeat` by supplying a
     * `filter` and/or `sort` property.  This may be a string that names a function on
     * the host, or a function may be assigned to the property directly.  The functions
     * should implemented following the standard `Array` filter/sort API.
     *
     * In order to re-run the filter or sort functions based on changes to sub-fields
     * of `items`, the `observe` property may be set as a space-separated list of
     * `item` sub-fields that should cause a re-filter/sort when modified.  If
     * the filter or sort function depends on properties not contained in `items`,
     * the user should observe changes to those properties and call `render` to update
     * the view based on the dependency change.
     *
     * For example, for an `dom-repeat` with a filter of the following:
     *
     * ```js
     * isEngineer(item) {
     *   return item.type == 'engineer' || item.manager.type == 'engineer';
     * }
     * ```
     *
     * Then the `observe` property should be configured as follows:
     *
     * ```html
     * <dom-repeat items="{{employees}}" filter="isEngineer" observe="type manager.type">
     * ```
     *
     * @customElement
     * @polymer
     * @extends {domRepeatBase}
     * @appliesMixin OptionalMutableData
     * @summary Custom element for stamping instance of a template bound to
     *   items in an array.
     */
    class DomRepeat extends domRepeatBase {

      // Not needed to find template; can be removed once the analyzer
      // can find the tag name from customElements.define call
      static get is() { return 'dom-repeat'; }

      static get template() { return null; }

      static get properties() {

        /**
         * Fired whenever DOM is added or removed by this template (by
         * default, rendering occurs lazily).  To force immediate rendering, call
         * `render`.
         *
         * @event dom-change
         */
        return {

          /**
           * An array containing items determining how many instances of the template
           * to stamp and that that each template instance should bind to.
           */
          items: {
            type: Array
          },

          /**
           * The name of the variable to add to the binding scope for the array
           * element associated with a given template instance.
           */
          as: {
            type: String,
            value: 'item'
          },

          /**
           * The name of the variable to add to the binding scope with the index
           * of the instance in the sorted and filtered list of rendered items.
           * Note, for the index in the `this.items` array, use the value of the
           * `itemsIndexAs` property.
           */
          indexAs: {
            type: String,
            value: 'index'
          },

          /**
           * The name of the variable to add to the binding scope with the index
           * of the instance in the `this.items` array. Note, for the index of
           * this instance in the sorted and filtered list of rendered items,
           * use the value of the `indexAs` property.
           */
          itemsIndexAs: {
            type: String,
            value: 'itemsIndex'
          },

          /**
           * A function that should determine the sort order of the items.  This
           * property should either be provided as a string, indicating a method
           * name on the element's host, or else be an actual function.  The
           * function should match the sort function passed to `Array.sort`.
           * Using a sort function has no effect on the underlying `items` array.
           */
          sort: {
            type: Function,
            observer: '__sortChanged'
          },

          /**
           * A function that can be used to filter items out of the view.  This
           * property should either be provided as a string, indicating a method
           * name on the element's host, or else be an actual function.  The
           * function should match the sort function passed to `Array.filter`.
           * Using a filter function has no effect on the underlying `items` array.
           */
          filter: {
            type: Function,
            observer: '__filterChanged'
          },

          /**
           * When using a `filter` or `sort` function, the `observe` property
           * should be set to a space-separated list of the names of item
           * sub-fields that should trigger a re-sort or re-filter when changed.
           * These should generally be fields of `item` that the sort or filter
           * function depends on.
           */
          observe: {
            type: String,
            observer: '__observeChanged'
          },

          /**
           * When using a `filter` or `sort` function, the `delay` property
           * determines a debounce time in ms after a change to observed item
           * properties that must pass before the filter or sort is re-run.
           * This is useful in rate-limiting shuffling of the view when
           * item changes may be frequent.
           */
          delay: Number,

          /**
           * Count of currently rendered items after `filter` (if any) has been applied.
           * If "chunking mode" is enabled, `renderedItemCount` is updated each time a
           * set of template instances is rendered.
           *
           */
          renderedItemCount: {
            type: Number,
            notify: !suppressTemplateNotifications,
            readOnly: true
          },

          /**
           * When greater than zero, defines an initial count of template instances
           * to render after setting the `items` array, before the next paint, and
           * puts the `dom-repeat` into "chunking mode".  The remaining items (and
           * any future items as a result of pushing onto the array) will be created
           * and rendered incrementally at each animation frame thereof until all
           * instances have been rendered.
           */
          initialCount: {
            type: Number
          },

          /**
           * When `initialCount` is used, this property defines a frame rate (in
           * fps) to target by throttling the number of instances rendered each
           * frame to not exceed the budget for the target frame rate.  The
           * framerate is effectively the number of `requestAnimationFrame`s that
           * it tries to allow to actually fire in a given second. It does this
           * by measuring the time between `rAF`s and continuously adjusting the
           * number of items created each `rAF` to maintain the target framerate.
           * Setting this to a higher number allows lower latency and higher
           * throughput for event handlers and other tasks, but results in a
           * longer time for the remaining items to complete rendering.
           */
          targetFramerate: {
            type: Number,
            value: 20
          },

          _targetFrameTime: {
            type: Number,
            computed: '__computeFrameTime(targetFramerate)'
          },

          /**
           * When the global `suppressTemplateNotifications` setting is used, setting
           * `notifyDomChange: true` will enable firing `dom-change` events on this
           * element.
           */
          notifyDomChange: {
            type: Boolean
          },

          /**
           * When chunking is enabled via `initialCount` and the `items` array is
           * set to a new array, this flag controls whether the previously rendered
           * instances are reused or not.
           *
           * When `true`, any previously rendered template instances are updated in
           * place to their new item values synchronously in one shot, and then any
           * further items (if any) are chunked out.  When `false`, the list is
           * returned back to its `initialCount` (any instances over the initial
           * count are discarded) and the remainder of the list is chunked back in.
           * Set this to `true` to avoid re-creating the list and losing scroll
           * position, although note that when changing the list to completely
           * different data the render thread will be blocked until all existing
           * instances are updated to their new data.
           */
          reuseChunkedInstances: {
            type: Boolean
          }

        };

      }

      static get observers() {
        return [ '__itemsChanged(items.*)' ];
      }

      constructor() {
        super();
        this.__instances = [];
        this.__renderDebouncer = null;
        this.__itemsIdxToInstIdx = {};
        this.__chunkCount = null;
        this.__renderStartTime = null;
        this.__itemsArrayChanged = false;
        this.__shouldMeasureChunk = false;
        this.__shouldContinueChunking = false;
        this.__chunkingId = 0;
        this.__sortFn = null;
        this.__filterFn = null;
        this.__observePaths = null;
        /** @type {?function(new:TemplateInstanceBase, Object=)} */
        this.__ctor = null;
        this.__isDetached = true;
        this.template = null;
        /** @type {TemplateInfo} */
        this._templateInfo;
      }

      /**
       * @override
       * @return {void}
       */
      disconnectedCallback() {
        super.disconnectedCallback();
        this.__isDetached = true;
        for (let i=0; i<this.__instances.length; i++) {
          this.__detachInstance(i);
        }
      }

      /**
       * @override
       * @return {void}
       */
      connectedCallback() {
        super.connectedCallback();
        if (!hideElementsGlobally()) {
          this.style.display = 'none';
        }
        // only perform attachment if the element was previously detached.
        if (this.__isDetached) {
          this.__isDetached = false;
          let wrappedParent = wrap(wrap(this).parentNode);
          for (let i=0; i<this.__instances.length; i++) {
            this.__attachInstance(i, wrappedParent);
          }
        }
      }

      __ensureTemplatized() {
        // Templatizing (generating the instance constructor) needs to wait
        // until ready, since won't have its template content handed back to
        // it until then
        if (!this.__ctor) {
          // When `removeNestedTemplates` is true, the "template" is the element
          // itself, which has been given a `_templateInfo` property
          const thisAsTemplate = /** @type {!HTMLTemplateElement} */ (
              /** @type {!HTMLElement} */ (this));
          let template = this.template = thisAsTemplate._templateInfo ?
              thisAsTemplate :
              /** @type {!HTMLTemplateElement} */ (this.querySelector('template'));
          if (!template) {
            // Wait until childList changes and template should be there by then
            let observer = new MutationObserver(() => {
              if (this.querySelector('template')) {
                observer.disconnect();
                this.__render();
              } else {
                throw new Error('dom-repeat requires a <template> child');
              }
            });
            observer.observe(this, {childList: true});
            return false;
          }
          // Template instance props that should be excluded from forwarding
          let instanceProps = {};
          instanceProps[this.as] = true;
          instanceProps[this.indexAs] = true;
          instanceProps[this.itemsIndexAs] = true;
          this.__ctor = templatize(template, this, {
            mutableData: this.mutableData,
            parentModel: true,
            instanceProps: instanceProps,
            /**
             * @this {DomRepeat}
             * @param {string} prop Property to set
             * @param {*} value Value to set property to
             */
            forwardHostProp: function(prop, value) {
              let i$ = this.__instances;
              for (let i=0, inst; (i<i$.length) && (inst=i$[i]); i++) {
                inst.forwardHostProp(prop, value);
              }
            },
            /**
             * @this {DomRepeat}
             * @param {Object} inst Instance to notify
             * @param {string} prop Property to notify
             * @param {*} value Value to notify
             */
            notifyInstanceProp: function(inst, prop, value) {
              if (matches(this.as, prop)) {
                let idx = inst[this.itemsIndexAs];
                if (prop == this.as) {
                  this.items[idx] = value;
                }
                let path = translate(this.as, `${JSCompiler_renameProperty('items', this)}.${idx}`, prop);
                this.notifyPath(path, value);
              }
            }
          });
        }
        return true;
      }

      __getMethodHost() {
        // Technically this should be the owner of the outermost template.
        // In shadow dom, this is always getRootNode().host, but we can
        // approximate this via cooperation with our dataHost always setting
        // `_methodHost` as long as there were bindings (or id's) on this
        // instance causing it to get a dataHost.
        return this.__dataHost._methodHost || this.__dataHost;
      }

      __functionFromPropertyValue(functionOrMethodName) {
        if (typeof functionOrMethodName === 'string') {
          let methodName = functionOrMethodName;
          let obj = this.__getMethodHost();
          return function() { return obj[methodName].apply(obj, arguments); };
        }

        return functionOrMethodName;
      }

      __sortChanged(sort) {
        this.__sortFn = this.__functionFromPropertyValue(sort);
        if (this.items) { this.__debounceRender(this.__render); }
      }

      __filterChanged(filter) {
        this.__filterFn = this.__functionFromPropertyValue(filter);
        if (this.items) { this.__debounceRender(this.__render); }
      }

      __computeFrameTime(rate) {
        return Math.ceil(1000/rate);
      }

      __observeChanged() {
        this.__observePaths = this.observe &&
          this.observe.replace('.*', '.').split(' ');
      }

      __handleObservedPaths(path) {
        // Handle cases where path changes should cause a re-sort/filter
        if (this.__sortFn || this.__filterFn) {
          if (!path) {
            // Always re-render if the item itself changed
            this.__debounceRender(this.__render, this.delay);
          } else if (this.__observePaths) {
            // Otherwise, re-render if the path changed matches an observed path
            let paths = this.__observePaths;
            for (let i=0; i<paths.length; i++) {
              if (path.indexOf(paths[i]) === 0) {
                this.__debounceRender(this.__render, this.delay);
              }
            }
          }
        }
      }

      __itemsChanged(change) {
        if (this.items && !Array.isArray(this.items)) {
          console.warn('dom-repeat expected array for `items`, found', this.items);
        }
        // If path was to an item (e.g. 'items.3' or 'items.3.foo'), forward the
        // path to that instance synchronously (returns false for non-item paths)
        if (!this.__handleItemPath(change.path, change.value)) {
          // Otherwise, the array was reset ('items') or spliced ('items.splices'),
          // so queue a render.  Restart chunking when the items changed (for
          // backward compatibility), unless `reuseChunkedInstances` option is set
          if (change.path === 'items') {
            this.__itemsArrayChanged = true;
          }
          this.__debounceRender(this.__render);
        }
      }

      /**
       * @param {function(this:DomRepeat)} fn Function to debounce.
       * @param {number=} delay Delay in ms to debounce by.
       */
      __debounceRender(fn, delay = 0) {
        this.__renderDebouncer = Debouncer.debounce(
              this.__renderDebouncer
            , delay > 0 ? timeOut.after(delay) : microTask
            , fn.bind(this));
        enqueueDebouncer(this.__renderDebouncer);
      }

      /**
       * Forces the element to render its content. Normally rendering is
       * asynchronous to a provoking change. This is done for efficiency so
       * that multiple changes trigger only a single render. The render method
       * should be called if, for example, template rendering is required to
       * validate application state.
       * @return {void}
       */
      render() {
        // Queue this repeater, then flush all in order
        this.__debounceRender(this.__render);
        flush();
      }

      __render() {
        if (!this.__ensureTemplatized()) {
          // No template found yet
          return;
        }
        let items = this.items || [];
        // Sort and filter the items into a mapping array from instance->item
        const isntIdxToItemsIdx = this.__sortAndFilterItems(items);
        // If we're chunking, increase the limit if there are new instances to
        // create and schedule the next chunk
        const limit = this.__calculateLimit(isntIdxToItemsIdx.length);
        // Create, update, and/or remove instances
        this.__updateInstances(items, limit, isntIdxToItemsIdx);
        // If we're chunking, schedule a rAF task to measure/continue chunking.     
        // Do this before any notifying events (renderedItemCount & dom-change)
        // since those could modify items and enqueue a new full render which will
        // pre-empt this measurement.
        if (this.initialCount &&
           (this.__shouldMeasureChunk || this.__shouldContinueChunking)) {
          cancelAnimationFrame(this.__chunkingId);
          this.__chunkingId = requestAnimationFrame(() => this.__continueChunking());
        }
        // Set rendered item count
        this._setRenderedItemCount(this.__instances.length);
        // Notify users
        if (!suppressTemplateNotifications || this.notifyDomChange) {
          this.dispatchEvent(new CustomEvent('dom-change', {
            bubbles: true,
            composed: true
          }));
        }
      }

      __sortAndFilterItems(items) {
        // Generate array maping the instance index to the items array index
        let isntIdxToItemsIdx = new Array(items.length);
        for (let i=0; i<items.length; i++) {
          isntIdxToItemsIdx[i] = i;
        }
        // Apply user filter
        if (this.__filterFn) {
          isntIdxToItemsIdx = isntIdxToItemsIdx.filter((i, idx, array) =>
            this.__filterFn(items[i], idx, array));
        }
        // Apply user sort
        if (this.__sortFn) {
          isntIdxToItemsIdx.sort((a, b) => this.__sortFn(items[a], items[b]));
        }
        return isntIdxToItemsIdx;
      }

      __calculateLimit(filteredItemCount) {
        let limit = filteredItemCount;
        const currentCount = this.__instances.length;
        // When chunking, we increase the limit from the currently rendered count
        // by the chunk count that is re-calculated after each rAF (with special
        // cases for reseting the limit to initialCount after changing items)
        if (this.initialCount) {
          let newCount;
          if (!this.__chunkCount ||
            (this.__itemsArrayChanged && !this.reuseChunkedInstances)) {
            // Limit next render to the initial count
            limit = Math.min(filteredItemCount, this.initialCount);
            // Subtract off any existing instances to determine the number of
            // instances that will be created
            newCount = Math.max(limit - currentCount, 0);
            // Initialize the chunk size with how many items we're creating
            this.__chunkCount = newCount || 1;
          } else {
            // The number of new instances that will be created is based on the
            // existing instances, the new list size, and the chunk size
            newCount = Math.min(
              Math.max(filteredItemCount - currentCount, 0), 
              this.__chunkCount);
            // Update the limit based on how many new items we're making, limited
            // buy the total size of the list
            limit = Math.min(currentCount + newCount, filteredItemCount);
          }
          // Record some state about chunking for use in `__continueChunking`
          this.__shouldMeasureChunk = newCount === this.__chunkCount;
          this.__shouldContinueChunking = limit < filteredItemCount;
          this.__renderStartTime = performance.now();
        }
        this.__itemsArrayChanged = false;
        return limit;
      }

      __continueChunking() {
        // Simple auto chunkSize throttling algorithm based on feedback loop:
        // measure actual time between frames and scale chunk count by ratio of
        // target/actual frame time.  Only modify chunk size if our measurement
        // reflects the cost of a creating a full chunk's worth of instances; this
        // avoids scaling up the chunk size if we e.g. quickly re-rendered instances
        // in place
        if (this.__shouldMeasureChunk) {
          const renderTime = performance.now() - this.__renderStartTime;
          const ratio = this._targetFrameTime / renderTime;
          this.__chunkCount = Math.round(this.__chunkCount * ratio) || 1;
        }
        // Enqueue a new render if we haven't reached the full size of the list
        if (this.__shouldContinueChunking) {
          this.__debounceRender(this.__render);
        }
      }
      
      __updateInstances(items, limit, isntIdxToItemsIdx) {
        // items->inst map kept for item path forwarding
        const itemsIdxToInstIdx = this.__itemsIdxToInstIdx = {};
        let instIdx;
        // Generate instances and assign items
        for (instIdx=0; instIdx<limit; instIdx++) {
          let inst = this.__instances[instIdx];
          let itemIdx = isntIdxToItemsIdx[instIdx];
          let item = items[itemIdx];
          itemsIdxToInstIdx[itemIdx] = instIdx;
          if (inst) {
            inst._setPendingProperty(this.as, item);
            inst._setPendingProperty(this.indexAs, instIdx);
            inst._setPendingProperty(this.itemsIndexAs, itemIdx);
            inst._flushProperties();
          } else {
            this.__insertInstance(item, instIdx, itemIdx);
          }
        }
        // Remove any extra instances from previous state
        for (let i=this.__instances.length-1; i>=instIdx; i--) {
          this.__detachAndRemoveInstance(i);
        }
      }

      __detachInstance(idx) {
        let inst = this.__instances[idx];
        const wrappedRoot = wrap(inst.root);
        for (let i=0; i<inst.children.length; i++) {
          let el = inst.children[i];
          wrappedRoot.appendChild(el);
        }
        return inst;
      }

      __attachInstance(idx, parent) {
        let inst = this.__instances[idx];
        // Note, this is pre-wrapped as an optimization
        parent.insertBefore(inst.root, this);
      }

      __detachAndRemoveInstance(idx) {
        this.__detachInstance(idx);
        this.__instances.splice(idx, 1);
      }

      __stampInstance(item, instIdx, itemIdx) {
        let model = {};
        model[this.as] = item;
        model[this.indexAs] = instIdx;
        model[this.itemsIndexAs] = itemIdx;
        return new this.__ctor(model);
      }

      __insertInstance(item, instIdx, itemIdx) {
        const inst = this.__stampInstance(item, instIdx, itemIdx);
        let beforeRow = this.__instances[instIdx + 1];
        let beforeNode = beforeRow ? beforeRow.children[0] : this;
        wrap(wrap(this).parentNode).insertBefore(inst.root, beforeNode);
        this.__instances[instIdx] = inst;
        return inst;
      }

      // Implements extension point from Templatize mixin
      /**
       * Shows or hides the template instance top level child elements. For
       * text nodes, `textContent` is removed while "hidden" and replaced when
       * "shown."
       * @param {boolean} hidden Set to true to hide the children;
       * set to false to show them.
       * @return {void}
       * @protected
       */
      _showHideChildren(hidden) {
        for (let i=0; i<this.__instances.length; i++) {
          this.__instances[i]._showHideChildren(hidden);
        }
      }

      // Called as a side effect of a host items.<key>.<path> path change,
      // responsible for notifying item.<path> changes to inst for key
      __handleItemPath(path, value) {
        let itemsPath = path.slice(6); // 'items.'.length == 6
        let dot = itemsPath.indexOf('.');
        let itemsIdx = dot < 0 ? itemsPath : itemsPath.substring(0, dot);
        // If path was index into array...
        if (itemsIdx == parseInt(itemsIdx, 10)) {
          let itemSubPath = dot < 0 ? '' : itemsPath.substring(dot+1);
          // If the path is observed, it will trigger a full refresh
          this.__handleObservedPaths(itemSubPath);
          // Note, even if a rull refresh is triggered, always do the path
          // notification because unless mutableData is used for dom-repeat
          // and all elements in the instance subtree, a full refresh may
          // not trigger the proper update.
          let instIdx = this.__itemsIdxToInstIdx[itemsIdx];
          let inst = this.__instances[instIdx];
          if (inst) {
            let itemPath = this.as + (itemSubPath ? '.' + itemSubPath : '');
            // This is effectively `notifyPath`, but avoids some of the overhead
            // of the public API
            inst._setPendingPropertyOrPath(itemPath, value, false, true);
            inst._flushProperties();
          }
          return true;
        }
      }

      /**
       * Returns the item associated with a given element stamped by
       * this `dom-repeat`.
       *
       * Note, to modify sub-properties of the item,
       * `modelForElement(el).set('item.<sub-prop>', value)`
       * should be used.
       *
       * @param {!HTMLElement} el Element for which to return the item.
       * @return {*} Item associated with the element.
       */
      itemForElement(el) {
        let instance = this.modelForElement(el);
        return instance && instance[this.as];
      }

      /**
       * Returns the inst index for a given element stamped by this `dom-repeat`.
       * If `sort` is provided, the index will reflect the sorted order (rather
       * than the original array order).
       *
       * @param {!HTMLElement} el Element for which to return the index.
       * @return {?number} Row index associated with the element (note this may
       *   not correspond to the array index if a user `sort` is applied).
       */
      indexForElement(el) {
        let instance = this.modelForElement(el);
        return instance && instance[this.indexAs];
      }

      /**
       * Returns the template "model" associated with a given element, which
       * serves as the binding scope for the template instance the element is
       * contained in. A template model
       * should be used to manipulate data associated with this template instance.
       *
       * Example:
       *
       *   let model = modelForElement(el);
       *   if (model.index < 10) {
       *     model.set('item.checked', true);
       *   }
       *
       * @param {!HTMLElement} el Element for which to return a template model.
       * @return {TemplateInstanceBase} Model representing the binding scope for
       *   the element.
       */
      modelForElement(el) {
        return modelForElement(this.template, el);
      }

    }

    customElements.define(DomRepeat.is, DomRepeat);

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    /**
     * @customElement
     * @polymer
     * @extends PolymerElement
     * @summary Base class for dom-if element; subclassed into concrete
     *   implementation.
     */
    class DomIfBase extends PolymerElement {

      // Not needed to find template; can be removed once the analyzer
      // can find the tag name from customElements.define call
      static get is() { return 'dom-if'; }

      static get template() { return null; }

      static get properties() {

        return {

          /**
           * Fired whenever DOM is added or removed/hidden by this template (by
           * default, rendering occurs lazily).  To force immediate rendering, call
           * `render`.
           *
           * @event dom-change
           */

          /**
           * A boolean indicating whether this template should stamp.
           */
          if: {
            type: Boolean,
            observer: '__debounceRender'
          },

          /**
           * When true, elements will be removed from DOM and discarded when `if`
           * becomes false and re-created and added back to the DOM when `if`
           * becomes true.  By default, stamped elements will be hidden but left
           * in the DOM when `if` becomes false, which is generally results
           * in better performance.
           */
          restamp: {
            type: Boolean,
            observer: '__debounceRender'
          },

          /**
           * When the global `suppressTemplateNotifications` setting is used, setting
           * `notifyDomChange: true` will enable firing `dom-change` events on this
           * element.
           */
          notifyDomChange: {
            type: Boolean
          }
        };

      }

      constructor() {
        super();
        this.__renderDebouncer = null;
        this._lastIf = false;
        this.__hideTemplateChildren__ = false;
        /** @type {!HTMLTemplateElement|undefined} */
        this.__template;
        /** @type {!TemplateInfo|undefined} */
        this._templateInfo;
      }

      __debounceRender() {
        // Render is async for 2 reasons:
        // 1. To eliminate dom creation trashing if user code thrashes `if` in the
        //    same turn. This was more common in 1.x where a compound computed
        //    property could result in the result changing multiple times, but is
        //    mitigated to a large extent by batched property processing in 2.x.
        // 2. To avoid double object propagation when a bag including values bound
        //    to the `if` property as well as one or more hostProps could enqueue
        //    the <dom-if> to flush before the <template>'s host property
        //    forwarding. In that scenario creating an instance would result in
        //    the host props being set once, and then the enqueued changes on the
        //    template would set properties a second time, potentially causing an
        //    object to be set to an instance more than once.  Creating the
        //    instance async from flushing data ensures this doesn't happen. If
        //    we wanted a sync option in the future, simply having <dom-if> flush
        //    (or clear) its template's pending host properties before creating
        //    the instance would also avoid the problem.
        this.__renderDebouncer = Debouncer.debounce(
              this.__renderDebouncer
            , microTask
            , () => this.__render());
        enqueueDebouncer(this.__renderDebouncer);
      }

      /**
       * @override
       * @return {void}
       */
      disconnectedCallback() {
        super.disconnectedCallback();
        const parent = wrap(this).parentNode;
        if (!parent || (parent.nodeType == Node.DOCUMENT_FRAGMENT_NODE &&
            !wrap(parent).host)) {
          this.__teardownInstance();
        }
      }

      /**
       * @override
       * @return {void}
       */
      connectedCallback() {
        super.connectedCallback();
        if (!hideElementsGlobally()) {
          this.style.display = 'none';
        }
        if (this.if) {
          this.__debounceRender();
        }
      }

      /**
       * Ensures a template has been assigned to `this.__template`.  If it has not
       * yet been, it querySelectors for it in its children and if it does not yet
       * exist (e.g. in parser-generated case), opens a mutation observer and
       * waits for it to appear (returns false if it has not yet been found,
       * otherwise true).  In the `removeNestedTemplates` case, the "template" will
       * be the `dom-if` element itself.
       *
       * @return {boolean} True when a template has been found, false otherwise
       */
      __ensureTemplate() {
        if (!this.__template) {
          // When `removeNestedTemplates` is true, the "template" is the element
          // itself, which has been given a `_templateInfo` property
          const thisAsTemplate = /** @type {!HTMLTemplateElement} */ (
              /** @type {!HTMLElement} */ (this));
          let template = thisAsTemplate._templateInfo ?
              thisAsTemplate :
              /** @type {!HTMLTemplateElement} */
              (wrap(thisAsTemplate).querySelector('template'));
          if (!template) {
            // Wait until childList changes and template should be there by then
            let observer = new MutationObserver(() => {
              if (wrap(this).querySelector('template')) {
                observer.disconnect();
                this.__render();
              } else {
                throw new Error('dom-if requires a <template> child');
              }
            });
            observer.observe(this, {childList: true});
            return false;
          }
          this.__template = template;
        }
        return true;
      }

      /**
       * Ensures a an instance of the template has been created and inserted. This
       * method may return false if the template has not yet been found or if
       * there is no `parentNode` to insert the template into (in either case,
       * connection or the template-finding mutation observer firing will queue
       * another render, causing this method to be called again at a more
       * appropriate time).
       *
       * Subclasses should implement the following methods called here:
       * - `__hasInstance`
       * - `__createAndInsertInstance`
       * - `__getInstanceNodes`
       *
       * @return {boolean} True if the instance was created, false otherwise.
       */
      __ensureInstance() {
        let parentNode = wrap(this).parentNode;
        if (!this.__hasInstance()) {
          // Guard against element being detached while render was queued
          if (!parentNode) {
            return false;
          }
          // Find the template (when false, there was no template yet)
          if (!this.__ensureTemplate()) {
            return false;
          }
          this.__createAndInsertInstance(parentNode);
        } else {
          // Move instance children if necessary
          let children = this.__getInstanceNodes();
          if (children && children.length) {
            // Detect case where dom-if was re-attached in new position
            let lastChild = wrap(this).previousSibling;
            if (lastChild !== children[children.length-1]) {
              for (let i=0, n; (i<children.length) && (n=children[i]); i++) {
                wrap(parentNode).insertBefore(n, this);
              }
            }
          }
        }
        return true;
      }

      /**
       * Forces the element to render its content. Normally rendering is
       * asynchronous to a provoking change. This is done for efficiency so
       * that multiple changes trigger only a single render. The render method
       * should be called if, for example, template rendering is required to
       * validate application state.
       *
       * @return {void}
       */
      render() {
        flush();
      }

      /**
       * Performs the key rendering steps:
       * 1. Ensure a template instance has been stamped (when true)
       * 2. Remove the template instance (when false and restamp:true)
       * 3. Sync the hidden state of the instance nodes with the if/restamp state
       * 4. Fires the `dom-change` event when necessary
       *
       * @return {void}
       */
      __render() {
        if (this.if) {
          if (!this.__ensureInstance()) {
            // No template found yet
            return;
          }
        } else if (this.restamp) {
          this.__teardownInstance();
        }
        this._showHideChildren();
        if ((!suppressTemplateNotifications || this.notifyDomChange)
            && this.if != this._lastIf) {
          this.dispatchEvent(new CustomEvent('dom-change', {
            bubbles: true,
            composed: true
          }));
          this._lastIf = this.if;
        }
      }

      // Ideally these would be annotated as abstract methods in an abstract class,
      // but closure compiler is finnicky
      /* eslint-disable valid-jsdoc */
      /**
       * Abstract API to be implemented by subclass: Returns true if a template
       * instance has been created and inserted.
       *
       * @protected
       * @return {boolean} True when an instance has been created.
       */
      __hasInstance() { }

      /**
       * Abstract API to be implemented by subclass: Returns the child nodes stamped
       * from a template instance.
       *
       * @protected
       * @return {Array<Node>} Array of child nodes stamped from the template
       * instance.
       */
      __getInstanceNodes() { }

      /**
       * Abstract API to be implemented by subclass: Creates an instance of the
       * template and inserts it into the given parent node.
       *
       * @protected
       * @param {Node} parentNode The parent node to insert the instance into
       * @return {void}
       */
      __createAndInsertInstance(parentNode) { } // eslint-disable-line no-unused-vars

      /**
       * Abstract API to be implemented by subclass: Removes nodes created by an
       * instance of a template and any associated cleanup.
       *
       * @protected
       * @return {void}
       */
      __teardownInstance() { }

      /**
       * Abstract API to be implemented by subclass: Shows or hides any template
       * instance childNodes based on the `if` state of the element and its
       * `__hideTemplateChildren__` property.
       *
       * @protected
       * @return {void}
       */
      _showHideChildren() { }
      /* eslint-enable valid-jsdoc */
    }

    /**
     * The version of DomIf used when `fastDomIf` setting is in use, which is
     * optimized for first-render (but adds a tax to all subsequent property updates
     * on the host, whether they were used in a given `dom-if` or not).
     *
     * This implementation avoids use of `Templatizer`, which introduces a new scope
     * (a non-element PropertyEffects instance), which is not strictly necessary
     * since `dom-if` never introduces new properties to its scope (unlike
     * `dom-repeat`). Taking advantage of this fact, the `dom-if` reaches up to its
     * `__dataHost` and stamps the template directly from the host using the host's
     * runtime `_stampTemplate` API, which binds the property effects of the
     * template directly to the host. This both avoids the intermediary
     * `Templatizer` instance, but also avoids the need to bind host properties to
     * the `<template>` element and forward those into the template instance.
     *
     * In this version of `dom-if`, the `this.__instance` method is the
     * `DocumentFragment` returned from `_stampTemplate`, which also serves as the
     * handle for later removing it using the `_removeBoundDom` method.
     */
    class DomIfFast extends DomIfBase {

      constructor() {
        super();
        this.__instance = null;
        this.__syncInfo = null;
      }

      /**
       * Implementation of abstract API needed by DomIfBase.
       *
       * @override
       * @return {boolean} True when an instance has been created.
       */
      __hasInstance() {
        return Boolean(this.__instance);
      }

      /**
       * Implementation of abstract API needed by DomIfBase.
       *
       * @override
       * @return {Array<Node>} Array of child nodes stamped from the template
       * instance.
       */
      __getInstanceNodes() {
        return this.__instance.templateInfo.childNodes;
      }

      /**
       * Implementation of abstract API needed by DomIfBase.
       *
       * Stamps the template by calling `_stampTemplate` on the `__dataHost` of this
       * element and then inserts the resulting nodes into the given `parentNode`.
       *
       * @override
       * @param {Node} parentNode The parent node to insert the instance into
       * @return {void}
       */
      __createAndInsertInstance(parentNode) {
        const host = this.__dataHost || this;
        if (strictTemplatePolicy) {
          if (!this.__dataHost) {
            throw new Error('strictTemplatePolicy: template owner not trusted');
          }
        }
        // Pre-bind and link the template into the effects system
        const templateInfo = host._bindTemplate(
            /** @type {!HTMLTemplateElement} */ (this.__template), true);
        // Install runEffects hook that prevents running property effects
        // (and any nested template effects) when the `if` is false
        templateInfo.runEffects = (runEffects, changedProps, hasPaths) => {
          let syncInfo = this.__syncInfo;
          if (this.if) {
            // Mix any props that changed while the `if` was false into `changedProps`
            if (syncInfo) {
              // If there were properties received while the `if` was false, it is
              // important to sync the hidden state with the element _first_, so that
              // new bindings to e.g. `textContent` do not get stomped on by
              // pre-hidden values if `_showHideChildren` were to be called later at
              // the next render. Clearing `__invalidProps` here ensures
              // `_showHideChildren`'s call to `__syncHostProperties` no-ops, so
              // that we don't call `runEffects` more often than necessary.
              this.__syncInfo = null;
              this._showHideChildren();
              changedProps = Object.assign(syncInfo.changedProps, changedProps);
            }
            runEffects(changedProps, hasPaths);
          } else {
            // Accumulate any values changed while `if` was false, along with the
            // runEffects method to sync them, so that we can replay them once `if`
            // becomes true
            if (this.__instance) {
              if (!syncInfo) {
                syncInfo = this.__syncInfo = { runEffects, changedProps: {} };
              }
              if (hasPaths) {
                // Store root object of any paths; this will ensure direct bindings
                // like [[obj.foo]] bindings run after a `set('obj.foo', v)`, but
                // note that path notifications like `set('obj.foo.bar', v)` will
                // not propagate. Since batched path notifications are not
                // supported, we cannot simply accumulate path notifications. This
                // is equivalent to the non-fastDomIf case, which stores root(p) in
                // __invalidProps.
                for (const p in changedProps) {
                  const rootProp = root(p);
                  syncInfo.changedProps[rootProp] = this.__dataHost[rootProp];
                }
              } else {
                Object.assign(syncInfo.changedProps, changedProps);
              }
            }
          }
        };
        // Stamp the template, and set its DocumentFragment to the "instance"
        this.__instance = host._stampTemplate(
            /** @type {!HTMLTemplateElement} */ (this.__template), templateInfo);
        wrap(parentNode).insertBefore(this.__instance, this);
      }

      /**
       * Run effects for any properties that changed while the `if` was false.
       *
       * @return {void}
       */
      __syncHostProperties() {
        const syncInfo = this.__syncInfo;
        if (syncInfo) {
          this.__syncInfo = null;
          syncInfo.runEffects(syncInfo.changedProps, false);
        }
      }

      /**
       * Implementation of abstract API needed by DomIfBase.
       *
       * Remove the instance and any nodes it created.  Uses the `__dataHost`'s
       * runtime `_removeBoundDom` method.
       *
       * @override
       * @return {void}
       */
      __teardownInstance() {
        const host = this.__dataHost || this;
        if (this.__instance) {
          host._removeBoundDom(this.__instance);
          this.__instance = null;
          this.__syncInfo = null;
        }
      }

      /**
       * Implementation of abstract API needed by DomIfBase.
       *
       * Shows or hides the template instance top level child nodes. For
       * text nodes, `textContent` is removed while "hidden" and replaced when
       * "shown."
       *
       * @override
       * @return {void}
       * @protected
       * @suppress {visibility}
       */
      _showHideChildren() {
        const hidden = this.__hideTemplateChildren__ || !this.if;
        if (this.__instance && Boolean(this.__instance.__hidden) !== hidden) {
          this.__instance.__hidden = hidden;
          showHideChildren(hidden, this.__instance.templateInfo.childNodes);
        }
        if (!hidden) {
          this.__syncHostProperties();
        }
      }
    }

    /**
     * The "legacy" implementation of `dom-if`, implemented using `Templatizer`.
     *
     * In this version, `this.__instance` is the `TemplateInstance` returned
     * from the templatized constructor.
     */
    class DomIfLegacy extends DomIfBase {

      constructor() {
        super();
        this.__ctor = null;
        this.__instance = null;
        this.__invalidProps = null;
      }

      /**
       * Implementation of abstract API needed by DomIfBase.
       *
       * @override
       * @return {boolean} True when an instance has been created.
       */
      __hasInstance() {
        return Boolean(this.__instance);
      }

      /**
       * Implementation of abstract API needed by DomIfBase.
       *
       * @override
       * @return {Array<Node>} Array of child nodes stamped from the template
       * instance.
       */
      __getInstanceNodes() {
        return this.__instance.children;
      }

      /**
       * Implementation of abstract API needed by DomIfBase.
       *
       * Stamps the template by creating a new instance of the templatized
       * constructor (which is created lazily if it does not yet exist), and then
       * inserts its resulting `root` doc fragment into the given `parentNode`.
       *
       * @override
       * @param {Node} parentNode The parent node to insert the instance into
       * @return {void}
       */
      __createAndInsertInstance(parentNode) {
        // Ensure we have an instance constructor
        if (!this.__ctor) {
          this.__ctor = templatize(
              /** @type {!HTMLTemplateElement} */ (this.__template), this, {
                // dom-if templatizer instances require `mutable: true`, as
                // `__syncHostProperties` relies on that behavior to sync objects
                mutableData: true,
                /**
                 * @param {string} prop Property to forward
                 * @param {*} value Value of property
                 * @this {DomIfLegacy}
                 */
                forwardHostProp: function(prop, value) {
                  if (this.__instance) {
                    if (this.if) {
                      this.__instance.forwardHostProp(prop, value);
                    } else {
                      // If we have an instance but are squelching host property
                      // forwarding due to if being false, note the invalidated
                      // properties so `__syncHostProperties` can sync them the next
                      // time `if` becomes true
                      this.__invalidProps =
                          this.__invalidProps || Object.create(null);
                      this.__invalidProps[root(prop)] = true;
                    }
                  }
                }
              });
        }
        // Create and insert the instance
        this.__instance = new this.__ctor();
        wrap(parentNode).insertBefore(this.__instance.root, this);
      }

      /**
       * Implementation of abstract API needed by DomIfBase.
       *
       * Removes the instance and any nodes it created.
       *
       * @override
       * @return {void}
       */
      __teardownInstance() {
        if (this.__instance) {
          let c$ = this.__instance.children;
          if (c$ && c$.length) {
            // use first child parent, for case when dom-if may have been detached
            let parent = wrap(c$[0]).parentNode;
            // Instance children may be disconnected from parents when dom-if
            // detaches if a tree was innerHTML'ed
            if (parent) {
              parent = wrap(parent);
              for (let i=0, n; (i<c$.length) && (n=c$[i]); i++) {
                parent.removeChild(n);
              }
            }
          }
          this.__invalidProps = null;
          this.__instance = null;
        }
      }

      /**
       * Forwards any properties that changed while the `if` was false into the
       * template instance and flushes it.
       *
       * @return {void}
       */
      __syncHostProperties() {
        let props = this.__invalidProps;
        if (props) {
          this.__invalidProps = null;
          for (let prop in props) {
            this.__instance._setPendingProperty(prop, this.__dataHost[prop]);
          }
          this.__instance._flushProperties();
        }
      }

      /**
       * Implementation of abstract API needed by DomIfBase.
       *
       * Shows or hides the template instance top level child elements. For
       * text nodes, `textContent` is removed while "hidden" and replaced when
       * "shown."
       *
       * @override
       * @protected
       * @return {void}
       * @suppress {visibility}
       */
      _showHideChildren() {
        const hidden = this.__hideTemplateChildren__ || !this.if;
        if (this.__instance && Boolean(this.__instance.__hidden) !== hidden) {
          this.__instance.__hidden = hidden;
          this.__instance._showHideChildren(hidden);
        }
        if (!hidden) {
          this.__syncHostProperties();
        }
      }
    }

    /**
     * The `<dom-if>` element will stamp a light-dom `<template>` child when
     * the `if` property becomes truthy, and the template can use Polymer
     * data-binding and declarative event features when used in the context of
     * a Polymer element's template.
     *
     * When `if` becomes falsy, the stamped content is hidden but not
     * removed from dom. When `if` subsequently becomes truthy again, the content
     * is simply re-shown. This approach is used due to its favorable performance
     * characteristics: the expense of creating template content is paid only
     * once and lazily.
     *
     * Set the `restamp` property to true to force the stamped content to be
     * created / destroyed when the `if` condition changes.
     *
     * @customElement
     * @polymer
     * @extends DomIfBase
     * @constructor
     * @summary Custom element that conditionally stamps and hides or removes
     *   template content based on a boolean flag.
     */
    const DomIf = fastDomIf ? DomIfFast : DomIfLegacy;

    customElements.define(DomIf.is, DomIf);

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    /**
     * Element mixin for recording dynamic associations between item paths in a
     * master `items` array and a `selected` array such that path changes to the
     * master array (at the host) element or elsewhere via data-binding) are
     * correctly propagated to items in the selected array and vice-versa.
     *
     * The `items` property accepts an array of user data, and via the
     * `select(item)` and `deselect(item)` API, updates the `selected` property
     * which may be bound to other parts of the application, and any changes to
     * sub-fields of `selected` item(s) will be kept in sync with items in the
     * `items` array.  When `multi` is false, `selected` is a property
     * representing the last selected item.  When `multi` is true, `selected`
     * is an array of multiply selected items.
     *
     * @polymer
     * @mixinFunction
     * @appliesMixin ElementMixin
     * @summary Element mixin for recording dynamic associations between item paths in a
     * master `items` array and a `selected` array
     */
    let ArraySelectorMixin = dedupingMixin(superClass => {

      /**
       * @constructor
       * @implements {Polymer_ElementMixin}
       * @private
       */
      let elementBase = ElementMixin(superClass);

      /**
       * @polymer
       * @mixinClass
       * @implements {Polymer_ArraySelectorMixin}
       * @unrestricted
       */
      class ArraySelectorMixin extends elementBase {

        static get properties() {
          return {

            /**
             * An array containing items from which selection will be made.
             */
            items: {
              type: Array,
            },

            /**
             * When `true`, multiple items may be selected at once (in this case,
             * `selected` is an array of currently selected items).  When `false`,
             * only one item may be selected at a time.
             */
            multi: {
              type: Boolean,
              value: false,
            },

            /**
             * When `multi` is true, this is an array that contains any selected.
             * When `multi` is false, this is the currently selected item, or `null`
             * if no item is selected.
             * @type {?Object|?Array<!Object>}
             */
            selected: {type: Object, notify: true},

            /**
             * When `multi` is false, this is the currently selected item, or `null`
             * if no item is selected.
             * @type {?Object}
             */
            selectedItem: {type: Object, notify: true},

            /**
             * When `true`, calling `select` on an item that is already selected
             * will deselect the item.
             */
            toggle: {type: Boolean, value: false}

          };
        }

        static get observers() {
          return ['__updateSelection(multi, items.*)'];
        }

        constructor() {
          super();
          this.__lastItems = null;
          this.__lastMulti = null;
          this.__selectedMap = null;
        }

        __updateSelection(multi, itemsInfo) {
          let path = itemsInfo.path;
          if (path == JSCompiler_renameProperty('items', this)) {
            // Case 1 - items array changed, so diff against previous array and
            // deselect any removed items and adjust selected indices
            let newItems = itemsInfo.base || [];
            let lastItems = this.__lastItems;
            let lastMulti = this.__lastMulti;
            if (multi !== lastMulti) {
              this.clearSelection();
            }
            if (lastItems) {
              let splices = calculateSplices(newItems, lastItems);
              this.__applySplices(splices);
            }
            this.__lastItems = newItems;
            this.__lastMulti = multi;
          } else if (itemsInfo.path == `${JSCompiler_renameProperty('items', this)}.splices`) {
            // Case 2 - got specific splice information describing the array mutation:
            // deselect any removed items and adjust selected indices
            this.__applySplices(itemsInfo.value.indexSplices);
          } else {
            // Case 3 - an array element was changed, so deselect the previous
            // item for that index if it was previously selected
            let part = path.slice(`${JSCompiler_renameProperty('items', this)}.`.length);
            let idx = parseInt(part, 10);
            if ((part.indexOf('.') < 0) && part == idx) {
              this.__deselectChangedIdx(idx);
            }
          }
        }

        __applySplices(splices) {
          let selected = this.__selectedMap;
          // Adjust selected indices and mark removals
          for (let i=0; i<splices.length; i++) {
            let s = splices[i];
            selected.forEach((idx, item) => {
              if (idx < s.index) ; else if (idx >= s.index + s.removed.length) {
                // adjust index
                selected.set(item, idx + s.addedCount - s.removed.length);
              } else {
                // remove index
                selected.set(item, -1);
              }
            });
            for (let j=0; j<s.addedCount; j++) {
              let idx = s.index + j;
              if (selected.has(this.items[idx])) {
                selected.set(this.items[idx], idx);
              }
            }
          }
          // Update linked paths
          this.__updateLinks();
          // Remove selected items that were removed from the items array
          let sidx = 0;
          selected.forEach((idx, item) => {
            if (idx < 0) {
              if (this.multi) {
                this.splice(JSCompiler_renameProperty('selected', this), sidx, 1);
              } else {
                this.selected = this.selectedItem = null;
              }
              selected.delete(item);
            } else {
              sidx++;
            }
          });
        }

        __updateLinks() {
          this.__dataLinkedPaths = {};
          if (this.multi) {
            let sidx = 0;
            this.__selectedMap.forEach(idx => {
              if (idx >= 0) {
                this.linkPaths(
                    `${JSCompiler_renameProperty('items', this)}.${idx}`,
                    `${JSCompiler_renameProperty('selected', this)}.${sidx++}`);
              }
            });
          } else {
            this.__selectedMap.forEach(idx => {
              this.linkPaths(
                  JSCompiler_renameProperty('selected', this),
                  `${JSCompiler_renameProperty('items', this)}.${idx}`);
              this.linkPaths(
                  JSCompiler_renameProperty('selectedItem', this),
                  `${JSCompiler_renameProperty('items', this)}.${idx}`);
            });
          }
        }

        /**
         * Clears the selection state.
         * @override
         * @return {void}
         */
        clearSelection() {
          // Unbind previous selection
          this.__dataLinkedPaths = {};
          // The selected map stores 3 pieces of information:
          // key: items array object
          // value: items array index
          // order: selected array index
          this.__selectedMap = new Map();
          // Initialize selection
          this.selected = this.multi ? [] : null;
          this.selectedItem = null;
        }

        /**
         * Returns whether the item is currently selected.
         *
         * @override
         * @param {*} item Item from `items` array to test
         * @return {boolean} Whether the item is selected
         */
        isSelected(item) {
          return this.__selectedMap.has(item);
        }

        /**
         * Returns whether the item is currently selected.
         *
         * @override
         * @param {number} idx Index from `items` array to test
         * @return {boolean} Whether the item is selected
         */
        isIndexSelected(idx) {
          return this.isSelected(this.items[idx]);
        }

        __deselectChangedIdx(idx) {
          let sidx = this.__selectedIndexForItemIndex(idx);
          if (sidx >= 0) {
            let i = 0;
            this.__selectedMap.forEach((idx, item) => {
              if (sidx == i++) {
                this.deselect(item);
              }
            });
          }
        }

        __selectedIndexForItemIndex(idx) {
          let selected = this.__dataLinkedPaths[`${JSCompiler_renameProperty('items', this)}.${idx}`];
          if (selected) {
            return parseInt(selected.slice(`${JSCompiler_renameProperty('selected', this)}.`.length), 10);
          }
        }

        /**
         * Deselects the given item if it is already selected.
         *
         * @override
         * @param {*} item Item from `items` array to deselect
         * @return {void}
         */
        deselect(item) {
          let idx = this.__selectedMap.get(item);
          if (idx >= 0) {
            this.__selectedMap.delete(item);
            let sidx;
            if (this.multi) {
              sidx = this.__selectedIndexForItemIndex(idx);
            }
            this.__updateLinks();
            if (this.multi) {
              this.splice(JSCompiler_renameProperty('selected', this), sidx, 1);
            } else {
              this.selected = this.selectedItem = null;
            }
          }
        }

        /**
         * Deselects the given index if it is already selected.
         *
         * @override
         * @param {number} idx Index from `items` array to deselect
         * @return {void}
         */
        deselectIndex(idx) {
          this.deselect(this.items[idx]);
        }

        /**
         * Selects the given item.  When `toggle` is true, this will automatically
         * deselect the item if already selected.
         *
         * @override
         * @param {*} item Item from `items` array to select
         * @return {void}
         */
        select(item) {
          this.selectIndex(this.items.indexOf(item));
        }

        /**
         * Selects the given index.  When `toggle` is true, this will automatically
         * deselect the item if already selected.
         *
         * @override
         * @param {number} idx Index from `items` array to select
         * @return {void}
         */
        selectIndex(idx) {
          let item = this.items[idx];
          if (!this.isSelected(item)) {
            if (!this.multi) {
              this.__selectedMap.clear();
            }
            this.__selectedMap.set(item, idx);
            this.__updateLinks();
            if (this.multi) {
              this.push(JSCompiler_renameProperty('selected', this), item);
            } else {
              this.selected = this.selectedItem = item;
            }
          } else if (this.toggle) {
            this.deselectIndex(idx);
          }
        }

      }

      return ArraySelectorMixin;

    });

    /**
     * @constructor
     * @extends {PolymerElement}
     * @implements {Polymer_ArraySelectorMixin}
     * @private
     */
    let baseArraySelector = ArraySelectorMixin(PolymerElement);

    /**
     * Element implementing the `ArraySelector` mixin, which records
     * dynamic associations between item paths in a master `items` array and a
     * `selected` array such that path changes to the master array (at the host)
     * element or elsewhere via data-binding) are correctly propagated to items
     * in the selected array and vice-versa.
     *
     * The `items` property accepts an array of user data, and via the
     * `select(item)` and `deselect(item)` API, updates the `selected` property
     * which may be bound to other parts of the application, and any changes to
     * sub-fields of `selected` item(s) will be kept in sync with items in the
     * `items` array.  When `multi` is false, `selected` is a property
     * representing the last selected item.  When `multi` is true, `selected`
     * is an array of multiply selected items.
     *
     * Example:
     *
     * ```js
     * import {PolymerElement} from '@polymer/polymer';
     * import '@polymer/polymer/lib/elements/array-selector.js';
     *
     * class EmployeeList extends PolymerElement {
     *   static get _template() {
     *     return html`
     *         <div> Employee list: </div>
     *         <dom-repeat id="employeeList" items="{{employees}}">
     *           <template>
     *             <div>First name: <span>{{item.first}}</span></div>
     *               <div>Last name: <span>{{item.last}}</span></div>
     *               <button on-click="toggleSelection">Select</button>
     *           </template>
     *         </dom-repeat>
     *
     *         <array-selector id="selector"
     *                         items="{{employees}}"
     *                         selected="{{selected}}"
     *                         multi toggle></array-selector>
     *
     *         <div> Selected employees: </div>
     *         <dom-repeat items="{{selected}}">
     *           <template>
     *             <div>First name: <span>{{item.first}}</span></div>
     *             <div>Last name: <span>{{item.last}}</span></div>
     *           </template>
     *         </dom-repeat>`;
     *   }
     *   static get is() { return 'employee-list'; }
     *   static get properties() {
     *     return {
     *       employees: {
     *         value() {
     *           return [
     *             {first: 'Bob', last: 'Smith'},
     *             {first: 'Sally', last: 'Johnson'},
     *             ...
     *           ];
     *         }
     *       }
     *     };
     *   }
     *   toggleSelection(e) {
     *     const item = this.$.employeeList.itemForElement(e.target);
     *     this.$.selector.select(item);
     *   }
     * }
     * ```
     *
     * @polymer
     * @customElement
     * @extends {baseArraySelector}
     * @appliesMixin ArraySelectorMixin
     * @summary Custom element that links paths between an input `items` array and
     *   an output `selected` item or array based on calls to its selection API.
     */
    class ArraySelector extends baseArraySelector {
      // Not needed to find template; can be removed once the analyzer
      // can find the tag name from customElements.define call
      static get is() { return 'array-selector'; }
      static get template() { return null; }
    }
    customElements.define(ArraySelector.is, ArraySelector);

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    const customStyleInterface = new CustomStyleInterface();

    if (!window.ShadyCSS) {
      window.ShadyCSS = {
        /**
         * @param {!HTMLTemplateElement} template
         * @param {string} elementName
         * @param {string=} elementExtends
         */
        prepareTemplate(template, elementName, elementExtends) {}, // eslint-disable-line no-unused-vars

        /**
         * @param {!HTMLTemplateElement} template
         * @param {string} elementName
         */
        prepareTemplateDom(template, elementName) {}, // eslint-disable-line no-unused-vars

        /**
         * @param {!HTMLTemplateElement} template
         * @param {string} elementName
         * @param {string=} elementExtends
         */
        prepareTemplateStyles(template, elementName, elementExtends) {}, // eslint-disable-line no-unused-vars

        /**
         * @param {Element} element
         * @param {Object=} properties
         */
        styleSubtree(element, properties) {
          customStyleInterface.processStyles();
          updateNativeProperties(element, properties);
        },

        /**
         * @param {Element} element
         */
        styleElement(element) { // eslint-disable-line no-unused-vars
          customStyleInterface.processStyles();
        },

        /**
         * @param {Object=} properties
         */
        styleDocument(properties) {
          customStyleInterface.processStyles();
          updateNativeProperties(document.body, properties);
        },

        /**
         * @param {Element} element
         * @param {string} property
         * @return {string}
         */
        getComputedStyleValue(element, property) {
          return getComputedStyleValue(element, property);
        },

        flushCustomStyles() {},
        nativeCss: nativeCssVariables,
        nativeShadow: nativeShadow,
        cssBuild: cssBuild,
        disableRuntime: disableRuntime,
      };
    }

    window.ShadyCSS.CustomStyleInterface = customStyleInterface;

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    const attr = 'include';

    const CustomStyleInterface$1 = window.ShadyCSS.CustomStyleInterface;

    /**
     * Custom element for defining styles in the main document that can take
     * advantage of [shady DOM](https://github.com/webcomponents/shadycss) shims
     * for style encapsulation, custom properties, and custom mixins.
     *
     * - Document styles defined in a `<custom-style>` are shimmed to ensure they
     *   do not leak into local DOM when running on browsers without native
     *   Shadow DOM.
     * - Custom properties can be defined in a `<custom-style>`. Use the `html` selector
     *   to define custom properties that apply to all custom elements.
     * - Custom mixins can be defined in a `<custom-style>`, if you import the optional
     *   [apply shim](https://github.com/webcomponents/shadycss#about-applyshim)
     *   (`shadycss/apply-shim.html`).
     *
     * To use:
     *
     * - Import `custom-style.html`.
     * - Place a `<custom-style>` element in the main document, wrapping an inline `<style>` tag that
     *   contains the CSS rules you want to shim.
     *
     * For example:
     *
     * ```html
     * <!-- import apply shim--only required if using mixins -->
     * <link rel="import" href="bower_components/shadycss/apply-shim.html">
     * <!-- import custom-style element -->
     * <link rel="import" href="bower_components/polymer/lib/elements/custom-style.html">
     *
     * <custom-style>
     *   <style>
     *     html {
     *       --custom-color: blue;
     *       --custom-mixin: {
     *         font-weight: bold;
     *         color: red;
     *       };
     *     }
     *   </style>
     * </custom-style>
     * ```
     *
     * @customElement
     * @extends HTMLElement
     * @summary Custom element for defining styles in the main document that can
     *   take advantage of Polymer's style scoping and custom properties shims.
     */
    class CustomStyle extends HTMLElement {
      constructor() {
        super();
        this._style = null;
        CustomStyleInterface$1.addCustomStyle(this);
      }
      /**
       * Returns the light-DOM `<style>` child this element wraps.  Upon first
       * call any style modules referenced via the `include` attribute will be
       * concatenated to this element's `<style>`.
       *
       * @export
       * @return {HTMLStyleElement} This element's light-DOM `<style>`
       */
      getStyle() {
        if (this._style) {
          return this._style;
        }
        const style = /** @type {HTMLStyleElement} */(this.querySelector('style'));
        if (!style) {
          return null;
        }
        this._style = style;
        const include = style.getAttribute(attr);
        if (include) {
          style.removeAttribute(attr);
          /** @suppress {deprecated} */
          style.textContent = cssFromModules(include) + style.textContent;
        }
        /*
        HTML Imports styling the main document are deprecated in Chrome
        https://crbug.com/523952

        If this element is not in the main document, then it must be in an HTML Import document.
        In that case, move the custom style to the main document.

        The ordering of `<custom-style>` should stay the same as when loaded by HTML Imports, but there may be odd
        cases of ordering w.r.t the main document styles.
        */
        if (this.ownerDocument !== window.document) {
          window.document.head.appendChild(this);
        }
        return this._style;
      }
    }

    window.customElements.define('custom-style', CustomStyle);

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    let mutablePropertyChange$1;
    /** @suppress {missingProperties} */
    (() => {
      mutablePropertyChange$1 = MutableData._mutablePropertyChange;
    })();

    /**
     * Legacy element behavior to add the optional ability to skip strict
     * dirty-checking for objects and arrays (always consider them to be
     * "dirty") by setting a `mutable-data` attribute on an element instance.
     *
     * By default, `Polymer.PropertyEffects` performs strict dirty checking on
     * objects, which means that any deep modifications to an object or array will
     * not be propagated unless "immutable" data patterns are used (i.e. all object
     * references from the root to the mutation were changed).
     *
     * Polymer also provides a proprietary data mutation and path notification API
     * (e.g. `notifyPath`, `set`, and array mutation API's) that allow efficient
     * mutation and notification of deep changes in an object graph to all elements
     * bound to the same object graph.
     *
     * In cases where neither immutable patterns nor the data mutation API can be
     * used, applying this mixin will allow Polymer to skip dirty checking for
     * objects and arrays (always consider them to be "dirty").  This allows a
     * user to make a deep modification to a bound object graph, and then either
     * simply re-set the object (e.g. `this.items = this.items`) or call `notifyPath`
     * (e.g. `this.notifyPath('items')`) to update the tree.  Note that all
     * elements that wish to be updated based on deep mutations must apply this
     * mixin or otherwise skip strict dirty checking for objects/arrays.
     * Specifically, any elements in the binding tree between the source of a
     * mutation and the consumption of it must enable this behavior or apply the
     * `Polymer.OptionalMutableDataBehavior`.
     *
     * While this behavior adds the ability to forgo Object/Array dirty checking,
     * the `mutableData` flag defaults to false and must be set on the instance.
     *
     * Note, the performance characteristics of propagating large object graphs
     * will be worse by relying on `mutableData: true` as opposed to using
     * strict dirty checking with immutable patterns or Polymer's path notification
     * API.
     *
     * @polymerBehavior
     * @summary Behavior to optionally skip strict dirty-checking for objects and
     *   arrays
     */
    const OptionalMutableDataBehavior = {

      properties: {
        /**
         * Instance-level flag for configuring the dirty-checking strategy
         * for this element.  When true, Objects and Arrays will skip dirty
         * checking, otherwise strict equality checking will be used.
         */
        mutableData: Boolean
      },

      /**
       * Overrides `Polymer.PropertyEffects` to skip strict equality checking
       * for Objects and Arrays.
       *
       * Pulls the value to dirty check against from the `__dataTemp` cache
       * (rather than the normal `__data` cache) for Objects.  Since the temp
       * cache is cleared at the end of a turn, this implementation allows
       * side-effects of deep object changes to be processed by re-setting the
       * same object (using the temp cache as an in-turn backstop to prevent
       * cycles due to 2-way notification).
       *
       * @param {string} property Property name
       * @param {*} value New property value
       * @param {*} old Previous property value
       * @return {boolean} Whether the property should be considered a change
       * @protected
       * @override
       */
      _shouldPropertyChange(property, value, old) {
        return mutablePropertyChange$1(this, property, value, old, this.mutableData);
      }
    };

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    // bc
    const Base = LegacyElementMixin(HTMLElement).prototype;

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    // Contains all connected resizables that do not have a parent.
    var ORPHANS = new Set();

    /**
     * `IronResizableBehavior` is a behavior that can be used in Polymer elements to
     * coordinate the flow of resize events between "resizers" (elements that
     *control the size or hidden state of their children) and "resizables" (elements
     *that need to be notified when they are resized or un-hidden by their parents
     *in order to take action on their new measurements).
     *
     * Elements that perform measurement should add the `IronResizableBehavior`
     *behavior to their element definition and listen for the `iron-resize` event on
     *themselves. This event will be fired when they become showing after having
     *been hidden, when they are resized explicitly by another resizable, or when
     *the window has been resized.
     *
     * Note, the `iron-resize` event is non-bubbling.
     *
     * @polymerBehavior
     * @demo demo/index.html
     **/
    const IronResizableBehavior = {
      properties: {
        /**
         * The closest ancestor element that implements `IronResizableBehavior`.
         */
        _parentResizable: {
          type: Object,
          observer: '_parentResizableChanged',
        },

        /**
         * True if this element is currently notifying its descendant elements of
         * resize.
         */
        _notifyingDescendant: {
          type: Boolean,
          value: false,
        }
      },

      listeners: {
        'iron-request-resize-notifications': '_onIronRequestResizeNotifications'
      },

      created: function() {
        // We don't really need property effects on these, and also we want them
        // to be created before the `_parentResizable` observer fires:
        this._interestedResizables = [];
        this._boundNotifyResize = this.notifyResize.bind(this);
        this._boundOnDescendantIronResize = this._onDescendantIronResize.bind(this);
      },

      attached: function() {
        this._requestResizeNotifications();
      },

      detached: function() {
        if (this._parentResizable) {
          this._parentResizable.stopResizeNotificationsFor(this);
        } else {
          ORPHANS.delete(this);
          window.removeEventListener('resize', this._boundNotifyResize);
        }

        this._parentResizable = null;
      },

      /**
       * Can be called to manually notify a resizable and its descendant
       * resizables of a resize change.
       */
      notifyResize: function() {
        if (!this.isAttached) {
          return;
        }

        this._interestedResizables.forEach(function(resizable) {
          if (this.resizerShouldNotify(resizable)) {
            this._notifyDescendant(resizable);
          }
        }, this);

        this._fireResize();
      },

      /**
       * Used to assign the closest resizable ancestor to this resizable
       * if the ancestor detects a request for notifications.
       */
      assignParentResizable: function(parentResizable) {
        if (this._parentResizable) {
          this._parentResizable.stopResizeNotificationsFor(this);
        }

        this._parentResizable = parentResizable;

        if (parentResizable &&
            parentResizable._interestedResizables.indexOf(this) === -1) {
          parentResizable._interestedResizables.push(this);
          parentResizable._subscribeIronResize(this);
        }
      },

      /**
       * Used to remove a resizable descendant from the list of descendants
       * that should be notified of a resize change.
       */
      stopResizeNotificationsFor: function(target) {
        var index = this._interestedResizables.indexOf(target);

        if (index > -1) {
          this._interestedResizables.splice(index, 1);
          this._unsubscribeIronResize(target);
        }
      },

      /**
       * Subscribe this element to listen to iron-resize events on the given target.
       *
       * Preferred over target.listen because the property renamer does not
       * understand to rename when the target is not specifically "this"
       *
       * @param {!HTMLElement} target Element to listen to for iron-resize events.
       */
      _subscribeIronResize: function(target) {
        target.addEventListener('iron-resize', this._boundOnDescendantIronResize);
      },

      /**
       * Unsubscribe this element from listening to to iron-resize events on the
       * given target.
       *
       * Preferred over target.unlisten because the property renamer does not
       * understand to rename when the target is not specifically "this"
       *
       * @param {!HTMLElement} target Element to listen to for iron-resize events.
       */
      _unsubscribeIronResize: function(target) {
        target.removeEventListener(
            'iron-resize', this._boundOnDescendantIronResize);
      },

      /**
       * This method can be overridden to filter nested elements that should or
       * should not be notified by the current element. Return true if an element
       * should be notified, or false if it should not be notified.
       *
       * @param {HTMLElement} element A candidate descendant element that
       * implements `IronResizableBehavior`.
       * @return {boolean} True if the `element` should be notified of resize.
       */
      resizerShouldNotify: function(element) {
        return true;
      },

      _onDescendantIronResize: function(event) {
        if (this._notifyingDescendant) {
          event.stopPropagation();
          return;
        }

        // no need to use this during shadow dom because of event retargeting
        if (!useShadow) {
          this._fireResize();
        }
      },

      _fireResize: function() {
        this.fire('iron-resize', null, {node: this, bubbles: false});
      },

      _onIronRequestResizeNotifications: function(event) {
        var target = /** @type {!EventTarget} */ (dom(event).rootTarget);
        if (target === this) {
          return;
        }

        target.assignParentResizable(this);
        this._notifyDescendant(target);

        event.stopPropagation();
      },

      _parentResizableChanged: function(parentResizable) {
        if (parentResizable) {
          window.removeEventListener('resize', this._boundNotifyResize);
        }
      },

      _notifyDescendant: function(descendant) {
        // NOTE(cdata): In IE10, attached is fired on children first, so it's
        // important not to notify them if the parent is not attached yet (or
        // else they will get redundantly notified when the parent attaches).
        if (!this.isAttached) {
          return;
        }

        this._notifyingDescendant = true;
        descendant.notifyResize();
        this._notifyingDescendant = false;
      },

      _requestResizeNotifications: function() {
        if (!this.isAttached) {
          return;
        }

        if (document.readyState === 'loading') {
          var _requestResizeNotifications =
              this._requestResizeNotifications.bind(this);
          document.addEventListener(
              'readystatechange', function readystatechanged() {
                document.removeEventListener('readystatechange', readystatechanged);
                _requestResizeNotifications();
              });
        } else {
          this._findParent();

          if (!this._parentResizable) {
            // If this resizable is an orphan, tell other orphans to try to find
            // their parent again, in case it's this resizable.
            ORPHANS.forEach(function(orphan) {
              if (orphan !== this) {
                orphan._findParent();
              }
            }, this);

            window.addEventListener('resize', this._boundNotifyResize);
            this.notifyResize();
          } else {
            // If this resizable has a parent, tell other child resizables of
            // that parent to try finding their parent again, in case it's this
            // resizable.
            this._parentResizable._interestedResizables
                .forEach(function(resizable) {
                  if (resizable !== this) {
                    resizable._findParent();
                  }
                }, this);
          }
        }
      },

      _findParent: function() {
        this.assignParentResizable(null);
        this.fire(
            'iron-request-resize-notifications',
            null,
            {node: this, bubbles: true, cancelable: true});

        if (!this._parentResizable) {
          ORPHANS.add(this);
        } else {
          ORPHANS.delete(this);
        }
      }
    };

    var IronResizableBehavior$1 = /*#__PURE__*/Object.freeze({
        __proto__: null,
        IronResizableBehavior: IronResizableBehavior
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
    `iron-collapse` creates a collapsible block of content.  By default, the content
    will be collapsed.  Use `opened` or `toggle()` to show/hide the content.

        <button on-click="toggle">toggle collapse</button>

        <iron-collapse id="collapse">
          <div>Content goes here...</div>
        </iron-collapse>

        ...

        toggle: function() {
          this.$.collapse.toggle();
        }

    `iron-collapse` adjusts the max-height/max-width of the collapsible element to
    show/hide the content.  So avoid putting padding/margin/border on the
    collapsible directly, and instead put a div inside and style that.

        <style>
          .collapse-content {
            padding: 15px;
            border: 1px solid #dedede;
          }
        </style>

        <iron-collapse>
          <div class="collapse-content">
            <div>Content goes here...</div>
          </div>
        </iron-collapse>

    ### Styling

    The following custom properties and mixins are available for styling:

    Custom property | Description | Default
    ----------------|-------------|----------
    `--iron-collapse-transition-duration` | Animation transition duration | `300ms`

    @group Iron Elements
    @hero hero.svg
    @demo demo/index.html
    @element iron-collapse
    */
    Polymer({
      _template: html`
    <style>
      :host {
        display: block;
        transition-duration: var(--iron-collapse-transition-duration, 300ms);
        /* Safari 10 needs this property prefixed to correctly apply the custom property */
        -webkit-transition-duration: var(--iron-collapse-transition-duration, 300ms);
        overflow: visible;
      }

      :host(.iron-collapse-closed) {
        display: none;
      }

      :host(:not(.iron-collapse-opened)) {
        overflow: hidden;
      }
    </style>

    <slot></slot>
`,

      is: 'iron-collapse',
      behaviors: [IronResizableBehavior],

      properties: {

        /**
         * If true, the orientation is horizontal; otherwise is vertical.
         *
         * @attribute horizontal
         */
        horizontal: {type: Boolean, value: false, observer: '_horizontalChanged'},

        /**
         * Set opened to true to show the collapse element and to false to hide it.
         *
         * @attribute opened
         */
        opened:
            {type: Boolean, value: false, notify: true, observer: '_openedChanged'},

        /**
         * When true, the element is transitioning its opened state. When false,
         * the element has finished opening/closing.
         *
         * @attribute transitioning
         */
        transitioning: {type: Boolean, notify: true, readOnly: true},

        /**
         * Set noAnimation to true to disable animations.
         *
         * @attribute noAnimation
         */
        noAnimation: {type: Boolean},

        /**
         * Stores the desired size of the collapse body.
         * @private
         */
        _desiredSize: {type: String, value: ''}
      },

      get dimension() {
        return this.horizontal ? 'width' : 'height';
      },

      /**
       * `maxWidth` or `maxHeight`.
       * @private
       */
      get _dimensionMax() {
        return this.horizontal ? 'maxWidth' : 'maxHeight';
      },

      /**
       * `max-width` or `max-height`.
       * @private
       */
      get _dimensionMaxCss() {
        return this.horizontal ? 'max-width' : 'max-height';
      },

      hostAttributes: {
        role: 'group',
        'aria-hidden': 'true',
      },

      listeners: {transitionend: '_onTransitionEnd'},

      /**
       * Toggle the opened state.
       *
       * @method toggle
       */
      toggle: function() {
        this.opened = !this.opened;
      },

      show: function() {
        this.opened = true;
      },

      hide: function() {
        this.opened = false;
      },

      /**
       * Updates the size of the element.
       * @param {string} size The new value for `maxWidth`/`maxHeight` as css property value, usually `auto` or `0px`.
       * @param {boolean=} animated if `true` updates the size with an animation, otherwise without.
       */
      updateSize: function(size, animated) {
        // Consider 'auto' as '', to take full size.
        size = size === 'auto' ? '' : size;

        var willAnimate = animated && !this.noAnimation && this.isAttached &&
            this._desiredSize !== size;

        this._desiredSize = size;

        this._updateTransition(false);
        // If we can animate, must do some prep work.
        if (willAnimate) {
          // Animation will start at the current size.
          var startSize = this._calcSize();
          // For `auto` we must calculate what is the final size for the animation.
          // After the transition is done, _transitionEnd will set the size back to
          // `auto`.
          if (size === '') {
            this.style[this._dimensionMax] = '';
            size = this._calcSize();
          }
          // Go to startSize without animation.
          this.style[this._dimensionMax] = startSize;
          // Force layout to ensure transition will go. Set scrollTop to itself
          // so that compilers won't remove it.
          this.scrollTop = this.scrollTop;
          // Enable animation.
          this._updateTransition(true);
          // If final size is the same as startSize it will not animate.
          willAnimate = (size !== startSize);
        }
        // Set the final size.
        this.style[this._dimensionMax] = size;
        // If it won't animate, call transitionEnd to set correct classes.
        if (!willAnimate) {
          this._transitionEnd();
        }
      },

      /**
       * enableTransition() is deprecated, but left over so it doesn't break
       * existing code. Please use `noAnimation` property instead.
       *
       * @method enableTransition
       * @deprecated since version 1.0.4
       */
      enableTransition: function(enabled) {
        Base._warn(
            '`enableTransition()` is deprecated, use `noAnimation` instead.');
        this.noAnimation = !enabled;
      },

      _updateTransition: function(enabled) {
        this.style.transitionDuration = (enabled && !this.noAnimation) ? '' : '0s';
      },

      _horizontalChanged: function() {
        this.style.transitionProperty = this._dimensionMaxCss;
        var otherDimension =
            this._dimensionMax === 'maxWidth' ? 'maxHeight' : 'maxWidth';
        this.style[otherDimension] = '';
        this.updateSize(this.opened ? 'auto' : '0px', false);
      },

      _openedChanged: function() {
        this.setAttribute('aria-hidden', !this.opened);

        this._setTransitioning(true);
        this.toggleClass('iron-collapse-closed', false);
        this.toggleClass('iron-collapse-opened', false);
        this.updateSize(this.opened ? 'auto' : '0px', true);

        // Focus the current collapse.
        if (this.opened) {
          this.focus();
        }
      },

      _transitionEnd: function() {
        this.style[this._dimensionMax] = this._desiredSize;
        this.toggleClass('iron-collapse-closed', !this.opened);
        this.toggleClass('iron-collapse-opened', this.opened);
        this._updateTransition(false);
        this.notifyResize();
        this._setTransitioning(false);
      },

      _onTransitionEnd: function(event) {
        if (dom(event).rootTarget === this) {
          this._transitionEnd();
        }
      },

      _calcSize: function() {
        return this.getBoundingClientRect()[this.dimension] + 'px';
      }
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /*
    A set of layout classes that let you specify layout properties directly in
    markup. You must include this file in every element that needs to use them.

    Sample use:

        import '@polymer/iron-flex-layout/iron-flex-layout-classes.js';

        const template = html`
          <style is="custom-style" include="iron-flex iron-flex-alignment"></style>
          <style>
            .test { width: 100px; }
          </style>
          <div class="layout horizontal center-center">
            <div class="test">horizontal layout center alignment</div>
          </div>
        `;
        document.body.appendChild(template.content);

    The following imports are available:
     - iron-flex
     - iron-flex-reverse
     - iron-flex-alignment
     - iron-flex-factors
     - iron-positioning
    */

    const template = html`
/* Most common used flex styles*/
<dom-module id="iron-flex">
  <template>
    <style>
      .layout.horizontal,
      .layout.vertical {
        display: -ms-flexbox;
        display: -webkit-flex;
        display: flex;
      }

      .layout.inline {
        display: -ms-inline-flexbox;
        display: -webkit-inline-flex;
        display: inline-flex;
      }

      .layout.horizontal {
        -ms-flex-direction: row;
        -webkit-flex-direction: row;
        flex-direction: row;
      }

      .layout.vertical {
        -ms-flex-direction: column;
        -webkit-flex-direction: column;
        flex-direction: column;
      }

      .layout.wrap {
        -ms-flex-wrap: wrap;
        -webkit-flex-wrap: wrap;
        flex-wrap: wrap;
      }

      .layout.no-wrap {
        -ms-flex-wrap: nowrap;
        -webkit-flex-wrap: nowrap;
        flex-wrap: nowrap;
      }

      .layout.center,
      .layout.center-center {
        -ms-flex-align: center;
        -webkit-align-items: center;
        align-items: center;
      }

      .layout.center-justified,
      .layout.center-center {
        -ms-flex-pack: center;
        -webkit-justify-content: center;
        justify-content: center;
      }

      .flex {
        -ms-flex: 1 1 0.000000001px;
        -webkit-flex: 1;
        flex: 1;
        -webkit-flex-basis: 0.000000001px;
        flex-basis: 0.000000001px;
      }

      .flex-auto {
        -ms-flex: 1 1 auto;
        -webkit-flex: 1 1 auto;
        flex: 1 1 auto;
      }

      .flex-none {
        -ms-flex: none;
        -webkit-flex: none;
        flex: none;
      }
    </style>
  </template>
</dom-module>
/* Basic flexbox reverse styles */
<dom-module id="iron-flex-reverse">
  <template>
    <style>
      .layout.horizontal-reverse,
      .layout.vertical-reverse {
        display: -ms-flexbox;
        display: -webkit-flex;
        display: flex;
      }

      .layout.horizontal-reverse {
        -ms-flex-direction: row-reverse;
        -webkit-flex-direction: row-reverse;
        flex-direction: row-reverse;
      }

      .layout.vertical-reverse {
        -ms-flex-direction: column-reverse;
        -webkit-flex-direction: column-reverse;
        flex-direction: column-reverse;
      }

      .layout.wrap-reverse {
        -ms-flex-wrap: wrap-reverse;
        -webkit-flex-wrap: wrap-reverse;
        flex-wrap: wrap-reverse;
      }
    </style>
  </template>
</dom-module>
/* Flexbox alignment */
<dom-module id="iron-flex-alignment">
  <template>
    <style>
      /**
       * Alignment in cross axis.
       */
      .layout.start {
        -ms-flex-align: start;
        -webkit-align-items: flex-start;
        align-items: flex-start;
      }

      .layout.center,
      .layout.center-center {
        -ms-flex-align: center;
        -webkit-align-items: center;
        align-items: center;
      }

      .layout.end {
        -ms-flex-align: end;
        -webkit-align-items: flex-end;
        align-items: flex-end;
      }

      .layout.baseline {
        -ms-flex-align: baseline;
        -webkit-align-items: baseline;
        align-items: baseline;
      }

      /**
       * Alignment in main axis.
       */
      .layout.start-justified {
        -ms-flex-pack: start;
        -webkit-justify-content: flex-start;
        justify-content: flex-start;
      }

      .layout.center-justified,
      .layout.center-center {
        -ms-flex-pack: center;
        -webkit-justify-content: center;
        justify-content: center;
      }

      .layout.end-justified {
        -ms-flex-pack: end;
        -webkit-justify-content: flex-end;
        justify-content: flex-end;
      }

      .layout.around-justified {
        -ms-flex-pack: distribute;
        -webkit-justify-content: space-around;
        justify-content: space-around;
      }

      .layout.justified {
        -ms-flex-pack: justify;
        -webkit-justify-content: space-between;
        justify-content: space-between;
      }

      /**
       * Self alignment.
       */
      .self-start {
        -ms-align-self: flex-start;
        -webkit-align-self: flex-start;
        align-self: flex-start;
      }

      .self-center {
        -ms-align-self: center;
        -webkit-align-self: center;
        align-self: center;
      }

      .self-end {
        -ms-align-self: flex-end;
        -webkit-align-self: flex-end;
        align-self: flex-end;
      }

      .self-stretch {
        -ms-align-self: stretch;
        -webkit-align-self: stretch;
        align-self: stretch;
      }

      .self-baseline {
        -ms-align-self: baseline;
        -webkit-align-self: baseline;
        align-self: baseline;
      }

      /**
       * multi-line alignment in main axis.
       */
      .layout.start-aligned {
        -ms-flex-line-pack: start;  /* IE10 */
        -ms-align-content: flex-start;
        -webkit-align-content: flex-start;
        align-content: flex-start;
      }

      .layout.end-aligned {
        -ms-flex-line-pack: end;  /* IE10 */
        -ms-align-content: flex-end;
        -webkit-align-content: flex-end;
        align-content: flex-end;
      }

      .layout.center-aligned {
        -ms-flex-line-pack: center;  /* IE10 */
        -ms-align-content: center;
        -webkit-align-content: center;
        align-content: center;
      }

      .layout.between-aligned {
        -ms-flex-line-pack: justify;  /* IE10 */
        -ms-align-content: space-between;
        -webkit-align-content: space-between;
        align-content: space-between;
      }

      .layout.around-aligned {
        -ms-flex-line-pack: distribute;  /* IE10 */
        -ms-align-content: space-around;
        -webkit-align-content: space-around;
        align-content: space-around;
      }
    </style>
  </template>
</dom-module>
/* Non-flexbox positioning helper styles */
<dom-module id="iron-flex-factors">
  <template>
    <style>
      .flex,
      .flex-1 {
        -ms-flex: 1 1 0.000000001px;
        -webkit-flex: 1;
        flex: 1;
        -webkit-flex-basis: 0.000000001px;
        flex-basis: 0.000000001px;
      }

      .flex-2 {
        -ms-flex: 2;
        -webkit-flex: 2;
        flex: 2;
      }

      .flex-3 {
        -ms-flex: 3;
        -webkit-flex: 3;
        flex: 3;
      }

      .flex-4 {
        -ms-flex: 4;
        -webkit-flex: 4;
        flex: 4;
      }

      .flex-5 {
        -ms-flex: 5;
        -webkit-flex: 5;
        flex: 5;
      }

      .flex-6 {
        -ms-flex: 6;
        -webkit-flex: 6;
        flex: 6;
      }

      .flex-7 {
        -ms-flex: 7;
        -webkit-flex: 7;
        flex: 7;
      }

      .flex-8 {
        -ms-flex: 8;
        -webkit-flex: 8;
        flex: 8;
      }

      .flex-9 {
        -ms-flex: 9;
        -webkit-flex: 9;
        flex: 9;
      }

      .flex-10 {
        -ms-flex: 10;
        -webkit-flex: 10;
        flex: 10;
      }

      .flex-11 {
        -ms-flex: 11;
        -webkit-flex: 11;
        flex: 11;
      }

      .flex-12 {
        -ms-flex: 12;
        -webkit-flex: 12;
        flex: 12;
      }
    </style>
  </template>
</dom-module>
<dom-module id="iron-positioning">
  <template>
    <style>
      .block {
        display: block;
      }

      [hidden] {
        display: none !important;
      }

      .invisible {
        visibility: hidden !important;
      }

      .relative {
        position: relative;
      }

      .fit {
        position: absolute;
        top: 0;
        right: 0;
        bottom: 0;
        left: 0;
      }

      body.fullbleed {
        margin: 0;
        height: 100vh;
      }

      .scroll {
        -webkit-overflow-scrolling: touch;
        overflow: auto;
      }

      /* fixed position */
      .fixed-bottom,
      .fixed-left,
      .fixed-right,
      .fixed-top {
        position: fixed;
      }

      .fixed-top {
        top: 0;
        left: 0;
        right: 0;
      }

      .fixed-right {
        top: 0;
        right: 0;
        bottom: 0;
      }

      .fixed-bottom {
        right: 0;
        bottom: 0;
        left: 0;
      }

      .fixed-left {
        top: 0;
        bottom: 0;
        left: 0;
      }
    </style>
  </template>
</dom-module>
`;
    template.setAttribute('style', 'display: none;');
    document.head.appendChild(template.content);

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
    The `<iron-flex-layout>` component provides simple ways to use
    [CSS flexible box
    layout](https://developer.mozilla.org/en-US/docs/Web/Guide/CSS/Flexible_boxes),
    also known as flexbox. Note that this is an old element, that was written
    before all modern browsers had non-prefixed flex styles. As such, nowadays you
    don't really need to use this element anymore, and can use CSS flex styles
    directly in your code.

    This component provides two different ways to use flexbox:

    1. [Layout
    classes](https://github.com/PolymerElements/iron-flex-layout/tree/master/iron-flex-layout-classes.html).
    The layout class stylesheet provides a simple set of class-based flexbox rules,
    that let you specify layout properties directly in markup. You must include this
    file in every element that needs to use them.

        Sample use:

        ```
        <custom-element-demo>
          <template>
            <script src="../webcomponentsjs/webcomponents-lite.js"></JAVA_SCRIIIIPT/>
            <next-code-block></next-code-block>
          </template>
        </custom-element-demo>
        ```

        ```js
        import {html} from '@polymer/polymer/lib/utils/html-tag.js';
        import '@polymer/iron-flex-layout/iron-flex-layout-classes.js';

        const template = html`
          <style is="custom-style" include="iron-flex iron-flex-alignment"></style>
          <style>
            .test { width: 100px; }
          </style>
          <div class="layout horizontal center-center">
            <div class="test">horizontal layout center alignment</div>
          </div>
        `;
        document.body.appendChild(template.content);
        ```

    2. [Custom CSS
    mixins](https://github.com/PolymerElements/iron-flex-layout/blob/master/iron-flex-layout.html).
    The mixin stylesheet includes custom CSS mixins that can be applied inside a CSS
    rule using the `@apply` function.

    Please note that the old [/deep/ layout
    classes](https://github.com/PolymerElements/iron-flex-layout/tree/master/classes)
    are deprecated, and should not be used. To continue using layout properties
    directly in markup, please switch to using the new `dom-module`-based
    [layout
    classes](https://github.com/PolymerElements/iron-flex-layout/tree/master/iron-flex-layout-classes.html).
    Please note that the new version does not use `/deep/`, and therefore requires
    you to import the `dom-modules` in every element that needs to use them.

    @group Iron Elements
    @pseudoElement iron-flex-layout
    @demo demo/index.html
    */
    const template$1 = html`
<custom-style>
  <style is="custom-style">
    [hidden] {
      display: none !important;
    }
  </style>
</custom-style>
<custom-style>
  <style is="custom-style">
    html {

      --layout: {
        display: -ms-flexbox;
        display: -webkit-flex;
        display: flex;
      };

      --layout-inline: {
        display: -ms-inline-flexbox;
        display: -webkit-inline-flex;
        display: inline-flex;
      };

      --layout-horizontal: {
        @apply --layout;

        -ms-flex-direction: row;
        -webkit-flex-direction: row;
        flex-direction: row;
      };

      --layout-horizontal-reverse: {
        @apply --layout;

        -ms-flex-direction: row-reverse;
        -webkit-flex-direction: row-reverse;
        flex-direction: row-reverse;
      };

      --layout-vertical: {
        @apply --layout;

        -ms-flex-direction: column;
        -webkit-flex-direction: column;
        flex-direction: column;
      };

      --layout-vertical-reverse: {
        @apply --layout;

        -ms-flex-direction: column-reverse;
        -webkit-flex-direction: column-reverse;
        flex-direction: column-reverse;
      };

      --layout-wrap: {
        -ms-flex-wrap: wrap;
        -webkit-flex-wrap: wrap;
        flex-wrap: wrap;
      };

      --layout-wrap-reverse: {
        -ms-flex-wrap: wrap-reverse;
        -webkit-flex-wrap: wrap-reverse;
        flex-wrap: wrap-reverse;
      };

      --layout-flex-auto: {
        -ms-flex: 1 1 auto;
        -webkit-flex: 1 1 auto;
        flex: 1 1 auto;
      };

      --layout-flex-none: {
        -ms-flex: none;
        -webkit-flex: none;
        flex: none;
      };

      --layout-flex: {
        -ms-flex: 1 1 0.000000001px;
        -webkit-flex: 1;
        flex: 1;
        -webkit-flex-basis: 0.000000001px;
        flex-basis: 0.000000001px;
      };

      --layout-flex-2: {
        -ms-flex: 2;
        -webkit-flex: 2;
        flex: 2;
      };

      --layout-flex-3: {
        -ms-flex: 3;
        -webkit-flex: 3;
        flex: 3;
      };

      --layout-flex-4: {
        -ms-flex: 4;
        -webkit-flex: 4;
        flex: 4;
      };

      --layout-flex-5: {
        -ms-flex: 5;
        -webkit-flex: 5;
        flex: 5;
      };

      --layout-flex-6: {
        -ms-flex: 6;
        -webkit-flex: 6;
        flex: 6;
      };

      --layout-flex-7: {
        -ms-flex: 7;
        -webkit-flex: 7;
        flex: 7;
      };

      --layout-flex-8: {
        -ms-flex: 8;
        -webkit-flex: 8;
        flex: 8;
      };

      --layout-flex-9: {
        -ms-flex: 9;
        -webkit-flex: 9;
        flex: 9;
      };

      --layout-flex-10: {
        -ms-flex: 10;
        -webkit-flex: 10;
        flex: 10;
      };

      --layout-flex-11: {
        -ms-flex: 11;
        -webkit-flex: 11;
        flex: 11;
      };

      --layout-flex-12: {
        -ms-flex: 12;
        -webkit-flex: 12;
        flex: 12;
      };

      /* alignment in cross axis */

      --layout-start: {
        -ms-flex-align: start;
        -webkit-align-items: flex-start;
        align-items: flex-start;
      };

      --layout-center: {
        -ms-flex-align: center;
        -webkit-align-items: center;
        align-items: center;
      };

      --layout-end: {
        -ms-flex-align: end;
        -webkit-align-items: flex-end;
        align-items: flex-end;
      };

      --layout-baseline: {
        -ms-flex-align: baseline;
        -webkit-align-items: baseline;
        align-items: baseline;
      };

      /* alignment in main axis */

      --layout-start-justified: {
        -ms-flex-pack: start;
        -webkit-justify-content: flex-start;
        justify-content: flex-start;
      };

      --layout-center-justified: {
        -ms-flex-pack: center;
        -webkit-justify-content: center;
        justify-content: center;
      };

      --layout-end-justified: {
        -ms-flex-pack: end;
        -webkit-justify-content: flex-end;
        justify-content: flex-end;
      };

      --layout-around-justified: {
        -ms-flex-pack: distribute;
        -webkit-justify-content: space-around;
        justify-content: space-around;
      };

      --layout-justified: {
        -ms-flex-pack: justify;
        -webkit-justify-content: space-between;
        justify-content: space-between;
      };

      --layout-center-center: {
        @apply --layout-center;
        @apply --layout-center-justified;
      };

      /* self alignment */

      --layout-self-start: {
        -ms-align-self: flex-start;
        -webkit-align-self: flex-start;
        align-self: flex-start;
      };

      --layout-self-center: {
        -ms-align-self: center;
        -webkit-align-self: center;
        align-self: center;
      };

      --layout-self-end: {
        -ms-align-self: flex-end;
        -webkit-align-self: flex-end;
        align-self: flex-end;
      };

      --layout-self-stretch: {
        -ms-align-self: stretch;
        -webkit-align-self: stretch;
        align-self: stretch;
      };

      --layout-self-baseline: {
        -ms-align-self: baseline;
        -webkit-align-self: baseline;
        align-self: baseline;
      };

      /* multi-line alignment in main axis */

      --layout-start-aligned: {
        -ms-flex-line-pack: start;  /* IE10 */
        -ms-align-content: flex-start;
        -webkit-align-content: flex-start;
        align-content: flex-start;
      };

      --layout-end-aligned: {
        -ms-flex-line-pack: end;  /* IE10 */
        -ms-align-content: flex-end;
        -webkit-align-content: flex-end;
        align-content: flex-end;
      };

      --layout-center-aligned: {
        -ms-flex-line-pack: center;  /* IE10 */
        -ms-align-content: center;
        -webkit-align-content: center;
        align-content: center;
      };

      --layout-between-aligned: {
        -ms-flex-line-pack: justify;  /* IE10 */
        -ms-align-content: space-between;
        -webkit-align-content: space-between;
        align-content: space-between;
      };

      --layout-around-aligned: {
        -ms-flex-line-pack: distribute;  /* IE10 */
        -ms-align-content: space-around;
        -webkit-align-content: space-around;
        align-content: space-around;
      };

      /*******************************
                Other Layout
      *******************************/

      --layout-block: {
        display: block;
      };

      --layout-invisible: {
        visibility: hidden !important;
      };

      --layout-relative: {
        position: relative;
      };

      --layout-fit: {
        position: absolute;
        top: 0;
        right: 0;
        bottom: 0;
        left: 0;
      };

      --layout-scroll: {
        -webkit-overflow-scrolling: touch;
        overflow: auto;
      };

      --layout-fullbleed: {
        margin: 0;
        height: 100vh;
      };

      /* fixed position */

      --layout-fixed-top: {
        position: fixed;
        top: 0;
        left: 0;
        right: 0;
      };

      --layout-fixed-right: {
        position: fixed;
        top: 0;
        right: 0;
        bottom: 0;
      };

      --layout-fixed-bottom: {
        position: fixed;
        right: 0;
        bottom: 0;
        left: 0;
      };

      --layout-fixed-left: {
        position: fixed;
        top: 0;
        bottom: 0;
        left: 0;
      };

    }
  </style>
</custom-style>`;

    template$1.setAttribute('style', 'display: none;');
    document.head.appendChild(template$1.content);

    var style = document.createElement('style');
    style.textContent = '[hidden] { display: none !important; }';
    document.head.appendChild(style);

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    class IronMeta {
      /**
       * @param {{
       *   type: (string|null|undefined),
       *   key: (string|null|undefined),
       *   value: *,
       * }=} options
       */
      constructor(options) {
        IronMeta[' '](options);

        /** @type {string} */
        this.type = (options && options.type) || 'default';
        /** @type {string|null|undefined} */
        this.key = options && options.key;
        if (options && 'value' in options) {
          /** @type {*} */
          this.value = options.value;
        }
      }

      /** @return {*} */
      get value() {
        var type = this.type;
        var key = this.key;

        if (type && key) {
          return IronMeta.types[type] && IronMeta.types[type][key];
        }
      }

      /** @param {*} value */
      set value(value) {
        var type = this.type;
        var key = this.key;

        if (type && key) {
          type = IronMeta.types[type] = IronMeta.types[type] || {};
          if (value == null) {
            delete type[key];
          } else {
            type[key] = value;
          }
        }
      }

      /** @return {!Array<*>} */
      get list() {
        var type = this.type;

        if (type) {
          var items = IronMeta.types[this.type];
          if (!items) {
            return [];
          }

          return Object.keys(items).map(function(key) {
            return metaDatas[this.type][key];
          }, this);
        }
      }

      /**
       * @param {string} key
       * @return {*}
       */
      byKey(key) {
        this.key = key;
        return this.value;
      }
    }
    // This function is used to convince Closure not to remove constructor calls
    // for instances that are not held anywhere. For example, when
    // `new IronMeta({...})` is used only for the side effect of adding a value.
    IronMeta[' '] = function() {};

    IronMeta.types = {};

    var metaDatas = IronMeta.types;

    /**
    `iron-meta` is a generic element you can use for sharing information across the
    DOM tree. It uses [monostate pattern](http://c2.com/cgi/wiki?MonostatePattern)
    such that any instance of iron-meta has access to the shared information. You
    can use `iron-meta` to share whatever you want (or create an extension [like
    x-meta] for enhancements).

    The `iron-meta` instances containing your actual data can be loaded in an
    import, or constructed in any way you see fit. The only requirement is that you
    create them before you try to access them.

    Examples:

    If I create an instance like this:

        <iron-meta key="info" value="foo/bar"></iron-meta>

    Note that value="foo/bar" is the metadata I've defined. I could define more
    attributes or use child nodes to define additional metadata.

    Now I can access that element (and it's metadata) from any iron-meta instance
    via the byKey method, e.g.

        meta.byKey('info');

    Pure imperative form would be like:

        document.createElement('iron-meta').byKey('info');

    Or, in a Polymer element, you can include a meta in your template:

        <iron-meta id="meta"></iron-meta>
        ...
        this.$.meta.byKey('info');

    @group Iron Elements
    @demo demo/index.html
    @element iron-meta
    */
    Polymer({

      is: 'iron-meta',

      properties: {

        /**
         * The type of meta-data.  All meta-data of the same type is stored
         * together.
         * @type {string}
         */
        type: {
          type: String,
          value: 'default',
        },

        /**
         * The key used to store `value` under the `type` namespace.
         * @type {?string}
         */
        key: {
          type: String,
        },

        /**
         * The meta-data to store or retrieve.
         * @type {*}
         */
        value: {
          type: String,
          notify: true,
        },

        /**
         * If true, `value` is set to the iron-meta instance itself.
         */
        self: {type: Boolean, observer: '_selfChanged'},

        __meta: {type: Boolean, computed: '__computeMeta(type, key, value)'}
      },

      hostAttributes: {hidden: true},

      __computeMeta: function(type, key, value) {
        var meta = new IronMeta({type: type, key: key});

        if (value !== undefined && value !== meta.value) {
          meta.value = value;
        } else if (this.value !== meta.value) {
          this.value = meta.value;
        }

        return meta;
      },

      get list() {
        return this.__meta && this.__meta.list;
      },

      _selfChanged: function(self) {
        if (self) {
          this.value = this;
        }
      },

      /**
       * Retrieves meta data value by key.
       *
       * @method byKey
       * @param {string} key The key of the meta-data to be returned.
       * @return {*}
       */
      byKey: function(key) {
        return new IronMeta({type: this.type, key: key}).value;
      }
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**

    The `iron-icon` element displays an icon. By default an icon renders as a 24px
    square.

    Example using src:

        <iron-icon src="star.png"></iron-icon>

    Example setting size to 32px x 32px:

        <iron-icon class="big" src="big_star.png"></iron-icon>

        <style is="custom-style">
          .big {
            --iron-icon-height: 32px;
            --iron-icon-width: 32px;
          }
        </style>

    The iron elements include several sets of icons. To use the default set of
    icons, import `iron-icons.js` and use the `icon` attribute to specify an icon:

        <script type="module">
          import "@polymer/iron-icons/iron-icons.js";
        </JAVA_SCRIIIIPT/>

        <iron-icon icon="menu"></iron-icon>

    To use a different built-in set of icons, import the specific
    `iron-icons/<iconset>-icons.js`, and specify the icon as `<iconset>:<icon>`.
    For example, to use a communication icon, you would use:

        <script type="module">
          import "@polymer/iron-icons/communication-icons.js";
        </JAVA_SCRIIIIPT/>

        <iron-icon icon="communication:email"></iron-icon>

    You can also create custom icon sets of bitmap or SVG icons.

    Example of using an icon named `cherry` from a custom iconset with the ID
    `fruit`:

        <iron-icon icon="fruit:cherry"></iron-icon>

    See `<iron-iconset>` and `<iron-iconset-svg>` for more information about how to
    create a custom iconset.

    See the `iron-icons` demo to see the icons available in the various iconsets.

    ### Styling

    The following custom properties are available for styling:

    Custom property | Description | Default
    ----------------|-------------|----------
    `--iron-icon` | Mixin applied to the icon | {}
    `--iron-icon-width` | Width of the icon | `24px`
    `--iron-icon-height` | Height of the icon | `24px`
    `--iron-icon-fill-color` | Fill color of the svg icon | `currentcolor`
    `--iron-icon-stroke-color` | Stroke color of the svg icon | none

    @group Iron Elements
    @element iron-icon
    @demo demo/index.html
    @hero hero.svg
    @homepage polymer.github.io
    */
    Polymer({
      _template: html`
    <style>
      :host {
        @apply --layout-inline;
        @apply --layout-center-center;
        position: relative;

        vertical-align: middle;

        fill: var(--iron-icon-fill-color, currentcolor);
        stroke: var(--iron-icon-stroke-color, none);

        width: var(--iron-icon-width, 24px);
        height: var(--iron-icon-height, 24px);
        @apply --iron-icon;
      }

      :host([hidden]) {
        display: none;
      }
    </style>
`,

      is: 'iron-icon',

      properties: {

        /**
         * The name of the icon to use. The name should be of the form:
         * `iconset_name:icon_name`.
         */
        icon: {type: String},

        /**
         * The name of the theme to used, if one is specified by the
         * iconset.
         */
        theme: {type: String},

        /**
         * If using iron-icon without an iconset, you can set the src to be
         * the URL of an individual icon image file. Note that this will take
         * precedence over a given icon attribute.
         */
        src: {type: String},

        /**
         * @type {!IronMeta}
         */
        _meta: {value: Base.create('iron-meta', {type: 'iconset'})}

      },

      observers: [
        '_updateIcon(_meta, isAttached)',
        '_updateIcon(theme, isAttached)',
        '_srcChanged(src, isAttached)',
        '_iconChanged(icon, isAttached)'
      ],

      _DEFAULT_ICONSET: 'icons',

      _iconChanged: function(icon) {
        var parts = (icon || '').split(':');
        this._iconName = parts.pop();
        this._iconsetName = parts.pop() || this._DEFAULT_ICONSET;
        this._updateIcon();
      },

      _srcChanged: function(src) {
        this._updateIcon();
      },

      _usesIconset: function() {
        return this.icon || !this.src;
      },

      /** @suppress {visibility} */
      _updateIcon: function() {
        if (this._usesIconset()) {
          if (this._img && this._img.parentNode) {
            dom(this.root).removeChild(this._img);
          }
          if (this._iconName === '') {
            if (this._iconset) {
              this._iconset.removeIcon(this);
            }
          } else if (this._iconsetName && this._meta) {
            this._iconset = /** @type {?Polymer.Iconset} */ (
                this._meta.byKey(this._iconsetName));
            if (this._iconset) {
              this._iconset.applyIcon(this, this._iconName, this.theme);
              this.unlisten(window, 'iron-iconset-added', '_updateIcon');
            } else {
              this.listen(window, 'iron-iconset-added', '_updateIcon');
            }
          }
        } else {
          if (this._iconset) {
            this._iconset.removeIcon(this);
          }
          if (!this._img) {
            this._img = document.createElement('img');
            this._img.style.width = '100%';
            this._img.style.height = '100%';
            this._img.draggable = false;
          }
          this._img.src = this.src;
          dom(this.root).appendChild(this._img);
        }
      }
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */
    /**
     * The `iron-iconset-svg` element allows users to define their own icon sets
     * that contain svg icons. The svg icon elements should be children of the
     * `iron-iconset-svg` element. Multiple icons should be given distinct id's.
     *
     * Using svg elements to create icons has a few advantages over traditional
     * bitmap graphics like jpg or png. Icons that use svg are vector based so
     * they are resolution independent and should look good on any device. They
     * are stylable via css. Icons can be themed, colorized, and even animated.
     *
     * Example:
     *
     *     <iron-iconset-svg name="my-svg-icons" size="24">
     *       <svg>
     *         <defs>
     *           <g id="shape">
     *             <rect x="12" y="0" width="12" height="24" />
     *             <circle cx="12" cy="12" r="12" />
     *           </g>
     *         </defs>
     *       </svg>
     *     </iron-iconset-svg>
     *
     * This will automatically register the icon set "my-svg-icons" to the iconset
     * database.  To use these icons from within another element, make a
     * `iron-iconset` element and call the `byId` method
     * to retrieve a given iconset. To apply a particular icon inside an
     * element use the `applyIcon` method. For example:
     *
     *     iconset.applyIcon(iconNode, 'car');
     *
     * @element iron-iconset-svg
     * @demo demo/index.html
     * @implements {Polymer.Iconset}
     */
    Polymer({
      is: 'iron-iconset-svg',

      properties: {

        /**
         * The name of the iconset.
         */
        name: {type: String, observer: '_nameChanged'},

        /**
         * The size of an individual icon. Note that icons must be square.
         */
        size: {type: Number, value: 24},

        /**
         * Set to true to enable mirroring of icons where specified when they are
         * stamped. Icons that should be mirrored should be decorated with a
         * `mirror-in-rtl` attribute.
         *
         * NOTE: For performance reasons, direction will be resolved once per
         * document per iconset, so moving icons in and out of RTL subtrees will
         * not cause their mirrored state to change.
         */
        rtlMirroring: {type: Boolean, value: false},

        /**
         * Set to true to measure RTL based on the dir attribute on the body or
         * html elements (measured on document.body or document.documentElement as
         * available).
         */
        useGlobalRtlAttribute: {type: Boolean, value: false}
      },

      created: function() {
        this._meta = new IronMeta({type: 'iconset', key: null, value: null});
      },

      attached: function() {
        this.style.display = 'none';
      },

      /**
       * Construct an array of all icon names in this iconset.
       *
       * @return {!Array} Array of icon names.
       */
      getIconNames: function() {
        this._icons = this._createIconMap();
        return Object.keys(this._icons).map(function(n) {
          return this.name + ':' + n;
        }, this);
      },

      /**
       * Applies an icon to the given element.
       *
       * An svg icon is prepended to the element's shadowRoot if it exists,
       * otherwise to the element itself.
       *
       * If RTL mirroring is enabled, and the icon is marked to be mirrored in
       * RTL, the element will be tested (once and only once ever for each
       * iconset) to determine the direction of the subtree the element is in.
       * This direction will apply to all future icon applications, although only
       * icons marked to be mirrored will be affected.
       *
       * @method applyIcon
       * @param {Element} element Element to which the icon is applied.
       * @param {string} iconName Name of the icon to apply.
       * @return {?Element} The svg element which renders the icon.
       */
      applyIcon: function(element, iconName) {
        // Remove old svg element
        this.removeIcon(element);
        // install new svg element
        var svg = this._cloneIcon(
            iconName, this.rtlMirroring && this._targetIsRTL(element));
        if (svg) {
          // insert svg element into shadow root, if it exists
          var pde = dom(element.root || element);
          pde.insertBefore(svg, pde.childNodes[0]);
          return element._svgIcon = svg;
        }
        return null;
      },

      /**
       * Remove an icon from the given element by undoing the changes effected
       * by `applyIcon`.
       *
       * @param {Element} element The element from which the icon is removed.
       */
      removeIcon: function(element) {
        // Remove old svg element
        if (element._svgIcon) {
          dom(element.root || element).removeChild(element._svgIcon);
          element._svgIcon = null;
        }
      },

      /**
       * Measures and memoizes the direction of the element. Note that this
       * measurement is only done once and the result is memoized for future
       * invocations.
       */
      _targetIsRTL: function(target) {
        if (this.__targetIsRTL == null) {
          if (this.useGlobalRtlAttribute) {
            var globalElement =
                (document.body && document.body.hasAttribute('dir')) ?
                document.body :
                document.documentElement;

            this.__targetIsRTL = globalElement.getAttribute('dir') === 'rtl';
          } else {
            if (target && target.nodeType !== Node.ELEMENT_NODE) {
              target = target.host;
            }

            this.__targetIsRTL =
                target && window.getComputedStyle(target)['direction'] === 'rtl';
          }
        }

        return this.__targetIsRTL;
      },

      /**
       *
       * When name is changed, register iconset metadata
       *
       */
      _nameChanged: function() {
        this._meta.value = null;
        this._meta.key = this.name;
        this._meta.value = this;

        this.async(function() {
          this.fire('iron-iconset-added', this, {node: window});
        });
      },

      /**
       * Create a map of child SVG elements by id.
       *
       * @return {!Object} Map of id's to SVG elements.
       */
      _createIconMap: function() {
        // Objects chained to Object.prototype (`{}`) have members. Specifically,
        // on FF there is a `watch` method that confuses the icon map, so we
        // need to use a null-based object here.
        var icons = Object.create(null);
        dom(this).querySelectorAll('[id]').forEach(function(icon) {
          icons[icon.id] = icon;
        });
        return icons;
      },

      /**
       * Produce installable clone of the SVG element matching `id` in this
       * iconset, or `undefined` if there is no matching element.
       *
       * @return {Element} Returns an installable clone of the SVG element
       * matching `id`.
       */
      _cloneIcon: function(id, mirrorAllowed) {
        // create the icon map on-demand, since the iconset itself has no discrete
        // signal to know when it's children are fully parsed
        this._icons = this._icons || this._createIconMap();
        return this._prepareSvgClone(this._icons[id], this.size, mirrorAllowed);
      },

      /**
       * @param {Element} sourceSvg
       * @param {number} size
       * @param {Boolean} mirrorAllowed
       * @return {Element}
       */
      _prepareSvgClone: function(sourceSvg, size, mirrorAllowed) {
        if (sourceSvg) {
          var content = sourceSvg.cloneNode(true),
              svg = document.createElementNS('http://www.w3.org/2000/svg', 'svg'),
              viewBox =
                  content.getAttribute('viewBox') || '0 0 ' + size + ' ' + size,
              cssText =
                  'pointer-events: none; display: block; width: 100%; height: 100%;';

          if (mirrorAllowed && content.hasAttribute('mirror-in-rtl')) {
            cssText +=
                '-webkit-transform:scale(-1,1);transform:scale(-1,1);transform-origin:center;';
          }

          svg.setAttribute('viewBox', viewBox);
          svg.setAttribute('preserveAspectRatio', 'xMidYMid meet');
          svg.setAttribute('focusable', 'false');
          // TODO(dfreedm): `pointer-events: none` works around
          // https://crbug.com/370136
          // TODO(sjmiles): inline style may not be ideal, but avoids requiring a
          // shadow-root
          svg.style.cssText = cssText;
          svg.appendChild(content).removeAttribute('id');
          return svg;
        }
        return null;
      }

    });

    /**
    @license
    Copyright (c) 2014 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    const template$2 = html`<iron-iconset-svg name="image" size="24">
<svg><defs>
<g id="add-a-photo"><path d="M3 4V1h2v3h3v2H5v3H3V6H0V4h3zm3 6V7h3V4h7l1.83 2H21c1.1 0 2 .9 2 2v12c0 1.1-.9 2-2 2H5c-1.1 0-2-.9-2-2V10h3zm7 9c2.76 0 5-2.24 5-5s-2.24-5-5-5-5 2.24-5 5 2.24 5 5 5zm-3.2-5c0 1.77 1.43 3.2 3.2 3.2s3.2-1.43 3.2-3.2-1.43-3.2-3.2-3.2-3.2 1.43-3.2 3.2z"></path></g>
<g id="add-to-photos"><path d="M4 6H2v14c0 1.1.9 2 2 2h14v-2H4V6zm16-4H8c-1.1 0-2 .9-2 2v12c0 1.1.9 2 2 2h12c1.1 0 2-.9 2-2V4c0-1.1-.9-2-2-2zm-1 9h-4v4h-2v-4H9V9h4V5h2v4h4v2z"></path></g>
<g id="adjust"><path d="M12 2C6.49 2 2 6.49 2 12s4.49 10 10 10 10-4.49 10-10S17.51 2 12 2zm0 18c-4.41 0-8-3.59-8-8s3.59-8 8-8 8 3.59 8 8-3.59 8-8 8zm3-8c0 1.66-1.34 3-3 3s-3-1.34-3-3 1.34-3 3-3 3 1.34 3 3z"></path></g>
<g id="assistant"><path d="M19 2H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h4l3 3 3-3h4c1.1 0 2-.9 2-2V4c0-1.1-.9-2-2-2zm-5.12 10.88L12 17l-1.88-4.12L6 11l4.12-1.88L12 5l1.88 4.12L18 11l-4.12 1.88z"></path></g>
<g id="assistant-photo"><path d="M14.4 6L14 4H5v17h2v-7h5.6l.4 2h7V6z"></path></g>
<g id="audiotrack"><path d="M12 3v9.28c-.47-.17-.97-.28-1.5-.28C8.01 12 6 14.01 6 16.5S8.01 21 10.5 21c2.31 0 4.2-1.75 4.45-4H15V6h4V3h-7z"></path></g>
<g id="blur-circular"><path d="M10 9c-.55 0-1 .45-1 1s.45 1 1 1 1-.45 1-1-.45-1-1-1zm0 4c-.55 0-1 .45-1 1s.45 1 1 1 1-.45 1-1-.45-1-1-1zM7 9.5c-.28 0-.5.22-.5.5s.22.5.5.5.5-.22.5-.5-.22-.5-.5-.5zm3 7c-.28 0-.5.22-.5.5s.22.5.5.5.5-.22.5-.5-.22-.5-.5-.5zm-3-3c-.28 0-.5.22-.5.5s.22.5.5.5.5-.22.5-.5-.22-.5-.5-.5zm3-6c.28 0 .5-.22.5-.5s-.22-.5-.5-.5-.5.22-.5.5.22.5.5.5zM14 9c-.55 0-1 .45-1 1s.45 1 1 1 1-.45 1-1-.45-1-1-1zm0-1.5c.28 0 .5-.22.5-.5s-.22-.5-.5-.5-.5.22-.5.5.22.5.5.5zm3 6c-.28 0-.5.22-.5.5s.22.5.5.5.5-.22.5-.5-.22-.5-.5-.5zm0-4c-.28 0-.5.22-.5.5s.22.5.5.5.5-.22.5-.5-.22-.5-.5-.5zM12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm0 18c-4.42 0-8-3.58-8-8s3.58-8 8-8 8 3.58 8 8-3.58 8-8 8zm2-3.5c-.28 0-.5.22-.5.5s.22.5.5.5.5-.22.5-.5-.22-.5-.5-.5zm0-3.5c-.55 0-1 .45-1 1s.45 1 1 1 1-.45 1-1-.45-1-1-1z"></path></g>
<g id="blur-linear"><path d="M5 17.5c.83 0 1.5-.67 1.5-1.5s-.67-1.5-1.5-1.5-1.5.67-1.5 1.5.67 1.5 1.5 1.5zM9 13c.55 0 1-.45 1-1s-.45-1-1-1-1 .45-1 1 .45 1 1 1zm0-4c.55 0 1-.45 1-1s-.45-1-1-1-1 .45-1 1 .45 1 1 1zM3 21h18v-2H3v2zM5 9.5c.83 0 1.5-.67 1.5-1.5S5.83 6.5 5 6.5 3.5 7.17 3.5 8 4.17 9.5 5 9.5zm0 4c.83 0 1.5-.67 1.5-1.5s-.67-1.5-1.5-1.5-1.5.67-1.5 1.5.67 1.5 1.5 1.5zM9 17c.55 0 1-.45 1-1s-.45-1-1-1-1 .45-1 1 .45 1 1 1zm8-.5c.28 0 .5-.22.5-.5s-.22-.5-.5-.5-.5.22-.5.5.22.5.5.5zM3 3v2h18V3H3zm14 5.5c.28 0 .5-.22.5-.5s-.22-.5-.5-.5-.5.22-.5.5.22.5.5.5zm0 4c.28 0 .5-.22.5-.5s-.22-.5-.5-.5-.5.22-.5.5.22.5.5.5zM13 9c.55 0 1-.45 1-1s-.45-1-1-1-1 .45-1 1 .45 1 1 1zm0 4c.55 0 1-.45 1-1s-.45-1-1-1-1 .45-1 1 .45 1 1 1zm0 4c.55 0 1-.45 1-1s-.45-1-1-1-1 .45-1 1 .45 1 1 1z"></path></g>
<g id="blur-off"><path d="M14 7c.55 0 1-.45 1-1s-.45-1-1-1-1 .45-1 1 .45 1 1 1zm-.2 4.48l.2.02c.83 0 1.5-.67 1.5-1.5s-.67-1.5-1.5-1.5-1.5.67-1.5 1.5l.02.2c.09.67.61 1.19 1.28 1.28zM14 3.5c.28 0 .5-.22.5-.5s-.22-.5-.5-.5-.5.22-.5.5.22.5.5.5zm-4 0c.28 0 .5-.22.5-.5s-.22-.5-.5-.5-.5.22-.5.5.22.5.5.5zm11 7c.28 0 .5-.22.5-.5s-.22-.5-.5-.5-.5.22-.5.5.22.5.5.5zM10 7c.55 0 1-.45 1-1s-.45-1-1-1-1 .45-1 1 .45 1 1 1zm8 8c.55 0 1-.45 1-1s-.45-1-1-1-1 .45-1 1 .45 1 1 1zm0-4c.55 0 1-.45 1-1s-.45-1-1-1-1 .45-1 1 .45 1 1 1zm0-4c.55 0 1-.45 1-1s-.45-1-1-1-1 .45-1 1 .45 1 1 1zm-4 13.5c-.28 0-.5.22-.5.5s.22.5.5.5.5-.22.5-.5-.22-.5-.5-.5zM2.5 5.27l3.78 3.78L6 9c-.55 0-1 .45-1 1s.45 1 1 1 1-.45 1-1c0-.1-.03-.19-.06-.28l2.81 2.81c-.71.11-1.25.73-1.25 1.47 0 .83.67 1.5 1.5 1.5.74 0 1.36-.54 1.47-1.25l2.81 2.81c-.09-.03-.18-.06-.28-.06-.55 0-1 .45-1 1s.45 1 1 1 1-.45 1-1c0-.1-.03-.19-.06-.28l3.78 3.78L20 20.23 3.77 4 2.5 5.27zM10 17c-.55 0-1 .45-1 1s.45 1 1 1 1-.45 1-1-.45-1-1-1zm11-3.5c-.28 0-.5.22-.5.5s.22.5.5.5.5-.22.5-.5-.22-.5-.5-.5zM6 13c-.55 0-1 .45-1 1s.45 1 1 1 1-.45 1-1-.45-1-1-1zM3 9.5c-.28 0-.5.22-.5.5s.22.5.5.5.5-.22.5-.5-.22-.5-.5-.5zm7 11c-.28 0-.5.22-.5.5s.22.5.5.5.5-.22.5-.5-.22-.5-.5-.5zM6 17c-.55 0-1 .45-1 1s.45 1 1 1 1-.45 1-1-.45-1-1-1zm-3-3.5c-.28 0-.5.22-.5.5s.22.5.5.5.5-.22.5-.5-.22-.5-.5-.5z"></path></g>
<g id="blur-on"><path d="M6 13c-.55 0-1 .45-1 1s.45 1 1 1 1-.45 1-1-.45-1-1-1zm0 4c-.55 0-1 .45-1 1s.45 1 1 1 1-.45 1-1-.45-1-1-1zm0-8c-.55 0-1 .45-1 1s.45 1 1 1 1-.45 1-1-.45-1-1-1zm-3 .5c-.28 0-.5.22-.5.5s.22.5.5.5.5-.22.5-.5-.22-.5-.5-.5zM6 5c-.55 0-1 .45-1 1s.45 1 1 1 1-.45 1-1-.45-1-1-1zm15 5.5c.28 0 .5-.22.5-.5s-.22-.5-.5-.5-.5.22-.5.5.22.5.5.5zM14 7c.55 0 1-.45 1-1s-.45-1-1-1-1 .45-1 1 .45 1 1 1zm0-3.5c.28 0 .5-.22.5-.5s-.22-.5-.5-.5-.5.22-.5.5.22.5.5.5zm-11 10c-.28 0-.5.22-.5.5s.22.5.5.5.5-.22.5-.5-.22-.5-.5-.5zm7 7c-.28 0-.5.22-.5.5s.22.5.5.5.5-.22.5-.5-.22-.5-.5-.5zm0-17c.28 0 .5-.22.5-.5s-.22-.5-.5-.5-.5.22-.5.5.22.5.5.5zM10 7c.55 0 1-.45 1-1s-.45-1-1-1-1 .45-1 1 .45 1 1 1zm0 5.5c-.83 0-1.5.67-1.5 1.5s.67 1.5 1.5 1.5 1.5-.67 1.5-1.5-.67-1.5-1.5-1.5zm8 .5c-.55 0-1 .45-1 1s.45 1 1 1 1-.45 1-1-.45-1-1-1zm0 4c-.55 0-1 .45-1 1s.45 1 1 1 1-.45 1-1-.45-1-1-1zm0-8c-.55 0-1 .45-1 1s.45 1 1 1 1-.45 1-1-.45-1-1-1zm0-4c-.55 0-1 .45-1 1s.45 1 1 1 1-.45 1-1-.45-1-1-1zm3 8.5c-.28 0-.5.22-.5.5s.22.5.5.5.5-.22.5-.5-.22-.5-.5-.5zM14 17c-.55 0-1 .45-1 1s.45 1 1 1 1-.45 1-1-.45-1-1-1zm0 3.5c-.28 0-.5.22-.5.5s.22.5.5.5.5-.22.5-.5-.22-.5-.5-.5zm-4-12c-.83 0-1.5.67-1.5 1.5s.67 1.5 1.5 1.5 1.5-.67 1.5-1.5-.67-1.5-1.5-1.5zm0 8.5c-.55 0-1 .45-1 1s.45 1 1 1 1-.45 1-1-.45-1-1-1zm4-4.5c-.83 0-1.5.67-1.5 1.5s.67 1.5 1.5 1.5 1.5-.67 1.5-1.5-.67-1.5-1.5-1.5zm0-4c-.83 0-1.5.67-1.5 1.5s.67 1.5 1.5 1.5 1.5-.67 1.5-1.5-.67-1.5-1.5-1.5z"></path></g>
<g id="brightness-1"><circle cx="12" cy="12" r="10"></circle></g>
<g id="brightness-2"><path d="M10 2c-1.82 0-3.53.5-5 1.35C7.99 5.08 10 8.3 10 12s-2.01 6.92-5 8.65C6.47 21.5 8.18 22 10 22c5.52 0 10-4.48 10-10S15.52 2 10 2z"></path></g>
<g id="brightness-3"><path d="M9 2c-1.05 0-2.05.16-3 .46 4.06 1.27 7 5.06 7 9.54 0 4.48-2.94 8.27-7 9.54.95.3 1.95.46 3 .46 5.52 0 10-4.48 10-10S14.52 2 9 2z"></path></g>
<g id="brightness-4"><path d="M20 8.69V4h-4.69L12 .69 8.69 4H4v4.69L.69 12 4 15.31V20h4.69L12 23.31 15.31 20H20v-4.69L23.31 12 20 8.69zM12 18c-.89 0-1.74-.2-2.5-.55C11.56 16.5 13 14.42 13 12s-1.44-4.5-3.5-5.45C10.26 6.2 11.11 6 12 6c3.31 0 6 2.69 6 6s-2.69 6-6 6z"></path></g>
<g id="brightness-5"><path d="M20 15.31L23.31 12 20 8.69V4h-4.69L12 .69 8.69 4H4v4.69L.69 12 4 15.31V20h4.69L12 23.31 15.31 20H20v-4.69zM12 18c-3.31 0-6-2.69-6-6s2.69-6 6-6 6 2.69 6 6-2.69 6-6 6z"></path></g>
<g id="brightness-6"><path d="M20 15.31L23.31 12 20 8.69V4h-4.69L12 .69 8.69 4H4v4.69L.69 12 4 15.31V20h4.69L12 23.31 15.31 20H20v-4.69zM12 18V6c3.31 0 6 2.69 6 6s-2.69 6-6 6z"></path></g>
<g id="brightness-7"><path d="M20 8.69V4h-4.69L12 .69 8.69 4H4v4.69L.69 12 4 15.31V20h4.69L12 23.31 15.31 20H20v-4.69L23.31 12 20 8.69zM12 18c-3.31 0-6-2.69-6-6s2.69-6 6-6 6 2.69 6 6-2.69 6-6 6zm0-10c-2.21 0-4 1.79-4 4s1.79 4 4 4 4-1.79 4-4-1.79-4-4-4z"></path></g>
<g id="broken-image"><path d="M21 5v6.59l-3-3.01-4 4.01-4-4-4 4-3-3.01V5c0-1.1.9-2 2-2h14c1.1 0 2 .9 2 2zm-3 6.42l3 3.01V19c0 1.1-.9 2-2 2H5c-1.1 0-2-.9-2-2v-6.58l3 2.99 4-4 4 4 4-3.99z"></path></g>
<g id="brush"><path d="M7 14c-1.66 0-3 1.34-3 3 0 1.31-1.16 2-2 2 .92 1.22 2.49 2 4 2 2.21 0 4-1.79 4-4 0-1.66-1.34-3-3-3zm13.71-9.37l-1.34-1.34c-.39-.39-1.02-.39-1.41 0L9 12.25 11.75 15l8.96-8.96c.39-.39.39-1.02 0-1.41z"></path></g>
<g id="burst-mode"><path d="M1 5h2v14H1zm4 0h2v14H5zm17 0H10c-.55 0-1 .45-1 1v12c0 .55.45 1 1 1h12c.55 0 1-.45 1-1V6c0-.55-.45-1-1-1zM11 17l2.5-3.15L15.29 16l2.5-3.22L21 17H11z"></path></g>
<g id="camera"><path d="M9.4 10.5l4.77-8.26C13.47 2.09 12.75 2 12 2c-2.4 0-4.6.85-6.32 2.25l3.66 6.35.06-.1zM21.54 9c-.92-2.92-3.15-5.26-6-6.34L11.88 9h9.66zm.26 1h-7.49l.29.5 4.76 8.25C21 16.97 22 14.61 22 12c0-.69-.07-1.35-.2-2zM8.54 12l-3.9-6.75C3.01 7.03 2 9.39 2 12c0 .69.07 1.35.2 2h7.49l-1.15-2zm-6.08 3c.92 2.92 3.15 5.26 6 6.34L12.12 15H2.46zm11.27 0l-3.9 6.76c.7.15 1.42.24 2.17.24 2.4 0 4.6-.85 6.32-2.25l-3.66-6.35-.93 1.6z"></path></g>
<g id="camera-alt"><circle cx="12" cy="12" r="3.2"></circle><path d="M9 2L7.17 4H4c-1.1 0-2 .9-2 2v12c0 1.1.9 2 2 2h16c1.1 0 2-.9 2-2V6c0-1.1-.9-2-2-2h-3.17L15 2H9zm3 15c-2.76 0-5-2.24-5-5s2.24-5 5-5 5 2.24 5 5-2.24 5-5 5z"></path></g>
<g id="camera-front"><path d="M10 20H5v2h5v2l3-3-3-3v2zm4 0v2h5v-2h-5zM12 8c1.1 0 2-.9 2-2s-.9-2-2-2-1.99.9-1.99 2S10.9 8 12 8zm5-8H7C5.9 0 5 .9 5 2v14c0 1.1.9 2 2 2h10c1.1 0 2-.9 2-2V2c0-1.1-.9-2-2-2zM7 2h10v10.5c0-1.67-3.33-2.5-5-2.5s-5 .83-5 2.5V2z"></path></g>
<g id="camera-rear"><path d="M10 20H5v2h5v2l3-3-3-3v2zm4 0v2h5v-2h-5zm3-20H7C5.9 0 5 .9 5 2v14c0 1.1.9 2 2 2h10c1.1 0 2-.9 2-2V2c0-1.1-.9-2-2-2zm-5 6c-1.11 0-2-.9-2-2s.89-2 1.99-2 2 .9 2 2C14 5.1 13.1 6 12 6z"></path></g>
<g id="camera-roll"><path d="M14 5c0-1.1-.9-2-2-2h-1V2c0-.55-.45-1-1-1H6c-.55 0-1 .45-1 1v1H4c-1.1 0-2 .9-2 2v15c0 1.1.9 2 2 2h8c1.1 0 2-.9 2-2h8V5h-8zm-2 13h-2v-2h2v2zm0-9h-2V7h2v2zm4 9h-2v-2h2v2zm0-9h-2V7h2v2zm4 9h-2v-2h2v2zm0-9h-2V7h2v2z"></path></g>
<g id="center-focus-strong"><path d="M12 8c-2.21 0-4 1.79-4 4s1.79 4 4 4 4-1.79 4-4-1.79-4-4-4zm-7 7H3v4c0 1.1.9 2 2 2h4v-2H5v-4zM5 5h4V3H5c-1.1 0-2 .9-2 2v4h2V5zm14-2h-4v2h4v4h2V5c0-1.1-.9-2-2-2zm0 16h-4v2h4c1.1 0 2-.9 2-2v-4h-2v4z"></path></g>
<g id="center-focus-weak"><path d="M5 15H3v4c0 1.1.9 2 2 2h4v-2H5v-4zM5 5h4V3H5c-1.1 0-2 .9-2 2v4h2V5zm14-2h-4v2h4v4h2V5c0-1.1-.9-2-2-2zm0 16h-4v2h4c1.1 0 2-.9 2-2v-4h-2v4zM12 8c-2.21 0-4 1.79-4 4s1.79 4 4 4 4-1.79 4-4-1.79-4-4-4zm0 6c-1.1 0-2-.9-2-2s.9-2 2-2 2 .9 2 2-.9 2-2 2z"></path></g>
<g id="collections"><path d="M22 16V4c0-1.1-.9-2-2-2H8c-1.1 0-2 .9-2 2v12c0 1.1.9 2 2 2h12c1.1 0 2-.9 2-2zm-11-4l2.03 2.71L16 11l4 5H8l3-4zM2 6v14c0 1.1.9 2 2 2h14v-2H4V6H2z"></path></g>
<g id="collections-bookmark"><path d="M4 6H2v14c0 1.1.9 2 2 2h14v-2H4V6zm16-4H8c-1.1 0-2 .9-2 2v12c0 1.1.9 2 2 2h12c1.1 0 2-.9 2-2V4c0-1.1-.9-2-2-2zm0 10l-2.5-1.5L15 12V4h5v8z"></path></g>
<g id="color-lens"><path d="M12 3c-4.97 0-9 4.03-9 9s4.03 9 9 9c.83 0 1.5-.67 1.5-1.5 0-.39-.15-.74-.39-1.01-.23-.26-.38-.61-.38-.99 0-.83.67-1.5 1.5-1.5H16c2.76 0 5-2.24 5-5 0-4.42-4.03-8-9-8zm-5.5 9c-.83 0-1.5-.67-1.5-1.5S5.67 9 6.5 9 8 9.67 8 10.5 7.33 12 6.5 12zm3-4C8.67 8 8 7.33 8 6.5S8.67 5 9.5 5s1.5.67 1.5 1.5S10.33 8 9.5 8zm5 0c-.83 0-1.5-.67-1.5-1.5S13.67 5 14.5 5s1.5.67 1.5 1.5S15.33 8 14.5 8zm3 4c-.83 0-1.5-.67-1.5-1.5S16.67 9 17.5 9s1.5.67 1.5 1.5-.67 1.5-1.5 1.5z"></path></g>
<g id="colorize"><path d="M20.71 5.63l-2.34-2.34c-.39-.39-1.02-.39-1.41 0l-3.12 3.12-1.93-1.91-1.41 1.41 1.42 1.42L3 16.25V21h4.75l8.92-8.92 1.42 1.42 1.41-1.41-1.92-1.92 3.12-3.12c.4-.4.4-1.03.01-1.42zM6.92 19L5 17.08l8.06-8.06 1.92 1.92L6.92 19z"></path></g>
<g id="compare"><path d="M10 3H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h5v2h2V1h-2v2zm0 15H5l5-6v6zm9-15h-5v2h5v13l-5-6v9h5c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2z"></path></g>
<g id="control-point"><path d="M13 7h-2v4H7v2h4v4h2v-4h4v-2h-4V7zm-1-5C6.49 2 2 6.49 2 12s4.49 10 10 10 10-4.49 10-10S17.51 2 12 2zm0 18c-4.41 0-8-3.59-8-8s3.59-8 8-8 8 3.59 8 8-3.59 8-8 8z"></path></g>
<g id="control-point-duplicate"><path d="M16 8h-2v3h-3v2h3v3h2v-3h3v-2h-3zM2 12c0-2.79 1.64-5.2 4.01-6.32V3.52C2.52 4.76 0 8.09 0 12s2.52 7.24 6.01 8.48v-2.16C3.64 17.2 2 14.79 2 12zm13-9c-4.96 0-9 4.04-9 9s4.04 9 9 9 9-4.04 9-9-4.04-9-9-9zm0 16c-3.86 0-7-3.14-7-7s3.14-7 7-7 7 3.14 7 7-3.14 7-7 7z"></path></g>
<g id="crop"><path d="M17 15h2V7c0-1.1-.9-2-2-2H9v2h8v8zM7 17V1H5v4H1v2h4v10c0 1.1.9 2 2 2h10v4h2v-4h4v-2H7z"></path></g>
<g id="crop-16-9"><path d="M19 6H5c-1.1 0-2 .9-2 2v8c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V8c0-1.1-.9-2-2-2zm0 10H5V8h14v8z"></path></g>
<g id="crop-3-2"><path d="M19 4H5c-1.1 0-2 .9-2 2v12c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V6c0-1.1-.9-2-2-2zm0 14H5V6h14v12z"></path></g>
<g id="crop-5-4"><path d="M19 5H5c-1.1 0-2 .9-2 2v10c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V7c0-1.1-.9-2-2-2zm0 12H5V7h14v10z"></path></g>
<g id="crop-7-5"><path d="M19 7H5c-1.1 0-2 .9-2 2v6c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V9c0-1.1-.9-2-2-2zm0 8H5V9h14v6z"></path></g>
<g id="crop-din"><path d="M19 3H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm0 16H5V5h14v14z"></path></g>
<g id="crop-free"><path d="M3 5v4h2V5h4V3H5c-1.1 0-2 .9-2 2zm2 10H3v4c0 1.1.9 2 2 2h4v-2H5v-4zm14 4h-4v2h4c1.1 0 2-.9 2-2v-4h-2v4zm0-16h-4v2h4v4h2V5c0-1.1-.9-2-2-2z"></path></g>
<g id="crop-landscape"><path d="M19 5H5c-1.1 0-2 .9-2 2v10c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V7c0-1.1-.9-2-2-2zm0 12H5V7h14v10z"></path></g>
<g id="crop-original"><path d="M19 3H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm0 16H5V5h14v14zm-5.04-6.71l-2.75 3.54-1.96-2.36L6.5 17h11l-3.54-4.71z"></path></g>
<g id="crop-portrait"><path d="M17 3H7c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h10c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm0 16H7V5h10v14z"></path></g>
<g id="crop-rotate"><path d="M7.47 21.49C4.2 19.93 1.86 16.76 1.5 13H0c.51 6.16 5.66 11 11.95 11 .23 0 .44-.02.66-.03L8.8 20.15l-1.33 1.34zM12.05 0c-.23 0-.44.02-.66.04l3.81 3.81 1.33-1.33C19.8 4.07 22.14 7.24 22.5 11H24c-.51-6.16-5.66-11-11.95-11zM16 14h2V8c0-1.11-.9-2-2-2h-6v2h6v6zm-8 2V4H6v2H4v2h2v8c0 1.1.89 2 2 2h8v2h2v-2h2v-2H8z"></path></g>
<g id="crop-square"><path d="M18 4H6c-1.1 0-2 .9-2 2v12c0 1.1.9 2 2 2h12c1.1 0 2-.9 2-2V6c0-1.1-.9-2-2-2zm0 14H6V6h12v12z"></path></g>
<g id="dehaze"><path d="M2 15.5v2h20v-2H2zm0-5v2h20v-2H2zm0-5v2h20v-2H2z"></path></g>
<g id="details"><path d="M3 4l9 16 9-16H3zm3.38 2h11.25L12 16 6.38 6z"></path></g>
<g id="edit"><path d="M3 17.25V21h3.75L17.81 9.94l-3.75-3.75L3 17.25zM20.71 7.04c.39-.39.39-1.02 0-1.41l-2.34-2.34c-.39-.39-1.02-.39-1.41 0l-1.83 1.83 3.75 3.75 1.83-1.83z"></path></g>
<g id="exposure"><path d="M15 17v2h2v-2h2v-2h-2v-2h-2v2h-2v2h2zm5-15H4c-1.1 0-2 .9-2 2v16c0 1.1.9 2 2 2h16c1.1 0 2-.9 2-2V4c0-1.1-.9-2-2-2zM5 5h6v2H5V5zm15 15H4L20 4v16z"></path></g>
<g id="exposure-neg-1"><path d="M4 11v2h8v-2H4zm15 7h-2V7.38L14 8.4V6.7L18.7 5h.3v13z"></path></g>
<g id="exposure-neg-2"><path d="M15.05 16.29l2.86-3.07c.38-.39.72-.79 1.04-1.18.32-.39.59-.78.82-1.17.23-.39.41-.78.54-1.17s.19-.79.19-1.18c0-.53-.09-1.02-.27-1.46-.18-.44-.44-.81-.78-1.11-.34-.31-.77-.54-1.26-.71-.51-.16-1.08-.24-1.72-.24-.69 0-1.31.11-1.85.32-.54.21-1 .51-1.36.88-.37.37-.65.8-.84 1.3-.18.47-.27.97-.28 1.5h2.14c.01-.31.05-.6.13-.87.09-.29.23-.54.4-.75.18-.21.41-.37.68-.49.27-.12.6-.18.96-.18.31 0 .58.05.81.15.23.1.43.25.59.43.16.18.28.4.37.65.08.25.13.52.13.81 0 .22-.03.43-.08.65-.06.22-.15.45-.29.7-.14.25-.32.53-.56.83-.23.3-.52.65-.88 1.03l-4.17 4.55V18H21v-1.71h-5.95zM2 11v2h8v-2H2z"></path></g>
<g id="exposure-plus-1"><path d="M10 7H8v4H4v2h4v4h2v-4h4v-2h-4V7zm10 11h-2V7.38L15 8.4V6.7L19.7 5h.3v13z"></path></g>
<g id="exposure-plus-2"><path d="M16.05 16.29l2.86-3.07c.38-.39.72-.79 1.04-1.18.32-.39.59-.78.82-1.17.23-.39.41-.78.54-1.17.13-.39.19-.79.19-1.18 0-.53-.09-1.02-.27-1.46-.18-.44-.44-.81-.78-1.11-.34-.31-.77-.54-1.26-.71-.51-.16-1.08-.24-1.72-.24-.69 0-1.31.11-1.85.32-.54.21-1 .51-1.36.88-.37.37-.65.8-.84 1.3-.18.47-.27.97-.28 1.5h2.14c.01-.31.05-.6.13-.87.09-.29.23-.54.4-.75.18-.21.41-.37.68-.49.27-.12.6-.18.96-.18.31 0 .58.05.81.15.23.1.43.25.59.43.16.18.28.4.37.65.08.25.13.52.13.81 0 .22-.03.43-.08.65-.06.22-.15.45-.29.7-.14.25-.32.53-.56.83-.23.3-.52.65-.88 1.03l-4.17 4.55V18H22v-1.71h-5.95zM8 7H6v4H2v2h4v4h2v-4h4v-2H8V7z"></path></g>
<g id="exposure-zero"><path d="M16.14 12.5c0 1-.1 1.85-.3 2.55-.2.7-.48 1.27-.83 1.7-.36.44-.79.75-1.3.95-.51.2-1.07.3-1.7.3-.62 0-1.18-.1-1.69-.3-.51-.2-.95-.51-1.31-.95-.36-.44-.65-1.01-.85-1.7-.2-.7-.3-1.55-.3-2.55v-2.04c0-1 .1-1.85.3-2.55.2-.7.48-1.26.84-1.69.36-.43.8-.74 1.31-.93C10.81 5.1 11.38 5 12 5c.63 0 1.19.1 1.7.29.51.19.95.5 1.31.93.36.43.64.99.84 1.69.2.7.3 1.54.3 2.55v2.04zm-2.11-2.36c0-.64-.05-1.18-.13-1.62-.09-.44-.22-.79-.4-1.06-.17-.27-.39-.46-.64-.58-.25-.13-.54-.19-.86-.19-.32 0-.61.06-.86.18s-.47.31-.64.58c-.17.27-.31.62-.4 1.06s-.13.98-.13 1.62v2.67c0 .64.05 1.18.14 1.62.09.45.23.81.4 1.09s.39.48.64.61.54.19.87.19c.33 0 .62-.06.87-.19s.46-.33.63-.61c.17-.28.3-.64.39-1.09.09-.45.13-.99.13-1.62v-2.66z"></path></g>
<g id="filter"><path d="M15.96 10.29l-2.75 3.54-1.96-2.36L8.5 15h11l-3.54-4.71zM3 5H1v16c0 1.1.9 2 2 2h16v-2H3V5zm18-4H7c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V3c0-1.1-.9-2-2-2zm0 16H7V3h14v14z"></path></g>
<g id="filter-1"><path d="M3 5H1v16c0 1.1.9 2 2 2h16v-2H3V5zm11 10h2V5h-4v2h2v8zm7-14H7c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V3c0-1.1-.9-2-2-2zm0 16H7V3h14v14z"></path></g>
<g id="filter-2"><path d="M3 5H1v16c0 1.1.9 2 2 2h16v-2H3V5zm18-4H7c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V3c0-1.1-.9-2-2-2zm0 16H7V3h14v14zm-4-4h-4v-2h2c1.1 0 2-.89 2-2V7c0-1.11-.9-2-2-2h-4v2h4v2h-2c-1.1 0-2 .89-2 2v4h6v-2z"></path></g>
<g id="filter-3"><path d="M21 1H7c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V3c0-1.1-.9-2-2-2zm0 16H7V3h14v14zM3 5H1v16c0 1.1.9 2 2 2h16v-2H3V5zm14 8v-1.5c0-.83-.67-1.5-1.5-1.5.83 0 1.5-.67 1.5-1.5V7c0-1.11-.9-2-2-2h-4v2h4v2h-2v2h2v2h-4v2h4c1.1 0 2-.89 2-2z"></path></g>
<g id="filter-4"><path d="M3 5H1v16c0 1.1.9 2 2 2h16v-2H3V5zm12 10h2V5h-2v4h-2V5h-2v6h4v4zm6-14H7c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V3c0-1.1-.9-2-2-2zm0 16H7V3h14v14z"></path></g>
<g id="filter-5"><path d="M21 1H7c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V3c0-1.1-.9-2-2-2zm0 16H7V3h14v14zM3 5H1v16c0 1.1.9 2 2 2h16v-2H3V5zm14 8v-2c0-1.11-.9-2-2-2h-2V7h4V5h-6v6h4v2h-4v2h4c1.1 0 2-.89 2-2z"></path></g>
<g id="filter-6"><path d="M3 5H1v16c0 1.1.9 2 2 2h16v-2H3V5zm18-4H7c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V3c0-1.1-.9-2-2-2zm0 16H7V3h14v14zm-8-2h2c1.1 0 2-.89 2-2v-2c0-1.11-.9-2-2-2h-2V7h4V5h-4c-1.1 0-2 .89-2 2v6c0 1.11.9 2 2 2zm0-4h2v2h-2v-2z"></path></g>
<g id="filter-7"><path d="M3 5H1v16c0 1.1.9 2 2 2h16v-2H3V5zm18-4H7c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V3c0-1.1-.9-2-2-2zm0 16H7V3h14v14zm-8-2l4-8V5h-6v2h4l-4 8h2z"></path></g>
<g id="filter-8"><path d="M3 5H1v16c0 1.1.9 2 2 2h16v-2H3V5zm18-4H7c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V3c0-1.1-.9-2-2-2zm0 16H7V3h14v14zm-8-2h2c1.1 0 2-.89 2-2v-1.5c0-.83-.67-1.5-1.5-1.5.83 0 1.5-.67 1.5-1.5V7c0-1.11-.9-2-2-2h-2c-1.1 0-2 .89-2 2v1.5c0 .83.67 1.5 1.5 1.5-.83 0-1.5.67-1.5 1.5V13c0 1.11.9 2 2 2zm0-8h2v2h-2V7zm0 4h2v2h-2v-2z"></path></g>
<g id="filter-9"><path d="M3 5H1v16c0 1.1.9 2 2 2h16v-2H3V5zm18-4H7c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V3c0-1.1-.9-2-2-2zm0 16H7V3h14v14zM15 5h-2c-1.1 0-2 .89-2 2v2c0 1.11.9 2 2 2h2v2h-4v2h4c1.1 0 2-.89 2-2V7c0-1.11-.9-2-2-2zm0 4h-2V7h2v2z"></path></g>
<g id="filter-9-plus"><path d="M3 5H1v16c0 1.1.9 2 2 2h16v-2H3V5zm11 7V8c0-1.11-.9-2-2-2h-1c-1.1 0-2 .89-2 2v1c0 1.11.9 2 2 2h1v1H9v2h3c1.1 0 2-.89 2-2zm-3-3V8h1v1h-1zm10-8H7c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V3c0-1.1-.9-2-2-2zm0 8h-2V7h-2v2h-2v2h2v2h2v-2h2v6H7V3h14v6z"></path></g>
<g id="filter-b-and-w"><path d="M19 3H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm0 16l-7-8v8H5l7-8V5h7v14z"></path></g>
<g id="filter-center-focus"><path d="M5 15H3v4c0 1.1.9 2 2 2h4v-2H5v-4zM5 5h4V3H5c-1.1 0-2 .9-2 2v4h2V5zm14-2h-4v2h4v4h2V5c0-1.1-.9-2-2-2zm0 16h-4v2h4c1.1 0 2-.9 2-2v-4h-2v4zM12 9c-1.66 0-3 1.34-3 3s1.34 3 3 3 3-1.34 3-3-1.34-3-3-3z"></path></g>
<g id="filter-drama"><path d="M19.35 10.04C18.67 6.59 15.64 4 12 4 9.11 4 6.61 5.64 5.36 8.04 2.35 8.36 0 10.9 0 14c0 3.31 2.69 6 6 6h13c2.76 0 5-2.24 5-5 0-2.64-2.05-4.78-4.65-4.96zM19 18H6c-2.21 0-4-1.79-4-4s1.79-4 4-4 4 1.79 4 4h2c0-2.76-1.86-5.08-4.4-5.78C8.61 6.88 10.2 6 12 6c3.03 0 5.5 2.47 5.5 5.5v.5H19c1.65 0 3 1.35 3 3s-1.35 3-3 3z"></path></g>
<g id="filter-frames"><path d="M20 4h-4l-4-4-4 4H4c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h16c1.1 0 2-.9 2-2V6c0-1.1-.9-2-2-2zm0 16H4V6h4.52l3.52-3.5L15.52 6H20v14zM18 8H6v10h12"></path></g>
<g id="filter-hdr"><path d="M14 6l-3.75 5 2.85 3.8-1.6 1.2C9.81 13.75 7 10 7 10l-6 8h22L14 6z"></path></g>
<g id="filter-none"><path d="M3 5H1v16c0 1.1.9 2 2 2h16v-2H3V5zm18-4H7c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V3c0-1.1-.9-2-2-2zm0 16H7V3h14v14z"></path></g>
<g id="filter-tilt-shift"><path d="M11 4.07V2.05c-2.01.2-3.84 1-5.32 2.21L7.1 5.69c1.11-.86 2.44-1.44 3.9-1.62zm7.32.19C16.84 3.05 15.01 2.25 13 2.05v2.02c1.46.18 2.79.76 3.9 1.62l1.42-1.43zM19.93 11h2.02c-.2-2.01-1-3.84-2.21-5.32L18.31 7.1c.86 1.11 1.44 2.44 1.62 3.9zM5.69 7.1L4.26 5.68C3.05 7.16 2.25 8.99 2.05 11h2.02c.18-1.46.76-2.79 1.62-3.9zM4.07 13H2.05c.2 2.01 1 3.84 2.21 5.32l1.43-1.43c-.86-1.1-1.44-2.43-1.62-3.89zM15 12c0-1.66-1.34-3-3-3s-3 1.34-3 3 1.34 3 3 3 3-1.34 3-3zm3.31 4.9l1.43 1.43c1.21-1.48 2.01-3.32 2.21-5.32h-2.02c-.18 1.45-.76 2.78-1.62 3.89zM13 19.93v2.02c2.01-.2 3.84-1 5.32-2.21l-1.43-1.43c-1.1.86-2.43 1.44-3.89 1.62zm-7.32-.19C7.16 20.95 9 21.75 11 21.95v-2.02c-1.46-.18-2.79-.76-3.9-1.62l-1.42 1.43z"></path></g>
<g id="filter-vintage"><path d="M18.7 12.4c-.28-.16-.57-.29-.86-.4.29-.11.58-.24.86-.4 1.92-1.11 2.99-3.12 3-5.19-1.79-1.03-4.07-1.11-6 0-.28.16-.54.35-.78.54.05-.31.08-.63.08-.95 0-2.22-1.21-4.15-3-5.19C10.21 1.85 9 3.78 9 6c0 .32.03.64.08.95-.24-.2-.5-.39-.78-.55-1.92-1.11-4.2-1.03-6 0 0 2.07 1.07 4.08 3 5.19.28.16.57.29.86.4-.29.11-.58.24-.86.4-1.92 1.11-2.99 3.12-3 5.19 1.79 1.03 4.07 1.11 6 0 .28-.16.54-.35.78-.54-.05.32-.08.64-.08.96 0 2.22 1.21 4.15 3 5.19 1.79-1.04 3-2.97 3-5.19 0-.32-.03-.64-.08-.95.24.2.5.38.78.54 1.92 1.11 4.2 1.03 6 0-.01-2.07-1.08-4.08-3-5.19zM12 16c-2.21 0-4-1.79-4-4s1.79-4 4-4 4 1.79 4 4-1.79 4-4 4z"></path></g>
<g id="flare"><path d="M7 11H1v2h6v-2zm2.17-3.24L7.05 5.64 5.64 7.05l2.12 2.12 1.41-1.41zM13 1h-2v6h2V1zm5.36 6.05l-1.41-1.41-2.12 2.12 1.41 1.41 2.12-2.12zM17 11v2h6v-2h-6zm-5-2c-1.66 0-3 1.34-3 3s1.34 3 3 3 3-1.34 3-3-1.34-3-3-3zm2.83 7.24l2.12 2.12 1.41-1.41-2.12-2.12-1.41 1.41zm-9.19.71l1.41 1.41 2.12-2.12-1.41-1.41-2.12 2.12zM11 23h2v-6h-2v6z"></path></g>
<g id="flash-auto"><path d="M3 2v12h3v9l7-12H9l4-9H3zm16 0h-2l-3.2 9h1.9l.7-2h3.2l.7 2h1.9L19 2zm-2.15 5.65L18 4l1.15 3.65h-2.3z"></path></g>
<g id="flash-off"><path d="M3.27 3L2 4.27l5 5V13h3v9l3.58-6.14L17.73 20 19 18.73 3.27 3zM17 10h-4l4-8H7v2.18l8.46 8.46L17 10z"></path></g>
<g id="flash-on"><path d="M7 2v11h3v9l7-12h-4l4-8z"></path></g>
<g id="flip"><path d="M15 21h2v-2h-2v2zm4-12h2V7h-2v2zM3 5v14c0 1.1.9 2 2 2h4v-2H5V5h4V3H5c-1.1 0-2 .9-2 2zm16-2v2h2c0-1.1-.9-2-2-2zm-8 20h2V1h-2v22zm8-6h2v-2h-2v2zM15 5h2V3h-2v2zm4 8h2v-2h-2v2zm0 8c1.1 0 2-.9 2-2h-2v2z"></path></g>
<g id="gradient"><path d="M11 9h2v2h-2zm-2 2h2v2H9zm4 0h2v2h-2zm2-2h2v2h-2zM7 9h2v2H7zm12-6H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zM9 18H7v-2h2v2zm4 0h-2v-2h2v2zm4 0h-2v-2h2v2zm2-7h-2v2h2v2h-2v-2h-2v2h-2v-2h-2v2H9v-2H7v2H5v-2h2v-2H5V5h14v6z"></path></g>
<g id="grain"><path d="M10 12c-1.1 0-2 .9-2 2s.9 2 2 2 2-.9 2-2-.9-2-2-2zM6 8c-1.1 0-2 .9-2 2s.9 2 2 2 2-.9 2-2-.9-2-2-2zm0 8c-1.1 0-2 .9-2 2s.9 2 2 2 2-.9 2-2-.9-2-2-2zm12-8c1.1 0 2-.9 2-2s-.9-2-2-2-2 .9-2 2 .9 2 2 2zm-4 8c-1.1 0-2 .9-2 2s.9 2 2 2 2-.9 2-2-.9-2-2-2zm4-4c-1.1 0-2 .9-2 2s.9 2 2 2 2-.9 2-2-.9-2-2-2zm-4-4c-1.1 0-2 .9-2 2s.9 2 2 2 2-.9 2-2-.9-2-2-2zm-4-4c-1.1 0-2 .9-2 2s.9 2 2 2 2-.9 2-2-.9-2-2-2z"></path></g>
<g id="grid-off"><path d="M8 4v1.45l2 2V4h4v4h-3.45l2 2H14v1.45l2 2V10h4v4h-3.45l2 2H20v1.45l2 2V4c0-1.1-.9-2-2-2H4.55l2 2H8zm8 0h4v4h-4V4zM1.27 1.27L0 2.55l2 2V20c0 1.1.9 2 2 2h15.46l2 2 1.27-1.27L1.27 1.27zM10 12.55L11.45 14H10v-1.45zm-6-6L5.45 8H4V6.55zM8 20H4v-4h4v4zm0-6H4v-4h3.45l.55.55V14zm6 6h-4v-4h3.45l.55.54V20zm2 0v-1.46L17.46 20H16z"></path></g>
<g id="grid-on"><path d="M20 2H4c-1.1 0-2 .9-2 2v16c0 1.1.9 2 2 2h16c1.1 0 2-.9 2-2V4c0-1.1-.9-2-2-2zM8 20H4v-4h4v4zm0-6H4v-4h4v4zm0-6H4V4h4v4zm6 12h-4v-4h4v4zm0-6h-4v-4h4v4zm0-6h-4V4h4v4zm6 12h-4v-4h4v4zm0-6h-4v-4h4v4zm0-6h-4V4h4v4z"></path></g>
<g id="hdr-off"><path d="M17.5 15v-2h1.1l.9 2H21l-.9-2.1c.5-.2.9-.8.9-1.4v-1c0-.8-.7-1.5-1.5-1.5H16v4.9l1.1 1.1h.4zm0-4.5h2v1h-2v-1zm-4.5 0v.4l1.5 1.5v-1.9c0-.8-.7-1.5-1.5-1.5h-1.9l1.5 1.5h.4zm-3.5-1l-7-7-1.1 1L6.9 9h-.4v2h-2V9H3v6h1.5v-2.5h2V15H8v-4.9l1.5 1.5V15h3.4l7.6 7.6 1.1-1.1-12.1-12z"></path></g>
<g id="hdr-on"><path d="M21 11.5v-1c0-.8-.7-1.5-1.5-1.5H16v6h1.5v-2h1.1l.9 2H21l-.9-2.1c.5-.3.9-.8.9-1.4zm-1.5 0h-2v-1h2v1zm-13-.5h-2V9H3v6h1.5v-2.5h2V15H8V9H6.5v2zM13 9H9.5v6H13c.8 0 1.5-.7 1.5-1.5v-3c0-.8-.7-1.5-1.5-1.5zm0 4.5h-2v-3h2v3z"></path></g>
<g id="hdr-strong"><path d="M17 6c-3.31 0-6 2.69-6 6s2.69 6 6 6 6-2.69 6-6-2.69-6-6-6zM5 8c-2.21 0-4 1.79-4 4s1.79 4 4 4 4-1.79 4-4-1.79-4-4-4zm0 6c-1.1 0-2-.9-2-2s.9-2 2-2 2 .9 2 2-.9 2-2 2z"></path></g>
<g id="hdr-weak"><path d="M5 8c-2.21 0-4 1.79-4 4s1.79 4 4 4 4-1.79 4-4-1.79-4-4-4zm12-2c-3.31 0-6 2.69-6 6s2.69 6 6 6 6-2.69 6-6-2.69-6-6-6zm0 10c-2.21 0-4-1.79-4-4s1.79-4 4-4 4 1.79 4 4-1.79 4-4 4z"></path></g>
<g id="healing"><path d="M17.73 12.02l3.98-3.98c.39-.39.39-1.02 0-1.41l-4.34-4.34c-.39-.39-1.02-.39-1.41 0l-3.98 3.98L8 2.29C7.8 2.1 7.55 2 7.29 2c-.25 0-.51.1-.7.29L2.25 6.63c-.39.39-.39 1.02 0 1.41l3.98 3.98L2.25 16c-.39.39-.39 1.02 0 1.41l4.34 4.34c.39.39 1.02.39 1.41 0l3.98-3.98 3.98 3.98c.2.2.45.29.71.29.26 0 .51-.1.71-.29l4.34-4.34c.39-.39.39-1.02 0-1.41l-3.99-3.98zM12 9c.55 0 1 .45 1 1s-.45 1-1 1-1-.45-1-1 .45-1 1-1zm-4.71 1.96L3.66 7.34l3.63-3.63 3.62 3.62-3.62 3.63zM10 13c-.55 0-1-.45-1-1s.45-1 1-1 1 .45 1 1-.45 1-1 1zm2 2c-.55 0-1-.45-1-1s.45-1 1-1 1 .45 1 1-.45 1-1 1zm2-4c.55 0 1 .45 1 1s-.45 1-1 1-1-.45-1-1 .45-1 1-1zm2.66 9.34l-3.63-3.62 3.63-3.63 3.62 3.62-3.62 3.63z"></path></g>
<g id="image"><path d="M21 19V5c0-1.1-.9-2-2-2H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2zM8.5 13.5l2.5 3.01L14.5 12l4.5 6H5l3.5-4.5z"></path></g>
<g id="image-aspect-ratio"><path d="M16 10h-2v2h2v-2zm0 4h-2v2h2v-2zm-8-4H6v2h2v-2zm4 0h-2v2h2v-2zm8-6H4c-1.1 0-2 .9-2 2v12c0 1.1.9 2 2 2h16c1.1 0 2-.9 2-2V6c0-1.1-.9-2-2-2zm0 14H4V6h16v12z"></path></g>
<g id="iso"><path d="M19 3H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zM5.5 7.5h2v-2H9v2h2V9H9v2H7.5V9h-2V7.5zM19 19H5L19 5v14zm-2-2v-1.5h-5V17h5z"></path></g>
<g id="landscape"><path d="M14 6l-3.75 5 2.85 3.8-1.6 1.2C9.81 13.75 7 10 7 10l-6 8h22L14 6z"></path></g>
<g id="leak-add"><path d="M6 3H3v3c1.66 0 3-1.34 3-3zm8 0h-2c0 4.97-4.03 9-9 9v2c6.08 0 11-4.93 11-11zm-4 0H8c0 2.76-2.24 5-5 5v2c3.87 0 7-3.13 7-7zm0 18h2c0-4.97 4.03-9 9-9v-2c-6.07 0-11 4.93-11 11zm8 0h3v-3c-1.66 0-3 1.34-3 3zm-4 0h2c0-2.76 2.24-5 5-5v-2c-3.87 0-7 3.13-7 7z"></path></g>
<g id="leak-remove"><path d="M10 3H8c0 .37-.04.72-.12 1.06l1.59 1.59C9.81 4.84 10 3.94 10 3zM3 4.27l2.84 2.84C5.03 7.67 4.06 8 3 8v2c1.61 0 3.09-.55 4.27-1.46L8.7 9.97C7.14 11.24 5.16 12 3 12v2c2.71 0 5.19-.99 7.11-2.62l2.5 2.5C10.99 15.81 10 18.29 10 21h2c0-2.16.76-4.14 2.03-5.69l1.43 1.43C14.55 17.91 14 19.39 14 21h2c0-1.06.33-2.03.89-2.84L19.73 21 21 19.73 4.27 3 3 4.27zM14 3h-2c0 1.5-.37 2.91-1.02 4.16l1.46 1.46C13.42 6.98 14 5.06 14 3zm5.94 13.12c.34-.08.69-.12 1.06-.12v-2c-.94 0-1.84.19-2.66.52l1.6 1.6zm-4.56-4.56l1.46 1.46C18.09 12.37 19.5 12 21 12v-2c-2.06 0-3.98.58-5.62 1.56z"></path></g>
<g id="lens"><path d="M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2z"></path></g>
<g id="linked-camera"><circle cx="12" cy="14" r="3.2"></circle><path d="M16 3.33c2.58 0 4.67 2.09 4.67 4.67H22c0-3.31-2.69-6-6-6v1.33M16 6c1.11 0 2 .89 2 2h1.33c0-1.84-1.49-3.33-3.33-3.33V6"></path><path d="M17 9c0-1.11-.89-2-2-2V4H9L7.17 6H4c-1.1 0-2 .9-2 2v12c0 1.1.9 2 2 2h16c1.1 0 2-.9 2-2V9h-5zm-5 10c-2.76 0-5-2.24-5-5s2.24-5 5-5 5 2.24 5 5-2.24 5-5 5z"></path></g>
<g id="looks"><path d="M12 10c-3.86 0-7 3.14-7 7h2c0-2.76 2.24-5 5-5s5 2.24 5 5h2c0-3.86-3.14-7-7-7zm0-4C5.93 6 1 10.93 1 17h2c0-4.96 4.04-9 9-9s9 4.04 9 9h2c0-6.07-4.93-11-11-11z"></path></g>
<g id="looks-3"><path d="M19.01 3h-14c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm-4 7.5c0 .83-.67 1.5-1.5 1.5.83 0 1.5.67 1.5 1.5V15c0 1.11-.9 2-2 2h-4v-2h4v-2h-2v-2h2V9h-4V7h4c1.1 0 2 .89 2 2v1.5z"></path></g>
<g id="looks-4"><path d="M19 3H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm-4 14h-2v-4H9V7h2v4h2V7h2v10z"></path></g>
<g id="looks-5"><path d="M19 3H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm-4 6h-4v2h2c1.1 0 2 .89 2 2v2c0 1.11-.9 2-2 2H9v-2h4v-2H9V7h6v2z"></path></g>
<g id="looks-6"><path d="M11 15h2v-2h-2v2zm8-12H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm-4 6h-4v2h2c1.1 0 2 .89 2 2v2c0 1.11-.9 2-2 2h-2c-1.1 0-2-.89-2-2V9c0-1.11.9-2 2-2h4v2z"></path></g>
<g id="looks-one"><path d="M19 3H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm-5 14h-2V9h-2V7h4v10z"></path></g>
<g id="looks-two"><path d="M19 3H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm-4 8c0 1.11-.9 2-2 2h-2v2h4v2H9v-4c0-1.11.9-2 2-2h2V9H9V7h4c1.1 0 2 .89 2 2v2z"></path></g>
<g id="loupe"><path d="M13 7h-2v4H7v2h4v4h2v-4h4v-2h-4V7zm-1-5C6.49 2 2 6.49 2 12s4.49 10 10 10h8c1.1 0 2-.9 2-2v-8c0-5.51-4.49-10-10-10zm0 18c-4.41 0-8-3.59-8-8s3.59-8 8-8 8 3.59 8 8-3.59 8-8 8z"></path></g>
<g id="monochrome-photos"><path d="M20 5h-3.2L15 3H9L7.2 5H4c-1.1 0-2 .9-2 2v12c0 1.1.9 2 2 2h16c1.1 0 2-.9 2-2V7c0-1.1-.9-2-2-2zm0 14h-8v-1c-2.8 0-5-2.2-5-5s2.2-5 5-5V7h8v12zm-3-6c0-2.8-2.2-5-5-5v1.8c1.8 0 3.2 1.4 3.2 3.2s-1.4 3.2-3.2 3.2V18c2.8 0 5-2.2 5-5zm-8.2 0c0 1.8 1.4 3.2 3.2 3.2V9.8c-1.8 0-3.2 1.4-3.2 3.2z"></path></g>
<g id="movie-creation"><path d="M18 4l2 4h-3l-2-4h-2l2 4h-3l-2-4H8l2 4H7L5 4H4c-1.1 0-1.99.9-1.99 2L2 18c0 1.1.9 2 2 2h16c1.1 0 2-.9 2-2V4h-4z"></path></g>
<g id="movie-filter"><path d="M18 4l2 3h-3l-2-3h-2l2 3h-3l-2-3H8l2 3H7L5 4H4c-1.1 0-1.99.9-1.99 2L2 18c0 1.1.9 2 2 2h16c1.1 0 2-.9 2-2V4h-4zm-6.75 11.25L10 18l-1.25-2.75L6 14l2.75-1.25L10 10l1.25 2.75L14 14l-2.75 1.25zm5.69-3.31L16 14l-.94-2.06L13 11l2.06-.94L16 8l.94 2.06L19 11l-2.06.94z"></path></g>
<g id="music-note"><path d="M12 3v10.55c-.59-.34-1.27-.55-2-.55-2.21 0-4 1.79-4 4s1.79 4 4 4 4-1.79 4-4V7h4V3h-6z"></path></g>
<g id="nature"><path d="M13 16.12c3.47-.41 6.17-3.36 6.17-6.95 0-3.87-3.13-7-7-7s-7 3.13-7 7c0 3.47 2.52 6.34 5.83 6.89V20H5v2h14v-2h-6v-3.88z"></path></g>
<g id="nature-people"><path d="M22.17 9.17c0-3.87-3.13-7-7-7s-7 3.13-7 7c0 3.47 2.52 6.34 5.83 6.89V20H6v-3h1v-4c0-.55-.45-1-1-1H3c-.55 0-1 .45-1 1v4h1v5h16v-2h-3v-3.88c3.47-.41 6.17-3.36 6.17-6.95zM4.5 11c.83 0 1.5-.67 1.5-1.5S5.33 8 4.5 8 3 8.67 3 9.5 3.67 11 4.5 11z"></path></g>
<g id="navigate-before"><path d="M15.41 7.41L14 6l-6 6 6 6 1.41-1.41L10.83 12z"></path></g>
<g id="navigate-next"><path d="M10 6L8.59 7.41 13.17 12l-4.58 4.59L10 18l6-6z"></path></g>
<g id="palette"><path d="M12 3c-4.97 0-9 4.03-9 9s4.03 9 9 9c.83 0 1.5-.67 1.5-1.5 0-.39-.15-.74-.39-1.01-.23-.26-.38-.61-.38-.99 0-.83.67-1.5 1.5-1.5H16c2.76 0 5-2.24 5-5 0-4.42-4.03-8-9-8zm-5.5 9c-.83 0-1.5-.67-1.5-1.5S5.67 9 6.5 9 8 9.67 8 10.5 7.33 12 6.5 12zm3-4C8.67 8 8 7.33 8 6.5S8.67 5 9.5 5s1.5.67 1.5 1.5S10.33 8 9.5 8zm5 0c-.83 0-1.5-.67-1.5-1.5S13.67 5 14.5 5s1.5.67 1.5 1.5S15.33 8 14.5 8zm3 4c-.83 0-1.5-.67-1.5-1.5S16.67 9 17.5 9s1.5.67 1.5 1.5-.67 1.5-1.5 1.5z"></path></g>
<g id="panorama"><path d="M23 18V6c0-1.1-.9-2-2-2H3c-1.1 0-2 .9-2 2v12c0 1.1.9 2 2 2h18c1.1 0 2-.9 2-2zM8.5 12.5l2.5 3.01L14.5 11l4.5 6H5l3.5-4.5z"></path></g>
<g id="panorama-fish-eye"><path d="M12 2C6.47 2 2 6.47 2 12s4.47 10 10 10 10-4.47 10-10S17.53 2 12 2zm0 18c-4.41 0-8-3.59-8-8s3.59-8 8-8 8 3.59 8 8-3.59 8-8 8z"></path></g>
<g id="panorama-horizontal"><path d="M20 6.54v10.91c-2.6-.77-5.28-1.16-8-1.16-2.72 0-5.4.39-8 1.16V6.54c2.6.77 5.28 1.16 8 1.16 2.72.01 5.4-.38 8-1.16M21.43 4c-.1 0-.2.02-.31.06C18.18 5.16 15.09 5.7 12 5.7c-3.09 0-6.18-.55-9.12-1.64-.11-.04-.22-.06-.31-.06-.34 0-.57.23-.57.63v14.75c0 .39.23.62.57.62.1 0 .2-.02.31-.06 2.94-1.1 6.03-1.64 9.12-1.64 3.09 0 6.18.55 9.12 1.64.11.04.21.06.31.06.33 0 .57-.23.57-.63V4.63c0-.4-.24-.63-.57-.63z"></path></g>
<g id="panorama-vertical"><path d="M19.94 21.12c-1.1-2.94-1.64-6.03-1.64-9.12 0-3.09.55-6.18 1.64-9.12.04-.11.06-.22.06-.31 0-.34-.23-.57-.63-.57H4.63c-.4 0-.63.23-.63.57 0 .1.02.2.06.31C5.16 5.82 5.71 8.91 5.71 12c0 3.09-.55 6.18-1.64 9.12-.05.11-.07.22-.07.31 0 .33.23.57.63.57h14.75c.39 0 .63-.24.63-.57-.01-.1-.03-.2-.07-.31zM6.54 20c.77-2.6 1.16-5.28 1.16-8 0-2.72-.39-5.4-1.16-8h10.91c-.77 2.6-1.16 5.28-1.16 8 0 2.72.39 5.4 1.16 8H6.54z"></path></g>
<g id="panorama-wide-angle"><path d="M12 6c2.45 0 4.71.2 7.29.64.47 1.78.71 3.58.71 5.36 0 1.78-.24 3.58-.71 5.36-2.58.44-4.84.64-7.29.64s-4.71-.2-7.29-.64C4.24 15.58 4 13.78 4 12c0-1.78.24-3.58.71-5.36C7.29 6.2 9.55 6 12 6m0-2c-2.73 0-5.22.24-7.95.72l-.93.16-.25.9C2.29 7.85 2 9.93 2 12s.29 4.15.87 6.22l.25.89.93.16c2.73.49 5.22.73 7.95.73s5.22-.24 7.95-.72l.93-.16.25-.89c.58-2.08.87-4.16.87-6.23s-.29-4.15-.87-6.22l-.25-.89-.93-.16C17.22 4.24 14.73 4 12 4z"></path></g>
<g id="photo"><path d="M21 19V5c0-1.1-.9-2-2-2H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2zM8.5 13.5l2.5 3.01L14.5 12l4.5 6H5l3.5-4.5z"></path></g>
<g id="photo-album"><path d="M18 2H6c-1.1 0-2 .9-2 2v16c0 1.1.9 2 2 2h12c1.1 0 2-.9 2-2V4c0-1.1-.9-2-2-2zM6 4h5v8l-2.5-1.5L6 12V4zm0 15l3-3.86 2.14 2.58 3-3.86L18 19H6z"></path></g>
<g id="photo-camera"><circle cx="12" cy="12" r="3.2"></circle><path d="M9 2L7.17 4H4c-1.1 0-2 .9-2 2v12c0 1.1.9 2 2 2h16c1.1 0 2-.9 2-2V6c0-1.1-.9-2-2-2h-3.17L15 2H9zm3 15c-2.76 0-5-2.24-5-5s2.24-5 5-5 5 2.24 5 5-2.24 5-5 5z"></path></g>
<g id="photo-filter"><path d="M19.02 10v9H5V5h9V3H5.02c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2v-9h-2zM17 10l.94-2.06L20 7l-2.06-.94L17 4l-.94 2.06L14 7l2.06.94zm-3.75.75L12 8l-1.25 2.75L8 12l2.75 1.25L12 16l1.25-2.75L16 12z"></path></g>
<g id="photo-library"><path d="M22 16V4c0-1.1-.9-2-2-2H8c-1.1 0-2 .9-2 2v12c0 1.1.9 2 2 2h12c1.1 0 2-.9 2-2zm-11-4l2.03 2.71L16 11l4 5H8l3-4zM2 6v14c0 1.1.9 2 2 2h14v-2H4V6H2z"></path></g>
<g id="photo-size-select-actual"><path d="M21 3H3C2 3 1 4 1 5v14c0 1.1.9 2 2 2h18c1 0 2-1 2-2V5c0-1-1-2-2-2zM5 17l3.5-4.5 2.5 3.01L14.5 11l4.5 6H5z"></path></g>
<g id="photo-size-select-large"><path d="M21 15h2v2h-2v-2zm0-4h2v2h-2v-2zm2 8h-2v2c1 0 2-1 2-2zM13 3h2v2h-2V3zm8 4h2v2h-2V7zm0-4v2h2c0-1-1-2-2-2zM1 7h2v2H1V7zm16-4h2v2h-2V3zm0 16h2v2h-2v-2zM3 3C2 3 1 4 1 5h2V3zm6 0h2v2H9V3zM5 3h2v2H5V3zm-4 8v8c0 1.1.9 2 2 2h12V11H1zm2 8l2.5-3.21 1.79 2.15 2.5-3.22L13 19H3z"></path></g>
<g id="photo-size-select-small"><path d="M23 15h-2v2h2v-2zm0-4h-2v2h2v-2zm0 8h-2v2c1 0 2-1 2-2zM15 3h-2v2h2V3zm8 4h-2v2h2V7zm-2-4v2h2c0-1-1-2-2-2zM3 21h8v-6H1v4c0 1.1.9 2 2 2zM3 7H1v2h2V7zm12 12h-2v2h2v-2zm4-16h-2v2h2V3zm0 16h-2v2h2v-2zM3 3C2 3 1 4 1 5h2V3zm0 8H1v2h2v-2zm8-8H9v2h2V3zM7 3H5v2h2V3z"></path></g>
<g id="picture-as-pdf"><path d="M20 2H8c-1.1 0-2 .9-2 2v12c0 1.1.9 2 2 2h12c1.1 0 2-.9 2-2V4c0-1.1-.9-2-2-2zm-8.5 7.5c0 .83-.67 1.5-1.5 1.5H9v2H7.5V7H10c.83 0 1.5.67 1.5 1.5v1zm5 2c0 .83-.67 1.5-1.5 1.5h-2.5V7H15c.83 0 1.5.67 1.5 1.5v3zm4-3H19v1h1.5V11H19v2h-1.5V7h3v1.5zM9 9.5h1v-1H9v1zM4 6H2v14c0 1.1.9 2 2 2h14v-2H4V6zm10 5.5h1v-3h-1v3z"></path></g>
<g id="portrait"><path d="M12 12.25c1.24 0 2.25-1.01 2.25-2.25S13.24 7.75 12 7.75 9.75 8.76 9.75 10s1.01 2.25 2.25 2.25zm4.5 4c0-1.5-3-2.25-4.5-2.25s-4.5.75-4.5 2.25V17h9v-.75zM19 3H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm0 16H5V5h14v14z"></path></g>
<g id="remove-red-eye"><path d="M12 4.5C7 4.5 2.73 7.61 1 12c1.73 4.39 6 7.5 11 7.5s9.27-3.11 11-7.5c-1.73-4.39-6-7.5-11-7.5zM12 17c-2.76 0-5-2.24-5-5s2.24-5 5-5 5 2.24 5 5-2.24 5-5 5zm0-8c-1.66 0-3 1.34-3 3s1.34 3 3 3 3-1.34 3-3-1.34-3-3-3z"></path></g>
<g id="rotate-90-degrees-ccw"><path d="M7.34 6.41L.86 12.9l6.49 6.48 6.49-6.48-6.5-6.49zM3.69 12.9l3.66-3.66L11 12.9l-3.66 3.66-3.65-3.66zm15.67-6.26C17.61 4.88 15.3 4 13 4V.76L8.76 5 13 9.24V6c1.79 0 3.58.68 4.95 2.05 2.73 2.73 2.73 7.17 0 9.9C16.58 19.32 14.79 20 13 20c-.97 0-1.94-.21-2.84-.61l-1.49 1.49C10.02 21.62 11.51 22 13 22c2.3 0 4.61-.88 6.36-2.64 3.52-3.51 3.52-9.21 0-12.72z"></path></g>
<g id="rotate-left"><path d="M7.11 8.53L5.7 7.11C4.8 8.27 4.24 9.61 4.07 11h2.02c.14-.87.49-1.72 1.02-2.47zM6.09 13H4.07c.17 1.39.72 2.73 1.62 3.89l1.41-1.42c-.52-.75-.87-1.59-1.01-2.47zm1.01 5.32c1.16.9 2.51 1.44 3.9 1.61V17.9c-.87-.15-1.71-.49-2.46-1.03L7.1 18.32zM13 4.07V1L8.45 5.55 13 10V6.09c2.84.48 5 2.94 5 5.91s-2.16 5.43-5 5.91v2.02c3.95-.49 7-3.85 7-7.93s-3.05-7.44-7-7.93z"></path></g>
<g id="rotate-right"><path d="M15.55 5.55L11 1v3.07C7.06 4.56 4 7.92 4 12s3.05 7.44 7 7.93v-2.02c-2.84-.48-5-2.94-5-5.91s2.16-5.43 5-5.91V10l4.55-4.45zM19.93 11c-.17-1.39-.72-2.73-1.62-3.89l-1.42 1.42c.54.75.88 1.6 1.02 2.47h2.02zM13 17.9v2.02c1.39-.17 2.74-.71 3.9-1.61l-1.44-1.44c-.75.54-1.59.89-2.46 1.03zm3.89-2.42l1.42 1.41c.9-1.16 1.45-2.5 1.62-3.89h-2.02c-.14.87-.48 1.72-1.02 2.48z"></path></g>
<g id="slideshow"><path d="M10 8v8l5-4-5-4zm9-5H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm0 16H5V5h14v14z"></path></g>
<g id="straighten"><path d="M21 6H3c-1.1 0-2 .9-2 2v8c0 1.1.9 2 2 2h18c1.1 0 2-.9 2-2V8c0-1.1-.9-2-2-2zm0 10H3V8h2v4h2V8h2v4h2V8h2v4h2V8h2v4h2V8h2v8z"></path></g>
<g id="style"><path d="M2.53 19.65l1.34.56v-9.03l-2.43 5.86c-.41 1.02.08 2.19 1.09 2.61zm19.5-3.7L17.07 3.98c-.31-.75-1.04-1.21-1.81-1.23-.26 0-.53.04-.79.15L7.1 5.95c-.75.31-1.21 1.03-1.23 1.8-.01.27.04.54.15.8l4.96 11.97c.31.76 1.05 1.22 1.83 1.23.26 0 .52-.05.77-.15l7.36-3.05c1.02-.42 1.51-1.59 1.09-2.6zM7.88 8.75c-.55 0-1-.45-1-1s.45-1 1-1 1 .45 1 1-.45 1-1 1zm-2 11c0 1.1.9 2 2 2h1.45l-3.45-8.34v6.34z"></path></g>
<g id="switch-camera"><path d="M20 4h-3.17L15 2H9L7.17 4H4c-1.1 0-2 .9-2 2v12c0 1.1.9 2 2 2h16c1.1 0 2-.9 2-2V6c0-1.1-.9-2-2-2zm-5 11.5V13H9v2.5L5.5 12 9 8.5V11h6V8.5l3.5 3.5-3.5 3.5z"></path></g>
<g id="switch-video"><path d="M18 9.5V6c0-.55-.45-1-1-1H3c-.55 0-1 .45-1 1v12c0 .55.45 1 1 1h14c.55 0 1-.45 1-1v-3.5l4 4v-13l-4 4zm-5 6V13H7v2.5L3.5 12 7 8.5V11h6V8.5l3.5 3.5-3.5 3.5z"></path></g>
<g id="tag-faces"><path d="M11.99 2C6.47 2 2 6.48 2 12s4.47 10 9.99 10C17.52 22 22 17.52 22 12S17.52 2 11.99 2zM12 20c-4.42 0-8-3.58-8-8s3.58-8 8-8 8 3.58 8 8-3.58 8-8 8zm3.5-9c.83 0 1.5-.67 1.5-1.5S16.33 8 15.5 8 14 8.67 14 9.5s.67 1.5 1.5 1.5zm-7 0c.83 0 1.5-.67 1.5-1.5S9.33 8 8.5 8 7 8.67 7 9.5 7.67 11 8.5 11zm3.5 6.5c2.33 0 4.31-1.46 5.11-3.5H6.89c.8 2.04 2.78 3.5 5.11 3.5z"></path></g>
<g id="texture"><path d="M19.51 3.08L3.08 19.51c.09.34.27.65.51.9.25.24.56.42.9.51L20.93 4.49c-.19-.69-.73-1.23-1.42-1.41zM11.88 3L3 11.88v2.83L14.71 3h-2.83zM5 3c-1.1 0-2 .9-2 2v2l4-4H5zm14 18c.55 0 1.05-.22 1.41-.59.37-.36.59-.86.59-1.41v-2l-4 4h2zm-9.71 0h2.83L21 12.12V9.29L9.29 21z"></path></g>
<g id="timelapse"><path d="M16.24 7.76C15.07 6.59 13.54 6 12 6v6l-4.24 4.24c2.34 2.34 6.14 2.34 8.49 0 2.34-2.34 2.34-6.14-.01-8.48zM12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm0 18c-4.42 0-8-3.58-8-8s3.58-8 8-8 8 3.58 8 8-3.58 8-8 8z"></path></g>
<g id="timer"><path d="M15 1H9v2h6V1zm-4 13h2V8h-2v6zm8.03-6.61l1.42-1.42c-.43-.51-.9-.99-1.41-1.41l-1.42 1.42C16.07 4.74 14.12 4 12 4c-4.97 0-9 4.03-9 9s4.02 9 9 9 9-4.03 9-9c0-2.12-.74-4.07-1.97-5.61zM12 20c-3.87 0-7-3.13-7-7s3.13-7 7-7 7 3.13 7 7-3.13 7-7 7z"></path></g>
<g id="timer-10"><path d="M0 7.72V9.4l3-1V18h2V6h-.25L0 7.72zm23.78 6.65c-.14-.28-.35-.53-.63-.74-.28-.21-.61-.39-1.01-.53s-.85-.27-1.35-.38c-.35-.07-.64-.15-.87-.23-.23-.08-.41-.16-.55-.25-.14-.09-.23-.19-.28-.3-.05-.11-.08-.24-.08-.39 0-.14.03-.28.09-.41.06-.13.15-.25.27-.34.12-.1.27-.18.45-.24s.4-.09.64-.09c.25 0 .47.04.66.11.19.07.35.17.48.29.13.12.22.26.29.42.06.16.1.32.1.49h1.95c0-.39-.08-.75-.24-1.09-.16-.34-.39-.63-.69-.88-.3-.25-.66-.44-1.09-.59C21.49 9.07 21 9 20.46 9c-.51 0-.98.07-1.39.21-.41.14-.77.33-1.06.57-.29.24-.51.52-.67.84-.16.32-.23.65-.23 1.01s.08.69.23.96c.15.28.36.52.64.73.27.21.6.38.98.53.38.14.81.26 1.27.36.39.08.71.17.95.26s.43.19.57.29c.13.1.22.22.27.34.05.12.07.25.07.39 0 .32-.13.57-.4.77-.27.2-.66.29-1.17.29-.22 0-.43-.02-.64-.08-.21-.05-.4-.13-.56-.24-.17-.11-.3-.26-.41-.44-.11-.18-.17-.41-.18-.67h-1.89c0 .36.08.71.24 1.05.16.34.39.65.7.93.31.27.69.49 1.15.66.46.17.98.25 1.58.25.53 0 1.01-.06 1.44-.19.43-.13.8-.31 1.11-.54.31-.23.54-.51.71-.83.17-.32.25-.67.25-1.06-.02-.4-.09-.74-.24-1.02zm-9.96-7.32c-.34-.4-.75-.7-1.23-.88-.47-.18-1.01-.27-1.59-.27-.58 0-1.11.09-1.59.27-.48.18-.89.47-1.23.88-.34.41-.6.93-.79 1.59-.18.65-.28 1.45-.28 2.39v1.92c0 .94.09 1.74.28 2.39.19.66.45 1.19.8 1.6.34.41.75.71 1.23.89.48.18 1.01.28 1.59.28.59 0 1.12-.09 1.59-.28.48-.18.88-.48 1.22-.89.34-.41.6-.94.78-1.6.18-.65.28-1.45.28-2.39v-1.92c0-.94-.09-1.74-.28-2.39-.18-.66-.44-1.19-.78-1.59zm-.92 6.17c0 .6-.04 1.11-.12 1.53-.08.42-.2.76-.36 1.02-.16.26-.36.45-.59.57-.23.12-.51.18-.82.18-.3 0-.58-.06-.82-.18s-.44-.31-.6-.57c-.16-.26-.29-.6-.38-1.02-.09-.42-.13-.93-.13-1.53v-2.5c0-.6.04-1.11.13-1.52.09-.41.21-.74.38-1 .16-.25.36-.43.6-.55.24-.11.51-.17.81-.17.31 0 .58.06.81.17.24.11.44.29.6.55.16.25.29.58.37.99.08.41.13.92.13 1.52v2.51z"></path></g>
<g id="timer-3"><path d="M11.61 12.97c-.16-.24-.36-.46-.62-.65-.25-.19-.56-.35-.93-.48.3-.14.57-.3.8-.5.23-.2.42-.41.57-.64.15-.23.27-.46.34-.71.08-.24.11-.49.11-.73 0-.55-.09-1.04-.28-1.46-.18-.42-.44-.77-.78-1.06-.33-.28-.73-.5-1.2-.64-.45-.13-.97-.2-1.53-.2-.55 0-1.06.08-1.52.24-.47.17-.87.4-1.2.69-.33.29-.6.63-.78 1.03-.2.39-.29.83-.29 1.29h1.98c0-.26.05-.49.14-.69.09-.2.22-.38.38-.52.17-.14.36-.25.58-.33.22-.08.46-.12.73-.12.61 0 1.06.16 1.36.47.3.31.44.75.44 1.32 0 .27-.04.52-.12.74-.08.22-.21.41-.38.57-.17.16-.38.28-.63.37-.25.09-.55.13-.89.13H6.72v1.57H7.9c.34 0 .64.04.91.11.27.08.5.19.69.35.19.16.34.36.44.61.1.24.16.54.16.87 0 .62-.18 1.09-.53 1.42-.35.33-.84.49-1.45.49-.29 0-.56-.04-.8-.13-.24-.08-.44-.2-.61-.36-.17-.16-.3-.34-.39-.56-.09-.22-.14-.46-.14-.72H4.19c0 .55.11 1.03.32 1.45.21.42.5.77.86 1.05s.77.49 1.24.63.96.21 1.48.21c.57 0 1.09-.08 1.58-.23.49-.15.91-.38 1.26-.68.36-.3.64-.66.84-1.1.2-.43.3-.93.3-1.48 0-.29-.04-.58-.11-.86-.08-.25-.19-.51-.35-.76zm9.26 1.4c-.14-.28-.35-.53-.63-.74-.28-.21-.61-.39-1.01-.53s-.85-.27-1.35-.38c-.35-.07-.64-.15-.87-.23-.23-.08-.41-.16-.55-.25-.14-.09-.23-.19-.28-.3-.05-.11-.08-.24-.08-.39s.03-.28.09-.41c.06-.13.15-.25.27-.34.12-.1.27-.18.45-.24s.4-.09.64-.09c.25 0 .47.04.66.11.19.07.35.17.48.29.13.12.22.26.29.42.06.16.1.32.1.49h1.95c0-.39-.08-.75-.24-1.09-.16-.34-.39-.63-.69-.88-.3-.25-.66-.44-1.09-.59-.43-.15-.92-.22-1.46-.22-.51 0-.98.07-1.39.21-.41.14-.77.33-1.06.57-.29.24-.51.52-.67.84-.16.32-.23.65-.23 1.01s.08.68.23.96c.15.28.37.52.64.73.27.21.6.38.98.53.38.14.81.26 1.27.36.39.08.71.17.95.26s.43.19.57.29c.13.1.22.22.27.34.05.12.07.25.07.39 0 .32-.13.57-.4.77-.27.2-.66.29-1.17.29-.22 0-.43-.02-.64-.08-.21-.05-.4-.13-.56-.24-.17-.11-.3-.26-.41-.44-.11-.18-.17-.41-.18-.67h-1.89c0 .36.08.71.24 1.05.16.34.39.65.7.93.31.27.69.49 1.15.66.46.17.98.25 1.58.25.53 0 1.01-.06 1.44-.19.43-.13.8-.31 1.11-.54.31-.23.54-.51.71-.83.17-.32.25-.67.25-1.06-.02-.4-.09-.74-.24-1.02z"></path></g>
<g id="timer-off"><path d="M19.04 4.55l-1.42 1.42C16.07 4.74 14.12 4 12 4c-1.83 0-3.53.55-4.95 1.48l1.46 1.46C9.53 6.35 10.73 6 12 6c3.87 0 7 3.13 7 7 0 1.27-.35 2.47-.94 3.49l1.45 1.45C20.45 16.53 21 14.83 21 13c0-2.12-.74-4.07-1.97-5.61l1.42-1.42-1.41-1.42zM15 1H9v2h6V1zm-4 8.44l2 2V8h-2v1.44zM3.02 4L1.75 5.27 4.5 8.03C3.55 9.45 3 11.16 3 13c0 4.97 4.02 9 9 9 1.84 0 3.55-.55 4.98-1.5l2.5 2.5 1.27-1.27-7.71-7.71L3.02 4zM12 20c-3.87 0-7-3.13-7-7 0-1.28.35-2.48.95-3.52l9.56 9.56c-1.03.61-2.23.96-3.51.96z"></path></g>
<g id="tonality"><path d="M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm-1 17.93c-3.94-.49-7-3.85-7-7.93s3.05-7.44 7-7.93v15.86zm2-15.86c1.03.13 2 .45 2.87.93H13v-.93zM13 7h5.24c.25.31.48.65.68 1H13V7zm0 3h6.74c.08.33.15.66.19 1H13v-1zm0 9.93V19h2.87c-.87.48-1.84.8-2.87.93zM18.24 17H13v-1h5.92c-.2.35-.43.69-.68 1zm1.5-3H13v-1h6.93c-.04.34-.11.67-.19 1z"></path></g>
<g id="transform"><path d="M22 18v-2H8V4h2L7 1 4 4h2v2H2v2h4v8c0 1.1.9 2 2 2h8v2h-2l3 3 3-3h-2v-2h4zM10 8h6v6h2V8c0-1.1-.9-2-2-2h-6v2z"></path></g>
<g id="tune"><path d="M3 17v2h6v-2H3zM3 5v2h10V5H3zm10 16v-2h8v-2h-8v-2h-2v6h2zM7 9v2H3v2h4v2h2V9H7zm14 4v-2H11v2h10zm-6-4h2V7h4V5h-4V3h-2v6z"></path></g>
<g id="view-comfy"><path d="M3 9h4V5H3v4zm0 5h4v-4H3v4zm5 0h4v-4H8v4zm5 0h4v-4h-4v4zM8 9h4V5H8v4zm5-4v4h4V5h-4zm5 9h4v-4h-4v4zM3 19h4v-4H3v4zm5 0h4v-4H8v4zm5 0h4v-4h-4v4zm5 0h4v-4h-4v4zm0-14v4h4V5h-4z"></path></g>
<g id="view-compact"><path d="M3 19h6v-7H3v7zm7 0h12v-7H10v7zM3 5v6h19V5H3z"></path></g>
<g id="vignette"><path d="M21 3H3c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h18c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm-9 15c-4.42 0-8-2.69-8-6s3.58-6 8-6 8 2.69 8 6-3.58 6-8 6z"></path></g>
<g id="wb-auto"><path d="M6.85 12.65h2.3L8 9l-1.15 3.65zM22 7l-1.2 6.29L19.3 7h-1.6l-1.49 6.29L15 7h-.76C12.77 5.17 10.53 4 8 4c-4.42 0-8 3.58-8 8s3.58 8 8 8c3.13 0 5.84-1.81 7.15-4.43l.1.43H17l1.5-6.1L20 16h1.75l2.05-9H22zm-11.7 9l-.7-2H6.4l-.7 2H3.8L7 7h2l3.2 9h-1.9z"></path></g>
<g id="wb-cloudy"><path d="M19.36 10.04C18.67 6.59 15.64 4 12 4 9.11 4 6.6 5.64 5.35 8.04 2.34 8.36 0 10.91 0 14c0 3.31 2.69 6 6 6h13c2.76 0 5-2.24 5-5 0-2.64-2.05-4.78-4.64-4.96z"></path></g>
<g id="wb-incandescent"><path d="M3.55 18.54l1.41 1.41 1.79-1.8-1.41-1.41-1.79 1.8zM11 22.45h2V19.5h-2v2.95zM4 10.5H1v2h3v-2zm11-4.19V1.5H9v4.81C7.21 7.35 6 9.28 6 11.5c0 3.31 2.69 6 6 6s6-2.69 6-6c0-2.22-1.21-4.15-3-5.19zm5 4.19v2h3v-2h-3zm-2.76 7.66l1.79 1.8 1.41-1.41-1.8-1.79-1.4 1.4z"></path></g>
<g id="wb-iridescent"><path d="M5 14.5h14v-6H5v6zM11 .55V3.5h2V.55h-2zm8.04 2.5l-1.79 1.79 1.41 1.41 1.8-1.79-1.42-1.41zM13 22.45V19.5h-2v2.95h2zm7.45-3.91l-1.8-1.79-1.41 1.41 1.79 1.8 1.42-1.42zM3.55 4.46l1.79 1.79 1.41-1.41-1.79-1.79-1.41 1.41zm1.41 15.49l1.79-1.8-1.41-1.41-1.79 1.79 1.41 1.42z"></path></g>
<g id="wb-sunny"><path d="M6.76 4.84l-1.8-1.79-1.41 1.41 1.79 1.79 1.42-1.41zM4 10.5H1v2h3v-2zm9-9.95h-2V3.5h2V.55zm7.45 3.91l-1.41-1.41-1.79 1.79 1.41 1.41 1.79-1.79zm-3.21 13.7l1.79 1.8 1.41-1.41-1.8-1.79-1.4 1.4zM20 10.5v2h3v-2h-3zm-8-5c-3.31 0-6 2.69-6 6s2.69 6 6 6 6-2.69 6-6-2.69-6-6-6zm-1 16.95h2V19.5h-2v2.95zm-7.45-3.91l1.41 1.41 1.79-1.8-1.41-1.41-1.79 1.8z"></path></g>
</defs></svg>
</iron-iconset-svg>`;

    document.head.appendChild(template$2.content);

    /**
    @license
    Copyright (c) 2014 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**

    `iron-icons` is a utility import that includes the definition for the
    `iron-icon` element, `iron-iconset-svg` element, as well as an import for the
    default icon set.

    The `iron-icons` directory also includes imports for additional icon sets that
    can be loaded into your project.

    Example loading icon set:

        <script type="module">
          import '@polymer/iron-icons/maps-icons.js';
        </JAVA_SCRIIIIPT/>

    To use an icon from one of these sets, first prefix your `iron-icon` with the
    icon set name, followed by a colon, ":", and then the icon id.

    Example using the directions-bus icon from the maps icon set:

        <iron-icon icon="maps:directions-bus"></iron-icon>

    See [iron-icon](https://www.webcomponents.org/element/@polymer/iron-icon) for
    more information about working with icons.

    See [iron-iconset](https://www.webcomponents.org/element/@polymer/iron-iconset)
    and
    [iron-iconset-svg](https://www.webcomponents.org/element/@polymer/iron-iconset-svg)
    for more information about how to create a custom iconset.

    @group Iron Elements
    @pseudoElement iron-icons
    @demo demo/index.html
    */

    const template$3 = html`<iron-iconset-svg name="icons" size="24">
<svg><defs>
<g id="3d-rotation"><path d="M7.52 21.48C4.25 19.94 1.91 16.76 1.55 13H.05C.56 19.16 5.71 24 12 24l.66-.03-3.81-3.81-1.33 1.32zm.89-6.52c-.19 0-.37-.03-.52-.08-.16-.06-.29-.13-.4-.24-.11-.1-.2-.22-.26-.37-.06-.14-.09-.3-.09-.47h-1.3c0 .36.07.68.21.95.14.27.33.5.56.69.24.18.51.32.82.41.3.1.62.15.96.15.37 0 .72-.05 1.03-.15.32-.1.6-.25.83-.44s.42-.43.55-.72c.13-.29.2-.61.2-.97 0-.19-.02-.38-.07-.56-.05-.18-.12-.35-.23-.51-.1-.16-.24-.3-.4-.43-.17-.13-.37-.23-.61-.31.2-.09.37-.2.52-.33.15-.13.27-.27.37-.42.1-.15.17-.3.22-.46.05-.16.07-.32.07-.48 0-.36-.06-.68-.18-.96-.12-.28-.29-.51-.51-.69-.2-.19-.47-.33-.77-.43C9.1 8.05 8.76 8 8.39 8c-.36 0-.69.05-1 .16-.3.11-.57.26-.79.45-.21.19-.38.41-.51.67-.12.26-.18.54-.18.85h1.3c0-.17.03-.32.09-.45s.14-.25.25-.34c.11-.09.23-.17.38-.22.15-.05.3-.08.48-.08.4 0 .7.1.89.31.19.2.29.49.29.86 0 .18-.03.34-.08.49-.05.15-.14.27-.25.37-.11.1-.25.18-.41.24-.16.06-.36.09-.58.09H7.5v1.03h.77c.22 0 .42.02.6.07s.33.13.45.23c.12.11.22.24.29.4.07.16.1.35.1.57 0 .41-.12.72-.35.93-.23.23-.55.33-.95.33zm8.55-5.92c-.32-.33-.7-.59-1.14-.77-.43-.18-.92-.27-1.46-.27H12v8h2.3c.55 0 1.06-.09 1.51-.27.45-.18.84-.43 1.16-.76.32-.33.57-.73.74-1.19.17-.47.26-.99.26-1.57v-.4c0-.58-.09-1.1-.26-1.57-.18-.47-.43-.87-.75-1.2zm-.39 3.16c0 .42-.05.79-.14 1.13-.1.33-.24.62-.43.85-.19.23-.43.41-.71.53-.29.12-.62.18-.99.18h-.91V9.12h.97c.72 0 1.27.23 1.64.69.38.46.57 1.12.57 1.99v.4zM12 0l-.66.03 3.81 3.81 1.33-1.33c3.27 1.55 5.61 4.72 5.96 8.48h1.5C23.44 4.84 18.29 0 12 0z"></path></g>
<g id="accessibility"><path d="M12 2c1.1 0 2 .9 2 2s-.9 2-2 2-2-.9-2-2 .9-2 2-2zm9 7h-6v13h-2v-6h-2v6H9V9H3V7h18v2z"></path></g>
<g id="accessible"><circle cx="12" cy="4" r="2"></circle><path d="M19 13v-2c-1.54.02-3.09-.75-4.07-1.83l-1.29-1.43c-.17-.19-.38-.34-.61-.45-.01 0-.01-.01-.02-.01H13c-.35-.2-.75-.3-1.19-.26C10.76 7.11 10 8.04 10 9.09V15c0 1.1.9 2 2 2h5v5h2v-5.5c0-1.1-.9-2-2-2h-3v-3.45c1.29 1.07 3.25 1.94 5 1.95zm-6.17 5c-.41 1.16-1.52 2-2.83 2-1.66 0-3-1.34-3-3 0-1.31.84-2.41 2-2.83V12.1c-2.28.46-4 2.48-4 4.9 0 2.76 2.24 5 5 5 2.42 0 4.44-1.72 4.9-4h-2.07z"></path></g>
<g id="account-balance"><path d="M4 10v7h3v-7H4zm6 0v7h3v-7h-3zM2 22h19v-3H2v3zm14-12v7h3v-7h-3zm-4.5-9L2 6v2h19V6l-9.5-5z"></path></g>
<g id="account-balance-wallet"><path d="M21 18v1c0 1.1-.9 2-2 2H5c-1.11 0-2-.9-2-2V5c0-1.1.89-2 2-2h14c1.1 0 2 .9 2 2v1h-9c-1.11 0-2 .9-2 2v8c0 1.1.89 2 2 2h9zm-9-2h10V8H12v8zm4-2.5c-.83 0-1.5-.67-1.5-1.5s.67-1.5 1.5-1.5 1.5.67 1.5 1.5-.67 1.5-1.5 1.5z"></path></g>
<g id="account-box"><path d="M3 5v14c0 1.1.89 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2H5c-1.11 0-2 .9-2 2zm12 4c0 1.66-1.34 3-3 3s-3-1.34-3-3 1.34-3 3-3 3 1.34 3 3zm-9 8c0-2 4-3.1 6-3.1s6 1.1 6 3.1v1H6v-1z"></path></g>
<g id="account-circle"><path d="M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm0 3c1.66 0 3 1.34 3 3s-1.34 3-3 3-3-1.34-3-3 1.34-3 3-3zm0 14.2c-2.5 0-4.71-1.28-6-3.22.03-1.99 4-3.08 6-3.08 1.99 0 5.97 1.09 6 3.08-1.29 1.94-3.5 3.22-6 3.22z"></path></g>
<g id="add"><path d="M19 13h-6v6h-2v-6H5v-2h6V5h2v6h6v2z"></path></g>
<g id="add-alert"><path d="M10.01 21.01c0 1.1.89 1.99 1.99 1.99s1.99-.89 1.99-1.99h-3.98zm8.87-4.19V11c0-3.25-2.25-5.97-5.29-6.69v-.72C13.59 2.71 12.88 2 12 2s-1.59.71-1.59 1.59v.72C7.37 5.03 5.12 7.75 5.12 11v5.82L3 18.94V20h18v-1.06l-2.12-2.12zM16 13.01h-3v3h-2v-3H8V11h3V8h2v3h3v2.01z"></path></g>
<g id="add-box"><path d="M19 3H5c-1.11 0-2 .9-2 2v14c0 1.1.89 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm-2 10h-4v4h-2v-4H7v-2h4V7h2v4h4v2z"></path></g>
<g id="add-circle"><path d="M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm5 11h-4v4h-2v-4H7v-2h4V7h2v4h4v2z"></path></g>
<g id="add-circle-outline"><path d="M13 7h-2v4H7v2h4v4h2v-4h4v-2h-4V7zm-1-5C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm0 18c-4.41 0-8-3.59-8-8s3.59-8 8-8 8 3.59 8 8-3.59 8-8 8z"></path></g>
<g id="add-shopping-cart"><path d="M11 9h2V6h3V4h-3V1h-2v3H8v2h3v3zm-4 9c-1.1 0-1.99.9-1.99 2S5.9 22 7 22s2-.9 2-2-.9-2-2-2zm10 0c-1.1 0-1.99.9-1.99 2s.89 2 1.99 2 2-.9 2-2-.9-2-2-2zm-9.83-3.25l.03-.12.9-1.63h7.45c.75 0 1.41-.41 1.75-1.03l3.86-7.01L19.42 4h-.01l-1.1 2-2.76 5H8.53l-.13-.27L6.16 6l-.95-2-.94-2H1v2h2l3.6 7.59-1.35 2.45c-.16.28-.25.61-.25.96 0 1.1.9 2 2 2h12v-2H7.42c-.13 0-.25-.11-.25-.25z"></path></g>
<g id="alarm"><path d="M22 5.72l-4.6-3.86-1.29 1.53 4.6 3.86L22 5.72zM7.88 3.39L6.6 1.86 2 5.71l1.29 1.53 4.59-3.85zM12.5 8H11v6l4.75 2.85.75-1.23-4-2.37V8zM12 4c-4.97 0-9 4.03-9 9s4.02 9 9 9c4.97 0 9-4.03 9-9s-4.03-9-9-9zm0 16c-3.87 0-7-3.13-7-7s3.13-7 7-7 7 3.13 7 7-3.13 7-7 7z"></path></g>
<g id="alarm-add"><path d="M7.88 3.39L6.6 1.86 2 5.71l1.29 1.53 4.59-3.85zM22 5.72l-4.6-3.86-1.29 1.53 4.6 3.86L22 5.72zM12 4c-4.97 0-9 4.03-9 9s4.02 9 9 9c4.97 0 9-4.03 9-9s-4.03-9-9-9zm0 16c-3.87 0-7-3.13-7-7s3.13-7 7-7 7 3.13 7 7-3.13 7-7 7zm1-11h-2v3H8v2h3v3h2v-3h3v-2h-3V9z"></path></g>
<g id="alarm-off"><path d="M12 6c3.87 0 7 3.13 7 7 0 .84-.16 1.65-.43 2.4l1.52 1.52c.58-1.19.91-2.51.91-3.92 0-4.97-4.03-9-9-9-1.41 0-2.73.33-3.92.91L9.6 6.43C10.35 6.16 11.16 6 12 6zm10-.28l-4.6-3.86-1.29 1.53 4.6 3.86L22 5.72zM2.92 2.29L1.65 3.57 2.98 4.9l-1.11.93 1.42 1.42 1.11-.94.8.8C3.83 8.69 3 10.75 3 13c0 4.97 4.02 9 9 9 2.25 0 4.31-.83 5.89-2.2l2.2 2.2 1.27-1.27L3.89 3.27l-.97-.98zm13.55 16.1C15.26 19.39 13.7 20 12 20c-3.87 0-7-3.13-7-7 0-1.7.61-3.26 1.61-4.47l9.86 9.86zM8.02 3.28L6.6 1.86l-.86.71 1.42 1.42.86-.71z"></path></g>
<g id="alarm-on"><path d="M22 5.72l-4.6-3.86-1.29 1.53 4.6 3.86L22 5.72zM7.88 3.39L6.6 1.86 2 5.71l1.29 1.53 4.59-3.85zM12 4c-4.97 0-9 4.03-9 9s4.02 9 9 9c4.97 0 9-4.03 9-9s-4.03-9-9-9zm0 16c-3.87 0-7-3.13-7-7s3.13-7 7-7 7 3.13 7 7-3.13 7-7 7zm-1.46-5.47L8.41 12.4l-1.06 1.06 3.18 3.18 6-6-1.06-1.06-4.93 4.95z"></path></g>
<g id="all-out"><path d="M16.21 4.16l4 4v-4zm4 12l-4 4h4zm-12 4l-4-4v4zm-4-12l4-4h-4zm12.95-.95c-2.73-2.73-7.17-2.73-9.9 0s-2.73 7.17 0 9.9 7.17 2.73 9.9 0 2.73-7.16 0-9.9zm-1.1 8.8c-2.13 2.13-5.57 2.13-7.7 0s-2.13-5.57 0-7.7 5.57-2.13 7.7 0 2.13 5.57 0 7.7z"></path></g>
<g id="android"><path d="M6 18c0 .55.45 1 1 1h1v3.5c0 .83.67 1.5 1.5 1.5s1.5-.67 1.5-1.5V19h2v3.5c0 .83.67 1.5 1.5 1.5s1.5-.67 1.5-1.5V19h1c.55 0 1-.45 1-1V8H6v10zM3.5 8C2.67 8 2 8.67 2 9.5v7c0 .83.67 1.5 1.5 1.5S5 17.33 5 16.5v-7C5 8.67 4.33 8 3.5 8zm17 0c-.83 0-1.5.67-1.5 1.5v7c0 .83.67 1.5 1.5 1.5s1.5-.67 1.5-1.5v-7c0-.83-.67-1.5-1.5-1.5zm-4.97-5.84l1.3-1.3c.2-.2.2-.51 0-.71-.2-.2-.51-.2-.71 0l-1.48 1.48C13.85 1.23 12.95 1 12 1c-.96 0-1.86.23-2.66.63L7.85.15c-.2-.2-.51-.2-.71 0-.2.2-.2.51 0 .71l1.31 1.31C6.97 3.26 6 5.01 6 7h12c0-1.99-.97-3.75-2.47-4.84zM10 5H9V4h1v1zm5 0h-1V4h1v1z"></path></g>
<g id="announcement"><path d="M20 2H4c-1.1 0-1.99.9-1.99 2L2 22l4-4h14c1.1 0 2-.9 2-2V4c0-1.1-.9-2-2-2zm-7 9h-2V5h2v6zm0 4h-2v-2h2v2z"></path></g>
<g id="apps"><path d="M4 8h4V4H4v4zm6 12h4v-4h-4v4zm-6 0h4v-4H4v4zm0-6h4v-4H4v4zm6 0h4v-4h-4v4zm6-10v4h4V4h-4zm-6 4h4V4h-4v4zm6 6h4v-4h-4v4zm0 6h4v-4h-4v4z"></path></g>
<g id="archive"><path d="M20.54 5.23l-1.39-1.68C18.88 3.21 18.47 3 18 3H6c-.47 0-.88.21-1.16.55L3.46 5.23C3.17 5.57 3 6.02 3 6.5V19c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V6.5c0-.48-.17-.93-.46-1.27zM12 17.5L6.5 12H10v-2h4v2h3.5L12 17.5zM5.12 5l.81-1h12l.94 1H5.12z"></path></g>
<g id="arrow-back"><path d="M20 11H7.83l5.59-5.59L12 4l-8 8 8 8 1.41-1.41L7.83 13H20v-2z"></path></g>
<g id="arrow-downward"><path d="M20 12l-1.41-1.41L13 16.17V4h-2v12.17l-5.58-5.59L4 12l8 8 8-8z"></path></g>
<g id="arrow-drop-down"><path d="M7 10l5 5 5-5z"></path></g>
<g id="arrow-drop-down-circle"><path d="M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm0 12l-4-4h8l-4 4z"></path></g>
<g id="arrow-drop-up"><path d="M7 14l5-5 5 5z"></path></g>
<g id="arrow-forward"><path d="M12 4l-1.41 1.41L16.17 11H4v2h12.17l-5.58 5.59L12 20l8-8z"></path></g>
<g id="arrow-upward"><path d="M4 12l1.41 1.41L11 7.83V20h2V7.83l5.58 5.59L20 12l-8-8-8 8z"></path></g>
<g id="aspect-ratio"><path d="M19 12h-2v3h-3v2h5v-5zM7 9h3V7H5v5h2V9zm14-6H3c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h18c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm0 16.01H3V4.99h18v14.02z"></path></g>
<g id="assessment"><path d="M19 3H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zM9 17H7v-7h2v7zm4 0h-2V7h2v10zm4 0h-2v-4h2v4z"></path></g>
<g id="assignment"><path d="M19 3h-4.18C14.4 1.84 13.3 1 12 1c-1.3 0-2.4.84-2.82 2H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm-7 0c.55 0 1 .45 1 1s-.45 1-1 1-1-.45-1-1 .45-1 1-1zm2 14H7v-2h7v2zm3-4H7v-2h10v2zm0-4H7V7h10v2z"></path></g>
<g id="assignment-ind"><path d="M19 3h-4.18C14.4 1.84 13.3 1 12 1c-1.3 0-2.4.84-2.82 2H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm-7 0c.55 0 1 .45 1 1s-.45 1-1 1-1-.45-1-1 .45-1 1-1zm0 4c1.66 0 3 1.34 3 3s-1.34 3-3 3-3-1.34-3-3 1.34-3 3-3zm6 12H6v-1.4c0-2 4-3.1 6-3.1s6 1.1 6 3.1V19z"></path></g>
<g id="assignment-late"><path d="M19 3h-4.18C14.4 1.84 13.3 1 12 1c-1.3 0-2.4.84-2.82 2H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm-6 15h-2v-2h2v2zm0-4h-2V8h2v6zm-1-9c-.55 0-1-.45-1-1s.45-1 1-1 1 .45 1 1-.45 1-1 1z"></path></g>
<g id="assignment-return"><path d="M19 3h-4.18C14.4 1.84 13.3 1 12 1c-1.3 0-2.4.84-2.82 2H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm-7 0c.55 0 1 .45 1 1s-.45 1-1 1-1-.45-1-1 .45-1 1-1zm4 12h-4v3l-5-5 5-5v3h4v4z"></path></g>
<g id="assignment-returned"><path d="M19 3h-4.18C14.4 1.84 13.3 1 12 1c-1.3 0-2.4.84-2.82 2H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm-7 0c.55 0 1 .45 1 1s-.45 1-1 1-1-.45-1-1 .45-1 1-1zm0 15l-5-5h3V9h4v4h3l-5 5z"></path></g>
<g id="assignment-turned-in"><path d="M19 3h-4.18C14.4 1.84 13.3 1 12 1c-1.3 0-2.4.84-2.82 2H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm-7 0c.55 0 1 .45 1 1s-.45 1-1 1-1-.45-1-1 .45-1 1-1zm-2 14l-4-4 1.41-1.41L10 14.17l6.59-6.59L18 9l-8 8z"></path></g>
<g id="attachment"><path d="M2 12.5C2 9.46 4.46 7 7.5 7H18c2.21 0 4 1.79 4 4s-1.79 4-4 4H9.5C8.12 15 7 13.88 7 12.5S8.12 10 9.5 10H17v2H9.41c-.55 0-.55 1 0 1H18c1.1 0 2-.9 2-2s-.9-2-2-2H7.5C5.57 9 4 10.57 4 12.5S5.57 16 7.5 16H17v2H7.5C4.46 18 2 15.54 2 12.5z"></path></g>
<g id="autorenew"><path d="M12 6v3l4-4-4-4v3c-4.42 0-8 3.58-8 8 0 1.57.46 3.03 1.24 4.26L6.7 14.8c-.45-.83-.7-1.79-.7-2.8 0-3.31 2.69-6 6-6zm6.76 1.74L17.3 9.2c.44.84.7 1.79.7 2.8 0 3.31-2.69 6-6 6v-3l-4 4 4 4v-3c4.42 0 8-3.58 8-8 0-1.57-.46-3.03-1.24-4.26z"></path></g>
<g id="backspace"><path d="M22 3H7c-.69 0-1.23.35-1.59.88L0 12l5.41 8.11c.36.53.9.89 1.59.89h15c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm-3 12.59L17.59 17 14 13.41 10.41 17 9 15.59 12.59 12 9 8.41 10.41 7 14 10.59 17.59 7 19 8.41 15.41 12 19 15.59z"></path></g>
<g id="backup"><path d="M19.35 10.04C18.67 6.59 15.64 4 12 4 9.11 4 6.6 5.64 5.35 8.04 2.34 8.36 0 10.91 0 14c0 3.31 2.69 6 6 6h13c2.76 0 5-2.24 5-5 0-2.64-2.05-4.78-4.65-4.96zM14 13v4h-4v-4H7l5-5 5 5h-3z"></path></g>
<g id="block"><path d="M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zM4 12c0-4.42 3.58-8 8-8 1.85 0 3.55.63 4.9 1.69L5.69 16.9C4.63 15.55 4 13.85 4 12zm8 8c-1.85 0-3.55-.63-4.9-1.69L18.31 7.1C19.37 8.45 20 10.15 20 12c0 4.42-3.58 8-8 8z"></path></g>
<g id="book"><path d="M18 2H6c-1.1 0-2 .9-2 2v16c0 1.1.9 2 2 2h12c1.1 0 2-.9 2-2V4c0-1.1-.9-2-2-2zM6 4h5v8l-2.5-1.5L6 12V4z"></path></g>
<g id="bookmark"><path d="M17 3H7c-1.1 0-1.99.9-1.99 2L5 21l7-3 7 3V5c0-1.1-.9-2-2-2z"></path></g>
<g id="bookmark-border"><path d="M17 3H7c-1.1 0-1.99.9-1.99 2L5 21l7-3 7 3V5c0-1.1-.9-2-2-2zm0 15l-5-2.18L7 18V5h10v13z"></path></g>
<g id="bug-report"><path d="M20 8h-2.81c-.45-.78-1.07-1.45-1.82-1.96L17 4.41 15.59 3l-2.17 2.17C12.96 5.06 12.49 5 12 5c-.49 0-.96.06-1.41.17L8.41 3 7 4.41l1.62 1.63C7.88 6.55 7.26 7.22 6.81 8H4v2h2.09c-.05.33-.09.66-.09 1v1H4v2h2v1c0 .34.04.67.09 1H4v2h2.81c1.04 1.79 2.97 3 5.19 3s4.15-1.21 5.19-3H20v-2h-2.09c.05-.33.09-.66.09-1v-1h2v-2h-2v-1c0-.34-.04-.67-.09-1H20V8zm-6 8h-4v-2h4v2zm0-4h-4v-2h4v2z"></path></g>
<g id="build"><path d="M22.7 19l-9.1-9.1c.9-2.3.4-5-1.5-6.9-2-2-5-2.4-7.4-1.3L9 6 6 9 1.6 4.7C.4 7.1.9 10.1 2.9 12.1c1.9 1.9 4.6 2.4 6.9 1.5l9.1 9.1c.4.4 1 .4 1.4 0l2.3-2.3c.5-.4.5-1.1.1-1.4z"></path></g>
<g id="cached"><path d="M19 8l-4 4h3c0 3.31-2.69 6-6 6-1.01 0-1.97-.25-2.8-.7l-1.46 1.46C8.97 19.54 10.43 20 12 20c4.42 0 8-3.58 8-8h3l-4-4zM6 12c0-3.31 2.69-6 6-6 1.01 0 1.97.25 2.8.7l1.46-1.46C15.03 4.46 13.57 4 12 4c-4.42 0-8 3.58-8 8H1l4 4 4-4H6z"></path></g>
<g id="camera-enhance"><path d="M9 3L7.17 5H4c-1.1 0-2 .9-2 2v12c0 1.1.9 2 2 2h16c1.1 0 2-.9 2-2V7c0-1.1-.9-2-2-2h-3.17L15 3H9zm3 15c-2.76 0-5-2.24-5-5s2.24-5 5-5 5 2.24 5 5-2.24 5-5 5zm0-1l1.25-2.75L16 13l-2.75-1.25L12 9l-1.25 2.75L8 13l2.75 1.25z"></path></g>
<g id="cancel"><path d="M12 2C6.47 2 2 6.47 2 12s4.47 10 10 10 10-4.47 10-10S17.53 2 12 2zm5 13.59L15.59 17 12 13.41 8.41 17 7 15.59 10.59 12 7 8.41 8.41 7 12 10.59 15.59 7 17 8.41 13.41 12 17 15.59z"></path></g>
<g id="card-giftcard"><path d="M20 6h-2.18c.11-.31.18-.65.18-1 0-1.66-1.34-3-3-3-1.05 0-1.96.54-2.5 1.35l-.5.67-.5-.68C10.96 2.54 10.05 2 9 2 7.34 2 6 3.34 6 5c0 .35.07.69.18 1H4c-1.11 0-1.99.89-1.99 2L2 19c0 1.11.89 2 2 2h16c1.11 0 2-.89 2-2V8c0-1.11-.89-2-2-2zm-5-2c.55 0 1 .45 1 1s-.45 1-1 1-1-.45-1-1 .45-1 1-1zM9 4c.55 0 1 .45 1 1s-.45 1-1 1-1-.45-1-1 .45-1 1-1zm11 15H4v-2h16v2zm0-5H4V8h5.08L7 10.83 8.62 12 11 8.76l1-1.36 1 1.36L15.38 12 17 10.83 14.92 8H20v6z"></path></g>
<g id="card-membership"><path d="M20 2H4c-1.11 0-2 .89-2 2v11c0 1.11.89 2 2 2h4v5l4-2 4 2v-5h4c1.11 0 2-.89 2-2V4c0-1.11-.89-2-2-2zm0 13H4v-2h16v2zm0-5H4V4h16v6z"></path></g>
<g id="card-travel"><path d="M20 6h-3V4c0-1.11-.89-2-2-2H9c-1.11 0-2 .89-2 2v2H4c-1.11 0-2 .89-2 2v11c0 1.11.89 2 2 2h16c1.11 0 2-.89 2-2V8c0-1.11-.89-2-2-2zM9 4h6v2H9V4zm11 15H4v-2h16v2zm0-5H4V8h3v2h2V8h6v2h2V8h3v6z"></path></g>
<g id="change-history"><path d="M12 7.77L18.39 18H5.61L12 7.77M12 4L2 20h20L12 4z"></path></g>
<g id="check"><path d="M9 16.17L4.83 12l-1.42 1.41L9 19 21 7l-1.41-1.41z"></path></g>
<g id="check-box"><path d="M19 3H5c-1.11 0-2 .9-2 2v14c0 1.1.89 2 2 2h14c1.11 0 2-.9 2-2V5c0-1.1-.89-2-2-2zm-9 14l-5-5 1.41-1.41L10 14.17l7.59-7.59L19 8l-9 9z"></path></g>
<g id="check-box-outline-blank"><path d="M19 5v14H5V5h14m0-2H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2z"></path></g>
<g id="check-circle"><path d="M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm-2 15l-5-5 1.41-1.41L10 14.17l7.59-7.59L19 8l-9 9z"></path></g>
<g id="chevron-left"><path d="M15.41 7.41L14 6l-6 6 6 6 1.41-1.41L10.83 12z"></path></g>
<g id="chevron-right"><path d="M10 6L8.59 7.41 13.17 12l-4.58 4.59L10 18l6-6z"></path></g>
<g id="chrome-reader-mode"><path d="M13 12h7v1.5h-7zm0-2.5h7V11h-7zm0 5h7V16h-7zM21 4H3c-1.1 0-2 .9-2 2v13c0 1.1.9 2 2 2h18c1.1 0 2-.9 2-2V6c0-1.1-.9-2-2-2zm0 15h-9V6h9v13z"></path></g>
<g id="class"><path d="M18 2H6c-1.1 0-2 .9-2 2v16c0 1.1.9 2 2 2h12c1.1 0 2-.9 2-2V4c0-1.1-.9-2-2-2zM6 4h5v8l-2.5-1.5L6 12V4z"></path></g>
<g id="clear"><path d="M19 6.41L17.59 5 12 10.59 6.41 5 5 6.41 10.59 12 5 17.59 6.41 19 12 13.41 17.59 19 19 17.59 13.41 12z"></path></g>
<g id="close"><path d="M19 6.41L17.59 5 12 10.59 6.41 5 5 6.41 10.59 12 5 17.59 6.41 19 12 13.41 17.59 19 19 17.59 13.41 12z"></path></g>
<g id="cloud"><path d="M19.35 10.04C18.67 6.59 15.64 4 12 4 9.11 4 6.6 5.64 5.35 8.04 2.34 8.36 0 10.91 0 14c0 3.31 2.69 6 6 6h13c2.76 0 5-2.24 5-5 0-2.64-2.05-4.78-4.65-4.96z"></path></g>
<g id="cloud-circle"><path d="M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm4.5 14H8c-1.66 0-3-1.34-3-3s1.34-3 3-3l.14.01C8.58 8.28 10.13 7 12 7c2.21 0 4 1.79 4 4h.5c1.38 0 2.5 1.12 2.5 2.5S17.88 16 16.5 16z"></path></g>
<g id="cloud-done"><path d="M19.35 10.04C18.67 6.59 15.64 4 12 4 9.11 4 6.6 5.64 5.35 8.04 2.34 8.36 0 10.91 0 14c0 3.31 2.69 6 6 6h13c2.76 0 5-2.24 5-5 0-2.64-2.05-4.78-4.65-4.96zM10 17l-3.5-3.5 1.41-1.41L10 14.17 15.18 9l1.41 1.41L10 17z"></path></g>
<g id="cloud-download"><path d="M19.35 10.04C18.67 6.59 15.64 4 12 4 9.11 4 6.6 5.64 5.35 8.04 2.34 8.36 0 10.91 0 14c0 3.31 2.69 6 6 6h13c2.76 0 5-2.24 5-5 0-2.64-2.05-4.78-4.65-4.96zM17 13l-5 5-5-5h3V9h4v4h3z"></path></g>
<g id="cloud-off"><path d="M19.35 10.04C18.67 6.59 15.64 4 12 4c-1.48 0-2.85.43-4.01 1.17l1.46 1.46C10.21 6.23 11.08 6 12 6c3.04 0 5.5 2.46 5.5 5.5v.5H19c1.66 0 3 1.34 3 3 0 1.13-.64 2.11-1.56 2.62l1.45 1.45C23.16 18.16 24 16.68 24 15c0-2.64-2.05-4.78-4.65-4.96zM3 5.27l2.75 2.74C2.56 8.15 0 10.77 0 14c0 3.31 2.69 6 6 6h11.73l2 2L21 20.73 4.27 4 3 5.27zM7.73 10l8 8H6c-2.21 0-4-1.79-4-4s1.79-4 4-4h1.73z"></path></g>
<g id="cloud-queue"><path d="M19.35 10.04C18.67 6.59 15.64 4 12 4 9.11 4 6.6 5.64 5.35 8.04 2.34 8.36 0 10.91 0 14c0 3.31 2.69 6 6 6h13c2.76 0 5-2.24 5-5 0-2.64-2.05-4.78-4.65-4.96zM19 18H6c-2.21 0-4-1.79-4-4s1.79-4 4-4h.71C7.37 7.69 9.48 6 12 6c3.04 0 5.5 2.46 5.5 5.5v.5H19c1.66 0 3 1.34 3 3s-1.34 3-3 3z"></path></g>
<g id="cloud-upload"><path d="M19.35 10.04C18.67 6.59 15.64 4 12 4 9.11 4 6.6 5.64 5.35 8.04 2.34 8.36 0 10.91 0 14c0 3.31 2.69 6 6 6h13c2.76 0 5-2.24 5-5 0-2.64-2.05-4.78-4.65-4.96zM14 13v4h-4v-4H7l5-5 5 5h-3z"></path></g>
<g id="code"><path d="M9.4 16.6L4.8 12l4.6-4.6L8 6l-6 6 6 6 1.4-1.4zm5.2 0l4.6-4.6-4.6-4.6L16 6l6 6-6 6-1.4-1.4z"></path></g>
<g id="compare-arrows"><path d="M9.01 14H2v2h7.01v3L13 15l-3.99-4v3zm5.98-1v-3H22V8h-7.01V5L11 9l3.99 4z"></path></g>
<g id="content-copy"><path d="M16 1H4c-1.1 0-2 .9-2 2v14h2V3h12V1zm3 4H8c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h11c1.1 0 2-.9 2-2V7c0-1.1-.9-2-2-2zm0 16H8V7h11v14z"></path></g>
<g id="content-cut"><path d="M9.64 7.64c.23-.5.36-1.05.36-1.64 0-2.21-1.79-4-4-4S2 3.79 2 6s1.79 4 4 4c.59 0 1.14-.13 1.64-.36L10 12l-2.36 2.36C7.14 14.13 6.59 14 6 14c-2.21 0-4 1.79-4 4s1.79 4 4 4 4-1.79 4-4c0-.59-.13-1.14-.36-1.64L12 14l7 7h3v-1L9.64 7.64zM6 8c-1.1 0-2-.89-2-2s.9-2 2-2 2 .89 2 2-.9 2-2 2zm0 12c-1.1 0-2-.89-2-2s.9-2 2-2 2 .89 2 2-.9 2-2 2zm6-7.5c-.28 0-.5-.22-.5-.5s.22-.5.5-.5.5.22.5.5-.22.5-.5.5zM19 3l-6 6 2 2 7-7V3z"></path></g>
<g id="content-paste"><path d="M19 2h-4.18C14.4.84 13.3 0 12 0c-1.3 0-2.4.84-2.82 2H5c-1.1 0-2 .9-2 2v16c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V4c0-1.1-.9-2-2-2zm-7 0c.55 0 1 .45 1 1s-.45 1-1 1-1-.45-1-1 .45-1 1-1zm7 18H5V4h2v3h10V4h2v16z"></path></g>
<g id="copyright"><path d="M10.08 10.86c.05-.33.16-.62.3-.87s.34-.46.59-.62c.24-.15.54-.22.91-.23.23.01.44.05.63.13.2.09.38.21.52.36s.25.33.34.53.13.42.14.64h1.79c-.02-.47-.11-.9-.28-1.29s-.4-.73-.7-1.01-.66-.5-1.08-.66-.88-.23-1.39-.23c-.65 0-1.22.11-1.7.34s-.88.53-1.2.92-.56.84-.71 1.36S8 11.29 8 11.87v.27c0 .58.08 1.12.23 1.64s.39.97.71 1.35.72.69 1.2.91 1.05.34 1.7.34c.47 0 .91-.08 1.32-.23s.77-.36 1.08-.63.56-.58.74-.94.29-.74.3-1.15h-1.79c-.01.21-.06.4-.15.58s-.21.33-.36.46-.32.23-.52.3c-.19.07-.39.09-.6.1-.36-.01-.66-.08-.89-.23-.25-.16-.45-.37-.59-.62s-.25-.55-.3-.88-.08-.67-.08-1v-.27c0-.35.03-.68.08-1.01zM12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm0 18c-4.41 0-8-3.59-8-8s3.59-8 8-8 8 3.59 8 8-3.59 8-8 8z"></path></g>
<g id="create"><path d="M3 17.25V21h3.75L17.81 9.94l-3.75-3.75L3 17.25zM20.71 7.04c.39-.39.39-1.02 0-1.41l-2.34-2.34c-.39-.39-1.02-.39-1.41 0l-1.83 1.83 3.75 3.75 1.83-1.83z"></path></g>
<g id="create-new-folder"><path d="M20 6h-8l-2-2H4c-1.11 0-1.99.89-1.99 2L2 18c0 1.11.89 2 2 2h16c1.11 0 2-.89 2-2V8c0-1.11-.89-2-2-2zm-1 8h-3v3h-2v-3h-3v-2h3V9h2v3h3v2z"></path></g>
<g id="credit-card"><path d="M20 4H4c-1.11 0-1.99.89-1.99 2L2 18c0 1.11.89 2 2 2h16c1.11 0 2-.89 2-2V6c0-1.11-.89-2-2-2zm0 14H4v-6h16v6zm0-10H4V6h16v2z"></path></g>
<g id="dashboard"><path d="M3 13h8V3H3v10zm0 8h8v-6H3v6zm10 0h8V11h-8v10zm0-18v6h8V3h-8z"></path></g>
<g id="date-range"><path d="M9 11H7v2h2v-2zm4 0h-2v2h2v-2zm4 0h-2v2h2v-2zm2-7h-1V2h-2v2H8V2H6v2H5c-1.11 0-1.99.9-1.99 2L3 20c0 1.1.89 2 2 2h14c1.1 0 2-.9 2-2V6c0-1.1-.9-2-2-2zm0 16H5V9h14v11z"></path></g>
<g id="delete"><path d="M6 19c0 1.1.9 2 2 2h8c1.1 0 2-.9 2-2V7H6v12zM19 4h-3.5l-1-1h-5l-1 1H5v2h14V4z"></path></g>
<g id="delete-forever"><path d="M6 19c0 1.1.9 2 2 2h8c1.1 0 2-.9 2-2V7H6v12zm2.46-7.12l1.41-1.41L12 12.59l2.12-2.12 1.41 1.41L13.41 14l2.12 2.12-1.41 1.41L12 15.41l-2.12 2.12-1.41-1.41L10.59 14l-2.13-2.12zM15.5 4l-1-1h-5l-1 1H5v2h14V4z"></path></g>
<g id="delete-sweep"><path d="M15 16h4v2h-4zm0-8h7v2h-7zm0 4h6v2h-6zM3 18c0 1.1.9 2 2 2h6c1.1 0 2-.9 2-2V8H3v10zM14 5h-3l-1-1H6L5 5H2v2h12z"></path></g>
<g id="description"><path d="M14 2H6c-1.1 0-1.99.9-1.99 2L4 20c0 1.1.89 2 1.99 2H18c1.1 0 2-.9 2-2V8l-6-6zm2 16H8v-2h8v2zm0-4H8v-2h8v2zm-3-5V3.5L18.5 9H13z"></path></g>
<g id="dns"><path d="M20 13H4c-.55 0-1 .45-1 1v6c0 .55.45 1 1 1h16c.55 0 1-.45 1-1v-6c0-.55-.45-1-1-1zM7 19c-1.1 0-2-.9-2-2s.9-2 2-2 2 .9 2 2-.9 2-2 2zM20 3H4c-.55 0-1 .45-1 1v6c0 .55.45 1 1 1h16c.55 0 1-.45 1-1V4c0-.55-.45-1-1-1zM7 9c-1.1 0-2-.9-2-2s.9-2 2-2 2 .9 2 2-.9 2-2 2z"></path></g>
<g id="done"><path d="M9 16.2L4.8 12l-1.4 1.4L9 19 21 7l-1.4-1.4L9 16.2z"></path></g>
<g id="done-all"><path d="M18 7l-1.41-1.41-6.34 6.34 1.41 1.41L18 7zm4.24-1.41L11.66 16.17 7.48 12l-1.41 1.41L11.66 19l12-12-1.42-1.41zM.41 13.41L6 19l1.41-1.41L1.83 12 .41 13.41z"></path></g>
<g id="donut-large"><path d="M11 5.08V2c-5 .5-9 4.81-9 10s4 9.5 9 10v-3.08c-3-.48-6-3.4-6-6.92s3-6.44 6-6.92zM18.97 11H22c-.47-5-4-8.53-9-9v3.08C16 5.51 18.54 8 18.97 11zM13 18.92V22c5-.47 8.53-4 9-9h-3.03c-.43 3-2.97 5.49-5.97 5.92z"></path></g>
<g id="donut-small"><path d="M11 9.16V2c-5 .5-9 4.79-9 10s4 9.5 9 10v-7.16c-1-.41-2-1.52-2-2.84s1-2.43 2-2.84zM14.86 11H22c-.48-4.75-4-8.53-9-9v7.16c1 .3 1.52.98 1.86 1.84zM13 14.84V22c5-.47 8.52-4.25 9-9h-7.14c-.34.86-.86 1.54-1.86 1.84z"></path></g>
<g id="drafts"><path d="M21.99 8c0-.72-.37-1.35-.94-1.7L12 1 2.95 6.3C2.38 6.65 2 7.28 2 8v10c0 1.1.9 2 2 2h16c1.1 0 2-.9 2-2l-.01-10zM12 13L3.74 7.84 12 3l8.26 4.84L12 13z"></path></g>
<g id="eject"><path d="M5 17h14v2H5zm7-12L5.33 15h13.34z"></path></g>
<g id="error"><path d="M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm1 15h-2v-2h2v2zm0-4h-2V7h2v6z"></path></g>
<g id="error-outline"><path d="M11 15h2v2h-2zm0-8h2v6h-2zm.99-5C6.47 2 2 6.48 2 12s4.47 10 9.99 10C17.52 22 22 17.52 22 12S17.52 2 11.99 2zM12 20c-4.42 0-8-3.58-8-8s3.58-8 8-8 8 3.58 8 8-3.58 8-8 8z"></path></g>
<g id="euro-symbol"><path d="M15 18.5c-2.51 0-4.68-1.42-5.76-3.5H15v-2H8.58c-.05-.33-.08-.66-.08-1s.03-.67.08-1H15V9H9.24C10.32 6.92 12.5 5.5 15 5.5c1.61 0 3.09.59 4.23 1.57L21 5.3C19.41 3.87 17.3 3 15 3c-3.92 0-7.24 2.51-8.48 6H3v2h3.06c-.04.33-.06.66-.06 1 0 .34.02.67.06 1H3v2h3.52c1.24 3.49 4.56 6 8.48 6 2.31 0 4.41-.87 6-2.3l-1.78-1.77c-1.13.98-2.6 1.57-4.22 1.57z"></path></g>
<g id="event"><path d="M17 12h-5v5h5v-5zM16 1v2H8V1H6v2H5c-1.11 0-1.99.9-1.99 2L3 19c0 1.1.89 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2h-1V1h-2zm3 18H5V8h14v11z"></path></g>
<g id="event-seat"><path d="M4 18v3h3v-3h10v3h3v-6H4zm15-8h3v3h-3zM2 10h3v3H2zm15 3H7V5c0-1.1.9-2 2-2h6c1.1 0 2 .9 2 2v8z"></path></g>
<g id="exit-to-app"><path d="M10.09 15.59L11.5 17l5-5-5-5-1.41 1.41L12.67 11H3v2h9.67l-2.58 2.59zM19 3H5c-1.11 0-2 .9-2 2v4h2V5h14v14H5v-4H3v4c0 1.1.89 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2z"></path></g>
<g id="expand-less"><path d="M12 8l-6 6 1.41 1.41L12 10.83l4.59 4.58L18 14z"></path></g>
<g id="expand-more"><path d="M16.59 8.59L12 13.17 7.41 8.59 6 10l6 6 6-6z"></path></g>
<g id="explore"><path d="M12 10.9c-.61 0-1.1.49-1.1 1.1s.49 1.1 1.1 1.1c.61 0 1.1-.49 1.1-1.1s-.49-1.1-1.1-1.1zM12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm2.19 12.19L6 18l3.81-8.19L18 6l-3.81 8.19z"></path></g>
<g id="extension"><path d="M20.5 11H19V7c0-1.1-.9-2-2-2h-4V3.5C13 2.12 11.88 1 10.5 1S8 2.12 8 3.5V5H4c-1.1 0-1.99.9-1.99 2v3.8H3.5c1.49 0 2.7 1.21 2.7 2.7s-1.21 2.7-2.7 2.7H2V20c0 1.1.9 2 2 2h3.8v-1.5c0-1.49 1.21-2.7 2.7-2.7 1.49 0 2.7 1.21 2.7 2.7V22H17c1.1 0 2-.9 2-2v-4h1.5c1.38 0 2.5-1.12 2.5-2.5S21.88 11 20.5 11z"></path></g>
<g id="face"><path d="M9 11.75c-.69 0-1.25.56-1.25 1.25s.56 1.25 1.25 1.25 1.25-.56 1.25-1.25-.56-1.25-1.25-1.25zm6 0c-.69 0-1.25.56-1.25 1.25s.56 1.25 1.25 1.25 1.25-.56 1.25-1.25-.56-1.25-1.25-1.25zM12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm0 18c-4.41 0-8-3.59-8-8 0-.29.02-.58.05-.86 2.36-1.05 4.23-2.98 5.21-5.37C11.07 8.33 14.05 10 17.42 10c.78 0 1.53-.09 2.25-.26.21.71.33 1.47.33 2.26 0 4.41-3.59 8-8 8z"></path></g>
<g id="favorite"><path d="M12 21.35l-1.45-1.32C5.4 15.36 2 12.28 2 8.5 2 5.42 4.42 3 7.5 3c1.74 0 3.41.81 4.5 2.09C13.09 3.81 14.76 3 16.5 3 19.58 3 22 5.42 22 8.5c0 3.78-3.4 6.86-8.55 11.54L12 21.35z"></path></g>
<g id="favorite-border"><path d="M16.5 3c-1.74 0-3.41.81-4.5 2.09C10.91 3.81 9.24 3 7.5 3 4.42 3 2 5.42 2 8.5c0 3.78 3.4 6.86 8.55 11.54L12 21.35l1.45-1.32C18.6 15.36 22 12.28 22 8.5 22 5.42 19.58 3 16.5 3zm-4.4 15.55l-.1.1-.1-.1C7.14 14.24 4 11.39 4 8.5 4 6.5 5.5 5 7.5 5c1.54 0 3.04.99 3.57 2.36h1.87C13.46 5.99 14.96 5 16.5 5c2 0 3.5 1.5 3.5 3.5 0 2.89-3.14 5.74-7.9 10.05z"></path></g>
<g id="feedback"><path d="M20 2H4c-1.1 0-1.99.9-1.99 2L2 22l4-4h14c1.1 0 2-.9 2-2V4c0-1.1-.9-2-2-2zm-7 12h-2v-2h2v2zm0-4h-2V6h2v4z"></path></g>
<g id="file-download"><path d="M19 9h-4V3H9v6H5l7 7 7-7zM5 18v2h14v-2H5z"></path></g>
<g id="file-upload"><path d="M9 16h6v-6h4l-7-7-7 7h4zm-4 2h14v2H5z"></path></g>
<g id="filter-list"><path d="M10 18h4v-2h-4v2zM3 6v2h18V6H3zm3 7h12v-2H6v2z"></path></g>
<g id="find-in-page"><path d="M20 19.59V8l-6-6H6c-1.1 0-1.99.9-1.99 2L4 20c0 1.1.89 2 1.99 2H18c.45 0 .85-.15 1.19-.4l-4.43-4.43c-.8.52-1.74.83-2.76.83-2.76 0-5-2.24-5-5s2.24-5 5-5 5 2.24 5 5c0 1.02-.31 1.96-.83 2.75L20 19.59zM9 13c0 1.66 1.34 3 3 3s3-1.34 3-3-1.34-3-3-3-3 1.34-3 3z"></path></g>
<g id="find-replace"><path d="M11 6c1.38 0 2.63.56 3.54 1.46L12 10h6V4l-2.05 2.05C14.68 4.78 12.93 4 11 4c-3.53 0-6.43 2.61-6.92 6H6.1c.46-2.28 2.48-4 4.9-4zm5.64 9.14c.66-.9 1.12-1.97 1.28-3.14H15.9c-.46 2.28-2.48 4-4.9 4-1.38 0-2.63-.56-3.54-1.46L10 12H4v6l2.05-2.05C7.32 17.22 9.07 18 11 18c1.55 0 2.98-.51 4.14-1.36L20 21.49 21.49 20l-4.85-4.86z"></path></g>
<g id="fingerprint"><path d="M17.81 4.47c-.08 0-.16-.02-.23-.06C15.66 3.42 14 3 12.01 3c-1.98 0-3.86.47-5.57 1.41-.24.13-.54.04-.68-.2-.13-.24-.04-.55.2-.68C7.82 2.52 9.86 2 12.01 2c2.13 0 3.99.47 6.03 1.52.25.13.34.43.21.67-.09.18-.26.28-.44.28zM3.5 9.72c-.1 0-.2-.03-.29-.09-.23-.16-.28-.47-.12-.7.99-1.4 2.25-2.5 3.75-3.27C9.98 4.04 14 4.03 17.15 5.65c1.5.77 2.76 1.86 3.75 3.25.16.22.11.54-.12.7-.23.16-.54.11-.7-.12-.9-1.26-2.04-2.25-3.39-2.94-2.87-1.47-6.54-1.47-9.4.01-1.36.7-2.5 1.7-3.4 2.96-.08.14-.23.21-.39.21zm6.25 12.07c-.13 0-.26-.05-.35-.15-.87-.87-1.34-1.43-2.01-2.64-.69-1.23-1.05-2.73-1.05-4.34 0-2.97 2.54-5.39 5.66-5.39s5.66 2.42 5.66 5.39c0 .28-.22.5-.5.5s-.5-.22-.5-.5c0-2.42-2.09-4.39-4.66-4.39-2.57 0-4.66 1.97-4.66 4.39 0 1.44.32 2.77.93 3.85.64 1.15 1.08 1.64 1.85 2.42.19.2.19.51 0 .71-.11.1-.24.15-.37.15zm7.17-1.85c-1.19 0-2.24-.3-3.1-.89-1.49-1.01-2.38-2.65-2.38-4.39 0-.28.22-.5.5-.5s.5.22.5.5c0 1.41.72 2.74 1.94 3.56.71.48 1.54.71 2.54.71.24 0 .64-.03 1.04-.1.27-.05.53.13.58.41.05.27-.13.53-.41.58-.57.11-1.07.12-1.21.12zM14.91 22c-.04 0-.09-.01-.13-.02-1.59-.44-2.63-1.03-3.72-2.1-1.4-1.39-2.17-3.24-2.17-5.22 0-1.62 1.38-2.94 3.08-2.94 1.7 0 3.08 1.32 3.08 2.94 0 1.07.93 1.94 2.08 1.94s2.08-.87 2.08-1.94c0-3.77-3.25-6.83-7.25-6.83-2.84 0-5.44 1.58-6.61 4.03-.39.81-.59 1.76-.59 2.8 0 .78.07 2.01.67 3.61.1.26-.03.55-.29.64-.26.1-.55-.04-.64-.29-.49-1.31-.73-2.61-.73-3.96 0-1.2.23-2.29.68-3.24 1.33-2.79 4.28-4.6 7.51-4.6 4.55 0 8.25 3.51 8.25 7.83 0 1.62-1.38 2.94-3.08 2.94s-3.08-1.32-3.08-2.94c0-1.07-.93-1.94-2.08-1.94s-2.08.87-2.08 1.94c0 1.71.66 3.31 1.87 4.51.95.94 1.86 1.46 3.27 1.85.27.07.42.35.35.61-.05.23-.26.38-.47.38z"></path></g>
<g id="first-page"><path d="M18.41 16.59L13.82 12l4.59-4.59L17 6l-6 6 6 6zM6 6h2v12H6z"></path></g>
<g id="flag"><path d="M14.4 6L14 4H5v17h2v-7h5.6l.4 2h7V6z"></path></g>
<g id="flight-land"><path d="M2.5 19h19v2h-19zm7.18-5.73l4.35 1.16 5.31 1.42c.8.21 1.62-.26 1.84-1.06.21-.8-.26-1.62-1.06-1.84l-5.31-1.42-2.76-9.02L10.12 2v8.28L5.15 8.95l-.93-2.32-1.45-.39v5.17l1.6.43 5.31 1.43z"></path></g>
<g id="flight-takeoff"><path d="M2.5 19h19v2h-19zm19.57-9.36c-.21-.8-1.04-1.28-1.84-1.06L14.92 10l-6.9-6.43-1.93.51 4.14 7.17-4.97 1.33-1.97-1.54-1.45.39 1.82 3.16.77 1.33 1.6-.43 5.31-1.42 4.35-1.16L21 11.49c.81-.23 1.28-1.05 1.07-1.85z"></path></g>
<g id="flip-to-back"><path d="M9 7H7v2h2V7zm0 4H7v2h2v-2zm0-8c-1.11 0-2 .9-2 2h2V3zm4 12h-2v2h2v-2zm6-12v2h2c0-1.1-.9-2-2-2zm-6 0h-2v2h2V3zM9 17v-2H7c0 1.1.89 2 2 2zm10-4h2v-2h-2v2zm0-4h2V7h-2v2zm0 8c1.1 0 2-.9 2-2h-2v2zM5 7H3v12c0 1.1.89 2 2 2h12v-2H5V7zm10-2h2V3h-2v2zm0 12h2v-2h-2v2z"></path></g>
<g id="flip-to-front"><path d="M3 13h2v-2H3v2zm0 4h2v-2H3v2zm2 4v-2H3c0 1.1.89 2 2 2zM3 9h2V7H3v2zm12 12h2v-2h-2v2zm4-18H9c-1.11 0-2 .9-2 2v10c0 1.1.89 2 2 2h10c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm0 12H9V5h10v10zm-8 6h2v-2h-2v2zm-4 0h2v-2H7v2z"></path></g>
<g id="folder"><path d="M10 4H4c-1.1 0-1.99.9-1.99 2L2 18c0 1.1.9 2 2 2h16c1.1 0 2-.9 2-2V8c0-1.1-.9-2-2-2h-8l-2-2z"></path></g>
<g id="folder-open"><path d="M20 6h-8l-2-2H4c-1.1 0-1.99.9-1.99 2L2 18c0 1.1.9 2 2 2h16c1.1 0 2-.9 2-2V8c0-1.1-.9-2-2-2zm0 12H4V8h16v10z"></path></g>
<g id="folder-shared"><path d="M20 6h-8l-2-2H4c-1.1 0-1.99.9-1.99 2L2 18c0 1.1.9 2 2 2h16c1.1 0 2-.9 2-2V8c0-1.1-.9-2-2-2zm-5 3c1.1 0 2 .9 2 2s-.9 2-2 2-2-.9-2-2 .9-2 2-2zm4 8h-8v-1c0-1.33 2.67-2 4-2s4 .67 4 2v1z"></path></g>
<g id="font-download"><path d="M9.93 13.5h4.14L12 7.98zM20 2H4c-1.1 0-2 .9-2 2v16c0 1.1.9 2 2 2h16c1.1 0 2-.9 2-2V4c0-1.1-.9-2-2-2zm-4.05 16.5l-1.14-3H9.17l-1.12 3H5.96l5.11-13h1.86l5.11 13h-2.09z"></path></g>
<g id="forward"><path d="M12 8V4l8 8-8 8v-4H4V8z"></path></g>
<g id="fullscreen"><path d="M7 14H5v5h5v-2H7v-3zm-2-4h2V7h3V5H5v5zm12 7h-3v2h5v-5h-2v3zM14 5v2h3v3h2V5h-5z"></path></g>
<g id="fullscreen-exit"><path d="M5 16h3v3h2v-5H5v2zm3-8H5v2h5V5H8v3zm6 11h2v-3h3v-2h-5v5zm2-11V5h-2v5h5V8h-3z"></path></g>
<g id="g-translate"><path d="M20 5h-9.12L10 2H4c-1.1 0-2 .9-2 2v13c0 1.1.9 2 2 2h7l1 3h8c1.1 0 2-.9 2-2V7c0-1.1-.9-2-2-2zM7.17 14.59c-2.25 0-4.09-1.83-4.09-4.09s1.83-4.09 4.09-4.09c1.04 0 1.99.37 2.74 1.07l.07.06-1.23 1.18-.06-.05c-.29-.27-.78-.59-1.52-.59-1.31 0-2.38 1.09-2.38 2.42s1.07 2.42 2.38 2.42c1.37 0 1.96-.87 2.12-1.46H7.08V9.91h3.95l.01.07c.04.21.05.4.05.61 0 2.35-1.61 4-3.92 4zm6.03-1.71c.33.6.74 1.18 1.19 1.7l-.54.53-.65-2.23zm.77-.76h-.99l-.31-1.04h3.99s-.34 1.31-1.56 2.74c-.52-.62-.89-1.23-1.13-1.7zM21 20c0 .55-.45 1-1 1h-7l2-2-.81-2.77.92-.92L17.79 18l.73-.73-2.71-2.68c.9-1.03 1.6-2.25 1.92-3.51H19v-1.04h-3.64V9h-1.04v1.04h-1.96L11.18 6H20c.55 0 1 .45 1 1v13z"></path></g>
<g id="gavel"><path d="M1 21h12v2H1zM5.245 8.07l2.83-2.827 14.14 14.142-2.828 2.828zM12.317 1l5.657 5.656-2.83 2.83-5.654-5.66zM3.825 9.485l5.657 5.657-2.828 2.828-5.657-5.657z"></path></g>
<g id="gesture"><path d="M4.59 6.89c.7-.71 1.4-1.35 1.71-1.22.5.2 0 1.03-.3 1.52-.25.42-2.86 3.89-2.86 6.31 0 1.28.48 2.34 1.34 2.98.75.56 1.74.73 2.64.46 1.07-.31 1.95-1.4 3.06-2.77 1.21-1.49 2.83-3.44 4.08-3.44 1.63 0 1.65 1.01 1.76 1.79-3.78.64-5.38 3.67-5.38 5.37 0 1.7 1.44 3.09 3.21 3.09 1.63 0 4.29-1.33 4.69-6.1H21v-2.5h-2.47c-.15-1.65-1.09-4.2-4.03-4.2-2.25 0-4.18 1.91-4.94 2.84-.58.73-2.06 2.48-2.29 2.72-.25.3-.68.84-1.11.84-.45 0-.72-.83-.36-1.92.35-1.09 1.4-2.86 1.85-3.52.78-1.14 1.3-1.92 1.3-3.28C8.95 3.69 7.31 3 6.44 3 5.12 3 3.97 4 3.72 4.25c-.36.36-.66.66-.88.93l1.75 1.71zm9.29 11.66c-.31 0-.74-.26-.74-.72 0-.6.73-2.2 2.87-2.76-.3 2.69-1.43 3.48-2.13 3.48z"></path></g>
<g id="get-app"><path d="M19 9h-4V3H9v6H5l7 7 7-7zM5 18v2h14v-2H5z"></path></g>
<g id="gif"><path d="M11.5 9H13v6h-1.5zM9 9H6c-.6 0-1 .5-1 1v4c0 .5.4 1 1 1h3c.6 0 1-.5 1-1v-2H8.5v1.5h-2v-3H10V10c0-.5-.4-1-1-1zm10 1.5V9h-4.5v6H16v-2h2v-1.5h-2v-1z"></path></g>
<g id="grade"><path d="M12 17.27L18.18 21l-1.64-7.03L22 9.24l-7.19-.61L12 2 9.19 8.63 2 9.24l5.46 4.73L5.82 21z"></path></g>
<g id="group-work"><path d="M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zM8 17.5c-1.38 0-2.5-1.12-2.5-2.5s1.12-2.5 2.5-2.5 2.5 1.12 2.5 2.5-1.12 2.5-2.5 2.5zM9.5 8c0-1.38 1.12-2.5 2.5-2.5s2.5 1.12 2.5 2.5-1.12 2.5-2.5 2.5S9.5 9.38 9.5 8zm6.5 9.5c-1.38 0-2.5-1.12-2.5-2.5s1.12-2.5 2.5-2.5 2.5 1.12 2.5 2.5-1.12 2.5-2.5 2.5z"></path></g>
<g id="help"><path d="M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm1 17h-2v-2h2v2zm2.07-7.75l-.9.92C13.45 12.9 13 13.5 13 15h-2v-.5c0-1.1.45-2.1 1.17-2.83l1.24-1.26c.37-.36.59-.86.59-1.41 0-1.1-.9-2-2-2s-2 .9-2 2H8c0-2.21 1.79-4 4-4s4 1.79 4 4c0 .88-.36 1.68-.93 2.25z"></path></g>
<g id="help-outline"><path d="M11 18h2v-2h-2v2zm1-16C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm0 18c-4.41 0-8-3.59-8-8s3.59-8 8-8 8 3.59 8 8-3.59 8-8 8zm0-14c-2.21 0-4 1.79-4 4h2c0-1.1.9-2 2-2s2 .9 2 2c0 2-3 1.75-3 5h2c0-2.25 3-2.5 3-5 0-2.21-1.79-4-4-4z"></path></g>
<g id="highlight-off"><path d="M14.59 8L12 10.59 9.41 8 8 9.41 10.59 12 8 14.59 9.41 16 12 13.41 14.59 16 16 14.59 13.41 12 16 9.41 14.59 8zM12 2C6.47 2 2 6.47 2 12s4.47 10 10 10 10-4.47 10-10S17.53 2 12 2zm0 18c-4.41 0-8-3.59-8-8s3.59-8 8-8 8 3.59 8 8-3.59 8-8 8z"></path></g>
<g id="history"><path d="M13 3c-4.97 0-9 4.03-9 9H1l3.89 3.89.07.14L9 12H6c0-3.87 3.13-7 7-7s7 3.13 7 7-3.13 7-7 7c-1.93 0-3.68-.79-4.94-2.06l-1.42 1.42C8.27 19.99 10.51 21 13 21c4.97 0 9-4.03 9-9s-4.03-9-9-9zm-1 5v5l4.28 2.54.72-1.21-3.5-2.08V8H12z"></path></g>
<g id="home"><path d="M10 20v-6h4v6h5v-8h3L12 3 2 12h3v8z"></path></g>
<g id="hourglass-empty"><path d="M6 2v6h.01L6 8.01 10 12l-4 4 .01.01H6V22h12v-5.99h-.01L18 16l-4-4 4-3.99-.01-.01H18V2H6zm10 14.5V20H8v-3.5l4-4 4 4zm-4-5l-4-4V4h8v3.5l-4 4z"></path></g>
<g id="hourglass-full"><path d="M6 2v6h.01L6 8.01 10 12l-4 4 .01.01H6V22h12v-5.99h-.01L18 16l-4-4 4-3.99-.01-.01H18V2H6z"></path></g>
<g id="http"><path d="M4.5 11h-2V9H1v6h1.5v-2.5h2V15H6V9H4.5v2zm2.5-.5h1.5V15H10v-4.5h1.5V9H7v1.5zm5.5 0H14V15h1.5v-4.5H17V9h-4.5v1.5zm9-1.5H18v6h1.5v-2h2c.8 0 1.5-.7 1.5-1.5v-1c0-.8-.7-1.5-1.5-1.5zm0 2.5h-2v-1h2v1z"></path></g>
<g id="https"><path d="M18 8h-1V6c0-2.76-2.24-5-5-5S7 3.24 7 6v2H6c-1.1 0-2 .9-2 2v10c0 1.1.9 2 2 2h12c1.1 0 2-.9 2-2V10c0-1.1-.9-2-2-2zm-6 9c-1.1 0-2-.9-2-2s.9-2 2-2 2 .9 2 2-.9 2-2 2zm3.1-9H8.9V6c0-1.71 1.39-3.1 3.1-3.1 1.71 0 3.1 1.39 3.1 3.1v2z"></path></g>
<g id="important-devices"><path d="M23 11.01L18 11c-.55 0-1 .45-1 1v9c0 .55.45 1 1 1h5c.55 0 1-.45 1-1v-9c0-.55-.45-.99-1-.99zM23 20h-5v-7h5v7zM20 2H2C.89 2 0 2.89 0 4v12c0 1.1.89 2 2 2h7v2H7v2h8v-2h-2v-2h2v-2H2V4h18v5h2V4c0-1.11-.9-2-2-2zm-8.03 7L11 6l-.97 3H7l2.47 1.76-.94 2.91 2.47-1.8 2.47 1.8-.94-2.91L15 9h-3.03z"></path></g>
<g id="inbox"><path d="M19 3H4.99c-1.11 0-1.98.89-1.98 2L3 19c0 1.1.88 2 1.99 2H19c1.1 0 2-.9 2-2V5c0-1.11-.9-2-2-2zm0 12h-4c0 1.66-1.35 3-3 3s-3-1.34-3-3H4.99V5H19v10z"></path></g>
<g id="indeterminate-check-box"><path d="M19 3H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm-2 10H7v-2h10v2z"></path></g>
<g id="info"><path d="M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm1 15h-2v-6h2v6zm0-8h-2V7h2v2z"></path></g>
<g id="info-outline"><path d="M11 17h2v-6h-2v6zm1-15C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm0 18c-4.41 0-8-3.59-8-8s3.59-8 8-8 8 3.59 8 8-3.59 8-8 8zM11 9h2V7h-2v2z"></path></g>
<g id="input"><path d="M21 3.01H3c-1.1 0-2 .9-2 2V9h2V4.99h18v14.03H3V15H1v4.01c0 1.1.9 1.98 2 1.98h18c1.1 0 2-.88 2-1.98v-14c0-1.11-.9-2-2-2zM11 16l4-4-4-4v3H1v2h10v3z"></path></g>
<g id="invert-colors"><path d="M17.66 7.93L12 2.27 6.34 7.93c-3.12 3.12-3.12 8.19 0 11.31C7.9 20.8 9.95 21.58 12 21.58c2.05 0 4.1-.78 5.66-2.34 3.12-3.12 3.12-8.19 0-11.31zM12 19.59c-1.6 0-3.11-.62-4.24-1.76C6.62 16.69 6 15.19 6 13.59s.62-3.11 1.76-4.24L12 5.1v14.49z"></path></g>
<g id="label"><path d="M17.63 5.84C17.27 5.33 16.67 5 16 5L5 5.01C3.9 5.01 3 5.9 3 7v10c0 1.1.9 1.99 2 1.99L16 19c.67 0 1.27-.33 1.63-.84L22 12l-4.37-6.16z"></path></g>
<g id="label-outline"><path d="M17.63 5.84C17.27 5.33 16.67 5 16 5L5 5.01C3.9 5.01 3 5.9 3 7v10c0 1.1.9 1.99 2 1.99L16 19c.67 0 1.27-.33 1.63-.84L22 12l-4.37-6.16zM16 17H5V7h11l3.55 5L16 17z"></path></g>
<g id="language"><path d="M11.99 2C6.47 2 2 6.48 2 12s4.47 10 9.99 10C17.52 22 22 17.52 22 12S17.52 2 11.99 2zm6.93 6h-2.95c-.32-1.25-.78-2.45-1.38-3.56 1.84.63 3.37 1.91 4.33 3.56zM12 4.04c.83 1.2 1.48 2.53 1.91 3.96h-3.82c.43-1.43 1.08-2.76 1.91-3.96zM4.26 14C4.1 13.36 4 12.69 4 12s.1-1.36.26-2h3.38c-.08.66-.14 1.32-.14 2 0 .68.06 1.34.14 2H4.26zm.82 2h2.95c.32 1.25.78 2.45 1.38 3.56-1.84-.63-3.37-1.9-4.33-3.56zm2.95-8H5.08c.96-1.66 2.49-2.93 4.33-3.56C8.81 5.55 8.35 6.75 8.03 8zM12 19.96c-.83-1.2-1.48-2.53-1.91-3.96h3.82c-.43 1.43-1.08 2.76-1.91 3.96zM14.34 14H9.66c-.09-.66-.16-1.32-.16-2 0-.68.07-1.35.16-2h4.68c.09.65.16 1.32.16 2 0 .68-.07 1.34-.16 2zm.25 5.56c.6-1.11 1.06-2.31 1.38-3.56h2.95c-.96 1.65-2.49 2.93-4.33 3.56zM16.36 14c.08-.66.14-1.32.14-2 0-.68-.06-1.34-.14-2h3.38c.16.64.26 1.31.26 2s-.1 1.36-.26 2h-3.38z"></path></g>
<g id="last-page"><path d="M5.59 7.41L10.18 12l-4.59 4.59L7 18l6-6-6-6zM16 6h2v12h-2z"></path></g>
<g id="launch"><path d="M19 19H5V5h7V3H5c-1.11 0-2 .9-2 2v14c0 1.1.89 2 2 2h14c1.1 0 2-.9 2-2v-7h-2v7zM14 3v2h3.59l-9.83 9.83 1.41 1.41L19 6.41V10h2V3h-7z"></path></g>
<g id="lightbulb-outline"><path d="M9 21c0 .55.45 1 1 1h4c.55 0 1-.45 1-1v-1H9v1zm3-19C8.14 2 5 5.14 5 9c0 2.38 1.19 4.47 3 5.74V17c0 .55.45 1 1 1h6c.55 0 1-.45 1-1v-2.26c1.81-1.27 3-3.36 3-5.74 0-3.86-3.14-7-7-7zm2.85 11.1l-.85.6V16h-4v-2.3l-.85-.6C7.8 12.16 7 10.63 7 9c0-2.76 2.24-5 5-5s5 2.24 5 5c0 1.63-.8 3.16-2.15 4.1z"></path></g>
<g id="line-style"><path d="M3 16h5v-2H3v2zm6.5 0h5v-2h-5v2zm6.5 0h5v-2h-5v2zM3 20h2v-2H3v2zm4 0h2v-2H7v2zm4 0h2v-2h-2v2zm4 0h2v-2h-2v2zm4 0h2v-2h-2v2zM3 12h8v-2H3v2zm10 0h8v-2h-8v2zM3 4v4h18V4H3z"></path></g>
<g id="line-weight"><path d="M3 17h18v-2H3v2zm0 3h18v-1H3v1zm0-7h18v-3H3v3zm0-9v4h18V4H3z"></path></g>
<g id="link"><path d="M3.9 12c0-1.71 1.39-3.1 3.1-3.1h4V7H7c-2.76 0-5 2.24-5 5s2.24 5 5 5h4v-1.9H7c-1.71 0-3.1-1.39-3.1-3.1zM8 13h8v-2H8v2zm9-6h-4v1.9h4c1.71 0 3.1 1.39 3.1 3.1s-1.39 3.1-3.1 3.1h-4V17h4c2.76 0 5-2.24 5-5s-2.24-5-5-5z"></path></g>
<g id="list"><path d="M3 13h2v-2H3v2zm0 4h2v-2H3v2zm0-8h2V7H3v2zm4 4h14v-2H7v2zm0 4h14v-2H7v2zM7 7v2h14V7H7z"></path></g>
<g id="lock"><path d="M18 8h-1V6c0-2.76-2.24-5-5-5S7 3.24 7 6v2H6c-1.1 0-2 .9-2 2v10c0 1.1.9 2 2 2h12c1.1 0 2-.9 2-2V10c0-1.1-.9-2-2-2zm-6 9c-1.1 0-2-.9-2-2s.9-2 2-2 2 .9 2 2-.9 2-2 2zm3.1-9H8.9V6c0-1.71 1.39-3.1 3.1-3.1 1.71 0 3.1 1.39 3.1 3.1v2z"></path></g>
<g id="lock-open"><path d="M12 17c1.1 0 2-.9 2-2s-.9-2-2-2-2 .9-2 2 .9 2 2 2zm6-9h-1V6c0-2.76-2.24-5-5-5S7 3.24 7 6h1.9c0-1.71 1.39-3.1 3.1-3.1 1.71 0 3.1 1.39 3.1 3.1v2H6c-1.1 0-2 .9-2 2v10c0 1.1.9 2 2 2h12c1.1 0 2-.9 2-2V10c0-1.1-.9-2-2-2zm0 12H6V10h12v10z"></path></g>
<g id="lock-outline"><path d="M12 17c1.1 0 2-.9 2-2s-.9-2-2-2-2 .9-2 2 .9 2 2 2zm6-9h-1V6c0-2.76-2.24-5-5-5S7 3.24 7 6v2H6c-1.1 0-2 .9-2 2v10c0 1.1.9 2 2 2h12c1.1 0 2-.9 2-2V10c0-1.1-.9-2-2-2zM8.9 6c0-1.71 1.39-3.1 3.1-3.1s3.1 1.39 3.1 3.1v2H8.9V6zM18 20H6V10h12v10z"></path></g>
<g id="low-priority"><path d="M14 5h8v2h-8zm0 5.5h8v2h-8zm0 5.5h8v2h-8zM2 11.5C2 15.08 4.92 18 8.5 18H9v2l3-3-3-3v2h-.5C6.02 16 4 13.98 4 11.5S6.02 7 8.5 7H12V5H8.5C4.92 5 2 7.92 2 11.5z"></path></g>
<g id="loyalty"><path d="M21.41 11.58l-9-9C12.05 2.22 11.55 2 11 2H4c-1.1 0-2 .9-2 2v7c0 .55.22 1.05.59 1.42l9 9c.36.36.86.58 1.41.58.55 0 1.05-.22 1.41-.59l7-7c.37-.36.59-.86.59-1.41 0-.55-.23-1.06-.59-1.42zM5.5 7C4.67 7 4 6.33 4 5.5S4.67 4 5.5 4 7 4.67 7 5.5 6.33 7 5.5 7zm11.77 8.27L13 19.54l-4.27-4.27C8.28 14.81 8 14.19 8 13.5c0-1.38 1.12-2.5 2.5-2.5.69 0 1.32.28 1.77.74l.73.72.73-.73c.45-.45 1.08-.73 1.77-.73 1.38 0 2.5 1.12 2.5 2.5 0 .69-.28 1.32-.73 1.77z"></path></g>
<g id="mail"><path d="M20 4H4c-1.1 0-1.99.9-1.99 2L2 18c0 1.1.9 2 2 2h16c1.1 0 2-.9 2-2V6c0-1.1-.9-2-2-2zm0 4l-8 5-8-5V6l8 5 8-5v2z"></path></g>
<g id="markunread"><path d="M20 4H4c-1.1 0-1.99.9-1.99 2L2 18c0 1.1.9 2 2 2h16c1.1 0 2-.9 2-2V6c0-1.1-.9-2-2-2zm0 4l-8 5-8-5V6l8 5 8-5v2z"></path></g>
<g id="markunread-mailbox"><path d="M20 6H10v6H8V4h6V0H6v6H4c-1.1 0-2 .9-2 2v12c0 1.1.9 2 2 2h16c1.1 0 2-.9 2-2V8c0-1.1-.9-2-2-2z"></path></g>
<g id="menu"><path d="M3 18h18v-2H3v2zm0-5h18v-2H3v2zm0-7v2h18V6H3z"></path></g>
<g id="more-horiz"><path d="M6 10c-1.1 0-2 .9-2 2s.9 2 2 2 2-.9 2-2-.9-2-2-2zm12 0c-1.1 0-2 .9-2 2s.9 2 2 2 2-.9 2-2-.9-2-2-2zm-6 0c-1.1 0-2 .9-2 2s.9 2 2 2 2-.9 2-2-.9-2-2-2z"></path></g>
<g id="more-vert"><path d="M12 8c1.1 0 2-.9 2-2s-.9-2-2-2-2 .9-2 2 .9 2 2 2zm0 2c-1.1 0-2 .9-2 2s.9 2 2 2 2-.9 2-2-.9-2-2-2zm0 6c-1.1 0-2 .9-2 2s.9 2 2 2 2-.9 2-2-.9-2-2-2z"></path></g>
<g id="motorcycle"><path d="M19.44 9.03L15.41 5H11v2h3.59l2 2H5c-2.8 0-5 2.2-5 5s2.2 5 5 5c2.46 0 4.45-1.69 4.9-4h1.65l2.77-2.77c-.21.54-.32 1.14-.32 1.77 0 2.8 2.2 5 5 5s5-2.2 5-5c0-2.65-1.97-4.77-4.56-4.97zM7.82 15C7.4 16.15 6.28 17 5 17c-1.63 0-3-1.37-3-3s1.37-3 3-3c1.28 0 2.4.85 2.82 2H5v2h2.82zM19 17c-1.66 0-3-1.34-3-3s1.34-3 3-3 3 1.34 3 3-1.34 3-3 3z"></path></g>
<g id="move-to-inbox"><path d="M19 3H4.99c-1.11 0-1.98.9-1.98 2L3 19c0 1.1.88 2 1.99 2H19c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm0 12h-4c0 1.66-1.35 3-3 3s-3-1.34-3-3H4.99V5H19v10zm-3-5h-2V7h-4v3H8l4 4 4-4z"></path></g>
<g id="next-week"><path d="M20 7h-4V5c0-.55-.22-1.05-.59-1.41C15.05 3.22 14.55 3 14 3h-4c-1.1 0-2 .9-2 2v2H4c-1.1 0-2 .9-2 2v11c0 1.1.9 2 2 2h16c1.1 0 2-.9 2-2V9c0-1.1-.9-2-2-2zM10 5h4v2h-4V5zm1 13.5l-1-1 3-3-3-3 1-1 4 4-4 4z"></path></g>
<g id="note-add"><path d="M14 2H6c-1.1 0-1.99.9-1.99 2L4 20c0 1.1.89 2 1.99 2H18c1.1 0 2-.9 2-2V8l-6-6zm2 14h-3v3h-2v-3H8v-2h3v-3h2v3h3v2zm-3-7V3.5L18.5 9H13z"></path></g>
<g id="offline-pin"><path d="M12 2C6.5 2 2 6.5 2 12s4.5 10 10 10 10-4.5 10-10S17.5 2 12 2zm5 16H7v-2h10v2zm-6.7-4L7 10.7l1.4-1.4 1.9 1.9 5.3-5.3L17 7.3 10.3 14z"></path></g>
<g id="opacity"><path d="M17.66 8L12 2.35 6.34 8C4.78 9.56 4 11.64 4 13.64s.78 4.11 2.34 5.67 3.61 2.35 5.66 2.35 4.1-.79 5.66-2.35S20 15.64 20 13.64 19.22 9.56 17.66 8zM6 14c.01-2 .62-3.27 1.76-4.4L12 5.27l4.24 4.38C17.38 10.77 17.99 12 18 14H6z"></path></g>
<g id="open-in-browser"><path d="M19 4H5c-1.11 0-2 .9-2 2v12c0 1.1.89 2 2 2h4v-2H5V8h14v10h-4v2h4c1.1 0 2-.9 2-2V6c0-1.1-.89-2-2-2zm-7 6l-4 4h3v6h2v-6h3l-4-4z"></path></g>
<g id="open-in-new"><path d="M19 19H5V5h7V3H5c-1.11 0-2 .9-2 2v14c0 1.1.89 2 2 2h14c1.1 0 2-.9 2-2v-7h-2v7zM14 3v2h3.59l-9.83 9.83 1.41 1.41L19 6.41V10h2V3h-7z"></path></g>
<g id="open-with"><path d="M10 9h4V6h3l-5-5-5 5h3v3zm-1 1H6V7l-5 5 5 5v-3h3v-4zm14 2l-5-5v3h-3v4h3v3l5-5zm-9 3h-4v3H7l5 5 5-5h-3v-3z"></path></g>
<g id="pageview"><path d="M11.5 9C10.12 9 9 10.12 9 11.5s1.12 2.5 2.5 2.5 2.5-1.12 2.5-2.5S12.88 9 11.5 9zM20 4H4c-1.1 0-2 .9-2 2v12c0 1.1.9 2 2 2h16c1.1 0 2-.9 2-2V6c0-1.1-.9-2-2-2zm-3.21 14.21l-2.91-2.91c-.69.44-1.51.7-2.39.7C9.01 16 7 13.99 7 11.5S9.01 7 11.5 7 16 9.01 16 11.5c0 .88-.26 1.69-.7 2.39l2.91 2.9-1.42 1.42z"></path></g>
<g id="pan-tool"><path d="M23 5.5V20c0 2.2-1.8 4-4 4h-7.3c-1.08 0-2.1-.43-2.85-1.19L1 14.83s1.26-1.23 1.3-1.25c.22-.19.49-.29.79-.29.22 0 .42.06.6.16.04.01 4.31 2.46 4.31 2.46V4c0-.83.67-1.5 1.5-1.5S11 3.17 11 4v7h1V1.5c0-.83.67-1.5 1.5-1.5S15 .67 15 1.5V11h1V2.5c0-.83.67-1.5 1.5-1.5s1.5.67 1.5 1.5V11h1V5.5c0-.83.67-1.5 1.5-1.5s1.5.67 1.5 1.5z"></path></g>
<g id="payment"><path d="M20 4H4c-1.11 0-1.99.89-1.99 2L2 18c0 1.11.89 2 2 2h16c1.11 0 2-.89 2-2V6c0-1.11-.89-2-2-2zm0 14H4v-6h16v6zm0-10H4V6h16v2z"></path></g>
<g id="perm-camera-mic"><path d="M20 5h-3.17L15 3H9L7.17 5H4c-1.1 0-2 .9-2 2v12c0 1.1.9 2 2 2h7v-2.09c-2.83-.48-5-2.94-5-5.91h2c0 2.21 1.79 4 4 4s4-1.79 4-4h2c0 2.97-2.17 5.43-5 5.91V21h7c1.1 0 2-.9 2-2V7c0-1.1-.9-2-2-2zm-6 8c0 1.1-.9 2-2 2s-2-.9-2-2V9c0-1.1.9-2 2-2s2 .9 2 2v4z"></path></g>
<g id="perm-contact-calendar"><path d="M19 3h-1V1h-2v2H8V1H6v2H5c-1.11 0-2 .9-2 2v14c0 1.1.89 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm-7 3c1.66 0 3 1.34 3 3s-1.34 3-3 3-3-1.34-3-3 1.34-3 3-3zm6 12H6v-1c0-2 4-3.1 6-3.1s6 1.1 6 3.1v1z"></path></g>
<g id="perm-data-setting"><path d="M18.99 11.5c.34 0 .67.03 1 .07L20 0 0 20h11.56c-.04-.33-.07-.66-.07-1 0-4.14 3.36-7.5 7.5-7.5zm3.71 7.99c.02-.16.04-.32.04-.49 0-.17-.01-.33-.04-.49l1.06-.83c.09-.08.12-.21.06-.32l-1-1.73c-.06-.11-.19-.15-.31-.11l-1.24.5c-.26-.2-.54-.37-.85-.49l-.19-1.32c-.01-.12-.12-.21-.24-.21h-2c-.12 0-.23.09-.25.21l-.19 1.32c-.3.13-.59.29-.85.49l-1.24-.5c-.11-.04-.24 0-.31.11l-1 1.73c-.06.11-.04.24.06.32l1.06.83c-.02.16-.03.32-.03.49 0 .17.01.33.03.49l-1.06.83c-.09.08-.12.21-.06.32l1 1.73c.06.11.19.15.31.11l1.24-.5c.26.2.54.37.85.49l.19 1.32c.02.12.12.21.25.21h2c.12 0 .23-.09.25-.21l.19-1.32c.3-.13.59-.29.84-.49l1.25.5c.11.04.24 0 .31-.11l1-1.73c.06-.11.03-.24-.06-.32l-1.07-.83zm-3.71 1.01c-.83 0-1.5-.67-1.5-1.5s.67-1.5 1.5-1.5 1.5.67 1.5 1.5-.67 1.5-1.5 1.5z"></path></g>
<g id="perm-device-information"><path d="M13 7h-2v2h2V7zm0 4h-2v6h2v-6zm4-9.99L7 1c-1.1 0-2 .9-2 2v18c0 1.1.9 2 2 2h10c1.1 0 2-.9 2-2V3c0-1.1-.9-1.99-2-1.99zM17 19H7V5h10v14z"></path></g>
<g id="perm-identity"><path d="M12 5.9c1.16 0 2.1.94 2.1 2.1s-.94 2.1-2.1 2.1S9.9 9.16 9.9 8s.94-2.1 2.1-2.1m0 9c2.97 0 6.1 1.46 6.1 2.1v1.1H5.9V17c0-.64 3.13-2.1 6.1-2.1M12 4C9.79 4 8 5.79 8 8s1.79 4 4 4 4-1.79 4-4-1.79-4-4-4zm0 9c-2.67 0-8 1.34-8 4v3h16v-3c0-2.66-5.33-4-8-4z"></path></g>
<g id="perm-media"><path d="M2 6H0v5h.01L0 20c0 1.1.9 2 2 2h18v-2H2V6zm20-2h-8l-2-2H6c-1.1 0-1.99.9-1.99 2L4 16c0 1.1.9 2 2 2h16c1.1 0 2-.9 2-2V6c0-1.1-.9-2-2-2zM7 15l4.5-6 3.5 4.51 2.5-3.01L21 15H7z"></path></g>
<g id="perm-phone-msg"><path d="M20 15.5c-1.25 0-2.45-.2-3.57-.57-.35-.11-.74-.03-1.02.24l-2.2 2.2c-2.83-1.44-5.15-3.75-6.59-6.58l2.2-2.21c.28-.27.36-.66.25-1.01C8.7 6.45 8.5 5.25 8.5 4c0-.55-.45-1-1-1H4c-.55 0-1 .45-1 1 0 9.39 7.61 17 17 17 .55 0 1-.45 1-1v-3.5c0-.55-.45-1-1-1zM12 3v10l3-3h6V3h-9z"></path></g>
<g id="perm-scan-wifi"><path d="M12 3C6.95 3 3.15 4.85 0 7.23L12 22 24 7.25C20.85 4.87 17.05 3 12 3zm1 13h-2v-6h2v6zm-2-8V6h2v2h-2z"></path></g>
<g id="pets"><circle cx="4.5" cy="9.5" r="2.5"></circle><circle cx="9" cy="5.5" r="2.5"></circle><circle cx="15" cy="5.5" r="2.5"></circle><circle cx="19.5" cy="9.5" r="2.5"></circle><path d="M17.34 14.86c-.87-1.02-1.6-1.89-2.48-2.91-.46-.54-1.05-1.08-1.75-1.32-.11-.04-.22-.07-.33-.09-.25-.04-.52-.04-.78-.04s-.53 0-.79.05c-.11.02-.22.05-.33.09-.7.24-1.28.78-1.75 1.32-.87 1.02-1.6 1.89-2.48 2.91-1.31 1.31-2.92 2.76-2.62 4.79.29 1.02 1.02 2.03 2.33 2.32.73.15 3.06-.44 5.54-.44h.18c2.48 0 4.81.58 5.54.44 1.31-.29 2.04-1.31 2.33-2.32.31-2.04-1.3-3.49-2.61-4.8z"></path></g>
<g id="picture-in-picture"><path d="M19 7h-8v6h8V7zm2-4H3c-1.1 0-2 .9-2 2v14c0 1.1.9 1.98 2 1.98h18c1.1 0 2-.88 2-1.98V5c0-1.1-.9-2-2-2zm0 16.01H3V4.98h18v14.03z"></path></g>
<g id="picture-in-picture-alt"><path d="M19 11h-8v6h8v-6zm4 8V4.98C23 3.88 22.1 3 21 3H3c-1.1 0-2 .88-2 1.98V19c0 1.1.9 2 2 2h18c1.1 0 2-.9 2-2zm-2 .02H3V4.97h18v14.05z"></path></g>
<g id="play-for-work"><path d="M11 5v5.59H7.5l4.5 4.5 4.5-4.5H13V5h-2zm-5 9c0 3.31 2.69 6 6 6s6-2.69 6-6h-2c0 2.21-1.79 4-4 4s-4-1.79-4-4H6z"></path></g>
<g id="polymer"><path d="M19 4h-4L7.11 16.63 4.5 12 9 4H5L.5 12 5 20h4l7.89-12.63L19.5 12 15 20h4l4.5-8z"></path></g>
<g id="power-settings-new"><path d="M13 3h-2v10h2V3zm4.83 2.17l-1.42 1.42C17.99 7.86 19 9.81 19 12c0 3.87-3.13 7-7 7s-7-3.13-7-7c0-2.19 1.01-4.14 2.58-5.42L6.17 5.17C4.23 6.82 3 9.26 3 12c0 4.97 4.03 9 9 9s9-4.03 9-9c0-2.74-1.23-5.18-3.17-6.83z"></path></g>
<g id="pregnant-woman"><path d="M9 4c0-1.11.89-2 2-2s2 .89 2 2-.89 2-2 2-2-.89-2-2zm7 9c-.01-1.34-.83-2.51-2-3 0-1.66-1.34-3-3-3s-3 1.34-3 3v7h2v5h3v-5h3v-4z"></path></g>
<g id="print"><path d="M19 8H5c-1.66 0-3 1.34-3 3v6h4v4h12v-4h4v-6c0-1.66-1.34-3-3-3zm-3 11H8v-5h8v5zm3-7c-.55 0-1-.45-1-1s.45-1 1-1 1 .45 1 1-.45 1-1 1zm-1-9H6v4h12V3z"></path></g>
<g id="query-builder"><path d="M11.99 2C6.47 2 2 6.48 2 12s4.47 10 9.99 10C17.52 22 22 17.52 22 12S17.52 2 11.99 2zM12 20c-4.42 0-8-3.58-8-8s3.58-8 8-8 8 3.58 8 8-3.58 8-8 8zm.5-13H11v6l5.25 3.15.75-1.23-4.5-2.67z"></path></g>
<g id="question-answer"><path d="M21 6h-2v9H6v2c0 .55.45 1 1 1h11l4 4V7c0-.55-.45-1-1-1zm-4 6V3c0-.55-.45-1-1-1H3c-.55 0-1 .45-1 1v14l4-4h10c.55 0 1-.45 1-1z"></path></g>
<g id="radio-button-checked"><path d="M12 7c-2.76 0-5 2.24-5 5s2.24 5 5 5 5-2.24 5-5-2.24-5-5-5zm0-5C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm0 18c-4.42 0-8-3.58-8-8s3.58-8 8-8 8 3.58 8 8-3.58 8-8 8z"></path></g>
<g id="radio-button-unchecked"><path d="M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm0 18c-4.42 0-8-3.58-8-8s3.58-8 8-8 8 3.58 8 8-3.58 8-8 8z"></path></g>
<g id="receipt"><path d="M18 17H6v-2h12v2zm0-4H6v-2h12v2zm0-4H6V7h12v2zM3 22l1.5-1.5L6 22l1.5-1.5L9 22l1.5-1.5L12 22l1.5-1.5L15 22l1.5-1.5L18 22l1.5-1.5L21 22V2l-1.5 1.5L18 2l-1.5 1.5L15 2l-1.5 1.5L12 2l-1.5 1.5L9 2 7.5 3.5 6 2 4.5 3.5 3 2v20z"></path></g>
<g id="record-voice-over"><circle cx="9" cy="9" r="4"></circle><path d="M9 15c-2.67 0-8 1.34-8 4v2h16v-2c0-2.66-5.33-4-8-4zm7.76-9.64l-1.68 1.69c.84 1.18.84 2.71 0 3.89l1.68 1.69c2.02-2.02 2.02-5.07 0-7.27zM20.07 2l-1.63 1.63c2.77 3.02 2.77 7.56 0 10.74L20.07 16c3.9-3.89 3.91-9.95 0-14z"></path></g>
<g id="redeem"><path d="M20 6h-2.18c.11-.31.18-.65.18-1 0-1.66-1.34-3-3-3-1.05 0-1.96.54-2.5 1.35l-.5.67-.5-.68C10.96 2.54 10.05 2 9 2 7.34 2 6 3.34 6 5c0 .35.07.69.18 1H4c-1.11 0-1.99.89-1.99 2L2 19c0 1.11.89 2 2 2h16c1.11 0 2-.89 2-2V8c0-1.11-.89-2-2-2zm-5-2c.55 0 1 .45 1 1s-.45 1-1 1-1-.45-1-1 .45-1 1-1zM9 4c.55 0 1 .45 1 1s-.45 1-1 1-1-.45-1-1 .45-1 1-1zm11 15H4v-2h16v2zm0-5H4V8h5.08L7 10.83 8.62 12 11 8.76l1-1.36 1 1.36L15.38 12 17 10.83 14.92 8H20v6z"></path></g>
<g id="redo"><path d="M18.4 10.6C16.55 8.99 14.15 8 11.5 8c-4.65 0-8.58 3.03-9.96 7.22L3.9 16c1.05-3.19 4.05-5.5 7.6-5.5 1.95 0 3.73.72 5.12 1.88L13 16h9V7l-3.6 3.6z"></path></g>
<g id="refresh"><path d="M17.65 6.35C16.2 4.9 14.21 4 12 4c-4.42 0-7.99 3.58-7.99 8s3.57 8 7.99 8c3.73 0 6.84-2.55 7.73-6h-2.08c-.82 2.33-3.04 4-5.65 4-3.31 0-6-2.69-6-6s2.69-6 6-6c1.66 0 3.14.69 4.22 1.78L13 11h7V4l-2.35 2.35z"></path></g>
<g id="remove"><path d="M19 13H5v-2h14v2z"></path></g>
<g id="remove-circle"><path d="M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm5 11H7v-2h10v2z"></path></g>
<g id="remove-circle-outline"><path d="M7 11v2h10v-2H7zm5-9C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm0 18c-4.41 0-8-3.59-8-8s3.59-8 8-8 8 3.59 8 8-3.59 8-8 8z"></path></g>
<g id="remove-shopping-cart"><path d="M22.73 22.73L2.77 2.77 2 2l-.73-.73L0 2.54l4.39 4.39 2.21 4.66-1.35 2.45c-.16.28-.25.61-.25.96 0 1.1.9 2 2 2h7.46l1.38 1.38c-.5.36-.83.95-.83 1.62 0 1.1.89 2 1.99 2 .67 0 1.26-.33 1.62-.84L21.46 24l1.27-1.27zM7.42 15c-.14 0-.25-.11-.25-.25l.03-.12.9-1.63h2.36l2 2H7.42zm8.13-2c.75 0 1.41-.41 1.75-1.03l3.58-6.49c.08-.14.12-.31.12-.48 0-.55-.45-1-1-1H6.54l9.01 9zM7 18c-1.1 0-1.99.9-1.99 2S5.9 22 7 22s2-.9 2-2-.9-2-2-2z"></path></g>
<g id="reorder"><path d="M3 15h18v-2H3v2zm0 4h18v-2H3v2zm0-8h18V9H3v2zm0-6v2h18V5H3z"></path></g>
<g id="reply"><path d="M10 9V5l-7 7 7 7v-4.1c5 0 8.5 1.6 11 5.1-1-5-4-10-11-11z"></path></g>
<g id="reply-all"><path d="M7 8V5l-7 7 7 7v-3l-4-4 4-4zm6 1V5l-7 7 7 7v-4.1c5 0 8.5 1.6 11 5.1-1-5-4-10-11-11z"></path></g>
<g id="report"><path d="M15.73 3H8.27L3 8.27v7.46L8.27 21h7.46L21 15.73V8.27L15.73 3zM12 17.3c-.72 0-1.3-.58-1.3-1.3 0-.72.58-1.3 1.3-1.3.72 0 1.3.58 1.3 1.3 0 .72-.58 1.3-1.3 1.3zm1-4.3h-2V7h2v6z"></path></g>
<g id="report-problem"><path d="M1 21h22L12 2 1 21zm12-3h-2v-2h2v2zm0-4h-2v-4h2v4z"></path></g>
<g id="restore"><path d="M13 3c-4.97 0-9 4.03-9 9H1l3.89 3.89.07.14L9 12H6c0-3.87 3.13-7 7-7s7 3.13 7 7-3.13 7-7 7c-1.93 0-3.68-.79-4.94-2.06l-1.42 1.42C8.27 19.99 10.51 21 13 21c4.97 0 9-4.03 9-9s-4.03-9-9-9zm-1 5v5l4.28 2.54.72-1.21-3.5-2.08V8H12z"></path></g>
<g id="restore-page"><path d="M14 2H6c-1.1 0-1.99.9-1.99 2L4 20c0 1.1.89 2 1.99 2H18c1.1 0 2-.9 2-2V8l-6-6zm-2 16c-2.05 0-3.81-1.24-4.58-3h1.71c.63.9 1.68 1.5 2.87 1.5 1.93 0 3.5-1.57 3.5-3.5S13.93 9.5 12 9.5c-1.35 0-2.52.78-3.1 1.9l1.6 1.6h-4V9l1.3 1.3C8.69 8.92 10.23 8 12 8c2.76 0 5 2.24 5 5s-2.24 5-5 5z"></path></g>
<g id="room"><path d="M12 2C8.13 2 5 5.13 5 9c0 5.25 7 13 7 13s7-7.75 7-13c0-3.87-3.13-7-7-7zm0 9.5c-1.38 0-2.5-1.12-2.5-2.5s1.12-2.5 2.5-2.5 2.5 1.12 2.5 2.5-1.12 2.5-2.5 2.5z"></path></g>
<g id="rounded-corner"><path d="M19 19h2v2h-2v-2zm0-2h2v-2h-2v2zM3 13h2v-2H3v2zm0 4h2v-2H3v2zm0-8h2V7H3v2zm0-4h2V3H3v2zm4 0h2V3H7v2zm8 16h2v-2h-2v2zm-4 0h2v-2h-2v2zm4 0h2v-2h-2v2zm-8 0h2v-2H7v2zm-4 0h2v-2H3v2zM21 8c0-2.76-2.24-5-5-5h-5v2h5c1.65 0 3 1.35 3 3v5h2V8z"></path></g>
<g id="rowing"><path d="M8.5 14.5L4 19l1.5 1.5L9 17h2l-2.5-2.5zM15 1c-1.1 0-2 .9-2 2s.9 2 2 2 2-.9 2-2-.9-2-2-2zm6 20.01L18 24l-2.99-3.01V19.5l-7.1-7.09c-.31.05-.61.07-.91.07v-2.16c1.66.03 3.61-.87 4.67-2.04l1.4-1.55c.19-.21.43-.38.69-.5.29-.14.62-.23.96-.23h.03C15.99 6.01 17 7.02 17 8.26v5.75c0 .84-.35 1.61-.92 2.16l-3.58-3.58v-2.27c-.63.52-1.43 1.02-2.29 1.39L16.5 18H18l3 3.01z"></path></g>
<g id="save"><path d="M17 3H5c-1.11 0-2 .9-2 2v14c0 1.1.89 2 2 2h14c1.1 0 2-.9 2-2V7l-4-4zm-5 16c-1.66 0-3-1.34-3-3s1.34-3 3-3 3 1.34 3 3-1.34 3-3 3zm3-10H5V5h10v4z"></path></g>
<g id="schedule"><path d="M11.99 2C6.47 2 2 6.48 2 12s4.47 10 9.99 10C17.52 22 22 17.52 22 12S17.52 2 11.99 2zM12 20c-4.42 0-8-3.58-8-8s3.58-8 8-8 8 3.58 8 8-3.58 8-8 8zm.5-13H11v6l5.25 3.15.75-1.23-4.5-2.67z"></path></g>
<g id="search"><path d="M15.5 14h-.79l-.28-.27C15.41 12.59 16 11.11 16 9.5 16 5.91 13.09 3 9.5 3S3 5.91 3 9.5 5.91 16 9.5 16c1.61 0 3.09-.59 4.23-1.57l.27.28v.79l5 4.99L20.49 19l-4.99-5zm-6 0C7.01 14 5 11.99 5 9.5S7.01 5 9.5 5 14 7.01 14 9.5 11.99 14 9.5 14z"></path></g>
<g id="select-all"><path d="M3 5h2V3c-1.1 0-2 .9-2 2zm0 8h2v-2H3v2zm4 8h2v-2H7v2zM3 9h2V7H3v2zm10-6h-2v2h2V3zm6 0v2h2c0-1.1-.9-2-2-2zM5 21v-2H3c0 1.1.9 2 2 2zm-2-4h2v-2H3v2zM9 3H7v2h2V3zm2 18h2v-2h-2v2zm8-8h2v-2h-2v2zm0 8c1.1 0 2-.9 2-2h-2v2zm0-12h2V7h-2v2zm0 8h2v-2h-2v2zm-4 4h2v-2h-2v2zm0-16h2V3h-2v2zM7 17h10V7H7v10zm2-8h6v6H9V9z"></path></g>
<g id="send"><path d="M2.01 21L23 12 2.01 3 2 10l15 2-15 2z"></path></g>
<g id="settings"><path d="M19.43 12.98c.04-.32.07-.64.07-.98s-.03-.66-.07-.98l2.11-1.65c.19-.15.24-.42.12-.64l-2-3.46c-.12-.22-.39-.3-.61-.22l-2.49 1c-.52-.4-1.08-.73-1.69-.98l-.38-2.65C14.46 2.18 14.25 2 14 2h-4c-.25 0-.46.18-.49.42l-.38 2.65c-.61.25-1.17.59-1.69.98l-2.49-1c-.23-.09-.49 0-.61.22l-2 3.46c-.13.22-.07.49.12.64l2.11 1.65c-.04.32-.07.65-.07.98s.03.66.07.98l-2.11 1.65c-.19.15-.24.42-.12.64l2 3.46c.12.22.39.3.61.22l2.49-1c.52.4 1.08.73 1.69.98l.38 2.65c.03.24.24.42.49.42h4c.25 0 .46-.18.49-.42l.38-2.65c.61-.25 1.17-.59 1.69-.98l2.49 1c.23.09.49 0 .61-.22l2-3.46c.12-.22.07-.49-.12-.64l-2.11-1.65zM12 15.5c-1.93 0-3.5-1.57-3.5-3.5s1.57-3.5 3.5-3.5 3.5 1.57 3.5 3.5-1.57 3.5-3.5 3.5z"></path></g>
<g id="settings-applications"><path d="M12 10c-1.1 0-2 .9-2 2s.9 2 2 2 2-.9 2-2-.9-2-2-2zm7-7H5c-1.11 0-2 .9-2 2v14c0 1.1.89 2 2 2h14c1.11 0 2-.9 2-2V5c0-1.1-.89-2-2-2zm-1.75 9c0 .23-.02.46-.05.68l1.48 1.16c.13.11.17.3.08.45l-1.4 2.42c-.09.15-.27.21-.43.15l-1.74-.7c-.36.28-.76.51-1.18.69l-.26 1.85c-.03.17-.18.3-.35.3h-2.8c-.17 0-.32-.13-.35-.29l-.26-1.85c-.43-.18-.82-.41-1.18-.69l-1.74.7c-.16.06-.34 0-.43-.15l-1.4-2.42c-.09-.15-.05-.34.08-.45l1.48-1.16c-.03-.23-.05-.46-.05-.69 0-.23.02-.46.05-.68l-1.48-1.16c-.13-.11-.17-.3-.08-.45l1.4-2.42c.09-.15.27-.21.43-.15l1.74.7c.36-.28.76-.51 1.18-.69l.26-1.85c.03-.17.18-.3.35-.3h2.8c.17 0 .32.13.35.29l.26 1.85c.43.18.82.41 1.18.69l1.74-.7c.16-.06.34 0 .43.15l1.4 2.42c.09.15.05.34-.08.45l-1.48 1.16c.03.23.05.46.05.69z"></path></g>
<g id="settings-backup-restore"><path d="M14 12c0-1.1-.9-2-2-2s-2 .9-2 2 .9 2 2 2 2-.9 2-2zm-2-9c-4.97 0-9 4.03-9 9H0l4 4 4-4H5c0-3.87 3.13-7 7-7s7 3.13 7 7-3.13 7-7 7c-1.51 0-2.91-.49-4.06-1.3l-1.42 1.44C8.04 20.3 9.94 21 12 21c4.97 0 9-4.03 9-9s-4.03-9-9-9z"></path></g>
<g id="settings-bluetooth"><path d="M11 24h2v-2h-2v2zm-4 0h2v-2H7v2zm8 0h2v-2h-2v2zm2.71-18.29L12 0h-1v7.59L6.41 3 5 4.41 10.59 10 5 15.59 6.41 17 11 12.41V20h1l5.71-5.71-4.3-4.29 4.3-4.29zM13 3.83l1.88 1.88L13 7.59V3.83zm1.88 10.46L13 16.17v-3.76l1.88 1.88z"></path></g>
<g id="settings-brightness"><path d="M21 3H3c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h18c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm0 16.01H3V4.99h18v14.02zM8 16h2.5l1.5 1.5 1.5-1.5H16v-2.5l1.5-1.5-1.5-1.5V8h-2.5L12 6.5 10.5 8H8v2.5L6.5 12 8 13.5V16zm4-7c1.66 0 3 1.34 3 3s-1.34 3-3 3V9z"></path></g>
<g id="settings-cell"><path d="M7 24h2v-2H7v2zm4 0h2v-2h-2v2zm4 0h2v-2h-2v2zM16 .01L8 0C6.9 0 6 .9 6 2v16c0 1.1.9 2 2 2h8c1.1 0 2-.9 2-2V2c0-1.1-.9-1.99-2-1.99zM16 16H8V4h8v12z"></path></g>
<g id="settings-ethernet"><path d="M7.77 6.76L6.23 5.48.82 12l5.41 6.52 1.54-1.28L3.42 12l4.35-5.24zM7 13h2v-2H7v2zm10-2h-2v2h2v-2zm-6 2h2v-2h-2v2zm6.77-7.52l-1.54 1.28L20.58 12l-4.35 5.24 1.54 1.28L23.18 12l-5.41-6.52z"></path></g>
<g id="settings-input-antenna"><path d="M12 5c-3.87 0-7 3.13-7 7h2c0-2.76 2.24-5 5-5s5 2.24 5 5h2c0-3.87-3.13-7-7-7zm1 9.29c.88-.39 1.5-1.26 1.5-2.29 0-1.38-1.12-2.5-2.5-2.5S9.5 10.62 9.5 12c0 1.02.62 1.9 1.5 2.29v3.3L7.59 21 9 22.41l3-3 3 3L16.41 21 13 17.59v-3.3zM12 1C5.93 1 1 5.93 1 12h2c0-4.97 4.03-9 9-9s9 4.03 9 9h2c0-6.07-4.93-11-11-11z"></path></g>
<g id="settings-input-component"><path d="M5 2c0-.55-.45-1-1-1s-1 .45-1 1v4H1v6h6V6H5V2zm4 14c0 1.3.84 2.4 2 2.82V23h2v-4.18c1.16-.41 2-1.51 2-2.82v-2H9v2zm-8 0c0 1.3.84 2.4 2 2.82V23h2v-4.18C6.16 18.4 7 17.3 7 16v-2H1v2zM21 6V2c0-.55-.45-1-1-1s-1 .45-1 1v4h-2v6h6V6h-2zm-8-4c0-.55-.45-1-1-1s-1 .45-1 1v4H9v6h6V6h-2V2zm4 14c0 1.3.84 2.4 2 2.82V23h2v-4.18c1.16-.41 2-1.51 2-2.82v-2h-6v2z"></path></g>
<g id="settings-input-composite"><path d="M5 2c0-.55-.45-1-1-1s-1 .45-1 1v4H1v6h6V6H5V2zm4 14c0 1.3.84 2.4 2 2.82V23h2v-4.18c1.16-.41 2-1.51 2-2.82v-2H9v2zm-8 0c0 1.3.84 2.4 2 2.82V23h2v-4.18C6.16 18.4 7 17.3 7 16v-2H1v2zM21 6V2c0-.55-.45-1-1-1s-1 .45-1 1v4h-2v6h6V6h-2zm-8-4c0-.55-.45-1-1-1s-1 .45-1 1v4H9v6h6V6h-2V2zm4 14c0 1.3.84 2.4 2 2.82V23h2v-4.18c1.16-.41 2-1.51 2-2.82v-2h-6v2z"></path></g>
<g id="settings-input-hdmi"><path d="M18 7V4c0-1.1-.9-2-2-2H8c-1.1 0-2 .9-2 2v3H5v6l3 6v3h8v-3l3-6V7h-1zM8 4h8v3h-2V5h-1v2h-2V5h-1v2H8V4z"></path></g>
<g id="settings-input-svideo"><path d="M8 11.5c0-.83-.67-1.5-1.5-1.5S5 10.67 5 11.5 5.67 13 6.5 13 8 12.33 8 11.5zm7-5c0-.83-.67-1.5-1.5-1.5h-3C9.67 5 9 5.67 9 6.5S9.67 8 10.5 8h3c.83 0 1.5-.67 1.5-1.5zM8.5 15c-.83 0-1.5.67-1.5 1.5S7.67 18 8.5 18s1.5-.67 1.5-1.5S9.33 15 8.5 15zM12 1C5.93 1 1 5.93 1 12s4.93 11 11 11 11-4.93 11-11S18.07 1 12 1zm0 20c-4.96 0-9-4.04-9-9s4.04-9 9-9 9 4.04 9 9-4.04 9-9 9zm5.5-11c-.83 0-1.5.67-1.5 1.5s.67 1.5 1.5 1.5 1.5-.67 1.5-1.5-.67-1.5-1.5-1.5zm-2 5c-.83 0-1.5.67-1.5 1.5s.67 1.5 1.5 1.5 1.5-.67 1.5-1.5-.67-1.5-1.5-1.5z"></path></g>
<g id="settings-overscan"><path d="M12.01 5.5L10 8h4l-1.99-2.5zM18 10v4l2.5-1.99L18 10zM6 10l-2.5 2.01L6 14v-4zm8 6h-4l2.01 2.5L14 16zm7-13H3c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h18c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm0 16.01H3V4.99h18v14.02z"></path></g>
<g id="settings-phone"><path d="M13 9h-2v2h2V9zm4 0h-2v2h2V9zm3 6.5c-1.25 0-2.45-.2-3.57-.57-.35-.11-.74-.03-1.02.24l-2.2 2.2c-2.83-1.44-5.15-3.75-6.59-6.58l2.2-2.21c.28-.27.36-.66.25-1.01C8.7 6.45 8.5 5.25 8.5 4c0-.55-.45-1-1-1H4c-.55 0-1 .45-1 1 0 9.39 7.61 17 17 17 .55 0 1-.45 1-1v-3.5c0-.55-.45-1-1-1zM19 9v2h2V9h-2z"></path></g>
<g id="settings-power"><path d="M7 24h2v-2H7v2zm4 0h2v-2h-2v2zm2-22h-2v10h2V2zm3.56 2.44l-1.45 1.45C16.84 6.94 18 8.83 18 11c0 3.31-2.69 6-6 6s-6-2.69-6-6c0-2.17 1.16-4.06 2.88-5.12L7.44 4.44C5.36 5.88 4 8.28 4 11c0 4.42 3.58 8 8 8s8-3.58 8-8c0-2.72-1.36-5.12-3.44-6.56zM15 24h2v-2h-2v2z"></path></g>
<g id="settings-remote"><path d="M15 9H9c-.55 0-1 .45-1 1v12c0 .55.45 1 1 1h6c.55 0 1-.45 1-1V10c0-.55-.45-1-1-1zm-3 6c-1.1 0-2-.9-2-2s.9-2 2-2 2 .9 2 2-.9 2-2 2zM7.05 6.05l1.41 1.41C9.37 6.56 10.62 6 12 6s2.63.56 3.54 1.46l1.41-1.41C15.68 4.78 13.93 4 12 4s-3.68.78-4.95 2.05zM12 0C8.96 0 6.21 1.23 4.22 3.22l1.41 1.41C7.26 3.01 9.51 2 12 2s4.74 1.01 6.36 2.64l1.41-1.41C17.79 1.23 15.04 0 12 0z"></path></g>
<g id="settings-voice"><path d="M7 24h2v-2H7v2zm5-11c1.66 0 2.99-1.34 2.99-3L15 4c0-1.66-1.34-3-3-3S9 2.34 9 4v6c0 1.66 1.34 3 3 3zm-1 11h2v-2h-2v2zm4 0h2v-2h-2v2zm4-14h-1.7c0 3-2.54 5.1-5.3 5.1S6.7 13 6.7 10H5c0 3.41 2.72 6.23 6 6.72V20h2v-3.28c3.28-.49 6-3.31 6-6.72z"></path></g>
<g id="shop"><path d="M16 6V4c0-1.11-.89-2-2-2h-4c-1.11 0-2 .89-2 2v2H2v13c0 1.11.89 2 2 2h16c1.11 0 2-.89 2-2V6h-6zm-6-2h4v2h-4V4zM9 18V9l7.5 4L9 18z"></path></g>
<g id="shop-two"><path d="M3 9H1v11c0 1.11.89 2 2 2h14c1.11 0 2-.89 2-2H3V9zm15-4V3c0-1.11-.89-2-2-2h-4c-1.11 0-2 .89-2 2v2H5v11c0 1.11.89 2 2 2h14c1.11 0 2-.89 2-2V5h-5zm-6-2h4v2h-4V3zm0 12V8l5.5 3-5.5 4z"></path></g>
<g id="shopping-basket"><path d="M17.21 9l-4.38-6.56c-.19-.28-.51-.42-.83-.42-.32 0-.64.14-.83.43L6.79 9H2c-.55 0-1 .45-1 1 0 .09.01.18.04.27l2.54 9.27c.23.84 1 1.46 1.92 1.46h13c.92 0 1.69-.62 1.93-1.46l2.54-9.27L23 10c0-.55-.45-1-1-1h-4.79zM9 9l3-4.4L15 9H9zm3 8c-1.1 0-2-.9-2-2s.9-2 2-2 2 .9 2 2-.9 2-2 2z"></path></g>
<g id="shopping-cart"><path d="M7 18c-1.1 0-1.99.9-1.99 2S5.9 22 7 22s2-.9 2-2-.9-2-2-2zM1 2v2h2l3.6 7.59-1.35 2.45c-.16.28-.25.61-.25.96 0 1.1.9 2 2 2h12v-2H7.42c-.14 0-.25-.11-.25-.25l.03-.12.9-1.63h7.45c.75 0 1.41-.41 1.75-1.03l3.58-6.49c.08-.14.12-.31.12-.48 0-.55-.45-1-1-1H5.21l-.94-2H1zm16 16c-1.1 0-1.99.9-1.99 2s.89 2 1.99 2 2-.9 2-2-.9-2-2-2z"></path></g>
<g id="sort"><path d="M3 18h6v-2H3v2zM3 6v2h18V6H3zm0 7h12v-2H3v2z"></path></g>
<g id="speaker-notes"><path d="M20 2H4c-1.1 0-1.99.9-1.99 2L2 22l4-4h14c1.1 0 2-.9 2-2V4c0-1.1-.9-2-2-2zM8 14H6v-2h2v2zm0-3H6V9h2v2zm0-3H6V6h2v2zm7 6h-5v-2h5v2zm3-3h-8V9h8v2zm0-3h-8V6h8v2z"></path></g>
<g id="speaker-notes-off"><path d="M10.54 11l-.54-.54L7.54 8 6 6.46 2.38 2.84 1.27 1.73 0 3l2.01 2.01L2 22l4-4h9l5.73 5.73L22 22.46 17.54 18l-7-7zM8 14H6v-2h2v2zm-2-3V9l2 2H6zm14-9H4.08L10 7.92V6h8v2h-7.92l1 1H18v2h-4.92l6.99 6.99C21.14 17.95 22 17.08 22 16V4c0-1.1-.9-2-2-2z"></path></g>
<g id="spellcheck"><path d="M12.45 16h2.09L9.43 3H7.57L2.46 16h2.09l1.12-3h5.64l1.14 3zm-6.02-5L8.5 5.48 10.57 11H6.43zm15.16.59l-8.09 8.09L9.83 16l-1.41 1.41 5.09 5.09L23 13l-1.41-1.41z"></path></g>
<g id="star"><path d="M12 17.27L18.18 21l-1.64-7.03L22 9.24l-7.19-.61L12 2 9.19 8.63 2 9.24l5.46 4.73L5.82 21z"></path></g>
<g id="star-border"><path d="M22 9.24l-7.19-.62L12 2 9.19 8.63 2 9.24l5.46 4.73L5.82 21 12 17.27 18.18 21l-1.63-7.03L22 9.24zM12 15.4l-3.76 2.27 1-4.28-3.32-2.88 4.38-.38L12 6.1l1.71 4.04 4.38.38-3.32 2.88 1 4.28L12 15.4z"></path></g>
<g id="star-half"><path d="M22 9.24l-7.19-.62L12 2 9.19 8.63 2 9.24l5.46 4.73L5.82 21 12 17.27 18.18 21l-1.63-7.03L22 9.24zM12 15.4V6.1l1.71 4.04 4.38.38-3.32 2.88 1 4.28L12 15.4z"></path></g>
<g id="stars"><path d="M11.99 2C6.47 2 2 6.48 2 12s4.47 10 9.99 10C17.52 22 22 17.52 22 12S17.52 2 11.99 2zm4.24 16L12 15.45 7.77 18l1.12-4.81-3.73-3.23 4.92-.42L12 5l1.92 4.53 4.92.42-3.73 3.23L16.23 18z"></path></g>
<g id="store"><path d="M20 4H4v2h16V4zm1 10v-2l-1-5H4l-1 5v2h1v6h10v-6h4v6h2v-6h1zm-9 4H6v-4h6v4z"></path></g>
<g id="subdirectory-arrow-left"><path d="M11 9l1.42 1.42L8.83 14H18V4h2v12H8.83l3.59 3.58L11 21l-6-6 6-6z"></path></g>
<g id="subdirectory-arrow-right"><path d="M19 15l-6 6-1.42-1.42L15.17 16H4V4h2v10h9.17l-3.59-3.58L13 9l6 6z"></path></g>
<g id="subject"><path d="M14 17H4v2h10v-2zm6-8H4v2h16V9zM4 15h16v-2H4v2zM4 5v2h16V5H4z"></path></g>
<g id="supervisor-account"><path d="M16.5 12c1.38 0 2.49-1.12 2.49-2.5S17.88 7 16.5 7C15.12 7 14 8.12 14 9.5s1.12 2.5 2.5 2.5zM9 11c1.66 0 2.99-1.34 2.99-3S10.66 5 9 5C7.34 5 6 6.34 6 8s1.34 3 3 3zm7.5 3c-1.83 0-5.5.92-5.5 2.75V19h11v-2.25c0-1.83-3.67-2.75-5.5-2.75zM9 13c-2.33 0-7 1.17-7 3.5V19h7v-2.25c0-.85.33-2.34 2.37-3.47C10.5 13.1 9.66 13 9 13z"></path></g>
<g id="swap-horiz"><path d="M6.99 11L3 15l3.99 4v-3H14v-2H6.99v-3zM21 9l-3.99-4v3H10v2h7.01v3L21 9z"></path></g>
<g id="swap-vert"><path d="M16 17.01V10h-2v7.01h-3L15 21l4-3.99h-3zM9 3L5 6.99h3V14h2V6.99h3L9 3z"></path></g>
<g id="swap-vertical-circle"><path d="M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zM6.5 9L10 5.5 13.5 9H11v4H9V9H6.5zm11 6L14 18.5 10.5 15H13v-4h2v4h2.5z"></path></g>
<g id="system-update-alt"><path d="M12 16.5l4-4h-3v-9h-2v9H8l4 4zm9-13h-6v1.99h6v14.03H3V5.49h6V3.5H3c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h18c1.1 0 2-.9 2-2v-14c0-1.1-.9-2-2-2z"></path></g>
<g id="tab"><path d="M21 3H3c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h18c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm0 16H3V5h10v4h8v10z"></path></g>
<g id="tab-unselected"><path d="M1 9h2V7H1v2zm0 4h2v-2H1v2zm0-8h2V3c-1.1 0-2 .9-2 2zm8 16h2v-2H9v2zm-8-4h2v-2H1v2zm2 4v-2H1c0 1.1.9 2 2 2zM21 3h-8v6h10V5c0-1.1-.9-2-2-2zm0 14h2v-2h-2v2zM9 5h2V3H9v2zM5 21h2v-2H5v2zM5 5h2V3H5v2zm16 16c1.1 0 2-.9 2-2h-2v2zm0-8h2v-2h-2v2zm-8 8h2v-2h-2v2zm4 0h2v-2h-2v2z"></path></g>
<g id="text-format"><path d="M5 17v2h14v-2H5zm4.5-4.2h5l.9 2.2h2.1L12.75 4h-1.5L6.5 15h2.1l.9-2.2zM12 5.98L13.87 11h-3.74L12 5.98z"></path></g>
<g id="theaters"><path d="M18 3v2h-2V3H8v2H6V3H4v18h2v-2h2v2h8v-2h2v2h2V3h-2zM8 17H6v-2h2v2zm0-4H6v-2h2v2zm0-4H6V7h2v2zm10 8h-2v-2h2v2zm0-4h-2v-2h2v2zm0-4h-2V7h2v2z"></path></g>
<g id="thumb-down"><path d="M15 3H6c-.83 0-1.54.5-1.84 1.22l-3.02 7.05c-.09.23-.14.47-.14.73v1.91l.01.01L1 14c0 1.1.9 2 2 2h6.31l-.95 4.57-.03.32c0 .41.17.79.44 1.06L9.83 23l6.59-6.59c.36-.36.58-.86.58-1.41V5c0-1.1-.9-2-2-2zm4 0v12h4V3h-4z"></path></g>
<g id="thumb-up"><path d="M1 21h4V9H1v12zm22-11c0-1.1-.9-2-2-2h-6.31l.95-4.57.03-.32c0-.41-.17-.79-.44-1.06L14.17 1 7.59 7.59C7.22 7.95 7 8.45 7 9v10c0 1.1.9 2 2 2h9c.83 0 1.54-.5 1.84-1.22l3.02-7.05c.09-.23.14-.47.14-.73v-1.91l-.01-.01L23 10z"></path></g>
<g id="thumbs-up-down"><path d="M12 6c0-.55-.45-1-1-1H5.82l.66-3.18.02-.23c0-.31-.13-.59-.33-.8L5.38 0 .44 4.94C.17 5.21 0 5.59 0 6v6.5c0 .83.67 1.5 1.5 1.5h6.75c.62 0 1.15-.38 1.38-.91l2.26-5.29c.07-.17.11-.36.11-.55V6zm10.5 4h-6.75c-.62 0-1.15.38-1.38.91l-2.26 5.29c-.07.17-.11.36-.11.55V18c0 .55.45 1 1 1h5.18l-.66 3.18-.02.24c0 .31.13.59.33.8l.79.78 4.94-4.94c.27-.27.44-.65.44-1.06v-6.5c0-.83-.67-1.5-1.5-1.5z"></path></g>
<g id="timeline"><path d="M23 8c0 1.1-.9 2-2 2-.18 0-.35-.02-.51-.07l-3.56 3.55c.05.16.07.34.07.52 0 1.1-.9 2-2 2s-2-.9-2-2c0-.18.02-.36.07-.52l-2.55-2.55c-.16.05-.34.07-.52.07s-.36-.02-.52-.07l-4.55 4.56c.05.16.07.33.07.51 0 1.1-.9 2-2 2s-2-.9-2-2 .9-2 2-2c.18 0 .35.02.51.07l4.56-4.55C8.02 9.36 8 9.18 8 9c0-1.1.9-2 2-2s2 .9 2 2c0 .18-.02.36-.07.52l2.55 2.55c.16-.05.34-.07.52-.07s.36.02.52.07l3.55-3.56C19.02 8.35 19 8.18 19 8c0-1.1.9-2 2-2s2 .9 2 2z"></path></g>
<g id="toc"><path d="M3 9h14V7H3v2zm0 4h14v-2H3v2zm0 4h14v-2H3v2zm16 0h2v-2h-2v2zm0-10v2h2V7h-2zm0 6h2v-2h-2v2z"></path></g>
<g id="today"><path d="M19 3h-1V1h-2v2H8V1H6v2H5c-1.11 0-1.99.9-1.99 2L3 19c0 1.1.89 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm0 16H5V8h14v11zM7 10h5v5H7z"></path></g>
<g id="toll"><path d="M15 4c-4.42 0-8 3.58-8 8s3.58 8 8 8 8-3.58 8-8-3.58-8-8-8zm0 14c-3.31 0-6-2.69-6-6s2.69-6 6-6 6 2.69 6 6-2.69 6-6 6zM3 12c0-2.61 1.67-4.83 4-5.65V4.26C3.55 5.15 1 8.27 1 12s2.55 6.85 6 7.74v-2.09c-2.33-.82-4-3.04-4-5.65z"></path></g>
<g id="touch-app"><path d="M9 11.24V7.5C9 6.12 10.12 5 11.5 5S14 6.12 14 7.5v3.74c1.21-.81 2-2.18 2-3.74C16 5.01 13.99 3 11.5 3S7 5.01 7 7.5c0 1.56.79 2.93 2 3.74zm9.84 4.63l-4.54-2.26c-.17-.07-.35-.11-.54-.11H13v-6c0-.83-.67-1.5-1.5-1.5S10 6.67 10 7.5v10.74l-3.43-.72c-.08-.01-.15-.03-.24-.03-.31 0-.59.13-.79.33l-.79.8 4.94 4.94c.27.27.65.44 1.06.44h6.79c.75 0 1.33-.55 1.44-1.28l.75-5.27c.01-.07.02-.14.02-.2 0-.62-.38-1.16-.91-1.38z"></path></g>
<g id="track-changes"><path d="M19.07 4.93l-1.41 1.41C19.1 7.79 20 9.79 20 12c0 4.42-3.58 8-8 8s-8-3.58-8-8c0-4.08 3.05-7.44 7-7.93v2.02C8.16 6.57 6 9.03 6 12c0 3.31 2.69 6 6 6s6-2.69 6-6c0-1.66-.67-3.16-1.76-4.24l-1.41 1.41C15.55 9.9 16 10.9 16 12c0 2.21-1.79 4-4 4s-4-1.79-4-4c0-1.86 1.28-3.41 3-3.86v2.14c-.6.35-1 .98-1 1.72 0 1.1.9 2 2 2s2-.9 2-2c0-.74-.4-1.38-1-1.72V2h-1C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10c0-2.76-1.12-5.26-2.93-7.07z"></path></g>
<g id="translate"><path d="M12.87 15.07l-2.54-2.51.03-.03c1.74-1.94 2.98-4.17 3.71-6.53H17V4h-7V2H8v2H1v1.99h11.17C11.5 7.92 10.44 9.75 9 11.35 8.07 10.32 7.3 9.19 6.69 8h-2c.73 1.63 1.73 3.17 2.98 4.56l-5.09 5.02L4 19l5-5 3.11 3.11.76-2.04zM18.5 10h-2L12 22h2l1.12-3h4.75L21 22h2l-4.5-12zm-2.62 7l1.62-4.33L19.12 17h-3.24z"></path></g>
<g id="trending-down"><path d="M16 18l2.29-2.29-4.88-4.88-4 4L2 7.41 3.41 6l6 6 4-4 6.3 6.29L22 12v6z"></path></g>
<g id="trending-flat"><path d="M22 12l-4-4v3H3v2h15v3z"></path></g>
<g id="trending-up"><path d="M16 6l2.29 2.29-4.88 4.88-4-4L2 16.59 3.41 18l6-6 4 4 6.3-6.29L22 12V6z"></path></g>
<g id="turned-in"><path d="M17 3H7c-1.1 0-1.99.9-1.99 2L5 21l7-3 7 3V5c0-1.1-.9-2-2-2z"></path></g>
<g id="turned-in-not"><path d="M17 3H7c-1.1 0-1.99.9-1.99 2L5 21l7-3 7 3V5c0-1.1-.9-2-2-2zm0 15l-5-2.18L7 18V5h10v13z"></path></g>
<g id="unarchive"><path d="M20.55 5.22l-1.39-1.68C18.88 3.21 18.47 3 18 3H6c-.47 0-.88.21-1.15.55L3.46 5.22C3.17 5.57 3 6.01 3 6.5V19c0 1.1.89 2 2 2h14c1.1 0 2-.9 2-2V6.5c0-.49-.17-.93-.45-1.28zM12 9.5l5.5 5.5H14v2h-4v-2H6.5L12 9.5zM5.12 5l.82-1h12l.93 1H5.12z"></path></g>
<g id="undo"><path d="M12.5 8c-2.65 0-5.05.99-6.9 2.6L2 7v9h9l-3.62-3.62c1.39-1.16 3.16-1.88 5.12-1.88 3.54 0 6.55 2.31 7.6 5.5l2.37-.78C21.08 11.03 17.15 8 12.5 8z"></path></g>
<g id="unfold-less"><path d="M7.41 18.59L8.83 20 12 16.83 15.17 20l1.41-1.41L12 14l-4.59 4.59zm9.18-13.18L15.17 4 12 7.17 8.83 4 7.41 5.41 12 10l4.59-4.59z"></path></g>
<g id="unfold-more"><path d="M12 5.83L15.17 9l1.41-1.41L12 3 7.41 7.59 8.83 9 12 5.83zm0 12.34L8.83 15l-1.41 1.41L12 21l4.59-4.59L15.17 15 12 18.17z"></path></g>
<g id="update"><path d="M21 10.12h-6.78l2.74-2.82c-2.73-2.7-7.15-2.8-9.88-.1-2.73 2.71-2.73 7.08 0 9.79 2.73 2.71 7.15 2.71 9.88 0C18.32 15.65 19 14.08 19 12.1h2c0 1.98-.88 4.55-2.64 6.29-3.51 3.48-9.21 3.48-12.72 0-3.5-3.47-3.53-9.11-.02-12.58 3.51-3.47 9.14-3.47 12.65 0L21 3v7.12zM12.5 8v4.25l3.5 2.08-.72 1.21L11 13V8h1.5z"></path></g>
<g id="verified-user"><path d="M12 1L3 5v6c0 5.55 3.84 10.74 9 12 5.16-1.26 9-6.45 9-12V5l-9-4zm-2 16l-4-4 1.41-1.41L10 14.17l6.59-6.59L18 9l-8 8z"></path></g>
<g id="view-agenda"><path d="M20 13H3c-.55 0-1 .45-1 1v6c0 .55.45 1 1 1h17c.55 0 1-.45 1-1v-6c0-.55-.45-1-1-1zm0-10H3c-.55 0-1 .45-1 1v6c0 .55.45 1 1 1h17c.55 0 1-.45 1-1V4c0-.55-.45-1-1-1z"></path></g>
<g id="view-array"><path d="M4 18h3V5H4v13zM18 5v13h3V5h-3zM8 18h9V5H8v13z"></path></g>
<g id="view-carousel"><path d="M7 19h10V4H7v15zm-5-2h4V6H2v11zM18 6v11h4V6h-4z"></path></g>
<g id="view-column"><path d="M10 18h5V5h-5v13zm-6 0h5V5H4v13zM16 5v13h5V5h-5z"></path></g>
<g id="view-day"><path d="M2 21h19v-3H2v3zM20 8H3c-.55 0-1 .45-1 1v6c0 .55.45 1 1 1h17c.55 0 1-.45 1-1V9c0-.55-.45-1-1-1zM2 3v3h19V3H2z"></path></g>
<g id="view-headline"><path d="M4 15h16v-2H4v2zm0 4h16v-2H4v2zm0-8h16V9H4v2zm0-6v2h16V5H4z"></path></g>
<g id="view-list"><path d="M4 14h4v-4H4v4zm0 5h4v-4H4v4zM4 9h4V5H4v4zm5 5h12v-4H9v4zm0 5h12v-4H9v4zM9 5v4h12V5H9z"></path></g>
<g id="view-module"><path d="M4 11h5V5H4v6zm0 7h5v-6H4v6zm6 0h5v-6h-5v6zm6 0h5v-6h-5v6zm-6-7h5V5h-5v6zm6-6v6h5V5h-5z"></path></g>
<g id="view-quilt"><path d="M10 18h5v-6h-5v6zm-6 0h5V5H4v13zm12 0h5v-6h-5v6zM10 5v6h11V5H10z"></path></g>
<g id="view-stream"><path d="M4 18h17v-6H4v6zM4 5v6h17V5H4z"></path></g>
<g id="view-week"><path d="M6 5H3c-.55 0-1 .45-1 1v12c0 .55.45 1 1 1h3c.55 0 1-.45 1-1V6c0-.55-.45-1-1-1zm14 0h-3c-.55 0-1 .45-1 1v12c0 .55.45 1 1 1h3c.55 0 1-.45 1-1V6c0-.55-.45-1-1-1zm-7 0h-3c-.55 0-1 .45-1 1v12c0 .55.45 1 1 1h3c.55 0 1-.45 1-1V6c0-.55-.45-1-1-1z"></path></g>
<g id="visibility"><path d="M12 4.5C7 4.5 2.73 7.61 1 12c1.73 4.39 6 7.5 11 7.5s9.27-3.11 11-7.5c-1.73-4.39-6-7.5-11-7.5zM12 17c-2.76 0-5-2.24-5-5s2.24-5 5-5 5 2.24 5 5-2.24 5-5 5zm0-8c-1.66 0-3 1.34-3 3s1.34 3 3 3 3-1.34 3-3-1.34-3-3-3z"></path></g>
<g id="visibility-off"><path d="M12 7c2.76 0 5 2.24 5 5 0 .65-.13 1.26-.36 1.83l2.92 2.92c1.51-1.26 2.7-2.89 3.43-4.75-1.73-4.39-6-7.5-11-7.5-1.4 0-2.74.25-3.98.7l2.16 2.16C10.74 7.13 11.35 7 12 7zM2 4.27l2.28 2.28.46.46C3.08 8.3 1.78 10.02 1 12c1.73 4.39 6 7.5 11 7.5 1.55 0 3.03-.3 4.38-.84l.42.42L19.73 22 21 20.73 3.27 3 2 4.27zM7.53 9.8l1.55 1.55c-.05.21-.08.43-.08.65 0 1.66 1.34 3 3 3 .22 0 .44-.03.65-.08l1.55 1.55c-.67.33-1.41.53-2.2.53-2.76 0-5-2.24-5-5 0-.79.2-1.53.53-2.2zm4.31-.78l3.15 3.15.02-.16c0-1.66-1.34-3-3-3l-.17.01z"></path></g>
<g id="warning"><path d="M1 21h22L12 2 1 21zm12-3h-2v-2h2v2zm0-4h-2v-4h2v4z"></path></g>
<g id="watch-later"><path d="M12 2C6.5 2 2 6.5 2 12s4.5 10 10 10 10-4.5 10-10S17.5 2 12 2zm4.2 14.2L11 13V7h1.5v5.2l4.5 2.7-.8 1.3z"></path></g>
<g id="weekend"><path d="M21 10c-1.1 0-2 .9-2 2v3H5v-3c0-1.1-.9-2-2-2s-2 .9-2 2v5c0 1.1.9 2 2 2h18c1.1 0 2-.9 2-2v-5c0-1.1-.9-2-2-2zm-3-5H6c-1.1 0-2 .9-2 2v2.15c1.16.41 2 1.51 2 2.82V14h12v-2.03c0-1.3.84-2.4 2-2.82V7c0-1.1-.9-2-2-2z"></path></g>
<g id="work"><path d="M20 6h-4V4c0-1.11-.89-2-2-2h-4c-1.11 0-2 .89-2 2v2H4c-1.11 0-1.99.89-1.99 2L2 19c0 1.11.89 2 2 2h16c1.11 0 2-.89 2-2V8c0-1.11-.89-2-2-2zm-6 0h-4V4h4v2z"></path></g>
<g id="youtube-searched-for"><path d="M17.01 14h-.8l-.27-.27c.98-1.14 1.57-2.61 1.57-4.23 0-3.59-2.91-6.5-6.5-6.5s-6.5 3-6.5 6.5H2l3.84 4 4.16-4H6.51C6.51 7 8.53 5 11.01 5s4.5 2.01 4.5 4.5c0 2.48-2.02 4.5-4.5 4.5-.65 0-1.26-.14-1.82-.38L7.71 15.1c.97.57 2.09.9 3.3.9 1.61 0 3.08-.59 4.22-1.57l.27.27v.79l5.01 4.99L22 19l-4.99-5z"></path></g>
<g id="zoom-in"><path d="M15.5 14h-.79l-.28-.27C15.41 12.59 16 11.11 16 9.5 16 5.91 13.09 3 9.5 3S3 5.91 3 9.5 5.91 16 9.5 16c1.61 0 3.09-.59 4.23-1.57l.27.28v.79l5 4.99L20.49 19l-4.99-5zm-6 0C7.01 14 5 11.99 5 9.5S7.01 5 9.5 5 14 7.01 14 9.5 11.99 14 9.5 14zm2.5-4h-2v2H9v-2H7V9h2V7h1v2h2v1z"></path></g>
<g id="zoom-out"><path d="M15.5 14h-.79l-.28-.27C15.41 12.59 16 11.11 16 9.5 16 5.91 13.09 3 9.5 3S3 5.91 3 9.5 5.91 16 9.5 16c1.61 0 3.09-.59 4.23-1.57l.27.28v.79l5 4.99L20.49 19l-4.99-5zm-6 0C7.01 14 5 11.99 5 9.5S7.01 5 9.5 5 14 7.01 14 9.5 11.99 14 9.5 14zM7 9h5v1H7z"></path></g>
</defs></svg>
</iron-iconset-svg>`;

    document.head.appendChild(template$3.content);

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
     * Chrome uses an older version of DOM Level 3 Keyboard Events
     *
     * Most keys are labeled as text, but some are Unicode codepoints.
     * Values taken from:
     * http://www.w3.org/TR/2007/WD-DOM-Level-3-Events-20071221/keyset.html#KeySet-Set
     */
    var KEY_IDENTIFIER = {
      'U+0008': 'backspace',
      'U+0009': 'tab',
      'U+001B': 'esc',
      'U+0020': 'space',
      'U+007F': 'del'
    };

    /**
     * Special table for KeyboardEvent.keyCode.
     * KeyboardEvent.keyIdentifier is better, and KeyBoardEvent.key is even better
     * than that.
     *
     * Values from:
     * https://developer.mozilla.org/en-US/docs/Web/API/KeyboardEvent.keyCode#Value_of_keyCode
     */
    var KEY_CODE = {
      8: 'backspace',
      9: 'tab',
      13: 'enter',
      27: 'esc',
      33: 'pageup',
      34: 'pagedown',
      35: 'end',
      36: 'home',
      32: 'space',
      37: 'left',
      38: 'up',
      39: 'right',
      40: 'down',
      46: 'del',
      106: '*'
    };

    /**
     * MODIFIER_KEYS maps the short name for modifier keys used in a key
     * combo string to the property name that references those same keys
     * in a KeyboardEvent instance.
     */
    var MODIFIER_KEYS = {
      'shift': 'shiftKey',
      'ctrl': 'ctrlKey',
      'alt': 'altKey',
      'meta': 'metaKey'
    };

    /**
     * KeyboardEvent.key is mostly represented by printable character made by
     * the keyboard, with unprintable keys labeled nicely.
     *
     * However, on OS X, Alt+char can make a Unicode character that follows an
     * Apple-specific mapping. In this case, we fall back to .keyCode.
     */
    var KEY_CHAR = /[a-z0-9*]/;

    /**
     * Matches a keyIdentifier string.
     */
    var IDENT_CHAR = /U\+/;

    /**
     * Matches arrow keys in Gecko 27.0+
     */
    var ARROW_KEY = /^arrow/;

    /**
     * Matches space keys everywhere (notably including IE10's exceptional name
     * `spacebar`).
     */
    var SPACE_KEY = /^space(bar)?/;

    /**
     * Matches ESC key.
     *
     * Value from: http://w3c.github.io/uievents-key/#key-Escape
     */
    var ESC_KEY = /^escape$/;

    /**
     * Transforms the key.
     * @param {string} key The KeyBoardEvent.key
     * @param {Boolean} [noSpecialChars] Limits the transformation to
     * alpha-numeric characters.
     */
    function transformKey(key, noSpecialChars) {
      var validKey = '';
      if (key) {
        var lKey = key.toLowerCase();
        if (lKey === ' ' || SPACE_KEY.test(lKey)) {
          validKey = 'space';
        } else if (ESC_KEY.test(lKey)) {
          validKey = 'esc';
        } else if (lKey.length == 1) {
          if (!noSpecialChars || KEY_CHAR.test(lKey)) {
            validKey = lKey;
          }
        } else if (ARROW_KEY.test(lKey)) {
          validKey = lKey.replace('arrow', '');
        } else if (lKey == 'multiply') {
          // numpad '*' can map to Multiply on IE/Windows
          validKey = '*';
        } else {
          validKey = lKey;
        }
      }
      return validKey;
    }

    function transformKeyIdentifier(keyIdent) {
      var validKey = '';
      if (keyIdent) {
        if (keyIdent in KEY_IDENTIFIER) {
          validKey = KEY_IDENTIFIER[keyIdent];
        } else if (IDENT_CHAR.test(keyIdent)) {
          keyIdent = parseInt(keyIdent.replace('U+', '0x'), 16);
          validKey = String.fromCharCode(keyIdent).toLowerCase();
        } else {
          validKey = keyIdent.toLowerCase();
        }
      }
      return validKey;
    }

    function transformKeyCode(keyCode) {
      var validKey = '';
      if (Number(keyCode)) {
        if (keyCode >= 65 && keyCode <= 90) {
          // ascii a-z
          // lowercase is 32 offset from uppercase
          validKey = String.fromCharCode(32 + keyCode);
        } else if (keyCode >= 112 && keyCode <= 123) {
          // function keys f1-f12
          validKey = 'f' + (keyCode - 112 + 1);
        } else if (keyCode >= 48 && keyCode <= 57) {
          // top 0-9 keys
          validKey = String(keyCode - 48);
        } else if (keyCode >= 96 && keyCode <= 105) {
          // num pad 0-9
          validKey = String(keyCode - 96);
        } else {
          validKey = KEY_CODE[keyCode];
        }
      }
      return validKey;
    }

    /**
     * Calculates the normalized key for a KeyboardEvent.
     * @param {KeyboardEvent} keyEvent
     * @param {Boolean} [noSpecialChars] Set to true to limit keyEvent.key
     * transformation to alpha-numeric chars. This is useful with key
     * combinations like shift + 2, which on FF for MacOS produces
     * keyEvent.key = @
     * To get 2 returned, set noSpecialChars = true
     * To get @ returned, set noSpecialChars = false
     */
    function normalizedKeyForEvent(keyEvent, noSpecialChars) {
      // Fall back from .key, to .detail.key for artifical keyboard events,
      // and then to deprecated .keyIdentifier and .keyCode.
      if (keyEvent.key) {
        return transformKey(keyEvent.key, noSpecialChars);
      }
      if (keyEvent.detail && keyEvent.detail.key) {
        return transformKey(keyEvent.detail.key, noSpecialChars);
      }
      return transformKeyIdentifier(keyEvent.keyIdentifier) ||
          transformKeyCode(keyEvent.keyCode) || '';
    }

    function keyComboMatchesEvent(keyCombo, event) {
      // For combos with modifiers we support only alpha-numeric keys
      var keyEvent = normalizedKeyForEvent(event, keyCombo.hasModifiers);
      return keyEvent === keyCombo.key &&
          (!keyCombo.hasModifiers ||
           (!!event.shiftKey === !!keyCombo.shiftKey &&
            !!event.ctrlKey === !!keyCombo.ctrlKey &&
            !!event.altKey === !!keyCombo.altKey &&
            !!event.metaKey === !!keyCombo.metaKey));
    }

    function parseKeyComboString(keyComboString) {
      if (keyComboString.length === 1) {
        return {combo: keyComboString, key: keyComboString, event: 'keydown'};
      }
      return keyComboString.split('+')
          .reduce(function(parsedKeyCombo, keyComboPart) {
            var eventParts = keyComboPart.split(':');
            var keyName = eventParts[0];
            var event = eventParts[1];

            if (keyName in MODIFIER_KEYS) {
              parsedKeyCombo[MODIFIER_KEYS[keyName]] = true;
              parsedKeyCombo.hasModifiers = true;
            } else {
              parsedKeyCombo.key = keyName;
              parsedKeyCombo.event = event || 'keydown';
            }

            return parsedKeyCombo;
          }, {combo: keyComboString.split(':').shift()});
    }

    function parseEventString(eventString) {
      return eventString.trim().split(' ').map(function(keyComboString) {
        return parseKeyComboString(keyComboString);
      });
    }

    /**
     * `Polymer.IronA11yKeysBehavior` provides a normalized interface for processing
     * keyboard commands that pertain to [WAI-ARIA best
     * practices](http://www.w3.org/TR/wai-aria-practices/#kbd_general_binding). The
     * element takes care of browser differences with respect to Keyboard events and
     * uses an expressive syntax to filter key presses.
     *
     * Use the `keyBindings` prototype property to express what combination of keys
     * will trigger the callback. A key binding has the format
     * `"KEY+MODIFIER:EVENT": "callback"` (`"KEY": "callback"` or
     * `"KEY:EVENT": "callback"` are valid as well). Some examples:
     *
     *      keyBindings: {
     *        'space': '_onKeydown', // same as 'space:keydown'
     *        'shift+tab': '_onKeydown',
     *        'enter:keypress': '_onKeypress',
     *        'esc:keyup': '_onKeyup'
     *      }
     *
     * The callback will receive with an event containing the following information
     * in `event.detail`:
     *
     *      _onKeydown: function(event) {
     *        console.log(event.detail.combo); // KEY+MODIFIER, e.g. "shift+tab"
     *        console.log(event.detail.key); // KEY only, e.g. "tab"
     *        console.log(event.detail.event); // EVENT, e.g. "keydown"
     *        console.log(event.detail.keyboardEvent); // the original KeyboardEvent
     *      }
     *
     * Use the `keyEventTarget` attribute to set up event handlers on a specific
     * node.
     *
     * See the [demo source
     * code](https://github.com/PolymerElements/iron-a11y-keys-behavior/blob/master/demo/x-key-aware.html)
     * for an example.
     *
     * @demo demo/index.html
     * @polymerBehavior
     */
    const IronA11yKeysBehavior = {
      properties: {
        /**
         * The EventTarget that will be firing relevant KeyboardEvents. Set it to
         * `null` to disable the listeners.
         * @type {?EventTarget}
         */
        keyEventTarget: {
          type: Object,
          value: function() {
            return this;
          }
        },

        /**
         * If true, this property will cause the implementing element to
         * automatically stop propagation on any handled KeyboardEvents.
         */
        stopKeyboardEventPropagation: {type: Boolean, value: false},

        _boundKeyHandlers: {
          type: Array,
          value: function() {
            return [];
          }
        },

        // We use this due to a limitation in IE10 where instances will have
        // own properties of everything on the "prototype".
        _imperativeKeyBindings: {
          type: Object,
          value: function() {
            return {};
          }
        }
      },

      observers: ['_resetKeyEventListeners(keyEventTarget, _boundKeyHandlers)'],


      /**
       * To be used to express what combination of keys  will trigger the relative
       * callback. e.g. `keyBindings: { 'esc': '_onEscPressed'}`
       * @type {!Object}
       */
      keyBindings: {},

      registered: function() {
        this._prepKeyBindings();
      },

      attached: function() {
        this._listenKeyEventListeners();
      },

      detached: function() {
        this._unlistenKeyEventListeners();
      },

      /**
       * Can be used to imperatively add a key binding to the implementing
       * element. This is the imperative equivalent of declaring a keybinding
       * in the `keyBindings` prototype property.
       *
       * @param {string} eventString
       * @param {string} handlerName
       */
      addOwnKeyBinding: function(eventString, handlerName) {
        this._imperativeKeyBindings[eventString] = handlerName;
        this._prepKeyBindings();
        this._resetKeyEventListeners();
      },

      /**
       * When called, will remove all imperatively-added key bindings.
       */
      removeOwnKeyBindings: function() {
        this._imperativeKeyBindings = {};
        this._prepKeyBindings();
        this._resetKeyEventListeners();
      },

      /**
       * Returns true if a keyboard event matches `eventString`.
       *
       * @param {KeyboardEvent} event
       * @param {string} eventString
       * @return {boolean}
       */
      keyboardEventMatchesKeys: function(event, eventString) {
        var keyCombos = parseEventString(eventString);
        for (var i = 0; i < keyCombos.length; ++i) {
          if (keyComboMatchesEvent(keyCombos[i], event)) {
            return true;
          }
        }
        return false;
      },

      _collectKeyBindings: function() {
        var keyBindings = this.behaviors.map(function(behavior) {
          return behavior.keyBindings;
        });

        if (keyBindings.indexOf(this.keyBindings) === -1) {
          keyBindings.push(this.keyBindings);
        }

        return keyBindings;
      },

      _prepKeyBindings: function() {
        this._keyBindings = {};

        this._collectKeyBindings().forEach(function(keyBindings) {
          for (var eventString in keyBindings) {
            this._addKeyBinding(eventString, keyBindings[eventString]);
          }
        }, this);

        for (var eventString in this._imperativeKeyBindings) {
          this._addKeyBinding(
              eventString, this._imperativeKeyBindings[eventString]);
        }

        // Give precedence to combos with modifiers to be checked first.
        for (var eventName in this._keyBindings) {
          this._keyBindings[eventName].sort(function(kb1, kb2) {
            var b1 = kb1[0].hasModifiers;
            var b2 = kb2[0].hasModifiers;
            return (b1 === b2) ? 0 : b1 ? -1 : 1;
          });
        }
      },

      _addKeyBinding: function(eventString, handlerName) {
        parseEventString(eventString).forEach(function(keyCombo) {
          this._keyBindings[keyCombo.event] =
              this._keyBindings[keyCombo.event] || [];

          this._keyBindings[keyCombo.event].push([keyCombo, handlerName]);
        }, this);
      },

      _resetKeyEventListeners: function() {
        this._unlistenKeyEventListeners();

        if (this.isAttached) {
          this._listenKeyEventListeners();
        }
      },

      _listenKeyEventListeners: function() {
        if (!this.keyEventTarget) {
          return;
        }
        Object.keys(this._keyBindings).forEach(function(eventName) {
          var keyBindings = this._keyBindings[eventName];
          var boundKeyHandler = this._onKeyBindingEvent.bind(this, keyBindings);

          this._boundKeyHandlers.push(
              [this.keyEventTarget, eventName, boundKeyHandler]);

          this.keyEventTarget.addEventListener(eventName, boundKeyHandler);
        }, this);
      },

      _unlistenKeyEventListeners: function() {
        var keyHandlerTuple;
        var keyEventTarget;
        var eventName;
        var boundKeyHandler;

        while (this._boundKeyHandlers.length) {
          // My kingdom for block-scope binding and destructuring assignment..
          keyHandlerTuple = this._boundKeyHandlers.pop();
          keyEventTarget = keyHandlerTuple[0];
          eventName = keyHandlerTuple[1];
          boundKeyHandler = keyHandlerTuple[2];

          keyEventTarget.removeEventListener(eventName, boundKeyHandler);
        }
      },

      _onKeyBindingEvent: function(keyBindings, event) {
        if (this.stopKeyboardEventPropagation) {
          event.stopPropagation();
        }

        // if event has been already prevented, don't do anything
        if (event.defaultPrevented) {
          return;
        }

        for (var i = 0; i < keyBindings.length; i++) {
          var keyCombo = keyBindings[i][0];
          var handlerName = keyBindings[i][1];
          if (keyComboMatchesEvent(keyCombo, event)) {
            this._triggerKeyHandler(keyCombo, handlerName, event);
            // exit the loop if eventDefault was prevented
            if (event.defaultPrevented) {
              return;
            }
          }
        }
      },

      _triggerKeyHandler: function(keyCombo, handlerName, keyboardEvent) {
        var detail = Object.create(keyCombo);
        detail.keyboardEvent = keyboardEvent;
        var event =
            new CustomEvent(keyCombo.event, {detail: detail, cancelable: true});
        this[handlerName].call(this, event);
        if (event.defaultPrevented) {
          keyboardEvent.preventDefault();
        }
      }
    };

    /**
    @license
    Copyright (c) 2016 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
     * `Polymer.IronScrollTargetBehavior` allows an element to respond to scroll
     * events from a designated scroll target.
     *
     * Elements that consume this behavior can override the `_scrollHandler`
     * method to add logic on the scroll event.
     *
     * @demo demo/scrolling-region.html Scrolling Region
     * @demo demo/document.html Document Element
     * @polymerBehavior
     */
    const IronScrollTargetBehavior = {

      properties: {

        /**
         * Specifies the element that will handle the scroll event
         * on the behalf of the current element. This is typically a reference to an
         *element, but there are a few more posibilities:
         *
         * ### Elements id
         *
         *```html
         * <div id="scrollable-element" style="overflow: auto;">
         *  <x-element scroll-target="scrollable-element">
         *    <!-- Content-->
         *  </x-element>
         * </div>
         *```
         * In this case, the `scrollTarget` will point to the outer div element.
         *
         * ### Document scrolling
         *
         * For document scrolling, you can use the reserved word `document`:
         *
         *```html
         * <x-element scroll-target="document">
         *   <!-- Content -->
         * </x-element>
         *```
         *
         * ### Elements reference
         *
         *```js
         * appHeader.scrollTarget = document.querySelector('#scrollable-element');
         *```
         *
         * @type {HTMLElement}
         * @default document
         */
        scrollTarget: {
          type: HTMLElement,
          value: function() {
            return this._defaultScrollTarget;
          }
        }
      },

      observers: ['_scrollTargetChanged(scrollTarget, isAttached)'],

      /**
       * True if the event listener should be installed.
       */
      _shouldHaveListener: true,

      _scrollTargetChanged: function(scrollTarget, isAttached) {

        if (this._oldScrollTarget) {
          this._toggleScrollListener(false, this._oldScrollTarget);
          this._oldScrollTarget = null;
        }
        if (!isAttached) {
          return;
        }
        // Support element id references
        if (scrollTarget === 'document') {
          this.scrollTarget = this._doc;

        } else if (typeof scrollTarget === 'string') {
          var domHost = this.domHost;

          this.scrollTarget = domHost && domHost.$ ?
              domHost.$[scrollTarget] :
              dom(this.ownerDocument).querySelector('#' + scrollTarget);

        } else if (this._isValidScrollTarget()) {
          this._oldScrollTarget = scrollTarget;
          this._toggleScrollListener(this._shouldHaveListener, scrollTarget);
        }
      },

      /**
       * Runs on every scroll event. Consumer of this behavior may override this
       * method.
       *
       * @protected
       */
      _scrollHandler: function scrollHandler() {},

      /**
       * The default scroll target. Consumers of this behavior may want to customize
       * the default scroll target.
       *
       * @type {Element}
       */
      get _defaultScrollTarget() {
        return this._doc;
      },

      /**
       * Shortcut for the document element
       *
       * @type {Element}
       */
      get _doc() {
        return this.ownerDocument.documentElement;
      },

      /**
       * Gets the number of pixels that the content of an element is scrolled
       * upward.
       *
       * @type {number}
       */
      get _scrollTop() {
        if (this._isValidScrollTarget()) {
          return this.scrollTarget === this._doc ? window.pageYOffset :
                                                   this.scrollTarget.scrollTop;
        }
        return 0;
      },

      /**
       * Gets the number of pixels that the content of an element is scrolled to the
       * left.
       *
       * @type {number}
       */
      get _scrollLeft() {
        if (this._isValidScrollTarget()) {
          return this.scrollTarget === this._doc ? window.pageXOffset :
                                                   this.scrollTarget.scrollLeft;
        }
        return 0;
      },

      /**
       * Sets the number of pixels that the content of an element is scrolled
       * upward.
       *
       * @type {number}
       */
      set _scrollTop(top) {
        if (this.scrollTarget === this._doc) {
          window.scrollTo(window.pageXOffset, top);
        } else if (this._isValidScrollTarget()) {
          this.scrollTarget.scrollTop = top;
        }
      },

      /**
       * Sets the number of pixels that the content of an element is scrolled to the
       * left.
       *
       * @type {number}
       */
      set _scrollLeft(left) {
        if (this.scrollTarget === this._doc) {
          window.scrollTo(left, window.pageYOffset);
        } else if (this._isValidScrollTarget()) {
          this.scrollTarget.scrollLeft = left;
        }
      },

      /**
       * Scrolls the content to a particular place.
       *
       * @method scroll
       * @param {number|!{left: number, top: number}} leftOrOptions The left position or scroll options
       * @param {number=} top The top position
       * @return {void}
       */
      scroll: function(leftOrOptions, top) {
        var left;

        if (typeof leftOrOptions === 'object') {
          left = leftOrOptions.left;
          top = leftOrOptions.top;
        } else {
          left = leftOrOptions;
        }

        left = left || 0;
        top = top || 0;
        if (this.scrollTarget === this._doc) {
          window.scrollTo(left, top);
        } else if (this._isValidScrollTarget()) {
          this.scrollTarget.scrollLeft = left;
          this.scrollTarget.scrollTop = top;
        }
      },

      /**
       * Gets the width of the scroll target.
       *
       * @type {number}
       */
      get _scrollTargetWidth() {
        if (this._isValidScrollTarget()) {
          return this.scrollTarget === this._doc ? window.innerWidth :
                                                   this.scrollTarget.offsetWidth;
        }
        return 0;
      },

      /**
       * Gets the height of the scroll target.
       *
       * @type {number}
       */
      get _scrollTargetHeight() {
        if (this._isValidScrollTarget()) {
          return this.scrollTarget === this._doc ? window.innerHeight :
                                                   this.scrollTarget.offsetHeight;
        }
        return 0;
      },

      /**
       * Returns true if the scroll target is a valid HTMLElement.
       *
       * @return {boolean}
       */
      _isValidScrollTarget: function() {
        return this.scrollTarget instanceof HTMLElement;
      },

      _toggleScrollListener: function(yes, scrollTarget) {
        var eventTarget = scrollTarget === this._doc ? window : scrollTarget;
        if (yes) {
          if (!this._boundScrollHandler) {
            this._boundScrollHandler = this._scrollHandler.bind(this);
            eventTarget.addEventListener('scroll', this._boundScrollHandler);
          }
        } else {
          if (this._boundScrollHandler) {
            eventTarget.removeEventListener('scroll', this._boundScrollHandler);
            this._boundScrollHandler = null;
          }
        }
      },

      /**
       * Enables or disables the scroll event listener.
       *
       * @param {boolean} yes True to add the event, False to remove it.
       */
      toggleScrollListener: function(yes) {
        this._shouldHaveListener = yes;
        this._toggleScrollListener(yes, this.scrollTarget);
      }

    };

    /**
    @license
    Copyright (c) 2016 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    var IOS = navigator.userAgent.match(/iP(?:hone|ad;(?: U;)? CPU) OS (\d+)/);
    var IOS_TOUCH_SCROLLING = IOS && IOS[1] >= 8;
    var DEFAULT_PHYSICAL_COUNT = 3;
    var HIDDEN_Y = '-10000px';
    var SECRET_TABINDEX = -100;

    /**

    `iron-list` displays a virtual, 'infinite' list. The template inside
    the iron-list element represents the DOM to create for each list item.
    The `items` property specifies an array of list item data.

    For performance reasons, not every item in the list is rendered at once;
    instead a small subset of actual template elements *(enough to fill the
    viewport)* are rendered and reused as the user scrolls. As such, it is important
    that all state of the list template is bound to the model driving it, since the
    view may be reused with a new model at any time. Particularly, any state that
    may change as the result of a user interaction with the list item must be bound
    to the model to avoid view state inconsistency.

    ### Sizing iron-list

    `iron-list` must either be explicitly sized, or delegate scrolling to an
    explicitly sized parent. By "explicitly sized", we mean it either has an
    explicit CSS `height` property set via a class or inline style, or else is sized
    by other layout means (e.g. the `flex` or `fit` classes).

    #### Flexbox - [jsbin](https://jsbin.com/vejoni/edit?html,output)

    ```html
    <template is="x-list">
      <style>
        :host {
          display: block;
          height: 100vh;
          display: flex;
          flex-direction: column;
        }

        iron-list {
          flex: 1 1 auto;
        }
      </style>
      <app-toolbar>App name</app-toolbar>
      <iron-list items="[[items]]">
        <template>
          <div>
            ...
          </div>
        </template>
      </iron-list>
    </template>
    ```
    #### Explicit size - [jsbin](https://jsbin.com/vopucus/edit?html,output)
    ```html
    <template is="x-list">
      <style>
        :host {
          display: block;
        }

        iron-list {
          height: 100vh; /* don't use % values unless the parent element is sized.
    *\/
        }
      </style>
      <iron-list items="[[items]]">
        <template>
          <div>
            ...
          </div>
        </template>
      </iron-list>
    </template>
    ```
    #### Main document scrolling -
    [jsbin](https://jsbin.com/wevirow/edit?html,output)
    ```html
    <head>
      <style>
        body {
          height: 100vh;
          margin: 0;
          display: flex;
          flex-direction: column;
        }

        app-toolbar {
          position: fixed;
          top: 0;
          left: 0;
          right: 0;
        }

        iron-list {
          /* add padding since the app-toolbar is fixed at the top *\/
          padding-top: 64px;
        }
      </style>
    </head>
    <body>
      <app-toolbar>App name</app-toolbar>
      <iron-list scroll-target="document">
        <template>
          <div>
            ...
          </div>
        </template>
      </iron-list>
    </body>
    ```

    `iron-list` must be given a `<template>` which contains exactly one element. In
    the examples above we used a `<div>`, but you can provide any element (including
    custom elements).

    ### Template model

    List item templates should bind to template models of the following structure:

    ```js
    {
      index: 0,        // index in the item array
      selected: false, // true if the current item is selected
      tabIndex: -1,    // a dynamically generated tabIndex for focus management
      item: {}         // user data corresponding to items[index]
    }
    ```

    Alternatively, you can change the property name used as data index by changing
    the `indexAs` property. The `as` property defines the name of the variable to
    add to the binding scope for the array.

    For example, given the following `data` array:

    ##### data.json

    ```js
    [
      {"name": "Bob"},
      {"name": "Tim"},
      {"name": "Mike"}
    ]
    ```

    The following code would render the list (note the name property is bound from
    the model object provided to the template scope):

    ```html
    <iron-ajax url="data.json" last-response="{{data}}" auto></iron-ajax>
    <iron-list items="[[data]]" as="item">
      <template>
        <div>
          Name: [[item.name]]
        </div>
      </template>
    </iron-list>
    ```

    ### Grid layout

    `iron-list` supports a grid layout in addition to linear layout by setting
    the `grid` attribute.  In this case, the list template item must have both fixed
    width and height (e.g. via CSS). Based on this, the number of items
    per row are determined automatically based on the size of the list viewport.

    ### Accessibility

    `iron-list` automatically manages the focus state for the items. It also
    provides a `tabIndex` property within the template scope that can be used for
    keyboard navigation. For example, users can press the up and down keys to move
    to previous and next items in the list:

    ```html
    <iron-list items="[[data]]" as="item">
      <template>
        <div tabindex$="[[tabIndex]]">
          Name: [[item.name]]
        </div>
      </template>
    </iron-list>
    ```

    ### Styling

    You can use the `--iron-list-items-container` mixin to style the container of
    items:

    ```css
    iron-list {
     --iron-list-items-container: {
        margin: auto;
      };
    }
    ```

    ### Resizing

    `iron-list` lays out the items when it receives a notification via the
    `iron-resize` event. This event is fired by any element that implements
    `IronResizableBehavior`.

    By default, elements such as `iron-pages`, `paper-tabs` or `paper-dialog` will
    trigger this event automatically. If you hide the list manually (e.g. you use
    `display: none`) you might want to implement `IronResizableBehavior` or fire
    this event manually right after the list became visible again. For example:

    ```js
    document.querySelector('iron-list').fire('iron-resize');
    ```

    ### When should `<iron-list>` be used?

    `iron-list` should be used when a page has significantly more DOM nodes than the
    ones visible on the screen. e.g. the page has 500 nodes, but only 20 are visible
    at a time. This is why we refer to it as a `virtual` list. In this case, a
    `dom-repeat` will still create 500 nodes which could slow down the web app, but
    `iron-list` will only create 20.

    However, having an `iron-list` does not mean that you can load all the data at
    once. Say you have a million records in the database, you want to split the data
    into pages so you can bring in a page at the time. The page could contain 500
    items, and iron-list will only render 20.

    @element iron-list
    @demo demo/index.html

    */
    Polymer({
      /** @override */
      _template: html`
    <style>
      :host {
        display: block;
      }

      @media only screen and (-webkit-max-device-pixel-ratio: 1) {
        :host {
          will-change: transform;
        }
      }

      #items {
        @apply --iron-list-items-container;
        position: relative;
      }

      :host(:not([grid])) #items > ::slotted(*) {
        width: 100%;
      }

      #items > ::slotted(*) {
        box-sizing: border-box;
        margin: 0;
        position: absolute;
        top: 0;
        will-change: transform;
      }
    </style>

    <array-selector id="selector" items="{{items}}" selected="{{selectedItems}}" selected-item="{{selectedItem}}"></array-selector>

    <div id="items">
      <slot></slot>
    </div>
`,

      is: 'iron-list',

      properties: {

        /**
         * An array containing items determining how many instances of the template
         * to stamp and that that each template instance should bind to.
         */
        items: {type: Array},

        /**
         * The name of the variable to add to the binding scope for the array
         * element associated with a given template instance.
         */
        as: {type: String, value: 'item'},

        /**
         * The name of the variable to add to the binding scope with the index
         * for the row.
         */
        indexAs: {type: String, value: 'index'},

        /**
         * The name of the variable to add to the binding scope to indicate
         * if the row is selected.
         */
        selectedAs: {type: String, value: 'selected'},

        /**
         * When true, the list is rendered as a grid. Grid items must have
         * fixed width and height set via CSS. e.g.
         *
         * ```html
         * <iron-list grid>
         *   <template>
         *      <div style="width: 100px; height: 100px;"> 100x100 </div>
         *   </template>
         * </iron-list>
         * ```
         */
        grid: {
          type: Boolean,
          value: false,
          reflectToAttribute: true,
          observer: '_gridChanged'
        },

        /**
         * When true, tapping a row will select the item, placing its data model
         * in the set of selected items retrievable via the selection property.
         *
         * Note that tapping focusable elements within the list item will not
         * result in selection, since they are presumed to have their * own action.
         */
        selectionEnabled: {type: Boolean, value: false},

        /**
         * When `multiSelection` is false, this is the currently selected item, or
         * `null` if no item is selected.
         */
        selectedItem: {type: Object, notify: true},

        /**
         * When `multiSelection` is true, this is an array that contains the
         * selected items.
         */
        selectedItems: {type: Object, notify: true},

        /**
         * When `true`, multiple items may be selected at once (in this case,
         * `selected` is an array of currently selected items).  When `false`,
         * only one item may be selected at a time.
         */
        multiSelection: {type: Boolean, value: false},

        /**
         * The offset top from the scrolling element to the iron-list element.
         * This value can be computed using the position returned by
         * `getBoundingClientRect()` although it's preferred to use a constant value
         * when possible.
         *
         * This property is useful when an external scrolling element is used and
         * there's some offset between the scrolling element and the list. For
         * example: a header is placed above the list.
         */
        scrollOffset: {type: Number, value: 0}
      },

      observers: [
        '_itemsChanged(items.*)',
        '_selectionEnabledChanged(selectionEnabled)',
        '_multiSelectionChanged(multiSelection)',
        '_setOverflow(scrollTarget, scrollOffset)'
      ],

      behaviors: [
        Templatizer,
        IronResizableBehavior,
        IronScrollTargetBehavior,
        OptionalMutableDataBehavior
      ],

      /**
       * The ratio of hidden tiles that should remain in the scroll direction.
       * Recommended value ~0.5, so it will distribute tiles evenly in both
       * directions.
       */
      _ratio: 0.5,

      /**
       * The padding-top value for the list.
       */
      _scrollerPaddingTop: 0,

      /**
       * This value is a cached value of `scrollTop` from the last `scroll` event.
       */
      _scrollPosition: 0,

      /**
       * The sum of the heights of all the tiles in the DOM.
       */
      _physicalSize: 0,

      /**
       * The average `offsetHeight` of the tiles observed till now.
       */
      _physicalAverage: 0,

      /**
       * The number of tiles which `offsetHeight` > 0 observed until now.
       */
      _physicalAverageCount: 0,

      /**
       * The Y position of the item rendered in the `_physicalStart`
       * tile relative to the scrolling list.
       */
      _physicalTop: 0,

      /**
       * The number of items in the list.
       */
      _virtualCount: 0,

      /**
       * The estimated scroll height based on `_physicalAverage`
       */
      _estScrollHeight: 0,

      /**
       * The scroll height of the dom node
       */
      _scrollHeight: 0,

      /**
       * The height of the list. This is referred as the viewport in the context of
       * list.
       */
      _viewportHeight: 0,

      /**
       * The width of the list. This is referred as the viewport in the context of
       * list.
       */
      _viewportWidth: 0,

      /**
       * An array of DOM nodes that are currently in the tree
       * @type {?Array<!HTMLElement>}
       */
      _physicalItems: null,

      /**
       * An array of heights for each item in `_physicalItems`
       * @type {?Array<number>}
       */
      _physicalSizes: null,

      /**
       * A cached value for the first visible index.
       * See `firstVisibleIndex`
       * @type {?number}
       */
      _firstVisibleIndexVal: null,

      /**
       * A cached value for the last visible index.
       * See `lastVisibleIndex`
       * @type {?number}
       */
      _lastVisibleIndexVal: null,

      /**
       * The max number of pages to render. One page is equivalent to the height of
       * the list.
       */
      _maxPages: 2,

      /**
       * The currently focused physical item.
       */
      _focusedItem: null,

      /**
       * The virtual index of the focused item.
       */
      _focusedVirtualIndex: -1,

      /**
       * The physical index of the focused item.
       */
      _focusedPhysicalIndex: -1,

      /**
       * The the item that is focused if it is moved offscreen.
       * @private {?HTMLElement}
       */
      _offscreenFocusedItem: null,

      /**
       * The item that backfills the `_offscreenFocusedItem` in the physical items
       * list when that item is moved offscreen.
       * @type {?HTMLElement}
       */
      _focusBackfillItem: null,

      /**
       * The maximum items per row
       */
      _itemsPerRow: 1,

      /**
       * The width of each grid item
       */
      _itemWidth: 0,

      /**
       * The height of the row in grid layout.
       */
      _rowHeight: 0,

      /**
       * The cost of stamping a template in ms.
       */
      _templateCost: 0,

      /**
       * Needed to pass event.model property to declarative event handlers -
       * see polymer/polymer#4339.
       */
      _parentModel: true,

      /**
       * The bottom of the physical content.
       */
      get _physicalBottom() {
        return this._physicalTop + this._physicalSize;
      },

      /**
       * The bottom of the scroll.
       */
      get _scrollBottom() {
        return this._scrollPosition + this._viewportHeight;
      },

      /**
       * The n-th item rendered in the last physical item.
       */
      get _virtualEnd() {
        return this._virtualStart + this._physicalCount - 1;
      },

      /**
       * The height of the physical content that isn't on the screen.
       */
      get _hiddenContentSize() {
        var size =
            this.grid ? this._physicalRows * this._rowHeight : this._physicalSize;
        return size - this._viewportHeight;
      },

      /**
       * The parent node for the _userTemplate.
       */
      get _itemsParent() {
        return dom(dom(this._userTemplate).parentNode);
      },

      /**
       * The maximum scroll top value.
       */
      get _maxScrollTop() {
        return this._estScrollHeight - this._viewportHeight + this._scrollOffset;
      },

      /**
       * The largest n-th value for an item such that it can be rendered in
       * `_physicalStart`.
       */
      get _maxVirtualStart() {
        var virtualCount = this._convertIndexToCompleteRow(this._virtualCount);
        return Math.max(0, virtualCount - this._physicalCount);
      },

      set _virtualStart(val) {
        val = this._clamp(val, 0, this._maxVirtualStart);
        if (this.grid) {
          val = val - (val % this._itemsPerRow);
        }
        this._virtualStartVal = val;
      },

      get _virtualStart() {
        return this._virtualStartVal || 0;
      },

      /**
       * The k-th tile that is at the top of the scrolling list.
       */
      set _physicalStart(val) {
        val = val % this._physicalCount;
        if (val < 0) {
          val = this._physicalCount + val;
        }
        if (this.grid) {
          val = val - (val % this._itemsPerRow);
        }
        this._physicalStartVal = val;
      },

      get _physicalStart() {
        return this._physicalStartVal || 0;
      },

      /**
       * The k-th tile that is at the bottom of the scrolling list.
       */
      get _physicalEnd() {
        return (this._physicalStart + this._physicalCount - 1) %
            this._physicalCount;
      },

      set _physicalCount(val) {
        this._physicalCountVal = val;
      },

      get _physicalCount() {
        return this._physicalCountVal || 0;
      },

      /**
       * An optimal physical size such that we will have enough physical items
       * to fill up the viewport and recycle when the user scrolls.
       *
       * This default value assumes that we will at least have the equivalent
       * to a viewport of physical items above and below the user's viewport.
       */
      get _optPhysicalSize() {
        return this._viewportHeight === 0 ? Infinity :
                                            this._viewportHeight * this._maxPages;
      },

      /**
       * True if the current list is visible.
       */
      get _isVisible() {
        return Boolean(this.offsetWidth || this.offsetHeight);
      },

      /**
       * Gets the index of the first visible item in the viewport.
       *
       * @type {number}
       */
      get firstVisibleIndex() {
        var idx = this._firstVisibleIndexVal;
        if (idx == null) {
          var physicalOffset = this._physicalTop + this._scrollOffset;

          idx = this._iterateItems(function(pidx, vidx) {
            physicalOffset += this._getPhysicalSizeIncrement(pidx);

            if (physicalOffset > this._scrollPosition) {
              return this.grid ? vidx - (vidx % this._itemsPerRow) : vidx;
            }
            // Handle a partially rendered final row in grid mode
            if (this.grid && this._virtualCount - 1 === vidx) {
              return vidx - (vidx % this._itemsPerRow);
            }
          }) ||
              0;
          this._firstVisibleIndexVal = idx;
        }
        return idx;
      },

      /**
       * Gets the index of the last visible item in the viewport.
       *
       * @type {number}
       */
      get lastVisibleIndex() {
        var idx = this._lastVisibleIndexVal;
        if (idx == null) {
          if (this.grid) {
            idx = Math.min(
                this._virtualCount,
                this.firstVisibleIndex + this._estRowsInView * this._itemsPerRow -
                    1);
          } else {
            var physicalOffset = this._physicalTop + this._scrollOffset;
            this._iterateItems(function(pidx, vidx) {
              if (physicalOffset < this._scrollBottom) {
                idx = vidx;
              }
              physicalOffset += this._getPhysicalSizeIncrement(pidx);
            });
          }
          this._lastVisibleIndexVal = idx;
        }
        return idx;
      },

      get _defaultScrollTarget() {
        return this;
      },

      get _virtualRowCount() {
        return Math.ceil(this._virtualCount / this._itemsPerRow);
      },

      get _estRowsInView() {
        return Math.ceil(this._viewportHeight / this._rowHeight);
      },

      get _physicalRows() {
        return Math.ceil(this._physicalCount / this._itemsPerRow);
      },

      get _scrollOffset() {
        return this._scrollerPaddingTop + this.scrollOffset;
      },

      /** @override */
      ready: function() {
        this.addEventListener('focus', this._didFocus.bind(this), true);
      },

      /** @override */
      attached: function() {
        this._debounce('_render', this._render, animationFrame);
        // `iron-resize` is fired when the list is attached if the event is added
        // before attached causing unnecessary work.
        this.listen(this, 'iron-resize', '_resizeHandler');
        this.listen(this, 'keydown', '_keydownHandler');
      },

      /** @override */
      detached: function() {
        this.unlisten(this, 'iron-resize', '_resizeHandler');
        this.unlisten(this, 'keydown', '_keydownHandler');
      },

      /**
       * Set the overflow property if this element has its own scrolling region
       */
      _setOverflow: function(scrollTarget) {
        this.style.webkitOverflowScrolling = scrollTarget === this ? 'touch' : '';
        this.style.overflowY = scrollTarget === this ? 'auto' : '';
        // Clear cache.
        this._lastVisibleIndexVal = null;
        this._firstVisibleIndexVal = null;
        this._debounce('_render', this._render, animationFrame);
      },

      /**
       * Invoke this method if you dynamically update the viewport's
       * size or CSS padding.
       *
       * @method updateViewportBoundaries
       */
      updateViewportBoundaries: function() {
        var styles = window.getComputedStyle(this);
        this._scrollerPaddingTop =
            this.scrollTarget === this ? 0 : parseInt(styles['padding-top'], 10);
        this._isRTL = Boolean(styles.direction === 'rtl');
        this._viewportWidth = this.$.items.offsetWidth;
        this._viewportHeight = this._scrollTargetHeight;
        this.grid && this._updateGridMetrics();
      },

      /**
       * Recycles the physical items when needed.
       */
      _scrollHandler: function() {
        var scrollTop = Math.max(0, Math.min(this._maxScrollTop, this._scrollTop));
        var delta = scrollTop - this._scrollPosition;
        var isScrollingDown = delta >= 0;
        // Track the current scroll position.
        this._scrollPosition = scrollTop;
        // Clear indexes for first and last visible indexes.
        this._firstVisibleIndexVal = null;
        this._lastVisibleIndexVal = null;
        // Random access.
        if (Math.abs(delta) > this._physicalSize && this._physicalSize > 0) {
          delta = delta - this._scrollOffset;
          var idxAdjustment =
              Math.round(delta / this._physicalAverage) * this._itemsPerRow;
          this._virtualStart = this._virtualStart + idxAdjustment;
          this._physicalStart = this._physicalStart + idxAdjustment;
          // Estimate new physical offset based on the virtual start index.
          // adjusts the physical start position to stay in sync with the clamped
          // virtual start index. It's critical not to let this value be
          // more than the scroll position however, since that would result in
          // the physical items not covering the viewport, and leading to
          // _increasePoolIfNeeded to run away creating items to try to fill it.
          this._physicalTop = Math.min(
              Math.floor(this._virtualStart / this._itemsPerRow) *
                  this._physicalAverage,
              this._scrollPosition);
          this._update();
        } else if (this._physicalCount > 0) {
          var reusables = this._getReusables(isScrollingDown);
          if (isScrollingDown) {
            this._physicalTop = reusables.physicalTop;
            this._virtualStart = this._virtualStart + reusables.indexes.length;
            this._physicalStart = this._physicalStart + reusables.indexes.length;
          } else {
            this._virtualStart = this._virtualStart - reusables.indexes.length;
            this._physicalStart = this._physicalStart - reusables.indexes.length;
          }
          this._update(
              reusables.indexes, isScrollingDown ? null : reusables.indexes);
          this._debounce(
              '_increasePoolIfNeeded',
              this._increasePoolIfNeeded.bind(this, 0),
              microTask);
        }
      },

      /**
       * Returns an object that contains the indexes of the physical items
       * that might be reused and the physicalTop.
       *
       * @param {boolean} fromTop If the potential reusable items are above the scrolling region.
       */
      _getReusables: function(fromTop) {
        var ith, lastIth, offsetContent, physicalItemHeight;
        var idxs = [];
        var protectedOffsetContent = this._hiddenContentSize * this._ratio;
        var virtualStart = this._virtualStart;
        var virtualEnd = this._virtualEnd;
        var physicalCount = this._physicalCount;
        var top = this._physicalTop + this._scrollOffset;
        var bottom = this._physicalBottom + this._scrollOffset;
        // This may be called outside of a scrollHandler, so use last cached position
        var scrollTop = this._scrollPosition;
        var scrollBottom = this._scrollBottom;

        if (fromTop) {
          ith = this._physicalStart;
          lastIth = this._physicalEnd;
          offsetContent = scrollTop - top;
        } else {
          ith = this._physicalEnd;
          lastIth = this._physicalStart;
          offsetContent = bottom - scrollBottom;
        }
        while (true) {
          physicalItemHeight = this._getPhysicalSizeIncrement(ith);
          offsetContent = offsetContent - physicalItemHeight;
          if (idxs.length >= physicalCount ||
              offsetContent <= protectedOffsetContent) {
            break;
          }
          if (fromTop) {
            // Check that index is within the valid range.
            if (virtualEnd + idxs.length + 1 >= this._virtualCount) {
              break;
            }
            // Check that the index is not visible.
            if (top + physicalItemHeight >= scrollTop - this._scrollOffset) {
              break;
            }
            idxs.push(ith);
            top = top + physicalItemHeight;
            ith = (ith + 1) % physicalCount;
          } else {
            // Check that index is within the valid range.
            if (virtualStart - idxs.length <= 0) {
              break;
            }
            // Check that the index is not visible.
            if (top + this._physicalSize - physicalItemHeight <= scrollBottom) {
              break;
            }
            idxs.push(ith);
            top = top - physicalItemHeight;
            ith = (ith === 0) ? physicalCount - 1 : ith - 1;
          }
        }
        return {indexes: idxs, physicalTop: top - this._scrollOffset};
      },

      /**
       * Update the list of items, starting from the `_virtualStart` item.
       * @param {!Array<number>=} itemSet
       * @param {!Array<number>=} movingUp
       */
      _update: function(itemSet, movingUp) {
        if ((itemSet && itemSet.length === 0) || this._physicalCount === 0) {
          return;
        }
        this._manageFocus();
        this._assignModels(itemSet);
        this._updateMetrics(itemSet);
        // Adjust offset after measuring.
        if (movingUp) {
          while (movingUp.length) {
            var idx = movingUp.pop();
            this._physicalTop -= this._getPhysicalSizeIncrement(idx);
          }
        }
        this._positionItems();
        this._updateScrollerSize();
      },

      /**
       * Creates a pool of DOM elements and attaches them to the local dom.
       *
       * @param {number} size Size of the pool
       */
      _createPool: function(size) {
        this._ensureTemplatized();
        var i, inst;
        var physicalItems = new Array(size);
        for (i = 0; i < size; i++) {
          inst = this.stamp(null);
          // TODO(blasten):
          // First element child is item; Safari doesn't support children[0]
          // on a doc fragment. Test this to see if it still matters.
          physicalItems[i] = inst.root.querySelector('*');
          this._itemsParent.appendChild(inst.root);
        }
        return physicalItems;
      },

      _isClientFull: function() {
        return this._scrollBottom != 0 &&
            this._physicalBottom - 1 >= this._scrollBottom &&
            this._physicalTop <= this._scrollPosition;
      },

      /**
       * Increases the pool size.
       */
      _increasePoolIfNeeded: function(count) {
        var nextPhysicalCount = this._clamp(
            this._physicalCount + count,
            DEFAULT_PHYSICAL_COUNT,
            this._virtualCount - this._virtualStart);
        nextPhysicalCount = this._convertIndexToCompleteRow(nextPhysicalCount);
        if (this.grid) {
          var correction = nextPhysicalCount % this._itemsPerRow;
          if (correction && nextPhysicalCount - correction <= this._physicalCount) {
            nextPhysicalCount += this._itemsPerRow;
          }
          nextPhysicalCount -= correction;
        }
        var delta = nextPhysicalCount - this._physicalCount;
        var nextIncrease = Math.round(this._physicalCount * 0.5);

        if (delta < 0) {
          return;
        }
        if (delta > 0) {
          var ts = window.performance.now();
          // Concat arrays in place.
          [].push.apply(this._physicalItems, this._createPool(delta));
          // Push 0s into physicalSizes. Can't use Array.fill because IE11 doesn't
          // support it.
          for (var i = 0; i < delta; i++) {
            this._physicalSizes.push(0);
          }
          this._physicalCount = this._physicalCount + delta;
          // Update the physical start if it needs to preserve the model of the
          // focused item. In this situation, the focused item is currently rendered
          // and its model would have changed after increasing the pool if the
          // physical start remained unchanged.
          if (this._physicalStart > this._physicalEnd &&
              this._isIndexRendered(this._focusedVirtualIndex) &&
              this._getPhysicalIndex(this._focusedVirtualIndex) <
                  this._physicalEnd) {
            this._physicalStart = this._physicalStart + delta;
          }
          this._update();
          this._templateCost = (window.performance.now() - ts) / delta;
          nextIncrease = Math.round(this._physicalCount * 0.5);
        }
        // The upper bounds is not fixed when dealing with a grid that doesn't
        // fill it's last row with the exact number of items per row.
        if (this._virtualEnd >= this._virtualCount - 1 || nextIncrease === 0) ; else if (!this._isClientFull()) {
          this._debounce(
              '_increasePoolIfNeeded',
              this._increasePoolIfNeeded.bind(this, nextIncrease),
              microTask);
        } else if (this._physicalSize < this._optPhysicalSize) {
          // Yield and increase the pool during idle time until the physical size is
          // optimal.
          this._debounce(
              '_increasePoolIfNeeded',
              this._increasePoolIfNeeded.bind(
                  this,
                  this._clamp(
                      Math.round(50 / this._templateCost), 1, nextIncrease)),
              idlePeriod);
        }
      },

      /**
       * Renders the a new list.
       */
      _render: function() {
        if (!this.isAttached || !this._isVisible) {
          return;
        }
        if (this._physicalCount !== 0) {
          var reusables = this._getReusables(true);
          this._physicalTop = reusables.physicalTop;
          this._virtualStart = this._virtualStart + reusables.indexes.length;
          this._physicalStart = this._physicalStart + reusables.indexes.length;
          this._update(reusables.indexes);
          this._update();
          this._increasePoolIfNeeded(0);
        } else if (this._virtualCount > 0) {
          // Initial render
          this.updateViewportBoundaries();
          this._increasePoolIfNeeded(DEFAULT_PHYSICAL_COUNT);
        }
      },

      /**
       * Templetizes the user template.
       */
      _ensureTemplatized: function() {
        if (this.ctor) {
          return;
        }
        this._userTemplate = /** @type {!HTMLTemplateElement} */ (
            this.queryEffectiveChildren('template'));
        if (!this._userTemplate) {
          console.warn('iron-list requires a template to be provided in light-dom');
        }
        var instanceProps = {};
        instanceProps.__key__ = true;
        instanceProps[this.as] = true;
        instanceProps[this.indexAs] = true;
        instanceProps[this.selectedAs] = true;
        instanceProps.tabIndex = true;
        this._instanceProps = instanceProps;
        this.templatize(this._userTemplate, this.mutableData);
      },

      _gridChanged: function(newGrid, oldGrid) {
        if (typeof oldGrid === 'undefined')
          return;
        this.notifyResize();
        flush();
        newGrid && this._updateGridMetrics();
      },

      /**
       * Called when the items have changed. That is, reassignments
       * to `items`, splices or updates to a single item.
       */
      _itemsChanged: function(change) {
        if (change.path === 'items') {
          this._virtualStart = 0;
          this._physicalTop = 0;
          this._virtualCount = this.items ? this.items.length : 0;
          this._physicalIndexForKey = {};
          this._firstVisibleIndexVal = null;
          this._lastVisibleIndexVal = null;
          this._physicalCount = this._physicalCount || 0;
          this._physicalItems = this._physicalItems || [];
          this._physicalSizes = this._physicalSizes || [];
          this._physicalStart = 0;
          if (this._scrollTop > this._scrollOffset) {
            this._resetScrollPosition(0);
          }
          this._removeFocusedItem();
          this._debounce('_render', this._render, animationFrame);
        } else if (change.path === 'items.splices') {
          this._adjustVirtualIndex(change.value.indexSplices);
          this._virtualCount = this.items ? this.items.length : 0;
          // Only blur if at least one item is added or removed.
          var itemAddedOrRemoved = change.value.indexSplices.some(function(splice) {
            return splice.addedCount > 0 || splice.removed.length > 0;
          });
          if (itemAddedOrRemoved) {
            // Only blur activeElement if it is a descendant of the list (#505,
            // #507).
            var activeElement = this._getActiveElement();
            if (this.contains(activeElement)) {
              activeElement.blur();
            }
          }
          // Render only if the affected index is rendered.
          var affectedIndexRendered =
              change.value.indexSplices.some(function(splice) {
                return splice.index + splice.addedCount >= this._virtualStart &&
                    splice.index <= this._virtualEnd;
              }, this);
          if (!this._isClientFull() || affectedIndexRendered) {
            this._debounce('_render', this._render, animationFrame);
          }
        } else if (change.path !== 'items.length') {
          this._forwardItemPath(change.path, change.value);
        }
      },

      _forwardItemPath: function(path, value) {
        path = path.slice(6);  // 'items.'.length == 6
        var dot = path.indexOf('.');
        if (dot === -1) {
          dot = path.length;
        }
        var isIndexRendered;
        var pidx;
        var inst;
        var offscreenInstance = this.modelForElement(this._offscreenFocusedItem);
        var vidx = parseInt(path.substring(0, dot), 10);
        isIndexRendered = this._isIndexRendered(vidx);
        if (isIndexRendered) {
          pidx = this._getPhysicalIndex(vidx);
          inst = this.modelForElement(this._physicalItems[pidx]);
        } else if (offscreenInstance) {
          inst = offscreenInstance;
        }

        if (!inst || inst[this.indexAs] !== vidx) {
          return;
        }
        path = path.substring(dot + 1);
        path = this.as + (path ? '.' + path : '');
        inst._setPendingPropertyOrPath(path, value, false, true);
        inst._flushProperties && inst._flushProperties();
        // TODO(blasten): V1 doesn't do this and it's a bug
        if (isIndexRendered) {
          this._updateMetrics([pidx]);
          this._positionItems();
          this._updateScrollerSize();
        }
      },

      /**
       * @param {!Array<!Object>} splices
       */
      _adjustVirtualIndex: function(splices) {
        splices.forEach(function(splice) {
          // deselect removed items
          splice.removed.forEach(this._removeItem, this);
          // We only need to care about changes happening above the current position
          if (splice.index < this._virtualStart) {
            var delta = Math.max(
                splice.addedCount - splice.removed.length,
                splice.index - this._virtualStart);
            this._virtualStart = this._virtualStart + delta;
            if (this._focusedVirtualIndex >= 0) {
              this._focusedVirtualIndex = this._focusedVirtualIndex + delta;
            }
          }
        }, this);
      },

      _removeItem: function(item) {
        this.$.selector.deselect(item);
        // remove the current focused item
        if (this._focusedItem &&
            this.modelForElement(this._focusedItem)[this.as] === item) {
          this._removeFocusedItem();
        }
      },

      /**
       * Executes a provided function per every physical index in `itemSet`
       * `itemSet` default value is equivalent to the entire set of physical
       * indexes.
       *
       * @param {!function(number, number)} fn
       * @param {!Array<number>=} itemSet
       */
      _iterateItems: function(fn, itemSet) {
        var pidx, vidx, rtn, i;

        if (arguments.length === 2 && itemSet) {
          for (i = 0; i < itemSet.length; i++) {
            pidx = itemSet[i];
            vidx = this._computeVidx(pidx);
            if ((rtn = fn.call(this, pidx, vidx)) != null) {
              return rtn;
            }
          }
        } else {
          pidx = this._physicalStart;
          vidx = this._virtualStart;
          for (; pidx < this._physicalCount; pidx++, vidx++) {
            if ((rtn = fn.call(this, pidx, vidx)) != null) {
              return rtn;
            }
          }
          for (pidx = 0; pidx < this._physicalStart; pidx++, vidx++) {
            if ((rtn = fn.call(this, pidx, vidx)) != null) {
              return rtn;
            }
          }
        }
      },

      /**
       * Returns the virtual index for a given physical index
       *
       * @param {number} pidx Physical index
       * @return {number}
       */
      _computeVidx: function(pidx) {
        if (pidx >= this._physicalStart) {
          return this._virtualStart + (pidx - this._physicalStart);
        }
        return this._virtualStart + (this._physicalCount - this._physicalStart) +
            pidx;
      },

      /**
       * Assigns the data models to a given set of items.
       * @param {!Array<number>=} itemSet
       */
      _assignModels: function(itemSet) {
        this._iterateItems(function(pidx, vidx) {
          var el = this._physicalItems[pidx];
          var item = this.items && this.items[vidx];
          if (item != null) {
            var inst = this.modelForElement(el);
            inst.__key__ = null;
            this._forwardProperty(inst, this.as, item);
            this._forwardProperty(
                inst, this.selectedAs, this.$.selector.isSelected(item));
            this._forwardProperty(inst, this.indexAs, vidx);
            this._forwardProperty(
                inst, 'tabIndex', this._focusedVirtualIndex === vidx ? 0 : -1);
            this._physicalIndexForKey[inst.__key__] = pidx;
            inst._flushProperties && inst._flushProperties(true);
            el.removeAttribute('hidden');
          } else {
            el.setAttribute('hidden', '');
          }
        }, itemSet);
      },

      /**
       * Updates the height for a given set of items.
       *
       * @param {!Array<number>=} itemSet
       */
      _updateMetrics: function(itemSet) {
        // Make sure we distributed all the physical items
        // so we can measure them.
        flush();

        var newPhysicalSize = 0;
        var oldPhysicalSize = 0;
        var prevAvgCount = this._physicalAverageCount;
        var prevPhysicalAvg = this._physicalAverage;

        this._iterateItems(function(pidx, vidx) {
          oldPhysicalSize += this._physicalSizes[pidx];
          this._physicalSizes[pidx] = this._physicalItems[pidx].offsetHeight;
          newPhysicalSize += this._physicalSizes[pidx];
          this._physicalAverageCount += this._physicalSizes[pidx] ? 1 : 0;
        }, itemSet);

        if (this.grid) {
          this._updateGridMetrics();
          this._physicalSize =
              Math.ceil(this._physicalCount / this._itemsPerRow) * this._rowHeight;
        } else {
          oldPhysicalSize = (this._itemsPerRow === 1) ?
              oldPhysicalSize :
              Math.ceil(this._physicalCount / this._itemsPerRow) * this._rowHeight;
          this._physicalSize =
              this._physicalSize + newPhysicalSize - oldPhysicalSize;
          this._itemsPerRow = 1;
        }
        // Update the average if it measured something.
        if (this._physicalAverageCount !== prevAvgCount) {
          this._physicalAverage = Math.round(
              ((prevPhysicalAvg * prevAvgCount) + newPhysicalSize) /
              this._physicalAverageCount);
        }
      },

      _updateGridMetrics: function() {
        this._itemWidth = this._physicalCount > 0 ?
            this._physicalItems[0].getBoundingClientRect().width :
            200;
        this._rowHeight =
            this._physicalCount > 0 ? this._physicalItems[0].offsetHeight : 200;
        this._itemsPerRow = this._itemWidth ?
            Math.floor(this._viewportWidth / this._itemWidth) :
            this._itemsPerRow;
      },

      /**
       * Updates the position of the physical items.
       */
      _positionItems: function() {
        this._adjustScrollPosition();

        var y = this._physicalTop;

        if (this.grid) {
          var totalItemWidth = this._itemsPerRow * this._itemWidth;
          var rowOffset = (this._viewportWidth - totalItemWidth) / 2;

          this._iterateItems(function(pidx, vidx) {
            var modulus = vidx % this._itemsPerRow;
            var x = Math.floor((modulus * this._itemWidth) + rowOffset);
            if (this._isRTL) {
              x = x * -1;
            }
            this.translate3d(x + 'px', y + 'px', 0, this._physicalItems[pidx]);
            if (this._shouldRenderNextRow(vidx)) {
              y += this._rowHeight;
            }
          });
        } else {
          const order = [];
          this._iterateItems(function(pidx, vidx) {
            const item = this._physicalItems[pidx];
            this.translate3d(0, y + 'px', 0, item);
            y += this._physicalSizes[pidx];
            const itemId = item.id;
            if (itemId) {
              order.push(itemId);
            }
          });
          if (order.length) {
            this.setAttribute('aria-owns', order.join(' '));
          }
        }
      },

      _getPhysicalSizeIncrement: function(pidx) {
        if (!this.grid) {
          return this._physicalSizes[pidx];
        }
        if (this._computeVidx(pidx) % this._itemsPerRow !== this._itemsPerRow - 1) {
          return 0;
        }
        return this._rowHeight;
      },

      /**
       * Returns, based on the current index,
       * whether or not the next index will need
       * to be rendered on a new row.
       *
       * @param {number} vidx Virtual index
       * @return {boolean}
       */
      _shouldRenderNextRow: function(vidx) {
        return vidx % this._itemsPerRow === this._itemsPerRow - 1;
      },

      /**
       * Adjusts the scroll position when it was overestimated.
       */
      _adjustScrollPosition: function() {
        var deltaHeight = this._virtualStart === 0 ?
            this._physicalTop :
            Math.min(this._scrollPosition + this._physicalTop, 0);
        // Note: the delta can be positive or negative.
        if (deltaHeight !== 0) {
          this._physicalTop = this._physicalTop - deltaHeight;
          // This may be called outside of a scrollHandler, so use last cached position
          var scrollTop = this._scrollPosition;
          // juking scroll position during interial scrolling on iOS is no bueno
          if (!IOS_TOUCH_SCROLLING && scrollTop > 0) {
            this._resetScrollPosition(scrollTop - deltaHeight);
          }
        }
      },

      /**
       * Sets the position of the scroll.
       */
      _resetScrollPosition: function(pos) {
        if (this.scrollTarget && pos >= 0) {
          this._scrollTop = pos;
          this._scrollPosition = this._scrollTop;
        }
      },

      /**
       * Sets the scroll height, that's the height of the content,
       *
       * @param {boolean=} forceUpdate If true, updates the height no matter what.
       */
      _updateScrollerSize: function(forceUpdate) {
        if (this.grid) {
          this._estScrollHeight = this._virtualRowCount * this._rowHeight;
        } else {
          this._estScrollHeight =
              (this._physicalBottom +
               Math.max(
                   this._virtualCount - this._physicalCount - this._virtualStart,
                   0) *
                   this._physicalAverage);
        }
        forceUpdate = forceUpdate || this._scrollHeight === 0;
        forceUpdate = forceUpdate ||
            this._scrollPosition >= this._estScrollHeight - this._physicalSize;
        forceUpdate = forceUpdate ||
            this.grid && this.$.items.style.height < this._estScrollHeight;
        // Amortize height adjustment, so it won't trigger large repaints too often.
        if (forceUpdate ||
            Math.abs(this._estScrollHeight - this._scrollHeight) >=
                this._viewportHeight) {
          this.$.items.style.height = this._estScrollHeight + 'px';
          this._scrollHeight = this._estScrollHeight;
        }
      },

      /**
       * Scroll to a specific item in the virtual list regardless
       * of the physical items in the DOM tree.
       *
       * @method scrollToItem
       * @param {(Object)} item The item to be scrolled to
       */
      scrollToItem: function(item) {
        return this.scrollToIndex(this.items.indexOf(item));
      },

      /**
       * Scroll to a specific index in the virtual list regardless
       * of the physical items in the DOM tree.
       *
       * @method scrollToIndex
       * @param {number} idx The index of the item
       */
      scrollToIndex: function(idx) {
        if (typeof idx !== 'number' || idx < 0 || idx > this.items.length - 1) {
          return;
        }
        flush();
        // Items should have been rendered prior scrolling to an index.
        if (this._physicalCount === 0) {
          return;
        }
        idx = this._clamp(idx, 0, this._virtualCount - 1);
        // Update the virtual start only when needed.
        if (!this._isIndexRendered(idx) || idx >= this._maxVirtualStart) {
          this._virtualStart =
              this.grid ? (idx - this._itemsPerRow * 2) : (idx - 1);
        }
        this._manageFocus();
        this._assignModels();
        this._updateMetrics();
        // Estimate new physical offset.
        this._physicalTop = Math.floor(this._virtualStart / this._itemsPerRow) *
            this._physicalAverage;

        var currentTopItem = this._physicalStart;
        var currentVirtualItem = this._virtualStart;
        var targetOffsetTop = 0;
        var hiddenContentSize = this._hiddenContentSize;
        // scroll to the item as much as we can.
        while (currentVirtualItem < idx && targetOffsetTop <= hiddenContentSize) {
          targetOffsetTop =
              targetOffsetTop + this._getPhysicalSizeIncrement(currentTopItem);
          currentTopItem = (currentTopItem + 1) % this._physicalCount;
          currentVirtualItem++;
        }
        this._updateScrollerSize(true);
        this._positionItems();
        this._resetScrollPosition(
            this._physicalTop + this._scrollOffset + targetOffsetTop);
        this._increasePoolIfNeeded(0);
        // clear cached visible index.
        this._firstVisibleIndexVal = null;
        this._lastVisibleIndexVal = null;
      },

      /**
       * Reset the physical average and the average count.
       */
      _resetAverage: function() {
        this._physicalAverage = 0;
        this._physicalAverageCount = 0;
      },

      /**
       * A handler for the `iron-resize` event triggered by `IronResizableBehavior`
       * when the element is resized.
       */
      _resizeHandler: function() {
        this._debounce('_render', function() {
          // clear cached visible index.
          this._firstVisibleIndexVal = null;
          this._lastVisibleIndexVal = null;
          if (this._isVisible) {
            this.updateViewportBoundaries();
            // Reinstall the scroll event listener.
            this.toggleScrollListener(true);
            this._resetAverage();
            this._render();
          } else {
            // Uninstall the scroll event listener.
            this.toggleScrollListener(false);
          }
        }, animationFrame);
      },

      /**
       * Selects the given item.
       *
       * @method selectItem
       * @param {Object} item The item instance.
       */
      selectItem: function(item) {
        return this.selectIndex(this.items.indexOf(item));
      },

      /**
       * Selects the item at the given index in the items array.
       *
       * @method selectIndex
       * @param {number} index The index of the item in the items array.
       */
      selectIndex: function(index) {
        if (index < 0 || index >= this._virtualCount) {
          return;
        }
        if (!this.multiSelection && this.selectedItem) {
          this.clearSelection();
        }
        if (this._isIndexRendered(index)) {
          var model = this.modelForElement(
              this._physicalItems[this._getPhysicalIndex(index)]);
          if (model) {
            model[this.selectedAs] = true;
          }
          this.updateSizeForIndex(index);
        }
        this.$.selector.selectIndex(index);
      },

      /**
       * Deselects the given item.
       *
       * @method deselect
       * @param {Object} item The item instance.
       */
      deselectItem: function(item) {
        return this.deselectIndex(this.items.indexOf(item));
      },

      /**
       * Deselects the item at the given index in the items array.
       *
       * @method deselectIndex
       * @param {number} index The index of the item in the items array.
       */
      deselectIndex: function(index) {
        if (index < 0 || index >= this._virtualCount) {
          return;
        }
        if (this._isIndexRendered(index)) {
          var model = this.modelForElement(
              this._physicalItems[this._getPhysicalIndex(index)]);
          model[this.selectedAs] = false;
          this.updateSizeForIndex(index);
        }
        this.$.selector.deselectIndex(index);
      },

      /**
       * Selects or deselects a given item depending on whether the item
       * has already been selected.
       *
       * @method toggleSelectionForItem
       * @param {Object} item The item object.
       */
      toggleSelectionForItem: function(item) {
        return this.toggleSelectionForIndex(this.items.indexOf(item));
      },

      /**
       * Selects or deselects the item at the given index in the items array
       * depending on whether the item has already been selected.
       *
       * @method toggleSelectionForIndex
       * @param {number} index The index of the item in the items array.
       */
      toggleSelectionForIndex: function(index) {
        var isSelected = this.$.selector.isIndexSelected ?
            this.$.selector.isIndexSelected(index) :
            this.$.selector.isSelected(this.items[index]);
        isSelected ? this.deselectIndex(index) : this.selectIndex(index);
      },

      /**
       * Clears the current selection in the list.
       *
       * @method clearSelection
       */
      clearSelection: function() {
        this._iterateItems(function(pidx, vidx) {
          this.modelForElement(this._physicalItems[pidx])[this.selectedAs] = false;
        });
        this.$.selector.clearSelection();
      },

      /**
       * Add an event listener to `tap` if `selectionEnabled` is true,
       * it will remove the listener otherwise.
       */
      _selectionEnabledChanged: function(selectionEnabled) {
        var handler = selectionEnabled ? this.listen : this.unlisten;
        handler.call(this, this, 'tap', '_selectionHandler');
      },

      /**
       * Select an item from an event object.
       */
      _selectionHandler: function(e) {
        var model = this.modelForElement(e.target);
        if (!model) {
          return;
        }
        var modelTabIndex, activeElTabIndex;
        var target = dom(e).path[0];
        var activeEl = this._getActiveElement();
        var physicalItem =
            this._physicalItems[this._getPhysicalIndex(model[this.indexAs])];
        // Safari does not focus certain form controls via mouse
        // https://bugs.webkit.org/show_bug.cgi?id=118043
        if (target.localName === 'input' || target.localName === 'button' ||
            target.localName === 'select') {
          return;
        }
        // Set a temporary tabindex
        modelTabIndex = model.tabIndex;
        model.tabIndex = SECRET_TABINDEX;
        activeElTabIndex = activeEl ? activeEl.tabIndex : -1;
        model.tabIndex = modelTabIndex;
        // Only select the item if the tap wasn't on a focusable child
        // or the element bound to `tabIndex`
        if (activeEl && physicalItem !== activeEl &&
            physicalItem.contains(activeEl) &&
            activeElTabIndex !== SECRET_TABINDEX) {
          return;
        }
        this.toggleSelectionForItem(model[this.as]);
      },

      _multiSelectionChanged: function(multiSelection) {
        this.clearSelection();
        this.$.selector.multi = multiSelection;
      },

      /**
       * Updates the size of a given list item.
       *
       * @method updateSizeForItem
       * @param {Object} item The item instance.
       */
      updateSizeForItem: function(item) {
        return this.updateSizeForIndex(this.items.indexOf(item));
      },

      /**
       * Updates the size of the item at the given index in the items array.
       *
       * @method updateSizeForIndex
       * @param {number} index The index of the item in the items array.
       */
      updateSizeForIndex: function(index) {
        if (!this._isIndexRendered(index)) {
          return null;
        }
        this._updateMetrics([this._getPhysicalIndex(index)]);
        this._positionItems();
        return null;
      },

      /**
       * Creates a temporary backfill item in the rendered pool of physical items
       * to replace the main focused item. The focused item has tabIndex = 0
       * and might be currently focused by the user.
       *
       * This dynamic replacement helps to preserve the focus state.
       */
      _manageFocus: function() {
        var fidx = this._focusedVirtualIndex;

        if (fidx >= 0 && fidx < this._virtualCount) {
          // if it's a valid index, check if that index is rendered
          // in a physical item.
          if (this._isIndexRendered(fidx)) {
            this._restoreFocusedItem();
          } else {
            this._createFocusBackfillItem();
          }
        } else if (this._virtualCount > 0 && this._physicalCount > 0) {
          // otherwise, assign the initial focused index.
          this._focusedPhysicalIndex = this._physicalStart;
          this._focusedVirtualIndex = this._virtualStart;
          this._focusedItem = this._physicalItems[this._physicalStart];
        }
      },

      /**
       * Converts a random index to the index of the item that completes it's row.
       * Allows for better order and fill computation when grid == true.
       */
      _convertIndexToCompleteRow: function(idx) {
        // when grid == false _itemPerRow can be unset.
        this._itemsPerRow = this._itemsPerRow || 1;
        return this.grid ? Math.ceil(idx / this._itemsPerRow) * this._itemsPerRow :
                           idx;
      },

      _isIndexRendered: function(idx) {
        return idx >= this._virtualStart && idx <= this._virtualEnd;
      },

      _isIndexVisible: function(idx) {
        return idx >= this.firstVisibleIndex && idx <= this.lastVisibleIndex;
      },

      _getPhysicalIndex: function(vidx) {
        return (this._physicalStart + (vidx - this._virtualStart)) %
            this._physicalCount;
      },

      focusItem: function(idx) {
        this._focusPhysicalItem(idx);
      },

      _focusPhysicalItem: function(idx) {
        if (idx < 0 || idx >= this._virtualCount) {
          return;
        }
        this._restoreFocusedItem();
        // scroll to index to make sure it's rendered
        if (!this._isIndexRendered(idx)) {
          this.scrollToIndex(idx);
        }
        var physicalItem = this._physicalItems[this._getPhysicalIndex(idx)];
        var model = this.modelForElement(physicalItem);
        var focusable;
        // set a secret tab index
        model.tabIndex = SECRET_TABINDEX;
        // check if focusable element is the physical item
        if (physicalItem.tabIndex === SECRET_TABINDEX) {
          focusable = physicalItem;
        }
        // search for the element which tabindex is bound to the secret tab index
        if (!focusable) {
          focusable = dom(physicalItem)
                          .querySelector('[tabindex="' + SECRET_TABINDEX + '"]');
        }
        // restore the tab index
        model.tabIndex = 0;
        // focus the focusable element
        this._focusedVirtualIndex = idx;
        focusable && focusable.focus();
      },

      _removeFocusedItem: function() {
        if (this._offscreenFocusedItem) {
          this._itemsParent.removeChild(this._offscreenFocusedItem);
        }
        this._offscreenFocusedItem = null;
        this._focusBackfillItem = null;
        this._focusedItem = null;
        this._focusedVirtualIndex = -1;
        this._focusedPhysicalIndex = -1;
      },

      _createFocusBackfillItem: function() {
        var fpidx = this._focusedPhysicalIndex;

        if (this._offscreenFocusedItem || this._focusedVirtualIndex < 0) {
          return;
        }
        if (!this._focusBackfillItem) {
          // Create a physical item.
          var inst = this.stamp(null);
          this._focusBackfillItem =
              /** @type {!HTMLElement} */ (inst.root.querySelector('*'));
          this._itemsParent.appendChild(inst.root);
        }
        // Set the offcreen focused physical item.
        this._offscreenFocusedItem = this._physicalItems[fpidx];
        this.modelForElement(this._offscreenFocusedItem).tabIndex = 0;
        this._physicalItems[fpidx] = this._focusBackfillItem;
        this._focusedPhysicalIndex = fpidx;
        // Hide the focused physical.
        this.translate3d(0, HIDDEN_Y, 0, this._offscreenFocusedItem);
      },

      _restoreFocusedItem: function() {
        if (!this._offscreenFocusedItem || this._focusedVirtualIndex < 0) {
          return;
        }
        // Assign models to the focused index.
        this._assignModels();
        // Get the new physical index for the focused index.
        var fpidx = this._focusedPhysicalIndex =
            this._getPhysicalIndex(this._focusedVirtualIndex);

        var onScreenItem = this._physicalItems[fpidx];
        if (!onScreenItem) {
          return;
        }
        var onScreenInstance = this.modelForElement(onScreenItem);
        var offScreenInstance = this.modelForElement(this._offscreenFocusedItem);
        // Restores the physical item only when it has the same model
        // as the offscreen one. Use key for comparison since users can set
        // a new item via set('items.idx').
        if (onScreenInstance[this.as] === offScreenInstance[this.as]) {
          // Flip the focus backfill.
          this._focusBackfillItem = onScreenItem;
          onScreenInstance.tabIndex = -1;
          // Restore the focused physical item.
          this._physicalItems[fpidx] = this._offscreenFocusedItem;
          // Hide the physical item that backfills.
          this.translate3d(0, HIDDEN_Y, 0, this._focusBackfillItem);
        } else {
          this._removeFocusedItem();
          this._focusBackfillItem = null;
        }
        this._offscreenFocusedItem = null;
      },

      _didFocus: function(e) {
        var targetModel = this.modelForElement(e.target);
        var focusedModel = this.modelForElement(this._focusedItem);
        var hasOffscreenFocusedItem = this._offscreenFocusedItem !== null;
        var fidx = this._focusedVirtualIndex;
        if (!targetModel) {
          return;
        }
        if (focusedModel === targetModel) {
          // If the user focused the same item, then bring it into view if it's not
          // visible.
          if (!this._isIndexVisible(fidx)) {
            this.scrollToIndex(fidx);
          }
        } else {
          this._restoreFocusedItem();
          // Restore tabIndex for the currently focused item.
          if (focusedModel) {
            focusedModel.tabIndex = -1;
          }
          // Set the tabIndex for the next focused item.
          targetModel.tabIndex = 0;
          fidx = targetModel[this.indexAs];
          this._focusedVirtualIndex = fidx;
          this._focusedPhysicalIndex = this._getPhysicalIndex(fidx);
          this._focusedItem = this._physicalItems[this._focusedPhysicalIndex];
          if (hasOffscreenFocusedItem && !this._offscreenFocusedItem) {
            this._update();
          }
        }
      },

      _keydownHandler: function(e) {
        switch (e.keyCode) {
          case /* ARROW_DOWN */ 40:
            if (this._focusedVirtualIndex < this._virtualCount - 1)
              e.preventDefault();
            this._focusPhysicalItem(
                this._focusedVirtualIndex + (this.grid ? this._itemsPerRow : 1));
            break;
          case /* ARROW_RIGHT */ 39:
            if (this.grid)
              this._focusPhysicalItem(
                  this._focusedVirtualIndex + (this._isRTL ? -1 : 1));
            break;
          case /* ARROW_UP */ 38:
            if (this._focusedVirtualIndex > 0)
              e.preventDefault();
            this._focusPhysicalItem(
                this._focusedVirtualIndex - (this.grid ? this._itemsPerRow : 1));
            break;
          case /* ARROW_LEFT */ 37:
            if (this.grid)
              this._focusPhysicalItem(
                  this._focusedVirtualIndex + (this._isRTL ? 1 : -1));
            break;
          case /* ENTER */ 13:
            this._focusPhysicalItem(this._focusedVirtualIndex);
            if (this.selectionEnabled)
              this._selectionHandler(e);
            break;
        }
      },

      _clamp: function(v, min, max) {
        return Math.min(max, Math.max(min, v));
      },

      _debounce: function(name, cb, asyncModule) {
        this._debouncers = this._debouncers || {};
        this._debouncers[name] =
            Debouncer.debounce(this._debouncers[name], asyncModule, cb.bind(this));
        enqueueDebouncer(this._debouncers[name]);
      },

      _forwardProperty: function(inst, name, value) {
        inst._setPendingProperty(name, value);
      },

      /* Templatizer bindings for v2 */
      _forwardHostPropV2: function(prop, value) {
        (this._physicalItems || [])
            .concat([this._offscreenFocusedItem, this._focusBackfillItem])
            .forEach(function(item) {
              if (item) {
                this.modelForElement(item).forwardHostProp(prop, value);
              }
            }, this);
      },

      _notifyInstancePropV2: function(inst, prop, value) {
        if (matches(this.as, prop)) {
          var idx = inst[this.indexAs];
          if (prop == this.as) {
            this.items[idx] = value;
          }
          this.notifyPath(translate(this.as, 'items.' + idx, prop), value);
        }
      },

      /* Templatizer bindings for v1 */
      _getStampedChildren: function() {
        return this._physicalItems;
      },

      _forwardInstancePath: function(inst, path, value) {
        if (path.indexOf(this.as + '.') === 0) {
          this.notifyPath(
              'items.' + inst.__key__ + '.' + path.slice(this.as.length + 1),
              value);
        }
      },

      _forwardParentPath: function(path, value) {
        (this._physicalItems || [])
            .concat([this._offscreenFocusedItem, this._focusBackfillItem])
            .forEach(function(item) {
              if (item) {
                this.modelForElement(item).notifyPath(path, value);
              }
            }, this);
      },

      _forwardParentProp: function(prop, value) {
        (this._physicalItems || [])
            .concat([this._offscreenFocusedItem, this._focusBackfillItem])
            .forEach(function(item) {
              if (item) {
                this.modelForElement(item)[prop] = value;
              }
            }, this);
      },

      /* Gets the activeElement of the shadow root/host that contains the list. */
      _getActiveElement: function() {
        var itemsHost = this._itemsParent.node.domHost;
        return dom(itemsHost ? itemsHost.root : document).activeElement;
      }
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    class IronSelection {
      /**
       * @param {!Function} selectCallback
       * @suppress {missingProvide}
       */
      constructor(selectCallback) {
        this.selection = [];
        this.selectCallback = selectCallback;
      }

      /**
       * Retrieves the selected item(s).
       *
       * @returns Returns the selected item(s). If the multi property is true,
       * `get` will return an array, otherwise it will return
       * the selected item or undefined if there is no selection.
       */
      get() {
        return this.multi ? this.selection.slice() : this.selection[0];
      }

      /**
       * Clears all the selection except the ones indicated.
       *
       * @param {Array} excludes items to be excluded.
       */
      clear(excludes) {
        this.selection.slice().forEach(function(item) {
          if (!excludes || excludes.indexOf(item) < 0) {
            this.setItemSelected(item, false);
          }
        }, this);
      }

      /**
       * Indicates if a given item is selected.
       *
       * @param {*} item The item whose selection state should be checked.
       * @return {boolean} Returns true if `item` is selected.
       */
      isSelected(item) {
        return this.selection.indexOf(item) >= 0;
      }

      /**
       * Sets the selection state for a given item to either selected or deselected.
       *
       * @param {*} item The item to select.
       * @param {boolean} isSelected True for selected, false for deselected.
       */
      setItemSelected(item, isSelected) {
        if (item != null) {
          if (isSelected !== this.isSelected(item)) {
            // proceed to update selection only if requested state differs from
            // current
            if (isSelected) {
              this.selection.push(item);
            } else {
              var i = this.selection.indexOf(item);
              if (i >= 0) {
                this.selection.splice(i, 1);
              }
            }
            if (this.selectCallback) {
              this.selectCallback(item, isSelected);
            }
          }
        }
      }

      /**
       * Sets the selection state for a given item. If the `multi` property
       * is true, then the selected state of `item` will be toggled; otherwise
       * the `item` will be selected.
       *
       * @param {*} item The item to select.
       */
      select(item) {
        if (this.multi) {
          this.toggle(item);
        } else if (this.get() !== item) {
          this.setItemSelected(this.get(), false);
          this.setItemSelected(item, true);
        }
      }

      /**
       * Toggles the selection state for `item`.
       *
       * @param {*} item The item to toggle.
       */
      toggle(item) {
        this.setItemSelected(item, !this.isSelected(item));
      }
    }

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
     * @polymerBehavior
     */
    const IronSelectableBehavior = {

      /**
       * Fired when iron-selector is activated (selected or deselected).
       * It is fired before the selected items are changed.
       * Cancel the event to abort selection.
       *
       * @event iron-activate
       */

      /**
       * Fired when an item is selected
       *
       * @event iron-select
       */

      /**
       * Fired when an item is deselected
       *
       * @event iron-deselect
       */

      /**
       * Fired when the list of selectable items changes (e.g., items are
       * added or removed). The detail of the event is a mutation record that
       * describes what changed.
       *
       * @event iron-items-changed
       */

      properties: {

        /**
         * If you want to use an attribute value or property of an element for
         * `selected` instead of the index, set this to the name of the attribute
         * or property. Hyphenated values are converted to camel case when used to
         * look up the property of a selectable element. Camel cased values are
         * *not* converted to hyphenated values for attribute lookup. It's
         * recommended that you provide the hyphenated form of the name so that
         * selection works in both cases. (Use `attr-or-property-name` instead of
         * `attrOrPropertyName`.)
         */
        attrForSelected: {type: String, value: null},

        /**
         * Gets or sets the selected element. The default is to use the index of the
         * item.
         * @type {string|number}
         */
        selected: {type: String, notify: true},

        /**
         * Returns the currently selected item.
         *
         * @type {?Object}
         */
        selectedItem: {type: Object, readOnly: true, notify: true},

        /**
         * The event that fires from items when they are selected. Selectable
         * will listen for this event from items and update the selection state.
         * Set to empty string to listen to no events.
         */
        activateEvent:
            {type: String, value: 'tap', observer: '_activateEventChanged'},

        /**
         * This is a CSS selector string.  If this is set, only items that match the
         * CSS selector are selectable.
         */
        selectable: String,

        /**
         * The class to set on elements when selected.
         */
        selectedClass: {type: String, value: 'iron-selected'},

        /**
         * The attribute to set on elements when selected.
         */
        selectedAttribute: {type: String, value: null},

        /**
         * Default fallback if the selection based on selected with
         * `attrForSelected` is not found.
         */
        fallbackSelection: {type: String, value: null},

        /**
         * The list of items from which a selection can be made.
         */
        items: {
          type: Array,
          readOnly: true,
          notify: true,
          value: function() {
            return [];
          }
        },

        /**
         * The set of excluded elements where the key is the `localName`
         * of the element that will be ignored from the item list.
         *
         * @default {template: 1}
         */
        _excludedLocalNames: {
          type: Object,
          value: function() {
            return {
              'template': 1,
              'dom-bind': 1,
              'dom-if': 1,
              'dom-repeat': 1,
            };
          }
        }
      },

      observers: [
        '_updateAttrForSelected(attrForSelected)',
        '_updateSelected(selected)',
        '_checkFallback(fallbackSelection)'
      ],

      created: function() {
        this._bindFilterItem = this._filterItem.bind(this);
        this._selection = new IronSelection(this._applySelection.bind(this));
      },

      attached: function() {
        this._observer = this._observeItems(this);
        this._addListener(this.activateEvent);
      },

      detached: function() {
        if (this._observer) {
          dom(this).unobserveNodes(this._observer);
        }
        this._removeListener(this.activateEvent);
      },

      /**
       * Returns the index of the given item.
       *
       * @method indexOf
       * @param {Object} item
       * @returns Returns the index of the item
       */
      indexOf: function(item) {
        return this.items ? this.items.indexOf(item) : -1;
      },

      /**
       * Selects the given value.
       *
       * @method select
       * @param {string|number} value the value to select.
       */
      select: function(value) {
        this.selected = value;
      },

      /**
       * Selects the previous item.
       *
       * @method selectPrevious
       */
      selectPrevious: function() {
        var length = this.items.length;
        var index = length - 1;
        if (this.selected !== undefined) {
          index = (Number(this._valueToIndex(this.selected)) - 1 + length) % length;
        }
        this.selected = this._indexToValue(index);
      },

      /**
       * Selects the next item.
       *
       * @method selectNext
       */
      selectNext: function() {
        var index = 0;
        if (this.selected !== undefined) {
          index =
              (Number(this._valueToIndex(this.selected)) + 1) % this.items.length;
        }
        this.selected = this._indexToValue(index);
      },

      /**
       * Selects the item at the given index.
       *
       * @method selectIndex
       */
      selectIndex: function(index) {
        this.select(this._indexToValue(index));
      },

      /**
       * Force a synchronous update of the `items` property.
       *
       * NOTE: Consider listening for the `iron-items-changed` event to respond to
       * updates to the set of selectable items after updates to the DOM list and
       * selection state have been made.
       *
       * WARNING: If you are using this method, you should probably consider an
       * alternate approach. Synchronously querying for items is potentially
       * slow for many use cases. The `items` property will update asynchronously
       * on its own to reflect selectable items in the DOM.
       */
      forceSynchronousItemUpdate: function() {
        if (this._observer && typeof this._observer.flush === 'function') {
          // NOTE(bicknellr): `dom.flush` above is no longer sufficient to trigger
          // `observeNodes` callbacks. Polymer 2.x returns an object from
          // `observeNodes` with a `flush` that synchronously gives the callback any
          // pending MutationRecords (retrieved with `takeRecords`). Any case where
          // ShadyDOM flushes were expected to synchronously trigger item updates
          // will now require calling `forceSynchronousItemUpdate`.
          this._observer.flush();
        } else {
          this._updateItems();
        }
      },

      // UNUSED, FOR API COMPATIBILITY
      get _shouldUpdateSelection() {
        return this.selected != null;
      },

      _checkFallback: function() {
        this._updateSelected();
      },

      _addListener: function(eventName) {
        this.listen(this, eventName, '_activateHandler');
      },

      _removeListener: function(eventName) {
        this.unlisten(this, eventName, '_activateHandler');
      },

      _activateEventChanged: function(eventName, old) {
        this._removeListener(old);
        this._addListener(eventName);
      },

      _updateItems: function() {
        var nodes = dom(this).queryDistributedElements(this.selectable || '*');
        nodes = Array.prototype.filter.call(nodes, this._bindFilterItem);
        this._setItems(nodes);
      },

      _updateAttrForSelected: function() {
        if (this.selectedItem) {
          this.selected = this._valueForItem(this.selectedItem);
        }
      },

      _updateSelected: function() {
        this._selectSelected(this.selected);
      },

      _selectSelected: function(selected) {
        if (!this.items) {
          return;
        }

        var item = this._valueToItem(this.selected);
        if (item) {
          this._selection.select(item);
        } else {
          this._selection.clear();
        }
        // Check for items, since this array is populated only when attached
        // Since Number(0) is falsy, explicitly check for undefined
        if (this.fallbackSelection && this.items.length &&
            (this._selection.get() === undefined)) {
          this.selected = this.fallbackSelection;
        }
      },

      _filterItem: function(node) {
        return !this._excludedLocalNames[node.localName];
      },

      _valueToItem: function(value) {
        return (value == null) ? null : this.items[this._valueToIndex(value)];
      },

      _valueToIndex: function(value) {
        if (this.attrForSelected) {
          for (var i = 0, item; item = this.items[i]; i++) {
            if (this._valueForItem(item) == value) {
              return i;
            }
          }
        } else {
          return Number(value);
        }
      },

      _indexToValue: function(index) {
        if (this.attrForSelected) {
          var item = this.items[index];
          if (item) {
            return this._valueForItem(item);
          }
        } else {
          return index;
        }
      },

      _valueForItem: function(item) {
        if (!item) {
          return null;
        }
        if (!this.attrForSelected) {
          var i = this.indexOf(item);
          return i === -1 ? null : i;
        }
        var propValue = item[dashToCamelCase(this.attrForSelected)];
        return propValue != undefined ? propValue :
                                        item.getAttribute(this.attrForSelected);
      },

      _applySelection: function(item, isSelected) {
        if (this.selectedClass) {
          this.toggleClass(this.selectedClass, isSelected, item);
        }
        if (this.selectedAttribute) {
          this.toggleAttribute(this.selectedAttribute, isSelected, item);
        }
        this._selectionChange();
        this.fire('iron-' + (isSelected ? 'select' : 'deselect'), {item: item});
      },

      _selectionChange: function() {
        this._setSelectedItem(this._selection.get());
      },

      // observe items change under the given node.
      _observeItems: function(node) {
        return dom(node).observeNodes(function(mutation) {
          this._updateItems();
          this._updateSelected();

          // Let other interested parties know about the change so that
          // we don't have to recreate mutation observers everywhere.
          this.fire(
              'iron-items-changed', mutation, {bubbles: false, cancelable: false});
        });
      },

      _activateHandler: function(e) {
        var t = e.target;
        var items = this.items;
        while (t && t != this) {
          var i = items.indexOf(t);
          if (i >= 0) {
            var value = this._indexToValue(i);
            this._itemActivate(value, t);
            return;
          }
          t = t.parentNode;
        }
      },

      _itemActivate: function(value, item) {
        if (!this.fire('iron-activate', {selected: value, item: item}, {
                   cancelable: true
                 })
                 .defaultPrevented) {
          this.select(value);
        }
      }

    };

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
    `iron-pages` is used to select one of its children to show. One use is to cycle
    through a list of children "pages".

    Example:

        <iron-pages selected="0">
          <div>One</div>
          <div>Two</div>
          <div>Three</div>
        </iron-pages>

        <script>
          document.addEventListener('click', function(e) {
            var pages = document.querySelector('iron-pages');
            pages.selectNext();
          });
        </JAVA_SCRIIIIPT/>

    @group Iron Elements
    @demo demo/index.html
    */
    Polymer({
      _template: html`
    <style>
      :host {
        display: block;
      }

      :host > ::slotted(:not(slot):not(.iron-selected)) {
        display: none !important;
      }
    </style>

    <slot></slot>
`,

      is: 'iron-pages',
      behaviors: [IronResizableBehavior, IronSelectableBehavior],

      properties: {

        // as the selected page is the only one visible, activateEvent
        // is both non-sensical and problematic; e.g. in cases where a user
        // handler attempts to change the page and the activateEvent
        // handler immediately changes it back
        activateEvent: {type: String, value: null}

      },

      observers: ['_selectedPageChanged(selected)'],

      _selectedPageChanged: function(selected, old) {
        this.async(this.notifyResize);
      }
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */
    const template$4 = html`
<custom-style>
  <style is="custom-style">
    html {

      --shadow-transition: {
        transition: box-shadow 0.28s cubic-bezier(0.4, 0, 0.2, 1);
      };

      --shadow-none: {
        box-shadow: none;
      };

      /* from http://codepen.io/shyndman/pen/c5394ddf2e8b2a5c9185904b57421cdb */

      --shadow-elevation-2dp: {
        box-shadow: 0 2px 2px 0 rgba(0, 0, 0, 0.14),
                    0 1px 5px 0 rgba(0, 0, 0, 0.12),
                    0 3px 1px -2px rgba(0, 0, 0, 0.2);
      };

      --shadow-elevation-3dp: {
        box-shadow: 0 3px 4px 0 rgba(0, 0, 0, 0.14),
                    0 1px 8px 0 rgba(0, 0, 0, 0.12),
                    0 3px 3px -2px rgba(0, 0, 0, 0.4);
      };

      --shadow-elevation-4dp: {
        box-shadow: 0 4px 5px 0 rgba(0, 0, 0, 0.14),
                    0 1px 10px 0 rgba(0, 0, 0, 0.12),
                    0 2px 4px -1px rgba(0, 0, 0, 0.4);
      };

      --shadow-elevation-6dp: {
        box-shadow: 0 6px 10px 0 rgba(0, 0, 0, 0.14),
                    0 1px 18px 0 rgba(0, 0, 0, 0.12),
                    0 3px 5px -1px rgba(0, 0, 0, 0.4);
      };

      --shadow-elevation-8dp: {
        box-shadow: 0 8px 10px 1px rgba(0, 0, 0, 0.14),
                    0 3px 14px 2px rgba(0, 0, 0, 0.12),
                    0 5px 5px -3px rgba(0, 0, 0, 0.4);
      };

      --shadow-elevation-12dp: {
        box-shadow: 0 12px 16px 1px rgba(0, 0, 0, 0.14),
                    0 4px 22px 3px rgba(0, 0, 0, 0.12),
                    0 6px 7px -4px rgba(0, 0, 0, 0.4);
      };

      --shadow-elevation-16dp: {
        box-shadow: 0 16px 24px 2px rgba(0, 0, 0, 0.14),
                    0  6px 30px 5px rgba(0, 0, 0, 0.12),
                    0  8px 10px -5px rgba(0, 0, 0, 0.4);
      };

      --shadow-elevation-24dp: {
        box-shadow: 0 24px 38px 3px rgba(0, 0, 0, 0.14),
                    0 9px 46px 8px rgba(0, 0, 0, 0.12),
                    0 11px 15px -7px rgba(0, 0, 0, 0.4);
      };
    }
  </style>
</custom-style>`;
    template$4.setAttribute('style', 'display: none;');
    document.head.appendChild(template$4.content);

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */
    const template$5 = html`
<dom-module id="paper-material-styles">
  <template>
    <style>
      html {
        --paper-material: {
          display: block;
          position: relative;
        };
        --paper-material-elevation-1: {
          @apply --shadow-elevation-2dp;
        };
        --paper-material-elevation-2: {
          @apply --shadow-elevation-4dp;
        };
        --paper-material-elevation-3: {
          @apply --shadow-elevation-6dp;
        };
        --paper-material-elevation-4: {
          @apply --shadow-elevation-8dp;
        };
        --paper-material-elevation-5: {
          @apply --shadow-elevation-16dp;
        };
      }
      .paper-material {
        @apply --paper-material;
      }
      .paper-material[elevation="1"] {
        @apply --paper-material-elevation-1;
      }
      .paper-material[elevation="2"] {
        @apply --paper-material-elevation-2;
      }
      .paper-material[elevation="3"] {
        @apply --paper-material-elevation-3;
      }
      .paper-material[elevation="4"] {
        @apply --paper-material-elevation-4;
      }
      .paper-material[elevation="5"] {
        @apply --paper-material-elevation-5;
      }

      /* Duplicate the styles because of https://github.com/webcomponents/shadycss/issues/193 */
      :host {
        --paper-material: {
          display: block;
          position: relative;
        };
        --paper-material-elevation-1: {
          @apply --shadow-elevation-2dp;
        };
        --paper-material-elevation-2: {
          @apply --shadow-elevation-4dp;
        };
        --paper-material-elevation-3: {
          @apply --shadow-elevation-6dp;
        };
        --paper-material-elevation-4: {
          @apply --shadow-elevation-8dp;
        };
        --paper-material-elevation-5: {
          @apply --shadow-elevation-16dp;
        };
      }
      :host(.paper-material) {
        @apply --paper-material;
      }
      :host(.paper-material[elevation="1"]) {
        @apply --paper-material-elevation-1;
      }
      :host(.paper-material[elevation="2"]) {
        @apply --paper-material-elevation-2;
      }
      :host(.paper-material[elevation="3"]) {
        @apply --paper-material-elevation-3;
      }
      :host(.paper-material[elevation="4"]) {
        @apply --paper-material-elevation-4;
      }
      :host(.paper-material[elevation="5"]) {
        @apply --paper-material-elevation-5;
      }
    </style>
  </template>
</dom-module>`;
    template$5.setAttribute('style', 'display: none;');
    document.head.appendChild(template$5.content);

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
     * @demo demo/index.html
     * @polymerBehavior
     */
    const IronControlState = {

      properties: {

        /**
         * If true, the element currently has focus.
         */
        focused: {
          type: Boolean,
          value: false,
          notify: true,
          readOnly: true,
          reflectToAttribute: true
        },

        /**
         * If true, the user cannot interact with this element.
         */
        disabled: {
          type: Boolean,
          value: false,
          notify: true,
          observer: '_disabledChanged',
          reflectToAttribute: true
        },

        /**
         * Value of the `tabindex` attribute before `disabled` was activated.
         * `null` means the attribute was not present.
         * @type {?string|undefined}
         */
        _oldTabIndex: {type: String},

        _boundFocusBlurHandler: {
          type: Function,
          value: function() {
            return this._focusBlurHandler.bind(this);
          }
        }
      },

      observers: ['_changedControlState(focused, disabled)'],

      /**
       * @return {void}
       */
      ready: function() {
        this.addEventListener('focus', this._boundFocusBlurHandler, true);
        this.addEventListener('blur', this._boundFocusBlurHandler, true);
      },

      _focusBlurHandler: function(event) {
        // Polymer takes care of retargeting events.
        this._setFocused(event.type === 'focus');
        return;
      },

      _disabledChanged: function(disabled, old) {
        this.setAttribute('aria-disabled', disabled ? 'true' : 'false');
        this.style.pointerEvents = disabled ? 'none' : '';
        if (disabled) {
          // Read the `tabindex` attribute instead of the `tabIndex` property.
          // The property returns `-1` if there is no `tabindex` attribute.
          // This distinction is important when restoring the value because
          // leaving `-1` hides shadow root children from the tab order.
          this._oldTabIndex = this.getAttribute('tabindex');
          this._setFocused(false);
          this.tabIndex = -1;
          this.blur();
        } else if (this._oldTabIndex !== undefined) {
          if (this._oldTabIndex === null) {
            this.removeAttribute('tabindex');
          } else {
            this.setAttribute('tabindex', this._oldTabIndex);
          }
        }
      },

      _changedControlState: function() {
        // _controlStateChanged is abstract, follow-on behaviors may implement it
        if (this._controlStateChanged) {
          this._controlStateChanged();
        }
      }

    };

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
     * @demo demo/index.html
     * @polymerBehavior IronButtonState
     */
    const IronButtonStateImpl = {

      properties: {

        /**
         * If true, the user is currently holding down the button.
         */
        pressed: {
          type: Boolean,
          readOnly: true,
          value: false,
          reflectToAttribute: true,
          observer: '_pressedChanged'
        },

        /**
         * If true, the button toggles the active state with each tap or press
         * of the spacebar.
         */
        toggles: {type: Boolean, value: false, reflectToAttribute: true},

        /**
         * If true, the button is a toggle and is currently in the active state.
         */
        active:
            {type: Boolean, value: false, notify: true, reflectToAttribute: true},

        /**
         * True if the element is currently being pressed by a "pointer," which
         * is loosely defined as mouse or touch input (but specifically excluding
         * keyboard input).
         */
        pointerDown: {type: Boolean, readOnly: true, value: false},

        /**
         * True if the input device that caused the element to receive focus
         * was a keyboard.
         */
        receivedFocusFromKeyboard: {type: Boolean, readOnly: true},

        /**
         * The aria attribute to be set if the button is a toggle and in the
         * active state.
         */
        ariaActiveAttribute: {
          type: String,
          value: 'aria-pressed',
          observer: '_ariaActiveAttributeChanged'
        }
      },

      listeners: {down: '_downHandler', up: '_upHandler', tap: '_tapHandler'},

      observers:
          ['_focusChanged(focused)', '_activeChanged(active, ariaActiveAttribute)'],

      /**
       * @type {!Object}
       */
      keyBindings: {
        'enter:keydown': '_asyncClick',
        'space:keydown': '_spaceKeyDownHandler',
        'space:keyup': '_spaceKeyUpHandler',
      },

      _mouseEventRe: /^mouse/,

      _tapHandler: function() {
        if (this.toggles) {
          // a tap is needed to toggle the active state
          this._userActivate(!this.active);
        } else {
          this.active = false;
        }
      },

      _focusChanged: function(focused) {
        this._detectKeyboardFocus(focused);

        if (!focused) {
          this._setPressed(false);
        }
      },

      _detectKeyboardFocus: function(focused) {
        this._setReceivedFocusFromKeyboard(!this.pointerDown && focused);
      },

      // to emulate native checkbox, (de-)activations from a user interaction fire
      // 'change' events
      _userActivate: function(active) {
        if (this.active !== active) {
          this.active = active;
          this.fire('change');
        }
      },

      _downHandler: function(event) {
        this._setPointerDown(true);
        this._setPressed(true);
        this._setReceivedFocusFromKeyboard(false);
      },

      _upHandler: function() {
        this._setPointerDown(false);
        this._setPressed(false);
      },

      /**
       * @param {!KeyboardEvent} event .
       */
      _spaceKeyDownHandler: function(event) {
        var keyboardEvent = event.detail.keyboardEvent;
        var target = dom(keyboardEvent).localTarget;

        // Ignore the event if this is coming from a focused light child, since that
        // element will deal with it.
        if (this.isLightDescendant(/** @type {Node} */ (target)))
          return;

        keyboardEvent.preventDefault();
        keyboardEvent.stopImmediatePropagation();
        this._setPressed(true);
      },

      /**
       * @param {!KeyboardEvent} event .
       */
      _spaceKeyUpHandler: function(event) {
        var keyboardEvent = event.detail.keyboardEvent;
        var target = dom(keyboardEvent).localTarget;

        // Ignore the event if this is coming from a focused light child, since that
        // element will deal with it.
        if (this.isLightDescendant(/** @type {Node} */ (target)))
          return;

        if (this.pressed) {
          this._asyncClick();
        }
        this._setPressed(false);
      },

      // trigger click asynchronously, the asynchrony is useful to allow one
      // event handler to unwind before triggering another event
      _asyncClick: function() {
        this.async(function() {
          this.click();
        }, 1);
      },

      // any of these changes are considered a change to button state

      _pressedChanged: function(pressed) {
        this._changedButtonState();
      },

      _ariaActiveAttributeChanged: function(value, oldValue) {
        if (oldValue && oldValue != value && this.hasAttribute(oldValue)) {
          this.removeAttribute(oldValue);
        }
      },

      _activeChanged: function(active, ariaActiveAttribute) {
        if (this.toggles) {
          this.setAttribute(this.ariaActiveAttribute, active ? 'true' : 'false');
        } else {
          this.removeAttribute(this.ariaActiveAttribute);
        }
        this._changedButtonState();
      },

      _controlStateChanged: function() {
        if (this.disabled) {
          this._setPressed(false);
        } else {
          this._changedButtonState();
        }
      },

      // provide hook for follow-on behaviors to react to button-state

      _changedButtonState: function() {
        if (this._buttonStateChanged) {
          this._buttonStateChanged();  // abstract
        }
      }

    };

    /** @polymerBehavior */
    const IronButtonState = [IronA11yKeysBehavior, IronButtonStateImpl];

    /**
    @license
    Copyright (c) 2014 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    var Utility = {
      distance: function(x1, y1, x2, y2) {
        var xDelta = (x1 - x2);
        var yDelta = (y1 - y2);

        return Math.sqrt(xDelta * xDelta + yDelta * yDelta);
      },

      now: window.performance && window.performance.now ?
          window.performance.now.bind(window.performance) :
          Date.now
    };

    /**
     * @param {HTMLElement} element
     * @constructor
     */
    function ElementMetrics(element) {
      this.element = element;
      this.width = this.boundingRect.width;
      this.height = this.boundingRect.height;

      this.size = Math.max(this.width, this.height);
    }

    ElementMetrics.prototype = {
      get boundingRect() {
        return this.element.getBoundingClientRect();
      },

      furthestCornerDistanceFrom: function(x, y) {
        var topLeft = Utility.distance(x, y, 0, 0);
        var topRight = Utility.distance(x, y, this.width, 0);
        var bottomLeft = Utility.distance(x, y, 0, this.height);
        var bottomRight = Utility.distance(x, y, this.width, this.height);

        return Math.max(topLeft, topRight, bottomLeft, bottomRight);
      }
    };

    /**
     * @param {HTMLElement} element
     * @constructor
     */
    function Ripple(element) {
      this.element = element;
      this.color = window.getComputedStyle(element).color;

      this.wave = document.createElement('div');
      this.waveContainer = document.createElement('div');
      this.wave.style.backgroundColor = this.color;
      this.wave.classList.add('wave');
      this.waveContainer.classList.add('wave-container');
      dom(this.waveContainer).appendChild(this.wave);

      this.resetInteractionState();
    }

    Ripple.MAX_RADIUS = 300;

    Ripple.prototype = {
      get recenters() {
        return this.element.recenters;
      },

      get center() {
        return this.element.center;
      },

      get mouseDownElapsed() {
        var elapsed;

        if (!this.mouseDownStart) {
          return 0;
        }

        elapsed = Utility.now() - this.mouseDownStart;

        if (this.mouseUpStart) {
          elapsed -= this.mouseUpElapsed;
        }

        return elapsed;
      },

      get mouseUpElapsed() {
        return this.mouseUpStart ? Utility.now() - this.mouseUpStart : 0;
      },

      get mouseDownElapsedSeconds() {
        return this.mouseDownElapsed / 1000;
      },

      get mouseUpElapsedSeconds() {
        return this.mouseUpElapsed / 1000;
      },

      get mouseInteractionSeconds() {
        return this.mouseDownElapsedSeconds + this.mouseUpElapsedSeconds;
      },

      get initialOpacity() {
        return this.element.initialOpacity;
      },

      get opacityDecayVelocity() {
        return this.element.opacityDecayVelocity;
      },

      get radius() {
        var width2 = this.containerMetrics.width * this.containerMetrics.width;
        var height2 = this.containerMetrics.height * this.containerMetrics.height;
        var waveRadius =
            Math.min(Math.sqrt(width2 + height2), Ripple.MAX_RADIUS) * 1.1 + 5;

        var duration = 1.1 - 0.2 * (waveRadius / Ripple.MAX_RADIUS);
        var timeNow = this.mouseInteractionSeconds / duration;
        var size = waveRadius * (1 - Math.pow(80, -timeNow));

        return Math.abs(size);
      },

      get opacity() {
        if (!this.mouseUpStart) {
          return this.initialOpacity;
        }

        return Math.max(
            0,
            this.initialOpacity -
                this.mouseUpElapsedSeconds * this.opacityDecayVelocity);
      },

      get outerOpacity() {
        // Linear increase in background opacity, capped at the opacity
        // of the wavefront (waveOpacity).
        var outerOpacity = this.mouseUpElapsedSeconds * 0.3;
        var waveOpacity = this.opacity;

        return Math.max(0, Math.min(outerOpacity, waveOpacity));
      },

      get isOpacityFullyDecayed() {
        return this.opacity < 0.01 &&
            this.radius >= Math.min(this.maxRadius, Ripple.MAX_RADIUS);
      },

      get isRestingAtMaxRadius() {
        return this.opacity >= this.initialOpacity &&
            this.radius >= Math.min(this.maxRadius, Ripple.MAX_RADIUS);
      },

      get isAnimationComplete() {
        return this.mouseUpStart ? this.isOpacityFullyDecayed :
                                   this.isRestingAtMaxRadius;
      },

      get translationFraction() {
        return Math.min(
            1, this.radius / this.containerMetrics.size * 2 / Math.sqrt(2));
      },

      get xNow() {
        if (this.xEnd) {
          return this.xStart + this.translationFraction * (this.xEnd - this.xStart);
        }

        return this.xStart;
      },

      get yNow() {
        if (this.yEnd) {
          return this.yStart + this.translationFraction * (this.yEnd - this.yStart);
        }

        return this.yStart;
      },

      get isMouseDown() {
        return this.mouseDownStart && !this.mouseUpStart;
      },

      resetInteractionState: function() {
        this.maxRadius = 0;
        this.mouseDownStart = 0;
        this.mouseUpStart = 0;

        this.xStart = 0;
        this.yStart = 0;
        this.xEnd = 0;
        this.yEnd = 0;
        this.slideDistance = 0;

        this.containerMetrics = new ElementMetrics(this.element);
      },

      draw: function() {
        var scale;
        var dx;
        var dy;

        this.wave.style.opacity = this.opacity;

        scale = this.radius / (this.containerMetrics.size / 2);
        dx = this.xNow - (this.containerMetrics.width / 2);
        dy = this.yNow - (this.containerMetrics.height / 2);


        // 2d transform for safari because of border-radius and overflow:hidden
        // clipping bug. https://bugs.webkit.org/show_bug.cgi?id=98538
        this.waveContainer.style.webkitTransform =
            'translate(' + dx + 'px, ' + dy + 'px)';
        this.waveContainer.style.transform =
            'translate3d(' + dx + 'px, ' + dy + 'px, 0)';
        this.wave.style.webkitTransform = 'scale(' + scale + ',' + scale + ')';
        this.wave.style.transform = 'scale3d(' + scale + ',' + scale + ',1)';
      },

      /** @param {Event=} event */
      downAction: function(event) {
        var xCenter = this.containerMetrics.width / 2;
        var yCenter = this.containerMetrics.height / 2;

        this.resetInteractionState();
        this.mouseDownStart = Utility.now();

        if (this.center) {
          this.xStart = xCenter;
          this.yStart = yCenter;
          this.slideDistance =
              Utility.distance(this.xStart, this.yStart, this.xEnd, this.yEnd);
        } else {
          this.xStart = event ?
              event.detail.x - this.containerMetrics.boundingRect.left :
              this.containerMetrics.width / 2;
          this.yStart = event ?
              event.detail.y - this.containerMetrics.boundingRect.top :
              this.containerMetrics.height / 2;
        }

        if (this.recenters) {
          this.xEnd = xCenter;
          this.yEnd = yCenter;
          this.slideDistance =
              Utility.distance(this.xStart, this.yStart, this.xEnd, this.yEnd);
        }

        this.maxRadius = this.containerMetrics.furthestCornerDistanceFrom(
            this.xStart, this.yStart);

        this.waveContainer.style.top =
            (this.containerMetrics.height - this.containerMetrics.size) / 2 + 'px';
        this.waveContainer.style.left =
            (this.containerMetrics.width - this.containerMetrics.size) / 2 + 'px';

        this.waveContainer.style.width = this.containerMetrics.size + 'px';
        this.waveContainer.style.height = this.containerMetrics.size + 'px';
      },

      /** @param {Event=} event */
      upAction: function(event) {
        if (!this.isMouseDown) {
          return;
        }

        this.mouseUpStart = Utility.now();
      },

      remove: function() {
        dom(dom(this.waveContainer).parentNode).removeChild(this.waveContainer);
      }
    };

    /**
    Material design: [Surface
    reaction](https://www.google.com/design/spec/animation/responsive-interaction.html#responsive-interaction-surface-reaction)

    `paper-ripple` provides a visual effect that other paper elements can
    use to simulate a rippling effect emanating from the point of contact.  The
    effect can be visualized as a concentric circle with motion.

    Example:

        <div style="position:relative">
          <paper-ripple></paper-ripple>
        </div>

    Note, it's important that the parent container of the ripple be relative
    position, otherwise the ripple will emanate outside of the desired container.

    `paper-ripple` listens to "mousedown" and "mouseup" events so it would display
    ripple effect when touches on it.  You can also defeat the default behavior and
    manually route the down and up actions to the ripple element.  Note that it is
    important if you call `downAction()` you will have to make sure to call
    `upAction()` so that `paper-ripple` would end the animation loop.

    Example:

        <paper-ripple id="ripple" style="pointer-events: none;"></paper-ripple>
        ...
        downAction: function(e) {
          this.$.ripple.downAction(e.detail);
        },
        upAction: function(e) {
          this.$.ripple.upAction();
        }

    Styling ripple effect:

      Use CSS color property to style the ripple:

        paper-ripple {
          color: #4285f4;
        }

      Note that CSS color property is inherited so it is not required to set it on
      the `paper-ripple` element directly.

    By default, the ripple is centered on the point of contact.  Apply the
    `recenters` attribute to have the ripple grow toward the center of its
    container.

        <paper-ripple recenters></paper-ripple>

    You can also  center the ripple inside its container from the start.

        <paper-ripple center></paper-ripple>

    Apply `circle` class to make the rippling effect within a circle.

        <paper-ripple class="circle"></paper-ripple>

    @element paper-ripple
    @demo demo/index.html
    */
    Polymer({
      /** @override */
      _template: html`
    <style>
      :host {
        display: block;
        position: absolute;
        border-radius: inherit;
        overflow: hidden;
        top: 0;
        left: 0;
        right: 0;
        bottom: 0;

        /* See PolymerElements/paper-behaviors/issues/34. On non-Chrome browsers,
         * creating a node (with a position:absolute) in the middle of an event
         * handler "interrupts" that event handler (which happens when the
         * ripple is created on demand) */
        pointer-events: none;
      }

      :host([animating]) {
        /* This resolves a rendering issue in Chrome (as of 40) where the
           ripple is not properly clipped by its parent (which may have
           rounded corners). See: http://jsbin.com/temexa/4

           Note: We only apply this style conditionally. Otherwise, the browser
           will create a new compositing layer for every ripple element on the
           page, and that would be bad. */
        -webkit-transform: translate(0, 0);
        transform: translate3d(0, 0, 0);
      }

      #background,
      #waves,
      .wave-container,
      .wave {
        pointer-events: none;
        position: absolute;
        top: 0;
        left: 0;
        width: 100%;
        height: 100%;
      }

      #background,
      .wave {
        opacity: 0;
      }

      #waves,
      .wave {
        overflow: hidden;
      }

      .wave-container,
      .wave {
        border-radius: 50%;
      }

      :host(.circle) #background,
      :host(.circle) #waves {
        border-radius: 50%;
      }

      :host(.circle) .wave-container {
        overflow: hidden;
      }
    </style>

    <div id="background"></div>
    <div id="waves"></div>
`,

      is: 'paper-ripple',
      behaviors: [IronA11yKeysBehavior],

      properties: {
        /**
         * The initial opacity set on the wave.
         * @type number
         * @default 0.25
         */
        initialOpacity: {type: Number, value: 0.25},

        /**
         * How fast (opacity per second) the wave fades out.
         *
         * @type number
         * @default 0.8
         */
        opacityDecayVelocity: {type: Number, value: 0.8},

        /**
         * If true, ripples will exhibit a gravitational pull towards
         * the center of their container as they fade away.
         *
         * @type boolean
         * @default false
         */
        recenters: {type: Boolean, value: false},

        /**
         * If true, ripples will center inside its container
         *
         * @type boolean
         * @default false
         */
        center: {type: Boolean, value: false},

        /**
         * A list of the visual ripples.
         *
         * @type Array
         * @default []
         */
        ripples: {
          type: Array,
          value: function() {
            return [];
          }
        },

        /**
         * True when there are visible ripples animating within the
         * element.
         */
        animating:
            {type: Boolean, readOnly: true, reflectToAttribute: true, value: false},

        /**
         * If true, the ripple will remain in the "down" state until `holdDown`
         * is set to false again.
         */
        holdDown: {type: Boolean, value: false, observer: '_holdDownChanged'},

        /**
         * If true, the ripple will not generate a ripple effect
         * via pointer interaction.
         * Calling ripple's imperative api like `simulatedRipple` will
         * still generate the ripple effect.
         */
        noink: {type: Boolean, value: false},

        _animating: {type: Boolean},

        _boundAnimate: {
          type: Function,
          value: function() {
            return this.animate.bind(this);
          }
        }
      },

      get target() {
        return this.keyEventTarget;
      },

      /**
       * @type {!Object}
       */
      keyBindings: {
        'enter:keydown': '_onEnterKeydown',
        'space:keydown': '_onSpaceKeydown',
        'space:keyup': '_onSpaceKeyup'
      },

      /** @override */
      attached: function() {
        // Set up a11yKeysBehavior to listen to key events on the target,
        // so that space and enter activate the ripple even if the target doesn't
        // handle key events. The key handlers deal with `noink` themselves.
        if (dom(this).parentNode.nodeType == 11) {  // DOCUMENT_FRAGMENT_NODE
          this.keyEventTarget = dom(this).getOwnerRoot().host;
        } else {
          this.keyEventTarget = dom(this).parentNode;
        }
        var keyEventTarget = /** @type {!EventTarget} */ (this.keyEventTarget);
        this.listen(keyEventTarget, 'up', 'uiUpAction');
        this.listen(keyEventTarget, 'down', 'uiDownAction');
      },

      /** @override */
      detached: function() {
        this.unlisten(this.keyEventTarget, 'up', 'uiUpAction');
        this.unlisten(this.keyEventTarget, 'down', 'uiDownAction');
        this.keyEventTarget = null;
      },

      get shouldKeepAnimating() {
        for (var index = 0; index < this.ripples.length; ++index) {
          if (!this.ripples[index].isAnimationComplete) {
            return true;
          }
        }

        return false;
      },

      simulatedRipple: function() {
        this.downAction(null);

        // Please see polymer/polymer#1305
        this.async(function() {
          this.upAction();
        }, 1);
      },

      /**
       * Provokes a ripple down effect via a UI event,
       * respecting the `noink` property.
       * @param {Event=} event
       */
      uiDownAction: function(event) {
        if (!this.noink) {
          this.downAction(event);
        }
      },

      /**
       * Provokes a ripple down effect via a UI event,
       * *not* respecting the `noink` property.
       * @param {Event=} event
       */
      downAction: function(event) {
        if (this.holdDown && this.ripples.length > 0) {
          return;
        }

        var ripple = this.addRipple();

        ripple.downAction(event);

        if (!this._animating) {
          this._animating = true;
          this.animate();
        }
      },

      /**
       * Provokes a ripple up effect via a UI event,
       * respecting the `noink` property.
       * @param {Event=} event
       */
      uiUpAction: function(event) {
        if (!this.noink) {
          this.upAction(event);
        }
      },

      /**
       * Provokes a ripple up effect via a UI event,
       * *not* respecting the `noink` property.
       * @param {Event=} event
       */
      upAction: function(event) {
        if (this.holdDown) {
          return;
        }

        this.ripples.forEach(function(ripple) {
          ripple.upAction(event);
        });

        this._animating = true;
        this.animate();
      },

      onAnimationComplete: function() {
        this._animating = false;
        this.$.background.style.backgroundColor = '';
        this.fire('transitionend');
      },

      addRipple: function() {
        var ripple = new Ripple(this);

        dom(this.$.waves).appendChild(ripple.waveContainer);
        this.$.background.style.backgroundColor = ripple.color;
        this.ripples.push(ripple);

        this._setAnimating(true);

        return ripple;
      },

      removeRipple: function(ripple) {
        var rippleIndex = this.ripples.indexOf(ripple);

        if (rippleIndex < 0) {
          return;
        }

        this.ripples.splice(rippleIndex, 1);

        ripple.remove();

        if (!this.ripples.length) {
          this._setAnimating(false);
        }
      },

      /**
       * Deprecated. Please use animateRipple() instead.
       *
       * This method name conflicts with Element#animate().
       * https://developer.mozilla.org/en-US/docs/Web/API/Element/animate.
       *
       * @suppress {checkTypes}
       * @override
       */
      animate: function() {
        if (!this._animating) {
          return;
        }
        var index;
        var ripple;

        for (index = 0; index < this.ripples.length; ++index) {
          ripple = this.ripples[index];

          ripple.draw();

          this.$.background.style.opacity = ripple.outerOpacity;

          if (ripple.isOpacityFullyDecayed && !ripple.isRestingAtMaxRadius) {
            this.removeRipple(ripple);
          }
        }

        if (!this.shouldKeepAnimating && this.ripples.length === 0) {
          this.onAnimationComplete();
        } else {
          window.requestAnimationFrame(this._boundAnimate);
        }
      },

      /**
       * An alias for animate() whose name does not conflict with the platform
       * Element.animate() method.
       */
      animateRipple: function() {
        return this.animate();
      },

      _onEnterKeydown: function() {
        this.uiDownAction();
        this.async(this.uiUpAction, 1);
      },

      _onSpaceKeydown: function() {
        this.uiDownAction();
      },

      _onSpaceKeyup: function() {
        this.uiUpAction();
      },

      // note: holdDown does not respect noink since it can be a focus based
      // effect.
      _holdDownChanged: function(newVal, oldVal) {
        if (oldVal === undefined) {
          return;
        }
        if (newVal) {
          this.downAction();
        } else {
          this.upAction();
        }
      }

      /**
      Fired when the animation finishes.
      This is useful if you want to wait until
      the ripple animation finishes to perform some action.

      @event transitionend
      @param {{node: Object}} detail Contains the animated node.
      */
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
     * `PaperRippleBehavior` dynamically implements a ripple when the element has
     * focus via pointer or keyboard.
     *
     * NOTE: This behavior is intended to be used in conjunction with and after
     * `IronButtonState` and `IronControlState`.
     *
     * @polymerBehavior PaperRippleBehavior
     */
    const PaperRippleBehavior = {
      properties: {
        /**
         * If true, the element will not produce a ripple effect when interacted
         * with via the pointer.
         */
        noink: {type: Boolean, observer: '_noinkChanged'},

        /**
         * @type {Element|undefined}
         */
        _rippleContainer: {
          type: Object,
        }
      },

      /**
       * Ensures a `<paper-ripple>` element is available when the element is
       * focused.
       */
      _buttonStateChanged: function() {
        if (this.focused) {
          this.ensureRipple();
        }
      },

      /**
       * In addition to the functionality provided in `IronButtonState`, ensures
       * a ripple effect is created when the element is in a `pressed` state.
       */
      _downHandler: function(event) {
        IronButtonStateImpl._downHandler.call(this, event);
        if (this.pressed) {
          this.ensureRipple(event);
        }
      },

      /**
       * Ensures this element contains a ripple effect. For startup efficiency
       * the ripple effect is dynamically on demand when needed.
       * @param {!Event=} optTriggeringEvent (optional) event that triggered the
       * ripple.
       */
      ensureRipple: function(optTriggeringEvent) {
        if (!this.hasRipple()) {
          this._ripple = this._createRipple();
          this._ripple.noink = this.noink;
          var rippleContainer = this._rippleContainer || this.root;
          if (rippleContainer) {
            dom(rippleContainer).appendChild(this._ripple);
          }
          if (optTriggeringEvent) {
            // Check if the event happened inside of the ripple container
            // Fall back to host instead of the root because distributed text
            // nodes are not valid event targets
            var domContainer = dom(this._rippleContainer || this);
            var target = dom(optTriggeringEvent).rootTarget;
            if (domContainer.deepContains(/** @type {Node} */ (target))) {
              this._ripple.uiDownAction(optTriggeringEvent);
            }
          }
        }
      },

      /**
       * Returns the `<paper-ripple>` element used by this element to create
       * ripple effects. The element's ripple is created on demand, when
       * necessary, and calling this method will force the
       * ripple to be created.
       */
      getRipple: function() {
        this.ensureRipple();
        return this._ripple;
      },

      /**
       * Returns true if this element currently contains a ripple effect.
       * @return {boolean}
       */
      hasRipple: function() {
        return Boolean(this._ripple);
      },

      /**
       * Create the element's ripple effect via creating a `<paper-ripple>`.
       * Override this method to customize the ripple element.
       * @return {!PaperRippleElement} Returns a `<paper-ripple>` element.
       */
      _createRipple: function() {
        var element = /** @type {!PaperRippleElement} */ (
            document.createElement('paper-ripple'));
        return element;
      },

      _noinkChanged: function(noink) {
        if (this.hasRipple()) {
          this._ripple.noink = noink;
        }
      }
    };

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /** @polymerBehavior PaperButtonBehavior */
    const PaperButtonBehaviorImpl = {
      properties: {
        /**
         * The z-depth of this element, from 0-5. Setting to 0 will remove the
         * shadow, and each increasing number greater than 0 will be "deeper"
         * than the last.
         *
         * @attribute elevation
         * @type number
         * @default 1
         */
        elevation: {type: Number, reflectToAttribute: true, readOnly: true}
      },

      observers: [
        '_calculateElevation(focused, disabled, active, pressed, receivedFocusFromKeyboard)',
        '_computeKeyboardClass(receivedFocusFromKeyboard)'
      ],

      hostAttributes: {role: 'button', tabindex: '0', animated: true},

      _calculateElevation: function() {
        var e = 1;
        if (this.disabled) {
          e = 0;
        } else if (this.active || this.pressed) {
          e = 4;
        } else if (this.receivedFocusFromKeyboard) {
          e = 3;
        }
        this._setElevation(e);
      },

      _computeKeyboardClass: function(receivedFocusFromKeyboard) {
        this.toggleClass('keyboard-focus', receivedFocusFromKeyboard);
      },

      /**
       * In addition to `IronButtonState` behavior, when space key goes down,
       * create a ripple down effect.
       *
       * @param {!KeyboardEvent} event .
       */
      _spaceKeyDownHandler: function(event) {
        IronButtonStateImpl._spaceKeyDownHandler.call(this, event);
        // Ensure that there is at most one ripple when the space key is held down.
        if (this.hasRipple() && this.getRipple().ripples.length < 1) {
          this._ripple.uiDownAction();
        }
      },

      /**
       * In addition to `IronButtonState` behavior, when space key goes up,
       * create a ripple up effect.
       *
       * @param {!KeyboardEvent} event .
       */
      _spaceKeyUpHandler: function(event) {
        IronButtonStateImpl._spaceKeyUpHandler.call(this, event);
        if (this.hasRipple()) {
          this._ripple.uiUpAction();
        }
      }
    };

    /** @polymerBehavior */
    const PaperButtonBehavior = [
      IronButtonState,
      IronControlState,
      PaperRippleBehavior,
      PaperButtonBehaviorImpl
    ];

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    const template$6 = html`
  <style include="paper-material-styles">
    /* Need to specify the same specificity as the styles imported from paper-material. */
    :host {
      @apply --layout-inline;
      @apply --layout-center-center;
      position: relative;
      box-sizing: border-box;
      min-width: 5.14em;
      margin: 0 0.29em;
      background: transparent;
      -webkit-tap-highlight-color: rgba(0, 0, 0, 0);
      -webkit-tap-highlight-color: transparent;
      font: inherit;
      text-transform: uppercase;
      outline-width: 0;
      border-radius: 3px;
      -moz-user-select: none;
      -ms-user-select: none;
      -webkit-user-select: none;
      user-select: none;
      cursor: pointer;
      z-index: 0;
      padding: 0.7em 0.57em;

      @apply --paper-font-common-base;
      @apply --paper-button;
    }

    :host([elevation="1"]) {
      @apply --paper-material-elevation-1;
    }

    :host([elevation="2"]) {
      @apply --paper-material-elevation-2;
    }

    :host([elevation="3"]) {
      @apply --paper-material-elevation-3;
    }

    :host([elevation="4"]) {
      @apply --paper-material-elevation-4;
    }

    :host([elevation="5"]) {
      @apply --paper-material-elevation-5;
    }

    :host([hidden]) {
      display: none !important;
    }

    :host([raised].keyboard-focus) {
      font-weight: bold;
      @apply --paper-button-raised-keyboard-focus;
    }

    :host(:not([raised]).keyboard-focus) {
      font-weight: bold;
      @apply --paper-button-flat-keyboard-focus;
    }

    :host([disabled]) {
      background: none;
      color: #a8a8a8;
      cursor: auto;
      pointer-events: none;

      @apply --paper-button-disabled;
    }

    :host([disabled][raised]) {
      background: #eaeaea;
    }


    :host([animated]) {
      @apply --shadow-transition;
    }

    paper-ripple {
      color: var(--paper-button-ink-color);
    }
  </style>

  <slot></slot>`;

    template$6.setAttribute('strip-whitespace', '');

    /**
    Material design:
    [Buttons](https://www.google.com/design/spec/components/buttons.html)

    `paper-button` is a button. When the user touches the button, a ripple effect
    emanates from the point of contact. It may be flat or raised. A raised button is
    styled with a shadow.

    Example:

        <paper-button>Flat button</paper-button>
        <paper-button raised>Raised button</paper-button>
        <paper-button noink>No ripple effect</paper-button>
        <paper-button toggles>Toggle-able button</paper-button>

    A button that has `toggles` true will remain `active` after being clicked (and
    will have an `active` attribute set). For more information, see the
    `IronButtonState` behavior.

    You may use custom DOM in the button body to create a variety of buttons. For
    example, to create a button with an icon and some text:

        <paper-button>
          <iron-icon icon="favorite"></iron-icon>
          custom button content
        </paper-button>

    To use `paper-button` as a link, wrap it in an anchor tag. Since `paper-button`
    will already receive focus, you may want to prevent the anchor tag from
    receiving focus as well by setting its tabindex to -1.

        <a href="https://www.polymer-project.org/" tabindex="-1">
          <paper-button raised>Polymer Project</paper-button>
        </a>

    ### Styling

    Style the button with CSS as you would a normal DOM element.

        paper-button.fancy {
          background: green;
          color: yellow;
        }

        paper-button.fancy:hover {
          background: lime;
        }

        paper-button[disabled],
        paper-button[toggles][active] {
          background: red;
        }

    By default, the ripple is the same color as the foreground at 25% opacity. You
    may customize the color using the `--paper-button-ink-color` custom property.

    The following custom properties and mixins are also available for styling:

    Custom property | Description | Default
    ----------------|-------------|----------
    `--paper-button-ink-color` | Background color of the ripple | `Based on the button's color`
    `--paper-button` | Mixin applied to the button | `{}`
    `--paper-button-disabled` | Mixin applied to the disabled button. Note that you can also use the `paper-button[disabled]` selector | `{}`
    `--paper-button-flat-keyboard-focus` | Mixin applied to a flat button after it's been focused using the keyboard | `{}`
    `--paper-button-raised-keyboard-focus` | Mixin applied to a raised button after it's been focused using the keyboard | `{}`

    @demo demo/index.html
    */
    Polymer({
      _template: template$6,

      is: 'paper-button',

      behaviors: [PaperButtonBehavior],

      properties: {
        /**
         * If true, the button should be styled with a shadow.
         */
        raised: {
          type: Boolean,
          reflectToAttribute: true,
          value: false,
          observer: '_calculateElevation',
        }
      },

      _calculateElevation: function() {
        if (!this.raised) {
          this._setElevation(0);
        } else {
          PaperButtonBehaviorImpl._calculateElevation.apply(this);
        }
      }

      /**
      Fired when the animation finishes.
      This is useful if you want to wait until
      the ripple animation finishes to perform some action.

      @event transitionend
      Event param: {{node: Object}} detail Contains the animated node.
      */
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */
    const template$7 = html`
<custom-style>
  <style is="custom-style">
    html {

      /* Material Design color palette for Google products */

      --google-red-100: #f4c7c3;
      --google-red-300: #e67c73;
      --google-red-500: #db4437;
      --google-red-700: #c53929;

      --google-blue-100: #c6dafc;
      --google-blue-300: #7baaf7;
      --google-blue-500: #4285f4;
      --google-blue-700: #3367d6;

      --google-green-100: #b7e1cd;
      --google-green-300: #57bb8a;
      --google-green-500: #0f9d58;
      --google-green-700: #0b8043;

      --google-yellow-100: #fce8b2;
      --google-yellow-300: #f7cb4d;
      --google-yellow-500: #f4b400;
      --google-yellow-700: #f09300;

      --google-grey-100: #f5f5f5;
      --google-grey-300: #e0e0e0;
      --google-grey-500: #9e9e9e;
      --google-grey-700: #616161;

      /* Material Design color palette from online spec document */

      --paper-red-50: #ffebee;
      --paper-red-100: #ffcdd2;
      --paper-red-200: #ef9a9a;
      --paper-red-300: #e57373;
      --paper-red-400: #ef5350;
      --paper-red-500: #f44336;
      --paper-red-600: #e53935;
      --paper-red-700: #d32f2f;
      --paper-red-800: #c62828;
      --paper-red-900: #b71c1c;
      --paper-red-a100: #ff8a80;
      --paper-red-a200: #ff5252;
      --paper-red-a400: #ff1744;
      --paper-red-a700: #d50000;

      --paper-pink-50: #fce4ec;
      --paper-pink-100: #f8bbd0;
      --paper-pink-200: #f48fb1;
      --paper-pink-300: #f06292;
      --paper-pink-400: #ec407a;
      --paper-pink-500: #e91e63;
      --paper-pink-600: #d81b60;
      --paper-pink-700: #c2185b;
      --paper-pink-800: #ad1457;
      --paper-pink-900: #880e4f;
      --paper-pink-a100: #ff80ab;
      --paper-pink-a200: #ff4081;
      --paper-pink-a400: #f50057;
      --paper-pink-a700: #c51162;

      --paper-purple-50: #f3e5f5;
      --paper-purple-100: #e1bee7;
      --paper-purple-200: #ce93d8;
      --paper-purple-300: #ba68c8;
      --paper-purple-400: #ab47bc;
      --paper-purple-500: #9c27b0;
      --paper-purple-600: #8e24aa;
      --paper-purple-700: #7b1fa2;
      --paper-purple-800: #6a1b9a;
      --paper-purple-900: #4a148c;
      --paper-purple-a100: #ea80fc;
      --paper-purple-a200: #e040fb;
      --paper-purple-a400: #d500f9;
      --paper-purple-a700: #aa00ff;

      --paper-deep-purple-50: #ede7f6;
      --paper-deep-purple-100: #d1c4e9;
      --paper-deep-purple-200: #b39ddb;
      --paper-deep-purple-300: #9575cd;
      --paper-deep-purple-400: #7e57c2;
      --paper-deep-purple-500: #673ab7;
      --paper-deep-purple-600: #5e35b1;
      --paper-deep-purple-700: #512da8;
      --paper-deep-purple-800: #4527a0;
      --paper-deep-purple-900: #311b92;
      --paper-deep-purple-a100: #b388ff;
      --paper-deep-purple-a200: #7c4dff;
      --paper-deep-purple-a400: #651fff;
      --paper-deep-purple-a700: #6200ea;

      --paper-indigo-50: #e8eaf6;
      --paper-indigo-100: #c5cae9;
      --paper-indigo-200: #9fa8da;
      --paper-indigo-300: #7986cb;
      --paper-indigo-400: #5c6bc0;
      --paper-indigo-500: #3f51b5;
      --paper-indigo-600: #3949ab;
      --paper-indigo-700: #303f9f;
      --paper-indigo-800: #283593;
      --paper-indigo-900: #1a237e;
      --paper-indigo-a100: #8c9eff;
      --paper-indigo-a200: #536dfe;
      --paper-indigo-a400: #3d5afe;
      --paper-indigo-a700: #304ffe;

      --paper-blue-50: #e3f2fd;
      --paper-blue-100: #bbdefb;
      --paper-blue-200: #90caf9;
      --paper-blue-300: #64b5f6;
      --paper-blue-400: #42a5f5;
      --paper-blue-500: #2196f3;
      --paper-blue-600: #1e88e5;
      --paper-blue-700: #1976d2;
      --paper-blue-800: #1565c0;
      --paper-blue-900: #0d47a1;
      --paper-blue-a100: #82b1ff;
      --paper-blue-a200: #448aff;
      --paper-blue-a400: #2979ff;
      --paper-blue-a700: #2962ff;

      --paper-light-blue-50: #e1f5fe;
      --paper-light-blue-100: #b3e5fc;
      --paper-light-blue-200: #81d4fa;
      --paper-light-blue-300: #4fc3f7;
      --paper-light-blue-400: #29b6f6;
      --paper-light-blue-500: #03a9f4;
      --paper-light-blue-600: #039be5;
      --paper-light-blue-700: #0288d1;
      --paper-light-blue-800: #0277bd;
      --paper-light-blue-900: #01579b;
      --paper-light-blue-a100: #80d8ff;
      --paper-light-blue-a200: #40c4ff;
      --paper-light-blue-a400: #00b0ff;
      --paper-light-blue-a700: #0091ea;

      --paper-cyan-50: #e0f7fa;
      --paper-cyan-100: #b2ebf2;
      --paper-cyan-200: #80deea;
      --paper-cyan-300: #4dd0e1;
      --paper-cyan-400: #26c6da;
      --paper-cyan-500: #00bcd4;
      --paper-cyan-600: #00acc1;
      --paper-cyan-700: #0097a7;
      --paper-cyan-800: #00838f;
      --paper-cyan-900: #006064;
      --paper-cyan-a100: #84ffff;
      --paper-cyan-a200: #18ffff;
      --paper-cyan-a400: #00e5ff;
      --paper-cyan-a700: #00b8d4;

      --paper-teal-50: #e0f2f1;
      --paper-teal-100: #b2dfdb;
      --paper-teal-200: #80cbc4;
      --paper-teal-300: #4db6ac;
      --paper-teal-400: #26a69a;
      --paper-teal-500: #009688;
      --paper-teal-600: #00897b;
      --paper-teal-700: #00796b;
      --paper-teal-800: #00695c;
      --paper-teal-900: #004d40;
      --paper-teal-a100: #a7ffeb;
      --paper-teal-a200: #64ffda;
      --paper-teal-a400: #1de9b6;
      --paper-teal-a700: #00bfa5;

      --paper-green-50: #e8f5e9;
      --paper-green-100: #c8e6c9;
      --paper-green-200: #a5d6a7;
      --paper-green-300: #81c784;
      --paper-green-400: #66bb6a;
      --paper-green-500: #4caf50;
      --paper-green-600: #43a047;
      --paper-green-700: #388e3c;
      --paper-green-800: #2e7d32;
      --paper-green-900: #1b5e20;
      --paper-green-a100: #b9f6ca;
      --paper-green-a200: #69f0ae;
      --paper-green-a400: #00e676;
      --paper-green-a700: #00c853;

      --paper-light-green-50: #f1f8e9;
      --paper-light-green-100: #dcedc8;
      --paper-light-green-200: #c5e1a5;
      --paper-light-green-300: #aed581;
      --paper-light-green-400: #9ccc65;
      --paper-light-green-500: #8bc34a;
      --paper-light-green-600: #7cb342;
      --paper-light-green-700: #689f38;
      --paper-light-green-800: #558b2f;
      --paper-light-green-900: #33691e;
      --paper-light-green-a100: #ccff90;
      --paper-light-green-a200: #b2ff59;
      --paper-light-green-a400: #76ff03;
      --paper-light-green-a700: #64dd17;

      --paper-lime-50: #f9fbe7;
      --paper-lime-100: #f0f4c3;
      --paper-lime-200: #e6ee9c;
      --paper-lime-300: #dce775;
      --paper-lime-400: #d4e157;
      --paper-lime-500: #cddc39;
      --paper-lime-600: #c0ca33;
      --paper-lime-700: #afb42b;
      --paper-lime-800: #9e9d24;
      --paper-lime-900: #827717;
      --paper-lime-a100: #f4ff81;
      --paper-lime-a200: #eeff41;
      --paper-lime-a400: #c6ff00;
      --paper-lime-a700: #aeea00;

      --paper-yellow-50: #fffde7;
      --paper-yellow-100: #fff9c4;
      --paper-yellow-200: #fff59d;
      --paper-yellow-300: #fff176;
      --paper-yellow-400: #ffee58;
      --paper-yellow-500: #ffeb3b;
      --paper-yellow-600: #fdd835;
      --paper-yellow-700: #fbc02d;
      --paper-yellow-800: #f9a825;
      --paper-yellow-900: #f57f17;
      --paper-yellow-a100: #ffff8d;
      --paper-yellow-a200: #ffff00;
      --paper-yellow-a400: #ffea00;
      --paper-yellow-a700: #ffd600;

      --paper-amber-50: #fff8e1;
      --paper-amber-100: #ffecb3;
      --paper-amber-200: #ffe082;
      --paper-amber-300: #ffd54f;
      --paper-amber-400: #ffca28;
      --paper-amber-500: #ffc107;
      --paper-amber-600: #ffb300;
      --paper-amber-700: #ffa000;
      --paper-amber-800: #ff8f00;
      --paper-amber-900: #ff6f00;
      --paper-amber-a100: #ffe57f;
      --paper-amber-a200: #ffd740;
      --paper-amber-a400: #ffc400;
      --paper-amber-a700: #ffab00;

      --paper-orange-50: #fff3e0;
      --paper-orange-100: #ffe0b2;
      --paper-orange-200: #ffcc80;
      --paper-orange-300: #ffb74d;
      --paper-orange-400: #ffa726;
      --paper-orange-500: #ff9800;
      --paper-orange-600: #fb8c00;
      --paper-orange-700: #f57c00;
      --paper-orange-800: #ef6c00;
      --paper-orange-900: #e65100;
      --paper-orange-a100: #ffd180;
      --paper-orange-a200: #ffab40;
      --paper-orange-a400: #ff9100;
      --paper-orange-a700: #ff6500;

      --paper-deep-orange-50: #fbe9e7;
      --paper-deep-orange-100: #ffccbc;
      --paper-deep-orange-200: #ffab91;
      --paper-deep-orange-300: #ff8a65;
      --paper-deep-orange-400: #ff7043;
      --paper-deep-orange-500: #ff5722;
      --paper-deep-orange-600: #f4511e;
      --paper-deep-orange-700: #e64a19;
      --paper-deep-orange-800: #d84315;
      --paper-deep-orange-900: #bf360c;
      --paper-deep-orange-a100: #ff9e80;
      --paper-deep-orange-a200: #ff6e40;
      --paper-deep-orange-a400: #ff3d00;
      --paper-deep-orange-a700: #dd2c00;

      --paper-brown-50: #efebe9;
      --paper-brown-100: #d7ccc8;
      --paper-brown-200: #bcaaa4;
      --paper-brown-300: #a1887f;
      --paper-brown-400: #8d6e63;
      --paper-brown-500: #795548;
      --paper-brown-600: #6d4c41;
      --paper-brown-700: #5d4037;
      --paper-brown-800: #4e342e;
      --paper-brown-900: #3e2723;

      --paper-grey-50: #fafafa;
      --paper-grey-100: #f5f5f5;
      --paper-grey-200: #eeeeee;
      --paper-grey-300: #e0e0e0;
      --paper-grey-400: #bdbdbd;
      --paper-grey-500: #9e9e9e;
      --paper-grey-600: #757575;
      --paper-grey-700: #616161;
      --paper-grey-800: #424242;
      --paper-grey-900: #212121;

      --paper-blue-grey-50: #eceff1;
      --paper-blue-grey-100: #cfd8dc;
      --paper-blue-grey-200: #b0bec5;
      --paper-blue-grey-300: #90a4ae;
      --paper-blue-grey-400: #78909c;
      --paper-blue-grey-500: #607d8b;
      --paper-blue-grey-600: #546e7a;
      --paper-blue-grey-700: #455a64;
      --paper-blue-grey-800: #37474f;
      --paper-blue-grey-900: #263238;

      /* opacity for dark text on a light background */
      --dark-divider-opacity: 0.12;
      --dark-disabled-opacity: 0.38; /* or hint text or icon */
      --dark-secondary-opacity: 0.54;
      --dark-primary-opacity: 0.87;

      /* opacity for light text on a dark background */
      --light-divider-opacity: 0.12;
      --light-disabled-opacity: 0.3; /* or hint text or icon */
      --light-secondary-opacity: 0.7;
      --light-primary-opacity: 1.0;

    }

  </style>
</custom-style>
`;
    template$7.setAttribute('style', 'display: none;');
    document.head.appendChild(template$7.content);

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */
    const template$8 = html`
<custom-style>
  <style is="custom-style">
    html {
      /*
       * You can use these generic variables in your elements for easy theming.
       * For example, if all your elements use \`--primary-text-color\` as its main
       * color, then switching from a light to a dark theme is just a matter of
       * changing the value of \`--primary-text-color\` in your application.
       */
      --primary-text-color: var(--light-theme-text-color);
      --primary-background-color: var(--light-theme-background-color);
      --secondary-text-color: var(--light-theme-secondary-color);
      --disabled-text-color: var(--light-theme-disabled-color);
      --divider-color: var(--light-theme-divider-color);
      --error-color: var(--paper-deep-orange-a700);

      /*
       * Primary and accent colors. Also see color.js for more colors.
       */
      --primary-color: var(--paper-indigo-500);
      --light-primary-color: var(--paper-indigo-100);
      --dark-primary-color: var(--paper-indigo-700);

      --accent-color: var(--paper-pink-a200);
      --light-accent-color: var(--paper-pink-a100);
      --dark-accent-color: var(--paper-pink-a400);


      /*
       * Material Design Light background theme
       */
      --light-theme-background-color: #ffffff;
      --light-theme-base-color: #000000;
      --light-theme-text-color: var(--paper-grey-900);
      --light-theme-secondary-color: #737373;  /* for secondary text and icons */
      --light-theme-disabled-color: #9b9b9b;  /* disabled/hint text */
      --light-theme-divider-color: #dbdbdb;

      /*
       * Material Design Dark background theme
       */
      --dark-theme-background-color: var(--paper-grey-900);
      --dark-theme-base-color: #ffffff;
      --dark-theme-text-color: #ffffff;
      --dark-theme-secondary-color: #bcbcbc;  /* for secondary text and icons */
      --dark-theme-disabled-color: #646464;  /* disabled/hint text */
      --dark-theme-divider-color: #3c3c3c;

      /*
       * Deprecated values because of their confusing names.
       */
      --text-primary-color: var(--dark-theme-text-color);
      --default-primary-color: var(--primary-color);
    }
  </style>
</custom-style>`;
    template$8.setAttribute('style', 'display: none;');
    document.head.appendChild(template$8.content);

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
      IronFormElementBehavior adds a `name`, `value` and `required` properties to
      a custom element. It mostly exists for backcompatibility with Polymer 1.x, and
      is probably not something you want to use.

      @demo demo/index.html
      @polymerBehavior
     */
    const IronFormElementBehavior = {

      properties: {
        /**
         * The name of this element.
         */
        name: {type: String},

        /**
         * The value for this element.
         * @type {*}
         */
        value: {notify: true, type: String},

        /**
         * Set to true to mark the input as required. If used in a form, a
         * custom element that uses this behavior should also use
         * IronValidatableBehavior and define a custom validation method.
         * Otherwise, a `required` element will always be considered valid.
         * It's also strongly recommended to provide a visual style for the element
         * when its value is invalid.
         */
        required: {type: Boolean, value: false},
      },

      // Empty implementations for backcompatibility.
      attached: function() {},
      detached: function() {}
    };

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
     * Singleton IronMeta instance.
     */
    let IronValidatableBehaviorMeta = null;

    /**
     * `Use IronValidatableBehavior` to implement an element that validates
     * user input. Use the related `IronValidatorBehavior` to add custom
     * validation logic to an iron-input.
     *
     * By default, an `<iron-form>` element validates its fields when the user
     * presses the submit button. To validate a form imperatively, call the form's
     * `validate()` method, which in turn will call `validate()` on all its
     * children. By using `IronValidatableBehavior`, your custom element
     * will get a public `validate()`, which will return the validity of the
     * element, and a corresponding `invalid` attribute, which can be used for
     * styling.
     *
     * To implement the custom validation logic of your element, you must override
     * the protected `_getValidity()` method of this behaviour, rather than
     * `validate()`. See
     * [this](https://github.com/PolymerElements/iron-form/blob/master/demo/simple-element.html)
     * for an example.
     *
     * ### Accessibility
     *
     * Changing the `invalid` property, either manually or by calling `validate()`
     * will update the `aria-invalid` attribute.
     *
     * @demo demo/index.html
     * @polymerBehavior
     */
    const IronValidatableBehavior = {

      properties: {
        /**
         * Name of the validator to use.
         */
        validator: {type: String},

        /**
         * True if the last call to `validate` is invalid.
         */
        invalid: {
          notify: true,
          reflectToAttribute: true,
          type: Boolean,
          value: false,
          observer: '_invalidChanged'
        },
      },

      registered: function() {
        IronValidatableBehaviorMeta = new IronMeta({type: 'validator'});
      },

      _invalidChanged: function() {
        if (this.invalid) {
          this.setAttribute('aria-invalid', 'true');
        } else {
          this.removeAttribute('aria-invalid');
        }
      },

      /* Recompute this every time it's needed, because we don't know if the
       * underlying IronValidatableBehaviorMeta has changed. */
      get _validator() {
        return IronValidatableBehaviorMeta &&
            IronValidatableBehaviorMeta.byKey(this.validator);
      },

      /**
       * @return {boolean} True if the validator `validator` exists.
       */
      hasValidator: function() {
        return this._validator != null;
      },

      /**
       * Returns true if the `value` is valid, and updates `invalid`. If you want
       * your element to have custom validation logic, do not override this method;
       * override `_getValidity(value)` instead.

       * @param {Object} value Deprecated: The value to be validated. By default,
       * it is passed to the validator's `validate()` function, if a validator is
       set.
       * If this argument is not specified, then the element's `value` property
       * is used, if it exists.
       * @return {boolean} True if `value` is valid.
       */
      validate: function(value) {
        // If this is an element that also has a value property, and there was
        // no explicit value argument passed, use the element's property instead.
        if (value === undefined && this.value !== undefined)
          this.invalid = !this._getValidity(this.value);
        else
          this.invalid = !this._getValidity(value);
        return !this.invalid;
      },

      /**
       * Returns true if `value` is valid.  By default, it is passed
       * to the validator's `validate()` function, if a validator is set. You
       * should override this method if you want to implement custom validity
       * logic for your element.
       *
       * @param {Object} value The value to be validated.
       * @return {boolean} True if `value` is valid.
       */

      _getValidity: function(value) {
        if (this.hasValidator()) {
          return this._validator.validate(value);
        }
        return true;
      }
    };

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
     * Use `IronCheckedElementBehavior` to implement a custom element that has a
     * `checked` property, which can be used for validation if the element is also
     * `required`. Element instances implementing this behavior will also be
     * registered for use in an `iron-form` element.
     *
     * @demo demo/index.html
     * @polymerBehavior IronCheckedElementBehavior
     */
    const IronCheckedElementBehaviorImpl = {

      properties: {
        /**
         * Fired when the checked state changes.
         *
         * @event iron-change
         */

        /**
         * Gets or sets the state, `true` is checked and `false` is unchecked.
         */
        checked: {
          type: Boolean,
          value: false,
          reflectToAttribute: true,
          notify: true,
          observer: '_checkedChanged'
        },

        /**
         * If true, the button toggles the active state with each tap or press
         * of the spacebar.
         */
        toggles: {type: Boolean, value: true, reflectToAttribute: true},

        /* Overriden from IronFormElementBehavior */
        value: {type: String, value: 'on', observer: '_valueChanged'}
      },

      observers: ['_requiredChanged(required)'],

      created: function() {
        // Used by `iron-form` to handle the case that an element with this behavior
        // doesn't have a role of 'checkbox' or 'radio', but should still only be
        // included when the form is serialized if `this.checked === true`.
        this._hasIronCheckedElementBehavior = true;
      },

      /**
       * Returns false if the element is required and not checked, and true
       * otherwise.
       * @param {*=} _value Ignored.
       * @return {boolean} true if `required` is false or if `checked` is true.
       */
      _getValidity: function(_value) {
        return this.disabled || !this.required || this.checked;
      },

      /**
       * Update the aria-required label when `required` is changed.
       */
      _requiredChanged: function() {
        if (this.required) {
          this.setAttribute('aria-required', 'true');
        } else {
          this.removeAttribute('aria-required');
        }
      },

      /**
       * Fire `iron-changed` when the checked state changes.
       */
      _checkedChanged: function() {
        this.active = this.checked;
        this.fire('iron-change');
      },

      /**
       * Reset value to 'on' if it is set to `undefined`.
       */
      _valueChanged: function() {
        if (this.value === undefined || this.value === null) {
          this.value = 'on';
        }
      }
    };

    /** @polymerBehavior */
    const IronCheckedElementBehavior = [
      IronFormElementBehavior,
      IronValidatableBehavior,
      IronCheckedElementBehaviorImpl
    ];

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
     * `PaperInkyFocusBehavior` implements a ripple when the element has keyboard
     * focus.
     *
     * @polymerBehavior PaperInkyFocusBehavior
     */
    const PaperInkyFocusBehaviorImpl = {
      observers: ['_focusedChanged(receivedFocusFromKeyboard)'],

      _focusedChanged: function(receivedFocusFromKeyboard) {
        if (receivedFocusFromKeyboard) {
          this.ensureRipple();
        }
        if (this.hasRipple()) {
          this._ripple.holdDown = receivedFocusFromKeyboard;
        }
      },

      _createRipple: function() {
        var ripple = PaperRippleBehavior._createRipple();
        ripple.id = 'ink';
        ripple.setAttribute('center', '');
        ripple.classList.add('circle');
        return ripple;
      }
    };

    /** @polymerBehavior */
    const PaperInkyFocusBehavior = [
      IronButtonState,
      IronControlState,
      PaperRippleBehavior,
      PaperInkyFocusBehaviorImpl
    ];

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
     * Use `PaperCheckedElementBehavior` to implement a custom element that has a
     * `checked` property similar to `IronCheckedElementBehavior` and is compatible
     * with having a ripple effect.
     * @polymerBehavior PaperCheckedElementBehavior
     */
    const PaperCheckedElementBehaviorImpl = {
      /**
       * Synchronizes the element's checked state with its ripple effect.
       */
      _checkedChanged: function() {
        IronCheckedElementBehaviorImpl._checkedChanged.call(this);
        if (this.hasRipple()) {
          if (this.checked) {
            this._ripple.setAttribute('checked', '');
          } else {
            this._ripple.removeAttribute('checked');
          }
        }
      },

      /**
       * Synchronizes the element's `active` and `checked` state.
       */
      _buttonStateChanged: function() {
        PaperRippleBehavior._buttonStateChanged.call(this);
        if (this.disabled) {
          return;
        }
        if (this.isAttached) {
          this.checked = this.active;
        }
      }
    };

    /** @polymerBehavior */
    const PaperCheckedElementBehavior = [
      PaperInkyFocusBehavior,
      IronCheckedElementBehavior,
      PaperCheckedElementBehaviorImpl
    ];

    /**
    @license
    Copyright (c) 2016 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    const template$9 = html`<style>
  :host {
    display: inline-block;
    white-space: nowrap;
    cursor: pointer;
    --calculated-paper-checkbox-size: var(--paper-checkbox-size, 18px);
    /* -1px is a sentinel for the default and is replaced in \`attached\`. */
    --calculated-paper-checkbox-ink-size: var(--paper-checkbox-ink-size, -1px);
    @apply --paper-font-common-base;
    line-height: 0;
    -webkit-tap-highlight-color: transparent;
  }

  :host([hidden]) {
    display: none !important;
  }

  :host(:focus) {
    outline: none;
  }

  .hidden {
    display: none;
  }

  #checkboxContainer {
    display: inline-block;
    position: relative;
    width: var(--calculated-paper-checkbox-size);
    height: var(--calculated-paper-checkbox-size);
    min-width: var(--calculated-paper-checkbox-size);
    margin: var(--paper-checkbox-margin, initial);
    vertical-align: var(--paper-checkbox-vertical-align, middle);
    background-color: var(--paper-checkbox-unchecked-background-color, transparent);
  }

  #ink {
    position: absolute;

    /* Center the ripple in the checkbox by negative offsetting it by
     * (inkWidth - rippleWidth) / 2 */
    top: calc(0px - (var(--calculated-paper-checkbox-ink-size) - var(--calculated-paper-checkbox-size)) / 2);
    left: calc(0px - (var(--calculated-paper-checkbox-ink-size) - var(--calculated-paper-checkbox-size)) / 2);
    width: var(--calculated-paper-checkbox-ink-size);
    height: var(--calculated-paper-checkbox-ink-size);
    color: var(--paper-checkbox-unchecked-ink-color, var(--primary-text-color));
    opacity: 0.6;
    pointer-events: none;
  }

  #ink:dir(rtl) {
    right: calc(0px - (var(--calculated-paper-checkbox-ink-size) - var(--calculated-paper-checkbox-size)) / 2);
    left: auto;
  }

  #ink[checked] {
    color: var(--paper-checkbox-checked-ink-color, var(--primary-color));
  }

  #checkbox {
    position: relative;
    box-sizing: border-box;
    height: 100%;
    border: solid 2px;
    border-color: var(--paper-checkbox-unchecked-color, var(--primary-text-color));
    border-radius: 2px;
    pointer-events: none;
    -webkit-transition: background-color 140ms, border-color 140ms;
    transition: background-color 140ms, border-color 140ms;

    -webkit-transition-duration: var(--paper-checkbox-animation-duration, 140ms);
    transition-duration: var(--paper-checkbox-animation-duration, 140ms);
  }

  /* checkbox checked animations */
  #checkbox.checked #checkmark {
    -webkit-animation: checkmark-expand 140ms ease-out forwards;
    animation: checkmark-expand 140ms ease-out forwards;

    -webkit-animation-duration: var(--paper-checkbox-animation-duration, 140ms);
    animation-duration: var(--paper-checkbox-animation-duration, 140ms);
  }

  @-webkit-keyframes checkmark-expand {
    0% {
      -webkit-transform: scale(0, 0) rotate(45deg);
    }
    100% {
      -webkit-transform: scale(1, 1) rotate(45deg);
    }
  }

  @keyframes checkmark-expand {
    0% {
      transform: scale(0, 0) rotate(45deg);
    }
    100% {
      transform: scale(1, 1) rotate(45deg);
    }
  }

  #checkbox.checked {
    background-color: var(--paper-checkbox-checked-color, var(--primary-color));
    border-color: var(--paper-checkbox-checked-color, var(--primary-color));
  }

  #checkmark {
    position: absolute;
    width: 36%;
    height: 70%;
    border-style: solid;
    border-top: none;
    border-left: none;
    border-right-width: calc(2/15 * var(--calculated-paper-checkbox-size));
    border-bottom-width: calc(2/15 * var(--calculated-paper-checkbox-size));
    border-color: var(--paper-checkbox-checkmark-color, white);
    -webkit-transform-origin: 97% 86%;
    transform-origin: 97% 86%;
    box-sizing: content-box; /* protect against page-level box-sizing */
  }

  #checkmark:dir(rtl) {
    -webkit-transform-origin: 50% 14%;
    transform-origin: 50% 14%;
  }

  /* label */
  #checkboxLabel {
    position: relative;
    display: inline-block;
    vertical-align: middle;
    padding-left: var(--paper-checkbox-label-spacing, 8px);
    white-space: normal;
    line-height: normal;
    color: var(--paper-checkbox-label-color, var(--primary-text-color));
    @apply --paper-checkbox-label;
  }

  :host([checked]) #checkboxLabel {
    color: var(--paper-checkbox-label-checked-color, var(--paper-checkbox-label-color, var(--primary-text-color)));
    @apply --paper-checkbox-label-checked;
  }

  #checkboxLabel:dir(rtl) {
    padding-right: var(--paper-checkbox-label-spacing, 8px);
    padding-left: 0;
  }

  #checkboxLabel[hidden] {
    display: none;
  }

  /* disabled state */

  :host([disabled]) #checkbox {
    opacity: 0.5;
    border-color: var(--paper-checkbox-unchecked-color, var(--primary-text-color));
  }

  :host([disabled][checked]) #checkbox {
    background-color: var(--paper-checkbox-unchecked-color, var(--primary-text-color));
    opacity: 0.5;
  }

  :host([disabled]) #checkboxLabel  {
    opacity: 0.65;
  }

  /* invalid state */
  #checkbox.invalid:not(.checked) {
    border-color: var(--paper-checkbox-error-color, var(--error-color));
  }
</style>

<div id="checkboxContainer">
  <div id="checkbox" class$="[[_computeCheckboxClass(checked, invalid)]]">
    <div id="checkmark" class$="[[_computeCheckmarkClass(checked)]]"></div>
  </div>
</div>

<div id="checkboxLabel"><slot></slot></div>`;
    template$9.setAttribute('strip-whitespace', '');

    /**
    Material design:
    [Checkbox](https://www.google.com/design/spec/components/selection-controls.html#selection-controls-checkbox)

    `paper-checkbox` is a button that can be either checked or unchecked. User can
    tap the checkbox to check or uncheck it. Usually you use checkboxes to allow
    user to select multiple options from a set. If you have a single ON/OFF option,
    avoid using a single checkbox and use `paper-toggle-button` instead.

    Example:

        <paper-checkbox>label</paper-checkbox>

        <paper-checkbox checked> label</paper-checkbox>

    ### Styling

    The following custom properties and mixins are available for styling:

    Custom property | Description | Default
    ----------------|-------------|----------
    `--paper-checkbox-unchecked-background-color` | Checkbox background color when the input is not checked | `transparent`
    `--paper-checkbox-unchecked-color` | Checkbox border color when the input is not checked | `--primary-text-color`
    `--paper-checkbox-unchecked-ink-color` | Selected/focus ripple color when the input is not checked | `--primary-text-color`
    `--paper-checkbox-checked-color` | Checkbox color when the input is checked | `--primary-color`
    `--paper-checkbox-checked-ink-color` | Selected/focus ripple color when the input is checked | `--primary-color`
    `--paper-checkbox-checkmark-color` | Checkmark color | `white`
    `--paper-checkbox-label-color` | Label color | `--primary-text-color`
    `--paper-checkbox-label-checked-color` | Label color when the input is checked | `--paper-checkbox-label-color`
    `--paper-checkbox-label-spacing` | Spacing between the label and the checkbox | `8px`
    `--paper-checkbox-label` | Mixin applied to the label | `{}`
    `--paper-checkbox-label-checked` | Mixin applied to the label when the input is checked | `{}`
    `--paper-checkbox-error-color` | Checkbox color when invalid | `--error-color`
    `--paper-checkbox-size` | Size of the checkbox | `18px`
    `--paper-checkbox-ink-size` | Size of the ripple | `48px`
    `--paper-checkbox-margin` | Margin around the checkbox container | `initial`
    `--paper-checkbox-vertical-align` | Vertical alignment of the checkbox container | `middle`

    This element applies the mixin `--paper-font-common-base` but does not import
    `paper-styles/typography.html`. In order to apply the `Roboto` font to this
    element, make sure you've imported `paper-styles/typography.html`.

    @demo demo/index.html
    */
    Polymer({
      _template: template$9,

      is: 'paper-checkbox',

      behaviors: [PaperCheckedElementBehavior],

      /** @private */
      hostAttributes: {role: 'checkbox', 'aria-checked': false, tabindex: 0},

      properties: {
        /**
         * Fired when the checked state changes due to user interaction.
         *
         * @event change
         */

        /**
         * Fired when the checked state changes.
         *
         * @event iron-change
         */
        ariaActiveAttribute: {type: String, value: 'aria-checked'}
      },

      attached: function() {
        // Wait until styles have resolved to check for the default sentinel.
        // See polymer#4009 for more details.
        afterNextRender(this, function() {
          var inkSize =
              this.getComputedStyleValue('--calculated-paper-checkbox-ink-size')
                  .trim();
          // If unset, compute and set the default `--paper-checkbox-ink-size`.
          if (inkSize === '-1px') {
            var checkboxSizeText =
                this.getComputedStyleValue('--calculated-paper-checkbox-size')
                    .trim();

            var units = 'px';
            var unitsMatches = checkboxSizeText.match(/[A-Za-z]+$/);
            if (unitsMatches !== null) {
              units = unitsMatches[0];
            }

            var checkboxSize = parseFloat(checkboxSizeText);
            var defaultInkSize = (8 / 3) * checkboxSize;

            if (units === 'px') {
              defaultInkSize = Math.floor(defaultInkSize);

              // The checkbox and ripple need to have the same parity so that their
              // centers align.
              if (defaultInkSize % 2 !== checkboxSize % 2) {
                defaultInkSize++;
              }
            }

            this.updateStyles({
              '--paper-checkbox-ink-size': defaultInkSize + units,
            });
          }
        });
      },

      _computeCheckboxClass: function(checked, invalid) {
        var className = '';
        if (checked) {
          className += 'checked ';
        }
        if (invalid) {
          className += 'invalid';
        }
        return className;
      },

      _computeCheckmarkClass: function(checked) {
        return checked ? '' : 'hidden';
      },

      // create ripple inside the checkboxContainer
      _createRipple: function() {
        this._rippleContainer = this.$.checkboxContainer;
        return PaperInkyFocusBehaviorImpl._createRipple.call(this);
      }

    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
    `Polymer.IronFitBehavior` fits an element in another element using `max-height`
    and `max-width`, and optionally centers it in the window or another element.

    The element will only be sized and/or positioned if it has not already been
    sized and/or positioned by CSS.

    CSS properties            | Action
    --------------------------|-------------------------------------------
    `position` set            | Element is not centered horizontally or vertically
    `top` or `bottom` set     | Element is not vertically centered
    `left` or `right` set     | Element is not horizontally centered
    `max-height` set          | Element respects `max-height`
    `max-width` set           | Element respects `max-width`

    `Polymer.IronFitBehavior` can position an element into another element using
    `verticalAlign` and `horizontalAlign`. This will override the element's css
    position.

        <div class="container">
          <iron-fit-impl vertical-align="top" horizontal-align="auto">
            Positioned into the container
          </iron-fit-impl>
        </div>

    Use `noOverlap` to position the element around another element without
    overlapping it.

        <div class="container">
          <iron-fit-impl no-overlap vertical-align="auto" horizontal-align="auto">
            Positioned around the container
          </iron-fit-impl>
        </div>

    Use `horizontalOffset, verticalOffset` to offset the element from its
    `positionTarget`; `Polymer.IronFitBehavior` will collapse these in order to
    keep the element within `fitInto` boundaries, while preserving the element's
    CSS margin values.

        <div class="container">
          <iron-fit-impl vertical-align="top" vertical-offset="20">
            With vertical offset
          </iron-fit-impl>
        </div>

    @demo demo/index.html
    @polymerBehavior
    */
    const IronFitBehavior = {

      properties: {

        /**
         * The element that will receive a `max-height`/`width`. By default it is
         * the same as `this`, but it can be set to a child element. This is useful,
         * for example, for implementing a scrolling region inside the element.
         * @type {!Element}
         */
        sizingTarget: {
          type: Object,
          value: function() {
            return this;
          }
        },

        /**
         * The element to fit `this` into.
         */
        fitInto: {type: Object, value: window},

        /**
         * Will position the element around the positionTarget without overlapping
         * it.
         */
        noOverlap: {type: Boolean},

        /**
         * The element that should be used to position the element. If not set, it
         * will default to the parent node.
         * @type {!Element}
         */
        positionTarget: {type: Element},

        /**
         * The orientation against which to align the element horizontally
         * relative to the `positionTarget`. Possible values are "left", "right",
         * "center", "auto".
         */
        horizontalAlign: {type: String},

        /**
         * The orientation against which to align the element vertically
         * relative to the `positionTarget`. Possible values are "top", "bottom",
         * "middle", "auto".
         */
        verticalAlign: {type: String},

        /**
         * If true, it will use `horizontalAlign` and `verticalAlign` values as
         * preferred alignment and if there's not enough space, it will pick the
         * values which minimize the cropping.
         */
        dynamicAlign: {type: Boolean},

        /**
         * A pixel value that will be added to the position calculated for the
         * given `horizontalAlign`, in the direction of alignment. You can think
         * of it as increasing or decreasing the distance to the side of the
         * screen given by `horizontalAlign`.
         *
         * If `horizontalAlign` is "left" or "center", this offset will increase or
         * decrease the distance to the left side of the screen: a negative offset
         * will move the dropdown to the left; a positive one, to the right.
         *
         * Conversely if `horizontalAlign` is "right", this offset will increase
         * or decrease the distance to the right side of the screen: a negative
         * offset will move the dropdown to the right; a positive one, to the left.
         */
        horizontalOffset: {type: Number, value: 0, notify: true},

        /**
         * A pixel value that will be added to the position calculated for the
         * given `verticalAlign`, in the direction of alignment. You can think
         * of it as increasing or decreasing the distance to the side of the
         * screen given by `verticalAlign`.
         *
         * If `verticalAlign` is "top" or "middle", this offset will increase or
         * decrease the distance to the top side of the screen: a negative offset
         * will move the dropdown upwards; a positive one, downwards.
         *
         * Conversely if `verticalAlign` is "bottom", this offset will increase
         * or decrease the distance to the bottom side of the screen: a negative
         * offset will move the dropdown downwards; a positive one, upwards.
         */
        verticalOffset: {type: Number, value: 0, notify: true},

        /**
         * Set to true to auto-fit on attach.
         */
        autoFitOnAttach: {type: Boolean, value: false},

        /** @type {?Object} */
        _fitInfo: {type: Object}
      },

      get _fitWidth() {
        var fitWidth;
        if (this.fitInto === window) {
          fitWidth = this.fitInto.innerWidth;
        } else {
          fitWidth = this.fitInto.getBoundingClientRect().width;
        }
        return fitWidth;
      },

      get _fitHeight() {
        var fitHeight;
        if (this.fitInto === window) {
          fitHeight = this.fitInto.innerHeight;
        } else {
          fitHeight = this.fitInto.getBoundingClientRect().height;
        }
        return fitHeight;
      },

      get _fitLeft() {
        var fitLeft;
        if (this.fitInto === window) {
          fitLeft = 0;
        } else {
          fitLeft = this.fitInto.getBoundingClientRect().left;
        }
        return fitLeft;
      },

      get _fitTop() {
        var fitTop;
        if (this.fitInto === window) {
          fitTop = 0;
        } else {
          fitTop = this.fitInto.getBoundingClientRect().top;
        }
        return fitTop;
      },

      /**
       * The element that should be used to position the element,
       * if no position target is configured.
       */
      get _defaultPositionTarget() {
        var parent = dom(this).parentNode;

        if (parent && parent.nodeType === Node.DOCUMENT_FRAGMENT_NODE) {
          parent = parent.host;
        }

        return parent;
      },

      /**
       * The horizontal align value, accounting for the RTL/LTR text direction.
       */
      get _localeHorizontalAlign() {
        if (this._isRTL) {
          // In RTL, "left" becomes "right".
          if (this.horizontalAlign === 'right') {
            return 'left';
          }
          if (this.horizontalAlign === 'left') {
            return 'right';
          }
        }
        return this.horizontalAlign;
      },

      /**
       * True if the element should be positioned instead of centered.
       * @private
       */
      get __shouldPosition() {
        return (this.horizontalAlign || this.verticalAlign) && this.positionTarget;
      },

      /**
       * True if the component is RTL.
       * @private
       */
      get _isRTL() {
        // Memoize this to avoid expensive calculations & relayouts.
        // Make sure we do it only once
        if (typeof this._memoizedIsRTL === 'undefined') {
          this._memoizedIsRTL = window.getComputedStyle(this).direction == 'rtl';
        }
        return this._memoizedIsRTL;
      },

      /** @override */
      attached: function() {
        this.positionTarget = this.positionTarget || this._defaultPositionTarget;
        if (this.autoFitOnAttach) {
          if (window.getComputedStyle(this).display === 'none') {
            setTimeout(function() {
              this.fit();
            }.bind(this));
          } else {
            // NOTE: shadydom applies distribution asynchronously
            // for performance reasons webcomponents/shadydom#120
            // Flush to get correct layout info.
            window.ShadyDOM && ShadyDOM.flush();
            this.fit();
          }
        }
      },

      /** @override */
      detached: function() {
        if (this.__deferredFit) {
          clearTimeout(this.__deferredFit);
          this.__deferredFit = null;
        }
      },

      /**
       * Positions and fits the element into the `fitInto` element.
       */
      fit: function() {
        this.position();
        this.constrain();
        this.center();
      },

      /**
       * Memoize information needed to position and size the target element.
       * @suppress {deprecated}
       */
      _discoverInfo: function() {
        if (this._fitInfo) {
          return;
        }
        var target = window.getComputedStyle(this);
        var sizer = window.getComputedStyle(this.sizingTarget);

        this._fitInfo = {
          inlineStyle: {
            top: this.style.top || '',
            left: this.style.left || '',
            position: this.style.position || ''
          },
          sizerInlineStyle: {
            maxWidth: this.sizingTarget.style.maxWidth || '',
            maxHeight: this.sizingTarget.style.maxHeight || '',
            boxSizing: this.sizingTarget.style.boxSizing || ''
          },
          positionedBy: {
            vertically: target.top !== 'auto' ?
                'top' :
                (target.bottom !== 'auto' ? 'bottom' : null),
            horizontally: target.left !== 'auto' ?
                'left' :
                (target.right !== 'auto' ? 'right' : null)
          },
          sizedBy: {
            height: sizer.maxHeight !== 'none',
            width: sizer.maxWidth !== 'none',
            minWidth: parseInt(sizer.minWidth, 10) || 0,
            minHeight: parseInt(sizer.minHeight, 10) || 0
          },
          margin: {
            top: parseInt(target.marginTop, 10) || 0,
            right: parseInt(target.marginRight, 10) || 0,
            bottom: parseInt(target.marginBottom, 10) || 0,
            left: parseInt(target.marginLeft, 10) || 0
          }
        };
      },

      /**
       * Resets the target element's position and size constraints, and clear
       * the memoized data.
       */
      resetFit: function() {
        var info = this._fitInfo || {};
        for (var property in info.sizerInlineStyle) {
          this.sizingTarget.style[property] = info.sizerInlineStyle[property];
        }
        for (var property in info.inlineStyle) {
          this.style[property] = info.inlineStyle[property];
        }

        this._fitInfo = null;
      },

      /**
       * Equivalent to calling `resetFit()` and `fit()`. Useful to call this after
       * the element or the `fitInto` element has been resized, or if any of the
       * positioning properties (e.g. `horizontalAlign, verticalAlign`) is updated.
       * It preserves the scroll position of the sizingTarget.
       */
      refit: function() {
        var scrollLeft = this.sizingTarget.scrollLeft;
        var scrollTop = this.sizingTarget.scrollTop;
        this.resetFit();
        this.fit();
        this.sizingTarget.scrollLeft = scrollLeft;
        this.sizingTarget.scrollTop = scrollTop;
      },

      /**
       * Positions the element according to `horizontalAlign, verticalAlign`.
       */
      position: function() {
        if (!this.__shouldPosition) {
          // needs to be centered, and it is done after constrain.
          return;
        }
        this._discoverInfo();

        this.style.position = 'fixed';
        // Need border-box for margin/padding.
        this.sizingTarget.style.boxSizing = 'border-box';
        // Set to 0, 0 in order to discover any offset caused by parent stacking
        // contexts.
        this.style.left = '0px';
        this.style.top = '0px';

        var rect = this.getBoundingClientRect();
        var positionRect = this.__getNormalizedRect(this.positionTarget);
        var fitRect = this.__getNormalizedRect(this.fitInto);

        var margin = this._fitInfo.margin;

        // Consider the margin as part of the size for position calculations.
        var size = {
          width: rect.width + margin.left + margin.right,
          height: rect.height + margin.top + margin.bottom
        };

        var position = this.__getPosition(
            this._localeHorizontalAlign,
            this.verticalAlign,
            size,
            rect,
            positionRect,
            fitRect);

        var left = position.left + margin.left;
        var top = position.top + margin.top;

        // We first limit right/bottom within fitInto respecting the margin,
        // then use those values to limit top/left.
        var right = Math.min(fitRect.right - margin.right, left + rect.width);
        var bottom = Math.min(fitRect.bottom - margin.bottom, top + rect.height);

        // Keep left/top within fitInto respecting the margin.
        left = Math.max(
            fitRect.left + margin.left,
            Math.min(left, right - this._fitInfo.sizedBy.minWidth));
        top = Math.max(
            fitRect.top + margin.top,
            Math.min(top, bottom - this._fitInfo.sizedBy.minHeight));

        // Use right/bottom to set maxWidth/maxHeight, and respect
        // minWidth/minHeight.
        this.sizingTarget.style.maxWidth =
            Math.max(right - left, this._fitInfo.sizedBy.minWidth) + 'px';
        this.sizingTarget.style.maxHeight =
            Math.max(bottom - top, this._fitInfo.sizedBy.minHeight) + 'px';

        // Remove the offset caused by any stacking context.
        this.style.left = (left - rect.left) + 'px';
        this.style.top = (top - rect.top) + 'px';
      },

      /**
       * Constrains the size of the element to `fitInto` by setting `max-height`
       * and/or `max-width`.
       */
      constrain: function() {
        if (this.__shouldPosition) {
          return;
        }
        this._discoverInfo();

        var info = this._fitInfo;
        // position at (0px, 0px) if not already positioned, so we can measure the
        // natural size.
        if (!info.positionedBy.vertically) {
          this.style.position = 'fixed';
          this.style.top = '0px';
        }
        if (!info.positionedBy.horizontally) {
          this.style.position = 'fixed';
          this.style.left = '0px';
        }

        // need border-box for margin/padding
        this.sizingTarget.style.boxSizing = 'border-box';
        // constrain the width and height if not already set
        var rect = this.getBoundingClientRect();
        if (!info.sizedBy.height) {
          this.__sizeDimension(
              rect, info.positionedBy.vertically, 'top', 'bottom', 'Height');
        }
        if (!info.sizedBy.width) {
          this.__sizeDimension(
              rect, info.positionedBy.horizontally, 'left', 'right', 'Width');
        }
      },

      /**
       * @protected
       * @deprecated
       */
      _sizeDimension: function(rect, positionedBy, start, end, extent) {
        this.__sizeDimension(rect, positionedBy, start, end, extent);
      },

      /**
       * @private
       */
      __sizeDimension: function(rect, positionedBy, start, end, extent) {
        var info = this._fitInfo;
        var fitRect = this.__getNormalizedRect(this.fitInto);
        var max = extent === 'Width' ? fitRect.width : fitRect.height;
        var flip = (positionedBy === end);
        var offset = flip ? max - rect[end] : rect[start];
        var margin = info.margin[flip ? start : end];
        var offsetExtent = 'offset' + extent;
        var sizingOffset = this[offsetExtent] - this.sizingTarget[offsetExtent];
        this.sizingTarget.style['max' + extent] =
            (max - margin - offset - sizingOffset) + 'px';
      },

      /**
       * Centers horizontally and vertically if not already positioned. This also
       * sets `position:fixed`.
       */
      center: function() {
        if (this.__shouldPosition) {
          return;
        }
        this._discoverInfo();

        var positionedBy = this._fitInfo.positionedBy;
        if (positionedBy.vertically && positionedBy.horizontally) {
          // Already positioned.
          return;
        }
        // Need position:fixed to center
        this.style.position = 'fixed';
        // Take into account the offset caused by parents that create stacking
        // contexts (e.g. with transform: translate3d). Translate to 0,0 and
        // measure the bounding rect.
        if (!positionedBy.vertically) {
          this.style.top = '0px';
        }
        if (!positionedBy.horizontally) {
          this.style.left = '0px';
        }
        // It will take in consideration margins and transforms
        var rect = this.getBoundingClientRect();
        var fitRect = this.__getNormalizedRect(this.fitInto);
        if (!positionedBy.vertically) {
          var top = fitRect.top - rect.top + (fitRect.height - rect.height) / 2;
          this.style.top = top + 'px';
        }
        if (!positionedBy.horizontally) {
          var left = fitRect.left - rect.left + (fitRect.width - rect.width) / 2;
          this.style.left = left + 'px';
        }
      },

      __getNormalizedRect: function(target) {
        if (target === document.documentElement || target === window) {
          return {
            top: 0,
            left: 0,
            width: window.innerWidth,
            height: window.innerHeight,
            right: window.innerWidth,
            bottom: window.innerHeight
          };
        }
        return target.getBoundingClientRect();
      },

      __getOffscreenArea: function(position, size, fitRect) {
        var verticalCrop = Math.min(0, position.top) +
            Math.min(0, fitRect.bottom - (position.top + size.height));
        var horizontalCrop = Math.min(0, position.left) +
            Math.min(0, fitRect.right - (position.left + size.width));
        return Math.abs(verticalCrop) * size.width +
            Math.abs(horizontalCrop) * size.height;
      },


      __getPosition: function(
          hAlign, vAlign, size, sizeNoMargins, positionRect, fitRect) {
        // All the possible configurations.
        // Ordered as top-left, top-right, bottom-left, bottom-right.
        var positions = [
          {
            verticalAlign: 'top',
            horizontalAlign: 'left',
            top: positionRect.top + this.verticalOffset,
            left: positionRect.left + this.horizontalOffset
          },
          {
            verticalAlign: 'top',
            horizontalAlign: 'right',
            top: positionRect.top + this.verticalOffset,
            left: positionRect.right - size.width - this.horizontalOffset
          },
          {
            verticalAlign: 'bottom',
            horizontalAlign: 'left',
            top: positionRect.bottom - size.height - this.verticalOffset,
            left: positionRect.left + this.horizontalOffset
          },
          {
            verticalAlign: 'bottom',
            horizontalAlign: 'right',
            top: positionRect.bottom - size.height - this.verticalOffset,
            left: positionRect.right - size.width - this.horizontalOffset
          }
        ];

        if (this.noOverlap) {
          // Duplicate.
          for (var i = 0, l = positions.length; i < l; i++) {
            var copy = {};
            for (var key in positions[i]) {
              copy[key] = positions[i][key];
            }
            positions.push(copy);
          }
          // Horizontal overlap only.
          positions[0].top = positions[1].top += positionRect.height;
          positions[2].top = positions[3].top -= positionRect.height;
          // Vertical overlap only.
          positions[4].left = positions[6].left += positionRect.width;
          positions[5].left = positions[7].left -= positionRect.width;
        }

        // Consider auto as null for coding convenience.
        vAlign = vAlign === 'auto' ? null : vAlign;
        hAlign = hAlign === 'auto' ? null : hAlign;

        if (!hAlign || hAlign === 'center') {
          positions.push({
            verticalAlign: 'top',
            horizontalAlign: 'center',
            top: positionRect.top + this.verticalOffset +
                (this.noOverlap ? positionRect.height : 0),
            left: positionRect.left - sizeNoMargins.width / 2 +
                positionRect.width / 2 + this.horizontalOffset
          });
          positions.push({
            verticalAlign: 'bottom',
            horizontalAlign: 'center',
            top: positionRect.bottom - size.height - this.verticalOffset -
                (this.noOverlap ? positionRect.height : 0),
            left: positionRect.left - sizeNoMargins.width / 2 +
                positionRect.width / 2 + this.horizontalOffset
          });
        }

        if (!vAlign || vAlign === 'middle') {
          positions.push({
            verticalAlign: 'middle',
            horizontalAlign: 'left',
            top: positionRect.top - sizeNoMargins.height / 2 +
                positionRect.height / 2 + this.verticalOffset,
            left: positionRect.left + this.horizontalOffset +
                (this.noOverlap ? positionRect.width : 0)
          });
          positions.push({
            verticalAlign: 'middle',
            horizontalAlign: 'right',
            top: positionRect.top - sizeNoMargins.height / 2 +
                positionRect.height / 2 + this.verticalOffset,
            left: positionRect.right - size.width - this.horizontalOffset -
                (this.noOverlap ? positionRect.width : 0)
          });
        }

        if (vAlign === 'middle' && hAlign === 'center') {
          positions.push({
            verticalAlign: 'middle',
            horizontalAlign: 'center',
            top: positionRect.top - sizeNoMargins.height / 2 +
                positionRect.height / 2 + this.verticalOffset,
            left: positionRect.left - sizeNoMargins.width / 2 +
                positionRect.width / 2 + this.horizontalOffset
          });
        }

        var position;
        for (var i = 0; i < positions.length; i++) {
          var candidate = positions[i];
          var vAlignOk = candidate.verticalAlign === vAlign;
          var hAlignOk = candidate.horizontalAlign === hAlign;

          // If both vAlign and hAlign are defined, return exact match.
          // For dynamicAlign and noOverlap we'll have more than one candidate, so
          // we'll have to check the offscreenArea to make the best choice.
          if (!this.dynamicAlign && !this.noOverlap && vAlignOk && hAlignOk) {
            position = candidate;
            break;
          }

          // Align is ok if alignment preferences are respected. If no preferences,
          // it is considered ok.
          var alignOk = (!vAlign || vAlignOk) && (!hAlign || hAlignOk);

          // Filter out elements that don't match the alignment (if defined).
          // With dynamicAlign, we need to consider all the positions to find the
          // one that minimizes the cropped area.
          if (!this.dynamicAlign && !alignOk) {
            continue;
          }

          candidate.offscreenArea =
              this.__getOffscreenArea(candidate, size, fitRect);
          // If not cropped and respects the align requirements, keep it.
          // This allows to prefer positions overlapping horizontally over the
          // ones overlapping vertically.
          if (candidate.offscreenArea === 0 && alignOk) {
            position = candidate;
            break;
          }
          position = position || candidate;
          var diff = candidate.offscreenArea - position.offscreenArea;
          // Check which crops less. If it crops equally, check if at least one
          // align setting is ok.
          if (diff < 0 || (diff === 0 && (vAlignOk || hAlignOk))) {
            position = candidate;
          }
        }

        return position;
      }

    };

    /**
    @license
    Copyright (c) 2016 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    var p$1 = Element.prototype;
    var matches$1 = p$1.matches || p$1.matchesSelector || p$1.mozMatchesSelector ||
        p$1.msMatchesSelector || p$1.oMatchesSelector || p$1.webkitMatchesSelector;

    class IronFocusablesHelperClass {
      /**
       * Returns a sorted array of tabbable nodes, including the root node.
       * It searches the tabbable nodes in the light and shadow dom of the chidren,
       * sorting the result by tabindex.
       * @param {!Node} node
       * @return {!Array<!HTMLElement>}
       */
      getTabbableNodes(node) {
        var result = [];
        // If there is at least one element with tabindex > 0, we need to sort
        // the final array by tabindex.
        var needsSortByTabIndex = this._collectTabbableNodes(node, result);
        if (needsSortByTabIndex) {
          return this._sortByTabIndex(result);
        }
        return result;
      }

      /**
       * Returns if a element is focusable.
       * @param {!HTMLElement} element
       * @return {boolean}
       */
      isFocusable(element) {
        // From http://stackoverflow.com/a/1600194/4228703:
        // There isn't a definite list, it's up to the browser. The only
        // standard we have is DOM Level 2 HTML
        // https://www.w3.org/TR/DOM-Level-2-HTML/html.html, according to which the
        // only elements that have a focus() method are HTMLInputElement,
        // HTMLSelectElement, HTMLTextAreaElement and HTMLAnchorElement. This
        // notably omits HTMLButtonElement and HTMLAreaElement. Referring to these
        // tests with tabbables in different browsers
        // http://allyjs.io/data-tables/focusable.html

        // Elements that cannot be focused if they have [disabled] attribute.
        if (matches$1.call(element, 'input, select, textarea, button, object')) {
          return matches$1.call(element, ':not([disabled])');
        }
        // Elements that can be focused even if they have [disabled] attribute.
        return matches$1.call(
            element, 'a[href], area[href], iframe, [tabindex], [contentEditable]');
      }

      /**
       * Returns if a element is tabbable. To be tabbable, a element must be
       * focusable, visible, and with a tabindex !== -1.
       * @param {!HTMLElement} element
       * @return {boolean}
       */
      isTabbable(element) {
        return this.isFocusable(element) &&
            matches$1.call(element, ':not([tabindex="-1"])') &&
            this._isVisible(element);
      }

      /**
       * Returns the normalized element tabindex. If not focusable, returns -1.
       * It checks for the attribute "tabindex" instead of the element property
       * `tabIndex` since browsers assign different values to it.
       * e.g. in Firefox `<div contenteditable>` has `tabIndex = -1`
       * @param {!HTMLElement} element
       * @return {!number}
       * @private
       */
      _normalizedTabIndex(element) {
        if (this.isFocusable(element)) {
          var tabIndex = element.getAttribute('tabindex') || 0;
          return Number(tabIndex);
        }
        return -1;
      }

      /**
       * Searches for nodes that are tabbable and adds them to the `result` array.
       * Returns if the `result` array needs to be sorted by tabindex.
       * @param {!Node} node The starting point for the search; added to `result`
       * if tabbable.
       * @param {!Array<!HTMLElement>} result
       * @return {boolean}
       * @private
       */
      _collectTabbableNodes(node, result) {
        // If not an element or not visible, no need to explore children.
        if (node.nodeType !== Node.ELEMENT_NODE) {
          return false;
        }
        var element = /** @type {!HTMLElement} */ (node);
        if (!this._isVisible(element)) {
          return false;
        }
        var tabIndex = this._normalizedTabIndex(element);
        var needsSort = tabIndex > 0;
        if (tabIndex >= 0) {
          result.push(element);
        }
        // In ShadowDOM v1, tab order is affected by the order of distrubution.
        // E.g. getTabbableNodes(#root) in ShadowDOM v1 should return [#A, #B];
        // in ShadowDOM v0 tab order is not affected by the distrubution order,
        // in fact getTabbableNodes(#root) returns [#B, #A].
        //  <div id="root">
        //   <!-- shadow -->
        //     <slot name="a">
        //     <slot name="b">
        //   <!-- /shadow -->
        //   <input id="A" slot="a">
        //   <input id="B" slot="b" tabindex="1">
        //  </div>
        // TODO(valdrin) support ShadowDOM v1 when upgrading to Polymer v2.0.
        var children;
        if (element.localName === 'content' || element.localName === 'slot') {
          children = dom(element).getDistributedNodes();
        } else {
          // Use shadow root if possible, will check for distributed nodes.
          children = dom(element.root || element).children;
        }
        for (var i = 0; i < children.length; i++) {
          // Ensure method is always invoked to collect tabbable children.
          needsSort = this._collectTabbableNodes(children[i], result) || needsSort;
        }
        return needsSort;
      }

      /**
       * Returns false if the element has `visibility: hidden` or `display: none`
       * @param {!HTMLElement} element
       * @return {boolean}
       * @private
       */
      _isVisible(element) {
        // Check inline style first to save a re-flow. If looks good, check also
        // computed style.
        var style = element.style;
        if (style.visibility !== 'hidden' && style.display !== 'none') {
          style = window.getComputedStyle(element);
          return (style.visibility !== 'hidden' && style.display !== 'none');
        }
        return false;
      }

      /**
       * Sorts an array of tabbable elements by tabindex. Returns a new array.
       * @param {!Array<!HTMLElement>} tabbables
       * @return {!Array<!HTMLElement>}
       * @private
       */
      _sortByTabIndex(tabbables) {
        // Implement a merge sort as Array.prototype.sort does a non-stable sort
        // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/sort
        var len = tabbables.length;
        if (len < 2) {
          return tabbables;
        }
        var pivot = Math.ceil(len / 2);
        var left = this._sortByTabIndex(tabbables.slice(0, pivot));
        var right = this._sortByTabIndex(tabbables.slice(pivot));
        return this._mergeSortByTabIndex(left, right);
      }

      /**
       * Merge sort iterator, merges the two arrays into one, sorted by tab index.
       * @param {!Array<!HTMLElement>} left
       * @param {!Array<!HTMLElement>} right
       * @return {!Array<!HTMLElement>}
       * @private
       */
      _mergeSortByTabIndex(left, right) {
        var result = [];
        while ((left.length > 0) && (right.length > 0)) {
          if (this._hasLowerTabOrder(left[0], right[0])) {
            result.push(right.shift());
          } else {
            result.push(left.shift());
          }
        }

        return result.concat(left, right);
      }

      /**
       * Returns if element `a` has lower tab order compared to element `b`
       * (both elements are assumed to be focusable and tabbable).
       * Elements with tabindex = 0 have lower tab order compared to elements
       * with tabindex > 0.
       * If both have same tabindex, it returns false.
       * @param {!HTMLElement} a
       * @param {!HTMLElement} b
       * @return {boolean}
       * @private
       */
      _hasLowerTabOrder(a, b) {
        // Normalize tabIndexes
        // e.g. in Firefox `<div contenteditable>` has `tabIndex = -1`
        var ati = Math.max(a.tabIndex, 0);
        var bti = Math.max(b.tabIndex, 0);
        return (ati === 0 || bti === 0) ? bti > ati : ati > bti;
      }
    }

    const IronFocusablesHelper = new IronFocusablesHelperClass();

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /*
    `iron-overlay-backdrop` is a backdrop used by `Polymer.IronOverlayBehavior`. It
    should be a singleton.

    ### Styling

    The following custom properties and mixins are available for styling.

    Custom property | Description | Default
    -------------------------------------------|------------------------|---------
    `--iron-overlay-backdrop-background-color` | Backdrop background color | #000
    `--iron-overlay-backdrop-opacity`          | Backdrop opacity | 0.6
    `--iron-overlay-backdrop`                  | Mixin applied to `iron-overlay-backdrop`.                      | {}
    `--iron-overlay-backdrop-opened`           | Mixin applied to `iron-overlay-backdrop` when it is displayed | {}
    */
    Polymer({
      /** @override */
      _template: html`
    <style>
      :host {
        position: fixed;
        top: 0;
        left: 0;
        width: 100%;
        height: 100%;
        background-color: var(--iron-overlay-backdrop-background-color, #000);
        opacity: 0;
        transition: opacity 0.2s;
        pointer-events: none;
        @apply --iron-overlay-backdrop;
      }

      :host(.opened) {
        opacity: var(--iron-overlay-backdrop-opacity, 0.6);
        pointer-events: auto;
        @apply --iron-overlay-backdrop-opened;
      }
    </style>

    <slot></slot>
`,

      is: 'iron-overlay-backdrop',

      properties: {

        /**
         * Returns true if the backdrop is opened.
         */
        opened: {
          reflectToAttribute: true,
          type: Boolean,
          value: false,
          observer: '_openedChanged',
        }

      },

      listeners: {
        'transitionend': '_onTransitionend',
      },

      /** @override */
      created: function() {
        // Used to cancel previous requestAnimationFrame calls when opened changes.
        this.__openedRaf = null;
      },

      /** @override */
      attached: function() {
        this.opened && this._openedChanged(this.opened);
      },

      /**
       * Appends the backdrop to document body if needed.
       */
      prepare: function() {
        if (this.opened && !this.parentNode) {
          dom(document.body).appendChild(this);
        }
      },

      /**
       * Shows the backdrop.
       */
      open: function() {
        this.opened = true;
      },

      /**
       * Hides the backdrop.
       */
      close: function() {
        this.opened = false;
      },

      /**
       * Removes the backdrop from document body if needed.
       */
      complete: function() {
        if (!this.opened && this.parentNode === document.body) {
          dom(this.parentNode).removeChild(this);
        }
      },

      _onTransitionend: function(event) {
        if (event && event.target === this) {
          this.complete();
        }
      },

      /**
       * @param {boolean} opened
       * @private
       */
      _openedChanged: function(opened) {
        if (opened) {
          // Auto-attach.
          this.prepare();
        } else {
          // Animation might be disabled via the mixin or opacity custom property.
          // If it is disabled in other ways, it's up to the user to call complete.
          var cs = window.getComputedStyle(this);
          if (cs.transitionDuration === '0s' || cs.opacity == 0) {
            this.complete();
          }
        }

        if (!this.isAttached) {
          return;
        }

        // Always cancel previous requestAnimationFrame.
        if (this.__openedRaf) {
          window.cancelAnimationFrame(this.__openedRaf);
          this.__openedRaf = null;
        }
        // Force relayout to ensure proper transitions.
        this.scrollTop = this.scrollTop;
        this.__openedRaf = window.requestAnimationFrame(function() {
          this.__openedRaf = null;
          this.toggleClass('opened', this.opened);
        }.bind(this));
      }
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
     * @package
     */
    class IronOverlayManagerClass {
      constructor() {
        /**
         * Used to keep track of the opened overlays.
         * @private {!Array<!Element>}
         */
        this._overlays = [];

        /**
         * iframes have a default z-index of 100,
         * so this default should be at least that.
         * @private {number}
         */
        this._minimumZ = 101;

        /**
         * Memoized backdrop element.
         * @private {Element|null}
         */
        this._backdropElement = null;

        // Enable document-wide tap recognizer.
        // NOTE: Use useCapture=true to avoid accidentally prevention of the closing
        // of an overlay via event.stopPropagation(). The only way to prevent
        // closing of an overlay should be through its APIs.
        // NOTE: enable tap on <html> to workaround Polymer/polymer#4459
        // Pass no-op function because MSEdge 15 doesn't handle null as 2nd argument
        // https://github.com/Microsoft/ChakraCore/issues/3863
        addListener(document.documentElement, 'tap', function() {});
        document.addEventListener('tap', this._onCaptureClick.bind(this), true);
        document.addEventListener('focus', this._onCaptureFocus.bind(this), true);
        document.addEventListener(
            'keydown', this._onCaptureKeyDown.bind(this), true);
      }

      /**
       * The shared backdrop element.
       * @return {!Element} backdropElement
       */
      get backdropElement() {
        if (!this._backdropElement) {
          this._backdropElement = document.createElement('iron-overlay-backdrop');
        }
        return this._backdropElement;
      }

      /**
       * The deepest active element.
       * @return {!Element} activeElement the active element
       */
      get deepActiveElement() {
        var active = document.activeElement;
        // document.activeElement can be null
        // https://developer.mozilla.org/en-US/docs/Web/API/Document/activeElement
        // In IE 11, it can also be an object when operating in iframes.
        // In these cases, default it to document.body.
        if (!active || active instanceof Element === false) {
          active = document.body;
        }
        while (active.root && dom(active.root).activeElement) {
          active = dom(active.root).activeElement;
        }
        return active;
      }

      /**
       * Brings the overlay at the specified index to the front.
       * @param {number} i
       * @private
       */
      _bringOverlayAtIndexToFront(i) {
        var overlay = this._overlays[i];
        if (!overlay) {
          return;
        }
        var lastI = this._overlays.length - 1;
        var currentOverlay = this._overlays[lastI];
        // Ensure always-on-top overlay stays on top.
        if (currentOverlay &&
            this._shouldBeBehindOverlay(overlay, currentOverlay)) {
          lastI--;
        }
        // If already the top element, return.
        if (i >= lastI) {
          return;
        }
        // Update z-index to be on top.
        var minimumZ = Math.max(this.currentOverlayZ(), this._minimumZ);
        if (this._getZ(overlay) <= minimumZ) {
          this._applyOverlayZ(overlay, minimumZ);
        }

        // Shift other overlays behind the new on top.
        while (i < lastI) {
          this._overlays[i] = this._overlays[i + 1];
          i++;
        }
        this._overlays[lastI] = overlay;
      }

      /**
       * Adds the overlay and updates its z-index if it's opened, or removes it if
       * it's closed. Also updates the backdrop z-index.
       * @param {!Element} overlay
       */
      addOrRemoveOverlay(overlay) {
        if (overlay.opened) {
          this.addOverlay(overlay);
        } else {
          this.removeOverlay(overlay);
        }
      }

      /**
       * Tracks overlays for z-index and focus management.
       * Ensures the last added overlay with always-on-top remains on top.
       * @param {!Element} overlay
       */
      addOverlay(overlay) {
        var i = this._overlays.indexOf(overlay);
        if (i >= 0) {
          this._bringOverlayAtIndexToFront(i);
          this.trackBackdrop();
          return;
        }
        var insertionIndex = this._overlays.length;
        var currentOverlay = this._overlays[insertionIndex - 1];
        var minimumZ = Math.max(this._getZ(currentOverlay), this._minimumZ);
        var newZ = this._getZ(overlay);

        // Ensure always-on-top overlay stays on top.
        if (currentOverlay &&
            this._shouldBeBehindOverlay(overlay, currentOverlay)) {
          // This bumps the z-index of +2.
          this._applyOverlayZ(currentOverlay, minimumZ);
          insertionIndex--;
          // Update minimumZ to match previous overlay's z-index.
          var previousOverlay = this._overlays[insertionIndex - 1];
          minimumZ = Math.max(this._getZ(previousOverlay), this._minimumZ);
        }

        // Update z-index and insert overlay.
        if (newZ <= minimumZ) {
          this._applyOverlayZ(overlay, minimumZ);
        }
        this._overlays.splice(insertionIndex, 0, overlay);

        this.trackBackdrop();
      }

      /**
       * @param {!Element} overlay
       */
      removeOverlay(overlay) {
        var i = this._overlays.indexOf(overlay);
        if (i === -1) {
          return;
        }
        this._overlays.splice(i, 1);

        this.trackBackdrop();
      }

      /**
       * Returns the current overlay.
       * @return {!Element|undefined}
       */
      currentOverlay() {
        var i = this._overlays.length - 1;
        return this._overlays[i];
      }

      /**
       * Returns the current overlay z-index.
       * @return {number}
       */
      currentOverlayZ() {
        return this._getZ(this.currentOverlay());
      }

      /**
       * Ensures that the minimum z-index of new overlays is at least `minimumZ`.
       * This does not effect the z-index of any existing overlays.
       * @param {number} minimumZ
       */
      ensureMinimumZ(minimumZ) {
        this._minimumZ = Math.max(this._minimumZ, minimumZ);
      }

      focusOverlay() {
        var current = /** @type {?} */ (this.currentOverlay());
        if (current) {
          current._applyFocus();
        }
      }

      /**
       * Updates the backdrop z-index.
       */
      trackBackdrop() {
        var overlay = this._overlayWithBackdrop();
        // Avoid creating the backdrop if there is no overlay with backdrop.
        if (!overlay && !this._backdropElement) {
          return;
        }
        this.backdropElement.style.zIndex = this._getZ(overlay) - 1;
        this.backdropElement.opened = !!overlay;
        // Property observers are not fired until element is attached
        // in Polymer 2.x, so we ensure element is attached if needed.
        // https://github.com/Polymer/polymer/issues/4526
        this.backdropElement.prepare();
      }

      /**
       * @return {!Array<!Element>}
       */
      getBackdrops() {
        var backdrops = [];
        for (var i = 0; i < this._overlays.length; i++) {
          if (this._overlays[i].withBackdrop) {
            backdrops.push(this._overlays[i]);
          }
        }
        return backdrops;
      }

      /**
       * Returns the z-index for the backdrop.
       * @return {number}
       */
      backdropZ() {
        return this._getZ(this._overlayWithBackdrop()) - 1;
      }

      /**
       * Returns the top opened overlay that has a backdrop.
       * @return {!Element|undefined}
       * @private
       */
      _overlayWithBackdrop() {
        for (var i = this._overlays.length - 1; i >= 0; i--) {
          if (this._overlays[i].withBackdrop) {
            return this._overlays[i];
          }
        }
      }

      /**
       * Calculates the minimum z-index for the overlay.
       * @param {Element=} overlay
       * @private
       */
      _getZ(overlay) {
        var z = this._minimumZ;
        if (overlay) {
          var z1 = Number(
              overlay.style.zIndex || window.getComputedStyle(overlay).zIndex);
          // Check if is a number
          // Number.isNaN not supported in IE 10+
          if (z1 === z1) {
            z = z1;
          }
        }
        return z;
      }

      /**
       * @param {!Element} element
       * @param {number|string} z
       * @private
       */
      _setZ(element, z) {
        element.style.zIndex = z;
      }

      /**
       * @param {!Element} overlay
       * @param {number} aboveZ
       * @private
       */
      _applyOverlayZ(overlay, aboveZ) {
        this._setZ(overlay, aboveZ + 2);
      }

      /**
       * Returns the deepest overlay in the path.
       * @param {!Array<!Element>=} path
       * @return {!Element|undefined}
       * @suppress {missingProperties}
       * @private
       */
      _overlayInPath(path) {
        path = path || [];
        for (var i = 0; i < path.length; i++) {
          if (path[i]._manager === this) {
            return path[i];
          }
        }
      }

      /**
       * Ensures the click event is delegated to the right overlay.
       * @param {!Event} event
       * @private
       */
      _onCaptureClick(event) {
        var i = this._overlays.length - 1;
        if (i === -1)
          return;
        var path = /** @type {!Array<!EventTarget>} */ (dom(event).path);
        var overlay;
        // Check if clicked outside of overlay.
        while ((overlay = /** @type {?} */ (this._overlays[i])) &&
               this._overlayInPath(path) !== overlay) {
          overlay._onCaptureClick(event);
          if (overlay.allowClickThrough) {
            i--;
          } else {
            break;
          }
        }
      }

      /**
       * Ensures the focus event is delegated to the right overlay.
       * @param {!Event} event
       * @private
       */
      _onCaptureFocus(event) {
        var overlay = /** @type {?} */ (this.currentOverlay());
        if (overlay) {
          overlay._onCaptureFocus(event);
        }
      }

      /**
       * Ensures TAB and ESC keyboard events are delegated to the right overlay.
       * @param {!Event} event
       * @private
       */
      _onCaptureKeyDown(event) {
        var overlay = /** @type {?} */ (this.currentOverlay());
        if (overlay) {
          if (IronA11yKeysBehavior.keyboardEventMatchesKeys(event, 'esc')) {
            overlay._onCaptureEsc(event);
          } else if (IronA11yKeysBehavior.keyboardEventMatchesKeys(event, 'tab')) {
            overlay._onCaptureTab(event);
          }
        }
      }

      /**
       * Returns if the overlay1 should be behind overlay2.
       * @param {!Element} overlay1
       * @param {!Element} overlay2
       * @return {boolean}
       * @suppress {missingProperties}
       * @private
       */
      _shouldBeBehindOverlay(overlay1, overlay2) {
        return !overlay1.alwaysOnTop && overlay2.alwaysOnTop;
      }
    }
    const IronOverlayManager = new IronOverlayManagerClass();

    /**
    @license
    Copyright (c) 2017 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */
    /**
     * Used to calculate the scroll direction during touch events.
     * @type {!Object}
     */
    var lastTouchPosition = {pageX: 0, pageY: 0};
    /**
     * Used to avoid computing event.path and filter scrollable nodes (better perf).
     * @type {?EventTarget}
     */
    var lastRootTarget = null;
    /**
     * @type {!Array<!Node>}
     */
    var lastScrollableNodes = [];
    /**
     * @type {!Array<string>}
     */
    var scrollEvents = [
      // Modern `wheel` event for mouse wheel scrolling:
      'wheel',
      // Older, non-standard `mousewheel` event for some FF:
      'mousewheel',
      // IE:
      'DOMMouseScroll',
      // Touch enabled devices
      'touchstart',
      'touchmove'
    ];
    // must be defined for modulizer
    var _boundScrollHandler;

    /**
     * The current element that defines the DOM boundaries of the
     * scroll lock. This is always the most recently locking element.
     *
     * @type {!Node|undefined}
     */
    var currentLockingElement;

    /**
     * Push an element onto the current scroll lock stack. The most recently
     * pushed element and its children will be considered scrollable. All
     * other elements will not be scrollable.
     *
     * Scroll locking is implemented as a stack so that cases such as
     * dropdowns within dropdowns are handled well.
     *
     * @param {!HTMLElement} element The element that should lock scroll.
     */
    function pushScrollLock(element) {
      // Prevent pushing the same element twice
      if (_lockingElements.indexOf(element) >= 0) {
        return;
      }

      if (_lockingElements.length === 0) {
        _lockScrollInteractions();
      }

      _lockingElements.push(element);
      currentLockingElement = _lockingElements[_lockingElements.length - 1];
    }

    /**
     * Remove an element from the scroll lock stack. The element being
     * removed does not need to be the most recently pushed element. However,
     * the scroll lock constraints only change when the most recently pushed
     * element is removed.
     *
     * @param {!HTMLElement} element The element to remove from the scroll
     * lock stack.
     */
    function removeScrollLock(element) {
      var index = _lockingElements.indexOf(element);

      if (index === -1) {
        return;
      }

      _lockingElements.splice(index, 1);
      currentLockingElement = _lockingElements[_lockingElements.length - 1];

      if (_lockingElements.length === 0) {
        _unlockScrollInteractions();
      }
    }

    const _lockingElements = [];

    function _scrollInteractionHandler(event) {
      // Avoid canceling an event with cancelable=false, e.g. scrolling is in
      // progress and cannot be interrupted.
      if (event.cancelable && _shouldPreventScrolling(event)) {
        event.preventDefault();
      }
      // If event has targetTouches (touch event), update last touch position.
      if (event.targetTouches) {
        var touch = event.targetTouches[0];
        lastTouchPosition.pageX = touch.pageX;
        lastTouchPosition.pageY = touch.pageY;
      }
    }

    function _lockScrollInteractions() {
      _boundScrollHandler =
          _boundScrollHandler || _scrollInteractionHandler.bind(undefined);
      for (var i = 0, l = scrollEvents.length; i < l; i++) {
        // NOTE: browsers that don't support objects as third arg will
        // interpret it as boolean, hence useCapture = true in this case.
        document.addEventListener(
            scrollEvents[i], _boundScrollHandler, {capture: true, passive: false});
      }
    }

    function _unlockScrollInteractions() {
      for (var i = 0, l = scrollEvents.length; i < l; i++) {
        // NOTE: browsers that don't support objects as third arg will
        // interpret it as boolean, hence useCapture = true in this case.
        document.removeEventListener(
            scrollEvents[i], _boundScrollHandler, {capture: true, passive: false});
      }
    }

    /**
     * Returns true if the event causes scroll outside the current locking
     * element, e.g. pointer/keyboard interactions, or scroll "leaking"
     * outside the locking element when it is already at its scroll boundaries.
     * @param {!Event} event
     * @return {boolean}
     * @package
     */
    function _shouldPreventScrolling(event) {
      // Update if root target changed. For touch events, ensure we don't
      // update during touchmove.
      var target = dom(event).rootTarget;
      if (event.type !== 'touchmove' && lastRootTarget !== target) {
        lastRootTarget = target;
        lastScrollableNodes = _getScrollableNodes(dom(event).path);
      }

      // Prevent event if no scrollable nodes.
      if (!lastScrollableNodes.length) {
        return true;
      }
      // Don't prevent touchstart event inside the locking element when it has
      // scrollable nodes.
      if (event.type === 'touchstart') {
        return false;
      }
      // Get deltaX/Y.
      var info = _getScrollInfo(event);
      // Prevent if there is no child that can scroll.
      return !_getScrollingNode(lastScrollableNodes, info.deltaX, info.deltaY);
    }

    /**
     * Returns an array of scrollable nodes up to the current locking element,
     * which is included too if scrollable.
     * @param {!Array<!Node>} nodes
     * @return {!Array<!Node>} scrollables
     * @package
     */
    function _getScrollableNodes(nodes) {
      var scrollables = [];
      var lockingIndex =
          nodes.indexOf(/** @type {!Node} */ (currentLockingElement));
      // Loop from root target to locking element (included).
      for (var i = 0; i <= lockingIndex; i++) {
        // Skip non-Element nodes.
        if (nodes[i].nodeType !== Node.ELEMENT_NODE) {
          continue;
        }
        var node = /** @type {!Element} */ (nodes[i]);
        // Check inline style before checking computed style.
        var style = node.style;
        if (style.overflow !== 'scroll' && style.overflow !== 'auto') {
          style = window.getComputedStyle(node);
        }
        if (style.overflow === 'scroll' || style.overflow === 'auto') {
          scrollables.push(node);
        }
      }
      return scrollables;
    }

    /**
     * Returns the node that is scrolling. If there is no scrolling,
     * returns undefined.
     * @param {!Array<!Node>} nodes
     * @param {number} deltaX Scroll delta on the x-axis
     * @param {number} deltaY Scroll delta on the y-axis
     * @return {!Node|undefined}
     * @package
     */
    function _getScrollingNode(nodes, deltaX, deltaY) {
      // No scroll.
      if (!deltaX && !deltaY) {
        return;
      }
      // Check only one axis according to where there is more scroll.
      // Prefer vertical to horizontal.
      var verticalScroll = Math.abs(deltaY) >= Math.abs(deltaX);
      for (var i = 0; i < nodes.length; i++) {
        var node = nodes[i];
        var canScroll = false;
        if (verticalScroll) {
          // delta < 0 is scroll up, delta > 0 is scroll down.
          canScroll = deltaY < 0 ?
              node.scrollTop > 0 :
              node.scrollTop < node.scrollHeight - node.clientHeight;
        } else {
          // delta < 0 is scroll left, delta > 0 is scroll right.
          canScroll = deltaX < 0 ?
              node.scrollLeft > 0 :
              node.scrollLeft < node.scrollWidth - node.clientWidth;
        }
        if (canScroll) {
          return node;
        }
      }
    }

    /**
     * Returns scroll `deltaX` and `deltaY`.
     * @param {!Event} event The scroll event
     * @return {{deltaX: number, deltaY: number}} Object containing the
     * x-axis scroll delta (positive: scroll right, negative: scroll left,
     * 0: no scroll), and the y-axis scroll delta (positive: scroll down,
     * negative: scroll up, 0: no scroll).
     * @package
     */
    function _getScrollInfo(event) {
      var info = {deltaX: event.deltaX, deltaY: event.deltaY};
      // Already available.
      if ('deltaX' in event) ;
      // Safari has scroll info in `wheelDeltaX/Y`.
      else if ('wheelDeltaX' in event && 'wheelDeltaY' in event) {
        info.deltaX = -event.wheelDeltaX;
        info.deltaY = -event.wheelDeltaY;
      }
      // IE10 has only vertical scroll info in `wheelDelta`.
      else if ('wheelDelta' in event) {
        info.deltaX = 0;
        info.deltaY = -event.wheelDelta;
      }
      // Firefox has scroll info in `detail` and `axis`.
      else if ('axis' in event) {
        info.deltaX = event.axis === 1 ? event.detail : 0;
        info.deltaY = event.axis === 2 ? event.detail : 0;
      }
      // On mobile devices, calculate scroll direction.
      else if (event.targetTouches) {
        var touch = event.targetTouches[0];
        // Touch moves from right to left => scrolling goes right.
        info.deltaX = lastTouchPosition.pageX - touch.pageX;
        // Touch moves from down to up => scrolling goes down.
        info.deltaY = lastTouchPosition.pageY - touch.pageY;
      }
      return info;
    }

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /** @polymerBehavior */
    const IronOverlayBehaviorImpl = {

      properties: {

        /**
         * True if the overlay is currently displayed.
         */
        opened:
            {observer: '_openedChanged', type: Boolean, value: false, notify: true},

        /**
         * True if the overlay was canceled when it was last closed.
         */
        canceled: {
          observer: '_canceledChanged',
          readOnly: true,
          type: Boolean,
          value: false
        },

        /**
         * Set to true to display a backdrop behind the overlay. It traps the focus
         * within the light DOM of the overlay.
         */
        withBackdrop: {
          observer: '_withBackdropChanged',
          type: Boolean,
        },

        /**
         * Set to true to disable auto-focusing the overlay or child nodes with
         * the `autofocus` attribute` when the overlay is opened.
         */
        noAutoFocus: {
          type: Boolean,
          value: false,
        },

        /**
         * Set to true to disable canceling the overlay with the ESC key.
         */
        noCancelOnEscKey: {
          type: Boolean,
          value: false,
        },

        /**
         * Set to true to disable canceling the overlay by clicking outside it.
         */
        noCancelOnOutsideClick: {
          type: Boolean,
          value: false,
        },

        /**
         * Contains the reason(s) this overlay was last closed (see
         * `iron-overlay-closed`). `IronOverlayBehavior` provides the `canceled`
         * reason; implementers of the behavior can provide other reasons in
         * addition to `canceled`.
         */
        closingReason: {
          // was a getter before, but needs to be a property so other
          // behaviors can override this.
          type: Object,
        },

        /**
         * Set to true to enable restoring of focus when overlay is closed.
         */
        restoreFocusOnClose: {
          type: Boolean,
          value: false,
        },

        /**
         * Set to true to allow clicks to go through overlays.
         * When the user clicks outside this overlay, the click may
         * close the overlay below.
         */
        allowClickThrough: {
          type: Boolean,
        },

        /**
         * Set to true to keep overlay always on top.
         */
        alwaysOnTop: {
          type: Boolean,
        },

        /**
         * Determines which action to perform when scroll outside an opened overlay
         * happens. Possible values: lock - blocks scrolling from happening, refit -
         * computes the new position on the overlay cancel - causes the overlay to
         * close
         */
        scrollAction: {
          type: String,
        },

        /**
         * Shortcut to access to the overlay manager.
         * @private
         * @type {!IronOverlayManagerClass}
         */
        _manager: {
          type: Object,
          value: IronOverlayManager,
        },

        /**
         * The node being focused.
         * @type {?Node}
         */
        _focusedChild: {
          type: Object,
        }

      },

      listeners: {'iron-resize': '_onIronResize'},

      observers: ['__updateScrollObservers(isAttached, opened, scrollAction)'],

      /**
       * The backdrop element.
       * @return {!Element}
       */
      get backdropElement() {
        return this._manager.backdropElement;
      },

      /**
       * Returns the node to give focus to.
       * @return {!Node}
       */
      get _focusNode() {
        return this._focusedChild || dom(this).querySelector('[autofocus]') || this;
      },

      /**
       * Array of nodes that can receive focus (overlay included), ordered by
       * `tabindex`. This is used to retrieve which is the first and last focusable
       * nodes in order to wrap the focus for overlays `with-backdrop`.
       *
       * If you know what is your content (specifically the first and last focusable
       * children), you can override this method to return only `[firstFocusable,
       * lastFocusable];`
       * @return {!Array<!Node>}
       * @protected
       */
      get _focusableNodes() {
        return IronFocusablesHelper.getTabbableNodes(this);
      },

      /**
       * @return {void}
       */
      ready: function() {
        // Used to skip calls to notifyResize and refit while the overlay is
        // animating.
        this.__isAnimating = false;
        // with-backdrop needs tabindex to be set in order to trap the focus.
        // If it is not set, IronOverlayBehavior will set it, and remove it if
        // with-backdrop = false.
        this.__shouldRemoveTabIndex = false;
        // Used for wrapping the focus on TAB / Shift+TAB.
        this.__firstFocusableNode = this.__lastFocusableNode = null;
        // Used by to keep track of the RAF callbacks.
        this.__rafs = {};
        // Focused node before overlay gets opened. Can be restored on close.
        this.__restoreFocusNode = null;
        // Scroll info to be restored.
        this.__scrollTop = this.__scrollLeft = null;
        this.__onCaptureScroll = this.__onCaptureScroll.bind(this);
        // Root nodes hosting the overlay, used to listen for scroll events on them.
        this.__rootNodes = null;
        this._ensureSetup();
      },

      /** @override */
      attached: function() {
        // Call _openedChanged here so that position can be computed correctly.
        if (this.opened) {
          this._openedChanged(this.opened);
        }
        this._observer = dom(this).observeNodes(this._onNodesChange);
      },

      /** @override */
      detached: function() {
        // TODO(bicknellr): Per spec, checking `this._observer` should never be
        // necessary because `connectedCallback` and `disconnectedCallback` should
        // always be called in alternating order. However, the custom elements
        // polyfill doesn't implement the reactions stack, so this can sometimes
        // happen, particularly if ShadyDOM is in noPatch mode where the custom
        // elements polyfill is installed before ShadyDOM. We should investigate
        // whether or not we can either implement the reactions stack without major
        // performance implications or patch ShadyDOM's functions to restore the
        // typical ShadyDOM-then-custom-elements order and remove this workaround.
        if (this._observer) {
          dom(this).unobserveNodes(this._observer);
        }
        this._observer = null;
        for (var cb in this.__rafs) {
          if (this.__rafs[cb] !== null) {
            cancelAnimationFrame(this.__rafs[cb]);
          }
        }
        this.__rafs = {};
        this._manager.removeOverlay(this);

        // We got detached while animating, ensure we show/hide the overlay
        // and fire iron-overlay-opened/closed event!
        if (this.__isAnimating) {
          if (this.opened) {
            this._finishRenderOpened();
          } else {
            // Restore the focus if necessary.
            this._applyFocus();
            this._finishRenderClosed();
          }
        }
      },

      /**
       * Toggle the opened state of the overlay.
       */
      toggle: function() {
        this._setCanceled(false);
        this.opened = !this.opened;
      },

      /**
       * Open the overlay.
       */
      open: function() {
        this._setCanceled(false);
        this.opened = true;
      },

      /**
       * Close the overlay.
       */
      close: function() {
        this._setCanceled(false);
        this.opened = false;
      },

      /**
       * Cancels the overlay.
       * @param {Event=} event The original event
       */
      cancel: function(event) {
        var cancelEvent =
            this.fire('iron-overlay-canceled', event, {cancelable: true});
        if (cancelEvent.defaultPrevented) {
          return;
        }

        this._setCanceled(true);
        this.opened = false;
      },

      /**
       * Invalidates the cached tabbable nodes. To be called when any of the
       * focusable content changes (e.g. a button is disabled).
       */
      invalidateTabbables: function() {
        this.__firstFocusableNode = this.__lastFocusableNode = null;
      },

      _ensureSetup: function() {
        if (this._overlaySetup) {
          return;
        }
        this._overlaySetup = true;
        this.style.outline = 'none';
        this.style.display = 'none';
      },

      /**
       * Called when `opened` changes.
       * @param {boolean=} opened
       * @protected
       */
      _openedChanged: function(opened) {
        if (opened) {
          this.removeAttribute('aria-hidden');
        } else {
          this.setAttribute('aria-hidden', 'true');
        }

        // Defer any animation-related code on attached
        // (_openedChanged gets called again on attached).
        if (!this.isAttached) {
          return;
        }

        this.__isAnimating = true;

        // Deraf for non-blocking rendering.
        this.__deraf('__openedChanged', this.__openedChanged);
      },

      _canceledChanged: function() {
        this.closingReason = this.closingReason || {};
        this.closingReason.canceled = this.canceled;
      },

      _withBackdropChanged: function() {
        // If tabindex is already set, no need to override it.
        if (this.withBackdrop && !this.hasAttribute('tabindex')) {
          this.setAttribute('tabindex', '-1');
          this.__shouldRemoveTabIndex = true;
        } else if (this.__shouldRemoveTabIndex) {
          this.removeAttribute('tabindex');
          this.__shouldRemoveTabIndex = false;
        }
        if (this.opened && this.isAttached) {
          this._manager.trackBackdrop();
        }
      },

      /**
       * tasks which must occur before opening; e.g. making the element visible.
       * @protected
       */
      _prepareRenderOpened: function() {
        // Store focused node.
        this.__restoreFocusNode = this._manager.deepActiveElement;

        // Needed to calculate the size of the overlay so that transitions on its
        // size will have the correct starting points.
        this._preparePositioning();
        this.refit();
        this._finishPositioning();

        // Safari will apply the focus to the autofocus element when displayed
        // for the first time, so we make sure to return the focus where it was.
        if (this.noAutoFocus && document.activeElement === this._focusNode) {
          this._focusNode.blur();
          this.__restoreFocusNode.focus();
        }
      },

      /**
       * Tasks which cause the overlay to actually open; typically play an
       * animation.
       * @protected
       */
      _renderOpened: function() {
        this._finishRenderOpened();
      },

      /**
       * Tasks which cause the overlay to actually close; typically play an
       * animation.
       * @protected
       */
      _renderClosed: function() {
        this._finishRenderClosed();
      },

      /**
       * Tasks to be performed at the end of open action. Will fire
       * `iron-overlay-opened`.
       * @protected
       */
      _finishRenderOpened: function() {
        this.notifyResize();
        this.__isAnimating = false;

        this.fire('iron-overlay-opened');
      },

      /**
       * Tasks to be performed at the end of close action. Will fire
       * `iron-overlay-closed`.
       * @protected
       */
      _finishRenderClosed: function() {
        // Hide the overlay.
        this.style.display = 'none';
        // Reset z-index only at the end of the animation.
        this.style.zIndex = '';
        this.notifyResize();
        this.__isAnimating = false;
        this.fire('iron-overlay-closed', this.closingReason);
      },

      _preparePositioning: function() {
        this.style.transition = this.style.webkitTransition = 'none';
        this.style.transform = this.style.webkitTransform = 'none';
        this.style.display = '';
      },

      _finishPositioning: function() {
        // First, make it invisible & reactivate animations.
        this.style.display = 'none';
        // Force reflow before re-enabling animations so that they don't start.
        // Set scrollTop to itself so that Closure Compiler doesn't remove this.
        this.scrollTop = this.scrollTop;
        this.style.transition = this.style.webkitTransition = '';
        this.style.transform = this.style.webkitTransform = '';
        // Now that animations are enabled, make it visible again
        this.style.display = '';
        // Force reflow, so that following animations are properly started.
        // Set scrollTop to itself so that Closure Compiler doesn't remove this.
        this.scrollTop = this.scrollTop;
      },

      /**
       * Applies focus according to the opened state.
       * @protected
       */
      _applyFocus: function() {
        if (this.opened) {
          if (!this.noAutoFocus) {
            this._focusNode.focus();
          }
        } else {
          // Restore focus.
          if (this.restoreFocusOnClose && this.__restoreFocusNode) {
            // If the activeElement is `<body>` or inside the overlay,
            // we are allowed to restore the focus. In all the other
            // cases focus might have been moved elsewhere by another
            // component or by an user interaction (e.g. click on a
            // button outside the overlay).
            var activeElement = this._manager.deepActiveElement;
            if (activeElement === document.body ||
                composedContains(this, activeElement)) {
              this.__restoreFocusNode.focus();
            }
          }
          this.__restoreFocusNode = null;
          this._focusNode.blur();
          this._focusedChild = null;
        }
      },

      /**
       * Cancels (closes) the overlay. Call when click happens outside the overlay.
       * @param {!Event} event
       * @protected
       */
      _onCaptureClick: function(event) {
        if (!this.noCancelOnOutsideClick) {
          this.cancel(event);
        }
      },

      /**
       * Keeps track of the focused child. If withBackdrop, traps focus within
       * overlay.
       * @param {!Event} event
       * @protected
       */
      _onCaptureFocus: function(event) {
        if (!this.withBackdrop) {
          return;
        }
        var path = dom(event).path;
        if (path.indexOf(this) === -1) {
          event.stopPropagation();
          this._applyFocus();
        } else {
          this._focusedChild = /** @type {Node} */ (path[0]);
        }
      },

      /**
       * Handles the ESC key event and cancels (closes) the overlay.
       * @param {!Event} event
       * @protected
       */
      _onCaptureEsc: function(event) {
        if (!this.noCancelOnEscKey) {
          this.cancel(event);
        }
      },

      /**
       * Handles TAB key events to track focus changes.
       * Will wrap focus for overlays withBackdrop.
       * @param {!Event} event
       * @protected
       */
      _onCaptureTab: function(event) {
        if (!this.withBackdrop) {
          return;
        }
        this.__ensureFirstLastFocusables();
        // TAB wraps from last to first focusable.
        // Shift + TAB wraps from first to last focusable.
        var shift = event.shiftKey;
        var nodeToCheck =
            shift ? this.__firstFocusableNode : this.__lastFocusableNode;
        var nodeToSet =
            shift ? this.__lastFocusableNode : this.__firstFocusableNode;
        var shouldWrap = false;
        if (nodeToCheck === nodeToSet) {
          // If nodeToCheck is the same as nodeToSet, it means we have an overlay
          // with 0 or 1 focusables; in either case we still need to trap the
          // focus within the overlay.
          shouldWrap = true;
        } else {
          // In dom=shadow, the manager will receive focus changes on the main
          // root but not the ones within other shadow roots, so we can't rely on
          // _focusedChild, but we should check the deepest active element.
          var focusedNode = this._manager.deepActiveElement;
          // If the active element is not the nodeToCheck but the overlay itself,
          // it means the focus is about to go outside the overlay, hence we
          // should prevent that (e.g. user opens the overlay and hit Shift+TAB).
          shouldWrap = (focusedNode === nodeToCheck || focusedNode === this);
        }

        if (shouldWrap) {
          // When the overlay contains the last focusable element of the document
          // and it's already focused, pressing TAB would move the focus outside
          // the document (e.g. to the browser search bar). Similarly, when the
          // overlay contains the first focusable element of the document and it's
          // already focused, pressing Shift+TAB would move the focus outside the
          // document (e.g. to the browser search bar).
          // In both cases, we would not receive a focus event, but only a blur.
          // In order to achieve focus wrapping, we prevent this TAB event and
          // force the focus. This will also prevent the focus to temporarily move
          // outside the overlay, which might cause scrolling.
          event.preventDefault();
          this._focusedChild = nodeToSet;
          this._applyFocus();
        }
      },

      /**
       * Refits if the overlay is opened and not animating.
       * @protected
       */
      _onIronResize: function() {
        if (this.opened && !this.__isAnimating) {
          this.__deraf('refit', this.refit);
        }
      },

      /**
       * Will call notifyResize if overlay is opened.
       * Can be overridden in order to avoid multiple observers on the same node.
       * @protected
       */
      _onNodesChange: function() {
        if (this.opened && !this.__isAnimating) {
          // It might have added focusable nodes, so invalidate cached values.
          this.invalidateTabbables();
          this.notifyResize();
        }
      },

      /**
       * Updates the references to the first and last focusable nodes.
       * @private
       */
      __ensureFirstLastFocusables: function() {
        var focusableNodes = this._focusableNodes;
        this.__firstFocusableNode = focusableNodes[0];
        this.__lastFocusableNode = focusableNodes[focusableNodes.length - 1];
      },

      /**
       * Tasks executed when opened changes: prepare for the opening, move the
       * focus, update the manager, render opened/closed.
       * @private
       */
      __openedChanged: function() {
        if (this.opened) {
          // Make overlay visible, then add it to the manager.
          this._prepareRenderOpened();
          this._manager.addOverlay(this);
          // Move the focus to the child node with [autofocus].
          this._applyFocus();

          this._renderOpened();
        } else {
          // Remove overlay, then restore the focus before actually closing.
          this._manager.removeOverlay(this);
          this._applyFocus();

          this._renderClosed();
        }
      },

      /**
       * Debounces the execution of a callback to the next animation frame.
       * @param {!string} jobname
       * @param {!Function} callback Always bound to `this`
       * @private
       */
      __deraf: function(jobname, callback) {
        var rafs = this.__rafs;
        if (rafs[jobname] !== null) {
          cancelAnimationFrame(rafs[jobname]);
        }
        rafs[jobname] = requestAnimationFrame(function nextAnimationFrame() {
          rafs[jobname] = null;
          callback.call(this);
        }.bind(this));
      },

      /**
       * @param {boolean} isAttached
       * @param {boolean} opened
       * @param {string=} scrollAction
       * @private
       */
      __updateScrollObservers: function(isAttached, opened, scrollAction) {
        if (!isAttached || !opened || !this.__isValidScrollAction(scrollAction)) {
          removeScrollLock(this);
          this.__removeScrollListeners();
        } else {
          if (scrollAction === 'lock') {
            this.__saveScrollPosition();
            pushScrollLock(this);
          }
          this.__addScrollListeners();
        }
      },

      /**
       * @private
       */
      __addScrollListeners: function() {
        if (!this.__rootNodes) {
          this.__rootNodes = [];
          // Listen for scroll events in all shadowRoots hosting this overlay only
          // when in native ShadowDOM.
          if (useShadow) {
            var node = this;
            while (node) {
              if (node.nodeType === Node.DOCUMENT_FRAGMENT_NODE && node.host) {
                this.__rootNodes.push(node);
              }
              node = node.host || node.assignedSlot || node.parentNode;
            }
          }
          this.__rootNodes.push(document);
        }
        this.__rootNodes.forEach(function(el) {
          el.addEventListener('scroll', this.__onCaptureScroll, {
            capture: true,
            passive: true,
          });
        }, this);
      },

      /**
       * @private
       */
      __removeScrollListeners: function() {
        if (this.__rootNodes) {
          this.__rootNodes.forEach(function(el) {
            el.removeEventListener('scroll', this.__onCaptureScroll, {
              capture: true,
              passive: true,
            });
          }, this);
        }
        if (!this.isAttached) {
          this.__rootNodes = null;
        }
      },

      /**
       * @param {string=} scrollAction
       * @return {boolean}
       * @private
       */
      __isValidScrollAction: function(scrollAction) {
        return scrollAction === 'lock' || scrollAction === 'refit' ||
            scrollAction === 'cancel';
      },

      /**
       * @private
       */
      __onCaptureScroll: function(event) {
        if (this.__isAnimating) {
          return;
        }
        // Check if scroll outside the overlay.
        if (dom(event).path.indexOf(this) >= 0) {
          return;
        }
        switch (this.scrollAction) {
          case 'lock':
            // NOTE: scrolling might happen if a scroll event is not cancellable, or
            // if user pressed keys that cause scrolling (they're not prevented in
            // order not to break a11y features like navigate with arrow keys).
            this.__restoreScrollPosition();
            break;
          case 'refit':
            this.__deraf('refit', this.refit);
            break;
          case 'cancel':
            this.cancel(event);
            break;
        }
      },

      /**
       * Memoizes the scroll position of the outside scrolling element.
       * @private
       */
      __saveScrollPosition: function() {
        if (document.scrollingElement) {
          this.__scrollTop = document.scrollingElement.scrollTop;
          this.__scrollLeft = document.scrollingElement.scrollLeft;
        } else {
          // Since we don't know if is the body or html, get max.
          this.__scrollTop =
              Math.max(document.documentElement.scrollTop, document.body.scrollTop);
          this.__scrollLeft = Math.max(
              document.documentElement.scrollLeft, document.body.scrollLeft);
        }
      },

      /**
       * Resets the scroll position of the outside scrolling element.
       * @private
       */
      __restoreScrollPosition: function() {
        if (document.scrollingElement) {
          document.scrollingElement.scrollTop = this.__scrollTop;
          document.scrollingElement.scrollLeft = this.__scrollLeft;
        } else {
          // Since we don't know if is the body or html, set both.
          document.documentElement.scrollTop = document.body.scrollTop =
              this.__scrollTop;
          document.documentElement.scrollLeft = document.body.scrollLeft =
              this.__scrollLeft;
        }
      },

    };

    const composedParent = node =>
        node.assignedSlot || node.parentNode || node.host;

    const composedContains = (ancestor, descendant) => {
      for (let element = descendant; element; element = composedParent(element)) {
        if (element === ancestor) {
          return true;
        }
      }
      return false;
    };

    /**
      Use `Polymer.IronOverlayBehavior` to implement an element that can be hidden
      or shown, and displays on top of other content. It includes an optional
      backdrop, and can be used to implement a variety of UI controls including
      dialogs and drop downs. Multiple overlays may be displayed at once.

      See the [demo source
      code](https://github.com/PolymerElements/iron-overlay-behavior/blob/master/demo/simple-overlay.html)
      for an example.

      ### Closing and canceling

      An overlay may be hidden by closing or canceling. The difference between close
      and cancel is user intent. Closing generally implies that the user
      acknowledged the content on the overlay. By default, it will cancel whenever
      the user taps outside it or presses the escape key. This behavior is
      configurable with the `no-cancel-on-esc-key` and the
      `no-cancel-on-outside-click` properties. `close()` should be called explicitly
      by the implementer when the user interacts with a control in the overlay
      element. When the dialog is canceled, the overlay fires an
      'iron-overlay-canceled' event. Call `preventDefault` on this event to prevent
      the overlay from closing.

      ### Positioning

      By default the element is sized and positioned to fit and centered inside the
      window. You can position and size it manually using CSS. See
      `Polymer.IronFitBehavior`.

      ### Backdrop

      Set the `with-backdrop` attribute to display a backdrop behind the overlay.
      The backdrop is appended to `<body>` and is of type `<iron-overlay-backdrop>`.
      See its doc page for styling options.

      In addition, `with-backdrop` will wrap the focus within the content in the
      light DOM. Override the [`_focusableNodes`
      getter](#Polymer.IronOverlayBehavior:property-_focusableNodes) to achieve a
      different behavior.

      ### Limitations

      The element is styled to appear on top of other content by setting its
      `z-index` property. You must ensure no element has a stacking context with a
      higher `z-index` than its parent stacking context. You should place this
      element as a child of `<body>` whenever possible.

      @demo demo/index.html
      @polymerBehavior
     */
    const IronOverlayBehavior =
        [IronFitBehavior, IronResizableBehavior, IronOverlayBehaviorImpl];

    /**
     * Fired after the overlay opens.
     * @event iron-overlay-opened
     */

    /**
     * Fired when the overlay is canceled, but before it is closed.
     * @event iron-overlay-canceled
     * @param {Event} event The closing of the overlay can be prevented
     * by calling `event.preventDefault()`. The `event.detail` is the original event
     * that originated the canceling (e.g. ESC keyboard event or click event outside
     * the overlay).
     */

    /**
     * Fired after the overlay closes.
     * @event iron-overlay-closed
     * @param {Event} event The `event.detail` is the `closingReason` property
     * (contains `canceled`, whether the overlay was canceled).
     */

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
      Use `Polymer.PaperDialogBehavior` and `paper-dialog-shared-styles.html` to
      implement a Material Design dialog.

      For example, if `<paper-dialog-impl>` implements this behavior:

          <paper-dialog-impl>
              <h2>Header</h2>
              <div>Dialog body</div>
              <div class="buttons">
                  <paper-button dialog-dismiss>Cancel</paper-button>
                  <paper-button dialog-confirm>Accept</paper-button>
              </div>
          </paper-dialog-impl>

      `paper-dialog-shared-styles.html` provide styles for a header, content area,
      and an action area for buttons. Use the `<h2>` tag for the header and the
      `buttons` class for the action area. You can use the `paper-dialog-scrollable`
      element (in its own repository) if you need a scrolling content area.

      Use the `dialog-dismiss` and `dialog-confirm` attributes on interactive
      controls to close the dialog. If the user dismisses the dialog with
      `dialog-confirm`, the `closingReason` will update to include `confirmed:
      true`.

      ### Accessibility

      This element has `role="dialog"` by default. Depending on the context, it may
      be more appropriate to override this attribute with `role="alertdialog"`.

      If `modal` is set, the element will prevent the focus from exiting the
      element. It will also ensure that focus remains in the dialog.

      @hero hero.svg
      @demo demo/index.html
      @polymerBehavior PaperDialogBehavior
     */
    const PaperDialogBehaviorImpl = {

      hostAttributes: {'role': 'dialog', 'tabindex': '-1'},

      properties: {

        /**
         * If `modal` is true, this implies `no-cancel-on-outside-click`,
         * `no-cancel-on-esc-key` and `with-backdrop`.
         */
        modal: {type: Boolean, value: false},

        __readied: {type: Boolean, value: false}

      },

      observers: ['_modalChanged(modal, __readied)'],

      listeners: {'tap': '_onDialogClick'},

      /**
       * @return {void}
       */
      ready: function() {
        // Only now these properties can be read.
        this.__prevNoCancelOnOutsideClick = this.noCancelOnOutsideClick;
        this.__prevNoCancelOnEscKey = this.noCancelOnEscKey;
        this.__prevWithBackdrop = this.withBackdrop;
        this.__readied = true;
      },

      _modalChanged: function(modal, readied) {
        // modal implies noCancelOnOutsideClick, noCancelOnEscKey and withBackdrop.
        // We need to wait for the element to be ready before we can read the
        // properties values.
        if (!readied) {
          return;
        }

        if (modal) {
          this.__prevNoCancelOnOutsideClick = this.noCancelOnOutsideClick;
          this.__prevNoCancelOnEscKey = this.noCancelOnEscKey;
          this.__prevWithBackdrop = this.withBackdrop;
          this.noCancelOnOutsideClick = true;
          this.noCancelOnEscKey = true;
          this.withBackdrop = true;
        } else {
          // If the value was changed to false, let it false.
          this.noCancelOnOutsideClick =
              this.noCancelOnOutsideClick && this.__prevNoCancelOnOutsideClick;
          this.noCancelOnEscKey =
              this.noCancelOnEscKey && this.__prevNoCancelOnEscKey;
          this.withBackdrop = this.withBackdrop && this.__prevWithBackdrop;
        }
      },

      _updateClosingReasonConfirmed: function(confirmed) {
        this.closingReason = this.closingReason || {};
        this.closingReason.confirmed = confirmed;
      },

      /**
       * Will dismiss the dialog if user clicked on an element with dialog-dismiss
       * or dialog-confirm attribute.
       */
      _onDialogClick: function(event) {
        // Search for the element with dialog-confirm or dialog-dismiss,
        // from the root target until this (excluded).
        var path = dom(event).path;
        for (var i = 0, l = path.indexOf(this); i < l; i++) {
          var target = path[i];
          if (target.hasAttribute &&
              (target.hasAttribute('dialog-dismiss') ||
               target.hasAttribute('dialog-confirm'))) {
            this._updateClosingReasonConfirmed(
                target.hasAttribute('dialog-confirm'));
            this.close();
            event.stopPropagation();
            break;
          }
        }
      }

    };

    /** @polymerBehavior */
    const PaperDialogBehavior =
        [IronOverlayBehavior, PaperDialogBehaviorImpl];

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
    Material design:
    [Dialogs](https://www.google.com/design/spec/components/dialogs.html)

    `paper-dialog-scrollable` implements a scrolling area used in a Material Design
    dialog. It shows a divider at the top and/or bottom indicating more content,
    depending on scroll position. Use this together with elements implementing
    `Polymer.PaperDialogBehavior`.

        <paper-dialog-impl>
          <h2>Header</h2>
          <paper-dialog-scrollable>
            Lorem ipsum...
          </paper-dialog-scrollable>
          <div class="buttons">
            <paper-button>OK</paper-button>
          </div>
        </paper-dialog-impl>

    It shows a top divider after scrolling if it is not the first child in its
    parent container, indicating there is more content above. It shows a bottom
    divider if it is scrollable and it is not the last child in its parent
    container, indicating there is more content below. The bottom divider is hidden
    if it is scrolled to the bottom.

    If `paper-dialog-scrollable` is not a direct child of the element implementing
    `Polymer.PaperDialogBehavior`, remember to set the `dialogElement`:

        <paper-dialog-impl id="myDialog">
          <h2>Header</h2>
          <div class="my-content-wrapper">
            <h4>Sub-header</h4>
            <paper-dialog-scrollable>
              Lorem ipsum...
            </paper-dialog-scrollable>
          </div>
          <div class="buttons">
            <paper-button>OK</paper-button>
          </div>
        </paper-dialog-impl>

        <script>
          var scrollable =
    Polymer.dom(myDialog).querySelector('paper-dialog-scrollable');
          scrollable.dialogElement = myDialog;
        </JAVA_SCRIIIIPT/>

    ### Styling
    The following custom properties and mixins are available for styling:

    Custom property | Description | Default
    ----------------|-------------|----------
    `--paper-dialog-scrollable` | Mixin for the scrollable content | {}

    @group Paper Elements
    @element paper-dialog-scrollable
    @demo demo/index.html
    @hero hero.svg
    */
    Polymer({
      _template: html`
    <style>

      :host {
        display: block;
        @apply --layout-relative;
      }

      :host(.is-scrolled:not(:first-child))::before {
        content: '';
        position: absolute;
        top: 0;
        left: 0;
        right: 0;
        height: 1px;
        background: var(--divider-color);
      }

      :host(.can-scroll:not(.scrolled-to-bottom):not(:last-child))::after {
        content: '';
        position: absolute;
        bottom: 0;
        left: 0;
        right: 0;
        height: 1px;
        background: var(--divider-color);
      }

      .scrollable {
        padding: 0 24px;

        @apply --layout-scroll;
        @apply --paper-dialog-scrollable;
      }

      .fit {
        @apply --layout-fit;
      }
    </style>

    <div id="scrollable" class="scrollable" on-scroll="updateScrollState">
      <slot></slot>
    </div>
`,

      is: 'paper-dialog-scrollable',

      properties: {

        /**
         * The dialog element that implements `Polymer.PaperDialogBehavior`
         * containing this element.
         * @type {?Node}
         */
        dialogElement: {type: Object}

      },

      /**
       * Returns the scrolling element.
       */
      get scrollTarget() {
        return this.$.scrollable;
      },

      ready: function() {
        this._ensureTarget();
        this.classList.add('no-padding');
      },

      attached: function() {
        this._ensureTarget();
        requestAnimationFrame(this.updateScrollState.bind(this));
      },

      updateScrollState: function() {
        this.toggleClass('is-scrolled', this.scrollTarget.scrollTop > 0);
        this.toggleClass(
            'can-scroll',
            this.scrollTarget.offsetHeight < this.scrollTarget.scrollHeight);
        this.toggleClass(
            'scrolled-to-bottom',
            this.scrollTarget.scrollTop + this.scrollTarget.offsetHeight >=
                this.scrollTarget.scrollHeight);
      },

      _ensureTarget: function() {
        // Read parentElement instead of parentNode in order to skip shadowRoots.
        this.dialogElement = this.dialogElement || this.parentElement;
        // Check if dialog implements paper-dialog-behavior. If not, fit
        // scrollTarget to host.
        if (this.dialogElement && this.dialogElement.behaviors &&
            this.dialogElement.behaviors.indexOf(PaperDialogBehaviorImpl) >= 0) {
          this.dialogElement.sizingTarget = this.scrollTarget;
          this.scrollTarget.classList.remove('fit');
        } else if (this.dialogElement) {
          this.scrollTarget.classList.add('fit');
        }
      }
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    // Give the user the choice to opt out of font loading.
    if (!window.polymerSkipLoadingFontRoboto) {
      const link = document.createElement('link');
      link.rel = 'stylesheet';
      link.type = 'text/css';
      link.crossOrigin = 'anonymous';
      link.href =
          'https://fonts.googleapis.com/css?family=Roboto+Mono:400,700|Roboto:400,300,300italic,400italic,500,500italic,700,700italic';
      document.head.appendChild(link);
    }

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */
    const template$a = html`<custom-style>
  <style is="custom-style">
    html {

      /* Shared Styles */
      --paper-font-common-base: {
        font-family: 'Roboto', 'Noto', sans-serif;
        -webkit-font-smoothing: antialiased;
      };

      --paper-font-common-code: {
        font-family: 'Roboto Mono', 'Consolas', 'Menlo', monospace;
        -webkit-font-smoothing: antialiased;
      };

      --paper-font-common-expensive-kerning: {
        text-rendering: optimizeLegibility;
      };

      --paper-font-common-nowrap: {
        white-space: nowrap;
        overflow: hidden;
        text-overflow: ellipsis;
      };

      /* Material Font Styles */

      --paper-font-display4: {
        @apply --paper-font-common-base;
        @apply --paper-font-common-nowrap;

        font-size: 112px;
        font-weight: 300;
        letter-spacing: -.044em;
        line-height: 120px;
      };

      --paper-font-display3: {
        @apply --paper-font-common-base;
        @apply --paper-font-common-nowrap;

        font-size: 56px;
        font-weight: 400;
        letter-spacing: -.026em;
        line-height: 60px;
      };

      --paper-font-display2: {
        @apply --paper-font-common-base;

        font-size: 45px;
        font-weight: 400;
        letter-spacing: -.018em;
        line-height: 48px;
      };

      --paper-font-display1: {
        @apply --paper-font-common-base;

        font-size: 34px;
        font-weight: 400;
        letter-spacing: -.01em;
        line-height: 40px;
      };

      --paper-font-headline: {
        @apply --paper-font-common-base;

        font-size: 24px;
        font-weight: 400;
        letter-spacing: -.012em;
        line-height: 32px;
      };

      --paper-font-title: {
        @apply --paper-font-common-base;
        @apply --paper-font-common-nowrap;

        font-size: 20px;
        font-weight: 500;
        line-height: 28px;
      };

      --paper-font-subhead: {
        @apply --paper-font-common-base;

        font-size: 16px;
        font-weight: 400;
        line-height: 24px;
      };

      --paper-font-body2: {
        @apply --paper-font-common-base;

        font-size: 14px;
        font-weight: 500;
        line-height: 24px;
      };

      --paper-font-body1: {
        @apply --paper-font-common-base;

        font-size: 14px;
        font-weight: 400;
        line-height: 20px;
      };

      --paper-font-caption: {
        @apply --paper-font-common-base;
        @apply --paper-font-common-nowrap;

        font-size: 12px;
        font-weight: 400;
        letter-spacing: 0.011em;
        line-height: 20px;
      };

      --paper-font-menu: {
        @apply --paper-font-common-base;
        @apply --paper-font-common-nowrap;

        font-size: 13px;
        font-weight: 500;
        line-height: 24px;
      };

      --paper-font-button: {
        @apply --paper-font-common-base;
        @apply --paper-font-common-nowrap;

        font-size: 14px;
        font-weight: 500;
        letter-spacing: 0.018em;
        line-height: 24px;
        text-transform: uppercase;
      };

      --paper-font-code2: {
        @apply --paper-font-common-code;

        font-size: 14px;
        font-weight: 700;
        line-height: 20px;
      };

      --paper-font-code1: {
        @apply --paper-font-common-code;

        font-size: 14px;
        font-weight: 500;
        line-height: 20px;
      };

    }

  </style>
</custom-style>`;
    template$a.setAttribute('style', 'display: none;');
    document.head.appendChild(template$a.content);

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */
    const $_documentContainer = document.createElement('template');
    $_documentContainer.setAttribute('style', 'display: none;');

    $_documentContainer.innerHTML = `<dom-module id="paper-dialog-shared-styles">
  <template>
    <style>
      :host {
        display: block;
        margin: 24px 40px;

        background: var(--paper-dialog-background-color, var(--primary-background-color));
        color: var(--paper-dialog-color, var(--primary-text-color));

        @apply --paper-font-body1;
        @apply --shadow-elevation-16dp;
        @apply --paper-dialog;
      }

      :host > ::slotted(*) {
        margin-top: 20px;
        padding: 0 24px;
      }

      :host > ::slotted(.no-padding) {
        padding: 0;
      }

      
      :host > ::slotted(*:first-child) {
        margin-top: 24px;
      }

      :host > ::slotted(*:last-child) {
        margin-bottom: 24px;
      }

      /* In 1.x, this selector was \`:host > ::content h2\`. In 2.x <slot> allows
      to select direct children only, which increases the weight of this
      selector, so we have to re-define first-child/last-child margins below. */
      :host > ::slotted(h2) {
        position: relative;
        margin: 0;

        @apply --paper-font-title;
        @apply --paper-dialog-title;
      }

      /* Apply mixin again, in case it sets margin-top. */
      :host > ::slotted(h2:first-child) {
        margin-top: 24px;
        @apply --paper-dialog-title;
      }

      /* Apply mixin again, in case it sets margin-bottom. */
      :host > ::slotted(h2:last-child) {
        margin-bottom: 24px;
        @apply --paper-dialog-title;
      }

      :host > ::slotted(.paper-dialog-buttons),
      :host > ::slotted(.buttons) {
        position: relative;
        padding: 8px 8px 8px 24px;
        margin: 0;

        color: var(--paper-dialog-button-color, var(--primary-color));

        @apply --layout-horizontal;
        @apply --layout-end-justified;
      }
    </style>
  </template>
</dom-module>`;

    document.head.appendChild($_documentContainer.content);

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
     * `NeonAnimatableBehavior` is implemented by elements containing
     * animations for use with elements implementing
     * `NeonAnimationRunnerBehavior`.
     * @polymerBehavior
     */
    const NeonAnimatableBehavior = {

      properties: {

        /**
         * Animation configuration. See README for more info.
         */
        animationConfig: {type: Object},

        /**
         * Convenience property for setting an 'entry' animation. Do not set
         * `animationConfig.entry` manually if using this. The animated node is set
         * to `this` if using this property.
         */
        entryAnimation: {
          observer: '_entryAnimationChanged',
          type: String,
        },

        /**
         * Convenience property for setting an 'exit' animation. Do not set
         * `animationConfig.exit` manually if using this. The animated node is set
         * to `this` if using this property.
         */
        exitAnimation: {
          observer: '_exitAnimationChanged',
          type: String,
        },

      },

      _entryAnimationChanged: function() {
        this.animationConfig = this.animationConfig || {};
        this.animationConfig['entry'] = [{name: this.entryAnimation, node: this}];
      },

      _exitAnimationChanged: function() {
        this.animationConfig = this.animationConfig || {};
        this.animationConfig['exit'] = [{name: this.exitAnimation, node: this}];
      },

      _copyProperties: function(config1, config2) {
        // shallowly copy properties from config2 to config1
        for (var property in config2) {
          config1[property] = config2[property];
        }
      },

      _cloneConfig: function(config) {
        var clone = {isClone: true};
        this._copyProperties(clone, config);
        return clone;
      },

      _getAnimationConfigRecursive: function(type, map, allConfigs) {
        if (!this.animationConfig) {
          return;
        }

        if (this.animationConfig.value &&
            typeof this.animationConfig.value === 'function') {
          this._warn(this._logf(
              'playAnimation',
              'Please put \'animationConfig\' inside of your components \'properties\' object instead of outside of it.'));
          return;
        }

        // type is optional
        var thisConfig;
        if (type) {
          thisConfig = this.animationConfig[type];
        } else {
          thisConfig = this.animationConfig;
        }

        if (!Array.isArray(thisConfig)) {
          thisConfig = [thisConfig];
        }

        // iterate animations and recurse to process configurations from child nodes
        if (thisConfig) {
          for (var config, index = 0; config = thisConfig[index]; index++) {
            if (config.animatable) {
              config.animatable._getAnimationConfigRecursive(
                  config.type || type, map, allConfigs);
            } else {
              if (config.id) {
                var cachedConfig = map[config.id];
                if (cachedConfig) {
                  // merge configurations with the same id, making a clone lazily
                  if (!cachedConfig.isClone) {
                    map[config.id] = this._cloneConfig(cachedConfig);
                    cachedConfig = map[config.id];
                  }
                  this._copyProperties(cachedConfig, config);
                } else {
                  // put any configs with an id into a map
                  map[config.id] = config;
                }
              } else {
                allConfigs.push(config);
              }
            }
          }
        }
      },

      /**
       * An element implementing `NeonAnimationRunnerBehavior` calls this
       * method to configure an animation with an optional type. Elements
       * implementing `NeonAnimatableBehavior` should define the property
       * `animationConfig`, which is either a configuration object or a map of
       * animation type to array of configuration objects.
       */
      getAnimationConfig: function(type) {
        var map = {};
        var allConfigs = [];
        this._getAnimationConfigRecursive(type, map, allConfigs);
        // append the configurations saved in the map to the array
        for (var key in map) {
          allConfigs.push(map[key]);
        }
        return allConfigs;
      }

    };

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
     * `NeonAnimationRunnerBehavior` adds a method to run animations.
     *
     * @polymerBehavior NeonAnimationRunnerBehavior
     */
    const NeonAnimationRunnerBehaviorImpl = {

      _configureAnimations: function(configs) {
        var results = [];
        var resultsToPlay = [];

        if (configs.length > 0) {
          for (let config, index = 0; config = configs[index]; index++) {
            let neonAnimation = document.createElement(config.name);
            // is this element actually a neon animation?
            if (neonAnimation.isNeonAnimation) {
              let result = null;
              // Closure compiler does not work well with a try / catch here.
              // .configure needs to be explicitly defined
              if (!neonAnimation.configure) {
                /**
                 * @param {Object} config
                 * @return {AnimationEffectReadOnly}
                 */
                neonAnimation.configure = function(config) {
                  return null;
                };
              }

              result = neonAnimation.configure(config);
              resultsToPlay.push({
                result: result,
                config: config,
                neonAnimation: neonAnimation,
              });
            } else {
              console.warn(this.is + ':', config.name, 'not found!');
            }
          }
        }

        for (var i = 0; i < resultsToPlay.length; i++) {
          let result = resultsToPlay[i].result;
          let config = resultsToPlay[i].config;
          let neonAnimation = resultsToPlay[i].neonAnimation;
          // configuration or play could fail if polyfills aren't loaded
          try {
            // Check if we have an Effect rather than an Animation
            if (typeof result.cancel != 'function') {
              result = document.timeline.play(result);
            }
          } catch (e) {
            result = null;
            console.warn('Couldnt play', '(', config.name, ').', e);
          }

          if (result) {
            results.push({
              neonAnimation: neonAnimation,
              config: config,
              animation: result,
            });
          }
        }

        return results;
      },

      _shouldComplete: function(activeEntries) {
        var finished = true;
        for (var i = 0; i < activeEntries.length; i++) {
          if (activeEntries[i].animation.playState != 'finished') {
            finished = false;
            break;
          }
        }
        return finished;
      },

      _complete: function(activeEntries) {
        for (var i = 0; i < activeEntries.length; i++) {
          activeEntries[i].neonAnimation.complete(activeEntries[i].config);
        }
        for (var i = 0; i < activeEntries.length; i++) {
          activeEntries[i].animation.cancel();
        }
      },

      /**
       * Plays an animation with an optional `type`.
       * @param {string=} type
       * @param {!Object=} cookie
       */
      playAnimation: function(type, cookie) {
        var configs = this.getAnimationConfig(type);
        if (!configs) {
          return;
        }
        this._active = this._active || {};
        if (this._active[type]) {
          this._complete(this._active[type]);
          delete this._active[type];
        }

        var activeEntries = this._configureAnimations(configs);

        if (activeEntries.length == 0) {
          this.fire('neon-animation-finish', cookie, {bubbles: false});
          return;
        }

        this._active[type] = activeEntries;

        for (var i = 0; i < activeEntries.length; i++) {
          activeEntries[i].animation.onfinish = function() {
            if (this._shouldComplete(activeEntries)) {
              this._complete(activeEntries);
              delete this._active[type];
              this.fire('neon-animation-finish', cookie, {bubbles: false});
            }
          }.bind(this);
        }
      },

      /**
       * Cancels the currently running animations.
       */
      cancelAnimation: function() {
        for (var k in this._active) {
          var entries = this._active[k];

                        for (var j in entries) {
            entries[j].animation.cancel();
          }
        }

        this._active = {};
      }
    };

    /** @polymerBehavior */
    const NeonAnimationRunnerBehavior =
        [NeonAnimatableBehavior, NeonAnimationRunnerBehaviorImpl];

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
    Material design:
    [Dialogs](https://www.google.com/design/spec/components/dialogs.html)

    `<paper-dialog>` is a dialog with Material Design styling and optional
    animations when it is opened or closed. It provides styles for a header, content
    area, and an action area for buttons. You can use the
    `<paper-dialog-scrollable>` element (in its own repository) if you need a
    scrolling content area. To autofocus a specific child element after opening the
    dialog, give it the `autofocus` attribute. See `Polymer.PaperDialogBehavior` and
    `Polymer.IronOverlayBehavior` for specifics.

    For example, the following code implements a dialog with a header, scrolling
    content area and buttons. Focus will be given to the `dialog-confirm` button
    when the dialog is opened.

        <paper-dialog>
          <h2>Header</h2>
          <paper-dialog-scrollable>
            Lorem ipsum...
          </paper-dialog-scrollable>
          <div class="buttons">
            <paper-button dialog-dismiss>Cancel</paper-button>
            <paper-button dialog-confirm autofocus>Accept</paper-button>
          </div>
        </paper-dialog>

    ### Styling

    See the docs for `Polymer.PaperDialogBehavior` for the custom properties
    available for styling this element.

    ### Animations

    Set the `entry-animation` and/or `exit-animation` attributes to add an animation
    when the dialog is opened or closed. See the documentation in
    [PolymerElements/neon-animation](https://github.com/PolymerElements/neon-animation)
    for more info.

    For example:

        <script type="module">
          import '@polymer/neon-animation/animations/fade-out-animation.js';
          import '@polymer/neon-animation/animations/scale-up-animation.js';
        </JAVA_SCRIIIIPT/>

        <paper-dialog entry-animation="scale-up-animation"
                      exit-animation="fade-out-animation">
          <h2>Header</h2>
          <div>Dialog body</div>
        </paper-dialog>

    ### Accessibility

    See the docs for `Polymer.PaperDialogBehavior` for accessibility features
    implemented by this element.

    @group Paper Elements
    @element paper-dialog
    @hero hero.svg
    @demo demo/index.html
    */
    Polymer({
      _template: html`
    <style include="paper-dialog-shared-styles"></style>
    <slot></slot>
`,

      is: 'paper-dialog',
      behaviors: [PaperDialogBehavior, NeonAnimationRunnerBehavior],
      listeners: {'neon-animation-finish': '_onNeonAnimationFinish'},

      _renderOpened: function() {
        this.cancelAnimation();
        this.playAnimation('entry');
      },

      _renderClosed: function() {
        this.cancelAnimation();
        this.playAnimation('exit');
      },

      _onNeonAnimationFinish: function() {
        if (this.opened) {
          this._finishRenderOpened();
        } else {
          this._finishRenderClosed();
        }
      }
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
    `iron-a11y-announcer` is a singleton element that is intended to add a11y
    to features that require on-demand announcement from screen readers. In
    order to make use of the announcer, it is best to request its availability
    in the announcing element.

    Example:

        Polymer({

          is: 'x-chatty',

          attached: function() {
            // This will create the singleton element if it has not
            // been created yet:
            Polymer.IronA11yAnnouncer.requestAvailability();
          }
        });

    After the `iron-a11y-announcer` has been made available, elements can
    make announces by firing bubbling `iron-announce` events.

    Example:

        this.fire('iron-announce', {
          text: 'This is an announcement!'
        }, { bubbles: true });

    Note: announcements are only audible if you have a screen reader enabled.

    @demo demo/index.html
    */
    const IronA11yAnnouncer = Polymer({
      /** @override */
      _template: html`
    <style>
      :host {
        display: inline-block;
        position: fixed;
        clip: rect(0px,0px,0px,0px);
      }
    </style>
    <div aria-live$="[[mode]]">[[_text]]</div>
`,

      is: 'iron-a11y-announcer',

      properties: {

        /**
         * The value of mode is used to set the `aria-live` attribute
         * for the element that will be announced. Valid values are: `off`,
         * `polite` and `assertive`.
         */
        mode: {type: String, value: 'polite'},

        /**
         * The timeout on refreshing the announcement text. Larger timeouts are
         * needed for certain screen readers to re-announce the same message.
         */
        timeout: {type: Number, value: 150},

        _text: {type: String, value: ''},
      },

      /** @override */
      created: function() {
        if (!IronA11yAnnouncer.instance) {
          IronA11yAnnouncer.instance = this;
        }

        document.addEventListener('iron-announce', this._onIronAnnounce.bind(this));
      },

      /**
       * Cause a text string to be announced by screen readers.
       *
       * @param {string} text The text that should be announced.
       */
      announce: function(text) {
        this._text = '';
        this.async(function() {
          this._text = text;
        }, this.timeout);
      },

      _onIronAnnounce: function(event) {
        if (event.detail && event.detail.text) {
          this.announce(event.detail.text);
        }
      }
    });

    IronA11yAnnouncer.instance = null;

    IronA11yAnnouncer.requestAvailability = function() {
      if (!IronA11yAnnouncer.instance) {
        IronA11yAnnouncer.instance = document.createElement('iron-a11y-announcer');
      }

      if (document.body) {
        document.body.appendChild(IronA11yAnnouncer.instance);
      } else {
        document.addEventListener('load', function() {
          document.body.appendChild(IronA11yAnnouncer.instance);
        });
      }
    };

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
    `<iron-input>` is a wrapper to a native `<input>` element, that adds two-way
    binding and prevention of invalid input. To use it, you must distribute a native
    `<input>` yourself. You can continue to use the native `input` as you would
    normally:

        <iron-input>
          <input>
        </iron-input>

        <iron-input>
          <input type="email" disabled>
        </iron-input>

    ### Two-way binding

    By default you can only get notified of changes to a native `<input>`'s `value`
    due to user input:

        <input value="{{myValue::input}}">

    This means that if you imperatively set the value (i.e. `someNativeInput.value =
    'foo'`), no events will be fired and this change cannot be observed.

    `iron-input` adds the `bind-value` property that mirrors the native `input`'s
    '`value` property; this property can be used for two-way data binding.
    `bind-value` will notify if it is changed either by user input or by script.

        <iron-input bind-value="{{myValue}}">
          <input>
        </iron-input>

    Note: this means that if you want to imperatively set the native `input`'s, you
    _must_ set `bind-value` instead, so that the wrapper `iron-input` can be
    notified.

    ### Validation

    `iron-input` uses the native `input`'s validation. For simplicity, `iron-input`
    has a `validate()` method (which internally just checks the distributed
    `input`'s validity), which sets an `invalid` attribute that can also be used for
    styling.

    To validate automatically as you type, you can use the `auto-validate`
    attribute.

    `iron-input` also fires an `iron-input-validate` event after `validate()` is
    called. You can use it to implement a custom validator:

        var CatsOnlyValidator = {
          validate: function(ironInput) {
            var valid = !ironInput.bindValue || ironInput.bindValue === 'cat';
            ironInput.invalid = !valid;
            return valid;
          }
        }
        ironInput.addEventListener('iron-input-validate', function() {
          CatsOnly.validate(input2);
        });

    You can also use an element implementing an
    [`IronValidatorBehavior`](/element/PolymerElements/iron-validatable-behavior).
    This example can also be found in the demo for this element:

        <iron-input validator="cats-only">
          <input>
        </iron-input>

    ### Preventing invalid input

    It may be desirable to only allow users to enter certain characters. You can use
    the `allowed-pattern` attribute to accomplish this. This feature is separate
    from validation, and `allowed-pattern` does not affect how the input is
    validated.

        // Only allow typing digits, but a valid input has exactly 5 digits.
        <iron-input allowed-pattern="[0-9]">
          <input pattern="\d{5}">
        </iron-input>

    @demo demo/index.html
    */
    Polymer({
      _template: html`
    <style>
      :host {
        display: inline-block;
      }
    </style>
    <slot id="content"></slot>
`,

      is: 'iron-input',
      behaviors: [IronValidatableBehavior],

      /**
       * Fired whenever `validate()` is called.
       *
       * @event iron-input-validate
       */

      properties: {

        /**
         * Use this property instead of `value` for two-way data binding, or to
         * set a default value for the input. **Do not** use the distributed
         * input's `value` property to set a default value.
         */
        bindValue: {type: String, value: ''},

        /**
         * Computed property that echoes `bindValue` (mostly used for Polymer 1.0
         * backcompatibility, if you were one-way binding to the Polymer 1.0
         * `input is="iron-input"` value attribute).
         */
        value: {type: String, computed: '_computeValue(bindValue)'},

        /**
         * Regex-like list of characters allowed as input; all characters not in the
         * list will be rejected. The recommended format should be a list of allowed
         * characters, for example, `[a-zA-Z0-9.+-!;:]`.
         *
         * This pattern represents the allowed characters for the field; as the user
         * inputs text, each individual character will be checked against the
         * pattern (rather than checking the entire value as a whole). If a
         * character is not a match, it will be rejected.
         *
         * Pasted input will have each character checked individually; if any
         * character doesn't match `allowedPattern`, the entire pasted string will
         * be rejected.
         *
         * Note: if you were using `iron-input` in 1.0, you were also required to
         * set `prevent-invalid-input`. This is no longer needed as of Polymer 2.0,
         * and will be set automatically for you if an `allowedPattern` is provided.
         *
         */
        allowedPattern: {type: String},

        /**
         * Set to true to auto-validate the input value as you type.
         */
        autoValidate: {type: Boolean, value: false},

        /**
         * The native input element.
         */
        _inputElement: Object,
      },

      observers: ['_bindValueChanged(bindValue, _inputElement)'],
      listeners: {'input': '_onInput', 'keypress': '_onKeypress'},

      created: function() {
        IronA11yAnnouncer.requestAvailability();
        this._previousValidInput = '';
        this._patternAlreadyChecked = false;
      },

      attached: function() {
        // If the input is added at a later time, update the internal reference.
        this._observer = dom(this).observeNodes(function(info) {
          this._initSlottedInput();
        }.bind(this));
      },

      detached: function() {
        if (this._observer) {
          dom(this).unobserveNodes(this._observer);
          this._observer = null;
        }
      },

      /**
       * Returns the distributed input element.
       */
      get inputElement() {
        return this._inputElement;
      },

      _initSlottedInput: function() {
        this._inputElement = this.getEffectiveChildren()[0];

        if (this.inputElement && this.inputElement.value) {
          this.bindValue = this.inputElement.value;
        }

        this.fire('iron-input-ready');
      },

      get _patternRegExp() {
        var pattern;
        if (this.allowedPattern) {
          pattern = new RegExp(this.allowedPattern);
        } else {
          switch (this.inputElement.type) {
            case 'number':
              pattern = /[0-9.,e-]/;
              break;
          }
        }
        return pattern;
      },

      /**
       * @suppress {checkTypes}
       */
      _bindValueChanged: function(bindValue, inputElement) {
        // The observer could have run before attached() when we have actually
        // initialized this property.
        if (!inputElement) {
          return;
        }

        if (bindValue === undefined) {
          inputElement.value = null;
        } else if (bindValue !== inputElement.value) {
          this.inputElement.value = bindValue;
        }

        if (this.autoValidate) {
          this.validate();
        }

        // manually notify because we don't want to notify until after setting value
        this.fire('bind-value-changed', {value: bindValue});
      },

      _onInput: function() {
        // Need to validate each of the characters pasted if they haven't
        // been validated inside `_onKeypress` already.
        if (this.allowedPattern && !this._patternAlreadyChecked) {
          var valid = this._checkPatternValidity();
          if (!valid) {
            this._announceInvalidCharacter(
                'Invalid string of characters not entered.');
            this.inputElement.value = this._previousValidInput;
          }
        }
        this.bindValue = this._previousValidInput = this.inputElement.value;
        this._patternAlreadyChecked = false;
      },

      _isPrintable: function(event) {
        // What a control/printable character is varies wildly based on the browser.
        // - most control characters (arrows, backspace) do not send a `keypress`
        // event
        //   in Chrome, but the *do* on Firefox
        // - in Firefox, when they do send a `keypress` event, control chars have
        //   a charCode = 0, keyCode = xx (for ex. 40 for down arrow)
        // - printable characters always send a keypress event.
        // - in Firefox, printable chars always have a keyCode = 0. In Chrome, the
        // keyCode
        //   always matches the charCode.
        // None of this makes any sense.

        // For these keys, ASCII code == browser keycode.
        var anyNonPrintable = (event.keyCode == 8) ||  // backspace
            (event.keyCode == 9) ||                    // tab
            (event.keyCode == 13) ||                   // enter
            (event.keyCode == 27);                     // escape

        // For these keys, make sure it's a browser keycode and not an ASCII code.
        var mozNonPrintable = (event.keyCode == 19) ||  // pause
            (event.keyCode == 20) ||                    // caps lock
            (event.keyCode == 45) ||                    // insert
            (event.keyCode == 46) ||                    // delete
            (event.keyCode == 144) ||                   // num lock
            (event.keyCode == 145) ||                   // scroll lock
            (event.keyCode > 32 &&
             event.keyCode < 41) ||  // page up/down, end, home, arrows
            (event.keyCode > 111 && event.keyCode < 124);  // fn keys

        return !anyNonPrintable && !(event.charCode == 0 && mozNonPrintable);
      },

      _onKeypress: function(event) {
        if (!this.allowedPattern && this.inputElement.type !== 'number') {
          return;
        }
        var regexp = this._patternRegExp;
        if (!regexp) {
          return;
        }

        // Handle special keys and backspace
        if (event.metaKey || event.ctrlKey || event.altKey) {
          return;
        }

        // Check the pattern either here or in `_onInput`, but not in both.
        this._patternAlreadyChecked = true;

        var thisChar = String.fromCharCode(event.charCode);
        if (this._isPrintable(event) && !regexp.test(thisChar)) {
          event.preventDefault();
          this._announceInvalidCharacter(
              'Invalid character ' + thisChar + ' not entered.');
        }
      },

      _checkPatternValidity: function() {
        var regexp = this._patternRegExp;
        if (!regexp) {
          return true;
        }
        for (var i = 0; i < this.inputElement.value.length; i++) {
          if (!regexp.test(this.inputElement.value[i])) {
            return false;
          }
        }
        return true;
      },

      /**
       * Returns true if `value` is valid. The validator provided in `validator`
       * will be used first, then any constraints.
       * @return {boolean} True if the value is valid.
       */
      validate: function() {
        if (!this.inputElement) {
          this.invalid = false;
          return true;
        }

        // Use the nested input's native validity.
        var valid = this.inputElement.checkValidity();

        // Only do extra checking if the browser thought this was valid.
        if (valid) {
          // Empty, required input is invalid
          if (this.required && this.bindValue === '') {
            valid = false;
          } else if (this.hasValidator()) {
            valid = IronValidatableBehavior.validate.call(this, this.bindValue);
          }
        }

        this.invalid = !valid;
        this.fire('iron-input-validate');
        return valid;
      },

      _announceInvalidCharacter: function(message) {
        this.fire('iron-announce', {text: message});
      },

      _computeValue: function(bindValue) {
        return bindValue;
      }
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
     * Use `Polymer.PaperInputAddonBehavior` to implement an add-on for
     * `<paper-input-container>`. A add-on appears below the input, and may display
     * information based on the input value and validity such as a character counter
     * or an error message.
     * @polymerBehavior
     */
    const PaperInputAddonBehavior = {
      attached: function() {
        this.fire('addon-attached');
      },

      /**
       * The function called by `<paper-input-container>` when the input value or
       * validity changes.
       * @param {{
       *   invalid: boolean,
       *   inputElement: (Element|undefined),
       *   value: (string|undefined)
       * }} state -
       *     inputElement: The input element.
       *     value: The input value.
       *     invalid: True if the input value is invalid.
       */
      update: function(state) {}

    };

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /*
    `<paper-input-char-counter>` is a character counter for use with
    `<paper-input-container>`. It shows the number of characters entered in the
    input and the max length if it is specified.

        <paper-input-container>
          <input maxlength="20">
          <paper-input-char-counter></paper-input-char-counter>
        </paper-input-container>

    ### Styling

    The following mixin is available for styling:

    Custom property | Description | Default
    ----------------|-------------|----------
    `--paper-input-char-counter` | Mixin applied to the element | `{}`
    */
    Polymer({
      /** @override */
      _template: html`
    <style>
      :host {
        display: inline-block;
        float: right;

        @apply --paper-font-caption;
        @apply --paper-input-char-counter;
      }

      :host([hidden]) {
        display: none !important;
      }

      :host(:dir(rtl)) {
        float: left;
      }
    </style>

    <span>[[_charCounterStr]]</span>
`,

      is: 'paper-input-char-counter',
      behaviors: [PaperInputAddonBehavior],
      properties: {_charCounterStr: {type: String, value: '0'}},

      /**
       * This overrides the update function in PaperInputAddonBehavior.
       * @param {{
       *   inputElement: (Element|undefined),
       *   value: (string|undefined),
       *   invalid: boolean
       * }} state -
       *     inputElement: The input element.
       *     value: The input value.
       *     invalid: True if the input value is invalid.
       */
      update: function(state) {
        if (!state.inputElement) {
          return;
        }

        state.value = state.value || '';

        var counter = state.value.toString().length.toString();

        if (state.inputElement.hasAttribute('maxlength')) {
          counter += '/' + state.inputElement.getAttribute('maxlength');
        }

        this._charCounterStr = counter;
      }
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */
    const template$b = html`
<custom-style>
  <style is="custom-style">
    html {
      --paper-input-container-shared-input-style: {
        position: relative; /* to make a stacking context */
        outline: none;
        box-shadow: none;
        padding: 0;
        margin: 0;
        width: 100%;
        max-width: 100%;
        background: transparent;
        border: none;
        color: var(--paper-input-container-input-color, var(--primary-text-color));
        -webkit-appearance: none;
        text-align: inherit;
        vertical-align: var(--paper-input-container-input-align, bottom);

        @apply --paper-font-subhead;
      };
    }
  </style>
</custom-style>
`;
    template$b.setAttribute('style', 'display: none;');
    document.head.appendChild(template$b.content);

    /*
    `<paper-input-container>` is a container for a `<label>`, an `<iron-input>` or
    `<textarea>` and optional add-on elements such as an error message or character
    counter, used to implement Material Design text fields.

    For example:

        <paper-input-container>
          <label slot="label">Your name</label>
          <iron-input slot="input">
            <input>
          </iron-input>
          // In Polymer 1.0, you would use `<input is="iron-input" slot="input">`
    instead of the above.
        </paper-input-container>

    You can style the nested `<input>` however you want; if you want it to look like
    a Material Design input, you can style it with the
    --paper-input-container-shared-input-style mixin.

    Do not wrap `<paper-input-container>` around elements that already include it,
    such as `<paper-input>`. Doing so may cause events to bounce infinitely between
    the container and its contained element.

    ### Listening for input changes

    By default, it listens for changes on the `bind-value` attribute on its children
    nodes and perform tasks such as auto-validating and label styling when the
    `bind-value` changes. You can configure the attribute it listens to with the
    `attr-for-value` attribute.

    ### Using a custom input element

    You can use a custom input element in a `<paper-input-container>`, for example
    to implement a compound input field like a social security number input. The
    custom input element should have the `paper-input-input` class, have a
    `notify:true` value property and optionally implements
    `Polymer.IronValidatableBehavior` if it is validatable.

        <paper-input-container attr-for-value="ssn-value">
          <label slot="label">Social security number</label>
          <ssn-input slot="input" class="paper-input-input"></ssn-input>
        </paper-input-container>


    If you're using a `<paper-input-container>` imperatively, it's important to make
    sure that you attach its children (the `iron-input` and the optional `label`)
    before you attach the `<paper-input-container>` itself, so that it can be set up
    correctly.

    ### Validation

    If the `auto-validate` attribute is set, the input container will validate the
    input and update the container styling when the input value changes.

    ### Add-ons

    Add-ons are child elements of a `<paper-input-container>` with the `add-on`
    attribute and implements the `Polymer.PaperInputAddonBehavior` behavior. They
    are notified when the input value or validity changes, and may implement
    functionality such as error messages or character counters. They appear at the
    bottom of the input.

    ### Prefixes and suffixes
    These are child elements of a `<paper-input-container>` with the `prefix`
    or `suffix` attribute, and are displayed inline with the input, before or after.

        <paper-input-container>
          <div slot="prefix">$</div>
          <label slot="label">Total</label>
          <iron-input slot="input">
            <input>
          </iron-input>
          // In Polymer 1.0, you would use `<input is="iron-input" slot="input">`
    instead of the above. <paper-icon-button slot="suffix"
    icon="clear"></paper-icon-button>
        </paper-input-container>

    ### Styling

    The following custom properties and mixins are available for styling:

    Custom property | Description | Default
    ----------------|-------------|----------
    `--paper-input-container-color` | Label and underline color when the input is not focused | `--secondary-text-color`
    `--paper-input-container-focus-color` | Label and underline color when the input is focused | `--primary-color`
    `--paper-input-container-invalid-color` | Label and underline color when the input is is invalid | `--error-color`
    `--paper-input-container-input-color` | Input foreground color | `--primary-text-color`
    `--paper-input-container` | Mixin applied to the container | `{}`
    `--paper-input-container-disabled` | Mixin applied to the container when it's disabled | `{}`
    `--paper-input-container-label` | Mixin applied to the label | `{}`
    `--paper-input-container-label-focus` | Mixin applied to the label when the input is focused | `{}`
    `--paper-input-container-label-floating` | Mixin applied to the label when floating | `{}`
    `--paper-input-container-input` | Mixin applied to the input | `{}`
    `--paper-input-container-input-align` | The vertical-align property of the input | `bottom`
    `--paper-input-container-input-disabled` | Mixin applied to the input when the component is disabled | `{}`
    `--paper-input-container-input-focus` | Mixin applied to the input when focused | `{}`
    `--paper-input-container-input-invalid` | Mixin applied to the input when invalid | `{}`
    `--paper-input-container-input-webkit-spinner` | Mixin applied to the webkit spinner | `{}`
    `--paper-input-container-input-webkit-clear` | Mixin applied to the webkit clear button | `{}`
    `--paper-input-container-input-webkit-calendar-picker-indicator` | Mixin applied to the webkit calendar picker indicator | `{}`
    `--paper-input-container-ms-clear` | Mixin applied to the Internet Explorer clear button | `{}`
    `--paper-input-container-underline` | Mixin applied to the underline | `{}`
    `--paper-input-container-underline-focus` | Mixin applied to the underline when the input is focused | `{}`
    `--paper-input-container-underline-disabled` | Mixin applied to the underline when the input is disabled | `{}`
    `--paper-input-prefix` | Mixin applied to the input prefix | `{}`
    `--paper-input-suffix` | Mixin applied to the input suffix | `{}`
    `--paper-input-container-label-before` | Mixin applied to label before pseudo element | {}
    `--paper-input-container-label-after` | Mixin applied to label after pseudo element (useful for required asterisk) | {}

    This element is `display:block` by default, but you can set the `inline`
    attribute to make it `display:inline-block`.
    */
    Polymer({
      /** @override */
      _template: html`
    <style>
      :host {
        display: block;
        padding: 8px 0;
        @apply --paper-input-container;
      }

      :host([inline]) {
        display: inline-block;
      }

      :host([disabled]) {
        pointer-events: none;
        opacity: 0.33;

        @apply --paper-input-container-disabled;
      }

      :host([hidden]) {
        display: none !important;
      }

      [hidden] {
        display: none !important;
      }

      .floated-label-placeholder {
        @apply --paper-font-caption;
      }

      .underline {
        height: 2px;
        position: relative;
      }

      .focused-line {
        @apply --layout-fit;
        border-bottom: 2px solid var(--paper-input-container-focus-color, var(--primary-color));

        -webkit-transform-origin: center center;
        transform-origin: center center;
        -webkit-transform: scale3d(0,1,1);
        transform: scale3d(0,1,1);

        @apply --paper-input-container-underline-focus;
      }

      .underline.is-highlighted .focused-line {
        -webkit-transform: none;
        transform: none;
        -webkit-transition: -webkit-transform 0.25s;
        transition: transform 0.25s;

        @apply --paper-transition-easing;
      }

      .underline.is-invalid .focused-line {
        border-color: var(--paper-input-container-invalid-color, var(--error-color));
        -webkit-transform: none;
        transform: none;
        -webkit-transition: -webkit-transform 0.25s;
        transition: transform 0.25s;

        @apply --paper-transition-easing;
      }

      .unfocused-line {
        @apply --layout-fit;
        border-bottom: 1px solid var(--paper-input-container-color, var(--secondary-text-color));
        @apply --paper-input-container-underline;
      }

      :host([disabled]) .unfocused-line {
        border-bottom: 1px dashed;
        border-color: var(--paper-input-container-color, var(--secondary-text-color));
        @apply --paper-input-container-underline-disabled;
      }

      .input-wrapper {
        @apply --layout-horizontal;
        @apply --layout-center;
        position: relative;
      }

      .input-content {
        @apply --layout-flex-auto;
        @apply --layout-relative;
        max-width: 100%;
      }

      .input-content ::slotted(label),
      .input-content ::slotted(.paper-input-label) {
        position: absolute;
        top: 0;
        left: 0;
        width: 100%;
        font: inherit;
        color: var(--paper-input-container-color, var(--secondary-text-color));
        -webkit-transition: -webkit-transform 0.25s, width 0.25s;
        transition: transform 0.25s, width 0.25s;
        -webkit-transform-origin: left top;
        transform-origin: left top;
        /* Fix for safari not focusing 0-height date/time inputs with -webkit-apperance: none; */
        min-height: 1px;

        @apply --paper-font-common-nowrap;
        @apply --paper-font-subhead;
        @apply --paper-input-container-label;
        @apply --paper-transition-easing;
      }


      .input-content ::slotted(label):before,
      .input-content ::slotted(.paper-input-label):before {
        @apply --paper-input-container-label-before;
      }

      .input-content ::slotted(label):after,
      .input-content ::slotted(.paper-input-label):after {
        @apply --paper-input-container-label-after;
      }

      .input-content.label-is-floating ::slotted(label),
      .input-content.label-is-floating ::slotted(.paper-input-label) {
        -webkit-transform: translateY(-75%) scale(0.75);
        transform: translateY(-75%) scale(0.75);

        /* Since we scale to 75/100 of the size, we actually have 100/75 of the
        original space now available */
        width: 133%;

        @apply --paper-input-container-label-floating;
      }

      :host(:dir(rtl)) .input-content.label-is-floating ::slotted(label),
      :host(:dir(rtl)) .input-content.label-is-floating ::slotted(.paper-input-label) {
        right: 0;
        left: auto;
        -webkit-transform-origin: right top;
        transform-origin: right top;
      }

      .input-content.label-is-highlighted ::slotted(label),
      .input-content.label-is-highlighted ::slotted(.paper-input-label) {
        color: var(--paper-input-container-focus-color, var(--primary-color));

        @apply --paper-input-container-label-focus;
      }

      .input-content.is-invalid ::slotted(label),
      .input-content.is-invalid ::slotted(.paper-input-label) {
        color: var(--paper-input-container-invalid-color, var(--error-color));
      }

      .input-content.label-is-hidden ::slotted(label),
      .input-content.label-is-hidden ::slotted(.paper-input-label) {
        visibility: hidden;
      }

      .input-content ::slotted(input),
      .input-content ::slotted(iron-input),
      .input-content ::slotted(textarea),
      .input-content ::slotted(iron-autogrow-textarea),
      .input-content ::slotted(.paper-input-input) {
        @apply --paper-input-container-shared-input-style;
        /* The apply shim doesn't apply the nested color custom property,
          so we have to re-apply it here. */
        color: var(--paper-input-container-input-color, var(--primary-text-color));
        @apply --paper-input-container-input;
      }

      .input-content ::slotted(input)::-webkit-outer-spin-button,
      .input-content ::slotted(input)::-webkit-inner-spin-button {
        @apply --paper-input-container-input-webkit-spinner;
      }

      .input-content.focused ::slotted(input),
      .input-content.focused ::slotted(iron-input),
      .input-content.focused ::slotted(textarea),
      .input-content.focused ::slotted(iron-autogrow-textarea),
      .input-content.focused ::slotted(.paper-input-input) {
        @apply --paper-input-container-input-focus;
      }

      .input-content.is-invalid ::slotted(input),
      .input-content.is-invalid ::slotted(iron-input),
      .input-content.is-invalid ::slotted(textarea),
      .input-content.is-invalid ::slotted(iron-autogrow-textarea),
      .input-content.is-invalid ::slotted(.paper-input-input) {
        @apply --paper-input-container-input-invalid;
      }

      .prefix ::slotted(*) {
        display: inline-block;
        @apply --paper-font-subhead;
        @apply --layout-flex-none;
        @apply --paper-input-prefix;
      }

      .suffix ::slotted(*) {
        display: inline-block;
        @apply --paper-font-subhead;
        @apply --layout-flex-none;

        @apply --paper-input-suffix;
      }

      /* Firefox sets a min-width on the input, which can cause layout issues */
      .input-content ::slotted(input) {
        min-width: 0;
      }

      .input-content ::slotted(textarea) {
        resize: none;
      }

      .add-on-content {
        position: relative;
      }

      .add-on-content.is-invalid ::slotted(*) {
        color: var(--paper-input-container-invalid-color, var(--error-color));
      }

      .add-on-content.is-highlighted ::slotted(*) {
        color: var(--paper-input-container-focus-color, var(--primary-color));
      }
    </style>

    <div class="floated-label-placeholder" aria-hidden="true" hidden="[[noLabelFloat]]">&nbsp;</div>

    <div class="input-wrapper">
      <span class="prefix"><slot name="prefix"></slot></span>

      <div class$="[[_computeInputContentClass(noLabelFloat,alwaysFloatLabel,focused,invalid,_inputHasContent)]]" id="labelAndInputContainer">
        <slot name="label"></slot>
        <slot name="input"></slot>
      </div>

      <span class="suffix"><slot name="suffix"></slot></span>
    </div>

    <div class$="[[_computeUnderlineClass(focused,invalid)]]">
      <div class="unfocused-line"></div>
      <div class="focused-line"></div>
    </div>

    <div class$="[[_computeAddOnContentClass(focused,invalid)]]">
      <slot name="add-on"></slot>
    </div>
`,

      is: 'paper-input-container',

      properties: {
        /**
         * Set to true to disable the floating label. The label disappears when the
         * input value is not null.
         */
        noLabelFloat: {type: Boolean, value: false},

        /**
         * Set to true to always float the floating label.
         */
        alwaysFloatLabel: {type: Boolean, value: false},

        /**
         * The attribute to listen for value changes on.
         */
        attrForValue: {type: String, value: 'bind-value'},

        /**
         * Set to true to auto-validate the input value when it changes.
         */
        autoValidate: {type: Boolean, value: false},

        /**
         * True if the input is invalid. This property is set automatically when the
         * input value changes if auto-validating, or when the `iron-input-validate`
         * event is heard from a child.
         */
        invalid: {observer: '_invalidChanged', type: Boolean, value: false},

        /**
         * True if the input has focus.
         */
        focused: {readOnly: true, type: Boolean, value: false, notify: true},

        _addons: {
          type: Array
          // do not set a default value here intentionally - it will be initialized
          // lazily when a distributed child is attached, which may occur before
          // configuration for this element in polyfill.
        },

        _inputHasContent: {type: Boolean, value: false},

        _inputSelector:
            {type: String, value: 'input,iron-input,textarea,.paper-input-input'},

        _boundOnFocus: {
          type: Function,
          value: function() {
            return this._onFocus.bind(this);
          }
        },

        _boundOnBlur: {
          type: Function,
          value: function() {
            return this._onBlur.bind(this);
          }
        },

        _boundOnInput: {
          type: Function,
          value: function() {
            return this._onInput.bind(this);
          }
        },

        _boundValueChanged: {
          type: Function,
          value: function() {
            return this._onValueChanged.bind(this);
          }
        }
      },

      listeners: {
        'addon-attached': '_onAddonAttached',
        'iron-input-validate': '_onIronInputValidate'
      },

      get _valueChangedEvent() {
        return this.attrForValue + '-changed';
      },

      get _propertyForValue() {
        return dashToCamelCase(this.attrForValue);
      },

      get _inputElement() {
        return dom(this).querySelector(this._inputSelector);
      },

      get _inputElementValue() {
        return this._inputElement[this._propertyForValue] ||
            this._inputElement.value;
      },

      /** @override */
      ready: function() {
        // Paper-input treats a value of undefined differently at startup than
        // the rest of the time (specifically: it does not validate it at startup,
        // but it does after that. We need to track whether the first time we
        // encounter the value is basically this first time, so that we can validate
        // it correctly the rest of the time. See
        // https://github.com/PolymerElements/paper-input/issues/605
        this.__isFirstValueUpdate = true;
        if (!this._addons) {
          this._addons = [];
        }
        this.addEventListener('focus', this._boundOnFocus, true);
        this.addEventListener('blur', this._boundOnBlur, true);
      },

      /** @override */
      attached: function() {
        if (this.attrForValue) {
          this._inputElement.addEventListener(
              this._valueChangedEvent, this._boundValueChanged);
        } else {
          this.addEventListener('input', this._onInput);
        }

        // Only validate when attached if the input already has a value.
        if (this._inputElementValue && this._inputElementValue != '') {
          this._handleValueAndAutoValidate(this._inputElement);
        } else {
          this._handleValue(this._inputElement);
        }
      },

      /** @private */
      _onAddonAttached: function(event) {
        if (!this._addons) {
          this._addons = [];
        }
        var target = event.target;
        if (this._addons.indexOf(target) === -1) {
          this._addons.push(target);
          if (this.isAttached) {
            this._handleValue(this._inputElement);
          }
        }
      },

      /** @private */
      _onFocus: function() {
        this._setFocused(true);
      },

      /** @private */
      _onBlur: function() {
        this._setFocused(false);
        this._handleValueAndAutoValidate(this._inputElement);
      },

      /** @private */
      _onInput: function(event) {
        this._handleValueAndAutoValidate(event.target);
      },

      /** @private */
      _onValueChanged: function(event) {
        var input = event.target;

        // Paper-input treats a value of undefined differently at startup than
        // the rest of the time (specifically: it does not validate it at startup,
        // but it does after that. If this is in fact the bootup case, ignore
        // validation, just this once.
        if (this.__isFirstValueUpdate) {
          this.__isFirstValueUpdate = false;
          if (input.value === undefined || input.value === '') {
            return;
          }
        }

        this._handleValueAndAutoValidate(event.target);
      },

      /** @private */
      _handleValue: function(inputElement) {
        var value = this._inputElementValue;

        // type="number" hack needed because this.value is empty until it's valid
        if (value || value === 0 ||
            (inputElement.type === 'number' && !inputElement.checkValidity())) {
          this._inputHasContent = true;
        } else {
          this._inputHasContent = false;
        }

        this.updateAddons(
            {inputElement: inputElement, value: value, invalid: this.invalid});
      },

      /** @private */
      _handleValueAndAutoValidate: function(inputElement) {
        if (this.autoValidate && inputElement) {
          var valid;

          if (inputElement.validate) {
            valid = inputElement.validate(this._inputElementValue);
          } else {
            valid = inputElement.checkValidity();
          }
          this.invalid = !valid;
        }

        // Call this last to notify the add-ons.
        this._handleValue(inputElement);
      },

      /** @private */
      _onIronInputValidate: function(event) {
        this.invalid = this._inputElement.invalid;
      },

      /** @private */
      _invalidChanged: function() {
        if (this._addons) {
          this.updateAddons({invalid: this.invalid});
        }
      },

      /**
       * Call this to update the state of add-ons.
       * @param {Object} state Add-on state.
       */
      updateAddons: function(state) {
        for (var addon, index = 0; addon = this._addons[index]; index++) {
          addon.update(state);
        }
      },

      /** @private */
      _computeInputContentClass: function(
          noLabelFloat, alwaysFloatLabel, focused, invalid, _inputHasContent) {
        var cls = 'input-content';
        if (!noLabelFloat) {
          var label = this.querySelector('label');

          if (alwaysFloatLabel || _inputHasContent) {
            cls += ' label-is-floating';
            // If the label is floating, ignore any offsets that may have been
            // applied from a prefix element.
            this.$.labelAndInputContainer.style.position = 'static';

            if (invalid) {
              cls += ' is-invalid';
            } else if (focused) {
              cls += ' label-is-highlighted';
            }
          } else {
            // When the label is not floating, it should overlap the input element.
            if (label) {
              this.$.labelAndInputContainer.style.position = 'relative';
            }
            if (invalid) {
              cls += ' is-invalid';
            }
          }
        } else {
          if (_inputHasContent) {
            cls += ' label-is-hidden';
          }
          if (invalid) {
            cls += ' is-invalid';
          }
        }
        if (focused) {
          cls += ' focused';
        }
        return cls;
      },

      /** @private */
      _computeUnderlineClass: function(focused, invalid) {
        var cls = 'underline';
        if (invalid) {
          cls += ' is-invalid';
        } else if (focused) {
          cls += ' is-highlighted';
        }
        return cls;
      },

      /** @private */
      _computeAddOnContentClass: function(focused, invalid) {
        var cls = 'add-on-content';
        if (invalid) {
          cls += ' is-invalid';
        } else if (focused) {
          cls += ' is-highlighted';
        }
        return cls;
      }
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /*
    `<paper-input-error>` is an error message for use with
    `<paper-input-container>`. The error is displayed when the
    `<paper-input-container>` is `invalid`.

        <paper-input-container>
          <input pattern="[0-9]*">
          <paper-input-error slot="add-on">Only numbers are
    allowed!</paper-input-error>
        </paper-input-container>

    ### Styling

    The following custom properties and mixins are available for styling:

    Custom property | Description | Default
    ----------------|-------------|----------
    `--paper-input-container-invalid-color` | The foreground color of the error | `--error-color`
    `--paper-input-error` | Mixin applied to the error | `{}`
    */
    Polymer({
      /** @override */
      _template: html`
    <style>
      :host {
        display: inline-block;
        visibility: hidden;

        color: var(--paper-input-container-invalid-color, var(--error-color));

        @apply --paper-font-caption;
        @apply --paper-input-error;
        position: absolute;
        left:0;
        right:0;
      }

      :host([invalid]) {
        visibility: visible;
      }

      #a11yWrapper {
        visibility: hidden;
      }

      :host([invalid]) #a11yWrapper {
        visibility: visible;
      }
    </style>

    <!--
    If the paper-input-error element is directly referenced by an
    \`aria-describedby\` attribute, such as when used as a paper-input add-on,
    then applying \`visibility: hidden;\` to the paper-input-error element itself
    does not hide the error.

    For more information, see:
    https://www.w3.org/TR/accname-1.1/#mapping_additional_nd_description
    -->
    <div id="a11yWrapper">
      <slot></slot>
    </div>
`,

      is: 'paper-input-error',
      behaviors: [PaperInputAddonBehavior],

      properties: {
        /**
         * True if the error is showing.
         */
        invalid: {readOnly: true, reflectToAttribute: true, type: Boolean}
      },

      /**
       * This overrides the update function in PaperInputAddonBehavior.
       * @param {{
       *   inputElement: (Element|undefined),
       *   value: (string|undefined),
       *   invalid: boolean
       * }} state -
       *     inputElement: The input element.
       *     value: The input value.
       *     invalid: True if the input value is invalid.
       */
      update: function(state) {
        this._setInvalid(state.invalid);
      }
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    // Generate unique, monotonically increasing IDs for labels (needed by
    // aria-labelledby) and add-ons.
    const PaperInputHelper = {};

    PaperInputHelper.NextLabelID = 1;
    PaperInputHelper.NextAddonID = 1;
    PaperInputHelper.NextInputID = 1;

    /**
     * Use `PaperInputBehavior` to implement inputs with `<paper-input-container>`.
     * This behavior is implemented by `<paper-input>`. It exposes a number of
     * properties from `<paper-input-container>` and `<input is="iron-input">` and
     * they should be bound in your template.
     *
     * The input element can be accessed by the `inputElement` property if you need
     * to access properties or methods that are not exposed.
     * @polymerBehavior PaperInputBehavior
     */
    const PaperInputBehaviorImpl = {

      properties: {
        /**
         * Fired when the input changes due to user interaction.
         *
         * @event change
         */

        /**
         * The label for this input. If you're using PaperInputBehavior to
         * implement your own paper-input-like element, bind this to
         * `<label>`'s content and `hidden` property, e.g.
         * `<label hidden$="[[!label]]">[[label]]</label>` in your `template`
         */
        label: {type: String},

        /**
         * The value for this input. If you're using PaperInputBehavior to
         * implement your own paper-input-like element, bind this to
         * the `<iron-input>`'s `bindValue`
         * property, or the value property of your input that is `notify:true`.
         * @type {*}
         */
        value: {notify: true, type: String},

        /**
         * Set to true to disable this input. If you're using PaperInputBehavior to
         * implement your own paper-input-like element, bind this to
         * both the `<paper-input-container>`'s and the input's `disabled` property.
         */
        disabled: {type: Boolean, value: false},

        /**
         * Returns true if the value is invalid. If you're using PaperInputBehavior
         * to implement your own paper-input-like element, bind this to both the
         * `<paper-input-container>`'s and the input's `invalid` property.
         *
         * If `autoValidate` is true, the `invalid` attribute is managed
         * automatically, which can clobber attempts to manage it manually.
         */
        invalid: {type: Boolean, value: false, notify: true},

        /**
         * Set this to specify the pattern allowed by `preventInvalidInput`. If
         * you're using PaperInputBehavior to implement your own paper-input-like
         * element, bind this to the `<input is="iron-input">`'s `allowedPattern`
         * property.
         */
        allowedPattern: {type: String},

        /**
         * The type of the input. The supported types are the
         * [native input's
         * types](https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input#Form_<input>_types).
         * If you're using PaperInputBehavior to implement your own paper-input-like
         * element, bind this to the (Polymer 1) `<input is="iron-input">`'s or
         * (Polymer 2)
         * `<iron-input>`'s `type` property.
         */
        type: {type: String},

        /**
         * The datalist of the input (if any). This should match the id of an
         * existing `<datalist>`. If you're using PaperInputBehavior to implement
         * your own paper-input-like element, bind this to the `<input
         * is="iron-input">`'s `list` property.
         */
        list: {type: String},

        /**
         * A pattern to validate the `input` with. If you're using
         * PaperInputBehavior to implement your own paper-input-like element, bind
         * this to the `<input is="iron-input">`'s `pattern` property.
         */
        pattern: {type: String},

        /**
         * Set to true to mark the input as required. If you're using
         * PaperInputBehavior to implement your own paper-input-like element, bind
         * this to the `<input is="iron-input">`'s `required` property.
         */
        required: {type: Boolean, value: false},

        /**
         * The error message to display when the input is invalid. If you're using
         * PaperInputBehavior to implement your own paper-input-like element,
         * bind this to the `<paper-input-error>`'s content, if using.
         */
        errorMessage: {type: String},

        /**
         * Set to true to show a character counter.
         */
        charCounter: {type: Boolean, value: false},

        /**
         * Set to true to disable the floating label. If you're using
         * PaperInputBehavior to implement your own paper-input-like element, bind
         * this to the `<paper-input-container>`'s `noLabelFloat` property.
         */
        noLabelFloat: {type: Boolean, value: false},

        /**
         * Set to true to always float the label. If you're using PaperInputBehavior
         * to implement your own paper-input-like element, bind this to the
         * `<paper-input-container>`'s `alwaysFloatLabel` property.
         */
        alwaysFloatLabel: {type: Boolean, value: false},

        /**
         * Set to true to auto-validate the input value. If you're using
         * PaperInputBehavior to implement your own paper-input-like element, bind
         * this to the `<paper-input-container>`'s `autoValidate` property.
         */
        autoValidate: {type: Boolean, value: false},

        /**
         * Name of the validator to use. If you're using PaperInputBehavior to
         * implement your own paper-input-like element, bind this to
         * the `<input is="iron-input">`'s `validator` property.
         */
        validator: {type: String},

        // HTMLInputElement attributes for binding if needed

        /**
         * If you're using PaperInputBehavior to implement your own paper-input-like
         * element, bind this to the `<input is="iron-input">`'s `autocomplete`
         * property.
         */
        autocomplete: {type: String, value: 'off'},

        /**
         * If you're using PaperInputBehavior to implement your own paper-input-like
         * element, bind this to the `<input is="iron-input">`'s `autofocus`
         * property.
         *
         * @type {!boolean}
         */
        autofocus: {type: Boolean, observer: '_autofocusChanged'},

        /**
         * If you're using PaperInputBehavior to implement your own paper-input-like
         * element, bind this to the `<input is="iron-input">`'s `inputmode`
         * property.
         */
        inputmode: {type: String},

        /**
         * The minimum length of the input value.
         * If you're using PaperInputBehavior to implement your own paper-input-like
         * element, bind this to the `<input is="iron-input">`'s `minlength`
         * property.
         */
        minlength: {type: Number},

        /**
         * The maximum length of the input value.
         * If you're using PaperInputBehavior to implement your own paper-input-like
         * element, bind this to the `<input is="iron-input">`'s `maxlength`
         * property.
         */
        maxlength: {type: Number},

        /**
         * The minimum (numeric or date-time) input value.
         * If you're using PaperInputBehavior to implement your own paper-input-like
         * element, bind this to the `<input is="iron-input">`'s `min` property.
         */
        min: {type: String},

        /**
         * The maximum (numeric or date-time) input value.
         * Can be a String (e.g. `"2000-01-01"`) or a Number (e.g. `2`).
         * If you're using PaperInputBehavior to implement your own paper-input-like
         * element, bind this to the `<input is="iron-input">`'s `max` property.
         */
        max: {type: String},

        /**
         * Limits the numeric or date-time increments.
         * If you're using PaperInputBehavior to implement your own paper-input-like
         * element, bind this to the `<input is="iron-input">`'s `step` property.
         */
        step: {type: String},

        /**
         * If you're using PaperInputBehavior to implement your own paper-input-like
         * element, bind this to the `<input is="iron-input">`'s `name` property.
         */
        name: {type: String},

        /**
         * A placeholder string in addition to the label. If this is set, the label
         * will always float.
         */
        placeholder: {
          type: String,
          // need to set a default so _computeAlwaysFloatLabel is run
          value: ''
        },

        /**
         * If you're using PaperInputBehavior to implement your own paper-input-like
         * element, bind this to the `<input is="iron-input">`'s `readonly`
         * property.
         */
        readonly: {type: Boolean, value: false},

        /**
         * If you're using PaperInputBehavior to implement your own paper-input-like
         * element, bind this to the `<input is="iron-input">`'s `size` property.
         */
        size: {type: Number},

        // Nonstandard attributes for binding if needed

        /**
         * If you're using PaperInputBehavior to implement your own paper-input-like
         * element, bind this to the `<input is="iron-input">`'s `autocapitalize`
         * property.
         *
         * @type {string}
         */
        autocapitalize: {type: String, value: 'none'},

        /**
         * If you're using PaperInputBehavior to implement your own paper-input-like
         * element, bind this to the `<input is="iron-input">`'s `autocorrect`
         * property.
         */
        autocorrect: {type: String, value: 'off'},

        /**
         * If you're using PaperInputBehavior to implement your own paper-input-like
         * element, bind this to the `<input is="iron-input">`'s `autosave`
         * property, used with type=search.
         */
        autosave: {type: String},

        /**
         * If you're using PaperInputBehavior to implement your own paper-input-like
         * element, bind this to the `<input is="iron-input">`'s `results` property,
         * used with type=search.
         */
        results: {type: Number},

        /**
         * If you're using PaperInputBehavior to implement your own paper-input-like
         * element, bind this to the `<input is="iron-input">`'s `accept` property,
         * used with type=file.
         */
        accept: {type: String},

        /**
         * If you're using PaperInputBehavior to implement your own paper-input-like
         * element, bind this to the`<input is="iron-input">`'s `multiple` property,
         * used with type=file.
         */
        multiple: {type: Boolean},

        /** @private */
        _ariaDescribedBy: {type: String, value: ''},

        /** @private */
        _ariaLabelledBy: {type: String, value: ''},

        /** @private */
        _inputId: {type: String, value: ''}
      },

      listeners: {
        'addon-attached': '_onAddonAttached',
      },

      /**
       * @type {!Object}
       */
      keyBindings: {'shift+tab:keydown': '_onShiftTabDown'},

      /** @private */
      hostAttributes: {tabindex: 0},

      /**
       * Returns a reference to the input element.
       * @return {!HTMLElement}
       */
      get inputElement() {
        // Chrome generates audit errors if an <input type="password"> has a
        // duplicate ID, which is almost always true in Shady DOM. Generate
        // a unique ID instead.
        if (!this.$) {
          this.$ = {};
        }
        if (!this.$.input) {
          this._generateInputId();
          this.$.input = this.$$('#' + this._inputId);
        }
        return this.$.input;
      },

      /**
       * Returns a reference to the focusable element.
       * @return {!HTMLElement}
       */
      get _focusableElement() {
        return this.inputElement;
      },

      /** @override */
      created: function() {
        // These types have some default placeholder text; overlapping
        // the label on top of it looks terrible. Auto-float the label in this case.
        this._typesThatHaveText =
            ['date', 'datetime', 'datetime-local', 'month', 'time', 'week', 'file'];
      },

      /** @override */
      attached: function() {
        this._updateAriaLabelledBy();

        // In the 2.0 version of the element, this is handled in `onIronInputReady`,
        // i.e. after the native input has finished distributing. In the 1.0
        // version, the input is in the shadow tree, so it's already available.
        if (!PolymerElement && this.inputElement &&
            this._typesThatHaveText.indexOf(this.inputElement.type) !== -1) {
          this.alwaysFloatLabel = true;
        }
      },

      _appendStringWithSpace: function(str, more) {
        if (str) {
          str = str + ' ' + more;
        } else {
          str = more;
        }
        return str;
      },

      _onAddonAttached: function(event) {
        var target = dom(event).rootTarget;
        if (target.id) {
          this._ariaDescribedBy =
              this._appendStringWithSpace(this._ariaDescribedBy, target.id);
        } else {
          var id = 'paper-input-add-on-' + PaperInputHelper.NextAddonID++;
          target.id = id;
          this._ariaDescribedBy =
              this._appendStringWithSpace(this._ariaDescribedBy, id);
        }
      },

      /**
       * Validates the input element and sets an error style if needed.
       *
       * @return {boolean}
       */
      validate: function() {
        return this.inputElement.validate();
      },

      /**
       * Forward focus to inputElement. Overriden from IronControlState.
       */
      _focusBlurHandler: function(event) {
        IronControlState._focusBlurHandler.call(this, event);

        // Forward the focus to the nested input.
        if (this.focused && !this._shiftTabPressed && this._focusableElement) {
          this._focusableElement.focus();
        }
      },

      /**
       * Handler that is called when a shift+tab keypress is detected by the menu.
       *
       * @param {CustomEvent} event A key combination event.
       */
      _onShiftTabDown: function(event) {
        var oldTabIndex = this.getAttribute('tabindex');
        this._shiftTabPressed = true;
        this.setAttribute('tabindex', '-1');
        this.async(function() {
          this.setAttribute('tabindex', oldTabIndex);
          this._shiftTabPressed = false;
        }, 1);
      },

      /**
       * If `autoValidate` is true, then validates the element.
       */
      _handleAutoValidate: function() {
        if (this.autoValidate)
          this.validate();
      },

      /**
       * Restores the cursor to its original position after updating the value.
       * @param {string} newValue The value that should be saved.
       */
      updateValueAndPreserveCaret: function(newValue) {
        // Not all elements might have selection, and even if they have the
        // right properties, accessing them might throw an exception (like for
        // <input type=number>)
        try {
          var start = this.inputElement.selectionStart;
          this.value = newValue;

          // The cursor automatically jumps to the end after re-setting the value,
          // so restore it to its original position.
          this.inputElement.selectionStart = start;
          this.inputElement.selectionEnd = start;
        } catch (e) {
          // Just set the value and give up on the caret.
          this.value = newValue;
        }
      },

      _computeAlwaysFloatLabel: function(alwaysFloatLabel, placeholder) {
        return placeholder || alwaysFloatLabel;
      },

      _updateAriaLabelledBy: function() {
        var label = dom(this.root).querySelector('label');
        if (!label) {
          this._ariaLabelledBy = '';
          return;
        }
        var labelledBy;
        if (label.id) {
          labelledBy = label.id;
        } else {
          labelledBy = 'paper-input-label-' + PaperInputHelper.NextLabelID++;
          label.id = labelledBy;
        }
        this._ariaLabelledBy = labelledBy;
      },

      _generateInputId: function() {
        if (!this._inputId || this._inputId === '') {
          this._inputId = 'input-' + PaperInputHelper.NextInputID++;
        }
      },

      _onChange: function(event) {
        // In the Shadow DOM, the `change` event is not leaked into the
        // ancestor tree, so we must do this manually.
        // See
        // https://w3c.github.io/webcomponents/spec/shadow/#events-that-are-not-leaked-into-ancestor-trees.
        if (this.shadowRoot) {
          this.fire(
              event.type,
              {sourceEvent: event},
              {node: this, bubbles: event.bubbles, cancelable: event.cancelable});
        }
      },

      _autofocusChanged: function() {
        // Firefox doesn't respect the autofocus attribute if it's applied after
        // the page is loaded (Chrome/WebKit do respect it), preventing an
        // autofocus attribute specified in markup from taking effect when the
        // element is upgraded. As a workaround, if the autofocus property is set,
        // and the focus hasn't already been moved elsewhere, we take focus.
        if (this.autofocus && this._focusableElement) {
          // In IE 11, the default document.activeElement can be the page's
          // outermost html element, but there are also cases (under the
          // polyfill?) in which the activeElement is not a real HTMLElement, but
          // just a plain object. We identify the latter case as having no valid
          // activeElement.
          var activeElement = document.activeElement;
          var isActiveElementValid = activeElement instanceof HTMLElement;

          // Has some other element has already taken the focus?
          var isSomeElementActive = isActiveElementValid &&
              activeElement !== document.body &&
              activeElement !== document.documentElement; /* IE 11 */
          if (!isSomeElementActive) {
            // No specific element has taken the focus yet, so we can take it.
            this._focusableElement.focus();
          }
        }
      }
    };

    /** @polymerBehavior */
    const PaperInputBehavior =
        [IronControlState, IronA11yKeysBehavior, PaperInputBehaviorImpl];

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
    Material design: [Text
    fields](https://www.google.com/design/spec/components/text-fields.html)

    `<paper-input>` is a single-line text field with Material Design styling.

        <paper-input label="Input label"></paper-input>

    It may include an optional error message or character counter.

        <paper-input error-message="Invalid input!" label="Input
        label"></paper-input> <paper-input char-counter label="Input
        label"></paper-input>

    It can also include custom prefix or suffix elements, which are displayed
    before or after the text input itself. In order for an element to be
    considered as a prefix, it must have the `prefix` attribute (and similarly
    for `suffix`).

        <paper-input label="total">
          <div prefix>$</div>
          <paper-icon-button slot="suffix" icon="clear"></paper-icon-button>
        </paper-input>

    A `paper-input` can use the native `type=search` or `type=file` features.
    However, since we can't control the native styling of the input (search icon,
    file button, date placeholder, etc.), in these cases the label will be
    automatically floated. The `placeholder` attribute can still be used for
    additional informational text.

        <paper-input label="search!" type="search"
            placeholder="search for cats" autosave="test" results="5">
        </paper-input>

    See `Polymer.PaperInputBehavior` for more API docs.

    ### Focus

    To focus a paper-input, you can call the native `focus()` method as long as the
    paper input has a tab index. Similarly, `blur()` will blur the element.

    ### Styling

    See `Polymer.PaperInputContainer` for a list of custom properties used to
    style this element.

    The following custom properties and mixins are available for styling:

    Custom property | Description | Default
    ----------------|-------------|----------
    `--paper-input-container-ms-clear` | Mixin applied to the Internet Explorer reveal button (the eyeball) | {}

    @element paper-input
    @demo demo/index.html
    */
    Polymer({
      is: 'paper-input',
      /** @override */
      _template: html`
    <style>
      :host {
        display: block;
      }

      :host([focused]) {
        outline: none;
      }

      :host([hidden]) {
        display: none !important;
      }

      input {
        /* Firefox sets a min-width on the input, which can cause layout issues */
        min-width: 0;
      }

      /* In 1.x, the <input> is distributed to paper-input-container, which styles it.
      In 2.x the <iron-input> is distributed to paper-input-container, which styles
      it, but in order for this to work correctly, we need to reset some
      of the native input's properties to inherit (from the iron-input) */
      iron-input > input {
        @apply --paper-input-container-shared-input-style;
        font-family: inherit;
        font-weight: inherit;
        font-size: inherit;
        letter-spacing: inherit;
        word-spacing: inherit;
        line-height: inherit;
        text-shadow: inherit;
        color: inherit;
        cursor: inherit;
      }

      input:disabled {
        @apply --paper-input-container-input-disabled;
      }

      input::-webkit-outer-spin-button,
      input::-webkit-inner-spin-button {
        @apply --paper-input-container-input-webkit-spinner;
      }

      input::-webkit-clear-button {
        @apply --paper-input-container-input-webkit-clear;
      }

      input::-webkit-calendar-picker-indicator {
        @apply --paper-input-container-input-webkit-calendar-picker-indicator;
      }

      input::-webkit-input-placeholder {
        color: var(--paper-input-container-color, var(--secondary-text-color));
      }

      input:-moz-placeholder {
        color: var(--paper-input-container-color, var(--secondary-text-color));
      }

      input::-moz-placeholder {
        color: var(--paper-input-container-color, var(--secondary-text-color));
      }

      input::-ms-clear {
        @apply --paper-input-container-ms-clear;
      }

      input::-ms-reveal {
        @apply --paper-input-container-ms-reveal;
      }

      input:-ms-input-placeholder {
        color: var(--paper-input-container-color, var(--secondary-text-color));
      }

      label {
        pointer-events: none;
      }
    </style>

    <paper-input-container id="container" no-label-float="[[noLabelFloat]]" always-float-label="[[_computeAlwaysFloatLabel(alwaysFloatLabel,placeholder)]]" auto-validate$="[[autoValidate]]" disabled$="[[disabled]]" invalid="[[invalid]]">

      <slot name="prefix" slot="prefix"></slot>

      <label hidden$="[[!label]]" aria-hidden="true" for$="[[_inputId]]" slot="label">[[label]]</label>

      <!-- Need to bind maxlength so that the paper-input-char-counter works correctly -->
      <iron-input bind-value="{{value}}" slot="input" class="input-element" id$="[[_inputId]]" maxlength$="[[maxlength]]" allowed-pattern="[[allowedPattern]]" invalid="{{invalid}}" validator="[[validator]]">
        <input aria-labelledby$="[[_ariaLabelledBy]]" aria-describedby$="[[_ariaDescribedBy]]" disabled$="[[disabled]]" title$="[[title]]" type$="[[type]]" pattern$="[[pattern]]" required$="[[required]]" autocomplete$="[[autocomplete]]" autofocus$="[[autofocus]]" inputmode$="[[inputmode]]" minlength$="[[minlength]]" maxlength$="[[maxlength]]" min$="[[min]]" max$="[[max]]" step$="[[step]]" name$="[[name]]" placeholder$="[[placeholder]]" readonly$="[[readonly]]" list$="[[list]]" size$="[[size]]" autocapitalize$="[[autocapitalize]]" autocorrect$="[[autocorrect]]" on-change="_onChange" tabindex$="[[tabIndex]]" autosave$="[[autosave]]" results$="[[results]]" accept$="[[accept]]" multiple$="[[multiple]]" role$="[[inputRole]]" aria-haspopup$="[[inputAriaHaspopup]]">
      </iron-input>

      <slot name="suffix" slot="suffix"></slot>

      <template is="dom-if" if="[[errorMessage]]">
        <paper-input-error aria-live="assertive" slot="add-on">[[errorMessage]]</paper-input-error>
      </template>

      <template is="dom-if" if="[[charCounter]]">
        <paper-input-char-counter slot="add-on"></paper-input-char-counter>
      </template>

    </paper-input-container>
  `,

      behaviors: [PaperInputBehavior, IronFormElementBehavior],

      properties: {
        value: {
          // Required for the correct TypeScript type-generation
          type: String
        },

        inputRole: {
          type: String,
          value: undefined,
        },

        inputAriaHaspopup: {
          type: String,
          value: undefined,
        },
      },

      /**
       * Returns a reference to the focusable element. Overridden from
       * PaperInputBehavior to correctly focus the native input.
       *
       * @return {!HTMLElement}
       */
      get _focusableElement() {
        return this.inputElement._inputElement;
      },

      // Note: This event is only available in the 1.0 version of this element.
      // In 2.0, the functionality of `_onIronInputReady` is done in
      // PaperInputBehavior::attached.
      listeners: {'iron-input-ready': '_onIronInputReady'},

      _onIronInputReady: function() {
        // Even though this is only used in the next line, save this for
        // backwards compatibility, since the native input had this ID until 2.0.5.
        if (!this.$.nativeInput) {
          this.$.nativeInput = /** @type {!Element} */ (this.$$('input'));
        }
        if (this.inputElement &&
            this._typesThatHaveText.indexOf(this.$.nativeInput.type) !== -1) {
          this.alwaysFloatLabel = true;
        }

        // Only validate when attached if the input already has a value.
        if (!!this.inputElement.bindValue) {
          this.$.container._handleValueAndAutoValidate(this.inputElement);
        }
      },
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
    `<iron-dropdown>` is a generalized element that is useful when you have
    hidden content (`dropdown-content`) that is revealed due to some change in
    state that should cause it to do so.

    Note that this is a low-level element intended to be used as part of other
    composite elements that cause dropdowns to be revealed.

    Examples of elements that might be implemented using an `iron-dropdown`
    include comboboxes, menubuttons, selects. The list goes on.

    The `<iron-dropdown>` element exposes attributes that allow the position
    of the `dropdown-content` relative to the `dropdown-trigger` to be
    configured.

        <iron-dropdown horizontal-align="right" vertical-align="top">
          <div slot="dropdown-content">Hello!</div>
        </iron-dropdown>

    In the above example, the `<div>` assigned to the `dropdown-content` slot will
    be hidden until the dropdown element has `opened` set to true, or when the
    `open` method is called on the element.

    @demo demo/index.html
    */
    Polymer({
      _template: html`
    <style>
      :host {
        position: fixed;
      }

      #contentWrapper ::slotted(*) {
        overflow: auto;
      }

      #contentWrapper.animating ::slotted(*) {
        overflow: hidden;
        pointer-events: none;
      }
    </style>

    <div id="contentWrapper">
      <slot id="content" name="dropdown-content"></slot>
    </div>
`,

      is: 'iron-dropdown',

      behaviors: [
        IronControlState,
        IronA11yKeysBehavior,
        IronOverlayBehavior,
        NeonAnimationRunnerBehavior
      ],

      properties: {
        /**
         * The orientation against which to align the dropdown content
         * horizontally relative to the dropdown trigger.
         * Overridden from `Polymer.IronFitBehavior`.
         */
        horizontalAlign: {type: String, value: 'left', reflectToAttribute: true},

        /**
         * The orientation against which to align the dropdown content
         * vertically relative to the dropdown trigger.
         * Overridden from `Polymer.IronFitBehavior`.
         */
        verticalAlign: {type: String, value: 'top', reflectToAttribute: true},

        /**
         * An animation config. If provided, this will be used to animate the
         * opening of the dropdown. Pass an Array for multiple animations.
         * See `neon-animation` documentation for more animation configuration
         * details.
         */
        openAnimationConfig: {type: Object},

        /**
         * An animation config. If provided, this will be used to animate the
         * closing of the dropdown. Pass an Array for multiple animations.
         * See `neon-animation` documentation for more animation configuration
         * details.
         */
        closeAnimationConfig: {type: Object},

        /**
         * If provided, this will be the element that will be focused when
         * the dropdown opens.
         */
        focusTarget: {type: Object},

        /**
         * Set to true to disable animations when opening and closing the
         * dropdown.
         */
        noAnimations: {type: Boolean, value: false},

        /**
         * By default, the dropdown will constrain scrolling on the page
         * to itself when opened.
         * Set to true in order to prevent scroll from being constrained
         * to the dropdown when it opens.
         * This property is a shortcut to set `scrollAction` to lock or refit.
         * Prefer directly setting the `scrollAction` property.
         */
        allowOutsideScroll:
            {type: Boolean, value: false, observer: '_allowOutsideScrollChanged'}
      },

      listeners: {'neon-animation-finish': '_onNeonAnimationFinish'},

      observers: [
        '_updateOverlayPosition(positionTarget, verticalAlign, horizontalAlign, verticalOffset, horizontalOffset)'
      ],

      /**
       * The element that is contained by the dropdown, if any.
       */
      get containedElement() {
        // Polymer 2.x returns slot.assignedNodes which can contain text nodes.
        var nodes = dom(this.$.content).getDistributedNodes();
        for (var i = 0, l = nodes.length; i < l; i++) {
          if (nodes[i].nodeType === Node.ELEMENT_NODE) {
            return nodes[i];
          }
        }
      },

      ready: function() {
        // Ensure scrollAction is set.
        if (!this.scrollAction) {
          this.scrollAction = this.allowOutsideScroll ? 'refit' : 'lock';
        }
        this._readied = true;
      },

      attached: function() {
        if (!this.sizingTarget || this.sizingTarget === this) {
          this.sizingTarget = this.containedElement || this;
        }
      },

      detached: function() {
        this.cancelAnimation();
      },

      /**
       * Called when the value of `opened` changes.
       * Overridden from `IronOverlayBehavior`
       */
      _openedChanged: function() {
        if (this.opened && this.disabled) {
          this.cancel();
        } else {
          this.cancelAnimation();
          this._updateAnimationConfig();
          IronOverlayBehaviorImpl._openedChanged.apply(this, arguments);
        }
      },

      /**
       * Overridden from `IronOverlayBehavior`.
       */
      _renderOpened: function() {
        if (!this.noAnimations && this.animationConfig.open) {
          this.$.contentWrapper.classList.add('animating');
          this.playAnimation('open');
        } else {
          IronOverlayBehaviorImpl._renderOpened.apply(this, arguments);
        }
      },

      /**
       * Overridden from `IronOverlayBehavior`.
       */
      _renderClosed: function() {
        if (!this.noAnimations && this.animationConfig.close) {
          this.$.contentWrapper.classList.add('animating');
          this.playAnimation('close');
        } else {
          IronOverlayBehaviorImpl._renderClosed.apply(this, arguments);
        }
      },

      /**
       * Called when animation finishes on the dropdown (when opening or
       * closing). Responsible for "completing" the process of opening or
       * closing the dropdown by positioning it or setting its display to
       * none.
       */
      _onNeonAnimationFinish: function() {
        this.$.contentWrapper.classList.remove('animating');
        if (this.opened) {
          this._finishRenderOpened();
        } else {
          this._finishRenderClosed();
        }
      },

      /**
       * Constructs the final animation config from different properties used
       * to configure specific parts of the opening and closing animations.
       */
      _updateAnimationConfig: function() {
        // Update the animation node to be the containedElement.
        var animationNode = this.containedElement;
        var animations = [].concat(this.openAnimationConfig || [])
                             .concat(this.closeAnimationConfig || []);
        for (var i = 0; i < animations.length; i++) {
          animations[i].node = animationNode;
        }
        this.animationConfig = {
          open: this.openAnimationConfig,
          close: this.closeAnimationConfig
        };
      },

      /**
       * Updates the overlay position based on configured horizontal
       * and vertical alignment.
       */
      _updateOverlayPosition: function() {
        if (this.isAttached) {
          // This triggers iron-resize, and iron-overlay-behavior will call refit if
          // needed.
          this.notifyResize();
        }
      },

      /**
       * Sets scrollAction according to the value of allowOutsideScroll.
       * Prefer setting directly scrollAction.
       */
      _allowOutsideScrollChanged: function(allowOutsideScroll) {
        // Wait until initial values are all set.
        if (!this._readied) {
          return;
        }
        if (!allowOutsideScroll) {
          this.scrollAction = 'lock';
        } else if (!this.scrollAction || this.scrollAction === 'lock') {
          this.scrollAction = 'refit';
        }
      },

      /**
       * Apply focus to focusTarget or containedElement
       */
      _applyFocus: function() {
        var focusTarget = this.focusTarget || this.containedElement;
        if (focusTarget && this.opened && !this.noAutoFocus) {
          focusTarget.focus();
        } else {
          IronOverlayBehaviorImpl._applyFocus.apply(this, arguments);
        }
      }
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
     * Use `NeonAnimationBehavior` to implement an animation.
     * @polymerBehavior
     */
    const NeonAnimationBehavior = {

      properties: {

        /**
         * Defines the animation timing.
         */
        animationTiming: {
          type: Object,
          value: function() {
            return {
              duration: 500, easing: 'cubic-bezier(0.4, 0, 0.2, 1)', fill: 'both'
            }
          }
        }

      },

      /**
       * Can be used to determine that elements implement this behavior.
       */
      isNeonAnimation: true,

      /**
       * Do any animation configuration here.
       */
      // configure: function(config) {
      // },

      created: function() {
        if (!document.body.animate) {
          console.warn(
              'No web animations detected. This element will not' +
              ' function without a web animations polyfill.');
        }
      },

      /**
       * Returns the animation timing by mixing in properties from `config` to the
       * defaults defined by the animation.
       */
      timingFromConfig: function(config) {
        if (config.timing) {
          for (var property in config.timing) {
            this.animationTiming[property] = config.timing[property];
          }
        }
        return this.animationTiming;
      },

      /**
       * Sets `transform` and `transformOrigin` properties along with the prefixed
       * versions.
       */
      setPrefixedProperty: function(node, property, value) {
        var map = {
          'transform': ['webkitTransform'],
          'transformOrigin': ['mozTransformOrigin', 'webkitTransformOrigin']
        };
        var prefixes = map[property];
        for (var prefix, index = 0; prefix = prefixes[index]; index++) {
          node.style[prefix] = value;
        }
        node.style[property] = value;
      },

      /**
       * Called when the animation finishes.
       */
      complete: function(config) {}

    };

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */
    /*
    `<fade-in-animation>` animates the opacity of an element from 0 to 1.

    Configuration:
    ```
    {
      name: 'fade-in-animation',
      node: <node>
      timing: <animation-timing>
    }
    ```
    */
    Polymer({

      is: 'fade-in-animation',

      behaviors: [NeonAnimationBehavior],

      configure: function(config) {
        var node = config.node;
        this._effect = new KeyframeEffect(
            node,
            [{'opacity': '0'}, {'opacity': '1'}],
            this.timingFromConfig(config));
        return this._effect;
      }

    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */
    /*
    `<fade-out-animation>` animates the opacity of an element from 1 to 0.

    Configuration:
    ```
    {
      name: 'fade-out-animation',
      node: <node>
      timing: <animation-timing>
    }
    ```
    */
    Polymer({

      is: 'fade-out-animation',

      behaviors: [NeonAnimationBehavior],

      configure: function(config) {
        var node = config.node;
        this._effect = new KeyframeEffect(
            node,
            [
              {'opacity': '1'},
              {'opacity': '0'},
            ],
            this.timingFromConfig(config));
        return this._effect;
      }

    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */
    Polymer({
      is: 'paper-menu-grow-height-animation',

      behaviors: [NeonAnimationBehavior],

      configure: function(config) {
        var node = config.node;
        var rect = node.getBoundingClientRect();
        var height = rect.height;

        this._effect = new KeyframeEffect(
            node,
            [{height: (height / 2) + 'px'}, {height: height + 'px'}],
            this.timingFromConfig(config));

        return this._effect;
      }
    });

    Polymer({
      is: 'paper-menu-grow-width-animation',

      behaviors: [NeonAnimationBehavior],

      configure: function(config) {
        var node = config.node;
        var rect = node.getBoundingClientRect();
        var width = rect.width;

        this._effect = new KeyframeEffect(
            node,
            [{width: (width / 2) + 'px'}, {width: width + 'px'}],
            this.timingFromConfig(config));

        return this._effect;
      }
    });

    Polymer({
      is: 'paper-menu-shrink-width-animation',

      behaviors: [NeonAnimationBehavior],

      configure: function(config) {
        var node = config.node;
        var rect = node.getBoundingClientRect();
        var width = rect.width;

        this._effect = new KeyframeEffect(
            node,
            [{width: width + 'px'}, {width: width - (width / 20) + 'px'}],
            this.timingFromConfig(config));

        return this._effect;
      }
    });

    Polymer({
      is: 'paper-menu-shrink-height-animation',

      behaviors: [NeonAnimationBehavior],

      configure: function(config) {
        var node = config.node;
        var rect = node.getBoundingClientRect();
        var height = rect.height;

        this.setPrefixedProperty(node, 'transformOrigin', '0 0');

        this._effect = new KeyframeEffect(
            node,
            [
              {height: height + 'px', transform: 'translateY(0)'},
              {height: height / 2 + 'px', transform: 'translateY(-20px)'}
            ],
            this.timingFromConfig(config));

        return this._effect;
      }
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    var config = {
      ANIMATION_CUBIC_BEZIER: 'cubic-bezier(.3,.95,.5,1)',
      MAX_ANIMATION_TIME_MS: 400
    };

    /**
    Material design: [Dropdown
    buttons](https://www.google.com/design/spec/components/buttons.html#buttons-dropdown-buttons)

    `paper-menu-button` allows one to compose a designated "trigger" element with
    another element that represents "content", to create a dropdown menu that
    displays the "content" when the "trigger" is clicked.

    The child element assigned to the `dropdown-trigger` slot will be used as the
    "trigger" element. The child element assigned to the `dropdown-content` slot
    will be used as the "content" element.

    The `paper-menu-button` is sensitive to its content's `iron-select` events. If
    the "content" element triggers an `iron-select` event, the `paper-menu-button`
    will close automatically.

    Example:

        <paper-menu-button>
          <paper-icon-button icon="menu"
    slot="dropdown-trigger"></paper-icon-button> <paper-listbox
    slot="dropdown-content"> <paper-item>Share</paper-item>
            <paper-item>Settings</paper-item>
            <paper-item>Help</paper-item>
          </paper-listbox>
        </paper-menu-button>

    ### Styling

    The following custom properties and mixins are also available for styling:

    Custom property | Description | Default
    ----------------|-------------|----------
    `--paper-menu-button-dropdown-background` | Background color of the paper-menu-button dropdown | `--primary-background-color`
    `--paper-menu-button` | Mixin applied to the paper-menu-button | `{}`
    `--paper-menu-button-disabled` | Mixin applied to the paper-menu-button when disabled | `{}`
    `--paper-menu-button-dropdown` | Mixin applied to the paper-menu-button dropdown | `{}`
    `--paper-menu-button-content` | Mixin applied to the paper-menu-button content | `{}`

    @hero hero.svg
    @demo demo/index.html
    */
    const PaperMenuButton = Polymer({
      _template: html`
    <style>
      :host {
        display: inline-block;
        position: relative;
        padding: 8px;
        outline: none;

        @apply --paper-menu-button;
      }

      :host([disabled]) {
        cursor: auto;
        color: var(--disabled-text-color);

        @apply --paper-menu-button-disabled;
      }

      iron-dropdown {
        @apply --paper-menu-button-dropdown;
      }

      .dropdown-content {
        @apply --shadow-elevation-2dp;

        position: relative;
        border-radius: 2px;
        background-color: var(--paper-menu-button-dropdown-background, var(--primary-background-color));

        @apply --paper-menu-button-content;
      }

      :host([vertical-align="top"]) .dropdown-content {
        margin-bottom: 20px;
        margin-top: -10px;
        top: 10px;
      }

      :host([vertical-align="bottom"]) .dropdown-content {
        bottom: 10px;
        margin-bottom: -10px;
        margin-top: 20px;
      }

      #trigger {
        cursor: pointer;
      }
    </style>

    <div id="trigger" on-tap="toggle">
      <slot name="dropdown-trigger"></slot>
    </div>

    <iron-dropdown id="dropdown" opened="{{opened}}" horizontal-align="[[horizontalAlign]]" vertical-align="[[verticalAlign]]" dynamic-align="[[dynamicAlign]]" horizontal-offset="[[horizontalOffset]]" vertical-offset="[[verticalOffset]]" no-overlap="[[noOverlap]]" open-animation-config="[[openAnimationConfig]]" close-animation-config="[[closeAnimationConfig]]" no-animations="[[noAnimations]]" focus-target="[[_dropdownContent]]" allow-outside-scroll="[[allowOutsideScroll]]" restore-focus-on-close="[[restoreFocusOnClose]]" on-iron-overlay-canceled="__onIronOverlayCanceled">
      <div slot="dropdown-content" class="dropdown-content">
        <slot id="content" name="dropdown-content"></slot>
      </div>
    </iron-dropdown>
`,

      is: 'paper-menu-button',

      /**
       * Fired when the dropdown opens.
       *
       * @event paper-dropdown-open
       */

      /**
       * Fired when the dropdown closes.
       *
       * @event paper-dropdown-close
       */

      behaviors: [IronA11yKeysBehavior, IronControlState],

      properties: {
        /**
         * True if the content is currently displayed.
         */
        opened:
            {type: Boolean, value: false, notify: true, observer: '_openedChanged'},

        /**
         * The orientation against which to align the menu dropdown
         * horizontally relative to the dropdown trigger.
         */
        horizontalAlign: {type: String, value: 'left', reflectToAttribute: true},

        /**
         * The orientation against which to align the menu dropdown
         * vertically relative to the dropdown trigger.
         */
        verticalAlign: {type: String, value: 'top', reflectToAttribute: true},

        /**
         * If true, the `horizontalAlign` and `verticalAlign` properties will
         * be considered preferences instead of strict requirements when
         * positioning the dropdown and may be changed if doing so reduces
         * the area of the dropdown falling outside of `fitInto`.
         */
        dynamicAlign: {type: Boolean},

        /**
         * A pixel value that will be added to the position calculated for the
         * given `horizontalAlign`. Use a negative value to offset to the
         * left, or a positive value to offset to the right.
         */
        horizontalOffset: {type: Number, value: 0, notify: true},

        /**
         * A pixel value that will be added to the position calculated for the
         * given `verticalAlign`. Use a negative value to offset towards the
         * top, or a positive value to offset towards the bottom.
         */
        verticalOffset: {type: Number, value: 0, notify: true},

        /**
         * If true, the dropdown will be positioned so that it doesn't overlap
         * the button.
         */
        noOverlap: {type: Boolean},

        /**
         * Set to true to disable animations when opening and closing the
         * dropdown.
         */
        noAnimations: {type: Boolean, value: false},

        /**
         * Set to true to disable automatically closing the dropdown after
         * a selection has been made.
         */
        ignoreSelect: {type: Boolean, value: false},

        /**
         * Set to true to enable automatically closing the dropdown after an
         * item has been activated, even if the selection did not change.
         */
        closeOnActivate: {type: Boolean, value: false},

        /**
         * An animation config. If provided, this will be used to animate the
         * opening of the dropdown.
         */
        openAnimationConfig: {
          type: Object,
          value: function() {
            return [
              {name: 'fade-in-animation', timing: {delay: 100, duration: 200}},
              {
                name: 'paper-menu-grow-width-animation',
                timing: {
                  delay: 100,
                  duration: 150,
                  easing: config.ANIMATION_CUBIC_BEZIER
                }
              },
              {
                name: 'paper-menu-grow-height-animation',
                timing: {
                  delay: 100,
                  duration: 275,
                  easing: config.ANIMATION_CUBIC_BEZIER
                }
              }
            ];
          }
        },

        /**
         * An animation config. If provided, this will be used to animate the
         * closing of the dropdown.
         */
        closeAnimationConfig: {
          type: Object,
          value: function() {
            return [
              {name: 'fade-out-animation', timing: {duration: 150}},
              {
                name: 'paper-menu-shrink-width-animation',
                timing: {
                  delay: 100,
                  duration: 50,
                  easing: config.ANIMATION_CUBIC_BEZIER
                }
              },
              {
                name: 'paper-menu-shrink-height-animation',
                timing: {duration: 200, easing: 'ease-in'}
              }
            ];
          }
        },

        /**
         * By default, the dropdown will constrain scrolling on the page
         * to itself when opened.
         * Set to true in order to prevent scroll from being constrained
         * to the dropdown when it opens.
         */
        allowOutsideScroll: {type: Boolean, value: false},

        /**
         * Whether focus should be restored to the button when the menu closes.
         */
        restoreFocusOnClose: {type: Boolean, value: true},

        /**
         * This is the element intended to be bound as the focus target
         * for the `iron-dropdown` contained by `paper-menu-button`.
         */
        _dropdownContent: {type: Object}
      },

      hostAttributes: {role: 'group', 'aria-haspopup': 'true'},

      listeners:
          {'iron-activate': '_onIronActivate', 'iron-select': '_onIronSelect'},

      /**
       * The content element that is contained by the menu button, if any.
       */
      get contentElement() {
        // Polymer 2.x returns slot.assignedNodes which can contain text nodes.
        var nodes = dom(this.$.content).getDistributedNodes();
        for (var i = 0, l = nodes.length; i < l; i++) {
          if (nodes[i].nodeType === Node.ELEMENT_NODE) {
            return nodes[i];
          }
        }
      },

      /**
       * Toggles the dropdown content between opened and closed.
       */
      toggle: function() {
        if (this.opened) {
          this.close();
        } else {
          this.open();
        }
      },

      /**
       * Make the dropdown content appear as an overlay positioned relative
       * to the dropdown trigger.
       */
      open: function() {
        if (this.disabled) {
          return;
        }

        this.$.dropdown.open();
      },

      /**
       * Hide the dropdown content.
       */
      close: function() {
        this.$.dropdown.close();
      },

      /**
       * When an `iron-select` event is received, the dropdown should
       * automatically close on the assumption that a value has been chosen.
       *
       * @param {CustomEvent} event A CustomEvent instance with type
       * set to `"iron-select"`.
       */
      _onIronSelect: function(event) {
        if (!this.ignoreSelect) {
          this.close();
        }
      },

      /**
       * Closes the dropdown when an `iron-activate` event is received if
       * `closeOnActivate` is true.
       *
       * @param {CustomEvent} event A CustomEvent of type 'iron-activate'.
       */
      _onIronActivate: function(event) {
        if (this.closeOnActivate) {
          this.close();
        }
      },

      /**
       * When the dropdown opens, the `paper-menu-button` fires `paper-open`.
       * When the dropdown closes, the `paper-menu-button` fires `paper-close`.
       *
       * @param {boolean} opened True if the dropdown is opened, otherwise false.
       * @param {boolean} oldOpened The previous value of `opened`.
       */
      _openedChanged: function(opened, oldOpened) {
        if (opened) {
          // TODO(cdata): Update this when we can measure changes in distributed
          // children in an idiomatic way.
          // We poke this property in case the element has changed. This will
          // cause the focus target for the `iron-dropdown` to be updated as
          // necessary:
          this._dropdownContent = this.contentElement;
          this.fire('paper-dropdown-open');
        } else if (oldOpened != null) {
          this.fire('paper-dropdown-close');
        }
      },

      /**
       * If the dropdown is open when disabled becomes true, close the
       * dropdown.
       *
       * @param {boolean} disabled True if disabled, otherwise false.
       */
      _disabledChanged: function(disabled) {
        IronControlState._disabledChanged.apply(this, arguments);
        if (disabled && this.opened) {
          this.close();
        }
      },

      __onIronOverlayCanceled: function(event) {
        var uiEvent = event.detail;
        var trigger = this.$.trigger;
        var path = dom(uiEvent).path;

        if (path.indexOf(trigger) > -1) {
          event.preventDefault();
        }
      }
    });

    Object.keys(config).forEach(function(key) {
      PaperMenuButton[key] = config[key];
    });

    /**
    @license
    Copyright (c) 2016 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */
    const $_documentContainer$1 = document.createElement('template');
    $_documentContainer$1.setAttribute('style', 'display: none;');

    $_documentContainer$1.innerHTML =
        `<iron-iconset-svg name="paper-dropdown-menu" size="24">
<svg><defs>
<g id="arrow-drop-down"><path d="M7 10l5 5 5-5z"></path></g>
</defs></svg>
</iron-iconset-svg>`;

    document.head.appendChild($_documentContainer$1.content);

    /**
    @license
    Copyright (c) 2016 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */
    const $_documentContainer$2 = document.createElement('template');
    $_documentContainer$2.setAttribute('style', 'display: none;');

    $_documentContainer$2.innerHTML =
        `<dom-module id="paper-dropdown-menu-shared-styles">
  <template>
    <style>
      :host {
        display: inline-block;
        position: relative;
        text-align: left;

        /* NOTE(cdata): Both values are needed, since some phones require the
         * value to be \`transparent\`.
         */
        -webkit-tap-highlight-color: rgba(0,0,0,0);
        -webkit-tap-highlight-color: transparent;

        --paper-input-container-input: {
          overflow: hidden;
          white-space: nowrap;
          text-overflow: ellipsis;
          max-width: 100%;
          box-sizing: border-box;
          cursor: pointer;
        };

        @apply --paper-dropdown-menu;
      }

      /* paper-dropdown-menu and paper-dropdown-menu-light both delegate focus
       * to other internal elements which manage focus styling. */
      :host(:focus) {
        outline: none;
      }

      :host(:dir(rtl)) {
        text-align: right;

        @apply(--paper-dropdown-menu);
      }

      :host([disabled]) {
        @apply --paper-dropdown-menu-disabled;
      }

      :host([noink]) paper-ripple {
        display: none;
      }

      :host([no-label-float]) paper-ripple {
        top: 8px;
      }

      paper-ripple {
        top: 12px;
        left: 0px;
        bottom: 8px;
        right: 0px;

        @apply --paper-dropdown-menu-ripple;
      }

      paper-menu-button {
        display: block;
        padding: 0;

        @apply --paper-dropdown-menu-button;
      }

      paper-input {
        @apply --paper-dropdown-menu-input;
      }

      iron-icon {
        color: var(--disabled-text-color);

        @apply --paper-dropdown-menu-icon;
      }
    </style>
  </template>
</dom-module>`;

    document.head.appendChild($_documentContainer$2.content);

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    // LegacyElementMixin dedupes and this is the base class for elements created
    // with the `Polymer` function, so this is only a cache lookup.
    // https://github.com/Polymer/polymer/blob/640bc80ac7177b761d46b2fa9c455c318f2b85c6/lib/legacy/class.js#L533-L534
    const LegacyPolymerElementBase = LegacyElementMixin(HTMLElement);

    /**
    Material design: [Dropdown
    menus](https://www.google.com/design/spec/components/buttons.html#buttons-dropdown-buttons)

    `paper-dropdown-menu` is similar to a native browser select element.
    `paper-dropdown-menu` works with selectable content. The currently selected
    item is displayed in the control. If no item is selected, the `label` is
    displayed instead.

    Example:

        <paper-dropdown-menu label="Your favourite pastry">
          <paper-listbox slot="dropdown-content">
            <paper-item>Croissant</paper-item>
            <paper-item>Donut</paper-item>
            <paper-item>Financier</paper-item>
            <paper-item>Madeleine</paper-item>
          </paper-listbox>
        </paper-dropdown-menu>

    This example renders a dropdown menu with 4 options.

    The child element with the slot `dropdown-content` is used as the dropdown
    menu. This can be a [`paper-listbox`](paper-listbox), or any other or
    element that acts like an [`iron-selector`](iron-selector).

    Specifically, the menu child must fire an
    [`iron-select`](iron-selector#event-iron-select) event when one of its
    children is selected, and an
    [`iron-deselect`](iron-selector#event-iron-deselect) event when a child is
    deselected. The selected or deselected item must be passed as the event's
    `detail.item` property.

    Applications can listen for the `iron-select` and `iron-deselect` events
    to react when options are selected and deselected.

    ### Styling

    The following custom properties and mixins are also available for styling:

    Custom property | Description | Default
    ----------------|-------------|----------
    `--paper-dropdown-menu` | A mixin that is applied to the element host | `{}`
    `--paper-dropdown-menu-disabled` | A mixin that is applied to the element host when disabled | `{}`
    `--paper-dropdown-menu-ripple` | A mixin that is applied to the internal ripple | `{}`
    `--paper-dropdown-menu-button` | A mixin that is applied to the internal menu button | `{}`
    `--paper-dropdown-menu-input` | A mixin that is applied to the internal paper input | `{}`
    `--paper-dropdown-menu-icon` | A mixin that is applied to the internal icon | `{}`

    You can also use any of the `paper-input-container` and `paper-menu-button`
    style mixins and custom properties to style the internal input and menu button
    respectively.

    @element paper-dropdown-menu
    @demo demo/index.html
    */
    Polymer({
      /** @override */
      _template: html`
    <style include="paper-dropdown-menu-shared-styles"></style>

    <paper-menu-button id="menuButton" vertical-align="[[verticalAlign]]" horizontal-align="[[horizontalAlign]]" dynamic-align="[[dynamicAlign]]" vertical-offset="[[_computeMenuVerticalOffset(noLabelFloat, verticalOffset)]]" disabled="[[disabled]]" no-animations="[[noAnimations]]" on-iron-select="_onIronSelect" on-iron-deselect="_onIronDeselect" opened="{{opened}}" close-on-activate allow-outside-scroll="[[allowOutsideScroll]]" restore-focus-on-close="[[restoreFocusOnClose]]">
      <!-- support hybrid mode: user might be using paper-menu-button 1.x which distributes via <content> -->
      <div class="dropdown-trigger" slot="dropdown-trigger">
        <paper-ripple></paper-ripple>
        <!-- paper-input has type="text" for a11y, do not remove -->
        <paper-input id="input" type="text" invalid="[[invalid]]" readonly disabled="[[disabled]]" value="[[value]]" placeholder="[[placeholder]]" error-message="[[errorMessage]]" always-float-label="[[alwaysFloatLabel]]" no-label-float="[[noLabelFloat]]" label="[[label]]" input-role="button" input-aria-haspopup="listbox" autocomplete="off">
          <!-- support hybrid mode: user might be using paper-input 1.x which distributes via <content> -->
          <iron-icon icon="paper-dropdown-menu:arrow-drop-down" suffix slot="suffix"></iron-icon>
        </paper-input>
      </div>
      <slot id="content" name="dropdown-content" slot="dropdown-content"></slot>
    </paper-menu-button>
`,

      is: 'paper-dropdown-menu',

      behaviors: [
        IronButtonState,
        IronControlState,
        IronFormElementBehavior,
        IronValidatableBehavior
      ],

      properties: {
        /**
         * The derived "label" of the currently selected item. This value
         * is the `label` property on the selected item if set, or else the
         * trimmed text content of the selected item.
         */
        selectedItemLabel: {type: String, notify: true, readOnly: true},

        /**
         * The last selected item. An item is selected if the dropdown menu has
         * a child with slot `dropdown-content`, and that child triggers an
         * `iron-select` event with the selected `item` in the `detail`.
         *
         * @type {?Object}
         */
        selectedItem: {type: Object, notify: true, readOnly: true},

        /**
         * The value for this element that will be used when submitting in
         * a form. It reflects the value of `selectedItemLabel`. If set directly,
         * it will not update the `selectedItemLabel` value.
         */
        value: {
          type: String,
          notify: true,
        },

        /**
         * The label for the dropdown.
         */
        label: {type: String},

        /**
         * The placeholder for the dropdown.
         */
        placeholder: {type: String},

        /**
         * The error message to display when invalid.
         */
        errorMessage: {type: String},

        /**
         * True if the dropdown is open. Otherwise, false.
         */
        opened:
            {type: Boolean, notify: true, value: false, observer: '_openedChanged'},

        /**
         * By default, the dropdown will constrain scrolling on the page
         * to itself when opened.
         * Set to true in order to prevent scroll from being constrained
         * to the dropdown when it opens.
         */
        allowOutsideScroll: {type: Boolean, value: false},

        /**
         * Set to true to disable the floating label. Bind this to the
         * `<paper-input-container>`'s `noLabelFloat` property.
         */
        noLabelFloat: {type: Boolean, value: false, reflectToAttribute: true},

        /**
         * Set to true to always float the label. Bind this to the
         * `<paper-input-container>`'s `alwaysFloatLabel` property.
         */
        alwaysFloatLabel: {type: Boolean, value: false},

        /**
         * Set to true to disable animations when opening and closing the
         * dropdown.
         */
        noAnimations: {type: Boolean, value: false},

        /**
         * The orientation against which to align the menu dropdown
         * horizontally relative to the dropdown trigger.
         */
        horizontalAlign: {type: String, value: 'right'},

        /**
         * The orientation against which to align the menu dropdown
         * vertically relative to the dropdown trigger.
         */
        verticalAlign: {type: String, value: 'top'},

        /**
         * Overrides the vertical offset computed in
         * _computeMenuVerticalOffset.
         */
        verticalOffset: Number,

        /**
         * If true, the `horizontalAlign` and `verticalAlign` properties will
         * be considered preferences instead of strict requirements when
         * positioning the dropdown and may be changed if doing so reduces
         * the area of the dropdown falling outside of `fitInto`.
         */
        dynamicAlign: {type: Boolean},

        /**
         * Whether focus should be restored to the dropdown when the menu closes.
         */
        restoreFocusOnClose: {type: Boolean, value: true},
      },

      listeners: {'tap': '_onTap'},

      /**
       * @type {!Object}
       */
      keyBindings: {'up down': 'open', 'esc': 'close'},

      observers: ['_selectedItemChanged(selectedItem)'],

      /**
       * Override `_attachDom` so that we can pass `delegatesFocus`. The overridden
       * implementation of `_attachDom` specifically skips the steps performed here
       * if the node already hosts a shadow root:
       * https://github.com/Polymer/polymer/blob/640bc80ac7177b761d46b2fa9c455c318f2b85c6/lib/mixins/element-mixin.js#L691-L694
       * @override
       */
      _attachDom(dom) {
        const wrappedThis = wrap(this);
        wrappedThis.attachShadow({
          mode: 'open',
          delegatesFocus: true,
          shadyUpgradeFragment: dom,
        });
        wrappedThis.shadowRoot.appendChild(dom);
        return LegacyPolymerElementBase.prototype._attachDom.call(this, dom);
      },

      /** @override */
      focus() {
        // When using Shady DOM and in browsers that don't support
        // `delegatesFocus`, attempting to focus this element with the browser's
        // native `HTMLElement#focus` will cause focus to be lost because this
        // element isn't focusable in those situations. To work around this, the
        // element in the shadow root that this element intends to delegate focus
        // to is manually focused instead.
        this.$.input._focusableElement.focus();
      },

      /** @override */
      attached: function() {
        // NOTE(cdata): Due to timing, a preselected value in a `IronSelectable`
        // child will cause an `iron-select` event to fire while the element is
        // still in a `DocumentFragment`. This has the effect of causing
        // handlers not to fire. So, we double check this value on attached:
        var contentElement = this.contentElement;
        if (contentElement && contentElement.selectedItem) {
          this._setSelectedItem(contentElement.selectedItem);
        }
      },

      /**
       * The content element that is contained by the dropdown menu, if any.
       */
      get contentElement() {
        // Polymer 2.x returns slot.assignedNodes which can contain text nodes.
        var nodes = dom(this.$.content).getDistributedNodes();
        for (var i = 0, l = nodes.length; i < l; i++) {
          if (nodes[i].nodeType === Node.ELEMENT_NODE) {
            return nodes[i];
          }
        }
      },

      /**
       * Show the dropdown content.
       */
      open: function() {
        this.$.menuButton.open();
      },

      /**
       * Hide the dropdown content.
       */
      close: function() {
        this.$.menuButton.close();
      },

      /**
       * A handler that is called when `iron-select` is fired.
       *
       * @param {CustomEvent} event An `iron-select` event.
       */
      _onIronSelect: function(event) {
        this._setSelectedItem(event.detail.item);
      },

      /**
       * A handler that is called when `iron-deselect` is fired.
       *
       * @param {CustomEvent} event An `iron-deselect` event.
       */
      _onIronDeselect: function(event) {
        this._setSelectedItem(null);
      },

      /**
       * A handler that is called when the dropdown is tapped.
       *
       * @param {CustomEvent} event A tap event.
       */
      _onTap: function(event) {
        if (findOriginalTarget(event) === this) {
          this.open();
        }
      },

      /**
       * Compute the label for the dropdown given a selected item.
       *
       * @param {Element} selectedItem A selected Element item, with an
       * optional `label` property.
       */
      _selectedItemChanged: function(selectedItem) {
        var value = '';
        if (!selectedItem) {
          value = '';
        } else {
          value = selectedItem.label || selectedItem.getAttribute('label') ||
              selectedItem.textContent.trim();
        }

        this.value = value;
        this._setSelectedItemLabel(value);
      },

      /**
       * Compute the vertical offset of the menu based on the value of
       * `noLabelFloat`.
       *
       * @param {boolean} noLabelFloat True if the label should not float
       * @param {number=} opt_verticalOffset Optional offset from the user
       * above the input, otherwise false.
       */
      _computeMenuVerticalOffset: function(noLabelFloat, opt_verticalOffset) {
        // Override offset if it's passed from the user.
        if (opt_verticalOffset) {
          return opt_verticalOffset;
        }

        // NOTE(cdata): These numbers are somewhat magical because they are
        // derived from the metrics of elements internal to `paper-input`'s
        // template. The metrics will change depending on whether or not the
        // input has a floating label.
        return noLabelFloat ? -4 : 8;
      },

      /**
       * Returns false if the element is required and does not have a selection,
       * and true otherwise.
       * @param {*=} _value Ignored.
       * @return {boolean} true if `required` is false, or if `required` is true
       * and the element has a valid selection.
       */
      _getValidity: function(_value) {
        return this.disabled || !this.required || (this.required && !!this.value);
      },

      _openedChanged: function() {
        var openState = this.opened ? 'true' : 'false';
        var e = this.contentElement;
        if (e) {
          e.setAttribute('aria-expanded', openState);
        }
      }
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    var SHADOW_WHEN_SCROLLING = 1;
    var SHADOW_ALWAYS = 2;
    var MODE_CONFIGS = {
      outerScroll: {'scroll': true},

      shadowMode: {
        'standard': SHADOW_ALWAYS,
        'waterfall': SHADOW_WHEN_SCROLLING,
        'waterfall-tall': SHADOW_WHEN_SCROLLING
      },

      tallMode: {'waterfall-tall': true}
    };

    /**
    **This element has been deprecated in favor of
    [app-layout](https://github.com/PolymerElements/app-layout).**

    `paper-header-panel` contains a header section and a content panel section.

    __Important:__ The `paper-header-panel` will not display if its parent does not
    have a height.

    Using layout classes, you can make the `paper-header-panel` fill the screen

        <body class="fullbleed layout vertical">
          <paper-header-panel class="flex">
            <paper-toolbar slot="header">
              <div>Hello World!</div>
            </paper-toolbar>
          </paper-header-panel>
        </body>

    Special support is provided for scrolling modes when one uses a paper-toolbar or
    equivalent for the header section.

    Example:

        <paper-header-panel>
          <paper-toolbar slot="header">Header</paper-toolbar>
          <div>Content goes here...</div>
        </paper-header-panel>

    To have the content fit to the main area, use the `fit` class.

    Example:

        <paper-header-panel>
          <div slot="header">standard</div>
          <div class="fit">content fits 100% below the header</div>
        </paper-header-panel>

    ### Modes

    Controls header and scrolling behavior. Options are `standard`, `seamed`,
    `waterfall`, `waterfall-tall`, `scroll` and `cover`. Default is `standard`.

    Mode | Description
    ----------------|-------------
    `standard` | The header is a step above the panel. The header will consume the panel at the point of entry, preventing it from passing through to the opposite side.
    `seamed` | The header is presented as seamed with the panel.
    `waterfall` | Similar to standard mode, but header is initially presented as seamed with panel, but then separates to form the step.
    `waterfall-tall` | The header is initially taller (`tall` class is added to the header). As the user scrolls, the header separates (forming an edge) while condensing (`tall` class is removed from the header).
    `scroll` | The header keeps its seam with the panel, and is pushed off screen.
    `cover` | The panel covers the whole `paper-header-panel` including the header. This allows user to style the panel in such a way that the panel is partially covering the header.

    ### Styling

    To change the shadow that shows up underneath the header:

        paper-header-panel {
          --paper-header-panel-shadow: {
              height: 6px;
              bottom: -6px;
              box-shadow: inset 0px 5px 6px -3px rgba(0, 0, 0, 0.4);
          };
        }

    To change the panel container:

        paper-header-panel {
          --paper-header-panel-container: {
            border: 1px solid gray;
          };

    To change the panel container in different modes:

        paper-header-panel {
          --paper-header-panel-standard-container: {
            border: 1px solid gray;
          };

          --paper-header-panel-seamed-container: {
            border: 1px solid gray;
          };

          --paper-header-panel-waterfall-container: {
            border: 1px solid gray;
          };

          --paper-header-panel-waterfall-tall-container: {
            border: 1px solid gray;
          };

          --paper-header-panel-scroll-container: {
            border: 1px solid gray;
          };

          --paper-header-panel-cover-container: {
            border: 1px solid gray;
          };
        }

    The following custom properties and mixins are available for styling:

    Custom property | Description | Default
    ----------------|-------------|----------
    `--paper-header-panel` | Mixin applied to the element | `{}`
    `--paper-header-panel-body` | Mixin applied to the element's body (i.e. everything below the toolbar) | `{}`
    `--paper-header-panel-container` | Mixin applied to the container in any mode | `{}`
    `--paper-header-panel-scroll-container` | Mixin applied to the container when in scroll mode | `{}`
    `--paper-header-panel-cover-container` | Mixin applied to the container when in cover mode | `{}`
    `--paper-header-panel-standard-container` | Mixin applied to the container when in standard mode | `{}`
    `--paper-header-panel-seamed-container` | Mixin applied to the container when in seamed mode | `{}`
    `--paper-header-panel-waterfall-container` | Mixin applied to the container when in waterfall mode | `{}`
    `--paper-header-panel-waterfall-tall-container` | Mixin applied to the container when in tall waterfall mode | `{}`
    `--paper-header-panel-shadow` | Mixin applied to the waterfall shadow | `{}`

    @group Paper Elements
    @element paper-header-panel
    @demo demo/index.html
    @hero hero.svg
    */
    Polymer({
      _template: html`
    <style>
      :host {
        @apply --layout-vertical;
        position: relative;
        height: 100%;
        @apply --paper-header-panel;
      }

      #mainContainer {
        @apply --layout-flex;
        position: relative;
        overflow-y: auto;
        overflow-x: hidden;
        -webkit-overflow-scrolling: touch;
      }

      #mainPanel {
        @apply --layout-vertical;
        @apply --layout-flex;
        position: relative;
        min-height: 0;
        @apply --paper-header-panel-body;
      }

      #mainContainer {
        @apply --paper-header-panel-container;
      }

      /*
       * mode: scroll
       */
      :host([mode=scroll]) #mainContainer {
        @apply --paper-header-panel-scroll-container;
        overflow: visible;
      }

      :host([mode=scroll]) {
        overflow-y: auto;
        overflow-x: hidden;
        -webkit-overflow-scrolling: touch;
      }

      /*
       * mode: cover
       */
      :host([mode=cover]) #mainContainer {
        @apply --paper-header-panel-cover-container;
        position: absolute;
        top: 0;
        right: 0;
        bottom: 0;
        left: 0;
      }

      :host([mode=cover]) #mainPanel {
        position: static;
      }

      /*
       * mode: standard
       */
      :host([mode=standard]) #mainContainer {
        @apply --paper-header-panel-standard-container;
      }

      /*
       * mode: seamed
       */
      :host([mode=seamed]) #mainContainer {
        @apply --paper-header-panel-seamed-container;
      }


      /*
       * mode: waterfall
       */
      :host([mode=waterfall]) #mainContainer {
        @apply --paper-header-panel-waterfall-container;
      }

      /*
       * mode: waterfall-tall
       */
      :host([mode=waterfall-tall]) #mainContainer {
        @apply --paper-header-panel-waterfall-tall-container;
      }

      #dropShadow {
        transition: opacity 0.5s;
        height: 6px;
        box-shadow: inset 0px 5px 6px -3px rgba(0, 0, 0, 0.4);
        @apply --paper-header-panel-shadow;
        position: absolute;
        top: 0;
        left: 0;
        right: 0;
        opacity: 0;
        pointer-events: none;
      }

      #dropShadow.has-shadow {
        opacity: 1;
      }

      #mainContainer > ::slotted(.fit) {
        @apply --layout-fit;
      }

    </style>

    <slot id="headerSlot" name="header"></slot>

    <div id="mainPanel">
      <div id="mainContainer" class\$="[[_computeMainContainerClass(mode)]]">
        <slot></slot>
      </div>
      <div id="dropShadow"></div>
    </div>
`,

      is: 'paper-header-panel',

      /**
       * Fired when the content has been scrolled. `event.detail.target` returns
       * the scrollable element which you can use to access scroll info such as
       * `scrollTop`.
       *
       *     <paper-header-panel on-content-scroll="scrollHandler">
       *       ...
       *     </paper-header-panel>
       *
       *
       *     scrollHandler: function(event) {
       *       var scroller = event.detail.target;
       *       console.log(scroller.scrollTop);
       *     }
       *
       * @event content-scroll
       */

      properties: {
        /**
         * Controls header and scrolling behavior. Options are
         * `standard`, `seamed`, `waterfall`, `waterfall-tall`, `scroll` and
         * `cover`. Default is `standard`.
         *
         * `standard`: The header is a step above the panel. The header will consume
         * the panel at the point of entry, preventing it from passing through to
         * the opposite side.
         *
         * `seamed`: The header is presented as seamed with the panel.
         *
         * `waterfall`: Similar to standard mode, but header is initially presented
         * as seamed with panel, but then separates to form the step.
         *
         * `waterfall-tall`: The header is initially taller (`tall` class is added
         * to the header).  As the user scrolls, the header separates (forming an
         * edge) while condensing (`tall` class is removed from the header).
         *
         * `scroll`: The header keeps its seam with the panel, and is pushed off
         * screen.
         *
         * `cover`: The panel covers the whole `paper-header-panel` including the
         * header. This allows user to style the panel in such a way that the panel
         * is partially covering the header.
         *
         *     <paper-header-panel mode="cover">
         *       <paper-toolbar slot="header" class="tall">
         *         <paper-icon-button icon="menu"></paper-icon-button>
         *       </paper-toolbar>
         *       <div></div>
         *     </paper-header-panel>
         */
        mode: {
          type: String,
          value: 'standard',
          observer: '_modeChanged',
          reflectToAttribute: true
        },

        /**
         * If true, the drop-shadow is always shown no matter what mode is set to.
         */
        shadow: {type: Boolean, value: false},

        /**
         * The class used in waterfall-tall mode.  Change this if the header
         * accepts a different class for toggling height, e.g. "medium-tall"
         */
        tallClass: {type: String, value: 'tall'},

        /**
         * If true, the scroller is at the top
         */
        atTop: {
          type: Boolean,
          value: true,
          notify: true,
          readOnly: true,
          reflectToAttribute: true
        }
      },

      observers: ['_computeDropShadowHidden(atTop, mode, shadow)'],

      attached: function() {
        this._addListener();
        // Run `scroll` logic once to initialize class names, etc.
        this._keepScrollingState();
      },

      detached: function() {
        this._removeListener();
      },

      ready: function() {
        this.scrollHandler = this._scroll.bind(this);
        console.warn(this.is, 'is deprecated. Please use app-layout instead!');
      },

      /**
       * Returns the header element
       * @return {!Element|undefined}
       */
      get header() {
        return dom(this.$.headerSlot).getDistributedNodes()[0];
      },

      /**
       * Returns the scrollable element.
       * @return {!HTMLElement}
       */
      get scroller() {
        return this._getScrollerForMode(this.mode);
      },

      /**
       * Returns true if the scroller has a visible shadow.
       * @return {boolean}
       */
      get visibleShadow() {
        return this.$.dropShadow.classList.contains('has-shadow');
      },

      _computeDropShadowHidden: function(atTop, mode, shadow) {
        var shadowMode = MODE_CONFIGS.shadowMode[mode];

        if (this.shadow) {
          this.toggleClass('has-shadow', true, this.$.dropShadow);
        } else if (shadowMode === SHADOW_ALWAYS) {
          this.toggleClass('has-shadow', true, this.$.dropShadow);
        } else if (shadowMode === SHADOW_WHEN_SCROLLING && !atTop) {
          this.toggleClass('has-shadow', true, this.$.dropShadow);
        } else {
          this.toggleClass('has-shadow', false, this.$.dropShadow);
        }
      },

      _computeMainContainerClass: function(mode) {
        // TODO:  It will be useful to have a utility for classes
        // e.g. Polymer.Utils.classes({ foo: true });

        var classes = {};

        classes['flex'] = mode !== 'cover';

        return Object.keys(classes)
            .filter(function(className) {
              return classes[className];
            })
            .join(' ');
      },

      _addListener: function() {
        this.scroller.addEventListener('scroll', this.scrollHandler);
      },

      _removeListener: function() {
        this.scroller.removeEventListener('scroll', this.scrollHandler);
      },

      _modeChanged: function(newMode, oldMode) {
        var configs = MODE_CONFIGS;
        var header = this.header;
        var animateDuration = 200;

        if (header) {
          // in tallMode it may add tallClass to the header; so do the cleanup
          // when mode is changed from tallMode to not tallMode
          if (configs.tallMode[oldMode] && !configs.tallMode[newMode]) {
            header.classList.remove(this.tallClass);
            this.async(function() {
              header.classList.remove('animate');
            }, animateDuration);
          } else {
            this.toggleClass('animate', configs.tallMode[newMode], header);
          }
        }
        this._keepScrollingState();
      },

      _keepScrollingState: function() {
        var main = this.scroller;
        var header = this.header;

        this._setAtTop(main.scrollTop === 0);

        if (header && this.tallClass && MODE_CONFIGS.tallMode[this.mode]) {
          this.toggleClass(
              this.tallClass,
              this.atTop ||
                  header.classList.contains(this.tallClass) &&
                      main.scrollHeight < this.offsetHeight,
              header);
        }
      },

      _scroll: function() {
        this._keepScrollingState();
        this.fire('content-scroll', {target: this.scroller}, {bubbles: false});
      },

      _getScrollerForMode: function(mode) {
        return MODE_CONFIGS.outerScroll[mode] ? this : this.$.mainContainer;
      }
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
    Material design: [Icon
    toggles](https://www.google.com/design/spec/components/buttons.html#buttons-toggle-buttons)

    `paper-icon-button` is a button with an image placed at the center. When the
    user touches the button, a ripple effect emanates from the center of the button.

    `paper-icon-button` does not include a default icon set. To use icons from the
    default set, include `PolymerElements/iron-icons/iron-icons.html`, and use the
    `icon` attribute to specify which icon from the icon set to use.

        <paper-icon-button icon="menu"></paper-icon-button>

    See [`iron-iconset`](iron-iconset) for more information about
    how to use a custom icon set.

    Example:

        <script type="module">
          import '@polymer/iron-icons/iron-icons.js';
        </JAVA_SCRIIIIPT/>

        <paper-icon-button icon="favorite"></paper-icon-button>
        <paper-icon-button src="star.png"></paper-icon-button>

    To use `paper-icon-button` as a link, wrap it in an anchor tag. Since
    `paper-icon-button` will already receive focus, you may want to prevent the
    anchor tag from receiving focus as well by setting its tabindex to -1.

        <a href="https://www.polymer-project.org" tabindex="-1">
          <paper-icon-button icon="polymer"></paper-icon-button>
        </a>

    ### Styling

    Style the button with CSS as you would a normal DOM element. If you are using
    the icons provided by `iron-icons`, they will inherit the foreground color of
    the button.

        /* make a red "favorite" button *\/
        <paper-icon-button icon="favorite" style="color: red;"></paper-icon-button>

    By default, the ripple is the same color as the foreground at 25% opacity. You
    may customize the color using the `--paper-icon-button-ink-color` custom
    property.

    The following custom properties and mixins are available for styling:

    Custom property | Description | Default
    ----------------|-------------|----------
    `--paper-icon-button-disabled-text` | The color of the disabled button | `--disabled-text-color`
    `--paper-icon-button-ink-color` | Selected/focus ripple color | `--primary-text-color`
    `--paper-icon-button` | Mixin for a button | `{}`
    `--paper-icon-button-disabled` | Mixin for a disabled button | `{}`
    `--paper-icon-button-hover` | Mixin for button on hover | `{}`

    @group Paper Elements
    @element paper-icon-button
    @demo demo/index.html
    */
    Polymer({
      is: 'paper-icon-button',

      _template: html`
    <style>
      :host {
        display: inline-block;
        position: relative;
        padding: 8px;
        outline: none;
        -webkit-user-select: none;
        -moz-user-select: none;
        -ms-user-select: none;
        user-select: none;
        cursor: pointer;
        z-index: 0;
        line-height: 1;

        width: 40px;
        height: 40px;

        /*
          NOTE: Both values are needed, since some phones require the value to
          be \`transparent\`.
        */
        -webkit-tap-highlight-color: rgba(0, 0, 0, 0);
        -webkit-tap-highlight-color: transparent;

        /* Because of polymer/2558, this style has lower specificity than * */
        box-sizing: border-box !important;

        @apply --paper-icon-button;
      }

      :host #ink {
        color: var(--paper-icon-button-ink-color, var(--primary-text-color));
        opacity: 0.6;
      }

      :host([disabled]) {
        color: var(--paper-icon-button-disabled-text, var(--disabled-text-color));
        pointer-events: none;
        cursor: auto;

        @apply --paper-icon-button-disabled;
      }

      :host([hidden]) {
        display: none !important;
      }

      :host(:hover) {
        @apply --paper-icon-button-hover;
      }

      iron-icon {
        --iron-icon-width: 100%;
        --iron-icon-height: 100%;
      }
    </style>

    <iron-icon id="icon" src="[[src]]" icon="[[icon]]"
               alt$="[[alt]]"></iron-icon>
  `,

      hostAttributes: {role: 'button', tabindex: '0'},

      behaviors: [PaperInkyFocusBehavior],

      registered: function() {
        this._template.setAttribute('strip-whitespace', '');
      },

      properties: {
        /**
         * The URL of an image for the icon. If the src property is specified,
         * the icon property should not be.
         */
        src: {type: String},

        /**
         * Specifies the icon name or index in the set of icons available in
         * the icon's icon set. If the icon property is specified,
         * the src property should not be.
         */
        icon: {type: String},

        /**
         * Specifies the alternate text for the button, for accessibility.
         */
        alt: {type: String, observer: '_altChanged'}
      },

      _altChanged: function(newValue, oldValue) {
        var label = this.getAttribute('aria-label');

        // Don't stomp over a user-set aria-label.
        if (!label || oldValue == label) {
          this.setAttribute('aria-label', newValue);
        }
      }
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
    `iron-autogrow-textarea` is an element containing a textarea that grows in
    height as more lines of input are entered. Unless an explicit height or the
    `maxRows` property is set, it will never scroll.

    Example:

        <iron-autogrow-textarea></iron-autogrow-textarea>

    ### Styling

    The following custom properties and mixins are available for styling:

    Custom property | Description | Default
    ----------------|-------------|----------
    `--iron-autogrow-textarea` | Mixin applied to the textarea | `{}`
    `--iron-autogrow-textarea-placeholder` | Mixin applied to the textarea placeholder | `{}`

    @demo demo/index.html
    */
    Polymer({
      /** @override */
      _template: html`
    <style>
      :host {
        display: inline-block;
        position: relative;
        width: 400px;
        border: 1px solid;
        padding: 2px;
        -moz-appearance: textarea;
        -webkit-appearance: textarea;
        overflow: hidden;
      }

      .mirror-text {
        visibility: hidden;
        word-wrap: break-word;
        @apply --iron-autogrow-textarea;
      }

      .fit {
        @apply --layout-fit;
      }

      textarea {
        position: relative;
        outline: none;
        border: none;
        resize: none;
        background: inherit;
        color: inherit;
        /* see comments in template */
        width: 100%;
        height: 100%;
        font-size: inherit;
        font-family: inherit;
        line-height: inherit;
        text-align: inherit;
        @apply --iron-autogrow-textarea;
      }

      textarea::-webkit-input-placeholder {
        @apply --iron-autogrow-textarea-placeholder;
      }

      textarea:-moz-placeholder {
        @apply --iron-autogrow-textarea-placeholder;
      }

      textarea::-moz-placeholder {
        @apply --iron-autogrow-textarea-placeholder;
      }

      textarea:-ms-input-placeholder {
        @apply --iron-autogrow-textarea-placeholder;
      }
    </style>

    <!-- the mirror sizes the input/textarea so it grows with typing -->
    <!-- use &#160; instead &nbsp; of to allow this element to be used in XHTML -->
    <div id="mirror" class="mirror-text" aria-hidden="true">&nbsp;</div>

    <!-- size the input/textarea with a div, because the textarea has intrinsic size in ff -->
    <div class="textarea-container fit">
      <textarea id="textarea" name$="[[name]]" aria-label$="[[label]]" autocomplete$="[[autocomplete]]" autofocus$="[[autofocus]]" autocapitalize$="[[autocapitalize]]" inputmode$="[[inputmode]]" placeholder$="[[placeholder]]" readonly$="[[readonly]]" required$="[[required]]" disabled$="[[disabled]]" rows$="[[rows]]" minlength$="[[minlength]]" maxlength$="[[maxlength]]"></textarea>
    </div>
`,

      is: 'iron-autogrow-textarea',
      behaviors: [IronValidatableBehavior, IronControlState],

      properties: {
        /**
         * Use this property instead of `bind-value` for two-way data binding.
         * @type {string|number}
         */
        value: {observer: '_valueChanged', type: String, notify: true},

        /**
         * This property is deprecated, and just mirrors `value`. Use `value`
         * instead.
         * @type {string|number}
         */
        bindValue: {observer: '_bindValueChanged', type: String, notify: true},

        /**
         * The initial number of rows.
         *
         * @attribute rows
         * @type number
         * @default 1
         */
        rows: {type: Number, value: 1, observer: '_updateCached'},

        /**
         * The maximum number of rows this element can grow to until it
         * scrolls. 0 means no maximum.
         *
         * @attribute maxRows
         * @type number
         * @default 0
         */
        maxRows: {type: Number, value: 0, observer: '_updateCached'},

        /**
         * Bound to the textarea's `autocomplete` attribute.
         */
        autocomplete: {type: String, value: 'off'},

        /**
         * Bound to the textarea's `autofocus` attribute.
         *
         * @type {!boolean}
         */
        autofocus: {type: Boolean, value: false},

        /**
         * Bound to the textarea's `autocapitalize` attribute.
         */
        autocapitalize: {type: String, value: 'none'},

        /**
         * Bound to the textarea's `inputmode` attribute.
         */
        inputmode: {type: String},

        /**
         * Bound to the textarea's `placeholder` attribute.
         */
        placeholder: {type: String},

        /**
         * Bound to the textarea's `readonly` attribute.
         */
        readonly: {type: String},

        /**
         * Set to true to mark the textarea as required.
         */
        required: {type: Boolean},

        /**
         * The minimum length of the input value.
         */
        minlength: {type: Number},

        /**
         * The maximum length of the input value.
         */
        maxlength: {type: Number},

        /**
         * Bound to the textarea's `aria-label` attribute.
         */
        label: {type: String}

      },

      listeners: {'input': '_onInput'},

      /**
       * Returns the underlying textarea.
       * @return {!HTMLTextAreaElement}
       */
      get textarea() {
        return /** @type {!HTMLTextAreaElement} */ (this.$.textarea);
      },

      /**
       * Returns textarea's selection start.
       * @return {number}
       */
      get selectionStart() {
        return this.$.textarea.selectionStart;
      },

      /**
       * Returns textarea's selection end.
       * @return {number}
       */
      get selectionEnd() {
        return this.$.textarea.selectionEnd;
      },

      /**
       * Sets the textarea's selection start.
       */
      set selectionStart(value) {
        this.$.textarea.selectionStart = value;
      },

      /**
       * Sets the textarea's selection end.
       */
      set selectionEnd(value) {
        this.$.textarea.selectionEnd = value;
      },

      /** @override */
      attached: function() {
        /* iOS has an arbitrary left margin of 3px that isn't present
         * in any other browser, and means that the paper-textarea's cursor
         * overlaps the label.
         * See https://github.com/PolymerElements/paper-input/issues/468.
         */
        var IS_IOS = navigator.userAgent.match(/iP(?:[oa]d|hone)/) &&
            !navigator.userAgent.match(/OS 1[3456789]/);
        if (IS_IOS) {
          this.$.textarea.style.marginLeft = '-3px';
        }
      },

      /**
       * Returns true if `value` is valid. The validator provided in `validator`
       * will be used first, if it exists; otherwise, the `textarea`'s validity
       * is used.
       * @return {boolean} True if the value is valid.
       */
      validate: function() {
        // Use the nested input's native validity.
        var valid = this.$.textarea.validity.valid;

        // Only do extra checking if the browser thought this was valid.
        if (valid) {
          // Empty, required input is invalid
          if (this.required && this.value === '') {
            valid = false;
          } else if (this.hasValidator()) {
            valid = IronValidatableBehavior.validate.call(this, this.value);
          }
        }

        this.invalid = !valid;
        this.fire('iron-input-validate');
        return valid;
      },

      _bindValueChanged: function(bindValue) {
        this.value = bindValue;
      },

      _valueChanged: function(value) {
        var textarea = this.textarea;
        if (!textarea) {
          return;
        }

        // If the bindValue changed manually, then we need to also update
        // the underlying textarea's value. Otherwise this change was probably
        // generated from the _onInput handler, and the two values are already
        // the same.
        if (textarea.value !== value) {
          textarea.value = !(value || value === 0) ? '' : value;
        }

        this.bindValue = value;
        this.$.mirror.innerHTML = this._valueForMirror();

        // Manually notify because we don't want to notify until after setting
        // value.
        this.fire('bind-value-changed', {value: this.bindValue});
      },

      _onInput: function(event) {
        var eventPath = dom(event).path;
        this.value = eventPath ? eventPath[0].value : event.target.value;
      },

      _constrain: function(tokens) {
        var _tokens;
        tokens = tokens || [''];
        // Enforce the min and max heights for a multiline input to avoid
        // measurement
        if (this.maxRows > 0 && tokens.length > this.maxRows) {
          _tokens = tokens.slice(0, this.maxRows);
        } else {
          _tokens = tokens.slice(0);
        }
        while (this.rows > 0 && _tokens.length < this.rows) {
          _tokens.push('');
        }
        // Use &#160; instead &nbsp; of to allow this element to be used in XHTML.
        return _tokens.join('<br/>') + '&#160;';
      },

      _valueForMirror: function() {
        var input = this.textarea;
        if (!input) {
          return;
        }
        this.tokens = (input && input.value) ? input.value.replace(/&/gm, '&amp;')
                                                   .replace(/"/gm, '&quot;')
                                                   .replace(/'/gm, '&#39;')
                                                   .replace(/</gm, '&lt;')
                                                   .replace(/>/gm, '&gt;')
                                                   .split('\n') :
                                               [''];
        return this._constrain(this.tokens);
      },

      _updateCached: function() {
        this.$.mirror.innerHTML = this._constrain(this.tokens);
      }
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /*
    `<paper-textarea>` is a multi-line text field with Material Design styling.

        <paper-textarea label="Textarea label"></paper-textarea>

    See `Polymer.PaperInputBehavior` for more API docs.

    ### Validation

    Currently only `required` and `maxlength` validation is supported.

    ### Styling

    See `Polymer.PaperInputContainer` for a list of custom properties used to
    style this element.
    */
    Polymer({
      /** @override */
      _template: html`
    <style>
      :host {
        display: block;
      }

      :host([hidden]) {
        display: none !important;
      }

      label {
        pointer-events: none;
      }
    </style>

    <paper-input-container no-label-float$="[[noLabelFloat]]" always-float-label="[[_computeAlwaysFloatLabel(alwaysFloatLabel,placeholder)]]" auto-validate$="[[autoValidate]]" disabled$="[[disabled]]" invalid="[[invalid]]">

      <label hidden$="[[!label]]" aria-hidden="true" for$="[[_inputId]]" slot="label">[[label]]</label>

      <iron-autogrow-textarea class="paper-input-input" slot="input" id$="[[_inputId]]" aria-labelledby$="[[_ariaLabelledBy]]" aria-describedby$="[[_ariaDescribedBy]]" bind-value="{{value}}" invalid="{{invalid}}" validator$="[[validator]]" disabled$="[[disabled]]" autocomplete$="[[autocomplete]]" autofocus$="[[autofocus]]" inputmode$="[[inputmode]]" name$="[[name]]" placeholder$="[[placeholder]]" readonly$="[[readonly]]" required$="[[required]]" minlength$="[[minlength]]" maxlength$="[[maxlength]]" autocapitalize$="[[autocapitalize]]" rows$="[[rows]]" max-rows$="[[maxRows]]" on-change="_onChange"></iron-autogrow-textarea>

      <template is="dom-if" if="[[errorMessage]]">
        <paper-input-error aria-live="assertive" slot="add-on">[[errorMessage]]</paper-input-error>
      </template>

      <template is="dom-if" if="[[charCounter]]">
        <paper-input-char-counter slot="add-on"></paper-input-char-counter>
      </template>

    </paper-input-container>
`,

      is: 'paper-textarea',

      behaviors: [
        PaperInputBehavior,
        IronFormElementBehavior,
      ],

      properties: {
        _ariaLabelledBy: {
          observer: '_ariaLabelledByChanged',
          type: String,
        },

        _ariaDescribedBy: {
          observer: '_ariaDescribedByChanged',
          type: String,
        },

        value: {
          // Required for the correct TypeScript type-generation
          type: String,
        },

        /**
         * The initial number of rows.
         * @type {number}
         * @default 1
         */
        rows: {
          type: Number,
          value: 1,
        },

        /**
         * The maximum number of rows this element can grow to until it
         * scrolls. 0 means no maximum.
         * @type {number}
         * @default 0
         */
        maxRows: {
          type: Number,
          value: 0,
        },
      },

      /**
       * @return {number}
       */
      get selectionStart() {
        return this.$.input.textarea.selectionStart;
      },

      set selectionStart(start) {
        this.$.input.textarea.selectionStart = start;
      },

      /**
       * @return {number}
       */
      get selectionEnd() {
        return this.$.input.textarea.selectionEnd;
      },

      set selectionEnd(end) {
        this.$.input.textarea.selectionEnd = end;
      },

      _ariaLabelledByChanged: function(ariaLabelledBy) {
        this._focusableElement.setAttribute('aria-labelledby', ariaLabelledBy);
      },

      _ariaDescribedByChanged: function(ariaDescribedBy) {
        this._focusableElement.setAttribute('aria-describedby', ariaDescribedBy);
      },

      get _focusableElement() {
        return this.inputElement.textarea;
      }
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */
    const $_documentContainer$3 = document.createElement('template');
    $_documentContainer$3.setAttribute('style', 'display: none;');

    $_documentContainer$3.innerHTML = `<dom-module id="paper-item-shared-styles">
  <template>
    <style>
      :host, .paper-item {
        display: block;
        position: relative;
        min-height: var(--paper-item-min-height, 48px);
        padding: 0px 16px;
      }

      .paper-item {
        @apply --paper-font-subhead;
        border:none;
        outline: none;
        background: white;
        width: 100%;
        text-align: left;
      }

      :host([hidden]), .paper-item[hidden] {
        display: none !important;
      }

      :host(.iron-selected), .paper-item.iron-selected {
        font-weight: var(--paper-item-selected-weight, bold);

        @apply --paper-item-selected;
      }

      :host([disabled]), .paper-item[disabled] {
        color: var(--paper-item-disabled-color, var(--disabled-text-color));

        @apply --paper-item-disabled;
      }

      :host(:focus), .paper-item:focus {
        position: relative;
        outline: 0;

        @apply --paper-item-focused;
      }

      :host(:focus):before, .paper-item:focus:before {
        @apply --layout-fit;

        background: currentColor;
        content: '';
        opacity: var(--dark-divider-opacity);
        pointer-events: none;

        @apply --paper-item-focused-before;
      }
    </style>
  </template>
</dom-module>`;

    document.head.appendChild($_documentContainer$3.content);

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /*
    `PaperItemBehavior` is a convenience behavior shared by <paper-item> and
    <paper-icon-item> that manages the shared control states and attributes of
    the items.
    */
    /** @polymerBehavior PaperItemBehavior */
    const PaperItemBehaviorImpl = {
      hostAttributes: {role: 'option', tabindex: '0'}
    };

    /** @polymerBehavior */
    const PaperItemBehavior =
        [IronButtonState, IronControlState, PaperItemBehaviorImpl];

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
    Material design:
    [Lists](https://www.google.com/design/spec/components/lists.html)

    `<paper-item>` is an interactive list item. By default, it is a horizontal
    flexbox.

        <paper-item>Item</paper-item>

    Use this element with `<paper-item-body>` to make Material Design styled
    two-line and three-line items.

        <paper-item>
          <paper-item-body two-line>
            <div>Show your status</div>
            <div secondary>Your status is visible to everyone</div>
          </paper-item-body>
          <iron-icon icon="warning"></iron-icon>
        </paper-item>

    To use `paper-item` as a link, wrap it in an anchor tag. Since `paper-item` will
    already receive focus, you may want to prevent the anchor tag from receiving
    focus as well by setting its tabindex to -1.

        <a href="https://www.polymer-project.org/" tabindex="-1">
          <paper-item raised>Polymer Project</paper-item>
        </a>

    If you are concerned about performance and want to use `paper-item` in a
    `paper-listbox` with many items, you can just use a native `button` with the
    `paper-item` class applied (provided you have correctly included the shared
    styles):

        <style is="custom-style" include="paper-item-shared-styles"></style>

        <paper-listbox>
          <button class="paper-item" role="option">Inbox</button>
          <button class="paper-item" role="option">Starred</button>
          <button class="paper-item" role="option">Sent mail</button>
        </paper-listbox>

    ### Styling

    The following custom properties and mixins are available for styling:

    Custom property | Description | Default
    ----------------|-------------|----------
    `--paper-item-min-height` | Minimum height of the item | `48px`
    `--paper-item` | Mixin applied to the item | `{}`
    `--paper-item-selected-weight` | The font weight of a selected item | `bold`
    `--paper-item-selected` | Mixin applied to selected paper-items | `{}`
    `--paper-item-disabled-color` | The color for disabled paper-items | `--disabled-text-color`
    `--paper-item-disabled` | Mixin applied to disabled paper-items | `{}`
    `--paper-item-focused` | Mixin applied to focused paper-items | `{}`
    `--paper-item-focused-before` | Mixin applied to :before focused paper-items | `{}`

    ### Accessibility

    This element has `role="listitem"` by default. Depending on usage, it may be
    more appropriate to set `role="menuitem"`, `role="menuitemcheckbox"` or
    `role="menuitemradio"`.

        <paper-item role="menuitemcheckbox">
          <paper-item-body>
            Show your status
          </paper-item-body>
          <paper-checkbox></paper-checkbox>
        </paper-item>

    @group Paper Elements
    @element paper-item
    @demo demo/index.html
    */
    Polymer({
      _template: html`
    <style include="paper-item-shared-styles">
      :host {
        @apply --layout-horizontal;
        @apply --layout-center;
        @apply --paper-font-subhead;

        @apply --paper-item;
      }
    </style>
    <slot></slot>
`,

      is: 'paper-item',
      behaviors: [PaperItemBehavior]
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /*
    Use `<paper-item-body>` in a `<paper-item>` or `<paper-icon-item>` to make two-
    or three- line items. It is a flex item that is a vertical flexbox.

        <paper-item>
          <paper-item-body two-line>
            <div>Show your status</div>
            <div secondary>Your status is visible to everyone</div>
          </paper-item-body>
        </paper-item>

    The child elements with the `secondary` attribute is given secondary text
    styling.

    ### Styling

    The following custom properties and mixins are available for styling:

    Custom property | Description | Default
    ----------------|-------------|----------
    `--paper-item-body-two-line-min-height` | Minimum height of a two-line item | `72px`
    `--paper-item-body-three-line-min-height` | Minimum height of a three-line item | `88px`
    `--paper-item-body-secondary-color` | Foreground color for the `secondary` area | `--secondary-text-color`
    `--paper-item-body-secondary` | Mixin applied to the `secondary` area | `{}`

    */
    Polymer({
      _template: html`
    <style>
      :host {
        overflow: hidden; /* needed for text-overflow: ellipsis to work on ff */
        @apply --layout-vertical;
        @apply --layout-center-justified;
        @apply --layout-flex;
      }

      :host([two-line]) {
        min-height: var(--paper-item-body-two-line-min-height, 72px);
      }

      :host([three-line]) {
        min-height: var(--paper-item-body-three-line-min-height, 88px);
      }

      :host > ::slotted(*) {
        overflow: hidden;
        text-overflow: ellipsis;
        white-space: nowrap;
      }

      :host > ::slotted([secondary]) {
        @apply --paper-font-body1;

        color: var(--paper-item-body-secondary-color, var(--secondary-text-color));

        @apply --paper-item-body-secondary;
      }
    </style>

    <slot></slot>
`,

      is: 'paper-item-body'
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /*
    `<paper-icon-item>` is a convenience element to make an item with icon. It is an
    interactive list item with a fixed-width icon area, according to Material
    Design. This is useful if the icons are of varying widths, but you want the item
    bodies to line up. Use this like a `<paper-item>`. The child node with the slot
    name `item-icon` is placed in the icon area.

        <paper-icon-item>
          <iron-icon icon="favorite" slot="item-icon"></iron-icon>
          Favorite
        </paper-icon-item>
        <paper-icon-item>
          <div class="avatar" slot="item-icon"></div>
          Avatar
        </paper-icon-item>

    ### Styling

    The following custom properties and mixins are available for styling:

    Custom property | Description | Default
    ----------------|-------------|----------
    `--paper-item-icon-width` | Width of the icon area | `56px`
    `--paper-item-icon` | Mixin applied to the icon area | `{}`
    `--paper-icon-item` | Mixin applied to the item | `{}`
    `--paper-item-selected-weight` | The font weight of a selected item | `bold`
    `--paper-item-selected` | Mixin applied to selected paper-items | `{}`
    `--paper-item-disabled-color` | The color for disabled paper-items | `--disabled-text-color`
    `--paper-item-disabled` | Mixin applied to disabled paper-items | `{}`
    `--paper-item-focused` | Mixin applied to focused paper-items | `{}`
    `--paper-item-focused-before` | Mixin applied to :before focused paper-items | `{}`

    */
    Polymer({
      _template: html`
    <style include="paper-item-shared-styles"></style>
    <style>
      :host {
        @apply --layout-horizontal;
        @apply --layout-center;
        @apply --paper-font-subhead;

        @apply --paper-item;
        @apply --paper-icon-item;
      }

      .content-icon {
        @apply --layout-horizontal;
        @apply --layout-center;

        width: var(--paper-item-icon-width, 56px);
        @apply --paper-item-icon;
      }
    </style>

    <div id="contentIcon" class="content-icon">
      <slot name="item-icon"></slot>
    </div>
    <slot></slot>
`,

      is: 'paper-icon-item',
      behaviors: [PaperItemBehavior]
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
     * @polymerBehavior IronMultiSelectableBehavior
     */
    const IronMultiSelectableBehaviorImpl = {
      properties: {

        /**
         * If true, multiple selections are allowed.
         */
        multi: {type: Boolean, value: false, observer: 'multiChanged'},

        /**
         * Gets or sets the selected elements. This is used instead of `selected`
         * when `multi` is true.
         */
        selectedValues: {
          type: Array,
          notify: true,
          value: function() {
            return [];
          }
        },

        /**
         * Returns an array of currently selected items.
         */
        selectedItems: {
          type: Array,
          readOnly: true,
          notify: true,
          value: function() {
            return [];
          }
        },

      },

      observers: ['_updateSelected(selectedValues.splices)'],

      /**
       * Selects the given value. If the `multi` property is true, then the selected
       * state of the `value` will be toggled; otherwise the `value` will be
       * selected.
       *
       * @method select
       * @param {string|number} value the value to select.
       */
      select: function(value) {
        if (this.multi) {
          this._toggleSelected(value);
        } else {
          this.selected = value;
        }
      },

      multiChanged: function(multi) {
        this._selection.multi = multi;
        this._updateSelected();
      },

      // UNUSED, FOR API COMPATIBILITY
      get _shouldUpdateSelection() {
        return this.selected != null ||
            (this.selectedValues != null && this.selectedValues.length);
      },

      _updateAttrForSelected: function() {
        if (!this.multi) {
          IronSelectableBehavior._updateAttrForSelected.apply(this);
        } else if (this.selectedItems && this.selectedItems.length > 0) {
          this.selectedValues =
              this.selectedItems
                  .map(
                      function(selectedItem) {
                        return this._indexToValue(this.indexOf(selectedItem));
                      },
                      this)
                  .filter(function(unfilteredValue) {
                    return unfilteredValue != null;
                  }, this);
        }
      },

      _updateSelected: function() {
        if (this.multi) {
          this._selectMulti(this.selectedValues);
        } else {
          this._selectSelected(this.selected);
        }
      },

      _selectMulti: function(values) {
        values = values || [];

        var selectedItems =
            (this._valuesToItems(values) || []).filter(function(item) {
              return item !== null && item !== undefined;
            });

        // clear all but the current selected items
        this._selection.clear(selectedItems);

        // select only those not selected yet
        for (var i = 0; i < selectedItems.length; i++) {
          this._selection.setItemSelected(selectedItems[i], true);
        }

        // Check for items, since this array is populated only when attached
        if (this.fallbackSelection && !this._selection.get().length) {
          var fallback = this._valueToItem(this.fallbackSelection);
          if (fallback) {
            this.select(this.fallbackSelection);
          }
        }
      },

      _selectionChange: function() {
        var s = this._selection.get();
        if (this.multi) {
          this._setSelectedItems(s);
          this._setSelectedItem(s.length ? s[0] : null);
        } else {
          if (s !== null && s !== undefined) {
            this._setSelectedItems([s]);
            this._setSelectedItem(s);
          } else {
            this._setSelectedItems([]);
            this._setSelectedItem(null);
          }
        }
      },

      _toggleSelected: function(value) {
        var i = this.selectedValues.indexOf(value);
        var unselected = i < 0;
        if (unselected) {
          this.push('selectedValues', value);
        } else {
          this.splice('selectedValues', i, 1);
        }
      },

      _valuesToItems: function(values) {
        return (values == null) ? null : values.map(function(value) {
          return this._valueToItem(value);
        }, this);
      }
    };

    /** @polymerBehavior */
    const IronMultiSelectableBehavior =
        [IronSelectableBehavior, IronMultiSelectableBehaviorImpl];

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
     * `IronMenuBehavior` implements accessible menu behavior.
     *
     * @demo demo/index.html
     * @polymerBehavior IronMenuBehavior
     */
    const IronMenuBehaviorImpl = {

      properties: {

        /**
         * Returns the currently focused item.
         * @type {?Object}
         */
        focusedItem:
            {observer: '_focusedItemChanged', readOnly: true, type: Object},

        /**
         * The attribute to use on menu items to look up the item title. Typing the
         * first letter of an item when the menu is open focuses that item. If
         * unset, `textContent` will be used.
         */
        attrForItemTitle: {type: String},

        /**
         * @type {boolean}
         */
        disabled: {
          type: Boolean,
          value: false,
          observer: '_disabledChanged',
        },
      },

      /**
       * The list of keys has been taken from
       * https://developer.mozilla.org/en-US/docs/Web/API/KeyboardEvent/getModifierState
       * @private
       */
      _MODIFIER_KEYS: [
        'Alt',
        'AltGraph',
        'CapsLock',
        'Control',
        'Fn',
        'FnLock',
        'Hyper',
        'Meta',
        'NumLock',
        'OS',
        'ScrollLock',
        'Shift',
        'Super',
        'Symbol',
        'SymbolLock'
      ],

      /** @private */
      _SEARCH_RESET_TIMEOUT_MS: 1000,

      /** @private */
      _previousTabIndex: 0,

      hostAttributes: {
        'role': 'menu',
      },

      observers: ['_updateMultiselectable(multi)'],

      listeners: {
        'focus': '_onFocus',
        'keydown': '_onKeydown',
        'iron-items-changed': '_onIronItemsChanged'
      },

      /**
       * @type {!Object}
       */
      keyBindings: {
        'up': '_onUpKey',
        'down': '_onDownKey',
        'esc': '_onEscKey',
        'shift+tab:keydown': '_onShiftTabDown'
      },

      attached: function() {
        this._resetTabindices();
      },

      /**
       * Selects the given value. If the `multi` property is true, then the selected
       * state of the `value` will be toggled; otherwise the `value` will be
       * selected.
       *
       * @param {string|number} value the value to select.
       */
      select: function(value) {
        // Cancel automatically focusing a default item if the menu received focus
        // through a user action selecting a particular item.
        if (this._defaultFocusAsync) {
          this.cancelAsync(this._defaultFocusAsync);
          this._defaultFocusAsync = null;
        }
        var item = this._valueToItem(value);
        if (item && item.hasAttribute('disabled'))
          return;
        this._setFocusedItem(item);
        IronMultiSelectableBehaviorImpl.select.apply(this, arguments);
      },

      /**
       * Resets all tabindex attributes to the appropriate value based on the
       * current selection state. The appropriate value is `0` (focusable) for
       * the default selected item, and `-1` (not keyboard focusable) for all
       * other items. Also sets the correct initial values for aria-selected
       * attribute, true for default selected item and false for others.
       */
      _resetTabindices: function() {
        var firstSelectedItem = this.multi ?
            (this.selectedItems && this.selectedItems[0]) :
            this.selectedItem;

        this.items.forEach(function(item) {
          item.setAttribute('tabindex', item === firstSelectedItem ? '0' : '-1');
          item.setAttribute('aria-selected', this._selection.isSelected(item));
        }, this);
      },

      /**
       * Sets appropriate ARIA based on whether or not the menu is meant to be
       * multi-selectable.
       *
       * @param {boolean} multi True if the menu should be multi-selectable.
       */
      _updateMultiselectable: function(multi) {
        if (multi) {
          this.setAttribute('aria-multiselectable', 'true');
        } else {
          this.removeAttribute('aria-multiselectable');
        }
      },

      /**
       * Given a KeyboardEvent, this method will focus the appropriate item in the
       * menu (if there is a relevant item, and it is possible to focus it).
       *
       * @param {KeyboardEvent} event A KeyboardEvent.
       */
      _focusWithKeyboardEvent: function(event) {
        // Make sure that the key pressed is not a modifier key.
        // getModifierState is not being used, as it is not available in Safari
        // earlier than 10.0.2 (https://trac.webkit.org/changeset/206725/webkit)
        if (this._MODIFIER_KEYS.indexOf(event.key) !== -1)
          return;

        this.cancelDebouncer('_clearSearchText');

        var searchText = this._searchText || '';
        var key = event.key && event.key.length == 1 ?
            event.key :
            String.fromCharCode(event.keyCode);
        searchText += key.toLocaleLowerCase();

        var searchLength = searchText.length;

        for (var i = 0, item; item = this.items[i]; i++) {
          if (item.hasAttribute('disabled')) {
            continue;
          }

          var attr = this.attrForItemTitle || 'textContent';
          var title = (item[attr] || item.getAttribute(attr) || '').trim();

          if (title.length < searchLength) {
            continue;
          }

          if (title.slice(0, searchLength).toLocaleLowerCase() == searchText) {
            this._setFocusedItem(item);
            break;
          }
        }

        this._searchText = searchText;
        this.debounce(
            '_clearSearchText',
            this._clearSearchText,
            this._SEARCH_RESET_TIMEOUT_MS);
      },

      _clearSearchText: function() {
        this._searchText = '';
      },

      /**
       * Focuses the previous item (relative to the currently focused item) in the
       * menu, disabled items will be skipped.
       * Loop until length + 1 to handle case of single item in menu.
       */
      _focusPrevious: function() {
        var length = this.items.length;
        var curFocusIndex = Number(this.indexOf(this.focusedItem));

        for (var i = 1; i < length + 1; i++) {
          var item = this.items[(curFocusIndex - i + length) % length];
          if (!item.hasAttribute('disabled')) {
            var owner = dom(item).getOwnerRoot() || document;
            this._setFocusedItem(item);

            // Focus might not have worked, if the element was hidden or not
            // focusable. In that case, try again.
            if (dom(owner).activeElement == item) {
              return;
            }
          }
        }
      },

      /**
       * Focuses the next item (relative to the currently focused item) in the
       * menu, disabled items will be skipped.
       * Loop until length + 1 to handle case of single item in menu.
       */
      _focusNext: function() {
        var length = this.items.length;
        var curFocusIndex = Number(this.indexOf(this.focusedItem));

        for (var i = 1; i < length + 1; i++) {
          var item = this.items[(curFocusIndex + i) % length];
          if (!item.hasAttribute('disabled')) {
            var owner = dom(item).getOwnerRoot() || document;
            this._setFocusedItem(item);

            // Focus might not have worked, if the element was hidden or not
            // focusable. In that case, try again.
            if (dom(owner).activeElement == item) {
              return;
            }
          }
        }
      },

      /**
       * Mutates items in the menu based on provided selection details, so that
       * all items correctly reflect selection state.
       *
       * @param {Element} item An item in the menu.
       * @param {boolean} isSelected True if the item should be shown in a
       * selected state, otherwise false.
       */
      _applySelection: function(item, isSelected) {
        if (isSelected) {
          item.setAttribute('aria-selected', 'true');
        } else {
          item.setAttribute('aria-selected', 'false');
        }
        IronSelectableBehavior._applySelection.apply(this, arguments);
      },

      /**
       * Discretely updates tabindex values among menu items as the focused item
       * changes.
       *
       * @param {Element} focusedItem The element that is currently focused.
       * @param {?Element} old The last element that was considered focused, if
       * applicable.
       */
      _focusedItemChanged: function(focusedItem, old) {
        old && old.setAttribute('tabindex', '-1');
        if (focusedItem && !focusedItem.hasAttribute('disabled') &&
            !this.disabled) {
          focusedItem.setAttribute('tabindex', '0');
          focusedItem.focus();
        }
      },

      /**
       * A handler that responds to mutation changes related to the list of items
       * in the menu.
       *
       * @param {CustomEvent} event An event containing mutation records as its
       * detail.
       */
      _onIronItemsChanged: function(event) {
        if (event.detail.addedNodes.length) {
          this._resetTabindices();
        }
      },

      /**
       * Handler that is called when a shift+tab keypress is detected by the menu.
       *
       * @param {CustomEvent} event A key combination event.
       */
      _onShiftTabDown: function(event) {
        var oldTabIndex = this.getAttribute('tabindex');

        IronMenuBehaviorImpl._shiftTabPressed = true;

        this._setFocusedItem(null);

        this.setAttribute('tabindex', '-1');

        this.async(function() {
          this.setAttribute('tabindex', oldTabIndex);
          IronMenuBehaviorImpl._shiftTabPressed = false;
          // NOTE(cdata): polymer/polymer#1305
        }, 1);
      },

      /**
       * Handler that is called when the menu receives focus.
       *
       * @param {FocusEvent} event A focus event.
       */
      _onFocus: function(event) {
        if (IronMenuBehaviorImpl._shiftTabPressed) {
          // do not focus the menu itself
          return;
        }

        // Do not focus the selected tab if the deepest target is part of the
        // menu element's local DOM and is focusable.
        var rootTarget =
            /** @type {?HTMLElement} */ (dom(event).rootTarget);
        if (rootTarget !== this && typeof rootTarget.tabIndex !== 'undefined' &&
            !this.isLightDescendant(rootTarget)) {
          return;
        }

        // clear the cached focus item
        this._defaultFocusAsync = this.async(function() {
          // focus the selected item when the menu receives focus, or the first item
          // if no item is selected
          var firstSelectedItem = this.multi ?
              (this.selectedItems && this.selectedItems[0]) :
              this.selectedItem;

          this._setFocusedItem(null);

          if (firstSelectedItem) {
            this._setFocusedItem(firstSelectedItem);
          } else if (this.items[0]) {
            // We find the first none-disabled item (if one exists)
            this._focusNext();
          }
        });
      },

      /**
       * Handler that is called when the up key is pressed.
       *
       * @param {CustomEvent} event A key combination event.
       */
      _onUpKey: function(event) {
        // up and down arrows moves the focus
        this._focusPrevious();
        event.detail.keyboardEvent.preventDefault();
      },

      /**
       * Handler that is called when the down key is pressed.
       *
       * @param {CustomEvent} event A key combination event.
       */
      _onDownKey: function(event) {
        this._focusNext();
        event.detail.keyboardEvent.preventDefault();
      },

      /**
       * Handler that is called when the esc key is pressed.
       *
       * @param {CustomEvent} event A key combination event.
       */
      _onEscKey: function(event) {
        var focusedItem = this.focusedItem;
        if (focusedItem) {
          focusedItem.blur();
        }
      },

      /**
       * Handler that is called when a keydown event is detected.
       *
       * @param {KeyboardEvent} event A keyboard event.
       */
      _onKeydown: function(event) {
        if (!this.keyboardEventMatchesKeys(event, 'up down esc')) {
          // all other keys focus the menu item starting with that character
          this._focusWithKeyboardEvent(event);
        }
        event.stopPropagation();
      },

      // override _activateHandler
      _activateHandler: function(event) {
        IronSelectableBehavior._activateHandler.call(this, event);
        event.stopPropagation();
      },

      /**
       * Updates this element's tab index when it's enabled/disabled.
       * @param {boolean} disabled
       */
      _disabledChanged: function(disabled) {
        if (disabled) {
          this._previousTabIndex =
              this.hasAttribute('tabindex') ? this.tabIndex : 0;
          this.removeAttribute(
              'tabindex');  // No tabindex means not tab-able or select-able.
        } else if (!this.hasAttribute('tabindex')) {
          this.setAttribute('tabindex', this._previousTabIndex);
        }
      }
    };

    IronMenuBehaviorImpl._shiftTabPressed = false;

    /** @polymerBehavior */
    const IronMenuBehavior =
        [IronMultiSelectableBehavior, IronA11yKeysBehavior, IronMenuBehaviorImpl];

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
    Material design:
    [Menus](https://www.google.com/design/spec/components/menus.html)

    `<paper-listbox>` implements an accessible listbox control with Material Design
    styling. The focused item is highlighted, and the selected item has bolded text.

        <paper-listbox>
          <paper-item>Item 1</paper-item>
          <paper-item>Item 2</paper-item>
        </paper-listbox>

    An initial selection can be specified with the `selected` attribute.

        <paper-listbox selected="0">
          <paper-item>Item 1</paper-item>
          <paper-item>Item 2</paper-item>
        </paper-listbox>

    Make a multi-select listbox with the `multi` attribute. Items in a multi-select
    listbox can be deselected, and multiple item can be selected.

        <paper-listbox multi>
          <paper-item>Item 1</paper-item>
          <paper-item>Item 2</paper-item>
        </paper-listbox>

    ### Styling

    The following custom properties and mixins are available for styling:

    Custom property | Description | Default
    ----------------|-------------|----------
    `--paper-listbox-background-color`   | Menu background color |
    `--primary-background-color`
    `--paper-listbox-color`              | Menu foreground color |
    `--primary-text-color`
    `--paper-listbox`                    | Mixin applied to the listbox | `{}`

    ### Accessibility

    `<paper-listbox>` has `role="listbox"` by default. A multi-select listbox will
    also have `aria-multiselectable` set. It implements key bindings to navigate
    through the listbox with the up and down arrow keys, esc to exit the listbox,
    and enter to activate a listbox item. Typing the first letter of a listbox item
    will also focus it.

    @group Paper Elements
    @element paper-listbox
    @demo demo/index.html
    */
    Polymer({
      _template: html`
    <style>
      :host {
        display: block;
        padding: 8px 0;

        background: var(--paper-listbox-background-color, var(--primary-background-color));
        color: var(--paper-listbox-color, var(--primary-text-color));

        @apply --paper-listbox;
      }
    </style>

    <slot></slot>
`,

      is: 'paper-listbox',
      behaviors: [IronMenuBehavior],

      /** @private */
      hostAttributes: {role: 'listbox'}
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */
    const template$c = html`
<dom-module id="paper-material-shared-styles">
  <template>
    <style>
      :host {
        display: block;
        position: relative;
      }

      :host([elevation="1"]) {
        @apply --shadow-elevation-2dp;
      }

      :host([elevation="2"]) {
        @apply --shadow-elevation-4dp;
      }

      :host([elevation="3"]) {
        @apply --shadow-elevation-6dp;
      }

      :host([elevation="4"]) {
        @apply --shadow-elevation-8dp;
      }

      :host([elevation="5"]) {
        @apply --shadow-elevation-16dp;
      }
    </style>
  </template>
</dom-module>
`;
    template$c.setAttribute('style', 'display: none;');
    document.body.appendChild(template$c.content);

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
    Material design:
    [Cards](https://www.google.com/design/spec/components/cards.html)

    `paper-material` is a container that renders two shadows on top of each other to
    create the effect of a lifted piece of paper.

    Example:

        <paper-material elevation="1">
          ... content ...
        </paper-material>

    @group Paper Elements
    @demo demo/index.html
    */
    Polymer({
      _template: html`
    <style include="paper-material-shared-styles"></style>
    <style>
      :host([animated]) {
        @apply --shadow-transition;
      }
      :host {
        @apply --paper-material;
      }
    </style>

    <slot></slot>
`,

      is: 'paper-material',

      properties: {
        /**
         * The z-depth of this element, from 0-5. Setting to 0 will remove the
         * shadow, and each increasing number greater than 0 will be "deeper"
         * than the last.
         *
         * @attribute elevation
         * @type number
         * @default 1
         */
        elevation: {type: Number, reflectToAttribute: true, value: 1},

        /**
         * Set this to true to animate the shadow when setting a new
         * `elevation` value.
         *
         * @attribute animated
         * @type boolean
         * @default false
         */
        animated: {type: Boolean, reflectToAttribute: true, value: false}
      }
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
     * `iron-range-behavior` provides the behavior for something with a minimum to
     * maximum range.
     *
     * @demo demo/index.html
     * @polymerBehavior
     */
    const IronRangeBehavior = {

      properties: {

        /**
         * The number that represents the current value.
         */
        value: {type: Number, value: 0, notify: true, reflectToAttribute: true},

        /**
         * The number that indicates the minimum value of the range.
         */
        min: {type: Number, value: 0, notify: true},

        /**
         * The number that indicates the maximum value of the range.
         */
        max: {type: Number, value: 100, notify: true},

        /**
         * Specifies the value granularity of the range's value.
         */
        step: {type: Number, value: 1, notify: true},

        /**
         * Returns the ratio of the value.
         */
        ratio: {type: Number, value: 0, readOnly: true, notify: true},
      },

      observers: ['_update(value, min, max, step)'],

      _calcRatio: function(value) {
        return (this._clampValue(value) - this.min) / (this.max - this.min);
      },

      _clampValue: function(value) {
        return Math.min(this.max, Math.max(this.min, this._calcStep(value)));
      },

      _calcStep: function(value) {
        // polymer/issues/2493
        value = parseFloat(value);

        if (!this.step) {
          return value;
        }

        var numSteps = Math.round((value - this.min) / this.step);
        if (this.step < 1) {
          /**
           * For small values of this.step, if we calculate the step using
           * `Math.round(value / step) * step` we may hit a precision point issue
           * eg. 0.1 * 0.2 =  0.020000000000000004
           * http://docs.oracle.com/cd/E19957-01/806-3568/ncg_goldberg.html
           *
           * as a work around we can divide by the reciprocal of `step`
           */
          return numSteps / (1 / this.step) + this.min;
        } else {
          return numSteps * this.step + this.min;
        }
      },

      _validateValue: function() {
        var v = this._clampValue(this.value);
        this.value = this.oldValue = isNaN(v) ? this.oldValue : v;
        return this.value !== v;
      },

      _update: function() {
        this._validateValue();
        this._setRatio(this._calcRatio(this.value) * 100);
      }

    };

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
    Material design: [Progress &
    activity](https://www.google.com/design/spec/components/progress-activity.html)

    The progress bars are for situations where the percentage completed can be
    determined. They give users a quick sense of how much longer an operation
    will take.

    Example:

        <paper-progress value="10"></paper-progress>

    There is also a secondary progress which is useful for displaying intermediate
    progress, such as the buffer level during a streaming playback progress bar.

    Example:

        <paper-progress value="10" secondary-progress="30"></paper-progress>

    ### Styling progress bar:

    To change the active progress bar color:

        paper-progress {
           --paper-progress-active-color: #e91e63;
        }

    To change the secondary progress bar color:

        paper-progress {
          --paper-progress-secondary-color: #f8bbd0;
        }

    To change the progress bar background color:

        paper-progress {
          --paper-progress-container-color: #64ffda;
        }

    Add the class `transiting` to a paper-progress to animate the progress bar when
    the value changed. You can also customize the transition:

        paper-progress {
          --paper-progress-transition-duration: 0.08s;
          --paper-progress-transition-timing-function: ease;
          --paper-progress-transition-delay: 0s;
        }

    To change the duration of the indeterminate cycle:

        paper-progress {
          --paper-progress-indeterminate-cycle-duration: 2s;
        }

    The following mixins are available for styling:

    Custom property | Description | Default
    ----------------|-------------|---------
    `--paper-progress-container` | Mixin applied to container | `{}`
    `--paper-progress-transition-duration` | Duration of the transition | `0.08s`
    `--paper-progress-transition-timing-function` | The timing function for the transition | `ease`
    `--paper-progress-transition-delay` | delay for the transition | `0s`
    `--paper-progress-container-color` | Color of the container | `--google-grey-300`
    `--paper-progress-active-color` | The color of the active bar | `--google-green-500`
    `--paper-progress-secondary-color` | The color of the secondary bar | `--google-green-100`
    `--paper-progress-disabled-active-color` | The color of the active bar if disabled | `--google-grey-500`
    `--paper-progress-disabled-secondary-color` | The color of the secondary bar if disabled  | `--google-grey-300`
    `--paper-progress-height` | The height of the progress bar | `4px`
    `--paper-progress-indeterminate-cycle-duration` | Duration of an indeterminate cycle | `2s`

    @group Paper Elements
    @element paper-progress
    @demo demo/index.html
    */
    Polymer({
      _template: html`
    <style>
      :host {
        display: block;
        width: 200px;
        position: relative;
        overflow: hidden;
      }

      :host([hidden]), [hidden] {
        display: none !important;
      }

      #progressContainer {
        @apply --paper-progress-container;
        position: relative;
      }

      #progressContainer,
      /* the stripe for the indeterminate animation*/
      .indeterminate::after {
        height: var(--paper-progress-height, 4px);
      }

      #primaryProgress,
      #secondaryProgress,
      .indeterminate::after {
        @apply --layout-fit;
      }

      #progressContainer,
      .indeterminate::after {
        background: var(--paper-progress-container-color, var(--google-grey-300));
      }

      :host(.transiting) #primaryProgress,
      :host(.transiting) #secondaryProgress {
        -webkit-transition-property: -webkit-transform;
        transition-property: transform;

        /* Duration */
        -webkit-transition-duration: var(--paper-progress-transition-duration, 0.08s);
        transition-duration: var(--paper-progress-transition-duration, 0.08s);

        /* Timing function */
        -webkit-transition-timing-function: var(--paper-progress-transition-timing-function, ease);
        transition-timing-function: var(--paper-progress-transition-timing-function, ease);

        /* Delay */
        -webkit-transition-delay: var(--paper-progress-transition-delay, 0s);
        transition-delay: var(--paper-progress-transition-delay, 0s);
      }

      #primaryProgress,
      #secondaryProgress {
        @apply --layout-fit;
        -webkit-transform-origin: left center;
        transform-origin: left center;
        -webkit-transform: scaleX(0);
        transform: scaleX(0);
        will-change: transform;
      }

      #primaryProgress {
        background: var(--paper-progress-active-color, var(--google-green-500));
      }

      #secondaryProgress {
        background: var(--paper-progress-secondary-color, var(--google-green-100));
      }

      :host([disabled]) #primaryProgress {
        background: var(--paper-progress-disabled-active-color, var(--google-grey-500));
      }

      :host([disabled]) #secondaryProgress {
        background: var(--paper-progress-disabled-secondary-color, var(--google-grey-300));
      }

      :host(:not([disabled])) #primaryProgress.indeterminate {
        -webkit-transform-origin: right center;
        transform-origin: right center;
        -webkit-animation: indeterminate-bar var(--paper-progress-indeterminate-cycle-duration, 2s) linear infinite;
        animation: indeterminate-bar var(--paper-progress-indeterminate-cycle-duration, 2s) linear infinite;
      }

      :host(:not([disabled])) #primaryProgress.indeterminate::after {
        content: "";
        -webkit-transform-origin: center center;
        transform-origin: center center;

        -webkit-animation: indeterminate-splitter var(--paper-progress-indeterminate-cycle-duration, 2s) linear infinite;
        animation: indeterminate-splitter var(--paper-progress-indeterminate-cycle-duration, 2s) linear infinite;
      }

      @-webkit-keyframes indeterminate-bar {
        0% {
          -webkit-transform: scaleX(1) translateX(-100%);
        }
        50% {
          -webkit-transform: scaleX(1) translateX(0%);
        }
        75% {
          -webkit-transform: scaleX(1) translateX(0%);
          -webkit-animation-timing-function: cubic-bezier(.28,.62,.37,.91);
        }
        100% {
          -webkit-transform: scaleX(0) translateX(0%);
        }
      }

      @-webkit-keyframes indeterminate-splitter {
        0% {
          -webkit-transform: scaleX(.75) translateX(-125%);
        }
        30% {
          -webkit-transform: scaleX(.75) translateX(-125%);
          -webkit-animation-timing-function: cubic-bezier(.42,0,.6,.8);
        }
        90% {
          -webkit-transform: scaleX(.75) translateX(125%);
        }
        100% {
          -webkit-transform: scaleX(.75) translateX(125%);
        }
      }

      @keyframes indeterminate-bar {
        0% {
          transform: scaleX(1) translateX(-100%);
        }
        50% {
          transform: scaleX(1) translateX(0%);
        }
        75% {
          transform: scaleX(1) translateX(0%);
          animation-timing-function: cubic-bezier(.28,.62,.37,.91);
        }
        100% {
          transform: scaleX(0) translateX(0%);
        }
      }

      @keyframes indeterminate-splitter {
        0% {
          transform: scaleX(.75) translateX(-125%);
        }
        30% {
          transform: scaleX(.75) translateX(-125%);
          animation-timing-function: cubic-bezier(.42,0,.6,.8);
        }
        90% {
          transform: scaleX(.75) translateX(125%);
        }
        100% {
          transform: scaleX(.75) translateX(125%);
        }
      }
    </style>

    <div id="progressContainer">
      <div id="secondaryProgress" hidden\$="[[_hideSecondaryProgress(secondaryRatio)]]"></div>
      <div id="primaryProgress"></div>
    </div>
`,

      is: 'paper-progress',
      behaviors: [IronRangeBehavior],

      properties: {
        /**
         * The number that represents the current secondary progress.
         */
        secondaryProgress: {type: Number, value: 0},

        /**
         * The secondary ratio
         */
        secondaryRatio: {type: Number, value: 0, readOnly: true},

        /**
         * Use an indeterminate progress indicator.
         */
        indeterminate:
            {type: Boolean, value: false, observer: '_toggleIndeterminate'},

        /**
         * True if the progress is disabled.
         */
        disabled: {
          type: Boolean,
          value: false,
          reflectToAttribute: true,
          observer: '_disabledChanged'
        }
      },

      observers:
          ['_progressChanged(secondaryProgress, value, min, max, indeterminate)'],

      hostAttributes: {role: 'progressbar'},

      _toggleIndeterminate: function(indeterminate) {
        // If we use attribute/class binding, the animation sometimes doesn't
        // translate properly on Safari 7.1. So instead, we toggle the class here in
        // the update method.
        this.toggleClass('indeterminate', indeterminate, this.$.primaryProgress);
      },

      _transformProgress: function(progress, ratio) {
        var transform = 'scaleX(' + (ratio / 100) + ')';
        progress.style.transform = progress.style.webkitTransform = transform;
      },

      _mainRatioChanged: function(ratio) {
        this._transformProgress(this.$.primaryProgress, ratio);
      },

      _progressChanged: function(
          secondaryProgress, value, min, max, indeterminate) {
        secondaryProgress = this._clampValue(secondaryProgress);
        value = this._clampValue(value);

        var secondaryRatio = this._calcRatio(secondaryProgress) * 100;
        var mainRatio = this._calcRatio(value) * 100;

        this._setSecondaryRatio(secondaryRatio);
        this._transformProgress(this.$.secondaryProgress, secondaryRatio);
        this._transformProgress(this.$.primaryProgress, mainRatio);

        this.secondaryProgress = secondaryProgress;

        if (indeterminate) {
          this.removeAttribute('aria-valuenow');
        } else {
          this.setAttribute('aria-valuenow', value);
        }
        this.setAttribute('aria-valuemin', min);
        this.setAttribute('aria-valuemax', max);
      },

      _disabledChanged: function(disabled) {
        this.setAttribute('aria-disabled', disabled ? 'true' : 'false');
      },

      _hideSecondaryProgress: function(secondaryRatio) {
        return secondaryRatio === 0;
      }
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    const template$d = html`
<style>
  :host {
    display: inline-block;
    line-height: 0;
    white-space: nowrap;
    cursor: pointer;
    @apply --paper-font-common-base;
    --calculated-paper-radio-button-size: var(--paper-radio-button-size, 16px);
    /* -1px is a sentinel for the default and is replace in \`attached\`. */
    --calculated-paper-radio-button-ink-size: var(--paper-radio-button-ink-size, -1px);
  }

  :host(:focus) {
    outline: none;
  }

  #radioContainer {
    @apply --layout-inline;
    @apply --layout-center-center;
    position: relative;
    width: var(--calculated-paper-radio-button-size);
    height: var(--calculated-paper-radio-button-size);
    vertical-align: middle;

    @apply --paper-radio-button-radio-container;
  }

  #ink {
    position: absolute;
    top: 50%;
    left: 50%;
    right: auto;
    width: var(--calculated-paper-radio-button-ink-size);
    height: var(--calculated-paper-radio-button-ink-size);
    color: var(--paper-radio-button-unchecked-ink-color, var(--primary-text-color));
    opacity: 0.6;
    pointer-events: none;
    -webkit-transform: translate(-50%, -50%);
    transform: translate(-50%, -50%);
  }

  #ink[checked] {
    color: var(--paper-radio-button-checked-ink-color, var(--primary-color));
  }

  #offRadio, #onRadio {
    position: absolute;
    box-sizing: border-box;
    top: 0;
    left: 0;
    width: 100%;
    height: 100%;
    border-radius: 50%;
  }

  #offRadio {
    border: 2px solid var(--paper-radio-button-unchecked-color, var(--primary-text-color));
    background-color: var(--paper-radio-button-unchecked-background-color, transparent);
    transition: border-color 0.28s;
  }

  #onRadio {
    background-color: var(--paper-radio-button-checked-color, var(--primary-color));
    -webkit-transform: scale(0);
    transform: scale(0);
    transition: -webkit-transform ease 0.28s;
    transition: transform ease 0.28s;
    will-change: transform;
  }

  :host([checked]) #offRadio {
    border-color: var(--paper-radio-button-checked-color, var(--primary-color));
  }

  :host([checked]) #onRadio {
    -webkit-transform: scale(0.5);
    transform: scale(0.5);
  }

  #radioLabel {
    line-height: normal;
    position: relative;
    display: inline-block;
    vertical-align: middle;
    margin-left: var(--paper-radio-button-label-spacing, 10px);
    white-space: normal;
    color: var(--paper-radio-button-label-color, var(--primary-text-color));

    @apply --paper-radio-button-label;
  }

  :host([checked]) #radioLabel {
    @apply --paper-radio-button-label-checked;
  }

  #radioLabel:dir(rtl) {
    margin-left: 0;
    margin-right: var(--paper-radio-button-label-spacing, 10px);
  }

  #radioLabel[hidden] {
    display: none;
  }

  /* disabled state */

  :host([disabled]) #offRadio {
    border-color: var(--paper-radio-button-unchecked-color, var(--primary-text-color));
    opacity: 0.5;
  }

  :host([disabled][checked]) #onRadio {
    background-color: var(--paper-radio-button-unchecked-color, var(--primary-text-color));
    opacity: 0.5;
  }

  :host([disabled]) #radioLabel {
    /* slightly darker than the button, so that it's readable */
    opacity: 0.65;
  }
</style>

<div id="radioContainer">
  <div id="offRadio"></div>
  <div id="onRadio"></div>
</div>

<div id="radioLabel"><slot></slot></div>`;
    template$d.setAttribute('strip-whitespace', '');

    /**
    Material design: [Radio button](https://www.google.com/design/spec/components/selection-controls.html#selection-controls-radio-button)

    `paper-radio-button` is a button that can be either checked or unchecked. The
    user can tap the radio button to check or uncheck it.

    Use a `<paper-radio-group>` to group a set of radio buttons. When radio buttons
    are inside a radio group, exactly one radio button in the group can be checked
    at any time.

    Example:

        <paper-radio-button></paper-radio-button>
        <paper-radio-button>Item label</paper-radio-button>

    ### Styling

    The following custom properties and mixins are available for styling:

    Custom property | Description | Default
    ----------------|-------------|----------
    `--paper-radio-button-unchecked-background-color` | Radio button background color when the input is not checked | `transparent`
    `--paper-radio-button-unchecked-color` | Radio button color when the input is not checked | `--primary-text-color`
    `--paper-radio-button-unchecked-ink-color` | Selected/focus ripple color when the input is not checked | `--primary-text-color`
    `--paper-radio-button-checked-color` | Radio button color when the input is checked | `--primary-color`
    `--paper-radio-button-checked-ink-color` | Selected/focus ripple color when the input is checked | `--primary-color`
    `--paper-radio-button-size` | Size of the radio button | `16px`
    `--paper-radio-button-ink-size` | Size of the ripple | `48px`
    `--paper-radio-button-label-color` | Label color | `--primary-text-color`
    `--paper-radio-button-label-spacing` | Spacing between the label and the button | `10px`
    `--paper-radio-button-radio-container` | A mixin applied to the internal radio container | `{}`
    `--paper-radio-button-label` | A mixin applied to the internal label | `{}`
    `--paper-radio-button-label-checked` | A mixin applied to the internal label when the radio button is checked | `{}`

    This element applies the mixin `--paper-font-common-base` but does not import
    `paper-styles/typography.html`. In order to apply the `Roboto` font to this
    element, make sure you've imported `paper-styles/typography.html`.

    @group Paper Elements
    @element paper-radio-button
    @demo demo/index.html
    */
    Polymer({
      _template: template$d,

      is: 'paper-radio-button',

      behaviors: [PaperCheckedElementBehavior],

      hostAttributes: {role: 'radio', 'aria-checked': false, tabindex: 0},

      properties: {
        /**
         * Fired when the checked state changes due to user interaction.
         *
         * @event change
         */

        /**
         * Fired when the checked state changes.
         *
         * @event iron-change
         */

        ariaActiveAttribute: {type: String, value: 'aria-checked'}
      },

      ready: function() {
        this._rippleContainer = this.$.radioContainer;
      },

      attached: function() {
        // Wait until styles have resolved to check for the default sentinel.
        // See polymer#4009 for more details.
        afterNextRender(this, function() {
          var inkSize =
              this.getComputedStyleValue('--calculated-paper-radio-button-ink-size')
                  .trim();
          // If unset, compute and set the default `--paper-radio-button-ink-size`.
          if (inkSize === '-1px') {
            var size = parseFloat(
                this.getComputedStyleValue('--calculated-paper-radio-button-size')
                    .trim());
            var defaultInkSize = Math.floor(3 * size);

            // The button and ripple need to have the same parity so that their
            // centers align.
            if (defaultInkSize % 2 !== size % 2) {
              defaultInkSize++;
            }

            this.updateStyles({
              '--paper-radio-button-ink-size': defaultInkSize + 'px',
            });
          }
        });
      },
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
     * `IronMenubarBehavior` implements accessible menubar behavior.
     *
     * @polymerBehavior IronMenubarBehavior
     */
    const IronMenubarBehaviorImpl = {

      hostAttributes: {'role': 'menubar'},

      /**
       * @type {!Object}
       */
      keyBindings: {'left': '_onLeftKey', 'right': '_onRightKey'},

      _onUpKey: function(event) {
        this.focusedItem.click();
        event.detail.keyboardEvent.preventDefault();
      },

      _onDownKey: function(event) {
        this.focusedItem.click();
        event.detail.keyboardEvent.preventDefault();
      },

      get _isRTL() {
        return window.getComputedStyle(this)['direction'] === 'rtl';
      },

      _onLeftKey: function(event) {
        if (this._isRTL) {
          this._focusNext();
        } else {
          this._focusPrevious();
        }
        event.detail.keyboardEvent.preventDefault();
      },

      _onRightKey: function(event) {
        if (this._isRTL) {
          this._focusPrevious();
        } else {
          this._focusNext();
        }
        event.detail.keyboardEvent.preventDefault();
      },

      _onKeydown: function(event) {
        if (this.keyboardEventMatchesKeys(event, 'up down left right esc')) {
          return;
        }

        // all other keys focus the menu item starting with that character
        this._focusWithKeyboardEvent(event);
      }

    };

    /** @polymerBehavior */
    const IronMenubarBehavior = [IronMenuBehavior, IronMenubarBehaviorImpl];

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
    Material design: [Radio
    button](https://www.google.com/design/spec/components/selection-controls.html#selection-controls-radio-button)

    `paper-radio-group` allows user to select at most one radio button from a set.
    Checking one radio button that belongs to a radio group unchecks any
    previously checked radio button within the same group. Use
    `selected` to get or set the selected radio button.

    The <paper-radio-buttons> inside the group must have the `name` attribute
    set.

    Example:

        <paper-radio-group selected="small">
          <paper-radio-button name="small">Small</paper-radio-button>
          <paper-radio-button name="medium">Medium</paper-radio-button>
          <paper-radio-button name="large">Large</paper-radio-button>
        </paper-radio-group>

    Radio-button-groups can be made optional, and allow zero buttons to be selected:

        <paper-radio-group selected="small" allow-empty-selection>
          <paper-radio-button name="small">Small</paper-radio-button>
          <paper-radio-button name="medium">Medium</paper-radio-button>
          <paper-radio-button name="large">Large</paper-radio-button>
        </paper-radio-group>

    See <a href="paper-radio-button">paper-radio-button</a> for more
    information about `paper-radio-button`.


    Custom property | Description | Default
    ----------------|-------------|----------
    `--paper-radio-group-item-padding` | The padding of the item | `12px`

    @group Paper Elements
    @element paper-radio-group
    @demo demo/index.html
    */
    Polymer({
      _template: html`
    <style>
      :host {
        display: inline-block;
      }

      :host ::slotted(*) {
        padding: var(--paper-radio-group-item-padding, 12px);
      }
    </style>

    <slot></slot>
`,

      is: 'paper-radio-group',
      behaviors: [IronMenubarBehavior],

      /** @private */
      hostAttributes: {
        role: 'radiogroup',
      },

      properties: {
        /**
         * Fired when the radio group selection changes.
         *
         * @event paper-radio-group-changed
         */

        /**
         * Overriden from Polymer.IronSelectableBehavior
         */
        attrForSelected: {type: String, value: 'name'},

        /**
         * Overriden from Polymer.IronSelectableBehavior
         */
        selectedAttribute: {type: String, value: 'checked'},

        /**
         * Overriden from Polymer.IronSelectableBehavior
         */
        selectable: {type: String, value: 'paper-radio-button'},

        /**
         * If true, radio-buttons can be deselected
         */
        allowEmptySelection: {type: Boolean, value: false}
      },

      /**
       * Selects the given value.
       */
      select: function(value) {
        var newItem = this._valueToItem(value);
        if (newItem && newItem.hasAttribute('disabled')) {
          return;
        }

        if (this.selected) {
          var oldItem = this._valueToItem(this.selected);

          if (this.selected == value) {
            // If deselecting is allowed we'll have to apply an empty selection.
            // Otherwise, we should force the selection to stay and make this
            // action a no-op.
            if (this.allowEmptySelection) {
              value = '';
            } else {
              if (oldItem)
                oldItem.checked = true;
              return;
            }
          }

          if (oldItem)
            oldItem.checked = false;
        }

        IronSelectableBehavior.select.apply(this, [value]);
        this.fire('paper-radio-group-changed');
      },

      _activateFocusedItem: function() {
        this._itemActivate(this._valueForItem(this.focusedItem), this.focusedItem);
      },

      _onUpKey: function(event) {
        this._focusPrevious();
        event.preventDefault();
        this._activateFocusedItem();
      },

      _onDownKey: function(event) {
        this._focusNext();
        event.preventDefault();
        this._activateFocusedItem();
      },

      _onLeftKey: function(event) {
        IronMenubarBehaviorImpl._onLeftKey.apply(this, arguments);
        this._activateFocusedItem();
      },

      _onRightKey: function(event) {
        IronMenubarBehaviorImpl._onRightKey.apply(this, arguments);
        this._activateFocusedItem();
      }
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    const template$e = html`
  <style>
    :host {
      @apply --layout;
      @apply --layout-justified;
      @apply --layout-center;
      width: 200px;
      cursor: default;
      -webkit-user-select: none;
      -moz-user-select: none;
      -ms-user-select: none;
      user-select: none;
      -webkit-tap-highlight-color: rgba(0, 0, 0, 0);
      --paper-progress-active-color: var(--paper-slider-active-color, var(--google-blue-700));
      --paper-progress-secondary-color: var(--paper-slider-secondary-color, var(--google-blue-300));
      --paper-progress-disabled-active-color: var(--paper-slider-disabled-active-color, var(--paper-grey-400));
      --paper-progress-disabled-secondary-color: var(--paper-slider-disabled-secondary-color, var(--paper-grey-400));
      --calculated-paper-slider-height: var(--paper-slider-height, 2px);
    }

    /* focus shows the ripple */
    :host(:focus) {
      outline: none;
    }

    /**
      * NOTE(keanulee): Though :host-context is not universally supported, some pages
      * still rely on paper-slider being flipped when dir="rtl" is set on body. For full
      * compatibility, dir="rtl" must be explicitly set on paper-slider.
      */
    :dir(rtl) #sliderContainer {
      -webkit-transform: scaleX(-1);
      transform: scaleX(-1);
    }

    /**
      * NOTE(keanulee): This is separate from the rule above because :host-context may
      * not be recognized.
      */
    :host([dir="rtl"]) #sliderContainer {
      -webkit-transform: scaleX(-1);
      transform: scaleX(-1);
    }

    /**
      * NOTE(keanulee): Needed to override the :host-context rule (where supported)
      * to support LTR sliders in RTL pages.
      */
    :host([dir="ltr"]) #sliderContainer {
      -webkit-transform: scaleX(1);
      transform: scaleX(1);
    }

    #sliderContainer {
      position: relative;
      width: 100%;
      height: calc(30px + var(--calculated-paper-slider-height));
      margin-left: calc(15px + var(--calculated-paper-slider-height)/2);
      margin-right: calc(15px + var(--calculated-paper-slider-height)/2);
    }

    #sliderContainer:focus {
      outline: 0;
    }

    #sliderContainer.editable {
      margin-top: 12px;
      margin-bottom: 12px;
    }

    .bar-container {
      position: absolute;
      top: 0;
      bottom: 0;
      left: 0;
      right: 0;
      overflow: hidden;
    }

    .ring > .bar-container {
      left: calc(5px + var(--calculated-paper-slider-height)/2);
      transition: left 0.18s ease;
    }

    .ring.expand.dragging > .bar-container {
      transition: none;
    }

    .ring.expand:not(.pin) > .bar-container {
      left: calc(8px + var(--calculated-paper-slider-height)/2);
    }

    #sliderBar {
      padding: 15px 0;
      width: 100%;
      background-color: var(--paper-slider-bar-color, transparent);
      --paper-progress-container-color: var(--paper-slider-container-color, var(--paper-grey-400));
      --paper-progress-height: var(--calculated-paper-slider-height);
    }

    .slider-markers {
      position: absolute;
      /* slider-knob is 30px + the slider-height so that the markers should start at a offset of 15px*/
      top: 15px;
      height: var(--calculated-paper-slider-height);
      left: 0;
      right: -1px;
      box-sizing: border-box;
      pointer-events: none;
      @apply --layout-horizontal;
    }

    .slider-marker {
      @apply --layout-flex;
    }
    .slider-markers::after,
    .slider-marker::after {
      content: "";
      display: block;
      margin-left: -1px;
      width: 2px;
      height: var(--calculated-paper-slider-height);
      border-radius: 50%;
      background-color: var(--paper-slider-markers-color, #000);
    }

    .slider-knob {
      position: absolute;
      left: 0;
      top: 0;
      margin-left: calc(-15px - var(--calculated-paper-slider-height)/2);
      width: calc(30px + var(--calculated-paper-slider-height));
      height: calc(30px + var(--calculated-paper-slider-height));
    }

    .transiting > .slider-knob {
      transition: left 0.08s ease;
    }

    .slider-knob:focus {
      outline: none;
    }

    .slider-knob.dragging {
      transition: none;
    }

    .snaps > .slider-knob.dragging {
      transition: -webkit-transform 0.08s ease;
      transition: transform 0.08s ease;
    }

    .slider-knob-inner {
      margin: 10px;
      width: calc(100% - 20px);
      height: calc(100% - 20px);
      background-color: var(--paper-slider-knob-color, var(--google-blue-700));
      border: 2px solid var(--paper-slider-knob-color, var(--google-blue-700));
      border-radius: 50%;

      -moz-box-sizing: border-box;
      box-sizing: border-box;

      transition-property: -webkit-transform, background-color, border;
      transition-property: transform, background-color, border;
      transition-duration: 0.18s;
      transition-timing-function: ease;
    }

    .expand:not(.pin) > .slider-knob > .slider-knob-inner {
      -webkit-transform: scale(1.5);
      transform: scale(1.5);
    }

    .ring > .slider-knob > .slider-knob-inner {
      background-color: var(--paper-slider-knob-start-color, transparent);
      border: 2px solid var(--paper-slider-knob-start-border-color, var(--paper-grey-400));
    }

    .slider-knob-inner::before {
      background-color: var(--paper-slider-pin-color, var(--google-blue-700));
    }

    .pin > .slider-knob > .slider-knob-inner::before {
      content: "";
      position: absolute;
      top: 0;
      left: 50%;
      margin-left: -13px;
      width: 26px;
      height: 26px;
      border-radius: 50% 50% 50% 0;

      -webkit-transform: rotate(-45deg) scale(0) translate(0);
      transform: rotate(-45deg) scale(0) translate(0);
    }

    .slider-knob-inner::before,
    .slider-knob-inner::after {
      transition: -webkit-transform .18s ease, background-color .18s ease;
      transition: transform .18s ease, background-color .18s ease;
    }

    .pin.ring > .slider-knob > .slider-knob-inner::before {
      background-color: var(--paper-slider-pin-start-color, var(--paper-grey-400));
    }

    .pin.expand > .slider-knob > .slider-knob-inner::before {
      -webkit-transform: rotate(-45deg) scale(1) translate(17px, -17px);
      transform: rotate(-45deg) scale(1) translate(17px, -17px);
    }

    .pin > .slider-knob > .slider-knob-inner::after {
      content: attr(value);
      position: absolute;
      top: 0;
      left: 50%;
      margin-left: -16px;
      width: 32px;
      height: 26px;
      text-align: center;
      color: var(--paper-slider-font-color, #fff);
      font-size: 10px;

      -webkit-transform: scale(0) translate(0);
      transform: scale(0) translate(0);
    }

    .pin.expand > .slider-knob > .slider-knob-inner::after {
      -webkit-transform: scale(1) translate(0, -17px);
      transform: scale(1) translate(0, -17px);
    }

    /* paper-input */
    .slider-input {
      width: 50px;
      overflow: hidden;
      --paper-input-container-input: {
        text-align: center;
        @apply --paper-slider-input-container-input;
      };
      @apply --paper-slider-input;
    }

    /* disabled state */
    #sliderContainer.disabled {
      pointer-events: none;
    }

    .disabled > .slider-knob > .slider-knob-inner {
      background-color: var(--paper-slider-disabled-knob-color, var(--paper-grey-400));
      border: 2px solid var(--paper-slider-disabled-knob-color, var(--paper-grey-400));
      -webkit-transform: scale3d(0.75, 0.75, 1);
      transform: scale3d(0.75, 0.75, 1);
    }

    .disabled.ring > .slider-knob > .slider-knob-inner {
      background-color: var(--paper-slider-knob-start-color, transparent);
      border: 2px solid var(--paper-slider-knob-start-border-color, var(--paper-grey-400));
    }

    paper-ripple {
      color: var(--paper-slider-knob-color, var(--google-blue-700));
    }
  </style>

  <div id="sliderContainer" class\$="[[_getClassNames(disabled, pin, snaps, immediateValue, min, expand, dragging, transiting, editable)]]">
    <div class="bar-container">
      <paper-progress disabled\$="[[disabled]]" id="sliderBar" aria-hidden="true" min="[[min]]" max="[[max]]" step="[[step]]" value="[[immediateValue]]" secondary-progress="[[secondaryProgress]]" on-down="_bardown" on-up="_resetKnob" on-track="_bartrack" on-tap="_barclick">
      </paper-progress>
    </div>

    <template is="dom-if" if="[[snaps]]">
      <div class="slider-markers">
        <template is="dom-repeat" items="[[markers]]">
          <div class="slider-marker"></div>
        </template>
      </div>
    </template>

    <div id="sliderKnob" class="slider-knob" on-down="_knobdown" on-up="_resetKnob" on-track="_onTrack" on-transitionend="_knobTransitionEnd">
        <div class="slider-knob-inner" value\$="[[immediateValue]]"></div>
    </div>
  </div>

  <template is="dom-if" if="[[editable]]">
    <paper-input id="input" type="number" step="[[step]]" min="[[min]]" max="[[max]]" class="slider-input" disabled\$="[[disabled]]" value="[[immediateValue]]" on-change="_changeValue" on-keydown="_inputKeyDown" no-label-float>
    </paper-input>
  </template>
`;
    template$e.setAttribute('strip-whitespace', '');

    /**
    Material design:
    [Sliders](https://www.google.com/design/spec/components/sliders.html)

    `paper-slider` allows user to select a value from a range of values by
    moving the slider thumb.  The interactive nature of the slider makes it a
    great choice for settings that reflect intensity levels, such as volume,
    brightness, or color saturation.

    Example:

        <paper-slider></paper-slider>

    Use `min` and `max` to specify the slider range.  Default is 0 to 100.

    Example:

        <paper-slider min="10" max="200" value="110"></paper-slider>

    ### Styling

    The following custom properties and mixins are available for styling:

    Custom property | Description | Default
    ----------------|-------------|----------
    `--paper-slider-container-color` | The background color of the bar | `--paper-grey-400`
    `--paper-slider-bar-color` | The background color of the slider | `transparent`
    `--paper-slider-active-color` | The progress bar color | `--google-blue-700`
    `--paper-slider-secondary-color` | The secondary progress bar color | `--google-blue-300`
    `--paper-slider-knob-color` | The knob color | `--google-blue-700`
    `--paper-slider-disabled-knob-color` | The disabled knob color | `--paper-grey-400`
    `--paper-slider-pin-color` | The pin color | `--google-blue-700`
    `--paper-slider-font-color` | The pin's text color | `#fff`
    `--paper-slider-markers-color` | The snaps markers color | `#000`
    `--paper-slider-disabled-active-color` | The disabled progress bar color | `--paper-grey-400`
    `--paper-slider-disabled-secondary-color` | The disabled secondary progress bar color | `--paper-grey-400`
    `--paper-slider-knob-start-color` | The fill color of the knob at the far left | `transparent`
    `--paper-slider-knob-start-border-color` | The border color of the knob at the far left | `--paper-grey-400`
    `--paper-slider-pin-start-color` | The color of the pin at the far left | `--paper-grey-400`
    `--paper-slider-height` | Height of the progress bar | `2px`
    `--paper-slider-input` | Mixin applied to the input in editable mode | `{}`
    `--paper-slider-input-container-input` | Mixin applied to the paper-input-container-input in editable mode | `{}`

    @group Paper Elements
    @element paper-slider
    @demo demo/index.html
    */
    Polymer({
      _template: template$e,

      is: 'paper-slider',

      behaviors: [
        IronA11yKeysBehavior,
        IronFormElementBehavior,
        PaperInkyFocusBehavior,
        IronRangeBehavior
      ],

      properties: {
        value: {
          type: Number,
          value: 0,
        },

        /**
         * If true, the slider thumb snaps to tick marks evenly spaced based
         * on the `step` property value.
         */
        snaps: {
          type: Boolean,
          value: false,
          notify: true,
        },

        /**
         * If true, a pin with numeric value label is shown when the slider thumb
         * is pressed. Use for settings for which users need to know the exact
         * value of the setting.
         */
        pin: {
          type: Boolean,
          value: false,
          notify: true,
        },

        /**
         * The number that represents the current secondary progress.
         */
        secondaryProgress: {
          type: Number,
          value: 0,
          notify: true,
          observer: '_secondaryProgressChanged',
        },

        /**
         * If true, an input is shown and user can use it to set the slider value.
         */
        editable: {
          type: Boolean,
          value: false,
        },

        /**
         * The immediate value of the slider.  This value is updated while the user
         * is dragging the slider.
         */
        immediateValue: {
          type: Number,
          value: 0,
          readOnly: true,
          notify: true,
        },

        /**
         * The maximum number of markers
         */
        maxMarkers: {
          type: Number,
          value: 0,
          notify: true,
        },

        /**
         * If true, the knob is expanded
         */
        expand: {
          type: Boolean,
          value: false,
          readOnly: true,
        },

        /**
         * If true, a touchmove on the slider bar doesn't drag the slider thunb.
         * Tapping on the slider bar still updates the slider's position
         */
        ignoreBarTouch: {
          type: Boolean,
          value: false,
        },

        /**
         * True when the user is dragging the slider.
         */
        dragging: {
          type: Boolean,
          value: false,
          readOnly: true,
          notify: true,
        },

        transiting: {
          type: Boolean,
          value: false,
          readOnly: true,
        },

        markers: {
          type: Array,
          readOnly: true,
          value: function() {
            return [];
          },
        },
      },

      observers: [
        '_updateKnob(value, min, max, snaps, step)',
        '_valueChanged(value)',
        '_immediateValueChanged(immediateValue)',
        '_updateMarkers(maxMarkers, min, max, snaps)',
      ],

      hostAttributes: {
        role: 'slider',
        tabindex: 0,
      },

      /** @type {!Object} */
      keyBindings: {
        'left': '_leftKey',
        'right': '_rightKey',
        'down pagedown home': '_decrementKey',
        'up pageup end': '_incrementKey',
      },

      ready: function() {
        if (this.ignoreBarTouch) {
          setTouchAction(this.$.sliderBar, 'auto');
        }
      },

      /**
       * Increases value by `step` but not above `max`.
       * @method increment
       */
      increment: function() {
        this.value = this._clampValue(this.value + this.step);
      },

      /**
       * Decreases value by `step` but not below `min`.
       * @method decrement
       */
      decrement: function() {
        this.value = this._clampValue(this.value - this.step);
      },

      _updateKnob: function(value, min, max, snaps, step) {
        this.setAttribute('aria-valuemin', min);
        this.setAttribute('aria-valuemax', max);
        this.setAttribute('aria-valuenow', value);

        this._positionKnob(this._calcRatio(value) * 100);
      },

      _valueChanged: function() {
        this.fire('value-change', {composed: true});
      },

      _immediateValueChanged: function() {
        if (this.dragging) {
          this.fire('immediate-value-change', {composed: true});
        } else {
          this.value = this.immediateValue;
        }
      },

      _secondaryProgressChanged: function() {
        this.secondaryProgress = this._clampValue(this.secondaryProgress);
      },

      _expandKnob: function() {
        this._setExpand(true);
      },

      _resetKnob: function() {
        this.cancelDebouncer('expandKnob');
        this._setExpand(false);
      },

      _positionKnob: function(ratio) {
        this._setImmediateValue(this._calcStep(this._calcKnobPosition(ratio)));
        this._setRatio(this._calcRatio(this.immediateValue) * 100);

        this.$.sliderKnob.style.left = this.ratio + '%';
        if (this.dragging) {
          this._knobstartx = (this.ratio * this._w) / 100;
          this.translate3d(0, 0, 0, this.$.sliderKnob);
        }
      },

      _calcKnobPosition: function(ratio) {
        return (this.max - this.min) * ratio / 100 + this.min;
      },

      _onTrack: function(event) {
        event.stopPropagation();
        switch (event.detail.state) {
          case 'start':
            this._trackStart(event);
            break;
          case 'track':
            this._trackX(event);
            break;
          case 'end':
            this._trackEnd();
            break;
        }
      },

      _trackStart: function(event) {
        this._setTransiting(false);
        this._w = this.$.sliderBar.offsetWidth;
        this._x = this.ratio * this._w / 100;
        this._startx = this._x;
        this._knobstartx = this._startx;
        this._minx = -this._startx;
        this._maxx = this._w - this._startx;
        this.$.sliderKnob.classList.add('dragging');
        this._setDragging(true);
      },

      _trackX: function(event) {
        if (!this.dragging) {
          this._trackStart(event);
        }

        var direction = this._isRTL ? -1 : 1;
        var dx =
            Math.min(this._maxx, Math.max(this._minx, event.detail.dx * direction));
        this._x = this._startx + dx;

        var immediateValue =
            this._calcStep(this._calcKnobPosition(this._x / this._w * 100));
        this._setImmediateValue(immediateValue);

        // update knob's position
        var translateX =
            ((this._calcRatio(this.immediateValue) * this._w) - this._knobstartx);
        this.translate3d(translateX + 'px', 0, 0, this.$.sliderKnob);
      },

      _trackEnd: function() {
        var s = this.$.sliderKnob.style;

        this.$.sliderKnob.classList.remove('dragging');
        this._setDragging(false);
        this._resetKnob();
        this.value = this.immediateValue;

        s.transform = s.webkitTransform = '';

        this.fire('change', {composed: true});
      },

      _knobdown: function(event) {
        this._expandKnob();

        // cancel selection
        event.preventDefault();

        // set the focus manually because we will called prevent default
        this.focus();
      },

      _bartrack: function(event) {
        if (this._allowBarEvent(event)) {
          this._onTrack(event);
        }
      },

      _barclick: function(event) {
        this._w = this.$.sliderBar.offsetWidth;
        var rect = this.$.sliderBar.getBoundingClientRect();
        var ratio = (event.detail.x - rect.left) / this._w * 100;
        if (this._isRTL) {
          ratio = 100 - ratio;
        }
        var prevRatio = this.ratio;

        this._setTransiting(true);
        this._positionKnob(ratio);

        // if the ratio doesn't change, sliderKnob's animation won't start
        // and `_knobTransitionEnd` won't be called
        // Therefore, we need to manually update the `transiting` state
        if (prevRatio === this.ratio) {
          this._setTransiting(false);
        }

        this.async(function() {
          this.fire('change', {composed: true});
        });

        // cancel selection
        event.preventDefault();

        // set the focus manually because we will called prevent default
        this.focus();
      },

      _bardown: function(event) {
        if (this._allowBarEvent(event)) {
          this.debounce('expandKnob', this._expandKnob, 60);
          this._barclick(event);
        }
      },

      _knobTransitionEnd: function(event) {
        if (event.target === this.$.sliderKnob) {
          this._setTransiting(false);
        }
      },

      _updateMarkers: function(maxMarkers, min, max, snaps) {
        if (!snaps) {
          this._setMarkers([]);
        }
        var steps = Math.round((max - min) / this.step);
        if (steps > maxMarkers) {
          steps = maxMarkers;
        }
        if (steps < 0 || !isFinite(steps)) {
          steps = 0;
        }
        this._setMarkers(new Array(steps));
      },

      _mergeClasses: function(classes) {
        return Object.keys(classes)
            .filter(function(className) {
              return classes[className];
            })
            .join(' ');
      },

      _getClassNames: function() {
        return this._mergeClasses({
          disabled: this.disabled,
          pin: this.pin,
          snaps: this.snaps,
          ring: this.immediateValue <= this.min,
          expand: this.expand,
          dragging: this.dragging,
          transiting: this.transiting,
          editable: this.editable
        });
      },

      _allowBarEvent: function(event) {
        return (
            !this.ignoreBarTouch ||
            (event.detail.sourceEvent instanceof MouseEvent))
      },

      get _isRTL() {
        if (this.__isRTL === undefined) {
          this.__isRTL = window.getComputedStyle(this)['direction'] === 'rtl';
        }
        return this.__isRTL;
      },

      _leftKey: function(event) {
        if (this._isRTL)
          this._incrementKey(event);
        else
          this._decrementKey(event);
      },

      _rightKey: function(event) {
        if (this._isRTL)
          this._decrementKey(event);
        else
          this._incrementKey(event);
      },

      _incrementKey: function(event) {
        if (!this.disabled) {
          if (event.detail.key === 'end') {
            this.value = this.max;
          } else {
            this.increment();
          }
          this.fire('change');
          event.preventDefault();
        }
      },

      _decrementKey: function(event) {
        if (!this.disabled) {
          if (event.detail.key === 'home') {
            this.value = this.min;
          } else {
            this.decrement();
          }
          this.fire('change');
          event.preventDefault();
        }
      },

      _changeValue: function(event) {
        this.value = event.target.value;
        this.fire('change', {composed: true});
      },

      _inputKeyDown: function(event) {
        event.stopPropagation();
      },

      // create the element ripple inside the `sliderKnob`
      _createRipple: function() {
        this._rippleContainer = this.$.sliderKnob;
        return PaperInkyFocusBehaviorImpl._createRipple.call(this);
      },

      // Hide the ripple when user is not interacting with keyboard.
      // This behavior is different from other ripple-y controls, but is
      // according to spec:
      // https://www.google.com/design/spec/components/sliders.html
      _focusedChanged: function(receivedFocusFromKeyboard) {
        if (receivedFocusFromKeyboard) {
          this.ensureRipple();
        }
        if (this.hasRipple()) {
          // note, ripple must be un-hidden prior to setting `holdDown`
          if (receivedFocusFromKeyboard) {
            this._ripple.style.display = '';
          } else {
            this._ripple.style.display = 'none';
          }
          this._ripple.holdDown = receivedFocusFromKeyboard;
        }
      }
      /**
       * Fired when the slider's value changes.
       *
       * @event value-change
       */

      /**
       * Fired when the slider's immediateValue changes. Only occurs while the
       * user is dragging.
       *
       * To detect changes to immediateValue that happen for any input (i.e.
       * dragging, tapping, clicking, etc.) listen for immediate-value-changed
       * instead.
       *
       * @event immediate-value-change
       */

      /**
       * Fired when the slider's value changes due to user interaction.
       *
       * Changes to the slider's value due to changes in an underlying
       * bound variable will not trigger this event.
       *
       * @event change
       */
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */
    const $_documentContainer$4 = document.createElement('template');
    $_documentContainer$4.setAttribute('style', 'display: none;');

    $_documentContainer$4.innerHTML = `<dom-module id="paper-spinner-styles">
  <template>
    <style>
      /*
      /**************************/
      /* STYLES FOR THE SPINNER */
      /**************************/

      /*
       * Constants:
       *      ARCSIZE     = 270 degrees (amount of circle the arc takes up)
       *      ARCTIME     = 1333ms (time it takes to expand and contract arc)
       *      ARCSTARTROT = 216 degrees (how much the start location of the arc
       *                                should rotate each time, 216 gives us a
       *                                5 pointed star shape (it's 360/5 * 3).
       *                                For a 7 pointed star, we might do
       *                                360/7 * 3 = 154.286)
       *      SHRINK_TIME = 400ms
       */

      :host {
        display: inline-block;
        position: relative;
        width: 28px;
        height: 28px;

        /* 360 * ARCTIME / (ARCSTARTROT + (360-ARCSIZE)) */
        --paper-spinner-container-rotation-duration: 1568ms;

        /* ARCTIME */
        --paper-spinner-expand-contract-duration: 1333ms;

        /* 4 * ARCTIME */
        --paper-spinner-full-cycle-duration: 5332ms;

        /* SHRINK_TIME */
        --paper-spinner-cooldown-duration: 400ms;
      }

      #spinnerContainer {
        width: 100%;
        height: 100%;

        /* The spinner does not have any contents that would have to be
         * flipped if the direction changes. Always use ltr so that the
         * style works out correctly in both cases. */
        direction: ltr;
      }

      #spinnerContainer.active {
        -webkit-animation: container-rotate var(--paper-spinner-container-rotation-duration) linear infinite;
        animation: container-rotate var(--paper-spinner-container-rotation-duration) linear infinite;
      }

      @-webkit-keyframes container-rotate {
        to { -webkit-transform: rotate(360deg) }
      }

      @keyframes container-rotate {
        to { transform: rotate(360deg) }
      }

      .spinner-layer {
        position: absolute;
        width: 100%;
        height: 100%;
        opacity: 0;
        white-space: nowrap;
        color: var(--paper-spinner-color, var(--google-blue-500));
      }

      .layer-1 {
        color: var(--paper-spinner-layer-1-color, var(--google-blue-500));
      }

      .layer-2 {
        color: var(--paper-spinner-layer-2-color, var(--google-red-500));
      }

      .layer-3 {
        color: var(--paper-spinner-layer-3-color, var(--google-yellow-500));
      }

      .layer-4 {
        color: var(--paper-spinner-layer-4-color, var(--google-green-500));
      }

      /**
       * IMPORTANT NOTE ABOUT CSS ANIMATION PROPERTIES (keanulee):
       *
       * iOS Safari (tested on iOS 8.1) does not handle animation-delay very well - it doesn't
       * guarantee that the animation will start _exactly_ after that value. So we avoid using
       * animation-delay and instead set custom keyframes for each color (as layer-2undant as it
       * seems).
       */
      .active .spinner-layer {
        -webkit-animation-name: fill-unfill-rotate;
        -webkit-animation-duration: var(--paper-spinner-full-cycle-duration);
        -webkit-animation-timing-function: cubic-bezier(0.4, 0.0, 0.2, 1);
        -webkit-animation-iteration-count: infinite;
        animation-name: fill-unfill-rotate;
        animation-duration: var(--paper-spinner-full-cycle-duration);
        animation-timing-function: cubic-bezier(0.4, 0.0, 0.2, 1);
        animation-iteration-count: infinite;
        opacity: 1;
      }

      .active .spinner-layer.layer-1 {
        -webkit-animation-name: fill-unfill-rotate, layer-1-fade-in-out;
        animation-name: fill-unfill-rotate, layer-1-fade-in-out;
      }

      .active .spinner-layer.layer-2 {
        -webkit-animation-name: fill-unfill-rotate, layer-2-fade-in-out;
        animation-name: fill-unfill-rotate, layer-2-fade-in-out;
      }

      .active .spinner-layer.layer-3 {
        -webkit-animation-name: fill-unfill-rotate, layer-3-fade-in-out;
        animation-name: fill-unfill-rotate, layer-3-fade-in-out;
      }

      .active .spinner-layer.layer-4 {
        -webkit-animation-name: fill-unfill-rotate, layer-4-fade-in-out;
        animation-name: fill-unfill-rotate, layer-4-fade-in-out;
      }

      @-webkit-keyframes fill-unfill-rotate {
        12.5% { -webkit-transform: rotate(135deg) } /* 0.5 * ARCSIZE */
        25%   { -webkit-transform: rotate(270deg) } /* 1   * ARCSIZE */
        37.5% { -webkit-transform: rotate(405deg) } /* 1.5 * ARCSIZE */
        50%   { -webkit-transform: rotate(540deg) } /* 2   * ARCSIZE */
        62.5% { -webkit-transform: rotate(675deg) } /* 2.5 * ARCSIZE */
        75%   { -webkit-transform: rotate(810deg) } /* 3   * ARCSIZE */
        87.5% { -webkit-transform: rotate(945deg) } /* 3.5 * ARCSIZE */
        to    { -webkit-transform: rotate(1080deg) } /* 4   * ARCSIZE */
      }

      @keyframes fill-unfill-rotate {
        12.5% { transform: rotate(135deg) } /* 0.5 * ARCSIZE */
        25%   { transform: rotate(270deg) } /* 1   * ARCSIZE */
        37.5% { transform: rotate(405deg) } /* 1.5 * ARCSIZE */
        50%   { transform: rotate(540deg) } /* 2   * ARCSIZE */
        62.5% { transform: rotate(675deg) } /* 2.5 * ARCSIZE */
        75%   { transform: rotate(810deg) } /* 3   * ARCSIZE */
        87.5% { transform: rotate(945deg) } /* 3.5 * ARCSIZE */
        to    { transform: rotate(1080deg) } /* 4   * ARCSIZE */
      }

      @-webkit-keyframes layer-1-fade-in-out {
        0% { opacity: 1 }
        25% { opacity: 1 }
        26% { opacity: 0 }
        89% { opacity: 0 }
        90% { opacity: 1 }
        to { opacity: 1 }
      }

      @keyframes layer-1-fade-in-out {
        0% { opacity: 1 }
        25% { opacity: 1 }
        26% { opacity: 0 }
        89% { opacity: 0 }
        90% { opacity: 1 }
        to { opacity: 1 }
      }

      @-webkit-keyframes layer-2-fade-in-out {
        0% { opacity: 0 }
        15% { opacity: 0 }
        25% { opacity: 1 }
        50% { opacity: 1 }
        51% { opacity: 0 }
        to { opacity: 0 }
      }

      @keyframes layer-2-fade-in-out {
        0% { opacity: 0 }
        15% { opacity: 0 }
        25% { opacity: 1 }
        50% { opacity: 1 }
        51% { opacity: 0 }
        to { opacity: 0 }
      }

      @-webkit-keyframes layer-3-fade-in-out {
        0% { opacity: 0 }
        40% { opacity: 0 }
        50% { opacity: 1 }
        75% { opacity: 1 }
        76% { opacity: 0 }
        to { opacity: 0 }
      }

      @keyframes layer-3-fade-in-out {
        0% { opacity: 0 }
        40% { opacity: 0 }
        50% { opacity: 1 }
        75% { opacity: 1 }
        76% { opacity: 0 }
        to { opacity: 0 }
      }

      @-webkit-keyframes layer-4-fade-in-out {
        0% { opacity: 0 }
        65% { opacity: 0 }
        75% { opacity: 1 }
        90% { opacity: 1 }
        to { opacity: 0 }
      }

      @keyframes layer-4-fade-in-out {
        0% { opacity: 0 }
        65% { opacity: 0 }
        75% { opacity: 1 }
        90% { opacity: 1 }
        to { opacity: 0 }
      }

      .circle-clipper {
        display: inline-block;
        position: relative;
        width: 50%;
        height: 100%;
        overflow: hidden;
      }

      /**
       * Patch the gap that appear between the two adjacent div.circle-clipper while the
       * spinner is rotating (appears on Chrome 50, Safari 9.1.1, and Edge).
       */
      .spinner-layer::after {
        content: '';
        left: 45%;
        width: 10%;
        border-top-style: solid;
      }

      .spinner-layer::after,
      .circle-clipper .circle {
        box-sizing: border-box;
        position: absolute;
        top: 0;
        border-width: var(--paper-spinner-stroke-width, 3px);
        border-radius: 50%;
      }

      .circle-clipper .circle {
        bottom: 0;
        width: 200%;
        border-style: solid;
        border-bottom-color: transparent !important;
      }

      .circle-clipper.left .circle {
        left: 0;
        border-right-color: transparent !important;
        -webkit-transform: rotate(129deg);
        transform: rotate(129deg);
      }

      .circle-clipper.right .circle {
        left: -100%;
        border-left-color: transparent !important;
        -webkit-transform: rotate(-129deg);
        transform: rotate(-129deg);
      }

      .active .gap-patch::after,
      .active .circle-clipper .circle {
        -webkit-animation-duration: var(--paper-spinner-expand-contract-duration);
        -webkit-animation-timing-function: cubic-bezier(0.4, 0.0, 0.2, 1);
        -webkit-animation-iteration-count: infinite;
        animation-duration: var(--paper-spinner-expand-contract-duration);
        animation-timing-function: cubic-bezier(0.4, 0.0, 0.2, 1);
        animation-iteration-count: infinite;
      }

      .active .circle-clipper.left .circle {
        -webkit-animation-name: left-spin;
        animation-name: left-spin;
      }

      .active .circle-clipper.right .circle {
        -webkit-animation-name: right-spin;
        animation-name: right-spin;
      }

      @-webkit-keyframes left-spin {
        0% { -webkit-transform: rotate(130deg) }
        50% { -webkit-transform: rotate(-5deg) }
        to { -webkit-transform: rotate(130deg) }
      }

      @keyframes left-spin {
        0% { transform: rotate(130deg) }
        50% { transform: rotate(-5deg) }
        to { transform: rotate(130deg) }
      }

      @-webkit-keyframes right-spin {
        0% { -webkit-transform: rotate(-130deg) }
        50% { -webkit-transform: rotate(5deg) }
        to { -webkit-transform: rotate(-130deg) }
      }

      @keyframes right-spin {
        0% { transform: rotate(-130deg) }
        50% { transform: rotate(5deg) }
        to { transform: rotate(-130deg) }
      }

      #spinnerContainer.cooldown {
        -webkit-animation: container-rotate var(--paper-spinner-container-rotation-duration) linear infinite, fade-out var(--paper-spinner-cooldown-duration) cubic-bezier(0.4, 0.0, 0.2, 1);
        animation: container-rotate var(--paper-spinner-container-rotation-duration) linear infinite, fade-out var(--paper-spinner-cooldown-duration) cubic-bezier(0.4, 0.0, 0.2, 1);
      }

      @-webkit-keyframes fade-out {
        0% { opacity: 1 }
        to { opacity: 0 }
      }

      @keyframes fade-out {
        0% { opacity: 1 }
        to { opacity: 0 }
      }
    </style>
  </template>
</dom-module>`;

    document.head.appendChild($_documentContainer$4.content);

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /** @polymerBehavior */
    const PaperSpinnerBehavior = {

      properties: {
        /**
         * Displays the spinner.
         */
        active: {
          type: Boolean,
          value: false,
          reflectToAttribute: true,
          observer: '__activeChanged'
        },

        /**
         * Alternative text content for accessibility support.
         * If alt is present, it will add an aria-label whose content matches alt
         * when active. If alt is not present, it will default to 'loading' as the
         * alt value.
         */
        alt: {type: String, value: 'loading', observer: '__altChanged'},

        __coolingDown: {type: Boolean, value: false}
      },

      __computeContainerClasses: function(active, coolingDown) {
        return [
          active || coolingDown ? 'active' : '',
          coolingDown ? 'cooldown' : ''
        ].join(' ');
      },

      __activeChanged: function(active, old) {
        this.__setAriaHidden(!active);
        this.__coolingDown = !active && old;
      },

      __altChanged: function(alt) {
        // user-provided `aria-label` takes precedence over prototype default
        if (alt === 'loading') {
          this.alt = this.getAttribute('aria-label') || alt;
        } else {
          this.__setAriaHidden(alt === '');
          this.setAttribute('aria-label', alt);
        }
      },

      __setAriaHidden: function(hidden) {
        var attr = 'aria-hidden';
        if (hidden) {
          this.setAttribute(attr, 'true');
        } else {
          this.removeAttribute(attr);
        }
      },

      __reset: function() {
        this.active = false;
        this.__coolingDown = false;
      }
    };

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    const template$f = html`
  <style include="paper-spinner-styles"></style>

  <div id="spinnerContainer" class-name="[[__computeContainerClasses(active, __coolingDown)]]" on-animationend="__reset" on-webkit-animation-end="__reset">
    <div class="spinner-layer">
      <div class="circle-clipper left">
        <div class="circle"></div>
      </div>
      <div class="circle-clipper right">
        <div class="circle"></div>
      </div>
    </div>
  </div>
`;
    template$f.setAttribute('strip-whitespace', '');

    /**
    Material design: [Progress &
    activity](https://www.google.com/design/spec/components/progress-activity.html)

    Element providing a single color material design circular spinner.

        <paper-spinner-lite active></paper-spinner-lite>

    The default spinner is blue. It can be customized to be a different color.

    ### Accessibility

    Alt attribute should be set to provide adequate context for accessibility. If
    not provided, it defaults to 'loading'. Empty alt can be provided to mark the
    element as decorative if alternative content is provided in another form (e.g. a
    text block following the spinner).

        <paper-spinner-lite alt="Loading contacts list" active></paper-spinner-lite>

    ### Styling

    The following custom properties and mixins are available for styling:

    Custom property | Description | Default
    ----------------|-------------|----------
    `--paper-spinner-color` | Color of the spinner | `--google-blue-500`
    `--paper-spinner-stroke-width` | The width of the spinner stroke | 3px

    @group Paper Elements
    @element paper-spinner-lite
    @hero hero.svg
    @demo demo/index.html
    */
    Polymer({
      _template: template$f,

      is: 'paper-spinner-lite',

      behaviors: [PaperSpinnerBehavior]
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    const template$g = html`
  <style include="paper-spinner-styles"></style>

  <div id="spinnerContainer" class-name="[[__computeContainerClasses(active, __coolingDown)]]" on-animationend="__reset" on-webkit-animation-end="__reset">
    <div class="spinner-layer layer-1">
      <div class="circle-clipper left">
        <div class="circle"></div>
      </div>
      <div class="circle-clipper right">
        <div class="circle"></div>
      </div>
    </div>

    <div class="spinner-layer layer-2">
      <div class="circle-clipper left">
        <div class="circle"></div>
      </div>
      <div class="circle-clipper right">
        <div class="circle"></div>
      </div>
    </div>

    <div class="spinner-layer layer-3">
      <div class="circle-clipper left">
        <div class="circle"></div>
      </div>
      <div class="circle-clipper right">
        <div class="circle"></div>
      </div>
    </div>

    <div class="spinner-layer layer-4">
      <div class="circle-clipper left">
        <div class="circle"></div>
      </div>
      <div class="circle-clipper right">
        <div class="circle"></div>
      </div>
    </div>
  </div>
`;
    template$g.setAttribute('strip-whitespace', '');

    /**
    Material design: [Progress &
    activity](https://www.google.com/design/spec/components/progress-activity.html)

    Element providing a multiple color material design circular spinner.

        <paper-spinner active></paper-spinner>

    The default spinner cycles between four layers of colors; by default they are
    blue, red, yellow and green. It can be customized to cycle between four
    different colors. Use <paper-spinner-lite> for single color spinners.

    ### Accessibility

    Alt attribute should be set to provide adequate context for accessibility. If
    not provided, it defaults to 'loading'. Empty alt can be provided to mark the
    element as decorative if alternative content is provided in another form (e.g. a
    text block following the spinner).

        <paper-spinner alt="Loading contacts list" active></paper-spinner>

    ### Styling

    The following custom properties and mixins are available for styling:

    Custom property | Description | Default
    ----------------|-------------|----------
    `--paper-spinner-layer-1-color` | Color of the first spinner rotation | `--google-blue-500`
    `--paper-spinner-layer-2-color` | Color of the second spinner rotation | `--google-red-500`
    `--paper-spinner-layer-3-color` | Color of the third spinner rotation | `--google-yellow-500`
    `--paper-spinner-layer-4-color` | Color of the fourth spinner rotation | `--google-green-500`
    `--paper-spinner-stroke-width` | The width of the spinner stroke | 3px

    @group Paper Elements
    @element paper-spinner
    @hero hero.svg
    @demo demo/index.html
    */
    Polymer({
      _template: template$g,

      is: 'paper-spinner',

      behaviors: [PaperSpinnerBehavior]
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    const template$h = html`<iron-iconset-svg name="paper-tabs" size="24">
<svg><defs>
<g id="chevron-left"><path d="M15.41 7.41L14 6l-6 6 6 6 1.41-1.41L10.83 12z"></path></g>
<g id="chevron-right"><path d="M10 6L8.59 7.41 13.17 12l-4.58 4.59L10 18l6-6z"></path></g>
</defs></svg>
</iron-iconset-svg>`;
    document.head.appendChild(template$h.content);

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    /*
    `paper-tab` is styled to look like a tab. It should be used in conjunction with
    `paper-tabs`.

    Example:

        <paper-tabs selected="0">
          <paper-tab>TAB 1</paper-tab>
          <paper-tab>TAB 2</paper-tab>
          <paper-tab>TAB 3</paper-tab>
        </paper-tabs>

    ### Styling

    The following custom properties and mixins are available for styling:

    Custom property | Description | Default
    ----------------|-------------|----------
    `--paper-tab-ink` | Ink color | `--paper-yellow-a100`
    `--paper-tab` | Mixin applied to the tab | `{}`
    `--paper-tab-content` | Mixin applied to the tab content | `{}`
    `--paper-tab-content-focused` | Mixin applied to the tab content when the tab is focused | `{}`
    `--paper-tab-content-unselected` | Mixin applied to the tab content when the tab is not selected | `{}`

    This element applies the mixin `--paper-font-common-base` but does not import
    `paper-styles/typography.html`. In order to apply the `Roboto` font to this
    element, make sure you've imported `paper-styles/typography.html`.
    */
    Polymer({
      _template: html`
    <style>
      :host {
        @apply --layout-inline;
        @apply --layout-center;
        @apply --layout-center-justified;
        @apply --layout-flex-auto;

        position: relative;
        padding: 0 12px;
        overflow: hidden;
        cursor: pointer;
        vertical-align: middle;

        @apply --paper-font-common-base;
        @apply --paper-tab;
      }

      :host(:focus) {
        outline: none;
      }

      :host([link]) {
        padding: 0;
      }

      .tab-content {
        height: 100%;
        transform: translateZ(0);
          -webkit-transform: translateZ(0);
        transition: opacity 0.1s cubic-bezier(0.4, 0.0, 1, 1);
        @apply --layout-horizontal;
        @apply --layout-center-center;
        @apply --layout-flex-auto;
        @apply --paper-tab-content;
      }

      :host(:not(.iron-selected)) > .tab-content {
        opacity: 0.8;

        @apply --paper-tab-content-unselected;
      }

      :host(:focus) .tab-content {
        opacity: 1;
        font-weight: 700;

        @apply --paper-tab-content-focused;
      }

      paper-ripple {
        color: var(--paper-tab-ink, var(--paper-yellow-a100));
      }

      .tab-content > ::slotted(a) {
        @apply --layout-flex-auto;

        height: 100%;
      }
    </style>

    <div class="tab-content">
      <slot></slot>
    </div>
`,

      is: 'paper-tab',

      behaviors: [IronControlState, IronButtonState, PaperRippleBehavior],

      properties: {

        /**
         * If true, the tab will forward keyboard clicks (enter/space) to
         * the first anchor element found in its descendants
         */
        link: {type: Boolean, value: false, reflectToAttribute: true}

      },

      /** @private */
      hostAttributes: {role: 'tab'},

      listeners: {down: '_updateNoink', tap: '_onTap'},

      attached: function() {
        this._updateNoink();
      },

      get _parentNoink() {
        var parent = dom(this).parentNode;
        return !!parent && !!parent.noink;
      },

      _updateNoink: function() {
        this.noink = !!this.noink || !!this._parentNoink;
      },

      _onTap: function(event) {
        if (this.link) {
          var anchor = this.queryEffectiveChildren('a');

          if (!anchor) {
            return;
          }

          // Don't get stuck in a loop delegating
          // the listener from the child anchor
          if (event.target === anchor) {
            return;
          }

          anchor.click();
        }
      }
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    /**
    Material design: [Tabs](https://www.google.com/design/spec/components/tabs.html)

    `paper-tabs` makes it easy to explore and switch between different views or
    functional aspects of an app, or to browse categorized data sets.

    Use `selected` property to get or set the selected tab.

    Example:

        <paper-tabs selected="0">
          <paper-tab>TAB 1</paper-tab>
          <paper-tab>TAB 2</paper-tab>
          <paper-tab>TAB 3</paper-tab>
        </paper-tabs>

    See <a href="?active=paper-tab">paper-tab</a> for more information about
    `paper-tab`.

    A common usage for `paper-tabs` is to use it along with `iron-pages` to switch
    between different views.

        <paper-tabs selected="{{selected}}">
          <paper-tab>Tab 1</paper-tab>
          <paper-tab>Tab 2</paper-tab>
          <paper-tab>Tab 3</paper-tab>
        </paper-tabs>

        <iron-pages selected="{{selected}}">
          <div>Page 1</div>
          <div>Page 2</div>
          <div>Page 3</div>
        </iron-pages>

    To use links in tabs, add `link` attribute to `paper-tab` and put an `<a>`
    element in `paper-tab` with a `tabindex` of -1.

    Example:

    <pre><code>
    &lt;style is="custom-style">
      .link {
        &#64;apply --layout-horizontal;
        &#64;apply --layout-center-center;
      }
    &lt;/style>

    &lt;paper-tabs selected="0">
      &lt;paper-tab link>
        &lt;a href="#link1" class="link" tabindex="-1">TAB ONE&lt;/a>
      &lt;/paper-tab>
      &lt;paper-tab link>
        &lt;a href="#link2" class="link" tabindex="-1">TAB TWO&lt;/a>
      &lt;/paper-tab>
      &lt;paper-tab link>
        &lt;a href="#link3" class="link" tabindex="-1">TAB THREE&lt;/a>
      &lt;/paper-tab>
    &lt;/paper-tabs>
    </code></pre>

    ### Styling

    The following custom properties and mixins are available for styling:

    Custom property | Description | Default
    ----------------|-------------|----------
    `--paper-tabs-selection-bar-color` | Color for the selection bar | `--paper-yellow-a100`
    `--paper-tabs-selection-bar` | Mixin applied to the selection bar | `{}`
    `--paper-tabs` | Mixin applied to the tabs | `{}`
    `--paper-tabs-content` | Mixin applied to the content container of tabs | `{}`
    `--paper-tabs-container` | Mixin applied to the layout container of tabs | `{}`

    @demo demo/index.html
    */
    Polymer({
      _template: html`
    <style>
      :host {
        @apply --layout;
        @apply --layout-center;

        height: 48px;
        font-size: 14px;
        font-weight: 500;
        overflow: hidden;
        -moz-user-select: none;
        -ms-user-select: none;
        -webkit-user-select: none;
        user-select: none;

        /* NOTE: Both values are needed, since some phones require the value to be \`transparent\`. */
        -webkit-tap-highlight-color: rgba(0, 0, 0, 0);
        -webkit-tap-highlight-color: transparent;

        @apply --paper-tabs;
      }

      :host(:dir(rtl)) {
        @apply --layout-horizontal-reverse;
      }

      #tabsContainer {
        position: relative;
        height: 100%;
        white-space: nowrap;
        overflow: hidden;
        @apply --layout-flex-auto;
        @apply --paper-tabs-container;
      }

      #tabsContent {
        height: 100%;
        -moz-flex-basis: auto;
        -ms-flex-basis: auto;
        flex-basis: auto;
        @apply --paper-tabs-content;
      }

      #tabsContent.scrollable {
        position: absolute;
        white-space: nowrap;
      }

      #tabsContent:not(.scrollable),
      #tabsContent.scrollable.fit-container {
        @apply --layout-horizontal;
      }

      #tabsContent.scrollable.fit-container {
        min-width: 100%;
      }

      #tabsContent.scrollable.fit-container > ::slotted(*) {
        /* IE - prevent tabs from compressing when they should scroll. */
        -ms-flex: 1 0 auto;
        -webkit-flex: 1 0 auto;
        flex: 1 0 auto;
      }

      .hidden {
        display: none;
      }

      .not-visible {
        opacity: 0;
        cursor: default;
      }

      paper-icon-button {
        width: 48px;
        height: 48px;
        padding: 12px;
        margin: 0 4px;
      }

      #selectionBar {
        position: absolute;
        height: 0;
        bottom: 0;
        left: 0;
        right: 0;
        border-bottom: 2px solid var(--paper-tabs-selection-bar-color, var(--paper-yellow-a100));
          -webkit-transform: scale(0);
        transform: scale(0);
          -webkit-transform-origin: left center;
        transform-origin: left center;
          transition: -webkit-transform;
        transition: transform;

        @apply --paper-tabs-selection-bar;
      }

      #selectionBar.align-bottom {
        top: 0;
        bottom: auto;
      }

      #selectionBar.expand {
        transition-duration: 0.15s;
        transition-timing-function: cubic-bezier(0.4, 0.0, 1, 1);
      }

      #selectionBar.contract {
        transition-duration: 0.18s;
        transition-timing-function: cubic-bezier(0.0, 0.0, 0.2, 1);
      }

      #tabsContent > ::slotted(:not(#selectionBar)) {
        height: 100%;
      }
    </style>

    <paper-icon-button icon="paper-tabs:chevron-left" class$="[[_computeScrollButtonClass(_leftHidden, scrollable, hideScrollButtons)]]" on-up="_onScrollButtonUp" on-down="_onLeftScrollButtonDown" tabindex="-1"></paper-icon-button>

    <div id="tabsContainer" on-track="_scroll" on-down="_down">
      <div id="tabsContent" class$="[[_computeTabsContentClass(scrollable, fitContainer)]]">
        <div id="selectionBar" class$="[[_computeSelectionBarClass(noBar, alignBottom)]]" on-transitionend="_onBarTransitionEnd"></div>
        <slot></slot>
      </div>
    </div>

    <paper-icon-button icon="paper-tabs:chevron-right" class$="[[_computeScrollButtonClass(_rightHidden, scrollable, hideScrollButtons)]]" on-up="_onScrollButtonUp" on-down="_onRightScrollButtonDown" tabindex="-1"></paper-icon-button>
`,

      is: 'paper-tabs',
      behaviors: [IronResizableBehavior, IronMenubarBehavior],

      properties: {
        /**
         * If true, ink ripple effect is disabled. When this property is changed,
         * all descendant `<paper-tab>` elements have their `noink` property
         * changed to the new value as well.
         */
        noink: {type: Boolean, value: false, observer: '_noinkChanged'},

        /**
         * If true, the bottom bar to indicate the selected tab will not be shown.
         */
        noBar: {type: Boolean, value: false},

        /**
         * If true, the slide effect for the bottom bar is disabled.
         */
        noSlide: {type: Boolean, value: false},

        /**
         * If true, tabs are scrollable and the tab width is based on the label
         * width.
         */
        scrollable: {type: Boolean, value: false},

        /**
         * If true, tabs expand to fit their container. This currently only applies
         * when scrollable is true.
         */
        fitContainer: {type: Boolean, value: false},

        /**
         * If true, dragging on the tabs to scroll is disabled.
         */
        disableDrag: {type: Boolean, value: false},

        /**
         * If true, scroll buttons (left/right arrow) will be hidden for scrollable
         * tabs.
         */
        hideScrollButtons: {type: Boolean, value: false},

        /**
         * If true, the tabs are aligned to bottom (the selection bar appears at the
         * top).
         */
        alignBottom: {type: Boolean, value: false},

        selectable: {type: String, value: 'paper-tab'},

        /**
         * If true, tabs are automatically selected when focused using the
         * keyboard.
         */
        autoselect: {type: Boolean, value: false},

        /**
         * The delay (in milliseconds) between when the user stops interacting
         * with the tabs through the keyboard and when the focused item is
         * automatically selected (if `autoselect` is true).
         */
        autoselectDelay: {type: Number, value: 0},

        _step: {type: Number, value: 10},

        _holdDelay: {type: Number, value: 1},

        _leftHidden: {type: Boolean, value: false},

        _rightHidden: {type: Boolean, value: false},

        _previousTab: {type: Object}
      },

      /** @private */
      hostAttributes: {role: 'tablist'},

      listeners: {
        'iron-resize': '_onTabSizingChanged',
        'iron-items-changed': '_onTabSizingChanged',
        'iron-select': '_onIronSelect',
        'iron-deselect': '_onIronDeselect'
      },

      /**
       * @type {!Object}
       */
      keyBindings: {'left:keyup right:keyup': '_onArrowKeyup'},

      created: function() {
        this._holdJob = null;
        this._pendingActivationItem = undefined;
        this._pendingActivationTimeout = undefined;
        this._bindDelayedActivationHandler =
            this._delayedActivationHandler.bind(this);
        this.addEventListener('blur', this._onBlurCapture.bind(this), true);
      },

      ready: function() {
        this.setScrollDirection('y', this.$.tabsContainer);
      },

      detached: function() {
        this._cancelPendingActivation();
      },

      _noinkChanged: function(noink) {
        var childTabs = dom(this).querySelectorAll('paper-tab');
        childTabs.forEach(
            noink ? this._setNoinkAttribute : this._removeNoinkAttribute);
      },

      _setNoinkAttribute: function(element) {
        element.setAttribute('noink', '');
      },

      _removeNoinkAttribute: function(element) {
        element.removeAttribute('noink');
      },

      _computeScrollButtonClass: function(
          hideThisButton, scrollable, hideScrollButtons) {
        if (!scrollable || hideScrollButtons) {
          return 'hidden';
        }

        if (hideThisButton) {
          return 'not-visible';
        }

        return '';
      },

      _computeTabsContentClass: function(scrollable, fitContainer) {
        return scrollable ? 'scrollable' + (fitContainer ? ' fit-container' : '') :
                            ' fit-container';
      },

      _computeSelectionBarClass: function(noBar, alignBottom) {
        if (noBar) {
          return 'hidden';
        } else if (alignBottom) {
          return 'align-bottom';
        }

        return '';
      },

      // TODO(cdata): Add `track` response back in when gesture lands.

      _onTabSizingChanged: function() {
        this.debounce('_onTabSizingChanged', function() {
          this._scroll();
          this._tabChanged(this.selectedItem);
        }, 10);
      },

      _onIronSelect: function(event) {
        this._tabChanged(event.detail.item, this._previousTab);
        this._previousTab = event.detail.item;
        this.cancelDebouncer('tab-changed');
      },

      _onIronDeselect: function(event) {
        this.debounce('tab-changed', function() {
          this._tabChanged(null, this._previousTab);
          this._previousTab = null;
          // See polymer/polymer#1305
        }, 1);
      },

      _activateHandler: function() {
        // Cancel item activations scheduled by keyboard events when any other
        // action causes an item to be activated (e.g. clicks).
        this._cancelPendingActivation();

        IronMenuBehaviorImpl._activateHandler.apply(this, arguments);
      },

      /**
       * Activates an item after a delay (in milliseconds).
       */
      _scheduleActivation: function(item, delay) {
        this._pendingActivationItem = item;
        this._pendingActivationTimeout =
            this.async(this._bindDelayedActivationHandler, delay);
      },

      /**
       * Activates the last item given to `_scheduleActivation`.
       */
      _delayedActivationHandler: function() {
        var item = this._pendingActivationItem;
        this._pendingActivationItem = undefined;
        this._pendingActivationTimeout = undefined;
        item.fire(this.activateEvent, null, {bubbles: true, cancelable: true});
      },

      /**
       * Cancels a previously scheduled item activation made with
       * `_scheduleActivation`.
       */
      _cancelPendingActivation: function() {
        if (this._pendingActivationTimeout !== undefined) {
          this.cancelAsync(this._pendingActivationTimeout);
          this._pendingActivationItem = undefined;
          this._pendingActivationTimeout = undefined;
        }
      },

      _onArrowKeyup: function(event) {
        if (this.autoselect) {
          this._scheduleActivation(this.focusedItem, this.autoselectDelay);
        }
      },

      _onBlurCapture: function(event) {
        // Cancel a scheduled item activation (if any) when that item is
        // blurred.
        if (event.target === this._pendingActivationItem) {
          this._cancelPendingActivation();
        }
      },

      get _tabContainerScrollSize() {
        return Math.max(
            0, this.$.tabsContainer.scrollWidth - this.$.tabsContainer.offsetWidth);
      },

      _scroll: function(e, detail) {
        if (!this.scrollable) {
          return;
        }

        var ddx = (detail && -detail.ddx) || 0;
        this._affectScroll(ddx);
      },

      _down: function(e) {
        // go one beat async to defeat IronMenuBehavior
        // autorefocus-on-no-selection timeout
        this.async(function() {
          if (this._defaultFocusAsync) {
            this.cancelAsync(this._defaultFocusAsync);
            this._defaultFocusAsync = null;
          }
        }, 1);
      },

      _affectScroll: function(dx) {
        this.$.tabsContainer.scrollLeft += dx;

        var scrollLeft = this.$.tabsContainer.scrollLeft;

        this._leftHidden = scrollLeft === 0;
        this._rightHidden = scrollLeft === this._tabContainerScrollSize;
      },

      _onLeftScrollButtonDown: function() {
        this._scrollToLeft();
        this._holdJob = setInterval(this._scrollToLeft.bind(this), this._holdDelay);
      },

      _onRightScrollButtonDown: function() {
        this._scrollToRight();
        this._holdJob =
            setInterval(this._scrollToRight.bind(this), this._holdDelay);
      },

      _onScrollButtonUp: function() {
        clearInterval(this._holdJob);
        this._holdJob = null;
      },

      _scrollToLeft: function() {
        this._affectScroll(-this._step);
      },

      _scrollToRight: function() {
        this._affectScroll(this._step);
      },

      _tabChanged: function(tab, old) {
        if (!tab) {
          // Remove the bar without animation.
          this.$.selectionBar.classList.remove('expand');
          this.$.selectionBar.classList.remove('contract');
          this._positionBar(0, 0);
          return;
        }

        var r = this.$.tabsContent.getBoundingClientRect();
        var w = r.width;
        var tabRect = tab.getBoundingClientRect();
        var tabOffsetLeft = tabRect.left - r.left;

        this._pos = {
          width: this._calcPercent(tabRect.width, w),
          left: this._calcPercent(tabOffsetLeft, w)
        };

        if (this.noSlide || old == null) {
          // Position the bar without animation.
          this.$.selectionBar.classList.remove('expand');
          this.$.selectionBar.classList.remove('contract');
          this._positionBar(this._pos.width, this._pos.left);
          return;
        }

        var oldRect = old.getBoundingClientRect();
        var oldIndex = this.items.indexOf(old);
        var index = this.items.indexOf(tab);
        var m = 5;

        // bar animation: expand
        this.$.selectionBar.classList.add('expand');

        var moveRight = oldIndex < index;
        var isRTL = this._isRTL;
        if (isRTL) {
          moveRight = !moveRight;
        }

        if (moveRight) {
          this._positionBar(
              this._calcPercent(tabRect.left + tabRect.width - oldRect.left, w) - m,
              this._left);
        } else {
          this._positionBar(
              this._calcPercent(oldRect.left + oldRect.width - tabRect.left, w) - m,
              this._calcPercent(tabOffsetLeft, w) + m);
        }

        if (this.scrollable) {
          this._scrollToSelectedIfNeeded(tabRect.width, tabOffsetLeft);
        }
      },

      _scrollToSelectedIfNeeded: function(tabWidth, tabOffsetLeft) {
        var l = tabOffsetLeft - this.$.tabsContainer.scrollLeft;
        if (l < 0) {
          this.$.tabsContainer.scrollLeft += l;
        } else {
          l += (tabWidth - this.$.tabsContainer.offsetWidth);
          if (l > 0) {
            this.$.tabsContainer.scrollLeft += l;
          }
        }
      },

      _calcPercent: function(w, w0) {
        return 100 * w / w0;
      },

      _positionBar: function(width, left) {
        width = width || 0;
        left = left || 0;

        this._width = width;
        this._left = left;
        this.transform(
            'translateX(' + left + '%) scaleX(' + (width / 100) + ')',
            this.$.selectionBar);
      },

      _onBarTransitionEnd: function(e) {
        var cl = this.$.selectionBar.classList;
        // bar animation: expand -> contract
        if (cl.contains('expand')) {
          cl.remove('expand');
          cl.add('contract');
          this._positionBar(this._pos.width, this._pos.left);
          // bar animation done
        } else if (cl.contains('contract')) {
          cl.remove('contract');
        }
      }
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    // Keeps track of the toast currently opened.
    var currentToast = null;

    /**
    Material design: [Snackbars &
    toasts](https://www.google.com/design/spec/components/snackbars-toasts.html)

    `paper-toast` provides a subtle notification toast. Only one `paper-toast` will
    be visible on screen.

    Use `opened` to show the toast:

    Example:

        <paper-toast text="Hello world!" opened></paper-toast>

    Also `open()` or `show()` can be used to show the toast:

    Example:

        <paper-button on-click="openToast">Open Toast</paper-button>
        <paper-toast id="toast" text="Hello world!"></paper-toast>

        ...

        openToast: function() {
          this.$.toast.open();
        }

    Set `duration` to 0, a negative number or Infinity to persist the toast on
    screen:

    Example:

        <paper-toast text="Terms and conditions" opened duration="0">
          <a href="#">Show more</a>
        </paper-toast>


    ### Styling
    The following custom properties and mixins are available for styling:

    Custom property | Description | Default
    ----------------|-------------|----------
    `--paper-toast-background-color` | The paper-toast background-color | `#323232`
    `--paper-toast-color` | The paper-toast color | `#f1f1f1`

    This element applies the mixin `--paper-font-common-base` but does not import
    `paper-styles/typography.html`. In order to apply the `Roboto` font to this
    element, make sure you've imported `paper-styles/typography.html`.

    @group Paper Elements
    @element paper-toast
    @demo demo/index.html
    @hero hero.svg
    */
    Polymer({
      _template: html`
    <style>
      :host {
        display: block;
        position: fixed;
        background-color: var(--paper-toast-background-color, #323232);
        color: var(--paper-toast-color, #f1f1f1);
        min-height: 48px;
        min-width: 288px;
        padding: 16px 24px;
        box-sizing: border-box;
        box-shadow: 0 2px 5px 0 rgba(0, 0, 0, 0.26);
        border-radius: 2px;
        margin: 12px;
        font-size: 14px;
        cursor: default;
        -webkit-transition: -webkit-transform 0.3s, opacity 0.3s;
        transition: transform 0.3s, opacity 0.3s;
        opacity: 0;
        -webkit-transform: translateY(100px);
        transform: translateY(100px);
        @apply --paper-font-common-base;
      }

      :host(.capsule) {
        border-radius: 24px;
      }

      :host(.fit-bottom) {
        width: 100%;
        min-width: 0;
        border-radius: 0;
        margin: 0;
      }

      :host(.paper-toast-open) {
        opacity: 1;
        -webkit-transform: translateY(0px);
        transform: translateY(0px);
      }
    </style>

    <span id="label">{{text}}</span>
    <slot></slot>
`,

      is: 'paper-toast',
      behaviors: [IronOverlayBehavior],

      properties: {
        /**
         * The element to fit `this` into.
         * Overridden from `Polymer.IronFitBehavior`.
         */
        fitInto: {type: Object, value: window, observer: '_onFitIntoChanged'},

        /**
         * The orientation against which to align the dropdown content
         * horizontally relative to `positionTarget`.
         * Overridden from `Polymer.IronFitBehavior`.
         */
        horizontalAlign: {type: String, value: 'left'},

        /**
         * The orientation against which to align the dropdown content
         * vertically relative to `positionTarget`.
         * Overridden from `Polymer.IronFitBehavior`.
         */
        verticalAlign: {type: String, value: 'bottom'},

        /**
         * The duration in milliseconds to show the toast.
         * Set to `0`, a negative number, or `Infinity`, to disable the
         * toast auto-closing.
         */
        duration: {type: Number, value: 3000},

        /**
         * The text to display in the toast.
         */
        text: {type: String, value: ''},

        /**
         * Overridden from `IronOverlayBehavior`.
         * Set to false to enable closing of the toast by clicking outside it.
         */
        noCancelOnOutsideClick: {type: Boolean, value: true},

        /**
         * Overridden from `IronOverlayBehavior`.
         * Set to true to disable auto-focusing the toast or child nodes with
         * the `autofocus` attribute` when the overlay is opened.
         */
        noAutoFocus: {type: Boolean, value: true}
      },

      listeners: {'transitionend': '__onTransitionEnd'},

      /**
       * Read-only. Deprecated. Use `opened` from `IronOverlayBehavior`.
       * @property visible
       * @deprecated
       */
      get visible() {
        Base._warn('`visible` is deprecated, use `opened` instead');
        return this.opened;
      },

      /**
       * Read-only. Can auto-close if duration is a positive finite number.
       * @property _canAutoClose
       */
      get _canAutoClose() {
        return this.duration > 0 && this.duration !== Infinity;
      },

      created: function() {
        this._autoClose = null;
        IronA11yAnnouncer.requestAvailability();
      },

      /**
       * Show the toast. Without arguments, this is the same as `open()` from
       * `IronOverlayBehavior`.
       * @param {(Object|string)=} properties Properties to be set before opening the toast.
       * e.g. `toast.show('hello')` or `toast.show({text: 'hello', duration: 3000})`
       */
      show: function(properties) {
        if (typeof properties == 'string') {
          properties = {text: properties};
        }
        for (var property in properties) {
          if (property.indexOf('_') === 0) {
            Base._warn(
                'The property "' + property + '" is private and was not set.');
          } else if (property in this) {
            this[property] = properties[property];
          } else {
            Base._warn('The property "' + property + '" is not valid.');
          }
        }
        this.open();
      },

      /**
       * Hide the toast. Same as `close()` from `IronOverlayBehavior`.
       */
      hide: function() {
        this.close();
      },

      /**
       * Called on transitions of the toast, indicating a finished animation
       * @private
       */
      __onTransitionEnd: function(e) {
        // there are different transitions that are happening when opening and
        // closing the toast. The last one so far is for `opacity`.
        // This marks the end of the transition, so we check for this to determine
        // if this is the correct event.
        if (e && e.target === this && e.propertyName === 'opacity') {
          if (this.opened) {
            this._finishRenderOpened();
          } else {
            this._finishRenderClosed();
          }
        }
      },

      /**
       * Overridden from `IronOverlayBehavior`.
       * Called when the value of `opened` changes.
       */
      _openedChanged: function() {
        if (this._autoClose !== null) {
          this.cancelAsync(this._autoClose);
          this._autoClose = null;
        }
        if (this.opened) {
          if (currentToast && currentToast !== this) {
            currentToast.close();
          }
          currentToast = this;
          this.fire('iron-announce', {text: this.text});
          if (this._canAutoClose) {
            this._autoClose = this.async(this.close, this.duration);
          }
        } else if (currentToast === this) {
          currentToast = null;
        }
        IronOverlayBehaviorImpl._openedChanged.apply(this, arguments);
      },

      /**
       * Overridden from `IronOverlayBehavior`.
       */
      _renderOpened: function() {
        this.classList.add('paper-toast-open');
      },

      /**
       * Overridden from `IronOverlayBehavior`.
       */
      _renderClosed: function() {
        this.classList.remove('paper-toast-open');
      },

      /**
       * @private
       */
      _onFitIntoChanged: function(fitInto) {
        this.positionTarget = fitInto;
      }

      /**
       * Fired when `paper-toast` is opened.
       *
       * @event 'iron-announce'
       * @param {{text: string}} detail Contains text that will be announced.
       */
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */

    const template$i = html`

    <style>
      :host {
        display: inline-block;
        @apply --layout-horizontal;
        @apply --layout-center;
        @apply --paper-font-common-base;
      }

      :host([disabled]) {
        pointer-events: none;
      }

      :host(:focus) {
        outline:none;
      }

      .toggle-bar {
        position: absolute;
        height: 100%;
        width: 100%;
        border-radius: 8px;
        pointer-events: none;
        opacity: 0.4;
        transition: background-color linear .08s;
        background-color: var(--paper-toggle-button-unchecked-bar-color, #000000);

        @apply --paper-toggle-button-unchecked-bar;
      }

      .toggle-button {
        position: absolute;
        top: -3px;
        left: 0;
        height: 20px;
        width: 20px;
        border-radius: 50%;
        box-shadow: 0 1px 5px 0 rgba(0, 0, 0, 0.6);
        transition: -webkit-transform linear .08s, background-color linear .08s;
        transition: transform linear .08s, background-color linear .08s;
        will-change: transform;
        background-color: var(--paper-toggle-button-unchecked-button-color, var(--paper-grey-50));

        @apply --paper-toggle-button-unchecked-button;
      }

      .toggle-button.dragging {
        -webkit-transition: none;
        transition: none;
      }

      :host([checked]:not([disabled])) .toggle-bar {
        opacity: 0.5;
        background-color: var(--paper-toggle-button-checked-bar-color, var(--primary-color));

        @apply --paper-toggle-button-checked-bar;
      }

      :host([disabled]) .toggle-bar {
        background-color: #000;
        opacity: 0.12;
      }

      :host([checked]) .toggle-button {
        -webkit-transform: translate(16px, 0);
        transform: translate(16px, 0);
      }

      :host([checked]:not([disabled])) .toggle-button {
        background-color: var(--paper-toggle-button-checked-button-color, var(--primary-color));

        @apply --paper-toggle-button-checked-button;
      }

      :host([disabled]) .toggle-button {
        background-color: #bdbdbd;
        opacity: 1;
      }

      .toggle-ink {
        position: absolute;
        top: -14px;
        left: -14px;
        right: auto;
        bottom: auto;
        width: 48px;
        height: 48px;
        opacity: 0.5;
        pointer-events: none;
        color: var(--paper-toggle-button-unchecked-ink-color, var(--primary-text-color));

        @apply --paper-toggle-button-unchecked-ink;
      }

      :host([checked]) .toggle-ink {
        color: var(--paper-toggle-button-checked-ink-color, var(--primary-color));

        @apply --paper-toggle-button-checked-ink;
      }

      .toggle-container {
        display: inline-block;
        position: relative;
        width: 36px;
        height: 14px;
        /* The toggle button has an absolute position of -3px; The extra 1px
        /* accounts for the toggle button shadow box. */
        margin: 4px 1px;
      }

      .toggle-label {
        position: relative;
        display: inline-block;
        vertical-align: middle;
        padding-left: var(--paper-toggle-button-label-spacing, 8px);
        pointer-events: none;
        color: var(--paper-toggle-button-label-color, var(--primary-text-color));
      }

      /* invalid state */
      :host([invalid]) .toggle-bar {
        background-color: var(--paper-toggle-button-invalid-bar-color, var(--error-color));
      }

      :host([invalid]) .toggle-button {
        background-color: var(--paper-toggle-button-invalid-button-color, var(--error-color));
      }

      :host([invalid]) .toggle-ink {
        color: var(--paper-toggle-button-invalid-ink-color, var(--error-color));
      }
    </style>

    <div class="toggle-container">
      <div id="toggleBar" class="toggle-bar"></div>
      <div id="toggleButton" class="toggle-button"></div>
    </div>

    <div class="toggle-label"><slot></slot></div>

  `;
    template$i.setAttribute('strip-whitespace', '');

    /**
    Material design: [Switch](https://www.google.com/design/spec/components/selection-controls.html#selection-controls-switch)

    `paper-toggle-button` provides a ON/OFF switch that user can toggle the state
    by tapping or by dragging the switch.

    Example:

        <paper-toggle-button></paper-toggle-button>

    ### Styling

    The following custom properties and mixins are available for styling:

    Custom property | Description | Default
    ----------------|-------------|----------
    `--paper-toggle-button-unchecked-bar-color` | Slider color when the input is not checked | `#000000`
    `--paper-toggle-button-unchecked-button-color` | Button color when the input is not checked | `--paper-grey-50`
    `--paper-toggle-button-unchecked-ink-color` | Selected/focus ripple color when the input is not checked | `--dark-primary-color`
    `--paper-toggle-button-checked-bar-color` | Slider button color when the input is checked | `--primary-color`
    `--paper-toggle-button-checked-button-color` | Button color when the input is checked | `--primary-color`
    `--paper-toggle-button-checked-ink-color` | Selected/focus ripple color when the input is checked | `--primary-color`
    `--paper-toggle-button-invalid-bar-color` | Slider button color when the input is invalid | `--error-color`
    `--paper-toggle-button-invalid-button-color` | Button color when the input is invalid | `--error-color`
    `--paper-toggle-button-invalid-ink-color` | Selected/focus ripple color when the input is invalid | `--error-color`
    `--paper-toggle-button-unchecked-bar` | Mixin applied to the slider when the input is not checked | `{}`
    `--paper-toggle-button-unchecked-button` | Mixin applied to the slider button when the input is not checked | `{}`
    `--paper-toggle-button-unchecked-ink` | Mixin applied to the ripple when the input is not checked | `{}`
    `--paper-toggle-button-checked-bar` | Mixin applied to the slider when the input is checked | `{}`
    `--paper-toggle-button-checked-button` | Mixin applied to the slider button when the input is checked | `{}`
    `--paper-toggle-button-checked-ink` | Mixin applied to the ripple when the input is checked | `{}`
    `--paper-toggle-button-label-color` | Label color | `--primary-text-color`
    `--paper-toggle-button-label-spacing` | Spacing between the label and the button | `8px`

    This element applies the mixin `--paper-font-common-base` but does not import `paper-styles/typography.html`.
    In order to apply the `Roboto` font to this element, make sure you've imported `paper-styles/typography.html`.

    @group Paper Elements
    @element paper-toggle-button
    @demo demo/index.html
    */
    Polymer({
      _template: template$i,

      is: 'paper-toggle-button',

      behaviors: [PaperCheckedElementBehavior],

      /** @private */
      hostAttributes: {role: 'button', 'aria-pressed': 'false', tabindex: 0},

      properties: {
          /**
           * Fired when the checked state changes due to user interaction.
           *
           * @event change
           */
          /**
           * Fired when the checked state changes.
           *
           * @event iron-change
           */
      },

      listeners: {track: '_ontrack'},

      attached: function() {
        afterNextRender(this, function() {
          setTouchAction(this, 'pan-y');
        });
      },

      _ontrack: function(event) {
        var track = event.detail;
        if (track.state === 'start') {
          this._trackStart(track);
        } else if (track.state === 'track') {
          this._trackMove(track);
        } else if (track.state === 'end') {
          this._trackEnd(track);
        }
      },

      _trackStart: function(track) {
        this._width = this.$.toggleBar.offsetWidth / 2;
        /*
         * keep an track-only check state to keep the dragging behavior smooth
         * while toggling activations
         */
        this._trackChecked = this.checked;
        this.$.toggleButton.classList.add('dragging');
      },

      _trackMove: function(track) {
        var dx = track.dx;
        this._x = Math.min(
            this._width, Math.max(0, this._trackChecked ? this._width + dx : dx));
        this.translate3d(this._x + 'px', 0, 0, this.$.toggleButton);
        this._userActivate(this._x > (this._width / 2));
      },

      _trackEnd: function(track) {
        this.$.toggleButton.classList.remove('dragging');
        this.transform('', this.$.toggleButton);
      },

      // customize the element's ripple
      _createRipple: function() {
        this._rippleContainer = this.$.toggleButton;
        var ripple = PaperRippleBehavior._createRipple();
        ripple.id = 'ink';
        ripple.setAttribute('recenters', '');
        ripple.classList.add('circle', 'toggle-ink');
        return ripple;
      }

    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
    **This element has been deprecated in favor of
    [app-layout](https://github.com/PolymerElements/app-layout).**

    Material design:
    [Toolbars](https://www.google.com/design/spec/components/toolbars.html)

    `paper-toolbar` is a horizontal bar containing items that can be used for
    label, navigation, search and actions.  The items placed inside the
    `paper-toolbar` are projected into a `class="horizontal center layout"`
    container inside of `paper-toolbar`'s Shadow DOM.  You can use flex attributes
    to control the items' sizing.

    Example:

    ```html
    <paper-toolbar>
      <paper-icon-button slot="top" icon="menu"
    on-tap="menuAction"></paper-icon-button> <div slot="top"
    class="title">Title</div> <paper-icon-button slot="top" icon="more-vert"
    on-tap="moreAction"></paper-icon-button>
    </paper-toolbar>
    ```

    `paper-toolbar` has a standard height, but can made be taller by setting `tall`
    class on the `paper-toolbar`. This will make the toolbar 3x the normal height.

    ```html
    <paper-toolbar class="tall">
      <paper-icon-button slot="top" icon="menu"></paper-icon-button>
    </paper-toolbar>
    ```

    Apply `medium-tall` class to make the toolbar medium tall.  This will make the
    toolbar 2x the normal height.

    ```html
    <paper-toolbar class="medium-tall">
      <paper-icon-button slot="top" icon="menu"></paper-icon-button>
    </paper-toolbar>
    ```

    When `tall`, items can pin to either the top (default), middle or bottom. Use
    `middle` slot for middle content and `bottom` slot for bottom content.

    ```html
    <paper-toolbar class="tall">
      <paper-icon-button slot="top" icon="menu"></paper-icon-button>
      <div slot="middle" class="title">Middle Title</div>
      <div slot="bottom" class="title">Bottom Title</div>
    </paper-toolbar>
    ```

    For `medium-tall` toolbar, the middle and bottom contents overlap and are
    pinned to the bottom. But `middleJustify` and `bottomJustify` attributes are
    still honored separately.

    To make an element completely fit at the bottom of the toolbar, use `fit` along
    with `bottom`.

    ```html
    <paper-toolbar class="tall">
      <div id="progressBar" slot="bottom" class="fit"></div>
    </paper-toolbar>
    ```

    When inside a `paper-header-panel` element with `mode="waterfall-tall"`,
    the class `.animate` is toggled to animate the height change in the toolbar.

    ### Styling

    The following custom properties and mixins are available for styling:

    Custom property | Description | Default
    ----------------|-------------|----------
    `--paper-toolbar-title`      | Mixin applied to the title of the toolbar | `{}`
    `--paper-toolbar-background` | Toolbar background color     | `--primary-color`
    `--paper-toolbar-color`      | Toolbar foreground color     | `--dark-theme-text-color`
    `--paper-toolbar-height`     | Custom height for toolbar    | `64px`
    `--paper-toolbar-sm-height`  | Custom height for small screen toolbar | `56px`
    `--paper-toolbar`            | Mixin applied to the toolbar | `{}`
    `--paper-toolbar-content`    | Mixin applied to the content section of the toolbar | `{}`
    `--paper-toolbar-medium`     | Mixin applied to medium height toolbar | `{}`
    `--paper-toolbar-tall`       | Mixin applied to tall height toolbar | `{}`
    `--paper-toolbar-transition` | Transition applied to the `.animate` class | `height 0.18s ease-in`

    ### Accessibility

    `<paper-toolbar>` has `role="toolbar"` by default. Any elements with the class
    `title` will be used as the label of the toolbar via `aria-labelledby`.

    ### Breaking change in 2.0

    * In Polymer 1.x, default content used to be distribuited automatically to the
    top toolbar. In v2, the you must set `slot="top"` on the default content to
    distribuite the content to the top toolbar.

    @demo demo/index.html
    */
    Polymer({
      _template: html`
    <style>
      :host {
        --calculated-paper-toolbar-height: var(--paper-toolbar-height, 64px);
        --calculated-paper-toolbar-sm-height: var(--paper-toolbar-sm-height, 56px);
        display: block;
        position: relative;
        box-sizing: border-box;
        -moz-box-sizing: border-box;
        height: var(--calculated-paper-toolbar-height);
        background: var(--paper-toolbar-background, var(--primary-color));
        color: var(--paper-toolbar-color, var(--dark-theme-text-color));
        @apply --paper-toolbar;
      }

      :host(.animate) {
        transition: var(--paper-toolbar-transition, height 0.18s ease-in);
      }

      :host(.medium-tall) {
        height: calc(var(--calculated-paper-toolbar-height) * 2);
        @apply --paper-toolbar-medium;
      }

      :host(.tall) {
        height: calc(var(--calculated-paper-toolbar-height) * 3);
        @apply --paper-toolbar-tall;
      }

      .toolbar-tools {
        position: relative;
        height: var(--calculated-paper-toolbar-height);
        padding: 0 16px;
        pointer-events: none;
        @apply --layout-horizontal;
        @apply --layout-center;
        @apply --paper-toolbar-content;
      }

      /*
       * TODO: Where should media query breakpoints live so they can be shared between elements?
       */

      @media (max-width: 600px) {
        :host {
          height: var(--calculated-paper-toolbar-sm-height);
        }

        :host(.medium-tall) {
          height: calc(var(--calculated-paper-toolbar-sm-height) * 2);
        }

        :host(.tall) {
          height: calc(var(--calculated-paper-toolbar-sm-height) * 3);
        }

        .toolbar-tools {
          height: var(--calculated-paper-toolbar-sm-height);
        }
      }

      #topBar {
        position: relative;
      }

      /* middle bar */
      #middleBar {
        position: absolute;
        top: 0;
        right: 0;
        left: 0;
      }

      :host(.tall) #middleBar,
      :host(.medium-tall) #middleBar {
        -webkit-transform: translateY(100%);
        transform: translateY(100%);
      }

      /* bottom bar */
      #bottomBar {
        position: absolute;
        right: 0;
        bottom: 0;
        left: 0;
      }

      /*
       * make elements (e.g. buttons) respond to mouse/touch events
       *
       * \`.toolbar-tools\` disables touch events so multiple toolbars can stack and not
       * absorb events. All children must have pointer events re-enabled to work as
       * expected.
       */
      .toolbar-tools > ::slotted(*:not([disabled])) {
        pointer-events: auto;
      }

      .toolbar-tools > ::slotted(.title) {
        @apply --paper-font-common-base;
        white-space: nowrap;
        overflow: hidden;
        text-overflow: ellipsis;
        font-size: 20px;
        font-weight: 400;
        line-height: 1;
        pointer-events: none;
        @apply --layout-flex;
      }

      .toolbar-tools > ::slotted(.title) {
        margin-left: 56px;
      }

      .toolbar-tools > ::slotted(paper-icon-button + .title) {
        margin-left: 0;
      }

      /**
       * The --paper-toolbar-title mixin is applied here instead of above to
       * fix the issue with margin-left being ignored due to css ordering.
       */
      .toolbar-tools > ::slotted(.title) {
        @apply --paper-toolbar-title;
      }

      .toolbar-tools > ::slotted(paper-icon-button[icon=menu]) {
        margin-right: 24px;
      }

      .toolbar-tools > ::slotted(.fit) {
        position: absolute;
        top: auto;
        right: 0;
        bottom: 0;
        left: 0;
        width: auto;
        margin: 0;
      }

      /* TODO(noms): Until we have a better solution for classes that don't use
       * /deep/ create our own.
       */
      .start-justified {
        @apply --layout-start-justified;
      }

      .center-justified {
        @apply --layout-center-justified;
      }

      .end-justified {
        @apply --layout-end-justified;
      }

      .around-justified {
        @apply --layout-around-justified;
      }

      .justified {
        @apply --layout-justified;
      }
    </style>

    <div id="topBar" class\$="toolbar-tools [[_computeBarExtraClasses(justify)]]">
      <slot name="top"></slot>
    </div>

    <div id="middleBar" class\$="toolbar-tools [[_computeBarExtraClasses(middleJustify)]]">
      <slot name="middle"></slot>
    </div>

    <div id="bottomBar" class\$="toolbar-tools [[_computeBarExtraClasses(bottomJustify)]]">
      <slot name="bottom"></slot>
    </div>
`,

      is: 'paper-toolbar',
      hostAttributes: {'role': 'toolbar'},

      properties: {
        /**
         * Controls how the items are aligned horizontally when they are placed
         * at the bottom.
         * Options are `start`, `center`, `end`, `justified` and `around`.
         */
        bottomJustify: {type: String, value: ''},

        /**
         * Controls how the items are aligned horizontally.
         * Options are `start`, `center`, `end`, `justified` and `around`.
         */
        justify: {type: String, value: ''},

        /**
         * Controls how the items are aligned horizontally when they are placed
         * in the middle.
         * Options are `start`, `center`, `end`, `justified` and `around`.
         */
        middleJustify: {type: String, value: ''}

      },

      ready: function() {
        console.warn(this.is, 'is deprecated. Please use app-layout instead!');
      },

      attached: function() {
        this._observer = this._observe(this);
        this._updateAriaLabelledBy();
      },

      detached: function() {
        if (this._observer) {
          this._observer.disconnect();
        }
      },

      _observe: function(node) {
        var observer = new MutationObserver(function() {
          this._updateAriaLabelledBy();
        }.bind(this));
        observer.observe(node, {childList: true, subtree: true});
        return observer;
      },

      _updateAriaLabelledBy: function() {
        flush();
        var labelledBy = [];
        var contents =
            Array.prototype.slice.call(dom(this.root).querySelectorAll('slot'))
                .concat(Array.prototype.slice.call(
                    dom(this.root).querySelectorAll('content')));

        for (var content, index = 0; content = contents[index]; index++) {
          var nodes = dom(content).getDistributedNodes();
          for (var node, jndex = 0; node = nodes[jndex]; jndex++) {
            if (node.classList && node.classList.contains('title')) {
              if (node.id) {
                labelledBy.push(node.id);
              } else {
                var id = 'paper-toolbar-label-' + Math.floor(Math.random() * 10000);
                node.id = id;
                labelledBy.push(id);
              }
            }
          }
        }
        if (labelledBy.length > 0) {
          this.setAttribute('aria-labelledby', labelledBy.join(' '));
        }
      },

      _computeBarExtraClasses: function(barJustify) {
        if (!barJustify)
          return '';
        return barJustify + (barJustify === 'justified' ? '' : '-justified');
      }
    });

    /**
    @license
    Copyright (c) 2015 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at
    http://polymer.github.io/LICENSE.txt The complete set of authors may be found at
    http://polymer.github.io/AUTHORS.txt The complete set of contributors may be
    found at http://polymer.github.io/CONTRIBUTORS.txt Code distributed by Google as
    part of the polymer project is also subject to an additional IP rights grant
    found at http://polymer.github.io/PATENTS.txt
    */

    /**
    Material design:
    [Tooltips](https://www.google.com/design/spec/components/tooltips.html)
    `<paper-tooltip>` is a label that appears on hover and focus when the user
    hovers over an element with the cursor or with the keyboard. It will be centered
    to an anchor element specified in the `for` attribute, or, if that doesn't
    exist, centered to the parent node containing it.
    Example:
        <div style="display:inline-block">
          <button>Click me!</button>
          <paper-tooltip>Tooltip text</paper-tooltip>
        </div>
        <div>
          <button id="btn">Click me!</button>
          <paper-tooltip for="btn">Tooltip text</paper-tooltip>
        </div>
    The tooltip can be positioned on the top|bottom|left|right of the anchor using
    the `position` attribute. The default position is bottom.
        <paper-tooltip for="btn" position="left">Tooltip text</paper-tooltip>
        <paper-tooltip for="btn" position="top">Tooltip text</paper-tooltip>

    ### Styling
    The following custom properties and mixins are available for styling:
    Custom property | Description | Default
    ----------------|-------------|----------
    `--paper-tooltip-background` | The background color of the tooltip | `#616161`
    `--paper-tooltip-opacity` | The opacity of the tooltip | `0.9`
    `--paper-tooltip-text-color` | The text color of the tooltip | `white`
    `--paper-tooltip` | Mixin applied to the tooltip | `{}`
    `--paper-tooltip-delay-in` | Delay before tooltip starts to fade in | `500`
    `--paper-tooltip-delay-out` | Delay before tooltip starts to fade out | `0`
    `--paper-tooltip-duration-in` | Timing for animation when showing tooltip | `500`
    `--paper-tooltip-duration-out` | Timing for animation when hiding tooltip | `0`
    `--paper-tooltip-animation` | Mixin applied to the tooltip animation | `{}`
    @group Paper Elements
    @element paper-tooltip
    @demo demo/index.html
    */
    Polymer({
      _template: html`
    <style>
      :host {
        display: block;
        position: absolute;
        outline: none;
        z-index: 1002;
        -moz-user-select: none;
        -ms-user-select: none;
        -webkit-user-select: none;
        user-select: none;
        cursor: default;
      }

      #tooltip {
        display: block;
        outline: none;
        @apply --paper-font-common-base;
        font-size: 10px;
        line-height: 1;
        background-color: var(--paper-tooltip-background, #616161);
        color: var(--paper-tooltip-text-color, white);
        padding: 8px;
        border-radius: 2px;
        @apply --paper-tooltip;
      }

      @keyframes keyFrameScaleUp {
        0% {
          transform: scale(0.0);
        }
        100% {
          transform: scale(1.0);
        }
      }

      @keyframes keyFrameScaleDown {
        0% {
          transform: scale(1.0);
        }
        100% {
          transform: scale(0.0);
        }
      }

      @keyframes keyFrameFadeInOpacity {
        0% {
          opacity: 0;
        }
        100% {
          opacity: var(--paper-tooltip-opacity, 0.9);
        }
      }

      @keyframes keyFrameFadeOutOpacity {
        0% {
          opacity: var(--paper-tooltip-opacity, 0.9);
        }
        100% {
          opacity: 0;
        }
      }

      @keyframes keyFrameSlideDownIn {
        0% {
          transform: translateY(-2000px);
          opacity: 0;
        }
        10% {
          opacity: 0.2;
        }
        100% {
          transform: translateY(0);
          opacity: var(--paper-tooltip-opacity, 0.9);
        }
      }

      @keyframes keyFrameSlideDownOut {
        0% {
          transform: translateY(0);
          opacity: var(--paper-tooltip-opacity, 0.9);
        }
        10% {
          opacity: 0.2;
        }
        100% {
          transform: translateY(-2000px);
          opacity: 0;
        }
      }

      .fade-in-animation {
        opacity: 0;
        animation-delay: var(--paper-tooltip-delay-in, 500ms);
        animation-name: keyFrameFadeInOpacity;
        animation-iteration-count: 1;
        animation-timing-function: ease-in;
        animation-duration: var(--paper-tooltip-duration-in, 500ms);
        animation-fill-mode: forwards;
        @apply --paper-tooltip-animation;
      }

      .fade-out-animation {
        opacity: var(--paper-tooltip-opacity, 0.9);
        animation-delay: var(--paper-tooltip-delay-out, 0ms);
        animation-name: keyFrameFadeOutOpacity;
        animation-iteration-count: 1;
        animation-timing-function: ease-in;
        animation-duration: var(--paper-tooltip-duration-out, 500ms);
        animation-fill-mode: forwards;
        @apply --paper-tooltip-animation;
      }

      .scale-up-animation {
        transform: scale(0);
        opacity: var(--paper-tooltip-opacity, 0.9);
        animation-delay: var(--paper-tooltip-delay-in, 500ms);
        animation-name: keyFrameScaleUp;
        animation-iteration-count: 1;
        animation-timing-function: ease-in;
        animation-duration: var(--paper-tooltip-duration-in, 500ms);
        animation-fill-mode: forwards;
        @apply --paper-tooltip-animation;
      }

      .scale-down-animation {
        transform: scale(1);
        opacity: var(--paper-tooltip-opacity, 0.9);
        animation-delay: var(--paper-tooltip-delay-out, 500ms);
        animation-name: keyFrameScaleDown;
        animation-iteration-count: 1;
        animation-timing-function: ease-in;
        animation-duration: var(--paper-tooltip-duration-out, 500ms);
        animation-fill-mode: forwards;
        @apply --paper-tooltip-animation;
      }

      .slide-down-animation {
        transform: translateY(-2000px);
        opacity: 0;
        animation-delay: var(--paper-tooltip-delay-out, 500ms);
        animation-name: keyFrameSlideDownIn;
        animation-iteration-count: 1;
        animation-timing-function: cubic-bezier(0.0, 0.0, 0.2, 1);
        animation-duration: var(--paper-tooltip-duration-out, 500ms);
        animation-fill-mode: forwards;
        @apply --paper-tooltip-animation;
      }

      .slide-down-animation-out {
        transform: translateY(0);
        opacity: var(--paper-tooltip-opacity, 0.9);
        animation-delay: var(--paper-tooltip-delay-out, 500ms);
        animation-name: keyFrameSlideDownOut;
        animation-iteration-count: 1;
        animation-timing-function: cubic-bezier(0.4, 0.0, 1, 1);
        animation-duration: var(--paper-tooltip-duration-out, 500ms);
        animation-fill-mode: forwards;
        @apply --paper-tooltip-animation;
      }

      .cancel-animation {
        animation-delay: -30s !important;
      }

      /* Thanks IE 10. */

      .hidden {
        display: none !important;
      }
    </style>

    <div id="tooltip" class="hidden">
      <slot></slot>
    </div>
`,

      is: 'paper-tooltip',
      hostAttributes: {role: 'tooltip', tabindex: -1},

      properties: {
        /**
         * The id of the element that the tooltip is anchored to. This element
         * must be a sibling of the tooltip. If this property is not set,
         * then the tooltip will be centered to the parent node containing it.
         */
        for: {type: String, observer: '_findTarget'},
        /**
         * Set this to true if you want to manually control when the tooltip
         * is shown or hidden.
         */
        manualMode: {type: Boolean, value: false, observer: '_manualModeChanged'},
        /**
         * Positions the tooltip to the top, right, bottom, left of its content.
         */
        position: {type: String, value: 'bottom'},
        /**
         * If true, no parts of the tooltip will ever be shown offscreen.
         */
        fitToVisibleBounds: {type: Boolean, value: false},
        /**
         * The spacing between the top of the tooltip and the element it is
         * anchored to.
         */
        offset: {type: Number, value: 14},
        /**
         * This property is deprecated, but left over so that it doesn't
         * break exiting code. Please use `offset` instead. If both `offset` and
         * `marginTop` are provided, `marginTop` will be ignored.
         * @deprecated since version 1.0.3
         */
        marginTop: {type: Number, value: 14},
        /**
         * The delay that will be applied before the `entry` animation is
         * played when showing the tooltip.
         */
        animationDelay: {type: Number, value: 500, observer: '_delayChange'},
        /**
         * The animation that will be played on entry.  This replaces the
         * deprecated animationConfig.  Entries here will override the
         * animationConfig settings.  You can enter your own animation
         * by setting it to the css class name.
         */
        animationEntry: {type: String, value: ''},
        /**
         * The animation that will be played on exit.  This replaces the
         * deprecated animationConfig.  Entries here will override the
         * animationConfig settings.  You can enter your own animation
         * by setting it to the css class name.
         */
        animationExit: {type: String, value: ''},
        /**
         * This property is deprecated.  Use --paper-tooltip-animation to change the
         * animation. The entry and exit animations that will be played when showing
         * and hiding the tooltip. If you want to override this, you must ensure
         * that your animationConfig has the exact format below.
         * @deprecated since version
         *
         * The entry and exit animations that will be played when showing and
         * hiding the tooltip. If you want to override this, you must ensure
         * that your animationConfig has the exact format below.
         */
        animationConfig: {
          type: Object,
          value: function() {
            return {
              'entry':
                  [{name: 'fade-in-animation', node: this, timing: {delay: 0}}],
                  'exit': [{name: 'fade-out-animation', node: this}]
            }
          }
        },
        _showing: {type: Boolean, value: false}
      },

      listeners: {
        'webkitAnimationEnd': '_onAnimationEnd',
      },

      /**
       * Returns the target element that this tooltip is anchored to. It is
       * either the element given by the `for` attribute, or the immediate parent
       * of the tooltip.
       *
       * @type {Node}
       */
      get target() {
        var parentNode = dom(this).parentNode;
        // If the parentNode is a document fragment, then we need to use the host.
        var ownerRoot = dom(this).getOwnerRoot();
        var target;
        if (this.for) {
          target = dom(ownerRoot).querySelector('#' + this.for);
        } else {
          target = parentNode.nodeType == Node.DOCUMENT_FRAGMENT_NODE ?
              ownerRoot.host :
              parentNode;
        }
        return target;
      },

      /**
       * @return {void}
       */
      attached: function() {
        this._findTarget();
      },

      /**
       * @return {void}
       */
      detached: function() {
        if (!this.manualMode)
          this._removeListeners();
      },

      /**
       * Replaces Neon-Animation playAnimation - just calls show and hide.
       * @deprecated Use show and hide instead.
       * @param {string} type Either `entry` or `exit`
       */
      playAnimation: function(type) {
        if (type === 'entry') {
          this.show();
        } else if (type === 'exit') {
          this.hide();
        }
      },

      /**
       * Cancels the animation and either fully shows or fully hides tooltip
       */
      cancelAnimation: function() {
        // Short-cut and cancel all animations and hide
        this.$.tooltip.classList.add('cancel-animation');
      },

      /**
       * Shows the tooltip programatically
       * @return {void}
       */
      show: function() {
        // If the tooltip is already showing, there's nothing to do.
        if (this._showing)
          return;

        if (dom(this).textContent.trim() === '') {
          // Check if effective children are also empty
          var allChildrenEmpty = true;
          var effectiveChildren = dom(this).getEffectiveChildNodes();
          for (var i = 0; i < effectiveChildren.length; i++) {
            if (effectiveChildren[i].textContent.trim() !== '') {
              allChildrenEmpty = false;
              break;
            }
          }
          if (allChildrenEmpty) {
            return;
          }
        }

        this._showing = true;
        this.$.tooltip.classList.remove('hidden');
        this.$.tooltip.classList.remove('cancel-animation');
        this.$.tooltip.classList.remove(this._getAnimationType('exit'));
        this.updatePosition();
        this._animationPlaying = true;
        this.$.tooltip.classList.add(this._getAnimationType('entry'));
      },

      /**
       * Hides the tooltip programatically
       * @return {void}
       */
      hide: function() {
        // If the tooltip is already hidden, there's nothing to do.
        if (!this._showing) {
          return;
        }

        // If the entry animation is still playing, don't try to play the exit
        // animation since this will reset the opacity to 1. Just end the animation.
        if (this._animationPlaying) {
          this._showing = false;
          this._cancelAnimation();
          return;
        } else {
          // Play Exit Animation
          this._onAnimationFinish();
        }

        this._showing = false;
        this._animationPlaying = true;
      },

      /**
       * @return {void}
       */
      updatePosition: function() {
        if (!this._target || !this.offsetParent)
          return;
        var offset = this.offset;
        // If a marginTop has been provided by the user (pre 1.0.3), use it.
        if (this.marginTop != 14 && this.offset == 14)
          offset = this.marginTop;
        var parentRect = this.offsetParent.getBoundingClientRect();
        var targetRect = this._target.getBoundingClientRect();
        var thisRect = this.getBoundingClientRect();
        var horizontalCenterOffset = (targetRect.width - thisRect.width) / 2;
        var verticalCenterOffset = (targetRect.height - thisRect.height) / 2;
        var targetLeft = targetRect.left - parentRect.left;
        var targetTop = targetRect.top - parentRect.top;
        var tooltipLeft, tooltipTop;
        switch (this.position) {
          case 'top':
            tooltipLeft = targetLeft + horizontalCenterOffset;
            tooltipTop = targetTop - thisRect.height - offset;
            break;
          case 'bottom':
            tooltipLeft = targetLeft + horizontalCenterOffset;
            tooltipTop = targetTop + targetRect.height + offset;
            break;
          case 'left':
            tooltipLeft = targetLeft - thisRect.width - offset;
            tooltipTop = targetTop + verticalCenterOffset;
            break;
          case 'right':
            tooltipLeft = targetLeft + targetRect.width + offset;
            tooltipTop = targetTop + verticalCenterOffset;
            break;
        }
        // TODO(noms): This should use IronFitBehavior if possible.
        if (this.fitToVisibleBounds) {
          // Clip the left/right side
          if (parentRect.left + tooltipLeft + thisRect.width > window.innerWidth) {
            this.style.right = '0px';
            this.style.left = 'auto';
          } else {
            this.style.left = Math.max(0, tooltipLeft) + 'px';
            this.style.right = 'auto';
          }
          // Clip the top/bottom side.
          if (parentRect.top + tooltipTop + thisRect.height > window.innerHeight) {
            this.style.bottom = (parentRect.height - targetTop + offset) + 'px';
            this.style.top = 'auto';
          } else {
            this.style.top = Math.max(-parentRect.top, tooltipTop) + 'px';
            this.style.bottom = 'auto';
          }
        } else {
          this.style.left = tooltipLeft + 'px';
          this.style.top = tooltipTop + 'px';
        }
      },

      _addListeners: function() {
        if (this._target) {
          this.listen(this._target, 'mouseenter', 'show');
          this.listen(this._target, 'focus', 'show');
          this.listen(this._target, 'mouseleave', 'hide');
          this.listen(this._target, 'blur', 'hide');
          this.listen(this._target, 'tap', 'hide');
        }
        this.listen(this.$.tooltip, 'animationend', '_onAnimationEnd');
        this.listen(this, 'mouseenter', 'hide');
      },

      _findTarget: function() {
        if (!this.manualMode)
          this._removeListeners();
        this._target = this.target;
        if (!this.manualMode)
          this._addListeners();
      },

      _delayChange: function(newValue) {
        // Only Update delay if different value set
        if (newValue !== 500) {
          this.updateStyles({'--paper-tooltip-delay-in': newValue + 'ms'});
        }
      },

      _manualModeChanged: function() {
        if (this.manualMode)
          this._removeListeners();
        else
          this._addListeners();
      },

      _cancelAnimation: function() {
        // Short-cut and cancel all animations and hide
        this.$.tooltip.classList.remove(this._getAnimationType('entry'));
        this.$.tooltip.classList.remove(this._getAnimationType('exit'));
        this.$.tooltip.classList.remove('cancel-animation');
        this.$.tooltip.classList.add('hidden');
      },

      _onAnimationFinish: function() {
        if (this._showing) {
          this.$.tooltip.classList.remove(this._getAnimationType('entry'));
          this.$.tooltip.classList.remove('cancel-animation');
          this.$.tooltip.classList.add(this._getAnimationType('exit'));
        }
      },

      _onAnimationEnd: function() {
        // If no longer showing add class hidden to completely hide tooltip
        this._animationPlaying = false;
        if (!this._showing) {
          this.$.tooltip.classList.remove(this._getAnimationType('exit'));
          this.$.tooltip.classList.add('hidden');
        }
      },

      _getAnimationType: function(type) {
        // These properties have priority over animationConfig values
        if ((type === 'entry') && (this.animationEntry !== '')) {
          return this.animationEntry;
        }
        if ((type === 'exit') && (this.animationExit !== '')) {
          return this.animationExit;
        }
        // If no results then return the legacy value from animationConfig
        if (this.animationConfig[type] &&
            typeof this.animationConfig[type][0].name === 'string') {
          // Checking Timing and Update if necessary - Legacy for animationConfig
          if (this.animationConfig[type][0].timing &&
              this.animationConfig[type][0].timing.delay &&
              this.animationConfig[type][0].timing.delay !== 0) {
            var timingDelay = this.animationConfig[type][0].timing.delay;
            // Has Timing Change - Update CSS
            if (type === 'entry') {
              this.updateStyles({'--paper-tooltip-delay-in': timingDelay + 'ms'});
            } else if (type === 'exit') {
              this.updateStyles({'--paper-tooltip-delay-out': timingDelay + 'ms'});
            }
          }
          return this.animationConfig[type][0].name;
        }
      },

      _removeListeners: function() {
        if (this._target) {
          this.unlisten(this._target, 'mouseenter', 'show');
          this.unlisten(this._target, 'focus', 'show');
          this.unlisten(this._target, 'mouseleave', 'hide');
          this.unlisten(this._target, 'blur', 'hide');
          this.unlisten(this._target, 'tap', 'hide');
        }
        this.unlisten(this.$.tooltip, 'animationend', '_onAnimationEnd');
        this.unlisten(this, 'mouseenter', 'hide');
      }
    });

    /* Copyright 2018 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    class ListenKey {
        constructor(listener) {
            this.listener = listener;
        }
    }
    const hashListeners = new Set();
    const storageListeners = new Set();
    window.addEventListener('hashchange', () => {
        hashListeners.forEach((listenKey) => listenKey.listener());
    });
    // [1]: The event only triggers when another tab edits the storage. Changing a
    // value in current browser tab will NOT trigger below event.
    window.addEventListener('storage', () => {
        storageListeners.forEach((listenKey) => listenKey.listener());
    });
    function addHashListener(fn) {
        const key = new ListenKey(fn);
        hashListeners.add(key);
        return key;
    }
    function addStorageListener(fn) {
        const key = new ListenKey(fn);
        storageListeners.add(key);
        return key;
    }
    function fireStorageChanged() {
        storageListeners.forEach((listenKey) => listenKey.listener());
    }
    function removeHashListenerByKey(key) {
        hashListeners.delete(key);
    }
    function removeStorageListenerByKey(key) {
        storageListeners.delete(key);
    }

    /* Copyright 2016 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    // If true, TensorBoard stores its hash in the URI state.
    // If false, tab switching in TensorBoard will not update location hash,
    // because hash updates interfere with wct_tests.
    let _useHash = false;
    function setUseHash(shouldUseHash) {
        _useHash = shouldUseHash;
    }
    function useHash() {
        return _useHash;
    }
    let _fakeHash = '';
    function setFakeHash(h) {
        _fakeHash = h;
    }
    function getFakeHash() {
        return _fakeHash;
    }

    var tf_globals = /*#__PURE__*/Object.freeze({
        __proto__: null,
        setUseHash: setUseHash,
        useHash: useHash,
        setFakeHash: setFakeHash,
        getFakeHash: getFakeHash
    });

    /* Copyright 2015 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * A keyword that users cannot use, since TensorBoard uses this to store info
     * about the active tab.
     */
    const TAB = '__tab__';
    /**
     * The name of the property for users to set on a Polymer component
     * in order for its stored properties to be stored in the URI unambiguously.
     * (No need to set this if you want multiple instances of the component to
     * share URI state)
     *
     * Example:
     * <my-component disambiguator="0"></my-component>
     *
     * The disambiguator should be set to any unique value so that multiple
     * instances of the component can store properties in URI storage.
     *
     * Because it's hard to dereference this variable in HTML property bindings,
     * it is NOT safe to change the disambiguator string without find+replace
     * across the codebase.
     */
    const DISAMBIGUATOR = 'disambiguator';
    // Keep an up-to-date store of URL params, which iframed plugins can request.
    let urlDict = {};
    function getUrlDict() {
        return urlDict;
    }
    addHashListener(() => {
        urlDict = componentToDict(readComponent());
    });
    const { get: getString, set: setString, getInitializer: getStringInitializer, getObserver: getStringObserver, disposeBinding: disposeStringBinding, } = makeBindings((x) => x, (x) => x);
    const { get: getBoolean, set: setBoolean, getInitializer: getBooleanInitializer, getObserver: getBooleanObserver, disposeBinding: disposeBooleanBinding, } = makeBindings((s) => (s === 'true' ? true : s === 'false' ? false : undefined), (b) => b.toString());
    const { get: getNumber, set: setNumber, getInitializer: getNumberInitializer, getObserver: getNumberObserver, disposeBinding: disposeNumberBinding, } = makeBindings((s) => +s, (n) => n.toString());
    const { get: getObject, set: setObject, getInitializer: getObjectInitializer, getObserver: getObjectObserver, disposeBinding: disposeObjectBinding, } = makeBindings((s) => JSON.parse(atob(s)), (o) => btoa(JSON.stringify(o)));
    function makeBindings(fromString, toString) {
        const hashListeners = [];
        const storageListeners = [];
        function get(key, options = {}) {
            const { defaultValue, useLocalStorage = false } = options;
            const value = useLocalStorage
                ? window.localStorage.getItem(key)
                : componentToDict(readComponent())[key];
            return value == undefined ? lodash.cloneDeep(defaultValue) : fromString(value);
        }
        function set(key, value, options = {}) {
            const { defaultValue, useLocalStorage = false, useLocationReplace = false, } = options;
            const stringValue = toString(value);
            if (useLocalStorage) {
                window.localStorage.setItem(key, stringValue);
                // Because of listeners.ts:[1], we need to manually notify all UI elements
                // listening to storage within the tab of a change.
                fireStorageChanged();
            }
            else if (!lodash.isEqual(value, get(key, { useLocalStorage }))) {
                if (lodash.isEqual(value, defaultValue)) {
                    unsetFromURI(key);
                }
                else {
                    const items = componentToDict(readComponent());
                    items[key] = stringValue;
                    writeComponent(dictToComponent(items), useLocationReplace);
                }
            }
        }
        /**
         * Returns a function that can be used on a `value` declaration to a Polymer
         * property. It updates the `polymerProperty` when storage changes -- i.e.,
         * when `useLocalStorage`, it listens to storage change from another tab and
         * when `useLocalStorage=false`, it listens to hashchange.
         */
        function getInitializer(key, options) {
            const fullOptions = Object.assign({ defaultValue: options.defaultValue, polymerProperty: key, useLocalStorage: false }, options);
            return function () {
                const uriStorageName = getURIStorageName(this, key);
                // setComponentValue will be called every time the underlying storage
                // changes and is responsible for ensuring that new state will propagate
                // to the component with specified property. It is important that this
                // function does not re-assign needlessly, to avoid Polymer observer
                // churn.
                const setComponentValue = () => {
                    const storedValue = get(uriStorageName, fullOptions);
                    const currentValue = this[fullOptions.polymerProperty];
                    if (!lodash.isEqual(storedValue, currentValue)) {
                        this[fullOptions.polymerProperty] = storedValue;
                    }
                };
                const addListener = fullOptions.useLocalStorage
                    ? addStorageListener
                    : addHashListener;
                // TODO(stephanwlee): When using fakeHash, it _should not_ listen to the
                //                    window.hashchange.
                const listenKey = addListener(() => setComponentValue());
                if (fullOptions.useLocalStorage) {
                    storageListeners.push(listenKey);
                }
                else {
                    hashListeners.push(listenKey);
                }
                // Set the value on the property.
                setComponentValue();
                return this[fullOptions.polymerProperty];
            };
        }
        function disposeBinding() {
            hashListeners.forEach((key) => removeHashListenerByKey(key));
            storageListeners.forEach((key) => removeStorageListenerByKey(key));
        }
        function getObserver(key, options) {
            const fullOptions = Object.assign({ defaultValue: options.defaultValue, polymerProperty: key, useLocalStorage: false }, options);
            return function () {
                const uriStorageName = getURIStorageName(this, key);
                const newVal = this[fullOptions.polymerProperty];
                set(uriStorageName, newVal, fullOptions);
            };
        }
        return { get, set, getInitializer, getObserver, disposeBinding };
    }
    function migrateLegacyURLScheme() {
        /**
         * TODO(psybuzz): move to some compatibility file.
         * For each WIT URL param in the legacy scheme, create another URL param
         * in the new scheme. Once WIT migrates to using the new plugin API
         * `getURLPluginData()`, we can update this method to delete the legacy
         * scheme params.
         *
         * This list of params was taken on 1/16/2020. Luckily, WIT only stored
         * strings, booleans.
         */
        const witUrlCompatibilitySet = new Set([
            'examplesPath',
            'hideModelPane2',
            'modelName1',
            'modelName2',
            'inferenceAddress1',
            'inferenceAddress2',
            'modelType',
            'modelVersion1',
            'modelVersion2',
            'modelSignature1',
            'modelSignature2',
            'maxExamples',
            'labelVocabPath',
            'multiClass',
            'sequenceExamples',
            'maxClassesToDisplay',
            'samplingOdds',
            'usePredictApi',
            'predictInputTensor',
            'predictOutputTensor',
        ]);
        const items = componentToDict(readComponent());
        if (items[TAB] === 'whatif') {
            for (let oldName of witUrlCompatibilitySet) {
                if (oldName in items) {
                    const oldValue = items[oldName];
                    items[`p.whatif.${oldName}`] = oldValue;
                }
            }
        }
        writeComponent(dictToComponent(items));
        urlDict = items;
    }
    /**
     * Get a unique storage name for a (Polymer component, propertyName) tuple.
     *
     * DISAMBIGUATOR must be set on the component, if other components use the
     * same propertyName.
     */
    function getURIStorageName(component, propertyName) {
        const d = component[DISAMBIGUATOR];
        const components = d == null ? [propertyName] : [d, propertyName];
        return components.join('.');
    }
    /**
     * Read component from URI (e.g. returns "events&runPrefix=train*").
     */
    function readComponent() {
        return useHash() ? window.location.hash.slice(1) : getFakeHash();
    }
    /**
     * Write component to URI.
     */
    function writeComponent(component, useLocationReplace = false) {
        if (useHash()) {
            if (useLocationReplace) {
                const url = new URL(window.location.href);
                url.hash = component;
                window.history.replaceState(null, '', url.toString());
            }
            else {
                window.location.hash = component;
            }
        }
        else {
            setFakeHash(component);
        }
    }
    /**
     * Convert dictionary of strings into a URI Component.
     * All key value entries get added as key value pairs in the component,
     * with the exception of a key with the TAB value, which if present
     * gets prepended to the URI Component string for backwards compatibility
     * reasons.
     */
    function dictToComponent(items) {
        let component = '';
        // Add the tab name e.g. 'events', 'images', 'histograms' as a prefix
        // for backwards compatbility.
        if (items[TAB] !== undefined) {
            component += items[TAB];
        }
        // Join other strings with &key=value notation
        const nonTab = Object.keys(items)
            .map((key) => [key, items[key]])
            .filter((pair) => pair[0] !== TAB)
            .map((pair) => {
            return encodeURIComponent(pair[0]) + '=' + encodeURIComponent(pair[1]);
        })
            .join('&');
        return nonTab.length > 0 ? component + '&' + nonTab : component;
    }
    /**
     * Convert a URI Component into a dictionary of strings.
     * Component should consist of key-value pairs joined by a delimiter
     * with the exception of the tabName.
     * Returns dict consisting of all key-value pairs and
     * dict[TAB] = tabName
     */
    function componentToDict(component) {
        const items = {};
        const tokens = component.split('&');
        tokens.forEach((token) => {
            const kv = token.split('=');
            // Special backwards compatibility for URI components like #scalars.
            if (kv.length === 1) {
                items[TAB] = kv[0];
            }
            else if (kv.length === 2) {
                items[decodeURIComponent(kv[0])] = decodeURIComponent(kv[1]);
            }
        });
        return items;
    }
    /**
     * Delete a key from the URI.
     */
    function unsetFromURI(key) {
        const items = componentToDict(readComponent());
        delete items[key];
        writeComponent(dictToComponent(items));
    }

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    let TfTagFilterer = class TfTagFilterer extends PolymerElement {
        constructor() {
            super(...arguments);
            this._tagFilter = getStringInitializer('tagFilter', {
                defaultValue: '',
                useLocalStorage: false,
                polymerProperty: '_tagFilter',
            }).call(this);
            this._tagFilterObserver = getStringObserver('tagFilter', {
                defaultValue: '',
                useLocalStorage: false,
                polymerProperty: '_tagFilter',
            });
        }
        _computeTagFilter() {
            return this._tagFilter;
        }
    };
    TfTagFilterer.template = html `
    <paper-input
      no-label-float=""
      label="Filter tags (regular expressions supported)"
      value="{{_tagFilter}}"
      class="search-input"
    >
      <iron-icon prefix="" icon="search" slot="prefix"></iron-icon>
    </paper-input>
    <style>
      :host {
        display: block;
        margin: 10px 5px 10px 10px;
      }
    </style>
  `;
    __decorate([
        property({
            type: String,
            notify: true,
            computed: '_computeTagFilter(_tagFilter)',
        }),
        __metadata("design:type", String)
    ], TfTagFilterer.prototype, "tagFilter", void 0);
    __decorate([
        property({
            type: String,
            observer: '_tagFilterObserver',
        }),
        __metadata("design:type", String)
    ], TfTagFilterer.prototype, "_tagFilter", void 0);
    TfTagFilterer = __decorate([
        customElement('tf-tag-filterer')
    ], TfTagFilterer);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * Interop for Polymer 3 styling
     *
     * From https://polymer-library.polymer-project.org/3.0/docs/devguide/style-shadow-dom:
     *   The following process is a workaround. While Polymer 3.0 does not use
     *   <dom-module> elements for templating, style modules do. The following process
     *   is a workaround for this fact. This process may be updated as required.
     */
    function registerStyleDomModule(args) {
        const { moduleName, styleContent } = args;
        const domModule = document.createElement('dom-module');
        const template = document.createElement('template');
        const styleIncludes = [];
        if (args.styleDependencies) {
            args.styleDependencies.forEach((dep) => {
                const style = document.createElement('style');
                style.setAttribute('include', dep);
                styleIncludes.push(style);
            });
        }
        const style = document.createElement('style');
        Object.assign(style, { textContent: styleContent });
        styleIncludes.forEach((styleElement) => {
            template.content.appendChild(styleElement);
        });
        template.content.appendChild(style);
        domModule.appendChild(template);
        domModule.register(moduleName);
    }

    /* Copyright 2016 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    registerStyleDomModule({
        moduleName: 'dashboard-style',
        styleDependencies: ['iron-flex'],
        styleContent: `
      :host {
        --sidebar-vertical-padding: 15px;
        --sidebar-left-padding: 30px;
      }

      [slot='sidebar'] {
        box-sizing: border-box;
        display: flex;
        flex-direction: column;
        height: 100%;
        margin-right: 10px;
        overflow-x: hidden;
        padding: 5px 0;
        text-overflow: ellipsis;
      }

      .settings {
        min-height: 50px;
        overflow-x: hidden;
        overflow-y: auto;
        will-change: transform;
      }

      .runs-selector {
        display: flex;
        flex-grow: 1;
        min-height: 200px;
      }

      tf-runs-selector {
        flex-grow: 1;
        flex-shrink: 1;
        left: var(--sidebar-left-padding);
        max-height: calc(100% - var(--sidebar-vertical-padding) * 2);
        overflow: hidden;
        position: absolute;
        right: 0;
      }

      .search-input {
        margin: 10px 5px 0 10px;
      }

      .sidebar-section {
        border-top: solid 1px rgba(0, 0, 0, 0.12);
        margin-right: 10px;
        padding: var(--sidebar-vertical-padding) 0
          var(--sidebar-vertical-padding) var(--sidebar-left-padding);
        position: relative;
        overflow: hidden;
      }

      .sidebar-section:first-of-type {
        border: none;
      }

      .sidebar-section paper-button {
        margin: 5px;
      }

      .sidebar-section paper-button:first-of-type {
        margin-left: 0 !important;
      }

      .sidebar-section paper-button:last-of-type {
        margin-right: 0 !important;
      }

      .sidebar-section > :first-child {
        margin-top: 0;
        padding-top: 0;
      }

      .sidebar-section > :last-child {
        margin-bottom: 0;
        padding-bottom: 0;
      }

      .sidebar-section h3 {
        color: var(--paper-grey-800);
        display: block;
        font-size: 14px;
        font-weight: normal;
        margin: 10px 0 5px;
        pointer-events: none;
      }

      paper-checkbox {
        --paper-checkbox-checked-color: var(--tb-ui-dark-accent);
        --paper-checkbox-unchecked-color: var(--tb-ui-dark-accent);
        font-size: 15px;
        margin-top: 5px;
      }
  `,
    });

    /* Copyright 2016 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    const style$1 = document.createElement('style');
    style$1.setAttribute('is', 'custom-style');
    style$1.textContent = `
  :root {
    --tb-orange-weak: #ffa726;
    --tb-orange-strong: #f57c00;
    --tb-orange-dark: #dc7320;
    --tb-grey-darker: #e2e2e2;
    --tb-grey-lighter: #f3f3f3;
    --tb-ui-dark-accent: #757575;
    --tb-ui-light-accent: #e0e0e0;
    --tb-graph-faded: #e0d4b3;
  }
`;
    document.head.appendChild(style$1);

    /* Copyright 2016 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    registerStyleDomModule({
        moduleName: 'scrollbar-style',
        styleContent: `
    .scrollbar::-webkit-scrollbar-track {
      visibility: hidden;
    }

    .scrollbar::-webkit-scrollbar {
      width: 10px;
    }

    .scrollbar::-webkit-scrollbar-thumb {
      border-radius: 10px;
      -webkit-box-shadow: inset 0 0 2px rgba(0, 0, 0, 0.3);
      background-color: var(--paper-grey-500);
      color: var(--paper-grey-900);
    }
    .scrollbar {
      box-sizing: border-box;
    }
  `,
    });

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    let TfDashboardLayout = class TfDashboardLayout extends PolymerElement {
    };
    TfDashboardLayout.template = html `
    <div id="sidebar">
      <slot name="sidebar"></slot>
    </div>

    <div id="center">
      <slot name="center" class="scollbar"></slot>
    </div>
    <style include="scrollbar-style"></style>
    <style>
      :host {
        display: flex;
        flex-direction: row;
        height: 100%;
      }

      #sidebar {
        flex: 0 0 var(--tf-dashboard-layout-sidebar-basis, 25%);
        height: 100%;
        max-width: var(--tf-dashboard-layout-sidebar-max-width, 350px);
        min-width: var(--tf-dashboard-layout-sidebar-min-width, 270px);
        overflow-y: auto;
        text-overflow: ellipsis;
      }

      #center {
        flex-grow: 1;
        flex-shrink: 1;
        height: 100%;
        overflow: hidden;
      }

      ::slotted([slot='center']) {
        contain: strict;
        height: 100%;
        overflow-x: hidden;
        overflow-y: auto;
        width: 100%;
        will-change: transform;
      }

      .tf-graph-dashboard #center {
        background: #fff;
      }
    </style>
  `;
    TfDashboardLayout = __decorate([
        customElement('tf-dashboard-layout')
    ], TfDashboardLayout);

    /* Copyright 2017 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    const LIMIT_LOCAL_STORAGE_KEY = 'TF.TensorBoard.PaginatedView.limit';
    const DEFAULT_LIMIT = 12; // reasonably small and has lots of factors
    let _limit = null; // cached localStorage value
    const listeners = new Set();
    /**
     * Register a listener (nullary function) to be called when the page
     * limit changes.
     */
    function addLimitListener(listener) {
        listeners.add(listener);
    }
    /**
     * Remove a listener registered with `addLimitListener`.
     */
    function removeLimitListener(listener) {
        listeners.delete(listener);
    }
    function getLimit() {
        if (_limit == null) {
            _limit = getNumber(LIMIT_LOCAL_STORAGE_KEY, {
                useLocalStorage: true,
            });
            if (_limit == null || !isFinite(_limit) || _limit <= 0) {
                _limit = DEFAULT_LIMIT;
            }
        }
        return _limit;
    }
    function setLimit(limit) {
        if (limit !== Math.floor(limit)) {
            throw new Error(`limit must be an integer, but got: ${limit}`);
        }
        if (limit <= 0) {
            throw new Error(`limit must be positive, but got: ${limit}`);
        }
        if (limit === _limit) {
            return;
        }
        _limit = limit;
        setNumber(LIMIT_LOCAL_STORAGE_KEY, _limit, {
            useLocalStorage: true,
        });
        listeners.forEach((listener) => {
            listener();
        });
    }

    /* Copyright 2018 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    class ArrayUpdateHelper extends PolymerElement {
        updateArrayProp(prop, value, getKey) {
            let orig = this.get(prop);
            const newVal = value;
            if (!Array.isArray(newVal)) {
                throw RangeError(`Expected new value to '${prop}' to be an array.`);
            }
            // In case using ComplexObserver, the method can be invoked before the prop
            // had a chance to initialize properly.
            if (!Array.isArray(orig)) {
                orig = [];
                this.set(prop, orig);
            }
            const lookup = new Set(newVal.map((item, i) => getKey(item, i)));
            let origInd = 0;
            let newValInd = 0;
            while (origInd < orig.length && newValInd < newVal.length) {
                if (!lookup.has(getKey(orig[origInd], origInd))) {
                    this.splice(prop, origInd, 1);
                    continue;
                }
                else if (getKey(orig[origInd], origInd) == getKey(newVal[newValInd], newValInd)) {
                    // update the element.
                    // TODO(stephanwlee): We may be able to update the original reference of
                    // the `value` by deep-copying the new value over.
                    this.set(`${prop}.${origInd}`, newVal[newValInd]);
                }
                else {
                    this.splice(prop, origInd, 0, newVal[newValInd]);
                }
                newValInd++;
                origInd++;
            }
            if (origInd < orig.length) {
                this.splice(prop, origInd);
            }
            if (newValInd < newVal.length) {
                this.push(prop, ...newVal.slice(newValInd));
            }
        }
    }

    /**
     * The TfDomRepeatBehavior re-implements part of the dom-repeat template.
     * It can be applied to any Polymer components to make it dom-repeat-like.
     * The major discrepency is in syntax when using:
     *
     *   <template is="dom-repeat">
     *     ...
     *   </template>
     *   // versus
     *   <foo-comp>
     *     <template>
     *       ...
     *     </template>
     *   </foo-comp>
     *
     * When implementing the behavior, the Polymer component has to invoke
     * `this.updateDom(newItems)`. Other protected APIs are `setGetItemKey` and
     * `setCacheSize` (please see respective doc for details).
     */
    class TfDomRepeat extends ArrayUpdateHelper {
        constructor() {
            super(...arguments);
            this.as = 'item';
            /**
             * Whether all stamped items are active or not.
             * @protected
             */
            this._contentActive = true;
            this._domBootstrapped = false;
            this._ctor = null;
            /**
             * A list of rendered and mounted items.
             */
            this._renderedItems = [];
            /**
             * A map of stamped child components.
             */
            this._renderedTemplateInst = new Map();
            /**
             * When item is removed, it is placed in a cache and the oldest item gets
             * removed when LRU grows more than size of the 2x_limit.
             */
            this._lruCachedItems = new Map();
            this._cacheSize = 10;
            this._getItemKey = (item) => JSON.stringify(item);
            this._isConnected = false;
        }
        connectedCallback() {
            super.connectedCallback();
            this._isConnected = true;
        }
        /**
         * Sets the size of the DOM cache used for optimizing performance. The
         * default cache size is 10.
         */
        setCacheSize(size) {
            this._cacheSize = size;
        }
        /**
         * Sets getItemKey, a function that requires a unique identifier for an
         * item. It is used to optimize performance of redraws. The default is
         * JSON.stringify(item).
         */
        setGetItemKey(getItemKey) {
            this._getItemKey = getItemKey;
        }
        /**
         * Updates DOM to reflect changes in `items`.
         */
        updateDom(items) {
            this.updateArrayProp('_renderedItems', items, this._getItemKey);
        }
        _ensureTemplatized() {
            // Polymer is not ready (and props/DOM for the components are not
            // populated)
            if (!this.isConnected)
                return false;
            if (!this._ctor) {
                const templateNode = this.querySelector('template');
                this._ctor = templatize(templateNode, this, {
                    parentModel: true,
                    instanceProps: {
                        [this.as]: true,
                        active: this._contentActive,
                    },
                    forwardHostProp: function (prop, value) {
                        this._renderedTemplateInst.forEach((inst) => {
                            inst.forwardHostProp(prop, value);
                        });
                    },
                });
            }
            return true;
        }
        _bootstrapDom() {
            if (!this._ensureTemplatized() || this._domBootstrapped) {
                return;
            }
            const observer = new MutationObserver((mutationsList) => {
                for (const mutation of mutationsList) {
                    if (mutation.type === 'childList') {
                        for (const node of Array.from(mutation.addedNodes)) {
                            if (node instanceof Element) {
                                node.setAttribute('slot', 'items');
                            }
                        }
                    }
                }
            });
            observer.observe(this, { childList: true });
            Array.from(this.children).forEach((child) => {
                this.removeChild(child);
            });
            this._lruCachedItems.clear();
            this._renderedItems.forEach((item, index) => this._insertItem(item, index));
            this._domBootstrapped = true;
        }
        _updateActive() {
            if (!this._domBootstrapped)
                return;
            Array.from(this._renderedTemplateInst.values()).forEach((inst) => {
                inst.notifyPath('active', this._contentActive);
            });
        }
        _updateDom(event // PolymerDeepPropertyChange<T, PolymerSpliceChange<T[]>>
        ) {
            if (!this._domBootstrapped)
                return;
            // These are uninteresting.
            if (event.path == '_renderedItems' ||
                event.path == '_renderedItems.length') {
                return;
            }
            if (event.path === '_renderedItems.splices') {
                const value = event.value; // PolymerSpliceChange<T[]>;
                value.indexSplices.forEach((splice) => {
                    const { index, addedCount, object, removed } = splice;
                    removed.forEach((item) => {
                        this._removeItem(item, this.children[index]);
                    });
                    object
                        .slice(index, index + addedCount)
                        .forEach((item, ind) => this._insertItem(item, index + ind));
                    this._trimCache();
                });
            }
            else {
                // Update the stamped and mounted DOM model by notifying.
                const key = this._getItemKey(event.value);
                if (this._renderedTemplateInst.has(key)) {
                    this._renderedTemplateInst.get(key).notifyPath(this.as, event.value);
                }
                else {
                    console.warn(`Expected '${key}' to exist in the DOM but ` + `could not find one.`);
                }
            }
        }
        _insertItem(item, atIndex) {
            if (!this._ensureTemplatized()) {
                throw new Error('Expected templatized before inserting an item');
            }
            let fragOrEl;
            const key = this._getItemKey(item);
            if (this._lruCachedItems.has(key)) {
                fragOrEl = this._lruCachedItems.get(key);
                this._lruCachedItems.delete(key);
                this._renderedTemplateInst
                    .get(key)
                    .notifyPath('active', this._contentActive);
            }
            else {
                const prop = { [this.as]: item, active: this._contentActive };
                const inst = new this._ctor(prop);
                fragOrEl = inst.root;
                this._renderedTemplateInst.set(key, inst);
            }
            if (this.children[atIndex]) {
                this.insertBefore(fragOrEl, this.children[atIndex]);
            }
            else {
                const els = fragOrEl.nodeType == Node.DOCUMENT_FRAGMENT_NODE
                    ? Array.from(fragOrEl.children)
                    : [fragOrEl];
                els.forEach((el) => el.setAttribute('slot', 'items'));
                this.appendChild(fragOrEl);
            }
        }
        _removeItem(item, node) {
            if (node.parentNode)
                node.parentNode.removeChild(node);
            const key = this._getItemKey(item);
            this._lruCachedItems.set(key, node);
            this._renderedTemplateInst.get(key).notifyPath('active', false);
        }
        _trimCache() {
            while (this._lruCachedItems.size > this._cacheSize) {
                const [firstKey] = this._lruCachedItems.keys();
                this._lruCachedItems.delete(firstKey);
                this._renderedTemplateInst.delete(firstKey);
            }
        }
    }
    __decorate([
        property({ type: String }),
        __metadata("design:type", Object)
    ], TfDomRepeat.prototype, "as", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfDomRepeat.prototype, "items", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfDomRepeat.prototype, "_contentActive", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Object)
    ], TfDomRepeat.prototype, "_domBootstrapped", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfDomRepeat.prototype, "_ctor", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfDomRepeat.prototype, "_renderedItems", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfDomRepeat.prototype, "_renderedTemplateInst", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfDomRepeat.prototype, "_lruCachedItems", void 0);
    __decorate([
        property({ type: Number }),
        __metadata("design:type", Object)
    ], TfDomRepeat.prototype, "_cacheSize", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfDomRepeat.prototype, "_getItemKey", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Object)
    ], TfDomRepeat.prototype, "_isConnected", void 0);
    __decorate([
        observe('_isConnected'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfDomRepeat.prototype, "_bootstrapDom", null);
    __decorate([
        observe('_contentActive'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfDomRepeat.prototype, "_updateActive", null);
    __decorate([
        observe('_renderedItems.*', '_domBootstrapped'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", [Object]),
        __metadata("design:returntype", void 0)
    ], TfDomRepeat.prototype, "_updateDom", null);
    __decorate([
        observe('_cacheSize'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfDomRepeat.prototype, "_trimCache", null);

    /* Copyright 2018 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    let TfCategoryPaginatedView = class TfCategoryPaginatedView extends TfDomRepeat {
        constructor() {
            super(...arguments);
            this.disablePagination = false;
            this.getCategoryItemKey = (item) => JSON.stringify(item);
            this._limit = 12;
            this._activeIndex = 0;
            this._pageInputRawValue = '';
            this._pageInputFocused = false;
        }
        _computeCount() {
            return this.category.items.length;
        }
        get _hasMultiple() {
            return this._count > 1;
        }
        _togglePane() {
            this.opened = !this.opened;
        }
        _changeContentActive(opened) {
            this._contentActive = opened;
        }
        _onPaneRenderedChanged(newRendered, oldRendered) {
            if (newRendered && newRendered !== oldRendered) {
                // Force dom-if render without waiting for one rAF.
                this.$.ifRendered.render();
            }
        }
        _computePaneRendered(category) {
            // Show a category unless it's a search results category where
            // there wasn't actually a search query.
            return !(category.metadata.type === CategoryType.SEARCH_RESULTS &&
                category.name === '');
        }
        get _itemsRendered() {
            return this._paneRendered && this.opened;
        }
        _computeIsSearchResults(type) {
            return type === CategoryType.SEARCH_RESULTS;
        }
        _computeIsInvalidSearchResults(metadata) {
            return (metadata.type === CategoryType.SEARCH_RESULTS && !metadata.validRegex);
        }
        _computeIsUniversalSearchQuery(metadata) {
            return (metadata.type === CategoryType.SEARCH_RESULTS && metadata.universalRegex);
        }
        _isCompositeSearch() {
            const { type, compositeSearch } = this.category.metadata;
            return compositeSearch && type === CategoryType.SEARCH_RESULTS;
        }
        ready() {
            super.ready();
            this.opened = this.initialOpened == null ? true : this.initialOpened;
            this._limitListener = () => {
                this.set('_limit', getLimit());
            };
            addLimitListener(this._limitListener);
            this._limitListener();
        }
        detached() {
            removeLimitListener(this._limitListener);
        }
        _updateRenderedItems() {
            var itemsRendered = this._itemsRendered;
            var limit = this._limit;
            var activeIndex = this._activeIndex;
            var disablePagination = this.disablePagination;
            if (!itemsRendered)
                return;
            const activePageIndex = Math.floor(activeIndex / limit);
            const items = this.category.items || [];
            const domItems = disablePagination
                ? items
                : items.slice(activePageIndex * limit, (activePageIndex + 1) * limit);
            this.updateDom(domItems);
        }
        _limitChanged(limit) {
            this.setCacheSize(limit * 2);
        }
        _getCategoryItemKeyChanged() {
            this.setGetItemKey(this.getCategoryItemKey);
        }
        get _currentPage() {
            var limit = this._limit;
            var activeIndex = this._activeIndex;
            return Math.floor(activeIndex / limit) + 1;
        }
        _computePageCount(_, limit) {
            return this.category ? Math.ceil(this.category.items.length / limit) : 0;
        }
        get _multiplePagesExist() {
            var pageCount = this._pageCount;
            var disablePagination = this.disablePagination;
            return !disablePagination && pageCount > 1;
        }
        get _hasPreviousPage() {
            var currentPage = this._currentPage;
            return currentPage > 1;
        }
        get _hasNextPage() {
            var currentPage = this._currentPage;
            var pageCount = this._pageCount;
            return currentPage < pageCount;
        }
        _computeInputWidth(pageCount) {
            // Add 20px for the +/- arrows added by browsers.
            return `calc(${pageCount.toString().length}em + 20px)`;
        }
        /**
         * Update _activeIndex, maintaining its range invariant.
         */
        _setActiveIndex(index) {
            const maxIndex = (this.category.items || []).length - 1;
            if (index > maxIndex) {
                index = maxIndex;
            }
            if (index < 0) {
                index = 0;
            }
            this.set('_activeIndex', index);
        }
        _clampActiveIndex() {
            this._setActiveIndex(this._activeIndex);
        }
        _performPreviousPage() {
            this._setActiveIndex(this._activeIndex - this._limit);
        }
        _performNextPage() {
            this._setActiveIndex(this._activeIndex + this._limit);
        }
        _computePageInputValue(focused, rawValue, currentPage) {
            return focused ? rawValue : currentPage.toString();
        }
        _handlePageInputEvent(e) {
            this.set('_pageInputRawValue', e.target.value);
            const oneIndexedPage = Number(e.target.value || NaN);
            if (isNaN(oneIndexedPage))
                return;
            const page = Math.max(1, Math.min(oneIndexedPage, this._pageCount)) - 1;
            this._setActiveIndex(this._limit * page);
        }
        _handlePageChangeEvent() {
            // Occurs on Enter, etc. Commit the true state.
            this.set('_pageInputRawValue', this._currentPage.toString());
        }
        _handlePageFocusEvent() {
            // Discard any old (or uninitialized) state before we grant focus.
            this.set('_pageInputRawValue', this._pageInputValue);
            this.set('_pageInputFocused', true);
        }
        _handlePageBlurEvent() {
            this.set('_pageInputFocused', false);
        }
        _updatePageInputValue(newValue) {
            // Force two-way binding.
            const pageInput = this.shadowRoot.querySelector('#page-input input');
            if (pageInput) {
                pageInput.value = newValue;
            }
        }
        _updateInputWidth() {
            this.updateStyles({
                '--tf-category-paginated-view-page-input-width': this._inputWidth,
            });
        }
    };
    TfCategoryPaginatedView.template = html `
    <template is="dom-if" if="[[_paneRendered]]" id="ifRendered">
      <button class="heading" on-tap="_togglePane" open-button$="[[opened]]">
        <span class="name">
          <template is="dom-if" if="[[_isSearchResults]]">
            <template is="dom-if" if="[[_isCompositeSearch(category)]]">
              <span>Tags matching multiple experiments</span>
              <template is="dom-if" if="[[_isInvalidSearchResults]]">
                <span
                  >&nbsp;<strong>(malformed regular expression)</strong></span
                >
              </template>
            </template>
            <template is="dom-if" if="[[!_isCompositeSearch(category)]]">
              <span class="light">Tags matching /</span>
              <span class="category-name" title$="[[category.name]]"
                >[[category.name]]</span
              >
              <span class="light">/</span>
              <template is="dom-if" if="[[_isUniversalSearchQuery]]">
                <span> (all tags)</span>
              </template>
              <template is="dom-if" if="[[_isInvalidSearchResults]]">
                <span> <strong>(malformed regular expression)</strong></span>
              </template>
            </template>
          </template>
          <template is="dom-if" if="[[!_isSearchResults]]">
            <span class="category-name" title$="[[category.name]]"
              >[[category.name]]</span
            >
          </template>
        </span>
        <span class="count">
          <template is="dom-if" if="[[_hasMultiple]]">
            <span>[[_count]]</span>
          </template>
          <iron-icon icon="expand-more" class="expand-arrow"></iron-icon>
        </span>
      </button>
      <!-- TODO(stephanwlee): investigate further. For some reason,
        transitionend that the iron-collapse relies on sometimes does not
        trigger when rendering a chart with a spinner. A toy example cannot
        reproduce this bug. -->
      <iron-collapse opened="[[opened]]" no-animation="">
        <div class="content">
          <span id="top-of-container"></span>
          <template is="dom-if" if="[[_multiplePagesExist]]">
            <div class="big-page-buttons" style="margin-bottom: 10px;">
              <paper-button
                on-tap="_performPreviousPage"
                disabled$="[[!_hasPreviousPage]]"
                >Previous page</paper-button
              >
              <paper-button
                on-tap="_performNextPage"
                disabled$="[[!_hasNextPage]]"
                >Next page</paper-button
              >
            </div>
          </template>

          <div id="items">
            <slot name="items"></slot>
          </div>
          <template is="dom-if" if="[[_multiplePagesExist]]">
            <div id="controls-container">
              <div style="display: inline-block; padding: 0 5px">
                Page
                <paper-input
                  id="page-input"
                  type="number"
                  no-label-float=""
                  min="1"
                  max="[[_pageCount]]"
                  value="[[_pageInputValue]]"
                  on-input="_handlePageInputEvent"
                  on-change="_handlePageChangeEvent"
                  on-focus="_handlePageFocusEvent"
                  on-blur="_handlePageBlurEvent"
                ></paper-input>
                of [[_pageCount]]
              </div>
            </div>

            <div class="big-page-buttons" style="margin-top: 10px;">
              <paper-button
                on-tap="_performPreviousPage"
                disabled$="[[!_hasPreviousPage]]"
                >Previous page</paper-button
              >
              <paper-button
                on-tap="_performNextPage"
                disabled$="[[!_hasNextPage]]"
                >Next page</paper-button
              >
            </div>
          </template>
        </div>
      </iron-collapse>
    </template>
    <style>
      :host {
        display: block;
        margin: 0 5px 1px 10px;
      }

      :host(:first-of-type) {
        margin-top: 10px;
      }

      :host(:last-of-type) {
        margin-bottom: 20px;
      }

      .heading {
        background-color: white;
        border: none;
        cursor: pointer;
        width: 100%;
        font-size: 15px;
        line-height: 1;
        box-shadow: 0 1px 5px rgba(0, 0, 0, 0.2);
        padding: 10px 15px;
        display: flex;
        align-items: center;
        justify-content: space-between;
      }

      .heading::-moz-focus-inner {
        padding: 10px 15px;
      }

      [open-button] {
        border-bottom-left-radius: 0 !important;
        border-bottom-right-radius: 0 !important;
      }

      [open-button] .expand-arrow {
        transform: rotateZ(180deg);
      }

      .name {
        display: inline-flex;
        overflow: hidden;
      }

      .light {
        color: var(--paper-grey-500);
      }

      .category-name {
        white-space: pre;
        overflow: hidden;
        text-overflow: ellipsis;
        padding: 2px 0;
      }

      .count {
        margin: 0 5px;
        font-size: 12px;
        color: var(--paper-grey-500);
        display: flex;
        align-items: center;
        flex: none;
      }

      .heading::-moz-focus-inner {
        padding: 10px 15px;
      }

      .content {
        display: flex;
        flex-direction: column;
        background: white;
        border-bottom-left-radius: 2px;
        border-bottom-right-radius: 2px;
        border-top: none;
        border: 1px solid #dedede;
        padding: 15px;
      }

      .light {
        color: var(--paper-grey-500);
      }

      #controls-container {
        justify-content: center;
        display: flex;
        flex-direction: row;
        flex-grow: 0;
        flex-shrink: 0;
        width: 100%;
      }

      #controls-container paper-button {
        display: inline-block;
      }

      .big-page-buttons {
        display: flex;
      }

      .big-page-buttons paper-button {
        background-color: var(--tb-ui-light-accent);
        color: var(--tb-ui-dark-accent);
        display: inline-block;
        flex-basis: 0;
        flex-grow: 1;
        flex-shrink: 1;
        font-size: 13px;
      }

      .big-page-buttons paper-button[disabled] {
        background: none;
      }

      slot {
        display: flex;
        flex-direction: row;
        flex-wrap: wrap;
      }

      ::slotted([slot='items']) {
        /* Tooltip for descriptions and others break with more strict ones. */
        contain: style;
      }

      #page-input {
        display: inline-block;
        width: var(--tf-category-paginated-view-page-input-width, 100%);
      }
    </style>
  `;
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfCategoryPaginatedView.prototype, "category", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfCategoryPaginatedView.prototype, "initialOpened", void 0);
    __decorate([
        property({
            type: Boolean,
            notify: true,
        }),
        __metadata("design:type", Boolean)
    ], TfCategoryPaginatedView.prototype, "opened", void 0);
    __decorate([
        property({
            type: Boolean,
        }),
        __metadata("design:type", Boolean)
    ], TfCategoryPaginatedView.prototype, "disablePagination", void 0);
    __decorate([
        property({
            type: Number,
            computed: '_computeCount(category.items.*)',
        }),
        __metadata("design:type", Number)
    ], TfCategoryPaginatedView.prototype, "_count", void 0);
    __decorate([
        property({
            type: Boolean,
            computed: '_computePaneRendered(category)',
            observer: '_onPaneRenderedChanged',
        }),
        __metadata("design:type", Boolean)
    ], TfCategoryPaginatedView.prototype, "_paneRendered", void 0);
    __decorate([
        property({
            type: Boolean,
            computed: '_computeIsSearchResults(category.metadata.type)',
        }),
        __metadata("design:type", Boolean)
    ], TfCategoryPaginatedView.prototype, "_isSearchResults", void 0);
    __decorate([
        property({
            type: Boolean,
            computed: '_computeIsInvalidSearchResults(category.metadata)',
        }),
        __metadata("design:type", Boolean)
    ], TfCategoryPaginatedView.prototype, "_isInvalidSearchResults", void 0);
    __decorate([
        property({
            type: Boolean,
            computed: '_computeIsUniversalSearchQuery(category.metadata)',
        }),
        __metadata("design:type", Boolean)
    ], TfCategoryPaginatedView.prototype, "_isUniversalSearchQuery", void 0);
    __decorate([
        property({
            type: Object,
            observer: '_getCategoryItemKeyChanged',
        }),
        __metadata("design:type", Object)
    ], TfCategoryPaginatedView.prototype, "getCategoryItemKey", void 0);
    __decorate([
        property({
            type: Number,
            observer: '_limitChanged',
        }),
        __metadata("design:type", Number)
    ], TfCategoryPaginatedView.prototype, "_limit", void 0);
    __decorate([
        property({
            type: Number,
        }),
        __metadata("design:type", Number)
    ], TfCategoryPaginatedView.prototype, "_activeIndex", void 0);
    __decorate([
        property({
            type: Number,
            computed: '_computePageCount(category.items.*, _limit)',
        }),
        __metadata("design:type", Number)
    ], TfCategoryPaginatedView.prototype, "_pageCount", void 0);
    __decorate([
        property({
            type: String,
            computed: '_computeInputWidth(_pageCount)',
            observer: '_updateInputWidth',
        }),
        __metadata("design:type", String)
    ], TfCategoryPaginatedView.prototype, "_inputWidth", void 0);
    __decorate([
        property({
            type: String,
            computed: '_computePageInputValue(_pageInputFocused, _pageInputRawValue, _currentPage)',
            observer: '_updatePageInputValue',
        }),
        __metadata("design:type", String)
    ], TfCategoryPaginatedView.prototype, "_pageInputValue", void 0);
    __decorate([
        property({
            type: String,
        }),
        __metadata("design:type", String)
    ], TfCategoryPaginatedView.prototype, "_pageInputRawValue", void 0);
    __decorate([
        property({
            type: Boolean,
        }),
        __metadata("design:type", Boolean)
    ], TfCategoryPaginatedView.prototype, "_pageInputFocused", void 0);
    __decorate([
        computed('_count'),
        __metadata("design:type", Boolean),
        __metadata("design:paramtypes", [])
    ], TfCategoryPaginatedView.prototype, "_hasMultiple", null);
    __decorate([
        observe('opened'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", [Boolean]),
        __metadata("design:returntype", void 0)
    ], TfCategoryPaginatedView.prototype, "_changeContentActive", null);
    __decorate([
        computed('opened', '_paneRendered'),
        __metadata("design:type", Boolean),
        __metadata("design:paramtypes", [])
    ], TfCategoryPaginatedView.prototype, "_itemsRendered", null);
    __decorate([
        observe('_itemsRendered', 'category.items.*', '_limit', '_activeIndex', '_pageCount', 'disablePagination'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfCategoryPaginatedView.prototype, "_updateRenderedItems", null);
    __decorate([
        computed('_limit', '_activeIndex'),
        __metadata("design:type", Number),
        __metadata("design:paramtypes", [])
    ], TfCategoryPaginatedView.prototype, "_currentPage", null);
    __decorate([
        computed('_pageCount', 'disablePagination'),
        __metadata("design:type", Boolean),
        __metadata("design:paramtypes", [])
    ], TfCategoryPaginatedView.prototype, "_multiplePagesExist", null);
    __decorate([
        computed('_currentPage'),
        __metadata("design:type", Boolean),
        __metadata("design:paramtypes", [])
    ], TfCategoryPaginatedView.prototype, "_hasPreviousPage", null);
    __decorate([
        computed('_currentPage', '_pageCount'),
        __metadata("design:type", Boolean),
        __metadata("design:paramtypes", [])
    ], TfCategoryPaginatedView.prototype, "_hasNextPage", null);
    __decorate([
        observe('category.items.*'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfCategoryPaginatedView.prototype, "_clampActiveIndex", null);
    TfCategoryPaginatedView = __decorate([
        customElement('tf-category-paginated-view')
    ], TfCategoryPaginatedView);

    /* Copyright 2018 The TensorFlow Authors. All Rights Reserved.
    +
    +Licensed under the Apache License, Version 2.0 (the "License");
    +you may not use this file except in compliance with the License.
    +You may obtain a copy of the License at
    +
    +    http://www.apache.org/licenses/LICENSE-2.0
    +
    +Unless required by applicable law or agreed to in writing, software
    +distributed under the License is distributed on an "AS IS" BASIS,
    +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    +See the License for the specific language governing permissions and
    +limitations under the License.
    +==============================================================================*/
    // A unique reference to a listener for an easier dereferencing.
    class ListenKey$1 {
        constructor(listener) {
            this.listener = listener;
        }
    }
    class BaseStore {
        constructor() {
            this.requestManager = new RequestManager(1 /* simultaneous request */);
            this._listeners = new Set();
            this.initialized = false;
        }
        refresh() {
            return this.load().then(() => {
                this.initialized = true;
            });
        }
        /**
         * Register a listener (nullary function) to be called when new runs are
         * available.
         */
        addListener(listener) {
            const key = new ListenKey$1(listener);
            this._listeners.add(key);
            return key;
        }
        /**
         * Remove a listener registered with `addListener`.
         */
        removeListenerByKey(listenKey) {
            this._listeners.delete(listenKey);
        }
        emitChange() {
            this._listeners.forEach((listenKey) => {
                try {
                    listenKey.listener();
                }
                catch (e) {
                    // ignore exceptions on the listener side.
                }
            });
        }
    }

    /* Copyright 2018 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    class EnvironmentStore extends BaseStore {
        load() {
            const url = getRouter().environment();
            return this.requestManager.request(url).then((result) => {
                const environment = {
                    dataLocation: result.data_location,
                    windowTitle: result.window_title,
                };
                if (result.experiment_name !== undefined) {
                    environment.experimentName = result.experiment_name;
                }
                if (result.experiment_description !== undefined) {
                    environment.experimentDescription = result.experiment_description;
                }
                if (result.creation_time !== undefined) {
                    environment.creationTime = result.creation_time;
                }
                if (lodash.isEqual(this.environment, environment))
                    return;
                this.environment = environment;
                this.emitChange();
            });
        }
        getDataLocation() {
            return this.environment ? this.environment.dataLocation : '';
        }
        getWindowTitle() {
            return this.environment ? this.environment.windowTitle : '';
        }
        getExperimentName() {
            return this.environment ? this.environment.experimentName : '';
        }
        getExperimentDescription() {
            return this.environment ? this.environment.experimentDescription : '';
        }
        getCreationTime() {
            return this.environment ? this.environment.creationTime : null;
        }
    }
    const environmentStore = new EnvironmentStore();

    /* Copyright 2017 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    class RunsStore extends BaseStore {
        constructor() {
            super(...arguments);
            this._runs = [];
        }
        load() {
            const url = getRouter().runs();
            return this.requestManager.request(url).then((newRuns) => {
                if (!lodash.isEqual(this._runs, newRuns)) {
                    this._runs = newRuns;
                    this.emitChange();
                }
            });
        }
        /**
         * Get the current list of runs. If no data is available, this will be
         * an empty array (i.e., there is no distinction between "no runs" and
         * "no runs yet").
         */
        getRuns() {
            return this._runs.slice();
        }
    }
    const runsStore = new RunsStore();

    var version$1 = "5.7.0";

    function ascending(a, b) {
      return a < b ? -1 : a > b ? 1 : a >= b ? 0 : NaN;
    }

    function bisector(compare) {
      if (compare.length === 1) compare = ascendingComparator(compare);
      return {
        left: function(a, x, lo, hi) {
          if (lo == null) lo = 0;
          if (hi == null) hi = a.length;
          while (lo < hi) {
            var mid = lo + hi >>> 1;
            if (compare(a[mid], x) < 0) lo = mid + 1;
            else hi = mid;
          }
          return lo;
        },
        right: function(a, x, lo, hi) {
          if (lo == null) lo = 0;
          if (hi == null) hi = a.length;
          while (lo < hi) {
            var mid = lo + hi >>> 1;
            if (compare(a[mid], x) > 0) hi = mid;
            else lo = mid + 1;
          }
          return lo;
        }
      };
    }

    function ascendingComparator(f) {
      return function(d, x) {
        return ascending(f(d), x);
      };
    }

    var ascendingBisect = bisector(ascending);
    var bisectRight = ascendingBisect.right;
    var bisectLeft = ascendingBisect.left;

    function pairs(array, f) {
      if (f == null) f = pair;
      var i = 0, n = array.length - 1, p = array[0], pairs = new Array(n < 0 ? 0 : n);
      while (i < n) pairs[i] = f(p, p = array[++i]);
      return pairs;
    }

    function pair(a, b) {
      return [a, b];
    }

    function cross(values0, values1, reduce) {
      var n0 = values0.length,
          n1 = values1.length,
          values = new Array(n0 * n1),
          i0,
          i1,
          i,
          value0;

      if (reduce == null) reduce = pair;

      for (i0 = i = 0; i0 < n0; ++i0) {
        for (value0 = values0[i0], i1 = 0; i1 < n1; ++i1, ++i) {
          values[i] = reduce(value0, values1[i1]);
        }
      }

      return values;
    }

    function descending(a, b) {
      return b < a ? -1 : b > a ? 1 : b >= a ? 0 : NaN;
    }

    function number(x) {
      return x === null ? NaN : +x;
    }

    function variance(values, valueof) {
      var n = values.length,
          m = 0,
          i = -1,
          mean = 0,
          value,
          delta,
          sum = 0;

      if (valueof == null) {
        while (++i < n) {
          if (!isNaN(value = number(values[i]))) {
            delta = value - mean;
            mean += delta / ++m;
            sum += delta * (value - mean);
          }
        }
      }

      else {
        while (++i < n) {
          if (!isNaN(value = number(valueof(values[i], i, values)))) {
            delta = value - mean;
            mean += delta / ++m;
            sum += delta * (value - mean);
          }
        }
      }

      if (m > 1) return sum / (m - 1);
    }

    function deviation(array, f) {
      var v = variance(array, f);
      return v ? Math.sqrt(v) : v;
    }

    function extent(values, valueof) {
      var n = values.length,
          i = -1,
          value,
          min,
          max;

      if (valueof == null) {
        while (++i < n) { // Find the first comparable value.
          if ((value = values[i]) != null && value >= value) {
            min = max = value;
            while (++i < n) { // Compare the remaining values.
              if ((value = values[i]) != null) {
                if (min > value) min = value;
                if (max < value) max = value;
              }
            }
          }
        }
      }

      else {
        while (++i < n) { // Find the first comparable value.
          if ((value = valueof(values[i], i, values)) != null && value >= value) {
            min = max = value;
            while (++i < n) { // Compare the remaining values.
              if ((value = valueof(values[i], i, values)) != null) {
                if (min > value) min = value;
                if (max < value) max = value;
              }
            }
          }
        }
      }

      return [min, max];
    }

    var array = Array.prototype;

    var slice = array.slice;
    var map = array.map;

    function constant(x) {
      return function() {
        return x;
      };
    }

    function identity(x) {
      return x;
    }

    function sequence(start, stop, step) {
      start = +start, stop = +stop, step = (n = arguments.length) < 2 ? (stop = start, start = 0, 1) : n < 3 ? 1 : +step;

      var i = -1,
          n = Math.max(0, Math.ceil((stop - start) / step)) | 0,
          range = new Array(n);

      while (++i < n) {
        range[i] = start + i * step;
      }

      return range;
    }

    var e10 = Math.sqrt(50),
        e5 = Math.sqrt(10),
        e2 = Math.sqrt(2);

    function ticks(start, stop, count) {
      var reverse,
          i = -1,
          n,
          ticks,
          step;

      stop = +stop, start = +start, count = +count;
      if (start === stop && count > 0) return [start];
      if (reverse = stop < start) n = start, start = stop, stop = n;
      if ((step = tickIncrement(start, stop, count)) === 0 || !isFinite(step)) return [];

      if (step > 0) {
        start = Math.ceil(start / step);
        stop = Math.floor(stop / step);
        ticks = new Array(n = Math.ceil(stop - start + 1));
        while (++i < n) ticks[i] = (start + i) * step;
      } else {
        start = Math.floor(start * step);
        stop = Math.ceil(stop * step);
        ticks = new Array(n = Math.ceil(start - stop + 1));
        while (++i < n) ticks[i] = (start - i) / step;
      }

      if (reverse) ticks.reverse();

      return ticks;
    }

    function tickIncrement(start, stop, count) {
      var step = (stop - start) / Math.max(0, count),
          power = Math.floor(Math.log(step) / Math.LN10),
          error = step / Math.pow(10, power);
      return power >= 0
          ? (error >= e10 ? 10 : error >= e5 ? 5 : error >= e2 ? 2 : 1) * Math.pow(10, power)
          : -Math.pow(10, -power) / (error >= e10 ? 10 : error >= e5 ? 5 : error >= e2 ? 2 : 1);
    }

    function tickStep(start, stop, count) {
      var step0 = Math.abs(stop - start) / Math.max(0, count),
          step1 = Math.pow(10, Math.floor(Math.log(step0) / Math.LN10)),
          error = step0 / step1;
      if (error >= e10) step1 *= 10;
      else if (error >= e5) step1 *= 5;
      else if (error >= e2) step1 *= 2;
      return stop < start ? -step1 : step1;
    }

    function thresholdSturges(values) {
      return Math.ceil(Math.log(values.length) / Math.LN2) + 1;
    }

    function histogram() {
      var value = identity,
          domain = extent,
          threshold = thresholdSturges;

      function histogram(data) {
        var i,
            n = data.length,
            x,
            values = new Array(n);

        for (i = 0; i < n; ++i) {
          values[i] = value(data[i], i, data);
        }

        var xz = domain(values),
            x0 = xz[0],
            x1 = xz[1],
            tz = threshold(values, x0, x1);

        // Convert number of thresholds into uniform thresholds.
        if (!Array.isArray(tz)) {
          tz = tickStep(x0, x1, tz);
          tz = sequence(Math.ceil(x0 / tz) * tz, x1, tz); // exclusive
        }

        // Remove any thresholds outside the domain.
        var m = tz.length;
        while (tz[0] <= x0) tz.shift(), --m;
        while (tz[m - 1] > x1) tz.pop(), --m;

        var bins = new Array(m + 1),
            bin;

        // Initialize bins.
        for (i = 0; i <= m; ++i) {
          bin = bins[i] = [];
          bin.x0 = i > 0 ? tz[i - 1] : x0;
          bin.x1 = i < m ? tz[i] : x1;
        }

        // Assign data to bins by value, ignoring any outside the domain.
        for (i = 0; i < n; ++i) {
          x = values[i];
          if (x0 <= x && x <= x1) {
            bins[bisectRight(tz, x, 0, m)].push(data[i]);
          }
        }

        return bins;
      }

      histogram.value = function(_) {
        return arguments.length ? (value = typeof _ === "function" ? _ : constant(_), histogram) : value;
      };

      histogram.domain = function(_) {
        return arguments.length ? (domain = typeof _ === "function" ? _ : constant([_[0], _[1]]), histogram) : domain;
      };

      histogram.thresholds = function(_) {
        return arguments.length ? (threshold = typeof _ === "function" ? _ : Array.isArray(_) ? constant(slice.call(_)) : constant(_), histogram) : threshold;
      };

      return histogram;
    }

    function threshold(values, p, valueof) {
      if (valueof == null) valueof = number;
      if (!(n = values.length)) return;
      if ((p = +p) <= 0 || n < 2) return +valueof(values[0], 0, values);
      if (p >= 1) return +valueof(values[n - 1], n - 1, values);
      var n,
          i = (n - 1) * p,
          i0 = Math.floor(i),
          value0 = +valueof(values[i0], i0, values),
          value1 = +valueof(values[i0 + 1], i0 + 1, values);
      return value0 + (value1 - value0) * (i - i0);
    }

    function freedmanDiaconis(values, min, max) {
      values = map.call(values, number).sort(ascending);
      return Math.ceil((max - min) / (2 * (threshold(values, 0.75) - threshold(values, 0.25)) * Math.pow(values.length, -1 / 3)));
    }

    function scott(values, min, max) {
      return Math.ceil((max - min) / (3.5 * deviation(values) * Math.pow(values.length, -1 / 3)));
    }

    function max(values, valueof) {
      var n = values.length,
          i = -1,
          value,
          max;

      if (valueof == null) {
        while (++i < n) { // Find the first comparable value.
          if ((value = values[i]) != null && value >= value) {
            max = value;
            while (++i < n) { // Compare the remaining values.
              if ((value = values[i]) != null && value > max) {
                max = value;
              }
            }
          }
        }
      }

      else {
        while (++i < n) { // Find the first comparable value.
          if ((value = valueof(values[i], i, values)) != null && value >= value) {
            max = value;
            while (++i < n) { // Compare the remaining values.
              if ((value = valueof(values[i], i, values)) != null && value > max) {
                max = value;
              }
            }
          }
        }
      }

      return max;
    }

    function mean(values, valueof) {
      var n = values.length,
          m = n,
          i = -1,
          value,
          sum = 0;

      if (valueof == null) {
        while (++i < n) {
          if (!isNaN(value = number(values[i]))) sum += value;
          else --m;
        }
      }

      else {
        while (++i < n) {
          if (!isNaN(value = number(valueof(values[i], i, values)))) sum += value;
          else --m;
        }
      }

      if (m) return sum / m;
    }

    function median(values, valueof) {
      var n = values.length,
          i = -1,
          value,
          numbers = [];

      if (valueof == null) {
        while (++i < n) {
          if (!isNaN(value = number(values[i]))) {
            numbers.push(value);
          }
        }
      }

      else {
        while (++i < n) {
          if (!isNaN(value = number(valueof(values[i], i, values)))) {
            numbers.push(value);
          }
        }
      }

      return threshold(numbers.sort(ascending), 0.5);
    }

    function merge(arrays) {
      var n = arrays.length,
          m,
          i = -1,
          j = 0,
          merged,
          array;

      while (++i < n) j += arrays[i].length;
      merged = new Array(j);

      while (--n >= 0) {
        array = arrays[n];
        m = array.length;
        while (--m >= 0) {
          merged[--j] = array[m];
        }
      }

      return merged;
    }

    function min(values, valueof) {
      var n = values.length,
          i = -1,
          value,
          min;

      if (valueof == null) {
        while (++i < n) { // Find the first comparable value.
          if ((value = values[i]) != null && value >= value) {
            min = value;
            while (++i < n) { // Compare the remaining values.
              if ((value = values[i]) != null && min > value) {
                min = value;
              }
            }
          }
        }
      }

      else {
        while (++i < n) { // Find the first comparable value.
          if ((value = valueof(values[i], i, values)) != null && value >= value) {
            min = value;
            while (++i < n) { // Compare the remaining values.
              if ((value = valueof(values[i], i, values)) != null && min > value) {
                min = value;
              }
            }
          }
        }
      }

      return min;
    }

    function permute(array, indexes) {
      var i = indexes.length, permutes = new Array(i);
      while (i--) permutes[i] = array[indexes[i]];
      return permutes;
    }

    function scan(values, compare) {
      if (!(n = values.length)) return;
      var n,
          i = 0,
          j = 0,
          xi,
          xj = values[j];

      if (compare == null) compare = ascending;

      while (++i < n) {
        if (compare(xi = values[i], xj) < 0 || compare(xj, xj) !== 0) {
          xj = xi, j = i;
        }
      }

      if (compare(xj, xj) === 0) return j;
    }

    function shuffle(array, i0, i1) {
      var m = (i1 == null ? array.length : i1) - (i0 = i0 == null ? 0 : +i0),
          t,
          i;

      while (m) {
        i = Math.random() * m-- | 0;
        t = array[m + i0];
        array[m + i0] = array[i + i0];
        array[i + i0] = t;
      }

      return array;
    }

    function sum(values, valueof) {
      var n = values.length,
          i = -1,
          value,
          sum = 0;

      if (valueof == null) {
        while (++i < n) {
          if (value = +values[i]) sum += value; // Note: zero and null are equivalent.
        }
      }

      else {
        while (++i < n) {
          if (value = +valueof(values[i], i, values)) sum += value;
        }
      }

      return sum;
    }

    function transpose(matrix) {
      if (!(n = matrix.length)) return [];
      for (var i = -1, m = min(matrix, length), transpose = new Array(m); ++i < m;) {
        for (var j = -1, n, row = transpose[i] = new Array(n); ++j < n;) {
          row[j] = matrix[j][i];
        }
      }
      return transpose;
    }

    function length(d) {
      return d.length;
    }

    function zip() {
      return transpose(arguments);
    }

    var slice$1 = Array.prototype.slice;

    function identity$1(x) {
      return x;
    }

    var top = 1,
        right = 2,
        bottom = 3,
        left = 4,
        epsilon = 1e-6;

    function translateX(x) {
      return "translate(" + (x + 0.5) + ",0)";
    }

    function translateY(y) {
      return "translate(0," + (y + 0.5) + ")";
    }

    function number$1(scale) {
      return function(d) {
        return +scale(d);
      };
    }

    function center(scale) {
      var offset = Math.max(0, scale.bandwidth() - 1) / 2; // Adjust for 0.5px offset.
      if (scale.round()) offset = Math.round(offset);
      return function(d) {
        return +scale(d) + offset;
      };
    }

    function entering() {
      return !this.__axis;
    }

    function axis(orient, scale) {
      var tickArguments = [],
          tickValues = null,
          tickFormat = null,
          tickSizeInner = 6,
          tickSizeOuter = 6,
          tickPadding = 3,
          k = orient === top || orient === left ? -1 : 1,
          x = orient === left || orient === right ? "x" : "y",
          transform = orient === top || orient === bottom ? translateX : translateY;

      function axis(context) {
        var values = tickValues == null ? (scale.ticks ? scale.ticks.apply(scale, tickArguments) : scale.domain()) : tickValues,
            format = tickFormat == null ? (scale.tickFormat ? scale.tickFormat.apply(scale, tickArguments) : identity$1) : tickFormat,
            spacing = Math.max(tickSizeInner, 0) + tickPadding,
            range = scale.range(),
            range0 = +range[0] + 0.5,
            range1 = +range[range.length - 1] + 0.5,
            position = (scale.bandwidth ? center : number$1)(scale.copy()),
            selection = context.selection ? context.selection() : context,
            path = selection.selectAll(".domain").data([null]),
            tick = selection.selectAll(".tick").data(values, scale).order(),
            tickExit = tick.exit(),
            tickEnter = tick.enter().append("g").attr("class", "tick"),
            line = tick.select("line"),
            text = tick.select("text");

        path = path.merge(path.enter().insert("path", ".tick")
            .attr("class", "domain")
            .attr("stroke", "currentColor"));

        tick = tick.merge(tickEnter);

        line = line.merge(tickEnter.append("line")
            .attr("stroke", "currentColor")
            .attr(x + "2", k * tickSizeInner));

        text = text.merge(tickEnter.append("text")
            .attr("fill", "currentColor")
            .attr(x, k * spacing)
            .attr("dy", orient === top ? "0em" : orient === bottom ? "0.71em" : "0.32em"));

        if (context !== selection) {
          path = path.transition(context);
          tick = tick.transition(context);
          line = line.transition(context);
          text = text.transition(context);

          tickExit = tickExit.transition(context)
              .attr("opacity", epsilon)
              .attr("transform", function(d) { return isFinite(d = position(d)) ? transform(d) : this.getAttribute("transform"); });

          tickEnter
              .attr("opacity", epsilon)
              .attr("transform", function(d) { var p = this.parentNode.__axis; return transform(p && isFinite(p = p(d)) ? p : position(d)); });
        }

        tickExit.remove();

        path
            .attr("d", orient === left || orient == right
                ? (tickSizeOuter ? "M" + k * tickSizeOuter + "," + range0 + "H0.5V" + range1 + "H" + k * tickSizeOuter : "M0.5," + range0 + "V" + range1)
                : (tickSizeOuter ? "M" + range0 + "," + k * tickSizeOuter + "V0.5H" + range1 + "V" + k * tickSizeOuter : "M" + range0 + ",0.5H" + range1));

        tick
            .attr("opacity", 1)
            .attr("transform", function(d) { return transform(position(d)); });

        line
            .attr(x + "2", k * tickSizeInner);

        text
            .attr(x, k * spacing)
            .text(format);

        selection.filter(entering)
            .attr("fill", "none")
            .attr("font-size", 10)
            .attr("font-family", "sans-serif")
            .attr("text-anchor", orient === right ? "start" : orient === left ? "end" : "middle");

        selection
            .each(function() { this.__axis = position; });
      }

      axis.scale = function(_) {
        return arguments.length ? (scale = _, axis) : scale;
      };

      axis.ticks = function() {
        return tickArguments = slice$1.call(arguments), axis;
      };

      axis.tickArguments = function(_) {
        return arguments.length ? (tickArguments = _ == null ? [] : slice$1.call(_), axis) : tickArguments.slice();
      };

      axis.tickValues = function(_) {
        return arguments.length ? (tickValues = _ == null ? null : slice$1.call(_), axis) : tickValues && tickValues.slice();
      };

      axis.tickFormat = function(_) {
        return arguments.length ? (tickFormat = _, axis) : tickFormat;
      };

      axis.tickSize = function(_) {
        return arguments.length ? (tickSizeInner = tickSizeOuter = +_, axis) : tickSizeInner;
      };

      axis.tickSizeInner = function(_) {
        return arguments.length ? (tickSizeInner = +_, axis) : tickSizeInner;
      };

      axis.tickSizeOuter = function(_) {
        return arguments.length ? (tickSizeOuter = +_, axis) : tickSizeOuter;
      };

      axis.tickPadding = function(_) {
        return arguments.length ? (tickPadding = +_, axis) : tickPadding;
      };

      return axis;
    }

    function axisTop(scale) {
      return axis(top, scale);
    }

    function axisRight(scale) {
      return axis(right, scale);
    }

    function axisBottom(scale) {
      return axis(bottom, scale);
    }

    function axisLeft(scale) {
      return axis(left, scale);
    }

    var noop = {value: function() {}};

    function dispatch() {
      for (var i = 0, n = arguments.length, _ = {}, t; i < n; ++i) {
        if (!(t = arguments[i] + "") || (t in _) || /[\s.]/.test(t)) throw new Error("illegal type: " + t);
        _[t] = [];
      }
      return new Dispatch(_);
    }

    function Dispatch(_) {
      this._ = _;
    }

    function parseTypenames(typenames, types) {
      return typenames.trim().split(/^|\s+/).map(function(t) {
        var name = "", i = t.indexOf(".");
        if (i >= 0) name = t.slice(i + 1), t = t.slice(0, i);
        if (t && !types.hasOwnProperty(t)) throw new Error("unknown type: " + t);
        return {type: t, name: name};
      });
    }

    Dispatch.prototype = dispatch.prototype = {
      constructor: Dispatch,
      on: function(typename, callback) {
        var _ = this._,
            T = parseTypenames(typename + "", _),
            t,
            i = -1,
            n = T.length;

        // If no callback was specified, return the callback of the given type and name.
        if (arguments.length < 2) {
          while (++i < n) if ((t = (typename = T[i]).type) && (t = get$1(_[t], typename.name))) return t;
          return;
        }

        // If a type was specified, set the callback for the given type and name.
        // Otherwise, if a null callback was specified, remove callbacks of the given name.
        if (callback != null && typeof callback !== "function") throw new Error("invalid callback: " + callback);
        while (++i < n) {
          if (t = (typename = T[i]).type) _[t] = set$1(_[t], typename.name, callback);
          else if (callback == null) for (t in _) _[t] = set$1(_[t], typename.name, null);
        }

        return this;
      },
      copy: function() {
        var copy = {}, _ = this._;
        for (var t in _) copy[t] = _[t].slice();
        return new Dispatch(copy);
      },
      call: function(type, that) {
        if ((n = arguments.length - 2) > 0) for (var args = new Array(n), i = 0, n, t; i < n; ++i) args[i] = arguments[i + 2];
        if (!this._.hasOwnProperty(type)) throw new Error("unknown type: " + type);
        for (t = this._[type], i = 0, n = t.length; i < n; ++i) t[i].value.apply(that, args);
      },
      apply: function(type, that, args) {
        if (!this._.hasOwnProperty(type)) throw new Error("unknown type: " + type);
        for (var t = this._[type], i = 0, n = t.length; i < n; ++i) t[i].value.apply(that, args);
      }
    };

    function get$1(type, name) {
      for (var i = 0, n = type.length, c; i < n; ++i) {
        if ((c = type[i]).name === name) {
          return c.value;
        }
      }
    }

    function set$1(type, name, callback) {
      for (var i = 0, n = type.length; i < n; ++i) {
        if (type[i].name === name) {
          type[i] = noop, type = type.slice(0, i).concat(type.slice(i + 1));
          break;
        }
      }
      if (callback != null) type.push({name: name, value: callback});
      return type;
    }

    var xhtml = "http://www.w3.org/1999/xhtml";

    var namespaces = {
      svg: "http://www.w3.org/2000/svg",
      xhtml: xhtml,
      xlink: "http://www.w3.org/1999/xlink",
      xml: "http://www.w3.org/XML/1998/namespace",
      xmlns: "http://www.w3.org/2000/xmlns/"
    };

    function namespace(name) {
      var prefix = name += "", i = prefix.indexOf(":");
      if (i >= 0 && (prefix = name.slice(0, i)) !== "xmlns") name = name.slice(i + 1);
      return namespaces.hasOwnProperty(prefix) ? {space: namespaces[prefix], local: name} : name;
    }

    function creatorInherit(name) {
      return function() {
        var document = this.ownerDocument,
            uri = this.namespaceURI;
        return uri === xhtml && document.documentElement.namespaceURI === xhtml
            ? document.createElement(name)
            : document.createElementNS(uri, name);
      };
    }

    function creatorFixed(fullname) {
      return function() {
        return this.ownerDocument.createElementNS(fullname.space, fullname.local);
      };
    }

    function creator(name) {
      var fullname = namespace(name);
      return (fullname.local
          ? creatorFixed
          : creatorInherit)(fullname);
    }

    function none() {}

    function selector(selector) {
      return selector == null ? none : function() {
        return this.querySelector(selector);
      };
    }

    function selection_select(select) {
      if (typeof select !== "function") select = selector(select);

      for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, subgroup = subgroups[j] = new Array(n), node, subnode, i = 0; i < n; ++i) {
          if ((node = group[i]) && (subnode = select.call(node, node.__data__, i, group))) {
            if ("__data__" in node) subnode.__data__ = node.__data__;
            subgroup[i] = subnode;
          }
        }
      }

      return new Selection(subgroups, this._parents);
    }

    function empty() {
      return [];
    }

    function selectorAll(selector) {
      return selector == null ? empty : function() {
        return this.querySelectorAll(selector);
      };
    }

    function selection_selectAll(select) {
      if (typeof select !== "function") select = selectorAll(select);

      for (var groups = this._groups, m = groups.length, subgroups = [], parents = [], j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, node, i = 0; i < n; ++i) {
          if (node = group[i]) {
            subgroups.push(select.call(node, node.__data__, i, group));
            parents.push(node);
          }
        }
      }

      return new Selection(subgroups, parents);
    }

    function matcher(selector) {
      return function() {
        return this.matches(selector);
      };
    }

    function selection_filter(match) {
      if (typeof match !== "function") match = matcher(match);

      for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, subgroup = subgroups[j] = [], node, i = 0; i < n; ++i) {
          if ((node = group[i]) && match.call(node, node.__data__, i, group)) {
            subgroup.push(node);
          }
        }
      }

      return new Selection(subgroups, this._parents);
    }

    function sparse(update) {
      return new Array(update.length);
    }

    function selection_enter() {
      return new Selection(this._enter || this._groups.map(sparse), this._parents);
    }

    function EnterNode(parent, datum) {
      this.ownerDocument = parent.ownerDocument;
      this.namespaceURI = parent.namespaceURI;
      this._next = null;
      this._parent = parent;
      this.__data__ = datum;
    }

    EnterNode.prototype = {
      constructor: EnterNode,
      appendChild: function(child) { return this._parent.insertBefore(child, this._next); },
      insertBefore: function(child, next) { return this._parent.insertBefore(child, next); },
      querySelector: function(selector) { return this._parent.querySelector(selector); },
      querySelectorAll: function(selector) { return this._parent.querySelectorAll(selector); }
    };

    function constant$1(x) {
      return function() {
        return x;
      };
    }

    var keyPrefix = "$"; // Protect against keys like “__proto__”.

    function bindIndex(parent, group, enter, update, exit, data) {
      var i = 0,
          node,
          groupLength = group.length,
          dataLength = data.length;

      // Put any non-null nodes that fit into update.
      // Put any null nodes into enter.
      // Put any remaining data into enter.
      for (; i < dataLength; ++i) {
        if (node = group[i]) {
          node.__data__ = data[i];
          update[i] = node;
        } else {
          enter[i] = new EnterNode(parent, data[i]);
        }
      }

      // Put any non-null nodes that don’t fit into exit.
      for (; i < groupLength; ++i) {
        if (node = group[i]) {
          exit[i] = node;
        }
      }
    }

    function bindKey(parent, group, enter, update, exit, data, key) {
      var i,
          node,
          nodeByKeyValue = {},
          groupLength = group.length,
          dataLength = data.length,
          keyValues = new Array(groupLength),
          keyValue;

      // Compute the key for each node.
      // If multiple nodes have the same key, the duplicates are added to exit.
      for (i = 0; i < groupLength; ++i) {
        if (node = group[i]) {
          keyValues[i] = keyValue = keyPrefix + key.call(node, node.__data__, i, group);
          if (keyValue in nodeByKeyValue) {
            exit[i] = node;
          } else {
            nodeByKeyValue[keyValue] = node;
          }
        }
      }

      // Compute the key for each datum.
      // If there a node associated with this key, join and add it to update.
      // If there is not (or the key is a duplicate), add it to enter.
      for (i = 0; i < dataLength; ++i) {
        keyValue = keyPrefix + key.call(parent, data[i], i, data);
        if (node = nodeByKeyValue[keyValue]) {
          update[i] = node;
          node.__data__ = data[i];
          nodeByKeyValue[keyValue] = null;
        } else {
          enter[i] = new EnterNode(parent, data[i]);
        }
      }

      // Add any remaining nodes that were not bound to data to exit.
      for (i = 0; i < groupLength; ++i) {
        if ((node = group[i]) && (nodeByKeyValue[keyValues[i]] === node)) {
          exit[i] = node;
        }
      }
    }

    function selection_data(value, key) {
      if (!value) {
        data = new Array(this.size()), j = -1;
        this.each(function(d) { data[++j] = d; });
        return data;
      }

      var bind = key ? bindKey : bindIndex,
          parents = this._parents,
          groups = this._groups;

      if (typeof value !== "function") value = constant$1(value);

      for (var m = groups.length, update = new Array(m), enter = new Array(m), exit = new Array(m), j = 0; j < m; ++j) {
        var parent = parents[j],
            group = groups[j],
            groupLength = group.length,
            data = value.call(parent, parent && parent.__data__, j, parents),
            dataLength = data.length,
            enterGroup = enter[j] = new Array(dataLength),
            updateGroup = update[j] = new Array(dataLength),
            exitGroup = exit[j] = new Array(groupLength);

        bind(parent, group, enterGroup, updateGroup, exitGroup, data, key);

        // Now connect the enter nodes to their following update node, such that
        // appendChild can insert the materialized enter node before this node,
        // rather than at the end of the parent node.
        for (var i0 = 0, i1 = 0, previous, next; i0 < dataLength; ++i0) {
          if (previous = enterGroup[i0]) {
            if (i0 >= i1) i1 = i0 + 1;
            while (!(next = updateGroup[i1]) && ++i1 < dataLength);
            previous._next = next || null;
          }
        }
      }

      update = new Selection(update, parents);
      update._enter = enter;
      update._exit = exit;
      return update;
    }

    function selection_exit() {
      return new Selection(this._exit || this._groups.map(sparse), this._parents);
    }

    function selection_join(onenter, onupdate, onexit) {
      var enter = this.enter(), update = this, exit = this.exit();
      enter = typeof onenter === "function" ? onenter(enter) : enter.append(onenter + "");
      if (onupdate != null) update = onupdate(update);
      if (onexit == null) exit.remove(); else onexit(exit);
      return enter && update ? enter.merge(update).order() : update;
    }

    function selection_merge(selection) {

      for (var groups0 = this._groups, groups1 = selection._groups, m0 = groups0.length, m1 = groups1.length, m = Math.min(m0, m1), merges = new Array(m0), j = 0; j < m; ++j) {
        for (var group0 = groups0[j], group1 = groups1[j], n = group0.length, merge = merges[j] = new Array(n), node, i = 0; i < n; ++i) {
          if (node = group0[i] || group1[i]) {
            merge[i] = node;
          }
        }
      }

      for (; j < m0; ++j) {
        merges[j] = groups0[j];
      }

      return new Selection(merges, this._parents);
    }

    function selection_order() {

      for (var groups = this._groups, j = -1, m = groups.length; ++j < m;) {
        for (var group = groups[j], i = group.length - 1, next = group[i], node; --i >= 0;) {
          if (node = group[i]) {
            if (next && node.compareDocumentPosition(next) ^ 4) next.parentNode.insertBefore(node, next);
            next = node;
          }
        }
      }

      return this;
    }

    function selection_sort(compare) {
      if (!compare) compare = ascending$1;

      function compareNode(a, b) {
        return a && b ? compare(a.__data__, b.__data__) : !a - !b;
      }

      for (var groups = this._groups, m = groups.length, sortgroups = new Array(m), j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, sortgroup = sortgroups[j] = new Array(n), node, i = 0; i < n; ++i) {
          if (node = group[i]) {
            sortgroup[i] = node;
          }
        }
        sortgroup.sort(compareNode);
      }

      return new Selection(sortgroups, this._parents).order();
    }

    function ascending$1(a, b) {
      return a < b ? -1 : a > b ? 1 : a >= b ? 0 : NaN;
    }

    function selection_call() {
      var callback = arguments[0];
      arguments[0] = this;
      callback.apply(null, arguments);
      return this;
    }

    function selection_nodes() {
      var nodes = new Array(this.size()), i = -1;
      this.each(function() { nodes[++i] = this; });
      return nodes;
    }

    function selection_node() {

      for (var groups = this._groups, j = 0, m = groups.length; j < m; ++j) {
        for (var group = groups[j], i = 0, n = group.length; i < n; ++i) {
          var node = group[i];
          if (node) return node;
        }
      }

      return null;
    }

    function selection_size() {
      var size = 0;
      this.each(function() { ++size; });
      return size;
    }

    function selection_empty() {
      return !this.node();
    }

    function selection_each(callback) {

      for (var groups = this._groups, j = 0, m = groups.length; j < m; ++j) {
        for (var group = groups[j], i = 0, n = group.length, node; i < n; ++i) {
          if (node = group[i]) callback.call(node, node.__data__, i, group);
        }
      }

      return this;
    }

    function attrRemove(name) {
      return function() {
        this.removeAttribute(name);
      };
    }

    function attrRemoveNS(fullname) {
      return function() {
        this.removeAttributeNS(fullname.space, fullname.local);
      };
    }

    function attrConstant(name, value) {
      return function() {
        this.setAttribute(name, value);
      };
    }

    function attrConstantNS(fullname, value) {
      return function() {
        this.setAttributeNS(fullname.space, fullname.local, value);
      };
    }

    function attrFunction(name, value) {
      return function() {
        var v = value.apply(this, arguments);
        if (v == null) this.removeAttribute(name);
        else this.setAttribute(name, v);
      };
    }

    function attrFunctionNS(fullname, value) {
      return function() {
        var v = value.apply(this, arguments);
        if (v == null) this.removeAttributeNS(fullname.space, fullname.local);
        else this.setAttributeNS(fullname.space, fullname.local, v);
      };
    }

    function selection_attr(name, value) {
      var fullname = namespace(name);

      if (arguments.length < 2) {
        var node = this.node();
        return fullname.local
            ? node.getAttributeNS(fullname.space, fullname.local)
            : node.getAttribute(fullname);
      }

      return this.each((value == null
          ? (fullname.local ? attrRemoveNS : attrRemove) : (typeof value === "function"
          ? (fullname.local ? attrFunctionNS : attrFunction)
          : (fullname.local ? attrConstantNS : attrConstant)))(fullname, value));
    }

    function defaultView(node) {
      return (node.ownerDocument && node.ownerDocument.defaultView) // node is a Node
          || (node.document && node) // node is a Window
          || node.defaultView; // node is a Document
    }

    function styleRemove(name) {
      return function() {
        this.style.removeProperty(name);
      };
    }

    function styleConstant(name, value, priority) {
      return function() {
        this.style.setProperty(name, value, priority);
      };
    }

    function styleFunction(name, value, priority) {
      return function() {
        var v = value.apply(this, arguments);
        if (v == null) this.style.removeProperty(name);
        else this.style.setProperty(name, v, priority);
      };
    }

    function selection_style(name, value, priority) {
      return arguments.length > 1
          ? this.each((value == null
                ? styleRemove : typeof value === "function"
                ? styleFunction
                : styleConstant)(name, value, priority == null ? "" : priority))
          : styleValue(this.node(), name);
    }

    function styleValue(node, name) {
      return node.style.getPropertyValue(name)
          || defaultView(node).getComputedStyle(node, null).getPropertyValue(name);
    }

    function propertyRemove(name) {
      return function() {
        delete this[name];
      };
    }

    function propertyConstant(name, value) {
      return function() {
        this[name] = value;
      };
    }

    function propertyFunction(name, value) {
      return function() {
        var v = value.apply(this, arguments);
        if (v == null) delete this[name];
        else this[name] = v;
      };
    }

    function selection_property(name, value) {
      return arguments.length > 1
          ? this.each((value == null
              ? propertyRemove : typeof value === "function"
              ? propertyFunction
              : propertyConstant)(name, value))
          : this.node()[name];
    }

    function classArray(string) {
      return string.trim().split(/^|\s+/);
    }

    function classList(node) {
      return node.classList || new ClassList(node);
    }

    function ClassList(node) {
      this._node = node;
      this._names = classArray(node.getAttribute("class") || "");
    }

    ClassList.prototype = {
      add: function(name) {
        var i = this._names.indexOf(name);
        if (i < 0) {
          this._names.push(name);
          this._node.setAttribute("class", this._names.join(" "));
        }
      },
      remove: function(name) {
        var i = this._names.indexOf(name);
        if (i >= 0) {
          this._names.splice(i, 1);
          this._node.setAttribute("class", this._names.join(" "));
        }
      },
      contains: function(name) {
        return this._names.indexOf(name) >= 0;
      }
    };

    function classedAdd(node, names) {
      var list = classList(node), i = -1, n = names.length;
      while (++i < n) list.add(names[i]);
    }

    function classedRemove(node, names) {
      var list = classList(node), i = -1, n = names.length;
      while (++i < n) list.remove(names[i]);
    }

    function classedTrue(names) {
      return function() {
        classedAdd(this, names);
      };
    }

    function classedFalse(names) {
      return function() {
        classedRemove(this, names);
      };
    }

    function classedFunction(names, value) {
      return function() {
        (value.apply(this, arguments) ? classedAdd : classedRemove)(this, names);
      };
    }

    function selection_classed(name, value) {
      var names = classArray(name + "");

      if (arguments.length < 2) {
        var list = classList(this.node()), i = -1, n = names.length;
        while (++i < n) if (!list.contains(names[i])) return false;
        return true;
      }

      return this.each((typeof value === "function"
          ? classedFunction : value
          ? classedTrue
          : classedFalse)(names, value));
    }

    function textRemove() {
      this.textContent = "";
    }

    function textConstant(value) {
      return function() {
        this.textContent = value;
      };
    }

    function textFunction(value) {
      return function() {
        var v = value.apply(this, arguments);
        this.textContent = v == null ? "" : v;
      };
    }

    function selection_text(value) {
      return arguments.length
          ? this.each(value == null
              ? textRemove : (typeof value === "function"
              ? textFunction
              : textConstant)(value))
          : this.node().textContent;
    }

    function htmlRemove() {
      this.innerHTML = "";
    }

    function htmlConstant(value) {
      return function() {
        this.innerHTML = value;
      };
    }

    function htmlFunction(value) {
      return function() {
        var v = value.apply(this, arguments);
        this.innerHTML = v == null ? "" : v;
      };
    }

    function selection_html(value) {
      return arguments.length
          ? this.each(value == null
              ? htmlRemove : (typeof value === "function"
              ? htmlFunction
              : htmlConstant)(value))
          : this.node().innerHTML;
    }

    function raise() {
      if (this.nextSibling) this.parentNode.appendChild(this);
    }

    function selection_raise() {
      return this.each(raise);
    }

    function lower() {
      if (this.previousSibling) this.parentNode.insertBefore(this, this.parentNode.firstChild);
    }

    function selection_lower() {
      return this.each(lower);
    }

    function selection_append(name) {
      var create = typeof name === "function" ? name : creator(name);
      return this.select(function() {
        return this.appendChild(create.apply(this, arguments));
      });
    }

    function constantNull() {
      return null;
    }

    function selection_insert(name, before) {
      var create = typeof name === "function" ? name : creator(name),
          select = before == null ? constantNull : typeof before === "function" ? before : selector(before);
      return this.select(function() {
        return this.insertBefore(create.apply(this, arguments), select.apply(this, arguments) || null);
      });
    }

    function remove() {
      var parent = this.parentNode;
      if (parent) parent.removeChild(this);
    }

    function selection_remove() {
      return this.each(remove);
    }

    function selection_cloneShallow() {
      var clone = this.cloneNode(false), parent = this.parentNode;
      return parent ? parent.insertBefore(clone, this.nextSibling) : clone;
    }

    function selection_cloneDeep() {
      var clone = this.cloneNode(true), parent = this.parentNode;
      return parent ? parent.insertBefore(clone, this.nextSibling) : clone;
    }

    function selection_clone(deep) {
      return this.select(deep ? selection_cloneDeep : selection_cloneShallow);
    }

    function selection_datum(value) {
      return arguments.length
          ? this.property("__data__", value)
          : this.node().__data__;
    }

    var filterEvents = {};

    var event = null;

    if (typeof document !== "undefined") {
      var element = document.documentElement;
      if (!("onmouseenter" in element)) {
        filterEvents = {mouseenter: "mouseover", mouseleave: "mouseout"};
      }
    }

    function filterContextListener(listener, index, group) {
      listener = contextListener(listener, index, group);
      return function(event) {
        var related = event.relatedTarget;
        if (!related || (related !== this && !(related.compareDocumentPosition(this) & 8))) {
          listener.call(this, event);
        }
      };
    }

    function contextListener(listener, index, group) {
      return function(event1) {
        var event0 = event; // Events can be reentrant (e.g., focus).
        event = event1;
        try {
          listener.call(this, this.__data__, index, group);
        } finally {
          event = event0;
        }
      };
    }

    function parseTypenames$1(typenames) {
      return typenames.trim().split(/^|\s+/).map(function(t) {
        var name = "", i = t.indexOf(".");
        if (i >= 0) name = t.slice(i + 1), t = t.slice(0, i);
        return {type: t, name: name};
      });
    }

    function onRemove(typename) {
      return function() {
        var on = this.__on;
        if (!on) return;
        for (var j = 0, i = -1, m = on.length, o; j < m; ++j) {
          if (o = on[j], (!typename.type || o.type === typename.type) && o.name === typename.name) {
            this.removeEventListener(o.type, o.listener, o.capture);
          } else {
            on[++i] = o;
          }
        }
        if (++i) on.length = i;
        else delete this.__on;
      };
    }

    function onAdd(typename, value, capture) {
      var wrap = filterEvents.hasOwnProperty(typename.type) ? filterContextListener : contextListener;
      return function(d, i, group) {
        var on = this.__on, o, listener = wrap(value, i, group);
        if (on) for (var j = 0, m = on.length; j < m; ++j) {
          if ((o = on[j]).type === typename.type && o.name === typename.name) {
            this.removeEventListener(o.type, o.listener, o.capture);
            this.addEventListener(o.type, o.listener = listener, o.capture = capture);
            o.value = value;
            return;
          }
        }
        this.addEventListener(typename.type, listener, capture);
        o = {type: typename.type, name: typename.name, value: value, listener: listener, capture: capture};
        if (!on) this.__on = [o];
        else on.push(o);
      };
    }

    function selection_on(typename, value, capture) {
      var typenames = parseTypenames$1(typename + ""), i, n = typenames.length, t;

      if (arguments.length < 2) {
        var on = this.node().__on;
        if (on) for (var j = 0, m = on.length, o; j < m; ++j) {
          for (i = 0, o = on[j]; i < n; ++i) {
            if ((t = typenames[i]).type === o.type && t.name === o.name) {
              return o.value;
            }
          }
        }
        return;
      }

      on = value ? onAdd : onRemove;
      if (capture == null) capture = false;
      for (i = 0; i < n; ++i) this.each(on(typenames[i], value, capture));
      return this;
    }

    function customEvent(event1, listener, that, args) {
      var event0 = event;
      event1.sourceEvent = event;
      event = event1;
      try {
        return listener.apply(that, args);
      } finally {
        event = event0;
      }
    }

    function dispatchEvent(node, type, params) {
      var window = defaultView(node),
          event = window.CustomEvent;

      if (typeof event === "function") {
        event = new event(type, params);
      } else {
        event = window.document.createEvent("Event");
        if (params) event.initEvent(type, params.bubbles, params.cancelable), event.detail = params.detail;
        else event.initEvent(type, false, false);
      }

      node.dispatchEvent(event);
    }

    function dispatchConstant(type, params) {
      return function() {
        return dispatchEvent(this, type, params);
      };
    }

    function dispatchFunction(type, params) {
      return function() {
        return dispatchEvent(this, type, params.apply(this, arguments));
      };
    }

    function selection_dispatch(type, params) {
      return this.each((typeof params === "function"
          ? dispatchFunction
          : dispatchConstant)(type, params));
    }

    var root$1 = [null];

    function Selection(groups, parents) {
      this._groups = groups;
      this._parents = parents;
    }

    function selection() {
      return new Selection([[document.documentElement]], root$1);
    }

    Selection.prototype = selection.prototype = {
      constructor: Selection,
      select: selection_select,
      selectAll: selection_selectAll,
      filter: selection_filter,
      data: selection_data,
      enter: selection_enter,
      exit: selection_exit,
      join: selection_join,
      merge: selection_merge,
      order: selection_order,
      sort: selection_sort,
      call: selection_call,
      nodes: selection_nodes,
      node: selection_node,
      size: selection_size,
      empty: selection_empty,
      each: selection_each,
      attr: selection_attr,
      style: selection_style,
      property: selection_property,
      classed: selection_classed,
      text: selection_text,
      html: selection_html,
      raise: selection_raise,
      lower: selection_lower,
      append: selection_append,
      insert: selection_insert,
      remove: selection_remove,
      clone: selection_clone,
      datum: selection_datum,
      on: selection_on,
      dispatch: selection_dispatch
    };

    function select(selector) {
      return typeof selector === "string"
          ? new Selection([[document.querySelector(selector)]], [document.documentElement])
          : new Selection([[selector]], root$1);
    }

    function create(name) {
      return select(creator(name).call(document.documentElement));
    }

    var nextId = 0;

    function local() {
      return new Local;
    }

    function Local() {
      this._ = "@" + (++nextId).toString(36);
    }

    Local.prototype = local.prototype = {
      constructor: Local,
      get: function(node) {
        var id = this._;
        while (!(id in node)) if (!(node = node.parentNode)) return;
        return node[id];
      },
      set: function(node, value) {
        return node[this._] = value;
      },
      remove: function(node) {
        return this._ in node && delete node[this._];
      },
      toString: function() {
        return this._;
      }
    };

    function sourceEvent() {
      var current = event, source;
      while (source = current.sourceEvent) current = source;
      return current;
    }

    function point(node, event) {
      var svg = node.ownerSVGElement || node;

      if (svg.createSVGPoint) {
        var point = svg.createSVGPoint();
        point.x = event.clientX, point.y = event.clientY;
        point = point.matrixTransform(node.getScreenCTM().inverse());
        return [point.x, point.y];
      }

      var rect = node.getBoundingClientRect();
      return [event.clientX - rect.left - node.clientLeft, event.clientY - rect.top - node.clientTop];
    }

    function mouse(node) {
      var event = sourceEvent();
      if (event.changedTouches) event = event.changedTouches[0];
      return point(node, event);
    }

    function selectAll(selector) {
      return typeof selector === "string"
          ? new Selection([document.querySelectorAll(selector)], [document.documentElement])
          : new Selection([selector == null ? [] : selector], root$1);
    }

    function touch(node, touches, identifier) {
      if (arguments.length < 3) identifier = touches, touches = sourceEvent().changedTouches;

      for (var i = 0, n = touches ? touches.length : 0, touch; i < n; ++i) {
        if ((touch = touches[i]).identifier === identifier) {
          return point(node, touch);
        }
      }

      return null;
    }

    function touches(node, touches) {
      if (touches == null) touches = sourceEvent().touches;

      for (var i = 0, n = touches ? touches.length : 0, points = new Array(n); i < n; ++i) {
        points[i] = point(node, touches[i]);
      }

      return points;
    }

    function nopropagation() {
      event.stopImmediatePropagation();
    }

    function noevent() {
      event.preventDefault();
      event.stopImmediatePropagation();
    }

    function dragDisable(view) {
      var root = view.document.documentElement,
          selection = select(view).on("dragstart.drag", noevent, true);
      if ("onselectstart" in root) {
        selection.on("selectstart.drag", noevent, true);
      } else {
        root.__noselect = root.style.MozUserSelect;
        root.style.MozUserSelect = "none";
      }
    }

    function yesdrag(view, noclick) {
      var root = view.document.documentElement,
          selection = select(view).on("dragstart.drag", null);
      if (noclick) {
        selection.on("click.drag", noevent, true);
        setTimeout(function() { selection.on("click.drag", null); }, 0);
      }
      if ("onselectstart" in root) {
        selection.on("selectstart.drag", null);
      } else {
        root.style.MozUserSelect = root.__noselect;
        delete root.__noselect;
      }
    }

    function constant$2(x) {
      return function() {
        return x;
      };
    }

    function DragEvent(target, type, subject, id, active, x, y, dx, dy, dispatch) {
      this.target = target;
      this.type = type;
      this.subject = subject;
      this.identifier = id;
      this.active = active;
      this.x = x;
      this.y = y;
      this.dx = dx;
      this.dy = dy;
      this._ = dispatch;
    }

    DragEvent.prototype.on = function() {
      var value = this._.on.apply(this._, arguments);
      return value === this._ ? this : value;
    };

    // Ignore right-click, since that should open the context menu.
    function defaultFilter() {
      return !event.ctrlKey && !event.button;
    }

    function defaultContainer() {
      return this.parentNode;
    }

    function defaultSubject(d) {
      return d == null ? {x: event.x, y: event.y} : d;
    }

    function defaultTouchable() {
      return navigator.maxTouchPoints || ("ontouchstart" in this);
    }

    function drag() {
      var filter = defaultFilter,
          container = defaultContainer,
          subject = defaultSubject,
          touchable = defaultTouchable,
          gestures = {},
          listeners = dispatch("start", "drag", "end"),
          active = 0,
          mousedownx,
          mousedowny,
          mousemoving,
          touchending,
          clickDistance2 = 0;

      function drag(selection) {
        selection
            .on("mousedown.drag", mousedowned)
          .filter(touchable)
            .on("touchstart.drag", touchstarted)
            .on("touchmove.drag", touchmoved)
            .on("touchend.drag touchcancel.drag", touchended)
            .style("touch-action", "none")
            .style("-webkit-tap-highlight-color", "rgba(0,0,0,0)");
      }

      function mousedowned() {
        if (touchending || !filter.apply(this, arguments)) return;
        var gesture = beforestart("mouse", container.apply(this, arguments), mouse, this, arguments);
        if (!gesture) return;
        select(event.view).on("mousemove.drag", mousemoved, true).on("mouseup.drag", mouseupped, true);
        dragDisable(event.view);
        nopropagation();
        mousemoving = false;
        mousedownx = event.clientX;
        mousedowny = event.clientY;
        gesture("start");
      }

      function mousemoved() {
        noevent();
        if (!mousemoving) {
          var dx = event.clientX - mousedownx, dy = event.clientY - mousedowny;
          mousemoving = dx * dx + dy * dy > clickDistance2;
        }
        gestures.mouse("drag");
      }

      function mouseupped() {
        select(event.view).on("mousemove.drag mouseup.drag", null);
        yesdrag(event.view, mousemoving);
        noevent();
        gestures.mouse("end");
      }

      function touchstarted() {
        if (!filter.apply(this, arguments)) return;
        var touches = event.changedTouches,
            c = container.apply(this, arguments),
            n = touches.length, i, gesture;

        for (i = 0; i < n; ++i) {
          if (gesture = beforestart(touches[i].identifier, c, touch, this, arguments)) {
            nopropagation();
            gesture("start");
          }
        }
      }

      function touchmoved() {
        var touches = event.changedTouches,
            n = touches.length, i, gesture;

        for (i = 0; i < n; ++i) {
          if (gesture = gestures[touches[i].identifier]) {
            noevent();
            gesture("drag");
          }
        }
      }

      function touchended() {
        var touches = event.changedTouches,
            n = touches.length, i, gesture;

        if (touchending) clearTimeout(touchending);
        touchending = setTimeout(function() { touchending = null; }, 500); // Ghost clicks are delayed!
        for (i = 0; i < n; ++i) {
          if (gesture = gestures[touches[i].identifier]) {
            nopropagation();
            gesture("end");
          }
        }
      }

      function beforestart(id, container, point, that, args) {
        var p = point(container, id), s, dx, dy,
            sublisteners = listeners.copy();

        if (!customEvent(new DragEvent(drag, "beforestart", s, id, active, p[0], p[1], 0, 0, sublisteners), function() {
          if ((event.subject = s = subject.apply(that, args)) == null) return false;
          dx = s.x - p[0] || 0;
          dy = s.y - p[1] || 0;
          return true;
        })) return;

        return function gesture(type) {
          var p0 = p, n;
          switch (type) {
            case "start": gestures[id] = gesture, n = active++; break;
            case "end": delete gestures[id], --active; // nobreak
            case "drag": p = point(container, id), n = active; break;
          }
          customEvent(new DragEvent(drag, type, s, id, n, p[0] + dx, p[1] + dy, p[0] - p0[0], p[1] - p0[1], sublisteners), sublisteners.apply, sublisteners, [type, that, args]);
        };
      }

      drag.filter = function(_) {
        return arguments.length ? (filter = typeof _ === "function" ? _ : constant$2(!!_), drag) : filter;
      };

      drag.container = function(_) {
        return arguments.length ? (container = typeof _ === "function" ? _ : constant$2(_), drag) : container;
      };

      drag.subject = function(_) {
        return arguments.length ? (subject = typeof _ === "function" ? _ : constant$2(_), drag) : subject;
      };

      drag.touchable = function(_) {
        return arguments.length ? (touchable = typeof _ === "function" ? _ : constant$2(!!_), drag) : touchable;
      };

      drag.on = function() {
        var value = listeners.on.apply(listeners, arguments);
        return value === listeners ? drag : value;
      };

      drag.clickDistance = function(_) {
        return arguments.length ? (clickDistance2 = (_ = +_) * _, drag) : Math.sqrt(clickDistance2);
      };

      return drag;
    }

    function define(constructor, factory, prototype) {
      constructor.prototype = factory.prototype = prototype;
      prototype.constructor = constructor;
    }

    function extend(parent, definition) {
      var prototype = Object.create(parent.prototype);
      for (var key in definition) prototype[key] = definition[key];
      return prototype;
    }

    function Color() {}

    var darker = 0.7;
    var brighter = 1 / darker;

    var reI = "\\s*([+-]?\\d+)\\s*",
        reN = "\\s*([+-]?\\d*\\.?\\d+(?:[eE][+-]?\\d+)?)\\s*",
        reP = "\\s*([+-]?\\d*\\.?\\d+(?:[eE][+-]?\\d+)?)%\\s*",
        reHex = /^#([0-9a-f]{3,8})$/,
        reRgbInteger = new RegExp("^rgb\\(" + [reI, reI, reI] + "\\)$"),
        reRgbPercent = new RegExp("^rgb\\(" + [reP, reP, reP] + "\\)$"),
        reRgbaInteger = new RegExp("^rgba\\(" + [reI, reI, reI, reN] + "\\)$"),
        reRgbaPercent = new RegExp("^rgba\\(" + [reP, reP, reP, reN] + "\\)$"),
        reHslPercent = new RegExp("^hsl\\(" + [reN, reP, reP] + "\\)$"),
        reHslaPercent = new RegExp("^hsla\\(" + [reN, reP, reP, reN] + "\\)$");

    var named = {
      aliceblue: 0xf0f8ff,
      antiquewhite: 0xfaebd7,
      aqua: 0x00ffff,
      aquamarine: 0x7fffd4,
      azure: 0xf0ffff,
      beige: 0xf5f5dc,
      bisque: 0xffe4c4,
      black: 0x000000,
      blanchedalmond: 0xffebcd,
      blue: 0x0000ff,
      blueviolet: 0x8a2be2,
      brown: 0xa52a2a,
      burlywood: 0xdeb887,
      cadetblue: 0x5f9ea0,
      chartreuse: 0x7fff00,
      chocolate: 0xd2691e,
      coral: 0xff7f50,
      cornflowerblue: 0x6495ed,
      cornsilk: 0xfff8dc,
      crimson: 0xdc143c,
      cyan: 0x00ffff,
      darkblue: 0x00008b,
      darkcyan: 0x008b8b,
      darkgoldenrod: 0xb8860b,
      darkgray: 0xa9a9a9,
      darkgreen: 0x006400,
      darkgrey: 0xa9a9a9,
      darkkhaki: 0xbdb76b,
      darkmagenta: 0x8b008b,
      darkolivegreen: 0x556b2f,
      darkorange: 0xff8c00,
      darkorchid: 0x9932cc,
      darkred: 0x8b0000,
      darksalmon: 0xe9967a,
      darkseagreen: 0x8fbc8f,
      darkslateblue: 0x483d8b,
      darkslategray: 0x2f4f4f,
      darkslategrey: 0x2f4f4f,
      darkturquoise: 0x00ced1,
      darkviolet: 0x9400d3,
      deeppink: 0xff1493,
      deepskyblue: 0x00bfff,
      dimgray: 0x696969,
      dimgrey: 0x696969,
      dodgerblue: 0x1e90ff,
      firebrick: 0xb22222,
      floralwhite: 0xfffaf0,
      forestgreen: 0x228b22,
      fuchsia: 0xff00ff,
      gainsboro: 0xdcdcdc,
      ghostwhite: 0xf8f8ff,
      gold: 0xffd700,
      goldenrod: 0xdaa520,
      gray: 0x808080,
      green: 0x008000,
      greenyellow: 0xadff2f,
      grey: 0x808080,
      honeydew: 0xf0fff0,
      hotpink: 0xff69b4,
      indianred: 0xcd5c5c,
      indigo: 0x4b0082,
      ivory: 0xfffff0,
      khaki: 0xf0e68c,
      lavender: 0xe6e6fa,
      lavenderblush: 0xfff0f5,
      lawngreen: 0x7cfc00,
      lemonchiffon: 0xfffacd,
      lightblue: 0xadd8e6,
      lightcoral: 0xf08080,
      lightcyan: 0xe0ffff,
      lightgoldenrodyellow: 0xfafad2,
      lightgray: 0xd3d3d3,
      lightgreen: 0x90ee90,
      lightgrey: 0xd3d3d3,
      lightpink: 0xffb6c1,
      lightsalmon: 0xffa07a,
      lightseagreen: 0x20b2aa,
      lightskyblue: 0x87cefa,
      lightslategray: 0x778899,
      lightslategrey: 0x778899,
      lightsteelblue: 0xb0c4de,
      lightyellow: 0xffffe0,
      lime: 0x00ff00,
      limegreen: 0x32cd32,
      linen: 0xfaf0e6,
      magenta: 0xff00ff,
      maroon: 0x800000,
      mediumaquamarine: 0x66cdaa,
      mediumblue: 0x0000cd,
      mediumorchid: 0xba55d3,
      mediumpurple: 0x9370db,
      mediumseagreen: 0x3cb371,
      mediumslateblue: 0x7b68ee,
      mediumspringgreen: 0x00fa9a,
      mediumturquoise: 0x48d1cc,
      mediumvioletred: 0xc71585,
      midnightblue: 0x191970,
      mintcream: 0xf5fffa,
      mistyrose: 0xffe4e1,
      moccasin: 0xffe4b5,
      navajowhite: 0xffdead,
      navy: 0x000080,
      oldlace: 0xfdf5e6,
      olive: 0x808000,
      olivedrab: 0x6b8e23,
      orange: 0xffa500,
      orangered: 0xff4500,
      orchid: 0xda70d6,
      palegoldenrod: 0xeee8aa,
      palegreen: 0x98fb98,
      paleturquoise: 0xafeeee,
      palevioletred: 0xdb7093,
      papayawhip: 0xffefd5,
      peachpuff: 0xffdab9,
      peru: 0xcd853f,
      pink: 0xffc0cb,
      plum: 0xdda0dd,
      powderblue: 0xb0e0e6,
      purple: 0x800080,
      rebeccapurple: 0x663399,
      red: 0xff0000,
      rosybrown: 0xbc8f8f,
      royalblue: 0x4169e1,
      saddlebrown: 0x8b4513,
      salmon: 0xfa8072,
      sandybrown: 0xf4a460,
      seagreen: 0x2e8b57,
      seashell: 0xfff5ee,
      sienna: 0xa0522d,
      silver: 0xc0c0c0,
      skyblue: 0x87ceeb,
      slateblue: 0x6a5acd,
      slategray: 0x708090,
      slategrey: 0x708090,
      snow: 0xfffafa,
      springgreen: 0x00ff7f,
      steelblue: 0x4682b4,
      tan: 0xd2b48c,
      teal: 0x008080,
      thistle: 0xd8bfd8,
      tomato: 0xff6347,
      turquoise: 0x40e0d0,
      violet: 0xee82ee,
      wheat: 0xf5deb3,
      white: 0xffffff,
      whitesmoke: 0xf5f5f5,
      yellow: 0xffff00,
      yellowgreen: 0x9acd32
    };

    define(Color, color, {
      copy: function(channels) {
        return Object.assign(new this.constructor, this, channels);
      },
      displayable: function() {
        return this.rgb().displayable();
      },
      hex: color_formatHex, // Deprecated! Use color.formatHex.
      formatHex: color_formatHex,
      formatHsl: color_formatHsl,
      formatRgb: color_formatRgb,
      toString: color_formatRgb
    });

    function color_formatHex() {
      return this.rgb().formatHex();
    }

    function color_formatHsl() {
      return hslConvert(this).formatHsl();
    }

    function color_formatRgb() {
      return this.rgb().formatRgb();
    }

    function color(format) {
      var m, l;
      format = (format + "").trim().toLowerCase();
      return (m = reHex.exec(format)) ? (l = m[1].length, m = parseInt(m[1], 16), l === 6 ? rgbn(m) // #ff0000
          : l === 3 ? new Rgb((m >> 8 & 0xf) | (m >> 4 & 0xf0), (m >> 4 & 0xf) | (m & 0xf0), ((m & 0xf) << 4) | (m & 0xf), 1) // #f00
          : l === 8 ? rgba(m >> 24 & 0xff, m >> 16 & 0xff, m >> 8 & 0xff, (m & 0xff) / 0xff) // #ff000000
          : l === 4 ? rgba((m >> 12 & 0xf) | (m >> 8 & 0xf0), (m >> 8 & 0xf) | (m >> 4 & 0xf0), (m >> 4 & 0xf) | (m & 0xf0), (((m & 0xf) << 4) | (m & 0xf)) / 0xff) // #f000
          : null) // invalid hex
          : (m = reRgbInteger.exec(format)) ? new Rgb(m[1], m[2], m[3], 1) // rgb(255, 0, 0)
          : (m = reRgbPercent.exec(format)) ? new Rgb(m[1] * 255 / 100, m[2] * 255 / 100, m[3] * 255 / 100, 1) // rgb(100%, 0%, 0%)
          : (m = reRgbaInteger.exec(format)) ? rgba(m[1], m[2], m[3], m[4]) // rgba(255, 0, 0, 1)
          : (m = reRgbaPercent.exec(format)) ? rgba(m[1] * 255 / 100, m[2] * 255 / 100, m[3] * 255 / 100, m[4]) // rgb(100%, 0%, 0%, 1)
          : (m = reHslPercent.exec(format)) ? hsla(m[1], m[2] / 100, m[3] / 100, 1) // hsl(120, 50%, 50%)
          : (m = reHslaPercent.exec(format)) ? hsla(m[1], m[2] / 100, m[3] / 100, m[4]) // hsla(120, 50%, 50%, 1)
          : named.hasOwnProperty(format) ? rgbn(named[format]) // eslint-disable-line no-prototype-builtins
          : format === "transparent" ? new Rgb(NaN, NaN, NaN, 0)
          : null;
    }

    function rgbn(n) {
      return new Rgb(n >> 16 & 0xff, n >> 8 & 0xff, n & 0xff, 1);
    }

    function rgba(r, g, b, a) {
      if (a <= 0) r = g = b = NaN;
      return new Rgb(r, g, b, a);
    }

    function rgbConvert(o) {
      if (!(o instanceof Color)) o = color(o);
      if (!o) return new Rgb;
      o = o.rgb();
      return new Rgb(o.r, o.g, o.b, o.opacity);
    }

    function rgb(r, g, b, opacity) {
      return arguments.length === 1 ? rgbConvert(r) : new Rgb(r, g, b, opacity == null ? 1 : opacity);
    }

    function Rgb(r, g, b, opacity) {
      this.r = +r;
      this.g = +g;
      this.b = +b;
      this.opacity = +opacity;
    }

    define(Rgb, rgb, extend(Color, {
      brighter: function(k) {
        k = k == null ? brighter : Math.pow(brighter, k);
        return new Rgb(this.r * k, this.g * k, this.b * k, this.opacity);
      },
      darker: function(k) {
        k = k == null ? darker : Math.pow(darker, k);
        return new Rgb(this.r * k, this.g * k, this.b * k, this.opacity);
      },
      rgb: function() {
        return this;
      },
      displayable: function() {
        return (-0.5 <= this.r && this.r < 255.5)
            && (-0.5 <= this.g && this.g < 255.5)
            && (-0.5 <= this.b && this.b < 255.5)
            && (0 <= this.opacity && this.opacity <= 1);
      },
      hex: rgb_formatHex, // Deprecated! Use color.formatHex.
      formatHex: rgb_formatHex,
      formatRgb: rgb_formatRgb,
      toString: rgb_formatRgb
    }));

    function rgb_formatHex() {
      return "#" + hex(this.r) + hex(this.g) + hex(this.b);
    }

    function rgb_formatRgb() {
      var a = this.opacity; a = isNaN(a) ? 1 : Math.max(0, Math.min(1, a));
      return (a === 1 ? "rgb(" : "rgba(")
          + Math.max(0, Math.min(255, Math.round(this.r) || 0)) + ", "
          + Math.max(0, Math.min(255, Math.round(this.g) || 0)) + ", "
          + Math.max(0, Math.min(255, Math.round(this.b) || 0))
          + (a === 1 ? ")" : ", " + a + ")");
    }

    function hex(value) {
      value = Math.max(0, Math.min(255, Math.round(value) || 0));
      return (value < 16 ? "0" : "") + value.toString(16);
    }

    function hsla(h, s, l, a) {
      if (a <= 0) h = s = l = NaN;
      else if (l <= 0 || l >= 1) h = s = NaN;
      else if (s <= 0) h = NaN;
      return new Hsl(h, s, l, a);
    }

    function hslConvert(o) {
      if (o instanceof Hsl) return new Hsl(o.h, o.s, o.l, o.opacity);
      if (!(o instanceof Color)) o = color(o);
      if (!o) return new Hsl;
      if (o instanceof Hsl) return o;
      o = o.rgb();
      var r = o.r / 255,
          g = o.g / 255,
          b = o.b / 255,
          min = Math.min(r, g, b),
          max = Math.max(r, g, b),
          h = NaN,
          s = max - min,
          l = (max + min) / 2;
      if (s) {
        if (r === max) h = (g - b) / s + (g < b) * 6;
        else if (g === max) h = (b - r) / s + 2;
        else h = (r - g) / s + 4;
        s /= l < 0.5 ? max + min : 2 - max - min;
        h *= 60;
      } else {
        s = l > 0 && l < 1 ? 0 : h;
      }
      return new Hsl(h, s, l, o.opacity);
    }

    function hsl(h, s, l, opacity) {
      return arguments.length === 1 ? hslConvert(h) : new Hsl(h, s, l, opacity == null ? 1 : opacity);
    }

    function Hsl(h, s, l, opacity) {
      this.h = +h;
      this.s = +s;
      this.l = +l;
      this.opacity = +opacity;
    }

    define(Hsl, hsl, extend(Color, {
      brighter: function(k) {
        k = k == null ? brighter : Math.pow(brighter, k);
        return new Hsl(this.h, this.s, this.l * k, this.opacity);
      },
      darker: function(k) {
        k = k == null ? darker : Math.pow(darker, k);
        return new Hsl(this.h, this.s, this.l * k, this.opacity);
      },
      rgb: function() {
        var h = this.h % 360 + (this.h < 0) * 360,
            s = isNaN(h) || isNaN(this.s) ? 0 : this.s,
            l = this.l,
            m2 = l + (l < 0.5 ? l : 1 - l) * s,
            m1 = 2 * l - m2;
        return new Rgb(
          hsl2rgb(h >= 240 ? h - 240 : h + 120, m1, m2),
          hsl2rgb(h, m1, m2),
          hsl2rgb(h < 120 ? h + 240 : h - 120, m1, m2),
          this.opacity
        );
      },
      displayable: function() {
        return (0 <= this.s && this.s <= 1 || isNaN(this.s))
            && (0 <= this.l && this.l <= 1)
            && (0 <= this.opacity && this.opacity <= 1);
      },
      formatHsl: function() {
        var a = this.opacity; a = isNaN(a) ? 1 : Math.max(0, Math.min(1, a));
        return (a === 1 ? "hsl(" : "hsla(")
            + (this.h || 0) + ", "
            + (this.s || 0) * 100 + "%, "
            + (this.l || 0) * 100 + "%"
            + (a === 1 ? ")" : ", " + a + ")");
      }
    }));

    /* From FvD 13.37, CSS Color Module Level 3 */
    function hsl2rgb(h, m1, m2) {
      return (h < 60 ? m1 + (m2 - m1) * h / 60
          : h < 180 ? m2
          : h < 240 ? m1 + (m2 - m1) * (240 - h) / 60
          : m1) * 255;
    }

    var deg2rad = Math.PI / 180;
    var rad2deg = 180 / Math.PI;

    // https://observablehq.com/@mbostock/lab-and-rgb
    var K = 18,
        Xn = 0.96422,
        Yn = 1,
        Zn = 0.82521,
        t0 = 4 / 29,
        t1 = 6 / 29,
        t2 = 3 * t1 * t1,
        t3 = t1 * t1 * t1;

    function labConvert(o) {
      if (o instanceof Lab) return new Lab(o.l, o.a, o.b, o.opacity);
      if (o instanceof Hcl) return hcl2lab(o);
      if (!(o instanceof Rgb)) o = rgbConvert(o);
      var r = rgb2lrgb(o.r),
          g = rgb2lrgb(o.g),
          b = rgb2lrgb(o.b),
          y = xyz2lab((0.2225045 * r + 0.7168786 * g + 0.0606169 * b) / Yn), x, z;
      if (r === g && g === b) x = z = y; else {
        x = xyz2lab((0.4360747 * r + 0.3850649 * g + 0.1430804 * b) / Xn);
        z = xyz2lab((0.0139322 * r + 0.0971045 * g + 0.7141733 * b) / Zn);
      }
      return new Lab(116 * y - 16, 500 * (x - y), 200 * (y - z), o.opacity);
    }

    function gray(l, opacity) {
      return new Lab(l, 0, 0, opacity == null ? 1 : opacity);
    }

    function lab(l, a, b, opacity) {
      return arguments.length === 1 ? labConvert(l) : new Lab(l, a, b, opacity == null ? 1 : opacity);
    }

    function Lab(l, a, b, opacity) {
      this.l = +l;
      this.a = +a;
      this.b = +b;
      this.opacity = +opacity;
    }

    define(Lab, lab, extend(Color, {
      brighter: function(k) {
        return new Lab(this.l + K * (k == null ? 1 : k), this.a, this.b, this.opacity);
      },
      darker: function(k) {
        return new Lab(this.l - K * (k == null ? 1 : k), this.a, this.b, this.opacity);
      },
      rgb: function() {
        var y = (this.l + 16) / 116,
            x = isNaN(this.a) ? y : y + this.a / 500,
            z = isNaN(this.b) ? y : y - this.b / 200;
        x = Xn * lab2xyz(x);
        y = Yn * lab2xyz(y);
        z = Zn * lab2xyz(z);
        return new Rgb(
          lrgb2rgb( 3.1338561 * x - 1.6168667 * y - 0.4906146 * z),
          lrgb2rgb(-0.9787684 * x + 1.9161415 * y + 0.0334540 * z),
          lrgb2rgb( 0.0719453 * x - 0.2289914 * y + 1.4052427 * z),
          this.opacity
        );
      }
    }));

    function xyz2lab(t) {
      return t > t3 ? Math.pow(t, 1 / 3) : t / t2 + t0;
    }

    function lab2xyz(t) {
      return t > t1 ? t * t * t : t2 * (t - t0);
    }

    function lrgb2rgb(x) {
      return 255 * (x <= 0.0031308 ? 12.92 * x : 1.055 * Math.pow(x, 1 / 2.4) - 0.055);
    }

    function rgb2lrgb(x) {
      return (x /= 255) <= 0.04045 ? x / 12.92 : Math.pow((x + 0.055) / 1.055, 2.4);
    }

    function hclConvert(o) {
      if (o instanceof Hcl) return new Hcl(o.h, o.c, o.l, o.opacity);
      if (!(o instanceof Lab)) o = labConvert(o);
      if (o.a === 0 && o.b === 0) return new Hcl(NaN, 0 < o.l && o.l < 100 ? 0 : NaN, o.l, o.opacity);
      var h = Math.atan2(o.b, o.a) * rad2deg;
      return new Hcl(h < 0 ? h + 360 : h, Math.sqrt(o.a * o.a + o.b * o.b), o.l, o.opacity);
    }

    function lch(l, c, h, opacity) {
      return arguments.length === 1 ? hclConvert(l) : new Hcl(h, c, l, opacity == null ? 1 : opacity);
    }

    function hcl(h, c, l, opacity) {
      return arguments.length === 1 ? hclConvert(h) : new Hcl(h, c, l, opacity == null ? 1 : opacity);
    }

    function Hcl(h, c, l, opacity) {
      this.h = +h;
      this.c = +c;
      this.l = +l;
      this.opacity = +opacity;
    }

    function hcl2lab(o) {
      if (isNaN(o.h)) return new Lab(o.l, 0, 0, o.opacity);
      var h = o.h * deg2rad;
      return new Lab(o.l, Math.cos(h) * o.c, Math.sin(h) * o.c, o.opacity);
    }

    define(Hcl, hcl, extend(Color, {
      brighter: function(k) {
        return new Hcl(this.h, this.c, this.l + K * (k == null ? 1 : k), this.opacity);
      },
      darker: function(k) {
        return new Hcl(this.h, this.c, this.l - K * (k == null ? 1 : k), this.opacity);
      },
      rgb: function() {
        return hcl2lab(this).rgb();
      }
    }));

    var A = -0.14861,
        B = +1.78277,
        C = -0.29227,
        D = -0.90649,
        E = +1.97294,
        ED = E * D,
        EB = E * B,
        BC_DA = B * C - D * A;

    function cubehelixConvert(o) {
      if (o instanceof Cubehelix) return new Cubehelix(o.h, o.s, o.l, o.opacity);
      if (!(o instanceof Rgb)) o = rgbConvert(o);
      var r = o.r / 255,
          g = o.g / 255,
          b = o.b / 255,
          l = (BC_DA * b + ED * r - EB * g) / (BC_DA + ED - EB),
          bl = b - l,
          k = (E * (g - l) - C * bl) / D,
          s = Math.sqrt(k * k + bl * bl) / (E * l * (1 - l)), // NaN if l=0 or l=1
          h = s ? Math.atan2(k, bl) * rad2deg - 120 : NaN;
      return new Cubehelix(h < 0 ? h + 360 : h, s, l, o.opacity);
    }

    function cubehelix(h, s, l, opacity) {
      return arguments.length === 1 ? cubehelixConvert(h) : new Cubehelix(h, s, l, opacity == null ? 1 : opacity);
    }

    function Cubehelix(h, s, l, opacity) {
      this.h = +h;
      this.s = +s;
      this.l = +l;
      this.opacity = +opacity;
    }

    define(Cubehelix, cubehelix, extend(Color, {
      brighter: function(k) {
        k = k == null ? brighter : Math.pow(brighter, k);
        return new Cubehelix(this.h, this.s, this.l * k, this.opacity);
      },
      darker: function(k) {
        k = k == null ? darker : Math.pow(darker, k);
        return new Cubehelix(this.h, this.s, this.l * k, this.opacity);
      },
      rgb: function() {
        var h = isNaN(this.h) ? 0 : (this.h + 120) * deg2rad,
            l = +this.l,
            a = isNaN(this.s) ? 0 : this.s * l * (1 - l),
            cosh = Math.cos(h),
            sinh = Math.sin(h);
        return new Rgb(
          255 * (l + a * (A * cosh + B * sinh)),
          255 * (l + a * (C * cosh + D * sinh)),
          255 * (l + a * (E * cosh)),
          this.opacity
        );
      }
    }));

    function basis(t1, v0, v1, v2, v3) {
      var t2 = t1 * t1, t3 = t2 * t1;
      return ((1 - 3 * t1 + 3 * t2 - t3) * v0
          + (4 - 6 * t2 + 3 * t3) * v1
          + (1 + 3 * t1 + 3 * t2 - 3 * t3) * v2
          + t3 * v3) / 6;
    }

    function basis$1(values) {
      var n = values.length - 1;
      return function(t) {
        var i = t <= 0 ? (t = 0) : t >= 1 ? (t = 1, n - 1) : Math.floor(t * n),
            v1 = values[i],
            v2 = values[i + 1],
            v0 = i > 0 ? values[i - 1] : 2 * v1 - v2,
            v3 = i < n - 1 ? values[i + 2] : 2 * v2 - v1;
        return basis((t - i / n) * n, v0, v1, v2, v3);
      };
    }

    function basisClosed(values) {
      var n = values.length;
      return function(t) {
        var i = Math.floor(((t %= 1) < 0 ? ++t : t) * n),
            v0 = values[(i + n - 1) % n],
            v1 = values[i % n],
            v2 = values[(i + 1) % n],
            v3 = values[(i + 2) % n];
        return basis((t - i / n) * n, v0, v1, v2, v3);
      };
    }

    function constant$3(x) {
      return function() {
        return x;
      };
    }

    function linear(a, d) {
      return function(t) {
        return a + t * d;
      };
    }

    function exponential(a, b, y) {
      return a = Math.pow(a, y), b = Math.pow(b, y) - a, y = 1 / y, function(t) {
        return Math.pow(a + t * b, y);
      };
    }

    function hue(a, b) {
      var d = b - a;
      return d ? linear(a, d > 180 || d < -180 ? d - 360 * Math.round(d / 360) : d) : constant$3(isNaN(a) ? b : a);
    }

    function gamma(y) {
      return (y = +y) === 1 ? nogamma : function(a, b) {
        return b - a ? exponential(a, b, y) : constant$3(isNaN(a) ? b : a);
      };
    }

    function nogamma(a, b) {
      var d = b - a;
      return d ? linear(a, d) : constant$3(isNaN(a) ? b : a);
    }

    var interpolateRgb = (function rgbGamma(y) {
      var color = gamma(y);

      function rgb$1(start, end) {
        var r = color((start = rgb(start)).r, (end = rgb(end)).r),
            g = color(start.g, end.g),
            b = color(start.b, end.b),
            opacity = nogamma(start.opacity, end.opacity);
        return function(t) {
          start.r = r(t);
          start.g = g(t);
          start.b = b(t);
          start.opacity = opacity(t);
          return start + "";
        };
      }

      rgb$1.gamma = rgbGamma;

      return rgb$1;
    })(1);

    function rgbSpline(spline) {
      return function(colors) {
        var n = colors.length,
            r = new Array(n),
            g = new Array(n),
            b = new Array(n),
            i, color;
        for (i = 0; i < n; ++i) {
          color = rgb(colors[i]);
          r[i] = color.r || 0;
          g[i] = color.g || 0;
          b[i] = color.b || 0;
        }
        r = spline(r);
        g = spline(g);
        b = spline(b);
        color.opacity = 1;
        return function(t) {
          color.r = r(t);
          color.g = g(t);
          color.b = b(t);
          return color + "";
        };
      };
    }

    var rgbBasis = rgbSpline(basis$1);
    var rgbBasisClosed = rgbSpline(basisClosed);

    function numberArray(a, b) {
      if (!b) b = [];
      var n = a ? Math.min(b.length, a.length) : 0,
          c = b.slice(),
          i;
      return function(t) {
        for (i = 0; i < n; ++i) c[i] = a[i] * (1 - t) + b[i] * t;
        return c;
      };
    }

    function isNumberArray(x) {
      return ArrayBuffer.isView(x) && !(x instanceof DataView);
    }

    function array$1(a, b) {
      return (isNumberArray(b) ? numberArray : genericArray)(a, b);
    }

    function genericArray(a, b) {
      var nb = b ? b.length : 0,
          na = a ? Math.min(nb, a.length) : 0,
          x = new Array(na),
          c = new Array(nb),
          i;

      for (i = 0; i < na; ++i) x[i] = interpolateValue(a[i], b[i]);
      for (; i < nb; ++i) c[i] = b[i];

      return function(t) {
        for (i = 0; i < na; ++i) c[i] = x[i](t);
        return c;
      };
    }

    function date(a, b) {
      var d = new Date;
      return a = +a, b = +b, function(t) {
        return d.setTime(a * (1 - t) + b * t), d;
      };
    }

    function interpolateNumber(a, b) {
      return a = +a, b = +b, function(t) {
        return a * (1 - t) + b * t;
      };
    }

    function object(a, b) {
      var i = {},
          c = {},
          k;

      if (a === null || typeof a !== "object") a = {};
      if (b === null || typeof b !== "object") b = {};

      for (k in b) {
        if (k in a) {
          i[k] = interpolateValue(a[k], b[k]);
        } else {
          c[k] = b[k];
        }
      }

      return function(t) {
        for (k in i) c[k] = i[k](t);
        return c;
      };
    }

    var reA = /[-+]?(?:\d+\.?\d*|\.?\d+)(?:[eE][-+]?\d+)?/g,
        reB = new RegExp(reA.source, "g");

    function zero(b) {
      return function() {
        return b;
      };
    }

    function one(b) {
      return function(t) {
        return b(t) + "";
      };
    }

    function interpolateString(a, b) {
      var bi = reA.lastIndex = reB.lastIndex = 0, // scan index for next number in b
          am, // current match in a
          bm, // current match in b
          bs, // string preceding current number in b, if any
          i = -1, // index in s
          s = [], // string constants and placeholders
          q = []; // number interpolators

      // Coerce inputs to strings.
      a = a + "", b = b + "";

      // Interpolate pairs of numbers in a & b.
      while ((am = reA.exec(a))
          && (bm = reB.exec(b))) {
        if ((bs = bm.index) > bi) { // a string precedes the next number in b
          bs = b.slice(bi, bs);
          if (s[i]) s[i] += bs; // coalesce with previous string
          else s[++i] = bs;
        }
        if ((am = am[0]) === (bm = bm[0])) { // numbers in a & b match
          if (s[i]) s[i] += bm; // coalesce with previous string
          else s[++i] = bm;
        } else { // interpolate non-matching numbers
          s[++i] = null;
          q.push({i: i, x: interpolateNumber(am, bm)});
        }
        bi = reB.lastIndex;
      }

      // Add remains of b.
      if (bi < b.length) {
        bs = b.slice(bi);
        if (s[i]) s[i] += bs; // coalesce with previous string
        else s[++i] = bs;
      }

      // Special optimization for only a single match.
      // Otherwise, interpolate each of the numbers and rejoin the string.
      return s.length < 2 ? (q[0]
          ? one(q[0].x)
          : zero(b))
          : (b = q.length, function(t) {
              for (var i = 0, o; i < b; ++i) s[(o = q[i]).i] = o.x(t);
              return s.join("");
            });
    }

    function interpolateValue(a, b) {
      var t = typeof b, c;
      return b == null || t === "boolean" ? constant$3(b)
          : (t === "number" ? interpolateNumber
          : t === "string" ? ((c = color(b)) ? (b = c, interpolateRgb) : interpolateString)
          : b instanceof color ? interpolateRgb
          : b instanceof Date ? date
          : isNumberArray(b) ? numberArray
          : Array.isArray(b) ? genericArray
          : typeof b.valueOf !== "function" && typeof b.toString !== "function" || isNaN(b) ? object
          : interpolateNumber)(a, b);
    }

    function discrete(range) {
      var n = range.length;
      return function(t) {
        return range[Math.max(0, Math.min(n - 1, Math.floor(t * n)))];
      };
    }

    function hue$1(a, b) {
      var i = hue(+a, +b);
      return function(t) {
        var x = i(t);
        return x - 360 * Math.floor(x / 360);
      };
    }

    function interpolateRound(a, b) {
      return a = +a, b = +b, function(t) {
        return Math.round(a * (1 - t) + b * t);
      };
    }

    var degrees = 180 / Math.PI;

    var identity$2 = {
      translateX: 0,
      translateY: 0,
      rotate: 0,
      skewX: 0,
      scaleX: 1,
      scaleY: 1
    };

    function decompose(a, b, c, d, e, f) {
      var scaleX, scaleY, skewX;
      if (scaleX = Math.sqrt(a * a + b * b)) a /= scaleX, b /= scaleX;
      if (skewX = a * c + b * d) c -= a * skewX, d -= b * skewX;
      if (scaleY = Math.sqrt(c * c + d * d)) c /= scaleY, d /= scaleY, skewX /= scaleY;
      if (a * d < b * c) a = -a, b = -b, skewX = -skewX, scaleX = -scaleX;
      return {
        translateX: e,
        translateY: f,
        rotate: Math.atan2(b, a) * degrees,
        skewX: Math.atan(skewX) * degrees,
        scaleX: scaleX,
        scaleY: scaleY
      };
    }

    var cssNode,
        cssRoot,
        cssView,
        svgNode;

    function parseCss$1(value) {
      if (value === "none") return identity$2;
      if (!cssNode) cssNode = document.createElement("DIV"), cssRoot = document.documentElement, cssView = document.defaultView;
      cssNode.style.transform = value;
      value = cssView.getComputedStyle(cssRoot.appendChild(cssNode), null).getPropertyValue("transform");
      cssRoot.removeChild(cssNode);
      value = value.slice(7, -1).split(",");
      return decompose(+value[0], +value[1], +value[2], +value[3], +value[4], +value[5]);
    }

    function parseSvg(value) {
      if (value == null) return identity$2;
      if (!svgNode) svgNode = document.createElementNS("http://www.w3.org/2000/svg", "g");
      svgNode.setAttribute("transform", value);
      if (!(value = svgNode.transform.baseVal.consolidate())) return identity$2;
      value = value.matrix;
      return decompose(value.a, value.b, value.c, value.d, value.e, value.f);
    }

    function interpolateTransform(parse, pxComma, pxParen, degParen) {

      function pop(s) {
        return s.length ? s.pop() + " " : "";
      }

      function translate(xa, ya, xb, yb, s, q) {
        if (xa !== xb || ya !== yb) {
          var i = s.push("translate(", null, pxComma, null, pxParen);
          q.push({i: i - 4, x: interpolateNumber(xa, xb)}, {i: i - 2, x: interpolateNumber(ya, yb)});
        } else if (xb || yb) {
          s.push("translate(" + xb + pxComma + yb + pxParen);
        }
      }

      function rotate(a, b, s, q) {
        if (a !== b) {
          if (a - b > 180) b += 360; else if (b - a > 180) a += 360; // shortest path
          q.push({i: s.push(pop(s) + "rotate(", null, degParen) - 2, x: interpolateNumber(a, b)});
        } else if (b) {
          s.push(pop(s) + "rotate(" + b + degParen);
        }
      }

      function skewX(a, b, s, q) {
        if (a !== b) {
          q.push({i: s.push(pop(s) + "skewX(", null, degParen) - 2, x: interpolateNumber(a, b)});
        } else if (b) {
          s.push(pop(s) + "skewX(" + b + degParen);
        }
      }

      function scale(xa, ya, xb, yb, s, q) {
        if (xa !== xb || ya !== yb) {
          var i = s.push(pop(s) + "scale(", null, ",", null, ")");
          q.push({i: i - 4, x: interpolateNumber(xa, xb)}, {i: i - 2, x: interpolateNumber(ya, yb)});
        } else if (xb !== 1 || yb !== 1) {
          s.push(pop(s) + "scale(" + xb + "," + yb + ")");
        }
      }

      return function(a, b) {
        var s = [], // string constants and placeholders
            q = []; // number interpolators
        a = parse(a), b = parse(b);
        translate(a.translateX, a.translateY, b.translateX, b.translateY, s, q);
        rotate(a.rotate, b.rotate, s, q);
        skewX(a.skewX, b.skewX, s, q);
        scale(a.scaleX, a.scaleY, b.scaleX, b.scaleY, s, q);
        a = b = null; // gc
        return function(t) {
          var i = -1, n = q.length, o;
          while (++i < n) s[(o = q[i]).i] = o.x(t);
          return s.join("");
        };
      };
    }

    var interpolateTransformCss = interpolateTransform(parseCss$1, "px, ", "px)", "deg)");
    var interpolateTransformSvg = interpolateTransform(parseSvg, ", ", ")", ")");

    var rho = Math.SQRT2,
        rho2 = 2,
        rho4 = 4,
        epsilon2 = 1e-12;

    function cosh(x) {
      return ((x = Math.exp(x)) + 1 / x) / 2;
    }

    function sinh(x) {
      return ((x = Math.exp(x)) - 1 / x) / 2;
    }

    function tanh(x) {
      return ((x = Math.exp(2 * x)) - 1) / (x + 1);
    }

    // p0 = [ux0, uy0, w0]
    // p1 = [ux1, uy1, w1]
    function interpolateZoom(p0, p1) {
      var ux0 = p0[0], uy0 = p0[1], w0 = p0[2],
          ux1 = p1[0], uy1 = p1[1], w1 = p1[2],
          dx = ux1 - ux0,
          dy = uy1 - uy0,
          d2 = dx * dx + dy * dy,
          i,
          S;

      // Special case for u0 ≅ u1.
      if (d2 < epsilon2) {
        S = Math.log(w1 / w0) / rho;
        i = function(t) {
          return [
            ux0 + t * dx,
            uy0 + t * dy,
            w0 * Math.exp(rho * t * S)
          ];
        };
      }

      // General case.
      else {
        var d1 = Math.sqrt(d2),
            b0 = (w1 * w1 - w0 * w0 + rho4 * d2) / (2 * w0 * rho2 * d1),
            b1 = (w1 * w1 - w0 * w0 - rho4 * d2) / (2 * w1 * rho2 * d1),
            r0 = Math.log(Math.sqrt(b0 * b0 + 1) - b0),
            r1 = Math.log(Math.sqrt(b1 * b1 + 1) - b1);
        S = (r1 - r0) / rho;
        i = function(t) {
          var s = t * S,
              coshr0 = cosh(r0),
              u = w0 / (rho2 * d1) * (coshr0 * tanh(rho * s + r0) - sinh(r0));
          return [
            ux0 + u * dx,
            uy0 + u * dy,
            w0 * coshr0 / cosh(rho * s + r0)
          ];
        };
      }

      i.duration = S * 1000;

      return i;
    }

    function hsl$1(hue) {
      return function(start, end) {
        var h = hue((start = hsl(start)).h, (end = hsl(end)).h),
            s = nogamma(start.s, end.s),
            l = nogamma(start.l, end.l),
            opacity = nogamma(start.opacity, end.opacity);
        return function(t) {
          start.h = h(t);
          start.s = s(t);
          start.l = l(t);
          start.opacity = opacity(t);
          return start + "";
        };
      }
    }

    var hsl$2 = hsl$1(hue);
    var hslLong = hsl$1(nogamma);

    function lab$1(start, end) {
      var l = nogamma((start = lab(start)).l, (end = lab(end)).l),
          a = nogamma(start.a, end.a),
          b = nogamma(start.b, end.b),
          opacity = nogamma(start.opacity, end.opacity);
      return function(t) {
        start.l = l(t);
        start.a = a(t);
        start.b = b(t);
        start.opacity = opacity(t);
        return start + "";
      };
    }

    function hcl$1(hue) {
      return function(start, end) {
        var h = hue((start = hcl(start)).h, (end = hcl(end)).h),
            c = nogamma(start.c, end.c),
            l = nogamma(start.l, end.l),
            opacity = nogamma(start.opacity, end.opacity);
        return function(t) {
          start.h = h(t);
          start.c = c(t);
          start.l = l(t);
          start.opacity = opacity(t);
          return start + "";
        };
      }
    }

    var hcl$2 = hcl$1(hue);
    var hclLong = hcl$1(nogamma);

    function cubehelix$1(hue) {
      return (function cubehelixGamma(y) {
        y = +y;

        function cubehelix$1(start, end) {
          var h = hue((start = cubehelix(start)).h, (end = cubehelix(end)).h),
              s = nogamma(start.s, end.s),
              l = nogamma(start.l, end.l),
              opacity = nogamma(start.opacity, end.opacity);
          return function(t) {
            start.h = h(t);
            start.s = s(t);
            start.l = l(Math.pow(t, y));
            start.opacity = opacity(t);
            return start + "";
          };
        }

        cubehelix$1.gamma = cubehelixGamma;

        return cubehelix$1;
      })(1);
    }

    var cubehelix$2 = cubehelix$1(hue);
    var cubehelixLong = cubehelix$1(nogamma);

    function piecewise(interpolate, values) {
      var i = 0, n = values.length - 1, v = values[0], I = new Array(n < 0 ? 0 : n);
      while (i < n) I[i] = interpolate(v, v = values[++i]);
      return function(t) {
        var i = Math.max(0, Math.min(n - 1, Math.floor(t *= n)));
        return I[i](t - i);
      };
    }

    function quantize(interpolator, n) {
      var samples = new Array(n);
      for (var i = 0; i < n; ++i) samples[i] = interpolator(i / (n - 1));
      return samples;
    }

    var frame = 0, // is an animation frame pending?
        timeout = 0, // is a timeout pending?
        interval = 0, // are any timers active?
        pokeDelay = 1000, // how frequently we check for clock skew
        taskHead,
        taskTail,
        clockLast = 0,
        clockNow = 0,
        clockSkew = 0,
        clock = typeof performance === "object" && performance.now ? performance : Date,
        setFrame = typeof window === "object" && window.requestAnimationFrame ? window.requestAnimationFrame.bind(window) : function(f) { setTimeout(f, 17); };

    function now() {
      return clockNow || (setFrame(clearNow), clockNow = clock.now() + clockSkew);
    }

    function clearNow() {
      clockNow = 0;
    }

    function Timer() {
      this._call =
      this._time =
      this._next = null;
    }

    Timer.prototype = timer.prototype = {
      constructor: Timer,
      restart: function(callback, delay, time) {
        if (typeof callback !== "function") throw new TypeError("callback is not a function");
        time = (time == null ? now() : +time) + (delay == null ? 0 : +delay);
        if (!this._next && taskTail !== this) {
          if (taskTail) taskTail._next = this;
          else taskHead = this;
          taskTail = this;
        }
        this._call = callback;
        this._time = time;
        sleep();
      },
      stop: function() {
        if (this._call) {
          this._call = null;
          this._time = Infinity;
          sleep();
        }
      }
    };

    function timer(callback, delay, time) {
      var t = new Timer;
      t.restart(callback, delay, time);
      return t;
    }

    function timerFlush() {
      now(); // Get the current time, if not already set.
      ++frame; // Pretend we’ve set an alarm, if we haven’t already.
      var t = taskHead, e;
      while (t) {
        if ((e = clockNow - t._time) >= 0) t._call.call(null, e);
        t = t._next;
      }
      --frame;
    }

    function wake() {
      clockNow = (clockLast = clock.now()) + clockSkew;
      frame = timeout = 0;
      try {
        timerFlush();
      } finally {
        frame = 0;
        nap();
        clockNow = 0;
      }
    }

    function poke() {
      var now = clock.now(), delay = now - clockLast;
      if (delay > pokeDelay) clockSkew -= delay, clockLast = now;
    }

    function nap() {
      var t0, t1 = taskHead, t2, time = Infinity;
      while (t1) {
        if (t1._call) {
          if (time > t1._time) time = t1._time;
          t0 = t1, t1 = t1._next;
        } else {
          t2 = t1._next, t1._next = null;
          t1 = t0 ? t0._next = t2 : taskHead = t2;
        }
      }
      taskTail = t0;
      sleep(time);
    }

    function sleep(time) {
      if (frame) return; // Soonest alarm already set, or will be.
      if (timeout) timeout = clearTimeout(timeout);
      var delay = time - clockNow; // Strictly less than if we recomputed clockNow.
      if (delay > 24) {
        if (time < Infinity) timeout = setTimeout(wake, time - clock.now() - clockSkew);
        if (interval) interval = clearInterval(interval);
      } else {
        if (!interval) clockLast = clock.now(), interval = setInterval(poke, pokeDelay);
        frame = 1, setFrame(wake);
      }
    }

    function timeout$1(callback, delay, time) {
      var t = new Timer;
      delay = delay == null ? 0 : +delay;
      t.restart(function(elapsed) {
        t.stop();
        callback(elapsed + delay);
      }, delay, time);
      return t;
    }

    function interval$1(callback, delay, time) {
      var t = new Timer, total = delay;
      if (delay == null) return t.restart(callback, delay, time), t;
      delay = +delay, time = time == null ? now() : +time;
      t.restart(function tick(elapsed) {
        elapsed += total;
        t.restart(tick, total += delay, time);
        callback(elapsed);
      }, delay, time);
      return t;
    }

    var emptyOn = dispatch("start", "end", "cancel", "interrupt");
    var emptyTween = [];

    var CREATED = 0;
    var SCHEDULED = 1;
    var STARTING = 2;
    var STARTED = 3;
    var RUNNING = 4;
    var ENDING = 5;
    var ENDED = 6;

    function schedule$1(node, name, id, index, group, timing) {
      var schedules = node.__transition;
      if (!schedules) node.__transition = {};
      else if (id in schedules) return;
      create$1(node, id, {
        name: name,
        index: index, // For context during callback.
        group: group, // For context during callback.
        on: emptyOn,
        tween: emptyTween,
        time: timing.time,
        delay: timing.delay,
        duration: timing.duration,
        ease: timing.ease,
        timer: null,
        state: CREATED
      });
    }

    function init(node, id) {
      var schedule = get$2(node, id);
      if (schedule.state > CREATED) throw new Error("too late; already scheduled");
      return schedule;
    }

    function set$2(node, id) {
      var schedule = get$2(node, id);
      if (schedule.state > STARTED) throw new Error("too late; already running");
      return schedule;
    }

    function get$2(node, id) {
      var schedule = node.__transition;
      if (!schedule || !(schedule = schedule[id])) throw new Error("transition not found");
      return schedule;
    }

    function create$1(node, id, self) {
      var schedules = node.__transition,
          tween;

      // Initialize the self timer when the transition is created.
      // Note the actual delay is not known until the first callback!
      schedules[id] = self;
      self.timer = timer(schedule, 0, self.time);

      function schedule(elapsed) {
        self.state = SCHEDULED;
        self.timer.restart(start, self.delay, self.time);

        // If the elapsed delay is less than our first sleep, start immediately.
        if (self.delay <= elapsed) start(elapsed - self.delay);
      }

      function start(elapsed) {
        var i, j, n, o;

        // If the state is not SCHEDULED, then we previously errored on start.
        if (self.state !== SCHEDULED) return stop();

        for (i in schedules) {
          o = schedules[i];
          if (o.name !== self.name) continue;

          // While this element already has a starting transition during this frame,
          // defer starting an interrupting transition until that transition has a
          // chance to tick (and possibly end); see d3/d3-transition#54!
          if (o.state === STARTED) return timeout$1(start);

          // Interrupt the active transition, if any.
          if (o.state === RUNNING) {
            o.state = ENDED;
            o.timer.stop();
            o.on.call("interrupt", node, node.__data__, o.index, o.group);
            delete schedules[i];
          }

          // Cancel any pre-empted transitions.
          else if (+i < id) {
            o.state = ENDED;
            o.timer.stop();
            o.on.call("cancel", node, node.__data__, o.index, o.group);
            delete schedules[i];
          }
        }

        // Defer the first tick to end of the current frame; see d3/d3#1576.
        // Note the transition may be canceled after start and before the first tick!
        // Note this must be scheduled before the start event; see d3/d3-transition#16!
        // Assuming this is successful, subsequent callbacks go straight to tick.
        timeout$1(function() {
          if (self.state === STARTED) {
            self.state = RUNNING;
            self.timer.restart(tick, self.delay, self.time);
            tick(elapsed);
          }
        });

        // Dispatch the start event.
        // Note this must be done before the tween are initialized.
        self.state = STARTING;
        self.on.call("start", node, node.__data__, self.index, self.group);
        if (self.state !== STARTING) return; // interrupted
        self.state = STARTED;

        // Initialize the tween, deleting null tween.
        tween = new Array(n = self.tween.length);
        for (i = 0, j = -1; i < n; ++i) {
          if (o = self.tween[i].value.call(node, node.__data__, self.index, self.group)) {
            tween[++j] = o;
          }
        }
        tween.length = j + 1;
      }

      function tick(elapsed) {
        var t = elapsed < self.duration ? self.ease.call(null, elapsed / self.duration) : (self.timer.restart(stop), self.state = ENDING, 1),
            i = -1,
            n = tween.length;

        while (++i < n) {
          tween[i].call(node, t);
        }

        // Dispatch the end event.
        if (self.state === ENDING) {
          self.on.call("end", node, node.__data__, self.index, self.group);
          stop();
        }
      }

      function stop() {
        self.state = ENDED;
        self.timer.stop();
        delete schedules[id];
        for (var i in schedules) return; // eslint-disable-line no-unused-vars
        delete node.__transition;
      }
    }

    function interrupt(node, name) {
      var schedules = node.__transition,
          schedule,
          active,
          empty = true,
          i;

      if (!schedules) return;

      name = name == null ? null : name + "";

      for (i in schedules) {
        if ((schedule = schedules[i]).name !== name) { empty = false; continue; }
        active = schedule.state > STARTING && schedule.state < ENDING;
        schedule.state = ENDED;
        schedule.timer.stop();
        schedule.on.call(active ? "interrupt" : "cancel", node, node.__data__, schedule.index, schedule.group);
        delete schedules[i];
      }

      if (empty) delete node.__transition;
    }

    function selection_interrupt(name) {
      return this.each(function() {
        interrupt(this, name);
      });
    }

    function tweenRemove(id, name) {
      var tween0, tween1;
      return function() {
        var schedule = set$2(this, id),
            tween = schedule.tween;

        // If this node shared tween with the previous node,
        // just assign the updated shared tween and we’re done!
        // Otherwise, copy-on-write.
        if (tween !== tween0) {
          tween1 = tween0 = tween;
          for (var i = 0, n = tween1.length; i < n; ++i) {
            if (tween1[i].name === name) {
              tween1 = tween1.slice();
              tween1.splice(i, 1);
              break;
            }
          }
        }

        schedule.tween = tween1;
      };
    }

    function tweenFunction(id, name, value) {
      var tween0, tween1;
      if (typeof value !== "function") throw new Error;
      return function() {
        var schedule = set$2(this, id),
            tween = schedule.tween;

        // If this node shared tween with the previous node,
        // just assign the updated shared tween and we’re done!
        // Otherwise, copy-on-write.
        if (tween !== tween0) {
          tween1 = (tween0 = tween).slice();
          for (var t = {name: name, value: value}, i = 0, n = tween1.length; i < n; ++i) {
            if (tween1[i].name === name) {
              tween1[i] = t;
              break;
            }
          }
          if (i === n) tween1.push(t);
        }

        schedule.tween = tween1;
      };
    }

    function transition_tween(name, value) {
      var id = this._id;

      name += "";

      if (arguments.length < 2) {
        var tween = get$2(this.node(), id).tween;
        for (var i = 0, n = tween.length, t; i < n; ++i) {
          if ((t = tween[i]).name === name) {
            return t.value;
          }
        }
        return null;
      }

      return this.each((value == null ? tweenRemove : tweenFunction)(id, name, value));
    }

    function tweenValue(transition, name, value) {
      var id = transition._id;

      transition.each(function() {
        var schedule = set$2(this, id);
        (schedule.value || (schedule.value = {}))[name] = value.apply(this, arguments);
      });

      return function(node) {
        return get$2(node, id).value[name];
      };
    }

    function interpolate(a, b) {
      var c;
      return (typeof b === "number" ? interpolateNumber
          : b instanceof color ? interpolateRgb
          : (c = color(b)) ? (b = c, interpolateRgb)
          : interpolateString)(a, b);
    }

    function attrRemove$1(name) {
      return function() {
        this.removeAttribute(name);
      };
    }

    function attrRemoveNS$1(fullname) {
      return function() {
        this.removeAttributeNS(fullname.space, fullname.local);
      };
    }

    function attrConstant$1(name, interpolate, value1) {
      var string00,
          string1 = value1 + "",
          interpolate0;
      return function() {
        var string0 = this.getAttribute(name);
        return string0 === string1 ? null
            : string0 === string00 ? interpolate0
            : interpolate0 = interpolate(string00 = string0, value1);
      };
    }

    function attrConstantNS$1(fullname, interpolate, value1) {
      var string00,
          string1 = value1 + "",
          interpolate0;
      return function() {
        var string0 = this.getAttributeNS(fullname.space, fullname.local);
        return string0 === string1 ? null
            : string0 === string00 ? interpolate0
            : interpolate0 = interpolate(string00 = string0, value1);
      };
    }

    function attrFunction$1(name, interpolate, value) {
      var string00,
          string10,
          interpolate0;
      return function() {
        var string0, value1 = value(this), string1;
        if (value1 == null) return void this.removeAttribute(name);
        string0 = this.getAttribute(name);
        string1 = value1 + "";
        return string0 === string1 ? null
            : string0 === string00 && string1 === string10 ? interpolate0
            : (string10 = string1, interpolate0 = interpolate(string00 = string0, value1));
      };
    }

    function attrFunctionNS$1(fullname, interpolate, value) {
      var string00,
          string10,
          interpolate0;
      return function() {
        var string0, value1 = value(this), string1;
        if (value1 == null) return void this.removeAttributeNS(fullname.space, fullname.local);
        string0 = this.getAttributeNS(fullname.space, fullname.local);
        string1 = value1 + "";
        return string0 === string1 ? null
            : string0 === string00 && string1 === string10 ? interpolate0
            : (string10 = string1, interpolate0 = interpolate(string00 = string0, value1));
      };
    }

    function transition_attr(name, value) {
      var fullname = namespace(name), i = fullname === "transform" ? interpolateTransformSvg : interpolate;
      return this.attrTween(name, typeof value === "function"
          ? (fullname.local ? attrFunctionNS$1 : attrFunction$1)(fullname, i, tweenValue(this, "attr." + name, value))
          : value == null ? (fullname.local ? attrRemoveNS$1 : attrRemove$1)(fullname)
          : (fullname.local ? attrConstantNS$1 : attrConstant$1)(fullname, i, value));
    }

    function attrInterpolate(name, i) {
      return function(t) {
        this.setAttribute(name, i.call(this, t));
      };
    }

    function attrInterpolateNS(fullname, i) {
      return function(t) {
        this.setAttributeNS(fullname.space, fullname.local, i.call(this, t));
      };
    }

    function attrTweenNS(fullname, value) {
      var t0, i0;
      function tween() {
        var i = value.apply(this, arguments);
        if (i !== i0) t0 = (i0 = i) && attrInterpolateNS(fullname, i);
        return t0;
      }
      tween._value = value;
      return tween;
    }

    function attrTween(name, value) {
      var t0, i0;
      function tween() {
        var i = value.apply(this, arguments);
        if (i !== i0) t0 = (i0 = i) && attrInterpolate(name, i);
        return t0;
      }
      tween._value = value;
      return tween;
    }

    function transition_attrTween(name, value) {
      var key = "attr." + name;
      if (arguments.length < 2) return (key = this.tween(key)) && key._value;
      if (value == null) return this.tween(key, null);
      if (typeof value !== "function") throw new Error;
      var fullname = namespace(name);
      return this.tween(key, (fullname.local ? attrTweenNS : attrTween)(fullname, value));
    }

    function delayFunction(id, value) {
      return function() {
        init(this, id).delay = +value.apply(this, arguments);
      };
    }

    function delayConstant(id, value) {
      return value = +value, function() {
        init(this, id).delay = value;
      };
    }

    function transition_delay(value) {
      var id = this._id;

      return arguments.length
          ? this.each((typeof value === "function"
              ? delayFunction
              : delayConstant)(id, value))
          : get$2(this.node(), id).delay;
    }

    function durationFunction(id, value) {
      return function() {
        set$2(this, id).duration = +value.apply(this, arguments);
      };
    }

    function durationConstant(id, value) {
      return value = +value, function() {
        set$2(this, id).duration = value;
      };
    }

    function transition_duration(value) {
      var id = this._id;

      return arguments.length
          ? this.each((typeof value === "function"
              ? durationFunction
              : durationConstant)(id, value))
          : get$2(this.node(), id).duration;
    }

    function easeConstant(id, value) {
      if (typeof value !== "function") throw new Error;
      return function() {
        set$2(this, id).ease = value;
      };
    }

    function transition_ease(value) {
      var id = this._id;

      return arguments.length
          ? this.each(easeConstant(id, value))
          : get$2(this.node(), id).ease;
    }

    function transition_filter(match) {
      if (typeof match !== "function") match = matcher(match);

      for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, subgroup = subgroups[j] = [], node, i = 0; i < n; ++i) {
          if ((node = group[i]) && match.call(node, node.__data__, i, group)) {
            subgroup.push(node);
          }
        }
      }

      return new Transition(subgroups, this._parents, this._name, this._id);
    }

    function transition_merge(transition) {
      if (transition._id !== this._id) throw new Error;

      for (var groups0 = this._groups, groups1 = transition._groups, m0 = groups0.length, m1 = groups1.length, m = Math.min(m0, m1), merges = new Array(m0), j = 0; j < m; ++j) {
        for (var group0 = groups0[j], group1 = groups1[j], n = group0.length, merge = merges[j] = new Array(n), node, i = 0; i < n; ++i) {
          if (node = group0[i] || group1[i]) {
            merge[i] = node;
          }
        }
      }

      for (; j < m0; ++j) {
        merges[j] = groups0[j];
      }

      return new Transition(merges, this._parents, this._name, this._id);
    }

    function start(name) {
      return (name + "").trim().split(/^|\s+/).every(function(t) {
        var i = t.indexOf(".");
        if (i >= 0) t = t.slice(0, i);
        return !t || t === "start";
      });
    }

    function onFunction(id, name, listener) {
      var on0, on1, sit = start(name) ? init : set$2;
      return function() {
        var schedule = sit(this, id),
            on = schedule.on;

        // If this node shared a dispatch with the previous node,
        // just assign the updated shared dispatch and we’re done!
        // Otherwise, copy-on-write.
        if (on !== on0) (on1 = (on0 = on).copy()).on(name, listener);

        schedule.on = on1;
      };
    }

    function transition_on(name, listener) {
      var id = this._id;

      return arguments.length < 2
          ? get$2(this.node(), id).on.on(name)
          : this.each(onFunction(id, name, listener));
    }

    function removeFunction(id) {
      return function() {
        var parent = this.parentNode;
        for (var i in this.__transition) if (+i !== id) return;
        if (parent) parent.removeChild(this);
      };
    }

    function transition_remove() {
      return this.on("end.remove", removeFunction(this._id));
    }

    function transition_select(select) {
      var name = this._name,
          id = this._id;

      if (typeof select !== "function") select = selector(select);

      for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, subgroup = subgroups[j] = new Array(n), node, subnode, i = 0; i < n; ++i) {
          if ((node = group[i]) && (subnode = select.call(node, node.__data__, i, group))) {
            if ("__data__" in node) subnode.__data__ = node.__data__;
            subgroup[i] = subnode;
            schedule$1(subgroup[i], name, id, i, subgroup, get$2(node, id));
          }
        }
      }

      return new Transition(subgroups, this._parents, name, id);
    }

    function transition_selectAll(select) {
      var name = this._name,
          id = this._id;

      if (typeof select !== "function") select = selectorAll(select);

      for (var groups = this._groups, m = groups.length, subgroups = [], parents = [], j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, node, i = 0; i < n; ++i) {
          if (node = group[i]) {
            for (var children = select.call(node, node.__data__, i, group), child, inherit = get$2(node, id), k = 0, l = children.length; k < l; ++k) {
              if (child = children[k]) {
                schedule$1(child, name, id, k, children, inherit);
              }
            }
            subgroups.push(children);
            parents.push(node);
          }
        }
      }

      return new Transition(subgroups, parents, name, id);
    }

    var Selection$1 = selection.prototype.constructor;

    function transition_selection() {
      return new Selection$1(this._groups, this._parents);
    }

    function styleNull(name, interpolate) {
      var string00,
          string10,
          interpolate0;
      return function() {
        var string0 = styleValue(this, name),
            string1 = (this.style.removeProperty(name), styleValue(this, name));
        return string0 === string1 ? null
            : string0 === string00 && string1 === string10 ? interpolate0
            : interpolate0 = interpolate(string00 = string0, string10 = string1);
      };
    }

    function styleRemove$1(name) {
      return function() {
        this.style.removeProperty(name);
      };
    }

    function styleConstant$1(name, interpolate, value1) {
      var string00,
          string1 = value1 + "",
          interpolate0;
      return function() {
        var string0 = styleValue(this, name);
        return string0 === string1 ? null
            : string0 === string00 ? interpolate0
            : interpolate0 = interpolate(string00 = string0, value1);
      };
    }

    function styleFunction$1(name, interpolate, value) {
      var string00,
          string10,
          interpolate0;
      return function() {
        var string0 = styleValue(this, name),
            value1 = value(this),
            string1 = value1 + "";
        if (value1 == null) string1 = value1 = (this.style.removeProperty(name), styleValue(this, name));
        return string0 === string1 ? null
            : string0 === string00 && string1 === string10 ? interpolate0
            : (string10 = string1, interpolate0 = interpolate(string00 = string0, value1));
      };
    }

    function styleMaybeRemove(id, name) {
      var on0, on1, listener0, key = "style." + name, event = "end." + key, remove;
      return function() {
        var schedule = set$2(this, id),
            on = schedule.on,
            listener = schedule.value[key] == null ? remove || (remove = styleRemove$1(name)) : undefined;

        // If this node shared a dispatch with the previous node,
        // just assign the updated shared dispatch and we’re done!
        // Otherwise, copy-on-write.
        if (on !== on0 || listener0 !== listener) (on1 = (on0 = on).copy()).on(event, listener0 = listener);

        schedule.on = on1;
      };
    }

    function transition_style(name, value, priority) {
      var i = (name += "") === "transform" ? interpolateTransformCss : interpolate;
      return value == null ? this
          .styleTween(name, styleNull(name, i))
          .on("end.style." + name, styleRemove$1(name))
        : typeof value === "function" ? this
          .styleTween(name, styleFunction$1(name, i, tweenValue(this, "style." + name, value)))
          .each(styleMaybeRemove(this._id, name))
        : this
          .styleTween(name, styleConstant$1(name, i, value), priority)
          .on("end.style." + name, null);
    }

    function styleInterpolate(name, i, priority) {
      return function(t) {
        this.style.setProperty(name, i.call(this, t), priority);
      };
    }

    function styleTween(name, value, priority) {
      var t, i0;
      function tween() {
        var i = value.apply(this, arguments);
        if (i !== i0) t = (i0 = i) && styleInterpolate(name, i, priority);
        return t;
      }
      tween._value = value;
      return tween;
    }

    function transition_styleTween(name, value, priority) {
      var key = "style." + (name += "");
      if (arguments.length < 2) return (key = this.tween(key)) && key._value;
      if (value == null) return this.tween(key, null);
      if (typeof value !== "function") throw new Error;
      return this.tween(key, styleTween(name, value, priority == null ? "" : priority));
    }

    function textConstant$1(value) {
      return function() {
        this.textContent = value;
      };
    }

    function textFunction$1(value) {
      return function() {
        var value1 = value(this);
        this.textContent = value1 == null ? "" : value1;
      };
    }

    function transition_text(value) {
      return this.tween("text", typeof value === "function"
          ? textFunction$1(tweenValue(this, "text", value))
          : textConstant$1(value == null ? "" : value + ""));
    }

    function textInterpolate(i) {
      return function(t) {
        this.textContent = i.call(this, t);
      };
    }

    function textTween(value) {
      var t0, i0;
      function tween() {
        var i = value.apply(this, arguments);
        if (i !== i0) t0 = (i0 = i) && textInterpolate(i);
        return t0;
      }
      tween._value = value;
      return tween;
    }

    function transition_textTween(value) {
      var key = "text";
      if (arguments.length < 1) return (key = this.tween(key)) && key._value;
      if (value == null) return this.tween(key, null);
      if (typeof value !== "function") throw new Error;
      return this.tween(key, textTween(value));
    }

    function transition_transition() {
      var name = this._name,
          id0 = this._id,
          id1 = newId();

      for (var groups = this._groups, m = groups.length, j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, node, i = 0; i < n; ++i) {
          if (node = group[i]) {
            var inherit = get$2(node, id0);
            schedule$1(node, name, id1, i, group, {
              time: inherit.time + inherit.delay + inherit.duration,
              delay: 0,
              duration: inherit.duration,
              ease: inherit.ease
            });
          }
        }
      }

      return new Transition(groups, this._parents, name, id1);
    }

    function transition_end() {
      var on0, on1, that = this, id = that._id, size = that.size();
      return new Promise(function(resolve, reject) {
        var cancel = {value: reject},
            end = {value: function() { if (--size === 0) resolve(); }};

        that.each(function() {
          var schedule = set$2(this, id),
              on = schedule.on;

          // If this node shared a dispatch with the previous node,
          // just assign the updated shared dispatch and we’re done!
          // Otherwise, copy-on-write.
          if (on !== on0) {
            on1 = (on0 = on).copy();
            on1._.cancel.push(cancel);
            on1._.interrupt.push(cancel);
            on1._.end.push(end);
          }

          schedule.on = on1;
        });
      });
    }

    var id = 0;

    function Transition(groups, parents, name, id) {
      this._groups = groups;
      this._parents = parents;
      this._name = name;
      this._id = id;
    }

    function transition(name) {
      return selection().transition(name);
    }

    function newId() {
      return ++id;
    }

    var selection_prototype = selection.prototype;

    Transition.prototype = transition.prototype = {
      constructor: Transition,
      select: transition_select,
      selectAll: transition_selectAll,
      filter: transition_filter,
      merge: transition_merge,
      selection: transition_selection,
      transition: transition_transition,
      call: selection_prototype.call,
      nodes: selection_prototype.nodes,
      node: selection_prototype.node,
      size: selection_prototype.size,
      empty: selection_prototype.empty,
      each: selection_prototype.each,
      on: transition_on,
      attr: transition_attr,
      attrTween: transition_attrTween,
      style: transition_style,
      styleTween: transition_styleTween,
      text: transition_text,
      textTween: transition_textTween,
      remove: transition_remove,
      tween: transition_tween,
      delay: transition_delay,
      duration: transition_duration,
      ease: transition_ease,
      end: transition_end
    };

    function linear$1(t) {
      return +t;
    }

    function quadIn(t) {
      return t * t;
    }

    function quadOut(t) {
      return t * (2 - t);
    }

    function quadInOut(t) {
      return ((t *= 2) <= 1 ? t * t : --t * (2 - t) + 1) / 2;
    }

    function cubicIn(t) {
      return t * t * t;
    }

    function cubicOut(t) {
      return --t * t * t + 1;
    }

    function cubicInOut(t) {
      return ((t *= 2) <= 1 ? t * t * t : (t -= 2) * t * t + 2) / 2;
    }

    var exponent = 3;

    var polyIn = (function custom(e) {
      e = +e;

      function polyIn(t) {
        return Math.pow(t, e);
      }

      polyIn.exponent = custom;

      return polyIn;
    })(exponent);

    var polyOut = (function custom(e) {
      e = +e;

      function polyOut(t) {
        return 1 - Math.pow(1 - t, e);
      }

      polyOut.exponent = custom;

      return polyOut;
    })(exponent);

    var polyInOut = (function custom(e) {
      e = +e;

      function polyInOut(t) {
        return ((t *= 2) <= 1 ? Math.pow(t, e) : 2 - Math.pow(2 - t, e)) / 2;
      }

      polyInOut.exponent = custom;

      return polyInOut;
    })(exponent);

    var pi = Math.PI,
        halfPi = pi / 2;

    function sinIn(t) {
      return 1 - Math.cos(t * halfPi);
    }

    function sinOut(t) {
      return Math.sin(t * halfPi);
    }

    function sinInOut(t) {
      return (1 - Math.cos(pi * t)) / 2;
    }

    function expIn(t) {
      return Math.pow(2, 10 * t - 10);
    }

    function expOut(t) {
      return 1 - Math.pow(2, -10 * t);
    }

    function expInOut(t) {
      return ((t *= 2) <= 1 ? Math.pow(2, 10 * t - 10) : 2 - Math.pow(2, 10 - 10 * t)) / 2;
    }

    function circleIn(t) {
      return 1 - Math.sqrt(1 - t * t);
    }

    function circleOut(t) {
      return Math.sqrt(1 - --t * t);
    }

    function circleInOut(t) {
      return ((t *= 2) <= 1 ? 1 - Math.sqrt(1 - t * t) : Math.sqrt(1 - (t -= 2) * t) + 1) / 2;
    }

    var b1 = 4 / 11,
        b2 = 6 / 11,
        b3 = 8 / 11,
        b4 = 3 / 4,
        b5 = 9 / 11,
        b6 = 10 / 11,
        b7 = 15 / 16,
        b8 = 21 / 22,
        b9 = 63 / 64,
        b0 = 1 / b1 / b1;

    function bounceIn(t) {
      return 1 - bounceOut(1 - t);
    }

    function bounceOut(t) {
      return (t = +t) < b1 ? b0 * t * t : t < b3 ? b0 * (t -= b2) * t + b4 : t < b6 ? b0 * (t -= b5) * t + b7 : b0 * (t -= b8) * t + b9;
    }

    function bounceInOut(t) {
      return ((t *= 2) <= 1 ? 1 - bounceOut(1 - t) : bounceOut(t - 1) + 1) / 2;
    }

    var overshoot = 1.70158;

    var backIn = (function custom(s) {
      s = +s;

      function backIn(t) {
        return t * t * ((s + 1) * t - s);
      }

      backIn.overshoot = custom;

      return backIn;
    })(overshoot);

    var backOut = (function custom(s) {
      s = +s;

      function backOut(t) {
        return --t * t * ((s + 1) * t + s) + 1;
      }

      backOut.overshoot = custom;

      return backOut;
    })(overshoot);

    var backInOut = (function custom(s) {
      s = +s;

      function backInOut(t) {
        return ((t *= 2) < 1 ? t * t * ((s + 1) * t - s) : (t -= 2) * t * ((s + 1) * t + s) + 2) / 2;
      }

      backInOut.overshoot = custom;

      return backInOut;
    })(overshoot);

    var tau = 2 * Math.PI,
        amplitude = 1,
        period = 0.3;

    var elasticIn = (function custom(a, p) {
      var s = Math.asin(1 / (a = Math.max(1, a))) * (p /= tau);

      function elasticIn(t) {
        return a * Math.pow(2, 10 * --t) * Math.sin((s - t) / p);
      }

      elasticIn.amplitude = function(a) { return custom(a, p * tau); };
      elasticIn.period = function(p) { return custom(a, p); };

      return elasticIn;
    })(amplitude, period);

    var elasticOut = (function custom(a, p) {
      var s = Math.asin(1 / (a = Math.max(1, a))) * (p /= tau);

      function elasticOut(t) {
        return 1 - a * Math.pow(2, -10 * (t = +t)) * Math.sin((t + s) / p);
      }

      elasticOut.amplitude = function(a) { return custom(a, p * tau); };
      elasticOut.period = function(p) { return custom(a, p); };

      return elasticOut;
    })(amplitude, period);

    var elasticInOut = (function custom(a, p) {
      var s = Math.asin(1 / (a = Math.max(1, a))) * (p /= tau);

      function elasticInOut(t) {
        return ((t = t * 2 - 1) < 0
            ? a * Math.pow(2, 10 * t) * Math.sin((s - t) / p)
            : 2 - a * Math.pow(2, -10 * t) * Math.sin((s + t) / p)) / 2;
      }

      elasticInOut.amplitude = function(a) { return custom(a, p * tau); };
      elasticInOut.period = function(p) { return custom(a, p); };

      return elasticInOut;
    })(amplitude, period);

    var src = /*#__PURE__*/Object.freeze({
        __proto__: null,
        easeLinear: linear$1,
        easeQuad: quadInOut,
        easeQuadIn: quadIn,
        easeQuadOut: quadOut,
        easeQuadInOut: quadInOut,
        easeCubic: cubicInOut,
        easeCubicIn: cubicIn,
        easeCubicOut: cubicOut,
        easeCubicInOut: cubicInOut,
        easePoly: polyInOut,
        easePolyIn: polyIn,
        easePolyOut: polyOut,
        easePolyInOut: polyInOut,
        easeSin: sinInOut,
        easeSinIn: sinIn,
        easeSinOut: sinOut,
        easeSinInOut: sinInOut,
        easeExp: expInOut,
        easeExpIn: expIn,
        easeExpOut: expOut,
        easeExpInOut: expInOut,
        easeCircle: circleInOut,
        easeCircleIn: circleIn,
        easeCircleOut: circleOut,
        easeCircleInOut: circleInOut,
        easeBounce: bounceOut,
        easeBounceIn: bounceIn,
        easeBounceOut: bounceOut,
        easeBounceInOut: bounceInOut,
        easeBack: backInOut,
        easeBackIn: backIn,
        easeBackOut: backOut,
        easeBackInOut: backInOut,
        easeElastic: elasticOut,
        easeElasticIn: elasticIn,
        easeElasticOut: elasticOut,
        easeElasticInOut: elasticInOut
    });

    var defaultTiming = {
      time: null, // Set on use.
      delay: 0,
      duration: 250,
      ease: cubicInOut
    };

    function inherit(node, id) {
      var timing;
      while (!(timing = node.__transition) || !(timing = timing[id])) {
        if (!(node = node.parentNode)) {
          return defaultTiming.time = now(), defaultTiming;
        }
      }
      return timing;
    }

    function selection_transition(name) {
      var id,
          timing;

      if (name instanceof Transition) {
        id = name._id, name = name._name;
      } else {
        id = newId(), (timing = defaultTiming).time = now(), name = name == null ? null : name + "";
      }

      for (var groups = this._groups, m = groups.length, j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, node, i = 0; i < n; ++i) {
          if (node = group[i]) {
            schedule$1(node, name, id, i, group, timing || inherit(node, id));
          }
        }
      }

      return new Transition(groups, this._parents, name, id);
    }

    selection.prototype.interrupt = selection_interrupt;
    selection.prototype.transition = selection_transition;

    var root$2 = [null];

    function active(node, name) {
      var schedules = node.__transition,
          schedule,
          i;

      if (schedules) {
        name = name == null ? null : name + "";
        for (i in schedules) {
          if ((schedule = schedules[i]).state > SCHEDULED && schedule.name === name) {
            return new Transition([[node]], root$2, name, +i);
          }
        }
      }

      return null;
    }

    function constant$4(x) {
      return function() {
        return x;
      };
    }

    function BrushEvent(target, type, selection) {
      this.target = target;
      this.type = type;
      this.selection = selection;
    }

    function nopropagation$1() {
      event.stopImmediatePropagation();
    }

    function noevent$1() {
      event.preventDefault();
      event.stopImmediatePropagation();
    }

    var MODE_DRAG = {name: "drag"},
        MODE_SPACE = {name: "space"},
        MODE_HANDLE = {name: "handle"},
        MODE_CENTER = {name: "center"};

    function number1(e) {
      return [+e[0], +e[1]];
    }

    function number2(e) {
      return [number1(e[0]), number1(e[1])];
    }

    function toucher(identifier) {
      return function(target) {
        return touch(target, event.touches, identifier);
      };
    }

    var X = {
      name: "x",
      handles: ["w", "e"].map(type),
      input: function(x, e) { return x == null ? null : [[+x[0], e[0][1]], [+x[1], e[1][1]]]; },
      output: function(xy) { return xy && [xy[0][0], xy[1][0]]; }
    };

    var Y = {
      name: "y",
      handles: ["n", "s"].map(type),
      input: function(y, e) { return y == null ? null : [[e[0][0], +y[0]], [e[1][0], +y[1]]]; },
      output: function(xy) { return xy && [xy[0][1], xy[1][1]]; }
    };

    var XY = {
      name: "xy",
      handles: ["n", "w", "e", "s", "nw", "ne", "sw", "se"].map(type),
      input: function(xy) { return xy == null ? null : number2(xy); },
      output: function(xy) { return xy; }
    };

    var cursors = {
      overlay: "crosshair",
      selection: "move",
      n: "ns-resize",
      e: "ew-resize",
      s: "ns-resize",
      w: "ew-resize",
      nw: "nwse-resize",
      ne: "nesw-resize",
      se: "nwse-resize",
      sw: "nesw-resize"
    };

    var flipX = {
      e: "w",
      w: "e",
      nw: "ne",
      ne: "nw",
      se: "sw",
      sw: "se"
    };

    var flipY = {
      n: "s",
      s: "n",
      nw: "sw",
      ne: "se",
      se: "ne",
      sw: "nw"
    };

    var signsX = {
      overlay: +1,
      selection: +1,
      n: null,
      e: +1,
      s: null,
      w: -1,
      nw: -1,
      ne: +1,
      se: +1,
      sw: -1
    };

    var signsY = {
      overlay: +1,
      selection: +1,
      n: -1,
      e: null,
      s: +1,
      w: null,
      nw: -1,
      ne: -1,
      se: +1,
      sw: +1
    };

    function type(t) {
      return {type: t};
    }

    // Ignore right-click, since that should open the context menu.
    function defaultFilter$1() {
      return !event.ctrlKey && !event.button;
    }

    function defaultExtent() {
      var svg = this.ownerSVGElement || this;
      if (svg.hasAttribute("viewBox")) {
        svg = svg.viewBox.baseVal;
        return [[svg.x, svg.y], [svg.x + svg.width, svg.y + svg.height]];
      }
      return [[0, 0], [svg.width.baseVal.value, svg.height.baseVal.value]];
    }

    function defaultTouchable$1() {
      return navigator.maxTouchPoints || ("ontouchstart" in this);
    }

    // Like d3.local, but with the name “__brush” rather than auto-generated.
    function local$1(node) {
      while (!node.__brush) if (!(node = node.parentNode)) return;
      return node.__brush;
    }

    function empty$1(extent) {
      return extent[0][0] === extent[1][0]
          || extent[0][1] === extent[1][1];
    }

    function brushSelection(node) {
      var state = node.__brush;
      return state ? state.dim.output(state.selection) : null;
    }

    function brushX() {
      return brush$1(X);
    }

    function brushY() {
      return brush$1(Y);
    }

    function brush() {
      return brush$1(XY);
    }

    function brush$1(dim) {
      var extent = defaultExtent,
          filter = defaultFilter$1,
          touchable = defaultTouchable$1,
          keys = true,
          listeners = dispatch("start", "brush", "end"),
          handleSize = 6,
          touchending;

      function brush(group) {
        var overlay = group
            .property("__brush", initialize)
          .selectAll(".overlay")
          .data([type("overlay")]);

        overlay.enter().append("rect")
            .attr("class", "overlay")
            .attr("pointer-events", "all")
            .attr("cursor", cursors.overlay)
          .merge(overlay)
            .each(function() {
              var extent = local$1(this).extent;
              select(this)
                  .attr("x", extent[0][0])
                  .attr("y", extent[0][1])
                  .attr("width", extent[1][0] - extent[0][0])
                  .attr("height", extent[1][1] - extent[0][1]);
            });

        group.selectAll(".selection")
          .data([type("selection")])
          .enter().append("rect")
            .attr("class", "selection")
            .attr("cursor", cursors.selection)
            .attr("fill", "#777")
            .attr("fill-opacity", 0.3)
            .attr("stroke", "#fff")
            .attr("shape-rendering", "crispEdges");

        var handle = group.selectAll(".handle")
          .data(dim.handles, function(d) { return d.type; });

        handle.exit().remove();

        handle.enter().append("rect")
            .attr("class", function(d) { return "handle handle--" + d.type; })
            .attr("cursor", function(d) { return cursors[d.type]; });

        group
            .each(redraw)
            .attr("fill", "none")
            .attr("pointer-events", "all")
            .on("mousedown.brush", started)
          .filter(touchable)
            .on("touchstart.brush", started)
            .on("touchmove.brush", touchmoved)
            .on("touchend.brush touchcancel.brush", touchended)
            .style("touch-action", "none")
            .style("-webkit-tap-highlight-color", "rgba(0,0,0,0)");
      }

      brush.move = function(group, selection) {
        if (group.selection) {
          group
              .on("start.brush", function() { emitter(this, arguments).beforestart().start(); })
              .on("interrupt.brush end.brush", function() { emitter(this, arguments).end(); })
              .tween("brush", function() {
                var that = this,
                    state = that.__brush,
                    emit = emitter(that, arguments),
                    selection0 = state.selection,
                    selection1 = dim.input(typeof selection === "function" ? selection.apply(this, arguments) : selection, state.extent),
                    i = interpolateValue(selection0, selection1);

                function tween(t) {
                  state.selection = t === 1 && selection1 === null ? null : i(t);
                  redraw.call(that);
                  emit.brush();
                }

                return selection0 !== null && selection1 !== null ? tween : tween(1);
              });
        } else {
          group
              .each(function() {
                var that = this,
                    args = arguments,
                    state = that.__brush,
                    selection1 = dim.input(typeof selection === "function" ? selection.apply(that, args) : selection, state.extent),
                    emit = emitter(that, args).beforestart();

                interrupt(that);
                state.selection = selection1 === null ? null : selection1;
                redraw.call(that);
                emit.start().brush().end();
              });
        }
      };

      brush.clear = function(group) {
        brush.move(group, null);
      };

      function redraw() {
        var group = select(this),
            selection = local$1(this).selection;

        if (selection) {
          group.selectAll(".selection")
              .style("display", null)
              .attr("x", selection[0][0])
              .attr("y", selection[0][1])
              .attr("width", selection[1][0] - selection[0][0])
              .attr("height", selection[1][1] - selection[0][1]);

          group.selectAll(".handle")
              .style("display", null)
              .attr("x", function(d) { return d.type[d.type.length - 1] === "e" ? selection[1][0] - handleSize / 2 : selection[0][0] - handleSize / 2; })
              .attr("y", function(d) { return d.type[0] === "s" ? selection[1][1] - handleSize / 2 : selection[0][1] - handleSize / 2; })
              .attr("width", function(d) { return d.type === "n" || d.type === "s" ? selection[1][0] - selection[0][0] + handleSize : handleSize; })
              .attr("height", function(d) { return d.type === "e" || d.type === "w" ? selection[1][1] - selection[0][1] + handleSize : handleSize; });
        }

        else {
          group.selectAll(".selection,.handle")
              .style("display", "none")
              .attr("x", null)
              .attr("y", null)
              .attr("width", null)
              .attr("height", null);
        }
      }

      function emitter(that, args, clean) {
        return (!clean && that.__brush.emitter) || new Emitter(that, args);
      }

      function Emitter(that, args) {
        this.that = that;
        this.args = args;
        this.state = that.__brush;
        this.active = 0;
      }

      Emitter.prototype = {
        beforestart: function() {
          if (++this.active === 1) this.state.emitter = this, this.starting = true;
          return this;
        },
        start: function() {
          if (this.starting) this.starting = false, this.emit("start");
          else this.emit("brush");
          return this;
        },
        brush: function() {
          this.emit("brush");
          return this;
        },
        end: function() {
          if (--this.active === 0) delete this.state.emitter, this.emit("end");
          return this;
        },
        emit: function(type) {
          customEvent(new BrushEvent(brush, type, dim.output(this.state.selection)), listeners.apply, listeners, [type, this.that, this.args]);
        }
      };

      function started() {
        if (touchending && !event.touches) return;
        if (!filter.apply(this, arguments)) return;

        var that = this,
            type = event.target.__data__.type,
            mode = (keys && event.metaKey ? type = "overlay" : type) === "selection" ? MODE_DRAG : (keys && event.altKey ? MODE_CENTER : MODE_HANDLE),
            signX = dim === Y ? null : signsX[type],
            signY = dim === X ? null : signsY[type],
            state = local$1(that),
            extent = state.extent,
            selection = state.selection,
            W = extent[0][0], w0, w1,
            N = extent[0][1], n0, n1,
            E = extent[1][0], e0, e1,
            S = extent[1][1], s0, s1,
            dx = 0,
            dy = 0,
            moving,
            shifting = signX && signY && keys && event.shiftKey,
            lockX,
            lockY,
            pointer = event.touches ? toucher(event.changedTouches[0].identifier) : mouse,
            point0 = pointer(that),
            point = point0,
            emit = emitter(that, arguments, true).beforestart();

        if (type === "overlay") {
          if (selection) moving = true;
          state.selection = selection = [
            [w0 = dim === Y ? W : point0[0], n0 = dim === X ? N : point0[1]],
            [e0 = dim === Y ? E : w0, s0 = dim === X ? S : n0]
          ];
        } else {
          w0 = selection[0][0];
          n0 = selection[0][1];
          e0 = selection[1][0];
          s0 = selection[1][1];
        }

        w1 = w0;
        n1 = n0;
        e1 = e0;
        s1 = s0;

        var group = select(that)
            .attr("pointer-events", "none");

        var overlay = group.selectAll(".overlay")
            .attr("cursor", cursors[type]);

        if (event.touches) {
          emit.moved = moved;
          emit.ended = ended;
        } else {
          var view = select(event.view)
              .on("mousemove.brush", moved, true)
              .on("mouseup.brush", ended, true);
          if (keys) view
              .on("keydown.brush", keydowned, true)
              .on("keyup.brush", keyupped, true);

          dragDisable(event.view);
        }

        nopropagation$1();
        interrupt(that);
        redraw.call(that);
        emit.start();

        function moved() {
          var point1 = pointer(that);
          if (shifting && !lockX && !lockY) {
            if (Math.abs(point1[0] - point[0]) > Math.abs(point1[1] - point[1])) lockY = true;
            else lockX = true;
          }
          point = point1;
          moving = true;
          noevent$1();
          move();
        }

        function move() {
          var t;

          dx = point[0] - point0[0];
          dy = point[1] - point0[1];

          switch (mode) {
            case MODE_SPACE:
            case MODE_DRAG: {
              if (signX) dx = Math.max(W - w0, Math.min(E - e0, dx)), w1 = w0 + dx, e1 = e0 + dx;
              if (signY) dy = Math.max(N - n0, Math.min(S - s0, dy)), n1 = n0 + dy, s1 = s0 + dy;
              break;
            }
            case MODE_HANDLE: {
              if (signX < 0) dx = Math.max(W - w0, Math.min(E - w0, dx)), w1 = w0 + dx, e1 = e0;
              else if (signX > 0) dx = Math.max(W - e0, Math.min(E - e0, dx)), w1 = w0, e1 = e0 + dx;
              if (signY < 0) dy = Math.max(N - n0, Math.min(S - n0, dy)), n1 = n0 + dy, s1 = s0;
              else if (signY > 0) dy = Math.max(N - s0, Math.min(S - s0, dy)), n1 = n0, s1 = s0 + dy;
              break;
            }
            case MODE_CENTER: {
              if (signX) w1 = Math.max(W, Math.min(E, w0 - dx * signX)), e1 = Math.max(W, Math.min(E, e0 + dx * signX));
              if (signY) n1 = Math.max(N, Math.min(S, n0 - dy * signY)), s1 = Math.max(N, Math.min(S, s0 + dy * signY));
              break;
            }
          }

          if (e1 < w1) {
            signX *= -1;
            t = w0, w0 = e0, e0 = t;
            t = w1, w1 = e1, e1 = t;
            if (type in flipX) overlay.attr("cursor", cursors[type = flipX[type]]);
          }

          if (s1 < n1) {
            signY *= -1;
            t = n0, n0 = s0, s0 = t;
            t = n1, n1 = s1, s1 = t;
            if (type in flipY) overlay.attr("cursor", cursors[type = flipY[type]]);
          }

          if (state.selection) selection = state.selection; // May be set by brush.move!
          if (lockX) w1 = selection[0][0], e1 = selection[1][0];
          if (lockY) n1 = selection[0][1], s1 = selection[1][1];

          if (selection[0][0] !== w1
              || selection[0][1] !== n1
              || selection[1][0] !== e1
              || selection[1][1] !== s1) {
            state.selection = [[w1, n1], [e1, s1]];
            redraw.call(that);
            emit.brush();
          }
        }

        function ended() {
          nopropagation$1();
          if (event.touches) {
            if (event.touches.length) return;
            if (touchending) clearTimeout(touchending);
            touchending = setTimeout(function() { touchending = null; }, 500); // Ghost clicks are delayed!
          } else {
            yesdrag(event.view, moving);
            view.on("keydown.brush keyup.brush mousemove.brush mouseup.brush", null);
          }
          group.attr("pointer-events", "all");
          overlay.attr("cursor", cursors.overlay);
          if (state.selection) selection = state.selection; // May be set by brush.move (on start)!
          if (empty$1(selection)) state.selection = null, redraw.call(that);
          emit.end();
        }

        function keydowned() {
          switch (event.keyCode) {
            case 16: { // SHIFT
              shifting = signX && signY;
              break;
            }
            case 18: { // ALT
              if (mode === MODE_HANDLE) {
                if (signX) e0 = e1 - dx * signX, w0 = w1 + dx * signX;
                if (signY) s0 = s1 - dy * signY, n0 = n1 + dy * signY;
                mode = MODE_CENTER;
                move();
              }
              break;
            }
            case 32: { // SPACE; takes priority over ALT
              if (mode === MODE_HANDLE || mode === MODE_CENTER) {
                if (signX < 0) e0 = e1 - dx; else if (signX > 0) w0 = w1 - dx;
                if (signY < 0) s0 = s1 - dy; else if (signY > 0) n0 = n1 - dy;
                mode = MODE_SPACE;
                overlay.attr("cursor", cursors.selection);
                move();
              }
              break;
            }
            default: return;
          }
          noevent$1();
        }

        function keyupped() {
          switch (event.keyCode) {
            case 16: { // SHIFT
              if (shifting) {
                lockX = lockY = shifting = false;
                move();
              }
              break;
            }
            case 18: { // ALT
              if (mode === MODE_CENTER) {
                if (signX < 0) e0 = e1; else if (signX > 0) w0 = w1;
                if (signY < 0) s0 = s1; else if (signY > 0) n0 = n1;
                mode = MODE_HANDLE;
                move();
              }
              break;
            }
            case 32: { // SPACE
              if (mode === MODE_SPACE) {
                if (event.altKey) {
                  if (signX) e0 = e1 - dx * signX, w0 = w1 + dx * signX;
                  if (signY) s0 = s1 - dy * signY, n0 = n1 + dy * signY;
                  mode = MODE_CENTER;
                } else {
                  if (signX < 0) e0 = e1; else if (signX > 0) w0 = w1;
                  if (signY < 0) s0 = s1; else if (signY > 0) n0 = n1;
                  mode = MODE_HANDLE;
                }
                overlay.attr("cursor", cursors[type]);
                move();
              }
              break;
            }
            default: return;
          }
          noevent$1();
        }
      }

      function touchmoved() {
        emitter(this, arguments).moved();
      }

      function touchended() {
        emitter(this, arguments).ended();
      }

      function initialize() {
        var state = this.__brush || {selection: null};
        state.extent = number2(extent.apply(this, arguments));
        state.dim = dim;
        return state;
      }

      brush.extent = function(_) {
        return arguments.length ? (extent = typeof _ === "function" ? _ : constant$4(number2(_)), brush) : extent;
      };

      brush.filter = function(_) {
        return arguments.length ? (filter = typeof _ === "function" ? _ : constant$4(!!_), brush) : filter;
      };

      brush.touchable = function(_) {
        return arguments.length ? (touchable = typeof _ === "function" ? _ : constant$4(!!_), brush) : touchable;
      };

      brush.handleSize = function(_) {
        return arguments.length ? (handleSize = +_, brush) : handleSize;
      };

      brush.keyModifiers = function(_) {
        return arguments.length ? (keys = !!_, brush) : keys;
      };

      brush.on = function() {
        var value = listeners.on.apply(listeners, arguments);
        return value === listeners ? brush : value;
      };

      return brush;
    }

    var cos = Math.cos;
    var sin = Math.sin;
    var pi$1 = Math.PI;
    var halfPi$1 = pi$1 / 2;
    var tau$1 = pi$1 * 2;
    var max$1 = Math.max;

    function compareValue(compare) {
      return function(a, b) {
        return compare(
          a.source.value + a.target.value,
          b.source.value + b.target.value
        );
      };
    }

    function chord() {
      var padAngle = 0,
          sortGroups = null,
          sortSubgroups = null,
          sortChords = null;

      function chord(matrix) {
        var n = matrix.length,
            groupSums = [],
            groupIndex = sequence(n),
            subgroupIndex = [],
            chords = [],
            groups = chords.groups = new Array(n),
            subgroups = new Array(n * n),
            k,
            x,
            x0,
            dx,
            i,
            j;

        // Compute the sum.
        k = 0, i = -1; while (++i < n) {
          x = 0, j = -1; while (++j < n) {
            x += matrix[i][j];
          }
          groupSums.push(x);
          subgroupIndex.push(sequence(n));
          k += x;
        }

        // Sort groups…
        if (sortGroups) groupIndex.sort(function(a, b) {
          return sortGroups(groupSums[a], groupSums[b]);
        });

        // Sort subgroups…
        if (sortSubgroups) subgroupIndex.forEach(function(d, i) {
          d.sort(function(a, b) {
            return sortSubgroups(matrix[i][a], matrix[i][b]);
          });
        });

        // Convert the sum to scaling factor for [0, 2pi].
        // TODO Allow start and end angle to be specified?
        // TODO Allow padding to be specified as percentage?
        k = max$1(0, tau$1 - padAngle * n) / k;
        dx = k ? padAngle : tau$1 / n;

        // Compute the start and end angle for each group and subgroup.
        // Note: Opera has a bug reordering object literal properties!
        x = 0, i = -1; while (++i < n) {
          x0 = x, j = -1; while (++j < n) {
            var di = groupIndex[i],
                dj = subgroupIndex[di][j],
                v = matrix[di][dj],
                a0 = x,
                a1 = x += v * k;
            subgroups[dj * n + di] = {
              index: di,
              subindex: dj,
              startAngle: a0,
              endAngle: a1,
              value: v
            };
          }
          groups[di] = {
            index: di,
            startAngle: x0,
            endAngle: x,
            value: groupSums[di]
          };
          x += dx;
        }

        // Generate chords for each (non-empty) subgroup-subgroup link.
        i = -1; while (++i < n) {
          j = i - 1; while (++j < n) {
            var source = subgroups[j * n + i],
                target = subgroups[i * n + j];
            if (source.value || target.value) {
              chords.push(source.value < target.value
                  ? {source: target, target: source}
                  : {source: source, target: target});
            }
          }
        }

        return sortChords ? chords.sort(sortChords) : chords;
      }

      chord.padAngle = function(_) {
        return arguments.length ? (padAngle = max$1(0, _), chord) : padAngle;
      };

      chord.sortGroups = function(_) {
        return arguments.length ? (sortGroups = _, chord) : sortGroups;
      };

      chord.sortSubgroups = function(_) {
        return arguments.length ? (sortSubgroups = _, chord) : sortSubgroups;
      };

      chord.sortChords = function(_) {
        return arguments.length ? (_ == null ? sortChords = null : (sortChords = compareValue(_))._ = _, chord) : sortChords && sortChords._;
      };

      return chord;
    }

    var slice$2 = Array.prototype.slice;

    function constant$5(x) {
      return function() {
        return x;
      };
    }

    var pi$2 = Math.PI,
        tau$2 = 2 * pi$2,
        epsilon$1 = 1e-6,
        tauEpsilon = tau$2 - epsilon$1;

    function Path() {
      this._x0 = this._y0 = // start of current subpath
      this._x1 = this._y1 = null; // end of current subpath
      this._ = "";
    }

    function path() {
      return new Path;
    }

    Path.prototype = path.prototype = {
      constructor: Path,
      moveTo: function(x, y) {
        this._ += "M" + (this._x0 = this._x1 = +x) + "," + (this._y0 = this._y1 = +y);
      },
      closePath: function() {
        if (this._x1 !== null) {
          this._x1 = this._x0, this._y1 = this._y0;
          this._ += "Z";
        }
      },
      lineTo: function(x, y) {
        this._ += "L" + (this._x1 = +x) + "," + (this._y1 = +y);
      },
      quadraticCurveTo: function(x1, y1, x, y) {
        this._ += "Q" + (+x1) + "," + (+y1) + "," + (this._x1 = +x) + "," + (this._y1 = +y);
      },
      bezierCurveTo: function(x1, y1, x2, y2, x, y) {
        this._ += "C" + (+x1) + "," + (+y1) + "," + (+x2) + "," + (+y2) + "," + (this._x1 = +x) + "," + (this._y1 = +y);
      },
      arcTo: function(x1, y1, x2, y2, r) {
        x1 = +x1, y1 = +y1, x2 = +x2, y2 = +y2, r = +r;
        var x0 = this._x1,
            y0 = this._y1,
            x21 = x2 - x1,
            y21 = y2 - y1,
            x01 = x0 - x1,
            y01 = y0 - y1,
            l01_2 = x01 * x01 + y01 * y01;

        // Is the radius negative? Error.
        if (r < 0) throw new Error("negative radius: " + r);

        // Is this path empty? Move to (x1,y1).
        if (this._x1 === null) {
          this._ += "M" + (this._x1 = x1) + "," + (this._y1 = y1);
        }

        // Or, is (x1,y1) coincident with (x0,y0)? Do nothing.
        else if (!(l01_2 > epsilon$1));

        // Or, are (x0,y0), (x1,y1) and (x2,y2) collinear?
        // Equivalently, is (x1,y1) coincident with (x2,y2)?
        // Or, is the radius zero? Line to (x1,y1).
        else if (!(Math.abs(y01 * x21 - y21 * x01) > epsilon$1) || !r) {
          this._ += "L" + (this._x1 = x1) + "," + (this._y1 = y1);
        }

        // Otherwise, draw an arc!
        else {
          var x20 = x2 - x0,
              y20 = y2 - y0,
              l21_2 = x21 * x21 + y21 * y21,
              l20_2 = x20 * x20 + y20 * y20,
              l21 = Math.sqrt(l21_2),
              l01 = Math.sqrt(l01_2),
              l = r * Math.tan((pi$2 - Math.acos((l21_2 + l01_2 - l20_2) / (2 * l21 * l01))) / 2),
              t01 = l / l01,
              t21 = l / l21;

          // If the start tangent is not coincident with (x0,y0), line to.
          if (Math.abs(t01 - 1) > epsilon$1) {
            this._ += "L" + (x1 + t01 * x01) + "," + (y1 + t01 * y01);
          }

          this._ += "A" + r + "," + r + ",0,0," + (+(y01 * x20 > x01 * y20)) + "," + (this._x1 = x1 + t21 * x21) + "," + (this._y1 = y1 + t21 * y21);
        }
      },
      arc: function(x, y, r, a0, a1, ccw) {
        x = +x, y = +y, r = +r, ccw = !!ccw;
        var dx = r * Math.cos(a0),
            dy = r * Math.sin(a0),
            x0 = x + dx,
            y0 = y + dy,
            cw = 1 ^ ccw,
            da = ccw ? a0 - a1 : a1 - a0;

        // Is the radius negative? Error.
        if (r < 0) throw new Error("negative radius: " + r);

        // Is this path empty? Move to (x0,y0).
        if (this._x1 === null) {
          this._ += "M" + x0 + "," + y0;
        }

        // Or, is (x0,y0) not coincident with the previous point? Line to (x0,y0).
        else if (Math.abs(this._x1 - x0) > epsilon$1 || Math.abs(this._y1 - y0) > epsilon$1) {
          this._ += "L" + x0 + "," + y0;
        }

        // Is this arc empty? We’re done.
        if (!r) return;

        // Does the angle go the wrong way? Flip the direction.
        if (da < 0) da = da % tau$2 + tau$2;

        // Is this a complete circle? Draw two arcs to complete the circle.
        if (da > tauEpsilon) {
          this._ += "A" + r + "," + r + ",0,1," + cw + "," + (x - dx) + "," + (y - dy) + "A" + r + "," + r + ",0,1," + cw + "," + (this._x1 = x0) + "," + (this._y1 = y0);
        }

        // Is this arc non-empty? Draw an arc!
        else if (da > epsilon$1) {
          this._ += "A" + r + "," + r + ",0," + (+(da >= pi$2)) + "," + cw + "," + (this._x1 = x + r * Math.cos(a1)) + "," + (this._y1 = y + r * Math.sin(a1));
        }
      },
      rect: function(x, y, w, h) {
        this._ += "M" + (this._x0 = this._x1 = +x) + "," + (this._y0 = this._y1 = +y) + "h" + (+w) + "v" + (+h) + "h" + (-w) + "Z";
      },
      toString: function() {
        return this._;
      }
    };

    function defaultSource(d) {
      return d.source;
    }

    function defaultTarget(d) {
      return d.target;
    }

    function defaultRadius(d) {
      return d.radius;
    }

    function defaultStartAngle(d) {
      return d.startAngle;
    }

    function defaultEndAngle(d) {
      return d.endAngle;
    }

    function ribbon() {
      var source = defaultSource,
          target = defaultTarget,
          radius = defaultRadius,
          startAngle = defaultStartAngle,
          endAngle = defaultEndAngle,
          context = null;

      function ribbon() {
        var buffer,
            argv = slice$2.call(arguments),
            s = source.apply(this, argv),
            t = target.apply(this, argv),
            sr = +radius.apply(this, (argv[0] = s, argv)),
            sa0 = startAngle.apply(this, argv) - halfPi$1,
            sa1 = endAngle.apply(this, argv) - halfPi$1,
            sx0 = sr * cos(sa0),
            sy0 = sr * sin(sa0),
            tr = +radius.apply(this, (argv[0] = t, argv)),
            ta0 = startAngle.apply(this, argv) - halfPi$1,
            ta1 = endAngle.apply(this, argv) - halfPi$1;

        if (!context) context = buffer = path();

        context.moveTo(sx0, sy0);
        context.arc(0, 0, sr, sa0, sa1);
        if (sa0 !== ta0 || sa1 !== ta1) { // TODO sr !== tr?
          context.quadraticCurveTo(0, 0, tr * cos(ta0), tr * sin(ta0));
          context.arc(0, 0, tr, ta0, ta1);
        }
        context.quadraticCurveTo(0, 0, sx0, sy0);
        context.closePath();

        if (buffer) return context = null, buffer + "" || null;
      }

      ribbon.radius = function(_) {
        return arguments.length ? (radius = typeof _ === "function" ? _ : constant$5(+_), ribbon) : radius;
      };

      ribbon.startAngle = function(_) {
        return arguments.length ? (startAngle = typeof _ === "function" ? _ : constant$5(+_), ribbon) : startAngle;
      };

      ribbon.endAngle = function(_) {
        return arguments.length ? (endAngle = typeof _ === "function" ? _ : constant$5(+_), ribbon) : endAngle;
      };

      ribbon.source = function(_) {
        return arguments.length ? (source = _, ribbon) : source;
      };

      ribbon.target = function(_) {
        return arguments.length ? (target = _, ribbon) : target;
      };

      ribbon.context = function(_) {
        return arguments.length ? ((context = _ == null ? null : _), ribbon) : context;
      };

      return ribbon;
    }

    var prefix = "$";

    function Map$1() {}

    Map$1.prototype = map$1.prototype = {
      constructor: Map$1,
      has: function(key) {
        return (prefix + key) in this;
      },
      get: function(key) {
        return this[prefix + key];
      },
      set: function(key, value) {
        this[prefix + key] = value;
        return this;
      },
      remove: function(key) {
        var property = prefix + key;
        return property in this && delete this[property];
      },
      clear: function() {
        for (var property in this) if (property[0] === prefix) delete this[property];
      },
      keys: function() {
        var keys = [];
        for (var property in this) if (property[0] === prefix) keys.push(property.slice(1));
        return keys;
      },
      values: function() {
        var values = [];
        for (var property in this) if (property[0] === prefix) values.push(this[property]);
        return values;
      },
      entries: function() {
        var entries = [];
        for (var property in this) if (property[0] === prefix) entries.push({key: property.slice(1), value: this[property]});
        return entries;
      },
      size: function() {
        var size = 0;
        for (var property in this) if (property[0] === prefix) ++size;
        return size;
      },
      empty: function() {
        for (var property in this) if (property[0] === prefix) return false;
        return true;
      },
      each: function(f) {
        for (var property in this) if (property[0] === prefix) f(this[property], property.slice(1), this);
      }
    };

    function map$1(object, f) {
      var map = new Map$1;

      // Copy constructor.
      if (object instanceof Map$1) object.each(function(value, key) { map.set(key, value); });

      // Index array by numeric index or specified key function.
      else if (Array.isArray(object)) {
        var i = -1,
            n = object.length,
            o;

        if (f == null) while (++i < n) map.set(i, object[i]);
        else while (++i < n) map.set(f(o = object[i], i, object), o);
      }

      // Convert object to map.
      else if (object) for (var key in object) map.set(key, object[key]);

      return map;
    }

    function nest() {
      var keys = [],
          sortKeys = [],
          sortValues,
          rollup,
          nest;

      function apply(array, depth, createResult, setResult) {
        if (depth >= keys.length) {
          if (sortValues != null) array.sort(sortValues);
          return rollup != null ? rollup(array) : array;
        }

        var i = -1,
            n = array.length,
            key = keys[depth++],
            keyValue,
            value,
            valuesByKey = map$1(),
            values,
            result = createResult();

        while (++i < n) {
          if (values = valuesByKey.get(keyValue = key(value = array[i]) + "")) {
            values.push(value);
          } else {
            valuesByKey.set(keyValue, [value]);
          }
        }

        valuesByKey.each(function(values, key) {
          setResult(result, key, apply(values, depth, createResult, setResult));
        });

        return result;
      }

      function entries(map, depth) {
        if (++depth > keys.length) return map;
        var array, sortKey = sortKeys[depth - 1];
        if (rollup != null && depth >= keys.length) array = map.entries();
        else array = [], map.each(function(v, k) { array.push({key: k, values: entries(v, depth)}); });
        return sortKey != null ? array.sort(function(a, b) { return sortKey(a.key, b.key); }) : array;
      }

      return nest = {
        object: function(array) { return apply(array, 0, createObject, setObject$1); },
        map: function(array) { return apply(array, 0, createMap, setMap); },
        entries: function(array) { return entries(apply(array, 0, createMap, setMap), 0); },
        key: function(d) { keys.push(d); return nest; },
        sortKeys: function(order) { sortKeys[keys.length - 1] = order; return nest; },
        sortValues: function(order) { sortValues = order; return nest; },
        rollup: function(f) { rollup = f; return nest; }
      };
    }

    function createObject() {
      return {};
    }

    function setObject$1(object, key, value) {
      object[key] = value;
    }

    function createMap() {
      return map$1();
    }

    function setMap(map, key, value) {
      map.set(key, value);
    }

    function Set$1() {}

    var proto$1 = map$1.prototype;

    Set$1.prototype = set$3.prototype = {
      constructor: Set$1,
      has: proto$1.has,
      add: function(value) {
        value += "";
        this[prefix + value] = value;
        return this;
      },
      remove: proto$1.remove,
      clear: proto$1.clear,
      values: proto$1.keys,
      size: proto$1.size,
      empty: proto$1.empty,
      each: proto$1.each
    };

    function set$3(object, f) {
      var set = new Set$1;

      // Copy constructor.
      if (object instanceof Set$1) object.each(function(value) { set.add(value); });

      // Otherwise, assume it’s an array.
      else if (object) {
        var i = -1, n = object.length;
        if (f == null) while (++i < n) set.add(object[i]);
        else while (++i < n) set.add(f(object[i], i, object));
      }

      return set;
    }

    function keys(map) {
      var keys = [];
      for (var key in map) keys.push(key);
      return keys;
    }

    function values(map) {
      var values = [];
      for (var key in map) values.push(map[key]);
      return values;
    }

    function entries(map) {
      var entries = [];
      for (var key in map) entries.push({key: key, value: map[key]});
      return entries;
    }

    var array$2 = Array.prototype;

    var slice$3 = array$2.slice;

    function ascending$2(a, b) {
      return a - b;
    }

    function area(ring) {
      var i = 0, n = ring.length, area = ring[n - 1][1] * ring[0][0] - ring[n - 1][0] * ring[0][1];
      while (++i < n) area += ring[i - 1][1] * ring[i][0] - ring[i - 1][0] * ring[i][1];
      return area;
    }

    function constant$6(x) {
      return function() {
        return x;
      };
    }

    function contains(ring, hole) {
      var i = -1, n = hole.length, c;
      while (++i < n) if (c = ringContains(ring, hole[i])) return c;
      return 0;
    }

    function ringContains(ring, point) {
      var x = point[0], y = point[1], contains = -1;
      for (var i = 0, n = ring.length, j = n - 1; i < n; j = i++) {
        var pi = ring[i], xi = pi[0], yi = pi[1], pj = ring[j], xj = pj[0], yj = pj[1];
        if (segmentContains(pi, pj, point)) return 0;
        if (((yi > y) !== (yj > y)) && ((x < (xj - xi) * (y - yi) / (yj - yi) + xi))) contains = -contains;
      }
      return contains;
    }

    function segmentContains(a, b, c) {
      var i; return collinear(a, b, c) && within(a[i = +(a[0] === b[0])], c[i], b[i]);
    }

    function collinear(a, b, c) {
      return (b[0] - a[0]) * (c[1] - a[1]) === (c[0] - a[0]) * (b[1] - a[1]);
    }

    function within(p, q, r) {
      return p <= q && q <= r || r <= q && q <= p;
    }

    function noop$1() {}

    var cases = [
      [],
      [[[1.0, 1.5], [0.5, 1.0]]],
      [[[1.5, 1.0], [1.0, 1.5]]],
      [[[1.5, 1.0], [0.5, 1.0]]],
      [[[1.0, 0.5], [1.5, 1.0]]],
      [[[1.0, 1.5], [0.5, 1.0]], [[1.0, 0.5], [1.5, 1.0]]],
      [[[1.0, 0.5], [1.0, 1.5]]],
      [[[1.0, 0.5], [0.5, 1.0]]],
      [[[0.5, 1.0], [1.0, 0.5]]],
      [[[1.0, 1.5], [1.0, 0.5]]],
      [[[0.5, 1.0], [1.0, 0.5]], [[1.5, 1.0], [1.0, 1.5]]],
      [[[1.5, 1.0], [1.0, 0.5]]],
      [[[0.5, 1.0], [1.5, 1.0]]],
      [[[1.0, 1.5], [1.5, 1.0]]],
      [[[0.5, 1.0], [1.0, 1.5]]],
      []
    ];

    function contours() {
      var dx = 1,
          dy = 1,
          threshold = thresholdSturges,
          smooth = smoothLinear;

      function contours(values) {
        var tz = threshold(values);

        // Convert number of thresholds into uniform thresholds.
        if (!Array.isArray(tz)) {
          var domain = extent(values), start = domain[0], stop = domain[1];
          tz = tickStep(start, stop, tz);
          tz = sequence(Math.floor(start / tz) * tz, Math.floor(stop / tz) * tz, tz);
        } else {
          tz = tz.slice().sort(ascending$2);
        }

        return tz.map(function(value) {
          return contour(values, value);
        });
      }

      // Accumulate, smooth contour rings, assign holes to exterior rings.
      // Based on https://github.com/mbostock/shapefile/blob/v0.6.2/shp/polygon.js
      function contour(values, value) {
        var polygons = [],
            holes = [];

        isorings(values, value, function(ring) {
          smooth(ring, values, value);
          if (area(ring) > 0) polygons.push([ring]);
          else holes.push(ring);
        });

        holes.forEach(function(hole) {
          for (var i = 0, n = polygons.length, polygon; i < n; ++i) {
            if (contains((polygon = polygons[i])[0], hole) !== -1) {
              polygon.push(hole);
              return;
            }
          }
        });

        return {
          type: "MultiPolygon",
          value: value,
          coordinates: polygons
        };
      }

      // Marching squares with isolines stitched into rings.
      // Based on https://github.com/topojson/topojson-client/blob/v3.0.0/src/stitch.js
      function isorings(values, value, callback) {
        var fragmentByStart = new Array,
            fragmentByEnd = new Array,
            x, y, t0, t1, t2, t3;

        // Special case for the first row (y = -1, t2 = t3 = 0).
        x = y = -1;
        t1 = values[0] >= value;
        cases[t1 << 1].forEach(stitch);
        while (++x < dx - 1) {
          t0 = t1, t1 = values[x + 1] >= value;
          cases[t0 | t1 << 1].forEach(stitch);
        }
        cases[t1 << 0].forEach(stitch);

        // General case for the intermediate rows.
        while (++y < dy - 1) {
          x = -1;
          t1 = values[y * dx + dx] >= value;
          t2 = values[y * dx] >= value;
          cases[t1 << 1 | t2 << 2].forEach(stitch);
          while (++x < dx - 1) {
            t0 = t1, t1 = values[y * dx + dx + x + 1] >= value;
            t3 = t2, t2 = values[y * dx + x + 1] >= value;
            cases[t0 | t1 << 1 | t2 << 2 | t3 << 3].forEach(stitch);
          }
          cases[t1 | t2 << 3].forEach(stitch);
        }

        // Special case for the last row (y = dy - 1, t0 = t1 = 0).
        x = -1;
        t2 = values[y * dx] >= value;
        cases[t2 << 2].forEach(stitch);
        while (++x < dx - 1) {
          t3 = t2, t2 = values[y * dx + x + 1] >= value;
          cases[t2 << 2 | t3 << 3].forEach(stitch);
        }
        cases[t2 << 3].forEach(stitch);

        function stitch(line) {
          var start = [line[0][0] + x, line[0][1] + y],
              end = [line[1][0] + x, line[1][1] + y],
              startIndex = index(start),
              endIndex = index(end),
              f, g;
          if (f = fragmentByEnd[startIndex]) {
            if (g = fragmentByStart[endIndex]) {
              delete fragmentByEnd[f.end];
              delete fragmentByStart[g.start];
              if (f === g) {
                f.ring.push(end);
                callback(f.ring);
              } else {
                fragmentByStart[f.start] = fragmentByEnd[g.end] = {start: f.start, end: g.end, ring: f.ring.concat(g.ring)};
              }
            } else {
              delete fragmentByEnd[f.end];
              f.ring.push(end);
              fragmentByEnd[f.end = endIndex] = f;
            }
          } else if (f = fragmentByStart[endIndex]) {
            if (g = fragmentByEnd[startIndex]) {
              delete fragmentByStart[f.start];
              delete fragmentByEnd[g.end];
              if (f === g) {
                f.ring.push(end);
                callback(f.ring);
              } else {
                fragmentByStart[g.start] = fragmentByEnd[f.end] = {start: g.start, end: f.end, ring: g.ring.concat(f.ring)};
              }
            } else {
              delete fragmentByStart[f.start];
              f.ring.unshift(start);
              fragmentByStart[f.start = startIndex] = f;
            }
          } else {
            fragmentByStart[startIndex] = fragmentByEnd[endIndex] = {start: startIndex, end: endIndex, ring: [start, end]};
          }
        }
      }

      function index(point) {
        return point[0] * 2 + point[1] * (dx + 1) * 4;
      }

      function smoothLinear(ring, values, value) {
        ring.forEach(function(point) {
          var x = point[0],
              y = point[1],
              xt = x | 0,
              yt = y | 0,
              v0,
              v1 = values[yt * dx + xt];
          if (x > 0 && x < dx && xt === x) {
            v0 = values[yt * dx + xt - 1];
            point[0] = x + (value - v0) / (v1 - v0) - 0.5;
          }
          if (y > 0 && y < dy && yt === y) {
            v0 = values[(yt - 1) * dx + xt];
            point[1] = y + (value - v0) / (v1 - v0) - 0.5;
          }
        });
      }

      contours.contour = contour;

      contours.size = function(_) {
        if (!arguments.length) return [dx, dy];
        var _0 = Math.ceil(_[0]), _1 = Math.ceil(_[1]);
        if (!(_0 > 0) || !(_1 > 0)) throw new Error("invalid size");
        return dx = _0, dy = _1, contours;
      };

      contours.thresholds = function(_) {
        return arguments.length ? (threshold = typeof _ === "function" ? _ : Array.isArray(_) ? constant$6(slice$3.call(_)) : constant$6(_), contours) : threshold;
      };

      contours.smooth = function(_) {
        return arguments.length ? (smooth = _ ? smoothLinear : noop$1, contours) : smooth === smoothLinear;
      };

      return contours;
    }

    // TODO Optimize edge cases.
    // TODO Optimize index calculation.
    // TODO Optimize arguments.
    function blurX(source, target, r) {
      var n = source.width,
          m = source.height,
          w = (r << 1) + 1;
      for (var j = 0; j < m; ++j) {
        for (var i = 0, sr = 0; i < n + r; ++i) {
          if (i < n) {
            sr += source.data[i + j * n];
          }
          if (i >= r) {
            if (i >= w) {
              sr -= source.data[i - w + j * n];
            }
            target.data[i - r + j * n] = sr / Math.min(i + 1, n - 1 + w - i, w);
          }
        }
      }
    }

    // TODO Optimize edge cases.
    // TODO Optimize index calculation.
    // TODO Optimize arguments.
    function blurY(source, target, r) {
      var n = source.width,
          m = source.height,
          w = (r << 1) + 1;
      for (var i = 0; i < n; ++i) {
        for (var j = 0, sr = 0; j < m + r; ++j) {
          if (j < m) {
            sr += source.data[i + j * n];
          }
          if (j >= r) {
            if (j >= w) {
              sr -= source.data[i + (j - w) * n];
            }
            target.data[i + (j - r) * n] = sr / Math.min(j + 1, m - 1 + w - j, w);
          }
        }
      }
    }

    function defaultX(d) {
      return d[0];
    }

    function defaultY(d) {
      return d[1];
    }

    function defaultWeight() {
      return 1;
    }

    function density() {
      var x = defaultX,
          y = defaultY,
          weight = defaultWeight,
          dx = 960,
          dy = 500,
          r = 20, // blur radius
          k = 2, // log2(grid cell size)
          o = r * 3, // grid offset, to pad for blur
          n = (dx + o * 2) >> k, // grid width
          m = (dy + o * 2) >> k, // grid height
          threshold = constant$6(20);

      function density(data) {
        var values0 = new Float32Array(n * m),
            values1 = new Float32Array(n * m);

        data.forEach(function(d, i, data) {
          var xi = (+x(d, i, data) + o) >> k,
              yi = (+y(d, i, data) + o) >> k,
              wi = +weight(d, i, data);
          if (xi >= 0 && xi < n && yi >= 0 && yi < m) {
            values0[xi + yi * n] += wi;
          }
        });

        // TODO Optimize.
        blurX({width: n, height: m, data: values0}, {width: n, height: m, data: values1}, r >> k);
        blurY({width: n, height: m, data: values1}, {width: n, height: m, data: values0}, r >> k);
        blurX({width: n, height: m, data: values0}, {width: n, height: m, data: values1}, r >> k);
        blurY({width: n, height: m, data: values1}, {width: n, height: m, data: values0}, r >> k);
        blurX({width: n, height: m, data: values0}, {width: n, height: m, data: values1}, r >> k);
        blurY({width: n, height: m, data: values1}, {width: n, height: m, data: values0}, r >> k);

        var tz = threshold(values0);

        // Convert number of thresholds into uniform thresholds.
        if (!Array.isArray(tz)) {
          var stop = max(values0);
          tz = tickStep(0, stop, tz);
          tz = sequence(0, Math.floor(stop / tz) * tz, tz);
          tz.shift();
        }

        return contours()
            .thresholds(tz)
            .size([n, m])
          (values0)
            .map(transform);
      }

      function transform(geometry) {
        geometry.value *= Math.pow(2, -2 * k); // Density in points per square pixel.
        geometry.coordinates.forEach(transformPolygon);
        return geometry;
      }

      function transformPolygon(coordinates) {
        coordinates.forEach(transformRing);
      }

      function transformRing(coordinates) {
        coordinates.forEach(transformPoint);
      }

      // TODO Optimize.
      function transformPoint(coordinates) {
        coordinates[0] = coordinates[0] * Math.pow(2, k) - o;
        coordinates[1] = coordinates[1] * Math.pow(2, k) - o;
      }

      function resize() {
        o = r * 3;
        n = (dx + o * 2) >> k;
        m = (dy + o * 2) >> k;
        return density;
      }

      density.x = function(_) {
        return arguments.length ? (x = typeof _ === "function" ? _ : constant$6(+_), density) : x;
      };

      density.y = function(_) {
        return arguments.length ? (y = typeof _ === "function" ? _ : constant$6(+_), density) : y;
      };

      density.weight = function(_) {
        return arguments.length ? (weight = typeof _ === "function" ? _ : constant$6(+_), density) : weight;
      };

      density.size = function(_) {
        if (!arguments.length) return [dx, dy];
        var _0 = Math.ceil(_[0]), _1 = Math.ceil(_[1]);
        if (!(_0 >= 0) && !(_0 >= 0)) throw new Error("invalid size");
        return dx = _0, dy = _1, resize();
      };

      density.cellSize = function(_) {
        if (!arguments.length) return 1 << k;
        if (!((_ = +_) >= 1)) throw new Error("invalid cell size");
        return k = Math.floor(Math.log(_) / Math.LN2), resize();
      };

      density.thresholds = function(_) {
        return arguments.length ? (threshold = typeof _ === "function" ? _ : Array.isArray(_) ? constant$6(slice$3.call(_)) : constant$6(_), density) : threshold;
      };

      density.bandwidth = function(_) {
        if (!arguments.length) return Math.sqrt(r * (r + 1));
        if (!((_ = +_) >= 0)) throw new Error("invalid bandwidth");
        return r = Math.round((Math.sqrt(4 * _ * _ + 1) - 1) / 2), resize();
      };

      return density;
    }

    var EOL = {},
        EOF = {},
        QUOTE = 34,
        NEWLINE = 10,
        RETURN = 13;

    function objectConverter(columns) {
      return new Function("d", "return {" + columns.map(function(name, i) {
        return JSON.stringify(name) + ": d[" + i + "] || \"\"";
      }).join(",") + "}");
    }

    function customConverter(columns, f) {
      var object = objectConverter(columns);
      return function(row, i) {
        return f(object(row), i, columns);
      };
    }

    // Compute unique columns in order of discovery.
    function inferColumns(rows) {
      var columnSet = Object.create(null),
          columns = [];

      rows.forEach(function(row) {
        for (var column in row) {
          if (!(column in columnSet)) {
            columns.push(columnSet[column] = column);
          }
        }
      });

      return columns;
    }

    function pad(value, width) {
      var s = value + "", length = s.length;
      return length < width ? new Array(width - length + 1).join(0) + s : s;
    }

    function formatYear(year) {
      return year < 0 ? "-" + pad(-year, 6)
        : year > 9999 ? "+" + pad(year, 6)
        : pad(year, 4);
    }

    function formatDate(date) {
      var hours = date.getUTCHours(),
          minutes = date.getUTCMinutes(),
          seconds = date.getUTCSeconds(),
          milliseconds = date.getUTCMilliseconds();
      return isNaN(date) ? "Invalid Date"
          : formatYear(date.getUTCFullYear()) + "-" + pad(date.getUTCMonth() + 1, 2) + "-" + pad(date.getUTCDate(), 2)
          + (milliseconds ? "T" + pad(hours, 2) + ":" + pad(minutes, 2) + ":" + pad(seconds, 2) + "." + pad(milliseconds, 3) + "Z"
          : seconds ? "T" + pad(hours, 2) + ":" + pad(minutes, 2) + ":" + pad(seconds, 2) + "Z"
          : minutes || hours ? "T" + pad(hours, 2) + ":" + pad(minutes, 2) + "Z"
          : "");
    }

    function dsvFormat(delimiter) {
      var reFormat = new RegExp("[\"" + delimiter + "\n\r]"),
          DELIMITER = delimiter.charCodeAt(0);

      function parse(text, f) {
        var convert, columns, rows = parseRows(text, function(row, i) {
          if (convert) return convert(row, i - 1);
          columns = row, convert = f ? customConverter(row, f) : objectConverter(row);
        });
        rows.columns = columns || [];
        return rows;
      }

      function parseRows(text, f) {
        var rows = [], // output rows
            N = text.length,
            I = 0, // current character index
            n = 0, // current line number
            t, // current token
            eof = N <= 0, // current token followed by EOF?
            eol = false; // current token followed by EOL?

        // Strip the trailing newline.
        if (text.charCodeAt(N - 1) === NEWLINE) --N;
        if (text.charCodeAt(N - 1) === RETURN) --N;

        function token() {
          if (eof) return EOF;
          if (eol) return eol = false, EOL;

          // Unescape quotes.
          var i, j = I, c;
          if (text.charCodeAt(j) === QUOTE) {
            while (I++ < N && text.charCodeAt(I) !== QUOTE || text.charCodeAt(++I) === QUOTE);
            if ((i = I) >= N) eof = true;
            else if ((c = text.charCodeAt(I++)) === NEWLINE) eol = true;
            else if (c === RETURN) { eol = true; if (text.charCodeAt(I) === NEWLINE) ++I; }
            return text.slice(j + 1, i - 1).replace(/""/g, "\"");
          }

          // Find next delimiter or newline.
          while (I < N) {
            if ((c = text.charCodeAt(i = I++)) === NEWLINE) eol = true;
            else if (c === RETURN) { eol = true; if (text.charCodeAt(I) === NEWLINE) ++I; }
            else if (c !== DELIMITER) continue;
            return text.slice(j, i);
          }

          // Return last token before EOF.
          return eof = true, text.slice(j, N);
        }

        while ((t = token()) !== EOF) {
          var row = [];
          while (t !== EOL && t !== EOF) row.push(t), t = token();
          if (f && (row = f(row, n++)) == null) continue;
          rows.push(row);
        }

        return rows;
      }

      function preformatBody(rows, columns) {
        return rows.map(function(row) {
          return columns.map(function(column) {
            return formatValue(row[column]);
          }).join(delimiter);
        });
      }

      function format(rows, columns) {
        if (columns == null) columns = inferColumns(rows);
        return [columns.map(formatValue).join(delimiter)].concat(preformatBody(rows, columns)).join("\n");
      }

      function formatBody(rows, columns) {
        if (columns == null) columns = inferColumns(rows);
        return preformatBody(rows, columns).join("\n");
      }

      function formatRows(rows) {
        return rows.map(formatRow).join("\n");
      }

      function formatRow(row) {
        return row.map(formatValue).join(delimiter);
      }

      function formatValue(value) {
        return value == null ? ""
            : value instanceof Date ? formatDate(value)
            : reFormat.test(value += "") ? "\"" + value.replace(/"/g, "\"\"") + "\""
            : value;
      }

      return {
        parse: parse,
        parseRows: parseRows,
        format: format,
        formatBody: formatBody,
        formatRows: formatRows,
        formatRow: formatRow,
        formatValue: formatValue
      };
    }

    var csv = dsvFormat(",");

    var csvParse = csv.parse;
    var csvParseRows = csv.parseRows;
    var csvFormat = csv.format;
    var csvFormatBody = csv.formatBody;
    var csvFormatRows = csv.formatRows;
    var csvFormatRow = csv.formatRow;
    var csvFormatValue = csv.formatValue;

    var tsv = dsvFormat("\t");

    var tsvParse = tsv.parse;
    var tsvParseRows = tsv.parseRows;
    var tsvFormat = tsv.format;
    var tsvFormatBody = tsv.formatBody;
    var tsvFormatRows = tsv.formatRows;
    var tsvFormatRow = tsv.formatRow;
    var tsvFormatValue = tsv.formatValue;

    function autoType(object) {
      for (var key in object) {
        var value = object[key].trim(), number, m;
        if (!value) value = null;
        else if (value === "true") value = true;
        else if (value === "false") value = false;
        else if (value === "NaN") value = NaN;
        else if (!isNaN(number = +value)) value = number;
        else if (m = value.match(/^([-+]\d{2})?\d{4}(-\d{2}(-\d{2})?)?(T\d{2}:\d{2}(:\d{2}(\.\d{3})?)?(Z|[-+]\d{2}:\d{2})?)?$/)) {
          if (fixtz && !!m[4] && !m[7]) value = value.replace(/-/g, "/").replace(/T/, " ");
          value = new Date(value);
        }
        else continue;
        object[key] = value;
      }
      return object;
    }

    // https://github.com/d3/d3-dsv/issues/45
    var fixtz = new Date("2019-01-01T00:00").getHours() || new Date("2019-07-01T00:00").getHours();

    function responseBlob(response) {
      if (!response.ok) throw new Error(response.status + " " + response.statusText);
      return response.blob();
    }

    function blob(input, init) {
      return fetch(input, init).then(responseBlob);
    }

    function responseArrayBuffer(response) {
      if (!response.ok) throw new Error(response.status + " " + response.statusText);
      return response.arrayBuffer();
    }

    function buffer(input, init) {
      return fetch(input, init).then(responseArrayBuffer);
    }

    function responseText(response) {
      if (!response.ok) throw new Error(response.status + " " + response.statusText);
      return response.text();
    }

    function text(input, init) {
      return fetch(input, init).then(responseText);
    }

    function dsvParse(parse) {
      return function(input, init, row) {
        if (arguments.length === 2 && typeof init === "function") row = init, init = undefined;
        return text(input, init).then(function(response) {
          return parse(response, row);
        });
      };
    }

    function dsv(delimiter, input, init, row) {
      if (arguments.length === 3 && typeof init === "function") row = init, init = undefined;
      var format = dsvFormat(delimiter);
      return text(input, init).then(function(response) {
        return format.parse(response, row);
      });
    }

    var csv$1 = dsvParse(csvParse);
    var tsv$1 = dsvParse(tsvParse);

    function image(input, init) {
      return new Promise(function(resolve, reject) {
        var image = new Image;
        for (var key in init) image[key] = init[key];
        image.onerror = reject;
        image.onload = function() { resolve(image); };
        image.src = input;
      });
    }

    function responseJson(response) {
      if (!response.ok) throw new Error(response.status + " " + response.statusText);
      if (response.status === 204 || response.status === 205) return;
      return response.json();
    }

    function json(input, init) {
      return fetch(input, init).then(responseJson);
    }

    function parser(type) {
      return function(input, init)  {
        return text(input, init).then(function(text) {
          return (new DOMParser).parseFromString(text, type);
        });
      };
    }

    var xml = parser("application/xml");

    var html$1 = parser("text/html");

    var svg = parser("image/svg+xml");

    function center$1(x, y) {
      var nodes;

      if (x == null) x = 0;
      if (y == null) y = 0;

      function force() {
        var i,
            n = nodes.length,
            node,
            sx = 0,
            sy = 0;

        for (i = 0; i < n; ++i) {
          node = nodes[i], sx += node.x, sy += node.y;
        }

        for (sx = sx / n - x, sy = sy / n - y, i = 0; i < n; ++i) {
          node = nodes[i], node.x -= sx, node.y -= sy;
        }
      }

      force.initialize = function(_) {
        nodes = _;
      };

      force.x = function(_) {
        return arguments.length ? (x = +_, force) : x;
      };

      force.y = function(_) {
        return arguments.length ? (y = +_, force) : y;
      };

      return force;
    }

    function constant$7(x) {
      return function() {
        return x;
      };
    }

    function jiggle() {
      return (Math.random() - 0.5) * 1e-6;
    }

    function tree_add(d) {
      var x = +this._x.call(null, d),
          y = +this._y.call(null, d);
      return add(this.cover(x, y), x, y, d);
    }

    function add(tree, x, y, d) {
      if (isNaN(x) || isNaN(y)) return tree; // ignore invalid points

      var parent,
          node = tree._root,
          leaf = {data: d},
          x0 = tree._x0,
          y0 = tree._y0,
          x1 = tree._x1,
          y1 = tree._y1,
          xm,
          ym,
          xp,
          yp,
          right,
          bottom,
          i,
          j;

      // If the tree is empty, initialize the root as a leaf.
      if (!node) return tree._root = leaf, tree;

      // Find the existing leaf for the new point, or add it.
      while (node.length) {
        if (right = x >= (xm = (x0 + x1) / 2)) x0 = xm; else x1 = xm;
        if (bottom = y >= (ym = (y0 + y1) / 2)) y0 = ym; else y1 = ym;
        if (parent = node, !(node = node[i = bottom << 1 | right])) return parent[i] = leaf, tree;
      }

      // Is the new point is exactly coincident with the existing point?
      xp = +tree._x.call(null, node.data);
      yp = +tree._y.call(null, node.data);
      if (x === xp && y === yp) return leaf.next = node, parent ? parent[i] = leaf : tree._root = leaf, tree;

      // Otherwise, split the leaf node until the old and new point are separated.
      do {
        parent = parent ? parent[i] = new Array(4) : tree._root = new Array(4);
        if (right = x >= (xm = (x0 + x1) / 2)) x0 = xm; else x1 = xm;
        if (bottom = y >= (ym = (y0 + y1) / 2)) y0 = ym; else y1 = ym;
      } while ((i = bottom << 1 | right) === (j = (yp >= ym) << 1 | (xp >= xm)));
      return parent[j] = node, parent[i] = leaf, tree;
    }

    function addAll(data) {
      var d, i, n = data.length,
          x,
          y,
          xz = new Array(n),
          yz = new Array(n),
          x0 = Infinity,
          y0 = Infinity,
          x1 = -Infinity,
          y1 = -Infinity;

      // Compute the points and their extent.
      for (i = 0; i < n; ++i) {
        if (isNaN(x = +this._x.call(null, d = data[i])) || isNaN(y = +this._y.call(null, d))) continue;
        xz[i] = x;
        yz[i] = y;
        if (x < x0) x0 = x;
        if (x > x1) x1 = x;
        if (y < y0) y0 = y;
        if (y > y1) y1 = y;
      }

      // If there were no (valid) points, abort.
      if (x0 > x1 || y0 > y1) return this;

      // Expand the tree to cover the new points.
      this.cover(x0, y0).cover(x1, y1);

      // Add the new points.
      for (i = 0; i < n; ++i) {
        add(this, xz[i], yz[i], data[i]);
      }

      return this;
    }

    function tree_cover(x, y) {
      if (isNaN(x = +x) || isNaN(y = +y)) return this; // ignore invalid points

      var x0 = this._x0,
          y0 = this._y0,
          x1 = this._x1,
          y1 = this._y1;

      // If the quadtree has no extent, initialize them.
      // Integer extent are necessary so that if we later double the extent,
      // the existing quadrant boundaries don’t change due to floating point error!
      if (isNaN(x0)) {
        x1 = (x0 = Math.floor(x)) + 1;
        y1 = (y0 = Math.floor(y)) + 1;
      }

      // Otherwise, double repeatedly to cover.
      else {
        var z = x1 - x0,
            node = this._root,
            parent,
            i;

        while (x0 > x || x >= x1 || y0 > y || y >= y1) {
          i = (y < y0) << 1 | (x < x0);
          parent = new Array(4), parent[i] = node, node = parent, z *= 2;
          switch (i) {
            case 0: x1 = x0 + z, y1 = y0 + z; break;
            case 1: x0 = x1 - z, y1 = y0 + z; break;
            case 2: x1 = x0 + z, y0 = y1 - z; break;
            case 3: x0 = x1 - z, y0 = y1 - z; break;
          }
        }

        if (this._root && this._root.length) this._root = node;
      }

      this._x0 = x0;
      this._y0 = y0;
      this._x1 = x1;
      this._y1 = y1;
      return this;
    }

    function tree_data() {
      var data = [];
      this.visit(function(node) {
        if (!node.length) do data.push(node.data); while (node = node.next)
      });
      return data;
    }

    function tree_extent(_) {
      return arguments.length
          ? this.cover(+_[0][0], +_[0][1]).cover(+_[1][0], +_[1][1])
          : isNaN(this._x0) ? undefined : [[this._x0, this._y0], [this._x1, this._y1]];
    }

    function Quad(node, x0, y0, x1, y1) {
      this.node = node;
      this.x0 = x0;
      this.y0 = y0;
      this.x1 = x1;
      this.y1 = y1;
    }

    function tree_find(x, y, radius) {
      var data,
          x0 = this._x0,
          y0 = this._y0,
          x1,
          y1,
          x2,
          y2,
          x3 = this._x1,
          y3 = this._y1,
          quads = [],
          node = this._root,
          q,
          i;

      if (node) quads.push(new Quad(node, x0, y0, x3, y3));
      if (radius == null) radius = Infinity;
      else {
        x0 = x - radius, y0 = y - radius;
        x3 = x + radius, y3 = y + radius;
        radius *= radius;
      }

      while (q = quads.pop()) {

        // Stop searching if this quadrant can’t contain a closer node.
        if (!(node = q.node)
            || (x1 = q.x0) > x3
            || (y1 = q.y0) > y3
            || (x2 = q.x1) < x0
            || (y2 = q.y1) < y0) continue;

        // Bisect the current quadrant.
        if (node.length) {
          var xm = (x1 + x2) / 2,
              ym = (y1 + y2) / 2;

          quads.push(
            new Quad(node[3], xm, ym, x2, y2),
            new Quad(node[2], x1, ym, xm, y2),
            new Quad(node[1], xm, y1, x2, ym),
            new Quad(node[0], x1, y1, xm, ym)
          );

          // Visit the closest quadrant first.
          if (i = (y >= ym) << 1 | (x >= xm)) {
            q = quads[quads.length - 1];
            quads[quads.length - 1] = quads[quads.length - 1 - i];
            quads[quads.length - 1 - i] = q;
          }
        }

        // Visit this point. (Visiting coincident points isn’t necessary!)
        else {
          var dx = x - +this._x.call(null, node.data),
              dy = y - +this._y.call(null, node.data),
              d2 = dx * dx + dy * dy;
          if (d2 < radius) {
            var d = Math.sqrt(radius = d2);
            x0 = x - d, y0 = y - d;
            x3 = x + d, y3 = y + d;
            data = node.data;
          }
        }
      }

      return data;
    }

    function tree_remove(d) {
      if (isNaN(x = +this._x.call(null, d)) || isNaN(y = +this._y.call(null, d))) return this; // ignore invalid points

      var parent,
          node = this._root,
          retainer,
          previous,
          next,
          x0 = this._x0,
          y0 = this._y0,
          x1 = this._x1,
          y1 = this._y1,
          x,
          y,
          xm,
          ym,
          right,
          bottom,
          i,
          j;

      // If the tree is empty, initialize the root as a leaf.
      if (!node) return this;

      // Find the leaf node for the point.
      // While descending, also retain the deepest parent with a non-removed sibling.
      if (node.length) while (true) {
        if (right = x >= (xm = (x0 + x1) / 2)) x0 = xm; else x1 = xm;
        if (bottom = y >= (ym = (y0 + y1) / 2)) y0 = ym; else y1 = ym;
        if (!(parent = node, node = node[i = bottom << 1 | right])) return this;
        if (!node.length) break;
        if (parent[(i + 1) & 3] || parent[(i + 2) & 3] || parent[(i + 3) & 3]) retainer = parent, j = i;
      }

      // Find the point to remove.
      while (node.data !== d) if (!(previous = node, node = node.next)) return this;
      if (next = node.next) delete node.next;

      // If there are multiple coincident points, remove just the point.
      if (previous) return (next ? previous.next = next : delete previous.next), this;

      // If this is the root point, remove it.
      if (!parent) return this._root = next, this;

      // Remove this leaf.
      next ? parent[i] = next : delete parent[i];

      // If the parent now contains exactly one leaf, collapse superfluous parents.
      if ((node = parent[0] || parent[1] || parent[2] || parent[3])
          && node === (parent[3] || parent[2] || parent[1] || parent[0])
          && !node.length) {
        if (retainer) retainer[j] = node;
        else this._root = node;
      }

      return this;
    }

    function removeAll(data) {
      for (var i = 0, n = data.length; i < n; ++i) this.remove(data[i]);
      return this;
    }

    function tree_root() {
      return this._root;
    }

    function tree_size() {
      var size = 0;
      this.visit(function(node) {
        if (!node.length) do ++size; while (node = node.next)
      });
      return size;
    }

    function tree_visit(callback) {
      var quads = [], q, node = this._root, child, x0, y0, x1, y1;
      if (node) quads.push(new Quad(node, this._x0, this._y0, this._x1, this._y1));
      while (q = quads.pop()) {
        if (!callback(node = q.node, x0 = q.x0, y0 = q.y0, x1 = q.x1, y1 = q.y1) && node.length) {
          var xm = (x0 + x1) / 2, ym = (y0 + y1) / 2;
          if (child = node[3]) quads.push(new Quad(child, xm, ym, x1, y1));
          if (child = node[2]) quads.push(new Quad(child, x0, ym, xm, y1));
          if (child = node[1]) quads.push(new Quad(child, xm, y0, x1, ym));
          if (child = node[0]) quads.push(new Quad(child, x0, y0, xm, ym));
        }
      }
      return this;
    }

    function tree_visitAfter(callback) {
      var quads = [], next = [], q;
      if (this._root) quads.push(new Quad(this._root, this._x0, this._y0, this._x1, this._y1));
      while (q = quads.pop()) {
        var node = q.node;
        if (node.length) {
          var child, x0 = q.x0, y0 = q.y0, x1 = q.x1, y1 = q.y1, xm = (x0 + x1) / 2, ym = (y0 + y1) / 2;
          if (child = node[0]) quads.push(new Quad(child, x0, y0, xm, ym));
          if (child = node[1]) quads.push(new Quad(child, xm, y0, x1, ym));
          if (child = node[2]) quads.push(new Quad(child, x0, ym, xm, y1));
          if (child = node[3]) quads.push(new Quad(child, xm, ym, x1, y1));
        }
        next.push(q);
      }
      while (q = next.pop()) {
        callback(q.node, q.x0, q.y0, q.x1, q.y1);
      }
      return this;
    }

    function defaultX$1(d) {
      return d[0];
    }

    function tree_x(_) {
      return arguments.length ? (this._x = _, this) : this._x;
    }

    function defaultY$1(d) {
      return d[1];
    }

    function tree_y(_) {
      return arguments.length ? (this._y = _, this) : this._y;
    }

    function quadtree(nodes, x, y) {
      var tree = new Quadtree(x == null ? defaultX$1 : x, y == null ? defaultY$1 : y, NaN, NaN, NaN, NaN);
      return nodes == null ? tree : tree.addAll(nodes);
    }

    function Quadtree(x, y, x0, y0, x1, y1) {
      this._x = x;
      this._y = y;
      this._x0 = x0;
      this._y0 = y0;
      this._x1 = x1;
      this._y1 = y1;
      this._root = undefined;
    }

    function leaf_copy(leaf) {
      var copy = {data: leaf.data}, next = copy;
      while (leaf = leaf.next) next = next.next = {data: leaf.data};
      return copy;
    }

    var treeProto = quadtree.prototype = Quadtree.prototype;

    treeProto.copy = function() {
      var copy = new Quadtree(this._x, this._y, this._x0, this._y0, this._x1, this._y1),
          node = this._root,
          nodes,
          child;

      if (!node) return copy;

      if (!node.length) return copy._root = leaf_copy(node), copy;

      nodes = [{source: node, target: copy._root = new Array(4)}];
      while (node = nodes.pop()) {
        for (var i = 0; i < 4; ++i) {
          if (child = node.source[i]) {
            if (child.length) nodes.push({source: child, target: node.target[i] = new Array(4)});
            else node.target[i] = leaf_copy(child);
          }
        }
      }

      return copy;
    };

    treeProto.add = tree_add;
    treeProto.addAll = addAll;
    treeProto.cover = tree_cover;
    treeProto.data = tree_data;
    treeProto.extent = tree_extent;
    treeProto.find = tree_find;
    treeProto.remove = tree_remove;
    treeProto.removeAll = removeAll;
    treeProto.root = tree_root;
    treeProto.size = tree_size;
    treeProto.visit = tree_visit;
    treeProto.visitAfter = tree_visitAfter;
    treeProto.x = tree_x;
    treeProto.y = tree_y;

    function x(d) {
      return d.x + d.vx;
    }

    function y(d) {
      return d.y + d.vy;
    }

    function collide(radius) {
      var nodes,
          radii,
          strength = 1,
          iterations = 1;

      if (typeof radius !== "function") radius = constant$7(radius == null ? 1 : +radius);

      function force() {
        var i, n = nodes.length,
            tree,
            node,
            xi,
            yi,
            ri,
            ri2;

        for (var k = 0; k < iterations; ++k) {
          tree = quadtree(nodes, x, y).visitAfter(prepare);
          for (i = 0; i < n; ++i) {
            node = nodes[i];
            ri = radii[node.index], ri2 = ri * ri;
            xi = node.x + node.vx;
            yi = node.y + node.vy;
            tree.visit(apply);
          }
        }

        function apply(quad, x0, y0, x1, y1) {
          var data = quad.data, rj = quad.r, r = ri + rj;
          if (data) {
            if (data.index > node.index) {
              var x = xi - data.x - data.vx,
                  y = yi - data.y - data.vy,
                  l = x * x + y * y;
              if (l < r * r) {
                if (x === 0) x = jiggle(), l += x * x;
                if (y === 0) y = jiggle(), l += y * y;
                l = (r - (l = Math.sqrt(l))) / l * strength;
                node.vx += (x *= l) * (r = (rj *= rj) / (ri2 + rj));
                node.vy += (y *= l) * r;
                data.vx -= x * (r = 1 - r);
                data.vy -= y * r;
              }
            }
            return;
          }
          return x0 > xi + r || x1 < xi - r || y0 > yi + r || y1 < yi - r;
        }
      }

      function prepare(quad) {
        if (quad.data) return quad.r = radii[quad.data.index];
        for (var i = quad.r = 0; i < 4; ++i) {
          if (quad[i] && quad[i].r > quad.r) {
            quad.r = quad[i].r;
          }
        }
      }

      function initialize() {
        if (!nodes) return;
        var i, n = nodes.length, node;
        radii = new Array(n);
        for (i = 0; i < n; ++i) node = nodes[i], radii[node.index] = +radius(node, i, nodes);
      }

      force.initialize = function(_) {
        nodes = _;
        initialize();
      };

      force.iterations = function(_) {
        return arguments.length ? (iterations = +_, force) : iterations;
      };

      force.strength = function(_) {
        return arguments.length ? (strength = +_, force) : strength;
      };

      force.radius = function(_) {
        return arguments.length ? (radius = typeof _ === "function" ? _ : constant$7(+_), initialize(), force) : radius;
      };

      return force;
    }

    function index(d) {
      return d.index;
    }

    function find(nodeById, nodeId) {
      var node = nodeById.get(nodeId);
      if (!node) throw new Error("missing: " + nodeId);
      return node;
    }

    function link(links) {
      var id = index,
          strength = defaultStrength,
          strengths,
          distance = constant$7(30),
          distances,
          nodes,
          count,
          bias,
          iterations = 1;

      if (links == null) links = [];

      function defaultStrength(link) {
        return 1 / Math.min(count[link.source.index], count[link.target.index]);
      }

      function force(alpha) {
        for (var k = 0, n = links.length; k < iterations; ++k) {
          for (var i = 0, link, source, target, x, y, l, b; i < n; ++i) {
            link = links[i], source = link.source, target = link.target;
            x = target.x + target.vx - source.x - source.vx || jiggle();
            y = target.y + target.vy - source.y - source.vy || jiggle();
            l = Math.sqrt(x * x + y * y);
            l = (l - distances[i]) / l * alpha * strengths[i];
            x *= l, y *= l;
            target.vx -= x * (b = bias[i]);
            target.vy -= y * b;
            source.vx += x * (b = 1 - b);
            source.vy += y * b;
          }
        }
      }

      function initialize() {
        if (!nodes) return;

        var i,
            n = nodes.length,
            m = links.length,
            nodeById = map$1(nodes, id),
            link;

        for (i = 0, count = new Array(n); i < m; ++i) {
          link = links[i], link.index = i;
          if (typeof link.source !== "object") link.source = find(nodeById, link.source);
          if (typeof link.target !== "object") link.target = find(nodeById, link.target);
          count[link.source.index] = (count[link.source.index] || 0) + 1;
          count[link.target.index] = (count[link.target.index] || 0) + 1;
        }

        for (i = 0, bias = new Array(m); i < m; ++i) {
          link = links[i], bias[i] = count[link.source.index] / (count[link.source.index] + count[link.target.index]);
        }

        strengths = new Array(m), initializeStrength();
        distances = new Array(m), initializeDistance();
      }

      function initializeStrength() {
        if (!nodes) return;

        for (var i = 0, n = links.length; i < n; ++i) {
          strengths[i] = +strength(links[i], i, links);
        }
      }

      function initializeDistance() {
        if (!nodes) return;

        for (var i = 0, n = links.length; i < n; ++i) {
          distances[i] = +distance(links[i], i, links);
        }
      }

      force.initialize = function(_) {
        nodes = _;
        initialize();
      };

      force.links = function(_) {
        return arguments.length ? (links = _, initialize(), force) : links;
      };

      force.id = function(_) {
        return arguments.length ? (id = _, force) : id;
      };

      force.iterations = function(_) {
        return arguments.length ? (iterations = +_, force) : iterations;
      };

      force.strength = function(_) {
        return arguments.length ? (strength = typeof _ === "function" ? _ : constant$7(+_), initializeStrength(), force) : strength;
      };

      force.distance = function(_) {
        return arguments.length ? (distance = typeof _ === "function" ? _ : constant$7(+_), initializeDistance(), force) : distance;
      };

      return force;
    }

    function x$1(d) {
      return d.x;
    }

    function y$1(d) {
      return d.y;
    }

    var initialRadius = 10,
        initialAngle = Math.PI * (3 - Math.sqrt(5));

    function simulation(nodes) {
      var simulation,
          alpha = 1,
          alphaMin = 0.001,
          alphaDecay = 1 - Math.pow(alphaMin, 1 / 300),
          alphaTarget = 0,
          velocityDecay = 0.6,
          forces = map$1(),
          stepper = timer(step),
          event = dispatch("tick", "end");

      if (nodes == null) nodes = [];

      function step() {
        tick();
        event.call("tick", simulation);
        if (alpha < alphaMin) {
          stepper.stop();
          event.call("end", simulation);
        }
      }

      function tick(iterations) {
        var i, n = nodes.length, node;

        if (iterations === undefined) iterations = 1;

        for (var k = 0; k < iterations; ++k) {
          alpha += (alphaTarget - alpha) * alphaDecay;

          forces.each(function (force) {
            force(alpha);
          });

          for (i = 0; i < n; ++i) {
            node = nodes[i];
            if (node.fx == null) node.x += node.vx *= velocityDecay;
            else node.x = node.fx, node.vx = 0;
            if (node.fy == null) node.y += node.vy *= velocityDecay;
            else node.y = node.fy, node.vy = 0;
          }
        }

        return simulation;
      }

      function initializeNodes() {
        for (var i = 0, n = nodes.length, node; i < n; ++i) {
          node = nodes[i], node.index = i;
          if (node.fx != null) node.x = node.fx;
          if (node.fy != null) node.y = node.fy;
          if (isNaN(node.x) || isNaN(node.y)) {
            var radius = initialRadius * Math.sqrt(i), angle = i * initialAngle;
            node.x = radius * Math.cos(angle);
            node.y = radius * Math.sin(angle);
          }
          if (isNaN(node.vx) || isNaN(node.vy)) {
            node.vx = node.vy = 0;
          }
        }
      }

      function initializeForce(force) {
        if (force.initialize) force.initialize(nodes);
        return force;
      }

      initializeNodes();

      return simulation = {
        tick: tick,

        restart: function() {
          return stepper.restart(step), simulation;
        },

        stop: function() {
          return stepper.stop(), simulation;
        },

        nodes: function(_) {
          return arguments.length ? (nodes = _, initializeNodes(), forces.each(initializeForce), simulation) : nodes;
        },

        alpha: function(_) {
          return arguments.length ? (alpha = +_, simulation) : alpha;
        },

        alphaMin: function(_) {
          return arguments.length ? (alphaMin = +_, simulation) : alphaMin;
        },

        alphaDecay: function(_) {
          return arguments.length ? (alphaDecay = +_, simulation) : +alphaDecay;
        },

        alphaTarget: function(_) {
          return arguments.length ? (alphaTarget = +_, simulation) : alphaTarget;
        },

        velocityDecay: function(_) {
          return arguments.length ? (velocityDecay = 1 - _, simulation) : 1 - velocityDecay;
        },

        force: function(name, _) {
          return arguments.length > 1 ? ((_ == null ? forces.remove(name) : forces.set(name, initializeForce(_))), simulation) : forces.get(name);
        },

        find: function(x, y, radius) {
          var i = 0,
              n = nodes.length,
              dx,
              dy,
              d2,
              node,
              closest;

          if (radius == null) radius = Infinity;
          else radius *= radius;

          for (i = 0; i < n; ++i) {
            node = nodes[i];
            dx = x - node.x;
            dy = y - node.y;
            d2 = dx * dx + dy * dy;
            if (d2 < radius) closest = node, radius = d2;
          }

          return closest;
        },

        on: function(name, _) {
          return arguments.length > 1 ? (event.on(name, _), simulation) : event.on(name);
        }
      };
    }

    function manyBody() {
      var nodes,
          node,
          alpha,
          strength = constant$7(-30),
          strengths,
          distanceMin2 = 1,
          distanceMax2 = Infinity,
          theta2 = 0.81;

      function force(_) {
        var i, n = nodes.length, tree = quadtree(nodes, x$1, y$1).visitAfter(accumulate);
        for (alpha = _, i = 0; i < n; ++i) node = nodes[i], tree.visit(apply);
      }

      function initialize() {
        if (!nodes) return;
        var i, n = nodes.length, node;
        strengths = new Array(n);
        for (i = 0; i < n; ++i) node = nodes[i], strengths[node.index] = +strength(node, i, nodes);
      }

      function accumulate(quad) {
        var strength = 0, q, c, weight = 0, x, y, i;

        // For internal nodes, accumulate forces from child quadrants.
        if (quad.length) {
          for (x = y = i = 0; i < 4; ++i) {
            if ((q = quad[i]) && (c = Math.abs(q.value))) {
              strength += q.value, weight += c, x += c * q.x, y += c * q.y;
            }
          }
          quad.x = x / weight;
          quad.y = y / weight;
        }

        // For leaf nodes, accumulate forces from coincident quadrants.
        else {
          q = quad;
          q.x = q.data.x;
          q.y = q.data.y;
          do strength += strengths[q.data.index];
          while (q = q.next);
        }

        quad.value = strength;
      }

      function apply(quad, x1, _, x2) {
        if (!quad.value) return true;

        var x = quad.x - node.x,
            y = quad.y - node.y,
            w = x2 - x1,
            l = x * x + y * y;

        // Apply the Barnes-Hut approximation if possible.
        // Limit forces for very close nodes; randomize direction if coincident.
        if (w * w / theta2 < l) {
          if (l < distanceMax2) {
            if (x === 0) x = jiggle(), l += x * x;
            if (y === 0) y = jiggle(), l += y * y;
            if (l < distanceMin2) l = Math.sqrt(distanceMin2 * l);
            node.vx += x * quad.value * alpha / l;
            node.vy += y * quad.value * alpha / l;
          }
          return true;
        }

        // Otherwise, process points directly.
        else if (quad.length || l >= distanceMax2) return;

        // Limit forces for very close nodes; randomize direction if coincident.
        if (quad.data !== node || quad.next) {
          if (x === 0) x = jiggle(), l += x * x;
          if (y === 0) y = jiggle(), l += y * y;
          if (l < distanceMin2) l = Math.sqrt(distanceMin2 * l);
        }

        do if (quad.data !== node) {
          w = strengths[quad.data.index] * alpha / l;
          node.vx += x * w;
          node.vy += y * w;
        } while (quad = quad.next);
      }

      force.initialize = function(_) {
        nodes = _;
        initialize();
      };

      force.strength = function(_) {
        return arguments.length ? (strength = typeof _ === "function" ? _ : constant$7(+_), initialize(), force) : strength;
      };

      force.distanceMin = function(_) {
        return arguments.length ? (distanceMin2 = _ * _, force) : Math.sqrt(distanceMin2);
      };

      force.distanceMax = function(_) {
        return arguments.length ? (distanceMax2 = _ * _, force) : Math.sqrt(distanceMax2);
      };

      force.theta = function(_) {
        return arguments.length ? (theta2 = _ * _, force) : Math.sqrt(theta2);
      };

      return force;
    }

    function radial(radius, x, y) {
      var nodes,
          strength = constant$7(0.1),
          strengths,
          radiuses;

      if (typeof radius !== "function") radius = constant$7(+radius);
      if (x == null) x = 0;
      if (y == null) y = 0;

      function force(alpha) {
        for (var i = 0, n = nodes.length; i < n; ++i) {
          var node = nodes[i],
              dx = node.x - x || 1e-6,
              dy = node.y - y || 1e-6,
              r = Math.sqrt(dx * dx + dy * dy),
              k = (radiuses[i] - r) * strengths[i] * alpha / r;
          node.vx += dx * k;
          node.vy += dy * k;
        }
      }

      function initialize() {
        if (!nodes) return;
        var i, n = nodes.length;
        strengths = new Array(n);
        radiuses = new Array(n);
        for (i = 0; i < n; ++i) {
          radiuses[i] = +radius(nodes[i], i, nodes);
          strengths[i] = isNaN(radiuses[i]) ? 0 : +strength(nodes[i], i, nodes);
        }
      }

      force.initialize = function(_) {
        nodes = _, initialize();
      };

      force.strength = function(_) {
        return arguments.length ? (strength = typeof _ === "function" ? _ : constant$7(+_), initialize(), force) : strength;
      };

      force.radius = function(_) {
        return arguments.length ? (radius = typeof _ === "function" ? _ : constant$7(+_), initialize(), force) : radius;
      };

      force.x = function(_) {
        return arguments.length ? (x = +_, force) : x;
      };

      force.y = function(_) {
        return arguments.length ? (y = +_, force) : y;
      };

      return force;
    }

    function x$2(x) {
      var strength = constant$7(0.1),
          nodes,
          strengths,
          xz;

      if (typeof x !== "function") x = constant$7(x == null ? 0 : +x);

      function force(alpha) {
        for (var i = 0, n = nodes.length, node; i < n; ++i) {
          node = nodes[i], node.vx += (xz[i] - node.x) * strengths[i] * alpha;
        }
      }

      function initialize() {
        if (!nodes) return;
        var i, n = nodes.length;
        strengths = new Array(n);
        xz = new Array(n);
        for (i = 0; i < n; ++i) {
          strengths[i] = isNaN(xz[i] = +x(nodes[i], i, nodes)) ? 0 : +strength(nodes[i], i, nodes);
        }
      }

      force.initialize = function(_) {
        nodes = _;
        initialize();
      };

      force.strength = function(_) {
        return arguments.length ? (strength = typeof _ === "function" ? _ : constant$7(+_), initialize(), force) : strength;
      };

      force.x = function(_) {
        return arguments.length ? (x = typeof _ === "function" ? _ : constant$7(+_), initialize(), force) : x;
      };

      return force;
    }

    function y$2(y) {
      var strength = constant$7(0.1),
          nodes,
          strengths,
          yz;

      if (typeof y !== "function") y = constant$7(y == null ? 0 : +y);

      function force(alpha) {
        for (var i = 0, n = nodes.length, node; i < n; ++i) {
          node = nodes[i], node.vy += (yz[i] - node.y) * strengths[i] * alpha;
        }
      }

      function initialize() {
        if (!nodes) return;
        var i, n = nodes.length;
        strengths = new Array(n);
        yz = new Array(n);
        for (i = 0; i < n; ++i) {
          strengths[i] = isNaN(yz[i] = +y(nodes[i], i, nodes)) ? 0 : +strength(nodes[i], i, nodes);
        }
      }

      force.initialize = function(_) {
        nodes = _;
        initialize();
      };

      force.strength = function(_) {
        return arguments.length ? (strength = typeof _ === "function" ? _ : constant$7(+_), initialize(), force) : strength;
      };

      force.y = function(_) {
        return arguments.length ? (y = typeof _ === "function" ? _ : constant$7(+_), initialize(), force) : y;
      };

      return force;
    }

    // Computes the decimal coefficient and exponent of the specified number x with
    // significant digits p, where x is positive and p is in [1, 21] or undefined.
    // For example, formatDecimal(1.23) returns ["123", 0].
    function formatDecimal(x, p) {
      if ((i = (x = p ? x.toExponential(p - 1) : x.toExponential()).indexOf("e")) < 0) return null; // NaN, ±Infinity
      var i, coefficient = x.slice(0, i);

      // The string returned by toExponential either has the form \d\.\d+e[-+]\d+
      // (e.g., 1.2e+3) or the form \de[-+]\d+ (e.g., 1e+3).
      return [
        coefficient.length > 1 ? coefficient[0] + coefficient.slice(2) : coefficient,
        +x.slice(i + 1)
      ];
    }

    function exponent$1(x) {
      return x = formatDecimal(Math.abs(x)), x ? x[1] : NaN;
    }

    function formatGroup(grouping, thousands) {
      return function(value, width) {
        var i = value.length,
            t = [],
            j = 0,
            g = grouping[0],
            length = 0;

        while (i > 0 && g > 0) {
          if (length + g + 1 > width) g = Math.max(1, width - length);
          t.push(value.substring(i -= g, i + g));
          if ((length += g + 1) > width) break;
          g = grouping[j = (j + 1) % grouping.length];
        }

        return t.reverse().join(thousands);
      };
    }

    function formatNumerals(numerals) {
      return function(value) {
        return value.replace(/[0-9]/g, function(i) {
          return numerals[+i];
        });
      };
    }

    // [[fill]align][sign][symbol][0][width][,][.precision][~][type]
    var re = /^(?:(.)?([<>=^]))?([+\-( ])?([$#])?(0)?(\d+)?(,)?(\.\d+)?(~)?([a-z%])?$/i;

    function formatSpecifier(specifier) {
      if (!(match = re.exec(specifier))) throw new Error("invalid format: " + specifier);
      var match;
      return new FormatSpecifier({
        fill: match[1],
        align: match[2],
        sign: match[3],
        symbol: match[4],
        zero: match[5],
        width: match[6],
        comma: match[7],
        precision: match[8] && match[8].slice(1),
        trim: match[9],
        type: match[10]
      });
    }

    formatSpecifier.prototype = FormatSpecifier.prototype; // instanceof

    function FormatSpecifier(specifier) {
      this.fill = specifier.fill === undefined ? " " : specifier.fill + "";
      this.align = specifier.align === undefined ? ">" : specifier.align + "";
      this.sign = specifier.sign === undefined ? "-" : specifier.sign + "";
      this.symbol = specifier.symbol === undefined ? "" : specifier.symbol + "";
      this.zero = !!specifier.zero;
      this.width = specifier.width === undefined ? undefined : +specifier.width;
      this.comma = !!specifier.comma;
      this.precision = specifier.precision === undefined ? undefined : +specifier.precision;
      this.trim = !!specifier.trim;
      this.type = specifier.type === undefined ? "" : specifier.type + "";
    }

    FormatSpecifier.prototype.toString = function() {
      return this.fill
          + this.align
          + this.sign
          + this.symbol
          + (this.zero ? "0" : "")
          + (this.width === undefined ? "" : Math.max(1, this.width | 0))
          + (this.comma ? "," : "")
          + (this.precision === undefined ? "" : "." + Math.max(0, this.precision | 0))
          + (this.trim ? "~" : "")
          + this.type;
    };

    // Trims insignificant zeros, e.g., replaces 1.2000k with 1.2k.
    function formatTrim(s) {
      out: for (var n = s.length, i = 1, i0 = -1, i1; i < n; ++i) {
        switch (s[i]) {
          case ".": i0 = i1 = i; break;
          case "0": if (i0 === 0) i0 = i; i1 = i; break;
          default: if (!+s[i]) break out; if (i0 > 0) i0 = 0; break;
        }
      }
      return i0 > 0 ? s.slice(0, i0) + s.slice(i1 + 1) : s;
    }

    var prefixExponent;

    function formatPrefixAuto(x, p) {
      var d = formatDecimal(x, p);
      if (!d) return x + "";
      var coefficient = d[0],
          exponent = d[1],
          i = exponent - (prefixExponent = Math.max(-8, Math.min(8, Math.floor(exponent / 3))) * 3) + 1,
          n = coefficient.length;
      return i === n ? coefficient
          : i > n ? coefficient + new Array(i - n + 1).join("0")
          : i > 0 ? coefficient.slice(0, i) + "." + coefficient.slice(i)
          : "0." + new Array(1 - i).join("0") + formatDecimal(x, Math.max(0, p + i - 1))[0]; // less than 1y!
    }

    function formatRounded(x, p) {
      var d = formatDecimal(x, p);
      if (!d) return x + "";
      var coefficient = d[0],
          exponent = d[1];
      return exponent < 0 ? "0." + new Array(-exponent).join("0") + coefficient
          : coefficient.length > exponent + 1 ? coefficient.slice(0, exponent + 1) + "." + coefficient.slice(exponent + 1)
          : coefficient + new Array(exponent - coefficient.length + 2).join("0");
    }

    var formatTypes = {
      "%": function(x, p) { return (x * 100).toFixed(p); },
      "b": function(x) { return Math.round(x).toString(2); },
      "c": function(x) { return x + ""; },
      "d": function(x) { return Math.round(x).toString(10); },
      "e": function(x, p) { return x.toExponential(p); },
      "f": function(x, p) { return x.toFixed(p); },
      "g": function(x, p) { return x.toPrecision(p); },
      "o": function(x) { return Math.round(x).toString(8); },
      "p": function(x, p) { return formatRounded(x * 100, p); },
      "r": formatRounded,
      "s": formatPrefixAuto,
      "X": function(x) { return Math.round(x).toString(16).toUpperCase(); },
      "x": function(x) { return Math.round(x).toString(16); }
    };

    function identity$3(x) {
      return x;
    }

    var map$2 = Array.prototype.map,
        prefixes = ["y","z","a","f","p","n","µ","m","","k","M","G","T","P","E","Z","Y"];

    function formatLocale(locale) {
      var group = locale.grouping === undefined || locale.thousands === undefined ? identity$3 : formatGroup(map$2.call(locale.grouping, Number), locale.thousands + ""),
          currencyPrefix = locale.currency === undefined ? "" : locale.currency[0] + "",
          currencySuffix = locale.currency === undefined ? "" : locale.currency[1] + "",
          decimal = locale.decimal === undefined ? "." : locale.decimal + "",
          numerals = locale.numerals === undefined ? identity$3 : formatNumerals(map$2.call(locale.numerals, String)),
          percent = locale.percent === undefined ? "%" : locale.percent + "",
          minus = locale.minus === undefined ? "-" : locale.minus + "",
          nan = locale.nan === undefined ? "NaN" : locale.nan + "";

      function newFormat(specifier) {
        specifier = formatSpecifier(specifier);

        var fill = specifier.fill,
            align = specifier.align,
            sign = specifier.sign,
            symbol = specifier.symbol,
            zero = specifier.zero,
            width = specifier.width,
            comma = specifier.comma,
            precision = specifier.precision,
            trim = specifier.trim,
            type = specifier.type;

        // The "n" type is an alias for ",g".
        if (type === "n") comma = true, type = "g";

        // The "" type, and any invalid type, is an alias for ".12~g".
        else if (!formatTypes[type]) precision === undefined && (precision = 12), trim = true, type = "g";

        // If zero fill is specified, padding goes after sign and before digits.
        if (zero || (fill === "0" && align === "=")) zero = true, fill = "0", align = "=";

        // Compute the prefix and suffix.
        // For SI-prefix, the suffix is lazily computed.
        var prefix = symbol === "$" ? currencyPrefix : symbol === "#" && /[boxX]/.test(type) ? "0" + type.toLowerCase() : "",
            suffix = symbol === "$" ? currencySuffix : /[%p]/.test(type) ? percent : "";

        // What format function should we use?
        // Is this an integer type?
        // Can this type generate exponential notation?
        var formatType = formatTypes[type],
            maybeSuffix = /[defgprs%]/.test(type);

        // Set the default precision if not specified,
        // or clamp the specified precision to the supported range.
        // For significant precision, it must be in [1, 21].
        // For fixed precision, it must be in [0, 20].
        precision = precision === undefined ? 6
            : /[gprs]/.test(type) ? Math.max(1, Math.min(21, precision))
            : Math.max(0, Math.min(20, precision));

        function format(value) {
          var valuePrefix = prefix,
              valueSuffix = suffix,
              i, n, c;

          if (type === "c") {
            valueSuffix = formatType(value) + valueSuffix;
            value = "";
          } else {
            value = +value;

            // Determine the sign. -0 is not less than 0, but 1 / -0 is!
            var valueNegative = value < 0 || 1 / value < 0;

            // Perform the initial formatting.
            value = isNaN(value) ? nan : formatType(Math.abs(value), precision);

            // Trim insignificant zeros.
            if (trim) value = formatTrim(value);

            // If a negative value rounds to zero after formatting, and no explicit positive sign is requested, hide the sign.
            if (valueNegative && +value === 0 && sign !== "+") valueNegative = false;

            // Compute the prefix and suffix.
            valuePrefix = (valueNegative ? (sign === "(" ? sign : minus) : sign === "-" || sign === "(" ? "" : sign) + valuePrefix;
            valueSuffix = (type === "s" ? prefixes[8 + prefixExponent / 3] : "") + valueSuffix + (valueNegative && sign === "(" ? ")" : "");

            // Break the formatted value into the integer “value” part that can be
            // grouped, and fractional or exponential “suffix” part that is not.
            if (maybeSuffix) {
              i = -1, n = value.length;
              while (++i < n) {
                if (c = value.charCodeAt(i), 48 > c || c > 57) {
                  valueSuffix = (c === 46 ? decimal + value.slice(i + 1) : value.slice(i)) + valueSuffix;
                  value = value.slice(0, i);
                  break;
                }
              }
            }
          }

          // If the fill character is not "0", grouping is applied before padding.
          if (comma && !zero) value = group(value, Infinity);

          // Compute the padding.
          var length = valuePrefix.length + value.length + valueSuffix.length,
              padding = length < width ? new Array(width - length + 1).join(fill) : "";

          // If the fill character is "0", grouping is applied after padding.
          if (comma && zero) value = group(padding + value, padding.length ? width - valueSuffix.length : Infinity), padding = "";

          // Reconstruct the final output based on the desired alignment.
          switch (align) {
            case "<": value = valuePrefix + value + valueSuffix + padding; break;
            case "=": value = valuePrefix + padding + value + valueSuffix; break;
            case "^": value = padding.slice(0, length = padding.length >> 1) + valuePrefix + value + valueSuffix + padding.slice(length); break;
            default: value = padding + valuePrefix + value + valueSuffix; break;
          }

          return numerals(value);
        }

        format.toString = function() {
          return specifier + "";
        };

        return format;
      }

      function formatPrefix(specifier, value) {
        var f = newFormat((specifier = formatSpecifier(specifier), specifier.type = "f", specifier)),
            e = Math.max(-8, Math.min(8, Math.floor(exponent$1(value) / 3))) * 3,
            k = Math.pow(10, -e),
            prefix = prefixes[8 + e / 3];
        return function(value) {
          return f(k * value) + prefix;
        };
      }

      return {
        format: newFormat,
        formatPrefix: formatPrefix
      };
    }

    var locale;
    var format;
    var formatPrefix;

    defaultLocale({
      decimal: ".",
      thousands: ",",
      grouping: [3],
      currency: ["$", ""],
      minus: "-"
    });

    function defaultLocale(definition) {
      locale = formatLocale(definition);
      format = locale.format;
      formatPrefix = locale.formatPrefix;
      return locale;
    }

    function precisionFixed(step) {
      return Math.max(0, -exponent$1(Math.abs(step)));
    }

    function precisionPrefix(step, value) {
      return Math.max(0, Math.max(-8, Math.min(8, Math.floor(exponent$1(value) / 3))) * 3 - exponent$1(Math.abs(step)));
    }

    function precisionRound(step, max) {
      step = Math.abs(step), max = Math.abs(max) - step;
      return Math.max(0, exponent$1(max) - exponent$1(step)) + 1;
    }

    // Adds floating point numbers with twice the normal precision.
    // Reference: J. R. Shewchuk, Adaptive Precision Floating-Point Arithmetic and
    // Fast Robust Geometric Predicates, Discrete & Computational Geometry 18(3)
    // 305–363 (1997).
    // Code adapted from GeographicLib by Charles F. F. Karney,
    // http://geographiclib.sourceforge.net/

    function adder() {
      return new Adder;
    }

    function Adder() {
      this.reset();
    }

    Adder.prototype = {
      constructor: Adder,
      reset: function() {
        this.s = // rounded value
        this.t = 0; // exact error
      },
      add: function(y) {
        add$1(temp, y, this.t);
        add$1(this, temp.s, this.s);
        if (this.s) this.t += temp.t;
        else this.s = temp.t;
      },
      valueOf: function() {
        return this.s;
      }
    };

    var temp = new Adder;

    function add$1(adder, a, b) {
      var x = adder.s = a + b,
          bv = x - a,
          av = x - bv;
      adder.t = (a - av) + (b - bv);
    }

    var epsilon$2 = 1e-6;
    var epsilon2$1 = 1e-12;
    var pi$3 = Math.PI;
    var halfPi$2 = pi$3 / 2;
    var quarterPi = pi$3 / 4;
    var tau$3 = pi$3 * 2;

    var degrees$1 = 180 / pi$3;
    var radians = pi$3 / 180;

    var abs = Math.abs;
    var atan = Math.atan;
    var atan2 = Math.atan2;
    var cos$1 = Math.cos;
    var ceil = Math.ceil;
    var exp = Math.exp;
    var log = Math.log;
    var pow = Math.pow;
    var sin$1 = Math.sin;
    var sign = Math.sign || function(x) { return x > 0 ? 1 : x < 0 ? -1 : 0; };
    var sqrt = Math.sqrt;
    var tan = Math.tan;

    function acos(x) {
      return x > 1 ? 0 : x < -1 ? pi$3 : Math.acos(x);
    }

    function asin(x) {
      return x > 1 ? halfPi$2 : x < -1 ? -halfPi$2 : Math.asin(x);
    }

    function haversin(x) {
      return (x = sin$1(x / 2)) * x;
    }

    function noop$2() {}

    function streamGeometry(geometry, stream) {
      if (geometry && streamGeometryType.hasOwnProperty(geometry.type)) {
        streamGeometryType[geometry.type](geometry, stream);
      }
    }

    var streamObjectType = {
      Feature: function(object, stream) {
        streamGeometry(object.geometry, stream);
      },
      FeatureCollection: function(object, stream) {
        var features = object.features, i = -1, n = features.length;
        while (++i < n) streamGeometry(features[i].geometry, stream);
      }
    };

    var streamGeometryType = {
      Sphere: function(object, stream) {
        stream.sphere();
      },
      Point: function(object, stream) {
        object = object.coordinates;
        stream.point(object[0], object[1], object[2]);
      },
      MultiPoint: function(object, stream) {
        var coordinates = object.coordinates, i = -1, n = coordinates.length;
        while (++i < n) object = coordinates[i], stream.point(object[0], object[1], object[2]);
      },
      LineString: function(object, stream) {
        streamLine(object.coordinates, stream, 0);
      },
      MultiLineString: function(object, stream) {
        var coordinates = object.coordinates, i = -1, n = coordinates.length;
        while (++i < n) streamLine(coordinates[i], stream, 0);
      },
      Polygon: function(object, stream) {
        streamPolygon(object.coordinates, stream);
      },
      MultiPolygon: function(object, stream) {
        var coordinates = object.coordinates, i = -1, n = coordinates.length;
        while (++i < n) streamPolygon(coordinates[i], stream);
      },
      GeometryCollection: function(object, stream) {
        var geometries = object.geometries, i = -1, n = geometries.length;
        while (++i < n) streamGeometry(geometries[i], stream);
      }
    };

    function streamLine(coordinates, stream, closed) {
      var i = -1, n = coordinates.length - closed, coordinate;
      stream.lineStart();
      while (++i < n) coordinate = coordinates[i], stream.point(coordinate[0], coordinate[1], coordinate[2]);
      stream.lineEnd();
    }

    function streamPolygon(coordinates, stream) {
      var i = -1, n = coordinates.length;
      stream.polygonStart();
      while (++i < n) streamLine(coordinates[i], stream, 1);
      stream.polygonEnd();
    }

    function geoStream(object, stream) {
      if (object && streamObjectType.hasOwnProperty(object.type)) {
        streamObjectType[object.type](object, stream);
      } else {
        streamGeometry(object, stream);
      }
    }

    var areaRingSum = adder();

    var areaSum = adder(),
        lambda00,
        phi00,
        lambda0,
        cosPhi0,
        sinPhi0;

    var areaStream = {
      point: noop$2,
      lineStart: noop$2,
      lineEnd: noop$2,
      polygonStart: function() {
        areaRingSum.reset();
        areaStream.lineStart = areaRingStart;
        areaStream.lineEnd = areaRingEnd;
      },
      polygonEnd: function() {
        var areaRing = +areaRingSum;
        areaSum.add(areaRing < 0 ? tau$3 + areaRing : areaRing);
        this.lineStart = this.lineEnd = this.point = noop$2;
      },
      sphere: function() {
        areaSum.add(tau$3);
      }
    };

    function areaRingStart() {
      areaStream.point = areaPointFirst;
    }

    function areaRingEnd() {
      areaPoint(lambda00, phi00);
    }

    function areaPointFirst(lambda, phi) {
      areaStream.point = areaPoint;
      lambda00 = lambda, phi00 = phi;
      lambda *= radians, phi *= radians;
      lambda0 = lambda, cosPhi0 = cos$1(phi = phi / 2 + quarterPi), sinPhi0 = sin$1(phi);
    }

    function areaPoint(lambda, phi) {
      lambda *= radians, phi *= radians;
      phi = phi / 2 + quarterPi; // half the angular distance from south pole

      // Spherical excess E for a spherical triangle with vertices: south pole,
      // previous point, current point.  Uses a formula derived from Cagnoli’s
      // theorem.  See Todhunter, Spherical Trig. (1871), Sec. 103, Eq. (2).
      var dLambda = lambda - lambda0,
          sdLambda = dLambda >= 0 ? 1 : -1,
          adLambda = sdLambda * dLambda,
          cosPhi = cos$1(phi),
          sinPhi = sin$1(phi),
          k = sinPhi0 * sinPhi,
          u = cosPhi0 * cosPhi + k * cos$1(adLambda),
          v = k * sdLambda * sin$1(adLambda);
      areaRingSum.add(atan2(v, u));

      // Advance the previous points.
      lambda0 = lambda, cosPhi0 = cosPhi, sinPhi0 = sinPhi;
    }

    function area$1(object) {
      areaSum.reset();
      geoStream(object, areaStream);
      return areaSum * 2;
    }

    function spherical(cartesian) {
      return [atan2(cartesian[1], cartesian[0]), asin(cartesian[2])];
    }

    function cartesian(spherical) {
      var lambda = spherical[0], phi = spherical[1], cosPhi = cos$1(phi);
      return [cosPhi * cos$1(lambda), cosPhi * sin$1(lambda), sin$1(phi)];
    }

    function cartesianDot(a, b) {
      return a[0] * b[0] + a[1] * b[1] + a[2] * b[2];
    }

    function cartesianCross(a, b) {
      return [a[1] * b[2] - a[2] * b[1], a[2] * b[0] - a[0] * b[2], a[0] * b[1] - a[1] * b[0]];
    }

    // TODO return a
    function cartesianAddInPlace(a, b) {
      a[0] += b[0], a[1] += b[1], a[2] += b[2];
    }

    function cartesianScale(vector, k) {
      return [vector[0] * k, vector[1] * k, vector[2] * k];
    }

    // TODO return d
    function cartesianNormalizeInPlace(d) {
      var l = sqrt(d[0] * d[0] + d[1] * d[1] + d[2] * d[2]);
      d[0] /= l, d[1] /= l, d[2] /= l;
    }

    var lambda0$1, phi0, lambda1, phi1, // bounds
        lambda2, // previous lambda-coordinate
        lambda00$1, phi00$1, // first point
        p0, // previous 3D point
        deltaSum = adder(),
        ranges,
        range;

    var boundsStream = {
      point: boundsPoint,
      lineStart: boundsLineStart,
      lineEnd: boundsLineEnd,
      polygonStart: function() {
        boundsStream.point = boundsRingPoint;
        boundsStream.lineStart = boundsRingStart;
        boundsStream.lineEnd = boundsRingEnd;
        deltaSum.reset();
        areaStream.polygonStart();
      },
      polygonEnd: function() {
        areaStream.polygonEnd();
        boundsStream.point = boundsPoint;
        boundsStream.lineStart = boundsLineStart;
        boundsStream.lineEnd = boundsLineEnd;
        if (areaRingSum < 0) lambda0$1 = -(lambda1 = 180), phi0 = -(phi1 = 90);
        else if (deltaSum > epsilon$2) phi1 = 90;
        else if (deltaSum < -epsilon$2) phi0 = -90;
        range[0] = lambda0$1, range[1] = lambda1;
      },
      sphere: function() {
        lambda0$1 = -(lambda1 = 180), phi0 = -(phi1 = 90);
      }
    };

    function boundsPoint(lambda, phi) {
      ranges.push(range = [lambda0$1 = lambda, lambda1 = lambda]);
      if (phi < phi0) phi0 = phi;
      if (phi > phi1) phi1 = phi;
    }

    function linePoint(lambda, phi) {
      var p = cartesian([lambda * radians, phi * radians]);
      if (p0) {
        var normal = cartesianCross(p0, p),
            equatorial = [normal[1], -normal[0], 0],
            inflection = cartesianCross(equatorial, normal);
        cartesianNormalizeInPlace(inflection);
        inflection = spherical(inflection);
        var delta = lambda - lambda2,
            sign = delta > 0 ? 1 : -1,
            lambdai = inflection[0] * degrees$1 * sign,
            phii,
            antimeridian = abs(delta) > 180;
        if (antimeridian ^ (sign * lambda2 < lambdai && lambdai < sign * lambda)) {
          phii = inflection[1] * degrees$1;
          if (phii > phi1) phi1 = phii;
        } else if (lambdai = (lambdai + 360) % 360 - 180, antimeridian ^ (sign * lambda2 < lambdai && lambdai < sign * lambda)) {
          phii = -inflection[1] * degrees$1;
          if (phii < phi0) phi0 = phii;
        } else {
          if (phi < phi0) phi0 = phi;
          if (phi > phi1) phi1 = phi;
        }
        if (antimeridian) {
          if (lambda < lambda2) {
            if (angle(lambda0$1, lambda) > angle(lambda0$1, lambda1)) lambda1 = lambda;
          } else {
            if (angle(lambda, lambda1) > angle(lambda0$1, lambda1)) lambda0$1 = lambda;
          }
        } else {
          if (lambda1 >= lambda0$1) {
            if (lambda < lambda0$1) lambda0$1 = lambda;
            if (lambda > lambda1) lambda1 = lambda;
          } else {
            if (lambda > lambda2) {
              if (angle(lambda0$1, lambda) > angle(lambda0$1, lambda1)) lambda1 = lambda;
            } else {
              if (angle(lambda, lambda1) > angle(lambda0$1, lambda1)) lambda0$1 = lambda;
            }
          }
        }
      } else {
        ranges.push(range = [lambda0$1 = lambda, lambda1 = lambda]);
      }
      if (phi < phi0) phi0 = phi;
      if (phi > phi1) phi1 = phi;
      p0 = p, lambda2 = lambda;
    }

    function boundsLineStart() {
      boundsStream.point = linePoint;
    }

    function boundsLineEnd() {
      range[0] = lambda0$1, range[1] = lambda1;
      boundsStream.point = boundsPoint;
      p0 = null;
    }

    function boundsRingPoint(lambda, phi) {
      if (p0) {
        var delta = lambda - lambda2;
        deltaSum.add(abs(delta) > 180 ? delta + (delta > 0 ? 360 : -360) : delta);
      } else {
        lambda00$1 = lambda, phi00$1 = phi;
      }
      areaStream.point(lambda, phi);
      linePoint(lambda, phi);
    }

    function boundsRingStart() {
      areaStream.lineStart();
    }

    function boundsRingEnd() {
      boundsRingPoint(lambda00$1, phi00$1);
      areaStream.lineEnd();
      if (abs(deltaSum) > epsilon$2) lambda0$1 = -(lambda1 = 180);
      range[0] = lambda0$1, range[1] = lambda1;
      p0 = null;
    }

    // Finds the left-right distance between two longitudes.
    // This is almost the same as (lambda1 - lambda0 + 360°) % 360°, except that we want
    // the distance between ±180° to be 360°.
    function angle(lambda0, lambda1) {
      return (lambda1 -= lambda0) < 0 ? lambda1 + 360 : lambda1;
    }

    function rangeCompare(a, b) {
      return a[0] - b[0];
    }

    function rangeContains(range, x) {
      return range[0] <= range[1] ? range[0] <= x && x <= range[1] : x < range[0] || range[1] < x;
    }

    function bounds(feature) {
      var i, n, a, b, merged, deltaMax, delta;

      phi1 = lambda1 = -(lambda0$1 = phi0 = Infinity);
      ranges = [];
      geoStream(feature, boundsStream);

      // First, sort ranges by their minimum longitudes.
      if (n = ranges.length) {
        ranges.sort(rangeCompare);

        // Then, merge any ranges that overlap.
        for (i = 1, a = ranges[0], merged = [a]; i < n; ++i) {
          b = ranges[i];
          if (rangeContains(a, b[0]) || rangeContains(a, b[1])) {
            if (angle(a[0], b[1]) > angle(a[0], a[1])) a[1] = b[1];
            if (angle(b[0], a[1]) > angle(a[0], a[1])) a[0] = b[0];
          } else {
            merged.push(a = b);
          }
        }

        // Finally, find the largest gap between the merged ranges.
        // The final bounding box will be the inverse of this gap.
        for (deltaMax = -Infinity, n = merged.length - 1, i = 0, a = merged[n]; i <= n; a = b, ++i) {
          b = merged[i];
          if ((delta = angle(a[1], b[0])) > deltaMax) deltaMax = delta, lambda0$1 = b[0], lambda1 = a[1];
        }
      }

      ranges = range = null;

      return lambda0$1 === Infinity || phi0 === Infinity
          ? [[NaN, NaN], [NaN, NaN]]
          : [[lambda0$1, phi0], [lambda1, phi1]];
    }

    var W0, W1,
        X0, Y0, Z0,
        X1, Y1, Z1,
        X2, Y2, Z2,
        lambda00$2, phi00$2, // first point
        x0, y0, z0; // previous point

    var centroidStream = {
      sphere: noop$2,
      point: centroidPoint,
      lineStart: centroidLineStart,
      lineEnd: centroidLineEnd,
      polygonStart: function() {
        centroidStream.lineStart = centroidRingStart;
        centroidStream.lineEnd = centroidRingEnd;
      },
      polygonEnd: function() {
        centroidStream.lineStart = centroidLineStart;
        centroidStream.lineEnd = centroidLineEnd;
      }
    };

    // Arithmetic mean of Cartesian vectors.
    function centroidPoint(lambda, phi) {
      lambda *= radians, phi *= radians;
      var cosPhi = cos$1(phi);
      centroidPointCartesian(cosPhi * cos$1(lambda), cosPhi * sin$1(lambda), sin$1(phi));
    }

    function centroidPointCartesian(x, y, z) {
      ++W0;
      X0 += (x - X0) / W0;
      Y0 += (y - Y0) / W0;
      Z0 += (z - Z0) / W0;
    }

    function centroidLineStart() {
      centroidStream.point = centroidLinePointFirst;
    }

    function centroidLinePointFirst(lambda, phi) {
      lambda *= radians, phi *= radians;
      var cosPhi = cos$1(phi);
      x0 = cosPhi * cos$1(lambda);
      y0 = cosPhi * sin$1(lambda);
      z0 = sin$1(phi);
      centroidStream.point = centroidLinePoint;
      centroidPointCartesian(x0, y0, z0);
    }

    function centroidLinePoint(lambda, phi) {
      lambda *= radians, phi *= radians;
      var cosPhi = cos$1(phi),
          x = cosPhi * cos$1(lambda),
          y = cosPhi * sin$1(lambda),
          z = sin$1(phi),
          w = atan2(sqrt((w = y0 * z - z0 * y) * w + (w = z0 * x - x0 * z) * w + (w = x0 * y - y0 * x) * w), x0 * x + y0 * y + z0 * z);
      W1 += w;
      X1 += w * (x0 + (x0 = x));
      Y1 += w * (y0 + (y0 = y));
      Z1 += w * (z0 + (z0 = z));
      centroidPointCartesian(x0, y0, z0);
    }

    function centroidLineEnd() {
      centroidStream.point = centroidPoint;
    }

    // See J. E. Brock, The Inertia Tensor for a Spherical Triangle,
    // J. Applied Mechanics 42, 239 (1975).
    function centroidRingStart() {
      centroidStream.point = centroidRingPointFirst;
    }

    function centroidRingEnd() {
      centroidRingPoint(lambda00$2, phi00$2);
      centroidStream.point = centroidPoint;
    }

    function centroidRingPointFirst(lambda, phi) {
      lambda00$2 = lambda, phi00$2 = phi;
      lambda *= radians, phi *= radians;
      centroidStream.point = centroidRingPoint;
      var cosPhi = cos$1(phi);
      x0 = cosPhi * cos$1(lambda);
      y0 = cosPhi * sin$1(lambda);
      z0 = sin$1(phi);
      centroidPointCartesian(x0, y0, z0);
    }

    function centroidRingPoint(lambda, phi) {
      lambda *= radians, phi *= radians;
      var cosPhi = cos$1(phi),
          x = cosPhi * cos$1(lambda),
          y = cosPhi * sin$1(lambda),
          z = sin$1(phi),
          cx = y0 * z - z0 * y,
          cy = z0 * x - x0 * z,
          cz = x0 * y - y0 * x,
          m = sqrt(cx * cx + cy * cy + cz * cz),
          w = asin(m), // line weight = angle
          v = m && -w / m; // area weight multiplier
      X2 += v * cx;
      Y2 += v * cy;
      Z2 += v * cz;
      W1 += w;
      X1 += w * (x0 + (x0 = x));
      Y1 += w * (y0 + (y0 = y));
      Z1 += w * (z0 + (z0 = z));
      centroidPointCartesian(x0, y0, z0);
    }

    function centroid(object) {
      W0 = W1 =
      X0 = Y0 = Z0 =
      X1 = Y1 = Z1 =
      X2 = Y2 = Z2 = 0;
      geoStream(object, centroidStream);

      var x = X2,
          y = Y2,
          z = Z2,
          m = x * x + y * y + z * z;

      // If the area-weighted ccentroid is undefined, fall back to length-weighted ccentroid.
      if (m < epsilon2$1) {
        x = X1, y = Y1, z = Z1;
        // If the feature has zero length, fall back to arithmetic mean of point vectors.
        if (W1 < epsilon$2) x = X0, y = Y0, z = Z0;
        m = x * x + y * y + z * z;
        // If the feature still has an undefined ccentroid, then return.
        if (m < epsilon2$1) return [NaN, NaN];
      }

      return [atan2(y, x) * degrees$1, asin(z / sqrt(m)) * degrees$1];
    }

    function constant$8(x) {
      return function() {
        return x;
      };
    }

    function compose(a, b) {

      function compose(x, y) {
        return x = a(x, y), b(x[0], x[1]);
      }

      if (a.invert && b.invert) compose.invert = function(x, y) {
        return x = b.invert(x, y), x && a.invert(x[0], x[1]);
      };

      return compose;
    }

    function rotationIdentity(lambda, phi) {
      return [abs(lambda) > pi$3 ? lambda + Math.round(-lambda / tau$3) * tau$3 : lambda, phi];
    }

    rotationIdentity.invert = rotationIdentity;

    function rotateRadians(deltaLambda, deltaPhi, deltaGamma) {
      return (deltaLambda %= tau$3) ? (deltaPhi || deltaGamma ? compose(rotationLambda(deltaLambda), rotationPhiGamma(deltaPhi, deltaGamma))
        : rotationLambda(deltaLambda))
        : (deltaPhi || deltaGamma ? rotationPhiGamma(deltaPhi, deltaGamma)
        : rotationIdentity);
    }

    function forwardRotationLambda(deltaLambda) {
      return function(lambda, phi) {
        return lambda += deltaLambda, [lambda > pi$3 ? lambda - tau$3 : lambda < -pi$3 ? lambda + tau$3 : lambda, phi];
      };
    }

    function rotationLambda(deltaLambda) {
      var rotation = forwardRotationLambda(deltaLambda);
      rotation.invert = forwardRotationLambda(-deltaLambda);
      return rotation;
    }

    function rotationPhiGamma(deltaPhi, deltaGamma) {
      var cosDeltaPhi = cos$1(deltaPhi),
          sinDeltaPhi = sin$1(deltaPhi),
          cosDeltaGamma = cos$1(deltaGamma),
          sinDeltaGamma = sin$1(deltaGamma);

      function rotation(lambda, phi) {
        var cosPhi = cos$1(phi),
            x = cos$1(lambda) * cosPhi,
            y = sin$1(lambda) * cosPhi,
            z = sin$1(phi),
            k = z * cosDeltaPhi + x * sinDeltaPhi;
        return [
          atan2(y * cosDeltaGamma - k * sinDeltaGamma, x * cosDeltaPhi - z * sinDeltaPhi),
          asin(k * cosDeltaGamma + y * sinDeltaGamma)
        ];
      }

      rotation.invert = function(lambda, phi) {
        var cosPhi = cos$1(phi),
            x = cos$1(lambda) * cosPhi,
            y = sin$1(lambda) * cosPhi,
            z = sin$1(phi),
            k = z * cosDeltaGamma - y * sinDeltaGamma;
        return [
          atan2(y * cosDeltaGamma + z * sinDeltaGamma, x * cosDeltaPhi + k * sinDeltaPhi),
          asin(k * cosDeltaPhi - x * sinDeltaPhi)
        ];
      };

      return rotation;
    }

    function rotation(rotate) {
      rotate = rotateRadians(rotate[0] * radians, rotate[1] * radians, rotate.length > 2 ? rotate[2] * radians : 0);

      function forward(coordinates) {
        coordinates = rotate(coordinates[0] * radians, coordinates[1] * radians);
        return coordinates[0] *= degrees$1, coordinates[1] *= degrees$1, coordinates;
      }

      forward.invert = function(coordinates) {
        coordinates = rotate.invert(coordinates[0] * radians, coordinates[1] * radians);
        return coordinates[0] *= degrees$1, coordinates[1] *= degrees$1, coordinates;
      };

      return forward;
    }

    // Generates a circle centered at [0°, 0°], with a given radius and precision.
    function circleStream(stream, radius, delta, direction, t0, t1) {
      if (!delta) return;
      var cosRadius = cos$1(radius),
          sinRadius = sin$1(radius),
          step = direction * delta;
      if (t0 == null) {
        t0 = radius + direction * tau$3;
        t1 = radius - step / 2;
      } else {
        t0 = circleRadius(cosRadius, t0);
        t1 = circleRadius(cosRadius, t1);
        if (direction > 0 ? t0 < t1 : t0 > t1) t0 += direction * tau$3;
      }
      for (var point, t = t0; direction > 0 ? t > t1 : t < t1; t -= step) {
        point = spherical([cosRadius, -sinRadius * cos$1(t), -sinRadius * sin$1(t)]);
        stream.point(point[0], point[1]);
      }
    }

    // Returns the signed angle of a cartesian point relative to [cosRadius, 0, 0].
    function circleRadius(cosRadius, point) {
      point = cartesian(point), point[0] -= cosRadius;
      cartesianNormalizeInPlace(point);
      var radius = acos(-point[1]);
      return ((-point[2] < 0 ? -radius : radius) + tau$3 - epsilon$2) % tau$3;
    }

    function circle() {
      var center = constant$8([0, 0]),
          radius = constant$8(90),
          precision = constant$8(6),
          ring,
          rotate,
          stream = {point: point};

      function point(x, y) {
        ring.push(x = rotate(x, y));
        x[0] *= degrees$1, x[1] *= degrees$1;
      }

      function circle() {
        var c = center.apply(this, arguments),
            r = radius.apply(this, arguments) * radians,
            p = precision.apply(this, arguments) * radians;
        ring = [];
        rotate = rotateRadians(-c[0] * radians, -c[1] * radians, 0).invert;
        circleStream(stream, r, p, 1);
        c = {type: "Polygon", coordinates: [ring]};
        ring = rotate = null;
        return c;
      }

      circle.center = function(_) {
        return arguments.length ? (center = typeof _ === "function" ? _ : constant$8([+_[0], +_[1]]), circle) : center;
      };

      circle.radius = function(_) {
        return arguments.length ? (radius = typeof _ === "function" ? _ : constant$8(+_), circle) : radius;
      };

      circle.precision = function(_) {
        return arguments.length ? (precision = typeof _ === "function" ? _ : constant$8(+_), circle) : precision;
      };

      return circle;
    }

    function clipBuffer() {
      var lines = [],
          line;
      return {
        point: function(x, y, m) {
          line.push([x, y, m]);
        },
        lineStart: function() {
          lines.push(line = []);
        },
        lineEnd: noop$2,
        rejoin: function() {
          if (lines.length > 1) lines.push(lines.pop().concat(lines.shift()));
        },
        result: function() {
          var result = lines;
          lines = [];
          line = null;
          return result;
        }
      };
    }

    function pointEqual(a, b) {
      return abs(a[0] - b[0]) < epsilon$2 && abs(a[1] - b[1]) < epsilon$2;
    }

    function Intersection(point, points, other, entry) {
      this.x = point;
      this.z = points;
      this.o = other; // another intersection
      this.e = entry; // is an entry?
      this.v = false; // visited
      this.n = this.p = null; // next & previous
    }

    // A generalized polygon clipping algorithm: given a polygon that has been cut
    // into its visible line segments, and rejoins the segments by interpolating
    // along the clip edge.
    function clipRejoin(segments, compareIntersection, startInside, interpolate, stream) {
      var subject = [],
          clip = [],
          i,
          n;

      segments.forEach(function(segment) {
        if ((n = segment.length - 1) <= 0) return;
        var n, p0 = segment[0], p1 = segment[n], x;

        if (pointEqual(p0, p1)) {
          if (!p0[2] && !p1[2]) {
            stream.lineStart();
            for (i = 0; i < n; ++i) stream.point((p0 = segment[i])[0], p0[1]);
            stream.lineEnd();
            return;
          }
          // handle degenerate cases by moving the point
          p1[0] += 2 * epsilon$2;
        }

        subject.push(x = new Intersection(p0, segment, null, true));
        clip.push(x.o = new Intersection(p0, null, x, false));
        subject.push(x = new Intersection(p1, segment, null, false));
        clip.push(x.o = new Intersection(p1, null, x, true));
      });

      if (!subject.length) return;

      clip.sort(compareIntersection);
      link$1(subject);
      link$1(clip);

      for (i = 0, n = clip.length; i < n; ++i) {
        clip[i].e = startInside = !startInside;
      }

      var start = subject[0],
          points,
          point;

      while (1) {
        // Find first unvisited intersection.
        var current = start,
            isSubject = true;
        while (current.v) if ((current = current.n) === start) return;
        points = current.z;
        stream.lineStart();
        do {
          current.v = current.o.v = true;
          if (current.e) {
            if (isSubject) {
              for (i = 0, n = points.length; i < n; ++i) stream.point((point = points[i])[0], point[1]);
            } else {
              interpolate(current.x, current.n.x, 1, stream);
            }
            current = current.n;
          } else {
            if (isSubject) {
              points = current.p.z;
              for (i = points.length - 1; i >= 0; --i) stream.point((point = points[i])[0], point[1]);
            } else {
              interpolate(current.x, current.p.x, -1, stream);
            }
            current = current.p;
          }
          current = current.o;
          points = current.z;
          isSubject = !isSubject;
        } while (!current.v);
        stream.lineEnd();
      }
    }

    function link$1(array) {
      if (!(n = array.length)) return;
      var n,
          i = 0,
          a = array[0],
          b;
      while (++i < n) {
        a.n = b = array[i];
        b.p = a;
        a = b;
      }
      a.n = b = array[0];
      b.p = a;
    }

    var sum$1 = adder();

    function longitude(point) {
      if (abs(point[0]) <= pi$3)
        return point[0];
      else
        return sign(point[0]) * ((abs(point[0]) + pi$3) % tau$3 - pi$3);
    }

    function polygonContains(polygon, point) {
      var lambda = longitude(point),
          phi = point[1],
          sinPhi = sin$1(phi),
          normal = [sin$1(lambda), -cos$1(lambda), 0],
          angle = 0,
          winding = 0;

      sum$1.reset();

      if (sinPhi === 1) phi = halfPi$2 + epsilon$2;
      else if (sinPhi === -1) phi = -halfPi$2 - epsilon$2;

      for (var i = 0, n = polygon.length; i < n; ++i) {
        if (!(m = (ring = polygon[i]).length)) continue;
        var ring,
            m,
            point0 = ring[m - 1],
            lambda0 = longitude(point0),
            phi0 = point0[1] / 2 + quarterPi,
            sinPhi0 = sin$1(phi0),
            cosPhi0 = cos$1(phi0);

        for (var j = 0; j < m; ++j, lambda0 = lambda1, sinPhi0 = sinPhi1, cosPhi0 = cosPhi1, point0 = point1) {
          var point1 = ring[j],
              lambda1 = longitude(point1),
              phi1 = point1[1] / 2 + quarterPi,
              sinPhi1 = sin$1(phi1),
              cosPhi1 = cos$1(phi1),
              delta = lambda1 - lambda0,
              sign = delta >= 0 ? 1 : -1,
              absDelta = sign * delta,
              antimeridian = absDelta > pi$3,
              k = sinPhi0 * sinPhi1;

          sum$1.add(atan2(k * sign * sin$1(absDelta), cosPhi0 * cosPhi1 + k * cos$1(absDelta)));
          angle += antimeridian ? delta + sign * tau$3 : delta;

          // Are the longitudes either side of the point’s meridian (lambda),
          // and are the latitudes smaller than the parallel (phi)?
          if (antimeridian ^ lambda0 >= lambda ^ lambda1 >= lambda) {
            var arc = cartesianCross(cartesian(point0), cartesian(point1));
            cartesianNormalizeInPlace(arc);
            var intersection = cartesianCross(normal, arc);
            cartesianNormalizeInPlace(intersection);
            var phiArc = (antimeridian ^ delta >= 0 ? -1 : 1) * asin(intersection[2]);
            if (phi > phiArc || phi === phiArc && (arc[0] || arc[1])) {
              winding += antimeridian ^ delta >= 0 ? 1 : -1;
            }
          }
        }
      }

      // First, determine whether the South pole is inside or outside:
      //
      // It is inside if:
      // * the polygon winds around it in a clockwise direction.
      // * the polygon does not (cumulatively) wind around it, but has a negative
      //   (counter-clockwise) area.
      //
      // Second, count the (signed) number of times a segment crosses a lambda
      // from the point to the South pole.  If it is zero, then the point is the
      // same side as the South pole.

      return (angle < -epsilon$2 || angle < epsilon$2 && sum$1 < -epsilon$2) ^ (winding & 1);
    }

    function clip(pointVisible, clipLine, interpolate, start) {
      return function(sink) {
        var line = clipLine(sink),
            ringBuffer = clipBuffer(),
            ringSink = clipLine(ringBuffer),
            polygonStarted = false,
            polygon,
            segments,
            ring;

        var clip = {
          point: point,
          lineStart: lineStart,
          lineEnd: lineEnd,
          polygonStart: function() {
            clip.point = pointRing;
            clip.lineStart = ringStart;
            clip.lineEnd = ringEnd;
            segments = [];
            polygon = [];
          },
          polygonEnd: function() {
            clip.point = point;
            clip.lineStart = lineStart;
            clip.lineEnd = lineEnd;
            segments = merge(segments);
            var startInside = polygonContains(polygon, start);
            if (segments.length) {
              if (!polygonStarted) sink.polygonStart(), polygonStarted = true;
              clipRejoin(segments, compareIntersection, startInside, interpolate, sink);
            } else if (startInside) {
              if (!polygonStarted) sink.polygonStart(), polygonStarted = true;
              sink.lineStart();
              interpolate(null, null, 1, sink);
              sink.lineEnd();
            }
            if (polygonStarted) sink.polygonEnd(), polygonStarted = false;
            segments = polygon = null;
          },
          sphere: function() {
            sink.polygonStart();
            sink.lineStart();
            interpolate(null, null, 1, sink);
            sink.lineEnd();
            sink.polygonEnd();
          }
        };

        function point(lambda, phi) {
          if (pointVisible(lambda, phi)) sink.point(lambda, phi);
        }

        function pointLine(lambda, phi) {
          line.point(lambda, phi);
        }

        function lineStart() {
          clip.point = pointLine;
          line.lineStart();
        }

        function lineEnd() {
          clip.point = point;
          line.lineEnd();
        }

        function pointRing(lambda, phi) {
          ring.push([lambda, phi]);
          ringSink.point(lambda, phi);
        }

        function ringStart() {
          ringSink.lineStart();
          ring = [];
        }

        function ringEnd() {
          pointRing(ring[0][0], ring[0][1]);
          ringSink.lineEnd();

          var clean = ringSink.clean(),
              ringSegments = ringBuffer.result(),
              i, n = ringSegments.length, m,
              segment,
              point;

          ring.pop();
          polygon.push(ring);
          ring = null;

          if (!n) return;

          // No intersections.
          if (clean & 1) {
            segment = ringSegments[0];
            if ((m = segment.length - 1) > 0) {
              if (!polygonStarted) sink.polygonStart(), polygonStarted = true;
              sink.lineStart();
              for (i = 0; i < m; ++i) sink.point((point = segment[i])[0], point[1]);
              sink.lineEnd();
            }
            return;
          }

          // Rejoin connected segments.
          // TODO reuse ringBuffer.rejoin()?
          if (n > 1 && clean & 2) ringSegments.push(ringSegments.pop().concat(ringSegments.shift()));

          segments.push(ringSegments.filter(validSegment));
        }

        return clip;
      };
    }

    function validSegment(segment) {
      return segment.length > 1;
    }

    // Intersections are sorted along the clip edge. For both antimeridian cutting
    // and circle clipping, the same comparison is used.
    function compareIntersection(a, b) {
      return ((a = a.x)[0] < 0 ? a[1] - halfPi$2 - epsilon$2 : halfPi$2 - a[1])
           - ((b = b.x)[0] < 0 ? b[1] - halfPi$2 - epsilon$2 : halfPi$2 - b[1]);
    }

    var clipAntimeridian = clip(
      function() { return true; },
      clipAntimeridianLine,
      clipAntimeridianInterpolate,
      [-pi$3, -halfPi$2]
    );

    // Takes a line and cuts into visible segments. Return values: 0 - there were
    // intersections or the line was empty; 1 - no intersections; 2 - there were
    // intersections, and the first and last segments should be rejoined.
    function clipAntimeridianLine(stream) {
      var lambda0 = NaN,
          phi0 = NaN,
          sign0 = NaN,
          clean; // no intersections

      return {
        lineStart: function() {
          stream.lineStart();
          clean = 1;
        },
        point: function(lambda1, phi1) {
          var sign1 = lambda1 > 0 ? pi$3 : -pi$3,
              delta = abs(lambda1 - lambda0);
          if (abs(delta - pi$3) < epsilon$2) { // line crosses a pole
            stream.point(lambda0, phi0 = (phi0 + phi1) / 2 > 0 ? halfPi$2 : -halfPi$2);
            stream.point(sign0, phi0);
            stream.lineEnd();
            stream.lineStart();
            stream.point(sign1, phi0);
            stream.point(lambda1, phi0);
            clean = 0;
          } else if (sign0 !== sign1 && delta >= pi$3) { // line crosses antimeridian
            if (abs(lambda0 - sign0) < epsilon$2) lambda0 -= sign0 * epsilon$2; // handle degeneracies
            if (abs(lambda1 - sign1) < epsilon$2) lambda1 -= sign1 * epsilon$2;
            phi0 = clipAntimeridianIntersect(lambda0, phi0, lambda1, phi1);
            stream.point(sign0, phi0);
            stream.lineEnd();
            stream.lineStart();
            stream.point(sign1, phi0);
            clean = 0;
          }
          stream.point(lambda0 = lambda1, phi0 = phi1);
          sign0 = sign1;
        },
        lineEnd: function() {
          stream.lineEnd();
          lambda0 = phi0 = NaN;
        },
        clean: function() {
          return 2 - clean; // if intersections, rejoin first and last segments
        }
      };
    }

    function clipAntimeridianIntersect(lambda0, phi0, lambda1, phi1) {
      var cosPhi0,
          cosPhi1,
          sinLambda0Lambda1 = sin$1(lambda0 - lambda1);
      return abs(sinLambda0Lambda1) > epsilon$2
          ? atan((sin$1(phi0) * (cosPhi1 = cos$1(phi1)) * sin$1(lambda1)
              - sin$1(phi1) * (cosPhi0 = cos$1(phi0)) * sin$1(lambda0))
              / (cosPhi0 * cosPhi1 * sinLambda0Lambda1))
          : (phi0 + phi1) / 2;
    }

    function clipAntimeridianInterpolate(from, to, direction, stream) {
      var phi;
      if (from == null) {
        phi = direction * halfPi$2;
        stream.point(-pi$3, phi);
        stream.point(0, phi);
        stream.point(pi$3, phi);
        stream.point(pi$3, 0);
        stream.point(pi$3, -phi);
        stream.point(0, -phi);
        stream.point(-pi$3, -phi);
        stream.point(-pi$3, 0);
        stream.point(-pi$3, phi);
      } else if (abs(from[0] - to[0]) > epsilon$2) {
        var lambda = from[0] < to[0] ? pi$3 : -pi$3;
        phi = direction * lambda / 2;
        stream.point(-lambda, phi);
        stream.point(0, phi);
        stream.point(lambda, phi);
      } else {
        stream.point(to[0], to[1]);
      }
    }

    function clipCircle(radius) {
      var cr = cos$1(radius),
          delta = 6 * radians,
          smallRadius = cr > 0,
          notHemisphere = abs(cr) > epsilon$2; // TODO optimise for this common case

      function interpolate(from, to, direction, stream) {
        circleStream(stream, radius, delta, direction, from, to);
      }

      function visible(lambda, phi) {
        return cos$1(lambda) * cos$1(phi) > cr;
      }

      // Takes a line and cuts into visible segments. Return values used for polygon
      // clipping: 0 - there were intersections or the line was empty; 1 - no
      // intersections 2 - there were intersections, and the first and last segments
      // should be rejoined.
      function clipLine(stream) {
        var point0, // previous point
            c0, // code for previous point
            v0, // visibility of previous point
            v00, // visibility of first point
            clean; // no intersections
        return {
          lineStart: function() {
            v00 = v0 = false;
            clean = 1;
          },
          point: function(lambda, phi) {
            var point1 = [lambda, phi],
                point2,
                v = visible(lambda, phi),
                c = smallRadius
                  ? v ? 0 : code(lambda, phi)
                  : v ? code(lambda + (lambda < 0 ? pi$3 : -pi$3), phi) : 0;
            if (!point0 && (v00 = v0 = v)) stream.lineStart();
            if (v !== v0) {
              point2 = intersect(point0, point1);
              if (!point2 || pointEqual(point0, point2) || pointEqual(point1, point2))
                point1[2] = 1;
            }
            if (v !== v0) {
              clean = 0;
              if (v) {
                // outside going in
                stream.lineStart();
                point2 = intersect(point1, point0);
                stream.point(point2[0], point2[1]);
              } else {
                // inside going out
                point2 = intersect(point0, point1);
                stream.point(point2[0], point2[1], 2);
                stream.lineEnd();
              }
              point0 = point2;
            } else if (notHemisphere && point0 && smallRadius ^ v) {
              var t;
              // If the codes for two points are different, or are both zero,
              // and there this segment intersects with the small circle.
              if (!(c & c0) && (t = intersect(point1, point0, true))) {
                clean = 0;
                if (smallRadius) {
                  stream.lineStart();
                  stream.point(t[0][0], t[0][1]);
                  stream.point(t[1][0], t[1][1]);
                  stream.lineEnd();
                } else {
                  stream.point(t[1][0], t[1][1]);
                  stream.lineEnd();
                  stream.lineStart();
                  stream.point(t[0][0], t[0][1], 3);
                }
              }
            }
            if (v && (!point0 || !pointEqual(point0, point1))) {
              stream.point(point1[0], point1[1]);
            }
            point0 = point1, v0 = v, c0 = c;
          },
          lineEnd: function() {
            if (v0) stream.lineEnd();
            point0 = null;
          },
          // Rejoin first and last segments if there were intersections and the first
          // and last points were visible.
          clean: function() {
            return clean | ((v00 && v0) << 1);
          }
        };
      }

      // Intersects the great circle between a and b with the clip circle.
      function intersect(a, b, two) {
        var pa = cartesian(a),
            pb = cartesian(b);

        // We have two planes, n1.p = d1 and n2.p = d2.
        // Find intersection line p(t) = c1 n1 + c2 n2 + t (n1 ⨯ n2).
        var n1 = [1, 0, 0], // normal
            n2 = cartesianCross(pa, pb),
            n2n2 = cartesianDot(n2, n2),
            n1n2 = n2[0], // cartesianDot(n1, n2),
            determinant = n2n2 - n1n2 * n1n2;

        // Two polar points.
        if (!determinant) return !two && a;

        var c1 =  cr * n2n2 / determinant,
            c2 = -cr * n1n2 / determinant,
            n1xn2 = cartesianCross(n1, n2),
            A = cartesianScale(n1, c1),
            B = cartesianScale(n2, c2);
        cartesianAddInPlace(A, B);

        // Solve |p(t)|^2 = 1.
        var u = n1xn2,
            w = cartesianDot(A, u),
            uu = cartesianDot(u, u),
            t2 = w * w - uu * (cartesianDot(A, A) - 1);

        if (t2 < 0) return;

        var t = sqrt(t2),
            q = cartesianScale(u, (-w - t) / uu);
        cartesianAddInPlace(q, A);
        q = spherical(q);

        if (!two) return q;

        // Two intersection points.
        var lambda0 = a[0],
            lambda1 = b[0],
            phi0 = a[1],
            phi1 = b[1],
            z;

        if (lambda1 < lambda0) z = lambda0, lambda0 = lambda1, lambda1 = z;

        var delta = lambda1 - lambda0,
            polar = abs(delta - pi$3) < epsilon$2,
            meridian = polar || delta < epsilon$2;

        if (!polar && phi1 < phi0) z = phi0, phi0 = phi1, phi1 = z;

        // Check that the first point is between a and b.
        if (meridian
            ? polar
              ? phi0 + phi1 > 0 ^ q[1] < (abs(q[0] - lambda0) < epsilon$2 ? phi0 : phi1)
              : phi0 <= q[1] && q[1] <= phi1
            : delta > pi$3 ^ (lambda0 <= q[0] && q[0] <= lambda1)) {
          var q1 = cartesianScale(u, (-w + t) / uu);
          cartesianAddInPlace(q1, A);
          return [q, spherical(q1)];
        }
      }

      // Generates a 4-bit vector representing the location of a point relative to
      // the small circle's bounding box.
      function code(lambda, phi) {
        var r = smallRadius ? radius : pi$3 - radius,
            code = 0;
        if (lambda < -r) code |= 1; // left
        else if (lambda > r) code |= 2; // right
        if (phi < -r) code |= 4; // below
        else if (phi > r) code |= 8; // above
        return code;
      }

      return clip(visible, clipLine, interpolate, smallRadius ? [0, -radius] : [-pi$3, radius - pi$3]);
    }

    function clipLine(a, b, x0, y0, x1, y1) {
      var ax = a[0],
          ay = a[1],
          bx = b[0],
          by = b[1],
          t0 = 0,
          t1 = 1,
          dx = bx - ax,
          dy = by - ay,
          r;

      r = x0 - ax;
      if (!dx && r > 0) return;
      r /= dx;
      if (dx < 0) {
        if (r < t0) return;
        if (r < t1) t1 = r;
      } else if (dx > 0) {
        if (r > t1) return;
        if (r > t0) t0 = r;
      }

      r = x1 - ax;
      if (!dx && r < 0) return;
      r /= dx;
      if (dx < 0) {
        if (r > t1) return;
        if (r > t0) t0 = r;
      } else if (dx > 0) {
        if (r < t0) return;
        if (r < t1) t1 = r;
      }

      r = y0 - ay;
      if (!dy && r > 0) return;
      r /= dy;
      if (dy < 0) {
        if (r < t0) return;
        if (r < t1) t1 = r;
      } else if (dy > 0) {
        if (r > t1) return;
        if (r > t0) t0 = r;
      }

      r = y1 - ay;
      if (!dy && r < 0) return;
      r /= dy;
      if (dy < 0) {
        if (r > t1) return;
        if (r > t0) t0 = r;
      } else if (dy > 0) {
        if (r < t0) return;
        if (r < t1) t1 = r;
      }

      if (t0 > 0) a[0] = ax + t0 * dx, a[1] = ay + t0 * dy;
      if (t1 < 1) b[0] = ax + t1 * dx, b[1] = ay + t1 * dy;
      return true;
    }

    var clipMax = 1e9, clipMin = -clipMax;

    // TODO Use d3-polygon’s polygonContains here for the ring check?
    // TODO Eliminate duplicate buffering in clipBuffer and polygon.push?

    function clipRectangle(x0, y0, x1, y1) {

      function visible(x, y) {
        return x0 <= x && x <= x1 && y0 <= y && y <= y1;
      }

      function interpolate(from, to, direction, stream) {
        var a = 0, a1 = 0;
        if (from == null
            || (a = corner(from, direction)) !== (a1 = corner(to, direction))
            || comparePoint(from, to) < 0 ^ direction > 0) {
          do stream.point(a === 0 || a === 3 ? x0 : x1, a > 1 ? y1 : y0);
          while ((a = (a + direction + 4) % 4) !== a1);
        } else {
          stream.point(to[0], to[1]);
        }
      }

      function corner(p, direction) {
        return abs(p[0] - x0) < epsilon$2 ? direction > 0 ? 0 : 3
            : abs(p[0] - x1) < epsilon$2 ? direction > 0 ? 2 : 1
            : abs(p[1] - y0) < epsilon$2 ? direction > 0 ? 1 : 0
            : direction > 0 ? 3 : 2; // abs(p[1] - y1) < epsilon
      }

      function compareIntersection(a, b) {
        return comparePoint(a.x, b.x);
      }

      function comparePoint(a, b) {
        var ca = corner(a, 1),
            cb = corner(b, 1);
        return ca !== cb ? ca - cb
            : ca === 0 ? b[1] - a[1]
            : ca === 1 ? a[0] - b[0]
            : ca === 2 ? a[1] - b[1]
            : b[0] - a[0];
      }

      return function(stream) {
        var activeStream = stream,
            bufferStream = clipBuffer(),
            segments,
            polygon,
            ring,
            x__, y__, v__, // first point
            x_, y_, v_, // previous point
            first,
            clean;

        var clipStream = {
          point: point,
          lineStart: lineStart,
          lineEnd: lineEnd,
          polygonStart: polygonStart,
          polygonEnd: polygonEnd
        };

        function point(x, y) {
          if (visible(x, y)) activeStream.point(x, y);
        }

        function polygonInside() {
          var winding = 0;

          for (var i = 0, n = polygon.length; i < n; ++i) {
            for (var ring = polygon[i], j = 1, m = ring.length, point = ring[0], a0, a1, b0 = point[0], b1 = point[1]; j < m; ++j) {
              a0 = b0, a1 = b1, point = ring[j], b0 = point[0], b1 = point[1];
              if (a1 <= y1) { if (b1 > y1 && (b0 - a0) * (y1 - a1) > (b1 - a1) * (x0 - a0)) ++winding; }
              else { if (b1 <= y1 && (b0 - a0) * (y1 - a1) < (b1 - a1) * (x0 - a0)) --winding; }
            }
          }

          return winding;
        }

        // Buffer geometry within a polygon and then clip it en masse.
        function polygonStart() {
          activeStream = bufferStream, segments = [], polygon = [], clean = true;
        }

        function polygonEnd() {
          var startInside = polygonInside(),
              cleanInside = clean && startInside,
              visible = (segments = merge(segments)).length;
          if (cleanInside || visible) {
            stream.polygonStart();
            if (cleanInside) {
              stream.lineStart();
              interpolate(null, null, 1, stream);
              stream.lineEnd();
            }
            if (visible) {
              clipRejoin(segments, compareIntersection, startInside, interpolate, stream);
            }
            stream.polygonEnd();
          }
          activeStream = stream, segments = polygon = ring = null;
        }

        function lineStart() {
          clipStream.point = linePoint;
          if (polygon) polygon.push(ring = []);
          first = true;
          v_ = false;
          x_ = y_ = NaN;
        }

        // TODO rather than special-case polygons, simply handle them separately.
        // Ideally, coincident intersection points should be jittered to avoid
        // clipping issues.
        function lineEnd() {
          if (segments) {
            linePoint(x__, y__);
            if (v__ && v_) bufferStream.rejoin();
            segments.push(bufferStream.result());
          }
          clipStream.point = point;
          if (v_) activeStream.lineEnd();
        }

        function linePoint(x, y) {
          var v = visible(x, y);
          if (polygon) ring.push([x, y]);
          if (first) {
            x__ = x, y__ = y, v__ = v;
            first = false;
            if (v) {
              activeStream.lineStart();
              activeStream.point(x, y);
            }
          } else {
            if (v && v_) activeStream.point(x, y);
            else {
              var a = [x_ = Math.max(clipMin, Math.min(clipMax, x_)), y_ = Math.max(clipMin, Math.min(clipMax, y_))],
                  b = [x = Math.max(clipMin, Math.min(clipMax, x)), y = Math.max(clipMin, Math.min(clipMax, y))];
              if (clipLine(a, b, x0, y0, x1, y1)) {
                if (!v_) {
                  activeStream.lineStart();
                  activeStream.point(a[0], a[1]);
                }
                activeStream.point(b[0], b[1]);
                if (!v) activeStream.lineEnd();
                clean = false;
              } else if (v) {
                activeStream.lineStart();
                activeStream.point(x, y);
                clean = false;
              }
            }
          }
          x_ = x, y_ = y, v_ = v;
        }

        return clipStream;
      };
    }

    function extent$1() {
      var x0 = 0,
          y0 = 0,
          x1 = 960,
          y1 = 500,
          cache,
          cacheStream,
          clip;

      return clip = {
        stream: function(stream) {
          return cache && cacheStream === stream ? cache : cache = clipRectangle(x0, y0, x1, y1)(cacheStream = stream);
        },
        extent: function(_) {
          return arguments.length ? (x0 = +_[0][0], y0 = +_[0][1], x1 = +_[1][0], y1 = +_[1][1], cache = cacheStream = null, clip) : [[x0, y0], [x1, y1]];
        }
      };
    }

    var lengthSum = adder(),
        lambda0$2,
        sinPhi0$1,
        cosPhi0$1;

    var lengthStream = {
      sphere: noop$2,
      point: noop$2,
      lineStart: lengthLineStart,
      lineEnd: noop$2,
      polygonStart: noop$2,
      polygonEnd: noop$2
    };

    function lengthLineStart() {
      lengthStream.point = lengthPointFirst;
      lengthStream.lineEnd = lengthLineEnd;
    }

    function lengthLineEnd() {
      lengthStream.point = lengthStream.lineEnd = noop$2;
    }

    function lengthPointFirst(lambda, phi) {
      lambda *= radians, phi *= radians;
      lambda0$2 = lambda, sinPhi0$1 = sin$1(phi), cosPhi0$1 = cos$1(phi);
      lengthStream.point = lengthPoint;
    }

    function lengthPoint(lambda, phi) {
      lambda *= radians, phi *= radians;
      var sinPhi = sin$1(phi),
          cosPhi = cos$1(phi),
          delta = abs(lambda - lambda0$2),
          cosDelta = cos$1(delta),
          sinDelta = sin$1(delta),
          x = cosPhi * sinDelta,
          y = cosPhi0$1 * sinPhi - sinPhi0$1 * cosPhi * cosDelta,
          z = sinPhi0$1 * sinPhi + cosPhi0$1 * cosPhi * cosDelta;
      lengthSum.add(atan2(sqrt(x * x + y * y), z));
      lambda0$2 = lambda, sinPhi0$1 = sinPhi, cosPhi0$1 = cosPhi;
    }

    function length$1(object) {
      lengthSum.reset();
      geoStream(object, lengthStream);
      return +lengthSum;
    }

    var coordinates = [null, null],
        object$1 = {type: "LineString", coordinates: coordinates};

    function distance(a, b) {
      coordinates[0] = a;
      coordinates[1] = b;
      return length$1(object$1);
    }

    var containsObjectType = {
      Feature: function(object, point) {
        return containsGeometry(object.geometry, point);
      },
      FeatureCollection: function(object, point) {
        var features = object.features, i = -1, n = features.length;
        while (++i < n) if (containsGeometry(features[i].geometry, point)) return true;
        return false;
      }
    };

    var containsGeometryType = {
      Sphere: function() {
        return true;
      },
      Point: function(object, point) {
        return containsPoint(object.coordinates, point);
      },
      MultiPoint: function(object, point) {
        var coordinates = object.coordinates, i = -1, n = coordinates.length;
        while (++i < n) if (containsPoint(coordinates[i], point)) return true;
        return false;
      },
      LineString: function(object, point) {
        return containsLine(object.coordinates, point);
      },
      MultiLineString: function(object, point) {
        var coordinates = object.coordinates, i = -1, n = coordinates.length;
        while (++i < n) if (containsLine(coordinates[i], point)) return true;
        return false;
      },
      Polygon: function(object, point) {
        return containsPolygon(object.coordinates, point);
      },
      MultiPolygon: function(object, point) {
        var coordinates = object.coordinates, i = -1, n = coordinates.length;
        while (++i < n) if (containsPolygon(coordinates[i], point)) return true;
        return false;
      },
      GeometryCollection: function(object, point) {
        var geometries = object.geometries, i = -1, n = geometries.length;
        while (++i < n) if (containsGeometry(geometries[i], point)) return true;
        return false;
      }
    };

    function containsGeometry(geometry, point) {
      return geometry && containsGeometryType.hasOwnProperty(geometry.type)
          ? containsGeometryType[geometry.type](geometry, point)
          : false;
    }

    function containsPoint(coordinates, point) {
      return distance(coordinates, point) === 0;
    }

    function containsLine(coordinates, point) {
      var ao, bo, ab;
      for (var i = 0, n = coordinates.length; i < n; i++) {
        bo = distance(coordinates[i], point);
        if (bo === 0) return true;
        if (i > 0) {
          ab = distance(coordinates[i], coordinates[i - 1]);
          if (
            ab > 0 &&
            ao <= ab &&
            bo <= ab &&
            (ao + bo - ab) * (1 - Math.pow((ao - bo) / ab, 2)) < epsilon2$1 * ab
          )
            return true;
        }
        ao = bo;
      }
      return false;
    }

    function containsPolygon(coordinates, point) {
      return !!polygonContains(coordinates.map(ringRadians), pointRadians(point));
    }

    function ringRadians(ring) {
      return ring = ring.map(pointRadians), ring.pop(), ring;
    }

    function pointRadians(point) {
      return [point[0] * radians, point[1] * radians];
    }

    function contains$1(object, point) {
      return (object && containsObjectType.hasOwnProperty(object.type)
          ? containsObjectType[object.type]
          : containsGeometry)(object, point);
    }

    function graticuleX(y0, y1, dy) {
      var y = sequence(y0, y1 - epsilon$2, dy).concat(y1);
      return function(x) { return y.map(function(y) { return [x, y]; }); };
    }

    function graticuleY(x0, x1, dx) {
      var x = sequence(x0, x1 - epsilon$2, dx).concat(x1);
      return function(y) { return x.map(function(x) { return [x, y]; }); };
    }

    function graticule() {
      var x1, x0, X1, X0,
          y1, y0, Y1, Y0,
          dx = 10, dy = dx, DX = 90, DY = 360,
          x, y, X, Y,
          precision = 2.5;

      function graticule() {
        return {type: "MultiLineString", coordinates: lines()};
      }

      function lines() {
        return sequence(ceil(X0 / DX) * DX, X1, DX).map(X)
            .concat(sequence(ceil(Y0 / DY) * DY, Y1, DY).map(Y))
            .concat(sequence(ceil(x0 / dx) * dx, x1, dx).filter(function(x) { return abs(x % DX) > epsilon$2; }).map(x))
            .concat(sequence(ceil(y0 / dy) * dy, y1, dy).filter(function(y) { return abs(y % DY) > epsilon$2; }).map(y));
      }

      graticule.lines = function() {
        return lines().map(function(coordinates) { return {type: "LineString", coordinates: coordinates}; });
      };

      graticule.outline = function() {
        return {
          type: "Polygon",
          coordinates: [
            X(X0).concat(
            Y(Y1).slice(1),
            X(X1).reverse().slice(1),
            Y(Y0).reverse().slice(1))
          ]
        };
      };

      graticule.extent = function(_) {
        if (!arguments.length) return graticule.extentMinor();
        return graticule.extentMajor(_).extentMinor(_);
      };

      graticule.extentMajor = function(_) {
        if (!arguments.length) return [[X0, Y0], [X1, Y1]];
        X0 = +_[0][0], X1 = +_[1][0];
        Y0 = +_[0][1], Y1 = +_[1][1];
        if (X0 > X1) _ = X0, X0 = X1, X1 = _;
        if (Y0 > Y1) _ = Y0, Y0 = Y1, Y1 = _;
        return graticule.precision(precision);
      };

      graticule.extentMinor = function(_) {
        if (!arguments.length) return [[x0, y0], [x1, y1]];
        x0 = +_[0][0], x1 = +_[1][0];
        y0 = +_[0][1], y1 = +_[1][1];
        if (x0 > x1) _ = x0, x0 = x1, x1 = _;
        if (y0 > y1) _ = y0, y0 = y1, y1 = _;
        return graticule.precision(precision);
      };

      graticule.step = function(_) {
        if (!arguments.length) return graticule.stepMinor();
        return graticule.stepMajor(_).stepMinor(_);
      };

      graticule.stepMajor = function(_) {
        if (!arguments.length) return [DX, DY];
        DX = +_[0], DY = +_[1];
        return graticule;
      };

      graticule.stepMinor = function(_) {
        if (!arguments.length) return [dx, dy];
        dx = +_[0], dy = +_[1];
        return graticule;
      };

      graticule.precision = function(_) {
        if (!arguments.length) return precision;
        precision = +_;
        x = graticuleX(y0, y1, 90);
        y = graticuleY(x0, x1, precision);
        X = graticuleX(Y0, Y1, 90);
        Y = graticuleY(X0, X1, precision);
        return graticule;
      };

      return graticule
          .extentMajor([[-180, -90 + epsilon$2], [180, 90 - epsilon$2]])
          .extentMinor([[-180, -80 - epsilon$2], [180, 80 + epsilon$2]]);
    }

    function graticule10() {
      return graticule()();
    }

    function interpolate$1(a, b) {
      var x0 = a[0] * radians,
          y0 = a[1] * radians,
          x1 = b[0] * radians,
          y1 = b[1] * radians,
          cy0 = cos$1(y0),
          sy0 = sin$1(y0),
          cy1 = cos$1(y1),
          sy1 = sin$1(y1),
          kx0 = cy0 * cos$1(x0),
          ky0 = cy0 * sin$1(x0),
          kx1 = cy1 * cos$1(x1),
          ky1 = cy1 * sin$1(x1),
          d = 2 * asin(sqrt(haversin(y1 - y0) + cy0 * cy1 * haversin(x1 - x0))),
          k = sin$1(d);

      var interpolate = d ? function(t) {
        var B = sin$1(t *= d) / k,
            A = sin$1(d - t) / k,
            x = A * kx0 + B * kx1,
            y = A * ky0 + B * ky1,
            z = A * sy0 + B * sy1;
        return [
          atan2(y, x) * degrees$1,
          atan2(z, sqrt(x * x + y * y)) * degrees$1
        ];
      } : function() {
        return [x0 * degrees$1, y0 * degrees$1];
      };

      interpolate.distance = d;

      return interpolate;
    }

    function identity$4(x) {
      return x;
    }

    var areaSum$1 = adder(),
        areaRingSum$1 = adder(),
        x00,
        y00,
        x0$1,
        y0$1;

    var areaStream$1 = {
      point: noop$2,
      lineStart: noop$2,
      lineEnd: noop$2,
      polygonStart: function() {
        areaStream$1.lineStart = areaRingStart$1;
        areaStream$1.lineEnd = areaRingEnd$1;
      },
      polygonEnd: function() {
        areaStream$1.lineStart = areaStream$1.lineEnd = areaStream$1.point = noop$2;
        areaSum$1.add(abs(areaRingSum$1));
        areaRingSum$1.reset();
      },
      result: function() {
        var area = areaSum$1 / 2;
        areaSum$1.reset();
        return area;
      }
    };

    function areaRingStart$1() {
      areaStream$1.point = areaPointFirst$1;
    }

    function areaPointFirst$1(x, y) {
      areaStream$1.point = areaPoint$1;
      x00 = x0$1 = x, y00 = y0$1 = y;
    }

    function areaPoint$1(x, y) {
      areaRingSum$1.add(y0$1 * x - x0$1 * y);
      x0$1 = x, y0$1 = y;
    }

    function areaRingEnd$1() {
      areaPoint$1(x00, y00);
    }

    var x0$2 = Infinity,
        y0$2 = x0$2,
        x1 = -x0$2,
        y1 = x1;

    var boundsStream$1 = {
      point: boundsPoint$1,
      lineStart: noop$2,
      lineEnd: noop$2,
      polygonStart: noop$2,
      polygonEnd: noop$2,
      result: function() {
        var bounds = [[x0$2, y0$2], [x1, y1]];
        x1 = y1 = -(y0$2 = x0$2 = Infinity);
        return bounds;
      }
    };

    function boundsPoint$1(x, y) {
      if (x < x0$2) x0$2 = x;
      if (x > x1) x1 = x;
      if (y < y0$2) y0$2 = y;
      if (y > y1) y1 = y;
    }

    // TODO Enforce positive area for exterior, negative area for interior?

    var X0$1 = 0,
        Y0$1 = 0,
        Z0$1 = 0,
        X1$1 = 0,
        Y1$1 = 0,
        Z1$1 = 0,
        X2$1 = 0,
        Y2$1 = 0,
        Z2$1 = 0,
        x00$1,
        y00$1,
        x0$3,
        y0$3;

    var centroidStream$1 = {
      point: centroidPoint$1,
      lineStart: centroidLineStart$1,
      lineEnd: centroidLineEnd$1,
      polygonStart: function() {
        centroidStream$1.lineStart = centroidRingStart$1;
        centroidStream$1.lineEnd = centroidRingEnd$1;
      },
      polygonEnd: function() {
        centroidStream$1.point = centroidPoint$1;
        centroidStream$1.lineStart = centroidLineStart$1;
        centroidStream$1.lineEnd = centroidLineEnd$1;
      },
      result: function() {
        var centroid = Z2$1 ? [X2$1 / Z2$1, Y2$1 / Z2$1]
            : Z1$1 ? [X1$1 / Z1$1, Y1$1 / Z1$1]
            : Z0$1 ? [X0$1 / Z0$1, Y0$1 / Z0$1]
            : [NaN, NaN];
        X0$1 = Y0$1 = Z0$1 =
        X1$1 = Y1$1 = Z1$1 =
        X2$1 = Y2$1 = Z2$1 = 0;
        return centroid;
      }
    };

    function centroidPoint$1(x, y) {
      X0$1 += x;
      Y0$1 += y;
      ++Z0$1;
    }

    function centroidLineStart$1() {
      centroidStream$1.point = centroidPointFirstLine;
    }

    function centroidPointFirstLine(x, y) {
      centroidStream$1.point = centroidPointLine;
      centroidPoint$1(x0$3 = x, y0$3 = y);
    }

    function centroidPointLine(x, y) {
      var dx = x - x0$3, dy = y - y0$3, z = sqrt(dx * dx + dy * dy);
      X1$1 += z * (x0$3 + x) / 2;
      Y1$1 += z * (y0$3 + y) / 2;
      Z1$1 += z;
      centroidPoint$1(x0$3 = x, y0$3 = y);
    }

    function centroidLineEnd$1() {
      centroidStream$1.point = centroidPoint$1;
    }

    function centroidRingStart$1() {
      centroidStream$1.point = centroidPointFirstRing;
    }

    function centroidRingEnd$1() {
      centroidPointRing(x00$1, y00$1);
    }

    function centroidPointFirstRing(x, y) {
      centroidStream$1.point = centroidPointRing;
      centroidPoint$1(x00$1 = x0$3 = x, y00$1 = y0$3 = y);
    }

    function centroidPointRing(x, y) {
      var dx = x - x0$3,
          dy = y - y0$3,
          z = sqrt(dx * dx + dy * dy);

      X1$1 += z * (x0$3 + x) / 2;
      Y1$1 += z * (y0$3 + y) / 2;
      Z1$1 += z;

      z = y0$3 * x - x0$3 * y;
      X2$1 += z * (x0$3 + x);
      Y2$1 += z * (y0$3 + y);
      Z2$1 += z * 3;
      centroidPoint$1(x0$3 = x, y0$3 = y);
    }

    function PathContext(context) {
      this._context = context;
    }

    PathContext.prototype = {
      _radius: 4.5,
      pointRadius: function(_) {
        return this._radius = _, this;
      },
      polygonStart: function() {
        this._line = 0;
      },
      polygonEnd: function() {
        this._line = NaN;
      },
      lineStart: function() {
        this._point = 0;
      },
      lineEnd: function() {
        if (this._line === 0) this._context.closePath();
        this._point = NaN;
      },
      point: function(x, y) {
        switch (this._point) {
          case 0: {
            this._context.moveTo(x, y);
            this._point = 1;
            break;
          }
          case 1: {
            this._context.lineTo(x, y);
            break;
          }
          default: {
            this._context.moveTo(x + this._radius, y);
            this._context.arc(x, y, this._radius, 0, tau$3);
            break;
          }
        }
      },
      result: noop$2
    };

    var lengthSum$1 = adder(),
        lengthRing,
        x00$2,
        y00$2,
        x0$4,
        y0$4;

    var lengthStream$1 = {
      point: noop$2,
      lineStart: function() {
        lengthStream$1.point = lengthPointFirst$1;
      },
      lineEnd: function() {
        if (lengthRing) lengthPoint$1(x00$2, y00$2);
        lengthStream$1.point = noop$2;
      },
      polygonStart: function() {
        lengthRing = true;
      },
      polygonEnd: function() {
        lengthRing = null;
      },
      result: function() {
        var length = +lengthSum$1;
        lengthSum$1.reset();
        return length;
      }
    };

    function lengthPointFirst$1(x, y) {
      lengthStream$1.point = lengthPoint$1;
      x00$2 = x0$4 = x, y00$2 = y0$4 = y;
    }

    function lengthPoint$1(x, y) {
      x0$4 -= x, y0$4 -= y;
      lengthSum$1.add(sqrt(x0$4 * x0$4 + y0$4 * y0$4));
      x0$4 = x, y0$4 = y;
    }

    function PathString() {
      this._string = [];
    }

    PathString.prototype = {
      _radius: 4.5,
      _circle: circle$1(4.5),
      pointRadius: function(_) {
        if ((_ = +_) !== this._radius) this._radius = _, this._circle = null;
        return this;
      },
      polygonStart: function() {
        this._line = 0;
      },
      polygonEnd: function() {
        this._line = NaN;
      },
      lineStart: function() {
        this._point = 0;
      },
      lineEnd: function() {
        if (this._line === 0) this._string.push("Z");
        this._point = NaN;
      },
      point: function(x, y) {
        switch (this._point) {
          case 0: {
            this._string.push("M", x, ",", y);
            this._point = 1;
            break;
          }
          case 1: {
            this._string.push("L", x, ",", y);
            break;
          }
          default: {
            if (this._circle == null) this._circle = circle$1(this._radius);
            this._string.push("M", x, ",", y, this._circle);
            break;
          }
        }
      },
      result: function() {
        if (this._string.length) {
          var result = this._string.join("");
          this._string = [];
          return result;
        } else {
          return null;
        }
      }
    };

    function circle$1(radius) {
      return "m0," + radius
          + "a" + radius + "," + radius + " 0 1,1 0," + -2 * radius
          + "a" + radius + "," + radius + " 0 1,1 0," + 2 * radius
          + "z";
    }

    function index$1(projection, context) {
      var pointRadius = 4.5,
          projectionStream,
          contextStream;

      function path(object) {
        if (object) {
          if (typeof pointRadius === "function") contextStream.pointRadius(+pointRadius.apply(this, arguments));
          geoStream(object, projectionStream(contextStream));
        }
        return contextStream.result();
      }

      path.area = function(object) {
        geoStream(object, projectionStream(areaStream$1));
        return areaStream$1.result();
      };

      path.measure = function(object) {
        geoStream(object, projectionStream(lengthStream$1));
        return lengthStream$1.result();
      };

      path.bounds = function(object) {
        geoStream(object, projectionStream(boundsStream$1));
        return boundsStream$1.result();
      };

      path.centroid = function(object) {
        geoStream(object, projectionStream(centroidStream$1));
        return centroidStream$1.result();
      };

      path.projection = function(_) {
        return arguments.length ? (projectionStream = _ == null ? (projection = null, identity$4) : (projection = _).stream, path) : projection;
      };

      path.context = function(_) {
        if (!arguments.length) return context;
        contextStream = _ == null ? (context = null, new PathString) : new PathContext(context = _);
        if (typeof pointRadius !== "function") contextStream.pointRadius(pointRadius);
        return path;
      };

      path.pointRadius = function(_) {
        if (!arguments.length) return pointRadius;
        pointRadius = typeof _ === "function" ? _ : (contextStream.pointRadius(+_), +_);
        return path;
      };

      return path.projection(projection).context(context);
    }

    function transform(methods) {
      return {
        stream: transformer(methods)
      };
    }

    function transformer(methods) {
      return function(stream) {
        var s = new TransformStream;
        for (var key in methods) s[key] = methods[key];
        s.stream = stream;
        return s;
      };
    }

    function TransformStream() {}

    TransformStream.prototype = {
      constructor: TransformStream,
      point: function(x, y) { this.stream.point(x, y); },
      sphere: function() { this.stream.sphere(); },
      lineStart: function() { this.stream.lineStart(); },
      lineEnd: function() { this.stream.lineEnd(); },
      polygonStart: function() { this.stream.polygonStart(); },
      polygonEnd: function() { this.stream.polygonEnd(); }
    };

    function fit(projection, fitBounds, object) {
      var clip = projection.clipExtent && projection.clipExtent();
      projection.scale(150).translate([0, 0]);
      if (clip != null) projection.clipExtent(null);
      geoStream(object, projection.stream(boundsStream$1));
      fitBounds(boundsStream$1.result());
      if (clip != null) projection.clipExtent(clip);
      return projection;
    }

    function fitExtent(projection, extent, object) {
      return fit(projection, function(b) {
        var w = extent[1][0] - extent[0][0],
            h = extent[1][1] - extent[0][1],
            k = Math.min(w / (b[1][0] - b[0][0]), h / (b[1][1] - b[0][1])),
            x = +extent[0][0] + (w - k * (b[1][0] + b[0][0])) / 2,
            y = +extent[0][1] + (h - k * (b[1][1] + b[0][1])) / 2;
        projection.scale(150 * k).translate([x, y]);
      }, object);
    }

    function fitSize(projection, size, object) {
      return fitExtent(projection, [[0, 0], size], object);
    }

    function fitWidth(projection, width, object) {
      return fit(projection, function(b) {
        var w = +width,
            k = w / (b[1][0] - b[0][0]),
            x = (w - k * (b[1][0] + b[0][0])) / 2,
            y = -k * b[0][1];
        projection.scale(150 * k).translate([x, y]);
      }, object);
    }

    function fitHeight(projection, height, object) {
      return fit(projection, function(b) {
        var h = +height,
            k = h / (b[1][1] - b[0][1]),
            x = -k * b[0][0],
            y = (h - k * (b[1][1] + b[0][1])) / 2;
        projection.scale(150 * k).translate([x, y]);
      }, object);
    }

    var maxDepth = 16, // maximum depth of subdivision
        cosMinDistance = cos$1(30 * radians); // cos(minimum angular distance)

    function resample(project, delta2) {
      return +delta2 ? resample$1(project, delta2) : resampleNone(project);
    }

    function resampleNone(project) {
      return transformer({
        point: function(x, y) {
          x = project(x, y);
          this.stream.point(x[0], x[1]);
        }
      });
    }

    function resample$1(project, delta2) {

      function resampleLineTo(x0, y0, lambda0, a0, b0, c0, x1, y1, lambda1, a1, b1, c1, depth, stream) {
        var dx = x1 - x0,
            dy = y1 - y0,
            d2 = dx * dx + dy * dy;
        if (d2 > 4 * delta2 && depth--) {
          var a = a0 + a1,
              b = b0 + b1,
              c = c0 + c1,
              m = sqrt(a * a + b * b + c * c),
              phi2 = asin(c /= m),
              lambda2 = abs(abs(c) - 1) < epsilon$2 || abs(lambda0 - lambda1) < epsilon$2 ? (lambda0 + lambda1) / 2 : atan2(b, a),
              p = project(lambda2, phi2),
              x2 = p[0],
              y2 = p[1],
              dx2 = x2 - x0,
              dy2 = y2 - y0,
              dz = dy * dx2 - dx * dy2;
          if (dz * dz / d2 > delta2 // perpendicular projected distance
              || abs((dx * dx2 + dy * dy2) / d2 - 0.5) > 0.3 // midpoint close to an end
              || a0 * a1 + b0 * b1 + c0 * c1 < cosMinDistance) { // angular distance
            resampleLineTo(x0, y0, lambda0, a0, b0, c0, x2, y2, lambda2, a /= m, b /= m, c, depth, stream);
            stream.point(x2, y2);
            resampleLineTo(x2, y2, lambda2, a, b, c, x1, y1, lambda1, a1, b1, c1, depth, stream);
          }
        }
      }
      return function(stream) {
        var lambda00, x00, y00, a00, b00, c00, // first point
            lambda0, x0, y0, a0, b0, c0; // previous point

        var resampleStream = {
          point: point,
          lineStart: lineStart,
          lineEnd: lineEnd,
          polygonStart: function() { stream.polygonStart(); resampleStream.lineStart = ringStart; },
          polygonEnd: function() { stream.polygonEnd(); resampleStream.lineStart = lineStart; }
        };

        function point(x, y) {
          x = project(x, y);
          stream.point(x[0], x[1]);
        }

        function lineStart() {
          x0 = NaN;
          resampleStream.point = linePoint;
          stream.lineStart();
        }

        function linePoint(lambda, phi) {
          var c = cartesian([lambda, phi]), p = project(lambda, phi);
          resampleLineTo(x0, y0, lambda0, a0, b0, c0, x0 = p[0], y0 = p[1], lambda0 = lambda, a0 = c[0], b0 = c[1], c0 = c[2], maxDepth, stream);
          stream.point(x0, y0);
        }

        function lineEnd() {
          resampleStream.point = point;
          stream.lineEnd();
        }

        function ringStart() {
          lineStart();
          resampleStream.point = ringPoint;
          resampleStream.lineEnd = ringEnd;
        }

        function ringPoint(lambda, phi) {
          linePoint(lambda00 = lambda, phi), x00 = x0, y00 = y0, a00 = a0, b00 = b0, c00 = c0;
          resampleStream.point = linePoint;
        }

        function ringEnd() {
          resampleLineTo(x0, y0, lambda0, a0, b0, c0, x00, y00, lambda00, a00, b00, c00, maxDepth, stream);
          resampleStream.lineEnd = lineEnd;
          lineEnd();
        }

        return resampleStream;
      };
    }

    var transformRadians = transformer({
      point: function(x, y) {
        this.stream.point(x * radians, y * radians);
      }
    });

    function transformRotate(rotate) {
      return transformer({
        point: function(x, y) {
          var r = rotate(x, y);
          return this.stream.point(r[0], r[1]);
        }
      });
    }

    function scaleTranslate(k, dx, dy, sx, sy) {
      function transform(x, y) {
        x *= sx; y *= sy;
        return [dx + k * x, dy - k * y];
      }
      transform.invert = function(x, y) {
        return [(x - dx) / k * sx, (dy - y) / k * sy];
      };
      return transform;
    }

    function scaleTranslateRotate(k, dx, dy, sx, sy, alpha) {
      var cosAlpha = cos$1(alpha),
          sinAlpha = sin$1(alpha),
          a = cosAlpha * k,
          b = sinAlpha * k,
          ai = cosAlpha / k,
          bi = sinAlpha / k,
          ci = (sinAlpha * dy - cosAlpha * dx) / k,
          fi = (sinAlpha * dx + cosAlpha * dy) / k;
      function transform(x, y) {
        x *= sx; y *= sy;
        return [a * x - b * y + dx, dy - b * x - a * y];
      }
      transform.invert = function(x, y) {
        return [sx * (ai * x - bi * y + ci), sy * (fi - bi * x - ai * y)];
      };
      return transform;
    }

    function projection(project) {
      return projectionMutator(function() { return project; })();
    }

    function projectionMutator(projectAt) {
      var project,
          k = 150, // scale
          x = 480, y = 250, // translate
          lambda = 0, phi = 0, // center
          deltaLambda = 0, deltaPhi = 0, deltaGamma = 0, rotate, // pre-rotate
          alpha = 0, // post-rotate angle
          sx = 1, // reflectX
          sy = 1, // reflectX
          theta = null, preclip = clipAntimeridian, // pre-clip angle
          x0 = null, y0, x1, y1, postclip = identity$4, // post-clip extent
          delta2 = 0.5, // precision
          projectResample,
          projectTransform,
          projectRotateTransform,
          cache,
          cacheStream;

      function projection(point) {
        return projectRotateTransform(point[0] * radians, point[1] * radians);
      }

      function invert(point) {
        point = projectRotateTransform.invert(point[0], point[1]);
        return point && [point[0] * degrees$1, point[1] * degrees$1];
      }

      projection.stream = function(stream) {
        return cache && cacheStream === stream ? cache : cache = transformRadians(transformRotate(rotate)(preclip(projectResample(postclip(cacheStream = stream)))));
      };

      projection.preclip = function(_) {
        return arguments.length ? (preclip = _, theta = undefined, reset()) : preclip;
      };

      projection.postclip = function(_) {
        return arguments.length ? (postclip = _, x0 = y0 = x1 = y1 = null, reset()) : postclip;
      };

      projection.clipAngle = function(_) {
        return arguments.length ? (preclip = +_ ? clipCircle(theta = _ * radians) : (theta = null, clipAntimeridian), reset()) : theta * degrees$1;
      };

      projection.clipExtent = function(_) {
        return arguments.length ? (postclip = _ == null ? (x0 = y0 = x1 = y1 = null, identity$4) : clipRectangle(x0 = +_[0][0], y0 = +_[0][1], x1 = +_[1][0], y1 = +_[1][1]), reset()) : x0 == null ? null : [[x0, y0], [x1, y1]];
      };

      projection.scale = function(_) {
        return arguments.length ? (k = +_, recenter()) : k;
      };

      projection.translate = function(_) {
        return arguments.length ? (x = +_[0], y = +_[1], recenter()) : [x, y];
      };

      projection.center = function(_) {
        return arguments.length ? (lambda = _[0] % 360 * radians, phi = _[1] % 360 * radians, recenter()) : [lambda * degrees$1, phi * degrees$1];
      };

      projection.rotate = function(_) {
        return arguments.length ? (deltaLambda = _[0] % 360 * radians, deltaPhi = _[1] % 360 * radians, deltaGamma = _.length > 2 ? _[2] % 360 * radians : 0, recenter()) : [deltaLambda * degrees$1, deltaPhi * degrees$1, deltaGamma * degrees$1];
      };

      projection.angle = function(_) {
        return arguments.length ? (alpha = _ % 360 * radians, recenter()) : alpha * degrees$1;
      };

      projection.reflectX = function(_) {
        return arguments.length ? (sx = _ ? -1 : 1, recenter()) : sx < 0;
      };

      projection.reflectY = function(_) {
        return arguments.length ? (sy = _ ? -1 : 1, recenter()) : sy < 0;
      };

      projection.precision = function(_) {
        return arguments.length ? (projectResample = resample(projectTransform, delta2 = _ * _), reset()) : sqrt(delta2);
      };

      projection.fitExtent = function(extent, object) {
        return fitExtent(projection, extent, object);
      };

      projection.fitSize = function(size, object) {
        return fitSize(projection, size, object);
      };

      projection.fitWidth = function(width, object) {
        return fitWidth(projection, width, object);
      };

      projection.fitHeight = function(height, object) {
        return fitHeight(projection, height, object);
      };

      function recenter() {
        var center = scaleTranslateRotate(k, 0, 0, sx, sy, alpha).apply(null, project(lambda, phi)),
            transform = (alpha ? scaleTranslateRotate : scaleTranslate)(k, x - center[0], y - center[1], sx, sy, alpha);
        rotate = rotateRadians(deltaLambda, deltaPhi, deltaGamma);
        projectTransform = compose(project, transform);
        projectRotateTransform = compose(rotate, projectTransform);
        projectResample = resample(projectTransform, delta2);
        return reset();
      }

      function reset() {
        cache = cacheStream = null;
        return projection;
      }

      return function() {
        project = projectAt.apply(this, arguments);
        projection.invert = project.invert && invert;
        return recenter();
      };
    }

    function conicProjection(projectAt) {
      var phi0 = 0,
          phi1 = pi$3 / 3,
          m = projectionMutator(projectAt),
          p = m(phi0, phi1);

      p.parallels = function(_) {
        return arguments.length ? m(phi0 = _[0] * radians, phi1 = _[1] * radians) : [phi0 * degrees$1, phi1 * degrees$1];
      };

      return p;
    }

    function cylindricalEqualAreaRaw(phi0) {
      var cosPhi0 = cos$1(phi0);

      function forward(lambda, phi) {
        return [lambda * cosPhi0, sin$1(phi) / cosPhi0];
      }

      forward.invert = function(x, y) {
        return [x / cosPhi0, asin(y * cosPhi0)];
      };

      return forward;
    }

    function conicEqualAreaRaw(y0, y1) {
      var sy0 = sin$1(y0), n = (sy0 + sin$1(y1)) / 2;

      // Are the parallels symmetrical around the Equator?
      if (abs(n) < epsilon$2) return cylindricalEqualAreaRaw(y0);

      var c = 1 + sy0 * (2 * n - sy0), r0 = sqrt(c) / n;

      function project(x, y) {
        var r = sqrt(c - 2 * n * sin$1(y)) / n;
        return [r * sin$1(x *= n), r0 - r * cos$1(x)];
      }

      project.invert = function(x, y) {
        var r0y = r0 - y,
            l = atan2(x, abs(r0y)) * sign(r0y);
        if (r0y * n < 0)
          l -= pi$3 * sign(x) * sign(r0y);
        return [l / n, asin((c - (x * x + r0y * r0y) * n * n) / (2 * n))];
      };

      return project;
    }

    function conicEqualArea() {
      return conicProjection(conicEqualAreaRaw)
          .scale(155.424)
          .center([0, 33.6442]);
    }

    function albers() {
      return conicEqualArea()
          .parallels([29.5, 45.5])
          .scale(1070)
          .translate([480, 250])
          .rotate([96, 0])
          .center([-0.6, 38.7]);
    }

    // The projections must have mutually exclusive clip regions on the sphere,
    // as this will avoid emitting interleaving lines and polygons.
    function multiplex(streams) {
      var n = streams.length;
      return {
        point: function(x, y) { var i = -1; while (++i < n) streams[i].point(x, y); },
        sphere: function() { var i = -1; while (++i < n) streams[i].sphere(); },
        lineStart: function() { var i = -1; while (++i < n) streams[i].lineStart(); },
        lineEnd: function() { var i = -1; while (++i < n) streams[i].lineEnd(); },
        polygonStart: function() { var i = -1; while (++i < n) streams[i].polygonStart(); },
        polygonEnd: function() { var i = -1; while (++i < n) streams[i].polygonEnd(); }
      };
    }

    // A composite projection for the United States, configured by default for
    // 960×500. The projection also works quite well at 960×600 if you change the
    // scale to 1285 and adjust the translate accordingly. The set of standard
    // parallels for each region comes from USGS, which is published here:
    // http://egsc.usgs.gov/isb/pubs/MapProjections/projections.html#albers
    function albersUsa() {
      var cache,
          cacheStream,
          lower48 = albers(), lower48Point,
          alaska = conicEqualArea().rotate([154, 0]).center([-2, 58.5]).parallels([55, 65]), alaskaPoint, // EPSG:3338
          hawaii = conicEqualArea().rotate([157, 0]).center([-3, 19.9]).parallels([8, 18]), hawaiiPoint, // ESRI:102007
          point, pointStream = {point: function(x, y) { point = [x, y]; }};

      function albersUsa(coordinates) {
        var x = coordinates[0], y = coordinates[1];
        return point = null,
            (lower48Point.point(x, y), point)
            || (alaskaPoint.point(x, y), point)
            || (hawaiiPoint.point(x, y), point);
      }

      albersUsa.invert = function(coordinates) {
        var k = lower48.scale(),
            t = lower48.translate(),
            x = (coordinates[0] - t[0]) / k,
            y = (coordinates[1] - t[1]) / k;
        return (y >= 0.120 && y < 0.234 && x >= -0.425 && x < -0.214 ? alaska
            : y >= 0.166 && y < 0.234 && x >= -0.214 && x < -0.115 ? hawaii
            : lower48).invert(coordinates);
      };

      albersUsa.stream = function(stream) {
        return cache && cacheStream === stream ? cache : cache = multiplex([lower48.stream(cacheStream = stream), alaska.stream(stream), hawaii.stream(stream)]);
      };

      albersUsa.precision = function(_) {
        if (!arguments.length) return lower48.precision();
        lower48.precision(_), alaska.precision(_), hawaii.precision(_);
        return reset();
      };

      albersUsa.scale = function(_) {
        if (!arguments.length) return lower48.scale();
        lower48.scale(_), alaska.scale(_ * 0.35), hawaii.scale(_);
        return albersUsa.translate(lower48.translate());
      };

      albersUsa.translate = function(_) {
        if (!arguments.length) return lower48.translate();
        var k = lower48.scale(), x = +_[0], y = +_[1];

        lower48Point = lower48
            .translate(_)
            .clipExtent([[x - 0.455 * k, y - 0.238 * k], [x + 0.455 * k, y + 0.238 * k]])
            .stream(pointStream);

        alaskaPoint = alaska
            .translate([x - 0.307 * k, y + 0.201 * k])
            .clipExtent([[x - 0.425 * k + epsilon$2, y + 0.120 * k + epsilon$2], [x - 0.214 * k - epsilon$2, y + 0.234 * k - epsilon$2]])
            .stream(pointStream);

        hawaiiPoint = hawaii
            .translate([x - 0.205 * k, y + 0.212 * k])
            .clipExtent([[x - 0.214 * k + epsilon$2, y + 0.166 * k + epsilon$2], [x - 0.115 * k - epsilon$2, y + 0.234 * k - epsilon$2]])
            .stream(pointStream);

        return reset();
      };

      albersUsa.fitExtent = function(extent, object) {
        return fitExtent(albersUsa, extent, object);
      };

      albersUsa.fitSize = function(size, object) {
        return fitSize(albersUsa, size, object);
      };

      albersUsa.fitWidth = function(width, object) {
        return fitWidth(albersUsa, width, object);
      };

      albersUsa.fitHeight = function(height, object) {
        return fitHeight(albersUsa, height, object);
      };

      function reset() {
        cache = cacheStream = null;
        return albersUsa;
      }

      return albersUsa.scale(1070);
    }

    function azimuthalRaw(scale) {
      return function(x, y) {
        var cx = cos$1(x),
            cy = cos$1(y),
            k = scale(cx * cy);
        return [
          k * cy * sin$1(x),
          k * sin$1(y)
        ];
      }
    }

    function azimuthalInvert(angle) {
      return function(x, y) {
        var z = sqrt(x * x + y * y),
            c = angle(z),
            sc = sin$1(c),
            cc = cos$1(c);
        return [
          atan2(x * sc, z * cc),
          asin(z && y * sc / z)
        ];
      }
    }

    var azimuthalEqualAreaRaw = azimuthalRaw(function(cxcy) {
      return sqrt(2 / (1 + cxcy));
    });

    azimuthalEqualAreaRaw.invert = azimuthalInvert(function(z) {
      return 2 * asin(z / 2);
    });

    function azimuthalEqualArea() {
      return projection(azimuthalEqualAreaRaw)
          .scale(124.75)
          .clipAngle(180 - 1e-3);
    }

    var azimuthalEquidistantRaw = azimuthalRaw(function(c) {
      return (c = acos(c)) && c / sin$1(c);
    });

    azimuthalEquidistantRaw.invert = azimuthalInvert(function(z) {
      return z;
    });

    function azimuthalEquidistant() {
      return projection(azimuthalEquidistantRaw)
          .scale(79.4188)
          .clipAngle(180 - 1e-3);
    }

    function mercatorRaw(lambda, phi) {
      return [lambda, log(tan((halfPi$2 + phi) / 2))];
    }

    mercatorRaw.invert = function(x, y) {
      return [x, 2 * atan(exp(y)) - halfPi$2];
    };

    function mercator() {
      return mercatorProjection(mercatorRaw)
          .scale(961 / tau$3);
    }

    function mercatorProjection(project) {
      var m = projection(project),
          center = m.center,
          scale = m.scale,
          translate = m.translate,
          clipExtent = m.clipExtent,
          x0 = null, y0, x1, y1; // clip extent

      m.scale = function(_) {
        return arguments.length ? (scale(_), reclip()) : scale();
      };

      m.translate = function(_) {
        return arguments.length ? (translate(_), reclip()) : translate();
      };

      m.center = function(_) {
        return arguments.length ? (center(_), reclip()) : center();
      };

      m.clipExtent = function(_) {
        return arguments.length ? ((_ == null ? x0 = y0 = x1 = y1 = null : (x0 = +_[0][0], y0 = +_[0][1], x1 = +_[1][0], y1 = +_[1][1])), reclip()) : x0 == null ? null : [[x0, y0], [x1, y1]];
      };

      function reclip() {
        var k = pi$3 * scale(),
            t = m(rotation(m.rotate()).invert([0, 0]));
        return clipExtent(x0 == null
            ? [[t[0] - k, t[1] - k], [t[0] + k, t[1] + k]] : project === mercatorRaw
            ? [[Math.max(t[0] - k, x0), y0], [Math.min(t[0] + k, x1), y1]]
            : [[x0, Math.max(t[1] - k, y0)], [x1, Math.min(t[1] + k, y1)]]);
      }

      return reclip();
    }

    function tany(y) {
      return tan((halfPi$2 + y) / 2);
    }

    function conicConformalRaw(y0, y1) {
      var cy0 = cos$1(y0),
          n = y0 === y1 ? sin$1(y0) : log(cy0 / cos$1(y1)) / log(tany(y1) / tany(y0)),
          f = cy0 * pow(tany(y0), n) / n;

      if (!n) return mercatorRaw;

      function project(x, y) {
        if (f > 0) { if (y < -halfPi$2 + epsilon$2) y = -halfPi$2 + epsilon$2; }
        else { if (y > halfPi$2 - epsilon$2) y = halfPi$2 - epsilon$2; }
        var r = f / pow(tany(y), n);
        return [r * sin$1(n * x), f - r * cos$1(n * x)];
      }

      project.invert = function(x, y) {
        var fy = f - y, r = sign(n) * sqrt(x * x + fy * fy),
          l = atan2(x, abs(fy)) * sign(fy);
        if (fy * n < 0)
          l -= pi$3 * sign(x) * sign(fy);
        return [l / n, 2 * atan(pow(f / r, 1 / n)) - halfPi$2];
      };

      return project;
    }

    function conicConformal() {
      return conicProjection(conicConformalRaw)
          .scale(109.5)
          .parallels([30, 30]);
    }

    function equirectangularRaw(lambda, phi) {
      return [lambda, phi];
    }

    equirectangularRaw.invert = equirectangularRaw;

    function equirectangular() {
      return projection(equirectangularRaw)
          .scale(152.63);
    }

    function conicEquidistantRaw(y0, y1) {
      var cy0 = cos$1(y0),
          n = y0 === y1 ? sin$1(y0) : (cy0 - cos$1(y1)) / (y1 - y0),
          g = cy0 / n + y0;

      if (abs(n) < epsilon$2) return equirectangularRaw;

      function project(x, y) {
        var gy = g - y, nx = n * x;
        return [gy * sin$1(nx), g - gy * cos$1(nx)];
      }

      project.invert = function(x, y) {
        var gy = g - y,
            l = atan2(x, abs(gy)) * sign(gy);
        if (gy * n < 0)
          l -= pi$3 * sign(x) * sign(gy);
        return [l / n, g - sign(n) * sqrt(x * x + gy * gy)];
      };

      return project;
    }

    function conicEquidistant() {
      return conicProjection(conicEquidistantRaw)
          .scale(131.154)
          .center([0, 13.9389]);
    }

    var A1 = 1.340264,
        A2 = -0.081106,
        A3 = 0.000893,
        A4 = 0.003796,
        M = sqrt(3) / 2,
        iterations = 12;

    function equalEarthRaw(lambda, phi) {
      var l = asin(M * sin$1(phi)), l2 = l * l, l6 = l2 * l2 * l2;
      return [
        lambda * cos$1(l) / (M * (A1 + 3 * A2 * l2 + l6 * (7 * A3 + 9 * A4 * l2))),
        l * (A1 + A2 * l2 + l6 * (A3 + A4 * l2))
      ];
    }

    equalEarthRaw.invert = function(x, y) {
      var l = y, l2 = l * l, l6 = l2 * l2 * l2;
      for (var i = 0, delta, fy, fpy; i < iterations; ++i) {
        fy = l * (A1 + A2 * l2 + l6 * (A3 + A4 * l2)) - y;
        fpy = A1 + 3 * A2 * l2 + l6 * (7 * A3 + 9 * A4 * l2);
        l -= delta = fy / fpy, l2 = l * l, l6 = l2 * l2 * l2;
        if (abs(delta) < epsilon2$1) break;
      }
      return [
        M * x * (A1 + 3 * A2 * l2 + l6 * (7 * A3 + 9 * A4 * l2)) / cos$1(l),
        asin(sin$1(l) / M)
      ];
    };

    function equalEarth() {
      return projection(equalEarthRaw)
          .scale(177.158);
    }

    function gnomonicRaw(x, y) {
      var cy = cos$1(y), k = cos$1(x) * cy;
      return [cy * sin$1(x) / k, sin$1(y) / k];
    }

    gnomonicRaw.invert = azimuthalInvert(atan);

    function gnomonic() {
      return projection(gnomonicRaw)
          .scale(144.049)
          .clipAngle(60);
    }

    function identity$5() {
      var k = 1, tx = 0, ty = 0, sx = 1, sy = 1, // scale, translate and reflect
          alpha = 0, ca, sa, // angle
          x0 = null, y0, x1, y1, // clip extent
          kx = 1, ky = 1,
          transform = transformer({
            point: function(x, y) {
              var p = projection([x, y]);
              this.stream.point(p[0], p[1]);
            }
          }),
          postclip = identity$4,
          cache,
          cacheStream;

      function reset() {
        kx = k * sx;
        ky = k * sy;
        cache = cacheStream = null;
        return projection;
      }

      function projection (p) {
        var x = p[0] * kx, y = p[1] * ky;
        if (alpha) {
          var t = y * ca - x * sa;
          x = x * ca + y * sa;
          y = t;
        }    
        return [x + tx, y + ty];
      }
      projection.invert = function(p) {
        var x = p[0] - tx, y = p[1] - ty;
        if (alpha) {
          var t = y * ca + x * sa;
          x = x * ca - y * sa;
          y = t;
        }
        return [x / kx, y / ky];
      };
      projection.stream = function(stream) {
        return cache && cacheStream === stream ? cache : cache = transform(postclip(cacheStream = stream));
      };
      projection.postclip = function(_) {
        return arguments.length ? (postclip = _, x0 = y0 = x1 = y1 = null, reset()) : postclip;
      };
      projection.clipExtent = function(_) {
        return arguments.length ? (postclip = _ == null ? (x0 = y0 = x1 = y1 = null, identity$4) : clipRectangle(x0 = +_[0][0], y0 = +_[0][1], x1 = +_[1][0], y1 = +_[1][1]), reset()) : x0 == null ? null : [[x0, y0], [x1, y1]];
      };
      projection.scale = function(_) {
        return arguments.length ? (k = +_, reset()) : k;
      };
      projection.translate = function(_) {
        return arguments.length ? (tx = +_[0], ty = +_[1], reset()) : [tx, ty];
      };
      projection.angle = function(_) {
        return arguments.length ? (alpha = _ % 360 * radians, sa = sin$1(alpha), ca = cos$1(alpha), reset()) : alpha * degrees$1;
      };
      projection.reflectX = function(_) {
        return arguments.length ? (sx = _ ? -1 : 1, reset()) : sx < 0;
      };
      projection.reflectY = function(_) {
        return arguments.length ? (sy = _ ? -1 : 1, reset()) : sy < 0;
      };
      projection.fitExtent = function(extent, object) {
        return fitExtent(projection, extent, object);
      };
      projection.fitSize = function(size, object) {
        return fitSize(projection, size, object);
      };
      projection.fitWidth = function(width, object) {
        return fitWidth(projection, width, object);
      };
      projection.fitHeight = function(height, object) {
        return fitHeight(projection, height, object);
      };

      return projection;
    }

    function naturalEarth1Raw(lambda, phi) {
      var phi2 = phi * phi, phi4 = phi2 * phi2;
      return [
        lambda * (0.8707 - 0.131979 * phi2 + phi4 * (-0.013791 + phi4 * (0.003971 * phi2 - 0.001529 * phi4))),
        phi * (1.007226 + phi2 * (0.015085 + phi4 * (-0.044475 + 0.028874 * phi2 - 0.005916 * phi4)))
      ];
    }

    naturalEarth1Raw.invert = function(x, y) {
      var phi = y, i = 25, delta;
      do {
        var phi2 = phi * phi, phi4 = phi2 * phi2;
        phi -= delta = (phi * (1.007226 + phi2 * (0.015085 + phi4 * (-0.044475 + 0.028874 * phi2 - 0.005916 * phi4))) - y) /
            (1.007226 + phi2 * (0.015085 * 3 + phi4 * (-0.044475 * 7 + 0.028874 * 9 * phi2 - 0.005916 * 11 * phi4)));
      } while (abs(delta) > epsilon$2 && --i > 0);
      return [
        x / (0.8707 + (phi2 = phi * phi) * (-0.131979 + phi2 * (-0.013791 + phi2 * phi2 * phi2 * (0.003971 - 0.001529 * phi2)))),
        phi
      ];
    };

    function naturalEarth1() {
      return projection(naturalEarth1Raw)
          .scale(175.295);
    }

    function orthographicRaw(x, y) {
      return [cos$1(y) * sin$1(x), sin$1(y)];
    }

    orthographicRaw.invert = azimuthalInvert(asin);

    function orthographic() {
      return projection(orthographicRaw)
          .scale(249.5)
          .clipAngle(90 + epsilon$2);
    }

    function stereographicRaw(x, y) {
      var cy = cos$1(y), k = 1 + cos$1(x) * cy;
      return [cy * sin$1(x) / k, sin$1(y) / k];
    }

    stereographicRaw.invert = azimuthalInvert(function(z) {
      return 2 * atan(z);
    });

    function stereographic() {
      return projection(stereographicRaw)
          .scale(250)
          .clipAngle(142);
    }

    function transverseMercatorRaw(lambda, phi) {
      return [log(tan((halfPi$2 + phi) / 2)), -lambda];
    }

    transverseMercatorRaw.invert = function(x, y) {
      return [-y, 2 * atan(exp(x)) - halfPi$2];
    };

    function transverseMercator() {
      var m = mercatorProjection(transverseMercatorRaw),
          center = m.center,
          rotate = m.rotate;

      m.center = function(_) {
        return arguments.length ? center([-_[1], _[0]]) : (_ = center(), [_[1], -_[0]]);
      };

      m.rotate = function(_) {
        return arguments.length ? rotate([_[0], _[1], _.length > 2 ? _[2] + 90 : 90]) : (_ = rotate(), [_[0], _[1], _[2] - 90]);
      };

      return rotate([0, 0, 90])
          .scale(159.155);
    }

    function defaultSeparation(a, b) {
      return a.parent === b.parent ? 1 : 2;
    }

    function meanX(children) {
      return children.reduce(meanXReduce, 0) / children.length;
    }

    function meanXReduce(x, c) {
      return x + c.x;
    }

    function maxY(children) {
      return 1 + children.reduce(maxYReduce, 0);
    }

    function maxYReduce(y, c) {
      return Math.max(y, c.y);
    }

    function leafLeft(node) {
      var children;
      while (children = node.children) node = children[0];
      return node;
    }

    function leafRight(node) {
      var children;
      while (children = node.children) node = children[children.length - 1];
      return node;
    }

    function cluster() {
      var separation = defaultSeparation,
          dx = 1,
          dy = 1,
          nodeSize = false;

      function cluster(root) {
        var previousNode,
            x = 0;

        // First walk, computing the initial x & y values.
        root.eachAfter(function(node) {
          var children = node.children;
          if (children) {
            node.x = meanX(children);
            node.y = maxY(children);
          } else {
            node.x = previousNode ? x += separation(node, previousNode) : 0;
            node.y = 0;
            previousNode = node;
          }
        });

        var left = leafLeft(root),
            right = leafRight(root),
            x0 = left.x - separation(left, right) / 2,
            x1 = right.x + separation(right, left) / 2;

        // Second walk, normalizing x & y to the desired size.
        return root.eachAfter(nodeSize ? function(node) {
          node.x = (node.x - root.x) * dx;
          node.y = (root.y - node.y) * dy;
        } : function(node) {
          node.x = (node.x - x0) / (x1 - x0) * dx;
          node.y = (1 - (root.y ? node.y / root.y : 1)) * dy;
        });
      }

      cluster.separation = function(x) {
        return arguments.length ? (separation = x, cluster) : separation;
      };

      cluster.size = function(x) {
        return arguments.length ? (nodeSize = false, dx = +x[0], dy = +x[1], cluster) : (nodeSize ? null : [dx, dy]);
      };

      cluster.nodeSize = function(x) {
        return arguments.length ? (nodeSize = true, dx = +x[0], dy = +x[1], cluster) : (nodeSize ? [dx, dy] : null);
      };

      return cluster;
    }

    function count(node) {
      var sum = 0,
          children = node.children,
          i = children && children.length;
      if (!i) sum = 1;
      else while (--i >= 0) sum += children[i].value;
      node.value = sum;
    }

    function node_count() {
      return this.eachAfter(count);
    }

    function node_each(callback) {
      var node = this, current, next = [node], children, i, n;
      do {
        current = next.reverse(), next = [];
        while (node = current.pop()) {
          callback(node), children = node.children;
          if (children) for (i = 0, n = children.length; i < n; ++i) {
            next.push(children[i]);
          }
        }
      } while (next.length);
      return this;
    }

    function node_eachBefore(callback) {
      var node = this, nodes = [node], children, i;
      while (node = nodes.pop()) {
        callback(node), children = node.children;
        if (children) for (i = children.length - 1; i >= 0; --i) {
          nodes.push(children[i]);
        }
      }
      return this;
    }

    function node_eachAfter(callback) {
      var node = this, nodes = [node], next = [], children, i, n;
      while (node = nodes.pop()) {
        next.push(node), children = node.children;
        if (children) for (i = 0, n = children.length; i < n; ++i) {
          nodes.push(children[i]);
        }
      }
      while (node = next.pop()) {
        callback(node);
      }
      return this;
    }

    function node_sum(value) {
      return this.eachAfter(function(node) {
        var sum = +value(node.data) || 0,
            children = node.children,
            i = children && children.length;
        while (--i >= 0) sum += children[i].value;
        node.value = sum;
      });
    }

    function node_sort(compare) {
      return this.eachBefore(function(node) {
        if (node.children) {
          node.children.sort(compare);
        }
      });
    }

    function node_path(end) {
      var start = this,
          ancestor = leastCommonAncestor(start, end),
          nodes = [start];
      while (start !== ancestor) {
        start = start.parent;
        nodes.push(start);
      }
      var k = nodes.length;
      while (end !== ancestor) {
        nodes.splice(k, 0, end);
        end = end.parent;
      }
      return nodes;
    }

    function leastCommonAncestor(a, b) {
      if (a === b) return a;
      var aNodes = a.ancestors(),
          bNodes = b.ancestors(),
          c = null;
      a = aNodes.pop();
      b = bNodes.pop();
      while (a === b) {
        c = a;
        a = aNodes.pop();
        b = bNodes.pop();
      }
      return c;
    }

    function node_ancestors() {
      var node = this, nodes = [node];
      while (node = node.parent) {
        nodes.push(node);
      }
      return nodes;
    }

    function node_descendants() {
      var nodes = [];
      this.each(function(node) {
        nodes.push(node);
      });
      return nodes;
    }

    function node_leaves() {
      var leaves = [];
      this.eachBefore(function(node) {
        if (!node.children) {
          leaves.push(node);
        }
      });
      return leaves;
    }

    function node_links() {
      var root = this, links = [];
      root.each(function(node) {
        if (node !== root) { // Don’t include the root’s parent, if any.
          links.push({source: node.parent, target: node});
        }
      });
      return links;
    }

    function hierarchy(data, children) {
      var root = new Node$1(data),
          valued = +data.value && (root.value = data.value),
          node,
          nodes = [root],
          child,
          childs,
          i,
          n;

      if (children == null) children = defaultChildren;

      while (node = nodes.pop()) {
        if (valued) node.value = +node.data.value;
        if ((childs = children(node.data)) && (n = childs.length)) {
          node.children = new Array(n);
          for (i = n - 1; i >= 0; --i) {
            nodes.push(child = node.children[i] = new Node$1(childs[i]));
            child.parent = node;
            child.depth = node.depth + 1;
          }
        }
      }

      return root.eachBefore(computeHeight);
    }

    function node_copy() {
      return hierarchy(this).eachBefore(copyData);
    }

    function defaultChildren(d) {
      return d.children;
    }

    function copyData(node) {
      node.data = node.data.data;
    }

    function computeHeight(node) {
      var height = 0;
      do node.height = height;
      while ((node = node.parent) && (node.height < ++height));
    }

    function Node$1(data) {
      this.data = data;
      this.depth =
      this.height = 0;
      this.parent = null;
    }

    Node$1.prototype = hierarchy.prototype = {
      constructor: Node$1,
      count: node_count,
      each: node_each,
      eachAfter: node_eachAfter,
      eachBefore: node_eachBefore,
      sum: node_sum,
      sort: node_sort,
      path: node_path,
      ancestors: node_ancestors,
      descendants: node_descendants,
      leaves: node_leaves,
      links: node_links,
      copy: node_copy
    };

    var slice$4 = Array.prototype.slice;

    function shuffle$1(array) {
      var m = array.length,
          t,
          i;

      while (m) {
        i = Math.random() * m-- | 0;
        t = array[m];
        array[m] = array[i];
        array[i] = t;
      }

      return array;
    }

    function enclose(circles) {
      var i = 0, n = (circles = shuffle$1(slice$4.call(circles))).length, B = [], p, e;

      while (i < n) {
        p = circles[i];
        if (e && enclosesWeak(e, p)) ++i;
        else e = encloseBasis(B = extendBasis(B, p)), i = 0;
      }

      return e;
    }

    function extendBasis(B, p) {
      var i, j;

      if (enclosesWeakAll(p, B)) return [p];

      // If we get here then B must have at least one element.
      for (i = 0; i < B.length; ++i) {
        if (enclosesNot(p, B[i])
            && enclosesWeakAll(encloseBasis2(B[i], p), B)) {
          return [B[i], p];
        }
      }

      // If we get here then B must have at least two elements.
      for (i = 0; i < B.length - 1; ++i) {
        for (j = i + 1; j < B.length; ++j) {
          if (enclosesNot(encloseBasis2(B[i], B[j]), p)
              && enclosesNot(encloseBasis2(B[i], p), B[j])
              && enclosesNot(encloseBasis2(B[j], p), B[i])
              && enclosesWeakAll(encloseBasis3(B[i], B[j], p), B)) {
            return [B[i], B[j], p];
          }
        }
      }

      // If we get here then something is very wrong.
      throw new Error;
    }

    function enclosesNot(a, b) {
      var dr = a.r - b.r, dx = b.x - a.x, dy = b.y - a.y;
      return dr < 0 || dr * dr < dx * dx + dy * dy;
    }

    function enclosesWeak(a, b) {
      var dr = a.r - b.r + 1e-6, dx = b.x - a.x, dy = b.y - a.y;
      return dr > 0 && dr * dr > dx * dx + dy * dy;
    }

    function enclosesWeakAll(a, B) {
      for (var i = 0; i < B.length; ++i) {
        if (!enclosesWeak(a, B[i])) {
          return false;
        }
      }
      return true;
    }

    function encloseBasis(B) {
      switch (B.length) {
        case 1: return encloseBasis1(B[0]);
        case 2: return encloseBasis2(B[0], B[1]);
        case 3: return encloseBasis3(B[0], B[1], B[2]);
      }
    }

    function encloseBasis1(a) {
      return {
        x: a.x,
        y: a.y,
        r: a.r
      };
    }

    function encloseBasis2(a, b) {
      var x1 = a.x, y1 = a.y, r1 = a.r,
          x2 = b.x, y2 = b.y, r2 = b.r,
          x21 = x2 - x1, y21 = y2 - y1, r21 = r2 - r1,
          l = Math.sqrt(x21 * x21 + y21 * y21);
      return {
        x: (x1 + x2 + x21 / l * r21) / 2,
        y: (y1 + y2 + y21 / l * r21) / 2,
        r: (l + r1 + r2) / 2
      };
    }

    function encloseBasis3(a, b, c) {
      var x1 = a.x, y1 = a.y, r1 = a.r,
          x2 = b.x, y2 = b.y, r2 = b.r,
          x3 = c.x, y3 = c.y, r3 = c.r,
          a2 = x1 - x2,
          a3 = x1 - x3,
          b2 = y1 - y2,
          b3 = y1 - y3,
          c2 = r2 - r1,
          c3 = r3 - r1,
          d1 = x1 * x1 + y1 * y1 - r1 * r1,
          d2 = d1 - x2 * x2 - y2 * y2 + r2 * r2,
          d3 = d1 - x3 * x3 - y3 * y3 + r3 * r3,
          ab = a3 * b2 - a2 * b3,
          xa = (b2 * d3 - b3 * d2) / (ab * 2) - x1,
          xb = (b3 * c2 - b2 * c3) / ab,
          ya = (a3 * d2 - a2 * d3) / (ab * 2) - y1,
          yb = (a2 * c3 - a3 * c2) / ab,
          A = xb * xb + yb * yb - 1,
          B = 2 * (r1 + xa * xb + ya * yb),
          C = xa * xa + ya * ya - r1 * r1,
          r = -(A ? (B + Math.sqrt(B * B - 4 * A * C)) / (2 * A) : C / B);
      return {
        x: x1 + xa + xb * r,
        y: y1 + ya + yb * r,
        r: r
      };
    }

    function place(b, a, c) {
      var dx = b.x - a.x, x, a2,
          dy = b.y - a.y, y, b2,
          d2 = dx * dx + dy * dy;
      if (d2) {
        a2 = a.r + c.r, a2 *= a2;
        b2 = b.r + c.r, b2 *= b2;
        if (a2 > b2) {
          x = (d2 + b2 - a2) / (2 * d2);
          y = Math.sqrt(Math.max(0, b2 / d2 - x * x));
          c.x = b.x - x * dx - y * dy;
          c.y = b.y - x * dy + y * dx;
        } else {
          x = (d2 + a2 - b2) / (2 * d2);
          y = Math.sqrt(Math.max(0, a2 / d2 - x * x));
          c.x = a.x + x * dx - y * dy;
          c.y = a.y + x * dy + y * dx;
        }
      } else {
        c.x = a.x + c.r;
        c.y = a.y;
      }
    }

    function intersects(a, b) {
      var dr = a.r + b.r - 1e-6, dx = b.x - a.x, dy = b.y - a.y;
      return dr > 0 && dr * dr > dx * dx + dy * dy;
    }

    function score(node) {
      var a = node._,
          b = node.next._,
          ab = a.r + b.r,
          dx = (a.x * b.r + b.x * a.r) / ab,
          dy = (a.y * b.r + b.y * a.r) / ab;
      return dx * dx + dy * dy;
    }

    function Node$2(circle) {
      this._ = circle;
      this.next = null;
      this.previous = null;
    }

    function packEnclose(circles) {
      if (!(n = circles.length)) return 0;

      var a, b, c, n, aa, ca, i, j, k, sj, sk;

      // Place the first circle.
      a = circles[0], a.x = 0, a.y = 0;
      if (!(n > 1)) return a.r;

      // Place the second circle.
      b = circles[1], a.x = -b.r, b.x = a.r, b.y = 0;
      if (!(n > 2)) return a.r + b.r;

      // Place the third circle.
      place(b, a, c = circles[2]);

      // Initialize the front-chain using the first three circles a, b and c.
      a = new Node$2(a), b = new Node$2(b), c = new Node$2(c);
      a.next = c.previous = b;
      b.next = a.previous = c;
      c.next = b.previous = a;

      // Attempt to place each remaining circle…
      pack: for (i = 3; i < n; ++i) {
        place(a._, b._, c = circles[i]), c = new Node$2(c);

        // Find the closest intersecting circle on the front-chain, if any.
        // “Closeness” is determined by linear distance along the front-chain.
        // “Ahead” or “behind” is likewise determined by linear distance.
        j = b.next, k = a.previous, sj = b._.r, sk = a._.r;
        do {
          if (sj <= sk) {
            if (intersects(j._, c._)) {
              b = j, a.next = b, b.previous = a, --i;
              continue pack;
            }
            sj += j._.r, j = j.next;
          } else {
            if (intersects(k._, c._)) {
              a = k, a.next = b, b.previous = a, --i;
              continue pack;
            }
            sk += k._.r, k = k.previous;
          }
        } while (j !== k.next);

        // Success! Insert the new circle c between a and b.
        c.previous = a, c.next = b, a.next = b.previous = b = c;

        // Compute the new closest circle pair to the centroid.
        aa = score(a);
        while ((c = c.next) !== b) {
          if ((ca = score(c)) < aa) {
            a = c, aa = ca;
          }
        }
        b = a.next;
      }

      // Compute the enclosing circle of the front chain.
      a = [b._], c = b; while ((c = c.next) !== b) a.push(c._); c = enclose(a);

      // Translate the circles to put the enclosing circle around the origin.
      for (i = 0; i < n; ++i) a = circles[i], a.x -= c.x, a.y -= c.y;

      return c.r;
    }

    function siblings(circles) {
      packEnclose(circles);
      return circles;
    }

    function optional(f) {
      return f == null ? null : required(f);
    }

    function required(f) {
      if (typeof f !== "function") throw new Error;
      return f;
    }

    function constantZero() {
      return 0;
    }

    function constant$9(x) {
      return function() {
        return x;
      };
    }

    function defaultRadius$1(d) {
      return Math.sqrt(d.value);
    }

    function index$2() {
      var radius = null,
          dx = 1,
          dy = 1,
          padding = constantZero;

      function pack(root) {
        root.x = dx / 2, root.y = dy / 2;
        if (radius) {
          root.eachBefore(radiusLeaf(radius))
              .eachAfter(packChildren(padding, 0.5))
              .eachBefore(translateChild(1));
        } else {
          root.eachBefore(radiusLeaf(defaultRadius$1))
              .eachAfter(packChildren(constantZero, 1))
              .eachAfter(packChildren(padding, root.r / Math.min(dx, dy)))
              .eachBefore(translateChild(Math.min(dx, dy) / (2 * root.r)));
        }
        return root;
      }

      pack.radius = function(x) {
        return arguments.length ? (radius = optional(x), pack) : radius;
      };

      pack.size = function(x) {
        return arguments.length ? (dx = +x[0], dy = +x[1], pack) : [dx, dy];
      };

      pack.padding = function(x) {
        return arguments.length ? (padding = typeof x === "function" ? x : constant$9(+x), pack) : padding;
      };

      return pack;
    }

    function radiusLeaf(radius) {
      return function(node) {
        if (!node.children) {
          node.r = Math.max(0, +radius(node) || 0);
        }
      };
    }

    function packChildren(padding, k) {
      return function(node) {
        if (children = node.children) {
          var children,
              i,
              n = children.length,
              r = padding(node) * k || 0,
              e;

          if (r) for (i = 0; i < n; ++i) children[i].r += r;
          e = packEnclose(children);
          if (r) for (i = 0; i < n; ++i) children[i].r -= r;
          node.r = e + r;
        }
      };
    }

    function translateChild(k) {
      return function(node) {
        var parent = node.parent;
        node.r *= k;
        if (parent) {
          node.x = parent.x + k * node.x;
          node.y = parent.y + k * node.y;
        }
      };
    }

    function roundNode(node) {
      node.x0 = Math.round(node.x0);
      node.y0 = Math.round(node.y0);
      node.x1 = Math.round(node.x1);
      node.y1 = Math.round(node.y1);
    }

    function treemapDice(parent, x0, y0, x1, y1) {
      var nodes = parent.children,
          node,
          i = -1,
          n = nodes.length,
          k = parent.value && (x1 - x0) / parent.value;

      while (++i < n) {
        node = nodes[i], node.y0 = y0, node.y1 = y1;
        node.x0 = x0, node.x1 = x0 += node.value * k;
      }
    }

    function partition() {
      var dx = 1,
          dy = 1,
          padding = 0,
          round = false;

      function partition(root) {
        var n = root.height + 1;
        root.x0 =
        root.y0 = padding;
        root.x1 = dx;
        root.y1 = dy / n;
        root.eachBefore(positionNode(dy, n));
        if (round) root.eachBefore(roundNode);
        return root;
      }

      function positionNode(dy, n) {
        return function(node) {
          if (node.children) {
            treemapDice(node, node.x0, dy * (node.depth + 1) / n, node.x1, dy * (node.depth + 2) / n);
          }
          var x0 = node.x0,
              y0 = node.y0,
              x1 = node.x1 - padding,
              y1 = node.y1 - padding;
          if (x1 < x0) x0 = x1 = (x0 + x1) / 2;
          if (y1 < y0) y0 = y1 = (y0 + y1) / 2;
          node.x0 = x0;
          node.y0 = y0;
          node.x1 = x1;
          node.y1 = y1;
        };
      }

      partition.round = function(x) {
        return arguments.length ? (round = !!x, partition) : round;
      };

      partition.size = function(x) {
        return arguments.length ? (dx = +x[0], dy = +x[1], partition) : [dx, dy];
      };

      partition.padding = function(x) {
        return arguments.length ? (padding = +x, partition) : padding;
      };

      return partition;
    }

    var keyPrefix$1 = "$", // Protect against keys like “__proto__”.
        preroot = {depth: -1},
        ambiguous = {};

    function defaultId(d) {
      return d.id;
    }

    function defaultParentId(d) {
      return d.parentId;
    }

    function stratify() {
      var id = defaultId,
          parentId = defaultParentId;

      function stratify(data) {
        var d,
            i,
            n = data.length,
            root,
            parent,
            node,
            nodes = new Array(n),
            nodeId,
            nodeKey,
            nodeByKey = {};

        for (i = 0; i < n; ++i) {
          d = data[i], node = nodes[i] = new Node$1(d);
          if ((nodeId = id(d, i, data)) != null && (nodeId += "")) {
            nodeKey = keyPrefix$1 + (node.id = nodeId);
            nodeByKey[nodeKey] = nodeKey in nodeByKey ? ambiguous : node;
          }
        }

        for (i = 0; i < n; ++i) {
          node = nodes[i], nodeId = parentId(data[i], i, data);
          if (nodeId == null || !(nodeId += "")) {
            if (root) throw new Error("multiple roots");
            root = node;
          } else {
            parent = nodeByKey[keyPrefix$1 + nodeId];
            if (!parent) throw new Error("missing: " + nodeId);
            if (parent === ambiguous) throw new Error("ambiguous: " + nodeId);
            if (parent.children) parent.children.push(node);
            else parent.children = [node];
            node.parent = parent;
          }
        }

        if (!root) throw new Error("no root");
        root.parent = preroot;
        root.eachBefore(function(node) { node.depth = node.parent.depth + 1; --n; }).eachBefore(computeHeight);
        root.parent = null;
        if (n > 0) throw new Error("cycle");

        return root;
      }

      stratify.id = function(x) {
        return arguments.length ? (id = required(x), stratify) : id;
      };

      stratify.parentId = function(x) {
        return arguments.length ? (parentId = required(x), stratify) : parentId;
      };

      return stratify;
    }

    function defaultSeparation$1(a, b) {
      return a.parent === b.parent ? 1 : 2;
    }

    // function radialSeparation(a, b) {
    //   return (a.parent === b.parent ? 1 : 2) / a.depth;
    // }

    // This function is used to traverse the left contour of a subtree (or
    // subforest). It returns the successor of v on this contour. This successor is
    // either given by the leftmost child of v or by the thread of v. The function
    // returns null if and only if v is on the highest level of its subtree.
    function nextLeft(v) {
      var children = v.children;
      return children ? children[0] : v.t;
    }

    // This function works analogously to nextLeft.
    function nextRight(v) {
      var children = v.children;
      return children ? children[children.length - 1] : v.t;
    }

    // Shifts the current subtree rooted at w+. This is done by increasing
    // prelim(w+) and mod(w+) by shift.
    function moveSubtree(wm, wp, shift) {
      var change = shift / (wp.i - wm.i);
      wp.c -= change;
      wp.s += shift;
      wm.c += change;
      wp.z += shift;
      wp.m += shift;
    }

    // All other shifts, applied to the smaller subtrees between w- and w+, are
    // performed by this function. To prepare the shifts, we have to adjust
    // change(w+), shift(w+), and change(w-).
    function executeShifts(v) {
      var shift = 0,
          change = 0,
          children = v.children,
          i = children.length,
          w;
      while (--i >= 0) {
        w = children[i];
        w.z += shift;
        w.m += shift;
        shift += w.s + (change += w.c);
      }
    }

    // If vi-’s ancestor is a sibling of v, returns vi-’s ancestor. Otherwise,
    // returns the specified (default) ancestor.
    function nextAncestor(vim, v, ancestor) {
      return vim.a.parent === v.parent ? vim.a : ancestor;
    }

    function TreeNode(node, i) {
      this._ = node;
      this.parent = null;
      this.children = null;
      this.A = null; // default ancestor
      this.a = this; // ancestor
      this.z = 0; // prelim
      this.m = 0; // mod
      this.c = 0; // change
      this.s = 0; // shift
      this.t = null; // thread
      this.i = i; // number
    }

    TreeNode.prototype = Object.create(Node$1.prototype);

    function treeRoot(root) {
      var tree = new TreeNode(root, 0),
          node,
          nodes = [tree],
          child,
          children,
          i,
          n;

      while (node = nodes.pop()) {
        if (children = node._.children) {
          node.children = new Array(n = children.length);
          for (i = n - 1; i >= 0; --i) {
            nodes.push(child = node.children[i] = new TreeNode(children[i], i));
            child.parent = node;
          }
        }
      }

      (tree.parent = new TreeNode(null, 0)).children = [tree];
      return tree;
    }

    // Node-link tree diagram using the Reingold-Tilford "tidy" algorithm
    function tree() {
      var separation = defaultSeparation$1,
          dx = 1,
          dy = 1,
          nodeSize = null;

      function tree(root) {
        var t = treeRoot(root);

        // Compute the layout using Buchheim et al.’s algorithm.
        t.eachAfter(firstWalk), t.parent.m = -t.z;
        t.eachBefore(secondWalk);

        // If a fixed node size is specified, scale x and y.
        if (nodeSize) root.eachBefore(sizeNode);

        // If a fixed tree size is specified, scale x and y based on the extent.
        // Compute the left-most, right-most, and depth-most nodes for extents.
        else {
          var left = root,
              right = root,
              bottom = root;
          root.eachBefore(function(node) {
            if (node.x < left.x) left = node;
            if (node.x > right.x) right = node;
            if (node.depth > bottom.depth) bottom = node;
          });
          var s = left === right ? 1 : separation(left, right) / 2,
              tx = s - left.x,
              kx = dx / (right.x + s + tx),
              ky = dy / (bottom.depth || 1);
          root.eachBefore(function(node) {
            node.x = (node.x + tx) * kx;
            node.y = node.depth * ky;
          });
        }

        return root;
      }

      // Computes a preliminary x-coordinate for v. Before that, FIRST WALK is
      // applied recursively to the children of v, as well as the function
      // APPORTION. After spacing out the children by calling EXECUTE SHIFTS, the
      // node v is placed to the midpoint of its outermost children.
      function firstWalk(v) {
        var children = v.children,
            siblings = v.parent.children,
            w = v.i ? siblings[v.i - 1] : null;
        if (children) {
          executeShifts(v);
          var midpoint = (children[0].z + children[children.length - 1].z) / 2;
          if (w) {
            v.z = w.z + separation(v._, w._);
            v.m = v.z - midpoint;
          } else {
            v.z = midpoint;
          }
        } else if (w) {
          v.z = w.z + separation(v._, w._);
        }
        v.parent.A = apportion(v, w, v.parent.A || siblings[0]);
      }

      // Computes all real x-coordinates by summing up the modifiers recursively.
      function secondWalk(v) {
        v._.x = v.z + v.parent.m;
        v.m += v.parent.m;
      }

      // The core of the algorithm. Here, a new subtree is combined with the
      // previous subtrees. Threads are used to traverse the inside and outside
      // contours of the left and right subtree up to the highest common level. The
      // vertices used for the traversals are vi+, vi-, vo-, and vo+, where the
      // superscript o means outside and i means inside, the subscript - means left
      // subtree and + means right subtree. For summing up the modifiers along the
      // contour, we use respective variables si+, si-, so-, and so+. Whenever two
      // nodes of the inside contours conflict, we compute the left one of the
      // greatest uncommon ancestors using the function ANCESTOR and call MOVE
      // SUBTREE to shift the subtree and prepare the shifts of smaller subtrees.
      // Finally, we add a new thread (if necessary).
      function apportion(v, w, ancestor) {
        if (w) {
          var vip = v,
              vop = v,
              vim = w,
              vom = vip.parent.children[0],
              sip = vip.m,
              sop = vop.m,
              sim = vim.m,
              som = vom.m,
              shift;
          while (vim = nextRight(vim), vip = nextLeft(vip), vim && vip) {
            vom = nextLeft(vom);
            vop = nextRight(vop);
            vop.a = v;
            shift = vim.z + sim - vip.z - sip + separation(vim._, vip._);
            if (shift > 0) {
              moveSubtree(nextAncestor(vim, v, ancestor), v, shift);
              sip += shift;
              sop += shift;
            }
            sim += vim.m;
            sip += vip.m;
            som += vom.m;
            sop += vop.m;
          }
          if (vim && !nextRight(vop)) {
            vop.t = vim;
            vop.m += sim - sop;
          }
          if (vip && !nextLeft(vom)) {
            vom.t = vip;
            vom.m += sip - som;
            ancestor = v;
          }
        }
        return ancestor;
      }

      function sizeNode(node) {
        node.x *= dx;
        node.y = node.depth * dy;
      }

      tree.separation = function(x) {
        return arguments.length ? (separation = x, tree) : separation;
      };

      tree.size = function(x) {
        return arguments.length ? (nodeSize = false, dx = +x[0], dy = +x[1], tree) : (nodeSize ? null : [dx, dy]);
      };

      tree.nodeSize = function(x) {
        return arguments.length ? (nodeSize = true, dx = +x[0], dy = +x[1], tree) : (nodeSize ? [dx, dy] : null);
      };

      return tree;
    }

    function treemapSlice(parent, x0, y0, x1, y1) {
      var nodes = parent.children,
          node,
          i = -1,
          n = nodes.length,
          k = parent.value && (y1 - y0) / parent.value;

      while (++i < n) {
        node = nodes[i], node.x0 = x0, node.x1 = x1;
        node.y0 = y0, node.y1 = y0 += node.value * k;
      }
    }

    var phi = (1 + Math.sqrt(5)) / 2;

    function squarifyRatio(ratio, parent, x0, y0, x1, y1) {
      var rows = [],
          nodes = parent.children,
          row,
          nodeValue,
          i0 = 0,
          i1 = 0,
          n = nodes.length,
          dx, dy,
          value = parent.value,
          sumValue,
          minValue,
          maxValue,
          newRatio,
          minRatio,
          alpha,
          beta;

      while (i0 < n) {
        dx = x1 - x0, dy = y1 - y0;

        // Find the next non-empty node.
        do sumValue = nodes[i1++].value; while (!sumValue && i1 < n);
        minValue = maxValue = sumValue;
        alpha = Math.max(dy / dx, dx / dy) / (value * ratio);
        beta = sumValue * sumValue * alpha;
        minRatio = Math.max(maxValue / beta, beta / minValue);

        // Keep adding nodes while the aspect ratio maintains or improves.
        for (; i1 < n; ++i1) {
          sumValue += nodeValue = nodes[i1].value;
          if (nodeValue < minValue) minValue = nodeValue;
          if (nodeValue > maxValue) maxValue = nodeValue;
          beta = sumValue * sumValue * alpha;
          newRatio = Math.max(maxValue / beta, beta / minValue);
          if (newRatio > minRatio) { sumValue -= nodeValue; break; }
          minRatio = newRatio;
        }

        // Position and record the row orientation.
        rows.push(row = {value: sumValue, dice: dx < dy, children: nodes.slice(i0, i1)});
        if (row.dice) treemapDice(row, x0, y0, x1, value ? y0 += dy * sumValue / value : y1);
        else treemapSlice(row, x0, y0, value ? x0 += dx * sumValue / value : x1, y1);
        value -= sumValue, i0 = i1;
      }

      return rows;
    }

    var squarify = (function custom(ratio) {

      function squarify(parent, x0, y0, x1, y1) {
        squarifyRatio(ratio, parent, x0, y0, x1, y1);
      }

      squarify.ratio = function(x) {
        return custom((x = +x) > 1 ? x : 1);
      };

      return squarify;
    })(phi);

    function index$3() {
      var tile = squarify,
          round = false,
          dx = 1,
          dy = 1,
          paddingStack = [0],
          paddingInner = constantZero,
          paddingTop = constantZero,
          paddingRight = constantZero,
          paddingBottom = constantZero,
          paddingLeft = constantZero;

      function treemap(root) {
        root.x0 =
        root.y0 = 0;
        root.x1 = dx;
        root.y1 = dy;
        root.eachBefore(positionNode);
        paddingStack = [0];
        if (round) root.eachBefore(roundNode);
        return root;
      }

      function positionNode(node) {
        var p = paddingStack[node.depth],
            x0 = node.x0 + p,
            y0 = node.y0 + p,
            x1 = node.x1 - p,
            y1 = node.y1 - p;
        if (x1 < x0) x0 = x1 = (x0 + x1) / 2;
        if (y1 < y0) y0 = y1 = (y0 + y1) / 2;
        node.x0 = x0;
        node.y0 = y0;
        node.x1 = x1;
        node.y1 = y1;
        if (node.children) {
          p = paddingStack[node.depth + 1] = paddingInner(node) / 2;
          x0 += paddingLeft(node) - p;
          y0 += paddingTop(node) - p;
          x1 -= paddingRight(node) - p;
          y1 -= paddingBottom(node) - p;
          if (x1 < x0) x0 = x1 = (x0 + x1) / 2;
          if (y1 < y0) y0 = y1 = (y0 + y1) / 2;
          tile(node, x0, y0, x1, y1);
        }
      }

      treemap.round = function(x) {
        return arguments.length ? (round = !!x, treemap) : round;
      };

      treemap.size = function(x) {
        return arguments.length ? (dx = +x[0], dy = +x[1], treemap) : [dx, dy];
      };

      treemap.tile = function(x) {
        return arguments.length ? (tile = required(x), treemap) : tile;
      };

      treemap.padding = function(x) {
        return arguments.length ? treemap.paddingInner(x).paddingOuter(x) : treemap.paddingInner();
      };

      treemap.paddingInner = function(x) {
        return arguments.length ? (paddingInner = typeof x === "function" ? x : constant$9(+x), treemap) : paddingInner;
      };

      treemap.paddingOuter = function(x) {
        return arguments.length ? treemap.paddingTop(x).paddingRight(x).paddingBottom(x).paddingLeft(x) : treemap.paddingTop();
      };

      treemap.paddingTop = function(x) {
        return arguments.length ? (paddingTop = typeof x === "function" ? x : constant$9(+x), treemap) : paddingTop;
      };

      treemap.paddingRight = function(x) {
        return arguments.length ? (paddingRight = typeof x === "function" ? x : constant$9(+x), treemap) : paddingRight;
      };

      treemap.paddingBottom = function(x) {
        return arguments.length ? (paddingBottom = typeof x === "function" ? x : constant$9(+x), treemap) : paddingBottom;
      };

      treemap.paddingLeft = function(x) {
        return arguments.length ? (paddingLeft = typeof x === "function" ? x : constant$9(+x), treemap) : paddingLeft;
      };

      return treemap;
    }

    function binary(parent, x0, y0, x1, y1) {
      var nodes = parent.children,
          i, n = nodes.length,
          sum, sums = new Array(n + 1);

      for (sums[0] = sum = i = 0; i < n; ++i) {
        sums[i + 1] = sum += nodes[i].value;
      }

      partition(0, n, parent.value, x0, y0, x1, y1);

      function partition(i, j, value, x0, y0, x1, y1) {
        if (i >= j - 1) {
          var node = nodes[i];
          node.x0 = x0, node.y0 = y0;
          node.x1 = x1, node.y1 = y1;
          return;
        }

        var valueOffset = sums[i],
            valueTarget = (value / 2) + valueOffset,
            k = i + 1,
            hi = j - 1;

        while (k < hi) {
          var mid = k + hi >>> 1;
          if (sums[mid] < valueTarget) k = mid + 1;
          else hi = mid;
        }

        if ((valueTarget - sums[k - 1]) < (sums[k] - valueTarget) && i + 1 < k) --k;

        var valueLeft = sums[k] - valueOffset,
            valueRight = value - valueLeft;

        if ((x1 - x0) > (y1 - y0)) {
          var xk = (x0 * valueRight + x1 * valueLeft) / value;
          partition(i, k, valueLeft, x0, y0, xk, y1);
          partition(k, j, valueRight, xk, y0, x1, y1);
        } else {
          var yk = (y0 * valueRight + y1 * valueLeft) / value;
          partition(i, k, valueLeft, x0, y0, x1, yk);
          partition(k, j, valueRight, x0, yk, x1, y1);
        }
      }
    }

    function sliceDice(parent, x0, y0, x1, y1) {
      (parent.depth & 1 ? treemapSlice : treemapDice)(parent, x0, y0, x1, y1);
    }

    var resquarify = (function custom(ratio) {

      function resquarify(parent, x0, y0, x1, y1) {
        if ((rows = parent._squarify) && (rows.ratio === ratio)) {
          var rows,
              row,
              nodes,
              i,
              j = -1,
              n,
              m = rows.length,
              value = parent.value;

          while (++j < m) {
            row = rows[j], nodes = row.children;
            for (i = row.value = 0, n = nodes.length; i < n; ++i) row.value += nodes[i].value;
            if (row.dice) treemapDice(row, x0, y0, x1, y0 += (y1 - y0) * row.value / value);
            else treemapSlice(row, x0, y0, x0 += (x1 - x0) * row.value / value, y1);
            value -= row.value;
          }
        } else {
          parent._squarify = rows = squarifyRatio(ratio, parent, x0, y0, x1, y1);
          rows.ratio = ratio;
        }
      }

      resquarify.ratio = function(x) {
        return custom((x = +x) > 1 ? x : 1);
      };

      return resquarify;
    })(phi);

    function area$2(polygon) {
      var i = -1,
          n = polygon.length,
          a,
          b = polygon[n - 1],
          area = 0;

      while (++i < n) {
        a = b;
        b = polygon[i];
        area += a[1] * b[0] - a[0] * b[1];
      }

      return area / 2;
    }

    function centroid$1(polygon) {
      var i = -1,
          n = polygon.length,
          x = 0,
          y = 0,
          a,
          b = polygon[n - 1],
          c,
          k = 0;

      while (++i < n) {
        a = b;
        b = polygon[i];
        k += c = a[0] * b[1] - b[0] * a[1];
        x += (a[0] + b[0]) * c;
        y += (a[1] + b[1]) * c;
      }

      return k *= 3, [x / k, y / k];
    }

    // Returns the 2D cross product of AB and AC vectors, i.e., the z-component of
    // the 3D cross product in a quadrant I Cartesian coordinate system (+x is
    // right, +y is up). Returns a positive value if ABC is counter-clockwise,
    // negative if clockwise, and zero if the points are collinear.
    function cross$1(a, b, c) {
      return (b[0] - a[0]) * (c[1] - a[1]) - (b[1] - a[1]) * (c[0] - a[0]);
    }

    function lexicographicOrder(a, b) {
      return a[0] - b[0] || a[1] - b[1];
    }

    // Computes the upper convex hull per the monotone chain algorithm.
    // Assumes points.length >= 3, is sorted by x, unique in y.
    // Returns an array of indices into points in left-to-right order.
    function computeUpperHullIndexes(points) {
      var n = points.length,
          indexes = [0, 1],
          size = 2;

      for (var i = 2; i < n; ++i) {
        while (size > 1 && cross$1(points[indexes[size - 2]], points[indexes[size - 1]], points[i]) <= 0) --size;
        indexes[size++] = i;
      }

      return indexes.slice(0, size); // remove popped points
    }

    function hull(points) {
      if ((n = points.length) < 3) return null;

      var i,
          n,
          sortedPoints = new Array(n),
          flippedPoints = new Array(n);

      for (i = 0; i < n; ++i) sortedPoints[i] = [+points[i][0], +points[i][1], i];
      sortedPoints.sort(lexicographicOrder);
      for (i = 0; i < n; ++i) flippedPoints[i] = [sortedPoints[i][0], -sortedPoints[i][1]];

      var upperIndexes = computeUpperHullIndexes(sortedPoints),
          lowerIndexes = computeUpperHullIndexes(flippedPoints);

      // Construct the hull polygon, removing possible duplicate endpoints.
      var skipLeft = lowerIndexes[0] === upperIndexes[0],
          skipRight = lowerIndexes[lowerIndexes.length - 1] === upperIndexes[upperIndexes.length - 1],
          hull = [];

      // Add upper hull in right-to-l order.
      // Then add lower hull in left-to-right order.
      for (i = upperIndexes.length - 1; i >= 0; --i) hull.push(points[sortedPoints[upperIndexes[i]][2]]);
      for (i = +skipLeft; i < lowerIndexes.length - skipRight; ++i) hull.push(points[sortedPoints[lowerIndexes[i]][2]]);

      return hull;
    }

    function contains$2(polygon, point) {
      var n = polygon.length,
          p = polygon[n - 1],
          x = point[0], y = point[1],
          x0 = p[0], y0 = p[1],
          x1, y1,
          inside = false;

      for (var i = 0; i < n; ++i) {
        p = polygon[i], x1 = p[0], y1 = p[1];
        if (((y1 > y) !== (y0 > y)) && (x < (x0 - x1) * (y - y1) / (y0 - y1) + x1)) inside = !inside;
        x0 = x1, y0 = y1;
      }

      return inside;
    }

    function length$2(polygon) {
      var i = -1,
          n = polygon.length,
          b = polygon[n - 1],
          xa,
          ya,
          xb = b[0],
          yb = b[1],
          perimeter = 0;

      while (++i < n) {
        xa = xb;
        ya = yb;
        b = polygon[i];
        xb = b[0];
        yb = b[1];
        xa -= xb;
        ya -= yb;
        perimeter += Math.sqrt(xa * xa + ya * ya);
      }

      return perimeter;
    }

    function defaultSource$1() {
      return Math.random();
    }

    var uniform = (function sourceRandomUniform(source) {
      function randomUniform(min, max) {
        min = min == null ? 0 : +min;
        max = max == null ? 1 : +max;
        if (arguments.length === 1) max = min, min = 0;
        else max -= min;
        return function() {
          return source() * max + min;
        };
      }

      randomUniform.source = sourceRandomUniform;

      return randomUniform;
    })(defaultSource$1);

    var normal = (function sourceRandomNormal(source) {
      function randomNormal(mu, sigma) {
        var x, r;
        mu = mu == null ? 0 : +mu;
        sigma = sigma == null ? 1 : +sigma;
        return function() {
          var y;

          // If available, use the second previously-generated uniform random.
          if (x != null) y = x, x = null;

          // Otherwise, generate a new x and y.
          else do {
            x = source() * 2 - 1;
            y = source() * 2 - 1;
            r = x * x + y * y;
          } while (!r || r > 1);

          return mu + sigma * y * Math.sqrt(-2 * Math.log(r) / r);
        };
      }

      randomNormal.source = sourceRandomNormal;

      return randomNormal;
    })(defaultSource$1);

    var logNormal = (function sourceRandomLogNormal(source) {
      function randomLogNormal() {
        var randomNormal = normal.source(source).apply(this, arguments);
        return function() {
          return Math.exp(randomNormal());
        };
      }

      randomLogNormal.source = sourceRandomLogNormal;

      return randomLogNormal;
    })(defaultSource$1);

    var irwinHall = (function sourceRandomIrwinHall(source) {
      function randomIrwinHall(n) {
        return function() {
          for (var sum = 0, i = 0; i < n; ++i) sum += source();
          return sum;
        };
      }

      randomIrwinHall.source = sourceRandomIrwinHall;

      return randomIrwinHall;
    })(defaultSource$1);

    var bates = (function sourceRandomBates(source) {
      function randomBates(n) {
        var randomIrwinHall = irwinHall.source(source)(n);
        return function() {
          return randomIrwinHall() / n;
        };
      }

      randomBates.source = sourceRandomBates;

      return randomBates;
    })(defaultSource$1);

    var exponential$1 = (function sourceRandomExponential(source) {
      function randomExponential(lambda) {
        return function() {
          return -Math.log(1 - source()) / lambda;
        };
      }

      randomExponential.source = sourceRandomExponential;

      return randomExponential;
    })(defaultSource$1);

    function initRange(domain, range) {
      switch (arguments.length) {
        case 0: break;
        case 1: this.range(domain); break;
        default: this.range(range).domain(domain); break;
      }
      return this;
    }

    function initInterpolator(domain, interpolator) {
      switch (arguments.length) {
        case 0: break;
        case 1: this.interpolator(domain); break;
        default: this.interpolator(interpolator).domain(domain); break;
      }
      return this;
    }

    var array$3 = Array.prototype;

    var map$3 = array$3.map;
    var slice$5 = array$3.slice;

    var implicit = {name: "implicit"};

    function ordinal() {
      var index = map$1(),
          domain = [],
          range = [],
          unknown = implicit;

      function scale(d) {
        var key = d + "", i = index.get(key);
        if (!i) {
          if (unknown !== implicit) return unknown;
          index.set(key, i = domain.push(d));
        }
        return range[(i - 1) % range.length];
      }

      scale.domain = function(_) {
        if (!arguments.length) return domain.slice();
        domain = [], index = map$1();
        var i = -1, n = _.length, d, key;
        while (++i < n) if (!index.has(key = (d = _[i]) + "")) index.set(key, domain.push(d));
        return scale;
      };

      scale.range = function(_) {
        return arguments.length ? (range = slice$5.call(_), scale) : range.slice();
      };

      scale.unknown = function(_) {
        return arguments.length ? (unknown = _, scale) : unknown;
      };

      scale.copy = function() {
        return ordinal(domain, range).unknown(unknown);
      };

      initRange.apply(scale, arguments);

      return scale;
    }

    function band() {
      var scale = ordinal().unknown(undefined),
          domain = scale.domain,
          ordinalRange = scale.range,
          range = [0, 1],
          step,
          bandwidth,
          round = false,
          paddingInner = 0,
          paddingOuter = 0,
          align = 0.5;

      delete scale.unknown;

      function rescale() {
        var n = domain().length,
            reverse = range[1] < range[0],
            start = range[reverse - 0],
            stop = range[1 - reverse];
        step = (stop - start) / Math.max(1, n - paddingInner + paddingOuter * 2);
        if (round) step = Math.floor(step);
        start += (stop - start - step * (n - paddingInner)) * align;
        bandwidth = step * (1 - paddingInner);
        if (round) start = Math.round(start), bandwidth = Math.round(bandwidth);
        var values = sequence(n).map(function(i) { return start + step * i; });
        return ordinalRange(reverse ? values.reverse() : values);
      }

      scale.domain = function(_) {
        return arguments.length ? (domain(_), rescale()) : domain();
      };

      scale.range = function(_) {
        return arguments.length ? (range = [+_[0], +_[1]], rescale()) : range.slice();
      };

      scale.rangeRound = function(_) {
        return range = [+_[0], +_[1]], round = true, rescale();
      };

      scale.bandwidth = function() {
        return bandwidth;
      };

      scale.step = function() {
        return step;
      };

      scale.round = function(_) {
        return arguments.length ? (round = !!_, rescale()) : round;
      };

      scale.padding = function(_) {
        return arguments.length ? (paddingInner = Math.min(1, paddingOuter = +_), rescale()) : paddingInner;
      };

      scale.paddingInner = function(_) {
        return arguments.length ? (paddingInner = Math.min(1, _), rescale()) : paddingInner;
      };

      scale.paddingOuter = function(_) {
        return arguments.length ? (paddingOuter = +_, rescale()) : paddingOuter;
      };

      scale.align = function(_) {
        return arguments.length ? (align = Math.max(0, Math.min(1, _)), rescale()) : align;
      };

      scale.copy = function() {
        return band(domain(), range)
            .round(round)
            .paddingInner(paddingInner)
            .paddingOuter(paddingOuter)
            .align(align);
      };

      return initRange.apply(rescale(), arguments);
    }

    function pointish(scale) {
      var copy = scale.copy;

      scale.padding = scale.paddingOuter;
      delete scale.paddingInner;
      delete scale.paddingOuter;

      scale.copy = function() {
        return pointish(copy());
      };

      return scale;
    }

    function point$1() {
      return pointish(band.apply(null, arguments).paddingInner(1));
    }

    function constant$a(x) {
      return function() {
        return x;
      };
    }

    function number$2(x) {
      return +x;
    }

    var unit = [0, 1];

    function identity$6(x) {
      return x;
    }

    function normalize$1(a, b) {
      return (b -= (a = +a))
          ? function(x) { return (x - a) / b; }
          : constant$a(isNaN(b) ? NaN : 0.5);
    }

    function clamper(domain) {
      var a = domain[0], b = domain[domain.length - 1], t;
      if (a > b) t = a, a = b, b = t;
      return function(x) { return Math.max(a, Math.min(b, x)); };
    }

    // normalize(a, b)(x) takes a domain value x in [a,b] and returns the corresponding parameter t in [0,1].
    // interpolate(a, b)(t) takes a parameter t in [0,1] and returns the corresponding range value x in [a,b].
    function bimap(domain, range, interpolate) {
      var d0 = domain[0], d1 = domain[1], r0 = range[0], r1 = range[1];
      if (d1 < d0) d0 = normalize$1(d1, d0), r0 = interpolate(r1, r0);
      else d0 = normalize$1(d0, d1), r0 = interpolate(r0, r1);
      return function(x) { return r0(d0(x)); };
    }

    function polymap(domain, range, interpolate) {
      var j = Math.min(domain.length, range.length) - 1,
          d = new Array(j),
          r = new Array(j),
          i = -1;

      // Reverse descending domains.
      if (domain[j] < domain[0]) {
        domain = domain.slice().reverse();
        range = range.slice().reverse();
      }

      while (++i < j) {
        d[i] = normalize$1(domain[i], domain[i + 1]);
        r[i] = interpolate(range[i], range[i + 1]);
      }

      return function(x) {
        var i = bisectRight(domain, x, 1, j) - 1;
        return r[i](d[i](x));
      };
    }

    function copy(source, target) {
      return target
          .domain(source.domain())
          .range(source.range())
          .interpolate(source.interpolate())
          .clamp(source.clamp())
          .unknown(source.unknown());
    }

    function transformer$1() {
      var domain = unit,
          range = unit,
          interpolate = interpolateValue,
          transform,
          untransform,
          unknown,
          clamp = identity$6,
          piecewise,
          output,
          input;

      function rescale() {
        piecewise = Math.min(domain.length, range.length) > 2 ? polymap : bimap;
        output = input = null;
        return scale;
      }

      function scale(x) {
        return isNaN(x = +x) ? unknown : (output || (output = piecewise(domain.map(transform), range, interpolate)))(transform(clamp(x)));
      }

      scale.invert = function(y) {
        return clamp(untransform((input || (input = piecewise(range, domain.map(transform), interpolateNumber)))(y)));
      };

      scale.domain = function(_) {
        return arguments.length ? (domain = map$3.call(_, number$2), clamp === identity$6 || (clamp = clamper(domain)), rescale()) : domain.slice();
      };

      scale.range = function(_) {
        return arguments.length ? (range = slice$5.call(_), rescale()) : range.slice();
      };

      scale.rangeRound = function(_) {
        return range = slice$5.call(_), interpolate = interpolateRound, rescale();
      };

      scale.clamp = function(_) {
        return arguments.length ? (clamp = _ ? clamper(domain) : identity$6, scale) : clamp !== identity$6;
      };

      scale.interpolate = function(_) {
        return arguments.length ? (interpolate = _, rescale()) : interpolate;
      };

      scale.unknown = function(_) {
        return arguments.length ? (unknown = _, scale) : unknown;
      };

      return function(t, u) {
        transform = t, untransform = u;
        return rescale();
      };
    }

    function continuous(transform, untransform) {
      return transformer$1()(transform, untransform);
    }

    function tickFormat(start, stop, count, specifier) {
      var step = tickStep(start, stop, count),
          precision;
      specifier = formatSpecifier(specifier == null ? ",f" : specifier);
      switch (specifier.type) {
        case "s": {
          var value = Math.max(Math.abs(start), Math.abs(stop));
          if (specifier.precision == null && !isNaN(precision = precisionPrefix(step, value))) specifier.precision = precision;
          return formatPrefix(specifier, value);
        }
        case "":
        case "e":
        case "g":
        case "p":
        case "r": {
          if (specifier.precision == null && !isNaN(precision = precisionRound(step, Math.max(Math.abs(start), Math.abs(stop))))) specifier.precision = precision - (specifier.type === "e");
          break;
        }
        case "f":
        case "%": {
          if (specifier.precision == null && !isNaN(precision = precisionFixed(step))) specifier.precision = precision - (specifier.type === "%") * 2;
          break;
        }
      }
      return format(specifier);
    }

    function linearish(scale) {
      var domain = scale.domain;

      scale.ticks = function(count) {
        var d = domain();
        return ticks(d[0], d[d.length - 1], count == null ? 10 : count);
      };

      scale.tickFormat = function(count, specifier) {
        var d = domain();
        return tickFormat(d[0], d[d.length - 1], count == null ? 10 : count, specifier);
      };

      scale.nice = function(count) {
        if (count == null) count = 10;

        var d = domain(),
            i0 = 0,
            i1 = d.length - 1,
            start = d[i0],
            stop = d[i1],
            step;

        if (stop < start) {
          step = start, start = stop, stop = step;
          step = i0, i0 = i1, i1 = step;
        }

        step = tickIncrement(start, stop, count);

        if (step > 0) {
          start = Math.floor(start / step) * step;
          stop = Math.ceil(stop / step) * step;
          step = tickIncrement(start, stop, count);
        } else if (step < 0) {
          start = Math.ceil(start * step) / step;
          stop = Math.floor(stop * step) / step;
          step = tickIncrement(start, stop, count);
        }

        if (step > 0) {
          d[i0] = Math.floor(start / step) * step;
          d[i1] = Math.ceil(stop / step) * step;
          domain(d);
        } else if (step < 0) {
          d[i0] = Math.ceil(start * step) / step;
          d[i1] = Math.floor(stop * step) / step;
          domain(d);
        }

        return scale;
      };

      return scale;
    }

    function linear$2() {
      var scale = continuous(identity$6, identity$6);

      scale.copy = function() {
        return copy(scale, linear$2());
      };

      initRange.apply(scale, arguments);

      return linearish(scale);
    }

    function identity$7(domain) {
      var unknown;

      function scale(x) {
        return isNaN(x = +x) ? unknown : x;
      }

      scale.invert = scale;

      scale.domain = scale.range = function(_) {
        return arguments.length ? (domain = map$3.call(_, number$2), scale) : domain.slice();
      };

      scale.unknown = function(_) {
        return arguments.length ? (unknown = _, scale) : unknown;
      };

      scale.copy = function() {
        return identity$7(domain).unknown(unknown);
      };

      domain = arguments.length ? map$3.call(domain, number$2) : [0, 1];

      return linearish(scale);
    }

    function nice(domain, interval) {
      domain = domain.slice();

      var i0 = 0,
          i1 = domain.length - 1,
          x0 = domain[i0],
          x1 = domain[i1],
          t;

      if (x1 < x0) {
        t = i0, i0 = i1, i1 = t;
        t = x0, x0 = x1, x1 = t;
      }

      domain[i0] = interval.floor(x0);
      domain[i1] = interval.ceil(x1);
      return domain;
    }

    function transformLog(x) {
      return Math.log(x);
    }

    function transformExp(x) {
      return Math.exp(x);
    }

    function transformLogn(x) {
      return -Math.log(-x);
    }

    function transformExpn(x) {
      return -Math.exp(-x);
    }

    function pow10(x) {
      return isFinite(x) ? +("1e" + x) : x < 0 ? 0 : x;
    }

    function powp(base) {
      return base === 10 ? pow10
          : base === Math.E ? Math.exp
          : function(x) { return Math.pow(base, x); };
    }

    function logp(base) {
      return base === Math.E ? Math.log
          : base === 10 && Math.log10
          || base === 2 && Math.log2
          || (base = Math.log(base), function(x) { return Math.log(x) / base; });
    }

    function reflect(f) {
      return function(x) {
        return -f(-x);
      };
    }

    function loggish(transform) {
      var scale = transform(transformLog, transformExp),
          domain = scale.domain,
          base = 10,
          logs,
          pows;

      function rescale() {
        logs = logp(base), pows = powp(base);
        if (domain()[0] < 0) {
          logs = reflect(logs), pows = reflect(pows);
          transform(transformLogn, transformExpn);
        } else {
          transform(transformLog, transformExp);
        }
        return scale;
      }

      scale.base = function(_) {
        return arguments.length ? (base = +_, rescale()) : base;
      };

      scale.domain = function(_) {
        return arguments.length ? (domain(_), rescale()) : domain();
      };

      scale.ticks = function(count) {
        var d = domain(),
            u = d[0],
            v = d[d.length - 1],
            r;

        if (r = v < u) i = u, u = v, v = i;

        var i = logs(u),
            j = logs(v),
            p,
            k,
            t,
            n = count == null ? 10 : +count,
            z = [];

        if (!(base % 1) && j - i < n) {
          i = Math.round(i) - 1, j = Math.round(j) + 1;
          if (u > 0) for (; i < j; ++i) {
            for (k = 1, p = pows(i); k < base; ++k) {
              t = p * k;
              if (t < u) continue;
              if (t > v) break;
              z.push(t);
            }
          } else for (; i < j; ++i) {
            for (k = base - 1, p = pows(i); k >= 1; --k) {
              t = p * k;
              if (t < u) continue;
              if (t > v) break;
              z.push(t);
            }
          }
        } else {
          z = ticks(i, j, Math.min(j - i, n)).map(pows);
        }

        return r ? z.reverse() : z;
      };

      scale.tickFormat = function(count, specifier) {
        if (specifier == null) specifier = base === 10 ? ".0e" : ",";
        if (typeof specifier !== "function") specifier = format(specifier);
        if (count === Infinity) return specifier;
        if (count == null) count = 10;
        var k = Math.max(1, base * count / scale.ticks().length); // TODO fast estimate?
        return function(d) {
          var i = d / pows(Math.round(logs(d)));
          if (i * base < base - 0.5) i *= base;
          return i <= k ? specifier(d) : "";
        };
      };

      scale.nice = function() {
        return domain(nice(domain(), {
          floor: function(x) { return pows(Math.floor(logs(x))); },
          ceil: function(x) { return pows(Math.ceil(logs(x))); }
        }));
      };

      return scale;
    }

    function log$1() {
      var scale = loggish(transformer$1()).domain([1, 10]);

      scale.copy = function() {
        return copy(scale, log$1()).base(scale.base());
      };

      initRange.apply(scale, arguments);

      return scale;
    }

    function transformSymlog(c) {
      return function(x) {
        return Math.sign(x) * Math.log1p(Math.abs(x / c));
      };
    }

    function transformSymexp(c) {
      return function(x) {
        return Math.sign(x) * Math.expm1(Math.abs(x)) * c;
      };
    }

    function symlogish(transform) {
      var c = 1, scale = transform(transformSymlog(c), transformSymexp(c));

      scale.constant = function(_) {
        return arguments.length ? transform(transformSymlog(c = +_), transformSymexp(c)) : c;
      };

      return linearish(scale);
    }

    function symlog() {
      var scale = symlogish(transformer$1());

      scale.copy = function() {
        return copy(scale, symlog()).constant(scale.constant());
      };

      return initRange.apply(scale, arguments);
    }

    function transformPow(exponent) {
      return function(x) {
        return x < 0 ? -Math.pow(-x, exponent) : Math.pow(x, exponent);
      };
    }

    function transformSqrt(x) {
      return x < 0 ? -Math.sqrt(-x) : Math.sqrt(x);
    }

    function transformSquare(x) {
      return x < 0 ? -x * x : x * x;
    }

    function powish(transform) {
      var scale = transform(identity$6, identity$6),
          exponent = 1;

      function rescale() {
        return exponent === 1 ? transform(identity$6, identity$6)
            : exponent === 0.5 ? transform(transformSqrt, transformSquare)
            : transform(transformPow(exponent), transformPow(1 / exponent));
      }

      scale.exponent = function(_) {
        return arguments.length ? (exponent = +_, rescale()) : exponent;
      };

      return linearish(scale);
    }

    function pow$1() {
      var scale = powish(transformer$1());

      scale.copy = function() {
        return copy(scale, pow$1()).exponent(scale.exponent());
      };

      initRange.apply(scale, arguments);

      return scale;
    }

    function sqrt$1() {
      return pow$1.apply(null, arguments).exponent(0.5);
    }

    function quantile() {
      var domain = [],
          range = [],
          thresholds = [],
          unknown;

      function rescale() {
        var i = 0, n = Math.max(1, range.length);
        thresholds = new Array(n - 1);
        while (++i < n) thresholds[i - 1] = threshold(domain, i / n);
        return scale;
      }

      function scale(x) {
        return isNaN(x = +x) ? unknown : range[bisectRight(thresholds, x)];
      }

      scale.invertExtent = function(y) {
        var i = range.indexOf(y);
        return i < 0 ? [NaN, NaN] : [
          i > 0 ? thresholds[i - 1] : domain[0],
          i < thresholds.length ? thresholds[i] : domain[domain.length - 1]
        ];
      };

      scale.domain = function(_) {
        if (!arguments.length) return domain.slice();
        domain = [];
        for (var i = 0, n = _.length, d; i < n; ++i) if (d = _[i], d != null && !isNaN(d = +d)) domain.push(d);
        domain.sort(ascending);
        return rescale();
      };

      scale.range = function(_) {
        return arguments.length ? (range = slice$5.call(_), rescale()) : range.slice();
      };

      scale.unknown = function(_) {
        return arguments.length ? (unknown = _, scale) : unknown;
      };

      scale.quantiles = function() {
        return thresholds.slice();
      };

      scale.copy = function() {
        return quantile()
            .domain(domain)
            .range(range)
            .unknown(unknown);
      };

      return initRange.apply(scale, arguments);
    }

    function quantize$1() {
      var x0 = 0,
          x1 = 1,
          n = 1,
          domain = [0.5],
          range = [0, 1],
          unknown;

      function scale(x) {
        return x <= x ? range[bisectRight(domain, x, 0, n)] : unknown;
      }

      function rescale() {
        var i = -1;
        domain = new Array(n);
        while (++i < n) domain[i] = ((i + 1) * x1 - (i - n) * x0) / (n + 1);
        return scale;
      }

      scale.domain = function(_) {
        return arguments.length ? (x0 = +_[0], x1 = +_[1], rescale()) : [x0, x1];
      };

      scale.range = function(_) {
        return arguments.length ? (n = (range = slice$5.call(_)).length - 1, rescale()) : range.slice();
      };

      scale.invertExtent = function(y) {
        var i = range.indexOf(y);
        return i < 0 ? [NaN, NaN]
            : i < 1 ? [x0, domain[0]]
            : i >= n ? [domain[n - 1], x1]
            : [domain[i - 1], domain[i]];
      };

      scale.unknown = function(_) {
        return arguments.length ? (unknown = _, scale) : scale;
      };

      scale.thresholds = function() {
        return domain.slice();
      };

      scale.copy = function() {
        return quantize$1()
            .domain([x0, x1])
            .range(range)
            .unknown(unknown);
      };

      return initRange.apply(linearish(scale), arguments);
    }

    function threshold$1() {
      var domain = [0.5],
          range = [0, 1],
          unknown,
          n = 1;

      function scale(x) {
        return x <= x ? range[bisectRight(domain, x, 0, n)] : unknown;
      }

      scale.domain = function(_) {
        return arguments.length ? (domain = slice$5.call(_), n = Math.min(domain.length, range.length - 1), scale) : domain.slice();
      };

      scale.range = function(_) {
        return arguments.length ? (range = slice$5.call(_), n = Math.min(domain.length, range.length - 1), scale) : range.slice();
      };

      scale.invertExtent = function(y) {
        var i = range.indexOf(y);
        return [domain[i - 1], domain[i]];
      };

      scale.unknown = function(_) {
        return arguments.length ? (unknown = _, scale) : unknown;
      };

      scale.copy = function() {
        return threshold$1()
            .domain(domain)
            .range(range)
            .unknown(unknown);
      };

      return initRange.apply(scale, arguments);
    }

    var t0$1 = new Date,
        t1$1 = new Date;

    function newInterval(floori, offseti, count, field) {

      function interval(date) {
        return floori(date = arguments.length === 0 ? new Date : new Date(+date)), date;
      }

      interval.floor = function(date) {
        return floori(date = new Date(+date)), date;
      };

      interval.ceil = function(date) {
        return floori(date = new Date(date - 1)), offseti(date, 1), floori(date), date;
      };

      interval.round = function(date) {
        var d0 = interval(date),
            d1 = interval.ceil(date);
        return date - d0 < d1 - date ? d0 : d1;
      };

      interval.offset = function(date, step) {
        return offseti(date = new Date(+date), step == null ? 1 : Math.floor(step)), date;
      };

      interval.range = function(start, stop, step) {
        var range = [], previous;
        start = interval.ceil(start);
        step = step == null ? 1 : Math.floor(step);
        if (!(start < stop) || !(step > 0)) return range; // also handles Invalid Date
        do range.push(previous = new Date(+start)), offseti(start, step), floori(start);
        while (previous < start && start < stop);
        return range;
      };

      interval.filter = function(test) {
        return newInterval(function(date) {
          if (date >= date) while (floori(date), !test(date)) date.setTime(date - 1);
        }, function(date, step) {
          if (date >= date) {
            if (step < 0) while (++step <= 0) {
              while (offseti(date, -1), !test(date)) {} // eslint-disable-line no-empty
            } else while (--step >= 0) {
              while (offseti(date, +1), !test(date)) {} // eslint-disable-line no-empty
            }
          }
        });
      };

      if (count) {
        interval.count = function(start, end) {
          t0$1.setTime(+start), t1$1.setTime(+end);
          floori(t0$1), floori(t1$1);
          return Math.floor(count(t0$1, t1$1));
        };

        interval.every = function(step) {
          step = Math.floor(step);
          return !isFinite(step) || !(step > 0) ? null
              : !(step > 1) ? interval
              : interval.filter(field
                  ? function(d) { return field(d) % step === 0; }
                  : function(d) { return interval.count(0, d) % step === 0; });
        };
      }

      return interval;
    }

    var millisecond = newInterval(function() {
      // noop
    }, function(date, step) {
      date.setTime(+date + step);
    }, function(start, end) {
      return end - start;
    });

    // An optimized implementation for this simple case.
    millisecond.every = function(k) {
      k = Math.floor(k);
      if (!isFinite(k) || !(k > 0)) return null;
      if (!(k > 1)) return millisecond;
      return newInterval(function(date) {
        date.setTime(Math.floor(date / k) * k);
      }, function(date, step) {
        date.setTime(+date + step * k);
      }, function(start, end) {
        return (end - start) / k;
      });
    };
    var milliseconds = millisecond.range;

    var durationSecond = 1e3;
    var durationMinute = 6e4;
    var durationHour = 36e5;
    var durationDay = 864e5;
    var durationWeek = 6048e5;

    var second = newInterval(function(date) {
      date.setTime(date - date.getMilliseconds());
    }, function(date, step) {
      date.setTime(+date + step * durationSecond);
    }, function(start, end) {
      return (end - start) / durationSecond;
    }, function(date) {
      return date.getUTCSeconds();
    });
    var seconds = second.range;

    var minute = newInterval(function(date) {
      date.setTime(date - date.getMilliseconds() - date.getSeconds() * durationSecond);
    }, function(date, step) {
      date.setTime(+date + step * durationMinute);
    }, function(start, end) {
      return (end - start) / durationMinute;
    }, function(date) {
      return date.getMinutes();
    });
    var minutes = minute.range;

    var hour = newInterval(function(date) {
      date.setTime(date - date.getMilliseconds() - date.getSeconds() * durationSecond - date.getMinutes() * durationMinute);
    }, function(date, step) {
      date.setTime(+date + step * durationHour);
    }, function(start, end) {
      return (end - start) / durationHour;
    }, function(date) {
      return date.getHours();
    });
    var hours = hour.range;

    var day = newInterval(function(date) {
      date.setHours(0, 0, 0, 0);
    }, function(date, step) {
      date.setDate(date.getDate() + step);
    }, function(start, end) {
      return (end - start - (end.getTimezoneOffset() - start.getTimezoneOffset()) * durationMinute) / durationDay;
    }, function(date) {
      return date.getDate() - 1;
    });
    var days = day.range;

    function weekday(i) {
      return newInterval(function(date) {
        date.setDate(date.getDate() - (date.getDay() + 7 - i) % 7);
        date.setHours(0, 0, 0, 0);
      }, function(date, step) {
        date.setDate(date.getDate() + step * 7);
      }, function(start, end) {
        return (end - start - (end.getTimezoneOffset() - start.getTimezoneOffset()) * durationMinute) / durationWeek;
      });
    }

    var sunday = weekday(0);
    var monday = weekday(1);
    var tuesday = weekday(2);
    var wednesday = weekday(3);
    var thursday = weekday(4);
    var friday = weekday(5);
    var saturday = weekday(6);

    var sundays = sunday.range;
    var mondays = monday.range;
    var tuesdays = tuesday.range;
    var wednesdays = wednesday.range;
    var thursdays = thursday.range;
    var fridays = friday.range;
    var saturdays = saturday.range;

    var month = newInterval(function(date) {
      date.setDate(1);
      date.setHours(0, 0, 0, 0);
    }, function(date, step) {
      date.setMonth(date.getMonth() + step);
    }, function(start, end) {
      return end.getMonth() - start.getMonth() + (end.getFullYear() - start.getFullYear()) * 12;
    }, function(date) {
      return date.getMonth();
    });
    var months = month.range;

    var year = newInterval(function(date) {
      date.setMonth(0, 1);
      date.setHours(0, 0, 0, 0);
    }, function(date, step) {
      date.setFullYear(date.getFullYear() + step);
    }, function(start, end) {
      return end.getFullYear() - start.getFullYear();
    }, function(date) {
      return date.getFullYear();
    });

    // An optimized implementation for this simple case.
    year.every = function(k) {
      return !isFinite(k = Math.floor(k)) || !(k > 0) ? null : newInterval(function(date) {
        date.setFullYear(Math.floor(date.getFullYear() / k) * k);
        date.setMonth(0, 1);
        date.setHours(0, 0, 0, 0);
      }, function(date, step) {
        date.setFullYear(date.getFullYear() + step * k);
      });
    };
    var years = year.range;

    var utcMinute = newInterval(function(date) {
      date.setUTCSeconds(0, 0);
    }, function(date, step) {
      date.setTime(+date + step * durationMinute);
    }, function(start, end) {
      return (end - start) / durationMinute;
    }, function(date) {
      return date.getUTCMinutes();
    });
    var utcMinutes = utcMinute.range;

    var utcHour = newInterval(function(date) {
      date.setUTCMinutes(0, 0, 0);
    }, function(date, step) {
      date.setTime(+date + step * durationHour);
    }, function(start, end) {
      return (end - start) / durationHour;
    }, function(date) {
      return date.getUTCHours();
    });
    var utcHours = utcHour.range;

    var utcDay = newInterval(function(date) {
      date.setUTCHours(0, 0, 0, 0);
    }, function(date, step) {
      date.setUTCDate(date.getUTCDate() + step);
    }, function(start, end) {
      return (end - start) / durationDay;
    }, function(date) {
      return date.getUTCDate() - 1;
    });
    var utcDays = utcDay.range;

    function utcWeekday(i) {
      return newInterval(function(date) {
        date.setUTCDate(date.getUTCDate() - (date.getUTCDay() + 7 - i) % 7);
        date.setUTCHours(0, 0, 0, 0);
      }, function(date, step) {
        date.setUTCDate(date.getUTCDate() + step * 7);
      }, function(start, end) {
        return (end - start) / durationWeek;
      });
    }

    var utcSunday = utcWeekday(0);
    var utcMonday = utcWeekday(1);
    var utcTuesday = utcWeekday(2);
    var utcWednesday = utcWeekday(3);
    var utcThursday = utcWeekday(4);
    var utcFriday = utcWeekday(5);
    var utcSaturday = utcWeekday(6);

    var utcSundays = utcSunday.range;
    var utcMondays = utcMonday.range;
    var utcTuesdays = utcTuesday.range;
    var utcWednesdays = utcWednesday.range;
    var utcThursdays = utcThursday.range;
    var utcFridays = utcFriday.range;
    var utcSaturdays = utcSaturday.range;

    var utcMonth = newInterval(function(date) {
      date.setUTCDate(1);
      date.setUTCHours(0, 0, 0, 0);
    }, function(date, step) {
      date.setUTCMonth(date.getUTCMonth() + step);
    }, function(start, end) {
      return end.getUTCMonth() - start.getUTCMonth() + (end.getUTCFullYear() - start.getUTCFullYear()) * 12;
    }, function(date) {
      return date.getUTCMonth();
    });
    var utcMonths = utcMonth.range;

    var utcYear = newInterval(function(date) {
      date.setUTCMonth(0, 1);
      date.setUTCHours(0, 0, 0, 0);
    }, function(date, step) {
      date.setUTCFullYear(date.getUTCFullYear() + step);
    }, function(start, end) {
      return end.getUTCFullYear() - start.getUTCFullYear();
    }, function(date) {
      return date.getUTCFullYear();
    });

    // An optimized implementation for this simple case.
    utcYear.every = function(k) {
      return !isFinite(k = Math.floor(k)) || !(k > 0) ? null : newInterval(function(date) {
        date.setUTCFullYear(Math.floor(date.getUTCFullYear() / k) * k);
        date.setUTCMonth(0, 1);
        date.setUTCHours(0, 0, 0, 0);
      }, function(date, step) {
        date.setUTCFullYear(date.getUTCFullYear() + step * k);
      });
    };
    var utcYears = utcYear.range;

    function localDate(d) {
      if (0 <= d.y && d.y < 100) {
        var date = new Date(-1, d.m, d.d, d.H, d.M, d.S, d.L);
        date.setFullYear(d.y);
        return date;
      }
      return new Date(d.y, d.m, d.d, d.H, d.M, d.S, d.L);
    }

    function utcDate(d) {
      if (0 <= d.y && d.y < 100) {
        var date = new Date(Date.UTC(-1, d.m, d.d, d.H, d.M, d.S, d.L));
        date.setUTCFullYear(d.y);
        return date;
      }
      return new Date(Date.UTC(d.y, d.m, d.d, d.H, d.M, d.S, d.L));
    }

    function newDate(y, m, d) {
      return {y: y, m: m, d: d, H: 0, M: 0, S: 0, L: 0};
    }

    function formatLocale$1(locale) {
      var locale_dateTime = locale.dateTime,
          locale_date = locale.date,
          locale_time = locale.time,
          locale_periods = locale.periods,
          locale_weekdays = locale.days,
          locale_shortWeekdays = locale.shortDays,
          locale_months = locale.months,
          locale_shortMonths = locale.shortMonths;

      var periodRe = formatRe(locale_periods),
          periodLookup = formatLookup(locale_periods),
          weekdayRe = formatRe(locale_weekdays),
          weekdayLookup = formatLookup(locale_weekdays),
          shortWeekdayRe = formatRe(locale_shortWeekdays),
          shortWeekdayLookup = formatLookup(locale_shortWeekdays),
          monthRe = formatRe(locale_months),
          monthLookup = formatLookup(locale_months),
          shortMonthRe = formatRe(locale_shortMonths),
          shortMonthLookup = formatLookup(locale_shortMonths);

      var formats = {
        "a": formatShortWeekday,
        "A": formatWeekday,
        "b": formatShortMonth,
        "B": formatMonth,
        "c": null,
        "d": formatDayOfMonth,
        "e": formatDayOfMonth,
        "f": formatMicroseconds,
        "H": formatHour24,
        "I": formatHour12,
        "j": formatDayOfYear,
        "L": formatMilliseconds,
        "m": formatMonthNumber,
        "M": formatMinutes,
        "p": formatPeriod,
        "q": formatQuarter,
        "Q": formatUnixTimestamp,
        "s": formatUnixTimestampSeconds,
        "S": formatSeconds,
        "u": formatWeekdayNumberMonday,
        "U": formatWeekNumberSunday,
        "V": formatWeekNumberISO,
        "w": formatWeekdayNumberSunday,
        "W": formatWeekNumberMonday,
        "x": null,
        "X": null,
        "y": formatYear$1,
        "Y": formatFullYear,
        "Z": formatZone,
        "%": formatLiteralPercent
      };

      var utcFormats = {
        "a": formatUTCShortWeekday,
        "A": formatUTCWeekday,
        "b": formatUTCShortMonth,
        "B": formatUTCMonth,
        "c": null,
        "d": formatUTCDayOfMonth,
        "e": formatUTCDayOfMonth,
        "f": formatUTCMicroseconds,
        "H": formatUTCHour24,
        "I": formatUTCHour12,
        "j": formatUTCDayOfYear,
        "L": formatUTCMilliseconds,
        "m": formatUTCMonthNumber,
        "M": formatUTCMinutes,
        "p": formatUTCPeriod,
        "q": formatUTCQuarter,
        "Q": formatUnixTimestamp,
        "s": formatUnixTimestampSeconds,
        "S": formatUTCSeconds,
        "u": formatUTCWeekdayNumberMonday,
        "U": formatUTCWeekNumberSunday,
        "V": formatUTCWeekNumberISO,
        "w": formatUTCWeekdayNumberSunday,
        "W": formatUTCWeekNumberMonday,
        "x": null,
        "X": null,
        "y": formatUTCYear,
        "Y": formatUTCFullYear,
        "Z": formatUTCZone,
        "%": formatLiteralPercent
      };

      var parses = {
        "a": parseShortWeekday,
        "A": parseWeekday,
        "b": parseShortMonth,
        "B": parseMonth,
        "c": parseLocaleDateTime,
        "d": parseDayOfMonth,
        "e": parseDayOfMonth,
        "f": parseMicroseconds,
        "H": parseHour24,
        "I": parseHour24,
        "j": parseDayOfYear,
        "L": parseMilliseconds,
        "m": parseMonthNumber,
        "M": parseMinutes,
        "p": parsePeriod,
        "q": parseQuarter,
        "Q": parseUnixTimestamp,
        "s": parseUnixTimestampSeconds,
        "S": parseSeconds,
        "u": parseWeekdayNumberMonday,
        "U": parseWeekNumberSunday,
        "V": parseWeekNumberISO,
        "w": parseWeekdayNumberSunday,
        "W": parseWeekNumberMonday,
        "x": parseLocaleDate,
        "X": parseLocaleTime,
        "y": parseYear,
        "Y": parseFullYear,
        "Z": parseZone,
        "%": parseLiteralPercent
      };

      // These recursive directive definitions must be deferred.
      formats.x = newFormat(locale_date, formats);
      formats.X = newFormat(locale_time, formats);
      formats.c = newFormat(locale_dateTime, formats);
      utcFormats.x = newFormat(locale_date, utcFormats);
      utcFormats.X = newFormat(locale_time, utcFormats);
      utcFormats.c = newFormat(locale_dateTime, utcFormats);

      function newFormat(specifier, formats) {
        return function(date) {
          var string = [],
              i = -1,
              j = 0,
              n = specifier.length,
              c,
              pad,
              format;

          if (!(date instanceof Date)) date = new Date(+date);

          while (++i < n) {
            if (specifier.charCodeAt(i) === 37) {
              string.push(specifier.slice(j, i));
              if ((pad = pads[c = specifier.charAt(++i)]) != null) c = specifier.charAt(++i);
              else pad = c === "e" ? " " : "0";
              if (format = formats[c]) c = format(date, pad);
              string.push(c);
              j = i + 1;
            }
          }

          string.push(specifier.slice(j, i));
          return string.join("");
        };
      }

      function newParse(specifier, Z) {
        return function(string) {
          var d = newDate(1900, undefined, 1),
              i = parseSpecifier(d, specifier, string += "", 0),
              week, day$1;
          if (i != string.length) return null;

          // If a UNIX timestamp is specified, return it.
          if ("Q" in d) return new Date(d.Q);
          if ("s" in d) return new Date(d.s * 1000 + ("L" in d ? d.L : 0));

          // If this is utcParse, never use the local timezone.
          if (Z && !("Z" in d)) d.Z = 0;

          // The am-pm flag is 0 for AM, and 1 for PM.
          if ("p" in d) d.H = d.H % 12 + d.p * 12;

          // If the month was not specified, inherit from the quarter.
          if (d.m === undefined) d.m = "q" in d ? d.q : 0;

          // Convert day-of-week and week-of-year to day-of-year.
          if ("V" in d) {
            if (d.V < 1 || d.V > 53) return null;
            if (!("w" in d)) d.w = 1;
            if ("Z" in d) {
              week = utcDate(newDate(d.y, 0, 1)), day$1 = week.getUTCDay();
              week = day$1 > 4 || day$1 === 0 ? utcMonday.ceil(week) : utcMonday(week);
              week = utcDay.offset(week, (d.V - 1) * 7);
              d.y = week.getUTCFullYear();
              d.m = week.getUTCMonth();
              d.d = week.getUTCDate() + (d.w + 6) % 7;
            } else {
              week = localDate(newDate(d.y, 0, 1)), day$1 = week.getDay();
              week = day$1 > 4 || day$1 === 0 ? monday.ceil(week) : monday(week);
              week = day.offset(week, (d.V - 1) * 7);
              d.y = week.getFullYear();
              d.m = week.getMonth();
              d.d = week.getDate() + (d.w + 6) % 7;
            }
          } else if ("W" in d || "U" in d) {
            if (!("w" in d)) d.w = "u" in d ? d.u % 7 : "W" in d ? 1 : 0;
            day$1 = "Z" in d ? utcDate(newDate(d.y, 0, 1)).getUTCDay() : localDate(newDate(d.y, 0, 1)).getDay();
            d.m = 0;
            d.d = "W" in d ? (d.w + 6) % 7 + d.W * 7 - (day$1 + 5) % 7 : d.w + d.U * 7 - (day$1 + 6) % 7;
          }

          // If a time zone is specified, all fields are interpreted as UTC and then
          // offset according to the specified time zone.
          if ("Z" in d) {
            d.H += d.Z / 100 | 0;
            d.M += d.Z % 100;
            return utcDate(d);
          }

          // Otherwise, all fields are in local time.
          return localDate(d);
        };
      }

      function parseSpecifier(d, specifier, string, j) {
        var i = 0,
            n = specifier.length,
            m = string.length,
            c,
            parse;

        while (i < n) {
          if (j >= m) return -1;
          c = specifier.charCodeAt(i++);
          if (c === 37) {
            c = specifier.charAt(i++);
            parse = parses[c in pads ? specifier.charAt(i++) : c];
            if (!parse || ((j = parse(d, string, j)) < 0)) return -1;
          } else if (c != string.charCodeAt(j++)) {
            return -1;
          }
        }

        return j;
      }

      function parsePeriod(d, string, i) {
        var n = periodRe.exec(string.slice(i));
        return n ? (d.p = periodLookup[n[0].toLowerCase()], i + n[0].length) : -1;
      }

      function parseShortWeekday(d, string, i) {
        var n = shortWeekdayRe.exec(string.slice(i));
        return n ? (d.w = shortWeekdayLookup[n[0].toLowerCase()], i + n[0].length) : -1;
      }

      function parseWeekday(d, string, i) {
        var n = weekdayRe.exec(string.slice(i));
        return n ? (d.w = weekdayLookup[n[0].toLowerCase()], i + n[0].length) : -1;
      }

      function parseShortMonth(d, string, i) {
        var n = shortMonthRe.exec(string.slice(i));
        return n ? (d.m = shortMonthLookup[n[0].toLowerCase()], i + n[0].length) : -1;
      }

      function parseMonth(d, string, i) {
        var n = monthRe.exec(string.slice(i));
        return n ? (d.m = monthLookup[n[0].toLowerCase()], i + n[0].length) : -1;
      }

      function parseLocaleDateTime(d, string, i) {
        return parseSpecifier(d, locale_dateTime, string, i);
      }

      function parseLocaleDate(d, string, i) {
        return parseSpecifier(d, locale_date, string, i);
      }

      function parseLocaleTime(d, string, i) {
        return parseSpecifier(d, locale_time, string, i);
      }

      function formatShortWeekday(d) {
        return locale_shortWeekdays[d.getDay()];
      }

      function formatWeekday(d) {
        return locale_weekdays[d.getDay()];
      }

      function formatShortMonth(d) {
        return locale_shortMonths[d.getMonth()];
      }

      function formatMonth(d) {
        return locale_months[d.getMonth()];
      }

      function formatPeriod(d) {
        return locale_periods[+(d.getHours() >= 12)];
      }

      function formatQuarter(d) {
        return 1 + ~~(d.getMonth() / 3);
      }

      function formatUTCShortWeekday(d) {
        return locale_shortWeekdays[d.getUTCDay()];
      }

      function formatUTCWeekday(d) {
        return locale_weekdays[d.getUTCDay()];
      }

      function formatUTCShortMonth(d) {
        return locale_shortMonths[d.getUTCMonth()];
      }

      function formatUTCMonth(d) {
        return locale_months[d.getUTCMonth()];
      }

      function formatUTCPeriod(d) {
        return locale_periods[+(d.getUTCHours() >= 12)];
      }

      function formatUTCQuarter(d) {
        return 1 + ~~(d.getUTCMonth() / 3);
      }

      return {
        format: function(specifier) {
          var f = newFormat(specifier += "", formats);
          f.toString = function() { return specifier; };
          return f;
        },
        parse: function(specifier) {
          var p = newParse(specifier += "", false);
          p.toString = function() { return specifier; };
          return p;
        },
        utcFormat: function(specifier) {
          var f = newFormat(specifier += "", utcFormats);
          f.toString = function() { return specifier; };
          return f;
        },
        utcParse: function(specifier) {
          var p = newParse(specifier += "", true);
          p.toString = function() { return specifier; };
          return p;
        }
      };
    }

    var pads = {"-": "", "_": " ", "0": "0"},
        numberRe = /^\s*\d+/, // note: ignores next directive
        percentRe = /^%/,
        requoteRe = /[\\^$*+?|[\]().{}]/g;

    function pad$1(value, fill, width) {
      var sign = value < 0 ? "-" : "",
          string = (sign ? -value : value) + "",
          length = string.length;
      return sign + (length < width ? new Array(width - length + 1).join(fill) + string : string);
    }

    function requote(s) {
      return s.replace(requoteRe, "\\$&");
    }

    function formatRe(names) {
      return new RegExp("^(?:" + names.map(requote).join("|") + ")", "i");
    }

    function formatLookup(names) {
      var map = {}, i = -1, n = names.length;
      while (++i < n) map[names[i].toLowerCase()] = i;
      return map;
    }

    function parseWeekdayNumberSunday(d, string, i) {
      var n = numberRe.exec(string.slice(i, i + 1));
      return n ? (d.w = +n[0], i + n[0].length) : -1;
    }

    function parseWeekdayNumberMonday(d, string, i) {
      var n = numberRe.exec(string.slice(i, i + 1));
      return n ? (d.u = +n[0], i + n[0].length) : -1;
    }

    function parseWeekNumberSunday(d, string, i) {
      var n = numberRe.exec(string.slice(i, i + 2));
      return n ? (d.U = +n[0], i + n[0].length) : -1;
    }

    function parseWeekNumberISO(d, string, i) {
      var n = numberRe.exec(string.slice(i, i + 2));
      return n ? (d.V = +n[0], i + n[0].length) : -1;
    }

    function parseWeekNumberMonday(d, string, i) {
      var n = numberRe.exec(string.slice(i, i + 2));
      return n ? (d.W = +n[0], i + n[0].length) : -1;
    }

    function parseFullYear(d, string, i) {
      var n = numberRe.exec(string.slice(i, i + 4));
      return n ? (d.y = +n[0], i + n[0].length) : -1;
    }

    function parseYear(d, string, i) {
      var n = numberRe.exec(string.slice(i, i + 2));
      return n ? (d.y = +n[0] + (+n[0] > 68 ? 1900 : 2000), i + n[0].length) : -1;
    }

    function parseZone(d, string, i) {
      var n = /^(Z)|([+-]\d\d)(?::?(\d\d))?/.exec(string.slice(i, i + 6));
      return n ? (d.Z = n[1] ? 0 : -(n[2] + (n[3] || "00")), i + n[0].length) : -1;
    }

    function parseQuarter(d, string, i) {
      var n = numberRe.exec(string.slice(i, i + 1));
      return n ? (d.q = n[0] * 3 - 3, i + n[0].length) : -1;
    }

    function parseMonthNumber(d, string, i) {
      var n = numberRe.exec(string.slice(i, i + 2));
      return n ? (d.m = n[0] - 1, i + n[0].length) : -1;
    }

    function parseDayOfMonth(d, string, i) {
      var n = numberRe.exec(string.slice(i, i + 2));
      return n ? (d.d = +n[0], i + n[0].length) : -1;
    }

    function parseDayOfYear(d, string, i) {
      var n = numberRe.exec(string.slice(i, i + 3));
      return n ? (d.m = 0, d.d = +n[0], i + n[0].length) : -1;
    }

    function parseHour24(d, string, i) {
      var n = numberRe.exec(string.slice(i, i + 2));
      return n ? (d.H = +n[0], i + n[0].length) : -1;
    }

    function parseMinutes(d, string, i) {
      var n = numberRe.exec(string.slice(i, i + 2));
      return n ? (d.M = +n[0], i + n[0].length) : -1;
    }

    function parseSeconds(d, string, i) {
      var n = numberRe.exec(string.slice(i, i + 2));
      return n ? (d.S = +n[0], i + n[0].length) : -1;
    }

    function parseMilliseconds(d, string, i) {
      var n = numberRe.exec(string.slice(i, i + 3));
      return n ? (d.L = +n[0], i + n[0].length) : -1;
    }

    function parseMicroseconds(d, string, i) {
      var n = numberRe.exec(string.slice(i, i + 6));
      return n ? (d.L = Math.floor(n[0] / 1000), i + n[0].length) : -1;
    }

    function parseLiteralPercent(d, string, i) {
      var n = percentRe.exec(string.slice(i, i + 1));
      return n ? i + n[0].length : -1;
    }

    function parseUnixTimestamp(d, string, i) {
      var n = numberRe.exec(string.slice(i));
      return n ? (d.Q = +n[0], i + n[0].length) : -1;
    }

    function parseUnixTimestampSeconds(d, string, i) {
      var n = numberRe.exec(string.slice(i));
      return n ? (d.s = +n[0], i + n[0].length) : -1;
    }

    function formatDayOfMonth(d, p) {
      return pad$1(d.getDate(), p, 2);
    }

    function formatHour24(d, p) {
      return pad$1(d.getHours(), p, 2);
    }

    function formatHour12(d, p) {
      return pad$1(d.getHours() % 12 || 12, p, 2);
    }

    function formatDayOfYear(d, p) {
      return pad$1(1 + day.count(year(d), d), p, 3);
    }

    function formatMilliseconds(d, p) {
      return pad$1(d.getMilliseconds(), p, 3);
    }

    function formatMicroseconds(d, p) {
      return formatMilliseconds(d, p) + "000";
    }

    function formatMonthNumber(d, p) {
      return pad$1(d.getMonth() + 1, p, 2);
    }

    function formatMinutes(d, p) {
      return pad$1(d.getMinutes(), p, 2);
    }

    function formatSeconds(d, p) {
      return pad$1(d.getSeconds(), p, 2);
    }

    function formatWeekdayNumberMonday(d) {
      var day = d.getDay();
      return day === 0 ? 7 : day;
    }

    function formatWeekNumberSunday(d, p) {
      return pad$1(sunday.count(year(d) - 1, d), p, 2);
    }

    function formatWeekNumberISO(d, p) {
      var day = d.getDay();
      d = (day >= 4 || day === 0) ? thursday(d) : thursday.ceil(d);
      return pad$1(thursday.count(year(d), d) + (year(d).getDay() === 4), p, 2);
    }

    function formatWeekdayNumberSunday(d) {
      return d.getDay();
    }

    function formatWeekNumberMonday(d, p) {
      return pad$1(monday.count(year(d) - 1, d), p, 2);
    }

    function formatYear$1(d, p) {
      return pad$1(d.getFullYear() % 100, p, 2);
    }

    function formatFullYear(d, p) {
      return pad$1(d.getFullYear() % 10000, p, 4);
    }

    function formatZone(d) {
      var z = d.getTimezoneOffset();
      return (z > 0 ? "-" : (z *= -1, "+"))
          + pad$1(z / 60 | 0, "0", 2)
          + pad$1(z % 60, "0", 2);
    }

    function formatUTCDayOfMonth(d, p) {
      return pad$1(d.getUTCDate(), p, 2);
    }

    function formatUTCHour24(d, p) {
      return pad$1(d.getUTCHours(), p, 2);
    }

    function formatUTCHour12(d, p) {
      return pad$1(d.getUTCHours() % 12 || 12, p, 2);
    }

    function formatUTCDayOfYear(d, p) {
      return pad$1(1 + utcDay.count(utcYear(d), d), p, 3);
    }

    function formatUTCMilliseconds(d, p) {
      return pad$1(d.getUTCMilliseconds(), p, 3);
    }

    function formatUTCMicroseconds(d, p) {
      return formatUTCMilliseconds(d, p) + "000";
    }

    function formatUTCMonthNumber(d, p) {
      return pad$1(d.getUTCMonth() + 1, p, 2);
    }

    function formatUTCMinutes(d, p) {
      return pad$1(d.getUTCMinutes(), p, 2);
    }

    function formatUTCSeconds(d, p) {
      return pad$1(d.getUTCSeconds(), p, 2);
    }

    function formatUTCWeekdayNumberMonday(d) {
      var dow = d.getUTCDay();
      return dow === 0 ? 7 : dow;
    }

    function formatUTCWeekNumberSunday(d, p) {
      return pad$1(utcSunday.count(utcYear(d) - 1, d), p, 2);
    }

    function formatUTCWeekNumberISO(d, p) {
      var day = d.getUTCDay();
      d = (day >= 4 || day === 0) ? utcThursday(d) : utcThursday.ceil(d);
      return pad$1(utcThursday.count(utcYear(d), d) + (utcYear(d).getUTCDay() === 4), p, 2);
    }

    function formatUTCWeekdayNumberSunday(d) {
      return d.getUTCDay();
    }

    function formatUTCWeekNumberMonday(d, p) {
      return pad$1(utcMonday.count(utcYear(d) - 1, d), p, 2);
    }

    function formatUTCYear(d, p) {
      return pad$1(d.getUTCFullYear() % 100, p, 2);
    }

    function formatUTCFullYear(d, p) {
      return pad$1(d.getUTCFullYear() % 10000, p, 4);
    }

    function formatUTCZone() {
      return "+0000";
    }

    function formatLiteralPercent() {
      return "%";
    }

    function formatUnixTimestamp(d) {
      return +d;
    }

    function formatUnixTimestampSeconds(d) {
      return Math.floor(+d / 1000);
    }

    var locale$1;
    var timeFormat;
    var timeParse;
    var utcFormat;
    var utcParse;

    defaultLocale$1({
      dateTime: "%x, %X",
      date: "%-m/%-d/%Y",
      time: "%-I:%M:%S %p",
      periods: ["AM", "PM"],
      days: ["Sunday", "Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday"],
      shortDays: ["Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"],
      months: ["January", "February", "March", "April", "May", "June", "July", "August", "September", "October", "November", "December"],
      shortMonths: ["Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"]
    });

    function defaultLocale$1(definition) {
      locale$1 = formatLocale$1(definition);
      timeFormat = locale$1.format;
      timeParse = locale$1.parse;
      utcFormat = locale$1.utcFormat;
      utcParse = locale$1.utcParse;
      return locale$1;
    }

    var isoSpecifier = "%Y-%m-%dT%H:%M:%S.%LZ";

    function formatIsoNative(date) {
      return date.toISOString();
    }

    var formatIso = Date.prototype.toISOString
        ? formatIsoNative
        : utcFormat(isoSpecifier);

    function parseIsoNative(string) {
      var date = new Date(string);
      return isNaN(date) ? null : date;
    }

    var parseIso = +new Date("2000-01-01T00:00:00.000Z")
        ? parseIsoNative
        : utcParse(isoSpecifier);

    var durationSecond$1 = 1000,
        durationMinute$1 = durationSecond$1 * 60,
        durationHour$1 = durationMinute$1 * 60,
        durationDay$1 = durationHour$1 * 24,
        durationWeek$1 = durationDay$1 * 7,
        durationMonth = durationDay$1 * 30,
        durationYear = durationDay$1 * 365;

    function date$1(t) {
      return new Date(t);
    }

    function number$3(t) {
      return t instanceof Date ? +t : +new Date(+t);
    }

    function calendar(year, month, week, day, hour, minute, second, millisecond, format) {
      var scale = continuous(identity$6, identity$6),
          invert = scale.invert,
          domain = scale.domain;

      var formatMillisecond = format(".%L"),
          formatSecond = format(":%S"),
          formatMinute = format("%I:%M"),
          formatHour = format("%I %p"),
          formatDay = format("%a %d"),
          formatWeek = format("%b %d"),
          formatMonth = format("%B"),
          formatYear = format("%Y");

      var tickIntervals = [
        [second,  1,      durationSecond$1],
        [second,  5,  5 * durationSecond$1],
        [second, 15, 15 * durationSecond$1],
        [second, 30, 30 * durationSecond$1],
        [minute,  1,      durationMinute$1],
        [minute,  5,  5 * durationMinute$1],
        [minute, 15, 15 * durationMinute$1],
        [minute, 30, 30 * durationMinute$1],
        [  hour,  1,      durationHour$1  ],
        [  hour,  3,  3 * durationHour$1  ],
        [  hour,  6,  6 * durationHour$1  ],
        [  hour, 12, 12 * durationHour$1  ],
        [   day,  1,      durationDay$1   ],
        [   day,  2,  2 * durationDay$1   ],
        [  week,  1,      durationWeek$1  ],
        [ month,  1,      durationMonth ],
        [ month,  3,  3 * durationMonth ],
        [  year,  1,      durationYear  ]
      ];

      function tickFormat(date) {
        return (second(date) < date ? formatMillisecond
            : minute(date) < date ? formatSecond
            : hour(date) < date ? formatMinute
            : day(date) < date ? formatHour
            : month(date) < date ? (week(date) < date ? formatDay : formatWeek)
            : year(date) < date ? formatMonth
            : formatYear)(date);
      }

      function tickInterval(interval, start, stop, step) {
        if (interval == null) interval = 10;

        // If a desired tick count is specified, pick a reasonable tick interval
        // based on the extent of the domain and a rough estimate of tick size.
        // Otherwise, assume interval is already a time interval and use it.
        if (typeof interval === "number") {
          var target = Math.abs(stop - start) / interval,
              i = bisector(function(i) { return i[2]; }).right(tickIntervals, target);
          if (i === tickIntervals.length) {
            step = tickStep(start / durationYear, stop / durationYear, interval);
            interval = year;
          } else if (i) {
            i = tickIntervals[target / tickIntervals[i - 1][2] < tickIntervals[i][2] / target ? i - 1 : i];
            step = i[1];
            interval = i[0];
          } else {
            step = Math.max(tickStep(start, stop, interval), 1);
            interval = millisecond;
          }
        }

        return step == null ? interval : interval.every(step);
      }

      scale.invert = function(y) {
        return new Date(invert(y));
      };

      scale.domain = function(_) {
        return arguments.length ? domain(map$3.call(_, number$3)) : domain().map(date$1);
      };

      scale.ticks = function(interval, step) {
        var d = domain(),
            t0 = d[0],
            t1 = d[d.length - 1],
            r = t1 < t0,
            t;
        if (r) t = t0, t0 = t1, t1 = t;
        t = tickInterval(interval, t0, t1, step);
        t = t ? t.range(t0, t1 + 1) : []; // inclusive stop
        return r ? t.reverse() : t;
      };

      scale.tickFormat = function(count, specifier) {
        return specifier == null ? tickFormat : format(specifier);
      };

      scale.nice = function(interval, step) {
        var d = domain();
        return (interval = tickInterval(interval, d[0], d[d.length - 1], step))
            ? domain(nice(d, interval))
            : scale;
      };

      scale.copy = function() {
        return copy(scale, calendar(year, month, week, day, hour, minute, second, millisecond, format));
      };

      return scale;
    }

    function time() {
      return initRange.apply(calendar(year, month, sunday, day, hour, minute, second, millisecond, timeFormat).domain([new Date(2000, 0, 1), new Date(2000, 0, 2)]), arguments);
    }

    function utcTime() {
      return initRange.apply(calendar(utcYear, utcMonth, utcSunday, utcDay, utcHour, utcMinute, second, millisecond, utcFormat).domain([Date.UTC(2000, 0, 1), Date.UTC(2000, 0, 2)]), arguments);
    }

    function transformer$2() {
      var x0 = 0,
          x1 = 1,
          t0,
          t1,
          k10,
          transform,
          interpolator = identity$6,
          clamp = false,
          unknown;

      function scale(x) {
        return isNaN(x = +x) ? unknown : interpolator(k10 === 0 ? 0.5 : (x = (transform(x) - t0) * k10, clamp ? Math.max(0, Math.min(1, x)) : x));
      }

      scale.domain = function(_) {
        return arguments.length ? (t0 = transform(x0 = +_[0]), t1 = transform(x1 = +_[1]), k10 = t0 === t1 ? 0 : 1 / (t1 - t0), scale) : [x0, x1];
      };

      scale.clamp = function(_) {
        return arguments.length ? (clamp = !!_, scale) : clamp;
      };

      scale.interpolator = function(_) {
        return arguments.length ? (interpolator = _, scale) : interpolator;
      };

      scale.unknown = function(_) {
        return arguments.length ? (unknown = _, scale) : unknown;
      };

      return function(t) {
        transform = t, t0 = t(x0), t1 = t(x1), k10 = t0 === t1 ? 0 : 1 / (t1 - t0);
        return scale;
      };
    }

    function copy$1(source, target) {
      return target
          .domain(source.domain())
          .interpolator(source.interpolator())
          .clamp(source.clamp())
          .unknown(source.unknown());
    }

    function sequential() {
      var scale = linearish(transformer$2()(identity$6));

      scale.copy = function() {
        return copy$1(scale, sequential());
      };

      return initInterpolator.apply(scale, arguments);
    }

    function sequentialLog() {
      var scale = loggish(transformer$2()).domain([1, 10]);

      scale.copy = function() {
        return copy$1(scale, sequentialLog()).base(scale.base());
      };

      return initInterpolator.apply(scale, arguments);
    }

    function sequentialSymlog() {
      var scale = symlogish(transformer$2());

      scale.copy = function() {
        return copy$1(scale, sequentialSymlog()).constant(scale.constant());
      };

      return initInterpolator.apply(scale, arguments);
    }

    function sequentialPow() {
      var scale = powish(transformer$2());

      scale.copy = function() {
        return copy$1(scale, sequentialPow()).exponent(scale.exponent());
      };

      return initInterpolator.apply(scale, arguments);
    }

    function sequentialSqrt() {
      return sequentialPow.apply(null, arguments).exponent(0.5);
    }

    function sequentialQuantile() {
      var domain = [],
          interpolator = identity$6;

      function scale(x) {
        if (!isNaN(x = +x)) return interpolator((bisectRight(domain, x) - 1) / (domain.length - 1));
      }

      scale.domain = function(_) {
        if (!arguments.length) return domain.slice();
        domain = [];
        for (var i = 0, n = _.length, d; i < n; ++i) if (d = _[i], d != null && !isNaN(d = +d)) domain.push(d);
        domain.sort(ascending);
        return scale;
      };

      scale.interpolator = function(_) {
        return arguments.length ? (interpolator = _, scale) : interpolator;
      };

      scale.copy = function() {
        return sequentialQuantile(interpolator).domain(domain);
      };

      return initInterpolator.apply(scale, arguments);
    }

    function transformer$3() {
      var x0 = 0,
          x1 = 0.5,
          x2 = 1,
          t0,
          t1,
          t2,
          k10,
          k21,
          interpolator = identity$6,
          transform,
          clamp = false,
          unknown;

      function scale(x) {
        return isNaN(x = +x) ? unknown : (x = 0.5 + ((x = +transform(x)) - t1) * (x < t1 ? k10 : k21), interpolator(clamp ? Math.max(0, Math.min(1, x)) : x));
      }

      scale.domain = function(_) {
        return arguments.length ? (t0 = transform(x0 = +_[0]), t1 = transform(x1 = +_[1]), t2 = transform(x2 = +_[2]), k10 = t0 === t1 ? 0 : 0.5 / (t1 - t0), k21 = t1 === t2 ? 0 : 0.5 / (t2 - t1), scale) : [x0, x1, x2];
      };

      scale.clamp = function(_) {
        return arguments.length ? (clamp = !!_, scale) : clamp;
      };

      scale.interpolator = function(_) {
        return arguments.length ? (interpolator = _, scale) : interpolator;
      };

      scale.unknown = function(_) {
        return arguments.length ? (unknown = _, scale) : unknown;
      };

      return function(t) {
        transform = t, t0 = t(x0), t1 = t(x1), t2 = t(x2), k10 = t0 === t1 ? 0 : 0.5 / (t1 - t0), k21 = t1 === t2 ? 0 : 0.5 / (t2 - t1);
        return scale;
      };
    }

    function diverging() {
      var scale = linearish(transformer$3()(identity$6));

      scale.copy = function() {
        return copy$1(scale, diverging());
      };

      return initInterpolator.apply(scale, arguments);
    }

    function divergingLog() {
      var scale = loggish(transformer$3()).domain([0.1, 1, 10]);

      scale.copy = function() {
        return copy$1(scale, divergingLog()).base(scale.base());
      };

      return initInterpolator.apply(scale, arguments);
    }

    function divergingSymlog() {
      var scale = symlogish(transformer$3());

      scale.copy = function() {
        return copy$1(scale, divergingSymlog()).constant(scale.constant());
      };

      return initInterpolator.apply(scale, arguments);
    }

    function divergingPow() {
      var scale = powish(transformer$3());

      scale.copy = function() {
        return copy$1(scale, divergingPow()).exponent(scale.exponent());
      };

      return initInterpolator.apply(scale, arguments);
    }

    function divergingSqrt() {
      return divergingPow.apply(null, arguments).exponent(0.5);
    }

    function colors(specifier) {
      var n = specifier.length / 6 | 0, colors = new Array(n), i = 0;
      while (i < n) colors[i] = "#" + specifier.slice(i * 6, ++i * 6);
      return colors;
    }

    var category10 = colors("1f77b4ff7f0e2ca02cd627289467bd8c564be377c27f7f7fbcbd2217becf");

    var Accent = colors("7fc97fbeaed4fdc086ffff99386cb0f0027fbf5b17666666");

    var Dark2 = colors("1b9e77d95f027570b3e7298a66a61ee6ab02a6761d666666");

    var Paired = colors("a6cee31f78b4b2df8a33a02cfb9a99e31a1cfdbf6fff7f00cab2d66a3d9affff99b15928");

    var Pastel1 = colors("fbb4aeb3cde3ccebc5decbe4fed9a6ffffcce5d8bdfddaecf2f2f2");

    var Pastel2 = colors("b3e2cdfdcdaccbd5e8f4cae4e6f5c9fff2aef1e2cccccccc");

    var Set1 = colors("e41a1c377eb84daf4a984ea3ff7f00ffff33a65628f781bf999999");

    var Set2 = colors("66c2a5fc8d628da0cbe78ac3a6d854ffd92fe5c494b3b3b3");

    var Set3 = colors("8dd3c7ffffb3bebadafb807280b1d3fdb462b3de69fccde5d9d9d9bc80bdccebc5ffed6f");

    var Tableau10 = colors("4e79a7f28e2ce1575976b7b259a14fedc949af7aa1ff9da79c755fbab0ab");

    function ramp(scheme) {
      return rgbBasis(scheme[scheme.length - 1]);
    }

    var scheme = new Array(3).concat(
      "d8b365f5f5f55ab4ac",
      "a6611adfc27d80cdc1018571",
      "a6611adfc27df5f5f580cdc1018571",
      "8c510ad8b365f6e8c3c7eae55ab4ac01665e",
      "8c510ad8b365f6e8c3f5f5f5c7eae55ab4ac01665e",
      "8c510abf812ddfc27df6e8c3c7eae580cdc135978f01665e",
      "8c510abf812ddfc27df6e8c3f5f5f5c7eae580cdc135978f01665e",
      "5430058c510abf812ddfc27df6e8c3c7eae580cdc135978f01665e003c30",
      "5430058c510abf812ddfc27df6e8c3f5f5f5c7eae580cdc135978f01665e003c30"
    ).map(colors);

    var BrBG = ramp(scheme);

    var scheme$1 = new Array(3).concat(
      "af8dc3f7f7f77fbf7b",
      "7b3294c2a5cfa6dba0008837",
      "7b3294c2a5cff7f7f7a6dba0008837",
      "762a83af8dc3e7d4e8d9f0d37fbf7b1b7837",
      "762a83af8dc3e7d4e8f7f7f7d9f0d37fbf7b1b7837",
      "762a839970abc2a5cfe7d4e8d9f0d3a6dba05aae611b7837",
      "762a839970abc2a5cfe7d4e8f7f7f7d9f0d3a6dba05aae611b7837",
      "40004b762a839970abc2a5cfe7d4e8d9f0d3a6dba05aae611b783700441b",
      "40004b762a839970abc2a5cfe7d4e8f7f7f7d9f0d3a6dba05aae611b783700441b"
    ).map(colors);

    var PRGn = ramp(scheme$1);

    var scheme$2 = new Array(3).concat(
      "e9a3c9f7f7f7a1d76a",
      "d01c8bf1b6dab8e1864dac26",
      "d01c8bf1b6daf7f7f7b8e1864dac26",
      "c51b7de9a3c9fde0efe6f5d0a1d76a4d9221",
      "c51b7de9a3c9fde0eff7f7f7e6f5d0a1d76a4d9221",
      "c51b7dde77aef1b6dafde0efe6f5d0b8e1867fbc414d9221",
      "c51b7dde77aef1b6dafde0eff7f7f7e6f5d0b8e1867fbc414d9221",
      "8e0152c51b7dde77aef1b6dafde0efe6f5d0b8e1867fbc414d9221276419",
      "8e0152c51b7dde77aef1b6dafde0eff7f7f7e6f5d0b8e1867fbc414d9221276419"
    ).map(colors);

    var PiYG = ramp(scheme$2);

    var scheme$3 = new Array(3).concat(
      "998ec3f7f7f7f1a340",
      "5e3c99b2abd2fdb863e66101",
      "5e3c99b2abd2f7f7f7fdb863e66101",
      "542788998ec3d8daebfee0b6f1a340b35806",
      "542788998ec3d8daebf7f7f7fee0b6f1a340b35806",
      "5427888073acb2abd2d8daebfee0b6fdb863e08214b35806",
      "5427888073acb2abd2d8daebf7f7f7fee0b6fdb863e08214b35806",
      "2d004b5427888073acb2abd2d8daebfee0b6fdb863e08214b358067f3b08",
      "2d004b5427888073acb2abd2d8daebf7f7f7fee0b6fdb863e08214b358067f3b08"
    ).map(colors);

    var PuOr = ramp(scheme$3);

    var scheme$4 = new Array(3).concat(
      "ef8a62f7f7f767a9cf",
      "ca0020f4a58292c5de0571b0",
      "ca0020f4a582f7f7f792c5de0571b0",
      "b2182bef8a62fddbc7d1e5f067a9cf2166ac",
      "b2182bef8a62fddbc7f7f7f7d1e5f067a9cf2166ac",
      "b2182bd6604df4a582fddbc7d1e5f092c5de4393c32166ac",
      "b2182bd6604df4a582fddbc7f7f7f7d1e5f092c5de4393c32166ac",
      "67001fb2182bd6604df4a582fddbc7d1e5f092c5de4393c32166ac053061",
      "67001fb2182bd6604df4a582fddbc7f7f7f7d1e5f092c5de4393c32166ac053061"
    ).map(colors);

    var RdBu = ramp(scheme$4);

    var scheme$5 = new Array(3).concat(
      "ef8a62ffffff999999",
      "ca0020f4a582bababa404040",
      "ca0020f4a582ffffffbababa404040",
      "b2182bef8a62fddbc7e0e0e09999994d4d4d",
      "b2182bef8a62fddbc7ffffffe0e0e09999994d4d4d",
      "b2182bd6604df4a582fddbc7e0e0e0bababa8787874d4d4d",
      "b2182bd6604df4a582fddbc7ffffffe0e0e0bababa8787874d4d4d",
      "67001fb2182bd6604df4a582fddbc7e0e0e0bababa8787874d4d4d1a1a1a",
      "67001fb2182bd6604df4a582fddbc7ffffffe0e0e0bababa8787874d4d4d1a1a1a"
    ).map(colors);

    var RdGy = ramp(scheme$5);

    var scheme$6 = new Array(3).concat(
      "fc8d59ffffbf91bfdb",
      "d7191cfdae61abd9e92c7bb6",
      "d7191cfdae61ffffbfabd9e92c7bb6",
      "d73027fc8d59fee090e0f3f891bfdb4575b4",
      "d73027fc8d59fee090ffffbfe0f3f891bfdb4575b4",
      "d73027f46d43fdae61fee090e0f3f8abd9e974add14575b4",
      "d73027f46d43fdae61fee090ffffbfe0f3f8abd9e974add14575b4",
      "a50026d73027f46d43fdae61fee090e0f3f8abd9e974add14575b4313695",
      "a50026d73027f46d43fdae61fee090ffffbfe0f3f8abd9e974add14575b4313695"
    ).map(colors);

    var RdYlBu = ramp(scheme$6);

    var scheme$7 = new Array(3).concat(
      "fc8d59ffffbf91cf60",
      "d7191cfdae61a6d96a1a9641",
      "d7191cfdae61ffffbfa6d96a1a9641",
      "d73027fc8d59fee08bd9ef8b91cf601a9850",
      "d73027fc8d59fee08bffffbfd9ef8b91cf601a9850",
      "d73027f46d43fdae61fee08bd9ef8ba6d96a66bd631a9850",
      "d73027f46d43fdae61fee08bffffbfd9ef8ba6d96a66bd631a9850",
      "a50026d73027f46d43fdae61fee08bd9ef8ba6d96a66bd631a9850006837",
      "a50026d73027f46d43fdae61fee08bffffbfd9ef8ba6d96a66bd631a9850006837"
    ).map(colors);

    var RdYlGn = ramp(scheme$7);

    var scheme$8 = new Array(3).concat(
      "fc8d59ffffbf99d594",
      "d7191cfdae61abdda42b83ba",
      "d7191cfdae61ffffbfabdda42b83ba",
      "d53e4ffc8d59fee08be6f59899d5943288bd",
      "d53e4ffc8d59fee08bffffbfe6f59899d5943288bd",
      "d53e4ff46d43fdae61fee08be6f598abdda466c2a53288bd",
      "d53e4ff46d43fdae61fee08bffffbfe6f598abdda466c2a53288bd",
      "9e0142d53e4ff46d43fdae61fee08be6f598abdda466c2a53288bd5e4fa2",
      "9e0142d53e4ff46d43fdae61fee08bffffbfe6f598abdda466c2a53288bd5e4fa2"
    ).map(colors);

    var Spectral = ramp(scheme$8);

    var scheme$9 = new Array(3).concat(
      "e5f5f999d8c92ca25f",
      "edf8fbb2e2e266c2a4238b45",
      "edf8fbb2e2e266c2a42ca25f006d2c",
      "edf8fbccece699d8c966c2a42ca25f006d2c",
      "edf8fbccece699d8c966c2a441ae76238b45005824",
      "f7fcfde5f5f9ccece699d8c966c2a441ae76238b45005824",
      "f7fcfde5f5f9ccece699d8c966c2a441ae76238b45006d2c00441b"
    ).map(colors);

    var BuGn = ramp(scheme$9);

    var scheme$a = new Array(3).concat(
      "e0ecf49ebcda8856a7",
      "edf8fbb3cde38c96c688419d",
      "edf8fbb3cde38c96c68856a7810f7c",
      "edf8fbbfd3e69ebcda8c96c68856a7810f7c",
      "edf8fbbfd3e69ebcda8c96c68c6bb188419d6e016b",
      "f7fcfde0ecf4bfd3e69ebcda8c96c68c6bb188419d6e016b",
      "f7fcfde0ecf4bfd3e69ebcda8c96c68c6bb188419d810f7c4d004b"
    ).map(colors);

    var BuPu = ramp(scheme$a);

    var scheme$b = new Array(3).concat(
      "e0f3dba8ddb543a2ca",
      "f0f9e8bae4bc7bccc42b8cbe",
      "f0f9e8bae4bc7bccc443a2ca0868ac",
      "f0f9e8ccebc5a8ddb57bccc443a2ca0868ac",
      "f0f9e8ccebc5a8ddb57bccc44eb3d32b8cbe08589e",
      "f7fcf0e0f3dbccebc5a8ddb57bccc44eb3d32b8cbe08589e",
      "f7fcf0e0f3dbccebc5a8ddb57bccc44eb3d32b8cbe0868ac084081"
    ).map(colors);

    var GnBu = ramp(scheme$b);

    var scheme$c = new Array(3).concat(
      "fee8c8fdbb84e34a33",
      "fef0d9fdcc8afc8d59d7301f",
      "fef0d9fdcc8afc8d59e34a33b30000",
      "fef0d9fdd49efdbb84fc8d59e34a33b30000",
      "fef0d9fdd49efdbb84fc8d59ef6548d7301f990000",
      "fff7ecfee8c8fdd49efdbb84fc8d59ef6548d7301f990000",
      "fff7ecfee8c8fdd49efdbb84fc8d59ef6548d7301fb300007f0000"
    ).map(colors);

    var OrRd = ramp(scheme$c);

    var scheme$d = new Array(3).concat(
      "ece2f0a6bddb1c9099",
      "f6eff7bdc9e167a9cf02818a",
      "f6eff7bdc9e167a9cf1c9099016c59",
      "f6eff7d0d1e6a6bddb67a9cf1c9099016c59",
      "f6eff7d0d1e6a6bddb67a9cf3690c002818a016450",
      "fff7fbece2f0d0d1e6a6bddb67a9cf3690c002818a016450",
      "fff7fbece2f0d0d1e6a6bddb67a9cf3690c002818a016c59014636"
    ).map(colors);

    var PuBuGn = ramp(scheme$d);

    var scheme$e = new Array(3).concat(
      "ece7f2a6bddb2b8cbe",
      "f1eef6bdc9e174a9cf0570b0",
      "f1eef6bdc9e174a9cf2b8cbe045a8d",
      "f1eef6d0d1e6a6bddb74a9cf2b8cbe045a8d",
      "f1eef6d0d1e6a6bddb74a9cf3690c00570b0034e7b",
      "fff7fbece7f2d0d1e6a6bddb74a9cf3690c00570b0034e7b",
      "fff7fbece7f2d0d1e6a6bddb74a9cf3690c00570b0045a8d023858"
    ).map(colors);

    var PuBu = ramp(scheme$e);

    var scheme$f = new Array(3).concat(
      "e7e1efc994c7dd1c77",
      "f1eef6d7b5d8df65b0ce1256",
      "f1eef6d7b5d8df65b0dd1c77980043",
      "f1eef6d4b9dac994c7df65b0dd1c77980043",
      "f1eef6d4b9dac994c7df65b0e7298ace125691003f",
      "f7f4f9e7e1efd4b9dac994c7df65b0e7298ace125691003f",
      "f7f4f9e7e1efd4b9dac994c7df65b0e7298ace125698004367001f"
    ).map(colors);

    var PuRd = ramp(scheme$f);

    var scheme$g = new Array(3).concat(
      "fde0ddfa9fb5c51b8a",
      "feebe2fbb4b9f768a1ae017e",
      "feebe2fbb4b9f768a1c51b8a7a0177",
      "feebe2fcc5c0fa9fb5f768a1c51b8a7a0177",
      "feebe2fcc5c0fa9fb5f768a1dd3497ae017e7a0177",
      "fff7f3fde0ddfcc5c0fa9fb5f768a1dd3497ae017e7a0177",
      "fff7f3fde0ddfcc5c0fa9fb5f768a1dd3497ae017e7a017749006a"
    ).map(colors);

    var RdPu = ramp(scheme$g);

    var scheme$h = new Array(3).concat(
      "edf8b17fcdbb2c7fb8",
      "ffffcca1dab441b6c4225ea8",
      "ffffcca1dab441b6c42c7fb8253494",
      "ffffccc7e9b47fcdbb41b6c42c7fb8253494",
      "ffffccc7e9b47fcdbb41b6c41d91c0225ea80c2c84",
      "ffffd9edf8b1c7e9b47fcdbb41b6c41d91c0225ea80c2c84",
      "ffffd9edf8b1c7e9b47fcdbb41b6c41d91c0225ea8253494081d58"
    ).map(colors);

    var YlGnBu = ramp(scheme$h);

    var scheme$i = new Array(3).concat(
      "f7fcb9addd8e31a354",
      "ffffccc2e69978c679238443",
      "ffffccc2e69978c67931a354006837",
      "ffffccd9f0a3addd8e78c67931a354006837",
      "ffffccd9f0a3addd8e78c67941ab5d238443005a32",
      "ffffe5f7fcb9d9f0a3addd8e78c67941ab5d238443005a32",
      "ffffe5f7fcb9d9f0a3addd8e78c67941ab5d238443006837004529"
    ).map(colors);

    var YlGn = ramp(scheme$i);

    var scheme$j = new Array(3).concat(
      "fff7bcfec44fd95f0e",
      "ffffd4fed98efe9929cc4c02",
      "ffffd4fed98efe9929d95f0e993404",
      "ffffd4fee391fec44ffe9929d95f0e993404",
      "ffffd4fee391fec44ffe9929ec7014cc4c028c2d04",
      "ffffe5fff7bcfee391fec44ffe9929ec7014cc4c028c2d04",
      "ffffe5fff7bcfee391fec44ffe9929ec7014cc4c02993404662506"
    ).map(colors);

    var YlOrBr = ramp(scheme$j);

    var scheme$k = new Array(3).concat(
      "ffeda0feb24cf03b20",
      "ffffb2fecc5cfd8d3ce31a1c",
      "ffffb2fecc5cfd8d3cf03b20bd0026",
      "ffffb2fed976feb24cfd8d3cf03b20bd0026",
      "ffffb2fed976feb24cfd8d3cfc4e2ae31a1cb10026",
      "ffffccffeda0fed976feb24cfd8d3cfc4e2ae31a1cb10026",
      "ffffccffeda0fed976feb24cfd8d3cfc4e2ae31a1cbd0026800026"
    ).map(colors);

    var YlOrRd = ramp(scheme$k);

    var scheme$l = new Array(3).concat(
      "deebf79ecae13182bd",
      "eff3ffbdd7e76baed62171b5",
      "eff3ffbdd7e76baed63182bd08519c",
      "eff3ffc6dbef9ecae16baed63182bd08519c",
      "eff3ffc6dbef9ecae16baed64292c62171b5084594",
      "f7fbffdeebf7c6dbef9ecae16baed64292c62171b5084594",
      "f7fbffdeebf7c6dbef9ecae16baed64292c62171b508519c08306b"
    ).map(colors);

    var Blues = ramp(scheme$l);

    var scheme$m = new Array(3).concat(
      "e5f5e0a1d99b31a354",
      "edf8e9bae4b374c476238b45",
      "edf8e9bae4b374c47631a354006d2c",
      "edf8e9c7e9c0a1d99b74c47631a354006d2c",
      "edf8e9c7e9c0a1d99b74c47641ab5d238b45005a32",
      "f7fcf5e5f5e0c7e9c0a1d99b74c47641ab5d238b45005a32",
      "f7fcf5e5f5e0c7e9c0a1d99b74c47641ab5d238b45006d2c00441b"
    ).map(colors);

    var Greens = ramp(scheme$m);

    var scheme$n = new Array(3).concat(
      "f0f0f0bdbdbd636363",
      "f7f7f7cccccc969696525252",
      "f7f7f7cccccc969696636363252525",
      "f7f7f7d9d9d9bdbdbd969696636363252525",
      "f7f7f7d9d9d9bdbdbd969696737373525252252525",
      "fffffff0f0f0d9d9d9bdbdbd969696737373525252252525",
      "fffffff0f0f0d9d9d9bdbdbd969696737373525252252525000000"
    ).map(colors);

    var Greys = ramp(scheme$n);

    var scheme$o = new Array(3).concat(
      "efedf5bcbddc756bb1",
      "f2f0f7cbc9e29e9ac86a51a3",
      "f2f0f7cbc9e29e9ac8756bb154278f",
      "f2f0f7dadaebbcbddc9e9ac8756bb154278f",
      "f2f0f7dadaebbcbddc9e9ac8807dba6a51a34a1486",
      "fcfbfdefedf5dadaebbcbddc9e9ac8807dba6a51a34a1486",
      "fcfbfdefedf5dadaebbcbddc9e9ac8807dba6a51a354278f3f007d"
    ).map(colors);

    var Purples = ramp(scheme$o);

    var scheme$p = new Array(3).concat(
      "fee0d2fc9272de2d26",
      "fee5d9fcae91fb6a4acb181d",
      "fee5d9fcae91fb6a4ade2d26a50f15",
      "fee5d9fcbba1fc9272fb6a4ade2d26a50f15",
      "fee5d9fcbba1fc9272fb6a4aef3b2ccb181d99000d",
      "fff5f0fee0d2fcbba1fc9272fb6a4aef3b2ccb181d99000d",
      "fff5f0fee0d2fcbba1fc9272fb6a4aef3b2ccb181da50f1567000d"
    ).map(colors);

    var Reds = ramp(scheme$p);

    var scheme$q = new Array(3).concat(
      "fee6cefdae6be6550d",
      "feeddefdbe85fd8d3cd94701",
      "feeddefdbe85fd8d3ce6550da63603",
      "feeddefdd0a2fdae6bfd8d3ce6550da63603",
      "feeddefdd0a2fdae6bfd8d3cf16913d948018c2d04",
      "fff5ebfee6cefdd0a2fdae6bfd8d3cf16913d948018c2d04",
      "fff5ebfee6cefdd0a2fdae6bfd8d3cf16913d94801a636037f2704"
    ).map(colors);

    var Oranges = ramp(scheme$q);

    function cividis(t) {
      t = Math.max(0, Math.min(1, t));
      return "rgb("
          + Math.max(0, Math.min(255, Math.round(-4.54 - t * (35.34 - t * (2381.73 - t * (6402.7 - t * (7024.72 - t * 2710.57))))))) + ", "
          + Math.max(0, Math.min(255, Math.round(32.49 + t * (170.73 + t * (52.82 - t * (131.46 - t * (176.58 - t * 67.37))))))) + ", "
          + Math.max(0, Math.min(255, Math.round(81.24 + t * (442.36 - t * (2482.43 - t * (6167.24 - t * (6614.94 - t * 2475.67)))))))
          + ")";
    }

    var cubehelix$3 = cubehelixLong(cubehelix(300, 0.5, 0.0), cubehelix(-240, 0.5, 1.0));

    var warm = cubehelixLong(cubehelix(-100, 0.75, 0.35), cubehelix(80, 1.50, 0.8));

    var cool = cubehelixLong(cubehelix(260, 0.75, 0.35), cubehelix(80, 1.50, 0.8));

    var c = cubehelix();

    function rainbow(t) {
      if (t < 0 || t > 1) t -= Math.floor(t);
      var ts = Math.abs(t - 0.5);
      c.h = 360 * t - 100;
      c.s = 1.5 - 1.5 * ts;
      c.l = 0.8 - 0.9 * ts;
      return c + "";
    }

    var c$1 = rgb(),
        pi_1_3 = Math.PI / 3,
        pi_2_3 = Math.PI * 2 / 3;

    function sinebow(t) {
      var x;
      t = (0.5 - t) * Math.PI;
      c$1.r = 255 * (x = Math.sin(t)) * x;
      c$1.g = 255 * (x = Math.sin(t + pi_1_3)) * x;
      c$1.b = 255 * (x = Math.sin(t + pi_2_3)) * x;
      return c$1 + "";
    }

    function turbo(t) {
      t = Math.max(0, Math.min(1, t));
      return "rgb("
          + Math.max(0, Math.min(255, Math.round(34.61 + t * (1172.33 - t * (10793.56 - t * (33300.12 - t * (38394.49 - t * 14825.05))))))) + ", "
          + Math.max(0, Math.min(255, Math.round(23.31 + t * (557.33 + t * (1225.33 - t * (3574.96 - t * (1073.77 + t * 707.56))))))) + ", "
          + Math.max(0, Math.min(255, Math.round(27.2 + t * (3211.1 - t * (15327.97 - t * (27814 - t * (22569.18 - t * 6838.66)))))))
          + ")";
    }

    function ramp$1(range) {
      var n = range.length;
      return function(t) {
        return range[Math.max(0, Math.min(n - 1, Math.floor(t * n)))];
      };
    }

    var viridis = ramp$1(colors("44015444025645045745055946075a46085c460a5d460b5e470d60470e6147106347116447136548146748166848176948186a481a6c481b6d481c6e481d6f481f70482071482173482374482475482576482677482878482979472a7a472c7a472d7b472e7c472f7d46307e46327e46337f463480453581453781453882443983443a83443b84433d84433e85423f854240864241864142874144874045884046883f47883f48893e49893e4a893e4c8a3d4d8a3d4e8a3c4f8a3c508b3b518b3b528b3a538b3a548c39558c39568c38588c38598c375a8c375b8d365c8d365d8d355e8d355f8d34608d34618d33628d33638d32648e32658e31668e31678e31688e30698e306a8e2f6b8e2f6c8e2e6d8e2e6e8e2e6f8e2d708e2d718e2c718e2c728e2c738e2b748e2b758e2a768e2a778e2a788e29798e297a8e297b8e287c8e287d8e277e8e277f8e27808e26818e26828e26828e25838e25848e25858e24868e24878e23888e23898e238a8d228b8d228c8d228d8d218e8d218f8d21908d21918c20928c20928c20938c1f948c1f958b1f968b1f978b1f988b1f998a1f9a8a1e9b8a1e9c891e9d891f9e891f9f881fa0881fa1881fa1871fa28720a38620a48621a58521a68522a78522a88423a98324aa8325ab8225ac8226ad8127ad8128ae8029af7f2ab07f2cb17e2db27d2eb37c2fb47c31b57b32b67a34b67935b77937b87838b9773aba763bbb753dbc743fbc7340bd7242be7144bf7046c06f48c16e4ac16d4cc26c4ec36b50c46a52c56954c56856c66758c7655ac8645cc8635ec96260ca6063cb5f65cb5e67cc5c69cd5b6ccd5a6ece5870cf5773d05675d05477d1537ad1517cd2507fd34e81d34d84d44b86d54989d5488bd6468ed64590d74393d74195d84098d83e9bd93c9dd93ba0da39a2da37a5db36a8db34aadc32addc30b0dd2fb2dd2db5de2bb8de29bade28bddf26c0df25c2df23c5e021c8e020cae11fcde11dd0e11cd2e21bd5e21ad8e219dae319dde318dfe318e2e418e5e419e7e419eae51aece51befe51cf1e51df4e61ef6e620f8e621fbe723fde725"));

    var magma = ramp$1(colors("00000401000501010601010802010902020b02020d03030f03031204041405041606051806051a07061c08071e0907200a08220b09240c09260d0a290e0b2b100b2d110c2f120d31130d34140e36150e38160f3b180f3d19103f1a10421c10441d11471e114920114b21114e22115024125325125527125829115a2a115c2c115f2d11612f116331116533106734106936106b38106c390f6e3b0f703d0f713f0f72400f74420f75440f764510774710784910784a10794c117a4e117b4f127b51127c52137c54137d56147d57157e59157e5a167e5c167f5d177f5f187f601880621980641a80651a80671b80681c816a1c816b1d816d1d816e1e81701f81721f817320817521817621817822817922827b23827c23827e24828025828125818326818426818627818827818928818b29818c29818e2a81902a81912b81932b80942c80962c80982d80992d809b2e7f9c2e7f9e2f7fa02f7fa1307ea3307ea5317ea6317da8327daa337dab337cad347cae347bb0357bb2357bb3367ab5367ab73779b83779ba3878bc3978bd3977bf3a77c03a76c23b75c43c75c53c74c73d73c83e73ca3e72cc3f71cd4071cf4070d0416fd2426fd3436ed5446dd6456cd8456cd9466bdb476adc4869de4968df4a68e04c67e24d66e34e65e44f64e55064e75263e85362e95462ea5661eb5760ec5860ed5a5fee5b5eef5d5ef05f5ef1605df2625df2645cf3655cf4675cf4695cf56b5cf66c5cf66e5cf7705cf7725cf8745cf8765cf9785df9795df97b5dfa7d5efa7f5efa815ffb835ffb8560fb8761fc8961fc8a62fc8c63fc8e64fc9065fd9266fd9467fd9668fd9869fd9a6afd9b6bfe9d6cfe9f6dfea16efea36ffea571fea772fea973feaa74feac76feae77feb078feb27afeb47bfeb67cfeb77efeb97ffebb81febd82febf84fec185fec287fec488fec68afec88cfeca8dfecc8ffecd90fecf92fed194fed395fed597fed799fed89afdda9cfddc9efddea0fde0a1fde2a3fde3a5fde5a7fde7a9fde9aafdebacfcecaefceeb0fcf0b2fcf2b4fcf4b6fcf6b8fcf7b9fcf9bbfcfbbdfcfdbf"));

    var inferno = ramp$1(colors("00000401000501010601010802010a02020c02020e03021004031204031405041706041907051b08051d09061f0a07220b07240c08260d08290e092b10092d110a30120a32140b34150b37160b39180c3c190c3e1b0c411c0c431e0c451f0c48210c4a230c4c240c4f260c51280b53290b552b0b572d0b592f0a5b310a5c320a5e340a5f3609613809623909633b09643d09653e0966400a67420a68440a68450a69470b6a490b6a4a0c6b4c0c6b4d0d6c4f0d6c510e6c520e6d540f6d550f6d57106e59106e5a116e5c126e5d126e5f136e61136e62146e64156e65156e67166e69166e6a176e6c186e6d186e6f196e71196e721a6e741a6e751b6e771c6d781c6d7a1d6d7c1d6d7d1e6d7f1e6c801f6c82206c84206b85216b87216b88226a8a226a8c23698d23698f24699025689225689326679526679727669827669a28659b29649d29649f2a63a02a63a22b62a32c61a52c60a62d60a82e5fa92e5eab2f5ead305dae305cb0315bb1325ab3325ab43359b63458b73557b93556ba3655bc3754bd3853bf3952c03a51c13a50c33b4fc43c4ec63d4dc73e4cc83f4bca404acb4149cc4248ce4347cf4446d04545d24644d34743d44842d54a41d74b3fd84c3ed94d3dda4e3cdb503bdd513ade5238df5337e05536e15635e25734e35933e45a31e55c30e65d2fe75e2ee8602de9612bea632aeb6429eb6628ec6726ed6925ee6a24ef6c23ef6e21f06f20f1711ff1731df2741cf3761bf37819f47918f57b17f57d15f67e14f68013f78212f78410f8850ff8870ef8890cf98b0bf98c0af98e09fa9008fa9207fa9407fb9606fb9706fb9906fb9b06fb9d07fc9f07fca108fca309fca50afca60cfca80dfcaa0ffcac11fcae12fcb014fcb216fcb418fbb61afbb81dfbba1ffbbc21fbbe23fac026fac228fac42afac62df9c72ff9c932f9cb35f8cd37f8cf3af7d13df7d340f6d543f6d746f5d949f5db4cf4dd4ff4df53f4e156f3e35af3e55df2e661f2e865f2ea69f1ec6df1ed71f1ef75f1f179f2f27df2f482f3f586f3f68af4f88ef5f992f6fa96f8fb9af9fc9dfafda1fcffa4"));

    var plasma = ramp$1(colors("0d088710078813078916078a19068c1b068d1d068e20068f2206902406912605912805922a05932c05942e05952f059631059733059735049837049938049a3a049a3c049b3e049c3f049c41049d43039e44039e46039f48039f4903a04b03a14c02a14e02a25002a25102a35302a35502a45601a45801a45901a55b01a55c01a65e01a66001a66100a76300a76400a76600a76700a86900a86a00a86c00a86e00a86f00a87100a87201a87401a87501a87701a87801a87a02a87b02a87d03a87e03a88004a88104a78305a78405a78606a68707a68808a68a09a58b0aa58d0ba58e0ca48f0da4910ea3920fa39410a29511a19613a19814a099159f9a169f9c179e9d189d9e199da01a9ca11b9ba21d9aa31e9aa51f99a62098a72197a82296aa2395ab2494ac2694ad2793ae2892b02991b12a90b22b8fb32c8eb42e8db52f8cb6308bb7318ab83289ba3388bb3488bc3587bd3786be3885bf3984c03a83c13b82c23c81c33d80c43e7fc5407ec6417dc7427cc8437bc9447aca457acb4679cc4778cc4977cd4a76ce4b75cf4c74d04d73d14e72d24f71d35171d45270d5536fd5546ed6556dd7566cd8576bd9586ada5a6ada5b69db5c68dc5d67dd5e66de5f65de6164df6263e06363e16462e26561e26660e3685fe4695ee56a5de56b5de66c5ce76e5be76f5ae87059e97158e97257ea7457eb7556eb7655ec7754ed7953ed7a52ee7b51ef7c51ef7e50f07f4ff0804ef1814df1834cf2844bf3854bf3874af48849f48948f58b47f58c46f68d45f68f44f79044f79143f79342f89441f89540f9973ff9983ef99a3efa9b3dfa9c3cfa9e3bfb9f3afba139fba238fca338fca537fca636fca835fca934fdab33fdac33fdae32fdaf31fdb130fdb22ffdb42ffdb52efeb72dfeb82cfeba2cfebb2bfebd2afebe2afec029fdc229fdc328fdc527fdc627fdc827fdca26fdcb26fccd25fcce25fcd025fcd225fbd324fbd524fbd724fad824fada24f9dc24f9dd25f8df25f8e125f7e225f7e425f6e626f6e826f5e926f5eb27f4ed27f3ee27f3f027f2f227f1f426f1f525f0f724f0f921"));

    function constant$b(x) {
      return function constant() {
        return x;
      };
    }

    var abs$1 = Math.abs;
    var atan2$1 = Math.atan2;
    var cos$2 = Math.cos;
    var max$2 = Math.max;
    var min$1 = Math.min;
    var sin$2 = Math.sin;
    var sqrt$2 = Math.sqrt;

    var epsilon$3 = 1e-12;
    var pi$4 = Math.PI;
    var halfPi$3 = pi$4 / 2;
    var tau$4 = 2 * pi$4;

    function acos$1(x) {
      return x > 1 ? 0 : x < -1 ? pi$4 : Math.acos(x);
    }

    function asin$1(x) {
      return x >= 1 ? halfPi$3 : x <= -1 ? -halfPi$3 : Math.asin(x);
    }

    function arcInnerRadius(d) {
      return d.innerRadius;
    }

    function arcOuterRadius(d) {
      return d.outerRadius;
    }

    function arcStartAngle(d) {
      return d.startAngle;
    }

    function arcEndAngle(d) {
      return d.endAngle;
    }

    function arcPadAngle(d) {
      return d && d.padAngle; // Note: optional!
    }

    function intersect(x0, y0, x1, y1, x2, y2, x3, y3) {
      var x10 = x1 - x0, y10 = y1 - y0,
          x32 = x3 - x2, y32 = y3 - y2,
          t = y32 * x10 - x32 * y10;
      if (t * t < epsilon$3) return;
      t = (x32 * (y0 - y2) - y32 * (x0 - x2)) / t;
      return [x0 + t * x10, y0 + t * y10];
    }

    // Compute perpendicular offset line of length rc.
    // http://mathworld.wolfram.com/Circle-LineIntersection.html
    function cornerTangents(x0, y0, x1, y1, r1, rc, cw) {
      var x01 = x0 - x1,
          y01 = y0 - y1,
          lo = (cw ? rc : -rc) / sqrt$2(x01 * x01 + y01 * y01),
          ox = lo * y01,
          oy = -lo * x01,
          x11 = x0 + ox,
          y11 = y0 + oy,
          x10 = x1 + ox,
          y10 = y1 + oy,
          x00 = (x11 + x10) / 2,
          y00 = (y11 + y10) / 2,
          dx = x10 - x11,
          dy = y10 - y11,
          d2 = dx * dx + dy * dy,
          r = r1 - rc,
          D = x11 * y10 - x10 * y11,
          d = (dy < 0 ? -1 : 1) * sqrt$2(max$2(0, r * r * d2 - D * D)),
          cx0 = (D * dy - dx * d) / d2,
          cy0 = (-D * dx - dy * d) / d2,
          cx1 = (D * dy + dx * d) / d2,
          cy1 = (-D * dx + dy * d) / d2,
          dx0 = cx0 - x00,
          dy0 = cy0 - y00,
          dx1 = cx1 - x00,
          dy1 = cy1 - y00;

      // Pick the closer of the two intersection points.
      // TODO Is there a faster way to determine which intersection to use?
      if (dx0 * dx0 + dy0 * dy0 > dx1 * dx1 + dy1 * dy1) cx0 = cx1, cy0 = cy1;

      return {
        cx: cx0,
        cy: cy0,
        x01: -ox,
        y01: -oy,
        x11: cx0 * (r1 / r - 1),
        y11: cy0 * (r1 / r - 1)
      };
    }

    function arc() {
      var innerRadius = arcInnerRadius,
          outerRadius = arcOuterRadius,
          cornerRadius = constant$b(0),
          padRadius = null,
          startAngle = arcStartAngle,
          endAngle = arcEndAngle,
          padAngle = arcPadAngle,
          context = null;

      function arc() {
        var buffer,
            r,
            r0 = +innerRadius.apply(this, arguments),
            r1 = +outerRadius.apply(this, arguments),
            a0 = startAngle.apply(this, arguments) - halfPi$3,
            a1 = endAngle.apply(this, arguments) - halfPi$3,
            da = abs$1(a1 - a0),
            cw = a1 > a0;

        if (!context) context = buffer = path();

        // Ensure that the outer radius is always larger than the inner radius.
        if (r1 < r0) r = r1, r1 = r0, r0 = r;

        // Is it a point?
        if (!(r1 > epsilon$3)) context.moveTo(0, 0);

        // Or is it a circle or annulus?
        else if (da > tau$4 - epsilon$3) {
          context.moveTo(r1 * cos$2(a0), r1 * sin$2(a0));
          context.arc(0, 0, r1, a0, a1, !cw);
          if (r0 > epsilon$3) {
            context.moveTo(r0 * cos$2(a1), r0 * sin$2(a1));
            context.arc(0, 0, r0, a1, a0, cw);
          }
        }

        // Or is it a circular or annular sector?
        else {
          var a01 = a0,
              a11 = a1,
              a00 = a0,
              a10 = a1,
              da0 = da,
              da1 = da,
              ap = padAngle.apply(this, arguments) / 2,
              rp = (ap > epsilon$3) && (padRadius ? +padRadius.apply(this, arguments) : sqrt$2(r0 * r0 + r1 * r1)),
              rc = min$1(abs$1(r1 - r0) / 2, +cornerRadius.apply(this, arguments)),
              rc0 = rc,
              rc1 = rc,
              t0,
              t1;

          // Apply padding? Note that since r1 ≥ r0, da1 ≥ da0.
          if (rp > epsilon$3) {
            var p0 = asin$1(rp / r0 * sin$2(ap)),
                p1 = asin$1(rp / r1 * sin$2(ap));
            if ((da0 -= p0 * 2) > epsilon$3) p0 *= (cw ? 1 : -1), a00 += p0, a10 -= p0;
            else da0 = 0, a00 = a10 = (a0 + a1) / 2;
            if ((da1 -= p1 * 2) > epsilon$3) p1 *= (cw ? 1 : -1), a01 += p1, a11 -= p1;
            else da1 = 0, a01 = a11 = (a0 + a1) / 2;
          }

          var x01 = r1 * cos$2(a01),
              y01 = r1 * sin$2(a01),
              x10 = r0 * cos$2(a10),
              y10 = r0 * sin$2(a10);

          // Apply rounded corners?
          if (rc > epsilon$3) {
            var x11 = r1 * cos$2(a11),
                y11 = r1 * sin$2(a11),
                x00 = r0 * cos$2(a00),
                y00 = r0 * sin$2(a00),
                oc;

            // Restrict the corner radius according to the sector angle.
            if (da < pi$4 && (oc = intersect(x01, y01, x00, y00, x11, y11, x10, y10))) {
              var ax = x01 - oc[0],
                  ay = y01 - oc[1],
                  bx = x11 - oc[0],
                  by = y11 - oc[1],
                  kc = 1 / sin$2(acos$1((ax * bx + ay * by) / (sqrt$2(ax * ax + ay * ay) * sqrt$2(bx * bx + by * by))) / 2),
                  lc = sqrt$2(oc[0] * oc[0] + oc[1] * oc[1]);
              rc0 = min$1(rc, (r0 - lc) / (kc - 1));
              rc1 = min$1(rc, (r1 - lc) / (kc + 1));
            }
          }

          // Is the sector collapsed to a line?
          if (!(da1 > epsilon$3)) context.moveTo(x01, y01);

          // Does the sector’s outer ring have rounded corners?
          else if (rc1 > epsilon$3) {
            t0 = cornerTangents(x00, y00, x01, y01, r1, rc1, cw);
            t1 = cornerTangents(x11, y11, x10, y10, r1, rc1, cw);

            context.moveTo(t0.cx + t0.x01, t0.cy + t0.y01);

            // Have the corners merged?
            if (rc1 < rc) context.arc(t0.cx, t0.cy, rc1, atan2$1(t0.y01, t0.x01), atan2$1(t1.y01, t1.x01), !cw);

            // Otherwise, draw the two corners and the ring.
            else {
              context.arc(t0.cx, t0.cy, rc1, atan2$1(t0.y01, t0.x01), atan2$1(t0.y11, t0.x11), !cw);
              context.arc(0, 0, r1, atan2$1(t0.cy + t0.y11, t0.cx + t0.x11), atan2$1(t1.cy + t1.y11, t1.cx + t1.x11), !cw);
              context.arc(t1.cx, t1.cy, rc1, atan2$1(t1.y11, t1.x11), atan2$1(t1.y01, t1.x01), !cw);
            }
          }

          // Or is the outer ring just a circular arc?
          else context.moveTo(x01, y01), context.arc(0, 0, r1, a01, a11, !cw);

          // Is there no inner ring, and it’s a circular sector?
          // Or perhaps it’s an annular sector collapsed due to padding?
          if (!(r0 > epsilon$3) || !(da0 > epsilon$3)) context.lineTo(x10, y10);

          // Does the sector’s inner ring (or point) have rounded corners?
          else if (rc0 > epsilon$3) {
            t0 = cornerTangents(x10, y10, x11, y11, r0, -rc0, cw);
            t1 = cornerTangents(x01, y01, x00, y00, r0, -rc0, cw);

            context.lineTo(t0.cx + t0.x01, t0.cy + t0.y01);

            // Have the corners merged?
            if (rc0 < rc) context.arc(t0.cx, t0.cy, rc0, atan2$1(t0.y01, t0.x01), atan2$1(t1.y01, t1.x01), !cw);

            // Otherwise, draw the two corners and the ring.
            else {
              context.arc(t0.cx, t0.cy, rc0, atan2$1(t0.y01, t0.x01), atan2$1(t0.y11, t0.x11), !cw);
              context.arc(0, 0, r0, atan2$1(t0.cy + t0.y11, t0.cx + t0.x11), atan2$1(t1.cy + t1.y11, t1.cx + t1.x11), cw);
              context.arc(t1.cx, t1.cy, rc0, atan2$1(t1.y11, t1.x11), atan2$1(t1.y01, t1.x01), !cw);
            }
          }

          // Or is the inner ring just a circular arc?
          else context.arc(0, 0, r0, a10, a00, cw);
        }

        context.closePath();

        if (buffer) return context = null, buffer + "" || null;
      }

      arc.centroid = function() {
        var r = (+innerRadius.apply(this, arguments) + +outerRadius.apply(this, arguments)) / 2,
            a = (+startAngle.apply(this, arguments) + +endAngle.apply(this, arguments)) / 2 - pi$4 / 2;
        return [cos$2(a) * r, sin$2(a) * r];
      };

      arc.innerRadius = function(_) {
        return arguments.length ? (innerRadius = typeof _ === "function" ? _ : constant$b(+_), arc) : innerRadius;
      };

      arc.outerRadius = function(_) {
        return arguments.length ? (outerRadius = typeof _ === "function" ? _ : constant$b(+_), arc) : outerRadius;
      };

      arc.cornerRadius = function(_) {
        return arguments.length ? (cornerRadius = typeof _ === "function" ? _ : constant$b(+_), arc) : cornerRadius;
      };

      arc.padRadius = function(_) {
        return arguments.length ? (padRadius = _ == null ? null : typeof _ === "function" ? _ : constant$b(+_), arc) : padRadius;
      };

      arc.startAngle = function(_) {
        return arguments.length ? (startAngle = typeof _ === "function" ? _ : constant$b(+_), arc) : startAngle;
      };

      arc.endAngle = function(_) {
        return arguments.length ? (endAngle = typeof _ === "function" ? _ : constant$b(+_), arc) : endAngle;
      };

      arc.padAngle = function(_) {
        return arguments.length ? (padAngle = typeof _ === "function" ? _ : constant$b(+_), arc) : padAngle;
      };

      arc.context = function(_) {
        return arguments.length ? ((context = _ == null ? null : _), arc) : context;
      };

      return arc;
    }

    function Linear(context) {
      this._context = context;
    }

    Linear.prototype = {
      areaStart: function() {
        this._line = 0;
      },
      areaEnd: function() {
        this._line = NaN;
      },
      lineStart: function() {
        this._point = 0;
      },
      lineEnd: function() {
        if (this._line || (this._line !== 0 && this._point === 1)) this._context.closePath();
        this._line = 1 - this._line;
      },
      point: function(x, y) {
        x = +x, y = +y;
        switch (this._point) {
          case 0: this._point = 1; this._line ? this._context.lineTo(x, y) : this._context.moveTo(x, y); break;
          case 1: this._point = 2; // proceed
          default: this._context.lineTo(x, y); break;
        }
      }
    };

    function curveLinear(context) {
      return new Linear(context);
    }

    function x$3(p) {
      return p[0];
    }

    function y$3(p) {
      return p[1];
    }

    function line() {
      var x = x$3,
          y = y$3,
          defined = constant$b(true),
          context = null,
          curve = curveLinear,
          output = null;

      function line(data) {
        var i,
            n = data.length,
            d,
            defined0 = false,
            buffer;

        if (context == null) output = curve(buffer = path());

        for (i = 0; i <= n; ++i) {
          if (!(i < n && defined(d = data[i], i, data)) === defined0) {
            if (defined0 = !defined0) output.lineStart();
            else output.lineEnd();
          }
          if (defined0) output.point(+x(d, i, data), +y(d, i, data));
        }

        if (buffer) return output = null, buffer + "" || null;
      }

      line.x = function(_) {
        return arguments.length ? (x = typeof _ === "function" ? _ : constant$b(+_), line) : x;
      };

      line.y = function(_) {
        return arguments.length ? (y = typeof _ === "function" ? _ : constant$b(+_), line) : y;
      };

      line.defined = function(_) {
        return arguments.length ? (defined = typeof _ === "function" ? _ : constant$b(!!_), line) : defined;
      };

      line.curve = function(_) {
        return arguments.length ? (curve = _, context != null && (output = curve(context)), line) : curve;
      };

      line.context = function(_) {
        return arguments.length ? (_ == null ? context = output = null : output = curve(context = _), line) : context;
      };

      return line;
    }

    function area$3() {
      var x0 = x$3,
          x1 = null,
          y0 = constant$b(0),
          y1 = y$3,
          defined = constant$b(true),
          context = null,
          curve = curveLinear,
          output = null;

      function area(data) {
        var i,
            j,
            k,
            n = data.length,
            d,
            defined0 = false,
            buffer,
            x0z = new Array(n),
            y0z = new Array(n);

        if (context == null) output = curve(buffer = path());

        for (i = 0; i <= n; ++i) {
          if (!(i < n && defined(d = data[i], i, data)) === defined0) {
            if (defined0 = !defined0) {
              j = i;
              output.areaStart();
              output.lineStart();
            } else {
              output.lineEnd();
              output.lineStart();
              for (k = i - 1; k >= j; --k) {
                output.point(x0z[k], y0z[k]);
              }
              output.lineEnd();
              output.areaEnd();
            }
          }
          if (defined0) {
            x0z[i] = +x0(d, i, data), y0z[i] = +y0(d, i, data);
            output.point(x1 ? +x1(d, i, data) : x0z[i], y1 ? +y1(d, i, data) : y0z[i]);
          }
        }

        if (buffer) return output = null, buffer + "" || null;
      }

      function arealine() {
        return line().defined(defined).curve(curve).context(context);
      }

      area.x = function(_) {
        return arguments.length ? (x0 = typeof _ === "function" ? _ : constant$b(+_), x1 = null, area) : x0;
      };

      area.x0 = function(_) {
        return arguments.length ? (x0 = typeof _ === "function" ? _ : constant$b(+_), area) : x0;
      };

      area.x1 = function(_) {
        return arguments.length ? (x1 = _ == null ? null : typeof _ === "function" ? _ : constant$b(+_), area) : x1;
      };

      area.y = function(_) {
        return arguments.length ? (y0 = typeof _ === "function" ? _ : constant$b(+_), y1 = null, area) : y0;
      };

      area.y0 = function(_) {
        return arguments.length ? (y0 = typeof _ === "function" ? _ : constant$b(+_), area) : y0;
      };

      area.y1 = function(_) {
        return arguments.length ? (y1 = _ == null ? null : typeof _ === "function" ? _ : constant$b(+_), area) : y1;
      };

      area.lineX0 =
      area.lineY0 = function() {
        return arealine().x(x0).y(y0);
      };

      area.lineY1 = function() {
        return arealine().x(x0).y(y1);
      };

      area.lineX1 = function() {
        return arealine().x(x1).y(y0);
      };

      area.defined = function(_) {
        return arguments.length ? (defined = typeof _ === "function" ? _ : constant$b(!!_), area) : defined;
      };

      area.curve = function(_) {
        return arguments.length ? (curve = _, context != null && (output = curve(context)), area) : curve;
      };

      area.context = function(_) {
        return arguments.length ? (_ == null ? context = output = null : output = curve(context = _), area) : context;
      };

      return area;
    }

    function descending$1(a, b) {
      return b < a ? -1 : b > a ? 1 : b >= a ? 0 : NaN;
    }

    function identity$8(d) {
      return d;
    }

    function pie() {
      var value = identity$8,
          sortValues = descending$1,
          sort = null,
          startAngle = constant$b(0),
          endAngle = constant$b(tau$4),
          padAngle = constant$b(0);

      function pie(data) {
        var i,
            n = data.length,
            j,
            k,
            sum = 0,
            index = new Array(n),
            arcs = new Array(n),
            a0 = +startAngle.apply(this, arguments),
            da = Math.min(tau$4, Math.max(-tau$4, endAngle.apply(this, arguments) - a0)),
            a1,
            p = Math.min(Math.abs(da) / n, padAngle.apply(this, arguments)),
            pa = p * (da < 0 ? -1 : 1),
            v;

        for (i = 0; i < n; ++i) {
          if ((v = arcs[index[i] = i] = +value(data[i], i, data)) > 0) {
            sum += v;
          }
        }

        // Optionally sort the arcs by previously-computed values or by data.
        if (sortValues != null) index.sort(function(i, j) { return sortValues(arcs[i], arcs[j]); });
        else if (sort != null) index.sort(function(i, j) { return sort(data[i], data[j]); });

        // Compute the arcs! They are stored in the original data's order.
        for (i = 0, k = sum ? (da - n * pa) / sum : 0; i < n; ++i, a0 = a1) {
          j = index[i], v = arcs[j], a1 = a0 + (v > 0 ? v * k : 0) + pa, arcs[j] = {
            data: data[j],
            index: i,
            value: v,
            startAngle: a0,
            endAngle: a1,
            padAngle: p
          };
        }

        return arcs;
      }

      pie.value = function(_) {
        return arguments.length ? (value = typeof _ === "function" ? _ : constant$b(+_), pie) : value;
      };

      pie.sortValues = function(_) {
        return arguments.length ? (sortValues = _, sort = null, pie) : sortValues;
      };

      pie.sort = function(_) {
        return arguments.length ? (sort = _, sortValues = null, pie) : sort;
      };

      pie.startAngle = function(_) {
        return arguments.length ? (startAngle = typeof _ === "function" ? _ : constant$b(+_), pie) : startAngle;
      };

      pie.endAngle = function(_) {
        return arguments.length ? (endAngle = typeof _ === "function" ? _ : constant$b(+_), pie) : endAngle;
      };

      pie.padAngle = function(_) {
        return arguments.length ? (padAngle = typeof _ === "function" ? _ : constant$b(+_), pie) : padAngle;
      };

      return pie;
    }

    var curveRadialLinear = curveRadial(curveLinear);

    function Radial(curve) {
      this._curve = curve;
    }

    Radial.prototype = {
      areaStart: function() {
        this._curve.areaStart();
      },
      areaEnd: function() {
        this._curve.areaEnd();
      },
      lineStart: function() {
        this._curve.lineStart();
      },
      lineEnd: function() {
        this._curve.lineEnd();
      },
      point: function(a, r) {
        this._curve.point(r * Math.sin(a), r * -Math.cos(a));
      }
    };

    function curveRadial(curve) {

      function radial(context) {
        return new Radial(curve(context));
      }

      radial._curve = curve;

      return radial;
    }

    function lineRadial(l) {
      var c = l.curve;

      l.angle = l.x, delete l.x;
      l.radius = l.y, delete l.y;

      l.curve = function(_) {
        return arguments.length ? c(curveRadial(_)) : c()._curve;
      };

      return l;
    }

    function lineRadial$1() {
      return lineRadial(line().curve(curveRadialLinear));
    }

    function areaRadial() {
      var a = area$3().curve(curveRadialLinear),
          c = a.curve,
          x0 = a.lineX0,
          x1 = a.lineX1,
          y0 = a.lineY0,
          y1 = a.lineY1;

      a.angle = a.x, delete a.x;
      a.startAngle = a.x0, delete a.x0;
      a.endAngle = a.x1, delete a.x1;
      a.radius = a.y, delete a.y;
      a.innerRadius = a.y0, delete a.y0;
      a.outerRadius = a.y1, delete a.y1;
      a.lineStartAngle = function() { return lineRadial(x0()); }, delete a.lineX0;
      a.lineEndAngle = function() { return lineRadial(x1()); }, delete a.lineX1;
      a.lineInnerRadius = function() { return lineRadial(y0()); }, delete a.lineY0;
      a.lineOuterRadius = function() { return lineRadial(y1()); }, delete a.lineY1;

      a.curve = function(_) {
        return arguments.length ? c(curveRadial(_)) : c()._curve;
      };

      return a;
    }

    function pointRadial(x, y) {
      return [(y = +y) * Math.cos(x -= Math.PI / 2), y * Math.sin(x)];
    }

    var slice$6 = Array.prototype.slice;

    function linkSource(d) {
      return d.source;
    }

    function linkTarget(d) {
      return d.target;
    }

    function link$2(curve) {
      var source = linkSource,
          target = linkTarget,
          x = x$3,
          y = y$3,
          context = null;

      function link() {
        var buffer, argv = slice$6.call(arguments), s = source.apply(this, argv), t = target.apply(this, argv);
        if (!context) context = buffer = path();
        curve(context, +x.apply(this, (argv[0] = s, argv)), +y.apply(this, argv), +x.apply(this, (argv[0] = t, argv)), +y.apply(this, argv));
        if (buffer) return context = null, buffer + "" || null;
      }

      link.source = function(_) {
        return arguments.length ? (source = _, link) : source;
      };

      link.target = function(_) {
        return arguments.length ? (target = _, link) : target;
      };

      link.x = function(_) {
        return arguments.length ? (x = typeof _ === "function" ? _ : constant$b(+_), link) : x;
      };

      link.y = function(_) {
        return arguments.length ? (y = typeof _ === "function" ? _ : constant$b(+_), link) : y;
      };

      link.context = function(_) {
        return arguments.length ? ((context = _ == null ? null : _), link) : context;
      };

      return link;
    }

    function curveHorizontal(context, x0, y0, x1, y1) {
      context.moveTo(x0, y0);
      context.bezierCurveTo(x0 = (x0 + x1) / 2, y0, x0, y1, x1, y1);
    }

    function curveVertical(context, x0, y0, x1, y1) {
      context.moveTo(x0, y0);
      context.bezierCurveTo(x0, y0 = (y0 + y1) / 2, x1, y0, x1, y1);
    }

    function curveRadial$1(context, x0, y0, x1, y1) {
      var p0 = pointRadial(x0, y0),
          p1 = pointRadial(x0, y0 = (y0 + y1) / 2),
          p2 = pointRadial(x1, y0),
          p3 = pointRadial(x1, y1);
      context.moveTo(p0[0], p0[1]);
      context.bezierCurveTo(p1[0], p1[1], p2[0], p2[1], p3[0], p3[1]);
    }

    function linkHorizontal() {
      return link$2(curveHorizontal);
    }

    function linkVertical() {
      return link$2(curveVertical);
    }

    function linkRadial() {
      var l = link$2(curveRadial$1);
      l.angle = l.x, delete l.x;
      l.radius = l.y, delete l.y;
      return l;
    }

    var circle$2 = {
      draw: function(context, size) {
        var r = Math.sqrt(size / pi$4);
        context.moveTo(r, 0);
        context.arc(0, 0, r, 0, tau$4);
      }
    };

    var cross$2 = {
      draw: function(context, size) {
        var r = Math.sqrt(size / 5) / 2;
        context.moveTo(-3 * r, -r);
        context.lineTo(-r, -r);
        context.lineTo(-r, -3 * r);
        context.lineTo(r, -3 * r);
        context.lineTo(r, -r);
        context.lineTo(3 * r, -r);
        context.lineTo(3 * r, r);
        context.lineTo(r, r);
        context.lineTo(r, 3 * r);
        context.lineTo(-r, 3 * r);
        context.lineTo(-r, r);
        context.lineTo(-3 * r, r);
        context.closePath();
      }
    };

    var tan30 = Math.sqrt(1 / 3),
        tan30_2 = tan30 * 2;

    var diamond = {
      draw: function(context, size) {
        var y = Math.sqrt(size / tan30_2),
            x = y * tan30;
        context.moveTo(0, -y);
        context.lineTo(x, 0);
        context.lineTo(0, y);
        context.lineTo(-x, 0);
        context.closePath();
      }
    };

    var ka = 0.89081309152928522810,
        kr = Math.sin(pi$4 / 10) / Math.sin(7 * pi$4 / 10),
        kx = Math.sin(tau$4 / 10) * kr,
        ky = -Math.cos(tau$4 / 10) * kr;

    var star = {
      draw: function(context, size) {
        var r = Math.sqrt(size * ka),
            x = kx * r,
            y = ky * r;
        context.moveTo(0, -r);
        context.lineTo(x, y);
        for (var i = 1; i < 5; ++i) {
          var a = tau$4 * i / 5,
              c = Math.cos(a),
              s = Math.sin(a);
          context.lineTo(s * r, -c * r);
          context.lineTo(c * x - s * y, s * x + c * y);
        }
        context.closePath();
      }
    };

    var square = {
      draw: function(context, size) {
        var w = Math.sqrt(size),
            x = -w / 2;
        context.rect(x, x, w, w);
      }
    };

    var sqrt3 = Math.sqrt(3);

    var triangle = {
      draw: function(context, size) {
        var y = -Math.sqrt(size / (sqrt3 * 3));
        context.moveTo(0, y * 2);
        context.lineTo(-sqrt3 * y, -y);
        context.lineTo(sqrt3 * y, -y);
        context.closePath();
      }
    };

    var c$2 = -0.5,
        s = Math.sqrt(3) / 2,
        k = 1 / Math.sqrt(12),
        a = (k / 2 + 1) * 3;

    var wye = {
      draw: function(context, size) {
        var r = Math.sqrt(size / a),
            x0 = r / 2,
            y0 = r * k,
            x1 = x0,
            y1 = r * k + r,
            x2 = -x1,
            y2 = y1;
        context.moveTo(x0, y0);
        context.lineTo(x1, y1);
        context.lineTo(x2, y2);
        context.lineTo(c$2 * x0 - s * y0, s * x0 + c$2 * y0);
        context.lineTo(c$2 * x1 - s * y1, s * x1 + c$2 * y1);
        context.lineTo(c$2 * x2 - s * y2, s * x2 + c$2 * y2);
        context.lineTo(c$2 * x0 + s * y0, c$2 * y0 - s * x0);
        context.lineTo(c$2 * x1 + s * y1, c$2 * y1 - s * x1);
        context.lineTo(c$2 * x2 + s * y2, c$2 * y2 - s * x2);
        context.closePath();
      }
    };

    var symbols = [
      circle$2,
      cross$2,
      diamond,
      square,
      star,
      triangle,
      wye
    ];

    function symbol() {
      var type = constant$b(circle$2),
          size = constant$b(64),
          context = null;

      function symbol() {
        var buffer;
        if (!context) context = buffer = path();
        type.apply(this, arguments).draw(context, +size.apply(this, arguments));
        if (buffer) return context = null, buffer + "" || null;
      }

      symbol.type = function(_) {
        return arguments.length ? (type = typeof _ === "function" ? _ : constant$b(_), symbol) : type;
      };

      symbol.size = function(_) {
        return arguments.length ? (size = typeof _ === "function" ? _ : constant$b(+_), symbol) : size;
      };

      symbol.context = function(_) {
        return arguments.length ? (context = _ == null ? null : _, symbol) : context;
      };

      return symbol;
    }

    function noop$3() {}

    function point$2(that, x, y) {
      that._context.bezierCurveTo(
        (2 * that._x0 + that._x1) / 3,
        (2 * that._y0 + that._y1) / 3,
        (that._x0 + 2 * that._x1) / 3,
        (that._y0 + 2 * that._y1) / 3,
        (that._x0 + 4 * that._x1 + x) / 6,
        (that._y0 + 4 * that._y1 + y) / 6
      );
    }

    function Basis(context) {
      this._context = context;
    }

    Basis.prototype = {
      areaStart: function() {
        this._line = 0;
      },
      areaEnd: function() {
        this._line = NaN;
      },
      lineStart: function() {
        this._x0 = this._x1 =
        this._y0 = this._y1 = NaN;
        this._point = 0;
      },
      lineEnd: function() {
        switch (this._point) {
          case 3: point$2(this, this._x1, this._y1); // proceed
          case 2: this._context.lineTo(this._x1, this._y1); break;
        }
        if (this._line || (this._line !== 0 && this._point === 1)) this._context.closePath();
        this._line = 1 - this._line;
      },
      point: function(x, y) {
        x = +x, y = +y;
        switch (this._point) {
          case 0: this._point = 1; this._line ? this._context.lineTo(x, y) : this._context.moveTo(x, y); break;
          case 1: this._point = 2; break;
          case 2: this._point = 3; this._context.lineTo((5 * this._x0 + this._x1) / 6, (5 * this._y0 + this._y1) / 6); // proceed
          default: point$2(this, x, y); break;
        }
        this._x0 = this._x1, this._x1 = x;
        this._y0 = this._y1, this._y1 = y;
      }
    };

    function basis$2(context) {
      return new Basis(context);
    }

    function BasisClosed(context) {
      this._context = context;
    }

    BasisClosed.prototype = {
      areaStart: noop$3,
      areaEnd: noop$3,
      lineStart: function() {
        this._x0 = this._x1 = this._x2 = this._x3 = this._x4 =
        this._y0 = this._y1 = this._y2 = this._y3 = this._y4 = NaN;
        this._point = 0;
      },
      lineEnd: function() {
        switch (this._point) {
          case 1: {
            this._context.moveTo(this._x2, this._y2);
            this._context.closePath();
            break;
          }
          case 2: {
            this._context.moveTo((this._x2 + 2 * this._x3) / 3, (this._y2 + 2 * this._y3) / 3);
            this._context.lineTo((this._x3 + 2 * this._x2) / 3, (this._y3 + 2 * this._y2) / 3);
            this._context.closePath();
            break;
          }
          case 3: {
            this.point(this._x2, this._y2);
            this.point(this._x3, this._y3);
            this.point(this._x4, this._y4);
            break;
          }
        }
      },
      point: function(x, y) {
        x = +x, y = +y;
        switch (this._point) {
          case 0: this._point = 1; this._x2 = x, this._y2 = y; break;
          case 1: this._point = 2; this._x3 = x, this._y3 = y; break;
          case 2: this._point = 3; this._x4 = x, this._y4 = y; this._context.moveTo((this._x0 + 4 * this._x1 + x) / 6, (this._y0 + 4 * this._y1 + y) / 6); break;
          default: point$2(this, x, y); break;
        }
        this._x0 = this._x1, this._x1 = x;
        this._y0 = this._y1, this._y1 = y;
      }
    };

    function basisClosed$1(context) {
      return new BasisClosed(context);
    }

    function BasisOpen(context) {
      this._context = context;
    }

    BasisOpen.prototype = {
      areaStart: function() {
        this._line = 0;
      },
      areaEnd: function() {
        this._line = NaN;
      },
      lineStart: function() {
        this._x0 = this._x1 =
        this._y0 = this._y1 = NaN;
        this._point = 0;
      },
      lineEnd: function() {
        if (this._line || (this._line !== 0 && this._point === 3)) this._context.closePath();
        this._line = 1 - this._line;
      },
      point: function(x, y) {
        x = +x, y = +y;
        switch (this._point) {
          case 0: this._point = 1; break;
          case 1: this._point = 2; break;
          case 2: this._point = 3; var x0 = (this._x0 + 4 * this._x1 + x) / 6, y0 = (this._y0 + 4 * this._y1 + y) / 6; this._line ? this._context.lineTo(x0, y0) : this._context.moveTo(x0, y0); break;
          case 3: this._point = 4; // proceed
          default: point$2(this, x, y); break;
        }
        this._x0 = this._x1, this._x1 = x;
        this._y0 = this._y1, this._y1 = y;
      }
    };

    function basisOpen(context) {
      return new BasisOpen(context);
    }

    function Bundle(context, beta) {
      this._basis = new Basis(context);
      this._beta = beta;
    }

    Bundle.prototype = {
      lineStart: function() {
        this._x = [];
        this._y = [];
        this._basis.lineStart();
      },
      lineEnd: function() {
        var x = this._x,
            y = this._y,
            j = x.length - 1;

        if (j > 0) {
          var x0 = x[0],
              y0 = y[0],
              dx = x[j] - x0,
              dy = y[j] - y0,
              i = -1,
              t;

          while (++i <= j) {
            t = i / j;
            this._basis.point(
              this._beta * x[i] + (1 - this._beta) * (x0 + t * dx),
              this._beta * y[i] + (1 - this._beta) * (y0 + t * dy)
            );
          }
        }

        this._x = this._y = null;
        this._basis.lineEnd();
      },
      point: function(x, y) {
        this._x.push(+x);
        this._y.push(+y);
      }
    };

    var bundle = (function custom(beta) {

      function bundle(context) {
        return beta === 1 ? new Basis(context) : new Bundle(context, beta);
      }

      bundle.beta = function(beta) {
        return custom(+beta);
      };

      return bundle;
    })(0.85);

    function point$3(that, x, y) {
      that._context.bezierCurveTo(
        that._x1 + that._k * (that._x2 - that._x0),
        that._y1 + that._k * (that._y2 - that._y0),
        that._x2 + that._k * (that._x1 - x),
        that._y2 + that._k * (that._y1 - y),
        that._x2,
        that._y2
      );
    }

    function Cardinal(context, tension) {
      this._context = context;
      this._k = (1 - tension) / 6;
    }

    Cardinal.prototype = {
      areaStart: function() {
        this._line = 0;
      },
      areaEnd: function() {
        this._line = NaN;
      },
      lineStart: function() {
        this._x0 = this._x1 = this._x2 =
        this._y0 = this._y1 = this._y2 = NaN;
        this._point = 0;
      },
      lineEnd: function() {
        switch (this._point) {
          case 2: this._context.lineTo(this._x2, this._y2); break;
          case 3: point$3(this, this._x1, this._y1); break;
        }
        if (this._line || (this._line !== 0 && this._point === 1)) this._context.closePath();
        this._line = 1 - this._line;
      },
      point: function(x, y) {
        x = +x, y = +y;
        switch (this._point) {
          case 0: this._point = 1; this._line ? this._context.lineTo(x, y) : this._context.moveTo(x, y); break;
          case 1: this._point = 2; this._x1 = x, this._y1 = y; break;
          case 2: this._point = 3; // proceed
          default: point$3(this, x, y); break;
        }
        this._x0 = this._x1, this._x1 = this._x2, this._x2 = x;
        this._y0 = this._y1, this._y1 = this._y2, this._y2 = y;
      }
    };

    var cardinal = (function custom(tension) {

      function cardinal(context) {
        return new Cardinal(context, tension);
      }

      cardinal.tension = function(tension) {
        return custom(+tension);
      };

      return cardinal;
    })(0);

    function CardinalClosed(context, tension) {
      this._context = context;
      this._k = (1 - tension) / 6;
    }

    CardinalClosed.prototype = {
      areaStart: noop$3,
      areaEnd: noop$3,
      lineStart: function() {
        this._x0 = this._x1 = this._x2 = this._x3 = this._x4 = this._x5 =
        this._y0 = this._y1 = this._y2 = this._y3 = this._y4 = this._y5 = NaN;
        this._point = 0;
      },
      lineEnd: function() {
        switch (this._point) {
          case 1: {
            this._context.moveTo(this._x3, this._y3);
            this._context.closePath();
            break;
          }
          case 2: {
            this._context.lineTo(this._x3, this._y3);
            this._context.closePath();
            break;
          }
          case 3: {
            this.point(this._x3, this._y3);
            this.point(this._x4, this._y4);
            this.point(this._x5, this._y5);
            break;
          }
        }
      },
      point: function(x, y) {
        x = +x, y = +y;
        switch (this._point) {
          case 0: this._point = 1; this._x3 = x, this._y3 = y; break;
          case 1: this._point = 2; this._context.moveTo(this._x4 = x, this._y4 = y); break;
          case 2: this._point = 3; this._x5 = x, this._y5 = y; break;
          default: point$3(this, x, y); break;
        }
        this._x0 = this._x1, this._x1 = this._x2, this._x2 = x;
        this._y0 = this._y1, this._y1 = this._y2, this._y2 = y;
      }
    };

    var cardinalClosed = (function custom(tension) {

      function cardinal(context) {
        return new CardinalClosed(context, tension);
      }

      cardinal.tension = function(tension) {
        return custom(+tension);
      };

      return cardinal;
    })(0);

    function CardinalOpen(context, tension) {
      this._context = context;
      this._k = (1 - tension) / 6;
    }

    CardinalOpen.prototype = {
      areaStart: function() {
        this._line = 0;
      },
      areaEnd: function() {
        this._line = NaN;
      },
      lineStart: function() {
        this._x0 = this._x1 = this._x2 =
        this._y0 = this._y1 = this._y2 = NaN;
        this._point = 0;
      },
      lineEnd: function() {
        if (this._line || (this._line !== 0 && this._point === 3)) this._context.closePath();
        this._line = 1 - this._line;
      },
      point: function(x, y) {
        x = +x, y = +y;
        switch (this._point) {
          case 0: this._point = 1; break;
          case 1: this._point = 2; break;
          case 2: this._point = 3; this._line ? this._context.lineTo(this._x2, this._y2) : this._context.moveTo(this._x2, this._y2); break;
          case 3: this._point = 4; // proceed
          default: point$3(this, x, y); break;
        }
        this._x0 = this._x1, this._x1 = this._x2, this._x2 = x;
        this._y0 = this._y1, this._y1 = this._y2, this._y2 = y;
      }
    };

    var cardinalOpen = (function custom(tension) {

      function cardinal(context) {
        return new CardinalOpen(context, tension);
      }

      cardinal.tension = function(tension) {
        return custom(+tension);
      };

      return cardinal;
    })(0);

    function point$4(that, x, y) {
      var x1 = that._x1,
          y1 = that._y1,
          x2 = that._x2,
          y2 = that._y2;

      if (that._l01_a > epsilon$3) {
        var a = 2 * that._l01_2a + 3 * that._l01_a * that._l12_a + that._l12_2a,
            n = 3 * that._l01_a * (that._l01_a + that._l12_a);
        x1 = (x1 * a - that._x0 * that._l12_2a + that._x2 * that._l01_2a) / n;
        y1 = (y1 * a - that._y0 * that._l12_2a + that._y2 * that._l01_2a) / n;
      }

      if (that._l23_a > epsilon$3) {
        var b = 2 * that._l23_2a + 3 * that._l23_a * that._l12_a + that._l12_2a,
            m = 3 * that._l23_a * (that._l23_a + that._l12_a);
        x2 = (x2 * b + that._x1 * that._l23_2a - x * that._l12_2a) / m;
        y2 = (y2 * b + that._y1 * that._l23_2a - y * that._l12_2a) / m;
      }

      that._context.bezierCurveTo(x1, y1, x2, y2, that._x2, that._y2);
    }

    function CatmullRom(context, alpha) {
      this._context = context;
      this._alpha = alpha;
    }

    CatmullRom.prototype = {
      areaStart: function() {
        this._line = 0;
      },
      areaEnd: function() {
        this._line = NaN;
      },
      lineStart: function() {
        this._x0 = this._x1 = this._x2 =
        this._y0 = this._y1 = this._y2 = NaN;
        this._l01_a = this._l12_a = this._l23_a =
        this._l01_2a = this._l12_2a = this._l23_2a =
        this._point = 0;
      },
      lineEnd: function() {
        switch (this._point) {
          case 2: this._context.lineTo(this._x2, this._y2); break;
          case 3: this.point(this._x2, this._y2); break;
        }
        if (this._line || (this._line !== 0 && this._point === 1)) this._context.closePath();
        this._line = 1 - this._line;
      },
      point: function(x, y) {
        x = +x, y = +y;

        if (this._point) {
          var x23 = this._x2 - x,
              y23 = this._y2 - y;
          this._l23_a = Math.sqrt(this._l23_2a = Math.pow(x23 * x23 + y23 * y23, this._alpha));
        }

        switch (this._point) {
          case 0: this._point = 1; this._line ? this._context.lineTo(x, y) : this._context.moveTo(x, y); break;
          case 1: this._point = 2; break;
          case 2: this._point = 3; // proceed
          default: point$4(this, x, y); break;
        }

        this._l01_a = this._l12_a, this._l12_a = this._l23_a;
        this._l01_2a = this._l12_2a, this._l12_2a = this._l23_2a;
        this._x0 = this._x1, this._x1 = this._x2, this._x2 = x;
        this._y0 = this._y1, this._y1 = this._y2, this._y2 = y;
      }
    };

    var catmullRom = (function custom(alpha) {

      function catmullRom(context) {
        return alpha ? new CatmullRom(context, alpha) : new Cardinal(context, 0);
      }

      catmullRom.alpha = function(alpha) {
        return custom(+alpha);
      };

      return catmullRom;
    })(0.5);

    function CatmullRomClosed(context, alpha) {
      this._context = context;
      this._alpha = alpha;
    }

    CatmullRomClosed.prototype = {
      areaStart: noop$3,
      areaEnd: noop$3,
      lineStart: function() {
        this._x0 = this._x1 = this._x2 = this._x3 = this._x4 = this._x5 =
        this._y0 = this._y1 = this._y2 = this._y3 = this._y4 = this._y5 = NaN;
        this._l01_a = this._l12_a = this._l23_a =
        this._l01_2a = this._l12_2a = this._l23_2a =
        this._point = 0;
      },
      lineEnd: function() {
        switch (this._point) {
          case 1: {
            this._context.moveTo(this._x3, this._y3);
            this._context.closePath();
            break;
          }
          case 2: {
            this._context.lineTo(this._x3, this._y3);
            this._context.closePath();
            break;
          }
          case 3: {
            this.point(this._x3, this._y3);
            this.point(this._x4, this._y4);
            this.point(this._x5, this._y5);
            break;
          }
        }
      },
      point: function(x, y) {
        x = +x, y = +y;

        if (this._point) {
          var x23 = this._x2 - x,
              y23 = this._y2 - y;
          this._l23_a = Math.sqrt(this._l23_2a = Math.pow(x23 * x23 + y23 * y23, this._alpha));
        }

        switch (this._point) {
          case 0: this._point = 1; this._x3 = x, this._y3 = y; break;
          case 1: this._point = 2; this._context.moveTo(this._x4 = x, this._y4 = y); break;
          case 2: this._point = 3; this._x5 = x, this._y5 = y; break;
          default: point$4(this, x, y); break;
        }

        this._l01_a = this._l12_a, this._l12_a = this._l23_a;
        this._l01_2a = this._l12_2a, this._l12_2a = this._l23_2a;
        this._x0 = this._x1, this._x1 = this._x2, this._x2 = x;
        this._y0 = this._y1, this._y1 = this._y2, this._y2 = y;
      }
    };

    var catmullRomClosed = (function custom(alpha) {

      function catmullRom(context) {
        return alpha ? new CatmullRomClosed(context, alpha) : new CardinalClosed(context, 0);
      }

      catmullRom.alpha = function(alpha) {
        return custom(+alpha);
      };

      return catmullRom;
    })(0.5);

    function CatmullRomOpen(context, alpha) {
      this._context = context;
      this._alpha = alpha;
    }

    CatmullRomOpen.prototype = {
      areaStart: function() {
        this._line = 0;
      },
      areaEnd: function() {
        this._line = NaN;
      },
      lineStart: function() {
        this._x0 = this._x1 = this._x2 =
        this._y0 = this._y1 = this._y2 = NaN;
        this._l01_a = this._l12_a = this._l23_a =
        this._l01_2a = this._l12_2a = this._l23_2a =
        this._point = 0;
      },
      lineEnd: function() {
        if (this._line || (this._line !== 0 && this._point === 3)) this._context.closePath();
        this._line = 1 - this._line;
      },
      point: function(x, y) {
        x = +x, y = +y;

        if (this._point) {
          var x23 = this._x2 - x,
              y23 = this._y2 - y;
          this._l23_a = Math.sqrt(this._l23_2a = Math.pow(x23 * x23 + y23 * y23, this._alpha));
        }

        switch (this._point) {
          case 0: this._point = 1; break;
          case 1: this._point = 2; break;
          case 2: this._point = 3; this._line ? this._context.lineTo(this._x2, this._y2) : this._context.moveTo(this._x2, this._y2); break;
          case 3: this._point = 4; // proceed
          default: point$4(this, x, y); break;
        }

        this._l01_a = this._l12_a, this._l12_a = this._l23_a;
        this._l01_2a = this._l12_2a, this._l12_2a = this._l23_2a;
        this._x0 = this._x1, this._x1 = this._x2, this._x2 = x;
        this._y0 = this._y1, this._y1 = this._y2, this._y2 = y;
      }
    };

    var catmullRomOpen = (function custom(alpha) {

      function catmullRom(context) {
        return alpha ? new CatmullRomOpen(context, alpha) : new CardinalOpen(context, 0);
      }

      catmullRom.alpha = function(alpha) {
        return custom(+alpha);
      };

      return catmullRom;
    })(0.5);

    function LinearClosed(context) {
      this._context = context;
    }

    LinearClosed.prototype = {
      areaStart: noop$3,
      areaEnd: noop$3,
      lineStart: function() {
        this._point = 0;
      },
      lineEnd: function() {
        if (this._point) this._context.closePath();
      },
      point: function(x, y) {
        x = +x, y = +y;
        if (this._point) this._context.lineTo(x, y);
        else this._point = 1, this._context.moveTo(x, y);
      }
    };

    function linearClosed(context) {
      return new LinearClosed(context);
    }

    function sign$1(x) {
      return x < 0 ? -1 : 1;
    }

    // Calculate the slopes of the tangents (Hermite-type interpolation) based on
    // the following paper: Steffen, M. 1990. A Simple Method for Monotonic
    // Interpolation in One Dimension. Astronomy and Astrophysics, Vol. 239, NO.
    // NOV(II), P. 443, 1990.
    function slope3(that, x2, y2) {
      var h0 = that._x1 - that._x0,
          h1 = x2 - that._x1,
          s0 = (that._y1 - that._y0) / (h0 || h1 < 0 && -0),
          s1 = (y2 - that._y1) / (h1 || h0 < 0 && -0),
          p = (s0 * h1 + s1 * h0) / (h0 + h1);
      return (sign$1(s0) + sign$1(s1)) * Math.min(Math.abs(s0), Math.abs(s1), 0.5 * Math.abs(p)) || 0;
    }

    // Calculate a one-sided slope.
    function slope2(that, t) {
      var h = that._x1 - that._x0;
      return h ? (3 * (that._y1 - that._y0) / h - t) / 2 : t;
    }

    // According to https://en.wikipedia.org/wiki/Cubic_Hermite_spline#Representations
    // "you can express cubic Hermite interpolation in terms of cubic Bézier curves
    // with respect to the four values p0, p0 + m0 / 3, p1 - m1 / 3, p1".
    function point$5(that, t0, t1) {
      var x0 = that._x0,
          y0 = that._y0,
          x1 = that._x1,
          y1 = that._y1,
          dx = (x1 - x0) / 3;
      that._context.bezierCurveTo(x0 + dx, y0 + dx * t0, x1 - dx, y1 - dx * t1, x1, y1);
    }

    function MonotoneX(context) {
      this._context = context;
    }

    MonotoneX.prototype = {
      areaStart: function() {
        this._line = 0;
      },
      areaEnd: function() {
        this._line = NaN;
      },
      lineStart: function() {
        this._x0 = this._x1 =
        this._y0 = this._y1 =
        this._t0 = NaN;
        this._point = 0;
      },
      lineEnd: function() {
        switch (this._point) {
          case 2: this._context.lineTo(this._x1, this._y1); break;
          case 3: point$5(this, this._t0, slope2(this, this._t0)); break;
        }
        if (this._line || (this._line !== 0 && this._point === 1)) this._context.closePath();
        this._line = 1 - this._line;
      },
      point: function(x, y) {
        var t1 = NaN;

        x = +x, y = +y;
        if (x === this._x1 && y === this._y1) return; // Ignore coincident points.
        switch (this._point) {
          case 0: this._point = 1; this._line ? this._context.lineTo(x, y) : this._context.moveTo(x, y); break;
          case 1: this._point = 2; break;
          case 2: this._point = 3; point$5(this, slope2(this, t1 = slope3(this, x, y)), t1); break;
          default: point$5(this, this._t0, t1 = slope3(this, x, y)); break;
        }

        this._x0 = this._x1, this._x1 = x;
        this._y0 = this._y1, this._y1 = y;
        this._t0 = t1;
      }
    };

    function MonotoneY(context) {
      this._context = new ReflectContext(context);
    }

    (MonotoneY.prototype = Object.create(MonotoneX.prototype)).point = function(x, y) {
      MonotoneX.prototype.point.call(this, y, x);
    };

    function ReflectContext(context) {
      this._context = context;
    }

    ReflectContext.prototype = {
      moveTo: function(x, y) { this._context.moveTo(y, x); },
      closePath: function() { this._context.closePath(); },
      lineTo: function(x, y) { this._context.lineTo(y, x); },
      bezierCurveTo: function(x1, y1, x2, y2, x, y) { this._context.bezierCurveTo(y1, x1, y2, x2, y, x); }
    };

    function monotoneX(context) {
      return new MonotoneX(context);
    }

    function monotoneY(context) {
      return new MonotoneY(context);
    }

    function Natural(context) {
      this._context = context;
    }

    Natural.prototype = {
      areaStart: function() {
        this._line = 0;
      },
      areaEnd: function() {
        this._line = NaN;
      },
      lineStart: function() {
        this._x = [];
        this._y = [];
      },
      lineEnd: function() {
        var x = this._x,
            y = this._y,
            n = x.length;

        if (n) {
          this._line ? this._context.lineTo(x[0], y[0]) : this._context.moveTo(x[0], y[0]);
          if (n === 2) {
            this._context.lineTo(x[1], y[1]);
          } else {
            var px = controlPoints(x),
                py = controlPoints(y);
            for (var i0 = 0, i1 = 1; i1 < n; ++i0, ++i1) {
              this._context.bezierCurveTo(px[0][i0], py[0][i0], px[1][i0], py[1][i0], x[i1], y[i1]);
            }
          }
        }

        if (this._line || (this._line !== 0 && n === 1)) this._context.closePath();
        this._line = 1 - this._line;
        this._x = this._y = null;
      },
      point: function(x, y) {
        this._x.push(+x);
        this._y.push(+y);
      }
    };

    // See https://www.particleincell.com/2012/bezier-splines/ for derivation.
    function controlPoints(x) {
      var i,
          n = x.length - 1,
          m,
          a = new Array(n),
          b = new Array(n),
          r = new Array(n);
      a[0] = 0, b[0] = 2, r[0] = x[0] + 2 * x[1];
      for (i = 1; i < n - 1; ++i) a[i] = 1, b[i] = 4, r[i] = 4 * x[i] + 2 * x[i + 1];
      a[n - 1] = 2, b[n - 1] = 7, r[n - 1] = 8 * x[n - 1] + x[n];
      for (i = 1; i < n; ++i) m = a[i] / b[i - 1], b[i] -= m, r[i] -= m * r[i - 1];
      a[n - 1] = r[n - 1] / b[n - 1];
      for (i = n - 2; i >= 0; --i) a[i] = (r[i] - a[i + 1]) / b[i];
      b[n - 1] = (x[n] + a[n - 1]) / 2;
      for (i = 0; i < n - 1; ++i) b[i] = 2 * x[i + 1] - a[i + 1];
      return [a, b];
    }

    function natural(context) {
      return new Natural(context);
    }

    function Step(context, t) {
      this._context = context;
      this._t = t;
    }

    Step.prototype = {
      areaStart: function() {
        this._line = 0;
      },
      areaEnd: function() {
        this._line = NaN;
      },
      lineStart: function() {
        this._x = this._y = NaN;
        this._point = 0;
      },
      lineEnd: function() {
        if (0 < this._t && this._t < 1 && this._point === 2) this._context.lineTo(this._x, this._y);
        if (this._line || (this._line !== 0 && this._point === 1)) this._context.closePath();
        if (this._line >= 0) this._t = 1 - this._t, this._line = 1 - this._line;
      },
      point: function(x, y) {
        x = +x, y = +y;
        switch (this._point) {
          case 0: this._point = 1; this._line ? this._context.lineTo(x, y) : this._context.moveTo(x, y); break;
          case 1: this._point = 2; // proceed
          default: {
            if (this._t <= 0) {
              this._context.lineTo(this._x, y);
              this._context.lineTo(x, y);
            } else {
              var x1 = this._x * (1 - this._t) + x * this._t;
              this._context.lineTo(x1, this._y);
              this._context.lineTo(x1, y);
            }
            break;
          }
        }
        this._x = x, this._y = y;
      }
    };

    function step(context) {
      return new Step(context, 0.5);
    }

    function stepBefore(context) {
      return new Step(context, 0);
    }

    function stepAfter(context) {
      return new Step(context, 1);
    }

    function none$1(series, order) {
      if (!((n = series.length) > 1)) return;
      for (var i = 1, j, s0, s1 = series[order[0]], n, m = s1.length; i < n; ++i) {
        s0 = s1, s1 = series[order[i]];
        for (j = 0; j < m; ++j) {
          s1[j][1] += s1[j][0] = isNaN(s0[j][1]) ? s0[j][0] : s0[j][1];
        }
      }
    }

    function none$2(series) {
      var n = series.length, o = new Array(n);
      while (--n >= 0) o[n] = n;
      return o;
    }

    function stackValue(d, key) {
      return d[key];
    }

    function stack() {
      var keys = constant$b([]),
          order = none$2,
          offset = none$1,
          value = stackValue;

      function stack(data) {
        var kz = keys.apply(this, arguments),
            i,
            m = data.length,
            n = kz.length,
            sz = new Array(n),
            oz;

        for (i = 0; i < n; ++i) {
          for (var ki = kz[i], si = sz[i] = new Array(m), j = 0, sij; j < m; ++j) {
            si[j] = sij = [0, +value(data[j], ki, j, data)];
            sij.data = data[j];
          }
          si.key = ki;
        }

        for (i = 0, oz = order(sz); i < n; ++i) {
          sz[oz[i]].index = i;
        }

        offset(sz, oz);
        return sz;
      }

      stack.keys = function(_) {
        return arguments.length ? (keys = typeof _ === "function" ? _ : constant$b(slice$6.call(_)), stack) : keys;
      };

      stack.value = function(_) {
        return arguments.length ? (value = typeof _ === "function" ? _ : constant$b(+_), stack) : value;
      };

      stack.order = function(_) {
        return arguments.length ? (order = _ == null ? none$2 : typeof _ === "function" ? _ : constant$b(slice$6.call(_)), stack) : order;
      };

      stack.offset = function(_) {
        return arguments.length ? (offset = _ == null ? none$1 : _, stack) : offset;
      };

      return stack;
    }

    function expand(series, order) {
      if (!((n = series.length) > 0)) return;
      for (var i, n, j = 0, m = series[0].length, y; j < m; ++j) {
        for (y = i = 0; i < n; ++i) y += series[i][j][1] || 0;
        if (y) for (i = 0; i < n; ++i) series[i][j][1] /= y;
      }
      none$1(series, order);
    }

    function diverging$1(series, order) {
      if (!((n = series.length) > 0)) return;
      for (var i, j = 0, d, dy, yp, yn, n, m = series[order[0]].length; j < m; ++j) {
        for (yp = yn = 0, i = 0; i < n; ++i) {
          if ((dy = (d = series[order[i]][j])[1] - d[0]) > 0) {
            d[0] = yp, d[1] = yp += dy;
          } else if (dy < 0) {
            d[1] = yn, d[0] = yn += dy;
          } else {
            d[0] = 0, d[1] = dy;
          }
        }
      }
    }

    function silhouette(series, order) {
      if (!((n = series.length) > 0)) return;
      for (var j = 0, s0 = series[order[0]], n, m = s0.length; j < m; ++j) {
        for (var i = 0, y = 0; i < n; ++i) y += series[i][j][1] || 0;
        s0[j][1] += s0[j][0] = -y / 2;
      }
      none$1(series, order);
    }

    function wiggle(series, order) {
      if (!((n = series.length) > 0) || !((m = (s0 = series[order[0]]).length) > 0)) return;
      for (var y = 0, j = 1, s0, m, n; j < m; ++j) {
        for (var i = 0, s1 = 0, s2 = 0; i < n; ++i) {
          var si = series[order[i]],
              sij0 = si[j][1] || 0,
              sij1 = si[j - 1][1] || 0,
              s3 = (sij0 - sij1) / 2;
          for (var k = 0; k < i; ++k) {
            var sk = series[order[k]],
                skj0 = sk[j][1] || 0,
                skj1 = sk[j - 1][1] || 0;
            s3 += skj0 - skj1;
          }
          s1 += sij0, s2 += s3 * sij0;
        }
        s0[j - 1][1] += s0[j - 1][0] = y;
        if (s1) y -= s2 / s1;
      }
      s0[j - 1][1] += s0[j - 1][0] = y;
      none$1(series, order);
    }

    function appearance(series) {
      var peaks = series.map(peak);
      return none$2(series).sort(function(a, b) { return peaks[a] - peaks[b]; });
    }

    function peak(series) {
      var i = -1, j = 0, n = series.length, vi, vj = -Infinity;
      while (++i < n) if ((vi = +series[i][1]) > vj) vj = vi, j = i;
      return j;
    }

    function ascending$3(series) {
      var sums = series.map(sum$2);
      return none$2(series).sort(function(a, b) { return sums[a] - sums[b]; });
    }

    function sum$2(series) {
      var s = 0, i = -1, n = series.length, v;
      while (++i < n) if (v = +series[i][1]) s += v;
      return s;
    }

    function descending$2(series) {
      return ascending$3(series).reverse();
    }

    function insideOut(series) {
      var n = series.length,
          i,
          j,
          sums = series.map(sum$2),
          order = appearance(series),
          top = 0,
          bottom = 0,
          tops = [],
          bottoms = [];

      for (i = 0; i < n; ++i) {
        j = order[i];
        if (top < bottom) {
          top += sums[j];
          tops.push(j);
        } else {
          bottom += sums[j];
          bottoms.push(j);
        }
      }

      return bottoms.reverse().concat(tops);
    }

    function reverse(series) {
      return none$2(series).reverse();
    }

    function constant$c(x) {
      return function() {
        return x;
      };
    }

    function x$4(d) {
      return d[0];
    }

    function y$4(d) {
      return d[1];
    }

    function RedBlackTree() {
      this._ = null; // root node
    }

    function RedBlackNode(node) {
      node.U = // parent node
      node.C = // color - true for red, false for black
      node.L = // left node
      node.R = // right node
      node.P = // previous node
      node.N = null; // next node
    }

    RedBlackTree.prototype = {
      constructor: RedBlackTree,

      insert: function(after, node) {
        var parent, grandpa, uncle;

        if (after) {
          node.P = after;
          node.N = after.N;
          if (after.N) after.N.P = node;
          after.N = node;
          if (after.R) {
            after = after.R;
            while (after.L) after = after.L;
            after.L = node;
          } else {
            after.R = node;
          }
          parent = after;
        } else if (this._) {
          after = RedBlackFirst(this._);
          node.P = null;
          node.N = after;
          after.P = after.L = node;
          parent = after;
        } else {
          node.P = node.N = null;
          this._ = node;
          parent = null;
        }
        node.L = node.R = null;
        node.U = parent;
        node.C = true;

        after = node;
        while (parent && parent.C) {
          grandpa = parent.U;
          if (parent === grandpa.L) {
            uncle = grandpa.R;
            if (uncle && uncle.C) {
              parent.C = uncle.C = false;
              grandpa.C = true;
              after = grandpa;
            } else {
              if (after === parent.R) {
                RedBlackRotateLeft(this, parent);
                after = parent;
                parent = after.U;
              }
              parent.C = false;
              grandpa.C = true;
              RedBlackRotateRight(this, grandpa);
            }
          } else {
            uncle = grandpa.L;
            if (uncle && uncle.C) {
              parent.C = uncle.C = false;
              grandpa.C = true;
              after = grandpa;
            } else {
              if (after === parent.L) {
                RedBlackRotateRight(this, parent);
                after = parent;
                parent = after.U;
              }
              parent.C = false;
              grandpa.C = true;
              RedBlackRotateLeft(this, grandpa);
            }
          }
          parent = after.U;
        }
        this._.C = false;
      },

      remove: function(node) {
        if (node.N) node.N.P = node.P;
        if (node.P) node.P.N = node.N;
        node.N = node.P = null;

        var parent = node.U,
            sibling,
            left = node.L,
            right = node.R,
            next,
            red;

        if (!left) next = right;
        else if (!right) next = left;
        else next = RedBlackFirst(right);

        if (parent) {
          if (parent.L === node) parent.L = next;
          else parent.R = next;
        } else {
          this._ = next;
        }

        if (left && right) {
          red = next.C;
          next.C = node.C;
          next.L = left;
          left.U = next;
          if (next !== right) {
            parent = next.U;
            next.U = node.U;
            node = next.R;
            parent.L = node;
            next.R = right;
            right.U = next;
          } else {
            next.U = parent;
            parent = next;
            node = next.R;
          }
        } else {
          red = node.C;
          node = next;
        }

        if (node) node.U = parent;
        if (red) return;
        if (node && node.C) { node.C = false; return; }

        do {
          if (node === this._) break;
          if (node === parent.L) {
            sibling = parent.R;
            if (sibling.C) {
              sibling.C = false;
              parent.C = true;
              RedBlackRotateLeft(this, parent);
              sibling = parent.R;
            }
            if ((sibling.L && sibling.L.C)
                || (sibling.R && sibling.R.C)) {
              if (!sibling.R || !sibling.R.C) {
                sibling.L.C = false;
                sibling.C = true;
                RedBlackRotateRight(this, sibling);
                sibling = parent.R;
              }
              sibling.C = parent.C;
              parent.C = sibling.R.C = false;
              RedBlackRotateLeft(this, parent);
              node = this._;
              break;
            }
          } else {
            sibling = parent.L;
            if (sibling.C) {
              sibling.C = false;
              parent.C = true;
              RedBlackRotateRight(this, parent);
              sibling = parent.L;
            }
            if ((sibling.L && sibling.L.C)
              || (sibling.R && sibling.R.C)) {
              if (!sibling.L || !sibling.L.C) {
                sibling.R.C = false;
                sibling.C = true;
                RedBlackRotateLeft(this, sibling);
                sibling = parent.L;
              }
              sibling.C = parent.C;
              parent.C = sibling.L.C = false;
              RedBlackRotateRight(this, parent);
              node = this._;
              break;
            }
          }
          sibling.C = true;
          node = parent;
          parent = parent.U;
        } while (!node.C);

        if (node) node.C = false;
      }
    };

    function RedBlackRotateLeft(tree, node) {
      var p = node,
          q = node.R,
          parent = p.U;

      if (parent) {
        if (parent.L === p) parent.L = q;
        else parent.R = q;
      } else {
        tree._ = q;
      }

      q.U = parent;
      p.U = q;
      p.R = q.L;
      if (p.R) p.R.U = p;
      q.L = p;
    }

    function RedBlackRotateRight(tree, node) {
      var p = node,
          q = node.L,
          parent = p.U;

      if (parent) {
        if (parent.L === p) parent.L = q;
        else parent.R = q;
      } else {
        tree._ = q;
      }

      q.U = parent;
      p.U = q;
      p.L = q.R;
      if (p.L) p.L.U = p;
      q.R = p;
    }

    function RedBlackFirst(node) {
      while (node.L) node = node.L;
      return node;
    }

    function createEdge(left, right, v0, v1) {
      var edge = [null, null],
          index = edges.push(edge) - 1;
      edge.left = left;
      edge.right = right;
      if (v0) setEdgeEnd(edge, left, right, v0);
      if (v1) setEdgeEnd(edge, right, left, v1);
      cells[left.index].halfedges.push(index);
      cells[right.index].halfedges.push(index);
      return edge;
    }

    function createBorderEdge(left, v0, v1) {
      var edge = [v0, v1];
      edge.left = left;
      return edge;
    }

    function setEdgeEnd(edge, left, right, vertex) {
      if (!edge[0] && !edge[1]) {
        edge[0] = vertex;
        edge.left = left;
        edge.right = right;
      } else if (edge.left === right) {
        edge[1] = vertex;
      } else {
        edge[0] = vertex;
      }
    }

    // Liang–Barsky line clipping.
    function clipEdge(edge, x0, y0, x1, y1) {
      var a = edge[0],
          b = edge[1],
          ax = a[0],
          ay = a[1],
          bx = b[0],
          by = b[1],
          t0 = 0,
          t1 = 1,
          dx = bx - ax,
          dy = by - ay,
          r;

      r = x0 - ax;
      if (!dx && r > 0) return;
      r /= dx;
      if (dx < 0) {
        if (r < t0) return;
        if (r < t1) t1 = r;
      } else if (dx > 0) {
        if (r > t1) return;
        if (r > t0) t0 = r;
      }

      r = x1 - ax;
      if (!dx && r < 0) return;
      r /= dx;
      if (dx < 0) {
        if (r > t1) return;
        if (r > t0) t0 = r;
      } else if (dx > 0) {
        if (r < t0) return;
        if (r < t1) t1 = r;
      }

      r = y0 - ay;
      if (!dy && r > 0) return;
      r /= dy;
      if (dy < 0) {
        if (r < t0) return;
        if (r < t1) t1 = r;
      } else if (dy > 0) {
        if (r > t1) return;
        if (r > t0) t0 = r;
      }

      r = y1 - ay;
      if (!dy && r < 0) return;
      r /= dy;
      if (dy < 0) {
        if (r > t1) return;
        if (r > t0) t0 = r;
      } else if (dy > 0) {
        if (r < t0) return;
        if (r < t1) t1 = r;
      }

      if (!(t0 > 0) && !(t1 < 1)) return true; // TODO Better check?

      if (t0 > 0) edge[0] = [ax + t0 * dx, ay + t0 * dy];
      if (t1 < 1) edge[1] = [ax + t1 * dx, ay + t1 * dy];
      return true;
    }

    function connectEdge(edge, x0, y0, x1, y1) {
      var v1 = edge[1];
      if (v1) return true;

      var v0 = edge[0],
          left = edge.left,
          right = edge.right,
          lx = left[0],
          ly = left[1],
          rx = right[0],
          ry = right[1],
          fx = (lx + rx) / 2,
          fy = (ly + ry) / 2,
          fm,
          fb;

      if (ry === ly) {
        if (fx < x0 || fx >= x1) return;
        if (lx > rx) {
          if (!v0) v0 = [fx, y0];
          else if (v0[1] >= y1) return;
          v1 = [fx, y1];
        } else {
          if (!v0) v0 = [fx, y1];
          else if (v0[1] < y0) return;
          v1 = [fx, y0];
        }
      } else {
        fm = (lx - rx) / (ry - ly);
        fb = fy - fm * fx;
        if (fm < -1 || fm > 1) {
          if (lx > rx) {
            if (!v0) v0 = [(y0 - fb) / fm, y0];
            else if (v0[1] >= y1) return;
            v1 = [(y1 - fb) / fm, y1];
          } else {
            if (!v0) v0 = [(y1 - fb) / fm, y1];
            else if (v0[1] < y0) return;
            v1 = [(y0 - fb) / fm, y0];
          }
        } else {
          if (ly < ry) {
            if (!v0) v0 = [x0, fm * x0 + fb];
            else if (v0[0] >= x1) return;
            v1 = [x1, fm * x1 + fb];
          } else {
            if (!v0) v0 = [x1, fm * x1 + fb];
            else if (v0[0] < x0) return;
            v1 = [x0, fm * x0 + fb];
          }
        }
      }

      edge[0] = v0;
      edge[1] = v1;
      return true;
    }

    function clipEdges(x0, y0, x1, y1) {
      var i = edges.length,
          edge;

      while (i--) {
        if (!connectEdge(edge = edges[i], x0, y0, x1, y1)
            || !clipEdge(edge, x0, y0, x1, y1)
            || !(Math.abs(edge[0][0] - edge[1][0]) > epsilon$4
                || Math.abs(edge[0][1] - edge[1][1]) > epsilon$4)) {
          delete edges[i];
        }
      }
    }

    function createCell(site) {
      return cells[site.index] = {
        site: site,
        halfedges: []
      };
    }

    function cellHalfedgeAngle(cell, edge) {
      var site = cell.site,
          va = edge.left,
          vb = edge.right;
      if (site === vb) vb = va, va = site;
      if (vb) return Math.atan2(vb[1] - va[1], vb[0] - va[0]);
      if (site === va) va = edge[1], vb = edge[0];
      else va = edge[0], vb = edge[1];
      return Math.atan2(va[0] - vb[0], vb[1] - va[1]);
    }

    function cellHalfedgeStart(cell, edge) {
      return edge[+(edge.left !== cell.site)];
    }

    function cellHalfedgeEnd(cell, edge) {
      return edge[+(edge.left === cell.site)];
    }

    function sortCellHalfedges() {
      for (var i = 0, n = cells.length, cell, halfedges, j, m; i < n; ++i) {
        if ((cell = cells[i]) && (m = (halfedges = cell.halfedges).length)) {
          var index = new Array(m),
              array = new Array(m);
          for (j = 0; j < m; ++j) index[j] = j, array[j] = cellHalfedgeAngle(cell, edges[halfedges[j]]);
          index.sort(function(i, j) { return array[j] - array[i]; });
          for (j = 0; j < m; ++j) array[j] = halfedges[index[j]];
          for (j = 0; j < m; ++j) halfedges[j] = array[j];
        }
      }
    }

    function clipCells(x0, y0, x1, y1) {
      var nCells = cells.length,
          iCell,
          cell,
          site,
          iHalfedge,
          halfedges,
          nHalfedges,
          start,
          startX,
          startY,
          end,
          endX,
          endY,
          cover = true;

      for (iCell = 0; iCell < nCells; ++iCell) {
        if (cell = cells[iCell]) {
          site = cell.site;
          halfedges = cell.halfedges;
          iHalfedge = halfedges.length;

          // Remove any dangling clipped edges.
          while (iHalfedge--) {
            if (!edges[halfedges[iHalfedge]]) {
              halfedges.splice(iHalfedge, 1);
            }
          }

          // Insert any border edges as necessary.
          iHalfedge = 0, nHalfedges = halfedges.length;
          while (iHalfedge < nHalfedges) {
            end = cellHalfedgeEnd(cell, edges[halfedges[iHalfedge]]), endX = end[0], endY = end[1];
            start = cellHalfedgeStart(cell, edges[halfedges[++iHalfedge % nHalfedges]]), startX = start[0], startY = start[1];
            if (Math.abs(endX - startX) > epsilon$4 || Math.abs(endY - startY) > epsilon$4) {
              halfedges.splice(iHalfedge, 0, edges.push(createBorderEdge(site, end,
                  Math.abs(endX - x0) < epsilon$4 && y1 - endY > epsilon$4 ? [x0, Math.abs(startX - x0) < epsilon$4 ? startY : y1]
                  : Math.abs(endY - y1) < epsilon$4 && x1 - endX > epsilon$4 ? [Math.abs(startY - y1) < epsilon$4 ? startX : x1, y1]
                  : Math.abs(endX - x1) < epsilon$4 && endY - y0 > epsilon$4 ? [x1, Math.abs(startX - x1) < epsilon$4 ? startY : y0]
                  : Math.abs(endY - y0) < epsilon$4 && endX - x0 > epsilon$4 ? [Math.abs(startY - y0) < epsilon$4 ? startX : x0, y0]
                  : null)) - 1);
              ++nHalfedges;
            }
          }

          if (nHalfedges) cover = false;
        }
      }

      // If there weren’t any edges, have the closest site cover the extent.
      // It doesn’t matter which corner of the extent we measure!
      if (cover) {
        var dx, dy, d2, dc = Infinity;

        for (iCell = 0, cover = null; iCell < nCells; ++iCell) {
          if (cell = cells[iCell]) {
            site = cell.site;
            dx = site[0] - x0;
            dy = site[1] - y0;
            d2 = dx * dx + dy * dy;
            if (d2 < dc) dc = d2, cover = cell;
          }
        }

        if (cover) {
          var v00 = [x0, y0], v01 = [x0, y1], v11 = [x1, y1], v10 = [x1, y0];
          cover.halfedges.push(
            edges.push(createBorderEdge(site = cover.site, v00, v01)) - 1,
            edges.push(createBorderEdge(site, v01, v11)) - 1,
            edges.push(createBorderEdge(site, v11, v10)) - 1,
            edges.push(createBorderEdge(site, v10, v00)) - 1
          );
        }
      }

      // Lastly delete any cells with no edges; these were entirely clipped.
      for (iCell = 0; iCell < nCells; ++iCell) {
        if (cell = cells[iCell]) {
          if (!cell.halfedges.length) {
            delete cells[iCell];
          }
        }
      }
    }

    var circlePool = [];

    var firstCircle;

    function Circle() {
      RedBlackNode(this);
      this.x =
      this.y =
      this.arc =
      this.site =
      this.cy = null;
    }

    function attachCircle(arc) {
      var lArc = arc.P,
          rArc = arc.N;

      if (!lArc || !rArc) return;

      var lSite = lArc.site,
          cSite = arc.site,
          rSite = rArc.site;

      if (lSite === rSite) return;

      var bx = cSite[0],
          by = cSite[1],
          ax = lSite[0] - bx,
          ay = lSite[1] - by,
          cx = rSite[0] - bx,
          cy = rSite[1] - by;

      var d = 2 * (ax * cy - ay * cx);
      if (d >= -epsilon2$2) return;

      var ha = ax * ax + ay * ay,
          hc = cx * cx + cy * cy,
          x = (cy * ha - ay * hc) / d,
          y = (ax * hc - cx * ha) / d;

      var circle = circlePool.pop() || new Circle;
      circle.arc = arc;
      circle.site = cSite;
      circle.x = x + bx;
      circle.y = (circle.cy = y + by) + Math.sqrt(x * x + y * y); // y bottom

      arc.circle = circle;

      var before = null,
          node = circles._;

      while (node) {
        if (circle.y < node.y || (circle.y === node.y && circle.x <= node.x)) {
          if (node.L) node = node.L;
          else { before = node.P; break; }
        } else {
          if (node.R) node = node.R;
          else { before = node; break; }
        }
      }

      circles.insert(before, circle);
      if (!before) firstCircle = circle;
    }

    function detachCircle(arc) {
      var circle = arc.circle;
      if (circle) {
        if (!circle.P) firstCircle = circle.N;
        circles.remove(circle);
        circlePool.push(circle);
        RedBlackNode(circle);
        arc.circle = null;
      }
    }

    var beachPool = [];

    function Beach() {
      RedBlackNode(this);
      this.edge =
      this.site =
      this.circle = null;
    }

    function createBeach(site) {
      var beach = beachPool.pop() || new Beach;
      beach.site = site;
      return beach;
    }

    function detachBeach(beach) {
      detachCircle(beach);
      beaches.remove(beach);
      beachPool.push(beach);
      RedBlackNode(beach);
    }

    function removeBeach(beach) {
      var circle = beach.circle,
          x = circle.x,
          y = circle.cy,
          vertex = [x, y],
          previous = beach.P,
          next = beach.N,
          disappearing = [beach];

      detachBeach(beach);

      var lArc = previous;
      while (lArc.circle
          && Math.abs(x - lArc.circle.x) < epsilon$4
          && Math.abs(y - lArc.circle.cy) < epsilon$4) {
        previous = lArc.P;
        disappearing.unshift(lArc);
        detachBeach(lArc);
        lArc = previous;
      }

      disappearing.unshift(lArc);
      detachCircle(lArc);

      var rArc = next;
      while (rArc.circle
          && Math.abs(x - rArc.circle.x) < epsilon$4
          && Math.abs(y - rArc.circle.cy) < epsilon$4) {
        next = rArc.N;
        disappearing.push(rArc);
        detachBeach(rArc);
        rArc = next;
      }

      disappearing.push(rArc);
      detachCircle(rArc);

      var nArcs = disappearing.length,
          iArc;
      for (iArc = 1; iArc < nArcs; ++iArc) {
        rArc = disappearing[iArc];
        lArc = disappearing[iArc - 1];
        setEdgeEnd(rArc.edge, lArc.site, rArc.site, vertex);
      }

      lArc = disappearing[0];
      rArc = disappearing[nArcs - 1];
      rArc.edge = createEdge(lArc.site, rArc.site, null, vertex);

      attachCircle(lArc);
      attachCircle(rArc);
    }

    function addBeach(site) {
      var x = site[0],
          directrix = site[1],
          lArc,
          rArc,
          dxl,
          dxr,
          node = beaches._;

      while (node) {
        dxl = leftBreakPoint(node, directrix) - x;
        if (dxl > epsilon$4) node = node.L; else {
          dxr = x - rightBreakPoint(node, directrix);
          if (dxr > epsilon$4) {
            if (!node.R) {
              lArc = node;
              break;
            }
            node = node.R;
          } else {
            if (dxl > -epsilon$4) {
              lArc = node.P;
              rArc = node;
            } else if (dxr > -epsilon$4) {
              lArc = node;
              rArc = node.N;
            } else {
              lArc = rArc = node;
            }
            break;
          }
        }
      }

      createCell(site);
      var newArc = createBeach(site);
      beaches.insert(lArc, newArc);

      if (!lArc && !rArc) return;

      if (lArc === rArc) {
        detachCircle(lArc);
        rArc = createBeach(lArc.site);
        beaches.insert(newArc, rArc);
        newArc.edge = rArc.edge = createEdge(lArc.site, newArc.site);
        attachCircle(lArc);
        attachCircle(rArc);
        return;
      }

      if (!rArc) { // && lArc
        newArc.edge = createEdge(lArc.site, newArc.site);
        return;
      }

      // else lArc !== rArc
      detachCircle(lArc);
      detachCircle(rArc);

      var lSite = lArc.site,
          ax = lSite[0],
          ay = lSite[1],
          bx = site[0] - ax,
          by = site[1] - ay,
          rSite = rArc.site,
          cx = rSite[0] - ax,
          cy = rSite[1] - ay,
          d = 2 * (bx * cy - by * cx),
          hb = bx * bx + by * by,
          hc = cx * cx + cy * cy,
          vertex = [(cy * hb - by * hc) / d + ax, (bx * hc - cx * hb) / d + ay];

      setEdgeEnd(rArc.edge, lSite, rSite, vertex);
      newArc.edge = createEdge(lSite, site, null, vertex);
      rArc.edge = createEdge(site, rSite, null, vertex);
      attachCircle(lArc);
      attachCircle(rArc);
    }

    function leftBreakPoint(arc, directrix) {
      var site = arc.site,
          rfocx = site[0],
          rfocy = site[1],
          pby2 = rfocy - directrix;

      if (!pby2) return rfocx;

      var lArc = arc.P;
      if (!lArc) return -Infinity;

      site = lArc.site;
      var lfocx = site[0],
          lfocy = site[1],
          plby2 = lfocy - directrix;

      if (!plby2) return lfocx;

      var hl = lfocx - rfocx,
          aby2 = 1 / pby2 - 1 / plby2,
          b = hl / plby2;

      if (aby2) return (-b + Math.sqrt(b * b - 2 * aby2 * (hl * hl / (-2 * plby2) - lfocy + plby2 / 2 + rfocy - pby2 / 2))) / aby2 + rfocx;

      return (rfocx + lfocx) / 2;
    }

    function rightBreakPoint(arc, directrix) {
      var rArc = arc.N;
      if (rArc) return leftBreakPoint(rArc, directrix);
      var site = arc.site;
      return site[1] === directrix ? site[0] : Infinity;
    }

    var epsilon$4 = 1e-6;
    var epsilon2$2 = 1e-12;
    var beaches;
    var cells;
    var circles;
    var edges;

    function triangleArea(a, b, c) {
      return (a[0] - c[0]) * (b[1] - a[1]) - (a[0] - b[0]) * (c[1] - a[1]);
    }

    function lexicographic(a, b) {
      return b[1] - a[1]
          || b[0] - a[0];
    }

    function Diagram(sites, extent) {
      var site = sites.sort(lexicographic).pop(),
          x,
          y,
          circle;

      edges = [];
      cells = new Array(sites.length);
      beaches = new RedBlackTree;
      circles = new RedBlackTree;

      while (true) {
        circle = firstCircle;
        if (site && (!circle || site[1] < circle.y || (site[1] === circle.y && site[0] < circle.x))) {
          if (site[0] !== x || site[1] !== y) {
            addBeach(site);
            x = site[0], y = site[1];
          }
          site = sites.pop();
        } else if (circle) {
          removeBeach(circle.arc);
        } else {
          break;
        }
      }

      sortCellHalfedges();

      if (extent) {
        var x0 = +extent[0][0],
            y0 = +extent[0][1],
            x1 = +extent[1][0],
            y1 = +extent[1][1];
        clipEdges(x0, y0, x1, y1);
        clipCells(x0, y0, x1, y1);
      }

      this.edges = edges;
      this.cells = cells;

      beaches =
      circles =
      edges =
      cells = null;
    }

    Diagram.prototype = {
      constructor: Diagram,

      polygons: function() {
        var edges = this.edges;

        return this.cells.map(function(cell) {
          var polygon = cell.halfedges.map(function(i) { return cellHalfedgeStart(cell, edges[i]); });
          polygon.data = cell.site.data;
          return polygon;
        });
      },

      triangles: function() {
        var triangles = [],
            edges = this.edges;

        this.cells.forEach(function(cell, i) {
          if (!(m = (halfedges = cell.halfedges).length)) return;
          var site = cell.site,
              halfedges,
              j = -1,
              m,
              s0,
              e1 = edges[halfedges[m - 1]],
              s1 = e1.left === site ? e1.right : e1.left;

          while (++j < m) {
            s0 = s1;
            e1 = edges[halfedges[j]];
            s1 = e1.left === site ? e1.right : e1.left;
            if (s0 && s1 && i < s0.index && i < s1.index && triangleArea(site, s0, s1) < 0) {
              triangles.push([site.data, s0.data, s1.data]);
            }
          }
        });

        return triangles;
      },

      links: function() {
        return this.edges.filter(function(edge) {
          return edge.right;
        }).map(function(edge) {
          return {
            source: edge.left.data,
            target: edge.right.data
          };
        });
      },

      find: function(x, y, radius) {
        var that = this, i0, i1 = that._found || 0, n = that.cells.length, cell;

        // Use the previously-found cell, or start with an arbitrary one.
        while (!(cell = that.cells[i1])) if (++i1 >= n) return null;
        var dx = x - cell.site[0], dy = y - cell.site[1], d2 = dx * dx + dy * dy;

        // Traverse the half-edges to find a closer cell, if any.
        do {
          cell = that.cells[i0 = i1], i1 = null;
          cell.halfedges.forEach(function(e) {
            var edge = that.edges[e], v = edge.left;
            if ((v === cell.site || !v) && !(v = edge.right)) return;
            var vx = x - v[0], vy = y - v[1], v2 = vx * vx + vy * vy;
            if (v2 < d2) d2 = v2, i1 = v.index;
          });
        } while (i1 !== null);

        that._found = i0;

        return radius == null || d2 <= radius * radius ? cell.site : null;
      }
    };

    function voronoi() {
      var x = x$4,
          y = y$4,
          extent = null;

      function voronoi(data) {
        return new Diagram(data.map(function(d, i) {
          var s = [Math.round(x(d, i, data) / epsilon$4) * epsilon$4, Math.round(y(d, i, data) / epsilon$4) * epsilon$4];
          s.index = i;
          s.data = d;
          return s;
        }), extent);
      }

      voronoi.polygons = function(data) {
        return voronoi(data).polygons();
      };

      voronoi.links = function(data) {
        return voronoi(data).links();
      };

      voronoi.triangles = function(data) {
        return voronoi(data).triangles();
      };

      voronoi.x = function(_) {
        return arguments.length ? (x = typeof _ === "function" ? _ : constant$c(+_), voronoi) : x;
      };

      voronoi.y = function(_) {
        return arguments.length ? (y = typeof _ === "function" ? _ : constant$c(+_), voronoi) : y;
      };

      voronoi.extent = function(_) {
        return arguments.length ? (extent = _ == null ? null : [[+_[0][0], +_[0][1]], [+_[1][0], +_[1][1]]], voronoi) : extent && [[extent[0][0], extent[0][1]], [extent[1][0], extent[1][1]]];
      };

      voronoi.size = function(_) {
        return arguments.length ? (extent = _ == null ? null : [[0, 0], [+_[0], +_[1]]], voronoi) : extent && [extent[1][0] - extent[0][0], extent[1][1] - extent[0][1]];
      };

      return voronoi;
    }

    function constant$d(x) {
      return function() {
        return x;
      };
    }

    function ZoomEvent(target, type, transform) {
      this.target = target;
      this.type = type;
      this.transform = transform;
    }

    function Transform(k, x, y) {
      this.k = k;
      this.x = x;
      this.y = y;
    }

    Transform.prototype = {
      constructor: Transform,
      scale: function(k) {
        return k === 1 ? this : new Transform(this.k * k, this.x, this.y);
      },
      translate: function(x, y) {
        return x === 0 & y === 0 ? this : new Transform(this.k, this.x + this.k * x, this.y + this.k * y);
      },
      apply: function(point) {
        return [point[0] * this.k + this.x, point[1] * this.k + this.y];
      },
      applyX: function(x) {
        return x * this.k + this.x;
      },
      applyY: function(y) {
        return y * this.k + this.y;
      },
      invert: function(location) {
        return [(location[0] - this.x) / this.k, (location[1] - this.y) / this.k];
      },
      invertX: function(x) {
        return (x - this.x) / this.k;
      },
      invertY: function(y) {
        return (y - this.y) / this.k;
      },
      rescaleX: function(x) {
        return x.copy().domain(x.range().map(this.invertX, this).map(x.invert, x));
      },
      rescaleY: function(y) {
        return y.copy().domain(y.range().map(this.invertY, this).map(y.invert, y));
      },
      toString: function() {
        return "translate(" + this.x + "," + this.y + ") scale(" + this.k + ")";
      }
    };

    var identity$9 = new Transform(1, 0, 0);

    transform$1.prototype = Transform.prototype;

    function transform$1(node) {
      while (!node.__zoom) if (!(node = node.parentNode)) return identity$9;
      return node.__zoom;
    }

    function nopropagation$2() {
      event.stopImmediatePropagation();
    }

    function noevent$2() {
      event.preventDefault();
      event.stopImmediatePropagation();
    }

    // Ignore right-click, since that should open the context menu.
    function defaultFilter$2() {
      return !event.ctrlKey && !event.button;
    }

    function defaultExtent$1() {
      var e = this;
      if (e instanceof SVGElement) {
        e = e.ownerSVGElement || e;
        if (e.hasAttribute("viewBox")) {
          e = e.viewBox.baseVal;
          return [[e.x, e.y], [e.x + e.width, e.y + e.height]];
        }
        return [[0, 0], [e.width.baseVal.value, e.height.baseVal.value]];
      }
      return [[0, 0], [e.clientWidth, e.clientHeight]];
    }

    function defaultTransform() {
      return this.__zoom || identity$9;
    }

    function defaultWheelDelta() {
      return -event.deltaY * (event.deltaMode === 1 ? 0.05 : event.deltaMode ? 1 : 0.002);
    }

    function defaultTouchable$2() {
      return navigator.maxTouchPoints || ("ontouchstart" in this);
    }

    function defaultConstrain(transform, extent, translateExtent) {
      var dx0 = transform.invertX(extent[0][0]) - translateExtent[0][0],
          dx1 = transform.invertX(extent[1][0]) - translateExtent[1][0],
          dy0 = transform.invertY(extent[0][1]) - translateExtent[0][1],
          dy1 = transform.invertY(extent[1][1]) - translateExtent[1][1];
      return transform.translate(
        dx1 > dx0 ? (dx0 + dx1) / 2 : Math.min(0, dx0) || Math.max(0, dx1),
        dy1 > dy0 ? (dy0 + dy1) / 2 : Math.min(0, dy0) || Math.max(0, dy1)
      );
    }

    function zoom() {
      var filter = defaultFilter$2,
          extent = defaultExtent$1,
          constrain = defaultConstrain,
          wheelDelta = defaultWheelDelta,
          touchable = defaultTouchable$2,
          scaleExtent = [0, Infinity],
          translateExtent = [[-Infinity, -Infinity], [Infinity, Infinity]],
          duration = 250,
          interpolate = interpolateZoom,
          listeners = dispatch("start", "zoom", "end"),
          touchstarting,
          touchending,
          touchDelay = 500,
          wheelDelay = 150,
          clickDistance2 = 0;

      function zoom(selection) {
        selection
            .property("__zoom", defaultTransform)
            .on("wheel.zoom", wheeled)
            .on("mousedown.zoom", mousedowned)
            .on("dblclick.zoom", dblclicked)
          .filter(touchable)
            .on("touchstart.zoom", touchstarted)
            .on("touchmove.zoom", touchmoved)
            .on("touchend.zoom touchcancel.zoom", touchended)
            .style("touch-action", "none")
            .style("-webkit-tap-highlight-color", "rgba(0,0,0,0)");
      }

      zoom.transform = function(collection, transform, point) {
        var selection = collection.selection ? collection.selection() : collection;
        selection.property("__zoom", defaultTransform);
        if (collection !== selection) {
          schedule(collection, transform, point);
        } else {
          selection.interrupt().each(function() {
            gesture(this, arguments)
                .start()
                .zoom(null, typeof transform === "function" ? transform.apply(this, arguments) : transform)
                .end();
          });
        }
      };

      zoom.scaleBy = function(selection, k, p) {
        zoom.scaleTo(selection, function() {
          var k0 = this.__zoom.k,
              k1 = typeof k === "function" ? k.apply(this, arguments) : k;
          return k0 * k1;
        }, p);
      };

      zoom.scaleTo = function(selection, k, p) {
        zoom.transform(selection, function() {
          var e = extent.apply(this, arguments),
              t0 = this.__zoom,
              p0 = p == null ? centroid(e) : typeof p === "function" ? p.apply(this, arguments) : p,
              p1 = t0.invert(p0),
              k1 = typeof k === "function" ? k.apply(this, arguments) : k;
          return constrain(translate(scale(t0, k1), p0, p1), e, translateExtent);
        }, p);
      };

      zoom.translateBy = function(selection, x, y) {
        zoom.transform(selection, function() {
          return constrain(this.__zoom.translate(
            typeof x === "function" ? x.apply(this, arguments) : x,
            typeof y === "function" ? y.apply(this, arguments) : y
          ), extent.apply(this, arguments), translateExtent);
        });
      };

      zoom.translateTo = function(selection, x, y, p) {
        zoom.transform(selection, function() {
          var e = extent.apply(this, arguments),
              t = this.__zoom,
              p0 = p == null ? centroid(e) : typeof p === "function" ? p.apply(this, arguments) : p;
          return constrain(identity$9.translate(p0[0], p0[1]).scale(t.k).translate(
            typeof x === "function" ? -x.apply(this, arguments) : -x,
            typeof y === "function" ? -y.apply(this, arguments) : -y
          ), e, translateExtent);
        }, p);
      };

      function scale(transform, k) {
        k = Math.max(scaleExtent[0], Math.min(scaleExtent[1], k));
        return k === transform.k ? transform : new Transform(k, transform.x, transform.y);
      }

      function translate(transform, p0, p1) {
        var x = p0[0] - p1[0] * transform.k, y = p0[1] - p1[1] * transform.k;
        return x === transform.x && y === transform.y ? transform : new Transform(transform.k, x, y);
      }

      function centroid(extent) {
        return [(+extent[0][0] + +extent[1][0]) / 2, (+extent[0][1] + +extent[1][1]) / 2];
      }

      function schedule(transition, transform, point) {
        transition
            .on("start.zoom", function() { gesture(this, arguments).start(); })
            .on("interrupt.zoom end.zoom", function() { gesture(this, arguments).end(); })
            .tween("zoom", function() {
              var that = this,
                  args = arguments,
                  g = gesture(that, args),
                  e = extent.apply(that, args),
                  p = point == null ? centroid(e) : typeof point === "function" ? point.apply(that, args) : point,
                  w = Math.max(e[1][0] - e[0][0], e[1][1] - e[0][1]),
                  a = that.__zoom,
                  b = typeof transform === "function" ? transform.apply(that, args) : transform,
                  i = interpolate(a.invert(p).concat(w / a.k), b.invert(p).concat(w / b.k));
              return function(t) {
                if (t === 1) t = b; // Avoid rounding error on end.
                else { var l = i(t), k = w / l[2]; t = new Transform(k, p[0] - l[0] * k, p[1] - l[1] * k); }
                g.zoom(null, t);
              };
            });
      }

      function gesture(that, args, clean) {
        return (!clean && that.__zooming) || new Gesture(that, args);
      }

      function Gesture(that, args) {
        this.that = that;
        this.args = args;
        this.active = 0;
        this.extent = extent.apply(that, args);
        this.taps = 0;
      }

      Gesture.prototype = {
        start: function() {
          if (++this.active === 1) {
            this.that.__zooming = this;
            this.emit("start");
          }
          return this;
        },
        zoom: function(key, transform) {
          if (this.mouse && key !== "mouse") this.mouse[1] = transform.invert(this.mouse[0]);
          if (this.touch0 && key !== "touch") this.touch0[1] = transform.invert(this.touch0[0]);
          if (this.touch1 && key !== "touch") this.touch1[1] = transform.invert(this.touch1[0]);
          this.that.__zoom = transform;
          this.emit("zoom");
          return this;
        },
        end: function() {
          if (--this.active === 0) {
            delete this.that.__zooming;
            this.emit("end");
          }
          return this;
        },
        emit: function(type) {
          customEvent(new ZoomEvent(zoom, type, this.that.__zoom), listeners.apply, listeners, [type, this.that, this.args]);
        }
      };

      function wheeled() {
        if (!filter.apply(this, arguments)) return;
        var g = gesture(this, arguments),
            t = this.__zoom,
            k = Math.max(scaleExtent[0], Math.min(scaleExtent[1], t.k * Math.pow(2, wheelDelta.apply(this, arguments)))),
            p = mouse(this);

        // If the mouse is in the same location as before, reuse it.
        // If there were recent wheel events, reset the wheel idle timeout.
        if (g.wheel) {
          if (g.mouse[0][0] !== p[0] || g.mouse[0][1] !== p[1]) {
            g.mouse[1] = t.invert(g.mouse[0] = p);
          }
          clearTimeout(g.wheel);
        }

        // If this wheel event won’t trigger a transform change, ignore it.
        else if (t.k === k) return;

        // Otherwise, capture the mouse point and location at the start.
        else {
          g.mouse = [p, t.invert(p)];
          interrupt(this);
          g.start();
        }

        noevent$2();
        g.wheel = setTimeout(wheelidled, wheelDelay);
        g.zoom("mouse", constrain(translate(scale(t, k), g.mouse[0], g.mouse[1]), g.extent, translateExtent));

        function wheelidled() {
          g.wheel = null;
          g.end();
        }
      }

      function mousedowned() {
        if (touchending || !filter.apply(this, arguments)) return;
        var g = gesture(this, arguments, true),
            v = select(event.view).on("mousemove.zoom", mousemoved, true).on("mouseup.zoom", mouseupped, true),
            p = mouse(this),
            x0 = event.clientX,
            y0 = event.clientY;

        dragDisable(event.view);
        nopropagation$2();
        g.mouse = [p, this.__zoom.invert(p)];
        interrupt(this);
        g.start();

        function mousemoved() {
          noevent$2();
          if (!g.moved) {
            var dx = event.clientX - x0, dy = event.clientY - y0;
            g.moved = dx * dx + dy * dy > clickDistance2;
          }
          g.zoom("mouse", constrain(translate(g.that.__zoom, g.mouse[0] = mouse(g.that), g.mouse[1]), g.extent, translateExtent));
        }

        function mouseupped() {
          v.on("mousemove.zoom mouseup.zoom", null);
          yesdrag(event.view, g.moved);
          noevent$2();
          g.end();
        }
      }

      function dblclicked() {
        if (!filter.apply(this, arguments)) return;
        var t0 = this.__zoom,
            p0 = mouse(this),
            p1 = t0.invert(p0),
            k1 = t0.k * (event.shiftKey ? 0.5 : 2),
            t1 = constrain(translate(scale(t0, k1), p0, p1), extent.apply(this, arguments), translateExtent);

        noevent$2();
        if (duration > 0) select(this).transition().duration(duration).call(schedule, t1, p0);
        else select(this).call(zoom.transform, t1);
      }

      function touchstarted() {
        if (!filter.apply(this, arguments)) return;
        var touches = event.touches,
            n = touches.length,
            g = gesture(this, arguments, event.changedTouches.length === n),
            started, i, t, p;

        nopropagation$2();
        for (i = 0; i < n; ++i) {
          t = touches[i], p = touch(this, touches, t.identifier);
          p = [p, this.__zoom.invert(p), t.identifier];
          if (!g.touch0) g.touch0 = p, started = true, g.taps = 1 + !!touchstarting;
          else if (!g.touch1 && g.touch0[2] !== p[2]) g.touch1 = p, g.taps = 0;
        }

        if (touchstarting) touchstarting = clearTimeout(touchstarting);

        if (started) {
          if (g.taps < 2) touchstarting = setTimeout(function() { touchstarting = null; }, touchDelay);
          interrupt(this);
          g.start();
        }
      }

      function touchmoved() {
        if (!this.__zooming) return;
        var g = gesture(this, arguments),
            touches = event.changedTouches,
            n = touches.length, i, t, p, l;

        noevent$2();
        if (touchstarting) touchstarting = clearTimeout(touchstarting);
        g.taps = 0;
        for (i = 0; i < n; ++i) {
          t = touches[i], p = touch(this, touches, t.identifier);
          if (g.touch0 && g.touch0[2] === t.identifier) g.touch0[0] = p;
          else if (g.touch1 && g.touch1[2] === t.identifier) g.touch1[0] = p;
        }
        t = g.that.__zoom;
        if (g.touch1) {
          var p0 = g.touch0[0], l0 = g.touch0[1],
              p1 = g.touch1[0], l1 = g.touch1[1],
              dp = (dp = p1[0] - p0[0]) * dp + (dp = p1[1] - p0[1]) * dp,
              dl = (dl = l1[0] - l0[0]) * dl + (dl = l1[1] - l0[1]) * dl;
          t = scale(t, Math.sqrt(dp / dl));
          p = [(p0[0] + p1[0]) / 2, (p0[1] + p1[1]) / 2];
          l = [(l0[0] + l1[0]) / 2, (l0[1] + l1[1]) / 2];
        }
        else if (g.touch0) p = g.touch0[0], l = g.touch0[1];
        else return;
        g.zoom("touch", constrain(translate(t, p, l), g.extent, translateExtent));
      }

      function touchended() {
        if (!this.__zooming) return;
        var g = gesture(this, arguments),
            touches = event.changedTouches,
            n = touches.length, i, t;

        nopropagation$2();
        if (touchending) clearTimeout(touchending);
        touchending = setTimeout(function() { touchending = null; }, touchDelay);
        for (i = 0; i < n; ++i) {
          t = touches[i];
          if (g.touch0 && g.touch0[2] === t.identifier) delete g.touch0;
          else if (g.touch1 && g.touch1[2] === t.identifier) delete g.touch1;
        }
        if (g.touch1 && !g.touch0) g.touch0 = g.touch1, delete g.touch1;
        if (g.touch0) g.touch0[1] = this.__zoom.invert(g.touch0[0]);
        else {
          g.end();
          // If this was a dbltap, reroute to the (optional) dblclick.zoom handler.
          if (g.taps === 2) {
            var p = select(this).on("dblclick.zoom");
            if (p) p.apply(this, arguments);
          }
        }
      }

      zoom.wheelDelta = function(_) {
        return arguments.length ? (wheelDelta = typeof _ === "function" ? _ : constant$d(+_), zoom) : wheelDelta;
      };

      zoom.filter = function(_) {
        return arguments.length ? (filter = typeof _ === "function" ? _ : constant$d(!!_), zoom) : filter;
      };

      zoom.touchable = function(_) {
        return arguments.length ? (touchable = typeof _ === "function" ? _ : constant$d(!!_), zoom) : touchable;
      };

      zoom.extent = function(_) {
        return arguments.length ? (extent = typeof _ === "function" ? _ : constant$d([[+_[0][0], +_[0][1]], [+_[1][0], +_[1][1]]]), zoom) : extent;
      };

      zoom.scaleExtent = function(_) {
        return arguments.length ? (scaleExtent[0] = +_[0], scaleExtent[1] = +_[1], zoom) : [scaleExtent[0], scaleExtent[1]];
      };

      zoom.translateExtent = function(_) {
        return arguments.length ? (translateExtent[0][0] = +_[0][0], translateExtent[1][0] = +_[1][0], translateExtent[0][1] = +_[0][1], translateExtent[1][1] = +_[1][1], zoom) : [[translateExtent[0][0], translateExtent[0][1]], [translateExtent[1][0], translateExtent[1][1]]];
      };

      zoom.constrain = function(_) {
        return arguments.length ? (constrain = _, zoom) : constrain;
      };

      zoom.duration = function(_) {
        return arguments.length ? (duration = +_, zoom) : duration;
      };

      zoom.interpolate = function(_) {
        return arguments.length ? (interpolate = _, zoom) : interpolate;
      };

      zoom.on = function() {
        var value = listeners.on.apply(listeners, arguments);
        return value === listeners ? zoom : value;
      };

      zoom.clickDistance = function(_) {
        return arguments.length ? (clickDistance2 = (_ = +_) * _, zoom) : Math.sqrt(clickDistance2);
      };

      return zoom;
    }

    var d3Typed = /*#__PURE__*/Object.freeze({
        __proto__: null,
        version: version$1,
        bisect: bisectRight,
        bisectRight: bisectRight,
        bisectLeft: bisectLeft,
        ascending: ascending,
        bisector: bisector,
        cross: cross,
        descending: descending,
        deviation: deviation,
        extent: extent,
        histogram: histogram,
        thresholdFreedmanDiaconis: freedmanDiaconis,
        thresholdScott: scott,
        thresholdSturges: thresholdSturges,
        max: max,
        mean: mean,
        median: median,
        merge: merge,
        min: min,
        pairs: pairs,
        permute: permute,
        quantile: threshold,
        range: sequence,
        scan: scan,
        shuffle: shuffle,
        sum: sum,
        ticks: ticks,
        tickIncrement: tickIncrement,
        tickStep: tickStep,
        transpose: transpose,
        variance: variance,
        zip: zip,
        axisTop: axisTop,
        axisRight: axisRight,
        axisBottom: axisBottom,
        axisLeft: axisLeft,
        brush: brush,
        brushX: brushX,
        brushY: brushY,
        brushSelection: brushSelection,
        chord: chord,
        ribbon: ribbon,
        nest: nest,
        set: set$3,
        map: map$1,
        keys: keys,
        values: values,
        entries: entries,
        color: color,
        rgb: rgb,
        hsl: hsl,
        lab: lab,
        hcl: hcl,
        lch: lch,
        gray: gray,
        cubehelix: cubehelix,
        contours: contours,
        contourDensity: density,
        dispatch: dispatch,
        drag: drag,
        dragDisable: dragDisable,
        dragEnable: yesdrag,
        dsvFormat: dsvFormat,
        csvParse: csvParse,
        csvParseRows: csvParseRows,
        csvFormat: csvFormat,
        csvFormatBody: csvFormatBody,
        csvFormatRows: csvFormatRows,
        csvFormatRow: csvFormatRow,
        csvFormatValue: csvFormatValue,
        tsvParse: tsvParse,
        tsvParseRows: tsvParseRows,
        tsvFormat: tsvFormat,
        tsvFormatBody: tsvFormatBody,
        tsvFormatRows: tsvFormatRows,
        tsvFormatRow: tsvFormatRow,
        tsvFormatValue: tsvFormatValue,
        autoType: autoType,
        easeLinear: linear$1,
        easeQuad: quadInOut,
        easeQuadIn: quadIn,
        easeQuadOut: quadOut,
        easeQuadInOut: quadInOut,
        easeCubic: cubicInOut,
        easeCubicIn: cubicIn,
        easeCubicOut: cubicOut,
        easeCubicInOut: cubicInOut,
        easePoly: polyInOut,
        easePolyIn: polyIn,
        easePolyOut: polyOut,
        easePolyInOut: polyInOut,
        easeSin: sinInOut,
        easeSinIn: sinIn,
        easeSinOut: sinOut,
        easeSinInOut: sinInOut,
        easeExp: expInOut,
        easeExpIn: expIn,
        easeExpOut: expOut,
        easeExpInOut: expInOut,
        easeCircle: circleInOut,
        easeCircleIn: circleIn,
        easeCircleOut: circleOut,
        easeCircleInOut: circleInOut,
        easeBounce: bounceOut,
        easeBounceIn: bounceIn,
        easeBounceOut: bounceOut,
        easeBounceInOut: bounceInOut,
        easeBack: backInOut,
        easeBackIn: backIn,
        easeBackOut: backOut,
        easeBackInOut: backInOut,
        easeElastic: elasticOut,
        easeElasticIn: elasticIn,
        easeElasticOut: elasticOut,
        easeElasticInOut: elasticInOut,
        blob: blob,
        buffer: buffer,
        dsv: dsv,
        csv: csv$1,
        tsv: tsv$1,
        image: image,
        json: json,
        text: text,
        xml: xml,
        html: html$1,
        svg: svg,
        forceCenter: center$1,
        forceCollide: collide,
        forceLink: link,
        forceManyBody: manyBody,
        forceRadial: radial,
        forceSimulation: simulation,
        forceX: x$2,
        forceY: y$2,
        formatDefaultLocale: defaultLocale,
        get format () { return format; },
        get formatPrefix () { return formatPrefix; },
        formatLocale: formatLocale,
        formatSpecifier: formatSpecifier,
        FormatSpecifier: FormatSpecifier,
        precisionFixed: precisionFixed,
        precisionPrefix: precisionPrefix,
        precisionRound: precisionRound,
        geoArea: area$1,
        geoBounds: bounds,
        geoCentroid: centroid,
        geoCircle: circle,
        geoClipAntimeridian: clipAntimeridian,
        geoClipCircle: clipCircle,
        geoClipExtent: extent$1,
        geoClipRectangle: clipRectangle,
        geoContains: contains$1,
        geoDistance: distance,
        geoGraticule: graticule,
        geoGraticule10: graticule10,
        geoInterpolate: interpolate$1,
        geoLength: length$1,
        geoPath: index$1,
        geoAlbers: albers,
        geoAlbersUsa: albersUsa,
        geoAzimuthalEqualArea: azimuthalEqualArea,
        geoAzimuthalEqualAreaRaw: azimuthalEqualAreaRaw,
        geoAzimuthalEquidistant: azimuthalEquidistant,
        geoAzimuthalEquidistantRaw: azimuthalEquidistantRaw,
        geoConicConformal: conicConformal,
        geoConicConformalRaw: conicConformalRaw,
        geoConicEqualArea: conicEqualArea,
        geoConicEqualAreaRaw: conicEqualAreaRaw,
        geoConicEquidistant: conicEquidistant,
        geoConicEquidistantRaw: conicEquidistantRaw,
        geoEqualEarth: equalEarth,
        geoEqualEarthRaw: equalEarthRaw,
        geoEquirectangular: equirectangular,
        geoEquirectangularRaw: equirectangularRaw,
        geoGnomonic: gnomonic,
        geoGnomonicRaw: gnomonicRaw,
        geoIdentity: identity$5,
        geoProjection: projection,
        geoProjectionMutator: projectionMutator,
        geoMercator: mercator,
        geoMercatorRaw: mercatorRaw,
        geoNaturalEarth1: naturalEarth1,
        geoNaturalEarth1Raw: naturalEarth1Raw,
        geoOrthographic: orthographic,
        geoOrthographicRaw: orthographicRaw,
        geoStereographic: stereographic,
        geoStereographicRaw: stereographicRaw,
        geoTransverseMercator: transverseMercator,
        geoTransverseMercatorRaw: transverseMercatorRaw,
        geoRotation: rotation,
        geoStream: geoStream,
        geoTransform: transform,
        cluster: cluster,
        hierarchy: hierarchy,
        pack: index$2,
        packSiblings: siblings,
        packEnclose: enclose,
        partition: partition,
        stratify: stratify,
        tree: tree,
        treemap: index$3,
        treemapBinary: binary,
        treemapDice: treemapDice,
        treemapSlice: treemapSlice,
        treemapSliceDice: sliceDice,
        treemapSquarify: squarify,
        treemapResquarify: resquarify,
        interpolate: interpolateValue,
        interpolateArray: array$1,
        interpolateBasis: basis$1,
        interpolateBasisClosed: basisClosed,
        interpolateDate: date,
        interpolateDiscrete: discrete,
        interpolateHue: hue$1,
        interpolateNumber: interpolateNumber,
        interpolateNumberArray: numberArray,
        interpolateObject: object,
        interpolateRound: interpolateRound,
        interpolateString: interpolateString,
        interpolateTransformCss: interpolateTransformCss,
        interpolateTransformSvg: interpolateTransformSvg,
        interpolateZoom: interpolateZoom,
        interpolateRgb: interpolateRgb,
        interpolateRgbBasis: rgbBasis,
        interpolateRgbBasisClosed: rgbBasisClosed,
        interpolateHsl: hsl$2,
        interpolateHslLong: hslLong,
        interpolateLab: lab$1,
        interpolateHcl: hcl$2,
        interpolateHclLong: hclLong,
        interpolateCubehelix: cubehelix$2,
        interpolateCubehelixLong: cubehelixLong,
        piecewise: piecewise,
        quantize: quantize,
        path: path,
        polygonArea: area$2,
        polygonCentroid: centroid$1,
        polygonHull: hull,
        polygonContains: contains$2,
        polygonLength: length$2,
        quadtree: quadtree,
        randomUniform: uniform,
        randomNormal: normal,
        randomLogNormal: logNormal,
        randomBates: bates,
        randomIrwinHall: irwinHall,
        randomExponential: exponential$1,
        scaleBand: band,
        scalePoint: point$1,
        scaleIdentity: identity$7,
        scaleLinear: linear$2,
        scaleLog: log$1,
        scaleSymlog: symlog,
        scaleOrdinal: ordinal,
        scaleImplicit: implicit,
        scalePow: pow$1,
        scaleSqrt: sqrt$1,
        scaleQuantile: quantile,
        scaleQuantize: quantize$1,
        scaleThreshold: threshold$1,
        scaleTime: time,
        scaleUtc: utcTime,
        scaleSequential: sequential,
        scaleSequentialLog: sequentialLog,
        scaleSequentialPow: sequentialPow,
        scaleSequentialSqrt: sequentialSqrt,
        scaleSequentialSymlog: sequentialSymlog,
        scaleSequentialQuantile: sequentialQuantile,
        scaleDiverging: diverging,
        scaleDivergingLog: divergingLog,
        scaleDivergingPow: divergingPow,
        scaleDivergingSqrt: divergingSqrt,
        scaleDivergingSymlog: divergingSymlog,
        tickFormat: tickFormat,
        schemeCategory10: category10,
        schemeAccent: Accent,
        schemeDark2: Dark2,
        schemePaired: Paired,
        schemePastel1: Pastel1,
        schemePastel2: Pastel2,
        schemeSet1: Set1,
        schemeSet2: Set2,
        schemeSet3: Set3,
        schemeTableau10: Tableau10,
        interpolateBrBG: BrBG,
        schemeBrBG: scheme,
        interpolatePRGn: PRGn,
        schemePRGn: scheme$1,
        interpolatePiYG: PiYG,
        schemePiYG: scheme$2,
        interpolatePuOr: PuOr,
        schemePuOr: scheme$3,
        interpolateRdBu: RdBu,
        schemeRdBu: scheme$4,
        interpolateRdGy: RdGy,
        schemeRdGy: scheme$5,
        interpolateRdYlBu: RdYlBu,
        schemeRdYlBu: scheme$6,
        interpolateRdYlGn: RdYlGn,
        schemeRdYlGn: scheme$7,
        interpolateSpectral: Spectral,
        schemeSpectral: scheme$8,
        interpolateBuGn: BuGn,
        schemeBuGn: scheme$9,
        interpolateBuPu: BuPu,
        schemeBuPu: scheme$a,
        interpolateGnBu: GnBu,
        schemeGnBu: scheme$b,
        interpolateOrRd: OrRd,
        schemeOrRd: scheme$c,
        interpolatePuBuGn: PuBuGn,
        schemePuBuGn: scheme$d,
        interpolatePuBu: PuBu,
        schemePuBu: scheme$e,
        interpolatePuRd: PuRd,
        schemePuRd: scheme$f,
        interpolateRdPu: RdPu,
        schemeRdPu: scheme$g,
        interpolateYlGnBu: YlGnBu,
        schemeYlGnBu: scheme$h,
        interpolateYlGn: YlGn,
        schemeYlGn: scheme$i,
        interpolateYlOrBr: YlOrBr,
        schemeYlOrBr: scheme$j,
        interpolateYlOrRd: YlOrRd,
        schemeYlOrRd: scheme$k,
        interpolateBlues: Blues,
        schemeBlues: scheme$l,
        interpolateGreens: Greens,
        schemeGreens: scheme$m,
        interpolateGreys: Greys,
        schemeGreys: scheme$n,
        interpolatePurples: Purples,
        schemePurples: scheme$o,
        interpolateReds: Reds,
        schemeReds: scheme$p,
        interpolateOranges: Oranges,
        schemeOranges: scheme$q,
        interpolateCividis: cividis,
        interpolateCubehelixDefault: cubehelix$3,
        interpolateRainbow: rainbow,
        interpolateWarm: warm,
        interpolateCool: cool,
        interpolateSinebow: sinebow,
        interpolateTurbo: turbo,
        interpolateViridis: viridis,
        interpolateMagma: magma,
        interpolateInferno: inferno,
        interpolatePlasma: plasma,
        create: create,
        creator: creator,
        local: local,
        matcher: matcher,
        mouse: mouse,
        namespace: namespace,
        namespaces: namespaces,
        clientPoint: point,
        select: select,
        selectAll: selectAll,
        selection: selection,
        selector: selector,
        selectorAll: selectorAll,
        style: styleValue,
        touch: touch,
        touches: touches,
        window: defaultView,
        get event () { return event; },
        customEvent: customEvent,
        arc: arc,
        area: area$3,
        line: line,
        pie: pie,
        areaRadial: areaRadial,
        radialArea: areaRadial,
        lineRadial: lineRadial$1,
        radialLine: lineRadial$1,
        pointRadial: pointRadial,
        linkHorizontal: linkHorizontal,
        linkVertical: linkVertical,
        linkRadial: linkRadial,
        symbol: symbol,
        symbols: symbols,
        symbolCircle: circle$2,
        symbolCross: cross$2,
        symbolDiamond: diamond,
        symbolSquare: square,
        symbolStar: star,
        symbolTriangle: triangle,
        symbolWye: wye,
        curveBasisClosed: basisClosed$1,
        curveBasisOpen: basisOpen,
        curveBasis: basis$2,
        curveBundle: bundle,
        curveCardinalClosed: cardinalClosed,
        curveCardinalOpen: cardinalOpen,
        curveCardinal: cardinal,
        curveCatmullRomClosed: catmullRomClosed,
        curveCatmullRomOpen: catmullRomOpen,
        curveCatmullRom: catmullRom,
        curveLinearClosed: linearClosed,
        curveLinear: curveLinear,
        curveMonotoneX: monotoneX,
        curveMonotoneY: monotoneY,
        curveNatural: natural,
        curveStep: step,
        curveStepAfter: stepAfter,
        curveStepBefore: stepBefore,
        stack: stack,
        stackOffsetExpand: expand,
        stackOffsetDiverging: diverging$1,
        stackOffsetNone: none$1,
        stackOffsetSilhouette: silhouette,
        stackOffsetWiggle: wiggle,
        stackOrderAppearance: appearance,
        stackOrderAscending: ascending$3,
        stackOrderDescending: descending$2,
        stackOrderInsideOut: insideOut,
        stackOrderNone: none$2,
        stackOrderReverse: reverse,
        timeInterval: newInterval,
        timeMillisecond: millisecond,
        timeMilliseconds: milliseconds,
        utcMillisecond: millisecond,
        utcMilliseconds: milliseconds,
        timeSecond: second,
        timeSeconds: seconds,
        utcSecond: second,
        utcSeconds: seconds,
        timeMinute: minute,
        timeMinutes: minutes,
        timeHour: hour,
        timeHours: hours,
        timeDay: day,
        timeDays: days,
        timeWeek: sunday,
        timeWeeks: sundays,
        timeSunday: sunday,
        timeSundays: sundays,
        timeMonday: monday,
        timeMondays: mondays,
        timeTuesday: tuesday,
        timeTuesdays: tuesdays,
        timeWednesday: wednesday,
        timeWednesdays: wednesdays,
        timeThursday: thursday,
        timeThursdays: thursdays,
        timeFriday: friday,
        timeFridays: fridays,
        timeSaturday: saturday,
        timeSaturdays: saturdays,
        timeMonth: month,
        timeMonths: months,
        timeYear: year,
        timeYears: years,
        utcMinute: utcMinute,
        utcMinutes: utcMinutes,
        utcHour: utcHour,
        utcHours: utcHours,
        utcDay: utcDay,
        utcDays: utcDays,
        utcWeek: utcSunday,
        utcWeeks: utcSundays,
        utcSunday: utcSunday,
        utcSundays: utcSundays,
        utcMonday: utcMonday,
        utcMondays: utcMondays,
        utcTuesday: utcTuesday,
        utcTuesdays: utcTuesdays,
        utcWednesday: utcWednesday,
        utcWednesdays: utcWednesdays,
        utcThursday: utcThursday,
        utcThursdays: utcThursdays,
        utcFriday: utcFriday,
        utcFridays: utcFridays,
        utcSaturday: utcSaturday,
        utcSaturdays: utcSaturdays,
        utcMonth: utcMonth,
        utcMonths: utcMonths,
        utcYear: utcYear,
        utcYears: utcYears,
        timeFormatDefaultLocale: defaultLocale$1,
        get timeFormat () { return timeFormat; },
        get timeParse () { return timeParse; },
        get utcFormat () { return utcFormat; },
        get utcParse () { return utcParse; },
        timeFormatLocale: formatLocale$1,
        isoFormat: formatIso,
        isoParse: parseIso,
        now: now,
        timer: timer,
        timerFlush: timerFlush,
        timeout: timeout$1,
        interval: interval$1,
        transition: transition,
        active: active,
        interrupt: interrupt,
        voronoi: voronoi,
        zoom: zoom,
        zoomTransform: transform$1,
        zoomIdentity: identity$9
    });

    /* Copyright 2016 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    const palettes = {
        googleStandard: [
            '#db4437',
            '#ff7043',
            '#f4b400',
            '#0f9d58',
            '#00796b',
            '#00acc1',
            '#4285f4',
            '#5c6bc0',
            '#ab47bc',
        ],
        googleCool: [
            '#9e9d24',
            '#0f9d58',
            '#00796b',
            '#00acc1',
            '#4285f4',
            '#5c6bc0',
            '#607d8b',
        ],
        googleWarm: [
            '#795548',
            '#ab47bc',
            '#f06292',
            '#c2185b',
            '#db4437',
            '#ff7043',
            '#f4b400',
        ],
        googleColorBlindAssist: [
            '#ff7043',
            '#00ACC1',
            '#AB47BC',
            '#2A56C6',
            '#0b8043',
            '#F7CB4D',
            '#c0ca33',
            '#5e35b1',
            '#A52714',
        ],
        // A colorblind-friendly palette designed for TensorBoard by Paul Tol
        // (https://personal.sron.nl/~pault/).
        tensorboardColorBlindAssist: [
            '#ff7043',
            '#0077bb',
            '#cc3311',
            '#33bbee',
            '#ee3377',
            '#009988',
            '#bbbbbb',
        ],
        // These palettes try to be better for color differentiation.
        // https://personal.sron.nl/~pault/
        colorBlindAssist1: [
            '#4477aa',
            '#44aaaa',
            '#aaaa44',
            '#aa7744',
            '#aa4455',
            '#aa4488',
        ],
        colorBlindAssist2: [
            '#88ccee',
            '#44aa99',
            '#117733',
            '#999933',
            '#ddcc77',
            '#cc6677',
            '#882255',
            '#aa4499',
        ],
        colorBlindAssist3: [
            '#332288',
            '#6699cc',
            '#88ccee',
            '#44aa99',
            '#117733',
            '#999933',
            '#ddcc77',
            '#cc6677',
            '#aa4466',
            '#882255',
            '#661100',
            '#aa4499',
        ],
        colorBlindAssist4: [
            // Paul Tol's "Alternative Scheme for Qualitative Data". Preferred
            // if `tensorboardColorBlindAssist` cannot be used for any reason.
            '#4477aa',
            '#66ccee',
            '#228833',
            '#ccbb44',
            '#ee6677',
            '#aa3377',
            '#bbbbbb',
        ],
        // based on this palette: http://mkweb.bcgsc.ca/biovis2012/
        colorBlindAssist5: [
            '#FF6DB6',
            '#920000',
            '#924900',
            '#DBD100',
            '#24FF24',
            '#006DDB',
            '#490092',
        ],
        mldash: [
            '#E47EAD',
            '#F4640D',
            '#FAA300',
            '#F5E636',
            '#00A077',
            '#0077B8',
            '#00B7ED',
        ],
    };
    const standard = palettes.tensorboardColorBlindAssist;

    /* Copyright 2018 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    class ExperimentsStore extends BaseStore {
        constructor() {
            super(...arguments);
            this._experiments = [];
        }
        load() {
            const url = getRouter().experiments();
            return this.requestManager.request(url).then((newExperiments) => {
                if (!lodash.isEqual(this._experiments, newExperiments)) {
                    this._experiments = newExperiments;
                    this.emitChange();
                }
            });
        }
        /**
         * Get the current list of experiments. If no data is available, this will be
         * an empty array (i.e., there is no distinction between "no experiment" and
         * "no experiment yet").
         */
        getExperiments() {
            return this._experiments.slice();
        }
    }
    const experimentsStore = new ExperimentsStore();

    /* Copyright 2015 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    // Example usage:
    // runs = ["train", "test", "test1", "test2"]
    // ccs = new ColorScale();
    // ccs.domain(runs);
    // ccs.getColor("train");
    // ccs.getColor("test1");
    class ColorScale {
        /**
         * Creates a color scale with optional custom palette.
         * @param {Array<string>} palette The color palette to use, as an
         *   Array of hex strings. Defaults to the standard palette.
         */
        constructor(palette = standard) {
            this.palette = palette;
            this.identifiers = map$1();
        }
        /**
         * Set the domain of strings.
         * @param {Array<string>} strings - An array of possible strings to use as the
         *     domain for your scale.
         */
        setDomain(strings) {
            this.identifiers = map$1();
            strings.forEach((s, i) => {
                this.identifiers.set(s, this.palette[i % this.palette.length]);
            });
            return this;
        }
        /**
         * Use the color scale to transform an element in the domain into a color.
         * @param {string} The input string to map to a color.
         * @return {string} The color corresponding to that input string.
         * @throws Will error if input string is not in the scale's domain.
         */
        getColor(s) {
            if (!this.identifiers.has(s)) {
                throw new Error(`String ${s} was not in the domain.`);
            }
            return this.identifiers.get(s);
        }
    }
    /**
     * A color scale of a domain from a store. Automatically updated when the store
     * emits a change.
     */
    function createAutoUpdateColorScale(store, getDomain) {
        const colorScale = new ColorScale();
        function updateRunsColorScale() {
            colorScale.setDomain(getDomain());
        }
        store.addListener(updateRunsColorScale);
        updateRunsColorScale();
        return (domain) => colorScale.getColor(domain);
    }
    const runsColorScale = createAutoUpdateColorScale(runsStore, () => runsStore.getRuns());
    const experimentsColorScale = createAutoUpdateColorScale(experimentsStore, () => {
        return experimentsStore.getExperiments().map(({ name }) => name);
    });

    /* Copyright 2016 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    registerStyleDomModule({
        moduleName: 'run-color-style',
        styleContent: `
    [color-class='light-blue'] paper-checkbox {
      --paper-checkbox-checked-color: var(--paper-light-blue-500);
      --paper-checkbox-checked-ink-color: var(--paper-light-blue-500);
      --paper-checkbox-unchecked-color: var(--paper-light-blue-900);
      --paper-checkbox-unchecked-ink-color: var(--paper-light-blue-900);
    }
    [color-class='red'] paper-checkbox {
      --paper-checkbox-checked-color: var(--paper-red-500);
      --paper-checkbox-checked-ink-color: var(--paper-red-500);
      --paper-checkbox-unchecked-color: var(--paper-red-900);
      --paper-checkbox-unchecked-ink-color: var(--paper-red-900);
    }
    [color-class='green'] paper-checkbox {
      --paper-checkbox-checked-color: var(--paper-green-500);
      --paper-checkbox-checked-ink-color: var(--paper-green-500);
      --paper-checkbox-unchecked-color: var(--paper-green-900);
      --paper-checkbox-unchecked-ink-color: var(--paper-green-900);
    }
    [color-class='purple'] paper-checkbox {
      --paper-checkbox-checked-color: var(--paper-purple-500);
      --paper-checkbox-checked-ink-color: var(--paper-purple-500);
      --paper-checkbox-unchecked-color: var(--paper-purple-900);
      --paper-checkbox-unchecked-ink-color: var(--paper-purple-900);
    }
    [color-class='teal'] paper-checkbox {
      --paper-checkbox-checked-color: var(--paper-teal-500);
      --paper-checkbox-checked-ink-color: var(--paper-teal-500);
      --paper-checkbox-unchecked-color: var(--paper-teal-900);
      --paper-checkbox-unchecked-ink-color: var(--paper-teal-900);
    }
    [color-class='pink'] paper-checkbox {
      --paper-checkbox-checked-color: var(--paper-pink-500);
      --paper-checkbox-checked-ink-color: var(--paper-pink-500);
      --paper-checkbox-unchecked-color: var(--paper-pink-900);
      --paper-checkbox-unchecked-ink-color: var(--paper-pink-900);
    }
    [color-class='orange'] paper-checkbox {
      --paper-checkbox-checked-color: var(--paper-orange-500);
      --paper-checkbox-checked-ink-color: var(--paper-orange-500);
      --paper-checkbox-unchecked-color: var(--paper-orange-900);
      --paper-checkbox-unchecked-ink-color: var(--paper-orange-900);
    }
    [color-class='brown'] paper-checkbox {
      --paper-checkbox-checked-color: var(--paper-brown-500);
      --paper-checkbox-checked-ink-color: var(--paper-brown-500);
      --paper-checkbox-unchecked-color: var(--paper-brown-900);
      --paper-checkbox-unchecked-ink-color: var(--paper-brown-900);
    }
    [color-class='indigo'] paper-checkbox {
      --paper-checkbox-checked-color: var(--paper-indigo-500);
      --paper-checkbox-checked-ink-color: var(--paper-indigo-500);
      --paper-checkbox-unchecked-color: var(--paper-indigo-900);
      --paper-checkbox-unchecked-ink-color: var(--paper-indigo-900);
    }
  `,
    });

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /*
    tf-multi-checkbox creates a list of checkboxes that can be used to toggle on or off
    a large number of values. Each checkbox displays a name, and may also have an
    associated tooltip value. Checkboxes can be highlighted, hidden, and re-ordered.

    tf-multi-checkbox assumes that the names may be very long compared to the width
    of the checkbox, and the number of names may also be very large, and works to
    handle these situations gracefully.
    */
    let TfMultiCheckbox = class TfMultiCheckbox extends LegacyElementMixin(PolymerElement) {
        constructor() {
            super(...arguments);
            // All the values of the checkbox.
            this.names = [];
            this.coloring = {
                getColor: () => '',
            };
            // Regex for filtering the names.
            this.regex = '';
            this.selectionState = {};
            this.maxNamesToEnableByDefault = 40;
            // Updating the regex can be slow, because it involves updating styles
            // on a large number of Polymer paper-checkboxes. We don't want to do
            // this while the user is typing, as it may make a bad, laggy UI.
            // So we debounce the updates that come from user typing.
            this._debouncedRegexChange = this._debouncedRegexChangeImpl();
        }
        _debouncedRegexChangeImpl() {
            var debounced = lodash.debounce((r) => {
                this.regex = r;
            }, 150, { leading: false });
            return function () {
                var r = this.$$('#names-regex').value;
                if (r == '') {
                    // If the user cleared the field, they may be done typing, so
                    // update more quickly.
                    this.async(() => {
                        this.regex = r;
                    }, 30);
                }
                else {
                    debounced(r);
                }
            };
        }
        get _regex() {
            var regexString = this.regex;
            try {
                return new RegExp(regexString);
            }
            catch (e) {
                return null;
            }
        }
        _setIsolatorIcon() {
            var selectionMap = this.selectionState;
            var numChecked = lodash.filter(lodash.values(selectionMap)).length;
            var buttons = Array.prototype.slice.call(this.root.querySelectorAll('.isolator'));
            buttons.forEach(function (b) {
                if (numChecked === 1 && selectionMap[b.name]) {
                    b.icon = 'radio-button-checked';
                }
                else {
                    b.icon = 'radio-button-unchecked';
                }
            });
        }
        computeNamesMatchingRegex(__, ___) {
            const regex = this._regex;
            return regex ? this.names.filter((n) => regex.test(n)) : this.names;
        }
        computeOutSelected(__, ___) {
            var selectionState = this.selectionState;
            var num = this.maxNamesToEnableByDefault;
            var allEnabled = this.namesMatchingRegex.length <= num;
            return this.namesMatchingRegex.filter((n) => {
                return selectionState[n] == null ? allEnabled : selectionState[n];
            });
        }
        synchronizeColors(e) {
            this._setIsolatorIcon();
            const checkboxes = this.root.querySelectorAll('paper-checkbox');
            checkboxes.forEach((p) => {
                const color = this.coloring.getColor(p.name);
                p.updateStyles({
                    '--paper-checkbox-checked-color': color,
                    '--paper-checkbox-checked-ink-color': color,
                    '--paper-checkbox-unchecked-color': color,
                    '--paper-checkbox-unchecked-ink-color': color,
                });
            });
            const buttons = this.root.querySelectorAll('.isolator');
            buttons.forEach((p) => {
                const color = this.coloring.getColor(p.name);
                p.style['color'] = color;
            });
            // The updateStyles call fails silently if the browser doesn't have focus,
            // e.g. if TensorBoard was opened into a new tab that isn't visible.
            // So we wait for requestAnimationFrame.
            window.requestAnimationFrame(() => {
                this.updateStyles();
            });
        }
        _isolateName(e) {
            // If user clicks on the label for one name, enable it and disable all other
            // names.
            var name = e.target.name;
            var selectionState = {};
            this.names.forEach(function (n) {
                selectionState[n] = n == name;
            });
            this.selectionState = selectionState;
        }
        _checkboxChange(e) {
            var target = e.target;
            const newSelectionState = lodash.clone(this.selectionState);
            newSelectionState[target.name] = target.checked;
            // n.b. notifyPath won't work because names may have periods.
            this.selectionState = newSelectionState;
        }
        _isChecked(item, outSelectedChange) {
            return this.outSelected.indexOf(item) != -1;
        }
        toggleAll() {
            // If any are toggled on, we turn everything off. Or, if none are toggled
            // on, we turn everything on.
            const anyToggledOn = this.namesMatchingRegex.some((name) => this.outSelected.includes(name));
            const newSelectionState = {};
            this.names.forEach((n) => {
                newSelectionState[n] = !anyToggledOn;
            });
            this.selectionState = newSelectionState;
        }
    };
    TfMultiCheckbox.template = html `
    <style include="scrollbar-style"></style>
    <style include="run-color-style"></style>

    <paper-input
      id="names-regex"
      no-label-float=""
      label="Write a regex to filter runs"
      value="[[regex]]"
      on-bind-value-changed="_debouncedRegexChange"
    ></paper-input>
    <div id="outer-container" class="scrollbar">
      <template
        is="dom-repeat"
        items="[[namesMatchingRegex]]"
        on-dom-change="synchronizeColors"
      >
        <div class="name-row">
          <div
            class="icon-container checkbox-container vertical-align-container"
          >
            <paper-checkbox
              class="checkbox vertical-align-center"
              id$="checkbox-[[item]]"
              name="[[item]]"
              checked$="[[_isChecked(item, selectionState.*)]]"
              on-change="_checkboxChange"
            ></paper-checkbox>
          </div>
          <div
            class="icon-container isolator-container vertical-align-container"
          >
            <paper-icon-button
              icon="radio-button-unchecked"
              class="isolator vertical-align-center"
              on-tap="_isolateName"
              name="[[item]]"
            ></paper-icon-button>
          </div>
          <div class="item-label-container">
            <span>[[item]]</span>
          </div>
        </div>
      </template>
    </div>
    <style>
      paper-input {
        --paper-input-container-focus-color: var(--tb-orange-strong);
        --paper-input-container-input: {
          font-size: 14px;
        }
        --paper-input-container-label: {
          font-size: 14px;
        }
      }
      :host {
        display: flex;
        flex-direction: column;
        height: 100%;
        overflow: hidden;
      }
      #outer-container {
        contain: content;
        flex-grow: 1;
        flex-shrink: 1;
        overflow-x: hidden;
        overflow-y: auto;
        width: 100%;
        will-change: transform;
        word-wrap: break-word;
      }
      .name-row {
        contain: content;
        padding-top: 5px;
        padding-bottom: 5px;
        display: flex;
        flex-direction: row;
        font-size: 13px;
        word-break: break-all; /* makes wrapping of hyperparam strings better */
      }
      .icon-container {
        flex-grow: 0;
        flex-shrink: 0;
        padding-left: 2px;
      }
      .checkbox {
        padding-left: 2px;
        width: 18px;
        height: 18px;
      }
      .isolator {
        width: 18px;
        height: 18px;
        padding: 0px;
      }
      .isolator-container {
        padding-left: 6px;
        padding-right: 3px;
      }
      .checkbox-container {
        padding-left: 2px;
      }
      .item-label-container {
        padding-left: 5px;
        flex-grow: 1;
        flex-shrink: 1;
        width: 0px; /* hack to get the flex-grow to work properly */
      }
      .tooltip-value-container {
        display: flex;
        justify-content: center;
        flex-grow: 0;
        flex-shrink: 0;
        text-align: right;
        padding-left: 2px;
      }
      .vertical-align-container {
        display: flex;
        justify-content: center;
      }
      .vertical-align-container .vertical-align-center {
        align-self: center;
      }
      .vertical-align-container .vertical-align-top {
        align-self: start;
      }
    </style>
  `;
    __decorate([
        property({
            type: Array,
        }),
        __metadata("design:type", Array)
    ], TfMultiCheckbox.prototype, "names", void 0);
    __decorate([
        property({
            type: Object,
        }),
        __metadata("design:type", Object)
    ], TfMultiCheckbox.prototype, "coloring", void 0);
    __decorate([
        property({
            type: String,
            notify: true,
        }),
        __metadata("design:type", String)
    ], TfMultiCheckbox.prototype, "regex", void 0);
    __decorate([
        property({
            type: Array,
            computed: 'computeNamesMatchingRegex(names.*, _regex)',
        }),
        __metadata("design:type", Array)
    ], TfMultiCheckbox.prototype, "namesMatchingRegex", void 0);
    __decorate([
        property({
            // If a name is explicitly enabled by user gesture, True, if explicitly
            // disabled, False. If undefined, default value (enable for first k
            // names, disable after).
            type: Object,
            notify: true,
        }),
        __metadata("design:type", Object)
    ], TfMultiCheckbox.prototype, "selectionState", void 0);
    __decorate([
        property({
            type: Array,
            notify: true,
            computed: 'computeOutSelected(namesMatchingRegex.*, selectionState.*)',
        }),
        __metadata("design:type", Array)
    ], TfMultiCheckbox.prototype, "outSelected", void 0);
    __decorate([
        property({
            // When TB first loads, if it has k or fewer names, they are all enabled
            // by default. If there are more, then they are all disabled.
            type: Number,
        }),
        __metadata("design:type", Number)
    ], TfMultiCheckbox.prototype, "maxNamesToEnableByDefault", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfMultiCheckbox.prototype, "_debouncedRegexChange", void 0);
    __decorate([
        computed('regex'),
        __metadata("design:type", RegExp),
        __metadata("design:paramtypes", [])
    ], TfMultiCheckbox.prototype, "_regex", null);
    __decorate([
        observe('selectionState', 'names'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfMultiCheckbox.prototype, "_setIsolatorIcon", null);
    TfMultiCheckbox = __decorate([
        customElement('tf-multi-checkbox')
    ], TfMultiCheckbox);

    /* Copyright 2019 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    // tf-wbr-string safely renders a string, with <wbr> word break elements inserted
    // after substrings that match a regular expression pattern.
    let TfWbrString = class TfWbrString extends PolymerElement {
        get _parts() {
            var value = this.value;
            var delimiterPattern = this.delimiterPattern;
            const result = [];
            while (true) {
                const delimiterRegExp = new RegExp(delimiterPattern, 'g');
                delimiterRegExp.test(value);
                if (delimiterRegExp.lastIndex === 0) {
                    result.push(value);
                    break;
                }
                else {
                    result.push(value.slice(0, delimiterRegExp.lastIndex));
                    value = value.slice(delimiterRegExp.lastIndex);
                }
            }
            return result;
        }
    };
    TfWbrString.template = html `
    <!--
      This ugly formatting is required to prevent spaces from slipping
      into the HTML.
    -->
    <template is="dom-repeat" items="[[_parts]]" as="part"
      >[[part]]<wbr
    /></template>
  `;
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfWbrString.prototype, "value", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfWbrString.prototype, "delimiterPattern", void 0);
    __decorate([
        computed('value', 'delimiterPattern'),
        __metadata("design:type", Array),
        __metadata("design:paramtypes", [])
    ], TfWbrString.prototype, "_parts", null);
    TfWbrString = __decorate([
        customElement('tf-wbr-string')
    ], TfWbrString);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    let TfRunsSelector = class TfRunsSelector extends LegacyElementMixin(PolymerElement) {
        constructor() {
            super(...arguments);
            this.runSelectionState = getObjectInitializer('runSelectionState', {
                defaultValue: {},
            })
                .call(this);
            this.regexInput = getStringInitializer('regexInput', {
                defaultValue: '',
            })
                .call(this);
            this._dataLocationClipLength = 250;
            this._dataLocationDelimiterPattern = '[/=_,-]';
            this.coloring = {
                getColor: runsColorScale,
            };
            this._storeRunSelectionState = getObjectObserver('runSelectionState', {
                defaultValue: {},
            });
            this._regexObserver = getStringObserver('regexInput', {
                defaultValue: '',
            });
        }
        attached() {
            this._runStoreListener = runsStore.addListener(() => {
                this.set('runs', runsStore.getRuns());
            });
            this.set('runs', runsStore.getRuns());
            this._envStoreListener = environmentStore.addListener(() => {
                this.set('dataLocation', environmentStore.getDataLocation());
            });
            this.set('dataLocation', environmentStore.getDataLocation());
        }
        detached() {
            runsStore.removeListenerByKey(this._runStoreListener);
            environmentStore.removeListenerByKey(this._envStoreListener);
        }
        _toggleAll() {
            this.$.multiCheckbox.toggleAll();
        }
        get _clippedDataLocation() {
            var dataLocation = this.dataLocation;
            var dataLocationClipLength = this._dataLocationClipLength;
            if (dataLocation === undefined) {
                // The dataLocation has not been set yet.
                return undefined;
            }
            if (dataLocation.length > dataLocationClipLength) {
                // Clip the dataLocation to avoid blocking the runs selector. Let the
                // user view a more full version of the dataLocation.
                return dataLocation.substring(0, dataLocationClipLength);
            }
            else {
                return dataLocation;
            }
        }
        _openDataLocationDialog(event) {
            event.preventDefault();
            this.$$('#data-location-dialog').open();
        }
        _shouldShowExpandDataLocationButton(dataLocation, _dataLocationClipLength) {
            return dataLocation && dataLocation.length > _dataLocationClipLength;
        }
    };
    TfRunsSelector.template = html `
    <paper-dialog with-backdrop="" id="data-location-dialog">
      <h2>Data Location</h2>
      <tf-wbr-string
        value="[[dataLocation]]"
        delimiter-pattern="[[_dataLocationDelimiterPattern]]"
      >
      </tf-wbr-string
    ></paper-dialog>
    <div id="top-text">
      <h3 id="tooltip-help" class="tooltip-container">Runs</h3>
    </div>
    <tf-multi-checkbox
      id="multiCheckbox"
      names="[[runs]]"
      selection-state="{{runSelectionState}}"
      out-selected="{{selectedRuns}}"
      regex="{{regexInput}}"
      coloring="[[coloring]]"
    ></tf-multi-checkbox>
    <paper-button class="x-button" id="toggle-all" on-tap="_toggleAll">
      Toggle All Runs
    </paper-button>
    <template is="dom-if" if="[[dataLocation]]">
      <div id="data-location">
        <tf-wbr-string
          value="[[_clippedDataLocation]]"
          delimiter-pattern="[[_dataLocationDelimiterPattern]]"
        ></tf-wbr-string
        ><!--
          We use HTML comments to remove spaces before the ellipsis.
        --><template
          is="dom-if"
          if="[[_shouldShowExpandDataLocationButton(dataLocation, _dataLocationClipLength)]]"
          ><!--
          --><a href="" on-click="_openDataLocationDialog">…</a>
        </template>
      </div>
    </template>
    <style>
      :host {
        box-sizing: border-box;
        display: flex;
        flex-direction: column;
        padding-bottom: 10px;
      }
      #top-text {
        width: 100%;
        flex-grow: 0;
        flex-shrink: 0;
        padding-right: 16px;
        box-sizing: border-box;
        color: var(--paper-grey-800);
      }
      tf-multi-checkbox {
        display: flex;
        flex-grow: 1;
        flex-shrink: 1;
        overflow: hidden;
      }
      .x-button {
        font-size: 13px;
        background-color: var(--tb-ui-light-accent);
        color: var(--tb-ui-dark-accent);
      }
      #tooltip-help {
        color: var(--paper-grey-800);
        margin: 0;
        font-weight: normal;
        font-size: 14px;
        margin-bottom: 5px;
      }
      paper-button {
        margin-left: 0;
      }
      #data-location {
        color: var(--tb-ui-dark-accent);
        font-size: 13px;
        margin: 5px 0 0 0;
        max-width: 288px;
      }
    </style>
  `;
    __decorate([
        property({
            type: Object,
            observer: '_storeRunSelectionState',
        }),
        __metadata("design:type", Object)
    ], TfRunsSelector.prototype, "runSelectionState", void 0);
    __decorate([
        property({
            type: String,
            observer: '_regexObserver',
        }),
        __metadata("design:type", String)
    ], TfRunsSelector.prototype, "regexInput", void 0);
    __decorate([
        property({
            type: Array,
            notify: true,
        }),
        __metadata("design:type", Array)
    ], TfRunsSelector.prototype, "selectedRuns", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfRunsSelector.prototype, "runs", void 0);
    __decorate([
        property({
            type: String,
            notify: true,
        }),
        __metadata("design:type", String)
    ], TfRunsSelector.prototype, "dataLocation", void 0);
    __decorate([
        property({
            type: Number,
        }),
        __metadata("design:type", Number)
    ], TfRunsSelector.prototype, "_dataLocationClipLength", void 0);
    __decorate([
        property({
            type: String,
        }),
        __metadata("design:type", String)
    ], TfRunsSelector.prototype, "_dataLocationDelimiterPattern", void 0);
    __decorate([
        property({
            type: Object,
        }),
        __metadata("design:type", Object)
    ], TfRunsSelector.prototype, "coloring", void 0);
    __decorate([
        computed('dataLocation', '_dataLocationClipLength'),
        __metadata("design:type", String),
        __metadata("design:paramtypes", [])
    ], TfRunsSelector.prototype, "_clippedDataLocation", null);
    TfRunsSelector = __decorate([
        customElement('tf-runs-selector')
    ], TfRunsSelector);

    /* Copyright 2017 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the 'License');
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an 'AS IS' BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * A class that allows marking promises as cancelled.
     *
     * This can be useful to, e.g., prevent old network requests from
     * stomping new ones and writing bad data.
     *
     * Usage:
     *
     *     const canceller = new Canceller();
     *     let myPromise: Promise<Foo> = getPromise();
     *     myPromise.then(canceller.cancellable(({value, cancelled} => {
     *       if (cancelled) {
     *         console.warn("Don't make promises you can't keep >:-{");
     *       }
     *       console.log("Enjoy your value:", value);
     *     }));
     *
     *     // If `myPromise` is resolved now, then `cancelled` will be `false`.
     *     canceller.cancelAll();
     *     // If `myPromise` is resolved now, then `cancelled` will be `true`.
     */
    class Canceller {
        constructor() {
            /**
             * How many times has `cancelAll` been called?
             */
            this.cancellationCount = 0;
        }
        /**
         * Create a cancellable task. This returns a new function that, when
         * invoked, will pass its argument to the provided function as well as
         * a `cancelled` argument. This argument will be `false` unless and
         * until `cancelAll` is invoked after the creation of this task.
         */
        cancellable(f) {
            const originalCancellationCount = this.cancellationCount;
            return (value) => {
                const cancelled = this.cancellationCount !== originalCancellationCount;
                return f({ value, cancelled });
            };
        }
        /**
         * Mark all outstanding tasks as cancelled. Tasks not yet created will
         * not be affected.
         */
        cancelAll() {
            this.cancellationCount++;
        }
    }

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * Does not actually sanitize html.
     *
     * This exported function is overriden in the internal Google repository for
     * compatibility with security libraries used internally at Google.
     */
    function sanitize(html) {
        return html;
    }

    /* Copyright 2017 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    // tf-markdown-view renders raw HTML that has been converted from
    // Markdown by some other agent. The HTML must be sanitized, and must be
    // safe to inject directly into the DOM.
    // TensorBoard plugins can use the `markdown_to_safe_html` function from
    // the `tensorboard.plugin_util` module on the backend to generate HTML
    // suitable for use with this component.
    let TfMarkdownView = class TfMarkdownView extends LegacyElementMixin(PolymerElement) {
        constructor() {
            super(...arguments);
            this.html = '';
        }
        get sanitizedHtml() {
            return sanitize(this.html);
        }
        attached() {
            window.requestAnimationFrame(() => {
                this.scopeSubtree(this.$.markdown, /*shouldObserve=*/ true);
            });
        }
    };
    TfMarkdownView.template = html `
    <div id="markdown" inner-h-t-m-l="[[sanitizedHtml]]"></div>
    <style>
      /*
       * Reduce topmost and bottommost margins from 16px to 0.3em (renders
       * at about 4.8px) to keep the layout compact. This improves the
       * appearance when there is only one line of text; standard Markdown
       * renderers will still include a \`<p>\` element.
       *
       * By targeting only the top-level, extremal elements, we preserve any
       * actual paragraph breaks and only change the padding against the
       * component edges.
       */
      #markdown > p:first-child {
        margin-top: 0.3em;
      }
      #markdown > p:last-child {
        margin-bottom: 0.3em;
      }

      /* Pleasant styles for Markdown tables. */
      #markdown table {
        border-collapse: collapse;
      }
      #markdown table th {
        font-weight: 600;
      }
      #markdown table th,
      #markdown table td {
        padding: 6px 13px;
        border: 1px solid #dfe2e5;
      }
      #markdown table tr {
        background-color: #fff;
        border-top: 1px solid #c6cbd1;
      }
    </style>
  `;
    __decorate([
        property({
            type: String,
        }),
        __metadata("design:type", String)
    ], TfMarkdownView.prototype, "html", void 0);
    __decorate([
        computed('html'),
        __metadata("design:type", Object),
        __metadata("design:paramtypes", [])
    ], TfMarkdownView.prototype, "sanitizedHtml", null);
    TfMarkdownView = __decorate([
        customElement('tf-markdown-view')
    ], TfMarkdownView);

    /* Copyright 2017 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

    http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    function formatDate$1(date) {
        if (!date) {
            return '';
        }
        // Turn things like "GMT-0700 (PDT)" into just "PDT".
        return date.toString().replace(/GMT-\d+ \(([^)]+)\)/, '$1');
    }
    /**
     * Returns CSS color that will contrast against background.
     * @param {?string} background RGB hex color code, e.g. #eee, #eeeeee.
     * @return {string}
     */
    function pickTextColor(background) {
        const rgb = convertHexToRgb(background);
        if (!rgb) {
            return 'inherit';
        }
        // See: http://www.w3.org/TR/AERT#color-contrast
        const brightness = Math.round((rgb[0] * 299 + rgb[1] * 587 + rgb[2] * 114) / 1000);
        return brightness > 125 ? 'inherit' : '#eee';
    }
    /**
     * Turns a hex string into an RGB array.
     * @param {?string} color RGB hex color code, e.g. #eee, #eeeeee.
     * @return {Array<number>}
     */
    function convertHexToRgb(color) {
        if (!color) {
            return null;
        }
        let m = color.match(/^#([0-9a-f]{1,2})([0-9a-f]{1,2})([0-9a-f]{1,2})$/);
        if (!m) {
            return null;
        }
        if (color.length == 4) {
            for (var i = 1; i <= 3; i++) {
                m[i] = m[i] + m[i];
            }
        }
        return [parseInt(m[1], 16), parseInt(m[2], 16), parseInt(m[3], 16)];
    }

    /* Copyright 2017 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    registerStyleDomModule({
        moduleName: 'tf-card-heading-style',
        styleContent: `
    figcaption {
      width: 100%;
    }

    /** Horizontal line of labels. */
    .heading-row {
      margin-top: -4px;
      display: flex;
      flex-direction: row;
      flex-wrap: wrap;
    }

    /** Piece of text in the figure caption. */
    .heading-label {
      flex-grow: 1;
      margin-top: 4px;
      max-width: 100%;
      word-wrap: break-word;
    }

    /** Makes label show on the right. */
    .heading-right {
      flex-grow: 0;
    }
  `,
    });

    /* Copyright 2017 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * A compact heading to appear above a single visualization, summarizing
     * the metadata about a tag or run-tag combination.
     *
     * Properties (all optional):
     *   - `displayName`, `tag`, and `description` are properties from the
     *     relevant Summary protobuf.
     *   - `run` is the name of the TensorFlow run, if applicable.
     *   - `color` can be set to display a colored border at the left of the
     *     card; if left unset, no border will be displayed.
     *   - Any children of this component will be rendered below the above
     *     information, and can be used to display (for example) some small
     *     control widgets.
     */
    let TfCardHeading = class TfCardHeading extends PolymerElement {
        constructor() {
            super(...arguments);
            this.displayName = null;
            this.tag = null;
            this.run = null;
            this.description = null;
            this.color = null;
        }
        _updateHeadingStyle() {
            this.updateStyles({
                '--tf-card-heading-background-color': this._runBackground,
                '--tf-card-heading-color': this._runColor,
            });
        }
        _computeRunBackground(color) {
            return color || 'none';
        }
        _computeRunColor(color) {
            return pickTextColor(color);
        }
        get _nameLabel() {
            var displayName = this.displayName;
            var tag = this.tag;
            return displayName || tag || '';
        }
        get _tagLabel() {
            var displayName = this.displayName;
            var tag = this.tag;
            return tag && tag !== displayName ? tag : '';
        }
        _toggleDescriptionDialog(e) {
            const dialog = this.$.descriptionDialog;
            dialog.positionTarget = e.target;
            dialog.toggle();
        }
    };
    TfCardHeading.template = html `
    <div class="container">
      <figcaption class="content">
        <div class="heading-row">
          <template is="dom-if" if="[[_nameLabel]]">
            <div itemprop="name" class="heading-label name">[[_nameLabel]]</div>
          </template>
          <template is="dom-if" if="[[run]]">
            <!-- Extra wrapping span needed to avoid flexbox blockification. -->
            <!-- (see flexbox spec, section 4 "Flex Items") -->
            <span>
              <span
                itemprop="run"
                id="heading-run"
                class="heading-label heading-right run"
                >[[run]]</span
              >
            </span>
          </template>
        </div>
        <template is="dom-if" if="[[_tagLabel]]">
          <div class="heading-row">
            <div class="heading-label">
              tag: <span itemprop="tag">[[_tagLabel]]</span>
            </div>
          </div>
        </template>
        <slot></slot>
      </figcaption>
      <template is="dom-if" if="[[description]]">
        <paper-icon-button
          icon="info"
          on-tap="_toggleDescriptionDialog"
          title="Show summary description"
        ></paper-icon-button>
      </template>
      <paper-dialog
        id="descriptionDialog"
        no-overlap=""
        horizontal-align="auto"
        vertical-align="auto"
      >
        <paper-dialog-scrollable>
          <tf-markdown-view html="[[description]]"></tf-markdown-view>
        </paper-dialog-scrollable>
      </paper-dialog>
    </div>
    <style include="tf-card-heading-style">
      .container {
        display: flex;
      }
      .content {
        font-size: 12px;
        flex-grow: 1;
      }
      .name {
        font-size: 14px;
      }
      .run {
        font-size: 11px;
        width: auto;
        border-radius: 3px;
        font-weight: bold;
        padding: 1px 4px 2px;
      }
      paper-icon-button {
        flex-grow: 0;
      }
      paper-dialog-scrollable {
        max-width: 640px;
      }
      #heading-run {
        background: var(--tf-card-heading-background-color);
        color: var(--tf-card-heading-color);
      }
    </style>
  `;
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfCardHeading.prototype, "displayName", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfCardHeading.prototype, "tag", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfCardHeading.prototype, "run", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfCardHeading.prototype, "description", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfCardHeading.prototype, "color", void 0);
    __decorate([
        property({
            type: String,
            computed: '_computeRunBackground(color)',
            readOnly: true,
            observer: '_updateHeadingStyle',
        }),
        __metadata("design:type", String)
    ], TfCardHeading.prototype, "_runBackground", void 0);
    __decorate([
        property({
            type: String,
            computed: '_computeRunColor(color)',
            readOnly: true,
            observer: '_updateHeadingStyle',
        }),
        __metadata("design:type", String)
    ], TfCardHeading.prototype, "_runColor", void 0);
    __decorate([
        computed('displayName', 'tag'),
        __metadata("design:type", String),
        __metadata("design:paramtypes", [])
    ], TfCardHeading.prototype, "_nameLabel", null);
    __decorate([
        computed('displayName', 'tag'),
        __metadata("design:type", String),
        __metadata("design:paramtypes", [])
    ], TfCardHeading.prototype, "_tagLabel", null);
    TfCardHeading = __decorate([
        customElement('tf-card-heading')
    ], TfCardHeading);

    /* Copyright 2016 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /*
    tf-audio-loader loads an individual audio clip from the TensorBoard
    backend.
    */
    let _TfAudioLoader = class _TfAudioLoader extends LegacyElementMixin(PolymerElement) {
        constructor() {
            super(...arguments);
            this._metadataCanceller = new Canceller();
            this._steps = [];
            this._attached = false;
        }
        get _runColor() {
            var run = this.run;
            return runsColorScale(run);
        }
        get _hasAtLeastOneStep() {
            var steps = this._steps;
            return !!steps && steps.length > 0;
        }
        get _hasMultipleSteps() {
            var steps = this._steps;
            return !!steps && steps.length > 1;
        }
        get _maxStepIndex() {
            var steps = this._steps;
            return steps.length - 1;
        }
        get _currentDatum() {
            var steps = this._steps;
            var stepIndex = this._stepIndex;
            return steps[stepIndex];
        }
        get _sampleText() {
            var sample = this.sample;
            return `${sample + 1}`;
        }
        get _hasMultipleSamples() {
            var totalSamples = this.totalSamples;
            return totalSamples > 1;
        }
        attached() {
            this._attached = true;
            this.reload();
        }
        _reloadOnRunTagChange() {
            this.reload();
        }
        reload() {
            if (!this._attached) {
                return;
            }
            this._metadataCanceller.cancelAll();
            const router = getRouter();
            const url = router.pluginRoute('audio', '/audio', new URLSearchParams({
                tag: this.tag,
                run: this.run,
                sample: String(this.sample),
            }));
            const updateSteps = this._metadataCanceller.cancellable((result) => {
                if (result.cancelled) {
                    return;
                }
                const data = result.value;
                const steps = data.map(this._createStepDatum.bind(this));
                this.set('_steps', steps);
                this.set('_stepIndex', steps.length - 1);
            });
            this.requestManager.request(url).then(updateSteps);
        }
        _createStepDatum(audioMetadata) {
            const searchParam = new URLSearchParams(audioMetadata.query);
            // Include wall_time just to disambiguate the URL and force
            // the browser to reload the audio when the URL changes. The
            // backend doesn't care about the value.
            searchParam.append('ts', String(audioMetadata.wall_time));
            const url = getRouter().pluginRoute('audio', '/individualAudio', searchParam);
            return {
                wall_time: formatDate$1(new Date(audioMetadata.wall_time * 1000)),
                step: audioMetadata.step,
                label: audioMetadata.label,
                contentType: audioMetadata.contentType,
                url,
            };
        }
    };
    _TfAudioLoader.template = html `
    <tf-card-heading
      tag="[[tag]]"
      run="[[run]]"
      display-name="[[tagMetadata.displayName]]"
      description="[[tagMetadata.description]]"
      color="[[_runColor]]"
    >
      <template is="dom-if" if="[[_hasMultipleSamples]]">
        <div class="heading-row">
          <div class="heading-label">
            sample: [[_sampleText]] of [[totalSamples]]
          </div>
        </div>
      </template>
      <template is="dom-if" if="[[_hasAtLeastOneStep]]">
        <div class="heading-row">
          <div class="heading-label">
            step <strong>[[_currentDatum.step]]</strong>
          </div>
          <template is="dom-if" if="[[_currentDatum.wall_time]]">
            <div class="heading-label heading-right">
              [[_currentDatum.wall_time]]
            </div>
          </template>
        </div>
      </template>
      <template is="dom-if" if="[[_hasMultipleSteps]]">
        <div class="heading-row">
          <paper-slider
            id="steps"
            immediate-value="{{_stepIndex}}"
            max="[[_maxStepIndex]]"
            max-markers="[[_maxStepIndex]]"
            snaps=""
            step="1"
            value="{{_stepIndex}}"
          ></paper-slider>
        </div>
      </template>
    </tf-card-heading>
    <template is="dom-if" if="[[_hasAtLeastOneStep]]">
      <audio
        controls=""
        src$="[[_currentDatum.url]]"
        type$="[[_currentDatum.contentType]]"
      ></audio>
      <tf-markdown-view html="[[_currentDatum.label]]"></tf-markdown-view>
    </template>
    <div id="main-audio-container"></div>

    <style include="tf-card-heading-style">
      :host {
        display: block;
        width: 350px;
        height: auto;
        position: relative;
        --step-slider-knob-color: #424242;
        margin-right: 15px;
        margin-bottom: 15px;
      }

      #steps {
        height: 15px;
        margin: 0 0 0 -15px;
        width: 100%;
        box-sizing: border-box;
        padding: 0 5px; /* so the slider knob doesn't butt out */
        margin-top: 5px;
        --paper-slider-active-color: var(--step-slider-knob-color);
        --paper-slider-knob-color: var(--step-slider-knob-color);
        --paper-slider-pin-color: var(--step-slider-knob-color);
        --paper-slider-knob-start-color: var(--step-slider-knob-color);
        --paper-slider-knob-start-border-color: var(--step-slider-knob-color);
        --paper-slider-pin-start-color: var(--step-slider-knob-color);
      }
    </style>
  `;
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], _TfAudioLoader.prototype, "run", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], _TfAudioLoader.prototype, "tag", void 0);
    __decorate([
        property({ type: Number }),
        __metadata("design:type", Number)
    ], _TfAudioLoader.prototype, "sample", void 0);
    __decorate([
        property({ type: Number }),
        __metadata("design:type", Number)
    ], _TfAudioLoader.prototype, "totalSamples", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], _TfAudioLoader.prototype, "tagMetadata", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", RequestManager)
    ], _TfAudioLoader.prototype, "requestManager", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Canceller)
    ], _TfAudioLoader.prototype, "_metadataCanceller", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], _TfAudioLoader.prototype, "_steps", void 0);
    __decorate([
        property({ type: Number }),
        __metadata("design:type", Number)
    ], _TfAudioLoader.prototype, "_stepIndex", void 0);
    __decorate([
        computed('run'),
        __metadata("design:type", String),
        __metadata("design:paramtypes", [])
    ], _TfAudioLoader.prototype, "_runColor", null);
    __decorate([
        computed('_steps'),
        __metadata("design:type", Boolean),
        __metadata("design:paramtypes", [])
    ], _TfAudioLoader.prototype, "_hasAtLeastOneStep", null);
    __decorate([
        computed('_steps'),
        __metadata("design:type", Boolean),
        __metadata("design:paramtypes", [])
    ], _TfAudioLoader.prototype, "_hasMultipleSteps", null);
    __decorate([
        computed('_steps'),
        __metadata("design:type", Number),
        __metadata("design:paramtypes", [])
    ], _TfAudioLoader.prototype, "_maxStepIndex", null);
    __decorate([
        computed('_steps', '_stepIndex'),
        __metadata("design:type", Object),
        __metadata("design:paramtypes", [])
    ], _TfAudioLoader.prototype, "_currentDatum", null);
    __decorate([
        computed('sample'),
        __metadata("design:type", String),
        __metadata("design:paramtypes", [])
    ], _TfAudioLoader.prototype, "_sampleText", null);
    __decorate([
        computed('totalSamples'),
        __metadata("design:type", Boolean),
        __metadata("design:paramtypes", [])
    ], _TfAudioLoader.prototype, "_hasMultipleSamples", null);
    __decorate([
        observe('run', 'tag'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], _TfAudioLoader.prototype, "_reloadOnRunTagChange", null);
    _TfAudioLoader = __decorate([
        customElement('tf-audio-loader')
    ], _TfAudioLoader);

    /* Copyright 2016 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /*
    tf-audio-dashboard displays a dashboard that loads audio from a TensorFlow run.
    */
    let TfAudioDashboard = class TfAudioDashboard extends LegacyElementMixin(PolymerElement) {
        constructor() {
            super(...arguments);
            this.reloadOnReady = true;
            this._tagFilter = '';
            this._requestManager = new RequestManager();
        }
        ready() {
            super.ready();
            if (this.reloadOnReady)
                this.reload();
        }
        reload() {
            this._fetchTags().then(() => {
                this._reloadAudio();
            });
        }
        _fetchTags() {
            const url = getRouter().pluginRoute('audio', '/tags');
            return this._requestManager.request(url).then((runToTagInfo) => {
                if (lodash.isEqual(runToTagInfo, this._runToTagInfo)) {
                    // No need to update anything if there are no changes.
                    return;
                }
                const runToTag = lodash.mapValues(runToTagInfo, (x) => Object.keys(x));
                const tags = getTags(runToTag);
                this.set('_dataNotFound', tags.length === 0);
                this.set('_runToTagInfo', runToTagInfo);
            });
        }
        _reloadAudio() {
            this.root.querySelectorAll('tf-audio-loader').forEach((audio) => {
                audio.reload();
            });
        }
        _shouldOpen(index) {
            return index <= 2;
        }
        get _categories() {
            var runToTagInfo = this._runToTagInfo;
            var selectedRuns = this._selectedRuns;
            var tagFilter = this._tagFilter;
            const runToTag = lodash.mapValues(runToTagInfo, (x) => Object.keys(x));
            const baseCategories = categorizeRunTagCombinations(runToTag, selectedRuns, tagFilter);
            function explodeItem(item) {
                const samples = runToTagInfo[item.run][item.tag].samples;
                return lodash.range(samples).map((i) => Object.assign({}, item, {
                    sample: i,
                    totalSamples: samples,
                }));
            }
            const withSamples = baseCategories.map((category) => Object.assign({}, category, {
                items: [].concat.apply([], category.items.map(explodeItem)),
            }));
            return withSamples;
        }
        _tagMetadata(runToTagInfo, run, tag) {
            return runToTagInfo[run][tag];
        }
    };
    TfAudioDashboard.template = html `
    <tf-dashboard-layout>
      <div class="sidebar" slot="sidebar">
        <div class="sidebar-section runs-selector">
          <tf-runs-selector
            id="runs-selector"
            selected-runs="{{_selectedRuns}}"
          ></tf-runs-selector>
        </div>
      </div>
      <div class="center" slot="center">
        <template is="dom-if" if="[[_dataNotFound]]">
          <div class="no-data-warning">
            <h3>No audio data was found.</h3>
            <p>Probable causes:</p>
            <ul>
              <li>You haven’t written any audio data to your event files.</li>
              <li>TensorBoard can’t find your event files.</li>
            </ul>

            <p>
              If you’re new to using TensorBoard, and want to find out how to
              add data and set up your event files, check out the
              <a
                href="https://github.com/tensorflow/tensorboard/blob/master/README.md"
                >README</a
              >
              and perhaps the
              <a
                href="https://www.tensorflow.org/get_started/summaries_and_tensorboard"
                >TensorBoard tutorial</a
              >.
            </p>

            <p>
              If you think TensorBoard is configured properly, please see
              <a
                href="https://github.com/tensorflow/tensorboard/blob/master/README.md#my-tensorboard-isnt-showing-any-data-whats-wrong"
                >the section of the README devoted to missing data problems</a
              >
              and consider filing an issue on GitHub.
            </p>
          </div>
        </template>
        <template is="dom-if" if="[[!_dataNotFound]]">
          <tf-tag-filterer tag-filter="{{_tagFilter}}"></tf-tag-filterer>
          <template is="dom-repeat" items="[[_categories]]" as="category">
            <tf-category-paginated-view
              category="[[category]]"
              initial-opened="[[_shouldOpen(index)]]"
            >
              <template>
                <tf-audio-loader
                  active="[[active]]"
                  run="[[item.run]]"
                  tag="[[item.tag]]"
                  sample="[[item.sample]]"
                  total-samples="[[item.totalSamples]]"
                  tag-metadata="[[_tagMetadata(_runToTagInfo, item.run, item.tag)]]"
                  request-manager="[[_requestManager]]"
                ></tf-audio-loader>
              </template>
            </tf-category-paginated-view>
          </template>
        </template>
      </div>
    </tf-dashboard-layout>
    <style include="dashboard-style"></style>
    <style>
      .no-data-warning {
        max-width: 540px;
        margin: 80px auto 0 auto;
      }
    </style>
  `;
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfAudioDashboard.prototype, "reloadOnReady", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfAudioDashboard.prototype, "_selectedRuns", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfAudioDashboard.prototype, "_runToTagInfo", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfAudioDashboard.prototype, "_dataNotFound", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfAudioDashboard.prototype, "_tagFilter", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", RequestManager)
    ], TfAudioDashboard.prototype, "_requestManager", void 0);
    __decorate([
        computed('_runToTagInfo', '_selectedRuns', '_tagFilter'),
        __metadata("design:type", Array),
        __metadata("design:paramtypes", [])
    ], TfAudioDashboard.prototype, "_categories", null);
    TfAudioDashboard = __decorate([
        customElement('tf-audio-dashboard')
    ], TfAudioDashboard);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    let TfOptionSelector = class TfOptionSelector extends LegacyElementMixin(PolymerElement) {
        attached() {
            this.async(function () {
                this.getEffectiveChildren().forEach(function (node) {
                    this.listen(node, 'tap', '_selectTarget');
                }.bind(this));
            });
        }
        _selectTarget(e) {
            this.selectedId = e.currentTarget.id;
        }
        _selectedIdChanged() {
            var selected = this.queryEffectiveChildren('#' + this.selectedId);
            if (!selected) {
                return;
            }
            this.getEffectiveChildren().forEach(function (node) {
                node.classList.remove('selected');
            });
            selected.classList.add('selected');
        }
    };
    TfOptionSelector.template = html `
    <div id="wrap">
      <h3>[[name]]</h3>
      <div class="content-wrapper"><slot></slot></div>
    </div>
    <style>
      .content-wrapper ::slotted(*) {
        background: none;
        color: var(--tb-ui-dark-accent);
        font-size: 13px;
        margin-top: 10px;
      }

      .content-wrapper ::slotted(*) {
        background: none;
        color: var(--tb-ui-dark-accent);
        font-size: 13px;
        margin-top: 10px;
      }

      .content-wrapper ::slotted(.selected) {
        background-color: var(--tb-ui-dark-accent);
        color: white !important;
      }

      h3 {
        color: var(--paper-grey-800);
        display: block;
        font-size: 14px;
        font-weight: normal;
        margin: 0 0 5px;
        pointer-events: none;
      }
    </style>
  `;
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfOptionSelector.prototype, "name", void 0);
    __decorate([
        property({
            type: String,
            notify: true,
            observer: '_selectedIdChanged',
        }),
        __metadata("design:type", String)
    ], TfOptionSelector.prototype, "selectedId", void 0);
    TfOptionSelector = __decorate([
        customElement('tf-option-selector')
    ], TfOptionSelector);

    /* Copyright 2017 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * Given many occurrences of tag info for a particular tag across
     * multiple runs, create a representative info object. This is useful
     * for plugins that display just one visualization per tag, instead of
     * one per run--tag combination: each run--tag combination can have its
     * own display name or description, so there is a dimension mismatch. We
     * reconcile this as follows:
     *
     *   - We only show a display name if all runs agree. Otherwise, or if
     *     there are no runs, we use the provided `defaultDisplayName`.
     *
     *   - If all runs agree on a description, we use it. Otherwise,
     *     we concatenate all descriptions, annotating which ones
     *     came from which run, and display them in a list.
     *
     * NOTE: Per TensorBoard convention, we assume that the provided
     * `description`s have sanitized HTML and are safe to render into the
     * DOM, while the `displayName` may be an arbitrary string. The output
     * of this function respects this convention as well.
     */
    function aggregateTagInfo(runToTagInfo, defaultDisplayName) {
        let unanimousDisplayName = undefined;
        const descriptionToRuns = {};
        Object.keys(runToTagInfo).forEach((run) => {
            const info = runToTagInfo[run];
            if (unanimousDisplayName === undefined) {
                unanimousDisplayName = info.displayName;
            }
            if (unanimousDisplayName !== info.displayName) {
                unanimousDisplayName = null;
            }
            if (descriptionToRuns[info.description] === undefined) {
                descriptionToRuns[info.description] = [];
            }
            descriptionToRuns[info.description].push(run);
        });
        const displayName = unanimousDisplayName != null ? unanimousDisplayName : defaultDisplayName;
        const description = (() => {
            const descriptions = Object.keys(descriptionToRuns);
            if (descriptions.length === 0) {
                return '';
            }
            else if (descriptions.length === 1) {
                return descriptions[0];
            }
            else {
                const items = descriptions.map((description) => {
                    const runs = descriptionToRuns[description].map((run) => {
                        // We're splicing potentially unsafe display names into
                        // sanitized descriptions, so we need to sanitize them.
                        const safeRun = run
                            .replace(/</g, '&lt;')
                            .replace(/>/g, '&gt;') // for symmetry :-)
                            .replace(/&/g, '&amp;');
                        return `<code>${safeRun}</code>`;
                    });
                    const joined = runs.length > 2
                        ? runs.slice(0, runs.length - 1).join(', ') +
                            ', and ' +
                            runs[runs.length - 1]
                        : runs.join(' and ');
                    const runNoun = ngettext(runs.length, 'run', 'runs');
                    return `<li><p>For ${runNoun} ${joined}:</p>${description}</li>`;
                });
                const prefix = '<p><strong>Multiple descriptions:</strong></p>';
                return `${prefix}<ul>${items.join('')}</ul>`;
            }
        })();
        return { displayName, description };
    }
    function ngettext(k, enSingular, enPlural) {
        // Potential extension point for proper i18n infrastructure, if we
        // ever implement it.
        return k === 1 ? enSingular : enPlural;
    }

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * tf-smoothing-input creates an input component for exponential smoothing.
     */
    let TfSmoothingInput = 
    // tslint:disable-next-line:no-unused-variable
    class TfSmoothingInput extends PolymerElement {
        constructor() {
            super(...arguments);
            this.weight = 0.6;
            this._updateWeight = lodash.debounce(function (val) {
                this.weight = val;
            }, 250);
        }
        _immediateWeightNumberForPaperSliderChanged() {
            this._inputWeightStringForPaperInput = this._immediateWeightNumberForPaperSlider.toString();
            this._updateWeight.call(this, this._immediateWeightNumberForPaperSlider);
        }
        _inputWeightStringForPaperInputChanged() {
            if (+this._inputWeightStringForPaperInput < 0) {
                this._inputWeightStringForPaperInput = '0';
            }
            else if (+this._inputWeightStringForPaperInput > 1) {
                this._inputWeightStringForPaperInput = '1';
            }
            var d = +this._inputWeightStringForPaperInput;
            if (!isNaN(d)) {
                this._updateWeight.call(this, d);
            }
        }
    };
    TfSmoothingInput.template = html `
    <h3 class="title">Smoothing</h3>
    <div class="smoothing-block">
      <paper-slider
        id="slider"
        immediate-value="{{_immediateWeightNumberForPaperSlider}}"
        max="[[max]]"
        min="[[min]]"
        pin
        step="[[step]]"
        type="number"
        value="{{weight}}"
      ></paper-slider>
      <paper-input
        id="input"
        label="weight"
        no-label-float
        value="{{_inputWeightStringForPaperInput}}"
        type="number"
        step="[[step]]"
        min="[[min]]"
        max="[[max]]"
      ></paper-input>
    </div>
    <style>
      .title {
        color: var(--paper-grey-800);
        margin: 0;
        font-weight: normal;
        font-size: 14px;
        margin-bottom: 5px;
      }

      .smoothing-block {
        display: flex;
      }

      paper-slider {
        --paper-slider-active-color: var(--tb-orange-strong);
        --paper-slider-knob-color: var(--tb-orange-strong);
        --paper-slider-knob-start-border-color: var(--tb-orange-strong);
        --paper-slider-knob-start-color: var(--tb-orange-strong);
        --paper-slider-markers-color: var(--tb-orange-strong);
        --paper-slider-pin-color: var(--tb-orange-strong);
        --paper-slider-pin-start-color: var(--tb-orange-strong);
        flex-grow: 2;
      }

      paper-input {
        --paper-input-container-focus-color: var(--tb-orange-strong);
        --paper-input-container-input: {
          font-size: 14px;
        }
        --paper-input-container-label: {
          font-size: 14px;
        }
        width: 60px;
      }
    </style>
  `;
    __decorate([
        property({ type: Number }),
        __metadata("design:type", Number)
    ], TfSmoothingInput.prototype, "step", void 0);
    __decorate([
        property({ type: Number }),
        __metadata("design:type", Number)
    ], TfSmoothingInput.prototype, "max", void 0);
    __decorate([
        property({ type: Number }),
        __metadata("design:type", Number)
    ], TfSmoothingInput.prototype, "min", void 0);
    __decorate([
        property({
            type: Number,
            notify: true,
        }),
        __metadata("design:type", Number)
    ], TfSmoothingInput.prototype, "weight", void 0);
    __decorate([
        property({
            type: Number,
            notify: true,
            observer: '_immediateWeightNumberForPaperSliderChanged',
        }),
        __metadata("design:type", Number)
    ], TfSmoothingInput.prototype, "_immediateWeightNumberForPaperSlider", void 0);
    __decorate([
        property({
            type: String,
            notify: true,
            observer: '_inputWeightStringForPaperInputChanged',
        }),
        __metadata("design:type", String)
    ], TfSmoothingInput.prototype, "_inputWeightStringForPaperInput", void 0);
    TfSmoothingInput = __decorate([
        customElement('tf-smoothing-input')
        // tslint:disable-next-line:no-unused-variable
    ], TfSmoothingInput);

    /* Copyright 2015 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the 'License');
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an 'AS IS' BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * Add query parameters to a URL. Values will be URL-encoded. The URL
     * may or may not already have query parameters. For convenience,
     * parameters whose value is `undefined` will be dropped.
     *
     * For example, the following expressions are equivalent:
     *
     *     addParams("http://foo", {a: "1", b: ["2", "3+4"], c: "5"})
     *     addParams("http://foo?a=1", {b: ["2", "3+4"], c: "5", d: undefined})
     *     "http://foo?a=1&b=2&b=3%2B4&c=5"
     *
     * @deprecated If used with `router.pluginRoute`, please use the queryParams
     * argument.
     */
    function addParams(baseURL, params) {
        const keys = Object.keys(params)
            .sort()
            .filter((k) => params[k] !== undefined);
        if (!keys.length) {
            return baseURL; // no need to change '/foo' to '/foo?'
        }
        const delimiter = baseURL.indexOf('?') !== -1 ? '&' : '?';
        const parts = [].concat(...keys.map((key) => {
            const rawValue = params[key];
            const values = Array.isArray(rawValue) ? rawValue : [rawValue];
            return values.map((value) => `${key}=${_encodeURIComponent(value)}`);
        }));
        const query = parts.join('&');
        return baseURL + delimiter + query;
    }
    function _encodeURIComponent(x) {
        // Replace parentheses for consistency with Python's urllib.urlencode.
        return encodeURIComponent(x).replace(/\(/g, '%28').replace(/\)/g, '%29');
    }

    /*! *****************************************************************************
    Copyright (c) Microsoft Corporation. All rights reserved.
    Licensed under the Apache License, Version 2.0 (the "License"); you may not use
    this file except in compliance with the License. You may obtain a copy of the
    License at http://www.apache.org/licenses/LICENSE-2.0

    THIS CODE IS PROVIDED ON AN *AS IS* BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
    KIND, EITHER EXPRESS OR IMPLIED, INCLUDING WITHOUT LIMITATION ANY IMPLIED
    WARRANTIES OR CONDITIONS OF TITLE, FITNESS FOR A PARTICULAR PURPOSE,
    MERCHANTABLITY OR NON-INFRINGEMENT.

    See the Apache Version 2.0 License for specific language governing permissions
    and limitations under the License.
    ***************************************************************************** */
    /* global Reflect, Promise */

    var extendStatics = Object.setPrototypeOf ||
        ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) ||
        function (d, b) { for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p]; };

    function __extends(d, b) {
        extendStatics(d, b);
        function __() { this.constructor = d; }
        d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __());
    }

    var __assign = Object.assign || function __assign(t) {
        for (var s, i = 1, n = arguments.length; i < n; i++) {
            s = arguments[i];
            for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p)) t[p] = s[p];
        }
        return t;
    };

    function __rest(s, e) {
        var t = {};
        for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p) && e.indexOf(p) < 0)
            t[p] = s[p];
        if (s != null && typeof Object.getOwnPropertySymbols === "function")
            for (var i = 0, p = Object.getOwnPropertySymbols(s); i < p.length; i++) if (e.indexOf(p[i]) < 0)
                t[p[i]] = s[p[i]];
        return t;
    }

    function __decorate$1(decorators, target, key, desc) {
        var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
        if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
        else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
        return c > 3 && r && Object.defineProperty(target, key, r), r;
    }

    function __param(paramIndex, decorator) {
        return function (target, key) { decorator(target, key, paramIndex); }
    }

    function __metadata$1(metadataKey, metadataValue) {
        if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(metadataKey, metadataValue);
    }

    function __awaiter$1(thisArg, _arguments, P, generator) {
        return new (P || (P = Promise))(function (resolve, reject) {
            function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
            function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
            function step(result) { result.done ? resolve(result.value) : new P(function (resolve) { resolve(result.value); }).then(fulfilled, rejected); }
            step((generator = generator.apply(thisArg, _arguments || [])).next());
        });
    }

    function __generator(thisArg, body) {
        var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g;
        return g = { next: verb(0), "throw": verb(1), "return": verb(2) }, typeof Symbol === "function" && (g[Symbol.iterator] = function() { return this; }), g;
        function verb(n) { return function (v) { return step([n, v]); }; }
        function step(op) {
            if (f) throw new TypeError("Generator is already executing.");
            while (_) try {
                if (f = 1, y && (t = y[op[0] & 2 ? "return" : op[0] ? "throw" : "next"]) && !(t = t.call(y, op[1])).done) return t;
                if (y = 0, t) op = [0, t.value];
                switch (op[0]) {
                    case 0: case 1: t = op; break;
                    case 4: _.label++; return { value: op[1], done: false };
                    case 5: _.label++; y = op[1]; op = [0]; continue;
                    case 7: op = _.ops.pop(); _.trys.pop(); continue;
                    default:
                        if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }
                        if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }
                        if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }
                        if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }
                        if (t[2]) _.ops.pop();
                        _.trys.pop(); continue;
                }
                op = body.call(thisArg, _);
            } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }
            if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };
        }
    }

    function __exportStar(m, exports) {
        for (var p in m) if (!exports.hasOwnProperty(p)) exports[p] = m[p];
    }

    function __values(o) {
        var m = typeof Symbol === "function" && o[Symbol.iterator], i = 0;
        if (m) return m.call(o);
        return {
            next: function () {
                if (o && i >= o.length) o = void 0;
                return { value: o && o[i++], done: !o };
            }
        };
    }

    function __read(o, n) {
        var m = typeof Symbol === "function" && o[Symbol.iterator];
        if (!m) return o;
        var i = m.call(o), r, ar = [], e;
        try {
            while ((n === void 0 || n-- > 0) && !(r = i.next()).done) ar.push(r.value);
        }
        catch (error) { e = { error: error }; }
        finally {
            try {
                if (r && !r.done && (m = i["return"])) m.call(i);
            }
            finally { if (e) throw e.error; }
        }
        return ar;
    }

    function __spread() {
        for (var ar = [], i = 0; i < arguments.length; i++)
            ar = ar.concat(__read(arguments[i]));
        return ar;
    }

    function __await(v) {
        return this instanceof __await ? (this.v = v, this) : new __await(v);
    }

    function __asyncGenerator(thisArg, _arguments, generator) {
        if (!Symbol.asyncIterator) throw new TypeError("Symbol.asyncIterator is not defined.");
        var g = generator.apply(thisArg, _arguments || []), i, q = [];
        return i = {}, verb("next"), verb("throw"), verb("return"), i[Symbol.asyncIterator] = function () { return this; }, i;
        function verb(n) { if (g[n]) i[n] = function (v) { return new Promise(function (a, b) { q.push([n, v, a, b]) > 1 || resume(n, v); }); }; }
        function resume(n, v) { try { step(g[n](v)); } catch (e) { settle(q[0][3], e); } }
        function step(r) { r.value instanceof __await ? Promise.resolve(r.value.v).then(fulfill, reject) : settle(q[0][2], r);  }
        function fulfill(value) { resume("next", value); }
        function reject(value) { resume("throw", value); }
        function settle(f, v) { if (f(v), q.shift(), q.length) resume(q[0][0], q[0][1]); }
    }

    function __asyncDelegator(o) {
        var i, p;
        return i = {}, verb("next"), verb("throw", function (e) { throw e; }), verb("return"), i[Symbol.iterator] = function () { return this; }, i;
        function verb(n, f) { if (o[n]) i[n] = function (v) { return (p = !p) ? { value: __await(o[n](v)), done: n === "return" } : f ? f(v) : v; }; }
    }

    function __asyncValues(o) {
        if (!Symbol.asyncIterator) throw new TypeError("Symbol.asyncIterator is not defined.");
        var m = o[Symbol.asyncIterator];
        return m ? m.call(o) : typeof __values === "function" ? __values(o) : o[Symbol.iterator]();
    }

    function __makeTemplateObject(cooked, raw) {
        if (Object.defineProperty) { Object.defineProperty(cooked, "raw", { value: raw }); } else { cooked.raw = raw; }
        return cooked;
    }

    var tslib_es6 = /*#__PURE__*/Object.freeze({
        __proto__: null,
        __extends: __extends,
        __assign: __assign,
        __rest: __rest,
        __decorate: __decorate$1,
        __param: __param,
        __metadata: __metadata$1,
        __awaiter: __awaiter$1,
        __generator: __generator,
        __exportStar: __exportStar,
        __values: __values,
        __read: __read,
        __spread: __spread,
        __await: __await,
        __asyncGenerator: __asyncGenerator,
        __asyncDelegator: __asyncDelegator,
        __asyncValues: __asyncValues,
        __makeTemplateObject: __makeTemplateObject
    });

    var version$2 = "4.13.0";

    function ascending$4(a, b) {
      return a < b ? -1 : a > b ? 1 : a >= b ? 0 : NaN;
    }

    function bisector$1(compare) {
      if (compare.length === 1) compare = ascendingComparator$1(compare);
      return {
        left: function(a, x, lo, hi) {
          if (lo == null) lo = 0;
          if (hi == null) hi = a.length;
          while (lo < hi) {
            var mid = lo + hi >>> 1;
            if (compare(a[mid], x) < 0) lo = mid + 1;
            else hi = mid;
          }
          return lo;
        },
        right: function(a, x, lo, hi) {
          if (lo == null) lo = 0;
          if (hi == null) hi = a.length;
          while (lo < hi) {
            var mid = lo + hi >>> 1;
            if (compare(a[mid], x) > 0) hi = mid;
            else lo = mid + 1;
          }
          return lo;
        }
      };
    }

    function ascendingComparator$1(f) {
      return function(d, x) {
        return ascending$4(f(d), x);
      };
    }

    var ascendingBisect$1 = bisector$1(ascending$4);
    var bisectRight$1 = ascendingBisect$1.right;
    var bisectLeft$1 = ascendingBisect$1.left;

    function pairs$1(array, f) {
      if (f == null) f = pair$1;
      var i = 0, n = array.length - 1, p = array[0], pairs = new Array(n < 0 ? 0 : n);
      while (i < n) pairs[i] = f(p, p = array[++i]);
      return pairs;
    }

    function pair$1(a, b) {
      return [a, b];
    }

    function cross$3(values0, values1, reduce) {
      var n0 = values0.length,
          n1 = values1.length,
          values = new Array(n0 * n1),
          i0,
          i1,
          i,
          value0;

      if (reduce == null) reduce = pair$1;

      for (i0 = i = 0; i0 < n0; ++i0) {
        for (value0 = values0[i0], i1 = 0; i1 < n1; ++i1, ++i) {
          values[i] = reduce(value0, values1[i1]);
        }
      }

      return values;
    }

    function descending$3(a, b) {
      return b < a ? -1 : b > a ? 1 : b >= a ? 0 : NaN;
    }

    function number$4(x) {
      return x === null ? NaN : +x;
    }

    function variance$1(values, valueof) {
      var n = values.length,
          m = 0,
          i = -1,
          mean = 0,
          value,
          delta,
          sum = 0;

      if (valueof == null) {
        while (++i < n) {
          if (!isNaN(value = number$4(values[i]))) {
            delta = value - mean;
            mean += delta / ++m;
            sum += delta * (value - mean);
          }
        }
      }

      else {
        while (++i < n) {
          if (!isNaN(value = number$4(valueof(values[i], i, values)))) {
            delta = value - mean;
            mean += delta / ++m;
            sum += delta * (value - mean);
          }
        }
      }

      if (m > 1) return sum / (m - 1);
    }

    function deviation$1(array, f) {
      var v = variance$1(array, f);
      return v ? Math.sqrt(v) : v;
    }

    function extent$2(values, valueof) {
      var n = values.length,
          i = -1,
          value,
          min,
          max;

      if (valueof == null) {
        while (++i < n) { // Find the first comparable value.
          if ((value = values[i]) != null && value >= value) {
            min = max = value;
            while (++i < n) { // Compare the remaining values.
              if ((value = values[i]) != null) {
                if (min > value) min = value;
                if (max < value) max = value;
              }
            }
          }
        }
      }

      else {
        while (++i < n) { // Find the first comparable value.
          if ((value = valueof(values[i], i, values)) != null && value >= value) {
            min = max = value;
            while (++i < n) { // Compare the remaining values.
              if ((value = valueof(values[i], i, values)) != null) {
                if (min > value) min = value;
                if (max < value) max = value;
              }
            }
          }
        }
      }

      return [min, max];
    }

    var array$4 = Array.prototype;

    var slice$7 = array$4.slice;
    var map$4 = array$4.map;

    function constant$e(x) {
      return function() {
        return x;
      };
    }

    function identity$a(x) {
      return x;
    }

    function range$1(start, stop, step) {
      start = +start, stop = +stop, step = (n = arguments.length) < 2 ? (stop = start, start = 0, 1) : n < 3 ? 1 : +step;

      var i = -1,
          n = Math.max(0, Math.ceil((stop - start) / step)) | 0,
          range = new Array(n);

      while (++i < n) {
        range[i] = start + i * step;
      }

      return range;
    }

    var e10$1 = Math.sqrt(50),
        e5$1 = Math.sqrt(10),
        e2$1 = Math.sqrt(2);

    function ticks$1(start, stop, count) {
      var reverse,
          i = -1,
          n,
          ticks,
          step;

      stop = +stop, start = +start, count = +count;
      if (start === stop && count > 0) return [start];
      if (reverse = stop < start) n = start, start = stop, stop = n;
      if ((step = tickIncrement$1(start, stop, count)) === 0 || !isFinite(step)) return [];

      if (step > 0) {
        start = Math.ceil(start / step);
        stop = Math.floor(stop / step);
        ticks = new Array(n = Math.ceil(stop - start + 1));
        while (++i < n) ticks[i] = (start + i) * step;
      } else {
        start = Math.floor(start * step);
        stop = Math.ceil(stop * step);
        ticks = new Array(n = Math.ceil(start - stop + 1));
        while (++i < n) ticks[i] = (start - i) / step;
      }

      if (reverse) ticks.reverse();

      return ticks;
    }

    function tickIncrement$1(start, stop, count) {
      var step = (stop - start) / Math.max(0, count),
          power = Math.floor(Math.log(step) / Math.LN10),
          error = step / Math.pow(10, power);
      return power >= 0
          ? (error >= e10$1 ? 10 : error >= e5$1 ? 5 : error >= e2$1 ? 2 : 1) * Math.pow(10, power)
          : -Math.pow(10, -power) / (error >= e10$1 ? 10 : error >= e5$1 ? 5 : error >= e2$1 ? 2 : 1);
    }

    function tickStep$1(start, stop, count) {
      var step0 = Math.abs(stop - start) / Math.max(0, count),
          step1 = Math.pow(10, Math.floor(Math.log(step0) / Math.LN10)),
          error = step0 / step1;
      if (error >= e10$1) step1 *= 10;
      else if (error >= e5$1) step1 *= 5;
      else if (error >= e2$1) step1 *= 2;
      return stop < start ? -step1 : step1;
    }

    function sturges(values) {
      return Math.ceil(Math.log(values.length) / Math.LN2) + 1;
    }

    function histogram$1() {
      var value = identity$a,
          domain = extent$2,
          threshold = sturges;

      function histogram(data) {
        var i,
            n = data.length,
            x,
            values = new Array(n);

        for (i = 0; i < n; ++i) {
          values[i] = value(data[i], i, data);
        }

        var xz = domain(values),
            x0 = xz[0],
            x1 = xz[1],
            tz = threshold(values, x0, x1);

        // Convert number of thresholds into uniform thresholds.
        if (!Array.isArray(tz)) {
          tz = tickStep$1(x0, x1, tz);
          tz = range$1(Math.ceil(x0 / tz) * tz, Math.floor(x1 / tz) * tz, tz); // exclusive
        }

        // Remove any thresholds outside the domain.
        var m = tz.length;
        while (tz[0] <= x0) tz.shift(), --m;
        while (tz[m - 1] > x1) tz.pop(), --m;

        var bins = new Array(m + 1),
            bin;

        // Initialize bins.
        for (i = 0; i <= m; ++i) {
          bin = bins[i] = [];
          bin.x0 = i > 0 ? tz[i - 1] : x0;
          bin.x1 = i < m ? tz[i] : x1;
        }

        // Assign data to bins by value, ignoring any outside the domain.
        for (i = 0; i < n; ++i) {
          x = values[i];
          if (x0 <= x && x <= x1) {
            bins[bisectRight$1(tz, x, 0, m)].push(data[i]);
          }
        }

        return bins;
      }

      histogram.value = function(_) {
        return arguments.length ? (value = typeof _ === "function" ? _ : constant$e(_), histogram) : value;
      };

      histogram.domain = function(_) {
        return arguments.length ? (domain = typeof _ === "function" ? _ : constant$e([_[0], _[1]]), histogram) : domain;
      };

      histogram.thresholds = function(_) {
        return arguments.length ? (threshold = typeof _ === "function" ? _ : Array.isArray(_) ? constant$e(slice$7.call(_)) : constant$e(_), histogram) : threshold;
      };

      return histogram;
    }

    function quantile$1(values, p, valueof) {
      if (valueof == null) valueof = number$4;
      if (!(n = values.length)) return;
      if ((p = +p) <= 0 || n < 2) return +valueof(values[0], 0, values);
      if (p >= 1) return +valueof(values[n - 1], n - 1, values);
      var n,
          i = (n - 1) * p,
          i0 = Math.floor(i),
          value0 = +valueof(values[i0], i0, values),
          value1 = +valueof(values[i0 + 1], i0 + 1, values);
      return value0 + (value1 - value0) * (i - i0);
    }

    function freedmanDiaconis$1(values, min, max) {
      values = map$4.call(values, number$4).sort(ascending$4);
      return Math.ceil((max - min) / (2 * (quantile$1(values, 0.75) - quantile$1(values, 0.25)) * Math.pow(values.length, -1 / 3)));
    }

    function scott$1(values, min, max) {
      return Math.ceil((max - min) / (3.5 * deviation$1(values) * Math.pow(values.length, -1 / 3)));
    }

    function max$3(values, valueof) {
      var n = values.length,
          i = -1,
          value,
          max;

      if (valueof == null) {
        while (++i < n) { // Find the first comparable value.
          if ((value = values[i]) != null && value >= value) {
            max = value;
            while (++i < n) { // Compare the remaining values.
              if ((value = values[i]) != null && value > max) {
                max = value;
              }
            }
          }
        }
      }

      else {
        while (++i < n) { // Find the first comparable value.
          if ((value = valueof(values[i], i, values)) != null && value >= value) {
            max = value;
            while (++i < n) { // Compare the remaining values.
              if ((value = valueof(values[i], i, values)) != null && value > max) {
                max = value;
              }
            }
          }
        }
      }

      return max;
    }

    function mean$1(values, valueof) {
      var n = values.length,
          m = n,
          i = -1,
          value,
          sum = 0;

      if (valueof == null) {
        while (++i < n) {
          if (!isNaN(value = number$4(values[i]))) sum += value;
          else --m;
        }
      }

      else {
        while (++i < n) {
          if (!isNaN(value = number$4(valueof(values[i], i, values)))) sum += value;
          else --m;
        }
      }

      if (m) return sum / m;
    }

    function median$1(values, valueof) {
      var n = values.length,
          i = -1,
          value,
          numbers = [];

      if (valueof == null) {
        while (++i < n) {
          if (!isNaN(value = number$4(values[i]))) {
            numbers.push(value);
          }
        }
      }

      else {
        while (++i < n) {
          if (!isNaN(value = number$4(valueof(values[i], i, values)))) {
            numbers.push(value);
          }
        }
      }

      return quantile$1(numbers.sort(ascending$4), 0.5);
    }

    function merge$1(arrays) {
      var n = arrays.length,
          m,
          i = -1,
          j = 0,
          merged,
          array;

      while (++i < n) j += arrays[i].length;
      merged = new Array(j);

      while (--n >= 0) {
        array = arrays[n];
        m = array.length;
        while (--m >= 0) {
          merged[--j] = array[m];
        }
      }

      return merged;
    }

    function min$2(values, valueof) {
      var n = values.length,
          i = -1,
          value,
          min;

      if (valueof == null) {
        while (++i < n) { // Find the first comparable value.
          if ((value = values[i]) != null && value >= value) {
            min = value;
            while (++i < n) { // Compare the remaining values.
              if ((value = values[i]) != null && min > value) {
                min = value;
              }
            }
          }
        }
      }

      else {
        while (++i < n) { // Find the first comparable value.
          if ((value = valueof(values[i], i, values)) != null && value >= value) {
            min = value;
            while (++i < n) { // Compare the remaining values.
              if ((value = valueof(values[i], i, values)) != null && min > value) {
                min = value;
              }
            }
          }
        }
      }

      return min;
    }

    function permute$1(array, indexes) {
      var i = indexes.length, permutes = new Array(i);
      while (i--) permutes[i] = array[indexes[i]];
      return permutes;
    }

    function scan$1(values, compare) {
      if (!(n = values.length)) return;
      var n,
          i = 0,
          j = 0,
          xi,
          xj = values[j];

      if (compare == null) compare = ascending$4;

      while (++i < n) {
        if (compare(xi = values[i], xj) < 0 || compare(xj, xj) !== 0) {
          xj = xi, j = i;
        }
      }

      if (compare(xj, xj) === 0) return j;
    }

    function shuffle$2(array, i0, i1) {
      var m = (i1 == null ? array.length : i1) - (i0 = i0 == null ? 0 : +i0),
          t,
          i;

      while (m) {
        i = Math.random() * m-- | 0;
        t = array[m + i0];
        array[m + i0] = array[i + i0];
        array[i + i0] = t;
      }

      return array;
    }

    function sum$3(values, valueof) {
      var n = values.length,
          i = -1,
          value,
          sum = 0;

      if (valueof == null) {
        while (++i < n) {
          if (value = +values[i]) sum += value; // Note: zero and null are equivalent.
        }
      }

      else {
        while (++i < n) {
          if (value = +valueof(values[i], i, values)) sum += value;
        }
      }

      return sum;
    }

    function transpose$1(matrix) {
      if (!(n = matrix.length)) return [];
      for (var i = -1, m = min$2(matrix, length$3), transpose = new Array(m); ++i < m;) {
        for (var j = -1, n, row = transpose[i] = new Array(n); ++j < n;) {
          row[j] = matrix[j][i];
        }
      }
      return transpose;
    }

    function length$3(d) {
      return d.length;
    }

    function zip$1() {
      return transpose$1(arguments);
    }

    var slice$8 = Array.prototype.slice;

    function identity$b(x) {
      return x;
    }

    var top$1 = 1,
        right$1 = 2,
        bottom$1 = 3,
        left$1 = 4,
        epsilon$5 = 1e-6;

    function translateX$1(x) {
      return "translate(" + (x + 0.5) + ",0)";
    }

    function translateY$1(y) {
      return "translate(0," + (y + 0.5) + ")";
    }

    function number$5(scale) {
      return function(d) {
        return +scale(d);
      };
    }

    function center$2(scale) {
      var offset = Math.max(0, scale.bandwidth() - 1) / 2; // Adjust for 0.5px offset.
      if (scale.round()) offset = Math.round(offset);
      return function(d) {
        return +scale(d) + offset;
      };
    }

    function entering$1() {
      return !this.__axis;
    }

    function axis$1(orient, scale) {
      var tickArguments = [],
          tickValues = null,
          tickFormat = null,
          tickSizeInner = 6,
          tickSizeOuter = 6,
          tickPadding = 3,
          k = orient === top$1 || orient === left$1 ? -1 : 1,
          x = orient === left$1 || orient === right$1 ? "x" : "y",
          transform = orient === top$1 || orient === bottom$1 ? translateX$1 : translateY$1;

      function axis(context) {
        var values = tickValues == null ? (scale.ticks ? scale.ticks.apply(scale, tickArguments) : scale.domain()) : tickValues,
            format = tickFormat == null ? (scale.tickFormat ? scale.tickFormat.apply(scale, tickArguments) : identity$b) : tickFormat,
            spacing = Math.max(tickSizeInner, 0) + tickPadding,
            range = scale.range(),
            range0 = +range[0] + 0.5,
            range1 = +range[range.length - 1] + 0.5,
            position = (scale.bandwidth ? center$2 : number$5)(scale.copy()),
            selection = context.selection ? context.selection() : context,
            path = selection.selectAll(".domain").data([null]),
            tick = selection.selectAll(".tick").data(values, scale).order(),
            tickExit = tick.exit(),
            tickEnter = tick.enter().append("g").attr("class", "tick"),
            line = tick.select("line"),
            text = tick.select("text");

        path = path.merge(path.enter().insert("path", ".tick")
            .attr("class", "domain")
            .attr("stroke", "#000"));

        tick = tick.merge(tickEnter);

        line = line.merge(tickEnter.append("line")
            .attr("stroke", "#000")
            .attr(x + "2", k * tickSizeInner));

        text = text.merge(tickEnter.append("text")
            .attr("fill", "#000")
            .attr(x, k * spacing)
            .attr("dy", orient === top$1 ? "0em" : orient === bottom$1 ? "0.71em" : "0.32em"));

        if (context !== selection) {
          path = path.transition(context);
          tick = tick.transition(context);
          line = line.transition(context);
          text = text.transition(context);

          tickExit = tickExit.transition(context)
              .attr("opacity", epsilon$5)
              .attr("transform", function(d) { return isFinite(d = position(d)) ? transform(d) : this.getAttribute("transform"); });

          tickEnter
              .attr("opacity", epsilon$5)
              .attr("transform", function(d) { var p = this.parentNode.__axis; return transform(p && isFinite(p = p(d)) ? p : position(d)); });
        }

        tickExit.remove();

        path
            .attr("d", orient === left$1 || orient == right$1
                ? "M" + k * tickSizeOuter + "," + range0 + "H0.5V" + range1 + "H" + k * tickSizeOuter
                : "M" + range0 + "," + k * tickSizeOuter + "V0.5H" + range1 + "V" + k * tickSizeOuter);

        tick
            .attr("opacity", 1)
            .attr("transform", function(d) { return transform(position(d)); });

        line
            .attr(x + "2", k * tickSizeInner);

        text
            .attr(x, k * spacing)
            .text(format);

        selection.filter(entering$1)
            .attr("fill", "none")
            .attr("font-size", 10)
            .attr("font-family", "sans-serif")
            .attr("text-anchor", orient === right$1 ? "start" : orient === left$1 ? "end" : "middle");

        selection
            .each(function() { this.__axis = position; });
      }

      axis.scale = function(_) {
        return arguments.length ? (scale = _, axis) : scale;
      };

      axis.ticks = function() {
        return tickArguments = slice$8.call(arguments), axis;
      };

      axis.tickArguments = function(_) {
        return arguments.length ? (tickArguments = _ == null ? [] : slice$8.call(_), axis) : tickArguments.slice();
      };

      axis.tickValues = function(_) {
        return arguments.length ? (tickValues = _ == null ? null : slice$8.call(_), axis) : tickValues && tickValues.slice();
      };

      axis.tickFormat = function(_) {
        return arguments.length ? (tickFormat = _, axis) : tickFormat;
      };

      axis.tickSize = function(_) {
        return arguments.length ? (tickSizeInner = tickSizeOuter = +_, axis) : tickSizeInner;
      };

      axis.tickSizeInner = function(_) {
        return arguments.length ? (tickSizeInner = +_, axis) : tickSizeInner;
      };

      axis.tickSizeOuter = function(_) {
        return arguments.length ? (tickSizeOuter = +_, axis) : tickSizeOuter;
      };

      axis.tickPadding = function(_) {
        return arguments.length ? (tickPadding = +_, axis) : tickPadding;
      };

      return axis;
    }

    function axisTop$1(scale) {
      return axis$1(top$1, scale);
    }

    function axisRight$1(scale) {
      return axis$1(right$1, scale);
    }

    function axisBottom$1(scale) {
      return axis$1(bottom$1, scale);
    }

    function axisLeft$1(scale) {
      return axis$1(left$1, scale);
    }

    var noop$4 = {value: function() {}};

    function dispatch$1() {
      for (var i = 0, n = arguments.length, _ = {}, t; i < n; ++i) {
        if (!(t = arguments[i] + "") || (t in _) || /[\s.]/.test(t)) throw new Error("illegal type: " + t);
        _[t] = [];
      }
      return new Dispatch$1(_);
    }

    function Dispatch$1(_) {
      this._ = _;
    }

    function parseTypenames$2(typenames, types) {
      return typenames.trim().split(/^|\s+/).map(function(t) {
        var name = "", i = t.indexOf(".");
        if (i >= 0) name = t.slice(i + 1), t = t.slice(0, i);
        if (t && !types.hasOwnProperty(t)) throw new Error("unknown type: " + t);
        return {type: t, name: name};
      });
    }

    Dispatch$1.prototype = dispatch$1.prototype = {
      constructor: Dispatch$1,
      on: function(typename, callback) {
        var _ = this._,
            T = parseTypenames$2(typename + "", _),
            t,
            i = -1,
            n = T.length;

        // If no callback was specified, return the callback of the given type and name.
        if (arguments.length < 2) {
          while (++i < n) if ((t = (typename = T[i]).type) && (t = get$3(_[t], typename.name))) return t;
          return;
        }

        // If a type was specified, set the callback for the given type and name.
        // Otherwise, if a null callback was specified, remove callbacks of the given name.
        if (callback != null && typeof callback !== "function") throw new Error("invalid callback: " + callback);
        while (++i < n) {
          if (t = (typename = T[i]).type) _[t] = set$4(_[t], typename.name, callback);
          else if (callback == null) for (t in _) _[t] = set$4(_[t], typename.name, null);
        }

        return this;
      },
      copy: function() {
        var copy = {}, _ = this._;
        for (var t in _) copy[t] = _[t].slice();
        return new Dispatch$1(copy);
      },
      call: function(type, that) {
        if ((n = arguments.length - 2) > 0) for (var args = new Array(n), i = 0, n, t; i < n; ++i) args[i] = arguments[i + 2];
        if (!this._.hasOwnProperty(type)) throw new Error("unknown type: " + type);
        for (t = this._[type], i = 0, n = t.length; i < n; ++i) t[i].value.apply(that, args);
      },
      apply: function(type, that, args) {
        if (!this._.hasOwnProperty(type)) throw new Error("unknown type: " + type);
        for (var t = this._[type], i = 0, n = t.length; i < n; ++i) t[i].value.apply(that, args);
      }
    };

    function get$3(type, name) {
      for (var i = 0, n = type.length, c; i < n; ++i) {
        if ((c = type[i]).name === name) {
          return c.value;
        }
      }
    }

    function set$4(type, name, callback) {
      for (var i = 0, n = type.length; i < n; ++i) {
        if (type[i].name === name) {
          type[i] = noop$4, type = type.slice(0, i).concat(type.slice(i + 1));
          break;
        }
      }
      if (callback != null) type.push({name: name, value: callback});
      return type;
    }

    var xhtml$1 = "http://www.w3.org/1999/xhtml";

    var namespaces$1 = {
      svg: "http://www.w3.org/2000/svg",
      xhtml: xhtml$1,
      xlink: "http://www.w3.org/1999/xlink",
      xml: "http://www.w3.org/XML/1998/namespace",
      xmlns: "http://www.w3.org/2000/xmlns/"
    };

    function namespace$1(name) {
      var prefix = name += "", i = prefix.indexOf(":");
      if (i >= 0 && (prefix = name.slice(0, i)) !== "xmlns") name = name.slice(i + 1);
      return namespaces$1.hasOwnProperty(prefix) ? {space: namespaces$1[prefix], local: name} : name;
    }

    function creatorInherit$1(name) {
      return function() {
        var document = this.ownerDocument,
            uri = this.namespaceURI;
        return uri === xhtml$1 && document.documentElement.namespaceURI === xhtml$1
            ? document.createElement(name)
            : document.createElementNS(uri, name);
      };
    }

    function creatorFixed$1(fullname) {
      return function() {
        return this.ownerDocument.createElementNS(fullname.space, fullname.local);
      };
    }

    function creator$1(name) {
      var fullname = namespace$1(name);
      return (fullname.local
          ? creatorFixed$1
          : creatorInherit$1)(fullname);
    }

    function none$3() {}

    function selector$1(selector) {
      return selector == null ? none$3 : function() {
        return this.querySelector(selector);
      };
    }

    function selection_select$1(select) {
      if (typeof select !== "function") select = selector$1(select);

      for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, subgroup = subgroups[j] = new Array(n), node, subnode, i = 0; i < n; ++i) {
          if ((node = group[i]) && (subnode = select.call(node, node.__data__, i, group))) {
            if ("__data__" in node) subnode.__data__ = node.__data__;
            subgroup[i] = subnode;
          }
        }
      }

      return new Selection$2(subgroups, this._parents);
    }

    function empty$2() {
      return [];
    }

    function selectorAll$1(selector) {
      return selector == null ? empty$2 : function() {
        return this.querySelectorAll(selector);
      };
    }

    function selection_selectAll$1(select) {
      if (typeof select !== "function") select = selectorAll$1(select);

      for (var groups = this._groups, m = groups.length, subgroups = [], parents = [], j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, node, i = 0; i < n; ++i) {
          if (node = group[i]) {
            subgroups.push(select.call(node, node.__data__, i, group));
            parents.push(node);
          }
        }
      }

      return new Selection$2(subgroups, parents);
    }

    function matcher$1(selector) {
      return function() {
        return this.matches(selector);
      };
    }

    function selection_filter$1(match) {
      if (typeof match !== "function") match = matcher$1(match);

      for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, subgroup = subgroups[j] = [], node, i = 0; i < n; ++i) {
          if ((node = group[i]) && match.call(node, node.__data__, i, group)) {
            subgroup.push(node);
          }
        }
      }

      return new Selection$2(subgroups, this._parents);
    }

    function sparse$1(update) {
      return new Array(update.length);
    }

    function selection_enter$1() {
      return new Selection$2(this._enter || this._groups.map(sparse$1), this._parents);
    }

    function EnterNode$1(parent, datum) {
      this.ownerDocument = parent.ownerDocument;
      this.namespaceURI = parent.namespaceURI;
      this._next = null;
      this._parent = parent;
      this.__data__ = datum;
    }

    EnterNode$1.prototype = {
      constructor: EnterNode$1,
      appendChild: function(child) { return this._parent.insertBefore(child, this._next); },
      insertBefore: function(child, next) { return this._parent.insertBefore(child, next); },
      querySelector: function(selector) { return this._parent.querySelector(selector); },
      querySelectorAll: function(selector) { return this._parent.querySelectorAll(selector); }
    };

    function constant$f(x) {
      return function() {
        return x;
      };
    }

    var keyPrefix$2 = "$"; // Protect against keys like “__proto__”.

    function bindIndex$1(parent, group, enter, update, exit, data) {
      var i = 0,
          node,
          groupLength = group.length,
          dataLength = data.length;

      // Put any non-null nodes that fit into update.
      // Put any null nodes into enter.
      // Put any remaining data into enter.
      for (; i < dataLength; ++i) {
        if (node = group[i]) {
          node.__data__ = data[i];
          update[i] = node;
        } else {
          enter[i] = new EnterNode$1(parent, data[i]);
        }
      }

      // Put any non-null nodes that don’t fit into exit.
      for (; i < groupLength; ++i) {
        if (node = group[i]) {
          exit[i] = node;
        }
      }
    }

    function bindKey$1(parent, group, enter, update, exit, data, key) {
      var i,
          node,
          nodeByKeyValue = {},
          groupLength = group.length,
          dataLength = data.length,
          keyValues = new Array(groupLength),
          keyValue;

      // Compute the key for each node.
      // If multiple nodes have the same key, the duplicates are added to exit.
      for (i = 0; i < groupLength; ++i) {
        if (node = group[i]) {
          keyValues[i] = keyValue = keyPrefix$2 + key.call(node, node.__data__, i, group);
          if (keyValue in nodeByKeyValue) {
            exit[i] = node;
          } else {
            nodeByKeyValue[keyValue] = node;
          }
        }
      }

      // Compute the key for each datum.
      // If there a node associated with this key, join and add it to update.
      // If there is not (or the key is a duplicate), add it to enter.
      for (i = 0; i < dataLength; ++i) {
        keyValue = keyPrefix$2 + key.call(parent, data[i], i, data);
        if (node = nodeByKeyValue[keyValue]) {
          update[i] = node;
          node.__data__ = data[i];
          nodeByKeyValue[keyValue] = null;
        } else {
          enter[i] = new EnterNode$1(parent, data[i]);
        }
      }

      // Add any remaining nodes that were not bound to data to exit.
      for (i = 0; i < groupLength; ++i) {
        if ((node = group[i]) && (nodeByKeyValue[keyValues[i]] === node)) {
          exit[i] = node;
        }
      }
    }

    function selection_data$1(value, key) {
      if (!value) {
        data = new Array(this.size()), j = -1;
        this.each(function(d) { data[++j] = d; });
        return data;
      }

      var bind = key ? bindKey$1 : bindIndex$1,
          parents = this._parents,
          groups = this._groups;

      if (typeof value !== "function") value = constant$f(value);

      for (var m = groups.length, update = new Array(m), enter = new Array(m), exit = new Array(m), j = 0; j < m; ++j) {
        var parent = parents[j],
            group = groups[j],
            groupLength = group.length,
            data = value.call(parent, parent && parent.__data__, j, parents),
            dataLength = data.length,
            enterGroup = enter[j] = new Array(dataLength),
            updateGroup = update[j] = new Array(dataLength),
            exitGroup = exit[j] = new Array(groupLength);

        bind(parent, group, enterGroup, updateGroup, exitGroup, data, key);

        // Now connect the enter nodes to their following update node, such that
        // appendChild can insert the materialized enter node before this node,
        // rather than at the end of the parent node.
        for (var i0 = 0, i1 = 0, previous, next; i0 < dataLength; ++i0) {
          if (previous = enterGroup[i0]) {
            if (i0 >= i1) i1 = i0 + 1;
            while (!(next = updateGroup[i1]) && ++i1 < dataLength);
            previous._next = next || null;
          }
        }
      }

      update = new Selection$2(update, parents);
      update._enter = enter;
      update._exit = exit;
      return update;
    }

    function selection_exit$1() {
      return new Selection$2(this._exit || this._groups.map(sparse$1), this._parents);
    }

    function selection_join$1(onenter, onupdate, onexit) {
      var enter = this.enter(), update = this, exit = this.exit();
      enter = typeof onenter === "function" ? onenter(enter) : enter.append(onenter + "");
      if (onupdate != null) update = onupdate(update);
      if (onexit == null) exit.remove(); else onexit(exit);
      return enter && update ? enter.merge(update).order() : update;
    }

    function selection_merge$1(selection) {

      for (var groups0 = this._groups, groups1 = selection._groups, m0 = groups0.length, m1 = groups1.length, m = Math.min(m0, m1), merges = new Array(m0), j = 0; j < m; ++j) {
        for (var group0 = groups0[j], group1 = groups1[j], n = group0.length, merge = merges[j] = new Array(n), node, i = 0; i < n; ++i) {
          if (node = group0[i] || group1[i]) {
            merge[i] = node;
          }
        }
      }

      for (; j < m0; ++j) {
        merges[j] = groups0[j];
      }

      return new Selection$2(merges, this._parents);
    }

    function selection_order$1() {

      for (var groups = this._groups, j = -1, m = groups.length; ++j < m;) {
        for (var group = groups[j], i = group.length - 1, next = group[i], node; --i >= 0;) {
          if (node = group[i]) {
            if (next && node.compareDocumentPosition(next) ^ 4) next.parentNode.insertBefore(node, next);
            next = node;
          }
        }
      }

      return this;
    }

    function selection_sort$1(compare) {
      if (!compare) compare = ascending$5;

      function compareNode(a, b) {
        return a && b ? compare(a.__data__, b.__data__) : !a - !b;
      }

      for (var groups = this._groups, m = groups.length, sortgroups = new Array(m), j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, sortgroup = sortgroups[j] = new Array(n), node, i = 0; i < n; ++i) {
          if (node = group[i]) {
            sortgroup[i] = node;
          }
        }
        sortgroup.sort(compareNode);
      }

      return new Selection$2(sortgroups, this._parents).order();
    }

    function ascending$5(a, b) {
      return a < b ? -1 : a > b ? 1 : a >= b ? 0 : NaN;
    }

    function selection_call$1() {
      var callback = arguments[0];
      arguments[0] = this;
      callback.apply(null, arguments);
      return this;
    }

    function selection_nodes$1() {
      var nodes = new Array(this.size()), i = -1;
      this.each(function() { nodes[++i] = this; });
      return nodes;
    }

    function selection_node$1() {

      for (var groups = this._groups, j = 0, m = groups.length; j < m; ++j) {
        for (var group = groups[j], i = 0, n = group.length; i < n; ++i) {
          var node = group[i];
          if (node) return node;
        }
      }

      return null;
    }

    function selection_size$1() {
      var size = 0;
      this.each(function() { ++size; });
      return size;
    }

    function selection_empty$1() {
      return !this.node();
    }

    function selection_each$1(callback) {

      for (var groups = this._groups, j = 0, m = groups.length; j < m; ++j) {
        for (var group = groups[j], i = 0, n = group.length, node; i < n; ++i) {
          if (node = group[i]) callback.call(node, node.__data__, i, group);
        }
      }

      return this;
    }

    function attrRemove$2(name) {
      return function() {
        this.removeAttribute(name);
      };
    }

    function attrRemoveNS$2(fullname) {
      return function() {
        this.removeAttributeNS(fullname.space, fullname.local);
      };
    }

    function attrConstant$2(name, value) {
      return function() {
        this.setAttribute(name, value);
      };
    }

    function attrConstantNS$2(fullname, value) {
      return function() {
        this.setAttributeNS(fullname.space, fullname.local, value);
      };
    }

    function attrFunction$2(name, value) {
      return function() {
        var v = value.apply(this, arguments);
        if (v == null) this.removeAttribute(name);
        else this.setAttribute(name, v);
      };
    }

    function attrFunctionNS$2(fullname, value) {
      return function() {
        var v = value.apply(this, arguments);
        if (v == null) this.removeAttributeNS(fullname.space, fullname.local);
        else this.setAttributeNS(fullname.space, fullname.local, v);
      };
    }

    function selection_attr$1(name, value) {
      var fullname = namespace$1(name);

      if (arguments.length < 2) {
        var node = this.node();
        return fullname.local
            ? node.getAttributeNS(fullname.space, fullname.local)
            : node.getAttribute(fullname);
      }

      return this.each((value == null
          ? (fullname.local ? attrRemoveNS$2 : attrRemove$2) : (typeof value === "function"
          ? (fullname.local ? attrFunctionNS$2 : attrFunction$2)
          : (fullname.local ? attrConstantNS$2 : attrConstant$2)))(fullname, value));
    }

    function defaultView$1(node) {
      return (node.ownerDocument && node.ownerDocument.defaultView) // node is a Node
          || (node.document && node) // node is a Window
          || node.defaultView; // node is a Document
    }

    function styleRemove$2(name) {
      return function() {
        this.style.removeProperty(name);
      };
    }

    function styleConstant$2(name, value, priority) {
      return function() {
        this.style.setProperty(name, value, priority);
      };
    }

    function styleFunction$2(name, value, priority) {
      return function() {
        var v = value.apply(this, arguments);
        if (v == null) this.style.removeProperty(name);
        else this.style.setProperty(name, v, priority);
      };
    }

    function selection_style$1(name, value, priority) {
      return arguments.length > 1
          ? this.each((value == null
                ? styleRemove$2 : typeof value === "function"
                ? styleFunction$2
                : styleConstant$2)(name, value, priority == null ? "" : priority))
          : styleValue$1(this.node(), name);
    }

    function styleValue$1(node, name) {
      return node.style.getPropertyValue(name)
          || defaultView$1(node).getComputedStyle(node, null).getPropertyValue(name);
    }

    function propertyRemove$1(name) {
      return function() {
        delete this[name];
      };
    }

    function propertyConstant$1(name, value) {
      return function() {
        this[name] = value;
      };
    }

    function propertyFunction$1(name, value) {
      return function() {
        var v = value.apply(this, arguments);
        if (v == null) delete this[name];
        else this[name] = v;
      };
    }

    function selection_property$1(name, value) {
      return arguments.length > 1
          ? this.each((value == null
              ? propertyRemove$1 : typeof value === "function"
              ? propertyFunction$1
              : propertyConstant$1)(name, value))
          : this.node()[name];
    }

    function classArray$1(string) {
      return string.trim().split(/^|\s+/);
    }

    function classList$1(node) {
      return node.classList || new ClassList$1(node);
    }

    function ClassList$1(node) {
      this._node = node;
      this._names = classArray$1(node.getAttribute("class") || "");
    }

    ClassList$1.prototype = {
      add: function(name) {
        var i = this._names.indexOf(name);
        if (i < 0) {
          this._names.push(name);
          this._node.setAttribute("class", this._names.join(" "));
        }
      },
      remove: function(name) {
        var i = this._names.indexOf(name);
        if (i >= 0) {
          this._names.splice(i, 1);
          this._node.setAttribute("class", this._names.join(" "));
        }
      },
      contains: function(name) {
        return this._names.indexOf(name) >= 0;
      }
    };

    function classedAdd$1(node, names) {
      var list = classList$1(node), i = -1, n = names.length;
      while (++i < n) list.add(names[i]);
    }

    function classedRemove$1(node, names) {
      var list = classList$1(node), i = -1, n = names.length;
      while (++i < n) list.remove(names[i]);
    }

    function classedTrue$1(names) {
      return function() {
        classedAdd$1(this, names);
      };
    }

    function classedFalse$1(names) {
      return function() {
        classedRemove$1(this, names);
      };
    }

    function classedFunction$1(names, value) {
      return function() {
        (value.apply(this, arguments) ? classedAdd$1 : classedRemove$1)(this, names);
      };
    }

    function selection_classed$1(name, value) {
      var names = classArray$1(name + "");

      if (arguments.length < 2) {
        var list = classList$1(this.node()), i = -1, n = names.length;
        while (++i < n) if (!list.contains(names[i])) return false;
        return true;
      }

      return this.each((typeof value === "function"
          ? classedFunction$1 : value
          ? classedTrue$1
          : classedFalse$1)(names, value));
    }

    function textRemove$1() {
      this.textContent = "";
    }

    function textConstant$2(value) {
      return function() {
        this.textContent = value;
      };
    }

    function textFunction$2(value) {
      return function() {
        var v = value.apply(this, arguments);
        this.textContent = v == null ? "" : v;
      };
    }

    function selection_text$1(value) {
      return arguments.length
          ? this.each(value == null
              ? textRemove$1 : (typeof value === "function"
              ? textFunction$2
              : textConstant$2)(value))
          : this.node().textContent;
    }

    function htmlRemove$1() {
      this.innerHTML = "";
    }

    function htmlConstant$1(value) {
      return function() {
        this.innerHTML = value;
      };
    }

    function htmlFunction$1(value) {
      return function() {
        var v = value.apply(this, arguments);
        this.innerHTML = v == null ? "" : v;
      };
    }

    function selection_html$1(value) {
      return arguments.length
          ? this.each(value == null
              ? htmlRemove$1 : (typeof value === "function"
              ? htmlFunction$1
              : htmlConstant$1)(value))
          : this.node().innerHTML;
    }

    function raise$1() {
      if (this.nextSibling) this.parentNode.appendChild(this);
    }

    function selection_raise$1() {
      return this.each(raise$1);
    }

    function lower$1() {
      if (this.previousSibling) this.parentNode.insertBefore(this, this.parentNode.firstChild);
    }

    function selection_lower$1() {
      return this.each(lower$1);
    }

    function selection_append$1(name) {
      var create = typeof name === "function" ? name : creator$1(name);
      return this.select(function() {
        return this.appendChild(create.apply(this, arguments));
      });
    }

    function constantNull$1() {
      return null;
    }

    function selection_insert$1(name, before) {
      var create = typeof name === "function" ? name : creator$1(name),
          select = before == null ? constantNull$1 : typeof before === "function" ? before : selector$1(before);
      return this.select(function() {
        return this.insertBefore(create.apply(this, arguments), select.apply(this, arguments) || null);
      });
    }

    function remove$1() {
      var parent = this.parentNode;
      if (parent) parent.removeChild(this);
    }

    function selection_remove$1() {
      return this.each(remove$1);
    }

    function selection_cloneShallow$1() {
      var clone = this.cloneNode(false), parent = this.parentNode;
      return parent ? parent.insertBefore(clone, this.nextSibling) : clone;
    }

    function selection_cloneDeep$1() {
      var clone = this.cloneNode(true), parent = this.parentNode;
      return parent ? parent.insertBefore(clone, this.nextSibling) : clone;
    }

    function selection_clone$1(deep) {
      return this.select(deep ? selection_cloneDeep$1 : selection_cloneShallow$1);
    }

    function selection_datum$1(value) {
      return arguments.length
          ? this.property("__data__", value)
          : this.node().__data__;
    }

    var filterEvents$1 = {};

    var event$1 = null;

    if (typeof document !== "undefined") {
      var element$1 = document.documentElement;
      if (!("onmouseenter" in element$1)) {
        filterEvents$1 = {mouseenter: "mouseover", mouseleave: "mouseout"};
      }
    }

    function filterContextListener$1(listener, index, group) {
      listener = contextListener$1(listener, index, group);
      return function(event) {
        var related = event.relatedTarget;
        if (!related || (related !== this && !(related.compareDocumentPosition(this) & 8))) {
          listener.call(this, event);
        }
      };
    }

    function contextListener$1(listener, index, group) {
      return function(event1) {
        var event0 = event$1; // Events can be reentrant (e.g., focus).
        event$1 = event1;
        try {
          listener.call(this, this.__data__, index, group);
        } finally {
          event$1 = event0;
        }
      };
    }

    function parseTypenames$3(typenames) {
      return typenames.trim().split(/^|\s+/).map(function(t) {
        var name = "", i = t.indexOf(".");
        if (i >= 0) name = t.slice(i + 1), t = t.slice(0, i);
        return {type: t, name: name};
      });
    }

    function onRemove$1(typename) {
      return function() {
        var on = this.__on;
        if (!on) return;
        for (var j = 0, i = -1, m = on.length, o; j < m; ++j) {
          if (o = on[j], (!typename.type || o.type === typename.type) && o.name === typename.name) {
            this.removeEventListener(o.type, o.listener, o.capture);
          } else {
            on[++i] = o;
          }
        }
        if (++i) on.length = i;
        else delete this.__on;
      };
    }

    function onAdd$1(typename, value, capture) {
      var wrap = filterEvents$1.hasOwnProperty(typename.type) ? filterContextListener$1 : contextListener$1;
      return function(d, i, group) {
        var on = this.__on, o, listener = wrap(value, i, group);
        if (on) for (var j = 0, m = on.length; j < m; ++j) {
          if ((o = on[j]).type === typename.type && o.name === typename.name) {
            this.removeEventListener(o.type, o.listener, o.capture);
            this.addEventListener(o.type, o.listener = listener, o.capture = capture);
            o.value = value;
            return;
          }
        }
        this.addEventListener(typename.type, listener, capture);
        o = {type: typename.type, name: typename.name, value: value, listener: listener, capture: capture};
        if (!on) this.__on = [o];
        else on.push(o);
      };
    }

    function selection_on$1(typename, value, capture) {
      var typenames = parseTypenames$3(typename + ""), i, n = typenames.length, t;

      if (arguments.length < 2) {
        var on = this.node().__on;
        if (on) for (var j = 0, m = on.length, o; j < m; ++j) {
          for (i = 0, o = on[j]; i < n; ++i) {
            if ((t = typenames[i]).type === o.type && t.name === o.name) {
              return o.value;
            }
          }
        }
        return;
      }

      on = value ? onAdd$1 : onRemove$1;
      if (capture == null) capture = false;
      for (i = 0; i < n; ++i) this.each(on(typenames[i], value, capture));
      return this;
    }

    function customEvent$1(event1, listener, that, args) {
      var event0 = event$1;
      event1.sourceEvent = event$1;
      event$1 = event1;
      try {
        return listener.apply(that, args);
      } finally {
        event$1 = event0;
      }
    }

    function dispatchEvent$1(node, type, params) {
      var window = defaultView$1(node),
          event = window.CustomEvent;

      if (typeof event === "function") {
        event = new event(type, params);
      } else {
        event = window.document.createEvent("Event");
        if (params) event.initEvent(type, params.bubbles, params.cancelable), event.detail = params.detail;
        else event.initEvent(type, false, false);
      }

      node.dispatchEvent(event);
    }

    function dispatchConstant$1(type, params) {
      return function() {
        return dispatchEvent$1(this, type, params);
      };
    }

    function dispatchFunction$1(type, params) {
      return function() {
        return dispatchEvent$1(this, type, params.apply(this, arguments));
      };
    }

    function selection_dispatch$1(type, params) {
      return this.each((typeof params === "function"
          ? dispatchFunction$1
          : dispatchConstant$1)(type, params));
    }

    var root$3 = [null];

    function Selection$2(groups, parents) {
      this._groups = groups;
      this._parents = parents;
    }

    function selection$1() {
      return new Selection$2([[document.documentElement]], root$3);
    }

    Selection$2.prototype = selection$1.prototype = {
      constructor: Selection$2,
      select: selection_select$1,
      selectAll: selection_selectAll$1,
      filter: selection_filter$1,
      data: selection_data$1,
      enter: selection_enter$1,
      exit: selection_exit$1,
      join: selection_join$1,
      merge: selection_merge$1,
      order: selection_order$1,
      sort: selection_sort$1,
      call: selection_call$1,
      nodes: selection_nodes$1,
      node: selection_node$1,
      size: selection_size$1,
      empty: selection_empty$1,
      each: selection_each$1,
      attr: selection_attr$1,
      style: selection_style$1,
      property: selection_property$1,
      classed: selection_classed$1,
      text: selection_text$1,
      html: selection_html$1,
      raise: selection_raise$1,
      lower: selection_lower$1,
      append: selection_append$1,
      insert: selection_insert$1,
      remove: selection_remove$1,
      clone: selection_clone$1,
      datum: selection_datum$1,
      on: selection_on$1,
      dispatch: selection_dispatch$1
    };

    function select$1(selector) {
      return typeof selector === "string"
          ? new Selection$2([[document.querySelector(selector)]], [document.documentElement])
          : new Selection$2([[selector]], root$3);
    }

    function sourceEvent$1() {
      var current = event$1, source;
      while (source = current.sourceEvent) current = source;
      return current;
    }

    function point$6(node, event) {
      var svg = node.ownerSVGElement || node;

      if (svg.createSVGPoint) {
        var point = svg.createSVGPoint();
        point.x = event.clientX, point.y = event.clientY;
        point = point.matrixTransform(node.getScreenCTM().inverse());
        return [point.x, point.y];
      }

      var rect = node.getBoundingClientRect();
      return [event.clientX - rect.left - node.clientLeft, event.clientY - rect.top - node.clientTop];
    }

    function mouse$1(node) {
      var event = sourceEvent$1();
      if (event.changedTouches) event = event.changedTouches[0];
      return point$6(node, event);
    }

    function noevent$3() {
      event$1.preventDefault();
      event$1.stopImmediatePropagation();
    }

    function dragDisable$1(view) {
      var root = view.document.documentElement,
          selection = select$1(view).on("dragstart.drag", noevent$3, true);
      if ("onselectstart" in root) {
        selection.on("selectstart.drag", noevent$3, true);
      } else {
        root.__noselect = root.style.MozUserSelect;
        root.style.MozUserSelect = "none";
      }
    }

    function yesdrag$1(view, noclick) {
      var root = view.document.documentElement,
          selection = select$1(view).on("dragstart.drag", null);
      if (noclick) {
        selection.on("click.drag", noevent$3, true);
        setTimeout(function() { selection.on("click.drag", null); }, 0);
      }
      if ("onselectstart" in root) {
        selection.on("selectstart.drag", null);
      } else {
        root.style.MozUserSelect = root.__noselect;
        delete root.__noselect;
      }
    }

    function define$1(constructor, factory, prototype) {
      constructor.prototype = factory.prototype = prototype;
      prototype.constructor = constructor;
    }

    function extend$1(parent, definition) {
      var prototype = Object.create(parent.prototype);
      for (var key in definition) prototype[key] = definition[key];
      return prototype;
    }

    function Color$1() {}

    var darker$1 = 0.7;
    var brighter$1 = 1 / darker$1;

    var reI$1 = "\\s*([+-]?\\d+)\\s*",
        reN$1 = "\\s*([+-]?\\d*\\.?\\d+(?:[eE][+-]?\\d+)?)\\s*",
        reP$1 = "\\s*([+-]?\\d*\\.?\\d+(?:[eE][+-]?\\d+)?)%\\s*",
        reHex$1 = /^#([0-9a-f]{3,8})$/,
        reRgbInteger$1 = new RegExp("^rgb\\(" + [reI$1, reI$1, reI$1] + "\\)$"),
        reRgbPercent$1 = new RegExp("^rgb\\(" + [reP$1, reP$1, reP$1] + "\\)$"),
        reRgbaInteger$1 = new RegExp("^rgba\\(" + [reI$1, reI$1, reI$1, reN$1] + "\\)$"),
        reRgbaPercent$1 = new RegExp("^rgba\\(" + [reP$1, reP$1, reP$1, reN$1] + "\\)$"),
        reHslPercent$1 = new RegExp("^hsl\\(" + [reN$1, reP$1, reP$1] + "\\)$"),
        reHslaPercent$1 = new RegExp("^hsla\\(" + [reN$1, reP$1, reP$1, reN$1] + "\\)$");

    var named$1 = {
      aliceblue: 0xf0f8ff,
      antiquewhite: 0xfaebd7,
      aqua: 0x00ffff,
      aquamarine: 0x7fffd4,
      azure: 0xf0ffff,
      beige: 0xf5f5dc,
      bisque: 0xffe4c4,
      black: 0x000000,
      blanchedalmond: 0xffebcd,
      blue: 0x0000ff,
      blueviolet: 0x8a2be2,
      brown: 0xa52a2a,
      burlywood: 0xdeb887,
      cadetblue: 0x5f9ea0,
      chartreuse: 0x7fff00,
      chocolate: 0xd2691e,
      coral: 0xff7f50,
      cornflowerblue: 0x6495ed,
      cornsilk: 0xfff8dc,
      crimson: 0xdc143c,
      cyan: 0x00ffff,
      darkblue: 0x00008b,
      darkcyan: 0x008b8b,
      darkgoldenrod: 0xb8860b,
      darkgray: 0xa9a9a9,
      darkgreen: 0x006400,
      darkgrey: 0xa9a9a9,
      darkkhaki: 0xbdb76b,
      darkmagenta: 0x8b008b,
      darkolivegreen: 0x556b2f,
      darkorange: 0xff8c00,
      darkorchid: 0x9932cc,
      darkred: 0x8b0000,
      darksalmon: 0xe9967a,
      darkseagreen: 0x8fbc8f,
      darkslateblue: 0x483d8b,
      darkslategray: 0x2f4f4f,
      darkslategrey: 0x2f4f4f,
      darkturquoise: 0x00ced1,
      darkviolet: 0x9400d3,
      deeppink: 0xff1493,
      deepskyblue: 0x00bfff,
      dimgray: 0x696969,
      dimgrey: 0x696969,
      dodgerblue: 0x1e90ff,
      firebrick: 0xb22222,
      floralwhite: 0xfffaf0,
      forestgreen: 0x228b22,
      fuchsia: 0xff00ff,
      gainsboro: 0xdcdcdc,
      ghostwhite: 0xf8f8ff,
      gold: 0xffd700,
      goldenrod: 0xdaa520,
      gray: 0x808080,
      green: 0x008000,
      greenyellow: 0xadff2f,
      grey: 0x808080,
      honeydew: 0xf0fff0,
      hotpink: 0xff69b4,
      indianred: 0xcd5c5c,
      indigo: 0x4b0082,
      ivory: 0xfffff0,
      khaki: 0xf0e68c,
      lavender: 0xe6e6fa,
      lavenderblush: 0xfff0f5,
      lawngreen: 0x7cfc00,
      lemonchiffon: 0xfffacd,
      lightblue: 0xadd8e6,
      lightcoral: 0xf08080,
      lightcyan: 0xe0ffff,
      lightgoldenrodyellow: 0xfafad2,
      lightgray: 0xd3d3d3,
      lightgreen: 0x90ee90,
      lightgrey: 0xd3d3d3,
      lightpink: 0xffb6c1,
      lightsalmon: 0xffa07a,
      lightseagreen: 0x20b2aa,
      lightskyblue: 0x87cefa,
      lightslategray: 0x778899,
      lightslategrey: 0x778899,
      lightsteelblue: 0xb0c4de,
      lightyellow: 0xffffe0,
      lime: 0x00ff00,
      limegreen: 0x32cd32,
      linen: 0xfaf0e6,
      magenta: 0xff00ff,
      maroon: 0x800000,
      mediumaquamarine: 0x66cdaa,
      mediumblue: 0x0000cd,
      mediumorchid: 0xba55d3,
      mediumpurple: 0x9370db,
      mediumseagreen: 0x3cb371,
      mediumslateblue: 0x7b68ee,
      mediumspringgreen: 0x00fa9a,
      mediumturquoise: 0x48d1cc,
      mediumvioletred: 0xc71585,
      midnightblue: 0x191970,
      mintcream: 0xf5fffa,
      mistyrose: 0xffe4e1,
      moccasin: 0xffe4b5,
      navajowhite: 0xffdead,
      navy: 0x000080,
      oldlace: 0xfdf5e6,
      olive: 0x808000,
      olivedrab: 0x6b8e23,
      orange: 0xffa500,
      orangered: 0xff4500,
      orchid: 0xda70d6,
      palegoldenrod: 0xeee8aa,
      palegreen: 0x98fb98,
      paleturquoise: 0xafeeee,
      palevioletred: 0xdb7093,
      papayawhip: 0xffefd5,
      peachpuff: 0xffdab9,
      peru: 0xcd853f,
      pink: 0xffc0cb,
      plum: 0xdda0dd,
      powderblue: 0xb0e0e6,
      purple: 0x800080,
      rebeccapurple: 0x663399,
      red: 0xff0000,
      rosybrown: 0xbc8f8f,
      royalblue: 0x4169e1,
      saddlebrown: 0x8b4513,
      salmon: 0xfa8072,
      sandybrown: 0xf4a460,
      seagreen: 0x2e8b57,
      seashell: 0xfff5ee,
      sienna: 0xa0522d,
      silver: 0xc0c0c0,
      skyblue: 0x87ceeb,
      slateblue: 0x6a5acd,
      slategray: 0x708090,
      slategrey: 0x708090,
      snow: 0xfffafa,
      springgreen: 0x00ff7f,
      steelblue: 0x4682b4,
      tan: 0xd2b48c,
      teal: 0x008080,
      thistle: 0xd8bfd8,
      tomato: 0xff6347,
      turquoise: 0x40e0d0,
      violet: 0xee82ee,
      wheat: 0xf5deb3,
      white: 0xffffff,
      whitesmoke: 0xf5f5f5,
      yellow: 0xffff00,
      yellowgreen: 0x9acd32
    };

    define$1(Color$1, color$1, {
      copy: function(channels) {
        return Object.assign(new this.constructor, this, channels);
      },
      displayable: function() {
        return this.rgb().displayable();
      },
      hex: color_formatHex$1, // Deprecated! Use color.formatHex.
      formatHex: color_formatHex$1,
      formatHsl: color_formatHsl$1,
      formatRgb: color_formatRgb$1,
      toString: color_formatRgb$1
    });

    function color_formatHex$1() {
      return this.rgb().formatHex();
    }

    function color_formatHsl$1() {
      return hslConvert$1(this).formatHsl();
    }

    function color_formatRgb$1() {
      return this.rgb().formatRgb();
    }

    function color$1(format) {
      var m, l;
      format = (format + "").trim().toLowerCase();
      return (m = reHex$1.exec(format)) ? (l = m[1].length, m = parseInt(m[1], 16), l === 6 ? rgbn$1(m) // #ff0000
          : l === 3 ? new Rgb$1((m >> 8 & 0xf) | (m >> 4 & 0xf0), (m >> 4 & 0xf) | (m & 0xf0), ((m & 0xf) << 4) | (m & 0xf), 1) // #f00
          : l === 8 ? rgba$1(m >> 24 & 0xff, m >> 16 & 0xff, m >> 8 & 0xff, (m & 0xff) / 0xff) // #ff000000
          : l === 4 ? rgba$1((m >> 12 & 0xf) | (m >> 8 & 0xf0), (m >> 8 & 0xf) | (m >> 4 & 0xf0), (m >> 4 & 0xf) | (m & 0xf0), (((m & 0xf) << 4) | (m & 0xf)) / 0xff) // #f000
          : null) // invalid hex
          : (m = reRgbInteger$1.exec(format)) ? new Rgb$1(m[1], m[2], m[3], 1) // rgb(255, 0, 0)
          : (m = reRgbPercent$1.exec(format)) ? new Rgb$1(m[1] * 255 / 100, m[2] * 255 / 100, m[3] * 255 / 100, 1) // rgb(100%, 0%, 0%)
          : (m = reRgbaInteger$1.exec(format)) ? rgba$1(m[1], m[2], m[3], m[4]) // rgba(255, 0, 0, 1)
          : (m = reRgbaPercent$1.exec(format)) ? rgba$1(m[1] * 255 / 100, m[2] * 255 / 100, m[3] * 255 / 100, m[4]) // rgb(100%, 0%, 0%, 1)
          : (m = reHslPercent$1.exec(format)) ? hsla$1(m[1], m[2] / 100, m[3] / 100, 1) // hsl(120, 50%, 50%)
          : (m = reHslaPercent$1.exec(format)) ? hsla$1(m[1], m[2] / 100, m[3] / 100, m[4]) // hsla(120, 50%, 50%, 1)
          : named$1.hasOwnProperty(format) ? rgbn$1(named$1[format]) // eslint-disable-line no-prototype-builtins
          : format === "transparent" ? new Rgb$1(NaN, NaN, NaN, 0)
          : null;
    }

    function rgbn$1(n) {
      return new Rgb$1(n >> 16 & 0xff, n >> 8 & 0xff, n & 0xff, 1);
    }

    function rgba$1(r, g, b, a) {
      if (a <= 0) r = g = b = NaN;
      return new Rgb$1(r, g, b, a);
    }

    function rgbConvert$1(o) {
      if (!(o instanceof Color$1)) o = color$1(o);
      if (!o) return new Rgb$1;
      o = o.rgb();
      return new Rgb$1(o.r, o.g, o.b, o.opacity);
    }

    function rgb$1(r, g, b, opacity) {
      return arguments.length === 1 ? rgbConvert$1(r) : new Rgb$1(r, g, b, opacity == null ? 1 : opacity);
    }

    function Rgb$1(r, g, b, opacity) {
      this.r = +r;
      this.g = +g;
      this.b = +b;
      this.opacity = +opacity;
    }

    define$1(Rgb$1, rgb$1, extend$1(Color$1, {
      brighter: function(k) {
        k = k == null ? brighter$1 : Math.pow(brighter$1, k);
        return new Rgb$1(this.r * k, this.g * k, this.b * k, this.opacity);
      },
      darker: function(k) {
        k = k == null ? darker$1 : Math.pow(darker$1, k);
        return new Rgb$1(this.r * k, this.g * k, this.b * k, this.opacity);
      },
      rgb: function() {
        return this;
      },
      displayable: function() {
        return (-0.5 <= this.r && this.r < 255.5)
            && (-0.5 <= this.g && this.g < 255.5)
            && (-0.5 <= this.b && this.b < 255.5)
            && (0 <= this.opacity && this.opacity <= 1);
      },
      hex: rgb_formatHex$1, // Deprecated! Use color.formatHex.
      formatHex: rgb_formatHex$1,
      formatRgb: rgb_formatRgb$1,
      toString: rgb_formatRgb$1
    }));

    function rgb_formatHex$1() {
      return "#" + hex$1(this.r) + hex$1(this.g) + hex$1(this.b);
    }

    function rgb_formatRgb$1() {
      var a = this.opacity; a = isNaN(a) ? 1 : Math.max(0, Math.min(1, a));
      return (a === 1 ? "rgb(" : "rgba(")
          + Math.max(0, Math.min(255, Math.round(this.r) || 0)) + ", "
          + Math.max(0, Math.min(255, Math.round(this.g) || 0)) + ", "
          + Math.max(0, Math.min(255, Math.round(this.b) || 0))
          + (a === 1 ? ")" : ", " + a + ")");
    }

    function hex$1(value) {
      value = Math.max(0, Math.min(255, Math.round(value) || 0));
      return (value < 16 ? "0" : "") + value.toString(16);
    }

    function hsla$1(h, s, l, a) {
      if (a <= 0) h = s = l = NaN;
      else if (l <= 0 || l >= 1) h = s = NaN;
      else if (s <= 0) h = NaN;
      return new Hsl$1(h, s, l, a);
    }

    function hslConvert$1(o) {
      if (o instanceof Hsl$1) return new Hsl$1(o.h, o.s, o.l, o.opacity);
      if (!(o instanceof Color$1)) o = color$1(o);
      if (!o) return new Hsl$1;
      if (o instanceof Hsl$1) return o;
      o = o.rgb();
      var r = o.r / 255,
          g = o.g / 255,
          b = o.b / 255,
          min = Math.min(r, g, b),
          max = Math.max(r, g, b),
          h = NaN,
          s = max - min,
          l = (max + min) / 2;
      if (s) {
        if (r === max) h = (g - b) / s + (g < b) * 6;
        else if (g === max) h = (b - r) / s + 2;
        else h = (r - g) / s + 4;
        s /= l < 0.5 ? max + min : 2 - max - min;
        h *= 60;
      } else {
        s = l > 0 && l < 1 ? 0 : h;
      }
      return new Hsl$1(h, s, l, o.opacity);
    }

    function hsl$3(h, s, l, opacity) {
      return arguments.length === 1 ? hslConvert$1(h) : new Hsl$1(h, s, l, opacity == null ? 1 : opacity);
    }

    function Hsl$1(h, s, l, opacity) {
      this.h = +h;
      this.s = +s;
      this.l = +l;
      this.opacity = +opacity;
    }

    define$1(Hsl$1, hsl$3, extend$1(Color$1, {
      brighter: function(k) {
        k = k == null ? brighter$1 : Math.pow(brighter$1, k);
        return new Hsl$1(this.h, this.s, this.l * k, this.opacity);
      },
      darker: function(k) {
        k = k == null ? darker$1 : Math.pow(darker$1, k);
        return new Hsl$1(this.h, this.s, this.l * k, this.opacity);
      },
      rgb: function() {
        var h = this.h % 360 + (this.h < 0) * 360,
            s = isNaN(h) || isNaN(this.s) ? 0 : this.s,
            l = this.l,
            m2 = l + (l < 0.5 ? l : 1 - l) * s,
            m1 = 2 * l - m2;
        return new Rgb$1(
          hsl2rgb$1(h >= 240 ? h - 240 : h + 120, m1, m2),
          hsl2rgb$1(h, m1, m2),
          hsl2rgb$1(h < 120 ? h + 240 : h - 120, m1, m2),
          this.opacity
        );
      },
      displayable: function() {
        return (0 <= this.s && this.s <= 1 || isNaN(this.s))
            && (0 <= this.l && this.l <= 1)
            && (0 <= this.opacity && this.opacity <= 1);
      },
      formatHsl: function() {
        var a = this.opacity; a = isNaN(a) ? 1 : Math.max(0, Math.min(1, a));
        return (a === 1 ? "hsl(" : "hsla(")
            + (this.h || 0) + ", "
            + (this.s || 0) * 100 + "%, "
            + (this.l || 0) * 100 + "%"
            + (a === 1 ? ")" : ", " + a + ")");
      }
    }));

    /* From FvD 13.37, CSS Color Module Level 3 */
    function hsl2rgb$1(h, m1, m2) {
      return (h < 60 ? m1 + (m2 - m1) * h / 60
          : h < 180 ? m2
          : h < 240 ? m1 + (m2 - m1) * (240 - h) / 60
          : m1) * 255;
    }

    function constant$g(x) {
      return function() {
        return x;
      };
    }

    function linear$3(a, d) {
      return function(t) {
        return a + t * d;
      };
    }

    function exponential$2(a, b, y) {
      return a = Math.pow(a, y), b = Math.pow(b, y) - a, y = 1 / y, function(t) {
        return Math.pow(a + t * b, y);
      };
    }

    function gamma$1(y) {
      return (y = +y) === 1 ? nogamma$1 : function(a, b) {
        return b - a ? exponential$2(a, b, y) : constant$g(isNaN(a) ? b : a);
      };
    }

    function nogamma$1(a, b) {
      var d = b - a;
      return d ? linear$3(a, d) : constant$g(isNaN(a) ? b : a);
    }

    var interpolateRgb$1 = (function rgbGamma(y) {
      var color = gamma$1(y);

      function rgb(start, end) {
        var r = color((start = rgb$1(start)).r, (end = rgb$1(end)).r),
            g = color(start.g, end.g),
            b = color(start.b, end.b),
            opacity = nogamma$1(start.opacity, end.opacity);
        return function(t) {
          start.r = r(t);
          start.g = g(t);
          start.b = b(t);
          start.opacity = opacity(t);
          return start + "";
        };
      }

      rgb.gamma = rgbGamma;

      return rgb;
    })(1);

    function numberArray$1(a, b) {
      if (!b) b = [];
      var n = a ? Math.min(b.length, a.length) : 0,
          c = b.slice(),
          i;
      return function(t) {
        for (i = 0; i < n; ++i) c[i] = a[i] * (1 - t) + b[i] * t;
        return c;
      };
    }

    function isNumberArray$1(x) {
      return ArrayBuffer.isView(x) && !(x instanceof DataView);
    }

    function genericArray$1(a, b) {
      var nb = b ? b.length : 0,
          na = a ? Math.min(nb, a.length) : 0,
          x = new Array(na),
          c = new Array(nb),
          i;

      for (i = 0; i < na; ++i) x[i] = interpolate$2(a[i], b[i]);
      for (; i < nb; ++i) c[i] = b[i];

      return function(t) {
        for (i = 0; i < na; ++i) c[i] = x[i](t);
        return c;
      };
    }

    function date$2(a, b) {
      var d = new Date;
      return a = +a, b = +b, function(t) {
        return d.setTime(a * (1 - t) + b * t), d;
      };
    }

    function interpolateNumber$1(a, b) {
      return a = +a, b = +b, function(t) {
        return a * (1 - t) + b * t;
      };
    }

    function object$2(a, b) {
      var i = {},
          c = {},
          k;

      if (a === null || typeof a !== "object") a = {};
      if (b === null || typeof b !== "object") b = {};

      for (k in b) {
        if (k in a) {
          i[k] = interpolate$2(a[k], b[k]);
        } else {
          c[k] = b[k];
        }
      }

      return function(t) {
        for (k in i) c[k] = i[k](t);
        return c;
      };
    }

    var reA$1 = /[-+]?(?:\d+\.?\d*|\.?\d+)(?:[eE][-+]?\d+)?/g,
        reB$1 = new RegExp(reA$1.source, "g");

    function zero$1(b) {
      return function() {
        return b;
      };
    }

    function one$1(b) {
      return function(t) {
        return b(t) + "";
      };
    }

    function interpolateString$1(a, b) {
      var bi = reA$1.lastIndex = reB$1.lastIndex = 0, // scan index for next number in b
          am, // current match in a
          bm, // current match in b
          bs, // string preceding current number in b, if any
          i = -1, // index in s
          s = [], // string constants and placeholders
          q = []; // number interpolators

      // Coerce inputs to strings.
      a = a + "", b = b + "";

      // Interpolate pairs of numbers in a & b.
      while ((am = reA$1.exec(a))
          && (bm = reB$1.exec(b))) {
        if ((bs = bm.index) > bi) { // a string precedes the next number in b
          bs = b.slice(bi, bs);
          if (s[i]) s[i] += bs; // coalesce with previous string
          else s[++i] = bs;
        }
        if ((am = am[0]) === (bm = bm[0])) { // numbers in a & b match
          if (s[i]) s[i] += bm; // coalesce with previous string
          else s[++i] = bm;
        } else { // interpolate non-matching numbers
          s[++i] = null;
          q.push({i: i, x: interpolateNumber$1(am, bm)});
        }
        bi = reB$1.lastIndex;
      }

      // Add remains of b.
      if (bi < b.length) {
        bs = b.slice(bi);
        if (s[i]) s[i] += bs; // coalesce with previous string
        else s[++i] = bs;
      }

      // Special optimization for only a single match.
      // Otherwise, interpolate each of the numbers and rejoin the string.
      return s.length < 2 ? (q[0]
          ? one$1(q[0].x)
          : zero$1(b))
          : (b = q.length, function(t) {
              for (var i = 0, o; i < b; ++i) s[(o = q[i]).i] = o.x(t);
              return s.join("");
            });
    }

    function interpolate$2(a, b) {
      var t = typeof b, c;
      return b == null || t === "boolean" ? constant$g(b)
          : (t === "number" ? interpolateNumber$1
          : t === "string" ? ((c = color$1(b)) ? (b = c, interpolateRgb$1) : interpolateString$1)
          : b instanceof color$1 ? interpolateRgb$1
          : b instanceof Date ? date$2
          : isNumberArray$1(b) ? numberArray$1
          : Array.isArray(b) ? genericArray$1
          : typeof b.valueOf !== "function" && typeof b.toString !== "function" || isNaN(b) ? object$2
          : interpolateNumber$1)(a, b);
    }

    var degrees$2 = 180 / Math.PI;

    var identity$c = {
      translateX: 0,
      translateY: 0,
      rotate: 0,
      skewX: 0,
      scaleX: 1,
      scaleY: 1
    };

    function decompose$1(a, b, c, d, e, f) {
      var scaleX, scaleY, skewX;
      if (scaleX = Math.sqrt(a * a + b * b)) a /= scaleX, b /= scaleX;
      if (skewX = a * c + b * d) c -= a * skewX, d -= b * skewX;
      if (scaleY = Math.sqrt(c * c + d * d)) c /= scaleY, d /= scaleY, skewX /= scaleY;
      if (a * d < b * c) a = -a, b = -b, skewX = -skewX, scaleX = -scaleX;
      return {
        translateX: e,
        translateY: f,
        rotate: Math.atan2(b, a) * degrees$2,
        skewX: Math.atan(skewX) * degrees$2,
        scaleX: scaleX,
        scaleY: scaleY
      };
    }

    var cssNode$1,
        cssRoot$1,
        cssView$1,
        svgNode$1;

    function parseCss$2(value) {
      if (value === "none") return identity$c;
      if (!cssNode$1) cssNode$1 = document.createElement("DIV"), cssRoot$1 = document.documentElement, cssView$1 = document.defaultView;
      cssNode$1.style.transform = value;
      value = cssView$1.getComputedStyle(cssRoot$1.appendChild(cssNode$1), null).getPropertyValue("transform");
      cssRoot$1.removeChild(cssNode$1);
      value = value.slice(7, -1).split(",");
      return decompose$1(+value[0], +value[1], +value[2], +value[3], +value[4], +value[5]);
    }

    function parseSvg$1(value) {
      if (value == null) return identity$c;
      if (!svgNode$1) svgNode$1 = document.createElementNS("http://www.w3.org/2000/svg", "g");
      svgNode$1.setAttribute("transform", value);
      if (!(value = svgNode$1.transform.baseVal.consolidate())) return identity$c;
      value = value.matrix;
      return decompose$1(value.a, value.b, value.c, value.d, value.e, value.f);
    }

    function interpolateTransform$1(parse, pxComma, pxParen, degParen) {

      function pop(s) {
        return s.length ? s.pop() + " " : "";
      }

      function translate(xa, ya, xb, yb, s, q) {
        if (xa !== xb || ya !== yb) {
          var i = s.push("translate(", null, pxComma, null, pxParen);
          q.push({i: i - 4, x: interpolateNumber$1(xa, xb)}, {i: i - 2, x: interpolateNumber$1(ya, yb)});
        } else if (xb || yb) {
          s.push("translate(" + xb + pxComma + yb + pxParen);
        }
      }

      function rotate(a, b, s, q) {
        if (a !== b) {
          if (a - b > 180) b += 360; else if (b - a > 180) a += 360; // shortest path
          q.push({i: s.push(pop(s) + "rotate(", null, degParen) - 2, x: interpolateNumber$1(a, b)});
        } else if (b) {
          s.push(pop(s) + "rotate(" + b + degParen);
        }
      }

      function skewX(a, b, s, q) {
        if (a !== b) {
          q.push({i: s.push(pop(s) + "skewX(", null, degParen) - 2, x: interpolateNumber$1(a, b)});
        } else if (b) {
          s.push(pop(s) + "skewX(" + b + degParen);
        }
      }

      function scale(xa, ya, xb, yb, s, q) {
        if (xa !== xb || ya !== yb) {
          var i = s.push(pop(s) + "scale(", null, ",", null, ")");
          q.push({i: i - 4, x: interpolateNumber$1(xa, xb)}, {i: i - 2, x: interpolateNumber$1(ya, yb)});
        } else if (xb !== 1 || yb !== 1) {
          s.push(pop(s) + "scale(" + xb + "," + yb + ")");
        }
      }

      return function(a, b) {
        var s = [], // string constants and placeholders
            q = []; // number interpolators
        a = parse(a), b = parse(b);
        translate(a.translateX, a.translateY, b.translateX, b.translateY, s, q);
        rotate(a.rotate, b.rotate, s, q);
        skewX(a.skewX, b.skewX, s, q);
        scale(a.scaleX, a.scaleY, b.scaleX, b.scaleY, s, q);
        a = b = null; // gc
        return function(t) {
          var i = -1, n = q.length, o;
          while (++i < n) s[(o = q[i]).i] = o.x(t);
          return s.join("");
        };
      };
    }

    var interpolateTransformCss$1 = interpolateTransform$1(parseCss$2, "px, ", "px)", "deg)");
    var interpolateTransformSvg$1 = interpolateTransform$1(parseSvg$1, ", ", ")", ")");

    var frame$1 = 0, // is an animation frame pending?
        timeout$2 = 0, // is a timeout pending?
        interval$2 = 0, // are any timers active?
        pokeDelay$1 = 1000, // how frequently we check for clock skew
        taskHead$1,
        taskTail$1,
        clockLast$1 = 0,
        clockNow$1 = 0,
        clockSkew$1 = 0,
        clock$1 = typeof performance === "object" && performance.now ? performance : Date,
        setFrame$1 = typeof window === "object" && window.requestAnimationFrame ? window.requestAnimationFrame.bind(window) : function(f) { setTimeout(f, 17); };

    function now$1() {
      return clockNow$1 || (setFrame$1(clearNow$1), clockNow$1 = clock$1.now() + clockSkew$1);
    }

    function clearNow$1() {
      clockNow$1 = 0;
    }

    function Timer$1() {
      this._call =
      this._time =
      this._next = null;
    }

    Timer$1.prototype = timer$1.prototype = {
      constructor: Timer$1,
      restart: function(callback, delay, time) {
        if (typeof callback !== "function") throw new TypeError("callback is not a function");
        time = (time == null ? now$1() : +time) + (delay == null ? 0 : +delay);
        if (!this._next && taskTail$1 !== this) {
          if (taskTail$1) taskTail$1._next = this;
          else taskHead$1 = this;
          taskTail$1 = this;
        }
        this._call = callback;
        this._time = time;
        sleep$1();
      },
      stop: function() {
        if (this._call) {
          this._call = null;
          this._time = Infinity;
          sleep$1();
        }
      }
    };

    function timer$1(callback, delay, time) {
      var t = new Timer$1;
      t.restart(callback, delay, time);
      return t;
    }

    function timerFlush$1() {
      now$1(); // Get the current time, if not already set.
      ++frame$1; // Pretend we’ve set an alarm, if we haven’t already.
      var t = taskHead$1, e;
      while (t) {
        if ((e = clockNow$1 - t._time) >= 0) t._call.call(null, e);
        t = t._next;
      }
      --frame$1;
    }

    function wake$1() {
      clockNow$1 = (clockLast$1 = clock$1.now()) + clockSkew$1;
      frame$1 = timeout$2 = 0;
      try {
        timerFlush$1();
      } finally {
        frame$1 = 0;
        nap$1();
        clockNow$1 = 0;
      }
    }

    function poke$1() {
      var now = clock$1.now(), delay = now - clockLast$1;
      if (delay > pokeDelay$1) clockSkew$1 -= delay, clockLast$1 = now;
    }

    function nap$1() {
      var t0, t1 = taskHead$1, t2, time = Infinity;
      while (t1) {
        if (t1._call) {
          if (time > t1._time) time = t1._time;
          t0 = t1, t1 = t1._next;
        } else {
          t2 = t1._next, t1._next = null;
          t1 = t0 ? t0._next = t2 : taskHead$1 = t2;
        }
      }
      taskTail$1 = t0;
      sleep$1(time);
    }

    function sleep$1(time) {
      if (frame$1) return; // Soonest alarm already set, or will be.
      if (timeout$2) timeout$2 = clearTimeout(timeout$2);
      var delay = time - clockNow$1; // Strictly less than if we recomputed clockNow.
      if (delay > 24) {
        if (time < Infinity) timeout$2 = setTimeout(wake$1, time - clock$1.now() - clockSkew$1);
        if (interval$2) interval$2 = clearInterval(interval$2);
      } else {
        if (!interval$2) clockLast$1 = clock$1.now(), interval$2 = setInterval(poke$1, pokeDelay$1);
        frame$1 = 1, setFrame$1(wake$1);
      }
    }

    function timeout$3(callback, delay, time) {
      var t = new Timer$1;
      delay = delay == null ? 0 : +delay;
      t.restart(function(elapsed) {
        t.stop();
        callback(elapsed + delay);
      }, delay, time);
      return t;
    }

    var emptyOn$1 = dispatch$1("start", "end", "cancel", "interrupt");
    var emptyTween$1 = [];

    var CREATED$1 = 0;
    var SCHEDULED$1 = 1;
    var STARTING$1 = 2;
    var STARTED$1 = 3;
    var RUNNING$1 = 4;
    var ENDING$1 = 5;
    var ENDED$1 = 6;

    function schedule$2(node, name, id, index, group, timing) {
      var schedules = node.__transition;
      if (!schedules) node.__transition = {};
      else if (id in schedules) return;
      create$2(node, id, {
        name: name,
        index: index, // For context during callback.
        group: group, // For context during callback.
        on: emptyOn$1,
        tween: emptyTween$1,
        time: timing.time,
        delay: timing.delay,
        duration: timing.duration,
        ease: timing.ease,
        timer: null,
        state: CREATED$1
      });
    }

    function init$1(node, id) {
      var schedule = get$4(node, id);
      if (schedule.state > CREATED$1) throw new Error("too late; already scheduled");
      return schedule;
    }

    function set$5(node, id) {
      var schedule = get$4(node, id);
      if (schedule.state > STARTED$1) throw new Error("too late; already running");
      return schedule;
    }

    function get$4(node, id) {
      var schedule = node.__transition;
      if (!schedule || !(schedule = schedule[id])) throw new Error("transition not found");
      return schedule;
    }

    function create$2(node, id, self) {
      var schedules = node.__transition,
          tween;

      // Initialize the self timer when the transition is created.
      // Note the actual delay is not known until the first callback!
      schedules[id] = self;
      self.timer = timer$1(schedule, 0, self.time);

      function schedule(elapsed) {
        self.state = SCHEDULED$1;
        self.timer.restart(start, self.delay, self.time);

        // If the elapsed delay is less than our first sleep, start immediately.
        if (self.delay <= elapsed) start(elapsed - self.delay);
      }

      function start(elapsed) {
        var i, j, n, o;

        // If the state is not SCHEDULED, then we previously errored on start.
        if (self.state !== SCHEDULED$1) return stop();

        for (i in schedules) {
          o = schedules[i];
          if (o.name !== self.name) continue;

          // While this element already has a starting transition during this frame,
          // defer starting an interrupting transition until that transition has a
          // chance to tick (and possibly end); see d3/d3-transition#54!
          if (o.state === STARTED$1) return timeout$3(start);

          // Interrupt the active transition, if any.
          if (o.state === RUNNING$1) {
            o.state = ENDED$1;
            o.timer.stop();
            o.on.call("interrupt", node, node.__data__, o.index, o.group);
            delete schedules[i];
          }

          // Cancel any pre-empted transitions.
          else if (+i < id) {
            o.state = ENDED$1;
            o.timer.stop();
            o.on.call("cancel", node, node.__data__, o.index, o.group);
            delete schedules[i];
          }
        }

        // Defer the first tick to end of the current frame; see d3/d3#1576.
        // Note the transition may be canceled after start and before the first tick!
        // Note this must be scheduled before the start event; see d3/d3-transition#16!
        // Assuming this is successful, subsequent callbacks go straight to tick.
        timeout$3(function() {
          if (self.state === STARTED$1) {
            self.state = RUNNING$1;
            self.timer.restart(tick, self.delay, self.time);
            tick(elapsed);
          }
        });

        // Dispatch the start event.
        // Note this must be done before the tween are initialized.
        self.state = STARTING$1;
        self.on.call("start", node, node.__data__, self.index, self.group);
        if (self.state !== STARTING$1) return; // interrupted
        self.state = STARTED$1;

        // Initialize the tween, deleting null tween.
        tween = new Array(n = self.tween.length);
        for (i = 0, j = -1; i < n; ++i) {
          if (o = self.tween[i].value.call(node, node.__data__, self.index, self.group)) {
            tween[++j] = o;
          }
        }
        tween.length = j + 1;
      }

      function tick(elapsed) {
        var t = elapsed < self.duration ? self.ease.call(null, elapsed / self.duration) : (self.timer.restart(stop), self.state = ENDING$1, 1),
            i = -1,
            n = tween.length;

        while (++i < n) {
          tween[i].call(node, t);
        }

        // Dispatch the end event.
        if (self.state === ENDING$1) {
          self.on.call("end", node, node.__data__, self.index, self.group);
          stop();
        }
      }

      function stop() {
        self.state = ENDED$1;
        self.timer.stop();
        delete schedules[id];
        for (var i in schedules) return; // eslint-disable-line no-unused-vars
        delete node.__transition;
      }
    }

    function interrupt$1(node, name) {
      var schedules = node.__transition,
          schedule,
          active,
          empty = true,
          i;

      if (!schedules) return;

      name = name == null ? null : name + "";

      for (i in schedules) {
        if ((schedule = schedules[i]).name !== name) { empty = false; continue; }
        active = schedule.state > STARTING$1 && schedule.state < ENDING$1;
        schedule.state = ENDED$1;
        schedule.timer.stop();
        schedule.on.call(active ? "interrupt" : "cancel", node, node.__data__, schedule.index, schedule.group);
        delete schedules[i];
      }

      if (empty) delete node.__transition;
    }

    function selection_interrupt$1(name) {
      return this.each(function() {
        interrupt$1(this, name);
      });
    }

    function tweenRemove$1(id, name) {
      var tween0, tween1;
      return function() {
        var schedule = set$5(this, id),
            tween = schedule.tween;

        // If this node shared tween with the previous node,
        // just assign the updated shared tween and we’re done!
        // Otherwise, copy-on-write.
        if (tween !== tween0) {
          tween1 = tween0 = tween;
          for (var i = 0, n = tween1.length; i < n; ++i) {
            if (tween1[i].name === name) {
              tween1 = tween1.slice();
              tween1.splice(i, 1);
              break;
            }
          }
        }

        schedule.tween = tween1;
      };
    }

    function tweenFunction$1(id, name, value) {
      var tween0, tween1;
      if (typeof value !== "function") throw new Error;
      return function() {
        var schedule = set$5(this, id),
            tween = schedule.tween;

        // If this node shared tween with the previous node,
        // just assign the updated shared tween and we’re done!
        // Otherwise, copy-on-write.
        if (tween !== tween0) {
          tween1 = (tween0 = tween).slice();
          for (var t = {name: name, value: value}, i = 0, n = tween1.length; i < n; ++i) {
            if (tween1[i].name === name) {
              tween1[i] = t;
              break;
            }
          }
          if (i === n) tween1.push(t);
        }

        schedule.tween = tween1;
      };
    }

    function transition_tween$1(name, value) {
      var id = this._id;

      name += "";

      if (arguments.length < 2) {
        var tween = get$4(this.node(), id).tween;
        for (var i = 0, n = tween.length, t; i < n; ++i) {
          if ((t = tween[i]).name === name) {
            return t.value;
          }
        }
        return null;
      }

      return this.each((value == null ? tweenRemove$1 : tweenFunction$1)(id, name, value));
    }

    function tweenValue$1(transition, name, value) {
      var id = transition._id;

      transition.each(function() {
        var schedule = set$5(this, id);
        (schedule.value || (schedule.value = {}))[name] = value.apply(this, arguments);
      });

      return function(node) {
        return get$4(node, id).value[name];
      };
    }

    function interpolate$3(a, b) {
      var c;
      return (typeof b === "number" ? interpolateNumber$1
          : b instanceof color$1 ? interpolateRgb$1
          : (c = color$1(b)) ? (b = c, interpolateRgb$1)
          : interpolateString$1)(a, b);
    }

    function attrRemove$3(name) {
      return function() {
        this.removeAttribute(name);
      };
    }

    function attrRemoveNS$3(fullname) {
      return function() {
        this.removeAttributeNS(fullname.space, fullname.local);
      };
    }

    function attrConstant$3(name, interpolate, value1) {
      var string00,
          string1 = value1 + "",
          interpolate0;
      return function() {
        var string0 = this.getAttribute(name);
        return string0 === string1 ? null
            : string0 === string00 ? interpolate0
            : interpolate0 = interpolate(string00 = string0, value1);
      };
    }

    function attrConstantNS$3(fullname, interpolate, value1) {
      var string00,
          string1 = value1 + "",
          interpolate0;
      return function() {
        var string0 = this.getAttributeNS(fullname.space, fullname.local);
        return string0 === string1 ? null
            : string0 === string00 ? interpolate0
            : interpolate0 = interpolate(string00 = string0, value1);
      };
    }

    function attrFunction$3(name, interpolate, value) {
      var string00,
          string10,
          interpolate0;
      return function() {
        var string0, value1 = value(this), string1;
        if (value1 == null) return void this.removeAttribute(name);
        string0 = this.getAttribute(name);
        string1 = value1 + "";
        return string0 === string1 ? null
            : string0 === string00 && string1 === string10 ? interpolate0
            : (string10 = string1, interpolate0 = interpolate(string00 = string0, value1));
      };
    }

    function attrFunctionNS$3(fullname, interpolate, value) {
      var string00,
          string10,
          interpolate0;
      return function() {
        var string0, value1 = value(this), string1;
        if (value1 == null) return void this.removeAttributeNS(fullname.space, fullname.local);
        string0 = this.getAttributeNS(fullname.space, fullname.local);
        string1 = value1 + "";
        return string0 === string1 ? null
            : string0 === string00 && string1 === string10 ? interpolate0
            : (string10 = string1, interpolate0 = interpolate(string00 = string0, value1));
      };
    }

    function transition_attr$1(name, value) {
      var fullname = namespace$1(name), i = fullname === "transform" ? interpolateTransformSvg$1 : interpolate$3;
      return this.attrTween(name, typeof value === "function"
          ? (fullname.local ? attrFunctionNS$3 : attrFunction$3)(fullname, i, tweenValue$1(this, "attr." + name, value))
          : value == null ? (fullname.local ? attrRemoveNS$3 : attrRemove$3)(fullname)
          : (fullname.local ? attrConstantNS$3 : attrConstant$3)(fullname, i, value));
    }

    function attrInterpolate$1(name, i) {
      return function(t) {
        this.setAttribute(name, i.call(this, t));
      };
    }

    function attrInterpolateNS$1(fullname, i) {
      return function(t) {
        this.setAttributeNS(fullname.space, fullname.local, i.call(this, t));
      };
    }

    function attrTweenNS$1(fullname, value) {
      var t0, i0;
      function tween() {
        var i = value.apply(this, arguments);
        if (i !== i0) t0 = (i0 = i) && attrInterpolateNS$1(fullname, i);
        return t0;
      }
      tween._value = value;
      return tween;
    }

    function attrTween$1(name, value) {
      var t0, i0;
      function tween() {
        var i = value.apply(this, arguments);
        if (i !== i0) t0 = (i0 = i) && attrInterpolate$1(name, i);
        return t0;
      }
      tween._value = value;
      return tween;
    }

    function transition_attrTween$1(name, value) {
      var key = "attr." + name;
      if (arguments.length < 2) return (key = this.tween(key)) && key._value;
      if (value == null) return this.tween(key, null);
      if (typeof value !== "function") throw new Error;
      var fullname = namespace$1(name);
      return this.tween(key, (fullname.local ? attrTweenNS$1 : attrTween$1)(fullname, value));
    }

    function delayFunction$1(id, value) {
      return function() {
        init$1(this, id).delay = +value.apply(this, arguments);
      };
    }

    function delayConstant$1(id, value) {
      return value = +value, function() {
        init$1(this, id).delay = value;
      };
    }

    function transition_delay$1(value) {
      var id = this._id;

      return arguments.length
          ? this.each((typeof value === "function"
              ? delayFunction$1
              : delayConstant$1)(id, value))
          : get$4(this.node(), id).delay;
    }

    function durationFunction$1(id, value) {
      return function() {
        set$5(this, id).duration = +value.apply(this, arguments);
      };
    }

    function durationConstant$1(id, value) {
      return value = +value, function() {
        set$5(this, id).duration = value;
      };
    }

    function transition_duration$1(value) {
      var id = this._id;

      return arguments.length
          ? this.each((typeof value === "function"
              ? durationFunction$1
              : durationConstant$1)(id, value))
          : get$4(this.node(), id).duration;
    }

    function easeConstant$1(id, value) {
      if (typeof value !== "function") throw new Error;
      return function() {
        set$5(this, id).ease = value;
      };
    }

    function transition_ease$1(value) {
      var id = this._id;

      return arguments.length
          ? this.each(easeConstant$1(id, value))
          : get$4(this.node(), id).ease;
    }

    function transition_filter$1(match) {
      if (typeof match !== "function") match = matcher$1(match);

      for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, subgroup = subgroups[j] = [], node, i = 0; i < n; ++i) {
          if ((node = group[i]) && match.call(node, node.__data__, i, group)) {
            subgroup.push(node);
          }
        }
      }

      return new Transition$1(subgroups, this._parents, this._name, this._id);
    }

    function transition_merge$1(transition) {
      if (transition._id !== this._id) throw new Error;

      for (var groups0 = this._groups, groups1 = transition._groups, m0 = groups0.length, m1 = groups1.length, m = Math.min(m0, m1), merges = new Array(m0), j = 0; j < m; ++j) {
        for (var group0 = groups0[j], group1 = groups1[j], n = group0.length, merge = merges[j] = new Array(n), node, i = 0; i < n; ++i) {
          if (node = group0[i] || group1[i]) {
            merge[i] = node;
          }
        }
      }

      for (; j < m0; ++j) {
        merges[j] = groups0[j];
      }

      return new Transition$1(merges, this._parents, this._name, this._id);
    }

    function start$1(name) {
      return (name + "").trim().split(/^|\s+/).every(function(t) {
        var i = t.indexOf(".");
        if (i >= 0) t = t.slice(0, i);
        return !t || t === "start";
      });
    }

    function onFunction$1(id, name, listener) {
      var on0, on1, sit = start$1(name) ? init$1 : set$5;
      return function() {
        var schedule = sit(this, id),
            on = schedule.on;

        // If this node shared a dispatch with the previous node,
        // just assign the updated shared dispatch and we’re done!
        // Otherwise, copy-on-write.
        if (on !== on0) (on1 = (on0 = on).copy()).on(name, listener);

        schedule.on = on1;
      };
    }

    function transition_on$1(name, listener) {
      var id = this._id;

      return arguments.length < 2
          ? get$4(this.node(), id).on.on(name)
          : this.each(onFunction$1(id, name, listener));
    }

    function removeFunction$1(id) {
      return function() {
        var parent = this.parentNode;
        for (var i in this.__transition) if (+i !== id) return;
        if (parent) parent.removeChild(this);
      };
    }

    function transition_remove$1() {
      return this.on("end.remove", removeFunction$1(this._id));
    }

    function transition_select$1(select) {
      var name = this._name,
          id = this._id;

      if (typeof select !== "function") select = selector$1(select);

      for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, subgroup = subgroups[j] = new Array(n), node, subnode, i = 0; i < n; ++i) {
          if ((node = group[i]) && (subnode = select.call(node, node.__data__, i, group))) {
            if ("__data__" in node) subnode.__data__ = node.__data__;
            subgroup[i] = subnode;
            schedule$2(subgroup[i], name, id, i, subgroup, get$4(node, id));
          }
        }
      }

      return new Transition$1(subgroups, this._parents, name, id);
    }

    function transition_selectAll$1(select) {
      var name = this._name,
          id = this._id;

      if (typeof select !== "function") select = selectorAll$1(select);

      for (var groups = this._groups, m = groups.length, subgroups = [], parents = [], j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, node, i = 0; i < n; ++i) {
          if (node = group[i]) {
            for (var children = select.call(node, node.__data__, i, group), child, inherit = get$4(node, id), k = 0, l = children.length; k < l; ++k) {
              if (child = children[k]) {
                schedule$2(child, name, id, k, children, inherit);
              }
            }
            subgroups.push(children);
            parents.push(node);
          }
        }
      }

      return new Transition$1(subgroups, parents, name, id);
    }

    var Selection$3 = selection$1.prototype.constructor;

    function transition_selection$1() {
      return new Selection$3(this._groups, this._parents);
    }

    function styleNull$1(name, interpolate) {
      var string00,
          string10,
          interpolate0;
      return function() {
        var string0 = styleValue$1(this, name),
            string1 = (this.style.removeProperty(name), styleValue$1(this, name));
        return string0 === string1 ? null
            : string0 === string00 && string1 === string10 ? interpolate0
            : interpolate0 = interpolate(string00 = string0, string10 = string1);
      };
    }

    function styleRemove$3(name) {
      return function() {
        this.style.removeProperty(name);
      };
    }

    function styleConstant$3(name, interpolate, value1) {
      var string00,
          string1 = value1 + "",
          interpolate0;
      return function() {
        var string0 = styleValue$1(this, name);
        return string0 === string1 ? null
            : string0 === string00 ? interpolate0
            : interpolate0 = interpolate(string00 = string0, value1);
      };
    }

    function styleFunction$3(name, interpolate, value) {
      var string00,
          string10,
          interpolate0;
      return function() {
        var string0 = styleValue$1(this, name),
            value1 = value(this),
            string1 = value1 + "";
        if (value1 == null) string1 = value1 = (this.style.removeProperty(name), styleValue$1(this, name));
        return string0 === string1 ? null
            : string0 === string00 && string1 === string10 ? interpolate0
            : (string10 = string1, interpolate0 = interpolate(string00 = string0, value1));
      };
    }

    function styleMaybeRemove$1(id, name) {
      var on0, on1, listener0, key = "style." + name, event = "end." + key, remove;
      return function() {
        var schedule = set$5(this, id),
            on = schedule.on,
            listener = schedule.value[key] == null ? remove || (remove = styleRemove$3(name)) : undefined;

        // If this node shared a dispatch with the previous node,
        // just assign the updated shared dispatch and we’re done!
        // Otherwise, copy-on-write.
        if (on !== on0 || listener0 !== listener) (on1 = (on0 = on).copy()).on(event, listener0 = listener);

        schedule.on = on1;
      };
    }

    function transition_style$1(name, value, priority) {
      var i = (name += "") === "transform" ? interpolateTransformCss$1 : interpolate$3;
      return value == null ? this
          .styleTween(name, styleNull$1(name, i))
          .on("end.style." + name, styleRemove$3(name))
        : typeof value === "function" ? this
          .styleTween(name, styleFunction$3(name, i, tweenValue$1(this, "style." + name, value)))
          .each(styleMaybeRemove$1(this._id, name))
        : this
          .styleTween(name, styleConstant$3(name, i, value), priority)
          .on("end.style." + name, null);
    }

    function styleInterpolate$1(name, i, priority) {
      return function(t) {
        this.style.setProperty(name, i.call(this, t), priority);
      };
    }

    function styleTween$1(name, value, priority) {
      var t, i0;
      function tween() {
        var i = value.apply(this, arguments);
        if (i !== i0) t = (i0 = i) && styleInterpolate$1(name, i, priority);
        return t;
      }
      tween._value = value;
      return tween;
    }

    function transition_styleTween$1(name, value, priority) {
      var key = "style." + (name += "");
      if (arguments.length < 2) return (key = this.tween(key)) && key._value;
      if (value == null) return this.tween(key, null);
      if (typeof value !== "function") throw new Error;
      return this.tween(key, styleTween$1(name, value, priority == null ? "" : priority));
    }

    function textConstant$3(value) {
      return function() {
        this.textContent = value;
      };
    }

    function textFunction$3(value) {
      return function() {
        var value1 = value(this);
        this.textContent = value1 == null ? "" : value1;
      };
    }

    function transition_text$1(value) {
      return this.tween("text", typeof value === "function"
          ? textFunction$3(tweenValue$1(this, "text", value))
          : textConstant$3(value == null ? "" : value + ""));
    }

    function textInterpolate$1(i) {
      return function(t) {
        this.textContent = i.call(this, t);
      };
    }

    function textTween$1(value) {
      var t0, i0;
      function tween() {
        var i = value.apply(this, arguments);
        if (i !== i0) t0 = (i0 = i) && textInterpolate$1(i);
        return t0;
      }
      tween._value = value;
      return tween;
    }

    function transition_textTween$1(value) {
      var key = "text";
      if (arguments.length < 1) return (key = this.tween(key)) && key._value;
      if (value == null) return this.tween(key, null);
      if (typeof value !== "function") throw new Error;
      return this.tween(key, textTween$1(value));
    }

    function transition_transition$1() {
      var name = this._name,
          id0 = this._id,
          id1 = newId$1();

      for (var groups = this._groups, m = groups.length, j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, node, i = 0; i < n; ++i) {
          if (node = group[i]) {
            var inherit = get$4(node, id0);
            schedule$2(node, name, id1, i, group, {
              time: inherit.time + inherit.delay + inherit.duration,
              delay: 0,
              duration: inherit.duration,
              ease: inherit.ease
            });
          }
        }
      }

      return new Transition$1(groups, this._parents, name, id1);
    }

    function transition_end$1() {
      var on0, on1, that = this, id = that._id, size = that.size();
      return new Promise(function(resolve, reject) {
        var cancel = {value: reject},
            end = {value: function() { if (--size === 0) resolve(); }};

        that.each(function() {
          var schedule = set$5(this, id),
              on = schedule.on;

          // If this node shared a dispatch with the previous node,
          // just assign the updated shared dispatch and we’re done!
          // Otherwise, copy-on-write.
          if (on !== on0) {
            on1 = (on0 = on).copy();
            on1._.cancel.push(cancel);
            on1._.interrupt.push(cancel);
            on1._.end.push(end);
          }

          schedule.on = on1;
        });
      });
    }

    var id$1 = 0;

    function Transition$1(groups, parents, name, id) {
      this._groups = groups;
      this._parents = parents;
      this._name = name;
      this._id = id;
    }

    function transition$1(name) {
      return selection$1().transition(name);
    }

    function newId$1() {
      return ++id$1;
    }

    var selection_prototype$1 = selection$1.prototype;

    Transition$1.prototype = transition$1.prototype = {
      constructor: Transition$1,
      select: transition_select$1,
      selectAll: transition_selectAll$1,
      filter: transition_filter$1,
      merge: transition_merge$1,
      selection: transition_selection$1,
      transition: transition_transition$1,
      call: selection_prototype$1.call,
      nodes: selection_prototype$1.nodes,
      node: selection_prototype$1.node,
      size: selection_prototype$1.size,
      empty: selection_prototype$1.empty,
      each: selection_prototype$1.each,
      on: transition_on$1,
      attr: transition_attr$1,
      attrTween: transition_attrTween$1,
      style: transition_style$1,
      styleTween: transition_styleTween$1,
      text: transition_text$1,
      textTween: transition_textTween$1,
      remove: transition_remove$1,
      tween: transition_tween$1,
      delay: transition_delay$1,
      duration: transition_duration$1,
      ease: transition_ease$1,
      end: transition_end$1
    };

    var defaultTiming$1 = {
      time: null, // Set on use.
      delay: 0,
      duration: 250,
      ease: cubicInOut
    };

    function inherit$1(node, id) {
      var timing;
      while (!(timing = node.__transition) || !(timing = timing[id])) {
        if (!(node = node.parentNode)) {
          return defaultTiming$1.time = now$1(), defaultTiming$1;
        }
      }
      return timing;
    }

    function selection_transition$1(name) {
      var id,
          timing;

      if (name instanceof Transition$1) {
        id = name._id, name = name._name;
      } else {
        id = newId$1(), (timing = defaultTiming$1).time = now$1(), name = name == null ? null : name + "";
      }

      for (var groups = this._groups, m = groups.length, j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, node, i = 0; i < n; ++i) {
          if (node = group[i]) {
            schedule$2(node, name, id, i, group, timing || inherit$1(node, id));
          }
        }
      }

      return new Transition$1(groups, this._parents, name, id);
    }

    selection$1.prototype.interrupt = selection_interrupt$1;
    selection$1.prototype.transition = selection_transition$1;

    function constant$h(x) {
      return function() {
        return x;
      };
    }

    function BrushEvent$1(target, type, selection) {
      this.target = target;
      this.type = type;
      this.selection = selection;
    }

    function nopropagation$3() {
      event$1.stopImmediatePropagation();
    }

    function noevent$4() {
      event$1.preventDefault();
      event$1.stopImmediatePropagation();
    }

    var MODE_DRAG$1 = {name: "drag"},
        MODE_SPACE$1 = {name: "space"},
        MODE_HANDLE$1 = {name: "handle"},
        MODE_CENTER$1 = {name: "center"};

    var X$1 = {
      name: "x",
      handles: ["e", "w"].map(type$1),
      input: function(x, e) { return x && [[x[0], e[0][1]], [x[1], e[1][1]]]; },
      output: function(xy) { return xy && [xy[0][0], xy[1][0]]; }
    };

    var Y$1 = {
      name: "y",
      handles: ["n", "s"].map(type$1),
      input: function(y, e) { return y && [[e[0][0], y[0]], [e[1][0], y[1]]]; },
      output: function(xy) { return xy && [xy[0][1], xy[1][1]]; }
    };

    var XY$1 = {
      name: "xy",
      handles: ["n", "e", "s", "w", "nw", "ne", "se", "sw"].map(type$1),
      input: function(xy) { return xy; },
      output: function(xy) { return xy; }
    };

    var cursors$1 = {
      overlay: "crosshair",
      selection: "move",
      n: "ns-resize",
      e: "ew-resize",
      s: "ns-resize",
      w: "ew-resize",
      nw: "nwse-resize",
      ne: "nesw-resize",
      se: "nwse-resize",
      sw: "nesw-resize"
    };

    var flipX$1 = {
      e: "w",
      w: "e",
      nw: "ne",
      ne: "nw",
      se: "sw",
      sw: "se"
    };

    var flipY$1 = {
      n: "s",
      s: "n",
      nw: "sw",
      ne: "se",
      se: "ne",
      sw: "nw"
    };

    var signsX$1 = {
      overlay: +1,
      selection: +1,
      n: null,
      e: +1,
      s: null,
      w: -1,
      nw: -1,
      ne: +1,
      se: +1,
      sw: -1
    };

    var signsY$1 = {
      overlay: +1,
      selection: +1,
      n: -1,
      e: null,
      s: +1,
      w: null,
      nw: -1,
      ne: -1,
      se: +1,
      sw: +1
    };

    function type$1(t) {
      return {type: t};
    }

    // Ignore right-click, since that should open the context menu.
    function defaultFilter$3() {
      return !event$1.button;
    }

    function defaultExtent$2() {
      var svg = this.ownerSVGElement || this;
      return [[0, 0], [svg.width.baseVal.value, svg.height.baseVal.value]];
    }

    // Like d3.local, but with the name “__brush” rather than auto-generated.
    function local$2(node) {
      while (!node.__brush) if (!(node = node.parentNode)) return;
      return node.__brush;
    }

    function empty$3(extent) {
      return extent[0][0] === extent[1][0]
          || extent[0][1] === extent[1][1];
    }

    function brushSelection$1(node) {
      var state = node.__brush;
      return state ? state.dim.output(state.selection) : null;
    }

    function brushX$1() {
      return brush$3(X$1);
    }

    function brushY$1() {
      return brush$3(Y$1);
    }

    function brush$2() {
      return brush$3(XY$1);
    }

    function brush$3(dim) {
      var extent = defaultExtent$2,
          filter = defaultFilter$3,
          listeners = dispatch$1(brush, "start", "brush", "end"),
          handleSize = 6,
          touchending;

      function brush(group) {
        var overlay = group
            .property("__brush", initialize)
          .selectAll(".overlay")
          .data([type$1("overlay")]);

        overlay.enter().append("rect")
            .attr("class", "overlay")
            .attr("pointer-events", "all")
            .attr("cursor", cursors$1.overlay)
          .merge(overlay)
            .each(function() {
              var extent = local$2(this).extent;
              select$1(this)
                  .attr("x", extent[0][0])
                  .attr("y", extent[0][1])
                  .attr("width", extent[1][0] - extent[0][0])
                  .attr("height", extent[1][1] - extent[0][1]);
            });

        group.selectAll(".selection")
          .data([type$1("selection")])
          .enter().append("rect")
            .attr("class", "selection")
            .attr("cursor", cursors$1.selection)
            .attr("fill", "#777")
            .attr("fill-opacity", 0.3)
            .attr("stroke", "#fff")
            .attr("shape-rendering", "crispEdges");

        var handle = group.selectAll(".handle")
          .data(dim.handles, function(d) { return d.type; });

        handle.exit().remove();

        handle.enter().append("rect")
            .attr("class", function(d) { return "handle handle--" + d.type; })
            .attr("cursor", function(d) { return cursors$1[d.type]; });

        group
            .each(redraw)
            .attr("fill", "none")
            .attr("pointer-events", "all")
            .style("-webkit-tap-highlight-color", "rgba(0,0,0,0)")
            .on("mousedown.brush touchstart.brush", started);
      }

      brush.move = function(group, selection) {
        if (group.selection) {
          group
              .on("start.brush", function() { emitter(this, arguments).beforestart().start(); })
              .on("interrupt.brush end.brush", function() { emitter(this, arguments).end(); })
              .tween("brush", function() {
                var that = this,
                    state = that.__brush,
                    emit = emitter(that, arguments),
                    selection0 = state.selection,
                    selection1 = dim.input(typeof selection === "function" ? selection.apply(this, arguments) : selection, state.extent),
                    i = interpolate$2(selection0, selection1);

                function tween(t) {
                  state.selection = t === 1 && empty$3(selection1) ? null : i(t);
                  redraw.call(that);
                  emit.brush();
                }

                return selection0 && selection1 ? tween : tween(1);
              });
        } else {
          group
              .each(function() {
                var that = this,
                    args = arguments,
                    state = that.__brush,
                    selection1 = dim.input(typeof selection === "function" ? selection.apply(that, args) : selection, state.extent),
                    emit = emitter(that, args).beforestart();

                interrupt$1(that);
                state.selection = selection1 == null || empty$3(selection1) ? null : selection1;
                redraw.call(that);
                emit.start().brush().end();
              });
        }
      };

      function redraw() {
        var group = select$1(this),
            selection = local$2(this).selection;

        if (selection) {
          group.selectAll(".selection")
              .style("display", null)
              .attr("x", selection[0][0])
              .attr("y", selection[0][1])
              .attr("width", selection[1][0] - selection[0][0])
              .attr("height", selection[1][1] - selection[0][1]);

          group.selectAll(".handle")
              .style("display", null)
              .attr("x", function(d) { return d.type[d.type.length - 1] === "e" ? selection[1][0] - handleSize / 2 : selection[0][0] - handleSize / 2; })
              .attr("y", function(d) { return d.type[0] === "s" ? selection[1][1] - handleSize / 2 : selection[0][1] - handleSize / 2; })
              .attr("width", function(d) { return d.type === "n" || d.type === "s" ? selection[1][0] - selection[0][0] + handleSize : handleSize; })
              .attr("height", function(d) { return d.type === "e" || d.type === "w" ? selection[1][1] - selection[0][1] + handleSize : handleSize; });
        }

        else {
          group.selectAll(".selection,.handle")
              .style("display", "none")
              .attr("x", null)
              .attr("y", null)
              .attr("width", null)
              .attr("height", null);
        }
      }

      function emitter(that, args) {
        return that.__brush.emitter || new Emitter(that, args);
      }

      function Emitter(that, args) {
        this.that = that;
        this.args = args;
        this.state = that.__brush;
        this.active = 0;
      }

      Emitter.prototype = {
        beforestart: function() {
          if (++this.active === 1) this.state.emitter = this, this.starting = true;
          return this;
        },
        start: function() {
          if (this.starting) this.starting = false, this.emit("start");
          return this;
        },
        brush: function() {
          this.emit("brush");
          return this;
        },
        end: function() {
          if (--this.active === 0) delete this.state.emitter, this.emit("end");
          return this;
        },
        emit: function(type) {
          customEvent$1(new BrushEvent$1(brush, type, dim.output(this.state.selection)), listeners.apply, listeners, [type, this.that, this.args]);
        }
      };

      function started() {
        if (event$1.touches) { if (event$1.changedTouches.length < event$1.touches.length) return noevent$4(); }
        else if (touchending) return;
        if (!filter.apply(this, arguments)) return;

        var that = this,
            type = event$1.target.__data__.type,
            mode = (event$1.metaKey ? type = "overlay" : type) === "selection" ? MODE_DRAG$1 : (event$1.altKey ? MODE_CENTER$1 : MODE_HANDLE$1),
            signX = dim === Y$1 ? null : signsX$1[type],
            signY = dim === X$1 ? null : signsY$1[type],
            state = local$2(that),
            extent = state.extent,
            selection = state.selection,
            W = extent[0][0], w0, w1,
            N = extent[0][1], n0, n1,
            E = extent[1][0], e0, e1,
            S = extent[1][1], s0, s1,
            dx,
            dy,
            moving,
            shifting = signX && signY && event$1.shiftKey,
            lockX,
            lockY,
            point0 = mouse$1(that),
            point = point0,
            emit = emitter(that, arguments).beforestart();

        if (type === "overlay") {
          state.selection = selection = [
            [w0 = dim === Y$1 ? W : point0[0], n0 = dim === X$1 ? N : point0[1]],
            [e0 = dim === Y$1 ? E : w0, s0 = dim === X$1 ? S : n0]
          ];
        } else {
          w0 = selection[0][0];
          n0 = selection[0][1];
          e0 = selection[1][0];
          s0 = selection[1][1];
        }

        w1 = w0;
        n1 = n0;
        e1 = e0;
        s1 = s0;

        var group = select$1(that)
            .attr("pointer-events", "none");

        var overlay = group.selectAll(".overlay")
            .attr("cursor", cursors$1[type]);

        if (event$1.touches) {
          group
              .on("touchmove.brush", moved, true)
              .on("touchend.brush touchcancel.brush", ended, true);
        } else {
          var view = select$1(event$1.view)
              .on("keydown.brush", keydowned, true)
              .on("keyup.brush", keyupped, true)
              .on("mousemove.brush", moved, true)
              .on("mouseup.brush", ended, true);

          dragDisable$1(event$1.view);
        }

        nopropagation$3();
        interrupt$1(that);
        redraw.call(that);
        emit.start();

        function moved() {
          var point1 = mouse$1(that);
          if (shifting && !lockX && !lockY) {
            if (Math.abs(point1[0] - point[0]) > Math.abs(point1[1] - point[1])) lockY = true;
            else lockX = true;
          }
          point = point1;
          moving = true;
          noevent$4();
          move();
        }

        function move() {
          var t;

          dx = point[0] - point0[0];
          dy = point[1] - point0[1];

          switch (mode) {
            case MODE_SPACE$1:
            case MODE_DRAG$1: {
              if (signX) dx = Math.max(W - w0, Math.min(E - e0, dx)), w1 = w0 + dx, e1 = e0 + dx;
              if (signY) dy = Math.max(N - n0, Math.min(S - s0, dy)), n1 = n0 + dy, s1 = s0 + dy;
              break;
            }
            case MODE_HANDLE$1: {
              if (signX < 0) dx = Math.max(W - w0, Math.min(E - w0, dx)), w1 = w0 + dx, e1 = e0;
              else if (signX > 0) dx = Math.max(W - e0, Math.min(E - e0, dx)), w1 = w0, e1 = e0 + dx;
              if (signY < 0) dy = Math.max(N - n0, Math.min(S - n0, dy)), n1 = n0 + dy, s1 = s0;
              else if (signY > 0) dy = Math.max(N - s0, Math.min(S - s0, dy)), n1 = n0, s1 = s0 + dy;
              break;
            }
            case MODE_CENTER$1: {
              if (signX) w1 = Math.max(W, Math.min(E, w0 - dx * signX)), e1 = Math.max(W, Math.min(E, e0 + dx * signX));
              if (signY) n1 = Math.max(N, Math.min(S, n0 - dy * signY)), s1 = Math.max(N, Math.min(S, s0 + dy * signY));
              break;
            }
          }

          if (e1 < w1) {
            signX *= -1;
            t = w0, w0 = e0, e0 = t;
            t = w1, w1 = e1, e1 = t;
            if (type in flipX$1) overlay.attr("cursor", cursors$1[type = flipX$1[type]]);
          }

          if (s1 < n1) {
            signY *= -1;
            t = n0, n0 = s0, s0 = t;
            t = n1, n1 = s1, s1 = t;
            if (type in flipY$1) overlay.attr("cursor", cursors$1[type = flipY$1[type]]);
          }

          if (state.selection) selection = state.selection; // May be set by brush.move!
          if (lockX) w1 = selection[0][0], e1 = selection[1][0];
          if (lockY) n1 = selection[0][1], s1 = selection[1][1];

          if (selection[0][0] !== w1
              || selection[0][1] !== n1
              || selection[1][0] !== e1
              || selection[1][1] !== s1) {
            state.selection = [[w1, n1], [e1, s1]];
            redraw.call(that);
            emit.brush();
          }
        }

        function ended() {
          nopropagation$3();
          if (event$1.touches) {
            if (event$1.touches.length) return;
            if (touchending) clearTimeout(touchending);
            touchending = setTimeout(function() { touchending = null; }, 500); // Ghost clicks are delayed!
            group.on("touchmove.brush touchend.brush touchcancel.brush", null);
          } else {
            yesdrag$1(event$1.view, moving);
            view.on("keydown.brush keyup.brush mousemove.brush mouseup.brush", null);
          }
          group.attr("pointer-events", "all");
          overlay.attr("cursor", cursors$1.overlay);
          if (state.selection) selection = state.selection; // May be set by brush.move (on start)!
          if (empty$3(selection)) state.selection = null, redraw.call(that);
          emit.end();
        }

        function keydowned() {
          switch (event$1.keyCode) {
            case 16: { // SHIFT
              shifting = signX && signY;
              break;
            }
            case 18: { // ALT
              if (mode === MODE_HANDLE$1) {
                if (signX) e0 = e1 - dx * signX, w0 = w1 + dx * signX;
                if (signY) s0 = s1 - dy * signY, n0 = n1 + dy * signY;
                mode = MODE_CENTER$1;
                move();
              }
              break;
            }
            case 32: { // SPACE; takes priority over ALT
              if (mode === MODE_HANDLE$1 || mode === MODE_CENTER$1) {
                if (signX < 0) e0 = e1 - dx; else if (signX > 0) w0 = w1 - dx;
                if (signY < 0) s0 = s1 - dy; else if (signY > 0) n0 = n1 - dy;
                mode = MODE_SPACE$1;
                overlay.attr("cursor", cursors$1.selection);
                move();
              }
              break;
            }
            default: return;
          }
          noevent$4();
        }

        function keyupped() {
          switch (event$1.keyCode) {
            case 16: { // SHIFT
              if (shifting) {
                lockX = lockY = shifting = false;
                move();
              }
              break;
            }
            case 18: { // ALT
              if (mode === MODE_CENTER$1) {
                if (signX < 0) e0 = e1; else if (signX > 0) w0 = w1;
                if (signY < 0) s0 = s1; else if (signY > 0) n0 = n1;
                mode = MODE_HANDLE$1;
                move();
              }
              break;
            }
            case 32: { // SPACE
              if (mode === MODE_SPACE$1) {
                if (event$1.altKey) {
                  if (signX) e0 = e1 - dx * signX, w0 = w1 + dx * signX;
                  if (signY) s0 = s1 - dy * signY, n0 = n1 + dy * signY;
                  mode = MODE_CENTER$1;
                } else {
                  if (signX < 0) e0 = e1; else if (signX > 0) w0 = w1;
                  if (signY < 0) s0 = s1; else if (signY > 0) n0 = n1;
                  mode = MODE_HANDLE$1;
                }
                overlay.attr("cursor", cursors$1[type]);
                move();
              }
              break;
            }
            default: return;
          }
          noevent$4();
        }
      }

      function initialize() {
        var state = this.__brush || {selection: null};
        state.extent = extent.apply(this, arguments);
        state.dim = dim;
        return state;
      }

      brush.extent = function(_) {
        return arguments.length ? (extent = typeof _ === "function" ? _ : constant$h([[+_[0][0], +_[0][1]], [+_[1][0], +_[1][1]]]), brush) : extent;
      };

      brush.filter = function(_) {
        return arguments.length ? (filter = typeof _ === "function" ? _ : constant$h(!!_), brush) : filter;
      };

      brush.handleSize = function(_) {
        return arguments.length ? (handleSize = +_, brush) : handleSize;
      };

      brush.on = function() {
        var value = listeners.on.apply(listeners, arguments);
        return value === listeners ? brush : value;
      };

      return brush;
    }

    function ascending$6(a, b) {
      return a < b ? -1 : a > b ? 1 : a >= b ? 0 : NaN;
    }

    function bisector$2(compare) {
      if (compare.length === 1) compare = ascendingComparator$2(compare);
      return {
        left: function(a, x, lo, hi) {
          if (lo == null) lo = 0;
          if (hi == null) hi = a.length;
          while (lo < hi) {
            var mid = lo + hi >>> 1;
            if (compare(a[mid], x) < 0) lo = mid + 1;
            else hi = mid;
          }
          return lo;
        },
        right: function(a, x, lo, hi) {
          if (lo == null) lo = 0;
          if (hi == null) hi = a.length;
          while (lo < hi) {
            var mid = lo + hi >>> 1;
            if (compare(a[mid], x) > 0) hi = mid;
            else lo = mid + 1;
          }
          return lo;
        }
      };
    }

    function ascendingComparator$2(f) {
      return function(d, x) {
        return ascending$6(f(d), x);
      };
    }

    var ascendingBisect$2 = bisector$2(ascending$6);

    function range$2(start, stop, step) {
      start = +start, stop = +stop, step = (n = arguments.length) < 2 ? (stop = start, start = 0, 1) : n < 3 ? 1 : +step;

      var i = -1,
          n = Math.max(0, Math.ceil((stop - start) / step)) | 0,
          range = new Array(n);

      while (++i < n) {
        range[i] = start + i * step;
      }

      return range;
    }

    var cos$3 = Math.cos;
    var sin$3 = Math.sin;
    var pi$5 = Math.PI;
    var halfPi$4 = pi$5 / 2;
    var tau$5 = pi$5 * 2;
    var max$4 = Math.max;

    function compareValue$1(compare) {
      return function(a, b) {
        return compare(
          a.source.value + a.target.value,
          b.source.value + b.target.value
        );
      };
    }

    function chord$1() {
      var padAngle = 0,
          sortGroups = null,
          sortSubgroups = null,
          sortChords = null;

      function chord(matrix) {
        var n = matrix.length,
            groupSums = [],
            groupIndex = range$2(n),
            subgroupIndex = [],
            chords = [],
            groups = chords.groups = new Array(n),
            subgroups = new Array(n * n),
            k,
            x,
            x0,
            dx,
            i,
            j;

        // Compute the sum.
        k = 0, i = -1; while (++i < n) {
          x = 0, j = -1; while (++j < n) {
            x += matrix[i][j];
          }
          groupSums.push(x);
          subgroupIndex.push(range$2(n));
          k += x;
        }

        // Sort groups…
        if (sortGroups) groupIndex.sort(function(a, b) {
          return sortGroups(groupSums[a], groupSums[b]);
        });

        // Sort subgroups…
        if (sortSubgroups) subgroupIndex.forEach(function(d, i) {
          d.sort(function(a, b) {
            return sortSubgroups(matrix[i][a], matrix[i][b]);
          });
        });

        // Convert the sum to scaling factor for [0, 2pi].
        // TODO Allow start and end angle to be specified?
        // TODO Allow padding to be specified as percentage?
        k = max$4(0, tau$5 - padAngle * n) / k;
        dx = k ? padAngle : tau$5 / n;

        // Compute the start and end angle for each group and subgroup.
        // Note: Opera has a bug reordering object literal properties!
        x = 0, i = -1; while (++i < n) {
          x0 = x, j = -1; while (++j < n) {
            var di = groupIndex[i],
                dj = subgroupIndex[di][j],
                v = matrix[di][dj],
                a0 = x,
                a1 = x += v * k;
            subgroups[dj * n + di] = {
              index: di,
              subindex: dj,
              startAngle: a0,
              endAngle: a1,
              value: v
            };
          }
          groups[di] = {
            index: di,
            startAngle: x0,
            endAngle: x,
            value: groupSums[di]
          };
          x += dx;
        }

        // Generate chords for each (non-empty) subgroup-subgroup link.
        i = -1; while (++i < n) {
          j = i - 1; while (++j < n) {
            var source = subgroups[j * n + i],
                target = subgroups[i * n + j];
            if (source.value || target.value) {
              chords.push(source.value < target.value
                  ? {source: target, target: source}
                  : {source: source, target: target});
            }
          }
        }

        return sortChords ? chords.sort(sortChords) : chords;
      }

      chord.padAngle = function(_) {
        return arguments.length ? (padAngle = max$4(0, _), chord) : padAngle;
      };

      chord.sortGroups = function(_) {
        return arguments.length ? (sortGroups = _, chord) : sortGroups;
      };

      chord.sortSubgroups = function(_) {
        return arguments.length ? (sortSubgroups = _, chord) : sortSubgroups;
      };

      chord.sortChords = function(_) {
        return arguments.length ? (_ == null ? sortChords = null : (sortChords = compareValue$1(_))._ = _, chord) : sortChords && sortChords._;
      };

      return chord;
    }

    var slice$9 = Array.prototype.slice;

    function constant$i(x) {
      return function() {
        return x;
      };
    }

    var pi$6 = Math.PI,
        tau$6 = 2 * pi$6,
        epsilon$6 = 1e-6,
        tauEpsilon$1 = tau$6 - epsilon$6;

    function Path$1() {
      this._x0 = this._y0 = // start of current subpath
      this._x1 = this._y1 = null; // end of current subpath
      this._ = "";
    }

    function path$1() {
      return new Path$1;
    }

    Path$1.prototype = path$1.prototype = {
      constructor: Path$1,
      moveTo: function(x, y) {
        this._ += "M" + (this._x0 = this._x1 = +x) + "," + (this._y0 = this._y1 = +y);
      },
      closePath: function() {
        if (this._x1 !== null) {
          this._x1 = this._x0, this._y1 = this._y0;
          this._ += "Z";
        }
      },
      lineTo: function(x, y) {
        this._ += "L" + (this._x1 = +x) + "," + (this._y1 = +y);
      },
      quadraticCurveTo: function(x1, y1, x, y) {
        this._ += "Q" + (+x1) + "," + (+y1) + "," + (this._x1 = +x) + "," + (this._y1 = +y);
      },
      bezierCurveTo: function(x1, y1, x2, y2, x, y) {
        this._ += "C" + (+x1) + "," + (+y1) + "," + (+x2) + "," + (+y2) + "," + (this._x1 = +x) + "," + (this._y1 = +y);
      },
      arcTo: function(x1, y1, x2, y2, r) {
        x1 = +x1, y1 = +y1, x2 = +x2, y2 = +y2, r = +r;
        var x0 = this._x1,
            y0 = this._y1,
            x21 = x2 - x1,
            y21 = y2 - y1,
            x01 = x0 - x1,
            y01 = y0 - y1,
            l01_2 = x01 * x01 + y01 * y01;

        // Is the radius negative? Error.
        if (r < 0) throw new Error("negative radius: " + r);

        // Is this path empty? Move to (x1,y1).
        if (this._x1 === null) {
          this._ += "M" + (this._x1 = x1) + "," + (this._y1 = y1);
        }

        // Or, is (x1,y1) coincident with (x0,y0)? Do nothing.
        else if (!(l01_2 > epsilon$6));

        // Or, are (x0,y0), (x1,y1) and (x2,y2) collinear?
        // Equivalently, is (x1,y1) coincident with (x2,y2)?
        // Or, is the radius zero? Line to (x1,y1).
        else if (!(Math.abs(y01 * x21 - y21 * x01) > epsilon$6) || !r) {
          this._ += "L" + (this._x1 = x1) + "," + (this._y1 = y1);
        }

        // Otherwise, draw an arc!
        else {
          var x20 = x2 - x0,
              y20 = y2 - y0,
              l21_2 = x21 * x21 + y21 * y21,
              l20_2 = x20 * x20 + y20 * y20,
              l21 = Math.sqrt(l21_2),
              l01 = Math.sqrt(l01_2),
              l = r * Math.tan((pi$6 - Math.acos((l21_2 + l01_2 - l20_2) / (2 * l21 * l01))) / 2),
              t01 = l / l01,
              t21 = l / l21;

          // If the start tangent is not coincident with (x0,y0), line to.
          if (Math.abs(t01 - 1) > epsilon$6) {
            this._ += "L" + (x1 + t01 * x01) + "," + (y1 + t01 * y01);
          }

          this._ += "A" + r + "," + r + ",0,0," + (+(y01 * x20 > x01 * y20)) + "," + (this._x1 = x1 + t21 * x21) + "," + (this._y1 = y1 + t21 * y21);
        }
      },
      arc: function(x, y, r, a0, a1, ccw) {
        x = +x, y = +y, r = +r, ccw = !!ccw;
        var dx = r * Math.cos(a0),
            dy = r * Math.sin(a0),
            x0 = x + dx,
            y0 = y + dy,
            cw = 1 ^ ccw,
            da = ccw ? a0 - a1 : a1 - a0;

        // Is the radius negative? Error.
        if (r < 0) throw new Error("negative radius: " + r);

        // Is this path empty? Move to (x0,y0).
        if (this._x1 === null) {
          this._ += "M" + x0 + "," + y0;
        }

        // Or, is (x0,y0) not coincident with the previous point? Line to (x0,y0).
        else if (Math.abs(this._x1 - x0) > epsilon$6 || Math.abs(this._y1 - y0) > epsilon$6) {
          this._ += "L" + x0 + "," + y0;
        }

        // Is this arc empty? We’re done.
        if (!r) return;

        // Does the angle go the wrong way? Flip the direction.
        if (da < 0) da = da % tau$6 + tau$6;

        // Is this a complete circle? Draw two arcs to complete the circle.
        if (da > tauEpsilon$1) {
          this._ += "A" + r + "," + r + ",0,1," + cw + "," + (x - dx) + "," + (y - dy) + "A" + r + "," + r + ",0,1," + cw + "," + (this._x1 = x0) + "," + (this._y1 = y0);
        }

        // Is this arc non-empty? Draw an arc!
        else if (da > epsilon$6) {
          this._ += "A" + r + "," + r + ",0," + (+(da >= pi$6)) + "," + cw + "," + (this._x1 = x + r * Math.cos(a1)) + "," + (this._y1 = y + r * Math.sin(a1));
        }
      },
      rect: function(x, y, w, h) {
        this._ += "M" + (this._x0 = this._x1 = +x) + "," + (this._y0 = this._y1 = +y) + "h" + (+w) + "v" + (+h) + "h" + (-w) + "Z";
      },
      toString: function() {
        return this._;
      }
    };

    function defaultSource$2(d) {
      return d.source;
    }

    function defaultTarget$1(d) {
      return d.target;
    }

    function defaultRadius$2(d) {
      return d.radius;
    }

    function defaultStartAngle$1(d) {
      return d.startAngle;
    }

    function defaultEndAngle$1(d) {
      return d.endAngle;
    }

    function ribbon$1() {
      var source = defaultSource$2,
          target = defaultTarget$1,
          radius = defaultRadius$2,
          startAngle = defaultStartAngle$1,
          endAngle = defaultEndAngle$1,
          context = null;

      function ribbon() {
        var buffer,
            argv = slice$9.call(arguments),
            s = source.apply(this, argv),
            t = target.apply(this, argv),
            sr = +radius.apply(this, (argv[0] = s, argv)),
            sa0 = startAngle.apply(this, argv) - halfPi$4,
            sa1 = endAngle.apply(this, argv) - halfPi$4,
            sx0 = sr * cos$3(sa0),
            sy0 = sr * sin$3(sa0),
            tr = +radius.apply(this, (argv[0] = t, argv)),
            ta0 = startAngle.apply(this, argv) - halfPi$4,
            ta1 = endAngle.apply(this, argv) - halfPi$4;

        if (!context) context = buffer = path$1();

        context.moveTo(sx0, sy0);
        context.arc(0, 0, sr, sa0, sa1);
        if (sa0 !== ta0 || sa1 !== ta1) { // TODO sr !== tr?
          context.quadraticCurveTo(0, 0, tr * cos$3(ta0), tr * sin$3(ta0));
          context.arc(0, 0, tr, ta0, ta1);
        }
        context.quadraticCurveTo(0, 0, sx0, sy0);
        context.closePath();

        if (buffer) return context = null, buffer + "" || null;
      }

      ribbon.radius = function(_) {
        return arguments.length ? (radius = typeof _ === "function" ? _ : constant$i(+_), ribbon) : radius;
      };

      ribbon.startAngle = function(_) {
        return arguments.length ? (startAngle = typeof _ === "function" ? _ : constant$i(+_), ribbon) : startAngle;
      };

      ribbon.endAngle = function(_) {
        return arguments.length ? (endAngle = typeof _ === "function" ? _ : constant$i(+_), ribbon) : endAngle;
      };

      ribbon.source = function(_) {
        return arguments.length ? (source = _, ribbon) : source;
      };

      ribbon.target = function(_) {
        return arguments.length ? (target = _, ribbon) : target;
      };

      ribbon.context = function(_) {
        return arguments.length ? ((context = _ == null ? null : _), ribbon) : context;
      };

      return ribbon;
    }

    var prefix$1 = "$";

    function Map$2() {}

    Map$2.prototype = map$5.prototype = {
      constructor: Map$2,
      has: function(key) {
        return (prefix$1 + key) in this;
      },
      get: function(key) {
        return this[prefix$1 + key];
      },
      set: function(key, value) {
        this[prefix$1 + key] = value;
        return this;
      },
      remove: function(key) {
        var property = prefix$1 + key;
        return property in this && delete this[property];
      },
      clear: function() {
        for (var property in this) if (property[0] === prefix$1) delete this[property];
      },
      keys: function() {
        var keys = [];
        for (var property in this) if (property[0] === prefix$1) keys.push(property.slice(1));
        return keys;
      },
      values: function() {
        var values = [];
        for (var property in this) if (property[0] === prefix$1) values.push(this[property]);
        return values;
      },
      entries: function() {
        var entries = [];
        for (var property in this) if (property[0] === prefix$1) entries.push({key: property.slice(1), value: this[property]});
        return entries;
      },
      size: function() {
        var size = 0;
        for (var property in this) if (property[0] === prefix$1) ++size;
        return size;
      },
      empty: function() {
        for (var property in this) if (property[0] === prefix$1) return false;
        return true;
      },
      each: function(f) {
        for (var property in this) if (property[0] === prefix$1) f(this[property], property.slice(1), this);
      }
    };

    function map$5(object, f) {
      var map = new Map$2;

      // Copy constructor.
      if (object instanceof Map$2) object.each(function(value, key) { map.set(key, value); });

      // Index array by numeric index or specified key function.
      else if (Array.isArray(object)) {
        var i = -1,
            n = object.length,
            o;

        if (f == null) while (++i < n) map.set(i, object[i]);
        else while (++i < n) map.set(f(o = object[i], i, object), o);
      }

      // Convert object to map.
      else if (object) for (var key in object) map.set(key, object[key]);

      return map;
    }

    function nest$1() {
      var keys = [],
          sortKeys = [],
          sortValues,
          rollup,
          nest;

      function apply(array, depth, createResult, setResult) {
        if (depth >= keys.length) {
          if (sortValues != null) array.sort(sortValues);
          return rollup != null ? rollup(array) : array;
        }

        var i = -1,
            n = array.length,
            key = keys[depth++],
            keyValue,
            value,
            valuesByKey = map$5(),
            values,
            result = createResult();

        while (++i < n) {
          if (values = valuesByKey.get(keyValue = key(value = array[i]) + "")) {
            values.push(value);
          } else {
            valuesByKey.set(keyValue, [value]);
          }
        }

        valuesByKey.each(function(values, key) {
          setResult(result, key, apply(values, depth, createResult, setResult));
        });

        return result;
      }

      function entries(map, depth) {
        if (++depth > keys.length) return map;
        var array, sortKey = sortKeys[depth - 1];
        if (rollup != null && depth >= keys.length) array = map.entries();
        else array = [], map.each(function(v, k) { array.push({key: k, values: entries(v, depth)}); });
        return sortKey != null ? array.sort(function(a, b) { return sortKey(a.key, b.key); }) : array;
      }

      return nest = {
        object: function(array) { return apply(array, 0, createObject$1, setObject$2); },
        map: function(array) { return apply(array, 0, createMap$1, setMap$1); },
        entries: function(array) { return entries(apply(array, 0, createMap$1, setMap$1), 0); },
        key: function(d) { keys.push(d); return nest; },
        sortKeys: function(order) { sortKeys[keys.length - 1] = order; return nest; },
        sortValues: function(order) { sortValues = order; return nest; },
        rollup: function(f) { rollup = f; return nest; }
      };
    }

    function createObject$1() {
      return {};
    }

    function setObject$2(object, key, value) {
      object[key] = value;
    }

    function createMap$1() {
      return map$5();
    }

    function setMap$1(map, key, value) {
      map.set(key, value);
    }

    function Set$2() {}

    var proto$2 = map$5.prototype;

    Set$2.prototype = set$6.prototype = {
      constructor: Set$2,
      has: proto$2.has,
      add: function(value) {
        value += "";
        this[prefix$1 + value] = value;
        return this;
      },
      remove: proto$2.remove,
      clear: proto$2.clear,
      values: proto$2.keys,
      size: proto$2.size,
      empty: proto$2.empty,
      each: proto$2.each
    };

    function set$6(object, f) {
      var set = new Set$2;

      // Copy constructor.
      if (object instanceof Set$2) object.each(function(value) { set.add(value); });

      // Otherwise, assume it’s an array.
      else if (object) {
        var i = -1, n = object.length;
        if (f == null) while (++i < n) set.add(object[i]);
        else while (++i < n) set.add(f(object[i], i, object));
      }

      return set;
    }

    function keys$1(map) {
      var keys = [];
      for (var key in map) keys.push(key);
      return keys;
    }

    function values$1(map) {
      var values = [];
      for (var key in map) values.push(map[key]);
      return values;
    }

    function entries$1(map) {
      var entries = [];
      for (var key in map) entries.push({key: key, value: map[key]});
      return entries;
    }

    function define$2(constructor, factory, prototype) {
      constructor.prototype = factory.prototype = prototype;
      prototype.constructor = constructor;
    }

    function extend$2(parent, definition) {
      var prototype = Object.create(parent.prototype);
      for (var key in definition) prototype[key] = definition[key];
      return prototype;
    }

    function Color$2() {}

    var darker$2 = 0.7;
    var brighter$2 = 1 / darker$2;

    var reI$2 = "\\s*([+-]?\\d+)\\s*",
        reN$2 = "\\s*([+-]?\\d*\\.?\\d+(?:[eE][+-]?\\d+)?)\\s*",
        reP$2 = "\\s*([+-]?\\d*\\.?\\d+(?:[eE][+-]?\\d+)?)%\\s*",
        reHex3 = /^#([0-9a-f]{3})$/,
        reHex6 = /^#([0-9a-f]{6})$/,
        reRgbInteger$2 = new RegExp("^rgb\\(" + [reI$2, reI$2, reI$2] + "\\)$"),
        reRgbPercent$2 = new RegExp("^rgb\\(" + [reP$2, reP$2, reP$2] + "\\)$"),
        reRgbaInteger$2 = new RegExp("^rgba\\(" + [reI$2, reI$2, reI$2, reN$2] + "\\)$"),
        reRgbaPercent$2 = new RegExp("^rgba\\(" + [reP$2, reP$2, reP$2, reN$2] + "\\)$"),
        reHslPercent$2 = new RegExp("^hsl\\(" + [reN$2, reP$2, reP$2] + "\\)$"),
        reHslaPercent$2 = new RegExp("^hsla\\(" + [reN$2, reP$2, reP$2, reN$2] + "\\)$");

    var named$2 = {
      aliceblue: 0xf0f8ff,
      antiquewhite: 0xfaebd7,
      aqua: 0x00ffff,
      aquamarine: 0x7fffd4,
      azure: 0xf0ffff,
      beige: 0xf5f5dc,
      bisque: 0xffe4c4,
      black: 0x000000,
      blanchedalmond: 0xffebcd,
      blue: 0x0000ff,
      blueviolet: 0x8a2be2,
      brown: 0xa52a2a,
      burlywood: 0xdeb887,
      cadetblue: 0x5f9ea0,
      chartreuse: 0x7fff00,
      chocolate: 0xd2691e,
      coral: 0xff7f50,
      cornflowerblue: 0x6495ed,
      cornsilk: 0xfff8dc,
      crimson: 0xdc143c,
      cyan: 0x00ffff,
      darkblue: 0x00008b,
      darkcyan: 0x008b8b,
      darkgoldenrod: 0xb8860b,
      darkgray: 0xa9a9a9,
      darkgreen: 0x006400,
      darkgrey: 0xa9a9a9,
      darkkhaki: 0xbdb76b,
      darkmagenta: 0x8b008b,
      darkolivegreen: 0x556b2f,
      darkorange: 0xff8c00,
      darkorchid: 0x9932cc,
      darkred: 0x8b0000,
      darksalmon: 0xe9967a,
      darkseagreen: 0x8fbc8f,
      darkslateblue: 0x483d8b,
      darkslategray: 0x2f4f4f,
      darkslategrey: 0x2f4f4f,
      darkturquoise: 0x00ced1,
      darkviolet: 0x9400d3,
      deeppink: 0xff1493,
      deepskyblue: 0x00bfff,
      dimgray: 0x696969,
      dimgrey: 0x696969,
      dodgerblue: 0x1e90ff,
      firebrick: 0xb22222,
      floralwhite: 0xfffaf0,
      forestgreen: 0x228b22,
      fuchsia: 0xff00ff,
      gainsboro: 0xdcdcdc,
      ghostwhite: 0xf8f8ff,
      gold: 0xffd700,
      goldenrod: 0xdaa520,
      gray: 0x808080,
      green: 0x008000,
      greenyellow: 0xadff2f,
      grey: 0x808080,
      honeydew: 0xf0fff0,
      hotpink: 0xff69b4,
      indianred: 0xcd5c5c,
      indigo: 0x4b0082,
      ivory: 0xfffff0,
      khaki: 0xf0e68c,
      lavender: 0xe6e6fa,
      lavenderblush: 0xfff0f5,
      lawngreen: 0x7cfc00,
      lemonchiffon: 0xfffacd,
      lightblue: 0xadd8e6,
      lightcoral: 0xf08080,
      lightcyan: 0xe0ffff,
      lightgoldenrodyellow: 0xfafad2,
      lightgray: 0xd3d3d3,
      lightgreen: 0x90ee90,
      lightgrey: 0xd3d3d3,
      lightpink: 0xffb6c1,
      lightsalmon: 0xffa07a,
      lightseagreen: 0x20b2aa,
      lightskyblue: 0x87cefa,
      lightslategray: 0x778899,
      lightslategrey: 0x778899,
      lightsteelblue: 0xb0c4de,
      lightyellow: 0xffffe0,
      lime: 0x00ff00,
      limegreen: 0x32cd32,
      linen: 0xfaf0e6,
      magenta: 0xff00ff,
      maroon: 0x800000,
      mediumaquamarine: 0x66cdaa,
      mediumblue: 0x0000cd,
      mediumorchid: 0xba55d3,
      mediumpurple: 0x9370db,
      mediumseagreen: 0x3cb371,
      mediumslateblue: 0x7b68ee,
      mediumspringgreen: 0x00fa9a,
      mediumturquoise: 0x48d1cc,
      mediumvioletred: 0xc71585,
      midnightblue: 0x191970,
      mintcream: 0xf5fffa,
      mistyrose: 0xffe4e1,
      moccasin: 0xffe4b5,
      navajowhite: 0xffdead,
      navy: 0x000080,
      oldlace: 0xfdf5e6,
      olive: 0x808000,
      olivedrab: 0x6b8e23,
      orange: 0xffa500,
      orangered: 0xff4500,
      orchid: 0xda70d6,
      palegoldenrod: 0xeee8aa,
      palegreen: 0x98fb98,
      paleturquoise: 0xafeeee,
      palevioletred: 0xdb7093,
      papayawhip: 0xffefd5,
      peachpuff: 0xffdab9,
      peru: 0xcd853f,
      pink: 0xffc0cb,
      plum: 0xdda0dd,
      powderblue: 0xb0e0e6,
      purple: 0x800080,
      rebeccapurple: 0x663399,
      red: 0xff0000,
      rosybrown: 0xbc8f8f,
      royalblue: 0x4169e1,
      saddlebrown: 0x8b4513,
      salmon: 0xfa8072,
      sandybrown: 0xf4a460,
      seagreen: 0x2e8b57,
      seashell: 0xfff5ee,
      sienna: 0xa0522d,
      silver: 0xc0c0c0,
      skyblue: 0x87ceeb,
      slateblue: 0x6a5acd,
      slategray: 0x708090,
      slategrey: 0x708090,
      snow: 0xfffafa,
      springgreen: 0x00ff7f,
      steelblue: 0x4682b4,
      tan: 0xd2b48c,
      teal: 0x008080,
      thistle: 0xd8bfd8,
      tomato: 0xff6347,
      turquoise: 0x40e0d0,
      violet: 0xee82ee,
      wheat: 0xf5deb3,
      white: 0xffffff,
      whitesmoke: 0xf5f5f5,
      yellow: 0xffff00,
      yellowgreen: 0x9acd32
    };

    define$2(Color$2, color$2, {
      displayable: function() {
        return this.rgb().displayable();
      },
      toString: function() {
        return this.rgb() + "";
      }
    });

    function color$2(format) {
      var m;
      format = (format + "").trim().toLowerCase();
      return (m = reHex3.exec(format)) ? (m = parseInt(m[1], 16), new Rgb$2((m >> 8 & 0xf) | (m >> 4 & 0x0f0), (m >> 4 & 0xf) | (m & 0xf0), ((m & 0xf) << 4) | (m & 0xf), 1)) // #f00
          : (m = reHex6.exec(format)) ? rgbn$2(parseInt(m[1], 16)) // #ff0000
          : (m = reRgbInteger$2.exec(format)) ? new Rgb$2(m[1], m[2], m[3], 1) // rgb(255, 0, 0)
          : (m = reRgbPercent$2.exec(format)) ? new Rgb$2(m[1] * 255 / 100, m[2] * 255 / 100, m[3] * 255 / 100, 1) // rgb(100%, 0%, 0%)
          : (m = reRgbaInteger$2.exec(format)) ? rgba$2(m[1], m[2], m[3], m[4]) // rgba(255, 0, 0, 1)
          : (m = reRgbaPercent$2.exec(format)) ? rgba$2(m[1] * 255 / 100, m[2] * 255 / 100, m[3] * 255 / 100, m[4]) // rgb(100%, 0%, 0%, 1)
          : (m = reHslPercent$2.exec(format)) ? hsla$2(m[1], m[2] / 100, m[3] / 100, 1) // hsl(120, 50%, 50%)
          : (m = reHslaPercent$2.exec(format)) ? hsla$2(m[1], m[2] / 100, m[3] / 100, m[4]) // hsla(120, 50%, 50%, 1)
          : named$2.hasOwnProperty(format) ? rgbn$2(named$2[format])
          : format === "transparent" ? new Rgb$2(NaN, NaN, NaN, 0)
          : null;
    }

    function rgbn$2(n) {
      return new Rgb$2(n >> 16 & 0xff, n >> 8 & 0xff, n & 0xff, 1);
    }

    function rgba$2(r, g, b, a) {
      if (a <= 0) r = g = b = NaN;
      return new Rgb$2(r, g, b, a);
    }

    function rgbConvert$2(o) {
      if (!(o instanceof Color$2)) o = color$2(o);
      if (!o) return new Rgb$2;
      o = o.rgb();
      return new Rgb$2(o.r, o.g, o.b, o.opacity);
    }

    function rgb$2(r, g, b, opacity) {
      return arguments.length === 1 ? rgbConvert$2(r) : new Rgb$2(r, g, b, opacity == null ? 1 : opacity);
    }

    function Rgb$2(r, g, b, opacity) {
      this.r = +r;
      this.g = +g;
      this.b = +b;
      this.opacity = +opacity;
    }

    define$2(Rgb$2, rgb$2, extend$2(Color$2, {
      brighter: function(k) {
        k = k == null ? brighter$2 : Math.pow(brighter$2, k);
        return new Rgb$2(this.r * k, this.g * k, this.b * k, this.opacity);
      },
      darker: function(k) {
        k = k == null ? darker$2 : Math.pow(darker$2, k);
        return new Rgb$2(this.r * k, this.g * k, this.b * k, this.opacity);
      },
      rgb: function() {
        return this;
      },
      displayable: function() {
        return (0 <= this.r && this.r <= 255)
            && (0 <= this.g && this.g <= 255)
            && (0 <= this.b && this.b <= 255)
            && (0 <= this.opacity && this.opacity <= 1);
      },
      toString: function() {
        var a = this.opacity; a = isNaN(a) ? 1 : Math.max(0, Math.min(1, a));
        return (a === 1 ? "rgb(" : "rgba(")
            + Math.max(0, Math.min(255, Math.round(this.r) || 0)) + ", "
            + Math.max(0, Math.min(255, Math.round(this.g) || 0)) + ", "
            + Math.max(0, Math.min(255, Math.round(this.b) || 0))
            + (a === 1 ? ")" : ", " + a + ")");
      }
    }));

    function hsla$2(h, s, l, a) {
      if (a <= 0) h = s = l = NaN;
      else if (l <= 0 || l >= 1) h = s = NaN;
      else if (s <= 0) h = NaN;
      return new Hsl$2(h, s, l, a);
    }

    function hslConvert$2(o) {
      if (o instanceof Hsl$2) return new Hsl$2(o.h, o.s, o.l, o.opacity);
      if (!(o instanceof Color$2)) o = color$2(o);
      if (!o) return new Hsl$2;
      if (o instanceof Hsl$2) return o;
      o = o.rgb();
      var r = o.r / 255,
          g = o.g / 255,
          b = o.b / 255,
          min = Math.min(r, g, b),
          max = Math.max(r, g, b),
          h = NaN,
          s = max - min,
          l = (max + min) / 2;
      if (s) {
        if (r === max) h = (g - b) / s + (g < b) * 6;
        else if (g === max) h = (b - r) / s + 2;
        else h = (r - g) / s + 4;
        s /= l < 0.5 ? max + min : 2 - max - min;
        h *= 60;
      } else {
        s = l > 0 && l < 1 ? 0 : h;
      }
      return new Hsl$2(h, s, l, o.opacity);
    }

    function hsl$4(h, s, l, opacity) {
      return arguments.length === 1 ? hslConvert$2(h) : new Hsl$2(h, s, l, opacity == null ? 1 : opacity);
    }

    function Hsl$2(h, s, l, opacity) {
      this.h = +h;
      this.s = +s;
      this.l = +l;
      this.opacity = +opacity;
    }

    define$2(Hsl$2, hsl$4, extend$2(Color$2, {
      brighter: function(k) {
        k = k == null ? brighter$2 : Math.pow(brighter$2, k);
        return new Hsl$2(this.h, this.s, this.l * k, this.opacity);
      },
      darker: function(k) {
        k = k == null ? darker$2 : Math.pow(darker$2, k);
        return new Hsl$2(this.h, this.s, this.l * k, this.opacity);
      },
      rgb: function() {
        var h = this.h % 360 + (this.h < 0) * 360,
            s = isNaN(h) || isNaN(this.s) ? 0 : this.s,
            l = this.l,
            m2 = l + (l < 0.5 ? l : 1 - l) * s,
            m1 = 2 * l - m2;
        return new Rgb$2(
          hsl2rgb$2(h >= 240 ? h - 240 : h + 120, m1, m2),
          hsl2rgb$2(h, m1, m2),
          hsl2rgb$2(h < 120 ? h + 240 : h - 120, m1, m2),
          this.opacity
        );
      },
      displayable: function() {
        return (0 <= this.s && this.s <= 1 || isNaN(this.s))
            && (0 <= this.l && this.l <= 1)
            && (0 <= this.opacity && this.opacity <= 1);
      }
    }));

    /* From FvD 13.37, CSS Color Module Level 3 */
    function hsl2rgb$2(h, m1, m2) {
      return (h < 60 ? m1 + (m2 - m1) * h / 60
          : h < 180 ? m2
          : h < 240 ? m1 + (m2 - m1) * (240 - h) / 60
          : m1) * 255;
    }

    var deg2rad$1 = Math.PI / 180;
    var rad2deg$1 = 180 / Math.PI;

    var Kn = 18,
        Xn$1 = 0.950470, // D65 standard referent
        Yn$1 = 1,
        Zn$1 = 1.088830,
        t0$2 = 4 / 29,
        t1$2 = 6 / 29,
        t2$1 = 3 * t1$2 * t1$2,
        t3$1 = t1$2 * t1$2 * t1$2;

    function labConvert$1(o) {
      if (o instanceof Lab$1) return new Lab$1(o.l, o.a, o.b, o.opacity);
      if (o instanceof Hcl$1) {
        var h = o.h * deg2rad$1;
        return new Lab$1(o.l, Math.cos(h) * o.c, Math.sin(h) * o.c, o.opacity);
      }
      if (!(o instanceof Rgb$2)) o = rgbConvert$2(o);
      var b = rgb2xyz(o.r),
          a = rgb2xyz(o.g),
          l = rgb2xyz(o.b),
          x = xyz2lab$1((0.4124564 * b + 0.3575761 * a + 0.1804375 * l) / Xn$1),
          y = xyz2lab$1((0.2126729 * b + 0.7151522 * a + 0.0721750 * l) / Yn$1),
          z = xyz2lab$1((0.0193339 * b + 0.1191920 * a + 0.9503041 * l) / Zn$1);
      return new Lab$1(116 * y - 16, 500 * (x - y), 200 * (y - z), o.opacity);
    }

    function lab$2(l, a, b, opacity) {
      return arguments.length === 1 ? labConvert$1(l) : new Lab$1(l, a, b, opacity == null ? 1 : opacity);
    }

    function Lab$1(l, a, b, opacity) {
      this.l = +l;
      this.a = +a;
      this.b = +b;
      this.opacity = +opacity;
    }

    define$2(Lab$1, lab$2, extend$2(Color$2, {
      brighter: function(k) {
        return new Lab$1(this.l + Kn * (k == null ? 1 : k), this.a, this.b, this.opacity);
      },
      darker: function(k) {
        return new Lab$1(this.l - Kn * (k == null ? 1 : k), this.a, this.b, this.opacity);
      },
      rgb: function() {
        var y = (this.l + 16) / 116,
            x = isNaN(this.a) ? y : y + this.a / 500,
            z = isNaN(this.b) ? y : y - this.b / 200;
        y = Yn$1 * lab2xyz$1(y);
        x = Xn$1 * lab2xyz$1(x);
        z = Zn$1 * lab2xyz$1(z);
        return new Rgb$2(
          xyz2rgb( 3.2404542 * x - 1.5371385 * y - 0.4985314 * z), // D65 -> sRGB
          xyz2rgb(-0.9692660 * x + 1.8760108 * y + 0.0415560 * z),
          xyz2rgb( 0.0556434 * x - 0.2040259 * y + 1.0572252 * z),
          this.opacity
        );
      }
    }));

    function xyz2lab$1(t) {
      return t > t3$1 ? Math.pow(t, 1 / 3) : t / t2$1 + t0$2;
    }

    function lab2xyz$1(t) {
      return t > t1$2 ? t * t * t : t2$1 * (t - t0$2);
    }

    function xyz2rgb(x) {
      return 255 * (x <= 0.0031308 ? 12.92 * x : 1.055 * Math.pow(x, 1 / 2.4) - 0.055);
    }

    function rgb2xyz(x) {
      return (x /= 255) <= 0.04045 ? x / 12.92 : Math.pow((x + 0.055) / 1.055, 2.4);
    }

    function hclConvert$1(o) {
      if (o instanceof Hcl$1) return new Hcl$1(o.h, o.c, o.l, o.opacity);
      if (!(o instanceof Lab$1)) o = labConvert$1(o);
      var h = Math.atan2(o.b, o.a) * rad2deg$1;
      return new Hcl$1(h < 0 ? h + 360 : h, Math.sqrt(o.a * o.a + o.b * o.b), o.l, o.opacity);
    }

    function hcl$3(h, c, l, opacity) {
      return arguments.length === 1 ? hclConvert$1(h) : new Hcl$1(h, c, l, opacity == null ? 1 : opacity);
    }

    function Hcl$1(h, c, l, opacity) {
      this.h = +h;
      this.c = +c;
      this.l = +l;
      this.opacity = +opacity;
    }

    define$2(Hcl$1, hcl$3, extend$2(Color$2, {
      brighter: function(k) {
        return new Hcl$1(this.h, this.c, this.l + Kn * (k == null ? 1 : k), this.opacity);
      },
      darker: function(k) {
        return new Hcl$1(this.h, this.c, this.l - Kn * (k == null ? 1 : k), this.opacity);
      },
      rgb: function() {
        return labConvert$1(this).rgb();
      }
    }));

    var A$1 = -0.14861,
        B$1 = +1.78277,
        C$1 = -0.29227,
        D$1 = -0.90649,
        E$1 = +1.97294,
        ED$1 = E$1 * D$1,
        EB$1 = E$1 * B$1,
        BC_DA$1 = B$1 * C$1 - D$1 * A$1;

    function cubehelixConvert$1(o) {
      if (o instanceof Cubehelix$1) return new Cubehelix$1(o.h, o.s, o.l, o.opacity);
      if (!(o instanceof Rgb$2)) o = rgbConvert$2(o);
      var r = o.r / 255,
          g = o.g / 255,
          b = o.b / 255,
          l = (BC_DA$1 * b + ED$1 * r - EB$1 * g) / (BC_DA$1 + ED$1 - EB$1),
          bl = b - l,
          k = (E$1 * (g - l) - C$1 * bl) / D$1,
          s = Math.sqrt(k * k + bl * bl) / (E$1 * l * (1 - l)), // NaN if l=0 or l=1
          h = s ? Math.atan2(k, bl) * rad2deg$1 - 120 : NaN;
      return new Cubehelix$1(h < 0 ? h + 360 : h, s, l, o.opacity);
    }

    function cubehelix$4(h, s, l, opacity) {
      return arguments.length === 1 ? cubehelixConvert$1(h) : new Cubehelix$1(h, s, l, opacity == null ? 1 : opacity);
    }

    function Cubehelix$1(h, s, l, opacity) {
      this.h = +h;
      this.s = +s;
      this.l = +l;
      this.opacity = +opacity;
    }

    define$2(Cubehelix$1, cubehelix$4, extend$2(Color$2, {
      brighter: function(k) {
        k = k == null ? brighter$2 : Math.pow(brighter$2, k);
        return new Cubehelix$1(this.h, this.s, this.l * k, this.opacity);
      },
      darker: function(k) {
        k = k == null ? darker$2 : Math.pow(darker$2, k);
        return new Cubehelix$1(this.h, this.s, this.l * k, this.opacity);
      },
      rgb: function() {
        var h = isNaN(this.h) ? 0 : (this.h + 120) * deg2rad$1,
            l = +this.l,
            a = isNaN(this.s) ? 0 : this.s * l * (1 - l),
            cosh = Math.cos(h),
            sinh = Math.sin(h);
        return new Rgb$2(
          255 * (l + a * (A$1 * cosh + B$1 * sinh)),
          255 * (l + a * (C$1 * cosh + D$1 * sinh)),
          255 * (l + a * (E$1 * cosh)),
          this.opacity
        );
      }
    }));

    var noop$5 = {value: function() {}};

    function dispatch$2() {
      for (var i = 0, n = arguments.length, _ = {}, t; i < n; ++i) {
        if (!(t = arguments[i] + "") || (t in _)) throw new Error("illegal type: " + t);
        _[t] = [];
      }
      return new Dispatch$2(_);
    }

    function Dispatch$2(_) {
      this._ = _;
    }

    function parseTypenames$4(typenames, types) {
      return typenames.trim().split(/^|\s+/).map(function(t) {
        var name = "", i = t.indexOf(".");
        if (i >= 0) name = t.slice(i + 1), t = t.slice(0, i);
        if (t && !types.hasOwnProperty(t)) throw new Error("unknown type: " + t);
        return {type: t, name: name};
      });
    }

    Dispatch$2.prototype = dispatch$2.prototype = {
      constructor: Dispatch$2,
      on: function(typename, callback) {
        var _ = this._,
            T = parseTypenames$4(typename + "", _),
            t,
            i = -1,
            n = T.length;

        // If no callback was specified, return the callback of the given type and name.
        if (arguments.length < 2) {
          while (++i < n) if ((t = (typename = T[i]).type) && (t = get$5(_[t], typename.name))) return t;
          return;
        }

        // If a type was specified, set the callback for the given type and name.
        // Otherwise, if a null callback was specified, remove callbacks of the given name.
        if (callback != null && typeof callback !== "function") throw new Error("invalid callback: " + callback);
        while (++i < n) {
          if (t = (typename = T[i]).type) _[t] = set$7(_[t], typename.name, callback);
          else if (callback == null) for (t in _) _[t] = set$7(_[t], typename.name, null);
        }

        return this;
      },
      copy: function() {
        var copy = {}, _ = this._;
        for (var t in _) copy[t] = _[t].slice();
        return new Dispatch$2(copy);
      },
      call: function(type, that) {
        if ((n = arguments.length - 2) > 0) for (var args = new Array(n), i = 0, n, t; i < n; ++i) args[i] = arguments[i + 2];
        if (!this._.hasOwnProperty(type)) throw new Error("unknown type: " + type);
        for (t = this._[type], i = 0, n = t.length; i < n; ++i) t[i].value.apply(that, args);
      },
      apply: function(type, that, args) {
        if (!this._.hasOwnProperty(type)) throw new Error("unknown type: " + type);
        for (var t = this._[type], i = 0, n = t.length; i < n; ++i) t[i].value.apply(that, args);
      }
    };

    function get$5(type, name) {
      for (var i = 0, n = type.length, c; i < n; ++i) {
        if ((c = type[i]).name === name) {
          return c.value;
        }
      }
    }

    function set$7(type, name, callback) {
      for (var i = 0, n = type.length; i < n; ++i) {
        if (type[i].name === name) {
          type[i] = noop$5, type = type.slice(0, i).concat(type.slice(i + 1));
          break;
        }
      }
      if (callback != null) type.push({name: name, value: callback});
      return type;
    }

    var noop$6 = {value: function() {}};

    function dispatch$3() {
      for (var i = 0, n = arguments.length, _ = {}, t; i < n; ++i) {
        if (!(t = arguments[i] + "") || (t in _) || /[\s.]/.test(t)) throw new Error("illegal type: " + t);
        _[t] = [];
      }
      return new Dispatch$3(_);
    }

    function Dispatch$3(_) {
      this._ = _;
    }

    function parseTypenames$5(typenames, types) {
      return typenames.trim().split(/^|\s+/).map(function(t) {
        var name = "", i = t.indexOf(".");
        if (i >= 0) name = t.slice(i + 1), t = t.slice(0, i);
        if (t && !types.hasOwnProperty(t)) throw new Error("unknown type: " + t);
        return {type: t, name: name};
      });
    }

    Dispatch$3.prototype = dispatch$3.prototype = {
      constructor: Dispatch$3,
      on: function(typename, callback) {
        var _ = this._,
            T = parseTypenames$5(typename + "", _),
            t,
            i = -1,
            n = T.length;

        // If no callback was specified, return the callback of the given type and name.
        if (arguments.length < 2) {
          while (++i < n) if ((t = (typename = T[i]).type) && (t = get$6(_[t], typename.name))) return t;
          return;
        }

        // If a type was specified, set the callback for the given type and name.
        // Otherwise, if a null callback was specified, remove callbacks of the given name.
        if (callback != null && typeof callback !== "function") throw new Error("invalid callback: " + callback);
        while (++i < n) {
          if (t = (typename = T[i]).type) _[t] = set$8(_[t], typename.name, callback);
          else if (callback == null) for (t in _) _[t] = set$8(_[t], typename.name, null);
        }

        return this;
      },
      copy: function() {
        var copy = {}, _ = this._;
        for (var t in _) copy[t] = _[t].slice();
        return new Dispatch$3(copy);
      },
      call: function(type, that) {
        if ((n = arguments.length - 2) > 0) for (var args = new Array(n), i = 0, n, t; i < n; ++i) args[i] = arguments[i + 2];
        if (!this._.hasOwnProperty(type)) throw new Error("unknown type: " + type);
        for (t = this._[type], i = 0, n = t.length; i < n; ++i) t[i].value.apply(that, args);
      },
      apply: function(type, that, args) {
        if (!this._.hasOwnProperty(type)) throw new Error("unknown type: " + type);
        for (var t = this._[type], i = 0, n = t.length; i < n; ++i) t[i].value.apply(that, args);
      }
    };

    function get$6(type, name) {
      for (var i = 0, n = type.length, c; i < n; ++i) {
        if ((c = type[i]).name === name) {
          return c.value;
        }
      }
    }

    function set$8(type, name, callback) {
      for (var i = 0, n = type.length; i < n; ++i) {
        if (type[i].name === name) {
          type[i] = noop$6, type = type.slice(0, i).concat(type.slice(i + 1));
          break;
        }
      }
      if (callback != null) type.push({name: name, value: callback});
      return type;
    }

    var xhtml$2 = "http://www.w3.org/1999/xhtml";

    var namespaces$2 = {
      svg: "http://www.w3.org/2000/svg",
      xhtml: xhtml$2,
      xlink: "http://www.w3.org/1999/xlink",
      xml: "http://www.w3.org/XML/1998/namespace",
      xmlns: "http://www.w3.org/2000/xmlns/"
    };

    function namespace$2(name) {
      var prefix = name += "", i = prefix.indexOf(":");
      if (i >= 0 && (prefix = name.slice(0, i)) !== "xmlns") name = name.slice(i + 1);
      return namespaces$2.hasOwnProperty(prefix) ? {space: namespaces$2[prefix], local: name} : name;
    }

    function creatorInherit$2(name) {
      return function() {
        var document = this.ownerDocument,
            uri = this.namespaceURI;
        return uri === xhtml$2 && document.documentElement.namespaceURI === xhtml$2
            ? document.createElement(name)
            : document.createElementNS(uri, name);
      };
    }

    function creatorFixed$2(fullname) {
      return function() {
        return this.ownerDocument.createElementNS(fullname.space, fullname.local);
      };
    }

    function creator$2(name) {
      var fullname = namespace$2(name);
      return (fullname.local
          ? creatorFixed$2
          : creatorInherit$2)(fullname);
    }

    function none$4() {}

    function selector$2(selector) {
      return selector == null ? none$4 : function() {
        return this.querySelector(selector);
      };
    }

    function selection_select$2(select) {
      if (typeof select !== "function") select = selector$2(select);

      for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, subgroup = subgroups[j] = new Array(n), node, subnode, i = 0; i < n; ++i) {
          if ((node = group[i]) && (subnode = select.call(node, node.__data__, i, group))) {
            if ("__data__" in node) subnode.__data__ = node.__data__;
            subgroup[i] = subnode;
          }
        }
      }

      return new Selection$4(subgroups, this._parents);
    }

    function empty$4() {
      return [];
    }

    function selectorAll$2(selector) {
      return selector == null ? empty$4 : function() {
        return this.querySelectorAll(selector);
      };
    }

    function selection_selectAll$2(select) {
      if (typeof select !== "function") select = selectorAll$2(select);

      for (var groups = this._groups, m = groups.length, subgroups = [], parents = [], j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, node, i = 0; i < n; ++i) {
          if (node = group[i]) {
            subgroups.push(select.call(node, node.__data__, i, group));
            parents.push(node);
          }
        }
      }

      return new Selection$4(subgroups, parents);
    }

    function matcher$2(selector) {
      return function() {
        return this.matches(selector);
      };
    }

    function selection_filter$2(match) {
      if (typeof match !== "function") match = matcher$2(match);

      for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, subgroup = subgroups[j] = [], node, i = 0; i < n; ++i) {
          if ((node = group[i]) && match.call(node, node.__data__, i, group)) {
            subgroup.push(node);
          }
        }
      }

      return new Selection$4(subgroups, this._parents);
    }

    function sparse$2(update) {
      return new Array(update.length);
    }

    function selection_enter$2() {
      return new Selection$4(this._enter || this._groups.map(sparse$2), this._parents);
    }

    function EnterNode$2(parent, datum) {
      this.ownerDocument = parent.ownerDocument;
      this.namespaceURI = parent.namespaceURI;
      this._next = null;
      this._parent = parent;
      this.__data__ = datum;
    }

    EnterNode$2.prototype = {
      constructor: EnterNode$2,
      appendChild: function(child) { return this._parent.insertBefore(child, this._next); },
      insertBefore: function(child, next) { return this._parent.insertBefore(child, next); },
      querySelector: function(selector) { return this._parent.querySelector(selector); },
      querySelectorAll: function(selector) { return this._parent.querySelectorAll(selector); }
    };

    function constant$j(x) {
      return function() {
        return x;
      };
    }

    var keyPrefix$3 = "$"; // Protect against keys like “__proto__”.

    function bindIndex$2(parent, group, enter, update, exit, data) {
      var i = 0,
          node,
          groupLength = group.length,
          dataLength = data.length;

      // Put any non-null nodes that fit into update.
      // Put any null nodes into enter.
      // Put any remaining data into enter.
      for (; i < dataLength; ++i) {
        if (node = group[i]) {
          node.__data__ = data[i];
          update[i] = node;
        } else {
          enter[i] = new EnterNode$2(parent, data[i]);
        }
      }

      // Put any non-null nodes that don’t fit into exit.
      for (; i < groupLength; ++i) {
        if (node = group[i]) {
          exit[i] = node;
        }
      }
    }

    function bindKey$2(parent, group, enter, update, exit, data, key) {
      var i,
          node,
          nodeByKeyValue = {},
          groupLength = group.length,
          dataLength = data.length,
          keyValues = new Array(groupLength),
          keyValue;

      // Compute the key for each node.
      // If multiple nodes have the same key, the duplicates are added to exit.
      for (i = 0; i < groupLength; ++i) {
        if (node = group[i]) {
          keyValues[i] = keyValue = keyPrefix$3 + key.call(node, node.__data__, i, group);
          if (keyValue in nodeByKeyValue) {
            exit[i] = node;
          } else {
            nodeByKeyValue[keyValue] = node;
          }
        }
      }

      // Compute the key for each datum.
      // If there a node associated with this key, join and add it to update.
      // If there is not (or the key is a duplicate), add it to enter.
      for (i = 0; i < dataLength; ++i) {
        keyValue = keyPrefix$3 + key.call(parent, data[i], i, data);
        if (node = nodeByKeyValue[keyValue]) {
          update[i] = node;
          node.__data__ = data[i];
          nodeByKeyValue[keyValue] = null;
        } else {
          enter[i] = new EnterNode$2(parent, data[i]);
        }
      }

      // Add any remaining nodes that were not bound to data to exit.
      for (i = 0; i < groupLength; ++i) {
        if ((node = group[i]) && (nodeByKeyValue[keyValues[i]] === node)) {
          exit[i] = node;
        }
      }
    }

    function selection_data$2(value, key) {
      if (!value) {
        data = new Array(this.size()), j = -1;
        this.each(function(d) { data[++j] = d; });
        return data;
      }

      var bind = key ? bindKey$2 : bindIndex$2,
          parents = this._parents,
          groups = this._groups;

      if (typeof value !== "function") value = constant$j(value);

      for (var m = groups.length, update = new Array(m), enter = new Array(m), exit = new Array(m), j = 0; j < m; ++j) {
        var parent = parents[j],
            group = groups[j],
            groupLength = group.length,
            data = value.call(parent, parent && parent.__data__, j, parents),
            dataLength = data.length,
            enterGroup = enter[j] = new Array(dataLength),
            updateGroup = update[j] = new Array(dataLength),
            exitGroup = exit[j] = new Array(groupLength);

        bind(parent, group, enterGroup, updateGroup, exitGroup, data, key);

        // Now connect the enter nodes to their following update node, such that
        // appendChild can insert the materialized enter node before this node,
        // rather than at the end of the parent node.
        for (var i0 = 0, i1 = 0, previous, next; i0 < dataLength; ++i0) {
          if (previous = enterGroup[i0]) {
            if (i0 >= i1) i1 = i0 + 1;
            while (!(next = updateGroup[i1]) && ++i1 < dataLength);
            previous._next = next || null;
          }
        }
      }

      update = new Selection$4(update, parents);
      update._enter = enter;
      update._exit = exit;
      return update;
    }

    function selection_exit$2() {
      return new Selection$4(this._exit || this._groups.map(sparse$2), this._parents);
    }

    function selection_join$2(onenter, onupdate, onexit) {
      var enter = this.enter(), update = this, exit = this.exit();
      enter = typeof onenter === "function" ? onenter(enter) : enter.append(onenter + "");
      if (onupdate != null) update = onupdate(update);
      if (onexit == null) exit.remove(); else onexit(exit);
      return enter && update ? enter.merge(update).order() : update;
    }

    function selection_merge$2(selection) {

      for (var groups0 = this._groups, groups1 = selection._groups, m0 = groups0.length, m1 = groups1.length, m = Math.min(m0, m1), merges = new Array(m0), j = 0; j < m; ++j) {
        for (var group0 = groups0[j], group1 = groups1[j], n = group0.length, merge = merges[j] = new Array(n), node, i = 0; i < n; ++i) {
          if (node = group0[i] || group1[i]) {
            merge[i] = node;
          }
        }
      }

      for (; j < m0; ++j) {
        merges[j] = groups0[j];
      }

      return new Selection$4(merges, this._parents);
    }

    function selection_order$2() {

      for (var groups = this._groups, j = -1, m = groups.length; ++j < m;) {
        for (var group = groups[j], i = group.length - 1, next = group[i], node; --i >= 0;) {
          if (node = group[i]) {
            if (next && node.compareDocumentPosition(next) ^ 4) next.parentNode.insertBefore(node, next);
            next = node;
          }
        }
      }

      return this;
    }

    function selection_sort$2(compare) {
      if (!compare) compare = ascending$7;

      function compareNode(a, b) {
        return a && b ? compare(a.__data__, b.__data__) : !a - !b;
      }

      for (var groups = this._groups, m = groups.length, sortgroups = new Array(m), j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, sortgroup = sortgroups[j] = new Array(n), node, i = 0; i < n; ++i) {
          if (node = group[i]) {
            sortgroup[i] = node;
          }
        }
        sortgroup.sort(compareNode);
      }

      return new Selection$4(sortgroups, this._parents).order();
    }

    function ascending$7(a, b) {
      return a < b ? -1 : a > b ? 1 : a >= b ? 0 : NaN;
    }

    function selection_call$2() {
      var callback = arguments[0];
      arguments[0] = this;
      callback.apply(null, arguments);
      return this;
    }

    function selection_nodes$2() {
      var nodes = new Array(this.size()), i = -1;
      this.each(function() { nodes[++i] = this; });
      return nodes;
    }

    function selection_node$2() {

      for (var groups = this._groups, j = 0, m = groups.length; j < m; ++j) {
        for (var group = groups[j], i = 0, n = group.length; i < n; ++i) {
          var node = group[i];
          if (node) return node;
        }
      }

      return null;
    }

    function selection_size$2() {
      var size = 0;
      this.each(function() { ++size; });
      return size;
    }

    function selection_empty$2() {
      return !this.node();
    }

    function selection_each$2(callback) {

      for (var groups = this._groups, j = 0, m = groups.length; j < m; ++j) {
        for (var group = groups[j], i = 0, n = group.length, node; i < n; ++i) {
          if (node = group[i]) callback.call(node, node.__data__, i, group);
        }
      }

      return this;
    }

    function attrRemove$4(name) {
      return function() {
        this.removeAttribute(name);
      };
    }

    function attrRemoveNS$4(fullname) {
      return function() {
        this.removeAttributeNS(fullname.space, fullname.local);
      };
    }

    function attrConstant$4(name, value) {
      return function() {
        this.setAttribute(name, value);
      };
    }

    function attrConstantNS$4(fullname, value) {
      return function() {
        this.setAttributeNS(fullname.space, fullname.local, value);
      };
    }

    function attrFunction$4(name, value) {
      return function() {
        var v = value.apply(this, arguments);
        if (v == null) this.removeAttribute(name);
        else this.setAttribute(name, v);
      };
    }

    function attrFunctionNS$4(fullname, value) {
      return function() {
        var v = value.apply(this, arguments);
        if (v == null) this.removeAttributeNS(fullname.space, fullname.local);
        else this.setAttributeNS(fullname.space, fullname.local, v);
      };
    }

    function selection_attr$2(name, value) {
      var fullname = namespace$2(name);

      if (arguments.length < 2) {
        var node = this.node();
        return fullname.local
            ? node.getAttributeNS(fullname.space, fullname.local)
            : node.getAttribute(fullname);
      }

      return this.each((value == null
          ? (fullname.local ? attrRemoveNS$4 : attrRemove$4) : (typeof value === "function"
          ? (fullname.local ? attrFunctionNS$4 : attrFunction$4)
          : (fullname.local ? attrConstantNS$4 : attrConstant$4)))(fullname, value));
    }

    function defaultView$2(node) {
      return (node.ownerDocument && node.ownerDocument.defaultView) // node is a Node
          || (node.document && node) // node is a Window
          || node.defaultView; // node is a Document
    }

    function styleRemove$4(name) {
      return function() {
        this.style.removeProperty(name);
      };
    }

    function styleConstant$4(name, value, priority) {
      return function() {
        this.style.setProperty(name, value, priority);
      };
    }

    function styleFunction$4(name, value, priority) {
      return function() {
        var v = value.apply(this, arguments);
        if (v == null) this.style.removeProperty(name);
        else this.style.setProperty(name, v, priority);
      };
    }

    function selection_style$2(name, value, priority) {
      return arguments.length > 1
          ? this.each((value == null
                ? styleRemove$4 : typeof value === "function"
                ? styleFunction$4
                : styleConstant$4)(name, value, priority == null ? "" : priority))
          : styleValue$2(this.node(), name);
    }

    function styleValue$2(node, name) {
      return node.style.getPropertyValue(name)
          || defaultView$2(node).getComputedStyle(node, null).getPropertyValue(name);
    }

    function propertyRemove$2(name) {
      return function() {
        delete this[name];
      };
    }

    function propertyConstant$2(name, value) {
      return function() {
        this[name] = value;
      };
    }

    function propertyFunction$2(name, value) {
      return function() {
        var v = value.apply(this, arguments);
        if (v == null) delete this[name];
        else this[name] = v;
      };
    }

    function selection_property$2(name, value) {
      return arguments.length > 1
          ? this.each((value == null
              ? propertyRemove$2 : typeof value === "function"
              ? propertyFunction$2
              : propertyConstant$2)(name, value))
          : this.node()[name];
    }

    function classArray$2(string) {
      return string.trim().split(/^|\s+/);
    }

    function classList$2(node) {
      return node.classList || new ClassList$2(node);
    }

    function ClassList$2(node) {
      this._node = node;
      this._names = classArray$2(node.getAttribute("class") || "");
    }

    ClassList$2.prototype = {
      add: function(name) {
        var i = this._names.indexOf(name);
        if (i < 0) {
          this._names.push(name);
          this._node.setAttribute("class", this._names.join(" "));
        }
      },
      remove: function(name) {
        var i = this._names.indexOf(name);
        if (i >= 0) {
          this._names.splice(i, 1);
          this._node.setAttribute("class", this._names.join(" "));
        }
      },
      contains: function(name) {
        return this._names.indexOf(name) >= 0;
      }
    };

    function classedAdd$2(node, names) {
      var list = classList$2(node), i = -1, n = names.length;
      while (++i < n) list.add(names[i]);
    }

    function classedRemove$2(node, names) {
      var list = classList$2(node), i = -1, n = names.length;
      while (++i < n) list.remove(names[i]);
    }

    function classedTrue$2(names) {
      return function() {
        classedAdd$2(this, names);
      };
    }

    function classedFalse$2(names) {
      return function() {
        classedRemove$2(this, names);
      };
    }

    function classedFunction$2(names, value) {
      return function() {
        (value.apply(this, arguments) ? classedAdd$2 : classedRemove$2)(this, names);
      };
    }

    function selection_classed$2(name, value) {
      var names = classArray$2(name + "");

      if (arguments.length < 2) {
        var list = classList$2(this.node()), i = -1, n = names.length;
        while (++i < n) if (!list.contains(names[i])) return false;
        return true;
      }

      return this.each((typeof value === "function"
          ? classedFunction$2 : value
          ? classedTrue$2
          : classedFalse$2)(names, value));
    }

    function textRemove$2() {
      this.textContent = "";
    }

    function textConstant$4(value) {
      return function() {
        this.textContent = value;
      };
    }

    function textFunction$4(value) {
      return function() {
        var v = value.apply(this, arguments);
        this.textContent = v == null ? "" : v;
      };
    }

    function selection_text$2(value) {
      return arguments.length
          ? this.each(value == null
              ? textRemove$2 : (typeof value === "function"
              ? textFunction$4
              : textConstant$4)(value))
          : this.node().textContent;
    }

    function htmlRemove$2() {
      this.innerHTML = "";
    }

    function htmlConstant$2(value) {
      return function() {
        this.innerHTML = value;
      };
    }

    function htmlFunction$2(value) {
      return function() {
        var v = value.apply(this, arguments);
        this.innerHTML = v == null ? "" : v;
      };
    }

    function selection_html$2(value) {
      return arguments.length
          ? this.each(value == null
              ? htmlRemove$2 : (typeof value === "function"
              ? htmlFunction$2
              : htmlConstant$2)(value))
          : this.node().innerHTML;
    }

    function raise$2() {
      if (this.nextSibling) this.parentNode.appendChild(this);
    }

    function selection_raise$2() {
      return this.each(raise$2);
    }

    function lower$2() {
      if (this.previousSibling) this.parentNode.insertBefore(this, this.parentNode.firstChild);
    }

    function selection_lower$2() {
      return this.each(lower$2);
    }

    function selection_append$2(name) {
      var create = typeof name === "function" ? name : creator$2(name);
      return this.select(function() {
        return this.appendChild(create.apply(this, arguments));
      });
    }

    function constantNull$2() {
      return null;
    }

    function selection_insert$2(name, before) {
      var create = typeof name === "function" ? name : creator$2(name),
          select = before == null ? constantNull$2 : typeof before === "function" ? before : selector$2(before);
      return this.select(function() {
        return this.insertBefore(create.apply(this, arguments), select.apply(this, arguments) || null);
      });
    }

    function remove$2() {
      var parent = this.parentNode;
      if (parent) parent.removeChild(this);
    }

    function selection_remove$2() {
      return this.each(remove$2);
    }

    function selection_cloneShallow$2() {
      var clone = this.cloneNode(false), parent = this.parentNode;
      return parent ? parent.insertBefore(clone, this.nextSibling) : clone;
    }

    function selection_cloneDeep$2() {
      var clone = this.cloneNode(true), parent = this.parentNode;
      return parent ? parent.insertBefore(clone, this.nextSibling) : clone;
    }

    function selection_clone$2(deep) {
      return this.select(deep ? selection_cloneDeep$2 : selection_cloneShallow$2);
    }

    function selection_datum$2(value) {
      return arguments.length
          ? this.property("__data__", value)
          : this.node().__data__;
    }

    var filterEvents$2 = {};

    var event$2 = null;

    if (typeof document !== "undefined") {
      var element$2 = document.documentElement;
      if (!("onmouseenter" in element$2)) {
        filterEvents$2 = {mouseenter: "mouseover", mouseleave: "mouseout"};
      }
    }

    function filterContextListener$2(listener, index, group) {
      listener = contextListener$2(listener, index, group);
      return function(event) {
        var related = event.relatedTarget;
        if (!related || (related !== this && !(related.compareDocumentPosition(this) & 8))) {
          listener.call(this, event);
        }
      };
    }

    function contextListener$2(listener, index, group) {
      return function(event1) {
        var event0 = event$2; // Events can be reentrant (e.g., focus).
        event$2 = event1;
        try {
          listener.call(this, this.__data__, index, group);
        } finally {
          event$2 = event0;
        }
      };
    }

    function parseTypenames$6(typenames) {
      return typenames.trim().split(/^|\s+/).map(function(t) {
        var name = "", i = t.indexOf(".");
        if (i >= 0) name = t.slice(i + 1), t = t.slice(0, i);
        return {type: t, name: name};
      });
    }

    function onRemove$2(typename) {
      return function() {
        var on = this.__on;
        if (!on) return;
        for (var j = 0, i = -1, m = on.length, o; j < m; ++j) {
          if (o = on[j], (!typename.type || o.type === typename.type) && o.name === typename.name) {
            this.removeEventListener(o.type, o.listener, o.capture);
          } else {
            on[++i] = o;
          }
        }
        if (++i) on.length = i;
        else delete this.__on;
      };
    }

    function onAdd$2(typename, value, capture) {
      var wrap = filterEvents$2.hasOwnProperty(typename.type) ? filterContextListener$2 : contextListener$2;
      return function(d, i, group) {
        var on = this.__on, o, listener = wrap(value, i, group);
        if (on) for (var j = 0, m = on.length; j < m; ++j) {
          if ((o = on[j]).type === typename.type && o.name === typename.name) {
            this.removeEventListener(o.type, o.listener, o.capture);
            this.addEventListener(o.type, o.listener = listener, o.capture = capture);
            o.value = value;
            return;
          }
        }
        this.addEventListener(typename.type, listener, capture);
        o = {type: typename.type, name: typename.name, value: value, listener: listener, capture: capture};
        if (!on) this.__on = [o];
        else on.push(o);
      };
    }

    function selection_on$2(typename, value, capture) {
      var typenames = parseTypenames$6(typename + ""), i, n = typenames.length, t;

      if (arguments.length < 2) {
        var on = this.node().__on;
        if (on) for (var j = 0, m = on.length, o; j < m; ++j) {
          for (i = 0, o = on[j]; i < n; ++i) {
            if ((t = typenames[i]).type === o.type && t.name === o.name) {
              return o.value;
            }
          }
        }
        return;
      }

      on = value ? onAdd$2 : onRemove$2;
      if (capture == null) capture = false;
      for (i = 0; i < n; ++i) this.each(on(typenames[i], value, capture));
      return this;
    }

    function customEvent$2(event1, listener, that, args) {
      var event0 = event$2;
      event1.sourceEvent = event$2;
      event$2 = event1;
      try {
        return listener.apply(that, args);
      } finally {
        event$2 = event0;
      }
    }

    function dispatchEvent$2(node, type, params) {
      var window = defaultView$2(node),
          event = window.CustomEvent;

      if (typeof event === "function") {
        event = new event(type, params);
      } else {
        event = window.document.createEvent("Event");
        if (params) event.initEvent(type, params.bubbles, params.cancelable), event.detail = params.detail;
        else event.initEvent(type, false, false);
      }

      node.dispatchEvent(event);
    }

    function dispatchConstant$2(type, params) {
      return function() {
        return dispatchEvent$2(this, type, params);
      };
    }

    function dispatchFunction$2(type, params) {
      return function() {
        return dispatchEvent$2(this, type, params.apply(this, arguments));
      };
    }

    function selection_dispatch$2(type, params) {
      return this.each((typeof params === "function"
          ? dispatchFunction$2
          : dispatchConstant$2)(type, params));
    }

    var root$4 = [null];

    function Selection$4(groups, parents) {
      this._groups = groups;
      this._parents = parents;
    }

    function selection$2() {
      return new Selection$4([[document.documentElement]], root$4);
    }

    Selection$4.prototype = selection$2.prototype = {
      constructor: Selection$4,
      select: selection_select$2,
      selectAll: selection_selectAll$2,
      filter: selection_filter$2,
      data: selection_data$2,
      enter: selection_enter$2,
      exit: selection_exit$2,
      join: selection_join$2,
      merge: selection_merge$2,
      order: selection_order$2,
      sort: selection_sort$2,
      call: selection_call$2,
      nodes: selection_nodes$2,
      node: selection_node$2,
      size: selection_size$2,
      empty: selection_empty$2,
      each: selection_each$2,
      attr: selection_attr$2,
      style: selection_style$2,
      property: selection_property$2,
      classed: selection_classed$2,
      text: selection_text$2,
      html: selection_html$2,
      raise: selection_raise$2,
      lower: selection_lower$2,
      append: selection_append$2,
      insert: selection_insert$2,
      remove: selection_remove$2,
      clone: selection_clone$2,
      datum: selection_datum$2,
      on: selection_on$2,
      dispatch: selection_dispatch$2
    };

    function select$2(selector) {
      return typeof selector === "string"
          ? new Selection$4([[document.querySelector(selector)]], [document.documentElement])
          : new Selection$4([[selector]], root$4);
    }

    function sourceEvent$2() {
      var current = event$2, source;
      while (source = current.sourceEvent) current = source;
      return current;
    }

    function point$7(node, event) {
      var svg = node.ownerSVGElement || node;

      if (svg.createSVGPoint) {
        var point = svg.createSVGPoint();
        point.x = event.clientX, point.y = event.clientY;
        point = point.matrixTransform(node.getScreenCTM().inverse());
        return [point.x, point.y];
      }

      var rect = node.getBoundingClientRect();
      return [event.clientX - rect.left - node.clientLeft, event.clientY - rect.top - node.clientTop];
    }

    function mouse$2(node) {
      var event = sourceEvent$2();
      if (event.changedTouches) event = event.changedTouches[0];
      return point$7(node, event);
    }

    function touch$1(node, touches, identifier) {
      if (arguments.length < 3) identifier = touches, touches = sourceEvent$2().changedTouches;

      for (var i = 0, n = touches ? touches.length : 0, touch; i < n; ++i) {
        if ((touch = touches[i]).identifier === identifier) {
          return point$7(node, touch);
        }
      }

      return null;
    }

    function nopropagation$4() {
      event$2.stopImmediatePropagation();
    }

    function noevent$5() {
      event$2.preventDefault();
      event$2.stopImmediatePropagation();
    }

    function nodrag(view) {
      var root = view.document.documentElement,
          selection = select$2(view).on("dragstart.drag", noevent$5, true);
      if ("onselectstart" in root) {
        selection.on("selectstart.drag", noevent$5, true);
      } else {
        root.__noselect = root.style.MozUserSelect;
        root.style.MozUserSelect = "none";
      }
    }

    function yesdrag$2(view, noclick) {
      var root = view.document.documentElement,
          selection = select$2(view).on("dragstart.drag", null);
      if (noclick) {
        selection.on("click.drag", noevent$5, true);
        setTimeout(function() { selection.on("click.drag", null); }, 0);
      }
      if ("onselectstart" in root) {
        selection.on("selectstart.drag", null);
      } else {
        root.style.MozUserSelect = root.__noselect;
        delete root.__noselect;
      }
    }

    function constant$k(x) {
      return function() {
        return x;
      };
    }

    function DragEvent$1(target, type, subject, id, active, x, y, dx, dy, dispatch) {
      this.target = target;
      this.type = type;
      this.subject = subject;
      this.identifier = id;
      this.active = active;
      this.x = x;
      this.y = y;
      this.dx = dx;
      this.dy = dy;
      this._ = dispatch;
    }

    DragEvent$1.prototype.on = function() {
      var value = this._.on.apply(this._, arguments);
      return value === this._ ? this : value;
    };

    // Ignore right-click, since that should open the context menu.
    function defaultFilter$4() {
      return !event$2.button;
    }

    function defaultContainer$1() {
      return this.parentNode;
    }

    function defaultSubject$1(d) {
      return d == null ? {x: event$2.x, y: event$2.y} : d;
    }

    function defaultTouchable$3() {
      return "ontouchstart" in this;
    }

    function drag$1() {
      var filter = defaultFilter$4,
          container = defaultContainer$1,
          subject = defaultSubject$1,
          touchable = defaultTouchable$3,
          gestures = {},
          listeners = dispatch$3("start", "drag", "end"),
          active = 0,
          mousedownx,
          mousedowny,
          mousemoving,
          touchending,
          clickDistance2 = 0;

      function drag(selection) {
        selection
            .on("mousedown.drag", mousedowned)
          .filter(touchable)
            .on("touchstart.drag", touchstarted)
            .on("touchmove.drag", touchmoved)
            .on("touchend.drag touchcancel.drag", touchended)
            .style("touch-action", "none")
            .style("-webkit-tap-highlight-color", "rgba(0,0,0,0)");
      }

      function mousedowned() {
        if (touchending || !filter.apply(this, arguments)) return;
        var gesture = beforestart("mouse", container.apply(this, arguments), mouse$2, this, arguments);
        if (!gesture) return;
        select$2(event$2.view).on("mousemove.drag", mousemoved, true).on("mouseup.drag", mouseupped, true);
        nodrag(event$2.view);
        nopropagation$4();
        mousemoving = false;
        mousedownx = event$2.clientX;
        mousedowny = event$2.clientY;
        gesture("start");
      }

      function mousemoved() {
        noevent$5();
        if (!mousemoving) {
          var dx = event$2.clientX - mousedownx, dy = event$2.clientY - mousedowny;
          mousemoving = dx * dx + dy * dy > clickDistance2;
        }
        gestures.mouse("drag");
      }

      function mouseupped() {
        select$2(event$2.view).on("mousemove.drag mouseup.drag", null);
        yesdrag$2(event$2.view, mousemoving);
        noevent$5();
        gestures.mouse("end");
      }

      function touchstarted() {
        if (!filter.apply(this, arguments)) return;
        var touches = event$2.changedTouches,
            c = container.apply(this, arguments),
            n = touches.length, i, gesture;

        for (i = 0; i < n; ++i) {
          if (gesture = beforestart(touches[i].identifier, c, touch$1, this, arguments)) {
            nopropagation$4();
            gesture("start");
          }
        }
      }

      function touchmoved() {
        var touches = event$2.changedTouches,
            n = touches.length, i, gesture;

        for (i = 0; i < n; ++i) {
          if (gesture = gestures[touches[i].identifier]) {
            noevent$5();
            gesture("drag");
          }
        }
      }

      function touchended() {
        var touches = event$2.changedTouches,
            n = touches.length, i, gesture;

        if (touchending) clearTimeout(touchending);
        touchending = setTimeout(function() { touchending = null; }, 500); // Ghost clicks are delayed!
        for (i = 0; i < n; ++i) {
          if (gesture = gestures[touches[i].identifier]) {
            nopropagation$4();
            gesture("end");
          }
        }
      }

      function beforestart(id, container, point, that, args) {
        var p = point(container, id), s, dx, dy,
            sublisteners = listeners.copy();

        if (!customEvent$2(new DragEvent$1(drag, "beforestart", s, id, active, p[0], p[1], 0, 0, sublisteners), function() {
          if ((event$2.subject = s = subject.apply(that, args)) == null) return false;
          dx = s.x - p[0] || 0;
          dy = s.y - p[1] || 0;
          return true;
        })) return;

        return function gesture(type) {
          var p0 = p, n;
          switch (type) {
            case "start": gestures[id] = gesture, n = active++; break;
            case "end": delete gestures[id], --active; // nobreak
            case "drag": p = point(container, id), n = active; break;
          }
          customEvent$2(new DragEvent$1(drag, type, s, id, n, p[0] + dx, p[1] + dy, p[0] - p0[0], p[1] - p0[1], sublisteners), sublisteners.apply, sublisteners, [type, that, args]);
        };
      }

      drag.filter = function(_) {
        return arguments.length ? (filter = typeof _ === "function" ? _ : constant$k(!!_), drag) : filter;
      };

      drag.container = function(_) {
        return arguments.length ? (container = typeof _ === "function" ? _ : constant$k(_), drag) : container;
      };

      drag.subject = function(_) {
        return arguments.length ? (subject = typeof _ === "function" ? _ : constant$k(_), drag) : subject;
      };

      drag.touchable = function(_) {
        return arguments.length ? (touchable = typeof _ === "function" ? _ : constant$k(!!_), drag) : touchable;
      };

      drag.on = function() {
        var value = listeners.on.apply(listeners, arguments);
        return value === listeners ? drag : value;
      };

      drag.clickDistance = function(_) {
        return arguments.length ? (clickDistance2 = (_ = +_) * _, drag) : Math.sqrt(clickDistance2);
      };

      return drag;
    }

    var EOL$1 = {},
        EOF$1 = {},
        QUOTE$1 = 34,
        NEWLINE$1 = 10,
        RETURN$1 = 13;

    function objectConverter$1(columns) {
      return new Function("d", "return {" + columns.map(function(name, i) {
        return JSON.stringify(name) + ": d[" + i + "]";
      }).join(",") + "}");
    }

    function customConverter$1(columns, f) {
      var object = objectConverter$1(columns);
      return function(row, i) {
        return f(object(row), i, columns);
      };
    }

    // Compute unique columns in order of discovery.
    function inferColumns$1(rows) {
      var columnSet = Object.create(null),
          columns = [];

      rows.forEach(function(row) {
        for (var column in row) {
          if (!(column in columnSet)) {
            columns.push(columnSet[column] = column);
          }
        }
      });

      return columns;
    }

    function dsv$1(delimiter) {
      var reFormat = new RegExp("[\"" + delimiter + "\n\r]"),
          DELIMITER = delimiter.charCodeAt(0);

      function parse(text, f) {
        var convert, columns, rows = parseRows(text, function(row, i) {
          if (convert) return convert(row, i - 1);
          columns = row, convert = f ? customConverter$1(row, f) : objectConverter$1(row);
        });
        rows.columns = columns || [];
        return rows;
      }

      function parseRows(text, f) {
        var rows = [], // output rows
            N = text.length,
            I = 0, // current character index
            n = 0, // current line number
            t, // current token
            eof = N <= 0, // current token followed by EOF?
            eol = false; // current token followed by EOL?

        // Strip the trailing newline.
        if (text.charCodeAt(N - 1) === NEWLINE$1) --N;
        if (text.charCodeAt(N - 1) === RETURN$1) --N;

        function token() {
          if (eof) return EOF$1;
          if (eol) return eol = false, EOL$1;

          // Unescape quotes.
          var i, j = I, c;
          if (text.charCodeAt(j) === QUOTE$1) {
            while (I++ < N && text.charCodeAt(I) !== QUOTE$1 || text.charCodeAt(++I) === QUOTE$1);
            if ((i = I) >= N) eof = true;
            else if ((c = text.charCodeAt(I++)) === NEWLINE$1) eol = true;
            else if (c === RETURN$1) { eol = true; if (text.charCodeAt(I) === NEWLINE$1) ++I; }
            return text.slice(j + 1, i - 1).replace(/""/g, "\"");
          }

          // Find next delimiter or newline.
          while (I < N) {
            if ((c = text.charCodeAt(i = I++)) === NEWLINE$1) eol = true;
            else if (c === RETURN$1) { eol = true; if (text.charCodeAt(I) === NEWLINE$1) ++I; }
            else if (c !== DELIMITER) continue;
            return text.slice(j, i);
          }

          // Return last token before EOF.
          return eof = true, text.slice(j, N);
        }

        while ((t = token()) !== EOF$1) {
          var row = [];
          while (t !== EOL$1 && t !== EOF$1) row.push(t), t = token();
          if (f && (row = f(row, n++)) == null) continue;
          rows.push(row);
        }

        return rows;
      }

      function format(rows, columns) {
        if (columns == null) columns = inferColumns$1(rows);
        return [columns.map(formatValue).join(delimiter)].concat(rows.map(function(row) {
          return columns.map(function(column) {
            return formatValue(row[column]);
          }).join(delimiter);
        })).join("\n");
      }

      function formatRows(rows) {
        return rows.map(formatRow).join("\n");
      }

      function formatRow(row) {
        return row.map(formatValue).join(delimiter);
      }

      function formatValue(text) {
        return text == null ? ""
            : reFormat.test(text += "") ? "\"" + text.replace(/"/g, "\"\"") + "\""
            : text;
      }

      return {
        parse: parse,
        parseRows: parseRows,
        format: format,
        formatRows: formatRows
      };
    }

    var csv$2 = dsv$1(",");

    var csvParse$1 = csv$2.parse;
    var csvParseRows$1 = csv$2.parseRows;
    var csvFormat$1 = csv$2.format;
    var csvFormatRows$1 = csv$2.formatRows;

    var tsv$2 = dsv$1("\t");

    var tsvParse$1 = tsv$2.parse;
    var tsvParseRows$1 = tsv$2.parseRows;
    var tsvFormat$1 = tsv$2.format;
    var tsvFormatRows$1 = tsv$2.formatRows;

    function linear$4(t) {
      return +t;
    }

    function quadIn$1(t) {
      return t * t;
    }

    function quadOut$1(t) {
      return t * (2 - t);
    }

    function quadInOut$1(t) {
      return ((t *= 2) <= 1 ? t * t : --t * (2 - t) + 1) / 2;
    }

    function cubicIn$1(t) {
      return t * t * t;
    }

    function cubicOut$1(t) {
      return --t * t * t + 1;
    }

    function cubicInOut$1(t) {
      return ((t *= 2) <= 1 ? t * t * t : (t -= 2) * t * t + 2) / 2;
    }

    var exponent$2 = 3;

    var polyIn$1 = (function custom(e) {
      e = +e;

      function polyIn(t) {
        return Math.pow(t, e);
      }

      polyIn.exponent = custom;

      return polyIn;
    })(exponent$2);

    var polyOut$1 = (function custom(e) {
      e = +e;

      function polyOut(t) {
        return 1 - Math.pow(1 - t, e);
      }

      polyOut.exponent = custom;

      return polyOut;
    })(exponent$2);

    var polyInOut$1 = (function custom(e) {
      e = +e;

      function polyInOut(t) {
        return ((t *= 2) <= 1 ? Math.pow(t, e) : 2 - Math.pow(2 - t, e)) / 2;
      }

      polyInOut.exponent = custom;

      return polyInOut;
    })(exponent$2);

    var pi$7 = Math.PI,
        halfPi$5 = pi$7 / 2;

    function sinIn$1(t) {
      return 1 - Math.cos(t * halfPi$5);
    }

    function sinOut$1(t) {
      return Math.sin(t * halfPi$5);
    }

    function sinInOut$1(t) {
      return (1 - Math.cos(pi$7 * t)) / 2;
    }

    function expIn$1(t) {
      return Math.pow(2, 10 * t - 10);
    }

    function expOut$1(t) {
      return 1 - Math.pow(2, -10 * t);
    }

    function expInOut$1(t) {
      return ((t *= 2) <= 1 ? Math.pow(2, 10 * t - 10) : 2 - Math.pow(2, 10 - 10 * t)) / 2;
    }

    function circleIn$1(t) {
      return 1 - Math.sqrt(1 - t * t);
    }

    function circleOut$1(t) {
      return Math.sqrt(1 - --t * t);
    }

    function circleInOut$1(t) {
      return ((t *= 2) <= 1 ? 1 - Math.sqrt(1 - t * t) : Math.sqrt(1 - (t -= 2) * t) + 1) / 2;
    }

    var b1$1 = 4 / 11,
        b2$1 = 6 / 11,
        b3$1 = 8 / 11,
        b4$1 = 3 / 4,
        b5$1 = 9 / 11,
        b6$1 = 10 / 11,
        b7$1 = 15 / 16,
        b8$1 = 21 / 22,
        b9$1 = 63 / 64,
        b0$1 = 1 / b1$1 / b1$1;

    function bounceIn$1(t) {
      return 1 - bounceOut$1(1 - t);
    }

    function bounceOut$1(t) {
      return (t = +t) < b1$1 ? b0$1 * t * t : t < b3$1 ? b0$1 * (t -= b2$1) * t + b4$1 : t < b6$1 ? b0$1 * (t -= b5$1) * t + b7$1 : b0$1 * (t -= b8$1) * t + b9$1;
    }

    function bounceInOut$1(t) {
      return ((t *= 2) <= 1 ? 1 - bounceOut$1(1 - t) : bounceOut$1(t - 1) + 1) / 2;
    }

    var overshoot$1 = 1.70158;

    var backIn$1 = (function custom(s) {
      s = +s;

      function backIn(t) {
        return t * t * ((s + 1) * t - s);
      }

      backIn.overshoot = custom;

      return backIn;
    })(overshoot$1);

    var backOut$1 = (function custom(s) {
      s = +s;

      function backOut(t) {
        return --t * t * ((s + 1) * t + s) + 1;
      }

      backOut.overshoot = custom;

      return backOut;
    })(overshoot$1);

    var backInOut$1 = (function custom(s) {
      s = +s;

      function backInOut(t) {
        return ((t *= 2) < 1 ? t * t * ((s + 1) * t - s) : (t -= 2) * t * ((s + 1) * t + s) + 2) / 2;
      }

      backInOut.overshoot = custom;

      return backInOut;
    })(overshoot$1);

    var tau$7 = 2 * Math.PI,
        amplitude$1 = 1,
        period$1 = 0.3;

    var elasticIn$1 = (function custom(a, p) {
      var s = Math.asin(1 / (a = Math.max(1, a))) * (p /= tau$7);

      function elasticIn(t) {
        return a * Math.pow(2, 10 * --t) * Math.sin((s - t) / p);
      }

      elasticIn.amplitude = function(a) { return custom(a, p * tau$7); };
      elasticIn.period = function(p) { return custom(a, p); };

      return elasticIn;
    })(amplitude$1, period$1);

    var elasticOut$1 = (function custom(a, p) {
      var s = Math.asin(1 / (a = Math.max(1, a))) * (p /= tau$7);

      function elasticOut(t) {
        return 1 - a * Math.pow(2, -10 * (t = +t)) * Math.sin((t + s) / p);
      }

      elasticOut.amplitude = function(a) { return custom(a, p * tau$7); };
      elasticOut.period = function(p) { return custom(a, p); };

      return elasticOut;
    })(amplitude$1, period$1);

    var elasticInOut$1 = (function custom(a, p) {
      var s = Math.asin(1 / (a = Math.max(1, a))) * (p /= tau$7);

      function elasticInOut(t) {
        return ((t = t * 2 - 1) < 0
            ? a * Math.pow(2, 10 * t) * Math.sin((s - t) / p)
            : 2 - a * Math.pow(2, -10 * t) * Math.sin((s + t) / p)) / 2;
      }

      elasticInOut.amplitude = function(a) { return custom(a, p * tau$7); };
      elasticInOut.period = function(p) { return custom(a, p); };

      return elasticInOut;
    })(amplitude$1, period$1);

    function center$3(x, y) {
      var nodes;

      if (x == null) x = 0;
      if (y == null) y = 0;

      function force() {
        var i,
            n = nodes.length,
            node,
            sx = 0,
            sy = 0;

        for (i = 0; i < n; ++i) {
          node = nodes[i], sx += node.x, sy += node.y;
        }

        for (sx = sx / n - x, sy = sy / n - y, i = 0; i < n; ++i) {
          node = nodes[i], node.x -= sx, node.y -= sy;
        }
      }

      force.initialize = function(_) {
        nodes = _;
      };

      force.x = function(_) {
        return arguments.length ? (x = +_, force) : x;
      };

      force.y = function(_) {
        return arguments.length ? (y = +_, force) : y;
      };

      return force;
    }

    function constant$l(x) {
      return function() {
        return x;
      };
    }

    function jiggle$1() {
      return (Math.random() - 0.5) * 1e-6;
    }

    function tree_add$1(d) {
      var x = +this._x.call(null, d),
          y = +this._y.call(null, d);
      return add$2(this.cover(x, y), x, y, d);
    }

    function add$2(tree, x, y, d) {
      if (isNaN(x) || isNaN(y)) return tree; // ignore invalid points

      var parent,
          node = tree._root,
          leaf = {data: d},
          x0 = tree._x0,
          y0 = tree._y0,
          x1 = tree._x1,
          y1 = tree._y1,
          xm,
          ym,
          xp,
          yp,
          right,
          bottom,
          i,
          j;

      // If the tree is empty, initialize the root as a leaf.
      if (!node) return tree._root = leaf, tree;

      // Find the existing leaf for the new point, or add it.
      while (node.length) {
        if (right = x >= (xm = (x0 + x1) / 2)) x0 = xm; else x1 = xm;
        if (bottom = y >= (ym = (y0 + y1) / 2)) y0 = ym; else y1 = ym;
        if (parent = node, !(node = node[i = bottom << 1 | right])) return parent[i] = leaf, tree;
      }

      // Is the new point is exactly coincident with the existing point?
      xp = +tree._x.call(null, node.data);
      yp = +tree._y.call(null, node.data);
      if (x === xp && y === yp) return leaf.next = node, parent ? parent[i] = leaf : tree._root = leaf, tree;

      // Otherwise, split the leaf node until the old and new point are separated.
      do {
        parent = parent ? parent[i] = new Array(4) : tree._root = new Array(4);
        if (right = x >= (xm = (x0 + x1) / 2)) x0 = xm; else x1 = xm;
        if (bottom = y >= (ym = (y0 + y1) / 2)) y0 = ym; else y1 = ym;
      } while ((i = bottom << 1 | right) === (j = (yp >= ym) << 1 | (xp >= xm)));
      return parent[j] = node, parent[i] = leaf, tree;
    }

    function addAll$1(data) {
      var d, i, n = data.length,
          x,
          y,
          xz = new Array(n),
          yz = new Array(n),
          x0 = Infinity,
          y0 = Infinity,
          x1 = -Infinity,
          y1 = -Infinity;

      // Compute the points and their extent.
      for (i = 0; i < n; ++i) {
        if (isNaN(x = +this._x.call(null, d = data[i])) || isNaN(y = +this._y.call(null, d))) continue;
        xz[i] = x;
        yz[i] = y;
        if (x < x0) x0 = x;
        if (x > x1) x1 = x;
        if (y < y0) y0 = y;
        if (y > y1) y1 = y;
      }

      // If there were no (valid) points, abort.
      if (x0 > x1 || y0 > y1) return this;

      // Expand the tree to cover the new points.
      this.cover(x0, y0).cover(x1, y1);

      // Add the new points.
      for (i = 0; i < n; ++i) {
        add$2(this, xz[i], yz[i], data[i]);
      }

      return this;
    }

    function tree_cover$1(x, y) {
      if (isNaN(x = +x) || isNaN(y = +y)) return this; // ignore invalid points

      var x0 = this._x0,
          y0 = this._y0,
          x1 = this._x1,
          y1 = this._y1;

      // If the quadtree has no extent, initialize them.
      // Integer extent are necessary so that if we later double the extent,
      // the existing quadrant boundaries don’t change due to floating point error!
      if (isNaN(x0)) {
        x1 = (x0 = Math.floor(x)) + 1;
        y1 = (y0 = Math.floor(y)) + 1;
      }

      // Otherwise, double repeatedly to cover.
      else {
        var z = x1 - x0,
            node = this._root,
            parent,
            i;

        while (x0 > x || x >= x1 || y0 > y || y >= y1) {
          i = (y < y0) << 1 | (x < x0);
          parent = new Array(4), parent[i] = node, node = parent, z *= 2;
          switch (i) {
            case 0: x1 = x0 + z, y1 = y0 + z; break;
            case 1: x0 = x1 - z, y1 = y0 + z; break;
            case 2: x1 = x0 + z, y0 = y1 - z; break;
            case 3: x0 = x1 - z, y0 = y1 - z; break;
          }
        }

        if (this._root && this._root.length) this._root = node;
      }

      this._x0 = x0;
      this._y0 = y0;
      this._x1 = x1;
      this._y1 = y1;
      return this;
    }

    function tree_data$1() {
      var data = [];
      this.visit(function(node) {
        if (!node.length) do data.push(node.data); while (node = node.next)
      });
      return data;
    }

    function tree_extent$1(_) {
      return arguments.length
          ? this.cover(+_[0][0], +_[0][1]).cover(+_[1][0], +_[1][1])
          : isNaN(this._x0) ? undefined : [[this._x0, this._y0], [this._x1, this._y1]];
    }

    function Quad$1(node, x0, y0, x1, y1) {
      this.node = node;
      this.x0 = x0;
      this.y0 = y0;
      this.x1 = x1;
      this.y1 = y1;
    }

    function tree_find$1(x, y, radius) {
      var data,
          x0 = this._x0,
          y0 = this._y0,
          x1,
          y1,
          x2,
          y2,
          x3 = this._x1,
          y3 = this._y1,
          quads = [],
          node = this._root,
          q,
          i;

      if (node) quads.push(new Quad$1(node, x0, y0, x3, y3));
      if (radius == null) radius = Infinity;
      else {
        x0 = x - radius, y0 = y - radius;
        x3 = x + radius, y3 = y + radius;
        radius *= radius;
      }

      while (q = quads.pop()) {

        // Stop searching if this quadrant can’t contain a closer node.
        if (!(node = q.node)
            || (x1 = q.x0) > x3
            || (y1 = q.y0) > y3
            || (x2 = q.x1) < x0
            || (y2 = q.y1) < y0) continue;

        // Bisect the current quadrant.
        if (node.length) {
          var xm = (x1 + x2) / 2,
              ym = (y1 + y2) / 2;

          quads.push(
            new Quad$1(node[3], xm, ym, x2, y2),
            new Quad$1(node[2], x1, ym, xm, y2),
            new Quad$1(node[1], xm, y1, x2, ym),
            new Quad$1(node[0], x1, y1, xm, ym)
          );

          // Visit the closest quadrant first.
          if (i = (y >= ym) << 1 | (x >= xm)) {
            q = quads[quads.length - 1];
            quads[quads.length - 1] = quads[quads.length - 1 - i];
            quads[quads.length - 1 - i] = q;
          }
        }

        // Visit this point. (Visiting coincident points isn’t necessary!)
        else {
          var dx = x - +this._x.call(null, node.data),
              dy = y - +this._y.call(null, node.data),
              d2 = dx * dx + dy * dy;
          if (d2 < radius) {
            var d = Math.sqrt(radius = d2);
            x0 = x - d, y0 = y - d;
            x3 = x + d, y3 = y + d;
            data = node.data;
          }
        }
      }

      return data;
    }

    function tree_remove$1(d) {
      if (isNaN(x = +this._x.call(null, d)) || isNaN(y = +this._y.call(null, d))) return this; // ignore invalid points

      var parent,
          node = this._root,
          retainer,
          previous,
          next,
          x0 = this._x0,
          y0 = this._y0,
          x1 = this._x1,
          y1 = this._y1,
          x,
          y,
          xm,
          ym,
          right,
          bottom,
          i,
          j;

      // If the tree is empty, initialize the root as a leaf.
      if (!node) return this;

      // Find the leaf node for the point.
      // While descending, also retain the deepest parent with a non-removed sibling.
      if (node.length) while (true) {
        if (right = x >= (xm = (x0 + x1) / 2)) x0 = xm; else x1 = xm;
        if (bottom = y >= (ym = (y0 + y1) / 2)) y0 = ym; else y1 = ym;
        if (!(parent = node, node = node[i = bottom << 1 | right])) return this;
        if (!node.length) break;
        if (parent[(i + 1) & 3] || parent[(i + 2) & 3] || parent[(i + 3) & 3]) retainer = parent, j = i;
      }

      // Find the point to remove.
      while (node.data !== d) if (!(previous = node, node = node.next)) return this;
      if (next = node.next) delete node.next;

      // If there are multiple coincident points, remove just the point.
      if (previous) return (next ? previous.next = next : delete previous.next), this;

      // If this is the root point, remove it.
      if (!parent) return this._root = next, this;

      // Remove this leaf.
      next ? parent[i] = next : delete parent[i];

      // If the parent now contains exactly one leaf, collapse superfluous parents.
      if ((node = parent[0] || parent[1] || parent[2] || parent[3])
          && node === (parent[3] || parent[2] || parent[1] || parent[0])
          && !node.length) {
        if (retainer) retainer[j] = node;
        else this._root = node;
      }

      return this;
    }

    function removeAll$1(data) {
      for (var i = 0, n = data.length; i < n; ++i) this.remove(data[i]);
      return this;
    }

    function tree_root$1() {
      return this._root;
    }

    function tree_size$1() {
      var size = 0;
      this.visit(function(node) {
        if (!node.length) do ++size; while (node = node.next)
      });
      return size;
    }

    function tree_visit$1(callback) {
      var quads = [], q, node = this._root, child, x0, y0, x1, y1;
      if (node) quads.push(new Quad$1(node, this._x0, this._y0, this._x1, this._y1));
      while (q = quads.pop()) {
        if (!callback(node = q.node, x0 = q.x0, y0 = q.y0, x1 = q.x1, y1 = q.y1) && node.length) {
          var xm = (x0 + x1) / 2, ym = (y0 + y1) / 2;
          if (child = node[3]) quads.push(new Quad$1(child, xm, ym, x1, y1));
          if (child = node[2]) quads.push(new Quad$1(child, x0, ym, xm, y1));
          if (child = node[1]) quads.push(new Quad$1(child, xm, y0, x1, ym));
          if (child = node[0]) quads.push(new Quad$1(child, x0, y0, xm, ym));
        }
      }
      return this;
    }

    function tree_visitAfter$1(callback) {
      var quads = [], next = [], q;
      if (this._root) quads.push(new Quad$1(this._root, this._x0, this._y0, this._x1, this._y1));
      while (q = quads.pop()) {
        var node = q.node;
        if (node.length) {
          var child, x0 = q.x0, y0 = q.y0, x1 = q.x1, y1 = q.y1, xm = (x0 + x1) / 2, ym = (y0 + y1) / 2;
          if (child = node[0]) quads.push(new Quad$1(child, x0, y0, xm, ym));
          if (child = node[1]) quads.push(new Quad$1(child, xm, y0, x1, ym));
          if (child = node[2]) quads.push(new Quad$1(child, x0, ym, xm, y1));
          if (child = node[3]) quads.push(new Quad$1(child, xm, ym, x1, y1));
        }
        next.push(q);
      }
      while (q = next.pop()) {
        callback(q.node, q.x0, q.y0, q.x1, q.y1);
      }
      return this;
    }

    function defaultX$2(d) {
      return d[0];
    }

    function tree_x$1(_) {
      return arguments.length ? (this._x = _, this) : this._x;
    }

    function defaultY$2(d) {
      return d[1];
    }

    function tree_y$1(_) {
      return arguments.length ? (this._y = _, this) : this._y;
    }

    function quadtree$1(nodes, x, y) {
      var tree = new Quadtree$1(x == null ? defaultX$2 : x, y == null ? defaultY$2 : y, NaN, NaN, NaN, NaN);
      return nodes == null ? tree : tree.addAll(nodes);
    }

    function Quadtree$1(x, y, x0, y0, x1, y1) {
      this._x = x;
      this._y = y;
      this._x0 = x0;
      this._y0 = y0;
      this._x1 = x1;
      this._y1 = y1;
      this._root = undefined;
    }

    function leaf_copy$1(leaf) {
      var copy = {data: leaf.data}, next = copy;
      while (leaf = leaf.next) next = next.next = {data: leaf.data};
      return copy;
    }

    var treeProto$1 = quadtree$1.prototype = Quadtree$1.prototype;

    treeProto$1.copy = function() {
      var copy = new Quadtree$1(this._x, this._y, this._x0, this._y0, this._x1, this._y1),
          node = this._root,
          nodes,
          child;

      if (!node) return copy;

      if (!node.length) return copy._root = leaf_copy$1(node), copy;

      nodes = [{source: node, target: copy._root = new Array(4)}];
      while (node = nodes.pop()) {
        for (var i = 0; i < 4; ++i) {
          if (child = node.source[i]) {
            if (child.length) nodes.push({source: child, target: node.target[i] = new Array(4)});
            else node.target[i] = leaf_copy$1(child);
          }
        }
      }

      return copy;
    };

    treeProto$1.add = tree_add$1;
    treeProto$1.addAll = addAll$1;
    treeProto$1.cover = tree_cover$1;
    treeProto$1.data = tree_data$1;
    treeProto$1.extent = tree_extent$1;
    treeProto$1.find = tree_find$1;
    treeProto$1.remove = tree_remove$1;
    treeProto$1.removeAll = removeAll$1;
    treeProto$1.root = tree_root$1;
    treeProto$1.size = tree_size$1;
    treeProto$1.visit = tree_visit$1;
    treeProto$1.visitAfter = tree_visitAfter$1;
    treeProto$1.x = tree_x$1;
    treeProto$1.y = tree_y$1;

    function x$5(d) {
      return d.x + d.vx;
    }

    function y$5(d) {
      return d.y + d.vy;
    }

    function collide$1(radius) {
      var nodes,
          radii,
          strength = 1,
          iterations = 1;

      if (typeof radius !== "function") radius = constant$l(radius == null ? 1 : +radius);

      function force() {
        var i, n = nodes.length,
            tree,
            node,
            xi,
            yi,
            ri,
            ri2;

        for (var k = 0; k < iterations; ++k) {
          tree = quadtree$1(nodes, x$5, y$5).visitAfter(prepare);
          for (i = 0; i < n; ++i) {
            node = nodes[i];
            ri = radii[node.index], ri2 = ri * ri;
            xi = node.x + node.vx;
            yi = node.y + node.vy;
            tree.visit(apply);
          }
        }

        function apply(quad, x0, y0, x1, y1) {
          var data = quad.data, rj = quad.r, r = ri + rj;
          if (data) {
            if (data.index > node.index) {
              var x = xi - data.x - data.vx,
                  y = yi - data.y - data.vy,
                  l = x * x + y * y;
              if (l < r * r) {
                if (x === 0) x = jiggle$1(), l += x * x;
                if (y === 0) y = jiggle$1(), l += y * y;
                l = (r - (l = Math.sqrt(l))) / l * strength;
                node.vx += (x *= l) * (r = (rj *= rj) / (ri2 + rj));
                node.vy += (y *= l) * r;
                data.vx -= x * (r = 1 - r);
                data.vy -= y * r;
              }
            }
            return;
          }
          return x0 > xi + r || x1 < xi - r || y0 > yi + r || y1 < yi - r;
        }
      }

      function prepare(quad) {
        if (quad.data) return quad.r = radii[quad.data.index];
        for (var i = quad.r = 0; i < 4; ++i) {
          if (quad[i] && quad[i].r > quad.r) {
            quad.r = quad[i].r;
          }
        }
      }

      function initialize() {
        if (!nodes) return;
        var i, n = nodes.length, node;
        radii = new Array(n);
        for (i = 0; i < n; ++i) node = nodes[i], radii[node.index] = +radius(node, i, nodes);
      }

      force.initialize = function(_) {
        nodes = _;
        initialize();
      };

      force.iterations = function(_) {
        return arguments.length ? (iterations = +_, force) : iterations;
      };

      force.strength = function(_) {
        return arguments.length ? (strength = +_, force) : strength;
      };

      force.radius = function(_) {
        return arguments.length ? (radius = typeof _ === "function" ? _ : constant$l(+_), initialize(), force) : radius;
      };

      return force;
    }

    var prefix$2 = "$";

    function Map$3() {}

    Map$3.prototype = map$6.prototype = {
      constructor: Map$3,
      has: function(key) {
        return (prefix$2 + key) in this;
      },
      get: function(key) {
        return this[prefix$2 + key];
      },
      set: function(key, value) {
        this[prefix$2 + key] = value;
        return this;
      },
      remove: function(key) {
        var property = prefix$2 + key;
        return property in this && delete this[property];
      },
      clear: function() {
        for (var property in this) if (property[0] === prefix$2) delete this[property];
      },
      keys: function() {
        var keys = [];
        for (var property in this) if (property[0] === prefix$2) keys.push(property.slice(1));
        return keys;
      },
      values: function() {
        var values = [];
        for (var property in this) if (property[0] === prefix$2) values.push(this[property]);
        return values;
      },
      entries: function() {
        var entries = [];
        for (var property in this) if (property[0] === prefix$2) entries.push({key: property.slice(1), value: this[property]});
        return entries;
      },
      size: function() {
        var size = 0;
        for (var property in this) if (property[0] === prefix$2) ++size;
        return size;
      },
      empty: function() {
        for (var property in this) if (property[0] === prefix$2) return false;
        return true;
      },
      each: function(f) {
        for (var property in this) if (property[0] === prefix$2) f(this[property], property.slice(1), this);
      }
    };

    function map$6(object, f) {
      var map = new Map$3;

      // Copy constructor.
      if (object instanceof Map$3) object.each(function(value, key) { map.set(key, value); });

      // Index array by numeric index or specified key function.
      else if (Array.isArray(object)) {
        var i = -1,
            n = object.length,
            o;

        if (f == null) while (++i < n) map.set(i, object[i]);
        else while (++i < n) map.set(f(o = object[i], i, object), o);
      }

      // Convert object to map.
      else if (object) for (var key in object) map.set(key, object[key]);

      return map;
    }

    function Set$3() {}

    var proto$3 = map$6.prototype;

    Set$3.prototype = set$9.prototype = {
      constructor: Set$3,
      has: proto$3.has,
      add: function(value) {
        value += "";
        this[prefix$2 + value] = value;
        return this;
      },
      remove: proto$3.remove,
      clear: proto$3.clear,
      values: proto$3.keys,
      size: proto$3.size,
      empty: proto$3.empty,
      each: proto$3.each
    };

    function set$9(object, f) {
      var set = new Set$3;

      // Copy constructor.
      if (object instanceof Set$3) object.each(function(value) { set.add(value); });

      // Otherwise, assume it’s an array.
      else if (object) {
        var i = -1, n = object.length;
        if (f == null) while (++i < n) set.add(object[i]);
        else while (++i < n) set.add(f(object[i], i, object));
      }

      return set;
    }

    function index$4(d) {
      return d.index;
    }

    function find$1(nodeById, nodeId) {
      var node = nodeById.get(nodeId);
      if (!node) throw new Error("missing: " + nodeId);
      return node;
    }

    function link$3(links) {
      var id = index$4,
          strength = defaultStrength,
          strengths,
          distance = constant$l(30),
          distances,
          nodes,
          count,
          bias,
          iterations = 1;

      if (links == null) links = [];

      function defaultStrength(link) {
        return 1 / Math.min(count[link.source.index], count[link.target.index]);
      }

      function force(alpha) {
        for (var k = 0, n = links.length; k < iterations; ++k) {
          for (var i = 0, link, source, target, x, y, l, b; i < n; ++i) {
            link = links[i], source = link.source, target = link.target;
            x = target.x + target.vx - source.x - source.vx || jiggle$1();
            y = target.y + target.vy - source.y - source.vy || jiggle$1();
            l = Math.sqrt(x * x + y * y);
            l = (l - distances[i]) / l * alpha * strengths[i];
            x *= l, y *= l;
            target.vx -= x * (b = bias[i]);
            target.vy -= y * b;
            source.vx += x * (b = 1 - b);
            source.vy += y * b;
          }
        }
      }

      function initialize() {
        if (!nodes) return;

        var i,
            n = nodes.length,
            m = links.length,
            nodeById = map$6(nodes, id),
            link;

        for (i = 0, count = new Array(n); i < m; ++i) {
          link = links[i], link.index = i;
          if (typeof link.source !== "object") link.source = find$1(nodeById, link.source);
          if (typeof link.target !== "object") link.target = find$1(nodeById, link.target);
          count[link.source.index] = (count[link.source.index] || 0) + 1;
          count[link.target.index] = (count[link.target.index] || 0) + 1;
        }

        for (i = 0, bias = new Array(m); i < m; ++i) {
          link = links[i], bias[i] = count[link.source.index] / (count[link.source.index] + count[link.target.index]);
        }

        strengths = new Array(m), initializeStrength();
        distances = new Array(m), initializeDistance();
      }

      function initializeStrength() {
        if (!nodes) return;

        for (var i = 0, n = links.length; i < n; ++i) {
          strengths[i] = +strength(links[i], i, links);
        }
      }

      function initializeDistance() {
        if (!nodes) return;

        for (var i = 0, n = links.length; i < n; ++i) {
          distances[i] = +distance(links[i], i, links);
        }
      }

      force.initialize = function(_) {
        nodes = _;
        initialize();
      };

      force.links = function(_) {
        return arguments.length ? (links = _, initialize(), force) : links;
      };

      force.id = function(_) {
        return arguments.length ? (id = _, force) : id;
      };

      force.iterations = function(_) {
        return arguments.length ? (iterations = +_, force) : iterations;
      };

      force.strength = function(_) {
        return arguments.length ? (strength = typeof _ === "function" ? _ : constant$l(+_), initializeStrength(), force) : strength;
      };

      force.distance = function(_) {
        return arguments.length ? (distance = typeof _ === "function" ? _ : constant$l(+_), initializeDistance(), force) : distance;
      };

      return force;
    }

    var noop$7 = {value: function() {}};

    function dispatch$4() {
      for (var i = 0, n = arguments.length, _ = {}, t; i < n; ++i) {
        if (!(t = arguments[i] + "") || (t in _) || /[\s.]/.test(t)) throw new Error("illegal type: " + t);
        _[t] = [];
      }
      return new Dispatch$4(_);
    }

    function Dispatch$4(_) {
      this._ = _;
    }

    function parseTypenames$7(typenames, types) {
      return typenames.trim().split(/^|\s+/).map(function(t) {
        var name = "", i = t.indexOf(".");
        if (i >= 0) name = t.slice(i + 1), t = t.slice(0, i);
        if (t && !types.hasOwnProperty(t)) throw new Error("unknown type: " + t);
        return {type: t, name: name};
      });
    }

    Dispatch$4.prototype = dispatch$4.prototype = {
      constructor: Dispatch$4,
      on: function(typename, callback) {
        var _ = this._,
            T = parseTypenames$7(typename + "", _),
            t,
            i = -1,
            n = T.length;

        // If no callback was specified, return the callback of the given type and name.
        if (arguments.length < 2) {
          while (++i < n) if ((t = (typename = T[i]).type) && (t = get$7(_[t], typename.name))) return t;
          return;
        }

        // If a type was specified, set the callback for the given type and name.
        // Otherwise, if a null callback was specified, remove callbacks of the given name.
        if (callback != null && typeof callback !== "function") throw new Error("invalid callback: " + callback);
        while (++i < n) {
          if (t = (typename = T[i]).type) _[t] = set$a(_[t], typename.name, callback);
          else if (callback == null) for (t in _) _[t] = set$a(_[t], typename.name, null);
        }

        return this;
      },
      copy: function() {
        var copy = {}, _ = this._;
        for (var t in _) copy[t] = _[t].slice();
        return new Dispatch$4(copy);
      },
      call: function(type, that) {
        if ((n = arguments.length - 2) > 0) for (var args = new Array(n), i = 0, n, t; i < n; ++i) args[i] = arguments[i + 2];
        if (!this._.hasOwnProperty(type)) throw new Error("unknown type: " + type);
        for (t = this._[type], i = 0, n = t.length; i < n; ++i) t[i].value.apply(that, args);
      },
      apply: function(type, that, args) {
        if (!this._.hasOwnProperty(type)) throw new Error("unknown type: " + type);
        for (var t = this._[type], i = 0, n = t.length; i < n; ++i) t[i].value.apply(that, args);
      }
    };

    function get$7(type, name) {
      for (var i = 0, n = type.length, c; i < n; ++i) {
        if ((c = type[i]).name === name) {
          return c.value;
        }
      }
    }

    function set$a(type, name, callback) {
      for (var i = 0, n = type.length; i < n; ++i) {
        if (type[i].name === name) {
          type[i] = noop$7, type = type.slice(0, i).concat(type.slice(i + 1));
          break;
        }
      }
      if (callback != null) type.push({name: name, value: callback});
      return type;
    }

    var frame$2 = 0, // is an animation frame pending?
        timeout$4 = 0, // is a timeout pending?
        interval$3 = 0, // are any timers active?
        pokeDelay$2 = 1000, // how frequently we check for clock skew
        taskHead$2,
        taskTail$2,
        clockLast$2 = 0,
        clockNow$2 = 0,
        clockSkew$2 = 0,
        clock$2 = typeof performance === "object" && performance.now ? performance : Date,
        setFrame$2 = typeof window === "object" && window.requestAnimationFrame ? window.requestAnimationFrame.bind(window) : function(f) { setTimeout(f, 17); };

    function now$2() {
      return clockNow$2 || (setFrame$2(clearNow$2), clockNow$2 = clock$2.now() + clockSkew$2);
    }

    function clearNow$2() {
      clockNow$2 = 0;
    }

    function Timer$2() {
      this._call =
      this._time =
      this._next = null;
    }

    Timer$2.prototype = timer$2.prototype = {
      constructor: Timer$2,
      restart: function(callback, delay, time) {
        if (typeof callback !== "function") throw new TypeError("callback is not a function");
        time = (time == null ? now$2() : +time) + (delay == null ? 0 : +delay);
        if (!this._next && taskTail$2 !== this) {
          if (taskTail$2) taskTail$2._next = this;
          else taskHead$2 = this;
          taskTail$2 = this;
        }
        this._call = callback;
        this._time = time;
        sleep$2();
      },
      stop: function() {
        if (this._call) {
          this._call = null;
          this._time = Infinity;
          sleep$2();
        }
      }
    };

    function timer$2(callback, delay, time) {
      var t = new Timer$2;
      t.restart(callback, delay, time);
      return t;
    }

    function timerFlush$2() {
      now$2(); // Get the current time, if not already set.
      ++frame$2; // Pretend we’ve set an alarm, if we haven’t already.
      var t = taskHead$2, e;
      while (t) {
        if ((e = clockNow$2 - t._time) >= 0) t._call.call(null, e);
        t = t._next;
      }
      --frame$2;
    }

    function wake$2() {
      clockNow$2 = (clockLast$2 = clock$2.now()) + clockSkew$2;
      frame$2 = timeout$4 = 0;
      try {
        timerFlush$2();
      } finally {
        frame$2 = 0;
        nap$2();
        clockNow$2 = 0;
      }
    }

    function poke$2() {
      var now = clock$2.now(), delay = now - clockLast$2;
      if (delay > pokeDelay$2) clockSkew$2 -= delay, clockLast$2 = now;
    }

    function nap$2() {
      var t0, t1 = taskHead$2, t2, time = Infinity;
      while (t1) {
        if (t1._call) {
          if (time > t1._time) time = t1._time;
          t0 = t1, t1 = t1._next;
        } else {
          t2 = t1._next, t1._next = null;
          t1 = t0 ? t0._next = t2 : taskHead$2 = t2;
        }
      }
      taskTail$2 = t0;
      sleep$2(time);
    }

    function sleep$2(time) {
      if (frame$2) return; // Soonest alarm already set, or will be.
      if (timeout$4) timeout$4 = clearTimeout(timeout$4);
      var delay = time - clockNow$2; // Strictly less than if we recomputed clockNow.
      if (delay > 24) {
        if (time < Infinity) timeout$4 = setTimeout(wake$2, time - clock$2.now() - clockSkew$2);
        if (interval$3) interval$3 = clearInterval(interval$3);
      } else {
        if (!interval$3) clockLast$2 = clock$2.now(), interval$3 = setInterval(poke$2, pokeDelay$2);
        frame$2 = 1, setFrame$2(wake$2);
      }
    }

    function x$6(d) {
      return d.x;
    }

    function y$6(d) {
      return d.y;
    }

    var initialRadius$1 = 10,
        initialAngle$1 = Math.PI * (3 - Math.sqrt(5));

    function simulation$1(nodes) {
      var simulation,
          alpha = 1,
          alphaMin = 0.001,
          alphaDecay = 1 - Math.pow(alphaMin, 1 / 300),
          alphaTarget = 0,
          velocityDecay = 0.6,
          forces = map$6(),
          stepper = timer$2(step),
          event = dispatch$4("tick", "end");

      if (nodes == null) nodes = [];

      function step() {
        tick();
        event.call("tick", simulation);
        if (alpha < alphaMin) {
          stepper.stop();
          event.call("end", simulation);
        }
      }

      function tick() {
        var i, n = nodes.length, node;

        alpha += (alphaTarget - alpha) * alphaDecay;

        forces.each(function(force) {
          force(alpha);
        });

        for (i = 0; i < n; ++i) {
          node = nodes[i];
          if (node.fx == null) node.x += node.vx *= velocityDecay;
          else node.x = node.fx, node.vx = 0;
          if (node.fy == null) node.y += node.vy *= velocityDecay;
          else node.y = node.fy, node.vy = 0;
        }
      }

      function initializeNodes() {
        for (var i = 0, n = nodes.length, node; i < n; ++i) {
          node = nodes[i], node.index = i;
          if (isNaN(node.x) || isNaN(node.y)) {
            var radius = initialRadius$1 * Math.sqrt(i), angle = i * initialAngle$1;
            node.x = radius * Math.cos(angle);
            node.y = radius * Math.sin(angle);
          }
          if (isNaN(node.vx) || isNaN(node.vy)) {
            node.vx = node.vy = 0;
          }
        }
      }

      function initializeForce(force) {
        if (force.initialize) force.initialize(nodes);
        return force;
      }

      initializeNodes();

      return simulation = {
        tick: tick,

        restart: function() {
          return stepper.restart(step), simulation;
        },

        stop: function() {
          return stepper.stop(), simulation;
        },

        nodes: function(_) {
          return arguments.length ? (nodes = _, initializeNodes(), forces.each(initializeForce), simulation) : nodes;
        },

        alpha: function(_) {
          return arguments.length ? (alpha = +_, simulation) : alpha;
        },

        alphaMin: function(_) {
          return arguments.length ? (alphaMin = +_, simulation) : alphaMin;
        },

        alphaDecay: function(_) {
          return arguments.length ? (alphaDecay = +_, simulation) : +alphaDecay;
        },

        alphaTarget: function(_) {
          return arguments.length ? (alphaTarget = +_, simulation) : alphaTarget;
        },

        velocityDecay: function(_) {
          return arguments.length ? (velocityDecay = 1 - _, simulation) : 1 - velocityDecay;
        },

        force: function(name, _) {
          return arguments.length > 1 ? ((_ == null ? forces.remove(name) : forces.set(name, initializeForce(_))), simulation) : forces.get(name);
        },

        find: function(x, y, radius) {
          var i = 0,
              n = nodes.length,
              dx,
              dy,
              d2,
              node,
              closest;

          if (radius == null) radius = Infinity;
          else radius *= radius;

          for (i = 0; i < n; ++i) {
            node = nodes[i];
            dx = x - node.x;
            dy = y - node.y;
            d2 = dx * dx + dy * dy;
            if (d2 < radius) closest = node, radius = d2;
          }

          return closest;
        },

        on: function(name, _) {
          return arguments.length > 1 ? (event.on(name, _), simulation) : event.on(name);
        }
      };
    }

    function manyBody$1() {
      var nodes,
          node,
          alpha,
          strength = constant$l(-30),
          strengths,
          distanceMin2 = 1,
          distanceMax2 = Infinity,
          theta2 = 0.81;

      function force(_) {
        var i, n = nodes.length, tree = quadtree$1(nodes, x$6, y$6).visitAfter(accumulate);
        for (alpha = _, i = 0; i < n; ++i) node = nodes[i], tree.visit(apply);
      }

      function initialize() {
        if (!nodes) return;
        var i, n = nodes.length, node;
        strengths = new Array(n);
        for (i = 0; i < n; ++i) node = nodes[i], strengths[node.index] = +strength(node, i, nodes);
      }

      function accumulate(quad) {
        var strength = 0, q, c, weight = 0, x, y, i;

        // For internal nodes, accumulate forces from child quadrants.
        if (quad.length) {
          for (x = y = i = 0; i < 4; ++i) {
            if ((q = quad[i]) && (c = Math.abs(q.value))) {
              strength += q.value, weight += c, x += c * q.x, y += c * q.y;
            }
          }
          quad.x = x / weight;
          quad.y = y / weight;
        }

        // For leaf nodes, accumulate forces from coincident quadrants.
        else {
          q = quad;
          q.x = q.data.x;
          q.y = q.data.y;
          do strength += strengths[q.data.index];
          while (q = q.next);
        }

        quad.value = strength;
      }

      function apply(quad, x1, _, x2) {
        if (!quad.value) return true;

        var x = quad.x - node.x,
            y = quad.y - node.y,
            w = x2 - x1,
            l = x * x + y * y;

        // Apply the Barnes-Hut approximation if possible.
        // Limit forces for very close nodes; randomize direction if coincident.
        if (w * w / theta2 < l) {
          if (l < distanceMax2) {
            if (x === 0) x = jiggle$1(), l += x * x;
            if (y === 0) y = jiggle$1(), l += y * y;
            if (l < distanceMin2) l = Math.sqrt(distanceMin2 * l);
            node.vx += x * quad.value * alpha / l;
            node.vy += y * quad.value * alpha / l;
          }
          return true;
        }

        // Otherwise, process points directly.
        else if (quad.length || l >= distanceMax2) return;

        // Limit forces for very close nodes; randomize direction if coincident.
        if (quad.data !== node || quad.next) {
          if (x === 0) x = jiggle$1(), l += x * x;
          if (y === 0) y = jiggle$1(), l += y * y;
          if (l < distanceMin2) l = Math.sqrt(distanceMin2 * l);
        }

        do if (quad.data !== node) {
          w = strengths[quad.data.index] * alpha / l;
          node.vx += x * w;
          node.vy += y * w;
        } while (quad = quad.next);
      }

      force.initialize = function(_) {
        nodes = _;
        initialize();
      };

      force.strength = function(_) {
        return arguments.length ? (strength = typeof _ === "function" ? _ : constant$l(+_), initialize(), force) : strength;
      };

      force.distanceMin = function(_) {
        return arguments.length ? (distanceMin2 = _ * _, force) : Math.sqrt(distanceMin2);
      };

      force.distanceMax = function(_) {
        return arguments.length ? (distanceMax2 = _ * _, force) : Math.sqrt(distanceMax2);
      };

      force.theta = function(_) {
        return arguments.length ? (theta2 = _ * _, force) : Math.sqrt(theta2);
      };

      return force;
    }

    function radial$1(radius, x, y) {
      var nodes,
          strength = constant$l(0.1),
          strengths,
          radiuses;

      if (typeof radius !== "function") radius = constant$l(+radius);
      if (x == null) x = 0;
      if (y == null) y = 0;

      function force(alpha) {
        for (var i = 0, n = nodes.length; i < n; ++i) {
          var node = nodes[i],
              dx = node.x - x || 1e-6,
              dy = node.y - y || 1e-6,
              r = Math.sqrt(dx * dx + dy * dy),
              k = (radiuses[i] - r) * strengths[i] * alpha / r;
          node.vx += dx * k;
          node.vy += dy * k;
        }
      }

      function initialize() {
        if (!nodes) return;
        var i, n = nodes.length;
        strengths = new Array(n);
        radiuses = new Array(n);
        for (i = 0; i < n; ++i) {
          radiuses[i] = +radius(nodes[i], i, nodes);
          strengths[i] = isNaN(radiuses[i]) ? 0 : +strength(nodes[i], i, nodes);
        }
      }

      force.initialize = function(_) {
        nodes = _, initialize();
      };

      force.strength = function(_) {
        return arguments.length ? (strength = typeof _ === "function" ? _ : constant$l(+_), initialize(), force) : strength;
      };

      force.radius = function(_) {
        return arguments.length ? (radius = typeof _ === "function" ? _ : constant$l(+_), initialize(), force) : radius;
      };

      force.x = function(_) {
        return arguments.length ? (x = +_, force) : x;
      };

      force.y = function(_) {
        return arguments.length ? (y = +_, force) : y;
      };

      return force;
    }

    function x$7(x) {
      var strength = constant$l(0.1),
          nodes,
          strengths,
          xz;

      if (typeof x !== "function") x = constant$l(x == null ? 0 : +x);

      function force(alpha) {
        for (var i = 0, n = nodes.length, node; i < n; ++i) {
          node = nodes[i], node.vx += (xz[i] - node.x) * strengths[i] * alpha;
        }
      }

      function initialize() {
        if (!nodes) return;
        var i, n = nodes.length;
        strengths = new Array(n);
        xz = new Array(n);
        for (i = 0; i < n; ++i) {
          strengths[i] = isNaN(xz[i] = +x(nodes[i], i, nodes)) ? 0 : +strength(nodes[i], i, nodes);
        }
      }

      force.initialize = function(_) {
        nodes = _;
        initialize();
      };

      force.strength = function(_) {
        return arguments.length ? (strength = typeof _ === "function" ? _ : constant$l(+_), initialize(), force) : strength;
      };

      force.x = function(_) {
        return arguments.length ? (x = typeof _ === "function" ? _ : constant$l(+_), initialize(), force) : x;
      };

      return force;
    }

    function y$7(y) {
      var strength = constant$l(0.1),
          nodes,
          strengths,
          yz;

      if (typeof y !== "function") y = constant$l(y == null ? 0 : +y);

      function force(alpha) {
        for (var i = 0, n = nodes.length, node; i < n; ++i) {
          node = nodes[i], node.vy += (yz[i] - node.y) * strengths[i] * alpha;
        }
      }

      function initialize() {
        if (!nodes) return;
        var i, n = nodes.length;
        strengths = new Array(n);
        yz = new Array(n);
        for (i = 0; i < n; ++i) {
          strengths[i] = isNaN(yz[i] = +y(nodes[i], i, nodes)) ? 0 : +strength(nodes[i], i, nodes);
        }
      }

      force.initialize = function(_) {
        nodes = _;
        initialize();
      };

      force.strength = function(_) {
        return arguments.length ? (strength = typeof _ === "function" ? _ : constant$l(+_), initialize(), force) : strength;
      };

      force.y = function(_) {
        return arguments.length ? (y = typeof _ === "function" ? _ : constant$l(+_), initialize(), force) : y;
      };

      return force;
    }

    // Computes the decimal coefficient and exponent of the specified number x with
    // significant digits p, where x is positive and p is in [1, 21] or undefined.
    // For example, formatDecimal(1.23) returns ["123", 0].
    function formatDecimal$1(x, p) {
      if ((i = (x = p ? x.toExponential(p - 1) : x.toExponential()).indexOf("e")) < 0) return null; // NaN, ±Infinity
      var i, coefficient = x.slice(0, i);

      // The string returned by toExponential either has the form \d\.\d+e[-+]\d+
      // (e.g., 1.2e+3) or the form \de[-+]\d+ (e.g., 1e+3).
      return [
        coefficient.length > 1 ? coefficient[0] + coefficient.slice(2) : coefficient,
        +x.slice(i + 1)
      ];
    }

    function exponent$3(x) {
      return x = formatDecimal$1(Math.abs(x)), x ? x[1] : NaN;
    }

    function formatGroup$1(grouping, thousands) {
      return function(value, width) {
        var i = value.length,
            t = [],
            j = 0,
            g = grouping[0],
            length = 0;

        while (i > 0 && g > 0) {
          if (length + g + 1 > width) g = Math.max(1, width - length);
          t.push(value.substring(i -= g, i + g));
          if ((length += g + 1) > width) break;
          g = grouping[j = (j + 1) % grouping.length];
        }

        return t.reverse().join(thousands);
      };
    }

    function formatNumerals$1(numerals) {
      return function(value) {
        return value.replace(/[0-9]/g, function(i) {
          return numerals[+i];
        });
      };
    }

    function formatDefault(x, p) {
      x = x.toPrecision(p);

      out: for (var n = x.length, i = 1, i0 = -1, i1; i < n; ++i) {
        switch (x[i]) {
          case ".": i0 = i1 = i; break;
          case "0": if (i0 === 0) i0 = i; i1 = i; break;
          case "e": break out;
          default: if (i0 > 0) i0 = 0; break;
        }
      }

      return i0 > 0 ? x.slice(0, i0) + x.slice(i1 + 1) : x;
    }

    var prefixExponent$1;

    function formatPrefixAuto$1(x, p) {
      var d = formatDecimal$1(x, p);
      if (!d) return x + "";
      var coefficient = d[0],
          exponent = d[1],
          i = exponent - (prefixExponent$1 = Math.max(-8, Math.min(8, Math.floor(exponent / 3))) * 3) + 1,
          n = coefficient.length;
      return i === n ? coefficient
          : i > n ? coefficient + new Array(i - n + 1).join("0")
          : i > 0 ? coefficient.slice(0, i) + "." + coefficient.slice(i)
          : "0." + new Array(1 - i).join("0") + formatDecimal$1(x, Math.max(0, p + i - 1))[0]; // less than 1y!
    }

    function formatRounded$1(x, p) {
      var d = formatDecimal$1(x, p);
      if (!d) return x + "";
      var coefficient = d[0],
          exponent = d[1];
      return exponent < 0 ? "0." + new Array(-exponent).join("0") + coefficient
          : coefficient.length > exponent + 1 ? coefficient.slice(0, exponent + 1) + "." + coefficient.slice(exponent + 1)
          : coefficient + new Array(exponent - coefficient.length + 2).join("0");
    }

    var formatTypes$1 = {
      "": formatDefault,
      "%": function(x, p) { return (x * 100).toFixed(p); },
      "b": function(x) { return Math.round(x).toString(2); },
      "c": function(x) { return x + ""; },
      "d": function(x) { return Math.round(x).toString(10); },
      "e": function(x, p) { return x.toExponential(p); },
      "f": function(x, p) { return x.toFixed(p); },
      "g": function(x, p) { return x.toPrecision(p); },
      "o": function(x) { return Math.round(x).toString(8); },
      "p": function(x, p) { return formatRounded$1(x * 100, p); },
      "r": formatRounded$1,
      "s": formatPrefixAuto$1,
      "X": function(x) { return Math.round(x).toString(16).toUpperCase(); },
      "x": function(x) { return Math.round(x).toString(16); }
    };

    // [[fill]align][sign][symbol][0][width][,][.precision][type]
    var re$1 = /^(?:(.)?([<>=^]))?([+\-\( ])?([$#])?(0)?(\d+)?(,)?(\.\d+)?([a-z%])?$/i;

    function formatSpecifier$1(specifier) {
      return new FormatSpecifier$1(specifier);
    }

    formatSpecifier$1.prototype = FormatSpecifier$1.prototype; // instanceof

    function FormatSpecifier$1(specifier) {
      if (!(match = re$1.exec(specifier))) throw new Error("invalid format: " + specifier);

      var match,
          fill = match[1] || " ",
          align = match[2] || ">",
          sign = match[3] || "-",
          symbol = match[4] || "",
          zero = !!match[5],
          width = match[6] && +match[6],
          comma = !!match[7],
          precision = match[8] && +match[8].slice(1),
          type = match[9] || "";

      // The "n" type is an alias for ",g".
      if (type === "n") comma = true, type = "g";

      // Map invalid types to the default format.
      else if (!formatTypes$1[type]) type = "";

      // If zero fill is specified, padding goes after sign and before digits.
      if (zero || (fill === "0" && align === "=")) zero = true, fill = "0", align = "=";

      this.fill = fill;
      this.align = align;
      this.sign = sign;
      this.symbol = symbol;
      this.zero = zero;
      this.width = width;
      this.comma = comma;
      this.precision = precision;
      this.type = type;
    }

    FormatSpecifier$1.prototype.toString = function() {
      return this.fill
          + this.align
          + this.sign
          + this.symbol
          + (this.zero ? "0" : "")
          + (this.width == null ? "" : Math.max(1, this.width | 0))
          + (this.comma ? "," : "")
          + (this.precision == null ? "" : "." + Math.max(0, this.precision | 0))
          + this.type;
    };

    function identity$d(x) {
      return x;
    }

    var prefixes$1 = ["y","z","a","f","p","n","µ","m","","k","M","G","T","P","E","Z","Y"];

    function formatLocale$2(locale) {
      var group = locale.grouping && locale.thousands ? formatGroup$1(locale.grouping, locale.thousands) : identity$d,
          currency = locale.currency,
          decimal = locale.decimal,
          numerals = locale.numerals ? formatNumerals$1(locale.numerals) : identity$d,
          percent = locale.percent || "%";

      function newFormat(specifier) {
        specifier = formatSpecifier$1(specifier);

        var fill = specifier.fill,
            align = specifier.align,
            sign = specifier.sign,
            symbol = specifier.symbol,
            zero = specifier.zero,
            width = specifier.width,
            comma = specifier.comma,
            precision = specifier.precision,
            type = specifier.type;

        // Compute the prefix and suffix.
        // For SI-prefix, the suffix is lazily computed.
        var prefix = symbol === "$" ? currency[0] : symbol === "#" && /[boxX]/.test(type) ? "0" + type.toLowerCase() : "",
            suffix = symbol === "$" ? currency[1] : /[%p]/.test(type) ? percent : "";

        // What format function should we use?
        // Is this an integer type?
        // Can this type generate exponential notation?
        var formatType = formatTypes$1[type],
            maybeSuffix = !type || /[defgprs%]/.test(type);

        // Set the default precision if not specified,
        // or clamp the specified precision to the supported range.
        // For significant precision, it must be in [1, 21].
        // For fixed precision, it must be in [0, 20].
        precision = precision == null ? (type ? 6 : 12)
            : /[gprs]/.test(type) ? Math.max(1, Math.min(21, precision))
            : Math.max(0, Math.min(20, precision));

        function format(value) {
          var valuePrefix = prefix,
              valueSuffix = suffix,
              i, n, c;

          if (type === "c") {
            valueSuffix = formatType(value) + valueSuffix;
            value = "";
          } else {
            value = +value;

            // Perform the initial formatting.
            var valueNegative = value < 0;
            value = formatType(Math.abs(value), precision);

            // If a negative value rounds to zero during formatting, treat as positive.
            if (valueNegative && +value === 0) valueNegative = false;

            // Compute the prefix and suffix.
            valuePrefix = (valueNegative ? (sign === "(" ? sign : "-") : sign === "-" || sign === "(" ? "" : sign) + valuePrefix;
            valueSuffix = (type === "s" ? prefixes$1[8 + prefixExponent$1 / 3] : "") + valueSuffix + (valueNegative && sign === "(" ? ")" : "");

            // Break the formatted value into the integer “value” part that can be
            // grouped, and fractional or exponential “suffix” part that is not.
            if (maybeSuffix) {
              i = -1, n = value.length;
              while (++i < n) {
                if (c = value.charCodeAt(i), 48 > c || c > 57) {
                  valueSuffix = (c === 46 ? decimal + value.slice(i + 1) : value.slice(i)) + valueSuffix;
                  value = value.slice(0, i);
                  break;
                }
              }
            }
          }

          // If the fill character is not "0", grouping is applied before padding.
          if (comma && !zero) value = group(value, Infinity);

          // Compute the padding.
          var length = valuePrefix.length + value.length + valueSuffix.length,
              padding = length < width ? new Array(width - length + 1).join(fill) : "";

          // If the fill character is "0", grouping is applied after padding.
          if (comma && zero) value = group(padding + value, padding.length ? width - valueSuffix.length : Infinity), padding = "";

          // Reconstruct the final output based on the desired alignment.
          switch (align) {
            case "<": value = valuePrefix + value + valueSuffix + padding; break;
            case "=": value = valuePrefix + padding + value + valueSuffix; break;
            case "^": value = padding.slice(0, length = padding.length >> 1) + valuePrefix + value + valueSuffix + padding.slice(length); break;
            default: value = padding + valuePrefix + value + valueSuffix; break;
          }

          return numerals(value);
        }

        format.toString = function() {
          return specifier + "";
        };

        return format;
      }

      function formatPrefix(specifier, value) {
        var f = newFormat((specifier = formatSpecifier$1(specifier), specifier.type = "f", specifier)),
            e = Math.max(-8, Math.min(8, Math.floor(exponent$3(value) / 3))) * 3,
            k = Math.pow(10, -e),
            prefix = prefixes$1[8 + e / 3];
        return function(value) {
          return f(k * value) + prefix;
        };
      }

      return {
        format: newFormat,
        formatPrefix: formatPrefix
      };
    }

    var locale$2;
    var format$1;
    var formatPrefix$1;

    defaultLocale$2({
      decimal: ".",
      thousands: ",",
      grouping: [3],
      currency: ["$", ""]
    });

    function defaultLocale$2(definition) {
      locale$2 = formatLocale$2(definition);
      format$1 = locale$2.format;
      formatPrefix$1 = locale$2.formatPrefix;
      return locale$2;
    }

    function precisionFixed$1(step) {
      return Math.max(0, -exponent$3(Math.abs(step)));
    }

    function precisionPrefix$1(step, value) {
      return Math.max(0, Math.max(-8, Math.min(8, Math.floor(exponent$3(value) / 3))) * 3 - exponent$3(Math.abs(step)));
    }

    function precisionRound$1(step, max) {
      step = Math.abs(step), max = Math.abs(max) - step;
      return Math.max(0, exponent$3(max) - exponent$3(step)) + 1;
    }

    // Adds floating point numbers with twice the normal precision.
    // Reference: J. R. Shewchuk, Adaptive Precision Floating-Point Arithmetic and
    // Fast Robust Geometric Predicates, Discrete & Computational Geometry 18(3)
    // 305–363 (1997).
    // Code adapted from GeographicLib by Charles F. F. Karney,
    // http://geographiclib.sourceforge.net/

    function adder$1() {
      return new Adder$1;
    }

    function Adder$1() {
      this.reset();
    }

    Adder$1.prototype = {
      constructor: Adder$1,
      reset: function() {
        this.s = // rounded value
        this.t = 0; // exact error
      },
      add: function(y) {
        add$3(temp$1, y, this.t);
        add$3(this, temp$1.s, this.s);
        if (this.s) this.t += temp$1.t;
        else this.s = temp$1.t;
      },
      valueOf: function() {
        return this.s;
      }
    };

    var temp$1 = new Adder$1;

    function add$3(adder, a, b) {
      var x = adder.s = a + b,
          bv = x - a,
          av = x - bv;
      adder.t = (a - av) + (b - bv);
    }

    var epsilon$7 = 1e-6;
    var epsilon2$3 = 1e-12;
    var pi$8 = Math.PI;
    var halfPi$6 = pi$8 / 2;
    var quarterPi$1 = pi$8 / 4;
    var tau$8 = pi$8 * 2;

    var degrees$3 = 180 / pi$8;
    var radians$1 = pi$8 / 180;

    var abs$2 = Math.abs;
    var atan$1 = Math.atan;
    var atan2$2 = Math.atan2;
    var cos$4 = Math.cos;
    var ceil$1 = Math.ceil;
    var exp$1 = Math.exp;
    var log$2 = Math.log;
    var pow$2 = Math.pow;
    var sin$4 = Math.sin;
    var sign$2 = Math.sign || function(x) { return x > 0 ? 1 : x < 0 ? -1 : 0; };
    var sqrt$3 = Math.sqrt;
    var tan$1 = Math.tan;

    function acos$2(x) {
      return x > 1 ? 0 : x < -1 ? pi$8 : Math.acos(x);
    }

    function asin$2(x) {
      return x > 1 ? halfPi$6 : x < -1 ? -halfPi$6 : Math.asin(x);
    }

    function haversin$1(x) {
      return (x = sin$4(x / 2)) * x;
    }

    function noop$8() {}

    function streamGeometry$1(geometry, stream) {
      if (geometry && streamGeometryType$1.hasOwnProperty(geometry.type)) {
        streamGeometryType$1[geometry.type](geometry, stream);
      }
    }

    var streamObjectType$1 = {
      Feature: function(object, stream) {
        streamGeometry$1(object.geometry, stream);
      },
      FeatureCollection: function(object, stream) {
        var features = object.features, i = -1, n = features.length;
        while (++i < n) streamGeometry$1(features[i].geometry, stream);
      }
    };

    var streamGeometryType$1 = {
      Sphere: function(object, stream) {
        stream.sphere();
      },
      Point: function(object, stream) {
        object = object.coordinates;
        stream.point(object[0], object[1], object[2]);
      },
      MultiPoint: function(object, stream) {
        var coordinates = object.coordinates, i = -1, n = coordinates.length;
        while (++i < n) object = coordinates[i], stream.point(object[0], object[1], object[2]);
      },
      LineString: function(object, stream) {
        streamLine$1(object.coordinates, stream, 0);
      },
      MultiLineString: function(object, stream) {
        var coordinates = object.coordinates, i = -1, n = coordinates.length;
        while (++i < n) streamLine$1(coordinates[i], stream, 0);
      },
      Polygon: function(object, stream) {
        streamPolygon$1(object.coordinates, stream);
      },
      MultiPolygon: function(object, stream) {
        var coordinates = object.coordinates, i = -1, n = coordinates.length;
        while (++i < n) streamPolygon$1(coordinates[i], stream);
      },
      GeometryCollection: function(object, stream) {
        var geometries = object.geometries, i = -1, n = geometries.length;
        while (++i < n) streamGeometry$1(geometries[i], stream);
      }
    };

    function streamLine$1(coordinates, stream, closed) {
      var i = -1, n = coordinates.length - closed, coordinate;
      stream.lineStart();
      while (++i < n) coordinate = coordinates[i], stream.point(coordinate[0], coordinate[1], coordinate[2]);
      stream.lineEnd();
    }

    function streamPolygon$1(coordinates, stream) {
      var i = -1, n = coordinates.length;
      stream.polygonStart();
      while (++i < n) streamLine$1(coordinates[i], stream, 1);
      stream.polygonEnd();
    }

    function geoStream$1(object, stream) {
      if (object && streamObjectType$1.hasOwnProperty(object.type)) {
        streamObjectType$1[object.type](object, stream);
      } else {
        streamGeometry$1(object, stream);
      }
    }

    var areaRingSum$2 = adder$1();

    var areaSum$2 = adder$1(),
        lambda00$3,
        phi00$3,
        lambda0$3,
        cosPhi0$2,
        sinPhi0$2;

    var areaStream$2 = {
      point: noop$8,
      lineStart: noop$8,
      lineEnd: noop$8,
      polygonStart: function() {
        areaRingSum$2.reset();
        areaStream$2.lineStart = areaRingStart$2;
        areaStream$2.lineEnd = areaRingEnd$2;
      },
      polygonEnd: function() {
        var areaRing = +areaRingSum$2;
        areaSum$2.add(areaRing < 0 ? tau$8 + areaRing : areaRing);
        this.lineStart = this.lineEnd = this.point = noop$8;
      },
      sphere: function() {
        areaSum$2.add(tau$8);
      }
    };

    function areaRingStart$2() {
      areaStream$2.point = areaPointFirst$2;
    }

    function areaRingEnd$2() {
      areaPoint$2(lambda00$3, phi00$3);
    }

    function areaPointFirst$2(lambda, phi) {
      areaStream$2.point = areaPoint$2;
      lambda00$3 = lambda, phi00$3 = phi;
      lambda *= radians$1, phi *= radians$1;
      lambda0$3 = lambda, cosPhi0$2 = cos$4(phi = phi / 2 + quarterPi$1), sinPhi0$2 = sin$4(phi);
    }

    function areaPoint$2(lambda, phi) {
      lambda *= radians$1, phi *= radians$1;
      phi = phi / 2 + quarterPi$1; // half the angular distance from south pole

      // Spherical excess E for a spherical triangle with vertices: south pole,
      // previous point, current point.  Uses a formula derived from Cagnoli’s
      // theorem.  See Todhunter, Spherical Trig. (1871), Sec. 103, Eq. (2).
      var dLambda = lambda - lambda0$3,
          sdLambda = dLambda >= 0 ? 1 : -1,
          adLambda = sdLambda * dLambda,
          cosPhi = cos$4(phi),
          sinPhi = sin$4(phi),
          k = sinPhi0$2 * sinPhi,
          u = cosPhi0$2 * cosPhi + k * cos$4(adLambda),
          v = k * sdLambda * sin$4(adLambda);
      areaRingSum$2.add(atan2$2(v, u));

      // Advance the previous points.
      lambda0$3 = lambda, cosPhi0$2 = cosPhi, sinPhi0$2 = sinPhi;
    }

    function area$4(object) {
      areaSum$2.reset();
      geoStream$1(object, areaStream$2);
      return areaSum$2 * 2;
    }

    function spherical$1(cartesian) {
      return [atan2$2(cartesian[1], cartesian[0]), asin$2(cartesian[2])];
    }

    function cartesian$1(spherical) {
      var lambda = spherical[0], phi = spherical[1], cosPhi = cos$4(phi);
      return [cosPhi * cos$4(lambda), cosPhi * sin$4(lambda), sin$4(phi)];
    }

    function cartesianDot$1(a, b) {
      return a[0] * b[0] + a[1] * b[1] + a[2] * b[2];
    }

    function cartesianCross$1(a, b) {
      return [a[1] * b[2] - a[2] * b[1], a[2] * b[0] - a[0] * b[2], a[0] * b[1] - a[1] * b[0]];
    }

    // TODO return a
    function cartesianAddInPlace$1(a, b) {
      a[0] += b[0], a[1] += b[1], a[2] += b[2];
    }

    function cartesianScale$1(vector, k) {
      return [vector[0] * k, vector[1] * k, vector[2] * k];
    }

    // TODO return d
    function cartesianNormalizeInPlace$1(d) {
      var l = sqrt$3(d[0] * d[0] + d[1] * d[1] + d[2] * d[2]);
      d[0] /= l, d[1] /= l, d[2] /= l;
    }

    var lambda0$4, phi0$1, lambda1$1, phi1$1, // bounds
        lambda2$1, // previous lambda-coordinate
        lambda00$4, phi00$4, // first point
        p0$1, // previous 3D point
        deltaSum$1 = adder$1(),
        ranges$1,
        range$3;

    var boundsStream$2 = {
      point: boundsPoint$2,
      lineStart: boundsLineStart$1,
      lineEnd: boundsLineEnd$1,
      polygonStart: function() {
        boundsStream$2.point = boundsRingPoint$1;
        boundsStream$2.lineStart = boundsRingStart$1;
        boundsStream$2.lineEnd = boundsRingEnd$1;
        deltaSum$1.reset();
        areaStream$2.polygonStart();
      },
      polygonEnd: function() {
        areaStream$2.polygonEnd();
        boundsStream$2.point = boundsPoint$2;
        boundsStream$2.lineStart = boundsLineStart$1;
        boundsStream$2.lineEnd = boundsLineEnd$1;
        if (areaRingSum$2 < 0) lambda0$4 = -(lambda1$1 = 180), phi0$1 = -(phi1$1 = 90);
        else if (deltaSum$1 > epsilon$7) phi1$1 = 90;
        else if (deltaSum$1 < -epsilon$7) phi0$1 = -90;
        range$3[0] = lambda0$4, range$3[1] = lambda1$1;
      }
    };

    function boundsPoint$2(lambda, phi) {
      ranges$1.push(range$3 = [lambda0$4 = lambda, lambda1$1 = lambda]);
      if (phi < phi0$1) phi0$1 = phi;
      if (phi > phi1$1) phi1$1 = phi;
    }

    function linePoint$1(lambda, phi) {
      var p = cartesian$1([lambda * radians$1, phi * radians$1]);
      if (p0$1) {
        var normal = cartesianCross$1(p0$1, p),
            equatorial = [normal[1], -normal[0], 0],
            inflection = cartesianCross$1(equatorial, normal);
        cartesianNormalizeInPlace$1(inflection);
        inflection = spherical$1(inflection);
        var delta = lambda - lambda2$1,
            sign = delta > 0 ? 1 : -1,
            lambdai = inflection[0] * degrees$3 * sign,
            phii,
            antimeridian = abs$2(delta) > 180;
        if (antimeridian ^ (sign * lambda2$1 < lambdai && lambdai < sign * lambda)) {
          phii = inflection[1] * degrees$3;
          if (phii > phi1$1) phi1$1 = phii;
        } else if (lambdai = (lambdai + 360) % 360 - 180, antimeridian ^ (sign * lambda2$1 < lambdai && lambdai < sign * lambda)) {
          phii = -inflection[1] * degrees$3;
          if (phii < phi0$1) phi0$1 = phii;
        } else {
          if (phi < phi0$1) phi0$1 = phi;
          if (phi > phi1$1) phi1$1 = phi;
        }
        if (antimeridian) {
          if (lambda < lambda2$1) {
            if (angle$1(lambda0$4, lambda) > angle$1(lambda0$4, lambda1$1)) lambda1$1 = lambda;
          } else {
            if (angle$1(lambda, lambda1$1) > angle$1(lambda0$4, lambda1$1)) lambda0$4 = lambda;
          }
        } else {
          if (lambda1$1 >= lambda0$4) {
            if (lambda < lambda0$4) lambda0$4 = lambda;
            if (lambda > lambda1$1) lambda1$1 = lambda;
          } else {
            if (lambda > lambda2$1) {
              if (angle$1(lambda0$4, lambda) > angle$1(lambda0$4, lambda1$1)) lambda1$1 = lambda;
            } else {
              if (angle$1(lambda, lambda1$1) > angle$1(lambda0$4, lambda1$1)) lambda0$4 = lambda;
            }
          }
        }
      } else {
        ranges$1.push(range$3 = [lambda0$4 = lambda, lambda1$1 = lambda]);
      }
      if (phi < phi0$1) phi0$1 = phi;
      if (phi > phi1$1) phi1$1 = phi;
      p0$1 = p, lambda2$1 = lambda;
    }

    function boundsLineStart$1() {
      boundsStream$2.point = linePoint$1;
    }

    function boundsLineEnd$1() {
      range$3[0] = lambda0$4, range$3[1] = lambda1$1;
      boundsStream$2.point = boundsPoint$2;
      p0$1 = null;
    }

    function boundsRingPoint$1(lambda, phi) {
      if (p0$1) {
        var delta = lambda - lambda2$1;
        deltaSum$1.add(abs$2(delta) > 180 ? delta + (delta > 0 ? 360 : -360) : delta);
      } else {
        lambda00$4 = lambda, phi00$4 = phi;
      }
      areaStream$2.point(lambda, phi);
      linePoint$1(lambda, phi);
    }

    function boundsRingStart$1() {
      areaStream$2.lineStart();
    }

    function boundsRingEnd$1() {
      boundsRingPoint$1(lambda00$4, phi00$4);
      areaStream$2.lineEnd();
      if (abs$2(deltaSum$1) > epsilon$7) lambda0$4 = -(lambda1$1 = 180);
      range$3[0] = lambda0$4, range$3[1] = lambda1$1;
      p0$1 = null;
    }

    // Finds the left-right distance between two longitudes.
    // This is almost the same as (lambda1 - lambda0 + 360°) % 360°, except that we want
    // the distance between ±180° to be 360°.
    function angle$1(lambda0, lambda1) {
      return (lambda1 -= lambda0) < 0 ? lambda1 + 360 : lambda1;
    }

    function rangeCompare$1(a, b) {
      return a[0] - b[0];
    }

    function rangeContains$1(range, x) {
      return range[0] <= range[1] ? range[0] <= x && x <= range[1] : x < range[0] || range[1] < x;
    }

    function bounds$1(feature) {
      var i, n, a, b, merged, deltaMax, delta;

      phi1$1 = lambda1$1 = -(lambda0$4 = phi0$1 = Infinity);
      ranges$1 = [];
      geoStream$1(feature, boundsStream$2);

      // First, sort ranges by their minimum longitudes.
      if (n = ranges$1.length) {
        ranges$1.sort(rangeCompare$1);

        // Then, merge any ranges that overlap.
        for (i = 1, a = ranges$1[0], merged = [a]; i < n; ++i) {
          b = ranges$1[i];
          if (rangeContains$1(a, b[0]) || rangeContains$1(a, b[1])) {
            if (angle$1(a[0], b[1]) > angle$1(a[0], a[1])) a[1] = b[1];
            if (angle$1(b[0], a[1]) > angle$1(a[0], a[1])) a[0] = b[0];
          } else {
            merged.push(a = b);
          }
        }

        // Finally, find the largest gap between the merged ranges.
        // The final bounding box will be the inverse of this gap.
        for (deltaMax = -Infinity, n = merged.length - 1, i = 0, a = merged[n]; i <= n; a = b, ++i) {
          b = merged[i];
          if ((delta = angle$1(a[1], b[0])) > deltaMax) deltaMax = delta, lambda0$4 = b[0], lambda1$1 = a[1];
        }
      }

      ranges$1 = range$3 = null;

      return lambda0$4 === Infinity || phi0$1 === Infinity
          ? [[NaN, NaN], [NaN, NaN]]
          : [[lambda0$4, phi0$1], [lambda1$1, phi1$1]];
    }

    var W0$1, W1$1,
        X0$2, Y0$2, Z0$2,
        X1$2, Y1$2, Z1$2,
        X2$2, Y2$2, Z2$2,
        lambda00$5, phi00$5, // first point
        x0$5, y0$5, z0$1; // previous point

    var centroidStream$2 = {
      sphere: noop$8,
      point: centroidPoint$2,
      lineStart: centroidLineStart$2,
      lineEnd: centroidLineEnd$2,
      polygonStart: function() {
        centroidStream$2.lineStart = centroidRingStart$2;
        centroidStream$2.lineEnd = centroidRingEnd$2;
      },
      polygonEnd: function() {
        centroidStream$2.lineStart = centroidLineStart$2;
        centroidStream$2.lineEnd = centroidLineEnd$2;
      }
    };

    // Arithmetic mean of Cartesian vectors.
    function centroidPoint$2(lambda, phi) {
      lambda *= radians$1, phi *= radians$1;
      var cosPhi = cos$4(phi);
      centroidPointCartesian$1(cosPhi * cos$4(lambda), cosPhi * sin$4(lambda), sin$4(phi));
    }

    function centroidPointCartesian$1(x, y, z) {
      ++W0$1;
      X0$2 += (x - X0$2) / W0$1;
      Y0$2 += (y - Y0$2) / W0$1;
      Z0$2 += (z - Z0$2) / W0$1;
    }

    function centroidLineStart$2() {
      centroidStream$2.point = centroidLinePointFirst$1;
    }

    function centroidLinePointFirst$1(lambda, phi) {
      lambda *= radians$1, phi *= radians$1;
      var cosPhi = cos$4(phi);
      x0$5 = cosPhi * cos$4(lambda);
      y0$5 = cosPhi * sin$4(lambda);
      z0$1 = sin$4(phi);
      centroidStream$2.point = centroidLinePoint$1;
      centroidPointCartesian$1(x0$5, y0$5, z0$1);
    }

    function centroidLinePoint$1(lambda, phi) {
      lambda *= radians$1, phi *= radians$1;
      var cosPhi = cos$4(phi),
          x = cosPhi * cos$4(lambda),
          y = cosPhi * sin$4(lambda),
          z = sin$4(phi),
          w = atan2$2(sqrt$3((w = y0$5 * z - z0$1 * y) * w + (w = z0$1 * x - x0$5 * z) * w + (w = x0$5 * y - y0$5 * x) * w), x0$5 * x + y0$5 * y + z0$1 * z);
      W1$1 += w;
      X1$2 += w * (x0$5 + (x0$5 = x));
      Y1$2 += w * (y0$5 + (y0$5 = y));
      Z1$2 += w * (z0$1 + (z0$1 = z));
      centroidPointCartesian$1(x0$5, y0$5, z0$1);
    }

    function centroidLineEnd$2() {
      centroidStream$2.point = centroidPoint$2;
    }

    // See J. E. Brock, The Inertia Tensor for a Spherical Triangle,
    // J. Applied Mechanics 42, 239 (1975).
    function centroidRingStart$2() {
      centroidStream$2.point = centroidRingPointFirst$1;
    }

    function centroidRingEnd$2() {
      centroidRingPoint$1(lambda00$5, phi00$5);
      centroidStream$2.point = centroidPoint$2;
    }

    function centroidRingPointFirst$1(lambda, phi) {
      lambda00$5 = lambda, phi00$5 = phi;
      lambda *= radians$1, phi *= radians$1;
      centroidStream$2.point = centroidRingPoint$1;
      var cosPhi = cos$4(phi);
      x0$5 = cosPhi * cos$4(lambda);
      y0$5 = cosPhi * sin$4(lambda);
      z0$1 = sin$4(phi);
      centroidPointCartesian$1(x0$5, y0$5, z0$1);
    }

    function centroidRingPoint$1(lambda, phi) {
      lambda *= radians$1, phi *= radians$1;
      var cosPhi = cos$4(phi),
          x = cosPhi * cos$4(lambda),
          y = cosPhi * sin$4(lambda),
          z = sin$4(phi),
          cx = y0$5 * z - z0$1 * y,
          cy = z0$1 * x - x0$5 * z,
          cz = x0$5 * y - y0$5 * x,
          m = sqrt$3(cx * cx + cy * cy + cz * cz),
          w = asin$2(m), // line weight = angle
          v = m && -w / m; // area weight multiplier
      X2$2 += v * cx;
      Y2$2 += v * cy;
      Z2$2 += v * cz;
      W1$1 += w;
      X1$2 += w * (x0$5 + (x0$5 = x));
      Y1$2 += w * (y0$5 + (y0$5 = y));
      Z1$2 += w * (z0$1 + (z0$1 = z));
      centroidPointCartesian$1(x0$5, y0$5, z0$1);
    }

    function centroid$2(object) {
      W0$1 = W1$1 =
      X0$2 = Y0$2 = Z0$2 =
      X1$2 = Y1$2 = Z1$2 =
      X2$2 = Y2$2 = Z2$2 = 0;
      geoStream$1(object, centroidStream$2);

      var x = X2$2,
          y = Y2$2,
          z = Z2$2,
          m = x * x + y * y + z * z;

      // If the area-weighted ccentroid is undefined, fall back to length-weighted ccentroid.
      if (m < epsilon2$3) {
        x = X1$2, y = Y1$2, z = Z1$2;
        // If the feature has zero length, fall back to arithmetic mean of point vectors.
        if (W1$1 < epsilon$7) x = X0$2, y = Y0$2, z = Z0$2;
        m = x * x + y * y + z * z;
        // If the feature still has an undefined ccentroid, then return.
        if (m < epsilon2$3) return [NaN, NaN];
      }

      return [atan2$2(y, x) * degrees$3, asin$2(z / sqrt$3(m)) * degrees$3];
    }

    function constant$m(x) {
      return function() {
        return x;
      };
    }

    function compose$1(a, b) {

      function compose(x, y) {
        return x = a(x, y), b(x[0], x[1]);
      }

      if (a.invert && b.invert) compose.invert = function(x, y) {
        return x = b.invert(x, y), x && a.invert(x[0], x[1]);
      };

      return compose;
    }

    function rotationIdentity$1(lambda, phi) {
      return [lambda > pi$8 ? lambda - tau$8 : lambda < -pi$8 ? lambda + tau$8 : lambda, phi];
    }

    rotationIdentity$1.invert = rotationIdentity$1;

    function rotateRadians$1(deltaLambda, deltaPhi, deltaGamma) {
      return (deltaLambda %= tau$8) ? (deltaPhi || deltaGamma ? compose$1(rotationLambda$1(deltaLambda), rotationPhiGamma$1(deltaPhi, deltaGamma))
        : rotationLambda$1(deltaLambda))
        : (deltaPhi || deltaGamma ? rotationPhiGamma$1(deltaPhi, deltaGamma)
        : rotationIdentity$1);
    }

    function forwardRotationLambda$1(deltaLambda) {
      return function(lambda, phi) {
        return lambda += deltaLambda, [lambda > pi$8 ? lambda - tau$8 : lambda < -pi$8 ? lambda + tau$8 : lambda, phi];
      };
    }

    function rotationLambda$1(deltaLambda) {
      var rotation = forwardRotationLambda$1(deltaLambda);
      rotation.invert = forwardRotationLambda$1(-deltaLambda);
      return rotation;
    }

    function rotationPhiGamma$1(deltaPhi, deltaGamma) {
      var cosDeltaPhi = cos$4(deltaPhi),
          sinDeltaPhi = sin$4(deltaPhi),
          cosDeltaGamma = cos$4(deltaGamma),
          sinDeltaGamma = sin$4(deltaGamma);

      function rotation(lambda, phi) {
        var cosPhi = cos$4(phi),
            x = cos$4(lambda) * cosPhi,
            y = sin$4(lambda) * cosPhi,
            z = sin$4(phi),
            k = z * cosDeltaPhi + x * sinDeltaPhi;
        return [
          atan2$2(y * cosDeltaGamma - k * sinDeltaGamma, x * cosDeltaPhi - z * sinDeltaPhi),
          asin$2(k * cosDeltaGamma + y * sinDeltaGamma)
        ];
      }

      rotation.invert = function(lambda, phi) {
        var cosPhi = cos$4(phi),
            x = cos$4(lambda) * cosPhi,
            y = sin$4(lambda) * cosPhi,
            z = sin$4(phi),
            k = z * cosDeltaGamma - y * sinDeltaGamma;
        return [
          atan2$2(y * cosDeltaGamma + z * sinDeltaGamma, x * cosDeltaPhi + k * sinDeltaPhi),
          asin$2(k * cosDeltaPhi - x * sinDeltaPhi)
        ];
      };

      return rotation;
    }

    function rotation$1(rotate) {
      rotate = rotateRadians$1(rotate[0] * radians$1, rotate[1] * radians$1, rotate.length > 2 ? rotate[2] * radians$1 : 0);

      function forward(coordinates) {
        coordinates = rotate(coordinates[0] * radians$1, coordinates[1] * radians$1);
        return coordinates[0] *= degrees$3, coordinates[1] *= degrees$3, coordinates;
      }

      forward.invert = function(coordinates) {
        coordinates = rotate.invert(coordinates[0] * radians$1, coordinates[1] * radians$1);
        return coordinates[0] *= degrees$3, coordinates[1] *= degrees$3, coordinates;
      };

      return forward;
    }

    // Generates a circle centered at [0°, 0°], with a given radius and precision.
    function circleStream$1(stream, radius, delta, direction, t0, t1) {
      if (!delta) return;
      var cosRadius = cos$4(radius),
          sinRadius = sin$4(radius),
          step = direction * delta;
      if (t0 == null) {
        t0 = radius + direction * tau$8;
        t1 = radius - step / 2;
      } else {
        t0 = circleRadius$1(cosRadius, t0);
        t1 = circleRadius$1(cosRadius, t1);
        if (direction > 0 ? t0 < t1 : t0 > t1) t0 += direction * tau$8;
      }
      for (var point, t = t0; direction > 0 ? t > t1 : t < t1; t -= step) {
        point = spherical$1([cosRadius, -sinRadius * cos$4(t), -sinRadius * sin$4(t)]);
        stream.point(point[0], point[1]);
      }
    }

    // Returns the signed angle of a cartesian point relative to [cosRadius, 0, 0].
    function circleRadius$1(cosRadius, point) {
      point = cartesian$1(point), point[0] -= cosRadius;
      cartesianNormalizeInPlace$1(point);
      var radius = acos$2(-point[1]);
      return ((-point[2] < 0 ? -radius : radius) + tau$8 - epsilon$7) % tau$8;
    }

    function circle$3() {
      var center = constant$m([0, 0]),
          radius = constant$m(90),
          precision = constant$m(6),
          ring,
          rotate,
          stream = {point: point};

      function point(x, y) {
        ring.push(x = rotate(x, y));
        x[0] *= degrees$3, x[1] *= degrees$3;
      }

      function circle() {
        var c = center.apply(this, arguments),
            r = radius.apply(this, arguments) * radians$1,
            p = precision.apply(this, arguments) * radians$1;
        ring = [];
        rotate = rotateRadians$1(-c[0] * radians$1, -c[1] * radians$1, 0).invert;
        circleStream$1(stream, r, p, 1);
        c = {type: "Polygon", coordinates: [ring]};
        ring = rotate = null;
        return c;
      }

      circle.center = function(_) {
        return arguments.length ? (center = typeof _ === "function" ? _ : constant$m([+_[0], +_[1]]), circle) : center;
      };

      circle.radius = function(_) {
        return arguments.length ? (radius = typeof _ === "function" ? _ : constant$m(+_), circle) : radius;
      };

      circle.precision = function(_) {
        return arguments.length ? (precision = typeof _ === "function" ? _ : constant$m(+_), circle) : precision;
      };

      return circle;
    }

    function clipBuffer$1() {
      var lines = [],
          line;
      return {
        point: function(x, y) {
          line.push([x, y]);
        },
        lineStart: function() {
          lines.push(line = []);
        },
        lineEnd: noop$8,
        rejoin: function() {
          if (lines.length > 1) lines.push(lines.pop().concat(lines.shift()));
        },
        result: function() {
          var result = lines;
          lines = [];
          line = null;
          return result;
        }
      };
    }

    function pointEqual$1(a, b) {
      return abs$2(a[0] - b[0]) < epsilon$7 && abs$2(a[1] - b[1]) < epsilon$7;
    }

    function Intersection$1(point, points, other, entry) {
      this.x = point;
      this.z = points;
      this.o = other; // another intersection
      this.e = entry; // is an entry?
      this.v = false; // visited
      this.n = this.p = null; // next & previous
    }

    // A generalized polygon clipping algorithm: given a polygon that has been cut
    // into its visible line segments, and rejoins the segments by interpolating
    // along the clip edge.
    function clipRejoin$1(segments, compareIntersection, startInside, interpolate, stream) {
      var subject = [],
          clip = [],
          i,
          n;

      segments.forEach(function(segment) {
        if ((n = segment.length - 1) <= 0) return;
        var n, p0 = segment[0], p1 = segment[n], x;

        // If the first and last points of a segment are coincident, then treat as a
        // closed ring. TODO if all rings are closed, then the winding order of the
        // exterior ring should be checked.
        if (pointEqual$1(p0, p1)) {
          stream.lineStart();
          for (i = 0; i < n; ++i) stream.point((p0 = segment[i])[0], p0[1]);
          stream.lineEnd();
          return;
        }

        subject.push(x = new Intersection$1(p0, segment, null, true));
        clip.push(x.o = new Intersection$1(p0, null, x, false));
        subject.push(x = new Intersection$1(p1, segment, null, false));
        clip.push(x.o = new Intersection$1(p1, null, x, true));
      });

      if (!subject.length) return;

      clip.sort(compareIntersection);
      link$4(subject);
      link$4(clip);

      for (i = 0, n = clip.length; i < n; ++i) {
        clip[i].e = startInside = !startInside;
      }

      var start = subject[0],
          points,
          point;

      while (1) {
        // Find first unvisited intersection.
        var current = start,
            isSubject = true;
        while (current.v) if ((current = current.n) === start) return;
        points = current.z;
        stream.lineStart();
        do {
          current.v = current.o.v = true;
          if (current.e) {
            if (isSubject) {
              for (i = 0, n = points.length; i < n; ++i) stream.point((point = points[i])[0], point[1]);
            } else {
              interpolate(current.x, current.n.x, 1, stream);
            }
            current = current.n;
          } else {
            if (isSubject) {
              points = current.p.z;
              for (i = points.length - 1; i >= 0; --i) stream.point((point = points[i])[0], point[1]);
            } else {
              interpolate(current.x, current.p.x, -1, stream);
            }
            current = current.p;
          }
          current = current.o;
          points = current.z;
          isSubject = !isSubject;
        } while (!current.v);
        stream.lineEnd();
      }
    }

    function link$4(array) {
      if (!(n = array.length)) return;
      var n,
          i = 0,
          a = array[0],
          b;
      while (++i < n) {
        a.n = b = array[i];
        b.p = a;
        a = b;
      }
      a.n = b = array[0];
      b.p = a;
    }

    var sum$4 = adder$1();

    function polygonContains$1(polygon, point) {
      var lambda = point[0],
          phi = point[1],
          normal = [sin$4(lambda), -cos$4(lambda), 0],
          angle = 0,
          winding = 0;

      sum$4.reset();

      for (var i = 0, n = polygon.length; i < n; ++i) {
        if (!(m = (ring = polygon[i]).length)) continue;
        var ring,
            m,
            point0 = ring[m - 1],
            lambda0 = point0[0],
            phi0 = point0[1] / 2 + quarterPi$1,
            sinPhi0 = sin$4(phi0),
            cosPhi0 = cos$4(phi0);

        for (var j = 0; j < m; ++j, lambda0 = lambda1, sinPhi0 = sinPhi1, cosPhi0 = cosPhi1, point0 = point1) {
          var point1 = ring[j],
              lambda1 = point1[0],
              phi1 = point1[1] / 2 + quarterPi$1,
              sinPhi1 = sin$4(phi1),
              cosPhi1 = cos$4(phi1),
              delta = lambda1 - lambda0,
              sign = delta >= 0 ? 1 : -1,
              absDelta = sign * delta,
              antimeridian = absDelta > pi$8,
              k = sinPhi0 * sinPhi1;

          sum$4.add(atan2$2(k * sign * sin$4(absDelta), cosPhi0 * cosPhi1 + k * cos$4(absDelta)));
          angle += antimeridian ? delta + sign * tau$8 : delta;

          // Are the longitudes either side of the point’s meridian (lambda),
          // and are the latitudes smaller than the parallel (phi)?
          if (antimeridian ^ lambda0 >= lambda ^ lambda1 >= lambda) {
            var arc = cartesianCross$1(cartesian$1(point0), cartesian$1(point1));
            cartesianNormalizeInPlace$1(arc);
            var intersection = cartesianCross$1(normal, arc);
            cartesianNormalizeInPlace$1(intersection);
            var phiArc = (antimeridian ^ delta >= 0 ? -1 : 1) * asin$2(intersection[2]);
            if (phi > phiArc || phi === phiArc && (arc[0] || arc[1])) {
              winding += antimeridian ^ delta >= 0 ? 1 : -1;
            }
          }
        }
      }

      // First, determine whether the South pole is inside or outside:
      //
      // It is inside if:
      // * the polygon winds around it in a clockwise direction.
      // * the polygon does not (cumulatively) wind around it, but has a negative
      //   (counter-clockwise) area.
      //
      // Second, count the (signed) number of times a segment crosses a lambda
      // from the point to the South pole.  If it is zero, then the point is the
      // same side as the South pole.

      return (angle < -epsilon$7 || angle < epsilon$7 && sum$4 < -epsilon$7) ^ (winding & 1);
    }

    function ascending$8(a, b) {
      return a < b ? -1 : a > b ? 1 : a >= b ? 0 : NaN;
    }

    function bisector$3(compare) {
      if (compare.length === 1) compare = ascendingComparator$3(compare);
      return {
        left: function(a, x, lo, hi) {
          if (lo == null) lo = 0;
          if (hi == null) hi = a.length;
          while (lo < hi) {
            var mid = lo + hi >>> 1;
            if (compare(a[mid], x) < 0) lo = mid + 1;
            else hi = mid;
          }
          return lo;
        },
        right: function(a, x, lo, hi) {
          if (lo == null) lo = 0;
          if (hi == null) hi = a.length;
          while (lo < hi) {
            var mid = lo + hi >>> 1;
            if (compare(a[mid], x) > 0) hi = mid;
            else lo = mid + 1;
          }
          return lo;
        }
      };
    }

    function ascendingComparator$3(f) {
      return function(d, x) {
        return ascending$8(f(d), x);
      };
    }

    var ascendingBisect$3 = bisector$3(ascending$8);

    function range$4(start, stop, step) {
      start = +start, stop = +stop, step = (n = arguments.length) < 2 ? (stop = start, start = 0, 1) : n < 3 ? 1 : +step;

      var i = -1,
          n = Math.max(0, Math.ceil((stop - start) / step)) | 0,
          range = new Array(n);

      while (++i < n) {
        range[i] = start + i * step;
      }

      return range;
    }

    function merge$2(arrays) {
      var n = arrays.length,
          m,
          i = -1,
          j = 0,
          merged,
          array;

      while (++i < n) j += arrays[i].length;
      merged = new Array(j);

      while (--n >= 0) {
        array = arrays[n];
        m = array.length;
        while (--m >= 0) {
          merged[--j] = array[m];
        }
      }

      return merged;
    }

    function clip$1(pointVisible, clipLine, interpolate, start) {
      return function(sink) {
        var line = clipLine(sink),
            ringBuffer = clipBuffer$1(),
            ringSink = clipLine(ringBuffer),
            polygonStarted = false,
            polygon,
            segments,
            ring;

        var clip = {
          point: point,
          lineStart: lineStart,
          lineEnd: lineEnd,
          polygonStart: function() {
            clip.point = pointRing;
            clip.lineStart = ringStart;
            clip.lineEnd = ringEnd;
            segments = [];
            polygon = [];
          },
          polygonEnd: function() {
            clip.point = point;
            clip.lineStart = lineStart;
            clip.lineEnd = lineEnd;
            segments = merge$2(segments);
            var startInside = polygonContains$1(polygon, start);
            if (segments.length) {
              if (!polygonStarted) sink.polygonStart(), polygonStarted = true;
              clipRejoin$1(segments, compareIntersection$1, startInside, interpolate, sink);
            } else if (startInside) {
              if (!polygonStarted) sink.polygonStart(), polygonStarted = true;
              sink.lineStart();
              interpolate(null, null, 1, sink);
              sink.lineEnd();
            }
            if (polygonStarted) sink.polygonEnd(), polygonStarted = false;
            segments = polygon = null;
          },
          sphere: function() {
            sink.polygonStart();
            sink.lineStart();
            interpolate(null, null, 1, sink);
            sink.lineEnd();
            sink.polygonEnd();
          }
        };

        function point(lambda, phi) {
          if (pointVisible(lambda, phi)) sink.point(lambda, phi);
        }

        function pointLine(lambda, phi) {
          line.point(lambda, phi);
        }

        function lineStart() {
          clip.point = pointLine;
          line.lineStart();
        }

        function lineEnd() {
          clip.point = point;
          line.lineEnd();
        }

        function pointRing(lambda, phi) {
          ring.push([lambda, phi]);
          ringSink.point(lambda, phi);
        }

        function ringStart() {
          ringSink.lineStart();
          ring = [];
        }

        function ringEnd() {
          pointRing(ring[0][0], ring[0][1]);
          ringSink.lineEnd();

          var clean = ringSink.clean(),
              ringSegments = ringBuffer.result(),
              i, n = ringSegments.length, m,
              segment,
              point;

          ring.pop();
          polygon.push(ring);
          ring = null;

          if (!n) return;

          // No intersections.
          if (clean & 1) {
            segment = ringSegments[0];
            if ((m = segment.length - 1) > 0) {
              if (!polygonStarted) sink.polygonStart(), polygonStarted = true;
              sink.lineStart();
              for (i = 0; i < m; ++i) sink.point((point = segment[i])[0], point[1]);
              sink.lineEnd();
            }
            return;
          }

          // Rejoin connected segments.
          // TODO reuse ringBuffer.rejoin()?
          if (n > 1 && clean & 2) ringSegments.push(ringSegments.pop().concat(ringSegments.shift()));

          segments.push(ringSegments.filter(validSegment$1));
        }

        return clip;
      };
    }

    function validSegment$1(segment) {
      return segment.length > 1;
    }

    // Intersections are sorted along the clip edge. For both antimeridian cutting
    // and circle clipping, the same comparison is used.
    function compareIntersection$1(a, b) {
      return ((a = a.x)[0] < 0 ? a[1] - halfPi$6 - epsilon$7 : halfPi$6 - a[1])
           - ((b = b.x)[0] < 0 ? b[1] - halfPi$6 - epsilon$7 : halfPi$6 - b[1]);
    }

    var clipAntimeridian$1 = clip$1(
      function() { return true; },
      clipAntimeridianLine$1,
      clipAntimeridianInterpolate$1,
      [-pi$8, -halfPi$6]
    );

    // Takes a line and cuts into visible segments. Return values: 0 - there were
    // intersections or the line was empty; 1 - no intersections; 2 - there were
    // intersections, and the first and last segments should be rejoined.
    function clipAntimeridianLine$1(stream) {
      var lambda0 = NaN,
          phi0 = NaN,
          sign0 = NaN,
          clean; // no intersections

      return {
        lineStart: function() {
          stream.lineStart();
          clean = 1;
        },
        point: function(lambda1, phi1) {
          var sign1 = lambda1 > 0 ? pi$8 : -pi$8,
              delta = abs$2(lambda1 - lambda0);
          if (abs$2(delta - pi$8) < epsilon$7) { // line crosses a pole
            stream.point(lambda0, phi0 = (phi0 + phi1) / 2 > 0 ? halfPi$6 : -halfPi$6);
            stream.point(sign0, phi0);
            stream.lineEnd();
            stream.lineStart();
            stream.point(sign1, phi0);
            stream.point(lambda1, phi0);
            clean = 0;
          } else if (sign0 !== sign1 && delta >= pi$8) { // line crosses antimeridian
            if (abs$2(lambda0 - sign0) < epsilon$7) lambda0 -= sign0 * epsilon$7; // handle degeneracies
            if (abs$2(lambda1 - sign1) < epsilon$7) lambda1 -= sign1 * epsilon$7;
            phi0 = clipAntimeridianIntersect$1(lambda0, phi0, lambda1, phi1);
            stream.point(sign0, phi0);
            stream.lineEnd();
            stream.lineStart();
            stream.point(sign1, phi0);
            clean = 0;
          }
          stream.point(lambda0 = lambda1, phi0 = phi1);
          sign0 = sign1;
        },
        lineEnd: function() {
          stream.lineEnd();
          lambda0 = phi0 = NaN;
        },
        clean: function() {
          return 2 - clean; // if intersections, rejoin first and last segments
        }
      };
    }

    function clipAntimeridianIntersect$1(lambda0, phi0, lambda1, phi1) {
      var cosPhi0,
          cosPhi1,
          sinLambda0Lambda1 = sin$4(lambda0 - lambda1);
      return abs$2(sinLambda0Lambda1) > epsilon$7
          ? atan$1((sin$4(phi0) * (cosPhi1 = cos$4(phi1)) * sin$4(lambda1)
              - sin$4(phi1) * (cosPhi0 = cos$4(phi0)) * sin$4(lambda0))
              / (cosPhi0 * cosPhi1 * sinLambda0Lambda1))
          : (phi0 + phi1) / 2;
    }

    function clipAntimeridianInterpolate$1(from, to, direction, stream) {
      var phi;
      if (from == null) {
        phi = direction * halfPi$6;
        stream.point(-pi$8, phi);
        stream.point(0, phi);
        stream.point(pi$8, phi);
        stream.point(pi$8, 0);
        stream.point(pi$8, -phi);
        stream.point(0, -phi);
        stream.point(-pi$8, -phi);
        stream.point(-pi$8, 0);
        stream.point(-pi$8, phi);
      } else if (abs$2(from[0] - to[0]) > epsilon$7) {
        var lambda = from[0] < to[0] ? pi$8 : -pi$8;
        phi = direction * lambda / 2;
        stream.point(-lambda, phi);
        stream.point(0, phi);
        stream.point(lambda, phi);
      } else {
        stream.point(to[0], to[1]);
      }
    }

    function clipCircle$1(radius) {
      var cr = cos$4(radius),
          delta = 6 * radians$1,
          smallRadius = cr > 0,
          notHemisphere = abs$2(cr) > epsilon$7; // TODO optimise for this common case

      function interpolate(from, to, direction, stream) {
        circleStream$1(stream, radius, delta, direction, from, to);
      }

      function visible(lambda, phi) {
        return cos$4(lambda) * cos$4(phi) > cr;
      }

      // Takes a line and cuts into visible segments. Return values used for polygon
      // clipping: 0 - there were intersections or the line was empty; 1 - no
      // intersections 2 - there were intersections, and the first and last segments
      // should be rejoined.
      function clipLine(stream) {
        var point0, // previous point
            c0, // code for previous point
            v0, // visibility of previous point
            v00, // visibility of first point
            clean; // no intersections
        return {
          lineStart: function() {
            v00 = v0 = false;
            clean = 1;
          },
          point: function(lambda, phi) {
            var point1 = [lambda, phi],
                point2,
                v = visible(lambda, phi),
                c = smallRadius
                  ? v ? 0 : code(lambda, phi)
                  : v ? code(lambda + (lambda < 0 ? pi$8 : -pi$8), phi) : 0;
            if (!point0 && (v00 = v0 = v)) stream.lineStart();
            // Handle degeneracies.
            // TODO ignore if not clipping polygons.
            if (v !== v0) {
              point2 = intersect(point0, point1);
              if (!point2 || pointEqual$1(point0, point2) || pointEqual$1(point1, point2)) {
                point1[0] += epsilon$7;
                point1[1] += epsilon$7;
                v = visible(point1[0], point1[1]);
              }
            }
            if (v !== v0) {
              clean = 0;
              if (v) {
                // outside going in
                stream.lineStart();
                point2 = intersect(point1, point0);
                stream.point(point2[0], point2[1]);
              } else {
                // inside going out
                point2 = intersect(point0, point1);
                stream.point(point2[0], point2[1]);
                stream.lineEnd();
              }
              point0 = point2;
            } else if (notHemisphere && point0 && smallRadius ^ v) {
              var t;
              // If the codes for two points are different, or are both zero,
              // and there this segment intersects with the small circle.
              if (!(c & c0) && (t = intersect(point1, point0, true))) {
                clean = 0;
                if (smallRadius) {
                  stream.lineStart();
                  stream.point(t[0][0], t[0][1]);
                  stream.point(t[1][0], t[1][1]);
                  stream.lineEnd();
                } else {
                  stream.point(t[1][0], t[1][1]);
                  stream.lineEnd();
                  stream.lineStart();
                  stream.point(t[0][0], t[0][1]);
                }
              }
            }
            if (v && (!point0 || !pointEqual$1(point0, point1))) {
              stream.point(point1[0], point1[1]);
            }
            point0 = point1, v0 = v, c0 = c;
          },
          lineEnd: function() {
            if (v0) stream.lineEnd();
            point0 = null;
          },
          // Rejoin first and last segments if there were intersections and the first
          // and last points were visible.
          clean: function() {
            return clean | ((v00 && v0) << 1);
          }
        };
      }

      // Intersects the great circle between a and b with the clip circle.
      function intersect(a, b, two) {
        var pa = cartesian$1(a),
            pb = cartesian$1(b);

        // We have two planes, n1.p = d1 and n2.p = d2.
        // Find intersection line p(t) = c1 n1 + c2 n2 + t (n1 ⨯ n2).
        var n1 = [1, 0, 0], // normal
            n2 = cartesianCross$1(pa, pb),
            n2n2 = cartesianDot$1(n2, n2),
            n1n2 = n2[0], // cartesianDot(n1, n2),
            determinant = n2n2 - n1n2 * n1n2;

        // Two polar points.
        if (!determinant) return !two && a;

        var c1 =  cr * n2n2 / determinant,
            c2 = -cr * n1n2 / determinant,
            n1xn2 = cartesianCross$1(n1, n2),
            A = cartesianScale$1(n1, c1),
            B = cartesianScale$1(n2, c2);
        cartesianAddInPlace$1(A, B);

        // Solve |p(t)|^2 = 1.
        var u = n1xn2,
            w = cartesianDot$1(A, u),
            uu = cartesianDot$1(u, u),
            t2 = w * w - uu * (cartesianDot$1(A, A) - 1);

        if (t2 < 0) return;

        var t = sqrt$3(t2),
            q = cartesianScale$1(u, (-w - t) / uu);
        cartesianAddInPlace$1(q, A);
        q = spherical$1(q);

        if (!two) return q;

        // Two intersection points.
        var lambda0 = a[0],
            lambda1 = b[0],
            phi0 = a[1],
            phi1 = b[1],
            z;

        if (lambda1 < lambda0) z = lambda0, lambda0 = lambda1, lambda1 = z;

        var delta = lambda1 - lambda0,
            polar = abs$2(delta - pi$8) < epsilon$7,
            meridian = polar || delta < epsilon$7;

        if (!polar && phi1 < phi0) z = phi0, phi0 = phi1, phi1 = z;

        // Check that the first point is between a and b.
        if (meridian
            ? polar
              ? phi0 + phi1 > 0 ^ q[1] < (abs$2(q[0] - lambda0) < epsilon$7 ? phi0 : phi1)
              : phi0 <= q[1] && q[1] <= phi1
            : delta > pi$8 ^ (lambda0 <= q[0] && q[0] <= lambda1)) {
          var q1 = cartesianScale$1(u, (-w + t) / uu);
          cartesianAddInPlace$1(q1, A);
          return [q, spherical$1(q1)];
        }
      }

      // Generates a 4-bit vector representing the location of a point relative to
      // the small circle's bounding box.
      function code(lambda, phi) {
        var r = smallRadius ? radius : pi$8 - radius,
            code = 0;
        if (lambda < -r) code |= 1; // left
        else if (lambda > r) code |= 2; // right
        if (phi < -r) code |= 4; // below
        else if (phi > r) code |= 8; // above
        return code;
      }

      return clip$1(visible, clipLine, interpolate, smallRadius ? [0, -radius] : [-pi$8, radius - pi$8]);
    }

    function clipLine$1(a, b, x0, y0, x1, y1) {
      var ax = a[0],
          ay = a[1],
          bx = b[0],
          by = b[1],
          t0 = 0,
          t1 = 1,
          dx = bx - ax,
          dy = by - ay,
          r;

      r = x0 - ax;
      if (!dx && r > 0) return;
      r /= dx;
      if (dx < 0) {
        if (r < t0) return;
        if (r < t1) t1 = r;
      } else if (dx > 0) {
        if (r > t1) return;
        if (r > t0) t0 = r;
      }

      r = x1 - ax;
      if (!dx && r < 0) return;
      r /= dx;
      if (dx < 0) {
        if (r > t1) return;
        if (r > t0) t0 = r;
      } else if (dx > 0) {
        if (r < t0) return;
        if (r < t1) t1 = r;
      }

      r = y0 - ay;
      if (!dy && r > 0) return;
      r /= dy;
      if (dy < 0) {
        if (r < t0) return;
        if (r < t1) t1 = r;
      } else if (dy > 0) {
        if (r > t1) return;
        if (r > t0) t0 = r;
      }

      r = y1 - ay;
      if (!dy && r < 0) return;
      r /= dy;
      if (dy < 0) {
        if (r > t1) return;
        if (r > t0) t0 = r;
      } else if (dy > 0) {
        if (r < t0) return;
        if (r < t1) t1 = r;
      }

      if (t0 > 0) a[0] = ax + t0 * dx, a[1] = ay + t0 * dy;
      if (t1 < 1) b[0] = ax + t1 * dx, b[1] = ay + t1 * dy;
      return true;
    }

    var clipMax$1 = 1e9, clipMin$1 = -clipMax$1;

    // TODO Use d3-polygon’s polygonContains here for the ring check?
    // TODO Eliminate duplicate buffering in clipBuffer and polygon.push?

    function clipRectangle$1(x0, y0, x1, y1) {

      function visible(x, y) {
        return x0 <= x && x <= x1 && y0 <= y && y <= y1;
      }

      function interpolate(from, to, direction, stream) {
        var a = 0, a1 = 0;
        if (from == null
            || (a = corner(from, direction)) !== (a1 = corner(to, direction))
            || comparePoint(from, to) < 0 ^ direction > 0) {
          do stream.point(a === 0 || a === 3 ? x0 : x1, a > 1 ? y1 : y0);
          while ((a = (a + direction + 4) % 4) !== a1);
        } else {
          stream.point(to[0], to[1]);
        }
      }

      function corner(p, direction) {
        return abs$2(p[0] - x0) < epsilon$7 ? direction > 0 ? 0 : 3
            : abs$2(p[0] - x1) < epsilon$7 ? direction > 0 ? 2 : 1
            : abs$2(p[1] - y0) < epsilon$7 ? direction > 0 ? 1 : 0
            : direction > 0 ? 3 : 2; // abs(p[1] - y1) < epsilon
      }

      function compareIntersection(a, b) {
        return comparePoint(a.x, b.x);
      }

      function comparePoint(a, b) {
        var ca = corner(a, 1),
            cb = corner(b, 1);
        return ca !== cb ? ca - cb
            : ca === 0 ? b[1] - a[1]
            : ca === 1 ? a[0] - b[0]
            : ca === 2 ? a[1] - b[1]
            : b[0] - a[0];
      }

      return function(stream) {
        var activeStream = stream,
            bufferStream = clipBuffer$1(),
            segments,
            polygon,
            ring,
            x__, y__, v__, // first point
            x_, y_, v_, // previous point
            first,
            clean;

        var clipStream = {
          point: point,
          lineStart: lineStart,
          lineEnd: lineEnd,
          polygonStart: polygonStart,
          polygonEnd: polygonEnd
        };

        function point(x, y) {
          if (visible(x, y)) activeStream.point(x, y);
        }

        function polygonInside() {
          var winding = 0;

          for (var i = 0, n = polygon.length; i < n; ++i) {
            for (var ring = polygon[i], j = 1, m = ring.length, point = ring[0], a0, a1, b0 = point[0], b1 = point[1]; j < m; ++j) {
              a0 = b0, a1 = b1, point = ring[j], b0 = point[0], b1 = point[1];
              if (a1 <= y1) { if (b1 > y1 && (b0 - a0) * (y1 - a1) > (b1 - a1) * (x0 - a0)) ++winding; }
              else { if (b1 <= y1 && (b0 - a0) * (y1 - a1) < (b1 - a1) * (x0 - a0)) --winding; }
            }
          }

          return winding;
        }

        // Buffer geometry within a polygon and then clip it en masse.
        function polygonStart() {
          activeStream = bufferStream, segments = [], polygon = [], clean = true;
        }

        function polygonEnd() {
          var startInside = polygonInside(),
              cleanInside = clean && startInside,
              visible = (segments = merge$2(segments)).length;
          if (cleanInside || visible) {
            stream.polygonStart();
            if (cleanInside) {
              stream.lineStart();
              interpolate(null, null, 1, stream);
              stream.lineEnd();
            }
            if (visible) {
              clipRejoin$1(segments, compareIntersection, startInside, interpolate, stream);
            }
            stream.polygonEnd();
          }
          activeStream = stream, segments = polygon = ring = null;
        }

        function lineStart() {
          clipStream.point = linePoint;
          if (polygon) polygon.push(ring = []);
          first = true;
          v_ = false;
          x_ = y_ = NaN;
        }

        // TODO rather than special-case polygons, simply handle them separately.
        // Ideally, coincident intersection points should be jittered to avoid
        // clipping issues.
        function lineEnd() {
          if (segments) {
            linePoint(x__, y__);
            if (v__ && v_) bufferStream.rejoin();
            segments.push(bufferStream.result());
          }
          clipStream.point = point;
          if (v_) activeStream.lineEnd();
        }

        function linePoint(x, y) {
          var v = visible(x, y);
          if (polygon) ring.push([x, y]);
          if (first) {
            x__ = x, y__ = y, v__ = v;
            first = false;
            if (v) {
              activeStream.lineStart();
              activeStream.point(x, y);
            }
          } else {
            if (v && v_) activeStream.point(x, y);
            else {
              var a = [x_ = Math.max(clipMin$1, Math.min(clipMax$1, x_)), y_ = Math.max(clipMin$1, Math.min(clipMax$1, y_))],
                  b = [x = Math.max(clipMin$1, Math.min(clipMax$1, x)), y = Math.max(clipMin$1, Math.min(clipMax$1, y))];
              if (clipLine$1(a, b, x0, y0, x1, y1)) {
                if (!v_) {
                  activeStream.lineStart();
                  activeStream.point(a[0], a[1]);
                }
                activeStream.point(b[0], b[1]);
                if (!v) activeStream.lineEnd();
                clean = false;
              } else if (v) {
                activeStream.lineStart();
                activeStream.point(x, y);
                clean = false;
              }
            }
          }
          x_ = x, y_ = y, v_ = v;
        }

        return clipStream;
      };
    }

    function extent$3() {
      var x0 = 0,
          y0 = 0,
          x1 = 960,
          y1 = 500,
          cache,
          cacheStream,
          clip;

      return clip = {
        stream: function(stream) {
          return cache && cacheStream === stream ? cache : cache = clipRectangle$1(x0, y0, x1, y1)(cacheStream = stream);
        },
        extent: function(_) {
          return arguments.length ? (x0 = +_[0][0], y0 = +_[0][1], x1 = +_[1][0], y1 = +_[1][1], cache = cacheStream = null, clip) : [[x0, y0], [x1, y1]];
        }
      };
    }

    var lengthSum$2 = adder$1(),
        lambda0$5,
        sinPhi0$3,
        cosPhi0$3;

    var lengthStream$2 = {
      sphere: noop$8,
      point: noop$8,
      lineStart: lengthLineStart$1,
      lineEnd: noop$8,
      polygonStart: noop$8,
      polygonEnd: noop$8
    };

    function lengthLineStart$1() {
      lengthStream$2.point = lengthPointFirst$2;
      lengthStream$2.lineEnd = lengthLineEnd$1;
    }

    function lengthLineEnd$1() {
      lengthStream$2.point = lengthStream$2.lineEnd = noop$8;
    }

    function lengthPointFirst$2(lambda, phi) {
      lambda *= radians$1, phi *= radians$1;
      lambda0$5 = lambda, sinPhi0$3 = sin$4(phi), cosPhi0$3 = cos$4(phi);
      lengthStream$2.point = lengthPoint$2;
    }

    function lengthPoint$2(lambda, phi) {
      lambda *= radians$1, phi *= radians$1;
      var sinPhi = sin$4(phi),
          cosPhi = cos$4(phi),
          delta = abs$2(lambda - lambda0$5),
          cosDelta = cos$4(delta),
          sinDelta = sin$4(delta),
          x = cosPhi * sinDelta,
          y = cosPhi0$3 * sinPhi - sinPhi0$3 * cosPhi * cosDelta,
          z = sinPhi0$3 * sinPhi + cosPhi0$3 * cosPhi * cosDelta;
      lengthSum$2.add(atan2$2(sqrt$3(x * x + y * y), z));
      lambda0$5 = lambda, sinPhi0$3 = sinPhi, cosPhi0$3 = cosPhi;
    }

    function length$4(object) {
      lengthSum$2.reset();
      geoStream$1(object, lengthStream$2);
      return +lengthSum$2;
    }

    var coordinates$1 = [null, null],
        object$3 = {type: "LineString", coordinates: coordinates$1};

    function distance$1(a, b) {
      coordinates$1[0] = a;
      coordinates$1[1] = b;
      return length$4(object$3);
    }

    var containsObjectType$1 = {
      Feature: function(object, point) {
        return containsGeometry$1(object.geometry, point);
      },
      FeatureCollection: function(object, point) {
        var features = object.features, i = -1, n = features.length;
        while (++i < n) if (containsGeometry$1(features[i].geometry, point)) return true;
        return false;
      }
    };

    var containsGeometryType$1 = {
      Sphere: function() {
        return true;
      },
      Point: function(object, point) {
        return containsPoint$1(object.coordinates, point);
      },
      MultiPoint: function(object, point) {
        var coordinates = object.coordinates, i = -1, n = coordinates.length;
        while (++i < n) if (containsPoint$1(coordinates[i], point)) return true;
        return false;
      },
      LineString: function(object, point) {
        return containsLine$1(object.coordinates, point);
      },
      MultiLineString: function(object, point) {
        var coordinates = object.coordinates, i = -1, n = coordinates.length;
        while (++i < n) if (containsLine$1(coordinates[i], point)) return true;
        return false;
      },
      Polygon: function(object, point) {
        return containsPolygon$1(object.coordinates, point);
      },
      MultiPolygon: function(object, point) {
        var coordinates = object.coordinates, i = -1, n = coordinates.length;
        while (++i < n) if (containsPolygon$1(coordinates[i], point)) return true;
        return false;
      },
      GeometryCollection: function(object, point) {
        var geometries = object.geometries, i = -1, n = geometries.length;
        while (++i < n) if (containsGeometry$1(geometries[i], point)) return true;
        return false;
      }
    };

    function containsGeometry$1(geometry, point) {
      return geometry && containsGeometryType$1.hasOwnProperty(geometry.type)
          ? containsGeometryType$1[geometry.type](geometry, point)
          : false;
    }

    function containsPoint$1(coordinates, point) {
      return distance$1(coordinates, point) === 0;
    }

    function containsLine$1(coordinates, point) {
      var ab = distance$1(coordinates[0], coordinates[1]),
          ao = distance$1(coordinates[0], point),
          ob = distance$1(point, coordinates[1]);
      return ao + ob <= ab + epsilon$7;
    }

    function containsPolygon$1(coordinates, point) {
      return !!polygonContains$1(coordinates.map(ringRadians$1), pointRadians$1(point));
    }

    function ringRadians$1(ring) {
      return ring = ring.map(pointRadians$1), ring.pop(), ring;
    }

    function pointRadians$1(point) {
      return [point[0] * radians$1, point[1] * radians$1];
    }

    function contains$3(object, point) {
      return (object && containsObjectType$1.hasOwnProperty(object.type)
          ? containsObjectType$1[object.type]
          : containsGeometry$1)(object, point);
    }

    function graticuleX$1(y0, y1, dy) {
      var y = range$4(y0, y1 - epsilon$7, dy).concat(y1);
      return function(x) { return y.map(function(y) { return [x, y]; }); };
    }

    function graticuleY$1(x0, x1, dx) {
      var x = range$4(x0, x1 - epsilon$7, dx).concat(x1);
      return function(y) { return x.map(function(x) { return [x, y]; }); };
    }

    function graticule$1() {
      var x1, x0, X1, X0,
          y1, y0, Y1, Y0,
          dx = 10, dy = dx, DX = 90, DY = 360,
          x, y, X, Y,
          precision = 2.5;

      function graticule() {
        return {type: "MultiLineString", coordinates: lines()};
      }

      function lines() {
        return range$4(ceil$1(X0 / DX) * DX, X1, DX).map(X)
            .concat(range$4(ceil$1(Y0 / DY) * DY, Y1, DY).map(Y))
            .concat(range$4(ceil$1(x0 / dx) * dx, x1, dx).filter(function(x) { return abs$2(x % DX) > epsilon$7; }).map(x))
            .concat(range$4(ceil$1(y0 / dy) * dy, y1, dy).filter(function(y) { return abs$2(y % DY) > epsilon$7; }).map(y));
      }

      graticule.lines = function() {
        return lines().map(function(coordinates) { return {type: "LineString", coordinates: coordinates}; });
      };

      graticule.outline = function() {
        return {
          type: "Polygon",
          coordinates: [
            X(X0).concat(
            Y(Y1).slice(1),
            X(X1).reverse().slice(1),
            Y(Y0).reverse().slice(1))
          ]
        };
      };

      graticule.extent = function(_) {
        if (!arguments.length) return graticule.extentMinor();
        return graticule.extentMajor(_).extentMinor(_);
      };

      graticule.extentMajor = function(_) {
        if (!arguments.length) return [[X0, Y0], [X1, Y1]];
        X0 = +_[0][0], X1 = +_[1][0];
        Y0 = +_[0][1], Y1 = +_[1][1];
        if (X0 > X1) _ = X0, X0 = X1, X1 = _;
        if (Y0 > Y1) _ = Y0, Y0 = Y1, Y1 = _;
        return graticule.precision(precision);
      };

      graticule.extentMinor = function(_) {
        if (!arguments.length) return [[x0, y0], [x1, y1]];
        x0 = +_[0][0], x1 = +_[1][0];
        y0 = +_[0][1], y1 = +_[1][1];
        if (x0 > x1) _ = x0, x0 = x1, x1 = _;
        if (y0 > y1) _ = y0, y0 = y1, y1 = _;
        return graticule.precision(precision);
      };

      graticule.step = function(_) {
        if (!arguments.length) return graticule.stepMinor();
        return graticule.stepMajor(_).stepMinor(_);
      };

      graticule.stepMajor = function(_) {
        if (!arguments.length) return [DX, DY];
        DX = +_[0], DY = +_[1];
        return graticule;
      };

      graticule.stepMinor = function(_) {
        if (!arguments.length) return [dx, dy];
        dx = +_[0], dy = +_[1];
        return graticule;
      };

      graticule.precision = function(_) {
        if (!arguments.length) return precision;
        precision = +_;
        x = graticuleX$1(y0, y1, 90);
        y = graticuleY$1(x0, x1, precision);
        X = graticuleX$1(Y0, Y1, 90);
        Y = graticuleY$1(X0, X1, precision);
        return graticule;
      };

      return graticule
          .extentMajor([[-180, -90 + epsilon$7], [180, 90 - epsilon$7]])
          .extentMinor([[-180, -80 - epsilon$7], [180, 80 + epsilon$7]]);
    }

    function graticule10$1() {
      return graticule$1()();
    }

    function interpolate$4(a, b) {
      var x0 = a[0] * radians$1,
          y0 = a[1] * radians$1,
          x1 = b[0] * radians$1,
          y1 = b[1] * radians$1,
          cy0 = cos$4(y0),
          sy0 = sin$4(y0),
          cy1 = cos$4(y1),
          sy1 = sin$4(y1),
          kx0 = cy0 * cos$4(x0),
          ky0 = cy0 * sin$4(x0),
          kx1 = cy1 * cos$4(x1),
          ky1 = cy1 * sin$4(x1),
          d = 2 * asin$2(sqrt$3(haversin$1(y1 - y0) + cy0 * cy1 * haversin$1(x1 - x0))),
          k = sin$4(d);

      var interpolate = d ? function(t) {
        var B = sin$4(t *= d) / k,
            A = sin$4(d - t) / k,
            x = A * kx0 + B * kx1,
            y = A * ky0 + B * ky1,
            z = A * sy0 + B * sy1;
        return [
          atan2$2(y, x) * degrees$3,
          atan2$2(z, sqrt$3(x * x + y * y)) * degrees$3
        ];
      } : function() {
        return [x0 * degrees$3, y0 * degrees$3];
      };

      interpolate.distance = d;

      return interpolate;
    }

    function identity$e(x) {
      return x;
    }

    var areaSum$3 = adder$1(),
        areaRingSum$3 = adder$1(),
        x00$3,
        y00$3,
        x0$6,
        y0$6;

    var areaStream$3 = {
      point: noop$8,
      lineStart: noop$8,
      lineEnd: noop$8,
      polygonStart: function() {
        areaStream$3.lineStart = areaRingStart$3;
        areaStream$3.lineEnd = areaRingEnd$3;
      },
      polygonEnd: function() {
        areaStream$3.lineStart = areaStream$3.lineEnd = areaStream$3.point = noop$8;
        areaSum$3.add(abs$2(areaRingSum$3));
        areaRingSum$3.reset();
      },
      result: function() {
        var area = areaSum$3 / 2;
        areaSum$3.reset();
        return area;
      }
    };

    function areaRingStart$3() {
      areaStream$3.point = areaPointFirst$3;
    }

    function areaPointFirst$3(x, y) {
      areaStream$3.point = areaPoint$3;
      x00$3 = x0$6 = x, y00$3 = y0$6 = y;
    }

    function areaPoint$3(x, y) {
      areaRingSum$3.add(y0$6 * x - x0$6 * y);
      x0$6 = x, y0$6 = y;
    }

    function areaRingEnd$3() {
      areaPoint$3(x00$3, y00$3);
    }

    var x0$7 = Infinity,
        y0$7 = x0$7,
        x1$1 = -x0$7,
        y1$1 = x1$1;

    var boundsStream$3 = {
      point: boundsPoint$3,
      lineStart: noop$8,
      lineEnd: noop$8,
      polygonStart: noop$8,
      polygonEnd: noop$8,
      result: function() {
        var bounds = [[x0$7, y0$7], [x1$1, y1$1]];
        x1$1 = y1$1 = -(y0$7 = x0$7 = Infinity);
        return bounds;
      }
    };

    function boundsPoint$3(x, y) {
      if (x < x0$7) x0$7 = x;
      if (x > x1$1) x1$1 = x;
      if (y < y0$7) y0$7 = y;
      if (y > y1$1) y1$1 = y;
    }

    // TODO Enforce positive area for exterior, negative area for interior?

    var X0$3 = 0,
        Y0$3 = 0,
        Z0$3 = 0,
        X1$3 = 0,
        Y1$3 = 0,
        Z1$3 = 0,
        X2$3 = 0,
        Y2$3 = 0,
        Z2$3 = 0,
        x00$4,
        y00$4,
        x0$8,
        y0$8;

    var centroidStream$3 = {
      point: centroidPoint$3,
      lineStart: centroidLineStart$3,
      lineEnd: centroidLineEnd$3,
      polygonStart: function() {
        centroidStream$3.lineStart = centroidRingStart$3;
        centroidStream$3.lineEnd = centroidRingEnd$3;
      },
      polygonEnd: function() {
        centroidStream$3.point = centroidPoint$3;
        centroidStream$3.lineStart = centroidLineStart$3;
        centroidStream$3.lineEnd = centroidLineEnd$3;
      },
      result: function() {
        var centroid = Z2$3 ? [X2$3 / Z2$3, Y2$3 / Z2$3]
            : Z1$3 ? [X1$3 / Z1$3, Y1$3 / Z1$3]
            : Z0$3 ? [X0$3 / Z0$3, Y0$3 / Z0$3]
            : [NaN, NaN];
        X0$3 = Y0$3 = Z0$3 =
        X1$3 = Y1$3 = Z1$3 =
        X2$3 = Y2$3 = Z2$3 = 0;
        return centroid;
      }
    };

    function centroidPoint$3(x, y) {
      X0$3 += x;
      Y0$3 += y;
      ++Z0$3;
    }

    function centroidLineStart$3() {
      centroidStream$3.point = centroidPointFirstLine$1;
    }

    function centroidPointFirstLine$1(x, y) {
      centroidStream$3.point = centroidPointLine$1;
      centroidPoint$3(x0$8 = x, y0$8 = y);
    }

    function centroidPointLine$1(x, y) {
      var dx = x - x0$8, dy = y - y0$8, z = sqrt$3(dx * dx + dy * dy);
      X1$3 += z * (x0$8 + x) / 2;
      Y1$3 += z * (y0$8 + y) / 2;
      Z1$3 += z;
      centroidPoint$3(x0$8 = x, y0$8 = y);
    }

    function centroidLineEnd$3() {
      centroidStream$3.point = centroidPoint$3;
    }

    function centroidRingStart$3() {
      centroidStream$3.point = centroidPointFirstRing$1;
    }

    function centroidRingEnd$3() {
      centroidPointRing$1(x00$4, y00$4);
    }

    function centroidPointFirstRing$1(x, y) {
      centroidStream$3.point = centroidPointRing$1;
      centroidPoint$3(x00$4 = x0$8 = x, y00$4 = y0$8 = y);
    }

    function centroidPointRing$1(x, y) {
      var dx = x - x0$8,
          dy = y - y0$8,
          z = sqrt$3(dx * dx + dy * dy);

      X1$3 += z * (x0$8 + x) / 2;
      Y1$3 += z * (y0$8 + y) / 2;
      Z1$3 += z;

      z = y0$8 * x - x0$8 * y;
      X2$3 += z * (x0$8 + x);
      Y2$3 += z * (y0$8 + y);
      Z2$3 += z * 3;
      centroidPoint$3(x0$8 = x, y0$8 = y);
    }

    function PathContext$1(context) {
      this._context = context;
    }

    PathContext$1.prototype = {
      _radius: 4.5,
      pointRadius: function(_) {
        return this._radius = _, this;
      },
      polygonStart: function() {
        this._line = 0;
      },
      polygonEnd: function() {
        this._line = NaN;
      },
      lineStart: function() {
        this._point = 0;
      },
      lineEnd: function() {
        if (this._line === 0) this._context.closePath();
        this._point = NaN;
      },
      point: function(x, y) {
        switch (this._point) {
          case 0: {
            this._context.moveTo(x, y);
            this._point = 1;
            break;
          }
          case 1: {
            this._context.lineTo(x, y);
            break;
          }
          default: {
            this._context.moveTo(x + this._radius, y);
            this._context.arc(x, y, this._radius, 0, tau$8);
            break;
          }
        }
      },
      result: noop$8
    };

    var lengthSum$3 = adder$1(),
        lengthRing$1,
        x00$5,
        y00$5,
        x0$9,
        y0$9;

    var lengthStream$3 = {
      point: noop$8,
      lineStart: function() {
        lengthStream$3.point = lengthPointFirst$3;
      },
      lineEnd: function() {
        if (lengthRing$1) lengthPoint$3(x00$5, y00$5);
        lengthStream$3.point = noop$8;
      },
      polygonStart: function() {
        lengthRing$1 = true;
      },
      polygonEnd: function() {
        lengthRing$1 = null;
      },
      result: function() {
        var length = +lengthSum$3;
        lengthSum$3.reset();
        return length;
      }
    };

    function lengthPointFirst$3(x, y) {
      lengthStream$3.point = lengthPoint$3;
      x00$5 = x0$9 = x, y00$5 = y0$9 = y;
    }

    function lengthPoint$3(x, y) {
      x0$9 -= x, y0$9 -= y;
      lengthSum$3.add(sqrt$3(x0$9 * x0$9 + y0$9 * y0$9));
      x0$9 = x, y0$9 = y;
    }

    function PathString$1() {
      this._string = [];
    }

    PathString$1.prototype = {
      _radius: 4.5,
      _circle: circle$4(4.5),
      pointRadius: function(_) {
        if ((_ = +_) !== this._radius) this._radius = _, this._circle = null;
        return this;
      },
      polygonStart: function() {
        this._line = 0;
      },
      polygonEnd: function() {
        this._line = NaN;
      },
      lineStart: function() {
        this._point = 0;
      },
      lineEnd: function() {
        if (this._line === 0) this._string.push("Z");
        this._point = NaN;
      },
      point: function(x, y) {
        switch (this._point) {
          case 0: {
            this._string.push("M", x, ",", y);
            this._point = 1;
            break;
          }
          case 1: {
            this._string.push("L", x, ",", y);
            break;
          }
          default: {
            if (this._circle == null) this._circle = circle$4(this._radius);
            this._string.push("M", x, ",", y, this._circle);
            break;
          }
        }
      },
      result: function() {
        if (this._string.length) {
          var result = this._string.join("");
          this._string = [];
          return result;
        } else {
          return null;
        }
      }
    };

    function circle$4(radius) {
      return "m0," + radius
          + "a" + radius + "," + radius + " 0 1,1 0," + -2 * radius
          + "a" + radius + "," + radius + " 0 1,1 0," + 2 * radius
          + "z";
    }

    function index$5(projection, context) {
      var pointRadius = 4.5,
          projectionStream,
          contextStream;

      function path(object) {
        if (object) {
          if (typeof pointRadius === "function") contextStream.pointRadius(+pointRadius.apply(this, arguments));
          geoStream$1(object, projectionStream(contextStream));
        }
        return contextStream.result();
      }

      path.area = function(object) {
        geoStream$1(object, projectionStream(areaStream$3));
        return areaStream$3.result();
      };

      path.measure = function(object) {
        geoStream$1(object, projectionStream(lengthStream$3));
        return lengthStream$3.result();
      };

      path.bounds = function(object) {
        geoStream$1(object, projectionStream(boundsStream$3));
        return boundsStream$3.result();
      };

      path.centroid = function(object) {
        geoStream$1(object, projectionStream(centroidStream$3));
        return centroidStream$3.result();
      };

      path.projection = function(_) {
        return arguments.length ? (projectionStream = _ == null ? (projection = null, identity$e) : (projection = _).stream, path) : projection;
      };

      path.context = function(_) {
        if (!arguments.length) return context;
        contextStream = _ == null ? (context = null, new PathString$1) : new PathContext$1(context = _);
        if (typeof pointRadius !== "function") contextStream.pointRadius(pointRadius);
        return path;
      };

      path.pointRadius = function(_) {
        if (!arguments.length) return pointRadius;
        pointRadius = typeof _ === "function" ? _ : (contextStream.pointRadius(+_), +_);
        return path;
      };

      return path.projection(projection).context(context);
    }

    function transform$2(methods) {
      return {
        stream: transformer$4(methods)
      };
    }

    function transformer$4(methods) {
      return function(stream) {
        var s = new TransformStream$1;
        for (var key in methods) s[key] = methods[key];
        s.stream = stream;
        return s;
      };
    }

    function TransformStream$1() {}

    TransformStream$1.prototype = {
      constructor: TransformStream$1,
      point: function(x, y) { this.stream.point(x, y); },
      sphere: function() { this.stream.sphere(); },
      lineStart: function() { this.stream.lineStart(); },
      lineEnd: function() { this.stream.lineEnd(); },
      polygonStart: function() { this.stream.polygonStart(); },
      polygonEnd: function() { this.stream.polygonEnd(); }
    };

    function fit$1(projection, fitBounds, object) {
      var clip = projection.clipExtent && projection.clipExtent();
      projection.scale(150).translate([0, 0]);
      if (clip != null) projection.clipExtent(null);
      geoStream$1(object, projection.stream(boundsStream$3));
      fitBounds(boundsStream$3.result());
      if (clip != null) projection.clipExtent(clip);
      return projection;
    }

    function fitExtent$1(projection, extent, object) {
      return fit$1(projection, function(b) {
        var w = extent[1][0] - extent[0][0],
            h = extent[1][1] - extent[0][1],
            k = Math.min(w / (b[1][0] - b[0][0]), h / (b[1][1] - b[0][1])),
            x = +extent[0][0] + (w - k * (b[1][0] + b[0][0])) / 2,
            y = +extent[0][1] + (h - k * (b[1][1] + b[0][1])) / 2;
        projection.scale(150 * k).translate([x, y]);
      }, object);
    }

    function fitSize$1(projection, size, object) {
      return fitExtent$1(projection, [[0, 0], size], object);
    }

    function fitWidth$1(projection, width, object) {
      return fit$1(projection, function(b) {
        var w = +width,
            k = w / (b[1][0] - b[0][0]),
            x = (w - k * (b[1][0] + b[0][0])) / 2,
            y = -k * b[0][1];
        projection.scale(150 * k).translate([x, y]);
      }, object);
    }

    function fitHeight$1(projection, height, object) {
      return fit$1(projection, function(b) {
        var h = +height,
            k = h / (b[1][1] - b[0][1]),
            x = -k * b[0][0],
            y = (h - k * (b[1][1] + b[0][1])) / 2;
        projection.scale(150 * k).translate([x, y]);
      }, object);
    }

    var maxDepth$1 = 16, // maximum depth of subdivision
        cosMinDistance$1 = cos$4(30 * radians$1); // cos(minimum angular distance)

    function resample$2(project, delta2) {
      return +delta2 ? resample$3(project, delta2) : resampleNone$1(project);
    }

    function resampleNone$1(project) {
      return transformer$4({
        point: function(x, y) {
          x = project(x, y);
          this.stream.point(x[0], x[1]);
        }
      });
    }

    function resample$3(project, delta2) {

      function resampleLineTo(x0, y0, lambda0, a0, b0, c0, x1, y1, lambda1, a1, b1, c1, depth, stream) {
        var dx = x1 - x0,
            dy = y1 - y0,
            d2 = dx * dx + dy * dy;
        if (d2 > 4 * delta2 && depth--) {
          var a = a0 + a1,
              b = b0 + b1,
              c = c0 + c1,
              m = sqrt$3(a * a + b * b + c * c),
              phi2 = asin$2(c /= m),
              lambda2 = abs$2(abs$2(c) - 1) < epsilon$7 || abs$2(lambda0 - lambda1) < epsilon$7 ? (lambda0 + lambda1) / 2 : atan2$2(b, a),
              p = project(lambda2, phi2),
              x2 = p[0],
              y2 = p[1],
              dx2 = x2 - x0,
              dy2 = y2 - y0,
              dz = dy * dx2 - dx * dy2;
          if (dz * dz / d2 > delta2 // perpendicular projected distance
              || abs$2((dx * dx2 + dy * dy2) / d2 - 0.5) > 0.3 // midpoint close to an end
              || a0 * a1 + b0 * b1 + c0 * c1 < cosMinDistance$1) { // angular distance
            resampleLineTo(x0, y0, lambda0, a0, b0, c0, x2, y2, lambda2, a /= m, b /= m, c, depth, stream);
            stream.point(x2, y2);
            resampleLineTo(x2, y2, lambda2, a, b, c, x1, y1, lambda1, a1, b1, c1, depth, stream);
          }
        }
      }
      return function(stream) {
        var lambda00, x00, y00, a00, b00, c00, // first point
            lambda0, x0, y0, a0, b0, c0; // previous point

        var resampleStream = {
          point: point,
          lineStart: lineStart,
          lineEnd: lineEnd,
          polygonStart: function() { stream.polygonStart(); resampleStream.lineStart = ringStart; },
          polygonEnd: function() { stream.polygonEnd(); resampleStream.lineStart = lineStart; }
        };

        function point(x, y) {
          x = project(x, y);
          stream.point(x[0], x[1]);
        }

        function lineStart() {
          x0 = NaN;
          resampleStream.point = linePoint;
          stream.lineStart();
        }

        function linePoint(lambda, phi) {
          var c = cartesian$1([lambda, phi]), p = project(lambda, phi);
          resampleLineTo(x0, y0, lambda0, a0, b0, c0, x0 = p[0], y0 = p[1], lambda0 = lambda, a0 = c[0], b0 = c[1], c0 = c[2], maxDepth$1, stream);
          stream.point(x0, y0);
        }

        function lineEnd() {
          resampleStream.point = point;
          stream.lineEnd();
        }

        function ringStart() {
          lineStart();
          resampleStream.point = ringPoint;
          resampleStream.lineEnd = ringEnd;
        }

        function ringPoint(lambda, phi) {
          linePoint(lambda00 = lambda, phi), x00 = x0, y00 = y0, a00 = a0, b00 = b0, c00 = c0;
          resampleStream.point = linePoint;
        }

        function ringEnd() {
          resampleLineTo(x0, y0, lambda0, a0, b0, c0, x00, y00, lambda00, a00, b00, c00, maxDepth$1, stream);
          resampleStream.lineEnd = lineEnd;
          lineEnd();
        }

        return resampleStream;
      };
    }

    var transformRadians$1 = transformer$4({
      point: function(x, y) {
        this.stream.point(x * radians$1, y * radians$1);
      }
    });

    function transformRotate$1(rotate) {
      return transformer$4({
        point: function(x, y) {
          var r = rotate(x, y);
          return this.stream.point(r[0], r[1]);
        }
      });
    }

    function projection$1(project) {
      return projectionMutator$1(function() { return project; })();
    }

    function projectionMutator$1(projectAt) {
      var project,
          k = 150, // scale
          x = 480, y = 250, // translate
          dx, dy, lambda = 0, phi = 0, // center
          deltaLambda = 0, deltaPhi = 0, deltaGamma = 0, rotate, projectRotate, // rotate
          theta = null, preclip = clipAntimeridian$1, // clip angle
          x0 = null, y0, x1, y1, postclip = identity$e, // clip extent
          delta2 = 0.5, projectResample = resample$2(projectTransform, delta2), // precision
          cache,
          cacheStream;

      function projection(point) {
        point = projectRotate(point[0] * radians$1, point[1] * radians$1);
        return [point[0] * k + dx, dy - point[1] * k];
      }

      function invert(point) {
        point = projectRotate.invert((point[0] - dx) / k, (dy - point[1]) / k);
        return point && [point[0] * degrees$3, point[1] * degrees$3];
      }

      function projectTransform(x, y) {
        return x = project(x, y), [x[0] * k + dx, dy - x[1] * k];
      }

      projection.stream = function(stream) {
        return cache && cacheStream === stream ? cache : cache = transformRadians$1(transformRotate$1(rotate)(preclip(projectResample(postclip(cacheStream = stream)))));
      };

      projection.preclip = function(_) {
        return arguments.length ? (preclip = _, theta = undefined, reset()) : preclip;
      };

      projection.postclip = function(_) {
        return arguments.length ? (postclip = _, x0 = y0 = x1 = y1 = null, reset()) : postclip;
      };

      projection.clipAngle = function(_) {
        return arguments.length ? (preclip = +_ ? clipCircle$1(theta = _ * radians$1) : (theta = null, clipAntimeridian$1), reset()) : theta * degrees$3;
      };

      projection.clipExtent = function(_) {
        return arguments.length ? (postclip = _ == null ? (x0 = y0 = x1 = y1 = null, identity$e) : clipRectangle$1(x0 = +_[0][0], y0 = +_[0][1], x1 = +_[1][0], y1 = +_[1][1]), reset()) : x0 == null ? null : [[x0, y0], [x1, y1]];
      };

      projection.scale = function(_) {
        return arguments.length ? (k = +_, recenter()) : k;
      };

      projection.translate = function(_) {
        return arguments.length ? (x = +_[0], y = +_[1], recenter()) : [x, y];
      };

      projection.center = function(_) {
        return arguments.length ? (lambda = _[0] % 360 * radians$1, phi = _[1] % 360 * radians$1, recenter()) : [lambda * degrees$3, phi * degrees$3];
      };

      projection.rotate = function(_) {
        return arguments.length ? (deltaLambda = _[0] % 360 * radians$1, deltaPhi = _[1] % 360 * radians$1, deltaGamma = _.length > 2 ? _[2] % 360 * radians$1 : 0, recenter()) : [deltaLambda * degrees$3, deltaPhi * degrees$3, deltaGamma * degrees$3];
      };

      projection.precision = function(_) {
        return arguments.length ? (projectResample = resample$2(projectTransform, delta2 = _ * _), reset()) : sqrt$3(delta2);
      };

      projection.fitExtent = function(extent, object) {
        return fitExtent$1(projection, extent, object);
      };

      projection.fitSize = function(size, object) {
        return fitSize$1(projection, size, object);
      };

      projection.fitWidth = function(width, object) {
        return fitWidth$1(projection, width, object);
      };

      projection.fitHeight = function(height, object) {
        return fitHeight$1(projection, height, object);
      };

      function recenter() {
        projectRotate = compose$1(rotate = rotateRadians$1(deltaLambda, deltaPhi, deltaGamma), project);
        var center = project(lambda, phi);
        dx = x - center[0] * k;
        dy = y + center[1] * k;
        return reset();
      }

      function reset() {
        cache = cacheStream = null;
        return projection;
      }

      return function() {
        project = projectAt.apply(this, arguments);
        projection.invert = project.invert && invert;
        return recenter();
      };
    }

    function conicProjection$1(projectAt) {
      var phi0 = 0,
          phi1 = pi$8 / 3,
          m = projectionMutator$1(projectAt),
          p = m(phi0, phi1);

      p.parallels = function(_) {
        return arguments.length ? m(phi0 = _[0] * radians$1, phi1 = _[1] * radians$1) : [phi0 * degrees$3, phi1 * degrees$3];
      };

      return p;
    }

    function cylindricalEqualAreaRaw$1(phi0) {
      var cosPhi0 = cos$4(phi0);

      function forward(lambda, phi) {
        return [lambda * cosPhi0, sin$4(phi) / cosPhi0];
      }

      forward.invert = function(x, y) {
        return [x / cosPhi0, asin$2(y * cosPhi0)];
      };

      return forward;
    }

    function conicEqualAreaRaw$1(y0, y1) {
      var sy0 = sin$4(y0), n = (sy0 + sin$4(y1)) / 2;

      // Are the parallels symmetrical around the Equator?
      if (abs$2(n) < epsilon$7) return cylindricalEqualAreaRaw$1(y0);

      var c = 1 + sy0 * (2 * n - sy0), r0 = sqrt$3(c) / n;

      function project(x, y) {
        var r = sqrt$3(c - 2 * n * sin$4(y)) / n;
        return [r * sin$4(x *= n), r0 - r * cos$4(x)];
      }

      project.invert = function(x, y) {
        var r0y = r0 - y;
        return [atan2$2(x, abs$2(r0y)) / n * sign$2(r0y), asin$2((c - (x * x + r0y * r0y) * n * n) / (2 * n))];
      };

      return project;
    }

    function conicEqualArea$1() {
      return conicProjection$1(conicEqualAreaRaw$1)
          .scale(155.424)
          .center([0, 33.6442]);
    }

    function albers$1() {
      return conicEqualArea$1()
          .parallels([29.5, 45.5])
          .scale(1070)
          .translate([480, 250])
          .rotate([96, 0])
          .center([-0.6, 38.7]);
    }

    // The projections must have mutually exclusive clip regions on the sphere,
    // as this will avoid emitting interleaving lines and polygons.
    function multiplex$1(streams) {
      var n = streams.length;
      return {
        point: function(x, y) { var i = -1; while (++i < n) streams[i].point(x, y); },
        sphere: function() { var i = -1; while (++i < n) streams[i].sphere(); },
        lineStart: function() { var i = -1; while (++i < n) streams[i].lineStart(); },
        lineEnd: function() { var i = -1; while (++i < n) streams[i].lineEnd(); },
        polygonStart: function() { var i = -1; while (++i < n) streams[i].polygonStart(); },
        polygonEnd: function() { var i = -1; while (++i < n) streams[i].polygonEnd(); }
      };
    }

    // A composite projection for the United States, configured by default for
    // 960×500. The projection also works quite well at 960×600 if you change the
    // scale to 1285 and adjust the translate accordingly. The set of standard
    // parallels for each region comes from USGS, which is published here:
    // http://egsc.usgs.gov/isb/pubs/MapProjections/projections.html#albers
    function albersUsa$1() {
      var cache,
          cacheStream,
          lower48 = albers$1(), lower48Point,
          alaska = conicEqualArea$1().rotate([154, 0]).center([-2, 58.5]).parallels([55, 65]), alaskaPoint, // EPSG:3338
          hawaii = conicEqualArea$1().rotate([157, 0]).center([-3, 19.9]).parallels([8, 18]), hawaiiPoint, // ESRI:102007
          point, pointStream = {point: function(x, y) { point = [x, y]; }};

      function albersUsa(coordinates) {
        var x = coordinates[0], y = coordinates[1];
        return point = null,
            (lower48Point.point(x, y), point)
            || (alaskaPoint.point(x, y), point)
            || (hawaiiPoint.point(x, y), point);
      }

      albersUsa.invert = function(coordinates) {
        var k = lower48.scale(),
            t = lower48.translate(),
            x = (coordinates[0] - t[0]) / k,
            y = (coordinates[1] - t[1]) / k;
        return (y >= 0.120 && y < 0.234 && x >= -0.425 && x < -0.214 ? alaska
            : y >= 0.166 && y < 0.234 && x >= -0.214 && x < -0.115 ? hawaii
            : lower48).invert(coordinates);
      };

      albersUsa.stream = function(stream) {
        return cache && cacheStream === stream ? cache : cache = multiplex$1([lower48.stream(cacheStream = stream), alaska.stream(stream), hawaii.stream(stream)]);
      };

      albersUsa.precision = function(_) {
        if (!arguments.length) return lower48.precision();
        lower48.precision(_), alaska.precision(_), hawaii.precision(_);
        return reset();
      };

      albersUsa.scale = function(_) {
        if (!arguments.length) return lower48.scale();
        lower48.scale(_), alaska.scale(_ * 0.35), hawaii.scale(_);
        return albersUsa.translate(lower48.translate());
      };

      albersUsa.translate = function(_) {
        if (!arguments.length) return lower48.translate();
        var k = lower48.scale(), x = +_[0], y = +_[1];

        lower48Point = lower48
            .translate(_)
            .clipExtent([[x - 0.455 * k, y - 0.238 * k], [x + 0.455 * k, y + 0.238 * k]])
            .stream(pointStream);

        alaskaPoint = alaska
            .translate([x - 0.307 * k, y + 0.201 * k])
            .clipExtent([[x - 0.425 * k + epsilon$7, y + 0.120 * k + epsilon$7], [x - 0.214 * k - epsilon$7, y + 0.234 * k - epsilon$7]])
            .stream(pointStream);

        hawaiiPoint = hawaii
            .translate([x - 0.205 * k, y + 0.212 * k])
            .clipExtent([[x - 0.214 * k + epsilon$7, y + 0.166 * k + epsilon$7], [x - 0.115 * k - epsilon$7, y + 0.234 * k - epsilon$7]])
            .stream(pointStream);

        return reset();
      };

      albersUsa.fitExtent = function(extent, object) {
        return fitExtent$1(albersUsa, extent, object);
      };

      albersUsa.fitSize = function(size, object) {
        return fitSize$1(albersUsa, size, object);
      };

      albersUsa.fitWidth = function(width, object) {
        return fitWidth$1(albersUsa, width, object);
      };

      albersUsa.fitHeight = function(height, object) {
        return fitHeight$1(albersUsa, height, object);
      };

      function reset() {
        cache = cacheStream = null;
        return albersUsa;
      }

      return albersUsa.scale(1070);
    }

    function azimuthalRaw$1(scale) {
      return function(x, y) {
        var cx = cos$4(x),
            cy = cos$4(y),
            k = scale(cx * cy);
        return [
          k * cy * sin$4(x),
          k * sin$4(y)
        ];
      }
    }

    function azimuthalInvert$1(angle) {
      return function(x, y) {
        var z = sqrt$3(x * x + y * y),
            c = angle(z),
            sc = sin$4(c),
            cc = cos$4(c);
        return [
          atan2$2(x * sc, z * cc),
          asin$2(z && y * sc / z)
        ];
      }
    }

    var azimuthalEqualAreaRaw$1 = azimuthalRaw$1(function(cxcy) {
      return sqrt$3(2 / (1 + cxcy));
    });

    azimuthalEqualAreaRaw$1.invert = azimuthalInvert$1(function(z) {
      return 2 * asin$2(z / 2);
    });

    function azimuthalEqualArea$1() {
      return projection$1(azimuthalEqualAreaRaw$1)
          .scale(124.75)
          .clipAngle(180 - 1e-3);
    }

    var azimuthalEquidistantRaw$1 = azimuthalRaw$1(function(c) {
      return (c = acos$2(c)) && c / sin$4(c);
    });

    azimuthalEquidistantRaw$1.invert = azimuthalInvert$1(function(z) {
      return z;
    });

    function azimuthalEquidistant$1() {
      return projection$1(azimuthalEquidistantRaw$1)
          .scale(79.4188)
          .clipAngle(180 - 1e-3);
    }

    function mercatorRaw$1(lambda, phi) {
      return [lambda, log$2(tan$1((halfPi$6 + phi) / 2))];
    }

    mercatorRaw$1.invert = function(x, y) {
      return [x, 2 * atan$1(exp$1(y)) - halfPi$6];
    };

    function mercator$1() {
      return mercatorProjection$1(mercatorRaw$1)
          .scale(961 / tau$8);
    }

    function mercatorProjection$1(project) {
      var m = projection$1(project),
          center = m.center,
          scale = m.scale,
          translate = m.translate,
          clipExtent = m.clipExtent,
          x0 = null, y0, x1, y1; // clip extent

      m.scale = function(_) {
        return arguments.length ? (scale(_), reclip()) : scale();
      };

      m.translate = function(_) {
        return arguments.length ? (translate(_), reclip()) : translate();
      };

      m.center = function(_) {
        return arguments.length ? (center(_), reclip()) : center();
      };

      m.clipExtent = function(_) {
        return arguments.length ? ((_ == null ? x0 = y0 = x1 = y1 = null : (x0 = +_[0][0], y0 = +_[0][1], x1 = +_[1][0], y1 = +_[1][1])), reclip()) : x0 == null ? null : [[x0, y0], [x1, y1]];
      };

      function reclip() {
        var k = pi$8 * scale(),
            t = m(rotation$1(m.rotate()).invert([0, 0]));
        return clipExtent(x0 == null
            ? [[t[0] - k, t[1] - k], [t[0] + k, t[1] + k]] : project === mercatorRaw$1
            ? [[Math.max(t[0] - k, x0), y0], [Math.min(t[0] + k, x1), y1]]
            : [[x0, Math.max(t[1] - k, y0)], [x1, Math.min(t[1] + k, y1)]]);
      }

      return reclip();
    }

    function tany$1(y) {
      return tan$1((halfPi$6 + y) / 2);
    }

    function conicConformalRaw$1(y0, y1) {
      var cy0 = cos$4(y0),
          n = y0 === y1 ? sin$4(y0) : log$2(cy0 / cos$4(y1)) / log$2(tany$1(y1) / tany$1(y0)),
          f = cy0 * pow$2(tany$1(y0), n) / n;

      if (!n) return mercatorRaw$1;

      function project(x, y) {
        if (f > 0) { if (y < -halfPi$6 + epsilon$7) y = -halfPi$6 + epsilon$7; }
        else { if (y > halfPi$6 - epsilon$7) y = halfPi$6 - epsilon$7; }
        var r = f / pow$2(tany$1(y), n);
        return [r * sin$4(n * x), f - r * cos$4(n * x)];
      }

      project.invert = function(x, y) {
        var fy = f - y, r = sign$2(n) * sqrt$3(x * x + fy * fy);
        return [atan2$2(x, abs$2(fy)) / n * sign$2(fy), 2 * atan$1(pow$2(f / r, 1 / n)) - halfPi$6];
      };

      return project;
    }

    function conicConformal$1() {
      return conicProjection$1(conicConformalRaw$1)
          .scale(109.5)
          .parallels([30, 30]);
    }

    function equirectangularRaw$1(lambda, phi) {
      return [lambda, phi];
    }

    equirectangularRaw$1.invert = equirectangularRaw$1;

    function equirectangular$1() {
      return projection$1(equirectangularRaw$1)
          .scale(152.63);
    }

    function conicEquidistantRaw$1(y0, y1) {
      var cy0 = cos$4(y0),
          n = y0 === y1 ? sin$4(y0) : (cy0 - cos$4(y1)) / (y1 - y0),
          g = cy0 / n + y0;

      if (abs$2(n) < epsilon$7) return equirectangularRaw$1;

      function project(x, y) {
        var gy = g - y, nx = n * x;
        return [gy * sin$4(nx), g - gy * cos$4(nx)];
      }

      project.invert = function(x, y) {
        var gy = g - y;
        return [atan2$2(x, abs$2(gy)) / n * sign$2(gy), g - sign$2(n) * sqrt$3(x * x + gy * gy)];
      };

      return project;
    }

    function conicEquidistant$1() {
      return conicProjection$1(conicEquidistantRaw$1)
          .scale(131.154)
          .center([0, 13.9389]);
    }

    function gnomonicRaw$1(x, y) {
      var cy = cos$4(y), k = cos$4(x) * cy;
      return [cy * sin$4(x) / k, sin$4(y) / k];
    }

    gnomonicRaw$1.invert = azimuthalInvert$1(atan$1);

    function gnomonic$1() {
      return projection$1(gnomonicRaw$1)
          .scale(144.049)
          .clipAngle(60);
    }

    function scaleTranslate$1(kx, ky, tx, ty) {
      return kx === 1 && ky === 1 && tx === 0 && ty === 0 ? identity$e : transformer$4({
        point: function(x, y) {
          this.stream.point(x * kx + tx, y * ky + ty);
        }
      });
    }

    function identity$f() {
      var k = 1, tx = 0, ty = 0, sx = 1, sy = 1, transform = identity$e, // scale, translate and reflect
          x0 = null, y0, x1, y1, // clip extent
          postclip = identity$e,
          cache,
          cacheStream,
          projection;

      function reset() {
        cache = cacheStream = null;
        return projection;
      }

      return projection = {
        stream: function(stream) {
          return cache && cacheStream === stream ? cache : cache = transform(postclip(cacheStream = stream));
        },
        postclip: function(_) {
          return arguments.length ? (postclip = _, x0 = y0 = x1 = y1 = null, reset()) : postclip;
        },
        clipExtent: function(_) {
          return arguments.length ? (postclip = _ == null ? (x0 = y0 = x1 = y1 = null, identity$e) : clipRectangle$1(x0 = +_[0][0], y0 = +_[0][1], x1 = +_[1][0], y1 = +_[1][1]), reset()) : x0 == null ? null : [[x0, y0], [x1, y1]];
        },
        scale: function(_) {
          return arguments.length ? (transform = scaleTranslate$1((k = +_) * sx, k * sy, tx, ty), reset()) : k;
        },
        translate: function(_) {
          return arguments.length ? (transform = scaleTranslate$1(k * sx, k * sy, tx = +_[0], ty = +_[1]), reset()) : [tx, ty];
        },
        reflectX: function(_) {
          return arguments.length ? (transform = scaleTranslate$1(k * (sx = _ ? -1 : 1), k * sy, tx, ty), reset()) : sx < 0;
        },
        reflectY: function(_) {
          return arguments.length ? (transform = scaleTranslate$1(k * sx, k * (sy = _ ? -1 : 1), tx, ty), reset()) : sy < 0;
        },
        fitExtent: function(extent, object) {
          return fitExtent$1(projection, extent, object);
        },
        fitSize: function(size, object) {
          return fitSize$1(projection, size, object);
        },
        fitWidth: function(width, object) {
          return fitWidth$1(projection, width, object);
        },
        fitHeight: function(height, object) {
          return fitHeight$1(projection, height, object);
        }
      };
    }

    function naturalEarth1Raw$1(lambda, phi) {
      var phi2 = phi * phi, phi4 = phi2 * phi2;
      return [
        lambda * (0.8707 - 0.131979 * phi2 + phi4 * (-0.013791 + phi4 * (0.003971 * phi2 - 0.001529 * phi4))),
        phi * (1.007226 + phi2 * (0.015085 + phi4 * (-0.044475 + 0.028874 * phi2 - 0.005916 * phi4)))
      ];
    }

    naturalEarth1Raw$1.invert = function(x, y) {
      var phi = y, i = 25, delta;
      do {
        var phi2 = phi * phi, phi4 = phi2 * phi2;
        phi -= delta = (phi * (1.007226 + phi2 * (0.015085 + phi4 * (-0.044475 + 0.028874 * phi2 - 0.005916 * phi4))) - y) /
            (1.007226 + phi2 * (0.015085 * 3 + phi4 * (-0.044475 * 7 + 0.028874 * 9 * phi2 - 0.005916 * 11 * phi4)));
      } while (abs$2(delta) > epsilon$7 && --i > 0);
      return [
        x / (0.8707 + (phi2 = phi * phi) * (-0.131979 + phi2 * (-0.013791 + phi2 * phi2 * phi2 * (0.003971 - 0.001529 * phi2)))),
        phi
      ];
    };

    function naturalEarth1$1() {
      return projection$1(naturalEarth1Raw$1)
          .scale(175.295);
    }

    function orthographicRaw$1(x, y) {
      return [cos$4(y) * sin$4(x), sin$4(y)];
    }

    orthographicRaw$1.invert = azimuthalInvert$1(asin$2);

    function orthographic$1() {
      return projection$1(orthographicRaw$1)
          .scale(249.5)
          .clipAngle(90 + epsilon$7);
    }

    function stereographicRaw$1(x, y) {
      var cy = cos$4(y), k = 1 + cos$4(x) * cy;
      return [cy * sin$4(x) / k, sin$4(y) / k];
    }

    stereographicRaw$1.invert = azimuthalInvert$1(function(z) {
      return 2 * atan$1(z);
    });

    function stereographic$1() {
      return projection$1(stereographicRaw$1)
          .scale(250)
          .clipAngle(142);
    }

    function transverseMercatorRaw$1(lambda, phi) {
      return [log$2(tan$1((halfPi$6 + phi) / 2)), -lambda];
    }

    transverseMercatorRaw$1.invert = function(x, y) {
      return [-y, 2 * atan$1(exp$1(x)) - halfPi$6];
    };

    function transverseMercator$1() {
      var m = mercatorProjection$1(transverseMercatorRaw$1),
          center = m.center,
          rotate = m.rotate;

      m.center = function(_) {
        return arguments.length ? center([-_[1], _[0]]) : (_ = center(), [_[1], -_[0]]);
      };

      m.rotate = function(_) {
        return arguments.length ? rotate([_[0], _[1], _.length > 2 ? _[2] + 90 : 90]) : (_ = rotate(), [_[0], _[1], _[2] - 90]);
      };

      return rotate([0, 0, 90])
          .scale(159.155);
    }

    function defaultSeparation$2(a, b) {
      return a.parent === b.parent ? 1 : 2;
    }

    function meanX$1(children) {
      return children.reduce(meanXReduce$1, 0) / children.length;
    }

    function meanXReduce$1(x, c) {
      return x + c.x;
    }

    function maxY$1(children) {
      return 1 + children.reduce(maxYReduce$1, 0);
    }

    function maxYReduce$1(y, c) {
      return Math.max(y, c.y);
    }

    function leafLeft$1(node) {
      var children;
      while (children = node.children) node = children[0];
      return node;
    }

    function leafRight$1(node) {
      var children;
      while (children = node.children) node = children[children.length - 1];
      return node;
    }

    function cluster$1() {
      var separation = defaultSeparation$2,
          dx = 1,
          dy = 1,
          nodeSize = false;

      function cluster(root) {
        var previousNode,
            x = 0;

        // First walk, computing the initial x & y values.
        root.eachAfter(function(node) {
          var children = node.children;
          if (children) {
            node.x = meanX$1(children);
            node.y = maxY$1(children);
          } else {
            node.x = previousNode ? x += separation(node, previousNode) : 0;
            node.y = 0;
            previousNode = node;
          }
        });

        var left = leafLeft$1(root),
            right = leafRight$1(root),
            x0 = left.x - separation(left, right) / 2,
            x1 = right.x + separation(right, left) / 2;

        // Second walk, normalizing x & y to the desired size.
        return root.eachAfter(nodeSize ? function(node) {
          node.x = (node.x - root.x) * dx;
          node.y = (root.y - node.y) * dy;
        } : function(node) {
          node.x = (node.x - x0) / (x1 - x0) * dx;
          node.y = (1 - (root.y ? node.y / root.y : 1)) * dy;
        });
      }

      cluster.separation = function(x) {
        return arguments.length ? (separation = x, cluster) : separation;
      };

      cluster.size = function(x) {
        return arguments.length ? (nodeSize = false, dx = +x[0], dy = +x[1], cluster) : (nodeSize ? null : [dx, dy]);
      };

      cluster.nodeSize = function(x) {
        return arguments.length ? (nodeSize = true, dx = +x[0], dy = +x[1], cluster) : (nodeSize ? [dx, dy] : null);
      };

      return cluster;
    }

    function count$1(node) {
      var sum = 0,
          children = node.children,
          i = children && children.length;
      if (!i) sum = 1;
      else while (--i >= 0) sum += children[i].value;
      node.value = sum;
    }

    function node_count$1() {
      return this.eachAfter(count$1);
    }

    function node_each$1(callback) {
      var node = this, current, next = [node], children, i, n;
      do {
        current = next.reverse(), next = [];
        while (node = current.pop()) {
          callback(node), children = node.children;
          if (children) for (i = 0, n = children.length; i < n; ++i) {
            next.push(children[i]);
          }
        }
      } while (next.length);
      return this;
    }

    function node_eachBefore$1(callback) {
      var node = this, nodes = [node], children, i;
      while (node = nodes.pop()) {
        callback(node), children = node.children;
        if (children) for (i = children.length - 1; i >= 0; --i) {
          nodes.push(children[i]);
        }
      }
      return this;
    }

    function node_eachAfter$1(callback) {
      var node = this, nodes = [node], next = [], children, i, n;
      while (node = nodes.pop()) {
        next.push(node), children = node.children;
        if (children) for (i = 0, n = children.length; i < n; ++i) {
          nodes.push(children[i]);
        }
      }
      while (node = next.pop()) {
        callback(node);
      }
      return this;
    }

    function node_sum$1(value) {
      return this.eachAfter(function(node) {
        var sum = +value(node.data) || 0,
            children = node.children,
            i = children && children.length;
        while (--i >= 0) sum += children[i].value;
        node.value = sum;
      });
    }

    function node_sort$1(compare) {
      return this.eachBefore(function(node) {
        if (node.children) {
          node.children.sort(compare);
        }
      });
    }

    function node_path$1(end) {
      var start = this,
          ancestor = leastCommonAncestor$1(start, end),
          nodes = [start];
      while (start !== ancestor) {
        start = start.parent;
        nodes.push(start);
      }
      var k = nodes.length;
      while (end !== ancestor) {
        nodes.splice(k, 0, end);
        end = end.parent;
      }
      return nodes;
    }

    function leastCommonAncestor$1(a, b) {
      if (a === b) return a;
      var aNodes = a.ancestors(),
          bNodes = b.ancestors(),
          c = null;
      a = aNodes.pop();
      b = bNodes.pop();
      while (a === b) {
        c = a;
        a = aNodes.pop();
        b = bNodes.pop();
      }
      return c;
    }

    function node_ancestors$1() {
      var node = this, nodes = [node];
      while (node = node.parent) {
        nodes.push(node);
      }
      return nodes;
    }

    function node_descendants$1() {
      var nodes = [];
      this.each(function(node) {
        nodes.push(node);
      });
      return nodes;
    }

    function node_leaves$1() {
      var leaves = [];
      this.eachBefore(function(node) {
        if (!node.children) {
          leaves.push(node);
        }
      });
      return leaves;
    }

    function node_links$1() {
      var root = this, links = [];
      root.each(function(node) {
        if (node !== root) { // Don’t include the root’s parent, if any.
          links.push({source: node.parent, target: node});
        }
      });
      return links;
    }

    function hierarchy$1(data, children) {
      var root = new Node$3(data),
          valued = +data.value && (root.value = data.value),
          node,
          nodes = [root],
          child,
          childs,
          i,
          n;

      if (children == null) children = defaultChildren$1;

      while (node = nodes.pop()) {
        if (valued) node.value = +node.data.value;
        if ((childs = children(node.data)) && (n = childs.length)) {
          node.children = new Array(n);
          for (i = n - 1; i >= 0; --i) {
            nodes.push(child = node.children[i] = new Node$3(childs[i]));
            child.parent = node;
            child.depth = node.depth + 1;
          }
        }
      }

      return root.eachBefore(computeHeight$1);
    }

    function node_copy$1() {
      return hierarchy$1(this).eachBefore(copyData$1);
    }

    function defaultChildren$1(d) {
      return d.children;
    }

    function copyData$1(node) {
      node.data = node.data.data;
    }

    function computeHeight$1(node) {
      var height = 0;
      do node.height = height;
      while ((node = node.parent) && (node.height < ++height));
    }

    function Node$3(data) {
      this.data = data;
      this.depth =
      this.height = 0;
      this.parent = null;
    }

    Node$3.prototype = hierarchy$1.prototype = {
      constructor: Node$3,
      count: node_count$1,
      each: node_each$1,
      eachAfter: node_eachAfter$1,
      eachBefore: node_eachBefore$1,
      sum: node_sum$1,
      sort: node_sort$1,
      path: node_path$1,
      ancestors: node_ancestors$1,
      descendants: node_descendants$1,
      leaves: node_leaves$1,
      links: node_links$1,
      copy: node_copy$1
    };

    var slice$a = Array.prototype.slice;

    function shuffle$3(array) {
      var m = array.length,
          t,
          i;

      while (m) {
        i = Math.random() * m-- | 0;
        t = array[m];
        array[m] = array[i];
        array[i] = t;
      }

      return array;
    }

    function enclose$1(circles) {
      var i = 0, n = (circles = shuffle$3(slice$a.call(circles))).length, B = [], p, e;

      while (i < n) {
        p = circles[i];
        if (e && enclosesWeak$1(e, p)) ++i;
        else e = encloseBasis$1(B = extendBasis$1(B, p)), i = 0;
      }

      return e;
    }

    function extendBasis$1(B, p) {
      var i, j;

      if (enclosesWeakAll$1(p, B)) return [p];

      // If we get here then B must have at least one element.
      for (i = 0; i < B.length; ++i) {
        if (enclosesNot$1(p, B[i])
            && enclosesWeakAll$1(encloseBasis2$1(B[i], p), B)) {
          return [B[i], p];
        }
      }

      // If we get here then B must have at least two elements.
      for (i = 0; i < B.length - 1; ++i) {
        for (j = i + 1; j < B.length; ++j) {
          if (enclosesNot$1(encloseBasis2$1(B[i], B[j]), p)
              && enclosesNot$1(encloseBasis2$1(B[i], p), B[j])
              && enclosesNot$1(encloseBasis2$1(B[j], p), B[i])
              && enclosesWeakAll$1(encloseBasis3$1(B[i], B[j], p), B)) {
            return [B[i], B[j], p];
          }
        }
      }

      // If we get here then something is very wrong.
      throw new Error;
    }

    function enclosesNot$1(a, b) {
      var dr = a.r - b.r, dx = b.x - a.x, dy = b.y - a.y;
      return dr < 0 || dr * dr < dx * dx + dy * dy;
    }

    function enclosesWeak$1(a, b) {
      var dr = a.r - b.r + 1e-6, dx = b.x - a.x, dy = b.y - a.y;
      return dr > 0 && dr * dr > dx * dx + dy * dy;
    }

    function enclosesWeakAll$1(a, B) {
      for (var i = 0; i < B.length; ++i) {
        if (!enclosesWeak$1(a, B[i])) {
          return false;
        }
      }
      return true;
    }

    function encloseBasis$1(B) {
      switch (B.length) {
        case 1: return encloseBasis1$1(B[0]);
        case 2: return encloseBasis2$1(B[0], B[1]);
        case 3: return encloseBasis3$1(B[0], B[1], B[2]);
      }
    }

    function encloseBasis1$1(a) {
      return {
        x: a.x,
        y: a.y,
        r: a.r
      };
    }

    function encloseBasis2$1(a, b) {
      var x1 = a.x, y1 = a.y, r1 = a.r,
          x2 = b.x, y2 = b.y, r2 = b.r,
          x21 = x2 - x1, y21 = y2 - y1, r21 = r2 - r1,
          l = Math.sqrt(x21 * x21 + y21 * y21);
      return {
        x: (x1 + x2 + x21 / l * r21) / 2,
        y: (y1 + y2 + y21 / l * r21) / 2,
        r: (l + r1 + r2) / 2
      };
    }

    function encloseBasis3$1(a, b, c) {
      var x1 = a.x, y1 = a.y, r1 = a.r,
          x2 = b.x, y2 = b.y, r2 = b.r,
          x3 = c.x, y3 = c.y, r3 = c.r,
          a2 = x1 - x2,
          a3 = x1 - x3,
          b2 = y1 - y2,
          b3 = y1 - y3,
          c2 = r2 - r1,
          c3 = r3 - r1,
          d1 = x1 * x1 + y1 * y1 - r1 * r1,
          d2 = d1 - x2 * x2 - y2 * y2 + r2 * r2,
          d3 = d1 - x3 * x3 - y3 * y3 + r3 * r3,
          ab = a3 * b2 - a2 * b3,
          xa = (b2 * d3 - b3 * d2) / (ab * 2) - x1,
          xb = (b3 * c2 - b2 * c3) / ab,
          ya = (a3 * d2 - a2 * d3) / (ab * 2) - y1,
          yb = (a2 * c3 - a3 * c2) / ab,
          A = xb * xb + yb * yb - 1,
          B = 2 * (r1 + xa * xb + ya * yb),
          C = xa * xa + ya * ya - r1 * r1,
          r = -(A ? (B + Math.sqrt(B * B - 4 * A * C)) / (2 * A) : C / B);
      return {
        x: x1 + xa + xb * r,
        y: y1 + ya + yb * r,
        r: r
      };
    }

    function place$1(a, b, c) {
      var ax = a.x,
          ay = a.y,
          da = b.r + c.r,
          db = a.r + c.r,
          dx = b.x - ax,
          dy = b.y - ay,
          dc = dx * dx + dy * dy;
      if (dc) {
        var x = 0.5 + ((db *= db) - (da *= da)) / (2 * dc),
            y = Math.sqrt(Math.max(0, 2 * da * (db + dc) - (db -= dc) * db - da * da)) / (2 * dc);
        c.x = ax + x * dx + y * dy;
        c.y = ay + x * dy - y * dx;
      } else {
        c.x = ax + db;
        c.y = ay;
      }
    }

    function intersects$1(a, b) {
      var dx = b.x - a.x,
          dy = b.y - a.y,
          dr = a.r + b.r;
      return dr * dr - 1e-6 > dx * dx + dy * dy;
    }

    function score$1(node) {
      var a = node._,
          b = node.next._,
          ab = a.r + b.r,
          dx = (a.x * b.r + b.x * a.r) / ab,
          dy = (a.y * b.r + b.y * a.r) / ab;
      return dx * dx + dy * dy;
    }

    function Node$4(circle) {
      this._ = circle;
      this.next = null;
      this.previous = null;
    }

    function packEnclose$1(circles) {
      if (!(n = circles.length)) return 0;

      var a, b, c, n, aa, ca, i, j, k, sj, sk;

      // Place the first circle.
      a = circles[0], a.x = 0, a.y = 0;
      if (!(n > 1)) return a.r;

      // Place the second circle.
      b = circles[1], a.x = -b.r, b.x = a.r, b.y = 0;
      if (!(n > 2)) return a.r + b.r;

      // Place the third circle.
      place$1(b, a, c = circles[2]);

      // Initialize the front-chain using the first three circles a, b and c.
      a = new Node$4(a), b = new Node$4(b), c = new Node$4(c);
      a.next = c.previous = b;
      b.next = a.previous = c;
      c.next = b.previous = a;

      // Attempt to place each remaining circle…
      pack: for (i = 3; i < n; ++i) {
        place$1(a._, b._, c = circles[i]), c = new Node$4(c);

        // Find the closest intersecting circle on the front-chain, if any.
        // “Closeness” is determined by linear distance along the front-chain.
        // “Ahead” or “behind” is likewise determined by linear distance.
        j = b.next, k = a.previous, sj = b._.r, sk = a._.r;
        do {
          if (sj <= sk) {
            if (intersects$1(j._, c._)) {
              b = j, a.next = b, b.previous = a, --i;
              continue pack;
            }
            sj += j._.r, j = j.next;
          } else {
            if (intersects$1(k._, c._)) {
              a = k, a.next = b, b.previous = a, --i;
              continue pack;
            }
            sk += k._.r, k = k.previous;
          }
        } while (j !== k.next);

        // Success! Insert the new circle c between a and b.
        c.previous = a, c.next = b, a.next = b.previous = b = c;

        // Compute the new closest circle pair to the centroid.
        aa = score$1(a);
        while ((c = c.next) !== b) {
          if ((ca = score$1(c)) < aa) {
            a = c, aa = ca;
          }
        }
        b = a.next;
      }

      // Compute the enclosing circle of the front chain.
      a = [b._], c = b; while ((c = c.next) !== b) a.push(c._); c = enclose$1(a);

      // Translate the circles to put the enclosing circle around the origin.
      for (i = 0; i < n; ++i) a = circles[i], a.x -= c.x, a.y -= c.y;

      return c.r;
    }

    function siblings$1(circles) {
      packEnclose$1(circles);
      return circles;
    }

    function optional$1(f) {
      return f == null ? null : required$1(f);
    }

    function required$1(f) {
      if (typeof f !== "function") throw new Error;
      return f;
    }

    function constantZero$1() {
      return 0;
    }

    function constant$n(x) {
      return function() {
        return x;
      };
    }

    function defaultRadius$3(d) {
      return Math.sqrt(d.value);
    }

    function index$6() {
      var radius = null,
          dx = 1,
          dy = 1,
          padding = constantZero$1;

      function pack(root) {
        root.x = dx / 2, root.y = dy / 2;
        if (radius) {
          root.eachBefore(radiusLeaf$1(radius))
              .eachAfter(packChildren$1(padding, 0.5))
              .eachBefore(translateChild$1(1));
        } else {
          root.eachBefore(radiusLeaf$1(defaultRadius$3))
              .eachAfter(packChildren$1(constantZero$1, 1))
              .eachAfter(packChildren$1(padding, root.r / Math.min(dx, dy)))
              .eachBefore(translateChild$1(Math.min(dx, dy) / (2 * root.r)));
        }
        return root;
      }

      pack.radius = function(x) {
        return arguments.length ? (radius = optional$1(x), pack) : radius;
      };

      pack.size = function(x) {
        return arguments.length ? (dx = +x[0], dy = +x[1], pack) : [dx, dy];
      };

      pack.padding = function(x) {
        return arguments.length ? (padding = typeof x === "function" ? x : constant$n(+x), pack) : padding;
      };

      return pack;
    }

    function radiusLeaf$1(radius) {
      return function(node) {
        if (!node.children) {
          node.r = Math.max(0, +radius(node) || 0);
        }
      };
    }

    function packChildren$1(padding, k) {
      return function(node) {
        if (children = node.children) {
          var children,
              i,
              n = children.length,
              r = padding(node) * k || 0,
              e;

          if (r) for (i = 0; i < n; ++i) children[i].r += r;
          e = packEnclose$1(children);
          if (r) for (i = 0; i < n; ++i) children[i].r -= r;
          node.r = e + r;
        }
      };
    }

    function translateChild$1(k) {
      return function(node) {
        var parent = node.parent;
        node.r *= k;
        if (parent) {
          node.x = parent.x + k * node.x;
          node.y = parent.y + k * node.y;
        }
      };
    }

    function roundNode$1(node) {
      node.x0 = Math.round(node.x0);
      node.y0 = Math.round(node.y0);
      node.x1 = Math.round(node.x1);
      node.y1 = Math.round(node.y1);
    }

    function treemapDice$1(parent, x0, y0, x1, y1) {
      var nodes = parent.children,
          node,
          i = -1,
          n = nodes.length,
          k = parent.value && (x1 - x0) / parent.value;

      while (++i < n) {
        node = nodes[i], node.y0 = y0, node.y1 = y1;
        node.x0 = x0, node.x1 = x0 += node.value * k;
      }
    }

    function partition$1() {
      var dx = 1,
          dy = 1,
          padding = 0,
          round = false;

      function partition(root) {
        var n = root.height + 1;
        root.x0 =
        root.y0 = padding;
        root.x1 = dx;
        root.y1 = dy / n;
        root.eachBefore(positionNode(dy, n));
        if (round) root.eachBefore(roundNode$1);
        return root;
      }

      function positionNode(dy, n) {
        return function(node) {
          if (node.children) {
            treemapDice$1(node, node.x0, dy * (node.depth + 1) / n, node.x1, dy * (node.depth + 2) / n);
          }
          var x0 = node.x0,
              y0 = node.y0,
              x1 = node.x1 - padding,
              y1 = node.y1 - padding;
          if (x1 < x0) x0 = x1 = (x0 + x1) / 2;
          if (y1 < y0) y0 = y1 = (y0 + y1) / 2;
          node.x0 = x0;
          node.y0 = y0;
          node.x1 = x1;
          node.y1 = y1;
        };
      }

      partition.round = function(x) {
        return arguments.length ? (round = !!x, partition) : round;
      };

      partition.size = function(x) {
        return arguments.length ? (dx = +x[0], dy = +x[1], partition) : [dx, dy];
      };

      partition.padding = function(x) {
        return arguments.length ? (padding = +x, partition) : padding;
      };

      return partition;
    }

    var keyPrefix$4 = "$", // Protect against keys like “__proto__”.
        preroot$1 = {depth: -1},
        ambiguous$1 = {};

    function defaultId$1(d) {
      return d.id;
    }

    function defaultParentId$1(d) {
      return d.parentId;
    }

    function stratify$1() {
      var id = defaultId$1,
          parentId = defaultParentId$1;

      function stratify(data) {
        var d,
            i,
            n = data.length,
            root,
            parent,
            node,
            nodes = new Array(n),
            nodeId,
            nodeKey,
            nodeByKey = {};

        for (i = 0; i < n; ++i) {
          d = data[i], node = nodes[i] = new Node$3(d);
          if ((nodeId = id(d, i, data)) != null && (nodeId += "")) {
            nodeKey = keyPrefix$4 + (node.id = nodeId);
            nodeByKey[nodeKey] = nodeKey in nodeByKey ? ambiguous$1 : node;
          }
        }

        for (i = 0; i < n; ++i) {
          node = nodes[i], nodeId = parentId(data[i], i, data);
          if (nodeId == null || !(nodeId += "")) {
            if (root) throw new Error("multiple roots");
            root = node;
          } else {
            parent = nodeByKey[keyPrefix$4 + nodeId];
            if (!parent) throw new Error("missing: " + nodeId);
            if (parent === ambiguous$1) throw new Error("ambiguous: " + nodeId);
            if (parent.children) parent.children.push(node);
            else parent.children = [node];
            node.parent = parent;
          }
        }

        if (!root) throw new Error("no root");
        root.parent = preroot$1;
        root.eachBefore(function(node) { node.depth = node.parent.depth + 1; --n; }).eachBefore(computeHeight$1);
        root.parent = null;
        if (n > 0) throw new Error("cycle");

        return root;
      }

      stratify.id = function(x) {
        return arguments.length ? (id = required$1(x), stratify) : id;
      };

      stratify.parentId = function(x) {
        return arguments.length ? (parentId = required$1(x), stratify) : parentId;
      };

      return stratify;
    }

    function defaultSeparation$3(a, b) {
      return a.parent === b.parent ? 1 : 2;
    }

    // function radialSeparation(a, b) {
    //   return (a.parent === b.parent ? 1 : 2) / a.depth;
    // }

    // This function is used to traverse the left contour of a subtree (or
    // subforest). It returns the successor of v on this contour. This successor is
    // either given by the leftmost child of v or by the thread of v. The function
    // returns null if and only if v is on the highest level of its subtree.
    function nextLeft$1(v) {
      var children = v.children;
      return children ? children[0] : v.t;
    }

    // This function works analogously to nextLeft.
    function nextRight$1(v) {
      var children = v.children;
      return children ? children[children.length - 1] : v.t;
    }

    // Shifts the current subtree rooted at w+. This is done by increasing
    // prelim(w+) and mod(w+) by shift.
    function moveSubtree$1(wm, wp, shift) {
      var change = shift / (wp.i - wm.i);
      wp.c -= change;
      wp.s += shift;
      wm.c += change;
      wp.z += shift;
      wp.m += shift;
    }

    // All other shifts, applied to the smaller subtrees between w- and w+, are
    // performed by this function. To prepare the shifts, we have to adjust
    // change(w+), shift(w+), and change(w-).
    function executeShifts$1(v) {
      var shift = 0,
          change = 0,
          children = v.children,
          i = children.length,
          w;
      while (--i >= 0) {
        w = children[i];
        w.z += shift;
        w.m += shift;
        shift += w.s + (change += w.c);
      }
    }

    // If vi-’s ancestor is a sibling of v, returns vi-’s ancestor. Otherwise,
    // returns the specified (default) ancestor.
    function nextAncestor$1(vim, v, ancestor) {
      return vim.a.parent === v.parent ? vim.a : ancestor;
    }

    function TreeNode$1(node, i) {
      this._ = node;
      this.parent = null;
      this.children = null;
      this.A = null; // default ancestor
      this.a = this; // ancestor
      this.z = 0; // prelim
      this.m = 0; // mod
      this.c = 0; // change
      this.s = 0; // shift
      this.t = null; // thread
      this.i = i; // number
    }

    TreeNode$1.prototype = Object.create(Node$3.prototype);

    function treeRoot$1(root) {
      var tree = new TreeNode$1(root, 0),
          node,
          nodes = [tree],
          child,
          children,
          i,
          n;

      while (node = nodes.pop()) {
        if (children = node._.children) {
          node.children = new Array(n = children.length);
          for (i = n - 1; i >= 0; --i) {
            nodes.push(child = node.children[i] = new TreeNode$1(children[i], i));
            child.parent = node;
          }
        }
      }

      (tree.parent = new TreeNode$1(null, 0)).children = [tree];
      return tree;
    }

    // Node-link tree diagram using the Reingold-Tilford "tidy" algorithm
    function tree$1() {
      var separation = defaultSeparation$3,
          dx = 1,
          dy = 1,
          nodeSize = null;

      function tree(root) {
        var t = treeRoot$1(root);

        // Compute the layout using Buchheim et al.’s algorithm.
        t.eachAfter(firstWalk), t.parent.m = -t.z;
        t.eachBefore(secondWalk);

        // If a fixed node size is specified, scale x and y.
        if (nodeSize) root.eachBefore(sizeNode);

        // If a fixed tree size is specified, scale x and y based on the extent.
        // Compute the left-most, right-most, and depth-most nodes for extents.
        else {
          var left = root,
              right = root,
              bottom = root;
          root.eachBefore(function(node) {
            if (node.x < left.x) left = node;
            if (node.x > right.x) right = node;
            if (node.depth > bottom.depth) bottom = node;
          });
          var s = left === right ? 1 : separation(left, right) / 2,
              tx = s - left.x,
              kx = dx / (right.x + s + tx),
              ky = dy / (bottom.depth || 1);
          root.eachBefore(function(node) {
            node.x = (node.x + tx) * kx;
            node.y = node.depth * ky;
          });
        }

        return root;
      }

      // Computes a preliminary x-coordinate for v. Before that, FIRST WALK is
      // applied recursively to the children of v, as well as the function
      // APPORTION. After spacing out the children by calling EXECUTE SHIFTS, the
      // node v is placed to the midpoint of its outermost children.
      function firstWalk(v) {
        var children = v.children,
            siblings = v.parent.children,
            w = v.i ? siblings[v.i - 1] : null;
        if (children) {
          executeShifts$1(v);
          var midpoint = (children[0].z + children[children.length - 1].z) / 2;
          if (w) {
            v.z = w.z + separation(v._, w._);
            v.m = v.z - midpoint;
          } else {
            v.z = midpoint;
          }
        } else if (w) {
          v.z = w.z + separation(v._, w._);
        }
        v.parent.A = apportion(v, w, v.parent.A || siblings[0]);
      }

      // Computes all real x-coordinates by summing up the modifiers recursively.
      function secondWalk(v) {
        v._.x = v.z + v.parent.m;
        v.m += v.parent.m;
      }

      // The core of the algorithm. Here, a new subtree is combined with the
      // previous subtrees. Threads are used to traverse the inside and outside
      // contours of the left and right subtree up to the highest common level. The
      // vertices used for the traversals are vi+, vi-, vo-, and vo+, where the
      // superscript o means outside and i means inside, the subscript - means left
      // subtree and + means right subtree. For summing up the modifiers along the
      // contour, we use respective variables si+, si-, so-, and so+. Whenever two
      // nodes of the inside contours conflict, we compute the left one of the
      // greatest uncommon ancestors using the function ANCESTOR and call MOVE
      // SUBTREE to shift the subtree and prepare the shifts of smaller subtrees.
      // Finally, we add a new thread (if necessary).
      function apportion(v, w, ancestor) {
        if (w) {
          var vip = v,
              vop = v,
              vim = w,
              vom = vip.parent.children[0],
              sip = vip.m,
              sop = vop.m,
              sim = vim.m,
              som = vom.m,
              shift;
          while (vim = nextRight$1(vim), vip = nextLeft$1(vip), vim && vip) {
            vom = nextLeft$1(vom);
            vop = nextRight$1(vop);
            vop.a = v;
            shift = vim.z + sim - vip.z - sip + separation(vim._, vip._);
            if (shift > 0) {
              moveSubtree$1(nextAncestor$1(vim, v, ancestor), v, shift);
              sip += shift;
              sop += shift;
            }
            sim += vim.m;
            sip += vip.m;
            som += vom.m;
            sop += vop.m;
          }
          if (vim && !nextRight$1(vop)) {
            vop.t = vim;
            vop.m += sim - sop;
          }
          if (vip && !nextLeft$1(vom)) {
            vom.t = vip;
            vom.m += sip - som;
            ancestor = v;
          }
        }
        return ancestor;
      }

      function sizeNode(node) {
        node.x *= dx;
        node.y = node.depth * dy;
      }

      tree.separation = function(x) {
        return arguments.length ? (separation = x, tree) : separation;
      };

      tree.size = function(x) {
        return arguments.length ? (nodeSize = false, dx = +x[0], dy = +x[1], tree) : (nodeSize ? null : [dx, dy]);
      };

      tree.nodeSize = function(x) {
        return arguments.length ? (nodeSize = true, dx = +x[0], dy = +x[1], tree) : (nodeSize ? [dx, dy] : null);
      };

      return tree;
    }

    function treemapSlice$1(parent, x0, y0, x1, y1) {
      var nodes = parent.children,
          node,
          i = -1,
          n = nodes.length,
          k = parent.value && (y1 - y0) / parent.value;

      while (++i < n) {
        node = nodes[i], node.x0 = x0, node.x1 = x1;
        node.y0 = y0, node.y1 = y0 += node.value * k;
      }
    }

    var phi$1 = (1 + Math.sqrt(5)) / 2;

    function squarifyRatio$1(ratio, parent, x0, y0, x1, y1) {
      var rows = [],
          nodes = parent.children,
          row,
          nodeValue,
          i0 = 0,
          i1 = 0,
          n = nodes.length,
          dx, dy,
          value = parent.value,
          sumValue,
          minValue,
          maxValue,
          newRatio,
          minRatio,
          alpha,
          beta;

      while (i0 < n) {
        dx = x1 - x0, dy = y1 - y0;

        // Find the next non-empty node.
        do sumValue = nodes[i1++].value; while (!sumValue && i1 < n);
        minValue = maxValue = sumValue;
        alpha = Math.max(dy / dx, dx / dy) / (value * ratio);
        beta = sumValue * sumValue * alpha;
        minRatio = Math.max(maxValue / beta, beta / minValue);

        // Keep adding nodes while the aspect ratio maintains or improves.
        for (; i1 < n; ++i1) {
          sumValue += nodeValue = nodes[i1].value;
          if (nodeValue < minValue) minValue = nodeValue;
          if (nodeValue > maxValue) maxValue = nodeValue;
          beta = sumValue * sumValue * alpha;
          newRatio = Math.max(maxValue / beta, beta / minValue);
          if (newRatio > minRatio) { sumValue -= nodeValue; break; }
          minRatio = newRatio;
        }

        // Position and record the row orientation.
        rows.push(row = {value: sumValue, dice: dx < dy, children: nodes.slice(i0, i1)});
        if (row.dice) treemapDice$1(row, x0, y0, x1, value ? y0 += dy * sumValue / value : y1);
        else treemapSlice$1(row, x0, y0, value ? x0 += dx * sumValue / value : x1, y1);
        value -= sumValue, i0 = i1;
      }

      return rows;
    }

    var squarify$1 = (function custom(ratio) {

      function squarify(parent, x0, y0, x1, y1) {
        squarifyRatio$1(ratio, parent, x0, y0, x1, y1);
      }

      squarify.ratio = function(x) {
        return custom((x = +x) > 1 ? x : 1);
      };

      return squarify;
    })(phi$1);

    function index$7() {
      var tile = squarify$1,
          round = false,
          dx = 1,
          dy = 1,
          paddingStack = [0],
          paddingInner = constantZero$1,
          paddingTop = constantZero$1,
          paddingRight = constantZero$1,
          paddingBottom = constantZero$1,
          paddingLeft = constantZero$1;

      function treemap(root) {
        root.x0 =
        root.y0 = 0;
        root.x1 = dx;
        root.y1 = dy;
        root.eachBefore(positionNode);
        paddingStack = [0];
        if (round) root.eachBefore(roundNode$1);
        return root;
      }

      function positionNode(node) {
        var p = paddingStack[node.depth],
            x0 = node.x0 + p,
            y0 = node.y0 + p,
            x1 = node.x1 - p,
            y1 = node.y1 - p;
        if (x1 < x0) x0 = x1 = (x0 + x1) / 2;
        if (y1 < y0) y0 = y1 = (y0 + y1) / 2;
        node.x0 = x0;
        node.y0 = y0;
        node.x1 = x1;
        node.y1 = y1;
        if (node.children) {
          p = paddingStack[node.depth + 1] = paddingInner(node) / 2;
          x0 += paddingLeft(node) - p;
          y0 += paddingTop(node) - p;
          x1 -= paddingRight(node) - p;
          y1 -= paddingBottom(node) - p;
          if (x1 < x0) x0 = x1 = (x0 + x1) / 2;
          if (y1 < y0) y0 = y1 = (y0 + y1) / 2;
          tile(node, x0, y0, x1, y1);
        }
      }

      treemap.round = function(x) {
        return arguments.length ? (round = !!x, treemap) : round;
      };

      treemap.size = function(x) {
        return arguments.length ? (dx = +x[0], dy = +x[1], treemap) : [dx, dy];
      };

      treemap.tile = function(x) {
        return arguments.length ? (tile = required$1(x), treemap) : tile;
      };

      treemap.padding = function(x) {
        return arguments.length ? treemap.paddingInner(x).paddingOuter(x) : treemap.paddingInner();
      };

      treemap.paddingInner = function(x) {
        return arguments.length ? (paddingInner = typeof x === "function" ? x : constant$n(+x), treemap) : paddingInner;
      };

      treemap.paddingOuter = function(x) {
        return arguments.length ? treemap.paddingTop(x).paddingRight(x).paddingBottom(x).paddingLeft(x) : treemap.paddingTop();
      };

      treemap.paddingTop = function(x) {
        return arguments.length ? (paddingTop = typeof x === "function" ? x : constant$n(+x), treemap) : paddingTop;
      };

      treemap.paddingRight = function(x) {
        return arguments.length ? (paddingRight = typeof x === "function" ? x : constant$n(+x), treemap) : paddingRight;
      };

      treemap.paddingBottom = function(x) {
        return arguments.length ? (paddingBottom = typeof x === "function" ? x : constant$n(+x), treemap) : paddingBottom;
      };

      treemap.paddingLeft = function(x) {
        return arguments.length ? (paddingLeft = typeof x === "function" ? x : constant$n(+x), treemap) : paddingLeft;
      };

      return treemap;
    }

    function binary$1(parent, x0, y0, x1, y1) {
      var nodes = parent.children,
          i, n = nodes.length,
          sum, sums = new Array(n + 1);

      for (sums[0] = sum = i = 0; i < n; ++i) {
        sums[i + 1] = sum += nodes[i].value;
      }

      partition(0, n, parent.value, x0, y0, x1, y1);

      function partition(i, j, value, x0, y0, x1, y1) {
        if (i >= j - 1) {
          var node = nodes[i];
          node.x0 = x0, node.y0 = y0;
          node.x1 = x1, node.y1 = y1;
          return;
        }

        var valueOffset = sums[i],
            valueTarget = (value / 2) + valueOffset,
            k = i + 1,
            hi = j - 1;

        while (k < hi) {
          var mid = k + hi >>> 1;
          if (sums[mid] < valueTarget) k = mid + 1;
          else hi = mid;
        }

        if ((valueTarget - sums[k - 1]) < (sums[k] - valueTarget) && i + 1 < k) --k;

        var valueLeft = sums[k] - valueOffset,
            valueRight = value - valueLeft;

        if ((x1 - x0) > (y1 - y0)) {
          var xk = (x0 * valueRight + x1 * valueLeft) / value;
          partition(i, k, valueLeft, x0, y0, xk, y1);
          partition(k, j, valueRight, xk, y0, x1, y1);
        } else {
          var yk = (y0 * valueRight + y1 * valueLeft) / value;
          partition(i, k, valueLeft, x0, y0, x1, yk);
          partition(k, j, valueRight, x0, yk, x1, y1);
        }
      }
    }

    function sliceDice$1(parent, x0, y0, x1, y1) {
      (parent.depth & 1 ? treemapSlice$1 : treemapDice$1)(parent, x0, y0, x1, y1);
    }

    var resquarify$1 = (function custom(ratio) {

      function resquarify(parent, x0, y0, x1, y1) {
        if ((rows = parent._squarify) && (rows.ratio === ratio)) {
          var rows,
              row,
              nodes,
              i,
              j = -1,
              n,
              m = rows.length,
              value = parent.value;

          while (++j < m) {
            row = rows[j], nodes = row.children;
            for (i = row.value = 0, n = nodes.length; i < n; ++i) row.value += nodes[i].value;
            if (row.dice) treemapDice$1(row, x0, y0, x1, y0 += (y1 - y0) * row.value / value);
            else treemapSlice$1(row, x0, y0, x0 += (x1 - x0) * row.value / value, y1);
            value -= row.value;
          }
        } else {
          parent._squarify = rows = squarifyRatio$1(ratio, parent, x0, y0, x1, y1);
          rows.ratio = ratio;
        }
      }

      resquarify.ratio = function(x) {
        return custom((x = +x) > 1 ? x : 1);
      };

      return resquarify;
    })(phi$1);

    function define$3(constructor, factory, prototype) {
      constructor.prototype = factory.prototype = prototype;
      prototype.constructor = constructor;
    }

    function extend$3(parent, definition) {
      var prototype = Object.create(parent.prototype);
      for (var key in definition) prototype[key] = definition[key];
      return prototype;
    }

    function Color$3() {}

    var darker$3 = 0.7;
    var brighter$3 = 1 / darker$3;

    var reI$3 = "\\s*([+-]?\\d+)\\s*",
        reN$3 = "\\s*([+-]?\\d*\\.?\\d+(?:[eE][+-]?\\d+)?)\\s*",
        reP$3 = "\\s*([+-]?\\d*\\.?\\d+(?:[eE][+-]?\\d+)?)%\\s*",
        reHex$2 = /^#([0-9a-f]{3,8})$/,
        reRgbInteger$3 = new RegExp("^rgb\\(" + [reI$3, reI$3, reI$3] + "\\)$"),
        reRgbPercent$3 = new RegExp("^rgb\\(" + [reP$3, reP$3, reP$3] + "\\)$"),
        reRgbaInteger$3 = new RegExp("^rgba\\(" + [reI$3, reI$3, reI$3, reN$3] + "\\)$"),
        reRgbaPercent$3 = new RegExp("^rgba\\(" + [reP$3, reP$3, reP$3, reN$3] + "\\)$"),
        reHslPercent$3 = new RegExp("^hsl\\(" + [reN$3, reP$3, reP$3] + "\\)$"),
        reHslaPercent$3 = new RegExp("^hsla\\(" + [reN$3, reP$3, reP$3, reN$3] + "\\)$");

    var named$3 = {
      aliceblue: 0xf0f8ff,
      antiquewhite: 0xfaebd7,
      aqua: 0x00ffff,
      aquamarine: 0x7fffd4,
      azure: 0xf0ffff,
      beige: 0xf5f5dc,
      bisque: 0xffe4c4,
      black: 0x000000,
      blanchedalmond: 0xffebcd,
      blue: 0x0000ff,
      blueviolet: 0x8a2be2,
      brown: 0xa52a2a,
      burlywood: 0xdeb887,
      cadetblue: 0x5f9ea0,
      chartreuse: 0x7fff00,
      chocolate: 0xd2691e,
      coral: 0xff7f50,
      cornflowerblue: 0x6495ed,
      cornsilk: 0xfff8dc,
      crimson: 0xdc143c,
      cyan: 0x00ffff,
      darkblue: 0x00008b,
      darkcyan: 0x008b8b,
      darkgoldenrod: 0xb8860b,
      darkgray: 0xa9a9a9,
      darkgreen: 0x006400,
      darkgrey: 0xa9a9a9,
      darkkhaki: 0xbdb76b,
      darkmagenta: 0x8b008b,
      darkolivegreen: 0x556b2f,
      darkorange: 0xff8c00,
      darkorchid: 0x9932cc,
      darkred: 0x8b0000,
      darksalmon: 0xe9967a,
      darkseagreen: 0x8fbc8f,
      darkslateblue: 0x483d8b,
      darkslategray: 0x2f4f4f,
      darkslategrey: 0x2f4f4f,
      darkturquoise: 0x00ced1,
      darkviolet: 0x9400d3,
      deeppink: 0xff1493,
      deepskyblue: 0x00bfff,
      dimgray: 0x696969,
      dimgrey: 0x696969,
      dodgerblue: 0x1e90ff,
      firebrick: 0xb22222,
      floralwhite: 0xfffaf0,
      forestgreen: 0x228b22,
      fuchsia: 0xff00ff,
      gainsboro: 0xdcdcdc,
      ghostwhite: 0xf8f8ff,
      gold: 0xffd700,
      goldenrod: 0xdaa520,
      gray: 0x808080,
      green: 0x008000,
      greenyellow: 0xadff2f,
      grey: 0x808080,
      honeydew: 0xf0fff0,
      hotpink: 0xff69b4,
      indianred: 0xcd5c5c,
      indigo: 0x4b0082,
      ivory: 0xfffff0,
      khaki: 0xf0e68c,
      lavender: 0xe6e6fa,
      lavenderblush: 0xfff0f5,
      lawngreen: 0x7cfc00,
      lemonchiffon: 0xfffacd,
      lightblue: 0xadd8e6,
      lightcoral: 0xf08080,
      lightcyan: 0xe0ffff,
      lightgoldenrodyellow: 0xfafad2,
      lightgray: 0xd3d3d3,
      lightgreen: 0x90ee90,
      lightgrey: 0xd3d3d3,
      lightpink: 0xffb6c1,
      lightsalmon: 0xffa07a,
      lightseagreen: 0x20b2aa,
      lightskyblue: 0x87cefa,
      lightslategray: 0x778899,
      lightslategrey: 0x778899,
      lightsteelblue: 0xb0c4de,
      lightyellow: 0xffffe0,
      lime: 0x00ff00,
      limegreen: 0x32cd32,
      linen: 0xfaf0e6,
      magenta: 0xff00ff,
      maroon: 0x800000,
      mediumaquamarine: 0x66cdaa,
      mediumblue: 0x0000cd,
      mediumorchid: 0xba55d3,
      mediumpurple: 0x9370db,
      mediumseagreen: 0x3cb371,
      mediumslateblue: 0x7b68ee,
      mediumspringgreen: 0x00fa9a,
      mediumturquoise: 0x48d1cc,
      mediumvioletred: 0xc71585,
      midnightblue: 0x191970,
      mintcream: 0xf5fffa,
      mistyrose: 0xffe4e1,
      moccasin: 0xffe4b5,
      navajowhite: 0xffdead,
      navy: 0x000080,
      oldlace: 0xfdf5e6,
      olive: 0x808000,
      olivedrab: 0x6b8e23,
      orange: 0xffa500,
      orangered: 0xff4500,
      orchid: 0xda70d6,
      palegoldenrod: 0xeee8aa,
      palegreen: 0x98fb98,
      paleturquoise: 0xafeeee,
      palevioletred: 0xdb7093,
      papayawhip: 0xffefd5,
      peachpuff: 0xffdab9,
      peru: 0xcd853f,
      pink: 0xffc0cb,
      plum: 0xdda0dd,
      powderblue: 0xb0e0e6,
      purple: 0x800080,
      rebeccapurple: 0x663399,
      red: 0xff0000,
      rosybrown: 0xbc8f8f,
      royalblue: 0x4169e1,
      saddlebrown: 0x8b4513,
      salmon: 0xfa8072,
      sandybrown: 0xf4a460,
      seagreen: 0x2e8b57,
      seashell: 0xfff5ee,
      sienna: 0xa0522d,
      silver: 0xc0c0c0,
      skyblue: 0x87ceeb,
      slateblue: 0x6a5acd,
      slategray: 0x708090,
      slategrey: 0x708090,
      snow: 0xfffafa,
      springgreen: 0x00ff7f,
      steelblue: 0x4682b4,
      tan: 0xd2b48c,
      teal: 0x008080,
      thistle: 0xd8bfd8,
      tomato: 0xff6347,
      turquoise: 0x40e0d0,
      violet: 0xee82ee,
      wheat: 0xf5deb3,
      white: 0xffffff,
      whitesmoke: 0xf5f5f5,
      yellow: 0xffff00,
      yellowgreen: 0x9acd32
    };

    define$3(Color$3, color$3, {
      copy: function(channels) {
        return Object.assign(new this.constructor, this, channels);
      },
      displayable: function() {
        return this.rgb().displayable();
      },
      hex: color_formatHex$2, // Deprecated! Use color.formatHex.
      formatHex: color_formatHex$2,
      formatHsl: color_formatHsl$2,
      formatRgb: color_formatRgb$2,
      toString: color_formatRgb$2
    });

    function color_formatHex$2() {
      return this.rgb().formatHex();
    }

    function color_formatHsl$2() {
      return hslConvert$3(this).formatHsl();
    }

    function color_formatRgb$2() {
      return this.rgb().formatRgb();
    }

    function color$3(format) {
      var m, l;
      format = (format + "").trim().toLowerCase();
      return (m = reHex$2.exec(format)) ? (l = m[1].length, m = parseInt(m[1], 16), l === 6 ? rgbn$3(m) // #ff0000
          : l === 3 ? new Rgb$3((m >> 8 & 0xf) | (m >> 4 & 0xf0), (m >> 4 & 0xf) | (m & 0xf0), ((m & 0xf) << 4) | (m & 0xf), 1) // #f00
          : l === 8 ? rgba$3(m >> 24 & 0xff, m >> 16 & 0xff, m >> 8 & 0xff, (m & 0xff) / 0xff) // #ff000000
          : l === 4 ? rgba$3((m >> 12 & 0xf) | (m >> 8 & 0xf0), (m >> 8 & 0xf) | (m >> 4 & 0xf0), (m >> 4 & 0xf) | (m & 0xf0), (((m & 0xf) << 4) | (m & 0xf)) / 0xff) // #f000
          : null) // invalid hex
          : (m = reRgbInteger$3.exec(format)) ? new Rgb$3(m[1], m[2], m[3], 1) // rgb(255, 0, 0)
          : (m = reRgbPercent$3.exec(format)) ? new Rgb$3(m[1] * 255 / 100, m[2] * 255 / 100, m[3] * 255 / 100, 1) // rgb(100%, 0%, 0%)
          : (m = reRgbaInteger$3.exec(format)) ? rgba$3(m[1], m[2], m[3], m[4]) // rgba(255, 0, 0, 1)
          : (m = reRgbaPercent$3.exec(format)) ? rgba$3(m[1] * 255 / 100, m[2] * 255 / 100, m[3] * 255 / 100, m[4]) // rgb(100%, 0%, 0%, 1)
          : (m = reHslPercent$3.exec(format)) ? hsla$3(m[1], m[2] / 100, m[3] / 100, 1) // hsl(120, 50%, 50%)
          : (m = reHslaPercent$3.exec(format)) ? hsla$3(m[1], m[2] / 100, m[3] / 100, m[4]) // hsla(120, 50%, 50%, 1)
          : named$3.hasOwnProperty(format) ? rgbn$3(named$3[format]) // eslint-disable-line no-prototype-builtins
          : format === "transparent" ? new Rgb$3(NaN, NaN, NaN, 0)
          : null;
    }

    function rgbn$3(n) {
      return new Rgb$3(n >> 16 & 0xff, n >> 8 & 0xff, n & 0xff, 1);
    }

    function rgba$3(r, g, b, a) {
      if (a <= 0) r = g = b = NaN;
      return new Rgb$3(r, g, b, a);
    }

    function rgbConvert$3(o) {
      if (!(o instanceof Color$3)) o = color$3(o);
      if (!o) return new Rgb$3;
      o = o.rgb();
      return new Rgb$3(o.r, o.g, o.b, o.opacity);
    }

    function rgb$3(r, g, b, opacity) {
      return arguments.length === 1 ? rgbConvert$3(r) : new Rgb$3(r, g, b, opacity == null ? 1 : opacity);
    }

    function Rgb$3(r, g, b, opacity) {
      this.r = +r;
      this.g = +g;
      this.b = +b;
      this.opacity = +opacity;
    }

    define$3(Rgb$3, rgb$3, extend$3(Color$3, {
      brighter: function(k) {
        k = k == null ? brighter$3 : Math.pow(brighter$3, k);
        return new Rgb$3(this.r * k, this.g * k, this.b * k, this.opacity);
      },
      darker: function(k) {
        k = k == null ? darker$3 : Math.pow(darker$3, k);
        return new Rgb$3(this.r * k, this.g * k, this.b * k, this.opacity);
      },
      rgb: function() {
        return this;
      },
      displayable: function() {
        return (-0.5 <= this.r && this.r < 255.5)
            && (-0.5 <= this.g && this.g < 255.5)
            && (-0.5 <= this.b && this.b < 255.5)
            && (0 <= this.opacity && this.opacity <= 1);
      },
      hex: rgb_formatHex$2, // Deprecated! Use color.formatHex.
      formatHex: rgb_formatHex$2,
      formatRgb: rgb_formatRgb$2,
      toString: rgb_formatRgb$2
    }));

    function rgb_formatHex$2() {
      return "#" + hex$2(this.r) + hex$2(this.g) + hex$2(this.b);
    }

    function rgb_formatRgb$2() {
      var a = this.opacity; a = isNaN(a) ? 1 : Math.max(0, Math.min(1, a));
      return (a === 1 ? "rgb(" : "rgba(")
          + Math.max(0, Math.min(255, Math.round(this.r) || 0)) + ", "
          + Math.max(0, Math.min(255, Math.round(this.g) || 0)) + ", "
          + Math.max(0, Math.min(255, Math.round(this.b) || 0))
          + (a === 1 ? ")" : ", " + a + ")");
    }

    function hex$2(value) {
      value = Math.max(0, Math.min(255, Math.round(value) || 0));
      return (value < 16 ? "0" : "") + value.toString(16);
    }

    function hsla$3(h, s, l, a) {
      if (a <= 0) h = s = l = NaN;
      else if (l <= 0 || l >= 1) h = s = NaN;
      else if (s <= 0) h = NaN;
      return new Hsl$3(h, s, l, a);
    }

    function hslConvert$3(o) {
      if (o instanceof Hsl$3) return new Hsl$3(o.h, o.s, o.l, o.opacity);
      if (!(o instanceof Color$3)) o = color$3(o);
      if (!o) return new Hsl$3;
      if (o instanceof Hsl$3) return o;
      o = o.rgb();
      var r = o.r / 255,
          g = o.g / 255,
          b = o.b / 255,
          min = Math.min(r, g, b),
          max = Math.max(r, g, b),
          h = NaN,
          s = max - min,
          l = (max + min) / 2;
      if (s) {
        if (r === max) h = (g - b) / s + (g < b) * 6;
        else if (g === max) h = (b - r) / s + 2;
        else h = (r - g) / s + 4;
        s /= l < 0.5 ? max + min : 2 - max - min;
        h *= 60;
      } else {
        s = l > 0 && l < 1 ? 0 : h;
      }
      return new Hsl$3(h, s, l, o.opacity);
    }

    function hsl$5(h, s, l, opacity) {
      return arguments.length === 1 ? hslConvert$3(h) : new Hsl$3(h, s, l, opacity == null ? 1 : opacity);
    }

    function Hsl$3(h, s, l, opacity) {
      this.h = +h;
      this.s = +s;
      this.l = +l;
      this.opacity = +opacity;
    }

    define$3(Hsl$3, hsl$5, extend$3(Color$3, {
      brighter: function(k) {
        k = k == null ? brighter$3 : Math.pow(brighter$3, k);
        return new Hsl$3(this.h, this.s, this.l * k, this.opacity);
      },
      darker: function(k) {
        k = k == null ? darker$3 : Math.pow(darker$3, k);
        return new Hsl$3(this.h, this.s, this.l * k, this.opacity);
      },
      rgb: function() {
        var h = this.h % 360 + (this.h < 0) * 360,
            s = isNaN(h) || isNaN(this.s) ? 0 : this.s,
            l = this.l,
            m2 = l + (l < 0.5 ? l : 1 - l) * s,
            m1 = 2 * l - m2;
        return new Rgb$3(
          hsl2rgb$3(h >= 240 ? h - 240 : h + 120, m1, m2),
          hsl2rgb$3(h, m1, m2),
          hsl2rgb$3(h < 120 ? h + 240 : h - 120, m1, m2),
          this.opacity
        );
      },
      displayable: function() {
        return (0 <= this.s && this.s <= 1 || isNaN(this.s))
            && (0 <= this.l && this.l <= 1)
            && (0 <= this.opacity && this.opacity <= 1);
      },
      formatHsl: function() {
        var a = this.opacity; a = isNaN(a) ? 1 : Math.max(0, Math.min(1, a));
        return (a === 1 ? "hsl(" : "hsla(")
            + (this.h || 0) + ", "
            + (this.s || 0) * 100 + "%, "
            + (this.l || 0) * 100 + "%"
            + (a === 1 ? ")" : ", " + a + ")");
      }
    }));

    /* From FvD 13.37, CSS Color Module Level 3 */
    function hsl2rgb$3(h, m1, m2) {
      return (h < 60 ? m1 + (m2 - m1) * h / 60
          : h < 180 ? m2
          : h < 240 ? m1 + (m2 - m1) * (240 - h) / 60
          : m1) * 255;
    }

    var deg2rad$2 = Math.PI / 180;
    var rad2deg$2 = 180 / Math.PI;

    // https://observablehq.com/@mbostock/lab-and-rgb
    var K$1 = 18,
        Xn$2 = 0.96422,
        Yn$2 = 1,
        Zn$2 = 0.82521,
        t0$3 = 4 / 29,
        t1$3 = 6 / 29,
        t2$2 = 3 * t1$3 * t1$3,
        t3$2 = t1$3 * t1$3 * t1$3;

    function labConvert$2(o) {
      if (o instanceof Lab$2) return new Lab$2(o.l, o.a, o.b, o.opacity);
      if (o instanceof Hcl$2) return hcl2lab$1(o);
      if (!(o instanceof Rgb$3)) o = rgbConvert$3(o);
      var r = rgb2lrgb$1(o.r),
          g = rgb2lrgb$1(o.g),
          b = rgb2lrgb$1(o.b),
          y = xyz2lab$2((0.2225045 * r + 0.7168786 * g + 0.0606169 * b) / Yn$2), x, z;
      if (r === g && g === b) x = z = y; else {
        x = xyz2lab$2((0.4360747 * r + 0.3850649 * g + 0.1430804 * b) / Xn$2);
        z = xyz2lab$2((0.0139322 * r + 0.0971045 * g + 0.7141733 * b) / Zn$2);
      }
      return new Lab$2(116 * y - 16, 500 * (x - y), 200 * (y - z), o.opacity);
    }

    function lab$3(l, a, b, opacity) {
      return arguments.length === 1 ? labConvert$2(l) : new Lab$2(l, a, b, opacity == null ? 1 : opacity);
    }

    function Lab$2(l, a, b, opacity) {
      this.l = +l;
      this.a = +a;
      this.b = +b;
      this.opacity = +opacity;
    }

    define$3(Lab$2, lab$3, extend$3(Color$3, {
      brighter: function(k) {
        return new Lab$2(this.l + K$1 * (k == null ? 1 : k), this.a, this.b, this.opacity);
      },
      darker: function(k) {
        return new Lab$2(this.l - K$1 * (k == null ? 1 : k), this.a, this.b, this.opacity);
      },
      rgb: function() {
        var y = (this.l + 16) / 116,
            x = isNaN(this.a) ? y : y + this.a / 500,
            z = isNaN(this.b) ? y : y - this.b / 200;
        x = Xn$2 * lab2xyz$2(x);
        y = Yn$2 * lab2xyz$2(y);
        z = Zn$2 * lab2xyz$2(z);
        return new Rgb$3(
          lrgb2rgb$1( 3.1338561 * x - 1.6168667 * y - 0.4906146 * z),
          lrgb2rgb$1(-0.9787684 * x + 1.9161415 * y + 0.0334540 * z),
          lrgb2rgb$1( 0.0719453 * x - 0.2289914 * y + 1.4052427 * z),
          this.opacity
        );
      }
    }));

    function xyz2lab$2(t) {
      return t > t3$2 ? Math.pow(t, 1 / 3) : t / t2$2 + t0$3;
    }

    function lab2xyz$2(t) {
      return t > t1$3 ? t * t * t : t2$2 * (t - t0$3);
    }

    function lrgb2rgb$1(x) {
      return 255 * (x <= 0.0031308 ? 12.92 * x : 1.055 * Math.pow(x, 1 / 2.4) - 0.055);
    }

    function rgb2lrgb$1(x) {
      return (x /= 255) <= 0.04045 ? x / 12.92 : Math.pow((x + 0.055) / 1.055, 2.4);
    }

    function hclConvert$2(o) {
      if (o instanceof Hcl$2) return new Hcl$2(o.h, o.c, o.l, o.opacity);
      if (!(o instanceof Lab$2)) o = labConvert$2(o);
      if (o.a === 0 && o.b === 0) return new Hcl$2(NaN, 0 < o.l && o.l < 100 ? 0 : NaN, o.l, o.opacity);
      var h = Math.atan2(o.b, o.a) * rad2deg$2;
      return new Hcl$2(h < 0 ? h + 360 : h, Math.sqrt(o.a * o.a + o.b * o.b), o.l, o.opacity);
    }

    function hcl$4(h, c, l, opacity) {
      return arguments.length === 1 ? hclConvert$2(h) : new Hcl$2(h, c, l, opacity == null ? 1 : opacity);
    }

    function Hcl$2(h, c, l, opacity) {
      this.h = +h;
      this.c = +c;
      this.l = +l;
      this.opacity = +opacity;
    }

    function hcl2lab$1(o) {
      if (isNaN(o.h)) return new Lab$2(o.l, 0, 0, o.opacity);
      var h = o.h * deg2rad$2;
      return new Lab$2(o.l, Math.cos(h) * o.c, Math.sin(h) * o.c, o.opacity);
    }

    define$3(Hcl$2, hcl$4, extend$3(Color$3, {
      brighter: function(k) {
        return new Hcl$2(this.h, this.c, this.l + K$1 * (k == null ? 1 : k), this.opacity);
      },
      darker: function(k) {
        return new Hcl$2(this.h, this.c, this.l - K$1 * (k == null ? 1 : k), this.opacity);
      },
      rgb: function() {
        return hcl2lab$1(this).rgb();
      }
    }));

    var A$2 = -0.14861,
        B$2 = +1.78277,
        C$2 = -0.29227,
        D$2 = -0.90649,
        E$2 = +1.97294,
        ED$2 = E$2 * D$2,
        EB$2 = E$2 * B$2,
        BC_DA$2 = B$2 * C$2 - D$2 * A$2;

    function cubehelixConvert$2(o) {
      if (o instanceof Cubehelix$2) return new Cubehelix$2(o.h, o.s, o.l, o.opacity);
      if (!(o instanceof Rgb$3)) o = rgbConvert$3(o);
      var r = o.r / 255,
          g = o.g / 255,
          b = o.b / 255,
          l = (BC_DA$2 * b + ED$2 * r - EB$2 * g) / (BC_DA$2 + ED$2 - EB$2),
          bl = b - l,
          k = (E$2 * (g - l) - C$2 * bl) / D$2,
          s = Math.sqrt(k * k + bl * bl) / (E$2 * l * (1 - l)), // NaN if l=0 or l=1
          h = s ? Math.atan2(k, bl) * rad2deg$2 - 120 : NaN;
      return new Cubehelix$2(h < 0 ? h + 360 : h, s, l, o.opacity);
    }

    function cubehelix$5(h, s, l, opacity) {
      return arguments.length === 1 ? cubehelixConvert$2(h) : new Cubehelix$2(h, s, l, opacity == null ? 1 : opacity);
    }

    function Cubehelix$2(h, s, l, opacity) {
      this.h = +h;
      this.s = +s;
      this.l = +l;
      this.opacity = +opacity;
    }

    define$3(Cubehelix$2, cubehelix$5, extend$3(Color$3, {
      brighter: function(k) {
        k = k == null ? brighter$3 : Math.pow(brighter$3, k);
        return new Cubehelix$2(this.h, this.s, this.l * k, this.opacity);
      },
      darker: function(k) {
        k = k == null ? darker$3 : Math.pow(darker$3, k);
        return new Cubehelix$2(this.h, this.s, this.l * k, this.opacity);
      },
      rgb: function() {
        var h = isNaN(this.h) ? 0 : (this.h + 120) * deg2rad$2,
            l = +this.l,
            a = isNaN(this.s) ? 0 : this.s * l * (1 - l),
            cosh = Math.cos(h),
            sinh = Math.sin(h);
        return new Rgb$3(
          255 * (l + a * (A$2 * cosh + B$2 * sinh)),
          255 * (l + a * (C$2 * cosh + D$2 * sinh)),
          255 * (l + a * (E$2 * cosh)),
          this.opacity
        );
      }
    }));

    function basis$3(t1, v0, v1, v2, v3) {
      var t2 = t1 * t1, t3 = t2 * t1;
      return ((1 - 3 * t1 + 3 * t2 - t3) * v0
          + (4 - 6 * t2 + 3 * t3) * v1
          + (1 + 3 * t1 + 3 * t2 - 3 * t3) * v2
          + t3 * v3) / 6;
    }

    function basis$4(values) {
      var n = values.length - 1;
      return function(t) {
        var i = t <= 0 ? (t = 0) : t >= 1 ? (t = 1, n - 1) : Math.floor(t * n),
            v1 = values[i],
            v2 = values[i + 1],
            v0 = i > 0 ? values[i - 1] : 2 * v1 - v2,
            v3 = i < n - 1 ? values[i + 2] : 2 * v2 - v1;
        return basis$3((t - i / n) * n, v0, v1, v2, v3);
      };
    }

    function basisClosed$2(values) {
      var n = values.length;
      return function(t) {
        var i = Math.floor(((t %= 1) < 0 ? ++t : t) * n),
            v0 = values[(i + n - 1) % n],
            v1 = values[i % n],
            v2 = values[(i + 1) % n],
            v3 = values[(i + 2) % n];
        return basis$3((t - i / n) * n, v0, v1, v2, v3);
      };
    }

    function constant$o(x) {
      return function() {
        return x;
      };
    }

    function linear$5(a, d) {
      return function(t) {
        return a + t * d;
      };
    }

    function exponential$3(a, b, y) {
      return a = Math.pow(a, y), b = Math.pow(b, y) - a, y = 1 / y, function(t) {
        return Math.pow(a + t * b, y);
      };
    }

    function hue$2(a, b) {
      var d = b - a;
      return d ? linear$5(a, d > 180 || d < -180 ? d - 360 * Math.round(d / 360) : d) : constant$o(isNaN(a) ? b : a);
    }

    function gamma$2(y) {
      return (y = +y) === 1 ? nogamma$2 : function(a, b) {
        return b - a ? exponential$3(a, b, y) : constant$o(isNaN(a) ? b : a);
      };
    }

    function nogamma$2(a, b) {
      var d = b - a;
      return d ? linear$5(a, d) : constant$o(isNaN(a) ? b : a);
    }

    var rgb$4 = (function rgbGamma(y) {
      var color = gamma$2(y);

      function rgb(start, end) {
        var r = color((start = rgb$3(start)).r, (end = rgb$3(end)).r),
            g = color(start.g, end.g),
            b = color(start.b, end.b),
            opacity = nogamma$2(start.opacity, end.opacity);
        return function(t) {
          start.r = r(t);
          start.g = g(t);
          start.b = b(t);
          start.opacity = opacity(t);
          return start + "";
        };
      }

      rgb.gamma = rgbGamma;

      return rgb;
    })(1);

    function rgbSpline$1(spline) {
      return function(colors) {
        var n = colors.length,
            r = new Array(n),
            g = new Array(n),
            b = new Array(n),
            i, color;
        for (i = 0; i < n; ++i) {
          color = rgb$3(colors[i]);
          r[i] = color.r || 0;
          g[i] = color.g || 0;
          b[i] = color.b || 0;
        }
        r = spline(r);
        g = spline(g);
        b = spline(b);
        color.opacity = 1;
        return function(t) {
          color.r = r(t);
          color.g = g(t);
          color.b = b(t);
          return color + "";
        };
      };
    }

    var rgbBasis$1 = rgbSpline$1(basis$4);
    var rgbBasisClosed$1 = rgbSpline$1(basisClosed$2);

    function array$5(a, b) {
      var nb = b ? b.length : 0,
          na = a ? Math.min(nb, a.length) : 0,
          x = new Array(na),
          c = new Array(nb),
          i;

      for (i = 0; i < na; ++i) x[i] = value(a[i], b[i]);
      for (; i < nb; ++i) c[i] = b[i];

      return function(t) {
        for (i = 0; i < na; ++i) c[i] = x[i](t);
        return c;
      };
    }

    function date$3(a, b) {
      var d = new Date;
      return a = +a, b -= a, function(t) {
        return d.setTime(a + b * t), d;
      };
    }

    function number$6(a, b) {
      return a = +a, b -= a, function(t) {
        return a + b * t;
      };
    }

    function object$4(a, b) {
      var i = {},
          c = {},
          k;

      if (a === null || typeof a !== "object") a = {};
      if (b === null || typeof b !== "object") b = {};

      for (k in b) {
        if (k in a) {
          i[k] = value(a[k], b[k]);
        } else {
          c[k] = b[k];
        }
      }

      return function(t) {
        for (k in i) c[k] = i[k](t);
        return c;
      };
    }

    var reA$2 = /[-+]?(?:\d+\.?\d*|\.?\d+)(?:[eE][-+]?\d+)?/g,
        reB$2 = new RegExp(reA$2.source, "g");

    function zero$2(b) {
      return function() {
        return b;
      };
    }

    function one$2(b) {
      return function(t) {
        return b(t) + "";
      };
    }

    function string(a, b) {
      var bi = reA$2.lastIndex = reB$2.lastIndex = 0, // scan index for next number in b
          am, // current match in a
          bm, // current match in b
          bs, // string preceding current number in b, if any
          i = -1, // index in s
          s = [], // string constants and placeholders
          q = []; // number interpolators

      // Coerce inputs to strings.
      a = a + "", b = b + "";

      // Interpolate pairs of numbers in a & b.
      while ((am = reA$2.exec(a))
          && (bm = reB$2.exec(b))) {
        if ((bs = bm.index) > bi) { // a string precedes the next number in b
          bs = b.slice(bi, bs);
          if (s[i]) s[i] += bs; // coalesce with previous string
          else s[++i] = bs;
        }
        if ((am = am[0]) === (bm = bm[0])) { // numbers in a & b match
          if (s[i]) s[i] += bm; // coalesce with previous string
          else s[++i] = bm;
        } else { // interpolate non-matching numbers
          s[++i] = null;
          q.push({i: i, x: number$6(am, bm)});
        }
        bi = reB$2.lastIndex;
      }

      // Add remains of b.
      if (bi < b.length) {
        bs = b.slice(bi);
        if (s[i]) s[i] += bs; // coalesce with previous string
        else s[++i] = bs;
      }

      // Special optimization for only a single match.
      // Otherwise, interpolate each of the numbers and rejoin the string.
      return s.length < 2 ? (q[0]
          ? one$2(q[0].x)
          : zero$2(b))
          : (b = q.length, function(t) {
              for (var i = 0, o; i < b; ++i) s[(o = q[i]).i] = o.x(t);
              return s.join("");
            });
    }

    function value(a, b) {
      var t = typeof b, c;
      return b == null || t === "boolean" ? constant$o(b)
          : (t === "number" ? number$6
          : t === "string" ? ((c = color$3(b)) ? (b = c, rgb$4) : string)
          : b instanceof color$3 ? rgb$4
          : b instanceof Date ? date$3
          : Array.isArray(b) ? array$5
          : typeof b.valueOf !== "function" && typeof b.toString !== "function" || isNaN(b) ? object$4
          : number$6)(a, b);
    }

    function round(a, b) {
      return a = +a, b -= a, function(t) {
        return Math.round(a + b * t);
      };
    }

    var degrees$4 = 180 / Math.PI;

    var identity$g = {
      translateX: 0,
      translateY: 0,
      rotate: 0,
      skewX: 0,
      scaleX: 1,
      scaleY: 1
    };

    function decompose$2(a, b, c, d, e, f) {
      var scaleX, scaleY, skewX;
      if (scaleX = Math.sqrt(a * a + b * b)) a /= scaleX, b /= scaleX;
      if (skewX = a * c + b * d) c -= a * skewX, d -= b * skewX;
      if (scaleY = Math.sqrt(c * c + d * d)) c /= scaleY, d /= scaleY, skewX /= scaleY;
      if (a * d < b * c) a = -a, b = -b, skewX = -skewX, scaleX = -scaleX;
      return {
        translateX: e,
        translateY: f,
        rotate: Math.atan2(b, a) * degrees$4,
        skewX: Math.atan(skewX) * degrees$4,
        scaleX: scaleX,
        scaleY: scaleY
      };
    }

    var cssNode$2,
        cssRoot$2,
        cssView$2,
        svgNode$2;

    function parseCss$3(value) {
      if (value === "none") return identity$g;
      if (!cssNode$2) cssNode$2 = document.createElement("DIV"), cssRoot$2 = document.documentElement, cssView$2 = document.defaultView;
      cssNode$2.style.transform = value;
      value = cssView$2.getComputedStyle(cssRoot$2.appendChild(cssNode$2), null).getPropertyValue("transform");
      cssRoot$2.removeChild(cssNode$2);
      value = value.slice(7, -1).split(",");
      return decompose$2(+value[0], +value[1], +value[2], +value[3], +value[4], +value[5]);
    }

    function parseSvg$2(value) {
      if (value == null) return identity$g;
      if (!svgNode$2) svgNode$2 = document.createElementNS("http://www.w3.org/2000/svg", "g");
      svgNode$2.setAttribute("transform", value);
      if (!(value = svgNode$2.transform.baseVal.consolidate())) return identity$g;
      value = value.matrix;
      return decompose$2(value.a, value.b, value.c, value.d, value.e, value.f);
    }

    function interpolateTransform$2(parse, pxComma, pxParen, degParen) {

      function pop(s) {
        return s.length ? s.pop() + " " : "";
      }

      function translate(xa, ya, xb, yb, s, q) {
        if (xa !== xb || ya !== yb) {
          var i = s.push("translate(", null, pxComma, null, pxParen);
          q.push({i: i - 4, x: number$6(xa, xb)}, {i: i - 2, x: number$6(ya, yb)});
        } else if (xb || yb) {
          s.push("translate(" + xb + pxComma + yb + pxParen);
        }
      }

      function rotate(a, b, s, q) {
        if (a !== b) {
          if (a - b > 180) b += 360; else if (b - a > 180) a += 360; // shortest path
          q.push({i: s.push(pop(s) + "rotate(", null, degParen) - 2, x: number$6(a, b)});
        } else if (b) {
          s.push(pop(s) + "rotate(" + b + degParen);
        }
      }

      function skewX(a, b, s, q) {
        if (a !== b) {
          q.push({i: s.push(pop(s) + "skewX(", null, degParen) - 2, x: number$6(a, b)});
        } else if (b) {
          s.push(pop(s) + "skewX(" + b + degParen);
        }
      }

      function scale(xa, ya, xb, yb, s, q) {
        if (xa !== xb || ya !== yb) {
          var i = s.push(pop(s) + "scale(", null, ",", null, ")");
          q.push({i: i - 4, x: number$6(xa, xb)}, {i: i - 2, x: number$6(ya, yb)});
        } else if (xb !== 1 || yb !== 1) {
          s.push(pop(s) + "scale(" + xb + "," + yb + ")");
        }
      }

      return function(a, b) {
        var s = [], // string constants and placeholders
            q = []; // number interpolators
        a = parse(a), b = parse(b);
        translate(a.translateX, a.translateY, b.translateX, b.translateY, s, q);
        rotate(a.rotate, b.rotate, s, q);
        skewX(a.skewX, b.skewX, s, q);
        scale(a.scaleX, a.scaleY, b.scaleX, b.scaleY, s, q);
        a = b = null; // gc
        return function(t) {
          var i = -1, n = q.length, o;
          while (++i < n) s[(o = q[i]).i] = o.x(t);
          return s.join("");
        };
      };
    }

    var interpolateTransformCss$2 = interpolateTransform$2(parseCss$3, "px, ", "px)", "deg)");
    var interpolateTransformSvg$2 = interpolateTransform$2(parseSvg$2, ", ", ")", ")");

    var rho$1 = Math.SQRT2,
        rho2$1 = 2,
        rho4$1 = 4,
        epsilon2$4 = 1e-12;

    function cosh$1(x) {
      return ((x = Math.exp(x)) + 1 / x) / 2;
    }

    function sinh$1(x) {
      return ((x = Math.exp(x)) - 1 / x) / 2;
    }

    function tanh$1(x) {
      return ((x = Math.exp(2 * x)) - 1) / (x + 1);
    }

    // p0 = [ux0, uy0, w0]
    // p1 = [ux1, uy1, w1]
    function zoom$1(p0, p1) {
      var ux0 = p0[0], uy0 = p0[1], w0 = p0[2],
          ux1 = p1[0], uy1 = p1[1], w1 = p1[2],
          dx = ux1 - ux0,
          dy = uy1 - uy0,
          d2 = dx * dx + dy * dy,
          i,
          S;

      // Special case for u0 ≅ u1.
      if (d2 < epsilon2$4) {
        S = Math.log(w1 / w0) / rho$1;
        i = function(t) {
          return [
            ux0 + t * dx,
            uy0 + t * dy,
            w0 * Math.exp(rho$1 * t * S)
          ];
        };
      }

      // General case.
      else {
        var d1 = Math.sqrt(d2),
            b0 = (w1 * w1 - w0 * w0 + rho4$1 * d2) / (2 * w0 * rho2$1 * d1),
            b1 = (w1 * w1 - w0 * w0 - rho4$1 * d2) / (2 * w1 * rho2$1 * d1),
            r0 = Math.log(Math.sqrt(b0 * b0 + 1) - b0),
            r1 = Math.log(Math.sqrt(b1 * b1 + 1) - b1);
        S = (r1 - r0) / rho$1;
        i = function(t) {
          var s = t * S,
              coshr0 = cosh$1(r0),
              u = w0 / (rho2$1 * d1) * (coshr0 * tanh$1(rho$1 * s + r0) - sinh$1(r0));
          return [
            ux0 + u * dx,
            uy0 + u * dy,
            w0 * coshr0 / cosh$1(rho$1 * s + r0)
          ];
        };
      }

      i.duration = S * 1000;

      return i;
    }

    function hsl$6(hue) {
      return function(start, end) {
        var h = hue((start = hsl$5(start)).h, (end = hsl$5(end)).h),
            s = nogamma$2(start.s, end.s),
            l = nogamma$2(start.l, end.l),
            opacity = nogamma$2(start.opacity, end.opacity);
        return function(t) {
          start.h = h(t);
          start.s = s(t);
          start.l = l(t);
          start.opacity = opacity(t);
          return start + "";
        };
      }
    }

    var hsl$7 = hsl$6(hue$2);
    var hslLong$1 = hsl$6(nogamma$2);

    function lab$4(start, end) {
      var l = nogamma$2((start = lab$3(start)).l, (end = lab$3(end)).l),
          a = nogamma$2(start.a, end.a),
          b = nogamma$2(start.b, end.b),
          opacity = nogamma$2(start.opacity, end.opacity);
      return function(t) {
        start.l = l(t);
        start.a = a(t);
        start.b = b(t);
        start.opacity = opacity(t);
        return start + "";
      };
    }

    function hcl$5(hue) {
      return function(start, end) {
        var h = hue((start = hcl$4(start)).h, (end = hcl$4(end)).h),
            c = nogamma$2(start.c, end.c),
            l = nogamma$2(start.l, end.l),
            opacity = nogamma$2(start.opacity, end.opacity);
        return function(t) {
          start.h = h(t);
          start.c = c(t);
          start.l = l(t);
          start.opacity = opacity(t);
          return start + "";
        };
      }
    }

    var hcl$6 = hcl$5(hue$2);
    var hclLong$1 = hcl$5(nogamma$2);

    function cubehelix$6(hue) {
      return (function cubehelixGamma(y) {
        y = +y;

        function cubehelix(start, end) {
          var h = hue((start = cubehelix$5(start)).h, (end = cubehelix$5(end)).h),
              s = nogamma$2(start.s, end.s),
              l = nogamma$2(start.l, end.l),
              opacity = nogamma$2(start.opacity, end.opacity);
          return function(t) {
            start.h = h(t);
            start.s = s(t);
            start.l = l(Math.pow(t, y));
            start.opacity = opacity(t);
            return start + "";
          };
        }

        cubehelix.gamma = cubehelixGamma;

        return cubehelix;
      })(1);
    }

    var cubehelix$7 = cubehelix$6(hue$2);
    var cubehelixLong$1 = cubehelix$6(nogamma$2);

    function quantize$2(interpolator, n) {
      var samples = new Array(n);
      for (var i = 0; i < n; ++i) samples[i] = interpolator(i / (n - 1));
      return samples;
    }

    var pi$9 = Math.PI,
        tau$9 = 2 * pi$9,
        epsilon$8 = 1e-6,
        tauEpsilon$2 = tau$9 - epsilon$8;

    function Path$2() {
      this._x0 = this._y0 = // start of current subpath
      this._x1 = this._y1 = null; // end of current subpath
      this._ = "";
    }

    function path$2() {
      return new Path$2;
    }

    Path$2.prototype = path$2.prototype = {
      constructor: Path$2,
      moveTo: function(x, y) {
        this._ += "M" + (this._x0 = this._x1 = +x) + "," + (this._y0 = this._y1 = +y);
      },
      closePath: function() {
        if (this._x1 !== null) {
          this._x1 = this._x0, this._y1 = this._y0;
          this._ += "Z";
        }
      },
      lineTo: function(x, y) {
        this._ += "L" + (this._x1 = +x) + "," + (this._y1 = +y);
      },
      quadraticCurveTo: function(x1, y1, x, y) {
        this._ += "Q" + (+x1) + "," + (+y1) + "," + (this._x1 = +x) + "," + (this._y1 = +y);
      },
      bezierCurveTo: function(x1, y1, x2, y2, x, y) {
        this._ += "C" + (+x1) + "," + (+y1) + "," + (+x2) + "," + (+y2) + "," + (this._x1 = +x) + "," + (this._y1 = +y);
      },
      arcTo: function(x1, y1, x2, y2, r) {
        x1 = +x1, y1 = +y1, x2 = +x2, y2 = +y2, r = +r;
        var x0 = this._x1,
            y0 = this._y1,
            x21 = x2 - x1,
            y21 = y2 - y1,
            x01 = x0 - x1,
            y01 = y0 - y1,
            l01_2 = x01 * x01 + y01 * y01;

        // Is the radius negative? Error.
        if (r < 0) throw new Error("negative radius: " + r);

        // Is this path empty? Move to (x1,y1).
        if (this._x1 === null) {
          this._ += "M" + (this._x1 = x1) + "," + (this._y1 = y1);
        }

        // Or, is (x1,y1) coincident with (x0,y0)? Do nothing.
        else if (!(l01_2 > epsilon$8)) ;

        // Or, are (x0,y0), (x1,y1) and (x2,y2) collinear?
        // Equivalently, is (x1,y1) coincident with (x2,y2)?
        // Or, is the radius zero? Line to (x1,y1).
        else if (!(Math.abs(y01 * x21 - y21 * x01) > epsilon$8) || !r) {
          this._ += "L" + (this._x1 = x1) + "," + (this._y1 = y1);
        }

        // Otherwise, draw an arc!
        else {
          var x20 = x2 - x0,
              y20 = y2 - y0,
              l21_2 = x21 * x21 + y21 * y21,
              l20_2 = x20 * x20 + y20 * y20,
              l21 = Math.sqrt(l21_2),
              l01 = Math.sqrt(l01_2),
              l = r * Math.tan((pi$9 - Math.acos((l21_2 + l01_2 - l20_2) / (2 * l21 * l01))) / 2),
              t01 = l / l01,
              t21 = l / l21;

          // If the start tangent is not coincident with (x0,y0), line to.
          if (Math.abs(t01 - 1) > epsilon$8) {
            this._ += "L" + (x1 + t01 * x01) + "," + (y1 + t01 * y01);
          }

          this._ += "A" + r + "," + r + ",0,0," + (+(y01 * x20 > x01 * y20)) + "," + (this._x1 = x1 + t21 * x21) + "," + (this._y1 = y1 + t21 * y21);
        }
      },
      arc: function(x, y, r, a0, a1, ccw) {
        x = +x, y = +y, r = +r;
        var dx = r * Math.cos(a0),
            dy = r * Math.sin(a0),
            x0 = x + dx,
            y0 = y + dy,
            cw = 1 ^ ccw,
            da = ccw ? a0 - a1 : a1 - a0;

        // Is the radius negative? Error.
        if (r < 0) throw new Error("negative radius: " + r);

        // Is this path empty? Move to (x0,y0).
        if (this._x1 === null) {
          this._ += "M" + x0 + "," + y0;
        }

        // Or, is (x0,y0) not coincident with the previous point? Line to (x0,y0).
        else if (Math.abs(this._x1 - x0) > epsilon$8 || Math.abs(this._y1 - y0) > epsilon$8) {
          this._ += "L" + x0 + "," + y0;
        }

        // Is this arc empty? We’re done.
        if (!r) return;

        // Does the angle go the wrong way? Flip the direction.
        if (da < 0) da = da % tau$9 + tau$9;

        // Is this a complete circle? Draw two arcs to complete the circle.
        if (da > tauEpsilon$2) {
          this._ += "A" + r + "," + r + ",0,1," + cw + "," + (x - dx) + "," + (y - dy) + "A" + r + "," + r + ",0,1," + cw + "," + (this._x1 = x0) + "," + (this._y1 = y0);
        }

        // Is this arc non-empty? Draw an arc!
        else if (da > epsilon$8) {
          this._ += "A" + r + "," + r + ",0," + (+(da >= pi$9)) + "," + cw + "," + (this._x1 = x + r * Math.cos(a1)) + "," + (this._y1 = y + r * Math.sin(a1));
        }
      },
      rect: function(x, y, w, h) {
        this._ += "M" + (this._x0 = this._x1 = +x) + "," + (this._y0 = this._y1 = +y) + "h" + (+w) + "v" + (+h) + "h" + (-w) + "Z";
      },
      toString: function() {
        return this._;
      }
    };

    function area$5(polygon) {
      var i = -1,
          n = polygon.length,
          a,
          b = polygon[n - 1],
          area = 0;

      while (++i < n) {
        a = b;
        b = polygon[i];
        area += a[1] * b[0] - a[0] * b[1];
      }

      return area / 2;
    }

    function centroid$3(polygon) {
      var i = -1,
          n = polygon.length,
          x = 0,
          y = 0,
          a,
          b = polygon[n - 1],
          c,
          k = 0;

      while (++i < n) {
        a = b;
        b = polygon[i];
        k += c = a[0] * b[1] - b[0] * a[1];
        x += (a[0] + b[0]) * c;
        y += (a[1] + b[1]) * c;
      }

      return k *= 3, [x / k, y / k];
    }

    // Returns the 2D cross product of AB and AC vectors, i.e., the z-component of
    // the 3D cross product in a quadrant I Cartesian coordinate system (+x is
    // right, +y is up). Returns a positive value if ABC is counter-clockwise,
    // negative if clockwise, and zero if the points are collinear.
    function cross$4(a, b, c) {
      return (b[0] - a[0]) * (c[1] - a[1]) - (b[1] - a[1]) * (c[0] - a[0]);
    }

    function lexicographicOrder$1(a, b) {
      return a[0] - b[0] || a[1] - b[1];
    }

    // Computes the upper convex hull per the monotone chain algorithm.
    // Assumes points.length >= 3, is sorted by x, unique in y.
    // Returns an array of indices into points in left-to-right order.
    function computeUpperHullIndexes$1(points) {
      var n = points.length,
          indexes = [0, 1],
          size = 2;

      for (var i = 2; i < n; ++i) {
        while (size > 1 && cross$4(points[indexes[size - 2]], points[indexes[size - 1]], points[i]) <= 0) --size;
        indexes[size++] = i;
      }

      return indexes.slice(0, size); // remove popped points
    }

    function hull$1(points) {
      if ((n = points.length) < 3) return null;

      var i,
          n,
          sortedPoints = new Array(n),
          flippedPoints = new Array(n);

      for (i = 0; i < n; ++i) sortedPoints[i] = [+points[i][0], +points[i][1], i];
      sortedPoints.sort(lexicographicOrder$1);
      for (i = 0; i < n; ++i) flippedPoints[i] = [sortedPoints[i][0], -sortedPoints[i][1]];

      var upperIndexes = computeUpperHullIndexes$1(sortedPoints),
          lowerIndexes = computeUpperHullIndexes$1(flippedPoints);

      // Construct the hull polygon, removing possible duplicate endpoints.
      var skipLeft = lowerIndexes[0] === upperIndexes[0],
          skipRight = lowerIndexes[lowerIndexes.length - 1] === upperIndexes[upperIndexes.length - 1],
          hull = [];

      // Add upper hull in right-to-l order.
      // Then add lower hull in left-to-right order.
      for (i = upperIndexes.length - 1; i >= 0; --i) hull.push(points[sortedPoints[upperIndexes[i]][2]]);
      for (i = +skipLeft; i < lowerIndexes.length - skipRight; ++i) hull.push(points[sortedPoints[lowerIndexes[i]][2]]);

      return hull;
    }

    function contains$4(polygon, point) {
      var n = polygon.length,
          p = polygon[n - 1],
          x = point[0], y = point[1],
          x0 = p[0], y0 = p[1],
          x1, y1,
          inside = false;

      for (var i = 0; i < n; ++i) {
        p = polygon[i], x1 = p[0], y1 = p[1];
        if (((y1 > y) !== (y0 > y)) && (x < (x0 - x1) * (y - y1) / (y0 - y1) + x1)) inside = !inside;
        x0 = x1, y0 = y1;
      }

      return inside;
    }

    function length$5(polygon) {
      var i = -1,
          n = polygon.length,
          b = polygon[n - 1],
          xa,
          ya,
          xb = b[0],
          yb = b[1],
          perimeter = 0;

      while (++i < n) {
        xa = xb;
        ya = yb;
        b = polygon[i];
        xb = b[0];
        yb = b[1];
        xa -= xb;
        ya -= yb;
        perimeter += Math.sqrt(xa * xa + ya * ya);
      }

      return perimeter;
    }

    function tree_add$2(d) {
      var x = +this._x.call(null, d),
          y = +this._y.call(null, d);
      return add$4(this.cover(x, y), x, y, d);
    }

    function add$4(tree, x, y, d) {
      if (isNaN(x) || isNaN(y)) return tree; // ignore invalid points

      var parent,
          node = tree._root,
          leaf = {data: d},
          x0 = tree._x0,
          y0 = tree._y0,
          x1 = tree._x1,
          y1 = tree._y1,
          xm,
          ym,
          xp,
          yp,
          right,
          bottom,
          i,
          j;

      // If the tree is empty, initialize the root as a leaf.
      if (!node) return tree._root = leaf, tree;

      // Find the existing leaf for the new point, or add it.
      while (node.length) {
        if (right = x >= (xm = (x0 + x1) / 2)) x0 = xm; else x1 = xm;
        if (bottom = y >= (ym = (y0 + y1) / 2)) y0 = ym; else y1 = ym;
        if (parent = node, !(node = node[i = bottom << 1 | right])) return parent[i] = leaf, tree;
      }

      // Is the new point is exactly coincident with the existing point?
      xp = +tree._x.call(null, node.data);
      yp = +tree._y.call(null, node.data);
      if (x === xp && y === yp) return leaf.next = node, parent ? parent[i] = leaf : tree._root = leaf, tree;

      // Otherwise, split the leaf node until the old and new point are separated.
      do {
        parent = parent ? parent[i] = new Array(4) : tree._root = new Array(4);
        if (right = x >= (xm = (x0 + x1) / 2)) x0 = xm; else x1 = xm;
        if (bottom = y >= (ym = (y0 + y1) / 2)) y0 = ym; else y1 = ym;
      } while ((i = bottom << 1 | right) === (j = (yp >= ym) << 1 | (xp >= xm)));
      return parent[j] = node, parent[i] = leaf, tree;
    }

    function addAll$2(data) {
      var d, i, n = data.length,
          x,
          y,
          xz = new Array(n),
          yz = new Array(n),
          x0 = Infinity,
          y0 = Infinity,
          x1 = -Infinity,
          y1 = -Infinity;

      // Compute the points and their extent.
      for (i = 0; i < n; ++i) {
        if (isNaN(x = +this._x.call(null, d = data[i])) || isNaN(y = +this._y.call(null, d))) continue;
        xz[i] = x;
        yz[i] = y;
        if (x < x0) x0 = x;
        if (x > x1) x1 = x;
        if (y < y0) y0 = y;
        if (y > y1) y1 = y;
      }

      // If there were no (valid) points, inherit the existing extent.
      if (x1 < x0) x0 = this._x0, x1 = this._x1;
      if (y1 < y0) y0 = this._y0, y1 = this._y1;

      // Expand the tree to cover the new points.
      this.cover(x0, y0).cover(x1, y1);

      // Add the new points.
      for (i = 0; i < n; ++i) {
        add$4(this, xz[i], yz[i], data[i]);
      }

      return this;
    }

    function tree_cover$2(x, y) {
      if (isNaN(x = +x) || isNaN(y = +y)) return this; // ignore invalid points

      var x0 = this._x0,
          y0 = this._y0,
          x1 = this._x1,
          y1 = this._y1;

      // If the quadtree has no extent, initialize them.
      // Integer extent are necessary so that if we later double the extent,
      // the existing quadrant boundaries don’t change due to floating point error!
      if (isNaN(x0)) {
        x1 = (x0 = Math.floor(x)) + 1;
        y1 = (y0 = Math.floor(y)) + 1;
      }

      // Otherwise, double repeatedly to cover.
      else if (x0 > x || x > x1 || y0 > y || y > y1) {
        var z = x1 - x0,
            node = this._root,
            parent,
            i;

        switch (i = (y < (y0 + y1) / 2) << 1 | (x < (x0 + x1) / 2)) {
          case 0: {
            do parent = new Array(4), parent[i] = node, node = parent;
            while (z *= 2, x1 = x0 + z, y1 = y0 + z, x > x1 || y > y1);
            break;
          }
          case 1: {
            do parent = new Array(4), parent[i] = node, node = parent;
            while (z *= 2, x0 = x1 - z, y1 = y0 + z, x0 > x || y > y1);
            break;
          }
          case 2: {
            do parent = new Array(4), parent[i] = node, node = parent;
            while (z *= 2, x1 = x0 + z, y0 = y1 - z, x > x1 || y0 > y);
            break;
          }
          case 3: {
            do parent = new Array(4), parent[i] = node, node = parent;
            while (z *= 2, x0 = x1 - z, y0 = y1 - z, x0 > x || y0 > y);
            break;
          }
        }

        if (this._root && this._root.length) this._root = node;
      }

      // If the quadtree covers the point already, just return.
      else return this;

      this._x0 = x0;
      this._y0 = y0;
      this._x1 = x1;
      this._y1 = y1;
      return this;
    }

    function tree_data$2() {
      var data = [];
      this.visit(function(node) {
        if (!node.length) do data.push(node.data); while (node = node.next)
      });
      return data;
    }

    function tree_extent$2(_) {
      return arguments.length
          ? this.cover(+_[0][0], +_[0][1]).cover(+_[1][0], +_[1][1])
          : isNaN(this._x0) ? undefined : [[this._x0, this._y0], [this._x1, this._y1]];
    }

    function Quad$2(node, x0, y0, x1, y1) {
      this.node = node;
      this.x0 = x0;
      this.y0 = y0;
      this.x1 = x1;
      this.y1 = y1;
    }

    function tree_find$2(x, y, radius) {
      var data,
          x0 = this._x0,
          y0 = this._y0,
          x1,
          y1,
          x2,
          y2,
          x3 = this._x1,
          y3 = this._y1,
          quads = [],
          node = this._root,
          q,
          i;

      if (node) quads.push(new Quad$2(node, x0, y0, x3, y3));
      if (radius == null) radius = Infinity;
      else {
        x0 = x - radius, y0 = y - radius;
        x3 = x + radius, y3 = y + radius;
        radius *= radius;
      }

      while (q = quads.pop()) {

        // Stop searching if this quadrant can’t contain a closer node.
        if (!(node = q.node)
            || (x1 = q.x0) > x3
            || (y1 = q.y0) > y3
            || (x2 = q.x1) < x0
            || (y2 = q.y1) < y0) continue;

        // Bisect the current quadrant.
        if (node.length) {
          var xm = (x1 + x2) / 2,
              ym = (y1 + y2) / 2;

          quads.push(
            new Quad$2(node[3], xm, ym, x2, y2),
            new Quad$2(node[2], x1, ym, xm, y2),
            new Quad$2(node[1], xm, y1, x2, ym),
            new Quad$2(node[0], x1, y1, xm, ym)
          );

          // Visit the closest quadrant first.
          if (i = (y >= ym) << 1 | (x >= xm)) {
            q = quads[quads.length - 1];
            quads[quads.length - 1] = quads[quads.length - 1 - i];
            quads[quads.length - 1 - i] = q;
          }
        }

        // Visit this point. (Visiting coincident points isn’t necessary!)
        else {
          var dx = x - +this._x.call(null, node.data),
              dy = y - +this._y.call(null, node.data),
              d2 = dx * dx + dy * dy;
          if (d2 < radius) {
            var d = Math.sqrt(radius = d2);
            x0 = x - d, y0 = y - d;
            x3 = x + d, y3 = y + d;
            data = node.data;
          }
        }
      }

      return data;
    }

    function tree_remove$2(d) {
      if (isNaN(x = +this._x.call(null, d)) || isNaN(y = +this._y.call(null, d))) return this; // ignore invalid points

      var parent,
          node = this._root,
          retainer,
          previous,
          next,
          x0 = this._x0,
          y0 = this._y0,
          x1 = this._x1,
          y1 = this._y1,
          x,
          y,
          xm,
          ym,
          right,
          bottom,
          i,
          j;

      // If the tree is empty, initialize the root as a leaf.
      if (!node) return this;

      // Find the leaf node for the point.
      // While descending, also retain the deepest parent with a non-removed sibling.
      if (node.length) while (true) {
        if (right = x >= (xm = (x0 + x1) / 2)) x0 = xm; else x1 = xm;
        if (bottom = y >= (ym = (y0 + y1) / 2)) y0 = ym; else y1 = ym;
        if (!(parent = node, node = node[i = bottom << 1 | right])) return this;
        if (!node.length) break;
        if (parent[(i + 1) & 3] || parent[(i + 2) & 3] || parent[(i + 3) & 3]) retainer = parent, j = i;
      }

      // Find the point to remove.
      while (node.data !== d) if (!(previous = node, node = node.next)) return this;
      if (next = node.next) delete node.next;

      // If there are multiple coincident points, remove just the point.
      if (previous) return (next ? previous.next = next : delete previous.next), this;

      // If this is the root point, remove it.
      if (!parent) return this._root = next, this;

      // Remove this leaf.
      next ? parent[i] = next : delete parent[i];

      // If the parent now contains exactly one leaf, collapse superfluous parents.
      if ((node = parent[0] || parent[1] || parent[2] || parent[3])
          && node === (parent[3] || parent[2] || parent[1] || parent[0])
          && !node.length) {
        if (retainer) retainer[j] = node;
        else this._root = node;
      }

      return this;
    }

    function removeAll$2(data) {
      for (var i = 0, n = data.length; i < n; ++i) this.remove(data[i]);
      return this;
    }

    function tree_root$2() {
      return this._root;
    }

    function tree_size$2() {
      var size = 0;
      this.visit(function(node) {
        if (!node.length) do ++size; while (node = node.next)
      });
      return size;
    }

    function tree_visit$2(callback) {
      var quads = [], q, node = this._root, child, x0, y0, x1, y1;
      if (node) quads.push(new Quad$2(node, this._x0, this._y0, this._x1, this._y1));
      while (q = quads.pop()) {
        if (!callback(node = q.node, x0 = q.x0, y0 = q.y0, x1 = q.x1, y1 = q.y1) && node.length) {
          var xm = (x0 + x1) / 2, ym = (y0 + y1) / 2;
          if (child = node[3]) quads.push(new Quad$2(child, xm, ym, x1, y1));
          if (child = node[2]) quads.push(new Quad$2(child, x0, ym, xm, y1));
          if (child = node[1]) quads.push(new Quad$2(child, xm, y0, x1, ym));
          if (child = node[0]) quads.push(new Quad$2(child, x0, y0, xm, ym));
        }
      }
      return this;
    }

    function tree_visitAfter$2(callback) {
      var quads = [], next = [], q;
      if (this._root) quads.push(new Quad$2(this._root, this._x0, this._y0, this._x1, this._y1));
      while (q = quads.pop()) {
        var node = q.node;
        if (node.length) {
          var child, x0 = q.x0, y0 = q.y0, x1 = q.x1, y1 = q.y1, xm = (x0 + x1) / 2, ym = (y0 + y1) / 2;
          if (child = node[0]) quads.push(new Quad$2(child, x0, y0, xm, ym));
          if (child = node[1]) quads.push(new Quad$2(child, xm, y0, x1, ym));
          if (child = node[2]) quads.push(new Quad$2(child, x0, ym, xm, y1));
          if (child = node[3]) quads.push(new Quad$2(child, xm, ym, x1, y1));
        }
        next.push(q);
      }
      while (q = next.pop()) {
        callback(q.node, q.x0, q.y0, q.x1, q.y1);
      }
      return this;
    }

    function defaultX$3(d) {
      return d[0];
    }

    function tree_x$2(_) {
      return arguments.length ? (this._x = _, this) : this._x;
    }

    function defaultY$3(d) {
      return d[1];
    }

    function tree_y$2(_) {
      return arguments.length ? (this._y = _, this) : this._y;
    }

    function quadtree$2(nodes, x, y) {
      var tree = new Quadtree$2(x == null ? defaultX$3 : x, y == null ? defaultY$3 : y, NaN, NaN, NaN, NaN);
      return nodes == null ? tree : tree.addAll(nodes);
    }

    function Quadtree$2(x, y, x0, y0, x1, y1) {
      this._x = x;
      this._y = y;
      this._x0 = x0;
      this._y0 = y0;
      this._x1 = x1;
      this._y1 = y1;
      this._root = undefined;
    }

    function leaf_copy$2(leaf) {
      var copy = {data: leaf.data}, next = copy;
      while (leaf = leaf.next) next = next.next = {data: leaf.data};
      return copy;
    }

    var treeProto$2 = quadtree$2.prototype = Quadtree$2.prototype;

    treeProto$2.copy = function() {
      var copy = new Quadtree$2(this._x, this._y, this._x0, this._y0, this._x1, this._y1),
          node = this._root,
          nodes,
          child;

      if (!node) return copy;

      if (!node.length) return copy._root = leaf_copy$2(node), copy;

      nodes = [{source: node, target: copy._root = new Array(4)}];
      while (node = nodes.pop()) {
        for (var i = 0; i < 4; ++i) {
          if (child = node.source[i]) {
            if (child.length) nodes.push({source: child, target: node.target[i] = new Array(4)});
            else node.target[i] = leaf_copy$2(child);
          }
        }
      }

      return copy;
    };

    treeProto$2.add = tree_add$2;
    treeProto$2.addAll = addAll$2;
    treeProto$2.cover = tree_cover$2;
    treeProto$2.data = tree_data$2;
    treeProto$2.extent = tree_extent$2;
    treeProto$2.find = tree_find$2;
    treeProto$2.remove = tree_remove$2;
    treeProto$2.removeAll = removeAll$2;
    treeProto$2.root = tree_root$2;
    treeProto$2.size = tree_size$2;
    treeProto$2.visit = tree_visit$2;
    treeProto$2.visitAfter = tree_visitAfter$2;
    treeProto$2.x = tree_x$2;
    treeProto$2.y = tree_y$2;

    var slice$b = [].slice;

    var noabort = {};

    function Queue(size) {
      this._size = size;
      this._call =
      this._error = null;
      this._tasks = [];
      this._data = [];
      this._waiting =
      this._active =
      this._ended =
      this._start = 0; // inside a synchronous task callback?
    }

    Queue.prototype = queue.prototype = {
      constructor: Queue,
      defer: function(callback) {
        if (typeof callback !== "function") throw new Error("invalid callback");
        if (this._call) throw new Error("defer after await");
        if (this._error != null) return this;
        var t = slice$b.call(arguments, 1);
        t.push(callback);
        ++this._waiting, this._tasks.push(t);
        poke$3(this);
        return this;
      },
      abort: function() {
        if (this._error == null) abort(this, new Error("abort"));
        return this;
      },
      await: function(callback) {
        if (typeof callback !== "function") throw new Error("invalid callback");
        if (this._call) throw new Error("multiple await");
        this._call = function(error, results) { callback.apply(null, [error].concat(results)); };
        maybeNotify(this);
        return this;
      },
      awaitAll: function(callback) {
        if (typeof callback !== "function") throw new Error("invalid callback");
        if (this._call) throw new Error("multiple await");
        this._call = callback;
        maybeNotify(this);
        return this;
      }
    };

    function poke$3(q) {
      if (!q._start) {
        try { start$2(q); } // let the current task complete
        catch (e) {
          if (q._tasks[q._ended + q._active - 1]) abort(q, e); // task errored synchronously
          else if (!q._data) throw e; // await callback errored synchronously
        }
      }
    }

    function start$2(q) {
      while (q._start = q._waiting && q._active < q._size) {
        var i = q._ended + q._active,
            t = q._tasks[i],
            j = t.length - 1,
            c = t[j];
        t[j] = end(q, i);
        --q._waiting, ++q._active;
        t = c.apply(null, t);
        if (!q._tasks[i]) continue; // task finished synchronously
        q._tasks[i] = t || noabort;
      }
    }

    function end(q, i) {
      return function(e, r) {
        if (!q._tasks[i]) return; // ignore multiple callbacks
        --q._active, ++q._ended;
        q._tasks[i] = null;
        if (q._error != null) return; // ignore secondary errors
        if (e != null) {
          abort(q, e);
        } else {
          q._data[i] = r;
          if (q._waiting) poke$3(q);
          else maybeNotify(q);
        }
      };
    }

    function abort(q, e) {
      var i = q._tasks.length, t;
      q._error = e; // ignore active callbacks
      q._data = undefined; // allow gc
      q._waiting = NaN; // prevent starting

      while (--i >= 0) {
        if (t = q._tasks[i]) {
          q._tasks[i] = null;
          if (t.abort) {
            try { t.abort(); }
            catch (e) { /* ignore */ }
          }
        }
      }

      q._active = NaN; // allow notification
      maybeNotify(q);
    }

    function maybeNotify(q) {
      if (!q._active && q._call) {
        var d = q._data;
        q._data = undefined; // allow gc
        q._call(q._error, d);
      }
    }

    function queue(concurrency) {
      if (concurrency == null) concurrency = Infinity;
      else if (!((concurrency = +concurrency) >= 1)) throw new Error("invalid concurrency");
      return new Queue(concurrency);
    }

    function defaultSource$3() {
      return Math.random();
    }

    var uniform$1 = (function sourceRandomUniform(source) {
      function randomUniform(min, max) {
        min = min == null ? 0 : +min;
        max = max == null ? 1 : +max;
        if (arguments.length === 1) max = min, min = 0;
        else max -= min;
        return function() {
          return source() * max + min;
        };
      }

      randomUniform.source = sourceRandomUniform;

      return randomUniform;
    })(defaultSource$3);

    var normal$1 = (function sourceRandomNormal(source) {
      function randomNormal(mu, sigma) {
        var x, r;
        mu = mu == null ? 0 : +mu;
        sigma = sigma == null ? 1 : +sigma;
        return function() {
          var y;

          // If available, use the second previously-generated uniform random.
          if (x != null) y = x, x = null;

          // Otherwise, generate a new x and y.
          else do {
            x = source() * 2 - 1;
            y = source() * 2 - 1;
            r = x * x + y * y;
          } while (!r || r > 1);

          return mu + sigma * y * Math.sqrt(-2 * Math.log(r) / r);
        };
      }

      randomNormal.source = sourceRandomNormal;

      return randomNormal;
    })(defaultSource$3);

    var logNormal$1 = (function sourceRandomLogNormal(source) {
      function randomLogNormal() {
        var randomNormal = normal$1.source(source).apply(this, arguments);
        return function() {
          return Math.exp(randomNormal());
        };
      }

      randomLogNormal.source = sourceRandomLogNormal;

      return randomLogNormal;
    })(defaultSource$3);

    var irwinHall$1 = (function sourceRandomIrwinHall(source) {
      function randomIrwinHall(n) {
        return function() {
          for (var sum = 0, i = 0; i < n; ++i) sum += source();
          return sum;
        };
      }

      randomIrwinHall.source = sourceRandomIrwinHall;

      return randomIrwinHall;
    })(defaultSource$3);

    var bates$1 = (function sourceRandomBates(source) {
      function randomBates(n) {
        var randomIrwinHall = irwinHall$1.source(source)(n);
        return function() {
          return randomIrwinHall() / n;
        };
      }

      randomBates.source = sourceRandomBates;

      return randomBates;
    })(defaultSource$3);

    var exponential$4 = (function sourceRandomExponential(source) {
      function randomExponential(lambda) {
        return function() {
          return -Math.log(1 - source()) / lambda;
        };
      }

      randomExponential.source = sourceRandomExponential;

      return randomExponential;
    })(defaultSource$3);

    function request(url, callback) {
      var request,
          event = dispatch("beforesend", "progress", "load", "error"),
          mimeType,
          headers = map$1(),
          xhr = new XMLHttpRequest,
          user = null,
          password = null,
          response,
          responseType,
          timeout = 0;

      // If IE does not support CORS, use XDomainRequest.
      if (typeof XDomainRequest !== "undefined"
          && !("withCredentials" in xhr)
          && /^(http(s)?:)?\/\//.test(url)) xhr = new XDomainRequest;

      "onload" in xhr
          ? xhr.onload = xhr.onerror = xhr.ontimeout = respond
          : xhr.onreadystatechange = function(o) { xhr.readyState > 3 && respond(o); };

      function respond(o) {
        var status = xhr.status, result;
        if (!status && hasResponse(xhr)
            || status >= 200 && status < 300
            || status === 304) {
          if (response) {
            try {
              result = response.call(request, xhr);
            } catch (e) {
              event.call("error", request, e);
              return;
            }
          } else {
            result = xhr;
          }
          event.call("load", request, result);
        } else {
          event.call("error", request, o);
        }
      }

      xhr.onprogress = function(e) {
        event.call("progress", request, e);
      };

      request = {
        header: function(name, value) {
          name = (name + "").toLowerCase();
          if (arguments.length < 2) return headers.get(name);
          if (value == null) headers.remove(name);
          else headers.set(name, value + "");
          return request;
        },

        // If mimeType is non-null and no Accept header is set, a default is used.
        mimeType: function(value) {
          if (!arguments.length) return mimeType;
          mimeType = value == null ? null : value + "";
          return request;
        },

        // Specifies what type the response value should take;
        // for instance, arraybuffer, blob, document, or text.
        responseType: function(value) {
          if (!arguments.length) return responseType;
          responseType = value;
          return request;
        },

        timeout: function(value) {
          if (!arguments.length) return timeout;
          timeout = +value;
          return request;
        },

        user: function(value) {
          return arguments.length < 1 ? user : (user = value == null ? null : value + "", request);
        },

        password: function(value) {
          return arguments.length < 1 ? password : (password = value == null ? null : value + "", request);
        },

        // Specify how to convert the response content to a specific type;
        // changes the callback value on "load" events.
        response: function(value) {
          response = value;
          return request;
        },

        // Alias for send("GET", …).
        get: function(data, callback) {
          return request.send("GET", data, callback);
        },

        // Alias for send("POST", …).
        post: function(data, callback) {
          return request.send("POST", data, callback);
        },

        // If callback is non-null, it will be used for error and load events.
        send: function(method, data, callback) {
          xhr.open(method, url, true, user, password);
          if (mimeType != null && !headers.has("accept")) headers.set("accept", mimeType + ",*/*");
          if (xhr.setRequestHeader) headers.each(function(value, name) { xhr.setRequestHeader(name, value); });
          if (mimeType != null && xhr.overrideMimeType) xhr.overrideMimeType(mimeType);
          if (responseType != null) xhr.responseType = responseType;
          if (timeout > 0) xhr.timeout = timeout;
          if (callback == null && typeof data === "function") callback = data, data = null;
          if (callback != null && callback.length === 1) callback = fixCallback(callback);
          if (callback != null) request.on("error", callback).on("load", function(xhr) { callback(null, xhr); });
          event.call("beforesend", request, xhr);
          xhr.send(data == null ? null : data);
          return request;
        },

        abort: function() {
          xhr.abort();
          return request;
        },

        on: function() {
          var value = event.on.apply(event, arguments);
          return value === event ? request : value;
        }
      };

      if (callback != null) {
        if (typeof callback !== "function") throw new Error("invalid callback: " + callback);
        return request.get(callback);
      }

      return request;
    }

    function fixCallback(callback) {
      return function(error, xhr) {
        callback(error == null ? xhr : null);
      };
    }

    function hasResponse(xhr) {
      var type = xhr.responseType;
      return type && type !== "text"
          ? xhr.response // null on error
          : xhr.responseText; // "" on error
    }

    function type$2(defaultMimeType, response) {
      return function(url, callback) {
        var r = request(url).mimeType(defaultMimeType).response(response);
        if (callback != null) {
          if (typeof callback !== "function") throw new Error("invalid callback: " + callback);
          return r.get(callback);
        }
        return r;
      };
    }

    var html$2 = type$2("text/html", function(xhr) {
      return document.createRange().createContextualFragment(xhr.responseText);
    });

    var json$1 = type$2("application/json", function(xhr) {
      return JSON.parse(xhr.responseText);
    });

    var text$1 = type$2("text/plain", function(xhr) {
      return xhr.responseText;
    });

    var xml$1 = type$2("application/xml", function(xhr) {
      var xml = xhr.responseXML;
      if (!xml) throw new Error("parse error");
      return xml;
    });

    function dsv$2(defaultMimeType, parse) {
      return function(url, row, callback) {
        if (arguments.length < 3) callback = row, row = null;
        var r = request(url).mimeType(defaultMimeType);
        r.row = function(_) { return arguments.length ? r.response(responseOf(parse, row = _)) : row; };
        r.row(row);
        return callback ? r.get(callback) : r;
      };
    }

    function responseOf(parse, row) {
      return function(request) {
        return parse(request.responseText, row);
      };
    }

    var csv$3 = dsv$2("text/csv", csvParse);

    var tsv$3 = dsv$2("text/tab-separated-values", tsvParse);

    function ascending$9(a, b) {
      return a < b ? -1 : a > b ? 1 : a >= b ? 0 : NaN;
    }

    function bisector$4(compare) {
      if (compare.length === 1) compare = ascendingComparator$4(compare);
      return {
        left: function(a, x, lo, hi) {
          if (lo == null) lo = 0;
          if (hi == null) hi = a.length;
          while (lo < hi) {
            var mid = lo + hi >>> 1;
            if (compare(a[mid], x) < 0) lo = mid + 1;
            else hi = mid;
          }
          return lo;
        },
        right: function(a, x, lo, hi) {
          if (lo == null) lo = 0;
          if (hi == null) hi = a.length;
          while (lo < hi) {
            var mid = lo + hi >>> 1;
            if (compare(a[mid], x) > 0) hi = mid;
            else lo = mid + 1;
          }
          return lo;
        }
      };
    }

    function ascendingComparator$4(f) {
      return function(d, x) {
        return ascending$9(f(d), x);
      };
    }

    var ascendingBisect$4 = bisector$4(ascending$9);
    var bisectRight$2 = ascendingBisect$4.right;

    function number$7(x) {
      return x === null ? NaN : +x;
    }

    function sequence$1(start, stop, step) {
      start = +start, stop = +stop, step = (n = arguments.length) < 2 ? (stop = start, start = 0, 1) : n < 3 ? 1 : +step;

      var i = -1,
          n = Math.max(0, Math.ceil((stop - start) / step)) | 0,
          range = new Array(n);

      while (++i < n) {
        range[i] = start + i * step;
      }

      return range;
    }

    var e10$2 = Math.sqrt(50),
        e5$2 = Math.sqrt(10),
        e2$2 = Math.sqrt(2);

    function ticks$2(start, stop, count) {
      var reverse,
          i = -1,
          n,
          ticks,
          step;

      stop = +stop, start = +start, count = +count;
      if (start === stop && count > 0) return [start];
      if (reverse = stop < start) n = start, start = stop, stop = n;
      if ((step = tickIncrement$2(start, stop, count)) === 0 || !isFinite(step)) return [];

      if (step > 0) {
        start = Math.ceil(start / step);
        stop = Math.floor(stop / step);
        ticks = new Array(n = Math.ceil(stop - start + 1));
        while (++i < n) ticks[i] = (start + i) * step;
      } else {
        start = Math.floor(start * step);
        stop = Math.ceil(stop * step);
        ticks = new Array(n = Math.ceil(start - stop + 1));
        while (++i < n) ticks[i] = (start - i) / step;
      }

      if (reverse) ticks.reverse();

      return ticks;
    }

    function tickIncrement$2(start, stop, count) {
      var step = (stop - start) / Math.max(0, count),
          power = Math.floor(Math.log(step) / Math.LN10),
          error = step / Math.pow(10, power);
      return power >= 0
          ? (error >= e10$2 ? 10 : error >= e5$2 ? 5 : error >= e2$2 ? 2 : 1) * Math.pow(10, power)
          : -Math.pow(10, -power) / (error >= e10$2 ? 10 : error >= e5$2 ? 5 : error >= e2$2 ? 2 : 1);
    }

    function tickStep$2(start, stop, count) {
      var step0 = Math.abs(stop - start) / Math.max(0, count),
          step1 = Math.pow(10, Math.floor(Math.log(step0) / Math.LN10)),
          error = step0 / step1;
      if (error >= e10$2) step1 *= 10;
      else if (error >= e5$2) step1 *= 5;
      else if (error >= e2$2) step1 *= 2;
      return stop < start ? -step1 : step1;
    }

    function threshold$2(values, p, valueof) {
      if (valueof == null) valueof = number$7;
      if (!(n = values.length)) return;
      if ((p = +p) <= 0 || n < 2) return +valueof(values[0], 0, values);
      if (p >= 1) return +valueof(values[n - 1], n - 1, values);
      var n,
          i = (n - 1) * p,
          i0 = Math.floor(i),
          value0 = +valueof(values[i0], i0, values),
          value1 = +valueof(values[i0 + 1], i0 + 1, values);
      return value0 + (value1 - value0) * (i - i0);
    }

    var prefix$3 = "$";

    function Map$4() {}

    Map$4.prototype = map$7.prototype = {
      constructor: Map$4,
      has: function(key) {
        return (prefix$3 + key) in this;
      },
      get: function(key) {
        return this[prefix$3 + key];
      },
      set: function(key, value) {
        this[prefix$3 + key] = value;
        return this;
      },
      remove: function(key) {
        var property = prefix$3 + key;
        return property in this && delete this[property];
      },
      clear: function() {
        for (var property in this) if (property[0] === prefix$3) delete this[property];
      },
      keys: function() {
        var keys = [];
        for (var property in this) if (property[0] === prefix$3) keys.push(property.slice(1));
        return keys;
      },
      values: function() {
        var values = [];
        for (var property in this) if (property[0] === prefix$3) values.push(this[property]);
        return values;
      },
      entries: function() {
        var entries = [];
        for (var property in this) if (property[0] === prefix$3) entries.push({key: property.slice(1), value: this[property]});
        return entries;
      },
      size: function() {
        var size = 0;
        for (var property in this) if (property[0] === prefix$3) ++size;
        return size;
      },
      empty: function() {
        for (var property in this) if (property[0] === prefix$3) return false;
        return true;
      },
      each: function(f) {
        for (var property in this) if (property[0] === prefix$3) f(this[property], property.slice(1), this);
      }
    };

    function map$7(object, f) {
      var map = new Map$4;

      // Copy constructor.
      if (object instanceof Map$4) object.each(function(value, key) { map.set(key, value); });

      // Index array by numeric index or specified key function.
      else if (Array.isArray(object)) {
        var i = -1,
            n = object.length,
            o;

        if (f == null) while (++i < n) map.set(i, object[i]);
        else while (++i < n) map.set(f(o = object[i], i, object), o);
      }

      // Convert object to map.
      else if (object) for (var key in object) map.set(key, object[key]);

      return map;
    }

    function Set$4() {}

    var proto$4 = map$7.prototype;

    Set$4.prototype = set$b.prototype = {
      constructor: Set$4,
      has: proto$4.has,
      add: function(value) {
        value += "";
        this[prefix$3 + value] = value;
        return this;
      },
      remove: proto$4.remove,
      clear: proto$4.clear,
      values: proto$4.keys,
      size: proto$4.size,
      empty: proto$4.empty,
      each: proto$4.each
    };

    function set$b(object, f) {
      var set = new Set$4;

      // Copy constructor.
      if (object instanceof Set$4) object.each(function(value) { set.add(value); });

      // Otherwise, assume it’s an array.
      else if (object) {
        var i = -1, n = object.length;
        if (f == null) while (++i < n) set.add(object[i]);
        else while (++i < n) set.add(f(object[i], i, object));
      }

      return set;
    }

    var array$6 = Array.prototype;

    var map$8 = array$6.map;
    var slice$c = array$6.slice;

    var implicit$1 = {name: "implicit"};

    function ordinal$1(range) {
      var index = map$7(),
          domain = [],
          unknown = implicit$1;

      range = range == null ? [] : slice$c.call(range);

      function scale(d) {
        var key = d + "", i = index.get(key);
        if (!i) {
          if (unknown !== implicit$1) return unknown;
          index.set(key, i = domain.push(d));
        }
        return range[(i - 1) % range.length];
      }

      scale.domain = function(_) {
        if (!arguments.length) return domain.slice();
        domain = [], index = map$7();
        var i = -1, n = _.length, d, key;
        while (++i < n) if (!index.has(key = (d = _[i]) + "")) index.set(key, domain.push(d));
        return scale;
      };

      scale.range = function(_) {
        return arguments.length ? (range = slice$c.call(_), scale) : range.slice();
      };

      scale.unknown = function(_) {
        return arguments.length ? (unknown = _, scale) : unknown;
      };

      scale.copy = function() {
        return ordinal$1()
            .domain(domain)
            .range(range)
            .unknown(unknown);
      };

      return scale;
    }

    function band$1() {
      var scale = ordinal$1().unknown(undefined),
          domain = scale.domain,
          ordinalRange = scale.range,
          range = [0, 1],
          step,
          bandwidth,
          round = false,
          paddingInner = 0,
          paddingOuter = 0,
          align = 0.5;

      delete scale.unknown;

      function rescale() {
        var n = domain().length,
            reverse = range[1] < range[0],
            start = range[reverse - 0],
            stop = range[1 - reverse];
        step = (stop - start) / Math.max(1, n - paddingInner + paddingOuter * 2);
        if (round) step = Math.floor(step);
        start += (stop - start - step * (n - paddingInner)) * align;
        bandwidth = step * (1 - paddingInner);
        if (round) start = Math.round(start), bandwidth = Math.round(bandwidth);
        var values = sequence$1(n).map(function(i) { return start + step * i; });
        return ordinalRange(reverse ? values.reverse() : values);
      }

      scale.domain = function(_) {
        return arguments.length ? (domain(_), rescale()) : domain();
      };

      scale.range = function(_) {
        return arguments.length ? (range = [+_[0], +_[1]], rescale()) : range.slice();
      };

      scale.rangeRound = function(_) {
        return range = [+_[0], +_[1]], round = true, rescale();
      };

      scale.bandwidth = function() {
        return bandwidth;
      };

      scale.step = function() {
        return step;
      };

      scale.round = function(_) {
        return arguments.length ? (round = !!_, rescale()) : round;
      };

      scale.padding = function(_) {
        return arguments.length ? (paddingInner = paddingOuter = Math.max(0, Math.min(1, _)), rescale()) : paddingInner;
      };

      scale.paddingInner = function(_) {
        return arguments.length ? (paddingInner = Math.max(0, Math.min(1, _)), rescale()) : paddingInner;
      };

      scale.paddingOuter = function(_) {
        return arguments.length ? (paddingOuter = Math.max(0, Math.min(1, _)), rescale()) : paddingOuter;
      };

      scale.align = function(_) {
        return arguments.length ? (align = Math.max(0, Math.min(1, _)), rescale()) : align;
      };

      scale.copy = function() {
        return band$1()
            .domain(domain())
            .range(range)
            .round(round)
            .paddingInner(paddingInner)
            .paddingOuter(paddingOuter)
            .align(align);
      };

      return rescale();
    }

    function pointish$1(scale) {
      var copy = scale.copy;

      scale.padding = scale.paddingOuter;
      delete scale.paddingInner;
      delete scale.paddingOuter;

      scale.copy = function() {
        return pointish$1(copy());
      };

      return scale;
    }

    function point$8() {
      return pointish$1(band$1().paddingInner(1));
    }

    function define$4(constructor, factory, prototype) {
      constructor.prototype = factory.prototype = prototype;
      prototype.constructor = constructor;
    }

    function extend$4(parent, definition) {
      var prototype = Object.create(parent.prototype);
      for (var key in definition) prototype[key] = definition[key];
      return prototype;
    }

    function Color$4() {}

    var darker$4 = 0.7;
    var brighter$4 = 1 / darker$4;

    var reI$4 = "\\s*([+-]?\\d+)\\s*",
        reN$4 = "\\s*([+-]?\\d*\\.?\\d+(?:[eE][+-]?\\d+)?)\\s*",
        reP$4 = "\\s*([+-]?\\d*\\.?\\d+(?:[eE][+-]?\\d+)?)%\\s*",
        reHex$3 = /^#([0-9a-f]{3,8})$/,
        reRgbInteger$4 = new RegExp("^rgb\\(" + [reI$4, reI$4, reI$4] + "\\)$"),
        reRgbPercent$4 = new RegExp("^rgb\\(" + [reP$4, reP$4, reP$4] + "\\)$"),
        reRgbaInteger$4 = new RegExp("^rgba\\(" + [reI$4, reI$4, reI$4, reN$4] + "\\)$"),
        reRgbaPercent$4 = new RegExp("^rgba\\(" + [reP$4, reP$4, reP$4, reN$4] + "\\)$"),
        reHslPercent$4 = new RegExp("^hsl\\(" + [reN$4, reP$4, reP$4] + "\\)$"),
        reHslaPercent$4 = new RegExp("^hsla\\(" + [reN$4, reP$4, reP$4, reN$4] + "\\)$");

    var named$4 = {
      aliceblue: 0xf0f8ff,
      antiquewhite: 0xfaebd7,
      aqua: 0x00ffff,
      aquamarine: 0x7fffd4,
      azure: 0xf0ffff,
      beige: 0xf5f5dc,
      bisque: 0xffe4c4,
      black: 0x000000,
      blanchedalmond: 0xffebcd,
      blue: 0x0000ff,
      blueviolet: 0x8a2be2,
      brown: 0xa52a2a,
      burlywood: 0xdeb887,
      cadetblue: 0x5f9ea0,
      chartreuse: 0x7fff00,
      chocolate: 0xd2691e,
      coral: 0xff7f50,
      cornflowerblue: 0x6495ed,
      cornsilk: 0xfff8dc,
      crimson: 0xdc143c,
      cyan: 0x00ffff,
      darkblue: 0x00008b,
      darkcyan: 0x008b8b,
      darkgoldenrod: 0xb8860b,
      darkgray: 0xa9a9a9,
      darkgreen: 0x006400,
      darkgrey: 0xa9a9a9,
      darkkhaki: 0xbdb76b,
      darkmagenta: 0x8b008b,
      darkolivegreen: 0x556b2f,
      darkorange: 0xff8c00,
      darkorchid: 0x9932cc,
      darkred: 0x8b0000,
      darksalmon: 0xe9967a,
      darkseagreen: 0x8fbc8f,
      darkslateblue: 0x483d8b,
      darkslategray: 0x2f4f4f,
      darkslategrey: 0x2f4f4f,
      darkturquoise: 0x00ced1,
      darkviolet: 0x9400d3,
      deeppink: 0xff1493,
      deepskyblue: 0x00bfff,
      dimgray: 0x696969,
      dimgrey: 0x696969,
      dodgerblue: 0x1e90ff,
      firebrick: 0xb22222,
      floralwhite: 0xfffaf0,
      forestgreen: 0x228b22,
      fuchsia: 0xff00ff,
      gainsboro: 0xdcdcdc,
      ghostwhite: 0xf8f8ff,
      gold: 0xffd700,
      goldenrod: 0xdaa520,
      gray: 0x808080,
      green: 0x008000,
      greenyellow: 0xadff2f,
      grey: 0x808080,
      honeydew: 0xf0fff0,
      hotpink: 0xff69b4,
      indianred: 0xcd5c5c,
      indigo: 0x4b0082,
      ivory: 0xfffff0,
      khaki: 0xf0e68c,
      lavender: 0xe6e6fa,
      lavenderblush: 0xfff0f5,
      lawngreen: 0x7cfc00,
      lemonchiffon: 0xfffacd,
      lightblue: 0xadd8e6,
      lightcoral: 0xf08080,
      lightcyan: 0xe0ffff,
      lightgoldenrodyellow: 0xfafad2,
      lightgray: 0xd3d3d3,
      lightgreen: 0x90ee90,
      lightgrey: 0xd3d3d3,
      lightpink: 0xffb6c1,
      lightsalmon: 0xffa07a,
      lightseagreen: 0x20b2aa,
      lightskyblue: 0x87cefa,
      lightslategray: 0x778899,
      lightslategrey: 0x778899,
      lightsteelblue: 0xb0c4de,
      lightyellow: 0xffffe0,
      lime: 0x00ff00,
      limegreen: 0x32cd32,
      linen: 0xfaf0e6,
      magenta: 0xff00ff,
      maroon: 0x800000,
      mediumaquamarine: 0x66cdaa,
      mediumblue: 0x0000cd,
      mediumorchid: 0xba55d3,
      mediumpurple: 0x9370db,
      mediumseagreen: 0x3cb371,
      mediumslateblue: 0x7b68ee,
      mediumspringgreen: 0x00fa9a,
      mediumturquoise: 0x48d1cc,
      mediumvioletred: 0xc71585,
      midnightblue: 0x191970,
      mintcream: 0xf5fffa,
      mistyrose: 0xffe4e1,
      moccasin: 0xffe4b5,
      navajowhite: 0xffdead,
      navy: 0x000080,
      oldlace: 0xfdf5e6,
      olive: 0x808000,
      olivedrab: 0x6b8e23,
      orange: 0xffa500,
      orangered: 0xff4500,
      orchid: 0xda70d6,
      palegoldenrod: 0xeee8aa,
      palegreen: 0x98fb98,
      paleturquoise: 0xafeeee,
      palevioletred: 0xdb7093,
      papayawhip: 0xffefd5,
      peachpuff: 0xffdab9,
      peru: 0xcd853f,
      pink: 0xffc0cb,
      plum: 0xdda0dd,
      powderblue: 0xb0e0e6,
      purple: 0x800080,
      rebeccapurple: 0x663399,
      red: 0xff0000,
      rosybrown: 0xbc8f8f,
      royalblue: 0x4169e1,
      saddlebrown: 0x8b4513,
      salmon: 0xfa8072,
      sandybrown: 0xf4a460,
      seagreen: 0x2e8b57,
      seashell: 0xfff5ee,
      sienna: 0xa0522d,
      silver: 0xc0c0c0,
      skyblue: 0x87ceeb,
      slateblue: 0x6a5acd,
      slategray: 0x708090,
      slategrey: 0x708090,
      snow: 0xfffafa,
      springgreen: 0x00ff7f,
      steelblue: 0x4682b4,
      tan: 0xd2b48c,
      teal: 0x008080,
      thistle: 0xd8bfd8,
      tomato: 0xff6347,
      turquoise: 0x40e0d0,
      violet: 0xee82ee,
      wheat: 0xf5deb3,
      white: 0xffffff,
      whitesmoke: 0xf5f5f5,
      yellow: 0xffff00,
      yellowgreen: 0x9acd32
    };

    define$4(Color$4, color$4, {
      copy: function(channels) {
        return Object.assign(new this.constructor, this, channels);
      },
      displayable: function() {
        return this.rgb().displayable();
      },
      hex: color_formatHex$3, // Deprecated! Use color.formatHex.
      formatHex: color_formatHex$3,
      formatHsl: color_formatHsl$3,
      formatRgb: color_formatRgb$3,
      toString: color_formatRgb$3
    });

    function color_formatHex$3() {
      return this.rgb().formatHex();
    }

    function color_formatHsl$3() {
      return hslConvert$4(this).formatHsl();
    }

    function color_formatRgb$3() {
      return this.rgb().formatRgb();
    }

    function color$4(format) {
      var m, l;
      format = (format + "").trim().toLowerCase();
      return (m = reHex$3.exec(format)) ? (l = m[1].length, m = parseInt(m[1], 16), l === 6 ? rgbn$4(m) // #ff0000
          : l === 3 ? new Rgb$4((m >> 8 & 0xf) | (m >> 4 & 0xf0), (m >> 4 & 0xf) | (m & 0xf0), ((m & 0xf) << 4) | (m & 0xf), 1) // #f00
          : l === 8 ? rgba$4(m >> 24 & 0xff, m >> 16 & 0xff, m >> 8 & 0xff, (m & 0xff) / 0xff) // #ff000000
          : l === 4 ? rgba$4((m >> 12 & 0xf) | (m >> 8 & 0xf0), (m >> 8 & 0xf) | (m >> 4 & 0xf0), (m >> 4 & 0xf) | (m & 0xf0), (((m & 0xf) << 4) | (m & 0xf)) / 0xff) // #f000
          : null) // invalid hex
          : (m = reRgbInteger$4.exec(format)) ? new Rgb$4(m[1], m[2], m[3], 1) // rgb(255, 0, 0)
          : (m = reRgbPercent$4.exec(format)) ? new Rgb$4(m[1] * 255 / 100, m[2] * 255 / 100, m[3] * 255 / 100, 1) // rgb(100%, 0%, 0%)
          : (m = reRgbaInteger$4.exec(format)) ? rgba$4(m[1], m[2], m[3], m[4]) // rgba(255, 0, 0, 1)
          : (m = reRgbaPercent$4.exec(format)) ? rgba$4(m[1] * 255 / 100, m[2] * 255 / 100, m[3] * 255 / 100, m[4]) // rgb(100%, 0%, 0%, 1)
          : (m = reHslPercent$4.exec(format)) ? hsla$4(m[1], m[2] / 100, m[3] / 100, 1) // hsl(120, 50%, 50%)
          : (m = reHslaPercent$4.exec(format)) ? hsla$4(m[1], m[2] / 100, m[3] / 100, m[4]) // hsla(120, 50%, 50%, 1)
          : named$4.hasOwnProperty(format) ? rgbn$4(named$4[format]) // eslint-disable-line no-prototype-builtins
          : format === "transparent" ? new Rgb$4(NaN, NaN, NaN, 0)
          : null;
    }

    function rgbn$4(n) {
      return new Rgb$4(n >> 16 & 0xff, n >> 8 & 0xff, n & 0xff, 1);
    }

    function rgba$4(r, g, b, a) {
      if (a <= 0) r = g = b = NaN;
      return new Rgb$4(r, g, b, a);
    }

    function rgbConvert$4(o) {
      if (!(o instanceof Color$4)) o = color$4(o);
      if (!o) return new Rgb$4;
      o = o.rgb();
      return new Rgb$4(o.r, o.g, o.b, o.opacity);
    }

    function rgb$5(r, g, b, opacity) {
      return arguments.length === 1 ? rgbConvert$4(r) : new Rgb$4(r, g, b, opacity == null ? 1 : opacity);
    }

    function Rgb$4(r, g, b, opacity) {
      this.r = +r;
      this.g = +g;
      this.b = +b;
      this.opacity = +opacity;
    }

    define$4(Rgb$4, rgb$5, extend$4(Color$4, {
      brighter: function(k) {
        k = k == null ? brighter$4 : Math.pow(brighter$4, k);
        return new Rgb$4(this.r * k, this.g * k, this.b * k, this.opacity);
      },
      darker: function(k) {
        k = k == null ? darker$4 : Math.pow(darker$4, k);
        return new Rgb$4(this.r * k, this.g * k, this.b * k, this.opacity);
      },
      rgb: function() {
        return this;
      },
      displayable: function() {
        return (-0.5 <= this.r && this.r < 255.5)
            && (-0.5 <= this.g && this.g < 255.5)
            && (-0.5 <= this.b && this.b < 255.5)
            && (0 <= this.opacity && this.opacity <= 1);
      },
      hex: rgb_formatHex$3, // Deprecated! Use color.formatHex.
      formatHex: rgb_formatHex$3,
      formatRgb: rgb_formatRgb$3,
      toString: rgb_formatRgb$3
    }));

    function rgb_formatHex$3() {
      return "#" + hex$3(this.r) + hex$3(this.g) + hex$3(this.b);
    }

    function rgb_formatRgb$3() {
      var a = this.opacity; a = isNaN(a) ? 1 : Math.max(0, Math.min(1, a));
      return (a === 1 ? "rgb(" : "rgba(")
          + Math.max(0, Math.min(255, Math.round(this.r) || 0)) + ", "
          + Math.max(0, Math.min(255, Math.round(this.g) || 0)) + ", "
          + Math.max(0, Math.min(255, Math.round(this.b) || 0))
          + (a === 1 ? ")" : ", " + a + ")");
    }

    function hex$3(value) {
      value = Math.max(0, Math.min(255, Math.round(value) || 0));
      return (value < 16 ? "0" : "") + value.toString(16);
    }

    function hsla$4(h, s, l, a) {
      if (a <= 0) h = s = l = NaN;
      else if (l <= 0 || l >= 1) h = s = NaN;
      else if (s <= 0) h = NaN;
      return new Hsl$4(h, s, l, a);
    }

    function hslConvert$4(o) {
      if (o instanceof Hsl$4) return new Hsl$4(o.h, o.s, o.l, o.opacity);
      if (!(o instanceof Color$4)) o = color$4(o);
      if (!o) return new Hsl$4;
      if (o instanceof Hsl$4) return o;
      o = o.rgb();
      var r = o.r / 255,
          g = o.g / 255,
          b = o.b / 255,
          min = Math.min(r, g, b),
          max = Math.max(r, g, b),
          h = NaN,
          s = max - min,
          l = (max + min) / 2;
      if (s) {
        if (r === max) h = (g - b) / s + (g < b) * 6;
        else if (g === max) h = (b - r) / s + 2;
        else h = (r - g) / s + 4;
        s /= l < 0.5 ? max + min : 2 - max - min;
        h *= 60;
      } else {
        s = l > 0 && l < 1 ? 0 : h;
      }
      return new Hsl$4(h, s, l, o.opacity);
    }

    function hsl$8(h, s, l, opacity) {
      return arguments.length === 1 ? hslConvert$4(h) : new Hsl$4(h, s, l, opacity == null ? 1 : opacity);
    }

    function Hsl$4(h, s, l, opacity) {
      this.h = +h;
      this.s = +s;
      this.l = +l;
      this.opacity = +opacity;
    }

    define$4(Hsl$4, hsl$8, extend$4(Color$4, {
      brighter: function(k) {
        k = k == null ? brighter$4 : Math.pow(brighter$4, k);
        return new Hsl$4(this.h, this.s, this.l * k, this.opacity);
      },
      darker: function(k) {
        k = k == null ? darker$4 : Math.pow(darker$4, k);
        return new Hsl$4(this.h, this.s, this.l * k, this.opacity);
      },
      rgb: function() {
        var h = this.h % 360 + (this.h < 0) * 360,
            s = isNaN(h) || isNaN(this.s) ? 0 : this.s,
            l = this.l,
            m2 = l + (l < 0.5 ? l : 1 - l) * s,
            m1 = 2 * l - m2;
        return new Rgb$4(
          hsl2rgb$4(h >= 240 ? h - 240 : h + 120, m1, m2),
          hsl2rgb$4(h, m1, m2),
          hsl2rgb$4(h < 120 ? h + 240 : h - 120, m1, m2),
          this.opacity
        );
      },
      displayable: function() {
        return (0 <= this.s && this.s <= 1 || isNaN(this.s))
            && (0 <= this.l && this.l <= 1)
            && (0 <= this.opacity && this.opacity <= 1);
      },
      formatHsl: function() {
        var a = this.opacity; a = isNaN(a) ? 1 : Math.max(0, Math.min(1, a));
        return (a === 1 ? "hsl(" : "hsla(")
            + (this.h || 0) + ", "
            + (this.s || 0) * 100 + "%, "
            + (this.l || 0) * 100 + "%"
            + (a === 1 ? ")" : ", " + a + ")");
      }
    }));

    /* From FvD 13.37, CSS Color Module Level 3 */
    function hsl2rgb$4(h, m1, m2) {
      return (h < 60 ? m1 + (m2 - m1) * h / 60
          : h < 180 ? m2
          : h < 240 ? m1 + (m2 - m1) * (240 - h) / 60
          : m1) * 255;
    }

    var deg2rad$3 = Math.PI / 180;
    var rad2deg$3 = 180 / Math.PI;

    var A$3 = -0.14861,
        B$3 = +1.78277,
        C$3 = -0.29227,
        D$3 = -0.90649,
        E$3 = +1.97294,
        ED$3 = E$3 * D$3,
        EB$3 = E$3 * B$3,
        BC_DA$3 = B$3 * C$3 - D$3 * A$3;

    function cubehelixConvert$3(o) {
      if (o instanceof Cubehelix$3) return new Cubehelix$3(o.h, o.s, o.l, o.opacity);
      if (!(o instanceof Rgb$4)) o = rgbConvert$4(o);
      var r = o.r / 255,
          g = o.g / 255,
          b = o.b / 255,
          l = (BC_DA$3 * b + ED$3 * r - EB$3 * g) / (BC_DA$3 + ED$3 - EB$3),
          bl = b - l,
          k = (E$3 * (g - l) - C$3 * bl) / D$3,
          s = Math.sqrt(k * k + bl * bl) / (E$3 * l * (1 - l)), // NaN if l=0 or l=1
          h = s ? Math.atan2(k, bl) * rad2deg$3 - 120 : NaN;
      return new Cubehelix$3(h < 0 ? h + 360 : h, s, l, o.opacity);
    }

    function cubehelix$8(h, s, l, opacity) {
      return arguments.length === 1 ? cubehelixConvert$3(h) : new Cubehelix$3(h, s, l, opacity == null ? 1 : opacity);
    }

    function Cubehelix$3(h, s, l, opacity) {
      this.h = +h;
      this.s = +s;
      this.l = +l;
      this.opacity = +opacity;
    }

    define$4(Cubehelix$3, cubehelix$8, extend$4(Color$4, {
      brighter: function(k) {
        k = k == null ? brighter$4 : Math.pow(brighter$4, k);
        return new Cubehelix$3(this.h, this.s, this.l * k, this.opacity);
      },
      darker: function(k) {
        k = k == null ? darker$4 : Math.pow(darker$4, k);
        return new Cubehelix$3(this.h, this.s, this.l * k, this.opacity);
      },
      rgb: function() {
        var h = isNaN(this.h) ? 0 : (this.h + 120) * deg2rad$3,
            l = +this.l,
            a = isNaN(this.s) ? 0 : this.s * l * (1 - l),
            cosh = Math.cos(h),
            sinh = Math.sin(h);
        return new Rgb$4(
          255 * (l + a * (A$3 * cosh + B$3 * sinh)),
          255 * (l + a * (C$3 * cosh + D$3 * sinh)),
          255 * (l + a * (E$3 * cosh)),
          this.opacity
        );
      }
    }));

    function constant$p(x) {
      return function() {
        return x;
      };
    }

    function linear$6(a, d) {
      return function(t) {
        return a + t * d;
      };
    }

    function exponential$5(a, b, y) {
      return a = Math.pow(a, y), b = Math.pow(b, y) - a, y = 1 / y, function(t) {
        return Math.pow(a + t * b, y);
      };
    }

    function hue$3(a, b) {
      var d = b - a;
      return d ? linear$6(a, d > 180 || d < -180 ? d - 360 * Math.round(d / 360) : d) : constant$p(isNaN(a) ? b : a);
    }

    function gamma$3(y) {
      return (y = +y) === 1 ? nogamma$3 : function(a, b) {
        return b - a ? exponential$5(a, b, y) : constant$p(isNaN(a) ? b : a);
      };
    }

    function nogamma$3(a, b) {
      var d = b - a;
      return d ? linear$6(a, d) : constant$p(isNaN(a) ? b : a);
    }

    var rgb$6 = (function rgbGamma(y) {
      var color = gamma$3(y);

      function rgb(start, end) {
        var r = color((start = rgb$5(start)).r, (end = rgb$5(end)).r),
            g = color(start.g, end.g),
            b = color(start.b, end.b),
            opacity = nogamma$3(start.opacity, end.opacity);
        return function(t) {
          start.r = r(t);
          start.g = g(t);
          start.b = b(t);
          start.opacity = opacity(t);
          return start + "";
        };
      }

      rgb.gamma = rgbGamma;

      return rgb;
    })(1);

    function numberArray$2(a, b) {
      if (!b) b = [];
      var n = a ? Math.min(b.length, a.length) : 0,
          c = b.slice(),
          i;
      return function(t) {
        for (i = 0; i < n; ++i) c[i] = a[i] * (1 - t) + b[i] * t;
        return c;
      };
    }

    function isNumberArray$2(x) {
      return ArrayBuffer.isView(x) && !(x instanceof DataView);
    }

    function genericArray$2(a, b) {
      var nb = b ? b.length : 0,
          na = a ? Math.min(nb, a.length) : 0,
          x = new Array(na),
          c = new Array(nb),
          i;

      for (i = 0; i < na; ++i) x[i] = interpolateValue$1(a[i], b[i]);
      for (; i < nb; ++i) c[i] = b[i];

      return function(t) {
        for (i = 0; i < na; ++i) c[i] = x[i](t);
        return c;
      };
    }

    function date$4(a, b) {
      var d = new Date;
      return a = +a, b = +b, function(t) {
        return d.setTime(a * (1 - t) + b * t), d;
      };
    }

    function reinterpolate(a, b) {
      return a = +a, b = +b, function(t) {
        return a * (1 - t) + b * t;
      };
    }

    function object$5(a, b) {
      var i = {},
          c = {},
          k;

      if (a === null || typeof a !== "object") a = {};
      if (b === null || typeof b !== "object") b = {};

      for (k in b) {
        if (k in a) {
          i[k] = interpolateValue$1(a[k], b[k]);
        } else {
          c[k] = b[k];
        }
      }

      return function(t) {
        for (k in i) c[k] = i[k](t);
        return c;
      };
    }

    var reA$3 = /[-+]?(?:\d+\.?\d*|\.?\d+)(?:[eE][-+]?\d+)?/g,
        reB$3 = new RegExp(reA$3.source, "g");

    function zero$3(b) {
      return function() {
        return b;
      };
    }

    function one$3(b) {
      return function(t) {
        return b(t) + "";
      };
    }

    function string$1(a, b) {
      var bi = reA$3.lastIndex = reB$3.lastIndex = 0, // scan index for next number in b
          am, // current match in a
          bm, // current match in b
          bs, // string preceding current number in b, if any
          i = -1, // index in s
          s = [], // string constants and placeholders
          q = []; // number interpolators

      // Coerce inputs to strings.
      a = a + "", b = b + "";

      // Interpolate pairs of numbers in a & b.
      while ((am = reA$3.exec(a))
          && (bm = reB$3.exec(b))) {
        if ((bs = bm.index) > bi) { // a string precedes the next number in b
          bs = b.slice(bi, bs);
          if (s[i]) s[i] += bs; // coalesce with previous string
          else s[++i] = bs;
        }
        if ((am = am[0]) === (bm = bm[0])) { // numbers in a & b match
          if (s[i]) s[i] += bm; // coalesce with previous string
          else s[++i] = bm;
        } else { // interpolate non-matching numbers
          s[++i] = null;
          q.push({i: i, x: reinterpolate(am, bm)});
        }
        bi = reB$3.lastIndex;
      }

      // Add remains of b.
      if (bi < b.length) {
        bs = b.slice(bi);
        if (s[i]) s[i] += bs; // coalesce with previous string
        else s[++i] = bs;
      }

      // Special optimization for only a single match.
      // Otherwise, interpolate each of the numbers and rejoin the string.
      return s.length < 2 ? (q[0]
          ? one$3(q[0].x)
          : zero$3(b))
          : (b = q.length, function(t) {
              for (var i = 0, o; i < b; ++i) s[(o = q[i]).i] = o.x(t);
              return s.join("");
            });
    }

    function interpolateValue$1(a, b) {
      var t = typeof b, c;
      return b == null || t === "boolean" ? constant$p(b)
          : (t === "number" ? reinterpolate
          : t === "string" ? ((c = color$4(b)) ? (b = c, rgb$6) : string$1)
          : b instanceof color$4 ? rgb$6
          : b instanceof Date ? date$4
          : isNumberArray$2(b) ? numberArray$2
          : Array.isArray(b) ? genericArray$2
          : typeof b.valueOf !== "function" && typeof b.toString !== "function" || isNaN(b) ? object$5
          : reinterpolate)(a, b);
    }

    function interpolateRound$1(a, b) {
      return a = +a, b = +b, function(t) {
        return Math.round(a * (1 - t) + b * t);
      };
    }

    function cubehelix$9(hue) {
      return (function cubehelixGamma(y) {
        y = +y;

        function cubehelix(start, end) {
          var h = hue((start = cubehelix$8(start)).h, (end = cubehelix$8(end)).h),
              s = nogamma$3(start.s, end.s),
              l = nogamma$3(start.l, end.l),
              opacity = nogamma$3(start.opacity, end.opacity);
          return function(t) {
            start.h = h(t);
            start.s = s(t);
            start.l = l(Math.pow(t, y));
            start.opacity = opacity(t);
            return start + "";
          };
        }

        cubehelix.gamma = cubehelixGamma;

        return cubehelix;
      })(1);
    }

    cubehelix$9(hue$3);
    var cubehelixLong$2 = cubehelix$9(nogamma$3);

    function constant$q(x) {
      return function() {
        return x;
      };
    }

    function number$8(x) {
      return +x;
    }

    var unit$1 = [0, 1];

    function deinterpolateLinear(a, b) {
      return (b -= (a = +a))
          ? function(x) { return (x - a) / b; }
          : constant$q(b);
    }

    function deinterpolateClamp(deinterpolate) {
      return function(a, b) {
        var d = deinterpolate(a = +a, b = +b);
        return function(x) { return x <= a ? 0 : x >= b ? 1 : d(x); };
      };
    }

    function reinterpolateClamp(reinterpolate) {
      return function(a, b) {
        var r = reinterpolate(a = +a, b = +b);
        return function(t) { return t <= 0 ? a : t >= 1 ? b : r(t); };
      };
    }

    function bimap$1(domain, range, deinterpolate, reinterpolate) {
      var d0 = domain[0], d1 = domain[1], r0 = range[0], r1 = range[1];
      if (d1 < d0) d0 = deinterpolate(d1, d0), r0 = reinterpolate(r1, r0);
      else d0 = deinterpolate(d0, d1), r0 = reinterpolate(r0, r1);
      return function(x) { return r0(d0(x)); };
    }

    function polymap$1(domain, range, deinterpolate, reinterpolate) {
      var j = Math.min(domain.length, range.length) - 1,
          d = new Array(j),
          r = new Array(j),
          i = -1;

      // Reverse descending domains.
      if (domain[j] < domain[0]) {
        domain = domain.slice().reverse();
        range = range.slice().reverse();
      }

      while (++i < j) {
        d[i] = deinterpolate(domain[i], domain[i + 1]);
        r[i] = reinterpolate(range[i], range[i + 1]);
      }

      return function(x) {
        var i = bisectRight$2(domain, x, 1, j) - 1;
        return r[i](d[i](x));
      };
    }

    function copy$2(source, target) {
      return target
          .domain(source.domain())
          .range(source.range())
          .interpolate(source.interpolate())
          .clamp(source.clamp());
    }

    // deinterpolate(a, b)(x) takes a domain value x in [a,b] and returns the corresponding parameter t in [0,1].
    // reinterpolate(a, b)(t) takes a parameter t in [0,1] and returns the corresponding domain value x in [a,b].
    function continuous$1(deinterpolate, reinterpolate) {
      var domain = unit$1,
          range = unit$1,
          interpolate = interpolateValue$1,
          clamp = false,
          piecewise,
          output,
          input;

      function rescale() {
        piecewise = Math.min(domain.length, range.length) > 2 ? polymap$1 : bimap$1;
        output = input = null;
        return scale;
      }

      function scale(x) {
        return (output || (output = piecewise(domain, range, clamp ? deinterpolateClamp(deinterpolate) : deinterpolate, interpolate)))(+x);
      }

      scale.invert = function(y) {
        return (input || (input = piecewise(range, domain, deinterpolateLinear, clamp ? reinterpolateClamp(reinterpolate) : reinterpolate)))(+y);
      };

      scale.domain = function(_) {
        return arguments.length ? (domain = map$8.call(_, number$8), rescale()) : domain.slice();
      };

      scale.range = function(_) {
        return arguments.length ? (range = slice$c.call(_), rescale()) : range.slice();
      };

      scale.rangeRound = function(_) {
        return range = slice$c.call(_), interpolate = interpolateRound$1, rescale();
      };

      scale.clamp = function(_) {
        return arguments.length ? (clamp = !!_, rescale()) : clamp;
      };

      scale.interpolate = function(_) {
        return arguments.length ? (interpolate = _, rescale()) : interpolate;
      };

      return rescale();
    }

    // Computes the decimal coefficient and exponent of the specified number x with
    // significant digits p, where x is positive and p is in [1, 21] or undefined.
    // For example, formatDecimal(1.23) returns ["123", 0].
    function formatDecimal$2(x, p) {
      if ((i = (x = p ? x.toExponential(p - 1) : x.toExponential()).indexOf("e")) < 0) return null; // NaN, ±Infinity
      var i, coefficient = x.slice(0, i);

      // The string returned by toExponential either has the form \d\.\d+e[-+]\d+
      // (e.g., 1.2e+3) or the form \de[-+]\d+ (e.g., 1e+3).
      return [
        coefficient.length > 1 ? coefficient[0] + coefficient.slice(2) : coefficient,
        +x.slice(i + 1)
      ];
    }

    function exponent$4(x) {
      return x = formatDecimal$2(Math.abs(x)), x ? x[1] : NaN;
    }

    function formatGroup$2(grouping, thousands) {
      return function(value, width) {
        var i = value.length,
            t = [],
            j = 0,
            g = grouping[0],
            length = 0;

        while (i > 0 && g > 0) {
          if (length + g + 1 > width) g = Math.max(1, width - length);
          t.push(value.substring(i -= g, i + g));
          if ((length += g + 1) > width) break;
          g = grouping[j = (j + 1) % grouping.length];
        }

        return t.reverse().join(thousands);
      };
    }

    function formatNumerals$2(numerals) {
      return function(value) {
        return value.replace(/[0-9]/g, function(i) {
          return numerals[+i];
        });
      };
    }

    // [[fill]align][sign][symbol][0][width][,][.precision][~][type]
    var re$2 = /^(?:(.)?([<>=^]))?([+\-( ])?([$#])?(0)?(\d+)?(,)?(\.\d+)?(~)?([a-z%])?$/i;

    function formatSpecifier$2(specifier) {
      if (!(match = re$2.exec(specifier))) throw new Error("invalid format: " + specifier);
      var match;
      return new FormatSpecifier$2({
        fill: match[1],
        align: match[2],
        sign: match[3],
        symbol: match[4],
        zero: match[5],
        width: match[6],
        comma: match[7],
        precision: match[8] && match[8].slice(1),
        trim: match[9],
        type: match[10]
      });
    }

    formatSpecifier$2.prototype = FormatSpecifier$2.prototype; // instanceof

    function FormatSpecifier$2(specifier) {
      this.fill = specifier.fill === undefined ? " " : specifier.fill + "";
      this.align = specifier.align === undefined ? ">" : specifier.align + "";
      this.sign = specifier.sign === undefined ? "-" : specifier.sign + "";
      this.symbol = specifier.symbol === undefined ? "" : specifier.symbol + "";
      this.zero = !!specifier.zero;
      this.width = specifier.width === undefined ? undefined : +specifier.width;
      this.comma = !!specifier.comma;
      this.precision = specifier.precision === undefined ? undefined : +specifier.precision;
      this.trim = !!specifier.trim;
      this.type = specifier.type === undefined ? "" : specifier.type + "";
    }

    FormatSpecifier$2.prototype.toString = function() {
      return this.fill
          + this.align
          + this.sign
          + this.symbol
          + (this.zero ? "0" : "")
          + (this.width === undefined ? "" : Math.max(1, this.width | 0))
          + (this.comma ? "," : "")
          + (this.precision === undefined ? "" : "." + Math.max(0, this.precision | 0))
          + (this.trim ? "~" : "")
          + this.type;
    };

    // Trims insignificant zeros, e.g., replaces 1.2000k with 1.2k.
    function formatTrim$1(s) {
      out: for (var n = s.length, i = 1, i0 = -1, i1; i < n; ++i) {
        switch (s[i]) {
          case ".": i0 = i1 = i; break;
          case "0": if (i0 === 0) i0 = i; i1 = i; break;
          default: if (!+s[i]) break out; if (i0 > 0) i0 = 0; break;
        }
      }
      return i0 > 0 ? s.slice(0, i0) + s.slice(i1 + 1) : s;
    }

    var prefixExponent$2;

    function formatPrefixAuto$2(x, p) {
      var d = formatDecimal$2(x, p);
      if (!d) return x + "";
      var coefficient = d[0],
          exponent = d[1],
          i = exponent - (prefixExponent$2 = Math.max(-8, Math.min(8, Math.floor(exponent / 3))) * 3) + 1,
          n = coefficient.length;
      return i === n ? coefficient
          : i > n ? coefficient + new Array(i - n + 1).join("0")
          : i > 0 ? coefficient.slice(0, i) + "." + coefficient.slice(i)
          : "0." + new Array(1 - i).join("0") + formatDecimal$2(x, Math.max(0, p + i - 1))[0]; // less than 1y!
    }

    function formatRounded$2(x, p) {
      var d = formatDecimal$2(x, p);
      if (!d) return x + "";
      var coefficient = d[0],
          exponent = d[1];
      return exponent < 0 ? "0." + new Array(-exponent).join("0") + coefficient
          : coefficient.length > exponent + 1 ? coefficient.slice(0, exponent + 1) + "." + coefficient.slice(exponent + 1)
          : coefficient + new Array(exponent - coefficient.length + 2).join("0");
    }

    var formatTypes$2 = {
      "%": function(x, p) { return (x * 100).toFixed(p); },
      "b": function(x) { return Math.round(x).toString(2); },
      "c": function(x) { return x + ""; },
      "d": function(x) { return Math.round(x).toString(10); },
      "e": function(x, p) { return x.toExponential(p); },
      "f": function(x, p) { return x.toFixed(p); },
      "g": function(x, p) { return x.toPrecision(p); },
      "o": function(x) { return Math.round(x).toString(8); },
      "p": function(x, p) { return formatRounded$2(x * 100, p); },
      "r": formatRounded$2,
      "s": formatPrefixAuto$2,
      "X": function(x) { return Math.round(x).toString(16).toUpperCase(); },
      "x": function(x) { return Math.round(x).toString(16); }
    };

    function identity$h(x) {
      return x;
    }

    var map$9 = Array.prototype.map,
        prefixes$2 = ["y","z","a","f","p","n","µ","m","","k","M","G","T","P","E","Z","Y"];

    function formatLocale$3(locale) {
      var group = locale.grouping === undefined || locale.thousands === undefined ? identity$h : formatGroup$2(map$9.call(locale.grouping, Number), locale.thousands + ""),
          currencyPrefix = locale.currency === undefined ? "" : locale.currency[0] + "",
          currencySuffix = locale.currency === undefined ? "" : locale.currency[1] + "",
          decimal = locale.decimal === undefined ? "." : locale.decimal + "",
          numerals = locale.numerals === undefined ? identity$h : formatNumerals$2(map$9.call(locale.numerals, String)),
          percent = locale.percent === undefined ? "%" : locale.percent + "",
          minus = locale.minus === undefined ? "-" : locale.minus + "",
          nan = locale.nan === undefined ? "NaN" : locale.nan + "";

      function newFormat(specifier) {
        specifier = formatSpecifier$2(specifier);

        var fill = specifier.fill,
            align = specifier.align,
            sign = specifier.sign,
            symbol = specifier.symbol,
            zero = specifier.zero,
            width = specifier.width,
            comma = specifier.comma,
            precision = specifier.precision,
            trim = specifier.trim,
            type = specifier.type;

        // The "n" type is an alias for ",g".
        if (type === "n") comma = true, type = "g";

        // The "" type, and any invalid type, is an alias for ".12~g".
        else if (!formatTypes$2[type]) precision === undefined && (precision = 12), trim = true, type = "g";

        // If zero fill is specified, padding goes after sign and before digits.
        if (zero || (fill === "0" && align === "=")) zero = true, fill = "0", align = "=";

        // Compute the prefix and suffix.
        // For SI-prefix, the suffix is lazily computed.
        var prefix = symbol === "$" ? currencyPrefix : symbol === "#" && /[boxX]/.test(type) ? "0" + type.toLowerCase() : "",
            suffix = symbol === "$" ? currencySuffix : /[%p]/.test(type) ? percent : "";

        // What format function should we use?
        // Is this an integer type?
        // Can this type generate exponential notation?
        var formatType = formatTypes$2[type],
            maybeSuffix = /[defgprs%]/.test(type);

        // Set the default precision if not specified,
        // or clamp the specified precision to the supported range.
        // For significant precision, it must be in [1, 21].
        // For fixed precision, it must be in [0, 20].
        precision = precision === undefined ? 6
            : /[gprs]/.test(type) ? Math.max(1, Math.min(21, precision))
            : Math.max(0, Math.min(20, precision));

        function format(value) {
          var valuePrefix = prefix,
              valueSuffix = suffix,
              i, n, c;

          if (type === "c") {
            valueSuffix = formatType(value) + valueSuffix;
            value = "";
          } else {
            value = +value;

            // Determine the sign. -0 is not less than 0, but 1 / -0 is!
            var valueNegative = value < 0 || 1 / value < 0;

            // Perform the initial formatting.
            value = isNaN(value) ? nan : formatType(Math.abs(value), precision);

            // Trim insignificant zeros.
            if (trim) value = formatTrim$1(value);

            // If a negative value rounds to zero after formatting, and no explicit positive sign is requested, hide the sign.
            if (valueNegative && +value === 0 && sign !== "+") valueNegative = false;

            // Compute the prefix and suffix.
            valuePrefix = (valueNegative ? (sign === "(" ? sign : minus) : sign === "-" || sign === "(" ? "" : sign) + valuePrefix;
            valueSuffix = (type === "s" ? prefixes$2[8 + prefixExponent$2 / 3] : "") + valueSuffix + (valueNegative && sign === "(" ? ")" : "");

            // Break the formatted value into the integer “value” part that can be
            // grouped, and fractional or exponential “suffix” part that is not.
            if (maybeSuffix) {
              i = -1, n = value.length;
              while (++i < n) {
                if (c = value.charCodeAt(i), 48 > c || c > 57) {
                  valueSuffix = (c === 46 ? decimal + value.slice(i + 1) : value.slice(i)) + valueSuffix;
                  value = value.slice(0, i);
                  break;
                }
              }
            }
          }

          // If the fill character is not "0", grouping is applied before padding.
          if (comma && !zero) value = group(value, Infinity);

          // Compute the padding.
          var length = valuePrefix.length + value.length + valueSuffix.length,
              padding = length < width ? new Array(width - length + 1).join(fill) : "";

          // If the fill character is "0", grouping is applied after padding.
          if (comma && zero) value = group(padding + value, padding.length ? width - valueSuffix.length : Infinity), padding = "";

          // Reconstruct the final output based on the desired alignment.
          switch (align) {
            case "<": value = valuePrefix + value + valueSuffix + padding; break;
            case "=": value = valuePrefix + padding + value + valueSuffix; break;
            case "^": value = padding.slice(0, length = padding.length >> 1) + valuePrefix + value + valueSuffix + padding.slice(length); break;
            default: value = padding + valuePrefix + value + valueSuffix; break;
          }

          return numerals(value);
        }

        format.toString = function() {
          return specifier + "";
        };

        return format;
      }

      function formatPrefix(specifier, value) {
        var f = newFormat((specifier = formatSpecifier$2(specifier), specifier.type = "f", specifier)),
            e = Math.max(-8, Math.min(8, Math.floor(exponent$4(value) / 3))) * 3,
            k = Math.pow(10, -e),
            prefix = prefixes$2[8 + e / 3];
        return function(value) {
          return f(k * value) + prefix;
        };
      }

      return {
        format: newFormat,
        formatPrefix: formatPrefix
      };
    }

    var locale$3;
    var format$2;
    var formatPrefix$2;

    defaultLocale$3({
      decimal: ".",
      thousands: ",",
      grouping: [3],
      currency: ["$", ""],
      minus: "-"
    });

    function defaultLocale$3(definition) {
      locale$3 = formatLocale$3(definition);
      format$2 = locale$3.format;
      formatPrefix$2 = locale$3.formatPrefix;
      return locale$3;
    }

    function precisionFixed$2(step) {
      return Math.max(0, -exponent$4(Math.abs(step)));
    }

    function precisionPrefix$2(step, value) {
      return Math.max(0, Math.max(-8, Math.min(8, Math.floor(exponent$4(value) / 3))) * 3 - exponent$4(Math.abs(step)));
    }

    function precisionRound$2(step, max) {
      step = Math.abs(step), max = Math.abs(max) - step;
      return Math.max(0, exponent$4(max) - exponent$4(step)) + 1;
    }

    function tickFormat$1(domain, count, specifier) {
      var start = domain[0],
          stop = domain[domain.length - 1],
          step = tickStep$2(start, stop, count == null ? 10 : count),
          precision;
      specifier = formatSpecifier$2(specifier == null ? ",f" : specifier);
      switch (specifier.type) {
        case "s": {
          var value = Math.max(Math.abs(start), Math.abs(stop));
          if (specifier.precision == null && !isNaN(precision = precisionPrefix$2(step, value))) specifier.precision = precision;
          return formatPrefix$2(specifier, value);
        }
        case "":
        case "e":
        case "g":
        case "p":
        case "r": {
          if (specifier.precision == null && !isNaN(precision = precisionRound$2(step, Math.max(Math.abs(start), Math.abs(stop))))) specifier.precision = precision - (specifier.type === "e");
          break;
        }
        case "f":
        case "%": {
          if (specifier.precision == null && !isNaN(precision = precisionFixed$2(step))) specifier.precision = precision - (specifier.type === "%") * 2;
          break;
        }
      }
      return format$2(specifier);
    }

    function linearish$1(scale) {
      var domain = scale.domain;

      scale.ticks = function(count) {
        var d = domain();
        return ticks$2(d[0], d[d.length - 1], count == null ? 10 : count);
      };

      scale.tickFormat = function(count, specifier) {
        return tickFormat$1(domain(), count, specifier);
      };

      scale.nice = function(count) {
        if (count == null) count = 10;

        var d = domain(),
            i0 = 0,
            i1 = d.length - 1,
            start = d[i0],
            stop = d[i1],
            step;

        if (stop < start) {
          step = start, start = stop, stop = step;
          step = i0, i0 = i1, i1 = step;
        }

        step = tickIncrement$2(start, stop, count);

        if (step > 0) {
          start = Math.floor(start / step) * step;
          stop = Math.ceil(stop / step) * step;
          step = tickIncrement$2(start, stop, count);
        } else if (step < 0) {
          start = Math.ceil(start * step) / step;
          stop = Math.floor(stop * step) / step;
          step = tickIncrement$2(start, stop, count);
        }

        if (step > 0) {
          d[i0] = Math.floor(start / step) * step;
          d[i1] = Math.ceil(stop / step) * step;
          domain(d);
        } else if (step < 0) {
          d[i0] = Math.ceil(start * step) / step;
          d[i1] = Math.floor(stop * step) / step;
          domain(d);
        }

        return scale;
      };

      return scale;
    }

    function linear$7() {
      var scale = continuous$1(deinterpolateLinear, reinterpolate);

      scale.copy = function() {
        return copy$2(scale, linear$7());
      };

      return linearish$1(scale);
    }

    function identity$i() {
      var domain = [0, 1];

      function scale(x) {
        return +x;
      }

      scale.invert = scale;

      scale.domain = scale.range = function(_) {
        return arguments.length ? (domain = map$8.call(_, number$8), scale) : domain.slice();
      };

      scale.copy = function() {
        return identity$i().domain(domain);
      };

      return linearish$1(scale);
    }

    function nice$1(domain, interval) {
      domain = domain.slice();

      var i0 = 0,
          i1 = domain.length - 1,
          x0 = domain[i0],
          x1 = domain[i1],
          t;

      if (x1 < x0) {
        t = i0, i0 = i1, i1 = t;
        t = x0, x0 = x1, x1 = t;
      }

      domain[i0] = interval.floor(x0);
      domain[i1] = interval.ceil(x1);
      return domain;
    }

    function deinterpolate(a, b) {
      return (b = Math.log(b / a))
          ? function(x) { return Math.log(x / a) / b; }
          : constant$q(b);
    }

    function reinterpolate$1(a, b) {
      return a < 0
          ? function(t) { return -Math.pow(-b, t) * Math.pow(-a, 1 - t); }
          : function(t) { return Math.pow(b, t) * Math.pow(a, 1 - t); };
    }

    function pow10$1(x) {
      return isFinite(x) ? +("1e" + x) : x < 0 ? 0 : x;
    }

    function powp$1(base) {
      return base === 10 ? pow10$1
          : base === Math.E ? Math.exp
          : function(x) { return Math.pow(base, x); };
    }

    function logp$1(base) {
      return base === Math.E ? Math.log
          : base === 10 && Math.log10
          || base === 2 && Math.log2
          || (base = Math.log(base), function(x) { return Math.log(x) / base; });
    }

    function reflect$1(f) {
      return function(x) {
        return -f(-x);
      };
    }

    function log$3() {
      var scale = continuous$1(deinterpolate, reinterpolate$1).domain([1, 10]),
          domain = scale.domain,
          base = 10,
          logs = logp$1(10),
          pows = powp$1(10);

      function rescale() {
        logs = logp$1(base), pows = powp$1(base);
        if (domain()[0] < 0) logs = reflect$1(logs), pows = reflect$1(pows);
        return scale;
      }

      scale.base = function(_) {
        return arguments.length ? (base = +_, rescale()) : base;
      };

      scale.domain = function(_) {
        return arguments.length ? (domain(_), rescale()) : domain();
      };

      scale.ticks = function(count) {
        var d = domain(),
            u = d[0],
            v = d[d.length - 1],
            r;

        if (r = v < u) i = u, u = v, v = i;

        var i = logs(u),
            j = logs(v),
            p,
            k,
            t,
            n = count == null ? 10 : +count,
            z = [];

        if (!(base % 1) && j - i < n) {
          i = Math.round(i) - 1, j = Math.round(j) + 1;
          if (u > 0) for (; i < j; ++i) {
            for (k = 1, p = pows(i); k < base; ++k) {
              t = p * k;
              if (t < u) continue;
              if (t > v) break;
              z.push(t);
            }
          } else for (; i < j; ++i) {
            for (k = base - 1, p = pows(i); k >= 1; --k) {
              t = p * k;
              if (t < u) continue;
              if (t > v) break;
              z.push(t);
            }
          }
        } else {
          z = ticks$2(i, j, Math.min(j - i, n)).map(pows);
        }

        return r ? z.reverse() : z;
      };

      scale.tickFormat = function(count, specifier) {
        if (specifier == null) specifier = base === 10 ? ".0e" : ",";
        if (typeof specifier !== "function") specifier = format$2(specifier);
        if (count === Infinity) return specifier;
        if (count == null) count = 10;
        var k = Math.max(1, base * count / scale.ticks().length); // TODO fast estimate?
        return function(d) {
          var i = d / pows(Math.round(logs(d)));
          if (i * base < base - 0.5) i *= base;
          return i <= k ? specifier(d) : "";
        };
      };

      scale.nice = function() {
        return domain(nice$1(domain(), {
          floor: function(x) { return pows(Math.floor(logs(x))); },
          ceil: function(x) { return pows(Math.ceil(logs(x))); }
        }));
      };

      scale.copy = function() {
        return copy$2(scale, log$3().base(base));
      };

      return scale;
    }

    function raise$3(x, exponent) {
      return x < 0 ? -Math.pow(-x, exponent) : Math.pow(x, exponent);
    }

    function pow$3() {
      var exponent = 1,
          scale = continuous$1(deinterpolate, reinterpolate),
          domain = scale.domain;

      function deinterpolate(a, b) {
        return (b = raise$3(b, exponent) - (a = raise$3(a, exponent)))
            ? function(x) { return (raise$3(x, exponent) - a) / b; }
            : constant$q(b);
      }

      function reinterpolate(a, b) {
        b = raise$3(b, exponent) - (a = raise$3(a, exponent));
        return function(t) { return raise$3(a + b * t, 1 / exponent); };
      }

      scale.exponent = function(_) {
        return arguments.length ? (exponent = +_, domain(domain())) : exponent;
      };

      scale.copy = function() {
        return copy$2(scale, pow$3().exponent(exponent));
      };

      return linearish$1(scale);
    }

    function sqrt$4() {
      return pow$3().exponent(0.5);
    }

    function quantile$2() {
      var domain = [],
          range = [],
          thresholds = [];

      function rescale() {
        var i = 0, n = Math.max(1, range.length);
        thresholds = new Array(n - 1);
        while (++i < n) thresholds[i - 1] = threshold$2(domain, i / n);
        return scale;
      }

      function scale(x) {
        if (!isNaN(x = +x)) return range[bisectRight$2(thresholds, x)];
      }

      scale.invertExtent = function(y) {
        var i = range.indexOf(y);
        return i < 0 ? [NaN, NaN] : [
          i > 0 ? thresholds[i - 1] : domain[0],
          i < thresholds.length ? thresholds[i] : domain[domain.length - 1]
        ];
      };

      scale.domain = function(_) {
        if (!arguments.length) return domain.slice();
        domain = [];
        for (var i = 0, n = _.length, d; i < n; ++i) if (d = _[i], d != null && !isNaN(d = +d)) domain.push(d);
        domain.sort(ascending$9);
        return rescale();
      };

      scale.range = function(_) {
        return arguments.length ? (range = slice$c.call(_), rescale()) : range.slice();
      };

      scale.quantiles = function() {
        return thresholds.slice();
      };

      scale.copy = function() {
        return quantile$2()
            .domain(domain)
            .range(range);
      };

      return scale;
    }

    function quantize$3() {
      var x0 = 0,
          x1 = 1,
          n = 1,
          domain = [0.5],
          range = [0, 1];

      function scale(x) {
        if (x <= x) return range[bisectRight$2(domain, x, 0, n)];
      }

      function rescale() {
        var i = -1;
        domain = new Array(n);
        while (++i < n) domain[i] = ((i + 1) * x1 - (i - n) * x0) / (n + 1);
        return scale;
      }

      scale.domain = function(_) {
        return arguments.length ? (x0 = +_[0], x1 = +_[1], rescale()) : [x0, x1];
      };

      scale.range = function(_) {
        return arguments.length ? (n = (range = slice$c.call(_)).length - 1, rescale()) : range.slice();
      };

      scale.invertExtent = function(y) {
        var i = range.indexOf(y);
        return i < 0 ? [NaN, NaN]
            : i < 1 ? [x0, domain[0]]
            : i >= n ? [domain[n - 1], x1]
            : [domain[i - 1], domain[i]];
      };

      scale.copy = function() {
        return quantize$3()
            .domain([x0, x1])
            .range(range);
      };

      return linearish$1(scale);
    }

    function threshold$3() {
      var domain = [0.5],
          range = [0, 1],
          n = 1;

      function scale(x) {
        if (x <= x) return range[bisectRight$2(domain, x, 0, n)];
      }

      scale.domain = function(_) {
        return arguments.length ? (domain = slice$c.call(_), n = Math.min(domain.length, range.length - 1), scale) : domain.slice();
      };

      scale.range = function(_) {
        return arguments.length ? (range = slice$c.call(_), n = Math.min(domain.length, range.length - 1), scale) : range.slice();
      };

      scale.invertExtent = function(y) {
        var i = range.indexOf(y);
        return [domain[i - 1], domain[i]];
      };

      scale.copy = function() {
        return threshold$3()
            .domain(domain)
            .range(range);
      };

      return scale;
    }

    var t0$4 = new Date,
        t1$4 = new Date;

    function newInterval$1(floori, offseti, count, field) {

      function interval(date) {
        return floori(date = arguments.length === 0 ? new Date : new Date(+date)), date;
      }

      interval.floor = function(date) {
        return floori(date = new Date(+date)), date;
      };

      interval.ceil = function(date) {
        return floori(date = new Date(date - 1)), offseti(date, 1), floori(date), date;
      };

      interval.round = function(date) {
        var d0 = interval(date),
            d1 = interval.ceil(date);
        return date - d0 < d1 - date ? d0 : d1;
      };

      interval.offset = function(date, step) {
        return offseti(date = new Date(+date), step == null ? 1 : Math.floor(step)), date;
      };

      interval.range = function(start, stop, step) {
        var range = [], previous;
        start = interval.ceil(start);
        step = step == null ? 1 : Math.floor(step);
        if (!(start < stop) || !(step > 0)) return range; // also handles Invalid Date
        do range.push(previous = new Date(+start)), offseti(start, step), floori(start);
        while (previous < start && start < stop);
        return range;
      };

      interval.filter = function(test) {
        return newInterval$1(function(date) {
          if (date >= date) while (floori(date), !test(date)) date.setTime(date - 1);
        }, function(date, step) {
          if (date >= date) {
            if (step < 0) while (++step <= 0) {
              while (offseti(date, -1), !test(date)) {} // eslint-disable-line no-empty
            } else while (--step >= 0) {
              while (offseti(date, +1), !test(date)) {} // eslint-disable-line no-empty
            }
          }
        });
      };

      if (count) {
        interval.count = function(start, end) {
          t0$4.setTime(+start), t1$4.setTime(+end);
          floori(t0$4), floori(t1$4);
          return Math.floor(count(t0$4, t1$4));
        };

        interval.every = function(step) {
          step = Math.floor(step);
          return !isFinite(step) || !(step > 0) ? null
              : !(step > 1) ? interval
              : interval.filter(field
                  ? function(d) { return field(d) % step === 0; }
                  : function(d) { return interval.count(0, d) % step === 0; });
        };
      }

      return interval;
    }

    var millisecond$1 = newInterval$1(function() {
      // noop
    }, function(date, step) {
      date.setTime(+date + step);
    }, function(start, end) {
      return end - start;
    });

    // An optimized implementation for this simple case.
    millisecond$1.every = function(k) {
      k = Math.floor(k);
      if (!isFinite(k) || !(k > 0)) return null;
      if (!(k > 1)) return millisecond$1;
      return newInterval$1(function(date) {
        date.setTime(Math.floor(date / k) * k);
      }, function(date, step) {
        date.setTime(+date + step * k);
      }, function(start, end) {
        return (end - start) / k;
      });
    };

    var durationSecond$2 = 1e3;
    var durationMinute$2 = 6e4;
    var durationHour$2 = 36e5;
    var durationDay$2 = 864e5;
    var durationWeek$2 = 6048e5;

    var second$1 = newInterval$1(function(date) {
      date.setTime(date - date.getMilliseconds());
    }, function(date, step) {
      date.setTime(+date + step * durationSecond$2);
    }, function(start, end) {
      return (end - start) / durationSecond$2;
    }, function(date) {
      return date.getUTCSeconds();
    });

    var minute$1 = newInterval$1(function(date) {
      date.setTime(date - date.getMilliseconds() - date.getSeconds() * durationSecond$2);
    }, function(date, step) {
      date.setTime(+date + step * durationMinute$2);
    }, function(start, end) {
      return (end - start) / durationMinute$2;
    }, function(date) {
      return date.getMinutes();
    });

    var hour$1 = newInterval$1(function(date) {
      date.setTime(date - date.getMilliseconds() - date.getSeconds() * durationSecond$2 - date.getMinutes() * durationMinute$2);
    }, function(date, step) {
      date.setTime(+date + step * durationHour$2);
    }, function(start, end) {
      return (end - start) / durationHour$2;
    }, function(date) {
      return date.getHours();
    });

    var day$1 = newInterval$1(function(date) {
      date.setHours(0, 0, 0, 0);
    }, function(date, step) {
      date.setDate(date.getDate() + step);
    }, function(start, end) {
      return (end - start - (end.getTimezoneOffset() - start.getTimezoneOffset()) * durationMinute$2) / durationDay$2;
    }, function(date) {
      return date.getDate() - 1;
    });

    function weekday$1(i) {
      return newInterval$1(function(date) {
        date.setDate(date.getDate() - (date.getDay() + 7 - i) % 7);
        date.setHours(0, 0, 0, 0);
      }, function(date, step) {
        date.setDate(date.getDate() + step * 7);
      }, function(start, end) {
        return (end - start - (end.getTimezoneOffset() - start.getTimezoneOffset()) * durationMinute$2) / durationWeek$2;
      });
    }

    var sunday$1 = weekday$1(0);
    var monday$1 = weekday$1(1);
    var tuesday$1 = weekday$1(2);
    var wednesday$1 = weekday$1(3);
    var thursday$1 = weekday$1(4);
    var friday$1 = weekday$1(5);
    var saturday$1 = weekday$1(6);

    var month$1 = newInterval$1(function(date) {
      date.setDate(1);
      date.setHours(0, 0, 0, 0);
    }, function(date, step) {
      date.setMonth(date.getMonth() + step);
    }, function(start, end) {
      return end.getMonth() - start.getMonth() + (end.getFullYear() - start.getFullYear()) * 12;
    }, function(date) {
      return date.getMonth();
    });

    var year$1 = newInterval$1(function(date) {
      date.setMonth(0, 1);
      date.setHours(0, 0, 0, 0);
    }, function(date, step) {
      date.setFullYear(date.getFullYear() + step);
    }, function(start, end) {
      return end.getFullYear() - start.getFullYear();
    }, function(date) {
      return date.getFullYear();
    });

    // An optimized implementation for this simple case.
    year$1.every = function(k) {
      return !isFinite(k = Math.floor(k)) || !(k > 0) ? null : newInterval$1(function(date) {
        date.setFullYear(Math.floor(date.getFullYear() / k) * k);
        date.setMonth(0, 1);
        date.setHours(0, 0, 0, 0);
      }, function(date, step) {
        date.setFullYear(date.getFullYear() + step * k);
      });
    };

    var utcMinute$1 = newInterval$1(function(date) {
      date.setUTCSeconds(0, 0);
    }, function(date, step) {
      date.setTime(+date + step * durationMinute$2);
    }, function(start, end) {
      return (end - start) / durationMinute$2;
    }, function(date) {
      return date.getUTCMinutes();
    });

    var utcHour$1 = newInterval$1(function(date) {
      date.setUTCMinutes(0, 0, 0);
    }, function(date, step) {
      date.setTime(+date + step * durationHour$2);
    }, function(start, end) {
      return (end - start) / durationHour$2;
    }, function(date) {
      return date.getUTCHours();
    });

    var utcDay$1 = newInterval$1(function(date) {
      date.setUTCHours(0, 0, 0, 0);
    }, function(date, step) {
      date.setUTCDate(date.getUTCDate() + step);
    }, function(start, end) {
      return (end - start) / durationDay$2;
    }, function(date) {
      return date.getUTCDate() - 1;
    });

    function utcWeekday$1(i) {
      return newInterval$1(function(date) {
        date.setUTCDate(date.getUTCDate() - (date.getUTCDay() + 7 - i) % 7);
        date.setUTCHours(0, 0, 0, 0);
      }, function(date, step) {
        date.setUTCDate(date.getUTCDate() + step * 7);
      }, function(start, end) {
        return (end - start) / durationWeek$2;
      });
    }

    var utcSunday$1 = utcWeekday$1(0);
    var utcMonday$1 = utcWeekday$1(1);
    var utcTuesday$1 = utcWeekday$1(2);
    var utcWednesday$1 = utcWeekday$1(3);
    var utcThursday$1 = utcWeekday$1(4);
    var utcFriday$1 = utcWeekday$1(5);
    var utcSaturday$1 = utcWeekday$1(6);

    var utcMonth$1 = newInterval$1(function(date) {
      date.setUTCDate(1);
      date.setUTCHours(0, 0, 0, 0);
    }, function(date, step) {
      date.setUTCMonth(date.getUTCMonth() + step);
    }, function(start, end) {
      return end.getUTCMonth() - start.getUTCMonth() + (end.getUTCFullYear() - start.getUTCFullYear()) * 12;
    }, function(date) {
      return date.getUTCMonth();
    });

    var utcYear$1 = newInterval$1(function(date) {
      date.setUTCMonth(0, 1);
      date.setUTCHours(0, 0, 0, 0);
    }, function(date, step) {
      date.setUTCFullYear(date.getUTCFullYear() + step);
    }, function(start, end) {
      return end.getUTCFullYear() - start.getUTCFullYear();
    }, function(date) {
      return date.getUTCFullYear();
    });

    // An optimized implementation for this simple case.
    utcYear$1.every = function(k) {
      return !isFinite(k = Math.floor(k)) || !(k > 0) ? null : newInterval$1(function(date) {
        date.setUTCFullYear(Math.floor(date.getUTCFullYear() / k) * k);
        date.setUTCMonth(0, 1);
        date.setUTCHours(0, 0, 0, 0);
      }, function(date, step) {
        date.setUTCFullYear(date.getUTCFullYear() + step * k);
      });
    };

    function localDate$1(d) {
      if (0 <= d.y && d.y < 100) {
        var date = new Date(-1, d.m, d.d, d.H, d.M, d.S, d.L);
        date.setFullYear(d.y);
        return date;
      }
      return new Date(d.y, d.m, d.d, d.H, d.M, d.S, d.L);
    }

    function utcDate$1(d) {
      if (0 <= d.y && d.y < 100) {
        var date = new Date(Date.UTC(-1, d.m, d.d, d.H, d.M, d.S, d.L));
        date.setUTCFullYear(d.y);
        return date;
      }
      return new Date(Date.UTC(d.y, d.m, d.d, d.H, d.M, d.S, d.L));
    }

    function newDate$1(y, m, d) {
      return {y: y, m: m, d: d, H: 0, M: 0, S: 0, L: 0};
    }

    function formatLocale$4(locale) {
      var locale_dateTime = locale.dateTime,
          locale_date = locale.date,
          locale_time = locale.time,
          locale_periods = locale.periods,
          locale_weekdays = locale.days,
          locale_shortWeekdays = locale.shortDays,
          locale_months = locale.months,
          locale_shortMonths = locale.shortMonths;

      var periodRe = formatRe$1(locale_periods),
          periodLookup = formatLookup$1(locale_periods),
          weekdayRe = formatRe$1(locale_weekdays),
          weekdayLookup = formatLookup$1(locale_weekdays),
          shortWeekdayRe = formatRe$1(locale_shortWeekdays),
          shortWeekdayLookup = formatLookup$1(locale_shortWeekdays),
          monthRe = formatRe$1(locale_months),
          monthLookup = formatLookup$1(locale_months),
          shortMonthRe = formatRe$1(locale_shortMonths),
          shortMonthLookup = formatLookup$1(locale_shortMonths);

      var formats = {
        "a": formatShortWeekday,
        "A": formatWeekday,
        "b": formatShortMonth,
        "B": formatMonth,
        "c": null,
        "d": formatDayOfMonth$1,
        "e": formatDayOfMonth$1,
        "f": formatMicroseconds$1,
        "H": formatHour24$1,
        "I": formatHour12$1,
        "j": formatDayOfYear$1,
        "L": formatMilliseconds$1,
        "m": formatMonthNumber$1,
        "M": formatMinutes$1,
        "p": formatPeriod,
        "q": formatQuarter,
        "Q": formatUnixTimestamp$1,
        "s": formatUnixTimestampSeconds$1,
        "S": formatSeconds$1,
        "u": formatWeekdayNumberMonday$1,
        "U": formatWeekNumberSunday$1,
        "V": formatWeekNumberISO$1,
        "w": formatWeekdayNumberSunday$1,
        "W": formatWeekNumberMonday$1,
        "x": null,
        "X": null,
        "y": formatYear$2,
        "Y": formatFullYear$1,
        "Z": formatZone$1,
        "%": formatLiteralPercent$1
      };

      var utcFormats = {
        "a": formatUTCShortWeekday,
        "A": formatUTCWeekday,
        "b": formatUTCShortMonth,
        "B": formatUTCMonth,
        "c": null,
        "d": formatUTCDayOfMonth$1,
        "e": formatUTCDayOfMonth$1,
        "f": formatUTCMicroseconds$1,
        "H": formatUTCHour24$1,
        "I": formatUTCHour12$1,
        "j": formatUTCDayOfYear$1,
        "L": formatUTCMilliseconds$1,
        "m": formatUTCMonthNumber$1,
        "M": formatUTCMinutes$1,
        "p": formatUTCPeriod,
        "q": formatUTCQuarter,
        "Q": formatUnixTimestamp$1,
        "s": formatUnixTimestampSeconds$1,
        "S": formatUTCSeconds$1,
        "u": formatUTCWeekdayNumberMonday$1,
        "U": formatUTCWeekNumberSunday$1,
        "V": formatUTCWeekNumberISO$1,
        "w": formatUTCWeekdayNumberSunday$1,
        "W": formatUTCWeekNumberMonday$1,
        "x": null,
        "X": null,
        "y": formatUTCYear$1,
        "Y": formatUTCFullYear$1,
        "Z": formatUTCZone$1,
        "%": formatLiteralPercent$1
      };

      var parses = {
        "a": parseShortWeekday,
        "A": parseWeekday,
        "b": parseShortMonth,
        "B": parseMonth,
        "c": parseLocaleDateTime,
        "d": parseDayOfMonth$1,
        "e": parseDayOfMonth$1,
        "f": parseMicroseconds$1,
        "H": parseHour24$1,
        "I": parseHour24$1,
        "j": parseDayOfYear$1,
        "L": parseMilliseconds$1,
        "m": parseMonthNumber$1,
        "M": parseMinutes$1,
        "p": parsePeriod,
        "q": parseQuarter$1,
        "Q": parseUnixTimestamp$1,
        "s": parseUnixTimestampSeconds$1,
        "S": parseSeconds$1,
        "u": parseWeekdayNumberMonday$1,
        "U": parseWeekNumberSunday$1,
        "V": parseWeekNumberISO$1,
        "w": parseWeekdayNumberSunday$1,
        "W": parseWeekNumberMonday$1,
        "x": parseLocaleDate,
        "X": parseLocaleTime,
        "y": parseYear$1,
        "Y": parseFullYear$1,
        "Z": parseZone$1,
        "%": parseLiteralPercent$1
      };

      // These recursive directive definitions must be deferred.
      formats.x = newFormat(locale_date, formats);
      formats.X = newFormat(locale_time, formats);
      formats.c = newFormat(locale_dateTime, formats);
      utcFormats.x = newFormat(locale_date, utcFormats);
      utcFormats.X = newFormat(locale_time, utcFormats);
      utcFormats.c = newFormat(locale_dateTime, utcFormats);

      function newFormat(specifier, formats) {
        return function(date) {
          var string = [],
              i = -1,
              j = 0,
              n = specifier.length,
              c,
              pad,
              format;

          if (!(date instanceof Date)) date = new Date(+date);

          while (++i < n) {
            if (specifier.charCodeAt(i) === 37) {
              string.push(specifier.slice(j, i));
              if ((pad = pads$1[c = specifier.charAt(++i)]) != null) c = specifier.charAt(++i);
              else pad = c === "e" ? " " : "0";
              if (format = formats[c]) c = format(date, pad);
              string.push(c);
              j = i + 1;
            }
          }

          string.push(specifier.slice(j, i));
          return string.join("");
        };
      }

      function newParse(specifier, Z) {
        return function(string) {
          var d = newDate$1(1900, undefined, 1),
              i = parseSpecifier(d, specifier, string += "", 0),
              week, day;
          if (i != string.length) return null;

          // If a UNIX timestamp is specified, return it.
          if ("Q" in d) return new Date(d.Q);
          if ("s" in d) return new Date(d.s * 1000 + ("L" in d ? d.L : 0));

          // If this is utcParse, never use the local timezone.
          if (Z && !("Z" in d)) d.Z = 0;

          // The am-pm flag is 0 for AM, and 1 for PM.
          if ("p" in d) d.H = d.H % 12 + d.p * 12;

          // If the month was not specified, inherit from the quarter.
          if (d.m === undefined) d.m = "q" in d ? d.q : 0;

          // Convert day-of-week and week-of-year to day-of-year.
          if ("V" in d) {
            if (d.V < 1 || d.V > 53) return null;
            if (!("w" in d)) d.w = 1;
            if ("Z" in d) {
              week = utcDate$1(newDate$1(d.y, 0, 1)), day = week.getUTCDay();
              week = day > 4 || day === 0 ? utcMonday$1.ceil(week) : utcMonday$1(week);
              week = utcDay$1.offset(week, (d.V - 1) * 7);
              d.y = week.getUTCFullYear();
              d.m = week.getUTCMonth();
              d.d = week.getUTCDate() + (d.w + 6) % 7;
            } else {
              week = localDate$1(newDate$1(d.y, 0, 1)), day = week.getDay();
              week = day > 4 || day === 0 ? monday$1.ceil(week) : monday$1(week);
              week = day$1.offset(week, (d.V - 1) * 7);
              d.y = week.getFullYear();
              d.m = week.getMonth();
              d.d = week.getDate() + (d.w + 6) % 7;
            }
          } else if ("W" in d || "U" in d) {
            if (!("w" in d)) d.w = "u" in d ? d.u % 7 : "W" in d ? 1 : 0;
            day = "Z" in d ? utcDate$1(newDate$1(d.y, 0, 1)).getUTCDay() : localDate$1(newDate$1(d.y, 0, 1)).getDay();
            d.m = 0;
            d.d = "W" in d ? (d.w + 6) % 7 + d.W * 7 - (day + 5) % 7 : d.w + d.U * 7 - (day + 6) % 7;
          }

          // If a time zone is specified, all fields are interpreted as UTC and then
          // offset according to the specified time zone.
          if ("Z" in d) {
            d.H += d.Z / 100 | 0;
            d.M += d.Z % 100;
            return utcDate$1(d);
          }

          // Otherwise, all fields are in local time.
          return localDate$1(d);
        };
      }

      function parseSpecifier(d, specifier, string, j) {
        var i = 0,
            n = specifier.length,
            m = string.length,
            c,
            parse;

        while (i < n) {
          if (j >= m) return -1;
          c = specifier.charCodeAt(i++);
          if (c === 37) {
            c = specifier.charAt(i++);
            parse = parses[c in pads$1 ? specifier.charAt(i++) : c];
            if (!parse || ((j = parse(d, string, j)) < 0)) return -1;
          } else if (c != string.charCodeAt(j++)) {
            return -1;
          }
        }

        return j;
      }

      function parsePeriod(d, string, i) {
        var n = periodRe.exec(string.slice(i));
        return n ? (d.p = periodLookup[n[0].toLowerCase()], i + n[0].length) : -1;
      }

      function parseShortWeekday(d, string, i) {
        var n = shortWeekdayRe.exec(string.slice(i));
        return n ? (d.w = shortWeekdayLookup[n[0].toLowerCase()], i + n[0].length) : -1;
      }

      function parseWeekday(d, string, i) {
        var n = weekdayRe.exec(string.slice(i));
        return n ? (d.w = weekdayLookup[n[0].toLowerCase()], i + n[0].length) : -1;
      }

      function parseShortMonth(d, string, i) {
        var n = shortMonthRe.exec(string.slice(i));
        return n ? (d.m = shortMonthLookup[n[0].toLowerCase()], i + n[0].length) : -1;
      }

      function parseMonth(d, string, i) {
        var n = monthRe.exec(string.slice(i));
        return n ? (d.m = monthLookup[n[0].toLowerCase()], i + n[0].length) : -1;
      }

      function parseLocaleDateTime(d, string, i) {
        return parseSpecifier(d, locale_dateTime, string, i);
      }

      function parseLocaleDate(d, string, i) {
        return parseSpecifier(d, locale_date, string, i);
      }

      function parseLocaleTime(d, string, i) {
        return parseSpecifier(d, locale_time, string, i);
      }

      function formatShortWeekday(d) {
        return locale_shortWeekdays[d.getDay()];
      }

      function formatWeekday(d) {
        return locale_weekdays[d.getDay()];
      }

      function formatShortMonth(d) {
        return locale_shortMonths[d.getMonth()];
      }

      function formatMonth(d) {
        return locale_months[d.getMonth()];
      }

      function formatPeriod(d) {
        return locale_periods[+(d.getHours() >= 12)];
      }

      function formatQuarter(d) {
        return 1 + ~~(d.getMonth() / 3);
      }

      function formatUTCShortWeekday(d) {
        return locale_shortWeekdays[d.getUTCDay()];
      }

      function formatUTCWeekday(d) {
        return locale_weekdays[d.getUTCDay()];
      }

      function formatUTCShortMonth(d) {
        return locale_shortMonths[d.getUTCMonth()];
      }

      function formatUTCMonth(d) {
        return locale_months[d.getUTCMonth()];
      }

      function formatUTCPeriod(d) {
        return locale_periods[+(d.getUTCHours() >= 12)];
      }

      function formatUTCQuarter(d) {
        return 1 + ~~(d.getUTCMonth() / 3);
      }

      return {
        format: function(specifier) {
          var f = newFormat(specifier += "", formats);
          f.toString = function() { return specifier; };
          return f;
        },
        parse: function(specifier) {
          var p = newParse(specifier += "", false);
          p.toString = function() { return specifier; };
          return p;
        },
        utcFormat: function(specifier) {
          var f = newFormat(specifier += "", utcFormats);
          f.toString = function() { return specifier; };
          return f;
        },
        utcParse: function(specifier) {
          var p = newParse(specifier += "", true);
          p.toString = function() { return specifier; };
          return p;
        }
      };
    }

    var pads$1 = {"-": "", "_": " ", "0": "0"},
        numberRe$1 = /^\s*\d+/, // note: ignores next directive
        percentRe$1 = /^%/,
        requoteRe$1 = /[\\^$*+?|[\]().{}]/g;

    function pad$2(value, fill, width) {
      var sign = value < 0 ? "-" : "",
          string = (sign ? -value : value) + "",
          length = string.length;
      return sign + (length < width ? new Array(width - length + 1).join(fill) + string : string);
    }

    function requote$1(s) {
      return s.replace(requoteRe$1, "\\$&");
    }

    function formatRe$1(names) {
      return new RegExp("^(?:" + names.map(requote$1).join("|") + ")", "i");
    }

    function formatLookup$1(names) {
      var map = {}, i = -1, n = names.length;
      while (++i < n) map[names[i].toLowerCase()] = i;
      return map;
    }

    function parseWeekdayNumberSunday$1(d, string, i) {
      var n = numberRe$1.exec(string.slice(i, i + 1));
      return n ? (d.w = +n[0], i + n[0].length) : -1;
    }

    function parseWeekdayNumberMonday$1(d, string, i) {
      var n = numberRe$1.exec(string.slice(i, i + 1));
      return n ? (d.u = +n[0], i + n[0].length) : -1;
    }

    function parseWeekNumberSunday$1(d, string, i) {
      var n = numberRe$1.exec(string.slice(i, i + 2));
      return n ? (d.U = +n[0], i + n[0].length) : -1;
    }

    function parseWeekNumberISO$1(d, string, i) {
      var n = numberRe$1.exec(string.slice(i, i + 2));
      return n ? (d.V = +n[0], i + n[0].length) : -1;
    }

    function parseWeekNumberMonday$1(d, string, i) {
      var n = numberRe$1.exec(string.slice(i, i + 2));
      return n ? (d.W = +n[0], i + n[0].length) : -1;
    }

    function parseFullYear$1(d, string, i) {
      var n = numberRe$1.exec(string.slice(i, i + 4));
      return n ? (d.y = +n[0], i + n[0].length) : -1;
    }

    function parseYear$1(d, string, i) {
      var n = numberRe$1.exec(string.slice(i, i + 2));
      return n ? (d.y = +n[0] + (+n[0] > 68 ? 1900 : 2000), i + n[0].length) : -1;
    }

    function parseZone$1(d, string, i) {
      var n = /^(Z)|([+-]\d\d)(?::?(\d\d))?/.exec(string.slice(i, i + 6));
      return n ? (d.Z = n[1] ? 0 : -(n[2] + (n[3] || "00")), i + n[0].length) : -1;
    }

    function parseQuarter$1(d, string, i) {
      var n = numberRe$1.exec(string.slice(i, i + 1));
      return n ? (d.q = n[0] * 3 - 3, i + n[0].length) : -1;
    }

    function parseMonthNumber$1(d, string, i) {
      var n = numberRe$1.exec(string.slice(i, i + 2));
      return n ? (d.m = n[0] - 1, i + n[0].length) : -1;
    }

    function parseDayOfMonth$1(d, string, i) {
      var n = numberRe$1.exec(string.slice(i, i + 2));
      return n ? (d.d = +n[0], i + n[0].length) : -1;
    }

    function parseDayOfYear$1(d, string, i) {
      var n = numberRe$1.exec(string.slice(i, i + 3));
      return n ? (d.m = 0, d.d = +n[0], i + n[0].length) : -1;
    }

    function parseHour24$1(d, string, i) {
      var n = numberRe$1.exec(string.slice(i, i + 2));
      return n ? (d.H = +n[0], i + n[0].length) : -1;
    }

    function parseMinutes$1(d, string, i) {
      var n = numberRe$1.exec(string.slice(i, i + 2));
      return n ? (d.M = +n[0], i + n[0].length) : -1;
    }

    function parseSeconds$1(d, string, i) {
      var n = numberRe$1.exec(string.slice(i, i + 2));
      return n ? (d.S = +n[0], i + n[0].length) : -1;
    }

    function parseMilliseconds$1(d, string, i) {
      var n = numberRe$1.exec(string.slice(i, i + 3));
      return n ? (d.L = +n[0], i + n[0].length) : -1;
    }

    function parseMicroseconds$1(d, string, i) {
      var n = numberRe$1.exec(string.slice(i, i + 6));
      return n ? (d.L = Math.floor(n[0] / 1000), i + n[0].length) : -1;
    }

    function parseLiteralPercent$1(d, string, i) {
      var n = percentRe$1.exec(string.slice(i, i + 1));
      return n ? i + n[0].length : -1;
    }

    function parseUnixTimestamp$1(d, string, i) {
      var n = numberRe$1.exec(string.slice(i));
      return n ? (d.Q = +n[0], i + n[0].length) : -1;
    }

    function parseUnixTimestampSeconds$1(d, string, i) {
      var n = numberRe$1.exec(string.slice(i));
      return n ? (d.s = +n[0], i + n[0].length) : -1;
    }

    function formatDayOfMonth$1(d, p) {
      return pad$2(d.getDate(), p, 2);
    }

    function formatHour24$1(d, p) {
      return pad$2(d.getHours(), p, 2);
    }

    function formatHour12$1(d, p) {
      return pad$2(d.getHours() % 12 || 12, p, 2);
    }

    function formatDayOfYear$1(d, p) {
      return pad$2(1 + day$1.count(year$1(d), d), p, 3);
    }

    function formatMilliseconds$1(d, p) {
      return pad$2(d.getMilliseconds(), p, 3);
    }

    function formatMicroseconds$1(d, p) {
      return formatMilliseconds$1(d, p) + "000";
    }

    function formatMonthNumber$1(d, p) {
      return pad$2(d.getMonth() + 1, p, 2);
    }

    function formatMinutes$1(d, p) {
      return pad$2(d.getMinutes(), p, 2);
    }

    function formatSeconds$1(d, p) {
      return pad$2(d.getSeconds(), p, 2);
    }

    function formatWeekdayNumberMonday$1(d) {
      var day = d.getDay();
      return day === 0 ? 7 : day;
    }

    function formatWeekNumberSunday$1(d, p) {
      return pad$2(sunday$1.count(year$1(d) - 1, d), p, 2);
    }

    function formatWeekNumberISO$1(d, p) {
      var day = d.getDay();
      d = (day >= 4 || day === 0) ? thursday$1(d) : thursday$1.ceil(d);
      return pad$2(thursday$1.count(year$1(d), d) + (year$1(d).getDay() === 4), p, 2);
    }

    function formatWeekdayNumberSunday$1(d) {
      return d.getDay();
    }

    function formatWeekNumberMonday$1(d, p) {
      return pad$2(monday$1.count(year$1(d) - 1, d), p, 2);
    }

    function formatYear$2(d, p) {
      return pad$2(d.getFullYear() % 100, p, 2);
    }

    function formatFullYear$1(d, p) {
      return pad$2(d.getFullYear() % 10000, p, 4);
    }

    function formatZone$1(d) {
      var z = d.getTimezoneOffset();
      return (z > 0 ? "-" : (z *= -1, "+"))
          + pad$2(z / 60 | 0, "0", 2)
          + pad$2(z % 60, "0", 2);
    }

    function formatUTCDayOfMonth$1(d, p) {
      return pad$2(d.getUTCDate(), p, 2);
    }

    function formatUTCHour24$1(d, p) {
      return pad$2(d.getUTCHours(), p, 2);
    }

    function formatUTCHour12$1(d, p) {
      return pad$2(d.getUTCHours() % 12 || 12, p, 2);
    }

    function formatUTCDayOfYear$1(d, p) {
      return pad$2(1 + utcDay$1.count(utcYear$1(d), d), p, 3);
    }

    function formatUTCMilliseconds$1(d, p) {
      return pad$2(d.getUTCMilliseconds(), p, 3);
    }

    function formatUTCMicroseconds$1(d, p) {
      return formatUTCMilliseconds$1(d, p) + "000";
    }

    function formatUTCMonthNumber$1(d, p) {
      return pad$2(d.getUTCMonth() + 1, p, 2);
    }

    function formatUTCMinutes$1(d, p) {
      return pad$2(d.getUTCMinutes(), p, 2);
    }

    function formatUTCSeconds$1(d, p) {
      return pad$2(d.getUTCSeconds(), p, 2);
    }

    function formatUTCWeekdayNumberMonday$1(d) {
      var dow = d.getUTCDay();
      return dow === 0 ? 7 : dow;
    }

    function formatUTCWeekNumberSunday$1(d, p) {
      return pad$2(utcSunday$1.count(utcYear$1(d) - 1, d), p, 2);
    }

    function formatUTCWeekNumberISO$1(d, p) {
      var day = d.getUTCDay();
      d = (day >= 4 || day === 0) ? utcThursday$1(d) : utcThursday$1.ceil(d);
      return pad$2(utcThursday$1.count(utcYear$1(d), d) + (utcYear$1(d).getUTCDay() === 4), p, 2);
    }

    function formatUTCWeekdayNumberSunday$1(d) {
      return d.getUTCDay();
    }

    function formatUTCWeekNumberMonday$1(d, p) {
      return pad$2(utcMonday$1.count(utcYear$1(d) - 1, d), p, 2);
    }

    function formatUTCYear$1(d, p) {
      return pad$2(d.getUTCFullYear() % 100, p, 2);
    }

    function formatUTCFullYear$1(d, p) {
      return pad$2(d.getUTCFullYear() % 10000, p, 4);
    }

    function formatUTCZone$1() {
      return "+0000";
    }

    function formatLiteralPercent$1() {
      return "%";
    }

    function formatUnixTimestamp$1(d) {
      return +d;
    }

    function formatUnixTimestampSeconds$1(d) {
      return Math.floor(+d / 1000);
    }

    var locale$4;
    var timeFormat$1;
    var timeParse$1;
    var utcFormat$1;
    var utcParse$1;

    defaultLocale$4({
      dateTime: "%x, %X",
      date: "%-m/%-d/%Y",
      time: "%-I:%M:%S %p",
      periods: ["AM", "PM"],
      days: ["Sunday", "Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday"],
      shortDays: ["Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"],
      months: ["January", "February", "March", "April", "May", "June", "July", "August", "September", "October", "November", "December"],
      shortMonths: ["Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"]
    });

    function defaultLocale$4(definition) {
      locale$4 = formatLocale$4(definition);
      timeFormat$1 = locale$4.format;
      timeParse$1 = locale$4.parse;
      utcFormat$1 = locale$4.utcFormat;
      utcParse$1 = locale$4.utcParse;
      return locale$4;
    }

    var durationSecond$3 = 1000,
        durationMinute$3 = durationSecond$3 * 60,
        durationHour$3 = durationMinute$3 * 60,
        durationDay$3 = durationHour$3 * 24,
        durationWeek$3 = durationDay$3 * 7,
        durationMonth$1 = durationDay$3 * 30,
        durationYear$1 = durationDay$3 * 365;

    function date$5(t) {
      return new Date(t);
    }

    function number$9(t) {
      return t instanceof Date ? +t : +new Date(+t);
    }

    function calendar$1(year, month, week, day, hour, minute, second, millisecond, format) {
      var scale = continuous$1(deinterpolateLinear, reinterpolate),
          invert = scale.invert,
          domain = scale.domain;

      var formatMillisecond = format(".%L"),
          formatSecond = format(":%S"),
          formatMinute = format("%I:%M"),
          formatHour = format("%I %p"),
          formatDay = format("%a %d"),
          formatWeek = format("%b %d"),
          formatMonth = format("%B"),
          formatYear = format("%Y");

      var tickIntervals = [
        [second,  1,      durationSecond$3],
        [second,  5,  5 * durationSecond$3],
        [second, 15, 15 * durationSecond$3],
        [second, 30, 30 * durationSecond$3],
        [minute,  1,      durationMinute$3],
        [minute,  5,  5 * durationMinute$3],
        [minute, 15, 15 * durationMinute$3],
        [minute, 30, 30 * durationMinute$3],
        [  hour,  1,      durationHour$3  ],
        [  hour,  3,  3 * durationHour$3  ],
        [  hour,  6,  6 * durationHour$3  ],
        [  hour, 12, 12 * durationHour$3  ],
        [   day,  1,      durationDay$3   ],
        [   day,  2,  2 * durationDay$3   ],
        [  week,  1,      durationWeek$3  ],
        [ month,  1,      durationMonth$1 ],
        [ month,  3,  3 * durationMonth$1 ],
        [  year,  1,      durationYear$1  ]
      ];

      function tickFormat(date) {
        return (second(date) < date ? formatMillisecond
            : minute(date) < date ? formatSecond
            : hour(date) < date ? formatMinute
            : day(date) < date ? formatHour
            : month(date) < date ? (week(date) < date ? formatDay : formatWeek)
            : year(date) < date ? formatMonth
            : formatYear)(date);
      }

      function tickInterval(interval, start, stop, step) {
        if (interval == null) interval = 10;

        // If a desired tick count is specified, pick a reasonable tick interval
        // based on the extent of the domain and a rough estimate of tick size.
        // Otherwise, assume interval is already a time interval and use it.
        if (typeof interval === "number") {
          var target = Math.abs(stop - start) / interval,
              i = bisector$4(function(i) { return i[2]; }).right(tickIntervals, target);
          if (i === tickIntervals.length) {
            step = tickStep$2(start / durationYear$1, stop / durationYear$1, interval);
            interval = year;
          } else if (i) {
            i = tickIntervals[target / tickIntervals[i - 1][2] < tickIntervals[i][2] / target ? i - 1 : i];
            step = i[1];
            interval = i[0];
          } else {
            step = Math.max(tickStep$2(start, stop, interval), 1);
            interval = millisecond;
          }
        }

        return step == null ? interval : interval.every(step);
      }

      scale.invert = function(y) {
        return new Date(invert(y));
      };

      scale.domain = function(_) {
        return arguments.length ? domain(map$8.call(_, number$9)) : domain().map(date$5);
      };

      scale.ticks = function(interval, step) {
        var d = domain(),
            t0 = d[0],
            t1 = d[d.length - 1],
            r = t1 < t0,
            t;
        if (r) t = t0, t0 = t1, t1 = t;
        t = tickInterval(interval, t0, t1, step);
        t = t ? t.range(t0, t1 + 1) : []; // inclusive stop
        return r ? t.reverse() : t;
      };

      scale.tickFormat = function(count, specifier) {
        return specifier == null ? tickFormat : format(specifier);
      };

      scale.nice = function(interval, step) {
        var d = domain();
        return (interval = tickInterval(interval, d[0], d[d.length - 1], step))
            ? domain(nice$1(d, interval))
            : scale;
      };

      scale.copy = function() {
        return copy$2(scale, calendar$1(year, month, week, day, hour, minute, second, millisecond, format));
      };

      return scale;
    }

    function time$1() {
      return calendar$1(year$1, month$1, sunday$1, day$1, hour$1, minute$1, second$1, millisecond$1, timeFormat$1).domain([new Date(2000, 0, 1), new Date(2000, 0, 2)]);
    }

    function utcTime$1() {
      return calendar$1(utcYear$1, utcMonth$1, utcSunday$1, utcDay$1, utcHour$1, utcMinute$1, second$1, millisecond$1, utcFormat$1).domain([Date.UTC(2000, 0, 1), Date.UTC(2000, 0, 2)]);
    }

    function colors$1(s) {
      return s.match(/.{6}/g).map(function(x) {
        return "#" + x;
      });
    }

    var category10$1 = colors$1("1f77b4ff7f0e2ca02cd627289467bd8c564be377c27f7f7fbcbd2217becf");

    var category20b = colors$1("393b795254a36b6ecf9c9ede6379398ca252b5cf6bcedb9c8c6d31bd9e39e7ba52e7cb94843c39ad494ad6616be7969c7b4173a55194ce6dbdde9ed6");

    var category20c = colors$1("3182bd6baed69ecae1c6dbefe6550dfd8d3cfdae6bfdd0a231a35474c476a1d99bc7e9c0756bb19e9ac8bcbddcdadaeb636363969696bdbdbdd9d9d9");

    var category20 = colors$1("1f77b4aec7e8ff7f0effbb782ca02c98df8ad62728ff98969467bdc5b0d58c564bc49c94e377c2f7b6d27f7f7fc7c7c7bcbd22dbdb8d17becf9edae5");

    var cubehelix$a = cubehelixLong$2(cubehelix$8(300, 0.5, 0.0), cubehelix$8(-240, 0.5, 1.0));

    var warm$1 = cubehelixLong$2(cubehelix$8(-100, 0.75, 0.35), cubehelix$8(80, 1.50, 0.8));

    var cool$1 = cubehelixLong$2(cubehelix$8(260, 0.75, 0.35), cubehelix$8(80, 1.50, 0.8));

    var rainbow$1 = cubehelix$8();

    function rainbow$2(t) {
      if (t < 0 || t > 1) t -= Math.floor(t);
      var ts = Math.abs(t - 0.5);
      rainbow$1.h = 360 * t - 100;
      rainbow$1.s = 1.5 - 1.5 * ts;
      rainbow$1.l = 0.8 - 0.9 * ts;
      return rainbow$1 + "";
    }

    function ramp$2(range) {
      var n = range.length;
      return function(t) {
        return range[Math.max(0, Math.min(n - 1, Math.floor(t * n)))];
      };
    }

    var viridis$1 = ramp$2(colors$1("44015444025645045745055946075a46085c460a5d460b5e470d60470e6147106347116447136548146748166848176948186a481a6c481b6d481c6e481d6f481f70482071482173482374482475482576482677482878482979472a7a472c7a472d7b472e7c472f7d46307e46327e46337f463480453581453781453882443983443a83443b84433d84433e85423f854240864241864142874144874045884046883f47883f48893e49893e4a893e4c8a3d4d8a3d4e8a3c4f8a3c508b3b518b3b528b3a538b3a548c39558c39568c38588c38598c375a8c375b8d365c8d365d8d355e8d355f8d34608d34618d33628d33638d32648e32658e31668e31678e31688e30698e306a8e2f6b8e2f6c8e2e6d8e2e6e8e2e6f8e2d708e2d718e2c718e2c728e2c738e2b748e2b758e2a768e2a778e2a788e29798e297a8e297b8e287c8e287d8e277e8e277f8e27808e26818e26828e26828e25838e25848e25858e24868e24878e23888e23898e238a8d228b8d228c8d228d8d218e8d218f8d21908d21918c20928c20928c20938c1f948c1f958b1f968b1f978b1f988b1f998a1f9a8a1e9b8a1e9c891e9d891f9e891f9f881fa0881fa1881fa1871fa28720a38620a48621a58521a68522a78522a88423a98324aa8325ab8225ac8226ad8127ad8128ae8029af7f2ab07f2cb17e2db27d2eb37c2fb47c31b57b32b67a34b67935b77937b87838b9773aba763bbb753dbc743fbc7340bd7242be7144bf7046c06f48c16e4ac16d4cc26c4ec36b50c46a52c56954c56856c66758c7655ac8645cc8635ec96260ca6063cb5f65cb5e67cc5c69cd5b6ccd5a6ece5870cf5773d05675d05477d1537ad1517cd2507fd34e81d34d84d44b86d54989d5488bd6468ed64590d74393d74195d84098d83e9bd93c9dd93ba0da39a2da37a5db36a8db34aadc32addc30b0dd2fb2dd2db5de2bb8de29bade28bddf26c0df25c2df23c5e021c8e020cae11fcde11dd0e11cd2e21bd5e21ad8e219dae319dde318dfe318e2e418e5e419e7e419eae51aece51befe51cf1e51df4e61ef6e620f8e621fbe723fde725"));

    var magma$1 = ramp$2(colors$1("00000401000501010601010802010902020b02020d03030f03031204041405041606051806051a07061c08071e0907200a08220b09240c09260d0a290e0b2b100b2d110c2f120d31130d34140e36150e38160f3b180f3d19103f1a10421c10441d11471e114920114b21114e22115024125325125527125829115a2a115c2c115f2d11612f116331116533106734106936106b38106c390f6e3b0f703d0f713f0f72400f74420f75440f764510774710784910784a10794c117a4e117b4f127b51127c52137c54137d56147d57157e59157e5a167e5c167f5d177f5f187f601880621980641a80651a80671b80681c816a1c816b1d816d1d816e1e81701f81721f817320817521817621817822817922827b23827c23827e24828025828125818326818426818627818827818928818b29818c29818e2a81902a81912b81932b80942c80962c80982d80992d809b2e7f9c2e7f9e2f7fa02f7fa1307ea3307ea5317ea6317da8327daa337dab337cad347cae347bb0357bb2357bb3367ab5367ab73779b83779ba3878bc3978bd3977bf3a77c03a76c23b75c43c75c53c74c73d73c83e73ca3e72cc3f71cd4071cf4070d0416fd2426fd3436ed5446dd6456cd8456cd9466bdb476adc4869de4968df4a68e04c67e24d66e34e65e44f64e55064e75263e85362e95462ea5661eb5760ec5860ed5a5fee5b5eef5d5ef05f5ef1605df2625df2645cf3655cf4675cf4695cf56b5cf66c5cf66e5cf7705cf7725cf8745cf8765cf9785df9795df97b5dfa7d5efa7f5efa815ffb835ffb8560fb8761fc8961fc8a62fc8c63fc8e64fc9065fd9266fd9467fd9668fd9869fd9a6afd9b6bfe9d6cfe9f6dfea16efea36ffea571fea772fea973feaa74feac76feae77feb078feb27afeb47bfeb67cfeb77efeb97ffebb81febd82febf84fec185fec287fec488fec68afec88cfeca8dfecc8ffecd90fecf92fed194fed395fed597fed799fed89afdda9cfddc9efddea0fde0a1fde2a3fde3a5fde5a7fde7a9fde9aafdebacfcecaefceeb0fcf0b2fcf2b4fcf4b6fcf6b8fcf7b9fcf9bbfcfbbdfcfdbf"));

    var inferno$1 = ramp$2(colors$1("00000401000501010601010802010a02020c02020e03021004031204031405041706041907051b08051d09061f0a07220b07240c08260d08290e092b10092d110a30120a32140b34150b37160b39180c3c190c3e1b0c411c0c431e0c451f0c48210c4a230c4c240c4f260c51280b53290b552b0b572d0b592f0a5b310a5c320a5e340a5f3609613809623909633b09643d09653e0966400a67420a68440a68450a69470b6a490b6a4a0c6b4c0c6b4d0d6c4f0d6c510e6c520e6d540f6d550f6d57106e59106e5a116e5c126e5d126e5f136e61136e62146e64156e65156e67166e69166e6a176e6c186e6d186e6f196e71196e721a6e741a6e751b6e771c6d781c6d7a1d6d7c1d6d7d1e6d7f1e6c801f6c82206c84206b85216b87216b88226a8a226a8c23698d23698f24699025689225689326679526679727669827669a28659b29649d29649f2a63a02a63a22b62a32c61a52c60a62d60a82e5fa92e5eab2f5ead305dae305cb0315bb1325ab3325ab43359b63458b73557b93556ba3655bc3754bd3853bf3952c03a51c13a50c33b4fc43c4ec63d4dc73e4cc83f4bca404acb4149cc4248ce4347cf4446d04545d24644d34743d44842d54a41d74b3fd84c3ed94d3dda4e3cdb503bdd513ade5238df5337e05536e15635e25734e35933e45a31e55c30e65d2fe75e2ee8602de9612bea632aeb6429eb6628ec6726ed6925ee6a24ef6c23ef6e21f06f20f1711ff1731df2741cf3761bf37819f47918f57b17f57d15f67e14f68013f78212f78410f8850ff8870ef8890cf98b0bf98c0af98e09fa9008fa9207fa9407fb9606fb9706fb9906fb9b06fb9d07fc9f07fca108fca309fca50afca60cfca80dfcaa0ffcac11fcae12fcb014fcb216fcb418fbb61afbb81dfbba1ffbbc21fbbe23fac026fac228fac42afac62df9c72ff9c932f9cb35f8cd37f8cf3af7d13df7d340f6d543f6d746f5d949f5db4cf4dd4ff4df53f4e156f3e35af3e55df2e661f2e865f2ea69f1ec6df1ed71f1ef75f1f179f2f27df2f482f3f586f3f68af4f88ef5f992f6fa96f8fb9af9fc9dfafda1fcffa4"));

    var plasma$1 = ramp$2(colors$1("0d088710078813078916078a19068c1b068d1d068e20068f2206902406912605912805922a05932c05942e05952f059631059733059735049837049938049a3a049a3c049b3e049c3f049c41049d43039e44039e46039f48039f4903a04b03a14c02a14e02a25002a25102a35302a35502a45601a45801a45901a55b01a55c01a65e01a66001a66100a76300a76400a76600a76700a86900a86a00a86c00a86e00a86f00a87100a87201a87401a87501a87701a87801a87a02a87b02a87d03a87e03a88004a88104a78305a78405a78606a68707a68808a68a09a58b0aa58d0ba58e0ca48f0da4910ea3920fa39410a29511a19613a19814a099159f9a169f9c179e9d189d9e199da01a9ca11b9ba21d9aa31e9aa51f99a62098a72197a82296aa2395ab2494ac2694ad2793ae2892b02991b12a90b22b8fb32c8eb42e8db52f8cb6308bb7318ab83289ba3388bb3488bc3587bd3786be3885bf3984c03a83c13b82c23c81c33d80c43e7fc5407ec6417dc7427cc8437bc9447aca457acb4679cc4778cc4977cd4a76ce4b75cf4c74d04d73d14e72d24f71d35171d45270d5536fd5546ed6556dd7566cd8576bd9586ada5a6ada5b69db5c68dc5d67dd5e66de5f65de6164df6263e06363e16462e26561e26660e3685fe4695ee56a5de56b5de66c5ce76e5be76f5ae87059e97158e97257ea7457eb7556eb7655ec7754ed7953ed7a52ee7b51ef7c51ef7e50f07f4ff0804ef1814df1834cf2844bf3854bf3874af48849f48948f58b47f58c46f68d45f68f44f79044f79143f79342f89441f89540f9973ff9983ef99a3efa9b3dfa9c3cfa9e3bfb9f3afba139fba238fca338fca537fca636fca835fca934fdab33fdac33fdae32fdaf31fdb130fdb22ffdb42ffdb52efeb72dfeb82cfeba2cfebb2bfebd2afebe2afec029fdc229fdc328fdc527fdc627fdc827fdca26fdcb26fccd25fcce25fcd025fcd225fbd324fbd524fbd724fad824fada24f9dc24f9dd25f8df25f8e125f7e225f7e425f6e626f6e826f5e926f5eb27f4ed27f3ee27f3f027f2f227f1f426f1f525f0f724f0f921"));

    function sequential$1(interpolator) {
      var x0 = 0,
          x1 = 1,
          clamp = false;

      function scale(x) {
        var t = (x - x0) / (x1 - x0);
        return interpolator(clamp ? Math.max(0, Math.min(1, t)) : t);
      }

      scale.domain = function(_) {
        return arguments.length ? (x0 = +_[0], x1 = +_[1], scale) : [x0, x1];
      };

      scale.clamp = function(_) {
        return arguments.length ? (clamp = !!_, scale) : clamp;
      };

      scale.interpolator = function(_) {
        return arguments.length ? (interpolator = _, scale) : interpolator;
      };

      scale.copy = function() {
        return sequential$1(interpolator).domain([x0, x1]).clamp(clamp);
      };

      return linearish$1(scale);
    }

    var xhtml$3 = "http://www.w3.org/1999/xhtml";

    var namespaces$3 = {
      svg: "http://www.w3.org/2000/svg",
      xhtml: xhtml$3,
      xlink: "http://www.w3.org/1999/xlink",
      xml: "http://www.w3.org/XML/1998/namespace",
      xmlns: "http://www.w3.org/2000/xmlns/"
    };

    function namespace$3(name) {
      var prefix = name += "", i = prefix.indexOf(":");
      if (i >= 0 && (prefix = name.slice(0, i)) !== "xmlns") name = name.slice(i + 1);
      return namespaces$3.hasOwnProperty(prefix) ? {space: namespaces$3[prefix], local: name} : name;
    }

    function creatorInherit$3(name) {
      return function() {
        var document = this.ownerDocument,
            uri = this.namespaceURI;
        return uri === xhtml$3 && document.documentElement.namespaceURI === xhtml$3
            ? document.createElement(name)
            : document.createElementNS(uri, name);
      };
    }

    function creatorFixed$3(fullname) {
      return function() {
        return this.ownerDocument.createElementNS(fullname.space, fullname.local);
      };
    }

    function creator$3(name) {
      var fullname = namespace$3(name);
      return (fullname.local
          ? creatorFixed$3
          : creatorInherit$3)(fullname);
    }

    function none$5() {}

    function selector$3(selector) {
      return selector == null ? none$5 : function() {
        return this.querySelector(selector);
      };
    }

    function selection_select$3(select) {
      if (typeof select !== "function") select = selector$3(select);

      for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, subgroup = subgroups[j] = new Array(n), node, subnode, i = 0; i < n; ++i) {
          if ((node = group[i]) && (subnode = select.call(node, node.__data__, i, group))) {
            if ("__data__" in node) subnode.__data__ = node.__data__;
            subgroup[i] = subnode;
          }
        }
      }

      return new Selection$5(subgroups, this._parents);
    }

    function empty$5() {
      return [];
    }

    function selectorAll$3(selector) {
      return selector == null ? empty$5 : function() {
        return this.querySelectorAll(selector);
      };
    }

    function selection_selectAll$3(select) {
      if (typeof select !== "function") select = selectorAll$3(select);

      for (var groups = this._groups, m = groups.length, subgroups = [], parents = [], j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, node, i = 0; i < n; ++i) {
          if (node = group[i]) {
            subgroups.push(select.call(node, node.__data__, i, group));
            parents.push(node);
          }
        }
      }

      return new Selection$5(subgroups, parents);
    }

    var matcher$3 = function(selector) {
      return function() {
        return this.matches(selector);
      };
    };

    if (typeof document !== "undefined") {
      var element$3 = document.documentElement;
      if (!element$3.matches) {
        var vendorMatches = element$3.webkitMatchesSelector
            || element$3.msMatchesSelector
            || element$3.mozMatchesSelector
            || element$3.oMatchesSelector;
        matcher$3 = function(selector) {
          return function() {
            return vendorMatches.call(this, selector);
          };
        };
      }
    }

    var matcher$4 = matcher$3;

    function selection_filter$3(match) {
      if (typeof match !== "function") match = matcher$4(match);

      for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, subgroup = subgroups[j] = [], node, i = 0; i < n; ++i) {
          if ((node = group[i]) && match.call(node, node.__data__, i, group)) {
            subgroup.push(node);
          }
        }
      }

      return new Selection$5(subgroups, this._parents);
    }

    function sparse$3(update) {
      return new Array(update.length);
    }

    function selection_enter$3() {
      return new Selection$5(this._enter || this._groups.map(sparse$3), this._parents);
    }

    function EnterNode$3(parent, datum) {
      this.ownerDocument = parent.ownerDocument;
      this.namespaceURI = parent.namespaceURI;
      this._next = null;
      this._parent = parent;
      this.__data__ = datum;
    }

    EnterNode$3.prototype = {
      constructor: EnterNode$3,
      appendChild: function(child) { return this._parent.insertBefore(child, this._next); },
      insertBefore: function(child, next) { return this._parent.insertBefore(child, next); },
      querySelector: function(selector) { return this._parent.querySelector(selector); },
      querySelectorAll: function(selector) { return this._parent.querySelectorAll(selector); }
    };

    function constant$r(x) {
      return function() {
        return x;
      };
    }

    var keyPrefix$5 = "$"; // Protect against keys like “__proto__”.

    function bindIndex$3(parent, group, enter, update, exit, data) {
      var i = 0,
          node,
          groupLength = group.length,
          dataLength = data.length;

      // Put any non-null nodes that fit into update.
      // Put any null nodes into enter.
      // Put any remaining data into enter.
      for (; i < dataLength; ++i) {
        if (node = group[i]) {
          node.__data__ = data[i];
          update[i] = node;
        } else {
          enter[i] = new EnterNode$3(parent, data[i]);
        }
      }

      // Put any non-null nodes that don’t fit into exit.
      for (; i < groupLength; ++i) {
        if (node = group[i]) {
          exit[i] = node;
        }
      }
    }

    function bindKey$3(parent, group, enter, update, exit, data, key) {
      var i,
          node,
          nodeByKeyValue = {},
          groupLength = group.length,
          dataLength = data.length,
          keyValues = new Array(groupLength),
          keyValue;

      // Compute the key for each node.
      // If multiple nodes have the same key, the duplicates are added to exit.
      for (i = 0; i < groupLength; ++i) {
        if (node = group[i]) {
          keyValues[i] = keyValue = keyPrefix$5 + key.call(node, node.__data__, i, group);
          if (keyValue in nodeByKeyValue) {
            exit[i] = node;
          } else {
            nodeByKeyValue[keyValue] = node;
          }
        }
      }

      // Compute the key for each datum.
      // If there a node associated with this key, join and add it to update.
      // If there is not (or the key is a duplicate), add it to enter.
      for (i = 0; i < dataLength; ++i) {
        keyValue = keyPrefix$5 + key.call(parent, data[i], i, data);
        if (node = nodeByKeyValue[keyValue]) {
          update[i] = node;
          node.__data__ = data[i];
          nodeByKeyValue[keyValue] = null;
        } else {
          enter[i] = new EnterNode$3(parent, data[i]);
        }
      }

      // Add any remaining nodes that were not bound to data to exit.
      for (i = 0; i < groupLength; ++i) {
        if ((node = group[i]) && (nodeByKeyValue[keyValues[i]] === node)) {
          exit[i] = node;
        }
      }
    }

    function selection_data$3(value, key) {
      if (!value) {
        data = new Array(this.size()), j = -1;
        this.each(function(d) { data[++j] = d; });
        return data;
      }

      var bind = key ? bindKey$3 : bindIndex$3,
          parents = this._parents,
          groups = this._groups;

      if (typeof value !== "function") value = constant$r(value);

      for (var m = groups.length, update = new Array(m), enter = new Array(m), exit = new Array(m), j = 0; j < m; ++j) {
        var parent = parents[j],
            group = groups[j],
            groupLength = group.length,
            data = value.call(parent, parent && parent.__data__, j, parents),
            dataLength = data.length,
            enterGroup = enter[j] = new Array(dataLength),
            updateGroup = update[j] = new Array(dataLength),
            exitGroup = exit[j] = new Array(groupLength);

        bind(parent, group, enterGroup, updateGroup, exitGroup, data, key);

        // Now connect the enter nodes to their following update node, such that
        // appendChild can insert the materialized enter node before this node,
        // rather than at the end of the parent node.
        for (var i0 = 0, i1 = 0, previous, next; i0 < dataLength; ++i0) {
          if (previous = enterGroup[i0]) {
            if (i0 >= i1) i1 = i0 + 1;
            while (!(next = updateGroup[i1]) && ++i1 < dataLength);
            previous._next = next || null;
          }
        }
      }

      update = new Selection$5(update, parents);
      update._enter = enter;
      update._exit = exit;
      return update;
    }

    function selection_exit$3() {
      return new Selection$5(this._exit || this._groups.map(sparse$3), this._parents);
    }

    function selection_merge$3(selection) {

      for (var groups0 = this._groups, groups1 = selection._groups, m0 = groups0.length, m1 = groups1.length, m = Math.min(m0, m1), merges = new Array(m0), j = 0; j < m; ++j) {
        for (var group0 = groups0[j], group1 = groups1[j], n = group0.length, merge = merges[j] = new Array(n), node, i = 0; i < n; ++i) {
          if (node = group0[i] || group1[i]) {
            merge[i] = node;
          }
        }
      }

      for (; j < m0; ++j) {
        merges[j] = groups0[j];
      }

      return new Selection$5(merges, this._parents);
    }

    function selection_order$3() {

      for (var groups = this._groups, j = -1, m = groups.length; ++j < m;) {
        for (var group = groups[j], i = group.length - 1, next = group[i], node; --i >= 0;) {
          if (node = group[i]) {
            if (next && next !== node.nextSibling) next.parentNode.insertBefore(node, next);
            next = node;
          }
        }
      }

      return this;
    }

    function selection_sort$3(compare) {
      if (!compare) compare = ascending$a;

      function compareNode(a, b) {
        return a && b ? compare(a.__data__, b.__data__) : !a - !b;
      }

      for (var groups = this._groups, m = groups.length, sortgroups = new Array(m), j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, sortgroup = sortgroups[j] = new Array(n), node, i = 0; i < n; ++i) {
          if (node = group[i]) {
            sortgroup[i] = node;
          }
        }
        sortgroup.sort(compareNode);
      }

      return new Selection$5(sortgroups, this._parents).order();
    }

    function ascending$a(a, b) {
      return a < b ? -1 : a > b ? 1 : a >= b ? 0 : NaN;
    }

    function selection_call$3() {
      var callback = arguments[0];
      arguments[0] = this;
      callback.apply(null, arguments);
      return this;
    }

    function selection_nodes$3() {
      var nodes = new Array(this.size()), i = -1;
      this.each(function() { nodes[++i] = this; });
      return nodes;
    }

    function selection_node$3() {

      for (var groups = this._groups, j = 0, m = groups.length; j < m; ++j) {
        for (var group = groups[j], i = 0, n = group.length; i < n; ++i) {
          var node = group[i];
          if (node) return node;
        }
      }

      return null;
    }

    function selection_size$3() {
      var size = 0;
      this.each(function() { ++size; });
      return size;
    }

    function selection_empty$3() {
      return !this.node();
    }

    function selection_each$3(callback) {

      for (var groups = this._groups, j = 0, m = groups.length; j < m; ++j) {
        for (var group = groups[j], i = 0, n = group.length, node; i < n; ++i) {
          if (node = group[i]) callback.call(node, node.__data__, i, group);
        }
      }

      return this;
    }

    function attrRemove$5(name) {
      return function() {
        this.removeAttribute(name);
      };
    }

    function attrRemoveNS$5(fullname) {
      return function() {
        this.removeAttributeNS(fullname.space, fullname.local);
      };
    }

    function attrConstant$5(name, value) {
      return function() {
        this.setAttribute(name, value);
      };
    }

    function attrConstantNS$5(fullname, value) {
      return function() {
        this.setAttributeNS(fullname.space, fullname.local, value);
      };
    }

    function attrFunction$5(name, value) {
      return function() {
        var v = value.apply(this, arguments);
        if (v == null) this.removeAttribute(name);
        else this.setAttribute(name, v);
      };
    }

    function attrFunctionNS$5(fullname, value) {
      return function() {
        var v = value.apply(this, arguments);
        if (v == null) this.removeAttributeNS(fullname.space, fullname.local);
        else this.setAttributeNS(fullname.space, fullname.local, v);
      };
    }

    function selection_attr$3(name, value) {
      var fullname = namespace$3(name);

      if (arguments.length < 2) {
        var node = this.node();
        return fullname.local
            ? node.getAttributeNS(fullname.space, fullname.local)
            : node.getAttribute(fullname);
      }

      return this.each((value == null
          ? (fullname.local ? attrRemoveNS$5 : attrRemove$5) : (typeof value === "function"
          ? (fullname.local ? attrFunctionNS$5 : attrFunction$5)
          : (fullname.local ? attrConstantNS$5 : attrConstant$5)))(fullname, value));
    }

    function defaultView$3(node) {
      return (node.ownerDocument && node.ownerDocument.defaultView) // node is a Node
          || (node.document && node) // node is a Window
          || node.defaultView; // node is a Document
    }

    function styleRemove$5(name) {
      return function() {
        this.style.removeProperty(name);
      };
    }

    function styleConstant$5(name, value, priority) {
      return function() {
        this.style.setProperty(name, value, priority);
      };
    }

    function styleFunction$5(name, value, priority) {
      return function() {
        var v = value.apply(this, arguments);
        if (v == null) this.style.removeProperty(name);
        else this.style.setProperty(name, v, priority);
      };
    }

    function selection_style$3(name, value, priority) {
      return arguments.length > 1
          ? this.each((value == null
                ? styleRemove$5 : typeof value === "function"
                ? styleFunction$5
                : styleConstant$5)(name, value, priority == null ? "" : priority))
          : styleValue$3(this.node(), name);
    }

    function styleValue$3(node, name) {
      return node.style.getPropertyValue(name)
          || defaultView$3(node).getComputedStyle(node, null).getPropertyValue(name);
    }

    function propertyRemove$3(name) {
      return function() {
        delete this[name];
      };
    }

    function propertyConstant$3(name, value) {
      return function() {
        this[name] = value;
      };
    }

    function propertyFunction$3(name, value) {
      return function() {
        var v = value.apply(this, arguments);
        if (v == null) delete this[name];
        else this[name] = v;
      };
    }

    function selection_property$3(name, value) {
      return arguments.length > 1
          ? this.each((value == null
              ? propertyRemove$3 : typeof value === "function"
              ? propertyFunction$3
              : propertyConstant$3)(name, value))
          : this.node()[name];
    }

    function classArray$3(string) {
      return string.trim().split(/^|\s+/);
    }

    function classList$3(node) {
      return node.classList || new ClassList$3(node);
    }

    function ClassList$3(node) {
      this._node = node;
      this._names = classArray$3(node.getAttribute("class") || "");
    }

    ClassList$3.prototype = {
      add: function(name) {
        var i = this._names.indexOf(name);
        if (i < 0) {
          this._names.push(name);
          this._node.setAttribute("class", this._names.join(" "));
        }
      },
      remove: function(name) {
        var i = this._names.indexOf(name);
        if (i >= 0) {
          this._names.splice(i, 1);
          this._node.setAttribute("class", this._names.join(" "));
        }
      },
      contains: function(name) {
        return this._names.indexOf(name) >= 0;
      }
    };

    function classedAdd$3(node, names) {
      var list = classList$3(node), i = -1, n = names.length;
      while (++i < n) list.add(names[i]);
    }

    function classedRemove$3(node, names) {
      var list = classList$3(node), i = -1, n = names.length;
      while (++i < n) list.remove(names[i]);
    }

    function classedTrue$3(names) {
      return function() {
        classedAdd$3(this, names);
      };
    }

    function classedFalse$3(names) {
      return function() {
        classedRemove$3(this, names);
      };
    }

    function classedFunction$3(names, value) {
      return function() {
        (value.apply(this, arguments) ? classedAdd$3 : classedRemove$3)(this, names);
      };
    }

    function selection_classed$3(name, value) {
      var names = classArray$3(name + "");

      if (arguments.length < 2) {
        var list = classList$3(this.node()), i = -1, n = names.length;
        while (++i < n) if (!list.contains(names[i])) return false;
        return true;
      }

      return this.each((typeof value === "function"
          ? classedFunction$3 : value
          ? classedTrue$3
          : classedFalse$3)(names, value));
    }

    function textRemove$3() {
      this.textContent = "";
    }

    function textConstant$5(value) {
      return function() {
        this.textContent = value;
      };
    }

    function textFunction$5(value) {
      return function() {
        var v = value.apply(this, arguments);
        this.textContent = v == null ? "" : v;
      };
    }

    function selection_text$3(value) {
      return arguments.length
          ? this.each(value == null
              ? textRemove$3 : (typeof value === "function"
              ? textFunction$5
              : textConstant$5)(value))
          : this.node().textContent;
    }

    function htmlRemove$3() {
      this.innerHTML = "";
    }

    function htmlConstant$3(value) {
      return function() {
        this.innerHTML = value;
      };
    }

    function htmlFunction$3(value) {
      return function() {
        var v = value.apply(this, arguments);
        this.innerHTML = v == null ? "" : v;
      };
    }

    function selection_html$3(value) {
      return arguments.length
          ? this.each(value == null
              ? htmlRemove$3 : (typeof value === "function"
              ? htmlFunction$3
              : htmlConstant$3)(value))
          : this.node().innerHTML;
    }

    function raise$4() {
      if (this.nextSibling) this.parentNode.appendChild(this);
    }

    function selection_raise$3() {
      return this.each(raise$4);
    }

    function lower$3() {
      if (this.previousSibling) this.parentNode.insertBefore(this, this.parentNode.firstChild);
    }

    function selection_lower$3() {
      return this.each(lower$3);
    }

    function selection_append$3(name) {
      var create = typeof name === "function" ? name : creator$3(name);
      return this.select(function() {
        return this.appendChild(create.apply(this, arguments));
      });
    }

    function constantNull$3() {
      return null;
    }

    function selection_insert$3(name, before) {
      var create = typeof name === "function" ? name : creator$3(name),
          select = before == null ? constantNull$3 : typeof before === "function" ? before : selector$3(before);
      return this.select(function() {
        return this.insertBefore(create.apply(this, arguments), select.apply(this, arguments) || null);
      });
    }

    function remove$3() {
      var parent = this.parentNode;
      if (parent) parent.removeChild(this);
    }

    function selection_remove$3() {
      return this.each(remove$3);
    }

    function selection_cloneShallow$3() {
      return this.parentNode.insertBefore(this.cloneNode(false), this.nextSibling);
    }

    function selection_cloneDeep$3() {
      return this.parentNode.insertBefore(this.cloneNode(true), this.nextSibling);
    }

    function selection_clone$3(deep) {
      return this.select(deep ? selection_cloneDeep$3 : selection_cloneShallow$3);
    }

    function selection_datum$3(value) {
      return arguments.length
          ? this.property("__data__", value)
          : this.node().__data__;
    }

    var filterEvents$3 = {};

    var event$3 = null;

    if (typeof document !== "undefined") {
      var element$4 = document.documentElement;
      if (!("onmouseenter" in element$4)) {
        filterEvents$3 = {mouseenter: "mouseover", mouseleave: "mouseout"};
      }
    }

    function filterContextListener$3(listener, index, group) {
      listener = contextListener$3(listener, index, group);
      return function(event) {
        var related = event.relatedTarget;
        if (!related || (related !== this && !(related.compareDocumentPosition(this) & 8))) {
          listener.call(this, event);
        }
      };
    }

    function contextListener$3(listener, index, group) {
      return function(event1) {
        var event0 = event$3; // Events can be reentrant (e.g., focus).
        event$3 = event1;
        try {
          listener.call(this, this.__data__, index, group);
        } finally {
          event$3 = event0;
        }
      };
    }

    function parseTypenames$8(typenames) {
      return typenames.trim().split(/^|\s+/).map(function(t) {
        var name = "", i = t.indexOf(".");
        if (i >= 0) name = t.slice(i + 1), t = t.slice(0, i);
        return {type: t, name: name};
      });
    }

    function onRemove$3(typename) {
      return function() {
        var on = this.__on;
        if (!on) return;
        for (var j = 0, i = -1, m = on.length, o; j < m; ++j) {
          if (o = on[j], (!typename.type || o.type === typename.type) && o.name === typename.name) {
            this.removeEventListener(o.type, o.listener, o.capture);
          } else {
            on[++i] = o;
          }
        }
        if (++i) on.length = i;
        else delete this.__on;
      };
    }

    function onAdd$3(typename, value, capture) {
      var wrap = filterEvents$3.hasOwnProperty(typename.type) ? filterContextListener$3 : contextListener$3;
      return function(d, i, group) {
        var on = this.__on, o, listener = wrap(value, i, group);
        if (on) for (var j = 0, m = on.length; j < m; ++j) {
          if ((o = on[j]).type === typename.type && o.name === typename.name) {
            this.removeEventListener(o.type, o.listener, o.capture);
            this.addEventListener(o.type, o.listener = listener, o.capture = capture);
            o.value = value;
            return;
          }
        }
        this.addEventListener(typename.type, listener, capture);
        o = {type: typename.type, name: typename.name, value: value, listener: listener, capture: capture};
        if (!on) this.__on = [o];
        else on.push(o);
      };
    }

    function selection_on$3(typename, value, capture) {
      var typenames = parseTypenames$8(typename + ""), i, n = typenames.length, t;

      if (arguments.length < 2) {
        var on = this.node().__on;
        if (on) for (var j = 0, m = on.length, o; j < m; ++j) {
          for (i = 0, o = on[j]; i < n; ++i) {
            if ((t = typenames[i]).type === o.type && t.name === o.name) {
              return o.value;
            }
          }
        }
        return;
      }

      on = value ? onAdd$3 : onRemove$3;
      if (capture == null) capture = false;
      for (i = 0; i < n; ++i) this.each(on(typenames[i], value, capture));
      return this;
    }

    function customEvent$3(event1, listener, that, args) {
      var event0 = event$3;
      event1.sourceEvent = event$3;
      event$3 = event1;
      try {
        return listener.apply(that, args);
      } finally {
        event$3 = event0;
      }
    }

    function dispatchEvent$3(node, type, params) {
      var window = defaultView$3(node),
          event = window.CustomEvent;

      if (typeof event === "function") {
        event = new event(type, params);
      } else {
        event = window.document.createEvent("Event");
        if (params) event.initEvent(type, params.bubbles, params.cancelable), event.detail = params.detail;
        else event.initEvent(type, false, false);
      }

      node.dispatchEvent(event);
    }

    function dispatchConstant$3(type, params) {
      return function() {
        return dispatchEvent$3(this, type, params);
      };
    }

    function dispatchFunction$3(type, params) {
      return function() {
        return dispatchEvent$3(this, type, params.apply(this, arguments));
      };
    }

    function selection_dispatch$3(type, params) {
      return this.each((typeof params === "function"
          ? dispatchFunction$3
          : dispatchConstant$3)(type, params));
    }

    var root$5 = [null];

    function Selection$5(groups, parents) {
      this._groups = groups;
      this._parents = parents;
    }

    function selection$3() {
      return new Selection$5([[document.documentElement]], root$5);
    }

    Selection$5.prototype = selection$3.prototype = {
      constructor: Selection$5,
      select: selection_select$3,
      selectAll: selection_selectAll$3,
      filter: selection_filter$3,
      data: selection_data$3,
      enter: selection_enter$3,
      exit: selection_exit$3,
      merge: selection_merge$3,
      order: selection_order$3,
      sort: selection_sort$3,
      call: selection_call$3,
      nodes: selection_nodes$3,
      node: selection_node$3,
      size: selection_size$3,
      empty: selection_empty$3,
      each: selection_each$3,
      attr: selection_attr$3,
      style: selection_style$3,
      property: selection_property$3,
      classed: selection_classed$3,
      text: selection_text$3,
      html: selection_html$3,
      raise: selection_raise$3,
      lower: selection_lower$3,
      append: selection_append$3,
      insert: selection_insert$3,
      remove: selection_remove$3,
      clone: selection_clone$3,
      datum: selection_datum$3,
      on: selection_on$3,
      dispatch: selection_dispatch$3
    };

    function select$3(selector) {
      return typeof selector === "string"
          ? new Selection$5([[document.querySelector(selector)]], [document.documentElement])
          : new Selection$5([[selector]], root$5);
    }

    function create$3(name) {
      return select$3(creator$3(name).call(document.documentElement));
    }

    var nextId$1 = 0;

    function local$3() {
      return new Local$1;
    }

    function Local$1() {
      this._ = "@" + (++nextId$1).toString(36);
    }

    Local$1.prototype = local$3.prototype = {
      constructor: Local$1,
      get: function(node) {
        var id = this._;
        while (!(id in node)) if (!(node = node.parentNode)) return;
        return node[id];
      },
      set: function(node, value) {
        return node[this._] = value;
      },
      remove: function(node) {
        return this._ in node && delete node[this._];
      },
      toString: function() {
        return this._;
      }
    };

    function sourceEvent$3() {
      var current = event$3, source;
      while (source = current.sourceEvent) current = source;
      return current;
    }

    function point$9(node, event) {
      var svg = node.ownerSVGElement || node;

      if (svg.createSVGPoint) {
        var point = svg.createSVGPoint();
        point.x = event.clientX, point.y = event.clientY;
        point = point.matrixTransform(node.getScreenCTM().inverse());
        return [point.x, point.y];
      }

      var rect = node.getBoundingClientRect();
      return [event.clientX - rect.left - node.clientLeft, event.clientY - rect.top - node.clientTop];
    }

    function mouse$3(node) {
      var event = sourceEvent$3();
      if (event.changedTouches) event = event.changedTouches[0];
      return point$9(node, event);
    }

    function selectAll$1(selector) {
      return typeof selector === "string"
          ? new Selection$5([document.querySelectorAll(selector)], [document.documentElement])
          : new Selection$5([selector == null ? [] : selector], root$5);
    }

    function touch$2(node, touches, identifier) {
      if (arguments.length < 3) identifier = touches, touches = sourceEvent$3().changedTouches;

      for (var i = 0, n = touches ? touches.length : 0, touch; i < n; ++i) {
        if ((touch = touches[i]).identifier === identifier) {
          return point$9(node, touch);
        }
      }

      return null;
    }

    function touches$1(node, touches) {
      if (touches == null) touches = sourceEvent$3().touches;

      for (var i = 0, n = touches ? touches.length : 0, points = new Array(n); i < n; ++i) {
        points[i] = point$9(node, touches[i]);
      }

      return points;
    }

    var pi$a = Math.PI,
        tau$a = 2 * pi$a,
        epsilon$9 = 1e-6,
        tauEpsilon$3 = tau$a - epsilon$9;

    function Path$3() {
      this._x0 = this._y0 = // start of current subpath
      this._x1 = this._y1 = null; // end of current subpath
      this._ = "";
    }

    function path$3() {
      return new Path$3;
    }

    Path$3.prototype = path$3.prototype = {
      constructor: Path$3,
      moveTo: function(x, y) {
        this._ += "M" + (this._x0 = this._x1 = +x) + "," + (this._y0 = this._y1 = +y);
      },
      closePath: function() {
        if (this._x1 !== null) {
          this._x1 = this._x0, this._y1 = this._y0;
          this._ += "Z";
        }
      },
      lineTo: function(x, y) {
        this._ += "L" + (this._x1 = +x) + "," + (this._y1 = +y);
      },
      quadraticCurveTo: function(x1, y1, x, y) {
        this._ += "Q" + (+x1) + "," + (+y1) + "," + (this._x1 = +x) + "," + (this._y1 = +y);
      },
      bezierCurveTo: function(x1, y1, x2, y2, x, y) {
        this._ += "C" + (+x1) + "," + (+y1) + "," + (+x2) + "," + (+y2) + "," + (this._x1 = +x) + "," + (this._y1 = +y);
      },
      arcTo: function(x1, y1, x2, y2, r) {
        x1 = +x1, y1 = +y1, x2 = +x2, y2 = +y2, r = +r;
        var x0 = this._x1,
            y0 = this._y1,
            x21 = x2 - x1,
            y21 = y2 - y1,
            x01 = x0 - x1,
            y01 = y0 - y1,
            l01_2 = x01 * x01 + y01 * y01;

        // Is the radius negative? Error.
        if (r < 0) throw new Error("negative radius: " + r);

        // Is this path empty? Move to (x1,y1).
        if (this._x1 === null) {
          this._ += "M" + (this._x1 = x1) + "," + (this._y1 = y1);
        }

        // Or, is (x1,y1) coincident with (x0,y0)? Do nothing.
        else if (!(l01_2 > epsilon$9));

        // Or, are (x0,y0), (x1,y1) and (x2,y2) collinear?
        // Equivalently, is (x1,y1) coincident with (x2,y2)?
        // Or, is the radius zero? Line to (x1,y1).
        else if (!(Math.abs(y01 * x21 - y21 * x01) > epsilon$9) || !r) {
          this._ += "L" + (this._x1 = x1) + "," + (this._y1 = y1);
        }

        // Otherwise, draw an arc!
        else {
          var x20 = x2 - x0,
              y20 = y2 - y0,
              l21_2 = x21 * x21 + y21 * y21,
              l20_2 = x20 * x20 + y20 * y20,
              l21 = Math.sqrt(l21_2),
              l01 = Math.sqrt(l01_2),
              l = r * Math.tan((pi$a - Math.acos((l21_2 + l01_2 - l20_2) / (2 * l21 * l01))) / 2),
              t01 = l / l01,
              t21 = l / l21;

          // If the start tangent is not coincident with (x0,y0), line to.
          if (Math.abs(t01 - 1) > epsilon$9) {
            this._ += "L" + (x1 + t01 * x01) + "," + (y1 + t01 * y01);
          }

          this._ += "A" + r + "," + r + ",0,0," + (+(y01 * x20 > x01 * y20)) + "," + (this._x1 = x1 + t21 * x21) + "," + (this._y1 = y1 + t21 * y21);
        }
      },
      arc: function(x, y, r, a0, a1, ccw) {
        x = +x, y = +y, r = +r, ccw = !!ccw;
        var dx = r * Math.cos(a0),
            dy = r * Math.sin(a0),
            x0 = x + dx,
            y0 = y + dy,
            cw = 1 ^ ccw,
            da = ccw ? a0 - a1 : a1 - a0;

        // Is the radius negative? Error.
        if (r < 0) throw new Error("negative radius: " + r);

        // Is this path empty? Move to (x0,y0).
        if (this._x1 === null) {
          this._ += "M" + x0 + "," + y0;
        }

        // Or, is (x0,y0) not coincident with the previous point? Line to (x0,y0).
        else if (Math.abs(this._x1 - x0) > epsilon$9 || Math.abs(this._y1 - y0) > epsilon$9) {
          this._ += "L" + x0 + "," + y0;
        }

        // Is this arc empty? We’re done.
        if (!r) return;

        // Does the angle go the wrong way? Flip the direction.
        if (da < 0) da = da % tau$a + tau$a;

        // Is this a complete circle? Draw two arcs to complete the circle.
        if (da > tauEpsilon$3) {
          this._ += "A" + r + "," + r + ",0,1," + cw + "," + (x - dx) + "," + (y - dy) + "A" + r + "," + r + ",0,1," + cw + "," + (this._x1 = x0) + "," + (this._y1 = y0);
        }

        // Is this arc non-empty? Draw an arc!
        else if (da > epsilon$9) {
          this._ += "A" + r + "," + r + ",0," + (+(da >= pi$a)) + "," + cw + "," + (this._x1 = x + r * Math.cos(a1)) + "," + (this._y1 = y + r * Math.sin(a1));
        }
      },
      rect: function(x, y, w, h) {
        this._ += "M" + (this._x0 = this._x1 = +x) + "," + (this._y0 = this._y1 = +y) + "h" + (+w) + "v" + (+h) + "h" + (-w) + "Z";
      },
      toString: function() {
        return this._;
      }
    };

    function constant$s(x) {
      return function constant() {
        return x;
      };
    }

    var abs$3 = Math.abs;
    var atan2$3 = Math.atan2;
    var cos$5 = Math.cos;
    var max$5 = Math.max;
    var min$3 = Math.min;
    var sin$5 = Math.sin;
    var sqrt$5 = Math.sqrt;

    var epsilon$a = 1e-12;
    var pi$b = Math.PI;
    var halfPi$7 = pi$b / 2;
    var tau$b = 2 * pi$b;

    function acos$3(x) {
      return x > 1 ? 0 : x < -1 ? pi$b : Math.acos(x);
    }

    function asin$3(x) {
      return x >= 1 ? halfPi$7 : x <= -1 ? -halfPi$7 : Math.asin(x);
    }

    function arcInnerRadius$1(d) {
      return d.innerRadius;
    }

    function arcOuterRadius$1(d) {
      return d.outerRadius;
    }

    function arcStartAngle$1(d) {
      return d.startAngle;
    }

    function arcEndAngle$1(d) {
      return d.endAngle;
    }

    function arcPadAngle$1(d) {
      return d && d.padAngle; // Note: optional!
    }

    function intersect$1(x0, y0, x1, y1, x2, y2, x3, y3) {
      var x10 = x1 - x0, y10 = y1 - y0,
          x32 = x3 - x2, y32 = y3 - y2,
          t = (x32 * (y0 - y2) - y32 * (x0 - x2)) / (y32 * x10 - x32 * y10);
      return [x0 + t * x10, y0 + t * y10];
    }

    // Compute perpendicular offset line of length rc.
    // http://mathworld.wolfram.com/Circle-LineIntersection.html
    function cornerTangents$1(x0, y0, x1, y1, r1, rc, cw) {
      var x01 = x0 - x1,
          y01 = y0 - y1,
          lo = (cw ? rc : -rc) / sqrt$5(x01 * x01 + y01 * y01),
          ox = lo * y01,
          oy = -lo * x01,
          x11 = x0 + ox,
          y11 = y0 + oy,
          x10 = x1 + ox,
          y10 = y1 + oy,
          x00 = (x11 + x10) / 2,
          y00 = (y11 + y10) / 2,
          dx = x10 - x11,
          dy = y10 - y11,
          d2 = dx * dx + dy * dy,
          r = r1 - rc,
          D = x11 * y10 - x10 * y11,
          d = (dy < 0 ? -1 : 1) * sqrt$5(max$5(0, r * r * d2 - D * D)),
          cx0 = (D * dy - dx * d) / d2,
          cy0 = (-D * dx - dy * d) / d2,
          cx1 = (D * dy + dx * d) / d2,
          cy1 = (-D * dx + dy * d) / d2,
          dx0 = cx0 - x00,
          dy0 = cy0 - y00,
          dx1 = cx1 - x00,
          dy1 = cy1 - y00;

      // Pick the closer of the two intersection points.
      // TODO Is there a faster way to determine which intersection to use?
      if (dx0 * dx0 + dy0 * dy0 > dx1 * dx1 + dy1 * dy1) cx0 = cx1, cy0 = cy1;

      return {
        cx: cx0,
        cy: cy0,
        x01: -ox,
        y01: -oy,
        x11: cx0 * (r1 / r - 1),
        y11: cy0 * (r1 / r - 1)
      };
    }

    function arc$1() {
      var innerRadius = arcInnerRadius$1,
          outerRadius = arcOuterRadius$1,
          cornerRadius = constant$s(0),
          padRadius = null,
          startAngle = arcStartAngle$1,
          endAngle = arcEndAngle$1,
          padAngle = arcPadAngle$1,
          context = null;

      function arc() {
        var buffer,
            r,
            r0 = +innerRadius.apply(this, arguments),
            r1 = +outerRadius.apply(this, arguments),
            a0 = startAngle.apply(this, arguments) - halfPi$7,
            a1 = endAngle.apply(this, arguments) - halfPi$7,
            da = abs$3(a1 - a0),
            cw = a1 > a0;

        if (!context) context = buffer = path$3();

        // Ensure that the outer radius is always larger than the inner radius.
        if (r1 < r0) r = r1, r1 = r0, r0 = r;

        // Is it a point?
        if (!(r1 > epsilon$a)) context.moveTo(0, 0);

        // Or is it a circle or annulus?
        else if (da > tau$b - epsilon$a) {
          context.moveTo(r1 * cos$5(a0), r1 * sin$5(a0));
          context.arc(0, 0, r1, a0, a1, !cw);
          if (r0 > epsilon$a) {
            context.moveTo(r0 * cos$5(a1), r0 * sin$5(a1));
            context.arc(0, 0, r0, a1, a0, cw);
          }
        }

        // Or is it a circular or annular sector?
        else {
          var a01 = a0,
              a11 = a1,
              a00 = a0,
              a10 = a1,
              da0 = da,
              da1 = da,
              ap = padAngle.apply(this, arguments) / 2,
              rp = (ap > epsilon$a) && (padRadius ? +padRadius.apply(this, arguments) : sqrt$5(r0 * r0 + r1 * r1)),
              rc = min$3(abs$3(r1 - r0) / 2, +cornerRadius.apply(this, arguments)),
              rc0 = rc,
              rc1 = rc,
              t0,
              t1;

          // Apply padding? Note that since r1 ≥ r0, da1 ≥ da0.
          if (rp > epsilon$a) {
            var p0 = asin$3(rp / r0 * sin$5(ap)),
                p1 = asin$3(rp / r1 * sin$5(ap));
            if ((da0 -= p0 * 2) > epsilon$a) p0 *= (cw ? 1 : -1), a00 += p0, a10 -= p0;
            else da0 = 0, a00 = a10 = (a0 + a1) / 2;
            if ((da1 -= p1 * 2) > epsilon$a) p1 *= (cw ? 1 : -1), a01 += p1, a11 -= p1;
            else da1 = 0, a01 = a11 = (a0 + a1) / 2;
          }

          var x01 = r1 * cos$5(a01),
              y01 = r1 * sin$5(a01),
              x10 = r0 * cos$5(a10),
              y10 = r0 * sin$5(a10);

          // Apply rounded corners?
          if (rc > epsilon$a) {
            var x11 = r1 * cos$5(a11),
                y11 = r1 * sin$5(a11),
                x00 = r0 * cos$5(a00),
                y00 = r0 * sin$5(a00);

            // Restrict the corner radius according to the sector angle.
            if (da < pi$b) {
              var oc = da0 > epsilon$a ? intersect$1(x01, y01, x00, y00, x11, y11, x10, y10) : [x10, y10],
                  ax = x01 - oc[0],
                  ay = y01 - oc[1],
                  bx = x11 - oc[0],
                  by = y11 - oc[1],
                  kc = 1 / sin$5(acos$3((ax * bx + ay * by) / (sqrt$5(ax * ax + ay * ay) * sqrt$5(bx * bx + by * by))) / 2),
                  lc = sqrt$5(oc[0] * oc[0] + oc[1] * oc[1]);
              rc0 = min$3(rc, (r0 - lc) / (kc - 1));
              rc1 = min$3(rc, (r1 - lc) / (kc + 1));
            }
          }

          // Is the sector collapsed to a line?
          if (!(da1 > epsilon$a)) context.moveTo(x01, y01);

          // Does the sector’s outer ring have rounded corners?
          else if (rc1 > epsilon$a) {
            t0 = cornerTangents$1(x00, y00, x01, y01, r1, rc1, cw);
            t1 = cornerTangents$1(x11, y11, x10, y10, r1, rc1, cw);

            context.moveTo(t0.cx + t0.x01, t0.cy + t0.y01);

            // Have the corners merged?
            if (rc1 < rc) context.arc(t0.cx, t0.cy, rc1, atan2$3(t0.y01, t0.x01), atan2$3(t1.y01, t1.x01), !cw);

            // Otherwise, draw the two corners and the ring.
            else {
              context.arc(t0.cx, t0.cy, rc1, atan2$3(t0.y01, t0.x01), atan2$3(t0.y11, t0.x11), !cw);
              context.arc(0, 0, r1, atan2$3(t0.cy + t0.y11, t0.cx + t0.x11), atan2$3(t1.cy + t1.y11, t1.cx + t1.x11), !cw);
              context.arc(t1.cx, t1.cy, rc1, atan2$3(t1.y11, t1.x11), atan2$3(t1.y01, t1.x01), !cw);
            }
          }

          // Or is the outer ring just a circular arc?
          else context.moveTo(x01, y01), context.arc(0, 0, r1, a01, a11, !cw);

          // Is there no inner ring, and it’s a circular sector?
          // Or perhaps it’s an annular sector collapsed due to padding?
          if (!(r0 > epsilon$a) || !(da0 > epsilon$a)) context.lineTo(x10, y10);

          // Does the sector’s inner ring (or point) have rounded corners?
          else if (rc0 > epsilon$a) {
            t0 = cornerTangents$1(x10, y10, x11, y11, r0, -rc0, cw);
            t1 = cornerTangents$1(x01, y01, x00, y00, r0, -rc0, cw);

            context.lineTo(t0.cx + t0.x01, t0.cy + t0.y01);

            // Have the corners merged?
            if (rc0 < rc) context.arc(t0.cx, t0.cy, rc0, atan2$3(t0.y01, t0.x01), atan2$3(t1.y01, t1.x01), !cw);

            // Otherwise, draw the two corners and the ring.
            else {
              context.arc(t0.cx, t0.cy, rc0, atan2$3(t0.y01, t0.x01), atan2$3(t0.y11, t0.x11), !cw);
              context.arc(0, 0, r0, atan2$3(t0.cy + t0.y11, t0.cx + t0.x11), atan2$3(t1.cy + t1.y11, t1.cx + t1.x11), cw);
              context.arc(t1.cx, t1.cy, rc0, atan2$3(t1.y11, t1.x11), atan2$3(t1.y01, t1.x01), !cw);
            }
          }

          // Or is the inner ring just a circular arc?
          else context.arc(0, 0, r0, a10, a00, cw);
        }

        context.closePath();

        if (buffer) return context = null, buffer + "" || null;
      }

      arc.centroid = function() {
        var r = (+innerRadius.apply(this, arguments) + +outerRadius.apply(this, arguments)) / 2,
            a = (+startAngle.apply(this, arguments) + +endAngle.apply(this, arguments)) / 2 - pi$b / 2;
        return [cos$5(a) * r, sin$5(a) * r];
      };

      arc.innerRadius = function(_) {
        return arguments.length ? (innerRadius = typeof _ === "function" ? _ : constant$s(+_), arc) : innerRadius;
      };

      arc.outerRadius = function(_) {
        return arguments.length ? (outerRadius = typeof _ === "function" ? _ : constant$s(+_), arc) : outerRadius;
      };

      arc.cornerRadius = function(_) {
        return arguments.length ? (cornerRadius = typeof _ === "function" ? _ : constant$s(+_), arc) : cornerRadius;
      };

      arc.padRadius = function(_) {
        return arguments.length ? (padRadius = _ == null ? null : typeof _ === "function" ? _ : constant$s(+_), arc) : padRadius;
      };

      arc.startAngle = function(_) {
        return arguments.length ? (startAngle = typeof _ === "function" ? _ : constant$s(+_), arc) : startAngle;
      };

      arc.endAngle = function(_) {
        return arguments.length ? (endAngle = typeof _ === "function" ? _ : constant$s(+_), arc) : endAngle;
      };

      arc.padAngle = function(_) {
        return arguments.length ? (padAngle = typeof _ === "function" ? _ : constant$s(+_), arc) : padAngle;
      };

      arc.context = function(_) {
        return arguments.length ? ((context = _ == null ? null : _), arc) : context;
      };

      return arc;
    }

    function Linear$1(context) {
      this._context = context;
    }

    Linear$1.prototype = {
      areaStart: function() {
        this._line = 0;
      },
      areaEnd: function() {
        this._line = NaN;
      },
      lineStart: function() {
        this._point = 0;
      },
      lineEnd: function() {
        if (this._line || (this._line !== 0 && this._point === 1)) this._context.closePath();
        this._line = 1 - this._line;
      },
      point: function(x, y) {
        x = +x, y = +y;
        switch (this._point) {
          case 0: this._point = 1; this._line ? this._context.lineTo(x, y) : this._context.moveTo(x, y); break;
          case 1: this._point = 2; // proceed
          default: this._context.lineTo(x, y); break;
        }
      }
    };

    function curveLinear$1(context) {
      return new Linear$1(context);
    }

    function x$8(p) {
      return p[0];
    }

    function y$8(p) {
      return p[1];
    }

    function line$1() {
      var x = x$8,
          y = y$8,
          defined = constant$s(true),
          context = null,
          curve = curveLinear$1,
          output = null;

      function line(data) {
        var i,
            n = data.length,
            d,
            defined0 = false,
            buffer;

        if (context == null) output = curve(buffer = path$3());

        for (i = 0; i <= n; ++i) {
          if (!(i < n && defined(d = data[i], i, data)) === defined0) {
            if (defined0 = !defined0) output.lineStart();
            else output.lineEnd();
          }
          if (defined0) output.point(+x(d, i, data), +y(d, i, data));
        }

        if (buffer) return output = null, buffer + "" || null;
      }

      line.x = function(_) {
        return arguments.length ? (x = typeof _ === "function" ? _ : constant$s(+_), line) : x;
      };

      line.y = function(_) {
        return arguments.length ? (y = typeof _ === "function" ? _ : constant$s(+_), line) : y;
      };

      line.defined = function(_) {
        return arguments.length ? (defined = typeof _ === "function" ? _ : constant$s(!!_), line) : defined;
      };

      line.curve = function(_) {
        return arguments.length ? (curve = _, context != null && (output = curve(context)), line) : curve;
      };

      line.context = function(_) {
        return arguments.length ? (_ == null ? context = output = null : output = curve(context = _), line) : context;
      };

      return line;
    }

    function area$6() {
      var x0 = x$8,
          x1 = null,
          y0 = constant$s(0),
          y1 = y$8,
          defined = constant$s(true),
          context = null,
          curve = curveLinear$1,
          output = null;

      function area(data) {
        var i,
            j,
            k,
            n = data.length,
            d,
            defined0 = false,
            buffer,
            x0z = new Array(n),
            y0z = new Array(n);

        if (context == null) output = curve(buffer = path$3());

        for (i = 0; i <= n; ++i) {
          if (!(i < n && defined(d = data[i], i, data)) === defined0) {
            if (defined0 = !defined0) {
              j = i;
              output.areaStart();
              output.lineStart();
            } else {
              output.lineEnd();
              output.lineStart();
              for (k = i - 1; k >= j; --k) {
                output.point(x0z[k], y0z[k]);
              }
              output.lineEnd();
              output.areaEnd();
            }
          }
          if (defined0) {
            x0z[i] = +x0(d, i, data), y0z[i] = +y0(d, i, data);
            output.point(x1 ? +x1(d, i, data) : x0z[i], y1 ? +y1(d, i, data) : y0z[i]);
          }
        }

        if (buffer) return output = null, buffer + "" || null;
      }

      function arealine() {
        return line$1().defined(defined).curve(curve).context(context);
      }

      area.x = function(_) {
        return arguments.length ? (x0 = typeof _ === "function" ? _ : constant$s(+_), x1 = null, area) : x0;
      };

      area.x0 = function(_) {
        return arguments.length ? (x0 = typeof _ === "function" ? _ : constant$s(+_), area) : x0;
      };

      area.x1 = function(_) {
        return arguments.length ? (x1 = _ == null ? null : typeof _ === "function" ? _ : constant$s(+_), area) : x1;
      };

      area.y = function(_) {
        return arguments.length ? (y0 = typeof _ === "function" ? _ : constant$s(+_), y1 = null, area) : y0;
      };

      area.y0 = function(_) {
        return arguments.length ? (y0 = typeof _ === "function" ? _ : constant$s(+_), area) : y0;
      };

      area.y1 = function(_) {
        return arguments.length ? (y1 = _ == null ? null : typeof _ === "function" ? _ : constant$s(+_), area) : y1;
      };

      area.lineX0 =
      area.lineY0 = function() {
        return arealine().x(x0).y(y0);
      };

      area.lineY1 = function() {
        return arealine().x(x0).y(y1);
      };

      area.lineX1 = function() {
        return arealine().x(x1).y(y0);
      };

      area.defined = function(_) {
        return arguments.length ? (defined = typeof _ === "function" ? _ : constant$s(!!_), area) : defined;
      };

      area.curve = function(_) {
        return arguments.length ? (curve = _, context != null && (output = curve(context)), area) : curve;
      };

      area.context = function(_) {
        return arguments.length ? (_ == null ? context = output = null : output = curve(context = _), area) : context;
      };

      return area;
    }

    function descending$4(a, b) {
      return b < a ? -1 : b > a ? 1 : b >= a ? 0 : NaN;
    }

    function identity$j(d) {
      return d;
    }

    function pie$1() {
      var value = identity$j,
          sortValues = descending$4,
          sort = null,
          startAngle = constant$s(0),
          endAngle = constant$s(tau$b),
          padAngle = constant$s(0);

      function pie(data) {
        var i,
            n = data.length,
            j,
            k,
            sum = 0,
            index = new Array(n),
            arcs = new Array(n),
            a0 = +startAngle.apply(this, arguments),
            da = Math.min(tau$b, Math.max(-tau$b, endAngle.apply(this, arguments) - a0)),
            a1,
            p = Math.min(Math.abs(da) / n, padAngle.apply(this, arguments)),
            pa = p * (da < 0 ? -1 : 1),
            v;

        for (i = 0; i < n; ++i) {
          if ((v = arcs[index[i] = i] = +value(data[i], i, data)) > 0) {
            sum += v;
          }
        }

        // Optionally sort the arcs by previously-computed values or by data.
        if (sortValues != null) index.sort(function(i, j) { return sortValues(arcs[i], arcs[j]); });
        else if (sort != null) index.sort(function(i, j) { return sort(data[i], data[j]); });

        // Compute the arcs! They are stored in the original data's order.
        for (i = 0, k = sum ? (da - n * pa) / sum : 0; i < n; ++i, a0 = a1) {
          j = index[i], v = arcs[j], a1 = a0 + (v > 0 ? v * k : 0) + pa, arcs[j] = {
            data: data[j],
            index: i,
            value: v,
            startAngle: a0,
            endAngle: a1,
            padAngle: p
          };
        }

        return arcs;
      }

      pie.value = function(_) {
        return arguments.length ? (value = typeof _ === "function" ? _ : constant$s(+_), pie) : value;
      };

      pie.sortValues = function(_) {
        return arguments.length ? (sortValues = _, sort = null, pie) : sortValues;
      };

      pie.sort = function(_) {
        return arguments.length ? (sort = _, sortValues = null, pie) : sort;
      };

      pie.startAngle = function(_) {
        return arguments.length ? (startAngle = typeof _ === "function" ? _ : constant$s(+_), pie) : startAngle;
      };

      pie.endAngle = function(_) {
        return arguments.length ? (endAngle = typeof _ === "function" ? _ : constant$s(+_), pie) : endAngle;
      };

      pie.padAngle = function(_) {
        return arguments.length ? (padAngle = typeof _ === "function" ? _ : constant$s(+_), pie) : padAngle;
      };

      return pie;
    }

    var curveRadialLinear$1 = curveRadial$2(curveLinear$1);

    function Radial$1(curve) {
      this._curve = curve;
    }

    Radial$1.prototype = {
      areaStart: function() {
        this._curve.areaStart();
      },
      areaEnd: function() {
        this._curve.areaEnd();
      },
      lineStart: function() {
        this._curve.lineStart();
      },
      lineEnd: function() {
        this._curve.lineEnd();
      },
      point: function(a, r) {
        this._curve.point(r * Math.sin(a), r * -Math.cos(a));
      }
    };

    function curveRadial$2(curve) {

      function radial(context) {
        return new Radial$1(curve(context));
      }

      radial._curve = curve;

      return radial;
    }

    function lineRadial$2(l) {
      var c = l.curve;

      l.angle = l.x, delete l.x;
      l.radius = l.y, delete l.y;

      l.curve = function(_) {
        return arguments.length ? c(curveRadial$2(_)) : c()._curve;
      };

      return l;
    }

    function lineRadial$3() {
      return lineRadial$2(line$1().curve(curveRadialLinear$1));
    }

    function areaRadial$1() {
      var a = area$6().curve(curveRadialLinear$1),
          c = a.curve,
          x0 = a.lineX0,
          x1 = a.lineX1,
          y0 = a.lineY0,
          y1 = a.lineY1;

      a.angle = a.x, delete a.x;
      a.startAngle = a.x0, delete a.x0;
      a.endAngle = a.x1, delete a.x1;
      a.radius = a.y, delete a.y;
      a.innerRadius = a.y0, delete a.y0;
      a.outerRadius = a.y1, delete a.y1;
      a.lineStartAngle = function() { return lineRadial$2(x0()); }, delete a.lineX0;
      a.lineEndAngle = function() { return lineRadial$2(x1()); }, delete a.lineX1;
      a.lineInnerRadius = function() { return lineRadial$2(y0()); }, delete a.lineY0;
      a.lineOuterRadius = function() { return lineRadial$2(y1()); }, delete a.lineY1;

      a.curve = function(_) {
        return arguments.length ? c(curveRadial$2(_)) : c()._curve;
      };

      return a;
    }

    function pointRadial$1(x, y) {
      return [(y = +y) * Math.cos(x -= Math.PI / 2), y * Math.sin(x)];
    }

    var slice$d = Array.prototype.slice;

    function linkSource$1(d) {
      return d.source;
    }

    function linkTarget$1(d) {
      return d.target;
    }

    function link$5(curve) {
      var source = linkSource$1,
          target = linkTarget$1,
          x = x$8,
          y = y$8,
          context = null;

      function link() {
        var buffer, argv = slice$d.call(arguments), s = source.apply(this, argv), t = target.apply(this, argv);
        if (!context) context = buffer = path$3();
        curve(context, +x.apply(this, (argv[0] = s, argv)), +y.apply(this, argv), +x.apply(this, (argv[0] = t, argv)), +y.apply(this, argv));
        if (buffer) return context = null, buffer + "" || null;
      }

      link.source = function(_) {
        return arguments.length ? (source = _, link) : source;
      };

      link.target = function(_) {
        return arguments.length ? (target = _, link) : target;
      };

      link.x = function(_) {
        return arguments.length ? (x = typeof _ === "function" ? _ : constant$s(+_), link) : x;
      };

      link.y = function(_) {
        return arguments.length ? (y = typeof _ === "function" ? _ : constant$s(+_), link) : y;
      };

      link.context = function(_) {
        return arguments.length ? ((context = _ == null ? null : _), link) : context;
      };

      return link;
    }

    function curveHorizontal$1(context, x0, y0, x1, y1) {
      context.moveTo(x0, y0);
      context.bezierCurveTo(x0 = (x0 + x1) / 2, y0, x0, y1, x1, y1);
    }

    function curveVertical$1(context, x0, y0, x1, y1) {
      context.moveTo(x0, y0);
      context.bezierCurveTo(x0, y0 = (y0 + y1) / 2, x1, y0, x1, y1);
    }

    function curveRadial$3(context, x0, y0, x1, y1) {
      var p0 = pointRadial$1(x0, y0),
          p1 = pointRadial$1(x0, y0 = (y0 + y1) / 2),
          p2 = pointRadial$1(x1, y0),
          p3 = pointRadial$1(x1, y1);
      context.moveTo(p0[0], p0[1]);
      context.bezierCurveTo(p1[0], p1[1], p2[0], p2[1], p3[0], p3[1]);
    }

    function linkHorizontal$1() {
      return link$5(curveHorizontal$1);
    }

    function linkVertical$1() {
      return link$5(curveVertical$1);
    }

    function linkRadial$1() {
      var l = link$5(curveRadial$3);
      l.angle = l.x, delete l.x;
      l.radius = l.y, delete l.y;
      return l;
    }

    var circle$5 = {
      draw: function(context, size) {
        var r = Math.sqrt(size / pi$b);
        context.moveTo(r, 0);
        context.arc(0, 0, r, 0, tau$b);
      }
    };

    var cross$5 = {
      draw: function(context, size) {
        var r = Math.sqrt(size / 5) / 2;
        context.moveTo(-3 * r, -r);
        context.lineTo(-r, -r);
        context.lineTo(-r, -3 * r);
        context.lineTo(r, -3 * r);
        context.lineTo(r, -r);
        context.lineTo(3 * r, -r);
        context.lineTo(3 * r, r);
        context.lineTo(r, r);
        context.lineTo(r, 3 * r);
        context.lineTo(-r, 3 * r);
        context.lineTo(-r, r);
        context.lineTo(-3 * r, r);
        context.closePath();
      }
    };

    var tan30$1 = Math.sqrt(1 / 3),
        tan30_2$1 = tan30$1 * 2;

    var diamond$1 = {
      draw: function(context, size) {
        var y = Math.sqrt(size / tan30_2$1),
            x = y * tan30$1;
        context.moveTo(0, -y);
        context.lineTo(x, 0);
        context.lineTo(0, y);
        context.lineTo(-x, 0);
        context.closePath();
      }
    };

    var ka$1 = 0.89081309152928522810,
        kr$1 = Math.sin(pi$b / 10) / Math.sin(7 * pi$b / 10),
        kx$1 = Math.sin(tau$b / 10) * kr$1,
        ky$1 = -Math.cos(tau$b / 10) * kr$1;

    var star$1 = {
      draw: function(context, size) {
        var r = Math.sqrt(size * ka$1),
            x = kx$1 * r,
            y = ky$1 * r;
        context.moveTo(0, -r);
        context.lineTo(x, y);
        for (var i = 1; i < 5; ++i) {
          var a = tau$b * i / 5,
              c = Math.cos(a),
              s = Math.sin(a);
          context.lineTo(s * r, -c * r);
          context.lineTo(c * x - s * y, s * x + c * y);
        }
        context.closePath();
      }
    };

    var square$1 = {
      draw: function(context, size) {
        var w = Math.sqrt(size),
            x = -w / 2;
        context.rect(x, x, w, w);
      }
    };

    var sqrt3$1 = Math.sqrt(3);

    var triangle$1 = {
      draw: function(context, size) {
        var y = -Math.sqrt(size / (sqrt3$1 * 3));
        context.moveTo(0, y * 2);
        context.lineTo(-sqrt3$1 * y, -y);
        context.lineTo(sqrt3$1 * y, -y);
        context.closePath();
      }
    };

    var c$3 = -0.5,
        s$1 = Math.sqrt(3) / 2,
        k$1 = 1 / Math.sqrt(12),
        a$1 = (k$1 / 2 + 1) * 3;

    var wye$1 = {
      draw: function(context, size) {
        var r = Math.sqrt(size / a$1),
            x0 = r / 2,
            y0 = r * k$1,
            x1 = x0,
            y1 = r * k$1 + r,
            x2 = -x1,
            y2 = y1;
        context.moveTo(x0, y0);
        context.lineTo(x1, y1);
        context.lineTo(x2, y2);
        context.lineTo(c$3 * x0 - s$1 * y0, s$1 * x0 + c$3 * y0);
        context.lineTo(c$3 * x1 - s$1 * y1, s$1 * x1 + c$3 * y1);
        context.lineTo(c$3 * x2 - s$1 * y2, s$1 * x2 + c$3 * y2);
        context.lineTo(c$3 * x0 + s$1 * y0, c$3 * y0 - s$1 * x0);
        context.lineTo(c$3 * x1 + s$1 * y1, c$3 * y1 - s$1 * x1);
        context.lineTo(c$3 * x2 + s$1 * y2, c$3 * y2 - s$1 * x2);
        context.closePath();
      }
    };

    var symbols$1 = [
      circle$5,
      cross$5,
      diamond$1,
      square$1,
      star$1,
      triangle$1,
      wye$1
    ];

    function symbol$1() {
      var type = constant$s(circle$5),
          size = constant$s(64),
          context = null;

      function symbol() {
        var buffer;
        if (!context) context = buffer = path$3();
        type.apply(this, arguments).draw(context, +size.apply(this, arguments));
        if (buffer) return context = null, buffer + "" || null;
      }

      symbol.type = function(_) {
        return arguments.length ? (type = typeof _ === "function" ? _ : constant$s(_), symbol) : type;
      };

      symbol.size = function(_) {
        return arguments.length ? (size = typeof _ === "function" ? _ : constant$s(+_), symbol) : size;
      };

      symbol.context = function(_) {
        return arguments.length ? (context = _ == null ? null : _, symbol) : context;
      };

      return symbol;
    }

    function noop$9() {}

    function point$a(that, x, y) {
      that._context.bezierCurveTo(
        (2 * that._x0 + that._x1) / 3,
        (2 * that._y0 + that._y1) / 3,
        (that._x0 + 2 * that._x1) / 3,
        (that._y0 + 2 * that._y1) / 3,
        (that._x0 + 4 * that._x1 + x) / 6,
        (that._y0 + 4 * that._y1 + y) / 6
      );
    }

    function Basis$1(context) {
      this._context = context;
    }

    Basis$1.prototype = {
      areaStart: function() {
        this._line = 0;
      },
      areaEnd: function() {
        this._line = NaN;
      },
      lineStart: function() {
        this._x0 = this._x1 =
        this._y0 = this._y1 = NaN;
        this._point = 0;
      },
      lineEnd: function() {
        switch (this._point) {
          case 3: point$a(this, this._x1, this._y1); // proceed
          case 2: this._context.lineTo(this._x1, this._y1); break;
        }
        if (this._line || (this._line !== 0 && this._point === 1)) this._context.closePath();
        this._line = 1 - this._line;
      },
      point: function(x, y) {
        x = +x, y = +y;
        switch (this._point) {
          case 0: this._point = 1; this._line ? this._context.lineTo(x, y) : this._context.moveTo(x, y); break;
          case 1: this._point = 2; break;
          case 2: this._point = 3; this._context.lineTo((5 * this._x0 + this._x1) / 6, (5 * this._y0 + this._y1) / 6); // proceed
          default: point$a(this, x, y); break;
        }
        this._x0 = this._x1, this._x1 = x;
        this._y0 = this._y1, this._y1 = y;
      }
    };

    function basis$5(context) {
      return new Basis$1(context);
    }

    function BasisClosed$1(context) {
      this._context = context;
    }

    BasisClosed$1.prototype = {
      areaStart: noop$9,
      areaEnd: noop$9,
      lineStart: function() {
        this._x0 = this._x1 = this._x2 = this._x3 = this._x4 =
        this._y0 = this._y1 = this._y2 = this._y3 = this._y4 = NaN;
        this._point = 0;
      },
      lineEnd: function() {
        switch (this._point) {
          case 1: {
            this._context.moveTo(this._x2, this._y2);
            this._context.closePath();
            break;
          }
          case 2: {
            this._context.moveTo((this._x2 + 2 * this._x3) / 3, (this._y2 + 2 * this._y3) / 3);
            this._context.lineTo((this._x3 + 2 * this._x2) / 3, (this._y3 + 2 * this._y2) / 3);
            this._context.closePath();
            break;
          }
          case 3: {
            this.point(this._x2, this._y2);
            this.point(this._x3, this._y3);
            this.point(this._x4, this._y4);
            break;
          }
        }
      },
      point: function(x, y) {
        x = +x, y = +y;
        switch (this._point) {
          case 0: this._point = 1; this._x2 = x, this._y2 = y; break;
          case 1: this._point = 2; this._x3 = x, this._y3 = y; break;
          case 2: this._point = 3; this._x4 = x, this._y4 = y; this._context.moveTo((this._x0 + 4 * this._x1 + x) / 6, (this._y0 + 4 * this._y1 + y) / 6); break;
          default: point$a(this, x, y); break;
        }
        this._x0 = this._x1, this._x1 = x;
        this._y0 = this._y1, this._y1 = y;
      }
    };

    function basisClosed$3(context) {
      return new BasisClosed$1(context);
    }

    function BasisOpen$1(context) {
      this._context = context;
    }

    BasisOpen$1.prototype = {
      areaStart: function() {
        this._line = 0;
      },
      areaEnd: function() {
        this._line = NaN;
      },
      lineStart: function() {
        this._x0 = this._x1 =
        this._y0 = this._y1 = NaN;
        this._point = 0;
      },
      lineEnd: function() {
        if (this._line || (this._line !== 0 && this._point === 3)) this._context.closePath();
        this._line = 1 - this._line;
      },
      point: function(x, y) {
        x = +x, y = +y;
        switch (this._point) {
          case 0: this._point = 1; break;
          case 1: this._point = 2; break;
          case 2: this._point = 3; var x0 = (this._x0 + 4 * this._x1 + x) / 6, y0 = (this._y0 + 4 * this._y1 + y) / 6; this._line ? this._context.lineTo(x0, y0) : this._context.moveTo(x0, y0); break;
          case 3: this._point = 4; // proceed
          default: point$a(this, x, y); break;
        }
        this._x0 = this._x1, this._x1 = x;
        this._y0 = this._y1, this._y1 = y;
      }
    };

    function basisOpen$1(context) {
      return new BasisOpen$1(context);
    }

    function Bundle$1(context, beta) {
      this._basis = new Basis$1(context);
      this._beta = beta;
    }

    Bundle$1.prototype = {
      lineStart: function() {
        this._x = [];
        this._y = [];
        this._basis.lineStart();
      },
      lineEnd: function() {
        var x = this._x,
            y = this._y,
            j = x.length - 1;

        if (j > 0) {
          var x0 = x[0],
              y0 = y[0],
              dx = x[j] - x0,
              dy = y[j] - y0,
              i = -1,
              t;

          while (++i <= j) {
            t = i / j;
            this._basis.point(
              this._beta * x[i] + (1 - this._beta) * (x0 + t * dx),
              this._beta * y[i] + (1 - this._beta) * (y0 + t * dy)
            );
          }
        }

        this._x = this._y = null;
        this._basis.lineEnd();
      },
      point: function(x, y) {
        this._x.push(+x);
        this._y.push(+y);
      }
    };

    var bundle$1 = (function custom(beta) {

      function bundle(context) {
        return beta === 1 ? new Basis$1(context) : new Bundle$1(context, beta);
      }

      bundle.beta = function(beta) {
        return custom(+beta);
      };

      return bundle;
    })(0.85);

    function point$b(that, x, y) {
      that._context.bezierCurveTo(
        that._x1 + that._k * (that._x2 - that._x0),
        that._y1 + that._k * (that._y2 - that._y0),
        that._x2 + that._k * (that._x1 - x),
        that._y2 + that._k * (that._y1 - y),
        that._x2,
        that._y2
      );
    }

    function Cardinal$1(context, tension) {
      this._context = context;
      this._k = (1 - tension) / 6;
    }

    Cardinal$1.prototype = {
      areaStart: function() {
        this._line = 0;
      },
      areaEnd: function() {
        this._line = NaN;
      },
      lineStart: function() {
        this._x0 = this._x1 = this._x2 =
        this._y0 = this._y1 = this._y2 = NaN;
        this._point = 0;
      },
      lineEnd: function() {
        switch (this._point) {
          case 2: this._context.lineTo(this._x2, this._y2); break;
          case 3: point$b(this, this._x1, this._y1); break;
        }
        if (this._line || (this._line !== 0 && this._point === 1)) this._context.closePath();
        this._line = 1 - this._line;
      },
      point: function(x, y) {
        x = +x, y = +y;
        switch (this._point) {
          case 0: this._point = 1; this._line ? this._context.lineTo(x, y) : this._context.moveTo(x, y); break;
          case 1: this._point = 2; this._x1 = x, this._y1 = y; break;
          case 2: this._point = 3; // proceed
          default: point$b(this, x, y); break;
        }
        this._x0 = this._x1, this._x1 = this._x2, this._x2 = x;
        this._y0 = this._y1, this._y1 = this._y2, this._y2 = y;
      }
    };

    var cardinal$1 = (function custom(tension) {

      function cardinal(context) {
        return new Cardinal$1(context, tension);
      }

      cardinal.tension = function(tension) {
        return custom(+tension);
      };

      return cardinal;
    })(0);

    function CardinalClosed$1(context, tension) {
      this._context = context;
      this._k = (1 - tension) / 6;
    }

    CardinalClosed$1.prototype = {
      areaStart: noop$9,
      areaEnd: noop$9,
      lineStart: function() {
        this._x0 = this._x1 = this._x2 = this._x3 = this._x4 = this._x5 =
        this._y0 = this._y1 = this._y2 = this._y3 = this._y4 = this._y5 = NaN;
        this._point = 0;
      },
      lineEnd: function() {
        switch (this._point) {
          case 1: {
            this._context.moveTo(this._x3, this._y3);
            this._context.closePath();
            break;
          }
          case 2: {
            this._context.lineTo(this._x3, this._y3);
            this._context.closePath();
            break;
          }
          case 3: {
            this.point(this._x3, this._y3);
            this.point(this._x4, this._y4);
            this.point(this._x5, this._y5);
            break;
          }
        }
      },
      point: function(x, y) {
        x = +x, y = +y;
        switch (this._point) {
          case 0: this._point = 1; this._x3 = x, this._y3 = y; break;
          case 1: this._point = 2; this._context.moveTo(this._x4 = x, this._y4 = y); break;
          case 2: this._point = 3; this._x5 = x, this._y5 = y; break;
          default: point$b(this, x, y); break;
        }
        this._x0 = this._x1, this._x1 = this._x2, this._x2 = x;
        this._y0 = this._y1, this._y1 = this._y2, this._y2 = y;
      }
    };

    var cardinalClosed$1 = (function custom(tension) {

      function cardinal(context) {
        return new CardinalClosed$1(context, tension);
      }

      cardinal.tension = function(tension) {
        return custom(+tension);
      };

      return cardinal;
    })(0);

    function CardinalOpen$1(context, tension) {
      this._context = context;
      this._k = (1 - tension) / 6;
    }

    CardinalOpen$1.prototype = {
      areaStart: function() {
        this._line = 0;
      },
      areaEnd: function() {
        this._line = NaN;
      },
      lineStart: function() {
        this._x0 = this._x1 = this._x2 =
        this._y0 = this._y1 = this._y2 = NaN;
        this._point = 0;
      },
      lineEnd: function() {
        if (this._line || (this._line !== 0 && this._point === 3)) this._context.closePath();
        this._line = 1 - this._line;
      },
      point: function(x, y) {
        x = +x, y = +y;
        switch (this._point) {
          case 0: this._point = 1; break;
          case 1: this._point = 2; break;
          case 2: this._point = 3; this._line ? this._context.lineTo(this._x2, this._y2) : this._context.moveTo(this._x2, this._y2); break;
          case 3: this._point = 4; // proceed
          default: point$b(this, x, y); break;
        }
        this._x0 = this._x1, this._x1 = this._x2, this._x2 = x;
        this._y0 = this._y1, this._y1 = this._y2, this._y2 = y;
      }
    };

    var cardinalOpen$1 = (function custom(tension) {

      function cardinal(context) {
        return new CardinalOpen$1(context, tension);
      }

      cardinal.tension = function(tension) {
        return custom(+tension);
      };

      return cardinal;
    })(0);

    function point$c(that, x, y) {
      var x1 = that._x1,
          y1 = that._y1,
          x2 = that._x2,
          y2 = that._y2;

      if (that._l01_a > epsilon$a) {
        var a = 2 * that._l01_2a + 3 * that._l01_a * that._l12_a + that._l12_2a,
            n = 3 * that._l01_a * (that._l01_a + that._l12_a);
        x1 = (x1 * a - that._x0 * that._l12_2a + that._x2 * that._l01_2a) / n;
        y1 = (y1 * a - that._y0 * that._l12_2a + that._y2 * that._l01_2a) / n;
      }

      if (that._l23_a > epsilon$a) {
        var b = 2 * that._l23_2a + 3 * that._l23_a * that._l12_a + that._l12_2a,
            m = 3 * that._l23_a * (that._l23_a + that._l12_a);
        x2 = (x2 * b + that._x1 * that._l23_2a - x * that._l12_2a) / m;
        y2 = (y2 * b + that._y1 * that._l23_2a - y * that._l12_2a) / m;
      }

      that._context.bezierCurveTo(x1, y1, x2, y2, that._x2, that._y2);
    }

    function CatmullRom$1(context, alpha) {
      this._context = context;
      this._alpha = alpha;
    }

    CatmullRom$1.prototype = {
      areaStart: function() {
        this._line = 0;
      },
      areaEnd: function() {
        this._line = NaN;
      },
      lineStart: function() {
        this._x0 = this._x1 = this._x2 =
        this._y0 = this._y1 = this._y2 = NaN;
        this._l01_a = this._l12_a = this._l23_a =
        this._l01_2a = this._l12_2a = this._l23_2a =
        this._point = 0;
      },
      lineEnd: function() {
        switch (this._point) {
          case 2: this._context.lineTo(this._x2, this._y2); break;
          case 3: this.point(this._x2, this._y2); break;
        }
        if (this._line || (this._line !== 0 && this._point === 1)) this._context.closePath();
        this._line = 1 - this._line;
      },
      point: function(x, y) {
        x = +x, y = +y;

        if (this._point) {
          var x23 = this._x2 - x,
              y23 = this._y2 - y;
          this._l23_a = Math.sqrt(this._l23_2a = Math.pow(x23 * x23 + y23 * y23, this._alpha));
        }

        switch (this._point) {
          case 0: this._point = 1; this._line ? this._context.lineTo(x, y) : this._context.moveTo(x, y); break;
          case 1: this._point = 2; break;
          case 2: this._point = 3; // proceed
          default: point$c(this, x, y); break;
        }

        this._l01_a = this._l12_a, this._l12_a = this._l23_a;
        this._l01_2a = this._l12_2a, this._l12_2a = this._l23_2a;
        this._x0 = this._x1, this._x1 = this._x2, this._x2 = x;
        this._y0 = this._y1, this._y1 = this._y2, this._y2 = y;
      }
    };

    var catmullRom$1 = (function custom(alpha) {

      function catmullRom(context) {
        return alpha ? new CatmullRom$1(context, alpha) : new Cardinal$1(context, 0);
      }

      catmullRom.alpha = function(alpha) {
        return custom(+alpha);
      };

      return catmullRom;
    })(0.5);

    function CatmullRomClosed$1(context, alpha) {
      this._context = context;
      this._alpha = alpha;
    }

    CatmullRomClosed$1.prototype = {
      areaStart: noop$9,
      areaEnd: noop$9,
      lineStart: function() {
        this._x0 = this._x1 = this._x2 = this._x3 = this._x4 = this._x5 =
        this._y0 = this._y1 = this._y2 = this._y3 = this._y4 = this._y5 = NaN;
        this._l01_a = this._l12_a = this._l23_a =
        this._l01_2a = this._l12_2a = this._l23_2a =
        this._point = 0;
      },
      lineEnd: function() {
        switch (this._point) {
          case 1: {
            this._context.moveTo(this._x3, this._y3);
            this._context.closePath();
            break;
          }
          case 2: {
            this._context.lineTo(this._x3, this._y3);
            this._context.closePath();
            break;
          }
          case 3: {
            this.point(this._x3, this._y3);
            this.point(this._x4, this._y4);
            this.point(this._x5, this._y5);
            break;
          }
        }
      },
      point: function(x, y) {
        x = +x, y = +y;

        if (this._point) {
          var x23 = this._x2 - x,
              y23 = this._y2 - y;
          this._l23_a = Math.sqrt(this._l23_2a = Math.pow(x23 * x23 + y23 * y23, this._alpha));
        }

        switch (this._point) {
          case 0: this._point = 1; this._x3 = x, this._y3 = y; break;
          case 1: this._point = 2; this._context.moveTo(this._x4 = x, this._y4 = y); break;
          case 2: this._point = 3; this._x5 = x, this._y5 = y; break;
          default: point$c(this, x, y); break;
        }

        this._l01_a = this._l12_a, this._l12_a = this._l23_a;
        this._l01_2a = this._l12_2a, this._l12_2a = this._l23_2a;
        this._x0 = this._x1, this._x1 = this._x2, this._x2 = x;
        this._y0 = this._y1, this._y1 = this._y2, this._y2 = y;
      }
    };

    var catmullRomClosed$1 = (function custom(alpha) {

      function catmullRom(context) {
        return alpha ? new CatmullRomClosed$1(context, alpha) : new CardinalClosed$1(context, 0);
      }

      catmullRom.alpha = function(alpha) {
        return custom(+alpha);
      };

      return catmullRom;
    })(0.5);

    function CatmullRomOpen$1(context, alpha) {
      this._context = context;
      this._alpha = alpha;
    }

    CatmullRomOpen$1.prototype = {
      areaStart: function() {
        this._line = 0;
      },
      areaEnd: function() {
        this._line = NaN;
      },
      lineStart: function() {
        this._x0 = this._x1 = this._x2 =
        this._y0 = this._y1 = this._y2 = NaN;
        this._l01_a = this._l12_a = this._l23_a =
        this._l01_2a = this._l12_2a = this._l23_2a =
        this._point = 0;
      },
      lineEnd: function() {
        if (this._line || (this._line !== 0 && this._point === 3)) this._context.closePath();
        this._line = 1 - this._line;
      },
      point: function(x, y) {
        x = +x, y = +y;

        if (this._point) {
          var x23 = this._x2 - x,
              y23 = this._y2 - y;
          this._l23_a = Math.sqrt(this._l23_2a = Math.pow(x23 * x23 + y23 * y23, this._alpha));
        }

        switch (this._point) {
          case 0: this._point = 1; break;
          case 1: this._point = 2; break;
          case 2: this._point = 3; this._line ? this._context.lineTo(this._x2, this._y2) : this._context.moveTo(this._x2, this._y2); break;
          case 3: this._point = 4; // proceed
          default: point$c(this, x, y); break;
        }

        this._l01_a = this._l12_a, this._l12_a = this._l23_a;
        this._l01_2a = this._l12_2a, this._l12_2a = this._l23_2a;
        this._x0 = this._x1, this._x1 = this._x2, this._x2 = x;
        this._y0 = this._y1, this._y1 = this._y2, this._y2 = y;
      }
    };

    var catmullRomOpen$1 = (function custom(alpha) {

      function catmullRom(context) {
        return alpha ? new CatmullRomOpen$1(context, alpha) : new CardinalOpen$1(context, 0);
      }

      catmullRom.alpha = function(alpha) {
        return custom(+alpha);
      };

      return catmullRom;
    })(0.5);

    function LinearClosed$1(context) {
      this._context = context;
    }

    LinearClosed$1.prototype = {
      areaStart: noop$9,
      areaEnd: noop$9,
      lineStart: function() {
        this._point = 0;
      },
      lineEnd: function() {
        if (this._point) this._context.closePath();
      },
      point: function(x, y) {
        x = +x, y = +y;
        if (this._point) this._context.lineTo(x, y);
        else this._point = 1, this._context.moveTo(x, y);
      }
    };

    function linearClosed$1(context) {
      return new LinearClosed$1(context);
    }

    function sign$3(x) {
      return x < 0 ? -1 : 1;
    }

    // Calculate the slopes of the tangents (Hermite-type interpolation) based on
    // the following paper: Steffen, M. 1990. A Simple Method for Monotonic
    // Interpolation in One Dimension. Astronomy and Astrophysics, Vol. 239, NO.
    // NOV(II), P. 443, 1990.
    function slope3$1(that, x2, y2) {
      var h0 = that._x1 - that._x0,
          h1 = x2 - that._x1,
          s0 = (that._y1 - that._y0) / (h0 || h1 < 0 && -0),
          s1 = (y2 - that._y1) / (h1 || h0 < 0 && -0),
          p = (s0 * h1 + s1 * h0) / (h0 + h1);
      return (sign$3(s0) + sign$3(s1)) * Math.min(Math.abs(s0), Math.abs(s1), 0.5 * Math.abs(p)) || 0;
    }

    // Calculate a one-sided slope.
    function slope2$1(that, t) {
      var h = that._x1 - that._x0;
      return h ? (3 * (that._y1 - that._y0) / h - t) / 2 : t;
    }

    // According to https://en.wikipedia.org/wiki/Cubic_Hermite_spline#Representations
    // "you can express cubic Hermite interpolation in terms of cubic Bézier curves
    // with respect to the four values p0, p0 + m0 / 3, p1 - m1 / 3, p1".
    function point$d(that, t0, t1) {
      var x0 = that._x0,
          y0 = that._y0,
          x1 = that._x1,
          y1 = that._y1,
          dx = (x1 - x0) / 3;
      that._context.bezierCurveTo(x0 + dx, y0 + dx * t0, x1 - dx, y1 - dx * t1, x1, y1);
    }

    function MonotoneX$1(context) {
      this._context = context;
    }

    MonotoneX$1.prototype = {
      areaStart: function() {
        this._line = 0;
      },
      areaEnd: function() {
        this._line = NaN;
      },
      lineStart: function() {
        this._x0 = this._x1 =
        this._y0 = this._y1 =
        this._t0 = NaN;
        this._point = 0;
      },
      lineEnd: function() {
        switch (this._point) {
          case 2: this._context.lineTo(this._x1, this._y1); break;
          case 3: point$d(this, this._t0, slope2$1(this, this._t0)); break;
        }
        if (this._line || (this._line !== 0 && this._point === 1)) this._context.closePath();
        this._line = 1 - this._line;
      },
      point: function(x, y) {
        var t1 = NaN;

        x = +x, y = +y;
        if (x === this._x1 && y === this._y1) return; // Ignore coincident points.
        switch (this._point) {
          case 0: this._point = 1; this._line ? this._context.lineTo(x, y) : this._context.moveTo(x, y); break;
          case 1: this._point = 2; break;
          case 2: this._point = 3; point$d(this, slope2$1(this, t1 = slope3$1(this, x, y)), t1); break;
          default: point$d(this, this._t0, t1 = slope3$1(this, x, y)); break;
        }

        this._x0 = this._x1, this._x1 = x;
        this._y0 = this._y1, this._y1 = y;
        this._t0 = t1;
      }
    };

    function MonotoneY$1(context) {
      this._context = new ReflectContext$1(context);
    }

    (MonotoneY$1.prototype = Object.create(MonotoneX$1.prototype)).point = function(x, y) {
      MonotoneX$1.prototype.point.call(this, y, x);
    };

    function ReflectContext$1(context) {
      this._context = context;
    }

    ReflectContext$1.prototype = {
      moveTo: function(x, y) { this._context.moveTo(y, x); },
      closePath: function() { this._context.closePath(); },
      lineTo: function(x, y) { this._context.lineTo(y, x); },
      bezierCurveTo: function(x1, y1, x2, y2, x, y) { this._context.bezierCurveTo(y1, x1, y2, x2, y, x); }
    };

    function monotoneX$1(context) {
      return new MonotoneX$1(context);
    }

    function monotoneY$1(context) {
      return new MonotoneY$1(context);
    }

    function Natural$1(context) {
      this._context = context;
    }

    Natural$1.prototype = {
      areaStart: function() {
        this._line = 0;
      },
      areaEnd: function() {
        this._line = NaN;
      },
      lineStart: function() {
        this._x = [];
        this._y = [];
      },
      lineEnd: function() {
        var x = this._x,
            y = this._y,
            n = x.length;

        if (n) {
          this._line ? this._context.lineTo(x[0], y[0]) : this._context.moveTo(x[0], y[0]);
          if (n === 2) {
            this._context.lineTo(x[1], y[1]);
          } else {
            var px = controlPoints$1(x),
                py = controlPoints$1(y);
            for (var i0 = 0, i1 = 1; i1 < n; ++i0, ++i1) {
              this._context.bezierCurveTo(px[0][i0], py[0][i0], px[1][i0], py[1][i0], x[i1], y[i1]);
            }
          }
        }

        if (this._line || (this._line !== 0 && n === 1)) this._context.closePath();
        this._line = 1 - this._line;
        this._x = this._y = null;
      },
      point: function(x, y) {
        this._x.push(+x);
        this._y.push(+y);
      }
    };

    // See https://www.particleincell.com/2012/bezier-splines/ for derivation.
    function controlPoints$1(x) {
      var i,
          n = x.length - 1,
          m,
          a = new Array(n),
          b = new Array(n),
          r = new Array(n);
      a[0] = 0, b[0] = 2, r[0] = x[0] + 2 * x[1];
      for (i = 1; i < n - 1; ++i) a[i] = 1, b[i] = 4, r[i] = 4 * x[i] + 2 * x[i + 1];
      a[n - 1] = 2, b[n - 1] = 7, r[n - 1] = 8 * x[n - 1] + x[n];
      for (i = 1; i < n; ++i) m = a[i] / b[i - 1], b[i] -= m, r[i] -= m * r[i - 1];
      a[n - 1] = r[n - 1] / b[n - 1];
      for (i = n - 2; i >= 0; --i) a[i] = (r[i] - a[i + 1]) / b[i];
      b[n - 1] = (x[n] + a[n - 1]) / 2;
      for (i = 0; i < n - 1; ++i) b[i] = 2 * x[i + 1] - a[i + 1];
      return [a, b];
    }

    function natural$1(context) {
      return new Natural$1(context);
    }

    function Step$1(context, t) {
      this._context = context;
      this._t = t;
    }

    Step$1.prototype = {
      areaStart: function() {
        this._line = 0;
      },
      areaEnd: function() {
        this._line = NaN;
      },
      lineStart: function() {
        this._x = this._y = NaN;
        this._point = 0;
      },
      lineEnd: function() {
        if (0 < this._t && this._t < 1 && this._point === 2) this._context.lineTo(this._x, this._y);
        if (this._line || (this._line !== 0 && this._point === 1)) this._context.closePath();
        if (this._line >= 0) this._t = 1 - this._t, this._line = 1 - this._line;
      },
      point: function(x, y) {
        x = +x, y = +y;
        switch (this._point) {
          case 0: this._point = 1; this._line ? this._context.lineTo(x, y) : this._context.moveTo(x, y); break;
          case 1: this._point = 2; // proceed
          default: {
            if (this._t <= 0) {
              this._context.lineTo(this._x, y);
              this._context.lineTo(x, y);
            } else {
              var x1 = this._x * (1 - this._t) + x * this._t;
              this._context.lineTo(x1, this._y);
              this._context.lineTo(x1, y);
            }
            break;
          }
        }
        this._x = x, this._y = y;
      }
    };

    function step$1(context) {
      return new Step$1(context, 0.5);
    }

    function stepBefore$1(context) {
      return new Step$1(context, 0);
    }

    function stepAfter$1(context) {
      return new Step$1(context, 1);
    }

    function none$6(series, order) {
      if (!((n = series.length) > 1)) return;
      for (var i = 1, j, s0, s1 = series[order[0]], n, m = s1.length; i < n; ++i) {
        s0 = s1, s1 = series[order[i]];
        for (j = 0; j < m; ++j) {
          s1[j][1] += s1[j][0] = isNaN(s0[j][1]) ? s0[j][0] : s0[j][1];
        }
      }
    }

    function none$7(series) {
      var n = series.length, o = new Array(n);
      while (--n >= 0) o[n] = n;
      return o;
    }

    function stackValue$1(d, key) {
      return d[key];
    }

    function stack$1() {
      var keys = constant$s([]),
          order = none$7,
          offset = none$6,
          value = stackValue$1;

      function stack(data) {
        var kz = keys.apply(this, arguments),
            i,
            m = data.length,
            n = kz.length,
            sz = new Array(n),
            oz;

        for (i = 0; i < n; ++i) {
          for (var ki = kz[i], si = sz[i] = new Array(m), j = 0, sij; j < m; ++j) {
            si[j] = sij = [0, +value(data[j], ki, j, data)];
            sij.data = data[j];
          }
          si.key = ki;
        }

        for (i = 0, oz = order(sz); i < n; ++i) {
          sz[oz[i]].index = i;
        }

        offset(sz, oz);
        return sz;
      }

      stack.keys = function(_) {
        return arguments.length ? (keys = typeof _ === "function" ? _ : constant$s(slice$d.call(_)), stack) : keys;
      };

      stack.value = function(_) {
        return arguments.length ? (value = typeof _ === "function" ? _ : constant$s(+_), stack) : value;
      };

      stack.order = function(_) {
        return arguments.length ? (order = _ == null ? none$7 : typeof _ === "function" ? _ : constant$s(slice$d.call(_)), stack) : order;
      };

      stack.offset = function(_) {
        return arguments.length ? (offset = _ == null ? none$6 : _, stack) : offset;
      };

      return stack;
    }

    function expand$1(series, order) {
      if (!((n = series.length) > 0)) return;
      for (var i, n, j = 0, m = series[0].length, y; j < m; ++j) {
        for (y = i = 0; i < n; ++i) y += series[i][j][1] || 0;
        if (y) for (i = 0; i < n; ++i) series[i][j][1] /= y;
      }
      none$6(series, order);
    }

    function diverging$2(series, order) {
      if (!((n = series.length) > 1)) return;
      for (var i, j = 0, d, dy, yp, yn, n, m = series[order[0]].length; j < m; ++j) {
        for (yp = yn = 0, i = 0; i < n; ++i) {
          if ((dy = (d = series[order[i]][j])[1] - d[0]) >= 0) {
            d[0] = yp, d[1] = yp += dy;
          } else if (dy < 0) {
            d[1] = yn, d[0] = yn += dy;
          } else {
            d[0] = yp;
          }
        }
      }
    }

    function silhouette$1(series, order) {
      if (!((n = series.length) > 0)) return;
      for (var j = 0, s0 = series[order[0]], n, m = s0.length; j < m; ++j) {
        for (var i = 0, y = 0; i < n; ++i) y += series[i][j][1] || 0;
        s0[j][1] += s0[j][0] = -y / 2;
      }
      none$6(series, order);
    }

    function wiggle$1(series, order) {
      if (!((n = series.length) > 0) || !((m = (s0 = series[order[0]]).length) > 0)) return;
      for (var y = 0, j = 1, s0, m, n; j < m; ++j) {
        for (var i = 0, s1 = 0, s2 = 0; i < n; ++i) {
          var si = series[order[i]],
              sij0 = si[j][1] || 0,
              sij1 = si[j - 1][1] || 0,
              s3 = (sij0 - sij1) / 2;
          for (var k = 0; k < i; ++k) {
            var sk = series[order[k]],
                skj0 = sk[j][1] || 0,
                skj1 = sk[j - 1][1] || 0;
            s3 += skj0 - skj1;
          }
          s1 += sij0, s2 += s3 * sij0;
        }
        s0[j - 1][1] += s0[j - 1][0] = y;
        if (s1) y -= s2 / s1;
      }
      s0[j - 1][1] += s0[j - 1][0] = y;
      none$6(series, order);
    }

    function ascending$b(series) {
      var sums = series.map(sum$5);
      return none$7(series).sort(function(a, b) { return sums[a] - sums[b]; });
    }

    function sum$5(series) {
      var s = 0, i = -1, n = series.length, v;
      while (++i < n) if (v = +series[i][1]) s += v;
      return s;
    }

    function descending$5(series) {
      return ascending$b(series).reverse();
    }

    function insideOut$1(series) {
      var n = series.length,
          i,
          j,
          sums = series.map(sum$5),
          order = none$7(series).sort(function(a, b) { return sums[b] - sums[a]; }),
          top = 0,
          bottom = 0,
          tops = [],
          bottoms = [];

      for (i = 0; i < n; ++i) {
        j = order[i];
        if (top < bottom) {
          top += sums[j];
          tops.push(j);
        } else {
          bottom += sums[j];
          bottoms.push(j);
        }
      }

      return bottoms.reverse().concat(tops);
    }

    function reverse$1(series) {
      return none$7(series).reverse();
    }

    var t0$5 = new Date,
        t1$5 = new Date;

    function newInterval$2(floori, offseti, count, field) {

      function interval(date) {
        return floori(date = new Date(+date)), date;
      }

      interval.floor = interval;

      interval.ceil = function(date) {
        return floori(date = new Date(date - 1)), offseti(date, 1), floori(date), date;
      };

      interval.round = function(date) {
        var d0 = interval(date),
            d1 = interval.ceil(date);
        return date - d0 < d1 - date ? d0 : d1;
      };

      interval.offset = function(date, step) {
        return offseti(date = new Date(+date), step == null ? 1 : Math.floor(step)), date;
      };

      interval.range = function(start, stop, step) {
        var range = [], previous;
        start = interval.ceil(start);
        step = step == null ? 1 : Math.floor(step);
        if (!(start < stop) || !(step > 0)) return range; // also handles Invalid Date
        do range.push(previous = new Date(+start)), offseti(start, step), floori(start);
        while (previous < start && start < stop);
        return range;
      };

      interval.filter = function(test) {
        return newInterval$2(function(date) {
          if (date >= date) while (floori(date), !test(date)) date.setTime(date - 1);
        }, function(date, step) {
          if (date >= date) {
            if (step < 0) while (++step <= 0) {
              while (offseti(date, -1), !test(date)) {} // eslint-disable-line no-empty
            } else while (--step >= 0) {
              while (offseti(date, +1), !test(date)) {} // eslint-disable-line no-empty
            }
          }
        });
      };

      if (count) {
        interval.count = function(start, end) {
          t0$5.setTime(+start), t1$5.setTime(+end);
          floori(t0$5), floori(t1$5);
          return Math.floor(count(t0$5, t1$5));
        };

        interval.every = function(step) {
          step = Math.floor(step);
          return !isFinite(step) || !(step > 0) ? null
              : !(step > 1) ? interval
              : interval.filter(field
                  ? function(d) { return field(d) % step === 0; }
                  : function(d) { return interval.count(0, d) % step === 0; });
        };
      }

      return interval;
    }

    var millisecond$2 = newInterval$2(function() {
      // noop
    }, function(date, step) {
      date.setTime(+date + step);
    }, function(start, end) {
      return end - start;
    });

    // An optimized implementation for this simple case.
    millisecond$2.every = function(k) {
      k = Math.floor(k);
      if (!isFinite(k) || !(k > 0)) return null;
      if (!(k > 1)) return millisecond$2;
      return newInterval$2(function(date) {
        date.setTime(Math.floor(date / k) * k);
      }, function(date, step) {
        date.setTime(+date + step * k);
      }, function(start, end) {
        return (end - start) / k;
      });
    };
    var milliseconds$1 = millisecond$2.range;

    var durationSecond$4 = 1e3;
    var durationMinute$4 = 6e4;
    var durationHour$4 = 36e5;
    var durationDay$4 = 864e5;
    var durationWeek$4 = 6048e5;

    var second$2 = newInterval$2(function(date) {
      date.setTime(Math.floor(date / durationSecond$4) * durationSecond$4);
    }, function(date, step) {
      date.setTime(+date + step * durationSecond$4);
    }, function(start, end) {
      return (end - start) / durationSecond$4;
    }, function(date) {
      return date.getUTCSeconds();
    });
    var seconds$1 = second$2.range;

    var minute$2 = newInterval$2(function(date) {
      date.setTime(Math.floor(date / durationMinute$4) * durationMinute$4);
    }, function(date, step) {
      date.setTime(+date + step * durationMinute$4);
    }, function(start, end) {
      return (end - start) / durationMinute$4;
    }, function(date) {
      return date.getMinutes();
    });
    var minutes$1 = minute$2.range;

    var hour$2 = newInterval$2(function(date) {
      var offset = date.getTimezoneOffset() * durationMinute$4 % durationHour$4;
      if (offset < 0) offset += durationHour$4;
      date.setTime(Math.floor((+date - offset) / durationHour$4) * durationHour$4 + offset);
    }, function(date, step) {
      date.setTime(+date + step * durationHour$4);
    }, function(start, end) {
      return (end - start) / durationHour$4;
    }, function(date) {
      return date.getHours();
    });
    var hours$1 = hour$2.range;

    var day$2 = newInterval$2(function(date) {
      date.setHours(0, 0, 0, 0);
    }, function(date, step) {
      date.setDate(date.getDate() + step);
    }, function(start, end) {
      return (end - start - (end.getTimezoneOffset() - start.getTimezoneOffset()) * durationMinute$4) / durationDay$4;
    }, function(date) {
      return date.getDate() - 1;
    });
    var days$1 = day$2.range;

    function weekday$2(i) {
      return newInterval$2(function(date) {
        date.setDate(date.getDate() - (date.getDay() + 7 - i) % 7);
        date.setHours(0, 0, 0, 0);
      }, function(date, step) {
        date.setDate(date.getDate() + step * 7);
      }, function(start, end) {
        return (end - start - (end.getTimezoneOffset() - start.getTimezoneOffset()) * durationMinute$4) / durationWeek$4;
      });
    }

    var sunday$2 = weekday$2(0);
    var monday$2 = weekday$2(1);
    var tuesday$2 = weekday$2(2);
    var wednesday$2 = weekday$2(3);
    var thursday$2 = weekday$2(4);
    var friday$2 = weekday$2(5);
    var saturday$2 = weekday$2(6);

    var sundays$1 = sunday$2.range;
    var mondays$1 = monday$2.range;
    var tuesdays$1 = tuesday$2.range;
    var wednesdays$1 = wednesday$2.range;
    var thursdays$1 = thursday$2.range;
    var fridays$1 = friday$2.range;
    var saturdays$1 = saturday$2.range;

    var month$2 = newInterval$2(function(date) {
      date.setDate(1);
      date.setHours(0, 0, 0, 0);
    }, function(date, step) {
      date.setMonth(date.getMonth() + step);
    }, function(start, end) {
      return end.getMonth() - start.getMonth() + (end.getFullYear() - start.getFullYear()) * 12;
    }, function(date) {
      return date.getMonth();
    });
    var months$1 = month$2.range;

    var year$2 = newInterval$2(function(date) {
      date.setMonth(0, 1);
      date.setHours(0, 0, 0, 0);
    }, function(date, step) {
      date.setFullYear(date.getFullYear() + step);
    }, function(start, end) {
      return end.getFullYear() - start.getFullYear();
    }, function(date) {
      return date.getFullYear();
    });

    // An optimized implementation for this simple case.
    year$2.every = function(k) {
      return !isFinite(k = Math.floor(k)) || !(k > 0) ? null : newInterval$2(function(date) {
        date.setFullYear(Math.floor(date.getFullYear() / k) * k);
        date.setMonth(0, 1);
        date.setHours(0, 0, 0, 0);
      }, function(date, step) {
        date.setFullYear(date.getFullYear() + step * k);
      });
    };
    var years$1 = year$2.range;

    var utcMinute$2 = newInterval$2(function(date) {
      date.setUTCSeconds(0, 0);
    }, function(date, step) {
      date.setTime(+date + step * durationMinute$4);
    }, function(start, end) {
      return (end - start) / durationMinute$4;
    }, function(date) {
      return date.getUTCMinutes();
    });
    var utcMinutes$1 = utcMinute$2.range;

    var utcHour$2 = newInterval$2(function(date) {
      date.setUTCMinutes(0, 0, 0);
    }, function(date, step) {
      date.setTime(+date + step * durationHour$4);
    }, function(start, end) {
      return (end - start) / durationHour$4;
    }, function(date) {
      return date.getUTCHours();
    });
    var utcHours$1 = utcHour$2.range;

    var utcDay$2 = newInterval$2(function(date) {
      date.setUTCHours(0, 0, 0, 0);
    }, function(date, step) {
      date.setUTCDate(date.getUTCDate() + step);
    }, function(start, end) {
      return (end - start) / durationDay$4;
    }, function(date) {
      return date.getUTCDate() - 1;
    });
    var utcDays$1 = utcDay$2.range;

    function utcWeekday$2(i) {
      return newInterval$2(function(date) {
        date.setUTCDate(date.getUTCDate() - (date.getUTCDay() + 7 - i) % 7);
        date.setUTCHours(0, 0, 0, 0);
      }, function(date, step) {
        date.setUTCDate(date.getUTCDate() + step * 7);
      }, function(start, end) {
        return (end - start) / durationWeek$4;
      });
    }

    var utcSunday$2 = utcWeekday$2(0);
    var utcMonday$2 = utcWeekday$2(1);
    var utcTuesday$2 = utcWeekday$2(2);
    var utcWednesday$2 = utcWeekday$2(3);
    var utcThursday$2 = utcWeekday$2(4);
    var utcFriday$2 = utcWeekday$2(5);
    var utcSaturday$2 = utcWeekday$2(6);

    var utcSundays$1 = utcSunday$2.range;
    var utcMondays$1 = utcMonday$2.range;
    var utcTuesdays$1 = utcTuesday$2.range;
    var utcWednesdays$1 = utcWednesday$2.range;
    var utcThursdays$1 = utcThursday$2.range;
    var utcFridays$1 = utcFriday$2.range;
    var utcSaturdays$1 = utcSaturday$2.range;

    var utcMonth$2 = newInterval$2(function(date) {
      date.setUTCDate(1);
      date.setUTCHours(0, 0, 0, 0);
    }, function(date, step) {
      date.setUTCMonth(date.getUTCMonth() + step);
    }, function(start, end) {
      return end.getUTCMonth() - start.getUTCMonth() + (end.getUTCFullYear() - start.getUTCFullYear()) * 12;
    }, function(date) {
      return date.getUTCMonth();
    });
    var utcMonths$1 = utcMonth$2.range;

    var utcYear$2 = newInterval$2(function(date) {
      date.setUTCMonth(0, 1);
      date.setUTCHours(0, 0, 0, 0);
    }, function(date, step) {
      date.setUTCFullYear(date.getUTCFullYear() + step);
    }, function(start, end) {
      return end.getUTCFullYear() - start.getUTCFullYear();
    }, function(date) {
      return date.getUTCFullYear();
    });

    // An optimized implementation for this simple case.
    utcYear$2.every = function(k) {
      return !isFinite(k = Math.floor(k)) || !(k > 0) ? null : newInterval$2(function(date) {
        date.setUTCFullYear(Math.floor(date.getUTCFullYear() / k) * k);
        date.setUTCMonth(0, 1);
        date.setUTCHours(0, 0, 0, 0);
      }, function(date, step) {
        date.setUTCFullYear(date.getUTCFullYear() + step * k);
      });
    };
    var utcYears$1 = utcYear$2.range;

    var t0$6 = new Date,
        t1$6 = new Date;

    function newInterval$3(floori, offseti, count, field) {

      function interval(date) {
        return floori(date = arguments.length === 0 ? new Date : new Date(+date)), date;
      }

      interval.floor = function(date) {
        return floori(date = new Date(+date)), date;
      };

      interval.ceil = function(date) {
        return floori(date = new Date(date - 1)), offseti(date, 1), floori(date), date;
      };

      interval.round = function(date) {
        var d0 = interval(date),
            d1 = interval.ceil(date);
        return date - d0 < d1 - date ? d0 : d1;
      };

      interval.offset = function(date, step) {
        return offseti(date = new Date(+date), step == null ? 1 : Math.floor(step)), date;
      };

      interval.range = function(start, stop, step) {
        var range = [], previous;
        start = interval.ceil(start);
        step = step == null ? 1 : Math.floor(step);
        if (!(start < stop) || !(step > 0)) return range; // also handles Invalid Date
        do range.push(previous = new Date(+start)), offseti(start, step), floori(start);
        while (previous < start && start < stop);
        return range;
      };

      interval.filter = function(test) {
        return newInterval$3(function(date) {
          if (date >= date) while (floori(date), !test(date)) date.setTime(date - 1);
        }, function(date, step) {
          if (date >= date) {
            if (step < 0) while (++step <= 0) {
              while (offseti(date, -1), !test(date)) {} // eslint-disable-line no-empty
            } else while (--step >= 0) {
              while (offseti(date, +1), !test(date)) {} // eslint-disable-line no-empty
            }
          }
        });
      };

      if (count) {
        interval.count = function(start, end) {
          t0$6.setTime(+start), t1$6.setTime(+end);
          floori(t0$6), floori(t1$6);
          return Math.floor(count(t0$6, t1$6));
        };

        interval.every = function(step) {
          step = Math.floor(step);
          return !isFinite(step) || !(step > 0) ? null
              : !(step > 1) ? interval
              : interval.filter(field
                  ? function(d) { return field(d) % step === 0; }
                  : function(d) { return interval.count(0, d) % step === 0; });
        };
      }

      return interval;
    }

    var durationMinute$5 = 6e4;
    var durationDay$5 = 864e5;
    var durationWeek$5 = 6048e5;

    var day$3 = newInterval$3(function(date) {
      date.setHours(0, 0, 0, 0);
    }, function(date, step) {
      date.setDate(date.getDate() + step);
    }, function(start, end) {
      return (end - start - (end.getTimezoneOffset() - start.getTimezoneOffset()) * durationMinute$5) / durationDay$5;
    }, function(date) {
      return date.getDate() - 1;
    });

    function weekday$3(i) {
      return newInterval$3(function(date) {
        date.setDate(date.getDate() - (date.getDay() + 7 - i) % 7);
        date.setHours(0, 0, 0, 0);
      }, function(date, step) {
        date.setDate(date.getDate() + step * 7);
      }, function(start, end) {
        return (end - start - (end.getTimezoneOffset() - start.getTimezoneOffset()) * durationMinute$5) / durationWeek$5;
      });
    }

    var sunday$3 = weekday$3(0);
    var monday$3 = weekday$3(1);
    var tuesday$3 = weekday$3(2);
    var wednesday$3 = weekday$3(3);
    var thursday$3 = weekday$3(4);
    var friday$3 = weekday$3(5);
    var saturday$3 = weekday$3(6);

    var year$3 = newInterval$3(function(date) {
      date.setMonth(0, 1);
      date.setHours(0, 0, 0, 0);
    }, function(date, step) {
      date.setFullYear(date.getFullYear() + step);
    }, function(start, end) {
      return end.getFullYear() - start.getFullYear();
    }, function(date) {
      return date.getFullYear();
    });

    // An optimized implementation for this simple case.
    year$3.every = function(k) {
      return !isFinite(k = Math.floor(k)) || !(k > 0) ? null : newInterval$3(function(date) {
        date.setFullYear(Math.floor(date.getFullYear() / k) * k);
        date.setMonth(0, 1);
        date.setHours(0, 0, 0, 0);
      }, function(date, step) {
        date.setFullYear(date.getFullYear() + step * k);
      });
    };

    var utcDay$3 = newInterval$3(function(date) {
      date.setUTCHours(0, 0, 0, 0);
    }, function(date, step) {
      date.setUTCDate(date.getUTCDate() + step);
    }, function(start, end) {
      return (end - start) / durationDay$5;
    }, function(date) {
      return date.getUTCDate() - 1;
    });

    function utcWeekday$3(i) {
      return newInterval$3(function(date) {
        date.setUTCDate(date.getUTCDate() - (date.getUTCDay() + 7 - i) % 7);
        date.setUTCHours(0, 0, 0, 0);
      }, function(date, step) {
        date.setUTCDate(date.getUTCDate() + step * 7);
      }, function(start, end) {
        return (end - start) / durationWeek$5;
      });
    }

    var utcSunday$3 = utcWeekday$3(0);
    var utcMonday$3 = utcWeekday$3(1);
    var utcTuesday$3 = utcWeekday$3(2);
    var utcWednesday$3 = utcWeekday$3(3);
    var utcThursday$3 = utcWeekday$3(4);
    var utcFriday$3 = utcWeekday$3(5);
    var utcSaturday$3 = utcWeekday$3(6);

    var utcYear$3 = newInterval$3(function(date) {
      date.setUTCMonth(0, 1);
      date.setUTCHours(0, 0, 0, 0);
    }, function(date, step) {
      date.setUTCFullYear(date.getUTCFullYear() + step);
    }, function(start, end) {
      return end.getUTCFullYear() - start.getUTCFullYear();
    }, function(date) {
      return date.getUTCFullYear();
    });

    // An optimized implementation for this simple case.
    utcYear$3.every = function(k) {
      return !isFinite(k = Math.floor(k)) || !(k > 0) ? null : newInterval$3(function(date) {
        date.setUTCFullYear(Math.floor(date.getUTCFullYear() / k) * k);
        date.setUTCMonth(0, 1);
        date.setUTCHours(0, 0, 0, 0);
      }, function(date, step) {
        date.setUTCFullYear(date.getUTCFullYear() + step * k);
      });
    };

    function localDate$2(d) {
      if (0 <= d.y && d.y < 100) {
        var date = new Date(-1, d.m, d.d, d.H, d.M, d.S, d.L);
        date.setFullYear(d.y);
        return date;
      }
      return new Date(d.y, d.m, d.d, d.H, d.M, d.S, d.L);
    }

    function utcDate$2(d) {
      if (0 <= d.y && d.y < 100) {
        var date = new Date(Date.UTC(-1, d.m, d.d, d.H, d.M, d.S, d.L));
        date.setUTCFullYear(d.y);
        return date;
      }
      return new Date(Date.UTC(d.y, d.m, d.d, d.H, d.M, d.S, d.L));
    }

    function newYear(y) {
      return {y: y, m: 0, d: 1, H: 0, M: 0, S: 0, L: 0};
    }

    function formatLocale$5(locale) {
      var locale_dateTime = locale.dateTime,
          locale_date = locale.date,
          locale_time = locale.time,
          locale_periods = locale.periods,
          locale_weekdays = locale.days,
          locale_shortWeekdays = locale.shortDays,
          locale_months = locale.months,
          locale_shortMonths = locale.shortMonths;

      var periodRe = formatRe$2(locale_periods),
          periodLookup = formatLookup$2(locale_periods),
          weekdayRe = formatRe$2(locale_weekdays),
          weekdayLookup = formatLookup$2(locale_weekdays),
          shortWeekdayRe = formatRe$2(locale_shortWeekdays),
          shortWeekdayLookup = formatLookup$2(locale_shortWeekdays),
          monthRe = formatRe$2(locale_months),
          monthLookup = formatLookup$2(locale_months),
          shortMonthRe = formatRe$2(locale_shortMonths),
          shortMonthLookup = formatLookup$2(locale_shortMonths);

      var formats = {
        "a": formatShortWeekday,
        "A": formatWeekday,
        "b": formatShortMonth,
        "B": formatMonth,
        "c": null,
        "d": formatDayOfMonth$2,
        "e": formatDayOfMonth$2,
        "f": formatMicroseconds$2,
        "H": formatHour24$2,
        "I": formatHour12$2,
        "j": formatDayOfYear$2,
        "L": formatMilliseconds$2,
        "m": formatMonthNumber$2,
        "M": formatMinutes$2,
        "p": formatPeriod,
        "Q": formatUnixTimestamp$2,
        "s": formatUnixTimestampSeconds$2,
        "S": formatSeconds$2,
        "u": formatWeekdayNumberMonday$2,
        "U": formatWeekNumberSunday$2,
        "V": formatWeekNumberISO$2,
        "w": formatWeekdayNumberSunday$2,
        "W": formatWeekNumberMonday$2,
        "x": null,
        "X": null,
        "y": formatYear$3,
        "Y": formatFullYear$2,
        "Z": formatZone$2,
        "%": formatLiteralPercent$2
      };

      var utcFormats = {
        "a": formatUTCShortWeekday,
        "A": formatUTCWeekday,
        "b": formatUTCShortMonth,
        "B": formatUTCMonth,
        "c": null,
        "d": formatUTCDayOfMonth$2,
        "e": formatUTCDayOfMonth$2,
        "f": formatUTCMicroseconds$2,
        "H": formatUTCHour24$2,
        "I": formatUTCHour12$2,
        "j": formatUTCDayOfYear$2,
        "L": formatUTCMilliseconds$2,
        "m": formatUTCMonthNumber$2,
        "M": formatUTCMinutes$2,
        "p": formatUTCPeriod,
        "Q": formatUnixTimestamp$2,
        "s": formatUnixTimestampSeconds$2,
        "S": formatUTCSeconds$2,
        "u": formatUTCWeekdayNumberMonday$2,
        "U": formatUTCWeekNumberSunday$2,
        "V": formatUTCWeekNumberISO$2,
        "w": formatUTCWeekdayNumberSunday$2,
        "W": formatUTCWeekNumberMonday$2,
        "x": null,
        "X": null,
        "y": formatUTCYear$2,
        "Y": formatUTCFullYear$2,
        "Z": formatUTCZone$2,
        "%": formatLiteralPercent$2
      };

      var parses = {
        "a": parseShortWeekday,
        "A": parseWeekday,
        "b": parseShortMonth,
        "B": parseMonth,
        "c": parseLocaleDateTime,
        "d": parseDayOfMonth$2,
        "e": parseDayOfMonth$2,
        "f": parseMicroseconds$2,
        "H": parseHour24$2,
        "I": parseHour24$2,
        "j": parseDayOfYear$2,
        "L": parseMilliseconds$2,
        "m": parseMonthNumber$2,
        "M": parseMinutes$2,
        "p": parsePeriod,
        "Q": parseUnixTimestamp$2,
        "s": parseUnixTimestampSeconds$2,
        "S": parseSeconds$2,
        "u": parseWeekdayNumberMonday$2,
        "U": parseWeekNumberSunday$2,
        "V": parseWeekNumberISO$2,
        "w": parseWeekdayNumberSunday$2,
        "W": parseWeekNumberMonday$2,
        "x": parseLocaleDate,
        "X": parseLocaleTime,
        "y": parseYear$2,
        "Y": parseFullYear$2,
        "Z": parseZone$2,
        "%": parseLiteralPercent$2
      };

      // These recursive directive definitions must be deferred.
      formats.x = newFormat(locale_date, formats);
      formats.X = newFormat(locale_time, formats);
      formats.c = newFormat(locale_dateTime, formats);
      utcFormats.x = newFormat(locale_date, utcFormats);
      utcFormats.X = newFormat(locale_time, utcFormats);
      utcFormats.c = newFormat(locale_dateTime, utcFormats);

      function newFormat(specifier, formats) {
        return function(date) {
          var string = [],
              i = -1,
              j = 0,
              n = specifier.length,
              c,
              pad,
              format;

          if (!(date instanceof Date)) date = new Date(+date);

          while (++i < n) {
            if (specifier.charCodeAt(i) === 37) {
              string.push(specifier.slice(j, i));
              if ((pad = pads$2[c = specifier.charAt(++i)]) != null) c = specifier.charAt(++i);
              else pad = c === "e" ? " " : "0";
              if (format = formats[c]) c = format(date, pad);
              string.push(c);
              j = i + 1;
            }
          }

          string.push(specifier.slice(j, i));
          return string.join("");
        };
      }

      function newParse(specifier, newDate) {
        return function(string) {
          var d = newYear(1900),
              i = parseSpecifier(d, specifier, string += "", 0),
              week, day;
          if (i != string.length) return null;

          // If a UNIX timestamp is specified, return it.
          if ("Q" in d) return new Date(d.Q);

          // The am-pm flag is 0 for AM, and 1 for PM.
          if ("p" in d) d.H = d.H % 12 + d.p * 12;

          // Convert day-of-week and week-of-year to day-of-year.
          if ("V" in d) {
            if (d.V < 1 || d.V > 53) return null;
            if (!("w" in d)) d.w = 1;
            if ("Z" in d) {
              week = utcDate$2(newYear(d.y)), day = week.getUTCDay();
              week = day > 4 || day === 0 ? utcMonday$3.ceil(week) : utcMonday$3(week);
              week = utcDay$3.offset(week, (d.V - 1) * 7);
              d.y = week.getUTCFullYear();
              d.m = week.getUTCMonth();
              d.d = week.getUTCDate() + (d.w + 6) % 7;
            } else {
              week = newDate(newYear(d.y)), day = week.getDay();
              week = day > 4 || day === 0 ? monday$3.ceil(week) : monday$3(week);
              week = day$3.offset(week, (d.V - 1) * 7);
              d.y = week.getFullYear();
              d.m = week.getMonth();
              d.d = week.getDate() + (d.w + 6) % 7;
            }
          } else if ("W" in d || "U" in d) {
            if (!("w" in d)) d.w = "u" in d ? d.u % 7 : "W" in d ? 1 : 0;
            day = "Z" in d ? utcDate$2(newYear(d.y)).getUTCDay() : newDate(newYear(d.y)).getDay();
            d.m = 0;
            d.d = "W" in d ? (d.w + 6) % 7 + d.W * 7 - (day + 5) % 7 : d.w + d.U * 7 - (day + 6) % 7;
          }

          // If a time zone is specified, all fields are interpreted as UTC and then
          // offset according to the specified time zone.
          if ("Z" in d) {
            d.H += d.Z / 100 | 0;
            d.M += d.Z % 100;
            return utcDate$2(d);
          }

          // Otherwise, all fields are in local time.
          return newDate(d);
        };
      }

      function parseSpecifier(d, specifier, string, j) {
        var i = 0,
            n = specifier.length,
            m = string.length,
            c,
            parse;

        while (i < n) {
          if (j >= m) return -1;
          c = specifier.charCodeAt(i++);
          if (c === 37) {
            c = specifier.charAt(i++);
            parse = parses[c in pads$2 ? specifier.charAt(i++) : c];
            if (!parse || ((j = parse(d, string, j)) < 0)) return -1;
          } else if (c != string.charCodeAt(j++)) {
            return -1;
          }
        }

        return j;
      }

      function parsePeriod(d, string, i) {
        var n = periodRe.exec(string.slice(i));
        return n ? (d.p = periodLookup[n[0].toLowerCase()], i + n[0].length) : -1;
      }

      function parseShortWeekday(d, string, i) {
        var n = shortWeekdayRe.exec(string.slice(i));
        return n ? (d.w = shortWeekdayLookup[n[0].toLowerCase()], i + n[0].length) : -1;
      }

      function parseWeekday(d, string, i) {
        var n = weekdayRe.exec(string.slice(i));
        return n ? (d.w = weekdayLookup[n[0].toLowerCase()], i + n[0].length) : -1;
      }

      function parseShortMonth(d, string, i) {
        var n = shortMonthRe.exec(string.slice(i));
        return n ? (d.m = shortMonthLookup[n[0].toLowerCase()], i + n[0].length) : -1;
      }

      function parseMonth(d, string, i) {
        var n = monthRe.exec(string.slice(i));
        return n ? (d.m = monthLookup[n[0].toLowerCase()], i + n[0].length) : -1;
      }

      function parseLocaleDateTime(d, string, i) {
        return parseSpecifier(d, locale_dateTime, string, i);
      }

      function parseLocaleDate(d, string, i) {
        return parseSpecifier(d, locale_date, string, i);
      }

      function parseLocaleTime(d, string, i) {
        return parseSpecifier(d, locale_time, string, i);
      }

      function formatShortWeekday(d) {
        return locale_shortWeekdays[d.getDay()];
      }

      function formatWeekday(d) {
        return locale_weekdays[d.getDay()];
      }

      function formatShortMonth(d) {
        return locale_shortMonths[d.getMonth()];
      }

      function formatMonth(d) {
        return locale_months[d.getMonth()];
      }

      function formatPeriod(d) {
        return locale_periods[+(d.getHours() >= 12)];
      }

      function formatUTCShortWeekday(d) {
        return locale_shortWeekdays[d.getUTCDay()];
      }

      function formatUTCWeekday(d) {
        return locale_weekdays[d.getUTCDay()];
      }

      function formatUTCShortMonth(d) {
        return locale_shortMonths[d.getUTCMonth()];
      }

      function formatUTCMonth(d) {
        return locale_months[d.getUTCMonth()];
      }

      function formatUTCPeriod(d) {
        return locale_periods[+(d.getUTCHours() >= 12)];
      }

      return {
        format: function(specifier) {
          var f = newFormat(specifier += "", formats);
          f.toString = function() { return specifier; };
          return f;
        },
        parse: function(specifier) {
          var p = newParse(specifier += "", localDate$2);
          p.toString = function() { return specifier; };
          return p;
        },
        utcFormat: function(specifier) {
          var f = newFormat(specifier += "", utcFormats);
          f.toString = function() { return specifier; };
          return f;
        },
        utcParse: function(specifier) {
          var p = newParse(specifier, utcDate$2);
          p.toString = function() { return specifier; };
          return p;
        }
      };
    }

    var pads$2 = {"-": "", "_": " ", "0": "0"},
        numberRe$2 = /^\s*\d+/, // note: ignores next directive
        percentRe$2 = /^%/,
        requoteRe$2 = /[\\^$*+?|[\]().{}]/g;

    function pad$3(value, fill, width) {
      var sign = value < 0 ? "-" : "",
          string = (sign ? -value : value) + "",
          length = string.length;
      return sign + (length < width ? new Array(width - length + 1).join(fill) + string : string);
    }

    function requote$2(s) {
      return s.replace(requoteRe$2, "\\$&");
    }

    function formatRe$2(names) {
      return new RegExp("^(?:" + names.map(requote$2).join("|") + ")", "i");
    }

    function formatLookup$2(names) {
      var map = {}, i = -1, n = names.length;
      while (++i < n) map[names[i].toLowerCase()] = i;
      return map;
    }

    function parseWeekdayNumberSunday$2(d, string, i) {
      var n = numberRe$2.exec(string.slice(i, i + 1));
      return n ? (d.w = +n[0], i + n[0].length) : -1;
    }

    function parseWeekdayNumberMonday$2(d, string, i) {
      var n = numberRe$2.exec(string.slice(i, i + 1));
      return n ? (d.u = +n[0], i + n[0].length) : -1;
    }

    function parseWeekNumberSunday$2(d, string, i) {
      var n = numberRe$2.exec(string.slice(i, i + 2));
      return n ? (d.U = +n[0], i + n[0].length) : -1;
    }

    function parseWeekNumberISO$2(d, string, i) {
      var n = numberRe$2.exec(string.slice(i, i + 2));
      return n ? (d.V = +n[0], i + n[0].length) : -1;
    }

    function parseWeekNumberMonday$2(d, string, i) {
      var n = numberRe$2.exec(string.slice(i, i + 2));
      return n ? (d.W = +n[0], i + n[0].length) : -1;
    }

    function parseFullYear$2(d, string, i) {
      var n = numberRe$2.exec(string.slice(i, i + 4));
      return n ? (d.y = +n[0], i + n[0].length) : -1;
    }

    function parseYear$2(d, string, i) {
      var n = numberRe$2.exec(string.slice(i, i + 2));
      return n ? (d.y = +n[0] + (+n[0] > 68 ? 1900 : 2000), i + n[0].length) : -1;
    }

    function parseZone$2(d, string, i) {
      var n = /^(Z)|([+-]\d\d)(?::?(\d\d))?/.exec(string.slice(i, i + 6));
      return n ? (d.Z = n[1] ? 0 : -(n[2] + (n[3] || "00")), i + n[0].length) : -1;
    }

    function parseMonthNumber$2(d, string, i) {
      var n = numberRe$2.exec(string.slice(i, i + 2));
      return n ? (d.m = n[0] - 1, i + n[0].length) : -1;
    }

    function parseDayOfMonth$2(d, string, i) {
      var n = numberRe$2.exec(string.slice(i, i + 2));
      return n ? (d.d = +n[0], i + n[0].length) : -1;
    }

    function parseDayOfYear$2(d, string, i) {
      var n = numberRe$2.exec(string.slice(i, i + 3));
      return n ? (d.m = 0, d.d = +n[0], i + n[0].length) : -1;
    }

    function parseHour24$2(d, string, i) {
      var n = numberRe$2.exec(string.slice(i, i + 2));
      return n ? (d.H = +n[0], i + n[0].length) : -1;
    }

    function parseMinutes$2(d, string, i) {
      var n = numberRe$2.exec(string.slice(i, i + 2));
      return n ? (d.M = +n[0], i + n[0].length) : -1;
    }

    function parseSeconds$2(d, string, i) {
      var n = numberRe$2.exec(string.slice(i, i + 2));
      return n ? (d.S = +n[0], i + n[0].length) : -1;
    }

    function parseMilliseconds$2(d, string, i) {
      var n = numberRe$2.exec(string.slice(i, i + 3));
      return n ? (d.L = +n[0], i + n[0].length) : -1;
    }

    function parseMicroseconds$2(d, string, i) {
      var n = numberRe$2.exec(string.slice(i, i + 6));
      return n ? (d.L = Math.floor(n[0] / 1000), i + n[0].length) : -1;
    }

    function parseLiteralPercent$2(d, string, i) {
      var n = percentRe$2.exec(string.slice(i, i + 1));
      return n ? i + n[0].length : -1;
    }

    function parseUnixTimestamp$2(d, string, i) {
      var n = numberRe$2.exec(string.slice(i));
      return n ? (d.Q = +n[0], i + n[0].length) : -1;
    }

    function parseUnixTimestampSeconds$2(d, string, i) {
      var n = numberRe$2.exec(string.slice(i));
      return n ? (d.Q = (+n[0]) * 1000, i + n[0].length) : -1;
    }

    function formatDayOfMonth$2(d, p) {
      return pad$3(d.getDate(), p, 2);
    }

    function formatHour24$2(d, p) {
      return pad$3(d.getHours(), p, 2);
    }

    function formatHour12$2(d, p) {
      return pad$3(d.getHours() % 12 || 12, p, 2);
    }

    function formatDayOfYear$2(d, p) {
      return pad$3(1 + day$3.count(year$3(d), d), p, 3);
    }

    function formatMilliseconds$2(d, p) {
      return pad$3(d.getMilliseconds(), p, 3);
    }

    function formatMicroseconds$2(d, p) {
      return formatMilliseconds$2(d, p) + "000";
    }

    function formatMonthNumber$2(d, p) {
      return pad$3(d.getMonth() + 1, p, 2);
    }

    function formatMinutes$2(d, p) {
      return pad$3(d.getMinutes(), p, 2);
    }

    function formatSeconds$2(d, p) {
      return pad$3(d.getSeconds(), p, 2);
    }

    function formatWeekdayNumberMonday$2(d) {
      var day = d.getDay();
      return day === 0 ? 7 : day;
    }

    function formatWeekNumberSunday$2(d, p) {
      return pad$3(sunday$3.count(year$3(d), d), p, 2);
    }

    function formatWeekNumberISO$2(d, p) {
      var day = d.getDay();
      d = (day >= 4 || day === 0) ? thursday$3(d) : thursday$3.ceil(d);
      return pad$3(thursday$3.count(year$3(d), d) + (year$3(d).getDay() === 4), p, 2);
    }

    function formatWeekdayNumberSunday$2(d) {
      return d.getDay();
    }

    function formatWeekNumberMonday$2(d, p) {
      return pad$3(monday$3.count(year$3(d), d), p, 2);
    }

    function formatYear$3(d, p) {
      return pad$3(d.getFullYear() % 100, p, 2);
    }

    function formatFullYear$2(d, p) {
      return pad$3(d.getFullYear() % 10000, p, 4);
    }

    function formatZone$2(d) {
      var z = d.getTimezoneOffset();
      return (z > 0 ? "-" : (z *= -1, "+"))
          + pad$3(z / 60 | 0, "0", 2)
          + pad$3(z % 60, "0", 2);
    }

    function formatUTCDayOfMonth$2(d, p) {
      return pad$3(d.getUTCDate(), p, 2);
    }

    function formatUTCHour24$2(d, p) {
      return pad$3(d.getUTCHours(), p, 2);
    }

    function formatUTCHour12$2(d, p) {
      return pad$3(d.getUTCHours() % 12 || 12, p, 2);
    }

    function formatUTCDayOfYear$2(d, p) {
      return pad$3(1 + utcDay$3.count(utcYear$3(d), d), p, 3);
    }

    function formatUTCMilliseconds$2(d, p) {
      return pad$3(d.getUTCMilliseconds(), p, 3);
    }

    function formatUTCMicroseconds$2(d, p) {
      return formatUTCMilliseconds$2(d, p) + "000";
    }

    function formatUTCMonthNumber$2(d, p) {
      return pad$3(d.getUTCMonth() + 1, p, 2);
    }

    function formatUTCMinutes$2(d, p) {
      return pad$3(d.getUTCMinutes(), p, 2);
    }

    function formatUTCSeconds$2(d, p) {
      return pad$3(d.getUTCSeconds(), p, 2);
    }

    function formatUTCWeekdayNumberMonday$2(d) {
      var dow = d.getUTCDay();
      return dow === 0 ? 7 : dow;
    }

    function formatUTCWeekNumberSunday$2(d, p) {
      return pad$3(utcSunday$3.count(utcYear$3(d), d), p, 2);
    }

    function formatUTCWeekNumberISO$2(d, p) {
      var day = d.getUTCDay();
      d = (day >= 4 || day === 0) ? utcThursday$3(d) : utcThursday$3.ceil(d);
      return pad$3(utcThursday$3.count(utcYear$3(d), d) + (utcYear$3(d).getUTCDay() === 4), p, 2);
    }

    function formatUTCWeekdayNumberSunday$2(d) {
      return d.getUTCDay();
    }

    function formatUTCWeekNumberMonday$2(d, p) {
      return pad$3(utcMonday$3.count(utcYear$3(d), d), p, 2);
    }

    function formatUTCYear$2(d, p) {
      return pad$3(d.getUTCFullYear() % 100, p, 2);
    }

    function formatUTCFullYear$2(d, p) {
      return pad$3(d.getUTCFullYear() % 10000, p, 4);
    }

    function formatUTCZone$2() {
      return "+0000";
    }

    function formatLiteralPercent$2() {
      return "%";
    }

    function formatUnixTimestamp$2(d) {
      return +d;
    }

    function formatUnixTimestampSeconds$2(d) {
      return Math.floor(+d / 1000);
    }

    var locale$5;
    var timeFormat$2;
    var timeParse$2;
    var utcFormat$2;
    var utcParse$2;

    defaultLocale$5({
      dateTime: "%x, %X",
      date: "%-m/%-d/%Y",
      time: "%-I:%M:%S %p",
      periods: ["AM", "PM"],
      days: ["Sunday", "Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday"],
      shortDays: ["Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"],
      months: ["January", "February", "March", "April", "May", "June", "July", "August", "September", "October", "November", "December"],
      shortMonths: ["Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"]
    });

    function defaultLocale$5(definition) {
      locale$5 = formatLocale$5(definition);
      timeFormat$2 = locale$5.format;
      timeParse$2 = locale$5.parse;
      utcFormat$2 = locale$5.utcFormat;
      utcParse$2 = locale$5.utcParse;
      return locale$5;
    }

    var isoSpecifier$1 = "%Y-%m-%dT%H:%M:%S.%LZ";

    function formatIsoNative$1(date) {
      return date.toISOString();
    }

    var formatIso$1 = Date.prototype.toISOString
        ? formatIsoNative$1
        : utcFormat$2(isoSpecifier$1);

    function parseIsoNative$1(string) {
      var date = new Date(string);
      return isNaN(date) ? null : date;
    }

    var parseIso$1 = +new Date("2000-01-01T00:00:00.000Z")
        ? parseIsoNative$1
        : utcParse$2(isoSpecifier$1);

    var frame$3 = 0, // is an animation frame pending?
        timeout$5 = 0, // is a timeout pending?
        interval$4 = 0, // are any timers active?
        pokeDelay$3 = 1000, // how frequently we check for clock skew
        taskHead$3,
        taskTail$3,
        clockLast$3 = 0,
        clockNow$3 = 0,
        clockSkew$3 = 0,
        clock$3 = typeof performance === "object" && performance.now ? performance : Date,
        setFrame$3 = typeof window === "object" && window.requestAnimationFrame ? window.requestAnimationFrame.bind(window) : function(f) { setTimeout(f, 17); };

    function now$3() {
      return clockNow$3 || (setFrame$3(clearNow$3), clockNow$3 = clock$3.now() + clockSkew$3);
    }

    function clearNow$3() {
      clockNow$3 = 0;
    }

    function Timer$3() {
      this._call =
      this._time =
      this._next = null;
    }

    Timer$3.prototype = timer$3.prototype = {
      constructor: Timer$3,
      restart: function(callback, delay, time) {
        if (typeof callback !== "function") throw new TypeError("callback is not a function");
        time = (time == null ? now$3() : +time) + (delay == null ? 0 : +delay);
        if (!this._next && taskTail$3 !== this) {
          if (taskTail$3) taskTail$3._next = this;
          else taskHead$3 = this;
          taskTail$3 = this;
        }
        this._call = callback;
        this._time = time;
        sleep$3();
      },
      stop: function() {
        if (this._call) {
          this._call = null;
          this._time = Infinity;
          sleep$3();
        }
      }
    };

    function timer$3(callback, delay, time) {
      var t = new Timer$3;
      t.restart(callback, delay, time);
      return t;
    }

    function timerFlush$3() {
      now$3(); // Get the current time, if not already set.
      ++frame$3; // Pretend we’ve set an alarm, if we haven’t already.
      var t = taskHead$3, e;
      while (t) {
        if ((e = clockNow$3 - t._time) >= 0) t._call.call(null, e);
        t = t._next;
      }
      --frame$3;
    }

    function wake$3() {
      clockNow$3 = (clockLast$3 = clock$3.now()) + clockSkew$3;
      frame$3 = timeout$5 = 0;
      try {
        timerFlush$3();
      } finally {
        frame$3 = 0;
        nap$3();
        clockNow$3 = 0;
      }
    }

    function poke$4() {
      var now = clock$3.now(), delay = now - clockLast$3;
      if (delay > pokeDelay$3) clockSkew$3 -= delay, clockLast$3 = now;
    }

    function nap$3() {
      var t0, t1 = taskHead$3, t2, time = Infinity;
      while (t1) {
        if (t1._call) {
          if (time > t1._time) time = t1._time;
          t0 = t1, t1 = t1._next;
        } else {
          t2 = t1._next, t1._next = null;
          t1 = t0 ? t0._next = t2 : taskHead$3 = t2;
        }
      }
      taskTail$3 = t0;
      sleep$3(time);
    }

    function sleep$3(time) {
      if (frame$3) return; // Soonest alarm already set, or will be.
      if (timeout$5) timeout$5 = clearTimeout(timeout$5);
      var delay = time - clockNow$3; // Strictly less than if we recomputed clockNow.
      if (delay > 24) {
        if (time < Infinity) timeout$5 = setTimeout(wake$3, time - clock$3.now() - clockSkew$3);
        if (interval$4) interval$4 = clearInterval(interval$4);
      } else {
        if (!interval$4) clockLast$3 = clock$3.now(), interval$4 = setInterval(poke$4, pokeDelay$3);
        frame$3 = 1, setFrame$3(wake$3);
      }
    }

    function timeout$6(callback, delay, time) {
      var t = new Timer$3;
      delay = delay == null ? 0 : +delay;
      t.restart(function(elapsed) {
        t.stop();
        callback(elapsed + delay);
      }, delay, time);
      return t;
    }

    function interval$5(callback, delay, time) {
      var t = new Timer$3, total = delay;
      if (delay == null) return t.restart(callback, delay, time), t;
      delay = +delay, time = time == null ? now$3() : +time;
      t.restart(function tick(elapsed) {
        elapsed += total;
        t.restart(tick, total += delay, time);
        callback(elapsed);
      }, delay, time);
      return t;
    }

    var xhtml$4 = "http://www.w3.org/1999/xhtml";

    var namespaces$4 = {
      svg: "http://www.w3.org/2000/svg",
      xhtml: xhtml$4,
      xlink: "http://www.w3.org/1999/xlink",
      xml: "http://www.w3.org/XML/1998/namespace",
      xmlns: "http://www.w3.org/2000/xmlns/"
    };

    function namespace$4(name) {
      var prefix = name += "", i = prefix.indexOf(":");
      if (i >= 0 && (prefix = name.slice(0, i)) !== "xmlns") name = name.slice(i + 1);
      return namespaces$4.hasOwnProperty(prefix) ? {space: namespaces$4[prefix], local: name} : name;
    }

    function creatorInherit$4(name) {
      return function() {
        var document = this.ownerDocument,
            uri = this.namespaceURI;
        return uri === xhtml$4 && document.documentElement.namespaceURI === xhtml$4
            ? document.createElement(name)
            : document.createElementNS(uri, name);
      };
    }

    function creatorFixed$4(fullname) {
      return function() {
        return this.ownerDocument.createElementNS(fullname.space, fullname.local);
      };
    }

    function creator$4(name) {
      var fullname = namespace$4(name);
      return (fullname.local
          ? creatorFixed$4
          : creatorInherit$4)(fullname);
    }

    function none$8() {}

    function selector$4(selector) {
      return selector == null ? none$8 : function() {
        return this.querySelector(selector);
      };
    }

    function selection_select$4(select) {
      if (typeof select !== "function") select = selector$4(select);

      for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, subgroup = subgroups[j] = new Array(n), node, subnode, i = 0; i < n; ++i) {
          if ((node = group[i]) && (subnode = select.call(node, node.__data__, i, group))) {
            if ("__data__" in node) subnode.__data__ = node.__data__;
            subgroup[i] = subnode;
          }
        }
      }

      return new Selection$6(subgroups, this._parents);
    }

    function empty$6() {
      return [];
    }

    function selectorAll$4(selector) {
      return selector == null ? empty$6 : function() {
        return this.querySelectorAll(selector);
      };
    }

    function selection_selectAll$4(select) {
      if (typeof select !== "function") select = selectorAll$4(select);

      for (var groups = this._groups, m = groups.length, subgroups = [], parents = [], j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, node, i = 0; i < n; ++i) {
          if (node = group[i]) {
            subgroups.push(select.call(node, node.__data__, i, group));
            parents.push(node);
          }
        }
      }

      return new Selection$6(subgroups, parents);
    }

    function matcher$5(selector) {
      return function() {
        return this.matches(selector);
      };
    }

    function selection_filter$4(match) {
      if (typeof match !== "function") match = matcher$5(match);

      for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, subgroup = subgroups[j] = [], node, i = 0; i < n; ++i) {
          if ((node = group[i]) && match.call(node, node.__data__, i, group)) {
            subgroup.push(node);
          }
        }
      }

      return new Selection$6(subgroups, this._parents);
    }

    function sparse$4(update) {
      return new Array(update.length);
    }

    function selection_enter$4() {
      return new Selection$6(this._enter || this._groups.map(sparse$4), this._parents);
    }

    function EnterNode$4(parent, datum) {
      this.ownerDocument = parent.ownerDocument;
      this.namespaceURI = parent.namespaceURI;
      this._next = null;
      this._parent = parent;
      this.__data__ = datum;
    }

    EnterNode$4.prototype = {
      constructor: EnterNode$4,
      appendChild: function(child) { return this._parent.insertBefore(child, this._next); },
      insertBefore: function(child, next) { return this._parent.insertBefore(child, next); },
      querySelector: function(selector) { return this._parent.querySelector(selector); },
      querySelectorAll: function(selector) { return this._parent.querySelectorAll(selector); }
    };

    function constant$t(x) {
      return function() {
        return x;
      };
    }

    var keyPrefix$6 = "$"; // Protect against keys like “__proto__”.

    function bindIndex$4(parent, group, enter, update, exit, data) {
      var i = 0,
          node,
          groupLength = group.length,
          dataLength = data.length;

      // Put any non-null nodes that fit into update.
      // Put any null nodes into enter.
      // Put any remaining data into enter.
      for (; i < dataLength; ++i) {
        if (node = group[i]) {
          node.__data__ = data[i];
          update[i] = node;
        } else {
          enter[i] = new EnterNode$4(parent, data[i]);
        }
      }

      // Put any non-null nodes that don’t fit into exit.
      for (; i < groupLength; ++i) {
        if (node = group[i]) {
          exit[i] = node;
        }
      }
    }

    function bindKey$4(parent, group, enter, update, exit, data, key) {
      var i,
          node,
          nodeByKeyValue = {},
          groupLength = group.length,
          dataLength = data.length,
          keyValues = new Array(groupLength),
          keyValue;

      // Compute the key for each node.
      // If multiple nodes have the same key, the duplicates are added to exit.
      for (i = 0; i < groupLength; ++i) {
        if (node = group[i]) {
          keyValues[i] = keyValue = keyPrefix$6 + key.call(node, node.__data__, i, group);
          if (keyValue in nodeByKeyValue) {
            exit[i] = node;
          } else {
            nodeByKeyValue[keyValue] = node;
          }
        }
      }

      // Compute the key for each datum.
      // If there a node associated with this key, join and add it to update.
      // If there is not (or the key is a duplicate), add it to enter.
      for (i = 0; i < dataLength; ++i) {
        keyValue = keyPrefix$6 + key.call(parent, data[i], i, data);
        if (node = nodeByKeyValue[keyValue]) {
          update[i] = node;
          node.__data__ = data[i];
          nodeByKeyValue[keyValue] = null;
        } else {
          enter[i] = new EnterNode$4(parent, data[i]);
        }
      }

      // Add any remaining nodes that were not bound to data to exit.
      for (i = 0; i < groupLength; ++i) {
        if ((node = group[i]) && (nodeByKeyValue[keyValues[i]] === node)) {
          exit[i] = node;
        }
      }
    }

    function selection_data$4(value, key) {
      if (!value) {
        data = new Array(this.size()), j = -1;
        this.each(function(d) { data[++j] = d; });
        return data;
      }

      var bind = key ? bindKey$4 : bindIndex$4,
          parents = this._parents,
          groups = this._groups;

      if (typeof value !== "function") value = constant$t(value);

      for (var m = groups.length, update = new Array(m), enter = new Array(m), exit = new Array(m), j = 0; j < m; ++j) {
        var parent = parents[j],
            group = groups[j],
            groupLength = group.length,
            data = value.call(parent, parent && parent.__data__, j, parents),
            dataLength = data.length,
            enterGroup = enter[j] = new Array(dataLength),
            updateGroup = update[j] = new Array(dataLength),
            exitGroup = exit[j] = new Array(groupLength);

        bind(parent, group, enterGroup, updateGroup, exitGroup, data, key);

        // Now connect the enter nodes to their following update node, such that
        // appendChild can insert the materialized enter node before this node,
        // rather than at the end of the parent node.
        for (var i0 = 0, i1 = 0, previous, next; i0 < dataLength; ++i0) {
          if (previous = enterGroup[i0]) {
            if (i0 >= i1) i1 = i0 + 1;
            while (!(next = updateGroup[i1]) && ++i1 < dataLength);
            previous._next = next || null;
          }
        }
      }

      update = new Selection$6(update, parents);
      update._enter = enter;
      update._exit = exit;
      return update;
    }

    function selection_exit$4() {
      return new Selection$6(this._exit || this._groups.map(sparse$4), this._parents);
    }

    function selection_join$3(onenter, onupdate, onexit) {
      var enter = this.enter(), update = this, exit = this.exit();
      enter = typeof onenter === "function" ? onenter(enter) : enter.append(onenter + "");
      if (onupdate != null) update = onupdate(update);
      if (onexit == null) exit.remove(); else onexit(exit);
      return enter && update ? enter.merge(update).order() : update;
    }

    function selection_merge$4(selection) {

      for (var groups0 = this._groups, groups1 = selection._groups, m0 = groups0.length, m1 = groups1.length, m = Math.min(m0, m1), merges = new Array(m0), j = 0; j < m; ++j) {
        for (var group0 = groups0[j], group1 = groups1[j], n = group0.length, merge = merges[j] = new Array(n), node, i = 0; i < n; ++i) {
          if (node = group0[i] || group1[i]) {
            merge[i] = node;
          }
        }
      }

      for (; j < m0; ++j) {
        merges[j] = groups0[j];
      }

      return new Selection$6(merges, this._parents);
    }

    function selection_order$4() {

      for (var groups = this._groups, j = -1, m = groups.length; ++j < m;) {
        for (var group = groups[j], i = group.length - 1, next = group[i], node; --i >= 0;) {
          if (node = group[i]) {
            if (next && node.compareDocumentPosition(next) ^ 4) next.parentNode.insertBefore(node, next);
            next = node;
          }
        }
      }

      return this;
    }

    function selection_sort$4(compare) {
      if (!compare) compare = ascending$c;

      function compareNode(a, b) {
        return a && b ? compare(a.__data__, b.__data__) : !a - !b;
      }

      for (var groups = this._groups, m = groups.length, sortgroups = new Array(m), j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, sortgroup = sortgroups[j] = new Array(n), node, i = 0; i < n; ++i) {
          if (node = group[i]) {
            sortgroup[i] = node;
          }
        }
        sortgroup.sort(compareNode);
      }

      return new Selection$6(sortgroups, this._parents).order();
    }

    function ascending$c(a, b) {
      return a < b ? -1 : a > b ? 1 : a >= b ? 0 : NaN;
    }

    function selection_call$4() {
      var callback = arguments[0];
      arguments[0] = this;
      callback.apply(null, arguments);
      return this;
    }

    function selection_nodes$4() {
      var nodes = new Array(this.size()), i = -1;
      this.each(function() { nodes[++i] = this; });
      return nodes;
    }

    function selection_node$4() {

      for (var groups = this._groups, j = 0, m = groups.length; j < m; ++j) {
        for (var group = groups[j], i = 0, n = group.length; i < n; ++i) {
          var node = group[i];
          if (node) return node;
        }
      }

      return null;
    }

    function selection_size$4() {
      var size = 0;
      this.each(function() { ++size; });
      return size;
    }

    function selection_empty$4() {
      return !this.node();
    }

    function selection_each$4(callback) {

      for (var groups = this._groups, j = 0, m = groups.length; j < m; ++j) {
        for (var group = groups[j], i = 0, n = group.length, node; i < n; ++i) {
          if (node = group[i]) callback.call(node, node.__data__, i, group);
        }
      }

      return this;
    }

    function attrRemove$6(name) {
      return function() {
        this.removeAttribute(name);
      };
    }

    function attrRemoveNS$6(fullname) {
      return function() {
        this.removeAttributeNS(fullname.space, fullname.local);
      };
    }

    function attrConstant$6(name, value) {
      return function() {
        this.setAttribute(name, value);
      };
    }

    function attrConstantNS$6(fullname, value) {
      return function() {
        this.setAttributeNS(fullname.space, fullname.local, value);
      };
    }

    function attrFunction$6(name, value) {
      return function() {
        var v = value.apply(this, arguments);
        if (v == null) this.removeAttribute(name);
        else this.setAttribute(name, v);
      };
    }

    function attrFunctionNS$6(fullname, value) {
      return function() {
        var v = value.apply(this, arguments);
        if (v == null) this.removeAttributeNS(fullname.space, fullname.local);
        else this.setAttributeNS(fullname.space, fullname.local, v);
      };
    }

    function selection_attr$4(name, value) {
      var fullname = namespace$4(name);

      if (arguments.length < 2) {
        var node = this.node();
        return fullname.local
            ? node.getAttributeNS(fullname.space, fullname.local)
            : node.getAttribute(fullname);
      }

      return this.each((value == null
          ? (fullname.local ? attrRemoveNS$6 : attrRemove$6) : (typeof value === "function"
          ? (fullname.local ? attrFunctionNS$6 : attrFunction$6)
          : (fullname.local ? attrConstantNS$6 : attrConstant$6)))(fullname, value));
    }

    function defaultView$4(node) {
      return (node.ownerDocument && node.ownerDocument.defaultView) // node is a Node
          || (node.document && node) // node is a Window
          || node.defaultView; // node is a Document
    }

    function styleRemove$6(name) {
      return function() {
        this.style.removeProperty(name);
      };
    }

    function styleConstant$6(name, value, priority) {
      return function() {
        this.style.setProperty(name, value, priority);
      };
    }

    function styleFunction$6(name, value, priority) {
      return function() {
        var v = value.apply(this, arguments);
        if (v == null) this.style.removeProperty(name);
        else this.style.setProperty(name, v, priority);
      };
    }

    function selection_style$4(name, value, priority) {
      return arguments.length > 1
          ? this.each((value == null
                ? styleRemove$6 : typeof value === "function"
                ? styleFunction$6
                : styleConstant$6)(name, value, priority == null ? "" : priority))
          : styleValue$4(this.node(), name);
    }

    function styleValue$4(node, name) {
      return node.style.getPropertyValue(name)
          || defaultView$4(node).getComputedStyle(node, null).getPropertyValue(name);
    }

    function propertyRemove$4(name) {
      return function() {
        delete this[name];
      };
    }

    function propertyConstant$4(name, value) {
      return function() {
        this[name] = value;
      };
    }

    function propertyFunction$4(name, value) {
      return function() {
        var v = value.apply(this, arguments);
        if (v == null) delete this[name];
        else this[name] = v;
      };
    }

    function selection_property$4(name, value) {
      return arguments.length > 1
          ? this.each((value == null
              ? propertyRemove$4 : typeof value === "function"
              ? propertyFunction$4
              : propertyConstant$4)(name, value))
          : this.node()[name];
    }

    function classArray$4(string) {
      return string.trim().split(/^|\s+/);
    }

    function classList$4(node) {
      return node.classList || new ClassList$4(node);
    }

    function ClassList$4(node) {
      this._node = node;
      this._names = classArray$4(node.getAttribute("class") || "");
    }

    ClassList$4.prototype = {
      add: function(name) {
        var i = this._names.indexOf(name);
        if (i < 0) {
          this._names.push(name);
          this._node.setAttribute("class", this._names.join(" "));
        }
      },
      remove: function(name) {
        var i = this._names.indexOf(name);
        if (i >= 0) {
          this._names.splice(i, 1);
          this._node.setAttribute("class", this._names.join(" "));
        }
      },
      contains: function(name) {
        return this._names.indexOf(name) >= 0;
      }
    };

    function classedAdd$4(node, names) {
      var list = classList$4(node), i = -1, n = names.length;
      while (++i < n) list.add(names[i]);
    }

    function classedRemove$4(node, names) {
      var list = classList$4(node), i = -1, n = names.length;
      while (++i < n) list.remove(names[i]);
    }

    function classedTrue$4(names) {
      return function() {
        classedAdd$4(this, names);
      };
    }

    function classedFalse$4(names) {
      return function() {
        classedRemove$4(this, names);
      };
    }

    function classedFunction$4(names, value) {
      return function() {
        (value.apply(this, arguments) ? classedAdd$4 : classedRemove$4)(this, names);
      };
    }

    function selection_classed$4(name, value) {
      var names = classArray$4(name + "");

      if (arguments.length < 2) {
        var list = classList$4(this.node()), i = -1, n = names.length;
        while (++i < n) if (!list.contains(names[i])) return false;
        return true;
      }

      return this.each((typeof value === "function"
          ? classedFunction$4 : value
          ? classedTrue$4
          : classedFalse$4)(names, value));
    }

    function textRemove$4() {
      this.textContent = "";
    }

    function textConstant$6(value) {
      return function() {
        this.textContent = value;
      };
    }

    function textFunction$6(value) {
      return function() {
        var v = value.apply(this, arguments);
        this.textContent = v == null ? "" : v;
      };
    }

    function selection_text$4(value) {
      return arguments.length
          ? this.each(value == null
              ? textRemove$4 : (typeof value === "function"
              ? textFunction$6
              : textConstant$6)(value))
          : this.node().textContent;
    }

    function htmlRemove$4() {
      this.innerHTML = "";
    }

    function htmlConstant$4(value) {
      return function() {
        this.innerHTML = value;
      };
    }

    function htmlFunction$4(value) {
      return function() {
        var v = value.apply(this, arguments);
        this.innerHTML = v == null ? "" : v;
      };
    }

    function selection_html$4(value) {
      return arguments.length
          ? this.each(value == null
              ? htmlRemove$4 : (typeof value === "function"
              ? htmlFunction$4
              : htmlConstant$4)(value))
          : this.node().innerHTML;
    }

    function raise$5() {
      if (this.nextSibling) this.parentNode.appendChild(this);
    }

    function selection_raise$4() {
      return this.each(raise$5);
    }

    function lower$4() {
      if (this.previousSibling) this.parentNode.insertBefore(this, this.parentNode.firstChild);
    }

    function selection_lower$4() {
      return this.each(lower$4);
    }

    function selection_append$4(name) {
      var create = typeof name === "function" ? name : creator$4(name);
      return this.select(function() {
        return this.appendChild(create.apply(this, arguments));
      });
    }

    function constantNull$4() {
      return null;
    }

    function selection_insert$4(name, before) {
      var create = typeof name === "function" ? name : creator$4(name),
          select = before == null ? constantNull$4 : typeof before === "function" ? before : selector$4(before);
      return this.select(function() {
        return this.insertBefore(create.apply(this, arguments), select.apply(this, arguments) || null);
      });
    }

    function remove$4() {
      var parent = this.parentNode;
      if (parent) parent.removeChild(this);
    }

    function selection_remove$4() {
      return this.each(remove$4);
    }

    function selection_cloneShallow$4() {
      var clone = this.cloneNode(false), parent = this.parentNode;
      return parent ? parent.insertBefore(clone, this.nextSibling) : clone;
    }

    function selection_cloneDeep$4() {
      var clone = this.cloneNode(true), parent = this.parentNode;
      return parent ? parent.insertBefore(clone, this.nextSibling) : clone;
    }

    function selection_clone$4(deep) {
      return this.select(deep ? selection_cloneDeep$4 : selection_cloneShallow$4);
    }

    function selection_datum$4(value) {
      return arguments.length
          ? this.property("__data__", value)
          : this.node().__data__;
    }

    var filterEvents$4 = {};

    if (typeof document !== "undefined") {
      var element$5 = document.documentElement;
      if (!("onmouseenter" in element$5)) {
        filterEvents$4 = {mouseenter: "mouseover", mouseleave: "mouseout"};
      }
    }

    function filterContextListener$4(listener, index, group) {
      listener = contextListener$4(listener, index, group);
      return function(event) {
        var related = event.relatedTarget;
        if (!related || (related !== this && !(related.compareDocumentPosition(this) & 8))) {
          listener.call(this, event);
        }
      };
    }

    function contextListener$4(listener, index, group) {
      return function(event1) {
        try {
          listener.call(this, this.__data__, index, group);
        } finally {
        }
      };
    }

    function parseTypenames$9(typenames) {
      return typenames.trim().split(/^|\s+/).map(function(t) {
        var name = "", i = t.indexOf(".");
        if (i >= 0) name = t.slice(i + 1), t = t.slice(0, i);
        return {type: t, name: name};
      });
    }

    function onRemove$4(typename) {
      return function() {
        var on = this.__on;
        if (!on) return;
        for (var j = 0, i = -1, m = on.length, o; j < m; ++j) {
          if (o = on[j], (!typename.type || o.type === typename.type) && o.name === typename.name) {
            this.removeEventListener(o.type, o.listener, o.capture);
          } else {
            on[++i] = o;
          }
        }
        if (++i) on.length = i;
        else delete this.__on;
      };
    }

    function onAdd$4(typename, value, capture) {
      var wrap = filterEvents$4.hasOwnProperty(typename.type) ? filterContextListener$4 : contextListener$4;
      return function(d, i, group) {
        var on = this.__on, o, listener = wrap(value, i, group);
        if (on) for (var j = 0, m = on.length; j < m; ++j) {
          if ((o = on[j]).type === typename.type && o.name === typename.name) {
            this.removeEventListener(o.type, o.listener, o.capture);
            this.addEventListener(o.type, o.listener = listener, o.capture = capture);
            o.value = value;
            return;
          }
        }
        this.addEventListener(typename.type, listener, capture);
        o = {type: typename.type, name: typename.name, value: value, listener: listener, capture: capture};
        if (!on) this.__on = [o];
        else on.push(o);
      };
    }

    function selection_on$4(typename, value, capture) {
      var typenames = parseTypenames$9(typename + ""), i, n = typenames.length, t;

      if (arguments.length < 2) {
        var on = this.node().__on;
        if (on) for (var j = 0, m = on.length, o; j < m; ++j) {
          for (i = 0, o = on[j]; i < n; ++i) {
            if ((t = typenames[i]).type === o.type && t.name === o.name) {
              return o.value;
            }
          }
        }
        return;
      }

      on = value ? onAdd$4 : onRemove$4;
      if (capture == null) capture = false;
      for (i = 0; i < n; ++i) this.each(on(typenames[i], value, capture));
      return this;
    }

    function dispatchEvent$4(node, type, params) {
      var window = defaultView$4(node),
          event = window.CustomEvent;

      if (typeof event === "function") {
        event = new event(type, params);
      } else {
        event = window.document.createEvent("Event");
        if (params) event.initEvent(type, params.bubbles, params.cancelable), event.detail = params.detail;
        else event.initEvent(type, false, false);
      }

      node.dispatchEvent(event);
    }

    function dispatchConstant$4(type, params) {
      return function() {
        return dispatchEvent$4(this, type, params);
      };
    }

    function dispatchFunction$4(type, params) {
      return function() {
        return dispatchEvent$4(this, type, params.apply(this, arguments));
      };
    }

    function selection_dispatch$4(type, params) {
      return this.each((typeof params === "function"
          ? dispatchFunction$4
          : dispatchConstant$4)(type, params));
    }

    var root$6 = [null];

    function Selection$6(groups, parents) {
      this._groups = groups;
      this._parents = parents;
    }

    function selection$4() {
      return new Selection$6([[document.documentElement]], root$6);
    }

    Selection$6.prototype = selection$4.prototype = {
      constructor: Selection$6,
      select: selection_select$4,
      selectAll: selection_selectAll$4,
      filter: selection_filter$4,
      data: selection_data$4,
      enter: selection_enter$4,
      exit: selection_exit$4,
      join: selection_join$3,
      merge: selection_merge$4,
      order: selection_order$4,
      sort: selection_sort$4,
      call: selection_call$4,
      nodes: selection_nodes$4,
      node: selection_node$4,
      size: selection_size$4,
      empty: selection_empty$4,
      each: selection_each$4,
      attr: selection_attr$4,
      style: selection_style$4,
      property: selection_property$4,
      classed: selection_classed$4,
      text: selection_text$4,
      html: selection_html$4,
      raise: selection_raise$4,
      lower: selection_lower$4,
      append: selection_append$4,
      insert: selection_insert$4,
      remove: selection_remove$4,
      clone: selection_clone$4,
      datum: selection_datum$4,
      on: selection_on$4,
      dispatch: selection_dispatch$4
    };

    var noop$a = {value: function() {}};

    function dispatch$5() {
      for (var i = 0, n = arguments.length, _ = {}, t; i < n; ++i) {
        if (!(t = arguments[i] + "") || (t in _) || /[\s.]/.test(t)) throw new Error("illegal type: " + t);
        _[t] = [];
      }
      return new Dispatch$5(_);
    }

    function Dispatch$5(_) {
      this._ = _;
    }

    function parseTypenames$a(typenames, types) {
      return typenames.trim().split(/^|\s+/).map(function(t) {
        var name = "", i = t.indexOf(".");
        if (i >= 0) name = t.slice(i + 1), t = t.slice(0, i);
        if (t && !types.hasOwnProperty(t)) throw new Error("unknown type: " + t);
        return {type: t, name: name};
      });
    }

    Dispatch$5.prototype = dispatch$5.prototype = {
      constructor: Dispatch$5,
      on: function(typename, callback) {
        var _ = this._,
            T = parseTypenames$a(typename + "", _),
            t,
            i = -1,
            n = T.length;

        // If no callback was specified, return the callback of the given type and name.
        if (arguments.length < 2) {
          while (++i < n) if ((t = (typename = T[i]).type) && (t = get$8(_[t], typename.name))) return t;
          return;
        }

        // If a type was specified, set the callback for the given type and name.
        // Otherwise, if a null callback was specified, remove callbacks of the given name.
        if (callback != null && typeof callback !== "function") throw new Error("invalid callback: " + callback);
        while (++i < n) {
          if (t = (typename = T[i]).type) _[t] = set$c(_[t], typename.name, callback);
          else if (callback == null) for (t in _) _[t] = set$c(_[t], typename.name, null);
        }

        return this;
      },
      copy: function() {
        var copy = {}, _ = this._;
        for (var t in _) copy[t] = _[t].slice();
        return new Dispatch$5(copy);
      },
      call: function(type, that) {
        if ((n = arguments.length - 2) > 0) for (var args = new Array(n), i = 0, n, t; i < n; ++i) args[i] = arguments[i + 2];
        if (!this._.hasOwnProperty(type)) throw new Error("unknown type: " + type);
        for (t = this._[type], i = 0, n = t.length; i < n; ++i) t[i].value.apply(that, args);
      },
      apply: function(type, that, args) {
        if (!this._.hasOwnProperty(type)) throw new Error("unknown type: " + type);
        for (var t = this._[type], i = 0, n = t.length; i < n; ++i) t[i].value.apply(that, args);
      }
    };

    function get$8(type, name) {
      for (var i = 0, n = type.length, c; i < n; ++i) {
        if ((c = type[i]).name === name) {
          return c.value;
        }
      }
    }

    function set$c(type, name, callback) {
      for (var i = 0, n = type.length; i < n; ++i) {
        if (type[i].name === name) {
          type[i] = noop$a, type = type.slice(0, i).concat(type.slice(i + 1));
          break;
        }
      }
      if (callback != null) type.push({name: name, value: callback});
      return type;
    }

    var frame$4 = 0, // is an animation frame pending?
        timeout$7 = 0, // is a timeout pending?
        interval$6 = 0, // are any timers active?
        pokeDelay$4 = 1000, // how frequently we check for clock skew
        taskHead$4,
        taskTail$4,
        clockLast$4 = 0,
        clockNow$4 = 0,
        clockSkew$4 = 0,
        clock$4 = typeof performance === "object" && performance.now ? performance : Date,
        setFrame$4 = typeof window === "object" && window.requestAnimationFrame ? window.requestAnimationFrame.bind(window) : function(f) { setTimeout(f, 17); };

    function now$4() {
      return clockNow$4 || (setFrame$4(clearNow$4), clockNow$4 = clock$4.now() + clockSkew$4);
    }

    function clearNow$4() {
      clockNow$4 = 0;
    }

    function Timer$4() {
      this._call =
      this._time =
      this._next = null;
    }

    Timer$4.prototype = timer$4.prototype = {
      constructor: Timer$4,
      restart: function(callback, delay, time) {
        if (typeof callback !== "function") throw new TypeError("callback is not a function");
        time = (time == null ? now$4() : +time) + (delay == null ? 0 : +delay);
        if (!this._next && taskTail$4 !== this) {
          if (taskTail$4) taskTail$4._next = this;
          else taskHead$4 = this;
          taskTail$4 = this;
        }
        this._call = callback;
        this._time = time;
        sleep$4();
      },
      stop: function() {
        if (this._call) {
          this._call = null;
          this._time = Infinity;
          sleep$4();
        }
      }
    };

    function timer$4(callback, delay, time) {
      var t = new Timer$4;
      t.restart(callback, delay, time);
      return t;
    }

    function timerFlush$4() {
      now$4(); // Get the current time, if not already set.
      ++frame$4; // Pretend we’ve set an alarm, if we haven’t already.
      var t = taskHead$4, e;
      while (t) {
        if ((e = clockNow$4 - t._time) >= 0) t._call.call(null, e);
        t = t._next;
      }
      --frame$4;
    }

    function wake$4() {
      clockNow$4 = (clockLast$4 = clock$4.now()) + clockSkew$4;
      frame$4 = timeout$7 = 0;
      try {
        timerFlush$4();
      } finally {
        frame$4 = 0;
        nap$4();
        clockNow$4 = 0;
      }
    }

    function poke$5() {
      var now = clock$4.now(), delay = now - clockLast$4;
      if (delay > pokeDelay$4) clockSkew$4 -= delay, clockLast$4 = now;
    }

    function nap$4() {
      var t0, t1 = taskHead$4, t2, time = Infinity;
      while (t1) {
        if (t1._call) {
          if (time > t1._time) time = t1._time;
          t0 = t1, t1 = t1._next;
        } else {
          t2 = t1._next, t1._next = null;
          t1 = t0 ? t0._next = t2 : taskHead$4 = t2;
        }
      }
      taskTail$4 = t0;
      sleep$4(time);
    }

    function sleep$4(time) {
      if (frame$4) return; // Soonest alarm already set, or will be.
      if (timeout$7) timeout$7 = clearTimeout(timeout$7);
      var delay = time - clockNow$4; // Strictly less than if we recomputed clockNow.
      if (delay > 24) {
        if (time < Infinity) timeout$7 = setTimeout(wake$4, time - clock$4.now() - clockSkew$4);
        if (interval$6) interval$6 = clearInterval(interval$6);
      } else {
        if (!interval$6) clockLast$4 = clock$4.now(), interval$6 = setInterval(poke$5, pokeDelay$4);
        frame$4 = 1, setFrame$4(wake$4);
      }
    }

    function timeout$8(callback, delay, time) {
      var t = new Timer$4;
      delay = delay == null ? 0 : +delay;
      t.restart(function(elapsed) {
        t.stop();
        callback(elapsed + delay);
      }, delay, time);
      return t;
    }

    var emptyOn$2 = dispatch$5("start", "end", "interrupt");
    var emptyTween$2 = [];

    var CREATED$2 = 0;
    var SCHEDULED$2 = 1;
    var STARTING$2 = 2;
    var STARTED$2 = 3;
    var RUNNING$2 = 4;
    var ENDING$2 = 5;
    var ENDED$2 = 6;

    function schedule$3(node, name, id, index, group, timing) {
      var schedules = node.__transition;
      if (!schedules) node.__transition = {};
      else if (id in schedules) return;
      create$4(node, id, {
        name: name,
        index: index, // For context during callback.
        group: group, // For context during callback.
        on: emptyOn$2,
        tween: emptyTween$2,
        time: timing.time,
        delay: timing.delay,
        duration: timing.duration,
        ease: timing.ease,
        timer: null,
        state: CREATED$2
      });
    }

    function init$2(node, id) {
      var schedule = get$9(node, id);
      if (schedule.state > CREATED$2) throw new Error("too late; already scheduled");
      return schedule;
    }

    function set$d(node, id) {
      var schedule = get$9(node, id);
      if (schedule.state > STARTING$2) throw new Error("too late; already started");
      return schedule;
    }

    function get$9(node, id) {
      var schedule = node.__transition;
      if (!schedule || !(schedule = schedule[id])) throw new Error("transition not found");
      return schedule;
    }

    function create$4(node, id, self) {
      var schedules = node.__transition,
          tween;

      // Initialize the self timer when the transition is created.
      // Note the actual delay is not known until the first callback!
      schedules[id] = self;
      self.timer = timer$4(schedule, 0, self.time);

      function schedule(elapsed) {
        self.state = SCHEDULED$2;
        self.timer.restart(start, self.delay, self.time);

        // If the elapsed delay is less than our first sleep, start immediately.
        if (self.delay <= elapsed) start(elapsed - self.delay);
      }

      function start(elapsed) {
        var i, j, n, o;

        // If the state is not SCHEDULED, then we previously errored on start.
        if (self.state !== SCHEDULED$2) return stop();

        for (i in schedules) {
          o = schedules[i];
          if (o.name !== self.name) continue;

          // While this element already has a starting transition during this frame,
          // defer starting an interrupting transition until that transition has a
          // chance to tick (and possibly end); see d3/d3-transition#54!
          if (o.state === STARTED$2) return timeout$8(start);

          // Interrupt the active transition, if any.
          // Dispatch the interrupt event.
          if (o.state === RUNNING$2) {
            o.state = ENDED$2;
            o.timer.stop();
            o.on.call("interrupt", node, node.__data__, o.index, o.group);
            delete schedules[i];
          }

          // Cancel any pre-empted transitions. No interrupt event is dispatched
          // because the cancelled transitions never started. Note that this also
          // removes this transition from the pending list!
          else if (+i < id) {
            o.state = ENDED$2;
            o.timer.stop();
            delete schedules[i];
          }
        }

        // Defer the first tick to end of the current frame; see d3/d3#1576.
        // Note the transition may be canceled after start and before the first tick!
        // Note this must be scheduled before the start event; see d3/d3-transition#16!
        // Assuming this is successful, subsequent callbacks go straight to tick.
        timeout$8(function() {
          if (self.state === STARTED$2) {
            self.state = RUNNING$2;
            self.timer.restart(tick, self.delay, self.time);
            tick(elapsed);
          }
        });

        // Dispatch the start event.
        // Note this must be done before the tween are initialized.
        self.state = STARTING$2;
        self.on.call("start", node, node.__data__, self.index, self.group);
        if (self.state !== STARTING$2) return; // interrupted
        self.state = STARTED$2;

        // Initialize the tween, deleting null tween.
        tween = new Array(n = self.tween.length);
        for (i = 0, j = -1; i < n; ++i) {
          if (o = self.tween[i].value.call(node, node.__data__, self.index, self.group)) {
            tween[++j] = o;
          }
        }
        tween.length = j + 1;
      }

      function tick(elapsed) {
        var t = elapsed < self.duration ? self.ease.call(null, elapsed / self.duration) : (self.timer.restart(stop), self.state = ENDING$2, 1),
            i = -1,
            n = tween.length;

        while (++i < n) {
          tween[i].call(null, t);
        }

        // Dispatch the end event.
        if (self.state === ENDING$2) {
          self.on.call("end", node, node.__data__, self.index, self.group);
          stop();
        }
      }

      function stop() {
        self.state = ENDED$2;
        self.timer.stop();
        delete schedules[id];
        for (var i in schedules) return; // eslint-disable-line no-unused-vars
        delete node.__transition;
      }
    }

    function interrupt$2(node, name) {
      var schedules = node.__transition,
          schedule,
          active,
          empty = true,
          i;

      if (!schedules) return;

      name = name == null ? null : name + "";

      for (i in schedules) {
        if ((schedule = schedules[i]).name !== name) { empty = false; continue; }
        active = schedule.state > STARTING$2 && schedule.state < ENDING$2;
        schedule.state = ENDED$2;
        schedule.timer.stop();
        if (active) schedule.on.call("interrupt", node, node.__data__, schedule.index, schedule.group);
        delete schedules[i];
      }

      if (empty) delete node.__transition;
    }

    function selection_interrupt$2(name) {
      return this.each(function() {
        interrupt$2(this, name);
      });
    }

    function define$5(constructor, factory, prototype) {
      constructor.prototype = factory.prototype = prototype;
      prototype.constructor = constructor;
    }

    function extend$5(parent, definition) {
      var prototype = Object.create(parent.prototype);
      for (var key in definition) prototype[key] = definition[key];
      return prototype;
    }

    function Color$5() {}

    var darker$5 = 0.7;
    var brighter$5 = 1 / darker$5;

    var reI$5 = "\\s*([+-]?\\d+)\\s*",
        reN$5 = "\\s*([+-]?\\d*\\.?\\d+(?:[eE][+-]?\\d+)?)\\s*",
        reP$5 = "\\s*([+-]?\\d*\\.?\\d+(?:[eE][+-]?\\d+)?)%\\s*",
        reHex$4 = /^#([0-9a-f]{3,8})$/,
        reRgbInteger$5 = new RegExp("^rgb\\(" + [reI$5, reI$5, reI$5] + "\\)$"),
        reRgbPercent$5 = new RegExp("^rgb\\(" + [reP$5, reP$5, reP$5] + "\\)$"),
        reRgbaInteger$5 = new RegExp("^rgba\\(" + [reI$5, reI$5, reI$5, reN$5] + "\\)$"),
        reRgbaPercent$5 = new RegExp("^rgba\\(" + [reP$5, reP$5, reP$5, reN$5] + "\\)$"),
        reHslPercent$5 = new RegExp("^hsl\\(" + [reN$5, reP$5, reP$5] + "\\)$"),
        reHslaPercent$5 = new RegExp("^hsla\\(" + [reN$5, reP$5, reP$5, reN$5] + "\\)$");

    var named$5 = {
      aliceblue: 0xf0f8ff,
      antiquewhite: 0xfaebd7,
      aqua: 0x00ffff,
      aquamarine: 0x7fffd4,
      azure: 0xf0ffff,
      beige: 0xf5f5dc,
      bisque: 0xffe4c4,
      black: 0x000000,
      blanchedalmond: 0xffebcd,
      blue: 0x0000ff,
      blueviolet: 0x8a2be2,
      brown: 0xa52a2a,
      burlywood: 0xdeb887,
      cadetblue: 0x5f9ea0,
      chartreuse: 0x7fff00,
      chocolate: 0xd2691e,
      coral: 0xff7f50,
      cornflowerblue: 0x6495ed,
      cornsilk: 0xfff8dc,
      crimson: 0xdc143c,
      cyan: 0x00ffff,
      darkblue: 0x00008b,
      darkcyan: 0x008b8b,
      darkgoldenrod: 0xb8860b,
      darkgray: 0xa9a9a9,
      darkgreen: 0x006400,
      darkgrey: 0xa9a9a9,
      darkkhaki: 0xbdb76b,
      darkmagenta: 0x8b008b,
      darkolivegreen: 0x556b2f,
      darkorange: 0xff8c00,
      darkorchid: 0x9932cc,
      darkred: 0x8b0000,
      darksalmon: 0xe9967a,
      darkseagreen: 0x8fbc8f,
      darkslateblue: 0x483d8b,
      darkslategray: 0x2f4f4f,
      darkslategrey: 0x2f4f4f,
      darkturquoise: 0x00ced1,
      darkviolet: 0x9400d3,
      deeppink: 0xff1493,
      deepskyblue: 0x00bfff,
      dimgray: 0x696969,
      dimgrey: 0x696969,
      dodgerblue: 0x1e90ff,
      firebrick: 0xb22222,
      floralwhite: 0xfffaf0,
      forestgreen: 0x228b22,
      fuchsia: 0xff00ff,
      gainsboro: 0xdcdcdc,
      ghostwhite: 0xf8f8ff,
      gold: 0xffd700,
      goldenrod: 0xdaa520,
      gray: 0x808080,
      green: 0x008000,
      greenyellow: 0xadff2f,
      grey: 0x808080,
      honeydew: 0xf0fff0,
      hotpink: 0xff69b4,
      indianred: 0xcd5c5c,
      indigo: 0x4b0082,
      ivory: 0xfffff0,
      khaki: 0xf0e68c,
      lavender: 0xe6e6fa,
      lavenderblush: 0xfff0f5,
      lawngreen: 0x7cfc00,
      lemonchiffon: 0xfffacd,
      lightblue: 0xadd8e6,
      lightcoral: 0xf08080,
      lightcyan: 0xe0ffff,
      lightgoldenrodyellow: 0xfafad2,
      lightgray: 0xd3d3d3,
      lightgreen: 0x90ee90,
      lightgrey: 0xd3d3d3,
      lightpink: 0xffb6c1,
      lightsalmon: 0xffa07a,
      lightseagreen: 0x20b2aa,
      lightskyblue: 0x87cefa,
      lightslategray: 0x778899,
      lightslategrey: 0x778899,
      lightsteelblue: 0xb0c4de,
      lightyellow: 0xffffe0,
      lime: 0x00ff00,
      limegreen: 0x32cd32,
      linen: 0xfaf0e6,
      magenta: 0xff00ff,
      maroon: 0x800000,
      mediumaquamarine: 0x66cdaa,
      mediumblue: 0x0000cd,
      mediumorchid: 0xba55d3,
      mediumpurple: 0x9370db,
      mediumseagreen: 0x3cb371,
      mediumslateblue: 0x7b68ee,
      mediumspringgreen: 0x00fa9a,
      mediumturquoise: 0x48d1cc,
      mediumvioletred: 0xc71585,
      midnightblue: 0x191970,
      mintcream: 0xf5fffa,
      mistyrose: 0xffe4e1,
      moccasin: 0xffe4b5,
      navajowhite: 0xffdead,
      navy: 0x000080,
      oldlace: 0xfdf5e6,
      olive: 0x808000,
      olivedrab: 0x6b8e23,
      orange: 0xffa500,
      orangered: 0xff4500,
      orchid: 0xda70d6,
      palegoldenrod: 0xeee8aa,
      palegreen: 0x98fb98,
      paleturquoise: 0xafeeee,
      palevioletred: 0xdb7093,
      papayawhip: 0xffefd5,
      peachpuff: 0xffdab9,
      peru: 0xcd853f,
      pink: 0xffc0cb,
      plum: 0xdda0dd,
      powderblue: 0xb0e0e6,
      purple: 0x800080,
      rebeccapurple: 0x663399,
      red: 0xff0000,
      rosybrown: 0xbc8f8f,
      royalblue: 0x4169e1,
      saddlebrown: 0x8b4513,
      salmon: 0xfa8072,
      sandybrown: 0xf4a460,
      seagreen: 0x2e8b57,
      seashell: 0xfff5ee,
      sienna: 0xa0522d,
      silver: 0xc0c0c0,
      skyblue: 0x87ceeb,
      slateblue: 0x6a5acd,
      slategray: 0x708090,
      slategrey: 0x708090,
      snow: 0xfffafa,
      springgreen: 0x00ff7f,
      steelblue: 0x4682b4,
      tan: 0xd2b48c,
      teal: 0x008080,
      thistle: 0xd8bfd8,
      tomato: 0xff6347,
      turquoise: 0x40e0d0,
      violet: 0xee82ee,
      wheat: 0xf5deb3,
      white: 0xffffff,
      whitesmoke: 0xf5f5f5,
      yellow: 0xffff00,
      yellowgreen: 0x9acd32
    };

    define$5(Color$5, color$5, {
      copy: function(channels) {
        return Object.assign(new this.constructor, this, channels);
      },
      displayable: function() {
        return this.rgb().displayable();
      },
      hex: color_formatHex$4, // Deprecated! Use color.formatHex.
      formatHex: color_formatHex$4,
      formatHsl: color_formatHsl$4,
      formatRgb: color_formatRgb$4,
      toString: color_formatRgb$4
    });

    function color_formatHex$4() {
      return this.rgb().formatHex();
    }

    function color_formatHsl$4() {
      return hslConvert$5(this).formatHsl();
    }

    function color_formatRgb$4() {
      return this.rgb().formatRgb();
    }

    function color$5(format) {
      var m, l;
      format = (format + "").trim().toLowerCase();
      return (m = reHex$4.exec(format)) ? (l = m[1].length, m = parseInt(m[1], 16), l === 6 ? rgbn$5(m) // #ff0000
          : l === 3 ? new Rgb$5((m >> 8 & 0xf) | (m >> 4 & 0xf0), (m >> 4 & 0xf) | (m & 0xf0), ((m & 0xf) << 4) | (m & 0xf), 1) // #f00
          : l === 8 ? rgba$5(m >> 24 & 0xff, m >> 16 & 0xff, m >> 8 & 0xff, (m & 0xff) / 0xff) // #ff000000
          : l === 4 ? rgba$5((m >> 12 & 0xf) | (m >> 8 & 0xf0), (m >> 8 & 0xf) | (m >> 4 & 0xf0), (m >> 4 & 0xf) | (m & 0xf0), (((m & 0xf) << 4) | (m & 0xf)) / 0xff) // #f000
          : null) // invalid hex
          : (m = reRgbInteger$5.exec(format)) ? new Rgb$5(m[1], m[2], m[3], 1) // rgb(255, 0, 0)
          : (m = reRgbPercent$5.exec(format)) ? new Rgb$5(m[1] * 255 / 100, m[2] * 255 / 100, m[3] * 255 / 100, 1) // rgb(100%, 0%, 0%)
          : (m = reRgbaInteger$5.exec(format)) ? rgba$5(m[1], m[2], m[3], m[4]) // rgba(255, 0, 0, 1)
          : (m = reRgbaPercent$5.exec(format)) ? rgba$5(m[1] * 255 / 100, m[2] * 255 / 100, m[3] * 255 / 100, m[4]) // rgb(100%, 0%, 0%, 1)
          : (m = reHslPercent$5.exec(format)) ? hsla$5(m[1], m[2] / 100, m[3] / 100, 1) // hsl(120, 50%, 50%)
          : (m = reHslaPercent$5.exec(format)) ? hsla$5(m[1], m[2] / 100, m[3] / 100, m[4]) // hsla(120, 50%, 50%, 1)
          : named$5.hasOwnProperty(format) ? rgbn$5(named$5[format]) // eslint-disable-line no-prototype-builtins
          : format === "transparent" ? new Rgb$5(NaN, NaN, NaN, 0)
          : null;
    }

    function rgbn$5(n) {
      return new Rgb$5(n >> 16 & 0xff, n >> 8 & 0xff, n & 0xff, 1);
    }

    function rgba$5(r, g, b, a) {
      if (a <= 0) r = g = b = NaN;
      return new Rgb$5(r, g, b, a);
    }

    function rgbConvert$5(o) {
      if (!(o instanceof Color$5)) o = color$5(o);
      if (!o) return new Rgb$5;
      o = o.rgb();
      return new Rgb$5(o.r, o.g, o.b, o.opacity);
    }

    function rgb$7(r, g, b, opacity) {
      return arguments.length === 1 ? rgbConvert$5(r) : new Rgb$5(r, g, b, opacity == null ? 1 : opacity);
    }

    function Rgb$5(r, g, b, opacity) {
      this.r = +r;
      this.g = +g;
      this.b = +b;
      this.opacity = +opacity;
    }

    define$5(Rgb$5, rgb$7, extend$5(Color$5, {
      brighter: function(k) {
        k = k == null ? brighter$5 : Math.pow(brighter$5, k);
        return new Rgb$5(this.r * k, this.g * k, this.b * k, this.opacity);
      },
      darker: function(k) {
        k = k == null ? darker$5 : Math.pow(darker$5, k);
        return new Rgb$5(this.r * k, this.g * k, this.b * k, this.opacity);
      },
      rgb: function() {
        return this;
      },
      displayable: function() {
        return (-0.5 <= this.r && this.r < 255.5)
            && (-0.5 <= this.g && this.g < 255.5)
            && (-0.5 <= this.b && this.b < 255.5)
            && (0 <= this.opacity && this.opacity <= 1);
      },
      hex: rgb_formatHex$4, // Deprecated! Use color.formatHex.
      formatHex: rgb_formatHex$4,
      formatRgb: rgb_formatRgb$4,
      toString: rgb_formatRgb$4
    }));

    function rgb_formatHex$4() {
      return "#" + hex$4(this.r) + hex$4(this.g) + hex$4(this.b);
    }

    function rgb_formatRgb$4() {
      var a = this.opacity; a = isNaN(a) ? 1 : Math.max(0, Math.min(1, a));
      return (a === 1 ? "rgb(" : "rgba(")
          + Math.max(0, Math.min(255, Math.round(this.r) || 0)) + ", "
          + Math.max(0, Math.min(255, Math.round(this.g) || 0)) + ", "
          + Math.max(0, Math.min(255, Math.round(this.b) || 0))
          + (a === 1 ? ")" : ", " + a + ")");
    }

    function hex$4(value) {
      value = Math.max(0, Math.min(255, Math.round(value) || 0));
      return (value < 16 ? "0" : "") + value.toString(16);
    }

    function hsla$5(h, s, l, a) {
      if (a <= 0) h = s = l = NaN;
      else if (l <= 0 || l >= 1) h = s = NaN;
      else if (s <= 0) h = NaN;
      return new Hsl$5(h, s, l, a);
    }

    function hslConvert$5(o) {
      if (o instanceof Hsl$5) return new Hsl$5(o.h, o.s, o.l, o.opacity);
      if (!(o instanceof Color$5)) o = color$5(o);
      if (!o) return new Hsl$5;
      if (o instanceof Hsl$5) return o;
      o = o.rgb();
      var r = o.r / 255,
          g = o.g / 255,
          b = o.b / 255,
          min = Math.min(r, g, b),
          max = Math.max(r, g, b),
          h = NaN,
          s = max - min,
          l = (max + min) / 2;
      if (s) {
        if (r === max) h = (g - b) / s + (g < b) * 6;
        else if (g === max) h = (b - r) / s + 2;
        else h = (r - g) / s + 4;
        s /= l < 0.5 ? max + min : 2 - max - min;
        h *= 60;
      } else {
        s = l > 0 && l < 1 ? 0 : h;
      }
      return new Hsl$5(h, s, l, o.opacity);
    }

    function hsl$9(h, s, l, opacity) {
      return arguments.length === 1 ? hslConvert$5(h) : new Hsl$5(h, s, l, opacity == null ? 1 : opacity);
    }

    function Hsl$5(h, s, l, opacity) {
      this.h = +h;
      this.s = +s;
      this.l = +l;
      this.opacity = +opacity;
    }

    define$5(Hsl$5, hsl$9, extend$5(Color$5, {
      brighter: function(k) {
        k = k == null ? brighter$5 : Math.pow(brighter$5, k);
        return new Hsl$5(this.h, this.s, this.l * k, this.opacity);
      },
      darker: function(k) {
        k = k == null ? darker$5 : Math.pow(darker$5, k);
        return new Hsl$5(this.h, this.s, this.l * k, this.opacity);
      },
      rgb: function() {
        var h = this.h % 360 + (this.h < 0) * 360,
            s = isNaN(h) || isNaN(this.s) ? 0 : this.s,
            l = this.l,
            m2 = l + (l < 0.5 ? l : 1 - l) * s,
            m1 = 2 * l - m2;
        return new Rgb$5(
          hsl2rgb$5(h >= 240 ? h - 240 : h + 120, m1, m2),
          hsl2rgb$5(h, m1, m2),
          hsl2rgb$5(h < 120 ? h + 240 : h - 120, m1, m2),
          this.opacity
        );
      },
      displayable: function() {
        return (0 <= this.s && this.s <= 1 || isNaN(this.s))
            && (0 <= this.l && this.l <= 1)
            && (0 <= this.opacity && this.opacity <= 1);
      },
      formatHsl: function() {
        var a = this.opacity; a = isNaN(a) ? 1 : Math.max(0, Math.min(1, a));
        return (a === 1 ? "hsl(" : "hsla(")
            + (this.h || 0) + ", "
            + (this.s || 0) * 100 + "%, "
            + (this.l || 0) * 100 + "%"
            + (a === 1 ? ")" : ", " + a + ")");
      }
    }));

    /* From FvD 13.37, CSS Color Module Level 3 */
    function hsl2rgb$5(h, m1, m2) {
      return (h < 60 ? m1 + (m2 - m1) * h / 60
          : h < 180 ? m2
          : h < 240 ? m1 + (m2 - m1) * (240 - h) / 60
          : m1) * 255;
    }

    function constant$u(x) {
      return function() {
        return x;
      };
    }

    function linear$8(a, d) {
      return function(t) {
        return a + t * d;
      };
    }

    function exponential$6(a, b, y) {
      return a = Math.pow(a, y), b = Math.pow(b, y) - a, y = 1 / y, function(t) {
        return Math.pow(a + t * b, y);
      };
    }

    function gamma$4(y) {
      return (y = +y) === 1 ? nogamma$4 : function(a, b) {
        return b - a ? exponential$6(a, b, y) : constant$u(isNaN(a) ? b : a);
      };
    }

    function nogamma$4(a, b) {
      var d = b - a;
      return d ? linear$8(a, d) : constant$u(isNaN(a) ? b : a);
    }

    var interpolateRgb$2 = (function rgbGamma(y) {
      var color = gamma$4(y);

      function rgb(start, end) {
        var r = color((start = rgb$7(start)).r, (end = rgb$7(end)).r),
            g = color(start.g, end.g),
            b = color(start.b, end.b),
            opacity = nogamma$4(start.opacity, end.opacity);
        return function(t) {
          start.r = r(t);
          start.g = g(t);
          start.b = b(t);
          start.opacity = opacity(t);
          return start + "";
        };
      }

      rgb.gamma = rgbGamma;

      return rgb;
    })(1);

    function interpolateNumber$2(a, b) {
      return a = +a, b = +b, function(t) {
        return a * (1 - t) + b * t;
      };
    }

    var reA$4 = /[-+]?(?:\d+\.?\d*|\.?\d+)(?:[eE][-+]?\d+)?/g,
        reB$4 = new RegExp(reA$4.source, "g");

    function zero$4(b) {
      return function() {
        return b;
      };
    }

    function one$4(b) {
      return function(t) {
        return b(t) + "";
      };
    }

    function interpolateString$2(a, b) {
      var bi = reA$4.lastIndex = reB$4.lastIndex = 0, // scan index for next number in b
          am, // current match in a
          bm, // current match in b
          bs, // string preceding current number in b, if any
          i = -1, // index in s
          s = [], // string constants and placeholders
          q = []; // number interpolators

      // Coerce inputs to strings.
      a = a + "", b = b + "";

      // Interpolate pairs of numbers in a & b.
      while ((am = reA$4.exec(a))
          && (bm = reB$4.exec(b))) {
        if ((bs = bm.index) > bi) { // a string precedes the next number in b
          bs = b.slice(bi, bs);
          if (s[i]) s[i] += bs; // coalesce with previous string
          else s[++i] = bs;
        }
        if ((am = am[0]) === (bm = bm[0])) { // numbers in a & b match
          if (s[i]) s[i] += bm; // coalesce with previous string
          else s[++i] = bm;
        } else { // interpolate non-matching numbers
          s[++i] = null;
          q.push({i: i, x: interpolateNumber$2(am, bm)});
        }
        bi = reB$4.lastIndex;
      }

      // Add remains of b.
      if (bi < b.length) {
        bs = b.slice(bi);
        if (s[i]) s[i] += bs; // coalesce with previous string
        else s[++i] = bs;
      }

      // Special optimization for only a single match.
      // Otherwise, interpolate each of the numbers and rejoin the string.
      return s.length < 2 ? (q[0]
          ? one$4(q[0].x)
          : zero$4(b))
          : (b = q.length, function(t) {
              for (var i = 0, o; i < b; ++i) s[(o = q[i]).i] = o.x(t);
              return s.join("");
            });
    }

    var degrees$5 = 180 / Math.PI;

    var identity$k = {
      translateX: 0,
      translateY: 0,
      rotate: 0,
      skewX: 0,
      scaleX: 1,
      scaleY: 1
    };

    function decompose$3(a, b, c, d, e, f) {
      var scaleX, scaleY, skewX;
      if (scaleX = Math.sqrt(a * a + b * b)) a /= scaleX, b /= scaleX;
      if (skewX = a * c + b * d) c -= a * skewX, d -= b * skewX;
      if (scaleY = Math.sqrt(c * c + d * d)) c /= scaleY, d /= scaleY, skewX /= scaleY;
      if (a * d < b * c) a = -a, b = -b, skewX = -skewX, scaleX = -scaleX;
      return {
        translateX: e,
        translateY: f,
        rotate: Math.atan2(b, a) * degrees$5,
        skewX: Math.atan(skewX) * degrees$5,
        scaleX: scaleX,
        scaleY: scaleY
      };
    }

    var cssNode$3,
        cssRoot$3,
        cssView$3,
        svgNode$3;

    function parseCss$4(value) {
      if (value === "none") return identity$k;
      if (!cssNode$3) cssNode$3 = document.createElement("DIV"), cssRoot$3 = document.documentElement, cssView$3 = document.defaultView;
      cssNode$3.style.transform = value;
      value = cssView$3.getComputedStyle(cssRoot$3.appendChild(cssNode$3), null).getPropertyValue("transform");
      cssRoot$3.removeChild(cssNode$3);
      value = value.slice(7, -1).split(",");
      return decompose$3(+value[0], +value[1], +value[2], +value[3], +value[4], +value[5]);
    }

    function parseSvg$3(value) {
      if (value == null) return identity$k;
      if (!svgNode$3) svgNode$3 = document.createElementNS("http://www.w3.org/2000/svg", "g");
      svgNode$3.setAttribute("transform", value);
      if (!(value = svgNode$3.transform.baseVal.consolidate())) return identity$k;
      value = value.matrix;
      return decompose$3(value.a, value.b, value.c, value.d, value.e, value.f);
    }

    function interpolateTransform$3(parse, pxComma, pxParen, degParen) {

      function pop(s) {
        return s.length ? s.pop() + " " : "";
      }

      function translate(xa, ya, xb, yb, s, q) {
        if (xa !== xb || ya !== yb) {
          var i = s.push("translate(", null, pxComma, null, pxParen);
          q.push({i: i - 4, x: interpolateNumber$2(xa, xb)}, {i: i - 2, x: interpolateNumber$2(ya, yb)});
        } else if (xb || yb) {
          s.push("translate(" + xb + pxComma + yb + pxParen);
        }
      }

      function rotate(a, b, s, q) {
        if (a !== b) {
          if (a - b > 180) b += 360; else if (b - a > 180) a += 360; // shortest path
          q.push({i: s.push(pop(s) + "rotate(", null, degParen) - 2, x: interpolateNumber$2(a, b)});
        } else if (b) {
          s.push(pop(s) + "rotate(" + b + degParen);
        }
      }

      function skewX(a, b, s, q) {
        if (a !== b) {
          q.push({i: s.push(pop(s) + "skewX(", null, degParen) - 2, x: interpolateNumber$2(a, b)});
        } else if (b) {
          s.push(pop(s) + "skewX(" + b + degParen);
        }
      }

      function scale(xa, ya, xb, yb, s, q) {
        if (xa !== xb || ya !== yb) {
          var i = s.push(pop(s) + "scale(", null, ",", null, ")");
          q.push({i: i - 4, x: interpolateNumber$2(xa, xb)}, {i: i - 2, x: interpolateNumber$2(ya, yb)});
        } else if (xb !== 1 || yb !== 1) {
          s.push(pop(s) + "scale(" + xb + "," + yb + ")");
        }
      }

      return function(a, b) {
        var s = [], // string constants and placeholders
            q = []; // number interpolators
        a = parse(a), b = parse(b);
        translate(a.translateX, a.translateY, b.translateX, b.translateY, s, q);
        rotate(a.rotate, b.rotate, s, q);
        skewX(a.skewX, b.skewX, s, q);
        scale(a.scaleX, a.scaleY, b.scaleX, b.scaleY, s, q);
        a = b = null; // gc
        return function(t) {
          var i = -1, n = q.length, o;
          while (++i < n) s[(o = q[i]).i] = o.x(t);
          return s.join("");
        };
      };
    }

    var interpolateTransformCss$3 = interpolateTransform$3(parseCss$4, "px, ", "px)", "deg)");
    var interpolateTransformSvg$3 = interpolateTransform$3(parseSvg$3, ", ", ")", ")");

    function tweenRemove$2(id, name) {
      var tween0, tween1;
      return function() {
        var schedule = set$d(this, id),
            tween = schedule.tween;

        // If this node shared tween with the previous node,
        // just assign the updated shared tween and we’re done!
        // Otherwise, copy-on-write.
        if (tween !== tween0) {
          tween1 = tween0 = tween;
          for (var i = 0, n = tween1.length; i < n; ++i) {
            if (tween1[i].name === name) {
              tween1 = tween1.slice();
              tween1.splice(i, 1);
              break;
            }
          }
        }

        schedule.tween = tween1;
      };
    }

    function tweenFunction$2(id, name, value) {
      var tween0, tween1;
      if (typeof value !== "function") throw new Error;
      return function() {
        var schedule = set$d(this, id),
            tween = schedule.tween;

        // If this node shared tween with the previous node,
        // just assign the updated shared tween and we’re done!
        // Otherwise, copy-on-write.
        if (tween !== tween0) {
          tween1 = (tween0 = tween).slice();
          for (var t = {name: name, value: value}, i = 0, n = tween1.length; i < n; ++i) {
            if (tween1[i].name === name) {
              tween1[i] = t;
              break;
            }
          }
          if (i === n) tween1.push(t);
        }

        schedule.tween = tween1;
      };
    }

    function transition_tween$2(name, value) {
      var id = this._id;

      name += "";

      if (arguments.length < 2) {
        var tween = get$9(this.node(), id).tween;
        for (var i = 0, n = tween.length, t; i < n; ++i) {
          if ((t = tween[i]).name === name) {
            return t.value;
          }
        }
        return null;
      }

      return this.each((value == null ? tweenRemove$2 : tweenFunction$2)(id, name, value));
    }

    function tweenValue$2(transition, name, value) {
      var id = transition._id;

      transition.each(function() {
        var schedule = set$d(this, id);
        (schedule.value || (schedule.value = {}))[name] = value.apply(this, arguments);
      });

      return function(node) {
        return get$9(node, id).value[name];
      };
    }

    function interpolate$5(a, b) {
      var c;
      return (typeof b === "number" ? interpolateNumber$2
          : b instanceof color$5 ? interpolateRgb$2
          : (c = color$5(b)) ? (b = c, interpolateRgb$2)
          : interpolateString$2)(a, b);
    }

    function attrRemove$7(name) {
      return function() {
        this.removeAttribute(name);
      };
    }

    function attrRemoveNS$7(fullname) {
      return function() {
        this.removeAttributeNS(fullname.space, fullname.local);
      };
    }

    function attrConstant$7(name, interpolate, value1) {
      var value00,
          interpolate0;
      return function() {
        var value0 = this.getAttribute(name);
        return value0 === value1 ? null
            : value0 === value00 ? interpolate0
            : interpolate0 = interpolate(value00 = value0, value1);
      };
    }

    function attrConstantNS$7(fullname, interpolate, value1) {
      var value00,
          interpolate0;
      return function() {
        var value0 = this.getAttributeNS(fullname.space, fullname.local);
        return value0 === value1 ? null
            : value0 === value00 ? interpolate0
            : interpolate0 = interpolate(value00 = value0, value1);
      };
    }

    function attrFunction$7(name, interpolate, value) {
      var value00,
          value10,
          interpolate0;
      return function() {
        var value0, value1 = value(this);
        if (value1 == null) return void this.removeAttribute(name);
        value0 = this.getAttribute(name);
        return value0 === value1 ? null
            : value0 === value00 && value1 === value10 ? interpolate0
            : interpolate0 = interpolate(value00 = value0, value10 = value1);
      };
    }

    function attrFunctionNS$7(fullname, interpolate, value) {
      var value00,
          value10,
          interpolate0;
      return function() {
        var value0, value1 = value(this);
        if (value1 == null) return void this.removeAttributeNS(fullname.space, fullname.local);
        value0 = this.getAttributeNS(fullname.space, fullname.local);
        return value0 === value1 ? null
            : value0 === value00 && value1 === value10 ? interpolate0
            : interpolate0 = interpolate(value00 = value0, value10 = value1);
      };
    }

    function transition_attr$2(name, value) {
      var fullname = namespace$4(name), i = fullname === "transform" ? interpolateTransformSvg$3 : interpolate$5;
      return this.attrTween(name, typeof value === "function"
          ? (fullname.local ? attrFunctionNS$7 : attrFunction$7)(fullname, i, tweenValue$2(this, "attr." + name, value))
          : value == null ? (fullname.local ? attrRemoveNS$7 : attrRemove$7)(fullname)
          : (fullname.local ? attrConstantNS$7 : attrConstant$7)(fullname, i, value + ""));
    }

    function attrTweenNS$2(fullname, value) {
      function tween() {
        var node = this, i = value.apply(node, arguments);
        return i && function(t) {
          node.setAttributeNS(fullname.space, fullname.local, i(t));
        };
      }
      tween._value = value;
      return tween;
    }

    function attrTween$2(name, value) {
      function tween() {
        var node = this, i = value.apply(node, arguments);
        return i && function(t) {
          node.setAttribute(name, i(t));
        };
      }
      tween._value = value;
      return tween;
    }

    function transition_attrTween$2(name, value) {
      var key = "attr." + name;
      if (arguments.length < 2) return (key = this.tween(key)) && key._value;
      if (value == null) return this.tween(key, null);
      if (typeof value !== "function") throw new Error;
      var fullname = namespace$4(name);
      return this.tween(key, (fullname.local ? attrTweenNS$2 : attrTween$2)(fullname, value));
    }

    function delayFunction$2(id, value) {
      return function() {
        init$2(this, id).delay = +value.apply(this, arguments);
      };
    }

    function delayConstant$2(id, value) {
      return value = +value, function() {
        init$2(this, id).delay = value;
      };
    }

    function transition_delay$2(value) {
      var id = this._id;

      return arguments.length
          ? this.each((typeof value === "function"
              ? delayFunction$2
              : delayConstant$2)(id, value))
          : get$9(this.node(), id).delay;
    }

    function durationFunction$2(id, value) {
      return function() {
        set$d(this, id).duration = +value.apply(this, arguments);
      };
    }

    function durationConstant$2(id, value) {
      return value = +value, function() {
        set$d(this, id).duration = value;
      };
    }

    function transition_duration$2(value) {
      var id = this._id;

      return arguments.length
          ? this.each((typeof value === "function"
              ? durationFunction$2
              : durationConstant$2)(id, value))
          : get$9(this.node(), id).duration;
    }

    function easeConstant$2(id, value) {
      if (typeof value !== "function") throw new Error;
      return function() {
        set$d(this, id).ease = value;
      };
    }

    function transition_ease$2(value) {
      var id = this._id;

      return arguments.length
          ? this.each(easeConstant$2(id, value))
          : get$9(this.node(), id).ease;
    }

    function transition_filter$2(match) {
      if (typeof match !== "function") match = matcher$5(match);

      for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, subgroup = subgroups[j] = [], node, i = 0; i < n; ++i) {
          if ((node = group[i]) && match.call(node, node.__data__, i, group)) {
            subgroup.push(node);
          }
        }
      }

      return new Transition$2(subgroups, this._parents, this._name, this._id);
    }

    function transition_merge$2(transition) {
      if (transition._id !== this._id) throw new Error;

      for (var groups0 = this._groups, groups1 = transition._groups, m0 = groups0.length, m1 = groups1.length, m = Math.min(m0, m1), merges = new Array(m0), j = 0; j < m; ++j) {
        for (var group0 = groups0[j], group1 = groups1[j], n = group0.length, merge = merges[j] = new Array(n), node, i = 0; i < n; ++i) {
          if (node = group0[i] || group1[i]) {
            merge[i] = node;
          }
        }
      }

      for (; j < m0; ++j) {
        merges[j] = groups0[j];
      }

      return new Transition$2(merges, this._parents, this._name, this._id);
    }

    function start$3(name) {
      return (name + "").trim().split(/^|\s+/).every(function(t) {
        var i = t.indexOf(".");
        if (i >= 0) t = t.slice(0, i);
        return !t || t === "start";
      });
    }

    function onFunction$2(id, name, listener) {
      var on0, on1, sit = start$3(name) ? init$2 : set$d;
      return function() {
        var schedule = sit(this, id),
            on = schedule.on;

        // If this node shared a dispatch with the previous node,
        // just assign the updated shared dispatch and we’re done!
        // Otherwise, copy-on-write.
        if (on !== on0) (on1 = (on0 = on).copy()).on(name, listener);

        schedule.on = on1;
      };
    }

    function transition_on$2(name, listener) {
      var id = this._id;

      return arguments.length < 2
          ? get$9(this.node(), id).on.on(name)
          : this.each(onFunction$2(id, name, listener));
    }

    function removeFunction$2(id) {
      return function() {
        var parent = this.parentNode;
        for (var i in this.__transition) if (+i !== id) return;
        if (parent) parent.removeChild(this);
      };
    }

    function transition_remove$2() {
      return this.on("end.remove", removeFunction$2(this._id));
    }

    function transition_select$2(select) {
      var name = this._name,
          id = this._id;

      if (typeof select !== "function") select = selector$4(select);

      for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, subgroup = subgroups[j] = new Array(n), node, subnode, i = 0; i < n; ++i) {
          if ((node = group[i]) && (subnode = select.call(node, node.__data__, i, group))) {
            if ("__data__" in node) subnode.__data__ = node.__data__;
            subgroup[i] = subnode;
            schedule$3(subgroup[i], name, id, i, subgroup, get$9(node, id));
          }
        }
      }

      return new Transition$2(subgroups, this._parents, name, id);
    }

    function transition_selectAll$2(select) {
      var name = this._name,
          id = this._id;

      if (typeof select !== "function") select = selectorAll$4(select);

      for (var groups = this._groups, m = groups.length, subgroups = [], parents = [], j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, node, i = 0; i < n; ++i) {
          if (node = group[i]) {
            for (var children = select.call(node, node.__data__, i, group), child, inherit = get$9(node, id), k = 0, l = children.length; k < l; ++k) {
              if (child = children[k]) {
                schedule$3(child, name, id, k, children, inherit);
              }
            }
            subgroups.push(children);
            parents.push(node);
          }
        }
      }

      return new Transition$2(subgroups, parents, name, id);
    }

    var Selection$7 = selection$4.prototype.constructor;

    function transition_selection$2() {
      return new Selection$7(this._groups, this._parents);
    }

    function styleRemove$7(name, interpolate) {
      var value00,
          value10,
          interpolate0;
      return function() {
        var value0 = styleValue$4(this, name),
            value1 = (this.style.removeProperty(name), styleValue$4(this, name));
        return value0 === value1 ? null
            : value0 === value00 && value1 === value10 ? interpolate0
            : interpolate0 = interpolate(value00 = value0, value10 = value1);
      };
    }

    function styleRemoveEnd(name) {
      return function() {
        this.style.removeProperty(name);
      };
    }

    function styleConstant$7(name, interpolate, value1) {
      var value00,
          interpolate0;
      return function() {
        var value0 = styleValue$4(this, name);
        return value0 === value1 ? null
            : value0 === value00 ? interpolate0
            : interpolate0 = interpolate(value00 = value0, value1);
      };
    }

    function styleFunction$7(name, interpolate, value) {
      var value00,
          value10,
          interpolate0;
      return function() {
        var value0 = styleValue$4(this, name),
            value1 = value(this);
        if (value1 == null) value1 = (this.style.removeProperty(name), styleValue$4(this, name));
        return value0 === value1 ? null
            : value0 === value00 && value1 === value10 ? interpolate0
            : interpolate0 = interpolate(value00 = value0, value10 = value1);
      };
    }

    function transition_style$2(name, value, priority) {
      var i = (name += "") === "transform" ? interpolateTransformCss$3 : interpolate$5;
      return value == null ? this
              .styleTween(name, styleRemove$7(name, i))
              .on("end.style." + name, styleRemoveEnd(name))
          : this.styleTween(name, typeof value === "function"
              ? styleFunction$7(name, i, tweenValue$2(this, "style." + name, value))
              : styleConstant$7(name, i, value + ""), priority);
    }

    function styleTween$2(name, value, priority) {
      function tween() {
        var node = this, i = value.apply(node, arguments);
        return i && function(t) {
          node.style.setProperty(name, i(t), priority);
        };
      }
      tween._value = value;
      return tween;
    }

    function transition_styleTween$2(name, value, priority) {
      var key = "style." + (name += "");
      if (arguments.length < 2) return (key = this.tween(key)) && key._value;
      if (value == null) return this.tween(key, null);
      if (typeof value !== "function") throw new Error;
      return this.tween(key, styleTween$2(name, value, priority == null ? "" : priority));
    }

    function textConstant$7(value) {
      return function() {
        this.textContent = value;
      };
    }

    function textFunction$7(value) {
      return function() {
        var value1 = value(this);
        this.textContent = value1 == null ? "" : value1;
      };
    }

    function transition_text$2(value) {
      return this.tween("text", typeof value === "function"
          ? textFunction$7(tweenValue$2(this, "text", value))
          : textConstant$7(value == null ? "" : value + ""));
    }

    function transition_transition$2() {
      var name = this._name,
          id0 = this._id,
          id1 = newId$2();

      for (var groups = this._groups, m = groups.length, j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, node, i = 0; i < n; ++i) {
          if (node = group[i]) {
            var inherit = get$9(node, id0);
            schedule$3(node, name, id1, i, group, {
              time: inherit.time + inherit.delay + inherit.duration,
              delay: 0,
              duration: inherit.duration,
              ease: inherit.ease
            });
          }
        }
      }

      return new Transition$2(groups, this._parents, name, id1);
    }

    var id$2 = 0;

    function Transition$2(groups, parents, name, id) {
      this._groups = groups;
      this._parents = parents;
      this._name = name;
      this._id = id;
    }

    function transition$2(name) {
      return selection$4().transition(name);
    }

    function newId$2() {
      return ++id$2;
    }

    var selection_prototype$2 = selection$4.prototype;

    Transition$2.prototype = transition$2.prototype = {
      constructor: Transition$2,
      select: transition_select$2,
      selectAll: transition_selectAll$2,
      filter: transition_filter$2,
      merge: transition_merge$2,
      selection: transition_selection$2,
      transition: transition_transition$2,
      call: selection_prototype$2.call,
      nodes: selection_prototype$2.nodes,
      node: selection_prototype$2.node,
      size: selection_prototype$2.size,
      empty: selection_prototype$2.empty,
      each: selection_prototype$2.each,
      on: transition_on$2,
      attr: transition_attr$2,
      attrTween: transition_attrTween$2,
      style: transition_style$2,
      styleTween: transition_styleTween$2,
      text: transition_text$2,
      remove: transition_remove$2,
      tween: transition_tween$2,
      delay: transition_delay$2,
      duration: transition_duration$2,
      ease: transition_ease$2
    };

    var defaultTiming$2 = {
      time: null, // Set on use.
      delay: 0,
      duration: 250,
      ease: cubicInOut
    };

    function inherit$2(node, id) {
      var timing;
      while (!(timing = node.__transition) || !(timing = timing[id])) {
        if (!(node = node.parentNode)) {
          return defaultTiming$2.time = now$4(), defaultTiming$2;
        }
      }
      return timing;
    }

    function selection_transition$2(name) {
      var id,
          timing;

      if (name instanceof Transition$2) {
        id = name._id, name = name._name;
      } else {
        id = newId$2(), (timing = defaultTiming$2).time = now$4(), name = name == null ? null : name + "";
      }

      for (var groups = this._groups, m = groups.length, j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, node, i = 0; i < n; ++i) {
          if (node = group[i]) {
            schedule$3(node, name, id, i, group, timing || inherit$2(node, id));
          }
        }
      }

      return new Transition$2(groups, this._parents, name, id);
    }

    selection$4.prototype.interrupt = selection_interrupt$2;
    selection$4.prototype.transition = selection_transition$2;

    var root$7 = [null];

    function active$1(node, name) {
      var schedules = node.__transition,
          schedule,
          i;

      if (schedules) {
        name = name == null ? null : name + "";
        for (i in schedules) {
          if ((schedule = schedules[i]).state > SCHEDULED$2 && schedule.name === name) {
            return new Transition$2([[node]], root$7, name, +i);
          }
        }
      }

      return null;
    }

    function constant$v(x) {
      return function() {
        return x;
      };
    }

    function x$9(d) {
      return d[0];
    }

    function y$9(d) {
      return d[1];
    }

    function RedBlackTree$1() {
      this._ = null; // root node
    }

    function RedBlackNode$1(node) {
      node.U = // parent node
      node.C = // color - true for red, false for black
      node.L = // left node
      node.R = // right node
      node.P = // previous node
      node.N = null; // next node
    }

    RedBlackTree$1.prototype = {
      constructor: RedBlackTree$1,

      insert: function(after, node) {
        var parent, grandpa, uncle;

        if (after) {
          node.P = after;
          node.N = after.N;
          if (after.N) after.N.P = node;
          after.N = node;
          if (after.R) {
            after = after.R;
            while (after.L) after = after.L;
            after.L = node;
          } else {
            after.R = node;
          }
          parent = after;
        } else if (this._) {
          after = RedBlackFirst$1(this._);
          node.P = null;
          node.N = after;
          after.P = after.L = node;
          parent = after;
        } else {
          node.P = node.N = null;
          this._ = node;
          parent = null;
        }
        node.L = node.R = null;
        node.U = parent;
        node.C = true;

        after = node;
        while (parent && parent.C) {
          grandpa = parent.U;
          if (parent === grandpa.L) {
            uncle = grandpa.R;
            if (uncle && uncle.C) {
              parent.C = uncle.C = false;
              grandpa.C = true;
              after = grandpa;
            } else {
              if (after === parent.R) {
                RedBlackRotateLeft$1(this, parent);
                after = parent;
                parent = after.U;
              }
              parent.C = false;
              grandpa.C = true;
              RedBlackRotateRight$1(this, grandpa);
            }
          } else {
            uncle = grandpa.L;
            if (uncle && uncle.C) {
              parent.C = uncle.C = false;
              grandpa.C = true;
              after = grandpa;
            } else {
              if (after === parent.L) {
                RedBlackRotateRight$1(this, parent);
                after = parent;
                parent = after.U;
              }
              parent.C = false;
              grandpa.C = true;
              RedBlackRotateLeft$1(this, grandpa);
            }
          }
          parent = after.U;
        }
        this._.C = false;
      },

      remove: function(node) {
        if (node.N) node.N.P = node.P;
        if (node.P) node.P.N = node.N;
        node.N = node.P = null;

        var parent = node.U,
            sibling,
            left = node.L,
            right = node.R,
            next,
            red;

        if (!left) next = right;
        else if (!right) next = left;
        else next = RedBlackFirst$1(right);

        if (parent) {
          if (parent.L === node) parent.L = next;
          else parent.R = next;
        } else {
          this._ = next;
        }

        if (left && right) {
          red = next.C;
          next.C = node.C;
          next.L = left;
          left.U = next;
          if (next !== right) {
            parent = next.U;
            next.U = node.U;
            node = next.R;
            parent.L = node;
            next.R = right;
            right.U = next;
          } else {
            next.U = parent;
            parent = next;
            node = next.R;
          }
        } else {
          red = node.C;
          node = next;
        }

        if (node) node.U = parent;
        if (red) return;
        if (node && node.C) { node.C = false; return; }

        do {
          if (node === this._) break;
          if (node === parent.L) {
            sibling = parent.R;
            if (sibling.C) {
              sibling.C = false;
              parent.C = true;
              RedBlackRotateLeft$1(this, parent);
              sibling = parent.R;
            }
            if ((sibling.L && sibling.L.C)
                || (sibling.R && sibling.R.C)) {
              if (!sibling.R || !sibling.R.C) {
                sibling.L.C = false;
                sibling.C = true;
                RedBlackRotateRight$1(this, sibling);
                sibling = parent.R;
              }
              sibling.C = parent.C;
              parent.C = sibling.R.C = false;
              RedBlackRotateLeft$1(this, parent);
              node = this._;
              break;
            }
          } else {
            sibling = parent.L;
            if (sibling.C) {
              sibling.C = false;
              parent.C = true;
              RedBlackRotateRight$1(this, parent);
              sibling = parent.L;
            }
            if ((sibling.L && sibling.L.C)
              || (sibling.R && sibling.R.C)) {
              if (!sibling.L || !sibling.L.C) {
                sibling.R.C = false;
                sibling.C = true;
                RedBlackRotateLeft$1(this, sibling);
                sibling = parent.L;
              }
              sibling.C = parent.C;
              parent.C = sibling.L.C = false;
              RedBlackRotateRight$1(this, parent);
              node = this._;
              break;
            }
          }
          sibling.C = true;
          node = parent;
          parent = parent.U;
        } while (!node.C);

        if (node) node.C = false;
      }
    };

    function RedBlackRotateLeft$1(tree, node) {
      var p = node,
          q = node.R,
          parent = p.U;

      if (parent) {
        if (parent.L === p) parent.L = q;
        else parent.R = q;
      } else {
        tree._ = q;
      }

      q.U = parent;
      p.U = q;
      p.R = q.L;
      if (p.R) p.R.U = p;
      q.L = p;
    }

    function RedBlackRotateRight$1(tree, node) {
      var p = node,
          q = node.L,
          parent = p.U;

      if (parent) {
        if (parent.L === p) parent.L = q;
        else parent.R = q;
      } else {
        tree._ = q;
      }

      q.U = parent;
      p.U = q;
      p.L = q.R;
      if (p.L) p.L.U = p;
      q.R = p;
    }

    function RedBlackFirst$1(node) {
      while (node.L) node = node.L;
      return node;
    }

    function createEdge$1(left, right, v0, v1) {
      var edge = [null, null],
          index = edges$1.push(edge) - 1;
      edge.left = left;
      edge.right = right;
      if (v0) setEdgeEnd$1(edge, left, right, v0);
      if (v1) setEdgeEnd$1(edge, right, left, v1);
      cells$1[left.index].halfedges.push(index);
      cells$1[right.index].halfedges.push(index);
      return edge;
    }

    function createBorderEdge$1(left, v0, v1) {
      var edge = [v0, v1];
      edge.left = left;
      return edge;
    }

    function setEdgeEnd$1(edge, left, right, vertex) {
      if (!edge[0] && !edge[1]) {
        edge[0] = vertex;
        edge.left = left;
        edge.right = right;
      } else if (edge.left === right) {
        edge[1] = vertex;
      } else {
        edge[0] = vertex;
      }
    }

    // Liang–Barsky line clipping.
    function clipEdge$1(edge, x0, y0, x1, y1) {
      var a = edge[0],
          b = edge[1],
          ax = a[0],
          ay = a[1],
          bx = b[0],
          by = b[1],
          t0 = 0,
          t1 = 1,
          dx = bx - ax,
          dy = by - ay,
          r;

      r = x0 - ax;
      if (!dx && r > 0) return;
      r /= dx;
      if (dx < 0) {
        if (r < t0) return;
        if (r < t1) t1 = r;
      } else if (dx > 0) {
        if (r > t1) return;
        if (r > t0) t0 = r;
      }

      r = x1 - ax;
      if (!dx && r < 0) return;
      r /= dx;
      if (dx < 0) {
        if (r > t1) return;
        if (r > t0) t0 = r;
      } else if (dx > 0) {
        if (r < t0) return;
        if (r < t1) t1 = r;
      }

      r = y0 - ay;
      if (!dy && r > 0) return;
      r /= dy;
      if (dy < 0) {
        if (r < t0) return;
        if (r < t1) t1 = r;
      } else if (dy > 0) {
        if (r > t1) return;
        if (r > t0) t0 = r;
      }

      r = y1 - ay;
      if (!dy && r < 0) return;
      r /= dy;
      if (dy < 0) {
        if (r > t1) return;
        if (r > t0) t0 = r;
      } else if (dy > 0) {
        if (r < t0) return;
        if (r < t1) t1 = r;
      }

      if (!(t0 > 0) && !(t1 < 1)) return true; // TODO Better check?

      if (t0 > 0) edge[0] = [ax + t0 * dx, ay + t0 * dy];
      if (t1 < 1) edge[1] = [ax + t1 * dx, ay + t1 * dy];
      return true;
    }

    function connectEdge$1(edge, x0, y0, x1, y1) {
      var v1 = edge[1];
      if (v1) return true;

      var v0 = edge[0],
          left = edge.left,
          right = edge.right,
          lx = left[0],
          ly = left[1],
          rx = right[0],
          ry = right[1],
          fx = (lx + rx) / 2,
          fy = (ly + ry) / 2,
          fm,
          fb;

      if (ry === ly) {
        if (fx < x0 || fx >= x1) return;
        if (lx > rx) {
          if (!v0) v0 = [fx, y0];
          else if (v0[1] >= y1) return;
          v1 = [fx, y1];
        } else {
          if (!v0) v0 = [fx, y1];
          else if (v0[1] < y0) return;
          v1 = [fx, y0];
        }
      } else {
        fm = (lx - rx) / (ry - ly);
        fb = fy - fm * fx;
        if (fm < -1 || fm > 1) {
          if (lx > rx) {
            if (!v0) v0 = [(y0 - fb) / fm, y0];
            else if (v0[1] >= y1) return;
            v1 = [(y1 - fb) / fm, y1];
          } else {
            if (!v0) v0 = [(y1 - fb) / fm, y1];
            else if (v0[1] < y0) return;
            v1 = [(y0 - fb) / fm, y0];
          }
        } else {
          if (ly < ry) {
            if (!v0) v0 = [x0, fm * x0 + fb];
            else if (v0[0] >= x1) return;
            v1 = [x1, fm * x1 + fb];
          } else {
            if (!v0) v0 = [x1, fm * x1 + fb];
            else if (v0[0] < x0) return;
            v1 = [x0, fm * x0 + fb];
          }
        }
      }

      edge[0] = v0;
      edge[1] = v1;
      return true;
    }

    function clipEdges$1(x0, y0, x1, y1) {
      var i = edges$1.length,
          edge;

      while (i--) {
        if (!connectEdge$1(edge = edges$1[i], x0, y0, x1, y1)
            || !clipEdge$1(edge, x0, y0, x1, y1)
            || !(Math.abs(edge[0][0] - edge[1][0]) > epsilon$b
                || Math.abs(edge[0][1] - edge[1][1]) > epsilon$b)) {
          delete edges$1[i];
        }
      }
    }

    function createCell$1(site) {
      return cells$1[site.index] = {
        site: site,
        halfedges: []
      };
    }

    function cellHalfedgeAngle$1(cell, edge) {
      var site = cell.site,
          va = edge.left,
          vb = edge.right;
      if (site === vb) vb = va, va = site;
      if (vb) return Math.atan2(vb[1] - va[1], vb[0] - va[0]);
      if (site === va) va = edge[1], vb = edge[0];
      else va = edge[0], vb = edge[1];
      return Math.atan2(va[0] - vb[0], vb[1] - va[1]);
    }

    function cellHalfedgeStart$1(cell, edge) {
      return edge[+(edge.left !== cell.site)];
    }

    function cellHalfedgeEnd$1(cell, edge) {
      return edge[+(edge.left === cell.site)];
    }

    function sortCellHalfedges$1() {
      for (var i = 0, n = cells$1.length, cell, halfedges, j, m; i < n; ++i) {
        if ((cell = cells$1[i]) && (m = (halfedges = cell.halfedges).length)) {
          var index = new Array(m),
              array = new Array(m);
          for (j = 0; j < m; ++j) index[j] = j, array[j] = cellHalfedgeAngle$1(cell, edges$1[halfedges[j]]);
          index.sort(function(i, j) { return array[j] - array[i]; });
          for (j = 0; j < m; ++j) array[j] = halfedges[index[j]];
          for (j = 0; j < m; ++j) halfedges[j] = array[j];
        }
      }
    }

    function clipCells$1(x0, y0, x1, y1) {
      var nCells = cells$1.length,
          iCell,
          cell,
          site,
          iHalfedge,
          halfedges,
          nHalfedges,
          start,
          startX,
          startY,
          end,
          endX,
          endY,
          cover = true;

      for (iCell = 0; iCell < nCells; ++iCell) {
        if (cell = cells$1[iCell]) {
          site = cell.site;
          halfedges = cell.halfedges;
          iHalfedge = halfedges.length;

          // Remove any dangling clipped edges.
          while (iHalfedge--) {
            if (!edges$1[halfedges[iHalfedge]]) {
              halfedges.splice(iHalfedge, 1);
            }
          }

          // Insert any border edges as necessary.
          iHalfedge = 0, nHalfedges = halfedges.length;
          while (iHalfedge < nHalfedges) {
            end = cellHalfedgeEnd$1(cell, edges$1[halfedges[iHalfedge]]), endX = end[0], endY = end[1];
            start = cellHalfedgeStart$1(cell, edges$1[halfedges[++iHalfedge % nHalfedges]]), startX = start[0], startY = start[1];
            if (Math.abs(endX - startX) > epsilon$b || Math.abs(endY - startY) > epsilon$b) {
              halfedges.splice(iHalfedge, 0, edges$1.push(createBorderEdge$1(site, end,
                  Math.abs(endX - x0) < epsilon$b && y1 - endY > epsilon$b ? [x0, Math.abs(startX - x0) < epsilon$b ? startY : y1]
                  : Math.abs(endY - y1) < epsilon$b && x1 - endX > epsilon$b ? [Math.abs(startY - y1) < epsilon$b ? startX : x1, y1]
                  : Math.abs(endX - x1) < epsilon$b && endY - y0 > epsilon$b ? [x1, Math.abs(startX - x1) < epsilon$b ? startY : y0]
                  : Math.abs(endY - y0) < epsilon$b && endX - x0 > epsilon$b ? [Math.abs(startY - y0) < epsilon$b ? startX : x0, y0]
                  : null)) - 1);
              ++nHalfedges;
            }
          }

          if (nHalfedges) cover = false;
        }
      }

      // If there weren’t any edges, have the closest site cover the extent.
      // It doesn’t matter which corner of the extent we measure!
      if (cover) {
        var dx, dy, d2, dc = Infinity;

        for (iCell = 0, cover = null; iCell < nCells; ++iCell) {
          if (cell = cells$1[iCell]) {
            site = cell.site;
            dx = site[0] - x0;
            dy = site[1] - y0;
            d2 = dx * dx + dy * dy;
            if (d2 < dc) dc = d2, cover = cell;
          }
        }

        if (cover) {
          var v00 = [x0, y0], v01 = [x0, y1], v11 = [x1, y1], v10 = [x1, y0];
          cover.halfedges.push(
            edges$1.push(createBorderEdge$1(site = cover.site, v00, v01)) - 1,
            edges$1.push(createBorderEdge$1(site, v01, v11)) - 1,
            edges$1.push(createBorderEdge$1(site, v11, v10)) - 1,
            edges$1.push(createBorderEdge$1(site, v10, v00)) - 1
          );
        }
      }

      // Lastly delete any cells with no edges; these were entirely clipped.
      for (iCell = 0; iCell < nCells; ++iCell) {
        if (cell = cells$1[iCell]) {
          if (!cell.halfedges.length) {
            delete cells$1[iCell];
          }
        }
      }
    }

    var circlePool$1 = [];

    var firstCircle$1;

    function Circle$1() {
      RedBlackNode$1(this);
      this.x =
      this.y =
      this.arc =
      this.site =
      this.cy = null;
    }

    function attachCircle$1(arc) {
      var lArc = arc.P,
          rArc = arc.N;

      if (!lArc || !rArc) return;

      var lSite = lArc.site,
          cSite = arc.site,
          rSite = rArc.site;

      if (lSite === rSite) return;

      var bx = cSite[0],
          by = cSite[1],
          ax = lSite[0] - bx,
          ay = lSite[1] - by,
          cx = rSite[0] - bx,
          cy = rSite[1] - by;

      var d = 2 * (ax * cy - ay * cx);
      if (d >= -epsilon2$5) return;

      var ha = ax * ax + ay * ay,
          hc = cx * cx + cy * cy,
          x = (cy * ha - ay * hc) / d,
          y = (ax * hc - cx * ha) / d;

      var circle = circlePool$1.pop() || new Circle$1;
      circle.arc = arc;
      circle.site = cSite;
      circle.x = x + bx;
      circle.y = (circle.cy = y + by) + Math.sqrt(x * x + y * y); // y bottom

      arc.circle = circle;

      var before = null,
          node = circles$1._;

      while (node) {
        if (circle.y < node.y || (circle.y === node.y && circle.x <= node.x)) {
          if (node.L) node = node.L;
          else { before = node.P; break; }
        } else {
          if (node.R) node = node.R;
          else { before = node; break; }
        }
      }

      circles$1.insert(before, circle);
      if (!before) firstCircle$1 = circle;
    }

    function detachCircle$1(arc) {
      var circle = arc.circle;
      if (circle) {
        if (!circle.P) firstCircle$1 = circle.N;
        circles$1.remove(circle);
        circlePool$1.push(circle);
        RedBlackNode$1(circle);
        arc.circle = null;
      }
    }

    var beachPool$1 = [];

    function Beach$1() {
      RedBlackNode$1(this);
      this.edge =
      this.site =
      this.circle = null;
    }

    function createBeach$1(site) {
      var beach = beachPool$1.pop() || new Beach$1;
      beach.site = site;
      return beach;
    }

    function detachBeach$1(beach) {
      detachCircle$1(beach);
      beaches$1.remove(beach);
      beachPool$1.push(beach);
      RedBlackNode$1(beach);
    }

    function removeBeach$1(beach) {
      var circle = beach.circle,
          x = circle.x,
          y = circle.cy,
          vertex = [x, y],
          previous = beach.P,
          next = beach.N,
          disappearing = [beach];

      detachBeach$1(beach);

      var lArc = previous;
      while (lArc.circle
          && Math.abs(x - lArc.circle.x) < epsilon$b
          && Math.abs(y - lArc.circle.cy) < epsilon$b) {
        previous = lArc.P;
        disappearing.unshift(lArc);
        detachBeach$1(lArc);
        lArc = previous;
      }

      disappearing.unshift(lArc);
      detachCircle$1(lArc);

      var rArc = next;
      while (rArc.circle
          && Math.abs(x - rArc.circle.x) < epsilon$b
          && Math.abs(y - rArc.circle.cy) < epsilon$b) {
        next = rArc.N;
        disappearing.push(rArc);
        detachBeach$1(rArc);
        rArc = next;
      }

      disappearing.push(rArc);
      detachCircle$1(rArc);

      var nArcs = disappearing.length,
          iArc;
      for (iArc = 1; iArc < nArcs; ++iArc) {
        rArc = disappearing[iArc];
        lArc = disappearing[iArc - 1];
        setEdgeEnd$1(rArc.edge, lArc.site, rArc.site, vertex);
      }

      lArc = disappearing[0];
      rArc = disappearing[nArcs - 1];
      rArc.edge = createEdge$1(lArc.site, rArc.site, null, vertex);

      attachCircle$1(lArc);
      attachCircle$1(rArc);
    }

    function addBeach$1(site) {
      var x = site[0],
          directrix = site[1],
          lArc,
          rArc,
          dxl,
          dxr,
          node = beaches$1._;

      while (node) {
        dxl = leftBreakPoint$1(node, directrix) - x;
        if (dxl > epsilon$b) node = node.L; else {
          dxr = x - rightBreakPoint$1(node, directrix);
          if (dxr > epsilon$b) {
            if (!node.R) {
              lArc = node;
              break;
            }
            node = node.R;
          } else {
            if (dxl > -epsilon$b) {
              lArc = node.P;
              rArc = node;
            } else if (dxr > -epsilon$b) {
              lArc = node;
              rArc = node.N;
            } else {
              lArc = rArc = node;
            }
            break;
          }
        }
      }

      createCell$1(site);
      var newArc = createBeach$1(site);
      beaches$1.insert(lArc, newArc);

      if (!lArc && !rArc) return;

      if (lArc === rArc) {
        detachCircle$1(lArc);
        rArc = createBeach$1(lArc.site);
        beaches$1.insert(newArc, rArc);
        newArc.edge = rArc.edge = createEdge$1(lArc.site, newArc.site);
        attachCircle$1(lArc);
        attachCircle$1(rArc);
        return;
      }

      if (!rArc) { // && lArc
        newArc.edge = createEdge$1(lArc.site, newArc.site);
        return;
      }

      // else lArc !== rArc
      detachCircle$1(lArc);
      detachCircle$1(rArc);

      var lSite = lArc.site,
          ax = lSite[0],
          ay = lSite[1],
          bx = site[0] - ax,
          by = site[1] - ay,
          rSite = rArc.site,
          cx = rSite[0] - ax,
          cy = rSite[1] - ay,
          d = 2 * (bx * cy - by * cx),
          hb = bx * bx + by * by,
          hc = cx * cx + cy * cy,
          vertex = [(cy * hb - by * hc) / d + ax, (bx * hc - cx * hb) / d + ay];

      setEdgeEnd$1(rArc.edge, lSite, rSite, vertex);
      newArc.edge = createEdge$1(lSite, site, null, vertex);
      rArc.edge = createEdge$1(site, rSite, null, vertex);
      attachCircle$1(lArc);
      attachCircle$1(rArc);
    }

    function leftBreakPoint$1(arc, directrix) {
      var site = arc.site,
          rfocx = site[0],
          rfocy = site[1],
          pby2 = rfocy - directrix;

      if (!pby2) return rfocx;

      var lArc = arc.P;
      if (!lArc) return -Infinity;

      site = lArc.site;
      var lfocx = site[0],
          lfocy = site[1],
          plby2 = lfocy - directrix;

      if (!plby2) return lfocx;

      var hl = lfocx - rfocx,
          aby2 = 1 / pby2 - 1 / plby2,
          b = hl / plby2;

      if (aby2) return (-b + Math.sqrt(b * b - 2 * aby2 * (hl * hl / (-2 * plby2) - lfocy + plby2 / 2 + rfocy - pby2 / 2))) / aby2 + rfocx;

      return (rfocx + lfocx) / 2;
    }

    function rightBreakPoint$1(arc, directrix) {
      var rArc = arc.N;
      if (rArc) return leftBreakPoint$1(rArc, directrix);
      var site = arc.site;
      return site[1] === directrix ? site[0] : Infinity;
    }

    var epsilon$b = 1e-6;
    var epsilon2$5 = 1e-12;
    var beaches$1;
    var cells$1;
    var circles$1;
    var edges$1;

    function triangleArea$1(a, b, c) {
      return (a[0] - c[0]) * (b[1] - a[1]) - (a[0] - b[0]) * (c[1] - a[1]);
    }

    function lexicographic$1(a, b) {
      return b[1] - a[1]
          || b[0] - a[0];
    }

    function Diagram$1(sites, extent) {
      var site = sites.sort(lexicographic$1).pop(),
          x,
          y,
          circle;

      edges$1 = [];
      cells$1 = new Array(sites.length);
      beaches$1 = new RedBlackTree$1;
      circles$1 = new RedBlackTree$1;

      while (true) {
        circle = firstCircle$1;
        if (site && (!circle || site[1] < circle.y || (site[1] === circle.y && site[0] < circle.x))) {
          if (site[0] !== x || site[1] !== y) {
            addBeach$1(site);
            x = site[0], y = site[1];
          }
          site = sites.pop();
        } else if (circle) {
          removeBeach$1(circle.arc);
        } else {
          break;
        }
      }

      sortCellHalfedges$1();

      if (extent) {
        var x0 = +extent[0][0],
            y0 = +extent[0][1],
            x1 = +extent[1][0],
            y1 = +extent[1][1];
        clipEdges$1(x0, y0, x1, y1);
        clipCells$1(x0, y0, x1, y1);
      }

      this.edges = edges$1;
      this.cells = cells$1;

      beaches$1 =
      circles$1 =
      edges$1 =
      cells$1 = null;
    }

    Diagram$1.prototype = {
      constructor: Diagram$1,

      polygons: function() {
        var edges = this.edges;

        return this.cells.map(function(cell) {
          var polygon = cell.halfedges.map(function(i) { return cellHalfedgeStart$1(cell, edges[i]); });
          polygon.data = cell.site.data;
          return polygon;
        });
      },

      triangles: function() {
        var triangles = [],
            edges = this.edges;

        this.cells.forEach(function(cell, i) {
          if (!(m = (halfedges = cell.halfedges).length)) return;
          var site = cell.site,
              halfedges,
              j = -1,
              m,
              s0,
              e1 = edges[halfedges[m - 1]],
              s1 = e1.left === site ? e1.right : e1.left;

          while (++j < m) {
            s0 = s1;
            e1 = edges[halfedges[j]];
            s1 = e1.left === site ? e1.right : e1.left;
            if (s0 && s1 && i < s0.index && i < s1.index && triangleArea$1(site, s0, s1) < 0) {
              triangles.push([site.data, s0.data, s1.data]);
            }
          }
        });

        return triangles;
      },

      links: function() {
        return this.edges.filter(function(edge) {
          return edge.right;
        }).map(function(edge) {
          return {
            source: edge.left.data,
            target: edge.right.data
          };
        });
      },

      find: function(x, y, radius) {
        var that = this, i0, i1 = that._found || 0, n = that.cells.length, cell;

        // Use the previously-found cell, or start with an arbitrary one.
        while (!(cell = that.cells[i1])) if (++i1 >= n) return null;
        var dx = x - cell.site[0], dy = y - cell.site[1], d2 = dx * dx + dy * dy;

        // Traverse the half-edges to find a closer cell, if any.
        do {
          cell = that.cells[i0 = i1], i1 = null;
          cell.halfedges.forEach(function(e) {
            var edge = that.edges[e], v = edge.left;
            if ((v === cell.site || !v) && !(v = edge.right)) return;
            var vx = x - v[0], vy = y - v[1], v2 = vx * vx + vy * vy;
            if (v2 < d2) d2 = v2, i1 = v.index;
          });
        } while (i1 !== null);

        that._found = i0;

        return radius == null || d2 <= radius * radius ? cell.site : null;
      }
    };

    function voronoi$1() {
      var x = x$9,
          y = y$9,
          extent = null;

      function voronoi(data) {
        return new Diagram$1(data.map(function(d, i) {
          var s = [Math.round(x(d, i, data) / epsilon$b) * epsilon$b, Math.round(y(d, i, data) / epsilon$b) * epsilon$b];
          s.index = i;
          s.data = d;
          return s;
        }), extent);
      }

      voronoi.polygons = function(data) {
        return voronoi(data).polygons();
      };

      voronoi.links = function(data) {
        return voronoi(data).links();
      };

      voronoi.triangles = function(data) {
        return voronoi(data).triangles();
      };

      voronoi.x = function(_) {
        return arguments.length ? (x = typeof _ === "function" ? _ : constant$v(+_), voronoi) : x;
      };

      voronoi.y = function(_) {
        return arguments.length ? (y = typeof _ === "function" ? _ : constant$v(+_), voronoi) : y;
      };

      voronoi.extent = function(_) {
        return arguments.length ? (extent = _ == null ? null : [[+_[0][0], +_[0][1]], [+_[1][0], +_[1][1]]], voronoi) : extent && [[extent[0][0], extent[0][1]], [extent[1][0], extent[1][1]]];
      };

      voronoi.size = function(_) {
        return arguments.length ? (extent = _ == null ? null : [[0, 0], [+_[0], +_[1]]], voronoi) : extent && [extent[1][0] - extent[0][0], extent[1][1] - extent[0][1]];
      };

      return voronoi;
    }

    var noop$b = {value: function() {}};

    function dispatch$6() {
      for (var i = 0, n = arguments.length, _ = {}, t; i < n; ++i) {
        if (!(t = arguments[i] + "") || (t in _) || /[\s.]/.test(t)) throw new Error("illegal type: " + t);
        _[t] = [];
      }
      return new Dispatch$6(_);
    }

    function Dispatch$6(_) {
      this._ = _;
    }

    function parseTypenames$b(typenames, types) {
      return typenames.trim().split(/^|\s+/).map(function(t) {
        var name = "", i = t.indexOf(".");
        if (i >= 0) name = t.slice(i + 1), t = t.slice(0, i);
        if (t && !types.hasOwnProperty(t)) throw new Error("unknown type: " + t);
        return {type: t, name: name};
      });
    }

    Dispatch$6.prototype = dispatch$6.prototype = {
      constructor: Dispatch$6,
      on: function(typename, callback) {
        var _ = this._,
            T = parseTypenames$b(typename + "", _),
            t,
            i = -1,
            n = T.length;

        // If no callback was specified, return the callback of the given type and name.
        if (arguments.length < 2) {
          while (++i < n) if ((t = (typename = T[i]).type) && (t = get$a(_[t], typename.name))) return t;
          return;
        }

        // If a type was specified, set the callback for the given type and name.
        // Otherwise, if a null callback was specified, remove callbacks of the given name.
        if (callback != null && typeof callback !== "function") throw new Error("invalid callback: " + callback);
        while (++i < n) {
          if (t = (typename = T[i]).type) _[t] = set$e(_[t], typename.name, callback);
          else if (callback == null) for (t in _) _[t] = set$e(_[t], typename.name, null);
        }

        return this;
      },
      copy: function() {
        var copy = {}, _ = this._;
        for (var t in _) copy[t] = _[t].slice();
        return new Dispatch$6(copy);
      },
      call: function(type, that) {
        if ((n = arguments.length - 2) > 0) for (var args = new Array(n), i = 0, n, t; i < n; ++i) args[i] = arguments[i + 2];
        if (!this._.hasOwnProperty(type)) throw new Error("unknown type: " + type);
        for (t = this._[type], i = 0, n = t.length; i < n; ++i) t[i].value.apply(that, args);
      },
      apply: function(type, that, args) {
        if (!this._.hasOwnProperty(type)) throw new Error("unknown type: " + type);
        for (var t = this._[type], i = 0, n = t.length; i < n; ++i) t[i].value.apply(that, args);
      }
    };

    function get$a(type, name) {
      for (var i = 0, n = type.length, c; i < n; ++i) {
        if ((c = type[i]).name === name) {
          return c.value;
        }
      }
    }

    function set$e(type, name, callback) {
      for (var i = 0, n = type.length; i < n; ++i) {
        if (type[i].name === name) {
          type[i] = noop$b, type = type.slice(0, i).concat(type.slice(i + 1));
          break;
        }
      }
      if (callback != null) type.push({name: name, value: callback});
      return type;
    }

    var xhtml$5 = "http://www.w3.org/1999/xhtml";

    var namespaces$5 = {
      svg: "http://www.w3.org/2000/svg",
      xhtml: xhtml$5,
      xlink: "http://www.w3.org/1999/xlink",
      xml: "http://www.w3.org/XML/1998/namespace",
      xmlns: "http://www.w3.org/2000/xmlns/"
    };

    function namespace$5(name) {
      var prefix = name += "", i = prefix.indexOf(":");
      if (i >= 0 && (prefix = name.slice(0, i)) !== "xmlns") name = name.slice(i + 1);
      return namespaces$5.hasOwnProperty(prefix) ? {space: namespaces$5[prefix], local: name} : name;
    }

    function creatorInherit$5(name) {
      return function() {
        var document = this.ownerDocument,
            uri = this.namespaceURI;
        return uri === xhtml$5 && document.documentElement.namespaceURI === xhtml$5
            ? document.createElement(name)
            : document.createElementNS(uri, name);
      };
    }

    function creatorFixed$5(fullname) {
      return function() {
        return this.ownerDocument.createElementNS(fullname.space, fullname.local);
      };
    }

    function creator$5(name) {
      var fullname = namespace$5(name);
      return (fullname.local
          ? creatorFixed$5
          : creatorInherit$5)(fullname);
    }

    function none$9() {}

    function selector$5(selector) {
      return selector == null ? none$9 : function() {
        return this.querySelector(selector);
      };
    }

    function selection_select$5(select) {
      if (typeof select !== "function") select = selector$5(select);

      for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, subgroup = subgroups[j] = new Array(n), node, subnode, i = 0; i < n; ++i) {
          if ((node = group[i]) && (subnode = select.call(node, node.__data__, i, group))) {
            if ("__data__" in node) subnode.__data__ = node.__data__;
            subgroup[i] = subnode;
          }
        }
      }

      return new Selection$8(subgroups, this._parents);
    }

    function empty$7() {
      return [];
    }

    function selectorAll$5(selector) {
      return selector == null ? empty$7 : function() {
        return this.querySelectorAll(selector);
      };
    }

    function selection_selectAll$5(select) {
      if (typeof select !== "function") select = selectorAll$5(select);

      for (var groups = this._groups, m = groups.length, subgroups = [], parents = [], j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, node, i = 0; i < n; ++i) {
          if (node = group[i]) {
            subgroups.push(select.call(node, node.__data__, i, group));
            parents.push(node);
          }
        }
      }

      return new Selection$8(subgroups, parents);
    }

    function matcher$6(selector) {
      return function() {
        return this.matches(selector);
      };
    }

    function selection_filter$5(match) {
      if (typeof match !== "function") match = matcher$6(match);

      for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, subgroup = subgroups[j] = [], node, i = 0; i < n; ++i) {
          if ((node = group[i]) && match.call(node, node.__data__, i, group)) {
            subgroup.push(node);
          }
        }
      }

      return new Selection$8(subgroups, this._parents);
    }

    function sparse$5(update) {
      return new Array(update.length);
    }

    function selection_enter$5() {
      return new Selection$8(this._enter || this._groups.map(sparse$5), this._parents);
    }

    function EnterNode$5(parent, datum) {
      this.ownerDocument = parent.ownerDocument;
      this.namespaceURI = parent.namespaceURI;
      this._next = null;
      this._parent = parent;
      this.__data__ = datum;
    }

    EnterNode$5.prototype = {
      constructor: EnterNode$5,
      appendChild: function(child) { return this._parent.insertBefore(child, this._next); },
      insertBefore: function(child, next) { return this._parent.insertBefore(child, next); },
      querySelector: function(selector) { return this._parent.querySelector(selector); },
      querySelectorAll: function(selector) { return this._parent.querySelectorAll(selector); }
    };

    function constant$w(x) {
      return function() {
        return x;
      };
    }

    var keyPrefix$7 = "$"; // Protect against keys like “__proto__”.

    function bindIndex$5(parent, group, enter, update, exit, data) {
      var i = 0,
          node,
          groupLength = group.length,
          dataLength = data.length;

      // Put any non-null nodes that fit into update.
      // Put any null nodes into enter.
      // Put any remaining data into enter.
      for (; i < dataLength; ++i) {
        if (node = group[i]) {
          node.__data__ = data[i];
          update[i] = node;
        } else {
          enter[i] = new EnterNode$5(parent, data[i]);
        }
      }

      // Put any non-null nodes that don’t fit into exit.
      for (; i < groupLength; ++i) {
        if (node = group[i]) {
          exit[i] = node;
        }
      }
    }

    function bindKey$5(parent, group, enter, update, exit, data, key) {
      var i,
          node,
          nodeByKeyValue = {},
          groupLength = group.length,
          dataLength = data.length,
          keyValues = new Array(groupLength),
          keyValue;

      // Compute the key for each node.
      // If multiple nodes have the same key, the duplicates are added to exit.
      for (i = 0; i < groupLength; ++i) {
        if (node = group[i]) {
          keyValues[i] = keyValue = keyPrefix$7 + key.call(node, node.__data__, i, group);
          if (keyValue in nodeByKeyValue) {
            exit[i] = node;
          } else {
            nodeByKeyValue[keyValue] = node;
          }
        }
      }

      // Compute the key for each datum.
      // If there a node associated with this key, join and add it to update.
      // If there is not (or the key is a duplicate), add it to enter.
      for (i = 0; i < dataLength; ++i) {
        keyValue = keyPrefix$7 + key.call(parent, data[i], i, data);
        if (node = nodeByKeyValue[keyValue]) {
          update[i] = node;
          node.__data__ = data[i];
          nodeByKeyValue[keyValue] = null;
        } else {
          enter[i] = new EnterNode$5(parent, data[i]);
        }
      }

      // Add any remaining nodes that were not bound to data to exit.
      for (i = 0; i < groupLength; ++i) {
        if ((node = group[i]) && (nodeByKeyValue[keyValues[i]] === node)) {
          exit[i] = node;
        }
      }
    }

    function selection_data$5(value, key) {
      if (!value) {
        data = new Array(this.size()), j = -1;
        this.each(function(d) { data[++j] = d; });
        return data;
      }

      var bind = key ? bindKey$5 : bindIndex$5,
          parents = this._parents,
          groups = this._groups;

      if (typeof value !== "function") value = constant$w(value);

      for (var m = groups.length, update = new Array(m), enter = new Array(m), exit = new Array(m), j = 0; j < m; ++j) {
        var parent = parents[j],
            group = groups[j],
            groupLength = group.length,
            data = value.call(parent, parent && parent.__data__, j, parents),
            dataLength = data.length,
            enterGroup = enter[j] = new Array(dataLength),
            updateGroup = update[j] = new Array(dataLength),
            exitGroup = exit[j] = new Array(groupLength);

        bind(parent, group, enterGroup, updateGroup, exitGroup, data, key);

        // Now connect the enter nodes to their following update node, such that
        // appendChild can insert the materialized enter node before this node,
        // rather than at the end of the parent node.
        for (var i0 = 0, i1 = 0, previous, next; i0 < dataLength; ++i0) {
          if (previous = enterGroup[i0]) {
            if (i0 >= i1) i1 = i0 + 1;
            while (!(next = updateGroup[i1]) && ++i1 < dataLength);
            previous._next = next || null;
          }
        }
      }

      update = new Selection$8(update, parents);
      update._enter = enter;
      update._exit = exit;
      return update;
    }

    function selection_exit$5() {
      return new Selection$8(this._exit || this._groups.map(sparse$5), this._parents);
    }

    function selection_join$4(onenter, onupdate, onexit) {
      var enter = this.enter(), update = this, exit = this.exit();
      enter = typeof onenter === "function" ? onenter(enter) : enter.append(onenter + "");
      if (onupdate != null) update = onupdate(update);
      if (onexit == null) exit.remove(); else onexit(exit);
      return enter && update ? enter.merge(update).order() : update;
    }

    function selection_merge$5(selection) {

      for (var groups0 = this._groups, groups1 = selection._groups, m0 = groups0.length, m1 = groups1.length, m = Math.min(m0, m1), merges = new Array(m0), j = 0; j < m; ++j) {
        for (var group0 = groups0[j], group1 = groups1[j], n = group0.length, merge = merges[j] = new Array(n), node, i = 0; i < n; ++i) {
          if (node = group0[i] || group1[i]) {
            merge[i] = node;
          }
        }
      }

      for (; j < m0; ++j) {
        merges[j] = groups0[j];
      }

      return new Selection$8(merges, this._parents);
    }

    function selection_order$5() {

      for (var groups = this._groups, j = -1, m = groups.length; ++j < m;) {
        for (var group = groups[j], i = group.length - 1, next = group[i], node; --i >= 0;) {
          if (node = group[i]) {
            if (next && node.compareDocumentPosition(next) ^ 4) next.parentNode.insertBefore(node, next);
            next = node;
          }
        }
      }

      return this;
    }

    function selection_sort$5(compare) {
      if (!compare) compare = ascending$d;

      function compareNode(a, b) {
        return a && b ? compare(a.__data__, b.__data__) : !a - !b;
      }

      for (var groups = this._groups, m = groups.length, sortgroups = new Array(m), j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, sortgroup = sortgroups[j] = new Array(n), node, i = 0; i < n; ++i) {
          if (node = group[i]) {
            sortgroup[i] = node;
          }
        }
        sortgroup.sort(compareNode);
      }

      return new Selection$8(sortgroups, this._parents).order();
    }

    function ascending$d(a, b) {
      return a < b ? -1 : a > b ? 1 : a >= b ? 0 : NaN;
    }

    function selection_call$5() {
      var callback = arguments[0];
      arguments[0] = this;
      callback.apply(null, arguments);
      return this;
    }

    function selection_nodes$5() {
      var nodes = new Array(this.size()), i = -1;
      this.each(function() { nodes[++i] = this; });
      return nodes;
    }

    function selection_node$5() {

      for (var groups = this._groups, j = 0, m = groups.length; j < m; ++j) {
        for (var group = groups[j], i = 0, n = group.length; i < n; ++i) {
          var node = group[i];
          if (node) return node;
        }
      }

      return null;
    }

    function selection_size$5() {
      var size = 0;
      this.each(function() { ++size; });
      return size;
    }

    function selection_empty$5() {
      return !this.node();
    }

    function selection_each$5(callback) {

      for (var groups = this._groups, j = 0, m = groups.length; j < m; ++j) {
        for (var group = groups[j], i = 0, n = group.length, node; i < n; ++i) {
          if (node = group[i]) callback.call(node, node.__data__, i, group);
        }
      }

      return this;
    }

    function attrRemove$8(name) {
      return function() {
        this.removeAttribute(name);
      };
    }

    function attrRemoveNS$8(fullname) {
      return function() {
        this.removeAttributeNS(fullname.space, fullname.local);
      };
    }

    function attrConstant$8(name, value) {
      return function() {
        this.setAttribute(name, value);
      };
    }

    function attrConstantNS$8(fullname, value) {
      return function() {
        this.setAttributeNS(fullname.space, fullname.local, value);
      };
    }

    function attrFunction$8(name, value) {
      return function() {
        var v = value.apply(this, arguments);
        if (v == null) this.removeAttribute(name);
        else this.setAttribute(name, v);
      };
    }

    function attrFunctionNS$8(fullname, value) {
      return function() {
        var v = value.apply(this, arguments);
        if (v == null) this.removeAttributeNS(fullname.space, fullname.local);
        else this.setAttributeNS(fullname.space, fullname.local, v);
      };
    }

    function selection_attr$5(name, value) {
      var fullname = namespace$5(name);

      if (arguments.length < 2) {
        var node = this.node();
        return fullname.local
            ? node.getAttributeNS(fullname.space, fullname.local)
            : node.getAttribute(fullname);
      }

      return this.each((value == null
          ? (fullname.local ? attrRemoveNS$8 : attrRemove$8) : (typeof value === "function"
          ? (fullname.local ? attrFunctionNS$8 : attrFunction$8)
          : (fullname.local ? attrConstantNS$8 : attrConstant$8)))(fullname, value));
    }

    function defaultView$5(node) {
      return (node.ownerDocument && node.ownerDocument.defaultView) // node is a Node
          || (node.document && node) // node is a Window
          || node.defaultView; // node is a Document
    }

    function styleRemove$8(name) {
      return function() {
        this.style.removeProperty(name);
      };
    }

    function styleConstant$8(name, value, priority) {
      return function() {
        this.style.setProperty(name, value, priority);
      };
    }

    function styleFunction$8(name, value, priority) {
      return function() {
        var v = value.apply(this, arguments);
        if (v == null) this.style.removeProperty(name);
        else this.style.setProperty(name, v, priority);
      };
    }

    function selection_style$5(name, value, priority) {
      return arguments.length > 1
          ? this.each((value == null
                ? styleRemove$8 : typeof value === "function"
                ? styleFunction$8
                : styleConstant$8)(name, value, priority == null ? "" : priority))
          : styleValue$5(this.node(), name);
    }

    function styleValue$5(node, name) {
      return node.style.getPropertyValue(name)
          || defaultView$5(node).getComputedStyle(node, null).getPropertyValue(name);
    }

    function propertyRemove$5(name) {
      return function() {
        delete this[name];
      };
    }

    function propertyConstant$5(name, value) {
      return function() {
        this[name] = value;
      };
    }

    function propertyFunction$5(name, value) {
      return function() {
        var v = value.apply(this, arguments);
        if (v == null) delete this[name];
        else this[name] = v;
      };
    }

    function selection_property$5(name, value) {
      return arguments.length > 1
          ? this.each((value == null
              ? propertyRemove$5 : typeof value === "function"
              ? propertyFunction$5
              : propertyConstant$5)(name, value))
          : this.node()[name];
    }

    function classArray$5(string) {
      return string.trim().split(/^|\s+/);
    }

    function classList$5(node) {
      return node.classList || new ClassList$5(node);
    }

    function ClassList$5(node) {
      this._node = node;
      this._names = classArray$5(node.getAttribute("class") || "");
    }

    ClassList$5.prototype = {
      add: function(name) {
        var i = this._names.indexOf(name);
        if (i < 0) {
          this._names.push(name);
          this._node.setAttribute("class", this._names.join(" "));
        }
      },
      remove: function(name) {
        var i = this._names.indexOf(name);
        if (i >= 0) {
          this._names.splice(i, 1);
          this._node.setAttribute("class", this._names.join(" "));
        }
      },
      contains: function(name) {
        return this._names.indexOf(name) >= 0;
      }
    };

    function classedAdd$5(node, names) {
      var list = classList$5(node), i = -1, n = names.length;
      while (++i < n) list.add(names[i]);
    }

    function classedRemove$5(node, names) {
      var list = classList$5(node), i = -1, n = names.length;
      while (++i < n) list.remove(names[i]);
    }

    function classedTrue$5(names) {
      return function() {
        classedAdd$5(this, names);
      };
    }

    function classedFalse$5(names) {
      return function() {
        classedRemove$5(this, names);
      };
    }

    function classedFunction$5(names, value) {
      return function() {
        (value.apply(this, arguments) ? classedAdd$5 : classedRemove$5)(this, names);
      };
    }

    function selection_classed$5(name, value) {
      var names = classArray$5(name + "");

      if (arguments.length < 2) {
        var list = classList$5(this.node()), i = -1, n = names.length;
        while (++i < n) if (!list.contains(names[i])) return false;
        return true;
      }

      return this.each((typeof value === "function"
          ? classedFunction$5 : value
          ? classedTrue$5
          : classedFalse$5)(names, value));
    }

    function textRemove$5() {
      this.textContent = "";
    }

    function textConstant$8(value) {
      return function() {
        this.textContent = value;
      };
    }

    function textFunction$8(value) {
      return function() {
        var v = value.apply(this, arguments);
        this.textContent = v == null ? "" : v;
      };
    }

    function selection_text$5(value) {
      return arguments.length
          ? this.each(value == null
              ? textRemove$5 : (typeof value === "function"
              ? textFunction$8
              : textConstant$8)(value))
          : this.node().textContent;
    }

    function htmlRemove$5() {
      this.innerHTML = "";
    }

    function htmlConstant$5(value) {
      return function() {
        this.innerHTML = value;
      };
    }

    function htmlFunction$5(value) {
      return function() {
        var v = value.apply(this, arguments);
        this.innerHTML = v == null ? "" : v;
      };
    }

    function selection_html$5(value) {
      return arguments.length
          ? this.each(value == null
              ? htmlRemove$5 : (typeof value === "function"
              ? htmlFunction$5
              : htmlConstant$5)(value))
          : this.node().innerHTML;
    }

    function raise$6() {
      if (this.nextSibling) this.parentNode.appendChild(this);
    }

    function selection_raise$5() {
      return this.each(raise$6);
    }

    function lower$5() {
      if (this.previousSibling) this.parentNode.insertBefore(this, this.parentNode.firstChild);
    }

    function selection_lower$5() {
      return this.each(lower$5);
    }

    function selection_append$5(name) {
      var create = typeof name === "function" ? name : creator$5(name);
      return this.select(function() {
        return this.appendChild(create.apply(this, arguments));
      });
    }

    function constantNull$5() {
      return null;
    }

    function selection_insert$5(name, before) {
      var create = typeof name === "function" ? name : creator$5(name),
          select = before == null ? constantNull$5 : typeof before === "function" ? before : selector$5(before);
      return this.select(function() {
        return this.insertBefore(create.apply(this, arguments), select.apply(this, arguments) || null);
      });
    }

    function remove$5() {
      var parent = this.parentNode;
      if (parent) parent.removeChild(this);
    }

    function selection_remove$5() {
      return this.each(remove$5);
    }

    function selection_cloneShallow$5() {
      var clone = this.cloneNode(false), parent = this.parentNode;
      return parent ? parent.insertBefore(clone, this.nextSibling) : clone;
    }

    function selection_cloneDeep$5() {
      var clone = this.cloneNode(true), parent = this.parentNode;
      return parent ? parent.insertBefore(clone, this.nextSibling) : clone;
    }

    function selection_clone$5(deep) {
      return this.select(deep ? selection_cloneDeep$5 : selection_cloneShallow$5);
    }

    function selection_datum$5(value) {
      return arguments.length
          ? this.property("__data__", value)
          : this.node().__data__;
    }

    var filterEvents$5 = {};

    var event$4 = null;

    if (typeof document !== "undefined") {
      var element$6 = document.documentElement;
      if (!("onmouseenter" in element$6)) {
        filterEvents$5 = {mouseenter: "mouseover", mouseleave: "mouseout"};
      }
    }

    function filterContextListener$5(listener, index, group) {
      listener = contextListener$5(listener, index, group);
      return function(event) {
        var related = event.relatedTarget;
        if (!related || (related !== this && !(related.compareDocumentPosition(this) & 8))) {
          listener.call(this, event);
        }
      };
    }

    function contextListener$5(listener, index, group) {
      return function(event1) {
        var event0 = event$4; // Events can be reentrant (e.g., focus).
        event$4 = event1;
        try {
          listener.call(this, this.__data__, index, group);
        } finally {
          event$4 = event0;
        }
      };
    }

    function parseTypenames$c(typenames) {
      return typenames.trim().split(/^|\s+/).map(function(t) {
        var name = "", i = t.indexOf(".");
        if (i >= 0) name = t.slice(i + 1), t = t.slice(0, i);
        return {type: t, name: name};
      });
    }

    function onRemove$5(typename) {
      return function() {
        var on = this.__on;
        if (!on) return;
        for (var j = 0, i = -1, m = on.length, o; j < m; ++j) {
          if (o = on[j], (!typename.type || o.type === typename.type) && o.name === typename.name) {
            this.removeEventListener(o.type, o.listener, o.capture);
          } else {
            on[++i] = o;
          }
        }
        if (++i) on.length = i;
        else delete this.__on;
      };
    }

    function onAdd$5(typename, value, capture) {
      var wrap = filterEvents$5.hasOwnProperty(typename.type) ? filterContextListener$5 : contextListener$5;
      return function(d, i, group) {
        var on = this.__on, o, listener = wrap(value, i, group);
        if (on) for (var j = 0, m = on.length; j < m; ++j) {
          if ((o = on[j]).type === typename.type && o.name === typename.name) {
            this.removeEventListener(o.type, o.listener, o.capture);
            this.addEventListener(o.type, o.listener = listener, o.capture = capture);
            o.value = value;
            return;
          }
        }
        this.addEventListener(typename.type, listener, capture);
        o = {type: typename.type, name: typename.name, value: value, listener: listener, capture: capture};
        if (!on) this.__on = [o];
        else on.push(o);
      };
    }

    function selection_on$5(typename, value, capture) {
      var typenames = parseTypenames$c(typename + ""), i, n = typenames.length, t;

      if (arguments.length < 2) {
        var on = this.node().__on;
        if (on) for (var j = 0, m = on.length, o; j < m; ++j) {
          for (i = 0, o = on[j]; i < n; ++i) {
            if ((t = typenames[i]).type === o.type && t.name === o.name) {
              return o.value;
            }
          }
        }
        return;
      }

      on = value ? onAdd$5 : onRemove$5;
      if (capture == null) capture = false;
      for (i = 0; i < n; ++i) this.each(on(typenames[i], value, capture));
      return this;
    }

    function customEvent$4(event1, listener, that, args) {
      var event0 = event$4;
      event1.sourceEvent = event$4;
      event$4 = event1;
      try {
        return listener.apply(that, args);
      } finally {
        event$4 = event0;
      }
    }

    function dispatchEvent$5(node, type, params) {
      var window = defaultView$5(node),
          event = window.CustomEvent;

      if (typeof event === "function") {
        event = new event(type, params);
      } else {
        event = window.document.createEvent("Event");
        if (params) event.initEvent(type, params.bubbles, params.cancelable), event.detail = params.detail;
        else event.initEvent(type, false, false);
      }

      node.dispatchEvent(event);
    }

    function dispatchConstant$5(type, params) {
      return function() {
        return dispatchEvent$5(this, type, params);
      };
    }

    function dispatchFunction$5(type, params) {
      return function() {
        return dispatchEvent$5(this, type, params.apply(this, arguments));
      };
    }

    function selection_dispatch$5(type, params) {
      return this.each((typeof params === "function"
          ? dispatchFunction$5
          : dispatchConstant$5)(type, params));
    }

    var root$8 = [null];

    function Selection$8(groups, parents) {
      this._groups = groups;
      this._parents = parents;
    }

    function selection$5() {
      return new Selection$8([[document.documentElement]], root$8);
    }

    Selection$8.prototype = selection$5.prototype = {
      constructor: Selection$8,
      select: selection_select$5,
      selectAll: selection_selectAll$5,
      filter: selection_filter$5,
      data: selection_data$5,
      enter: selection_enter$5,
      exit: selection_exit$5,
      join: selection_join$4,
      merge: selection_merge$5,
      order: selection_order$5,
      sort: selection_sort$5,
      call: selection_call$5,
      nodes: selection_nodes$5,
      node: selection_node$5,
      size: selection_size$5,
      empty: selection_empty$5,
      each: selection_each$5,
      attr: selection_attr$5,
      style: selection_style$5,
      property: selection_property$5,
      classed: selection_classed$5,
      text: selection_text$5,
      html: selection_html$5,
      raise: selection_raise$5,
      lower: selection_lower$5,
      append: selection_append$5,
      insert: selection_insert$5,
      remove: selection_remove$5,
      clone: selection_clone$5,
      datum: selection_datum$5,
      on: selection_on$5,
      dispatch: selection_dispatch$5
    };

    function select$4(selector) {
      return typeof selector === "string"
          ? new Selection$8([[document.querySelector(selector)]], [document.documentElement])
          : new Selection$8([[selector]], root$8);
    }

    function sourceEvent$4() {
      var current = event$4, source;
      while (source = current.sourceEvent) current = source;
      return current;
    }

    function point$e(node, event) {
      var svg = node.ownerSVGElement || node;

      if (svg.createSVGPoint) {
        var point = svg.createSVGPoint();
        point.x = event.clientX, point.y = event.clientY;
        point = point.matrixTransform(node.getScreenCTM().inverse());
        return [point.x, point.y];
      }

      var rect = node.getBoundingClientRect();
      return [event.clientX - rect.left - node.clientLeft, event.clientY - rect.top - node.clientTop];
    }

    function mouse$4(node) {
      var event = sourceEvent$4();
      if (event.changedTouches) event = event.changedTouches[0];
      return point$e(node, event);
    }

    function touch$3(node, touches, identifier) {
      if (arguments.length < 3) identifier = touches, touches = sourceEvent$4().changedTouches;

      for (var i = 0, n = touches ? touches.length : 0, touch; i < n; ++i) {
        if ((touch = touches[i]).identifier === identifier) {
          return point$e(node, touch);
        }
      }

      return null;
    }

    function noevent$6() {
      event$4.preventDefault();
      event$4.stopImmediatePropagation();
    }

    function dragDisable$2(view) {
      var root = view.document.documentElement,
          selection = select$4(view).on("dragstart.drag", noevent$6, true);
      if ("onselectstart" in root) {
        selection.on("selectstart.drag", noevent$6, true);
      } else {
        root.__noselect = root.style.MozUserSelect;
        root.style.MozUserSelect = "none";
      }
    }

    function yesdrag$3(view, noclick) {
      var root = view.document.documentElement,
          selection = select$4(view).on("dragstart.drag", null);
      if (noclick) {
        selection.on("click.drag", noevent$6, true);
        setTimeout(function() { selection.on("click.drag", null); }, 0);
      }
      if ("onselectstart" in root) {
        selection.on("selectstart.drag", null);
      } else {
        root.style.MozUserSelect = root.__noselect;
        delete root.__noselect;
      }
    }

    function define$6(constructor, factory, prototype) {
      constructor.prototype = factory.prototype = prototype;
      prototype.constructor = constructor;
    }

    function extend$6(parent, definition) {
      var prototype = Object.create(parent.prototype);
      for (var key in definition) prototype[key] = definition[key];
      return prototype;
    }

    function Color$6() {}

    var darker$6 = 0.7;
    var brighter$6 = 1 / darker$6;

    var reI$6 = "\\s*([+-]?\\d+)\\s*",
        reN$6 = "\\s*([+-]?\\d*\\.?\\d+(?:[eE][+-]?\\d+)?)\\s*",
        reP$6 = "\\s*([+-]?\\d*\\.?\\d+(?:[eE][+-]?\\d+)?)%\\s*",
        reHex$5 = /^#([0-9a-f]{3,8})$/,
        reRgbInteger$6 = new RegExp("^rgb\\(" + [reI$6, reI$6, reI$6] + "\\)$"),
        reRgbPercent$6 = new RegExp("^rgb\\(" + [reP$6, reP$6, reP$6] + "\\)$"),
        reRgbaInteger$6 = new RegExp("^rgba\\(" + [reI$6, reI$6, reI$6, reN$6] + "\\)$"),
        reRgbaPercent$6 = new RegExp("^rgba\\(" + [reP$6, reP$6, reP$6, reN$6] + "\\)$"),
        reHslPercent$6 = new RegExp("^hsl\\(" + [reN$6, reP$6, reP$6] + "\\)$"),
        reHslaPercent$6 = new RegExp("^hsla\\(" + [reN$6, reP$6, reP$6, reN$6] + "\\)$");

    var named$6 = {
      aliceblue: 0xf0f8ff,
      antiquewhite: 0xfaebd7,
      aqua: 0x00ffff,
      aquamarine: 0x7fffd4,
      azure: 0xf0ffff,
      beige: 0xf5f5dc,
      bisque: 0xffe4c4,
      black: 0x000000,
      blanchedalmond: 0xffebcd,
      blue: 0x0000ff,
      blueviolet: 0x8a2be2,
      brown: 0xa52a2a,
      burlywood: 0xdeb887,
      cadetblue: 0x5f9ea0,
      chartreuse: 0x7fff00,
      chocolate: 0xd2691e,
      coral: 0xff7f50,
      cornflowerblue: 0x6495ed,
      cornsilk: 0xfff8dc,
      crimson: 0xdc143c,
      cyan: 0x00ffff,
      darkblue: 0x00008b,
      darkcyan: 0x008b8b,
      darkgoldenrod: 0xb8860b,
      darkgray: 0xa9a9a9,
      darkgreen: 0x006400,
      darkgrey: 0xa9a9a9,
      darkkhaki: 0xbdb76b,
      darkmagenta: 0x8b008b,
      darkolivegreen: 0x556b2f,
      darkorange: 0xff8c00,
      darkorchid: 0x9932cc,
      darkred: 0x8b0000,
      darksalmon: 0xe9967a,
      darkseagreen: 0x8fbc8f,
      darkslateblue: 0x483d8b,
      darkslategray: 0x2f4f4f,
      darkslategrey: 0x2f4f4f,
      darkturquoise: 0x00ced1,
      darkviolet: 0x9400d3,
      deeppink: 0xff1493,
      deepskyblue: 0x00bfff,
      dimgray: 0x696969,
      dimgrey: 0x696969,
      dodgerblue: 0x1e90ff,
      firebrick: 0xb22222,
      floralwhite: 0xfffaf0,
      forestgreen: 0x228b22,
      fuchsia: 0xff00ff,
      gainsboro: 0xdcdcdc,
      ghostwhite: 0xf8f8ff,
      gold: 0xffd700,
      goldenrod: 0xdaa520,
      gray: 0x808080,
      green: 0x008000,
      greenyellow: 0xadff2f,
      grey: 0x808080,
      honeydew: 0xf0fff0,
      hotpink: 0xff69b4,
      indianred: 0xcd5c5c,
      indigo: 0x4b0082,
      ivory: 0xfffff0,
      khaki: 0xf0e68c,
      lavender: 0xe6e6fa,
      lavenderblush: 0xfff0f5,
      lawngreen: 0x7cfc00,
      lemonchiffon: 0xfffacd,
      lightblue: 0xadd8e6,
      lightcoral: 0xf08080,
      lightcyan: 0xe0ffff,
      lightgoldenrodyellow: 0xfafad2,
      lightgray: 0xd3d3d3,
      lightgreen: 0x90ee90,
      lightgrey: 0xd3d3d3,
      lightpink: 0xffb6c1,
      lightsalmon: 0xffa07a,
      lightseagreen: 0x20b2aa,
      lightskyblue: 0x87cefa,
      lightslategray: 0x778899,
      lightslategrey: 0x778899,
      lightsteelblue: 0xb0c4de,
      lightyellow: 0xffffe0,
      lime: 0x00ff00,
      limegreen: 0x32cd32,
      linen: 0xfaf0e6,
      magenta: 0xff00ff,
      maroon: 0x800000,
      mediumaquamarine: 0x66cdaa,
      mediumblue: 0x0000cd,
      mediumorchid: 0xba55d3,
      mediumpurple: 0x9370db,
      mediumseagreen: 0x3cb371,
      mediumslateblue: 0x7b68ee,
      mediumspringgreen: 0x00fa9a,
      mediumturquoise: 0x48d1cc,
      mediumvioletred: 0xc71585,
      midnightblue: 0x191970,
      mintcream: 0xf5fffa,
      mistyrose: 0xffe4e1,
      moccasin: 0xffe4b5,
      navajowhite: 0xffdead,
      navy: 0x000080,
      oldlace: 0xfdf5e6,
      olive: 0x808000,
      olivedrab: 0x6b8e23,
      orange: 0xffa500,
      orangered: 0xff4500,
      orchid: 0xda70d6,
      palegoldenrod: 0xeee8aa,
      palegreen: 0x98fb98,
      paleturquoise: 0xafeeee,
      palevioletred: 0xdb7093,
      papayawhip: 0xffefd5,
      peachpuff: 0xffdab9,
      peru: 0xcd853f,
      pink: 0xffc0cb,
      plum: 0xdda0dd,
      powderblue: 0xb0e0e6,
      purple: 0x800080,
      rebeccapurple: 0x663399,
      red: 0xff0000,
      rosybrown: 0xbc8f8f,
      royalblue: 0x4169e1,
      saddlebrown: 0x8b4513,
      salmon: 0xfa8072,
      sandybrown: 0xf4a460,
      seagreen: 0x2e8b57,
      seashell: 0xfff5ee,
      sienna: 0xa0522d,
      silver: 0xc0c0c0,
      skyblue: 0x87ceeb,
      slateblue: 0x6a5acd,
      slategray: 0x708090,
      slategrey: 0x708090,
      snow: 0xfffafa,
      springgreen: 0x00ff7f,
      steelblue: 0x4682b4,
      tan: 0xd2b48c,
      teal: 0x008080,
      thistle: 0xd8bfd8,
      tomato: 0xff6347,
      turquoise: 0x40e0d0,
      violet: 0xee82ee,
      wheat: 0xf5deb3,
      white: 0xffffff,
      whitesmoke: 0xf5f5f5,
      yellow: 0xffff00,
      yellowgreen: 0x9acd32
    };

    define$6(Color$6, color$6, {
      copy: function(channels) {
        return Object.assign(new this.constructor, this, channels);
      },
      displayable: function() {
        return this.rgb().displayable();
      },
      hex: color_formatHex$5, // Deprecated! Use color.formatHex.
      formatHex: color_formatHex$5,
      formatHsl: color_formatHsl$5,
      formatRgb: color_formatRgb$5,
      toString: color_formatRgb$5
    });

    function color_formatHex$5() {
      return this.rgb().formatHex();
    }

    function color_formatHsl$5() {
      return hslConvert$6(this).formatHsl();
    }

    function color_formatRgb$5() {
      return this.rgb().formatRgb();
    }

    function color$6(format) {
      var m, l;
      format = (format + "").trim().toLowerCase();
      return (m = reHex$5.exec(format)) ? (l = m[1].length, m = parseInt(m[1], 16), l === 6 ? rgbn$6(m) // #ff0000
          : l === 3 ? new Rgb$6((m >> 8 & 0xf) | (m >> 4 & 0xf0), (m >> 4 & 0xf) | (m & 0xf0), ((m & 0xf) << 4) | (m & 0xf), 1) // #f00
          : l === 8 ? rgba$6(m >> 24 & 0xff, m >> 16 & 0xff, m >> 8 & 0xff, (m & 0xff) / 0xff) // #ff000000
          : l === 4 ? rgba$6((m >> 12 & 0xf) | (m >> 8 & 0xf0), (m >> 8 & 0xf) | (m >> 4 & 0xf0), (m >> 4 & 0xf) | (m & 0xf0), (((m & 0xf) << 4) | (m & 0xf)) / 0xff) // #f000
          : null) // invalid hex
          : (m = reRgbInteger$6.exec(format)) ? new Rgb$6(m[1], m[2], m[3], 1) // rgb(255, 0, 0)
          : (m = reRgbPercent$6.exec(format)) ? new Rgb$6(m[1] * 255 / 100, m[2] * 255 / 100, m[3] * 255 / 100, 1) // rgb(100%, 0%, 0%)
          : (m = reRgbaInteger$6.exec(format)) ? rgba$6(m[1], m[2], m[3], m[4]) // rgba(255, 0, 0, 1)
          : (m = reRgbaPercent$6.exec(format)) ? rgba$6(m[1] * 255 / 100, m[2] * 255 / 100, m[3] * 255 / 100, m[4]) // rgb(100%, 0%, 0%, 1)
          : (m = reHslPercent$6.exec(format)) ? hsla$6(m[1], m[2] / 100, m[3] / 100, 1) // hsl(120, 50%, 50%)
          : (m = reHslaPercent$6.exec(format)) ? hsla$6(m[1], m[2] / 100, m[3] / 100, m[4]) // hsla(120, 50%, 50%, 1)
          : named$6.hasOwnProperty(format) ? rgbn$6(named$6[format]) // eslint-disable-line no-prototype-builtins
          : format === "transparent" ? new Rgb$6(NaN, NaN, NaN, 0)
          : null;
    }

    function rgbn$6(n) {
      return new Rgb$6(n >> 16 & 0xff, n >> 8 & 0xff, n & 0xff, 1);
    }

    function rgba$6(r, g, b, a) {
      if (a <= 0) r = g = b = NaN;
      return new Rgb$6(r, g, b, a);
    }

    function rgbConvert$6(o) {
      if (!(o instanceof Color$6)) o = color$6(o);
      if (!o) return new Rgb$6;
      o = o.rgb();
      return new Rgb$6(o.r, o.g, o.b, o.opacity);
    }

    function rgb$8(r, g, b, opacity) {
      return arguments.length === 1 ? rgbConvert$6(r) : new Rgb$6(r, g, b, opacity == null ? 1 : opacity);
    }

    function Rgb$6(r, g, b, opacity) {
      this.r = +r;
      this.g = +g;
      this.b = +b;
      this.opacity = +opacity;
    }

    define$6(Rgb$6, rgb$8, extend$6(Color$6, {
      brighter: function(k) {
        k = k == null ? brighter$6 : Math.pow(brighter$6, k);
        return new Rgb$6(this.r * k, this.g * k, this.b * k, this.opacity);
      },
      darker: function(k) {
        k = k == null ? darker$6 : Math.pow(darker$6, k);
        return new Rgb$6(this.r * k, this.g * k, this.b * k, this.opacity);
      },
      rgb: function() {
        return this;
      },
      displayable: function() {
        return (-0.5 <= this.r && this.r < 255.5)
            && (-0.5 <= this.g && this.g < 255.5)
            && (-0.5 <= this.b && this.b < 255.5)
            && (0 <= this.opacity && this.opacity <= 1);
      },
      hex: rgb_formatHex$5, // Deprecated! Use color.formatHex.
      formatHex: rgb_formatHex$5,
      formatRgb: rgb_formatRgb$5,
      toString: rgb_formatRgb$5
    }));

    function rgb_formatHex$5() {
      return "#" + hex$5(this.r) + hex$5(this.g) + hex$5(this.b);
    }

    function rgb_formatRgb$5() {
      var a = this.opacity; a = isNaN(a) ? 1 : Math.max(0, Math.min(1, a));
      return (a === 1 ? "rgb(" : "rgba(")
          + Math.max(0, Math.min(255, Math.round(this.r) || 0)) + ", "
          + Math.max(0, Math.min(255, Math.round(this.g) || 0)) + ", "
          + Math.max(0, Math.min(255, Math.round(this.b) || 0))
          + (a === 1 ? ")" : ", " + a + ")");
    }

    function hex$5(value) {
      value = Math.max(0, Math.min(255, Math.round(value) || 0));
      return (value < 16 ? "0" : "") + value.toString(16);
    }

    function hsla$6(h, s, l, a) {
      if (a <= 0) h = s = l = NaN;
      else if (l <= 0 || l >= 1) h = s = NaN;
      else if (s <= 0) h = NaN;
      return new Hsl$6(h, s, l, a);
    }

    function hslConvert$6(o) {
      if (o instanceof Hsl$6) return new Hsl$6(o.h, o.s, o.l, o.opacity);
      if (!(o instanceof Color$6)) o = color$6(o);
      if (!o) return new Hsl$6;
      if (o instanceof Hsl$6) return o;
      o = o.rgb();
      var r = o.r / 255,
          g = o.g / 255,
          b = o.b / 255,
          min = Math.min(r, g, b),
          max = Math.max(r, g, b),
          h = NaN,
          s = max - min,
          l = (max + min) / 2;
      if (s) {
        if (r === max) h = (g - b) / s + (g < b) * 6;
        else if (g === max) h = (b - r) / s + 2;
        else h = (r - g) / s + 4;
        s /= l < 0.5 ? max + min : 2 - max - min;
        h *= 60;
      } else {
        s = l > 0 && l < 1 ? 0 : h;
      }
      return new Hsl$6(h, s, l, o.opacity);
    }

    function hsl$a(h, s, l, opacity) {
      return arguments.length === 1 ? hslConvert$6(h) : new Hsl$6(h, s, l, opacity == null ? 1 : opacity);
    }

    function Hsl$6(h, s, l, opacity) {
      this.h = +h;
      this.s = +s;
      this.l = +l;
      this.opacity = +opacity;
    }

    define$6(Hsl$6, hsl$a, extend$6(Color$6, {
      brighter: function(k) {
        k = k == null ? brighter$6 : Math.pow(brighter$6, k);
        return new Hsl$6(this.h, this.s, this.l * k, this.opacity);
      },
      darker: function(k) {
        k = k == null ? darker$6 : Math.pow(darker$6, k);
        return new Hsl$6(this.h, this.s, this.l * k, this.opacity);
      },
      rgb: function() {
        var h = this.h % 360 + (this.h < 0) * 360,
            s = isNaN(h) || isNaN(this.s) ? 0 : this.s,
            l = this.l,
            m2 = l + (l < 0.5 ? l : 1 - l) * s,
            m1 = 2 * l - m2;
        return new Rgb$6(
          hsl2rgb$6(h >= 240 ? h - 240 : h + 120, m1, m2),
          hsl2rgb$6(h, m1, m2),
          hsl2rgb$6(h < 120 ? h + 240 : h - 120, m1, m2),
          this.opacity
        );
      },
      displayable: function() {
        return (0 <= this.s && this.s <= 1 || isNaN(this.s))
            && (0 <= this.l && this.l <= 1)
            && (0 <= this.opacity && this.opacity <= 1);
      },
      formatHsl: function() {
        var a = this.opacity; a = isNaN(a) ? 1 : Math.max(0, Math.min(1, a));
        return (a === 1 ? "hsl(" : "hsla(")
            + (this.h || 0) + ", "
            + (this.s || 0) * 100 + "%, "
            + (this.l || 0) * 100 + "%"
            + (a === 1 ? ")" : ", " + a + ")");
      }
    }));

    /* From FvD 13.37, CSS Color Module Level 3 */
    function hsl2rgb$6(h, m1, m2) {
      return (h < 60 ? m1 + (m2 - m1) * h / 60
          : h < 180 ? m2
          : h < 240 ? m1 + (m2 - m1) * (240 - h) / 60
          : m1) * 255;
    }

    function constant$x(x) {
      return function() {
        return x;
      };
    }

    function linear$9(a, d) {
      return function(t) {
        return a + t * d;
      };
    }

    function exponential$7(a, b, y) {
      return a = Math.pow(a, y), b = Math.pow(b, y) - a, y = 1 / y, function(t) {
        return Math.pow(a + t * b, y);
      };
    }

    function gamma$5(y) {
      return (y = +y) === 1 ? nogamma$5 : function(a, b) {
        return b - a ? exponential$7(a, b, y) : constant$x(isNaN(a) ? b : a);
      };
    }

    function nogamma$5(a, b) {
      var d = b - a;
      return d ? linear$9(a, d) : constant$x(isNaN(a) ? b : a);
    }

    var interpolateRgb$3 = (function rgbGamma(y) {
      var color = gamma$5(y);

      function rgb(start, end) {
        var r = color((start = rgb$8(start)).r, (end = rgb$8(end)).r),
            g = color(start.g, end.g),
            b = color(start.b, end.b),
            opacity = nogamma$5(start.opacity, end.opacity);
        return function(t) {
          start.r = r(t);
          start.g = g(t);
          start.b = b(t);
          start.opacity = opacity(t);
          return start + "";
        };
      }

      rgb.gamma = rgbGamma;

      return rgb;
    })(1);

    function interpolateNumber$3(a, b) {
      return a = +a, b = +b, function(t) {
        return a * (1 - t) + b * t;
      };
    }

    var reA$5 = /[-+]?(?:\d+\.?\d*|\.?\d+)(?:[eE][-+]?\d+)?/g,
        reB$5 = new RegExp(reA$5.source, "g");

    function zero$5(b) {
      return function() {
        return b;
      };
    }

    function one$5(b) {
      return function(t) {
        return b(t) + "";
      };
    }

    function interpolateString$3(a, b) {
      var bi = reA$5.lastIndex = reB$5.lastIndex = 0, // scan index for next number in b
          am, // current match in a
          bm, // current match in b
          bs, // string preceding current number in b, if any
          i = -1, // index in s
          s = [], // string constants and placeholders
          q = []; // number interpolators

      // Coerce inputs to strings.
      a = a + "", b = b + "";

      // Interpolate pairs of numbers in a & b.
      while ((am = reA$5.exec(a))
          && (bm = reB$5.exec(b))) {
        if ((bs = bm.index) > bi) { // a string precedes the next number in b
          bs = b.slice(bi, bs);
          if (s[i]) s[i] += bs; // coalesce with previous string
          else s[++i] = bs;
        }
        if ((am = am[0]) === (bm = bm[0])) { // numbers in a & b match
          if (s[i]) s[i] += bm; // coalesce with previous string
          else s[++i] = bm;
        } else { // interpolate non-matching numbers
          s[++i] = null;
          q.push({i: i, x: interpolateNumber$3(am, bm)});
        }
        bi = reB$5.lastIndex;
      }

      // Add remains of b.
      if (bi < b.length) {
        bs = b.slice(bi);
        if (s[i]) s[i] += bs; // coalesce with previous string
        else s[++i] = bs;
      }

      // Special optimization for only a single match.
      // Otherwise, interpolate each of the numbers and rejoin the string.
      return s.length < 2 ? (q[0]
          ? one$5(q[0].x)
          : zero$5(b))
          : (b = q.length, function(t) {
              for (var i = 0, o; i < b; ++i) s[(o = q[i]).i] = o.x(t);
              return s.join("");
            });
    }

    var degrees$6 = 180 / Math.PI;

    var identity$l = {
      translateX: 0,
      translateY: 0,
      rotate: 0,
      skewX: 0,
      scaleX: 1,
      scaleY: 1
    };

    function decompose$4(a, b, c, d, e, f) {
      var scaleX, scaleY, skewX;
      if (scaleX = Math.sqrt(a * a + b * b)) a /= scaleX, b /= scaleX;
      if (skewX = a * c + b * d) c -= a * skewX, d -= b * skewX;
      if (scaleY = Math.sqrt(c * c + d * d)) c /= scaleY, d /= scaleY, skewX /= scaleY;
      if (a * d < b * c) a = -a, b = -b, skewX = -skewX, scaleX = -scaleX;
      return {
        translateX: e,
        translateY: f,
        rotate: Math.atan2(b, a) * degrees$6,
        skewX: Math.atan(skewX) * degrees$6,
        scaleX: scaleX,
        scaleY: scaleY
      };
    }

    var cssNode$4,
        cssRoot$4,
        cssView$4,
        svgNode$4;

    function parseCss$5(value) {
      if (value === "none") return identity$l;
      if (!cssNode$4) cssNode$4 = document.createElement("DIV"), cssRoot$4 = document.documentElement, cssView$4 = document.defaultView;
      cssNode$4.style.transform = value;
      value = cssView$4.getComputedStyle(cssRoot$4.appendChild(cssNode$4), null).getPropertyValue("transform");
      cssRoot$4.removeChild(cssNode$4);
      value = value.slice(7, -1).split(",");
      return decompose$4(+value[0], +value[1], +value[2], +value[3], +value[4], +value[5]);
    }

    function parseSvg$4(value) {
      if (value == null) return identity$l;
      if (!svgNode$4) svgNode$4 = document.createElementNS("http://www.w3.org/2000/svg", "g");
      svgNode$4.setAttribute("transform", value);
      if (!(value = svgNode$4.transform.baseVal.consolidate())) return identity$l;
      value = value.matrix;
      return decompose$4(value.a, value.b, value.c, value.d, value.e, value.f);
    }

    function interpolateTransform$4(parse, pxComma, pxParen, degParen) {

      function pop(s) {
        return s.length ? s.pop() + " " : "";
      }

      function translate(xa, ya, xb, yb, s, q) {
        if (xa !== xb || ya !== yb) {
          var i = s.push("translate(", null, pxComma, null, pxParen);
          q.push({i: i - 4, x: interpolateNumber$3(xa, xb)}, {i: i - 2, x: interpolateNumber$3(ya, yb)});
        } else if (xb || yb) {
          s.push("translate(" + xb + pxComma + yb + pxParen);
        }
      }

      function rotate(a, b, s, q) {
        if (a !== b) {
          if (a - b > 180) b += 360; else if (b - a > 180) a += 360; // shortest path
          q.push({i: s.push(pop(s) + "rotate(", null, degParen) - 2, x: interpolateNumber$3(a, b)});
        } else if (b) {
          s.push(pop(s) + "rotate(" + b + degParen);
        }
      }

      function skewX(a, b, s, q) {
        if (a !== b) {
          q.push({i: s.push(pop(s) + "skewX(", null, degParen) - 2, x: interpolateNumber$3(a, b)});
        } else if (b) {
          s.push(pop(s) + "skewX(" + b + degParen);
        }
      }

      function scale(xa, ya, xb, yb, s, q) {
        if (xa !== xb || ya !== yb) {
          var i = s.push(pop(s) + "scale(", null, ",", null, ")");
          q.push({i: i - 4, x: interpolateNumber$3(xa, xb)}, {i: i - 2, x: interpolateNumber$3(ya, yb)});
        } else if (xb !== 1 || yb !== 1) {
          s.push(pop(s) + "scale(" + xb + "," + yb + ")");
        }
      }

      return function(a, b) {
        var s = [], // string constants and placeholders
            q = []; // number interpolators
        a = parse(a), b = parse(b);
        translate(a.translateX, a.translateY, b.translateX, b.translateY, s, q);
        rotate(a.rotate, b.rotate, s, q);
        skewX(a.skewX, b.skewX, s, q);
        scale(a.scaleX, a.scaleY, b.scaleX, b.scaleY, s, q);
        a = b = null; // gc
        return function(t) {
          var i = -1, n = q.length, o;
          while (++i < n) s[(o = q[i]).i] = o.x(t);
          return s.join("");
        };
      };
    }

    var interpolateTransformCss$4 = interpolateTransform$4(parseCss$5, "px, ", "px)", "deg)");
    var interpolateTransformSvg$4 = interpolateTransform$4(parseSvg$4, ", ", ")", ")");

    var rho$2 = Math.SQRT2,
        rho2$2 = 2,
        rho4$2 = 4,
        epsilon2$6 = 1e-12;

    function cosh$2(x) {
      return ((x = Math.exp(x)) + 1 / x) / 2;
    }

    function sinh$2(x) {
      return ((x = Math.exp(x)) - 1 / x) / 2;
    }

    function tanh$2(x) {
      return ((x = Math.exp(2 * x)) - 1) / (x + 1);
    }

    // p0 = [ux0, uy0, w0]
    // p1 = [ux1, uy1, w1]
    function interpolateZoom$1(p0, p1) {
      var ux0 = p0[0], uy0 = p0[1], w0 = p0[2],
          ux1 = p1[0], uy1 = p1[1], w1 = p1[2],
          dx = ux1 - ux0,
          dy = uy1 - uy0,
          d2 = dx * dx + dy * dy,
          i,
          S;

      // Special case for u0 ≅ u1.
      if (d2 < epsilon2$6) {
        S = Math.log(w1 / w0) / rho$2;
        i = function(t) {
          return [
            ux0 + t * dx,
            uy0 + t * dy,
            w0 * Math.exp(rho$2 * t * S)
          ];
        };
      }

      // General case.
      else {
        var d1 = Math.sqrt(d2),
            b0 = (w1 * w1 - w0 * w0 + rho4$2 * d2) / (2 * w0 * rho2$2 * d1),
            b1 = (w1 * w1 - w0 * w0 - rho4$2 * d2) / (2 * w1 * rho2$2 * d1),
            r0 = Math.log(Math.sqrt(b0 * b0 + 1) - b0),
            r1 = Math.log(Math.sqrt(b1 * b1 + 1) - b1);
        S = (r1 - r0) / rho$2;
        i = function(t) {
          var s = t * S,
              coshr0 = cosh$2(r0),
              u = w0 / (rho2$2 * d1) * (coshr0 * tanh$2(rho$2 * s + r0) - sinh$2(r0));
          return [
            ux0 + u * dx,
            uy0 + u * dy,
            w0 * coshr0 / cosh$2(rho$2 * s + r0)
          ];
        };
      }

      i.duration = S * 1000;

      return i;
    }

    var frame$5 = 0, // is an animation frame pending?
        timeout$9 = 0, // is a timeout pending?
        interval$7 = 0, // are any timers active?
        pokeDelay$5 = 1000, // how frequently we check for clock skew
        taskHead$5,
        taskTail$5,
        clockLast$5 = 0,
        clockNow$5 = 0,
        clockSkew$5 = 0,
        clock$5 = typeof performance === "object" && performance.now ? performance : Date,
        setFrame$5 = typeof window === "object" && window.requestAnimationFrame ? window.requestAnimationFrame.bind(window) : function(f) { setTimeout(f, 17); };

    function now$5() {
      return clockNow$5 || (setFrame$5(clearNow$5), clockNow$5 = clock$5.now() + clockSkew$5);
    }

    function clearNow$5() {
      clockNow$5 = 0;
    }

    function Timer$5() {
      this._call =
      this._time =
      this._next = null;
    }

    Timer$5.prototype = timer$5.prototype = {
      constructor: Timer$5,
      restart: function(callback, delay, time) {
        if (typeof callback !== "function") throw new TypeError("callback is not a function");
        time = (time == null ? now$5() : +time) + (delay == null ? 0 : +delay);
        if (!this._next && taskTail$5 !== this) {
          if (taskTail$5) taskTail$5._next = this;
          else taskHead$5 = this;
          taskTail$5 = this;
        }
        this._call = callback;
        this._time = time;
        sleep$5();
      },
      stop: function() {
        if (this._call) {
          this._call = null;
          this._time = Infinity;
          sleep$5();
        }
      }
    };

    function timer$5(callback, delay, time) {
      var t = new Timer$5;
      t.restart(callback, delay, time);
      return t;
    }

    function timerFlush$5() {
      now$5(); // Get the current time, if not already set.
      ++frame$5; // Pretend we’ve set an alarm, if we haven’t already.
      var t = taskHead$5, e;
      while (t) {
        if ((e = clockNow$5 - t._time) >= 0) t._call.call(null, e);
        t = t._next;
      }
      --frame$5;
    }

    function wake$5() {
      clockNow$5 = (clockLast$5 = clock$5.now()) + clockSkew$5;
      frame$5 = timeout$9 = 0;
      try {
        timerFlush$5();
      } finally {
        frame$5 = 0;
        nap$5();
        clockNow$5 = 0;
      }
    }

    function poke$6() {
      var now = clock$5.now(), delay = now - clockLast$5;
      if (delay > pokeDelay$5) clockSkew$5 -= delay, clockLast$5 = now;
    }

    function nap$5() {
      var t0, t1 = taskHead$5, t2, time = Infinity;
      while (t1) {
        if (t1._call) {
          if (time > t1._time) time = t1._time;
          t0 = t1, t1 = t1._next;
        } else {
          t2 = t1._next, t1._next = null;
          t1 = t0 ? t0._next = t2 : taskHead$5 = t2;
        }
      }
      taskTail$5 = t0;
      sleep$5(time);
    }

    function sleep$5(time) {
      if (frame$5) return; // Soonest alarm already set, or will be.
      if (timeout$9) timeout$9 = clearTimeout(timeout$9);
      var delay = time - clockNow$5; // Strictly less than if we recomputed clockNow.
      if (delay > 24) {
        if (time < Infinity) timeout$9 = setTimeout(wake$5, time - clock$5.now() - clockSkew$5);
        if (interval$7) interval$7 = clearInterval(interval$7);
      } else {
        if (!interval$7) clockLast$5 = clock$5.now(), interval$7 = setInterval(poke$6, pokeDelay$5);
        frame$5 = 1, setFrame$5(wake$5);
      }
    }

    function timeout$a(callback, delay, time) {
      var t = new Timer$5;
      delay = delay == null ? 0 : +delay;
      t.restart(function(elapsed) {
        t.stop();
        callback(elapsed + delay);
      }, delay, time);
      return t;
    }

    var emptyOn$3 = dispatch$6("start", "end", "cancel", "interrupt");
    var emptyTween$3 = [];

    var CREATED$3 = 0;
    var SCHEDULED$3 = 1;
    var STARTING$3 = 2;
    var STARTED$3 = 3;
    var RUNNING$3 = 4;
    var ENDING$3 = 5;
    var ENDED$3 = 6;

    function schedule$4(node, name, id, index, group, timing) {
      var schedules = node.__transition;
      if (!schedules) node.__transition = {};
      else if (id in schedules) return;
      create$5(node, id, {
        name: name,
        index: index, // For context during callback.
        group: group, // For context during callback.
        on: emptyOn$3,
        tween: emptyTween$3,
        time: timing.time,
        delay: timing.delay,
        duration: timing.duration,
        ease: timing.ease,
        timer: null,
        state: CREATED$3
      });
    }

    function init$3(node, id) {
      var schedule = get$b(node, id);
      if (schedule.state > CREATED$3) throw new Error("too late; already scheduled");
      return schedule;
    }

    function set$f(node, id) {
      var schedule = get$b(node, id);
      if (schedule.state > STARTED$3) throw new Error("too late; already running");
      return schedule;
    }

    function get$b(node, id) {
      var schedule = node.__transition;
      if (!schedule || !(schedule = schedule[id])) throw new Error("transition not found");
      return schedule;
    }

    function create$5(node, id, self) {
      var schedules = node.__transition,
          tween;

      // Initialize the self timer when the transition is created.
      // Note the actual delay is not known until the first callback!
      schedules[id] = self;
      self.timer = timer$5(schedule, 0, self.time);

      function schedule(elapsed) {
        self.state = SCHEDULED$3;
        self.timer.restart(start, self.delay, self.time);

        // If the elapsed delay is less than our first sleep, start immediately.
        if (self.delay <= elapsed) start(elapsed - self.delay);
      }

      function start(elapsed) {
        var i, j, n, o;

        // If the state is not SCHEDULED, then we previously errored on start.
        if (self.state !== SCHEDULED$3) return stop();

        for (i in schedules) {
          o = schedules[i];
          if (o.name !== self.name) continue;

          // While this element already has a starting transition during this frame,
          // defer starting an interrupting transition until that transition has a
          // chance to tick (and possibly end); see d3/d3-transition#54!
          if (o.state === STARTED$3) return timeout$a(start);

          // Interrupt the active transition, if any.
          if (o.state === RUNNING$3) {
            o.state = ENDED$3;
            o.timer.stop();
            o.on.call("interrupt", node, node.__data__, o.index, o.group);
            delete schedules[i];
          }

          // Cancel any pre-empted transitions.
          else if (+i < id) {
            o.state = ENDED$3;
            o.timer.stop();
            o.on.call("cancel", node, node.__data__, o.index, o.group);
            delete schedules[i];
          }
        }

        // Defer the first tick to end of the current frame; see d3/d3#1576.
        // Note the transition may be canceled after start and before the first tick!
        // Note this must be scheduled before the start event; see d3/d3-transition#16!
        // Assuming this is successful, subsequent callbacks go straight to tick.
        timeout$a(function() {
          if (self.state === STARTED$3) {
            self.state = RUNNING$3;
            self.timer.restart(tick, self.delay, self.time);
            tick(elapsed);
          }
        });

        // Dispatch the start event.
        // Note this must be done before the tween are initialized.
        self.state = STARTING$3;
        self.on.call("start", node, node.__data__, self.index, self.group);
        if (self.state !== STARTING$3) return; // interrupted
        self.state = STARTED$3;

        // Initialize the tween, deleting null tween.
        tween = new Array(n = self.tween.length);
        for (i = 0, j = -1; i < n; ++i) {
          if (o = self.tween[i].value.call(node, node.__data__, self.index, self.group)) {
            tween[++j] = o;
          }
        }
        tween.length = j + 1;
      }

      function tick(elapsed) {
        var t = elapsed < self.duration ? self.ease.call(null, elapsed / self.duration) : (self.timer.restart(stop), self.state = ENDING$3, 1),
            i = -1,
            n = tween.length;

        while (++i < n) {
          tween[i].call(node, t);
        }

        // Dispatch the end event.
        if (self.state === ENDING$3) {
          self.on.call("end", node, node.__data__, self.index, self.group);
          stop();
        }
      }

      function stop() {
        self.state = ENDED$3;
        self.timer.stop();
        delete schedules[id];
        for (var i in schedules) return; // eslint-disable-line no-unused-vars
        delete node.__transition;
      }
    }

    function interrupt$3(node, name) {
      var schedules = node.__transition,
          schedule,
          active,
          empty = true,
          i;

      if (!schedules) return;

      name = name == null ? null : name + "";

      for (i in schedules) {
        if ((schedule = schedules[i]).name !== name) { empty = false; continue; }
        active = schedule.state > STARTING$3 && schedule.state < ENDING$3;
        schedule.state = ENDED$3;
        schedule.timer.stop();
        schedule.on.call(active ? "interrupt" : "cancel", node, node.__data__, schedule.index, schedule.group);
        delete schedules[i];
      }

      if (empty) delete node.__transition;
    }

    function selection_interrupt$3(name) {
      return this.each(function() {
        interrupt$3(this, name);
      });
    }

    function tweenRemove$3(id, name) {
      var tween0, tween1;
      return function() {
        var schedule = set$f(this, id),
            tween = schedule.tween;

        // If this node shared tween with the previous node,
        // just assign the updated shared tween and we’re done!
        // Otherwise, copy-on-write.
        if (tween !== tween0) {
          tween1 = tween0 = tween;
          for (var i = 0, n = tween1.length; i < n; ++i) {
            if (tween1[i].name === name) {
              tween1 = tween1.slice();
              tween1.splice(i, 1);
              break;
            }
          }
        }

        schedule.tween = tween1;
      };
    }

    function tweenFunction$3(id, name, value) {
      var tween0, tween1;
      if (typeof value !== "function") throw new Error;
      return function() {
        var schedule = set$f(this, id),
            tween = schedule.tween;

        // If this node shared tween with the previous node,
        // just assign the updated shared tween and we’re done!
        // Otherwise, copy-on-write.
        if (tween !== tween0) {
          tween1 = (tween0 = tween).slice();
          for (var t = {name: name, value: value}, i = 0, n = tween1.length; i < n; ++i) {
            if (tween1[i].name === name) {
              tween1[i] = t;
              break;
            }
          }
          if (i === n) tween1.push(t);
        }

        schedule.tween = tween1;
      };
    }

    function transition_tween$3(name, value) {
      var id = this._id;

      name += "";

      if (arguments.length < 2) {
        var tween = get$b(this.node(), id).tween;
        for (var i = 0, n = tween.length, t; i < n; ++i) {
          if ((t = tween[i]).name === name) {
            return t.value;
          }
        }
        return null;
      }

      return this.each((value == null ? tweenRemove$3 : tweenFunction$3)(id, name, value));
    }

    function tweenValue$3(transition, name, value) {
      var id = transition._id;

      transition.each(function() {
        var schedule = set$f(this, id);
        (schedule.value || (schedule.value = {}))[name] = value.apply(this, arguments);
      });

      return function(node) {
        return get$b(node, id).value[name];
      };
    }

    function interpolate$6(a, b) {
      var c;
      return (typeof b === "number" ? interpolateNumber$3
          : b instanceof color$6 ? interpolateRgb$3
          : (c = color$6(b)) ? (b = c, interpolateRgb$3)
          : interpolateString$3)(a, b);
    }

    function attrRemove$9(name) {
      return function() {
        this.removeAttribute(name);
      };
    }

    function attrRemoveNS$9(fullname) {
      return function() {
        this.removeAttributeNS(fullname.space, fullname.local);
      };
    }

    function attrConstant$9(name, interpolate, value1) {
      var string00,
          string1 = value1 + "",
          interpolate0;
      return function() {
        var string0 = this.getAttribute(name);
        return string0 === string1 ? null
            : string0 === string00 ? interpolate0
            : interpolate0 = interpolate(string00 = string0, value1);
      };
    }

    function attrConstantNS$9(fullname, interpolate, value1) {
      var string00,
          string1 = value1 + "",
          interpolate0;
      return function() {
        var string0 = this.getAttributeNS(fullname.space, fullname.local);
        return string0 === string1 ? null
            : string0 === string00 ? interpolate0
            : interpolate0 = interpolate(string00 = string0, value1);
      };
    }

    function attrFunction$9(name, interpolate, value) {
      var string00,
          string10,
          interpolate0;
      return function() {
        var string0, value1 = value(this), string1;
        if (value1 == null) return void this.removeAttribute(name);
        string0 = this.getAttribute(name);
        string1 = value1 + "";
        return string0 === string1 ? null
            : string0 === string00 && string1 === string10 ? interpolate0
            : (string10 = string1, interpolate0 = interpolate(string00 = string0, value1));
      };
    }

    function attrFunctionNS$9(fullname, interpolate, value) {
      var string00,
          string10,
          interpolate0;
      return function() {
        var string0, value1 = value(this), string1;
        if (value1 == null) return void this.removeAttributeNS(fullname.space, fullname.local);
        string0 = this.getAttributeNS(fullname.space, fullname.local);
        string1 = value1 + "";
        return string0 === string1 ? null
            : string0 === string00 && string1 === string10 ? interpolate0
            : (string10 = string1, interpolate0 = interpolate(string00 = string0, value1));
      };
    }

    function transition_attr$3(name, value) {
      var fullname = namespace$5(name), i = fullname === "transform" ? interpolateTransformSvg$4 : interpolate$6;
      return this.attrTween(name, typeof value === "function"
          ? (fullname.local ? attrFunctionNS$9 : attrFunction$9)(fullname, i, tweenValue$3(this, "attr." + name, value))
          : value == null ? (fullname.local ? attrRemoveNS$9 : attrRemove$9)(fullname)
          : (fullname.local ? attrConstantNS$9 : attrConstant$9)(fullname, i, value));
    }

    function attrInterpolate$2(name, i) {
      return function(t) {
        this.setAttribute(name, i.call(this, t));
      };
    }

    function attrInterpolateNS$2(fullname, i) {
      return function(t) {
        this.setAttributeNS(fullname.space, fullname.local, i.call(this, t));
      };
    }

    function attrTweenNS$3(fullname, value) {
      var t0, i0;
      function tween() {
        var i = value.apply(this, arguments);
        if (i !== i0) t0 = (i0 = i) && attrInterpolateNS$2(fullname, i);
        return t0;
      }
      tween._value = value;
      return tween;
    }

    function attrTween$3(name, value) {
      var t0, i0;
      function tween() {
        var i = value.apply(this, arguments);
        if (i !== i0) t0 = (i0 = i) && attrInterpolate$2(name, i);
        return t0;
      }
      tween._value = value;
      return tween;
    }

    function transition_attrTween$3(name, value) {
      var key = "attr." + name;
      if (arguments.length < 2) return (key = this.tween(key)) && key._value;
      if (value == null) return this.tween(key, null);
      if (typeof value !== "function") throw new Error;
      var fullname = namespace$5(name);
      return this.tween(key, (fullname.local ? attrTweenNS$3 : attrTween$3)(fullname, value));
    }

    function delayFunction$3(id, value) {
      return function() {
        init$3(this, id).delay = +value.apply(this, arguments);
      };
    }

    function delayConstant$3(id, value) {
      return value = +value, function() {
        init$3(this, id).delay = value;
      };
    }

    function transition_delay$3(value) {
      var id = this._id;

      return arguments.length
          ? this.each((typeof value === "function"
              ? delayFunction$3
              : delayConstant$3)(id, value))
          : get$b(this.node(), id).delay;
    }

    function durationFunction$3(id, value) {
      return function() {
        set$f(this, id).duration = +value.apply(this, arguments);
      };
    }

    function durationConstant$3(id, value) {
      return value = +value, function() {
        set$f(this, id).duration = value;
      };
    }

    function transition_duration$3(value) {
      var id = this._id;

      return arguments.length
          ? this.each((typeof value === "function"
              ? durationFunction$3
              : durationConstant$3)(id, value))
          : get$b(this.node(), id).duration;
    }

    function easeConstant$3(id, value) {
      if (typeof value !== "function") throw new Error;
      return function() {
        set$f(this, id).ease = value;
      };
    }

    function transition_ease$3(value) {
      var id = this._id;

      return arguments.length
          ? this.each(easeConstant$3(id, value))
          : get$b(this.node(), id).ease;
    }

    function transition_filter$3(match) {
      if (typeof match !== "function") match = matcher$6(match);

      for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, subgroup = subgroups[j] = [], node, i = 0; i < n; ++i) {
          if ((node = group[i]) && match.call(node, node.__data__, i, group)) {
            subgroup.push(node);
          }
        }
      }

      return new Transition$3(subgroups, this._parents, this._name, this._id);
    }

    function transition_merge$3(transition) {
      if (transition._id !== this._id) throw new Error;

      for (var groups0 = this._groups, groups1 = transition._groups, m0 = groups0.length, m1 = groups1.length, m = Math.min(m0, m1), merges = new Array(m0), j = 0; j < m; ++j) {
        for (var group0 = groups0[j], group1 = groups1[j], n = group0.length, merge = merges[j] = new Array(n), node, i = 0; i < n; ++i) {
          if (node = group0[i] || group1[i]) {
            merge[i] = node;
          }
        }
      }

      for (; j < m0; ++j) {
        merges[j] = groups0[j];
      }

      return new Transition$3(merges, this._parents, this._name, this._id);
    }

    function start$4(name) {
      return (name + "").trim().split(/^|\s+/).every(function(t) {
        var i = t.indexOf(".");
        if (i >= 0) t = t.slice(0, i);
        return !t || t === "start";
      });
    }

    function onFunction$3(id, name, listener) {
      var on0, on1, sit = start$4(name) ? init$3 : set$f;
      return function() {
        var schedule = sit(this, id),
            on = schedule.on;

        // If this node shared a dispatch with the previous node,
        // just assign the updated shared dispatch and we’re done!
        // Otherwise, copy-on-write.
        if (on !== on0) (on1 = (on0 = on).copy()).on(name, listener);

        schedule.on = on1;
      };
    }

    function transition_on$3(name, listener) {
      var id = this._id;

      return arguments.length < 2
          ? get$b(this.node(), id).on.on(name)
          : this.each(onFunction$3(id, name, listener));
    }

    function removeFunction$3(id) {
      return function() {
        var parent = this.parentNode;
        for (var i in this.__transition) if (+i !== id) return;
        if (parent) parent.removeChild(this);
      };
    }

    function transition_remove$3() {
      return this.on("end.remove", removeFunction$3(this._id));
    }

    function transition_select$3(select) {
      var name = this._name,
          id = this._id;

      if (typeof select !== "function") select = selector$5(select);

      for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, subgroup = subgroups[j] = new Array(n), node, subnode, i = 0; i < n; ++i) {
          if ((node = group[i]) && (subnode = select.call(node, node.__data__, i, group))) {
            if ("__data__" in node) subnode.__data__ = node.__data__;
            subgroup[i] = subnode;
            schedule$4(subgroup[i], name, id, i, subgroup, get$b(node, id));
          }
        }
      }

      return new Transition$3(subgroups, this._parents, name, id);
    }

    function transition_selectAll$3(select) {
      var name = this._name,
          id = this._id;

      if (typeof select !== "function") select = selectorAll$5(select);

      for (var groups = this._groups, m = groups.length, subgroups = [], parents = [], j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, node, i = 0; i < n; ++i) {
          if (node = group[i]) {
            for (var children = select.call(node, node.__data__, i, group), child, inherit = get$b(node, id), k = 0, l = children.length; k < l; ++k) {
              if (child = children[k]) {
                schedule$4(child, name, id, k, children, inherit);
              }
            }
            subgroups.push(children);
            parents.push(node);
          }
        }
      }

      return new Transition$3(subgroups, parents, name, id);
    }

    var Selection$9 = selection$5.prototype.constructor;

    function transition_selection$3() {
      return new Selection$9(this._groups, this._parents);
    }

    function styleNull$2(name, interpolate) {
      var string00,
          string10,
          interpolate0;
      return function() {
        var string0 = styleValue$5(this, name),
            string1 = (this.style.removeProperty(name), styleValue$5(this, name));
        return string0 === string1 ? null
            : string0 === string00 && string1 === string10 ? interpolate0
            : interpolate0 = interpolate(string00 = string0, string10 = string1);
      };
    }

    function styleRemove$9(name) {
      return function() {
        this.style.removeProperty(name);
      };
    }

    function styleConstant$9(name, interpolate, value1) {
      var string00,
          string1 = value1 + "",
          interpolate0;
      return function() {
        var string0 = styleValue$5(this, name);
        return string0 === string1 ? null
            : string0 === string00 ? interpolate0
            : interpolate0 = interpolate(string00 = string0, value1);
      };
    }

    function styleFunction$9(name, interpolate, value) {
      var string00,
          string10,
          interpolate0;
      return function() {
        var string0 = styleValue$5(this, name),
            value1 = value(this),
            string1 = value1 + "";
        if (value1 == null) string1 = value1 = (this.style.removeProperty(name), styleValue$5(this, name));
        return string0 === string1 ? null
            : string0 === string00 && string1 === string10 ? interpolate0
            : (string10 = string1, interpolate0 = interpolate(string00 = string0, value1));
      };
    }

    function styleMaybeRemove$2(id, name) {
      var on0, on1, listener0, key = "style." + name, event = "end." + key, remove;
      return function() {
        var schedule = set$f(this, id),
            on = schedule.on,
            listener = schedule.value[key] == null ? remove || (remove = styleRemove$9(name)) : undefined;

        // If this node shared a dispatch with the previous node,
        // just assign the updated shared dispatch and we’re done!
        // Otherwise, copy-on-write.
        if (on !== on0 || listener0 !== listener) (on1 = (on0 = on).copy()).on(event, listener0 = listener);

        schedule.on = on1;
      };
    }

    function transition_style$3(name, value, priority) {
      var i = (name += "") === "transform" ? interpolateTransformCss$4 : interpolate$6;
      return value == null ? this
          .styleTween(name, styleNull$2(name, i))
          .on("end.style." + name, styleRemove$9(name))
        : typeof value === "function" ? this
          .styleTween(name, styleFunction$9(name, i, tweenValue$3(this, "style." + name, value)))
          .each(styleMaybeRemove$2(this._id, name))
        : this
          .styleTween(name, styleConstant$9(name, i, value), priority)
          .on("end.style." + name, null);
    }

    function styleInterpolate$2(name, i, priority) {
      return function(t) {
        this.style.setProperty(name, i.call(this, t), priority);
      };
    }

    function styleTween$3(name, value, priority) {
      var t, i0;
      function tween() {
        var i = value.apply(this, arguments);
        if (i !== i0) t = (i0 = i) && styleInterpolate$2(name, i, priority);
        return t;
      }
      tween._value = value;
      return tween;
    }

    function transition_styleTween$3(name, value, priority) {
      var key = "style." + (name += "");
      if (arguments.length < 2) return (key = this.tween(key)) && key._value;
      if (value == null) return this.tween(key, null);
      if (typeof value !== "function") throw new Error;
      return this.tween(key, styleTween$3(name, value, priority == null ? "" : priority));
    }

    function textConstant$9(value) {
      return function() {
        this.textContent = value;
      };
    }

    function textFunction$9(value) {
      return function() {
        var value1 = value(this);
        this.textContent = value1 == null ? "" : value1;
      };
    }

    function transition_text$3(value) {
      return this.tween("text", typeof value === "function"
          ? textFunction$9(tweenValue$3(this, "text", value))
          : textConstant$9(value == null ? "" : value + ""));
    }

    function textInterpolate$2(i) {
      return function(t) {
        this.textContent = i.call(this, t);
      };
    }

    function textTween$2(value) {
      var t0, i0;
      function tween() {
        var i = value.apply(this, arguments);
        if (i !== i0) t0 = (i0 = i) && textInterpolate$2(i);
        return t0;
      }
      tween._value = value;
      return tween;
    }

    function transition_textTween$2(value) {
      var key = "text";
      if (arguments.length < 1) return (key = this.tween(key)) && key._value;
      if (value == null) return this.tween(key, null);
      if (typeof value !== "function") throw new Error;
      return this.tween(key, textTween$2(value));
    }

    function transition_transition$3() {
      var name = this._name,
          id0 = this._id,
          id1 = newId$3();

      for (var groups = this._groups, m = groups.length, j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, node, i = 0; i < n; ++i) {
          if (node = group[i]) {
            var inherit = get$b(node, id0);
            schedule$4(node, name, id1, i, group, {
              time: inherit.time + inherit.delay + inherit.duration,
              delay: 0,
              duration: inherit.duration,
              ease: inherit.ease
            });
          }
        }
      }

      return new Transition$3(groups, this._parents, name, id1);
    }

    function transition_end$2() {
      var on0, on1, that = this, id = that._id, size = that.size();
      return new Promise(function(resolve, reject) {
        var cancel = {value: reject},
            end = {value: function() { if (--size === 0) resolve(); }};

        that.each(function() {
          var schedule = set$f(this, id),
              on = schedule.on;

          // If this node shared a dispatch with the previous node,
          // just assign the updated shared dispatch and we’re done!
          // Otherwise, copy-on-write.
          if (on !== on0) {
            on1 = (on0 = on).copy();
            on1._.cancel.push(cancel);
            on1._.interrupt.push(cancel);
            on1._.end.push(end);
          }

          schedule.on = on1;
        });
      });
    }

    var id$3 = 0;

    function Transition$3(groups, parents, name, id) {
      this._groups = groups;
      this._parents = parents;
      this._name = name;
      this._id = id;
    }

    function transition$3(name) {
      return selection$5().transition(name);
    }

    function newId$3() {
      return ++id$3;
    }

    var selection_prototype$3 = selection$5.prototype;

    Transition$3.prototype = transition$3.prototype = {
      constructor: Transition$3,
      select: transition_select$3,
      selectAll: transition_selectAll$3,
      filter: transition_filter$3,
      merge: transition_merge$3,
      selection: transition_selection$3,
      transition: transition_transition$3,
      call: selection_prototype$3.call,
      nodes: selection_prototype$3.nodes,
      node: selection_prototype$3.node,
      size: selection_prototype$3.size,
      empty: selection_prototype$3.empty,
      each: selection_prototype$3.each,
      on: transition_on$3,
      attr: transition_attr$3,
      attrTween: transition_attrTween$3,
      style: transition_style$3,
      styleTween: transition_styleTween$3,
      text: transition_text$3,
      textTween: transition_textTween$2,
      remove: transition_remove$3,
      tween: transition_tween$3,
      delay: transition_delay$3,
      duration: transition_duration$3,
      ease: transition_ease$3,
      end: transition_end$2
    };

    var defaultTiming$3 = {
      time: null, // Set on use.
      delay: 0,
      duration: 250,
      ease: cubicInOut
    };

    function inherit$3(node, id) {
      var timing;
      while (!(timing = node.__transition) || !(timing = timing[id])) {
        if (!(node = node.parentNode)) {
          return defaultTiming$3.time = now$5(), defaultTiming$3;
        }
      }
      return timing;
    }

    function selection_transition$3(name) {
      var id,
          timing;

      if (name instanceof Transition$3) {
        id = name._id, name = name._name;
      } else {
        id = newId$3(), (timing = defaultTiming$3).time = now$5(), name = name == null ? null : name + "";
      }

      for (var groups = this._groups, m = groups.length, j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, node, i = 0; i < n; ++i) {
          if (node = group[i]) {
            schedule$4(node, name, id, i, group, timing || inherit$3(node, id));
          }
        }
      }

      return new Transition$3(groups, this._parents, name, id);
    }

    selection$5.prototype.interrupt = selection_interrupt$3;
    selection$5.prototype.transition = selection_transition$3;

    function constant$y(x) {
      return function() {
        return x;
      };
    }

    function ZoomEvent$1(target, type, transform) {
      this.target = target;
      this.type = type;
      this.transform = transform;
    }

    function Transform$1(k, x, y) {
      this.k = k;
      this.x = x;
      this.y = y;
    }

    Transform$1.prototype = {
      constructor: Transform$1,
      scale: function(k) {
        return k === 1 ? this : new Transform$1(this.k * k, this.x, this.y);
      },
      translate: function(x, y) {
        return x === 0 & y === 0 ? this : new Transform$1(this.k, this.x + this.k * x, this.y + this.k * y);
      },
      apply: function(point) {
        return [point[0] * this.k + this.x, point[1] * this.k + this.y];
      },
      applyX: function(x) {
        return x * this.k + this.x;
      },
      applyY: function(y) {
        return y * this.k + this.y;
      },
      invert: function(location) {
        return [(location[0] - this.x) / this.k, (location[1] - this.y) / this.k];
      },
      invertX: function(x) {
        return (x - this.x) / this.k;
      },
      invertY: function(y) {
        return (y - this.y) / this.k;
      },
      rescaleX: function(x) {
        return x.copy().domain(x.range().map(this.invertX, this).map(x.invert, x));
      },
      rescaleY: function(y) {
        return y.copy().domain(y.range().map(this.invertY, this).map(y.invert, y));
      },
      toString: function() {
        return "translate(" + this.x + "," + this.y + ") scale(" + this.k + ")";
      }
    };

    var identity$m = new Transform$1(1, 0, 0);

    transform$3.prototype = Transform$1.prototype;

    function transform$3(node) {
      return node.__zoom || identity$m;
    }

    function nopropagation$5() {
      event$4.stopImmediatePropagation();
    }

    function noevent$7() {
      event$4.preventDefault();
      event$4.stopImmediatePropagation();
    }

    // Ignore right-click, since that should open the context menu.
    function defaultFilter$5() {
      return !event$4.button;
    }

    function defaultExtent$3() {
      var e = this, w, h;
      if (e instanceof SVGElement) {
        e = e.ownerSVGElement || e;
        w = e.width.baseVal.value;
        h = e.height.baseVal.value;
      } else {
        w = e.clientWidth;
        h = e.clientHeight;
      }
      return [[0, 0], [w, h]];
    }

    function defaultTransform$1() {
      return this.__zoom || identity$m;
    }

    function defaultWheelDelta$1() {
      return -event$4.deltaY * (event$4.deltaMode ? 120 : 1) / 500;
    }

    function defaultTouchable$4() {
      return "ontouchstart" in this;
    }

    function defaultConstrain$1(transform, extent, translateExtent) {
      var dx0 = transform.invertX(extent[0][0]) - translateExtent[0][0],
          dx1 = transform.invertX(extent[1][0]) - translateExtent[1][0],
          dy0 = transform.invertY(extent[0][1]) - translateExtent[0][1],
          dy1 = transform.invertY(extent[1][1]) - translateExtent[1][1];
      return transform.translate(
        dx1 > dx0 ? (dx0 + dx1) / 2 : Math.min(0, dx0) || Math.max(0, dx1),
        dy1 > dy0 ? (dy0 + dy1) / 2 : Math.min(0, dy0) || Math.max(0, dy1)
      );
    }

    function zoom$2() {
      var filter = defaultFilter$5,
          extent = defaultExtent$3,
          constrain = defaultConstrain$1,
          wheelDelta = defaultWheelDelta$1,
          touchable = defaultTouchable$4,
          scaleExtent = [0, Infinity],
          translateExtent = [[-Infinity, -Infinity], [Infinity, Infinity]],
          duration = 250,
          interpolate = interpolateZoom$1,
          gestures = [],
          listeners = dispatch$6("start", "zoom", "end"),
          touchstarting,
          touchending,
          touchDelay = 500,
          wheelDelay = 150,
          clickDistance2 = 0;

      function zoom(selection) {
        selection
            .property("__zoom", defaultTransform$1)
            .on("wheel.zoom", wheeled)
            .on("mousedown.zoom", mousedowned)
            .on("dblclick.zoom", dblclicked)
          .filter(touchable)
            .on("touchstart.zoom", touchstarted)
            .on("touchmove.zoom", touchmoved)
            .on("touchend.zoom touchcancel.zoom", touchended)
            .style("touch-action", "none")
            .style("-webkit-tap-highlight-color", "rgba(0,0,0,0)");
      }

      zoom.transform = function(collection, transform) {
        var selection = collection.selection ? collection.selection() : collection;
        selection.property("__zoom", defaultTransform$1);
        if (collection !== selection) {
          schedule(collection, transform);
        } else {
          selection.interrupt().each(function() {
            gesture(this, arguments)
                .start()
                .zoom(null, typeof transform === "function" ? transform.apply(this, arguments) : transform)
                .end();
          });
        }
      };

      zoom.scaleBy = function(selection, k) {
        zoom.scaleTo(selection, function() {
          var k0 = this.__zoom.k,
              k1 = typeof k === "function" ? k.apply(this, arguments) : k;
          return k0 * k1;
        });
      };

      zoom.scaleTo = function(selection, k) {
        zoom.transform(selection, function() {
          var e = extent.apply(this, arguments),
              t0 = this.__zoom,
              p0 = centroid(e),
              p1 = t0.invert(p0),
              k1 = typeof k === "function" ? k.apply(this, arguments) : k;
          return constrain(translate(scale(t0, k1), p0, p1), e, translateExtent);
        });
      };

      zoom.translateBy = function(selection, x, y) {
        zoom.transform(selection, function() {
          return constrain(this.__zoom.translate(
            typeof x === "function" ? x.apply(this, arguments) : x,
            typeof y === "function" ? y.apply(this, arguments) : y
          ), extent.apply(this, arguments), translateExtent);
        });
      };

      zoom.translateTo = function(selection, x, y) {
        zoom.transform(selection, function() {
          var e = extent.apply(this, arguments),
              t = this.__zoom,
              p = centroid(e);
          return constrain(identity$m.translate(p[0], p[1]).scale(t.k).translate(
            typeof x === "function" ? -x.apply(this, arguments) : -x,
            typeof y === "function" ? -y.apply(this, arguments) : -y
          ), e, translateExtent);
        });
      };

      function scale(transform, k) {
        k = Math.max(scaleExtent[0], Math.min(scaleExtent[1], k));
        return k === transform.k ? transform : new Transform$1(k, transform.x, transform.y);
      }

      function translate(transform, p0, p1) {
        var x = p0[0] - p1[0] * transform.k, y = p0[1] - p1[1] * transform.k;
        return x === transform.x && y === transform.y ? transform : new Transform$1(transform.k, x, y);
      }

      function centroid(extent) {
        return [(+extent[0][0] + +extent[1][0]) / 2, (+extent[0][1] + +extent[1][1]) / 2];
      }

      function schedule(transition, transform, center) {
        transition
            .on("start.zoom", function() { gesture(this, arguments).start(); })
            .on("interrupt.zoom end.zoom", function() { gesture(this, arguments).end(); })
            .tween("zoom", function() {
              var that = this,
                  args = arguments,
                  g = gesture(that, args),
                  e = extent.apply(that, args),
                  p = center || centroid(e),
                  w = Math.max(e[1][0] - e[0][0], e[1][1] - e[0][1]),
                  a = that.__zoom,
                  b = typeof transform === "function" ? transform.apply(that, args) : transform,
                  i = interpolate(a.invert(p).concat(w / a.k), b.invert(p).concat(w / b.k));
              return function(t) {
                if (t === 1) t = b; // Avoid rounding error on end.
                else { var l = i(t), k = w / l[2]; t = new Transform$1(k, p[0] - l[0] * k, p[1] - l[1] * k); }
                g.zoom(null, t);
              };
            });
      }

      function gesture(that, args) {
        for (var i = 0, n = gestures.length, g; i < n; ++i) {
          if ((g = gestures[i]).that === that) {
            return g;
          }
        }
        return new Gesture(that, args);
      }

      function Gesture(that, args) {
        this.that = that;
        this.args = args;
        this.index = -1;
        this.active = 0;
        this.extent = extent.apply(that, args);
      }

      Gesture.prototype = {
        start: function() {
          if (++this.active === 1) {
            this.index = gestures.push(this) - 1;
            this.emit("start");
          }
          return this;
        },
        zoom: function(key, transform) {
          if (this.mouse && key !== "mouse") this.mouse[1] = transform.invert(this.mouse[0]);
          if (this.touch0 && key !== "touch") this.touch0[1] = transform.invert(this.touch0[0]);
          if (this.touch1 && key !== "touch") this.touch1[1] = transform.invert(this.touch1[0]);
          this.that.__zoom = transform;
          this.emit("zoom");
          return this;
        },
        end: function() {
          if (--this.active === 0) {
            gestures.splice(this.index, 1);
            this.index = -1;
            this.emit("end");
          }
          return this;
        },
        emit: function(type) {
          customEvent$4(new ZoomEvent$1(zoom, type, this.that.__zoom), listeners.apply, listeners, [type, this.that, this.args]);
        }
      };

      function wheeled() {
        if (!filter.apply(this, arguments)) return;
        var g = gesture(this, arguments),
            t = this.__zoom,
            k = Math.max(scaleExtent[0], Math.min(scaleExtent[1], t.k * Math.pow(2, wheelDelta.apply(this, arguments)))),
            p = mouse$4(this);

        // If the mouse is in the same location as before, reuse it.
        // If there were recent wheel events, reset the wheel idle timeout.
        if (g.wheel) {
          if (g.mouse[0][0] !== p[0] || g.mouse[0][1] !== p[1]) {
            g.mouse[1] = t.invert(g.mouse[0] = p);
          }
          clearTimeout(g.wheel);
        }

        // If this wheel event won’t trigger a transform change, ignore it.
        else if (t.k === k) return;

        // Otherwise, capture the mouse point and location at the start.
        else {
          g.mouse = [p, t.invert(p)];
          interrupt$3(this);
          g.start();
        }

        noevent$7();
        g.wheel = setTimeout(wheelidled, wheelDelay);
        g.zoom("mouse", constrain(translate(scale(t, k), g.mouse[0], g.mouse[1]), g.extent, translateExtent));

        function wheelidled() {
          g.wheel = null;
          g.end();
        }
      }

      function mousedowned() {
        if (touchending || !filter.apply(this, arguments)) return;
        var g = gesture(this, arguments),
            v = select$4(event$4.view).on("mousemove.zoom", mousemoved, true).on("mouseup.zoom", mouseupped, true),
            p = mouse$4(this),
            x0 = event$4.clientX,
            y0 = event$4.clientY;

        dragDisable$2(event$4.view);
        nopropagation$5();
        g.mouse = [p, this.__zoom.invert(p)];
        interrupt$3(this);
        g.start();

        function mousemoved() {
          noevent$7();
          if (!g.moved) {
            var dx = event$4.clientX - x0, dy = event$4.clientY - y0;
            g.moved = dx * dx + dy * dy > clickDistance2;
          }
          g.zoom("mouse", constrain(translate(g.that.__zoom, g.mouse[0] = mouse$4(g.that), g.mouse[1]), g.extent, translateExtent));
        }

        function mouseupped() {
          v.on("mousemove.zoom mouseup.zoom", null);
          yesdrag$3(event$4.view, g.moved);
          noevent$7();
          g.end();
        }
      }

      function dblclicked() {
        if (!filter.apply(this, arguments)) return;
        var t0 = this.__zoom,
            p0 = mouse$4(this),
            p1 = t0.invert(p0),
            k1 = t0.k * (event$4.shiftKey ? 0.5 : 2),
            t1 = constrain(translate(scale(t0, k1), p0, p1), extent.apply(this, arguments), translateExtent);

        noevent$7();
        if (duration > 0) select$4(this).transition().duration(duration).call(schedule, t1, p0);
        else select$4(this).call(zoom.transform, t1);
      }

      function touchstarted() {
        if (!filter.apply(this, arguments)) return;
        var g = gesture(this, arguments),
            touches = event$4.changedTouches,
            started,
            n = touches.length, i, t, p;

        nopropagation$5();
        for (i = 0; i < n; ++i) {
          t = touches[i], p = touch$3(this, touches, t.identifier);
          p = [p, this.__zoom.invert(p), t.identifier];
          if (!g.touch0) g.touch0 = p, started = true;
          else if (!g.touch1) g.touch1 = p;
        }

        // If this is a dbltap, reroute to the (optional) dblclick.zoom handler.
        if (touchstarting) {
          touchstarting = clearTimeout(touchstarting);
          if (!g.touch1) {
            g.end();
            p = select$4(this).on("dblclick.zoom");
            if (p) p.apply(this, arguments);
            return;
          }
        }

        if (started) {
          touchstarting = setTimeout(function() { touchstarting = null; }, touchDelay);
          interrupt$3(this);
          g.start();
        }
      }

      function touchmoved() {
        var g = gesture(this, arguments),
            touches = event$4.changedTouches,
            n = touches.length, i, t, p, l;

        noevent$7();
        if (touchstarting) touchstarting = clearTimeout(touchstarting);
        for (i = 0; i < n; ++i) {
          t = touches[i], p = touch$3(this, touches, t.identifier);
          if (g.touch0 && g.touch0[2] === t.identifier) g.touch0[0] = p;
          else if (g.touch1 && g.touch1[2] === t.identifier) g.touch1[0] = p;
        }
        t = g.that.__zoom;
        if (g.touch1) {
          var p0 = g.touch0[0], l0 = g.touch0[1],
              p1 = g.touch1[0], l1 = g.touch1[1],
              dp = (dp = p1[0] - p0[0]) * dp + (dp = p1[1] - p0[1]) * dp,
              dl = (dl = l1[0] - l0[0]) * dl + (dl = l1[1] - l0[1]) * dl;
          t = scale(t, Math.sqrt(dp / dl));
          p = [(p0[0] + p1[0]) / 2, (p0[1] + p1[1]) / 2];
          l = [(l0[0] + l1[0]) / 2, (l0[1] + l1[1]) / 2];
        }
        else if (g.touch0) p = g.touch0[0], l = g.touch0[1];
        else return;
        g.zoom("touch", constrain(translate(t, p, l), g.extent, translateExtent));
      }

      function touchended() {
        var g = gesture(this, arguments),
            touches = event$4.changedTouches,
            n = touches.length, i, t;

        nopropagation$5();
        if (touchending) clearTimeout(touchending);
        touchending = setTimeout(function() { touchending = null; }, touchDelay);
        for (i = 0; i < n; ++i) {
          t = touches[i];
          if (g.touch0 && g.touch0[2] === t.identifier) delete g.touch0;
          else if (g.touch1 && g.touch1[2] === t.identifier) delete g.touch1;
        }
        if (g.touch1 && !g.touch0) g.touch0 = g.touch1, delete g.touch1;
        if (g.touch0) g.touch0[1] = this.__zoom.invert(g.touch0[0]);
        else g.end();
      }

      zoom.wheelDelta = function(_) {
        return arguments.length ? (wheelDelta = typeof _ === "function" ? _ : constant$y(+_), zoom) : wheelDelta;
      };

      zoom.filter = function(_) {
        return arguments.length ? (filter = typeof _ === "function" ? _ : constant$y(!!_), zoom) : filter;
      };

      zoom.touchable = function(_) {
        return arguments.length ? (touchable = typeof _ === "function" ? _ : constant$y(!!_), zoom) : touchable;
      };

      zoom.extent = function(_) {
        return arguments.length ? (extent = typeof _ === "function" ? _ : constant$y([[+_[0][0], +_[0][1]], [+_[1][0], +_[1][1]]]), zoom) : extent;
      };

      zoom.scaleExtent = function(_) {
        return arguments.length ? (scaleExtent[0] = +_[0], scaleExtent[1] = +_[1], zoom) : [scaleExtent[0], scaleExtent[1]];
      };

      zoom.translateExtent = function(_) {
        return arguments.length ? (translateExtent[0][0] = +_[0][0], translateExtent[1][0] = +_[1][0], translateExtent[0][1] = +_[0][1], translateExtent[1][1] = +_[1][1], zoom) : [[translateExtent[0][0], translateExtent[0][1]], [translateExtent[1][0], translateExtent[1][1]]];
      };

      zoom.constrain = function(_) {
        return arguments.length ? (constrain = _, zoom) : constrain;
      };

      zoom.duration = function(_) {
        return arguments.length ? (duration = +_, zoom) : duration;
      };

      zoom.interpolate = function(_) {
        return arguments.length ? (interpolate = _, zoom) : interpolate;
      };

      zoom.on = function() {
        var value = listeners.on.apply(listeners, arguments);
        return value === listeners ? zoom : value;
      };

      zoom.clickDistance = function(_) {
        return arguments.length ? (clickDistance2 = (_ = +_) * _, zoom) : Math.sqrt(clickDistance2);
      };

      return zoom;
    }

    var d3 = /*#__PURE__*/Object.freeze({
        __proto__: null,
        version: version$2,
        bisect: bisectRight$1,
        bisectRight: bisectRight$1,
        bisectLeft: bisectLeft$1,
        ascending: ascending$4,
        bisector: bisector$1,
        cross: cross$3,
        descending: descending$3,
        deviation: deviation$1,
        extent: extent$2,
        histogram: histogram$1,
        thresholdFreedmanDiaconis: freedmanDiaconis$1,
        thresholdScott: scott$1,
        thresholdSturges: sturges,
        max: max$3,
        mean: mean$1,
        median: median$1,
        merge: merge$1,
        min: min$2,
        pairs: pairs$1,
        permute: permute$1,
        quantile: quantile$1,
        range: range$1,
        scan: scan$1,
        shuffle: shuffle$2,
        sum: sum$3,
        ticks: ticks$1,
        tickIncrement: tickIncrement$1,
        tickStep: tickStep$1,
        transpose: transpose$1,
        variance: variance$1,
        zip: zip$1,
        axisTop: axisTop$1,
        axisRight: axisRight$1,
        axisBottom: axisBottom$1,
        axisLeft: axisLeft$1,
        brush: brush$2,
        brushX: brushX$1,
        brushY: brushY$1,
        brushSelection: brushSelection$1,
        chord: chord$1,
        ribbon: ribbon$1,
        nest: nest$1,
        set: set$6,
        map: map$5,
        keys: keys$1,
        values: values$1,
        entries: entries$1,
        color: color$2,
        rgb: rgb$2,
        hsl: hsl$4,
        lab: lab$2,
        hcl: hcl$3,
        cubehelix: cubehelix$4,
        dispatch: dispatch$2,
        drag: drag$1,
        dragDisable: nodrag,
        dragEnable: yesdrag$2,
        dsvFormat: dsv$1,
        csvParse: csvParse$1,
        csvParseRows: csvParseRows$1,
        csvFormat: csvFormat$1,
        csvFormatRows: csvFormatRows$1,
        tsvParse: tsvParse$1,
        tsvParseRows: tsvParseRows$1,
        tsvFormat: tsvFormat$1,
        tsvFormatRows: tsvFormatRows$1,
        easeLinear: linear$4,
        easeQuad: quadInOut$1,
        easeQuadIn: quadIn$1,
        easeQuadOut: quadOut$1,
        easeQuadInOut: quadInOut$1,
        easeCubic: cubicInOut$1,
        easeCubicIn: cubicIn$1,
        easeCubicOut: cubicOut$1,
        easeCubicInOut: cubicInOut$1,
        easePoly: polyInOut$1,
        easePolyIn: polyIn$1,
        easePolyOut: polyOut$1,
        easePolyInOut: polyInOut$1,
        easeSin: sinInOut$1,
        easeSinIn: sinIn$1,
        easeSinOut: sinOut$1,
        easeSinInOut: sinInOut$1,
        easeExp: expInOut$1,
        easeExpIn: expIn$1,
        easeExpOut: expOut$1,
        easeExpInOut: expInOut$1,
        easeCircle: circleInOut$1,
        easeCircleIn: circleIn$1,
        easeCircleOut: circleOut$1,
        easeCircleInOut: circleInOut$1,
        easeBounce: bounceOut$1,
        easeBounceIn: bounceIn$1,
        easeBounceOut: bounceOut$1,
        easeBounceInOut: bounceInOut$1,
        easeBack: backInOut$1,
        easeBackIn: backIn$1,
        easeBackOut: backOut$1,
        easeBackInOut: backInOut$1,
        easeElastic: elasticOut$1,
        easeElasticIn: elasticIn$1,
        easeElasticOut: elasticOut$1,
        easeElasticInOut: elasticInOut$1,
        forceCenter: center$3,
        forceCollide: collide$1,
        forceLink: link$3,
        forceManyBody: manyBody$1,
        forceRadial: radial$1,
        forceSimulation: simulation$1,
        forceX: x$7,
        forceY: y$7,
        formatDefaultLocale: defaultLocale$2,
        get format () { return format$1; },
        get formatPrefix () { return formatPrefix$1; },
        formatLocale: formatLocale$2,
        formatSpecifier: formatSpecifier$1,
        precisionFixed: precisionFixed$1,
        precisionPrefix: precisionPrefix$1,
        precisionRound: precisionRound$1,
        geoArea: area$4,
        geoBounds: bounds$1,
        geoCentroid: centroid$2,
        geoCircle: circle$3,
        geoClipAntimeridian: clipAntimeridian$1,
        geoClipCircle: clipCircle$1,
        geoClipExtent: extent$3,
        geoClipRectangle: clipRectangle$1,
        geoContains: contains$3,
        geoDistance: distance$1,
        geoGraticule: graticule$1,
        geoGraticule10: graticule10$1,
        geoInterpolate: interpolate$4,
        geoLength: length$4,
        geoPath: index$5,
        geoAlbers: albers$1,
        geoAlbersUsa: albersUsa$1,
        geoAzimuthalEqualArea: azimuthalEqualArea$1,
        geoAzimuthalEqualAreaRaw: azimuthalEqualAreaRaw$1,
        geoAzimuthalEquidistant: azimuthalEquidistant$1,
        geoAzimuthalEquidistantRaw: azimuthalEquidistantRaw$1,
        geoConicConformal: conicConformal$1,
        geoConicConformalRaw: conicConformalRaw$1,
        geoConicEqualArea: conicEqualArea$1,
        geoConicEqualAreaRaw: conicEqualAreaRaw$1,
        geoConicEquidistant: conicEquidistant$1,
        geoConicEquidistantRaw: conicEquidistantRaw$1,
        geoEquirectangular: equirectangular$1,
        geoEquirectangularRaw: equirectangularRaw$1,
        geoGnomonic: gnomonic$1,
        geoGnomonicRaw: gnomonicRaw$1,
        geoIdentity: identity$f,
        geoProjection: projection$1,
        geoProjectionMutator: projectionMutator$1,
        geoMercator: mercator$1,
        geoMercatorRaw: mercatorRaw$1,
        geoNaturalEarth1: naturalEarth1$1,
        geoNaturalEarth1Raw: naturalEarth1Raw$1,
        geoOrthographic: orthographic$1,
        geoOrthographicRaw: orthographicRaw$1,
        geoStereographic: stereographic$1,
        geoStereographicRaw: stereographicRaw$1,
        geoTransverseMercator: transverseMercator$1,
        geoTransverseMercatorRaw: transverseMercatorRaw$1,
        geoRotation: rotation$1,
        geoStream: geoStream$1,
        geoTransform: transform$2,
        cluster: cluster$1,
        hierarchy: hierarchy$1,
        pack: index$6,
        packSiblings: siblings$1,
        packEnclose: enclose$1,
        partition: partition$1,
        stratify: stratify$1,
        tree: tree$1,
        treemap: index$7,
        treemapBinary: binary$1,
        treemapDice: treemapDice$1,
        treemapSlice: treemapSlice$1,
        treemapSliceDice: sliceDice$1,
        treemapSquarify: squarify$1,
        treemapResquarify: resquarify$1,
        interpolate: value,
        interpolateArray: array$5,
        interpolateBasis: basis$4,
        interpolateBasisClosed: basisClosed$2,
        interpolateDate: date$3,
        interpolateNumber: number$6,
        interpolateObject: object$4,
        interpolateRound: round,
        interpolateString: string,
        interpolateTransformCss: interpolateTransformCss$2,
        interpolateTransformSvg: interpolateTransformSvg$2,
        interpolateZoom: zoom$1,
        interpolateRgb: rgb$4,
        interpolateRgbBasis: rgbBasis$1,
        interpolateRgbBasisClosed: rgbBasisClosed$1,
        interpolateHsl: hsl$7,
        interpolateHslLong: hslLong$1,
        interpolateLab: lab$4,
        interpolateHcl: hcl$6,
        interpolateHclLong: hclLong$1,
        interpolateCubehelix: cubehelix$7,
        interpolateCubehelixLong: cubehelixLong$1,
        quantize: quantize$2,
        path: path$2,
        polygonArea: area$5,
        polygonCentroid: centroid$3,
        polygonHull: hull$1,
        polygonContains: contains$4,
        polygonLength: length$5,
        quadtree: quadtree$2,
        queue: queue,
        randomUniform: uniform$1,
        randomNormal: normal$1,
        randomLogNormal: logNormal$1,
        randomBates: bates$1,
        randomIrwinHall: irwinHall$1,
        randomExponential: exponential$4,
        request: request,
        html: html$2,
        json: json$1,
        text: text$1,
        xml: xml$1,
        csv: csv$3,
        tsv: tsv$3,
        scaleBand: band$1,
        scalePoint: point$8,
        scaleIdentity: identity$i,
        scaleLinear: linear$7,
        scaleLog: log$3,
        scaleOrdinal: ordinal$1,
        scaleImplicit: implicit$1,
        scalePow: pow$3,
        scaleSqrt: sqrt$4,
        scaleQuantile: quantile$2,
        scaleQuantize: quantize$3,
        scaleThreshold: threshold$3,
        scaleTime: time$1,
        scaleUtc: utcTime$1,
        schemeCategory10: category10$1,
        schemeCategory20b: category20b,
        schemeCategory20c: category20c,
        schemeCategory20: category20,
        interpolateCubehelixDefault: cubehelix$a,
        interpolateRainbow: rainbow$2,
        interpolateWarm: warm$1,
        interpolateCool: cool$1,
        interpolateViridis: viridis$1,
        interpolateMagma: magma$1,
        interpolateInferno: inferno$1,
        interpolatePlasma: plasma$1,
        scaleSequential: sequential$1,
        create: create$3,
        creator: creator$3,
        local: local$3,
        matcher: matcher$4,
        mouse: mouse$3,
        namespace: namespace$3,
        namespaces: namespaces$3,
        clientPoint: point$9,
        select: select$3,
        selectAll: selectAll$1,
        selection: selection$3,
        selector: selector$3,
        selectorAll: selectorAll$3,
        style: styleValue$3,
        touch: touch$2,
        touches: touches$1,
        window: defaultView$3,
        get event () { return event$3; },
        customEvent: customEvent$3,
        arc: arc$1,
        area: area$6,
        line: line$1,
        pie: pie$1,
        areaRadial: areaRadial$1,
        radialArea: areaRadial$1,
        lineRadial: lineRadial$3,
        radialLine: lineRadial$3,
        pointRadial: pointRadial$1,
        linkHorizontal: linkHorizontal$1,
        linkVertical: linkVertical$1,
        linkRadial: linkRadial$1,
        symbol: symbol$1,
        symbols: symbols$1,
        symbolCircle: circle$5,
        symbolCross: cross$5,
        symbolDiamond: diamond$1,
        symbolSquare: square$1,
        symbolStar: star$1,
        symbolTriangle: triangle$1,
        symbolWye: wye$1,
        curveBasisClosed: basisClosed$3,
        curveBasisOpen: basisOpen$1,
        curveBasis: basis$5,
        curveBundle: bundle$1,
        curveCardinalClosed: cardinalClosed$1,
        curveCardinalOpen: cardinalOpen$1,
        curveCardinal: cardinal$1,
        curveCatmullRomClosed: catmullRomClosed$1,
        curveCatmullRomOpen: catmullRomOpen$1,
        curveCatmullRom: catmullRom$1,
        curveLinearClosed: linearClosed$1,
        curveLinear: curveLinear$1,
        curveMonotoneX: monotoneX$1,
        curveMonotoneY: monotoneY$1,
        curveNatural: natural$1,
        curveStep: step$1,
        curveStepAfter: stepAfter$1,
        curveStepBefore: stepBefore$1,
        stack: stack$1,
        stackOffsetExpand: expand$1,
        stackOffsetDiverging: diverging$2,
        stackOffsetNone: none$6,
        stackOffsetSilhouette: silhouette$1,
        stackOffsetWiggle: wiggle$1,
        stackOrderAscending: ascending$b,
        stackOrderDescending: descending$5,
        stackOrderInsideOut: insideOut$1,
        stackOrderNone: none$7,
        stackOrderReverse: reverse$1,
        timeInterval: newInterval$2,
        timeMillisecond: millisecond$2,
        timeMilliseconds: milliseconds$1,
        utcMillisecond: millisecond$2,
        utcMilliseconds: milliseconds$1,
        timeSecond: second$2,
        timeSeconds: seconds$1,
        utcSecond: second$2,
        utcSeconds: seconds$1,
        timeMinute: minute$2,
        timeMinutes: minutes$1,
        timeHour: hour$2,
        timeHours: hours$1,
        timeDay: day$2,
        timeDays: days$1,
        timeWeek: sunday$2,
        timeWeeks: sundays$1,
        timeSunday: sunday$2,
        timeSundays: sundays$1,
        timeMonday: monday$2,
        timeMondays: mondays$1,
        timeTuesday: tuesday$2,
        timeTuesdays: tuesdays$1,
        timeWednesday: wednesday$2,
        timeWednesdays: wednesdays$1,
        timeThursday: thursday$2,
        timeThursdays: thursdays$1,
        timeFriday: friday$2,
        timeFridays: fridays$1,
        timeSaturday: saturday$2,
        timeSaturdays: saturdays$1,
        timeMonth: month$2,
        timeMonths: months$1,
        timeYear: year$2,
        timeYears: years$1,
        utcMinute: utcMinute$2,
        utcMinutes: utcMinutes$1,
        utcHour: utcHour$2,
        utcHours: utcHours$1,
        utcDay: utcDay$2,
        utcDays: utcDays$1,
        utcWeek: utcSunday$2,
        utcWeeks: utcSundays$1,
        utcSunday: utcSunday$2,
        utcSundays: utcSundays$1,
        utcMonday: utcMonday$2,
        utcMondays: utcMondays$1,
        utcTuesday: utcTuesday$2,
        utcTuesdays: utcTuesdays$1,
        utcWednesday: utcWednesday$2,
        utcWednesdays: utcWednesdays$1,
        utcThursday: utcThursday$2,
        utcThursdays: utcThursdays$1,
        utcFriday: utcFriday$2,
        utcFridays: utcFridays$1,
        utcSaturday: utcSaturday$2,
        utcSaturdays: utcSaturdays$1,
        utcMonth: utcMonth$2,
        utcMonths: utcMonths$1,
        utcYear: utcYear$2,
        utcYears: utcYears$1,
        timeFormatDefaultLocale: defaultLocale$5,
        get timeFormat () { return timeFormat$2; },
        get timeParse () { return timeParse$2; },
        get utcFormat () { return utcFormat$2; },
        get utcParse () { return utcParse$2; },
        timeFormatLocale: formatLocale$5,
        isoFormat: formatIso$1,
        isoParse: parseIso$1,
        now: now$3,
        timer: timer$3,
        timerFlush: timerFlush$3,
        timeout: timeout$6,
        interval: interval$5,
        transition: transition$2,
        active: active$1,
        interrupt: interrupt$2,
        voronoi: voronoi$1,
        zoom: zoom$2,
        zoomTransform: transform$3,
        zoomIdentity: identity$m
    });

    var addD3SelectionMulti = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     *
     * @fileoverview manually add d3-selection-multi to d3 default bundle. Most of this code is
     * copied from d3-selection-multi@1.0.0.
     * See https://github.com/d3/d3-selection-multi/issues/11 for why we have to do this
     */
    Object.defineProperty(exports, "__esModule", { value: true });

    var d3Selection = d3;
    var d3Transition = d3;
    function attrsFunction(selection, map) {
        return selection.each(function () {
            var x = map.apply(this, arguments), s = d3Selection.select(this);
            for (var name_1 in x)
                s.attr(name_1, x[name_1]);
        });
    }
    function attrsObject(selection, map) {
        for (var name_2 in map)
            selection.attr(name_2, map[name_2]);
        return selection;
    }
    function selection_attrs(map) {
        return (typeof map === "function" ? attrsFunction : attrsObject)(this, map);
    }
    function stylesFunction(selection, map, priority) {
        return selection.each(function () {
            var x = map.apply(this, arguments), s = d3Selection.select(this);
            for (var name_3 in x)
                s.style(name_3, x[name_3], priority);
        });
    }
    function stylesObject(selection, map, priority) {
        for (var name_4 in map)
            selection.style(name_4, map[name_4], priority);
        return selection;
    }
    function selection_styles(map, priority) {
        return (typeof map === "function" ? stylesFunction : stylesObject)(this, map, priority == null ? "" : priority);
    }
    function propertiesFunction(selection, map) {
        return selection.each(function () {
            var x = map.apply(this, arguments), s = d3Selection.select(this);
            for (var name_5 in x)
                s.property(name_5, x[name_5]);
        });
    }
    function propertiesObject(selection, map) {
        for (var name_6 in map)
            selection.property(name_6, map[name_6]);
        return selection;
    }
    function selection_properties(map) {
        return (typeof map === "function" ? propertiesFunction : propertiesObject)(this, map);
    }
    function attrsFunction$1(transition, map) {
        return transition.each(function () {
            var x = map.apply(this, arguments), t = d3Selection.select(this).transition(transition);
            for (var name_7 in x)
                t.attr(name_7, x[name_7]);
        });
    }
    function attrsObject$1(transition, map) {
        for (var name_8 in map)
            transition.attr(name_8, map[name_8]);
        return transition;
    }
    function transition_attrs(map) {
        return (typeof map === "function" ? attrsFunction$1 : attrsObject$1)(this, map);
    }
    function stylesFunction$1(transition, map, priority) {
        return transition.each(function () {
            var x = map.apply(this, arguments), t = d3Selection.select(this).transition(transition);
            for (var name_9 in x)
                t.style(name_9, x[name_9], priority);
        });
    }
    function stylesObject$1(transition, map, priority) {
        for (var name_10 in map)
            transition.style(name_10, map[name_10], priority);
        return transition;
    }
    function transition_styles(map, priority) {
        return (typeof map === "function" ? stylesFunction$1 : stylesObject$1)(this, map, priority == null ? "" : priority);
    }
    d3Selection.selection.prototype.attrs = selection_attrs;
    d3Selection.selection.prototype.styles = selection_styles;
    d3Selection.selection.prototype.properties = selection_properties;
    d3Transition.transition.prototype.attrs = transition_attrs;
    d3Transition.transition.prototype.styles = transition_styles;
    });

    var coerceD3 = createCommonjsModule(function (module, exports) {
    Object.defineProperty(exports, "__esModule", { value: true });

    /**
     * Coerce possibly external d3 instance into our own instance of d3 so we can use d3-selection-multi.
     * See https://github.com/d3/d3-selection-multi/issues/11 for why we have to do this.
     *
     * Any public facing API that accepts a d3 selection should first pass that user-supplied selection
     * through here - this ensures all selection objects that go through the Plottable codebase are "vetted".
     */
    function coerceExternalD3(externalD3Selection) {
        // if .attrs isn't defined; convert the selection
        if (externalD3Selection.attrs == null) {
            if (externalD3Selection.nodes == null) {
                // nodes isn't defined; this is probably a d3v3 selection. handle it accordingly
                var nodes_1 = [];
                externalD3Selection.each(function () {
                    nodes_1.push(this);
                });
                return d3.selectAll(nodes_1);
            }
            else {
                return d3.selectAll(externalD3Selection.nodes());
            }
        }
        else {
            return externalD3Selection;
        }
    }
    exports.coerceExternalD3 = coerceExternalD3;
    });

    var makeEnum_1 = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });
    function makeEnum(values) {
        return values.reduce(function (obj, v) {
            obj[v] = v;
            return obj;
        }, {});
    }
    exports.makeEnum = makeEnum;
    });

    var easingAnimator = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });



    var EASE_NAME_MAPPING = {
        linear: src.easeLinear,
        quad: src.easeQuad,
        quadIn: src.easeQuadIn,
        quadOut: src.easeQuadOut,
        quadInOut: src.easeQuadInOut,
        cubic: src.easeCubic,
        cubicIn: src.easeCubicIn,
        cubicOut: src.easeCubicOut,
        cubicInOut: src.easeCubicInOut,
        poly: src.easePoly,
        polyIn: src.easePolyIn,
        polyOut: src.easePolyOut,
        polyInOut: src.easePolyInOut,
        sin: src.easeSin,
        sinIn: src.easeSinIn,
        sinOut: src.easeSinOut,
        sinInOut: src.easeSinInOut,
        exp: src.easeExp,
        expIn: src.easeExpIn,
        expOut: src.easeExpOut,
        expInOut: src.easeExpInOut,
        circle: src.easeCircle,
        circleIn: src.easeCircleIn,
        circleOut: src.easeCircleOut,
        circleInOut: src.easeCircleInOut,
        bounce: src.easeBounce,
        bounceIn: src.easeBounceIn,
        bounceOut: src.easeBounceOut,
        bounceInOut: src.easeBounceInOut,
        back: src.easeBack,
        backIn: src.easeBackIn,
        backOut: src.easeBackOut,
        backInOut: src.easeBackInOut,
        elastic: src.easeElastic,
        elasticIn: src.easeElasticIn,
        elasticOut: src.easeElasticOut,
        elasticInOut: src.easeElasticInOut,
    };
    exports.EaseName = makeEnum_1.makeEnum([
        "linear",
        "quad",
        "quadIn",
        "quadOut",
        "quadInOut",
        "cubic",
        "cubicIn",
        "cubicOut",
        "cubicInOut",
        "poly",
        "polyIn",
        "polyOut",
        "polyInOut",
        "sin",
        "sinIn",
        "sinOut",
        "sinInOut",
        "exp",
        "expIn",
        "expOut",
        "expInOut",
        "circle",
        "circleIn",
        "circleOut",
        "circleInOut",
        "bounce",
        "bounceIn",
        "bounceOut",
        "bounceInOut",
        "back",
        "backIn",
        "backOut",
        "backInOut",
        "elastic",
        "elasticIn",
        "elasticOut",
        "elasticInOut",
    ]);
    /**
     * An Animator with easing and configurable durations and delays.
     */
    var Easing = /** @class */ (function () {
        /**
         * Constructs the default animator
         *
         * @constructor
         */
        function Easing() {
            this._startDelay = Easing._DEFAULT_START_DELAY_MILLISECONDS;
            this._stepDuration = Easing._DEFAULT_STEP_DURATION_MILLISECONDS;
            this._stepDelay = Easing._DEFAULT_ITERATIVE_DELAY_MILLISECONDS;
            this._maxTotalDuration = Easing._DEFAULT_MAX_TOTAL_DURATION_MILLISECONDS;
            this._easingMode = Easing._DEFAULT_EASING_MODE;
        }
        Easing.prototype.totalTime = function (numberOfSteps) {
            var adjustedIterativeDelay = this._getAdjustedIterativeDelay(numberOfSteps);
            return this.startDelay() + adjustedIterativeDelay * (Math.max(numberOfSteps - 1, 0)) + this.stepDuration();
        };
        Easing.prototype.animate = function (selection, attrToAppliedProjector) {
            var _this = this;
            selection = coerceD3.coerceExternalD3(selection);
            var numberOfSteps = selection.size();
            var adjustedIterativeDelay = this._getAdjustedIterativeDelay(numberOfSteps);
            return selection.transition()
                .ease(this._getEaseFactory())
                .duration(this.stepDuration())
                .delay(function (d, i) { return _this.startDelay() + adjustedIterativeDelay * i; })
                .attrs(attrToAppliedProjector);
        };
        Easing.prototype.startDelay = function (startDelay) {
            if (startDelay == null) {
                return this._startDelay;
            }
            else {
                this._startDelay = startDelay;
                return this;
            }
        };
        Easing.prototype.stepDuration = function (stepDuration) {
            if (stepDuration == null) {
                return Math.min(this._stepDuration, this._maxTotalDuration);
            }
            else {
                this._stepDuration = stepDuration;
                return this;
            }
        };
        Easing.prototype.stepDelay = function (stepDelay) {
            if (stepDelay == null) {
                return this._stepDelay;
            }
            else {
                this._stepDelay = stepDelay;
                return this;
            }
        };
        Easing.prototype.maxTotalDuration = function (maxTotalDuration) {
            if (maxTotalDuration == null) {
                return this._maxTotalDuration;
            }
            else {
                this._maxTotalDuration = maxTotalDuration;
                return this;
            }
        };
        Easing.prototype.easingMode = function (easingMode) {
            if (easingMode == null) {
                return this._easingMode;
            }
            else {
                this._easingMode = easingMode;
                return this;
            }
        };
        Easing.prototype._getEaseFactory = function () {
            var ease = this.easingMode();
            if (typeof ease === "string") {
                var maybeEaseFunction = EASE_NAME_MAPPING[ease];
                if (maybeEaseFunction == null) {
                    // oops; name is wrong - default to linear instead
                    return EASE_NAME_MAPPING["linear"];
                }
                else {
                    return maybeEaseFunction;
                }
            }
            else {
                return ease;
            }
        };
        /**
         * Adjust the iterative delay, such that it takes into account the maxTotalDuration constraint
         */
        Easing.prototype._getAdjustedIterativeDelay = function (numberOfSteps) {
            var stepStartTimeInterval = this.maxTotalDuration() - this.stepDuration();
            stepStartTimeInterval = Math.max(stepStartTimeInterval, 0);
            var maxPossibleIterativeDelay = stepStartTimeInterval / Math.max(numberOfSteps - 1, 1);
            return Math.min(this.stepDelay(), maxPossibleIterativeDelay);
        };
        /**
         * The default starting delay of the animation in milliseconds
         */
        Easing._DEFAULT_START_DELAY_MILLISECONDS = 0;
        /**
         * The default duration of one animation step in milliseconds
         */
        Easing._DEFAULT_STEP_DURATION_MILLISECONDS = 300;
        /**
         * The default maximum start delay between each step of an animation
         */
        Easing._DEFAULT_ITERATIVE_DELAY_MILLISECONDS = 15;
        /**
         * The default maximum total animation duration
         */
        Easing._DEFAULT_MAX_TOTAL_DURATION_MILLISECONDS = Infinity;
        /**
         * The default easing of the animation
         */
        Easing._DEFAULT_EASING_MODE = "expOut";
        return Easing;
    }());
    exports.Easing = Easing;
    });

    var nullAnimator = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });

    /**
     * An animator implementation with no animation. The attributes are
     * immediately set on the selection.
     */
    var Null = /** @class */ (function () {
        function Null() {
        }
        Null.prototype.totalTime = function (selection) {
            return 0;
        };
        Null.prototype.animate = function (selection, attrToAppliedProjector) {
            selection = coerceD3.coerceExternalD3(selection);
            return selection.attrs(attrToAppliedProjector);
        };
        return Null;
    }());
    exports.Null = Null;
    });

    var animators = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });

    tslib_es6.__exportStar(easingAnimator, exports);
    tslib_es6.__exportStar(nullAnimator, exports);
    });

    var cache = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2017-present Palantir Technologies, Inc. All rights reserved.
     * Licensed under the MIT License (the "License"); you may obtain a copy of the
     * license at https://github.com/palantir/typesettable/blob/develop/LICENSE
     */
    Object.defineProperty(exports, "__esModule", { value: true });
    var Cache = (function () {
        /**
         * @constructor
         *
         * @param {string} compute The function whose results will be cached.
         */
        function Cache(compute) {
            this.cache = {};
            this.compute = compute;
        }
        /**
         * Attempt to look up k in the cache, computing the result if it isn't
         * found.
         *
         * @param {string} k The key to look up in the cache.
         * @return {T} The value associated with k; the result of compute(k).
         */
        Cache.prototype.get = function (k) {
            if (!this.cache.hasOwnProperty(k)) {
                this.cache[k] = this.compute(k);
            }
            return this.cache[k];
        };
        /**
         * Reset the cache empty.
         *
         * @return {Cache<T>} The calling Cache.
         */
        Cache.prototype.clear = function () {
            this.cache = {};
            return this;
        };
        return Cache;
    }());
    exports.Cache = Cache;

    });

    var methods = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2017-present Palantir Technologies, Inc. All rights reserved.
     * Licensed under the MIT License (the "License"); you may obtain a copy of the
     * license at https://github.com/palantir/typesettable/blob/develop/LICENSE
     */
    Object.defineProperty(exports, "__esModule", { value: true });
    var Methods = (function () {
        function Methods() {
        }
        /**
         * Check if two arrays are equal by strict equality.
         */
        Methods.arrayEq = function (a, b) {
            // Technically, null and undefined are arrays too
            if (a == null || b == null) {
                return a === b;
            }
            if (a.length !== b.length) {
                return false;
            }
            for (var i = 0; i < a.length; i++) {
                if (a[i] !== b[i]) {
                    return false;
                }
            }
            return true;
        };
        /**
         * @param {any} a Object to check against b for equality.
         * @param {any} b Object to check against a for equality.
         *
         * @returns {boolean} whether or not two objects share the same keys, and
         *          values associated with those keys. Values will be compared
         *          with ===.
         */
        Methods.objEq = function (a, b) {
            if (a == null || b == null) {
                return a === b;
            }
            var keysA = Object.keys(a).sort();
            var keysB = Object.keys(b).sort();
            var valuesA = keysA.map(function (k) { return a[k]; });
            var valuesB = keysB.map(function (k) { return b[k]; });
            return Methods.arrayEq(keysA, keysB) && Methods.arrayEq(valuesA, valuesB);
        };
        Methods.strictEq = function (a, b) {
            return a === b;
        };
        /**
         * Shim for _.defaults
         */
        Methods.defaults = function (target) {
            var objects = [];
            for (var _i = 1; _i < arguments.length; _i++) {
                objects[_i - 1] = arguments[_i];
            }
            if (target == null) {
                throw new TypeError("Cannot convert undefined or null to object");
            }
            var result = Object(target);
            objects.forEach(function (obj) {
                if (obj != null) {
                    for (var key in obj) {
                        if (Object.prototype.hasOwnProperty.call(obj, key)) {
                            result[key] = obj[key];
                        }
                    }
                }
            });
            return result;
        };
        return Methods;
    }());
    exports.Methods = Methods;

    });

    var stringMethods = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2017-present Palantir Technologies, Inc. All rights reserved.
     * Licensed under the MIT License (the "License"); you may obtain a copy of the
     * license at https://github.com/palantir/typesettable/blob/develop/LICENSE
     */
    Object.defineProperty(exports, "__esModule", { value: true });
    var StringMethods = (function () {
        function StringMethods() {
        }
        /**
         * Treat all sequences of consecutive spaces as a single " ".
         */
        StringMethods.combineWhitespace = function (str) {
            return str.replace(/[ \t]+/g, " ");
        };
        StringMethods.isNotEmptyString = function (str) {
            return str && str.trim() !== "";
        };
        StringMethods.trimStart = function (str, splitter) {
            if (!str) {
                return str;
            }
            var chars = str.split("");
            var reduceFunction = splitter ? function (s) { return s.split(splitter).some(StringMethods.isNotEmptyString); }
                : StringMethods.isNotEmptyString;
            return chars.reduce(function (s, c) { return reduceFunction(s + c) ? s + c : s; }, "");
        };
        StringMethods.trimEnd = function (str, c) {
            if (!str) {
                return str;
            }
            var reversedChars = str.split("");
            reversedChars.reverse();
            reversedChars = StringMethods.trimStart(reversedChars.join(""), c).split("");
            reversedChars.reverse();
            return reversedChars.join("");
        };
        return StringMethods;
    }());
    exports.StringMethods = StringMethods;

    });

    var tokenizer = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2017-present Palantir Technologies, Inc. All rights reserved.
     * Licensed under the MIT License (the "License"); you may obtain a copy of the
     * license at https://github.com/palantir/typesettable/blob/develop/LICENSE
     */
    Object.defineProperty(exports, "__esModule", { value: true });
    var Tokenizer = (function () {
        function Tokenizer() {
            this.WordDividerRegExp = new RegExp("\\W");
            this.WhitespaceRegExp = new RegExp("\\s");
        }
        Tokenizer.prototype.tokenize = function (line) {
            var _this = this;
            return line.split("").reduce(function (tokens, c) {
                return tokens.slice(0, -1).concat(_this.shouldCreateNewToken(tokens[tokens.length - 1], c));
            }, [""]);
        };
        Tokenizer.prototype.shouldCreateNewToken = function (token, newCharacter) {
            if (!token) {
                return [newCharacter];
            }
            var lastCharacter = token[token.length - 1];
            if (this.WhitespaceRegExp.test(lastCharacter) && this.WhitespaceRegExp.test(newCharacter)) {
                return [token + newCharacter];
            }
            else if (this.WhitespaceRegExp.test(lastCharacter) || this.WhitespaceRegExp.test(newCharacter)) {
                return [token, newCharacter];
            }
            else if (!(this.WordDividerRegExp.test(lastCharacter))) {
                return [token + newCharacter];
            }
            else if (lastCharacter === newCharacter) {
                return [token + newCharacter];
            }
            else {
                return [token, newCharacter];
            }
        };
        return Tokenizer;
    }());
    exports.Tokenizer = Tokenizer;

    });

    var utils = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2017-present Palantir Technologies, Inc. All rights reserved.
     * Licensed under the MIT License (the "License"); you may obtain a copy of the
     * license at https://github.com/palantir/typesettable/blob/develop/LICENSE
     */
    function __export(m) {
        for (var p in m) if (!exports.hasOwnProperty(p)) exports[p] = m[p];
    }
    Object.defineProperty(exports, "__esModule", { value: true });
    __export(cache);
    __export(methods);
    __export(stringMethods);
    __export(tokenizer);

    });

    var writer = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2017-present Palantir Technologies, Inc. All rights reserved.
     * Licensed under the MIT License (the "License"); you may obtain a copy of the
     * license at https://github.com/palantir/typesettable/blob/develop/LICENSE
     */
    Object.defineProperty(exports, "__esModule", { value: true });

    var DEFAULT_WRITE_OPTIONS = {
        textRotation: 0,
        textShear: 0,
        xAlign: "left",
        yAlign: "top",
    };
    var Writer = (function () {
        function Writer(_measurer, _penFactory, _wrapper) {
            this._measurer = _measurer;
            this._penFactory = _penFactory;
            this._wrapper = _wrapper;
        }
        Writer.prototype.measurer = function (newMeasurer) {
            this._measurer = newMeasurer;
            return this;
        };
        Writer.prototype.wrapper = function (newWrapper) {
            this._wrapper = newWrapper;
            return this;
        };
        Writer.prototype.penFactory = function (newPenFactory) {
            this._penFactory = newPenFactory;
            return this;
        };
        /**
         * Writes the text into the container. If no container is specified, the pen's
         * default container will be used.
         */
        Writer.prototype.write = function (text, width, height, options, container) {
            if (options === void 0) { options = {}; }
            // apply default options
            options = utils.Methods.defaults({}, DEFAULT_WRITE_OPTIONS, options);
            // validate input
            if (Writer.SupportedRotation.indexOf(options.textRotation) === -1) {
                throw new Error("unsupported rotation - " + options.textRotation +
                    ". Supported rotations are " + Writer.SupportedRotation.join(", "));
            }
            if (options.textShear != null && options.textShear < -80 || options.textShear > 80) {
                throw new Error("unsupported shear angle - " + options.textShear + ". Must be between -80 and 80");
            }
            var orientHorizontally = Math.abs(Math.abs(options.textRotation) - 90) > 45;
            var primaryDimension = orientHorizontally ? width : height;
            var secondaryDimension = orientHorizontally ? height : width;
            // compute shear parameters
            var shearDegrees = options.textShear;
            var shearRadians = shearDegrees * Math.PI / 180;
            var lineHeight = this._measurer.measure().height;
            var shearShift = lineHeight * Math.tan(shearRadians);
            // When we apply text shear, the primary axis grows and the secondary axis
            // shrinks, due to trigonometry. The text shear feature uses the normal
            // wrapping logic with a subsituted bounding box of the corrected size
            // (computed below). When rendering the wrapped lines, we rotate the text
            // container by the text rotation angle AND the shear angle then carefully
            // offset each one so that they are still aligned to the primary alignment
            // option.
            var shearCorrectedPrimaryDimension = primaryDimension / Math.cos(shearRadians) - Math.abs(shearShift);
            var shearCorrectedSecondaryDimension = secondaryDimension * Math.cos(shearRadians);
            // normalize and wrap text
            var normalizedText = utils.StringMethods.combineWhitespace(text);
            var wrappedText = this._wrapper ?
                this._wrapper.wrap(normalizedText, this._measurer, shearCorrectedPrimaryDimension, shearCorrectedSecondaryDimension).wrappedText : normalizedText;
            var lines = wrappedText.split("\n");
            // correct the intial x/y offset of the text container accounting shear and alignment
            var shearCorrectedXOffset = Writer.XOffsetFactor[options.xAlign] *
                shearCorrectedPrimaryDimension * Math.sin(shearRadians);
            var shearCorrectedYOffset = Writer.YOffsetFactor[options.yAlign] *
                (shearCorrectedSecondaryDimension - lines.length * lineHeight);
            var shearCorrection = shearCorrectedXOffset - shearCorrectedYOffset;
            // compute transform
            var translate = [0, 0];
            var rotate = options.textRotation + shearDegrees;
            switch (options.textRotation) {
                case 90:
                    translate = [width + shearCorrection, 0];
                    break;
                case -90:
                    translate = [-shearCorrection, height];
                    break;
                case 180:
                    translate = [width, height + shearCorrection];
                    break;
                default:
                    translate = [0, -shearCorrection];
                    break;
            }
            // create a new pen and write the lines
            var linePen = this._penFactory.createPen(text, { translate: translate, rotate: rotate }, container);
            this.writeLines(lines, linePen, shearCorrectedPrimaryDimension, lineHeight, shearShift, options.xAlign);
            if (linePen.destroy != null) {
                linePen.destroy();
            }
        };
        Writer.prototype.writeLines = function (lines, linePen, width, lineHeight, shearShift, xAlign) {
            lines.forEach(function (line, i) {
                var xShearOffset = (shearShift > 0) ? (i + 1) * shearShift : (i) * shearShift;
                linePen.write(line, width, xAlign, xShearOffset, (i + 1) * lineHeight);
            });
        };
        return Writer;
    }());
    Writer.XOffsetFactor = {
        center: 0.5,
        left: 0,
        right: 1,
    };
    Writer.YOffsetFactor = {
        bottom: 1,
        center: 0.5,
        top: 0,
    };
    Writer.SupportedRotation = [-90, 0, 180, 90];
    exports.Writer = Writer;

    });

    var writers = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2017-present Palantir Technologies, Inc. All rights reserved.
     * Licensed under the MIT License (the "License"); you may obtain a copy of the
     * license at https://github.com/palantir/typesettable/blob/develop/LICENSE
     */
    function __export(m) {
        for (var p in m) if (!exports.hasOwnProperty(p)) exports[p] = m[p];
    }
    Object.defineProperty(exports, "__esModule", { value: true });
    __export(writer);

    });

    var html$3 = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2017-present Palantir Technologies, Inc. All rights reserved.
     * Licensed under the MIT License (the "License"); you may obtain a copy of the
     * license at https://github.com/palantir/typesettable/blob/develop/LICENSE
     */
    Object.defineProperty(exports, "__esModule", { value: true });
    var HtmlUtils = (function () {
        function HtmlUtils() {
        }
        /**
         * Appends an HTML element with the specified tag name to the provided element.
         * The variadic classnames are added to the new element.
         *
         * Returns the new element.
         */
        HtmlUtils.append = function (element, tagName) {
            var classNames = [];
            for (var _i = 2; _i < arguments.length; _i++) {
                classNames[_i - 2] = arguments[_i];
            }
            var child = HtmlUtils.create.apply(HtmlUtils, [tagName].concat(classNames));
            element.appendChild(child);
            return child;
        };
        /**
         * Creates and returns a new HTMLElement with the attached classnames.
         */
        HtmlUtils.create = function (tagName) {
            var classNames = [];
            for (var _i = 1; _i < arguments.length; _i++) {
                classNames[_i - 1] = arguments[_i];
            }
            var element = document.createElement(tagName);
            HtmlUtils.addClasses.apply(HtmlUtils, [element].concat(classNames));
            return element;
        };
        /**
         * Adds the variadic classnames to the Element
         */
        HtmlUtils.addClasses = function (element) {
            var classNames = [];
            for (var _i = 1; _i < arguments.length; _i++) {
                classNames[_i - 1] = arguments[_i];
            }
            classNames = classNames.filter(function (c) { return c != null; });
            if (element.classList != null) {
                classNames.forEach(function (className) {
                    element.classList.add(className);
                });
            }
            else {
                // IE 11 does not support classList
                element.setAttribute("class", classNames.join(" "));
            }
        };
        /**
         * Returns the width/height of HTMLElement's bounding box
         */
        HtmlUtils.getDimensions = function (element) {
            // using feature detection, safely return the bounding box dimensions of the
            // provided html element
            if (element.getBoundingClientRect) {
                try {
                    var _a = element.getBoundingClientRect(), width = _a.width, height = _a.height;
                    // copy to prevent NoModificationAllowedError
                    return { width: width, height: height };
                }
                catch (err) {
                    // swallow any errors that occur (Firefox Linux)
                }
            }
            // if can't get valid bbox, return 0,0
            return { height: 0, width: 0 };
        };
        return HtmlUtils;
    }());
    exports.HtmlUtils = HtmlUtils;
    /**
     * A typesetter context for HTML.
     */
    var HtmlContext = (function () {
        /**
         * @param element - The CSS font styles applied to `element` will determine the
         * size of text measurements. Also the default text block container.
         * @param className - added to new text blocks
         * @param addTitle - enable title attribute to be added to new text blocks.
         */
        function HtmlContext(element, className, addTitle) {
            if (addTitle === void 0) { addTitle = false; }
            var _this = this;
            this.element = element;
            this.className = className;
            this.addTitle = addTitle;
            this.createRuler = function () {
                return function (text) {
                    var textElement = HtmlUtils.append(_this.element, "span", "text-tmp", _this.className);
                    textElement.textContent = text;
                    var dimensions = HtmlUtils.getDimensions(textElement);
                    _this.element.removeChild(textElement); // element.remove() doesn't work in IE11
                    return dimensions;
                };
            };
            this.createPen = function (text, transform, element) {
                if (element == null) {
                    element = _this.element;
                }
                var textBlock = HtmlUtils.append(element, "div", "text-block", _this.className);
                textBlock.style.position = "relative";
                textBlock.style.transform =
                    "translate(0, -1em) " +
                        ("translate(" + transform.translate[0] + "px, " + transform.translate[1] + "px) ") +
                        ("rotate(" + transform.rotate + "deg)");
                // This awkward transform origin matches the SVG origin
                textBlock.style.transformOrigin = "0 1.2em";
                // attach optional title
                if (_this.addTitle) {
                    textBlock.setAttribute("title", text);
                }
                return _this.createHtmlLinePen(textBlock);
            };
        }
        HtmlContext.prototype.setAddTitle = function (addTitle) {
            this.addTitle = addTitle;
        };
        HtmlContext.prototype.createHtmlLinePen = function (textBlock) {
            return {
                write: function (line, width, xAlign, xOffset, yOffset) {
                    var textLine = HtmlUtils.append(textBlock, "div", "text-line");
                    textLine.textContent = line;
                    textLine.style.width = width + "px";
                    textLine.style.textAlign = xAlign;
                    textLine.style.position = "absolute";
                    textLine.style.whiteSpace = "nowrap";
                    textLine.style.top = yOffset + "px";
                    textLine.style.left = xOffset + "px";
                },
            };
        };
        return HtmlContext;
    }());
    exports.HtmlContext = HtmlContext;

    });

    var svg$1 = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2017-present Palantir Technologies, Inc. All rights reserved.
     * Licensed under the MIT License (the "License"); you may obtain a copy of the
     * license at https://github.com/palantir/typesettable/blob/develop/LICENSE
     */
    Object.defineProperty(exports, "__esModule", { value: true });


    var SvgUtils = (function () {
        function SvgUtils() {
        }
        /**
         * Appends an SVG element with the specified tag name to the provided element.
         * The variadic classnames are added to the new element.
         *
         * Returns the new element.
         */
        SvgUtils.append = function (element, tagName) {
            var classNames = [];
            for (var _i = 2; _i < arguments.length; _i++) {
                classNames[_i - 2] = arguments[_i];
            }
            var child = SvgUtils.create.apply(SvgUtils, [tagName].concat(classNames));
            element.appendChild(child);
            return child;
        };
        /**
         * Creates and returns a new SVGElement with the attached classnames.
         */
        SvgUtils.create = function (tagName) {
            var classNames = [];
            for (var _i = 1; _i < arguments.length; _i++) {
                classNames[_i - 1] = arguments[_i];
            }
            var element = document.createElementNS(SvgUtils.SVG_NS, tagName);
            html$3.HtmlUtils.addClasses.apply(html$3.HtmlUtils, [element].concat(classNames));
            return element;
        };
        /**
         * Returns the width/height of svg element's bounding box
         */
        SvgUtils.getDimensions = function (element) {
            // using feature detection, safely return the bounding box dimensions of the
            // provided svg element
            if (element.getBBox) {
                try {
                    var _a = element.getBBox(), width = _a.width, height = _a.height;
                    // copy to prevent NoModificationAllowedError
                    return { width: width, height: height };
                }
                catch (err) {
                    // swallow any errors that occur (Firefox Linux)
                }
            }
            // if can't get valid bbox, return 0,0
            return { height: 0, width: 0 };
        };
        return SvgUtils;
    }());
    SvgUtils.SVG_NS = "http://www.w3.org/2000/svg";
    exports.SvgUtils = SvgUtils;
    /**
     * A typesetter context for SVG.
     *
     * @param element - The CSS font styles applied to `element` will determine the
     * size of text measurements. Also the default text block container.
     * @param className - added to new text blocks
     * @param addTitleElement - enable title tags to be added to new text blocks.
     */
    var SvgContext = (function () {
        function SvgContext(element, className, addTitleElement) {
            if (addTitleElement === void 0) { addTitleElement = false; }
            var _this = this;
            this.element = element;
            this.className = className;
            this.addTitleElement = addTitleElement;
            this.createRuler = function () {
                var _a = _this.getTextElements(_this.element), parentElement = _a.parentElement, containerElement = _a.containerElement, textElement = _a.textElement;
                return function (text) {
                    parentElement.appendChild(containerElement);
                    textElement.textContent = text;
                    var dimensions = SvgUtils.getDimensions(textElement);
                    parentElement.removeChild(containerElement); // element.remove() doesn't work in IE11
                    return dimensions;
                };
            };
            this.createPen = function (text, transform, element) {
                if (element == null) {
                    element = _this.element;
                }
                var textContainer = SvgUtils.append(element, "g", "text-container", _this.className);
                // attach optional title
                if (_this.addTitleElement) {
                    SvgUtils.append(textContainer, "title").textContent = text;
                    textContainer.setAttribute("title", text);
                }
                // create and transform text block group
                var textBlockGroup = SvgUtils.append(textContainer, "g", "text-area");
                textBlockGroup.setAttribute("transform", "translate(" + transform.translate[0] + "," + transform.translate[1] + ")" +
                    ("rotate(" + transform.rotate + ")"));
                return _this.createSvgLinePen(textBlockGroup);
            };
        }
        SvgContext.prototype.setAddTitleElement = function (addTitleElement) {
            this.addTitleElement = addTitleElement;
        };
        SvgContext.prototype.createSvgLinePen = function (textBlockGroup) {
            return {
                write: function (line, width, xAlign, xOffset, yOffset) {
                    xOffset += width * writers.Writer.XOffsetFactor[xAlign];
                    var element = SvgUtils.append(textBlockGroup, "text", "text-line");
                    element.textContent = line;
                    element.setAttribute("text-anchor", SvgContext.AnchorMap[xAlign]);
                    element.setAttribute("transform", "translate(" + xOffset + "," + yOffset + ")");
                    element.setAttribute("y", "-0.25em");
                },
            };
        };
        SvgContext.prototype.getTextElements = function (element) {
            // if element is already a text element, return it
            if (element.tagName === "text") {
                var parentElement = element.parentElement;
                if (parentElement == null) {
                    parentElement = element.parentNode;
                }
                // must be removed from parent since we re-add it on every measurement
                parentElement.removeChild(element);
                return {
                    containerElement: element,
                    parentElement: parentElement,
                    textElement: element,
                };
            }
            // if element has a text element descendent, select it and return it
            var selected = element.querySelector("text");
            if (selected != null) {
                var parentElement = selected.parentElement;
                if (parentElement == null) {
                    parentElement = selected.parentNode;
                }
                // must be removed from parent since we re-add it on every measurement
                parentElement.removeChild(selected);
                return {
                    containerElement: selected,
                    parentElement: parentElement,
                    textElement: selected,
                };
            }
            // otherwise create a new text element
            var created = SvgUtils.create("text", this.className);
            return {
                containerElement: created,
                parentElement: element,
                textElement: created,
            };
        };
        return SvgContext;
    }());
    SvgContext.AnchorMap = {
        center: "middle",
        left: "start",
        right: "end",
    };
    exports.SvgContext = SvgContext;

    });

    var canvas = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2017-present Palantir Technologies, Inc. All rights reserved.
     * Licensed under the MIT License (the "License"); you may obtain a copy of the
     * license at https://github.com/palantir/typesettable/blob/develop/LICENSE
     */
    Object.defineProperty(exports, "__esModule", { value: true });

    var DEFAULT_FILL_COLOR = "#444";
    /**
     * A typesetter context for HTML5 Canvas.
     *
     * Due to the Canvas API, you must explicitly define the line height, and any
     * styling for the font must also be explicitly defined in the optional
     * `ICanvasFontStyle` object.
     */
    var CanvasContext = (function () {
        function CanvasContext(ctx, lineHeight, style) {
            if (lineHeight === void 0) { lineHeight = 10; }
            if (style === void 0) { style = {}; }
            var _this = this;
            this.ctx = ctx;
            this.lineHeight = lineHeight;
            this.style = style;
            this.createRuler = function () {
                return function (text) {
                    _this.ctx.font = _this.style.font;
                    var width = _this.ctx.measureText(text).width;
                    return { width: width, height: _this.lineHeight };
                };
            };
            this.createPen = function (_text, transform, ctx) {
                if (ctx == null) {
                    ctx = _this.ctx;
                }
                ctx.save();
                ctx.translate(transform.translate[0], transform.translate[1]);
                ctx.rotate(transform.rotate * Math.PI / 180.0);
                return _this.createCanvasPen(ctx);
            };
            if (this.style.fill === undefined) {
                this.style.fill = DEFAULT_FILL_COLOR;
            }
        }
        CanvasContext.prototype.createCanvasPen = function (ctx) {
            var _this = this;
            return {
                destroy: function () {
                    ctx.restore();
                },
                write: function (line, width, xAlign, xOffset, yOffset) {
                    xOffset += width * writers.Writer.XOffsetFactor[xAlign];
                    ctx.textAlign = xAlign;
                    if (_this.style.font != null) {
                        ctx.font = _this.style.font;
                    }
                    if (_this.style.fill != null) {
                        ctx.fillStyle = _this.style.fill;
                        ctx.fillText(line, xOffset, yOffset);
                    }
                    if (_this.style.stroke != null) {
                        ctx.strokeStyle = _this.style.fill;
                        ctx.strokeText(line, xOffset, yOffset);
                    }
                },
            };
        };
        return CanvasContext;
    }());
    exports.CanvasContext = CanvasContext;

    });

    var contexts = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2017-present Palantir Technologies, Inc. All rights reserved.
     * Licensed under the MIT License (the "License"); you may obtain a copy of the
     * license at https://github.com/palantir/typesettable/blob/develop/LICENSE
     */
    function __export(m) {
        for (var p in m) if (!exports.hasOwnProperty(p)) exports[p] = m[p];
    }
    Object.defineProperty(exports, "__esModule", { value: true });
    __export(svg$1);
    __export(canvas);
    __export(html$3);

    });

    var abstractMeasurer = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2017-present Palantir Technologies, Inc. All rights reserved.
     * Licensed under the MIT License (the "License"); you may obtain a copy of the
     * license at https://github.com/palantir/typesettable/blob/develop/LICENSE
     */
    Object.defineProperty(exports, "__esModule", { value: true });
    var AbstractMeasurer = (function () {
        function AbstractMeasurer(ruler) {
            if (ruler.createRuler != null) {
                this.ruler = ruler.createRuler();
            }
            else {
                this.ruler = ruler;
            }
        }
        AbstractMeasurer.prototype.measure = function (text) {
            if (text === void 0) { text = AbstractMeasurer.HEIGHT_TEXT; }
            return this.ruler(text);
        };
        return AbstractMeasurer;
    }());
    /**
     * A string representing the full ascender/descender range of your text.
     *
     * Note that this is really only applicable to western alphabets. If you are
     * using a different locale language such as arabic or chinese, you may want
     * to override this.
     */
    AbstractMeasurer.HEIGHT_TEXT = "bdpql";
    exports.AbstractMeasurer = AbstractMeasurer;

    });

    var measurer = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2017-present Palantir Technologies, Inc. All rights reserved.
     * Licensed under the MIT License (the "License"); you may obtain a copy of the
     * license at https://github.com/palantir/typesettable/blob/develop/LICENSE
     */
    var __extends = (commonjsGlobal && commonjsGlobal.__extends) || (function () {
        var extendStatics = Object.setPrototypeOf ||
            ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) ||
            function (d, b) { for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p]; };
        return function (d, b) {
            extendStatics(d, b);
            function __() { this.constructor = d; }
            d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __());
        };
    })();
    Object.defineProperty(exports, "__esModule", { value: true });

    var Measurer = (function (_super) {
        __extends(Measurer, _super);
        function Measurer(ruler, useGuards) {
            if (useGuards === void 0) { useGuards = false; }
            var _this = _super.call(this, ruler) || this;
            _this.useGuards = useGuards;
            return _this;
        }
        // Guards assures same line height and width of whitespaces on both ends.
        Measurer.prototype._addGuards = function (text) {
            return abstractMeasurer.AbstractMeasurer.HEIGHT_TEXT + text + abstractMeasurer.AbstractMeasurer.HEIGHT_TEXT;
        };
        Measurer.prototype._measureLine = function (line, forceGuards) {
            if (forceGuards === void 0) { forceGuards = false; }
            var useGuards = this.useGuards || forceGuards || /^[\t ]$/.test(line);
            var measuredLine = useGuards ? this._addGuards(line) : line;
            var measuredLineDimensions = _super.prototype.measure.call(this, measuredLine);
            measuredLineDimensions.width -= useGuards ? (2 * this.getGuardWidth()) : 0;
            return measuredLineDimensions;
        };
        Measurer.prototype.measure = function (text) {
            var _this = this;
            if (text === void 0) { text = abstractMeasurer.AbstractMeasurer.HEIGHT_TEXT; }
            if (text.trim() === "") {
                return { width: 0, height: 0 };
            }
            var linesDimensions = text.trim().split("\n").map(function (line) { return _this._measureLine(line); });
            return {
                height: linesDimensions.reduce(function (acc, dim) { return acc + dim.height; }, 0),
                width: linesDimensions.reduce(function (acc, dim) { return Math.max(acc, dim.width); }, 0),
            };
        };
        Measurer.prototype.getGuardWidth = function () {
            if (this.guardWidth == null) {
                this.guardWidth = _super.prototype.measure.call(this).width;
            }
            return this.guardWidth;
        };
        return Measurer;
    }(abstractMeasurer.AbstractMeasurer));
    exports.Measurer = Measurer;

    });

    var characterMeasurer = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2017-present Palantir Technologies, Inc. All rights reserved.
     * Licensed under the MIT License (the "License"); you may obtain a copy of the
     * license at https://github.com/palantir/typesettable/blob/develop/LICENSE
     */
    /* istanbul ignore next */
    var __extends = (commonjsGlobal && commonjsGlobal.__extends) || (function () {
        var extendStatics = Object.setPrototypeOf ||
            ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) ||
            function (d, b) { for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p]; };
        return function (d, b) {
            extendStatics(d, b);
            function __() { this.constructor = d; }
            d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __());
        };
    })();
    Object.defineProperty(exports, "__esModule", { value: true });

    var CharacterMeasurer = (function (_super) {
        __extends(CharacterMeasurer, _super);
        function CharacterMeasurer() {
            return _super !== null && _super.apply(this, arguments) || this;
        }
        CharacterMeasurer.prototype._measureCharacter = function (c) {
            return _super.prototype._measureLine.call(this, c);
        };
        CharacterMeasurer.prototype._measureLine = function (line) {
            var _this = this;
            var charactersDimensions = line.split("").map(function (c) { return _this._measureCharacter(c); });
            return {
                height: charactersDimensions.reduce(function (acc, dim) { return Math.max(acc, dim.height); }, 0),
                width: charactersDimensions.reduce(function (acc, dim) { return acc + dim.width; }, 0),
            };
        };
        return CharacterMeasurer;
    }(measurer.Measurer));
    exports.CharacterMeasurer = CharacterMeasurer;

    });

    var cacheCharacterMeasurer = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2017-present Palantir Technologies, Inc. All rights reserved.
     * Licensed under the MIT License (the "License"); you may obtain a copy of the
     * license at https://github.com/palantir/typesettable/blob/develop/LICENSE
     */
    /* istanbul ignore next */
    var __extends = (commonjsGlobal && commonjsGlobal.__extends) || (function () {
        var extendStatics = Object.setPrototypeOf ||
            ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) ||
            function (d, b) { for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p]; };
        return function (d, b) {
            extendStatics(d, b);
            function __() { this.constructor = d; }
            d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __());
        };
    })();
    Object.defineProperty(exports, "__esModule", { value: true });


    var CacheCharacterMeasurer = (function (_super) {
        __extends(CacheCharacterMeasurer, _super);
        function CacheCharacterMeasurer(ruler, useGuards) {
            var _this = _super.call(this, ruler, useGuards) || this;
            _this.cache = new utils.Cache(function (c) {
                return _this._measureCharacterNotFromCache(c);
            });
            return _this;
        }
        CacheCharacterMeasurer.prototype._measureCharacterNotFromCache = function (c) {
            return _super.prototype._measureCharacter.call(this, c);
        };
        CacheCharacterMeasurer.prototype._measureCharacter = function (c) {
            return this.cache.get(c);
        };
        CacheCharacterMeasurer.prototype.reset = function () {
            this.cache.clear();
        };
        return CacheCharacterMeasurer;
    }(characterMeasurer.CharacterMeasurer));
    exports.CacheCharacterMeasurer = CacheCharacterMeasurer;

    });

    var cacheMeasurer = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2017-present Palantir Technologies, Inc. All rights reserved.
     * Licensed under the MIT License (the "License"); you may obtain a copy of the
     * license at https://github.com/palantir/typesettable/blob/develop/LICENSE
     */
    /* istanbul ignore next */
    var __extends = (commonjsGlobal && commonjsGlobal.__extends) || (function () {
        var extendStatics = Object.setPrototypeOf ||
            ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) ||
            function (d, b) { for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p]; };
        return function (d, b) {
            extendStatics(d, b);
            function __() { this.constructor = d; }
            d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __());
        };
    })();
    Object.defineProperty(exports, "__esModule", { value: true });



    var CacheMeasurer = (function (_super) {
        __extends(CacheMeasurer, _super);
        function CacheMeasurer(ruler) {
            var _this = _super.call(this, ruler) || this;
            _this.dimCache = new utils.Cache(function (s) {
                return _this._measureNotFromCache(s);
            });
            return _this;
        }
        CacheMeasurer.prototype._measureNotFromCache = function (s) {
            return _super.prototype.measure.call(this, s);
        };
        CacheMeasurer.prototype.measure = function (s) {
            if (s === void 0) { s = abstractMeasurer.AbstractMeasurer.HEIGHT_TEXT; }
            return this.dimCache.get(s);
        };
        CacheMeasurer.prototype.reset = function () {
            this.dimCache.clear();
            _super.prototype.reset.call(this);
        };
        return CacheMeasurer;
    }(cacheCharacterMeasurer.CacheCharacterMeasurer));
    exports.CacheMeasurer = CacheMeasurer;

    });

    var measurers = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2017-present Palantir Technologies, Inc. All rights reserved.
     * Licensed under the MIT License (the "License"); you may obtain a copy of the
     * license at https://github.com/palantir/typesettable/blob/develop/LICENSE
     */
    function __export(m) {
        for (var p in m) if (!exports.hasOwnProperty(p)) exports[p] = m[p];
    }
    Object.defineProperty(exports, "__esModule", { value: true });
    __export(abstractMeasurer);
    __export(cacheCharacterMeasurer);
    __export(cacheMeasurer);
    __export(characterMeasurer);
    __export(measurer);

    });

    var wrapper = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2017-present Palantir Technologies, Inc. All rights reserved.
     * Licensed under the MIT License (the "License"); you may obtain a copy of the
     * license at https://github.com/palantir/typesettable/blob/develop/LICENSE
     */
    Object.defineProperty(exports, "__esModule", { value: true });

    var Wrapper = (function () {
        function Wrapper() {
            this.maxLines(Infinity);
            this.textTrimming("ellipsis");
            this.allowBreakingWords(false);
            this._tokenizer = new utils.Tokenizer();
            this._breakingCharacter = "-";
        }
        Wrapper.prototype.maxLines = function (noLines) {
            if (noLines == null) {
                return this._maxLines;
            }
            else {
                this._maxLines = noLines;
                return this;
            }
        };
        Wrapper.prototype.textTrimming = function (option) {
            if (option == null) {
                return this._textTrimming;
            }
            else {
                if (option !== "ellipsis" && option !== "none") {
                    throw new Error(option + " - unsupported text trimming option.");
                }
                this._textTrimming = option;
                return this;
            }
        };
        Wrapper.prototype.allowBreakingWords = function (allow) {
            if (allow == null) {
                return this._allowBreakingWords;
            }
            else {
                this._allowBreakingWords = allow;
                return this;
            }
        };
        Wrapper.prototype.wrap = function (text, measurer, width, height) {
            var _this = this;
            if (height === void 0) { height = Infinity; }
            var initialWrappingResult = {
                noBrokeWords: 0,
                noLines: 0,
                originalText: text,
                truncatedText: "",
                wrappedText: "",
            };
            var state = {
                availableLines: Math.min(Math.floor(height / measurer.measure().height), this._maxLines),
                availableWidth: width,
                canFitText: true,
                currentLine: "",
                wrapping: initialWrappingResult,
            };
            var lines = text.split("\n");
            return lines.reduce(function (s, line, i) {
                return _this.breakLineToFitWidth(s, line, i !== lines.length - 1, measurer);
            }, state).wrapping;
        };
        Wrapper.prototype.breakLineToFitWidth = function (state, line, hasNextLine, measurer) {
            var _this = this;
            if (!state.canFitText && state.wrapping.truncatedText !== "") {
                state.wrapping.truncatedText += "\n";
            }
            var tokens = this._tokenizer.tokenize(line);
            state = tokens.reduce(function (s, token) {
                return _this.wrapNextToken(token, s, measurer);
            }, state);
            var wrappedText = utils.StringMethods.trimEnd(state.currentLine);
            state.wrapping.noLines += +(wrappedText !== "");
            if (state.wrapping.noLines === state.availableLines && this._textTrimming !== "none" && hasNextLine) {
                // Note: no need to add more ellipses, they were added in `wrapNextToken`
                state.canFitText = false;
            }
            else {
                state.wrapping.wrappedText += wrappedText;
            }
            state.currentLine = "\n";
            return state;
        };
        Wrapper.prototype.canFitToken = function (token, width, measurer) {
            var _this = this;
            var possibleBreaks = token.split("").map(function (c, i) { return (i !== token.length - 1) ? c + _this._breakingCharacter : c; });
            return (measurer.measure(token).width <= width) || possibleBreaks.every(function (c) { return measurer.measure(c).width <= width; });
        };
        Wrapper.prototype.addEllipsis = function (line, width, measurer) {
            if (this._textTrimming === "none") {
                return {
                    remainingToken: "",
                    wrappedToken: line,
                };
            }
            var truncatedLine = line.substring(0).trim();
            var lineWidth = measurer.measure(truncatedLine).width;
            var ellipsesWidth = measurer.measure("...").width;
            var prefix = (line.length > 0 && line[0] === "\n") ? "\n" : "";
            if (width <= ellipsesWidth) {
                var periodWidth = ellipsesWidth / 3;
                var numPeriodsThatFit = Math.floor(width / periodWidth);
                return {
                    remainingToken: line,
                    wrappedToken: prefix + "...".substr(0, numPeriodsThatFit),
                };
            }
            while (lineWidth + ellipsesWidth > width) {
                truncatedLine = utils.StringMethods.trimEnd(truncatedLine.substr(0, truncatedLine.length - 1));
                lineWidth = measurer.measure(truncatedLine).width;
            }
            return {
                remainingToken: utils.StringMethods.trimEnd(line.substring(truncatedLine.length), "-").trim(),
                wrappedToken: prefix + truncatedLine + "...",
            };
        };
        Wrapper.prototype.wrapNextToken = function (token, state, measurer) {
            if (!state.canFitText ||
                state.availableLines === state.wrapping.noLines ||
                !this.canFitToken(token, state.availableWidth, measurer)) {
                return this.finishWrapping(token, state, measurer);
            }
            var remainingToken = token;
            while (remainingToken) {
                var result = this.breakTokenToFitInWidth(remainingToken, state.currentLine, state.availableWidth, measurer);
                state.currentLine = result.line;
                remainingToken = result.remainingToken;
                if (remainingToken != null) {
                    state.wrapping.noBrokeWords += +result.breakWord;
                    ++state.wrapping.noLines;
                    if (state.availableLines === state.wrapping.noLines) {
                        var ellipsisResult = this.addEllipsis(state.currentLine, state.availableWidth, measurer);
                        state.wrapping.wrappedText += ellipsisResult.wrappedToken;
                        state.wrapping.truncatedText += ellipsisResult.remainingToken + remainingToken;
                        state.currentLine = "\n";
                        return state;
                    }
                    else {
                        state.wrapping.wrappedText += utils.StringMethods.trimEnd(state.currentLine);
                        state.currentLine = "\n";
                    }
                }
            }
            return state;
        };
        Wrapper.prototype.finishWrapping = function (token, state, measurer) {
            // Token is really long, but we have a space to put part of the word.
            if (state.canFitText &&
                state.availableLines !== state.wrapping.noLines &&
                this._textTrimming !== "none") {
                var res = this.addEllipsis(state.currentLine + token, state.availableWidth, measurer);
                state.wrapping.wrappedText += res.wrappedToken;
                state.wrapping.truncatedText += res.remainingToken;
                state.wrapping.noBrokeWords += +(res.remainingToken.length < token.length);
                state.wrapping.noLines += +(res.wrappedToken.length > 0);
                state.currentLine = "";
            }
            else {
                state.wrapping.truncatedText += token;
            }
            state.canFitText = false;
            return state;
        };
        /**
         * Breaks single token to fit current line.
         * If token contains only whitespaces then they will not be populated to next line.
         */
        Wrapper.prototype.breakTokenToFitInWidth = function (token, line, availableWidth, measurer, breakingCharacter) {
            if (breakingCharacter === void 0) { breakingCharacter = this._breakingCharacter; }
            if (measurer.measure(line + token).width <= availableWidth) {
                return {
                    breakWord: false,
                    line: line + token,
                    remainingToken: null,
                };
            }
            if (token.trim() === "") {
                return {
                    breakWord: false,
                    line: line,
                    remainingToken: "",
                };
            }
            // if we don't allow breaking words AND the token isn't the only thing on the current line.
            if (!this._allowBreakingWords && line.trim() !== "") {
                return {
                    breakWord: false,
                    line: line,
                    remainingToken: token,
                };
            }
            var fitTokenLength = 0;
            while (fitTokenLength < token.length) {
                if (measurer.measure(line + token.substring(0, fitTokenLength + 1) + breakingCharacter).width <= availableWidth) {
                    ++fitTokenLength;
                }
                else {
                    break;
                }
            }
            var suffix = "";
            if (fitTokenLength > 0) {
                suffix = breakingCharacter;
            }
            return {
                breakWord: fitTokenLength > 0,
                line: line + token.substring(0, fitTokenLength) + suffix,
                remainingToken: token.substring(fitTokenLength),
            };
        };
        return Wrapper;
    }());
    exports.Wrapper = Wrapper;

    });

    var singleLineWrapper = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2017-present Palantir Technologies, Inc. All rights reserved.
     * Licensed under the MIT License (the "License"); you may obtain a copy of the
     * license at https://github.com/palantir/typesettable/blob/develop/LICENSE
     */
    var __extends = (commonjsGlobal && commonjsGlobal.__extends) || (function () {
        var extendStatics = Object.setPrototypeOf ||
            ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) ||
            function (d, b) { for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p]; };
        return function (d, b) {
            extendStatics(d, b);
            function __() { this.constructor = d; }
            d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __());
        };
    })();
    Object.defineProperty(exports, "__esModule", { value: true });

    var SingleLineWrapper = (function (_super) {
        __extends(SingleLineWrapper, _super);
        function SingleLineWrapper() {
            return _super !== null && _super.apply(this, arguments) || this;
        }
        SingleLineWrapper.prototype.wrap = function (text, measurer, width, height) {
            var _this = this;
            if (height === void 0) { height = Infinity; }
            var lines = text.split("\n");
            if (lines.length > 1) {
                throw new Error("SingleLineWrapper is designed to work only on single line");
            }
            var wrapFN = function (w) { return _super.prototype.wrap.call(_this, text, measurer, w, height); };
            var result = wrapFN(width);
            if (result.noLines < 2) {
                return result;
            }
            var left = 0;
            var right = width;
            for (var i = 0; i < SingleLineWrapper.NO_WRAP_ITERATIONS && right > left; ++i) {
                var currentWidth = (right + left) / 2;
                var currentResult = wrapFN(currentWidth);
                if (this.areSameResults(result, currentResult)) {
                    right = currentWidth;
                    result = currentResult;
                }
                else {
                    left = currentWidth;
                }
            }
            return result;
        };
        SingleLineWrapper.prototype.areSameResults = function (one, two) {
            return one.noLines === two.noLines && one.truncatedText === two.truncatedText;
        };
        return SingleLineWrapper;
    }(wrapper.Wrapper));
    SingleLineWrapper.NO_WRAP_ITERATIONS = 5;
    exports.SingleLineWrapper = SingleLineWrapper;

    });

    var wrappers = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2017-present Palantir Technologies, Inc. All rights reserved.
     * Licensed under the MIT License (the "License"); you may obtain a copy of the
     * license at https://github.com/palantir/typesettable/blob/develop/LICENSE
     */
    function __export(m) {
        for (var p in m) if (!exports.hasOwnProperty(p)) exports[p] = m[p];
    }
    Object.defineProperty(exports, "__esModule", { value: true });
    __export(singleLineWrapper);
    __export(wrapper);

    });

    var typesetter = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2017-present Palantir Technologies, Inc. All rights reserved.
     * Licensed under the MIT License (the "License"); you may obtain a copy of the
     * license at https://github.com/palantir/typesettable/blob/develop/LICENSE
     */
    Object.defineProperty(exports, "__esModule", { value: true });




    /**
     * This is a convenience interface for typesetting strings using the default
     * measurer/wrapper/writer setup.
     */
    var Typesetter = (function () {
        function Typesetter(context) {
            this.context = context;
            this.measurer = new measurers.CacheMeasurer(this.context);
            this.wrapper = new wrappers.Wrapper();
            this.writer = new writers.Writer(this.measurer, this.context, this.wrapper);
        }
        Typesetter.svg = function (element, className, addTitleElement) {
            return new Typesetter(new contexts.SvgContext(element, className, addTitleElement));
        };
        Typesetter.canvas = function (ctx, lineHeight, style) {
            return new Typesetter(new contexts.CanvasContext(ctx, lineHeight, style));
        };
        Typesetter.html = function (element, className, addTitle) {
            return new Typesetter(new contexts.HtmlContext(element, className, addTitle));
        };
        /**
         * Wraps the given string into the width/height and writes it into the
         * canvas or SVG (depending on context).
         *
         * Delegates to `Writer.write` using the internal `ITypesetterContext`.
         */
        Typesetter.prototype.write = function (text, width, height, options, into) {
            this.writer.write(text, width, height, options, into);
        };
        /**
         * Clears the `Measurer`'s CacheMeasurer.
         *
         * Call this if your font style changee in SVG.
         */
        Typesetter.prototype.clearMeasurerCache = function () {
            this.measurer.reset();
        };
        return Typesetter;
    }());
    exports.Typesetter = Typesetter;

    });

    var src$1 = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2017-present Palantir Technologies, Inc. All rights reserved.
     * Licensed under the MIT License (the "License"); you may obtain a copy of the
     * license at https://github.com/palantir/typesettable/blob/develop/LICENSE
     */
    function __export(m) {
        for (var p in m) if (!exports.hasOwnProperty(p)) exports[p] = m[p];
    }
    Object.defineProperty(exports, "__esModule", { value: true });
    __export(contexts);
    __export(measurers);
    __export(typesetter);
    __export(utils);
    __export(wrappers);
    __export(writers);

    });

    var arrayUtils = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });

    var nativeArray = window.Array;
    /**
     * Takes two arrays of numbers and adds them together
     *
     * @param {number[]} aList The first array of numbers
     * @param {number[]} bList The second array of numbers
     * @return {number[]} An array of numbers where x[i] = aList[i] + bList[i]
     */
    function add(aList, bList) {
        if (aList.length !== bList.length) {
            throw new Error("attempted to add arrays of unequal length");
        }
        return aList.map(function (_, i) { return aList[i] + bList[i]; });
    }
    exports.add = add;
    /**
     * Take an array of values, and return the unique values.
     * Will work iff ∀ a, b, a.toString() == b.toString() => a == b; will break on Object inputs
     *
     * @param {T[]} values The values to find uniqueness for
     * @return {T[]} The unique values
     */
    function uniq(arr) {
        var seen = d3.set();
        var result = [];
        arr.forEach(function (x) {
            if (!seen.has(String(x))) {
                seen.add(String(x));
                result.push(x);
            }
        });
        return result;
    }
    exports.uniq = uniq;
    /**
     * @param {T[][]} a The 2D array that will have its elements joined together.
     * @return {T[]} Every array in a, concatenated together in the order they appear.
     */
    function flatten(a) {
        return nativeArray.prototype.concat.apply([], a);
    }
    exports.flatten = flatten;
    /**
     * Creates an array of length `count`, filled with value or (if value is a function), value()
     *
     * @param {T | ((index?: number) => T)} value The value to fill the array with or a value generator (called with index as arg)
     * @param {number} count The length of the array to generate
     * @return {any[]}
     */
    function createFilledArray(value, count) {
        var out = [];
        for (var i = 0; i < count; i++) {
            out[i] = typeof (value) === "function" ? value(i) : value;
        }
        return out;
    }
    exports.createFilledArray = createFilledArray;
    });

    var colorUtils = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });

    var nativeMath = window.Math;
    /**
     * Return contrast ratio between two colors
     * Based on implementation from chroma.js by Gregor Aisch (gka) (licensed under BSD)
     * chroma.js may be found here: https://github.com/gka/chroma.js
     * License may be found here: https://github.com/gka/chroma.js/blob/master/LICENSE
     * see http://www.w3.org/TR/2008/REC-WCAG20-20081211/#contrast-ratiodef
     */
    function contrast(a, b) {
        var l1 = luminance(a) + 0.05;
        var l2 = luminance(b) + 0.05;
        return l1 > l2 ? l1 / l2 : l2 / l1;
    }
    exports.contrast = contrast;
    /**
     * Returns a brighter copy of this color. Each channel is multiplied by 0.7 ^ -factor.
     * Channel values are capped at the maximum value of 255, and the minimum value of 30.
     */
    function lightenColor(color, factor) {
        var brightened = d3.color(color).brighter(factor);
        return brightened.rgb().toString();
    }
    exports.lightenColor = lightenColor;
    /**
     * Gets the Hex Code of the color resulting by applying the className CSS class to the
     * colorTester selection. Returns null if the tester is transparent.
     *
     * @param {d3.Selection<void>} colorTester The d3 selection to apply the CSS class to
     * @param {string} className The name of the class to be applied
     * @return {string} The hex code of the computed color
     */
    function colorTest(colorTester, className) {
        colorTester.classed(className, true);
        // Use regex to get the text inside the rgb parentheses
        var colorStyle = colorTester.style("background-color");
        if (colorStyle === "transparent") {
            return null;
        }
        var match = /\((.+)\)/.exec(colorStyle);
        if (!match) {
            return null;
        }
        var rgb = match[1]
            .split(",")
            .map(function (colorValue) {
            var colorNumber = +colorValue;
            var hexValue = colorNumber.toString(16);
            return colorNumber < 16 ? "0" + hexValue : hexValue;
        });
        if (rgb.length === 4 && rgb[3] === "00") {
            return null;
        }
        var hexCode = "#" + rgb.join("");
        colorTester.classed(className, false);
        return hexCode;
    }
    exports.colorTest = colorTest;
    /**
     * Return relative luminance (defined here: http://www.w3.org/TR/2008/REC-WCAG20-20081211/#relativeluminancedef)
     * Based on implementation from chroma.js by Gregor Aisch (gka) (licensed under BSD)
     * chroma.js may be found here: https://github.com/gka/chroma.js
     * License may be found here: https://github.com/gka/chroma.js/blob/master/LICENSE
     */
    function luminance(color) {
        var rgb = d3.rgb(color);
        var lum = function (x) {
            x = x / 255;
            return x <= 0.03928 ? x / 12.92 : nativeMath.pow((x + 0.055) / 1.055, 2.4);
        };
        var r = lum(rgb.r);
        var g = lum(rgb.g);
        var b = lum(rgb.b);
        return 0.2126 * r + 0.7152 * g + 0.0722 * b;
    }
    });

    var domUtils = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });

    var nativeMath = window.Math;
    /**
     * Returns whether the child is in fact a child of the parent
     */
    function contains(parent, child) {
        var maybeParent = child;
        while (maybeParent != null && maybeParent !== parent) {
            maybeParent = maybeParent.parentNode;
        }
        return maybeParent === parent;
    }
    exports.contains = contains;
    /**
     * Gets the bounding box of an element.
     * @param {d3.Selection} element
     * @returns {SVGRed} The bounding box.
     */
    function elementBBox(element) {
        var bbox;
        // HACKHACK: Firefox won't correctly measure nodes with style "display: none" or their descendents (FF Bug 612118).
        try {
            bbox = element.node().getBBox();
        }
        catch (err) {
            bbox = { x: 0, y: 0, width: 0, height: 0 };
        }
        return bbox;
    }
    exports.elementBBox = elementBBox;
    function entityBounds(element) {
        if (element instanceof SVGElement) {
            return elementBBox(d3.select(element));
        }
        else if (element instanceof HTMLElement) {
            var rect = element.getBoundingClientRect();
            return { x: rect.left, y: rect.top, width: rect.width, height: rect.height };
        }
        else {
            return { x: 0, y: 0, width: 0, height: 0 };
        }
    }
    exports.entityBounds = entityBounds;
    /**
     * Screen refresh rate which is assumed to be 60fps
     */
    exports.SCREEN_REFRESH_RATE_MILLISECONDS = 1000 / 60;
    /**
     * Polyfill for `window.requestAnimationFrame`.
     * If the function exists, then we use the function directly.
     * Otherwise, we set a timeout on `SCREEN_REFRESH_RATE_MILLISECONDS` and then perform the function.
     *
     * @param {() => void} callback The callback to call in the next animation frame
     */
    function requestAnimationFramePolyfill(callback) {
        if (window.requestAnimationFrame != null) {
            window.requestAnimationFrame(callback);
        }
        else {
            setTimeout(callback, exports.SCREEN_REFRESH_RATE_MILLISECONDS);
        }
    }
    exports.requestAnimationFramePolyfill = requestAnimationFramePolyfill;
    /**
     * Calculates the width of the element.
     * The width includes the padding and the border on the element's left and right sides.
     *
     * @param {Element} element The element to query
     * @returns {number} The width of the element.
     */
    function elementWidth(elementOrSelection) {
        var element = elementOrSelection instanceof d3.selection
            ? elementOrSelection.node()
            : elementOrSelection;
        var style = window.getComputedStyle(element);
        return _parseStyleValue(style, "width")
            + _parseStyleValue(style, "padding-left")
            + _parseStyleValue(style, "padding-right")
            + _parseStyleValue(style, "border-left-width")
            + _parseStyleValue(style, "border-right-width");
    }
    exports.elementWidth = elementWidth;
    /**
     * Calculates the height of the element.
     * The height includes the padding the and the border on the element's top and bottom sides.
     *
     * @param {Element} element The element to query
     * @returns {number} The height of the element
     */
    function elementHeight(elementOrSelection) {
        var element = elementOrSelection instanceof d3.selection
            ? elementOrSelection.node()
            : elementOrSelection;
        var style = window.getComputedStyle(element);
        return _parseStyleValue(style, "height")
            + _parseStyleValue(style, "padding-top")
            + _parseStyleValue(style, "padding-bottom")
            + _parseStyleValue(style, "border-top-width")
            + _parseStyleValue(style, "border-bottom-width");
    }
    exports.elementHeight = elementHeight;
    // taken from the BNF at https://www.w3.org/TR/SVG/coords.html
    var WSP = "\\s";
    var NUMBER = "(?:[-+]?[0-9]*\\.?[0-9]+)";
    var COMMA_WSP = "(?:(?:" + WSP + "+,?" + WSP + "*)|(?:," + WSP + "*))";
    var TRANSLATE_REGEX = new RegExp("translate" + WSP + "*\\(" + WSP + "*(" + NUMBER + ")(?:" + COMMA_WSP + "(" + NUMBER + "))?" + WSP + "*\\)");
    var ROTATE_REGEX = new RegExp("rotate" + WSP + "*\\(" + WSP + "*(" + NUMBER + ")" + WSP + "*\\)");
    var SCALE_REGEX = new RegExp("scale" + WSP + "*\\(" + WSP + "*(" + NUMBER + ")(?:" + COMMA_WSP + "(" + NUMBER + "))?" + WSP + "*\\)");
    /**
     * Accepts selections whose .transform contain a "translate(a, b)" and extracts the a and b
     */
    function getTranslateValues(el) {
        var match = TRANSLATE_REGEX.exec(el.attr("transform"));
        if (match != null) {
            var translateX = match[1], _a = match[2], translateY = _a === void 0 ? 0 : _a;
            return [+translateX, +translateY];
        }
        else {
            return [0, 0];
        }
    }
    exports.getTranslateValues = getTranslateValues;
    /**
     * Accepts selections whose .transform contain a "rotate(angle)" and returns the angle
     */
    function getRotate(el) {
        var match = ROTATE_REGEX.exec(el.attr("transform"));
        if (match != null) {
            var rotation = match[1];
            return +rotation;
        }
        else {
            return 0;
        }
    }
    exports.getRotate = getRotate;
    function getScaleValues(el) {
        var match = SCALE_REGEX.exec(el.attr("transform"));
        if (match != null) {
            var scaleX = match[1], scaleY = match[2];
            return [+scaleX, scaleY == null ? +scaleX : +scaleY];
        }
        else {
            return [0, 0];
        }
    }
    exports.getScaleValues = getScaleValues;
    /**
     * Checks if the first ClientRect overlaps the second.
     *
     * @param {ClientRect} clientRectA The first ClientRect
     * @param {ClientRect} clientRectB The second ClientRect
     * @returns {boolean} If the ClientRects overlap each other.
     */
    function clientRectsOverlap(clientRectA, clientRectB) {
        if (nativeMath.floor(clientRectA.right) <= nativeMath.ceil(clientRectB.left)) {
            return false;
        }
        if (nativeMath.ceil(clientRectA.left) >= nativeMath.floor(clientRectB.right)) {
            return false;
        }
        if (nativeMath.floor(clientRectA.bottom) <= nativeMath.ceil(clientRectB.top)) {
            return false;
        }
        if (nativeMath.ceil(clientRectA.top) >= nativeMath.floor(clientRectB.bottom)) {
            return false;
        }
        return true;
    }
    exports.clientRectsOverlap = clientRectsOverlap;
    /**
     * Return a new ClientRect that is the old ClientRect expanded by amount in all directions.
     * @param rect
     * @param amount
     */
    function expandRect(rect, amount) {
        return {
            left: rect.left - amount,
            top: rect.top - amount,
            right: rect.right + amount,
            bottom: rect.bottom + amount,
            width: rect.width + amount * 2,
            height: rect.height + amount * 2,
        };
    }
    exports.expandRect = expandRect;
    /**
     * Returns true if and only if innerClientRect is inside outerClientRect.
     *
     * @param {ClientRect} innerClientRect The first ClientRect
     * @param {ClientRect} outerClientRect The second ClientRect
     * @returns {boolean} If and only if the innerClientRect is inside outerClientRect.
     */
    function clientRectInside(innerClientRect, outerClientRect) {
        return (nativeMath.floor(outerClientRect.left) <= nativeMath.ceil(innerClientRect.left) &&
            nativeMath.floor(outerClientRect.top) <= nativeMath.ceil(innerClientRect.top) &&
            nativeMath.floor(innerClientRect.right) <= nativeMath.ceil(outerClientRect.right) &&
            nativeMath.floor(innerClientRect.bottom) <= nativeMath.ceil(outerClientRect.bottom));
    }
    exports.clientRectInside = clientRectInside;
    /**
     * Returns true if the supplied coordinates or Ranges intersect or are contained by bbox.
     *
     * @param {number | Range} xValOrRange The x coordinate or Range to test
     * @param {number | Range} yValOrRange The y coordinate or Range to test
     * @param {SVGRect} bbox The bbox
     * @param {number} tolerance Amount by which to expand bbox, in each dimension, before
     * testing intersection
     *
     * @returns {boolean} True if the supplied coordinates or Ranges intersect or are
     * contained by bbox, false otherwise.
     */
    function intersectsBBox(xValOrRange, yValOrRange, bbox, tolerance) {
        if (tolerance === void 0) { tolerance = 0.5; }
        var xRange = _parseRange(xValOrRange);
        var yRange = _parseRange(yValOrRange);
        // SVGRects are positioned with sub-pixel accuracy (the default unit
        // for the x, y, height & width attributes), but user selections (e.g. via
        // mouse events) usually have pixel accuracy. A tolerance of half-a-pixel
        // seems appropriate.
        return bbox.x + bbox.width >= xRange.min - tolerance &&
            bbox.x <= xRange.max + tolerance &&
            bbox.y + bbox.height >= yRange.min - tolerance &&
            bbox.y <= yRange.max + tolerance;
    }
    exports.intersectsBBox = intersectsBBox;
    /**
     * Create a Range from a number or an object with "min" and "max" defined.
     *
     * @param {any} input The object to parse
     *
     * @returns {Range} The generated Range
     */
    function _parseRange(input) {
        if (typeof (input) === "number") {
            var value = input;
            return { min: value, max: value };
        }
        var range = input;
        if (range instanceof Object && "min" in range && "max" in range) {
            return range;
        }
        throw new Error("input '" + input + "' can't be parsed as an Range");
    }
    function _parseStyleValue(style, property) {
        var value = style.getPropertyValue(property);
        var parsedValue = parseFloat(value);
        return parsedValue || 0;
    }
    /**
     * Returns an array containing all ancestor `HTMLElement`s, starting at the
     * provided element and usually ending with the `<body>` element.
     */
    function getHtmlElementAncestors(elem) {
        var elems = [];
        while (elem && elem instanceof HTMLElement) {
            elems.push(elem);
            elem = elem.parentElement;
        }
        return elems;
    }
    exports.getHtmlElementAncestors = getHtmlElementAncestors;
    /**
     * Returns the `ICssTransformMatrix` of an element, if defined in its computed
     * style. Returns `null` if there is no transform on the element.
     */
    function getElementTransform(elem) {
        var style = window.getComputedStyle(elem, null);
        var transform = style.getPropertyValue("-webkit-transform") ||
            style.getPropertyValue("-moz-transform") ||
            style.getPropertyValue("-ms-transform") ||
            style.getPropertyValue("-o-transform") ||
            style.getPropertyValue("transform");
        return parseTransformMatrix(transform);
    }
    exports.getElementTransform = getElementTransform;
    var _MATRIX_REGEX = /^matrix\(([^)]+)\)$/;
    var _SPLIT_REGEX = /[, ]+/;
    /**
     * Attempts to parse a string such as `"matrix(1, 0, 1, 1, 100, 0)"` into an
     * array such as `[1, 0, 1, 1, 100, 0]`.
     *
     * If unable to do so, `null` is returned.
     */
    function parseTransformMatrix(transform) {
        if (transform == null || transform === "none") {
            return null;
        }
        var matrixStrings = transform.match(_MATRIX_REGEX);
        if (matrixStrings == null || matrixStrings.length < 2) {
            return null;
        }
        var matrix = matrixStrings[1].split(_SPLIT_REGEX).map(function (v) { return parseFloat(v); });
        if (matrix.length != 6) {
            return null;
        }
        return matrix;
    }
    });

    var mathUtils = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });


    var nativeMath = window.Math;
    var _IDENTITY_TRANSFORM = [1, 0, 0, 1, 0, 0];
    /**
     * Checks if x is between a and b.
     *
     * @param {number} x The value to test if in range
     * @param {number} a The beginning of the (inclusive) range
     * @param {number} b The ending of the (inclusive) range
     * @return {boolean} Whether x is in [a, b]
     */
    function inRange(x, a, b) {
        return (nativeMath.min(a, b) <= x && x <= nativeMath.max(a, b));
    }
    exports.inRange = inRange;
    /**
     * Clamps x to the range [min, max].
     *
     * @param {number} x The value to be clamped.
     * @param {number} min The minimum value.
     * @param {number} max The maximum value.
     * @return {number} A clamped value in the range [min, max].
     */
    function clamp(x, min, max) {
        return nativeMath.min(nativeMath.max(min, x), max);
    }
    exports.clamp = clamp;
    function max(array, firstArg, secondArg) {
        var accessor = typeof (firstArg) === "function" ? firstArg : null;
        var defaultValue = accessor == null ? firstArg : secondArg;
        var maxValue = accessor == null ? d3.max(array) : d3.max(array, accessor);
        return maxValue !== undefined ? maxValue : defaultValue;
    }
    exports.max = max;
    function min(array, firstArg, secondArg) {
        var accessor = typeof (firstArg) === "function" ? firstArg : null;
        var defaultValue = accessor == null ? firstArg : secondArg;
        var minValue = accessor == null ? d3.min(array) : d3.min(array, accessor);
        return minValue !== undefined ? minValue : defaultValue;
    }
    exports.min = min;
    /**
     * Returns true **only** if x is NaN
     */
    function isNaN(n) {
        return n !== n;
    }
    exports.isNaN = isNaN;
    /**
     * Returns true if the argument is a number, which is not NaN
     * Numbers represented as strings do not pass this function
     */
    function isValidNumber(n) {
        return typeof n === "number" && n - n < 1;
    }
    exports.isValidNumber = isValidNumber;
    /**
     * Generates an array of consecutive, strictly increasing numbers
     * in the range [start, stop) separeted by step
     */
    function range(start, stop, step) {
        if (step === void 0) { step = 1; }
        if (step === 0) {
            throw new Error("step cannot be 0");
        }
        var length = nativeMath.max(nativeMath.ceil((stop - start) / step), 0);
        var range = [];
        for (var i = 0; i < length; ++i) {
            range[i] = start + step * i;
        }
        return range;
    }
    exports.range = range;
    /**
     * Returns the square of the distance between two points
     *
     * @param {Point} p1
     * @param {Point} p2
     * @return {number} dist(p1, p2)^2
     */
    function distanceSquared(p1, p2) {
        return nativeMath.pow(p2.y - p1.y, 2) + nativeMath.pow(p2.x - p1.x, 2);
    }
    exports.distanceSquared = distanceSquared;
    function degreesToRadians(degree) {
        return degree / 360 * nativeMath.PI * 2;
    }
    exports.degreesToRadians = degreesToRadians;
    /**
     * Returns if the point is within the bounds. Points along
     * the bounds are considered "within" as well.
     * @param {Point} p Point in considerations.
     * @param {Bounds} bounds Bounds within which to check for inclusion.
     */
    function within(p, bounds) {
        return bounds.topLeft.x <= p.x
            && bounds.bottomRight.x >= p.x
            && bounds.topLeft.y <= p.y
            && bounds.bottomRight.y >= p.y;
    }
    exports.within = within;
    /**
     * Returns whether the first bounds intersects the second bounds.
     * Pass primitive numbers directly for performance.
     *
     * Assumes width and heights are positive.
     */
    function boundsIntersects(aX, aY, aWidth, aHeight, bX, bY, bWidth, bHeight) {
        return aX <= bX + bWidth &&
            bX <= aX + aWidth &&
            aY <= bY + bHeight &&
            bY <= aY + aHeight;
    }
    exports.boundsIntersects = boundsIntersects;
    /**
     * Returns a `ICssTransformMatrix` representing the cumulative transformation of
     * the element and all its parents. This transform converts from top-level
     * clientX/clientY coordinates (such as document mouse events) to internal
     * component offsetX/offsetY coordinates.
     *
     * Use `applyTransform` to convert from client coordinates to element
     * coordinates, accounting for all CSS transforms applied to that element.
     *
     * Note that this handles css `transform` but does not handle css
     * `transform-origin` values other than default ("50% 50%").
     */
    function getCumulativeTransform(element) {
        var elems = domUtils.getHtmlElementAncestors(element);
        var transform = _IDENTITY_TRANSFORM;
        var offsetParent = null;
        for (var _i = 0, elems_1 = elems; _i < elems_1.length; _i++) {
            var elem = elems_1[_i];
            // apply css transform from any ancestor element
            var elementTransform = domUtils.getElementTransform(elem);
            if (elementTransform != null) {
                var midX = elem.clientWidth / 2;
                var midY = elem.clientHeight / 2;
                transform = multiplyTranslate(transform, [midX, midY]);
                transform = multiplyMatrix(transform, invertMatrix(elementTransform));
                transform = multiplyTranslate(transform, [-midX, -midY]);
            }
            // apply scroll offsets from any ancestor element
            var offsetX = elem.scrollLeft;
            var offsetY = elem.scrollTop;
            // apply client+offset from only acenstor "offsetParent"
            if (offsetParent === null || elem === offsetParent) {
                offsetX -= elem.offsetLeft + elem.clientLeft;
                offsetY -= elem.offsetTop + elem.clientTop;
                offsetParent = elem.offsetParent;
            }
            transform = multiplyTranslate(transform, [offsetX, offsetY]);
        }
        return transform;
    }
    exports.getCumulativeTransform = getCumulativeTransform;
    /**
     * Straightforward matrix multiplication of homogenized css transform matrices.
     */
    function multiplyMatrix(a, b) {
        return [
            a[0] * b[0] + a[2] * b[1],
            a[1] * b[0] + a[3] * b[1],
            a[0] * b[2] + a[2] * b[3],
            a[1] * b[2] + a[3] * b[3],
            a[0] * b[4] + a[2] * b[5] + a[4],
            a[1] * b[4] + a[3] * b[5] + a[5],
        ];
    }
    exports.multiplyMatrix = multiplyMatrix;
    /**
     * Prepends translation to transformation matrix.
     *
     * Equivalent to `multiplyMatrix([1, 0, 0, 1, ...v], b)`
     */
    function premultiplyTranslate(v, b) {
        return [
            b[0], b[1], b[2], b[3],
            b[4] + v[0],
            b[5] + v[1],
        ];
    }
    exports.premultiplyTranslate = premultiplyTranslate;
    /**
     * Appends translation to transformation matrix.
     *
     * Equivalent to `multiplyMatrix(a, [1, 0, 0, 1, ...v])`
     */
    function multiplyTranslate(a, v) {
        return [
            a[0], a[1], a[2], a[3],
            a[0] * v[0] + a[2] * v[1] + a[4],
            a[1] * v[0] + a[3] * v[1] + a[5],
        ];
    }
    exports.multiplyTranslate = multiplyTranslate;
    /**
     * Analytical inverse of a `ICssTransformMatrix` analogous to a non-singular
     * homogenous 3x3 matrix.
     *
     * http://mathworld.wolfram.com/MatrixInverse.html
     * https://stackoverflow.com/questions/2624422/efficient-4x4-matrix-inverse-affine-transform
     */
    function invertMatrix(a) {
        var determinant = a[0] * a[3] - a[1] * a[2];
        if (determinant === 0) {
            throw new Error("singular matrix");
        }
        var inverseDeterminant = 1 / determinant;
        return [
            inverseDeterminant * a[3],
            inverseDeterminant * -a[1],
            inverseDeterminant * -a[2],
            inverseDeterminant * a[0],
            inverseDeterminant * (-a[3] * a[4] + a[2] * a[5]),
            inverseDeterminant * (a[1] * a[4] + -a[0] * a[5]),
        ];
    }
    exports.invertMatrix = invertMatrix;
    /**
     * Applies the `ICssTransformMatrix` to the `Point`.
     *
     * Returns a new `Point`.
     */
    function applyTransform(a, p) {
        return {
            x: a[0] * p.x + a[2] * p.y + a[4],
            y: a[1] * p.x + a[3] * p.y + a[5],
        };
    }
    exports.applyTransform = applyTransform;
    });

    var rTreeSplitStrategies = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2017-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });
    var SplitStrategyTrivial = /** @class */ (function () {
        function SplitStrategyTrivial() {
        }
        SplitStrategyTrivial.prototype.split = function (entries, nodes) {
            // Create simple middle split
            var mid = Math.ceil(entries.length / 2);
            for (var i = 0; i < mid; i++) {
                nodes[0].insert(entries[i]);
            }
            for (var i = mid; i < entries.length; i++) {
                nodes[1].insert(entries[i]);
            }
        };
        return SplitStrategyTrivial;
    }());
    exports.SplitStrategyTrivial = SplitStrategyTrivial;
    // Linear split method adapted from https://github.com/imbcmdth/RTree/blob/master/src/rtree.js
    var SplitStrategyLinear = /** @class */ (function () {
        function SplitStrategyLinear() {
        }
        SplitStrategyLinear.prototype.split = function (entries, nodes) {
            // copy entries before we mutate it
            entries = entries.slice();
            this.chooseFirstSplit(entries, nodes);
            while (entries.length > 0) {
                this.addNext(entries, nodes);
            }
        };
        /**
         * Choose the two farthest-apart entries to begin the split.
         */
        SplitStrategyLinear.prototype.chooseFirstSplit = function (entries, nodes) {
            // Determine entry indices that have min/max x/y coordinates
            var minXH = 0;
            var minYH = 0;
            var maxXL = entries.length - 1;
            var maxYL = entries.length - 1;
            for (var i = 1; i < entries.length - 1; i++) {
                var entry = entries[i];
                if (entry.bounds.xl > entries[maxXL].bounds.xl) {
                    maxXL = i;
                }
                else if (entry.bounds.xh < entries[minXH].bounds.xh) {
                    minXH = i;
                }
                if (entry.bounds.yl > entries[maxYL].bounds.yl) {
                    maxYL = i;
                }
                else if (entry.bounds.yh < entries[minYH].bounds.yh) {
                    minYH = i;
                }
            }
            // Choose to split x or y based on greatest difference
            var dx = Math.abs(entries[minXH].bounds.xh - entries[maxXL].bounds.xl);
            var dy = Math.abs(entries[minYH].bounds.yh - entries[maxYL].bounds.yl);
            var _a = dx > dy ? [minXH, maxXL] : [minYH, maxYL], i0 = _a[0], i1 = _a[1];
            // if no detectable split, just use first/last entries
            if (i0 === i1) {
                i0 = 0;
                i1 = entries.length - 1;
            }
            // Split off nodes. We splice with the max index first to make sure we
            // don't change the index of the second splice call
            nodes[0].insert(entries.splice(Math.max(i0, i1), 1)[0]);
            nodes[1].insert(entries.splice(Math.min(i0, i1), 1)[0]);
        };
        /**
         * Split the next entry. Choose the entry that expands its parent node's
         * area the least.
         */
        SplitStrategyLinear.prototype.addNext = function (entries, nodes) {
            var index = null;
            var minDiff = null;
            var minDiffNode = null;
            for (var i = 0; i < entries.length; i++) {
                var entry = entries[i];
                var areaDiff0 = nodes[0].unionAreaDifference(entry.bounds);
                var areaDiff1 = nodes[1].unionAreaDifference(entry.bounds);
                if (areaDiff0 < minDiff || index == null) {
                    index = i;
                    minDiff = areaDiff0;
                    minDiffNode = nodes[0];
                }
                if (areaDiff1 < minDiff) {
                    index = i;
                    minDiff = areaDiff1;
                    minDiffNode = nodes[1];
                }
            }
            minDiffNode.insert(entries.splice(index, 1)[0]);
        };
        return SplitStrategyLinear;
    }());
    exports.SplitStrategyLinear = SplitStrategyLinear;
    });

    var rTree = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2017-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });

    /**
     * The maximum number of children in an r-tree node before we attempt to split.
     * This must be >= 2.
     */
    var DEFAULT_MAX_NODE_CHILDREN = 5;
    /**
     * There are several strategies for splitting nodes that contain overlapping
     * regions. By default we use `SplitStrategyLinear` which minimizes the change
     * in node bounding box area.
     */
    var DEFAULT_SPLIT_STRATEGY = new rTreeSplitStrategies.SplitStrategyLinear();
    /**
     * The return result of predicates used with `RTree.queryNodes`.
     *
     * The `PASS_AND_OVERWRITE` value will overwrite previous results
     * when the predicate finds a more optimal result.
     */
    var QueryPredicateResult;
    (function (QueryPredicateResult) {
        QueryPredicateResult[QueryPredicateResult["PASS"] = 0] = "PASS";
        QueryPredicateResult[QueryPredicateResult["FAIL"] = 1] = "FAIL";
        QueryPredicateResult[QueryPredicateResult["PASS_AND_OVERWRITE"] = 2] = "PASS_AND_OVERWRITE";
    })(QueryPredicateResult = exports.QueryPredicateResult || (exports.QueryPredicateResult = {}));
    /**
     * Creates a node predicate for use with `RTree.queryNodes`
     *
     * @param point - the query point
     * @param nearFn - an `IDistanceFunction` from the query point to the nearest
     * point on the node bounds
     * @param farFn - an `IDistanceFunction` from the query point to the farthest
     * point on the node bounds
     */
    function createMinimizingNodePredicate(point, nearFn, farFn) {
        var nearestLeafDistance = Infinity;
        var farthestBranchDistance = Infinity;
        return function (node) {
            var near = nearFn(node.bounds, point);
            var far = farFn(node.bounds, point);
            // assumption: node.value indicates that parent is a leaf
            if (node.value != null) {
                if (near < nearestLeafDistance) {
                    nearestLeafDistance = near;
                    farthestBranchDistance = far;
                    return QueryPredicateResult.PASS_AND_OVERWRITE;
                }
                else if (near === nearestLeafDistance) {
                    return QueryPredicateResult.PASS;
                }
                else {
                    return QueryPredicateResult.FAIL;
                }
            }
            else {
                if (near > farthestBranchDistance) {
                    return QueryPredicateResult.FAIL;
                }
                else {
                    farthestBranchDistance = Math.max(far, farthestBranchDistance);
                    return QueryPredicateResult.PASS;
                }
            }
        };
    }
    exports.createMinimizingNodePredicate = createMinimizingNodePredicate;
    /**
     * Create a `Array.sort` function from a query point and a distance function.
     */
    function createNodeSort(point, distanceFn) {
        return function (a, b) {
            return distanceFn(b.bounds, point) - distanceFn(a.bounds, point);
        };
    }
    exports.createNodeSort = createNodeSort;
    /**
     * R-Tree is a multidimensional spatial region tree. It stores entries that have
     * arbitrarily overlapping bounding boxes and supports efficient point and
     * bounding box overlap queries.
     *
     * Average search time complexity is O(log_M(N)) where M = max children per node
     * and N is number of values stored in tree.
     *
     * It is similar in purpose to a quadtree except quadtrees can only store a
     * single point per entry. Also, the space-partitioning structure of quadtrees
     * provides guarantees that any given value has no neighbors closer than its
     * node's bounds, whereas r-trees provide no such guarantees.
     */
    var RTree = /** @class */ (function () {
        function RTree(maxNodeChildren, splitStrategy) {
            if (maxNodeChildren === void 0) { maxNodeChildren = DEFAULT_MAX_NODE_CHILDREN; }
            if (splitStrategy === void 0) { splitStrategy = DEFAULT_SPLIT_STRATEGY; }
            this.maxNodeChildren = maxNodeChildren;
            this.splitStrategy = splitStrategy;
            this.root = new RTreeNode(true);
            this.size = 0;
        }
        RTree.prototype.getRoot = function () {
            return this.root;
        };
        RTree.prototype.clear = function () {
            this.root = new RTreeNode(true);
            this.size = 0;
        };
        RTree.prototype.insert = function (bounds, value) {
            var node = this.root;
            // Choose subtree until we find a leaf
            while (!node.leaf) {
                node = node.subtree(bounds);
            }
            // Insert new value node into leaf node
            var valueNode = RTreeNode.valueNode(bounds, value);
            node.insert(valueNode);
            this.size += 1;
            // While node overflows, split and walk up
            while (node.overflow(this.maxNodeChildren)) {
                node = node.split(this.splitStrategy);
                if (node.parent == null) {
                    this.root = node;
                }
            }
            return valueNode;
        };
        RTree.prototype.locate = function (xy) {
            return this.query(function (b) { return b.contains(xy); });
        };
        /**
         * Returns an array of `T` values that are the "nearest" to the query point.
         *
         * Nearness is measured as the absolute distance from the query point to the
         * nearest edge of the node bounds. If the node bounds contains the query
         * point, the distance is 0.
         */
        RTree.prototype.locateNearest = function (xy) {
            var predicate = createMinimizingNodePredicate(xy, RTreeBounds.distanceSquaredToNearEdge, RTreeBounds.distanceSquaredToFarEdge);
            var nodes = this.queryNodes(predicate);
            return nodes.map(function (node) { return node.value; });
        };
        /**
         * Returns an array of `T` values that are the "nearest" to the query point.
         *
         * Nearness is measured as the 1-dimensional absolute distance from the
         * query's x point to the nearest edge of the node bounds. If the node
         * bounds contains the query point, the distance is 0.
         *
         * The results are sorted by y-coordinate nearness.
         */
        RTree.prototype.locateNearestX = function (xy) {
            var predicate = createMinimizingNodePredicate(xy, RTreeBounds.absoluteDistanceToNearEdgeX, RTreeBounds.absoluteDistanceToFarEdgeX);
            var nodes = this.queryNodes(predicate);
            nodes.sort(createNodeSort(xy, RTreeBounds.absoluteDistanceToNearEdgeY));
            return nodes.map(function (node) { return node.value; });
        };
        /**
         * Returns an array of `T` values that are the "nearest" to the query point.
         *
         * Nearness is measured as the 1-dimensional absolute distance from the
         * query's y point to the nearest edge of the node bounds. If the node
         * bounds contains the query point, the distance is 0.
         *
         * The results are sorted by x-coordinate nearness.
         */
        RTree.prototype.locateNearestY = function (xy) {
            var predicate = createMinimizingNodePredicate(xy, RTreeBounds.absoluteDistanceToNearEdgeY, RTreeBounds.absoluteDistanceToFarEdgeY);
            var nodes = this.queryNodes(predicate);
            nodes.sort(createNodeSort(xy, RTreeBounds.absoluteDistanceToNearEdgeX));
            return nodes.map(function (node) { return node.value; });
        };
        RTree.prototype.intersect = function (bounds) {
            return this.query(function (b) { return RTreeBounds.isBoundsOverlapBounds(b, bounds); });
        };
        RTree.prototype.intersectX = function (bounds) {
            return this.query(function (b) { return RTreeBounds.isBoundsOverlapX(b, bounds); });
        };
        RTree.prototype.intersectY = function (bounds) {
            return this.query(function (b) { return RTreeBounds.isBoundsOverlapY(b, bounds); });
        };
        RTree.prototype.query = function (predicate) {
            var results = [];
            if (this.root.bounds != null && !predicate(this.root.bounds)) {
                return results;
            }
            var candidates = [this.root];
            while (candidates.length > 0) {
                var candidate = candidates.shift();
                for (var i = 0; i < candidate.entries.length; i++) {
                    var entry = candidate.entries[i];
                    if (predicate(entry.bounds)) {
                        if (candidate.leaf) {
                            results.push(entry.value);
                        }
                        else {
                            candidates.push(entry);
                        }
                    }
                }
            }
            return results;
        };
        RTree.prototype.queryNodes = function (predicate) {
            var results = [];
            if (this.root.bounds != null && predicate(this.root) === QueryPredicateResult.FAIL) {
                return results;
            }
            var candidates = [this.root];
            while (candidates.length > 0) {
                var candidate = candidates.shift();
                for (var i = 0; i < candidate.entries.length; i++) {
                    var entry = candidate.entries[i];
                    var p = predicate(entry);
                    if (p === QueryPredicateResult.PASS_AND_OVERWRITE) {
                        results = [];
                    }
                    if (p === QueryPredicateResult.PASS || p === QueryPredicateResult.PASS_AND_OVERWRITE) {
                        if (candidate.leaf) {
                            results.push(entry);
                        }
                        else {
                            candidates.push(entry);
                        }
                    }
                }
            }
            return results;
        };
        return RTree;
    }());
    exports.RTree = RTree;
    var RTreeNode = /** @class */ (function () {
        function RTreeNode(leaf) {
            this.leaf = leaf;
            this.bounds = null;
            this.entries = [];
            this.parent = null;
            this.value = null;
        }
        RTreeNode.valueNode = function (bounds, value) {
            var node = new RTreeNode(true);
            node.bounds = bounds;
            node.value = value;
            return node;
        };
        /**
         * Returns `true` iff this node has more children than the `maxNodeChildren`
         * parameter.
         */
        RTreeNode.prototype.overflow = function (maxNodeChildren) {
            return this.entries.length > maxNodeChildren;
        };
        /**
         * Inserts a child node and updates the ancestry bounds.
         */
        RTreeNode.prototype.insert = function (node) {
            this.entries.push(node);
            node.parent = this;
            // Update ancestor bounds
            var ancestor = this;
            while (ancestor != null) {
                ancestor.bounds = RTreeBounds.unionAll([ancestor.bounds, node.bounds]);
                ancestor = ancestor.parent;
            }
            return this;
        };
        /**
         * Removes a child node and updates the ancestry bounds.
         *
         * If the node argument is not a child, do nothing.
         */
        RTreeNode.prototype.remove = function (node) {
            var i = this.entries.indexOf(node);
            if (i >= 0) {
                this.entries.splice(i, 1);
                // Update ancestor bounds
                var ancestor = this;
                while (ancestor != null) {
                    ancestor.bounds = RTreeBounds.unionAll(ancestor.entries.map(function (e) { return e.bounds; }));
                    ancestor = ancestor.parent;
                }
            }
            return this;
        };
        /**
         * Chooses an node from then entries that minimizes the area difference that
         * adding the bounds the each entry would cause.
         */
        RTreeNode.prototype.subtree = function (bounds) {
            var minDiff = Infinity;
            var minEntry = null;
            // choose entry for which the addition least increases the entry's area
            for (var i = 0; i < this.entries.length; i++) {
                var entry = this.entries[i];
                var diffArea = entry.unionAreaDifference(bounds);
                if (diffArea < minDiff || (
                // break ties to node with fewest children
                diffArea === minDiff &&
                    minEntry != null &&
                    entry.entries.length < minEntry.entries.length)) {
                    minEntry = entry;
                }
            }
            return minEntry;
        };
        /**
         * Splits this node by creating two new nodes and dividing the this node's
         * children between them. This node is removed from its parent and the two
         * new nodes are added.
         *
         * If this node is the root, a new parent node is created.
         *
         * Returns the parent node.
         */
        RTreeNode.prototype.split = function (strategy) {
            // Remove self from parent.
            if (this.parent != null) {
                this.parent.remove(this);
            }
            // Create children from split
            var children = [
                new RTreeNode(this.leaf),
                new RTreeNode(this.leaf),
            ];
            strategy.split(this.entries, children);
            // Add new nodes to parent
            // If root, create new non-leaf node as parent.
            var parent = this.parent != null ? this.parent : new RTreeNode(false);
            parent.insert(children[0]);
            parent.insert(children[1]);
            // Always make the parent a non-leaf after split
            parent.leaf = false;
            return parent;
        };
        /**
         * Returns the difference in area that adding an entry `bounds` to the node
         * would cause.
         */
        RTreeNode.prototype.unionAreaDifference = function (bounds) {
            return Math.abs(RTreeBounds.union(this.bounds, bounds).area() - this.bounds.area());
        };
        /**
         * Returns the depth from this node to the deepest leaf descendant.
         */
        RTreeNode.prototype.maxDepth = function () {
            if (this.leaf)
                return 1;
            return 1 + this.entries.map(function (e) { return e.maxDepth(); }).reduce(function (a, b) { return Math.max(a, b); });
        };
        return RTreeNode;
    }());
    exports.RTreeNode = RTreeNode;
    var RTreeBounds = /** @class */ (function () {
        function RTreeBounds(xl, yl, xh, yh) {
            this.xl = xl;
            this.yl = yl;
            this.xh = xh;
            this.yh = yh;
            this.width = this.xh - this.xl;
            this.height = this.yh - this.yl;
        }
        RTreeBounds.xywh = function (x, y, w, h) {
            return new RTreeBounds(x, y, x + w, y + h);
        };
        RTreeBounds.entityBounds = function (bounds) {
            return new RTreeBounds(bounds.x, bounds.y, bounds.x + bounds.width, bounds.y + bounds.height);
        };
        RTreeBounds.bounds = function (bounds) {
            return RTreeBounds.pointPair(bounds.topLeft, bounds.bottomRight);
        };
        RTreeBounds.pointPair = function (p0, p1) {
            return new RTreeBounds(Math.min(p0.x, p1.x), Math.min(p0.y, p1.y), Math.max(p0.x, p1.x), Math.max(p0.y, p1.y));
        };
        RTreeBounds.points = function (points) {
            if (points.length < 2) {
                throw new Error("need at least 2 points to create bounds");
            }
            var xs = points.map(function (p) { return p.x; });
            var ys = points.map(function (p) { return p.y; });
            return new RTreeBounds(xs.reduce(function (a, b) { return Math.min(a, b); }), ys.reduce(function (a, b) { return Math.min(a, b); }), xs.reduce(function (a, b) { return Math.max(a, b); }), ys.reduce(function (a, b) { return Math.max(a, b); }));
        };
        RTreeBounds.union = function (b0, b1) {
            return new RTreeBounds(Math.min(b0.xl, b1.xl), Math.min(b0.yl, b1.yl), Math.max(b0.xh, b1.xh), Math.max(b0.yh, b1.yh));
        };
        RTreeBounds.unionAll = function (bounds) {
            bounds = bounds.filter(function (b) { return b != null; });
            if (bounds.length === 0) {
                return null;
            }
            return bounds.reduce(function (b0, b1) { return RTreeBounds.union(b0, b1); });
        };
        /**
         * Returns true if `a` overlaps `b` in the x and y axes.
         *
         * Touching counts as overlap.
         */
        RTreeBounds.isBoundsOverlapBounds = function (a, b) {
            return RTreeBounds.isBoundsOverlapX(a, b) && RTreeBounds.isBoundsOverlapY(a, b);
        };
        /**
         * Returns true if `a` overlaps `b` in the x axis only.
         *
         * Touching counts as overlap.
         */
        RTreeBounds.isBoundsOverlapX = function (a, b) {
            return !(a.xh < b.xl) && !(a.xl > b.xh);
        };
        /**
         * Returns true if `a` overlaps `b` in the y axis only.
         *
         * Touching counts as overlap.
         */
        RTreeBounds.isBoundsOverlapY = function (a, b) {
            return !(a.yh < b.yl) && !(a.yl > b.yh);
        };
        /**
         * Returns the orthogonal absolute distance in the x-dimension from point
         * `p` to the nearest edge of `bounds`.
         *
         * If `p.x` is inside the bounds returns `0`.
         */
        RTreeBounds.absoluteDistanceToNearEdgeX = function (bounds, p) {
            var half = bounds.width / 2;
            var mid = bounds.xl + half;
            return Math.max(Math.abs(p.x - mid) - half, 0);
        };
        /**
         * Returns the orthogonal absolute distance in the y-dimension from point
         * `p` to the nearest edge of `bounds`.
         *
         * If `p.y` is inside the bounds returns `0`.
         */
        RTreeBounds.absoluteDistanceToNearEdgeY = function (bounds, p) {
            var half = bounds.height / 2;
            var mid = bounds.yl + half;
            return Math.max(Math.abs(p.y - mid) - half, 0);
        };
        /**
         * Returns the orthogonal absolute distance in the x-dimension from point
         * `p` to the farthest edge of `bounds`.
         *
         * If `p.x` is inside the bounds returns `0`.
         */
        RTreeBounds.absoluteDistanceToFarEdgeX = function (bounds, p) {
            var near = RTreeBounds.absoluteDistanceToNearEdgeX(bounds, p);
            return near === 0 ? 0 : near + bounds.width;
        };
        /**
         * Returns the orthogonal absolute distance in the y-dimension from point
         * `p` to the farthest edge of `bounds`.
         *
         * If `p.y` is inside the bounds returns `0`.
         */
        RTreeBounds.absoluteDistanceToFarEdgeY = function (bounds, p) {
            var near = RTreeBounds.absoluteDistanceToNearEdgeY(bounds, p);
            return near === 0 ? 0 : near + bounds.height;
        };
        /**
         * Returns the distance squared from `p` to the nearest edge of `bounds`. If
         * the point touches or is inside the bounds, returns `0`;
         *
         * https://gamedev.stackexchange.com/questions/44483/how-do-i-calculate-distance-between-a-point-and-an-axis-aligned-rectangle
         */
        RTreeBounds.distanceSquaredToNearEdge = function (bounds, p) {
            var dx = RTreeBounds.absoluteDistanceToNearEdgeX(bounds, p);
            var dy = RTreeBounds.absoluteDistanceToNearEdgeY(bounds, p);
            return dx * dx + dy * dy;
        };
        RTreeBounds.distanceSquaredToFarEdge = function (bounds, p) {
            var dx = RTreeBounds.absoluteDistanceToFarEdgeX(bounds, p);
            var dy = RTreeBounds.absoluteDistanceToFarEdgeY(bounds, p);
            return dx * dx + dy * dy;
        };
        RTreeBounds.prototype.area = function () {
            if (this.areaCached == null) {
                this.areaCached = (this.xh - this.xl) * (this.yh - this.yl);
            }
            return this.areaCached;
        };
        RTreeBounds.prototype.contains = function (xy) {
            return this.xl <= xy.x && this.xh >= xy.x && this.yl <= xy.y && this.yh >= xy.y;
        };
        return RTreeBounds;
    }());
    exports.RTreeBounds = RTreeBounds;
    });

    var stackingUtils = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });




    /**
     * Option type for stacking direction. By default, stacked bar and area charts
     * put the first data series at the bottom of the axis ("bottomup"), but this
     * can be reversed with the "topdown" option, which produces a stacking order
     * that matches the order of series in the legend.
     */
    exports.IStackingOrder = makeEnum_1.makeEnum(["topdown", "bottomup"]);
    var nativeMath = window.Math;
    /**
     * Computes the StackingResult (value and offset) for each data point in each Dataset.
     *
     * @param {Dataset[]} datasets The Datasets to be stacked on top of each other in the order of stacking
     * @param {Accessor<any>} keyAccessor Accessor for the key of the data
     * @param {Accessor<number>} valueAccessor Accessor for the value of the data
     * @param {IStackingOrder} stackingOrder The order of stacking (default "bottomup")
     * @return {StackingResult} value and offset for each datapoint in each Dataset
     */
    function stack(datasets, keyAccessor, valueAccessor, stackingOrder) {
        if (stackingOrder === void 0) { stackingOrder = "bottomup"; }
        var positiveOffsets = d3.map();
        var negativeOffsets = d3.map();
        var datasetToKeyToStackedDatum = new utils$1.Map();
        if (stackingOrder === "topdown") {
            datasets = datasets.slice();
            datasets.reverse();
        }
        for (var _i = 0, datasets_1 = datasets; _i < datasets_1.length; _i++) {
            var dataset = datasets_1[_i];
            var keyToStackedDatum = new utils$1.Map();
            var data = dataset.data();
            var dataLen = data.length;
            for (var index = 0; index < dataLen; index++) {
                var datum = data[index];
                var accessedKey = keyAccessor(datum, index, dataset);
                var key = exports.normalizeKey(accessedKey);
                var value = +valueAccessor(datum, index, dataset);
                var offset = void 0;
                var offsetMap = (value >= 0) ? positiveOffsets : negativeOffsets;
                if (offsetMap.has(key)) {
                    offset = offsetMap.get(key);
                    offsetMap.set(key, offset + value);
                }
                else {
                    offset = 0;
                    offsetMap.set(key, value);
                }
                keyToStackedDatum.set(key, {
                    offset: offset,
                    value: value,
                    axisValue: accessedKey,
                    originalDatum: datum,
                    originalDataset: dataset,
                    originalIndex: index,
                });
            }
            datasetToKeyToStackedDatum.set(dataset, keyToStackedDatum);
        }
        return datasetToKeyToStackedDatum;
    }
    exports.stack = stack;
    /**
     * Computes the maximum and minimum extents of each stack individually.
     *
     * @param {GenericStackingResult} stackingResult The value and offset information for each datapoint in each dataset
     * @return { { maximumExtents: Utils.Map<D, number>, minimumExtents: Utils.Map<D, number> } } The maximum and minimum extents
     * of each individual stack.
     */
    function stackedExtents(stackingResult) {
        var maximumExtents = new utils$1.Map();
        var minimumExtents = new utils$1.Map();
        stackingResult.forEach(function (stack) {
            stack.forEach(function (datum, key) {
                // correctly handle negative bar stacks
                var offsetValue = datum.offset + datum.value;
                var maximalValue = utils$1.Math.max([offsetValue, datum.offset], datum.offset);
                var minimalValue = utils$1.Math.min([offsetValue, datum.offset], datum.offset);
                var axisValue = datum.axisValue;
                if (!maximumExtents.has(key)) {
                    maximumExtents.set(key, { extent: maximalValue, axisValue: axisValue, stackedDatum: datum });
                }
                else if (maximumExtents.get(key).extent < maximalValue) {
                    maximumExtents.set(key, { extent: maximalValue, axisValue: axisValue, stackedDatum: datum });
                }
                if (!minimumExtents.has(key)) {
                    minimumExtents.set(key, { extent: minimalValue, axisValue: axisValue, stackedDatum: datum });
                }
                else if (minimumExtents.get(key).extent > (minimalValue)) {
                    minimumExtents.set(key, { extent: minimalValue, axisValue: axisValue, stackedDatum: datum });
                }
            });
        });
        return { maximumExtents: maximumExtents, minimumExtents: minimumExtents };
    }
    exports.stackedExtents = stackedExtents;
    /**
     * Computes the total extent over all data points in all Datasets, taking stacking into consideration.
     *
     * @param {StackingResult} stackingResult The value and offset information for each datapoint in each dataset
     * @param {Accessor<any>} keyAccessor Accessor for the key of the data existent in the stackingResult
     * @param {Accessor<boolean>} filter A filter for data to be considered when computing the total extent
     * @return {[number, number]} The total extent
     */
    function stackedExtent(stackingResult, keyAccessor, filter) {
        var extents = [];
        stackingResult.forEach(function (stackedDatumMap, dataset) {
            var data = dataset.data();
            var dataLen = data.length;
            for (var index = 0; index < dataLen; index++) {
                var datum = data[index];
                if (filter != null && !filter(datum, index, dataset)) {
                    continue;
                }
                var stackedDatum = stackedDatumMap.get(exports.normalizeKey(keyAccessor(datum, index, dataset)));
                extents.push(stackedDatum.value + stackedDatum.offset);
            }
        });
        var maxStackExtent = utils$1.Math.max(extents, 0);
        var minStackExtent = utils$1.Math.min(extents, 0);
        return [nativeMath.min(minStackExtent, 0), nativeMath.max(0, maxStackExtent)];
    }
    exports.stackedExtent = stackedExtent;
    /**
     * Normalizes a key used for stacking
     *
     * @param {any} key The key to be normalized
     * @return {string} The stringified key
     */
    exports.normalizeKey = lodash.memoize(function (key) {
        return String(key);
    });
    });

    var config$1 = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });
    /**
     * Specifies if Plottable should show warnings.
     */
    exports.SHOW_WARNINGS = true;
    /**
     * Specifies if Plottable should add <title> elements to text.
     */
    exports.ADD_TITLE_ELEMENTS = true;
    });

    var windowUtils = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });

    /**
     * Print a warning message to the console, if it is available.
     *
     * @param {string} The warnings to print
     */
    function warn(warning) {
        if (!config$1.SHOW_WARNINGS) {
            return;
        }
        // tslint:disable-next-line:no-console
        console.warn(warning);
    }
    exports.warn = warn;
    /**
     * Is like setTimeout, but activates synchronously if time=0
     * We special case 0 because of an observed issue where calling setTimeout causes visible flickering.
     * We believe this is because when requestAnimationFrame calls into the paint function, as soon as that function finishes
     * evaluating, the results are painted to the screen. As a result, if we want something to occur immediately but call setTimeout
     * with time=0, then it is pushed to the call stack and rendered in the next frame, so the component that was rendered via
     * setTimeout appears out-of-sync with the rest of the plot.
     */
    function setTimeout(f, time) {
        var args = [];
        for (var _i = 2; _i < arguments.length; _i++) {
            args[_i - 2] = arguments[_i];
        }
        if (time === 0) {
            f(args);
            return -1;
        }
        else {
            return window.setTimeout(f, time, args);
        }
    }
    exports.setTimeout = setTimeout;
    /**
     * Debounces the supplied callback and returns a function with the same
     * arguments.
     *
     * The callback is schedule for invocation every time the returned function is
     * invoked. If the returned function is called within the debounce time, the
     * previously scheduled call is canceled and the callback is schedule again.
     *
     * If debounced, the callback will be called with the most recent arguments.
     *
     * @param {number} msec - the debounce time in milliseconds
     * @param {T} callback - the callback invoked after the debounce time
     * @param {any} context  - the `this` argument used to invoke the callback
     */
    function debounce(msec, callback, context) {
        var timeoutToken = null;
        var args = [];
        var deferredCallback = function () {
            callback.apply(context, args);
        };
        // coerce to T
        return function () {
            args = Array.prototype.slice.call(arguments);
            clearTimeout(timeoutToken);
            timeoutToken = setTimeout(deferredCallback, msec);
        };
    }
    exports.debounce = debounce;
    /**
     * Sends a deprecation warning to the console. The warning includes the name of the deprecated method,
     * version number of the deprecation, and an optional message.
     *
     * To be used in the first line of a deprecated method.
     *
     * @param {string} callingMethod The name of the method being deprecated
     * @param {string} version The version when the tagged method became obsolete
     * @param {string?} message Optional message to be shown with the warning
     */
    function deprecated(callingMethod, version, message) {
        if (message === void 0) { message = ""; }
        warn("Method " + callingMethod + " has been deprecated in version " + version +
            ". Please refer to the release notes. " + message);
    }
    exports.deprecated = deprecated;
    });

    var bucket = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2017-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });
    /**
     * This class keeps track of bucketing state while collapsing dense line
     * geometry in a line and area plots.
     */
    var Bucket = /** @class */ (function () {
        function Bucket(index, xValue, yValue) {
            this.entryIndex = index;
            this.exitIndex = index;
            this.minIndex = index;
            this.maxIndex = index;
            this.bucketValue = xValue;
            this.minValue = yValue;
            this.maxValue = yValue;
        }
        Bucket.prototype.isInBucket = function (value) {
            return value == this.bucketValue;
        };
        Bucket.prototype.addToBucket = function (value, index) {
            if (value < this.minValue) {
                this.minValue = value;
                this.minIndex = index;
            }
            if (value > this.maxValue) {
                this.maxValue = value;
                this.maxIndex = index;
            }
            this.exitIndex = index;
        };
        Bucket.prototype.getUniqueIndices = function () {
            var idxs = [this.entryIndex, this.maxIndex, this.minIndex, this.exitIndex];
            return idxs.filter(function (idx, i) { return i == 0 || idx != idxs[i - 1]; });
        };
        return Bucket;
    }());
    exports.Bucket = Bucket;
    });

    var set$g = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });
    /**
     * Shim for ES6 set.
     * https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Set
     */
    var Set = /** @class */ (function () {
        function Set() {
            if (typeof window.Set === "function") {
                this._es6Set = new window.Set();
            }
            else {
                this._values = [];
            }
            this.size = 0;
        }
        Set.prototype.add = function (value) {
            if (this._es6Set != null) {
                this._es6Set.add(value);
                this.size = this._es6Set.size;
                return this;
            }
            if (!this.has(value)) {
                this._values.push(value);
                this.size = this._values.length;
            }
            return this;
        };
        Set.prototype.delete = function (value) {
            if (this._es6Set != null) {
                var deleted = this._es6Set.delete(value);
                this.size = this._es6Set.size;
                return deleted;
            }
            var index = this._values.indexOf(value);
            if (index !== -1) {
                this._values.splice(index, 1);
                this.size = this._values.length;
                return true;
            }
            return false;
        };
        Set.prototype.has = function (value) {
            if (this._es6Set != null) {
                return this._es6Set.has(value);
            }
            return this._values.indexOf(value) !== -1;
        };
        Set.prototype.forEach = function (callback, thisArg) {
            var _this = this;
            if (this._es6Set != null) {
                var callbackWrapper = function (value, value2) { return callback.call(thisArg, value, value2, _this); };
                this._es6Set.forEach(callbackWrapper, thisArg);
                return;
            }
            this._values.forEach(function (value) {
                callback.call(thisArg, value, value, _this);
            });
        };
        return Set;
    }());
    exports.Set = Set;
    });

    var callbackSet = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });


    /**
     * A set of callbacks which can be all invoked at once.
     * Each callback exists at most once in the set (based on reference equality).
     * All callbacks should have the same signature.
     */
    var CallbackSet = /** @class */ (function (_super) {
        tslib_es6.__extends(CallbackSet, _super);
        function CallbackSet() {
            return _super !== null && _super.apply(this, arguments) || this;
        }
        CallbackSet.prototype.callCallbacks = function () {
            var _this = this;
            var args = [];
            for (var _i = 0; _i < arguments.length; _i++) {
                args[_i] = arguments[_i];
            }
            this.forEach(function (callback) {
                callback.apply(_this, args);
            });
            return this;
        };
        return CallbackSet;
    }(set$g.Set));
    exports.CallbackSet = CallbackSet;
    });

    var entityStore = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });

    /**
     * Implementation of {IEntityStore} that uses an array for easy iteration as
     * well as a quad tree for fast nearest-point queries.
     *
     * Note that if the position of your entities changes, you MUST rebuild the
     * entity store for the `entityNearest` method to work since the quadtree does
     * not know that its nodes have moved.
     */
    var EntityStore = /** @class */ (function () {
        function EntityStore() {
            this._entities = [];
            this._rtree = new rTree.RTree();
        }
        EntityStore.prototype.addAll = function (entities, entityBoundsFactory, bounds) {
            this._entities = this._entities.concat(entities);
            // filter out of bounds entities if bounds is defined
            if (bounds !== undefined) {
                var filterBounds = rTree.RTreeBounds.bounds(bounds);
                for (var i = 0; i < entities.length; i++) {
                    var entity = entities[i];
                    var entityBounds = rTree.RTreeBounds.entityBounds(entityBoundsFactory(entity));
                    if (rTree.RTreeBounds.isBoundsOverlapBounds(filterBounds, entityBounds)) {
                        this._rtree.insert(entityBounds, entity);
                    }
                }
            }
            else {
                for (var i = 0; i < entities.length; i++) {
                    var entity = entities[i];
                    var entityBounds = rTree.RTreeBounds.entityBounds(entityBoundsFactory(entity));
                    this._rtree.insert(entityBounds, entity);
                }
            }
        };
        EntityStore.prototype.entityNearest = function (queryPoint) {
            return this._rtree.locateNearest(queryPoint).pop();
        };
        EntityStore.prototype.entityNearestX = function (queryPoint) {
            return this._rtree.locateNearestX(queryPoint).pop();
        };
        EntityStore.prototype.entityNearestY = function (queryPoint) {
            return this._rtree.locateNearestY(queryPoint).pop();
        };
        EntityStore.prototype.entitiesInBounds = function (bounds) {
            return this._rtree.intersect(rTree.RTreeBounds.entityBounds(bounds));
        };
        EntityStore.prototype.entitiesInXBounds = function (bounds) {
            return this._rtree.intersectX(rTree.RTreeBounds.entityBounds(bounds));
        };
        EntityStore.prototype.entitiesInYBounds = function (bounds) {
            return this._rtree.intersectY(rTree.RTreeBounds.entityBounds(bounds));
        };
        EntityStore.prototype.entities = function () {
            return this._entities;
        };
        return EntityStore;
    }());
    exports.EntityStore = EntityStore;
    });

    var map$a = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });

    /**
     * Shim for ES6 map.
     * https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map
     */
    var Map = /** @class */ (function () {
        function Map() {
            if (typeof window.Map === "function") {
                this._es6Map = new window.Map();
            }
            else {
                this._keyValuePairs = [];
            }
        }
        Map.prototype.set = function (key, value) {
            if (mathUtils.isNaN(key)) {
                throw new Error("NaN may not be used as a key to the Map");
            }
            if (this._es6Map != null) {
                this._es6Map.set(key, value);
                return this;
            }
            for (var i = 0; i < this._keyValuePairs.length; i++) {
                if (this._keyValuePairs[i].key === key) {
                    this._keyValuePairs[i].value = value;
                    return this;
                }
            }
            this._keyValuePairs.push({ key: key, value: value });
            return this;
        };
        Map.prototype.get = function (key) {
            if (this._es6Map != null) {
                return this._es6Map.get(key);
            }
            for (var i = 0; i < this._keyValuePairs.length; i++) {
                if (this._keyValuePairs[i].key === key) {
                    return this._keyValuePairs[i].value;
                }
            }
            return undefined;
        };
        Map.prototype.has = function (key) {
            if (this._es6Map != null) {
                return this._es6Map.has(key);
            }
            for (var i = 0; i < this._keyValuePairs.length; i++) {
                if (this._keyValuePairs[i].key === key) {
                    return true;
                }
            }
            return false;
        };
        Map.prototype.forEach = function (callbackFn, thisArg) {
            var _this = this;
            if (this._es6Map != null) {
                var callbackWrapper = function (value, key) { return callbackFn.call(thisArg, value, key, _this); };
                this._es6Map.forEach(callbackWrapper, thisArg);
                return;
            }
            this._keyValuePairs.forEach(function (keyValuePair) {
                callbackFn.call(thisArg, keyValuePair.value, keyValuePair.key, _this);
            });
        };
        Map.prototype.delete = function (key) {
            if (this._es6Map != null) {
                return this._es6Map.delete(key);
            }
            for (var i = 0; i < this._keyValuePairs.length; i++) {
                if (this._keyValuePairs[i].key === key) {
                    this._keyValuePairs.splice(i, 1);
                    return true;
                }
            }
            return false;
        };
        return Map;
    }());
    exports.Map = Map;
    });

    var objectUtils = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2017-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });
    /**
     * Polyfill for Object.assign
     */
    function assign() {
        var objs = [];
        for (var _i = 0; _i < arguments.length; _i++) {
            objs[_i] = arguments[_i];
        }
        var result = {};
        for (var _a = 0, objs_1 = objs; _a < objs_1.length; _a++) {
            var obj = objs_1[_a];
            var keys = Object.keys(obj);
            for (var _b = 0, keys_1 = keys; _b < keys_1.length; _b++) {
                var key = keys_1[_b];
                result[key] = obj[key];
            }
        }
        return result;
    }
    exports.assign = assign;
    });

    var transformAwareTranslator = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2017-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });

    var _TRANSLATOR_KEY = "__Plottable_ClientTranslator";
    /**
     * Returns a singleton-ized `Translator` instance associated with the component.
     */
    function getTranslator(component) {
        var rootElement = component.root().rootElement().node();
        var translator = rootElement[_TRANSLATOR_KEY];
        if (translator == null) {
            translator = new Translator(rootElement);
            rootElement[_TRANSLATOR_KEY] = translator;
        }
        return translator;
    }
    exports.getTranslator = getTranslator;
    /**
     * The translator implements CSS transform aware position measuring. We manually
     * compute a cumulative CSS3 of the root element ancestors up to `<body>`.
     */
    var Translator = /** @class */ (function () {
        function Translator(_rootElement) {
            this._rootElement = _rootElement;
        }
        /**
         * Given `document` client coordinates, computes the position relative to the
         * `Component`'s root element, taking into account the cumulative CSS3
         * transforms of the root element ancestors up to `<body>`.
         *
         * This triggers a layout but doesn't further modify the DOM, so causes a
         * maximum of one layout per frame.
         *
         * Does not support `transform-origin` CSS property other than the default.
         */
        Translator.prototype.computePosition = function (clientX, clientY) {
            var clientPosition = {
                x: clientX,
                y: clientY,
            };
            var transform = utils$1.Math.getCumulativeTransform(this._rootElement);
            if (transform == null) {
                return clientPosition;
            }
            var transformed = utils$1.Math.applyTransform(transform, clientPosition);
            return transformed;
        };
        /**
         * Is the event's target part of the given component's DOM tree?
         */
        Translator.isEventInside = function (component, e) {
            return utils$1.DOM.contains(component.root().rootElement().node(), e.target);
        };
        return Translator;
    }());
    exports.Translator = Translator;
    });

    var utils$1 = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });


    exports.Array = arrayUtils;

    exports.Color = colorUtils;

    exports.DOM = domUtils;

    exports.Math = mathUtils;

    exports.RTree = rTree;

    exports.Stacking = stackingUtils;

    exports.Window = windowUtils;
    tslib_es6.__exportStar(bucket, exports);
    tslib_es6.__exportStar(callbackSet, exports);
    tslib_es6.__exportStar(coerceD3, exports);
    tslib_es6.__exportStar(entityStore, exports);
    tslib_es6.__exportStar(map$a, exports);
    tslib_es6.__exportStar(objectUtils, exports);
    tslib_es6.__exportStar(set$g, exports);
    tslib_es6.__exportStar(transformAwareTranslator, exports);
    });

    var renderPolicy = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });


    /**
     * Renders Components immediately after they are enqueued.
     * Useful for debugging, horrible for performance.
     */
    var Immediate = /** @class */ (function () {
        function Immediate() {
        }
        Immediate.prototype.render = function () {
            renderController.flush();
        };
        return Immediate;
    }());
    exports.Immediate = Immediate;
    /**
     * The default way to render, which only tries to render every frame
     * (usually, 1/60th of a second).
     */
    var AnimationFrame = /** @class */ (function () {
        function AnimationFrame() {
        }
        AnimationFrame.prototype.render = function () {
            utils$1.DOM.requestAnimationFramePolyfill(renderController.flush);
        };
        return AnimationFrame;
    }());
    exports.AnimationFrame = AnimationFrame;
    /**
     * Renders with `setTimeout()`.
     * Generally an inferior way to render compared to `requestAnimationFrame`,
     * but useful for browsers that don't suppoort `requestAnimationFrame`.
     */
    var Timeout = /** @class */ (function () {
        function Timeout() {
            this._timeoutMsec = utils$1.DOM.SCREEN_REFRESH_RATE_MILLISECONDS;
        }
        Timeout.prototype.render = function () {
            setTimeout(renderController.flush, this._timeoutMsec);
        };
        return Timeout;
    }());
    exports.Timeout = Timeout;
    });

    var renderController = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });



    /**
     * The RenderController is responsible for enqueueing and synchronizing
     * layout and render calls for Components.
     *
     * Layout and render calls occur inside an animation callback
     * (window.requestAnimationFrame if available).
     *
     * RenderController.flush() immediately lays out and renders all Components currently enqueued.
     *
     * To always have immediate rendering (useful for debugging), call
     * ```typescript
     * Plottable.RenderController.setRenderPolicy(
     *   new Plottable.RenderPolicies.Immediate()
     * );
     * ```
     */
    var _componentsNeedingRender = new utils$1.Set();
    var _componentsNeedingComputeLayout = new utils$1.Set();
    var _animationRequested = false;
    var _isCurrentlyFlushing = false;
    exports.Policy = makeEnum_1.makeEnum(["immediate", "animationFrame", "timeout"]);
    var _renderPolicy = new renderPolicy.AnimationFrame();
    function renderPolicy$1(renderPolicy$1) {
        if (renderPolicy$1 == null) {
            return _renderPolicy;
        }
        switch (renderPolicy$1) {
            case exports.Policy.immediate:
                _renderPolicy = new renderPolicy.Immediate();
                break;
            case exports.Policy.animationFrame:
                _renderPolicy = new renderPolicy.AnimationFrame();
                break;
            case exports.Policy.timeout:
                _renderPolicy = new renderPolicy.Timeout();
                break;
            default:
                utils$1.Window.warn("Unrecognized renderPolicy: " + renderPolicy$1);
        }
    }
    exports.renderPolicy = renderPolicy$1;
    /**
     * Enqueues the Component for rendering.
     *
     * @param {Component} component
     */
    function registerToRender(component) {
        if (_isCurrentlyFlushing) {
            utils$1.Window.warn("Registered to render while other components are flushing: request may be ignored");
        }
        _componentsNeedingRender.add(component);
        requestRender();
    }
    exports.registerToRender = registerToRender;
    /**
     * Enqueues the Component for layout and rendering.
     *
     * @param {Component} component
     */
    function registerToComputeLayoutAndRender(component) {
        _componentsNeedingComputeLayout.add(component);
        _componentsNeedingRender.add(component);
        requestRender();
    }
    exports.registerToComputeLayoutAndRender = registerToComputeLayoutAndRender;
    /**
     * Enqueues the Component for layout and rendering.
     *
     * @param {Component} component
     * @deprecated This method has been renamed to `RenderController.registerToComputeLayoutAndRender()`.
     */
    function registerToComputeLayout(component) {
        registerToComputeLayoutAndRender(component);
    }
    exports.registerToComputeLayout = registerToComputeLayout;
    function requestRender() {
        // Only run or enqueue flush on first request.
        if (!_animationRequested) {
            _animationRequested = true;
            _renderPolicy.render();
        }
    }
    /**
     * Renders all Components waiting to be rendered immediately
     * instead of waiting until the next frame. Flush is idempotent (given there are no intermediate registrations).
     *
     * Useful to call when debugging.
     */
    function flush() {
        if (_animationRequested) {
            // Layout
            _componentsNeedingComputeLayout.forEach(function (component) { return component.computeLayout(); });
            // Top level render; Containers will put their children in the toRender queue
            _componentsNeedingRender.forEach(function (component) { return component.render(); });
            _isCurrentlyFlushing = true;
            var failed_1 = new utils$1.Set();
            _componentsNeedingRender.forEach(function (component) {
                try {
                    component.renderImmediately();
                }
                catch (err) {
                    // throw error with timeout to avoid interrupting further renders
                    window.setTimeout(function () {
                        throw err;
                    }, 0);
                    failed_1.add(component);
                }
            });
            _componentsNeedingComputeLayout = new utils$1.Set();
            _componentsNeedingRender = failed_1;
            _animationRequested = false;
            _isCurrentlyFlushing = false;
        }
    }
    exports.flush = flush;
    });

    var component = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });






    exports.XAlignment = makeEnum_1.makeEnum(["left", "center", "right"]);
    exports.YAlignment = makeEnum_1.makeEnum(["top", "center", "bottom"]);
    /**
     * Components are the core logical units that build Plottable visualizations.
     *
     * This class deals with Component lifecycle (anchoring, getting a size, and rendering
     * infrastructure), as well as building the framework of DOM elements for all Components.
     */
    var Component = /** @class */ (function () {
        function Component() {
            /**
             * Subclasses should set this to true in their constructor to prevent content from overflowing.
             */
            this._overflowHidden = false;
            /**
             * Origin of this Component relative to its parent.
             */
            this._origin = { x: 0, y: 0 };
            this._xAlignment = "left";
            this._yAlignment = "top";
            this._isSetup = false;
            this._isAnchored = false;
            this._cssClasses = new utils$1.Set();
            /**
             * If .destroy() has been called on this Component.
             */
            this._destroyed = false;
            this._onAnchorCallbacks = new utils$1.CallbackSet();
            this._onDetachCallbacks = new utils$1.CallbackSet();
            this._cssClasses.add("component");
        }
        /**
         * Attaches the Component as a child of a given d3 Selection.
         *
         * @param {d3.Selection} selection.
         * @returns {Component} The calling Component.
         */
        Component.prototype.anchor = function (selection) {
            selection = coerceD3.coerceExternalD3(selection);
            if (this._destroyed) {
                throw new Error("Can't reuse destroy()-ed Components!");
            }
            if (this.isRoot()) {
                this._rootElement = selection;
                // rootElement gets the "plottable" CSS class
                this._rootElement.classed("plottable", true);
            }
            if (this._element != null) {
                // reattach existing element
                selection.node().appendChild(this._element.node());
            }
            else {
                this._element = selection.append("div");
                this._setup();
            }
            this._isAnchored = true;
            this._onAnchorCallbacks.callCallbacks(this);
            return this;
        };
        /**
         * Adds a callback to be called on anchoring the Component to the DOM.
         * If the Component is already anchored, the callback is called immediately.
         *
         * @param {ComponentCallback} callback
         * @return {Component}
         */
        Component.prototype.onAnchor = function (callback) {
            if (this._isAnchored) {
                callback(this);
            }
            this._onAnchorCallbacks.add(callback);
            return this;
        };
        /**
         * Removes a callback that would be called on anchoring the Component to the DOM.
         * The callback is identified by reference equality.
         *
         * @param {ComponentCallback} callback
         * @return {Component}
         */
        Component.prototype.offAnchor = function (callback) {
            this._onAnchorCallbacks.delete(callback);
            return this;
        };
        /**
         * Creates additional elements as necessary for the Component to function.
         * Called during anchor() if the Component's element has not been created yet.
         * Override in subclasses to provide additional functionality.
         */
        Component.prototype._setup = function () {
            var _this = this;
            if (this._isSetup) {
                return;
            }
            this._cssClasses.forEach(function (cssClass) {
                _this._element.classed(cssClass, true);
            });
            this._cssClasses = new utils$1.Set();
            this._backgroundContainer = this._element.append("svg").classed("background-container", true);
            this._content = this._element.append("svg").classed("content", true);
            this._foregroundContainer = this._element.append("svg").classed("foreground-container", true);
            if (this._overflowHidden) {
                this._content.classed("component-overflow-hidden", true);
            }
            else {
                this._content.classed("component-overflow-visible", true);
            }
            this._isSetup = true;
        };
        /**
         * Given available space in pixels, returns the minimum width and height this Component will need.
         *
         * @param {number} availableWidth
         * @param {number} availableHeight
         * @returns {SpaceRequest}
         */
        Component.prototype.requestedSpace = function (availableWidth, availableHeight) {
            return {
                minWidth: 0,
                minHeight: 0,
            };
        };
        /**
         * Computes and sets the size, position, and alignment of the Component from the specified values.
         * If no parameters are supplied and the Component is a root node,
         * they are inferred from the size of the Component's element.
         *
         * @param {Point} [origin] Origin of the space offered to the Component.
         * @param {number} [availableWidth] Available width in pixels.
         * @param {number} [availableHeight] Available height in pixels.
         * @returns {Component} The calling Component.
         */
        Component.prototype.computeLayout = function (origin, availableWidth, availableHeight) {
            if (origin == null || availableWidth == null || availableHeight == null) {
                if (this._element == null) {
                    throw new Error("anchor() must be called before computeLayout()");
                }
                else if (this._rootElement != null) {
                    // retrieve height/width from rootElement
                    origin = { x: 0, y: 0 };
                    var elem = this._rootElement.node();
                    availableWidth = utils$1.DOM.elementWidth(elem);
                    availableHeight = utils$1.DOM.elementHeight(elem);
                }
                else {
                    throw new Error("null arguments cannot be passed to computeLayout() on a non-root, unanchored node");
                }
            }
            var _a = this._sizeFromOffer(availableWidth, availableHeight), height = _a.height, width = _a.width;
            var xAlignProportion = Component._xAlignToProportion[this._xAlignment];
            var yAlignProportion = Component._yAlignToProportion[this._yAlignment];
            var originX = origin.x + (availableWidth - width) * xAlignProportion;
            var originY = origin.y + (availableHeight - height) * yAlignProportion;
            this.setBounds(width, height, originX, originY);
            return this;
        };
        /**
         * Directly sets component size and, optionally, its origin.
         *
         * Preferably, layout is accomplished by placing components in a table.
         * However, if you need to directly override the component size, you may call
         * this method.
         *
         * Note that this method styles the anchored element, so this is usually only
         * useful after the component has been anchored. If the component has not been
         * anchored to an element, the internal properties will be set but no styles
         * will be applied.
         *
         * @param {number} [width] width in pixels
         * @param {number} [height] height in pixels
         * @param {number} [originX] left offset in pixels
         * @param {number} [originY] top offset in pixels
         */
        Component.prototype.setBounds = function (width, height, originX, originY) {
            if (originX === void 0) { originX = 0; }
            if (originY === void 0) { originY = 0; }
            this._width = width;
            this._height = height;
            this._origin = {
                x: originX,
                y: originY,
            };
            if (this._element != null) {
                this._element.styles({
                    left: originX + "px",
                    height: height + "px",
                    top: originY + "px",
                    width: width + "px",
                });
            }
            if (this._resizeHandler != null) {
                this._resizeHandler({ width: width, height: height });
            }
            return this;
        };
        Component.prototype._sizeFromOffer = function (availableWidth, availableHeight) {
            var requestedSpace = this.requestedSpace(availableWidth, availableHeight);
            return {
                width: this.fixedWidth() ? Math.min(availableWidth, requestedSpace.minWidth) : availableWidth,
                height: this.fixedHeight() ? Math.min(availableHeight, requestedSpace.minHeight) : availableHeight,
            };
        };
        /**
         * Queues the Component for rendering.
         *
         * @returns {Component} The calling Component.
         */
        Component.prototype.render = function () {
            if (this._isAnchored && this._isSetup && this.width() >= 0 && this.height() >= 0) {
                renderController.registerToRender(this);
            }
            return this;
        };
        Component.prototype.renderLowPriority = function () {
            return this.render();
        };
        Component.prototype._scheduleComputeLayout = function () {
            if (this._isAnchored && this._isSetup) {
                renderController.registerToComputeLayoutAndRender(this);
            }
        };
        /**
         * Sets a callback that gets called when the component resizes. The size change
         * is not guaranteed to be reflected by the DOM at the time the callback is fired.
         *
         * @param {IResizeHandler} [resizeHandler] Callback to be called when component resizes
         */
        Component.prototype.onResize = function (resizeHandler) {
            this._resizeHandler = resizeHandler;
            return this;
        };
        /**
         * Renders the Component without waiting for the next frame. This method is a no-op on
         * Component, Table, and Group; render them immediately with .renderTo() instead.
         */
        Component.prototype.renderImmediately = function () {
            return this;
        };
        /**
         * Causes the Component to re-layout and render.
         *
         * @returns {Component} The calling Component.
         */
        Component.prototype.redraw = function () {
            if (this._isAnchored && this._isSetup) {
                if (this.isRoot()) {
                    this._scheduleComputeLayout();
                }
                else {
                    this.parent().redraw();
                }
            }
            return this;
        };
        /**
         * Tell this component to invalidate any caching. This function should be
         * called when a CSS change has occurred that could influence the layout
         * of the Component, such as changing the font size.
         *
         * Subclasses should override.
         */
        Component.prototype.invalidateCache = function () {
            // Core component has no caching.
        };
        /**
         * Renders the Component to a given HTML Element.
         *
         * @param {String|d3.Selection} element The element, a selector string for the element, or a d3.Selection for the element.
         * @returns {Component} The calling Component.
         */
        Component.prototype.renderTo = function (element) {
            this.detach();
            if (element != null) {
                var selection = void 0;
                if (typeof (element) === "string") {
                    selection = d3.select(element);
                }
                else if (lodash.isElement(element)) {
                    selection = d3.select(element);
                }
                else {
                    selection = coerceD3.coerceExternalD3(element);
                }
                if (!selection.node() || selection.node().nodeName == null) {
                    throw new Error("Plottable requires a valid Element to renderTo");
                }
                if (selection.node().nodeName === "svg") {
                    throw new Error("Plottable 3.x and later can only renderTo an HTML component; pass a div instead!");
                }
                this.anchor(selection);
            }
            if (this._element == null) {
                throw new Error("If a Component has never been rendered before, then renderTo must be given a node to render to, " +
                    "or a d3.Selection, or a selector string");
            }
            renderController.registerToComputeLayoutAndRender(this);
            // flush so that consumers can immediately attach to stuff we create in the DOM
            renderController.flush();
            return this;
        };
        Component.prototype.xAlignment = function (xAlignment) {
            if (xAlignment == null) {
                return this._xAlignment;
            }
            xAlignment = xAlignment.toLowerCase();
            if (Component._xAlignToProportion[xAlignment] == null) {
                throw new Error("Unsupported alignment: " + xAlignment);
            }
            this._xAlignment = xAlignment;
            this.redraw();
            return this;
        };
        Component.prototype.yAlignment = function (yAlignment) {
            if (yAlignment == null) {
                return this._yAlignment;
            }
            yAlignment = yAlignment.toLowerCase();
            if (Component._yAlignToProportion[yAlignment] == null) {
                throw new Error("Unsupported alignment: " + yAlignment);
            }
            this._yAlignment = yAlignment;
            this.redraw();
            return this;
        };
        /**
         * Checks if the Component has a given CSS class.
         *
         * @param {string} cssClass The CSS class to check for.
         */
        Component.prototype.hasClass = function (cssClass) {
            if (cssClass == null) {
                return false;
            }
            if (this._element == null) {
                return this._cssClasses.has(cssClass);
            }
            else {
                return this._element.classed(cssClass);
            }
        };
        /**
         * Adds a given CSS class to the Component.
         *
         * @param {string} cssClass The CSS class to add.
         * @returns {Component} The calling Component.
         */
        Component.prototype.addClass = function (cssClass) {
            if (cssClass == null) {
                return this;
            }
            if (this._element == null) {
                this._cssClasses.add(cssClass);
            }
            else {
                this._element.classed(cssClass, true);
            }
            return this;
        };
        /**
         * Removes a given CSS class from the Component.
         *
         * @param {string} cssClass The CSS class to remove.
         * @returns {Component} The calling Component.
         */
        Component.prototype.removeClass = function (cssClass) {
            if (cssClass == null) {
                return this;
            }
            if (this._element == null) {
                this._cssClasses.delete(cssClass);
            }
            else {
                this._element.classed(cssClass, false);
            }
            return this;
        };
        /**
         * Checks if the Component has a fixed width or if it grows to fill available space.
         * Returns false by default on the base Component class.
         */
        Component.prototype.fixedWidth = function () {
            return false;
        };
        /**
         * Checks if the Component has a fixed height or if it grows to fill available space.
         * Returns false by default on the base Component class.
         */
        Component.prototype.fixedHeight = function () {
            return false;
        };
        /**
         * Detaches a Component from the DOM. The Component can be reused.
         *
         * This should only be used if you plan on reusing the calling Component. Otherwise, use destroy().
         *
         * @returns The calling Component.
         */
        Component.prototype.detach = function () {
            this.parent(null);
            if (this._isAnchored) {
                this._element.remove();
            }
            this._isAnchored = false;
            this._onDetachCallbacks.callCallbacks(this);
            return this;
        };
        /**
         * Adds a callback to be called when the Component is detach()-ed.
         *
         * @param {ComponentCallback} callback
         * @return {Component} The calling Component.
         */
        Component.prototype.onDetach = function (callback) {
            this._onDetachCallbacks.add(callback);
            return this;
        };
        /**
         * Removes a callback to be called when the Component is detach()-ed.
         * The callback is identified by reference equality.
         *
         * @param {ComponentCallback} callback
         * @return {Component} The calling Component.
         */
        Component.prototype.offDetach = function (callback) {
            this._onDetachCallbacks.delete(callback);
            return this;
        };
        Component.prototype.parent = function (parent) {
            if (parent === undefined) {
                return this._parent;
            }
            if (parent !== null && !parent.has(this)) {
                throw new Error("Passed invalid parent");
            }
            this._parent = parent;
            return this;
        };
        /**
         * @returns {Bounds} for the component in pixel space, where the topLeft
         * represents the component's minimum x and y values and the bottomRight represents
         * the component's maximum x and y values.
         */
        Component.prototype.bounds = function () {
            var topLeft = this.origin();
            return {
                topLeft: topLeft,
                bottomRight: {
                    x: topLeft.x + this.width(),
                    y: topLeft.y + this.height(),
                },
            };
        };
        /**
         * Removes a Component from the DOM and disconnects all listeners.
         */
        Component.prototype.destroy = function () {
            this._destroyed = true;
            this.detach();
        };
        /**
         * Gets the width of the Component in pixels.
         */
        Component.prototype.width = function () {
            return this._width;
        };
        /**
         * Gets the height of the Component in pixels.
         */
        Component.prototype.height = function () {
            return this._height;
        };
        /**
         * Gets the origin of the Component relative to its parent.
         *
         * @return {Point}
         */
        Component.prototype.origin = function () {
            return {
                x: this._origin.x,
                y: this._origin.y,
            };
        };
        /**
         * Gets the origin of the Component relative to the root Component.
         *
         * @return {Point}
         */
        Component.prototype.originToRoot = function () {
            var origin = this.origin();
            var ancestor = this.parent();
            while (ancestor != null) {
                var ancestorOrigin = ancestor.origin();
                origin.x += ancestorOrigin.x;
                origin.y += ancestorOrigin.y;
                ancestor = ancestor.parent();
            }
            return origin;
        };
        /**
         * Gets the root component of the hierarchy. If the provided
         * component is the root, that component will be returned.
         */
        Component.prototype.root = function () {
            var component = this;
            while (!component.isRoot()) {
                component = component.parent();
            }
            return component;
        };
        Component.prototype.isRoot = function () {
            return this.parent() == null;
        };
        /**
         * Gets the Selection containing the <g> in front of the visual elements of the Component.
         *
         * Will return undefined if the Component has not been anchored.
         *
         * @return {d3.Selection}
         */
        Component.prototype.foreground = function () {
            return this._foregroundContainer;
        };
        /**
         * Gets the SVG that holds the visual elements of the Component.
         *
         * Will return undefined if the Component has not been anchored.
         *
         * @return {d3.Selection} content selection for the Component
         */
        Component.prototype.content = function () {
            return this._content;
        };
        /**
         * Returns the HTML Element at the root of this component's DOM tree.
         */
        Component.prototype.element = function () {
            return this._element;
        };
        /**
         * Returns the top-level user supplied element that roots the tree that this Component lives in.
         */
        Component.prototype.rootElement = function () {
            return this.root()._rootElement;
        };
        /**
         * Gets the Selection containing the <g> behind the visual elements of the Component.
         *
         * Will return undefined if the Component has not been anchored.
         *
         * @return {d3.Selection} background selection for the Component
         */
        Component.prototype.background = function () {
            return this._backgroundContainer;
        };
        Component._xAlignToProportion = {
            left: 0,
            center: 0.5,
            right: 1,
        };
        Component._yAlignToProportion = {
            top: 0,
            center: 0.5,
            bottom: 1,
        };
        return Component;
    }());
    exports.Component = Component;
    });

    var formatters = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });

    // Do not use utc by default
    var DEFAULT_USE_UTC = false;
    /**
     * Creates a formatter for currency values.
     *
     * @param {number} [precision] The number of decimal places to show (default 2).
     * @param {string} [symbol] The currency symbol to use (default "$").
     * @param {boolean} [prefix] Whether to prepend or append the currency symbol (default true).
     *
     * @returns {Formatter} A formatter for currency values.
     */
    function currency(precision, symbol, prefix) {
        if (precision === void 0) { precision = 2; }
        if (symbol === void 0) { symbol = "$"; }
        if (prefix === void 0) { prefix = true; }
        var fixedFormatter = fixed(precision);
        return function (d) {
            var formattedValue = fixedFormatter(Math.abs(d));
            if (formattedValue !== "") {
                if (prefix) {
                    formattedValue = symbol + formattedValue;
                }
                else {
                    formattedValue += symbol;
                }
                if (d < 0) {
                    formattedValue = "-" + formattedValue;
                }
            }
            return formattedValue;
        };
    }
    exports.currency = currency;
    /**
     * Creates a formatter that displays exactly [precision] decimal places.
     *
     * @param {number} [precision] The number of decimal places to show (default 3).
     *
     * @returns {Formatter} A formatter that displays exactly [precision] decimal places.
     */
    function fixed(precision) {
        if (precision === void 0) { precision = 3; }
        verifyPrecision(precision);
        return function (d) { return d.toFixed(precision); };
    }
    exports.fixed = fixed;
    /**
     * Creates a formatter that formats numbers to show no more than
     * [maxNumberOfDecimalPlaces] decimal places. All other values are stringified.
     *
     * @param {number} [maxNumberOfDecimalPlaces] The number of decimal places to show (default 3).
     *
     * @returns {Formatter} A formatter for general values.
     */
    function general(maxNumberOfDecimalPlaces) {
        if (maxNumberOfDecimalPlaces === void 0) { maxNumberOfDecimalPlaces = 3; }
        verifyPrecision(maxNumberOfDecimalPlaces);
        return function (d) {
            if (typeof d === "number") {
                var multiplier = Math.pow(10, maxNumberOfDecimalPlaces);
                return String(Math.round(d * multiplier) / multiplier);
            }
            else {
                return String(d);
            }
        };
    }
    exports.general = general;
    /**
     * Creates a formatter that stringifies its input.
     *
     * @returns {Formatter} A formatter that stringifies its input.
     */
    function identity() {
        return function (d) { return String(d); };
    }
    exports.identity = identity;
    /**
     * Creates a formatter for percentage values.
     * Multiplies the input by 100 and appends "%".
     *
     * @param {number} [precision] The number of decimal places to show (default 0).
     *
     * @returns {Formatter} A formatter for percentage values.
     */
    function percentage(precision) {
        if (precision === void 0) { precision = 0; }
        var fixedFormatter = fixed(precision);
        return function (d) {
            var valToFormat = d * 100;
            // Account for float imprecision
            var valString = d.toString();
            var integerPowerTen = Math.pow(10, valString.length - (valString.indexOf(".") + 1));
            valToFormat = parseInt((valToFormat * integerPowerTen).toString(), 10) / integerPowerTen;
            return fixedFormatter(valToFormat) + "%";
        };
    }
    exports.percentage = percentage;
    /**
     * Creates a formatter for values that displays [numberOfSignificantFigures] significant figures
     * and puts SI notation.
     *
     * @param {number} [numberOfSignificantFigures] The number of significant figures to show (default 3).
     *
     * @returns {Formatter} A formatter for SI values.
     */
    function siSuffix(numberOfSignificantFigures) {
        if (numberOfSignificantFigures === void 0) { numberOfSignificantFigures = 3; }
        verifyPrecision(numberOfSignificantFigures);
        return function (d) { return d3.format("." + numberOfSignificantFigures + "s")(d); };
    }
    exports.siSuffix = siSuffix;
    /**
     * Creates a formatter for values that displays abbreviated values
     * and uses standard short scale suffixes
     * - K - thousands - 10 ^ 3
     * - M - millions - 10 ^ 6
     * - B - billions - 10 ^ 9
     * - T - trillions - 10 ^ 12
     * - Q - quadrillions - 10 ^ 15
     *
     * Numbers with a magnitude outside of (10 ^ (-precision), 10 ^ 15) are shown using
     * scientific notation to avoid creating extremely long decimal strings.
     *
     * @param {number} [precision] the number of decimal places to show (default 3)
     * @returns {Formatter} A formatter with short scale formatting
     */
    function shortScale(precision) {
        if (precision === void 0) { precision = 3; }
        verifyPrecision(precision);
        var suffixes = "KMBTQ";
        var exponentFormatter = d3.format("." + precision + "e");
        var fixedFormatter = d3.format("." + precision + "f");
        var max = Math.pow(10, (3 * (suffixes.length + 1)));
        var min = Math.pow(10, -precision);
        return function (num) {
            var absNum = Math.abs(num);
            if ((absNum < min || absNum >= max) && absNum !== 0) {
                return exponentFormatter(num);
            }
            var idx = -1;
            while (absNum >= Math.pow(1000, idx + 2) && idx < (suffixes.length - 1)) {
                idx++;
            }
            var output = "";
            if (idx === -1) {
                output = fixedFormatter(num);
            }
            else {
                output = fixedFormatter(num / Math.pow(1000, idx + 1)) + suffixes[idx];
            }
            // catch rounding by the underlying d3 formatter
            if ((num > 0 && output.substr(0, 4) === "1000") || (num < 0 && output.substr(0, 5) === "-1000")) {
                if (idx < suffixes.length - 1) {
                    idx++;
                    output = fixedFormatter(num / Math.pow(1000, idx + 1)) + suffixes[idx];
                }
                else {
                    output = exponentFormatter(num);
                }
            }
            return output;
        };
    }
    exports.shortScale = shortScale;
    /**
     * Creates a multi time formatter that displays dates.
     *
     * @returns {Formatter} A formatter for time/date values.
     */
    function multiTime() {
        // Formatter tiers going from shortest time scale to largest - these were taken from d3
        // https://github.com/mbostock/d3/wiki/Time-Formatting#format_multi
        var candidateFormats = [
            {
                specifier: ".%L",
                predicate: function (d) { return d.getMilliseconds() !== 0; },
            },
            {
                specifier: ":%S",
                predicate: function (d) { return d.getSeconds() !== 0; },
            },
            {
                specifier: "%I:%M",
                predicate: function (d) { return d.getMinutes() !== 0; },
            },
            {
                specifier: "%I %p",
                predicate: function (d) { return d.getHours() !== 0; },
            },
            {
                specifier: "%a %d",
                predicate: function (d) { return d.getDay() !== 0 && d.getDate() !== 1; },
            },
            {
                specifier: "%b %d",
                predicate: function (d) { return d.getDate() !== 1; },
            },
            {
                specifier: "%b",
                predicate: function (d) { return d.getMonth() !== 0; },
            },
        ];
        return function (d) {
            var acceptableFormats = candidateFormats.filter(function (candidate) { return candidate.predicate(d); });
            var specifier = acceptableFormats.length > 0
                ? acceptableFormats[0].specifier
                : "%Y";
            return d3.timeFormat(specifier)(d);
        };
    }
    exports.multiTime = multiTime;
    /**
     * Creates a time formatter that displays time/date using given specifier.
     *
     * List of directives can be found on: https://github.com/mbostock/d3/wiki/Time-Formatting#format
     *
     * @param {string} [specifier] The specifier for the formatter.
     * @param {boolean} [useUTC] Displays time in UTC if true, local time if false. Defaults to false.
     *
     * @returns {Formatter} A formatter for time/date values.
     */
    function time(specifier, useUTC) {
        if (useUTC === void 0) { useUTC = DEFAULT_USE_UTC; }
        if (useUTC) {
            return d3.utcFormat(specifier);
        }
        return d3.timeFormat(specifier);
    }
    exports.time = time;
    function verifyPrecision(precision) {
        if (precision < 0 || precision > 20) {
            throw new RangeError("Formatter precision must be between 0 and 20");
        }
        if (precision !== Math.floor(precision)) {
            throw new RangeError("Formatter precision must be an integer");
        }
    }
    });

    var axis$2 = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });







    exports.AxisOrientation = makeEnum_1.makeEnum(["bottom", "left", "right", "top"]);
    var Axis = /** @class */ (function (_super) {
        tslib_es6.__extends(Axis, _super);
        /**
         * Constructs an Axis.
         * An Axis is a visual representation of a Scale.
         *
         * @constructor
         * @param {Scale} scale
         * @param {AxisOrientation} orientation Orientation of this Axis.
         */
        function Axis(scale, orientation) {
            var _this = _super.call(this) || this;
            _this._endTickLength = 5;
            _this._innerTickLength = 5;
            _this._tickLabelPadding = 10;
            _this._margin = 15;
            _this._showEndTickLabels = false;
            _this._annotationsEnabled = false;
            _this._annotationTierCount = 1;
            if (scale == null || orientation == null) {
                throw new Error("Axis requires a scale and orientation");
            }
            _this._scale = scale;
            _this.orientation(orientation);
            _this._setDefaultAlignment();
            _this.addClass("axis");
            if (_this.isHorizontal()) {
                _this.addClass("x-axis");
            }
            else {
                _this.addClass("y-axis");
            }
            _this.formatter(formatters.identity());
            _this._rescaleCallback = function (newScale) { return _this._rescale(); };
            _this._scale.onUpdate(_this._rescaleCallback);
            _this._annotatedTicks = [];
            _this._annotationFormatter = formatters.identity();
            return _this;
        }
        Axis.prototype.destroy = function () {
            _super.prototype.destroy.call(this);
            this._scale.offUpdate(this._rescaleCallback);
        };
        /**
         * Gets the tick label data on a element. Element in argument must be a descendent of a tick label element.
         *
         * @param {Element} element
         */
        Axis.prototype.tickLabelDataOnElement = function (element) {
            if (element == null) {
                return undefined;
            }
            var tickLabel;
            // go up DOM tree to find tick label element in ancestor elements
            while ((element != null) && (element.classList) && (tickLabel === undefined)) {
                if (element.classList.contains(Axis.TICK_LABEL_CLASS)) {
                    tickLabel = element;
                }
                else {
                    element = element.parentNode;
                }
            }
            return element === undefined ? undefined : d3.select(element).datum();
        };
        Axis.prototype._computeWidth = function () {
            // to be overridden by subclass logic
            return this._maxLabelTickLength();
        };
        Axis.prototype._computeHeight = function () {
            // to be overridden by subclass logic
            return this._maxLabelTickLength();
        };
        Axis.prototype.requestedSpace = function (offeredWidth, offeredHeight) {
            var requestedWidth = 0;
            var requestedHeight = 0;
            if (this.isHorizontal()) {
                requestedHeight = this._computeHeight() + this._margin;
                if (this.annotationsEnabled()) {
                    var tierHeight = this._annotationMeasurer.measure().height + 2 * Axis._ANNOTATION_LABEL_PADDING;
                    requestedHeight += tierHeight * this.annotationTierCount();
                }
            }
            else {
                requestedWidth = this._computeWidth() + this._margin;
                if (this.annotationsEnabled()) {
                    var tierHeight = this._annotationMeasurer.measure().height + 2 * Axis._ANNOTATION_LABEL_PADDING;
                    requestedWidth += tierHeight * this.annotationTierCount();
                }
            }
            return {
                minWidth: requestedWidth,
                minHeight: requestedHeight,
            };
        };
        Axis.prototype.fixedHeight = function () {
            return this.isHorizontal();
        };
        Axis.prototype.fixedWidth = function () {
            return !this.isHorizontal();
        };
        Axis.prototype._rescale = function () {
            // default implementation; subclasses may call redraw() here
            this.render();
        };
        Axis.prototype.computeLayout = function (origin, availableWidth, availableHeight) {
            _super.prototype.computeLayout.call(this, origin, availableWidth, availableHeight);
            if (this.isHorizontal()) {
                this._scale.range([0, this.width()]);
            }
            else {
                this._scale.range([this.height(), 0]);
            }
            return this;
        };
        Axis.prototype._sizeFromOffer = function (availableWidth, availableHeight) {
            var requestedSpace = this.requestedSpace(availableWidth, availableHeight);
            if (this.isHorizontal()) {
                return {
                    width: availableWidth,
                    // always keep the height to be what we request; Axes tell the outside what height they are.
                    // this allows blueprint-chart to put the Axis in document flow by removing absolute positioning
                    height: requestedSpace.minHeight,
                };
            }
            else {
                return {
                    height: availableHeight,
                    width: requestedSpace.minWidth,
                };
            }
        };
        Axis.prototype._setup = function () {
            _super.prototype._setup.call(this);
            this._tickMarkContainer = this.content().append("g")
                .classed(Axis.TICK_MARK_CLASS + "-container", true);
            this._tickLabelContainer = this.content().append("g")
                .classed(Axis.TICK_LABEL_CLASS + "-container", true);
            this._baseline = this.content().append("line").classed("baseline", true);
            this._annotationContainer = this.content().append("g")
                .classed("annotation-container", true);
            this._annotationContainer.append("g").classed("annotation-line-container", true);
            this._annotationContainer.append("g").classed("annotation-circle-container", true);
            this._annotationContainer.append("g").classed("annotation-rect-container", true);
            var annotationLabelContainer = this._annotationContainer.append("g").classed("annotation-label-container", true);
            var typesetterContext = new src$1.SvgContext(annotationLabelContainer.node());
            this._annotationMeasurer = new src$1.CacheMeasurer(typesetterContext);
            this._annotationWriter = new src$1.Writer(this._annotationMeasurer, typesetterContext);
        };
        /*
         * Function for generating tick values in data-space (as opposed to pixel values).
         * To be implemented by subclasses.
         */
        Axis.prototype._getTickValues = function () {
            return [];
        };
        /**
         * Render tick marks, baseline, and annotations. Should be super called by subclasses and then overridden to draw
         * other relevant aspects of this Axis.
         */
        Axis.prototype.renderImmediately = function () {
            var tickMarkValues = this._getTickValues();
            var tickMarksUpdate = this._tickMarkContainer.selectAll("." + Axis.TICK_MARK_CLASS).data(tickMarkValues);
            var tickMarks = tickMarksUpdate
                .enter()
                .append("line")
                .classed(Axis.TICK_MARK_CLASS, true)
                .merge(tickMarksUpdate);
            tickMarks.attrs(this._generateTickMarkAttrHash());
            d3.select(tickMarks.nodes()[0]).classed(Axis.END_TICK_MARK_CLASS, true)
                .attrs(this._generateTickMarkAttrHash(true));
            d3.select(tickMarks.nodes()[tickMarkValues.length - 1]).classed(Axis.END_TICK_MARK_CLASS, true)
                .attrs(this._generateTickMarkAttrHash(true));
            tickMarksUpdate.exit().remove();
            this._baseline.attrs(this._generateBaselineAttrHash());
            if (this.annotationsEnabled()) {
                this._drawAnnotations();
            }
            else {
                this._removeAnnotations();
            }
            return this;
        };
        Axis.prototype.annotatedTicks = function (annotatedTicks) {
            if (annotatedTicks == null) {
                return this._annotatedTicks;
            }
            this._annotatedTicks = annotatedTicks;
            this.render();
            return this;
        };
        Axis.prototype.annotationFormatter = function (annotationFormatter) {
            if (annotationFormatter == null) {
                return this._annotationFormatter;
            }
            this._annotationFormatter = annotationFormatter;
            this.render();
            return this;
        };
        Axis.prototype.annotationsEnabled = function (annotationsEnabled) {
            if (annotationsEnabled == null) {
                return this._annotationsEnabled;
            }
            this._annotationsEnabled = annotationsEnabled;
            this.redraw();
            return this;
        };
        Axis.prototype.annotationTierCount = function (annotationTierCount) {
            if (annotationTierCount == null) {
                return this._annotationTierCount;
            }
            if (annotationTierCount < 0) {
                throw new Error("annotationTierCount cannot be negative");
            }
            this._annotationTierCount = annotationTierCount;
            this.redraw();
            return this;
        };
        Axis.prototype._drawAnnotations = function () {
            var _this = this;
            var labelPadding = Axis._ANNOTATION_LABEL_PADDING;
            var measurements = new utils$1.Map();
            var annotatedTicks = this._annotatedTicksToRender();
            annotatedTicks.forEach(function (annotatedTick) {
                var measurement = _this._annotationMeasurer.measure(_this.annotationFormatter()(annotatedTick));
                var paddedMeasurement = {
                    width: measurement.width + 2 * labelPadding,
                    height: measurement.height + 2 * labelPadding,
                };
                measurements.set(annotatedTick, paddedMeasurement);
            });
            var tierHeight = this._annotationMeasurer.measure().height + 2 * labelPadding;
            var annotationToTier = this._annotationToTier(measurements);
            var hiddenAnnotations = new utils$1.Set();
            var axisHeight = this.isHorizontal() ? this.height() : this.width();
            var axisHeightWithoutMarginAndAnnotations = this._coreSize();
            var numTiers = Math.min(this.annotationTierCount(), Math.floor((axisHeight - axisHeightWithoutMarginAndAnnotations) / tierHeight));
            annotationToTier.forEach(function (tier, annotation) {
                if (tier === -1 || tier >= numTiers) {
                    hiddenAnnotations.add(annotation);
                }
            });
            var bindElements = function (selection, elementName, className) {
                var elementsUpdate = selection.selectAll("." + className).data(annotatedTicks);
                var elements = elementsUpdate
                    .enter()
                    .append(elementName)
                    .classed(className, true)
                    .merge(elementsUpdate);
                elementsUpdate.exit().remove();
                return elements;
            };
            var offsetF = function (d) {
                switch (_this.orientation()) {
                    case "bottom":
                    case "right":
                        return annotationToTier.get(d) * tierHeight + axisHeightWithoutMarginAndAnnotations;
                    case "top":
                    case "left":
                        return axisHeight - axisHeightWithoutMarginAndAnnotations - annotationToTier.get(d) * tierHeight;
                }
            };
            var positionF = function (d) { return _this._scale.scale(d); };
            var visibilityF = function (d) { return hiddenAnnotations.has(d) ? "hidden" : "visible"; };
            var secondaryPosition;
            switch (this.orientation()) {
                case "bottom":
                case "right":
                    secondaryPosition = 0;
                    break;
                case "top":
                    secondaryPosition = this.height();
                    break;
                case "left":
                    secondaryPosition = this.width();
                    break;
            }
            var isHorizontal = this.isHorizontal();
            bindElements(this._annotationContainer.select(".annotation-line-container"), "line", Axis.ANNOTATION_LINE_CLASS)
                .attrs({
                x1: isHorizontal ? positionF : secondaryPosition,
                x2: isHorizontal ? positionF : offsetF,
                y1: isHorizontal ? secondaryPosition : positionF,
                y2: isHorizontal ? offsetF : positionF,
                visibility: visibilityF,
            });
            bindElements(this._annotationContainer.select(".annotation-circle-container"), "circle", Axis.ANNOTATION_CIRCLE_CLASS)
                .attrs({
                cx: isHorizontal ? positionF : secondaryPosition,
                cy: isHorizontal ? secondaryPosition : positionF,
                r: 3,
            });
            var rectangleOffsetF = function (d) {
                switch (_this.orientation()) {
                    case "bottom":
                    case "right":
                        return offsetF(d);
                    case "top":
                    case "left":
                        return offsetF(d) - measurements.get(d).height;
                }
            };
            bindElements(this._annotationContainer.select(".annotation-rect-container"), "rect", Axis.ANNOTATION_RECT_CLASS)
                .attrs({
                x: isHorizontal ? positionF : rectangleOffsetF,
                y: isHorizontal ? rectangleOffsetF : positionF,
                width: isHorizontal ? function (d) { return measurements.get(d).width; } : function (d) { return measurements.get(d).height; },
                height: isHorizontal ? function (d) { return measurements.get(d).height; } : function (d) { return measurements.get(d).width; },
                visibility: visibilityF,
            });
            var annotationWriter = this._annotationWriter;
            var annotationFormatter = this.annotationFormatter();
            var annotationLabels = bindElements(this._annotationContainer.select(".annotation-label-container"), "g", Axis.ANNOTATION_LABEL_CLASS);
            annotationLabels.selectAll(".text-container").remove();
            annotationLabels.attrs({
                transform: function (d) {
                    var xTranslate = isHorizontal ? positionF(d) : rectangleOffsetF(d);
                    var yTranslate = isHorizontal ? rectangleOffsetF(d) : positionF(d);
                    return "translate(" + xTranslate + "," + yTranslate + ")";
                },
                visibility: visibilityF,
            })
                .each(function (annotationLabel) {
                annotationWriter.write(annotationFormatter(annotationLabel), isHorizontal ? measurements.get(annotationLabel).width : measurements.get(annotationLabel).height, isHorizontal ? measurements.get(annotationLabel).height : measurements.get(annotationLabel).width, {
                    xAlign: "center",
                    yAlign: "center",
                    textRotation: isHorizontal ? 0 : 90,
                }, d3.select(this).node());
            });
        };
        Axis.prototype._annotatedTicksToRender = function () {
            var _this = this;
            var scaleRange = this._scale.range();
            return utils$1.Array.uniq(this.annotatedTicks().filter(function (tick) {
                if (tick == null) {
                    return false;
                }
                return utils$1.Math.inRange(_this._scale.scale(tick), scaleRange[0], scaleRange[1]);
            }));
        };
        /**
         * Retrieves the size of the core pieces.
         *
         * The core pieces include the labels, the end tick marks, the inner tick marks, and the tick label padding.
         */
        Axis.prototype._coreSize = function () {
            var relevantDimension = this.isHorizontal() ? this.height() : this.width();
            var axisHeightWithoutMargin = this.isHorizontal() ? this._computeHeight() : this._computeWidth();
            return Math.min(axisHeightWithoutMargin, relevantDimension);
        };
        Axis.prototype._annotationTierHeight = function () {
            return this._annotationMeasurer.measure().height + 2 * Axis._ANNOTATION_LABEL_PADDING;
        };
        Axis.prototype._annotationToTier = function (measurements) {
            var _this = this;
            var annotationTiers = [[]];
            var annotationToTier = new utils$1.Map();
            var dimension = this.isHorizontal() ? this.width() : this.height();
            this._annotatedTicksToRender().forEach(function (annotatedTick) {
                var position = _this._scale.scale(annotatedTick);
                var length = measurements.get(annotatedTick).width;
                if (position < 0 || position + length > dimension) {
                    annotationToTier.set(annotatedTick, -1);
                    return;
                }
                var tierHasCollision = function (testTier) { return annotationTiers[testTier].some(function (testTick) {
                    var testPosition = _this._scale.scale(testTick);
                    var testLength = measurements.get(testTick).width;
                    return position + length >= testPosition && position <= testPosition + testLength;
                }); };
                var tier = 0;
                while (tierHasCollision(tier)) {
                    tier++;
                    if (annotationTiers.length === tier) {
                        annotationTiers.push([]);
                    }
                }
                annotationTiers[tier].push(annotatedTick);
                annotationToTier.set(annotatedTick, tier);
            });
            return annotationToTier;
        };
        Axis.prototype._removeAnnotations = function () {
            this._annotationContainer.selectAll(".annotation-line").remove();
            this._annotationContainer.selectAll(".annotation-circle").remove();
            this._annotationContainer.selectAll(".annotation-rect").remove();
            this._annotationContainer.selectAll(".annotation-label").remove();
        };
        Axis.prototype._generateBaselineAttrHash = function () {
            var baselineAttrHash = {
                x1: 0,
                y1: 0,
                x2: 0,
                y2: 0,
            };
            switch (this._orientation) {
                case "bottom":
                    baselineAttrHash["x2"] = this.width();
                    break;
                case "top":
                    baselineAttrHash["x2"] = this.width();
                    baselineAttrHash["y1"] = this.height();
                    baselineAttrHash["y2"] = this.height();
                    break;
                case "left":
                    baselineAttrHash["x1"] = this.width();
                    baselineAttrHash["x2"] = this.width();
                    baselineAttrHash["y2"] = this.height();
                    break;
                case "right":
                    baselineAttrHash["y2"] = this.height();
                    break;
            }
            return baselineAttrHash;
        };
        Axis.prototype._generateTickMarkAttrHash = function (isEndTickMark) {
            var _this = this;
            if (isEndTickMark === void 0) { isEndTickMark = false; }
            var tickMarkAttrHash = {
                x1: 0,
                y1: 0,
                x2: 0,
                y2: 0,
            };
            var scalingFunction = function (d) { return _this._scale.scale(d); };
            if (this.isHorizontal()) {
                tickMarkAttrHash["x1"] = scalingFunction;
                tickMarkAttrHash["x2"] = scalingFunction;
            }
            else {
                tickMarkAttrHash["y1"] = scalingFunction;
                tickMarkAttrHash["y2"] = scalingFunction;
            }
            var tickLength = isEndTickMark ? this._endTickLength : this._innerTickLength;
            switch (this._orientation) {
                case "bottom":
                    tickMarkAttrHash["y2"] = tickLength;
                    break;
                case "top":
                    tickMarkAttrHash["y1"] = this.height();
                    tickMarkAttrHash["y2"] = this.height() - tickLength;
                    break;
                case "left":
                    tickMarkAttrHash["x1"] = this.width();
                    tickMarkAttrHash["x2"] = this.width() - tickLength;
                    break;
                case "right":
                    tickMarkAttrHash["x2"] = tickLength;
                    break;
            }
            return tickMarkAttrHash;
        };
        Axis.prototype._setDefaultAlignment = function () {
            switch (this._orientation) {
                case "bottom":
                    this.yAlignment("top");
                    break;
                case "top":
                    this.yAlignment("bottom");
                    break;
                case "left":
                    this.xAlignment("right");
                    break;
                case "right":
                    this.xAlignment("left");
                    break;
            }
        };
        /**
         * Get whether this axis is horizontal (orientation is "top" or "bottom") or vertical.
         * @returns {boolean} - true for horizontal, false for vertical.
         */
        Axis.prototype.isHorizontal = function () {
            return this._orientation === "top" || this._orientation === "bottom";
        };
        /**
         * Get the scale that this axis is associated with.
         * @returns {Scale<D, number>}
         */
        Axis.prototype.getScale = function () {
            return this._scale;
        };
        Axis.prototype.formatter = function (formatter) {
            if (formatter == null) {
                return this._formatter;
            }
            this._formatter = formatter;
            this.redraw();
            return this;
        };
        Axis.prototype.innerTickLength = function (length) {
            if (length == null) {
                return this._innerTickLength;
            }
            else {
                if (length < 0) {
                    throw new Error("inner tick length must be positive");
                }
                this._innerTickLength = length;
                this.redraw();
                return this;
            }
        };
        Axis.prototype.endTickLength = function (length) {
            if (length == null) {
                return this._endTickLength;
            }
            else {
                if (length < 0) {
                    throw new Error("end tick length must be positive");
                }
                this._endTickLength = length;
                this.redraw();
                return this;
            }
        };
        /**
         * Gets the maximum pixel length over all ticks on this axis.
         * @returns {number}
         */
        Axis.prototype._maxLabelTickLength = function () {
            if (this.showEndTickLabels()) {
                return Math.max(this.innerTickLength(), this.endTickLength());
            }
            else {
                return this.innerTickLength();
            }
        };
        Axis.prototype.tickLabelPadding = function (padding) {
            if (padding == null) {
                return this._tickLabelPadding;
            }
            else {
                if (padding < 0) {
                    throw new Error("tick label padding must be positive");
                }
                this._tickLabelPadding = padding;
                this.redraw();
                return this;
            }
        };
        Axis.prototype.margin = function (size) {
            if (size == null) {
                return this._margin;
            }
            else {
                if (size < 0) {
                    throw new Error("margin size must be positive");
                }
                this._margin = size;
                this.redraw();
                return this;
            }
        };
        Axis.prototype.orientation = function (orientation) {
            if (orientation == null) {
                return this._orientation;
            }
            else {
                // ensure backwards compatibility for older versions that supply orientation in different cases
                var newOrientationLC = orientation.toLowerCase();
                if (newOrientationLC !== "top" &&
                    newOrientationLC !== "bottom" &&
                    newOrientationLC !== "left" &&
                    newOrientationLC !== "right") {
                    throw new Error("unsupported orientation");
                }
                this._orientation = newOrientationLC;
                this.redraw();
                return this;
            }
        };
        Axis.prototype.showEndTickLabels = function (show) {
            if (show == null) {
                return this._showEndTickLabels;
            }
            this._showEndTickLabels = show;
            this.render();
            return this;
        };
        Axis.prototype._showAllTickMarks = function () {
            this._tickMarkContainer.selectAll("." + Axis.TICK_MARK_CLASS)
                .each(function () {
                d3.select(this).style("visibility", "inherit");
            });
        };
        Axis.prototype._showAllTickLabels = function () {
            this._tickLabelContainer.selectAll("." + Axis.TICK_LABEL_CLASS)
                .each(function () {
                d3.select(this).style("visibility", "inherit");
            });
        };
        /**
         * Responsible for hiding any tick labels that break out of the bounding
         * container.
         */
        Axis.prototype._hideOverflowingTickLabels = function () {
            var boundingBox = this.element().node().getBoundingClientRect();
            var tickLabels = this._tickLabelContainer.selectAll("." + Axis.TICK_LABEL_CLASS);
            if (tickLabels.empty()) {
                return;
            }
            tickLabels.each(function (d, i) {
                if (!utils$1.DOM.clientRectInside(this.getBoundingClientRect(), boundingBox)) {
                    d3.select(this).style("visibility", "hidden");
                }
            });
        };
        /**
         * Hides the Tick Marks which have no corresponding Tick Labels
         */
        Axis.prototype._hideTickMarksWithoutLabel = function () {
            var visibleTickMarks = this._tickMarkContainer.selectAll("." + Axis.TICK_MARK_CLASS);
            var visibleTickLabels = this._tickLabelContainer
                .selectAll("." + Axis.TICK_LABEL_CLASS)
                .filter(function (d, i) {
                var visibility = d3.select(this).style("visibility");
                return (visibility === "inherit") || (visibility === "visible");
            });
            var labelNumbersShown = visibleTickLabels.data();
            visibleTickMarks.each(function (e, i) {
                if (labelNumbersShown.indexOf(e) === -1) {
                    d3.select(this).style("visibility", "hidden");
                }
            });
        };
        Axis.prototype.invalidateCache = function () {
            _super.prototype.invalidateCache.call(this);
            this._annotationMeasurer.reset();
        };
        /**
         * The css class applied to each end tick mark (the line on the end tick).
         */
        Axis.END_TICK_MARK_CLASS = "end-tick-mark";
        /**
         * The css class applied to each tick mark (the line on the tick).
         */
        Axis.TICK_MARK_CLASS = "tick-mark";
        /**
         * The css class applied to each tick label (the text associated with the tick).
         */
        Axis.TICK_LABEL_CLASS = "tick-label";
        /**
         * The css class applied to each annotation line, which extends from the axis to the rect.
         */
        Axis.ANNOTATION_LINE_CLASS = "annotation-line";
        /**
         * The css class applied to each annotation rect, which surrounds the annotation label.
         */
        Axis.ANNOTATION_RECT_CLASS = "annotation-rect";
        /**
         * The css class applied to each annotation circle, which denotes which tick is being annotated.
         */
        Axis.ANNOTATION_CIRCLE_CLASS = "annotation-circle";
        /**
         * The css class applied to each annotation label, which shows the formatted annotation text.
         */
        Axis.ANNOTATION_LABEL_CLASS = "annotation-label";
        Axis._ANNOTATION_LABEL_PADDING = 4;
        return Axis;
    }(component.Component));
    exports.Axis = Axis;
    });

    var categoryAxis = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });






    var Category = /** @class */ (function (_super) {
        tslib_es6.__extends(Category, _super);
        /**
         * Constructs a Category Axis.
         *
         * A Category Axis is a visual representation of a Category Scale.
         *
         * @constructor
         * @param {Scales.Category} scale
         * @param {AxisOrientation} [orientation="bottom"] Orientation of this Category Axis.
         */
        function Category(scale, orientation) {
            if (orientation === void 0) { orientation = "bottom"; }
            var _this = _super.call(this, scale, orientation) || this;
            /**
             * The rotation angle of tick label text. Only 0, 90, -90 are supported
             */
            _this._tickLabelAngle = 0;
            /**
             * The shear angle of the tick label text. Only values -80 <= x <= 80 are supported
             */
            _this._tickLabelShearAngle = 0;
            _this.addClass("category-axis");
            return _this;
        }
        Object.defineProperty(Category.prototype, "_wrapper", {
            /**
             * A Wrapper configured according to the other properties on this axis.
             * @returns {Typesettable.Wrapper}
             */
            get: function () {
                var wrapper = new src$1.Wrapper();
                if (this._tickLabelMaxLines != null) {
                    wrapper.maxLines(this._tickLabelMaxLines);
                }
                return wrapper;
            },
            enumerable: true,
            configurable: true
        });
        Object.defineProperty(Category.prototype, "_writer", {
            /**
             * A Writer attached to this measurer and wrapper.
             * @returns {Typesettable.Writer}
             */
            get: function () {
                return new src$1.Writer(this._measurer, this._typesetterContext, this._wrapper);
            },
            enumerable: true,
            configurable: true
        });
        Category.prototype._setup = function () {
            _super.prototype._setup.call(this);
            this._typesetterContext = new src$1.SvgContext(this._tickLabelContainer.node());
            this._measurer = new src$1.CacheMeasurer(this._typesetterContext);
        };
        Category.prototype._rescale = function () {
            return this.redraw();
        };
        /**
         * Compute space requirements for this Category Axis. Category Axes have two primary space requirements:
         *
         * 1) width/height needed by the tick lines (including annotations, padding, and margins).
         * 2) width/height needed by the tick text.
         *
         * We requested space is the sum of the lines and text.
         * @param offeredWidth
         * @param offeredHeight
         * @returns {any}
         */
        Category.prototype.requestedSpace = function (offeredWidth, offeredHeight) {
            var widthRequiredByTicks = this.isHorizontal() ? 0 : this._tickSpaceRequired() + this.margin();
            var heightRequiredByTicks = this.isHorizontal() ? this._tickSpaceRequired() + this.margin() : 0;
            if (this._scale.domain().length === 0) {
                return {
                    minWidth: 0,
                    minHeight: 0,
                };
            }
            if (this.annotationsEnabled()) {
                var tierTotalHeight = this._annotationTierHeight() * this.annotationTierCount();
                if (this.isHorizontal()) {
                    heightRequiredByTicks += tierTotalHeight;
                }
                else {
                    widthRequiredByTicks += tierTotalHeight;
                }
            }
            var measureResult = this._measureTickLabels(offeredWidth, offeredHeight);
            return {
                minWidth: measureResult.usedWidth + widthRequiredByTicks,
                minHeight: measureResult.usedHeight + heightRequiredByTicks,
            };
        };
        Category.prototype._coreSize = function () {
            var relevantDimension = this.isHorizontal() ? this.height() : this.width();
            var relevantRequestedSpaceDimension = this.isHorizontal() ?
                this.requestedSpace(this.width(), this.height()).minHeight :
                this.requestedSpace(this.width(), this.height()).minWidth;
            var marginAndAnnotationSize = this.margin() + this._annotationTierHeight();
            var axisHeightWithoutMargin = relevantRequestedSpaceDimension - marginAndAnnotationSize;
            return Math.min(axisHeightWithoutMargin, relevantDimension);
        };
        Category.prototype._getTickValues = function () {
            return this.getDownsampleInfo().domain;
        };
        Category.prototype._sizeFromOffer = function (availableWidth, availableHeight) {
            // hack: continue using Component._sizeFromOffer to prevent angled axis ticks
            // from overflowing their container
            return component.Component.prototype._sizeFromOffer.call(this, availableWidth, availableHeight);
        };
        /**
         * Take the scale and drop ticks at regular intervals such that the resultant ticks are all a reasonable minimum
         * distance apart. Return the resultant ticks to render, as well as the new stepWidth between them.
         *
         * @param {Scales.Category} scale - The scale being downsampled. Defaults to this Axis' scale.
         * @return {DownsampleInfo} an object holding the resultant domain and new stepWidth.
         */
        Category.prototype.getDownsampleInfo = function (scale, domain) {
            if (scale === void 0) { scale = this._scale; }
            if (domain === void 0) { domain = scale.invertRange(); }
            // account for how shearing tightens the space between vertically oriented ticks
            var shearFactor = this._tickLabelAngle === 0 ? 1 : 1 / Math.cos(this._tickLabelShearAngle / 180 * Math.PI);
            var shearedMinimumWidth = Category._MINIMUM_WIDTH_PER_LABEL_PX * shearFactor;
            var downsampleRatio = Math.ceil(shearedMinimumWidth / scale.stepWidth());
            return {
                domain: domain.filter(function (d, i) { return i % downsampleRatio === 0; }),
                stepWidth: downsampleRatio * scale.stepWidth(),
            };
        };
        Category.prototype.tickLabelAngle = function (angle) {
            if (angle == null) {
                return this._tickLabelAngle;
            }
            if (angle !== 0 && angle !== 90 && angle !== -90) {
                throw new Error("Angle " + angle + " not supported; only 0, 90, and -90 are valid values");
            }
            this._tickLabelAngle = angle;
            this.redraw();
            return this;
        };
        Category.prototype.tickLabelShearAngle = function (angle) {
            if (angle == null) {
                return this._tickLabelShearAngle;
            }
            if (angle < -80 || angle > 80) {
                throw new Error("Angle " + angle + " not supported; Must be between [-80, 80]");
            }
            this._tickLabelShearAngle = angle;
            this.redraw();
            return this;
        };
        Category.prototype.tickLabelMaxWidth = function (maxWidth) {
            // allow user to un-set tickLabelMaxWidth by passing in null or undefined explicitly
            if (arguments.length === 0) {
                return this._tickLabelMaxWidth;
            }
            this._tickLabelMaxWidth = maxWidth;
            this.redraw();
            return this;
        };
        Category.prototype.tickLabelMaxLines = function (maxLines) {
            // allow user to un-set tickLabelMaxLines by passing in null or undefined explicitly
            if (arguments.length === 0) {
                return this._tickLabelMaxLines;
            }
            this._tickLabelMaxLines = maxLines;
            this.redraw();
            return this;
        };
        /**
         * Return the space required by the ticks, padding included.
         * @returns {number}
         */
        Category.prototype._tickSpaceRequired = function () {
            return this._maxLabelTickLength() + this.tickLabelPadding();
        };
        /**
         * Write ticks to the DOM.
         * @param {Plottable.Scales.Category} scale The scale this axis is representing.
         * @param {d3.Selection} ticks The tick elements to write.
         */
        Category.prototype._drawTicks = function (stepWidth, ticks) {
            var self = this;
            var xAlign;
            var yAlign;
            switch (this.tickLabelAngle()) {
                case 0:
                    xAlign = { left: "right", right: "left", top: "center", bottom: "center" };
                    yAlign = { left: "center", right: "center", top: "bottom", bottom: "top" };
                    break;
                case 90:
                    xAlign = { left: "center", right: "center", top: "right", bottom: "left" };
                    yAlign = { left: "top", right: "bottom", top: "center", bottom: "center" };
                    break;
                case -90:
                    xAlign = { left: "center", right: "center", top: "left", bottom: "right" };
                    yAlign = { left: "bottom", right: "top", top: "center", bottom: "center" };
                    break;
            }
            ticks.each(function (d) {
                var container = d3.select(this);
                var width = self.isHorizontal() ? stepWidth : self.width() - self._tickSpaceRequired();
                var height = self.isHorizontal() ? self.height() - self._tickSpaceRequired() : stepWidth;
                var writeOptions = {
                    xAlign: xAlign[self.orientation()],
                    yAlign: yAlign[self.orientation()],
                    textRotation: self.tickLabelAngle(),
                    textShear: self.tickLabelShearAngle(),
                };
                if (self._tickLabelMaxWidth != null) {
                    // for left-oriented axes, we must move the ticks by the amount we've cut off in order to keep the text
                    // aligned with the side of the ticks
                    if (self.orientation() === "left" && width > self._tickLabelMaxWidth) {
                        var cutOffWidth = width - self._tickLabelMaxWidth;
                        var newTransform = container.attr("transform") + " translate(" + cutOffWidth + ", 0)";
                        container.attr("transform", newTransform);
                    }
                    width = Math.min(width, self._tickLabelMaxWidth);
                }
                self._writer.write(self.formatter()(d), width, height, writeOptions, container.node());
            });
        };
        /**
         * Measures the size of the tick labels without making any (permanent) DOM changes.
         *
         * @param {number} axisWidth Width available for this axis.
         * @param {number} axisHeight Height available for this axis.
         * @param {Plottable.Scales.Category} scale The scale this axis is representing.
         * @param {string[]} ticks The strings that will be printed on the ticks.
         */
        Category.prototype._measureTickLabels = function (axisWidth, axisHeight) {
            var _this = this;
            var thisScale = this._scale;
            // set up a test scale to simulate rendering ticks with the given width and height.
            var scale = thisScale.cloneWithoutProviders()
                .range([0, this.isHorizontal() ? axisWidth : axisHeight]);
            var _a = this.getDownsampleInfo(scale), domain = _a.domain, stepWidth = _a.stepWidth;
            // HACKHACK: https://github.com/palantir/svg-typewriter/issues/25
            // the width (x-axis specific) available to a single tick label.
            var width = axisWidth - this._tickSpaceRequired(); // default for left/right
            if (this.isHorizontal()) {
                width = stepWidth; // defaults to the band width
                if (this._tickLabelAngle !== 0) {
                    width = axisHeight - this._tickSpaceRequired(); // use the axis height
                }
                // HACKHACK: Wrapper fails under negative circumstances
                width = Math.max(width, 0);
            }
            // HACKHACK: https://github.com/palantir/svg-typewriter/issues/25
            // the height (y-axis specific) available to a single tick label.
            var height = stepWidth; // default for left/right
            if (this.isHorizontal()) {
                height = axisHeight - this._tickSpaceRequired();
                if (this._tickLabelAngle !== 0) {
                    height = axisWidth - this._tickSpaceRequired();
                }
                // HACKHACK: Wrapper fails under negative circumstances
                height = Math.max(height, 0);
            }
            if (this._tickLabelMaxWidth != null) {
                width = Math.min(width, this._tickLabelMaxWidth);
            }
            var wrappingResults = domain.map(function (s) {
                return _this._wrapper.wrap(_this.formatter()(s), _this._measurer, width, height);
            });
            // HACKHACK: https://github.com/palantir/svg-typewriter/issues/25
            var widthFn = (this.isHorizontal() && this._tickLabelAngle === 0) ? d3.sum : utils$1.Math.max;
            var heightFn = (this.isHorizontal() && this._tickLabelAngle === 0) ? utils$1.Math.max : d3.sum;
            var usedWidth = widthFn(wrappingResults, function (t) { return _this._measurer.measure(t.wrappedText).width; }, 0);
            var usedHeight = heightFn(wrappingResults, function (t) { return _this._measurer.measure(t.wrappedText).height; }, 0);
            // If the tick labels are rotated, reverse usedWidth and usedHeight
            // HACKHACK: https://github.com/palantir/svg-typewriter/issues/25
            if (this._tickLabelAngle !== 0) {
                _b = [usedHeight, usedWidth], usedWidth = _b[0], usedHeight = _b[1];
            }
            return {
                usedWidth: usedWidth,
                usedHeight: usedHeight,
            };
            var _b;
        };
        Category.prototype.renderImmediately = function () {
            var _this = this;
            _super.prototype.renderImmediately.call(this);
            var catScale = this._scale;
            var _a = this.getDownsampleInfo(catScale), domain = _a.domain, stepWidth = _a.stepWidth;
            // Give each tick a stepWidth of space which will partition the entire axis evenly
            var availableTextSpace = stepWidth;
            if (this.isHorizontal() && this._tickLabelMaxWidth != null) {
                availableTextSpace = Math.min(availableTextSpace, this._tickLabelMaxWidth);
            }
            var getTickLabelTransform = function (d, i) {
                // scale(d) will give the center of the band, so subtract half of the text width to get the left (top-most)
                // coordinate that the tick label should be transformed to.
                var tickLabelEdge = catScale.scale(d) - availableTextSpace / 2;
                var x = _this.isHorizontal() ? tickLabelEdge : 0;
                var y = _this.isHorizontal() ? 0 : tickLabelEdge;
                return "translate(" + x + "," + y + ")";
            };
            var tickLabelsUpdate = this._tickLabelContainer.selectAll("." + axis$2.Axis.TICK_LABEL_CLASS).data(domain);
            var tickLabels = tickLabelsUpdate
                .enter()
                .append("g")
                .classed(axis$2.Axis.TICK_LABEL_CLASS, true)
                .merge(tickLabelsUpdate);
            tickLabelsUpdate.exit().remove();
            tickLabels.attr("transform", getTickLabelTransform);
            // erase all text first, then rewrite
            tickLabels.text("");
            this._drawTicks(stepWidth, tickLabels);
            var xTranslate = this.orientation() === "right" ? this._tickSpaceRequired() : 0;
            var yTranslate = this.orientation() === "bottom" ? this._tickSpaceRequired() : 0;
            this._tickLabelContainer.attr("transform", "translate(" + xTranslate + "," + yTranslate + ")");
            // hide ticks and labels that overflow the axis
            this._showAllTickMarks();
            this._showAllTickLabels();
            this._hideTickMarksWithoutLabel();
            return this;
        };
        Category.prototype.computeLayout = function (origin, availableWidth, availableHeight) {
            _super.prototype.computeLayout.call(this, origin, availableWidth, availableHeight);
            if (!this.isHorizontal()) {
                this._scale.range([0, this.height()]);
            }
            return this;
        };
        Category.prototype.invalidateCache = function () {
            _super.prototype.invalidateCache.call(this);
            this._measurer.reset();
        };
        /**
         * How many pixels to give labels at minimum before downsampling takes effect.
         */
        Category._MINIMUM_WIDTH_PER_LABEL_PX = 15;
        return Category;
    }(axis$2.Axis));
    exports.Category = Category;
    });

    var numericAxis = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });






    var Numeric = /** @class */ (function (_super) {
        tslib_es6.__extends(Numeric, _super);
        /**
         * Constructs a Numeric Axis.
         *
         * A Numeric Axis is a visual representation of a QuantitativeScale.
         *
         * @constructor
         * @param {QuantitativeScale} scale
         * @param {AxisOrientation} orientation Orientation of this Numeric Axis.
         */
        function Numeric(scale, orientation) {
            var _this = _super.call(this, scale, orientation) || this;
            _this._tickLabelPositioning = "center";
            _this._usesTextWidthApproximation = false;
            _this.formatter(formatters.general());
            return _this;
        }
        Numeric.prototype._setup = function () {
            _super.prototype._setup.call(this);
            var context = new src$1.SvgContext(this._tickLabelContainer.node(), axis$2.Axis.TICK_LABEL_CLASS);
            this._measurer = new src$1.CacheMeasurer(context);
            this._wrapper = new src$1.Wrapper().maxLines(1);
        };
        Numeric.prototype._computeWidth = function () {
            var maxTextWidth = this._usesTextWidthApproximation ? this._computeApproximateTextWidth() : this._computeExactTextWidth();
            if (this._tickLabelPositioning === "center") {
                return this._maxLabelTickLength() + this.tickLabelPadding() + maxTextWidth;
            }
            else {
                return Math.max(this._maxLabelTickLength(), this.tickLabelPadding() + maxTextWidth);
            }
        };
        Numeric.prototype._computeExactTextWidth = function () {
            var _this = this;
            var tickValues = this._getTickValues();
            var textLengths = tickValues.map(function (v) {
                var formattedValue = _this.formatter()(v);
                return _this._measurer.measure(formattedValue).width;
            });
            return utils$1.Math.max(textLengths, 0);
        };
        Numeric.prototype._computeApproximateTextWidth = function () {
            var _this = this;
            var tickValues = this._getTickValues();
            var mWidth = this._measurer.measure("M").width;
            var textLengths = tickValues.map(function (v) {
                var formattedValue = _this.formatter()(v);
                return formattedValue.length * mWidth;
            });
            return utils$1.Math.max(textLengths, 0);
        };
        Numeric.prototype._computeHeight = function () {
            var textHeight = this._measurer.measure().height;
            if (this._tickLabelPositioning === "center") {
                return this._maxLabelTickLength() + this.tickLabelPadding() + textHeight;
            }
            else {
                return Math.max(this._maxLabelTickLength(), this.tickLabelPadding() + textHeight);
            }
        };
        Numeric.prototype._getTickValues = function () {
            var scale = this._scale;
            var domain = scale.domain();
            var min = domain[0] <= domain[1] ? domain[0] : domain[1];
            var max = domain[0] >= domain[1] ? domain[0] : domain[1];
            return scale.ticks().filter(function (i) { return i >= min && i <= max; });
        };
        Numeric.prototype._rescale = function () {
            if (!this._isSetup) {
                return;
            }
            if (!this.isHorizontal()) {
                var reComputedWidth = this._computeWidth();
                if (reComputedWidth > this.width() || reComputedWidth < (this.width() - this.margin())) {
                    this.redraw();
                    return;
                }
            }
            this.render();
        };
        Numeric.prototype.renderImmediately = function () {
            var _this = this;
            _super.prototype.renderImmediately.call(this);
            var tickLabelAttrHash = {
                x: 0,
                y: 0,
                dx: "0em",
                dy: "0.3em",
            };
            var tickMarkLength = this._maxLabelTickLength();
            var tickLabelPadding = this.tickLabelPadding();
            var tickLabelTextAnchor = "middle";
            var labelGroupTransformX = 0;
            var labelGroupTransformY = 0;
            var labelGroupShiftX = 0;
            var labelGroupShiftY = 0;
            if (this.isHorizontal()) {
                switch (this._tickLabelPositioning) {
                    case "left":
                        tickLabelTextAnchor = "end";
                        labelGroupTransformX = -tickLabelPadding;
                        labelGroupShiftY = tickLabelPadding;
                        break;
                    case "center":
                        labelGroupShiftY = tickMarkLength + tickLabelPadding;
                        break;
                    case "right":
                        tickLabelTextAnchor = "start";
                        labelGroupTransformX = tickLabelPadding;
                        labelGroupShiftY = tickLabelPadding;
                        break;
                }
            }
            else {
                switch (this._tickLabelPositioning) {
                    case "top":
                        tickLabelAttrHash["dy"] = "-0.3em";
                        labelGroupShiftX = tickLabelPadding;
                        labelGroupTransformY = -tickLabelPadding;
                        break;
                    case "center":
                        labelGroupShiftX = tickMarkLength + tickLabelPadding;
                        break;
                    case "bottom":
                        tickLabelAttrHash["dy"] = "1em";
                        labelGroupShiftX = tickLabelPadding;
                        labelGroupTransformY = tickLabelPadding;
                        break;
                }
            }
            var tickMarkAttrHash = this._generateTickMarkAttrHash();
            switch (this.orientation()) {
                case "bottom":
                    tickLabelAttrHash["x"] = tickMarkAttrHash["x1"];
                    tickLabelAttrHash["dy"] = "0.95em";
                    labelGroupTransformY = tickMarkAttrHash["y1"] + labelGroupShiftY;
                    break;
                case "top":
                    tickLabelAttrHash["x"] = tickMarkAttrHash["x1"];
                    tickLabelAttrHash["dy"] = "-.25em";
                    labelGroupTransformY = tickMarkAttrHash["y1"] - labelGroupShiftY;
                    break;
                case "left":
                    tickLabelTextAnchor = "end";
                    labelGroupTransformX = tickMarkAttrHash["x1"] - labelGroupShiftX;
                    tickLabelAttrHash["y"] = tickMarkAttrHash["y1"];
                    break;
                case "right":
                    tickLabelTextAnchor = "start";
                    labelGroupTransformX = tickMarkAttrHash["x1"] + labelGroupShiftX;
                    tickLabelAttrHash["y"] = tickMarkAttrHash["y1"];
                    break;
            }
            var tickLabelValues = this._getTickValues();
            var tickLabelsUpdate = this._tickLabelContainer.selectAll("." + axis$2.Axis.TICK_LABEL_CLASS).data(tickLabelValues);
            tickLabelsUpdate.exit().remove();
            var tickLabels = tickLabelsUpdate
                .enter()
                .append("text")
                .classed(axis$2.Axis.TICK_LABEL_CLASS, true)
                .merge(tickLabelsUpdate);
            tickLabels.style("text-anchor", tickLabelTextAnchor)
                .style("visibility", "inherit")
                .attrs(tickLabelAttrHash)
                .text(function (s) { return _this.formatter()(s); });
            var labelGroupTransform = "translate(" + labelGroupTransformX + ", " + labelGroupTransformY + ")";
            this._tickLabelContainer.attr("transform", labelGroupTransform);
            this._showAllTickMarks();
            if (!this.showEndTickLabels()) {
                this._hideEndTickLabels();
            }
            this._hideOverflowingTickLabels();
            this._hideOverlappingTickLabels();
            if (this._tickLabelPositioning !== "center") {
                this._hideTickMarksWithoutLabel();
            }
            return this;
        };
        Numeric.prototype.tickLabelPosition = function (position) {
            if (position == null) {
                return this._tickLabelPositioning;
            }
            else {
                var positionLC = position.toLowerCase();
                if (this.isHorizontal()) {
                    if (!(positionLC === "left" || positionLC === "center" || positionLC === "right")) {
                        throw new Error(positionLC + " is not a valid tick label position for a horizontal NumericAxis");
                    }
                }
                else {
                    if (!(positionLC === "top" || positionLC === "center" || positionLC === "bottom")) {
                        throw new Error(positionLC + " is not a valid tick label position for a vertical NumericAxis");
                    }
                }
                this._tickLabelPositioning = positionLC;
                this.redraw();
                return this;
            }
        };
        Numeric.prototype.usesTextWidthApproximation = function (enable) {
            if (enable == null) {
                return this._usesTextWidthApproximation;
            }
            else {
                this._usesTextWidthApproximation = enable;
                return this;
            }
        };
        Numeric.prototype._hideEndTickLabels = function () {
            var boundingBox = this.element().node().getBoundingClientRect();
            var tickLabels = this._tickLabelContainer.selectAll("." + axis$2.Axis.TICK_LABEL_CLASS);
            if (tickLabels.size() === 0) {
                return;
            }
            var firstTickLabel = tickLabels.nodes()[0];
            if (!utils$1.DOM.clientRectInside(firstTickLabel.getBoundingClientRect(), boundingBox)) {
                d3.select(firstTickLabel).style("visibility", "hidden");
            }
            var lastTickLabel = tickLabels.nodes()[tickLabels.size() - 1];
            if (!utils$1.DOM.clientRectInside(lastTickLabel.getBoundingClientRect(), boundingBox)) {
                d3.select(lastTickLabel).style("visibility", "hidden");
            }
        };
        Numeric.prototype._hideOverlappingTickLabels = function () {
            var visibleTickLabels = this._tickLabelContainer
                .selectAll("." + axis$2.Axis.TICK_LABEL_CLASS)
                .filter(function (d, i) {
                var visibility = d3.select(this).style("visibility");
                return (visibility === "inherit") || (visibility === "visible");
            });
            var visibleTickLabelRects = visibleTickLabels.nodes().map(function (label) { return label.getBoundingClientRect(); });
            var interval = 1;
            while (!this._hasOverlapWithInterval(interval, visibleTickLabelRects) && interval < visibleTickLabelRects.length) {
                interval += 1;
            }
            visibleTickLabels.each(function (d, i) {
                var tickLabel = d3.select(this);
                if (i % interval !== 0) {
                    tickLabel.style("visibility", "hidden");
                }
            });
        };
        /**
         * The method is responsible for evenly spacing the labels on the axis.
         * @return test to see if taking every `interval` recrangle from `rects`
         *         will result in labels not overlapping
         *
         * For top, bottom, left, right positioning of the thicks, we want the padding
         * between the labels to be 3x, such that the label will be  `padding` distance
         * from the tick and 2 * `padding` distance (or more) from the next tick:
         * see https://github.com/palantir/plottable/pull/1812
         */
        Numeric.prototype._hasOverlapWithInterval = function (interval, rects) {
            var padding = (this._tickLabelPositioning === "center")
                ? this.tickLabelPadding()
                : this.tickLabelPadding() * 3;
            var rectsWithPadding = rects.map(function (rect) { return utils$1.DOM.expandRect(rect, padding); });
            for (var i = 0; i < rectsWithPadding.length - interval; i += interval) {
                var currRect = rectsWithPadding[i];
                var nextRect = rectsWithPadding[i + interval];
                if (utils$1.DOM.clientRectsOverlap(currRect, nextRect)) {
                    return false;
                }
            }
            return true;
        };
        Numeric.prototype.invalidateCache = function () {
            _super.prototype.invalidateCache.call(this);
            this._measurer.reset();
        };
        return Numeric;
    }(axis$2.Axis));
    exports.Numeric = Numeric;
    });

    var tickGenerators = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });

    /**
     * Creates a TickGenerator using the specified interval.
     *
     * Generates ticks at multiples of the interval while also including the domain boundaries.
     *
     * @param {number} interval
     * @returns {TickGenerator}
     */
    function intervalTickGenerator(interval) {
        if (interval <= 0) {
            throw new Error("interval must be positive number");
        }
        return function (s) {
            var domain = s.domain();
            var low = Math.min(domain[0], domain[1]);
            var high = Math.max(domain[0], domain[1]);
            var firstTick = Math.ceil(low / interval) * interval;
            var numTicks = Math.floor((high - firstTick) / interval) + 1;
            var lowTicks = low % interval === 0 ? [] : [low];
            var middleTicks = utils$1.Math.range(0, numTicks).map(function (t) { return firstTick + t * interval; });
            var highTicks = high % interval === 0 ? [] : [high];
            return lowTicks.concat(middleTicks).concat(highTicks);
        };
    }
    exports.intervalTickGenerator = intervalTickGenerator;
    /**
     * Creates a TickGenerator returns only integer tick values.
     *
     * @returns {TickGenerator}
     */
    function integerTickGenerator() {
        return function (s) {
            var defaultTicks = s.defaultTicks();
            return defaultTicks.filter(function (tick, i) { return (tick % 1 === 0) || (i === 0) || (i === defaultTicks.length - 1); });
        };
    }
    exports.integerTickGenerator = integerTickGenerator;
    });

    var panZoomConstraints = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2017-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });
    /**
     * Return `value` if its distance to `center` was scaled by `zoom`.
     *
     * e.g. zoomOut(100, 2, 50) -> 150
     * e.g. zoomOut(0, 2, 50) -> -50
     * e.g. zoomOut(100, 0.5, 50) -> 75
     * e.g. zoomOut(0, 0.5, 50) -> 25
     */
    function zoomOut(value, zoom, center) {
        return center - (center - value) * zoom;
    }
    exports.zoomOut = zoomOut;
    /**
     * This is the zoomOut method algebra-ed to solve for the `center` value
     * given the other three. The "target" is the return value of zoomOut.
     */
    function getZoomOutCenter(value, zoom, target) {
        return (value * zoom - target) / (zoom - 1);
    }
    /**
     * Return possibly different zoomAmount and centerPoint values such that applying the zoom
     * to the scale respects the given min/max extents and values.
     */
    function constrainedZoom(scale, zoomAmount, centerPoint, minDomainExtent, maxDomainExtent, minDomainValue, maxDomainValue) {
        zoomAmount = constrainZoomExtents(scale, zoomAmount, minDomainExtent, maxDomainExtent);
        return constrainZoomValues(scale, zoomAmount, centerPoint, minDomainValue, maxDomainValue);
    }
    exports.constrainedZoom = constrainedZoom;
    function constrainZoomExtents(scale, zoomAmount, minDomainExtent, maxDomainExtent) {
        var extentIncreasing = zoomAmount > 1;
        var boundingDomainExtent = extentIncreasing ? maxDomainExtent : minDomainExtent;
        if (boundingDomainExtent == null) {
            return zoomAmount;
        }
        var _a = scale.getTransformationDomain(), scaleDomainMin = _a[0], scaleDomainMax = _a[1];
        var domainExtent = Math.abs(scaleDomainMax - scaleDomainMin);
        var compareF = extentIncreasing ? Math.min : Math.max;
        return compareF(zoomAmount, boundingDomainExtent / domainExtent);
    }
    exports.constrainZoomExtents = constrainZoomExtents;
    /**
     * Modify zoomAmount and centerPoint such that a panzoom will stay inside
     * the bounds defined by minDomainValue and maxDomainValue.
     */
    function constrainZoomValues(scale, zoomAmount, centerPoint, minDomainValue, maxDomainValue) {
        // when zooming in, we don't have to worry about overflowing domain
        if (zoomAmount <= 1) {
            return { centerPoint: centerPoint, zoomAmount: zoomAmount };
        }
        // if no constraints set, we're done
        if (minDomainValue == null && maxDomainValue == null) {
            return { centerPoint: centerPoint, zoomAmount: zoomAmount };
        }
        var rangeReversed = isRangeReversed(scale);
        var domainReversed = isDomainReversed(scale);
        var defaultDomainMin = domainReversed ? Infinity : -Infinity;
        var defaultDomainMax = domainReversed ? -Infinity : Infinity;
        minDomainValue = minDomainValue == null ? defaultDomainMin : minDomainValue;
        maxDomainValue = maxDomainValue == null ? defaultDomainMax : maxDomainValue;
        var _a = scale.getTransformationDomain(), scaleDomainMin = _a[0], scaleDomainMax = _a[1];
        var maxRange = scale.scaleTransformation(maxDomainValue);
        var currentMaxRange = scale.scaleTransformation(scaleDomainMax);
        var newMaxRange = zoomOut(currentMaxRange, zoomAmount, centerPoint);
        var minRange = scale.scaleTransformation(minDomainValue);
        var currentMinRange = scale.scaleTransformation(scaleDomainMin);
        var newMinRange = zoomOut(currentMinRange, zoomAmount, centerPoint);
        var minMaxLength = Math.abs(maxRange - minRange);
        var newRangeLength = Math.abs(newMaxRange - newMinRange);
        if (newRangeLength > minMaxLength) {
            // The new zoom simply won't fit. Instead just set the zoom amount to a full zoom out.
            var wantedZoomAmount = (maxRange - minRange) / (currentMaxRange - currentMinRange);
            if (wantedZoomAmount !== 1) {
                // only solve for centerPoint if wantedZoomAmount isn't 1 to prevent NaN.
                var wantedCenterPoint = getZoomOutCenter(currentMaxRange, wantedZoomAmount, maxRange);
                return {
                    centerPoint: wantedCenterPoint,
                    zoomAmount: wantedZoomAmount,
                };
            }
            else {
                // the centerPoint doesn't matter at all here but we must include it
                return {
                    centerPoint: centerPoint,
                    zoomAmount: wantedZoomAmount,
                };
            }
        }
        else {
            // the zoom does fit, but one end is outside. In this case just nudge the edge in
            if (newMaxRange > maxRange != rangeReversed) {
                // prevent out of bounds on max edge.
                return {
                    centerPoint: getZoomOutCenter(currentMaxRange, zoomAmount, maxRange),
                    zoomAmount: zoomAmount,
                };
            }
            else if (newMinRange < minRange != rangeReversed) {
                // prevent out of bounds on min edge.
                return {
                    centerPoint: getZoomOutCenter(currentMinRange, zoomAmount, minRange),
                    zoomAmount: zoomAmount,
                };
            }
            else {
                return { centerPoint: centerPoint, zoomAmount: zoomAmount };
            }
        }
    }
    exports.constrainZoomValues = constrainZoomValues;
    /**
     * Returns a new translation value that respects domain min/max value
     * constraints.
     */
    function constrainedTranslation(scale, translation, minDomainValue, maxDomainValue) {
        var _a = scale.getTransformationDomain(), scaleDomainMin = _a[0], scaleDomainMax = _a[1];
        var reversed = isRangeReversed(scale);
        if (translation > 0 !== reversed) {
            var bound = maxDomainValue;
            if (bound != null) {
                var currentMaxRange = scale.scaleTransformation(scaleDomainMax);
                var maxRange = scale.scaleTransformation(bound);
                translation = (reversed ? Math.max : Math.min)(currentMaxRange + translation, maxRange) - currentMaxRange;
            }
        }
        else {
            var bound = minDomainValue;
            if (bound != null) {
                var currentMinRange = scale.scaleTransformation(scaleDomainMin);
                var minRange = scale.scaleTransformation(bound);
                translation = (reversed ? Math.min : Math.max)(currentMinRange + translation, minRange) - currentMinRange;
            }
        }
        return translation;
    }
    exports.constrainedTranslation = constrainedTranslation;
    /**
     * Returns true iff the scale.range[1] < scale.range[0].
     */
    function isRangeReversed(scale) {
        var range = scale.range();
        return range[1] < range[0];
    }
    function isDomainReversed(scale) {
        var domain = scale.getTransformationDomain();
        return domain[1] < domain[0];
    }
    });

    var scale = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });

    var Scale = /** @class */ (function () {
        /**
         * A Scale is a function (in the mathematical sense) that maps values from a domain to a range.
         *
         * @constructor
         */
        function Scale() {
            this._autoDomainAutomatically = true;
            this._domainModificationInProgress = false;
            this._updateId = 0;
            this._callbacks = new utils$1.CallbackSet();
            this._includedValuesProviders = new utils$1.Set();
        }
        /**
         * Given an array of potential domain values, computes the extent of those values.
         *
         * @param {D[]} values
         * @returns {D[]} The extent of the input values.
         */
        Scale.prototype.extentOfValues = function (values) {
            return []; // this should be overwritten
        };
        Scale.prototype._getAllIncludedValues = function (ignoreAttachState) {
            var _this = this;
            if (ignoreAttachState === void 0) { ignoreAttachState = false; }
            var providerArray = [];
            this._includedValuesProviders.forEach(function (provider) {
                var extents = provider(_this, ignoreAttachState);
                providerArray = providerArray.concat(extents);
            });
            return providerArray;
        };
        Scale.prototype._getExtent = function () {
            return []; // this should be overwritten
        };
        /**
         * Adds a callback to be called when the Scale updates.
         *
         * @param {ScaleCallback} callback.
         * @returns {Scale} The calling Scale.
         */
        Scale.prototype.onUpdate = function (callback) {
            this._callbacks.add(callback);
            return this;
        };
        /**
         * Removes a callback that would be called when the Scale updates.
         *
         * @param {ScaleCallback} callback.
         * @returns {Scale} The calling Scale.
         */
        Scale.prototype.offUpdate = function (callback) {
            this._callbacks.delete(callback);
            return this;
        };
        Scale.prototype._dispatchUpdate = function () {
            this._updateId++;
            this._callbacks.callCallbacks(this);
        };
        /**
         * Sets the Scale's domain so that it spans the Extents of all its ExtentsProviders.
         *
         * @returns {Scale} The calling Scale.
         */
        Scale.prototype.autoDomain = function () {
            this._autoDomainAutomatically = true;
            this._setDomain(this._getExtent());
            return this;
        };
        /**
         * Triggers `.autoDomain()` if the domain is not explicitly set.
         */
        Scale.prototype.autoDomainIfAutomaticMode = function () {
            if (this._autoDomainAutomatically) {
                this.autoDomain();
            }
        };
        /**
         * Computes the range value corresponding to a given domain value.
         *
         * @param {D} value
         * @returns {R} The range value corresponding to the supplied domain value.
         */
        Scale.prototype.scale = function (value) {
            throw new Error("Subclasses should override scale");
        };
        /**
         * Gets an array of tick values spanning the domain.
         *
         * @returns {D[]}
         */
        Scale.prototype.ticks = function () {
            return this.domain();
        };
        Scale.prototype.domain = function (values) {
            if (values == null) {
                return this._getDomain();
            }
            else {
                this._autoDomainAutomatically = false;
                this._setDomain(values);
                return this;
            }
        };
        Scale.prototype._getDomain = function () {
            throw new Error("Subclasses should override _getDomain");
        };
        Scale.prototype._setDomain = function (values) {
            if (!this._domainModificationInProgress) {
                this._domainModificationInProgress = true;
                this._backingScaleDomain(values);
                this._dispatchUpdate();
                this._domainModificationInProgress = false;
            }
        };
        Scale.prototype._backingScaleDomain = function (values) {
            throw new Error("Subclasses should override _backingDomain");
        };
        Scale.prototype.range = function (values) {
            if (values == null) {
                return this._getRange();
            }
            else {
                this._setRange(values);
                return this;
            }
        };
        Scale.prototype._getRange = function () {
            throw new Error("Subclasses should override _getRange");
        };
        Scale.prototype._setRange = function (values) {
            throw new Error("Subclasses should override _setRange");
        };
        /**
         * Adds an IncludedValuesProvider to the Scale.
         *
         * @param {Scales.IncludedValuesProvider} provider
         * @returns {Scale} The calling Scale.
         */
        Scale.prototype.addIncludedValuesProvider = function (provider) {
            this._includedValuesProviders.add(provider);
            this.autoDomainIfAutomaticMode();
            return this;
        };
        /**
         * Removes the IncludedValuesProvider from the Scale.
         *
         * @param {Scales.IncludedValuesProvider} provider
         * @returns {Scale} The calling Scale.
         */
        Scale.prototype.removeIncludedValuesProvider = function (provider) {
            this._includedValuesProviders.delete(provider);
            this.autoDomainIfAutomaticMode();
            return this;
        };
        Scale.prototype.updateId = function () {
            return this._updateId;
        };
        return Scale;
    }());
    exports.Scale = Scale;
    });

    var categoryScale = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });





    var TRANSFORMATION_SPACE = [0, 1];
    var Category = /** @class */ (function (_super) {
        tslib_es6.__extends(Category, _super);
        /**
         * A Category Scale maps strings to numbers.
         *
         * @constructor
         */
        function Category() {
            var _this = _super.call(this) || this;
            _this._range = [0, 1];
            _this._d3Scale = d3.scaleBand();
            _this._d3Scale.range(TRANSFORMATION_SPACE);
            _this._d3TransformationScale = d3.scaleLinear();
            _this._d3TransformationScale.domain(TRANSFORMATION_SPACE);
            var d3InnerPadding = 0.3;
            _this._innerPadding = Category._convertToPlottableInnerPadding(d3InnerPadding);
            _this._outerPadding = Category._convertToPlottableOuterPadding(0.5, d3InnerPadding);
            return _this;
        }
        /**
         * Return a clone of this category scale that holds the same pan/zoom, padding, domain and range, but
         * without any included values providers.
         */
        Category.prototype.cloneWithoutProviders = function () {
            var scale = new Category()
                .domain(this.domain())
                .range(this.range())
                .innerPadding(this.innerPadding())
                .outerPadding(this.outerPadding());
            scale._d3TransformationScale.domain(this._d3TransformationScale.domain());
            return scale;
        };
        Category.prototype.extentOfValues = function (values) {
            return utils$1.Array.uniq(values);
        };
        Category.prototype._getExtent = function () {
            return utils$1.Array.uniq(this._getAllIncludedValues());
        };
        Category.prototype.domain = function (values) {
            return _super.prototype.domain.call(this, values);
        };
        /**
         * Returns domain values that lie inside the given range.
         * @param range
         * @returns {string[]}
         */
        Category.prototype.invertRange = function (range) {
            var _this = this;
            if (range === void 0) { range = this.range(); }
            var rangeBand = this._d3Scale.bandwidth();
            var domainStartNormalized = this.invertedTransformation(range[0]);
            var domainEndNormalized = this.invertedTransformation(range[1]);
            var domain = this._d3Scale.domain();
            // map ["a", "b", "c"] to the normalized center position (e.g. [0.25, .5, 0.75]). We add
            // half the rangeBand to consider the center of the bars
            var normalizedDomain = domain.map(function (d) { return _this._d3Scale(d) + rangeBand / 2; });
            var domainStart = d3.bisect(normalizedDomain, domainStartNormalized);
            var domainEnd = d3.bisect(normalizedDomain, domainEndNormalized);
            return domain.slice(domainStart, domainEnd);
        };
        Category.prototype.range = function (values) {
            return _super.prototype.range.call(this, values);
        };
        Category._convertToPlottableInnerPadding = function (d3InnerPadding) {
            return 1 / (1 - d3InnerPadding) - 1;
        };
        Category._convertToPlottableOuterPadding = function (d3OuterPadding, d3InnerPadding) {
            return d3OuterPadding / (1 - d3InnerPadding);
        };
        Category.prototype._setBands = function () {
            var d3InnerPadding = 1 - 1 / (1 + this.innerPadding());
            var d3OuterPadding = this.outerPadding() / (1 + this.innerPadding());
            this._d3Scale.paddingInner(d3InnerPadding);
            this._d3Scale.paddingOuter(d3OuterPadding);
        };
        /**
         * Returns the width of the range band.
         *
         * @returns {number} The range band width
         */
        Category.prototype.rangeBand = function () {
            return this._rescaleBand(this._d3Scale.bandwidth());
        };
        /**
         * Returns the step width of the scale.
         *
         * The step width is the pixel distance between adjacent values in the domain.
         *
         * @returns {number}
         */
        Category.prototype.stepWidth = function () {
            // todo consider replacing this with _d3Scale.step()
            return this._rescaleBand(this._d3Scale.bandwidth() * (1 + this.innerPadding()));
        };
        Category.prototype.ticks = function () {
            return this.domain();
        };
        Category.prototype.innerPadding = function (innerPadding) {
            if (innerPadding == null) {
                return this._innerPadding;
            }
            this._innerPadding = innerPadding;
            this.range(this.range());
            this._dispatchUpdate();
            return this;
        };
        Category.prototype.outerPadding = function (outerPadding) {
            if (outerPadding == null) {
                return this._outerPadding;
            }
            this._outerPadding = outerPadding;
            this.range(this.range());
            this._dispatchUpdate();
            return this;
        };
        Category.prototype.scale = function (value) {
            // Determine the middle of the range band for the value
            var untransformed = this._d3Scale(value) + this._d3Scale.bandwidth() / 2;
            // Convert to screen space
            return this._d3TransformationScale(untransformed);
        };
        Category.prototype.zoom = function (magnifyAmount, centerValue) {
            var _this = this;
            var magnifyTransform = function (rangeValue) {
                return _this._d3TransformationScale.invert(panZoomConstraints.zoomOut(rangeValue, magnifyAmount, centerValue));
            };
            this._d3TransformationScale.domain(this._d3TransformationScale.range().map(magnifyTransform));
            this._dispatchUpdate();
        };
        Category.prototype.pan = function (translateAmount) {
            var _this = this;
            var translateTransform = function (rangeValue) {
                return _this._d3TransformationScale.invert(rangeValue + translateAmount);
            };
            this._d3TransformationScale.domain(this._d3TransformationScale.range().map(translateTransform));
            this._dispatchUpdate();
        };
        Category.prototype.scaleTransformation = function (value) {
            return this._d3TransformationScale(value);
        };
        Category.prototype.invertedTransformation = function (value) {
            return this._d3TransformationScale.invert(value);
        };
        Category.prototype.getTransformationExtent = function () {
            return TRANSFORMATION_SPACE;
        };
        Category.prototype.getTransformationDomain = function () {
            return this._d3TransformationScale.domain();
        };
        Category.prototype.setTransformationDomain = function (domain) {
            this._d3TransformationScale.domain(domain);
            this._dispatchUpdate();
        };
        Category.prototype._getDomain = function () {
            return this._backingScaleDomain();
        };
        Category.prototype._backingScaleDomain = function (values) {
            if (values == null) {
                return this._d3Scale.domain();
            }
            else {
                this._d3Scale.domain(values);
                this._setBands();
                return this;
            }
        };
        Category.prototype._getRange = function () {
            return this._range;
        };
        Category.prototype._setRange = function (values) {
            this._range = values;
            this._d3TransformationScale.range(values);
            this._setBands();
        };
        /**
         * Converts a width or height in *Transformation Space* into *Screen Space*.
         */
        Category.prototype._rescaleBand = function (band) {
            return Math.abs(this._d3TransformationScale(band) - this._d3TransformationScale(0));
        };
        return Category;
    }(scale.Scale));
    exports.Category = Category;
    });

    var colorScale = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });




    /**
     * Workaround for bad d3 behavior.
     *
     * d3's color scales, which are oridinal scales, have side-effects when invoked.
     * When you call the `scale(value)` as a function, it will implicitly add new
     * values to the domain.
     *
     * These side-effects cause us to rely on looking up the current scale state
     * with the `.domain()` and `.range()` methods. However, these methods have poor
     * performance implications since they will always slice their internal values
     * before returning. Inside the inner render loop of color scatter plot points,
     * these slices add up.
     */
    var ImplicitSeriesTracker = /** @class */ (function () {
        function ImplicitSeriesTracker() {
            this.count = 0;
            this.tracker = {};
        }
        ImplicitSeriesTracker.prototype.getIndex = function (value) {
            if (this.tracker[value] != null) {
                return this.tracker[value];
            }
            else {
                var idx = this.count;
                this.tracker[value] = idx;
                this.count += 1;
                return idx;
            }
        };
        ImplicitSeriesTracker.prototype.clear = function () {
            this.count = 0;
            this.tracker = {};
        };
        return ImplicitSeriesTracker;
    }());
    var Color = /** @class */ (function (_super) {
        tslib_es6.__extends(Color, _super);
        /**
         * A Color Scale maps string values to color hex values expressed as a string.
         *
         * @constructor
         * @param {string} [scaleType] One of "Category10"/"Category20"/"Category20b"/"Category20c".
         *   (see https://github.com/mbostock/d3/wiki/Ordinal-Scales#categorical-colors)
         *   If not supplied, reads the colors defined using CSS -- see plottable.css.
         */
        function Color(scaleType) {
            var _this = _super.call(this) || this;
            // Cache the number of range value to avoid calling `scale.range().length`
            _this._rangeLength = 1;
            _this._tracker = new ImplicitSeriesTracker();
            var scale;
            switch (scaleType) {
                case null:
                case undefined:
                    if (Color._plottableColorCache == null) {
                        Color._plottableColorCache = Color._getPlottableColors();
                    }
                    scale = d3.scaleOrdinal().range(Color._plottableColorCache);
                    break;
                case "Category10":
                case "category10":
                case "10":
                    scale = d3.scaleOrdinal(d3.schemeCategory10);
                    break;
                case "Category20":
                case "category20":
                case "20":
                    scale = d3.scaleOrdinal(d3.schemeCategory20);
                    break;
                case "Category20b":
                case "category20b":
                case "20b":
                    scale = d3.scaleOrdinal(d3.schemeCategory20b);
                    break;
                case "Category20c":
                case "category20c":
                case "20c":
                    scale = d3.scaleOrdinal(d3.schemeCategory20c);
                    break;
                default:
                    throw new Error("Unsupported ColorScale type");
            }
            _this._d3Scale = scale;
            _this._rangeLength = _this._d3Scale.range().length;
            return _this;
        }
        Color.prototype.extentOfValues = function (values) {
            return utils$1.Array.uniq(values);
        };
        // Duplicated from OrdinalScale._getExtent - should be removed in #388
        Color.prototype._getExtent = function () {
            return utils$1.Array.uniq(this._getAllIncludedValues());
        };
        Color.invalidateColorCache = function () {
            Color._plottableColorCache = null;
        };
        Color._getPlottableColors = function () {
            var plottableDefaultColors = [];
            var colorTester = d3.select("body").append("plottable-color-tester");
            var defaultColorHex = utils$1.Color.colorTest(colorTester, "");
            var i = 0;
            var colorHex = utils$1.Color.colorTest(colorTester, "plottable-colors-0");
            while (colorHex != null && i < this._MAXIMUM_COLORS_FROM_CSS) {
                if (colorHex === defaultColorHex && colorHex === plottableDefaultColors[plottableDefaultColors.length - 1]) {
                    break;
                }
                plottableDefaultColors.push(colorHex);
                i++;
                colorHex = utils$1.Color.colorTest(colorTester, "plottable-colors-" + i);
            }
            colorTester.remove();
            return plottableDefaultColors;
        };
        /**
         * Returns the color-string corresponding to a given string.
         *
         * If there are not enough colors in the range(), a lightened version of an
         * existing color will be used.
         *
         * @param {string} value
         * @returns {string}
         */
        Color.prototype.scale = function (value) {
            var color = this._d3Scale(value);
            var index = this._tracker.getIndex(value);
            var numLooped = Math.floor(index / this._rangeLength);
            if (numLooped === 0) {
                return color;
            }
            var modifyFactor = Math.log(numLooped * Color._LOOP_LIGHTEN_FACTOR + 1);
            return utils$1.Color.lightenColor(color, modifyFactor);
        };
        Color.prototype._getDomain = function () {
            return this._backingScaleDomain();
        };
        Color.prototype._backingScaleDomain = function (values) {
            if (values == null) {
                return this._d3Scale.domain();
            }
            else {
                this._d3Scale.domain(values);
                this._tracker.clear();
                return this;
            }
        };
        Color.prototype._getRange = function () {
            return this._d3Scale.range();
        };
        Color.prototype._setRange = function (values) {
            this._d3Scale.range(values);
            this._rangeLength = values.length;
        };
        Color._LOOP_LIGHTEN_FACTOR = 1.6;
        // The maximum number of colors we are getting from CSS stylesheets
        Color._MAXIMUM_COLORS_FROM_CSS = 256;
        return Color;
    }(scale.Scale));
    exports.Color = Color;
    });

    var interpolatedColorScale = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });




    var InterpolatedColor = /** @class */ (function (_super) {
        tslib_es6.__extends(InterpolatedColor, _super);
        /**
         * An InterpolatedColor Scale maps numbers to color hex values, expressed as strings.
         *
         * @param {string} [scaleType="linear"] One of "linear"/"log"/"sqrt"/"pow".
         */
        function InterpolatedColor(scaleType) {
            if (scaleType === void 0) { scaleType = "linear"; }
            var _this = _super.call(this) || this;
            switch (scaleType) {
                case "linear":
                    _this._colorScale = d3.scaleLinear();
                    break;
                case "log":
                    _this._colorScale = d3.scaleLog();
                    break;
                case "sqrt":
                    _this._colorScale = d3.scaleSqrt();
                    break;
                case "pow":
                    _this._colorScale = d3.scalePow();
                    break;
            }
            if (_this._colorScale == null) {
                throw new Error("unknown QuantitativeScale scale type " + scaleType);
            }
            _this.range(InterpolatedColor.REDS);
            return _this;
        }
        InterpolatedColor.prototype.extentOfValues = function (values) {
            var extent = d3.extent(values);
            if (extent[0] == null || extent[1] == null) {
                return [];
            }
            else {
                return extent;
            }
        };
        /**
         * Generates the converted QuantitativeScale.
         */
        InterpolatedColor.prototype._d3InterpolatedScale = function () {
            return this._colorScale.range([0, 1]).interpolate(this._interpolateColors());
        };
        /**
         * Generates the d3 interpolator for colors.
         */
        InterpolatedColor.prototype._interpolateColors = function () {
            var colors = this._colorRange;
            if (colors.length < 2) {
                throw new Error("Color scale arrays must have at least two elements.");
            }
            return function (a, b) {
                return function (t) {
                    // Clamp t parameter to [0,1]
                    t = Math.max(0, Math.min(1, t));
                    // Determine indices for colors
                    var tScaled = t * (colors.length - 1);
                    var i0 = Math.floor(tScaled);
                    var i1 = Math.ceil(tScaled);
                    var frac = (tScaled - i0);
                    // Interpolate in the L*a*b color space
                    return d3.interpolateLab(colors[i0], colors[i1])(frac);
                };
            };
        };
        InterpolatedColor.prototype._resetScale = function () {
            this._d3Scale = this._d3InterpolatedScale();
            this.autoDomainIfAutomaticMode();
            this._dispatchUpdate();
        };
        InterpolatedColor.prototype.autoDomain = function () {
            // InterpolatedColorScales do not pad
            var includedValues = this._getAllIncludedValues();
            if (includedValues.length > 0) {
                this._setDomain([utils$1.Math.min(includedValues, 0), utils$1.Math.max(includedValues, 0)]);
            }
            return this;
        };
        InterpolatedColor.prototype.scale = function (value) {
            return this._d3Scale(value);
        };
        InterpolatedColor.prototype._getDomain = function () {
            return this._backingScaleDomain();
        };
        InterpolatedColor.prototype._backingScaleDomain = function (values) {
            if (values == null) {
                return this._d3Scale.domain();
            }
            else {
                this._d3Scale.domain(values);
                return this;
            }
        };
        InterpolatedColor.prototype._getRange = function () {
            return this._colorRange;
        };
        InterpolatedColor.prototype._setRange = function (range) {
            this._colorRange = range;
            this._resetScale();
        };
        InterpolatedColor.REDS = [
            "#FFFFFF",
            "#FFF6E1",
            "#FEF4C0",
            "#FED976",
            "#FEB24C",
            "#FD8D3C",
            "#FC4E2A",
            "#E31A1C",
            "#B10026",
        ];
        InterpolatedColor.BLUES = [
            "#FFFFFF",
            "#CCFFFF",
            "#A5FFFD",
            "#85F7FB",
            "#6ED3EF",
            "#55A7E0",
            "#417FD0",
            "#2545D3",
            "#0B02E1",
        ];
        InterpolatedColor.POSNEG = [
            "#0B02E1",
            "#2545D3",
            "#417FD0",
            "#55A7E0",
            "#6ED3EF",
            "#85F7FB",
            "#A5FFFD",
            "#CCFFFF",
            "#FFFFFF",
            "#FFF6E1",
            "#FEF4C0",
            "#FED976",
            "#FEB24C",
            "#FD8D3C",
            "#FC4E2A",
            "#E31A1C",
            "#B10026",
        ];
        return InterpolatedColor;
    }(scale.Scale));
    exports.InterpolatedColor = InterpolatedColor;
    });

    var quantitativeScale = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });





    var QuantitativeScale = /** @class */ (function (_super) {
        tslib_es6.__extends(QuantitativeScale, _super);
        /**
         * A QuantitativeScale is a Scale that maps number-like values to numbers.
         * It is invertible and continuous.
         *
         * @constructor
         */
        function QuantitativeScale() {
            var _this = _super.call(this) || this;
            _this._tickGenerator = function (scale) { return scale.defaultTicks(); };
            _this._padProportion = 0.05;
            _this._snappingDomainEnabled = true;
            _this._paddingExceptionsProviders = new utils$1.Set();
            return _this;
        }
        QuantitativeScale.prototype.autoDomain = function () {
            this._domainMin = null;
            this._domainMax = null;
            _super.prototype.autoDomain.call(this);
            return this;
        };
        QuantitativeScale.prototype.autoDomainIfAutomaticMode = function () {
            if (this._domainMin != null && this._domainMax != null) {
                this._setDomain([this._domainMin, this._domainMax]);
                return;
            }
            var computedExtent = this._getExtent();
            if (this._domainMin != null) {
                var maxValue = computedExtent[1];
                if (this._domainMin >= maxValue) {
                    maxValue = this._expandSingleValueDomain([this._domainMin, this._domainMin])[1];
                }
                this._setDomain([this._domainMin, maxValue]);
                return;
            }
            if (this._domainMax != null) {
                var minValue = computedExtent[0];
                if (this._domainMax <= minValue) {
                    minValue = this._expandSingleValueDomain([this._domainMax, this._domainMax])[0];
                }
                this._setDomain([minValue, this._domainMax]);
                return;
            }
            _super.prototype.autoDomainIfAutomaticMode.call(this);
        };
        QuantitativeScale.prototype._getUnboundedExtent = function (ignoreAttachState) {
            if (ignoreAttachState === void 0) { ignoreAttachState = false; }
            var includedValues = this._getAllIncludedValues(ignoreAttachState);
            var extent = this._defaultExtent();
            if (includedValues.length !== 0) {
                var combinedExtent = [
                    utils$1.Math.min(includedValues, extent[0]),
                    utils$1.Math.max(includedValues, extent[1]),
                ];
                extent = this._padDomain(combinedExtent);
            }
            return extent;
        };
        QuantitativeScale.prototype._getExtent = function () {
            var extent = this._getUnboundedExtent();
            if (this._domainMin != null) {
                extent[0] = this._domainMin;
            }
            if (this._domainMax != null) {
                extent[1] = this._domainMax;
            }
            return extent;
        };
        /**
         * Adds a padding exception provider.
         * If one end of the domain is set to an excepted value as a result of autoDomain()-ing,
         * that end of the domain will not be padded.
         *
         * @param {Scales.PaddingExceptionProvider<D>} provider The provider function.
         * @returns {QuantitativeScale} The calling QuantitativeScale.
         */
        QuantitativeScale.prototype.addPaddingExceptionsProvider = function (provider) {
            this._paddingExceptionsProviders.add(provider);
            this.autoDomainIfAutomaticMode();
            return this;
        };
        /**
         * Removes the padding exception provider.
         *
         * @param {Scales.PaddingExceptionProvider<D>} provider The provider function.
         * @returns {QuantitativeScale} The calling QuantitativeScale.
         */
        QuantitativeScale.prototype.removePaddingExceptionsProvider = function (provider) {
            this._paddingExceptionsProviders.delete(provider);
            this.autoDomainIfAutomaticMode();
            return this;
        };
        QuantitativeScale.prototype.padProportion = function (padProportion) {
            if (padProportion == null) {
                return this._padProportion;
            }
            if (padProportion < 0) {
                throw new Error("padProportion must be non-negative");
            }
            this._padProportion = padProportion;
            this.autoDomainIfAutomaticMode();
            return this;
        };
        QuantitativeScale.prototype._padDomain = function (domain) {
            var _this = this;
            if (domain[0].valueOf() === domain[1].valueOf()) {
                return this._expandSingleValueDomain(domain);
            }
            if (this._padProportion === 0) {
                return domain;
            }
            var p = this._padProportion / 2;
            var min = domain[0];
            var max = domain[1];
            var minExistsInExceptions = false;
            var maxExistsInExceptions = false;
            this._paddingExceptionsProviders.forEach(function (provider) {
                var values = provider(_this);
                values.forEach(function (value) {
                    if (value.valueOf() === min.valueOf()) {
                        minExistsInExceptions = true;
                    }
                    if (value.valueOf() === max.valueOf()) {
                        maxExistsInExceptions = true;
                    }
                });
            });
            var originalDomain = this._backingScaleDomain();
            this._backingScaleDomain(domain);
            var newMin = minExistsInExceptions ? min : this.invert(this.scale(min) - (this.scale(max) - this.scale(min)) * p);
            var newMax = maxExistsInExceptions ? max : this.invert(this.scale(max) + (this.scale(max) - this.scale(min)) * p);
            this._backingScaleDomain(originalDomain);
            if (this._snappingDomainEnabled) {
                return this._niceDomain([newMin, newMax]);
            }
            return ([newMin, newMax]);
        };
        QuantitativeScale.prototype.snappingDomainEnabled = function (snappingDomainEnabled) {
            if (snappingDomainEnabled == null) {
                return this._snappingDomainEnabled;
            }
            this._snappingDomainEnabled = snappingDomainEnabled;
            this.autoDomainIfAutomaticMode();
            return this;
        };
        QuantitativeScale.prototype._expandSingleValueDomain = function (singleValueDomain) {
            return singleValueDomain;
        };
        /**
         * Computes the domain value corresponding to a supplied range value.
         *
         * @param {number} value: A value from the Scale's range.
         * @returns {D} The domain value corresponding to the supplied range value.
         */
        QuantitativeScale.prototype.invert = function (value) {
            throw new Error("Subclasses should override invert");
        };
        QuantitativeScale.prototype.domain = function (values) {
            if (values != null) {
                this._domainMin = values[0];
                this._domainMax = values[1];
            }
            return _super.prototype.domain.call(this, values);
        };
        QuantitativeScale.prototype.domainMin = function (domainMin) {
            if (domainMin == null) {
                return this.domain()[0];
            }
            this._domainMin = domainMin;
            this.autoDomainIfAutomaticMode();
            return this;
        };
        QuantitativeScale.prototype.domainMax = function (domainMax) {
            if (domainMax == null) {
                return this.domain()[1];
            }
            this._domainMax = domainMax;
            this.autoDomainIfAutomaticMode();
            return this;
        };
        QuantitativeScale.prototype.extentOfValues = function (values) {
            // HACKHACK: TS1.4 doesn't consider numbers to be Number-like (valueOf() returning number), so D can't be typed correctly
            var extent = d3.extent(values.filter(function (value) { return utils$1.Math.isValidNumber(+value); }));
            if (extent[0] == null || extent[1] == null) {
                return [];
            }
            else {
                return extent;
            }
        };
        QuantitativeScale.prototype.zoom = function (magnifyAmount, centerValue) {
            var _this = this;
            var magnifyTransform = function (rangeValue) { return _this.invert(panZoomConstraints.zoomOut(rangeValue, magnifyAmount, centerValue)); };
            this.domain(this.range().map(magnifyTransform));
        };
        QuantitativeScale.prototype.pan = function (translateAmount) {
            var _this = this;
            var translateTransform = function (rangeValue) { return _this.invert(rangeValue + translateAmount); };
            this.domain(this.range().map(translateTransform));
        };
        QuantitativeScale.prototype.scaleTransformation = function (value) {
            throw new Error("Subclasses should override scaleTransformation");
        };
        QuantitativeScale.prototype.invertedTransformation = function (value) {
            throw new Error("Subclasses should override invertedTransformation");
        };
        QuantitativeScale.prototype.getTransformationExtent = function () {
            throw new Error("Subclasses should override getTransformationExtent");
        };
        QuantitativeScale.prototype.getTransformationDomain = function () {
            throw new Error("Subclasses should override getTransformationDomain");
        };
        QuantitativeScale.prototype.setTransformationDomain = function (domain) {
            throw new Error("Subclasses should override setTransformationDomain");
        };
        QuantitativeScale.prototype._setDomain = function (values) {
            var isNaNOrInfinity = function (x) { return utils$1.Math.isNaN(x) || x === Infinity || x === -Infinity; };
            if (isNaNOrInfinity(values[0]) || isNaNOrInfinity(values[1])) {
                utils$1.Window.warn("Warning: QuantitativeScales cannot take NaN or Infinity as a domain value. Ignoring.");
                return;
            }
            _super.prototype._setDomain.call(this, values);
        };
        /**
         * Gets the array of tick values generated by the default algorithm.
         */
        QuantitativeScale.prototype.defaultTicks = function () {
            throw new Error("Subclasses should override _getDefaultTicks");
        };
        /**
         * Gets an array of tick values spanning the domain.
         *
         * @returns {D[]}
         */
        QuantitativeScale.prototype.ticks = function () {
            return this._tickGenerator(this);
        };
        /**
         * Given a domain, expands its domain onto "nice" values, e.g. whole
         * numbers.
         */
        QuantitativeScale.prototype._niceDomain = function (domain, count) {
            throw new Error("Subclasses should override _niceDomain");
        };
        QuantitativeScale.prototype._defaultExtent = function () {
            throw new Error("Subclasses should override _defaultExtent");
        };
        QuantitativeScale.prototype.tickGenerator = function (generator) {
            if (generator == null) {
                return this._tickGenerator;
            }
            else {
                this._tickGenerator = generator;
                return this;
            }
        };
        QuantitativeScale._DEFAULT_NUM_TICKS = 10;
        return QuantitativeScale;
    }(scale.Scale));
    exports.QuantitativeScale = QuantitativeScale;
    });

    var linearScale = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });



    var Linear = /** @class */ (function (_super) {
        tslib_es6.__extends(Linear, _super);
        /**
         * @constructor
         */
        function Linear() {
            var _this = _super.call(this) || this;
            _this._d3Scale = d3.scaleLinear();
            return _this;
        }
        Linear.prototype._defaultExtent = function () {
            return [0, 1];
        };
        Linear.prototype._expandSingleValueDomain = function (singleValueDomain) {
            if (singleValueDomain[0] === singleValueDomain[1]) {
                return [singleValueDomain[0] - 1, singleValueDomain[1] + 1];
            }
            return singleValueDomain;
        };
        Linear.prototype.scale = function (value) {
            return this._d3Scale(value);
        };
        Linear.prototype.scaleTransformation = function (value) {
            return this.scale(value);
        };
        Linear.prototype.invertedTransformation = function (value) {
            return this.invert(value);
        };
        Linear.prototype.getTransformationExtent = function () {
            return this._getUnboundedExtent(true);
        };
        Linear.prototype.getTransformationDomain = function () {
            return this.domain();
        };
        Linear.prototype.setTransformationDomain = function (domain) {
            this.domain(domain);
        };
        Linear.prototype._getDomain = function () {
            return this._backingScaleDomain();
        };
        Linear.prototype._backingScaleDomain = function (values) {
            if (values == null) {
                return this._d3Scale.domain();
            }
            else {
                this._d3Scale.domain(values);
                return this;
            }
        };
        Linear.prototype._getRange = function () {
            return this._d3Scale.range();
        };
        Linear.prototype._setRange = function (values) {
            this._d3Scale.range(values);
        };
        Linear.prototype.invert = function (value) {
            return this._d3Scale.invert(value);
        };
        Linear.prototype.defaultTicks = function () {
            return this._d3Scale.ticks(Linear._DEFAULT_NUM_TICKS);
        };
        Linear.prototype._niceDomain = function (domain, count) {
            return this._d3Scale.copy().domain(domain).nice(count).domain();
        };
        return Linear;
    }(quantitativeScale.QuantitativeScale));
    exports.Linear = Linear;
    });

    var logScale = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });



    var Log = /** @class */ (function (_super) {
        tslib_es6.__extends(Log, _super);
        /**
         * @constructor
         */
        function Log(base) {
            if (base === void 0) { base = 10; }
            var _this = _super.call(this) || this;
            _this._d3Scale = d3.scaleLog().base(base);
            _this._setDomain(_this._defaultExtent());
            return _this;
        }
        Log.prototype._defaultExtent = function () {
            return [1, this._d3Scale.base()];
        };
        Log.prototype._expandSingleValueDomain = function (singleValueDomain) {
            if (singleValueDomain[0] === singleValueDomain[1]) {
                return [singleValueDomain[0] / this._d3Scale.base(),
                    singleValueDomain[1] * this._d3Scale.base()];
            }
            return singleValueDomain;
        };
        Log.prototype.scale = function (value) {
            return this._d3Scale(value);
        };
        Log.prototype.scaleTransformation = function (value) {
            return this.scale(value);
        };
        Log.prototype.invertedTransformation = function (value) {
            return this.invert(value);
        };
        Log.prototype.getTransformationExtent = function () {
            return this._getUnboundedExtent(true);
        };
        Log.prototype.getTransformationDomain = function () {
            return this.domain();
        };
        Log.prototype.setTransformationDomain = function (domain) {
            this.domain(domain);
        };
        Log.prototype._getDomain = function () {
            return this._backingScaleDomain();
        };
        Log.prototype._backingScaleDomain = function (values) {
            if (values == null) {
                return this._d3Scale.domain();
            }
            else {
                this._d3Scale.domain(values);
                return this;
            }
        };
        Log.prototype._getRange = function () {
            return this._d3Scale.range();
        };
        Log.prototype._setRange = function (values) {
            this._d3Scale.range(values);
        };
        Log.prototype.invert = function (value) {
            return this._d3Scale.invert(value);
        };
        Log.prototype.defaultTicks = function () {
            return this._d3Scale.ticks(Log._DEFAULT_NUM_TICKS);
        };
        Log.prototype._niceDomain = function (domain, count) {
            return this._d3Scale.copy().domain(domain).nice().domain();
        };
        return Log;
    }(quantitativeScale.QuantitativeScale));
    exports.Log = Log;
    });

    var modifiedLogScale = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });





    var ModifiedLog = /** @class */ (function (_super) {
        tslib_es6.__extends(ModifiedLog, _super);
        /**
         * A ModifiedLog Scale acts as a regular log scale for large numbers.
         * As it approaches 0, it gradually becomes linear.
         * Consequently, a ModifiedLog Scale can process 0 and negative numbers.
         *
         * For x >= base, scale(x) = log(x).
         *
         * For 0 < x < base, scale(x) will become more and more
         * linear as it approaches 0.
         *
         * At x == 0, scale(x) == 0.
         *
         * For negative values, scale(-x) = -scale(x).
         *
         * The range and domain for the scale should also be set, using the
         * range() and domain() accessors, respectively.
         *
         * For `range`, provide a two-element array giving the minimum and
         * maximum of values produced when scaling.
         *
         * For `domain` provide a two-element array giving the minimum and
         * maximum of the values that will be scaled.
         *
         * @constructor
         * @param {number} [base=10]
         *        The base of the log. Must be > 1.
         *
         */
        function ModifiedLog(base) {
            if (base === void 0) { base = 10; }
            var _this = _super.call(this) || this;
            _this._logTickGenerator = function (scale) {
                // Say your domain is [-100, 100] and your pivot is 10.
                // then we're going to draw negative log ticks from -100 to -10,
                // linear ticks from -10 to 10, and positive log ticks from 10 to 100.
                var middle = function (x, y, z) { return [x, y, z].sort(function (a, b) { return a - b; })[1]; };
                var min = utils$1.Math.min(_this._untransformedDomain, 0);
                var max = utils$1.Math.max(_this._untransformedDomain, 0);
                var negativeLower = min;
                var negativeUpper = middle(min, max, -_this._pivot);
                var positiveLower = middle(min, max, _this._pivot);
                var positiveUpper = max;
                var negativeLogTicks = _this._logTicks(-negativeUpper, -negativeLower).map(function (x) { return -x; }).reverse();
                var positiveLogTicks = _this._logTicks(positiveLower, positiveUpper);
                var linearMin = Math.max(min, -_this._pivot);
                var linearMax = Math.min(max, _this._pivot);
                var linearTicks = d3.scaleLinear().domain([linearMin, linearMax]).ticks(_this._howManyTicks(linearMin, linearMax));
                var ticks = negativeLogTicks.concat(linearTicks).concat(positiveLogTicks);
                // If you only have 1 tick, you can't tell how big the scale is.
                if (ticks.length <= 1) {
                    ticks = d3.scaleLinear().domain([min, max]).ticks(scales.ModifiedLog._DEFAULT_NUM_TICKS);
                }
                return ticks;
            };
            _this._d3Scale = d3.scaleLinear();
            _this._base = base;
            _this._pivot = _this._base;
            _this._setDomain(_this._defaultExtent());
            _this.tickGenerator(_this._logTickGenerator);
            if (base <= 1) {
                throw new Error("ModifiedLogScale: The base must be > 1");
            }
            return _this;
        }
        /**
         * Returns an adjusted log10 value for graphing purposes.  The first
         * adjustment is that negative values are changed to positive during
         * the calculations, and then the answer is negated at the end.  The
         * second is that, for values less than 10, an increasingly large
         * (0 to 1) scaling factor is added such that at 0 the value is
         * adjusted to 1, resulting in a returned result of 0.
         */
        ModifiedLog.prototype._adjustedLog = function (x) {
            var negationFactor = x < 0 ? -1 : 1;
            x *= negationFactor;
            if (x < this._pivot) {
                x += (this._pivot - x) / this._pivot;
            }
            x = Math.log(x) / Math.log(this._base);
            x *= negationFactor;
            return x;
        };
        ModifiedLog.prototype._invertedAdjustedLog = function (x) {
            var negationFactor = x < 0 ? -1 : 1;
            x *= negationFactor;
            x = Math.pow(this._base, x);
            if (x < this._pivot) {
                x = (this._pivot * (x - 1)) / (this._pivot - 1);
            }
            x *= negationFactor;
            return x;
        };
        ModifiedLog.prototype.scale = function (x) {
            return this._d3Scale(this._adjustedLog(x));
        };
        ModifiedLog.prototype.invert = function (x) {
            return this._invertedAdjustedLog(this._d3Scale.invert(x));
        };
        ModifiedLog.prototype.scaleTransformation = function (value) {
            return this.scale(value);
        };
        ModifiedLog.prototype.invertedTransformation = function (value) {
            return this.invert(value);
        };
        ModifiedLog.prototype.getTransformationExtent = function () {
            return this._getUnboundedExtent(true);
        };
        ModifiedLog.prototype.getTransformationDomain = function () {
            return this.domain();
        };
        ModifiedLog.prototype.setTransformationDomain = function (domain) {
            this.domain(domain);
        };
        ModifiedLog.prototype._getDomain = function () {
            return this._untransformedDomain;
        };
        ModifiedLog.prototype._setDomain = function (values) {
            this._untransformedDomain = values;
            var transformedDomain = [this._adjustedLog(values[0]), this._adjustedLog(values[1])];
            _super.prototype._setDomain.call(this, transformedDomain);
        };
        ModifiedLog.prototype._backingScaleDomain = function (values) {
            if (values == null) {
                return this._d3Scale.domain();
            }
            else {
                this._d3Scale.domain(values);
                return this;
            }
        };
        /**
         * Return an appropriate number of ticks from lower to upper.
         *
         * This will first try to fit as many powers of this.base as it can from
         * lower to upper.
         *
         * If it still has ticks after that, it will generate ticks in "clusters",
         * e.g. [20, 30, ... 90, 100] would be a cluster, [200, 300, ... 900, 1000]
         * would be another cluster.
         *
         * This function will generate clusters as large as it can while not
         * drastically exceeding its number of ticks.
         */
        ModifiedLog.prototype._logTicks = function (lower, upper) {
            var _this = this;
            var nTicks = this._howManyTicks(lower, upper);
            if (nTicks === 0) {
                return [];
            }
            var startLogged = Math.floor(Math.log(lower) / Math.log(this._base));
            var endLogged = Math.ceil(Math.log(upper) / Math.log(this._base));
            var bases = d3.range(endLogged, startLogged, -Math.ceil((endLogged - startLogged) / nTicks));
            var multiples = d3.range(this._base, 1, -(this._base - 1)).map(Math.floor);
            var uniqMultiples = utils$1.Array.uniq(multiples);
            var clusters = bases.map(function (b) { return uniqMultiples.map(function (x) { return Math.pow(_this._base, b - 1) * x; }); });
            var flattened = utils$1.Array.flatten(clusters);
            var filtered = flattened.filter(function (x) { return lower <= x && x <= upper; });
            var sorted = filtered.sort(function (x, y) { return x - y; });
            return sorted;
        };
        /**
         * How many ticks does the range [lower, upper] deserve?
         *
         * e.g. if your domain was [10, 1000] and I asked _howManyTicks(10, 100),
         * I would get 1/2 of the ticks. The range 10, 100 takes up 1/2 of the
         * distance when plotted.
         */
        ModifiedLog.prototype._howManyTicks = function (lower, upper) {
            var adjustedMin = this._adjustedLog(utils$1.Math.min(this._untransformedDomain, 0));
            var adjustedMax = this._adjustedLog(utils$1.Math.max(this._untransformedDomain, 0));
            var adjustedLower = this._adjustedLog(lower);
            var adjustedUpper = this._adjustedLog(upper);
            var proportion = (adjustedUpper - adjustedLower) / (adjustedMax - adjustedMin);
            var ticks = Math.ceil(proportion * scales.ModifiedLog._DEFAULT_NUM_TICKS);
            return ticks;
        };
        ModifiedLog.prototype._niceDomain = function (domain, count) {
            return domain;
        };
        ModifiedLog.prototype._defaultExtent = function () {
            return [0, this._base];
        };
        ModifiedLog.prototype._expandSingleValueDomain = function (singleValueDomain) {
            if (singleValueDomain[0] === singleValueDomain[1]) {
                var singleValue = singleValueDomain[0];
                if (singleValue > 0) {
                    return [singleValue / this._base, singleValue * this._base];
                }
                else if (singleValue === 0) {
                    return [-this._base, this._base];
                }
                else {
                    return [singleValue * this._base, singleValue / this._base];
                }
            }
            return singleValueDomain;
        };
        ModifiedLog.prototype._getRange = function () {
            return this._d3Scale.range();
        };
        ModifiedLog.prototype._setRange = function (values) {
            this._d3Scale.range(values);
        };
        ModifiedLog.prototype.defaultTicks = function () {
            return this._d3Scale.ticks(scales.ModifiedLog._DEFAULT_NUM_TICKS);
        };
        return ModifiedLog;
    }(quantitativeScale.QuantitativeScale));
    exports.ModifiedLog = ModifiedLog;
    });

    var timeScale = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });




    var Time = /** @class */ (function (_super) {
        tslib_es6.__extends(Time, _super);
        /**
         * A Time Scale maps Date objects to numbers.
         *
         * @constructor
         */
        function Time() {
            var _this = _super.call(this) || this;
            _this._d3Scale = d3.scaleTime();
            _this.autoDomain();
            return _this;
        }
        /**
         * Returns an array of ticks values separated by the specified interval.
         *
         * @param {string} interval A string specifying the interval unit.
         * @param {number?} [step] The number of multiples of the interval between consecutive ticks.
         * @return {Date[]}
         */
        Time.prototype.tickInterval = function (interval, step, useUTC) {
            if (step === void 0) { step = 1; }
            if (useUTC === void 0) { useUTC = false; }
            // temporarily creats a time scale from our linear scale into a time scale so we can get access to its api
            var tempScale = d3.scaleTime();
            var d3Interval = Time.timeIntervalToD3Time(interval, useUTC).every(step);
            tempScale.domain(this.domain());
            tempScale.range(this.range());
            return tempScale.ticks(d3Interval);
        };
        Time.prototype._setDomain = function (values) {
            if (values[1] < values[0]) {
                throw new Error("Scale.Time domain values must be in chronological order");
            }
            return _super.prototype._setDomain.call(this, values);
        };
        Time.prototype._defaultExtent = function () {
            return [new Date("1970-01-01"), new Date("1970-01-02")];
        };
        Time.prototype._expandSingleValueDomain = function (singleValueDomain) {
            var startTime = singleValueDomain[0].getTime();
            var endTime = singleValueDomain[1].getTime();
            if (startTime === endTime) {
                var startDate = new Date(startTime);
                startDate.setDate(startDate.getDate() - 1);
                var endDate = new Date(endTime);
                endDate.setDate(endDate.getDate() + 1);
                return [startDate, endDate];
            }
            return singleValueDomain;
        };
        Time.prototype.scale = function (value) {
            return this._d3Scale(value);
        };
        Time.prototype.scaleTransformation = function (value) {
            return this.scale(new Date(value));
        };
        Time.prototype.invertedTransformation = function (value) {
            return this.invert(value).getTime();
        };
        Time.prototype.getTransformationExtent = function () {
            var extent = this._getUnboundedExtent(true);
            return [extent[0].valueOf(), extent[1].valueOf()];
        };
        Time.prototype.getTransformationDomain = function () {
            var dates = this.domain();
            return [dates[0].valueOf(), dates[1].valueOf()];
        };
        Time.prototype.setTransformationDomain = function (_a) {
            var domainMin = _a[0], domainMax = _a[1];
            this.domain([new Date(domainMin), new Date(domainMax)]);
        };
        Time.prototype._getDomain = function () {
            return this._backingScaleDomain();
        };
        Time.prototype._backingScaleDomain = function (values) {
            if (values == null) {
                return this._d3Scale.domain();
            }
            else {
                this._d3Scale.domain(values);
                return this;
            }
        };
        Time.prototype._getRange = function () {
            return this._d3Scale.range();
        };
        Time.prototype._setRange = function (values) {
            this._d3Scale.range(values);
        };
        Time.prototype.invert = function (value) {
            return this._d3Scale.invert(value);
        };
        Time.prototype.defaultTicks = function () {
            return this._d3Scale.ticks(Time._DEFAULT_NUM_TICKS);
        };
        Time.prototype._niceDomain = function (domain) {
            return this._d3Scale.copy().domain(domain).nice().domain();
        };
        /**
         * Transforms the Plottable TimeInterval string into a d3 time interval equivalent.
         * If the provided TimeInterval is incorrect, the default is d3.timeYear
         */
        Time.timeIntervalToD3Time = function (timeInterval, useUTC) {
            switch (timeInterval) {
                case timeAxis.TimeInterval.second:
                    return useUTC ? d3.utcSecond : d3.timeSecond;
                case timeAxis.TimeInterval.minute:
                    return useUTC ? d3.utcMinute : d3.timeMinute;
                case timeAxis.TimeInterval.hour:
                    return useUTC ? d3.utcHour : d3.timeHour;
                case timeAxis.TimeInterval.day:
                    return useUTC ? d3.utcDay : d3.timeDay;
                case timeAxis.TimeInterval.week:
                    return useUTC ? d3.utcWeek : d3.timeWeek;
                case timeAxis.TimeInterval.month:
                    return useUTC ? d3.utcMonth : d3.timeMonth;
                case timeAxis.TimeInterval.year:
                    return useUTC ? d3.utcYear : d3.timeYear;
                default:
                    throw Error("TimeInterval specified does not exist: " + timeInterval);
            }
        };
        return Time;
    }(quantitativeScale.QuantitativeScale));
    exports.Time = Time;
    });

    var scales = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });


    exports.TickGenerators = tickGenerators;
    tslib_es6.__exportStar(categoryScale, exports);
    tslib_es6.__exportStar(colorScale, exports);
    tslib_es6.__exportStar(interpolatedColorScale, exports);
    tslib_es6.__exportStar(linearScale, exports);
    tslib_es6.__exportStar(logScale, exports);
    tslib_es6.__exportStar(modifiedLogScale, exports);
    tslib_es6.__exportStar(timeScale, exports);
    // ---------------------------------------------------------
    var categoryScale_1 = categoryScale;

    /**
     * Type guarded function to check if the scale implements the
     * `TransformableScale` interface. Unfortunately, there is no way to do
     * runtime interface typechecking, so we have to explicitly list all classes
     * that implement the interface.
     */
    function isTransformable(scale) {
        return (scale instanceof quantitativeScale.QuantitativeScale ||
            scale instanceof categoryScale_1.Category);
    }
    exports.isTransformable = isTransformable;
    });

    var timeAxis = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });








    exports.TimeInterval = makeEnum_1.makeEnum([
        "second",
        "minute",
        "hour",
        "day",
        "week",
        "month",
        "year",
    ]);
    /**
     * Possible orientations for a Time Axis.
     */
    exports.TimeAxisOrientation = makeEnum_1.makeEnum(["top", "bottom"]);
    exports.TierLabelPosition = makeEnum_1.makeEnum(["between", "center"]);
    var Time = /** @class */ (function (_super) {
        tslib_es6.__extends(Time, _super);
        /**
         * Constructs a Time Axis.
         *
         * A Time Axis is a visual representation of a Time Scale.
         *
         * @constructor
         * @param {Scales.Time} scale
         * @param {AxisOrientation} orientation Orientation of this Time Axis. Time Axes can only have "top" or "bottom"
         * @param {boolean} useUTC Displays date object in UTC if true, local time if false. Defaults to false.
         * orientations.
         */
        function Time(scale, orientation, useUTC) {
            var _this = _super.call(this, scale, orientation) || this;
            _this._maxTimeIntervalPrecision = null;
            _this._tierLabelPositions = [];
            _this._useUTC = useUTC;
            _this.addClass("time-axis");
            _this.tickLabelPadding(5);
            _this.axisConfigurations(Time._DEFAULT_TIME_AXIS_CONFIGURATIONS(_this._useUTC));
            _this.annotationFormatter(formatters.time("%a %b %d, %Y", _this._useUTC));
            return _this;
        }
        Time.prototype.tierLabelPositions = function (newPositions) {
            if (newPositions == null) {
                return this._tierLabelPositions;
            }
            else {
                if (!newPositions.every(function (pos) { return pos.toLowerCase() === "between" || pos.toLowerCase() === "center"; })) {
                    throw new Error("Unsupported position for tier labels");
                }
                this._tierLabelPositions = newPositions;
                this.redraw();
                return this;
            }
        };
        Time.prototype.maxTimeIntervalPrecision = function (newPrecision) {
            if (newPrecision == null) {
                return this._maxTimeIntervalPrecision;
            }
            else {
                this._maxTimeIntervalPrecision = newPrecision;
                this.redraw();
                return this;
            }
        };
        /**
         * Returns the current `TimeAxisConfiguration` used to render the axes.
         *
         * Note that this is only valid after the axis had been rendered and the
         * most precise valid configuration is determined from the available space
         * and maximum precision constraints.
         *
         * @returns {TimeAxisConfiguration} The currently used `TimeAxisConfiguration` or `undefined`.
         */
        Time.prototype.currentAxisConfiguration = function () {
            return this._possibleTimeAxisConfigurations[this._mostPreciseConfigIndex];
        };
        Time.prototype.axisConfigurations = function (configurations) {
            if (configurations == null) {
                return this._possibleTimeAxisConfigurations;
            }
            this._possibleTimeAxisConfigurations = configurations;
            this._numTiers = utils$1.Math.max(this._possibleTimeAxisConfigurations.map(function (config) { return config.length; }), 0);
            if (this._isAnchored) {
                this._setupDomElements();
            }
            var oldLabelPositions = this.tierLabelPositions();
            var newLabelPositions = [];
            for (var i = 0; i < this._numTiers; i++) {
                newLabelPositions.push(oldLabelPositions[i] || "between");
            }
            this.tierLabelPositions(newLabelPositions);
            this.redraw();
            return this;
        };
        /**
         * Gets the index of the most precise TimeAxisConfiguration that will fit in the current width.
         */
        Time.prototype._getMostPreciseConfigurationIndex = function () {
            var _this = this;
            var mostPreciseIndex = this._possibleTimeAxisConfigurations.length;
            this._possibleTimeAxisConfigurations.forEach(function (interval, index) {
                if (index < mostPreciseIndex && interval.every(function (tier) {
                    return _this._checkTimeAxisTierConfiguration(tier);
                })) {
                    mostPreciseIndex = index;
                }
            });
            if (mostPreciseIndex === this._possibleTimeAxisConfigurations.length) {
                utils$1.Window.warn("zoomed out too far: could not find suitable interval to display labels");
                --mostPreciseIndex;
            }
            return mostPreciseIndex;
        };
        Time.prototype.orientation = function (orientation) {
            if (orientation && (orientation.toLowerCase() === "right" || orientation.toLowerCase() === "left")) {
                throw new Error(orientation + " is not a supported orientation for TimeAxis - only horizontal orientations are supported");
            }
            return _super.prototype.orientation.call(this, orientation); // maintains getter-setter functionality
        };
        Time.prototype._computeHeight = function () {
            var textHeight = this._measurer.measure().height;
            this._tierHeights = [];
            for (var i = 0; i < this._numTiers; i++) {
                this._tierHeights.push(textHeight + this.tickLabelPadding() +
                    ((this._tierLabelPositions[i]) === "between" ? 0 : this._maxLabelTickLength()));
            }
            return d3.sum(this._tierHeights);
        };
        Time.prototype._getIntervalLength = function (config) {
            var startDate = this._scale.domain()[0];
            var d3Interval = scales.Time.timeIntervalToD3Time(config.interval, this._useUTC);
            var endDate = d3Interval.offset(startDate, config.step);
            if (endDate > this._scale.domain()[1]) {
                // this offset is too large, so just return available width
                return this.width();
            }
            // measure how much space one date can get
            var stepLength = Math.abs(this._scale.scale(endDate) - this._scale.scale(startDate));
            return stepLength;
        };
        Time.prototype._maxWidthForInterval = function (config) {
            return this._measurer.measure(config.formatter(Time._LONG_DATE)).width;
        };
        /**
         * Check if tier configuration fits in the current width and satisfied the
         * max TimeInterval precision limit.
         */
        Time.prototype._checkTimeAxisTierConfiguration = function (config) {
            // Use the sorted index to determine if the teir configuration contains a
            // time interval that is too precise for the maxTimeIntervalPrecision
            // setting (if set).
            if (this._maxTimeIntervalPrecision != null) {
                var precisionLimit = Time._SORTED_TIME_INTERVAL_INDEX[this._maxTimeIntervalPrecision];
                var configPrecision = Time._SORTED_TIME_INTERVAL_INDEX[config.interval];
                if (precisionLimit != null && configPrecision != null && configPrecision < precisionLimit) {
                    return false;
                }
            }
            var worstWidth = this._maxWidthForInterval(config) + 2 * this.tickLabelPadding();
            return Math.min(this._getIntervalLength(config), this.width()) >= worstWidth;
        };
        Time.prototype._sizeFromOffer = function (availableWidth, availableHeight) {
            // Makes sure that the size it requires is a multiple of tier sizes, such that
            // we have no leftover tiers
            var size = _super.prototype._sizeFromOffer.call(this, availableWidth, availableHeight);
            var tierHeights = this._tierHeights.reduce(function (prevValue, currValue, index, arr) {
                return (prevValue + currValue > size.height) ? prevValue : (prevValue + currValue);
            });
            var nonCoreHeight = this.margin() + (this.annotationsEnabled() ? this.annotationTierCount() * this._annotationTierHeight() : 0);
            size.height = Math.min(size.height, tierHeights + nonCoreHeight);
            return size;
        };
        Time.prototype._setup = function () {
            _super.prototype._setup.call(this);
            this._setupDomElements();
        };
        Time.prototype._setupDomElements = function () {
            this.content().selectAll("." + Time.TIME_AXIS_TIER_CLASS).remove();
            this._tierLabelContainers = [];
            this._tierMarkContainers = [];
            this._tierBaselines = [];
            this._tickLabelContainer.remove();
            this._baseline.remove();
            for (var i = 0; i < this._numTiers; ++i) {
                var tierContainer = this.content().append("g").classed(Time.TIME_AXIS_TIER_CLASS, true);
                this._tierLabelContainers.push(tierContainer.append("g").classed(axis$2.Axis.TICK_LABEL_CLASS + "-container", true));
                this._tierMarkContainers.push(tierContainer.append("g").classed(axis$2.Axis.TICK_MARK_CLASS + "-container", true));
                this._tierBaselines.push(tierContainer.append("line").classed("baseline", true));
            }
            var context = new src$1.SvgContext(this._tierLabelContainers[0].node());
            this._measurer = new src$1.CacheMeasurer(context);
        };
        Time.prototype._getTickIntervalValues = function (config) {
            return this._scale.tickInterval(config.interval, config.step, this._useUTC);
        };
        Time.prototype._getTickValues = function () {
            var _this = this;
            return this._possibleTimeAxisConfigurations[this._mostPreciseConfigIndex].reduce(function (ticks, config) { return ticks.concat(_this._getTickIntervalValues(config)); }, []);
        };
        Time.prototype._cleanTiers = function () {
            for (var index = 0; index < this._tierLabelContainers.length; index++) {
                this._tierLabelContainers[index].selectAll("." + axis$2.Axis.TICK_LABEL_CLASS).remove();
                this._tierMarkContainers[index].selectAll("." + axis$2.Axis.TICK_MARK_CLASS).remove();
                this._tierBaselines[index].style("visibility", "hidden");
            }
        };
        Time.prototype._getTickValuesForConfiguration = function (config) {
            var tickPos = this._scale.tickInterval(config.interval, config.step, this._useUTC);
            var domain = this._scale.domain();
            var tickPosValues = tickPos.map(function (d) { return d.valueOf(); }); // can't indexOf with objects
            if (tickPosValues.indexOf(domain[0].valueOf()) === -1) {
                tickPos.unshift(domain[0]);
            }
            if (tickPosValues.indexOf(domain[1].valueOf()) === -1) {
                tickPos.push(domain[1]);
            }
            return tickPos;
        };
        Time.prototype._renderTierLabels = function (container, config, index) {
            var _this = this;
            var tickPos = this._getTickValuesForConfiguration(config);
            var labelPos = [];
            if (this._tierLabelPositions[index] === "between" && config.step === 1) {
                tickPos.map(function (datum, i) {
                    if (i + 1 >= tickPos.length) {
                        return;
                    }
                    labelPos.push(new Date((tickPos[i + 1].valueOf() - tickPos[i].valueOf()) / 2 + tickPos[i].valueOf()));
                });
            }
            else {
                labelPos = tickPos;
            }
            var tickLabelsUpdate = container.selectAll("." + axis$2.Axis.TICK_LABEL_CLASS).data(labelPos, function (d) { return String(d.valueOf()); });
            var tickLabelsEnter = tickLabelsUpdate
                .enter()
                .append("g")
                .classed(axis$2.Axis.TICK_LABEL_CLASS, true);
            tickLabelsEnter.append("text");
            var xTranslate = (this._tierLabelPositions[index] === "center" || config.step === 1) ? 0 : this.tickLabelPadding();
            var yTranslate;
            if (this.orientation() === "bottom") {
                yTranslate = d3.sum(this._tierHeights.slice(0, index + 1)) - this.tickLabelPadding();
            }
            else {
                if (this._tierLabelPositions[index] === "center") {
                    yTranslate = this.height() - d3.sum(this._tierHeights.slice(0, index)) - this.tickLabelPadding() - this._maxLabelTickLength();
                }
                else {
                    yTranslate = this.height() - d3.sum(this._tierHeights.slice(0, index)) - this.tickLabelPadding();
                }
            }
            var tickLabels = tickLabelsUpdate.merge(tickLabelsEnter);
            var textSelection = tickLabels.selectAll("text");
            if (textSelection.size() > 0) {
                textSelection.attr("transform", "translate(" + xTranslate + "," + yTranslate + ")");
            }
            tickLabelsUpdate.exit().remove();
            tickLabels.attr("transform", function (d) { return "translate(" + _this._scale.scale(d) + ",0)"; });
            var anchor = (this._tierLabelPositions[index] === "center" || config.step === 1) ? "middle" : "start";
            tickLabels.selectAll("text").text(config.formatter).style("text-anchor", anchor);
        };
        Time.prototype._renderTickMarks = function (tickValues, index) {
            var tickMarksUpdate = this._tierMarkContainers[index].selectAll("." + axis$2.Axis.TICK_MARK_CLASS).data(tickValues);
            var tickMarks = tickMarksUpdate
                .enter()
                .append("line")
                .classed(axis$2.Axis.TICK_MARK_CLASS, true)
                .merge(tickMarksUpdate);
            var attr = this._generateTickMarkAttrHash();
            var offset = this._tierHeights.slice(0, index).reduce(function (translate, height) { return translate + height; }, 0);
            if (this.orientation() === "bottom") {
                attr["y1"] = offset;
                attr["y2"] = offset + (this._tierLabelPositions[index] === "center" ? this.innerTickLength() : this._tierHeights[index]);
            }
            else {
                attr["y1"] = this.height() - offset;
                attr["y2"] = this.height() - (offset + (this._tierLabelPositions[index] === "center" ?
                    this.innerTickLength() : this._tierHeights[index]));
            }
            tickMarks.attrs(attr);
            if (this.orientation() === "bottom") {
                attr["y1"] = offset;
                attr["y2"] = offset + (this._tierLabelPositions[index] === "center" ? this.endTickLength() : this._tierHeights[index]);
            }
            else {
                attr["y1"] = this.height() - offset;
                attr["y2"] = this.height() - (offset + (this._tierLabelPositions[index] === "center" ?
                    this.endTickLength() : this._tierHeights[index]));
            }
            d3.select(tickMarks.nodes()[0]).attrs(attr);
            d3.select(tickMarks.nodes()[tickMarks.size() - 1]).attrs(attr);
            // Add end-tick classes to first and last tick for CSS customization purposes
            d3.select(tickMarks.nodes()[0]).classed(axis$2.Axis.END_TICK_MARK_CLASS, true);
            d3.select(tickMarks.nodes()[tickMarks.size() - 1]).classed(axis$2.Axis.END_TICK_MARK_CLASS, true);
            tickMarksUpdate.exit().remove();
        };
        Time.prototype._renderLabellessTickMarks = function (tickValues) {
            var tickMarksUpdate = this._tickMarkContainer.selectAll("." + axis$2.Axis.TICK_MARK_CLASS).data(tickValues);
            var tickMarks = tickMarksUpdate
                .enter()
                .append("line")
                .classed(axis$2.Axis.TICK_MARK_CLASS, true)
                .merge(tickMarksUpdate);
            var attr = this._generateTickMarkAttrHash();
            attr["y2"] = (this.orientation() === "bottom") ? this.tickLabelPadding() : this.height() - this.tickLabelPadding();
            tickMarks.attrs(attr);
            tickMarksUpdate.exit().remove();
        };
        Time.prototype._generateLabellessTicks = function () {
            if (this._mostPreciseConfigIndex < 1) {
                return [];
            }
            return this._getTickIntervalValues(this._possibleTimeAxisConfigurations[this._mostPreciseConfigIndex - 1][0]);
        };
        Time.prototype.renderImmediately = function () {
            var _this = this;
            this._mostPreciseConfigIndex = this._getMostPreciseConfigurationIndex();
            var tierConfigs = this._possibleTimeAxisConfigurations[this._mostPreciseConfigIndex];
            this._cleanTiers();
            tierConfigs.forEach(function (config, i) {
                return _this._renderTierLabels(_this._tierLabelContainers[i], config, i);
            });
            var tierTicks = tierConfigs.map(function (config, i) {
                return _this._getTickValuesForConfiguration(config);
            });
            var baselineOffset = 0;
            for (var i = 0; i < Math.max(tierConfigs.length, 1); ++i) {
                var attr = this._generateBaselineAttrHash();
                attr["y1"] += (this.orientation() === "bottom") ? baselineOffset : -baselineOffset;
                attr["y2"] = attr["y1"];
                this._tierBaselines[i].attrs(attr).style("visibility", "inherit");
                baselineOffset += this._tierHeights[i];
            }
            var labelLessTicks = [];
            var domain = this._scale.domain();
            var totalLength = this._scale.scale(domain[1]) - this._scale.scale(domain[0]);
            if (this._getIntervalLength(tierConfigs[0]) * 1.5 >= totalLength) {
                labelLessTicks = this._generateLabellessTicks();
            }
            this._renderLabellessTickMarks(labelLessTicks);
            this._hideOverflowingTiers();
            for (var i = 0; i < tierConfigs.length; ++i) {
                this._renderTickMarks(tierTicks[i], i);
                this._hideOverlappingAndCutOffLabels(i);
            }
            if (this.annotationsEnabled()) {
                this._drawAnnotations();
            }
            else {
                this._removeAnnotations();
            }
            return this;
        };
        Time.prototype._hideOverflowingTiers = function () {
            var _this = this;
            var availableHeight = this.height();
            var usedHeight = 0;
            this.content()
                .selectAll("." + Time.TIME_AXIS_TIER_CLASS)
                .attr("visibility", function (d, i) {
                usedHeight += _this._tierHeights[i];
                return usedHeight <= availableHeight ? "inherit" : "hidden";
            });
        };
        Time.prototype._hideOverlappingAndCutOffLabels = function (index) {
            var _this = this;
            var boundingBox = this.element().node().getBoundingClientRect();
            var isInsideBBox = function (tickBox) {
                return (Math.floor(boundingBox.left) <= Math.ceil(tickBox.left) &&
                    Math.floor(boundingBox.top) <= Math.ceil(tickBox.top) &&
                    Math.floor(tickBox.right) <= Math.ceil(boundingBox.left + _this.width()) &&
                    Math.floor(tickBox.bottom) <= Math.ceil(boundingBox.top + _this.height()));
            };
            var visibleTickMarks = this._tierMarkContainers[index]
                .selectAll("." + axis$2.Axis.TICK_MARK_CLASS)
                .filter(function (d, i) {
                var visibility = d3.select(this).style("visibility");
                return visibility === "visible" || visibility === "inherit";
            });
            // We use the ClientRects because x1/x2 attributes are not comparable to ClientRects of labels
            var visibleTickMarkRects = visibleTickMarks.nodes().map(function (mark) { return mark.getBoundingClientRect(); });
            var visibleTickLabels = this._tierLabelContainers[index]
                .selectAll("." + axis$2.Axis.TICK_LABEL_CLASS)
                .filter(function (d, i) {
                var visibility = d3.select(this).style("visibility");
                return visibility === "visible" || visibility === "inherit";
            });
            var lastLabelClientRect;
            visibleTickLabels.each(function (d, i) {
                var clientRect = this.getBoundingClientRect();
                var tickLabel = d3.select(this);
                var leadingTickMark = visibleTickMarkRects[i];
                var trailingTickMark = visibleTickMarkRects[i + 1];
                var isOverlappingLastLabel = (lastLabelClientRect != null && utils$1.DOM.clientRectsOverlap(clientRect, lastLabelClientRect));
                var isOverlappingLeadingTickMark = (leadingTickMark != null && utils$1.DOM.clientRectsOverlap(clientRect, leadingTickMark));
                var isOverlappingTrailingTickMark = (trailingTickMark != null && utils$1.DOM.clientRectsOverlap(clientRect, trailingTickMark));
                if (!isInsideBBox(clientRect) || isOverlappingLastLabel || isOverlappingLeadingTickMark || isOverlappingTrailingTickMark) {
                    tickLabel.style("visibility", "hidden");
                }
                else {
                    lastLabelClientRect = clientRect;
                    tickLabel.style("visibility", "inherit");
                }
            });
        };
        Time.prototype.invalidateCache = function () {
            _super.prototype.invalidateCache.call(this);
            this._measurer.reset();
        };
        /**
         * The CSS class applied to each Time Axis tier
         */
        Time.TIME_AXIS_TIER_CLASS = "time-axis-tier";
        Time._SORTED_TIME_INTERVAL_INDEX = (_a = {},
            _a[exports.TimeInterval.second] = 0,
            _a[exports.TimeInterval.minute] = 1,
            _a[exports.TimeInterval.hour] = 2,
            _a[exports.TimeInterval.day] = 3,
            _a[exports.TimeInterval.week] = 4,
            _a[exports.TimeInterval.month] = 5,
            _a[exports.TimeInterval.year] = 6,
            _a);
        Time._DEFAULT_TIME_AXIS_CONFIGURATIONS = function (useUTC) {
            var formatter = function (specifier) { return formatters.time(specifier, useUTC); };
            return [
                [
                    { interval: exports.TimeInterval.second, step: 1, formatter: formatter("%I:%M:%S %p") },
                    { interval: exports.TimeInterval.day, step: 1, formatter: formatter("%B %e, %Y") },
                ],
                [
                    { interval: exports.TimeInterval.second, step: 5, formatter: formatter("%I:%M:%S %p") },
                    { interval: exports.TimeInterval.day, step: 1, formatter: formatter("%B %e, %Y") },
                ],
                [
                    { interval: exports.TimeInterval.second, step: 10, formatter: formatter("%I:%M:%S %p") },
                    { interval: exports.TimeInterval.day, step: 1, formatter: formatter("%B %e, %Y") },
                ],
                [
                    { interval: exports.TimeInterval.second, step: 15, formatter: formatter("%I:%M:%S %p") },
                    { interval: exports.TimeInterval.day, step: 1, formatter: formatter("%B %e, %Y") },
                ],
                [
                    { interval: exports.TimeInterval.second, step: 30, formatter: formatter("%I:%M:%S %p") },
                    { interval: exports.TimeInterval.day, step: 1, formatter: formatter("%B %e, %Y") },
                ],
                [
                    { interval: exports.TimeInterval.minute, step: 1, formatter: formatter("%I:%M %p") },
                    { interval: exports.TimeInterval.day, step: 1, formatter: formatter("%B %e, %Y") },
                ],
                [
                    { interval: exports.TimeInterval.minute, step: 5, formatter: formatter("%I:%M %p") },
                    { interval: exports.TimeInterval.day, step: 1, formatter: formatter("%B %e, %Y") },
                ],
                [
                    { interval: exports.TimeInterval.minute, step: 10, formatter: formatter("%I:%M %p") },
                    { interval: exports.TimeInterval.day, step: 1, formatter: formatter("%B %e, %Y") },
                ],
                [
                    { interval: exports.TimeInterval.minute, step: 15, formatter: formatter("%I:%M %p") },
                    { interval: exports.TimeInterval.day, step: 1, formatter: formatter("%B %e, %Y") },
                ],
                [
                    { interval: exports.TimeInterval.minute, step: 30, formatter: formatter("%I:%M %p") },
                    { interval: exports.TimeInterval.day, step: 1, formatter: formatter("%B %e, %Y") },
                ],
                [
                    { interval: exports.TimeInterval.hour, step: 1, formatter: formatter("%I %p") },
                    { interval: exports.TimeInterval.day, step: 1, formatter: formatter("%B %e, %Y") },
                ],
                [
                    { interval: exports.TimeInterval.hour, step: 3, formatter: formatter("%I %p") },
                    { interval: exports.TimeInterval.day, step: 1, formatter: formatter("%B %e, %Y") },
                ],
                [
                    { interval: exports.TimeInterval.hour, step: 6, formatter: formatter("%I %p") },
                    { interval: exports.TimeInterval.day, step: 1, formatter: formatter("%B %e, %Y") },
                ],
                [
                    { interval: exports.TimeInterval.hour, step: 12, formatter: formatter("%I %p") },
                    { interval: exports.TimeInterval.day, step: 1, formatter: formatter("%B %e, %Y") },
                ],
                [
                    { interval: exports.TimeInterval.day, step: 1, formatter: formatter("%a %e") },
                    { interval: exports.TimeInterval.month, step: 1, formatter: formatter("%B %Y") },
                ],
                [
                    { interval: exports.TimeInterval.day, step: 1, formatter: formatter("%e") },
                    { interval: exports.TimeInterval.month, step: 1, formatter: formatter("%B %Y") },
                ],
                [
                    { interval: exports.TimeInterval.month, step: 1, formatter: formatter("%B") },
                    { interval: exports.TimeInterval.year, step: 1, formatter: formatter("%Y") },
                ],
                [
                    { interval: exports.TimeInterval.month, step: 1, formatter: formatter("%b") },
                    { interval: exports.TimeInterval.year, step: 1, formatter: formatter("%Y") },
                ],
                [
                    { interval: exports.TimeInterval.month, step: 3, formatter: formatter("%b") },
                    { interval: exports.TimeInterval.year, step: 1, formatter: formatter("%Y") },
                ],
                [
                    { interval: exports.TimeInterval.month, step: 6, formatter: formatter("%b") },
                    { interval: exports.TimeInterval.year, step: 1, formatter: formatter("%Y") },
                ],
                [
                    { interval: exports.TimeInterval.year, step: 1, formatter: formatter("%Y") },
                ],
                [
                    { interval: exports.TimeInterval.year, step: 1, formatter: formatter("%y") },
                ],
                [
                    { interval: exports.TimeInterval.year, step: 5, formatter: formatter("%Y") },
                ],
                [
                    { interval: exports.TimeInterval.year, step: 25, formatter: formatter("%Y") },
                ],
                [
                    { interval: exports.TimeInterval.year, step: 50, formatter: formatter("%Y") },
                ],
                [
                    { interval: exports.TimeInterval.year, step: 100, formatter: formatter("%Y") },
                ],
                [
                    { interval: exports.TimeInterval.year, step: 200, formatter: formatter("%Y") },
                ],
                [
                    { interval: exports.TimeInterval.year, step: 500, formatter: formatter("%Y") },
                ],
                [
                    { interval: exports.TimeInterval.year, step: 1000, formatter: formatter("%Y") },
                ],
            ];
        };
        Time._LONG_DATE = new Date(9999, 8, 29, 12, 59, 9999);
        return Time;
    }(axis$2.Axis));
    exports.Time = Time;
    var _a;
    });

    var axes = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });

    tslib_es6.__exportStar(categoryAxis, exports);
    tslib_es6.__exportStar(numericAxis, exports);
    tslib_es6.__exportStar(timeAxis, exports);
    });

    var dispatcher = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });

    var Dispatcher = /** @class */ (function () {
        function Dispatcher() {
            /**
             * Subclasses set these in constructor. Then, these get attached to the event
             * target in _connect
             *
             * eventname is a DOM event name like "mouseup", "touchstart", etc. The
             * callback is simply registered to the event callback with bubbling.
             */
            this._eventToProcessingFunction = {};
            /**
             * All listeners are registered to this `EventTarget` and events are then
             * dispatched to callbacks from `_eventNameToCallbackSet` manually.
             *
             * Subclasses set their own event target instead of `document`.
             */
            this._eventTarget = document;
            this._eventNameToCallbackSet = {};
            this._connected = false;
        }
        Dispatcher.prototype._hasNoCallbacks = function () {
            var eventNames = Object.keys(this._eventNameToCallbackSet);
            for (var i = 0; i < eventNames.length; i++) {
                if (this._eventNameToCallbackSet[eventNames[i]].size !== 0) {
                    return false;
                }
            }
            return true;
        };
        Dispatcher.prototype._connect = function () {
            var _this = this;
            if (this._connected) {
                return;
            }
            Object.keys(this._eventToProcessingFunction).forEach(function (event) {
                var processingFunction = _this._eventToProcessingFunction[event];
                // Add `{ passive: false }` option because Chrome 73 broke this.
                var options = event === "wheel" ? { passive: false } : undefined;
                _this._eventTarget.addEventListener(event, processingFunction, options);
            });
            this._connected = true;
        };
        Dispatcher.prototype._disconnect = function () {
            var _this = this;
            if (this._connected && this._hasNoCallbacks()) {
                Object.keys(this._eventToProcessingFunction).forEach(function (event) {
                    var processingFunction = _this._eventToProcessingFunction[event];
                    _this._eventTarget.removeEventListener(event, processingFunction);
                });
                this._connected = false;
            }
        };
        Dispatcher.prototype._addCallbackForEvent = function (eventName, callback) {
            if (this._eventNameToCallbackSet[eventName] == null) {
                this._eventNameToCallbackSet[eventName] = new utils$1.CallbackSet();
            }
            this._eventNameToCallbackSet[eventName].add(callback);
            this._connect();
        };
        Dispatcher.prototype._removeCallbackForEvent = function (eventName, callback) {
            if (this._eventNameToCallbackSet[eventName] != null) {
                this._eventNameToCallbackSet[eventName].delete(callback);
            }
            this._disconnect();
        };
        Dispatcher.prototype._callCallbacksForEvent = function (eventName) {
            var args = [];
            for (var _i = 1; _i < arguments.length; _i++) {
                args[_i - 1] = arguments[_i];
            }
            var callbackSet = this._eventNameToCallbackSet[eventName];
            if (callbackSet != null) {
                callbackSet.callCallbacks.apply(callbackSet, args);
            }
        };
        return Dispatcher;
    }());
    exports.Dispatcher = Dispatcher;
    });

    var keyDispatcher = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });


    var Key = /** @class */ (function (_super) {
        tslib_es6.__extends(Key, _super);
        /**
         * This constructor should not be invoked directly.
         *
         * @constructor
         */
        function Key() {
            var _this = _super.call(this) || this;
            _this._eventToProcessingFunction[Key._KEYDOWN_EVENT_NAME] = function (e) { return _this._processKeydown(e); };
            _this._eventToProcessingFunction[Key._KEYUP_EVENT_NAME] = function (e) { return _this._processKeyup(e); };
            return _this;
        }
        /**
         * Gets a Key Dispatcher. If one already exists it will be returned;
         * otherwise, a new one will be created.
         *
         * @return {Dispatchers.Key}
         */
        Key.getDispatcher = function () {
            var dispatcher = document[Key._DISPATCHER_KEY];
            if (dispatcher == null) {
                dispatcher = new Key();
                document[Key._DISPATCHER_KEY] = dispatcher;
            }
            return dispatcher;
        };
        Key.prototype._processKeydown = function (event) {
            this._callCallbacksForEvent(Key._KEYDOWN_EVENT_NAME, event.keyCode, event);
        };
        Key.prototype._processKeyup = function (event) {
            this._callCallbacksForEvent(Key._KEYUP_EVENT_NAME, event.keyCode, event);
        };
        /**
         * Registers a callback to be called whenever a key is pressed.
         *
         * @param {KeyCallback} callback
         * @return {Dispatchers.Key} The calling Key Dispatcher.
         */
        Key.prototype.onKeyDown = function (callback) {
            this._addCallbackForEvent(Key._KEYDOWN_EVENT_NAME, callback);
            return this;
        };
        /**
         * Removes the callback to be called whenever a key is pressed.
         *
         * @param {KeyCallback} callback
         * @return {Dispatchers.Key} The calling Key Dispatcher.
         */
        Key.prototype.offKeyDown = function (callback) {
            this._removeCallbackForEvent(Key._KEYDOWN_EVENT_NAME, callback);
            return this;
        };
        /** Registers a callback to be called whenever a key is released.
         *
         * @param {KeyCallback} callback
         * @return {Dispatchers.Key} The calling Key Dispatcher.
         */
        Key.prototype.onKeyUp = function (callback) {
            this._addCallbackForEvent(Key._KEYUP_EVENT_NAME, callback);
            return this;
        };
        /**
         * Removes the callback to be called whenever a key is released.
         *
         * @param {KeyCallback} callback
         * @return {Dispatchers.Key} The calling Key Dispatcher.
         */
        Key.prototype.offKeyUp = function (callback) {
            this._removeCallbackForEvent(Key._KEYUP_EVENT_NAME, callback);
            return this;
        };
        Key._DISPATCHER_KEY = "__Plottable_Dispatcher_Key";
        Key._KEYDOWN_EVENT_NAME = "keydown";
        Key._KEYUP_EVENT_NAME = "keyup";
        return Key;
    }(dispatcher.Dispatcher));
    exports.Key = Key;
    });

    var mouseDispatcher = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });



    var Mouse = /** @class */ (function (_super) {
        tslib_es6.__extends(Mouse, _super);
        /**
         * This constructor not be invoked directly.
         *
         * @constructor
         */
        function Mouse(component) {
            var _this = _super.call(this) || this;
            _this._lastMousePosition = { x: -1, y: -1 };
            _this._translator = utils$1.getTranslator(component);
            var processMoveCallback = function (e) { return _this._measureAndDispatch(component, e, Mouse._MOUSEMOVE_EVENT_NAME, "page"); };
            _this._eventToProcessingFunction[Mouse._MOUSEOVER_EVENT_NAME] = processMoveCallback;
            _this._eventToProcessingFunction[Mouse._MOUSEMOVE_EVENT_NAME] = processMoveCallback;
            _this._eventToProcessingFunction[Mouse._MOUSEOUT_EVENT_NAME] = processMoveCallback;
            _this._eventToProcessingFunction[Mouse._MOUSEDOWN_EVENT_NAME] =
                function (e) { return _this._measureAndDispatch(component, e, Mouse._MOUSEDOWN_EVENT_NAME); };
            _this._eventToProcessingFunction[Mouse._MOUSEUP_EVENT_NAME] =
                function (e) { return _this._measureAndDispatch(component, e, Mouse._MOUSEUP_EVENT_NAME, "page"); };
            _this._eventToProcessingFunction[Mouse._WHEEL_EVENT_NAME] =
                function (e) { return _this._measureAndDispatch(component, e, Mouse._WHEEL_EVENT_NAME); };
            _this._eventToProcessingFunction[Mouse._DBLCLICK_EVENT_NAME] =
                function (e) { return _this._measureAndDispatch(component, e, Mouse._DBLCLICK_EVENT_NAME); };
            return _this;
        }
        /**
         * Get a Mouse Dispatcher for the component tree.
         * If one already exists on that <svg>, it will be returned; otherwise, a new one will be created.
         *
         * @param {SVGElement} elem
         * @return {Dispatchers.Mouse}
         */
        Mouse.getDispatcher = function (component) {
            var element = component.root().rootElement();
            var dispatcher = element[Mouse._DISPATCHER_KEY];
            if (dispatcher == null) {
                dispatcher = new Mouse(component);
                element[Mouse._DISPATCHER_KEY] = dispatcher;
            }
            return dispatcher;
        };
        /**
         * Registers a callback to be called when the mouse position changes.
         *
         * @param {MouseCallback} callback
         * @return {Dispatchers.Mouse} The calling Mouse Dispatcher.
         */
        Mouse.prototype.onMouseMove = function (callback) {
            this._addCallbackForEvent(Mouse._MOUSEMOVE_EVENT_NAME, callback);
            return this;
        };
        /**
         * Removes a callback that would be called when the mouse position changes.
         *
         * @param {MouseCallback} callback
         * @return {Dispatchers.Mouse} The calling Mouse Dispatcher.
         */
        Mouse.prototype.offMouseMove = function (callback) {
            this._removeCallbackForEvent(Mouse._MOUSEMOVE_EVENT_NAME, callback);
            return this;
        };
        /**
         * Registers a callback to be called when a mousedown occurs.
         *
         * @param {MouseCallback} callback
         * @return {Dispatchers.Mouse} The calling Mouse Dispatcher.
         */
        Mouse.prototype.onMouseDown = function (callback) {
            this._addCallbackForEvent(Mouse._MOUSEDOWN_EVENT_NAME, callback);
            return this;
        };
        /**
         * Removes a callback that would be called when a mousedown occurs.
         *
         * @param {MouseCallback} callback
         * @return {Dispatchers.Mouse} The calling Mouse Dispatcher.
         */
        Mouse.prototype.offMouseDown = function (callback) {
            this._removeCallbackForEvent(Mouse._MOUSEDOWN_EVENT_NAME, callback);
            return this;
        };
        /**
         * Registers a callback to be called when a mouseup occurs.
         *
         * @param {MouseCallback} callback
         * @return {Dispatchers.Mouse} The calling Mouse Dispatcher.
         */
        Mouse.prototype.onMouseUp = function (callback) {
            this._addCallbackForEvent(Mouse._MOUSEUP_EVENT_NAME, callback);
            return this;
        };
        /**
         * Removes a callback that would be called when a mouseup occurs.
         *
         * @param {MouseCallback} callback
         * @return {Dispatchers.Mouse} The calling Mouse Dispatcher.
         */
        Mouse.prototype.offMouseUp = function (callback) {
            this._removeCallbackForEvent(Mouse._MOUSEUP_EVENT_NAME, callback);
            return this;
        };
        /**
         * Registers a callback to be called when a wheel event occurs.
         *
         * @param {MouseCallback} callback
         * @return {Dispatchers.Mouse} The calling Mouse Dispatcher.
         */
        Mouse.prototype.onWheel = function (callback) {
            this._addCallbackForEvent(Mouse._WHEEL_EVENT_NAME, callback);
            return this;
        };
        /**
         * Removes a callback that would be called when a wheel event occurs.
         *
         * @param {MouseCallback} callback
         * @return {Dispatchers.Mouse} The calling Mouse Dispatcher.
         */
        Mouse.prototype.offWheel = function (callback) {
            this._removeCallbackForEvent(Mouse._WHEEL_EVENT_NAME, callback);
            return this;
        };
        /**
         * Registers a callback to be called when a dblClick occurs.
         *
         * @param {MouseCallback} callback
         * @return {Dispatchers.Mouse} The calling Mouse Dispatcher.
         */
        Mouse.prototype.onDblClick = function (callback) {
            this._addCallbackForEvent(Mouse._DBLCLICK_EVENT_NAME, callback);
            return this;
        };
        /**
         * Removes a callback that would be called when a dblClick occurs.
         *
         * @param {MouseCallback} callback
         * @return {Dispatchers.Mouse} The calling Mouse Dispatcher.
         */
        Mouse.prototype.offDblClick = function (callback) {
            this._removeCallbackForEvent(Mouse._DBLCLICK_EVENT_NAME, callback);
            return this;
        };
        /**
         * Computes the mouse position from the given event, and if successful
         * calls all the callbacks in the provided callbackSet.
         */
        Mouse.prototype._measureAndDispatch = function (component, event, eventName, scope) {
            if (scope === void 0) { scope = "element"; }
            if (scope !== "page" && scope !== "element") {
                throw new Error("Invalid scope '" + scope + "', must be 'element' or 'page'");
            }
            if (scope === "page" || this.eventInside(component, event)) {
                var position = this._translator.computePosition(event.clientX, event.clientY);
                this._lastMousePosition = position;
                this._callCallbacksForEvent(eventName, this.lastMousePosition(), event);
            }
        };
        Mouse.prototype.eventInside = function (component, event) {
            return utils$1.Translator.isEventInside(component, event);
        };
        /**
         * Returns the last computed mouse position in <svg> coordinate space.
         *
         * @return {Point}
         */
        Mouse.prototype.lastMousePosition = function () {
            return this._lastMousePosition;
        };
        Mouse._DISPATCHER_KEY = "__Plottable_Dispatcher_Mouse";
        Mouse._MOUSEOVER_EVENT_NAME = "mouseover";
        Mouse._MOUSEMOVE_EVENT_NAME = "mousemove";
        Mouse._MOUSEOUT_EVENT_NAME = "mouseout";
        Mouse._MOUSEDOWN_EVENT_NAME = "mousedown";
        Mouse._MOUSEUP_EVENT_NAME = "mouseup";
        Mouse._WHEEL_EVENT_NAME = "wheel";
        Mouse._DBLCLICK_EVENT_NAME = "dblclick";
        return Mouse;
    }(dispatcher.Dispatcher));
    exports.Mouse = Mouse;
    });

    var touchDispatcher = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });



    var Touch = /** @class */ (function (_super) {
        tslib_es6.__extends(Touch, _super);
        /**
         * This constructor should not be invoked directly.
         *
         * @param {SVGElement} svg The root <svg> to attach to.
         */
        function Touch(component) {
            var _this = _super.call(this) || this;
            _this._translator = utils$1.getTranslator(component);
            _this._eventToProcessingFunction[Touch._TOUCHSTART_EVENT_NAME] =
                function (e) { return _this._measureAndDispatch(component, e, Touch._TOUCHSTART_EVENT_NAME, "page"); };
            _this._eventToProcessingFunction[Touch._TOUCHMOVE_EVENT_NAME] =
                function (e) { return _this._measureAndDispatch(component, e, Touch._TOUCHMOVE_EVENT_NAME, "page"); };
            _this._eventToProcessingFunction[Touch._TOUCHEND_EVENT_NAME] =
                function (e) { return _this._measureAndDispatch(component, e, Touch._TOUCHEND_EVENT_NAME, "page"); };
            _this._eventToProcessingFunction[Touch._TOUCHCANCEL_EVENT_NAME] =
                function (e) { return _this._measureAndDispatch(component, e, Touch._TOUCHCANCEL_EVENT_NAME, "page"); };
            return _this;
        }
        /**
         * Gets a Touch Dispatcher for the component.
         * If one already exists, it will be returned; otherwise, a new one will be created.
         *
         * @param component
         * @return {Dispatchers.Touch}
         */
        Touch.getDispatcher = function (component) {
            var svg = component.root().rootElement();
            var dispatcher = svg[Touch._DISPATCHER_KEY];
            if (dispatcher == null) {
                dispatcher = new Touch(component);
                svg[Touch._DISPATCHER_KEY] = dispatcher;
            }
            return dispatcher;
        };
        /**
         * Registers a callback to be called when a touch starts.
         *
         * @param {TouchCallback} callback
         * @return {Dispatchers.Touch} The calling Touch Dispatcher.
         */
        Touch.prototype.onTouchStart = function (callback) {
            this._addCallbackForEvent(Touch._TOUCHSTART_EVENT_NAME, callback);
            return this;
        };
        /**
         * Removes a callback that would be called when a touch starts.
         *
         * @param {TouchCallback} callback
         * @return {Dispatchers.Touch} The calling Touch Dispatcher.
         */
        Touch.prototype.offTouchStart = function (callback) {
            this._removeCallbackForEvent(Touch._TOUCHSTART_EVENT_NAME, callback);
            return this;
        };
        /**
         * Registers a callback to be called when the touch position changes.
         *
         * @param {TouchCallback} callback
         * @return {Dispatchers.Touch} The calling Touch Dispatcher.
         */
        Touch.prototype.onTouchMove = function (callback) {
            this._addCallbackForEvent(Touch._TOUCHMOVE_EVENT_NAME, callback);
            return this;
        };
        /**
         * Removes a callback that would be called when the touch position changes.
         *
         * @param {TouchCallback} callback
         * @return {Dispatchers.Touch} The calling Touch Dispatcher.
         */
        Touch.prototype.offTouchMove = function (callback) {
            this._removeCallbackForEvent(Touch._TOUCHMOVE_EVENT_NAME, callback);
            return this;
        };
        /**
         * Registers a callback to be called when a touch ends.
         *
         * @param {TouchCallback} callback
         * @return {Dispatchers.Touch} The calling Touch Dispatcher.
         */
        Touch.prototype.onTouchEnd = function (callback) {
            this._addCallbackForEvent(Touch._TOUCHEND_EVENT_NAME, callback);
            return this;
        };
        /**
         * Removes a callback that would be called when a touch ends.
         *
         * @param {TouchCallback} callback
         * @return {Dispatchers.Touch} The calling Touch Dispatcher.
         */
        Touch.prototype.offTouchEnd = function (callback) {
            this._removeCallbackForEvent(Touch._TOUCHEND_EVENT_NAME, callback);
            return this;
        };
        /**
         * Registers a callback to be called when a touch is cancelled.
         *
         * @param {TouchCallback} callback
         * @return {Dispatchers.Touch} The calling Touch Dispatcher.
         */
        Touch.prototype.onTouchCancel = function (callback) {
            this._addCallbackForEvent(Touch._TOUCHCANCEL_EVENT_NAME, callback);
            return this;
        };
        /**
         * Removes a callback that would be called when a touch is cancelled.
         *
         * @param {TouchCallback} callback
         * @return {Dispatchers.Touch} The calling Touch Dispatcher.
         */
        Touch.prototype.offTouchCancel = function (callback) {
            this._removeCallbackForEvent(Touch._TOUCHCANCEL_EVENT_NAME, callback);
            return this;
        };
        /**
         * Computes the Touch position from the given event, and if successful
         * calls all the callbacks in the provided callbackSet.
         */
        Touch.prototype._measureAndDispatch = function (component, event, eventName, scope) {
            if (scope === void 0) { scope = "element"; }
            if (scope !== "page" && scope !== "element") {
                throw new Error("Invalid scope '" + scope + "', must be 'element' or 'page'");
            }
            if (scope === "element" && !this.eventInside(component, event)) {
                return;
            }
            var touches = event.changedTouches;
            var touchPositions = {};
            var touchIdentifiers = [];
            for (var i = 0; i < touches.length; i++) {
                var touch = touches[i];
                var touchID = touch.identifier;
                var newTouchPosition = this._translator.computePosition(touch.clientX, touch.clientY);
                if (newTouchPosition != null) {
                    touchPositions[touchID] = newTouchPosition;
                    touchIdentifiers.push(touchID);
                }
            }
            if (touchIdentifiers.length > 0) {
                this._callCallbacksForEvent(eventName, touchIdentifiers, touchPositions, event);
            }
        };
        Touch.prototype.eventInside = function (component, event) {
            return utils$1.Translator.isEventInside(component, event);
        };
        Touch._DISPATCHER_KEY = "__Plottable_Dispatcher_Touch";
        Touch._TOUCHSTART_EVENT_NAME = "touchstart";
        Touch._TOUCHMOVE_EVENT_NAME = "touchmove";
        Touch._TOUCHEND_EVENT_NAME = "touchend";
        Touch._TOUCHCANCEL_EVENT_NAME = "touchcancel";
        return Touch;
    }(dispatcher.Dispatcher));
    exports.Touch = Touch;
    });

    var dispatchers = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });

    tslib_es6.__exportStar(keyDispatcher, exports);
    tslib_es6.__exportStar(mouseDispatcher, exports);
    tslib_es6.__exportStar(touchDispatcher, exports);
    });

    var interaction = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });
    var Interaction = /** @class */ (function () {
        function Interaction() {
            var _this = this;
            this._anchorCallback = function (component) { return _this._anchor(component); };
            this._enabled = true;
        }
        /**
         * Attaches this Interaction to a Component.
         * If the Interaction was already attached to a Component, it first detaches itself from the old Component.
         *
         * @param {Component} component
         * @returns {Interaction} The calling Interaction.
         */
        Interaction.prototype.attachTo = function (component) {
            this._disconnect();
            this._componentAttachedTo = component;
            this._connect();
            return this;
        };
        /**
         * @deprecated renamed to .detach().
         */
        Interaction.prototype.detachFrom = function (_component) {
            return this.detach();
        };
        /**
         * Detaches this Interaction from whatever component it was attached to.
         * This Interaction can be reused.
         *
         * @returns {Interaction} The calling Interaction.
         */
        Interaction.prototype.detach = function () {
            this._disconnect();
            this._componentAttachedTo = null;
            return this;
        };
        Interaction.prototype.enabled = function (enabled) {
            if (enabled == null) {
                return this._enabled;
            }
            this._enabled = enabled;
            if (this._enabled) {
                this._connect();
            }
            else {
                this._disconnect();
            }
            return this;
        };
        Interaction.prototype._anchor = function (component) {
            this._isAnchored = true;
        };
        Interaction.prototype._unanchor = function () {
            this._isAnchored = false;
        };
        /**
         * Translates an <svg>-coordinate-space point to Component-space coordinates.
         *
         * @param {Point} p A Point in <svg>-space coordinates.
         * @return {Point} The same location in Component-space coordinates.
         */
        Interaction.prototype._translateToComponentSpace = function (p) {
            var origin = this._componentAttachedTo.originToRoot();
            return {
                x: p.x - origin.x,
                y: p.y - origin.y,
            };
        };
        /**
         * Checks whether a Component-coordinate-space Point is inside the Component.
         *
         * @param {Point} p A Point in Compoennt-space coordinates.
         * @return {boolean} Whether or not the point is inside the Component.
         */
        Interaction.prototype._isInsideComponent = function (p) {
            return 0 <= p.x && 0 <= p.y
                && p.x <= this._componentAttachedTo.width()
                && p.y <= this._componentAttachedTo.height();
        };
        Interaction.prototype._connect = function () {
            if (this.enabled() && this._componentAttachedTo != null && !this._isAnchored) {
                this._componentAttachedTo.onAnchor(this._anchorCallback);
            }
        };
        Interaction.prototype._disconnect = function () {
            if (this._isAnchored) {
                this._unanchor();
            }
            if (this._componentAttachedTo != null) {
                this._componentAttachedTo.offAnchor(this._anchorCallback);
            }
        };
        return Interaction;
    }());
    exports.Interaction = Interaction;
    });

    var clickInteraction = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });




    var Click = /** @class */ (function (_super) {
        tslib_es6.__extends(Click, _super);
        function Click() {
            var _this = _super !== null && _super.apply(this, arguments) || this;
            _this._clickedDown = false;
            _this._doubleClicking = false;
            _this._onClickCallbacks = new utils$1.CallbackSet();
            _this._onDoubleClickCallbacks = new utils$1.CallbackSet();
            /**
             * Note: we bind to mousedown, mouseup, touchstart and touchend because browsers
             * have a 300ms delay between touchstart and click to allow for scrolling cancelling etc.
             */
            _this._mouseDownCallback = function (p, event) { return _this._handleClickDown(p, event); };
            _this._mouseUpCallback = function (p, event) { return _this._handleClickUp(p, event); };
            _this._dblClickCallback = function (p, event) { return _this._handleDblClick(p, event); };
            _this._touchStartCallback = function (ids, idToPoint, event) { return _this._handleClickDown(idToPoint[ids[0]], event); };
            _this._touchEndCallback = function (ids, idToPoint, event) { return _this._handleClickUp(idToPoint[ids[0]], event); };
            _this._touchCancelCallback = function (ids, idToPoint) { return _this._clickedDown = false; };
            return _this;
        }
        Click.prototype._anchor = function (component) {
            _super.prototype._anchor.call(this, component);
            this._mouseDispatcher = dispatchers.Mouse.getDispatcher(component);
            this._mouseDispatcher.onMouseDown(this._mouseDownCallback);
            this._mouseDispatcher.onMouseUp(this._mouseUpCallback);
            this._mouseDispatcher.onDblClick(this._dblClickCallback);
            this._touchDispatcher = dispatchers.Touch.getDispatcher(component);
            this._touchDispatcher.onTouchStart(this._touchStartCallback);
            this._touchDispatcher.onTouchEnd(this._touchEndCallback);
            this._touchDispatcher.onTouchCancel(this._touchCancelCallback);
        };
        Click.prototype._unanchor = function () {
            _super.prototype._unanchor.call(this);
            this._mouseDispatcher.offMouseDown(this._mouseDownCallback);
            this._mouseDispatcher.offMouseUp(this._mouseUpCallback);
            this._mouseDispatcher.offDblClick(this._dblClickCallback);
            this._mouseDispatcher = null;
            this._touchDispatcher.offTouchStart(this._touchStartCallback);
            this._touchDispatcher.offTouchEnd(this._touchEndCallback);
            this._touchDispatcher.offTouchCancel(this._touchCancelCallback);
            this._touchDispatcher = null;
        };
        Click.prototype._handleClickDown = function (p, event) {
            var translatedP = this._translateToComponentSpace(p);
            if (this._isInsideComponent(translatedP)) {
                this._clickedDown = true;
                this._clickedPoint = translatedP;
            }
        };
        Click.prototype._handleClickUp = function (p, event) {
            var _this = this;
            var translatedP = this._translateToComponentSpace(p);
            if (this._clickedDown && Click._pointsEqual(translatedP, this._clickedPoint)) {
                setTimeout(function () {
                    if (!_this._doubleClicking) {
                        _this._onClickCallbacks.callCallbacks(translatedP, event);
                    }
                }, 0);
            }
            this._clickedDown = false;
        };
        Click.prototype._handleDblClick = function (p, event) {
            var _this = this;
            var translatedP = this._translateToComponentSpace(p);
            this._doubleClicking = true;
            this._onDoubleClickCallbacks.callCallbacks(translatedP, event);
            setTimeout(function () { return _this._doubleClicking = false; }, 0);
        };
        Click._pointsEqual = function (p1, p2) {
            return p1.x === p2.x && p1.y === p2.y;
        };
        /**
         * Adds a callback to be called when the Component is clicked.
         *
         * @param {ClickCallback} callback
         * @return {Interactions.Click} The calling Click Interaction.
         */
        Click.prototype.onClick = function (callback) {
            this._onClickCallbacks.add(callback);
            return this;
        };
        /**
         * Removes a callback that would be called when the Component is clicked.
         *
         * @param {ClickCallback} callback
         * @return {Interactions.Click} The calling Click Interaction.
         */
        Click.prototype.offClick = function (callback) {
            this._onClickCallbacks.delete(callback);
            return this;
        };
        /**
         * Adds a callback to be called when the Component is double-clicked.
         *
         * @param {ClickCallback} callback
         * @return {Interactions.Click} The calling Click Interaction.
         */
        Click.prototype.onDoubleClick = function (callback) {
            this._onDoubleClickCallbacks.add(callback);
            return this;
        };
        /**
         * Removes a callback that would be called when the Component is double-clicked.
         *
         * @param {ClickCallback} callback
         * @return {Interactions.Click} The calling Click Interaction.
         */
        Click.prototype.offDoubleClick = function (callback) {
            this._onDoubleClickCallbacks.delete(callback);
            return this;
        };
        return Click;
    }(interaction.Interaction));
    exports.Click = Click;
    });

    var dragInteraction = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });




    var Drag = /** @class */ (function (_super) {
        tslib_es6.__extends(Drag, _super);
        function Drag(mouseButton) {
            var _this = _super.call(this) || this;
            _this._dragging = false;
            _this._constrainedToComponent = true;
            /**
             * Only emit events when the mouseFilter is true for the source mouse
             * events. Use this to define custom filters (e.g. only right click,
             * require shift to be held down, etc.)
             */
            _this._mouseFilter = Drag._DEFAULT_MOUSE_FILTER;
            _this._dragStartCallbacks = new utils$1.CallbackSet();
            _this._dragCallbacks = new utils$1.CallbackSet();
            _this._dragEndCallbacks = new utils$1.CallbackSet();
            _this._mouseDownCallback = function (p, e) { return _this._startDrag(p, e); };
            _this._mouseMoveCallback = function (p, e) { return _this._doDrag(p, e); };
            _this._mouseUpCallback = function (p, e) { return _this._endDrag(p, e); };
            _this._touchStartCallback = function (ids, idToPoint, e) { return _this._startDrag(idToPoint[ids[0]], e); };
            _this._touchMoveCallback = function (ids, idToPoint, e) { return _this._doDrag(idToPoint[ids[0]], e); };
            _this._touchEndCallback = function (ids, idToPoint, e) { return _this._endDrag(idToPoint[ids[0]], e); };
            _this._mouseButton = mouseButton !== undefined ? mouseButton : 0;
            return _this;
        }
        Drag.prototype._anchor = function (component) {
            _super.prototype._anchor.call(this, component);
            this._mouseDispatcher = dispatchers.Mouse.getDispatcher(this._componentAttachedTo);
            this._mouseDispatcher.onMouseDown(this._mouseDownCallback);
            this._mouseDispatcher.onMouseMove(this._mouseMoveCallback);
            this._mouseDispatcher.onMouseUp(this._mouseUpCallback);
            this._touchDispatcher = dispatchers.Touch.getDispatcher(this._componentAttachedTo);
            this._touchDispatcher.onTouchStart(this._touchStartCallback);
            this._touchDispatcher.onTouchMove(this._touchMoveCallback);
            this._touchDispatcher.onTouchEnd(this._touchEndCallback);
        };
        Drag.prototype._unanchor = function () {
            _super.prototype._unanchor.call(this);
            this._mouseDispatcher.offMouseDown(this._mouseDownCallback);
            this._mouseDispatcher.offMouseMove(this._mouseMoveCallback);
            this._mouseDispatcher.offMouseUp(this._mouseUpCallback);
            this._mouseDispatcher = null;
            this._touchDispatcher.offTouchStart(this._touchStartCallback);
            this._touchDispatcher.offTouchMove(this._touchMoveCallback);
            this._touchDispatcher.offTouchEnd(this._touchEndCallback);
            this._touchDispatcher = null;
        };
        Drag.prototype._translateAndConstrain = function (p) {
            var translatedP = this._translateToComponentSpace(p);
            if (!this._constrainedToComponent) {
                return translatedP;
            }
            return {
                x: utils$1.Math.clamp(translatedP.x, 0, this._componentAttachedTo.width()),
                y: utils$1.Math.clamp(translatedP.y, 0, this._componentAttachedTo.height()),
            };
        };
        Drag.prototype._startDrag = function (point, event) {
            if (event instanceof MouseEvent && !this._mouseFilter(event)) {
                return;
            }
            var translatedP = this._translateToComponentSpace(point);
            if (this._isInsideComponent(translatedP)) {
                event.preventDefault();
                this._dragging = true;
                this._dragOrigin = translatedP;
                this._dragStartCallbacks.callCallbacks(this._dragOrigin);
            }
        };
        Drag.prototype._doDrag = function (point, event) {
            if (this._dragging) {
                this._dragCallbacks.callCallbacks(this._dragOrigin, this._translateAndConstrain(point));
            }
        };
        Drag.prototype._endDrag = function (point, event) {
            if (event instanceof MouseEvent && event.button !== this._mouseButton) {
                return;
            }
            if (this._dragging) {
                this._dragging = false;
                this._dragEndCallbacks.callCallbacks(this._dragOrigin, this._translateAndConstrain(point));
            }
        };
        Drag.prototype.constrainedToComponent = function (constrainedToComponent) {
            if (constrainedToComponent == null) {
                return this._constrainedToComponent;
            }
            this._constrainedToComponent = constrainedToComponent;
            return this;
        };
        Drag.prototype.mouseFilter = function (filter) {
            if (arguments.length === 0) {
                return this._mouseFilter;
            }
            this._mouseFilter = filter;
            return this;
        };
        /**
         * Adds a callback to be called when dragging starts.
         *
         * @param {DragCallback} callback
         * @returns {Drag} The calling Drag Interaction.
         */
        Drag.prototype.onDragStart = function (callback) {
            this._dragStartCallbacks.add(callback);
            return this;
        };
        /**
         * Removes a callback that would be called when dragging starts.
         *
         * @param {DragCallback} callback
         * @returns {Drag} The calling Drag Interaction.
         */
        Drag.prototype.offDragStart = function (callback) {
            this._dragStartCallbacks.delete(callback);
            return this;
        };
        /**
         * Adds a callback to be called during dragging.
         *
         * @param {DragCallback} callback
         * @returns {Drag} The calling Drag Interaction.
         */
        Drag.prototype.onDrag = function (callback) {
            this._dragCallbacks.add(callback);
            return this;
        };
        /**
         * Removes a callback that would be called during dragging.
         *
         * @param {DragCallback} callback
         * @returns {Drag} The calling Drag Interaction.
         */
        Drag.prototype.offDrag = function (callback) {
            this._dragCallbacks.delete(callback);
            return this;
        };
        /**
         * Adds a callback to be called when dragging ends.
         *
         * @param {DragCallback} callback
         * @returns {Drag} The calling Drag Interaction.
         */
        Drag.prototype.onDragEnd = function (callback) {
            this._dragEndCallbacks.add(callback);
            return this;
        };
        /**
         * Removes a callback that would be called when dragging ends.
         *
         * @param {DragCallback} callback
         * @returns {Drag} The calling Drag Interaction.
         */
        Drag.prototype.offDragEnd = function (callback) {
            this._dragEndCallbacks.delete(callback);
            return this;
        };
        Drag._DEFAULT_MOUSE_FILTER = function (event) { return event.button === 0; };
        return Drag;
    }(interaction.Interaction));
    exports.Drag = Drag;
    });

    var keyInteraction = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });




    var Key = /** @class */ (function (_super) {
        tslib_es6.__extends(Key, _super);
        function Key() {
            var _this = _super !== null && _super.apply(this, arguments) || this;
            _this._keyPressCallbacks = {};
            _this._keyReleaseCallbacks = {};
            _this._mouseMoveCallback = function (point) { return false; }; // HACKHACK: registering a listener
            _this._downedKeys = new utils$1.Set();
            _this._keyDownCallback = function (keyCode, event) { return _this._handleKeyDownEvent(keyCode, event); };
            _this._keyUpCallback = function (keyCode) { return _this._handleKeyUpEvent(keyCode); };
            return _this;
        }
        Key.prototype._anchor = function (component) {
            _super.prototype._anchor.call(this, component);
            this._positionDispatcher = dispatchers.Mouse.getDispatcher(this._componentAttachedTo);
            this._positionDispatcher.onMouseMove(this._mouseMoveCallback);
            this._keyDispatcher = dispatchers.Key.getDispatcher();
            this._keyDispatcher.onKeyDown(this._keyDownCallback);
            this._keyDispatcher.onKeyUp(this._keyUpCallback);
        };
        Key.prototype._unanchor = function () {
            _super.prototype._unanchor.call(this);
            this._positionDispatcher.offMouseMove(this._mouseMoveCallback);
            this._positionDispatcher = null;
            this._keyDispatcher.offKeyDown(this._keyDownCallback);
            this._keyDispatcher.offKeyUp(this._keyUpCallback);
            this._keyDispatcher = null;
        };
        Key.prototype._handleKeyDownEvent = function (keyCode, event) {
            var p = this._translateToComponentSpace(this._positionDispatcher.lastMousePosition());
            if (this._isInsideComponent(p) && !event.repeat) {
                if (this._keyPressCallbacks[keyCode]) {
                    this._keyPressCallbacks[keyCode].callCallbacks(keyCode);
                }
                this._downedKeys.add(keyCode);
            }
        };
        Key.prototype._handleKeyUpEvent = function (keyCode) {
            if (this._downedKeys.has(keyCode) && this._keyReleaseCallbacks[keyCode]) {
                this._keyReleaseCallbacks[keyCode].callCallbacks(keyCode);
            }
            this._downedKeys.delete(keyCode);
        };
        /**
         * Adds a callback to be called when the key with the given keyCode is
         * pressed and the user is moused over the Component.
         *
         * @param {number} keyCode
         * @param {KeyCallback} callback
         * @returns {Interactions.Key} The calling Key Interaction.
         */
        Key.prototype.onKeyPress = function (keyCode, callback) {
            if (!this._keyPressCallbacks[keyCode]) {
                this._keyPressCallbacks[keyCode] = new utils$1.CallbackSet();
            }
            this._keyPressCallbacks[keyCode].add(callback);
            return this;
        };
        /**
         * Removes a callback that would be called when the key with the given keyCode is
         * pressed and the user is moused over the Component.
         *
         * @param {number} keyCode
         * @param {KeyCallback} callback
         * @returns {Interactions.Key} The calling Key Interaction.
         */
        Key.prototype.offKeyPress = function (keyCode, callback) {
            this._keyPressCallbacks[keyCode].delete(callback);
            if (this._keyPressCallbacks[keyCode].size === 0) {
                delete this._keyPressCallbacks[keyCode];
            }
            return this;
        };
        /**
         * Adds a callback to be called when the key with the given keyCode is
         * released if the key was pressed with the mouse inside of the Component.
         *
         * @param {number} keyCode
         * @param {KeyCallback} callback
         * @returns {Interactions.Key} The calling Key Interaction.
         */
        Key.prototype.onKeyRelease = function (keyCode, callback) {
            if (!this._keyReleaseCallbacks[keyCode]) {
                this._keyReleaseCallbacks[keyCode] = new utils$1.CallbackSet();
            }
            this._keyReleaseCallbacks[keyCode].add(callback);
            return this;
        };
        /**
         * Removes a callback that would be called when the key with the given keyCode is
         * released if the key was pressed with the mouse inside of the Component.
         *
         * @param {number} keyCode
         * @param {KeyCallback} callback
         * @returns {Interactions.Key} The calling Key Interaction.
         */
        Key.prototype.offKeyRelease = function (keyCode, callback) {
            this._keyReleaseCallbacks[keyCode].delete(callback);
            if (this._keyReleaseCallbacks[keyCode].size === 0) {
                delete this._keyReleaseCallbacks[keyCode];
            }
            return this;
        };
        return Key;
    }(interaction.Interaction));
    exports.Key = Key;
    });

    var panZoomInteraction = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });








    var PanZoom = /** @class */ (function (_super) {
        tslib_es6.__extends(PanZoom, _super);
        /**
         * A PanZoom Interaction updates the domains of an x-scale and/or a y-scale
         * in response to the user panning or zooming.
         *
         * @constructor
         * @param {TransformableScale} [xScale] The x-scale to update on panning/zooming.
         * @param {TransformableScale} [yScale] The y-scale to update on panning/zooming.
         */
        function PanZoom(xScale, yScale) {
            var _this = _super.call(this) || this;
            _this._wheelFilter = function (e) { return true; };
            _this._wheelCallback = function (p, e) { return _this._handleWheelEvent(p, e); };
            _this._touchStartCallback = function (ids, idToPoint, e) { return _this._handleTouchStart(ids, idToPoint, e); };
            _this._touchMoveCallback = function (ids, idToPoint, e) { return _this._handlePinch(ids, idToPoint, e); };
            _this._touchEndCallback = function (ids, idToPoint, e) { return _this._handleTouchEnd(ids, idToPoint, e); };
            _this._touchCancelCallback = function (ids, idToPoint, e) { return _this._handleTouchEnd(ids, idToPoint, e); };
            _this._panEndCallbacks = new utils$1.CallbackSet();
            _this._zoomEndCallbacks = new utils$1.CallbackSet();
            _this._panZoomUpdateCallbacks = new utils$1.CallbackSet();
            _this._xScales = new utils$1.Set();
            _this._yScales = new utils$1.Set();
            _this._dragInteraction = new interactions.Drag();
            _this._setupDragInteraction();
            _this._touchIds = d3.map();
            _this._minDomainExtents = new utils$1.Map();
            _this._maxDomainExtents = new utils$1.Map();
            _this._minDomainValues = new utils$1.Map();
            _this._maxDomainValues = new utils$1.Map();
            if (xScale != null) {
                _this.addXScale(xScale);
            }
            if (yScale != null) {
                _this.addYScale(yScale);
            }
            return _this;
        }
        /**
         * Get the backing drag interaction. Useful to customize the panzoom interaction.
         * @returns {Drag}
         */
        PanZoom.prototype.dragInteraction = function () {
            return this._dragInteraction;
        };
        PanZoom.prototype.wheelFilter = function (filter) {
            if (arguments.length === 0) {
                return this._wheelFilter;
            }
            this._wheelFilter = filter;
            return this;
        };
        /**
         * Pans the chart by a specified amount
         *
         * @param {Plottable.Point} [translateAmount] The amount by which to translate the x and y scales.
         */
        PanZoom.prototype.pan = function (translateAmount) {
            var _this = this;
            this.xScales().forEach(function (xScale) {
                xScale.pan(_this._constrainedTranslation(xScale, translateAmount.x));
            });
            this.yScales().forEach(function (yScale) {
                yScale.pan(_this._constrainedTranslation(yScale, translateAmount.y));
            });
            this._panZoomUpdateCallbacks.callCallbacks();
        };
        /**
         * Zooms the chart by a specified amount around a specific point
         *
         * @param {number} [zoomAmount] The percentage by which to zoom the x and y scale.
         * A value of 0.9 zooms in by 10%. A value of 1.1 zooms out by 10%. A value of 1 has
         * no effect.
         * @param {Plottable.Point} [centerValue] The center in pixels around which to zoom.
         * By default, `centerValue` is the center of the x and y range of each scale.
         * @param {boolean} [constrained] Whether to respect the zoom extents and min/max
         * values. Default true.
         */
        PanZoom.prototype.zoom = function (zoomAmount, centerValue, constrained) {
            var _this = this;
            if (constrained === void 0) { constrained = true; }
            var centerX;
            var centerY;
            if (centerValue != null) {
                centerX = centerValue.x;
                centerY = centerValue.y;
                if (constrained) {
                    this.xScales().forEach(function (xScale) {
                        var constrained = _this._constrainedZoom(xScale, zoomAmount, centerX);
                        centerX = constrained.centerPoint;
                        zoomAmount = constrained.zoomAmount;
                    });
                    this.yScales().forEach(function (yScale) {
                        var constrained = _this._constrainedZoom(yScale, zoomAmount, centerY);
                        centerY = constrained.centerPoint;
                        zoomAmount = constrained.zoomAmount;
                    });
                }
            }
            this.xScales().forEach(function (xScale) {
                var range = xScale.range();
                var xCenter = centerX == null
                    ? (range[1] + range[0]) / 2
                    : centerX;
                xScale.zoom(zoomAmount, xCenter);
            });
            this.yScales().forEach(function (yScale) {
                var range = yScale.range();
                var yCenter = centerY == null
                    ? (range[1] + range[0]) / 2
                    : centerY;
                yScale.zoom(zoomAmount, yCenter);
            });
            this._panZoomUpdateCallbacks.callCallbacks();
            return { zoomAmount: zoomAmount, centerValue: { centerX: centerX, centerY: centerY } };
        };
        PanZoom.prototype._anchor = function (component) {
            _super.prototype._anchor.call(this, component);
            this._dragInteraction.attachTo(component);
            this._mouseDispatcher = dispatchers.Mouse.getDispatcher(this._componentAttachedTo);
            this._mouseDispatcher.onWheel(this._wheelCallback);
            this._touchDispatcher = dispatchers.Touch.getDispatcher(this._componentAttachedTo);
            this._touchDispatcher.onTouchStart(this._touchStartCallback);
            this._touchDispatcher.onTouchMove(this._touchMoveCallback);
            this._touchDispatcher.onTouchEnd(this._touchEndCallback);
            this._touchDispatcher.onTouchCancel(this._touchCancelCallback);
        };
        PanZoom.prototype._unanchor = function () {
            _super.prototype._unanchor.call(this);
            this._mouseDispatcher.offWheel(this._wheelCallback);
            this._mouseDispatcher = null;
            this._touchDispatcher.offTouchStart(this._touchStartCallback);
            this._touchDispatcher.offTouchMove(this._touchMoveCallback);
            this._touchDispatcher.offTouchEnd(this._touchEndCallback);
            this._touchDispatcher.offTouchCancel(this._touchCancelCallback);
            this._touchDispatcher = null;
            this._dragInteraction.detach();
        };
        PanZoom.prototype._handleTouchStart = function (ids, idToPoint, e) {
            for (var i = 0; i < ids.length && this._touchIds.size() < 2; i++) {
                var id = ids[i];
                this._touchIds.set(id.toString(), this._translateToComponentSpace(idToPoint[id]));
            }
        };
        PanZoom.prototype._handlePinch = function (ids, idToPoint, e) {
            var _this = this;
            if (this._touchIds.size() < 2) {
                return;
            }
            var oldPoints = this._touchIds.values();
            if (!this._isInsideComponent(this._translateToComponentSpace(oldPoints[0])) || !this._isInsideComponent(this._translateToComponentSpace(oldPoints[1]))) {
                return;
            }
            var oldCornerDistance = PanZoom._pointDistance(oldPoints[0], oldPoints[1]);
            if (oldCornerDistance === 0) {
                return;
            }
            ids.forEach(function (id) {
                if (_this._touchIds.has(id.toString())) {
                    _this._touchIds.set(id.toString(), _this._translateToComponentSpace(idToPoint[id]));
                }
            });
            var points = this._touchIds.values();
            var newCornerDistance = PanZoom._pointDistance(points[0], points[1]);
            if (newCornerDistance === 0) {
                return;
            }
            var initMagnifyAmount = oldCornerDistance / newCornerDistance;
            var normalizedPointDiffs = points.map(function (point, i) {
                return {
                    x: (point.x - oldPoints[i].x) / initMagnifyAmount,
                    y: (point.y - oldPoints[i].y) / initMagnifyAmount,
                };
            });
            var oldCenterPoint = PanZoom.centerPoint(oldPoints[0], oldPoints[1]);
            var _a = this.zoom(initMagnifyAmount, oldCenterPoint), centerValue = _a.centerValue, zoomAmount = _a.zoomAmount;
            var centerX = centerValue.centerX, centerY = centerValue.centerY;
            var constrainedPoints = oldPoints.map(function (oldPoint, i) {
                return {
                    x: normalizedPointDiffs[i].x * zoomAmount + oldPoint.x,
                    y: normalizedPointDiffs[i].y * zoomAmount + oldPoint.y,
                };
            });
            var translateAmount = {
                x: centerX - ((constrainedPoints[0].x + constrainedPoints[1].x) / 2),
                y: centerY - ((constrainedPoints[0].y + constrainedPoints[1].y) / 2),
            };
            this.pan(translateAmount);
        };
        PanZoom.centerPoint = function (point1, point2) {
            var leftX = Math.min(point1.x, point2.x);
            var rightX = Math.max(point1.x, point2.x);
            var topY = Math.min(point1.y, point2.y);
            var bottomY = Math.max(point1.y, point2.y);
            return { x: (leftX + rightX) / 2, y: (bottomY + topY) / 2 };
        };
        PanZoom._pointDistance = function (point1, point2) {
            var leftX = Math.min(point1.x, point2.x);
            var rightX = Math.max(point1.x, point2.x);
            var topY = Math.min(point1.y, point2.y);
            var bottomY = Math.max(point1.y, point2.y);
            return Math.sqrt(Math.pow(rightX - leftX, 2) + Math.pow(bottomY - topY, 2));
        };
        PanZoom.prototype._handleTouchEnd = function (ids, idToPoint, e) {
            var _this = this;
            ids.forEach(function (id) {
                _this._touchIds.remove(id.toString());
            });
            if (this._touchIds.size() > 0) {
                this._zoomEndCallbacks.callCallbacks();
            }
        };
        PanZoom.prototype._handleWheelEvent = function (p, e) {
            if (!this._wheelFilter(e)) {
                return;
            }
            var translatedP = this._translateToComponentSpace(p);
            if (this._isInsideComponent(translatedP)) {
                e.preventDefault();
                // in cases where only horizontal scroll is present, use that in lieu of nothing.
                var deltaAmount = e.deltaY !== 0 ? e.deltaY : e.deltaX;
                var deltaPixelAmount = deltaAmount * (e.deltaMode ? PanZoom._PIXELS_PER_LINE : 1);
                var zoomAmount = Math.pow(2, deltaPixelAmount * .002);
                this.zoom(zoomAmount, translatedP);
                this._zoomEndCallbacks.callCallbacks();
            }
        };
        PanZoom.prototype._constrainedZoom = function (scale, zoomAmount, centerPoint) {
            return panZoomConstraints.constrainedZoom(scale, zoomAmount, centerPoint, this.minDomainExtent(scale), this.maxDomainExtent(scale), this.minDomainValue(scale), this.maxDomainValue(scale));
        };
        PanZoom.prototype._constrainedTranslation = function (scale, translation) {
            return panZoomConstraints.constrainedTranslation(scale, translation, this.minDomainValue(scale), this.maxDomainValue(scale));
        };
        PanZoom.prototype._setupDragInteraction = function () {
            var _this = this;
            this._dragInteraction.constrainedToComponent(false);
            var lastDragPoint;
            this._dragInteraction.onDragStart(function () { return lastDragPoint = null; });
            this._dragInteraction.onDrag(function (startPoint, endPoint) {
                if (_this._touchIds.size() >= 2) {
                    return;
                }
                var translateAmount = {
                    x: (lastDragPoint == null ? startPoint.x : lastDragPoint.x) - endPoint.x,
                    y: (lastDragPoint == null ? startPoint.y : lastDragPoint.y) - endPoint.y,
                };
                _this.pan(translateAmount);
                lastDragPoint = endPoint;
            });
            this._dragInteraction.onDragEnd(function () { return _this._panEndCallbacks.callCallbacks(); });
        };
        PanZoom.prototype._nonLinearScaleWithExtents = function (scale) {
            return this.minDomainExtent(scale) != null && this.maxDomainExtent(scale) != null && !(scale instanceof scales.Linear) && !(scale instanceof scales.Time);
        };
        PanZoom.prototype.xScales = function (xScales) {
            var _this = this;
            if (xScales == null) {
                var scales_1 = [];
                this._xScales.forEach(function (xScale) {
                    scales_1.push(xScale);
                });
                return scales_1;
            }
            this._xScales = new utils$1.Set();
            xScales.forEach(function (xScale) {
                _this.addXScale(xScale);
            });
            return this;
        };
        PanZoom.prototype.yScales = function (yScales) {
            var _this = this;
            if (yScales == null) {
                var scales_2 = [];
                this._yScales.forEach(function (yScale) {
                    scales_2.push(yScale);
                });
                return scales_2;
            }
            this._yScales = new utils$1.Set();
            yScales.forEach(function (yScale) {
                _this.addYScale(yScale);
            });
            return this;
        };
        /**
         * Adds an x scale to this PanZoom Interaction
         *
         * @param {TransformableScale} An x scale to add
         * @returns {Interactions.PanZoom} The calling PanZoom Interaction.
         */
        PanZoom.prototype.addXScale = function (xScale) {
            this._xScales.add(xScale);
            return this;
        };
        /**
         * Removes an x scale from this PanZoom Interaction
         *
         * @param {TransformableScale} An x scale to remove
         * @returns {Interactions.PanZoom} The calling PanZoom Interaction.
         */
        PanZoom.prototype.removeXScale = function (xScale) {
            this._xScales.delete(xScale);
            this._minDomainExtents.delete(xScale);
            this._maxDomainExtents.delete(xScale);
            this._minDomainValues.delete(xScale);
            this._maxDomainValues.delete(xScale);
            return this;
        };
        /**
         * Adds a y scale to this PanZoom Interaction
         *
         * @param {TransformableScale} A y scale to add
         * @returns {Interactions.PanZoom} The calling PanZoom Interaction.
         */
        PanZoom.prototype.addYScale = function (yScale) {
            this._yScales.add(yScale);
            return this;
        };
        /**
         * Removes a y scale from this PanZoom Interaction
         *
         * @param {TransformableScale} A y scale to remove
         * @returns {Interactions.PanZoom} The calling PanZoom Interaction.
         */
        PanZoom.prototype.removeYScale = function (yScale) {
            this._yScales.delete(yScale);
            this._minDomainExtents.delete(yScale);
            this._maxDomainExtents.delete(yScale);
            this._minDomainValues.delete(yScale);
            this._maxDomainValues.delete(yScale);
            return this;
        };
        PanZoom.prototype.minDomainExtent = function (scale, minDomainExtent) {
            if (minDomainExtent == null) {
                return this._minDomainExtents.get(scale);
            }
            if (minDomainExtent.valueOf() < 0) {
                throw new Error("extent must be non-negative");
            }
            var maxExtentForScale = this.maxDomainExtent(scale);
            if (maxExtentForScale != null && maxExtentForScale.valueOf() < minDomainExtent.valueOf()) {
                throw new Error("minDomainExtent must be smaller than maxDomainExtent for the same Scale");
            }
            if (this._nonLinearScaleWithExtents(scale)) {
                utils$1.Window.warn("Panning and zooming with extents on a nonlinear scale may have unintended behavior.");
            }
            this._minDomainExtents.set(scale, minDomainExtent);
            return this;
        };
        PanZoom.prototype.maxDomainExtent = function (scale, maxDomainExtent) {
            if (maxDomainExtent == null) {
                return this._maxDomainExtents.get(scale);
            }
            if (maxDomainExtent.valueOf() <= 0) {
                throw new Error("extent must be positive");
            }
            var minExtentForScale = this.minDomainExtent(scale);
            if (minExtentForScale != null && maxDomainExtent.valueOf() < minExtentForScale.valueOf()) {
                throw new Error("maxDomainExtent must be larger than minDomainExtent for the same Scale");
            }
            if (this._nonLinearScaleWithExtents(scale)) {
                utils$1.Window.warn("Panning and zooming with extents on a nonlinear scale may have unintended behavior.");
            }
            this._maxDomainExtents.set(scale, maxDomainExtent);
            return this;
        };
        PanZoom.prototype.minDomainValue = function (scale, minDomainValue) {
            if (minDomainValue == null) {
                return this._minDomainValues.get(scale);
            }
            this._minDomainValues.set(scale, minDomainValue);
            return this;
        };
        PanZoom.prototype.maxDomainValue = function (scale, maxDomainValue) {
            if (maxDomainValue == null) {
                return this._maxDomainValues.get(scale);
            }
            this._maxDomainValues.set(scale, maxDomainValue);
            return this;
        };
        /**
         * Uses the current domain of the scale to apply a minimum and maximum
         * domain value for that scale.
         *
         * This constrains the pan/zoom interaction to show no more than the domain
         * of the scale.
         */
        PanZoom.prototype.setMinMaxDomainValuesTo = function (scale) {
            this._minDomainValues.delete(scale);
            this._maxDomainValues.delete(scale);
            var _a = scale.getTransformationDomain(), domainMin = _a[0], domainMax = _a[1];
            this.minDomainValue(scale, domainMin);
            this.maxDomainValue(scale, domainMax);
            return this;
        };
        /**
         * Adds a callback to be called when panning ends.
         *
         * @param {PanCallback} callback
         * @returns {this} The calling PanZoom Interaction.
         */
        PanZoom.prototype.onPanEnd = function (callback) {
            this._panEndCallbacks.add(callback);
            return this;
        };
        /**
         * Removes a callback that would be called when panning ends.
         *
         * @param {PanCallback} callback
         * @returns {this} The calling PanZoom Interaction.
         */
        PanZoom.prototype.offPanEnd = function (callback) {
            this._panEndCallbacks.delete(callback);
            return this;
        };
        /**
         * Adds a callback to be called when zooming ends.
         *
         * @param {ZoomCallback} callback
         * @returns {this} The calling PanZoom Interaction.
         */
        PanZoom.prototype.onZoomEnd = function (callback) {
            this._zoomEndCallbacks.add(callback);
            return this;
        };
        /**
         * Removes a callback that would be called when zooming ends.
         *
         * @param {ZoomCallback} callback
         * @returns {this} The calling PanZoom Interaction.
         */
        PanZoom.prototype.offZoomEnd = function (callback) {
            this._zoomEndCallbacks.delete(callback);
            return this;
        };
        /**
         * Adds a callback to be called when any pan or zoom update occurs. This is
         * called on every frame, so be sure your callback is fast.
         *
         * @param {PanZoomUpdateCallback} callback
         * @returns {this} The calling PanZoom Interaction.
         */
        PanZoom.prototype.onPanZoomUpdate = function (callback) {
            this._panZoomUpdateCallbacks.add(callback);
            return this;
        };
        /**
         * Removes a callback that would be called when any pan or zoom update occurs.
         *
         * @param {PanZoomUpdateCallback} callback
         * @returns {this} The calling PanZoom Interaction.
         */
        PanZoom.prototype.offPanZoomUpdate = function (callback) {
            this._panZoomUpdateCallbacks.delete(callback);
            return this;
        };
        /**
         * The number of pixels occupied in a line.
         */
        PanZoom._PIXELS_PER_LINE = 120;
        return PanZoom;
    }(interaction.Interaction));
    exports.PanZoom = PanZoom;
    });

    var pointerInteraction = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });




    var Pointer = /** @class */ (function (_super) {
        tslib_es6.__extends(Pointer, _super);
        function Pointer() {
            var _this = _super !== null && _super.apply(this, arguments) || this;
            _this._overComponent = false;
            _this._pointerEnterCallbacks = new utils$1.CallbackSet();
            _this._pointerMoveCallbacks = new utils$1.CallbackSet();
            _this._pointerExitCallbacks = new utils$1.CallbackSet();
            _this._mouseMoveCallback = function (p, e) { return _this._handleMouseEvent(p, e); };
            _this._touchStartCallback = function (ids, idToPoint, e) { return _this._handleTouchEvent(idToPoint[ids[0]], e); };
            return _this;
        }
        Pointer.prototype._anchor = function (component) {
            _super.prototype._anchor.call(this, component);
            this._mouseDispatcher = dispatchers.Mouse.getDispatcher(this._componentAttachedTo);
            this._mouseDispatcher.onMouseMove(this._mouseMoveCallback);
            this._touchDispatcher = dispatchers.Touch.getDispatcher(this._componentAttachedTo);
            this._touchDispatcher.onTouchStart(this._touchStartCallback);
        };
        Pointer.prototype._unanchor = function () {
            _super.prototype._unanchor.call(this);
            this._mouseDispatcher.offMouseMove(this._mouseMoveCallback);
            this._mouseDispatcher = null;
            this._touchDispatcher.offTouchStart(this._touchStartCallback);
            this._touchDispatcher = null;
        };
        Pointer.prototype._handleMouseEvent = function (p, e) {
            var insideSVG = this._mouseDispatcher.eventInside(this._componentAttachedTo, e);
            this._handlePointerEvent(p, insideSVG);
        };
        Pointer.prototype._handleTouchEvent = function (p, e) {
            var insideSVG = this._touchDispatcher.eventInside(this._componentAttachedTo, e);
            this._handlePointerEvent(p, insideSVG);
        };
        Pointer.prototype._handlePointerEvent = function (p, insideSVG) {
            var translatedP = this._translateToComponentSpace(p);
            var overComponent = this._isInsideComponent(translatedP);
            if (overComponent && insideSVG) {
                if (!this._overComponent) {
                    this._pointerEnterCallbacks.callCallbacks(translatedP);
                }
                this._pointerMoveCallbacks.callCallbacks(translatedP);
            }
            else if (this._overComponent) {
                this._pointerExitCallbacks.callCallbacks(translatedP);
            }
            this._overComponent = overComponent && insideSVG;
        };
        /**
         * Adds a callback to be called when the pointer enters the Component.
         *
         * @param {PointerCallback} callback
         * @return {Interactions.Pointer} The calling Pointer Interaction.
         */
        Pointer.prototype.onPointerEnter = function (callback) {
            this._pointerEnterCallbacks.add(callback);
            return this;
        };
        /**
         * Removes a callback that would be called when the pointer enters the Component.
         *
         * @param {PointerCallback} callback
         * @return {Interactions.Pointer} The calling Pointer Interaction.
         */
        Pointer.prototype.offPointerEnter = function (callback) {
            this._pointerEnterCallbacks.delete(callback);
            return this;
        };
        /**
         * Adds a callback to be called when the pointer moves within the Component.
         *
         * @param {PointerCallback} callback
         * @return {Interactions.Pointer} The calling Pointer Interaction.
         */
        Pointer.prototype.onPointerMove = function (callback) {
            this._pointerMoveCallbacks.add(callback);
            return this;
        };
        /**
         * Removes a callback that would be called when the pointer moves within the Component.
         *
         * @param {PointerCallback} callback
         * @return {Interactions.Pointer} The calling Pointer Interaction.
         */
        Pointer.prototype.offPointerMove = function (callback) {
            this._pointerMoveCallbacks.delete(callback);
            return this;
        };
        /**
         * Adds a callback to be called when the pointer exits the Component.
         *
         * @param {PointerCallback} callback
         * @return {Interactions.Pointer} The calling Pointer Interaction.
         */
        Pointer.prototype.onPointerExit = function (callback) {
            this._pointerExitCallbacks.add(callback);
            return this;
        };
        /**
         * Removes a callback that would be called when the pointer exits the Component.
         *
         * @param {PointerCallback} callback
         * @return {Interactions.Pointer} The calling Pointer Interaction.
         */
        Pointer.prototype.offPointerExit = function (callback) {
            this._pointerExitCallbacks.delete(callback);
            return this;
        };
        return Pointer;
    }(interaction.Interaction));
    exports.Pointer = Pointer;
    });

    var interactions = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });

    tslib_es6.__exportStar(clickInteraction, exports);
    tslib_es6.__exportStar(dragInteraction, exports);
    tslib_es6.__exportStar(keyInteraction, exports);
    tslib_es6.__exportStar(panZoomInteraction, exports);
    tslib_es6.__exportStar(pointerInteraction, exports);

    exports.zoomOut = panZoomConstraints.zoomOut;
    });

    var selectionBoxLayer = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });



    var PropertyMode;
    (function (PropertyMode) {
        PropertyMode[PropertyMode["VALUE"] = 0] = "VALUE";
        PropertyMode[PropertyMode["PIXEL"] = 1] = "PIXEL";
    })(PropertyMode = exports.PropertyMode || (exports.PropertyMode = {}));
    var SelectionBoxLayer = /** @class */ (function (_super) {
        tslib_es6.__extends(SelectionBoxLayer, _super);
        function SelectionBoxLayer() {
            var _this = _super.call(this) || this;
            _this._boxVisible = false;
            _this._boxBounds = {
                topLeft: { x: 0, y: 0 },
                bottomRight: { x: 0, y: 0 },
            };
            _this._xBoundsMode = PropertyMode.PIXEL;
            _this._yBoundsMode = PropertyMode.PIXEL;
            _this.addClass("selection-box-layer");
            _this._adjustBoundsCallback = function () {
                _this.render();
            };
            _this._overflowHidden = true;
            _this._xExtent = [undefined, undefined];
            _this._yExtent = [undefined, undefined];
            return _this;
        }
        SelectionBoxLayer.prototype._setup = function () {
            _super.prototype._setup.call(this);
            this._box = this.content().append("g").classed("selection-box", true).remove();
            this._boxArea = this._box.append("rect").classed("selection-area", true);
        };
        SelectionBoxLayer.prototype._sizeFromOffer = function (availableWidth, availableHeight) {
            return {
                width: availableWidth,
                height: availableHeight,
            };
        };
        SelectionBoxLayer.prototype.bounds = function (newBounds) {
            if (newBounds == null) {
                return this._getBounds();
            }
            this._setBounds(newBounds);
            this._xBoundsMode = PropertyMode.PIXEL;
            this._yBoundsMode = PropertyMode.PIXEL;
            this.render();
            return this;
        };
        SelectionBoxLayer.prototype._setBounds = function (newBounds) {
            var topLeft = {
                x: Math.min(newBounds.topLeft.x, newBounds.bottomRight.x),
                y: Math.min(newBounds.topLeft.y, newBounds.bottomRight.y),
            };
            var bottomRight = {
                x: Math.max(newBounds.topLeft.x, newBounds.bottomRight.x),
                y: Math.max(newBounds.topLeft.y, newBounds.bottomRight.y),
            };
            this._boxBounds = {
                topLeft: topLeft,
                bottomRight: bottomRight,
            };
        };
        SelectionBoxLayer.prototype._getBounds = function () {
            return {
                topLeft: {
                    x: this._xBoundsMode === PropertyMode.PIXEL ?
                        this._boxBounds.topLeft.x :
                        (this._xScale == null ?
                            0 :
                            Math.min(this.xScale().scale(this.xExtent()[0]), this.xScale().scale(this.xExtent()[1]))),
                    y: this._yBoundsMode === PropertyMode.PIXEL ?
                        this._boxBounds.topLeft.y :
                        (this._yScale == null ?
                            0 :
                            Math.min(this.yScale().scale(this.yExtent()[0]), this.yScale().scale(this.yExtent()[1]))),
                },
                bottomRight: {
                    x: this._xBoundsMode === PropertyMode.PIXEL ?
                        this._boxBounds.bottomRight.x :
                        (this._xScale == null ?
                            0 :
                            Math.max(this.xScale().scale(this.xExtent()[0]), this.xScale().scale(this.xExtent()[1]))),
                    y: this._yBoundsMode === PropertyMode.PIXEL ?
                        this._boxBounds.bottomRight.y :
                        (this._yScale == null ?
                            0 :
                            Math.max(this.yScale().scale(this.yExtent()[0]), this.yScale().scale(this.yExtent()[1]))),
                },
            };
        };
        SelectionBoxLayer.prototype.renderImmediately = function () {
            _super.prototype.renderImmediately.call(this);
            if (this._boxVisible) {
                var bounds = this.bounds();
                var t = bounds.topLeft.y;
                var b = bounds.bottomRight.y;
                var l = bounds.topLeft.x;
                var r = bounds.bottomRight.x;
                if (!(utils$1.Math.isValidNumber(t) &&
                    utils$1.Math.isValidNumber(b) &&
                    utils$1.Math.isValidNumber(l) &&
                    utils$1.Math.isValidNumber(r))) {
                    throw new Error("bounds have not been properly set");
                }
                this._boxArea.attrs({
                    x: l, y: t, width: r - l, height: b - t,
                });
                this.content().node().appendChild(this._box.node());
            }
            else {
                this._box.remove();
            }
            return this;
        };
        SelectionBoxLayer.prototype.boxVisible = function (show) {
            if (show == null) {
                return this._boxVisible;
            }
            this._boxVisible = show;
            this.render();
            return this;
        };
        SelectionBoxLayer.prototype.fixedWidth = function () {
            return true;
        };
        SelectionBoxLayer.prototype.fixedHeight = function () {
            return true;
        };
        SelectionBoxLayer.prototype.xScale = function (xScale) {
            if (xScale == null) {
                return this._xScale;
            }
            if (this._xScale != null) {
                this._xScale.offUpdate(this._adjustBoundsCallback);
            }
            this._xScale = xScale;
            this._xBoundsMode = PropertyMode.VALUE;
            this._xScale.onUpdate(this._adjustBoundsCallback);
            this.render();
            return this;
        };
        SelectionBoxLayer.prototype.yScale = function (yScale) {
            if (yScale == null) {
                return this._yScale;
            }
            if (this._yScale != null) {
                this._yScale.offUpdate(this._adjustBoundsCallback);
            }
            this._yScale = yScale;
            this._yBoundsMode = PropertyMode.VALUE;
            this._yScale.onUpdate(this._adjustBoundsCallback);
            this.render();
            return this;
        };
        SelectionBoxLayer.prototype.xExtent = function (xExtent) {
            // Explicit typing for Typescript 1.4
            if (xExtent == null) {
                return this._getXExtent();
            }
            this._setXExtent(xExtent);
            this._xBoundsMode = PropertyMode.VALUE;
            this.render();
            return this;
        };
        SelectionBoxLayer.prototype._getXExtent = function () {
            return this._xBoundsMode === PropertyMode.VALUE ?
                this._xExtent :
                (this._xScale == null ?
                    [undefined, undefined] :
                    [
                        this._xScale.invert(this._boxBounds.topLeft.x),
                        this._xScale.invert(this._boxBounds.bottomRight.x),
                    ]);
        };
        SelectionBoxLayer.prototype._setXExtent = function (xExtent) {
            this._xExtent = xExtent;
        };
        SelectionBoxLayer.prototype.yExtent = function (yExtent) {
            // Explicit typing for Typescript 1.4
            if (yExtent == null) {
                return this._getYExtent();
            }
            this._setYExtent(yExtent);
            this._yBoundsMode = PropertyMode.VALUE;
            this.render();
            return this;
        };
        SelectionBoxLayer.prototype._getYExtent = function () {
            return this._yBoundsMode === PropertyMode.VALUE ?
                this._yExtent :
                (this._yScale == null ?
                    [undefined, undefined] :
                    [
                        this._yScale.invert(this._boxBounds.topLeft.y),
                        this._yScale.invert(this._boxBounds.bottomRight.y),
                    ]);
        };
        SelectionBoxLayer.prototype._setYExtent = function (yExtent) {
            this._yExtent = yExtent;
        };
        SelectionBoxLayer.prototype.destroy = function () {
            _super.prototype.destroy.call(this);
            if (this._xScale != null) {
                this.xScale().offUpdate(this._adjustBoundsCallback);
            }
            if (this._yScale != null) {
                this.yScale().offUpdate(this._adjustBoundsCallback);
            }
        };
        return SelectionBoxLayer;
    }(component.Component));
    exports.SelectionBoxLayer = SelectionBoxLayer;
    });

    var dragBoxLayer = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });






    var DragBoxLayer = /** @class */ (function (_super) {
        tslib_es6.__extends(DragBoxLayer, _super);
        /**
         * Constructs a DragBoxLayer.
         *
         * A DragBoxLayer is a SelectionBoxLayer with a built-in Drag Interaction.
         * A drag gesture will set the Bounds of the box.
         * If resizing is enabled using resizable(true), the edges of box can be repositioned.
         *
         * @constructor
         */
        function DragBoxLayer() {
            var _this = _super.call(this) || this;
            _this._detectionRadius = 3;
            _this._resizable = false;
            _this._movable = false;
            _this._hasCorners = true;
            _this.addClass("drag-box-layer");
            _this._dragInteraction = new interactions.Drag();
            _this._setUpCallbacks();
            _this._dragInteraction.attachTo(_this);
            _this._dragStartCallbacks = new utils$1.CallbackSet();
            _this._dragCallbacks = new utils$1.CallbackSet();
            _this._dragEndCallbacks = new utils$1.CallbackSet();
            return _this;
        }
        DragBoxLayer.prototype._setUpCallbacks = function () {
            var _this = this;
            var resizingEdges;
            var topLeft;
            var bottomRight;
            var lastEndPoint;
            var DRAG_MODES = {
                newBox: 0,
                resize: 1,
                move: 2,
            };
            var mode = DRAG_MODES.newBox;
            var onDragStartCallback = function (startPoint) {
                resizingEdges = _this._getResizingEdges(startPoint);
                var bounds = _this.bounds();
                var isInsideBox = bounds.topLeft.x <= startPoint.x && startPoint.x <= bounds.bottomRight.x &&
                    bounds.topLeft.y <= startPoint.y && startPoint.y <= bounds.bottomRight.y;
                if (_this.boxVisible() && (resizingEdges.top || resizingEdges.bottom || resizingEdges.left || resizingEdges.right)) {
                    mode = DRAG_MODES.resize;
                }
                else if (_this.boxVisible() && _this.movable() && isInsideBox) {
                    mode = DRAG_MODES.move;
                }
                else {
                    mode = DRAG_MODES.newBox;
                    _this._setBounds({
                        topLeft: startPoint,
                        bottomRight: startPoint,
                    });
                    if (_this._xBoundsMode === components.PropertyMode.VALUE && _this.xScale() != null) {
                        _this._setXExtent([_this.xScale().invert(startPoint.x), _this.xScale().invert(startPoint.x)]);
                    }
                    if (_this._yBoundsMode === components.PropertyMode.VALUE && _this.yScale() != null) {
                        _this._setYExtent([_this.yScale().invert(startPoint.y), _this.yScale().invert(startPoint.y)]);
                    }
                    _this.render();
                }
                _this.boxVisible(true);
                bounds = _this.bounds();
                // copy points so changes to topLeft and bottomRight don't mutate bounds
                topLeft = { x: bounds.topLeft.x, y: bounds.topLeft.y };
                bottomRight = { x: bounds.bottomRight.x, y: bounds.bottomRight.y };
                lastEndPoint = startPoint;
                _this._dragStartCallbacks.callCallbacks(bounds);
            };
            var onDragCallback = function (startPoint, endPoint) {
                switch (mode) {
                    case DRAG_MODES.newBox:
                        bottomRight.x = endPoint.x;
                        bottomRight.y = endPoint.y;
                        break;
                    case DRAG_MODES.resize:
                        if (resizingEdges.bottom) {
                            bottomRight.y = endPoint.y;
                        }
                        else if (resizingEdges.top) {
                            topLeft.y = endPoint.y;
                        }
                        if (resizingEdges.right) {
                            bottomRight.x = endPoint.x;
                        }
                        else if (resizingEdges.left) {
                            topLeft.x = endPoint.x;
                        }
                        break;
                    case DRAG_MODES.move:
                        var dx = endPoint.x - lastEndPoint.x;
                        var dy = endPoint.y - lastEndPoint.y;
                        topLeft.x += dx;
                        topLeft.y += dy;
                        bottomRight.x += dx;
                        bottomRight.y += dy;
                        lastEndPoint = endPoint;
                        break;
                }
                _this._setBounds({
                    topLeft: topLeft,
                    bottomRight: bottomRight,
                });
                if (_this._xBoundsMode === components.PropertyMode.VALUE && _this.xScale() != null) {
                    _this._setXExtent([_this.xScale().invert(topLeft.x), _this.xScale().invert(bottomRight.x)]);
                }
                if (_this._yBoundsMode === components.PropertyMode.VALUE && _this.yScale() != null) {
                    _this._setYExtent([_this.yScale().invert(topLeft.y), _this.yScale().invert(bottomRight.y)]);
                }
                _this.render();
                _this._dragCallbacks.callCallbacks(_this.bounds());
            };
            var onDragEndCallback = function (startPoint, endPoint) {
                if (mode === DRAG_MODES.newBox && startPoint.x === endPoint.x && startPoint.y === endPoint.y) {
                    _this.boxVisible(false);
                }
                _this._dragEndCallbacks.callCallbacks(_this.bounds());
            };
            this._dragInteraction.onDragStart(onDragStartCallback);
            this._dragInteraction.onDrag(onDragCallback);
            this._dragInteraction.onDragEnd(onDragEndCallback);
            this._disconnectInteraction = function () {
                _this._dragInteraction.offDragStart(onDragStartCallback);
                _this._dragInteraction.offDrag(onDragCallback);
                _this._dragInteraction.offDragEnd(onDragEndCallback);
                _this._dragInteraction.detach();
            };
        };
        DragBoxLayer.prototype._setup = function () {
            var _this = this;
            _super.prototype._setup.call(this);
            var createLine = function () { return _this._box.append("line").styles({
                opacity: 0,
                stroke: "pink",
                "pointer-events": "visibleStroke",
            }); };
            this._detectionEdgeT = createLine().classed("drag-edge-tb", true);
            this._detectionEdgeB = createLine().classed("drag-edge-tb", true);
            this._detectionEdgeL = createLine().classed("drag-edge-lr", true);
            this._detectionEdgeR = createLine().classed("drag-edge-lr", true);
            if (this._hasCorners) {
                var createCorner = function () { return _this._box.append("circle")
                    .styles({
                    opacity: 0,
                    fill: "pink",
                    "pointer-events": "visibleFill",
                }); };
                this._detectionCornerTL = createCorner().classed("drag-corner-tl", true);
                this._detectionCornerTR = createCorner().classed("drag-corner-tr", true);
                this._detectionCornerBL = createCorner().classed("drag-corner-bl", true);
                this._detectionCornerBR = createCorner().classed("drag-corner-br", true);
            }
        };
        DragBoxLayer.prototype._getResizingEdges = function (p) {
            var edges = {
                top: false,
                bottom: false,
                left: false,
                right: false,
            };
            if (!this.resizable()) {
                return edges;
            }
            var bounds = this.bounds();
            var t = bounds.topLeft.y;
            var b = bounds.bottomRight.y;
            var l = bounds.topLeft.x;
            var r = bounds.bottomRight.x;
            var rad = this._detectionRadius;
            if (l - rad <= p.x && p.x <= r + rad) {
                edges.top = (t - rad <= p.y && p.y <= t + rad);
                edges.bottom = (b - rad <= p.y && p.y <= b + rad);
            }
            if (t - rad <= p.y && p.y <= b + rad) {
                edges.left = (l - rad <= p.x && p.x <= l + rad);
                edges.right = (r - rad <= p.x && p.x <= r + rad);
            }
            return edges;
        };
        DragBoxLayer.prototype.renderImmediately = function () {
            _super.prototype.renderImmediately.call(this);
            if (this.boxVisible()) {
                var bounds = this.bounds();
                var t = bounds.topLeft.y;
                var b = bounds.bottomRight.y;
                var l = bounds.topLeft.x;
                var r = bounds.bottomRight.x;
                this._detectionEdgeT.attrs({
                    x1: l, y1: t, x2: r, y2: t,
                    "stroke-width": this._detectionRadius * 2,
                });
                this._detectionEdgeB.attrs({
                    x1: l, y1: b, x2: r, y2: b,
                    "stroke-width": this._detectionRadius * 2,
                });
                this._detectionEdgeL.attrs({
                    x1: l, y1: t, x2: l, y2: b,
                    "stroke-width": this._detectionRadius * 2,
                });
                this._detectionEdgeR.attrs({
                    x1: r, y1: t, x2: r, y2: b,
                    "stroke-width": this._detectionRadius * 2,
                });
                if (this._hasCorners) {
                    this._detectionCornerTL.attrs({ cx: l, cy: t, r: this._detectionRadius });
                    this._detectionCornerTR.attrs({ cx: r, cy: t, r: this._detectionRadius });
                    this._detectionCornerBL.attrs({ cx: l, cy: b, r: this._detectionRadius });
                    this._detectionCornerBR.attrs({ cx: r, cy: b, r: this._detectionRadius });
                }
            }
            return this;
        };
        DragBoxLayer.prototype.detectionRadius = function (r) {
            if (r == null) {
                return this._detectionRadius;
            }
            if (r < 0) {
                throw new Error("detection radius cannot be negative.");
            }
            this._detectionRadius = r;
            this.render();
            return this;
        };
        DragBoxLayer.prototype.resizable = function (canResize) {
            if (canResize == null) {
                return this._resizable;
            }
            this._resizable = canResize;
            this._setResizableClasses(canResize);
            return this;
        };
        // Sets resizable classes. Overridden by subclasses that only resize in one dimension.
        DragBoxLayer.prototype._setResizableClasses = function (canResize) {
            if (canResize && this.enabled()) {
                this.addClass("x-resizable");
                this.addClass("y-resizable");
            }
            else {
                this.removeClass("x-resizable");
                this.removeClass("y-resizable");
            }
        };
        DragBoxLayer.prototype.movable = function (movable) {
            if (movable == null) {
                return this._movable;
            }
            this._movable = movable;
            this._setMovableClass();
            return this;
        };
        DragBoxLayer.prototype._setMovableClass = function () {
            if (this.movable() && this.enabled()) {
                this.addClass("movable");
            }
            else {
                this.removeClass("movable");
            }
        };
        /**
         * Sets the callback to be called when dragging starts.
         *
         * @param {DragBoxCallback} callback
         * @returns {DragBoxLayer} The calling DragBoxLayer.
         */
        DragBoxLayer.prototype.onDragStart = function (callback) {
            this._dragStartCallbacks.add(callback);
            return this;
        };
        /**
         * Removes a callback to be called when dragging starts.
         *
         * @param {DragBoxCallback} callback
         * @returns {DragBoxLayer} The calling DragBoxLayer.
         */
        DragBoxLayer.prototype.offDragStart = function (callback) {
            this._dragStartCallbacks.delete(callback);
            return this;
        };
        /**
         * Sets a callback to be called during dragging.
         *
         * @param {DragBoxCallback} callback
         * @returns {DragBoxLayer} The calling DragBoxLayer.
         */
        DragBoxLayer.prototype.onDrag = function (callback) {
            this._dragCallbacks.add(callback);
            return this;
        };
        /**
         * Removes a callback to be called during dragging.
         *
         * @param {DragBoxCallback} callback
         * @returns {DragBoxLayer} The calling DragBoxLayer.
         */
        DragBoxLayer.prototype.offDrag = function (callback) {
            this._dragCallbacks.delete(callback);
            return this;
        };
        /**
         * Sets a callback to be called when dragging ends.
         *
         * @param {DragBoxCallback} callback
         * @returns {DragBoxLayer} The calling DragBoxLayer.
         */
        DragBoxLayer.prototype.onDragEnd = function (callback) {
            this._dragEndCallbacks.add(callback);
            return this;
        };
        /**
         * Removes a callback to be called when dragging ends.
         *
         * @param {DragBoxCallback} callback
         * @returns {DragBoxLayer} The calling DragBoxLayer.
         */
        DragBoxLayer.prototype.offDragEnd = function (callback) {
            this._dragEndCallbacks.delete(callback);
            return this;
        };
        /**
         * Gets the internal Interactions.Drag of the DragBoxLayer.
         */
        DragBoxLayer.prototype.dragInteraction = function () {
            return this._dragInteraction;
        };
        DragBoxLayer.prototype.enabled = function (enabled) {
            if (enabled == null) {
                return this._dragInteraction.enabled();
            }
            this._dragInteraction.enabled(enabled);
            this._setResizableClasses(this.resizable());
            this._setMovableClass();
            return this;
        };
        DragBoxLayer.prototype.destroy = function () {
            var _this = this;
            _super.prototype.destroy.call(this);
            this._dragStartCallbacks.forEach(function (callback) { return _this._dragCallbacks.delete(callback); });
            this._dragCallbacks.forEach(function (callback) { return _this._dragCallbacks.delete(callback); });
            this._dragEndCallbacks.forEach(function (callback) { return _this._dragEndCallbacks.delete(callback); });
            this._disconnectInteraction();
        };
        DragBoxLayer.prototype.detach = function () {
            this._resetState();
            this._dragInteraction.detach();
            _super.prototype.detach.call(this);
            return this;
        };
        DragBoxLayer.prototype.anchor = function (selection) {
            selection = coerceD3.coerceExternalD3(selection);
            this._dragInteraction.attachTo(this);
            _super.prototype.anchor.call(this, selection);
            return this;
        };
        DragBoxLayer.prototype._resetState = function () {
            this.bounds({
                topLeft: { x: 0, y: 0 },
                bottomRight: { x: 0, y: 0 },
            });
        };
        return DragBoxLayer;
    }(selectionBoxLayer.SelectionBoxLayer));
    exports.DragBoxLayer = DragBoxLayer;
    });

    var guideLineLayer = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });



    var PropertyMode;
    (function (PropertyMode) {
        PropertyMode[PropertyMode["VALUE"] = 0] = "VALUE";
        PropertyMode[PropertyMode["PIXEL"] = 1] = "PIXEL";
    })(PropertyMode || (PropertyMode = {}));
    var GuideLineLayer = /** @class */ (function (_super) {
        tslib_es6.__extends(GuideLineLayer, _super);
        function GuideLineLayer(orientation) {
            var _this = _super.call(this) || this;
            _this._mode = PropertyMode.VALUE;
            if (orientation !== GuideLineLayer.ORIENTATION_VERTICAL && orientation !== GuideLineLayer.ORIENTATION_HORIZONTAL) {
                throw new Error(orientation + " is not a valid orientation for GuideLineLayer");
            }
            _this._orientation = orientation;
            _this._overflowHidden = true;
            _this.addClass("guide-line-layer");
            if (_this._isVertical()) {
                _this.addClass("vertical");
            }
            else {
                _this.addClass("horizontal");
            }
            _this._scaleUpdateCallback = function () {
                _this._syncPixelPositionAndValue();
                _this.render();
            };
            return _this;
        }
        GuideLineLayer.prototype._setup = function () {
            _super.prototype._setup.call(this);
            this._guideLine = this.content().append("line").classed("guide-line", true);
        };
        GuideLineLayer.prototype._sizeFromOffer = function (availableWidth, availableHeight) {
            return {
                width: availableWidth,
                height: availableHeight,
            };
        };
        GuideLineLayer.prototype._isVertical = function () {
            return this._orientation === GuideLineLayer.ORIENTATION_VERTICAL;
        };
        GuideLineLayer.prototype.fixedWidth = function () {
            return true;
        };
        GuideLineLayer.prototype.fixedHeight = function () {
            return true;
        };
        GuideLineLayer.prototype.computeLayout = function (origin, availableWidth, availableHeight) {
            _super.prototype.computeLayout.call(this, origin, availableWidth, availableHeight);
            if (this.scale() != null) {
                if (this._isVertical()) {
                    this.scale().range([0, this.width()]);
                }
                else {
                    this.scale().range([this.height(), 0]);
                }
            }
            return this;
        };
        GuideLineLayer.prototype.renderImmediately = function () {
            _super.prototype.renderImmediately.call(this);
            this._syncPixelPositionAndValue();
            this._guideLine.attrs({
                x1: this._isVertical() ? this.pixelPosition() : 0,
                y1: this._isVertical() ? 0 : this.pixelPosition(),
                x2: this._isVertical() ? this.pixelPosition() : this.width(),
                y2: this._isVertical() ? this.height() : this.pixelPosition(),
            });
            return this;
        };
        // sets pixelPosition() or value() based on the other, depending on which was the last one set
        GuideLineLayer.prototype._syncPixelPositionAndValue = function () {
            if (this.scale() == null) {
                return;
            }
            if (this._mode === PropertyMode.VALUE && this.value() != null) {
                this._pixelPosition = this.scale().scale(this.value());
            }
            else if (this._mode === PropertyMode.PIXEL && this.pixelPosition() != null) {
                this._value = this.scale().invert(this.pixelPosition());
            }
        };
        GuideLineLayer.prototype._setPixelPositionWithoutChangingMode = function (pixelPosition) {
            this._pixelPosition = pixelPosition;
            if (this.scale() != null) {
                this._value = this.scale().invert(this.pixelPosition());
            }
            this.render();
        };
        GuideLineLayer.prototype.scale = function (scale) {
            if (scale == null) {
                return this._scale;
            }
            var previousScale = this._scale;
            if (previousScale != null) {
                previousScale.offUpdate(this._scaleUpdateCallback);
            }
            this._scale = scale;
            this._scale.onUpdate(this._scaleUpdateCallback);
            this._syncPixelPositionAndValue();
            this.redraw();
            return this;
        };
        GuideLineLayer.prototype.value = function (value) {
            if (value == null) {
                return this._value;
            }
            this._value = value;
            this._mode = PropertyMode.VALUE;
            this._syncPixelPositionAndValue();
            this.render();
            return this;
        };
        GuideLineLayer.prototype.pixelPosition = function (pixelPosition) {
            if (pixelPosition == null) {
                return this._pixelPosition;
            }
            if (!utils$1.Math.isValidNumber(pixelPosition)) {
                throw new Error("pixelPosition must be a finite number");
            }
            this._pixelPosition = pixelPosition;
            this._mode = PropertyMode.PIXEL;
            this._syncPixelPositionAndValue();
            this.render();
            return this;
        };
        GuideLineLayer.prototype.destroy = function () {
            _super.prototype.destroy.call(this);
            if (this.scale() != null) {
                this.scale().offUpdate(this._scaleUpdateCallback);
            }
        };
        GuideLineLayer.ORIENTATION_VERTICAL = "vertical";
        GuideLineLayer.ORIENTATION_HORIZONTAL = "horizontal";
        return GuideLineLayer;
    }(component.Component));
    exports.GuideLineLayer = GuideLineLayer;
    });

    var dragLineLayer = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });




    var DragLineLayer = /** @class */ (function (_super) {
        tslib_es6.__extends(DragLineLayer, _super);
        function DragLineLayer(orientation) {
            var _this = _super.call(this, orientation) || this;
            _this._detectionRadius = 3;
            _this._enabled = true;
            _this.addClass("drag-line-layer");
            _this.addClass("enabled");
            _this._dragInteraction = new interactions.Drag();
            _this._dragInteraction.attachTo(_this);
            var onLine = function (p) {
                return (_this._isVertical() &&
                    _this.pixelPosition() - _this.detectionRadius() <= p.x &&
                    p.x <= _this.pixelPosition() + _this.detectionRadius()) ||
                    (!_this._isVertical() &&
                        _this.pixelPosition() - _this.detectionRadius() <= p.y &&
                        p.y <= _this.pixelPosition() + _this.detectionRadius());
            };
            var dragging = false;
            var interactionDragStartCallback = function (start) {
                if (onLine(start)) {
                    dragging = true;
                    _this._dragStartCallbacks.callCallbacks(_this);
                }
            };
            _this._dragInteraction.onDragStart(interactionDragStartCallback);
            var interactionDragCallback = function (start, end) {
                if (dragging) {
                    _this._setPixelPositionWithoutChangingMode(_this._isVertical() ? end.x : end.y);
                    _this._dragCallbacks.callCallbacks(_this);
                }
            };
            _this._dragInteraction.onDrag(interactionDragCallback);
            var interactionDragEndCallback = function (start, end) {
                if (dragging) {
                    dragging = false;
                    _this._dragEndCallbacks.callCallbacks(_this);
                }
            };
            _this._dragInteraction.onDragEnd(interactionDragEndCallback);
            _this._disconnectInteraction = function () {
                _this._dragInteraction.offDragStart(interactionDragStartCallback);
                _this._dragInteraction.offDrag(interactionDragCallback);
                _this._dragInteraction.offDragEnd(interactionDragEndCallback);
                _this._dragInteraction.detach();
            };
            _this._dragStartCallbacks = new utils$1.CallbackSet();
            _this._dragCallbacks = new utils$1.CallbackSet();
            _this._dragEndCallbacks = new utils$1.CallbackSet();
            return _this;
        }
        DragLineLayer.prototype._setup = function () {
            _super.prototype._setup.call(this);
            this._detectionEdge = this.content().append("line").styles({
                opacity: 0,
                stroke: "pink",
                "pointer-events": "visibleStroke",
            }).classed("drag-edge", true);
        };
        DragLineLayer.prototype.renderImmediately = function () {
            _super.prototype.renderImmediately.call(this);
            this._detectionEdge.attrs({
                x1: this._isVertical() ? this.pixelPosition() : 0,
                y1: this._isVertical() ? 0 : this.pixelPosition(),
                x2: this._isVertical() ? this.pixelPosition() : this.width(),
                y2: this._isVertical() ? this.height() : this.pixelPosition(),
                "stroke-width": this._detectionRadius * 2,
            });
            return this;
        };
        DragLineLayer.prototype.detectionRadius = function (detectionRadius) {
            if (detectionRadius == null) {
                return this._detectionRadius;
            }
            if (detectionRadius < 0) {
                throw new Error("detection radius cannot be negative.");
            }
            this._detectionRadius = detectionRadius;
            this.render();
            return this;
        };
        DragLineLayer.prototype.enabled = function (enabled) {
            if (enabled == null) {
                return this._enabled;
            }
            this._enabled = enabled;
            if (enabled) {
                this.addClass("enabled");
            }
            else {
                this.removeClass("enabled");
            }
            this._dragInteraction.enabled(enabled);
            return this;
        };
        /**
         * Sets the callback to be called when dragging starts.
         * The callback will be passed the calling DragLineLayer.
         *
         * @param {DragLineCallback<D>} callback
         * @returns {DragLineLayer<D>} The calling DragLineLayer.
         */
        DragLineLayer.prototype.onDragStart = function (callback) {
            this._dragStartCallbacks.add(callback);
            return this;
        };
        /**
         * Removes a callback that would be called when dragging starts.
         *
         * @param {DragLineCallback<D>} callback
         * @returns {DragLineLayer<D>} The calling DragLineLayer.
         */
        DragLineLayer.prototype.offDragStart = function (callback) {
            this._dragStartCallbacks.delete(callback);
            return this;
        };
        /**
         * Sets a callback to be called during dragging.
         * The callback will be passed the calling DragLineLayer.
         *
         * @param {DragLineCallback<D>} callback
         * @returns {DragLineLayer<D>} The calling DragLineLayer.
         */
        DragLineLayer.prototype.onDrag = function (callback) {
            this._dragCallbacks.add(callback);
            return this;
        };
        /**
         * Removes a callback that would be called during dragging.
         *
         * @param {DragLineCallback<D>} callback
         * @returns {DragLineLayer<D>} The calling DragLineLayer.
         */
        DragLineLayer.prototype.offDrag = function (callback) {
            this._dragCallbacks.delete(callback);
            return this;
        };
        /**
         * Sets a callback to be called when dragging ends.
         * The callback will be passed the calling DragLineLayer.
         *
         * @param {DragLineCallback<D>} callback
         * @returns {DragLineLayer<D>} The calling DragLineLayer.
         */
        DragLineLayer.prototype.onDragEnd = function (callback) {
            this._dragEndCallbacks.add(callback);
            return this;
        };
        /**
         * Removes a callback that would be called when dragging ends.
         *
         * @param {DragLineCallback<D>} callback
         * @returns {DragLineLayer<D>} The calling DragLineLayer.
         */
        DragLineLayer.prototype.offDragEnd = function (callback) {
            this._dragEndCallbacks.delete(callback);
            return this;
        };
        DragLineLayer.prototype.destroy = function () {
            var _this = this;
            _super.prototype.destroy.call(this);
            this._dragStartCallbacks.forEach(function (callback) { return _this._dragStartCallbacks.delete(callback); });
            this._dragCallbacks.forEach(function (callback) { return _this._dragCallbacks.delete(callback); });
            this._dragEndCallbacks.forEach(function (callback) { return _this._dragEndCallbacks.delete(callback); });
            this._disconnectInteraction();
        };
        return DragLineLayer;
    }(guideLineLayer.GuideLineLayer));
    exports.DragLineLayer = DragLineLayer;
    });

    var gridlines = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });


    /**
     * Returns next grid position based on tick value
     *
     * @param scale Scale used by the grid
     * @param between Value denoting whether the grid renders between ticks or on ticks
     * @param orderedTicks All ticks in order. only needed when rendering lines between ticks
     */
    function gridPositionFactory(scale, between, orderedTicks) {
        var previousTick = {};
        if (orderedTicks !== undefined) {
            for (var i = 0; i < orderedTicks.length; i++) {
                var previous = orderedTicks[i - 1];
                var current = orderedTicks[i];
                previousTick[current] = previous;
            }
        }
        return function (tickVal) {
            var position = scale.scale(tickVal);
            if (!between) {
                return position;
            }
            var gridPosition;
            var previousPosition = previousTick[tickVal] === undefined
                ? undefined
                : scale.scale(previousTick[tickVal]);
            if (previousPosition !== undefined) {
                gridPosition = previousPosition + (position - previousPosition) / 2;
            }
            return gridPosition;
        };
    }
    var Gridlines = /** @class */ (function (_super) {
        tslib_es6.__extends(Gridlines, _super);
        /**
         * @constructor
         * @param {Scale} xScale The scale to base the x gridlines on. Pass null if no gridlines are desired.
         * @param {Scale} yScale The scale to base the y gridlines on. Pass null if no gridlines are desired.
         */
        function Gridlines(xScale, yScale) {
            var _this = _super.call(this) || this;
            _this.addClass("gridlines");
            _this._xScale = xScale;
            _this._yScale = yScale;
            _this._renderCallback = function (scale) { return _this.render(); };
            if (_this._xScale) {
                _this._xScale.onUpdate(_this._renderCallback);
            }
            if (_this._yScale) {
                _this._yScale.onUpdate(_this._renderCallback);
            }
            return _this;
        }
        Gridlines.prototype.betweenX = function (_betweenX) {
            if (_betweenX === undefined) {
                return this._betweenX;
            }
            if (_betweenX !== this._betweenX) {
                this._betweenX = _betweenX;
                this.render();
            }
            return this;
        };
        Gridlines.prototype.betweenY = function (_betweenY) {
            if (_betweenY === undefined) {
                return this._betweenY;
            }
            if (_betweenY !== this._betweenY) {
                this._betweenY = _betweenY;
                this.render();
            }
            return this;
        };
        Gridlines.prototype.destroy = function () {
            _super.prototype.destroy.call(this);
            if (this._xScale) {
                this._xScale.offUpdate(this._renderCallback);
            }
            if (this._yScale) {
                this._yScale.offUpdate(this._renderCallback);
            }
            return this;
        };
        Gridlines.prototype._setup = function () {
            _super.prototype._setup.call(this);
            this._xLinesContainer = this.content().append("g").classed("x-gridlines", true);
            this._yLinesContainer = this.content().append("g").classed("y-gridlines", true);
        };
        Gridlines.prototype.renderImmediately = function () {
            _super.prototype.renderImmediately.call(this);
            this._redrawXLines();
            this._redrawYLines();
            return this;
        };
        Gridlines.prototype.computeLayout = function (origin, availableWidth, availableHeight) {
            _super.prototype.computeLayout.call(this, origin, availableWidth, availableHeight);
            if (this._xScale != null) {
                this._xScale.range([0, this.width()]);
            }
            if (this._yScale != null) {
                this._yScale.range([this.height(), 0]);
            }
            return this;
        };
        Gridlines.prototype._redrawXLines = function () {
            if (this._xScale) {
                var between = this.betweenX();
                var xTicks = this._xScale.ticks().slice(between ? 1 : 0);
                var xLinesUpdate = this._xLinesContainer.selectAll("line").data(xTicks);
                var xLines = xLinesUpdate.enter().append("line").merge(xLinesUpdate);
                xLines.attr("x1", gridPositionFactory(this._xScale, between, this._xScale.ticks()))
                    .attr("y1", 0)
                    .attr("x2", gridPositionFactory(this._xScale, between, this._xScale.ticks()))
                    .attr("y2", this.height())
                    .classed("betweenline", between)
                    .classed("zeroline", function (t) { return t === 0; });
                xLinesUpdate.exit().remove();
            }
        };
        Gridlines.prototype._redrawYLines = function () {
            if (this._yScale) {
                var between = this.betweenY();
                var yTicks = this._yScale.ticks().slice(between ? 1 : 0);
                var yLinesUpdate = this._yLinesContainer.selectAll("line").data(yTicks);
                var yLines = yLinesUpdate.enter().append("line").merge(yLinesUpdate);
                yLines.attr("x1", 0)
                    .attr("y1", gridPositionFactory(this._yScale, between, this._yScale.ticks()))
                    .attr("x2", this.width())
                    .attr("y2", gridPositionFactory(this._yScale, between, this._yScale.ticks()))
                    .classed("betweenline", between)
                    .classed("zeroline", function (t) { return t === 0; });
                yLinesUpdate.exit().remove();
            }
        };
        return Gridlines;
    }(component.Component));
    exports.Gridlines = Gridlines;
    });

    var componentContainer = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });



    /*
     * ComponentContainer class encapsulates Table and ComponentGroup's shared functionality.
     * It will not do anything if instantiated directly.
     */
    var ComponentContainer = /** @class */ (function (_super) {
        tslib_es6.__extends(ComponentContainer, _super);
        function ComponentContainer() {
            var _this = _super.call(this) || this;
            _this._detachCallback = function (component) { return _this.remove(component); };
            return _this;
        }
        ComponentContainer.prototype.anchor = function (selection) {
            var _this = this;
            selection = coerceD3.coerceExternalD3(selection);
            _super.prototype.anchor.call(this, selection);
            this._forEach(function (c) { return c.anchor(_this.element()); });
            return this;
        };
        ComponentContainer.prototype.render = function () {
            this._forEach(function (c) { return c.render(); });
            return this;
        };
        /**
         * Checks whether the specified Component is in the ComponentContainer.
         */
        ComponentContainer.prototype.has = function (component) {
            throw new Error("has() is not implemented on ComponentContainer");
        };
        ComponentContainer.prototype._adoptAndAnchor = function (component) {
            component.parent(this);
            component.onDetach(this._detachCallback);
            if (this._isAnchored) {
                component.anchor(this.element());
            }
        };
        /**
         * Removes the specified Component from the ComponentContainer.
         */
        ComponentContainer.prototype.remove = function (component) {
            if (this.has(component)) {
                component.offDetach(this._detachCallback);
                this._remove(component);
                component.detach();
                this.redraw();
            }
            return this;
        };
        /**
         * Carry out the actual removal of a Component.
         * Implementation dependent on the type of container.
         *
         * @return {boolean} true if the Component was successfully removed, false otherwise.
         */
        ComponentContainer.prototype._remove = function (component) {
            return false;
        };
        /**
         * Invokes a callback on each Component in the ComponentContainer.
         */
        ComponentContainer.prototype._forEach = function (callback) {
            throw new Error("_forEach() is not implemented on ComponentContainer");
        };
        /**
         * Destroys the ComponentContainer and all Components within it.
         */
        ComponentContainer.prototype.destroy = function () {
            _super.prototype.destroy.call(this);
            this._forEach(function (c) { return c.destroy(); });
        };
        ComponentContainer.prototype.invalidateCache = function () {
            this._forEach(function (c) { return c.invalidateCache(); });
        };
        return ComponentContainer;
    }(component.Component));
    exports.ComponentContainer = ComponentContainer;
    });

    var group = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });



    var Group = /** @class */ (function (_super) {
        tslib_es6.__extends(Group, _super);
        /**
         * Constructs a Group.
         *
         * A Group contains Components that will be rendered on top of each other.
         * Components added later will be rendered above Components already in the Group.
         *
         * @constructor
         * @param {Component[]} [components=[]] Components to be added to the Group.
         */
        function Group(components) {
            if (components === void 0) { components = []; }
            var _this = _super.call(this) || this;
            _this._components = [];
            _this.addClass("component-group");
            components.forEach(function (c) { return _this.append(c); });
            return _this;
        }
        Group.prototype._forEach = function (callback) {
            this.components().forEach(callback);
        };
        /**
         * Checks whether the specified Component is in the Group.
         */
        Group.prototype.has = function (component) {
            return this._components.indexOf(component) >= 0;
        };
        Group.prototype.requestedSpace = function (offeredWidth, offeredHeight) {
            var requests = this._components.map(function (c) { return c.requestedSpace(offeredWidth, offeredHeight); });
            return {
                minWidth: utils$1.Math.max(requests, function (request) { return request.minWidth; }, 0),
                minHeight: utils$1.Math.max(requests, function (request) { return request.minHeight; }, 0),
            };
        };
        Group.prototype.computeLayout = function (origin, availableWidth, availableHeight) {
            var _this = this;
            _super.prototype.computeLayout.call(this, origin, availableWidth, availableHeight);
            this._forEach(function (component) {
                component.computeLayout({ x: 0, y: 0 }, _this.width(), _this.height());
            });
            return this;
        };
        Group.prototype._sizeFromOffer = function (availableWidth, availableHeight) {
            return {
                width: availableWidth,
                height: availableHeight,
            };
        };
        Group.prototype.fixedWidth = function () {
            return this._components.every(function (c) { return c.fixedWidth(); });
        };
        Group.prototype.fixedHeight = function () {
            return this._components.every(function (c) { return c.fixedHeight(); });
        };
        /**
         * @return {Component[]} The Components in this Group.
         */
        Group.prototype.components = function () {
            return this._components.slice();
        };
        /**
         * Adds a Component to this Group.
         * The added Component will be rendered above Components already in the Group.
         */
        Group.prototype.append = function (component) {
            if (component != null && !this.has(component)) {
                component.detach();
                this._components.push(component);
                this._adoptAndAnchor(component);
                this.redraw();
            }
            return this;
        };
        Group.prototype._remove = function (component) {
            var removeIndex = this._components.indexOf(component);
            if (removeIndex >= 0) {
                this._components.splice(removeIndex, 1);
                return true;
            }
            return false;
        };
        return Group;
    }(componentContainer.ComponentContainer));
    exports.Group = Group;
    });

    var interpolatedColorLegend = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });






    var InterpolatedColorLegend = /** @class */ (function (_super) {
        tslib_es6.__extends(InterpolatedColorLegend, _super);
        /**
         * Creates an InterpolatedColorLegend.
         *
         * The InterpolatedColorLegend consists of a sequence of swatches that show the
         * associated InterpolatedColor Scale sampled at various points.
         * Two labels show the maximum and minimum values of the InterpolatedColor Scale.
         *
         * @constructor
         * @param {Scales.InterpolatedColor} interpolatedColorScale
         */
        function InterpolatedColorLegend(interpolatedColorScale) {
            var _this = _super.call(this) || this;
            _this._textPadding = 5;
            if (interpolatedColorScale == null) {
                throw new Error("InterpolatedColorLegend requires a interpolatedColorScale");
            }
            _this._scale = interpolatedColorScale;
            _this._redrawCallback = function (scale) { return _this.redraw(); };
            _this._scale.onUpdate(_this._redrawCallback);
            _this._formatter = formatters.general();
            _this._orientation = "horizontal";
            _this._expands = false;
            _this.addClass("legend");
            _this.addClass("interpolated-color-legend");
            return _this;
        }
        InterpolatedColorLegend.prototype.destroy = function () {
            _super.prototype.destroy.call(this);
            this._scale.offUpdate(this._redrawCallback);
        };
        InterpolatedColorLegend.prototype.formatter = function (formatter) {
            if (formatter === undefined) {
                return this._formatter;
            }
            this._formatter = formatter;
            this.redraw();
            return this;
        };
        InterpolatedColorLegend.prototype.expands = function (expands) {
            if (expands == null) {
                return this._expands;
            }
            this._expands = expands;
            this.redraw();
            return this;
        };
        InterpolatedColorLegend._ensureOrientation = function (orientation) {
            orientation = orientation.toLowerCase();
            if (orientation === "horizontal" || orientation === "left" || orientation === "right") {
                return orientation;
            }
            else {
                throw new Error("\"" + orientation + "\" is not a valid orientation for InterpolatedColorLegend");
            }
        };
        InterpolatedColorLegend.prototype.orientation = function (orientation) {
            if (orientation == null) {
                return this._orientation;
            }
            else {
                this._orientation = InterpolatedColorLegend._ensureOrientation(orientation);
                this.redraw();
                return this;
            }
        };
        InterpolatedColorLegend.prototype.fixedWidth = function () {
            return !this.expands() || this._isVertical();
        };
        InterpolatedColorLegend.prototype.fixedHeight = function () {
            return !this.expands() || !this._isVertical();
        };
        InterpolatedColorLegend.prototype._generateTicks = function (numSwatches) {
            if (numSwatches === void 0) { numSwatches = InterpolatedColorLegend._DEFAULT_NUM_SWATCHES; }
            var domain = this._scale.domain();
            if (numSwatches === 1) {
                return [domain[0]];
            }
            var slope = (domain[1] - domain[0]) / (numSwatches - 1);
            var ticks = [];
            for (var i = 0; i < numSwatches; i++) {
                ticks.push(domain[0] + slope * i);
            }
            return ticks;
        };
        InterpolatedColorLegend.prototype._setup = function () {
            _super.prototype._setup.call(this);
            this._swatchContainer = this.content().append("g").classed("swatch-container", true);
            this._swatchBoundingBox = this.content().append("rect").classed("swatch-bounding-box", true);
            this._lowerLabel = this.content().append("g").classed(InterpolatedColorLegend.LEGEND_LABEL_CLASS, true);
            this._upperLabel = this.content().append("g").classed(InterpolatedColorLegend.LEGEND_LABEL_CLASS, true);
            var context = new src$1.SvgContext(this.content().node());
            this._measurer = new src$1.Measurer(context);
            this._wrapper = new src$1.Wrapper();
            this._writer = new src$1.Writer(this._measurer, context, this._wrapper);
        };
        InterpolatedColorLegend.prototype.requestedSpace = function (offeredWidth, offeredHeight) {
            var _this = this;
            var textHeight = this._measurer.measure().height;
            var padding = textHeight;
            var domain = this._scale.domain();
            var labelWidths = domain.map(function (d) { return _this._measurer.measure(_this._formatter(d)).width; });
            var desiredHeight;
            var desiredWidth;
            var numSwatches = InterpolatedColorLegend._DEFAULT_NUM_SWATCHES;
            if (this._isVertical()) {
                var longestWidth = utils$1.Math.max(labelWidths, 0);
                desiredWidth = padding + textHeight + this._textPadding + longestWidth + this._textPadding;
                desiredHeight = numSwatches * textHeight;
            }
            else {
                desiredHeight = padding + textHeight + padding;
                desiredWidth = this._textPadding + labelWidths[0] + numSwatches * textHeight
                    + labelWidths[1] + this._textPadding;
            }
            return {
                minWidth: desiredWidth,
                minHeight: desiredHeight,
            };
        };
        InterpolatedColorLegend.prototype._isVertical = function () {
            return this._orientation !== "horizontal";
        };
        InterpolatedColorLegend.prototype.renderImmediately = function () {
            var _this = this;
            _super.prototype.renderImmediately.call(this);
            var domain = this._scale.domain();
            var text0 = this._formatter(domain[0]);
            var text0Width = this._measurer.measure(text0).width;
            var text1 = this._formatter(domain[1]);
            var text1Width = this._measurer.measure(text1).width;
            var textHeight = this._measurer.measure().height;
            var textPadding = this._textPadding;
            var upperLabelShift = { x: 0, y: 0 };
            var lowerLabelShift = { x: 0, y: 0 };
            var lowerWriteOptions = {
                xAlign: "center",
                yAlign: "center",
                textRotation: 0,
            };
            var upperWriteOptions = {
                xAlign: "center",
                yAlign: "center",
                textRotation: 0,
            };
            var swatchWidth;
            var swatchHeight;
            var swatchX;
            var swatchY;
            var boundingBoxAttr = {
                x: 0,
                y: 0,
                width: 0,
                height: 0,
            };
            var padding;
            var numSwatches;
            if (this._isVertical()) {
                numSwatches = Math.floor(this.height());
                var longestTextWidth_1 = Math.max(text0Width, text1Width);
                padding = (this.width() - longestTextWidth_1 - 2 * this._textPadding) / 2;
                swatchWidth = Math.max(this.width() - padding - 2 * textPadding - longestTextWidth_1, 0);
                swatchHeight = 1;
                swatchY = function (d, i) { return _this.height() - (i + 1); };
                upperWriteOptions.yAlign = "top";
                upperLabelShift.y = 0;
                lowerWriteOptions.yAlign = "bottom";
                lowerLabelShift.y = 0;
                if (this._orientation === "left") {
                    swatchX = function (d, i) { return textPadding + longestTextWidth_1 + textPadding; };
                    upperWriteOptions.xAlign = "right";
                    upperLabelShift.x = -(padding + swatchWidth + textPadding);
                    lowerWriteOptions.xAlign = "right";
                    lowerLabelShift.x = -(padding + swatchWidth + textPadding);
                }
                else {
                    swatchX = function (d, i) { return padding; };
                    upperWriteOptions.xAlign = "left";
                    upperLabelShift.x = padding + swatchWidth + textPadding;
                    lowerWriteOptions.xAlign = "left";
                    lowerLabelShift.x = padding + swatchWidth + textPadding;
                }
                boundingBoxAttr["width"] = swatchWidth;
                boundingBoxAttr["height"] = numSwatches * swatchHeight;
            }
            else {
                padding = Math.max(textPadding, (this.height() - textHeight) / 2);
                numSwatches = Math.max(Math.floor(this.width() - textPadding * 4 - text0Width - text1Width), 0);
                swatchWidth = 1;
                swatchHeight = Math.max((this.height() - 2 * padding), 0);
                swatchX = function (d, i) { return Math.floor(text0Width + 2 * textPadding) + i; };
                swatchY = function (d, i) { return padding; };
                upperWriteOptions.xAlign = "right";
                upperLabelShift.x = -textPadding;
                lowerWriteOptions.xAlign = "left";
                lowerLabelShift.x = textPadding;
                boundingBoxAttr["y"] = padding;
                boundingBoxAttr["width"] = numSwatches * swatchWidth;
                boundingBoxAttr["height"] = swatchHeight;
            }
            boundingBoxAttr["x"] = swatchX(null, 0); // position of the first swatch
            this._upperLabel.text(""); // clear the upper label
            this._writer.write(text1, this.width(), this.height(), upperWriteOptions, this._upperLabel.node());
            var upperTranslateString = "translate(" + upperLabelShift.x + ", " + upperLabelShift.y + ")";
            this._upperLabel.attr("transform", upperTranslateString);
            this._lowerLabel.text(""); // clear the lower label
            this._writer.write(text0, this.width(), this.height(), lowerWriteOptions, this._lowerLabel.node());
            var lowerTranslateString = "translate(" + lowerLabelShift.x + ", " + lowerLabelShift.y + ")";
            this._lowerLabel.attr("transform", lowerTranslateString);
            this._swatchBoundingBox.attrs(boundingBoxAttr);
            var ticks = this._generateTicks(numSwatches);
            var swatchesUpdate = this._swatchContainer.selectAll("rect.swatch").data(ticks);
            var rects = swatchesUpdate.enter().append("rect").classed("swatch", true);
            var swatches = swatchesUpdate.merge(rects);
            swatchesUpdate.exit().remove();
            swatches.attrs({
                fill: function (d, i) { return _this._scale.scale(d); },
                width: swatchWidth,
                height: swatchHeight,
                x: swatchX,
                y: swatchY,
                "shape-rendering": "crispEdges",
            });
            if (config$1.ADD_TITLE_ELEMENTS) {
                rects.append("title").text(function (d) { return _this._formatter(d); });
            }
            return this;
        };
        InterpolatedColorLegend._DEFAULT_NUM_SWATCHES = 11;
        /**
         * The css class applied to the legend labels.
         */
        InterpolatedColorLegend.LEGEND_LABEL_CLASS = "legend-label";
        return InterpolatedColorLegend;
    }(component.Component));
    exports.InterpolatedColorLegend = InterpolatedColorLegend;
    });

    var label = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });



    var Label = /** @class */ (function (_super) {
        tslib_es6.__extends(Label, _super);
        /**
         * A Label is a Component that displays a single line of text.
         *
         * @constructor
         * @param {string} [displayText=""] The text of the Label.
         * @param {number} [angle=0] The angle of the Label in degrees (-90/0/90). 0 is horizontal.
         */
        function Label(displayText, angle) {
            if (displayText === void 0) { displayText = ""; }
            if (angle === void 0) { angle = 0; }
            var _this = _super.call(this) || this;
            _this.addClass("label");
            _this.text(displayText);
            _this.angle(angle);
            _this.xAlignment("center").yAlignment("center");
            _this._padding = 0;
            return _this;
        }
        Label.prototype.requestedSpace = function (offeredWidth, offeredHeight) {
            var desiredWH = this._measurer.measure(this._text);
            var desiredWidth = (this.angle() === 0 ? desiredWH.width : desiredWH.height) + 2 * this.padding();
            var desiredHeight = (this.angle() === 0 ? desiredWH.height : desiredWH.width) + 2 * this.padding();
            return {
                minWidth: desiredWidth,
                minHeight: desiredHeight,
            };
        };
        Label.prototype._setup = function () {
            _super.prototype._setup.call(this);
            this._textContainer = this.content().append("g");
            var context = new src$1.SvgContext(this._textContainer.node());
            this._measurer = new src$1.CacheMeasurer(context);
            this._wrapper = new src$1.Wrapper();
            this._writer = new src$1.Writer(this._measurer, context, this._wrapper);
            this.text(this._text);
        };
        Label.prototype.text = function (displayText) {
            if (displayText == null) {
                return this._text;
            }
            else {
                if (typeof displayText !== "string") {
                    throw new Error("Label.text() only takes strings as input");
                }
                this._text = displayText;
                this.redraw();
                return this;
            }
        };
        Label.prototype.angle = function (angle) {
            if (angle == null) {
                return this._angle;
            }
            else {
                angle %= 360;
                if (angle > 180) {
                    angle -= 360;
                }
                else if (angle < -180) {
                    angle += 360;
                }
                if (angle === -90 || angle === 0 || angle === 90) {
                    this._angle = angle;
                }
                else {
                    throw new Error(angle + " is not a valid angle for Label");
                }
                this.redraw();
                return this;
            }
        };
        Label.prototype.padding = function (padAmount) {
            if (padAmount == null) {
                return this._padding;
            }
            else {
                padAmount = +padAmount;
                if (padAmount < 0) {
                    throw new Error(padAmount + " is not a valid padding value. Cannot be less than 0.");
                }
                this._padding = padAmount;
                this.redraw();
                return this;
            }
        };
        Label.prototype.fixedWidth = function () {
            return true;
        };
        Label.prototype.fixedHeight = function () {
            return true;
        };
        Label.prototype.renderImmediately = function () {
            _super.prototype.renderImmediately.call(this);
            // HACKHACK Typesettable.remove existing content - #21 on Typesettable.
            this._textContainer.selectAll("g").remove();
            var textMeasurement = this._measurer.measure(this._text);
            var heightPadding = Math.max(Math.min((this.height() - textMeasurement.height) / 2, this.padding()), 0);
            var widthPadding = Math.max(Math.min((this.width() - textMeasurement.width) / 2, this.padding()), 0);
            this._textContainer.attr("transform", "translate(" + widthPadding + "," + heightPadding + ")");
            var writeWidth = this.width() - 2 * widthPadding;
            var writeHeight = this.height() - 2 * heightPadding;
            var writeOptions = {
                xAlign: this.xAlignment(),
                yAlign: this.yAlignment(),
                textRotation: this.angle(),
            };
            this._writer.write(this._text, writeWidth, writeHeight, writeOptions);
            return this;
        };
        Label.prototype.invalidateCache = function () {
            _super.prototype.invalidateCache.call(this);
            this._measurer.reset();
        };
        return Label;
    }(component.Component));
    exports.Label = Label;
    var TitleLabel = /** @class */ (function (_super) {
        tslib_es6.__extends(TitleLabel, _super);
        /**
         * @constructor
         * @param {string} [text]
         * @param {number} [angle] One of -90/0/90. 0 is horizontal.
         */
        function TitleLabel(text, angle) {
            var _this = _super.call(this, text, angle) || this;
            _this.addClass(TitleLabel.TITLE_LABEL_CLASS);
            return _this;
        }
        TitleLabel.TITLE_LABEL_CLASS = "title-label";
        return TitleLabel;
    }(Label));
    exports.TitleLabel = TitleLabel;
    var AxisLabel = /** @class */ (function (_super) {
        tslib_es6.__extends(AxisLabel, _super);
        /**
         * @constructor
         * @param {string} [text]
         * @param {number} [angle] One of -90/0/90. 0 is horizontal.
         */
        function AxisLabel(text, angle) {
            var _this = _super.call(this, text, angle) || this;
            _this.addClass(AxisLabel.AXIS_LABEL_CLASS);
            return _this;
        }
        AxisLabel.AXIS_LABEL_CLASS = "axis-label";
        return AxisLabel;
    }(Label));
    exports.AxisLabel = AxisLabel;
    });

    var symbolFactories = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });

    function circle() {
        return function (symbolSize) {
            return d3.symbol().type(d3.symbolCircle).size(Math.PI * Math.pow(symbolSize / 2, 2));
        };
    }
    exports.circle = circle;
    function square() {
        return function (symbolSize) {
            return d3.symbol().type(d3.symbolSquare).size(Math.pow(symbolSize, 2));
        };
    }
    exports.square = square;
    function cross() {
        return function (symbolSize) {
            return d3.symbol().type(d3.symbolCross).size((5 / 9) * Math.pow(symbolSize, 2));
        };
    }
    exports.cross = cross;
    function diamond() {
        return function (symbolSize) {
            return d3.symbol().type(d3.symbolDiamond).size(Math.tan(Math.PI / 6) * Math.pow(symbolSize, 2) / 2);
        };
    }
    exports.diamond = diamond;
    function triangle() {
        return function (symbolSize) {
            return d3.symbol().type(d3.symbolTriangle).size(Math.sqrt(3) * Math.pow(symbolSize / 2, 2));
        };
    }
    exports.triangle = triangle;
    // copied from https://github.com/d3/d3-shape/blob/e2e57722004acba754ed9edff020282682450c5c/src/symbol/star.js#L3
    var ka = 0.89081309152928522810;
    function star() {
        return function (symbolSize) {
            return d3.symbol().type(d3.symbolStar).size(ka * Math.pow(symbolSize / 2, 2));
        };
    }
    exports.star = star;
    // copied from https://github.com/d3/d3-shape/blob/c35b2303eb4836aba3171642f01c2653e4228b9c/src/symbol/wye.js#L2
    var a = ((1 / Math.sqrt(12)) / 2 + 1) * 3;
    function wye() {
        return function (symbolSize) {
            return d3.symbol().type(d3.symbolWye).size(a * Math.pow(symbolSize / 2.4, 2));
        };
    }
    exports.wye = wye;
    });

    var legend = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });








    /**
     * The Legend's row representations. Stores positioning information
     * and column data.
     */
    var LegendRow = /** @class */ (function () {
        function LegendRow(
        /**
         * Columns within the row
         * @param {LegendColumn<any>[]} columns
         */
        columns, 
        /**
         * Padding applied below the row. Affects the spacing between rows. Defaults to 0.
         * @param {bottomPadding} number
         */
        bottomPadding, 
        /**
         * Sets the maximum allowable width of this column.
         * @param {number} maxWidth
         */
        maxWidth) {
            if (columns === void 0) { columns = []; }
            if (bottomPadding === void 0) { bottomPadding = 0; }
            if (maxWidth === void 0) { maxWidth = Infinity; }
            this.columns = columns;
            this.bottomPadding = bottomPadding;
            this.maxWidth = maxWidth;
        }
        /**
         * Adds a column to the list of columns within the row. May readjust the size of the
         * column to fit within the row
         *
         * @param {LegendColumn<any>} column
         */
        LegendRow.prototype.addColumn = function (column) {
            var desiredColumnWidth = column.width;
            // choose the smaller of 1) remaining space, 2) desired width
            var widthRemaining = this.getWidthAvailable();
            column.width = Math.min(widthRemaining, desiredColumnWidth);
            this.columns.push(column);
        };
        /**
         * Returns the bounds the column, relative to the row.
         * @param {number} columnIndex The index of the column in question
         * @returns {Bounds} bounds
         */
        LegendRow.prototype.getBounds = function (columnIndex) {
            var column = this.columns[columnIndex];
            var columnXOffset = 0;
            for (var i = 0; i < columnIndex; i++) {
                columnXOffset += this.columns[i].width;
            }
            return {
                topLeft: { x: columnXOffset, y: 0 },
                bottomRight: {
                    x: columnXOffset + column.width,
                    y: column.height,
                },
            };
        };
        /**
         * Returns the height of the row, including the bottomPadding.
         * @return {number} height
         */
        LegendRow.prototype.getHeight = function () {
            return utils$1.Math.max(this.columns.map(function (_a) {
                var height = _a.height;
                return height;
            }), 0) + this.bottomPadding;
        };
        /**
         * Returns the current width of the row constrained by maxWidth, if set.
         * @returns {number} width
         */
        LegendRow.prototype.getWidth = function () {
            return Math.min(this.columns.reduce(function (sum, _a) {
                var width = _a.width;
                return sum + width;
            }, 0), this.maxWidth);
        };
        /**
         * Returns the remaining width available in the row based on the maximum
         * width of this row.
         * @returns {number} widthRemaining
         */
        LegendRow.prototype.getWidthAvailable = function () {
            var widthConsumed = this.getWidth();
            return Math.max(this.maxWidth - widthConsumed, 0);
        };
        return LegendRow;
    }());
    /**
     * Stores LegendRows. Useful for calculating and maintaining
     * positioning information about the Legend.
     */
    var LegendTable = /** @class */ (function () {
        function LegendTable(maxWidth, maxHeight, padding, rows) {
            if (maxWidth === void 0) { maxWidth = Infinity; }
            if (maxHeight === void 0) { maxHeight = Infinity; }
            if (padding === void 0) { padding = 0; }
            if (rows === void 0) { rows = []; }
            this.maxWidth = maxWidth;
            this.maxHeight = maxHeight;
            this.padding = padding;
            this.rows = rows;
        }
        LegendTable.prototype.addRow = function (row) {
            row.maxWidth = this.maxWidth - this.padding * 2;
            this.rows.push(row);
        };
        /**
         * Returns the bounds of the column relative to the parent and siblings of the
         * column.
         *
         * @param {number} rowIndex The parent row containing the desired column.
         * @param {number} columnIndex The column to calculate bounds.
         * @returns {Bounds}
         */
        LegendTable.prototype.getColumnBounds = function (rowIndex, columnIndex) {
            var rowBounds = this.getRowBounds(rowIndex);
            var columnBounds = this.rows[rowIndex].getBounds(columnIndex);
            columnBounds.topLeft.x += rowBounds.topLeft.x;
            columnBounds.bottomRight.x += rowBounds.topLeft.x;
            columnBounds.topLeft.y += rowBounds.topLeft.y;
            columnBounds.bottomRight.y += rowBounds.topLeft.y;
            return columnBounds;
        };
        /**
         * Returns the bounds relative to the parent and siblings of the row.
         *
         * @param {number} rowIndex The row to calculate bounds
         * @returns {Bounds}
         */
        LegendTable.prototype.getRowBounds = function (rowIndex) {
            var rowXOffset = this.padding;
            var rowYOffset = this.padding;
            for (var i = 0; i < rowIndex; i++) {
                rowYOffset += this.rows[i].getHeight();
            }
            var rowBounds = {
                topLeft: { x: rowXOffset, y: rowYOffset },
                bottomRight: {
                    x: rowXOffset + this.rows[rowIndex].getWidth(),
                    y: rowYOffset + this.rows[rowIndex].getHeight(),
                },
            };
            return rowBounds;
        };
        /**
         * Returns the height of the Table, constrained by a maximum height, if set.
         * The height includes the padding, if set.
         * @returns {number} height
         */
        LegendTable.prototype.getHeight = function () {
            return Math.min(this.rows.reduce(function (sum, row) { return sum + row.getHeight(); }, 0) + this.padding * 2, this.maxHeight);
        };
        /**
         * Returns the width of the table, constrained by the maximum width, if set.
         * The width includes the padding, if set.
         * @returns {number} width
         */
        LegendTable.prototype.getWidth = function () {
            return Math.min(utils$1.Math.max(this.rows.map(function (row) { return row.getWidth(); }), 0) + this.padding * 2, this.maxWidth);
        };
        return LegendTable;
    }());
    var Legend = /** @class */ (function (_super) {
        tslib_es6.__extends(Legend, _super);
        /**
         * The Legend consists of a series of entries, each with a color and label taken from the Color Scale.
         *
         * @constructor
         * @param {Scale.Color} scale
         */
        function Legend(colorScale) {
            var _this = _super.call(this) || this;
            _this._padding = 5;
            _this._rowBottomPadding = 3;
            _this.addClass("legend");
            _this.maxEntriesPerRow(1);
            if (colorScale == null) {
                throw new Error("Legend requires a colorScale");
            }
            _this._colorScale = colorScale;
            _this._redrawCallback = function (scale) { return _this.redraw(); };
            _this._colorScale.onUpdate(_this._redrawCallback);
            _this._formatter = formatters.identity();
            _this.maxLinesPerEntry(1);
            _this.xAlignment("right").yAlignment("top");
            _this.comparator(function (a, b) {
                var formattedText = _this._colorScale.domain().slice().map(function (d) { return _this._formatter(d); });
                return formattedText.indexOf(a) - formattedText.indexOf(b);
            });
            _this._symbolFactoryAccessor = function () { return symbolFactories.circle(); };
            _this._symbolOpacityAccessor = function () { return 1; };
            return _this;
        }
        Legend.prototype._setup = function () {
            _super.prototype._setup.call(this);
            var fakeLegendRow = this.content().append("g").classed(Legend.LEGEND_ROW_CLASS, true);
            var fakeLegendEntry = fakeLegendRow.append("g").classed(Legend.LEGEND_ENTRY_CLASS, true);
            fakeLegendEntry.append("text");
            var context = new src$1.SvgContext(fakeLegendRow.node(), null, config$1.ADD_TITLE_ELEMENTS);
            this._measurer = new src$1.CacheMeasurer(context);
            this._wrapper = new src$1.Wrapper().maxLines(this.maxLinesPerEntry());
            this._writer = new src$1.Writer(this._measurer, context, this._wrapper);
        };
        Legend.prototype.formatter = function (formatter) {
            if (formatter == null) {
                return this._formatter;
            }
            this._formatter = formatter;
            this.redraw();
            return this;
        };
        Legend.prototype.maxEntriesPerRow = function (maxEntriesPerRow) {
            if (maxEntriesPerRow == null) {
                return this._maxEntriesPerRow;
            }
            else {
                this._maxEntriesPerRow = maxEntriesPerRow;
                this.redraw();
                return this;
            }
        };
        Legend.prototype.maxLinesPerEntry = function (maxLinesPerEntry) {
            if (maxLinesPerEntry == null) {
                return this._maxLinesPerEntry;
            }
            else {
                this._maxLinesPerEntry = maxLinesPerEntry;
                this.redraw();
                return this;
            }
        };
        Legend.prototype.maxWidth = function (maxWidth) {
            if (maxWidth == null) {
                return this._maxWidth;
            }
            else {
                this._maxWidth = maxWidth;
                this.redraw();
                return this;
            }
        };
        Legend.prototype.comparator = function (comparator) {
            if (comparator == null) {
                return this._comparator;
            }
            else {
                this._comparator = comparator;
                this.redraw();
                return this;
            }
        };
        Legend.prototype.colorScale = function (colorScale) {
            if (colorScale != null) {
                this._colorScale.offUpdate(this._redrawCallback);
                this._colorScale = colorScale;
                this._colorScale.onUpdate(this._redrawCallback);
                this.redraw();
                return this;
            }
            else {
                return this._colorScale;
            }
        };
        Legend.prototype.destroy = function () {
            _super.prototype.destroy.call(this);
            this._colorScale.offUpdate(this._redrawCallback);
        };
        Legend.prototype._buildLegendTable = function (width, height) {
            var _this = this;
            var textHeight = this._measurer.measure().height;
            var table = new LegendTable(width, height, this._padding);
            var entryNames = this._colorScale.domain().slice().sort(function (a, b) { return _this._comparator(_this._formatter(a), _this._formatter(b)); });
            var row = new LegendRow();
            table.addRow(row);
            row.bottomPadding = this._rowBottomPadding;
            entryNames.forEach(function (name, index) {
                if (row.columns.length / 2 === _this.maxEntriesPerRow()) {
                    // we add two columns per entry, a symbol column and a name column
                    // if the current row is full, according to the number of entries
                    // we're allowed to have per row, we need to allocate new space
                    row = new LegendRow();
                    row.bottomPadding = _this._rowBottomPadding;
                    table.addRow(row);
                }
                var availableWidth = row.getWidthAvailable();
                var formattedName = _this._formatter(name);
                // this is the width of the series name without any line wrapping
                // it is the most optimal presentation of the name
                var unwrappedNameWidth = _this._measurer.measure(formattedName).width;
                var willBeSquished = (availableWidth - textHeight - unwrappedNameWidth) < 0;
                if (willBeSquished && row.columns.length > 1) {
                    // adding the entry to this row will squish this
                    // entry. The row already contains entries so create
                    // a new row to add this entry to for optimal display
                    row = new LegendRow();
                    row.bottomPadding = _this._rowBottomPadding;
                    table.addRow(row);
                }
                var symbolColumn = { width: textHeight, height: textHeight, data: { name: name, type: "symbol" } };
                row.addColumn(symbolColumn);
                // the space consumed by the name field is the minimum of the space available in the table
                // and the actual width consumed by the name
                availableWidth = row.getWidthAvailable();
                var usedNameWidth = Math.min(availableWidth, unwrappedNameWidth);
                _this._wrapper.maxLines(_this.maxLinesPerEntry());
                var numberOfRows = _this._wrapper.wrap(formattedName, _this._measurer, usedNameWidth).noLines;
                var nameColumnHeight = numberOfRows * textHeight;
                var nameColumn = { width: usedNameWidth, height: nameColumnHeight, data: { name: name, type: "text" } };
                row.addColumn(nameColumn);
            });
            return table;
        };
        Legend.prototype.requestedSpace = function (offeredWidth, offeredHeight) {
            // if max width is set, the table is guaranteed to be at most maxWidth wide.
            // if max width is not set, the table will be as wide as the longest untruncated row
            var table = this._buildLegendTable(utils$1.Math.min([this.maxWidth(), offeredWidth], offeredWidth), offeredHeight);
            return {
                minHeight: table.getHeight(),
                minWidth: table.getWidth(),
            };
        };
        /**
         * Gets the Entities (representing Legend entries) at a particular point.
         * Returns an empty array if no Entities are present at that location.
         *
         * @param {Point} p
         * @returns {Entity<Legend>[]}
         */
        Legend.prototype.entitiesAt = function (p) {
            var _this = this;
            if (!this._isSetup) {
                return [];
            }
            var table = this._buildLegendTable(this.width(), this.height());
            return table.rows.reduce(function (entity, row, rowIndex) {
                if (entity.length !== 0) {
                    // we've already found the nearest entity; just return it.
                    return entity;
                }
                var rowBounds = table.getRowBounds(rowIndex);
                var withinRow = utils$1.Math.within(p, rowBounds);
                if (!withinRow) {
                    // the nearest entity isn't within this row, continue;
                    return entity;
                }
                return row.columns.reduce(function (entity, column, columnIndex) {
                    var columnBounds = table.getColumnBounds(rowIndex, columnIndex);
                    var withinColumn = utils$1.Math.within(p, columnBounds);
                    if (withinColumn) {
                        var rowElement = _this.content().selectAll("." + Legend.LEGEND_ROW_CLASS).nodes()[rowIndex];
                        // HACKHACK The 2.x API chooses the symbol element as the "selection" to return, regardless of what
                        // was actually selected
                        var entryElement = d3.select(rowElement)
                            .selectAll("." + Legend.LEGEND_ENTRY_CLASS).nodes()[Math.floor(columnIndex / 2)];
                        var symbolElement = d3.select(entryElement).select("." + Legend.LEGEND_SYMBOL_CLASS);
                        // HACKHACK The 2.x API returns the center {x, y} of the symbol as the position.
                        var rowTranslate = utils$1.DOM.getTranslateValues(d3.select(rowElement));
                        var symbolTranslate = utils$1.DOM.getTranslateValues(symbolElement);
                        return [{
                                bounds: utils$1.DOM.elementBBox(d3.select(rowElement)),
                                datum: column.data.name,
                                position: {
                                    x: rowTranslate[0] + symbolTranslate[0],
                                    y: rowTranslate[1] + symbolTranslate[1],
                                },
                                selection: d3.select(entryElement),
                                component: _this,
                            }];
                    }
                    return entity;
                }, entity);
            }, []);
        };
        Legend.prototype.renderImmediately = function () {
            _super.prototype.renderImmediately.call(this);
            var table = this._buildLegendTable(this.width(), this.height());
            // clear content from previous renders
            this.content().selectAll("*").remove();
            var rowsUpdate = this.content().selectAll("g." + Legend.LEGEND_ROW_CLASS).data(table.rows);
            var rows = rowsUpdate
                .enter()
                .append("g")
                .classed(Legend.LEGEND_ROW_CLASS, true)
                .merge(rowsUpdate);
            rowsUpdate.exit().remove();
            rows.attr("transform", function (row, rowIndex) {
                var rowBounds = table.getRowBounds(rowIndex);
                return "translate(" + rowBounds.topLeft.x + ", " + rowBounds.topLeft.y + ")";
            });
            var self = this;
            rows.each(function (row, rowIndex) {
                var symbolEntryPairs = [];
                for (var i = 0; i < row.columns.length; i += 2) {
                    symbolEntryPairs.push([row.columns[i], row.columns[i + 1]]);
                }
                var entriesUpdate = d3.select(this).selectAll("g." + Legend.LEGEND_ENTRY_CLASS).data(symbolEntryPairs);
                var entriesEnter = entriesUpdate
                    .enter()
                    .append("g")
                    .classed(Legend.LEGEND_ENTRY_CLASS, true)
                    .merge(entriesUpdate);
                entriesEnter.append("path")
                    .attr("d", function (symbolEntryPair, columnIndex) {
                    var symbol = symbolEntryPair[0];
                    return self.symbol()(symbol.data.name, rowIndex)(symbol.height * 0.6)(null);
                })
                    .attr("transform", function (symbolEntryPair, i) {
                    var symbol = symbolEntryPair[0];
                    var columnIndex = table.rows[rowIndex].columns.indexOf(symbol);
                    var columnBounds = table.getColumnBounds(rowIndex, columnIndex);
                    return "translate(" + (columnBounds.topLeft.x + symbol.width / 2) + ", " + symbol.height / 2 + ")";
                })
                    .attr("fill", function (symbolEntryPair) { return self._colorScale.scale(symbolEntryPair[0].data.name); })
                    .attr("opacity", function (symbolEntryPair, _columnIndex) {
                    return self.symbolOpacity()(symbolEntryPair[0].data.name, rowIndex);
                })
                    .classed(Legend.LEGEND_SYMBOL_CLASS, true);
                entriesEnter.append("g").classed("text-container", true)
                    .attr("transform", function (symbolEntryPair, i) {
                    var entry = symbolEntryPair[1];
                    var columnIndex = table.rows[rowIndex].columns.indexOf(entry);
                    var columnBounds = table.getColumnBounds(rowIndex, columnIndex);
                    return "translate(" + columnBounds.topLeft.x + ", 0)";
                })
                    .each(function (symbolEntryPair, i, rowIndex) {
                    var textContainer = d3.select(this);
                    var column = symbolEntryPair[1];
                    var writeOptions = {
                        xAlign: "left",
                        yAlign: "top",
                        textRotation: 0,
                    };
                    self._writer.write(self._formatter(column.data.name), column.width, self.height(), writeOptions, textContainer.node());
                });
                entriesUpdate.exit().remove();
            });
            return this;
        };
        Legend.prototype.symbol = function (symbol) {
            if (symbol == null) {
                return this._symbolFactoryAccessor;
            }
            else {
                this._symbolFactoryAccessor = symbol;
                this.render();
                return this;
            }
        };
        Legend.prototype.symbolOpacity = function (symbolOpacity) {
            if (symbolOpacity == null) {
                return this._symbolOpacityAccessor;
            }
            else if (typeof symbolOpacity === "number") {
                this._symbolOpacityAccessor = function () { return symbolOpacity; };
            }
            else {
                this._symbolOpacityAccessor = symbolOpacity;
            }
            this.render();
            return this;
        };
        Legend.prototype.fixedWidth = function () {
            return true;
        };
        Legend.prototype.fixedHeight = function () {
            return true;
        };
        Legend.prototype.invalidateCache = function () {
            _super.prototype.invalidateCache.call(this);
            this._measurer.reset();
        };
        /**
         * The css class applied to each legend row
         */
        Legend.LEGEND_ROW_CLASS = "legend-row";
        /**
         * The css class applied to each legend entry
         */
        Legend.LEGEND_ENTRY_CLASS = "legend-entry";
        /**
         * The css class applied to each legend symbol
         */
        Legend.LEGEND_SYMBOL_CLASS = "legend-symbol";
        return Legend;
    }(component.Component));
    exports.Legend = Legend;
    });

    var commons = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });
    var Animator;
    (function (Animator) {
        Animator.MAIN = "main";
        Animator.RESET = "reset";
    })(Animator = exports.Animator || (exports.Animator = {}));
    });

    /*!
     * isobject <https://github.com/jonschlinkert/isobject>
     *
     * Copyright (c) 2014-2017, Jon Schlinkert.
     * Released under the MIT License.
     */

    var isobject = function isObject(val) {
      return val != null && typeof val === 'object' && Array.isArray(val) === false;
    };

    function isObjectObject(o) {
      return isobject(o) === true
        && Object.prototype.toString.call(o) === '[object Object]';
    }

    var isPlainObject = function isPlainObject(o) {
      var ctor,prot;

      if (isObjectObject(o) === false) return false;

      // If has modified constructor
      ctor = o.constructor;
      if (typeof ctor !== 'function') return false;

      // If has modified prototype
      prot = ctor.prototype;
      if (isObjectObject(prot) === false) return false;

      // If constructor does not have an Object-specific method
      if (prot.hasOwnProperty('isPrototypeOf') === false) {
        return false;
      }

      // Most likely a plain Object
      return true;
    };

    var dataset = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });

    var UPDATE_MONOTONIC = 0;
    var Dataset = /** @class */ (function () {
        /**
         * A Dataset contains an array of data and some metadata.
         * Changes to the data or metadata will cause anything subscribed to the Dataset to update.
         *
         * @constructor
         * @param {any[]} [data=[]] The data for this Dataset.
         * @param {any} [metadata={}] An object containing additional information.
         */
        function Dataset(data, metadata) {
            if (data === void 0) { data = []; }
            if (metadata === void 0) { metadata = {}; }
            this._updateId = UPDATE_MONOTONIC++;
            this._data = data;
            this._metadata = metadata;
            this._callbacks = new utils$1.CallbackSet();
        }
        /**
         * Adds a callback to be called when the Dataset updates.
         *
         * @param {DatasetCallback} callback.
         * @returns {Dataset} The calling Dataset.
         */
        Dataset.prototype.onUpdate = function (callback) {
            this._callbacks.add(callback);
            return this;
        };
        /**
         * Removes a callback that would be called when the Dataset updates.
         *
         * @param {DatasetCallback} callback
         * @returns {Dataset} The calling Dataset.
         */
        Dataset.prototype.offUpdate = function (callback) {
            this._callbacks.delete(callback);
            return this;
        };
        Dataset.prototype.data = function (data) {
            if (data == null) {
                return this._data;
            }
            else {
                this._data = data;
                this._dispatchUpdate();
                return this;
            }
        };
        Dataset.prototype.metadata = function (metadata) {
            if (metadata == null) {
                return this._metadata;
            }
            else {
                this._metadata = metadata;
                this._dispatchUpdate();
                return this;
            }
        };
        Dataset.prototype.updateId = function () {
            return this._updateId;
        };
        Dataset.prototype._dispatchUpdate = function () {
            this._updateId = UPDATE_MONOTONIC++;
            this._callbacks.callCallbacks(this);
        };
        return Dataset;
    }());
    exports.Dataset = Dataset;
    });

    var signature = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     * @fileoverview Implements the Signature API to help in comparing when two
     * Plottable objects have "changed".
     *
     * Memoization in Plottable is complicated by mutable scales and datasets. We cannot simply
     * reference compare two e.g. scales since it may have internally mutated. To resolve this,
     * we write a recursive Signature interface that holds an immutable snapshot of whatever
     * state the scale/data was in at the time. Then on memoized function invocation we sign the
     * new inputs and compare the signatures to decide if we should recompute.
     *
     * We must hand-write a signature for each custom class we wish to support.
     */
    Object.defineProperty(exports, "__esModule", { value: true });




    /**
     * Generic signature factory - pass any value and get a signature for it.
     *
     * Datasets and Scales are handled specially - see their respective signing methods.
     *
     * If the input is already a signature, simply return it.
     *
     * @param a
     * @returns {Signature}
     */
    function sign(a) {
        if (a instanceof Signature) {
            return a;
        }
        else if (a instanceof Date) {
            return signRef(a.valueOf());
        }
        else if (a instanceof scale.Scale) {
            return signScale(a);
        }
        else if (a instanceof dataset.Dataset) {
            return signDataset(a);
        }
        else if (isPlainObject(a)) {
            return signObj(a);
        }
        else if (Array.isArray(a)) {
            return signArray(a);
        }
        else {
            return signRef(a);
        }
    }
    exports.sign = sign;
    function signScale(scale) {
        var scaleObj = {
            // required in case the domain has changed without the updateId changing
            domain: scale.domain(),
            // required in case the range has changed without the updateId changing
            range: scale.range(),
            // generic catch-all for scale updates (existing code doesn't fully catch all
            // updates, but it's good for something)
            updateId: scale.updateId(),
            // keep a ref to the scale in case the ref changes
            ref: signRef(scale),
        };
        return signObj(scaleObj);
    }
    exports.signScale = signScale;
    function signDataset(dataset) {
        var datasetObj = {
            ref: signRef(dataset),
            // only sign updateId since only data() and metadata() exist as properties
            // and both update updateId
            updateId: dataset.updateId(),
        };
        return signObj(datasetObj);
    }
    exports.signDataset = signDataset;
    function signRef(a) {
        return new ReferenceSignature(a);
    }
    exports.signRef = signRef;
    function signArray(a) {
        return new ArraySignature(a.map(function (element) { return sign(element); }));
    }
    exports.signArray = signArray;
    function signObj(obj) {
        var signatureRecord = {};
        for (var key in obj) {
            if (obj.hasOwnProperty(key)) {
                signatureRecord[key] = sign(obj[key]);
            }
        }
        return new ObjectSignature(signatureRecord);
    }
    exports.signObj = signObj;
    /**
     * Base signature. Subclasses should implement isSignatureDifferent. All classes
     * should be immutable.
     *
     * Users should only call `isDifferent`, not `isSignatureDifferent`.
     */
    var Signature = /** @class */ (function () {
        function Signature() {
        }
        Signature.prototype.isDifferent = function (other) {
            if (other instanceof this.constructor) {
                return this.isSignatureDifferent(other);
            }
            else {
                return true;
            }
        };
        return Signature;
    }());
    exports.Signature = Signature;
    /**
     * A signature for an array.
     */
    var ArraySignature = /** @class */ (function (_super) {
        tslib_es6.__extends(ArraySignature, _super);
        function ArraySignature(array) {
            var _this = _super.call(this) || this;
            _this.array = array;
            return _this;
        }
        /**
         * An array of signatures is different if any of the elements isDifferent.
         */
        ArraySignature.prototype.isSignatureDifferent = function (other) {
            if (other.array.length !== this.array.length) {
                return true;
            }
            else {
                for (var i = 0; i < this.array.length; i++) {
                    if (this.array[i].isDifferent(other.array[i])) {
                        return true;
                    }
                }
                return false;
            }
        };
        return ArraySignature;
    }(Signature));
    exports.ArraySignature = ArraySignature;
    var ReferenceSignature = /** @class */ (function (_super) {
        tslib_es6.__extends(ReferenceSignature, _super);
        function ReferenceSignature(ref) {
            var _this = _super.call(this) || this;
            _this.ref = ref;
            return _this;
        }
        ReferenceSignature.prototype.isSignatureDifferent = function (other) {
            return this.ref !== other.ref;
        };
        return ReferenceSignature;
    }(Signature));
    exports.ReferenceSignature = ReferenceSignature;
    /**
     * A signature for a plain js object.
     */
    var ObjectSignature = /** @class */ (function (_super) {
        tslib_es6.__extends(ObjectSignature, _super);
        function ObjectSignature(obj) {
            var _this = _super.call(this) || this;
            _this.obj = obj;
            return _this;
        }
        /**
         * An object signature is different if any of the elements isDifferent.
         */
        ObjectSignature.prototype.isSignatureDifferent = function (other) {
            var myKeys = Object.keys(this.obj);
            var otherKeys = Object.keys(other.obj);
            if (myKeys.length !== otherKeys.length) {
                return true;
            }
            for (var _i = 0, myKeys_1 = myKeys; _i < myKeys_1.length; _i++) {
                var key = myKeys_1[_i];
                if (!other.obj.hasOwnProperty(key)) {
                    return true;
                }
                if (this.obj[key].isDifferent(other.obj[key])) {
                    return true;
                }
            }
            return false;
        };
        return ObjectSignature;
    }(Signature));
    exports.ObjectSignature = ObjectSignature;
    });

    var memoize_1 = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     * @fileoverview Implements a function memoizer using the Signature API.
     */
    Object.defineProperty(exports, "__esModule", { value: true });

    /**
     * Return a memoized version of the input function. The memoized function
     * reduces unnecessary invocations of the input by keeping a cache of the
     * return value of compute:
     *
     * <pre>
     * function compute(a, b) { return a + b }
     * const memoizedCompute = memoize(compute);
     *
     * compute(3, 7) == 10
     * compute(3, 7) == 10 // cache hit
     * </pre>
     *
     * Cache invalidation is complicated by mutable classes (Scales and Datasets).
     * The Signature API is built to solve this issue by constructing an immutable
     * snapshot of Scales/Datasets on memoized function invocation, which is itself
     * a performance hit. Thus we introduce a "doLocked" method that momentarily
     * bypasses sign/comparison logic and simply returns the cached value.
     *
     * See the Signature API for more information.
     *
     * @param {F} compute
     * @returns {MemoizedFunction<F extends Function>}
     */
    function memoize(compute) {
        var lastSignature = undefined;
        var lastValue;
        var locked = false;
        var logPerformance = false;
        var memoizeFn = function () {
            var args = [];
            for (var _i = 0; _i < arguments.length; _i++) {
                args[_i] = arguments[_i];
            }
            if (locked) {
                return lastValue;
            }
            var inputSignature = signature.signArray(args);
            if (lastSignature === undefined
                || lastSignature.isDifferent(inputSignature)) {
                if (logPerformance) {
                    console.warn("cache miss! computing");
                }
                lastSignature = inputSignature;
                lastValue = compute.apply(this, args);
            }
            else {
                if (logPerformance) {
                    console.warn("cache hit!");
                }
            }
            return lastValue;
        };
        memoizeFn.doLocked = function (cb) {
            if (locked) {
                throw new Error("Locking an already locked memoize function!");
            }
            locked = true;
            var retVal = cb.apply(this);
            locked = false;
            return retVal;
        };
        memoizeFn.logPerformance = function (log) {
            if (log === void 0) { log = true; }
            logPerformance = log;
            return this;
        };
        return memoizeFn;
    }
    exports.memoize = memoize;
    });

    var memoizeProjectors_1 = createCommonjsModule(function (module, exports) {
    Object.defineProperty(exports, "__esModule", { value: true });

    /**
     * An index that stores values by numeric key.
     *
     * Internally this uses prototype-less objects as key lookups are marginally
     * faster than `{}`s.
     */
    var IndexMap = /** @class */ (function () {
        function IndexMap() {
            this.map = Object.create(null);
            this.exists = Object.create(null);
        }
        IndexMap.prototype.delete = function (key) {
            delete this.map[key];
            delete this.exists[key];
            return true;
        };
        IndexMap.prototype.get = function (key) {
            return this.map[key];
        };
        IndexMap.prototype.has = function (key) {
            return !!this.exists[key];
        };
        IndexMap.prototype.set = function (key, value) {
            this.map[key] = value;
            this.exists[key] = true;
            return this;
        };
        return IndexMap;
    }());
    /**
     * A lodash-style `MapCache` that utilizes a [number, number] key to create a
     * fast-lookup 2D index. This is much faster than stringifying the key.
     */
    var DatasetIndexCache = /** @class */ (function () {
        function DatasetIndexCache() {
            this.map = new IndexMap();
        }
        DatasetIndexCache.prototype.get = function (key) {
            return this.map.get(key[0]).get(key[1]);
        };
        DatasetIndexCache.prototype.has = function (key) {
            return this.map.has(key[0]) && this.map.get(key[0]).has(key[1]);
        };
        DatasetIndexCache.prototype.set = function (key, value) {
            this.map.has(key[0]) || this.map.set(key[0], new IndexMap());
            this.map.get(key[0]).set(key[1], value);
            return this;
        };
        DatasetIndexCache.prototype.delete = function (key) {
            // NOTE: this can potentially leave dangling `IndexMap`s if we delete
            // all the keys from the index instead of using `clear`. The overhead is
            // minimal, so this is fine.
            this.map.has(key[0]) && this.map.get(key[0]).delete(key[1]);
            return true;
        };
        DatasetIndexCache.prototype.clear = function () {
            this.map = new IndexMap();
        };
        DatasetIndexCache.resolver = function (d, i, dataset) { return [dataset.updateId(), i]; };
        return DatasetIndexCache;
    }());
    function memoizeProjector(projector) {
        var memo = lodash.memoize(projector, DatasetIndexCache.resolver);
        memo.cache = new DatasetIndexCache();
        return memo;
    }
    exports.memoizeProjector = memoizeProjector;
    function memoizeProjectors(attrToProjector) {
        Object.keys(attrToProjector).forEach(function (key) {
            attrToProjector[key] = memoizeProjector(attrToProjector[key]);
        });
        return attrToProjector;
    }
    exports.memoizeProjectors = memoizeProjectors;
    });

    var memThunk_1 = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     * @fileoverview Implements a convenient thunk function to handle the common case
     * of creating a memoized function that takes its inputs from mutable class properties.
     */
    Object.defineProperty(exports, "__esModule", { value: true });

    /**
     * First pass argument thunks that will be evaluated whenever the memThunk
     * is accessed. This should be fast and simple.
     *
     * Then pass a pure function that, when given the argument thunks' values,
     * will output some computed value. It should not use `this` in the body.
     *
     * We memoize and return this pure function.
     *
     * This way, memThunk lets you implement a performant, always-up-to-date "computed"
     * value getter.
     */
    function memThunk() {
        var argsAndCompute = [];
        for (var _i = 0; _i < arguments.length; _i++) {
            argsAndCompute[_i] = arguments[_i];
        }
        var inputs = argsAndCompute.slice(0, -1);
        var compute = argsAndCompute[argsAndCompute.length - 1];
        var memoizedCompute = memoize.memoize(compute);
        var memoizedThunk = function () {
            var _this = this;
            var inputEval = inputs.map(function (inputFn) { return inputFn.apply(_this); });
            return memoizedCompute.apply(undefined, inputEval);
        };
        return memoizedThunk;
    }
    exports.memThunk = memThunk;
    });

    var memoize = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });

    tslib_es6.__exportStar(memoize_1, exports);
    tslib_es6.__exportStar(memoizeProjectors_1, exports);
    tslib_es6.__exportStar(memThunk_1, exports);

    exports.sign = signature.sign;
    });

    var canvasDrawer = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });

    /**
     * A CanvasDrawer draws data onto a supplied Canvas Context.
     *
     * This class is immutable (but has internal state) and shouldn't be extended.
     */
    var CanvasDrawer = /** @class */ (function () {
        /**
         * @param _context The context for a canvas that this drawer will draw to.
         * @param _drawStep The draw step logic that actually draws.
         */
        function CanvasDrawer(_context, _drawStep) {
            this._context = _context;
            this._drawStep = _drawStep;
        }
        // public for testing
        CanvasDrawer.prototype.getDrawStep = function () {
            return this._drawStep;
        };
        CanvasDrawer.prototype.draw = function (data, appliedDrawSteps) {
            var projector = appliedDrawSteps[appliedDrawSteps.length - 1].attrToAppliedProjector;
            // don't support animations for now; just draw the last draw step immediately
            this._context.save();
            this._drawStep(this._context, data, projector);
            this._context.restore();
        };
        CanvasDrawer.prototype.getVisualPrimitives = function () {
            return [];
        };
        CanvasDrawer.prototype.getVisualPrimitiveAtIndex = function (index) {
            return null;
        };
        CanvasDrawer.prototype.remove = function () {
            // NO op - canvas element owns the canvas; context is free
        };
        return CanvasDrawer;
    }());
    exports.CanvasDrawer = CanvasDrawer;
    exports.ContextStyleAttrs = [
        "fill-opacity",
        "fill",
        "opacity",
        "stroke-opacity",
        "stroke-width",
        "stroke",
        "stroke-dasharray",
    ];
    /**
     * DEPRECATED. Precompute the concatted array and use `resolveAttributes`
     * instead of using this in the inner loop.
     */
    function resolveAttributesSubsetWithStyles(projector, extraKeys, datum, index) {
        var attrKeys = exports.ContextStyleAttrs.concat(extraKeys);
        return resolveAttributes(projector, attrKeys, datum, index);
    }
    exports.resolveAttributesSubsetWithStyles = resolveAttributesSubsetWithStyles;
    function resolveAttributes(projector, attrKeys, datum, index) {
        var attrs = {};
        for (var _i = 0, attrKeys_1 = attrKeys; _i < attrKeys_1.length; _i++) {
            var attrKey = attrKeys_1[_i];
            if (projector.hasOwnProperty(attrKey)) {
                attrs[attrKey] = projector[attrKey](datum, index);
            }
        }
        return attrs;
    }
    exports.resolveAttributes = resolveAttributes;
    function getStrokeOpacity(style) {
        var baseOpacity = style["opacity"] != null ? parseFloat(style["opacity"]) : 1;
        var strokeOpacity = style["stroke-opacity"] != null ? parseFloat(style["stroke-opacity"]) : 1;
        return strokeOpacity * baseOpacity;
    }
    function getFillOpacity(style) {
        var baseOpacity = style["opacity"] != null ? parseFloat(style["opacity"]) : 1;
        var fillOpacity = style["fill-opacity"] != null ? parseFloat(style["fill-opacity"]) : 1;
        return fillOpacity * baseOpacity;
    }
    function getStrokeWidth(style) {
        return style["stroke-width"] != null ? parseFloat(style["stroke-width"]) : 1;
    }
    exports.getStrokeWidth = getStrokeWidth;
    function getStrokeDashArray(style) {
        var rawValue = style["stroke-dasharray"];
        if (rawValue != null) {
            try {
                return rawValue.split(/[ ,]+/).map(function (x) { return parseInt(x, 10); });
            }
            catch (e) {
                console.error("getStrokeDashArray failed with: " + e);
                return [];
            }
        }
        return [];
    }
    exports.getStrokeDashArray = getStrokeDashArray;
    function renderArea(context, d3Area, data, style) {
        context.save();
        context.beginPath();
        d3Area.context(context);
        d3Area(data);
        context.lineJoin = "round";
        renderPathWithStyle(context, style);
        context.restore();
    }
    exports.renderArea = renderArea;
    function renderLine(context, d3Line, data, style) {
        context.save();
        context.beginPath();
        d3Line.context(context);
        d3Line(data);
        context.lineJoin = "round";
        renderPathWithStyle(context, style);
        context.restore();
    }
    exports.renderLine = renderLine;
    function renderPathWithStyle(context, style) {
        if (style["stroke"]) {
            context.lineWidth = getStrokeWidth(style);
            var strokeColor = d3.color(style["stroke"]);
            var strokeDashArray = getStrokeDashArray(style);
            context.setLineDash(strokeDashArray);
            strokeColor.opacity *= getStrokeOpacity(style);
            context.strokeStyle = strokeColor.toString();
            context.stroke();
        }
        if (style["fill"]) {
            var fillColor = d3.color(style["fill"]);
            fillColor.opacity *= getFillOpacity(style);
            context.fillStyle = fillColor.toString();
            context.fill();
        }
    }
    exports.renderPathWithStyle = renderPathWithStyle;
    });

    var drawer = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });
    /**
     * A Drawer is a stateful class that holds one SVGDrawer and one CanvasDrawer, and can switch between
     * the two.
     */
    var ProxyDrawer = /** @class */ (function () {
        /**
         * A Drawer draws svg elements based on the input Dataset.
         *
         * @constructor
         * @param _svgDrawerFactory A factory that will be invoked to create an SVGDrawer whenever useSVG is called
         * @param _canvasDrawStep The DrawStep to be fed into a new CanvasDrawer whenever useCanvas is called
         */
        function ProxyDrawer(_svgDrawerFactory, _canvasDrawerFactory) {
            this._svgDrawerFactory = _svgDrawerFactory;
            this._canvasDrawerFactory = _canvasDrawerFactory;
        }
        /**
         * Remove the old drawer and use SVG rendering from now on.
         */
        ProxyDrawer.prototype.useSVG = function (parent) {
            if (this._currentDrawer != null) {
                this._currentDrawer.remove();
            }
            var svgDrawer = this._svgDrawerFactory();
            svgDrawer.attachTo(parent);
            this._currentDrawer = svgDrawer;
        };
        /**
         * Remove the old drawer and use Canvas rendering from now on.
         */
        ProxyDrawer.prototype.useCanvas = function (canvas) {
            if (this._currentDrawer != null) {
                this._currentDrawer.remove();
            }
            this._currentDrawer = this._canvasDrawerFactory(canvas.node().getContext("2d"));
        };
        // public for testing
        ProxyDrawer.prototype.getDrawer = function () {
            return this._currentDrawer;
        };
        /**
         * Removes this Drawer's renderArea
         */
        ProxyDrawer.prototype.remove = function () {
            if (this._currentDrawer != null) {
                this._currentDrawer.remove();
            }
        };
        ProxyDrawer.prototype.draw = function (data, drawSteps) {
            this._currentDrawer.draw(data, drawSteps);
        };
        ProxyDrawer.prototype.getVisualPrimitives = function () {
            return this._currentDrawer.getVisualPrimitives();
        };
        ProxyDrawer.prototype.getVisualPrimitiveAtIndex = function (index) {
            return this._currentDrawer.getVisualPrimitiveAtIndex(index);
        };
        return ProxyDrawer;
    }());
    exports.ProxyDrawer = ProxyDrawer;
    });

    var svgDrawer = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });


    /**
     * An SVGDrawer draws data by creating DOM elements and setting specific attributes on them
     * to accurately reflect the data being passed in.
     *
     * This class is immutable (but has internal state).
     */
    var SVGDrawer = /** @class */ (function () {
        /**
         * @param svgElementName an HTML/SVG tag name to be created, one per datum.
         * @param className CSS classes to be applied to the drawn primitives.
         * @param applyDefaultAttributes
         */
        function SVGDrawer(svgElementName, className) {
            this._root = d3.select(document.createElementNS("http://www.w3.org/2000/svg", "g"));
            this._className = className;
            this._svgElementName = svgElementName;
        }
        SVGDrawer.prototype.draw = function (data, appliedDrawSteps) {
            var _this = this;
            /*
             * Draw to the DOM by clearing old DOM elements, adding new DOM elements,
             * and then passing those DOM elements to the animator, which will set the
             * appropriate attributes on the DOM.
             */
            this._createAndDestroyDOMElements(data);
            var delay = 0;
            var drawLength = appliedDrawSteps.length;
            var _loop_1 = function (i) {
                var drawStep = appliedDrawSteps[i];
                utils$1.Window.setTimeout(function () { return _this._drawStep(drawStep); }, delay);
                delay += drawStep.animator.totalTime(data.length);
            };
            for (var i = 0; i < drawLength; i++) {
                _loop_1(i);
            }
        };
        SVGDrawer.prototype.getVisualPrimitives = function () {
            if (this._cachedVisualPrimitivesNodes == null) {
                this._cachedVisualPrimitivesNodes = this._selection.nodes();
            }
            return this._cachedVisualPrimitivesNodes;
        };
        SVGDrawer.prototype.getVisualPrimitiveAtIndex = function (index) {
            if (this._cachedVisualPrimitivesNodeMap == null) {
                return null;
            }
            return this._cachedVisualPrimitivesNodeMap.get(index);
        };
        SVGDrawer.prototype.remove = function () {
            this._root.remove();
        };
        SVGDrawer.prototype.attachTo = function (parent) {
            parent.node().appendChild(this._root.node());
        };
        // public for testing
        SVGDrawer.prototype.getRoot = function () {
            return this._root;
        };
        /**
         * Returns the CSS selector for this Drawer's visual elements.
         */
        SVGDrawer.prototype.selector = function () {
            return this._svgElementName;
        };
        SVGDrawer.prototype._applyDefaultAttributes = function (selection) {
            // subclasses may override
        };
        SVGDrawer.prototype._createAndDestroyDOMElements = function (data) {
            // first store source data index before any filtering takes place
            var mappedData = data.map(function (d, i) { return d != null ? { d: d, i: i } : null; });
            // Whereas canvas renders can cope with null data values, svg renderer
            // attribute accessors assume non-null data values, so we must filter them
            // out. Unfortunately, this means the index passed to each accessor will
            // not necessarily match the index of the datum in the dataset.
            var filteredData = mappedData.filter(function (d) { return d != null; });
            var dataElementsUpdate = this._root.selectAll(this.selector()).data(filteredData);
            this._selection = dataElementsUpdate
                .enter()
                .append(this._svgElementName)
                .merge(dataElementsUpdate);
            dataElementsUpdate.exit().remove();
            // build map of source data index to Element
            var newMap = new utils$1.Map();
            this._selection.each(function (dataAndIndex) {
                // TODO optionally attach data attributes to Element
                // this.setAttribute("data-source-data", dataAndIndex.d);
                // this.setAttribute("data-source-index", dataAndIndex.i);
                newMap.set(dataAndIndex.i, this);
            });
            this._cachedVisualPrimitivesNodeMap = newMap;
            this._cachedVisualPrimitivesNodes = null;
            // unwrap data now that elements match their original data index
            this._selection.data(this._selection.data().map(function (_a) {
                var d = _a.d;
                return d;
            }));
            if (this._className != null) {
                this._selection.classed(this._className, true);
            }
            this._applyDefaultAttributes(this._selection);
        };
        /**
         * Draws data using one step
         *
         * @param{AppliedDrawStep} step The step, how data should be drawn.
         */
        SVGDrawer.prototype._drawStep = function (step) {
            var _this = this;
            var colorAttributes = ["fill", "stroke"];
            colorAttributes.forEach(function (colorAttribute) {
                if (step.attrToAppliedProjector[colorAttribute] != null) {
                    _this._selection.attr(colorAttribute, step.attrToAppliedProjector[colorAttribute]);
                }
            });
            step.animator.animate(this._selection, step.attrToAppliedProjector);
            if (this._className != null) {
                this._selection.classed(this._className, true);
            }
        };
        return SVGDrawer;
    }());
    exports.SVGDrawer = SVGDrawer;
    });

    var deferredRenderer = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2017-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });
    /**
     * Stores the deferred transformation state for a single scale.
     */
    var DomainTransform = /** @class */ (function () {
        function DomainTransform() {
            var _this = this;
            this.scale = 0;
            this.translate = 0;
            this.cachedDomain = [null, null];
            this.lastSeenDomain = [null, null];
            this.updateDomain = function (scale) {
                _this.lastSeenDomain = scale.getTransformationDomain();
                var cachedLength = scale.scaleTransformation(_this.cachedDomain[1]) - scale.scaleTransformation(_this.cachedDomain[0]);
                var lastSeenLength = scale.scaleTransformation(_this.lastSeenDomain[1]) - scale.scaleTransformation(_this.lastSeenDomain[0]);
                _this.scale = (cachedLength / lastSeenLength) || 1;
                _this.translate = scale.scaleTransformation(_this.cachedDomain[0]) - scale.scaleTransformation(_this.lastSeenDomain[0]) * _this.scale || 0;
            };
        }
        DomainTransform.prototype.reset = function () {
            this.scale = 1;
            this.translate = 0;
            this.cachedDomain = this.lastSeenDomain;
        };
        DomainTransform.prototype.setDomain = function (scale) {
            this.cachedDomain = scale.getTransformationDomain();
        };
        return DomainTransform;
    }());
    /**
     * Manages deferred rendering callbacks.
     *
     * Call `setDomains` when deferred rendering is initially enabled to fix the
     * current domain values.
     *
     * Call `updateDomains` when scale domains change, which uses the domain to
     * compute CSS-tyle transform parameters passed to `applyTransformCallback`,
     * mimicking the result of a full re-render. After a deferred timeout, invoke
     * `applyTransformCallback` again with an identity transform and finally invoke
     * `renderCallback`, which should actually redraw the plot.
     *
     * Call `resetTransforms` just prior to re-rendering into the canvas. This
     * ensures that the canvas is at 1:1 scaling.
     */
    var DeferredRenderer = /** @class */ (function () {
        function DeferredRenderer(renderCallback, applyTransformCallback) {
            var _this = this;
            this.renderCallback = renderCallback;
            this.applyTransformCallback = applyTransformCallback;
            this.domainTransformX = new DomainTransform();
            this.domainTransformY = new DomainTransform();
            this.renderDeferred = function () {
                _this.applyTransform();
                clearTimeout(_this.timeoutToken);
                _this.timeoutToken = setTimeout(function () {
                    _this.renderCallback();
                }, DeferredRenderer.DEFERRED_RENDERING_DELAY);
            };
        }
        DeferredRenderer.prototype.setDomains = function (scaleX, scaleY) {
            if (scaleX) {
                this.domainTransformX.setDomain(scaleX);
            }
            if (scaleY) {
                this.domainTransformY.setDomain(scaleY);
            }
            this.renderDeferred();
        };
        DeferredRenderer.prototype.updateDomains = function (scaleX, scaleY) {
            if (scaleX) {
                this.domainTransformX.updateDomain(scaleX);
            }
            if (scaleY) {
                this.domainTransformY.updateDomain(scaleY);
            }
            this.renderDeferred();
        };
        DeferredRenderer.prototype.resetTransforms = function () {
            this.domainTransformX.reset();
            this.domainTransformY.reset();
            this.applyTransform();
        };
        DeferredRenderer.prototype.applyTransform = function () {
            this.applyTransformCallback(this.domainTransformX.translate, this.domainTransformY.translate, this.domainTransformX.scale, this.domainTransformY.scale);
        };
        DeferredRenderer.DEFERRED_RENDERING_DELAY = 200;
        return DeferredRenderer;
    }());
    exports.DeferredRenderer = DeferredRenderer;
    });

    var plot = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });













    exports.Renderer = makeEnum_1.makeEnum(["svg", "canvas"]);
    var Plot = /** @class */ (function (_super) {
        tslib_es6.__extends(Plot, _super);
        /**
         * A Plot draws some visualization of the inputted Datasets.
         *
         * @constructor
         */
        function Plot() {
            var _this = _super.call(this) || this;
            /**
             * Whether the backing datasets have changed since this plot's last render.
             */
            _this._dataChanged = false;
            /**
             * Mapping from attribute names to the extents ([min, max]) values that that attribute takes on.
             */
            _this._attrExtents = {};
            _this._animate = false;
            /**
             * The Animators for this plot. Each plot exposes a set of "animator key" strings that
             * define how different parts of that particular Plot animates. For instance, Rectangle
             * Plots have a "rectangles" animator key which controls how the <rect>s are animated.
             * @see animator()
             *
             * There are two common animators that most Plots respect: "main" and "reset". In general,
             * Plots draw in two steps: first they "reset" their visual elements (e.g. scatter plots set
             * all the dots to size 0), and then they do the "main" animation into the correct visualization
             * (e.g. scatter plot dots grow to their specified size).
             */
            _this._animators = {};
            /**
             * Mapping from property names to the extents ([min, max]) values that that
             * property takes on.
             */
            _this._propertyExtents = {};
            _this._resetEntityStore = function () {
                _this._cachedEntityStore = undefined;
            };
            _this._overflowHidden = true;
            _this.addClass("plot");
            _this._datasetToDrawer = new utils$1.Map();
            _this._attrBindings = d3.map();
            _this._includedValuesProvider = function (scale, ignoreAnchorState) {
                return _this._includedValuesForScale(scale, ignoreAnchorState);
            };
            _this._renderCallback = function () { return _this.render(); };
            _this._onDatasetUpdateCallback = function () { return _this._onDatasetUpdate(); };
            _this._propertyBindings = d3.map();
            var mainAnimator = new animators.Easing().maxTotalDuration(Plot._ANIMATION_MAX_DURATION);
            _this.animator(commons.Animator.MAIN, mainAnimator);
            _this.animator(commons.Animator.RESET, new animators.Null());
            _this._deferredResetEntityStore = utils$1.Window.debounce(deferredRenderer.DeferredRenderer.DEFERRED_RENDERING_DELAY, _this._resetEntityStore);
            return _this;
        }
        Plot.getTotalDrawTime = function (data, drawSteps) {
            return drawSteps.reduce(function (time, drawStep) { return time + drawStep.animator.totalTime(data.length); }, 0);
        };
        Plot.applyDrawSteps = function (drawSteps, dataset) {
            var appliedDrawSteps = drawSteps.map(function (drawStep) {
                var attrToProjector = drawStep.attrToProjector;
                var attrToAppliedProjector = {};
                Object.keys(attrToProjector).forEach(function (attr) {
                    attrToAppliedProjector[attr] =
                        function (datum, index) { return attrToProjector[attr](datum, index, dataset); };
                });
                return {
                    attrToAppliedProjector: attrToAppliedProjector,
                    animator: drawStep.animator,
                };
            });
            return appliedDrawSteps;
        };
        Plot.prototype.anchor = function (selection) {
            selection = coerceD3.coerceExternalD3(selection);
            _super.prototype.anchor.call(this, selection);
            this._dataChanged = true;
            this._resetEntityStore();
            this._updateExtents();
            return this;
        };
        Plot.prototype._setup = function () {
            var _this = this;
            if (this._isSetup) {
                return;
            }
            _super.prototype._setup.call(this);
            if (this._canvas != null) {
                this._appendCanvasNode();
            }
            this._renderArea = this.content().append("g").classed("render-area", true);
            this.datasets().forEach(function (dataset) { return _this._createNodesForDataset(dataset); });
        };
        Plot.prototype._appendCanvasNode = function () {
            var canvasContainer = this.element().select(".plot-canvas-container");
            if (canvasContainer.empty()) {
                canvasContainer = this.element().append("div").classed("plot-canvas-container", true);
                canvasContainer.node().appendChild(this._canvas.node());
            }
        };
        Plot.prototype.setBounds = function (width, height, originX, originY) {
            _super.prototype.setBounds.call(this, width, height, originX, originY);
            this._updateExtents();
            if (this._canvas != null) {
                if (this._bufferCanvas && !this._bufferCanvasValid) {
                    // copy current canvas to buffer 1:1
                    //
                    // Why use a buffer canvas?
                    // As soon as we change the size of a canvas with css or attributes, it
                    // clears the contents. Without a buffer canvas, this requires
                    // drag-resizable charts to immediately do a full redraw while you
                    // drag-resize, which can cause jank. To avoid that, this buffer canvas
                    // stores the current canvas contents when the resize starts and redraws
                    // it into the resized canvas. Eventually, the deferred rendering
                    // callback will trigger and do a full-rez redraw. If deferred rendering
                    // is disabled, the buffer copy will be overwritten immediately by a
                    // full redraw.
                    this._bufferCanvas.attr("width", this._canvas.attr("width"));
                    this._bufferCanvas.attr("height", this._canvas.attr("height"));
                    var btx = this._bufferCanvas.node().getContext("2d");
                    // for headless test compat (jest)
                    if (btx) {
                        var originalCanvas = this._canvas.node();
                        if (originalCanvas.width > 0 && originalCanvas.height > 0) {
                            if (btx.canvas.width > 0 && btx.canvas.height > 0) {
                                btx.drawImage(originalCanvas, 0, 0);
                            }
                        }
                        else {
                            console.warn("Failed to fill buffer canvas with with 0x0 canvas");
                        }
                    }
                    this._bufferCanvasValid = true;
                }
                // update canvas size
                var ratio = (window.devicePixelRatio != null) ? window.devicePixelRatio : 1;
                // update canvas width/height taking into account retina displays.
                // This will also clear the canvas of any drawn elements so we should
                // be sure not to computeLayout() without a render() in the future.
                this._canvas.attr("width", width * ratio);
                this._canvas.attr("height", height * ratio);
                // reset the transform then set the scale factor
                var ctx = this._canvas.node().getContext("2d");
                // null check for headless test compat (jest)
                if (ctx) {
                    ctx.setTransform(ratio, 0, 0, ratio, 0, 0);
                    if (this._bufferCanvas) {
                        var bufferCanvas = this._bufferCanvas.node();
                        if (bufferCanvas.width > 0 && bufferCanvas.height > 0) {
                            if (ctx.canvas.width > 0 && ctx.canvas.height > 0) {
                                // draw buffer to current canvas at new size
                                ctx.drawImage(bufferCanvas, 0, 0, width, height);
                            }
                        }
                        else {
                            console.warn("Failed to fill canvas with 0x0 buffer canvas");
                        }
                    }
                }
            }
            return this;
        };
        Plot.prototype.destroy = function () {
            var _this = this;
            _super.prototype.destroy.call(this);
            this._scales().forEach(function (scale) { return scale.offUpdate(_this._renderCallback); });
            this.datasets([]);
        };
        /**
         * Setup the DOM nodes for the given dataset. This is a separate
         * step from "creating the Drawer" since the element may not be setup yet
         * (in which case the _renderArea is null because the .element() and .content()
         * are null). Also because subclasses may do more than just configure one
         * single drawer (e.g. adding text drawing capabilities).
         */
        Plot.prototype._createNodesForDataset = function (dataset) {
            var drawer = this._datasetToDrawer.get(dataset);
            if (this.renderer() === "svg") {
                drawer.useSVG(this._renderArea);
            }
            else {
                drawer.useCanvas(this._canvas);
            }
            return drawer;
        };
        /**
         * Create a new Drawer. Subclasses should override this to return
         * a Drawer that draws the correct shapes for this plot.
         */
        Plot.prototype._createDrawer = function (dataset) {
            return new drawer.ProxyDrawer(function () { return new svgDrawer.SVGDrawer("path", ""); }, function (ctx) { return new canvasDrawer.CanvasDrawer(ctx, function () { }); });
        };
        Plot.prototype._getAnimator = function (key) {
            if (this._animateOnNextRender()) {
                return this._animators[key] || new animators.Null();
            }
            else {
                return new animators.Null();
            }
        };
        Plot.prototype._onDatasetUpdate = function () {
            this._updateExtents();
            this._dataChanged = true;
            this._resetEntityStore();
            this.renderLowPriority();
        };
        Plot.prototype.attr = function (attr, attrValue, scale) {
            if (attrValue == null) {
                return this._attrBindings.get(attr);
            }
            this._bindAttr(attr, attrValue, scale);
            this.render(); // queue a re-render upon changing projector
            return this;
        };
        Plot.prototype._bindProperty = function (property, valueOrFn, scale, postScale) {
            var binding = this._propertyBindings.get(property);
            var oldScale = binding != null ? binding.scale : null;
            var accessor = typeof valueOrFn === "function" ? valueOrFn : function () { return valueOrFn; };
            this._propertyBindings.set(property, { accessor: accessor, scale: scale, postScale: postScale });
            if (oldScale != null) {
                this._uninstallScaleForKey(oldScale, property);
            }
            if (scale != null) {
                this._installScaleForKey(scale, property);
            }
            this._clearAttrToProjectorCache();
        };
        Plot.prototype._bindAttr = function (attr, valueOrFn, scale) {
            var binding = this._attrBindings.get(attr);
            var oldScale = binding != null ? binding.scale : null;
            var accessor = typeof valueOrFn === "function" ? valueOrFn : function () { return valueOrFn; };
            this._attrBindings.set(attr, { accessor: accessor, scale: scale });
            if (oldScale != null) {
                this._uninstallScaleForKey(oldScale, attr);
            }
            if (scale != null) {
                this._installScaleForKey(scale, attr);
            }
            this._clearAttrToProjectorCache();
        };
        Plot.prototype._clearAttrToProjectorCache = function () {
            delete this._cachedAttrToProjector;
        };
        Plot.prototype._getAttrToProjector = function () {
            if (this._cachedAttrToProjector == null) {
                var projectors = this._generateAttrToProjector();
                if (Plot.OPTIMIZE_MEMOIZE_PROJECTORS) {
                    projectors = memoize.memoizeProjectors(projectors);
                }
                this._cachedAttrToProjector = projectors;
            }
            // return shallow clone of cached projector
            return utils$1.assign({}, this._cachedAttrToProjector);
        };
        Plot.prototype._generateAttrToProjector = function () {
            var h = {};
            this._attrBindings.each(function (binding, attr) {
                h[attr] = Plot._scaledAccessor(binding);
            });
            var propertyProjectors = this._propertyProjectors();
            Object.keys(propertyProjectors).forEach(function (key) {
                if (h[key] == null) {
                    h[key] = propertyProjectors[key];
                }
            });
            return h;
        };
        Plot.prototype.renderImmediately = function () {
            _super.prototype.renderImmediately.call(this);
            if (this._isAnchored) {
                this._paint();
                this._dataChanged = false;
            }
            return this;
        };
        Plot.prototype.renderLowPriority = function () {
            this._renderCallback();
            return this;
        };
        Plot.prototype.animated = function (willAnimate) {
            if (willAnimate == null) {
                return this._animate;
            }
            this._animate = willAnimate;
            return this;
        };
        Plot.prototype.detach = function () {
            _super.prototype.detach.call(this);
            // make the domain resize
            this._updateExtents();
            return this;
        };
        /**
         * @returns {Scale[]} A unique array of all scales currently used by the Plot.
         */
        Plot.prototype._scales = function () {
            var scales = [];
            this._attrBindings.each(function (binding, attr) {
                var scale = binding.scale;
                if (scale != null && scales.indexOf(scale) === -1) {
                    scales.push(scale);
                }
            });
            this._propertyBindings.each(function (binding, property) {
                var scale = binding.scale;
                if (scale != null && scales.indexOf(scale) === -1) {
                    scales.push(scale);
                }
            });
            return scales;
        };
        /**
         * Updates the extents associated with each attribute, then autodomains all scales the Plot uses.
         */
        Plot.prototype._updateExtents = function () {
            var _this = this;
            this._resetEntityStore();
            this._scales().forEach(function (scale) { return scale.addIncludedValuesProvider(_this._includedValuesProvider); });
        };
        Plot.prototype._filterForProperty = function (property) {
            return null;
        };
        Plot.prototype.getExtentsForAttr = function (attr) {
            var _this = this;
            if (this._attrExtents[attr] == null) {
                var thunk = memoize.memThunk(function () { return _this.datasets(); }, function () { return _this._attrBindings.get(attr); }, function (datasets, accScaleBinding) {
                    if (accScaleBinding == null || accScaleBinding.accessor == null) {
                        return null;
                    }
                    return datasets.map(function (dataset) { return computeExtent(dataset, accScaleBinding, null); });
                });
                this._attrExtents[attr] = thunk;
            }
            return this._attrExtents[attr]();
        };
        /**
         * Override in subclass to add special extents, such as included values
         */
        Plot.prototype.getExtentsForProperty = function (property) {
            var _this = this;
            if (this._propertyExtents[property] == null) {
                var thunk = memoize.memThunk(function () { return _this.datasets(); }, function () { return _this._propertyBindings.get(property); }, function () { return _this._filterForProperty(property); }, function (datasets, accScaleBinding, filter) {
                    if (accScaleBinding == null || accScaleBinding.accessor == null) {
                        return null;
                    }
                    return datasets.map(function (dataset) { return computeExtent(dataset, accScaleBinding, filter); });
                });
                this._propertyExtents[property] = thunk;
            }
            return this._propertyExtents[property]();
        };
        Plot.prototype._includedValuesForScale = function (scale, ignoreAttachState) {
            var _this = this;
            if (!this._isAnchored && !ignoreAttachState) {
                return [];
            }
            var includedValues = [];
            this._attrBindings.each(function (binding, attr) {
                if (binding.scale === scale) {
                    var extents = _this.getExtentsForAttr(attr);
                    if (extents != null) {
                        includedValues = includedValues.concat(d3.merge(extents));
                    }
                }
            });
            this._propertyBindings.each(function (binding, property) {
                if (binding.scale === scale) {
                    var extents = _this.getExtentsForProperty(property);
                    if (extents != null) {
                        includedValues = includedValues.concat(d3.merge(extents));
                    }
                }
            });
            return includedValues;
        };
        Plot.prototype.animator = function (animatorKey, animator) {
            if (animator === undefined) {
                return this._animators[animatorKey];
            }
            else {
                this._animators[animatorKey] = animator;
                return this;
            }
        };
        Plot.prototype.renderer = function (renderer) {
            var _this = this;
            if (renderer === undefined) {
                return this._canvas == null ? "svg" : "canvas";
            }
            else {
                if (this._canvas == null && renderer === "canvas") {
                    // construct the canvas, remove drawer's renderAreas, set drawer's canvas
                    this._canvas = d3.select(document.createElement("canvas")).classed("plot-canvas", true);
                    this._bufferCanvas = d3.select(document.createElement("canvas"));
                    if (this.element() != null) {
                        this._appendCanvasNode();
                    }
                    this._datasetToDrawer.forEach(function (drawer) {
                        drawer.useCanvas(_this._canvas);
                    });
                    this.render();
                }
                else if (this._canvas != null && renderer == "svg") {
                    this._canvas.remove();
                    this._canvas = null;
                    this._bufferCanvas = null;
                    this._datasetToDrawer.forEach(function (drawer) {
                        drawer.useSVG(_this._renderArea);
                    });
                    this.render();
                }
                return this;
            }
        };
        /**
         * Adds a Dataset to the Plot.
         *
         * @param {Dataset} dataset
         * @returns {Plot} The calling Plot.
         */
        Plot.prototype.addDataset = function (dataset) {
            this._addDataset(dataset);
            this._onDatasetUpdate();
            return this;
        };
        Plot.prototype._addDataset = function (dataset) {
            this._removeDataset(dataset);
            var drawer = this._createDrawer(dataset);
            this._datasetToDrawer.set(dataset, drawer);
            if (this._isSetup) {
                this._createNodesForDataset(dataset);
            }
            dataset.onUpdate(this._onDatasetUpdateCallback);
            return this;
        };
        /**
         * Removes a Dataset from the Plot.
         *
         * @param {Dataset} dataset
         * @returns {Plot} The calling Plot.
         */
        Plot.prototype.removeDataset = function (dataset) {
            this._removeDataset(dataset);
            this._onDatasetUpdate();
            return this;
        };
        Plot.prototype._removeDataset = function (dataset) {
            if (this.datasets().indexOf(dataset) === -1) {
                return this;
            }
            this._removeDatasetNodes(dataset);
            dataset.offUpdate(this._onDatasetUpdateCallback);
            this._datasetToDrawer.delete(dataset);
            return this;
        };
        Plot.prototype._removeDatasetNodes = function (dataset) {
            var drawer = this._datasetToDrawer.get(dataset);
            drawer.remove();
        };
        Plot.prototype.datasets = function (datasets) {
            var _this = this;
            var currentDatasets = [];
            this._datasetToDrawer.forEach(function (drawer, dataset) { return currentDatasets.push(dataset); });
            if (datasets == null) {
                return currentDatasets;
            }
            currentDatasets.forEach(function (dataset) { return _this._removeDataset(dataset); });
            datasets.forEach(function (dataset) { return _this._addDataset(dataset); });
            this._onDatasetUpdate();
            return this;
        };
        Plot.prototype._generateDrawSteps = function () {
            return [{ attrToProjector: this._getAttrToProjector(), animator: new animators.Null() }];
        };
        Plot.prototype._additionalPaint = function (time) {
            // no-op
        };
        /**
         * _buildLightweightPlotEntities constucts {LightweightPlotEntity[]} from
         * all the entities in the plot
         * @param {Dataset[]} [datasets] - datasets comprising this plot
         */
        Plot.prototype._buildLightweightPlotEntities = function (datasets) {
            var _this = this;
            var lightweightPlotEntities = [];
            datasets.forEach(function (dataset, datasetIndex) {
                var drawer = _this._datasetToDrawer.get(dataset);
                var validDatumIndex = 0;
                var data = dataset.data();
                var dataLen = data.length;
                var _loop_1 = function (datumIndex) {
                    var datum = data[datumIndex];
                    var position = _this._pixelPoint(datum, datumIndex, dataset);
                    if (utils$1.Math.isNaN(position.x) || utils$1.Math.isNaN(position.y)) {
                        return "continue";
                    }
                    var plot = _this;
                    lightweightPlotEntities.push({
                        datum: datum,
                        get position() {
                            // only calculate position when needing to improve pan zoom performance #3159
                            return plot._pixelPoint.call(plot, datum, datumIndex, dataset);
                        },
                        index: datumIndex,
                        dataset: dataset,
                        datasetIndex: datasetIndex,
                        component: _this,
                        drawer: drawer,
                        validDatumIndex: validDatumIndex,
                    });
                    validDatumIndex++;
                };
                for (var datumIndex = 0; datumIndex < dataLen; datumIndex++) {
                    _loop_1(datumIndex);
                }
            });
            return lightweightPlotEntities;
        };
        Plot.prototype._getDataToDraw = function () {
            var dataToDraw = new utils$1.Map();
            this.datasets().forEach(function (dataset) { return dataToDraw.set(dataset, dataset.data()); });
            return dataToDraw;
        };
        Plot.prototype._paint = function () {
            var _this = this;
            delete this._cachedAttrToProjector;
            var drawSteps = this._generateDrawSteps();
            var dataToDraw = this._getDataToDraw();
            var drawers = this.datasets().map(function (dataset) { return _this._datasetToDrawer.get(dataset); });
            if (this.renderer() === "canvas") {
                var canvas = this._canvas.node();
                var context_1 = canvas.getContext("2d");
                context_1.clearRect(0, 0, canvas.clientWidth, canvas.clientHeight);
                this._bufferCanvasValid = false;
            }
            this.datasets().forEach(function (ds, i) {
                var appliedDrawSteps = Plot.applyDrawSteps(drawSteps, ds);
                drawers[i].draw(dataToDraw.get(ds), appliedDrawSteps);
            });
            var times = this.datasets().map(function (ds, i) { return Plot.getTotalDrawTime(dataToDraw.get(ds), drawSteps); });
            var maxTime = utils$1.Math.max(times, 0);
            this._additionalPaint(maxTime);
        };
        /**
         * Retrieves the drawn visual elements for the specified Datasets as a d3 Selection.
         * Not supported on canvas renderer.
         *
         * @param {Dataset[]} [datasets] The Datasets to retrieve the Selections for.
         *   If not provided, Selections will be retrieved for all Datasets on the Plot.
         * @returns {d3.Selection}
         */
        Plot.prototype.selections = function (datasets) {
            var _this = this;
            if (datasets === void 0) { datasets = this.datasets(); }
            if (this.renderer() === "canvas") {
                // return empty selection
                return d3.selectAll();
            }
            else {
                var selections_1 = [];
                datasets.forEach(function (dataset) {
                    var drawer = _this._datasetToDrawer.get(dataset);
                    if (drawer == null) {
                        return;
                    }
                    var visualPrimitives = drawer.getVisualPrimitives();
                    selections_1.push.apply(selections_1, visualPrimitives);
                });
                return d3.selectAll(selections_1);
            }
        };
        /**
         * Gets the Entities associated with the specified Datasets.
         *
         * @param {Dataset[]} datasets The Datasets to retrieve the Entities for.
         *   If not provided, returns defaults to all Datasets on the Plot.
         * @return {Plots.PlotEntity[]}
         */
        Plot.prototype.entities = function (datasets) {
            var _this = this;
            return this._getEntityStore(datasets).entities().map(function (entity) { return _this._lightweightPlotEntityToPlotEntity(entity); });
        };
        /**
         * Gets the `Plots.PlotEntity`s in the plot, whereby the
         * `Plots.ILightweightPlotEntity`s are filtered by the provided predicate.
         * Since `_lightweightPlotEntityToPlotEntity` can be expensive, this method
         * can be useful for performance optimization.
         *
         * @param where A filter predicate that is applied to lightweight entities
         * before upconverting them to full plot entities
         * @return {Plots.PlotEntity[]}
         */
        Plot.prototype.filterEntities = function (where) {
            var _this = this;
            return this._getEntityStore()
                .entities()
                .filter(where)
                .map(function (entity) { return _this._lightweightPlotEntityToPlotEntity(entity); });
        };
        /**
         * _getEntityStore returns the store of all Entities associated with the specified dataset
         *
         * @param {Dataset[]} [datasets] - The datasets with which to construct the store. If no datasets
         * are specified all datasets will be used.
         */
        Plot.prototype._getEntityStore = function (datasets) {
            var _this = this;
            var entityBoundsFactory = function (entity) { return _this._entityBounds(entity); };
            if (datasets !== undefined) {
                var entityStore = new utils$1.EntityStore();
                entityStore.addAll(this._buildLightweightPlotEntities(datasets), entityBoundsFactory, this._localOriginBounds());
                return entityStore;
            }
            else if (this._cachedEntityStore === undefined) {
                var entityStore = new utils$1.EntityStore();
                entityStore.addAll(this._buildLightweightPlotEntities(this.datasets()), entityBoundsFactory, this._localOriginBounds());
                this._cachedEntityStore = entityStore;
            }
            return this._cachedEntityStore;
        };
        /**
         * _localOriginBounds returns bounds of the plot from its own origin, rather than from parent origin (as provided by `this.bounds()`)
         *
         * @return {Bounds}
         */
        Plot.prototype._localOriginBounds = function () {
            return {
                topLeft: { x: 0, y: 0 },
                bottomRight: { x: this.width(), y: this.height() },
            };
        };
        Plot.prototype._entityBounds = function (entity) {
            var datum = entity.datum, index = entity.index, dataset = entity.dataset;
            var _a = this._pixelPoint(datum, index, dataset), x = _a.x, y = _a.y;
            return { x: x, y: y, width: 0, height: 0 };
        };
        Plot.prototype._lightweightPlotEntityToPlotEntity = function (entity) {
            var plotEntity = {
                bounds: this._entityBounds(entity),
                component: entity.component,
                dataset: entity.dataset,
                datasetIndex: entity.datasetIndex,
                datum: entity.datum,
                index: entity.index,
                position: entity.position,
                selection: d3.select(entity.drawer.getVisualPrimitiveAtIndex(entity.validDatumIndex)),
            };
            return plotEntity;
        };
        /**
         * Gets the PlotEntities at a particular Point.
         *
         * Each plot type determines how to locate entities at or near the query
         * point. For example, line and area charts will return the nearest entity,
         * but bar charts will only return the entities that fully contain the query
         * point.
         *
         * @param {Point} point The point to query.
         * @returns {PlotEntity[]} The PlotEntities at the particular point
         */
        Plot.prototype.entitiesAt = function (point) {
            throw new Error("plots must implement entitiesAt");
        };
        /**
         * Returns the {Plots.PlotEntity} nearest to the query point,
         * or undefined if no {Plots.PlotEntity} can be found.
         *
         * @param {Point} queryPoint
         * @returns {Plots.PlotEntity} The nearest PlotEntity, or undefined if no {Plots.PlotEntity} can be found.
         */
        Plot.prototype.entityNearest = function (queryPoint) {
            var nearest = this._getEntityStore().entityNearest(queryPoint);
            return nearest === undefined ? undefined : this._lightweightPlotEntityToPlotEntity(nearest);
        };
        Plot.prototype.entitiesIn = function (xRangeOrBounds, yRange) {
            var queryBounds;
            if (yRange == null) {
                var bounds = xRangeOrBounds;
                queryBounds = {
                    x: bounds.topLeft.x,
                    y: bounds.topLeft.y,
                    width: bounds.bottomRight.x - bounds.topLeft.x,
                    height: bounds.bottomRight.y - bounds.topLeft.y,
                };
            }
            else {
                var xRange = xRangeOrBounds;
                queryBounds = {
                    x: xRange.min,
                    y: yRange.min,
                    width: xRange.max - xRange.min,
                    height: yRange.max - yRange.min,
                };
            }
            return this.entitiesInBounds(queryBounds);
        };
        /**
         * Returns the entites whose bounding boxes overlap the parameter.
         *
         * `queryBounds` are in pixel space, measured from the origin of this plot.
         */
        Plot.prototype.entitiesInBounds = function (queryBounds) {
            var _this = this;
            var found = this._getEntityStore().entitiesInBounds(queryBounds);
            if (!found) {
                return undefined;
            }
            return found.map(function (entity) { return _this._lightweightPlotEntityToPlotEntity(entity); });
        };
        /**
         * Returns the entites whose bounding boxes overlap the `queryBounds`
         * parameter on the x-axis.
         *
         * `queryBounds` are in pixel space, measured from the origin of this plot.
         */
        Plot.prototype.entitiesInXBounds = function (queryBounds) {
            var _this = this;
            var found = this._getEntityStore().entitiesInXBounds(queryBounds);
            if (!found) {
                return undefined;
            }
            return found.map(function (entity) { return _this._lightweightPlotEntityToPlotEntity(entity); });
        };
        /**
         * Returns the entites whose bounding boxes overlap the `queryBounds`
         * parameter on the y-axis.
         *
         * `queryBounds` are in pixel space, measured from the origin of this plot.
         */
        Plot.prototype.entitiesInYBounds = function (queryBounds) {
            var _this = this;
            var found = this._getEntityStore().entitiesInYBounds(queryBounds);
            if (!found) {
                return undefined;
            }
            return found.map(function (entity) { return _this._lightweightPlotEntityToPlotEntity(entity); });
        };
        Plot.prototype._uninstallScaleForKey = function (scale, key) {
            scale.offUpdate(this._renderCallback);
            scale.offUpdate(this._deferredResetEntityStore);
            scale.removeIncludedValuesProvider(this._includedValuesProvider);
        };
        Plot.prototype._installScaleForKey = function (scale, key) {
            scale.onUpdate(this._renderCallback);
            scale.onUpdate(this._deferredResetEntityStore);
            scale.addIncludedValuesProvider(this._includedValuesProvider);
        };
        Plot.prototype._propertyProjectors = function () {
            return {};
        };
        Plot._scaledAccessor = function (binding) {
            var scale = binding.scale, accessor = binding.accessor, postScale = binding.postScale;
            // if provided, apply scale
            var scaledAccesor = scale == null ? accessor :
                function (d, i, ds) { return scale.scale(accessor(d, i, ds)); };
            // if provided, apply post scale
            var postScaledAccesor = postScale == null ? scaledAccesor :
                function (d, i, ds) { return postScale(scaledAccesor(d, i, ds), d, i, ds); };
            return postScaledAccesor;
        };
        Plot.prototype._pixelPoint = function (datum, index, dataset) {
            return { x: 0, y: 0 };
        };
        Plot.prototype._animateOnNextRender = function () {
            return this._animate && this._dataChanged;
        };
        Plot.OPTIMIZE_MEMOIZE_PROJECTORS = false;
        Plot._ANIMATION_MAX_DURATION = 600;
        return Plot;
    }(component.Component));
    exports.Plot = Plot;
    function computeExtent(dataset, accScaleBinding, filter) {
        var accessor = accScaleBinding.accessor;
        var scale = accScaleBinding.scale;
        if (scale == null) {
            return [];
        }
        var data = dataset.data();
        if (filter != null) {
            data = data.filter(function (d, i) { return filter(d, i, dataset); });
        }
        var appliedAccessor = function (d, i) { return accessor(d, i, dataset); };
        var mappedData = data.map(appliedAccessor);
        return scale.extentOfValues(mappedData);
    }
    });

    var plotGroup = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });




    var PlotGroup = /** @class */ (function (_super) {
        tslib_es6.__extends(PlotGroup, _super);
        function PlotGroup() {
            return _super !== null && _super.apply(this, arguments) || this;
        }
        PlotGroup.prototype.entityNearest = function (point) {
            var closestPlotEntity;
            var minDistSquared = Infinity;
            this.components().forEach(function (plotComponent) {
                // we know it's a Plot since .append() throws a runtime error otherwise
                var plot = plotComponent;
                var candidatePlotEntity = plot.entityNearest(point);
                if (candidatePlotEntity == null) {
                    return;
                }
                var distSquared = utils$1.Math.distanceSquared(candidatePlotEntity.position, point);
                if (distSquared <= minDistSquared) {
                    minDistSquared = distSquared;
                    closestPlotEntity = candidatePlotEntity;
                }
            });
            return closestPlotEntity;
        };
        /**
         * Adds a Plot to this Plot Group.
         * The added Plot will be rendered above Plots already in the Group.
         */
        PlotGroup.prototype.append = function (plot$1) {
            if (plot$1 != null && !(plot$1 instanceof plot.Plot)) {
                throw new Error("Plot Group only accepts plots");
            }
            _super.prototype.append.call(this, plot$1);
            return this;
        };
        return PlotGroup;
    }(group.Group));
    exports.PlotGroup = PlotGroup;
    });

    var table = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });




    var Table = /** @class */ (function (_super) {
        tslib_es6.__extends(Table, _super);
        /**
         * A Table combines Components in the form of a grid. A
         * common case is combining a y-axis, x-axis, and the plotted data via
         * ```typescript
         * new Table([[yAxis, plot],
         *            [null,  xAxis]]);
         * ```
         *
         * @constructor
         * @param {(Component|null|undefined)[][]} [rows=[]] A 2-D array of Components to be added to the Table.
         *   null can be used if a cell is empty.
         */
        function Table(rows) {
            if (rows === void 0) { rows = []; }
            var _this = _super.call(this) || this;
            _this._rowPadding = 0;
            _this._columnPadding = 0;
            _this._rows = [];
            _this._rowWeights = [];
            _this._columnWeights = [];
            _this._nRows = 0;
            _this._nCols = 0;
            _this._calculatedLayout = null;
            _this.addClass("table");
            rows.forEach(function (row, rowIndex) {
                row.forEach(function (component, colIndex) {
                    if (component != null) {
                        _this.add(component, rowIndex, colIndex);
                    }
                });
            });
            return _this;
        }
        Table.prototype._forEach = function (callback) {
            for (var r = 0; r < this._nRows; r++) {
                for (var c = 0; c < this._nCols; c++) {
                    if (this._rows[r][c] != null) {
                        callback(this._rows[r][c]);
                    }
                }
            }
        };
        /**
         * Checks whether the specified Component is in the Table.
         */
        Table.prototype.has = function (component) {
            for (var r = 0; r < this._nRows; r++) {
                for (var c = 0; c < this._nCols; c++) {
                    if (this._rows[r][c] === component) {
                        return true;
                    }
                }
            }
            return false;
        };
        /**
         * Returns the Component at the specified row and column index.
         *
         * @param {number} rowIndex
         * @param {number} columnIndex
         * @returns {Component} The Component at the specified position, or null if no Component is there.
         */
        Table.prototype.componentAt = function (rowIndex, columnIndex) {
            if (rowIndex < 0 || rowIndex >= this._nRows || columnIndex < 0 || columnIndex >= this._nCols) {
                return null;
            }
            return this._rows[rowIndex][columnIndex];
        };
        /**
         * Adds a Component in the specified row and column position.
         *
         * For example, instead of calling `new Table([[a, b], [null, c]])`, you
         * could call
         * var table = new Plottable.Components.Table();
         * table.add(a, 0, 0);
         * table.add(b, 0, 1);
         * table.add(c, 1, 1);
         *
         * @param {Component} component The Component to be added.
         * @param {number} row
         * @param {number} col
         * @returns {Table} The calling Table.
         */
        Table.prototype.add = function (component, row, col) {
            if (component == null) {
                throw Error("Cannot add null to a table cell");
            }
            if (!this.has(component)) {
                var currentComponent = this._rows[row] && this._rows[row][col];
                if (currentComponent != null) {
                    throw new Error("cell is occupied");
                }
                component.detach();
                this._nRows = Math.max(row + 1, this._nRows);
                this._nCols = Math.max(col + 1, this._nCols);
                this._padTableToSize(this._nRows, this._nCols);
                this._rows[row][col] = component;
                this._adoptAndAnchor(component);
                this.redraw();
            }
            return this;
        };
        Table.prototype._remove = function (component) {
            for (var r = 0; r < this._nRows; r++) {
                for (var c = 0; c < this._nCols; c++) {
                    if (this._rows[r][c] === component) {
                        this._rows[r][c] = null;
                        return true;
                    }
                }
            }
            return false;
        };
        Table.prototype._iterateLayout = function (availableWidth, availableHeight, isFinalOffer) {
            if (isFinalOffer === void 0) { isFinalOffer = false; }
            /*
             * Given availableWidth and availableHeight, figure out how to allocate it between rows and columns using an iterative algorithm.
             *
             * For both dimensions, keeps track of "guaranteedSpace", which the fixed-size components have requested, and
             * "proportionalSpace", which is being given to proportionally-growing components according to the weights on the table.
             * Here is how it works (example uses width but it is the same for height). First, columns are guaranteed no width, and
             * the free width is allocated to columns based on their colWeights. Then, in determineGuarantees, every component is
             * offered its column's width and may request some amount of it, which increases that column's guaranteed
             * width. If there are some components that were not satisfied with the width they were offered, and there is free
             * width that has not already been guaranteed, then the remaining width is allocated to the unsatisfied columns and the
             * algorithm runs again. If all components are satisfied, then the remaining width is allocated as proportional space
             * according to the colWeights.
             *
             * The guaranteed width for each column is monotonically increasing as the algorithm iterates. Since it is deterministic
             * and monotonically increasing, if the freeWidth does not change during an iteration it implies that no further progress
             * is possible, so the algorithm will not continue iterating on that dimension's account.
             *
             * If the algorithm runs more than 5 times, we stop and just use whatever we arrived at. It's not clear under what
             * circumstances this will happen or if it will happen at all. A message will be printed to the console if this occurs.
             *
             */
            var rows = this._rows;
            var cols = d3.transpose(this._rows);
            var availableWidthAfterPadding = availableWidth - this._columnPadding * (this._nCols - 1);
            var availableHeightAfterPadding = availableHeight - this._rowPadding * (this._nRows - 1);
            var rowWeights = Table._calcComponentWeights(this._rowWeights, rows, function (c) { return (c == null) || c.fixedHeight(); });
            var colWeights = Table._calcComponentWeights(this._columnWeights, cols, function (c) { return (c == null) || c.fixedWidth(); });
            // To give the table a good starting position to iterate from, we give the fixed-width components half-weight
            // so that they will get some initial space allocated to work with
            var heuristicColWeights = colWeights.map(function (c) { return c === 0 ? 0.5 : c; });
            var heuristicRowWeights = rowWeights.map(function (c) { return c === 0 ? 0.5 : c; });
            var colProportionalSpace = Table._calcProportionalSpace(heuristicColWeights, availableWidthAfterPadding);
            var rowProportionalSpace = Table._calcProportionalSpace(heuristicRowWeights, availableHeightAfterPadding);
            var guaranteedWidths = utils$1.Array.createFilledArray(0, this._nCols);
            var guaranteedHeights = utils$1.Array.createFilledArray(0, this._nRows);
            var freeWidth;
            var freeHeight;
            var nIterations = 0;
            var guarantees;
            var wantsWidth;
            var wantsHeight;
            while (true) {
                var offeredHeights = utils$1.Array.add(guaranteedHeights, rowProportionalSpace);
                var offeredWidths = utils$1.Array.add(guaranteedWidths, colProportionalSpace);
                guarantees = this._determineGuarantees(offeredWidths, offeredHeights, isFinalOffer);
                guaranteedWidths = guarantees.guaranteedWidths;
                guaranteedHeights = guarantees.guaranteedHeights;
                wantsWidth = guarantees.wantsWidthArr.some(function (x) { return x; });
                wantsHeight = guarantees.wantsHeightArr.some(function (x) { return x; });
                var lastFreeWidth = freeWidth;
                var lastFreeHeight = freeHeight;
                freeWidth = availableWidthAfterPadding - d3.sum(guarantees.guaranteedWidths);
                freeHeight = availableHeightAfterPadding - d3.sum(guarantees.guaranteedHeights);
                var xWeights = void 0;
                if (wantsWidth) {
                    xWeights = guarantees.wantsWidthArr.map(function (x) { return x ? 0.1 : 0; });
                    xWeights = utils$1.Array.add(xWeights, colWeights);
                }
                else {
                    xWeights = colWeights;
                }
                var yWeights = void 0;
                if (wantsHeight) {
                    yWeights = guarantees.wantsHeightArr.map(function (x) { return x ? 0.1 : 0; });
                    yWeights = utils$1.Array.add(yWeights, rowWeights);
                }
                else {
                    yWeights = rowWeights;
                }
                colProportionalSpace = Table._calcProportionalSpace(xWeights, freeWidth);
                rowProportionalSpace = Table._calcProportionalSpace(yWeights, freeHeight);
                nIterations++;
                var canImproveWidthAllocation = freeWidth > 0 && freeWidth !== lastFreeWidth;
                var canImproveHeightAllocation = freeHeight > 0 && freeHeight !== lastFreeHeight;
                if (!(canImproveWidthAllocation || canImproveHeightAllocation)) {
                    break;
                }
                if (nIterations > 5) {
                    break;
                }
            }
            // Redo the proportional space one last time, to ensure we use the real weights not the wantsWidth/Height weights
            freeWidth = availableWidthAfterPadding - d3.sum(guarantees.guaranteedWidths);
            freeHeight = availableHeightAfterPadding - d3.sum(guarantees.guaranteedHeights);
            colProportionalSpace = Table._calcProportionalSpace(colWeights, freeWidth);
            rowProportionalSpace = Table._calcProportionalSpace(rowWeights, freeHeight);
            return {
                colProportionalSpace: colProportionalSpace,
                rowProportionalSpace: rowProportionalSpace,
                guaranteedWidths: guarantees.guaranteedWidths,
                guaranteedHeights: guarantees.guaranteedHeights,
                wantsWidth: wantsWidth,
                wantsHeight: wantsHeight,
            };
        };
        Table.prototype._determineGuarantees = function (offeredWidths, offeredHeights, isFinalOffer) {
            if (isFinalOffer === void 0) { isFinalOffer = false; }
            var requestedWidths = utils$1.Array.createFilledArray(0, this._nCols);
            var requestedHeights = utils$1.Array.createFilledArray(0, this._nRows);
            var columnNeedsWidth = utils$1.Array.createFilledArray(false, this._nCols);
            var rowNeedsHeight = utils$1.Array.createFilledArray(false, this._nRows);
            this._rows.forEach(function (row, rowIndex) {
                row.forEach(function (component, colIndex) {
                    var spaceRequest;
                    if (component != null) {
                        spaceRequest = component.requestedSpace(offeredWidths[colIndex], offeredHeights[rowIndex]);
                    }
                    else {
                        spaceRequest = {
                            minWidth: 0,
                            minHeight: 0,
                        };
                    }
                    var columnWidth = isFinalOffer ? Math.min(spaceRequest.minWidth, offeredWidths[colIndex]) : spaceRequest.minWidth;
                    requestedWidths[colIndex] = Math.max(requestedWidths[colIndex], columnWidth);
                    var rowHeight = isFinalOffer ? Math.min(spaceRequest.minHeight, offeredHeights[rowIndex]) : spaceRequest.minHeight;
                    requestedHeights[rowIndex] = Math.max(requestedHeights[rowIndex], rowHeight);
                    var componentNeedsWidth = spaceRequest.minWidth > offeredWidths[colIndex];
                    columnNeedsWidth[colIndex] = columnNeedsWidth[colIndex] || componentNeedsWidth;
                    var componentNeedsHeight = spaceRequest.minHeight > offeredHeights[rowIndex];
                    rowNeedsHeight[rowIndex] = rowNeedsHeight[rowIndex] || componentNeedsHeight;
                });
            });
            return {
                guaranteedWidths: requestedWidths,
                guaranteedHeights: requestedHeights,
                wantsWidthArr: columnNeedsWidth,
                wantsHeightArr: rowNeedsHeight,
            };
        };
        Table.prototype.requestedSpace = function (offeredWidth, offeredHeight) {
            this._calculatedLayout = this._iterateLayout(offeredWidth, offeredHeight);
            return {
                minWidth: d3.sum(this._calculatedLayout.guaranteedWidths),
                minHeight: d3.sum(this._calculatedLayout.guaranteedHeights),
            };
        };
        Table.prototype.computeLayout = function (origin, availableWidth, availableHeight) {
            var _this = this;
            _super.prototype.computeLayout.call(this, origin, availableWidth, availableHeight);
            var lastLayoutWidth = d3.sum(this._calculatedLayout.guaranteedWidths);
            var lastLayoutHeight = d3.sum(this._calculatedLayout.guaranteedHeights);
            var layout = this._calculatedLayout;
            if (lastLayoutWidth > this.width() || lastLayoutHeight > this.height()) {
                layout = this._iterateLayout(this.width(), this.height(), true);
            }
            var childYOrigin = 0;
            var rowHeights = utils$1.Array.add(layout.rowProportionalSpace, layout.guaranteedHeights);
            var colWidths = utils$1.Array.add(layout.colProportionalSpace, layout.guaranteedWidths);
            this._rows.forEach(function (row, rowIndex) {
                var childXOrigin = 0;
                row.forEach(function (component, colIndex) {
                    // recursively compute layout
                    if (component != null) {
                        component.computeLayout({ x: childXOrigin, y: childYOrigin }, colWidths[colIndex], rowHeights[rowIndex]);
                    }
                    childXOrigin += colWidths[colIndex] + _this._columnPadding;
                });
                childYOrigin += rowHeights[rowIndex] + _this._rowPadding;
            });
            return this;
        };
        Table.prototype.rowPadding = function (rowPadding) {
            if (rowPadding == null) {
                return this._rowPadding;
            }
            if (!utils$1.Math.isValidNumber(rowPadding) || rowPadding < 0) {
                throw Error("rowPadding must be a non-negative finite value");
            }
            this._rowPadding = rowPadding;
            this.redraw();
            return this;
        };
        Table.prototype.columnPadding = function (columnPadding) {
            if (columnPadding == null) {
                return this._columnPadding;
            }
            if (!utils$1.Math.isValidNumber(columnPadding) || columnPadding < 0) {
                throw Error("columnPadding must be a non-negative finite value");
            }
            this._columnPadding = columnPadding;
            this.redraw();
            return this;
        };
        Table.prototype.rowWeight = function (index, weight) {
            if (weight == null) {
                return this._rowWeights[index];
            }
            if (!utils$1.Math.isValidNumber(weight) || weight < 0) {
                throw Error("rowWeight must be a non-negative finite value");
            }
            this._rowWeights[index] = weight;
            this.redraw();
            return this;
        };
        Table.prototype.columnWeight = function (index, weight) {
            if (weight == null) {
                return this._columnWeights[index];
            }
            if (!utils$1.Math.isValidNumber(weight) || weight < 0) {
                throw Error("columnWeight must be a non-negative finite value");
            }
            this._columnWeights[index] = weight;
            this.redraw();
            return this;
        };
        Table.prototype.fixedWidth = function () {
            var cols = d3.transpose(this._rows);
            return Table._fixedSpace(cols, function (c) { return (c == null) || c.fixedWidth(); });
        };
        Table.prototype.fixedHeight = function () {
            return Table._fixedSpace(this._rows, function (c) { return (c == null) || c.fixedHeight(); });
        };
        Table.prototype._padTableToSize = function (nRows, nCols) {
            for (var i = 0; i < nRows; i++) {
                if (this._rows[i] === undefined) {
                    this._rows[i] = [];
                    this._rowWeights[i] = null;
                }
                for (var j = 0; j < nCols; j++) {
                    if (this._rows[i][j] === undefined) {
                        this._rows[i][j] = null;
                    }
                }
            }
            for (var j = 0; j < nCols; j++) {
                if (this._columnWeights[j] === undefined) {
                    this._columnWeights[j] = null;
                }
            }
        };
        Table._calcComponentWeights = function (setWeights, componentGroups, fixityAccessor) {
            // If the row/col weight was explicitly set, then return it outright
            // If the weight was not explicitly set, then guess it using the heuristic that if all components are fixed-space
            // then weight is 0, otherwise weight is 1
            return setWeights.map(function (w, i) {
                if (w != null) {
                    return w;
                }
                var fixities = componentGroups[i].map(fixityAccessor);
                var allFixed = fixities.reduce(function (a, b) { return a && b; }, true);
                return allFixed ? 0 : 1;
            });
        };
        Table._calcProportionalSpace = function (weights, freeSpace) {
            var weightSum = d3.sum(weights);
            if (weightSum === 0) {
                return utils$1.Array.createFilledArray(0, weights.length);
            }
            else {
                return weights.map(function (w) { return freeSpace * w / weightSum; });
            }
        };
        Table._fixedSpace = function (componentGroup, fixityAccessor) {
            var all = function (bools) { return bools.reduce(function (a, b) { return a && b; }, true); };
            var groupIsFixed = function (components) { return all(components.map(fixityAccessor)); };
            return all(componentGroup.map(groupIsFixed));
        };
        return Table;
    }(componentContainer.ComponentContainer));
    exports.Table = Table;
    });

    var wrappedLabel = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });


    var DEFAULT_MAX_LINES = 2;
    var WrappedLabel = /** @class */ (function (_super) {
        tslib_es6.__extends(WrappedLabel, _super);
        function WrappedLabel() {
            var _this = _super !== null && _super.apply(this, arguments) || this;
            _this._maxLines = DEFAULT_MAX_LINES;
            return _this;
        }
        WrappedLabel.prototype.requestedSpace = function (offeredWidth, offeredHeight) {
            this._wrapper.maxLines(this._maxLines);
            var offeredLineLength = this.angle() === 0 ? offeredWidth : offeredHeight;
            if (offeredLineLength === 0) {
                offeredLineLength = Infinity;
            }
            var wrapped = this._wrapper.wrap(this._text, this._measurer, offeredLineLength);
            var measuredWrap = this._measurer.measure(wrapped.wrappedText);
            var minWidth = (this.angle() === 0 ? measuredWrap.width : measuredWrap.height) + 2 * this.padding();
            var minHeight = (this.angle() === 0 ? measuredWrap.height : measuredWrap.width) + 2 * this.padding();
            return { minWidth: minWidth, minHeight: minHeight };
        };
        WrappedLabel.prototype.maxLines = function (maxLines) {
            // allow user to un-set by passing in null or undefined explicitly
            if (arguments.length === 0) {
                return this._maxLines;
            }
            this._maxLines = maxLines;
            this.redraw();
            return this;
        };
        return WrappedLabel;
    }(label.Label));
    exports.WrappedLabel = WrappedLabel;
    });

    var xDragBoxLayer = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });


    var XDragBoxLayer = /** @class */ (function (_super) {
        tslib_es6.__extends(XDragBoxLayer, _super);
        /**
         * An XDragBoxLayer is a DragBoxLayer whose size can only be set in the X-direction.
         * The y-values of the bounds() are always set to 0 and the height() of the XDragBoxLayer.
         *
         * @constructor
         */
        function XDragBoxLayer() {
            var _this = _super.call(this) || this;
            _this.addClass("x-drag-box-layer");
            _this._hasCorners = false;
            return _this;
        }
        XDragBoxLayer.prototype.computeLayout = function (origin, availableWidth, availableHeight) {
            _super.prototype.computeLayout.call(this, origin, availableWidth, availableHeight);
            // set correct bounds when width/height changes
            this._setBounds(this.bounds());
            return this;
        };
        XDragBoxLayer.prototype._setBounds = function (newBounds) {
            _super.prototype._setBounds.call(this, {
                topLeft: { x: newBounds.topLeft.x, y: 0 },
                bottomRight: { x: newBounds.bottomRight.x, y: this.height() },
            });
        };
        XDragBoxLayer.prototype._setResizableClasses = function (canResize) {
            if (canResize && this.enabled()) {
                this.addClass("x-resizable");
            }
            else {
                this.removeClass("x-resizable");
            }
        };
        XDragBoxLayer.prototype.yScale = function (yScale) {
            if (yScale == null) {
                return _super.prototype.yScale.call(this);
            }
            throw new Error("yScales cannot be set on an XDragBoxLayer");
        };
        XDragBoxLayer.prototype.yExtent = function (yExtent) {
            if (yExtent == null) {
                return _super.prototype.yExtent.call(this);
            }
            throw new Error("XDragBoxLayer has no yExtent");
        };
        return XDragBoxLayer;
    }(dragBoxLayer.DragBoxLayer));
    exports.XDragBoxLayer = XDragBoxLayer;
    });

    var yDragBoxLayer = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });


    var YDragBoxLayer = /** @class */ (function (_super) {
        tslib_es6.__extends(YDragBoxLayer, _super);
        /**
         * A YDragBoxLayer is a DragBoxLayer whose size can only be set in the Y-direction.
         * The x-values of the bounds() are always set to 0 and the width() of the YDragBoxLayer.
         *
         * @constructor
         */
        function YDragBoxLayer() {
            var _this = _super.call(this) || this;
            _this.addClass("y-drag-box-layer");
            _this._hasCorners = false;
            return _this;
        }
        YDragBoxLayer.prototype.computeLayout = function (origin, availableWidth, availableHeight) {
            _super.prototype.computeLayout.call(this, origin, availableWidth, availableHeight);
            // set correct bounds when width/height changes
            this._setBounds(this.bounds());
            return this;
        };
        YDragBoxLayer.prototype._setBounds = function (newBounds) {
            _super.prototype._setBounds.call(this, {
                topLeft: { x: 0, y: newBounds.topLeft.y },
                bottomRight: { x: this.width(), y: newBounds.bottomRight.y },
            });
        };
        YDragBoxLayer.prototype._setResizableClasses = function (canResize) {
            if (canResize && this.enabled()) {
                this.addClass("y-resizable");
            }
            else {
                this.removeClass("y-resizable");
            }
        };
        YDragBoxLayer.prototype.xScale = function (xScale) {
            if (xScale == null) {
                return _super.prototype.xScale.call(this);
            }
            throw new Error("xScales cannot be set on an YDragBoxLayer");
        };
        YDragBoxLayer.prototype.xExtent = function (xExtent) {
            if (xExtent == null) {
                return _super.prototype.xExtent.call(this);
            }
            throw new Error("YDragBoxLayer has no xExtent");
        };
        return YDragBoxLayer;
    }(dragBoxLayer.DragBoxLayer));
    exports.YDragBoxLayer = YDragBoxLayer;
    });

    var components = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });

    tslib_es6.__exportStar(dragBoxLayer, exports);
    tslib_es6.__exportStar(dragLineLayer, exports);
    tslib_es6.__exportStar(gridlines, exports);
    tslib_es6.__exportStar(group, exports);
    tslib_es6.__exportStar(guideLineLayer, exports);
    tslib_es6.__exportStar(interpolatedColorLegend, exports);
    tslib_es6.__exportStar(label, exports);
    tslib_es6.__exportStar(legend, exports);
    tslib_es6.__exportStar(plotGroup, exports);
    tslib_es6.__exportStar(selectionBoxLayer, exports);
    tslib_es6.__exportStar(table, exports);
    tslib_es6.__exportStar(wrappedLabel, exports);
    tslib_es6.__exportStar(xDragBoxLayer, exports);
    tslib_es6.__exportStar(yDragBoxLayer, exports);
    });

    var arcDrawer = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });


    var ArcSVGDrawer = /** @class */ (function (_super) {
        tslib_es6.__extends(ArcSVGDrawer, _super);
        function ArcSVGDrawer() {
            return _super.call(this, "path", "arc fill") || this;
        }
        ArcSVGDrawer.prototype._applyDefaultAttributes = function (selection) {
            selection.style("stroke", "none");
        };
        return ArcSVGDrawer;
    }(svgDrawer.SVGDrawer));
    exports.ArcSVGDrawer = ArcSVGDrawer;
    });

    var arcOutlineDrawer = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });


    var ArcOutlineSVGDrawer = /** @class */ (function (_super) {
        tslib_es6.__extends(ArcOutlineSVGDrawer, _super);
        function ArcOutlineSVGDrawer() {
            return _super.call(this, "path", "arc outline") || this;
        }
        ArcOutlineSVGDrawer.prototype._applyDefaultAttributes = function (selection) {
            selection.style("fill", "none");
        };
        return ArcOutlineSVGDrawer;
    }(svgDrawer.SVGDrawer));
    exports.ArcOutlineSVGDrawer = ArcOutlineSVGDrawer;
    });

    var areaDrawer = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });



    var AreaSVGDrawer = /** @class */ (function (_super) {
        tslib_es6.__extends(AreaSVGDrawer, _super);
        function AreaSVGDrawer() {
            return _super.call(this, "path", "area") || this;
        }
        AreaSVGDrawer.prototype._applyDefaultAttributes = function (selection) {
            selection.style("stroke", "none");
        };
        AreaSVGDrawer.prototype.getVisualPrimitiveAtIndex = function (index) {
            // areas are represented by one single element; always get that element
            // regardless of the data index.
            return _super.prototype.getVisualPrimitiveAtIndex.call(this, 0);
        };
        return AreaSVGDrawer;
    }(svgDrawer.SVGDrawer));
    exports.AreaSVGDrawer = AreaSVGDrawer;
    var AREA_FILL_ATTRS = ["opacity", "fill", "fill-opacity"];
    var AREA_STROKE_ATTRS = ["opacity", "stroke", "stroke-width"];
    function makeAreaCanvasDrawStep(d3AreaFactory, d3LineFactory) {
        return function (context, data, projector) {
            var fillAttrs = canvasDrawer.resolveAttributes(projector, AREA_FILL_ATTRS, data[0], 0);
            canvasDrawer.renderArea(context, d3AreaFactory(), data[0], fillAttrs);
            var strokeAttrs = canvasDrawer.resolveAttributes(projector, AREA_STROKE_ATTRS, data[0], 0);
            canvasDrawer.renderLine(context, d3LineFactory(), data[0], strokeAttrs);
        };
    }
    exports.makeAreaCanvasDrawStep = makeAreaCanvasDrawStep;
    });

    var lineDrawer = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });



    var LineSVGDrawer = /** @class */ (function (_super) {
        tslib_es6.__extends(LineSVGDrawer, _super);
        function LineSVGDrawer() {
            return _super.call(this, "path", "line") || this;
        }
        LineSVGDrawer.prototype._applyDefaultAttributes = function (selection) {
            selection.style("fill", "none");
        };
        LineSVGDrawer.prototype.getVisualPrimitiveAtIndex = function (index) {
            return _super.prototype.getVisualPrimitiveAtIndex.call(this, 0);
        };
        return LineSVGDrawer;
    }(svgDrawer.SVGDrawer));
    exports.LineSVGDrawer = LineSVGDrawer;
    var LINE_ATTRIBUTES = [
        "opacity",
        "stroke-opacity",
        "stroke-width",
        "stroke",
        "stroke-dasharray",
    ];
    /**
     * @param d3LineFactory A callback that gives this Line Drawer a d3.Line object which will be
     * used to draw with.
     *
     * TODO put the d3.Line into the attrToAppliedProjector directly
     */
    function makeLineCanvasDrawStep(d3LineFactory) {
        return function (context, data, attrToAppliedProjector) {
            var lineStyle = canvasDrawer.resolveAttributes(attrToAppliedProjector, LINE_ATTRIBUTES, data[0], 0);
            canvasDrawer.renderLine(context, d3LineFactory(), data[0], lineStyle);
        };
    }
    exports.makeLineCanvasDrawStep = makeLineCanvasDrawStep;
    });

    var rectangleDrawer = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });



    var RectangleSVGDrawer = /** @class */ (function (_super) {
        tslib_es6.__extends(RectangleSVGDrawer, _super);
        function RectangleSVGDrawer(_rootClassName) {
            if (_rootClassName === void 0) { _rootClassName = ""; }
            var _this = _super.call(this, "rect", "") || this;
            _this._rootClassName = _rootClassName;
            _this._root.classed(_this._rootClassName, true);
            return _this;
        }
        return RectangleSVGDrawer;
    }(svgDrawer.SVGDrawer));
    exports.RectangleSVGDrawer = RectangleSVGDrawer;
    var RECT_ATTRS = canvasDrawer.ContextStyleAttrs.concat(["x", "y", "width", "height"]);
    exports.RectangleCanvasDrawStep = function (context, data, projector) {
        context.save();
        var dataLen = data.length;
        for (var index = 0; index < dataLen; index++) {
            var datum = data[index];
            if (datum == null) {
                continue;
            }
            var attrs = canvasDrawer.resolveAttributes(projector, RECT_ATTRS, datum, index);
            context.beginPath();
            context.rect(attrs["x"], attrs["y"], attrs["width"], attrs["height"]);
            canvasDrawer.renderPathWithStyle(context, attrs);
        }
        context.restore();
    };
    var RectangleCanvasDrawer = /** @class */ (function (_super) {
        tslib_es6.__extends(RectangleCanvasDrawer, _super);
        function RectangleCanvasDrawer(ctx) {
            return _super.call(this, ctx, exports.RectangleCanvasDrawStep) || this;
        }
        return RectangleCanvasDrawer;
    }(canvasDrawer.CanvasDrawer));
    exports.RectangleCanvasDrawer = RectangleCanvasDrawer;
    });

    var segmentDrawer = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });


    var SegmentSVGDrawer = /** @class */ (function (_super) {
        tslib_es6.__extends(SegmentSVGDrawer, _super);
        function SegmentSVGDrawer() {
            return _super.call(this, "line", "") || this;
        }
        return SegmentSVGDrawer;
    }(svgDrawer.SVGDrawer));
    exports.SegmentSVGDrawer = SegmentSVGDrawer;
    });

    var canvasBuffer = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2017-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });
    var CanvasBuffer = /** @class */ (function () {
        function CanvasBuffer(screenWidth, screenHeight, devicePixelRatio) {
            if (devicePixelRatio === void 0) { devicePixelRatio = window.devicePixelRatio; }
            this.screenWidth = screenWidth;
            this.screenHeight = screenHeight;
            this.devicePixelRatio = devicePixelRatio;
            this.pixelWidth = screenWidth * devicePixelRatio;
            this.pixelHeight = screenHeight * devicePixelRatio;
            this.canvas = document.createElement("canvas");
            this.ctx = this.canvas.getContext("2d");
            CanvasBuffer.sizePixels(this.ctx, screenWidth, screenHeight, devicePixelRatio);
        }
        /**
         * Resizes the canvas' internal pixel buffer to match the devicePixelRatio
         */
        CanvasBuffer.sizePixels = function (ctx, screenWidth, screenHeight, devicePixelRatio) {
            var canvas = ctx.canvas;
            canvas.width = screenWidth * devicePixelRatio;
            canvas.height = screenHeight * devicePixelRatio;
            canvas.style.width = screenWidth + "px";
            canvas.style.height = screenHeight + "px";
            ctx.setTransform(1, 0, 0, 1, 0, 0);
            ctx.scale(devicePixelRatio, devicePixelRatio);
        };
        CanvasBuffer.prototype.blit = function (ctx, x, y) {
            if (x === void 0) { x = 0; }
            if (y === void 0) { y = 0; }
            ctx.drawImage(this.canvas, x, y, this.screenWidth, this.screenHeight);
        };
        CanvasBuffer.prototype.blitCenter = function (ctx, x, y) {
            if (x === void 0) { x = 0; }
            if (y === void 0) { y = 0; }
            this.blit(ctx, Math.floor(x - this.screenWidth / 2), Math.floor(y - this.screenHeight / 2));
        };
        /**
         * Changes the size of the underlying canvas in screen space, respecting the
         * current devicePixelRatio.
         *
         * @param center - optionally enable a translate transformation moving the
         *                 origin to the center of the canvas.
         */
        CanvasBuffer.prototype.resize = function (screenWidth, screenHeight, center) {
            if (center === void 0) { center = false; }
            var devicePixelRatio = this.devicePixelRatio;
            this.screenWidth = screenWidth;
            this.screenHeight = screenHeight;
            this.pixelWidth = screenWidth * devicePixelRatio;
            this.pixelHeight = screenHeight * devicePixelRatio;
            CanvasBuffer.sizePixels(this.ctx, screenWidth, screenHeight, devicePixelRatio);
            if (center) {
                this.ctx.translate(screenWidth / 2, screenWidth / 2);
            }
            return this;
        };
        /**
         * Temporarily resets the current context transformation and fills the
         * entire canvas with the provided color. If no color is provided, the
         * canvas is cleared instead.
         */
        CanvasBuffer.prototype.clear = function (color) {
            var _a = this, pixelWidth = _a.pixelWidth, pixelHeight = _a.pixelHeight, ctx = _a.ctx;
            ctx.save();
            ctx.setTransform(1, 0, 0, 1, 0, 0);
            if (color == null) {
                ctx.clearRect(0, 0, pixelWidth, pixelHeight);
            }
            else {
                ctx.fillStyle = color;
                ctx.fillRect(0, 0, pixelWidth, pixelHeight);
            }
            ctx.restore();
            return this;
        };
        CanvasBuffer.prototype.getImageData = function () {
            return this.ctx.getImageData(0, 0, this.pixelWidth, this.pixelHeight);
        };
        return CanvasBuffer;
    }());
    exports.CanvasBuffer = CanvasBuffer;
    });

    var symbolDrawer = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });




    var SymbolSVGDrawer = /** @class */ (function (_super) {
        tslib_es6.__extends(SymbolSVGDrawer, _super);
        function SymbolSVGDrawer() {
            return _super.call(this, "path", "symbol") || this;
        }
        return SymbolSVGDrawer;
    }(svgDrawer.SVGDrawer));
    exports.SymbolSVGDrawer = SymbolSVGDrawer;
    var SYMBOL_ATTRS = canvasDrawer.ContextStyleAttrs.concat(["x", "y"]);
    function makeSymbolCanvasDrawStep(dataset, symbolProjector, sizeProjector, stepBuffer) {
        var _this = this;
        return function (context, data, attrToAppliedProjector) {
            var _a = context.canvas, clientWidth = _a.clientWidth, clientHeight = _a.clientHeight;
            var buffer = (stepBuffer === undefined) ? new canvasBuffer.CanvasBuffer(0, 0) : stepBuffer;
            var symbolAccessor = symbolProjector();
            var sizeAccessor = sizeProjector();
            var prevAttrs = null;
            var prevSymbolGenerator = null;
            var prevSymbolSize = null;
            for (var index = 0; index < data.length; index++) {
                var datum = data[index];
                if (datum == null) {
                    continue;
                }
                // check symbol is in viewport
                var attrs = canvasDrawer.resolveAttributes(attrToAppliedProjector, SYMBOL_ATTRS, datum, index);
                var symbolSize = sizeAccessor(datum, index, dataset);
                if (!squareOverlapsBounds(clientWidth, clientHeight, attrs["x"], attrs["y"], symbolSize)) {
                    continue;
                }
                // check attributes and symbol type
                var attrsSame = isAttributeValuesEqual(prevAttrs, attrs, canvasDrawer.ContextStyleAttrs);
                var symbolGenerator = symbolAccessor(datum, index, _this._dataset);
                if (attrsSame && prevSymbolSize == symbolSize && prevSymbolGenerator == symbolGenerator) ;
                else {
                    // make room for bigger symbol if needed
                    var strokeWidth = canvasDrawer.getStrokeWidth(attrs);
                    // +1 to account for subpixel aliasing
                    var wantedBufferSize = symbolSize + strokeWidth + 1;
                    if (wantedBufferSize > buffer.screenWidth || wantedBufferSize > buffer.screenHeight) {
                        buffer.resize(wantedBufferSize, wantedBufferSize, true);
                    }
                    // draw actual symbol into buffer
                    buffer.clear();
                    var bufferCtx = buffer.ctx;
                    bufferCtx.beginPath();
                    symbolGenerator(symbolSize).context(bufferCtx)(null);
                    bufferCtx.closePath();
                    canvasDrawer.renderPathWithStyle(bufferCtx, attrs);
                    // save the values that are in the buffer
                    prevSymbolGenerator = symbolGenerator;
                    prevSymbolSize = symbolSize;
                    prevAttrs = attrs;
                }
                // blit the buffer to the canvas
                buffer.blitCenter(context, attrs["x"], attrs["y"]);
            }
        };
    }
    exports.makeSymbolCanvasDrawStep = makeSymbolCanvasDrawStep;
    function squareOverlapsBounds(width, height, x, y, size) {
        return (x + size >= 0 && x - size <= width &&
            y + size >= 0 && y - size <= height);
    }
    function isAttributeValuesEqual(prevAttrs, attrs, attrKeys) {
        if (prevAttrs == null) {
            return false;
        }
        for (var i = 0; i < attrKeys.length; i++) {
            var attrKey = attrKeys[i];
            if (prevAttrs[attrKey] != attrs[attrKey]) {
                return false;
            }
        }
        return true;
    }
    });

    var drawers = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });

    tslib_es6.__exportStar(arcDrawer, exports);
    tslib_es6.__exportStar(arcOutlineDrawer, exports);
    tslib_es6.__exportStar(areaDrawer, exports);
    tslib_es6.__exportStar(canvasDrawer, exports);
    tslib_es6.__exportStar(drawer, exports);
    tslib_es6.__exportStar(lineDrawer, exports);
    tslib_es6.__exportStar(rectangleDrawer, exports);
    tslib_es6.__exportStar(segmentDrawer, exports);
    tslib_es6.__exportStar(svgDrawer, exports);
    tslib_es6.__exportStar(symbolDrawer, exports);
    });

    var xyPlot = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });





    var XYPlot = /** @class */ (function (_super) {
        tslib_es6.__extends(XYPlot, _super);
        /**
         * An XYPlot is a Plot that displays data along two primary directions, X and Y.
         *
         * @constructor
         * @param {Scale} xScale The x scale to use.
         * @param {Scale} yScale The y scale to use.
         */
        function XYPlot() {
            var _this = _super.call(this) || this;
            _this._autoAdjustXScaleDomain = false;
            _this._autoAdjustYScaleDomain = false;
            _this._deferredRendering = false;
            _this._applyDeferredRenderingTransform = function (tx, ty, sx, sy) {
                if (!_this._isAnchored) {
                    return;
                }
                if (_this._renderArea != null) {
                    _this._renderArea.attr("transform", "translate(" + tx + ", " + ty + ") scale(" + sx + ", " + sy + ")");
                }
                if (_this._canvas != null) {
                    _this._canvas.style("transform", "translate(" + tx + "px, " + ty + "px) scale(" + sx + ", " + sy + ")");
                }
            };
            _this.addClass("xy-plot");
            _this._adjustYDomainOnChangeFromXCallback = function (scale) { return _this._adjustYDomainOnChangeFromX(); };
            _this._adjustXDomainOnChangeFromYCallback = function (scale) { return _this._adjustXDomainOnChangeFromY(); };
            _this._renderCallback = function () {
                if (_this.deferredRendering()) {
                    var scaleX = _this.x() && _this.x().scale;
                    var scaleY = _this.y() && _this.y().scale;
                    _this._deferredRenderer.updateDomains(scaleX, scaleY);
                }
                else {
                    _this.render();
                }
            };
            _this._deferredRenderer = new deferredRenderer.DeferredRenderer(function () { return _this.render(); }, _this._applyDeferredRenderingTransform);
            return _this;
        }
        XYPlot.prototype.render = function () {
            if (this.deferredRendering()) {
                this._deferredRenderer.resetTransforms();
            }
            return _super.prototype.render.call(this);
        };
        XYPlot.prototype.deferredRendering = function (deferredRendering) {
            if (deferredRendering == null) {
                return this._deferredRendering;
            }
            if (deferredRendering) {
                var scaleX = this.x() && this.x().scale;
                var scaleY = this.y() && this.y().scale;
                this._deferredRenderer.setDomains(scaleX, scaleY);
            }
            this._deferredRendering = deferredRendering;
            return this;
        };
        XYPlot.prototype.x = function (x, xScale, postScale) {
            if (x == null) {
                return this._propertyBindings.get(XYPlot._X_KEY);
            }
            this._bindProperty(XYPlot._X_KEY, x, xScale, postScale);
            var width = this.width();
            if (xScale != null && width != null) {
                xScale.range([0, width]);
            }
            if (this._autoAdjustYScaleDomain) {
                this._updateYExtentsAndAutodomain();
            }
            this.render();
            return this;
        };
        XYPlot.prototype.y = function (y, yScale, postScale) {
            if (y == null) {
                return this._propertyBindings.get(XYPlot._Y_KEY);
            }
            this._bindProperty(XYPlot._Y_KEY, y, yScale, postScale);
            var height = this.height();
            if (yScale != null && height != null) {
                if (yScale instanceof scales.Category) {
                    yScale.range([0, height]);
                }
                else {
                    yScale.range([height, 0]);
                }
            }
            if (this._autoAdjustXScaleDomain) {
                this._updateXExtentsAndAutodomain();
            }
            this.render();
            return this;
        };
        XYPlot.prototype._filterForProperty = function (property) {
            if (property === "x" && this._autoAdjustXScaleDomain) {
                return this._makeFilterByProperty("y");
            }
            else if ((property === "y" || property === "y0") && this._autoAdjustYScaleDomain) {
                return this._makeFilterByProperty("x");
            }
            return null;
        };
        XYPlot.prototype._makeFilterByProperty = function (property) {
            var binding = this._propertyBindings.get(property);
            if (binding != null) {
                var accessor_1 = binding.accessor;
                var scale_1 = binding.scale;
                if (scale_1 != null) {
                    return function (datum, index, dataset) {
                        var range = scale_1.range();
                        return utils$1.Math.inRange(scale_1.scale(accessor_1(datum, index, dataset)), range[0], range[1]);
                    };
                }
            }
            return null;
        };
        XYPlot.prototype._uninstallScaleForKey = function (scale, key) {
            _super.prototype._uninstallScaleForKey.call(this, scale, key);
            var adjustCallback = key === XYPlot._X_KEY ? this._adjustYDomainOnChangeFromXCallback
                : this._adjustXDomainOnChangeFromYCallback;
            scale.offUpdate(adjustCallback);
        };
        XYPlot.prototype._installScaleForKey = function (scale, key) {
            _super.prototype._installScaleForKey.call(this, scale, key);
            var adjustCallback = key === XYPlot._X_KEY ? this._adjustYDomainOnChangeFromXCallback
                : this._adjustXDomainOnChangeFromYCallback;
            scale.onUpdate(adjustCallback);
        };
        XYPlot.prototype.destroy = function () {
            _super.prototype.destroy.call(this);
            if (this.x().scale) {
                this.x().scale.offUpdate(this._adjustYDomainOnChangeFromXCallback);
            }
            if (this.y().scale) {
                this.y().scale.offUpdate(this._adjustXDomainOnChangeFromYCallback);
            }
            return this;
        };
        XYPlot.prototype.autorangeMode = function (autorangeMode) {
            if (autorangeMode == null) {
                if (this._autoAdjustXScaleDomain) {
                    return "x";
                }
                if (this._autoAdjustYScaleDomain) {
                    return "y";
                }
                return "none";
            }
            switch (autorangeMode) {
                case "x":
                    this._autoAdjustXScaleDomain = true;
                    this._autoAdjustYScaleDomain = false;
                    this._adjustXDomainOnChangeFromY();
                    break;
                case "y":
                    this._autoAdjustXScaleDomain = false;
                    this._autoAdjustYScaleDomain = true;
                    this._adjustYDomainOnChangeFromX();
                    break;
                case "none":
                    this._autoAdjustXScaleDomain = false;
                    this._autoAdjustYScaleDomain = false;
                    break;
                default:
                    throw new Error("Invalid scale name '" + autorangeMode + "', must be 'x', 'y' or 'none'");
            }
            return this;
        };
        XYPlot.prototype.computeLayout = function (origin, availableWidth, availableHeight) {
            _super.prototype.computeLayout.call(this, origin, availableWidth, availableHeight);
            var xBinding = this.x();
            var xScale = xBinding && xBinding.scale;
            if (xScale != null) {
                xScale.range([0, this.width()]);
            }
            var yBinding = this.y();
            var yScale = yBinding && yBinding.scale;
            if (yScale != null) {
                if (yScale instanceof scales.Category) {
                    yScale.range([0, this.height()]);
                }
                else {
                    yScale.range([this.height(), 0]);
                }
            }
            return this;
        };
        XYPlot.prototype._updateXExtentsAndAutodomain = function () {
            var xScale = this.x().scale;
            if (xScale != null) {
                xScale.autoDomain();
            }
        };
        XYPlot.prototype._updateYExtentsAndAutodomain = function () {
            var yScale = this.y().scale;
            if (yScale != null) {
                yScale.autoDomain();
            }
        };
        /**
         * Adjusts the domains of both X and Y scales to show all data.
         * This call does not override the autorange() behavior.
         *
         * @returns {XYPlot} The calling XYPlot.
         */
        XYPlot.prototype.showAllData = function () {
            this._updateXExtentsAndAutodomain();
            this._updateYExtentsAndAutodomain();
            return this;
        };
        XYPlot.prototype._adjustYDomainOnChangeFromX = function () {
            if (!this._projectorsReady()) {
                return;
            }
            if (this._autoAdjustYScaleDomain) {
                this._updateYExtentsAndAutodomain();
            }
        };
        XYPlot.prototype._adjustXDomainOnChangeFromY = function () {
            if (!this._projectorsReady()) {
                return;
            }
            if (this._autoAdjustXScaleDomain) {
                this._updateXExtentsAndAutodomain();
            }
        };
        XYPlot.prototype._projectorsReady = function () {
            var xBinding = this.x();
            var yBinding = this.y();
            return xBinding != null &&
                xBinding.accessor != null &&
                yBinding != null &&
                yBinding.accessor != null;
        };
        XYPlot.prototype._pixelPoint = function (datum, index, dataset) {
            var xProjector = plot.Plot._scaledAccessor(this.x());
            var yProjector = plot.Plot._scaledAccessor(this.y());
            return { x: xProjector(datum, index, dataset), y: yProjector(datum, index, dataset) };
        };
        XYPlot.prototype._getDataToDraw = function () {
            var _this = this;
            var dataToDraw = _super.prototype._getDataToDraw.call(this);
            var definedAttr = this.attr("defined");
            var definedFunction = function (d, i, dataset) {
                var positionX = plot.Plot._scaledAccessor(_this.x())(d, i, dataset);
                var positionY = plot.Plot._scaledAccessor(_this.y())(d, i, dataset);
                if (definedAttr && definedAttr.accessor(d, i, dataset) === false) {
                    return false;
                }
                return utils$1.Math.isValidNumber(positionX) &&
                    utils$1.Math.isValidNumber(positionY);
            };
            this.datasets().forEach(function (dataset) {
                dataToDraw.set(dataset, dataToDraw.get(dataset).filter(function (d, i) { return definedFunction(d, i, dataset); }));
            });
            return dataToDraw;
        };
        XYPlot._X_KEY = "x";
        XYPlot._Y_KEY = "y";
        return XYPlot;
    }(plot.Plot));
    exports.XYPlot = XYPlot;
    });

    var linePlot = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });













    var CURVE_NAME_MAPPING = {
        linear: d3.curveLinear,
        linearClosed: d3.curveLinearClosed,
        step: d3.curveStep,
        stepBefore: d3.curveStepBefore,
        stepAfter: d3.curveStepAfter,
        basis: d3.curveBasis,
        basisOpen: d3.curveBasisOpen,
        basisClosed: d3.curveBasisClosed,
        bundle: d3.curveBundle,
        cardinal: d3.curveCardinal,
        cardinalOpen: d3.curveCardinalOpen,
        cardinalClosed: d3.curveCardinalClosed,
        monotone: d3.curveMonotoneX,
    };
    /**
     * Known curve types that line and area plot's .curve() methods understand
     */
    exports.CurveName = makeEnum_1.makeEnum([
        "linear",
        "linearClosed",
        "step",
        "stepBefore",
        "stepAfter",
        "basis",
        "basisOpen",
        "basisClosed",
        "bundle",
        "cardinal",
        "cardinalOpen",
        "cardinalClosed",
        "monotone",
    ]);
    var Line = /** @class */ (function (_super) {
        tslib_es6.__extends(Line, _super);
        /**
         * A Line Plot draws line segments starting from the first data point to the next.
         *
         * @constructor
         */
        function Line() {
            var _this = _super.call(this) || this;
            _this._curve = "linear";
            _this._autorangeSmooth = false;
            _this._croppedRenderingEnabled = true;
            _this._collapseDenseVerticalLinesEnabled = false;
            _this._downsamplingEnabled = false;
            _this.addClass("line-plot");
            var animator = new animators.Easing();
            animator.stepDuration(plot.Plot._ANIMATION_MAX_DURATION);
            animator.easingMode("expInOut");
            animator.maxTotalDuration(plot.Plot._ANIMATION_MAX_DURATION);
            _this.animator(plots.Animator.MAIN, animator);
            _this.attr("stroke", new scales.Color().range()[0]);
            _this.attr("stroke-width", "2px");
            return _this;
        }
        Line.prototype.x = function (x, xScale, postScale) {
            if (x == null) {
                return _super.prototype.x.call(this);
            }
            else {
                _super.prototype.x.call(this, x, xScale, postScale);
                this._setScaleSnapping();
                return this;
            }
        };
        Line.prototype.y = function (y, yScale, postScale) {
            if (y == null) {
                return _super.prototype.y.call(this);
            }
            else {
                _super.prototype.y.call(this, y, yScale, postScale);
                this._setScaleSnapping();
                return this;
            }
        };
        Line.prototype.autorangeMode = function (autorangeMode) {
            if (autorangeMode == null) {
                return _super.prototype.autorangeMode.call(this);
            }
            _super.prototype.autorangeMode.call(this, autorangeMode);
            this._setScaleSnapping();
            return this;
        };
        Line.prototype.autorangeSmooth = function (autorangeSmooth) {
            if (autorangeSmooth == null) {
                return this._autorangeSmooth;
            }
            this._autorangeSmooth = autorangeSmooth;
            this._setScaleSnapping();
            return this;
        };
        Line.prototype._setScaleSnapping = function () {
            if (this.autorangeMode() === "x" && this.x() && this.x().scale && this.x().scale instanceof quantitativeScale.QuantitativeScale) {
                this.x().scale.snappingDomainEnabled(!this.autorangeSmooth());
            }
            if (this.autorangeMode() === "y" && this.y() && this.y().scale && this.y().scale instanceof quantitativeScale.QuantitativeScale) {
                this.y().scale.snappingDomainEnabled(!this.autorangeSmooth());
            }
        };
        Line.prototype.curve = function (curve) {
            if (curve == null) {
                return this._curve;
            }
            this._curve = curve;
            this.render();
            return this;
        };
        Line.prototype.downsamplingEnabled = function (downsampling) {
            if (downsampling == null) {
                return this._downsamplingEnabled;
            }
            this._downsamplingEnabled = downsampling;
            return this;
        };
        Line.prototype.croppedRenderingEnabled = function (croppedRendering) {
            if (croppedRendering == null) {
                return this._croppedRenderingEnabled;
            }
            this._croppedRenderingEnabled = croppedRendering;
            this.render();
            return this;
        };
        Line.prototype.collapseDenseLinesEnabled = function (collapseDenseLines) {
            if (collapseDenseLines == null) {
                return this._collapseDenseVerticalLinesEnabled;
            }
            this._collapseDenseVerticalLinesEnabled = collapseDenseLines;
            this.render();
            return this;
        };
        Line.prototype._createDrawer = function (dataset) {
            var _this = this;
            return new drawer.ProxyDrawer(function () { return new lineDrawer.LineSVGDrawer(); }, function (ctx) { return new drawers.CanvasDrawer(ctx, lineDrawer.makeLineCanvasDrawStep(function () { return _this._d3LineFactory(dataset); })); });
        };
        Line.prototype.getExtentsForProperty = function (property) {
            var extents = _super.prototype.getExtentsForProperty.call(this, property);
            if (!this._autorangeSmooth) {
                return extents;
            }
            if (this.autorangeMode() !== property) {
                return extents;
            }
            if (this.autorangeMode() !== "x" && this.autorangeMode() !== "y") {
                return extents;
            }
            var edgeIntersectionPoints = this._getEdgeIntersectionPoints();
            var includedValues;
            if (this.autorangeMode() === "y") {
                includedValues = edgeIntersectionPoints.left.concat(edgeIntersectionPoints.right).map(function (point) { return point.y; });
            }
            else {
                includedValues = edgeIntersectionPoints.top.concat(edgeIntersectionPoints.bottom).map(function (point) { return point.x; });
            }
            return extents.map(function (extent) { return d3.extent(d3.merge([extent, includedValues])); });
        };
        Line.prototype._getEdgeIntersectionPoints = function () {
            var _this = this;
            if (!(this.y().scale instanceof quantitativeScale.QuantitativeScale && this.x().scale instanceof quantitativeScale.QuantitativeScale)) {
                return {
                    left: [],
                    right: [],
                    top: [],
                    bottom: [],
                };
            }
            var yScale = this.y().scale;
            var xScale = this.x().scale;
            var intersectionPoints = {
                left: [],
                right: [],
                top: [],
                bottom: [],
            };
            var leftX = xScale.scale(xScale.domain()[0]);
            var rightX = xScale.scale(xScale.domain()[1]);
            var bottomY = yScale.scale(yScale.domain()[0]);
            var topY = yScale.scale(yScale.domain()[1]);
            this.datasets().forEach(function (dataset) {
                var data = dataset.data();
                var x1, x2, y1, y2;
                var prevX, prevY, currX, currY;
                for (var i = 1; i < data.length; i++) {
                    prevX = currX || xScale.scale(_this.x().accessor(data[i - 1], i - 1, dataset));
                    prevY = currY || yScale.scale(_this.y().accessor(data[i - 1], i - 1, dataset));
                    currX = xScale.scale(_this.x().accessor(data[i], i, dataset));
                    currY = yScale.scale(_this.y().accessor(data[i], i, dataset));
                    // If values crossed left edge
                    if ((prevX < leftX) === (leftX <= currX)) {
                        x1 = leftX - prevX;
                        x2 = currX - prevX;
                        y2 = currY - prevY;
                        y1 = x1 * y2 / x2;
                        intersectionPoints.left.push({
                            x: leftX,
                            y: yScale.invert(prevY + y1),
                        });
                    }
                    // If values crossed right edge
                    if ((prevX < rightX) === (rightX <= currX)) {
                        x1 = rightX - prevX;
                        x2 = currX - prevX;
                        y2 = currY - prevY;
                        y1 = x1 * y2 / x2;
                        intersectionPoints.right.push({
                            x: rightX,
                            y: yScale.invert(prevY + y1),
                        });
                    }
                    // If values crossed upper edge
                    if ((prevY < topY) === (topY <= currY)) {
                        x2 = currX - prevX;
                        y1 = topY - prevY;
                        y2 = currY - prevY;
                        x1 = y1 * x2 / y2;
                        intersectionPoints.top.push({
                            x: xScale.invert(prevX + x1),
                            y: topY,
                        });
                    }
                    // If values crossed lower edge
                    if ((prevY < bottomY) === (bottomY <= currY)) {
                        x2 = currX - prevX;
                        y1 = bottomY - prevY;
                        y2 = currY - prevY;
                        x1 = y1 * x2 / y2;
                        intersectionPoints.bottom.push({
                            x: xScale.invert(prevX + x1),
                            y: bottomY,
                        });
                    }
                }
            });
            return intersectionPoints;
        };
        Line.prototype._getResetYFunction = function () {
            // gets the y-value generator for the animation start point
            var yDomain = this.y().scale.domain();
            var domainMax = Math.max(yDomain[0], yDomain[1]);
            var domainMin = Math.min(yDomain[0], yDomain[1]);
            // start from zero, or the closest domain value to zero
            // avoids lines zooming on from offscreen.
            var startValue = (domainMax < 0 && domainMax) || (domainMin > 0 && domainMin) || 0;
            var scaledStartValue = this.y().scale.scale(startValue);
            return function (d, i, dataset) { return scaledStartValue; };
        };
        Line.prototype._generateDrawSteps = function () {
            var drawSteps = [];
            if (this._animateOnNextRender()) {
                var attrToProjector = this._getAttrToProjector();
                attrToProjector["d"] = this._constructLineProjector(plot.Plot._scaledAccessor(this.x()), this._getResetYFunction());
                drawSteps.push({ attrToProjector: attrToProjector, animator: this._getAnimator(plots.Animator.RESET) });
            }
            drawSteps.push({
                attrToProjector: this._getAttrToProjector(),
                animator: this._getAnimator(plots.Animator.MAIN),
            });
            return drawSteps;
        };
        Line.prototype._generateAttrToProjector = function () {
            var attrToProjector = _super.prototype._generateAttrToProjector.call(this);
            Object.keys(attrToProjector).forEach(function (attribute) {
                if (attribute === "d") {
                    return;
                }
                var projector = attrToProjector[attribute];
                attrToProjector[attribute] = function (data, i, dataset) {
                    return data.length > 0 ? projector(data[0], i, dataset) : null;
                };
            });
            return attrToProjector;
        };
        Line.prototype.entitiesAt = function (point) {
            var entity = this.entityNearestByXThenY(point);
            if (entity != null) {
                return [entity];
            }
            else {
                return [];
            }
        };
        /**
         * Returns the PlotEntity nearest to the query point by X then by Y, or undefined if no PlotEntity can be found.
         *
         * @param {Point} queryPoint
         * @returns {PlotEntity} The nearest PlotEntity, or undefined if no PlotEntity can be found.
         */
        Line.prototype.entityNearestByXThenY = function (queryPoint) {
            var minXDist = Infinity;
            var minYDist = Infinity;
            var closest;
            var chartBounds = this.bounds();
            var entities = this.entities();
            var entityLen = entities.length;
            for (var i = 0; i < entityLen; i++) {
                var entity = entities[i];
                if (!utils$1.Math.within(entity.position, chartBounds)) {
                    continue;
                }
                var xDist = Math.abs(queryPoint.x - entity.position.x);
                var yDist = Math.abs(queryPoint.y - entity.position.y);
                if (xDist < minXDist || xDist === minXDist && yDist < minYDist) {
                    closest = entity;
                    minXDist = xDist;
                    minYDist = yDist;
                }
            }
            return closest;
        };
        Line.prototype._propertyProjectors = function () {
            var propertyToProjectors = _super.prototype._propertyProjectors.call(this);
            propertyToProjectors["d"] = this._constructLineProjector(plot.Plot._scaledAccessor(this.x()), plot.Plot._scaledAccessor(this.y()));
            return propertyToProjectors;
        };
        Line.prototype._constructLineProjector = function (xProjector, yProjector) {
            var _this = this;
            return function (datum, index, dataset) {
                return _this._d3LineFactory(dataset, xProjector, yProjector)(datum);
            };
        };
        /**
         * Return a d3.Line whose .x, .y, and .defined accessors are hooked up to the xProjector and yProjector
         * after they've been fed the dataset, and whose curve is configured to this plot's curve.
         * @param dataset
         * @param xProjector
         * @param yProjector
         * @returns {Line<[number,number]>}
         * @private
         */
        Line.prototype._d3LineFactory = function (dataset, xProjector, yProjector) {
            if (xProjector === void 0) { xProjector = plot.Plot._scaledAccessor(this.x()); }
            if (yProjector === void 0) { yProjector = plot.Plot._scaledAccessor(this.y()); }
            var definedProjector = function (d, i, dataset) {
                var positionX = xProjector(d, i, dataset);
                var positionY = yProjector(d, i, dataset);
                return utils$1.Math.isValidNumber(positionX) && utils$1.Math.isValidNumber(positionY);
            };
            return d3.line()
                .x(function (innerDatum, innerIndex) { return xProjector(innerDatum, innerIndex, dataset); })
                .y(function (innerDatum, innerIndex) { return yProjector(innerDatum, innerIndex, dataset); })
                .curve(this._getCurveFactory())
                .defined(function (innerDatum, innerIndex) { return definedProjector(innerDatum, innerIndex, dataset); });
        };
        Line.prototype._getCurveFactory = function () {
            var curve = this.curve();
            if (typeof curve === "string") {
                var maybeCurveFunction = CURVE_NAME_MAPPING[curve];
                if (maybeCurveFunction == null) {
                    // oops; name is wrong - default to linear instead
                    return CURVE_NAME_MAPPING["linear"];
                }
                else {
                    return maybeCurveFunction;
                }
            }
            else {
                return curve;
            }
        };
        /**
         * Line plots represent each dataset with a single <path> element, so we wrap the dataset data in a single element array.
         * @returns {Map<Dataset, any[]>}
         * @private
         */
        Line.prototype._getDataToDraw = function () {
            var _this = this;
            var dataToDraw = new utils$1.Map();
            this.datasets().forEach(function (dataset) {
                var data = dataset.data();
                if (!_this._croppedRenderingEnabled && !_this._downsamplingEnabled) {
                    dataToDraw.set(dataset, [data]);
                    return;
                }
                var filteredDataIndices = [];
                var dataLen = data.length;
                for (var i = 0; i < dataLen; i++) {
                    filteredDataIndices[i] = i;
                }
                if (_this._croppedRenderingEnabled) {
                    filteredDataIndices = _this._filterCroppedRendering(dataset, filteredDataIndices);
                }
                if (_this._downsamplingEnabled) {
                    filteredDataIndices = _this._filterDownsampling(dataset, filteredDataIndices);
                }
                if (_this._collapseDenseVerticalLinesEnabled) {
                    filteredDataIndices = _this._filterDenseLines(dataset, filteredDataIndices);
                }
                var filteredData = [];
                var filteredIndicesLen = filteredDataIndices.length;
                for (var i = 0; i < filteredIndicesLen; i++) {
                    var index = filteredDataIndices[i];
                    filteredData[i] = data[index];
                }
                dataToDraw.set(dataset, [filteredData]);
            });
            return dataToDraw;
        };
        Line.prototype._filterCroppedRendering = function (dataset, indices) {
            var _this = this;
            var xProjector = plot.Plot._scaledAccessor(this.x());
            var yProjector = plot.Plot._scaledAccessor(this.y());
            var data = dataset.data();
            var filteredDataIndices = [];
            var pointInViewport = function (x, y) {
                return utils$1.Math.inRange(x, 0, _this.width()) &&
                    utils$1.Math.inRange(y, 0, _this.height());
            };
            for (var i = 0; i < indices.length; i++) {
                var currXPoint = xProjector(data[indices[i]], indices[i], dataset);
                var currYPoint = yProjector(data[indices[i]], indices[i], dataset);
                var shouldShow = pointInViewport(currXPoint, currYPoint);
                if (!shouldShow && indices[i - 1] != null && data[indices[i - 1]] != null) {
                    var prevXPoint = xProjector(data[indices[i - 1]], indices[i - 1], dataset);
                    var prevYPoint = yProjector(data[indices[i - 1]], indices[i - 1], dataset);
                    shouldShow = shouldShow || pointInViewport(prevXPoint, prevYPoint);
                }
                if (!shouldShow && indices[i + 1] != null && data[indices[i + 1]] != null) {
                    var nextXPoint = xProjector(data[indices[i + 1]], indices[i + 1], dataset);
                    var nextYPoint = yProjector(data[indices[i + 1]], indices[i + 1], dataset);
                    shouldShow = shouldShow || pointInViewport(nextXPoint, nextYPoint);
                }
                if (shouldShow) {
                    filteredDataIndices.push(indices[i]);
                }
            }
            return filteredDataIndices;
        };
        Line.prototype._filterDownsampling = function (dataset, indices) {
            if (indices.length === 0) {
                return [];
            }
            var data = dataset.data();
            var scaledXAccessor = plot.Plot._scaledAccessor(this.x());
            var scaledYAccessor = plot.Plot._scaledAccessor(this.y());
            var filteredIndices = [indices[0]];
            var indexOnCurrentSlope = function (i, currentSlope) {
                var p1x = scaledXAccessor(data[indices[i]], indices[i], dataset);
                var p1y = scaledYAccessor(data[indices[i]], indices[i], dataset);
                var p2x = scaledXAccessor(data[indices[i + 1]], indices[i + 1], dataset);
                var p2y = scaledYAccessor(data[indices[i + 1]], indices[i + 1], dataset);
                if (currentSlope === Infinity) {
                    return Math.floor(p1x) === Math.floor(p2x);
                }
                else {
                    var expectedP2y = p1y + (p2x - p1x) * currentSlope;
                    return Math.floor(p2y) === Math.floor(expectedP2y);
                }
            };
            for (var i = 0; i < indices.length - 1;) {
                var indexFirst = indices[i];
                var p1x = scaledXAccessor(data[indices[i]], indices[i], dataset);
                var p1y = scaledYAccessor(data[indices[i]], indices[i], dataset);
                var p2x = scaledXAccessor(data[indices[i + 1]], indices[i + 1], dataset);
                var p2y = scaledYAccessor(data[indices[i + 1]], indices[i + 1], dataset);
                var currentSlope = (Math.floor(p1x) === Math.floor(p2x)) ? Infinity : (p2y - p1y) / (p2x - p1x);
                var indexMin = indices[i];
                var minScaledValue = (currentSlope === Infinity) ? p1y : p1x;
                var indexMax = indexMin;
                var maxScaledValue = minScaledValue;
                var firstIndexOnCurrentSlope = true;
                while (i < indices.length - 1 && (firstIndexOnCurrentSlope || indexOnCurrentSlope(i, currentSlope))) {
                    i++;
                    firstIndexOnCurrentSlope = false;
                    var currScaledValue = currentSlope === Infinity ? scaledYAccessor(data[indices[i]], indices[i], dataset) :
                        scaledXAccessor(data[indices[i]], indices[i], dataset);
                    if (currScaledValue > maxScaledValue) {
                        maxScaledValue = currScaledValue;
                        indexMax = indices[i];
                    }
                    if (currScaledValue < minScaledValue) {
                        minScaledValue = currScaledValue;
                        indexMin = indices[i];
                    }
                }
                var indexLast = indices[i];
                if (indexMin !== indexFirst) {
                    filteredIndices.push(indexMin);
                }
                if (indexMax !== indexMin && indexMax !== indexFirst) {
                    filteredIndices.push(indexMax);
                }
                if (indexLast !== indexFirst && indexLast !== indexMin && indexLast !== indexMax) {
                    filteredIndices.push(indexLast);
                }
            }
            return filteredIndices;
        };
        /**
         * Collapse line geometry
         *
         * Assuming that there are many points that are drawn on the same coordinate,
         * we can save a lot of render time by just drawing one line from the min to
         * max y-coordinate of all those points.
         */
        Line.prototype._filterDenseLines = function (dataset, indices) {
            if (indices.length === 0) {
                return [];
            }
            var data = dataset.data();
            var scaledXAccessor = plot.Plot._scaledAccessor(this.x());
            var scaledYAccessor = plot.Plot._scaledAccessor(this.y());
            var xFn = function (i) { return scaledXAccessor(data[i], i, dataset); };
            var yFn = function (i) { return scaledYAccessor(data[i], i, dataset); };
            // TODO determine if we should collapse x or y or not collapse at all.
            // For now, we assume line charts are always a function of x
            return this._bucketByX(dataset, indices, xFn, yFn);
        };
        /**
         * Iterates over the line points collapsing points that fall on the same
         * floored x coordinate.
         *
         * Once all the points with the same x coordinate are detected, we draw a
         * single line from the min to max y coorindate.
         *
         * The "entrance" and "exit" lines to/from this collapsed vertical line are
         * also drawn. This allows lines with no collapsed segments to render
         * correctly.
         */
        Line.prototype._bucketByX = function (dataset, indices, xFn, yFn) {
            var filteredIndices = [];
            var data = dataset.data();
            var bucket = null;
            var indicesLength = indices.length;
            for (var ii = 0; ii <= indicesLength; ++ii) {
                var i = indices[ii];
                if (data[i] == null) {
                    continue;
                }
                var x = Math.floor(xFn(i));
                var y = yFn(i);
                if (bucket == null) {
                    bucket = new utils$1.Bucket(i, x, y);
                }
                else if (bucket.isInBucket(x)) {
                    bucket.addToBucket(y, i);
                }
                else {
                    filteredIndices.push.apply(filteredIndices, bucket.getUniqueIndices());
                    bucket = new utils$1.Bucket(i, x, y);
                }
            }
            if (bucket != null) {
                filteredIndices.push.apply(filteredIndices, bucket.getUniqueIndices());
            }
            return filteredIndices;
        };
        return Line;
    }(xyPlot.XYPlot));
    exports.Line = Line;
    });

    var areaPlot = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });











    var Area = /** @class */ (function (_super) {
        tslib_es6.__extends(Area, _super);
        /**
         * An Area Plot draws a filled region (area) between Y and Y0.
         *
         * @constructor
         */
        function Area() {
            var _this = _super.call(this) || this;
            _this.addClass("area-plot");
            _this.y0(0); // default
            _this.attr("fill-opacity", 0.25);
            _this.attr("fill", new scales.Color().range()[0]);
            _this._lineDrawers = new utils$1.Map();
            return _this;
        }
        Area.prototype.y = function (y, yScale) {
            if (y == null) {
                return _super.prototype.y.call(this);
            }
            if (yScale == null) {
                _super.prototype.y.call(this, y);
            }
            else {
                _super.prototype.y.call(this, y, yScale);
            }
            if (yScale != null) {
                var y0 = this.y0().accessor;
                if (y0 != null) {
                    this._bindProperty(Area._Y0_KEY, y0, yScale);
                }
                this._updateYScale();
            }
            return this;
        };
        Area.prototype.y0 = function (y0) {
            if (y0 == null) {
                return this._propertyBindings.get(Area._Y0_KEY);
            }
            var yBinding = this.y();
            var yScale = yBinding && yBinding.scale;
            this._bindProperty(Area._Y0_KEY, y0, yScale);
            this._updateYScale();
            this.render();
            return this;
        };
        Area.prototype._onDatasetUpdate = function () {
            _super.prototype._onDatasetUpdate.call(this);
            this._updateYScale();
        };
        Area.prototype._addDataset = function (dataset) {
            var _this = this;
            this._lineDrawers.set(dataset, new drawers.ProxyDrawer(function () { return new lineDrawer.LineSVGDrawer(); }, function (ctx) { return new drawers.CanvasDrawer(ctx, lineDrawer.makeLineCanvasDrawStep(function () {
                var xProjector = plot.Plot._scaledAccessor(_this.x());
                var yProjector = plot.Plot._scaledAccessor(_this.y());
                return _this._d3LineFactory(dataset, xProjector, yProjector);
            })); }));
            _super.prototype._addDataset.call(this, dataset);
            return this;
        };
        Area.prototype._createNodesForDataset = function (dataset) {
            _super.prototype._createNodesForDataset.call(this, dataset);
            var drawer = this._lineDrawers.get(dataset);
            if (this.renderer() === "svg") {
                drawer.useSVG(this._renderArea);
            }
            else {
                drawer.useCanvas(this._canvas);
            }
            return drawer;
        };
        Area.prototype._removeDatasetNodes = function (dataset) {
            _super.prototype._removeDatasetNodes.call(this, dataset);
            this._lineDrawers.get(dataset).remove();
        };
        Area.prototype._additionalPaint = function () {
            var _this = this;
            var drawSteps = this._generateLineDrawSteps();
            var dataToDraw = this._getDataToDraw();
            this.datasets().forEach(function (dataset) {
                var appliedDrawSteps = plot.Plot.applyDrawSteps(drawSteps, dataset);
                _this._lineDrawers.get(dataset).draw(dataToDraw.get(dataset), appliedDrawSteps);
            });
        };
        Area.prototype._generateLineDrawSteps = function () {
            var drawSteps = [];
            if (this._animateOnNextRender()) {
                var attrToProjector = this._generateLineAttrToProjector();
                attrToProjector["d"] = this._constructLineProjector(plot.Plot._scaledAccessor(this.x()), this._getResetYFunction());
                drawSteps.push({ attrToProjector: attrToProjector, animator: this._getAnimator(plots.Animator.RESET) });
            }
            drawSteps.push({
                attrToProjector: this._generateLineAttrToProjector(),
                animator: this._getAnimator(plots.Animator.MAIN),
            });
            return drawSteps;
        };
        Area.prototype._generateLineAttrToProjector = function () {
            var lineAttrToProjector = this._getAttrToProjector();
            lineAttrToProjector["d"] = this._constructLineProjector(plot.Plot._scaledAccessor(this.x()), plot.Plot._scaledAccessor(this.y()));
            return lineAttrToProjector;
        };
        Area.prototype._createDrawer = function (dataset) {
            var _this = this;
            return new drawer.ProxyDrawer(function () { return new areaDrawer.AreaSVGDrawer(); }, function (ctx) {
                return new drawers.CanvasDrawer(ctx, areaDrawer.makeAreaCanvasDrawStep(function () {
                    var _a = _this._coordinateProjectors(), xProjector = _a[0], yProjector = _a[1], y0Projector = _a[2];
                    var definedProjector = _this._createDefinedProjector(xProjector, yProjector);
                    return _this._createAreaGenerator(xProjector, yProjector, y0Projector, definedProjector, dataset);
                }, function () {
                    var _a = _this._coordinateProjectors(), xProjector = _a[0], yProjector = _a[1];
                    var definedProjector = _this._createDefinedProjector(xProjector, yProjector);
                    return _this._createTopLineGenerator(xProjector, yProjector, definedProjector, dataset);
                }));
            });
        };
        Area.prototype._generateDrawSteps = function () {
            var drawSteps = [];
            if (this._animateOnNextRender()) {
                var attrToProjector = this._getAttrToProjector();
                attrToProjector["d"] = this._constructAreaProjector(plot.Plot._scaledAccessor(this.x()), this._getResetYFunction(), plot.Plot._scaledAccessor(this.y0()));
                drawSteps.push({ attrToProjector: attrToProjector, animator: this._getAnimator(plots.Animator.RESET) });
            }
            drawSteps.push({
                attrToProjector: this._getAttrToProjector(),
                animator: this._getAnimator(plots.Animator.MAIN),
            });
            return drawSteps;
        };
        Area.prototype._updateYScale = function () {
            var extents = this.getExtentsForProperty("y0");
            var extent = utils$1.Array.flatten(extents);
            var uniqExtentVals = utils$1.Array.uniq(extent);
            var constantBaseline = uniqExtentVals.length === 1 ? uniqExtentVals[0] : null;
            var yBinding = this.y();
            var yScale = (yBinding && yBinding.scale);
            if (yScale == null) {
                return;
            }
            if (this._constantBaselineValueProvider != null) {
                yScale.removePaddingExceptionsProvider(this._constantBaselineValueProvider);
                this._constantBaselineValueProvider = null;
            }
            if (constantBaseline != null) {
                this._constantBaselineValueProvider = function () { return [constantBaseline]; };
                yScale.addPaddingExceptionsProvider(this._constantBaselineValueProvider);
            }
        };
        Area.prototype._getResetYFunction = function () {
            return plot.Plot._scaledAccessor(this.y0());
        };
        Area.prototype._coordinateProjectors = function () {
            return [
                plot.Plot._scaledAccessor(this.x()),
                plot.Plot._scaledAccessor(this.y()),
                plot.Plot._scaledAccessor(this.y0()),
            ];
        };
        Area.prototype._propertyProjectors = function () {
            var propertyToProjectors = _super.prototype._propertyProjectors.call(this);
            var _a = this._coordinateProjectors(), xProject = _a[0], yProjector = _a[1], y0Projector = _a[2];
            propertyToProjectors["d"] = this._constructAreaProjector(xProject, yProjector, y0Projector);
            return propertyToProjectors;
        };
        Area.prototype.selections = function (datasets) {
            var _this = this;
            if (datasets === void 0) { datasets = this.datasets(); }
            if (this.renderer() === "canvas") {
                return d3.selectAll();
            }
            var allSelections = _super.prototype.selections.call(this, datasets).nodes();
            var lineDrawers = datasets.map(function (dataset) { return _this._lineDrawers.get(dataset); })
                .filter(function (drawer) { return drawer != null; });
            lineDrawers.forEach(function (ld) { return allSelections.push.apply(allSelections, ld.getVisualPrimitives()); });
            return d3.selectAll(allSelections);
        };
        Area.prototype._constructAreaProjector = function (xProjector, yProjector, y0Projector) {
            var _this = this;
            var definedProjector = this._createDefinedProjector(plot.Plot._scaledAccessor(this.x()), plot.Plot._scaledAccessor(this.y()));
            return function (datum, index, dataset) {
                var areaGenerator = _this._createAreaGenerator(xProjector, yProjector, y0Projector, definedProjector, dataset);
                return areaGenerator(datum);
            };
        };
        Area.prototype._createDefinedProjector = function (xProjector, yProjector) {
            return function (d, i, dataset) {
                var positionX = xProjector(d, i, dataset);
                var positionY = yProjector(d, i, dataset);
                return utils$1.Math.isValidNumber(positionX) && utils$1.Math.isValidNumber(positionY);
            };
        };
        Area.prototype._createAreaGenerator = function (xProjector, yProjector, y0Projector, definedProjector, dataset) {
            // just runtime error if user passes curveBundle to area plot
            var curveFactory = this._getCurveFactory();
            var areaGenerator = d3.area()
                .x(function (innerDatum, innerIndex) { return xProjector(innerDatum, innerIndex, dataset); })
                .y1(function (innerDatum, innerIndex) { return yProjector(innerDatum, innerIndex, dataset); })
                .y0(function (innerDatum, innerIndex) { return y0Projector(innerDatum, innerIndex, dataset); })
                .curve(curveFactory)
                .defined(function (innerDatum, innerIndex) { return definedProjector(innerDatum, innerIndex, dataset); });
            return areaGenerator;
        };
        Area.prototype._createTopLineGenerator = function (xProjector, yProjector, definedProjector, dataset) {
            var curveFactory = this._getCurveFactory();
            var areaGenerator = d3.line()
                .x(function (innerDatum, innerIndex) { return xProjector(innerDatum, innerIndex, dataset); })
                .y(function (innerDatum, innerIndex) { return yProjector(innerDatum, innerIndex, dataset); })
                .curve(curveFactory)
                .defined(function (innerDatum, innerIndex) { return definedProjector(innerDatum, innerIndex, dataset); });
            return areaGenerator;
        };
        Area._Y0_KEY = "y0";
        return Area;
    }(linePlot.Line));
    exports.Area = Area;
    });

    var barPlot = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });
















    exports.BarOrientation = makeEnum_1.makeEnum(["vertical", "horizontal"]);
    exports.LabelsPosition = makeEnum_1.makeEnum(["start", "middle", "end", "outside"]);
    exports.BarAlignment = makeEnum_1.makeEnum(["start", "middle", "end"]);
    var Bar = /** @class */ (function (_super) {
        tslib_es6.__extends(Bar, _super);
        /**
         * A Bar Plot draws bars growing out from a baseline to some value
         *
         * @constructor
         * @param {string} [orientation="vertical"] One of "vertical"/"horizontal".
         */
        function Bar(orientation) {
            if (orientation === void 0) { orientation = "vertical"; }
            var _this = _super.call(this) || this;
            _this._labelFormatter = formatters.identity();
            _this._labelsEnabled = false;
            _this._labelsPosition = exports.LabelsPosition.end;
            _this._hideBarsIfAnyAreTooWide = true;
            _this._barAlignment = "middle";
            _this._computeBarPixelThickness = memoize.memoize(computeBarPixelThickness);
            /**
             * Whether all the bars in this barPlot have the same pixel thickness.
             * If so, use the _barPixelThickness property to access the thickness.
             */
            _this._fixedBarPixelThickness = true;
            _this.addClass("bar-plot");
            if (orientation !== "vertical" && orientation !== "horizontal") {
                throw new Error(orientation + " is not a valid orientation for Plots.Bar");
            }
            _this._isVertical = orientation === "vertical";
            _this.animator("baseline", new animators.Null());
            _this.attr("fill", new scales.Color().range()[0]);
            _this.attr(Bar._BAR_THICKNESS_KEY, function () { return _this._barPixelThickness(); });
            _this._labelConfig = new utils$1.Map();
            _this._baselineValueProvider = function () { return [_this.baselineValue()]; };
            return _this;
        }
        Bar.prototype.computeLayout = function (origin, availableWidth, availableHeight) {
            _super.prototype.computeLayout.call(this, origin, availableWidth, availableHeight);
            this._updateExtents();
            return this;
        };
        Bar.prototype.x = function (x, xScale) {
            if (x == null) {
                return _super.prototype.x.call(this);
            }
            if (xScale == null) {
                _super.prototype.x.call(this, x);
            }
            else {
                _super.prototype.x.call(this, x, xScale);
            }
            this._updateThicknessAttr();
            this._updateLengthScale();
            return this;
        };
        Bar.prototype.y = function (y, yScale) {
            if (y == null) {
                return _super.prototype.y.call(this);
            }
            if (yScale == null) {
                _super.prototype.y.call(this, y);
            }
            else {
                _super.prototype.y.call(this, y, yScale);
            }
            this._updateLengthScale();
            return this;
        };
        /**
         * The binding associated with bar length. Length is the count or value the bar is trying to show.
         * This is the .y() for a vertical plot and .x() for a horizontal plot.
         */
        Bar.prototype.length = function () {
            return this._isVertical ? this.y() : this.x();
        };
        /**
         * The binding associated with bar position. Position separates the different bar categories.
         * This is the .x() for a vertical plot and .y() for a horizontal plot.
         */
        Bar.prototype.position = function () {
            return this._isVertical ? this.x() : this.y();
        };
        Bar.prototype.barEnd = function (end) {
            if (end == null) {
                return this._propertyBindings.get(Bar._BAR_END_KEY);
            }
            var binding = this.position();
            var scale = binding && binding.scale;
            this._bindProperty(Bar._BAR_END_KEY, end, scale);
            this._updateThicknessAttr();
            this._updateLengthScale();
            this.render();
            return this;
        };
        Bar.prototype.barAlignment = function (align) {
            if (align == null) {
                return this._barAlignment;
            }
            this._barAlignment = align;
            this._clearAttrToProjectorCache();
            this.render();
            return this;
        };
        /**
         * Gets the orientation of the plot
         *
         * @return "vertical" | "horizontal"
         */
        Bar.prototype.orientation = function () {
            return this._isVertical ? "vertical" : "horizontal";
        };
        Bar.prototype._createDrawer = function () {
            return new drawer.ProxyDrawer(function () { return new rectangleDrawer.RectangleSVGDrawer(Bar._BAR_AREA_CLASS); }, function (ctx) { return new drawers.RectangleCanvasDrawer(ctx); });
        };
        Bar.prototype._setup = function () {
            _super.prototype._setup.call(this);
            this._baseline = this._renderArea.append("line").classed("baseline", true);
        };
        Bar.prototype.baselineValue = function (value) {
            if (value == null) {
                if (this._baselineValue != null) {
                    return this._baselineValue;
                }
                if (!this._projectorsReady()) {
                    return 0;
                }
                var lengthScale = this.length().scale;
                if (!lengthScale) {
                    return 0;
                }
                if (lengthScale instanceof scales.Time) {
                    return new Date(0);
                }
                return 0;
            }
            this._baselineValue = value;
            this._updateLengthScale();
            this._clearAttrToProjectorCache();
            this.render();
            return this;
        };
        Bar.prototype.addDataset = function (dataset) {
            _super.prototype.addDataset.call(this, dataset);
            return this;
        };
        Bar.prototype._addDataset = function (dataset) {
            _super.prototype._addDataset.call(this, dataset);
            return this;
        };
        Bar.prototype.removeDataset = function (dataset) {
            _super.prototype.removeDataset.call(this, dataset);
            return this;
        };
        Bar.prototype._removeDataset = function (dataset) {
            _super.prototype._removeDataset.call(this, dataset);
            return this;
        };
        Bar.prototype.datasets = function (datasets) {
            if (datasets == null) {
                return _super.prototype.datasets.call(this);
            }
            _super.prototype.datasets.call(this, datasets);
            return this;
        };
        Bar.prototype.labelsEnabled = function (enabled, labelsPosition) {
            if (enabled == null) {
                return this._labelsEnabled;
            }
            else {
                this._labelsEnabled = enabled;
                if (labelsPosition != null) {
                    this._labelsPosition = labelsPosition;
                }
                this._clearAttrToProjectorCache();
                this.render();
                return this;
            }
        };
        Bar.prototype.labelFormatter = function (formatter) {
            if (formatter == null) {
                return this._labelFormatter;
            }
            else {
                this._labelFormatter = formatter;
                this._clearAttrToProjectorCache();
                this.render();
                return this;
            }
        };
        Bar.prototype._createNodesForDataset = function (dataset) {
            var drawer = _super.prototype._createNodesForDataset.call(this, dataset);
            var labelArea = this._renderArea.append("g").classed(Bar._LABEL_AREA_CLASS, true);
            var context = new src$1.SvgContext(labelArea.node());
            var measurer = new src$1.CacheMeasurer(context);
            var writer = new src$1.Writer(measurer, context);
            this._labelConfig.set(dataset, { labelArea: labelArea, measurer: measurer, writer: writer });
            return drawer;
        };
        Bar.prototype._removeDatasetNodes = function (dataset) {
            _super.prototype._removeDatasetNodes.call(this, dataset);
            var labelConfig = this._labelConfig.get(dataset);
            if (labelConfig != null) {
                labelConfig.labelArea.remove();
                this._labelConfig.delete(dataset);
            }
        };
        /**
         * Returns the PlotEntity nearest to the query point according to the following algorithm:
         *   - If the query point is inside a bar, returns the PlotEntity for that bar.
         *   - Otherwise, gets the nearest PlotEntity by the positional direction (X for vertical, Y for horizontal),
         *     breaking ties with the secondary direction.
         * Returns undefined if no PlotEntity can be found.
         *
         * @param {Point} queryPoint
         * @returns {PlotEntity} The nearest PlotEntity, or undefined if no PlotEntity can be found.
         */
        Bar.prototype.entityNearest = function (queryPoint) {
            var _this = this;
            var worker = function () {
                var nearest = _this._isVertical ?
                    _this._getEntityStore().entityNearestX(queryPoint) :
                    _this._getEntityStore().entityNearestY(queryPoint);
                return nearest === undefined ? undefined : _this._lightweightPlotEntityToPlotEntity(nearest);
            };
            return this._fixedBarPixelThickness ? this._computeBarPixelThickness.doLocked(worker) : worker();
        };
        /**
         * Gets the Entities at a particular Point.
         *
         * @param {Point} p
         * @returns {PlotEntity[]}
         */
        Bar.prototype.entitiesAt = function (p) {
            var _this = this;
            var worker = function () { return _this._entitiesIntersecting(p.x, p.y); };
            return this._fixedBarPixelThickness ? this._computeBarPixelThickness.doLocked(worker) : worker();
        };
        Bar.prototype.entitiesInBounds = function (queryBounds) {
            var _this = this;
            var worker = function () { return _super.prototype.entitiesInBounds.call(_this, queryBounds); };
            return this._fixedBarPixelThickness ? this._computeBarPixelThickness.doLocked(worker) : worker();
        };
        Bar.prototype.entitiesInXBounds = function (queryBounds) {
            var _this = this;
            var worker = function () { return _super.prototype.entitiesInXBounds.call(_this, queryBounds); };
            return this._fixedBarPixelThickness ? this._computeBarPixelThickness.doLocked(worker) : worker();
        };
        Bar.prototype.entitiesInYBounds = function (queryBounds) {
            var _this = this;
            var worker = function () { return _super.prototype.entitiesInYBounds.call(_this, queryBounds); };
            return this._fixedBarPixelThickness ? this._computeBarPixelThickness.doLocked(worker) : worker();
        };
        Bar.prototype._entitiesIntersecting = function (xValOrRange, yValOrRange) {
            var intersected = [];
            var entities = this._getEntityStore().entities();
            var entitiesLen = entities.length;
            for (var i = 0; i < entitiesLen; i++) {
                var entity = entities[i];
                if (utils$1.DOM.intersectsBBox(xValOrRange, yValOrRange, this._entityBounds(entity))) {
                    intersected.push(this._lightweightPlotEntityToPlotEntity(entity));
                }
            }
            return intersected;
        };
        Bar.prototype._updateLengthScale = function () {
            if (!this._projectorsReady()) {
                return;
            }
            var lengthScale = this.length().scale;
            if (lengthScale instanceof quantitativeScale.QuantitativeScale) {
                lengthScale.addPaddingExceptionsProvider(this._baselineValueProvider);
                lengthScale.addIncludedValuesProvider(this._baselineValueProvider);
            }
        };
        Bar.prototype.renderImmediately = function () {
            var _this = this;
            // HACK update bar pixel thickness
            this._barPixelThickness();
            return this._computeBarPixelThickness.doLocked(function () { return _super.prototype.renderImmediately.call(_this); });
        };
        Bar.prototype._additionalPaint = function (time) {
            var _this = this;
            var lengthScale = this.length().scale;
            var scaledBaseline = lengthScale.scale(this.baselineValue());
            var baselineAttr = {
                x1: this._isVertical ? 0 : scaledBaseline,
                y1: this._isVertical ? scaledBaseline : 0,
                x2: this._isVertical ? this.width() : scaledBaseline,
                y2: this._isVertical ? scaledBaseline : this.height(),
            };
            this._getAnimator("baseline").animate(this._baseline, baselineAttr);
            this.datasets().forEach(function (dataset) { return _this._labelConfig.get(dataset).labelArea.selectAll("g").remove(); });
            if (this._labelsEnabled) {
                utils$1.Window.setTimeout(function () { return _this._drawLabels(); }, time);
            }
        };
        /**
         * Makes sure the extent takes into account the widths of the bars
         */
        Bar.prototype.getExtentsForProperty = function (property) {
            var _this = this;
            var extents = _super.prototype.getExtentsForProperty.call(this, property);
            var accScaleBinding;
            if (property === "x" && this._isVertical) {
                accScaleBinding = this.x();
            }
            else if (property === "y" && !this._isVertical) {
                accScaleBinding = this.y();
            }
            else {
                return extents;
            }
            if (!(accScaleBinding && accScaleBinding.scale && accScaleBinding.scale instanceof quantitativeScale.QuantitativeScale)) {
                return extents;
            }
            var scale = accScaleBinding.scale;
            var width = this._barPixelThickness();
            // To account for inverted domains
            extents = extents.map(function (extent) { return d3.extent([
                scale.invert(_this._getPositionAttr(scale.scale(extent[0]), width)),
                scale.invert(_this._getPositionAttr(scale.scale(extent[0]), width) + width),
                scale.invert(_this._getPositionAttr(scale.scale(extent[1]), width)),
                scale.invert(_this._getPositionAttr(scale.scale(extent[1]), width) + width),
            ]); });
            return extents;
        };
        /**
         * Return the <rect>'s x or y attr value given the position and thickness of
         * that bar. This method is responsible for account for barAlignment, in particular.
         */
        Bar.prototype._getPositionAttr = function (position, thickness) {
            // account for flipped vertical axis
            if (!this._isVertical) {
                position -= thickness;
                thickness *= -1;
            }
            switch (this._barAlignment) {
                case "start":
                    return position;
                case "end":
                    return position - thickness;
                case "middle":
                default:
                    return position - thickness / 2;
            }
        };
        Bar.prototype._drawLabels = function () {
            var _this = this;
            var dataToDraw = this._getDataToDraw();
            var attrToProjector = this._getAttrToProjector();
            var anyLabelTooWide = this.datasets().some(function (dataset) {
                return dataToDraw.get(dataset).some(function (datum, index) {
                    if (datum == null) {
                        return false;
                    }
                    return _this._drawLabel(datum, index, dataset, attrToProjector);
                });
            });
            if (this._hideBarsIfAnyAreTooWide && anyLabelTooWide) {
                this.datasets().forEach(function (dataset) { return _this._labelConfig.get(dataset).labelArea.selectAll("g").remove(); });
            }
        };
        Bar.prototype._drawLabel = function (datum, index, dataset, attrToProjector) {
            var _a = this._labelConfig.get(dataset), labelArea = _a.labelArea, measurer = _a.measurer, writer = _a.writer;
            var lengthAccessor = this.length().accessor;
            var length = lengthAccessor(datum, index, dataset);
            var lengthScale = this.length().scale;
            var scaledLength = lengthScale != null ? lengthScale.scale(length) : length;
            var scaledBaseline = lengthScale != null ? lengthScale.scale(this.baselineValue()) : this.baselineValue();
            var barCoordinates = { x: attrToProjector["x"](datum, index, dataset), y: attrToProjector["y"](datum, index, dataset) };
            var barDimensions = { width: attrToProjector["width"](datum, index, dataset), height: attrToProjector["height"](datum, index, dataset) };
            var text = this._labelFormatter(length, datum, index, dataset);
            var measurement = measurer.measure(text);
            var showLabelOnBar = this._shouldShowLabelOnBar(barCoordinates, barDimensions, measurement);
            // show label on right when value === baseline for horizontal plots
            var aboveOrLeftOfBaseline = this._isVertical ? scaledLength <= scaledBaseline : scaledLength < scaledBaseline;
            var _b = this._calculateLabelProperties(barCoordinates, barDimensions, measurement, showLabelOnBar, aboveOrLeftOfBaseline), containerDimensions = _b.containerDimensions, labelContainerOrigin = _b.labelContainerOrigin, labelOrigin = _b.labelOrigin, alignment = _b.alignment;
            var color = attrToProjector["fill"](datum, index, dataset);
            var labelContainer = this._createLabelContainer(labelArea, labelContainerOrigin, labelOrigin, measurement, showLabelOnBar, color);
            var writeOptions = { xAlign: alignment.x, yAlign: alignment.y };
            writer.write(text, containerDimensions.width, containerDimensions.height, writeOptions, labelContainer.node());
            var tooWide = this._isVertical
                ? barDimensions.width < (measurement.width)
                : barDimensions.height < (measurement.height);
            return tooWide;
        };
        /**
         * Labels are "on-bar" by default, but if the bar is not long enough to fit the text,
         * we can try putting the label "off-bar", if there's enough space outside of the bar
         * to fit it.
         */
        Bar.prototype._shouldShowLabelOnBar = function (barCoordinates, barDimensions, labelDimensions) {
            if (this._labelsPosition === exports.LabelsPosition.outside) {
                return false;
            }
            var barStart = this._isVertical ? barCoordinates.y : barCoordinates.x;
            var barLength = this._isVertical ? barDimensions.height : barDimensions.width;
            var totalLength = this._isVertical ? this.height() : this.width();
            var labelLength = this._isVertical ? labelDimensions.height : labelDimensions.width;
            var barEnd = barStart + barLength;
            var barLengthVisibleOnScreen = barLength;
            if (barEnd > totalLength) {
                barLengthVisibleOnScreen = totalLength - barStart;
            }
            else if (barStart < 0) {
                barLengthVisibleOnScreen = barEnd;
            }
            return (labelLength + Bar._LABEL_MARGIN_INSIDE_BAR <= barLengthVisibleOnScreen);
        };
        Bar.prototype._calculateLabelProperties = function (barCoordinates, barDimensions, measurement, showLabelOnBar, aboveOrLeftOfBaseline) {
            var _this = this;
            var barCoordinate = this._isVertical ? barCoordinates.y : barCoordinates.x;
            var barDimension = this._isVertical ? barDimensions.height : barDimensions.width;
            var measurementDimension = this._isVertical ? measurement.height : measurement.width;
            var alignmentDimension = "center";
            var containerDimension = barDimension;
            var labelContainerOriginCoordinate = barCoordinate;
            var labelOriginCoordinate = barCoordinate;
            var updateCoordinates = function (position) {
                switch (position) {
                    case "topLeft":
                        alignmentDimension = _this._isVertical ? "top" : "left";
                        labelContainerOriginCoordinate += Bar._LABEL_MARGIN_INSIDE_BAR;
                        labelOriginCoordinate += Bar._LABEL_MARGIN_INSIDE_BAR;
                        return;
                    case "center":
                        labelOriginCoordinate += (barDimension + measurementDimension) / 2;
                        return;
                    case "bottomRight":
                        alignmentDimension = _this._isVertical ? "bottom" : "right";
                        labelContainerOriginCoordinate -= Bar._LABEL_MARGIN_INSIDE_BAR;
                        labelOriginCoordinate += containerDimension - Bar._LABEL_MARGIN_INSIDE_BAR - measurementDimension;
                        return;
                }
            };
            if (showLabelOnBar) {
                switch (this._labelsPosition) {
                    case exports.LabelsPosition.start:
                        aboveOrLeftOfBaseline ? updateCoordinates("bottomRight") : updateCoordinates("topLeft");
                        break;
                    case exports.LabelsPosition.middle:
                        updateCoordinates("center");
                        break;
                    case exports.LabelsPosition.end:
                        aboveOrLeftOfBaseline ? updateCoordinates("topLeft") : updateCoordinates("bottomRight");
                        break;
                }
            }
            else {
                if (aboveOrLeftOfBaseline) {
                    alignmentDimension = this._isVertical ? "top" : "left";
                    containerDimension = barDimension + Bar._LABEL_MARGIN_INSIDE_BAR + measurementDimension;
                    labelContainerOriginCoordinate -= Bar._LABEL_MARGIN_INSIDE_BAR + measurementDimension;
                    labelOriginCoordinate -= Bar._LABEL_MARGIN_INSIDE_BAR + measurementDimension;
                }
                else {
                    alignmentDimension = this._isVertical ? "bottom" : "right";
                    containerDimension = barDimension + Bar._LABEL_MARGIN_INSIDE_BAR + measurementDimension;
                    labelOriginCoordinate += barDimension + Bar._LABEL_MARGIN_INSIDE_BAR;
                }
            }
            return {
                containerDimensions: {
                    width: this._isVertical ? barDimensions.width : containerDimension,
                    height: this._isVertical ? containerDimension : barDimensions.height,
                },
                labelContainerOrigin: {
                    x: this._isVertical ? barCoordinates.x : labelContainerOriginCoordinate,
                    y: this._isVertical ? labelContainerOriginCoordinate : barCoordinates.y,
                },
                labelOrigin: {
                    x: this._isVertical ? (barCoordinates.x + barDimensions.width / 2 - measurement.width / 2) : labelOriginCoordinate,
                    y: this._isVertical ? labelOriginCoordinate : (barCoordinates.y + barDimensions.height / 2 - measurement.height / 2),
                },
                alignment: {
                    x: this._isVertical ? "center" : alignmentDimension,
                    y: this._isVertical ? alignmentDimension : "center",
                },
            };
        };
        Bar.prototype._createLabelContainer = function (labelArea, labelContainerOrigin, labelOrigin, measurement, showLabelOnBar, color) {
            var labelContainer = labelArea.append("g").attr("transform", "translate(" + labelContainerOrigin.x + ", " + labelContainerOrigin.y + ")");
            if (showLabelOnBar) {
                labelContainer.classed("on-bar-label", true);
                var dark = utils$1.Color.contrast("white", color) * 1.6 < utils$1.Color.contrast("black", color);
                labelContainer.classed(dark ? "dark-label" : "light-label", true);
            }
            else {
                labelContainer.classed("off-bar-label", true);
            }
            return labelContainer;
        };
        Bar.prototype._generateDrawSteps = function () {
            var drawSteps = [];
            if (this._animateOnNextRender()) {
                var resetAttrToProjector = this._getAttrToProjector();
                var lengthScale = this.length().scale;
                var scaledBaseline_1 = lengthScale.scale(this.baselineValue());
                var lengthAttr = this._isVertical ? "y" : "x";
                var thicknessAttr = this._isVertical ? "height" : "width";
                resetAttrToProjector[lengthAttr] = function () { return scaledBaseline_1; };
                resetAttrToProjector[thicknessAttr] = function () { return 0; };
                drawSteps.push({ attrToProjector: resetAttrToProjector, animator: this._getAnimator(plots.Animator.RESET) });
            }
            drawSteps.push({
                attrToProjector: this._getAttrToProjector(),
                animator: this._getAnimator(plots.Animator.MAIN),
            });
            return drawSteps;
        };
        Bar.prototype._generateAttrToProjector = function () {
            var _this = this;
            var attrToProjector = _super.prototype._generateAttrToProjector.call(this);
            var lengthScale = this.length().scale;
            var scaledBaseline = lengthScale.scale(this.baselineValue());
            var lengthAttr = this._isVertical ? "y" : "x";
            var positionAttr = this._isVertical ? "x" : "y";
            var positionF = plot.Plot._scaledAccessor(this.position());
            var originalLengthFn = plot.Plot._scaledAccessor(this.length());
            var lengthFn = function (d, i, dataset) {
                return Math.abs(scaledBaseline - originalLengthFn(d, i, dataset));
            };
            var thicknessF = attrToProjector[Bar._BAR_THICKNESS_KEY];
            var gapF = attrToProjector["gap"];
            var thicknessMinusGap = gapF == null ? thicknessF : function (d, i, dataset) {
                var thick = thicknessF(d, i, dataset);
                // only subtract gap if bars are at least 2 pixels wide, otherwise canvas
                // interpolation can cause bars to become invisible due to subpixel
                // sampling
                return thick < Bar._BAR_GAPLESS_THRESHOLD_PX ? thick : thick - gapF(d, i, dataset);
            };
            // re-interpret "width" attr from representing "thickness" to actually meaning
            // width (that is, x-direction specific) again
            attrToProjector["width"] = this._isVertical ? thicknessMinusGap : lengthFn;
            attrToProjector["height"] = this._isVertical ? lengthFn : thicknessMinusGap;
            attrToProjector[lengthAttr] = function (d, i, dataset) {
                var originalLength = originalLengthFn(d, i, dataset);
                // If it is past the baseline, it should start at the baseline then width/height
                // carries it over. If it's not past the baseline, leave it at original position and
                // then width/height carries it to baseline
                return (originalLength > scaledBaseline) ? scaledBaseline : originalLength;
            };
            attrToProjector[positionAttr] = function (d, i, dataset) {
                return _this._getPositionAttr(positionF(d, i, dataset), thicknessF(d, i, dataset));
            };
            return attrToProjector;
        };
        Bar.prototype._updateThicknessAttr = function () {
            var _this = this;
            var startProj = this.position();
            var endProj = this.barEnd();
            if (startProj != null && endProj != null) {
                this._fixedBarPixelThickness = false;
                this.attr(Bar._BAR_THICKNESS_KEY, function (d, i, data) {
                    var v1 = startProj.accessor(d, i, data);
                    var v2 = endProj.accessor(d, i, data);
                    v1 = startProj.scale ? startProj.scale.scale(v1) : v1;
                    v2 = endProj.scale ? endProj.scale.scale(v2) : v2;
                    return Math.abs(v2 - v1);
                });
            }
            else {
                this._fixedBarPixelThickness = true;
                this.attr(Bar._BAR_THICKNESS_KEY, function () { return _this._barPixelThickness(); });
            }
        };
        Bar.prototype._barPixelThickness = function () {
            if (this._fixedBarPixelThickness) {
                if (this._projectorsReady()) {
                    return this._computeBarPixelThickness(this.position(), this.datasets(), this._isVertical ? this.width() : this.height());
                }
                else {
                    return 0;
                }
            }
            else {
                return 0;
            }
        };
        Bar.prototype.entities = function (datasets) {
            if (datasets === void 0) { datasets = this.datasets(); }
            if (!this._projectorsReady()) {
                return [];
            }
            var entities = _super.prototype.entities.call(this, datasets);
            return entities;
        };
        Bar.prototype._entityBounds = function (entity) {
            var datum = entity.datum, index = entity.index, dataset = entity.dataset;
            return this._pixelBounds(datum, index, dataset);
        };
        /**
         * The rectangular bounds of a bar. Note that the x/y coordinates are not the
         * same as the "pixel point" because they are always at the top/left of the
         * bar.
         */
        Bar.prototype._pixelBounds = function (datum, index, dataset) {
            var attrToProjector = this._getAttrToProjector();
            return {
                x: attrToProjector["x"](datum, index, dataset),
                y: attrToProjector["y"](datum, index, dataset),
                width: attrToProjector["width"](datum, index, dataset),
                height: attrToProjector["height"](datum, index, dataset),
            };
        };
        /**
         * The "pixel point" of a bar is the farthest point from the baseline.
         *
         * For example, in a vertical bar chart with positive bar values, the pixel
         * point will be at the top of the bar. For negative bar values, the pixel
         * point will be at the bottom of the bar.
         */
        Bar.prototype._pixelPoint = function (datum, index, dataset) {
            var rect = this._pixelBounds(datum, index, dataset);
            var originalPosition = (this._isVertical ? plot.Plot._scaledAccessor(this.y()) : plot.Plot._scaledAccessor(this.x()))(datum, index, dataset);
            var scaledBaseline = (this._isVertical ? this.y().scale : this.x().scale).scale(this.baselineValue());
            return this._pixelPointBar(originalPosition, scaledBaseline, rect);
        };
        Bar.prototype._pixelPointBar = function (originalPosition, scaledBaseline, rect) {
            var x, y;
            if (this._isVertical) {
                x = rect.x + rect.width / 2;
                y = originalPosition <= scaledBaseline ? rect.y : rect.y + rect.height;
            }
            else {
                x = originalPosition >= scaledBaseline ? rect.x + rect.width : rect.x;
                y = rect.y + rect.height / 2;
            }
            return { x: x, y: y };
        };
        Bar.prototype._uninstallScaleForKey = function (scale, key) {
            _super.prototype._uninstallScaleForKey.call(this, scale, key);
        };
        Bar.prototype._getDataToDraw = function () {
            var _this = this;
            var dataToDraw = new utils$1.Map();
            var attrToProjector = this._getAttrToProjector();
            var plotWidth = this.width();
            var plotHeight = this.height();
            this.datasets().forEach(function (dataset) {
                var data = dataset.data().map(function (d, i) {
                    var isValid = _this._isDatumOnScreen(attrToProjector, plotWidth, plotHeight, d, i, dataset);
                    return isValid ? d : null;
                });
                dataToDraw.set(dataset, data);
            });
            return dataToDraw;
        };
        Bar.prototype._isDatumOnScreen = function (attrToProjector, plotWidth, plotHeight, d, i, dataset) {
            var pixelX = attrToProjector["x"](d, i, dataset);
            var pixelY = attrToProjector["y"](d, i, dataset);
            var pixelWidth = attrToProjector["width"](d, i, dataset);
            var pixelHeight = attrToProjector["height"](d, i, dataset);
            var isValid = utils$1.Math.isValidNumber(pixelX) &&
                utils$1.Math.isValidNumber(pixelY) &&
                utils$1.Math.isValidNumber(pixelWidth) &&
                utils$1.Math.isValidNumber(pixelHeight);
            if (!isValid) {
                return false;
            }
            return utils$1.Math.boundsIntersects(pixelX, pixelY, pixelWidth, pixelHeight, 0, 0, plotWidth, plotHeight);
        };
        Bar.prototype.invalidateCache = function () {
            var _this = this;
            _super.prototype.invalidateCache.call(this);
            this.datasets().forEach(function (dataset) { return _this._labelConfig.get(dataset).measurer.reset(); });
        };
        Bar._BAR_THICKNESS_RATIO = 0.95;
        Bar._BAR_GAPLESS_THRESHOLD_PX = 3;
        Bar._SINGLE_BAR_DIMENSION_RATIO = 0.4;
        Bar._BAR_AREA_CLASS = "bar-area";
        Bar._BAR_END_KEY = "barEnd";
        // we special case the "width" property to represent the bar thickness
        // (aka the distance between adjacent bar positions); in _generateAttrToProjector
        // we re-assign "width" to specifically refer to <rect>'s width attribute
        Bar._BAR_THICKNESS_KEY = "width";
        Bar._LABEL_AREA_CLASS = "bar-label-text-area";
        /**
         * In the case of "start" or "end" LabelPositions, put the label this many px away
         * from the bar's length distance edge
         */
        Bar._LABEL_MARGIN_INSIDE_BAR = 10;
        return Bar;
    }(xyPlot.XYPlot));
    exports.Bar = Bar;
    /**
     * Computes the barPixelThickness of all the bars in the plot.
     *
     * If the position scale of the plot is a CategoryScale and in bands mode, then the rangeBands function will be used.
     * If the position scale of the plot is a QuantitativeScale, then the bar thickness is equal to the smallest distance between
     * two adjacent data points, padded for visualisation.
     *
     * This is ignored when explicitly setting the barEnd.
     */
    function computeBarPixelThickness(positionBinding, datasets, plotPositionDimensionLength) {
        var barPixelThickness;
        var positionScale = positionBinding.scale;
        if (positionScale instanceof scales.Category) {
            barPixelThickness = positionScale.rangeBand();
        }
        else {
            var positionAccessor_1 = positionBinding.accessor;
            var numberBarAccessorData = d3.set(utils$1.Array.flatten(datasets.map(function (dataset) {
                return dataset.data().map(function (d, i) { return positionAccessor_1(d, i, dataset); })
                    .filter(function (d) { return d != null; })
                    .map(function (d) { return d.valueOf(); });
            }))).values().map(function (value) { return +value; });
            numberBarAccessorData.sort(function (a, b) { return a - b; });
            var scaledData = numberBarAccessorData.map(function (datum) { return positionScale.scale(datum); });
            var barAccessorDataPairs = d3.pairs(scaledData);
            barPixelThickness = utils$1.Math.min(barAccessorDataPairs, function (pair, i) {
                return Math.abs(pair[1] - pair[0]);
            }, plotPositionDimensionLength * Bar._SINGLE_BAR_DIMENSION_RATIO);
            barPixelThickness *= Bar._BAR_THICKNESS_RATIO;
        }
        return barPixelThickness;
    }
    });

    var clusteredBarPlot = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });





    var ClusteredBar = /** @class */ (function (_super) {
        tslib_es6.__extends(ClusteredBar, _super);
        /**
         * A ClusteredBar Plot groups bars across Datasets based on the primary value of the bars.
         *   On a vertical ClusteredBar Plot, the bars with the same X value are grouped.
         *   On a horizontal ClusteredBar Plot, the bars with the same Y value are grouped.
         *
         * @constructor
         * @param {string} [orientation="vertical"] One of "vertical"/"horizontal".
         */
        function ClusteredBar(orientation) {
            if (orientation === void 0) { orientation = "vertical"; }
            var _this = _super.call(this, orientation) || this;
            _this._clusterOffsets = new utils$1.Map();
            return _this;
        }
        ClusteredBar.prototype._generateAttrToProjector = function () {
            var _this = this;
            var attrToProjector = _super.prototype._generateAttrToProjector.call(this);
            // the width is constant, so set the inner scale range to that
            var innerScale = this._makeInnerScale();
            var innerWidthF = function (d, i) { return innerScale.rangeBand(); };
            attrToProjector["width"] = this._isVertical ? innerWidthF : attrToProjector["width"];
            attrToProjector["height"] = !this._isVertical ? innerWidthF : attrToProjector["height"];
            var xAttr = attrToProjector["x"];
            var yAttr = attrToProjector["y"];
            attrToProjector["x"] = this._isVertical ?
                function (d, i, ds) { return xAttr(d, i, ds) + _this._clusterOffsets.get(ds); } :
                function (d, i, ds) { return xAttr(d, i, ds); };
            attrToProjector["y"] = this._isVertical ?
                function (d, i, ds) { return yAttr(d, i, ds); } :
                function (d, i, ds) { return yAttr(d, i, ds) + _this._clusterOffsets.get(ds); };
            return attrToProjector;
        };
        ClusteredBar.prototype._updateClusterPosition = function () {
            var _this = this;
            var innerScale = this._makeInnerScale();
            this.datasets().forEach(function (d, i) { return _this._clusterOffsets.set(d, innerScale.scale(String(i)) - innerScale.rangeBand() / 2); });
        };
        ClusteredBar.prototype._makeInnerScale = function () {
            var innerScale = new scales.Category();
            innerScale.domain(this.datasets().map(function (d, i) { return String(i); }));
            var widthProjector = plot.Plot._scaledAccessor(this.attr(barPlot.Bar._BAR_THICKNESS_KEY));
            innerScale.range([0, widthProjector(null, 0, null)]);
            return innerScale;
        };
        ClusteredBar.prototype._getDataToDraw = function () {
            this._updateClusterPosition();
            return _super.prototype._getDataToDraw.call(this);
        };
        return ClusteredBar;
    }(barPlot.Bar));
    exports.ClusteredBar = ClusteredBar;
    });

    var piePlot = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });












    var Pie = /** @class */ (function (_super) {
        tslib_es6.__extends(Pie, _super);
        /**
         * @constructor
         */
        function Pie() {
            var _this = _super.call(this) || this;
            _this._startAngle = 0;
            _this._endAngle = 2 * Math.PI;
            _this._labelFormatter = formatters.identity();
            _this._labelsEnabled = false;
            _this.innerRadius(0);
            _this.outerRadius(function () {
                var pieCenter = _this._pieCenter();
                return Math.min(Math.max(_this.width() - pieCenter.x, pieCenter.x), Math.max(_this.height() - pieCenter.y, pieCenter.y));
            });
            _this.addClass("pie-plot");
            _this.attr("fill", function (d, i) { return String(i); }, new scales.Color());
            _this._strokeDrawers = new utils$1.Map();
            return _this;
        }
        Pie.prototype._setup = function () {
            var _this = this;
            _super.prototype._setup.call(this);
            this._strokeDrawers.forEach(function (d) { return d.attachTo(_this._renderArea); });
        };
        Pie.prototype.computeLayout = function (origin, availableWidth, availableHeight) {
            _super.prototype.computeLayout.call(this, origin, availableWidth, availableHeight);
            var pieCenter = this._pieCenter();
            this._renderArea.attr("transform", "translate(" + pieCenter.x + "," + pieCenter.y + ")");
            var radiusLimit = Math.min(Math.max(this.width() - pieCenter.x, pieCenter.x), Math.max(this.height() - pieCenter.y, pieCenter.y));
            if (this.innerRadius().scale != null) {
                this.innerRadius().scale.range([0, radiusLimit]);
            }
            if (this.outerRadius().scale != null) {
                this.outerRadius().scale.range([0, radiusLimit]);
            }
            return this;
        };
        Pie.prototype.addDataset = function (dataset) {
            _super.prototype.addDataset.call(this, dataset);
            return this;
        };
        Pie.prototype._addDataset = function (dataset) {
            if (this.datasets().length === 1) {
                utils$1.Window.warn("Only one dataset is supported in Pie plots");
                return this;
            }
            this._updatePieAngles();
            _super.prototype._addDataset.call(this, dataset);
            var strokeDrawer = new arcOutlineDrawer.ArcOutlineSVGDrawer();
            if (this._isSetup) {
                strokeDrawer.attachTo(this._renderArea);
            }
            this._strokeDrawers.set(dataset, strokeDrawer);
            return this;
        };
        Pie.prototype.removeDataset = function (dataset) {
            _super.prototype.removeDataset.call(this, dataset);
            return this;
        };
        Pie.prototype._removeDatasetNodes = function (dataset) {
            _super.prototype._removeDatasetNodes.call(this, dataset);
            this._strokeDrawers.get(dataset).remove();
        };
        Pie.prototype._removeDataset = function (dataset) {
            _super.prototype._removeDataset.call(this, dataset);
            this._strokeDrawers.delete(dataset);
            this._startAngles = [];
            this._endAngles = [];
            return this;
        };
        Pie.prototype.selections = function (datasets) {
            var _this = this;
            if (datasets === void 0) { datasets = this.datasets(); }
            var allSelections = _super.prototype.selections.call(this, datasets).nodes();
            datasets.forEach(function (dataset) {
                var drawer = _this._strokeDrawers.get(dataset);
                if (drawer == null) {
                    return;
                }
                allSelections.push.apply(allSelections, drawer.getVisualPrimitives());
            });
            return d3.selectAll(allSelections);
        };
        Pie.prototype._onDatasetUpdate = function () {
            _super.prototype._onDatasetUpdate.call(this);
            this._updatePieAngles();
            this.render();
        };
        Pie.prototype._createDrawer = function () {
            return new drawer.ProxyDrawer(function () { return new arcDrawer.ArcSVGDrawer(); }, function () {
                windowUtils.warn("canvas renderer is not supported on Pie Plot!");
                return null;
            });
        };
        Pie.prototype.entities = function (datasets) {
            var _this = this;
            if (datasets === void 0) { datasets = this.datasets(); }
            var entities = _super.prototype.entities.call(this, datasets);
            return entities.map(function (entity) {
                entity.position.x += _this.width() / 2;
                entity.position.y += _this.height() / 2;
                var stroke = d3.select(_this._strokeDrawers.get(entity.dataset).getVisualPrimitiveAtIndex(entity.index));
                var piePlotEntity = entity;
                piePlotEntity.strokeSelection = stroke;
                return piePlotEntity;
            });
        };
        Pie.prototype.sectorValue = function (sectorValue, scale) {
            if (sectorValue == null) {
                return this._propertyBindings.get(Pie._SECTOR_VALUE_KEY);
            }
            this._bindProperty(Pie._SECTOR_VALUE_KEY, sectorValue, scale);
            this._updatePieAngles();
            this.render();
            return this;
        };
        Pie.prototype.innerRadius = function (innerRadius, scale) {
            if (innerRadius == null) {
                return this._propertyBindings.get(Pie._INNER_RADIUS_KEY);
            }
            this._bindProperty(Pie._INNER_RADIUS_KEY, innerRadius, scale);
            this.render();
            return this;
        };
        Pie.prototype.outerRadius = function (outerRadius, scale) {
            if (outerRadius == null) {
                return this._propertyBindings.get(Pie._OUTER_RADIUS_KEY);
            }
            this._bindProperty(Pie._OUTER_RADIUS_KEY, outerRadius, scale);
            this.render();
            return this;
        };
        Pie.prototype.startAngle = function (angle) {
            if (angle == null) {
                return this._startAngle;
            }
            else {
                this._startAngle = angle;
                this._updatePieAngles();
                this.render();
                return this;
            }
        };
        Pie.prototype.endAngle = function (angle) {
            if (angle == null) {
                return this._endAngle;
            }
            else {
                this._endAngle = angle;
                this._updatePieAngles();
                this.render();
                return this;
            }
        };
        Pie.prototype.labelsEnabled = function (enabled) {
            if (enabled == null) {
                return this._labelsEnabled;
            }
            else {
                this._labelsEnabled = enabled;
                this.render();
                return this;
            }
        };
        Pie.prototype.labelFormatter = function (formatter) {
            if (formatter == null) {
                return this._labelFormatter;
            }
            else {
                this._labelFormatter = formatter;
                this.render();
                return this;
            }
        };
        /*
         * Gets the Entities at a particular Point.
         *
         * @param {Point} p
         * @param {PlotEntity[]}
         */
        Pie.prototype.entitiesAt = function (queryPoint) {
            var center = { x: this.width() / 2, y: this.height() / 2 };
            var adjustedQueryPoint = { x: queryPoint.x - center.x, y: queryPoint.y - center.y };
            var index = this._sliceIndexForPoint(adjustedQueryPoint);
            return index == null ? [] : [this.entities()[index]];
        };
        Pie.prototype._propertyProjectors = function () {
            var _this = this;
            var attrToProjector = _super.prototype._propertyProjectors.call(this);
            var innerRadiusAccessor = plot.Plot._scaledAccessor(this.innerRadius());
            var outerRadiusAccessor = plot.Plot._scaledAccessor(this.outerRadius());
            attrToProjector["d"] = function (datum, index, ds) {
                return d3.arc().innerRadius(innerRadiusAccessor(datum, index, ds))
                    .outerRadius(outerRadiusAccessor(datum, index, ds))
                    .startAngle(_this._startAngles[index])
                    .endAngle(_this._endAngles[index])(datum, index);
            };
            return attrToProjector;
        };
        Pie.prototype._updatePieAngles = function () {
            if (this.sectorValue() == null) {
                return;
            }
            if (this.datasets().length === 0) {
                return;
            }
            var sectorValueAccessor = plot.Plot._scaledAccessor(this.sectorValue());
            var dataset = this.datasets()[0];
            var data = this._getDataToDraw().get(dataset);
            var pie = d3.pie().sort(null).startAngle(this._startAngle).endAngle(this._endAngle)
                .value(function (d, i) { return sectorValueAccessor(d, i, dataset); })(data);
            this._startAngles = pie.map(function (slice) { return slice.startAngle; });
            this._endAngles = pie.map(function (slice) { return slice.endAngle; });
        };
        Pie.prototype._pieCenter = function () {
            var a = this._startAngle < this._endAngle ? this._startAngle : this._endAngle;
            var b = this._startAngle < this._endAngle ? this._endAngle : this._startAngle;
            var sinA = Math.sin(a);
            var cosA = Math.cos(a);
            var sinB = Math.sin(b);
            var cosB = Math.cos(b);
            var hTop;
            var hBottom;
            var wRight;
            var wLeft;
            /**
             *  The center of the pie is computed using the sine and cosine of the start angle and the end angle
             *  The sine indicates whether the start and end fall on the right half or the left half of the pie
             *  The cosine indicates whether the start and end fall on the top or the bottom half of the pie
             *  Different combinations provide the different heights and widths the pie needs from the center to the sides
             */
            if (sinA >= 0 && sinB >= 0) {
                if (cosA >= 0 && cosB >= 0) {
                    hTop = cosA;
                    hBottom = 0;
                    wLeft = 0;
                    wRight = sinB;
                }
                else if (cosA < 0 && cosB < 0) {
                    hTop = 0;
                    hBottom = -cosB;
                    wLeft = 0;
                    wRight = sinA;
                }
                else if (cosA >= 0 && cosB < 0) {
                    hTop = cosA;
                    hBottom = -cosB;
                    wLeft = 0;
                    wRight = sinA;
                }
                else if (cosA < 0 && cosB >= 0) {
                    hTop = 1;
                    hBottom = 1;
                    wLeft = 1;
                    wRight = Math.max(sinA, sinB);
                }
            }
            else if (sinA >= 0 && sinB < 0) {
                if (cosA >= 0 && cosB >= 0) {
                    hTop = Math.max(cosA, cosB);
                    hBottom = 1;
                    wLeft = 1;
                    wRight = 1;
                }
                else if (cosA < 0 && cosB < 0) {
                    hTop = 0;
                    hBottom = 1;
                    wLeft = -sinB;
                    wRight = sinA;
                }
                else if (cosA >= 0 && cosB < 0) {
                    hTop = cosA;
                    hBottom = 1;
                    wLeft = -sinB;
                    wRight = 1;
                }
                else if (cosA < 0 && cosB >= 0) {
                    hTop = cosB;
                    hBottom = 1;
                    wLeft = 1;
                    wRight = sinA;
                }
            }
            else if (sinA < 0 && sinB >= 0) {
                if (cosA >= 0 && cosB >= 0) {
                    hTop = 1;
                    hBottom = 0;
                    wLeft = -sinA;
                    wRight = sinB;
                }
                else if (cosA < 0 && cosB < 0) {
                    hTop = 1;
                    hBottom = Math.max(-cosA, -cosB);
                    wLeft = 1;
                    wRight = 1;
                }
                else if (cosA >= 0 && cosB < 0) {
                    hTop = 1;
                    hBottom = -cosB;
                    wLeft = -sinA;
                    wRight = 1;
                }
                else if (cosA < 0 && cosB >= 0) {
                    hTop = 1;
                    hBottom = -cosA;
                    wLeft = 1;
                    wRight = sinB;
                }
            }
            else if (sinA < 0 && sinB < 0) {
                if (cosA >= 0 && cosB >= 0) {
                    hTop = cosB;
                    hBottom = 0;
                    wLeft = -sinA;
                    wRight = 0;
                }
                else if (cosA < 0 && cosB < 0) {
                    hTop = 0;
                    hBottom = -cosA;
                    wLeft = -sinB;
                    wRight = 0;
                }
                else if (cosA >= 0 && cosB < 0) {
                    hTop = 1;
                    hBottom = 1;
                    wLeft = Math.max(cosA, -cosB);
                    wRight = 1;
                }
                else if (cosA < 0 && cosB >= 0) {
                    hTop = cosB;
                    hBottom = -cosA;
                    wLeft = 1;
                    wRight = 0;
                }
            }
            return {
                x: wLeft + wRight == 0 ? 0 : (wLeft / (wLeft + wRight)) * this.width(),
                y: hTop + hBottom == 0 ? 0 : (hTop / (hTop + hBottom)) * this.height(),
            };
        };
        Pie.prototype._getDataToDraw = function () {
            var dataToDraw = _super.prototype._getDataToDraw.call(this);
            if (this.datasets().length === 0) {
                return dataToDraw;
            }
            var sectorValueAccessor = plot.Plot._scaledAccessor(this.sectorValue());
            var ds = this.datasets()[0];
            var data = dataToDraw.get(ds);
            var filteredData = data.filter(function (d, i) { return Pie._isValidData(sectorValueAccessor(d, i, ds)); });
            dataToDraw.set(ds, filteredData);
            return dataToDraw;
        };
        Pie._isValidData = function (value) {
            return utils$1.Math.isValidNumber(value) && value >= 0;
        };
        Pie.prototype._pixelPoint = function (datum, index, dataset) {
            var scaledValueAccessor = plot.Plot._scaledAccessor(this.sectorValue());
            if (!Pie._isValidData(scaledValueAccessor(datum, index, dataset))) {
                return { x: NaN, y: NaN };
            }
            var innerRadius = plot.Plot._scaledAccessor(this.innerRadius())(datum, index, dataset);
            var outerRadius = plot.Plot._scaledAccessor(this.outerRadius())(datum, index, dataset);
            var avgRadius = (innerRadius + outerRadius) / 2;
            var pie = d3.pie()
                .sort(null)
                .value(function (d, i) {
                var value = scaledValueAccessor(d, i, dataset);
                return Pie._isValidData(value) ? value : 0;
            }).startAngle(this._startAngle).endAngle(this._endAngle)(dataset.data());
            var startAngle = pie[index].startAngle;
            var endAngle = pie[index].endAngle;
            var avgAngle = (startAngle + endAngle) / 2;
            return { x: avgRadius * Math.sin(avgAngle), y: -avgRadius * Math.cos(avgAngle) };
        };
        Pie.prototype._additionalPaint = function (time) {
            var _this = this;
            this._renderArea.select(".label-area").remove();
            if (this._labelsEnabled) {
                utils$1.Window.setTimeout(function () { return _this._drawLabels(); }, time);
            }
            var drawSteps = this._generateStrokeDrawSteps();
            var dataToDraw = this._getDataToDraw();
            this.datasets().forEach(function (dataset) {
                var appliedDrawSteps = plot.Plot.applyDrawSteps(drawSteps, dataset);
                _this._strokeDrawers.get(dataset).draw(dataToDraw.get(dataset), appliedDrawSteps);
            });
        };
        Pie.prototype._generateStrokeDrawSteps = function () {
            var attrToProjector = this._getAttrToProjector();
            return [{ attrToProjector: attrToProjector, animator: new animators.Null() }];
        };
        Pie.prototype._sliceIndexForPoint = function (p) {
            var pointRadius = Math.sqrt(Math.pow(p.x, 2) + Math.pow(p.y, 2));
            var pointAngle = Math.acos(-p.y / pointRadius);
            if (p.x < 0) {
                pointAngle = Math.PI * 2 - pointAngle;
            }
            var index;
            for (var i = 0; i < this._startAngles.length; i++) {
                if (this._startAngles[i] < pointAngle && this._endAngles[i] > pointAngle) {
                    index = i;
                    break;
                }
            }
            if (index !== undefined) {
                var dataset = this.datasets()[0];
                var datum = dataset.data()[index];
                var innerRadius = this.innerRadius().accessor(datum, index, dataset);
                var outerRadius = this.outerRadius().accessor(datum, index, dataset);
                if (pointRadius > innerRadius && pointRadius < outerRadius) {
                    return index;
                }
            }
            return null;
        };
        Pie.prototype._drawLabels = function () {
            var _this = this;
            var attrToProjector = this._getAttrToProjector();
            var labelArea = this._renderArea.append("g").classed("label-area", true);
            var context = new src$1.SvgContext(labelArea.node());
            var measurer = new src$1.CacheMeasurer(context);
            var writer = new src$1.Writer(measurer, context);
            var dataset = this.datasets()[0];
            var data = this._getDataToDraw().get(dataset);
            var dataLen = data.length;
            var _loop_1 = function (datumIndex) {
                var datum = data[datumIndex];
                var value = this_1.sectorValue().accessor(datum, datumIndex, dataset);
                if (!utils$1.Math.isValidNumber(value)) {
                    return "continue";
                }
                value = this_1._labelFormatter(value, datum, datumIndex, dataset);
                var measurement = measurer.measure(value);
                var theta = (this_1._endAngles[datumIndex] + this_1._startAngles[datumIndex]) / 2;
                var outerRadius = this_1.outerRadius().accessor(datum, datumIndex, dataset);
                if (this_1.outerRadius().scale) {
                    outerRadius = this_1.outerRadius().scale.scale(outerRadius);
                }
                var innerRadius = this_1.innerRadius().accessor(datum, datumIndex, dataset);
                if (this_1.innerRadius().scale) {
                    innerRadius = this_1.innerRadius().scale.scale(innerRadius);
                }
                var labelRadius = (outerRadius + innerRadius) / 2;
                var x = Math.sin(theta) * labelRadius - measurement.width / 2;
                var y = -Math.cos(theta) * labelRadius - measurement.height / 2;
                var corners = [
                    { x: x, y: y },
                    { x: x, y: y + measurement.height },
                    { x: x + measurement.width, y: y },
                    { x: x + measurement.width, y: y + measurement.height },
                ];
                var showLabel = corners.every(function (corner) {
                    return Math.abs(corner.x) <= _this.width() / 2 && Math.abs(corner.y) <= _this.height() / 2;
                });
                if (showLabel) {
                    var sliceIndices = corners.map(function (corner) { return _this._sliceIndexForPoint(corner); });
                    showLabel = sliceIndices.every(function (index) { return index === datumIndex; });
                }
                var color = attrToProjector["fill"](datum, datumIndex, dataset);
                var dark = utils$1.Color.contrast("white", color) * 1.6 < utils$1.Color.contrast("black", color);
                var g = labelArea.append("g").attr("transform", "translate(" + x + "," + y + ")");
                var className = dark ? "dark-label" : "light-label";
                g.classed(className, true);
                g.style("visibility", showLabel ? "inherit" : "hidden");
                writer.write(value, measurement.width, measurement.height, {
                    xAlign: "center",
                    yAlign: "center",
                }, g.node());
            };
            var this_1 = this;
            for (var datumIndex = 0; datumIndex < dataLen; datumIndex++) {
                _loop_1(datumIndex);
            }
        };
        Pie._INNER_RADIUS_KEY = "inner-radius";
        Pie._OUTER_RADIUS_KEY = "outer-radius";
        Pie._SECTOR_VALUE_KEY = "sector-value";
        return Pie;
    }(plot.Plot));
    exports.Pie = Pie;
    });

    var rectanglePlot = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });











    var Rectangle = /** @class */ (function (_super) {
        tslib_es6.__extends(Rectangle, _super);
        /**
         * A Rectangle Plot displays rectangles based on the data.
         * The left and right edges of each rectangle can be set with x() and x2().
         *   If only x() is set the Rectangle Plot will attempt to compute the correct left and right edge positions.
         * The top and bottom edges of each rectangle can be set with y() and y2().
         *   If only y() is set the Rectangle Plot will attempt to compute the correct top and bottom edge positions.
         *
         * @constructor
         * @param {Scale.Scale} xScale
         * @param {Scale.Scale} yScale
         */
        function Rectangle() {
            var _this = _super.call(this) || this;
            _this._labelsEnabled = false;
            _this._label = null;
            _this.animator("rectangles", new animators.Null());
            _this.addClass("rectangle-plot");
            _this.attr("fill", new scales.Color().range()[0]);
            return _this;
        }
        Rectangle.prototype._createDrawer = function () {
            return new drawer.ProxyDrawer(function () { return new rectangleDrawer.RectangleSVGDrawer(); }, function (ctx) { return new drawers.RectangleCanvasDrawer(ctx); });
        };
        Rectangle.prototype._generateAttrToProjector = function () {
            var _this = this;
            var attrToProjector = _super.prototype._generateAttrToProjector.call(this);
            // Copy each of the different projectors.
            var xAttr = plot.Plot._scaledAccessor(this.x());
            var x2Attr = attrToProjector[Rectangle._X2_KEY];
            var yAttr = plot.Plot._scaledAccessor(this.y());
            var y2Attr = attrToProjector[Rectangle._Y2_KEY];
            var xScale = this.x().scale;
            var yScale = this.y().scale;
            if (x2Attr != null) {
                attrToProjector["width"] = function (d, i, dataset) { return Math.abs(x2Attr(d, i, dataset) - xAttr(d, i, dataset)); };
                attrToProjector["x"] = function (d, i, dataset) { return Math.min(x2Attr(d, i, dataset), xAttr(d, i, dataset)); };
            }
            else {
                attrToProjector["width"] = function (d, i, dataset) { return _this._rectangleWidth(xScale); };
                attrToProjector["x"] = function (d, i, dataset) { return xAttr(d, i, dataset) - 0.5 * attrToProjector["width"](d, i, dataset); };
            }
            if (y2Attr != null) {
                attrToProjector["height"] = function (d, i, dataset) { return Math.abs(y2Attr(d, i, dataset) - yAttr(d, i, dataset)); };
                attrToProjector["y"] = function (d, i, dataset) {
                    return Math.max(y2Attr(d, i, dataset), yAttr(d, i, dataset)) - attrToProjector["height"](d, i, dataset);
                };
            }
            else {
                attrToProjector["height"] = function (d, i, dataset) { return _this._rectangleWidth(yScale); };
                attrToProjector["y"] = function (d, i, dataset) { return yAttr(d, i, dataset) - 0.5 * attrToProjector["height"](d, i, dataset); };
            }
            // Clean up the attributes projected onto the SVG elements
            delete attrToProjector[Rectangle._X2_KEY];
            delete attrToProjector[Rectangle._Y2_KEY];
            return attrToProjector;
        };
        Rectangle.prototype._generateDrawSteps = function () {
            return [{ attrToProjector: this._getAttrToProjector(), animator: this._getAnimator("rectangles") }];
        };
        Rectangle.prototype._filterForProperty = function (property) {
            if (property === "x2") {
                return _super.prototype._filterForProperty.call(this, "x");
            }
            else if (property === "y2") {
                return _super.prototype._filterForProperty.call(this, "y");
            }
            return _super.prototype._filterForProperty.call(this, property);
        };
        Rectangle.prototype.x = function (x, xScale, postScale) {
            if (x == null) {
                return _super.prototype.x.call(this);
            }
            if (xScale == null) {
                _super.prototype.x.call(this, x);
            }
            else {
                _super.prototype.x.call(this, x, xScale, postScale);
            }
            if (xScale != null) {
                var x2Binding = this.x2();
                var x2 = x2Binding && x2Binding.accessor;
                if (x2 != null) {
                    this._bindProperty(Rectangle._X2_KEY, x2, xScale, x2Binding.postScale);
                }
            }
            // The x and y scales should render in bands with no padding for category scales
            if (xScale instanceof scales.Category) {
                xScale.innerPadding(0).outerPadding(0);
            }
            return this;
        };
        Rectangle.prototype.x2 = function (x2, postScale) {
            if (x2 == null) {
                return this._propertyBindings.get(Rectangle._X2_KEY);
            }
            var xBinding = this.x();
            var xScale = xBinding && xBinding.scale;
            this._bindProperty(Rectangle._X2_KEY, x2, xScale, postScale);
            this.render();
            return this;
        };
        Rectangle.prototype.y = function (y, yScale, postScale) {
            if (y == null) {
                return _super.prototype.y.call(this);
            }
            if (yScale == null) {
                _super.prototype.y.call(this, y);
            }
            else {
                _super.prototype.y.call(this, y, yScale, postScale);
            }
            if (yScale != null) {
                var y2Binding = this.y2();
                var y2 = y2Binding && y2Binding.accessor;
                if (y2 != null) {
                    this._bindProperty(Rectangle._Y2_KEY, y2, yScale, y2Binding.postScale);
                }
            }
            // The x and y scales should render in bands with no padding for category scales
            if (yScale instanceof scales.Category) {
                yScale.innerPadding(0).outerPadding(0);
            }
            return this;
        };
        Rectangle.prototype.y2 = function (y2, postScale) {
            if (y2 == null) {
                return this._propertyBindings.get(Rectangle._Y2_KEY);
            }
            var yBinding = this.y();
            var yScale = yBinding && yBinding.scale;
            this._bindProperty(Rectangle._Y2_KEY, y2, yScale, postScale);
            this.render();
            return this;
        };
        /**
         * Gets the PlotEntities at a particular Point.
         *
         * @param {Point} point The point to query.
         * @returns {PlotEntity[]} The PlotEntities at the particular point
         */
        Rectangle.prototype.entitiesAt = function (point) {
            var attrToProjector = this._getAttrToProjector();
            return this.entities().filter(function (entity) {
                var datum = entity.datum;
                var index = entity.index;
                var dataset = entity.dataset;
                var x = attrToProjector["x"](datum, index, dataset);
                var y = attrToProjector["y"](datum, index, dataset);
                var width = attrToProjector["width"](datum, index, dataset);
                var height = attrToProjector["height"](datum, index, dataset);
                return x <= point.x && point.x <= x + width && y <= point.y && point.y <= y + height;
            });
        };
        Rectangle.prototype._entityBounds = function (entity) {
            var datum = entity.datum, index = entity.index, dataset = entity.dataset;
            return this._entityBBox(datum, index, dataset, this._getAttrToProjector());
        };
        Rectangle.prototype._entityBBox = function (datum, index, dataset, attrToProjector) {
            return {
                x: attrToProjector["x"](datum, index, dataset),
                y: attrToProjector["y"](datum, index, dataset),
                width: attrToProjector["width"](datum, index, dataset),
                height: attrToProjector["height"](datum, index, dataset),
            };
        };
        Rectangle.prototype.label = function (label) {
            if (label == null) {
                return this._label;
            }
            this._label = label;
            this.render();
            return this;
        };
        Rectangle.prototype.labelsEnabled = function (enabled) {
            if (enabled == null) {
                return this._labelsEnabled;
            }
            else {
                this._labelsEnabled = enabled;
                this.render();
                return this;
            }
        };
        Rectangle.prototype._propertyProjectors = function () {
            var attrToProjector = _super.prototype._propertyProjectors.call(this);
            if (this.x2() != null) {
                attrToProjector["x2"] = plot.Plot._scaledAccessor(this.x2());
            }
            if (this.y2() != null) {
                attrToProjector["y2"] = plot.Plot._scaledAccessor(this.y2());
            }
            return attrToProjector;
        };
        Rectangle.prototype._pixelPoint = function (datum, index, dataset) {
            var attrToProjector = this._getAttrToProjector();
            var rectX = attrToProjector["x"](datum, index, dataset);
            var rectY = attrToProjector["y"](datum, index, dataset);
            var rectWidth = attrToProjector["width"](datum, index, dataset);
            var rectHeight = attrToProjector["height"](datum, index, dataset);
            var x = rectX + rectWidth / 2;
            var y = rectY + rectHeight / 2;
            return { x: x, y: y };
        };
        Rectangle.prototype._rectangleWidth = function (scale) {
            if (scale instanceof scales.Category) {
                return scale.rangeBand();
            }
            else {
                var accessor_1 = scale === this.x().scale ? this.x().accessor : this.y().accessor;
                var accessorData = d3.set(utils$1.Array.flatten(this.datasets().map(function (dataset) {
                    return dataset.data().map(function (d, i) { return accessor_1(d, i, dataset).valueOf(); });
                }))).values().map(function (value) { return +value; });
                // Get the absolute difference between min and max
                var min = utils$1.Math.min(accessorData, 0);
                var max = utils$1.Math.max(accessorData, 0);
                var scaledMin = scale.scale(min);
                var scaledMax = scale.scale(max);
                return (scaledMax - scaledMin) / Math.abs(max - min);
            }
        };
        Rectangle.prototype._getDataToDraw = function () {
            var dataToDraw = new utils$1.Map();
            var attrToProjector = this._getAttrToProjector();
            this.datasets().forEach(function (dataset) {
                var data = dataset.data().map(function (d, i) {
                    var isValid = (utils$1.Math.isValidNumber(attrToProjector["x"](d, i, dataset)) &&
                        utils$1.Math.isValidNumber(attrToProjector["y"](d, i, dataset)) &&
                        utils$1.Math.isValidNumber(attrToProjector["width"](d, i, dataset)) &&
                        utils$1.Math.isValidNumber(attrToProjector["height"](d, i, dataset)));
                    return isValid ? d : null;
                });
                dataToDraw.set(dataset, data);
            });
            return dataToDraw;
        };
        Rectangle.prototype._additionalPaint = function (time) {
            var _this = this;
            this._renderArea.selectAll(".label-area").remove();
            if (this._labelsEnabled && this.label() != null) {
                utils$1.Window.setTimeout(function () { return _this._drawLabels(); }, time);
            }
        };
        Rectangle.prototype._drawLabels = function () {
            var _this = this;
            var dataToDraw = this._getDataToDraw();
            this.datasets().forEach(function (dataset, i) { return _this._drawLabel(dataToDraw, dataset, i); });
        };
        Rectangle.prototype._drawLabel = function (dataToDraw, dataset, datasetIndex) {
            var attrToProjector = this._getAttrToProjector();
            var labelArea = this._renderArea.append("g").classed("label-area", true);
            var context = new src$1.SvgContext(labelArea.node());
            var measurer = new src$1.CacheMeasurer(context);
            var writer = new src$1.Writer(measurer, context);
            var xRange = this.x().scale.range();
            var yRange = this.y().scale.range();
            var xMin = Math.min.apply(null, xRange);
            var xMax = Math.max.apply(null, xRange);
            var yMin = Math.min.apply(null, yRange);
            var yMax = Math.max.apply(null, yRange);
            var data = dataToDraw.get(dataset);
            var dataLength = data.length;
            for (var datumIndex = 0; datumIndex < dataLength; datumIndex++) {
                var datum = data[datumIndex];
                if (datum == null) {
                    continue;
                }
                var label = "" + this.label()(datum, datumIndex, dataset);
                var measurement = measurer.measure(label);
                var x = attrToProjector["x"](datum, datumIndex, dataset);
                var y = attrToProjector["y"](datum, datumIndex, dataset);
                var width = attrToProjector["width"](datum, datumIndex, dataset);
                var height = attrToProjector["height"](datum, datumIndex, dataset);
                if (measurement.height <= height && measurement.width <= width) {
                    var horizontalOffset = (width - measurement.width) / 2;
                    var verticalOffset = (height - measurement.height) / 2;
                    x += horizontalOffset;
                    y += verticalOffset;
                    var xLabelRange = { min: x, max: x + measurement.width };
                    var yLabelRange = { min: y, max: y + measurement.height };
                    if (xLabelRange.min < xMin || xLabelRange.max > xMax || yLabelRange.min < yMin || yLabelRange.max > yMax) {
                        continue;
                    }
                    if (this._overlayLabel(xLabelRange, yLabelRange, datumIndex, datasetIndex, dataToDraw)) {
                        continue;
                    }
                    var color = attrToProjector["fill"](datum, datumIndex, dataset);
                    var dark = utils$1.Color.contrast("white", color) * 1.6 < utils$1.Color.contrast("black", color);
                    var g = labelArea.append("g").attr("transform", "translate(" + x + "," + y + ")");
                    var className = dark ? "dark-label" : "light-label";
                    g.classed(className, true);
                    writer.write(label, measurement.width, measurement.height, {
                        xAlign: "center",
                        yAlign: "center",
                    }, g.node());
                }
            }
        };
        Rectangle.prototype._overlayLabel = function (labelXRange, labelYRange, datumIndex, datasetIndex, dataToDraw) {
            var attrToProjector = this._getAttrToProjector();
            var datasets = this.datasets();
            for (var i = datasetIndex; i < datasets.length; i++) {
                var dataset = datasets[i];
                var data = dataToDraw.get(dataset);
                var dataLen = data.length;
                for (var j = (i === datasetIndex ? datumIndex + 1 : 0); j < dataLen; j++) {
                    if (utils$1.DOM.intersectsBBox(labelXRange, labelYRange, this._entityBBox(data[j], j, dataset, attrToProjector))) {
                        return true;
                    }
                }
            }
            return false;
        };
        Rectangle._X2_KEY = "x2";
        Rectangle._Y2_KEY = "y2";
        return Rectangle;
    }(xyPlot.XYPlot));
    exports.Rectangle = Rectangle;
    });

    var scatterPlot = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });













    var Scatter = /** @class */ (function (_super) {
        tslib_es6.__extends(Scatter, _super);
        /**
         * A Scatter Plot draws a symbol at each data point.
         *
         * @constructor
         */
        function Scatter() {
            var _this = _super.call(this) || this;
            _this._labelFormatter = formatters.identity();
            _this._labelsEnabled = false;
            _this.addClass("scatter-plot");
            var animator = new animators.Easing();
            animator.startDelay(5);
            animator.stepDuration(250);
            animator.maxTotalDuration(plot.Plot._ANIMATION_MAX_DURATION);
            _this.animator(plots.Animator.MAIN, animator);
            _this.attr("opacity", 0.6);
            _this.attr("fill", new scales.Color().range()[0]);
            _this.size(6);
            var circleSymbolFactory = symbolFactories.circle();
            _this.symbol(function () { return circleSymbolFactory; });
            _this._labelConfig = new utils$1.Map();
            return _this;
        }
        Scatter.prototype._buildLightweightPlotEntities = function (datasets) {
            var _this = this;
            var lightweightPlotEntities = _super.prototype._buildLightweightPlotEntities.call(this, datasets);
            return lightweightPlotEntities.map(function (lightweightPlotEntity) {
                var diameter = plot.Plot._scaledAccessor(_this.size())(lightweightPlotEntity.datum, lightweightPlotEntity.index, lightweightPlotEntity.dataset);
                lightweightPlotEntity.diameter = diameter;
                return lightweightPlotEntity;
            });
        };
        Scatter.prototype._createDrawer = function (dataset) {
            var _this = this;
            return new drawer.ProxyDrawer(function () { return new symbolDrawer.SymbolSVGDrawer(); }, function (ctx) {
                return new drawers.CanvasDrawer(ctx, symbolDrawer.makeSymbolCanvasDrawStep(dataset, function () { return plot.Plot._scaledAccessor(_this.symbol()); }, function () { return plot.Plot._scaledAccessor(_this.size()); }));
            });
        };
        Scatter.prototype.size = function (size, scale) {
            if (size == null) {
                return this._propertyBindings.get(Scatter._SIZE_KEY);
            }
            this._bindProperty(Scatter._SIZE_KEY, size, scale);
            this.render();
            return this;
        };
        Scatter.prototype.symbol = function (symbol) {
            if (symbol == null) {
                return this._propertyBindings.get(Scatter._SYMBOL_KEY);
            }
            this._propertyBindings.set(Scatter._SYMBOL_KEY, { accessor: symbol });
            this.render();
            return this;
        };
        Scatter.prototype._generateDrawSteps = function () {
            var drawSteps = [];
            if (this._animateOnNextRender()) {
                var attrToProjector = this._getAttrToProjector();
                var symbolProjector_1 = plot.Plot._scaledAccessor(this.symbol());
                attrToProjector["d"] = function (datum, index, dataset) { return symbolProjector_1(datum, index, dataset)(0)(null); };
                drawSteps.push({ attrToProjector: attrToProjector, animator: this._getAnimator(plots.Animator.RESET) });
            }
            drawSteps.push({
                attrToProjector: this._getAttrToProjector(),
                animator: this._getAnimator(plots.Animator.MAIN),
            });
            return drawSteps;
        };
        Scatter.prototype._propertyProjectors = function () {
            var propertyToProjectors = _super.prototype._propertyProjectors.call(this);
            var xProjector = plot.Plot._scaledAccessor(this.x());
            var yProjector = plot.Plot._scaledAccessor(this.y());
            propertyToProjectors["x"] = xProjector;
            propertyToProjectors["y"] = yProjector;
            propertyToProjectors["transform"] = function (datum, index, dataset) {
                return "translate(" + xProjector(datum, index, dataset) + "," + yProjector(datum, index, dataset) + ")";
            };
            propertyToProjectors["d"] = this._constructSymbolGenerator();
            return propertyToProjectors;
        };
        Scatter.prototype._constructSymbolGenerator = function () {
            var symbolProjector = plot.Plot._scaledAccessor(this.symbol());
            var sizeProjector = plot.Plot._scaledAccessor(this.size());
            return function (datum, index, dataset) {
                return symbolProjector(datum, index, dataset)(sizeProjector(datum, index, dataset))(null);
            };
        };
        Scatter.prototype._entityBounds = function (entity) {
            return {
                x: entity.position.x - entity.diameter / 2,
                y: entity.position.y - entity.diameter / 2,
                width: entity.diameter,
                height: entity.diameter,
            };
        };
        Scatter.prototype._entityVisibleOnPlot = function (entity, bounds) {
            var xRange = { min: bounds.topLeft.x, max: bounds.bottomRight.x };
            var yRange = { min: bounds.topLeft.y, max: bounds.bottomRight.y };
            var translatedBbox = this._entityBounds(entity);
            return utils$1.DOM.intersectsBBox(xRange, yRange, translatedBbox);
        };
        /**
         * Gets the Entities at a particular Point.
         *
         * @param {Point} p
         * @returns {PlotEntity[]}
         */
        Scatter.prototype.entitiesAt = function (p) {
            var xProjector = plot.Plot._scaledAccessor(this.x());
            var yProjector = plot.Plot._scaledAccessor(this.y());
            var sizeProjector = plot.Plot._scaledAccessor(this.size());
            return this.entities().filter(function (entity) {
                var datum = entity.datum;
                var index = entity.index;
                var dataset = entity.dataset;
                var x = xProjector(datum, index, dataset);
                var y = yProjector(datum, index, dataset);
                var size = sizeProjector(datum, index, dataset);
                return x - size / 2 <= p.x && p.x <= x + size / 2 && y - size / 2 <= p.y && p.y <= y + size / 2;
            });
        };
        Scatter.prototype.labelsEnabled = function (enabled) {
            if (enabled == null) {
                return this._labelsEnabled;
            }
            else {
                this._labelsEnabled = enabled;
                this._clearAttrToProjectorCache();
                this.render();
                return this;
            }
        };
        Scatter.prototype._createNodesForDataset = function (dataset) {
            var drawer = _super.prototype._createNodesForDataset.call(this, dataset);
            var labelArea = this._renderArea.append("g").classed(Scatter._LABEL_AREA_CLASS, true);
            var context = new src$1.SvgContext(labelArea.node());
            var measurer = new src$1.CacheMeasurer(context);
            var writer = new src$1.Writer(measurer, context);
            this._labelConfig.set(dataset, { labelArea: labelArea, measurer: measurer, writer: writer });
            return drawer;
        };
        Scatter.prototype._removeDatasetNodes = function (dataset) {
            _super.prototype._removeDatasetNodes.call(this, dataset);
            var labelConfig = this._labelConfig.get(dataset);
            if (labelConfig != null) {
                labelConfig.labelArea.remove();
                this._labelConfig.delete(dataset);
            }
        };
        Scatter.prototype._additionalPaint = function (time) {
            var _this = this;
            this.datasets().forEach(function (dataset) { return _this._labelConfig.get(dataset).labelArea.selectAll("g").remove(); });
            if (this._labelsEnabled) {
                utils$1.Window.setTimeout(function () { return _this._drawLabels(); }, time);
            }
        };
        Scatter.prototype._drawLabels = function () {
            var _this = this;
            var dataToDraw = this._getDataToDraw();
            var attrToProjector = this._getAttrToProjector();
            this.datasets().forEach(function (dataset) {
                var data = dataToDraw.get(dataset);
                var dataLen = data.length;
                for (var index = 0; index < dataLen; index++) {
                    var datum = data[index];
                    if (datum == null) {
                        continue;
                    }
                    _this._drawLabel(datum, index, dataset, attrToProjector);
                }
            });
        };
        Scatter.prototype._drawLabel = function (datum, index, dataset, attrToProjector) {
            if (datum.label == null) {
                return;
            }
            var _a = this._labelConfig.get(dataset), labelArea = _a.labelArea, measurer = _a.measurer, writer = _a.writer;
            var scatterCoordinates = { x: attrToProjector["x"](datum, index, dataset), y: attrToProjector["y"](datum, index, dataset) };
            var sizeProjector = plot.Plot._scaledAccessor(this.size());
            var diameter = sizeProjector(datum, index, dataset);
            var label = this._labelFormatter(datum.label, datum, index, dataset);
            var measurement = measurer.measure(label);
            var _b = this._calculateLabelProperties(scatterCoordinates, diameter, measurement), containerDimensions = _b.containerDimensions, labelContainerOrigin = _b.labelContainerOrigin, labelOrigin = _b.labelOrigin, alignment = _b.alignment;
            var labelContainer = this._createLabelContainer(labelArea, labelContainerOrigin, labelOrigin, measurement);
            var writeOptions = { xAlign: alignment.x, yAlign: alignment.y };
            writer.write(label, containerDimensions.width, containerDimensions.height, writeOptions, labelContainer.node());
        };
        Scatter.prototype._calculateLabelProperties = function (pointCoordinates, diameter, measurement) {
            // If diameter is smaller than font size, put label above
            var labelShift = diameter < measurement.height ? diameter / 2 + Scatter._LABEL_MARGIN_FROM_BUBBLE : 0;
            return {
                containerDimensions: {
                    width: measurement.width,
                    height: measurement.height,
                },
                labelContainerOrigin: {
                    x: pointCoordinates.x - measurement.width / 2,
                    y: pointCoordinates.y - measurement.height / 2 + labelShift,
                },
                labelOrigin: {
                    x: pointCoordinates.x,
                    y: pointCoordinates.y,
                },
                alignment: {
                    x: "center",
                    y: "center",
                },
            };
        };
        Scatter.prototype._createLabelContainer = function (labelArea, labelContainerOrigin, labelOrigin, measurement) {
            var labelContainer = labelArea.append("g")
                .attr("transform", "translate(" + labelContainerOrigin.x + ", " + labelContainerOrigin.y + ")");
            labelContainer.classed("on-bar-label", true);
            return labelContainer;
        };
        Scatter._SIZE_KEY = "size";
        Scatter._SYMBOL_KEY = "symbol";
        // label stuff
        Scatter._LABEL_AREA_CLASS = "scatter-label-text-area";
        Scatter._LABEL_MARGIN_FROM_BUBBLE = 15;
        return Scatter;
    }(xyPlot.XYPlot));
    exports.Scatter = Scatter;
    });

    var segmentPlot = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });








    var Segment = /** @class */ (function (_super) {
        tslib_es6.__extends(Segment, _super);
        /**
         * A Segment Plot displays line segments based on the data.
         *
         * @constructor
         */
        function Segment() {
            var _this = _super.call(this) || this;
            _this.addClass("segment-plot");
            _this.attr("stroke", new scales.Color().range()[0]);
            _this.attr("stroke-width", "2px");
            return _this;
        }
        Segment.prototype._createDrawer = function () {
            return new drawer.ProxyDrawer(function () { return new segmentDrawer.SegmentSVGDrawer(); }, function () {
                windowUtils.warn("canvas renderer is not supported on Segment Plot!");
                return null;
            });
        };
        Segment.prototype._generateDrawSteps = function () {
            return [{ attrToProjector: this._getAttrToProjector(), animator: new animators.Null() }];
        };
        Segment.prototype._filterForProperty = function (property) {
            if (property === "x2") {
                return _super.prototype._filterForProperty.call(this, "x");
            }
            else if (property === "y2") {
                return _super.prototype._filterForProperty.call(this, "y");
            }
            return _super.prototype._filterForProperty.call(this, property);
        };
        Segment.prototype.x = function (x, xScale) {
            if (x == null) {
                return _super.prototype.x.call(this);
            }
            if (xScale == null) {
                _super.prototype.x.call(this, x);
            }
            else {
                _super.prototype.x.call(this, x, xScale);
                var x2Binding = this.x2();
                var x2 = x2Binding && x2Binding.accessor;
                if (x2 != null) {
                    this._bindProperty(Segment._X2_KEY, x2, xScale);
                }
            }
            return this;
        };
        Segment.prototype.x2 = function (x2) {
            if (x2 == null) {
                return this._propertyBindings.get(Segment._X2_KEY);
            }
            var xBinding = this.x();
            var xScale = xBinding && xBinding.scale;
            this._bindProperty(Segment._X2_KEY, x2, xScale);
            this.render();
            return this;
        };
        Segment.prototype.y = function (y, yScale) {
            if (y == null) {
                return _super.prototype.y.call(this);
            }
            if (yScale == null) {
                _super.prototype.y.call(this, y);
            }
            else {
                _super.prototype.y.call(this, y, yScale);
                var y2Binding = this.y2();
                var y2 = y2Binding && y2Binding.accessor;
                if (y2 != null) {
                    this._bindProperty(Segment._Y2_KEY, y2, yScale);
                }
            }
            return this;
        };
        Segment.prototype.y2 = function (y2) {
            if (y2 == null) {
                return this._propertyBindings.get(Segment._Y2_KEY);
            }
            var yBinding = this.y();
            var yScale = yBinding && yBinding.scale;
            this._bindProperty(Segment._Y2_KEY, y2, yScale);
            this.render();
            return this;
        };
        Segment.prototype._propertyProjectors = function () {
            var attrToProjector = _super.prototype._propertyProjectors.call(this);
            attrToProjector["x1"] = plot.Plot._scaledAccessor(this.x());
            attrToProjector["x2"] = this.x2() == null ? plot.Plot._scaledAccessor(this.x()) : plot.Plot._scaledAccessor(this.x2());
            attrToProjector["y1"] = plot.Plot._scaledAccessor(this.y());
            attrToProjector["y2"] = this.y2() == null ? plot.Plot._scaledAccessor(this.y()) : plot.Plot._scaledAccessor(this.y2());
            return attrToProjector;
        };
        Segment.prototype.entitiesAt = function (point) {
            var entity = this.entityNearest(point);
            if (entity != null) {
                return [entity];
            }
            else {
                return [];
            }
        };
        Segment.prototype.entitiesIn = function (xRangeOrBounds, yRange) {
            var dataXRange;
            var dataYRange;
            if (yRange == null) {
                var bounds = xRangeOrBounds;
                dataXRange = { min: bounds.topLeft.x, max: bounds.bottomRight.x };
                dataYRange = { min: bounds.topLeft.y, max: bounds.bottomRight.y };
            }
            else {
                dataXRange = xRangeOrBounds;
                dataYRange = yRange;
            }
            return this._entitiesIntersecting(dataXRange, dataYRange);
        };
        Segment.prototype._entitiesIntersecting = function (xRange, yRange) {
            var intersected = [];
            var attrToProjector = this._getAttrToProjector();
            var entities = this.entities();
            var entitiesLen = entities.length;
            for (var i = 0; i < entitiesLen; i++) {
                var entity = entities[i];
                if (this._lineIntersectsBox(entity, xRange, yRange, attrToProjector)) {
                    intersected.push(entity);
                }
            }
            return intersected;
        };
        Segment.prototype._lineIntersectsBox = function (entity, xRange, yRange, attrToProjector) {
            var _this = this;
            var x1 = attrToProjector["x1"](entity.datum, entity.index, entity.dataset);
            var x2 = attrToProjector["x2"](entity.datum, entity.index, entity.dataset);
            var y1 = attrToProjector["y1"](entity.datum, entity.index, entity.dataset);
            var y2 = attrToProjector["y2"](entity.datum, entity.index, entity.dataset);
            // check if any of end points of the segment is inside the box
            if ((xRange.min <= x1 && x1 <= xRange.max && yRange.min <= y1 && y1 <= yRange.max) ||
                (xRange.min <= x2 && x2 <= xRange.max && yRange.min <= y2 && y2 <= yRange.max)) {
                return true;
            }
            var startPoint = { x: x1, y: y1 };
            var endPoint = { x: x2, y: y2 };
            var corners = [
                { x: xRange.min, y: yRange.min },
                { x: xRange.min, y: yRange.max },
                { x: xRange.max, y: yRange.max },
                { x: xRange.max, y: yRange.min },
            ];
            var intersections = corners.filter(function (point, index) {
                if (index !== 0) {
                    // return true if border formed by conecting current corner and previous corner intersects with the segment
                    return _this._lineIntersectsSegment(startPoint, endPoint, point, corners[index - 1]) &&
                        _this._lineIntersectsSegment(point, corners[index - 1], startPoint, endPoint);
                }
                return false;
            });
            return intersections.length > 0;
        };
        Segment.prototype._lineIntersectsSegment = function (point1, point2, point3, point4) {
            // tslint:disable-next-line:no-shadowed-variable
            var calcOrientation = function (point1, point2, point) {
                return (point2.x - point1.x) * (point.y - point2.y) - (point2.y - point1.y) * (point.x - point2.x);
            };
            // point3 and point4 are on different sides of line formed by point1 and point2
            return calcOrientation(point1, point2, point3) * calcOrientation(point1, point2, point4) < 0;
        };
        Segment._X2_KEY = "x2";
        Segment._Y2_KEY = "y2";
        return Segment;
    }(xyPlot.XYPlot));
    exports.Segment = Segment;
    });

    var stackedAreaPlot = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });







    var StackedArea = /** @class */ (function (_super) {
        tslib_es6.__extends(StackedArea, _super);
        /**
         * @constructor
         */
        function StackedArea() {
            var _this = _super.call(this) || this;
            _this._stackingResult = memoize.memThunk(function () { return _this.datasets(); }, function () { return _this.x().accessor; }, function () { return _this.y().accessor; }, function () { return _this._stackingOrder; }, function (datasets, keyAccessor, valueAccessor, stackingOrder) {
                return utils$1.Stacking.stack(datasets, keyAccessor, valueAccessor, stackingOrder);
            });
            _this._stackedExtent = memoize.memThunk(_this._stackingResult, function () { return _this.x().accessor; }, function () { return _this._filterForProperty("y"); }, function (stackingResult, keyAccessor, filter) {
                return utils$1.Stacking.stackedExtent(stackingResult, keyAccessor, filter);
            });
            _this._baselineValue = 0;
            _this._stackingOrder = "bottomup";
            _this.addClass("stacked-area-plot");
            _this._baselineValueProvider = function () { return [_this._baselineValue]; };
            _this.croppedRenderingEnabled(false);
            return _this;
        }
        StackedArea.prototype.croppedRenderingEnabled = function (croppedRendering) {
            if (croppedRendering == null) {
                return _super.prototype.croppedRenderingEnabled.call(this);
            }
            if (croppedRendering) {
                // HACKHACK #3032: cropped rendering doesn't currently work correctly on StackedArea
                utils$1.Window.warn("Warning: Stacked Area Plot does not support cropped rendering.");
                return this;
            }
            return _super.prototype.croppedRenderingEnabled.call(this, croppedRendering);
        };
        StackedArea.prototype._getAnimator = function (key) {
            return new animators.Null();
        };
        StackedArea.prototype._setup = function () {
            _super.prototype._setup.call(this);
            this._baseline = this._renderArea.append("line").classed("baseline", true);
        };
        StackedArea.prototype.x = function (x, xScale) {
            if (x == null) {
                return _super.prototype.x.call(this);
            }
            if (xScale == null) {
                _super.prototype.x.call(this, x);
            }
            else {
                _super.prototype.x.call(this, x, xScale);
            }
            this._checkSameDomain();
            return this;
        };
        StackedArea.prototype.y = function (y, yScale) {
            if (y == null) {
                return _super.prototype.y.call(this);
            }
            if (yScale == null) {
                _super.prototype.y.call(this, y);
            }
            else {
                _super.prototype.y.call(this, y, yScale);
            }
            this._checkSameDomain();
            return this;
        };
        /**
         * Gets the offset of the y value corresponding to an x value of a given dataset. This allows other plots to plot
         * points corresponding to their stacked value in the graph.
         * @param dataset The dataset from which to retrieve the y value offset
         * @param x The x value corresponding to the y-value of interest.
         */
        StackedArea.prototype.yOffset = function (dataset, x) {
            var stackingResult = this._stackingResult();
            if (stackingResult == null) {
                return undefined;
            }
            var datasetStackingResult = stackingResult.get(dataset);
            if (datasetStackingResult == null) {
                return undefined;
            }
            var result = datasetStackingResult.get(String(x));
            if (result == null) {
                return undefined;
            }
            return result.offset;
        };
        StackedArea.prototype.stackingOrder = function (stackingOrder) {
            if (stackingOrder == null) {
                return this._stackingOrder;
            }
            this._stackingOrder = stackingOrder;
            this._onDatasetUpdate();
            return this;
        };
        StackedArea.prototype.downsamplingEnabled = function (downsampling) {
            if (downsampling == null) {
                return _super.prototype.downsamplingEnabled.call(this);
            }
            utils$1.Window.warn("Warning: Stacked Area Plot does not support downsampling");
            return this;
        };
        StackedArea.prototype._additionalPaint = function () {
            var scaledBaseline = this.y().scale.scale(this._baselineValue);
            var baselineAttr = {
                x1: 0,
                y1: scaledBaseline,
                x2: this.width(),
                y2: scaledBaseline,
            };
            this._getAnimator("baseline").animate(this._baseline, baselineAttr);
        };
        StackedArea.prototype._updateYScale = function () {
            var yBinding = this.y();
            var scale = (yBinding && yBinding.scale);
            if (scale == null) {
                return;
            }
            scale.addPaddingExceptionsProvider(this._baselineValueProvider);
            scale.addIncludedValuesProvider(this._baselineValueProvider);
        };
        StackedArea.prototype._onDatasetUpdate = function () {
            this._checkSameDomain();
            _super.prototype._onDatasetUpdate.call(this);
            return this;
        };
        StackedArea.prototype.getExtentsForProperty = function (attr) {
            var primaryAttr = "y";
            if (attr === primaryAttr) {
                return [this._stackedExtent()];
            }
            else {
                return _super.prototype.getExtentsForProperty.call(this, attr);
            }
        };
        StackedArea.prototype._checkSameDomain = function () {
            if (!this._projectorsReady()) {
                return;
            }
            var datasets = this.datasets();
            var keyAccessor = this.x().accessor;
            var keySets = datasets.map(function (dataset) {
                return d3.set(dataset.data().map(function (datum, i) { return utils$1.Stacking.normalizeKey(keyAccessor(datum, i, dataset)); })).values();
            });
            var domainKeys = StackedArea._domainKeys(datasets, keyAccessor);
            if (keySets.some(function (keySet) { return keySet.length !== domainKeys.length; })) {
                utils$1.Window.warn("the domains across the datasets are not the same. Plot may produce unintended behavior.");
            }
        };
        /**
         * Given an array of Datasets and the accessor function for the key, computes the
         * set reunion (no duplicates) of the domain of each Dataset. The keys are stringified
         * before being returned.
         *
         * @param {Dataset[]} datasets The Datasets for which we extract the domain keys
         * @param {Accessor<any>} keyAccessor The accessor for the key of the data
         * @return {string[]} An array of stringified keys
         */
        StackedArea._domainKeys = function (datasets, keyAccessor) {
            var domainKeys = d3.set();
            datasets.forEach(function (dataset) {
                var data = dataset.data();
                var dataLen = data.length;
                for (var index = 0; index < dataLen; index++) {
                    var datum = data[index];
                    domainKeys.add(keyAccessor(datum, index, dataset));
                }
            });
            return domainKeys.values();
        };
        StackedArea.prototype._coordinateProjectors = function () {
            var _this = this;
            var xProjector = plot.Plot._scaledAccessor(this.x());
            var yAccessor = this.y().accessor;
            var xAccessor = this.x().accessor;
            var normalizedXAccessor = function (datum, index, dataset) {
                return utils$1.Stacking.normalizeKey(xAccessor(datum, index, dataset));
            };
            var stackingResult = this._stackingResult();
            var stackYProjector = function (d, i, dataset) {
                var y = +yAccessor(d, i, dataset);
                var offset = stackingResult.get(dataset).get(normalizedXAccessor(d, i, dataset)).offset;
                return _this.y().scale.scale(y + offset);
            };
            var stackY0Projector = function (d, i, dataset) {
                var offset = stackingResult.get(dataset).get(normalizedXAccessor(d, i, dataset)).offset;
                return _this.y().scale.scale(offset);
            };
            return [
                xProjector,
                stackYProjector,
                stackY0Projector,
            ];
        };
        StackedArea.prototype._propertyProjectors = function () {
            var propertyToProjectors = _super.prototype._propertyProjectors.call(this);
            var _a = this._coordinateProjectors(), xProjector = _a[0], stackYProjector = _a[1], stackY0Projector = _a[2];
            propertyToProjectors["d"] = this._constructAreaProjector(xProjector, stackYProjector, stackY0Projector);
            return propertyToProjectors;
        };
        StackedArea.prototype._pixelPoint = function (datum, index, dataset) {
            var pixelPoint = _super.prototype._pixelPoint.call(this, datum, index, dataset);
            var xValue = this.x().accessor(datum, index, dataset);
            var yValue = this.y().accessor(datum, index, dataset);
            var scaledYValue = this.y().scale.scale(+yValue + this._stackingResult().get(dataset).get(utils$1.Stacking.normalizeKey(xValue)).offset);
            return { x: pixelPoint.x, y: scaledYValue };
        };
        return StackedArea;
    }(areaPlot.Area));
    exports.StackedArea = StackedArea;
    });

    var stackedBarPlot = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });







    var StackedBar = /** @class */ (function (_super) {
        tslib_es6.__extends(StackedBar, _super);
        /**
         * A StackedBar Plot stacks bars across Datasets based on the primary value of the bars.
         *   On a vertical StackedBar Plot, the bars with the same X value are stacked.
         *   On a horizontal StackedBar Plot, the bars with the same Y value are stacked.
         *
         * @constructor
         * @param {Scale} xScale
         * @param {Scale} yScale
         * @param {string} [orientation="vertical"] One of "vertical"/"horizontal".
         */
        function StackedBar(orientation) {
            if (orientation === void 0) { orientation = "vertical"; }
            var _this = _super.call(this, orientation) || this;
            _this._extremaFormatter = formatters.identity();
            _this._stackingResult = memoize.memThunk(function () { return _this.datasets(); }, function () { return _this.position().accessor; }, function () { return _this.length().accessor; }, function () { return _this._stackingOrder; }, function (datasets, positionAccessor, lengthAccessor, stackingOrder) {
                return utils$1.Stacking.stack(datasets, positionAccessor, lengthAccessor, stackingOrder);
            });
            _this._stackedExtent = memoize.memThunk(_this._stackingResult, function () { return _this.position().accessor; }, function () { return _this._filterForProperty(_this._isVertical ? "y" : "x"); }, function (stackingResult, positionAccessor, filter) {
                return utils$1.Stacking.stackedExtent(stackingResult, positionAccessor, filter);
            });
            _this.addClass("stacked-bar-plot");
            _this._stackingOrder = "bottomup";
            return _this;
        }
        StackedBar.prototype.stackingOrder = function (stackingOrder) {
            if (stackingOrder == null) {
                return this._stackingOrder;
            }
            this._stackingOrder = stackingOrder;
            this._onDatasetUpdate();
            return this;
        };
        StackedBar.prototype.extremaFormatter = function (formatter) {
            if (arguments.length === 0) {
                return this._extremaFormatter;
            }
            else {
                this._extremaFormatter = formatter;
                this.render();
                return this;
            }
        };
        StackedBar.prototype._setup = function () {
            _super.prototype._setup.call(this);
            this._labelArea = this._renderArea.append("g").classed(barPlot.Bar._LABEL_AREA_CLASS, true);
            var context = new src$1.SvgContext(this._labelArea.node());
            this._measurer = new src$1.CacheMeasurer(context);
            this._writer = new src$1.Writer(this._measurer, context);
        };
        StackedBar.prototype._drawLabels = function () {
            var _this = this;
            _super.prototype._drawLabels.call(this);
            // remove all current labels before redrawing
            this._labelArea.selectAll("g").remove();
            var baselineValue = +this.baselineValue();
            var positionScale = this.position().scale;
            var lengthScale = this.length().scale;
            var _a = utils$1.Stacking.stackedExtents(this._stackingResult()), maximumExtents = _a.maximumExtents, minimumExtents = _a.minimumExtents;
            var anyTooWide = [];
            /**
             * Try drawing the text at the center of the bounds. This method does not draw
             * the text if the text would overflow outside of the plot.
             *
             * @param text
             * @param bounds
             * @returns {boolean}
             */
            var maybeDrawLabel = function (text, bounds, barThickness) {
                var _a = bounds.topLeft, x = _a.x, y = _a.y;
                var width = bounds.bottomRight.x - bounds.topLeft.x;
                var height = bounds.bottomRight.y - bounds.topLeft.y;
                var textTooLong = _this._isVertical
                    ? width > barThickness
                    : height > barThickness;
                if (!textTooLong) {
                    var labelContainer = _this._labelArea.append("g").attr("transform", "translate(" + x + ", " + y + ")");
                    labelContainer.classed("stacked-bar-label", true);
                    var writeOptions = {
                        xAlign: "center",
                        yAlign: "center",
                    };
                    _this._writer.write(text, width, height, writeOptions, labelContainer.node());
                }
                return textTooLong;
            };
            var drawLabelsForExtents = function (stacks, computeLabelTopLeft) {
                var attrToProjector = _this._generateAttrToProjector();
                var plotWidth = _this.width();
                var plotHeight = _this.height();
                stacks.forEach(function (stack) {
                    if (stack.extent !== baselineValue) {
                        // only draw sums for values not at the baseline
                        var text = _this.extremaFormatter()(stack.extent);
                        var textDimensions = _this._measurer.measure(text);
                        var stackedDatum = stack.stackedDatum;
                        var originalDatum = stackedDatum.originalDatum, originalIndex = stackedDatum.originalIndex, originalDataset = stackedDatum.originalDataset;
                        // only consider stack extents that are on the screen
                        if (!_this._isDatumOnScreen(attrToProjector, plotWidth, plotHeight, originalDatum, originalIndex, originalDataset)) {
                            return;
                        }
                        var barThickness = plot.Plot._scaledAccessor(_this.attr(barPlot.Bar._BAR_THICKNESS_KEY))(originalDatum, originalIndex, originalDataset);
                        /*
                         * The stackEdge is aligned at the edge of the stack in the length dimension,
                         * and in the center of the stack in the thickness dimension.
                         */
                        var stackEdgeLength = lengthScale.scale(stack.extent);
                        var stackCenterPosition = _this._getPositionAttr(positionScale.scale(stack.axisValue), barThickness) + barThickness / 2;
                        var stackEdge = _this._isVertical
                            ? {
                                x: stackCenterPosition,
                                y: stackEdgeLength,
                            }
                            : {
                                x: stackEdgeLength,
                                y: stackCenterPosition,
                            };
                        var topLeft = computeLabelTopLeft(stackEdge, textDimensions, barThickness);
                        var isTooWide = maybeDrawLabel(text, {
                            topLeft: topLeft,
                            bottomRight: {
                                x: topLeft.x + textDimensions.width,
                                y: topLeft.y + textDimensions.height,
                            },
                        }, barThickness);
                        anyTooWide.push(isTooWide);
                    }
                });
            };
            drawLabelsForExtents(maximumExtents, function (stackEdge, measurement, thickness) {
                var primaryTextMeasurement = _this._isVertical ? measurement.width : measurement.height;
                var secondaryTextMeasurement = _this._isVertical ? measurement.height : measurement.width;
                return {
                    x: _this._isVertical
                        ? stackEdge.x - primaryTextMeasurement / 2
                        : stackEdge.x + StackedBar._EXTREMA_LABEL_MARGIN_FROM_BAR,
                    y: _this._isVertical
                        ? stackEdge.y - secondaryTextMeasurement
                        : stackEdge.y - primaryTextMeasurement / 2,
                };
            });
            drawLabelsForExtents(minimumExtents, function (stackEdge, measurement, thickness) {
                var primaryTextMeasurement = _this._isVertical ? measurement.width : measurement.height;
                var secondaryTextMeasurement = _this._isVertical ? measurement.height : measurement.width;
                return {
                    x: _this._isVertical
                        ? stackEdge.x - primaryTextMeasurement / 2
                        : stackEdge.x - secondaryTextMeasurement,
                    y: _this._isVertical
                        ? stackEdge.y + StackedBar._EXTREMA_LABEL_MARGIN_FROM_BAR
                        : stackEdge.y - primaryTextMeasurement / 2,
                };
            });
            if (anyTooWide.some(function (d) { return d; })) {
                this._labelArea.selectAll("g").remove();
            }
        };
        StackedBar.prototype._generateAttrToProjector = function () {
            var _this = this;
            var attrToProjector = _super.prototype._generateAttrToProjector.call(this);
            var valueAttr = this._isVertical ? "y" : "x";
            var lengthScale = this.length().scale;
            var lengthAccessor = this.length().accessor;
            var positionAccessor = this.position().accessor;
            var normalizedPositionAccessor = function (datum, index, dataset) {
                return utils$1.Stacking.normalizeKey(positionAccessor(datum, index, dataset));
            };
            var stackingResult = this._stackingResult();
            var getStart = function (d, i, dataset) {
                return lengthScale.scale(stackingResult.get(dataset).get(normalizedPositionAccessor(d, i, dataset)).offset);
            };
            var getEnd = function (d, i, dataset) {
                return lengthScale.scale(+lengthAccessor(d, i, dataset) +
                    stackingResult.get(dataset).get(normalizedPositionAccessor(d, i, dataset)).offset);
            };
            var heightF = function (d, i, dataset) {
                return Math.abs(getEnd(d, i, dataset) - getStart(d, i, dataset));
            };
            attrToProjector[this._isVertical ? "height" : "width"] = heightF;
            var attrFunction = function (d, i, dataset) {
                return +lengthAccessor(d, i, dataset) < 0 ? getStart(d, i, dataset) : getEnd(d, i, dataset);
            };
            attrToProjector[valueAttr] = function (d, i, dataset) {
                return _this._isVertical ? attrFunction(d, i, dataset) : attrFunction(d, i, dataset) - heightF(d, i, dataset);
            };
            return attrToProjector;
        };
        StackedBar.prototype.getExtentsForProperty = function (attr) {
            var primaryAttr = this._isVertical ? "y" : "x";
            if (attr === primaryAttr) {
                return [this._stackedExtent()];
            }
            else {
                return _super.prototype.getExtentsForProperty.call(this, attr);
            }
        };
        StackedBar.prototype.invalidateCache = function () {
            _super.prototype.invalidateCache.call(this);
            this._measurer.reset();
        };
        StackedBar._EXTREMA_LABEL_MARGIN_FROM_BAR = 5;
        return StackedBar;
    }(barPlot.Bar));
    exports.StackedBar = StackedBar;
    });

    var waterfallPlot = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });




    var Waterfall = /** @class */ (function (_super) {
        tslib_es6.__extends(Waterfall, _super);
        function Waterfall() {
            var _this = _super.call(this) || this;
            _this._connectorsEnabled = false;
            _this.addClass("waterfall-plot");
            return _this;
        }
        Waterfall.prototype.connectorsEnabled = function (enabled) {
            if (enabled == null) {
                return this._connectorsEnabled;
            }
            this._connectorsEnabled = enabled;
            return this;
        };
        Waterfall.prototype.total = function (total) {
            if (total == null) {
                return this._propertyBindings.get(Waterfall._TOTAL_KEY);
            }
            this._bindProperty(Waterfall._TOTAL_KEY, total, null);
            return this;
        };
        Waterfall.prototype._additionalPaint = function (time) {
            var _this = this;
            this._connectorArea.selectAll("line").remove();
            if (this._connectorsEnabled) {
                utils$1.Window.setTimeout(function () { return _this._drawConnectors(); }, time);
            }
        };
        Waterfall.prototype._createNodesForDataset = function (dataset) {
            var drawer = _super.prototype._createNodesForDataset.call(this, dataset);
            this._connectorArea = this._renderArea.append("g").classed(Waterfall._CONNECTOR_AREA_CLASS, true);
            return drawer;
        };
        Waterfall.prototype.getExtentsForProperty = function (attr) {
            var primaryAttr = "y";
            if (attr === primaryAttr) {
                return [this._extent];
            }
            else {
                return _super.prototype.getExtentsForProperty.call(this, attr);
            }
        };
        Waterfall.prototype._generateAttrToProjector = function () {
            var _this = this;
            var attrToProjector = _super.prototype._generateAttrToProjector.call(this);
            var yScale = this.y().scale;
            var totalAccessor = plot.Plot._scaledAccessor(this.total());
            var yAttr = this.attr("y");
            if (yAttr == null) {
                attrToProjector["y"] = function (d, i, dataset) {
                    var currentValue = _this.y().accessor(d, i, dataset);
                    var isTotal = totalAccessor(d, i, dataset);
                    if (isTotal) {
                        return Math.min(yScale.scale(currentValue), yScale.scale(0));
                    }
                    else {
                        var currentSubtotal = _this._subtotals[i];
                        if (i === 0) {
                            if (currentValue < 0) {
                                return yScale.scale(currentSubtotal - currentValue);
                            }
                            else {
                                return yScale.scale(currentSubtotal);
                            }
                        }
                        var priorSubtotal = _this._subtotals[i - 1];
                        if (currentSubtotal > priorSubtotal) {
                            return yScale.scale(currentSubtotal);
                        }
                        else {
                            return yScale.scale(priorSubtotal);
                        }
                    }
                };
            }
            var heightAttr = this.attr("height");
            if (heightAttr == null) {
                attrToProjector["height"] = function (d, i, dataset) {
                    var isTotal = totalAccessor(d, i, dataset);
                    var currentValue = _this.y().accessor(d, i, dataset);
                    if (isTotal) {
                        return Math.abs(yScale.scale(currentValue) - yScale.scale(0));
                    }
                    else {
                        var currentSubtotal = _this._subtotals[i];
                        if (i === 0) {
                            return Math.abs(yScale.scale(currentSubtotal) - yScale.scale(currentSubtotal - currentValue));
                        }
                        else {
                            var priorSubtotal = _this._subtotals[i - 1];
                            return Math.abs(yScale.scale(currentSubtotal) - yScale.scale(priorSubtotal));
                        }
                    }
                };
            }
            attrToProjector["class"] = function (d, i, dataset) {
                var baseClass = "";
                if (_this.attr("class") != null) {
                    baseClass = _this.attr("class").accessor(d, i, dataset) + " ";
                }
                var isTotal = totalAccessor(d, i, dataset);
                if (isTotal) {
                    return baseClass + Waterfall._BAR_TOTAL_CLASS;
                }
                else {
                    var delta = _this.y().accessor(d, i, dataset);
                    return baseClass + (delta > 0 ? Waterfall._BAR_GROWTH_CLASS : Waterfall._BAR_DECLINE_CLASS);
                }
            };
            return attrToProjector;
        };
        Waterfall.prototype._onDatasetUpdate = function () {
            this._updateSubtotals();
            _super.prototype._onDatasetUpdate.call(this);
            return this;
        };
        Waterfall.prototype._calculateSubtotalsAndExtent = function (dataset) {
            var min = Number.MAX_VALUE;
            var max = Number.MIN_VALUE;
            var total = 0;
            var hasStarted = false;
            var data = dataset.data();
            var dataLen = data.length;
            for (var index = 0; index < dataLen; index++) {
                var datum = data[index];
                var currentValue = this.y().accessor(datum, index, dataset);
                var isTotal = this.total().accessor(datum, index, dataset);
                if (!isTotal || index === 0) {
                    total += currentValue;
                }
                this._subtotals.push(total);
                if (total < min) {
                    min = total;
                }
                if (total > max) {
                    max = total;
                }
                if (isTotal) {
                    if (currentValue < min) {
                        min = currentValue;
                    }
                    if (currentValue > max) {
                        max = currentValue;
                    }
                }
                if (!hasStarted && isTotal) {
                    var startTotal = currentValue - total;
                    for (var i = 0; i < this._subtotals.length; i++) {
                        this._subtotals[i] += startTotal;
                    }
                    hasStarted = true;
                    total += startTotal;
                    min += startTotal;
                    max += startTotal;
                }
            }
            this._extent = [min, max];
        };
        Waterfall.prototype._drawConnectors = function () {
            var attrToProjector = this._getAttrToProjector();
            var dataset = this.datasets()[0];
            for (var datumIndex = 1; datumIndex < dataset.data().length; datumIndex++) {
                var prevIndex = datumIndex - 1;
                var datum = dataset.data()[datumIndex];
                var prevDatum = dataset.data()[prevIndex];
                var x = attrToProjector["x"](prevDatum, prevIndex, dataset);
                var x2 = attrToProjector["x"](datum, datumIndex, dataset) + attrToProjector["width"](datum, datumIndex, dataset);
                var y = attrToProjector["y"](datum, datumIndex, dataset);
                if ((this._subtotals[datumIndex] > 0 && this._subtotals[datumIndex] > this._subtotals[prevIndex]) ||
                    (this._subtotals[datumIndex] < 0 && this._subtotals[datumIndex] >= this._subtotals[prevIndex])) {
                    y = attrToProjector["y"](datum, datumIndex, dataset) + attrToProjector["height"](datum, datumIndex, dataset);
                }
                this._connectorArea.append("line").classed(Waterfall._CONNECTOR_CLASS, true)
                    .attr("x1", x).attr("x2", x2).attr("y1", y).attr("y2", y);
            }
        };
        Waterfall.prototype._updateSubtotals = function () {
            var datasets = this.datasets();
            if (datasets.length > 0) {
                var dataset = datasets[datasets.length - 1];
                this._subtotals = new Array();
                this._calculateSubtotalsAndExtent(dataset);
            }
        };
        Waterfall._BAR_DECLINE_CLASS = "waterfall-decline";
        Waterfall._BAR_GROWTH_CLASS = "waterfall-growth";
        Waterfall._BAR_TOTAL_CLASS = "waterfall-total";
        Waterfall._CONNECTOR_CLASS = "connector";
        Waterfall._CONNECTOR_AREA_CLASS = "connector-area";
        Waterfall._TOTAL_KEY = "total";
        return Waterfall;
    }(barPlot.Bar));
    exports.Waterfall = Waterfall;
    });

    var plots = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });

    tslib_es6.__exportStar(areaPlot, exports);
    tslib_es6.__exportStar(barPlot, exports);
    tslib_es6.__exportStar(commons, exports);
    tslib_es6.__exportStar(clusteredBarPlot, exports);
    tslib_es6.__exportStar(linePlot, exports);
    tslib_es6.__exportStar(piePlot, exports);
    tslib_es6.__exportStar(rectanglePlot, exports);
    tslib_es6.__exportStar(scatterPlot, exports);
    tslib_es6.__exportStar(segmentPlot, exports);
    tslib_es6.__exportStar(stackedAreaPlot, exports);
    tslib_es6.__exportStar(stackedBarPlot, exports);
    tslib_es6.__exportStar(waterfallPlot, exports);
    });

    var version$3 = createCommonjsModule(function (module, exports) {
    /**
     * Copyright 2014-present Palantir Technologies
     * @license MIT
     */
    Object.defineProperty(exports, "__esModule", { value: true });
    /*
     * WARNING: The js output of this expression is searched by string (yes, I know) and replaced with a
     * real version number during the dist phase for for npm module publishing. Modifying this line should
     * be accompanied by modifying the "sed-version" task in package.json accordingly.
     */
    exports.version = "3.9.0";
    });

    var src$2 = createCommonjsModule(function (module, exports) {
    Object.defineProperty(exports, "__esModule", { value: true });

    // HACKHACK d3-selection-multi doesn't play well with default "d3" package in a
    // bundler environment (e.g. webpack) - see https://github.com/d3/d3-selection-multi/issues/11
    // we add it manually to the default "d3" bundle


    exports.Animators = animators;

    exports.Axes = axes;

    exports.Components = components;

    exports.Configs = config$1;

    exports.Formatters = formatters;

    exports.RenderController = renderController;

    exports.RenderPolicies = renderPolicy;

    exports.SymbolFactories = symbolFactories;

    exports.Dispatchers = dispatchers;

    exports.Drawers = drawers;

    exports.Interactions = interactions;

    exports.Plots = plots;

    exports.Scales = scales;

    exports.Utils = utils$1;
    tslib_es6.__exportStar(axis$2, exports);

    exports.TimeInterval = timeAxis.TimeInterval;
    tslib_es6.__exportStar(component, exports);
    tslib_es6.__exportStar(componentContainer, exports);
    tslib_es6.__exportStar(dataset, exports);

    exports.version = version$3.version;
    tslib_es6.__exportStar(dispatcher, exports);
    tslib_es6.__exportStar(drawer, exports);
    tslib_es6.__exportStar(interaction, exports);
    tslib_es6.__exportStar(keyInteraction, exports);
    tslib_es6.__exportStar(xyPlot, exports);
    tslib_es6.__exportStar(plot, exports);
    tslib_es6.__exportStar(quantitativeScale, exports);
    tslib_es6.__exportStar(scale, exports);
    });

    /* Copyright 2015 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * A list of symbols that line charts can cycle through per data series.
     */
    const SYMBOLS_LIST = [
        {
            character: '\u25FC',
            method: src$2.SymbolFactories.square,
        },
        {
            character: '\u25C6',
            method: src$2.SymbolFactories.diamond,
        },
        {
            character: '\u25B2',
            method: src$2.SymbolFactories.triangle,
        },
        {
            character: '\u2605',
            method: src$2.SymbolFactories.star,
        },
        {
            character: '\u271A',
            method: src$2.SymbolFactories.cross,
        },
    ];
    /** X axis choices for TensorBoard charts. */
    var XType;
    (function (XType) {
        /** Linear scale using the "step" property of the datum. */
        XType["STEP"] = "step";
        /** Temporal scale using the "wall_time" property of the datum. */
        XType["RELATIVE"] = "relative";
        /**
         * Temporal scale using the "relative" property of the datum if it is present
         * or calculating from "wall_time" if it isn't.
         */
        XType["WALL_TIME"] = "wall_time";
    })(XType || (XType = {}));
    let Y_TOOLTIP_FORMATTER_PRECISION = 4;
    let STEP_FORMATTER_PRECISION = 4;
    let Y_AXIS_FORMATTER_PRECISION = 3;
    let TOOLTIP_Y_PIXEL_OFFSET = 20;
    let TOOLTIP_CIRCLE_SIZE = 4;
    let NAN_SYMBOL_SIZE = 6;
    /* Create a formatter function that will switch between exponential and
     * regular display depending on the scale of the number being formatted,
     * and show `digits` significant digits.
     */
    function multiscaleFormatter(digits) {
        return (v) => {
            let absv = Math.abs(v);
            if (absv < 1e-15) {
                // Sometimes zero-like values get an annoying representation
                absv = 0;
            }
            let f;
            if (absv >= 10000) {
                f = format('.' + digits + '~e');
            }
            else if (absv > 0 && absv < 0.01) {
                f = format('.' + digits + '~e');
            }
            else {
                f = format('.' + digits + '~g');
            }
            return f(v);
        };
    }
    const stepFormatter = format(`.${STEP_FORMATTER_PRECISION}~s`);
    function stepX() {
        let scale = new src$2.Scales.Linear();
        scale.tickGenerator(src$2.Scales.TickGenerators.integerTickGenerator());
        let axis = new src$2.Axes.Numeric(scale, 'bottom');
        axis.formatter(stepFormatter);
        return {
            scale: scale,
            axis: axis,
            accessor: (d) => d.step,
        };
    }
    let timeFormatter = src$2.Formatters.time('%a %b %e, %H:%M:%S');
    function wallX() {
        let scale = new src$2.Scales.Time();
        return {
            scale: scale,
            axis: new src$2.Axes.Time(scale, 'bottom'),
            accessor: (d) => d.wall_time,
        };
    }
    let relativeAccessor = 
    // tslint:disable-next-line:no-any be quiet tsc
    (d, index, dataset) => {
        // We may be rendering the final-point datum for scatterplot.
        // If so, we will have already provided the 'relative' property
        if (d.relative != null) {
            return d.relative;
        }
        let data = dataset.data();
        // I can't imagine how this function would be called when the data is
        // empty (after all, it iterates over the data), but lets guard just
        // to be safe.
        let first = data.length > 0 ? +data[0].wall_time : 0;
        return (+d.wall_time - first) / (60 * 60 * 1000); // ms to hours
    };
    let relativeFormatter = (n) => {
        // we will always show 2 units of precision, e.g days and hours, or
        // minutes and seconds, but not hours and minutes and seconds
        let ret = '';
        let days = Math.floor(n / 24);
        n -= days * 24;
        if (days) {
            ret += days + 'd ';
        }
        let hours = Math.floor(n);
        n -= hours;
        n *= 60;
        if (hours || days) {
            ret += hours + 'h ';
        }
        let minutes = Math.floor(n);
        n -= minutes;
        n *= 60;
        if (minutes || hours || days) {
            ret += minutes + 'm ';
        }
        let seconds = Math.floor(n);
        return ret + seconds + 's';
    };
    function relativeX() {
        let scale = new src$2.Scales.Linear();
        return {
            scale: scale,
            axis: new src$2.Axes.Numeric(scale, 'bottom'),
            accessor: relativeAccessor,
        };
    }
    function getXComponents(xType) {
        switch (xType) {
            case XType.STEP:
                return stepX();
            case XType.WALL_TIME:
                return wallX();
            case XType.RELATIVE:
                return relativeX();
            default:
                throw new Error('invalid xType: ' + xType);
        }
    }

    // NOT_LOADED is implicit
    var LoadState;
    (function (LoadState) {
        LoadState[LoadState["LOADING"] = 0] = "LOADING";
        LoadState[LoadState["LOADED"] = 1] = "LOADED";
    })(LoadState || (LoadState = {}));
    function DataLoaderBehavior(superClass) {
        return class DataLoaderBehaviorImpl extends superClass {
            constructor() {
                super(...arguments);
                /**
                 * A unique identifiable string. When changes, it expunges the data
                 * cache.
                 */
                this.loadKey = '';
                // List of items to be loaded. By default, items are passed to
                // `requestData` to fetch data. When the request resolves, invokes
                // `loadDataCallback` with the datum and its response.
                this.dataToLoad = [];
                /**
                 * A function that takes an item as an input and returns a unique
                 * identifiable string. Used for caching purposes.
                 */
                this.getDataLoadName = (item) => String(item);
                this.dataLoading = false;
                // The standard Node.isConnected doesn't seem to be set reliably, so we
                // wire up our own property manually.
                this._isConnected = false;
                /*
                 * A map of a cache key to LoadState. If a cacheKey does not exist in the
                 * map, it is considered NOT_LOADED.
                 * Invoking `reload` or a change in `loadKey` clears the cache.
                 */
                this._dataLoadState = new Map();
                this._canceller = new Canceller();
                this._loadDataAsync = null;
                this._loadData = lodash.throttle(this._loadDataImpl, 100, {
                    leading: true,
                    trailing: true,
                });
            }
            connectedCallback() {
                super.connectedCallback();
                this._isConnected = true;
            }
            disconnectedCallback() {
                super.disconnectedCallback();
                this._isConnected = false;
            }
            static get properties() {
                return {
                    active: {
                        type: Boolean,
                        observer: '_loadDataIfActive',
                    },
                    _isConnected: { type: Boolean },
                    loadKey: { type: String },
                    dataToLoad: { type: Array },
                    getDataLoadName: { type: Object },
                    loadDataCallback: { type: Object },
                    requestData: { type: Object },
                };
            }
            static get observers() {
                return ['_dataToLoadChanged(_isConnected, dataToLoad.*)'];
            }
            onLoadFinish() {
                // Override to do something useful.
            }
            reload() {
                this._dataLoadState.clear();
                this._loadData();
            }
            reset() {
                // https://github.com/tensorflow/tensorboard/issues/1499
                // Cannot use the observer to observe `loadKey` changes directly.
                if (this._loadDataAsync != null) {
                    clearTimeout(this._loadDataAsync);
                    this._loadDataAsync = null;
                }
                if (this._canceller)
                    this._canceller.cancelAll();
                if (this._dataLoadState)
                    this._dataLoadState.clear();
                if (this._isConnected)
                    this._loadData();
            }
            _dataToLoadChanged() {
                if (this._isConnected)
                    this._loadData();
            }
            detached() {
                // Note: Cannot call canceller.cancelAll since it will poison the cache.
                // detached gets called when a component gets unmounted from the document
                // but it can be re-mounted. When remounted, poisoned cache will manifest.
                // t=0: dataLoadState: 'a' = loading
                // t=10: unmount
                // t=20: request for 'a' resolves but we do not change the loadState
                // because we do not want to set one if, instead, it was resetted at t=10.
                if (this._loadDataAsync != null) {
                    clearTimeout(this._loadDataAsync);
                    this._loadDataAsync = null;
                }
            }
            _loadDataIfActive() {
                if (this.active) {
                    this._loadData();
                }
            }
            _loadDataImpl() {
                if (!this.active)
                    return;
                if (this._loadDataAsync !== null)
                    clearTimeout(this._loadDataAsync);
                this._loadDataAsync = setTimeout(this._canceller.cancellable((result) => {
                    if (result.cancelled) {
                        return;
                    }
                    this.dataLoading = true;
                    const dirtyItems = this.dataToLoad.filter((datum) => {
                        const cacheKey = this.getDataLoadName(datum);
                        return !this._dataLoadState.has(cacheKey);
                    });
                    for (const item of dirtyItems) {
                        const cacheKey = this.getDataLoadName(item);
                        this._dataLoadState.set(cacheKey, LoadState.LOADING);
                    }
                    const onLoad = this._canceller.cancellable((result) => {
                        if (result.cancelled) {
                            return;
                        }
                        const { item, data } = result.value;
                        const cacheKey = this.getDataLoadName(item);
                        this._dataLoadState.set(cacheKey, LoadState.LOADED);
                        this.loadDataCallback(this, item, data);
                    });
                    const onFinish = this._canceller.cancellable((result) => {
                        // Only notify of data load if the load was not cancelled.
                        if (!result.cancelled) {
                            const keysFetched = result.value;
                            const fetched = new Set(dirtyItems.map((item) => this.getDataLoadName(item)));
                            const shouldNotify = this.dataToLoad.some((datum) => fetched.has(this.getDataLoadName(datum)));
                            if (shouldNotify) {
                                this.onLoadFinish();
                            }
                            this._loadDataAsync = null;
                        }
                        const isDataFetchPending = Array.from(this._dataLoadState.values()).includes(LoadState.LOADING);
                        if (!isDataFetchPending) {
                            this.dataLoading = false;
                        }
                    });
                    this.requestData(dirtyItems, onLoad, () => onFinish(undefined));
                }));
            }
        };
    }

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    registerStyleDomModule({
        moduleName: 'plottable-style',
        styleContent: `
    
.plottable-colors-0 {
  background-color: #5279c7; /* INDIGO */
}

.plottable-colors-1 {
  background-color: #fd373e; /* CORAL_RED */
}

.plottable-colors-2 {
  background-color: #63c261; /* FERN */
}

.plottable-colors-3 {
  background-color: #fad419; /* BRIGHT_SUN */
}

.plottable-colors-4 {
  background-color: #2c2b6f; /* JACARTA */
}

.plottable-colors-5 {
  background-color: #ff7939; /* BURNING_ORANGE */
}

.plottable-colors-6 {
  background-color: #db2e65; /* CERISE_RED */
}

.plottable-colors-7 {
  background-color: #99ce50; /* CONIFER */
}

.plottable-colors-8 {
  background-color: #962565; /* ROYAL_HEATH */
}

.plottable-colors-9 {
  background-color: #06cccc; /* ROBINS_EGG_BLUE */
}

/**
 * User-supplied renderTo element.
 */
.plottable {
  display: block; /* must be block elements for width/height calculations to work in Firefox. */
  pointer-events: visibleFill;
  position: relative;
  /**
   * Pre 3.0, users could set the dimension of the root element in two ways: either using CSS
   * (inline or through a stylesheet), or using the SVG width/height attributes. By default, we
   * set the SVG width/height attributes to 100%.
   *
   * Post 3.0 the root element is always a normal div and the only way to set the dimensions is
   * to use CSS. To replicate the "100%-by-default" behavior, we apply width/height 100%.
   */
  width: 100%;
  height: 100%;
}

/**
 * The _element that roots each Component's DOM.
 */
.plottable .component {
  /* Allow components to be positioned with explicit left/top/width/height styles */
  position: absolute;
}

.plottable .background-container,
.plottable .content,
.plottable .foreground-container {
  position: absolute;
  width: 100%;
  height: 100%;
}

/**
 * Don't allow svg elements above the content to steal events
 */
.plottable .foreground-container {
  pointer-events: none;
}

.plottable .component-overflow-hidden {
  overflow: hidden;
}

.plottable .component-overflow-visible {
  overflow: visible;
}

.plottable .plot-canvas-container {
  width: 100%;
  height: 100%;
  overflow: hidden;
}

.plottable .plot-canvas {
  width: 100%;
  height: 100%;
  /**
   * Play well with deferred rendering.
   */
  transform-origin: 0px 0px 0px;
}

.plottable text {
  text-rendering: geometricPrecision;
}

.plottable .label text {
  fill: #32313F;
}

.plottable .bar-label-text-area text,
.plottable .scatter-label-text-area text {
  font-size: 12px;
}

.plottable .label-area text {
  fill: #32313F;
  font-size: 14px;
}

.plottable .light-label text {
  fill: white;
}

.plottable .dark-label text {
  fill: #32313F;
}

.plottable .off-bar-label text {
  fill: #32313F;
}

.plottable .stacked-bar-label text {
  fill: #32313F;
  font-style: normal;
}

.plottable .stacked-bar-plot .off-bar-label {
  /* HACKHACK #2795: correct off-bar label logic to be implemented on StackedBar */
  visibility: hidden !important;
}

.plottable .axis-label text {
  font-size: 10px;
  font-weight: bold;
  letter-spacing: 1px;
  line-height: normal;
  text-transform: uppercase;
}

.plottable .title-label text {
  font-size: 20px;
  font-weight: bold;
}

.plottable .axis line.baseline {
  stroke: #CCC;
  stroke-width: 1px;
}

.plottable .axis line.tick-mark {
  stroke: #CCC;
  stroke-width: 1px;
}

.plottable .axis text {
  fill: #32313F;
  font-size: 12px;
  font-weight: 200;
  line-height: normal;
}

.plottable .axis .annotation-circle {
  fill: white;
  stroke-width: 1px;
  stroke: #CCC;
}

.plottable .axis .annotation-line {
  stroke: #CCC;
  stroke-width: 1px;
}

.plottable .axis .annotation-rect {
  stroke: #CCC;
  stroke-width: 1px;
  fill: white;
}

.plottable .bar-plot .baseline {
  stroke: #999;
}

.plottable .gridlines line {
  stroke: #3C3C3C; /* hackhack: gridlines should be solid; see #820 */
  opacity: 0.25;
  stroke-width: 1px;
}

.plottable .selection-box-layer .selection-area {
  fill: black;
  fill-opacity: 0.03;
  stroke: #CCC;
}
/* DragBoxLayer */
.plottable .drag-box-layer.x-resizable .drag-edge-lr {
  cursor: ew-resize;
}
.plottable .drag-box-layer.y-resizable .drag-edge-tb {
  cursor: ns-resize;
}

.plottable .drag-box-layer.x-resizable.y-resizable .drag-corner-tl {
  cursor: nwse-resize;
}
.plottable .drag-box-layer.x-resizable.y-resizable .drag-corner-tr {
  cursor: nesw-resize;
}
.plottable .drag-box-layer.x-resizable.y-resizable .drag-corner-bl {
  cursor: nesw-resize;
}
.plottable .drag-box-layer.x-resizable.y-resizable .drag-corner-br {
  cursor: nwse-resize;
}

.plottable .drag-box-layer.movable .selection-area {
  cursor: move; /* IE fallback */
  cursor: -moz-grab;
  cursor: -webkit-grab;
  cursor: grab;
}

.plottable .drag-box-layer.movable .selection-area:active {
  cursor: -moz-grabbing;
  cursor: -webkit-grabbing;
  cursor: grabbing;
}
/* /DragBoxLayer */

.plottable .guide-line-layer line.guide-line {
  stroke: #CCC;
  stroke-width: 1px;
}

.plottable .drag-line-layer.enabled.vertical line.drag-edge {
  cursor: ew-resize;
}

.plottable .drag-line-layer.enabled.horizontal line.drag-edge {
  cursor: ns-resize;
}

.plottable .legend text {
  fill: #32313F;
  font-size: 12px;
  font-weight: bold;
  line-height: normal;
}

.plottable .interpolated-color-legend rect.swatch-bounding-box {
  fill: none;
  stroke: #CCC;
  stroke-width: 1px;
  pointer-events: none;
}

.plottable .waterfall-plot line.connector {
  stroke: #CCC;
  stroke-width: 1px;
}

.plottable .pie-plot .arc.outline {
  stroke-linejoin: round;
}

`,
    });

    /* Copyright 2019 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    function getHtmlElementAncestors(elem) {
        const elems = [];
        while (elem && elem instanceof HTMLElement) {
            elems.push(elem);
            if (elem.assignedSlot) {
                elem = elem.assignedSlot;
            }
            else if (!elem.parentElement) {
                const parentNode = elem.parentNode;
                if (parentNode instanceof DocumentFragment) {
                    elem = parentNode.host;
                }
                else {
                    // <html>.parentNode == <html>
                    elem = parentNode !== elem ? parentNode : null;
                }
            }
            else {
                elem = elem.parentElement;
            }
        }
        return elems;
    }
    const _IDENTITY_TRANSFORM = [1, 0, 0, 1, 0, 0];
    // Forked from https://github.com/palantir/plottable/blob/b6e36fbd4d8d7cba579d853b9f35cc260d1243bf/src/utils/mathUtils.ts#L173-L202
    // The only difference is the implementation of the getHtmlElementAncestors.
    function getCumulativeTransform(element) {
        const elems = getHtmlElementAncestors(element);
        let transform = _IDENTITY_TRANSFORM;
        let offsetParent = null;
        for (const elem of elems) {
            // apply css transform from any ancestor element
            const elementTransform = src$2.Utils.DOM.getElementTransform(elem);
            if (elementTransform != null) {
                const midX = elem.clientWidth / 2;
                const midY = elem.clientHeight / 2;
                transform = src$2.Utils.Math.multiplyTranslate(transform, [
                    midX,
                    midY,
                ]);
                transform = src$2.Utils.Math.multiplyMatrix(transform, src$2.Utils.Math.invertMatrix(elementTransform));
                transform = src$2.Utils.Math.multiplyTranslate(transform, [
                    -midX,
                    -midY,
                ]);
            }
            // apply scroll offsets from any ancestor element
            let offsetX = elem.scrollLeft;
            let offsetY = elem.scrollTop;
            // apply client+offset from only acenstor "offsetParent"
            if (offsetParent === null || elem === offsetParent) {
                offsetX -= elem.offsetLeft + elem.clientLeft;
                offsetY -= elem.offsetTop + elem.clientTop;
                offsetParent = elem.offsetParent;
            }
            transform = src$2.Utils.Math.multiplyTranslate(transform, [
                offsetX,
                offsetY,
            ]);
        }
        return transform;
    }
    class CustomTranslator extends src$2.Utils.Translator {
        computePosition(clientX, clientY) {
            const clientPosition = {
                x: clientX,
                y: clientY,
            };
            const transform = getCumulativeTransform(this._rootElement);
            if (transform == null) {
                return clientPosition;
            }
            const transformed = src$2.Utils.Math.applyTransform(transform, clientPosition);
            return transformed;
        }
    }
    class MouseDispatcher extends src$2.Dispatchers.Mouse {
        constructor(component) {
            super(component);
            // eventTarget is `document` by default. Change it to the root of chart.
            this._eventTarget = component.root().rootElement().node();
            // Requires custom translator that uses correct DOM traversal (with
            // WebComponents) to change pointer position to relative to the root node.
            this._translator = new CustomTranslator(component.root().rootElement().node());
        }
        static getDispatcher(component) {
            const element = component.root().rootElement();
            let dispatcher = element[MouseDispatcher._DISPATCHER_KEY];
            if (!dispatcher) {
                dispatcher = new MouseDispatcher(component);
                element[MouseDispatcher._DISPATCHER_KEY] = dispatcher;
            }
            return dispatcher;
        }
    }
    class TouchDispatcher extends src$2.Dispatchers.Touch {
        constructor(component) {
            super(component);
            // eventTarget is `document` by default. Change it to the root of chart.
            this._eventTarget = component.root().rootElement().node();
            // Requires custom translator that uses correct DOM traversal (with
            // WebComponents) to change pointer position to relative to the root node.
            this._translator = new CustomTranslator(component.root().rootElement().node());
        }
        static getDispatcher(component) {
            const element = component.root().rootElement();
            let dispatcher = element[TouchDispatcher._DISPATCHER_KEY];
            if (!dispatcher) {
                dispatcher = new TouchDispatcher(component);
                element[TouchDispatcher._DISPATCHER_KEY] = dispatcher;
            }
            return dispatcher;
        }
    }
    /**
     * Fixes #3282.
     *
     * Repro: when tooltip is shown, slowly move the mouse to an edge of a chart.
     * We expect the tooltip the disappear when the cursor is on the edge of the
     * chart.
     *
     * Cause:
     * 1. For Polymer 2 and its Shadow DOM compatibility, TensorBoard opted out of
     * the event delegation of Plottable. Plottable, by default, attaches a set of
     * event listeners on document.body and invokes appropriate callbacks
     * depending on the circumstances. However, with the Shadow DOM, the event
     * re-targetting broke (harder to identify `event.target`), so TensorBoard,
     * instead, attaches the event listeners on every Plottable container, SVGs.
     *
     * 2. When mouse leaves (mouseout) the container, Plottable remaps the event
     * as mouse move and calculate whether the cursor is inside a component
     * (Interaction.prototype._isInsideComponent, specifically) to trigger
     * appropriate callback. The method, however is flawed since it returns, for a
     * component that is, for instance, at <0, 0> with size of <100, 100>, true
     * when pointer is at <100, 100>. It should only return true for [0, 100) for
     * a given dimension, instead.  As a result, the mouseout event occurred at
     * <100, 100> was treated as an event inside the component but all the
     * subsequent mouse movements are not captured since they are events that
     * occurred outside of the event target. In vanilla Plottable, this bug do not
     * manifest since event delegation on the entire document will eventually
     * trigger mouse out when cursor is at, for instance, <101, 100>.
     */
    src$2.Interaction.prototype._isInsideComponent = function (p) {
        return (0 <= p.x &&
            0 <= p.y &&
            // Delta: `<` instead of `<=` here and below.
            p.x < this._componentAttachedTo.width() &&
            p.y < this._componentAttachedTo.height());
    };
    class PointerInteraction extends src$2.Interactions.Pointer {
        _anchor(component) {
            const anyThis = this;
            anyThis._isAnchored = true;
            anyThis._mouseDispatcher = MouseDispatcher.getDispatcher(anyThis._componentAttachedTo);
            anyThis._mouseDispatcher.onMouseMove(anyThis._mouseMoveCallback);
            anyThis._touchDispatcher = TouchDispatcher.getDispatcher(anyThis._componentAttachedTo);
            anyThis._touchDispatcher.onTouchStart(anyThis._touchStartCallback);
        }
    }

    var TooltipPosition;
    (function (TooltipPosition) {
        /**
         * Positions the tooltip to the bottom of the chart in most case. Positions
         * the tooltip above the chart if there isn't sufficient space below.
         */
        TooltipPosition["AUTO"] = "auto";
        /**
         * Position the tooltip on the bottom of the chart.
         */
        TooltipPosition["BOTTOM"] = "bottom";
        /**
         * Positions the tooltip to the right of the chart.
         */
        TooltipPosition["RIGHT"] = "right";
    })(TooltipPosition || (TooltipPosition = {}));
    const DEFAULT_TOOLTIP_STYLE = {
        boxShadow: '0 1px 4px rgba(0, 0, 0, .3)',
        opacity: 0,
        position: 'fixed',
        willChange: 'transform',
        zIndex: 5,
    };
    let VzChartTooltip = class VzChartTooltip extends LegacyElementMixin(PolymerElement) {
        constructor() {
            super(...arguments);
            this.position = TooltipPosition.AUTO;
            this.minDistFromEdge = 15;
            this._styleCache = null;
            this._raf = null;
            this._tunnel = null;
        }
        ready() {
            this._styleCache = null;
            this._raf = null;
            this._tunnel = null;
        }
        attached() {
            this._tunnel = this._createTunnel();
            this._hideOnBlur = () => {
                if (document.hidden)
                    this.hide();
            };
            window.addEventListener('visibilitychange', this._hideOnBlur);
        }
        detached() {
            this.hide();
            this._removeTunnel(this._tunnel);
            this._tunnel = null;
            window.removeEventListener('visibilitychange', this._hideOnBlur);
        }
        content() {
            return this._tunnel.shadowRoot;
        }
        hide() {
            window.cancelAnimationFrame(this._raf);
            this._styleCache = null;
            this._tunnel.style.opacity = 0;
        }
        /**
         * CSS Scopes the newly added DOM (in most tooltip where columns are
         * invariable, only newly added rows are necessary to be scoped) and positions
         * the tooltip with respect to the anchorNode.
         */
        updateAndPosition(anchorNode) {
            window.cancelAnimationFrame(this._raf);
            this._raf = window.requestAnimationFrame(() => {
                if (!this.isAttached)
                    return;
                this._repositionImpl(anchorNode);
            });
        }
        _repositionImpl(anchorNode) {
            const tooltipContent = this._tunnel;
            const nodeRect = anchorNode.getBoundingClientRect();
            const tooltipRect = tooltipContent.getBoundingClientRect();
            const viewportHeight = window.innerHeight;
            const documentWidth = document.body.clientWidth;
            const anchorTop = nodeRect.top;
            const anchorBottom = anchorTop + nodeRect.height;
            const effectiveTooltipHeight = tooltipRect.height + TOOLTIP_Y_PIXEL_OFFSET;
            let bottom = null;
            let left = Math.max(this.minDistFromEdge, nodeRect.left);
            let right = null;
            let top = anchorTop;
            if (this.position == TooltipPosition.RIGHT) {
                left = nodeRect.right;
            }
            else {
                top = anchorBottom + TOOLTIP_Y_PIXEL_OFFSET;
                // prevent it from falling off the right side of the screen.
                if (documentWidth < left + tooltipRect.width + this.minDistFromEdge) {
                    left = null;
                    right = this.minDistFromEdge;
                }
            }
            // If there is not enough space to render tooltip below the anchorNode in
            // the viewport and there is enough space above, place it above the
            // anchorNode.
            if (this.position == TooltipPosition.AUTO &&
                nodeRect.top - effectiveTooltipHeight > 0 &&
                viewportHeight < nodeRect.top + nodeRect.height + effectiveTooltipHeight) {
                top = null;
                bottom = viewportHeight - anchorTop + TOOLTIP_Y_PIXEL_OFFSET;
            }
            const newStyle = {
                contain: 'content',
                opacity: 1,
                left: left ? `${left}px` : null,
                right: right ? `${right}px` : null,
                top: top ? `${top}px` : null,
                bottom: bottom ? `${bottom}px` : null,
            };
            // Do not update the style (which can cause re-layout) if it has not
            // changed.
            if (!lodash.isEqual(this._styleCache, newStyle)) {
                Object.assign(tooltipContent.style, newStyle);
                this._styleCache = newStyle;
            }
        }
        _createTunnel() {
            if (!this.contentComponentName) {
                throw new RangeError('Require `contentComponentName` to be a name of a Polymer component');
            }
            const tunnel = document.createElement(this.contentComponentName);
            Object.assign(tunnel.style, DEFAULT_TOOLTIP_STYLE);
            document.body.appendChild(tunnel);
            return tunnel;
        }
        _removeTunnel(tunnel) {
            document.body.removeChild(tunnel);
        }
    };
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], VzChartTooltip.prototype, "contentComponentName", void 0);
    __decorate([
        property({
            type: String,
        }),
        __metadata("design:type", String)
    ], VzChartTooltip.prototype, "position", void 0);
    __decorate([
        property({
            type: Number,
        }),
        __metadata("design:type", Number)
    ], VzChartTooltip.prototype, "minDistFromEdge", void 0);
    VzChartTooltip = __decorate([
        customElement('vz-chart-tooltip')
    ], VzChartTooltip);

    /* Copyright 2018 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    class LinearScale extends src$2.Scales.Linear {
        constructor() {
            super();
            this._ignoreOutlier = false;
            this.padProportion(0.2);
        }
        setValueProviderForDomain(provider) {
            this._valueProviderForDomain = provider;
            return this;
        }
        /**
         * TODO(nickfelt): Examine whether we truly require `c`.
         * Adds some padding to a given domain. Specifically, it:
         * - returns about [-0.1a - c, 2.1a + c] when a = b and a >= 0.
         * - returns about [-2.1|a| - c, -0.1|a| + c] when a = b and a < 0.
         * - returns [-0.1b, b + padProportion * (b-a)] if b > 2a and a > 0
         * - else, pads by `padProportion`
         * Note that `c` is a constant offset which specifically is 1.1. Please refer
         * to [1] for its rationale.
         * @override
         */
        _niceDomain(domain, count) {
            const [a, b] = domain;
            let padding;
            const span = b - a;
            if (span === 0) {
                // If b===a, we would create an empty range. We instead select the range
                // [0, 2*a] if a > 0, or [-2*a, 0] if a < 0, plus a little bit of
                // extra padding on the top and bottom of the plot.
                padding = Math.abs(a) * 1.1 + 1.1;
            }
            else {
                padding = span * this.padProportion();
            }
            let lower;
            if (a >= 0 && a < span) {
                // [1]: We include the intercept (y = 0) if doing so less than doubles the
                // span of the y-axis. (We actually select a lower bound that's slightly
                // less than 0 so that 0.00 will clearly be written on the lower edge of
                // the chart. The label on the lowest tick is often filtered out.)
                lower = -0.1 * b;
            }
            else {
                lower = a - padding;
            }
            return super._niceDomain([lower, b + padding], count);
        }
        /**
         * @override to remove default padding logic.
         */
        _getUnboundedExtent(ignoreAttachState) {
            const includedValues = this._getAllIncludedValues(ignoreAttachState);
            let extent = this._defaultExtent();
            if (includedValues.length !== 0) {
                const combinedExtent = [
                    src$2.Utils.Math.min(includedValues, extent[0]),
                    src$2.Utils.Math.max(includedValues, extent[1]),
                ];
                extent = this._niceDomain(combinedExtent);
            }
            return extent;
        }
        /**
         * @override
         */
        _getAllIncludedValues(ignoreAttachState = false) {
            const values = this._valueProviderForDomain
                ? this._valueProviderForDomain()
                : [];
            return this.extentOfValues(values);
        }
        /**
         * @override to apply the outlier logic.
         */
        extentOfValues(values) {
            const legalValues = values.filter((x) => src$2.Utils.Math.isValidNumber(x));
            let filteredValues = legalValues;
            if (this.ignoreOutlier()) {
                const sortedValues = legalValues.sort((a, b) => a - b);
                const a = threshold(sortedValues, 0.05);
                const b = threshold(sortedValues, 0.95);
                filteredValues = legalValues.filter((x) => x >= a && x <= b);
            }
            const extent$1 = extent(filteredValues);
            return extent$1[0] == null || extent$1[1] == null ? [] : extent$1;
        }
        ignoreOutlier(ignore) {
            if (typeof ignore == 'boolean') {
                this._ignoreOutlier = ignore;
                return this;
            }
            return this._ignoreOutlier;
        }
    }

    /* Copyright 2018 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    class TfScale extends src$2.QuantitativeScale {
        constructor() {
            super(...arguments);
            this._ignoreOutlier = false;
        }
        setValueProviderForDomain(provider) {
            this._valueProviderForDomain = provider;
            return this;
        }
        ignoreOutlier(ignore) {
            if (typeof ignore == 'boolean') {
                this._ignoreOutlier = ignore;
                return this;
            }
            return this._ignoreOutlier;
        }
        /**
         * Returns possible `extent`s for a dataset. Note that a dataset can contain
         * multiple series. Unlike the method name suggests, it uses values from each
         * series to return `extent`s.
         * @override
         */
        _getAllIncludedValues(ignoreAttachState = false) {
            const values = this._valueProviderForDomain
                ? this._valueProviderForDomain()
                : [];
            return this.extentOfValues(values);
        }
    }

    /* Copyright 2018 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    // Smallest positive non-zero value represented by IEEE 754 binary (64 bit)	import * as d3 from 'd3';
    // floating-point number.
    // https://www.ecma-international.org/ecma-262/5.1/#sec-8.5
    const MIN_POSITIVE_VALUE = Math.pow(2, -1074);
    function log$4(x) {
        return Math.log10(x);
    }
    function pow$4(x) {
        return Math.pow(10, x);
    }
    /**
     * A logarithmic scale that returns NaN for all non-positive values as it
     * mathematically is supposed to be -Infinity. Also, due to the floating point
     * precision issue, it treats all values smaller than MIN_POSITIVE_VALUE as
     * non-positive. Lastly, if using autoDomain feature and if all values are the
     * same value, it pads 10% of the value.
     */
    class LogScale extends TfScale {
        constructor() {
            super();
            this._d3LogScale = log$1();
            this.padProportion(0.2);
        }
        scale(x) {
            // Returning NaN makes sure line plot does not plot illegal values.
            if (x <= 0)
                return NaN;
            return this._d3LogScale(x);
        }
        invert(x) {
            return this._d3LogScale.invert(x);
        }
        scaleTransformation(value) {
            return this.scale(value);
        }
        invertedTransformation(value) {
            return this.invert(value);
        }
        getTransformationDomain() {
            return this.domain();
        }
        setTransformationDomain(domain) {
            this.domain(domain);
        }
        getTransformationExtent() {
            return this._getUnboundedExtent(true);
        }
        _getDomain() {
            return this._untransformedDomain;
        }
        _setDomain(values) {
            this._untransformedDomain = values;
            const [min, max] = values;
            super._setDomain([Math.max(MIN_POSITIVE_VALUE, min), max]);
        }
        /**
         * Given a domain, pad it and clip the lower bound to MIN_POSITIVE_VALUE.
         */
        _niceDomain(domain, count) {
            const [low, high] = domain;
            const adjustedLogLow = Math.max(log$4(MIN_POSITIVE_VALUE), log$4(low));
            const logHigh = log$4(high);
            const spread = logHigh - adjustedLogLow;
            const pad = spread ? spread * this.padProportion() : 1;
            return [
                pow$4(Math.max(log$4(MIN_POSITIVE_VALUE), adjustedLogLow - pad)),
                pow$4(logHigh + pad),
            ];
        }
        /**
         * Generates a possible extent based on data from all plots the scale is
         * connected to by taking the minimum and maximum values of all extents for
         * lower and upper bound, respectively.
         * @override to remove default padding logic.
         */
        _getUnboundedExtent(ignoreAttachState) {
            const includedValues = this._getAllIncludedValues(ignoreAttachState);
            let extent = this._defaultExtent();
            if (includedValues.length !== 0) {
                const combinedExtent = [
                    src$2.Utils.Math.min(includedValues, extent[0]),
                    src$2.Utils.Math.max(includedValues, extent[1]),
                ];
                extent = this._niceDomain(combinedExtent);
            }
            return extent;
        }
        _getAllIncludedValues(ignoreAttachState = false) {
            const values = super._getAllIncludedValues();
            // For log scale, the value cannot be smaller or equal to 0. They are
            // negative infinity.
            return values.map((x) => (x > 0 ? x : MIN_POSITIVE_VALUE));
        }
        _defaultExtent() {
            return [1, 10];
        }
        _backingScaleDomain(values) {
            if (values == null) {
                return this._d3LogScale.domain();
            }
            else {
                this._d3LogScale.domain(values);
                return this;
            }
        }
        _getRange() {
            return this._d3LogScale.range();
        }
        _setRange(values) {
            this._d3LogScale.range(values);
        }
        defaultTicks() {
            return this._d3LogScale.ticks(1);
        }
        ticks() {
            return this._d3LogScale.ticks();
        }
        /**
         * Returns an `extent` for a data series. In log-scale, we must omit all
         * non-positive values when computing a `domain`.
         * @override
         */
        extentOfValues(values) {
            // Log can only take positive values.
            const legalValues = values.filter((x) => src$2.Utils.Math.isValidNumber(x) && x > 0);
            let filteredValues = legalValues;
            if (this.ignoreOutlier()) {
                const logValues = legalValues.map(log$4);
                const sortedLogValues = logValues.sort((a, b) => a - b);
                const a = threshold(sortedLogValues, 0.05);
                const b = threshold(sortedLogValues, 0.95);
                filteredValues = sortedLogValues.filter((x) => x >= a && x <= b).map(pow$4);
            }
            const extent$1 = extent(filteredValues);
            return extent$1[0] == null || extent$1[1] == null ? [] : extent$1;
        }
    }

    /* Copyright 2015 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the 'License');
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an 'AS IS' BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    class DragZoomLayer extends src$2.Components.SelectionBoxLayer {
        /**
         * Constructs a SelectionBoxLayer with an attached DragInteraction and
         * ClickInteraction. On drag, it triggers an animated zoom into the box
         * that was dragged. On double click, it zooms back out to the original
         * view, before any zooming.
         * The zoom animation uses an easing function (default
         * d3.ease('cubic-in-out')) and is customizable.
         * Usage: Construct the selection box layer and attach x and y scales,
         * and then add the layer over the plot you are zooming on using a
         * Component Group.
         * TODO(@decentralion) - merge this into Plottable
         */
        constructor(xScale, yScale, unzoomMethod) {
            super();
            this.easeFn = cubicInOut;
            this._animationTime = 750;
            this.xScale(xScale);
            this.yScale(yScale);
            this._dragInteraction = new src$2.Interactions.Drag();
            this._doubleClickInteraction = new src$2.Interactions.Click();
            this.setupCallbacks();
            this.unzoomMethod = unzoomMethod;
            // Activate interaction only when the component is mounted onto DOM.
            this.onDetach(() => {
                this._doubleClickInteraction.detachFrom(this);
                this._dragInteraction.detachFrom(this);
            });
            this.onAnchor(() => {
                this._doubleClickInteraction.attachTo(this);
                this._dragInteraction.attachTo(this);
            });
        }
        /**
         * Register a method that calls when the DragZoom interaction starts.
         */
        interactionStart(cb) {
            this.onStart = cb;
        }
        /**
         * Register a method that calls when the DragZoom interaction ends.
         */
        interactionEnd(cb) {
            this.onEnd = cb;
        }
        /**
         * Returns backing drag interaction. Useful for customization to the
         * interaction.
         */
        dragInteraction() {
            return this._dragInteraction;
        }
        setupCallbacks() {
            let dragging = false;
            this._dragInteraction.onDragStart((startPoint) => {
                this.bounds({
                    topLeft: startPoint,
                    bottomRight: startPoint,
                });
                this.onStart();
            });
            this._dragInteraction.onDrag((startPoint, endPoint) => {
                this.bounds({ topLeft: startPoint, bottomRight: endPoint });
                this.boxVisible(true);
                dragging = true;
            });
            this._dragInteraction.onDragEnd((startPoint, endPoint) => {
                this.boxVisible(false);
                this.bounds({ topLeft: startPoint, bottomRight: endPoint });
                if (dragging) {
                    this.zoom();
                }
                else {
                    this.onEnd();
                }
                dragging = false;
            });
            this._doubleClickInteraction.onDoubleClick(this.unzoom.bind(this));
        }
        animationTime(animationTime) {
            if (animationTime == null) {
                return this._animationTime;
            }
            if (animationTime < 0) {
                throw new Error('animationTime cannot be negative');
            }
            this._animationTime = animationTime;
            return this;
        }
        /**
         * Set the easing function, which determines how the zoom interpolates
         * over time.
         */
        ease(fn) {
            if (typeof fn !== 'function') {
                throw new Error('ease function must be a function');
            }
            if (fn(0) !== 0 || fn(1) !== 1) {
                src$2.Utils.Window.warn('Easing function does not maintain invariant ' +
                    'f(0)==0 && f(1)==1. Bad behavior may result.');
            }
            this.easeFn = fn;
            return this;
        }
        // Zoom into extent of the selection box bounds
        zoom() {
            let x0 = this.xExtent()[0].valueOf();
            let x1 = this.xExtent()[1].valueOf();
            let y0 = this.yExtent()[1].valueOf();
            let y1 = this.yExtent()[0].valueOf();
            if (x0 === x1 || y0 === y1) {
                return;
            }
            this.interpolateZoom(x0, x1, y0, y1);
        }
        // Restore the scales to their state before any zoom
        unzoom() {
            // We need to reset the zoom domain unconditionally, as the data or the
            // smoothing may have updated, such that we are not longer fully zoomed out.
            let xScale = this.xScale();
            xScale._domainMin = null;
            xScale._domainMax = null;
            let xDomain = xScale._getExtent();
            this.xScale().domain(xDomain);
            this.unzoomMethod();
        }
        // If we are zooming, disable interactions, to avoid contention
        isZooming(isZooming) {
            this._dragInteraction.enabled(!isZooming);
            this._doubleClickInteraction.enabled(!isZooming);
        }
        interpolateZoom(x0f, x1f, y0f, y1f) {
            let x0s = this.xScale().domain()[0].valueOf();
            let x1s = this.xScale().domain()[1].valueOf();
            let y0s = this.yScale().domain()[0].valueOf();
            let y1s = this.yScale().domain()[1].valueOf();
            // Copy a ref to the ease fn, so that changing ease wont affect zooms in
            // progress.
            let ease = this.easeFn;
            let interpolator = (a, b, p) => interpolateNumber(a, b)(ease(p));
            this.isZooming(true);
            let start = Date.now();
            let draw = () => {
                let now = Date.now();
                let passed = now - start;
                let p = this._animationTime === 0
                    ? 1
                    : Math.min(1, passed / this._animationTime);
                let x0 = interpolator(x0s, x0f, p);
                let x1 = interpolator(x1s, x1f, p);
                let y0 = interpolator(y0s, y0f, p);
                let y1 = interpolator(y1s, y1f, p);
                this.xScale().domain([x0, x1]);
                this.yScale().domain([y0, y1]);
                if (p < 1) {
                    src$2.Utils.DOM.requestAnimationFramePolyfill(draw);
                }
                else {
                    this.onEnd();
                    this.isZooming(false);
                }
            };
            draw();
        }
    }

    /* Copyright 2018 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    var State;
    (function (State) {
        State[State["NONE"] = 0] = "NONE";
        State[State["DRAG_ZOOMING"] = 1] = "DRAG_ZOOMING";
        State[State["PANNING"] = 2] = "PANNING";
    })(State || (State = {}));
    class PanZoomDragLayer extends src$2.Components.Group {
        /**
         * A Plottable component/layer with a complex interaction for the line chart.
         * When not pressing alt-key, it behaves like DragZoomLayer -- dragging a
         * region zooms the area under the gray box and double clicking resets the
         * zoom. When pressing alt-key, it lets user pan around while having mousedown
         * on the chart and let user zoom-in/out of cursor when scroll with alt key
         * pressed.
         */
        constructor(xScale, yScale, unzoomMethod) {
            super();
            this.state = State.NONE;
            this.panStartCallback = new src$2.Utils.CallbackSet();
            this.panEndCallback = new src$2.Utils.CallbackSet();
            this.panZoom = new src$2.Interactions.PanZoom(xScale, yScale);
            this.panZoom.dragInteraction().mouseFilter((event) => {
                return PanZoomDragLayer.isPanKey(event) && event.button === 0;
            });
            this.panZoom.wheelFilter(this.canScrollZoom);
            this.dragZoomLayer = new DragZoomLayer(xScale, yScale, unzoomMethod);
            this.dragZoomLayer.dragInteraction().mouseFilter((event) => {
                return !PanZoomDragLayer.isPanKey(event) && event.button === 0;
            });
            this.append(this.dragZoomLayer);
            const onWheel = this.onWheel.bind(this);
            this.onAnchor(() => {
                this._mouseDispatcher = src$2.Dispatchers.Mouse.getDispatcher(this);
                this._mouseDispatcher.onWheel(onWheel);
                this.panZoom.attachTo(this);
            });
            this.onDetach(() => {
                this.panZoom.detachFrom(this);
                // onDetach can be invoked before onAnchor
                if (this._mouseDispatcher) {
                    this._mouseDispatcher.offWheel(onWheel);
                    this._mouseDispatcher = null;
                }
            });
            this.panZoom.dragInteraction().onDragStart(() => {
                if (this.state == State.NONE)
                    this.setState(State.PANNING);
            });
            this.panZoom.dragInteraction().onDragEnd(() => {
                if (this.state == State.PANNING)
                    this.setState(State.NONE);
            });
            this.dragZoomLayer.dragInteraction().onDragStart(() => {
                if (this.state == State.NONE)
                    this.setState(State.DRAG_ZOOMING);
            });
            this.dragZoomLayer.dragInteraction().onDragEnd(() => {
                if (this.state == State.DRAG_ZOOMING)
                    this.setState(State.NONE);
            });
        }
        onWheel(_, event) {
            if (this.canScrollZoom(event))
                return;
            const helpContainer = this.element();
            if (!helpContainer.select('.help').empty())
                return;
            const help = helpContainer.append('div').classed('help', true);
            help.append('span').text('Alt + Scroll to Zoom');
            // Please see vz-pan-zoom-style for the definition of the animation.
            help.on('animationend', () => void help.remove());
        }
        static isPanKey(event) {
            return Boolean(event.altKey) || Boolean(event.shiftKey);
        }
        canScrollZoom(event) {
            return event.altKey;
        }
        setState(nextState) {
            if (this.state == nextState)
                return;
            const prevState = this.state;
            this.state = nextState;
            this.root().removeClass(this.stateClassName(prevState));
            this.root().addClass(this.stateClassName(nextState));
            if (prevState == State.PANNING) {
                this.panEndCallback.callCallbacks();
            }
            if (nextState == State.PANNING) {
                this.panStartCallback.callCallbacks();
            }
        }
        stateClassName(state) {
            switch (state) {
                case State.PANNING:
                    return 'panning';
                case State.DRAG_ZOOMING:
                    return 'drag-zooming';
                case State.NONE:
                default:
                    return '';
            }
        }
        onPanStart(cb) {
            this.panStartCallback.add(cb);
        }
        onPanEnd(cb) {
            this.panEndCallback.add(cb);
        }
        onScrollZoom(cb) {
            this.panZoom.onZoomEnd(cb);
        }
        onDragZoomStart(cb) {
            this.dragZoomLayer.interactionStart(cb);
        }
        onDragZoomEnd(cb) {
            this.dragZoomLayer.interactionEnd(cb);
        }
    }

    /* Copyright 2018 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    var TooltipColumnEvalType;
    (function (TooltipColumnEvalType) {
        TooltipColumnEvalType[TooltipColumnEvalType["TEXT"] = 0] = "TEXT";
        TooltipColumnEvalType[TooltipColumnEvalType["DOM"] = 1] = "DOM";
    })(TooltipColumnEvalType || (TooltipColumnEvalType = {}));
    var YScaleType;
    (function (YScaleType) {
        YScaleType["LOG"] = "log";
        YScaleType["LINEAR"] = "linear";
    })(YScaleType || (YScaleType = {}));
    /**
     * The maximum number of marker symbols within any line for a data series. Too
     * many markers clutter the chart.
     */
    const _MAX_MARKERS = 20;
    class LineChart {
        constructor(xComponentsCreationMethod, yValueAccessor, yScaleType, colorScale, tooltip, // VzChartTooltip
        tooltipColumns, fillArea, defaultXRange, defaultYRange, symbolFunction, xAxisFormatter) {
            this.dirtyDatasets = new Set();
            this.seriesNames = [];
            this.name2datasets = {};
            this.colorScale = colorScale;
            this.tooltip = tooltip;
            this.datasets = [];
            this._ignoreYOutliers = false;
            // lastPointDataset is a dataset that contains just the last point of
            // every dataset we're currently drawing.
            this.lastPointsDataset = new src$2.Dataset();
            this.nanDataset = new src$2.Dataset();
            this.yValueAccessor = yValueAccessor;
            // The symbol function maps series to marker. It uses a special dataset that
            // varies based on whether smoothing is enabled.
            this.symbolFunction = symbolFunction;
            this._defaultXRange = defaultXRange;
            this._defaultYRange = defaultYRange;
            this.tooltipColumns = tooltipColumns;
            this.buildChart(xComponentsCreationMethod, yValueAccessor, yScaleType, fillArea, xAxisFormatter);
        }
        buildChart(xComponentsCreationMethod, yValueAccessor, yScaleType, fillArea, xAxisFormatter) {
            this.destroy();
            const xComponents = xComponentsCreationMethod();
            this.xAccessor = xComponents.accessor;
            this.xScale = xComponents.scale;
            this.xAxis = xComponents.axis;
            this.xAxis.margin(0).tickLabelPadding(3);
            if (xAxisFormatter) {
                this.xAxis.formatter(xAxisFormatter);
            }
            this.yScale = LineChart.getYScaleFromType(yScaleType);
            this.yScale.setValueProviderForDomain(() => this.getValuesForYAxisDomainCompute());
            this.yAxis = new src$2.Axes.Numeric(this.yScale, 'left');
            let yFormatter = multiscaleFormatter(Y_AXIS_FORMATTER_PRECISION);
            this.yAxis.margin(0)
                .tickLabelPadding(5)
                .formatter(yFormatter);
            this.yAxis.usesTextWidthApproximation(true);
            this.fillArea = fillArea;
            const panZoomLayer = new PanZoomDragLayer(this.xScale, this.yScale, () => this.resetDomain());
            this.tooltipInteraction = this.createTooltipInteraction(panZoomLayer);
            this.tooltipPointsComponent = new src$2.Component();
            const plot = this.buildPlot(this.xScale, this.yScale, fillArea);
            this.gridlines = new src$2.Components.Gridlines(this.xScale, this.yScale);
            let xZeroLine = null;
            if (yScaleType !== YScaleType.LOG) {
                xZeroLine = new src$2.Components.GuideLineLayer('horizontal');
                xZeroLine.scale(this.yScale).value(0);
            }
            let yZeroLine = new src$2.Components.GuideLineLayer('vertical');
            yZeroLine.scale(this.xScale).value(0);
            this.center = new src$2.Components.Group([
                this.gridlines,
                xZeroLine,
                yZeroLine,
                plot,
                this.tooltipPointsComponent,
                panZoomLayer,
            ]);
            this.center.addClass('main');
            this.outer = new src$2.Components.Table([
                [this.yAxis, this.center],
                [null, this.xAxis],
            ]);
        }
        buildPlot(xScale, yScale, fillArea) {
            if (fillArea) {
                this.marginAreaPlot = new src$2.Plots.Area();
                this.marginAreaPlot.x(this.xAccessor, xScale);
                this.marginAreaPlot.y(fillArea.higherAccessor, yScale);
                this.marginAreaPlot.y0(fillArea.lowerAccessor);
                this.marginAreaPlot.attr('fill', (d, i, dataset) => this.colorScale.scale(dataset.metadata().name));
                this.marginAreaPlot.attr('fill-opacity', 0.3);
                this.marginAreaPlot.attr('stroke-width', 0);
            }
            this.smoothedAccessor = (d) => d.smoothed;
            let linePlot = new src$2.Plots.Line();
            linePlot.x(this.xAccessor, xScale);
            linePlot.y(this.yValueAccessor, yScale);
            linePlot.attr('stroke', (d, i, dataset) => this.colorScale.scale(dataset.metadata().name));
            this.linePlot = linePlot;
            this.setupTooltips(linePlot);
            let smoothLinePlot = new src$2.Plots.Line();
            smoothLinePlot.x(this.xAccessor, xScale);
            smoothLinePlot.y(this.smoothedAccessor, yScale);
            smoothLinePlot.attr('stroke', (d, i, dataset) => this.colorScale.scale(dataset.metadata().name));
            this.smoothLinePlot = smoothLinePlot;
            if (this.symbolFunction) {
                const markersScatterPlot = new src$2.Plots.Scatter();
                markersScatterPlot.x(this.xAccessor, xScale);
                markersScatterPlot.y(this.yValueAccessor, yScale);
                markersScatterPlot.attr('fill', (d, i, dataset) => this.colorScale.scale(dataset.metadata().name));
                markersScatterPlot.attr('opacity', 1);
                markersScatterPlot.size(TOOLTIP_CIRCLE_SIZE * 2);
                markersScatterPlot.symbol((d, i, dataset) => {
                    return this.symbolFunction(dataset.metadata().name);
                });
                // Use a special dataset because this scatter plot should use the accesor
                // that depends on whether smoothing is enabled.
                this.markersScatterPlot = markersScatterPlot;
            }
            // The scatterPlot will display the last point for each dataset.
            // This way, if there is only one datum for the series, it is still
            // visible. We hide it when tooltips are active to keep things clean.
            let scatterPlot = new src$2.Plots.Scatter();
            scatterPlot.x(this.xAccessor, xScale);
            scatterPlot.y(this.yValueAccessor, yScale);
            scatterPlot.attr('fill', (d) => this.colorScale.scale(d.name));
            scatterPlot.attr('opacity', 1);
            scatterPlot.size(TOOLTIP_CIRCLE_SIZE * 2);
            scatterPlot.datasets([this.lastPointsDataset]);
            this.scatterPlot = scatterPlot;
            let nanDisplay = new src$2.Plots.Scatter();
            nanDisplay.x(this.xAccessor, xScale);
            nanDisplay.y((x) => x.displayY, yScale);
            nanDisplay.attr('fill', (d) => this.colorScale.scale(d.name));
            nanDisplay.attr('opacity', 1);
            nanDisplay.size(NAN_SYMBOL_SIZE * 2);
            nanDisplay.datasets([this.nanDataset]);
            nanDisplay.symbol(src$2.SymbolFactories.triangle);
            this.nanDisplay = nanDisplay;
            const groups = [nanDisplay, scatterPlot, smoothLinePlot, linePlot];
            if (this.marginAreaPlot) {
                groups.push(this.marginAreaPlot);
            }
            if (this.markersScatterPlot) {
                groups.push(this.markersScatterPlot);
            }
            return new src$2.Components.Group(groups);
        }
        ignoreYOutliers(ignoreYOutliers) {
            if (ignoreYOutliers !== this._ignoreYOutliers) {
                this._ignoreYOutliers = ignoreYOutliers;
                this.updateSpecialDatasets();
                this.yScale.ignoreOutlier(ignoreYOutliers);
                this.resetYDomain();
            }
        }
        getValuesForYAxisDomainCompute() {
            const accessors = this.getAccessorsForComputingYRange();
            let datasetToValues = (d) => {
                return accessors.map((accessor) => d.data().map((x) => accessor(x, -1, d)));
            };
            return lodash.flattenDeep(this.datasets.map(datasetToValues)).filter(isFinite);
        }
        /** Constructs special datasets. Each special dataset contains exceptional
         * values from all of the regular datasets, e.g. last points in series, or
         * NaN values. Those points will have a `name` and `relative` property added
         * (since usually those are context in the surrounding dataset).
         */
        updateSpecialDatasets() {
            const accessor = this.getYAxisAccessor();
            let lastPointsData = this.datasets
                .map((d) => {
                let datum = null;
                // filter out NaNs to ensure last point is a clean one
                let nonNanData = d.data().filter((x) => !isNaN(accessor(x, -1, d)));
                if (nonNanData.length > 0) {
                    let idx = nonNanData.length - 1;
                    datum = nonNanData[idx];
                    datum.name = d.metadata().name;
                    datum.relative = relativeAccessor(datum, -1, d);
                }
                return datum;
            })
                .filter((x) => x != null);
            this.lastPointsDataset.data(lastPointsData);
            if (this.markersScatterPlot) {
                this.markersScatterPlot.datasets(this.datasets.map(this.createSampledDatasetForMarkers));
            }
            // Take a dataset, return an array of NaN data points
            // the NaN points will have a "displayY" property which is the
            // y-value of a nearby point that was not NaN (0 if all points are NaN)
            let datasetToNaNData = (d) => {
                let displayY = null;
                let data = d.data();
                let i = 0;
                while (i < data.length && displayY == null) {
                    if (!isNaN(accessor(data[i], -1, d))) {
                        displayY = accessor(data[i], -1, d);
                    }
                    i++;
                }
                if (displayY == null) {
                    displayY = 0;
                }
                let nanData = [];
                for (i = 0; i < data.length; i++) {
                    if (!isNaN(accessor(data[i], -1, d))) {
                        displayY = accessor(data[i], -1, d);
                    }
                    else {
                        data[i].name = d.metadata().name;
                        data[i].displayY = displayY;
                        data[i].relative = relativeAccessor(data[i], -1, d);
                        nanData.push(data[i]);
                    }
                }
                return nanData;
            };
            let nanData = lodash.flatten(this.datasets.map(datasetToNaNData));
            this.nanDataset.data(nanData);
        }
        resetDomain() {
            this.resetXDomain();
            this.resetYDomain();
        }
        resetXDomain() {
            let xDomain;
            if (this._defaultXRange != null) {
                // Use the range specified by the caller.
                xDomain = this._defaultXRange;
            }
            else {
                // (Copied from vz_line_chart.DragZoomLayer.unzoom.)
                const xScale = this.xScale;
                xScale._domainMin = null;
                xScale._domainMax = null;
                xDomain = xScale._getExtent();
            }
            this.xScale.domain(xDomain);
        }
        resetYDomain() {
            if (this._defaultYRange != null) {
                // Use the range specified by the caller.
                this.yScale.domain(this._defaultYRange);
            }
            else {
                // TfScale has all the logics for scaling and we manually trigger it with
                // `autoDomain`. However, this enables the autoDomain mode which updates
                // the domain on any dataset change and this is not desirably especially
                // when a run is not finished yet; we don't want the graph to change in
                // scale while user is inspecting the graph. By setting the `domain`
                // explicitly, we can turn the feature off.
                this.yScale.autoDomain();
                this.yScale.domain(this.yScale.domain());
            }
        }
        getAccessorsForComputingYRange() {
            const accessors = [this.getYAxisAccessor()];
            if (this.fillArea) {
                // Make the Y domain take margins into account.
                accessors.push(this.fillArea.lowerAccessor, this.fillArea.higherAccessor);
            }
            return accessors;
        }
        getYAxisAccessor() {
            return this.smoothingEnabled ? this.smoothedAccessor : this.yValueAccessor;
        }
        createTooltipInteraction(pzdl) {
            const pi = new PointerInteraction();
            // Disable interaction while drag zooming.
            const disableTooltipUpdate = () => {
                pi.enabled(false);
                this.hideTooltips();
            };
            const enableTooltipUpdate = () => pi.enabled(true);
            pzdl.onPanStart(disableTooltipUpdate);
            pzdl.onDragZoomStart(disableTooltipUpdate);
            pzdl.onPanEnd(enableTooltipUpdate);
            pzdl.onDragZoomEnd(enableTooltipUpdate);
            // When using wheel, cursor position does not change. Redraw the tooltip
            // using the last known mouse position.
            pzdl.onScrollZoom(() => this.updateTooltipContent(this._lastMousePosition));
            pi.onPointerMove((p) => {
                this._lastMousePosition = p;
                this.updateTooltipContent(p);
            });
            pi.onPointerExit(() => this.hideTooltips());
            return pi;
        }
        updateTooltipContent(p) {
            // Line plot must be initialized to draw.
            if (!this.linePlot)
                return;
            window.cancelAnimationFrame(this._tooltipUpdateAnimationFrame);
            this._tooltipUpdateAnimationFrame = window.requestAnimationFrame(() => {
                let target = {
                    x: p.x,
                    y: p.y,
                    datum: null,
                    dataset: null,
                };
                let bbox = this.gridlines.content().node().getBBox();
                // pts is the closets point to the tooltip for each dataset
                let pts = this.linePlot
                    .datasets()
                    .map((dataset) => this.findClosestPoint(target, dataset))
                    .filter(Boolean);
                let intersectsBBox = src$2.Utils.DOM.intersectsBBox;
                // We draw tooltips for points that are NaN, or are currently visible
                let ptsForTooltips = pts.filter((p) => intersectsBBox(p.x, p.y, bbox) ||
                    isNaN(this.yValueAccessor(p.datum, 0, p.dataset)));
                // Only draw little indicator circles for the non-NaN points
                let ptsToCircle = ptsForTooltips.filter((p) => !isNaN(this.yValueAccessor(p.datum, 0, p.dataset)));
                if (pts.length !== 0) {
                    this.scatterPlot.attr('display', 'none');
                    const ptsSelection = this.tooltipPointsComponent
                        .content()
                        .selectAll('.point')
                        .data(ptsToCircle, (p) => p.dataset.metadata().name);
                    ptsSelection.enter().append('circle').classed('point', true);
                    ptsSelection
                        .attr('r', TOOLTIP_CIRCLE_SIZE)
                        .attr('cx', (p) => p.x)
                        .attr('cy', (p) => p.y)
                        .style('stroke', 'none')
                        .attr('fill', (p) => this.colorScale.scale(p.dataset.metadata().name));
                    ptsSelection.exit().remove();
                    this.drawTooltips(ptsForTooltips, target, this.tooltipColumns);
                }
                else {
                    this.hideTooltips();
                }
            });
        }
        hideTooltips() {
            window.cancelAnimationFrame(this._tooltipUpdateAnimationFrame);
            this.tooltip.hide();
            this.scatterPlot.attr('display', 'block');
            this.tooltipPointsComponent.content().selectAll('.point').remove();
        }
        setupTooltips(plot) {
            plot.onDetach(() => {
                this.tooltipInteraction.detachFrom(plot);
                this.tooltipInteraction.enabled(false);
            });
            plot.onAnchor(() => {
                this.tooltipInteraction.attachTo(plot);
                this.tooltipInteraction.enabled(true);
            });
        }
        drawTooltips(points, target, tooltipColumns) {
            if (!points.length) {
                this.tooltip.hide();
                return;
            }
            const { colorScale } = this;
            const swatchCol = {
                title: '',
                static: false,
                evalType: TooltipColumnEvalType.DOM,
                evaluate(d) {
                    select(this)
                        .select('span')
                        .style('background-color', () => colorScale.scale(d.dataset.metadata().name));
                    return '';
                },
                enter(d) {
                    select(this)
                        .append('span')
                        .classed('swatch', true)
                        .style('background-color', () => colorScale.scale(d.dataset.metadata().name));
                },
            };
            tooltipColumns = [swatchCol, ...tooltipColumns];
            const dist = (p) => Math.pow(p.x - target.x, 2) + Math.pow(p.y - target.y, 2);
            const closestDist = lodash.min(points.map(dist));
            const valueSortMethod = this.smoothingEnabled
                ? this.smoothedAccessor
                : this.yValueAccessor;
            if (this.tooltipSortingMethod === 'ascending') {
                points = lodash.sortBy(points, (d) => valueSortMethod(d.datum, -1, d.dataset));
            }
            else if (this.tooltipSortingMethod === 'descending') {
                points = lodash.sortBy(points, (d) => valueSortMethod(d.datum, -1, d.dataset)).reverse();
            }
            else if (this.tooltipSortingMethod === 'nearest') {
                points = lodash.sortBy(points, dist);
            }
            else {
                // The 'default' sorting method maintains the order of names passed to
                // setVisibleSeries(). However we reverse that order when defining the
                // datasets. So we must call reverse again to restore the order.
                points = points.slice(0).reverse();
            }
            const self = this;
            const table = select(this.tooltip.content()).select('table');
            const header = table
                .select('thead')
                .selectAll('th')
                .data(tooltipColumns, (column, _, __) => {
                return column.title;
            });
            header
                .enter()
                .append('th')
                .text((col) => col.title)
                .nodes();
            header.exit().remove();
            const rows = table
                .select('tbody')
                .selectAll('tr')
                .data(points, (pt, _, __) => {
                return pt.dataset.metadata().name;
            });
            rows
                .classed('distant', (d) => {
                // Grey out the point if any of the following are true:
                // - The cursor is outside of the x-extent of the dataset
                // - The point's y value is NaN
                let firstPoint = d.dataset.data()[0];
                let lastPoint = lodash.last(d.dataset.data());
                let firstX = this.xScale.scale(this.xAccessor(firstPoint, 0, d.dataset));
                let lastX = this.xScale.scale(this.xAccessor(lastPoint, 0, d.dataset));
                let s = this.smoothingEnabled
                    ? d.datum.smoothed
                    : this.yValueAccessor(d.datum, 0, d.dataset);
                return target.x < firstX || target.x > lastX || isNaN(s);
            })
                .classed('closest', (p) => dist(p) === closestDist)
                .each(function (point) {
                self.drawTooltipRow(this, tooltipColumns, point);
            })
                // reorders DOM to match the ordering of the `data`.
                .order();
            rows.exit().remove();
            rows
                .enter()
                .append('tr')
                .each(function (point) {
                self.drawTooltipRow(this, tooltipColumns, point);
            })
                .nodes();
            this.tooltip.updateAndPosition(this.targetSVG.node());
        }
        drawTooltipRow(row, tooltipColumns, point) {
            const self = this;
            const columns = select(row).selectAll('td').data(tooltipColumns);
            columns.each(function (col) {
                // Skip column value update when the column is static.
                if (col.static)
                    return;
                self.drawTooltipColumn.call(self, this, col, point);
            });
            columns.exit().remove();
            columns
                .enter()
                .append('td')
                .each(function (col) {
                if ('enter' in col && col.enter) {
                    const customTooltip = col;
                    customTooltip.enter.call(this, point);
                }
                self.drawTooltipColumn.call(self, this, col, point);
            });
        }
        drawTooltipColumn(column, tooltipCol, point) {
            const { smoothingEnabled } = this;
            if ('evalType' in tooltipCol &&
                tooltipCol.evalType == TooltipColumnEvalType.DOM) {
                tooltipCol.evaluate.call(column, point, { smoothingEnabled });
            }
            else {
                select(column).text(tooltipCol.evaluate.call(column, point, { smoothingEnabled }));
            }
        }
        findClosestPoint(target, dataset) {
            const xPoints = dataset
                .data()
                .map((d, i) => this.xScale.scale(this.xAccessor(d, i, dataset)));
            let idx = lodash.sortedIndex(xPoints, target.x);
            if (xPoints.length == 0)
                return null;
            if (idx === xPoints.length) {
                idx = idx - 1;
            }
            else if (idx !== 0) {
                const prevDist = Math.abs(xPoints[idx - 1] - target.x);
                const nextDist = Math.abs(xPoints[idx] - target.x);
                idx = prevDist < nextDist ? idx - 1 : idx;
            }
            const datum = dataset.data()[idx];
            const y = this.smoothingEnabled
                ? this.smoothedAccessor(datum, idx, dataset)
                : this.yValueAccessor(datum, idx, dataset);
            return {
                x: xPoints[idx],
                y: this.yScale.scale(y),
                datum,
                dataset,
            };
        }
        resmoothDataset(dataset) {
            let data = dataset.data();
            const smoothingWeight = this.smoothingWeight;
            // 1st-order IIR low-pass filter to attenuate the higher-
            // frequency components of the time-series.
            let last = data.length > 0 ? 0 : NaN;
            let numAccum = 0;
            const yValues = data.map((d, i) => this.yValueAccessor(d, i, dataset));
            // See #786.
            const isConstant = yValues.every((v) => v == yValues[0]);
            data.forEach((d, i) => {
                const nextVal = yValues[i];
                if (isConstant || !Number.isFinite(nextVal)) {
                    d.smoothed = nextVal;
                }
                else {
                    last = last * smoothingWeight + (1 - smoothingWeight) * nextVal;
                    numAccum++;
                    // The uncorrected moving average is biased towards the initial value.
                    // For example, if initialized with `0`, with smoothingWeight `s`, where
                    // every data point is `c`, after `t` steps the moving average is
                    // ```
                    //   EMA = 0*s^(t) + c*(1 - s)*s^(t-1) + c*(1 - s)*s^(t-2) + ...
                    //       = c*(1 - s^t)
                    // ```
                    // If initialized with `0`, dividing by (1 - s^t) is enough to debias
                    // the moving average. We count the number of finite data points and
                    // divide appropriately before storing the data.
                    let debiasWeight = 1;
                    if (smoothingWeight !== 1) {
                        debiasWeight = 1 - Math.pow(smoothingWeight, numAccum);
                    }
                    d.smoothed = last / debiasWeight;
                }
            });
        }
        getDataset(name) {
            if (this.name2datasets[name] === undefined) {
                this.name2datasets[name] = new src$2.Dataset([], {
                    name,
                    meta: null,
                });
            }
            return this.name2datasets[name];
        }
        static getYScaleFromType(yScaleType) {
            if (yScaleType === YScaleType.LOG) {
                return new LogScale();
            }
            else if (yScaleType === YScaleType.LINEAR) {
                return new LinearScale();
            }
            else {
                throw new Error('Unrecognized yScale type ' + yScaleType);
            }
        }
        /**
         * Stages update of visible series on the chart.
         *
         * Please call `commitChanges` for the changes to be reflected on the chart
         * after making all the changes.
         */
        setVisibleSeries(names) {
            this.disableChanges();
            names = names.sort();
            names.reverse(); // draw first series on top
            this.seriesNames = names;
        }
        disableChanges() {
            if (!this.dirtyDatasets.size) {
                // Prevent plots from reacting to the dataset changes.
                this.linePlot.datasets([]);
                if (this.smoothLinePlot) {
                    this.smoothLinePlot.datasets([]);
                }
                if (this.marginAreaPlot) {
                    this.marginAreaPlot.datasets([]);
                }
            }
        }
        commitChanges() {
            this.datasets = this.seriesNames.map((r) => this.getDataset(r));
            [...this.dirtyDatasets].forEach((d) => {
                if (this.smoothingEnabled) {
                    this.resmoothDataset(this.getDataset(d));
                }
            });
            this.updateSpecialDatasets();
            this.linePlot.datasets(this.datasets);
            if (this.smoothingEnabled) {
                this.smoothLinePlot.datasets(this.datasets);
            }
            if (this.marginAreaPlot) {
                this.marginAreaPlot.datasets(this.datasets);
            }
            this.measureBBoxAndMaybeInvalidateLayoutInRaf();
            this.dirtyDatasets.clear();
        }
        /**
         * Samples a dataset so that it contains no more than _MAX_MARKERS number of
         * data points. This function returns the original dataset if it does not
         * exceed that many points.
         */
        createSampledDatasetForMarkers(original) {
            const originalData = original.data();
            if (originalData.length <= _MAX_MARKERS) {
                // This dataset is small enough. Do not sample.
                return original;
            }
            // Downsample the data. Otherwise, too many markers clutter the chart.
            const skipLength = Math.ceil(originalData.length / _MAX_MARKERS);
            const data = new Array(Math.floor(originalData.length / skipLength));
            for (let i = 0, j = 0; i < data.length; i++, j += skipLength) {
                data[i] = originalData[j];
            }
            return new src$2.Dataset(data, original.metadata());
        }
        /**
         * Stages a data change of a series on the chart.
         *
         * Please call `commitChanges` for the changes to be reflected on the chart
         * after making all the changes.
         */
        setSeriesData(name, data) {
            this.disableChanges();
            this.getDataset(name).data(data);
            this.dirtyDatasets.add(name);
        }
        /**
         * Sets a metadata change of a series on the chart.
         *
         * Please call `commitChanges` for the changes to be reflected on the chart
         * after making all the changes.
         */
        setSeriesMetadata(name, meta) {
            this.disableChanges();
            this.getDataset(name).metadata(Object.assign(Object.assign({}, this.getDataset(name).metadata()), { meta }));
            this.dirtyDatasets.add(name);
        }
        smoothingUpdate(weight) {
            this.smoothingWeight = weight;
            this.datasets.forEach((d) => this.resmoothDataset(d));
            if (!this.smoothingEnabled) {
                this.linePlot.addClass('ghost');
                this.scatterPlot.y(this.smoothedAccessor, this.yScale);
                this.smoothingEnabled = true;
                this.smoothLinePlot.datasets(this.datasets);
            }
            if (this.markersScatterPlot) {
                // Use the correct accessor for marker positioning.
                this.markersScatterPlot.y(this.getYAxisAccessor(), this.yScale);
            }
            this.updateSpecialDatasets();
        }
        smoothingDisable() {
            if (this.smoothingEnabled) {
                this.linePlot.removeClass('ghost');
                this.scatterPlot.y(this.yValueAccessor, this.yScale);
                this.smoothLinePlot.datasets([]);
                this.smoothingEnabled = false;
                this.updateSpecialDatasets();
            }
            if (this.markersScatterPlot) {
                // Use the correct accessor (which depends on whether smoothing is
                // enabled) for marker positioning.
                this.markersScatterPlot.y(this.getYAxisAccessor(), this.yScale);
            }
        }
        setColorScale(colorScale) {
            this.colorScale = colorScale;
        }
        setTooltipColumns(tooltipColumns) {
            this.tooltipColumns = tooltipColumns;
        }
        setTooltipSortingMethod(method) {
            this.tooltipSortingMethod = method;
        }
        renderTo(targetSVG) {
            this.targetSVG = targetSVG;
            this.outer.renderTo(targetSVG);
            if (this._defaultXRange != null) {
                // A higher-level component provided a default range for the X axis.
                // Start with that range.
                this.resetXDomain();
            }
            if (this._defaultYRange != null) {
                // A higher-level component provided a default range for the Y axis.
                // Start with that range.
                this.resetYDomain();
            }
            this.measureBBoxAndMaybeInvalidateLayoutInRaf();
        }
        redraw() {
            window.cancelAnimationFrame(this._redrawRaf);
            this._redrawRaf = window.requestAnimationFrame(() => {
                this.measureBBoxAndMaybeInvalidateLayout();
                this.outer.redraw();
            });
        }
        measureBBoxAndMaybeInvalidateLayoutInRaf() {
            window.cancelAnimationFrame(this._invalidateLayoutRaf);
            this._invalidateLayoutRaf = window.requestAnimationFrame(() => {
                this.measureBBoxAndMaybeInvalidateLayout();
            });
        }
        /**
         * Measures bounding box of the anchor node and determines whether the layout
         * needs to be re-done with measurement cache invalidated. Plottable improved
         * performance of rendering by caching expensive DOM measurement but this
         * cache can be poisoned in case the anchor node is in a wrong state -- namely
         * `display: none` where all dimensions are 0.
         */
        measureBBoxAndMaybeInvalidateLayout() {
            if (this._lastDrawBBox) {
                const { width: prevWidth } = this._lastDrawBBox;
                const { width } = this.targetSVG.node().getBoundingClientRect();
                if (prevWidth == 0 && prevWidth < width)
                    this.outer.invalidateCache();
            }
            this._lastDrawBBox = this.targetSVG.node().getBoundingClientRect();
        }
        destroy() {
            // Destroying outer destroys all subcomponents recursively.
            window.cancelAnimationFrame(this._redrawRaf);
            window.cancelAnimationFrame(this._invalidateLayoutRaf);
            if (this.outer)
                this.outer.destroy();
        }
        onAnchor(fn) {
            if (this.outer)
                this.outer.onAnchor(fn);
        }
        /**
         * Returns whether the extent of rendered data values fits the current
         * chart viewport domain (includes smoothing and outlier detection).
         *
         * This is true when there is no data, and false when the domain has been
         * transformed from the extent via transformations (pan, zoom).
         */
        isDataFitToDomain() {
            return (isDataFitToDomain(this.xAxis.getScale()) &&
                isDataFitToDomain(this.yAxis.getScale()));
            function isDataFitToDomain(scale) {
                /**
                 * Domain represents the currently displayed region, possibly a zoomed
                 * in or zoomed out view of the data.
                 *
                 * Extent represents the extent of the data, the range of all provided
                 * datum values.
                 */
                const domain = scale.getTransformationDomain();
                const extent = scale.getTransformationExtent();
                return extent[0] === domain[0] && extent[1] === domain[1];
            }
        }
    }

    /* Copyright 2018 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    var NodeName;
    (function (NodeName) {
        NodeName["GROUP"] = "G";
        NodeName["DIV"] = "DIV";
        NodeName["SVG"] = "SVG";
        NodeName["TEXT"] = "TEXT";
    })(NodeName || (NodeName = {}));
    class PlottableExporter {
        constructor(rootEl) {
            this.uniqueId = 0;
            this.root = rootEl;
        }
        exportAsString() {
            const convertedNodes = this.convert(this.root);
            if (!convertedNodes)
                return '';
            const svg = this.createRootSvg();
            svg.appendChild(convertedNodes);
            return svg.outerHTML;
        }
        createUniqueId(prefix) {
            return `${prefix}_${this.uniqueId++}`;
        }
        getSize() {
            return this.root.getBoundingClientRect();
        }
        createRootSvg() {
            const svg = document.createElement('svg');
            const rect = this.getSize();
            // case on `viewBox` is sensitive.
            svg.setAttributeNS('svg', 'viewBox', `0 0 ${rect.width} ${rect.height}`);
            svg.setAttribute('xmlns', 'http://www.w3.org/2000/svg');
            return svg;
        }
        convert(node) {
            let newNode = null;
            const nodeName = node.nodeName.toUpperCase();
            if (node.nodeType == Node.ELEMENT_NODE &&
                (nodeName == NodeName.DIV || nodeName == NodeName.SVG)) {
                newNode = document.createElement(NodeName.GROUP);
                const style = window.getComputedStyle(node);
                const left = parseInt(style.left, 10);
                const top = parseInt(style.top, 10);
                if (left || top) {
                    const clipId = this.createUniqueId('clip');
                    newNode.setAttribute('transform', `translate(${left}, ${top})`);
                    newNode.setAttribute('clip-path', `url(#${clipId})`);
                    const width = parseInt(style.width, 10);
                    const height = parseInt(style.height, 10);
                    const rect = document.createElement('rect');
                    rect.setAttribute('width', String(width));
                    rect.setAttribute('height', String(height));
                    const clipPath = document.createElementNS('svg', 'clipPath');
                    clipPath.id = clipId;
                    clipPath.appendChild(rect);
                    newNode.appendChild(clipPath);
                }
            }
            else {
                newNode = node.cloneNode();
            }
            Array.from(node.childNodes)
                .map((node) => this.convert(node))
                .filter(Boolean)
                .forEach((el) => newNode.appendChild(el));
            // Remove empty grouping. They add too much noise.
            const shouldOmit = (newNode.nodeName.toUpperCase() == NodeName.GROUP &&
                !newNode.hasChildNodes()) ||
                this.shouldOmitNode(node);
            if (shouldOmit)
                return null;
            return this.stripClass(this.transferStyle(node, newNode));
        }
        stripClass(node) {
            if (node.nodeType == Node.ELEMENT_NODE) {
                node.removeAttribute('class');
            }
            return node;
        }
        transferStyle(origNode, node) {
            if (node.nodeType != Node.ELEMENT_NODE)
                return node;
            const el = node;
            const nodeName = node.nodeName.toUpperCase();
            const style = window.getComputedStyle(origNode);
            if (nodeName == NodeName.TEXT) {
                Object.assign(el.style, {
                    fontFamily: style.fontFamily,
                    fontSize: style.fontSize,
                    fontWeight: style.fontWeight,
                });
            }
            if (nodeName != NodeName.GROUP) {
                el.setAttribute('fill', style.fill);
                el.setAttribute('stroke', style.stroke);
                el.setAttribute('stroke-width', style.strokeWidth);
            }
            if (style.opacity != '1')
                el.setAttribute('opacity', style.opacity);
            return node;
        }
        shouldOmitNode(node) {
            return false;
        }
    }
    class LineChartExporter extends PlottableExporter {
        shouldOmitNode(node) {
            // Scatter plot is useful for tooltip. Tooltip is meaningless in the
            // exported svg.
            if (node.nodeType == Node.ELEMENT_NODE) {
                return node.classList.contains('scatter-plot');
            }
            return false;
        }
    }

    /* Copyright 2016 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    registerStyleDomModule({
        moduleName: 'vz-pan-zoom-style',
        styleContent: `
    .help {
      align-items: center;
      animation-delay: 1s;
      animation-duration: 1s;
      animation-name: fade-out;
      background: rgba(30, 30, 30, 0.6);
      bottom: 0;
      color: #fff;
      display: flex;
      justify-content: center;
      left: 0;
      opacity: 1;
      padding: 20px;
      pointer-events: none;
      position: absolute;
      right: 0;
      top: 0;
    }

    .help > span {
      white-space: normal;
    }

    @keyframes fade-out {
      0% {
        opacity: 1;
      }

      100% {
        opacity: 0;
      }
    }
  `,
    });

    const valueFormatter = multiscaleFormatter(Y_TOOLTIP_FORMATTER_PRECISION);
    const formatValueOrNaN = (x) => (isNaN(x) ? 'NaN' : valueFormatter(x));
    const DEFAULT_TOOLTIP_COLUMNS = [
        {
            title: 'Name',
            evaluate: (d) => d.dataset.metadata().name,
        },
        {
            title: 'Smoothed',
            evaluate(d, statusObject) {
                const { smoothingEnabled } = statusObject;
                return formatValueOrNaN(smoothingEnabled ? d.datum.smoothed : d.datum.scalar);
            },
        },
        {
            title: 'Value',
            evaluate: (d) => formatValueOrNaN(d.datum.scalar),
        },
        {
            title: 'Step',
            evaluate: (d) => stepFormatter(d.datum.step),
        },
        {
            title: 'Time',
            evaluate: (d) => timeFormatter(d.datum.wall_time),
        },
        {
            title: 'Relative',
            evaluate: (d) => relativeFormatter(relativeAccessor(d.datum, -1, d.dataset)),
        },
    ];
    let VzLineChart2 = class VzLineChart2 extends LegacyElementMixin(PolymerElement) {
        constructor() {
            super(...arguments);
            /**
             * Scale that maps series names to colors. The default colors are from
             * d3.schemeCategory10. Use this property to replace the default line
             * colors with colors of your own choice.
             */
            this.colorScale = new src$2.Scales.Color().range(category10.slice(0));
            /**
             * Whether smoothing is enabled or not. If true, smoothed lines will be
             * plotted in the chart while the unsmoothed lines will be ghosted in
             * the background.
             */
            this.smoothingEnabled = false;
            /**
             * Weight (between 0.0 and 1.0) of the smoothing. A value of 0.0
             * means very little smoothing, possibly no smoothing at all. A
             * value of 1.0 means a whole lot of smoothing, possibly so much as
             * to make the whole plot appear as a constant function.
             *
             * Has no effect when `smoothingEnabled` is `false`.
             */
            this.smoothingWeight = 0.6;
            /**
             * This is a helper field for automatically generating commonly used
             * functions for xComponentsCreationMethod. Valid values are what can
             * be processed by vz_chart_helpers.getXComponents() and include
             * "step", "wall_time", and "relative".
             */
            this.xType = null;
            /**
             * We accept a function for creating an XComponents object instead of such
             * an object itself because the Axis must be made right when we make the
             * LineChart object, lest we use a previously destroyed Axis. See the async
             * logic below that uses this property.
             *
             * Note that this function returns a function because polymer calls the
             * outer function to compute the value. We actually want the value of this
             * property to be the inner function.
             */
            this.xComponentsCreationMethod = null;
            /**
             * A method that implements the Plottable.IAccessor<number> interface. Used
             * for accessing the y value from a data point.
             *
             * Note that this function returns a function because polymer calls the
             * outer function to compute the value. We actually want the value of this
             * property to be the inner function.
             */
            this.yValueAccessor = (d) => d.scalar;
            /**
             * An array of ChartHelper.TooltipColumn objects. Used to populate the table
             * within the tooltip. The table contains 1 row per run.
             *
             * Note that this function returns a function because polymer calls the
             * outer function to compute the value. We actually want the value of this
             * property to be the inner function.
             */
            this.tooltipColumns = DEFAULT_TOOLTIP_COLUMNS;
            /**
             * The scale for the y-axis. Allows:
             * - "linear" - linear scale (Plottable.Scales.Linear)
             * - "log" - modified-log scale (Plottable.Scales.ModifiedLog)
             */
            this.yScaleType = YScaleType.LINEAR;
            /**
             * Whether to ignore outlier data when computing the yScale domain.
             */
            this.ignoreYOutliers = false;
            /**
             * Change how the tooltip is sorted. Allows:
             * - "default" - Sort the tooltip by input order.
             * - "ascending" - Sort the tooltip by ascending value.
             * - "descending" - Sort the tooltip by descending value.
             * - "nearest" - Sort the tooltip by closest to cursor.
             */
            this.tooltipSortingMethod = 'default';
            /**
             * Changes how the tooltip is positioned. Allows:
             * - "bottom" - Position the tooltip on the bottom of the chart.
             * - "right" - Position the tooltip to the right of the chart.
             * - "auto" - Position the tooltip to the bottom of the chart in most case.
             *            Position the tooltip above the chart if there isn't sufficient
             *            space below.
             */
            this.tooltipPosition = TooltipPosition.BOTTOM;
            this._visibleSeriesCache = [];
            this._seriesDataCache = {};
            this._seriesMetadataCache = {};
            this._makeChartAsyncCallbackId = null;
        }
        ready() {
            super.ready();
            this.scopeSubtree(this.$.chartdiv, true);
        }
        attached() {
            // `capture` ensures that no handler can stop propagation and break the
            // handler. `passive` ensures that browser does not wait renderer thread
            // on JS handler (which can prevent default and impact rendering).
            const option = { capture: true, passive: true };
            this._listen(this, 'mousedown', this._onMouseDown.bind(this), option);
            this._listen(this, 'mouseup', this._onMouseUp.bind(this), option);
            this._listen(window, 'keydown', this._onKeyDown.bind(this), option);
            this._listen(window, 'keyup', this._onKeyUp.bind(this), option);
        }
        detached() {
            this.cancelAsync(this._makeChartAsyncCallbackId);
            if (this._chart) {
                this._chart.destroy();
                this._chart = undefined;
            }
            if (this._listeners) {
                this._listeners.forEach(({ node, eventName, func, option }) => {
                    node.removeEventListener(eventName, func, option);
                });
                this._listeners.clear();
            }
        }
        _listen(node, eventName, func, option = {}) {
            if (!this._listeners)
                this._listeners = new Set();
            this._listeners.add({ node, eventName, func, option });
            node.addEventListener(eventName, func, option);
        }
        _onKeyDown(event) {
            this.toggleClass('pankey', PanZoomDragLayer.isPanKey(event));
        }
        _onKeyUp(event) {
            this.toggleClass('pankey', PanZoomDragLayer.isPanKey(event));
        }
        _onMouseDown(event) {
            this.toggleClass('mousedown', true);
        }
        _onMouseUp(event) {
            this.toggleClass('mousedown', false);
        }
        /**
         * Returns whether the extent of rendered data values fits the current
         * chart viewport domain (includes smoothing and outlier detection).
         *
         * This is true when there is no data, and false when the domain has been
         * transformed from the extent via transformations (pan, zoom).
         */
        isDataFitToDomain() {
            return this._chart ? this._chart.isDataFitToDomain() : true;
        }
        /**
         * Sets the series that the chart displays. Series with other names will
         * not be displayed.
         *
         * @param {Array<String>} names Array with the names of the series to
         * display.
         */
        setVisibleSeries(names) {
            if (lodash.isEqual(this._visibleSeriesCache, names))
                return;
            this._visibleSeriesCache = names;
        }
        /**
         * Sets the data of one of the series. Note that to display this series
         * its name must be in the setVisibleSeries() array.
         *
         * @param {string} name Name of the series.
         * @param {Array<!vz_chart_helpers.ScalarDatum>} data Data of the series.
         * This is an array of objects with at least the following properties:
         * - step: (Number) - index of the datum.
         * - wall_time: (Date) - Date object with the datum's time.
         * - scalar: (Number) - Value of the datum.
         */
        setSeriesData(name, data) {
            this._seriesDataCache[name] = data;
            if (this._chart) {
                this._chart.setSeriesData(name, data);
            }
        }
        /**
         * Sets the metadata of one of the series.
         *
         * @param {string} name Name of the series.
         * @param {*} meta Metadata of the dataset used for later
         */
        setSeriesMetadata(name, meta) {
            this._seriesMetadataCache[name] = meta;
            if (this._chart) {
                this._chart.setSeriesMetadata(name, meta);
            }
        }
        commitChanges() {
            if (!this._chart)
                return;
            this._chart.commitChanges();
        }
        /**
         * Reset the chart domain. If the chart has not rendered yet, a call to this
         * method no-ops.
         */
        resetDomain() {
            if (this._chart) {
                this._chart.resetDomain();
            }
        }
        /**
         * Re-renders the chart. Useful if e.g. the container size changed.
         */
        redraw() {
            if (this._chart) {
                this._chart.redraw();
            }
        }
        /**
         * Creates a chart, and asynchronously renders it. Fires a chart-rendered
         * event after the chart is rendered.
         */
        _makeChart() {
            if (this._makeChartAsyncCallbackId !== null) {
                this.cancelAsync(this._makeChartAsyncCallbackId);
                this._makeChartAsyncCallbackId = null;
            }
            this._makeChartAsyncCallbackId = this.async(function () {
                this._makeChartAsyncCallbackId = null;
                // Find the actual xComponentsCreationMethod.
                let normalXComponentsCreationMethod = this.xComponentsCreationMethod;
                if (!this.xType && !normalXComponentsCreationMethod) {
                    normalXComponentsCreationMethod = stepX;
                }
                else if (this.xType) {
                    normalXComponentsCreationMethod = () => getXComponents(this.xType);
                }
                if (!normalXComponentsCreationMethod ||
                    !this.yValueAccessor ||
                    !this.tooltipColumns) {
                    return;
                }
                // We directly reference properties of `this` because this call is
                // asynchronous, and values may have changed in between the call being
                // initiated and actually being run.
                var chart = new LineChart(normalXComponentsCreationMethod, this.yValueAccessor, this.yScaleType, this.colorScale, this.$.tooltip, this.tooltipColumns, this.fillArea, this.defaultXRange, this.defaultYRange, this.symbolFunction, this.xAxisFormatter);
                var div = select(this.$.chartdiv);
                chart.renderTo(div);
                if (this._chart) {
                    this._chart.destroy();
                }
                this._chart = chart;
                this._chart.onAnchor(() => this.fire('chart-attached'));
            }, 350);
        }
        _reloadFromCache() {
            if (!this._chart)
                return;
            this._visibleSeriesCache.forEach((name) => {
                this._chart.setSeriesData(name, this._seriesDataCache[name] || []);
            });
            this._visibleSeriesCache
                .filter((name) => this._seriesMetadataCache[name])
                .forEach((name) => {
                this._chart.setSeriesMetadata(name, this._seriesMetadataCache[name]);
            });
            this._chart.setVisibleSeries(this._visibleSeriesCache);
            this._chart.commitChanges();
        }
        _smoothingChanged() {
            if (!this._chart)
                return;
            if (this.smoothingEnabled) {
                this._chart.smoothingUpdate(this.smoothingWeight);
            }
            else {
                this._chart.smoothingDisable();
            }
        }
        _outliersChanged() {
            if (!this._chart)
                return;
            this._chart.ignoreYOutliers(this.ignoreYOutliers);
        }
        _colorScaleChanged() {
            if (!this._chart)
                return;
            this._chart.setColorScale(this.colorScale);
            this._chart.redraw();
        }
        _tooltipColumnsChanged() {
            if (!this._chart)
                return;
            this._chart.setTooltipColumns(this.tooltipColumns);
        }
        _tooltipSortingMethodChanged() {
            if (!this._chart)
                return;
            this._chart.setTooltipSortingMethod(this.tooltipSortingMethod);
        }
        getExporter() {
            return new LineChartExporter(this.$.chartdiv);
        }
    };
    VzLineChart2.template = html `
    <div id="chartdiv"></div>
    <vz-chart-tooltip
      id="tooltip"
      position="[[tooltipPosition]]"
      content-component-name="vz-line-chart-tooltip"
    ></vz-chart-tooltip>
    <style include="plottable-style"></style>
    <style include="vz-pan-zoom-style"></style>
    <style>
      :host {
        -moz-user-select: none;
        -webkit-user-select: none;
        display: flex;
        flex-direction: column;
        flex-grow: 1;
        flex-shrink: 1;
        outline: none;
        position: relative;
        white-space: nowrap;
      }
      div {
        -webkit-user-select: none;
        -moz-user-select: none;
        flex-grow: 1;
        flex-shrink: 1;
      }

      #chartdiv .main {
        contain: strict;
        cursor: crosshair;
      }

      :host(.pankey) #chartdiv :not(.drag-zooming) .main {
        cursor: -webkit-grab;
        cursor: grab;
      }

      :host(.mousedown) #chartdiv .panning .main {
        cursor: -webkit-grabbing;
        cursor: grabbing;
      }

      #chartdiv {
        contain: strict;
      }

      #chartdiv line.guide-line {
        stroke: #999;
        stroke-width: 1.5px;
      }
      #chartdiv:hover .main {
        will-change: transform;
      }

      .ghost {
        opacity: 0.2;
        stroke-width: 1px;
      }
    </style>
  `;
    __decorate([
        property({ type: Object }),
        __metadata("design:type", src$2.Scales.Color)
    ], VzLineChart2.prototype, "colorScale", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Function)
    ], VzLineChart2.prototype, "symbolFunction", void 0);
    __decorate([
        property({
            type: Boolean,
            notify: true,
        }),
        __metadata("design:type", Boolean)
    ], VzLineChart2.prototype, "smoothingEnabled", void 0);
    __decorate([
        property({ type: Number }),
        __metadata("design:type", Number)
    ], VzLineChart2.prototype, "smoothingWeight", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], VzLineChart2.prototype, "xType", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Function)
    ], VzLineChart2.prototype, "xComponentsCreationMethod", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Function)
    ], VzLineChart2.prototype, "xAxisFormatter", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Function)
    ], VzLineChart2.prototype, "yValueAccessor", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], VzLineChart2.prototype, "tooltipColumns", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], VzLineChart2.prototype, "fillArea", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], VzLineChart2.prototype, "defaultXRange", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], VzLineChart2.prototype, "defaultYRange", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], VzLineChart2.prototype, "yScaleType", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], VzLineChart2.prototype, "ignoreYOutliers", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], VzLineChart2.prototype, "tooltipSortingMethod", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], VzLineChart2.prototype, "tooltipPosition", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], VzLineChart2.prototype, "_chart", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], VzLineChart2.prototype, "_visibleSeriesCache", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], VzLineChart2.prototype, "_seriesDataCache", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], VzLineChart2.prototype, "_seriesMetadataCache", void 0);
    __decorate([
        property({ type: Number }),
        __metadata("design:type", Number)
    ], VzLineChart2.prototype, "_makeChartAsyncCallbackId", void 0);
    __decorate([
        observe('xComponentsCreationMethod', 'xType', 'yValueAccessor', 'yScaleType', 'isAttached'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], VzLineChart2.prototype, "_makeChart", null);
    __decorate([
        observe('_chart', '_visibleSeriesCache'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], VzLineChart2.prototype, "_reloadFromCache", null);
    __decorate([
        observe('smoothingEnabled', 'smoothingWeight', '_chart'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], VzLineChart2.prototype, "_smoothingChanged", null);
    __decorate([
        observe('ignoreYOutliers', '_chart'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], VzLineChart2.prototype, "_outliersChanged", null);
    __decorate([
        observe('colorScale'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], VzLineChart2.prototype, "_colorScaleChanged", null);
    __decorate([
        observe('tooltipColumns'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], VzLineChart2.prototype, "_tooltipColumnsChanged", null);
    __decorate([
        observe('tooltipSortingMethod', '_chart'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], VzLineChart2.prototype, "_tooltipSortingMethodChanged", null);
    VzLineChart2 = __decorate([
        customElement('vz-line-chart2')
    ], VzLineChart2);
    let VzLineChartTooltip = class VzLineChartTooltip extends PolymerElement {
    };
    VzLineChartTooltip.template = html `
    <div class="content">
      <table>
        <thead></thead>
        <tbody></tbody>
      </table>
    </div>
    <style>
      :host {
        pointer-events: none;
      }

      .content {
        background: rgba(0, 0, 0, 0.8);
        border-radius: 4px;
        color: #fff;
        overflow: hidden;
        pointer-events: none;
      }

      table {
        font-size: 13px;
        line-height: 1.4em;
        margin-top: 10px;
        padding: 8px;
      }

      thead {
        font-size: 14px;
      }

      tbody {
        font-size: 13px;
        line-height: 21px;
        white-space: nowrap;
      }

      td {
        padding: 0 5px;
      }

      .swatch {
        border-radius: 50%;
        display: block;
        height: 18px;
        width: 18px;
      }

      .closest .swatch {
        box-shadow: inset 0 0 0 2px #fff;
      }

      th {
        padding: 0 5px;
        text-align: left;
      }

      .distant td:not(.swatch) {
        opacity: 0.8;
      }

      .ghost {
        opacity: 0.2;
        stroke-width: 1px;
      }
    </style>
  `;
    VzLineChartTooltip = __decorate([
        customElement('vz-line-chart-tooltip')
    ], VzLineChartTooltip);

    // The chart can sometimes get in a bad state, when it redraws while
    // it is display: none due to the user having switched to a different
    // page. This code implements a cascading queue to redraw the bad charts
    // one-by-one once they are active again.
    // We use a cascading queue becuase we don't want to block the UI / make the
    // ripples very slow while everything synchronously redraws.
    const redrawQueue = [];
    let redrawRaf = 0;
    const cascadingRedraw = lodash.throttle(function internalRedraw() {
        if (redrawQueue.length == 0)
            return;
        const x = redrawQueue.shift();
        if (x && x.active) {
            x.redraw();
            x._maybeRenderedInBadState = false;
        }
        window.cancelAnimationFrame(redrawRaf);
        window.requestAnimationFrame(internalRedraw);
    }, 100);
    // A component that fetches data from the TensorBoard server and renders it into
    // a vz-line-chart.
    let _TfLineChartDataLoader = class _TfLineChartDataLoader extends DataLoaderBehavior(LegacyElementMixin(PolymerElement)) {
        constructor() {
            super(...arguments);
            this._redrawRaf = null;
            this.active = false;
            this.logScaleActive = false;
            this.colorScale = { scale: runsColorScale };
            this._resetDomainOnNextLoad = true;
            this._maybeRenderedInBadState = false;
        }
        onLoadFinish() {
            this.commitChanges();
            if (this.dataToLoad.length > 0 && this._resetDomainOnNextLoad) {
                // (Don't unset _resetDomainOnNextLoad when we didn't
                // load any runs: this has the effect that if all our
                // runs are deselected, then we toggle them all on, we
                // properly size the domain once all the data is loaded
                // instead of just when we're first rendered.)
                this._resetDomainOnNextLoad = false;
                this.getChart().resetDomain();
            }
            this.redraw();
        }
        disconnectedCallback() {
            super.disconnectedCallback();
            if (this._redrawRaf !== null)
                cancelAnimationFrame(this._redrawRaf);
        }
        exportAsSvgString() {
            const exporter = this.getChart().getExporter();
            return exporter.exportAsString();
        }
        getChart() {
            return this.$.chart;
        }
        resetDomain() {
            this.getChart().resetDomain();
        }
        setSeriesData(name, data) {
            this.getChart().setSeriesData(name, data);
        }
        setSeriesMetadata(name, metadata) {
            this.getChart().setSeriesMetadata(name, metadata);
        }
        commitChanges() {
            this.getChart().commitChanges();
        }
        redraw() {
            if (this._redrawRaf !== null)
                cancelAnimationFrame(this._redrawRaf);
            this._redrawRaf = window.requestAnimationFrame(() => {
                if (this.active) {
                    this.getChart().redraw();
                }
                else {
                    // If we reached a point where we should render while the page
                    // is not active, we've gotten into a bad state.
                    this._maybeRenderedInBadState = true;
                }
            });
        }
        _loadKeyChanged() {
            this.reset();
            this._resetDomainOnNextLoad = true;
        }
        _dataSeriesChanged() {
            // Setting visible series redraws the chart.
            this.getChart().setVisibleSeries(this.dataSeries);
        }
        _logScaleChanged(logScaleActive) {
            const chart = this.getChart();
            chart.yScaleType = logScaleActive ? YScaleType.LOG : YScaleType.LINEAR;
            this.redraw();
        }
        _fixBadStateWhenActive() {
            // When the chart enters a potentially bad state (because it should
            // redraw, but the page is not currently active), we set the
            // _maybeRenderedInBadState flag. Whenever the chart becomes active,
            // we test this and schedule a redraw of the bad charts.
            if (this.active && this._maybeRenderedInBadState) {
                redrawQueue.push(this);
                cascadingRedraw();
            }
        }
        _onChartAttached() {
            if (!this.active) {
                this._maybeRenderedInBadState = true;
            }
        }
    };
    _TfLineChartDataLoader.template = html `
    <div id="chart-and-spinner-container">
      <vz-line-chart2
        id="chart"
        data-loading$="[[dataLoading]]"
        color-scale="[[colorScale]]"
        default-x-range="[[defaultXRange]]"
        default-y-range="[[defaultYRange]]"
        fill-area="[[fillArea]]"
        ignore-y-outliers="[[ignoreYOutliers]]"
        on-chart-attached="_onChartAttached"
        smoothing-enabled="[[smoothingEnabled]]"
        smoothing-weight="[[smoothingWeight]]"
        symbol-function="[[symbolFunction]]"
        tooltip-columns="[[tooltipColumns]]"
        tooltip-position="[[tooltipPosition]]"
        tooltip-sorting-method="[[tooltipSortingMethod]]"
        x-components-creation-method="[[xComponentsCreationMethod]]"
        x-type="[[xType]]"
        y-value-accessor="[[yValueAccessor]]"
      ></vz-line-chart2>
      <template is="dom-if" if="[[dataLoading]]">
        <div id="loading-spinner-container">
          <paper-spinner-lite active=""></paper-spinner-lite>
        </div>
      </template>
    </div>
    <style>
      :host {
        height: 100%;
        width: 100%;
        display: flex;
        flex-direction: column;
      }

      :host([_maybe-rendered-in-bad-state]) vz-line-chart {
        visibility: hidden;
      }

      #chart-and-spinner-container {
        display: flex;
        flex-grow: 1;
        position: relative;
      }

      #loading-spinner-container {
        align-items: center;
        bottom: 0;
        display: flex;
        display: flex;
        justify-content: center;
        left: 0;
        pointer-events: none;
        position: absolute;
        right: 0;
        top: 0;
      }

      vz-line-chart2 {
        -webkit-user-select: none;
        -moz-user-select: none;
      }

      vz-line-chart2[data-loading] {
        opacity: 0.3;
      }
    </style>
  `;
    __decorate([
        property({
            type: Boolean,
            observer: '_fixBadStateWhenActive',
        }),
        __metadata("design:type", Boolean)
    ], _TfLineChartDataLoader.prototype, "active", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], _TfLineChartDataLoader.prototype, "dataSeries", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", RequestManager)
    ], _TfLineChartDataLoader.prototype, "requestManager", void 0);
    __decorate([
        property({
            type: Boolean,
            observer: '_logScaleChanged',
        }),
        __metadata("design:type", Boolean)
    ], _TfLineChartDataLoader.prototype, "logScaleActive", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], _TfLineChartDataLoader.prototype, "xComponentsCreationMethod", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], _TfLineChartDataLoader.prototype, "xType", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Function)
    ], _TfLineChartDataLoader.prototype, "yValueAccessor", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], _TfLineChartDataLoader.prototype, "fillArea", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], _TfLineChartDataLoader.prototype, "smoothingEnabled", void 0);
    __decorate([
        property({ type: Number }),
        __metadata("design:type", Number)
    ], _TfLineChartDataLoader.prototype, "smoothingWeight", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], _TfLineChartDataLoader.prototype, "tooltipColumns", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", Object)
    ], _TfLineChartDataLoader.prototype, "tooltipSortingMethod", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], _TfLineChartDataLoader.prototype, "tooltipPosition", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], _TfLineChartDataLoader.prototype, "ignoreYOutliers", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], _TfLineChartDataLoader.prototype, "defaultXRange", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], _TfLineChartDataLoader.prototype, "defaultYRange", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Function)
    ], _TfLineChartDataLoader.prototype, "symbolFunction", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], _TfLineChartDataLoader.prototype, "colorScale", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], _TfLineChartDataLoader.prototype, "_resetDomainOnNextLoad", void 0);
    __decorate([
        property({
            type: Boolean,
            reflectToAttribute: true,
        }),
        __metadata("design:type", Boolean)
    ], _TfLineChartDataLoader.prototype, "_maybeRenderedInBadState", void 0);
    __decorate([
        observe('loadKey'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], _TfLineChartDataLoader.prototype, "_loadKeyChanged", null);
    __decorate([
        observe('dataSeries.*'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], _TfLineChartDataLoader.prototype, "_dataSeriesChanged", null);
    _TfLineChartDataLoader = __decorate([
        customElement('tf-line-chart-data-loader')
    ], _TfLineChartDataLoader);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    registerStyleDomModule({
        moduleName: 'tf-custom-scalar-card-style',
        styleContent: `
    :host {
      margin: 5px 10px;
      display: inline-block;
      width: 330px;
      vertical-align: text-top;
    }

    :host([_expanded]) {
      width: 100%;
    }

    :host([_expanded]) #tf-line-chart-data-loader-container {
      height: 400px;
    }

    h1 {
      font-size: 19px;
      font-weight: normal;
    }

    #tf-line-chart-data-loader-container {
      height: 200px;
      width: 100%;
    }

    #buttons {
      display: flex;
      flex-direction: row;
    }

    paper-icon-button {
      color: #2196f3;
      border-radius: 100%;
      width: 32px;
      height: 32px;
      padding: 4px;
    }

    paper-icon-button[selected] {
      background: var(--tb-ui-light-accent);
    }

    .download-links {
      display: flex;
      height: 32px;
    }

    .download-links a {
      font-size: 10px;
      align-self: center;
      margin: 2px;
    }

    .download-links paper-dropdown-menu {
      width: 100px;
      --paper-input-container-label: {
        font-size: 10px;
      }
      --paper-input-container-input: {
        font-size: 10px;
      }
    }
  `,
    });

    /* Copyright 2017 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * A class that represents a data series for a custom scalars chart.
     */
    class DataSeries {
        constructor(run, tag, name, scalarData, symbol) {
            this.run = run;
            this.tag = tag;
            this.name = name;
            this.scalarData = scalarData;
            this.symbol = symbol;
        }
        getName() {
            return this.name;
        }
        setData(scalarData) {
            this.scalarData = scalarData;
        }
        getData() {
            return this.scalarData;
        }
        getRun() {
            return this.run;
        }
        getTag() {
            return this.tag;
        }
        getSymbol() {
            return this.symbol;
        }
    }
    function generateDataSeriesName(run, tag) {
        return `${tag} (${run})`;
    }
    /**
     * A color scale that wraps the usual color scale that relies on runs. This
     * particular color scale parses the run from a series name and defers to that
     * former color scale.
     */
    class DataSeriesColorScale {
        constructor(runBasedColorScale) {
            this.runBasedColorScale = runBasedColorScale;
        }
        /**
         * Obtains the correct color based on the run.
         * @param {string} dataSeries
         * @return {string} The color.
         */
        scale(dataSeries) {
            return this.runBasedColorScale.scale(this.parseRunName(dataSeries));
        }
        /**
         * Parses the run name from a data series string. Returns the empty string if
         * parsing fails.
         */
        parseRunName(dataSeries) {
            const match = dataSeries.match(/\((.*)\)$/);
            if (!match) {
                // No match found.
                return '';
            }
            return match[1];
        }
    }

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    let _TfCustomScalarMarginChartCard = class _TfCustomScalarMarginChartCard extends LegacyElementMixin(PolymerElement) {
        constructor() {
            super(...arguments);
            this.active = true;
            this._colorScale = new DataSeriesColorScale({
                scale: runsColorScale,
            });
            this._nameToDataSeries = {};
            this._expanded = false;
            this._requestData = (items, onLoad, onFinish) => {
                const router = getRouter();
                const baseUrl = router.pluginRoute('custom_scalars', '/scalars');
                Promise.all(items.map((item) => {
                    const run = item;
                    const tag = this._tagFilter;
                    const url = addParams(baseUrl, { tag, run });
                    return this.requestManager
                        .request(url)
                        .then((data) => void onLoad({ item, data }));
                })).finally(() => void onFinish());
            };
            this._runToNextAvailableSymbolIndex = {};
            this._matchesListOpened = false;
            this._fillArea = {
                lowerAccessor: (d) => d.lower,
                higherAccessor: (d) => d.upper,
            };
            this._tooltipColumns = (() => {
                const valueFormatter = multiscaleFormatter(Y_TOOLTIP_FORMATTER_PRECISION);
                const formatValueOrNaN = (x) => (isNaN(x) ? 'NaN' : valueFormatter(x));
                return [
                    {
                        title: 'Name',
                        evaluate: (d) => d.dataset.metadata().name,
                    },
                    {
                        title: 'Value',
                        evaluate: (d) => formatValueOrNaN(d.datum.scalar),
                    },
                    {
                        title: 'Lower Margin',
                        evaluate: (d) => formatValueOrNaN(d.datum.lower),
                    },
                    {
                        title: 'Upper Margin',
                        evaluate: (d) => formatValueOrNaN(d.datum.upper),
                    },
                    {
                        title: 'Step',
                        evaluate: (d) => stepFormatter(d.datum.step),
                    },
                    {
                        title: 'Time',
                        evaluate: (d) => timeFormatter(d.datum.wall_time),
                    },
                    {
                        title: 'Relative',
                        evaluate: (d) => relativeFormatter(relativeAccessor(d.datum, -1, d.dataset)),
                    },
                ];
            })();
            this._missingTags = [];
            this._missingTagsCollapsibleOpened = false;
        }
        reload() {
            this.$.loader.reload();
        }
        redraw() {
            this.$.loader.redraw();
        }
        _toggleExpanded(e) {
            this.set('_expanded', !this._expanded);
            this.redraw();
        }
        _toggleLogScale() {
            this.set('_logScaleActive', !this._logScaleActive);
        }
        _resetDomain() {
            const chart = this.$.loader;
            if (chart) {
                chart.resetDomain();
            }
        }
        _csvUrl(_nameToDataSeries, dataSeriesName) {
            if (!dataSeriesName)
                return '';
            const baseUrl = this._downloadDataUrl(_nameToDataSeries, dataSeriesName);
            return addParams(baseUrl, { format: 'csv' });
        }
        _jsonUrl(_nameToDataSeries, dataSeriesName) {
            if (!dataSeriesName)
                return '';
            const baseUrl = this._downloadDataUrl(_nameToDataSeries, dataSeriesName);
            return addParams(baseUrl, { format: 'json' });
        }
        _downloadDataUrl(_nameToDataSeries, dataSeriesName) {
            const dataSeries = _nameToDataSeries[dataSeriesName];
            const getVars = {
                tag: dataSeries.getTag(),
                run: dataSeries.getRun(),
            };
            return addParams(getRouter().pluginRoute('custom_scalars', '/download_data'), getVars);
        }
        _createProcessDataFunction(marginChartSeries) {
            // This function is called when data is received from the backend.
            return (scalarChart, run, data) => {
                if (!data.regex_valid) {
                    // The regular expression is constructed from frontend logic that
                    // pieces together different tags. Hence, this case should never be
                    // reached if the dashboard behaves correctly.
                    this.set('_tagFilterInvalid', true);
                    return;
                }
                // The user's regular expression was valid.
                // Incorporate these newly loaded values.
                const newMapping = lodash.clone(this._nameToDataSeries);
                const tagsNotFound = [];
                lodash.forEach(marginChartSeries, (tagsObject) => {
                    let tagNotFound = false;
                    const scalarEvents = data.tag_to_events[tagsObject.value];
                    const lowerBounds = data.tag_to_events[tagsObject.lower];
                    const upperBounds = data.tag_to_events[tagsObject.upper];
                    // Make sure that data is found for each of the tags.
                    if (lodash.isUndefined(scalarEvents)) {
                        tagsNotFound.push(tagsObject.value);
                        tagNotFound = true;
                    }
                    if (lodash.isUndefined(lowerBounds)) {
                        tagsNotFound.push(tagsObject.lower);
                        tagNotFound = true;
                    }
                    if (lodash.isUndefined(upperBounds)) {
                        tagsNotFound.push(tagsObject.upper);
                        tagNotFound = true;
                    }
                    // At least one of the tags lacks data. We terminate early because
                    // the line chart requires all 3 pieces of data (value, lower bound,
                    // and upper bound).
                    if (tagNotFound) {
                        return;
                    }
                    // Make sure that steps for all the lists correspond with each
                    // other. Otherwise, display an error message.
                    const obtainStep = (datum) => datum[1];
                    const stepsMismatch = this._findStepMismatch(tagsObject, scalarEvents.map(obtainStep), lowerBounds.map(obtainStep), upperBounds.map(obtainStep));
                    if (stepsMismatch) {
                        this.set('_stepsMismatch', stepsMismatch);
                        return;
                    }
                    // Create data points that the line chart can parse.
                    const obtainNumber = (datum) => datum[2];
                    const dataPoints = scalarEvents.map((datum, i) => ({
                        wall_time: new Date(datum[0] * 1000),
                        step: obtainStep(datum),
                        scalar: obtainNumber(datum),
                        lower: obtainNumber(lowerBounds[i]),
                        upper: obtainNumber(upperBounds[i]),
                    }));
                    // Compute the series name, which is based on both the run and the
                    // tag of the value.
                    const seriesName = generateDataSeriesName(run, tagsObject.value);
                    const series = newMapping[seriesName];
                    if (series) {
                        // This series already exists.
                        series.setData(dataPoints);
                    }
                    else {
                        const series = this._createNewDataSeries(run, tagsObject.value, seriesName, dataPoints);
                        newMapping[seriesName] = series;
                    }
                });
                this.set('_nameToDataSeries', newMapping);
                const entryIndex = lodash.findIndex(this._missingTags, (entry) => {
                    return entry.run === run;
                });
                if (tagsNotFound.length && tagsNotFound.length != 3) {
                    // Some but not all tags were found. Show a warning message.
                    const entry = {
                        run: run,
                        tags: tagsNotFound,
                    };
                    if (entryIndex >= 0) {
                        // Remove the previous entry. Insert the new one.
                        this.splice('_missingTags', entryIndex, 1, entry);
                    }
                    else {
                        // Insert a new entry.
                        this.push('_missingTags', entry);
                    }
                }
                else if (entryIndex >= 0) {
                    // Remove the previous entry if it exists.
                    this.splice('_missingTags', entryIndex, 1);
                }
            };
        }
        _findStepMismatch(tagsObject, valueSteps, lowerSteps, upperSteps) {
            if (lodash.isEqual(lowerSteps, valueSteps) &&
                lodash.isEqual(upperSteps, valueSteps)) {
                // There is no mismatch.
                return null;
            }
            return {
                seriesObject: tagsObject,
                valueSteps: valueSteps,
                lowerSteps: lowerSteps,
                upperSteps: upperSteps,
            };
        }
        _createNewDataSeries(run, tag, seriesName, dataPoints) {
            // If the run has not been seen before, define the next
            // available marker index.
            this._runToNextAvailableSymbolIndex[run] |= 0;
            // Every data series within a run has a unique symbol.
            const lineChartSymbol = SYMBOLS_LIST[this._runToNextAvailableSymbolIndex[run]];
            // Create a series with this name.
            const series = new DataSeries(run, tag, seriesName, dataPoints, lineChartSymbol);
            // Loop back to the beginning if we are out of symbols.
            const numSymbols = SYMBOLS_LIST.length;
            this._runToNextAvailableSymbolIndex[run] =
                (this._runToNextAvailableSymbolIndex[run] + 1) % numSymbols;
            return series;
        }
        _updateChart() {
            var _nameToDataSeries = this._nameToDataSeries;
            // Add new data series.
            lodash.forOwn(_nameToDataSeries, (dataSeries) => {
                this.$.loader.setSeriesData(dataSeries.getName(), dataSeries.getData());
            });
            this.$.loader.commitChanges();
        }
        get _seriesNames() {
            const runLookup = new Set(this.runs);
            return Object.entries(this._nameToDataSeries)
                .filter(([_, series]) => runLookup.has(series.run))
                .map(([name]) => name);
        }
        _determineColor(colorScale, seriesName) {
            return colorScale.scale(seriesName);
        }
        _refreshDataSeries() {
            var _tagFilter = this._tagFilter;
            this.set('_nameToDataSeries', {});
        }
        _createSymbolFunction() {
            return (seriesName) => this._nameToDataSeries[seriesName].getSymbol().method();
        }
        _determineSymbol(_nameToDataSeries, seriesName) {
            return _nameToDataSeries[seriesName].getSymbol().character;
        }
        get _tagFilter() {
            var marginChartSeries = this.marginChartSeries;
            const tags = lodash.flatten(marginChartSeries.map((series) => [
                series.value,
                series.lower,
                series.upper,
            ]));
            const escapedTags = tags.map((r) => '(' + this._escapeRegexCharacters(r) + ')');
            // Combine the different regexes into a single regex.
            return escapedTags.join('|');
        }
        _escapeRegexCharacters(stringValue) {
            return stringValue.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
        }
        _getToggleCollapsibleIcon(listOpened) {
            return listOpened ? 'expand-less' : 'expand-more';
        }
        _toggleMatchesOpen() {
            this.set('_matchesListOpened', !this._matchesListOpened);
        }
        get _titleDisplayString() {
            var title = this.title;
            // If no title is provided, use the tag filter, which is a combination
            // of the tags for the value, lower, and upper fields of each series.
            return title || 'untitled';
        }
        _separateWithCommas(numbers) {
            return numbers.join(', ');
        }
        _toggleMissingTagsCollapsibleOpen() {
            this.set('_missingTagsCollapsibleOpened', !this._missingTagsCollapsibleOpened);
        }
        _matchListEntryColorUpdated() {
            const domRepeat = this.$$('#match-list-repeat');
            if (!domRepeat) {
                return;
            }
            this.root
                .querySelectorAll('.match-list-entry')
                .forEach((entryElement) => {
                const seriesName = domRepeat.itemForElement(entryElement);
                entryElement.style.color = this._determineColor(this._colorScale, seriesName);
            });
        }
    };
    _TfCustomScalarMarginChartCard.template = html `
    <tf-card-heading display-name="[[_titleDisplayString]]"></tf-card-heading>
    <div id="tf-line-chart-data-loader-container">
      <tf-line-chart-data-loader
        id="loader"
        active="[[active]]"
        color-scale="[[_colorScale]]"
        data-series="[[_seriesNames]]"
        fill-area="[[_fillArea]]"
        ignore-y-outliers="[[ignoreYOutliers]]"
        load-key="[[_tagFilter]]"
        data-to-load="[[runs]]"
        request-data="[[_requestData]]"
        log-scale-active="[[_logScaleActive]]"
        load-data-callback="[[_createProcessDataFunction(marginChartSeries)]]"
        request-manager="[[requestManager]]"
        symbol-function="[[_createSymbolFunction()]]"
        tooltip-columns="[[_tooltipColumns]]"
        tooltip-sorting-method="[[tooltipSortingMethod]]"
        x-type="[[xType]]"
      >
      </tf-line-chart-data-loader>
    </div>
    <div id="buttons">
      <paper-icon-button
        selected$="[[_expanded]]"
        icon="fullscreen"
        on-tap="_toggleExpanded"
      ></paper-icon-button>
      <paper-icon-button
        selected$="[[_logScaleActive]]"
        icon="line-weight"
        on-tap="_toggleLogScale"
        title="Toggle y-axis log scale"
      ></paper-icon-button>
      <paper-icon-button
        icon="settings-overscan"
        on-tap="_resetDomain"
        title="Fit domain to data"
      ></paper-icon-button>
      <span style="flex-grow: 1"></span>
      <template is="dom-if" if="[[showDownloadLinks]]">
        <div class="download-links">
          <paper-dropdown-menu
            no-label-float="true"
            label="series to download"
            selected-item-label="{{_dataSeriesNameToDownload}}"
          >
            <paper-listbox class="dropdown-content" slot="dropdown-content">
              <template
                is="dom-repeat"
                items="[[_seriesNames]]"
                as="dataSeriesName"
              >
                <paper-item no-label-float="true"
                  >[[dataSeriesName]]</paper-item
                >
              </template>
            </paper-listbox>
          </paper-dropdown-menu>
          <a
            download="[[_dataSeriesNameToDownload]].csv"
            href="[[_csvUrl(_nameToDataSeries, _dataSeriesNameToDownload)]]"
            >CSV</a
          >
          <a
            download="[[_dataSeriesNameToDownload]].json"
            href="[[_jsonUrl(_nameToDataSeries, _dataSeriesNameToDownload)]]"
            >JSON</a
          >
        </div>
      </template>
    </div>

    <!-- here -->
    <template is="dom-if" if="[[_missingTags.length]]">
      <div class="collapsible-list-title">
        <paper-icon-button
          icon="[[_getToggleCollapsibleIcon(_missingTagsCollapsibleOpened)]]"
          on-click="_toggleMissingTagsCollapsibleOpen"
          class="toggle-collapsible-button"
        >
        </paper-icon-button>
        <span class="collapsible-title-text">
          <iron-icon icon="icons:error"></iron-icon> Missing Tags
        </span>
      </div>
      <iron-collapse opened="[[_missingTagsCollapsibleOpened]]">
        <div class="error-content">
          <iron-icon class="error-icon" icon="icons:error"></iron-icon>
          <template is="dom-repeat" items="[[_missingTags]]" as="missingEntry">
            <div class="missing-tags-for-run-container">
              Run "[[missingEntry.run]]" lacks data for tags
              <ul>
                <template
                  is="dom-repeat"
                  items="[[missingEntry.tags]]"
                  as="tag"
                >
                  <li>[[tag]]</li>
                </template>
              </ul>
            </div>
          </template>
        </div>
      </iron-collapse>
    </template>

    <template is="dom-if" if="[[_tagFilterInvalid]]">
      <div class="error-content">
        <iron-icon class="error-icon" icon="icons:error"></iron-icon>
        This regular expresion is invalid:<br />
        <span class="invalid-regex">[[_tagFilter]]</span>
      </div>
    </template>

    <template is="dom-if" if="[[_stepsMismatch]]">
      <div class="error-content">
        <iron-icon class="error-icon" icon="icons:error"></iron-icon>
        The steps for value, lower, and upper tags do not match:
        <ul>
          <li>
            <span class="tag-name">[[_stepsMismatch.seriesObject.value]]</span>:
            [[_separateWithCommas(_stepsMismatch.valueSteps)]]
          </li>
          <li>
            <span class="tag-name">[[_stepsMismatch.seriesObject.lower]]</span>:
            [[_separateWithCommas(_stepsMismatch.lowerSteps)]]
          </li>
          <li>
            <span class="tag-name">[[_stepsMismatch.seriesObject.upper]]</span>:
            [[_separateWithCommas(_stepsMismatch.upperSteps)]]
          </li>
        </ul>
      </div>
    </template>

    <div id="matches-container">
      <div class="collapsible-list-title">
        <template is="dom-if" if="[[_seriesNames.length]]">
          <paper-icon-button
            icon="[[_getToggleCollapsibleIcon(_matchesListOpened)]]"
            on-click="_toggleMatchesOpen"
            class="toggle-matches-button"
          >
          </paper-icon-button>
        </template>

        <span class="collapsible-title-text">
          Matches ([[_seriesNames.length]])
        </span>
      </div>
      <template is="dom-if" if="[[_seriesNames.length]]">
        <iron-collapse opened="[[_matchesListOpened]]">
          <div id="matches-list">
            <template
              is="dom-repeat"
              items="[[_seriesNames]]"
              as="seriesName"
              id="match-list-repeat"
              on-dom-change="_matchListEntryColorUpdated"
            >
              <div class="match-list-entry">
                <span class="match-entry-symbol">
                  [[_determineSymbol(_nameToDataSeries, seriesName)]]
                </span>
                [[seriesName]]
              </div>
            </template>
          </div>
        </iron-collapse>
      </template>
    </div>

    <style include="tf-custom-scalar-card-style"></style>
    <style>
      .error-content {
        background: #f00;
        border-radius: 5px;
        color: #fff;
        margin: 10px 0 0 0;
        padding: 10px;
      }

      .error-icon {
        display: block;
        fill: #fff;
        margin: 0 auto 5px auto;
      }

      .invalid-regex {
        font-weight: bold;
      }

      .error-content ul {
        margin: 1px 0 0 0;
        padding: 0 0 0 19px;
      }

      .tag-name {
        font-weight: bold;
      }

      .collapsible-list-title {
        margin: 10px 0 5px 0;
      }

      .collapsible-title-text {
        vertical-align: middle;
      }

      #matches-list {
        max-height: 200px;
        overflow-y: auto;
      }

      .match-list-entry {
        margin: 0 0 5px 0;
      }

      .match-entry-symbol {
        font-family: arial, sans-serif;
        display: inline-block;
        width: 10px;
      }

      .missing-tags-for-run-container {
        margin: 8px 0 0 0;
      }
    </style>
  `;
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], _TfCustomScalarMarginChartCard.prototype, "runs", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], _TfCustomScalarMarginChartCard.prototype, "xType", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], _TfCustomScalarMarginChartCard.prototype, "active", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], _TfCustomScalarMarginChartCard.prototype, "title", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], _TfCustomScalarMarginChartCard.prototype, "marginChartSeries", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], _TfCustomScalarMarginChartCard.prototype, "ignoreYOutliers", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", RequestManager)
    ], _TfCustomScalarMarginChartCard.prototype, "requestManager", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], _TfCustomScalarMarginChartCard.prototype, "showDownloadLinks", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], _TfCustomScalarMarginChartCard.prototype, "tagMetadata", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], _TfCustomScalarMarginChartCard.prototype, "tooltipSortingMethod", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], _TfCustomScalarMarginChartCard.prototype, "_colorScale", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], _TfCustomScalarMarginChartCard.prototype, "_tagFilterInvalid", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], _TfCustomScalarMarginChartCard.prototype, "_nameToDataSeries", void 0);
    __decorate([
        property({
            type: Boolean,
            reflectToAttribute: true,
        }),
        __metadata("design:type", Boolean)
    ], _TfCustomScalarMarginChartCard.prototype, "_expanded", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], _TfCustomScalarMarginChartCard.prototype, "_logScaleActive", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Function)
    ], _TfCustomScalarMarginChartCard.prototype, "_requestData", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], _TfCustomScalarMarginChartCard.prototype, "_runToNextAvailableSymbolIndex", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], _TfCustomScalarMarginChartCard.prototype, "_matchesListOpened", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], _TfCustomScalarMarginChartCard.prototype, "_fillArea", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], _TfCustomScalarMarginChartCard.prototype, "_tooltipColumns", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], _TfCustomScalarMarginChartCard.prototype, "_missingTags", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], _TfCustomScalarMarginChartCard.prototype, "_missingTagsCollapsibleOpened", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], _TfCustomScalarMarginChartCard.prototype, "_stepsMismatch", void 0);
    __decorate([
        observe('_nameToDataSeries'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], _TfCustomScalarMarginChartCard.prototype, "_updateChart", null);
    __decorate([
        computed('_nameToDataSeries', 'runs'),
        __metadata("design:type", Object),
        __metadata("design:paramtypes", [])
    ], _TfCustomScalarMarginChartCard.prototype, "_seriesNames", null);
    __decorate([
        observe('_tagFilter'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], _TfCustomScalarMarginChartCard.prototype, "_refreshDataSeries", null);
    __decorate([
        computed('marginChartSeries'),
        __metadata("design:type", String),
        __metadata("design:paramtypes", [])
    ], _TfCustomScalarMarginChartCard.prototype, "_tagFilter", null);
    __decorate([
        computed('title'),
        __metadata("design:type", String),
        __metadata("design:paramtypes", [])
    ], _TfCustomScalarMarginChartCard.prototype, "_titleDisplayString", null);
    _TfCustomScalarMarginChartCard = __decorate([
        customElement('tf-custom-scalar-margin-chart-card')
    ], _TfCustomScalarMarginChartCard);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/

    var tf_backend = /*#__PURE__*/Object.freeze({
        __proto__: null,
        TYPES: TYPES$1,
        getRunsNamed: getRunsNamed,
        getTags: getTags,
        filterTags: filterTags,
        ListenKey: ListenKey$1,
        BaseStore: BaseStore,
        Canceller: Canceller,
        EnvironmentStore: EnvironmentStore,
        environmentStore: environmentStore,
        ExperimentsStore: ExperimentsStore,
        experimentsStore: experimentsStore,
        RequestCancellationError: RequestCancellationError,
        InvalidRequestOptionsError: InvalidRequestOptionsError,
        RequestNetworkError: RequestNetworkError,
        get HttpMethodType () { return HttpMethodType; },
        RequestOptions: RequestOptions,
        RequestManager: RequestManager,
        createRouter: createRouter,
        getRouter: getRouter,
        setRouter: setRouter,
        createSearchParam: createSearchParam,
        RunsStore: RunsStore,
        runsStore: runsStore,
        addParams: addParams
    });

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    let _TfCustomScalarMultiLineChartCard = class _TfCustomScalarMultiLineChartCard extends LegacyElementMixin(PolymerElement) {
        constructor() {
            super(...arguments);
            this.active = true;
            this._colorScale = new DataSeriesColorScale({
                scale: runsColorScale,
            });
            this._nameToDataSeries = {};
            this._expanded = false;
            this._requestData = (items, onLoad, onFinish) => {
                const router = getRouter();
                const baseUrl = router.pluginRoute('custom_scalars', '/scalars');
                Promise.all(items.map((item) => {
                    const run = item;
                    const tag = this._tagFilter;
                    const url = addParams(baseUrl, { tag, run });
                    return this.requestManager
                        .request(url)
                        .then((data) => void onLoad({ item, data }));
                })).finally(() => void onFinish());
            };
            this._runToNextAvailableSymbolIndex = {};
            this._matchesListOpened = false;
        }
        reload() {
            this.$.loader.reload();
        }
        redraw() {
            this.$.loader.redraw();
        }
        _toggleExpanded(e) {
            this.set('_expanded', !this._expanded);
            this.redraw();
        }
        _toggleLogScale() {
            this.set('_logScaleActive', !this._logScaleActive);
        }
        _resetDomain() {
            const chart = this.$.loader;
            if (chart) {
                chart.resetDomain();
            }
        }
        _csvUrl(nameToSeries, dataSeriesName) {
            if (!dataSeriesName)
                return '';
            const baseUrl = this._downloadDataUrl(nameToSeries, dataSeriesName);
            return addParams(baseUrl, { format: 'csv' });
        }
        _jsonUrl(nameToSeries, dataSeriesName) {
            if (!dataSeriesName)
                return '';
            const baseUrl = this._downloadDataUrl(nameToSeries, dataSeriesName);
            return addParams(baseUrl, { format: 'json' });
        }
        _downloadDataUrl(nameToSeries, dataSeriesName) {
            const dataSeries = nameToSeries[dataSeriesName];
            const getVars = {
                tag: dataSeries.getTag(),
                run: dataSeries.getRun(),
            };
            return addParams(getRouter().pluginRoute('custom_scalars', '/download_data'), getVars);
        }
        _createProcessDataFunction() {
            // This function is called when data is received from the backend.
            return (scalarChart, run, data) => {
                if (data.regex_valid) {
                    // The user's regular expression was valid.
                    // Incorporate these newly loaded values.
                    const newMapping = lodash.clone(this._nameToDataSeries);
                    lodash.forOwn(data.tag_to_events, (scalarEvents, tag) => {
                        const data = scalarEvents.map((datum) => ({
                            wall_time: new Date(datum[0] * 1000),
                            step: datum[1],
                            scalar: datum[2],
                        }));
                        const seriesName = generateDataSeriesName(run, tag);
                        const datum = newMapping[seriesName];
                        if (datum) {
                            // This series already exists.
                            datum.setData(data);
                        }
                        else {
                            if (lodash.isUndefined(this._runToNextAvailableSymbolIndex[run])) {
                                // The run has not been seen before. Define the next available
                                // marker index.
                                this._runToNextAvailableSymbolIndex[run] = 0;
                            }
                            // Every data series within a run has a unique symbol.
                            const lineChartSymbol = SYMBOLS_LIST[this._runToNextAvailableSymbolIndex[run]];
                            // Create a series with this name.
                            const series = new DataSeries(run, tag, seriesName, data, lineChartSymbol);
                            newMapping[seriesName] = series;
                            // Loop back to the beginning if we are out of symbols.
                            const numSymbols = SYMBOLS_LIST.length;
                            this._runToNextAvailableSymbolIndex[run] =
                                (this._runToNextAvailableSymbolIndex[run] + 1) % numSymbols;
                        }
                    });
                    this.set('_nameToDataSeries', newMapping);
                }
            };
        }
        _updateChart() {
            var _nameToDataSeries = this._nameToDataSeries;
            // Add new data series.
            Object.entries(_nameToDataSeries).forEach(([name, series]) => {
                this.$.loader.setSeriesData(name, series.getData());
            });
            this.$.loader.commitChanges();
        }
        _computeSelectedRunsSet(runs) {
            const mapping = {};
            lodash.forEach(runs, (run) => {
                mapping[run] = 1;
            });
            return mapping;
        }
        get _seriesNames() {
            const runLookup = new Set(this.runs);
            return Object.entries(this._nameToDataSeries)
                .filter(([_, series]) => runLookup.has(series.run))
                .map(([name]) => name);
        }
        _determineColor(colorScale, seriesName) {
            return colorScale.scale(seriesName);
        }
        _refreshDataSeries() {
            var _tagFilter = this._tagFilter;
            this.set('_nameToDataSeries', {});
        }
        _createSymbolFunction() {
            return (seriesName) => this._nameToDataSeries[seriesName].getSymbol().method();
        }
        _determineSymbol(nameToSeries, seriesName) {
            return nameToSeries[seriesName].getSymbol().character;
        }
        get _tagFilter() {
            var tagRegexes = this.tagRegexes;
            if (tagRegexes.length === 1) {
                return tagRegexes[0];
            }
            // Combine the different regexes into a single regex.
            return tagRegexes.map((r) => '(' + r + ')').join('|');
        }
        _getToggleMatchesIcon(matchesListOpened) {
            return matchesListOpened ? 'expand-less' : 'expand-more';
        }
        _toggleMatchesOpen() {
            this.set('_matchesListOpened', !this._matchesListOpened);
        }
        get _titleDisplayString() {
            var title = this.title;
            // If no title is provided, use a placeholder string.
            return title || 'untitled';
        }
        _matchListEntryColorUpdated(event) {
            const domRepeat = this.$$('#match-list-repeat');
            if (!domRepeat) {
                return;
            }
            this.root
                .querySelectorAll('.match-list-entry')
                .forEach((entryElement) => {
                const seriesName = domRepeat.itemForElement(entryElement);
                entryElement.style.color = this._determineColor(this._colorScale, seriesName);
            });
        }
    };
    _TfCustomScalarMultiLineChartCard.template = html `
    <tf-card-heading display-name="[[_titleDisplayString]]"></tf-card-heading>
    <div id="tf-line-chart-data-loader-container">
      <tf-line-chart-data-loader
        id="loader"
        active="[[active]]"
        color-scale="[[_colorScale]]"
        data-series="[[_seriesNames]]"
        ignore-y-outliers="[[ignoreYOutliers]]"
        load-key="[[_tagFilter]]"
        data-to-load="[[runs]]"
        request-data="[[_requestData]]"
        log-scale-active="[[_logScaleActive]]"
        load-data-callback="[[_createProcessDataFunction()]]"
        request-manager="[[requestManager]]"
        smoothing-enabled="[[smoothingEnabled]]"
        smoothing-weight="[[smoothingWeight]]"
        symbol-function="[[_createSymbolFunction()]]"
        tooltip-sorting-method="[[tooltipSortingMethod]]"
        x-type="[[xType]]"
      >
      </tf-line-chart-data-loader>
    </div>
    <div id="buttons">
      <paper-icon-button
        selected$="[[_expanded]]"
        icon="fullscreen"
        on-tap="_toggleExpanded"
      ></paper-icon-button>
      <paper-icon-button
        selected$="[[_logScaleActive]]"
        icon="line-weight"
        on-tap="_toggleLogScale"
        title="Toggle y-axis log scale"
      ></paper-icon-button>
      <paper-icon-button
        icon="settings-overscan"
        on-tap="_resetDomain"
        title="Fit domain to data"
      ></paper-icon-button>
      <span style="flex-grow: 1"></span>
      <template is="dom-if" if="[[showDownloadLinks]]">
        <div class="download-links">
          <paper-dropdown-menu
            no-label-float="true"
            label="series to download"
            selected-item-label="{{_dataSeriesNameToDownload}}"
          >
            <paper-listbox class="dropdown-content" slot="dropdown-content">
              <template
                is="dom-repeat"
                items="[[_seriesNames]]"
                as="dataSeriesName"
              >
                <paper-item no-label-float="true"
                  >[[dataSeriesName]]</paper-item
                >
              </template>
            </paper-listbox>
          </paper-dropdown-menu>
          <a
            download="[[_dataSeriesNameToDownload]].csv"
            href="[[_csvUrl(_nameToDataSeries, _dataSeriesNameToDownload)]]"
            >CSV</a
          >
          <a
            download="[[_dataSeriesNameToDownload]].json"
            href="[[_jsonUrl(_nameToDataSeries, _dataSeriesNameToDownload)]]"
            >JSON</a
          >
        </div>
      </template>
    </div>
    <div id="matches-container">
      <div id="matches-list-title">
        <template is="dom-if" if="[[_seriesNames.length]]">
          <paper-icon-button
            icon="[[_getToggleMatchesIcon(_matchesListOpened)]]"
            on-click="_toggleMatchesOpen"
            class="toggle-matches-button"
          >
          </paper-icon-button>
        </template>

        <span class="matches-text"> Matches ([[_seriesNames.length]]) </span>
      </div>
      <template is="dom-if" if="[[_seriesNames.length]]">
        <iron-collapse opened="[[_matchesListOpened]]">
          <div id="matches-list">
            <template
              is="dom-repeat"
              items="[[_seriesNames]]"
              as="seriesName"
              id="match-list-repeat"
              on-dom-change="_matchListEntryColorUpdated"
            >
              <div class="match-list-entry">
                <span class="match-entry-symbol">
                  [[_determineSymbol(_nameToDataSeries, seriesName)]]
                </span>
                [[seriesName]]
              </div>
            </template>
          </div>
        </iron-collapse>
      </template>
    </div>

    <style include="tf-custom-scalar-card-style"></style>
    <style>
      #matches-list-title {
        margin: 10px 0 5px 0;
      }

      #matches-list {
        max-height: 200px;
        overflow-y: auto;
      }

      .match-list-entry {
        margin: 0 0 5px 0;
      }

      .match-entry-symbol {
        font-family: arial, sans-serif;
        display: inline-block;
        width: 10px;
      }

      .matches-text {
        vertical-align: middle;
      }
    </style>
  `;
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], _TfCustomScalarMultiLineChartCard.prototype, "runs", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], _TfCustomScalarMultiLineChartCard.prototype, "xType", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], _TfCustomScalarMultiLineChartCard.prototype, "active", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], _TfCustomScalarMultiLineChartCard.prototype, "title", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], _TfCustomScalarMultiLineChartCard.prototype, "tagRegexes", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], _TfCustomScalarMultiLineChartCard.prototype, "ignoreYOutliers", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", RequestManager)
    ], _TfCustomScalarMultiLineChartCard.prototype, "requestManager", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], _TfCustomScalarMultiLineChartCard.prototype, "showDownloadLinks", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], _TfCustomScalarMultiLineChartCard.prototype, "smoothingEnabled", void 0);
    __decorate([
        property({ type: Number }),
        __metadata("design:type", Number)
    ], _TfCustomScalarMultiLineChartCard.prototype, "smoothingWeight", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], _TfCustomScalarMultiLineChartCard.prototype, "tagMetadata", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], _TfCustomScalarMultiLineChartCard.prototype, "tooltipSortingMethod", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", DataSeriesColorScale)
    ], _TfCustomScalarMultiLineChartCard.prototype, "_colorScale", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], _TfCustomScalarMultiLineChartCard.prototype, "_nameToDataSeries", void 0);
    __decorate([
        property({
            type: Boolean,
            reflectToAttribute: true,
        }),
        __metadata("design:type", Boolean)
    ], _TfCustomScalarMultiLineChartCard.prototype, "_expanded", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], _TfCustomScalarMultiLineChartCard.prototype, "_logScaleActive", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Function)
    ], _TfCustomScalarMultiLineChartCard.prototype, "_requestData", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], _TfCustomScalarMultiLineChartCard.prototype, "_runToNextAvailableSymbolIndex", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], _TfCustomScalarMultiLineChartCard.prototype, "_matchesListOpened", void 0);
    __decorate([
        observe('_nameToDataSeries'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], _TfCustomScalarMultiLineChartCard.prototype, "_updateChart", null);
    __decorate([
        computed('_nameToDataSeries', 'runs'),
        __metadata("design:type", Object),
        __metadata("design:paramtypes", [])
    ], _TfCustomScalarMultiLineChartCard.prototype, "_seriesNames", null);
    __decorate([
        observe('_tagFilter'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], _TfCustomScalarMultiLineChartCard.prototype, "_refreshDataSeries", null);
    __decorate([
        computed('tagRegexes'),
        __metadata("design:type", String),
        __metadata("design:paramtypes", [])
    ], _TfCustomScalarMultiLineChartCard.prototype, "_tagFilter", null);
    __decorate([
        computed('title'),
        __metadata("design:type", String),
        __metadata("design:paramtypes", [])
    ], _TfCustomScalarMultiLineChartCard.prototype, "_titleDisplayString", null);
    _TfCustomScalarMultiLineChartCard = __decorate([
        customElement('tf-custom-scalar-multi-line-chart-card')
    ], _TfCustomScalarMultiLineChartCard);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    let TfCustomScalarDashboard = class TfCustomScalarDashboard extends PolymerElement {
        constructor() {
            super(...arguments);
            this._requestManager = new RequestManager(50);
            this._canceller = new Canceller();
            this._showDownloadLinks = getBooleanInitializer('_showDownloadLinks', {
                defaultValue: false,
                useLocalStorage: true,
            }).call(this);
            this._smoothingWeight = getNumberInitializer('_smoothingWeight', {
                defaultValue: 0.6,
            }).call(this);
            this._ignoreYOutliers = getBooleanInitializer('_ignoreYOutliers', {
                defaultValue: true,
                useLocalStorage: true,
            }).call(this);
            this._xType = 'step';
            this._active = true;
            this.reloadOnReady = true;
            this._showDownloadLinksObserver = getBooleanObserver('_showDownloadLinks', {
                defaultValue: false,
                useLocalStorage: true,
            });
            this._smoothingWeightObserver = getNumberObserver('_smoothingWeight', {
                defaultValue: 0.6,
            });
            this._ignoreYOutliersObserver = getBooleanObserver('_ignoreYOutliers', {
                defaultValue: true,
                useLocalStorage: true,
            });
        }
        ready() {
            super.ready();
            if (this.reloadOnReady)
                this.reload();
        }
        reload() {
            const url = getRouter().pluginsListing();
            const handlePluginsListingResponse = this._canceller.cancellable((result) => {
                if (result.cancelled) {
                    return;
                }
                this.set('_dataNotFound', !result.value['custom_scalars']);
                if (this._dataNotFound) {
                    return;
                }
                this._retrieveLayoutAndData();
            });
            this._requestManager.request(url).then(handlePluginsListingResponse);
        }
        _reloadCharts() {
            const charts = this.root.querySelectorAll('tf-custom-scalar-margin-chart-card, ' +
                'tf-custom-scalar-multi-line-chart-card');
            charts.forEach((chart) => {
                chart.reload();
            });
        }
        _retrieveLayoutAndData() {
            const url = getRouter().pluginRoute('custom_scalars', '/layout');
            const update = this._canceller.cancellable((result) => {
                if (result.cancelled) {
                    return;
                }
                // This plugin is only active if data is available.
                this.set('_layout', result.value);
                if (!this._dataNotFound) {
                    this._reloadCharts();
                }
            });
            this._requestManager.request(url).then(update);
        }
        get _smoothingEnabled() {
            var _smoothingWeight = this._smoothingWeight;
            return _smoothingWeight > 0;
        }
        get _categories() {
            var layout = this._layout;
            if (!layout.category) {
                return [];
            }
            let firstTimeLoad = false;
            if (!this._openedCategories) {
                // This is the first time the user loads the categories. Start storing
                // which categories are open.
                firstTimeLoad = true;
                this._openedCategories = {};
            }
            const categories = layout.category.map((category) => {
                if (firstTimeLoad && !category /* ??? */.closed) {
                    // Remember whether this category is currently open.
                    this._openedCategories[category.title] = true;
                }
                return {
                    name: category.title,
                    items: category.chart,
                    metadata: {
                        type: CategoryType.PREFIX_GROUP,
                        opened: !!this._openedCategories[category.title],
                    },
                };
            });
            return categories;
        }
        _categoryOpenedToggled(event) {
            const pane = event.target;
            if (pane.opened) {
                this._openedCategories[pane.category.name] = true;
            }
            else {
                delete this._openedCategories[pane.category.name];
            }
        }
    };
    TfCustomScalarDashboard.template = html `
    <tf-dashboard-layout>
      <div class="sidebar" slot="sidebar">
        <div class="settings">
          <div class="sidebar-section">
            <div class="line-item">
              <paper-checkbox checked="{{_showDownloadLinks}}"
                >Show data download links</paper-checkbox
              >
            </div>
            <div class="line-item">
              <paper-checkbox checked="{{_ignoreYOutliers}}"
                >Ignore outliers in chart scaling</paper-checkbox
              >
            </div>
            <div id="tooltip-sorting">
              <div id="tooltip-sorting-label">Tooltip sorting method:</div>
              <paper-dropdown-menu
                no-label-float=""
                selected-item-label="{{_tooltipSortingMethod}}"
              >
                <paper-listbox
                  class="dropdown-content"
                  selected="0"
                  slot="dropdown-content"
                >
                  <paper-item>default</paper-item>
                  <paper-item>descending</paper-item>
                  <paper-item>ascending</paper-item>
                  <paper-item>nearest</paper-item>
                </paper-listbox>
              </paper-dropdown-menu>
            </div>
          </div>
          <div class="sidebar-section">
            <tf-smoothing-input
              weight="{{_smoothingWeight}}"
              step="0.001"
              min="0"
              max="1"
            ></tf-smoothing-input>
          </div>
          <div class="sidebar-section">
            <tf-option-selector
              id="x-type-selector"
              name="Horizontal Axis"
              selected-id="{{_xType}}"
            >
              <paper-button id="step">step</paper-button
              ><!--
            --><paper-button id="relative">relative</paper-button
              ><!--
            --><paper-button id="wall_time">wall</paper-button>
            </tf-option-selector>
          </div>
        </div>
        <div class="sidebar-section runs-selector">
          <tf-runs-selector selected-runs="{{_selectedRuns}}">
          </tf-runs-selector>
        </div>
      </div>
      <div class="center" slot="center" id="categories-container">
        <template is="dom-if" if="[[_dataNotFound]]">
          <div class="no-data-warning">
            <h3>The custom scalars dashboard is inactive.</h3>
            <p>Probable causes:</p>
            <ol>
              <li>You haven't laid out the dashboard.</li>
              <li>You haven’t written any scalar data to your event files.</li>
            </ol>

            <p>
              To lay out the dashboard, pass a <code>Layout</code> protobuffer
              to the <code>set_layout</code> method. For example,
            </p>
            <pre>
from tensorboard import summary
from tensorboard.plugins.custom_scalar import layout_pb2
...
# This action does not have to be performed at every step, so the action is not
# taken care of by an op in the graph. We only need to specify the layout once
# (instead of per step).
layout_summary = summary_lib.custom_scalar_pb(layout_pb2.Layout(
  category=[
    layout_pb2.Category(
      title='losses',
      chart=[
          layout_pb2.Chart(
              title='losses',
              multiline=layout_pb2.MultilineChartContent(
                tag=[r'loss.*'],
              )),
          layout_pb2.Chart(
              title='baz',
              margin=layout_pb2.MarginChartContent(
                series=[
                  layout_pb2.MarginChartContent.Series(
                    value='loss/baz/scalar_summary',
                    lower='baz_lower/baz/scalar_summary',
                    upper='baz_upper/baz/scalar_summary'),
                ],
              )),
      ]),
    layout_pb2.Category(
      title='trig functions',
      chart=[
          layout_pb2.Chart(
              title='wave trig functions',
              multiline=layout_pb2.MultilineChartContent(
                tag=[r'trigFunctions/cosine', r'trigFunctions/sine'],
              )),
          # The range of tangent is different. Let's give it its own chart.
          layout_pb2.Chart(
              title='tan',
              multiline=layout_pb2.MultilineChartContent(
                tag=[r'trigFunctions/tangent'],
              )),
      ],
      # This category we care less about. Let's make it initially closed.
      closed=True),
  ]))
writer.add_summary(layout_summary)
</pre
            >
            <p>
              If you’re new to using TensorBoard, and want to find out how to
              add data and set up your event files, check out the
              <a
                href="https://github.com/tensorflow/tensorboard/blob/master/README.md"
                >README</a
              >
              and perhaps the
              <a
                href="https://www.tensorflow.org/get_started/summaries_and_tensorboard"
                >TensorBoard tutorial</a
              >.
            </p>
          </div>
        </template>
        <template is="dom-if" if="[[!_dataNotFound]]">
          <template is="dom-repeat" items="[[_categories]]" as="category">
            <tf-category-paginated-view
              as="chart"
              category="[[category]]"
              disable-pagination
              initial-opened="[[category.metadata.opened]]"
            >
              <template>
                <template is="dom-if" if="[[chart.multiline]]">
                  <tf-custom-scalar-multi-line-chart-card
                    active="[[active]]"
                    request-manager="[[_requestManager]]"
                    runs="[[_selectedRuns]]"
                    title="[[chart.title]]"
                    x-type="[[_xType]]"
                    smoothing-enabled="[[_smoothingEnabled]]"
                    smoothing-weight="[[_smoothingWeight]]"
                    tooltip-sorting-method="[[tooltipSortingMethod]]"
                    ignore-y-outliers="[[_ignoreYOutliers]]"
                    show-download-links="[[_showDownloadLinks]]"
                    tag-regexes="[[chart.multiline.tag]]"
                  ></tf-custom-scalar-multi-line-chart-card>
                </template>
                <template is="dom-if" if="[[chart.margin]]">
                  <tf-custom-scalar-margin-chart-card
                    active="[[active]]"
                    request-manager="[[_requestManager]]"
                    runs="[[_selectedRuns]]"
                    title="[[chart.title]]"
                    x-type="[[_xType]]"
                    tooltip-sorting-method="[[tooltipSortingMethod]]"
                    ignore-y-outliers="[[_ignoreYOutliers]]"
                    show-download-links="[[_showDownloadLinks]]"
                    margin-chart-series="[[chart.margin.series]]"
                  ></tf-custom-scalar-margin-chart-card>
                </template>
              </template>
            </tf-category-paginated-view>
          </template>
        </template>
      </div>
    </tf-dashboard-layout>

    <style include="dashboard-style"></style>
    <style>
      #tooltip-sorting {
        align-items: center;
        display: flex;
        font-size: 14px;
        margin-top: 15px;
      }
      #tooltip-sorting paper-dropdown-menu {
        margin-left: 10px;
        --paper-input-container-focus-color: var(--tb-orange-strong);
        width: 105px;
      }
      .line-item {
        display: block;
        padding-top: 5px;
      }
      .no-data-warning {
        max-width: 540px;
        margin: 80px auto 0 auto;
      }
    </style>
  `;
    __decorate([
        property({ type: Object }),
        __metadata("design:type", RequestManager)
    ], TfCustomScalarDashboard.prototype, "_requestManager", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Canceller)
    ], TfCustomScalarDashboard.prototype, "_canceller", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfCustomScalarDashboard.prototype, "_selectedRuns", void 0);
    __decorate([
        property({
            type: Boolean,
            notify: true,
            observer: '_showDownloadLinksObserver',
        }),
        __metadata("design:type", Boolean)
    ], TfCustomScalarDashboard.prototype, "_showDownloadLinks", void 0);
    __decorate([
        property({
            type: Number,
            notify: true,
            observer: '_smoothingWeightObserver',
        }),
        __metadata("design:type", Number)
    ], TfCustomScalarDashboard.prototype, "_smoothingWeight", void 0);
    __decorate([
        property({
            type: Boolean,
            observer: '_ignoreYOutliersObserver',
        }),
        __metadata("design:type", Boolean)
    ], TfCustomScalarDashboard.prototype, "_ignoreYOutliers", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfCustomScalarDashboard.prototype, "_xType", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfCustomScalarDashboard.prototype, "_layout", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfCustomScalarDashboard.prototype, "_dataNotFound", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfCustomScalarDashboard.prototype, "_openedCategories", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfCustomScalarDashboard.prototype, "_active", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfCustomScalarDashboard.prototype, "reloadOnReady", void 0);
    __decorate([
        computed('_smoothingWeight'),
        __metadata("design:type", Boolean),
        __metadata("design:paramtypes", [])
    ], TfCustomScalarDashboard.prototype, "_smoothingEnabled", null);
    __decorate([
        computed('_layout'),
        __metadata("design:type", Array),
        __metadata("design:paramtypes", [])
    ], TfCustomScalarDashboard.prototype, "_categories", null);
    TfCustomScalarDashboard = __decorate([
        customElement('tf-custom-scalar-dashboard')
    ], TfCustomScalarDashboard);

    /* Copyright 2016 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    class DistributionChart {
        constructor(xType, colorScale) {
            this.run2datasets = {};
            this.colorScale = colorScale;
            this.buildChart(xType);
        }
        getDataset(run) {
            if (this.run2datasets[run] === undefined) {
                this.run2datasets[run] = new src$2.Dataset([], { run: run });
            }
            return this.run2datasets[run];
        }
        buildChart(xType) {
            if (this.outer) {
                this.outer.destroy();
            }
            let xComponents = getXComponents(xType);
            this.xAccessor = xComponents.accessor;
            this.xScale = xComponents.scale;
            this.xAxis = xComponents.axis;
            this.xAxis.margin(0);
            this.xAxis.tickLabelPadding(3);
            this.yScale = new src$2.Scales.Linear();
            this.yAxis = new src$2.Axes.Numeric(this.yScale, 'left');
            let yFormatter = multiscaleFormatter(Y_AXIS_FORMATTER_PRECISION);
            this.yAxis.margin(0).tickLabelPadding(5).formatter(yFormatter);
            this.yAxis.usesTextWidthApproximation(true);
            let center = this.buildPlot(this.xAccessor, this.xScale, this.yScale);
            this.gridlines = new src$2.Components.Gridlines(this.xScale, this.yScale);
            this.center = new src$2.Components.Group([this.gridlines, center]);
            this.outer = new src$2.Components.Table([
                [this.yAxis, this.center],
                [null, this.xAxis],
            ]);
        }
        buildPlot(xAccessor, xScale, yScale) {
            let percents = [0, 228, 1587, 3085, 5000, 6915, 8413, 9772, 10000];
            let opacities = lodash.range(percents.length - 1).map((i) => (percents[i + 1] - percents[i]) / 2500);
            let accessors = percents.map((p, i) => (datum) => datum[i][1]);
            let median = 4;
            let medianAccessor = accessors[median];
            let plots = lodash.range(accessors.length - 1).map((i) => {
                let p = new src$2.Plots.Area();
                p.x(xAccessor, xScale);
                let y0 = i > median ? accessors[i] : accessors[i + 1];
                let y = i > median ? accessors[i + 1] : accessors[i];
                p.y(y, yScale);
                p.y0(y0);
                p.attr('fill', (d, i, dataset) => this.colorScale.scale(dataset.metadata().run));
                p.attr('stroke', (d, i, dataset) => this.colorScale.scale(dataset.metadata().run));
                p.attr('stroke-weight', (d, i, m) => '0.5px');
                p.attr('stroke-opacity', () => opacities[i]);
                p.attr('fill-opacity', () => opacities[i]);
                return p;
            });
            let medianPlot = new src$2.Plots.Line();
            medianPlot.x(xAccessor, xScale);
            medianPlot.y(medianAccessor, yScale);
            medianPlot.attr('stroke', (d, i, m) => this.colorScale.scale(m.run));
            this.plots = plots;
            return new src$2.Components.Group(plots);
        }
        setVisibleSeries(runs) {
            this.runs = runs;
            let datasets = runs.map((r) => this.getDataset(r));
            this.plots.forEach((p) => p.datasets(datasets));
        }
        /**
         * Set the data of a series on the chart.
         */
        setSeriesData(name, data) {
            this.getDataset(name).data(data);
        }
        renderTo(targetSVG) {
            this.targetSVG = targetSVG;
            this.outer.renderTo(targetSVG);
        }
        redraw() {
            this.outer.redraw();
        }
        destroy() {
            this.outer.destroy();
        }
    }
    let _VzDistributionChart = class _VzDistributionChart extends LegacyElementMixin(PolymerElement) {
        constructor() {
            super(...arguments);
            this.colorScale = new src$2.Scales.Color().range(category10.slice());
            this.xType = 'step';
            this._visibleSeriesCache = [];
            this._seriesDataCache = {};
            this._makeChartAsyncCallbackId = null;
        }
        setVisibleSeries(names) {
            this._visibleSeriesCache = names;
            if (this._chart) {
                this._chart.setVisibleSeries(names);
                this.redraw();
            }
        }
        setSeriesData(name, data) {
            this._seriesDataCache[name] = data;
            if (this._chart) {
                this._chart.setSeriesData(name, data);
            }
        }
        redraw() {
            this._chart.redraw();
        }
        _makeChart() {
            var xType = this.xType;
            var colorScale = this.colorScale;
            var _attached = this._attached;
            if (this._makeChartAsyncCallbackId === null) {
                this.cancelAsync(this._makeChartAsyncCallbackId);
            }
            this._makeChartAsyncCallbackId = this.async(function () {
                this._makeChartAsyncCallbackId = null;
                if (!_attached)
                    return;
                if (this._chart)
                    this._chart.destroy();
                var chart = new DistributionChart(xType, colorScale);
                var svg = select(this.$.chartdiv);
                chart.renderTo(svg);
                this._chart = chart;
            }, 350);
        }
        _reloadFromCache() {
            if (this._chart) {
                this._chart.setVisibleSeries(this._visibleSeriesCache);
                this._visibleSeriesCache.forEach(function (name) {
                    this._chart.setSeriesData(name, this._seriesDataCache[name] || []);
                }.bind(this));
            }
        }
        attached() {
            this._attached = true;
        }
        detached() {
            this._attached = false;
        }
    };
    _VzDistributionChart.template = html `
    <style include="plottable-style"></style>
    <div id="chartdiv"></div>
    <style>
      :host {
        -webkit-user-select: none;
        -moz-user-select: none;
        display: flex;
        flex-direction: column;
        flex-grow: 1;
        flex-shrink: 1;
        position: relative;
      }
      #chartdiv {
        -webkit-user-select: none;
        -moz-user-select: none;
        flex-grow: 1;
        flex-shrink: 1;
      }
    </style>
  `;
    __decorate([
        property({ type: Object }),
        __metadata("design:type", src$2.Scales.Color)
    ], _VzDistributionChart.prototype, "colorScale", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], _VzDistributionChart.prototype, "xType", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], _VzDistributionChart.prototype, "_attached", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", DistributionChart)
    ], _VzDistributionChart.prototype, "_chart", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], _VzDistributionChart.prototype, "_visibleSeriesCache", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], _VzDistributionChart.prototype, "_seriesDataCache", void 0);
    __decorate([
        property({ type: Number }),
        __metadata("design:type", Number)
    ], _VzDistributionChart.prototype, "_makeChartAsyncCallbackId", void 0);
    __decorate([
        observe('xType', 'colorScale', '_attached'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], _VzDistributionChart.prototype, "_makeChart", null);
    __decorate([
        observe('_chart'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], _VzDistributionChart.prototype, "_reloadFromCache", null);
    _VzDistributionChart = __decorate([
        customElement('vz-distribution-chart')
    ], _VzDistributionChart);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
      tf-distribution-loader loads an individual distribution from the
      TensorBoard backend, and renders it into a vz-distribution-chart.
    */
    let _TfDistributionLoader = class _TfDistributionLoader extends DataLoaderBehavior(LegacyElementMixin(PolymerElement)) {
        constructor() {
            super(...arguments);
            this.getDataLoadName = ({ run }) => run;
            this.requestData = (items, onLoad, onFinish) => {
                const router = getRouter();
                const baseUrl = router.pluginRoute('distributions', '/distributions');
                Promise.all(items.map((item) => {
                    const url = addParams(baseUrl, { tag: item.tag, run: item.run });
                    return this.requestManager
                        .request(url)
                        .then((data) => void onLoad({ item, data }));
                })).finally(() => void onFinish());
            };
            this.loadDataCallback = (_, datum, backendData) => {
                const data = backendData.map((datum) => {
                    // `vz-distribution-chart` wants each datum as an array with
                    // extra `wall_time` and `step` properties.
                    const [wall_time, step, bins] = datum;
                    bins.wall_time = new Date(wall_time * 1000);
                    bins.step = step;
                    return bins;
                });
                const name = this.getDataLoadName(datum);
                this.$.chart.setSeriesData(name, data);
                this.$.chart.setVisibleSeries([name]);
            };
            this._colorScale = { scale: runsColorScale };
            this._expanded = false;
            this._canceller = new Canceller();
        }
        _reloadOnRunTagChange() {
            this.reload();
        }
        _updateDataToLoad() {
            var run = this.run;
            var tag = this.tag;
            this.dataToLoad = [{ run, tag }];
        }
        get _runColor() {
            var run = this.run;
            return this._colorScale.scale(run);
        }
        /**
         * Ask the distribution chart to redraw itself. This should be
         * called whenever the dimensions of the view change (e.g., when
         * the card is expanded), as the distribution chart will need to
         * recalculate its layout.
         */
        redraw() {
            this.$.chart.redraw();
        }
        _toggleExpanded(e) {
            this.set('_expanded', !this._expanded);
            this.redraw();
        }
    };
    _TfDistributionLoader.template = html `
    <tf-card-heading
      tag="[[tag]]"
      run="[[run]]"
      display-name="[[tagMetadata.displayName]]"
      description="[[tagMetadata.description]]"
      color="[[_runColor]]"
    ></tf-card-heading>
    <!--
      The main distribution that we render. Data is set directly with
      \`setSeriesData\`, not with a bound property.
    -->
    <vz-distribution-chart
      id="chart"
      x-type="[[xType]]"
      color-scale="[[_colorScale]]"
    ></vz-distribution-chart>
    <div style="display: flex; flex-direction: row;">
      <paper-icon-button
        selected$="[[_expanded]]"
        icon="fullscreen"
        on-tap="_toggleExpanded"
      ></paper-icon-button>
    </div>
    <style>
      :host {
        display: flex;
        flex-direction: column;
        width: 330px;
        height: 235px;
        margin-right: 10px;
        margin-bottom: 15px;
      }
      :host([_expanded]) {
        width: 700px;
        height: 500px;
      }

      vz-histogram-timeseries {
        -moz-user-select: none;
        -webkit-user-select: none;
      }

      paper-icon-button {
        color: #2196f3;
        border-radius: 100%;
        width: 32px;
        height: 32px;
        padding: 4px;
      }
      paper-icon-button[selected] {
        background: var(--tb-ui-light-accent);
      }

      tf-card-heading {
        margin-bottom: 10px;
      }
    </style>
  `;
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], _TfDistributionLoader.prototype, "run", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], _TfDistributionLoader.prototype, "tag", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], _TfDistributionLoader.prototype, "tagMetadata", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], _TfDistributionLoader.prototype, "xType", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], _TfDistributionLoader.prototype, "getDataLoadName", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], _TfDistributionLoader.prototype, "loadDataCallback", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], _TfDistributionLoader.prototype, "_colorScale", void 0);
    __decorate([
        property({
            type: Boolean,
            reflectToAttribute: true,
        }),
        __metadata("design:type", Boolean)
    ], _TfDistributionLoader.prototype, "_expanded", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", RequestManager)
    ], _TfDistributionLoader.prototype, "requestManager", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Canceller)
    ], _TfDistributionLoader.prototype, "_canceller", void 0);
    __decorate([
        observe('run', 'tag'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], _TfDistributionLoader.prototype, "_reloadOnRunTagChange", null);
    __decorate([
        observe('run', 'tag'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], _TfDistributionLoader.prototype, "_updateDataToLoad", null);
    __decorate([
        computed('run'),
        __metadata("design:type", String),
        __metadata("design:paramtypes", [])
    ], _TfDistributionLoader.prototype, "_runColor", null);
    _TfDistributionLoader = __decorate([
        customElement('tf-distribution-loader')
    ], _TfDistributionLoader);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
      A frontend that displays a set of tf-distribution-loaders, each of
      which displays the distribution for a single tag on a single run. This
      dashboard provides a categorizer and abcissa seletor (step, relative,
      or wall time).
    */
    let TfDistributionDashboard = class TfDistributionDashboard extends LegacyElementMixin(PolymerElement) {
        constructor() {
            super(...arguments);
            this.reloadOnReady = true;
            this._xType = 'step';
            this._requestManager = new RequestManager();
        }
        ready() {
            super.ready();
            if (this.reloadOnReady)
                this.reload();
        }
        reload() {
            this._fetchTags().then(() => {
                this._reloadDistributions();
            });
        }
        _fetchTags() {
            const url = getRouter().pluginRoute('distributions', '/tags');
            return this._requestManager.request(url).then((runToTagInfo) => {
                if (lodash.isEqual(runToTagInfo, this._runToTagInfo)) {
                    // No need to update anything if there are no changes.
                    return;
                }
                const runToTag = lodash.mapValues(runToTagInfo, (x) => Object.keys(x));
                const tags = getTags(runToTag);
                this.set('_dataNotFound', tags.length === 0);
                this.set('_runToTag', runToTag);
                this.set('_runToTagInfo', runToTagInfo);
                this.async(() => {
                    // See the comment above `_categoriesDomReady`.
                    this.set('_categoriesDomReady', true);
                });
            });
        }
        _reloadDistributions() {
            this.root.querySelectorAll('tf-distribution-loader').forEach((loader) => {
                loader.reload();
            });
        }
        _shouldOpen(index) {
            return index <= 2;
        }
        get _categories() {
            var runToTag = this._runToTag;
            var selectedRuns = this._selectedRuns;
            var tagFilter = this._tagFilter;
            var categoriesDomReady = this._categoriesDomReady;
            return categorizeRunTagCombinations(runToTag, selectedRuns, tagFilter);
        }
        _tagMetadata(runToTagInfo, run, tag) {
            return runToTagInfo[run][tag];
        }
    };
    TfDistributionDashboard.template = html `
    <tf-dashboard-layout>
      <div class="sidebar" slot="sidebar">
        <div class="settings">
          <div class="sidebar-section">
            <tf-option-selector
              id="xTypeSelector"
              name="Horizontal axis"
              selected-id="{{_xType}}"
            >
              <paper-button id="step">step</paper-button>
              <paper-button id="relative">relative</paper-button>
              <paper-button id="wall_time">wall</paper-button>
            </tf-option-selector>
          </div>
        </div>
        <div class="sidebar-section runs-selector">
          <tf-runs-selector selected-runs="{{_selectedRuns}}">
          </tf-runs-selector>
        </div>
      </div>

      <div class="center" slot="center">
        <template is="dom-if" if="[[_dataNotFound]]">
          <div class="no-data-warning">
            <h3>No distribution data was found.</h3>
            <p>Probable causes:</p>
            <ul>
              <li>
                You haven’t written any histogram data to your event files.
                (Histograms and distributions both use the histogram summary
                operation.)
              </li>

              <li>TensorBoard can’t find your event files.</li>
            </ul>

            <p>
              If you’re new to using TensorBoard, and want to find out how to
              add data and set up your event files, check out the
              <a
                href="https://github.com/tensorflow/tensorboard/blob/master/README.md"
                >README</a
              >
              and perhaps the
              <a
                href="https://www.tensorflow.org/get_started/summaries_and_tensorboard"
                >TensorBoard tutorial</a
              >.
            </p>

            <p>
              If you think TensorBoard is configured properly, please see
              <a
                href="https://github.com/tensorflow/tensorboard/blob/master/README.md#my-tensorboard-isnt-showing-any-data-whats-wrong"
                >the section of the README devoted to missing data problems</a
              >
              and consider filing an issue on GitHub.
            </p>
          </div>
        </template>
        <template is="dom-if" if="[[!_dataNotFound]]">
          <tf-tag-filterer tag-filter="{{_tagFilter}}"></tf-tag-filterer>
          <template is="dom-repeat" items="[[_categories]]" as="category">
            <tf-category-paginated-view
              category="[[category]]"
              initial-opened="[[_shouldOpen(index)]]"
            >
              <template>
                <tf-distribution-loader
                  active="[[active]]"
                  run="[[item.run]]"
                  tag="[[item.tag]]"
                  tag-metadata="[[_tagMetadata(_runToTagInfo, item.run, item.tag)]]"
                  x-type="[[_xType]]"
                  request-manager="[[_requestManager]]"
                ></tf-distribution-loader>
              </template>
            </tf-category-paginated-view>
          </template>
        </template>
      </div>
    </tf-dashboard-layout>

    <style include="dashboard-style"></style>
    <style>
      .no-data-warning {
        max-width: 540px;
        margin: 80px auto 0 auto;
      }
    </style>
  `;
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfDistributionDashboard.prototype, "reloadOnReady", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfDistributionDashboard.prototype, "_xType", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfDistributionDashboard.prototype, "_selectedRuns", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfDistributionDashboard.prototype, "_runToTag", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfDistributionDashboard.prototype, "_runToTagInfo", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfDistributionDashboard.prototype, "_dataNotFound", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfDistributionDashboard.prototype, "_tagFilter", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfDistributionDashboard.prototype, "_categoriesDomReady", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", RequestManager)
    ], TfDistributionDashboard.prototype, "_requestManager", void 0);
    __decorate([
        computed('_runToTag', '_selectedRuns', '_tagFilter', '_categoriesDomReady'),
        __metadata("design:type", Array),
        __metadata("design:paramtypes", [])
    ], TfDistributionDashboard.prototype, "_categories", null);
    TfDistributionDashboard = __decorate([
        customElement('tf-distribution-dashboard')
    ], TfDistributionDashboard);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/

    var tf_storage = /*#__PURE__*/Object.freeze({
        __proto__: null,
        ListenKey: ListenKey,
        addHashListener: addHashListener,
        addStorageListener: addStorageListener,
        fireStorageChanged: fireStorageChanged,
        removeHashListenerByKey: removeHashListenerByKey,
        removeStorageListenerByKey: removeStorageListenerByKey,
        TAB: TAB,
        DISAMBIGUATOR: DISAMBIGUATOR,
        getUrlDict: getUrlDict,
        getString: getString,
        setString: setString,
        getStringInitializer: getStringInitializer,
        getStringObserver: getStringObserver,
        disposeStringBinding: disposeStringBinding,
        getBoolean: getBoolean,
        setBoolean: setBoolean,
        getBooleanInitializer: getBooleanInitializer,
        getBooleanObserver: getBooleanObserver,
        disposeBooleanBinding: disposeBooleanBinding,
        getNumber: getNumber,
        setNumber: setNumber,
        getNumberInitializer: getNumberInitializer,
        getNumberObserver: getNumberObserver,
        disposeNumberBinding: disposeNumberBinding,
        getObject: getObject,
        setObject: setObject,
        getObjectInitializer: getObjectInitializer,
        getObjectObserver: getObjectObserver,
        disposeObjectBinding: disposeObjectBinding,
        makeBindings: makeBindings,
        migrateLegacyURLScheme: migrateLegacyURLScheme
    });

    /* Copyright 2015 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the 'License');
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an 'AS IS' BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    // Note that tf-graph-control depends on the value of the enum.
    // Polymer does not let one use JS variable as a prop.
    var SelectionType;
    (function (SelectionType) {
        SelectionType["OP_GRAPH"] = "op_graph";
        SelectionType["CONCEPTUAL_GRAPH"] = "conceptual_graph";
        SelectionType["PROFILE"] = "profile";
    })(SelectionType || (SelectionType = {}));
    /** Enums element class of objects in the scene */
    let Class$1 = {
        Node: {
            // <g> element that contains nodes.
            CONTAINER: 'nodes',
            // <g> element that contains detail about a node.
            GROUP: 'node',
            // <g> element that contains visual elements (like rect, ellipse).
            SHAPE: 'nodeshape',
            // <*> element(s) under SHAPE that should receive color updates.
            COLOR_TARGET: 'nodecolortarget',
            // <text> element showing the node's label.
            LABEL: 'nodelabel',
            // <g> element that contains all visuals for the expand/collapse
            // button for expandable group nodes.
            BUTTON_CONTAINER: 'buttoncontainer',
            // <circle> element that surrounds expand/collapse buttons.
            BUTTON_CIRCLE: 'buttoncircle',
            // <path> element of the expand button.
            EXPAND_BUTTON: 'expandbutton',
            // <path> element of the collapse button.
            COLLAPSE_BUTTON: 'collapsebutton',
        },
        Edge: {
            CONTAINER: 'edges',
            GROUP: 'edge',
            LINE: 'edgeline',
            REFERENCE_EDGE: 'referenceedge',
            REF_LINE: 'refline',
            SELECTABLE: 'selectableedge',
            SELECTED: 'selectededge',
            STRUCTURAL: 'structural',
        },
        Annotation: {
            OUTBOX: 'out-annotations',
            INBOX: 'in-annotations',
            GROUP: 'annotation',
            NODE: 'annotation-node',
            EDGE: 'annotation-edge',
            CONTROL_EDGE: 'annotation-control-edge',
            LABEL: 'annotation-label',
            ELLIPSIS: 'annotation-ellipsis',
        },
        Scene: {
            GROUP: 'scene',
            CORE: 'core',
            FUNCTION_LIBRARY: 'function-library',
            INEXTRACT: 'in-extract',
            OUTEXTRACT: 'out-extract',
        },
        Subscene: { GROUP: 'subscene' },
        OPNODE: 'op',
        METANODE: 'meta',
        SERIESNODE: 'series',
        BRIDGENODE: 'bridge',
        ELLIPSISNODE: 'ellipsis',
    };
    const SVG_NAMESPACE = 'http://www.w3.org/2000/svg';
    /**
     * Given a container d3 selection, select a child element of a given tag and
     * class. If multiple children matches the tag and class name, returns only
     * the first one.
     *
     * @param container
     * @param tagName tag name.
     * @param className (optional) Class name or list of class names.
     * @return selection of the element, or an empty selection
     */
    function selectChild(container, tagName, className) {
        let children = container.node().childNodes;
        for (let i = 0; i < children.length; i++) {
            let child = children[i];
            if (child.tagName === tagName) {
                if (className instanceof Array) {
                    let hasAllClasses = true;
                    for (let j = 0; j < className.length; j++) {
                        hasAllClasses =
                            hasAllClasses && child.classList.contains(className[j]);
                    }
                    if (hasAllClasses) {
                        return select(child);
                    }
                }
                else if (!className || child.classList.contains(className)) {
                    return select(child);
                }
            }
        }
        return select(null);
    }
    /**
     * Given a container d3 selection, select a child svg element of a given tag
     * and class if exists or append / insert one otherwise.  If multiple children
     * matches the tag and class name, returns only the first one.
     *
     * @param container
     * @param tagName tag name.
     * @param className (optional) Class name or a list of class names.
     * @param before (optional) reference DOM node for insertion.
     * @return selection of the element
     */
    function selectOrCreateChild(container, tagName, className, before) {
        let child = selectChild(container, tagName, className);
        if (!child.empty()) {
            return child;
        }
        let newElement = document.createElementNS('http://www.w3.org/2000/svg', tagName);
        if (className instanceof Array) {
            for (let i = 0; i < className.length; i++) {
                newElement.classList.add(className[i]);
            }
        }
        else {
            newElement.classList.add(className);
        }
        if (before) {
            // if before exists, insert
            container.node().insertBefore(newElement, before);
        }
        else {
            // otherwise, append
            container.node().appendChild(newElement);
        }
        return (select(newElement)
            // need to bind data to emulate d3_selection.append
            .datum(container.datum()));
    }
    /**
     * Execution stats for the node.
     */
    class NodeStats {
        constructor(outputSize) {
            /**
             * Total number of bytes used for the node. Sum of all children
             * if it is a Group node.
             */
            this.totalBytes = 0;
            this.outputSize = outputSize;
        }
        /**
         * Add the start and end time for a particular kernel execution of this op.
         * Ops can have multiple kernel executions within the same session run.
         */
        addExecutionTime(startTime, endTime) {
            if (this.startTime != null) {
                this.startTime = Math.min(this.startTime, startTime);
            }
            else {
                this.startTime = startTime;
            }
            if (this.endTime != null) {
                this.endTime = Math.max(this.endTime, endTime);
            }
            else {
                this.endTime = endTime;
            }
        }
        /**
         * Add the bytes allocated for a particular kernel execution of this op.
         * Ops can have multiple kernel executions within the same session run.
         */
        addBytesAllocation(totalBytes) {
            if (this.totalBytes != null) {
                this.totalBytes = Math.max(this.totalBytes, totalBytes);
            }
            else {
                this.totalBytes = totalBytes;
            }
        }
        /**
         * Combines the specified stats with the current stats.
         * Modifies the current object. This method is used to
         * compute aggregate stats for group nodes.
         */
        combine(stats) {
            if (stats.totalBytes != null) {
                this.totalBytes += stats.totalBytes;
            }
            if (stats.getTotalMicros() != null) {
                this.addExecutionTime(stats.startTime, stats.endTime);
            }
        }
        /**
         * Total number of compute time in microseconds used for the node.
         * Sum of all children if it is a Group node. Null if it is unknown.
         * This method can not be scaffolded under a getter attribute because
         * ECMAScript 5 does not support getter attributes.
         */
        getTotalMicros() {
            if (this.startTime == null || this.endTime == null) {
                return null;
            }
            return this.endTime - this.startTime;
        }
    }
    /** The minimum stroke width of an edge. */
    const MIN_EDGE_WIDTH = 0.75;
    /** The maximum stroke width of an edge. */
    const MAX_EDGE_WIDTH = 12;
    /** The exponent used in the power scale for edge thickness. */
    const EDGE_WIDTH_SCALE_EXPONENT = 0.3;
    /** The domain (min and max value) for the edge width. */
    const DOMAIN_EDGE_WIDTH_SCALE = [1, 5000000];
    const EDGE_WIDTH_SIZE_BASED_SCALE = pow$1()
        .exponent(EDGE_WIDTH_SCALE_EXPONENT)
        .domain(DOMAIN_EDGE_WIDTH_SCALE)
        .range([MIN_EDGE_WIDTH, MAX_EDGE_WIDTH])
        .clamp(true);

    /**
     * Removes all key-value entries from the list cache.
     *
     * @private
     * @name clear
     * @memberOf ListCache
     */
    function listCacheClear() {
      this.__data__ = [];
      this.size = 0;
    }

    var _listCacheClear = listCacheClear;

    /**
     * Performs a
     * [`SameValueZero`](http://ecma-international.org/ecma-262/7.0/#sec-samevaluezero)
     * comparison between two values to determine if they are equivalent.
     *
     * @static
     * @memberOf _
     * @since 4.0.0
     * @category Lang
     * @param {*} value The value to compare.
     * @param {*} other The other value to compare.
     * @returns {boolean} Returns `true` if the values are equivalent, else `false`.
     * @example
     *
     * var object = { 'a': 1 };
     * var other = { 'a': 1 };
     *
     * _.eq(object, object);
     * // => true
     *
     * _.eq(object, other);
     * // => false
     *
     * _.eq('a', 'a');
     * // => true
     *
     * _.eq('a', Object('a'));
     * // => false
     *
     * _.eq(NaN, NaN);
     * // => true
     */
    function eq(value, other) {
      return value === other || (value !== value && other !== other);
    }

    var eq_1 = eq;

    /**
     * Gets the index at which the `key` is found in `array` of key-value pairs.
     *
     * @private
     * @param {Array} array The array to inspect.
     * @param {*} key The key to search for.
     * @returns {number} Returns the index of the matched value, else `-1`.
     */
    function assocIndexOf(array, key) {
      var length = array.length;
      while (length--) {
        if (eq_1(array[length][0], key)) {
          return length;
        }
      }
      return -1;
    }

    var _assocIndexOf = assocIndexOf;

    /** Used for built-in method references. */
    var arrayProto = Array.prototype;

    /** Built-in value references. */
    var splice = arrayProto.splice;

    /**
     * Removes `key` and its value from the list cache.
     *
     * @private
     * @name delete
     * @memberOf ListCache
     * @param {string} key The key of the value to remove.
     * @returns {boolean} Returns `true` if the entry was removed, else `false`.
     */
    function listCacheDelete(key) {
      var data = this.__data__,
          index = _assocIndexOf(data, key);

      if (index < 0) {
        return false;
      }
      var lastIndex = data.length - 1;
      if (index == lastIndex) {
        data.pop();
      } else {
        splice.call(data, index, 1);
      }
      --this.size;
      return true;
    }

    var _listCacheDelete = listCacheDelete;

    /**
     * Gets the list cache value for `key`.
     *
     * @private
     * @name get
     * @memberOf ListCache
     * @param {string} key The key of the value to get.
     * @returns {*} Returns the entry value.
     */
    function listCacheGet(key) {
      var data = this.__data__,
          index = _assocIndexOf(data, key);

      return index < 0 ? undefined : data[index][1];
    }

    var _listCacheGet = listCacheGet;

    /**
     * Checks if a list cache value for `key` exists.
     *
     * @private
     * @name has
     * @memberOf ListCache
     * @param {string} key The key of the entry to check.
     * @returns {boolean} Returns `true` if an entry for `key` exists, else `false`.
     */
    function listCacheHas(key) {
      return _assocIndexOf(this.__data__, key) > -1;
    }

    var _listCacheHas = listCacheHas;

    /**
     * Sets the list cache `key` to `value`.
     *
     * @private
     * @name set
     * @memberOf ListCache
     * @param {string} key The key of the value to set.
     * @param {*} value The value to set.
     * @returns {Object} Returns the list cache instance.
     */
    function listCacheSet(key, value) {
      var data = this.__data__,
          index = _assocIndexOf(data, key);

      if (index < 0) {
        ++this.size;
        data.push([key, value]);
      } else {
        data[index][1] = value;
      }
      return this;
    }

    var _listCacheSet = listCacheSet;

    /**
     * Creates an list cache object.
     *
     * @private
     * @constructor
     * @param {Array} [entries] The key-value pairs to cache.
     */
    function ListCache(entries) {
      var index = -1,
          length = entries == null ? 0 : entries.length;

      this.clear();
      while (++index < length) {
        var entry = entries[index];
        this.set(entry[0], entry[1]);
      }
    }

    // Add methods to `ListCache`.
    ListCache.prototype.clear = _listCacheClear;
    ListCache.prototype['delete'] = _listCacheDelete;
    ListCache.prototype.get = _listCacheGet;
    ListCache.prototype.has = _listCacheHas;
    ListCache.prototype.set = _listCacheSet;

    var _ListCache = ListCache;

    /**
     * Removes all key-value entries from the stack.
     *
     * @private
     * @name clear
     * @memberOf Stack
     */
    function stackClear() {
      this.__data__ = new _ListCache;
      this.size = 0;
    }

    var _stackClear = stackClear;

    /**
     * Removes `key` and its value from the stack.
     *
     * @private
     * @name delete
     * @memberOf Stack
     * @param {string} key The key of the value to remove.
     * @returns {boolean} Returns `true` if the entry was removed, else `false`.
     */
    function stackDelete(key) {
      var data = this.__data__,
          result = data['delete'](key);

      this.size = data.size;
      return result;
    }

    var _stackDelete = stackDelete;

    /**
     * Gets the stack value for `key`.
     *
     * @private
     * @name get
     * @memberOf Stack
     * @param {string} key The key of the value to get.
     * @returns {*} Returns the entry value.
     */
    function stackGet(key) {
      return this.__data__.get(key);
    }

    var _stackGet = stackGet;

    /**
     * Checks if a stack value for `key` exists.
     *
     * @private
     * @name has
     * @memberOf Stack
     * @param {string} key The key of the entry to check.
     * @returns {boolean} Returns `true` if an entry for `key` exists, else `false`.
     */
    function stackHas(key) {
      return this.__data__.has(key);
    }

    var _stackHas = stackHas;

    /** Detect free variable `global` from Node.js. */
    var freeGlobal = typeof commonjsGlobal == 'object' && commonjsGlobal && commonjsGlobal.Object === Object && commonjsGlobal;

    var _freeGlobal = freeGlobal;

    /** Detect free variable `self`. */
    var freeSelf = typeof self == 'object' && self && self.Object === Object && self;

    /** Used as a reference to the global object. */
    var root$9 = _freeGlobal || freeSelf || Function('return this')();

    var _root = root$9;

    /** Built-in value references. */
    var Symbol$1 = _root.Symbol;

    var _Symbol = Symbol$1;

    /** Used for built-in method references. */
    var objectProto = Object.prototype;

    /** Used to check objects for own properties. */
    var hasOwnProperty = objectProto.hasOwnProperty;

    /**
     * Used to resolve the
     * [`toStringTag`](http://ecma-international.org/ecma-262/7.0/#sec-object.prototype.tostring)
     * of values.
     */
    var nativeObjectToString = objectProto.toString;

    /** Built-in value references. */
    var symToStringTag = _Symbol ? _Symbol.toStringTag : undefined;

    /**
     * A specialized version of `baseGetTag` which ignores `Symbol.toStringTag` values.
     *
     * @private
     * @param {*} value The value to query.
     * @returns {string} Returns the raw `toStringTag`.
     */
    function getRawTag(value) {
      var isOwn = hasOwnProperty.call(value, symToStringTag),
          tag = value[symToStringTag];

      try {
        value[symToStringTag] = undefined;
        var unmasked = true;
      } catch (e) {}

      var result = nativeObjectToString.call(value);
      if (unmasked) {
        if (isOwn) {
          value[symToStringTag] = tag;
        } else {
          delete value[symToStringTag];
        }
      }
      return result;
    }

    var _getRawTag = getRawTag;

    /** Used for built-in method references. */
    var objectProto$1 = Object.prototype;

    /**
     * Used to resolve the
     * [`toStringTag`](http://ecma-international.org/ecma-262/7.0/#sec-object.prototype.tostring)
     * of values.
     */
    var nativeObjectToString$1 = objectProto$1.toString;

    /**
     * Converts `value` to a string using `Object.prototype.toString`.
     *
     * @private
     * @param {*} value The value to convert.
     * @returns {string} Returns the converted string.
     */
    function objectToString(value) {
      return nativeObjectToString$1.call(value);
    }

    var _objectToString = objectToString;

    /** `Object#toString` result references. */
    var nullTag = '[object Null]',
        undefinedTag = '[object Undefined]';

    /** Built-in value references. */
    var symToStringTag$1 = _Symbol ? _Symbol.toStringTag : undefined;

    /**
     * The base implementation of `getTag` without fallbacks for buggy environments.
     *
     * @private
     * @param {*} value The value to query.
     * @returns {string} Returns the `toStringTag`.
     */
    function baseGetTag(value) {
      if (value == null) {
        return value === undefined ? undefinedTag : nullTag;
      }
      return (symToStringTag$1 && symToStringTag$1 in Object(value))
        ? _getRawTag(value)
        : _objectToString(value);
    }

    var _baseGetTag = baseGetTag;

    /**
     * Checks if `value` is the
     * [language type](http://www.ecma-international.org/ecma-262/7.0/#sec-ecmascript-language-types)
     * of `Object`. (e.g. arrays, functions, objects, regexes, `new Number(0)`, and `new String('')`)
     *
     * @static
     * @memberOf _
     * @since 0.1.0
     * @category Lang
     * @param {*} value The value to check.
     * @returns {boolean} Returns `true` if `value` is an object, else `false`.
     * @example
     *
     * _.isObject({});
     * // => true
     *
     * _.isObject([1, 2, 3]);
     * // => true
     *
     * _.isObject(_.noop);
     * // => true
     *
     * _.isObject(null);
     * // => false
     */
    function isObject(value) {
      var type = typeof value;
      return value != null && (type == 'object' || type == 'function');
    }

    var isObject_1 = isObject;

    /** `Object#toString` result references. */
    var asyncTag = '[object AsyncFunction]',
        funcTag = '[object Function]',
        genTag = '[object GeneratorFunction]',
        proxyTag = '[object Proxy]';

    /**
     * Checks if `value` is classified as a `Function` object.
     *
     * @static
     * @memberOf _
     * @since 0.1.0
     * @category Lang
     * @param {*} value The value to check.
     * @returns {boolean} Returns `true` if `value` is a function, else `false`.
     * @example
     *
     * _.isFunction(_);
     * // => true
     *
     * _.isFunction(/abc/);
     * // => false
     */
    function isFunction(value) {
      if (!isObject_1(value)) {
        return false;
      }
      // The use of `Object#toString` avoids issues with the `typeof` operator
      // in Safari 9 which returns 'object' for typed arrays and other constructors.
      var tag = _baseGetTag(value);
      return tag == funcTag || tag == genTag || tag == asyncTag || tag == proxyTag;
    }

    var isFunction_1 = isFunction;

    /** Used to detect overreaching core-js shims. */
    var coreJsData = _root['__core-js_shared__'];

    var _coreJsData = coreJsData;

    /** Used to detect methods masquerading as native. */
    var maskSrcKey = (function() {
      var uid = /[^.]+$/.exec(_coreJsData && _coreJsData.keys && _coreJsData.keys.IE_PROTO || '');
      return uid ? ('Symbol(src)_1.' + uid) : '';
    }());

    /**
     * Checks if `func` has its source masked.
     *
     * @private
     * @param {Function} func The function to check.
     * @returns {boolean} Returns `true` if `func` is masked, else `false`.
     */
    function isMasked(func) {
      return !!maskSrcKey && (maskSrcKey in func);
    }

    var _isMasked = isMasked;

    /** Used for built-in method references. */
    var funcProto = Function.prototype;

    /** Used to resolve the decompiled source of functions. */
    var funcToString = funcProto.toString;

    /**
     * Converts `func` to its source code.
     *
     * @private
     * @param {Function} func The function to convert.
     * @returns {string} Returns the source code.
     */
    function toSource(func) {
      if (func != null) {
        try {
          return funcToString.call(func);
        } catch (e) {}
        try {
          return (func + '');
        } catch (e) {}
      }
      return '';
    }

    var _toSource = toSource;

    /**
     * Used to match `RegExp`
     * [syntax characters](http://ecma-international.org/ecma-262/7.0/#sec-patterns).
     */
    var reRegExpChar = /[\\^$.*+?()[\]{}|]/g;

    /** Used to detect host constructors (Safari). */
    var reIsHostCtor = /^\[object .+?Constructor\]$/;

    /** Used for built-in method references. */
    var funcProto$1 = Function.prototype,
        objectProto$2 = Object.prototype;

    /** Used to resolve the decompiled source of functions. */
    var funcToString$1 = funcProto$1.toString;

    /** Used to check objects for own properties. */
    var hasOwnProperty$1 = objectProto$2.hasOwnProperty;

    /** Used to detect if a method is native. */
    var reIsNative = RegExp('^' +
      funcToString$1.call(hasOwnProperty$1).replace(reRegExpChar, '\\$&')
      .replace(/hasOwnProperty|(function).*?(?=\\\()| for .+?(?=\\\])/g, '$1.*?') + '$'
    );

    /**
     * The base implementation of `_.isNative` without bad shim checks.
     *
     * @private
     * @param {*} value The value to check.
     * @returns {boolean} Returns `true` if `value` is a native function,
     *  else `false`.
     */
    function baseIsNative(value) {
      if (!isObject_1(value) || _isMasked(value)) {
        return false;
      }
      var pattern = isFunction_1(value) ? reIsNative : reIsHostCtor;
      return pattern.test(_toSource(value));
    }

    var _baseIsNative = baseIsNative;

    /**
     * Gets the value at `key` of `object`.
     *
     * @private
     * @param {Object} [object] The object to query.
     * @param {string} key The key of the property to get.
     * @returns {*} Returns the property value.
     */
    function getValue(object, key) {
      return object == null ? undefined : object[key];
    }

    var _getValue = getValue;

    /**
     * Gets the native function at `key` of `object`.
     *
     * @private
     * @param {Object} object The object to query.
     * @param {string} key The key of the method to get.
     * @returns {*} Returns the function if it's native, else `undefined`.
     */
    function getNative(object, key) {
      var value = _getValue(object, key);
      return _baseIsNative(value) ? value : undefined;
    }

    var _getNative = getNative;

    /* Built-in method references that are verified to be native. */
    var Map$5 = _getNative(_root, 'Map');

    var _Map = Map$5;

    /* Built-in method references that are verified to be native. */
    var nativeCreate = _getNative(Object, 'create');

    var _nativeCreate = nativeCreate;

    /**
     * Removes all key-value entries from the hash.
     *
     * @private
     * @name clear
     * @memberOf Hash
     */
    function hashClear() {
      this.__data__ = _nativeCreate ? _nativeCreate(null) : {};
      this.size = 0;
    }

    var _hashClear = hashClear;

    /**
     * Removes `key` and its value from the hash.
     *
     * @private
     * @name delete
     * @memberOf Hash
     * @param {Object} hash The hash to modify.
     * @param {string} key The key of the value to remove.
     * @returns {boolean} Returns `true` if the entry was removed, else `false`.
     */
    function hashDelete(key) {
      var result = this.has(key) && delete this.__data__[key];
      this.size -= result ? 1 : 0;
      return result;
    }

    var _hashDelete = hashDelete;

    /** Used to stand-in for `undefined` hash values. */
    var HASH_UNDEFINED = '__lodash_hash_undefined__';

    /** Used for built-in method references. */
    var objectProto$3 = Object.prototype;

    /** Used to check objects for own properties. */
    var hasOwnProperty$2 = objectProto$3.hasOwnProperty;

    /**
     * Gets the hash value for `key`.
     *
     * @private
     * @name get
     * @memberOf Hash
     * @param {string} key The key of the value to get.
     * @returns {*} Returns the entry value.
     */
    function hashGet(key) {
      var data = this.__data__;
      if (_nativeCreate) {
        var result = data[key];
        return result === HASH_UNDEFINED ? undefined : result;
      }
      return hasOwnProperty$2.call(data, key) ? data[key] : undefined;
    }

    var _hashGet = hashGet;

    /** Used for built-in method references. */
    var objectProto$4 = Object.prototype;

    /** Used to check objects for own properties. */
    var hasOwnProperty$3 = objectProto$4.hasOwnProperty;

    /**
     * Checks if a hash value for `key` exists.
     *
     * @private
     * @name has
     * @memberOf Hash
     * @param {string} key The key of the entry to check.
     * @returns {boolean} Returns `true` if an entry for `key` exists, else `false`.
     */
    function hashHas(key) {
      var data = this.__data__;
      return _nativeCreate ? (data[key] !== undefined) : hasOwnProperty$3.call(data, key);
    }

    var _hashHas = hashHas;

    /** Used to stand-in for `undefined` hash values. */
    var HASH_UNDEFINED$1 = '__lodash_hash_undefined__';

    /**
     * Sets the hash `key` to `value`.
     *
     * @private
     * @name set
     * @memberOf Hash
     * @param {string} key The key of the value to set.
     * @param {*} value The value to set.
     * @returns {Object} Returns the hash instance.
     */
    function hashSet(key, value) {
      var data = this.__data__;
      this.size += this.has(key) ? 0 : 1;
      data[key] = (_nativeCreate && value === undefined) ? HASH_UNDEFINED$1 : value;
      return this;
    }

    var _hashSet = hashSet;

    /**
     * Creates a hash object.
     *
     * @private
     * @constructor
     * @param {Array} [entries] The key-value pairs to cache.
     */
    function Hash(entries) {
      var index = -1,
          length = entries == null ? 0 : entries.length;

      this.clear();
      while (++index < length) {
        var entry = entries[index];
        this.set(entry[0], entry[1]);
      }
    }

    // Add methods to `Hash`.
    Hash.prototype.clear = _hashClear;
    Hash.prototype['delete'] = _hashDelete;
    Hash.prototype.get = _hashGet;
    Hash.prototype.has = _hashHas;
    Hash.prototype.set = _hashSet;

    var _Hash = Hash;

    /**
     * Removes all key-value entries from the map.
     *
     * @private
     * @name clear
     * @memberOf MapCache
     */
    function mapCacheClear() {
      this.size = 0;
      this.__data__ = {
        'hash': new _Hash,
        'map': new (_Map || _ListCache),
        'string': new _Hash
      };
    }

    var _mapCacheClear = mapCacheClear;

    /**
     * Checks if `value` is suitable for use as unique object key.
     *
     * @private
     * @param {*} value The value to check.
     * @returns {boolean} Returns `true` if `value` is suitable, else `false`.
     */
    function isKeyable(value) {
      var type = typeof value;
      return (type == 'string' || type == 'number' || type == 'symbol' || type == 'boolean')
        ? (value !== '__proto__')
        : (value === null);
    }

    var _isKeyable = isKeyable;

    /**
     * Gets the data for `map`.
     *
     * @private
     * @param {Object} map The map to query.
     * @param {string} key The reference key.
     * @returns {*} Returns the map data.
     */
    function getMapData(map, key) {
      var data = map.__data__;
      return _isKeyable(key)
        ? data[typeof key == 'string' ? 'string' : 'hash']
        : data.map;
    }

    var _getMapData = getMapData;

    /**
     * Removes `key` and its value from the map.
     *
     * @private
     * @name delete
     * @memberOf MapCache
     * @param {string} key The key of the value to remove.
     * @returns {boolean} Returns `true` if the entry was removed, else `false`.
     */
    function mapCacheDelete(key) {
      var result = _getMapData(this, key)['delete'](key);
      this.size -= result ? 1 : 0;
      return result;
    }

    var _mapCacheDelete = mapCacheDelete;

    /**
     * Gets the map value for `key`.
     *
     * @private
     * @name get
     * @memberOf MapCache
     * @param {string} key The key of the value to get.
     * @returns {*} Returns the entry value.
     */
    function mapCacheGet(key) {
      return _getMapData(this, key).get(key);
    }

    var _mapCacheGet = mapCacheGet;

    /**
     * Checks if a map value for `key` exists.
     *
     * @private
     * @name has
     * @memberOf MapCache
     * @param {string} key The key of the entry to check.
     * @returns {boolean} Returns `true` if an entry for `key` exists, else `false`.
     */
    function mapCacheHas(key) {
      return _getMapData(this, key).has(key);
    }

    var _mapCacheHas = mapCacheHas;

    /**
     * Sets the map `key` to `value`.
     *
     * @private
     * @name set
     * @memberOf MapCache
     * @param {string} key The key of the value to set.
     * @param {*} value The value to set.
     * @returns {Object} Returns the map cache instance.
     */
    function mapCacheSet(key, value) {
      var data = _getMapData(this, key),
          size = data.size;

      data.set(key, value);
      this.size += data.size == size ? 0 : 1;
      return this;
    }

    var _mapCacheSet = mapCacheSet;

    /**
     * Creates a map cache object to store key-value pairs.
     *
     * @private
     * @constructor
     * @param {Array} [entries] The key-value pairs to cache.
     */
    function MapCache(entries) {
      var index = -1,
          length = entries == null ? 0 : entries.length;

      this.clear();
      while (++index < length) {
        var entry = entries[index];
        this.set(entry[0], entry[1]);
      }
    }

    // Add methods to `MapCache`.
    MapCache.prototype.clear = _mapCacheClear;
    MapCache.prototype['delete'] = _mapCacheDelete;
    MapCache.prototype.get = _mapCacheGet;
    MapCache.prototype.has = _mapCacheHas;
    MapCache.prototype.set = _mapCacheSet;

    var _MapCache = MapCache;

    /** Used as the size to enable large array optimizations. */
    var LARGE_ARRAY_SIZE = 200;

    /**
     * Sets the stack `key` to `value`.
     *
     * @private
     * @name set
     * @memberOf Stack
     * @param {string} key The key of the value to set.
     * @param {*} value The value to set.
     * @returns {Object} Returns the stack cache instance.
     */
    function stackSet(key, value) {
      var data = this.__data__;
      if (data instanceof _ListCache) {
        var pairs = data.__data__;
        if (!_Map || (pairs.length < LARGE_ARRAY_SIZE - 1)) {
          pairs.push([key, value]);
          this.size = ++data.size;
          return this;
        }
        data = this.__data__ = new _MapCache(pairs);
      }
      data.set(key, value);
      this.size = data.size;
      return this;
    }

    var _stackSet = stackSet;

    /**
     * Creates a stack cache object to store key-value pairs.
     *
     * @private
     * @constructor
     * @param {Array} [entries] The key-value pairs to cache.
     */
    function Stack(entries) {
      var data = this.__data__ = new _ListCache(entries);
      this.size = data.size;
    }

    // Add methods to `Stack`.
    Stack.prototype.clear = _stackClear;
    Stack.prototype['delete'] = _stackDelete;
    Stack.prototype.get = _stackGet;
    Stack.prototype.has = _stackHas;
    Stack.prototype.set = _stackSet;

    var _Stack = Stack;

    /**
     * A specialized version of `_.forEach` for arrays without support for
     * iteratee shorthands.
     *
     * @private
     * @param {Array} [array] The array to iterate over.
     * @param {Function} iteratee The function invoked per iteration.
     * @returns {Array} Returns `array`.
     */
    function arrayEach(array, iteratee) {
      var index = -1,
          length = array == null ? 0 : array.length;

      while (++index < length) {
        if (iteratee(array[index], index, array) === false) {
          break;
        }
      }
      return array;
    }

    var _arrayEach = arrayEach;

    var defineProperty = (function() {
      try {
        var func = _getNative(Object, 'defineProperty');
        func({}, '', {});
        return func;
      } catch (e) {}
    }());

    var _defineProperty = defineProperty;

    /**
     * The base implementation of `assignValue` and `assignMergeValue` without
     * value checks.
     *
     * @private
     * @param {Object} object The object to modify.
     * @param {string} key The key of the property to assign.
     * @param {*} value The value to assign.
     */
    function baseAssignValue(object, key, value) {
      if (key == '__proto__' && _defineProperty) {
        _defineProperty(object, key, {
          'configurable': true,
          'enumerable': true,
          'value': value,
          'writable': true
        });
      } else {
        object[key] = value;
      }
    }

    var _baseAssignValue = baseAssignValue;

    /** Used for built-in method references. */
    var objectProto$5 = Object.prototype;

    /** Used to check objects for own properties. */
    var hasOwnProperty$4 = objectProto$5.hasOwnProperty;

    /**
     * Assigns `value` to `key` of `object` if the existing value is not equivalent
     * using [`SameValueZero`](http://ecma-international.org/ecma-262/7.0/#sec-samevaluezero)
     * for equality comparisons.
     *
     * @private
     * @param {Object} object The object to modify.
     * @param {string} key The key of the property to assign.
     * @param {*} value The value to assign.
     */
    function assignValue(object, key, value) {
      var objValue = object[key];
      if (!(hasOwnProperty$4.call(object, key) && eq_1(objValue, value)) ||
          (value === undefined && !(key in object))) {
        _baseAssignValue(object, key, value);
      }
    }

    var _assignValue = assignValue;

    /**
     * Copies properties of `source` to `object`.
     *
     * @private
     * @param {Object} source The object to copy properties from.
     * @param {Array} props The property identifiers to copy.
     * @param {Object} [object={}] The object to copy properties to.
     * @param {Function} [customizer] The function to customize copied values.
     * @returns {Object} Returns `object`.
     */
    function copyObject(source, props, object, customizer) {
      var isNew = !object;
      object || (object = {});

      var index = -1,
          length = props.length;

      while (++index < length) {
        var key = props[index];

        var newValue = customizer
          ? customizer(object[key], source[key], key, object, source)
          : undefined;

        if (newValue === undefined) {
          newValue = source[key];
        }
        if (isNew) {
          _baseAssignValue(object, key, newValue);
        } else {
          _assignValue(object, key, newValue);
        }
      }
      return object;
    }

    var _copyObject = copyObject;

    /**
     * The base implementation of `_.times` without support for iteratee shorthands
     * or max array length checks.
     *
     * @private
     * @param {number} n The number of times to invoke `iteratee`.
     * @param {Function} iteratee The function invoked per iteration.
     * @returns {Array} Returns the array of results.
     */
    function baseTimes(n, iteratee) {
      var index = -1,
          result = Array(n);

      while (++index < n) {
        result[index] = iteratee(index);
      }
      return result;
    }

    var _baseTimes = baseTimes;

    /**
     * Checks if `value` is object-like. A value is object-like if it's not `null`
     * and has a `typeof` result of "object".
     *
     * @static
     * @memberOf _
     * @since 4.0.0
     * @category Lang
     * @param {*} value The value to check.
     * @returns {boolean} Returns `true` if `value` is object-like, else `false`.
     * @example
     *
     * _.isObjectLike({});
     * // => true
     *
     * _.isObjectLike([1, 2, 3]);
     * // => true
     *
     * _.isObjectLike(_.noop);
     * // => false
     *
     * _.isObjectLike(null);
     * // => false
     */
    function isObjectLike(value) {
      return value != null && typeof value == 'object';
    }

    var isObjectLike_1 = isObjectLike;

    /** `Object#toString` result references. */
    var argsTag = '[object Arguments]';

    /**
     * The base implementation of `_.isArguments`.
     *
     * @private
     * @param {*} value The value to check.
     * @returns {boolean} Returns `true` if `value` is an `arguments` object,
     */
    function baseIsArguments(value) {
      return isObjectLike_1(value) && _baseGetTag(value) == argsTag;
    }

    var _baseIsArguments = baseIsArguments;

    /** Used for built-in method references. */
    var objectProto$6 = Object.prototype;

    /** Used to check objects for own properties. */
    var hasOwnProperty$5 = objectProto$6.hasOwnProperty;

    /** Built-in value references. */
    var propertyIsEnumerable = objectProto$6.propertyIsEnumerable;

    /**
     * Checks if `value` is likely an `arguments` object.
     *
     * @static
     * @memberOf _
     * @since 0.1.0
     * @category Lang
     * @param {*} value The value to check.
     * @returns {boolean} Returns `true` if `value` is an `arguments` object,
     *  else `false`.
     * @example
     *
     * _.isArguments(function() { return arguments; }());
     * // => true
     *
     * _.isArguments([1, 2, 3]);
     * // => false
     */
    var isArguments = _baseIsArguments(function() { return arguments; }()) ? _baseIsArguments : function(value) {
      return isObjectLike_1(value) && hasOwnProperty$5.call(value, 'callee') &&
        !propertyIsEnumerable.call(value, 'callee');
    };

    var isArguments_1 = isArguments;

    /**
     * Checks if `value` is classified as an `Array` object.
     *
     * @static
     * @memberOf _
     * @since 0.1.0
     * @category Lang
     * @param {*} value The value to check.
     * @returns {boolean} Returns `true` if `value` is an array, else `false`.
     * @example
     *
     * _.isArray([1, 2, 3]);
     * // => true
     *
     * _.isArray(document.body.children);
     * // => false
     *
     * _.isArray('abc');
     * // => false
     *
     * _.isArray(_.noop);
     * // => false
     */
    var isArray = Array.isArray;

    var isArray_1 = isArray;

    /**
     * This method returns `false`.
     *
     * @static
     * @memberOf _
     * @since 4.13.0
     * @category Util
     * @returns {boolean} Returns `false`.
     * @example
     *
     * _.times(2, _.stubFalse);
     * // => [false, false]
     */
    function stubFalse() {
      return false;
    }

    var stubFalse_1 = stubFalse;

    var isBuffer_1 = createCommonjsModule(function (module, exports) {
    /** Detect free variable `exports`. */
    var freeExports =  exports && !exports.nodeType && exports;

    /** Detect free variable `module`. */
    var freeModule = freeExports && 'object' == 'object' && module && !module.nodeType && module;

    /** Detect the popular CommonJS extension `module.exports`. */
    var moduleExports = freeModule && freeModule.exports === freeExports;

    /** Built-in value references. */
    var Buffer = moduleExports ? _root.Buffer : undefined;

    /* Built-in method references for those with the same name as other `lodash` methods. */
    var nativeIsBuffer = Buffer ? Buffer.isBuffer : undefined;

    /**
     * Checks if `value` is a buffer.
     *
     * @static
     * @memberOf _
     * @since 4.3.0
     * @category Lang
     * @param {*} value The value to check.
     * @returns {boolean} Returns `true` if `value` is a buffer, else `false`.
     * @example
     *
     * _.isBuffer(new Buffer(2));
     * // => true
     *
     * _.isBuffer(new Uint8Array(2));
     * // => false
     */
    var isBuffer = nativeIsBuffer || stubFalse_1;

    module.exports = isBuffer;
    });

    /** Used as references for various `Number` constants. */
    var MAX_SAFE_INTEGER = 9007199254740991;

    /** Used to detect unsigned integer values. */
    var reIsUint = /^(?:0|[1-9]\d*)$/;

    /**
     * Checks if `value` is a valid array-like index.
     *
     * @private
     * @param {*} value The value to check.
     * @param {number} [length=MAX_SAFE_INTEGER] The upper bounds of a valid index.
     * @returns {boolean} Returns `true` if `value` is a valid index, else `false`.
     */
    function isIndex(value, length) {
      var type = typeof value;
      length = length == null ? MAX_SAFE_INTEGER : length;

      return !!length &&
        (type == 'number' ||
          (type != 'symbol' && reIsUint.test(value))) &&
            (value > -1 && value % 1 == 0 && value < length);
    }

    var _isIndex = isIndex;

    /** Used as references for various `Number` constants. */
    var MAX_SAFE_INTEGER$1 = 9007199254740991;

    /**
     * Checks if `value` is a valid array-like length.
     *
     * **Note:** This method is loosely based on
     * [`ToLength`](http://ecma-international.org/ecma-262/7.0/#sec-tolength).
     *
     * @static
     * @memberOf _
     * @since 4.0.0
     * @category Lang
     * @param {*} value The value to check.
     * @returns {boolean} Returns `true` if `value` is a valid length, else `false`.
     * @example
     *
     * _.isLength(3);
     * // => true
     *
     * _.isLength(Number.MIN_VALUE);
     * // => false
     *
     * _.isLength(Infinity);
     * // => false
     *
     * _.isLength('3');
     * // => false
     */
    function isLength(value) {
      return typeof value == 'number' &&
        value > -1 && value % 1 == 0 && value <= MAX_SAFE_INTEGER$1;
    }

    var isLength_1 = isLength;

    /** `Object#toString` result references. */
    var argsTag$1 = '[object Arguments]',
        arrayTag = '[object Array]',
        boolTag = '[object Boolean]',
        dateTag = '[object Date]',
        errorTag = '[object Error]',
        funcTag$1 = '[object Function]',
        mapTag = '[object Map]',
        numberTag = '[object Number]',
        objectTag = '[object Object]',
        regexpTag = '[object RegExp]',
        setTag = '[object Set]',
        stringTag = '[object String]',
        weakMapTag = '[object WeakMap]';

    var arrayBufferTag = '[object ArrayBuffer]',
        dataViewTag = '[object DataView]',
        float32Tag = '[object Float32Array]',
        float64Tag = '[object Float64Array]',
        int8Tag = '[object Int8Array]',
        int16Tag = '[object Int16Array]',
        int32Tag = '[object Int32Array]',
        uint8Tag = '[object Uint8Array]',
        uint8ClampedTag = '[object Uint8ClampedArray]',
        uint16Tag = '[object Uint16Array]',
        uint32Tag = '[object Uint32Array]';

    /** Used to identify `toStringTag` values of typed arrays. */
    var typedArrayTags = {};
    typedArrayTags[float32Tag] = typedArrayTags[float64Tag] =
    typedArrayTags[int8Tag] = typedArrayTags[int16Tag] =
    typedArrayTags[int32Tag] = typedArrayTags[uint8Tag] =
    typedArrayTags[uint8ClampedTag] = typedArrayTags[uint16Tag] =
    typedArrayTags[uint32Tag] = true;
    typedArrayTags[argsTag$1] = typedArrayTags[arrayTag] =
    typedArrayTags[arrayBufferTag] = typedArrayTags[boolTag] =
    typedArrayTags[dataViewTag] = typedArrayTags[dateTag] =
    typedArrayTags[errorTag] = typedArrayTags[funcTag$1] =
    typedArrayTags[mapTag] = typedArrayTags[numberTag] =
    typedArrayTags[objectTag] = typedArrayTags[regexpTag] =
    typedArrayTags[setTag] = typedArrayTags[stringTag] =
    typedArrayTags[weakMapTag] = false;

    /**
     * The base implementation of `_.isTypedArray` without Node.js optimizations.
     *
     * @private
     * @param {*} value The value to check.
     * @returns {boolean} Returns `true` if `value` is a typed array, else `false`.
     */
    function baseIsTypedArray(value) {
      return isObjectLike_1(value) &&
        isLength_1(value.length) && !!typedArrayTags[_baseGetTag(value)];
    }

    var _baseIsTypedArray = baseIsTypedArray;

    /**
     * The base implementation of `_.unary` without support for storing metadata.
     *
     * @private
     * @param {Function} func The function to cap arguments for.
     * @returns {Function} Returns the new capped function.
     */
    function baseUnary(func) {
      return function(value) {
        return func(value);
      };
    }

    var _baseUnary = baseUnary;

    var _nodeUtil = createCommonjsModule(function (module, exports) {
    /** Detect free variable `exports`. */
    var freeExports =  exports && !exports.nodeType && exports;

    /** Detect free variable `module`. */
    var freeModule = freeExports && 'object' == 'object' && module && !module.nodeType && module;

    /** Detect the popular CommonJS extension `module.exports`. */
    var moduleExports = freeModule && freeModule.exports === freeExports;

    /** Detect free variable `process` from Node.js. */
    var freeProcess = moduleExports && _freeGlobal.process;

    /** Used to access faster Node.js helpers. */
    var nodeUtil = (function() {
      try {
        // Use `util.types` for Node.js 10+.
        var types = freeModule && freeModule.require && freeModule.require('util').types;

        if (types) {
          return types;
        }

        // Legacy `process.binding('util')` for Node.js < 10.
        return freeProcess && freeProcess.binding && freeProcess.binding('util');
      } catch (e) {}
    }());

    module.exports = nodeUtil;
    });

    /* Node.js helper references. */
    var nodeIsTypedArray = _nodeUtil && _nodeUtil.isTypedArray;

    /**
     * Checks if `value` is classified as a typed array.
     *
     * @static
     * @memberOf _
     * @since 3.0.0
     * @category Lang
     * @param {*} value The value to check.
     * @returns {boolean} Returns `true` if `value` is a typed array, else `false`.
     * @example
     *
     * _.isTypedArray(new Uint8Array);
     * // => true
     *
     * _.isTypedArray([]);
     * // => false
     */
    var isTypedArray = nodeIsTypedArray ? _baseUnary(nodeIsTypedArray) : _baseIsTypedArray;

    var isTypedArray_1 = isTypedArray;

    /** Used for built-in method references. */
    var objectProto$7 = Object.prototype;

    /** Used to check objects for own properties. */
    var hasOwnProperty$6 = objectProto$7.hasOwnProperty;

    /**
     * Creates an array of the enumerable property names of the array-like `value`.
     *
     * @private
     * @param {*} value The value to query.
     * @param {boolean} inherited Specify returning inherited property names.
     * @returns {Array} Returns the array of property names.
     */
    function arrayLikeKeys(value, inherited) {
      var isArr = isArray_1(value),
          isArg = !isArr && isArguments_1(value),
          isBuff = !isArr && !isArg && isBuffer_1(value),
          isType = !isArr && !isArg && !isBuff && isTypedArray_1(value),
          skipIndexes = isArr || isArg || isBuff || isType,
          result = skipIndexes ? _baseTimes(value.length, String) : [],
          length = result.length;

      for (var key in value) {
        if ((inherited || hasOwnProperty$6.call(value, key)) &&
            !(skipIndexes && (
               // Safari 9 has enumerable `arguments.length` in strict mode.
               key == 'length' ||
               // Node.js 0.10 has enumerable non-index properties on buffers.
               (isBuff && (key == 'offset' || key == 'parent')) ||
               // PhantomJS 2 has enumerable non-index properties on typed arrays.
               (isType && (key == 'buffer' || key == 'byteLength' || key == 'byteOffset')) ||
               // Skip index properties.
               _isIndex(key, length)
            ))) {
          result.push(key);
        }
      }
      return result;
    }

    var _arrayLikeKeys = arrayLikeKeys;

    /** Used for built-in method references. */
    var objectProto$8 = Object.prototype;

    /**
     * Checks if `value` is likely a prototype object.
     *
     * @private
     * @param {*} value The value to check.
     * @returns {boolean} Returns `true` if `value` is a prototype, else `false`.
     */
    function isPrototype(value) {
      var Ctor = value && value.constructor,
          proto = (typeof Ctor == 'function' && Ctor.prototype) || objectProto$8;

      return value === proto;
    }

    var _isPrototype = isPrototype;

    /**
     * Creates a unary function that invokes `func` with its argument transformed.
     *
     * @private
     * @param {Function} func The function to wrap.
     * @param {Function} transform The argument transform.
     * @returns {Function} Returns the new function.
     */
    function overArg(func, transform) {
      return function(arg) {
        return func(transform(arg));
      };
    }

    var _overArg = overArg;

    /* Built-in method references for those with the same name as other `lodash` methods. */
    var nativeKeys = _overArg(Object.keys, Object);

    var _nativeKeys = nativeKeys;

    /** Used for built-in method references. */
    var objectProto$9 = Object.prototype;

    /** Used to check objects for own properties. */
    var hasOwnProperty$7 = objectProto$9.hasOwnProperty;

    /**
     * The base implementation of `_.keys` which doesn't treat sparse arrays as dense.
     *
     * @private
     * @param {Object} object The object to query.
     * @returns {Array} Returns the array of property names.
     */
    function baseKeys(object) {
      if (!_isPrototype(object)) {
        return _nativeKeys(object);
      }
      var result = [];
      for (var key in Object(object)) {
        if (hasOwnProperty$7.call(object, key) && key != 'constructor') {
          result.push(key);
        }
      }
      return result;
    }

    var _baseKeys = baseKeys;

    /**
     * Checks if `value` is array-like. A value is considered array-like if it's
     * not a function and has a `value.length` that's an integer greater than or
     * equal to `0` and less than or equal to `Number.MAX_SAFE_INTEGER`.
     *
     * @static
     * @memberOf _
     * @since 4.0.0
     * @category Lang
     * @param {*} value The value to check.
     * @returns {boolean} Returns `true` if `value` is array-like, else `false`.
     * @example
     *
     * _.isArrayLike([1, 2, 3]);
     * // => true
     *
     * _.isArrayLike(document.body.children);
     * // => true
     *
     * _.isArrayLike('abc');
     * // => true
     *
     * _.isArrayLike(_.noop);
     * // => false
     */
    function isArrayLike(value) {
      return value != null && isLength_1(value.length) && !isFunction_1(value);
    }

    var isArrayLike_1 = isArrayLike;

    /**
     * Creates an array of the own enumerable property names of `object`.
     *
     * **Note:** Non-object values are coerced to objects. See the
     * [ES spec](http://ecma-international.org/ecma-262/7.0/#sec-object.keys)
     * for more details.
     *
     * @static
     * @since 0.1.0
     * @memberOf _
     * @category Object
     * @param {Object} object The object to query.
     * @returns {Array} Returns the array of property names.
     * @example
     *
     * function Foo() {
     *   this.a = 1;
     *   this.b = 2;
     * }
     *
     * Foo.prototype.c = 3;
     *
     * _.keys(new Foo);
     * // => ['a', 'b'] (iteration order is not guaranteed)
     *
     * _.keys('hi');
     * // => ['0', '1']
     */
    function keys$2(object) {
      return isArrayLike_1(object) ? _arrayLikeKeys(object) : _baseKeys(object);
    }

    var keys_1 = keys$2;

    /**
     * The base implementation of `_.assign` without support for multiple sources
     * or `customizer` functions.
     *
     * @private
     * @param {Object} object The destination object.
     * @param {Object} source The source object.
     * @returns {Object} Returns `object`.
     */
    function baseAssign(object, source) {
      return object && _copyObject(source, keys_1(source), object);
    }

    var _baseAssign = baseAssign;

    /**
     * This function is like
     * [`Object.keys`](http://ecma-international.org/ecma-262/7.0/#sec-object.keys)
     * except that it includes inherited enumerable properties.
     *
     * @private
     * @param {Object} object The object to query.
     * @returns {Array} Returns the array of property names.
     */
    function nativeKeysIn(object) {
      var result = [];
      if (object != null) {
        for (var key in Object(object)) {
          result.push(key);
        }
      }
      return result;
    }

    var _nativeKeysIn = nativeKeysIn;

    /** Used for built-in method references. */
    var objectProto$a = Object.prototype;

    /** Used to check objects for own properties. */
    var hasOwnProperty$8 = objectProto$a.hasOwnProperty;

    /**
     * The base implementation of `_.keysIn` which doesn't treat sparse arrays as dense.
     *
     * @private
     * @param {Object} object The object to query.
     * @returns {Array} Returns the array of property names.
     */
    function baseKeysIn(object) {
      if (!isObject_1(object)) {
        return _nativeKeysIn(object);
      }
      var isProto = _isPrototype(object),
          result = [];

      for (var key in object) {
        if (!(key == 'constructor' && (isProto || !hasOwnProperty$8.call(object, key)))) {
          result.push(key);
        }
      }
      return result;
    }

    var _baseKeysIn = baseKeysIn;

    /**
     * Creates an array of the own and inherited enumerable property names of `object`.
     *
     * **Note:** Non-object values are coerced to objects.
     *
     * @static
     * @memberOf _
     * @since 3.0.0
     * @category Object
     * @param {Object} object The object to query.
     * @returns {Array} Returns the array of property names.
     * @example
     *
     * function Foo() {
     *   this.a = 1;
     *   this.b = 2;
     * }
     *
     * Foo.prototype.c = 3;
     *
     * _.keysIn(new Foo);
     * // => ['a', 'b', 'c'] (iteration order is not guaranteed)
     */
    function keysIn$1(object) {
      return isArrayLike_1(object) ? _arrayLikeKeys(object, true) : _baseKeysIn(object);
    }

    var keysIn_1 = keysIn$1;

    /**
     * The base implementation of `_.assignIn` without support for multiple sources
     * or `customizer` functions.
     *
     * @private
     * @param {Object} object The destination object.
     * @param {Object} source The source object.
     * @returns {Object} Returns `object`.
     */
    function baseAssignIn(object, source) {
      return object && _copyObject(source, keysIn_1(source), object);
    }

    var _baseAssignIn = baseAssignIn;

    var _cloneBuffer = createCommonjsModule(function (module, exports) {
    /** Detect free variable `exports`. */
    var freeExports =  exports && !exports.nodeType && exports;

    /** Detect free variable `module`. */
    var freeModule = freeExports && 'object' == 'object' && module && !module.nodeType && module;

    /** Detect the popular CommonJS extension `module.exports`. */
    var moduleExports = freeModule && freeModule.exports === freeExports;

    /** Built-in value references. */
    var Buffer = moduleExports ? _root.Buffer : undefined,
        allocUnsafe = Buffer ? Buffer.allocUnsafe : undefined;

    /**
     * Creates a clone of  `buffer`.
     *
     * @private
     * @param {Buffer} buffer The buffer to clone.
     * @param {boolean} [isDeep] Specify a deep clone.
     * @returns {Buffer} Returns the cloned buffer.
     */
    function cloneBuffer(buffer, isDeep) {
      if (isDeep) {
        return buffer.slice();
      }
      var length = buffer.length,
          result = allocUnsafe ? allocUnsafe(length) : new buffer.constructor(length);

      buffer.copy(result);
      return result;
    }

    module.exports = cloneBuffer;
    });

    /**
     * Copies the values of `source` to `array`.
     *
     * @private
     * @param {Array} source The array to copy values from.
     * @param {Array} [array=[]] The array to copy values to.
     * @returns {Array} Returns `array`.
     */
    function copyArray(source, array) {
      var index = -1,
          length = source.length;

      array || (array = Array(length));
      while (++index < length) {
        array[index] = source[index];
      }
      return array;
    }

    var _copyArray = copyArray;

    /**
     * A specialized version of `_.filter` for arrays without support for
     * iteratee shorthands.
     *
     * @private
     * @param {Array} [array] The array to iterate over.
     * @param {Function} predicate The function invoked per iteration.
     * @returns {Array} Returns the new filtered array.
     */
    function arrayFilter(array, predicate) {
      var index = -1,
          length = array == null ? 0 : array.length,
          resIndex = 0,
          result = [];

      while (++index < length) {
        var value = array[index];
        if (predicate(value, index, array)) {
          result[resIndex++] = value;
        }
      }
      return result;
    }

    var _arrayFilter = arrayFilter;

    /**
     * This method returns a new empty array.
     *
     * @static
     * @memberOf _
     * @since 4.13.0
     * @category Util
     * @returns {Array} Returns the new empty array.
     * @example
     *
     * var arrays = _.times(2, _.stubArray);
     *
     * console.log(arrays);
     * // => [[], []]
     *
     * console.log(arrays[0] === arrays[1]);
     * // => false
     */
    function stubArray() {
      return [];
    }

    var stubArray_1 = stubArray;

    /** Used for built-in method references. */
    var objectProto$b = Object.prototype;

    /** Built-in value references. */
    var propertyIsEnumerable$1 = objectProto$b.propertyIsEnumerable;

    /* Built-in method references for those with the same name as other `lodash` methods. */
    var nativeGetSymbols = Object.getOwnPropertySymbols;

    /**
     * Creates an array of the own enumerable symbols of `object`.
     *
     * @private
     * @param {Object} object The object to query.
     * @returns {Array} Returns the array of symbols.
     */
    var getSymbols = !nativeGetSymbols ? stubArray_1 : function(object) {
      if (object == null) {
        return [];
      }
      object = Object(object);
      return _arrayFilter(nativeGetSymbols(object), function(symbol) {
        return propertyIsEnumerable$1.call(object, symbol);
      });
    };

    var _getSymbols = getSymbols;

    /**
     * Copies own symbols of `source` to `object`.
     *
     * @private
     * @param {Object} source The object to copy symbols from.
     * @param {Object} [object={}] The object to copy symbols to.
     * @returns {Object} Returns `object`.
     */
    function copySymbols(source, object) {
      return _copyObject(source, _getSymbols(source), object);
    }

    var _copySymbols = copySymbols;

    /**
     * Appends the elements of `values` to `array`.
     *
     * @private
     * @param {Array} array The array to modify.
     * @param {Array} values The values to append.
     * @returns {Array} Returns `array`.
     */
    function arrayPush(array, values) {
      var index = -1,
          length = values.length,
          offset = array.length;

      while (++index < length) {
        array[offset + index] = values[index];
      }
      return array;
    }

    var _arrayPush = arrayPush;

    /** Built-in value references. */
    var getPrototype = _overArg(Object.getPrototypeOf, Object);

    var _getPrototype = getPrototype;

    /* Built-in method references for those with the same name as other `lodash` methods. */
    var nativeGetSymbols$1 = Object.getOwnPropertySymbols;

    /**
     * Creates an array of the own and inherited enumerable symbols of `object`.
     *
     * @private
     * @param {Object} object The object to query.
     * @returns {Array} Returns the array of symbols.
     */
    var getSymbolsIn = !nativeGetSymbols$1 ? stubArray_1 : function(object) {
      var result = [];
      while (object) {
        _arrayPush(result, _getSymbols(object));
        object = _getPrototype(object);
      }
      return result;
    };

    var _getSymbolsIn = getSymbolsIn;

    /**
     * Copies own and inherited symbols of `source` to `object`.
     *
     * @private
     * @param {Object} source The object to copy symbols from.
     * @param {Object} [object={}] The object to copy symbols to.
     * @returns {Object} Returns `object`.
     */
    function copySymbolsIn(source, object) {
      return _copyObject(source, _getSymbolsIn(source), object);
    }

    var _copySymbolsIn = copySymbolsIn;

    /**
     * The base implementation of `getAllKeys` and `getAllKeysIn` which uses
     * `keysFunc` and `symbolsFunc` to get the enumerable property names and
     * symbols of `object`.
     *
     * @private
     * @param {Object} object The object to query.
     * @param {Function} keysFunc The function to get the keys of `object`.
     * @param {Function} symbolsFunc The function to get the symbols of `object`.
     * @returns {Array} Returns the array of property names and symbols.
     */
    function baseGetAllKeys(object, keysFunc, symbolsFunc) {
      var result = keysFunc(object);
      return isArray_1(object) ? result : _arrayPush(result, symbolsFunc(object));
    }

    var _baseGetAllKeys = baseGetAllKeys;

    /**
     * Creates an array of own enumerable property names and symbols of `object`.
     *
     * @private
     * @param {Object} object The object to query.
     * @returns {Array} Returns the array of property names and symbols.
     */
    function getAllKeys(object) {
      return _baseGetAllKeys(object, keys_1, _getSymbols);
    }

    var _getAllKeys = getAllKeys;

    /**
     * Creates an array of own and inherited enumerable property names and
     * symbols of `object`.
     *
     * @private
     * @param {Object} object The object to query.
     * @returns {Array} Returns the array of property names and symbols.
     */
    function getAllKeysIn(object) {
      return _baseGetAllKeys(object, keysIn_1, _getSymbolsIn);
    }

    var _getAllKeysIn = getAllKeysIn;

    /* Built-in method references that are verified to be native. */
    var DataView$1 = _getNative(_root, 'DataView');

    var _DataView = DataView$1;

    /* Built-in method references that are verified to be native. */
    var Promise$1 = _getNative(_root, 'Promise');

    var _Promise = Promise$1;

    /* Built-in method references that are verified to be native. */
    var Set$5 = _getNative(_root, 'Set');

    var _Set = Set$5;

    /* Built-in method references that are verified to be native. */
    var WeakMap$1 = _getNative(_root, 'WeakMap');

    var _WeakMap = WeakMap$1;

    /** `Object#toString` result references. */
    var mapTag$1 = '[object Map]',
        objectTag$1 = '[object Object]',
        promiseTag = '[object Promise]',
        setTag$1 = '[object Set]',
        weakMapTag$1 = '[object WeakMap]';

    var dataViewTag$1 = '[object DataView]';

    /** Used to detect maps, sets, and weakmaps. */
    var dataViewCtorString = _toSource(_DataView),
        mapCtorString = _toSource(_Map),
        promiseCtorString = _toSource(_Promise),
        setCtorString = _toSource(_Set),
        weakMapCtorString = _toSource(_WeakMap);

    /**
     * Gets the `toStringTag` of `value`.
     *
     * @private
     * @param {*} value The value to query.
     * @returns {string} Returns the `toStringTag`.
     */
    var getTag = _baseGetTag;

    // Fallback for data views, maps, sets, and weak maps in IE 11 and promises in Node.js < 6.
    if ((_DataView && getTag(new _DataView(new ArrayBuffer(1))) != dataViewTag$1) ||
        (_Map && getTag(new _Map) != mapTag$1) ||
        (_Promise && getTag(_Promise.resolve()) != promiseTag) ||
        (_Set && getTag(new _Set) != setTag$1) ||
        (_WeakMap && getTag(new _WeakMap) != weakMapTag$1)) {
      getTag = function(value) {
        var result = _baseGetTag(value),
            Ctor = result == objectTag$1 ? value.constructor : undefined,
            ctorString = Ctor ? _toSource(Ctor) : '';

        if (ctorString) {
          switch (ctorString) {
            case dataViewCtorString: return dataViewTag$1;
            case mapCtorString: return mapTag$1;
            case promiseCtorString: return promiseTag;
            case setCtorString: return setTag$1;
            case weakMapCtorString: return weakMapTag$1;
          }
        }
        return result;
      };
    }

    var _getTag = getTag;

    /** Used for built-in method references. */
    var objectProto$c = Object.prototype;

    /** Used to check objects for own properties. */
    var hasOwnProperty$9 = objectProto$c.hasOwnProperty;

    /**
     * Initializes an array clone.
     *
     * @private
     * @param {Array} array The array to clone.
     * @returns {Array} Returns the initialized clone.
     */
    function initCloneArray(array) {
      var length = array.length,
          result = new array.constructor(length);

      // Add properties assigned by `RegExp#exec`.
      if (length && typeof array[0] == 'string' && hasOwnProperty$9.call(array, 'index')) {
        result.index = array.index;
        result.input = array.input;
      }
      return result;
    }

    var _initCloneArray = initCloneArray;

    /** Built-in value references. */
    var Uint8Array$1 = _root.Uint8Array;

    var _Uint8Array = Uint8Array$1;

    /**
     * Creates a clone of `arrayBuffer`.
     *
     * @private
     * @param {ArrayBuffer} arrayBuffer The array buffer to clone.
     * @returns {ArrayBuffer} Returns the cloned array buffer.
     */
    function cloneArrayBuffer(arrayBuffer) {
      var result = new arrayBuffer.constructor(arrayBuffer.byteLength);
      new _Uint8Array(result).set(new _Uint8Array(arrayBuffer));
      return result;
    }

    var _cloneArrayBuffer = cloneArrayBuffer;

    /**
     * Creates a clone of `dataView`.
     *
     * @private
     * @param {Object} dataView The data view to clone.
     * @param {boolean} [isDeep] Specify a deep clone.
     * @returns {Object} Returns the cloned data view.
     */
    function cloneDataView(dataView, isDeep) {
      var buffer = isDeep ? _cloneArrayBuffer(dataView.buffer) : dataView.buffer;
      return new dataView.constructor(buffer, dataView.byteOffset, dataView.byteLength);
    }

    var _cloneDataView = cloneDataView;

    /** Used to match `RegExp` flags from their coerced string values. */
    var reFlags = /\w*$/;

    /**
     * Creates a clone of `regexp`.
     *
     * @private
     * @param {Object} regexp The regexp to clone.
     * @returns {Object} Returns the cloned regexp.
     */
    function cloneRegExp(regexp) {
      var result = new regexp.constructor(regexp.source, reFlags.exec(regexp));
      result.lastIndex = regexp.lastIndex;
      return result;
    }

    var _cloneRegExp = cloneRegExp;

    /** Used to convert symbols to primitives and strings. */
    var symbolProto = _Symbol ? _Symbol.prototype : undefined,
        symbolValueOf = symbolProto ? symbolProto.valueOf : undefined;

    /**
     * Creates a clone of the `symbol` object.
     *
     * @private
     * @param {Object} symbol The symbol object to clone.
     * @returns {Object} Returns the cloned symbol object.
     */
    function cloneSymbol(symbol) {
      return symbolValueOf ? Object(symbolValueOf.call(symbol)) : {};
    }

    var _cloneSymbol = cloneSymbol;

    /**
     * Creates a clone of `typedArray`.
     *
     * @private
     * @param {Object} typedArray The typed array to clone.
     * @param {boolean} [isDeep] Specify a deep clone.
     * @returns {Object} Returns the cloned typed array.
     */
    function cloneTypedArray(typedArray, isDeep) {
      var buffer = isDeep ? _cloneArrayBuffer(typedArray.buffer) : typedArray.buffer;
      return new typedArray.constructor(buffer, typedArray.byteOffset, typedArray.length);
    }

    var _cloneTypedArray = cloneTypedArray;

    /** `Object#toString` result references. */
    var boolTag$1 = '[object Boolean]',
        dateTag$1 = '[object Date]',
        mapTag$2 = '[object Map]',
        numberTag$1 = '[object Number]',
        regexpTag$1 = '[object RegExp]',
        setTag$2 = '[object Set]',
        stringTag$1 = '[object String]',
        symbolTag = '[object Symbol]';

    var arrayBufferTag$1 = '[object ArrayBuffer]',
        dataViewTag$2 = '[object DataView]',
        float32Tag$1 = '[object Float32Array]',
        float64Tag$1 = '[object Float64Array]',
        int8Tag$1 = '[object Int8Array]',
        int16Tag$1 = '[object Int16Array]',
        int32Tag$1 = '[object Int32Array]',
        uint8Tag$1 = '[object Uint8Array]',
        uint8ClampedTag$1 = '[object Uint8ClampedArray]',
        uint16Tag$1 = '[object Uint16Array]',
        uint32Tag$1 = '[object Uint32Array]';

    /**
     * Initializes an object clone based on its `toStringTag`.
     *
     * **Note:** This function only supports cloning values with tags of
     * `Boolean`, `Date`, `Error`, `Map`, `Number`, `RegExp`, `Set`, or `String`.
     *
     * @private
     * @param {Object} object The object to clone.
     * @param {string} tag The `toStringTag` of the object to clone.
     * @param {boolean} [isDeep] Specify a deep clone.
     * @returns {Object} Returns the initialized clone.
     */
    function initCloneByTag(object, tag, isDeep) {
      var Ctor = object.constructor;
      switch (tag) {
        case arrayBufferTag$1:
          return _cloneArrayBuffer(object);

        case boolTag$1:
        case dateTag$1:
          return new Ctor(+object);

        case dataViewTag$2:
          return _cloneDataView(object, isDeep);

        case float32Tag$1: case float64Tag$1:
        case int8Tag$1: case int16Tag$1: case int32Tag$1:
        case uint8Tag$1: case uint8ClampedTag$1: case uint16Tag$1: case uint32Tag$1:
          return _cloneTypedArray(object, isDeep);

        case mapTag$2:
          return new Ctor;

        case numberTag$1:
        case stringTag$1:
          return new Ctor(object);

        case regexpTag$1:
          return _cloneRegExp(object);

        case setTag$2:
          return new Ctor;

        case symbolTag:
          return _cloneSymbol(object);
      }
    }

    var _initCloneByTag = initCloneByTag;

    /** Built-in value references. */
    var objectCreate = Object.create;

    /**
     * The base implementation of `_.create` without support for assigning
     * properties to the created object.
     *
     * @private
     * @param {Object} proto The object to inherit from.
     * @returns {Object} Returns the new object.
     */
    var baseCreate = (function() {
      function object() {}
      return function(proto) {
        if (!isObject_1(proto)) {
          return {};
        }
        if (objectCreate) {
          return objectCreate(proto);
        }
        object.prototype = proto;
        var result = new object;
        object.prototype = undefined;
        return result;
      };
    }());

    var _baseCreate = baseCreate;

    /**
     * Initializes an object clone.
     *
     * @private
     * @param {Object} object The object to clone.
     * @returns {Object} Returns the initialized clone.
     */
    function initCloneObject(object) {
      return (typeof object.constructor == 'function' && !_isPrototype(object))
        ? _baseCreate(_getPrototype(object))
        : {};
    }

    var _initCloneObject = initCloneObject;

    /** `Object#toString` result references. */
    var mapTag$3 = '[object Map]';

    /**
     * The base implementation of `_.isMap` without Node.js optimizations.
     *
     * @private
     * @param {*} value The value to check.
     * @returns {boolean} Returns `true` if `value` is a map, else `false`.
     */
    function baseIsMap(value) {
      return isObjectLike_1(value) && _getTag(value) == mapTag$3;
    }

    var _baseIsMap = baseIsMap;

    /* Node.js helper references. */
    var nodeIsMap = _nodeUtil && _nodeUtil.isMap;

    /**
     * Checks if `value` is classified as a `Map` object.
     *
     * @static
     * @memberOf _
     * @since 4.3.0
     * @category Lang
     * @param {*} value The value to check.
     * @returns {boolean} Returns `true` if `value` is a map, else `false`.
     * @example
     *
     * _.isMap(new Map);
     * // => true
     *
     * _.isMap(new WeakMap);
     * // => false
     */
    var isMap = nodeIsMap ? _baseUnary(nodeIsMap) : _baseIsMap;

    var isMap_1 = isMap;

    /** `Object#toString` result references. */
    var setTag$3 = '[object Set]';

    /**
     * The base implementation of `_.isSet` without Node.js optimizations.
     *
     * @private
     * @param {*} value The value to check.
     * @returns {boolean} Returns `true` if `value` is a set, else `false`.
     */
    function baseIsSet(value) {
      return isObjectLike_1(value) && _getTag(value) == setTag$3;
    }

    var _baseIsSet = baseIsSet;

    /* Node.js helper references. */
    var nodeIsSet = _nodeUtil && _nodeUtil.isSet;

    /**
     * Checks if `value` is classified as a `Set` object.
     *
     * @static
     * @memberOf _
     * @since 4.3.0
     * @category Lang
     * @param {*} value The value to check.
     * @returns {boolean} Returns `true` if `value` is a set, else `false`.
     * @example
     *
     * _.isSet(new Set);
     * // => true
     *
     * _.isSet(new WeakSet);
     * // => false
     */
    var isSet = nodeIsSet ? _baseUnary(nodeIsSet) : _baseIsSet;

    var isSet_1 = isSet;

    /** Used to compose bitmasks for cloning. */
    var CLONE_DEEP_FLAG = 1,
        CLONE_FLAT_FLAG = 2,
        CLONE_SYMBOLS_FLAG = 4;

    /** `Object#toString` result references. */
    var argsTag$2 = '[object Arguments]',
        arrayTag$1 = '[object Array]',
        boolTag$2 = '[object Boolean]',
        dateTag$2 = '[object Date]',
        errorTag$1 = '[object Error]',
        funcTag$2 = '[object Function]',
        genTag$1 = '[object GeneratorFunction]',
        mapTag$4 = '[object Map]',
        numberTag$2 = '[object Number]',
        objectTag$2 = '[object Object]',
        regexpTag$2 = '[object RegExp]',
        setTag$4 = '[object Set]',
        stringTag$2 = '[object String]',
        symbolTag$1 = '[object Symbol]',
        weakMapTag$2 = '[object WeakMap]';

    var arrayBufferTag$2 = '[object ArrayBuffer]',
        dataViewTag$3 = '[object DataView]',
        float32Tag$2 = '[object Float32Array]',
        float64Tag$2 = '[object Float64Array]',
        int8Tag$2 = '[object Int8Array]',
        int16Tag$2 = '[object Int16Array]',
        int32Tag$2 = '[object Int32Array]',
        uint8Tag$2 = '[object Uint8Array]',
        uint8ClampedTag$2 = '[object Uint8ClampedArray]',
        uint16Tag$2 = '[object Uint16Array]',
        uint32Tag$2 = '[object Uint32Array]';

    /** Used to identify `toStringTag` values supported by `_.clone`. */
    var cloneableTags = {};
    cloneableTags[argsTag$2] = cloneableTags[arrayTag$1] =
    cloneableTags[arrayBufferTag$2] = cloneableTags[dataViewTag$3] =
    cloneableTags[boolTag$2] = cloneableTags[dateTag$2] =
    cloneableTags[float32Tag$2] = cloneableTags[float64Tag$2] =
    cloneableTags[int8Tag$2] = cloneableTags[int16Tag$2] =
    cloneableTags[int32Tag$2] = cloneableTags[mapTag$4] =
    cloneableTags[numberTag$2] = cloneableTags[objectTag$2] =
    cloneableTags[regexpTag$2] = cloneableTags[setTag$4] =
    cloneableTags[stringTag$2] = cloneableTags[symbolTag$1] =
    cloneableTags[uint8Tag$2] = cloneableTags[uint8ClampedTag$2] =
    cloneableTags[uint16Tag$2] = cloneableTags[uint32Tag$2] = true;
    cloneableTags[errorTag$1] = cloneableTags[funcTag$2] =
    cloneableTags[weakMapTag$2] = false;

    /**
     * The base implementation of `_.clone` and `_.cloneDeep` which tracks
     * traversed objects.
     *
     * @private
     * @param {*} value The value to clone.
     * @param {boolean} bitmask The bitmask flags.
     *  1 - Deep clone
     *  2 - Flatten inherited properties
     *  4 - Clone symbols
     * @param {Function} [customizer] The function to customize cloning.
     * @param {string} [key] The key of `value`.
     * @param {Object} [object] The parent object of `value`.
     * @param {Object} [stack] Tracks traversed objects and their clone counterparts.
     * @returns {*} Returns the cloned value.
     */
    function baseClone(value, bitmask, customizer, key, object, stack) {
      var result,
          isDeep = bitmask & CLONE_DEEP_FLAG,
          isFlat = bitmask & CLONE_FLAT_FLAG,
          isFull = bitmask & CLONE_SYMBOLS_FLAG;

      if (customizer) {
        result = object ? customizer(value, key, object, stack) : customizer(value);
      }
      if (result !== undefined) {
        return result;
      }
      if (!isObject_1(value)) {
        return value;
      }
      var isArr = isArray_1(value);
      if (isArr) {
        result = _initCloneArray(value);
        if (!isDeep) {
          return _copyArray(value, result);
        }
      } else {
        var tag = _getTag(value),
            isFunc = tag == funcTag$2 || tag == genTag$1;

        if (isBuffer_1(value)) {
          return _cloneBuffer(value, isDeep);
        }
        if (tag == objectTag$2 || tag == argsTag$2 || (isFunc && !object)) {
          result = (isFlat || isFunc) ? {} : _initCloneObject(value);
          if (!isDeep) {
            return isFlat
              ? _copySymbolsIn(value, _baseAssignIn(result, value))
              : _copySymbols(value, _baseAssign(result, value));
          }
        } else {
          if (!cloneableTags[tag]) {
            return object ? value : {};
          }
          result = _initCloneByTag(value, tag, isDeep);
        }
      }
      // Check for circular references and return its corresponding clone.
      stack || (stack = new _Stack);
      var stacked = stack.get(value);
      if (stacked) {
        return stacked;
      }
      stack.set(value, result);

      if (isSet_1(value)) {
        value.forEach(function(subValue) {
          result.add(baseClone(subValue, bitmask, customizer, subValue, value, stack));
        });
      } else if (isMap_1(value)) {
        value.forEach(function(subValue, key) {
          result.set(key, baseClone(subValue, bitmask, customizer, key, value, stack));
        });
      }

      var keysFunc = isFull
        ? (isFlat ? _getAllKeysIn : _getAllKeys)
        : (isFlat ? keysIn : keys_1);

      var props = isArr ? undefined : keysFunc(value);
      _arrayEach(props || value, function(subValue, key) {
        if (props) {
          key = subValue;
          subValue = value[key];
        }
        // Recursively populate clone (susceptible to call stack limits).
        _assignValue(result, key, baseClone(subValue, bitmask, customizer, key, value, stack));
      });
      return result;
    }

    var _baseClone = baseClone;

    /** Used to compose bitmasks for cloning. */
    var CLONE_SYMBOLS_FLAG$1 = 4;

    /**
     * Creates a shallow clone of `value`.
     *
     * **Note:** This method is loosely based on the
     * [structured clone algorithm](https://mdn.io/Structured_clone_algorithm)
     * and supports cloning arrays, array buffers, booleans, date objects, maps,
     * numbers, `Object` objects, regexes, sets, strings, symbols, and typed
     * arrays. The own enumerable properties of `arguments` objects are cloned
     * as plain objects. An empty object is returned for uncloneable values such
     * as error objects, functions, DOM nodes, and WeakMaps.
     *
     * @static
     * @memberOf _
     * @since 0.1.0
     * @category Lang
     * @param {*} value The value to clone.
     * @returns {*} Returns the cloned value.
     * @see _.cloneDeep
     * @example
     *
     * var objects = [{ 'a': 1 }, { 'b': 2 }];
     *
     * var shallow = _.clone(objects);
     * console.log(shallow[0] === objects[0]);
     * // => true
     */
    function clone(value) {
      return _baseClone(value, CLONE_SYMBOLS_FLAG$1);
    }

    var clone_1 = clone;

    /**
     * Creates a function that returns `value`.
     *
     * @static
     * @memberOf _
     * @since 2.4.0
     * @category Util
     * @param {*} value The value to return from the new function.
     * @returns {Function} Returns the new constant function.
     * @example
     *
     * var objects = _.times(2, _.constant({ 'a': 1 }));
     *
     * console.log(objects);
     * // => [{ 'a': 1 }, { 'a': 1 }]
     *
     * console.log(objects[0] === objects[1]);
     * // => true
     */
    function constant$z(value) {
      return function() {
        return value;
      };
    }

    var constant_1 = constant$z;

    /**
     * Creates a base function for methods like `_.forIn` and `_.forOwn`.
     *
     * @private
     * @param {boolean} [fromRight] Specify iterating from right to left.
     * @returns {Function} Returns the new base function.
     */
    function createBaseFor(fromRight) {
      return function(object, iteratee, keysFunc) {
        var index = -1,
            iterable = Object(object),
            props = keysFunc(object),
            length = props.length;

        while (length--) {
          var key = props[fromRight ? length : ++index];
          if (iteratee(iterable[key], key, iterable) === false) {
            break;
          }
        }
        return object;
      };
    }

    var _createBaseFor = createBaseFor;

    /**
     * The base implementation of `baseForOwn` which iterates over `object`
     * properties returned by `keysFunc` and invokes `iteratee` for each property.
     * Iteratee functions may exit iteration early by explicitly returning `false`.
     *
     * @private
     * @param {Object} object The object to iterate over.
     * @param {Function} iteratee The function invoked per iteration.
     * @param {Function} keysFunc The function to get the keys of `object`.
     * @returns {Object} Returns `object`.
     */
    var baseFor = _createBaseFor();

    var _baseFor = baseFor;

    /**
     * The base implementation of `_.forOwn` without support for iteratee shorthands.
     *
     * @private
     * @param {Object} object The object to iterate over.
     * @param {Function} iteratee The function invoked per iteration.
     * @returns {Object} Returns `object`.
     */
    function baseForOwn(object, iteratee) {
      return object && _baseFor(object, iteratee, keys_1);
    }

    var _baseForOwn = baseForOwn;

    /**
     * Creates a `baseEach` or `baseEachRight` function.
     *
     * @private
     * @param {Function} eachFunc The function to iterate over a collection.
     * @param {boolean} [fromRight] Specify iterating from right to left.
     * @returns {Function} Returns the new base function.
     */
    function createBaseEach(eachFunc, fromRight) {
      return function(collection, iteratee) {
        if (collection == null) {
          return collection;
        }
        if (!isArrayLike_1(collection)) {
          return eachFunc(collection, iteratee);
        }
        var length = collection.length,
            index = fromRight ? length : -1,
            iterable = Object(collection);

        while ((fromRight ? index-- : ++index < length)) {
          if (iteratee(iterable[index], index, iterable) === false) {
            break;
          }
        }
        return collection;
      };
    }

    var _createBaseEach = createBaseEach;

    /**
     * The base implementation of `_.forEach` without support for iteratee shorthands.
     *
     * @private
     * @param {Array|Object} collection The collection to iterate over.
     * @param {Function} iteratee The function invoked per iteration.
     * @returns {Array|Object} Returns `collection`.
     */
    var baseEach = _createBaseEach(_baseForOwn);

    var _baseEach = baseEach;

    /**
     * This method returns the first argument it receives.
     *
     * @static
     * @since 0.1.0
     * @memberOf _
     * @category Util
     * @param {*} value Any value.
     * @returns {*} Returns `value`.
     * @example
     *
     * var object = { 'a': 1 };
     *
     * console.log(_.identity(object) === object);
     * // => true
     */
    function identity$n(value) {
      return value;
    }

    var identity_1 = identity$n;

    /**
     * Casts `value` to `identity` if it's not a function.
     *
     * @private
     * @param {*} value The value to inspect.
     * @returns {Function} Returns cast function.
     */
    function castFunction(value) {
      return typeof value == 'function' ? value : identity_1;
    }

    var _castFunction = castFunction;

    /**
     * Iterates over elements of `collection` and invokes `iteratee` for each element.
     * The iteratee is invoked with three arguments: (value, index|key, collection).
     * Iteratee functions may exit iteration early by explicitly returning `false`.
     *
     * **Note:** As with other "Collections" methods, objects with a "length"
     * property are iterated like arrays. To avoid this behavior use `_.forIn`
     * or `_.forOwn` for object iteration.
     *
     * @static
     * @memberOf _
     * @since 0.1.0
     * @alias each
     * @category Collection
     * @param {Array|Object} collection The collection to iterate over.
     * @param {Function} [iteratee=_.identity] The function invoked per iteration.
     * @returns {Array|Object} Returns `collection`.
     * @see _.forEachRight
     * @example
     *
     * _.forEach([1, 2], function(value) {
     *   console.log(value);
     * });
     * // => Logs `1` then `2`.
     *
     * _.forEach({ 'a': 1, 'b': 2 }, function(value, key) {
     *   console.log(key);
     * });
     * // => Logs 'a' then 'b' (iteration order is not guaranteed).
     */
    function forEach(collection, iteratee) {
      var func = isArray_1(collection) ? _arrayEach : _baseEach;
      return func(collection, _castFunction(iteratee));
    }

    var forEach_1 = forEach;

    var each = forEach_1;

    /**
     * The base implementation of `_.filter` without support for iteratee shorthands.
     *
     * @private
     * @param {Array|Object} collection The collection to iterate over.
     * @param {Function} predicate The function invoked per iteration.
     * @returns {Array} Returns the new filtered array.
     */
    function baseFilter(collection, predicate) {
      var result = [];
      _baseEach(collection, function(value, index, collection) {
        if (predicate(value, index, collection)) {
          result.push(value);
        }
      });
      return result;
    }

    var _baseFilter = baseFilter;

    /** Used to stand-in for `undefined` hash values. */
    var HASH_UNDEFINED$2 = '__lodash_hash_undefined__';

    /**
     * Adds `value` to the array cache.
     *
     * @private
     * @name add
     * @memberOf SetCache
     * @alias push
     * @param {*} value The value to cache.
     * @returns {Object} Returns the cache instance.
     */
    function setCacheAdd(value) {
      this.__data__.set(value, HASH_UNDEFINED$2);
      return this;
    }

    var _setCacheAdd = setCacheAdd;

    /**
     * Checks if `value` is in the array cache.
     *
     * @private
     * @name has
     * @memberOf SetCache
     * @param {*} value The value to search for.
     * @returns {number} Returns `true` if `value` is found, else `false`.
     */
    function setCacheHas(value) {
      return this.__data__.has(value);
    }

    var _setCacheHas = setCacheHas;

    /**
     *
     * Creates an array cache object to store unique values.
     *
     * @private
     * @constructor
     * @param {Array} [values] The values to cache.
     */
    function SetCache(values) {
      var index = -1,
          length = values == null ? 0 : values.length;

      this.__data__ = new _MapCache;
      while (++index < length) {
        this.add(values[index]);
      }
    }

    // Add methods to `SetCache`.
    SetCache.prototype.add = SetCache.prototype.push = _setCacheAdd;
    SetCache.prototype.has = _setCacheHas;

    var _SetCache = SetCache;

    /**
     * A specialized version of `_.some` for arrays without support for iteratee
     * shorthands.
     *
     * @private
     * @param {Array} [array] The array to iterate over.
     * @param {Function} predicate The function invoked per iteration.
     * @returns {boolean} Returns `true` if any element passes the predicate check,
     *  else `false`.
     */
    function arraySome(array, predicate) {
      var index = -1,
          length = array == null ? 0 : array.length;

      while (++index < length) {
        if (predicate(array[index], index, array)) {
          return true;
        }
      }
      return false;
    }

    var _arraySome = arraySome;

    /**
     * Checks if a `cache` value for `key` exists.
     *
     * @private
     * @param {Object} cache The cache to query.
     * @param {string} key The key of the entry to check.
     * @returns {boolean} Returns `true` if an entry for `key` exists, else `false`.
     */
    function cacheHas(cache, key) {
      return cache.has(key);
    }

    var _cacheHas = cacheHas;

    /** Used to compose bitmasks for value comparisons. */
    var COMPARE_PARTIAL_FLAG = 1,
        COMPARE_UNORDERED_FLAG = 2;

    /**
     * A specialized version of `baseIsEqualDeep` for arrays with support for
     * partial deep comparisons.
     *
     * @private
     * @param {Array} array The array to compare.
     * @param {Array} other The other array to compare.
     * @param {number} bitmask The bitmask flags. See `baseIsEqual` for more details.
     * @param {Function} customizer The function to customize comparisons.
     * @param {Function} equalFunc The function to determine equivalents of values.
     * @param {Object} stack Tracks traversed `array` and `other` objects.
     * @returns {boolean} Returns `true` if the arrays are equivalent, else `false`.
     */
    function equalArrays(array, other, bitmask, customizer, equalFunc, stack) {
      var isPartial = bitmask & COMPARE_PARTIAL_FLAG,
          arrLength = array.length,
          othLength = other.length;

      if (arrLength != othLength && !(isPartial && othLength > arrLength)) {
        return false;
      }
      // Assume cyclic values are equal.
      var stacked = stack.get(array);
      if (stacked && stack.get(other)) {
        return stacked == other;
      }
      var index = -1,
          result = true,
          seen = (bitmask & COMPARE_UNORDERED_FLAG) ? new _SetCache : undefined;

      stack.set(array, other);
      stack.set(other, array);

      // Ignore non-index properties.
      while (++index < arrLength) {
        var arrValue = array[index],
            othValue = other[index];

        if (customizer) {
          var compared = isPartial
            ? customizer(othValue, arrValue, index, other, array, stack)
            : customizer(arrValue, othValue, index, array, other, stack);
        }
        if (compared !== undefined) {
          if (compared) {
            continue;
          }
          result = false;
          break;
        }
        // Recursively compare arrays (susceptible to call stack limits).
        if (seen) {
          if (!_arraySome(other, function(othValue, othIndex) {
                if (!_cacheHas(seen, othIndex) &&
                    (arrValue === othValue || equalFunc(arrValue, othValue, bitmask, customizer, stack))) {
                  return seen.push(othIndex);
                }
              })) {
            result = false;
            break;
          }
        } else if (!(
              arrValue === othValue ||
                equalFunc(arrValue, othValue, bitmask, customizer, stack)
            )) {
          result = false;
          break;
        }
      }
      stack['delete'](array);
      stack['delete'](other);
      return result;
    }

    var _equalArrays = equalArrays;

    /**
     * Converts `map` to its key-value pairs.
     *
     * @private
     * @param {Object} map The map to convert.
     * @returns {Array} Returns the key-value pairs.
     */
    function mapToArray(map) {
      var index = -1,
          result = Array(map.size);

      map.forEach(function(value, key) {
        result[++index] = [key, value];
      });
      return result;
    }

    var _mapToArray = mapToArray;

    /**
     * Converts `set` to an array of its values.
     *
     * @private
     * @param {Object} set The set to convert.
     * @returns {Array} Returns the values.
     */
    function setToArray(set) {
      var index = -1,
          result = Array(set.size);

      set.forEach(function(value) {
        result[++index] = value;
      });
      return result;
    }

    var _setToArray = setToArray;

    /** Used to compose bitmasks for value comparisons. */
    var COMPARE_PARTIAL_FLAG$1 = 1,
        COMPARE_UNORDERED_FLAG$1 = 2;

    /** `Object#toString` result references. */
    var boolTag$3 = '[object Boolean]',
        dateTag$3 = '[object Date]',
        errorTag$2 = '[object Error]',
        mapTag$5 = '[object Map]',
        numberTag$3 = '[object Number]',
        regexpTag$3 = '[object RegExp]',
        setTag$5 = '[object Set]',
        stringTag$3 = '[object String]',
        symbolTag$2 = '[object Symbol]';

    var arrayBufferTag$3 = '[object ArrayBuffer]',
        dataViewTag$4 = '[object DataView]';

    /** Used to convert symbols to primitives and strings. */
    var symbolProto$1 = _Symbol ? _Symbol.prototype : undefined,
        symbolValueOf$1 = symbolProto$1 ? symbolProto$1.valueOf : undefined;

    /**
     * A specialized version of `baseIsEqualDeep` for comparing objects of
     * the same `toStringTag`.
     *
     * **Note:** This function only supports comparing values with tags of
     * `Boolean`, `Date`, `Error`, `Number`, `RegExp`, or `String`.
     *
     * @private
     * @param {Object} object The object to compare.
     * @param {Object} other The other object to compare.
     * @param {string} tag The `toStringTag` of the objects to compare.
     * @param {number} bitmask The bitmask flags. See `baseIsEqual` for more details.
     * @param {Function} customizer The function to customize comparisons.
     * @param {Function} equalFunc The function to determine equivalents of values.
     * @param {Object} stack Tracks traversed `object` and `other` objects.
     * @returns {boolean} Returns `true` if the objects are equivalent, else `false`.
     */
    function equalByTag(object, other, tag, bitmask, customizer, equalFunc, stack) {
      switch (tag) {
        case dataViewTag$4:
          if ((object.byteLength != other.byteLength) ||
              (object.byteOffset != other.byteOffset)) {
            return false;
          }
          object = object.buffer;
          other = other.buffer;

        case arrayBufferTag$3:
          if ((object.byteLength != other.byteLength) ||
              !equalFunc(new _Uint8Array(object), new _Uint8Array(other))) {
            return false;
          }
          return true;

        case boolTag$3:
        case dateTag$3:
        case numberTag$3:
          // Coerce booleans to `1` or `0` and dates to milliseconds.
          // Invalid dates are coerced to `NaN`.
          return eq_1(+object, +other);

        case errorTag$2:
          return object.name == other.name && object.message == other.message;

        case regexpTag$3:
        case stringTag$3:
          // Coerce regexes to strings and treat strings, primitives and objects,
          // as equal. See http://www.ecma-international.org/ecma-262/7.0/#sec-regexp.prototype.tostring
          // for more details.
          return object == (other + '');

        case mapTag$5:
          var convert = _mapToArray;

        case setTag$5:
          var isPartial = bitmask & COMPARE_PARTIAL_FLAG$1;
          convert || (convert = _setToArray);

          if (object.size != other.size && !isPartial) {
            return false;
          }
          // Assume cyclic values are equal.
          var stacked = stack.get(object);
          if (stacked) {
            return stacked == other;
          }
          bitmask |= COMPARE_UNORDERED_FLAG$1;

          // Recursively compare objects (susceptible to call stack limits).
          stack.set(object, other);
          var result = _equalArrays(convert(object), convert(other), bitmask, customizer, equalFunc, stack);
          stack['delete'](object);
          return result;

        case symbolTag$2:
          if (symbolValueOf$1) {
            return symbolValueOf$1.call(object) == symbolValueOf$1.call(other);
          }
      }
      return false;
    }

    var _equalByTag = equalByTag;

    /** Used to compose bitmasks for value comparisons. */
    var COMPARE_PARTIAL_FLAG$2 = 1;

    /** Used for built-in method references. */
    var objectProto$d = Object.prototype;

    /** Used to check objects for own properties. */
    var hasOwnProperty$a = objectProto$d.hasOwnProperty;

    /**
     * A specialized version of `baseIsEqualDeep` for objects with support for
     * partial deep comparisons.
     *
     * @private
     * @param {Object} object The object to compare.
     * @param {Object} other The other object to compare.
     * @param {number} bitmask The bitmask flags. See `baseIsEqual` for more details.
     * @param {Function} customizer The function to customize comparisons.
     * @param {Function} equalFunc The function to determine equivalents of values.
     * @param {Object} stack Tracks traversed `object` and `other` objects.
     * @returns {boolean} Returns `true` if the objects are equivalent, else `false`.
     */
    function equalObjects(object, other, bitmask, customizer, equalFunc, stack) {
      var isPartial = bitmask & COMPARE_PARTIAL_FLAG$2,
          objProps = _getAllKeys(object),
          objLength = objProps.length,
          othProps = _getAllKeys(other),
          othLength = othProps.length;

      if (objLength != othLength && !isPartial) {
        return false;
      }
      var index = objLength;
      while (index--) {
        var key = objProps[index];
        if (!(isPartial ? key in other : hasOwnProperty$a.call(other, key))) {
          return false;
        }
      }
      // Assume cyclic values are equal.
      var stacked = stack.get(object);
      if (stacked && stack.get(other)) {
        return stacked == other;
      }
      var result = true;
      stack.set(object, other);
      stack.set(other, object);

      var skipCtor = isPartial;
      while (++index < objLength) {
        key = objProps[index];
        var objValue = object[key],
            othValue = other[key];

        if (customizer) {
          var compared = isPartial
            ? customizer(othValue, objValue, key, other, object, stack)
            : customizer(objValue, othValue, key, object, other, stack);
        }
        // Recursively compare objects (susceptible to call stack limits).
        if (!(compared === undefined
              ? (objValue === othValue || equalFunc(objValue, othValue, bitmask, customizer, stack))
              : compared
            )) {
          result = false;
          break;
        }
        skipCtor || (skipCtor = key == 'constructor');
      }
      if (result && !skipCtor) {
        var objCtor = object.constructor,
            othCtor = other.constructor;

        // Non `Object` object instances with different constructors are not equal.
        if (objCtor != othCtor &&
            ('constructor' in object && 'constructor' in other) &&
            !(typeof objCtor == 'function' && objCtor instanceof objCtor &&
              typeof othCtor == 'function' && othCtor instanceof othCtor)) {
          result = false;
        }
      }
      stack['delete'](object);
      stack['delete'](other);
      return result;
    }

    var _equalObjects = equalObjects;

    /** Used to compose bitmasks for value comparisons. */
    var COMPARE_PARTIAL_FLAG$3 = 1;

    /** `Object#toString` result references. */
    var argsTag$3 = '[object Arguments]',
        arrayTag$2 = '[object Array]',
        objectTag$3 = '[object Object]';

    /** Used for built-in method references. */
    var objectProto$e = Object.prototype;

    /** Used to check objects for own properties. */
    var hasOwnProperty$b = objectProto$e.hasOwnProperty;

    /**
     * A specialized version of `baseIsEqual` for arrays and objects which performs
     * deep comparisons and tracks traversed objects enabling objects with circular
     * references to be compared.
     *
     * @private
     * @param {Object} object The object to compare.
     * @param {Object} other The other object to compare.
     * @param {number} bitmask The bitmask flags. See `baseIsEqual` for more details.
     * @param {Function} customizer The function to customize comparisons.
     * @param {Function} equalFunc The function to determine equivalents of values.
     * @param {Object} [stack] Tracks traversed `object` and `other` objects.
     * @returns {boolean} Returns `true` if the objects are equivalent, else `false`.
     */
    function baseIsEqualDeep(object, other, bitmask, customizer, equalFunc, stack) {
      var objIsArr = isArray_1(object),
          othIsArr = isArray_1(other),
          objTag = objIsArr ? arrayTag$2 : _getTag(object),
          othTag = othIsArr ? arrayTag$2 : _getTag(other);

      objTag = objTag == argsTag$3 ? objectTag$3 : objTag;
      othTag = othTag == argsTag$3 ? objectTag$3 : othTag;

      var objIsObj = objTag == objectTag$3,
          othIsObj = othTag == objectTag$3,
          isSameTag = objTag == othTag;

      if (isSameTag && isBuffer_1(object)) {
        if (!isBuffer_1(other)) {
          return false;
        }
        objIsArr = true;
        objIsObj = false;
      }
      if (isSameTag && !objIsObj) {
        stack || (stack = new _Stack);
        return (objIsArr || isTypedArray_1(object))
          ? _equalArrays(object, other, bitmask, customizer, equalFunc, stack)
          : _equalByTag(object, other, objTag, bitmask, customizer, equalFunc, stack);
      }
      if (!(bitmask & COMPARE_PARTIAL_FLAG$3)) {
        var objIsWrapped = objIsObj && hasOwnProperty$b.call(object, '__wrapped__'),
            othIsWrapped = othIsObj && hasOwnProperty$b.call(other, '__wrapped__');

        if (objIsWrapped || othIsWrapped) {
          var objUnwrapped = objIsWrapped ? object.value() : object,
              othUnwrapped = othIsWrapped ? other.value() : other;

          stack || (stack = new _Stack);
          return equalFunc(objUnwrapped, othUnwrapped, bitmask, customizer, stack);
        }
      }
      if (!isSameTag) {
        return false;
      }
      stack || (stack = new _Stack);
      return _equalObjects(object, other, bitmask, customizer, equalFunc, stack);
    }

    var _baseIsEqualDeep = baseIsEqualDeep;

    /**
     * The base implementation of `_.isEqual` which supports partial comparisons
     * and tracks traversed objects.
     *
     * @private
     * @param {*} value The value to compare.
     * @param {*} other The other value to compare.
     * @param {boolean} bitmask The bitmask flags.
     *  1 - Unordered comparison
     *  2 - Partial comparison
     * @param {Function} [customizer] The function to customize comparisons.
     * @param {Object} [stack] Tracks traversed `value` and `other` objects.
     * @returns {boolean} Returns `true` if the values are equivalent, else `false`.
     */
    function baseIsEqual(value, other, bitmask, customizer, stack) {
      if (value === other) {
        return true;
      }
      if (value == null || other == null || (!isObjectLike_1(value) && !isObjectLike_1(other))) {
        return value !== value && other !== other;
      }
      return _baseIsEqualDeep(value, other, bitmask, customizer, baseIsEqual, stack);
    }

    var _baseIsEqual = baseIsEqual;

    /** Used to compose bitmasks for value comparisons. */
    var COMPARE_PARTIAL_FLAG$4 = 1,
        COMPARE_UNORDERED_FLAG$2 = 2;

    /**
     * The base implementation of `_.isMatch` without support for iteratee shorthands.
     *
     * @private
     * @param {Object} object The object to inspect.
     * @param {Object} source The object of property values to match.
     * @param {Array} matchData The property names, values, and compare flags to match.
     * @param {Function} [customizer] The function to customize comparisons.
     * @returns {boolean} Returns `true` if `object` is a match, else `false`.
     */
    function baseIsMatch(object, source, matchData, customizer) {
      var index = matchData.length,
          length = index,
          noCustomizer = !customizer;

      if (object == null) {
        return !length;
      }
      object = Object(object);
      while (index--) {
        var data = matchData[index];
        if ((noCustomizer && data[2])
              ? data[1] !== object[data[0]]
              : !(data[0] in object)
            ) {
          return false;
        }
      }
      while (++index < length) {
        data = matchData[index];
        var key = data[0],
            objValue = object[key],
            srcValue = data[1];

        if (noCustomizer && data[2]) {
          if (objValue === undefined && !(key in object)) {
            return false;
          }
        } else {
          var stack = new _Stack;
          if (customizer) {
            var result = customizer(objValue, srcValue, key, object, source, stack);
          }
          if (!(result === undefined
                ? _baseIsEqual(srcValue, objValue, COMPARE_PARTIAL_FLAG$4 | COMPARE_UNORDERED_FLAG$2, customizer, stack)
                : result
              )) {
            return false;
          }
        }
      }
      return true;
    }

    var _baseIsMatch = baseIsMatch;

    /**
     * Checks if `value` is suitable for strict equality comparisons, i.e. `===`.
     *
     * @private
     * @param {*} value The value to check.
     * @returns {boolean} Returns `true` if `value` if suitable for strict
     *  equality comparisons, else `false`.
     */
    function isStrictComparable(value) {
      return value === value && !isObject_1(value);
    }

    var _isStrictComparable = isStrictComparable;

    /**
     * Gets the property names, values, and compare flags of `object`.
     *
     * @private
     * @param {Object} object The object to query.
     * @returns {Array} Returns the match data of `object`.
     */
    function getMatchData(object) {
      var result = keys_1(object),
          length = result.length;

      while (length--) {
        var key = result[length],
            value = object[key];

        result[length] = [key, value, _isStrictComparable(value)];
      }
      return result;
    }

    var _getMatchData = getMatchData;

    /**
     * A specialized version of `matchesProperty` for source values suitable
     * for strict equality comparisons, i.e. `===`.
     *
     * @private
     * @param {string} key The key of the property to get.
     * @param {*} srcValue The value to match.
     * @returns {Function} Returns the new spec function.
     */
    function matchesStrictComparable(key, srcValue) {
      return function(object) {
        if (object == null) {
          return false;
        }
        return object[key] === srcValue &&
          (srcValue !== undefined || (key in Object(object)));
      };
    }

    var _matchesStrictComparable = matchesStrictComparable;

    /**
     * The base implementation of `_.matches` which doesn't clone `source`.
     *
     * @private
     * @param {Object} source The object of property values to match.
     * @returns {Function} Returns the new spec function.
     */
    function baseMatches(source) {
      var matchData = _getMatchData(source);
      if (matchData.length == 1 && matchData[0][2]) {
        return _matchesStrictComparable(matchData[0][0], matchData[0][1]);
      }
      return function(object) {
        return object === source || _baseIsMatch(object, source, matchData);
      };
    }

    var _baseMatches = baseMatches;

    /** `Object#toString` result references. */
    var symbolTag$3 = '[object Symbol]';

    /**
     * Checks if `value` is classified as a `Symbol` primitive or object.
     *
     * @static
     * @memberOf _
     * @since 4.0.0
     * @category Lang
     * @param {*} value The value to check.
     * @returns {boolean} Returns `true` if `value` is a symbol, else `false`.
     * @example
     *
     * _.isSymbol(Symbol.iterator);
     * // => true
     *
     * _.isSymbol('abc');
     * // => false
     */
    function isSymbol(value) {
      return typeof value == 'symbol' ||
        (isObjectLike_1(value) && _baseGetTag(value) == symbolTag$3);
    }

    var isSymbol_1 = isSymbol;

    /** Used to match property names within property paths. */
    var reIsDeepProp = /\.|\[(?:[^[\]]*|(["'])(?:(?!\1)[^\\]|\\.)*?\1)\]/,
        reIsPlainProp = /^\w*$/;

    /**
     * Checks if `value` is a property name and not a property path.
     *
     * @private
     * @param {*} value The value to check.
     * @param {Object} [object] The object to query keys on.
     * @returns {boolean} Returns `true` if `value` is a property name, else `false`.
     */
    function isKey(value, object) {
      if (isArray_1(value)) {
        return false;
      }
      var type = typeof value;
      if (type == 'number' || type == 'symbol' || type == 'boolean' ||
          value == null || isSymbol_1(value)) {
        return true;
      }
      return reIsPlainProp.test(value) || !reIsDeepProp.test(value) ||
        (object != null && value in Object(object));
    }

    var _isKey = isKey;

    /** Error message constants. */
    var FUNC_ERROR_TEXT = 'Expected a function';

    /**
     * Creates a function that memoizes the result of `func`. If `resolver` is
     * provided, it determines the cache key for storing the result based on the
     * arguments provided to the memoized function. By default, the first argument
     * provided to the memoized function is used as the map cache key. The `func`
     * is invoked with the `this` binding of the memoized function.
     *
     * **Note:** The cache is exposed as the `cache` property on the memoized
     * function. Its creation may be customized by replacing the `_.memoize.Cache`
     * constructor with one whose instances implement the
     * [`Map`](http://ecma-international.org/ecma-262/7.0/#sec-properties-of-the-map-prototype-object)
     * method interface of `clear`, `delete`, `get`, `has`, and `set`.
     *
     * @static
     * @memberOf _
     * @since 0.1.0
     * @category Function
     * @param {Function} func The function to have its output memoized.
     * @param {Function} [resolver] The function to resolve the cache key.
     * @returns {Function} Returns the new memoized function.
     * @example
     *
     * var object = { 'a': 1, 'b': 2 };
     * var other = { 'c': 3, 'd': 4 };
     *
     * var values = _.memoize(_.values);
     * values(object);
     * // => [1, 2]
     *
     * values(other);
     * // => [3, 4]
     *
     * object.a = 2;
     * values(object);
     * // => [1, 2]
     *
     * // Modify the result cache.
     * values.cache.set(object, ['a', 'b']);
     * values(object);
     * // => ['a', 'b']
     *
     * // Replace `_.memoize.Cache`.
     * _.memoize.Cache = WeakMap;
     */
    function memoize$1(func, resolver) {
      if (typeof func != 'function' || (resolver != null && typeof resolver != 'function')) {
        throw new TypeError(FUNC_ERROR_TEXT);
      }
      var memoized = function() {
        var args = arguments,
            key = resolver ? resolver.apply(this, args) : args[0],
            cache = memoized.cache;

        if (cache.has(key)) {
          return cache.get(key);
        }
        var result = func.apply(this, args);
        memoized.cache = cache.set(key, result) || cache;
        return result;
      };
      memoized.cache = new (memoize$1.Cache || _MapCache);
      return memoized;
    }

    // Expose `MapCache`.
    memoize$1.Cache = _MapCache;

    var memoize_1$1 = memoize$1;

    /** Used as the maximum memoize cache size. */
    var MAX_MEMOIZE_SIZE = 500;

    /**
     * A specialized version of `_.memoize` which clears the memoized function's
     * cache when it exceeds `MAX_MEMOIZE_SIZE`.
     *
     * @private
     * @param {Function} func The function to have its output memoized.
     * @returns {Function} Returns the new memoized function.
     */
    function memoizeCapped(func) {
      var result = memoize_1$1(func, function(key) {
        if (cache.size === MAX_MEMOIZE_SIZE) {
          cache.clear();
        }
        return key;
      });

      var cache = result.cache;
      return result;
    }

    var _memoizeCapped = memoizeCapped;

    /** Used to match property names within property paths. */
    var rePropName = /[^.[\]]+|\[(?:(-?\d+(?:\.\d+)?)|(["'])((?:(?!\2)[^\\]|\\.)*?)\2)\]|(?=(?:\.|\[\])(?:\.|\[\]|$))/g;

    /** Used to match backslashes in property paths. */
    var reEscapeChar = /\\(\\)?/g;

    /**
     * Converts `string` to a property path array.
     *
     * @private
     * @param {string} string The string to convert.
     * @returns {Array} Returns the property path array.
     */
    var stringToPath = _memoizeCapped(function(string) {
      var result = [];
      if (string.charCodeAt(0) === 46 /* . */) {
        result.push('');
      }
      string.replace(rePropName, function(match, number, quote, subString) {
        result.push(quote ? subString.replace(reEscapeChar, '$1') : (number || match));
      });
      return result;
    });

    var _stringToPath = stringToPath;

    /**
     * A specialized version of `_.map` for arrays without support for iteratee
     * shorthands.
     *
     * @private
     * @param {Array} [array] The array to iterate over.
     * @param {Function} iteratee The function invoked per iteration.
     * @returns {Array} Returns the new mapped array.
     */
    function arrayMap(array, iteratee) {
      var index = -1,
          length = array == null ? 0 : array.length,
          result = Array(length);

      while (++index < length) {
        result[index] = iteratee(array[index], index, array);
      }
      return result;
    }

    var _arrayMap = arrayMap;

    /** Used as references for various `Number` constants. */
    var INFINITY = 1 / 0;

    /** Used to convert symbols to primitives and strings. */
    var symbolProto$2 = _Symbol ? _Symbol.prototype : undefined,
        symbolToString = symbolProto$2 ? symbolProto$2.toString : undefined;

    /**
     * The base implementation of `_.toString` which doesn't convert nullish
     * values to empty strings.
     *
     * @private
     * @param {*} value The value to process.
     * @returns {string} Returns the string.
     */
    function baseToString(value) {
      // Exit early for strings to avoid a performance hit in some environments.
      if (typeof value == 'string') {
        return value;
      }
      if (isArray_1(value)) {
        // Recursively convert values (susceptible to call stack limits).
        return _arrayMap(value, baseToString) + '';
      }
      if (isSymbol_1(value)) {
        return symbolToString ? symbolToString.call(value) : '';
      }
      var result = (value + '');
      return (result == '0' && (1 / value) == -INFINITY) ? '-0' : result;
    }

    var _baseToString = baseToString;

    /**
     * Converts `value` to a string. An empty string is returned for `null`
     * and `undefined` values. The sign of `-0` is preserved.
     *
     * @static
     * @memberOf _
     * @since 4.0.0
     * @category Lang
     * @param {*} value The value to convert.
     * @returns {string} Returns the converted string.
     * @example
     *
     * _.toString(null);
     * // => ''
     *
     * _.toString(-0);
     * // => '-0'
     *
     * _.toString([1, 2, 3]);
     * // => '1,2,3'
     */
    function toString(value) {
      return value == null ? '' : _baseToString(value);
    }

    var toString_1 = toString;

    /**
     * Casts `value` to a path array if it's not one.
     *
     * @private
     * @param {*} value The value to inspect.
     * @param {Object} [object] The object to query keys on.
     * @returns {Array} Returns the cast property path array.
     */
    function castPath(value, object) {
      if (isArray_1(value)) {
        return value;
      }
      return _isKey(value, object) ? [value] : _stringToPath(toString_1(value));
    }

    var _castPath = castPath;

    /** Used as references for various `Number` constants. */
    var INFINITY$1 = 1 / 0;

    /**
     * Converts `value` to a string key if it's not a string or symbol.
     *
     * @private
     * @param {*} value The value to inspect.
     * @returns {string|symbol} Returns the key.
     */
    function toKey(value) {
      if (typeof value == 'string' || isSymbol_1(value)) {
        return value;
      }
      var result = (value + '');
      return (result == '0' && (1 / value) == -INFINITY$1) ? '-0' : result;
    }

    var _toKey = toKey;

    /**
     * The base implementation of `_.get` without support for default values.
     *
     * @private
     * @param {Object} object The object to query.
     * @param {Array|string} path The path of the property to get.
     * @returns {*} Returns the resolved value.
     */
    function baseGet(object, path) {
      path = _castPath(path, object);

      var index = 0,
          length = path.length;

      while (object != null && index < length) {
        object = object[_toKey(path[index++])];
      }
      return (index && index == length) ? object : undefined;
    }

    var _baseGet = baseGet;

    /**
     * Gets the value at `path` of `object`. If the resolved value is
     * `undefined`, the `defaultValue` is returned in its place.
     *
     * @static
     * @memberOf _
     * @since 3.7.0
     * @category Object
     * @param {Object} object The object to query.
     * @param {Array|string} path The path of the property to get.
     * @param {*} [defaultValue] The value returned for `undefined` resolved values.
     * @returns {*} Returns the resolved value.
     * @example
     *
     * var object = { 'a': [{ 'b': { 'c': 3 } }] };
     *
     * _.get(object, 'a[0].b.c');
     * // => 3
     *
     * _.get(object, ['a', '0', 'b', 'c']);
     * // => 3
     *
     * _.get(object, 'a.b.c', 'default');
     * // => 'default'
     */
    function get$c(object, path, defaultValue) {
      var result = object == null ? undefined : _baseGet(object, path);
      return result === undefined ? defaultValue : result;
    }

    var get_1 = get$c;

    /**
     * The base implementation of `_.hasIn` without support for deep paths.
     *
     * @private
     * @param {Object} [object] The object to query.
     * @param {Array|string} key The key to check.
     * @returns {boolean} Returns `true` if `key` exists, else `false`.
     */
    function baseHasIn(object, key) {
      return object != null && key in Object(object);
    }

    var _baseHasIn = baseHasIn;

    /**
     * Checks if `path` exists on `object`.
     *
     * @private
     * @param {Object} object The object to query.
     * @param {Array|string} path The path to check.
     * @param {Function} hasFunc The function to check properties.
     * @returns {boolean} Returns `true` if `path` exists, else `false`.
     */
    function hasPath(object, path, hasFunc) {
      path = _castPath(path, object);

      var index = -1,
          length = path.length,
          result = false;

      while (++index < length) {
        var key = _toKey(path[index]);
        if (!(result = object != null && hasFunc(object, key))) {
          break;
        }
        object = object[key];
      }
      if (result || ++index != length) {
        return result;
      }
      length = object == null ? 0 : object.length;
      return !!length && isLength_1(length) && _isIndex(key, length) &&
        (isArray_1(object) || isArguments_1(object));
    }

    var _hasPath = hasPath;

    /**
     * Checks if `path` is a direct or inherited property of `object`.
     *
     * @static
     * @memberOf _
     * @since 4.0.0
     * @category Object
     * @param {Object} object The object to query.
     * @param {Array|string} path The path to check.
     * @returns {boolean} Returns `true` if `path` exists, else `false`.
     * @example
     *
     * var object = _.create({ 'a': _.create({ 'b': 2 }) });
     *
     * _.hasIn(object, 'a');
     * // => true
     *
     * _.hasIn(object, 'a.b');
     * // => true
     *
     * _.hasIn(object, ['a', 'b']);
     * // => true
     *
     * _.hasIn(object, 'b');
     * // => false
     */
    function hasIn(object, path) {
      return object != null && _hasPath(object, path, _baseHasIn);
    }

    var hasIn_1 = hasIn;

    /** Used to compose bitmasks for value comparisons. */
    var COMPARE_PARTIAL_FLAG$5 = 1,
        COMPARE_UNORDERED_FLAG$3 = 2;

    /**
     * The base implementation of `_.matchesProperty` which doesn't clone `srcValue`.
     *
     * @private
     * @param {string} path The path of the property to get.
     * @param {*} srcValue The value to match.
     * @returns {Function} Returns the new spec function.
     */
    function baseMatchesProperty(path, srcValue) {
      if (_isKey(path) && _isStrictComparable(srcValue)) {
        return _matchesStrictComparable(_toKey(path), srcValue);
      }
      return function(object) {
        var objValue = get_1(object, path);
        return (objValue === undefined && objValue === srcValue)
          ? hasIn_1(object, path)
          : _baseIsEqual(srcValue, objValue, COMPARE_PARTIAL_FLAG$5 | COMPARE_UNORDERED_FLAG$3);
      };
    }

    var _baseMatchesProperty = baseMatchesProperty;

    /**
     * The base implementation of `_.property` without support for deep paths.
     *
     * @private
     * @param {string} key The key of the property to get.
     * @returns {Function} Returns the new accessor function.
     */
    function baseProperty(key) {
      return function(object) {
        return object == null ? undefined : object[key];
      };
    }

    var _baseProperty = baseProperty;

    /**
     * A specialized version of `baseProperty` which supports deep paths.
     *
     * @private
     * @param {Array|string} path The path of the property to get.
     * @returns {Function} Returns the new accessor function.
     */
    function basePropertyDeep(path) {
      return function(object) {
        return _baseGet(object, path);
      };
    }

    var _basePropertyDeep = basePropertyDeep;

    /**
     * Creates a function that returns the value at `path` of a given object.
     *
     * @static
     * @memberOf _
     * @since 2.4.0
     * @category Util
     * @param {Array|string} path The path of the property to get.
     * @returns {Function} Returns the new accessor function.
     * @example
     *
     * var objects = [
     *   { 'a': { 'b': 2 } },
     *   { 'a': { 'b': 1 } }
     * ];
     *
     * _.map(objects, _.property('a.b'));
     * // => [2, 1]
     *
     * _.map(_.sortBy(objects, _.property(['a', 'b'])), 'a.b');
     * // => [1, 2]
     */
    function property$1(path) {
      return _isKey(path) ? _baseProperty(_toKey(path)) : _basePropertyDeep(path);
    }

    var property_1 = property$1;

    /**
     * The base implementation of `_.iteratee`.
     *
     * @private
     * @param {*} [value=_.identity] The value to convert to an iteratee.
     * @returns {Function} Returns the iteratee.
     */
    function baseIteratee(value) {
      // Don't store the `typeof` result in a variable to avoid a JIT bug in Safari 9.
      // See https://bugs.webkit.org/show_bug.cgi?id=156034 for more details.
      if (typeof value == 'function') {
        return value;
      }
      if (value == null) {
        return identity_1;
      }
      if (typeof value == 'object') {
        return isArray_1(value)
          ? _baseMatchesProperty(value[0], value[1])
          : _baseMatches(value);
      }
      return property_1(value);
    }

    var _baseIteratee = baseIteratee;

    /**
     * Iterates over elements of `collection`, returning an array of all elements
     * `predicate` returns truthy for. The predicate is invoked with three
     * arguments: (value, index|key, collection).
     *
     * **Note:** Unlike `_.remove`, this method returns a new array.
     *
     * @static
     * @memberOf _
     * @since 0.1.0
     * @category Collection
     * @param {Array|Object} collection The collection to iterate over.
     * @param {Function} [predicate=_.identity] The function invoked per iteration.
     * @returns {Array} Returns the new filtered array.
     * @see _.reject
     * @example
     *
     * var users = [
     *   { 'user': 'barney', 'age': 36, 'active': true },
     *   { 'user': 'fred',   'age': 40, 'active': false }
     * ];
     *
     * _.filter(users, function(o) { return !o.active; });
     * // => objects for ['fred']
     *
     * // The `_.matches` iteratee shorthand.
     * _.filter(users, { 'age': 36, 'active': true });
     * // => objects for ['barney']
     *
     * // The `_.matchesProperty` iteratee shorthand.
     * _.filter(users, ['active', false]);
     * // => objects for ['fred']
     *
     * // The `_.property` iteratee shorthand.
     * _.filter(users, 'active');
     * // => objects for ['barney']
     */
    function filter(collection, predicate) {
      var func = isArray_1(collection) ? _arrayFilter : _baseFilter;
      return func(collection, _baseIteratee(predicate));
    }

    var filter_1 = filter;

    /** Used for built-in method references. */
    var objectProto$f = Object.prototype;

    /** Used to check objects for own properties. */
    var hasOwnProperty$c = objectProto$f.hasOwnProperty;

    /**
     * The base implementation of `_.has` without support for deep paths.
     *
     * @private
     * @param {Object} [object] The object to query.
     * @param {Array|string} key The key to check.
     * @returns {boolean} Returns `true` if `key` exists, else `false`.
     */
    function baseHas(object, key) {
      return object != null && hasOwnProperty$c.call(object, key);
    }

    var _baseHas = baseHas;

    /**
     * Checks if `path` is a direct property of `object`.
     *
     * @static
     * @since 0.1.0
     * @memberOf _
     * @category Object
     * @param {Object} object The object to query.
     * @param {Array|string} path The path to check.
     * @returns {boolean} Returns `true` if `path` exists, else `false`.
     * @example
     *
     * var object = { 'a': { 'b': 2 } };
     * var other = _.create({ 'a': _.create({ 'b': 2 }) });
     *
     * _.has(object, 'a');
     * // => true
     *
     * _.has(object, 'a.b');
     * // => true
     *
     * _.has(object, ['a', 'b']);
     * // => true
     *
     * _.has(other, 'a');
     * // => false
     */
    function has(object, path) {
      return object != null && _hasPath(object, path, _baseHas);
    }

    var has_1 = has;

    /** `Object#toString` result references. */
    var mapTag$6 = '[object Map]',
        setTag$6 = '[object Set]';

    /** Used for built-in method references. */
    var objectProto$g = Object.prototype;

    /** Used to check objects for own properties. */
    var hasOwnProperty$d = objectProto$g.hasOwnProperty;

    /**
     * Checks if `value` is an empty object, collection, map, or set.
     *
     * Objects are considered empty if they have no own enumerable string keyed
     * properties.
     *
     * Array-like values such as `arguments` objects, arrays, buffers, strings, or
     * jQuery-like collections are considered empty if they have a `length` of `0`.
     * Similarly, maps and sets are considered empty if they have a `size` of `0`.
     *
     * @static
     * @memberOf _
     * @since 0.1.0
     * @category Lang
     * @param {*} value The value to check.
     * @returns {boolean} Returns `true` if `value` is empty, else `false`.
     * @example
     *
     * _.isEmpty(null);
     * // => true
     *
     * _.isEmpty(true);
     * // => true
     *
     * _.isEmpty(1);
     * // => true
     *
     * _.isEmpty([1, 2, 3]);
     * // => false
     *
     * _.isEmpty({ 'a': 1 });
     * // => false
     */
    function isEmpty(value) {
      if (value == null) {
        return true;
      }
      if (isArrayLike_1(value) &&
          (isArray_1(value) || typeof value == 'string' || typeof value.splice == 'function' ||
            isBuffer_1(value) || isTypedArray_1(value) || isArguments_1(value))) {
        return !value.length;
      }
      var tag = _getTag(value);
      if (tag == mapTag$6 || tag == setTag$6) {
        return !value.size;
      }
      if (_isPrototype(value)) {
        return !_baseKeys(value).length;
      }
      for (var key in value) {
        if (hasOwnProperty$d.call(value, key)) {
          return false;
        }
      }
      return true;
    }

    var isEmpty_1 = isEmpty;

    /**
     * Checks if `value` is `undefined`.
     *
     * @static
     * @since 0.1.0
     * @memberOf _
     * @category Lang
     * @param {*} value The value to check.
     * @returns {boolean} Returns `true` if `value` is `undefined`, else `false`.
     * @example
     *
     * _.isUndefined(void 0);
     * // => true
     *
     * _.isUndefined(null);
     * // => false
     */
    function isUndefined(value) {
      return value === undefined;
    }

    var isUndefined_1 = isUndefined;

    /**
     * The base implementation of `_.map` without support for iteratee shorthands.
     *
     * @private
     * @param {Array|Object} collection The collection to iterate over.
     * @param {Function} iteratee The function invoked per iteration.
     * @returns {Array} Returns the new mapped array.
     */
    function baseMap(collection, iteratee) {
      var index = -1,
          result = isArrayLike_1(collection) ? Array(collection.length) : [];

      _baseEach(collection, function(value, key, collection) {
        result[++index] = iteratee(value, key, collection);
      });
      return result;
    }

    var _baseMap = baseMap;

    /**
     * Creates an array of values by running each element in `collection` thru
     * `iteratee`. The iteratee is invoked with three arguments:
     * (value, index|key, collection).
     *
     * Many lodash methods are guarded to work as iteratees for methods like
     * `_.every`, `_.filter`, `_.map`, `_.mapValues`, `_.reject`, and `_.some`.
     *
     * The guarded methods are:
     * `ary`, `chunk`, `curry`, `curryRight`, `drop`, `dropRight`, `every`,
     * `fill`, `invert`, `parseInt`, `random`, `range`, `rangeRight`, `repeat`,
     * `sampleSize`, `slice`, `some`, `sortBy`, `split`, `take`, `takeRight`,
     * `template`, `trim`, `trimEnd`, `trimStart`, and `words`
     *
     * @static
     * @memberOf _
     * @since 0.1.0
     * @category Collection
     * @param {Array|Object} collection The collection to iterate over.
     * @param {Function} [iteratee=_.identity] The function invoked per iteration.
     * @returns {Array} Returns the new mapped array.
     * @example
     *
     * function square(n) {
     *   return n * n;
     * }
     *
     * _.map([4, 8], square);
     * // => [16, 64]
     *
     * _.map({ 'a': 4, 'b': 8 }, square);
     * // => [16, 64] (iteration order is not guaranteed)
     *
     * var users = [
     *   { 'user': 'barney' },
     *   { 'user': 'fred' }
     * ];
     *
     * // The `_.property` iteratee shorthand.
     * _.map(users, 'user');
     * // => ['barney', 'fred']
     */
    function map$b(collection, iteratee) {
      var func = isArray_1(collection) ? _arrayMap : _baseMap;
      return func(collection, _baseIteratee(iteratee));
    }

    var map_1 = map$b;

    /**
     * A specialized version of `_.reduce` for arrays without support for
     * iteratee shorthands.
     *
     * @private
     * @param {Array} [array] The array to iterate over.
     * @param {Function} iteratee The function invoked per iteration.
     * @param {*} [accumulator] The initial value.
     * @param {boolean} [initAccum] Specify using the first element of `array` as
     *  the initial value.
     * @returns {*} Returns the accumulated value.
     */
    function arrayReduce(array, iteratee, accumulator, initAccum) {
      var index = -1,
          length = array == null ? 0 : array.length;

      if (initAccum && length) {
        accumulator = array[++index];
      }
      while (++index < length) {
        accumulator = iteratee(accumulator, array[index], index, array);
      }
      return accumulator;
    }

    var _arrayReduce = arrayReduce;

    /**
     * The base implementation of `_.reduce` and `_.reduceRight`, without support
     * for iteratee shorthands, which iterates over `collection` using `eachFunc`.
     *
     * @private
     * @param {Array|Object} collection The collection to iterate over.
     * @param {Function} iteratee The function invoked per iteration.
     * @param {*} accumulator The initial value.
     * @param {boolean} initAccum Specify using the first or last element of
     *  `collection` as the initial value.
     * @param {Function} eachFunc The function to iterate over `collection`.
     * @returns {*} Returns the accumulated value.
     */
    function baseReduce(collection, iteratee, accumulator, initAccum, eachFunc) {
      eachFunc(collection, function(value, index, collection) {
        accumulator = initAccum
          ? (initAccum = false, value)
          : iteratee(accumulator, value, index, collection);
      });
      return accumulator;
    }

    var _baseReduce = baseReduce;

    /**
     * Reduces `collection` to a value which is the accumulated result of running
     * each element in `collection` thru `iteratee`, where each successive
     * invocation is supplied the return value of the previous. If `accumulator`
     * is not given, the first element of `collection` is used as the initial
     * value. The iteratee is invoked with four arguments:
     * (accumulator, value, index|key, collection).
     *
     * Many lodash methods are guarded to work as iteratees for methods like
     * `_.reduce`, `_.reduceRight`, and `_.transform`.
     *
     * The guarded methods are:
     * `assign`, `defaults`, `defaultsDeep`, `includes`, `merge`, `orderBy`,
     * and `sortBy`
     *
     * @static
     * @memberOf _
     * @since 0.1.0
     * @category Collection
     * @param {Array|Object} collection The collection to iterate over.
     * @param {Function} [iteratee=_.identity] The function invoked per iteration.
     * @param {*} [accumulator] The initial value.
     * @returns {*} Returns the accumulated value.
     * @see _.reduceRight
     * @example
     *
     * _.reduce([1, 2], function(sum, n) {
     *   return sum + n;
     * }, 0);
     * // => 3
     *
     * _.reduce({ 'a': 1, 'b': 2, 'c': 1 }, function(result, value, key) {
     *   (result[value] || (result[value] = [])).push(key);
     *   return result;
     * }, {});
     * // => { '1': ['a', 'c'], '2': ['b'] } (iteration order is not guaranteed)
     */
    function reduce(collection, iteratee, accumulator) {
      var func = isArray_1(collection) ? _arrayReduce : _baseReduce,
          initAccum = arguments.length < 3;

      return func(collection, _baseIteratee(iteratee), accumulator, initAccum, _baseEach);
    }

    var reduce_1 = reduce;

    /** `Object#toString` result references. */
    var stringTag$4 = '[object String]';

    /**
     * Checks if `value` is classified as a `String` primitive or object.
     *
     * @static
     * @since 0.1.0
     * @memberOf _
     * @category Lang
     * @param {*} value The value to check.
     * @returns {boolean} Returns `true` if `value` is a string, else `false`.
     * @example
     *
     * _.isString('abc');
     * // => true
     *
     * _.isString(1);
     * // => false
     */
    function isString(value) {
      return typeof value == 'string' ||
        (!isArray_1(value) && isObjectLike_1(value) && _baseGetTag(value) == stringTag$4);
    }

    var isString_1 = isString;

    /**
     * Gets the size of an ASCII `string`.
     *
     * @private
     * @param {string} string The string inspect.
     * @returns {number} Returns the string size.
     */
    var asciiSize = _baseProperty('length');

    var _asciiSize = asciiSize;

    /** Used to compose unicode character classes. */
    var rsAstralRange = '\\ud800-\\udfff',
        rsComboMarksRange = '\\u0300-\\u036f',
        reComboHalfMarksRange = '\\ufe20-\\ufe2f',
        rsComboSymbolsRange = '\\u20d0-\\u20ff',
        rsComboRange = rsComboMarksRange + reComboHalfMarksRange + rsComboSymbolsRange,
        rsVarRange = '\\ufe0e\\ufe0f';

    /** Used to compose unicode capture groups. */
    var rsZWJ = '\\u200d';

    /** Used to detect strings with [zero-width joiners or code points from the astral planes](http://eev.ee/blog/2015/09/12/dark-corners-of-unicode/). */
    var reHasUnicode = RegExp('[' + rsZWJ + rsAstralRange  + rsComboRange + rsVarRange + ']');

    /**
     * Checks if `string` contains Unicode symbols.
     *
     * @private
     * @param {string} string The string to inspect.
     * @returns {boolean} Returns `true` if a symbol is found, else `false`.
     */
    function hasUnicode(string) {
      return reHasUnicode.test(string);
    }

    var _hasUnicode = hasUnicode;

    /** Used to compose unicode character classes. */
    var rsAstralRange$1 = '\\ud800-\\udfff',
        rsComboMarksRange$1 = '\\u0300-\\u036f',
        reComboHalfMarksRange$1 = '\\ufe20-\\ufe2f',
        rsComboSymbolsRange$1 = '\\u20d0-\\u20ff',
        rsComboRange$1 = rsComboMarksRange$1 + reComboHalfMarksRange$1 + rsComboSymbolsRange$1,
        rsVarRange$1 = '\\ufe0e\\ufe0f';

    /** Used to compose unicode capture groups. */
    var rsAstral = '[' + rsAstralRange$1 + ']',
        rsCombo = '[' + rsComboRange$1 + ']',
        rsFitz = '\\ud83c[\\udffb-\\udfff]',
        rsModifier = '(?:' + rsCombo + '|' + rsFitz + ')',
        rsNonAstral = '[^' + rsAstralRange$1 + ']',
        rsRegional = '(?:\\ud83c[\\udde6-\\uddff]){2}',
        rsSurrPair = '[\\ud800-\\udbff][\\udc00-\\udfff]',
        rsZWJ$1 = '\\u200d';

    /** Used to compose unicode regexes. */
    var reOptMod = rsModifier + '?',
        rsOptVar = '[' + rsVarRange$1 + ']?',
        rsOptJoin = '(?:' + rsZWJ$1 + '(?:' + [rsNonAstral, rsRegional, rsSurrPair].join('|') + ')' + rsOptVar + reOptMod + ')*',
        rsSeq = rsOptVar + reOptMod + rsOptJoin,
        rsSymbol = '(?:' + [rsNonAstral + rsCombo + '?', rsCombo, rsRegional, rsSurrPair, rsAstral].join('|') + ')';

    /** Used to match [string symbols](https://mathiasbynens.be/notes/javascript-unicode). */
    var reUnicode = RegExp(rsFitz + '(?=' + rsFitz + ')|' + rsSymbol + rsSeq, 'g');

    /**
     * Gets the size of a Unicode `string`.
     *
     * @private
     * @param {string} string The string inspect.
     * @returns {number} Returns the string size.
     */
    function unicodeSize(string) {
      var result = reUnicode.lastIndex = 0;
      while (reUnicode.test(string)) {
        ++result;
      }
      return result;
    }

    var _unicodeSize = unicodeSize;

    /**
     * Gets the number of symbols in `string`.
     *
     * @private
     * @param {string} string The string to inspect.
     * @returns {number} Returns the string size.
     */
    function stringSize(string) {
      return _hasUnicode(string)
        ? _unicodeSize(string)
        : _asciiSize(string);
    }

    var _stringSize = stringSize;

    /** `Object#toString` result references. */
    var mapTag$7 = '[object Map]',
        setTag$7 = '[object Set]';

    /**
     * Gets the size of `collection` by returning its length for array-like
     * values or the number of own enumerable string keyed properties for objects.
     *
     * @static
     * @memberOf _
     * @since 0.1.0
     * @category Collection
     * @param {Array|Object|string} collection The collection to inspect.
     * @returns {number} Returns the collection size.
     * @example
     *
     * _.size([1, 2, 3]);
     * // => 3
     *
     * _.size({ 'a': 1, 'b': 2 });
     * // => 2
     *
     * _.size('pebbles');
     * // => 7
     */
    function size(collection) {
      if (collection == null) {
        return 0;
      }
      if (isArrayLike_1(collection)) {
        return isString_1(collection) ? _stringSize(collection) : collection.length;
      }
      var tag = _getTag(collection);
      if (tag == mapTag$7 || tag == setTag$7) {
        return collection.size;
      }
      return _baseKeys(collection).length;
    }

    var size_1 = size;

    /**
     * An alternative to `_.reduce`; this method transforms `object` to a new
     * `accumulator` object which is the result of running each of its own
     * enumerable string keyed properties thru `iteratee`, with each invocation
     * potentially mutating the `accumulator` object. If `accumulator` is not
     * provided, a new object with the same `[[Prototype]]` will be used. The
     * iteratee is invoked with four arguments: (accumulator, value, key, object).
     * Iteratee functions may exit iteration early by explicitly returning `false`.
     *
     * @static
     * @memberOf _
     * @since 1.3.0
     * @category Object
     * @param {Object} object The object to iterate over.
     * @param {Function} [iteratee=_.identity] The function invoked per iteration.
     * @param {*} [accumulator] The custom accumulator value.
     * @returns {*} Returns the accumulated value.
     * @example
     *
     * _.transform([2, 3, 4], function(result, n) {
     *   result.push(n *= n);
     *   return n % 2 == 0;
     * }, []);
     * // => [4, 9]
     *
     * _.transform({ 'a': 1, 'b': 2, 'c': 1 }, function(result, value, key) {
     *   (result[value] || (result[value] = [])).push(key);
     * }, {});
     * // => { '1': ['a', 'c'], '2': ['b'] }
     */
    function transform$4(object, iteratee, accumulator) {
      var isArr = isArray_1(object),
          isArrLike = isArr || isBuffer_1(object) || isTypedArray_1(object);

      iteratee = _baseIteratee(iteratee);
      if (accumulator == null) {
        var Ctor = object && object.constructor;
        if (isArrLike) {
          accumulator = isArr ? new Ctor : [];
        }
        else if (isObject_1(object)) {
          accumulator = isFunction_1(Ctor) ? _baseCreate(_getPrototype(object)) : {};
        }
        else {
          accumulator = {};
        }
      }
      (isArrLike ? _arrayEach : _baseForOwn)(object, function(value, index, object) {
        return iteratee(accumulator, value, index, object);
      });
      return accumulator;
    }

    var transform_1 = transform$4;

    /** Built-in value references. */
    var spreadableSymbol = _Symbol ? _Symbol.isConcatSpreadable : undefined;

    /**
     * Checks if `value` is a flattenable `arguments` object or array.
     *
     * @private
     * @param {*} value The value to check.
     * @returns {boolean} Returns `true` if `value` is flattenable, else `false`.
     */
    function isFlattenable(value) {
      return isArray_1(value) || isArguments_1(value) ||
        !!(spreadableSymbol && value && value[spreadableSymbol]);
    }

    var _isFlattenable = isFlattenable;

    /**
     * The base implementation of `_.flatten` with support for restricting flattening.
     *
     * @private
     * @param {Array} array The array to flatten.
     * @param {number} depth The maximum recursion depth.
     * @param {boolean} [predicate=isFlattenable] The function invoked per iteration.
     * @param {boolean} [isStrict] Restrict to values that pass `predicate` checks.
     * @param {Array} [result=[]] The initial result value.
     * @returns {Array} Returns the new flattened array.
     */
    function baseFlatten(array, depth, predicate, isStrict, result) {
      var index = -1,
          length = array.length;

      predicate || (predicate = _isFlattenable);
      result || (result = []);

      while (++index < length) {
        var value = array[index];
        if (depth > 0 && predicate(value)) {
          if (depth > 1) {
            // Recursively flatten arrays (susceptible to call stack limits).
            baseFlatten(value, depth - 1, predicate, isStrict, result);
          } else {
            _arrayPush(result, value);
          }
        } else if (!isStrict) {
          result[result.length] = value;
        }
      }
      return result;
    }

    var _baseFlatten = baseFlatten;

    /**
     * A faster alternative to `Function#apply`, this function invokes `func`
     * with the `this` binding of `thisArg` and the arguments of `args`.
     *
     * @private
     * @param {Function} func The function to invoke.
     * @param {*} thisArg The `this` binding of `func`.
     * @param {Array} args The arguments to invoke `func` with.
     * @returns {*} Returns the result of `func`.
     */
    function apply(func, thisArg, args) {
      switch (args.length) {
        case 0: return func.call(thisArg);
        case 1: return func.call(thisArg, args[0]);
        case 2: return func.call(thisArg, args[0], args[1]);
        case 3: return func.call(thisArg, args[0], args[1], args[2]);
      }
      return func.apply(thisArg, args);
    }

    var _apply = apply;

    /* Built-in method references for those with the same name as other `lodash` methods. */
    var nativeMax = Math.max;

    /**
     * A specialized version of `baseRest` which transforms the rest array.
     *
     * @private
     * @param {Function} func The function to apply a rest parameter to.
     * @param {number} [start=func.length-1] The start position of the rest parameter.
     * @param {Function} transform The rest array transform.
     * @returns {Function} Returns the new function.
     */
    function overRest(func, start, transform) {
      start = nativeMax(start === undefined ? (func.length - 1) : start, 0);
      return function() {
        var args = arguments,
            index = -1,
            length = nativeMax(args.length - start, 0),
            array = Array(length);

        while (++index < length) {
          array[index] = args[start + index];
        }
        index = -1;
        var otherArgs = Array(start + 1);
        while (++index < start) {
          otherArgs[index] = args[index];
        }
        otherArgs[start] = transform(array);
        return _apply(func, this, otherArgs);
      };
    }

    var _overRest = overRest;

    /**
     * The base implementation of `setToString` without support for hot loop shorting.
     *
     * @private
     * @param {Function} func The function to modify.
     * @param {Function} string The `toString` result.
     * @returns {Function} Returns `func`.
     */
    var baseSetToString = !_defineProperty ? identity_1 : function(func, string) {
      return _defineProperty(func, 'toString', {
        'configurable': true,
        'enumerable': false,
        'value': constant_1(string),
        'writable': true
      });
    };

    var _baseSetToString = baseSetToString;

    /** Used to detect hot functions by number of calls within a span of milliseconds. */
    var HOT_COUNT = 800,
        HOT_SPAN = 16;

    /* Built-in method references for those with the same name as other `lodash` methods. */
    var nativeNow = Date.now;

    /**
     * Creates a function that'll short out and invoke `identity` instead
     * of `func` when it's called `HOT_COUNT` or more times in `HOT_SPAN`
     * milliseconds.
     *
     * @private
     * @param {Function} func The function to restrict.
     * @returns {Function} Returns the new shortable function.
     */
    function shortOut(func) {
      var count = 0,
          lastCalled = 0;

      return function() {
        var stamp = nativeNow(),
            remaining = HOT_SPAN - (stamp - lastCalled);

        lastCalled = stamp;
        if (remaining > 0) {
          if (++count >= HOT_COUNT) {
            return arguments[0];
          }
        } else {
          count = 0;
        }
        return func.apply(undefined, arguments);
      };
    }

    var _shortOut = shortOut;

    /**
     * Sets the `toString` method of `func` to return `string`.
     *
     * @private
     * @param {Function} func The function to modify.
     * @param {Function} string The `toString` result.
     * @returns {Function} Returns `func`.
     */
    var setToString = _shortOut(_baseSetToString);

    var _setToString = setToString;

    /**
     * The base implementation of `_.rest` which doesn't validate or coerce arguments.
     *
     * @private
     * @param {Function} func The function to apply a rest parameter to.
     * @param {number} [start=func.length-1] The start position of the rest parameter.
     * @returns {Function} Returns the new function.
     */
    function baseRest(func, start) {
      return _setToString(_overRest(func, start, identity_1), func + '');
    }

    var _baseRest = baseRest;

    /**
     * The base implementation of `_.findIndex` and `_.findLastIndex` without
     * support for iteratee shorthands.
     *
     * @private
     * @param {Array} array The array to inspect.
     * @param {Function} predicate The function invoked per iteration.
     * @param {number} fromIndex The index to search from.
     * @param {boolean} [fromRight] Specify iterating from right to left.
     * @returns {number} Returns the index of the matched value, else `-1`.
     */
    function baseFindIndex(array, predicate, fromIndex, fromRight) {
      var length = array.length,
          index = fromIndex + (fromRight ? 1 : -1);

      while ((fromRight ? index-- : ++index < length)) {
        if (predicate(array[index], index, array)) {
          return index;
        }
      }
      return -1;
    }

    var _baseFindIndex = baseFindIndex;

    /**
     * The base implementation of `_.isNaN` without support for number objects.
     *
     * @private
     * @param {*} value The value to check.
     * @returns {boolean} Returns `true` if `value` is `NaN`, else `false`.
     */
    function baseIsNaN(value) {
      return value !== value;
    }

    var _baseIsNaN = baseIsNaN;

    /**
     * A specialized version of `_.indexOf` which performs strict equality
     * comparisons of values, i.e. `===`.
     *
     * @private
     * @param {Array} array The array to inspect.
     * @param {*} value The value to search for.
     * @param {number} fromIndex The index to search from.
     * @returns {number} Returns the index of the matched value, else `-1`.
     */
    function strictIndexOf(array, value, fromIndex) {
      var index = fromIndex - 1,
          length = array.length;

      while (++index < length) {
        if (array[index] === value) {
          return index;
        }
      }
      return -1;
    }

    var _strictIndexOf = strictIndexOf;

    /**
     * The base implementation of `_.indexOf` without `fromIndex` bounds checks.
     *
     * @private
     * @param {Array} array The array to inspect.
     * @param {*} value The value to search for.
     * @param {number} fromIndex The index to search from.
     * @returns {number} Returns the index of the matched value, else `-1`.
     */
    function baseIndexOf(array, value, fromIndex) {
      return value === value
        ? _strictIndexOf(array, value, fromIndex)
        : _baseFindIndex(array, _baseIsNaN, fromIndex);
    }

    var _baseIndexOf = baseIndexOf;

    /**
     * A specialized version of `_.includes` for arrays without support for
     * specifying an index to search from.
     *
     * @private
     * @param {Array} [array] The array to inspect.
     * @param {*} target The value to search for.
     * @returns {boolean} Returns `true` if `target` is found, else `false`.
     */
    function arrayIncludes(array, value) {
      var length = array == null ? 0 : array.length;
      return !!length && _baseIndexOf(array, value, 0) > -1;
    }

    var _arrayIncludes = arrayIncludes;

    /**
     * This function is like `arrayIncludes` except that it accepts a comparator.
     *
     * @private
     * @param {Array} [array] The array to inspect.
     * @param {*} target The value to search for.
     * @param {Function} comparator The comparator invoked per element.
     * @returns {boolean} Returns `true` if `target` is found, else `false`.
     */
    function arrayIncludesWith(array, value, comparator) {
      var index = -1,
          length = array == null ? 0 : array.length;

      while (++index < length) {
        if (comparator(value, array[index])) {
          return true;
        }
      }
      return false;
    }

    var _arrayIncludesWith = arrayIncludesWith;

    /**
     * This method returns `undefined`.
     *
     * @static
     * @memberOf _
     * @since 2.3.0
     * @category Util
     * @example
     *
     * _.times(2, _.noop);
     * // => [undefined, undefined]
     */
    function noop$c() {
      // No operation performed.
    }

    var noop_1 = noop$c;

    /** Used as references for various `Number` constants. */
    var INFINITY$2 = 1 / 0;

    /**
     * Creates a set object of `values`.
     *
     * @private
     * @param {Array} values The values to add to the set.
     * @returns {Object} Returns the new set.
     */
    var createSet = !(_Set && (1 / _setToArray(new _Set([,-0]))[1]) == INFINITY$2) ? noop_1 : function(values) {
      return new _Set(values);
    };

    var _createSet = createSet;

    /** Used as the size to enable large array optimizations. */
    var LARGE_ARRAY_SIZE$1 = 200;

    /**
     * The base implementation of `_.uniqBy` without support for iteratee shorthands.
     *
     * @private
     * @param {Array} array The array to inspect.
     * @param {Function} [iteratee] The iteratee invoked per element.
     * @param {Function} [comparator] The comparator invoked per element.
     * @returns {Array} Returns the new duplicate free array.
     */
    function baseUniq(array, iteratee, comparator) {
      var index = -1,
          includes = _arrayIncludes,
          length = array.length,
          isCommon = true,
          result = [],
          seen = result;

      if (comparator) {
        isCommon = false;
        includes = _arrayIncludesWith;
      }
      else if (length >= LARGE_ARRAY_SIZE$1) {
        var set = iteratee ? null : _createSet(array);
        if (set) {
          return _setToArray(set);
        }
        isCommon = false;
        includes = _cacheHas;
        seen = new _SetCache;
      }
      else {
        seen = iteratee ? [] : result;
      }
      outer:
      while (++index < length) {
        var value = array[index],
            computed = iteratee ? iteratee(value) : value;

        value = (comparator || value !== 0) ? value : 0;
        if (isCommon && computed === computed) {
          var seenIndex = seen.length;
          while (seenIndex--) {
            if (seen[seenIndex] === computed) {
              continue outer;
            }
          }
          if (iteratee) {
            seen.push(computed);
          }
          result.push(value);
        }
        else if (!includes(seen, computed, comparator)) {
          if (seen !== result) {
            seen.push(computed);
          }
          result.push(value);
        }
      }
      return result;
    }

    var _baseUniq = baseUniq;

    /**
     * This method is like `_.isArrayLike` except that it also checks if `value`
     * is an object.
     *
     * @static
     * @memberOf _
     * @since 4.0.0
     * @category Lang
     * @param {*} value The value to check.
     * @returns {boolean} Returns `true` if `value` is an array-like object,
     *  else `false`.
     * @example
     *
     * _.isArrayLikeObject([1, 2, 3]);
     * // => true
     *
     * _.isArrayLikeObject(document.body.children);
     * // => true
     *
     * _.isArrayLikeObject('abc');
     * // => false
     *
     * _.isArrayLikeObject(_.noop);
     * // => false
     */
    function isArrayLikeObject(value) {
      return isObjectLike_1(value) && isArrayLike_1(value);
    }

    var isArrayLikeObject_1 = isArrayLikeObject;

    /**
     * Creates an array of unique values, in order, from all given arrays using
     * [`SameValueZero`](http://ecma-international.org/ecma-262/7.0/#sec-samevaluezero)
     * for equality comparisons.
     *
     * @static
     * @memberOf _
     * @since 0.1.0
     * @category Array
     * @param {...Array} [arrays] The arrays to inspect.
     * @returns {Array} Returns the new array of combined values.
     * @example
     *
     * _.union([2], [1, 2]);
     * // => [2, 1]
     */
    var union = _baseRest(function(arrays) {
      return _baseUniq(_baseFlatten(arrays, 1, isArrayLikeObject_1, true));
    });

    var union_1 = union;

    /**
     * The base implementation of `_.values` and `_.valuesIn` which creates an
     * array of `object` property values corresponding to the property names
     * of `props`.
     *
     * @private
     * @param {Object} object The object to query.
     * @param {Array} props The property names to get values for.
     * @returns {Object} Returns the array of property values.
     */
    function baseValues(object, props) {
      return _arrayMap(props, function(key) {
        return object[key];
      });
    }

    var _baseValues = baseValues;

    /**
     * Creates an array of the own enumerable string keyed property values of `object`.
     *
     * **Note:** Non-object values are coerced to objects.
     *
     * @static
     * @since 0.1.0
     * @memberOf _
     * @category Object
     * @param {Object} object The object to query.
     * @returns {Array} Returns the array of property values.
     * @example
     *
     * function Foo() {
     *   this.a = 1;
     *   this.b = 2;
     * }
     *
     * Foo.prototype.c = 3;
     *
     * _.values(new Foo);
     * // => [1, 2] (iteration order is not guaranteed)
     *
     * _.values('hi');
     * // => ['h', 'i']
     */
    function values$2(object) {
      return object == null ? [] : _baseValues(object, keys_1(object));
    }

    var values_1 = values$2;

    /* global window */

    var lodash$1;

    if (typeof commonjsRequire === "function") {
      try {
        lodash$1 = {
          clone: clone_1,
          constant: constant_1,
          each: each,
          filter: filter_1,
          has:  has_1,
          isArray: isArray_1,
          isEmpty: isEmpty_1,
          isFunction: isFunction_1,
          isUndefined: isUndefined_1,
          keys: keys_1,
          map: map_1,
          reduce: reduce_1,
          size: size_1,
          transform: transform_1,
          union: union_1,
          values: values_1
        };
      } catch (e) {
        // continue regardless of error
      }
    }

    if (!lodash$1) {
      lodash$1 = window._;
    }

    var lodash_1 = lodash$1;

    var graph = Graph;

    var DEFAULT_EDGE_NAME = "\x00";
    var GRAPH_NODE = "\x00";
    var EDGE_KEY_DELIM = "\x01";

    // Implementation notes:
    //
    //  * Node id query functions should return string ids for the nodes
    //  * Edge id query functions should return an "edgeObj", edge object, that is
    //    composed of enough information to uniquely identify an edge: {v, w, name}.
    //  * Internally we use an "edgeId", a stringified form of the edgeObj, to
    //    reference edges. This is because we need a performant way to look these
    //    edges up and, object properties, which have string keys, are the closest
    //    we're going to get to a performant hashtable in JavaScript.

    function Graph(opts) {
      this._isDirected = lodash_1.has(opts, "directed") ? opts.directed : true;
      this._isMultigraph = lodash_1.has(opts, "multigraph") ? opts.multigraph : false;
      this._isCompound = lodash_1.has(opts, "compound") ? opts.compound : false;

      // Label for the graph itself
      this._label = undefined;

      // Defaults to be set when creating a new node
      this._defaultNodeLabelFn = lodash_1.constant(undefined);

      // Defaults to be set when creating a new edge
      this._defaultEdgeLabelFn = lodash_1.constant(undefined);

      // v -> label
      this._nodes = {};

      if (this._isCompound) {
        // v -> parent
        this._parent = {};

        // v -> children
        this._children = {};
        this._children[GRAPH_NODE] = {};
      }

      // v -> edgeObj
      this._in = {};

      // u -> v -> Number
      this._preds = {};

      // v -> edgeObj
      this._out = {};

      // v -> w -> Number
      this._sucs = {};

      // e -> edgeObj
      this._edgeObjs = {};

      // e -> label
      this._edgeLabels = {};
    }

    /* Number of nodes in the graph. Should only be changed by the implementation. */
    Graph.prototype._nodeCount = 0;

    /* Number of edges in the graph. Should only be changed by the implementation. */
    Graph.prototype._edgeCount = 0;


    /* === Graph functions ========= */

    Graph.prototype.isDirected = function() {
      return this._isDirected;
    };

    Graph.prototype.isMultigraph = function() {
      return this._isMultigraph;
    };

    Graph.prototype.isCompound = function() {
      return this._isCompound;
    };

    Graph.prototype.setGraph = function(label) {
      this._label = label;
      return this;
    };

    Graph.prototype.graph = function() {
      return this._label;
    };


    /* === Node functions ========== */

    Graph.prototype.setDefaultNodeLabel = function(newDefault) {
      if (!lodash_1.isFunction(newDefault)) {
        newDefault = lodash_1.constant(newDefault);
      }
      this._defaultNodeLabelFn = newDefault;
      return this;
    };

    Graph.prototype.nodeCount = function() {
      return this._nodeCount;
    };

    Graph.prototype.nodes = function() {
      return lodash_1.keys(this._nodes);
    };

    Graph.prototype.sources = function() {
      var self = this;
      return lodash_1.filter(this.nodes(), function(v) {
        return lodash_1.isEmpty(self._in[v]);
      });
    };

    Graph.prototype.sinks = function() {
      var self = this;
      return lodash_1.filter(this.nodes(), function(v) {
        return lodash_1.isEmpty(self._out[v]);
      });
    };

    Graph.prototype.setNodes = function(vs, value) {
      var args = arguments;
      var self = this;
      lodash_1.each(vs, function(v) {
        if (args.length > 1) {
          self.setNode(v, value);
        } else {
          self.setNode(v);
        }
      });
      return this;
    };

    Graph.prototype.setNode = function(v, value) {
      if (lodash_1.has(this._nodes, v)) {
        if (arguments.length > 1) {
          this._nodes[v] = value;
        }
        return this;
      }

      this._nodes[v] = arguments.length > 1 ? value : this._defaultNodeLabelFn(v);
      if (this._isCompound) {
        this._parent[v] = GRAPH_NODE;
        this._children[v] = {};
        this._children[GRAPH_NODE][v] = true;
      }
      this._in[v] = {};
      this._preds[v] = {};
      this._out[v] = {};
      this._sucs[v] = {};
      ++this._nodeCount;
      return this;
    };

    Graph.prototype.node = function(v) {
      return this._nodes[v];
    };

    Graph.prototype.hasNode = function(v) {
      return lodash_1.has(this._nodes, v);
    };

    Graph.prototype.removeNode =  function(v) {
      var self = this;
      if (lodash_1.has(this._nodes, v)) {
        var removeEdge = function(e) { self.removeEdge(self._edgeObjs[e]); };
        delete this._nodes[v];
        if (this._isCompound) {
          this._removeFromParentsChildList(v);
          delete this._parent[v];
          lodash_1.each(this.children(v), function(child) {
            self.setParent(child);
          });
          delete this._children[v];
        }
        lodash_1.each(lodash_1.keys(this._in[v]), removeEdge);
        delete this._in[v];
        delete this._preds[v];
        lodash_1.each(lodash_1.keys(this._out[v]), removeEdge);
        delete this._out[v];
        delete this._sucs[v];
        --this._nodeCount;
      }
      return this;
    };

    Graph.prototype.setParent = function(v, parent) {
      if (!this._isCompound) {
        throw new Error("Cannot set parent in a non-compound graph");
      }

      if (lodash_1.isUndefined(parent)) {
        parent = GRAPH_NODE;
      } else {
        // Coerce parent to string
        parent += "";
        for (var ancestor = parent;
          !lodash_1.isUndefined(ancestor);
          ancestor = this.parent(ancestor)) {
          if (ancestor === v) {
            throw new Error("Setting " + parent+ " as parent of " + v +
                            " would create a cycle");
          }
        }

        this.setNode(parent);
      }

      this.setNode(v);
      this._removeFromParentsChildList(v);
      this._parent[v] = parent;
      this._children[parent][v] = true;
      return this;
    };

    Graph.prototype._removeFromParentsChildList = function(v) {
      delete this._children[this._parent[v]][v];
    };

    Graph.prototype.parent = function(v) {
      if (this._isCompound) {
        var parent = this._parent[v];
        if (parent !== GRAPH_NODE) {
          return parent;
        }
      }
    };

    Graph.prototype.children = function(v) {
      if (lodash_1.isUndefined(v)) {
        v = GRAPH_NODE;
      }

      if (this._isCompound) {
        var children = this._children[v];
        if (children) {
          return lodash_1.keys(children);
        }
      } else if (v === GRAPH_NODE) {
        return this.nodes();
      } else if (this.hasNode(v)) {
        return [];
      }
    };

    Graph.prototype.predecessors = function(v) {
      var predsV = this._preds[v];
      if (predsV) {
        return lodash_1.keys(predsV);
      }
    };

    Graph.prototype.successors = function(v) {
      var sucsV = this._sucs[v];
      if (sucsV) {
        return lodash_1.keys(sucsV);
      }
    };

    Graph.prototype.neighbors = function(v) {
      var preds = this.predecessors(v);
      if (preds) {
        return lodash_1.union(preds, this.successors(v));
      }
    };

    Graph.prototype.isLeaf = function (v) {
      var neighbors;
      if (this.isDirected()) {
        neighbors = this.successors(v);
      } else {
        neighbors = this.neighbors(v);
      }
      return neighbors.length === 0;
    };

    Graph.prototype.filterNodes = function(filter) {
      var copy = new this.constructor({
        directed: this._isDirected,
        multigraph: this._isMultigraph,
        compound: this._isCompound
      });

      copy.setGraph(this.graph());

      var self = this;
      lodash_1.each(this._nodes, function(value, v) {
        if (filter(v)) {
          copy.setNode(v, value);
        }
      });

      lodash_1.each(this._edgeObjs, function(e) {
        if (copy.hasNode(e.v) && copy.hasNode(e.w)) {
          copy.setEdge(e, self.edge(e));
        }
      });

      var parents = {};
      function findParent(v) {
        var parent = self.parent(v);
        if (parent === undefined || copy.hasNode(parent)) {
          parents[v] = parent;
          return parent;
        } else if (parent in parents) {
          return parents[parent];
        } else {
          return findParent(parent);
        }
      }

      if (this._isCompound) {
        lodash_1.each(copy.nodes(), function(v) {
          copy.setParent(v, findParent(v));
        });
      }

      return copy;
    };

    /* === Edge functions ========== */

    Graph.prototype.setDefaultEdgeLabel = function(newDefault) {
      if (!lodash_1.isFunction(newDefault)) {
        newDefault = lodash_1.constant(newDefault);
      }
      this._defaultEdgeLabelFn = newDefault;
      return this;
    };

    Graph.prototype.edgeCount = function() {
      return this._edgeCount;
    };

    Graph.prototype.edges = function() {
      return lodash_1.values(this._edgeObjs);
    };

    Graph.prototype.setPath = function(vs, value) {
      var self = this;
      var args = arguments;
      lodash_1.reduce(vs, function(v, w) {
        if (args.length > 1) {
          self.setEdge(v, w, value);
        } else {
          self.setEdge(v, w);
        }
        return w;
      });
      return this;
    };

    /*
     * setEdge(v, w, [value, [name]])
     * setEdge({ v, w, [name] }, [value])
     */
    Graph.prototype.setEdge = function() {
      var v, w, name, value;
      var valueSpecified = false;
      var arg0 = arguments[0];

      if (typeof arg0 === "object" && arg0 !== null && "v" in arg0) {
        v = arg0.v;
        w = arg0.w;
        name = arg0.name;
        if (arguments.length === 2) {
          value = arguments[1];
          valueSpecified = true;
        }
      } else {
        v = arg0;
        w = arguments[1];
        name = arguments[3];
        if (arguments.length > 2) {
          value = arguments[2];
          valueSpecified = true;
        }
      }

      v = "" + v;
      w = "" + w;
      if (!lodash_1.isUndefined(name)) {
        name = "" + name;
      }

      var e = edgeArgsToId(this._isDirected, v, w, name);
      if (lodash_1.has(this._edgeLabels, e)) {
        if (valueSpecified) {
          this._edgeLabels[e] = value;
        }
        return this;
      }

      if (!lodash_1.isUndefined(name) && !this._isMultigraph) {
        throw new Error("Cannot set a named edge when isMultigraph = false");
      }

      // It didn't exist, so we need to create it.
      // First ensure the nodes exist.
      this.setNode(v);
      this.setNode(w);

      this._edgeLabels[e] = valueSpecified ? value : this._defaultEdgeLabelFn(v, w, name);

      var edgeObj = edgeArgsToObj(this._isDirected, v, w, name);
      // Ensure we add undirected edges in a consistent way.
      v = edgeObj.v;
      w = edgeObj.w;

      Object.freeze(edgeObj);
      this._edgeObjs[e] = edgeObj;
      incrementOrInitEntry(this._preds[w], v);
      incrementOrInitEntry(this._sucs[v], w);
      this._in[w][e] = edgeObj;
      this._out[v][e] = edgeObj;
      this._edgeCount++;
      return this;
    };

    Graph.prototype.edge = function(v, w, name) {
      var e = (arguments.length === 1
        ? edgeObjToId(this._isDirected, arguments[0])
        : edgeArgsToId(this._isDirected, v, w, name));
      return this._edgeLabels[e];
    };

    Graph.prototype.hasEdge = function(v, w, name) {
      var e = (arguments.length === 1
        ? edgeObjToId(this._isDirected, arguments[0])
        : edgeArgsToId(this._isDirected, v, w, name));
      return lodash_1.has(this._edgeLabels, e);
    };

    Graph.prototype.removeEdge = function(v, w, name) {
      var e = (arguments.length === 1
        ? edgeObjToId(this._isDirected, arguments[0])
        : edgeArgsToId(this._isDirected, v, w, name));
      var edge = this._edgeObjs[e];
      if (edge) {
        v = edge.v;
        w = edge.w;
        delete this._edgeLabels[e];
        delete this._edgeObjs[e];
        decrementOrRemoveEntry(this._preds[w], v);
        decrementOrRemoveEntry(this._sucs[v], w);
        delete this._in[w][e];
        delete this._out[v][e];
        this._edgeCount--;
      }
      return this;
    };

    Graph.prototype.inEdges = function(v, u) {
      var inV = this._in[v];
      if (inV) {
        var edges = lodash_1.values(inV);
        if (!u) {
          return edges;
        }
        return lodash_1.filter(edges, function(edge) { return edge.v === u; });
      }
    };

    Graph.prototype.outEdges = function(v, w) {
      var outV = this._out[v];
      if (outV) {
        var edges = lodash_1.values(outV);
        if (!w) {
          return edges;
        }
        return lodash_1.filter(edges, function(edge) { return edge.w === w; });
      }
    };

    Graph.prototype.nodeEdges = function(v, w) {
      var inEdges = this.inEdges(v, w);
      if (inEdges) {
        return inEdges.concat(this.outEdges(v, w));
      }
    };

    function incrementOrInitEntry(map, k) {
      if (map[k]) {
        map[k]++;
      } else {
        map[k] = 1;
      }
    }

    function decrementOrRemoveEntry(map, k) {
      if (!--map[k]) { delete map[k]; }
    }

    function edgeArgsToId(isDirected, v_, w_, name) {
      var v = "" + v_;
      var w = "" + w_;
      if (!isDirected && v > w) {
        var tmp = v;
        v = w;
        w = tmp;
      }
      return v + EDGE_KEY_DELIM + w + EDGE_KEY_DELIM +
                 (lodash_1.isUndefined(name) ? DEFAULT_EDGE_NAME : name);
    }

    function edgeArgsToObj(isDirected, v_, w_, name) {
      var v = "" + v_;
      var w = "" + w_;
      if (!isDirected && v > w) {
        var tmp = v;
        v = w;
        w = tmp;
      }
      var edgeObj =  { v: v, w: w };
      if (name) {
        edgeObj.name = name;
      }
      return edgeObj;
    }

    function edgeObjToId(isDirected, edgeObj) {
      return edgeArgsToId(isDirected, edgeObj.v, edgeObj.w, edgeObj.name);
    }

    var version$4 = '2.1.8';

    // Includes only the "core" of graphlib
    var lib = {
      Graph: graph,
      version: version$4
    };

    var json$2 = {
      write: write,
      read: read
    };

    function write(g) {
      var json = {
        options: {
          directed: g.isDirected(),
          multigraph: g.isMultigraph(),
          compound: g.isCompound()
        },
        nodes: writeNodes(g),
        edges: writeEdges(g)
      };
      if (!lodash_1.isUndefined(g.graph())) {
        json.value = lodash_1.clone(g.graph());
      }
      return json;
    }

    function writeNodes(g) {
      return lodash_1.map(g.nodes(), function(v) {
        var nodeValue = g.node(v);
        var parent = g.parent(v);
        var node = { v: v };
        if (!lodash_1.isUndefined(nodeValue)) {
          node.value = nodeValue;
        }
        if (!lodash_1.isUndefined(parent)) {
          node.parent = parent;
        }
        return node;
      });
    }

    function writeEdges(g) {
      return lodash_1.map(g.edges(), function(e) {
        var edgeValue = g.edge(e);
        var edge = { v: e.v, w: e.w };
        if (!lodash_1.isUndefined(e.name)) {
          edge.name = e.name;
        }
        if (!lodash_1.isUndefined(edgeValue)) {
          edge.value = edgeValue;
        }
        return edge;
      });
    }

    function read(json) {
      var g = new graph(json.options).setGraph(json.value);
      lodash_1.each(json.nodes, function(entry) {
        g.setNode(entry.v, entry.value);
        if (entry.parent) {
          g.setParent(entry.v, entry.parent);
        }
      });
      lodash_1.each(json.edges, function(entry) {
        g.setEdge({ v: entry.v, w: entry.w, name: entry.name }, entry.value);
      });
      return g;
    }

    var components_1 = components$1;

    function components$1(g) {
      var visited = {};
      var cmpts = [];
      var cmpt;

      function dfs(v) {
        if (lodash_1.has(visited, v)) return;
        visited[v] = true;
        cmpt.push(v);
        lodash_1.each(g.successors(v), dfs);
        lodash_1.each(g.predecessors(v), dfs);
      }

      lodash_1.each(g.nodes(), function(v) {
        cmpt = [];
        dfs(v);
        if (cmpt.length) {
          cmpts.push(cmpt);
        }
      });

      return cmpts;
    }

    var priorityQueue = PriorityQueue;

    /**
     * A min-priority queue data structure. This algorithm is derived from Cormen,
     * et al., "Introduction to Algorithms". The basic idea of a min-priority
     * queue is that you can efficiently (in O(1) time) get the smallest key in
     * the queue. Adding and removing elements takes O(log n) time. A key can
     * have its priority decreased in O(log n) time.
     */
    function PriorityQueue() {
      this._arr = [];
      this._keyIndices = {};
    }

    /**
     * Returns the number of elements in the queue. Takes `O(1)` time.
     */
    PriorityQueue.prototype.size = function() {
      return this._arr.length;
    };

    /**
     * Returns the keys that are in the queue. Takes `O(n)` time.
     */
    PriorityQueue.prototype.keys = function() {
      return this._arr.map(function(x) { return x.key; });
    };

    /**
     * Returns `true` if **key** is in the queue and `false` if not.
     */
    PriorityQueue.prototype.has = function(key) {
      return lodash_1.has(this._keyIndices, key);
    };

    /**
     * Returns the priority for **key**. If **key** is not present in the queue
     * then this function returns `undefined`. Takes `O(1)` time.
     *
     * @param {Object} key
     */
    PriorityQueue.prototype.priority = function(key) {
      var index = this._keyIndices[key];
      if (index !== undefined) {
        return this._arr[index].priority;
      }
    };

    /**
     * Returns the key for the minimum element in this queue. If the queue is
     * empty this function throws an Error. Takes `O(1)` time.
     */
    PriorityQueue.prototype.min = function() {
      if (this.size() === 0) {
        throw new Error("Queue underflow");
      }
      return this._arr[0].key;
    };

    /**
     * Inserts a new key into the priority queue. If the key already exists in
     * the queue this function returns `false`; otherwise it will return `true`.
     * Takes `O(n)` time.
     *
     * @param {Object} key the key to add
     * @param {Number} priority the initial priority for the key
     */
    PriorityQueue.prototype.add = function(key, priority) {
      var keyIndices = this._keyIndices;
      key = String(key);
      if (!lodash_1.has(keyIndices, key)) {
        var arr = this._arr;
        var index = arr.length;
        keyIndices[key] = index;
        arr.push({key: key, priority: priority});
        this._decrease(index);
        return true;
      }
      return false;
    };

    /**
     * Removes and returns the smallest key in the queue. Takes `O(log n)` time.
     */
    PriorityQueue.prototype.removeMin = function() {
      this._swap(0, this._arr.length - 1);
      var min = this._arr.pop();
      delete this._keyIndices[min.key];
      this._heapify(0);
      return min.key;
    };

    /**
     * Decreases the priority for **key** to **priority**. If the new priority is
     * greater than the previous priority, this function will throw an Error.
     *
     * @param {Object} key the key for which to raise priority
     * @param {Number} priority the new priority for the key
     */
    PriorityQueue.prototype.decrease = function(key, priority) {
      var index = this._keyIndices[key];
      if (priority > this._arr[index].priority) {
        throw new Error("New priority is greater than current priority. " +
            "Key: " + key + " Old: " + this._arr[index].priority + " New: " + priority);
      }
      this._arr[index].priority = priority;
      this._decrease(index);
    };

    PriorityQueue.prototype._heapify = function(i) {
      var arr = this._arr;
      var l = 2 * i;
      var r = l + 1;
      var largest = i;
      if (l < arr.length) {
        largest = arr[l].priority < arr[largest].priority ? l : largest;
        if (r < arr.length) {
          largest = arr[r].priority < arr[largest].priority ? r : largest;
        }
        if (largest !== i) {
          this._swap(i, largest);
          this._heapify(largest);
        }
      }
    };

    PriorityQueue.prototype._decrease = function(index) {
      var arr = this._arr;
      var priority = arr[index].priority;
      var parent;
      while (index !== 0) {
        parent = index >> 1;
        if (arr[parent].priority < priority) {
          break;
        }
        this._swap(index, parent);
        index = parent;
      }
    };

    PriorityQueue.prototype._swap = function(i, j) {
      var arr = this._arr;
      var keyIndices = this._keyIndices;
      var origArrI = arr[i];
      var origArrJ = arr[j];
      arr[i] = origArrJ;
      arr[j] = origArrI;
      keyIndices[origArrJ.key] = i;
      keyIndices[origArrI.key] = j;
    };

    var dijkstra_1 = dijkstra;

    var DEFAULT_WEIGHT_FUNC = lodash_1.constant(1);

    function dijkstra(g, source, weightFn, edgeFn) {
      return runDijkstra(g, String(source),
        weightFn || DEFAULT_WEIGHT_FUNC,
        edgeFn || function(v) { return g.outEdges(v); });
    }

    function runDijkstra(g, source, weightFn, edgeFn) {
      var results = {};
      var pq = new priorityQueue();
      var v, vEntry;

      var updateNeighbors = function(edge) {
        var w = edge.v !== v ? edge.v : edge.w;
        var wEntry = results[w];
        var weight = weightFn(edge);
        var distance = vEntry.distance + weight;

        if (weight < 0) {
          throw new Error("dijkstra does not allow negative edge weights. " +
                          "Bad edge: " + edge + " Weight: " + weight);
        }

        if (distance < wEntry.distance) {
          wEntry.distance = distance;
          wEntry.predecessor = v;
          pq.decrease(w, distance);
        }
      };

      g.nodes().forEach(function(v) {
        var distance = v === source ? 0 : Number.POSITIVE_INFINITY;
        results[v] = { distance: distance };
        pq.add(v, distance);
      });

      while (pq.size() > 0) {
        v = pq.removeMin();
        vEntry = results[v];
        if (vEntry.distance === Number.POSITIVE_INFINITY) {
          break;
        }

        edgeFn(v).forEach(updateNeighbors);
      }

      return results;
    }

    var dijkstraAll_1 = dijkstraAll;

    function dijkstraAll(g, weightFunc, edgeFunc) {
      return lodash_1.transform(g.nodes(), function(acc, v) {
        acc[v] = dijkstra_1(g, v, weightFunc, edgeFunc);
      }, {});
    }

    var tarjan_1 = tarjan;

    function tarjan(g) {
      var index = 0;
      var stack = [];
      var visited = {}; // node id -> { onStack, lowlink, index }
      var results = [];

      function dfs(v) {
        var entry = visited[v] = {
          onStack: true,
          lowlink: index,
          index: index++
        };
        stack.push(v);

        g.successors(v).forEach(function(w) {
          if (!lodash_1.has(visited, w)) {
            dfs(w);
            entry.lowlink = Math.min(entry.lowlink, visited[w].lowlink);
          } else if (visited[w].onStack) {
            entry.lowlink = Math.min(entry.lowlink, visited[w].index);
          }
        });

        if (entry.lowlink === entry.index) {
          var cmpt = [];
          var w;
          do {
            w = stack.pop();
            visited[w].onStack = false;
            cmpt.push(w);
          } while (v !== w);
          results.push(cmpt);
        }
      }

      g.nodes().forEach(function(v) {
        if (!lodash_1.has(visited, v)) {
          dfs(v);
        }
      });

      return results;
    }

    var findCycles_1 = findCycles;

    function findCycles(g) {
      return lodash_1.filter(tarjan_1(g), function(cmpt) {
        return cmpt.length > 1 || (cmpt.length === 1 && g.hasEdge(cmpt[0], cmpt[0]));
      });
    }

    var floydWarshall_1 = floydWarshall;

    var DEFAULT_WEIGHT_FUNC$1 = lodash_1.constant(1);

    function floydWarshall(g, weightFn, edgeFn) {
      return runFloydWarshall(g,
        weightFn || DEFAULT_WEIGHT_FUNC$1,
        edgeFn || function(v) { return g.outEdges(v); });
    }

    function runFloydWarshall(g, weightFn, edgeFn) {
      var results = {};
      var nodes = g.nodes();

      nodes.forEach(function(v) {
        results[v] = {};
        results[v][v] = { distance: 0 };
        nodes.forEach(function(w) {
          if (v !== w) {
            results[v][w] = { distance: Number.POSITIVE_INFINITY };
          }
        });
        edgeFn(v).forEach(function(edge) {
          var w = edge.v === v ? edge.w : edge.v;
          var d = weightFn(edge);
          results[v][w] = { distance: d, predecessor: v };
        });
      });

      nodes.forEach(function(k) {
        var rowK = results[k];
        nodes.forEach(function(i) {
          var rowI = results[i];
          nodes.forEach(function(j) {
            var ik = rowI[k];
            var kj = rowK[j];
            var ij = rowI[j];
            var altDistance = ik.distance + kj.distance;
            if (altDistance < ij.distance) {
              ij.distance = altDistance;
              ij.predecessor = kj.predecessor;
            }
          });
        });
      });

      return results;
    }

    var topsort_1 = topsort;
    topsort.CycleException = CycleException;

    function topsort(g) {
      var visited = {};
      var stack = {};
      var results = [];

      function visit(node) {
        if (lodash_1.has(stack, node)) {
          throw new CycleException();
        }

        if (!lodash_1.has(visited, node)) {
          stack[node] = true;
          visited[node] = true;
          lodash_1.each(g.predecessors(node), visit);
          delete stack[node];
          results.push(node);
        }
      }

      lodash_1.each(g.sinks(), visit);

      if (lodash_1.size(visited) !== g.nodeCount()) {
        throw new CycleException();
      }

      return results;
    }

    function CycleException() {}
    CycleException.prototype = new Error(); // must be an instance of Error to pass testing

    var isAcyclic_1 = isAcyclic;

    function isAcyclic(g) {
      try {
        topsort_1(g);
      } catch (e) {
        if (e instanceof topsort_1.CycleException) {
          return false;
        }
        throw e;
      }
      return true;
    }

    var dfs_1 = dfs;

    /*
     * A helper that preforms a pre- or post-order traversal on the input graph
     * and returns the nodes in the order they were visited. If the graph is
     * undirected then this algorithm will navigate using neighbors. If the graph
     * is directed then this algorithm will navigate using successors.
     *
     * Order must be one of "pre" or "post".
     */
    function dfs(g, vs, order) {
      if (!lodash_1.isArray(vs)) {
        vs = [vs];
      }

      var navigation = (g.isDirected() ? g.successors : g.neighbors).bind(g);

      var acc = [];
      var visited = {};
      lodash_1.each(vs, function(v) {
        if (!g.hasNode(v)) {
          throw new Error("Graph does not have node: " + v);
        }

        doDfs(g, v, order === "post", visited, navigation, acc);
      });
      return acc;
    }

    function doDfs(g, v, postorder, visited, navigation, acc) {
      if (!lodash_1.has(visited, v)) {
        visited[v] = true;

        if (!postorder) { acc.push(v); }
        lodash_1.each(navigation(v), function(w) {
          doDfs(g, w, postorder, visited, navigation, acc);
        });
        if (postorder) { acc.push(v); }
      }
    }

    var postorder_1 = postorder;

    function postorder(g, vs) {
      return dfs_1(g, vs, "post");
    }

    var preorder_1 = preorder;

    function preorder(g, vs) {
      return dfs_1(g, vs, "pre");
    }

    var prim_1 = prim;

    function prim(g, weightFunc) {
      var result = new graph();
      var parents = {};
      var pq = new priorityQueue();
      var v;

      function updateNeighbors(edge) {
        var w = edge.v === v ? edge.w : edge.v;
        var pri = pq.priority(w);
        if (pri !== undefined) {
          var edgeWeight = weightFunc(edge);
          if (edgeWeight < pri) {
            parents[w] = v;
            pq.decrease(w, edgeWeight);
          }
        }
      }

      if (g.nodeCount() === 0) {
        return result;
      }

      lodash_1.each(g.nodes(), function(v) {
        pq.add(v, Number.POSITIVE_INFINITY);
        result.setNode(v);
      });

      // Start from an arbitrary node
      pq.decrease(g.nodes()[0], 0);

      var init = false;
      while (pq.size() > 0) {
        v = pq.removeMin();
        if (lodash_1.has(parents, v)) {
          result.setEdge(v, parents[v]);
        } else if (init) {
          throw new Error("Input graph is not connected: " + g);
        } else {
          init = true;
        }

        g.nodeEdges(v).forEach(updateNeighbors);
      }

      return result;
    }

    var alg = {
      components: components_1,
      dijkstra: dijkstra_1,
      dijkstraAll: dijkstraAll_1,
      findCycles: findCycles_1,
      floydWarshall: floydWarshall_1,
      isAcyclic: isAcyclic_1,
      postorder: postorder_1,
      preorder: preorder_1,
      prim: prim_1,
      tarjan: tarjan_1,
      topsort: topsort_1
    };

    /**
     * Copyright (c) 2014, Chris Pettitt
     * All rights reserved.
     *
     * Redistribution and use in source and binary forms, with or without
     * modification, are permitted provided that the following conditions are met:
     *
     * 1. Redistributions of source code must retain the above copyright notice, this
     * list of conditions and the following disclaimer.
     *
     * 2. Redistributions in binary form must reproduce the above copyright notice,
     * this list of conditions and the following disclaimer in the documentation
     * and/or other materials provided with the distribution.
     *
     * 3. Neither the name of the copyright holder nor the names of its contributors
     * may be used to endorse or promote products derived from this software without
     * specific prior written permission.
     *
     * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
     * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
     * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
     * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
     * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
     * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
     * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
     * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
     * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
     * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
     */



    var graphlib = {
      Graph: lib.Graph,
      json: json$2,
      alg: alg,
      version: lib.version
    };

    /* global window */

    var graphlib$1;

    if (typeof commonjsRequire === "function") {
      try {
        graphlib$1 = graphlib;
      } catch (e) {
        // continue regardless of error
      }
    }

    if (!graphlib$1) {
      graphlib$1 = window.graphlib;
    }

    var graphlib_1 = graphlib$1;

    /** Used to compose bitmasks for cloning. */
    var CLONE_DEEP_FLAG$1 = 1,
        CLONE_SYMBOLS_FLAG$2 = 4;

    /**
     * This method is like `_.clone` except that it recursively clones `value`.
     *
     * @static
     * @memberOf _
     * @since 1.0.0
     * @category Lang
     * @param {*} value The value to recursively clone.
     * @returns {*} Returns the deep cloned value.
     * @see _.clone
     * @example
     *
     * var objects = [{ 'a': 1 }, { 'b': 2 }];
     *
     * var deep = _.cloneDeep(objects);
     * console.log(deep[0] === objects[0]);
     * // => false
     */
    function cloneDeep(value) {
      return _baseClone(value, CLONE_DEEP_FLAG$1 | CLONE_SYMBOLS_FLAG$2);
    }

    var cloneDeep_1 = cloneDeep;

    /**
     * Checks if the given arguments are from an iteratee call.
     *
     * @private
     * @param {*} value The potential iteratee value argument.
     * @param {*} index The potential iteratee index or key argument.
     * @param {*} object The potential iteratee object argument.
     * @returns {boolean} Returns `true` if the arguments are from an iteratee call,
     *  else `false`.
     */
    function isIterateeCall(value, index, object) {
      if (!isObject_1(object)) {
        return false;
      }
      var type = typeof index;
      if (type == 'number'
            ? (isArrayLike_1(object) && _isIndex(index, object.length))
            : (type == 'string' && index in object)
          ) {
        return eq_1(object[index], value);
      }
      return false;
    }

    var _isIterateeCall = isIterateeCall;

    /** Used for built-in method references. */
    var objectProto$h = Object.prototype;

    /** Used to check objects for own properties. */
    var hasOwnProperty$e = objectProto$h.hasOwnProperty;

    /**
     * Assigns own and inherited enumerable string keyed properties of source
     * objects to the destination object for all destination properties that
     * resolve to `undefined`. Source objects are applied from left to right.
     * Once a property is set, additional values of the same property are ignored.
     *
     * **Note:** This method mutates `object`.
     *
     * @static
     * @since 0.1.0
     * @memberOf _
     * @category Object
     * @param {Object} object The destination object.
     * @param {...Object} [sources] The source objects.
     * @returns {Object} Returns `object`.
     * @see _.defaultsDeep
     * @example
     *
     * _.defaults({ 'a': 1 }, { 'b': 2 }, { 'a': 3 });
     * // => { 'a': 1, 'b': 2 }
     */
    var defaults = _baseRest(function(object, sources) {
      object = Object(object);

      var index = -1;
      var length = sources.length;
      var guard = length > 2 ? sources[2] : undefined;

      if (guard && _isIterateeCall(sources[0], sources[1], guard)) {
        length = 1;
      }

      while (++index < length) {
        var source = sources[index];
        var props = keysIn_1(source);
        var propsIndex = -1;
        var propsLength = props.length;

        while (++propsIndex < propsLength) {
          var key = props[propsIndex];
          var value = object[key];

          if (value === undefined ||
              (eq_1(value, objectProto$h[key]) && !hasOwnProperty$e.call(object, key))) {
            object[key] = source[key];
          }
        }
      }

      return object;
    });

    var defaults_1 = defaults;

    /**
     * Creates a `_.find` or `_.findLast` function.
     *
     * @private
     * @param {Function} findIndexFunc The function to find the collection index.
     * @returns {Function} Returns the new find function.
     */
    function createFind(findIndexFunc) {
      return function(collection, predicate, fromIndex) {
        var iterable = Object(collection);
        if (!isArrayLike_1(collection)) {
          var iteratee = _baseIteratee(predicate);
          collection = keys_1(collection);
          predicate = function(key) { return iteratee(iterable[key], key, iterable); };
        }
        var index = findIndexFunc(collection, predicate, fromIndex);
        return index > -1 ? iterable[iteratee ? collection[index] : index] : undefined;
      };
    }

    var _createFind = createFind;

    /** Used as references for various `Number` constants. */
    var NAN = 0 / 0;

    /** Used to match leading and trailing whitespace. */
    var reTrim = /^\s+|\s+$/g;

    /** Used to detect bad signed hexadecimal string values. */
    var reIsBadHex = /^[-+]0x[0-9a-f]+$/i;

    /** Used to detect binary string values. */
    var reIsBinary = /^0b[01]+$/i;

    /** Used to detect octal string values. */
    var reIsOctal = /^0o[0-7]+$/i;

    /** Built-in method references without a dependency on `root`. */
    var freeParseInt = parseInt;

    /**
     * Converts `value` to a number.
     *
     * @static
     * @memberOf _
     * @since 4.0.0
     * @category Lang
     * @param {*} value The value to process.
     * @returns {number} Returns the number.
     * @example
     *
     * _.toNumber(3.2);
     * // => 3.2
     *
     * _.toNumber(Number.MIN_VALUE);
     * // => 5e-324
     *
     * _.toNumber(Infinity);
     * // => Infinity
     *
     * _.toNumber('3.2');
     * // => 3.2
     */
    function toNumber(value) {
      if (typeof value == 'number') {
        return value;
      }
      if (isSymbol_1(value)) {
        return NAN;
      }
      if (isObject_1(value)) {
        var other = typeof value.valueOf == 'function' ? value.valueOf() : value;
        value = isObject_1(other) ? (other + '') : other;
      }
      if (typeof value != 'string') {
        return value === 0 ? value : +value;
      }
      value = value.replace(reTrim, '');
      var isBinary = reIsBinary.test(value);
      return (isBinary || reIsOctal.test(value))
        ? freeParseInt(value.slice(2), isBinary ? 2 : 8)
        : (reIsBadHex.test(value) ? NAN : +value);
    }

    var toNumber_1 = toNumber;

    /** Used as references for various `Number` constants. */
    var INFINITY$3 = 1 / 0,
        MAX_INTEGER = 1.7976931348623157e+308;

    /**
     * Converts `value` to a finite number.
     *
     * @static
     * @memberOf _
     * @since 4.12.0
     * @category Lang
     * @param {*} value The value to convert.
     * @returns {number} Returns the converted number.
     * @example
     *
     * _.toFinite(3.2);
     * // => 3.2
     *
     * _.toFinite(Number.MIN_VALUE);
     * // => 5e-324
     *
     * _.toFinite(Infinity);
     * // => 1.7976931348623157e+308
     *
     * _.toFinite('3.2');
     * // => 3.2
     */
    function toFinite(value) {
      if (!value) {
        return value === 0 ? value : 0;
      }
      value = toNumber_1(value);
      if (value === INFINITY$3 || value === -INFINITY$3) {
        var sign = (value < 0 ? -1 : 1);
        return sign * MAX_INTEGER;
      }
      return value === value ? value : 0;
    }

    var toFinite_1 = toFinite;

    /**
     * Converts `value` to an integer.
     *
     * **Note:** This method is loosely based on
     * [`ToInteger`](http://www.ecma-international.org/ecma-262/7.0/#sec-tointeger).
     *
     * @static
     * @memberOf _
     * @since 4.0.0
     * @category Lang
     * @param {*} value The value to convert.
     * @returns {number} Returns the converted integer.
     * @example
     *
     * _.toInteger(3.2);
     * // => 3
     *
     * _.toInteger(Number.MIN_VALUE);
     * // => 0
     *
     * _.toInteger(Infinity);
     * // => 1.7976931348623157e+308
     *
     * _.toInteger('3.2');
     * // => 3
     */
    function toInteger(value) {
      var result = toFinite_1(value),
          remainder = result % 1;

      return result === result ? (remainder ? result - remainder : result) : 0;
    }

    var toInteger_1 = toInteger;

    /* Built-in method references for those with the same name as other `lodash` methods. */
    var nativeMax$1 = Math.max;

    /**
     * This method is like `_.find` except that it returns the index of the first
     * element `predicate` returns truthy for instead of the element itself.
     *
     * @static
     * @memberOf _
     * @since 1.1.0
     * @category Array
     * @param {Array} array The array to inspect.
     * @param {Function} [predicate=_.identity] The function invoked per iteration.
     * @param {number} [fromIndex=0] The index to search from.
     * @returns {number} Returns the index of the found element, else `-1`.
     * @example
     *
     * var users = [
     *   { 'user': 'barney',  'active': false },
     *   { 'user': 'fred',    'active': false },
     *   { 'user': 'pebbles', 'active': true }
     * ];
     *
     * _.findIndex(users, function(o) { return o.user == 'barney'; });
     * // => 0
     *
     * // The `_.matches` iteratee shorthand.
     * _.findIndex(users, { 'user': 'fred', 'active': false });
     * // => 1
     *
     * // The `_.matchesProperty` iteratee shorthand.
     * _.findIndex(users, ['active', false]);
     * // => 0
     *
     * // The `_.property` iteratee shorthand.
     * _.findIndex(users, 'active');
     * // => 2
     */
    function findIndex(array, predicate, fromIndex) {
      var length = array == null ? 0 : array.length;
      if (!length) {
        return -1;
      }
      var index = fromIndex == null ? 0 : toInteger_1(fromIndex);
      if (index < 0) {
        index = nativeMax$1(length + index, 0);
      }
      return _baseFindIndex(array, _baseIteratee(predicate), index);
    }

    var findIndex_1 = findIndex;

    /**
     * Iterates over elements of `collection`, returning the first element
     * `predicate` returns truthy for. The predicate is invoked with three
     * arguments: (value, index|key, collection).
     *
     * @static
     * @memberOf _
     * @since 0.1.0
     * @category Collection
     * @param {Array|Object} collection The collection to inspect.
     * @param {Function} [predicate=_.identity] The function invoked per iteration.
     * @param {number} [fromIndex=0] The index to search from.
     * @returns {*} Returns the matched element, else `undefined`.
     * @example
     *
     * var users = [
     *   { 'user': 'barney',  'age': 36, 'active': true },
     *   { 'user': 'fred',    'age': 40, 'active': false },
     *   { 'user': 'pebbles', 'age': 1,  'active': true }
     * ];
     *
     * _.find(users, function(o) { return o.age < 40; });
     * // => object for 'barney'
     *
     * // The `_.matches` iteratee shorthand.
     * _.find(users, { 'age': 1, 'active': true });
     * // => object for 'pebbles'
     *
     * // The `_.matchesProperty` iteratee shorthand.
     * _.find(users, ['active', false]);
     * // => object for 'fred'
     *
     * // The `_.property` iteratee shorthand.
     * _.find(users, 'active');
     * // => object for 'barney'
     */
    var find$2 = _createFind(findIndex_1);

    var find_1 = find$2;

    /**
     * Flattens `array` a single level deep.
     *
     * @static
     * @memberOf _
     * @since 0.1.0
     * @category Array
     * @param {Array} array The array to flatten.
     * @returns {Array} Returns the new flattened array.
     * @example
     *
     * _.flatten([1, [2, [3, [4]], 5]]);
     * // => [1, 2, [3, [4]], 5]
     */
    function flatten(array) {
      var length = array == null ? 0 : array.length;
      return length ? _baseFlatten(array, 1) : [];
    }

    var flatten_1 = flatten;

    /**
     * Iterates over own and inherited enumerable string keyed properties of an
     * object and invokes `iteratee` for each property. The iteratee is invoked
     * with three arguments: (value, key, object). Iteratee functions may exit
     * iteration early by explicitly returning `false`.
     *
     * @static
     * @memberOf _
     * @since 0.3.0
     * @category Object
     * @param {Object} object The object to iterate over.
     * @param {Function} [iteratee=_.identity] The function invoked per iteration.
     * @returns {Object} Returns `object`.
     * @see _.forInRight
     * @example
     *
     * function Foo() {
     *   this.a = 1;
     *   this.b = 2;
     * }
     *
     * Foo.prototype.c = 3;
     *
     * _.forIn(new Foo, function(value, key) {
     *   console.log(key);
     * });
     * // => Logs 'a', 'b', then 'c' (iteration order is not guaranteed).
     */
    function forIn(object, iteratee) {
      return object == null
        ? object
        : _baseFor(object, _castFunction(iteratee), keysIn_1);
    }

    var forIn_1 = forIn;

    /**
     * Gets the last element of `array`.
     *
     * @static
     * @memberOf _
     * @since 0.1.0
     * @category Array
     * @param {Array} array The array to query.
     * @returns {*} Returns the last element of `array`.
     * @example
     *
     * _.last([1, 2, 3]);
     * // => 3
     */
    function last(array) {
      var length = array == null ? 0 : array.length;
      return length ? array[length - 1] : undefined;
    }

    var last_1 = last;

    /**
     * Creates an object with the same keys as `object` and values generated
     * by running each own enumerable string keyed property of `object` thru
     * `iteratee`. The iteratee is invoked with three arguments:
     * (value, key, object).
     *
     * @static
     * @memberOf _
     * @since 2.4.0
     * @category Object
     * @param {Object} object The object to iterate over.
     * @param {Function} [iteratee=_.identity] The function invoked per iteration.
     * @returns {Object} Returns the new mapped object.
     * @see _.mapKeys
     * @example
     *
     * var users = {
     *   'fred':    { 'user': 'fred',    'age': 40 },
     *   'pebbles': { 'user': 'pebbles', 'age': 1 }
     * };
     *
     * _.mapValues(users, function(o) { return o.age; });
     * // => { 'fred': 40, 'pebbles': 1 } (iteration order is not guaranteed)
     *
     * // The `_.property` iteratee shorthand.
     * _.mapValues(users, 'age');
     * // => { 'fred': 40, 'pebbles': 1 } (iteration order is not guaranteed)
     */
    function mapValues(object, iteratee) {
      var result = {};
      iteratee = _baseIteratee(iteratee);

      _baseForOwn(object, function(value, key, object) {
        _baseAssignValue(result, key, iteratee(value, key, object));
      });
      return result;
    }

    var mapValues_1 = mapValues;

    /**
     * The base implementation of methods like `_.max` and `_.min` which accepts a
     * `comparator` to determine the extremum value.
     *
     * @private
     * @param {Array} array The array to iterate over.
     * @param {Function} iteratee The iteratee invoked per iteration.
     * @param {Function} comparator The comparator used to compare values.
     * @returns {*} Returns the extremum value.
     */
    function baseExtremum(array, iteratee, comparator) {
      var index = -1,
          length = array.length;

      while (++index < length) {
        var value = array[index],
            current = iteratee(value);

        if (current != null && (computed === undefined
              ? (current === current && !isSymbol_1(current))
              : comparator(current, computed)
            )) {
          var computed = current,
              result = value;
        }
      }
      return result;
    }

    var _baseExtremum = baseExtremum;

    /**
     * The base implementation of `_.gt` which doesn't coerce arguments.
     *
     * @private
     * @param {*} value The value to compare.
     * @param {*} other The other value to compare.
     * @returns {boolean} Returns `true` if `value` is greater than `other`,
     *  else `false`.
     */
    function baseGt(value, other) {
      return value > other;
    }

    var _baseGt = baseGt;

    /**
     * Computes the maximum value of `array`. If `array` is empty or falsey,
     * `undefined` is returned.
     *
     * @static
     * @since 0.1.0
     * @memberOf _
     * @category Math
     * @param {Array} array The array to iterate over.
     * @returns {*} Returns the maximum value.
     * @example
     *
     * _.max([4, 2, 8, 6]);
     * // => 8
     *
     * _.max([]);
     * // => undefined
     */
    function max$6(array) {
      return (array && array.length)
        ? _baseExtremum(array, identity_1, _baseGt)
        : undefined;
    }

    var max_1 = max$6;

    /**
     * This function is like `assignValue` except that it doesn't assign
     * `undefined` values.
     *
     * @private
     * @param {Object} object The object to modify.
     * @param {string} key The key of the property to assign.
     * @param {*} value The value to assign.
     */
    function assignMergeValue(object, key, value) {
      if ((value !== undefined && !eq_1(object[key], value)) ||
          (value === undefined && !(key in object))) {
        _baseAssignValue(object, key, value);
      }
    }

    var _assignMergeValue = assignMergeValue;

    /** `Object#toString` result references. */
    var objectTag$4 = '[object Object]';

    /** Used for built-in method references. */
    var funcProto$2 = Function.prototype,
        objectProto$i = Object.prototype;

    /** Used to resolve the decompiled source of functions. */
    var funcToString$2 = funcProto$2.toString;

    /** Used to check objects for own properties. */
    var hasOwnProperty$f = objectProto$i.hasOwnProperty;

    /** Used to infer the `Object` constructor. */
    var objectCtorString = funcToString$2.call(Object);

    /**
     * Checks if `value` is a plain object, that is, an object created by the
     * `Object` constructor or one with a `[[Prototype]]` of `null`.
     *
     * @static
     * @memberOf _
     * @since 0.8.0
     * @category Lang
     * @param {*} value The value to check.
     * @returns {boolean} Returns `true` if `value` is a plain object, else `false`.
     * @example
     *
     * function Foo() {
     *   this.a = 1;
     * }
     *
     * _.isPlainObject(new Foo);
     * // => false
     *
     * _.isPlainObject([1, 2, 3]);
     * // => false
     *
     * _.isPlainObject({ 'x': 0, 'y': 0 });
     * // => true
     *
     * _.isPlainObject(Object.create(null));
     * // => true
     */
    function isPlainObject$1(value) {
      if (!isObjectLike_1(value) || _baseGetTag(value) != objectTag$4) {
        return false;
      }
      var proto = _getPrototype(value);
      if (proto === null) {
        return true;
      }
      var Ctor = hasOwnProperty$f.call(proto, 'constructor') && proto.constructor;
      return typeof Ctor == 'function' && Ctor instanceof Ctor &&
        funcToString$2.call(Ctor) == objectCtorString;
    }

    var isPlainObject_1 = isPlainObject$1;

    /**
     * Gets the value at `key`, unless `key` is "__proto__" or "constructor".
     *
     * @private
     * @param {Object} object The object to query.
     * @param {string} key The key of the property to get.
     * @returns {*} Returns the property value.
     */
    function safeGet(object, key) {
      if (key === 'constructor' && typeof object[key] === 'function') {
        return;
      }

      if (key == '__proto__') {
        return;
      }

      return object[key];
    }

    var _safeGet = safeGet;

    /**
     * Converts `value` to a plain object flattening inherited enumerable string
     * keyed properties of `value` to own properties of the plain object.
     *
     * @static
     * @memberOf _
     * @since 3.0.0
     * @category Lang
     * @param {*} value The value to convert.
     * @returns {Object} Returns the converted plain object.
     * @example
     *
     * function Foo() {
     *   this.b = 2;
     * }
     *
     * Foo.prototype.c = 3;
     *
     * _.assign({ 'a': 1 }, new Foo);
     * // => { 'a': 1, 'b': 2 }
     *
     * _.assign({ 'a': 1 }, _.toPlainObject(new Foo));
     * // => { 'a': 1, 'b': 2, 'c': 3 }
     */
    function toPlainObject(value) {
      return _copyObject(value, keysIn_1(value));
    }

    var toPlainObject_1 = toPlainObject;

    /**
     * A specialized version of `baseMerge` for arrays and objects which performs
     * deep merges and tracks traversed objects enabling objects with circular
     * references to be merged.
     *
     * @private
     * @param {Object} object The destination object.
     * @param {Object} source The source object.
     * @param {string} key The key of the value to merge.
     * @param {number} srcIndex The index of `source`.
     * @param {Function} mergeFunc The function to merge values.
     * @param {Function} [customizer] The function to customize assigned values.
     * @param {Object} [stack] Tracks traversed source values and their merged
     *  counterparts.
     */
    function baseMergeDeep(object, source, key, srcIndex, mergeFunc, customizer, stack) {
      var objValue = _safeGet(object, key),
          srcValue = _safeGet(source, key),
          stacked = stack.get(srcValue);

      if (stacked) {
        _assignMergeValue(object, key, stacked);
        return;
      }
      var newValue = customizer
        ? customizer(objValue, srcValue, (key + ''), object, source, stack)
        : undefined;

      var isCommon = newValue === undefined;

      if (isCommon) {
        var isArr = isArray_1(srcValue),
            isBuff = !isArr && isBuffer_1(srcValue),
            isTyped = !isArr && !isBuff && isTypedArray_1(srcValue);

        newValue = srcValue;
        if (isArr || isBuff || isTyped) {
          if (isArray_1(objValue)) {
            newValue = objValue;
          }
          else if (isArrayLikeObject_1(objValue)) {
            newValue = _copyArray(objValue);
          }
          else if (isBuff) {
            isCommon = false;
            newValue = _cloneBuffer(srcValue, true);
          }
          else if (isTyped) {
            isCommon = false;
            newValue = _cloneTypedArray(srcValue, true);
          }
          else {
            newValue = [];
          }
        }
        else if (isPlainObject_1(srcValue) || isArguments_1(srcValue)) {
          newValue = objValue;
          if (isArguments_1(objValue)) {
            newValue = toPlainObject_1(objValue);
          }
          else if (!isObject_1(objValue) || isFunction_1(objValue)) {
            newValue = _initCloneObject(srcValue);
          }
        }
        else {
          isCommon = false;
        }
      }
      if (isCommon) {
        // Recursively merge objects and arrays (susceptible to call stack limits).
        stack.set(srcValue, newValue);
        mergeFunc(newValue, srcValue, srcIndex, customizer, stack);
        stack['delete'](srcValue);
      }
      _assignMergeValue(object, key, newValue);
    }

    var _baseMergeDeep = baseMergeDeep;

    /**
     * The base implementation of `_.merge` without support for multiple sources.
     *
     * @private
     * @param {Object} object The destination object.
     * @param {Object} source The source object.
     * @param {number} srcIndex The index of `source`.
     * @param {Function} [customizer] The function to customize merged values.
     * @param {Object} [stack] Tracks traversed source values and their merged
     *  counterparts.
     */
    function baseMerge(object, source, srcIndex, customizer, stack) {
      if (object === source) {
        return;
      }
      _baseFor(source, function(srcValue, key) {
        stack || (stack = new _Stack);
        if (isObject_1(srcValue)) {
          _baseMergeDeep(object, source, key, srcIndex, baseMerge, customizer, stack);
        }
        else {
          var newValue = customizer
            ? customizer(_safeGet(object, key), srcValue, (key + ''), object, source, stack)
            : undefined;

          if (newValue === undefined) {
            newValue = srcValue;
          }
          _assignMergeValue(object, key, newValue);
        }
      }, keysIn_1);
    }

    var _baseMerge = baseMerge;

    /**
     * Creates a function like `_.assign`.
     *
     * @private
     * @param {Function} assigner The function to assign values.
     * @returns {Function} Returns the new assigner function.
     */
    function createAssigner(assigner) {
      return _baseRest(function(object, sources) {
        var index = -1,
            length = sources.length,
            customizer = length > 1 ? sources[length - 1] : undefined,
            guard = length > 2 ? sources[2] : undefined;

        customizer = (assigner.length > 3 && typeof customizer == 'function')
          ? (length--, customizer)
          : undefined;

        if (guard && _isIterateeCall(sources[0], sources[1], guard)) {
          customizer = length < 3 ? undefined : customizer;
          length = 1;
        }
        object = Object(object);
        while (++index < length) {
          var source = sources[index];
          if (source) {
            assigner(object, source, index, customizer);
          }
        }
        return object;
      });
    }

    var _createAssigner = createAssigner;

    /**
     * This method is like `_.assign` except that it recursively merges own and
     * inherited enumerable string keyed properties of source objects into the
     * destination object. Source properties that resolve to `undefined` are
     * skipped if a destination value exists. Array and plain object properties
     * are merged recursively. Other objects and value types are overridden by
     * assignment. Source objects are applied from left to right. Subsequent
     * sources overwrite property assignments of previous sources.
     *
     * **Note:** This method mutates `object`.
     *
     * @static
     * @memberOf _
     * @since 0.5.0
     * @category Object
     * @param {Object} object The destination object.
     * @param {...Object} [sources] The source objects.
     * @returns {Object} Returns `object`.
     * @example
     *
     * var object = {
     *   'a': [{ 'b': 2 }, { 'd': 4 }]
     * };
     *
     * var other = {
     *   'a': [{ 'c': 3 }, { 'e': 5 }]
     * };
     *
     * _.merge(object, other);
     * // => { 'a': [{ 'b': 2, 'c': 3 }, { 'd': 4, 'e': 5 }] }
     */
    var merge$3 = _createAssigner(function(object, source, srcIndex) {
      _baseMerge(object, source, srcIndex);
    });

    var merge_1 = merge$3;

    /**
     * The base implementation of `_.lt` which doesn't coerce arguments.
     *
     * @private
     * @param {*} value The value to compare.
     * @param {*} other The other value to compare.
     * @returns {boolean} Returns `true` if `value` is less than `other`,
     *  else `false`.
     */
    function baseLt(value, other) {
      return value < other;
    }

    var _baseLt = baseLt;

    /**
     * Computes the minimum value of `array`. If `array` is empty or falsey,
     * `undefined` is returned.
     *
     * @static
     * @since 0.1.0
     * @memberOf _
     * @category Math
     * @param {Array} array The array to iterate over.
     * @returns {*} Returns the minimum value.
     * @example
     *
     * _.min([4, 2, 8, 6]);
     * // => 2
     *
     * _.min([]);
     * // => undefined
     */
    function min$4(array) {
      return (array && array.length)
        ? _baseExtremum(array, identity_1, _baseLt)
        : undefined;
    }

    var min_1 = min$4;

    /**
     * This method is like `_.min` except that it accepts `iteratee` which is
     * invoked for each element in `array` to generate the criterion by which
     * the value is ranked. The iteratee is invoked with one argument: (value).
     *
     * @static
     * @memberOf _
     * @since 4.0.0
     * @category Math
     * @param {Array} array The array to iterate over.
     * @param {Function} [iteratee=_.identity] The iteratee invoked per element.
     * @returns {*} Returns the minimum value.
     * @example
     *
     * var objects = [{ 'n': 1 }, { 'n': 2 }];
     *
     * _.minBy(objects, function(o) { return o.n; });
     * // => { 'n': 1 }
     *
     * // The `_.property` iteratee shorthand.
     * _.minBy(objects, 'n');
     * // => { 'n': 1 }
     */
    function minBy(array, iteratee) {
      return (array && array.length)
        ? _baseExtremum(array, _baseIteratee(iteratee), _baseLt)
        : undefined;
    }

    var minBy_1 = minBy;

    /**
     * Gets the timestamp of the number of milliseconds that have elapsed since
     * the Unix epoch (1 January 1970 00:00:00 UTC).
     *
     * @static
     * @memberOf _
     * @since 2.4.0
     * @category Date
     * @returns {number} Returns the timestamp.
     * @example
     *
     * _.defer(function(stamp) {
     *   console.log(_.now() - stamp);
     * }, _.now());
     * // => Logs the number of milliseconds it took for the deferred invocation.
     */
    var now$6 = function() {
      return _root.Date.now();
    };

    var now_1 = now$6;

    /**
     * The base implementation of `_.set`.
     *
     * @private
     * @param {Object} object The object to modify.
     * @param {Array|string} path The path of the property to set.
     * @param {*} value The value to set.
     * @param {Function} [customizer] The function to customize path creation.
     * @returns {Object} Returns `object`.
     */
    function baseSet(object, path, value, customizer) {
      if (!isObject_1(object)) {
        return object;
      }
      path = _castPath(path, object);

      var index = -1,
          length = path.length,
          lastIndex = length - 1,
          nested = object;

      while (nested != null && ++index < length) {
        var key = _toKey(path[index]),
            newValue = value;

        if (index != lastIndex) {
          var objValue = nested[key];
          newValue = customizer ? customizer(objValue, key, nested) : undefined;
          if (newValue === undefined) {
            newValue = isObject_1(objValue)
              ? objValue
              : (_isIndex(path[index + 1]) ? [] : {});
          }
        }
        _assignValue(nested, key, newValue);
        nested = nested[key];
      }
      return object;
    }

    var _baseSet = baseSet;

    /**
     * The base implementation of  `_.pickBy` without support for iteratee shorthands.
     *
     * @private
     * @param {Object} object The source object.
     * @param {string[]} paths The property paths to pick.
     * @param {Function} predicate The function invoked per property.
     * @returns {Object} Returns the new object.
     */
    function basePickBy(object, paths, predicate) {
      var index = -1,
          length = paths.length,
          result = {};

      while (++index < length) {
        var path = paths[index],
            value = _baseGet(object, path);

        if (predicate(value, path)) {
          _baseSet(result, _castPath(path, object), value);
        }
      }
      return result;
    }

    var _basePickBy = basePickBy;

    /**
     * The base implementation of `_.pick` without support for individual
     * property identifiers.
     *
     * @private
     * @param {Object} object The source object.
     * @param {string[]} paths The property paths to pick.
     * @returns {Object} Returns the new object.
     */
    function basePick(object, paths) {
      return _basePickBy(object, paths, function(value, path) {
        return hasIn_1(object, path);
      });
    }

    var _basePick = basePick;

    /**
     * A specialized version of `baseRest` which flattens the rest array.
     *
     * @private
     * @param {Function} func The function to apply a rest parameter to.
     * @returns {Function} Returns the new function.
     */
    function flatRest(func) {
      return _setToString(_overRest(func, undefined, flatten_1), func + '');
    }

    var _flatRest = flatRest;

    /**
     * Creates an object composed of the picked `object` properties.
     *
     * @static
     * @since 0.1.0
     * @memberOf _
     * @category Object
     * @param {Object} object The source object.
     * @param {...(string|string[])} [paths] The property paths to pick.
     * @returns {Object} Returns the new object.
     * @example
     *
     * var object = { 'a': 1, 'b': '2', 'c': 3 };
     *
     * _.pick(object, ['a', 'c']);
     * // => { 'a': 1, 'c': 3 }
     */
    var pick = _flatRest(function(object, paths) {
      return object == null ? {} : _basePick(object, paths);
    });

    var pick_1 = pick;

    /* Built-in method references for those with the same name as other `lodash` methods. */
    var nativeCeil = Math.ceil,
        nativeMax$2 = Math.max;

    /**
     * The base implementation of `_.range` and `_.rangeRight` which doesn't
     * coerce arguments.
     *
     * @private
     * @param {number} start The start of the range.
     * @param {number} end The end of the range.
     * @param {number} step The value to increment or decrement by.
     * @param {boolean} [fromRight] Specify iterating from right to left.
     * @returns {Array} Returns the range of numbers.
     */
    function baseRange(start, end, step, fromRight) {
      var index = -1,
          length = nativeMax$2(nativeCeil((end - start) / (step || 1)), 0),
          result = Array(length);

      while (length--) {
        result[fromRight ? length : ++index] = start;
        start += step;
      }
      return result;
    }

    var _baseRange = baseRange;

    /**
     * Creates a `_.range` or `_.rangeRight` function.
     *
     * @private
     * @param {boolean} [fromRight] Specify iterating from right to left.
     * @returns {Function} Returns the new range function.
     */
    function createRange(fromRight) {
      return function(start, end, step) {
        if (step && typeof step != 'number' && _isIterateeCall(start, end, step)) {
          end = step = undefined;
        }
        // Ensure the sign of `-0` is preserved.
        start = toFinite_1(start);
        if (end === undefined) {
          end = start;
          start = 0;
        } else {
          end = toFinite_1(end);
        }
        step = step === undefined ? (start < end ? 1 : -1) : toFinite_1(step);
        return _baseRange(start, end, step, fromRight);
      };
    }

    var _createRange = createRange;

    /**
     * Creates an array of numbers (positive and/or negative) progressing from
     * `start` up to, but not including, `end`. A step of `-1` is used if a negative
     * `start` is specified without an `end` or `step`. If `end` is not specified,
     * it's set to `start` with `start` then set to `0`.
     *
     * **Note:** JavaScript follows the IEEE-754 standard for resolving
     * floating-point values which can produce unexpected results.
     *
     * @static
     * @since 0.1.0
     * @memberOf _
     * @category Util
     * @param {number} [start=0] The start of the range.
     * @param {number} end The end of the range.
     * @param {number} [step=1] The value to increment or decrement by.
     * @returns {Array} Returns the range of numbers.
     * @see _.inRange, _.rangeRight
     * @example
     *
     * _.range(4);
     * // => [0, 1, 2, 3]
     *
     * _.range(-4);
     * // => [0, -1, -2, -3]
     *
     * _.range(1, 5);
     * // => [1, 2, 3, 4]
     *
     * _.range(0, 20, 5);
     * // => [0, 5, 10, 15]
     *
     * _.range(0, -4, -1);
     * // => [0, -1, -2, -3]
     *
     * _.range(1, 4, 0);
     * // => [1, 1, 1]
     *
     * _.range(0);
     * // => []
     */
    var range$5 = _createRange();

    var range_1 = range$5;

    /**
     * The base implementation of `_.sortBy` which uses `comparer` to define the
     * sort order of `array` and replaces criteria objects with their corresponding
     * values.
     *
     * @private
     * @param {Array} array The array to sort.
     * @param {Function} comparer The function to define sort order.
     * @returns {Array} Returns `array`.
     */
    function baseSortBy(array, comparer) {
      var length = array.length;

      array.sort(comparer);
      while (length--) {
        array[length] = array[length].value;
      }
      return array;
    }

    var _baseSortBy = baseSortBy;

    /**
     * Compares values to sort them in ascending order.
     *
     * @private
     * @param {*} value The value to compare.
     * @param {*} other The other value to compare.
     * @returns {number} Returns the sort order indicator for `value`.
     */
    function compareAscending(value, other) {
      if (value !== other) {
        var valIsDefined = value !== undefined,
            valIsNull = value === null,
            valIsReflexive = value === value,
            valIsSymbol = isSymbol_1(value);

        var othIsDefined = other !== undefined,
            othIsNull = other === null,
            othIsReflexive = other === other,
            othIsSymbol = isSymbol_1(other);

        if ((!othIsNull && !othIsSymbol && !valIsSymbol && value > other) ||
            (valIsSymbol && othIsDefined && othIsReflexive && !othIsNull && !othIsSymbol) ||
            (valIsNull && othIsDefined && othIsReflexive) ||
            (!valIsDefined && othIsReflexive) ||
            !valIsReflexive) {
          return 1;
        }
        if ((!valIsNull && !valIsSymbol && !othIsSymbol && value < other) ||
            (othIsSymbol && valIsDefined && valIsReflexive && !valIsNull && !valIsSymbol) ||
            (othIsNull && valIsDefined && valIsReflexive) ||
            (!othIsDefined && valIsReflexive) ||
            !othIsReflexive) {
          return -1;
        }
      }
      return 0;
    }

    var _compareAscending = compareAscending;

    /**
     * Used by `_.orderBy` to compare multiple properties of a value to another
     * and stable sort them.
     *
     * If `orders` is unspecified, all values are sorted in ascending order. Otherwise,
     * specify an order of "desc" for descending or "asc" for ascending sort order
     * of corresponding values.
     *
     * @private
     * @param {Object} object The object to compare.
     * @param {Object} other The other object to compare.
     * @param {boolean[]|string[]} orders The order to sort by for each property.
     * @returns {number} Returns the sort order indicator for `object`.
     */
    function compareMultiple(object, other, orders) {
      var index = -1,
          objCriteria = object.criteria,
          othCriteria = other.criteria,
          length = objCriteria.length,
          ordersLength = orders.length;

      while (++index < length) {
        var result = _compareAscending(objCriteria[index], othCriteria[index]);
        if (result) {
          if (index >= ordersLength) {
            return result;
          }
          var order = orders[index];
          return result * (order == 'desc' ? -1 : 1);
        }
      }
      // Fixes an `Array#sort` bug in the JS engine embedded in Adobe applications
      // that causes it, under certain circumstances, to provide the same value for
      // `object` and `other`. See https://github.com/jashkenas/underscore/pull/1247
      // for more details.
      //
      // This also ensures a stable sort in V8 and other engines.
      // See https://bugs.chromium.org/p/v8/issues/detail?id=90 for more details.
      return object.index - other.index;
    }

    var _compareMultiple = compareMultiple;

    /**
     * The base implementation of `_.orderBy` without param guards.
     *
     * @private
     * @param {Array|Object} collection The collection to iterate over.
     * @param {Function[]|Object[]|string[]} iteratees The iteratees to sort by.
     * @param {string[]} orders The sort orders of `iteratees`.
     * @returns {Array} Returns the new sorted array.
     */
    function baseOrderBy(collection, iteratees, orders) {
      var index = -1;
      iteratees = _arrayMap(iteratees.length ? iteratees : [identity_1], _baseUnary(_baseIteratee));

      var result = _baseMap(collection, function(value, key, collection) {
        var criteria = _arrayMap(iteratees, function(iteratee) {
          return iteratee(value);
        });
        return { 'criteria': criteria, 'index': ++index, 'value': value };
      });

      return _baseSortBy(result, function(object, other) {
        return _compareMultiple(object, other, orders);
      });
    }

    var _baseOrderBy = baseOrderBy;

    /**
     * Creates an array of elements, sorted in ascending order by the results of
     * running each element in a collection thru each iteratee. This method
     * performs a stable sort, that is, it preserves the original sort order of
     * equal elements. The iteratees are invoked with one argument: (value).
     *
     * @static
     * @memberOf _
     * @since 0.1.0
     * @category Collection
     * @param {Array|Object} collection The collection to iterate over.
     * @param {...(Function|Function[])} [iteratees=[_.identity]]
     *  The iteratees to sort by.
     * @returns {Array} Returns the new sorted array.
     * @example
     *
     * var users = [
     *   { 'user': 'fred',   'age': 48 },
     *   { 'user': 'barney', 'age': 36 },
     *   { 'user': 'fred',   'age': 40 },
     *   { 'user': 'barney', 'age': 34 }
     * ];
     *
     * _.sortBy(users, [function(o) { return o.user; }]);
     * // => objects for [['barney', 36], ['barney', 34], ['fred', 48], ['fred', 40]]
     *
     * _.sortBy(users, ['user', 'age']);
     * // => objects for [['barney', 34], ['barney', 36], ['fred', 40], ['fred', 48]]
     */
    var sortBy = _baseRest(function(collection, iteratees) {
      if (collection == null) {
        return [];
      }
      var length = iteratees.length;
      if (length > 1 && _isIterateeCall(collection, iteratees[0], iteratees[1])) {
        iteratees = [];
      } else if (length > 2 && _isIterateeCall(iteratees[0], iteratees[1], iteratees[2])) {
        iteratees = [iteratees[0]];
      }
      return _baseOrderBy(collection, _baseFlatten(iteratees, 1), []);
    });

    var sortBy_1 = sortBy;

    /** Used to generate unique IDs. */
    var idCounter = 0;

    /**
     * Generates a unique ID. If `prefix` is given, the ID is appended to it.
     *
     * @static
     * @since 0.1.0
     * @memberOf _
     * @category Util
     * @param {string} [prefix=''] The value to prefix the ID with.
     * @returns {string} Returns the unique ID.
     * @example
     *
     * _.uniqueId('contact_');
     * // => 'contact_104'
     *
     * _.uniqueId();
     * // => '105'
     */
    function uniqueId(prefix) {
      var id = ++idCounter;
      return toString_1(prefix) + id;
    }

    var uniqueId_1 = uniqueId;

    /**
     * This base implementation of `_.zipObject` which assigns values using `assignFunc`.
     *
     * @private
     * @param {Array} props The property identifiers.
     * @param {Array} values The property values.
     * @param {Function} assignFunc The function to assign values.
     * @returns {Object} Returns the new object.
     */
    function baseZipObject(props, values, assignFunc) {
      var index = -1,
          length = props.length,
          valsLength = values.length,
          result = {};

      while (++index < length) {
        var value = index < valsLength ? values[index] : undefined;
        assignFunc(result, props[index], value);
      }
      return result;
    }

    var _baseZipObject = baseZipObject;

    /**
     * This method is like `_.fromPairs` except that it accepts two arrays,
     * one of property identifiers and one of corresponding values.
     *
     * @static
     * @memberOf _
     * @since 0.4.0
     * @category Array
     * @param {Array} [props=[]] The property identifiers.
     * @param {Array} [values=[]] The property values.
     * @returns {Object} Returns the new object.
     * @example
     *
     * _.zipObject(['a', 'b'], [1, 2]);
     * // => { 'a': 1, 'b': 2 }
     */
    function zipObject(props, values) {
      return _baseZipObject(props || [], values || [], _assignValue);
    }

    var zipObject_1 = zipObject;

    /* global window */

    var lodash$2;

    if (typeof commonjsRequire === "function") {
      try {
        lodash$2 = {
          cloneDeep: cloneDeep_1,
          constant: constant_1,
          defaults: defaults_1,
          each: each,
          filter: filter_1,
          find: find_1,
          flatten: flatten_1,
          forEach: forEach_1,
          forIn: forIn_1,
          has:  has_1,
          isUndefined: isUndefined_1,
          last: last_1,
          map: map_1,
          mapValues: mapValues_1,
          max: max_1,
          merge: merge_1,
          min: min_1,
          minBy: minBy_1,
          now: now_1,
          pick: pick_1,
          range: range_1,
          reduce: reduce_1,
          sortBy: sortBy_1,
          uniqueId: uniqueId_1,
          values: values_1,
          zipObject: zipObject_1,
        };
      } catch (e) {
        // continue regardless of error
      }
    }

    if (!lodash$2) {
      lodash$2 = window._;
    }

    var lodash_1$1 = lodash$2;

    /*
     * Simple doubly linked list implementation derived from Cormen, et al.,
     * "Introduction to Algorithms".
     */

    var list = List;

    function List() {
      var sentinel = {};
      sentinel._next = sentinel._prev = sentinel;
      this._sentinel = sentinel;
    }

    List.prototype.dequeue = function() {
      var sentinel = this._sentinel;
      var entry = sentinel._prev;
      if (entry !== sentinel) {
        unlink(entry);
        return entry;
      }
    };

    List.prototype.enqueue = function(entry) {
      var sentinel = this._sentinel;
      if (entry._prev && entry._next) {
        unlink(entry);
      }
      entry._next = sentinel._next;
      sentinel._next._prev = entry;
      sentinel._next = entry;
      entry._prev = sentinel;
    };

    List.prototype.toString = function() {
      var strs = [];
      var sentinel = this._sentinel;
      var curr = sentinel._prev;
      while (curr !== sentinel) {
        strs.push(JSON.stringify(curr, filterOutLinks));
        curr = curr._prev;
      }
      return "[" + strs.join(", ") + "]";
    };

    function unlink(entry) {
      entry._prev._next = entry._next;
      entry._next._prev = entry._prev;
      delete entry._next;
      delete entry._prev;
    }

    function filterOutLinks(k, v) {
      if (k !== "_next" && k !== "_prev") {
        return v;
      }
    }

    var Graph$1 = graphlib_1.Graph;


    /*
     * A greedy heuristic for finding a feedback arc set for a graph. A feedback
     * arc set is a set of edges that can be removed to make a graph acyclic.
     * The algorithm comes from: P. Eades, X. Lin, and W. F. Smyth, "A fast and
     * effective heuristic for the feedback arc set problem." This implementation
     * adjusts that from the paper to allow for weighted edges.
     */
    var greedyFas = greedyFAS;

    var DEFAULT_WEIGHT_FN = lodash_1$1.constant(1);

    function greedyFAS(g, weightFn) {
      if (g.nodeCount() <= 1) {
        return [];
      }
      var state = buildState(g, weightFn || DEFAULT_WEIGHT_FN);
      var results = doGreedyFAS(state.graph, state.buckets, state.zeroIdx);

      // Expand multi-edges
      return lodash_1$1.flatten(lodash_1$1.map(results, function(e) {
        return g.outEdges(e.v, e.w);
      }), true);
    }

    function doGreedyFAS(g, buckets, zeroIdx) {
      var results = [];
      var sources = buckets[buckets.length - 1];
      var sinks = buckets[0];

      var entry;
      while (g.nodeCount()) {
        while ((entry = sinks.dequeue()))   { removeNode(g, buckets, zeroIdx, entry); }
        while ((entry = sources.dequeue())) { removeNode(g, buckets, zeroIdx, entry); }
        if (g.nodeCount()) {
          for (var i = buckets.length - 2; i > 0; --i) {
            entry = buckets[i].dequeue();
            if (entry) {
              results = results.concat(removeNode(g, buckets, zeroIdx, entry, true));
              break;
            }
          }
        }
      }

      return results;
    }

    function removeNode(g, buckets, zeroIdx, entry, collectPredecessors) {
      var results = collectPredecessors ? [] : undefined;

      lodash_1$1.forEach(g.inEdges(entry.v), function(edge) {
        var weight = g.edge(edge);
        var uEntry = g.node(edge.v);

        if (collectPredecessors) {
          results.push({ v: edge.v, w: edge.w });
        }

        uEntry.out -= weight;
        assignBucket(buckets, zeroIdx, uEntry);
      });

      lodash_1$1.forEach(g.outEdges(entry.v), function(edge) {
        var weight = g.edge(edge);
        var w = edge.w;
        var wEntry = g.node(w);
        wEntry["in"] -= weight;
        assignBucket(buckets, zeroIdx, wEntry);
      });

      g.removeNode(entry.v);

      return results;
    }

    function buildState(g, weightFn) {
      var fasGraph = new Graph$1();
      var maxIn = 0;
      var maxOut = 0;

      lodash_1$1.forEach(g.nodes(), function(v) {
        fasGraph.setNode(v, { v: v, "in": 0, out: 0 });
      });

      // Aggregate weights on nodes, but also sum the weights across multi-edges
      // into a single edge for the fasGraph.
      lodash_1$1.forEach(g.edges(), function(e) {
        var prevWeight = fasGraph.edge(e.v, e.w) || 0;
        var weight = weightFn(e);
        var edgeWeight = prevWeight + weight;
        fasGraph.setEdge(e.v, e.w, edgeWeight);
        maxOut = Math.max(maxOut, fasGraph.node(e.v).out += weight);
        maxIn  = Math.max(maxIn,  fasGraph.node(e.w)["in"]  += weight);
      });

      var buckets = lodash_1$1.range(maxOut + maxIn + 3).map(function() { return new list(); });
      var zeroIdx = maxIn + 1;

      lodash_1$1.forEach(fasGraph.nodes(), function(v) {
        assignBucket(buckets, zeroIdx, fasGraph.node(v));
      });

      return { graph: fasGraph, buckets: buckets, zeroIdx: zeroIdx };
    }

    function assignBucket(buckets, zeroIdx, entry) {
      if (!entry.out) {
        buckets[0].enqueue(entry);
      } else if (!entry["in"]) {
        buckets[buckets.length - 1].enqueue(entry);
      } else {
        buckets[entry.out - entry["in"] + zeroIdx].enqueue(entry);
      }
    }

    var acyclic = {
      run: run,
      undo: undo
    };

    function run(g) {
      var fas = (g.graph().acyclicer === "greedy"
        ? greedyFas(g, weightFn(g))
        : dfsFAS(g));
      lodash_1$1.forEach(fas, function(e) {
        var label = g.edge(e);
        g.removeEdge(e);
        label.forwardName = e.name;
        label.reversed = true;
        g.setEdge(e.w, e.v, label, lodash_1$1.uniqueId("rev"));
      });

      function weightFn(g) {
        return function(e) {
          return g.edge(e).weight;
        };
      }
    }

    function dfsFAS(g) {
      var fas = [];
      var stack = {};
      var visited = {};

      function dfs(v) {
        if (lodash_1$1.has(visited, v)) {
          return;
        }
        visited[v] = true;
        stack[v] = true;
        lodash_1$1.forEach(g.outEdges(v), function(e) {
          if (lodash_1$1.has(stack, e.w)) {
            fas.push(e);
          } else {
            dfs(e.w);
          }
        });
        delete stack[v];
      }

      lodash_1$1.forEach(g.nodes(), dfs);
      return fas;
    }

    function undo(g) {
      lodash_1$1.forEach(g.edges(), function(e) {
        var label = g.edge(e);
        if (label.reversed) {
          g.removeEdge(e);

          var forwardName = label.forwardName;
          delete label.reversed;
          delete label.forwardName;
          g.setEdge(e.w, e.v, label, forwardName);
        }
      });
    }

    var Graph$2 = graphlib_1.Graph;

    var util = {
      addDummyNode: addDummyNode,
      simplify: simplify,
      asNonCompoundGraph: asNonCompoundGraph,
      successorWeights: successorWeights,
      predecessorWeights: predecessorWeights,
      intersectRect: intersectRect,
      buildLayerMatrix: buildLayerMatrix,
      normalizeRanks: normalizeRanks,
      removeEmptyRanks: removeEmptyRanks,
      addBorderNode: addBorderNode,
      maxRank: maxRank,
      partition: partition$2,
      time: time$2,
      notime: notime
    };

    /*
     * Adds a dummy node to the graph and return v.
     */
    function addDummyNode(g, type, attrs, name) {
      var v;
      do {
        v = lodash_1$1.uniqueId(name);
      } while (g.hasNode(v));

      attrs.dummy = type;
      g.setNode(v, attrs);
      return v;
    }

    /*
     * Returns a new graph with only simple edges. Handles aggregation of data
     * associated with multi-edges.
     */
    function simplify(g) {
      var simplified = new Graph$2().setGraph(g.graph());
      lodash_1$1.forEach(g.nodes(), function(v) { simplified.setNode(v, g.node(v)); });
      lodash_1$1.forEach(g.edges(), function(e) {
        var simpleLabel = simplified.edge(e.v, e.w) || { weight: 0, minlen: 1 };
        var label = g.edge(e);
        simplified.setEdge(e.v, e.w, {
          weight: simpleLabel.weight + label.weight,
          minlen: Math.max(simpleLabel.minlen, label.minlen)
        });
      });
      return simplified;
    }

    function asNonCompoundGraph(g) {
      var simplified = new Graph$2({ multigraph: g.isMultigraph() }).setGraph(g.graph());
      lodash_1$1.forEach(g.nodes(), function(v) {
        if (!g.children(v).length) {
          simplified.setNode(v, g.node(v));
        }
      });
      lodash_1$1.forEach(g.edges(), function(e) {
        simplified.setEdge(e, g.edge(e));
      });
      return simplified;
    }

    function successorWeights(g) {
      var weightMap = lodash_1$1.map(g.nodes(), function(v) {
        var sucs = {};
        lodash_1$1.forEach(g.outEdges(v), function(e) {
          sucs[e.w] = (sucs[e.w] || 0) + g.edge(e).weight;
        });
        return sucs;
      });
      return lodash_1$1.zipObject(g.nodes(), weightMap);
    }

    function predecessorWeights(g) {
      var weightMap = lodash_1$1.map(g.nodes(), function(v) {
        var preds = {};
        lodash_1$1.forEach(g.inEdges(v), function(e) {
          preds[e.v] = (preds[e.v] || 0) + g.edge(e).weight;
        });
        return preds;
      });
      return lodash_1$1.zipObject(g.nodes(), weightMap);
    }

    /*
     * Finds where a line starting at point ({x, y}) would intersect a rectangle
     * ({x, y, width, height}) if it were pointing at the rectangle's center.
     */
    function intersectRect(rect, point) {
      var x = rect.x;
      var y = rect.y;

      // Rectangle intersection algorithm from:
      // http://math.stackexchange.com/questions/108113/find-edge-between-two-boxes
      var dx = point.x - x;
      var dy = point.y - y;
      var w = rect.width / 2;
      var h = rect.height / 2;

      if (!dx && !dy) {
        throw new Error("Not possible to find intersection inside of the rectangle");
      }

      var sx, sy;
      if (Math.abs(dy) * w > Math.abs(dx) * h) {
        // Intersection is top or bottom of rect.
        if (dy < 0) {
          h = -h;
        }
        sx = h * dx / dy;
        sy = h;
      } else {
        // Intersection is left or right of rect.
        if (dx < 0) {
          w = -w;
        }
        sx = w;
        sy = w * dy / dx;
      }

      return { x: x + sx, y: y + sy };
    }

    /*
     * Given a DAG with each node assigned "rank" and "order" properties, this
     * function will produce a matrix with the ids of each node.
     */
    function buildLayerMatrix(g) {
      var layering = lodash_1$1.map(lodash_1$1.range(maxRank(g) + 1), function() { return []; });
      lodash_1$1.forEach(g.nodes(), function(v) {
        var node = g.node(v);
        var rank = node.rank;
        if (!lodash_1$1.isUndefined(rank)) {
          layering[rank][node.order] = v;
        }
      });
      return layering;
    }

    /*
     * Adjusts the ranks for all nodes in the graph such that all nodes v have
     * rank(v) >= 0 and at least one node w has rank(w) = 0.
     */
    function normalizeRanks(g) {
      var min = lodash_1$1.min(lodash_1$1.map(g.nodes(), function(v) { return g.node(v).rank; }));
      lodash_1$1.forEach(g.nodes(), function(v) {
        var node = g.node(v);
        if (lodash_1$1.has(node, "rank")) {
          node.rank -= min;
        }
      });
    }

    function removeEmptyRanks(g) {
      // Ranks may not start at 0, so we need to offset them
      var offset = lodash_1$1.min(lodash_1$1.map(g.nodes(), function(v) { return g.node(v).rank; }));

      var layers = [];
      lodash_1$1.forEach(g.nodes(), function(v) {
        var rank = g.node(v).rank - offset;
        if (!layers[rank]) {
          layers[rank] = [];
        }
        layers[rank].push(v);
      });

      var delta = 0;
      var nodeRankFactor = g.graph().nodeRankFactor;
      lodash_1$1.forEach(layers, function(vs, i) {
        if (lodash_1$1.isUndefined(vs) && i % nodeRankFactor !== 0) {
          --delta;
        } else if (delta) {
          lodash_1$1.forEach(vs, function(v) { g.node(v).rank += delta; });
        }
      });
    }

    function addBorderNode(g, prefix, rank, order) {
      var node = {
        width: 0,
        height: 0
      };
      if (arguments.length >= 4) {
        node.rank = rank;
        node.order = order;
      }
      return addDummyNode(g, "border", node, prefix);
    }

    function maxRank(g) {
      return lodash_1$1.max(lodash_1$1.map(g.nodes(), function(v) {
        var rank = g.node(v).rank;
        if (!lodash_1$1.isUndefined(rank)) {
          return rank;
        }
      }));
    }

    /*
     * Partition a collection into two groups: `lhs` and `rhs`. If the supplied
     * function returns true for an entry it goes into `lhs`. Otherwise it goes
     * into `rhs.
     */
    function partition$2(collection, fn) {
      var result = { lhs: [], rhs: [] };
      lodash_1$1.forEach(collection, function(value) {
        if (fn(value)) {
          result.lhs.push(value);
        } else {
          result.rhs.push(value);
        }
      });
      return result;
    }

    /*
     * Returns a new function that wraps `fn` with a timer. The wrapper logs the
     * time it takes to execute the function.
     */
    function time$2(name, fn) {
      var start = lodash_1$1.now();
      try {
        return fn();
      } finally {
        console.log(name + " time: " + (lodash_1$1.now() - start) + "ms");
      }
    }

    function notime(name, fn) {
      return fn();
    }

    var normalize$2 = {
      run: run$1,
      undo: undo$1
    };

    /*
     * Breaks any long edges in the graph into short segments that span 1 layer
     * each. This operation is undoable with the denormalize function.
     *
     * Pre-conditions:
     *
     *    1. The input graph is a DAG.
     *    2. Each node in the graph has a "rank" property.
     *
     * Post-condition:
     *
     *    1. All edges in the graph have a length of 1.
     *    2. Dummy nodes are added where edges have been split into segments.
     *    3. The graph is augmented with a "dummyChains" attribute which contains
     *       the first dummy in each chain of dummy nodes produced.
     */
    function run$1(g) {
      g.graph().dummyChains = [];
      lodash_1$1.forEach(g.edges(), function(edge) { normalizeEdge(g, edge); });
    }

    function normalizeEdge(g, e) {
      var v = e.v;
      var vRank = g.node(v).rank;
      var w = e.w;
      var wRank = g.node(w).rank;
      var name = e.name;
      var edgeLabel = g.edge(e);
      var labelRank = edgeLabel.labelRank;

      if (wRank === vRank + 1) return;

      g.removeEdge(e);

      var dummy, attrs, i;
      for (i = 0, ++vRank; vRank < wRank; ++i, ++vRank) {
        edgeLabel.points = [];
        attrs = {
          width: 0, height: 0,
          edgeLabel: edgeLabel, edgeObj: e,
          rank: vRank
        };
        dummy = util.addDummyNode(g, "edge", attrs, "_d");
        if (vRank === labelRank) {
          attrs.width = edgeLabel.width;
          attrs.height = edgeLabel.height;
          attrs.dummy = "edge-label";
          attrs.labelpos = edgeLabel.labelpos;
        }
        g.setEdge(v, dummy, { weight: edgeLabel.weight }, name);
        if (i === 0) {
          g.graph().dummyChains.push(dummy);
        }
        v = dummy;
      }

      g.setEdge(v, w, { weight: edgeLabel.weight }, name);
    }

    function undo$1(g) {
      lodash_1$1.forEach(g.graph().dummyChains, function(v) {
        var node = g.node(v);
        var origLabel = node.edgeLabel;
        var w;
        g.setEdge(node.edgeObj, origLabel);
        while (node.dummy) {
          w = g.successors(v)[0];
          g.removeNode(v);
          origLabel.points.push({ x: node.x, y: node.y });
          if (node.dummy === "edge-label") {
            origLabel.x = node.x;
            origLabel.y = node.y;
            origLabel.width = node.width;
            origLabel.height = node.height;
          }
          v = w;
          node = g.node(v);
        }
      });
    }

    var util$1 = {
      longestPath: longestPath,
      slack: slack
    };

    /*
     * Initializes ranks for the input graph using the longest path algorithm. This
     * algorithm scales well and is fast in practice, it yields rather poor
     * solutions. Nodes are pushed to the lowest layer possible, leaving the bottom
     * ranks wide and leaving edges longer than necessary. However, due to its
     * speed, this algorithm is good for getting an initial ranking that can be fed
     * into other algorithms.
     *
     * This algorithm does not normalize layers because it will be used by other
     * algorithms in most cases. If using this algorithm directly, be sure to
     * run normalize at the end.
     *
     * Pre-conditions:
     *
     *    1. Input graph is a DAG.
     *    2. Input graph node labels can be assigned properties.
     *
     * Post-conditions:
     *
     *    1. Each node will be assign an (unnormalized) "rank" property.
     */
    function longestPath(g) {
      var visited = {};

      function dfs(v) {
        var label = g.node(v);
        if (lodash_1$1.has(visited, v)) {
          return label.rank;
        }
        visited[v] = true;

        var rank = lodash_1$1.min(lodash_1$1.map(g.outEdges(v), function(e) {
          return dfs(e.w) - g.edge(e).minlen;
        }));

        if (rank === Number.POSITIVE_INFINITY || // return value of _.map([]) for Lodash 3
            rank === undefined || // return value of _.map([]) for Lodash 4
            rank === null) { // return value of _.map([null])
          rank = 0;
        }

        return (label.rank = rank);
      }

      lodash_1$1.forEach(g.sources(), dfs);
    }

    /*
     * Returns the amount of slack for the given edge. The slack is defined as the
     * difference between the length of the edge and its minimum length.
     */
    function slack(g, e) {
      return g.node(e.w).rank - g.node(e.v).rank - g.edge(e).minlen;
    }

    var Graph$3 = graphlib_1.Graph;
    var slack$1 = util$1.slack;

    var feasibleTree_1 = feasibleTree;

    /*
     * Constructs a spanning tree with tight edges and adjusted the input node's
     * ranks to achieve this. A tight edge is one that is has a length that matches
     * its "minlen" attribute.
     *
     * The basic structure for this function is derived from Gansner, et al., "A
     * Technique for Drawing Directed Graphs."
     *
     * Pre-conditions:
     *
     *    1. Graph must be a DAG.
     *    2. Graph must be connected.
     *    3. Graph must have at least one node.
     *    5. Graph nodes must have been previously assigned a "rank" property that
     *       respects the "minlen" property of incident edges.
     *    6. Graph edges must have a "minlen" property.
     *
     * Post-conditions:
     *
     *    - Graph nodes will have their rank adjusted to ensure that all edges are
     *      tight.
     *
     * Returns a tree (undirected graph) that is constructed using only "tight"
     * edges.
     */
    function feasibleTree(g) {
      var t = new Graph$3({ directed: false });

      // Choose arbitrary node from which to start our tree
      var start = g.nodes()[0];
      var size = g.nodeCount();
      t.setNode(start, {});

      var edge, delta;
      while (tightTree(t, g) < size) {
        edge = findMinSlackEdge(t, g);
        delta = t.hasNode(edge.v) ? slack$1(g, edge) : -slack$1(g, edge);
        shiftRanks(t, g, delta);
      }

      return t;
    }

    /*
     * Finds a maximal tree of tight edges and returns the number of nodes in the
     * tree.
     */
    function tightTree(t, g) {
      function dfs(v) {
        lodash_1$1.forEach(g.nodeEdges(v), function(e) {
          var edgeV = e.v,
            w = (v === edgeV) ? e.w : edgeV;
          if (!t.hasNode(w) && !slack$1(g, e)) {
            t.setNode(w, {});
            t.setEdge(v, w, {});
            dfs(w);
          }
        });
      }

      lodash_1$1.forEach(t.nodes(), dfs);
      return t.nodeCount();
    }

    /*
     * Finds the edge with the smallest slack that is incident on tree and returns
     * it.
     */
    function findMinSlackEdge(t, g) {
      return lodash_1$1.minBy(g.edges(), function(e) {
        if (t.hasNode(e.v) !== t.hasNode(e.w)) {
          return slack$1(g, e);
        }
      });
    }

    function shiftRanks(t, g, delta) {
      lodash_1$1.forEach(t.nodes(), function(v) {
        g.node(v).rank += delta;
      });
    }

    var slack$2 = util$1.slack;
    var initRank = util$1.longestPath;
    var preorder$1 = graphlib_1.alg.preorder;
    var postorder$1 = graphlib_1.alg.postorder;
    var simplify$1 = util.simplify;

    var networkSimplex_1 = networkSimplex;

    // Expose some internals for testing purposes
    networkSimplex.initLowLimValues = initLowLimValues;
    networkSimplex.initCutValues = initCutValues;
    networkSimplex.calcCutValue = calcCutValue;
    networkSimplex.leaveEdge = leaveEdge;
    networkSimplex.enterEdge = enterEdge;
    networkSimplex.exchangeEdges = exchangeEdges;

    /*
     * The network simplex algorithm assigns ranks to each node in the input graph
     * and iteratively improves the ranking to reduce the length of edges.
     *
     * Preconditions:
     *
     *    1. The input graph must be a DAG.
     *    2. All nodes in the graph must have an object value.
     *    3. All edges in the graph must have "minlen" and "weight" attributes.
     *
     * Postconditions:
     *
     *    1. All nodes in the graph will have an assigned "rank" attribute that has
     *       been optimized by the network simplex algorithm. Ranks start at 0.
     *
     *
     * A rough sketch of the algorithm is as follows:
     *
     *    1. Assign initial ranks to each node. We use the longest path algorithm,
     *       which assigns ranks to the lowest position possible. In general this
     *       leads to very wide bottom ranks and unnecessarily long edges.
     *    2. Construct a feasible tight tree. A tight tree is one such that all
     *       edges in the tree have no slack (difference between length of edge
     *       and minlen for the edge). This by itself greatly improves the assigned
     *       rankings by shorting edges.
     *    3. Iteratively find edges that have negative cut values. Generally a
     *       negative cut value indicates that the edge could be removed and a new
     *       tree edge could be added to produce a more compact graph.
     *
     * Much of the algorithms here are derived from Gansner, et al., "A Technique
     * for Drawing Directed Graphs." The structure of the file roughly follows the
     * structure of the overall algorithm.
     */
    function networkSimplex(g) {
      g = simplify$1(g);
      initRank(g);
      var t = feasibleTree_1(g);
      initLowLimValues(t);
      initCutValues(t, g);

      var e, f;
      while ((e = leaveEdge(t))) {
        f = enterEdge(t, g, e);
        exchangeEdges(t, g, e, f);
      }
    }

    /*
     * Initializes cut values for all edges in the tree.
     */
    function initCutValues(t, g) {
      var vs = postorder$1(t, t.nodes());
      vs = vs.slice(0, vs.length - 1);
      lodash_1$1.forEach(vs, function(v) {
        assignCutValue(t, g, v);
      });
    }

    function assignCutValue(t, g, child) {
      var childLab = t.node(child);
      var parent = childLab.parent;
      t.edge(child, parent).cutvalue = calcCutValue(t, g, child);
    }

    /*
     * Given the tight tree, its graph, and a child in the graph calculate and
     * return the cut value for the edge between the child and its parent.
     */
    function calcCutValue(t, g, child) {
      var childLab = t.node(child);
      var parent = childLab.parent;
      // True if the child is on the tail end of the edge in the directed graph
      var childIsTail = true;
      // The graph's view of the tree edge we're inspecting
      var graphEdge = g.edge(child, parent);
      // The accumulated cut value for the edge between this node and its parent
      var cutValue = 0;

      if (!graphEdge) {
        childIsTail = false;
        graphEdge = g.edge(parent, child);
      }

      cutValue = graphEdge.weight;

      lodash_1$1.forEach(g.nodeEdges(child), function(e) {
        var isOutEdge = e.v === child,
          other = isOutEdge ? e.w : e.v;

        if (other !== parent) {
          var pointsToHead = isOutEdge === childIsTail,
            otherWeight = g.edge(e).weight;

          cutValue += pointsToHead ? otherWeight : -otherWeight;
          if (isTreeEdge(t, child, other)) {
            var otherCutValue = t.edge(child, other).cutvalue;
            cutValue += pointsToHead ? -otherCutValue : otherCutValue;
          }
        }
      });

      return cutValue;
    }

    function initLowLimValues(tree, root) {
      if (arguments.length < 2) {
        root = tree.nodes()[0];
      }
      dfsAssignLowLim(tree, {}, 1, root);
    }

    function dfsAssignLowLim(tree, visited, nextLim, v, parent) {
      var low = nextLim;
      var label = tree.node(v);

      visited[v] = true;
      lodash_1$1.forEach(tree.neighbors(v), function(w) {
        if (!lodash_1$1.has(visited, w)) {
          nextLim = dfsAssignLowLim(tree, visited, nextLim, w, v);
        }
      });

      label.low = low;
      label.lim = nextLim++;
      if (parent) {
        label.parent = parent;
      } else {
        // TODO should be able to remove this when we incrementally update low lim
        delete label.parent;
      }

      return nextLim;
    }

    function leaveEdge(tree) {
      return lodash_1$1.find(tree.edges(), function(e) {
        return tree.edge(e).cutvalue < 0;
      });
    }

    function enterEdge(t, g, edge) {
      var v = edge.v;
      var w = edge.w;

      // For the rest of this function we assume that v is the tail and w is the
      // head, so if we don't have this edge in the graph we should flip it to
      // match the correct orientation.
      if (!g.hasEdge(v, w)) {
        v = edge.w;
        w = edge.v;
      }

      var vLabel = t.node(v);
      var wLabel = t.node(w);
      var tailLabel = vLabel;
      var flip = false;

      // If the root is in the tail of the edge then we need to flip the logic that
      // checks for the head and tail nodes in the candidates function below.
      if (vLabel.lim > wLabel.lim) {
        tailLabel = wLabel;
        flip = true;
      }

      var candidates = lodash_1$1.filter(g.edges(), function(edge) {
        return flip === isDescendant$1(t, t.node(edge.v), tailLabel) &&
               flip !== isDescendant$1(t, t.node(edge.w), tailLabel);
      });

      return lodash_1$1.minBy(candidates, function(edge) { return slack$2(g, edge); });
    }

    function exchangeEdges(t, g, e, f) {
      var v = e.v;
      var w = e.w;
      t.removeEdge(v, w);
      t.setEdge(f.v, f.w, {});
      initLowLimValues(t);
      initCutValues(t, g);
      updateRanks(t, g);
    }

    function updateRanks(t, g) {
      var root = lodash_1$1.find(t.nodes(), function(v) { return !g.node(v).parent; });
      var vs = preorder$1(t, root);
      vs = vs.slice(1);
      lodash_1$1.forEach(vs, function(v) {
        var parent = t.node(v).parent,
          edge = g.edge(v, parent),
          flipped = false;

        if (!edge) {
          edge = g.edge(parent, v);
          flipped = true;
        }

        g.node(v).rank = g.node(parent).rank + (flipped ? edge.minlen : -edge.minlen);
      });
    }

    /*
     * Returns true if the edge is in the tree.
     */
    function isTreeEdge(tree, u, v) {
      return tree.hasEdge(u, v);
    }

    /*
     * Returns true if the specified node is descendant of the root node per the
     * assigned low and lim attributes in the tree.
     */
    function isDescendant$1(tree, vLabel, rootLabel) {
      return rootLabel.low <= vLabel.lim && vLabel.lim <= rootLabel.lim;
    }

    var longestPath$1 = util$1.longestPath;



    var rank_1 = rank;

    /*
     * Assigns a rank to each node in the input graph that respects the "minlen"
     * constraint specified on edges between nodes.
     *
     * This basic structure is derived from Gansner, et al., "A Technique for
     * Drawing Directed Graphs."
     *
     * Pre-conditions:
     *
     *    1. Graph must be a connected DAG
     *    2. Graph nodes must be objects
     *    3. Graph edges must have "weight" and "minlen" attributes
     *
     * Post-conditions:
     *
     *    1. Graph nodes will have a "rank" attribute based on the results of the
     *       algorithm. Ranks can start at any index (including negative), we'll
     *       fix them up later.
     */
    function rank(g) {
      switch(g.graph().ranker) {
      case "network-simplex": networkSimplexRanker(g); break;
      case "tight-tree": tightTreeRanker(g); break;
      case "longest-path": longestPathRanker(g); break;
      default: networkSimplexRanker(g);
      }
    }

    // A fast and simple ranker, but results are far from optimal.
    var longestPathRanker = longestPath$1;

    function tightTreeRanker(g) {
      longestPath$1(g);
      feasibleTree_1(g);
    }

    function networkSimplexRanker(g) {
      networkSimplex_1(g);
    }

    var parentDummyChains_1 = parentDummyChains;

    function parentDummyChains(g) {
      var postorderNums = postorder$2(g);

      lodash_1$1.forEach(g.graph().dummyChains, function(v) {
        var node = g.node(v);
        var edgeObj = node.edgeObj;
        var pathData = findPath(g, postorderNums, edgeObj.v, edgeObj.w);
        var path = pathData.path;
        var lca = pathData.lca;
        var pathIdx = 0;
        var pathV = path[pathIdx];
        var ascending = true;

        while (v !== edgeObj.w) {
          node = g.node(v);

          if (ascending) {
            while ((pathV = path[pathIdx]) !== lca &&
                   g.node(pathV).maxRank < node.rank) {
              pathIdx++;
            }

            if (pathV === lca) {
              ascending = false;
            }
          }

          if (!ascending) {
            while (pathIdx < path.length - 1 &&
                   g.node(pathV = path[pathIdx + 1]).minRank <= node.rank) {
              pathIdx++;
            }
            pathV = path[pathIdx];
          }

          g.setParent(v, pathV);
          v = g.successors(v)[0];
        }
      });
    }

    // Find a path from v to w through the lowest common ancestor (LCA). Return the
    // full path and the LCA.
    function findPath(g, postorderNums, v, w) {
      var vPath = [];
      var wPath = [];
      var low = Math.min(postorderNums[v].low, postorderNums[w].low);
      var lim = Math.max(postorderNums[v].lim, postorderNums[w].lim);
      var parent;
      var lca;

      // Traverse up from v to find the LCA
      parent = v;
      do {
        parent = g.parent(parent);
        vPath.push(parent);
      } while (parent &&
               (postorderNums[parent].low > low || lim > postorderNums[parent].lim));
      lca = parent;

      // Traverse from w to LCA
      parent = w;
      while ((parent = g.parent(parent)) !== lca) {
        wPath.push(parent);
      }

      return { path: vPath.concat(wPath.reverse()), lca: lca };
    }

    function postorder$2(g) {
      var result = {};
      var lim = 0;

      function dfs(v) {
        var low = lim;
        lodash_1$1.forEach(g.children(v), dfs);
        result[v] = { low: low, lim: lim++ };
      }
      lodash_1$1.forEach(g.children(), dfs);

      return result;
    }

    var nestingGraph = {
      run: run$2,
      cleanup: cleanup
    };

    /*
     * A nesting graph creates dummy nodes for the tops and bottoms of subgraphs,
     * adds appropriate edges to ensure that all cluster nodes are placed between
     * these boundries, and ensures that the graph is connected.
     *
     * In addition we ensure, through the use of the minlen property, that nodes
     * and subgraph border nodes to not end up on the same rank.
     *
     * Preconditions:
     *
     *    1. Input graph is a DAG
     *    2. Nodes in the input graph has a minlen attribute
     *
     * Postconditions:
     *
     *    1. Input graph is connected.
     *    2. Dummy nodes are added for the tops and bottoms of subgraphs.
     *    3. The minlen attribute for nodes is adjusted to ensure nodes do not
     *       get placed on the same rank as subgraph border nodes.
     *
     * The nesting graph idea comes from Sander, "Layout of Compound Directed
     * Graphs."
     */
    function run$2(g) {
      var root = util.addDummyNode(g, "root", {}, "_root");
      var depths = treeDepths(g);
      var height = lodash_1$1.max(lodash_1$1.values(depths)) - 1; // Note: depths is an Object not an array
      var nodeSep = 2 * height + 1;

      g.graph().nestingRoot = root;

      // Multiply minlen by nodeSep to align nodes on non-border ranks.
      lodash_1$1.forEach(g.edges(), function(e) { g.edge(e).minlen *= nodeSep; });

      // Calculate a weight that is sufficient to keep subgraphs vertically compact
      var weight = sumWeights(g) + 1;

      // Create border nodes and link them up
      lodash_1$1.forEach(g.children(), function(child) {
        dfs$1(g, root, nodeSep, weight, height, depths, child);
      });

      // Save the multiplier for node layers for later removal of empty border
      // layers.
      g.graph().nodeRankFactor = nodeSep;
    }

    function dfs$1(g, root, nodeSep, weight, height, depths, v) {
      var children = g.children(v);
      if (!children.length) {
        if (v !== root) {
          g.setEdge(root, v, { weight: 0, minlen: nodeSep });
        }
        return;
      }

      var top = util.addBorderNode(g, "_bt");
      var bottom = util.addBorderNode(g, "_bb");
      var label = g.node(v);

      g.setParent(top, v);
      label.borderTop = top;
      g.setParent(bottom, v);
      label.borderBottom = bottom;

      lodash_1$1.forEach(children, function(child) {
        dfs$1(g, root, nodeSep, weight, height, depths, child);

        var childNode = g.node(child);
        var childTop = childNode.borderTop ? childNode.borderTop : child;
        var childBottom = childNode.borderBottom ? childNode.borderBottom : child;
        var thisWeight = childNode.borderTop ? weight : 2 * weight;
        var minlen = childTop !== childBottom ? 1 : height - depths[v] + 1;

        g.setEdge(top, childTop, {
          weight: thisWeight,
          minlen: minlen,
          nestingEdge: true
        });

        g.setEdge(childBottom, bottom, {
          weight: thisWeight,
          minlen: minlen,
          nestingEdge: true
        });
      });

      if (!g.parent(v)) {
        g.setEdge(root, top, { weight: 0, minlen: height + depths[v] });
      }
    }

    function treeDepths(g) {
      var depths = {};
      function dfs(v, depth) {
        var children = g.children(v);
        if (children && children.length) {
          lodash_1$1.forEach(children, function(child) {
            dfs(child, depth + 1);
          });
        }
        depths[v] = depth;
      }
      lodash_1$1.forEach(g.children(), function(v) { dfs(v, 1); });
      return depths;
    }

    function sumWeights(g) {
      return lodash_1$1.reduce(g.edges(), function(acc, e) {
        return acc + g.edge(e).weight;
      }, 0);
    }

    function cleanup(g) {
      var graphLabel = g.graph();
      g.removeNode(graphLabel.nestingRoot);
      delete graphLabel.nestingRoot;
      lodash_1$1.forEach(g.edges(), function(e) {
        var edge = g.edge(e);
        if (edge.nestingEdge) {
          g.removeEdge(e);
        }
      });
    }

    var addBorderSegments_1 = addBorderSegments;

    function addBorderSegments(g) {
      function dfs(v) {
        var children = g.children(v);
        var node = g.node(v);
        if (children.length) {
          lodash_1$1.forEach(children, dfs);
        }

        if (lodash_1$1.has(node, "minRank")) {
          node.borderLeft = [];
          node.borderRight = [];
          for (var rank = node.minRank, maxRank = node.maxRank + 1;
            rank < maxRank;
            ++rank) {
            addBorderNode$1(g, "borderLeft", "_bl", v, node, rank);
            addBorderNode$1(g, "borderRight", "_br", v, node, rank);
          }
        }
      }

      lodash_1$1.forEach(g.children(), dfs);
    }

    function addBorderNode$1(g, prop, prefix, sg, sgNode, rank) {
      var label = { width: 0, height: 0, rank: rank, borderType: prop };
      var prev = sgNode[prop][rank - 1];
      var curr = util.addDummyNode(g, "border", label, prefix);
      sgNode[prop][rank] = curr;
      g.setParent(curr, sg);
      if (prev) {
        g.setEdge(prev, curr, { weight: 1 });
      }
    }

    var coordinateSystem = {
      adjust: adjust,
      undo: undo$2
    };

    function adjust(g) {
      var rankDir = g.graph().rankdir.toLowerCase();
      if (rankDir === "lr" || rankDir === "rl") {
        swapWidthHeight(g);
      }
    }

    function undo$2(g) {
      var rankDir = g.graph().rankdir.toLowerCase();
      if (rankDir === "bt" || rankDir === "rl") {
        reverseY(g);
      }

      if (rankDir === "lr" || rankDir === "rl") {
        swapXY(g);
        swapWidthHeight(g);
      }
    }

    function swapWidthHeight(g) {
      lodash_1$1.forEach(g.nodes(), function(v) { swapWidthHeightOne(g.node(v)); });
      lodash_1$1.forEach(g.edges(), function(e) { swapWidthHeightOne(g.edge(e)); });
    }

    function swapWidthHeightOne(attrs) {
      var w = attrs.width;
      attrs.width = attrs.height;
      attrs.height = w;
    }

    function reverseY(g) {
      lodash_1$1.forEach(g.nodes(), function(v) { reverseYOne(g.node(v)); });

      lodash_1$1.forEach(g.edges(), function(e) {
        var edge = g.edge(e);
        lodash_1$1.forEach(edge.points, reverseYOne);
        if (lodash_1$1.has(edge, "y")) {
          reverseYOne(edge);
        }
      });
    }

    function reverseYOne(attrs) {
      attrs.y = -attrs.y;
    }

    function swapXY(g) {
      lodash_1$1.forEach(g.nodes(), function(v) { swapXYOne(g.node(v)); });

      lodash_1$1.forEach(g.edges(), function(e) {
        var edge = g.edge(e);
        lodash_1$1.forEach(edge.points, swapXYOne);
        if (lodash_1$1.has(edge, "x")) {
          swapXYOne(edge);
        }
      });
    }

    function swapXYOne(attrs) {
      var x = attrs.x;
      attrs.x = attrs.y;
      attrs.y = x;
    }

    var initOrder_1 = initOrder;

    /*
     * Assigns an initial order value for each node by performing a DFS search
     * starting from nodes in the first rank. Nodes are assigned an order in their
     * rank as they are first visited.
     *
     * This approach comes from Gansner, et al., "A Technique for Drawing Directed
     * Graphs."
     *
     * Returns a layering matrix with an array per layer and each layer sorted by
     * the order of its nodes.
     */
    function initOrder(g) {
      var visited = {};
      var simpleNodes = lodash_1$1.filter(g.nodes(), function(v) {
        return !g.children(v).length;
      });
      var maxRank = lodash_1$1.max(lodash_1$1.map(simpleNodes, function(v) { return g.node(v).rank; }));
      var layers = lodash_1$1.map(lodash_1$1.range(maxRank + 1), function() { return []; });

      function dfs(v) {
        if (lodash_1$1.has(visited, v)) return;
        visited[v] = true;
        var node = g.node(v);
        layers[node.rank].push(v);
        lodash_1$1.forEach(g.successors(v), dfs);
      }

      var orderedVs = lodash_1$1.sortBy(simpleNodes, function(v) { return g.node(v).rank; });
      lodash_1$1.forEach(orderedVs, dfs);

      return layers;
    }

    var crossCount_1 = crossCount;

    /*
     * A function that takes a layering (an array of layers, each with an array of
     * ordererd nodes) and a graph and returns a weighted crossing count.
     *
     * Pre-conditions:
     *
     *    1. Input graph must be simple (not a multigraph), directed, and include
     *       only simple edges.
     *    2. Edges in the input graph must have assigned weights.
     *
     * Post-conditions:
     *
     *    1. The graph and layering matrix are left unchanged.
     *
     * This algorithm is derived from Barth, et al., "Bilayer Cross Counting."
     */
    function crossCount(g, layering) {
      var cc = 0;
      for (var i = 1; i < layering.length; ++i) {
        cc += twoLayerCrossCount(g, layering[i-1], layering[i]);
      }
      return cc;
    }

    function twoLayerCrossCount(g, northLayer, southLayer) {
      // Sort all of the edges between the north and south layers by their position
      // in the north layer and then the south. Map these edges to the position of
      // their head in the south layer.
      var southPos = lodash_1$1.zipObject(southLayer,
        lodash_1$1.map(southLayer, function (v, i) { return i; }));
      var southEntries = lodash_1$1.flatten(lodash_1$1.map(northLayer, function(v) {
        return lodash_1$1.sortBy(lodash_1$1.map(g.outEdges(v), function(e) {
          return { pos: southPos[e.w], weight: g.edge(e).weight };
        }), "pos");
      }), true);

      // Build the accumulator tree
      var firstIndex = 1;
      while (firstIndex < southLayer.length) firstIndex <<= 1;
      var treeSize = 2 * firstIndex - 1;
      firstIndex -= 1;
      var tree = lodash_1$1.map(new Array(treeSize), function() { return 0; });

      // Calculate the weighted crossings
      var cc = 0;
      lodash_1$1.forEach(southEntries.forEach(function(entry) {
        var index = entry.pos + firstIndex;
        tree[index] += entry.weight;
        var weightSum = 0;
        while (index > 0) {
          if (index % 2) {
            weightSum += tree[index + 1];
          }
          index = (index - 1) >> 1;
          tree[index] += entry.weight;
        }
        cc += entry.weight * weightSum;
      }));

      return cc;
    }

    var barycenter_1 = barycenter;

    function barycenter(g, movable) {
      return lodash_1$1.map(movable, function(v) {
        var inV = g.inEdges(v);
        if (!inV.length) {
          return { v: v };
        } else {
          var result = lodash_1$1.reduce(inV, function(acc, e) {
            var edge = g.edge(e),
              nodeU = g.node(e.v);
            return {
              sum: acc.sum + (edge.weight * nodeU.order),
              weight: acc.weight + edge.weight
            };
          }, { sum: 0, weight: 0 });

          return {
            v: v,
            barycenter: result.sum / result.weight,
            weight: result.weight
          };
        }
      });
    }

    var resolveConflicts_1 = resolveConflicts;

    /*
     * Given a list of entries of the form {v, barycenter, weight} and a
     * constraint graph this function will resolve any conflicts between the
     * constraint graph and the barycenters for the entries. If the barycenters for
     * an entry would violate a constraint in the constraint graph then we coalesce
     * the nodes in the conflict into a new node that respects the contraint and
     * aggregates barycenter and weight information.
     *
     * This implementation is based on the description in Forster, "A Fast and
     * Simple Hueristic for Constrained Two-Level Crossing Reduction," thought it
     * differs in some specific details.
     *
     * Pre-conditions:
     *
     *    1. Each entry has the form {v, barycenter, weight}, or if the node has
     *       no barycenter, then {v}.
     *
     * Returns:
     *
     *    A new list of entries of the form {vs, i, barycenter, weight}. The list
     *    `vs` may either be a singleton or it may be an aggregation of nodes
     *    ordered such that they do not violate constraints from the constraint
     *    graph. The property `i` is the lowest original index of any of the
     *    elements in `vs`.
     */
    function resolveConflicts(entries, cg) {
      var mappedEntries = {};
      lodash_1$1.forEach(entries, function(entry, i) {
        var tmp = mappedEntries[entry.v] = {
          indegree: 0,
          "in": [],
          out: [],
          vs: [entry.v],
          i: i
        };
        if (!lodash_1$1.isUndefined(entry.barycenter)) {
          tmp.barycenter = entry.barycenter;
          tmp.weight = entry.weight;
        }
      });

      lodash_1$1.forEach(cg.edges(), function(e) {
        var entryV = mappedEntries[e.v];
        var entryW = mappedEntries[e.w];
        if (!lodash_1$1.isUndefined(entryV) && !lodash_1$1.isUndefined(entryW)) {
          entryW.indegree++;
          entryV.out.push(mappedEntries[e.w]);
        }
      });

      var sourceSet = lodash_1$1.filter(mappedEntries, function(entry) {
        return !entry.indegree;
      });

      return doResolveConflicts(sourceSet);
    }

    function doResolveConflicts(sourceSet) {
      var entries = [];

      function handleIn(vEntry) {
        return function(uEntry) {
          if (uEntry.merged) {
            return;
          }
          if (lodash_1$1.isUndefined(uEntry.barycenter) ||
              lodash_1$1.isUndefined(vEntry.barycenter) ||
              uEntry.barycenter >= vEntry.barycenter) {
            mergeEntries(vEntry, uEntry);
          }
        };
      }

      function handleOut(vEntry) {
        return function(wEntry) {
          wEntry["in"].push(vEntry);
          if (--wEntry.indegree === 0) {
            sourceSet.push(wEntry);
          }
        };
      }

      while (sourceSet.length) {
        var entry = sourceSet.pop();
        entries.push(entry);
        lodash_1$1.forEach(entry["in"].reverse(), handleIn(entry));
        lodash_1$1.forEach(entry.out, handleOut(entry));
      }

      return lodash_1$1.map(lodash_1$1.filter(entries, function(entry) { return !entry.merged; }),
        function(entry) {
          return lodash_1$1.pick(entry, ["vs", "i", "barycenter", "weight"]);
        });

    }

    function mergeEntries(target, source) {
      var sum = 0;
      var weight = 0;

      if (target.weight) {
        sum += target.barycenter * target.weight;
        weight += target.weight;
      }

      if (source.weight) {
        sum += source.barycenter * source.weight;
        weight += source.weight;
      }

      target.vs = source.vs.concat(target.vs);
      target.barycenter = sum / weight;
      target.weight = weight;
      target.i = Math.min(source.i, target.i);
      source.merged = true;
    }

    var sort_1 = sort;

    function sort(entries, biasRight) {
      var parts = util.partition(entries, function(entry) {
        return lodash_1$1.has(entry, "barycenter");
      });
      var sortable = parts.lhs,
        unsortable = lodash_1$1.sortBy(parts.rhs, function(entry) { return -entry.i; }),
        vs = [],
        sum = 0,
        weight = 0,
        vsIndex = 0;

      sortable.sort(compareWithBias(!!biasRight));

      vsIndex = consumeUnsortable(vs, unsortable, vsIndex);

      lodash_1$1.forEach(sortable, function (entry) {
        vsIndex += entry.vs.length;
        vs.push(entry.vs);
        sum += entry.barycenter * entry.weight;
        weight += entry.weight;
        vsIndex = consumeUnsortable(vs, unsortable, vsIndex);
      });

      var result = { vs: lodash_1$1.flatten(vs, true) };
      if (weight) {
        result.barycenter = sum / weight;
        result.weight = weight;
      }
      return result;
    }

    function consumeUnsortable(vs, unsortable, index) {
      var last;
      while (unsortable.length && (last = lodash_1$1.last(unsortable)).i <= index) {
        unsortable.pop();
        vs.push(last.vs);
        index++;
      }
      return index;
    }

    function compareWithBias(bias) {
      return function(entryV, entryW) {
        if (entryV.barycenter < entryW.barycenter) {
          return -1;
        } else if (entryV.barycenter > entryW.barycenter) {
          return 1;
        }

        return !bias ? entryV.i - entryW.i : entryW.i - entryV.i;
      };
    }

    var sortSubgraph_1 = sortSubgraph;

    function sortSubgraph(g, v, cg, biasRight) {
      var movable = g.children(v);
      var node = g.node(v);
      var bl = node ? node.borderLeft : undefined;
      var br = node ? node.borderRight: undefined;
      var subgraphs = {};

      if (bl) {
        movable = lodash_1$1.filter(movable, function(w) {
          return w !== bl && w !== br;
        });
      }

      var barycenters = barycenter_1(g, movable);
      lodash_1$1.forEach(barycenters, function(entry) {
        if (g.children(entry.v).length) {
          var subgraphResult = sortSubgraph(g, entry.v, cg, biasRight);
          subgraphs[entry.v] = subgraphResult;
          if (lodash_1$1.has(subgraphResult, "barycenter")) {
            mergeBarycenters(entry, subgraphResult);
          }
        }
      });

      var entries = resolveConflicts_1(barycenters, cg);
      expandSubgraphs(entries, subgraphs);

      var result = sort_1(entries, biasRight);

      if (bl) {
        result.vs = lodash_1$1.flatten([bl, result.vs, br], true);
        if (g.predecessors(bl).length) {
          var blPred = g.node(g.predecessors(bl)[0]),
            brPred = g.node(g.predecessors(br)[0]);
          if (!lodash_1$1.has(result, "barycenter")) {
            result.barycenter = 0;
            result.weight = 0;
          }
          result.barycenter = (result.barycenter * result.weight +
                               blPred.order + brPred.order) / (result.weight + 2);
          result.weight += 2;
        }
      }

      return result;
    }

    function expandSubgraphs(entries, subgraphs) {
      lodash_1$1.forEach(entries, function(entry) {
        entry.vs = lodash_1$1.flatten(entry.vs.map(function(v) {
          if (subgraphs[v]) {
            return subgraphs[v].vs;
          }
          return v;
        }), true);
      });
    }

    function mergeBarycenters(target, other) {
      if (!lodash_1$1.isUndefined(target.barycenter)) {
        target.barycenter = (target.barycenter * target.weight +
                             other.barycenter * other.weight) /
                            (target.weight + other.weight);
        target.weight += other.weight;
      } else {
        target.barycenter = other.barycenter;
        target.weight = other.weight;
      }
    }

    var Graph$4 = graphlib_1.Graph;

    var buildLayerGraph_1 = buildLayerGraph;

    /*
     * Constructs a graph that can be used to sort a layer of nodes. The graph will
     * contain all base and subgraph nodes from the request layer in their original
     * hierarchy and any edges that are incident on these nodes and are of the type
     * requested by the "relationship" parameter.
     *
     * Nodes from the requested rank that do not have parents are assigned a root
     * node in the output graph, which is set in the root graph attribute. This
     * makes it easy to walk the hierarchy of movable nodes during ordering.
     *
     * Pre-conditions:
     *
     *    1. Input graph is a DAG
     *    2. Base nodes in the input graph have a rank attribute
     *    3. Subgraph nodes in the input graph has minRank and maxRank attributes
     *    4. Edges have an assigned weight
     *
     * Post-conditions:
     *
     *    1. Output graph has all nodes in the movable rank with preserved
     *       hierarchy.
     *    2. Root nodes in the movable layer are made children of the node
     *       indicated by the root attribute of the graph.
     *    3. Non-movable nodes incident on movable nodes, selected by the
     *       relationship parameter, are included in the graph (without hierarchy).
     *    4. Edges incident on movable nodes, selected by the relationship
     *       parameter, are added to the output graph.
     *    5. The weights for copied edges are aggregated as need, since the output
     *       graph is not a multi-graph.
     */
    function buildLayerGraph(g, rank, relationship) {
      var root = createRootNode(g),
        result = new Graph$4({ compound: true }).setGraph({ root: root })
          .setDefaultNodeLabel(function(v) { return g.node(v); });

      lodash_1$1.forEach(g.nodes(), function(v) {
        var node = g.node(v),
          parent = g.parent(v);

        if (node.rank === rank || node.minRank <= rank && rank <= node.maxRank) {
          result.setNode(v);
          result.setParent(v, parent || root);

          // This assumes we have only short edges!
          lodash_1$1.forEach(g[relationship](v), function(e) {
            var u = e.v === v ? e.w : e.v,
              edge = result.edge(u, v),
              weight = !lodash_1$1.isUndefined(edge) ? edge.weight : 0;
            result.setEdge(u, v, { weight: g.edge(e).weight + weight });
          });

          if (lodash_1$1.has(node, "minRank")) {
            result.setNode(v, {
              borderLeft: node.borderLeft[rank],
              borderRight: node.borderRight[rank]
            });
          }
        }
      });

      return result;
    }

    function createRootNode(g) {
      var v;
      while (g.hasNode((v = lodash_1$1.uniqueId("_root"))));
      return v;
    }

    var addSubgraphConstraints_1 = addSubgraphConstraints;

    function addSubgraphConstraints(g, cg, vs) {
      var prev = {},
        rootPrev;

      lodash_1$1.forEach(vs, function(v) {
        var child = g.parent(v),
          parent,
          prevChild;
        while (child) {
          parent = g.parent(child);
          if (parent) {
            prevChild = prev[parent];
            prev[parent] = child;
          } else {
            prevChild = rootPrev;
            rootPrev = child;
          }
          if (prevChild && prevChild !== child) {
            cg.setEdge(prevChild, child);
            return;
          }
          child = parent;
        }
      });

      /*
      function dfs(v) {
        var children = v ? g.children(v) : g.children();
        if (children.length) {
          var min = Number.POSITIVE_INFINITY,
              subgraphs = [];
          _.each(children, function(child) {
            var childMin = dfs(child);
            if (g.children(child).length) {
              subgraphs.push({ v: child, order: childMin });
            }
            min = Math.min(min, childMin);
          });
          _.reduce(_.sortBy(subgraphs, "order"), function(prev, curr) {
            cg.setEdge(prev.v, curr.v);
            return curr;
          });
          return min;
        }
        return g.node(v).order;
      }
      dfs(undefined);
      */
    }

    var Graph$5 = graphlib_1.Graph;


    var order_1 = order;

    /*
     * Applies heuristics to minimize edge crossings in the graph and sets the best
     * order solution as an order attribute on each node.
     *
     * Pre-conditions:
     *
     *    1. Graph must be DAG
     *    2. Graph nodes must be objects with a "rank" attribute
     *    3. Graph edges must have the "weight" attribute
     *
     * Post-conditions:
     *
     *    1. Graph nodes will have an "order" attribute based on the results of the
     *       algorithm.
     */
    function order(g) {
      var maxRank = util.maxRank(g),
        downLayerGraphs = buildLayerGraphs(g, lodash_1$1.range(1, maxRank + 1), "inEdges"),
        upLayerGraphs = buildLayerGraphs(g, lodash_1$1.range(maxRank - 1, -1, -1), "outEdges");

      var layering = initOrder_1(g);
      assignOrder(g, layering);

      var bestCC = Number.POSITIVE_INFINITY,
        best;

      for (var i = 0, lastBest = 0; lastBest < 4; ++i, ++lastBest) {
        sweepLayerGraphs(i % 2 ? downLayerGraphs : upLayerGraphs, i % 4 >= 2);

        layering = util.buildLayerMatrix(g);
        var cc = crossCount_1(g, layering);
        if (cc < bestCC) {
          lastBest = 0;
          best = lodash_1$1.cloneDeep(layering);
          bestCC = cc;
        }
      }

      assignOrder(g, best);
    }

    function buildLayerGraphs(g, ranks, relationship) {
      return lodash_1$1.map(ranks, function(rank) {
        return buildLayerGraph_1(g, rank, relationship);
      });
    }

    function sweepLayerGraphs(layerGraphs, biasRight) {
      var cg = new Graph$5();
      lodash_1$1.forEach(layerGraphs, function(lg) {
        var root = lg.graph().root;
        var sorted = sortSubgraph_1(lg, root, cg, biasRight);
        lodash_1$1.forEach(sorted.vs, function(v, i) {
          lg.node(v).order = i;
        });
        addSubgraphConstraints_1(lg, cg, sorted.vs);
      });
    }

    function assignOrder(g, layering) {
      lodash_1$1.forEach(layering, function(layer) {
        lodash_1$1.forEach(layer, function(v, i) {
          g.node(v).order = i;
        });
      });
    }

    var Graph$6 = graphlib_1.Graph;


    /*
     * This module provides coordinate assignment based on Brandes and Köpf, "Fast
     * and Simple Horizontal Coordinate Assignment."
     */

    var bk = {
      positionX: positionX,
      findType1Conflicts: findType1Conflicts,
      findType2Conflicts: findType2Conflicts,
      addConflict: addConflict,
      hasConflict: hasConflict,
      verticalAlignment: verticalAlignment,
      horizontalCompaction: horizontalCompaction,
      alignCoordinates: alignCoordinates,
      findSmallestWidthAlignment: findSmallestWidthAlignment,
      balance: balance
    };

    /*
     * Marks all edges in the graph with a type-1 conflict with the "type1Conflict"
     * property. A type-1 conflict is one where a non-inner segment crosses an
     * inner segment. An inner segment is an edge with both incident nodes marked
     * with the "dummy" property.
     *
     * This algorithm scans layer by layer, starting with the second, for type-1
     * conflicts between the current layer and the previous layer. For each layer
     * it scans the nodes from left to right until it reaches one that is incident
     * on an inner segment. It then scans predecessors to determine if they have
     * edges that cross that inner segment. At the end a final scan is done for all
     * nodes on the current rank to see if they cross the last visited inner
     * segment.
     *
     * This algorithm (safely) assumes that a dummy node will only be incident on a
     * single node in the layers being scanned.
     */
    function findType1Conflicts(g, layering) {
      var conflicts = {};

      function visitLayer(prevLayer, layer) {
        var
          // last visited node in the previous layer that is incident on an inner
          // segment.
          k0 = 0,
          // Tracks the last node in this layer scanned for crossings with a type-1
          // segment.
          scanPos = 0,
          prevLayerLength = prevLayer.length,
          lastNode = lodash_1$1.last(layer);

        lodash_1$1.forEach(layer, function(v, i) {
          var w = findOtherInnerSegmentNode(g, v),
            k1 = w ? g.node(w).order : prevLayerLength;

          if (w || v === lastNode) {
            lodash_1$1.forEach(layer.slice(scanPos, i +1), function(scanNode) {
              lodash_1$1.forEach(g.predecessors(scanNode), function(u) {
                var uLabel = g.node(u),
                  uPos = uLabel.order;
                if ((uPos < k0 || k1 < uPos) &&
                    !(uLabel.dummy && g.node(scanNode).dummy)) {
                  addConflict(conflicts, u, scanNode);
                }
              });
            });
            scanPos = i + 1;
            k0 = k1;
          }
        });

        return layer;
      }

      lodash_1$1.reduce(layering, visitLayer);
      return conflicts;
    }

    function findType2Conflicts(g, layering) {
      var conflicts = {};

      function scan(south, southPos, southEnd, prevNorthBorder, nextNorthBorder) {
        var v;
        lodash_1$1.forEach(lodash_1$1.range(southPos, southEnd), function(i) {
          v = south[i];
          if (g.node(v).dummy) {
            lodash_1$1.forEach(g.predecessors(v), function(u) {
              var uNode = g.node(u);
              if (uNode.dummy &&
                  (uNode.order < prevNorthBorder || uNode.order > nextNorthBorder)) {
                addConflict(conflicts, u, v);
              }
            });
          }
        });
      }


      function visitLayer(north, south) {
        var prevNorthPos = -1,
          nextNorthPos,
          southPos = 0;

        lodash_1$1.forEach(south, function(v, southLookahead) {
          if (g.node(v).dummy === "border") {
            var predecessors = g.predecessors(v);
            if (predecessors.length) {
              nextNorthPos = g.node(predecessors[0]).order;
              scan(south, southPos, southLookahead, prevNorthPos, nextNorthPos);
              southPos = southLookahead;
              prevNorthPos = nextNorthPos;
            }
          }
          scan(south, southPos, south.length, nextNorthPos, north.length);
        });

        return south;
      }

      lodash_1$1.reduce(layering, visitLayer);
      return conflicts;
    }

    function findOtherInnerSegmentNode(g, v) {
      if (g.node(v).dummy) {
        return lodash_1$1.find(g.predecessors(v), function(u) {
          return g.node(u).dummy;
        });
      }
    }

    function addConflict(conflicts, v, w) {
      if (v > w) {
        var tmp = v;
        v = w;
        w = tmp;
      }

      var conflictsV = conflicts[v];
      if (!conflictsV) {
        conflicts[v] = conflictsV = {};
      }
      conflictsV[w] = true;
    }

    function hasConflict(conflicts, v, w) {
      if (v > w) {
        var tmp = v;
        v = w;
        w = tmp;
      }
      return lodash_1$1.has(conflicts[v], w);
    }

    /*
     * Try to align nodes into vertical "blocks" where possible. This algorithm
     * attempts to align a node with one of its median neighbors. If the edge
     * connecting a neighbor is a type-1 conflict then we ignore that possibility.
     * If a previous node has already formed a block with a node after the node
     * we're trying to form a block with, we also ignore that possibility - our
     * blocks would be split in that scenario.
     */
    function verticalAlignment(g, layering, conflicts, neighborFn) {
      var root = {},
        align = {},
        pos = {};

      // We cache the position here based on the layering because the graph and
      // layering may be out of sync. The layering matrix is manipulated to
      // generate different extreme alignments.
      lodash_1$1.forEach(layering, function(layer) {
        lodash_1$1.forEach(layer, function(v, order) {
          root[v] = v;
          align[v] = v;
          pos[v] = order;
        });
      });

      lodash_1$1.forEach(layering, function(layer) {
        var prevIdx = -1;
        lodash_1$1.forEach(layer, function(v) {
          var ws = neighborFn(v);
          if (ws.length) {
            ws = lodash_1$1.sortBy(ws, function(w) { return pos[w]; });
            var mp = (ws.length - 1) / 2;
            for (var i = Math.floor(mp), il = Math.ceil(mp); i <= il; ++i) {
              var w = ws[i];
              if (align[v] === v &&
                  prevIdx < pos[w] &&
                  !hasConflict(conflicts, v, w)) {
                align[w] = v;
                align[v] = root[v] = root[w];
                prevIdx = pos[w];
              }
            }
          }
        });
      });

      return { root: root, align: align };
    }

    function horizontalCompaction(g, layering, root, align, reverseSep) {
      // This portion of the algorithm differs from BK due to a number of problems.
      // Instead of their algorithm we construct a new block graph and do two
      // sweeps. The first sweep places blocks with the smallest possible
      // coordinates. The second sweep removes unused space by moving blocks to the
      // greatest coordinates without violating separation.
      var xs = {},
        blockG = buildBlockGraph(g, layering, root, reverseSep),
        borderType = reverseSep ? "borderLeft" : "borderRight";

      function iterate(setXsFunc, nextNodesFunc) {
        var stack = blockG.nodes();
        var elem = stack.pop();
        var visited = {};
        while (elem) {
          if (visited[elem]) {
            setXsFunc(elem);
          } else {
            visited[elem] = true;
            stack.push(elem);
            stack = stack.concat(nextNodesFunc(elem));
          }

          elem = stack.pop();
        }
      }

      // First pass, assign smallest coordinates
      function pass1(elem) {
        xs[elem] = blockG.inEdges(elem).reduce(function(acc, e) {
          return Math.max(acc, xs[e.v] + blockG.edge(e));
        }, 0);
      }

      // Second pass, assign greatest coordinates
      function pass2(elem) {
        var min = blockG.outEdges(elem).reduce(function(acc, e) {
          return Math.min(acc, xs[e.w] - blockG.edge(e));
        }, Number.POSITIVE_INFINITY);

        var node = g.node(elem);
        if (min !== Number.POSITIVE_INFINITY && node.borderType !== borderType) {
          xs[elem] = Math.max(xs[elem], min);
        }
      }

      iterate(pass1, blockG.predecessors.bind(blockG));
      iterate(pass2, blockG.successors.bind(blockG));

      // Assign x coordinates to all nodes
      lodash_1$1.forEach(align, function(v) {
        xs[v] = xs[root[v]];
      });

      return xs;
    }


    function buildBlockGraph(g, layering, root, reverseSep) {
      var blockGraph = new Graph$6(),
        graphLabel = g.graph(),
        sepFn = sep(graphLabel.nodesep, graphLabel.edgesep, reverseSep);

      lodash_1$1.forEach(layering, function(layer) {
        var u;
        lodash_1$1.forEach(layer, function(v) {
          var vRoot = root[v];
          blockGraph.setNode(vRoot);
          if (u) {
            var uRoot = root[u],
              prevMax = blockGraph.edge(uRoot, vRoot);
            blockGraph.setEdge(uRoot, vRoot, Math.max(sepFn(g, v, u), prevMax || 0));
          }
          u = v;
        });
      });

      return blockGraph;
    }

    /*
     * Returns the alignment that has the smallest width of the given alignments.
     */
    function findSmallestWidthAlignment(g, xss) {
      return lodash_1$1.minBy(lodash_1$1.values(xss), function (xs) {
        var max = Number.NEGATIVE_INFINITY;
        var min = Number.POSITIVE_INFINITY;

        lodash_1$1.forIn(xs, function (x, v) {
          var halfWidth = width(g, v) / 2;

          max = Math.max(x + halfWidth, max);
          min = Math.min(x - halfWidth, min);
        });

        return max - min;
      });
    }

    /*
     * Align the coordinates of each of the layout alignments such that
     * left-biased alignments have their minimum coordinate at the same point as
     * the minimum coordinate of the smallest width alignment and right-biased
     * alignments have their maximum coordinate at the same point as the maximum
     * coordinate of the smallest width alignment.
     */
    function alignCoordinates(xss, alignTo) {
      var alignToVals = lodash_1$1.values(alignTo),
        alignToMin = lodash_1$1.min(alignToVals),
        alignToMax = lodash_1$1.max(alignToVals);

      lodash_1$1.forEach(["u", "d"], function(vert) {
        lodash_1$1.forEach(["l", "r"], function(horiz) {
          var alignment = vert + horiz,
            xs = xss[alignment],
            delta;
          if (xs === alignTo) return;

          var xsVals = lodash_1$1.values(xs);
          delta = horiz === "l" ? alignToMin - lodash_1$1.min(xsVals) : alignToMax - lodash_1$1.max(xsVals);

          if (delta) {
            xss[alignment] = lodash_1$1.mapValues(xs, function(x) { return x + delta; });
          }
        });
      });
    }

    function balance(xss, align) {
      return lodash_1$1.mapValues(xss.ul, function(ignore, v) {
        if (align) {
          return xss[align.toLowerCase()][v];
        } else {
          var xs = lodash_1$1.sortBy(lodash_1$1.map(xss, v));
          return (xs[1] + xs[2]) / 2;
        }
      });
    }

    function positionX(g) {
      var layering = util.buildLayerMatrix(g);
      var conflicts = lodash_1$1.merge(
        findType1Conflicts(g, layering),
        findType2Conflicts(g, layering));

      var xss = {};
      var adjustedLayering;
      lodash_1$1.forEach(["u", "d"], function(vert) {
        adjustedLayering = vert === "u" ? layering : lodash_1$1.values(layering).reverse();
        lodash_1$1.forEach(["l", "r"], function(horiz) {
          if (horiz === "r") {
            adjustedLayering = lodash_1$1.map(adjustedLayering, function(inner) {
              return lodash_1$1.values(inner).reverse();
            });
          }

          var neighborFn = (vert === "u" ? g.predecessors : g.successors).bind(g);
          var align = verticalAlignment(g, adjustedLayering, conflicts, neighborFn);
          var xs = horizontalCompaction(g, adjustedLayering,
            align.root, align.align, horiz === "r");
          if (horiz === "r") {
            xs = lodash_1$1.mapValues(xs, function(x) { return -x; });
          }
          xss[vert + horiz] = xs;
        });
      });

      var smallestWidth = findSmallestWidthAlignment(g, xss);
      alignCoordinates(xss, smallestWidth);
      return balance(xss, g.graph().align);
    }

    function sep(nodeSep, edgeSep, reverseSep) {
      return function(g, v, w) {
        var vLabel = g.node(v);
        var wLabel = g.node(w);
        var sum = 0;
        var delta;

        sum += vLabel.width / 2;
        if (lodash_1$1.has(vLabel, "labelpos")) {
          switch (vLabel.labelpos.toLowerCase()) {
          case "l": delta = -vLabel.width / 2; break;
          case "r": delta = vLabel.width / 2; break;
          }
        }
        if (delta) {
          sum += reverseSep ? delta : -delta;
        }
        delta = 0;

        sum += (vLabel.dummy ? edgeSep : nodeSep) / 2;
        sum += (wLabel.dummy ? edgeSep : nodeSep) / 2;

        sum += wLabel.width / 2;
        if (lodash_1$1.has(wLabel, "labelpos")) {
          switch (wLabel.labelpos.toLowerCase()) {
          case "l": delta = wLabel.width / 2; break;
          case "r": delta = -wLabel.width / 2; break;
          }
        }
        if (delta) {
          sum += reverseSep ? delta : -delta;
        }
        delta = 0;

        return sum;
      };
    }

    function width(g, v) {
      return g.node(v).width;
    }

    var positionX$1 = bk.positionX;

    var position_1 = position;

    function position(g) {
      g = util.asNonCompoundGraph(g);

      positionY(g);
      lodash_1$1.forEach(positionX$1(g), function(x, v) {
        g.node(v).x = x;
      });
    }

    function positionY(g) {
      var layering = util.buildLayerMatrix(g);
      var rankSep = g.graph().ranksep;
      var prevY = 0;
      lodash_1$1.forEach(layering, function(layer) {
        var maxHeight = lodash_1$1.max(lodash_1$1.map(layer, function(v) { return g.node(v).height; }));
        lodash_1$1.forEach(layer, function(v) {
          g.node(v).y = prevY + maxHeight / 2;
        });
        prevY += maxHeight + rankSep;
      });
    }

    var normalizeRanks$1 = util.normalizeRanks;

    var removeEmptyRanks$1 = util.removeEmptyRanks;





    var util$2 = util;
    var Graph$7 = graphlib_1.Graph;

    var layout_1 = layout;

    function layout(g, opts) {
      var time = opts && opts.debugTiming ? util$2.time : util$2.notime;
      time("layout", function() {
        var layoutGraph = 
          time("  buildLayoutGraph", function() { return buildLayoutGraph(g); });
        time("  runLayout",        function() { runLayout(layoutGraph, time); });
        time("  updateInputGraph", function() { updateInputGraph(g, layoutGraph); });
      });
    }

    function runLayout(g, time) {
      time("    makeSpaceForEdgeLabels", function() { makeSpaceForEdgeLabels(g); });
      time("    removeSelfEdges",        function() { removeSelfEdges(g); });
      time("    acyclic",                function() { acyclic.run(g); });
      time("    nestingGraph.run",       function() { nestingGraph.run(g); });
      time("    rank",                   function() { rank_1(util$2.asNonCompoundGraph(g)); });
      time("    injectEdgeLabelProxies", function() { injectEdgeLabelProxies(g); });
      time("    removeEmptyRanks",       function() { removeEmptyRanks$1(g); });
      time("    nestingGraph.cleanup",   function() { nestingGraph.cleanup(g); });
      time("    normalizeRanks",         function() { normalizeRanks$1(g); });
      time("    assignRankMinMax",       function() { assignRankMinMax(g); });
      time("    removeEdgeLabelProxies", function() { removeEdgeLabelProxies(g); });
      time("    normalize.run",          function() { normalize$2.run(g); });
      time("    parentDummyChains",      function() { parentDummyChains_1(g); });
      time("    addBorderSegments",      function() { addBorderSegments_1(g); });
      time("    order",                  function() { order_1(g); });
      time("    insertSelfEdges",        function() { insertSelfEdges(g); });
      time("    adjustCoordinateSystem", function() { coordinateSystem.adjust(g); });
      time("    position",               function() { position_1(g); });
      time("    positionSelfEdges",      function() { positionSelfEdges(g); });
      time("    removeBorderNodes",      function() { removeBorderNodes(g); });
      time("    normalize.undo",         function() { normalize$2.undo(g); });
      time("    fixupEdgeLabelCoords",   function() { fixupEdgeLabelCoords(g); });
      time("    undoCoordinateSystem",   function() { coordinateSystem.undo(g); });
      time("    translateGraph",         function() { translateGraph(g); });
      time("    assignNodeIntersects",   function() { assignNodeIntersects(g); });
      time("    reversePoints",          function() { reversePointsForReversedEdges(g); });
      time("    acyclic.undo",           function() { acyclic.undo(g); });
    }

    /*
     * Copies final layout information from the layout graph back to the input
     * graph. This process only copies whitelisted attributes from the layout graph
     * to the input graph, so it serves as a good place to determine what
     * attributes can influence layout.
     */
    function updateInputGraph(inputGraph, layoutGraph) {
      lodash_1$1.forEach(inputGraph.nodes(), function(v) {
        var inputLabel = inputGraph.node(v);
        var layoutLabel = layoutGraph.node(v);

        if (inputLabel) {
          inputLabel.x = layoutLabel.x;
          inputLabel.y = layoutLabel.y;

          if (layoutGraph.children(v).length) {
            inputLabel.width = layoutLabel.width;
            inputLabel.height = layoutLabel.height;
          }
        }
      });

      lodash_1$1.forEach(inputGraph.edges(), function(e) {
        var inputLabel = inputGraph.edge(e);
        var layoutLabel = layoutGraph.edge(e);

        inputLabel.points = layoutLabel.points;
        if (lodash_1$1.has(layoutLabel, "x")) {
          inputLabel.x = layoutLabel.x;
          inputLabel.y = layoutLabel.y;
        }
      });

      inputGraph.graph().width = layoutGraph.graph().width;
      inputGraph.graph().height = layoutGraph.graph().height;
    }

    var graphNumAttrs = ["nodesep", "edgesep", "ranksep", "marginx", "marginy"];
    var graphDefaults = { ranksep: 50, edgesep: 20, nodesep: 50, rankdir: "tb" };
    var graphAttrs = ["acyclicer", "ranker", "rankdir", "align"];
    var nodeNumAttrs = ["width", "height"];
    var nodeDefaults = { width: 0, height: 0 };
    var edgeNumAttrs = ["minlen", "weight", "width", "height", "labeloffset"];
    var edgeDefaults = {
      minlen: 1, weight: 1, width: 0, height: 0,
      labeloffset: 10, labelpos: "r"
    };
    var edgeAttrs = ["labelpos"];

    /*
     * Constructs a new graph from the input graph, which can be used for layout.
     * This process copies only whitelisted attributes from the input graph to the
     * layout graph. Thus this function serves as a good place to determine what
     * attributes can influence layout.
     */
    function buildLayoutGraph(inputGraph) {
      var g = new Graph$7({ multigraph: true, compound: true });
      var graph = canonicalize(inputGraph.graph());

      g.setGraph(lodash_1$1.merge({},
        graphDefaults,
        selectNumberAttrs(graph, graphNumAttrs),
        lodash_1$1.pick(graph, graphAttrs)));

      lodash_1$1.forEach(inputGraph.nodes(), function(v) {
        var node = canonicalize(inputGraph.node(v));
        g.setNode(v, lodash_1$1.defaults(selectNumberAttrs(node, nodeNumAttrs), nodeDefaults));
        g.setParent(v, inputGraph.parent(v));
      });

      lodash_1$1.forEach(inputGraph.edges(), function(e) {
        var edge = canonicalize(inputGraph.edge(e));
        g.setEdge(e, lodash_1$1.merge({},
          edgeDefaults,
          selectNumberAttrs(edge, edgeNumAttrs),
          lodash_1$1.pick(edge, edgeAttrs)));
      });

      return g;
    }

    /*
     * This idea comes from the Gansner paper: to account for edge labels in our
     * layout we split each rank in half by doubling minlen and halving ranksep.
     * Then we can place labels at these mid-points between nodes.
     *
     * We also add some minimal padding to the width to push the label for the edge
     * away from the edge itself a bit.
     */
    function makeSpaceForEdgeLabels(g) {
      var graph = g.graph();
      graph.ranksep /= 2;
      lodash_1$1.forEach(g.edges(), function(e) {
        var edge = g.edge(e);
        edge.minlen *= 2;
        if (edge.labelpos.toLowerCase() !== "c") {
          if (graph.rankdir === "TB" || graph.rankdir === "BT") {
            edge.width += edge.labeloffset;
          } else {
            edge.height += edge.labeloffset;
          }
        }
      });
    }

    /*
     * Creates temporary dummy nodes that capture the rank in which each edge's
     * label is going to, if it has one of non-zero width and height. We do this
     * so that we can safely remove empty ranks while preserving balance for the
     * label's position.
     */
    function injectEdgeLabelProxies(g) {
      lodash_1$1.forEach(g.edges(), function(e) {
        var edge = g.edge(e);
        if (edge.width && edge.height) {
          var v = g.node(e.v);
          var w = g.node(e.w);
          var label = { rank: (w.rank - v.rank) / 2 + v.rank, e: e };
          util$2.addDummyNode(g, "edge-proxy", label, "_ep");
        }
      });
    }

    function assignRankMinMax(g) {
      var maxRank = 0;
      lodash_1$1.forEach(g.nodes(), function(v) {
        var node = g.node(v);
        if (node.borderTop) {
          node.minRank = g.node(node.borderTop).rank;
          node.maxRank = g.node(node.borderBottom).rank;
          maxRank = lodash_1$1.max(maxRank, node.maxRank);
        }
      });
      g.graph().maxRank = maxRank;
    }

    function removeEdgeLabelProxies(g) {
      lodash_1$1.forEach(g.nodes(), function(v) {
        var node = g.node(v);
        if (node.dummy === "edge-proxy") {
          g.edge(node.e).labelRank = node.rank;
          g.removeNode(v);
        }
      });
    }

    function translateGraph(g) {
      var minX = Number.POSITIVE_INFINITY;
      var maxX = 0;
      var minY = Number.POSITIVE_INFINITY;
      var maxY = 0;
      var graphLabel = g.graph();
      var marginX = graphLabel.marginx || 0;
      var marginY = graphLabel.marginy || 0;

      function getExtremes(attrs) {
        var x = attrs.x;
        var y = attrs.y;
        var w = attrs.width;
        var h = attrs.height;
        minX = Math.min(minX, x - w / 2);
        maxX = Math.max(maxX, x + w / 2);
        minY = Math.min(minY, y - h / 2);
        maxY = Math.max(maxY, y + h / 2);
      }

      lodash_1$1.forEach(g.nodes(), function(v) { getExtremes(g.node(v)); });
      lodash_1$1.forEach(g.edges(), function(e) {
        var edge = g.edge(e);
        if (lodash_1$1.has(edge, "x")) {
          getExtremes(edge);
        }
      });

      minX -= marginX;
      minY -= marginY;

      lodash_1$1.forEach(g.nodes(), function(v) {
        var node = g.node(v);
        node.x -= minX;
        node.y -= minY;
      });

      lodash_1$1.forEach(g.edges(), function(e) {
        var edge = g.edge(e);
        lodash_1$1.forEach(edge.points, function(p) {
          p.x -= minX;
          p.y -= minY;
        });
        if (lodash_1$1.has(edge, "x")) { edge.x -= minX; }
        if (lodash_1$1.has(edge, "y")) { edge.y -= minY; }
      });

      graphLabel.width = maxX - minX + marginX;
      graphLabel.height = maxY - minY + marginY;
    }

    function assignNodeIntersects(g) {
      lodash_1$1.forEach(g.edges(), function(e) {
        var edge = g.edge(e);
        var nodeV = g.node(e.v);
        var nodeW = g.node(e.w);
        var p1, p2;
        if (!edge.points) {
          edge.points = [];
          p1 = nodeW;
          p2 = nodeV;
        } else {
          p1 = edge.points[0];
          p2 = edge.points[edge.points.length - 1];
        }
        edge.points.unshift(util$2.intersectRect(nodeV, p1));
        edge.points.push(util$2.intersectRect(nodeW, p2));
      });
    }

    function fixupEdgeLabelCoords(g) {
      lodash_1$1.forEach(g.edges(), function(e) {
        var edge = g.edge(e);
        if (lodash_1$1.has(edge, "x")) {
          if (edge.labelpos === "l" || edge.labelpos === "r") {
            edge.width -= edge.labeloffset;
          }
          switch (edge.labelpos) {
          case "l": edge.x -= edge.width / 2 + edge.labeloffset; break;
          case "r": edge.x += edge.width / 2 + edge.labeloffset; break;
          }
        }
      });
    }

    function reversePointsForReversedEdges(g) {
      lodash_1$1.forEach(g.edges(), function(e) {
        var edge = g.edge(e);
        if (edge.reversed) {
          edge.points.reverse();
        }
      });
    }

    function removeBorderNodes(g) {
      lodash_1$1.forEach(g.nodes(), function(v) {
        if (g.children(v).length) {
          var node = g.node(v);
          var t = g.node(node.borderTop);
          var b = g.node(node.borderBottom);
          var l = g.node(lodash_1$1.last(node.borderLeft));
          var r = g.node(lodash_1$1.last(node.borderRight));

          node.width = Math.abs(r.x - l.x);
          node.height = Math.abs(b.y - t.y);
          node.x = l.x + node.width / 2;
          node.y = t.y + node.height / 2;
        }
      });

      lodash_1$1.forEach(g.nodes(), function(v) {
        if (g.node(v).dummy === "border") {
          g.removeNode(v);
        }
      });
    }

    function removeSelfEdges(g) {
      lodash_1$1.forEach(g.edges(), function(e) {
        if (e.v === e.w) {
          var node = g.node(e.v);
          if (!node.selfEdges) {
            node.selfEdges = [];
          }
          node.selfEdges.push({ e: e, label: g.edge(e) });
          g.removeEdge(e);
        }
      });
    }

    function insertSelfEdges(g) {
      var layers = util$2.buildLayerMatrix(g);
      lodash_1$1.forEach(layers, function(layer) {
        var orderShift = 0;
        lodash_1$1.forEach(layer, function(v, i) {
          var node = g.node(v);
          node.order = i + orderShift;
          lodash_1$1.forEach(node.selfEdges, function(selfEdge) {
            util$2.addDummyNode(g, "selfedge", {
              width: selfEdge.label.width,
              height: selfEdge.label.height,
              rank: node.rank,
              order: i + (++orderShift),
              e: selfEdge.e,
              label: selfEdge.label
            }, "_se");
          });
          delete node.selfEdges;
        });
      });
    }

    function positionSelfEdges(g) {
      lodash_1$1.forEach(g.nodes(), function(v) {
        var node = g.node(v);
        if (node.dummy === "selfedge") {
          var selfNode = g.node(node.e.v);
          var x = selfNode.x + selfNode.width / 2;
          var y = selfNode.y;
          var dx = node.x - x;
          var dy = selfNode.height / 2;
          g.setEdge(node.e, node.label);
          g.removeNode(v);
          node.label.points = [
            { x: x + 2 * dx / 3, y: y - dy },
            { x: x + 5 * dx / 6, y: y - dy },
            { x: x +     dx    , y: y },
            { x: x + 5 * dx / 6, y: y + dy },
            { x: x + 2 * dx / 3, y: y + dy }
          ];
          node.label.x = node.x;
          node.label.y = node.y;
        }
      });
    }

    function selectNumberAttrs(obj, attrs) {
      return lodash_1$1.mapValues(lodash_1$1.pick(obj, attrs), Number);
    }

    function canonicalize(attrs) {
      var newAttrs = {};
      lodash_1$1.forEach(attrs, function(v, k) {
        newAttrs[k.toLowerCase()] = v;
      });
      return newAttrs;
    }

    var Graph$8 = graphlib_1.Graph;

    var debug = {
      debugOrdering: debugOrdering
    };

    /* istanbul ignore next */
    function debugOrdering(g) {
      var layerMatrix = util.buildLayerMatrix(g);

      var h = new Graph$8({ compound: true, multigraph: true }).setGraph({});

      lodash_1$1.forEach(g.nodes(), function(v) {
        h.setNode(v, { label: v });
        h.setParent(v, "layer" + g.node(v).rank);
      });

      lodash_1$1.forEach(g.edges(), function(e) {
        h.setEdge(e.v, e.w, {}, e.name);
      });

      lodash_1$1.forEach(layerMatrix, function(layer, i) {
        var layerV = "layer" + i;
        h.setNode(layerV, { rank: "same" });
        lodash_1$1.reduce(layer, function(u, v) {
          h.setEdge(u, v, { style: "invis" });
          return v;
        });
      });

      return h;
    }

    var version$5 = "0.8.5";

    /*
    Copyright (c) 2012-2014 Chris Pettitt

    Permission is hereby granted, free of charge, to any person obtaining a copy
    of this software and associated documentation files (the "Software"), to deal
    in the Software without restriction, including without limitation the rights
    to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
    copies of the Software, and to permit persons to whom the Software is
    furnished to do so, subject to the following conditions:

    The above copyright notice and this permission notice shall be included in
    all copies or substantial portions of the Software.

    THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
    IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
    FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
    AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
    LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
    OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
    THE SOFTWARE.
    */

    var dagre = {
      graphlib: graphlib_1,

      layout: layout_1,
      debug: debug,
      util: {
        time: util.time,
        notime: util.notime
      },
      version: version$5
    };

    /* Copyright 2015 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the 'License');
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an 'AS IS' BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    const ASYNC_TASK_DELAY = 20;
    /**
     * Measure and log a synchronous task.
     */
    function time$3(msg, task) {
        let start = Date.now();
        let result = task();
        /* tslint:disable */
        console.log(msg, ':', Date.now() - start, 'ms');
        /* tslint:enable */
        return result;
    }
    /**
     * Creates a tracker that sets the progress property of the
     * provided polymer component. The provided component must have
     * a property called 'progress' that is not read-only. The progress
     * property is an object with a numerical 'value' property and a
     * string 'msg' property.
     */
    function getTracker(polymerComponent) {
        return {
            setMessage: function (msg) {
                polymerComponent.set('progress', {
                    value: polymerComponent.progress.value,
                    msg: msg,
                });
            },
            updateProgress: function (value) {
                polymerComponent.set('progress', {
                    value: polymerComponent.progress.value + value,
                    msg: polymerComponent.progress.msg,
                });
            },
            reportError: function (msg, err) {
                // Log the stack trace in the console.
                console.error(err.stack);
                // And send a user-friendly message to the UI.
                polymerComponent.set('progress', {
                    value: polymerComponent.progress.value,
                    msg: msg,
                    error: true,
                });
            },
        };
    }
    /**
     * Creates a tracker for a subtask given the parent tracker, the total
     * progress
     * of the subtask and the subtask message. The parent task should pass a
     * subtracker to its subtasks. The subtask reports its own progress which
     * becomes relative to the main task.
     */
    function getSubtaskTracker(parentTracker, impactOnTotalProgress, subtaskMsg) {
        return {
            setMessage: function (progressMsg) {
                // The parent should show a concatenation of its message along with
                // its subtask tracker message.
                parentTracker.setMessage(subtaskMsg + ': ' + progressMsg);
            },
            updateProgress: function (incrementValue) {
                // Update the parent progress relative to the child progress.
                // For example, if the sub-task progresses by 30%, and the impact on the
                // total progress is 50%, then the task progresses by 30% * 50% = 15%.
                parentTracker.updateProgress((incrementValue * impactOnTotalProgress) / 100);
            },
            reportError: function (msg, err) {
                // The parent should show a concatenation of its message along with
                // its subtask error message.
                parentTracker.reportError(subtaskMsg + ': ' + msg, err);
            },
        };
    }
    /**
     * Runs a synchronous expensive task and return the result.
     * Please use runAsyncPromiseTask in case a task returns a Promise.
     */
    function runTask(msg, incProgressValue, task, tracker) {
        // Update the progress message to say the current running task.
        tracker.setMessage(msg);
        // Run the expensive task with a delay that gives enough time for the
        // UI to update.
        try {
            let result = time$3(msg, task);
            // Update the progress value.
            tracker.updateProgress(incProgressValue);
            // Return the result to be used by other tasks.
            return result;
        }
        catch (e) {
            // Errors that happen inside asynchronous tasks are
            // reported to the tracker using a user-friendly message.
            tracker.reportError('Failed ' + msg, e);
        }
    }
    /**
     * Runs an expensive task asynchronously and returns a promise of the result.
     */
    function runAsyncTask(msg, incProgressValue, task, tracker) {
        return new Promise((resolve, reject) => {
            // Update the progress message to say the current running task.
            tracker.setMessage(msg);
            // Run the expensive task with a delay that gives enough time for the
            // UI to update.
            setTimeout(function () {
                try {
                    let result = time$3(msg, task);
                    // Update the progress value.
                    tracker.updateProgress(incProgressValue);
                    // Return the result to be used by other tasks.
                    resolve(result);
                }
                catch (e) {
                    // Errors that happen inside asynchronous tasks are
                    // reported to the tracker using a user-friendly message.
                    tracker.reportError('Failed ' + msg, e);
                }
            }, ASYNC_TASK_DELAY);
        });
    }
    /**
     * Asynchronously runs an expensive task that returns a promise. Updates the
     * tracker's progress after the promise resolves. Returns a new promise that
     * resolves after the progress is updated.
     */
    function runAsyncPromiseTask(msg, incProgressValue, task, tracker) {
        return new Promise((resolve, reject) => {
            let handleError = function (e) {
                // Errors that happen inside asynchronous tasks are
                // reported to the tracker using a user-friendly message.
                tracker.reportError('Failed ' + msg, e);
                reject(e);
            };
            // Update the progress message to say the current running task.
            tracker.setMessage(msg);
            // Run the expensive task with a delay that gives enough time for the
            // UI to update.
            setTimeout(function () {
                try {
                    let start = Date.now();
                    task()
                        .then(function (value) {
                        /* tslint:disable */
                        console.log(msg, ':', Date.now() - start, 'ms');
                        // Update the progress value.
                        tracker.updateProgress(incProgressValue);
                        // Return the result to be used by other tasks.
                        resolve(value);
                    })
                        .catch(handleError);
                }
                catch (e) {
                    handleError(e);
                }
            }, ASYNC_TASK_DELAY);
        });
    }
    /**
     * Returns a query selector with escaped special characters that are not
     * allowed in a query selector.
     */
    function escapeQuerySelector(querySelector) {
        return querySelector.replace(/([:.\[\],/\\\(\)])/g, '\\$1');
    }
    // For unit conversion.
    const MEMORY_UNITS = [
        // Atomic unit.
        { symbol: 'B' },
        // numUnits specifies how many previous units this unit contains.
        { symbol: 'KB', numUnits: 1024 },
        { symbol: 'MB', numUnits: 1024 },
        { symbol: 'GB', numUnits: 1024 },
        { symbol: 'TB', numUnits: 1024 },
        { symbol: 'PB', numUnits: 1024 },
    ];
    const TIME_UNITS = [
        // Atomic unit. Finest granularity in TensorFlow stat collection.
        { symbol: 'µs' },
        // numUnits specifies how many previous units this unit contains.
        { symbol: 'ms', numUnits: 1000 },
        { symbol: 's', numUnits: 1000 },
        { symbol: 'min', numUnits: 60 },
        { symbol: 'hr', numUnits: 60 },
        { symbol: 'days', numUnits: 24 },
    ];
    /**
     * Returns the human readable version of the unit.
     * (e.g. 1.35 GB, 23 MB, 34 ms, 6.53 min etc).
     */
    function convertUnitsToHumanReadable(value, units, unitIndex = 0) {
        if (unitIndex + 1 < units.length && value >= units[unitIndex + 1].numUnits) {
            return convertUnitsToHumanReadable(value / units[unitIndex + 1].numUnits, units, unitIndex + 1);
        }
        // toPrecision() has the tendency to return a number in scientific
        // notation and casting back to a number brings it back to normal notation.
        // e.g.,
        //   > value = 213; value.toPrecision(1)
        //   < "2e+2"
        //   > Number(value.toPrecision(1))
        //   < 200
        return Number(value.toPrecision(3)) + ' ' + units[unitIndex].symbol;
    }
    function hasDisplayableNodeStats(stats) {
        if (stats &&
            (stats.totalBytes > 0 || stats.getTotalMicros() > 0 || stats.outputSize)) {
            return true;
        }
        return false;
    }
    /**
     * Given a list of strings, it returns a new list of strings with the longest
     * common prefix removed. If the common prefix is one of the strings in the
     * list, it returns the original strings.
     */
    function removeCommonPrefix(strings) {
        if (strings.length < 2) {
            return strings;
        }
        let index = 0;
        let largestIndex = 0;
        // Find the shortest name across all strings.
        let minLength = lodash.min(lodash.map(strings, (str) => str.length));
        while (true) {
            index++;
            let prefixes = lodash.map(strings, (str) => str.substring(0, index));
            let allTheSame = prefixes.every((prefix, i) => {
                return i === 0 ? true : prefix === prefixes[i - 1];
            });
            if (allTheSame) {
                if (index >= minLength) {
                    // There is a string whose whole name is a prefix to other string.
                    // In this case, we return the original list of string.
                    return strings;
                }
                largestIndex = index;
            }
            else {
                break;
            }
        }
        return lodash.map(strings, (str) => str.substring(largestIndex));
    }
    /**
     * Given a timestamp in microseconds, return a human-friendly string denoting
     * how long ago the timestamp was.
     */
    function computeHumanFriendlyTime(timeInMicroseconds) {
        var timeDifferenceInMs = +new Date() - +new Date(timeInMicroseconds / 1000);
        if (timeDifferenceInMs < 30000) {
            return 'just now';
        }
        else if (timeDifferenceInMs < 60000) {
            return Math.floor(timeDifferenceInMs / 1000) + ' seconds ago';
        }
        else if (timeDifferenceInMs < 120000) {
            return 'a minute ago';
        }
        else if (timeDifferenceInMs < 3600000) {
            return Math.floor(timeDifferenceInMs / 60000) + ' minutes ago';
        }
        else if (Math.floor(timeDifferenceInMs / 3600000) == 1) {
            return 'an hour ago';
        }
        else if (timeDifferenceInMs < 86400000) {
            return Math.floor(timeDifferenceInMs / 3600000) + ' hours ago';
        }
        else if (timeDifferenceInMs < 172800000) {
            return 'yesterday';
        }
        return Math.floor(timeDifferenceInMs / 86400000) + ' days ago';
    }

    /* Copyright 2015 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the 'License');
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an 'AS IS' BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    const NAMESPACE_DELIM = '/';
    const ROOT_NAME = '__root__';
    const FUNCTION_LIBRARY_NODE_PREFIX = '__function_library__';
    /** Attribute key used for storing attributes that are too large. */
    const LARGE_ATTRS_KEY = '_too_large_attrs';
    // Separator between the source and the destination name of the edge.
    const EDGE_KEY_DELIM$1 = '--';
    var GraphType;
    (function (GraphType) {
        GraphType[GraphType["FULL"] = 0] = "FULL";
        GraphType[GraphType["EMBEDDED"] = 1] = "EMBEDDED";
        GraphType[GraphType["META"] = 2] = "META";
        GraphType[GraphType["SERIES"] = 3] = "SERIES";
        GraphType[GraphType["CORE"] = 4] = "CORE";
        GraphType[GraphType["SHADOW"] = 5] = "SHADOW";
        GraphType[GraphType["BRIDGE"] = 6] = "BRIDGE";
        GraphType[GraphType["EDGE"] = 7] = "EDGE";
    })(GraphType || (GraphType = {}));
    var NodeType;
    (function (NodeType) {
        NodeType[NodeType["META"] = 0] = "META";
        NodeType[NodeType["OP"] = 1] = "OP";
        NodeType[NodeType["SERIES"] = 2] = "SERIES";
        NodeType[NodeType["BRIDGE"] = 3] = "BRIDGE";
        NodeType[NodeType["ELLIPSIS"] = 4] = "ELLIPSIS";
    })(NodeType || (NodeType = {}));
    /** Indicates if a node is to be included in the main graph when rendered. */
    var InclusionType;
    (function (InclusionType) {
        InclusionType[InclusionType["INCLUDE"] = 0] = "INCLUDE";
        InclusionType[InclusionType["EXCLUDE"] = 1] = "EXCLUDE";
        InclusionType[InclusionType["UNSPECIFIED"] = 2] = "UNSPECIFIED";
    })(InclusionType || (InclusionType = {}));
    /** Indicates if a series is to be grouped in the graph when rendered. */
    var SeriesGroupingType;
    (function (SeriesGroupingType) {
        SeriesGroupingType[SeriesGroupingType["GROUP"] = 0] = "GROUP";
        SeriesGroupingType[SeriesGroupingType["UNGROUP"] = 1] = "UNGROUP";
    })(SeriesGroupingType || (SeriesGroupingType = {}));
    /** Attribute key reserved for the shapes of the output tensors. */
    const OUTPUT_SHAPES_KEY = '_output_shapes';
    /** Attribute key reserved for the XLA cluster that an op runs on. */
    const _XLA_CLUSTER_KEY = '_XlaCluster';
    /**
     * A SlimGraph is inspired by graphlib.Graph, but having only the functionality
     * that we need.
     */
    class SlimGraph {
        constructor() {
            this.nodes = {};
            this.edges = [];
        }
    }
    class EllipsisNodeImpl {
        /**
         * Constructs a new ellipsis annotation node.
         *
         * @param numNodes The number of additional annotations this node represents.
         */
        constructor(numNodes) {
            this.type = NodeType.ELLIPSIS;
            this.isGroupNode = false;
            this.cardinality = 1;
            this.parentNode = null;
            this.stats = null;
            this.setNumMoreNodes(numNodes);
            this.include = InclusionType.UNSPECIFIED;
        }
        setNumMoreNodes(numNodes) {
            this.numMoreNodes = numNodes;
            this.name = '... ' + numNodes + ' more';
        }
    }
    /**
     * A label object for nodes in the full graph and leaf nodes in the render
     * graph.
     */
    class OpNodeImpl {
        /**
         * Constructs a new Op node.
         *
         * @param rawNode The raw node.
         */
        constructor(rawNode) {
            this.op = rawNode.op;
            this.name = rawNode.name;
            this.device = rawNode.device;
            this.attr = rawNode.attr;
            // An array of normalized inputs that denote the incoming edges to
            // the current node. Each input contains the normalized name of the
            // source node, whether it has a number part and whether it is a
            // control dependency.
            this.inputs = normalizeInputs(rawNode.input);
            this.outputShapes = extractOutputShapes(rawNode.attr);
            this.xlaCluster = extractXlaCluster(rawNode.attr);
            this.compatible = false;
            // additional properties
            this.type = NodeType.OP;
            this.isGroupNode = false;
            this.cardinality = 1;
            this.inEmbeddings = [];
            this.outEmbeddings = [];
            this.parentNode = null;
            this.include = InclusionType.UNSPECIFIED;
            this.owningSeries = null;
        }
    }
    function createMetanode(name, opt = {}) {
        return new MetanodeImpl(name, opt);
    }
    /**
     * Joins the information from the stats file (memory, compute time) with the
     * graph information.
     */
    function joinStatsInfoWithGraph(graph, stats, devicesForStats) {
        // Reset stats for each node.
        lodash.each(graph.nodes, (node) => {
            node.stats = null;
        });
        lodash.each(stats.dev_stats, (devStats) => {
            // Ignore devices that are not selected.
            if (devicesForStats && !devicesForStats[devStats.device]) {
                return;
            }
            lodash.each(devStats.node_stats, (nodeStats) => {
                // Lookup the node in the graph by its original name, e.g. A/B. If not
                // found, lookup by the rewritten name A/B/(B) in case the name is both
                // a namespace and a node name.
                let nodeName = nodeStats.node_name in graph.nodes
                    ? nodeStats.node_name
                    : getStrictName(nodeStats.node_name);
                // Couldn't find a matching node.
                if (!(nodeName in graph.nodes)) {
                    return;
                }
                // Compute the total bytes used.
                let totalBytes = 0;
                if (nodeStats.memory) {
                    lodash.each(nodeStats.memory, (alloc) => {
                        if (alloc.total_bytes) {
                            if (alloc.total_bytes > 0) {
                                totalBytes += Number(alloc.total_bytes);
                            }
                            else {
                                /* tslint:disable */
                                console.log('ignoring negative memory allocation for ' + nodeName);
                                /* tslint:enable */
                            }
                        }
                    });
                }
                let outputSize = null;
                if (nodeStats.output) {
                    outputSize = lodash.map(nodeStats.output, (output) => {
                        return lodash.map(output.tensor_description.shape.dim, (dim) => Number(dim.size));
                    });
                }
                graph.nodes[nodeName].device = devStats.device;
                if (graph.nodes[nodeName].stats == null) {
                    graph.nodes[nodeName].stats = new NodeStats(outputSize);
                }
                graph.nodes[nodeName].stats.addBytesAllocation(totalBytes);
                if (nodeStats.all_end_rel_micros) {
                    if (nodeStats.all_end_rel_micros > 0) {
                        graph.nodes[nodeName].stats.addExecutionTime(nodeStats.all_start_micros, nodeStats.all_start_micros + nodeStats.all_end_rel_micros);
                    }
                    else {
                        /* tslint:disable */
                        console.log('ignoring negative runtime for ' + nodeName);
                        /* tslint:enable */
                    }
                }
            });
        });
    }
    class MetanodeImpl {
        /** A label object for meta-nodes in the graph hierarchy */
        constructor(name, opt = {}) {
            this.name = name;
            this.type = NodeType.META;
            /** number of levels under this group */
            this.depth = 1;
            this.isGroupNode = true;
            /** # of leaf nodes (including embedded ones) */
            this.cardinality = 0;
            /** graph contains metanodes, nodes, edges
             * and metaedges for main items within this metanode
             */
            this.metagraph = createGraph(name, GraphType.META, opt);
            /** bridgegraph must be constructed lazily-see hierarchy.getBridgegraph() */
            this.bridgegraph = null;
            /**
             * A dictionary that count ops type of nodes in this metanode
             * (op type => count).
             */
            this.opHistogram = {};
            this.deviceHistogram = {};
            this.xlaClusterHistogram = {};
            this.compatibilityHistogram = { compatible: 0, incompatible: 0 };
            /** unique id for a metanode of similar subgraph */
            this.templateId = null;
            /** Metanode which contains this node, if any */
            this.parentNode = null;
            this.hasNonControlEdges = false;
            this.include = InclusionType.UNSPECIFIED;
            this.associatedFunction = '';
        }
        getFirstChild() {
            return this.metagraph.node(this.metagraph.nodes()[0]);
        }
        /**
         * Returns the op node associated with the metanode.
         * For example, if the metanode is 'sgd', the associated
         * op node is sgd/(sgd).
         */
        getRootOp() {
            let nameSplit = this.name.split('/');
            let rootOpName = this.name + '/(' + nameSplit[nameSplit.length - 1] + ')';
            return this.metagraph.node(rootOpName);
        }
        /**
         * Return an array of the names of all the leaves (non-GroupNodes) inside
         * this metanode. This performs a breadth-first search of the tree, so
         * immediate child leaves will appear earlier in the output array than
         * descendant leaves.
         */
        leaves() {
            let leaves = [];
            let queue = [this];
            let metagraph; // Defined here due to a limitation of ES6->5 compilation.
            while (queue.length) {
                let node = queue.shift();
                if (node.isGroupNode) {
                    metagraph = node.metagraph;
                    lodash.each(metagraph.nodes(), (name) => queue.push(metagraph.node(name)));
                }
                else {
                    leaves.push(node.name);
                }
            }
            return leaves;
        }
    }
    function createMetaedge(v, w) {
        return new MetaedgeImpl(v, w);
    }
    /**
     * A label object for edges between metanodes of subgraphs in the render graph.
     */
    class MetaedgeImpl {
        constructor(v, w) {
            this.v = v;
            this.w = w;
            this.baseEdgeList = [];
            this.inbound = null;
            this.numRegularEdges = 0;
            this.numControlEdges = 0;
            this.numRefEdges = 0;
            this.totalSize = 0;
        }
        addBaseEdge(edge, h) {
            this.baseEdgeList.push(edge);
            if (edge.isControlDependency) {
                this.numControlEdges += 1;
            }
            else {
                this.numRegularEdges += 1;
            }
            if (edge.isReferenceEdge) {
                this.numRefEdges += 1;
            }
            // Compute the size of the tensor flowing through this
            // base edge.
            this.totalSize += MetaedgeImpl.computeSizeOfEdge(edge, h);
            h.maxMetaEdgeSize = Math.max(h.maxMetaEdgeSize, this.totalSize);
        }
        static computeSizeOfEdge(edge, h) {
            let opNode = h.node(edge.v);
            if (!opNode.outputShapes) {
                // No shape information. Asssume a single number. This gives
                // a lower bound for the total size.
                return 1;
            }
            h.hasShapeInfo = true;
            // Sum the sizes of all output tensors.
            // TODO(stephanwlee): Use Object.values after es2017.
            const values = Object.keys(opNode.outputShapes)
                .map((k) => opNode.outputShapes[k])
                .map((shape) => {
                // If the shape is unknown, treat it as 1 when computing
                // total size. This gives a lower bound for the total size.
                if (shape == null) {
                    return 1;
                }
                // Multiply all shapes to get the total size of the tensor.
                // E.g. The total size of [4, 2, 1] is 4 * 2 * 1.
                return shape.reduce((accumulated, currSize) => {
                    // If this particular dimension is unknown, treat
                    // it as 1 when computing total size. This gives a lower bound
                    // for the total size.
                    if (currSize === -1) {
                        currSize = 1;
                    }
                    return accumulated * currSize;
                }, 1);
            });
            return lodash.sum(values);
        }
    }
    function createSeriesNode(prefix, suffix, parent, clusterId, name, graphOptions) {
        return new SeriesNodeImpl(prefix, suffix, parent, clusterId, name, graphOptions);
    }
    function getSeriesNodeName(prefix, suffix, parent, startId, endId) {
        let numRepresentation = typeof startId !== 'undefined' && typeof endId !== 'undefined'
            ? '[' + startId + '-' + endId + ']'
            : '#';
        let pattern = prefix + numRepresentation + suffix;
        return (parent ? parent + '/' : '') + pattern;
    }
    class SeriesNodeImpl {
        constructor(prefix, suffix, parent, clusterId, name, graphOptions) {
            this.name = name || getSeriesNodeName(prefix, suffix, parent);
            this.type = NodeType.SERIES;
            this.hasLoop = false;
            this.prefix = prefix;
            this.suffix = suffix;
            this.clusterId = clusterId;
            this.ids = [];
            this.parent = parent;
            this.isGroupNode = true;
            this.cardinality = 0;
            this.metagraph = createGraph(name, GraphType.SERIES, graphOptions);
            // bridgegraph must be constructed lazily-see hierarchy.getBridgegraph()
            this.bridgegraph = null;
            this.parentNode = null;
            this.deviceHistogram = {};
            this.xlaClusterHistogram = {};
            this.compatibilityHistogram = { compatible: 0, incompatible: 0 };
            this.hasNonControlEdges = false;
            this.include = InclusionType.UNSPECIFIED;
        }
    }
    /**
     * Extracts the shapes of the output tensors from the attr property in the
     * node proto.
     */
    // tslint:disable-next-line:no-any
    function extractOutputShapes(attr) {
        // We don't know anything about the output tensors.
        if (!attr) {
            return null;
        }
        for (let i = 0; i < attr.length; i++) {
            let { key, value } = attr[i];
            if (key === OUTPUT_SHAPES_KEY) {
                if (!value.list.shape) {
                    // The OUTPUT_SHAPES_KEY lacks a value. We know nothing about the shape.
                    return null;
                }
                // Map all output tensors into array of numbers denoting their shape.
                let result = value.list.shape.map((shape) => {
                    if (shape.unknown_rank) {
                        // This output tensor is of unknown rank. We don't know if it is a
                        // scalar, or a tensor, or of what shape it is.
                        return null;
                    }
                    if (shape.dim == null ||
                        (shape.dim.length === 1 && shape.dim[0].size == null)) {
                        // This output tensor is a scalar.
                        return [];
                    }
                    // This output tensor has a known rank. Map each dimension size
                    // into a number.
                    return shape.dim.map((dim) => {
                        // Size can be -1 if this particular dimension is unknown.
                        return dim.size;
                    });
                });
                // Since we already processed it, remove the entry from the attribute
                // list (saves memory).
                attr.splice(i, 1);
                return result;
            }
        }
        // We didn't find OUTPUT_SHAPES_KEY in attributes, so we don't know anything
        // about the output tensors.
        return null;
    }
    /**
     * Extracts the XLA Cluster that an op runs on from the attrs of the OpNode.
     * @param attr The attr property.
     * @return A string that is the name of the cluster. Or null if it could not be
     *     determined.
     */
    // tslint:disable-next-line:no-any
    function extractXlaCluster(attr) {
        if (!attr) {
            return null;
        }
        // Find the attribute for XLA cluster if there is one.
        for (let i = 0; i < attr.length; i++) {
            if (attr[i].key === _XLA_CLUSTER_KEY) {
                return attr[i].value['s'] || null;
            }
        }
        return null;
    }
    /**
     * Normalizes the inputs and extracts associated metadata:
     * 1) Inputs can contain a colon followed by a suffix of characters.
     *    That suffix may be a single number (e.g. inputName:1) or several word
     *    characters separated from a number by a colon (e.g. inputName:foo:1). The
     *    latter case is used to denote inputs and outputs of functions.
     * 2) Control dependency inputs contain caret at the beginning and we
     *    remove this and annotate the edge as a control dependency.
     * @param inputs Array of unnormalized names of input nodes.
     */
    function normalizeInputs(inputs) {
        let normalizedInputs = [];
        lodash.each(inputs, (inputName) => {
            let isControlDependency = inputName[0] === '^';
            if (isControlDependency) {
                // The carat merely indicates whether this input is a control dependency.
                // It should not be part of the name.
                inputName = inputName.substring(1);
            }
            let name = inputName;
            let outputTensorKey = '0';
            let match = inputName.match(/(.*):(\w+:\d+)$/);
            if (match) {
                // The output string consists of several characters and a number separated
                // by a colon.
                name = match[1];
                outputTensorKey = match[2];
            }
            else {
                match = inputName.match(/(.*):(\d+)$/);
                if (match) {
                    // The output string consists of a single number.
                    name = match[1];
                    outputTensorKey = match[2];
                }
            }
            if (normalizedInputs.length === 0 ||
                name !== normalizedInputs[normalizedInputs.length - 1].name) {
                normalizedInputs.push({
                    name: name,
                    outputTensorKey: outputTensorKey,
                    isControlDependency: isControlDependency,
                });
            }
        });
        return normalizedInputs;
    }
    function addEdgeToGraph(graph, inputName, outputNode, input, params, index) {
        // Don't allow loops in the graph.
        if (inputName === outputNode.name) {
            return;
        }
        // Check if this op type and input number corresponds to a
        // reference edge using the refEdges dictionary in the params.
        let isRefEdge = params.refEdges[outputNode.op + ' ' + index] === true;
        graph.edges.push({
            v: inputName,
            w: outputNode.name,
            outputTensorKey: input.outputTensorKey,
            isControlDependency: input.isControlDependency,
            isReferenceEdge: isRefEdge,
        });
    }
    const DefaultBuildParams = {
        enableEmbedding: true,
        inEmbeddingTypes: ['Const'],
        outEmbeddingTypes: ['^[a-zA-Z]+Summary$'],
        // This is the whitelist of inputs on op types that are considered
        // reference edges. "Assign 0" indicates that the first input to
        // an OpNode with operation type "Assign" is a reference edge.
        refEdges: {
            'Assign 0': true,
            'AssignAdd 0': true,
            'AssignSub 0': true,
            'assign 0': true,
            'assign_add 0': true,
            'assign_sub 0': true,
            'count_up_to 0': true,
            'ScatterAdd 0': true,
            'ScatterSub 0': true,
            'ScatterUpdate 0': true,
            'scatter_add 0': true,
            'scatter_sub 0': true,
            'scatter_update 0': true,
        },
    };
    function build(graphDef, params, tracker) {
        /**
         * A dictionary that maps each in-embedding node name to the node
         * object.
         */
        let inEmbedding = {};
        /**
         * A dictionary that maps each out-embedding node name to the node
         * object.
         */
        let outEmbedding = {};
        /**
         * A dictionary that maps each node name to an array of the node's
         * out-embedding node label objects.
         */
        let outEmbeddings = {};
        let isInEmbeddedPred = getEmbedPredicate(params.inEmbeddingTypes);
        let isOutEmbeddedPred = getEmbedPredicate(params.outEmbeddingTypes);
        let embeddingNodeNames = [];
        let rawNodes = graphDef.node;
        /**
         * A list of all the non-embedding node names which appear in the processed
         * list of raw nodes. Here we pre-allocate enough room for all the rawNodes,
         * even though there will some number of embeddings. The excess array length
         * is spliced off later.
         *
         * Experimentation shows that around 30% of the array will go unused, and
         * even for very large networks that amounts to less than 10k spaces.
         */
        let nodeNames = new Array(rawNodes.length);
        return runAsyncTask('Normalizing names', 30, () => {
            let opNodes = new Array(rawNodes.length);
            let index = 0;
            const processRawNode = (rawNode) => {
                let opNode = new OpNodeImpl(rawNode);
                if (isInEmbeddedPred(opNode)) {
                    embeddingNodeNames.push(opNode.name);
                    inEmbedding[opNode.name] = opNode;
                    return opNode;
                }
                if (isOutEmbeddedPred(opNode)) {
                    embeddingNodeNames.push(opNode.name);
                    outEmbedding[opNode.name] = opNode;
                    lodash.each(opNode.inputs, (input) => {
                        let inputName = input.name;
                        outEmbeddings[inputName] = outEmbeddings[inputName] || [];
                        outEmbeddings[inputName].push(opNode);
                    });
                    return opNode;
                }
                // The node is not an embedding, so add it to the names and nodes
                // lists.
                opNodes[index] = opNode;
                nodeNames[index] = opNode.name;
                index++;
                return opNode;
            };
            lodash.each(rawNodes, processRawNode);
            const processFunction = (func) => {
                // Give the function itself a node.
                const functionNodeName = FUNCTION_LIBRARY_NODE_PREFIX + func.signature.name;
                // Create an op node for the function. Mark it as part of a
                // function library.
                processRawNode({
                    name: functionNodeName,
                    input: [],
                    device: '',
                    op: '',
                    attr: [],
                });
                // If the function has inputs, make nodes out of them.
                if (func.signature.input_arg) {
                    // Makes an OpNode out of either an input_arg of a library
                    // function.
                    let currentInputIndex = 0;
                    const processInput = (arg) => {
                        const opNode = processRawNode({
                            name: functionNodeName + NAMESPACE_DELIM + arg.name,
                            input: [],
                            device: '',
                            op: 'input_arg',
                            attr: [
                                {
                                    key: 'T',
                                    value: {
                                        type: arg.type,
                                    },
                                },
                            ],
                        });
                        opNode.functionInputIndex = currentInputIndex;
                        currentInputIndex++;
                    };
                    // Make nodes for input args of the function. Unfortunately, the
                    // pbtxt configuration language is not rich enough to
                    // differentiate between an array with 1 item vs 1 object
                    // property.
                    if (func.signature.input_arg['name']) {
                        // There is only 1 input arg.
                        processInput(func.signature.input_arg);
                    }
                    else {
                        // There are several input args.
                        lodash.each(func.signature.input_arg, processInput);
                    }
                }
                // Make nodes for output args of the function. Track the names of
                // output args within the keys of this object. Unlike the
                // input_args, the output_args are already defined within the
                // node_defs of the library function.
                let currentOutputIndex = 0;
                const outputArgNames = {};
                // If the function has outputs, make nodes out of them.
                if (func.signature.output_arg) {
                    const processOutput = (arg) => {
                        outputArgNames[functionNodeName + NAMESPACE_DELIM + arg.name] = currentOutputIndex;
                        currentOutputIndex++;
                    };
                    if (func.signature.output_arg['name']) {
                        // There is only 1 output arg.
                        processOutput(func.signature.output_arg);
                    }
                    else {
                        // There are several output args.
                        lodash.each(func.signature.output_arg, processOutput);
                    }
                }
                lodash.each(func.node_def, (rawNode) => {
                    // Prefix with the name of the function so that the graph
                    // correctly computes the hierarchy (and makes metanodes).
                    rawNode.name = functionNodeName + '/' + rawNode.name;
                    if (typeof rawNode.input === 'string') {
                        rawNode.input = [rawNode.input];
                    }
                    const opNode = processRawNode(rawNode);
                    if (lodash.isNumber(outputArgNames[rawNode.name])) {
                        // Mark the node as one of the outputs of the function.
                        opNode.functionOutputIndex = outputArgNames[rawNode.name];
                    }
                    lodash.each(opNode.inputs, (normalizedInput) => {
                        normalizedInput.name =
                            functionNodeName + NAMESPACE_DELIM + normalizedInput.name;
                    });
                });
            };
            if (graphDef.library && graphDef.library.function) {
                // This graph contains functions.
                lodash.each(graphDef.library.function, processFunction);
            }
            opNodes.splice(index);
            nodeNames.splice(index);
            return opNodes;
        }, tracker)
            .then((opNodes) => {
            // Create the graph data structure from the graphlib library.
            return runAsyncTask('Building the data structure', 70, () => {
                let normalizedNameDict = mapStrictHierarchy(nodeNames, embeddingNodeNames);
                let graph = new SlimGraph();
                // Add the nodes to the graph.
                lodash.each(opNodes, (opNode) => {
                    let normalizedName = normalizedNameDict[opNode.name] || opNode.name;
                    graph.nodes[normalizedName] = opNode;
                    // Check if the node has out-embeddings. If yes, add them to the
                    // node.
                    if (opNode.name in outEmbeddings) {
                        opNode.outEmbeddings = outEmbeddings[opNode.name];
                        // Normalize the names of the out-embeddings.
                        lodash.each(opNode.outEmbeddings, (node) => {
                            node.name = normalizedNameDict[node.name] || node.name;
                        });
                    }
                    // Update the name of the node.
                    opNode.name = normalizedName;
                });
                // Visit each node's inputs to add the edges to the graph. If the
                // input
                // is an in-embedding, then add it to the node's in-embeddings
                // instead.
                lodash.each(opNodes, (opNode) => {
                    lodash.each(opNode.inputs, (input, i) => {
                        let inputName = input.name;
                        if (inputName in inEmbedding) {
                            let inEmbedNode = inEmbedding[inputName];
                            opNode.inEmbeddings.push(inEmbedNode);
                            // Move the inputs of the in-embedding node into incoming
                            // edges of
                            // the main node. E.g. the control dependency of a constant
                            // node
                            // should be moved to the op node where the constant is
                            // embedded.
                            for (let embedInput of inEmbedNode.inputs) {
                                addEdgeToGraph(graph, normalizedNameDict[embedInput.name] || embedInput.name, opNode, embedInput, params, i);
                            }
                        }
                        else if (inputName in outEmbedding) {
                            // Move the inputs of the out-embedding node into inputs of
                            // the main node where the out-embedding points to.
                            let outEmbedNode = outEmbedding[inputName];
                            for (let embedInput of outEmbedNode.inputs) {
                                addEdgeToGraph(graph, normalizedNameDict[embedInput.name] || embedInput.name, opNode, input, params, i);
                            }
                        }
                        else {
                            addEdgeToGraph(graph, normalizedNameDict[inputName] || inputName, opNode, input, params, i);
                        }
                    });
                });
                // Normalize the names of in-embeddings.
                lodash.each(inEmbedding, (node, name) => {
                    node.name = normalizedNameDict[node.name] || node.name;
                });
                return graph;
            }, tracker);
        });
    }
    /**
     * Create a new graphlib.Graph() instance with default parameters
     */
    function createGraph(name, type, opt) {
        const graphOptions = opt || {};
        let graph = new dagre.graphlib.Graph(graphOptions);
        graph.setGraph({
            name: name,
            rankdir: graphOptions.rankdir || 'BT',
            type: type,
        });
        return graph;
    }
    /**
     * Create a predicate for checking whether a node should be embedded based on
     * the specified types.
     */
    function getEmbedPredicate(types) {
        return function (node) {
            // check types
            for (let i = 0; i < types.length; i++) {
                let regExp = new RegExp(types[i]);
                if (typeof node.op === 'string' && node.op.match(regExp)) {
                    return true;
                }
            }
            return false;
        };
    }
    /**
     * Returns a strict node name (name => name/(name)) to avoid conflicts
     * where the node name is also a namespace.
     */
    function getStrictName(name) {
        let parts = name.split(NAMESPACE_DELIM);
        return name + NAMESPACE_DELIM + '(' + parts[parts.length - 1] + ')';
    }
    /**
     * For each op node (embedding or non-embedding), rename it if there is a
     * non-embedding node under its namespace. For example, assume node name 'A'.
     * If there is a non-embedding node under its namespace (e.g. 'A/B'), 'A' will
     * be renamed to 'A/(A)'. Then the namespace 'A' will contain 2 nodes: '(A)'
     * and 'B'. If all the nodes under 'A' are embedding nodes (e.g. constant and
     * summary), keep 'A' as an Op node and don't create a namespace.
     *
     * @param nodeNames An array of regular (non-embedding) node names.
     * @param embeddingNodeNames An array of embedding node names.
     * @return Dictionary object mapping names that need to be renamed to
     *     new names.
     */
    function mapStrictHierarchy(nodeNames, embeddingNodeNames) {
        /** Dictionary that maps the old new to the new name */
        let newNameDictionary = {};
        /** Set used to store all namespaces. */
        let namespaceSet = {};
        // sort the nodes to make prefix check faster
        nodeNames.sort();
        // look for nodes with a prefix a,a/b -> a/(a),a/b
        for (let i = 0; i < nodeNames.length - 1; ++i) {
            let a = nodeNames[i];
            // Get all the parent namespaces of the current node
            // and add them in the namespace set.
            lodash.each(getHierarchicalPath(a).slice(0, -1), (ns) => {
                namespaceSet[ns] = true;
            });
            for (let j = i + 1; j < nodeNames.length; ++j) {
                let b = nodeNames[j];
                if (lodash.startsWith(b, a)) {
                    if (b.length > a.length && b.charAt(a.length) === NAMESPACE_DELIM) {
                        newNameDictionary[a] = getStrictName(a);
                        break;
                    }
                }
                else {
                    break;
                }
            }
        }
        // Go through all the embedding node names and rename them in case they
        // collide with namespaces.
        lodash.each(embeddingNodeNames, (embeddingName) => {
            if (embeddingName in namespaceSet) {
                // Rename to follow strict hierarchy.
                newNameDictionary[embeddingName] = getStrictName(embeddingName);
            }
        });
        return newNameDictionary;
    }
    /**
     * Returns a list of the degrees of each node in the graph.
     */
    function degreeSequence(graph) {
        let degrees = graph.nodes().map(function (name) {
            return graph.neighbors(name).length;
        });
        degrees.sort();
        return degrees;
    }
    /**
     * Returns if the degree sequence of the two graphs is the same.
     */
    function hasSimilarDegreeSequence(graph1, graph2) {
        let dg1 = degreeSequence(graph1);
        let dg2 = degreeSequence(graph2);
        for (let i = 0; i < dg1.length; i++) {
            if (dg1[i] !== dg2[i]) {
                return false;
            }
        }
        return true;
    }
    /**
     * Returns the hierarchical path of the current node, based on the node's name.
     * For example, if the name is 'a/b/c', the returned path is
     * ['a', 'a/b', 'a/b/c'].
     */
    function getHierarchicalPath(name, seriesNames) {
        let path = [];
        let i = name.indexOf(NAMESPACE_DELIM);
        // Push all parent portions of the path.
        while (i >= 0) {
            path.push(name.substring(0, i));
            i = name.indexOf(NAMESPACE_DELIM, i + 1);
        }
        // If the node's path is under a series, then add the series node name to the
        // hierarchical path as the parent of the leaf.
        if (seriesNames) {
            let seriesName = seriesNames[name];
            if (seriesName) {
                path.push(seriesName);
            }
        }
        // Push the leaf of the path.
        path.push(name);
        return path;
    }
    /**
     * Returns the string for the node inclusion toggle button, dependant
     * on the provided current InclusionType.
     */
    function getIncludeNodeButtonString(include) {
        if (include === InclusionType.EXCLUDE) {
            return 'Add to main graph';
        }
        else {
            return 'Remove from main graph';
        }
    }
    /**
     * Returns the string for the series node grouping toggle button, dependant
     * on the provided current SeriesGroupingType.
     */
    function getGroupSeriesNodeButtonString(group) {
        if (group === SeriesGroupingType.GROUP) {
            return 'Ungroup this series of nodes';
        }
        else {
            return 'Group this series of nodes';
        }
    }
    /**
     * Toggle the node series grouping option in the provided map, setting it
     * to ungroup if the series is not already in the map.
     */
    function toggleNodeSeriesGroup(map, name) {
        if (!(name in map) || map[name] === SeriesGroupingType.GROUP) {
            map[name] = SeriesGroupingType.UNGROUP;
        }
        else {
            map[name] = SeriesGroupingType.GROUP;
        }
    }

    /* Copyright 2015 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the 'License');
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an 'AS IS' BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * Color parameters for op nodes.
     */
    let OpNodeColors = {
        DEFAULT_FILL: '#ffffff',
        DEFAULT_STROKE: '#b2b2b2',
        COMPATIBLE: '#0f9d58',
        INCOMPATIBLE: '#db4437',
    };
    /**
     * Color parameters for node encoding.
     * @type {Object}
     */
    let MetanodeColors = {
        /**
         * Default fill and stroke to use when no other information is available.
         */
        DEFAULT_FILL: '#d9d9d9',
        DEFAULT_STROKE: '#a6a6a6',
        SATURATION: 0.6,
        LIGHTNESS: 0.85,
        /**
         * Neutral color to use when the node is expanded (used when coloring by
         * compute time, memory and device).
         */
        EXPANDED_COLOR: '#f0f0f0',
        /**
         * Standard hue values for node color palette.
         */
        HUES: [220, 100, 180, 40, 20, 340, 260, 300, 140, 60],
        STRUCTURE_PALETTE(id, lightened) {
            // The code below is a flexible way to computationally create a set
            // of colors that go well together.
            let hues = MetanodeColors.HUES;
            let n = hues.length;
            let hue = hues[id % n];
            let m = Math.sin((hue * Math.PI) / 360);
            let sat = lightened ? 30 : 90 - 60 * m;
            let light = lightened ? 95 : 80;
            return hsl(hue, 0.01 * sat, 0.01 * light).toString();
        },
        DEVICE_PALETTE(index) {
            return MetanodeColors.STRUCTURE_PALETTE(index);
        },
        XLA_CLUSTER_PALETTE(index) {
            return MetanodeColors.STRUCTURE_PALETTE(index);
        },
        UNKNOWN: '#eee',
        GRADIENT_OUTLINE: '#888',
    };
    /**
     * Color parameters for op nodes.
     */
    let SeriesNodeColors = {
        DEFAULT_FILL: 'white',
        DEFAULT_STROKE: '#b2b2b2',
    };
    /**
     * Parameters that affect how the graph is rendered on the screen.
     */
    const PARAMS = {
        /**
         * Whether to extract high degree nodes from the core part of the graph.
         */
        enableExtraction: true,
        /**
         * The minimum number of nodes for a graph to have in order for high in and
         * out degree nodes to be extracted in auxiliary. The aim here is to prevent
         * nodes from being extracted from small graphs.
         */
        minNodeCountForExtraction: 15,
        /**
         * The minimum in or out degree a node must have in order to be possibly
         * extracted.
         */
        minDegreeForExtraction: 5,
        /**
         * Maximum number of control edges a node can have before they aren't
         * displayed.
         */
        maxControlDegree: 4,
        /**
         * Maximum in (for outbound bridge paths) or out (for inbound bridge paths)
         * degree of a node allowed for a bridge path to be rendered to it from a
         * subhierarchy of nodes. Having a max prevents having too many nodes emanate
         * from a subhierarchy and crowding up.
         */
        maxBridgePathDegree: 4,
        /**
         * Types patterns for predefined out-extract nodes, which are
         * sink-like nodes that will be extracted from the main graph.
         */
        outExtractTypes: ['NoOp'],
        /**
         * Types patterns for predefined in-extract nodes, which are
         * source-like nodes that will be extracted from the main graph.
         */
        inExtractTypes: [],
        /**
         * When removing edges from a high degree node, remove all of its edges if
         * detachAllEdgesForHighDegree is true.  Otherwise remove all in-edges if
         * the node has high in-degree, or all out-edges if the node has high
         * out-degree.
         */
        detachAllEdgesForHighDegree: true,
        /**
         * After extracting high in/out degree nodes and predefined
         * source-like/sink-like, extract isolated nodes to the side
         * if this extractIsolatedNodesWithAnnotationsOnOneSide is true.
         */
        extractIsolatedNodesWithAnnotationsOnOneSide: true,
        /**
         * Whether to add bridge nodes and edges to the core when building the
         * subhierarchy of an expanded metanode. See buildSubhierarchy().
         */
        enableBridgegraph: true,
        /**
         * 2 colors, for the minimum and maximum value respectively, whenever we
         * have a gradient scale.
         */
        minMaxColors: ['#fff5f0', '#fb6a4a'],
        /**
         * Maximum number of annotations to be displayed on a node before an
         * ellipsis is used.
         */
        maxAnnotations: 5,
    };
    /**
     * The regular expression to use when parsing for the string that is
     * used to label a function node in the graph. We strip away a prefix
     * indicating that the node represents a function definition. We also
     * remove an arbitrary hexadecimal suffix and the number following it
     * if it is present. To be clear, we extract foo from
     * __function_library__foo_deadb00f_42.
     */
    const nodeDisplayNameRegex = new RegExp('^(?:' +
        FUNCTION_LIBRARY_NODE_PREFIX +
        ')?(\\w+)_[a-z0-9]{8}(?:_\\d+)?$');
    /**
     * Stores the rendering information, such as x and y coordinates,
     * for each node in the graph.
     */
    class RenderGraphInfo {
        constructor(hierarchy, displayingStats) {
            this.hierarchy = hierarchy;
            this.displayingStats = displayingStats;
            this.index = {};
            this.renderedOpNames = [];
            this.computeScales();
            // Maps node name to whether the rendering hierarchy was already
            // constructed.
            this.hasSubhierarchy = {};
            this.root = new RenderGroupNodeInfo(hierarchy.root, hierarchy.graphOptions);
            this.index[hierarchy.root.name] = this.root;
            this.renderedOpNames.push(hierarchy.root.name);
            this.buildSubhierarchy(hierarchy.root.name);
            this.root.expanded = true;
            this.traceInputs = false;
        }
        computeScales() {
            this.deviceColorMap = ordinal()
                .domain(this.hierarchy.devices)
                .range(lodash.map(sequence(this.hierarchy.devices.length), MetanodeColors.DEVICE_PALETTE));
            this.xlaClusterColorMap = ordinal()
                .domain(this.hierarchy.xlaClusters)
                .range(lodash.map(sequence(this.hierarchy.xlaClusters.length), MetanodeColors.XLA_CLUSTER_PALETTE));
            let topLevelGraph = this.hierarchy.root.metagraph;
            // Find the maximum memory usage. Use 0 as the minimum.
            let maxMemory = max(topLevelGraph.nodes(), (nodeName, index) => {
                let node = topLevelGraph.node(nodeName);
                // Some ops don't have stats at all.
                if (node.stats != null) {
                    return node.stats.totalBytes;
                }
            });
            this.memoryUsageScale = linear$2()
                .domain([0, maxMemory])
                .range(PARAMS.minMaxColors);
            // Find the maximum compute time. Use 0 as the minimum.
            let maxComputeTime = max(topLevelGraph.nodes(), (nodeName, index) => {
                let node = topLevelGraph.node(nodeName);
                // Some ops don't have stats at all.
                if (node.stats != null) {
                    return node.stats.getTotalMicros();
                }
            });
            this.computeTimeScale = linear$2()
                .domain([0, maxComputeTime])
                .range(PARAMS.minMaxColors);
            this.edgeWidthSizedBasedScale = this.hierarchy.hasShapeInfo
                ? EDGE_WIDTH_SIZE_BASED_SCALE
                : linear$2()
                    .domain([1, this.hierarchy.maxMetaEdgeSize])
                    .range([
                    MIN_EDGE_WIDTH,
                    MAX_EDGE_WIDTH,
                ]);
        }
        /**
         * Get a previously created RenderNodeInfo by its node name.
         */
        getRenderNodeByName(nodeName) {
            return this.index[nodeName];
        }
        /**
         * Get the underlying node in the hierarchical graph by its name.
         */
        getNodeByName(nodeName) {
            return this.hierarchy.node(nodeName);
        }
        colorHistogram(histogram, colors) {
            if (Object.keys(histogram).length > 0) {
                // Compute the total # of items.
                const numItems = lodash.sum(Object.keys(histogram).map((key) => histogram[key]));
                return Object.keys(histogram).map((key) => ({
                    color: colors(key),
                    // Normalize to a proportion of total # of items.
                    proportion: histogram[key] / numItems,
                }));
            }
            console.info('no pairs found!');
            return null;
        }
        /**
         * Get a previously created RenderNodeInfo for the specified node name,
         * or create one if it hasn't been created yet.
         */
        getOrCreateRenderNodeByName(nodeName) {
            // Polymer may invoke this with null.
            if (!nodeName) {
                return null;
            }
            if (nodeName in this.index) {
                return this.index[nodeName];
            }
            let node = this.hierarchy.node(nodeName);
            // Exit early if the node does not exist in the hierarchy. This can happen
            // when a graph is reloaded while the infocard points to a node not visible
            // at the top-level.
            if (!node) {
                return null;
            }
            let renderInfo = node.isGroupNode
                ? new RenderGroupNodeInfo(node, this.hierarchy.graphOptions)
                : new RenderNodeInfo(node);
            this.index[nodeName] = renderInfo;
            this.renderedOpNames.push(nodeName);
            if (node.stats) {
                renderInfo.memoryColor = this.memoryUsageScale(node.stats.totalBytes);
                renderInfo.computeTimeColor = this.computeTimeScale(node.stats.getTotalMicros());
            }
            // We only fade nodes when we're displaying stats.
            renderInfo.isFadedOut =
                this.displayingStats &&
                    !hasDisplayableNodeStats(node.stats);
            var deviceHistogram = null;
            var xlaClusterHistogram = null;
            var opCompatibility = null;
            if (node.isGroupNode) {
                deviceHistogram = node.deviceHistogram;
                xlaClusterHistogram = node.xlaClusterHistogram;
                let compat = node.compatibilityHistogram.compatible;
                let incompat = node.compatibilityHistogram.incompatible;
                if (compat != 0 || incompat != 0) {
                    opCompatibility = compat / (compat + incompat);
                }
            }
            else {
                let device = renderInfo.node.device;
                if (device) {
                    deviceHistogram = { [device]: 1 };
                }
                let xlaCluster = renderInfo.node.xlaCluster;
                if (xlaCluster) {
                    xlaClusterHistogram = { [xlaCluster]: 1 };
                }
                if (renderInfo.node.type === NodeType.OP) {
                    opCompatibility = renderInfo.node.compatible ? 1 : 0;
                }
            }
            if (deviceHistogram) {
                renderInfo.deviceColors = this.colorHistogram(deviceHistogram, this.deviceColorMap);
            }
            if (xlaClusterHistogram) {
                renderInfo.xlaClusterColors = this.colorHistogram(xlaClusterHistogram, this.xlaClusterColorMap);
            }
            if (opCompatibility != null) {
                renderInfo.compatibilityColors = [
                    {
                        color: OpNodeColors.COMPATIBLE,
                        proportion: opCompatibility,
                    },
                    {
                        color: OpNodeColors.INCOMPATIBLE,
                        proportion: 1 - opCompatibility,
                    },
                ];
            }
            return this.index[nodeName];
        }
        /**
         * Return the nearest ancestor node, including itself, that is visible
         * in the visualization. This method is used so that we can select
         * (highlight) a node that isn't drawn yet, by selecting (highlighting)
         * its nearest ancestor that has been drawn.
         */
        getNearestVisibleAncestor(name) {
            let path = getHierarchicalPath(name);
            let i = 0;
            let renderNode = null;
            // Fallthrough. If everything was expanded return the node.
            let nodeName = name;
            for (; i < path.length; i++) {
                nodeName = path[i];
                renderNode = this.getRenderNodeByName(nodeName);
                // Op nodes have expanded set to false by default.
                if (!renderNode.expanded) {
                    break;
                }
            }
            // Check case where highlighted node is an embedded node whose parent node
            // is also its hierarchical parent. In this case, we want to return the
            // embedded node name, as it is also displayed if its parent has been
            // displayed.
            if (i == path.length - 2) {
                let nextName = path[i + 1];
                if (renderNode.inAnnotations.nodeNames[nextName]) {
                    return nextName;
                }
                if (renderNode.outAnnotations.nodeNames[nextName]) {
                    return nextName;
                }
            }
            return nodeName;
        }
        // TODO: Delete this an any code it touches (all deprecated).
        setDepth(depth) {
            setGroupNodeDepth(this.root, +depth);
        }
        /**
         * Returns true if the renderNode is an isolated node within its parent node.
         */
        isNodeAuxiliary(renderNode) {
            let parentNode = (this.getRenderNodeByName(renderNode.node.parentNode.name));
            let found = lodash.find(parentNode.isolatedInExtract, (node) => {
                return node.node.name === renderNode.node.name;
            });
            if (found) {
                return true;
            }
            found = lodash.find(parentNode.isolatedOutExtract, (node) => {
                return node.node.name === renderNode.node.name;
            });
            return !!found;
        }
        /**
         * Returns a list of ops that have been rendered so far for this graph. More
         * ops may later be rendered if the user expands nodes for instance. The list
         * returned here can only stay the same size or grow on successive calls.
         */
        getNamesOfRenderedOps() {
            return this.renderedOpNames;
        }
        /**
         * Clones an op node and adds it to a metagraph. Does nothing if an op node
         * with the same new name has already been created within the metagraph. This
         * method is used when duplicating a library function to be injected within a
         * metanode representing a function call.
         * @param parentMetanode The parent metanode on which to add the new node.
         * @param node The op node to clone.
         * @param newPrefix The prefix string to use in lieu of the one that merely
         *     indicates that the metanode represents a function defined in the
         *     library. This prefix should reflect graph hierarchy.
         * @return The newly created op node (the clone of the original).
         */
        cloneAndAddFunctionOpNode(parentMetanode, libraryFunctionNodeName, node, newPrefix) {
            const newName = node.name.replace(libraryFunctionNodeName, newPrefix);
            let newOpNode = parentMetanode.metagraph.node(newName);
            if (newOpNode) {
                // This node had already been created and added to the graph.
                return newOpNode;
            }
            // Create a new op node.
            newOpNode = new OpNodeImpl({
                name: newName,
                input: [],
                device: node.device,
                op: node.op,
                attr: lodash.cloneDeep(node.attr),
            });
            // Update various properties.
            newOpNode.cardinality = node.cardinality;
            newOpNode.include = node.include;
            newOpNode.outputShapes = lodash.cloneDeep(node.outputShapes);
            newOpNode.xlaCluster = node.xlaCluster;
            newOpNode.functionInputIndex = node.functionInputIndex;
            newOpNode.functionOutputIndex = node.functionOutputIndex;
            // Update the inputs of the new node to reflect the new path.
            newOpNode.inputs = node.inputs.map((normalizedInput) => {
                const newNormalizedInput = lodash.clone(normalizedInput);
                newNormalizedInput.name = normalizedInput.name.replace(libraryFunctionNodeName, newPrefix);
                return newNormalizedInput;
            });
            // Add the new op node to the hierarchy and metagraph. Also add it to its
            // parent metanode.
            newOpNode.parentNode = parentMetanode;
            parentMetanode.metagraph.setNode(newOpNode.name, newOpNode);
            this.hierarchy.setNode(newOpNode.name, newOpNode);
            // Update embeddings.
            const updateEmbeddingOpNode = (embeddingNode) => {
                return this.cloneAndAddFunctionOpNode(parentMetanode, libraryFunctionNodeName, embeddingNode, newPrefix);
            };
            newOpNode.inEmbeddings = node.inEmbeddings.map(updateEmbeddingOpNode);
            newOpNode.outEmbeddings = node.outEmbeddings.map(updateEmbeddingOpNode);
            return newOpNode;
        }
        /**
         * Clones a Metanode that represents a function defined in the graph library.
         * We dynamically inject a clone of a function into a meta graph when the user
         * expands a function call. We cannot do this at the beginning because the
         * functions may recursively call themselves or other functions.
         * @param metagraph The metagraph we are currently rendering the sub-hierarchy
         *     for.
         * @param opNodeToReplace The op node in the graph to replace with a new
         *     (expandable) metanode that visualizes the innards of a function.
         * @param libraryMetanode The metanode for a library function to clone.
         * @param oldPrefix The old prefix to replace (that just reflects how this
         *     node is for a library function).
         * @param newPrefix The prefix string to use in lieu of the one that merely
         *     indicates that the metanode represents a function defined in the
         *     library. This prefix should reflect graph hierarchy.
         */
        cloneFunctionLibraryMetanode(metagraph, opNodeToReplace, libraryMetanode, oldPrefix, newPrefix) {
            // Make a mapping between function output index and the new node for the
            // output.
            const functionOutputIndexToNode = {};
            const newMetanode = this.cloneFunctionLibraryMetanodeHelper(metagraph, opNodeToReplace, libraryMetanode, oldPrefix, newPrefix, functionOutputIndexToNode);
            if (!lodash.isEmpty(functionOutputIndexToNode)) {
                // After we have cloned the edges within the metanode, we still must add
                // edges that emanate out of output ops within the function.
                this.patchEdgesFromFunctionOutputs(opNodeToReplace, functionOutputIndexToNode);
            }
            return newMetanode;
        }
        /**
         * A helper subroutine that performs the bulk of the logic for
         * `cloneFunctionLibraryMetanode`.
         * @param metagraph The metagraph we are currently rendering the sub-hierarchy
         *     for.
         * @param opNodeToReplace The op node in the graph to replace with a new
         *     (expandable) metanode that visualizes the innards of a function.
         * @param libraryMetanode The metanode for a library function to clone.
         * @param oldPrefix The old prefix to replace (that just reflects how this
         *     node is for a library function).
         * @param newPrefix The prefix string to use in lieu of the one that merely
         *     indicates that the metanode represents a function defined in the
         *     library. This prefix should reflect graph hierarchy.
         * @param functionOutputIndexToNode A mapping between function output index
         *     and the corresponding output node. Used to connect outputs with
         *     destinations outside of the function metanode.
         */
        cloneFunctionLibraryMetanodeHelper(metagraph, opNodeToReplace, libraryMetanode, oldPrefix, newPrefix, functionOutputIndexToNode) {
            const newMetanode = createMetanode(libraryMetanode.name.replace(oldPrefix, newPrefix));
            // Copy over various properties.
            newMetanode.depth = libraryMetanode.depth;
            newMetanode.cardinality = libraryMetanode.cardinality;
            newMetanode.templateId = libraryMetanode.templateId;
            newMetanode.opHistogram = lodash.clone(libraryMetanode.opHistogram);
            newMetanode.deviceHistogram = lodash.clone(libraryMetanode.deviceHistogram);
            newMetanode.xlaClusterHistogram = lodash.clone(libraryMetanode.xlaClusterHistogram);
            newMetanode.hasNonControlEdges = libraryMetanode.hasNonControlEdges;
            newMetanode.include = libraryMetanode.include;
            newMetanode.nodeAttributes = lodash.clone(libraryMetanode.nodeAttributes);
            newMetanode.associatedFunction = libraryMetanode.associatedFunction;
            // Recursively duplicate the children nodes.
            lodash.each(libraryMetanode.metagraph.nodes(), (nodeName) => {
                const node = libraryMetanode.metagraph.node(nodeName);
                switch (node.type) {
                    case NodeType.META:
                        // Recursively duplicate the metanode.
                        const newNode = this.cloneFunctionLibraryMetanodeHelper(metagraph, opNodeToReplace, node, oldPrefix, newPrefix, functionOutputIndexToNode);
                        // Add the new node to the graph.
                        newNode.parentNode = newMetanode;
                        newMetanode.metagraph.setNode(newNode.name, newNode);
                        this.hierarchy.setNode(newNode.name, newNode);
                        break;
                    case NodeType.OP:
                        // Duplicate the op node.
                        const newOpNode = this.cloneAndAddFunctionOpNode(newMetanode, oldPrefix, node, newPrefix);
                        if (lodash.isNumber(newOpNode.functionInputIndex)) {
                            // This node represents an input_arg of the library function. Give
                            // it edges so that its bridge edges are created correctly.
                            this.patchEdgesIntoFunctionInputs(opNodeToReplace, newOpNode);
                        }
                        if (lodash.isNumber(newOpNode.functionOutputIndex)) {
                            functionOutputIndexToNode[newOpNode.functionOutputIndex] = newOpNode;
                        }
                        break;
                    default:
                        // This logic should never run because the meta graph should only
                        // contain meta and op nodes.
                        console.warn(node.name + ' is oddly neither a metanode nor an opnode.');
                }
            });
            // Clone the edges within the function library metanode.
            this.cloneLibraryMetanodeEdges(libraryMetanode, newMetanode, oldPrefix, newPrefix);
            return newMetanode;
        }
        /**
         * Clones the edges within `libraryMetanode` and adds them to `newMetanode`.
         * The names of edge sources and destinations have their prefixes replaced
         * with new prefixes that reflect their hierarchical positions in the graph
         * instead of within the function library template. This is a subroutine for
         * dynamically injecting a function metanode into the graph.
         */
        cloneLibraryMetanodeEdges(libraryMetanode, newMetanode, oldPrefix, newPrefix) {
            lodash.each(libraryMetanode.metagraph.edges(), (edgeObject) => {
                const edge = libraryMetanode.metagraph.edge(edgeObject);
                const newV = edge.v.replace(oldPrefix, newPrefix);
                const newW = edge.w.replace(oldPrefix, newPrefix);
                const newMetaEdge = new MetaedgeImpl(newV, newW);
                // Duplicate various properties.
                newMetaEdge.inbound = edge.inbound;
                newMetaEdge.numRegularEdges = edge.numRegularEdges;
                newMetaEdge.numControlEdges = edge.numControlEdges;
                newMetaEdge.numRefEdges = edge.numRefEdges;
                newMetaEdge.totalSize = edge.totalSize;
                if (edge.baseEdgeList) {
                    newMetaEdge.baseEdgeList = edge.baseEdgeList.map((baseEdge) => {
                        const newBaseEdge = lodash.clone(baseEdge);
                        newBaseEdge.v = baseEdge.v.replace(oldPrefix, newPrefix);
                        newBaseEdge.w = baseEdge.w.replace(oldPrefix, newPrefix);
                        return newBaseEdge;
                    });
                }
                // Set the direction of the edge based on whether it is inbound. The edge
                // is inbound if its destination is within the metagraph.
                if (newMetanode.metagraph.node(newW)) {
                    newMetanode.metagraph.setEdge(newV, newW, newMetaEdge);
                }
                else {
                    newMetanode.metagraph.setEdge(newW, newV, newMetaEdge);
                }
            });
        }
        /**
         * When a metanode representing a function is cloned and placed into the
         * graph, we must create edges between inputs into the function call and the
         * input ops within the function. This function performs that patching.
         */
        patchEdgesIntoFunctionInputs(opNodeToReplace, newOpNode) {
            // If the last few raw inputs are the same node, previous graph logic
            // collapses them into a single normalized input.
            let inputIndex = Math.min(newOpNode.functionInputIndex, opNodeToReplace.inputs.length - 1);
            let newInput = lodash.clone(opNodeToReplace.inputs[inputIndex]);
            while (newInput.isControlDependency) {
                // Ignore control dependencies - they are not assigned to
                // input_args.
                inputIndex++;
                newInput = opNodeToReplace.inputs[inputIndex];
            }
            // Clone the normalized input object.
            newOpNode.inputs.push(newInput);
            // Update values in the corresponding edge in the high-level
            // metagraph.
            const originalMetaEdges = this.hierarchy.getPredecessors(opNodeToReplace.name);
            // Find the metaedge that the input index corresponds to.
            // A metaedge may correspond to several edges. For instance,
            // an edge may enter a series node.
            let originalMetaEdge;
            let regularEdgeCount = 0;
            lodash.each(originalMetaEdges.regular, (metaEdge) => {
                regularEdgeCount += metaEdge.numRegularEdges;
                if (regularEdgeCount > inputIndex) {
                    originalMetaEdge = metaEdge;
                    // Terminate the loop.
                    return false;
                }
            });
            // Also change any base edges that point into the original node to
            // point to the input arg within the function. These are used to
            // make bridge edges.
            lodash.each(originalMetaEdge.baseEdgeList, (edge) => {
                if (edge.w === opNodeToReplace.name) {
                    edge.w = newOpNode.name;
                }
                if (edge.v === opNodeToReplace.name) {
                    edge.v = newOpNode.name;
                }
            });
        }
        /**
         * When a metanode representing a function is cloned and placed into the
         * graph, we must create edges between output ops within the new function
         * metanode to its successors. This function does that after scanning the
         * successors of the function call.
         */
        patchEdgesFromFunctionOutputs(opNodeToReplace, functionOutputIndexToDestinationNode) {
            // Connect the outputs of the function to other ops.
            const originalMetaEdges = this.hierarchy.getSuccessors(opNodeToReplace.name);
            lodash.each(originalMetaEdges.regular, (metaedge) => {
                lodash.each(metaedge.baseEdgeList, (baseEdge) => {
                    // Destination nodes within regular base edges are op nodes.
                    const destinationNode = this.hierarchy.node(baseEdge.w);
                    lodash.each(destinationNode.inputs, (normalizedInput) => {
                        // If an output of the function is an input into the op, map it back
                        // to the output within the function so bridge edges are computed.
                        if (normalizedInput.name === opNodeToReplace.name) {
                            // Map the output tensor index (which in this case is for sure
                            // numeric because it is an output of a metanode) to the correct
                            // function output.
                            const outputNode = functionOutputIndexToDestinationNode[normalizedInput.outputTensorKey];
                            normalizedInput.name = outputNode.name;
                            normalizedInput.outputTensorKey = baseEdge.outputTensorKey;
                        }
                    });
                });
                // Modify the list of base edges to point from the output so that bridge
                // edges are correct.
                lodash.each(metaedge.baseEdgeList, (baseEdge) => {
                    baseEdge.v =
                        functionOutputIndexToDestinationNode[baseEdge.outputTensorKey].name;
                    baseEdge.outputTensorKey = '0';
                });
            });
        }
        buildSubhierarchy(nodeName) {
            // Terminate if the rendering hierarchy was already constructed
            // for this node.
            if (nodeName in this.hasSubhierarchy) {
                return;
            }
            // Record that we constructed the rendering hierarchy for this node, so we
            // don't construct it another time.
            this.hasSubhierarchy[nodeName] = true;
            let renderNodeInfo = this.index[nodeName];
            // If it is not a meta node or a series node, don't do anything.
            if (renderNodeInfo.node.type !== NodeType.META &&
                renderNodeInfo.node.type !== NodeType.SERIES) {
                return;
            }
            // At this point we know the rendering information is about a group node.
            let renderGroupNodeInfo = renderNodeInfo;
            let metagraph = renderGroupNodeInfo.node.metagraph;
            let coreGraph = renderGroupNodeInfo.coreGraph;
            const nodesThatGotCloned = [];
            const functionCallMetanodesToAdd = [];
            if (!lodash.isEmpty(this.hierarchy.libraryFunctions)) {
                // This graph has library functions. Add them to the current
                // sub-hierarchy if necessary.
                lodash.each(metagraph.nodes(), (childName) => {
                    // Why is this so often undefined?
                    const originalNode = metagraph.node(childName);
                    const libraryFunctionData = this.hierarchy.libraryFunctions[originalNode.op];
                    if (!libraryFunctionData) {
                        // This node is not a function call.
                        return;
                    }
                    if (childName.indexOf(FUNCTION_LIBRARY_NODE_PREFIX) === 0) {
                        // Do not replace library functions in the graph. The library
                        // functions serve as templates for other nodes.
                        return;
                    }
                    // We later replace the node that is a function call with a copy of the
                    // function metagraph. We do not do so now because we are also looping
                    // through all the nodes.
                    const clonedMetanode = this.cloneFunctionLibraryMetanode(metagraph, originalNode, libraryFunctionData.node, libraryFunctionData.node.name, originalNode.name);
                    nodesThatGotCloned.push(originalNode);
                    functionCallMetanodesToAdd.push(clonedMetanode);
                });
                // Perform node replacement.
                lodash.each(functionCallMetanodesToAdd, (clonedMetanode, i) => {
                    const originalNode = nodesThatGotCloned[i];
                    clonedMetanode.parentNode = originalNode.parentNode;
                    metagraph.setNode(originalNode.name, clonedMetanode);
                    this.hierarchy.setNode(originalNode.name, clonedMetanode);
                });
            }
            // Create render nodes to represent each child from the metagraph. Although
            // these will initially be added to the coreGraph, they may later be
            // extracted. Also, due to extraction, the coreGraph may contain disjoint
            // groups between which there is no visible path (other than annotations).
            lodash.each(metagraph.nodes(), (childName) => {
                let childRenderInfo = this.getOrCreateRenderNodeByName(childName);
                let childNode = childRenderInfo.node;
                coreGraph.setNode(childName, childRenderInfo);
                if (!childNode.isGroupNode) {
                    lodash.each(childNode.inEmbeddings, (embedding) => {
                        let renderMetaedgeInfo = new RenderMetaedgeInfo(null);
                        let renderNodeInfo = new RenderNodeInfo(embedding);
                        addInAnnotation(childRenderInfo, embedding, renderNodeInfo, renderMetaedgeInfo, AnnotationType.CONSTANT);
                        this.index[embedding.name] = renderNodeInfo;
                    });
                    lodash.each(childNode.outEmbeddings, (embedding) => {
                        let renderMetaedgeInfo = new RenderMetaedgeInfo(null);
                        let renderNodeInfo = new RenderNodeInfo(embedding);
                        addOutAnnotation(childRenderInfo, embedding, renderNodeInfo, renderMetaedgeInfo, AnnotationType.SUMMARY);
                        this.index[embedding.name] = renderNodeInfo;
                    });
                }
            });
            // Add render metaedge info for edges in the metagraph.
            lodash.each(metagraph.edges(), (edgeObj) => {
                let metaedge = metagraph.edge(edgeObj);
                let renderMetaedgeInfo = new RenderMetaedgeInfo(metaedge);
                renderMetaedgeInfo.isFadedOut =
                    this.index[edgeObj.v].isFadedOut || this.index[edgeObj.w].isFadedOut;
                coreGraph.setEdge(edgeObj.v, edgeObj.w, renderMetaedgeInfo);
            });
            if (
                renderGroupNodeInfo.node.type === NodeType.META) {
                extractHighDegrees(renderGroupNodeInfo);
            }
            // If there are functions, it is possible for metanodes to be dynamically
            // added later. Construct the hierarchies for nodes that are predecessors to
            // nodes in the current hierarchy so that edges are drawn correctly.
            if (!lodash.isEmpty(this.hierarchy.libraryFunctions)) {
                this.buildSubhierarchiesForNeededFunctions(metagraph);
            }
            if (nodeName === ROOT_NAME) {
                // Add all metanodes representing library function templates into the
                // library function scene group for the root node.
                lodash.forOwn(this.hierarchy.libraryFunctions, (libraryFunctionData, functionName) => {
                    const node = libraryFunctionData.node;
                    const childRenderInfo = this.getOrCreateRenderNodeByName(node.name);
                    renderGroupNodeInfo.libraryFunctionsExtract.push(childRenderInfo);
                    // Do not render function definitions in the core graph.
                    childRenderInfo.node.include = InclusionType.EXCLUDE;
                    coreGraph.removeNode(node.name);
                });
            }
            // Look up the parent node's render information and short circuit if none.
            let parentNode = renderGroupNodeInfo.node.parentNode;
            if (!parentNode) {
                return;
            }
            let parentNodeInfo = this.index[parentNode.name];
            // Utility function for computing the name of a bridge node.
            let getBridgeNodeName = (inbound, ...rest) => rest.concat([inbound ? 'IN' : 'OUT']).join('~~');
            // Build out the bridgegraph.
            let bridgegraph = this.hierarchy.getBridgegraph(nodeName);
            // Look for popular nodes so we can make annotations instead of paths.
            let otherCounts = {
                // Counts of edges coming INTO other nodes by name (outgoing from self).
                in: {},
                // Counts of edges going OUT from other nodes by name (coming into self).
                out: {},
                // Counts of all control edges involving other nodes by name.
                control: {},
            };
            lodash.each(bridgegraph.edges(), (e) => {
                // An edge is inbound if its destination node is in the metagraph.
                let inbound = !!metagraph.node(e.w);
                let otherName = inbound ? e.v : e.w;
                let metaedge = bridgegraph.edge(e);
                if (!metaedge.numRegularEdges) {
                    otherCounts.control[otherName] =
                        (otherCounts.control[otherName] || 0) + 1;
                }
                else if (inbound) {
                    otherCounts.out[otherName] = (otherCounts.out[otherName] || 0) + 1;
                }
                else {
                    otherCounts.in[otherName] = (otherCounts.in[otherName] || 0) + 1;
                }
            });
            // Add annotations and edges for bridgegraph relationships.
            let hierarchyNodeMap = this.hierarchy.getNodeMap();
            lodash.each(bridgegraph.edges(), (bridgeEdgeObj) => {
                let bridgeMetaedge = bridgegraph.edge(bridgeEdgeObj);
                // Determine whether this bridge edge is incoming by checking the
                // metagraph for a node that matches the destination end.
                let inbound = !!metagraph.node(bridgeEdgeObj.w);
                // Based on the direction of the edge, one endpoint will be an immediate
                // child of this renderNodeInfo, and the other endpoint will be a sibling
                // of the parent (or an ancestor further up).
                let [childName, otherName] = inbound
                    ? [bridgeEdgeObj.w, bridgeEdgeObj.v]
                    : [bridgeEdgeObj.v, bridgeEdgeObj.w];
                let childRenderInfo = this.index[childName];
                let otherRenderInfo = this.index[otherName];
                let otherNode = otherRenderInfo
                    ? otherRenderInfo.node
                    : hierarchyNodeMap[otherName];
                // Determine whether this edge is a control edge between nodes where
                // either node is high-degree with respect to control edges. This will
                // be a signal to show it as an annotation instead of a bridge edge.
                let isHighDegreeControlEdge = !bridgeMetaedge.numRegularEdges &&
                    otherCounts.control[otherName] > PARAMS.maxControlDegree;
                let [, childAnnotations] = inbound
                    ? [renderNodeInfo.inAnnotations, childRenderInfo.inAnnotations]
                    : [renderNodeInfo.outAnnotations, childRenderInfo.outAnnotations];
                // Don't render a bridge path if the other node has in or out degree above
                // a threshold, lest bridge paths emanating out of a metagraph crowd up,
                // as was the case for the Fatcat LSTM lstm_1 > lstm_1 metagraph.
                let otherDegreeCount = (inbound ? otherCounts.out : otherCounts.in)[otherName];
                let isOtherHighDegree = otherDegreeCount > PARAMS.maxBridgePathDegree;
                // The adjoining render metaedge info from the parent's coreGraph, if any.
                // It will either be a Metaedge involving this node directly, if it
                // previously came from a metagraph, or it'll be a Metaedge involving
                // a previously created bridge node standing in for the other node.
                let adjoiningMetaedge = null;
                // We can only hope to render a bridge path if:
                //  - bridgegraph paths are enabled,
                //  - the other node is not too high-degree,
                //  - the child is in the core (not extracted for being high-degree), and
                //  - there's a path (in the traversal sense) between child and other.
                let canDrawBridgePath = false;
                if (
                    !isOtherHighDegree &&
                    !isHighDegreeControlEdge &&
                    childRenderInfo.isInCore()) {
                    // Utility function for finding an adjoining metaedge.
                    let findAdjoiningMetaedge = (targetName) => {
                        let adjoiningEdgeObj = inbound
                            ? { v: targetName, w: nodeName }
                            : { v: nodeName, w: targetName };
                        return (parentNodeInfo.coreGraph.edge(adjoiningEdgeObj));
                    };
                    adjoiningMetaedge = findAdjoiningMetaedge(otherName);
                    if (!adjoiningMetaedge) {
                        adjoiningMetaedge = findAdjoiningMetaedge(getBridgeNodeName(inbound, otherName, parentNode.name));
                    }
                    canDrawBridgePath = !!adjoiningMetaedge;
                }
                // Although dataflow edges are acyclic, control dependency edges may
                // actually point 'backwards' in the graph. If this bridgeMetaedge is
                // a control dependency, we need to determine whether it's backwards
                // pointing so that we render it appropriately.
                //
                // For instance, say we're rendering a graph with nodes named A/B and Z/Y,
                // and we're currently rendering the bridgegraph for A. Further, let's say
                // that there was an original BaseEdge from A/B->Z/Y and a CONTROL EDGE
                // from Z/Y=>A/B.
                //
                //     +----------------+
                //     | A              |
                //     |  +-----+       |         +------+
                //     |  | B   |>----->|>------->| Z    |
                //     |  |     |       |         |      |
                //     |  |     |   *   |         |      |
                //     |  |     |<=====<|<=======<|      |
                //     |  +-----+       |         +------+
                //     +----------------+
                //
                // When we render the subhierarchy for Metanode A, we'll come across a
                // control-only Metaedge in the bridgegraph from Z=>A/B (*). The question
                // is whether this edge is backwards.
                //
                // To answer that question, we follow the chain of adjoining metaedges
                // until we reach the topmost one. In this case, that's the control-only
                // Metaedge Z=>A in the ROOT's metagraph. We determine that this edge
                // is backwards by looking at the topological ordering of ROOT's metagraph
                // (which ignores control edges) and seeing that Z comes AFTER A.
                //
                // The property of being backwards is independent of whether the edge
                // is inbound or outbound. In the preceding example, if we were building
                // the subhierarchy for Z, we'd find bridge edge Z/Y=>A, walk to its
                // topmost adjoining metaedge Z=>A and discover that it's backwards.
                let backwards = false;
                if (adjoiningMetaedge && !bridgeMetaedge.numRegularEdges) {
                    // Find the top-most adjoining render metaedge information, and the
                    // GroupNode whose metagraph must contain the associated metaedge.
                    let topAdjoiningMetaedge = adjoiningMetaedge;
                    let topGroupNode = parentNodeInfo.node;
                    while (topAdjoiningMetaedge.adjoiningMetaedge) {
                        topAdjoiningMetaedge = topAdjoiningMetaedge.adjoiningMetaedge;
                        topGroupNode = topGroupNode.parentNode;
                    }
                    // Check against the topological ordering for the top node. The current
                    // bridge metaedge we're evaluating is backwards if its source comes
                    // after its destination.
                    let ordering = this.hierarchy.getTopologicalOrdering(topGroupNode.name);
                    let e = topAdjoiningMetaedge.metaedge;
                    backwards = ordering[e.v] > ordering[e.w];
                }
                // Render backwards control edges as annotations.
                canDrawBridgePath = canDrawBridgePath && !backwards;
                // If we can't make a bridge path for any reason, then we add an
                // annotation instead.
                if (!canDrawBridgePath) {
                    childAnnotations.push(new Annotation(otherNode, otherRenderInfo, new RenderMetaedgeInfo(bridgeMetaedge), AnnotationType.SHORTCUT, inbound));
                    return;
                }
                // At this point, all conditions have been met for drawing a bridge path.
                // Find or create the IN/OUT node representing otherNode.
                let bridgeContainerName = getBridgeNodeName(inbound, nodeName);
                let bridgeNodeName = getBridgeNodeName(inbound, otherName, nodeName);
                let bridgeNodeRenderInfo = coreGraph.node(bridgeNodeName);
                if (!bridgeNodeRenderInfo) {
                    // Find or create the directional container for the bridge node.
                    let bridgeContainerInfo = coreGraph.node(bridgeContainerName);
                    if (!bridgeContainerInfo) {
                        let bridgeContainerNode = {
                            // Important node properties.
                            name: bridgeContainerName,
                            type: NodeType.BRIDGE,
                            // Unused node properties.
                            isGroupNode: false,
                            cardinality: 0,
                            parentNode: null,
                            stats: null,
                            include: InclusionType.UNSPECIFIED,
                            // BridgeNode properties.
                            inbound: inbound,
                            nodeAttributes: {},
                        };
                        bridgeContainerInfo = new RenderNodeInfo(bridgeContainerNode);
                        this.index[bridgeContainerName] = bridgeContainerInfo;
                        coreGraph.setNode(bridgeContainerName, bridgeContainerInfo);
                    }
                    let bridgeNode = {
                        // Important node properties.
                        name: bridgeNodeName,
                        type: NodeType.BRIDGE,
                        // Unimportant node properties.
                        isGroupNode: false,
                        cardinality: 1,
                        parentNode: null,
                        stats: null,
                        include: InclusionType.UNSPECIFIED,
                        // BridgeNode properties.
                        inbound: inbound,
                        nodeAttributes: {},
                    };
                    bridgeNodeRenderInfo = new RenderNodeInfo(bridgeNode);
                    this.index[bridgeNodeName] = bridgeNodeRenderInfo;
                    coreGraph.setNode(bridgeNodeName, bridgeNodeRenderInfo);
                    // Set bridgeNode to be a graphlib child of the container node.
                    coreGraph.setParent(bridgeNodeName, bridgeContainerName);
                    bridgeContainerInfo.node.cardinality++;
                }
                // Create and add a bridge render metaedge.
                let bridgeRenderMetaedge = new RenderMetaedgeInfo(bridgeMetaedge);
                bridgeRenderMetaedge.adjoiningMetaedge = adjoiningMetaedge;
                inbound
                    ? coreGraph.setEdge(bridgeNodeName, childName, bridgeRenderMetaedge)
                    : coreGraph.setEdge(childName, bridgeNodeName, bridgeRenderMetaedge);
            }); // End _.each(bridgegraph.edges).
            // For each bridge container (IN and/or OUT), add structural edges between
            // terminal nodes and that container. A terminal node is one which has no
            // non-bridge edges in the direction of the container.
            //
            // For example, consider a Metanode A which contains two child nodes A/B
            // and A/C. Let's say it has one edge in the metagraph from A/B->A/C, and
            // one edge in the bridgegraph from Z->A/C.
            //
            // At this point, we've added a container bridge node IN to house all
            // incoming bridge nodes. We've also added a bridge node Z' (with parent IN)
            // to A, and a bridge edge from Z'->C.
            //
            //     +----------------------+
            //     | A          +---+     |
            //     |    +------>| C |     |
            //     |    |       +---+     |
            //     |    |         ^       |
            //     |    |         |       |
            //     |    |    +----|----+  |
            //     |    |    | IN |    |  |
            //     |  +---+  |  +---+  |  |
            //     |  | B |  |  | Z'|  |  |
            //     |  +---+  |  +---+  |  |
            //     |         +---------+  |
            //     +----------------------+
            //
            // With no other help, dagre would lay out B and Z' on the same level,
            // because both of them have no incoming edges. In other words, B is a
            // terminal node in the INCOMING direction.
            //
            // But we want to force dagre to lay out Z' (and everything in IN) lower
            // than all non-bridge nodes, so that there's enough room for the bridge
            // edges after they've been adjusted to meet up with paths coming in from
            // outside.
            //
            // To force Z' (and all other bridge nodes) to be lowest in the graph, we
            // identify terminal nodes like B and give them structural edges to
            // a new structural bridge node S which we add to IN.
            //
            //     +----------------------+
            //     | A          +---+     |
            //     |       +--->| C |     |
            //     |       |    +---+     |
            //     |     +---+    ^       |
            //     |     | B |    |       |
            //     |     +---+    |       |
            //     |       ^      |       |
            //     |       |      |       |
            //     |  +----|------|----+  |
            //     |  |IN  |      |    |  |
            //     |  |  +---+  +---+  |  |
            //     |  |  | S |  | Z'|  |  |
            //     |  |  +---+  +---+  |  |
            //     |  +----------------+  |
            //     +----------------------+
            //
            // This ensures that dagre will lay out the bridge containers strictly at
            // the ends of the graph. The structural edges will never be seen in the
            // visualization except as a debugging aid.
            lodash.each([true, false], (inbound) => {
                let bridgeContainerName = getBridgeNodeName(inbound, nodeName);
                let bridgeContainerInfo = coreGraph.node(bridgeContainerName);
                if (!bridgeContainerInfo) {
                    return;
                }
                lodash.each(coreGraph.nodes(), (childName) => {
                    // Short-circuit if this child is a bridge node or it's not a terminal
                    // node in the direction we're interested in.
                    let childNodeInfo = coreGraph.node(childName);
                    if (childNodeInfo.node.type === NodeType.BRIDGE) {
                        return;
                    }
                    let isTerminal = inbound
                        ? !coreGraph.predecessors(childName).length
                        : !coreGraph.successors(childName).length;
                    if (!isTerminal) {
                        return;
                    }
                    // Find or create a bridge node in the container for all structural
                    // metaedges. It would have been nice to skip this step and simply
                    // set a metaedge between the terminal node and the container node, but
                    // in that case, something about the graph upsets dagre.layout()'s
                    // longestPath algorithm (was getting errors due to an undefined).
                    let structuralNodeName = getBridgeNodeName(inbound, nodeName, 'STRUCTURAL_TARGET');
                    let structuralRenderInfo = coreGraph.node(structuralNodeName);
                    if (!structuralRenderInfo) {
                        let bridgeNode = {
                            // Important Node properties.
                            name: structuralNodeName,
                            type: NodeType.BRIDGE,
                            // Unimportant Node properties.
                            isGroupNode: false,
                            cardinality: 1,
                            parentNode: null,
                            stats: null,
                            include: InclusionType.UNSPECIFIED,
                            // BridgeNode properties.
                            inbound: inbound,
                            nodeAttributes: {},
                        };
                        structuralRenderInfo = new RenderNodeInfo(bridgeNode);
                        structuralRenderInfo.structural = true;
                        this.index[structuralNodeName] = structuralRenderInfo;
                        coreGraph.setNode(structuralNodeName, structuralRenderInfo);
                        bridgeContainerInfo.node.cardinality++;
                        coreGraph.setParent(structuralNodeName, bridgeContainerName);
                    }
                    // Create the structural Metaedge and insert it.
                    let structuralMetaedgeInfo = new RenderMetaedgeInfo(null);
                    structuralMetaedgeInfo.structural = true;
                    structuralMetaedgeInfo.weight--; // Reduce weight for dagre layout.
                    inbound
                        ? coreGraph.setEdge(structuralNodeName, childName, structuralMetaedgeInfo)
                        : coreGraph.setEdge(childName, structuralNodeName, structuralMetaedgeInfo);
                });
            });
        }
        /**
         * This method builds subhierarchies for function calls that are needed for
         * rendering edges in the current subhierarchy being built.
         *
         * When building subhierarchies for a metagraph M, the subhierarchies of
         * metanodes containing endpoint nodes for edges within metagraph M must
         * already be built. Otherwise, bridge edges will be missing from the graph.
         */
        buildSubhierarchiesForNeededFunctions(metagraph) {
            lodash.each(metagraph.edges(), (edgeObj) => {
                let metaedge = metagraph.edge(edgeObj);
                let renderMetaedgeInfo = new RenderMetaedgeInfo(metaedge);
                lodash.forEach(renderMetaedgeInfo.metaedge.baseEdgeList, (baseEdge) => {
                    const sourcePathList = baseEdge.v.split(NAMESPACE_DELIM);
                    for (let i = sourcePathList.length; i >= 0; i--) {
                        const fromBeginningPathList = sourcePathList.slice(0, i);
                        const node = this.hierarchy.node(fromBeginningPathList.join(NAMESPACE_DELIM));
                        if (node) {
                            if (node.type === NodeType.OP &&
                                this.hierarchy.libraryFunctions[node.op]) {
                                for (let j = 1; j < fromBeginningPathList.length; j++) {
                                    // Expand all hierarchies including the parent.
                                    const currentNodeName = fromBeginningPathList
                                        .slice(0, j)
                                        .join(NAMESPACE_DELIM);
                                    if (!currentNodeName) {
                                        continue;
                                    }
                                    // Build the hierarchy for this current level.
                                    this.buildSubhierarchy(currentNodeName);
                                }
                            }
                            // No need to analyze the other higher hierarchies.
                            break;
                        }
                    }
                });
            });
        }
    }
    /**
     * A class for rendering annotation object which contains label
     * about the node embedded as annotation, type of annotation and the location
     * of both the annotation's node and edge.
     *
     * Annotation objects include embedded constants, embedded summary, and
     * edge shortcuts.
     */
    class Annotation {
        /**
         * Creates a new Annotation.
         *
         * @param node The underlying node this annotation points to.
         * @param renderNodeInfo The render information for the underlying node
         *     this annotation points to. This can be null if the annotation
         *     denotes an embedding (constant, summary), in which case we
         *     use the node property.
         * @param renderMetaedgeInfo The render information for the edge associated
         *     with the annotation.
         * @param type The type of the annotation.
         * @param isIn True if it is an in-annotation. False if it is an
         *     out-annotation.
         */
        constructor(node, renderNodeInfo, renderMetaedgeInfo, type, isIn) {
            this.node = node;
            this.renderNodeInfo = renderNodeInfo;
            this.renderMetaedgeInfo = renderMetaedgeInfo;
            this.annotationType = type;
            // Properties specified by layout
            this.dx = 0;
            this.dy = 0;
            this.width = 0;
            this.height = 0;
            // Properties needed for generating an ID for the edge's path element if
            // this annotation is associated with a metaedge.
            if (renderMetaedgeInfo && renderMetaedgeInfo.metaedge) {
                this.v = renderMetaedgeInfo.metaedge.v;
                this.w = renderMetaedgeInfo.metaedge.w;
            }
            this.isIn = isIn;
            this.points = [];
        }
    }
    var AnnotationType;
    (function (AnnotationType) {
        AnnotationType[AnnotationType["SHORTCUT"] = 0] = "SHORTCUT";
        AnnotationType[AnnotationType["CONSTANT"] = 1] = "CONSTANT";
        AnnotationType[AnnotationType["SUMMARY"] = 2] = "SUMMARY";
        AnnotationType[AnnotationType["ELLIPSIS"] = 3] = "ELLIPSIS";
    })(AnnotationType || (AnnotationType = {}));
    /**
     * Manages a list of annotations. Two will be used for each
     * RenderNodeInfo, one for in annotations and one for out annotations.
     */
    class AnnotationList {
        constructor() {
            this.list = [];
            this.nodeNames = {};
        }
        /**
         * Append an annotation to the list, or a stand-in ellipsis annotation instead
         * if this would make it too many.
         */
        push(annotation) {
            if (annotation.node.name in this.nodeNames) {
                return; // Skip duplicate annotation.
            }
            this.nodeNames[annotation.node.name] = true;
            if (this.list.length < PARAMS.maxAnnotations) {
                this.list.push(annotation);
                return;
            }
            let lastAnnotation = this.list[this.list.length - 1];
            if (lastAnnotation.annotationType === AnnotationType.ELLIPSIS) {
                let ellipsisNode = lastAnnotation.node;
                ellipsisNode.setNumMoreNodes(++ellipsisNode.numMoreNodes);
                return;
            }
            let ellipsisNode = new EllipsisNodeImpl(1);
            this.list.push(new Annotation(ellipsisNode, new RenderNodeInfo(ellipsisNode), null, AnnotationType.ELLIPSIS, annotation.isIn));
        }
    }
    /**
     * Contains rendering information about a node in the hierarchical graph.
     */
    class RenderNodeInfo {
        constructor(node) {
            this.node = node;
            this.expanded = false;
            this.inAnnotations = new AnnotationList();
            this.outAnnotations = new AnnotationList();
            // Params specified by layout
            this.x = 0;
            this.y = 0;
            this.width = 0;
            this.height = 0;
            this.inboxWidth = 0;
            this.outboxWidth = 0;
            this.excluded = false;
            // Params for bridge paths.
            this.structural = false;
            // Params for node box.
            this.labelOffset = 0;
            this.radius = 0;
            // Params for expanded node
            this.labelHeight = 0;
            this.paddingTop = 0;
            this.paddingLeft = 0;
            this.paddingRight = 0;
            this.paddingBottom = 0;
            this.isInExtract = false;
            this.isOutExtract = false;
            this.coreBox = { width: 0, height: 0 };
            // By default, we don't fade nodes out. Default to false for safety.
            this.isFadedOut = false;
            // Only use the portion beyond the last delimiter as the display
            // name.
            this.displayName = node.name.substring(node.name.lastIndexOf(NAMESPACE_DELIM) + 1);
            if (node.type === NodeType.META && node.associatedFunction) {
                // Function names are suffixed with a length-8 hexadecimal string
                // followed by an optional number. We remove that suffix because
                // the user did not generate that suffix. That suffix merely
                // serves to differentiate between functions with different
                // signatures but the same name otherwise.
                // Furthermore, we remove the prefix that merely ascertains this
                // node as a function definition. There is no reason for the user
                // to see that in the graph, as the node would already be within
                // the functions scene group.
                const match = this.displayName.match(nodeDisplayNameRegex);
                if (match) {
                    // The display name had been successfully extracted. This is the most
                    // common scenario.
                    this.displayName = match[1];
                }
                else if (lodash.startsWith(this.displayName, FUNCTION_LIBRARY_NODE_PREFIX)) {
                    // The string does not match the usual pattern for how functions are
                    // named. Just use the entire second portion of the string as the name
                    // if we can successfully remove the prefix.
                    this.displayName = this.displayName.substring(FUNCTION_LIBRARY_NODE_PREFIX.length);
                }
            }
        }
        isInCore() {
            return !this.isInExtract && !this.isOutExtract && !this.isLibraryFunction;
        }
    }
    /**
     * Contains rendering information about a Metaedge from the underlying
     * hierarchical graph. It may be from either a metagraph or a bridgegraph.
     */
    class RenderMetaedgeInfo {
        constructor(metaedge) {
            this.metaedge = metaedge;
            this.adjoiningMetaedge = null;
            this.structural = false;
            this.weight = 1;
            this.isFadedOut = false;
        }
    }
    function addInAnnotation(node, predecessor, predecessorRenderInfo, edge, type) {
        let annotation = new Annotation(predecessor, predecessorRenderInfo, edge, type, true);
        node.inAnnotations.push(annotation);
    }
    function addOutAnnotation(node, successor, successorRenderInfo, edge, type) {
        let annotation = new Annotation(successor, successorRenderInfo, edge, type, false);
        node.outAnnotations.push(annotation);
    }
    function setGraphDepth(graph, depth) {
        lodash.each(graph.nodes(), (nodeName) => {
            let child = graph.node(nodeName);
            child.expanded = depth > 1; // set all child of depth 1 to collapsed
            if (depth > 0) {
                switch (child.node.type) {
                    case NodeType.META:
                    case NodeType.SERIES:
                        setGroupNodeDepth(child, depth - 1);
                        break;
                    // Do nothing for leaf
                }
            }
        });
    }
    class RenderGroupNodeInfo extends RenderNodeInfo {
        constructor(groupNode, graphOptions) {
            super(groupNode);
            let metagraph = groupNode.metagraph;
            let gl = metagraph.graph();
            graphOptions.compound = true;
            this.coreGraph = createGraph(gl.name, GraphType.CORE, graphOptions);
            this.inExtractBox = { width: 0, height: 0 };
            this.outExtractBox = { width: 0, height: 0 };
            this.libraryFunctionsBox = { width: 0, height: 0 };
            this.isolatedInExtract = [];
            this.isolatedOutExtract = [];
            this.libraryFunctionsExtract = [];
        }
    }
    function setGroupNodeDepth(renderInfo, depth) {
        if (renderInfo.coreGraph) {
            setGraphDepth(renderInfo.coreGraph, depth);
        }
    }
    /**
     * Remove an edge from the graph and add annotations to both ends of the edge.
     *
     * @param The core graph.
     * @param v Source name.
     * @param w Sink name.
     */
    function createShortcut(graph, v, w) {
        let src = graph.node(v);
        let sink = graph.node(w);
        let edge = graph.edge(v, w);
        // If either of the nodes is explicitly included in the main graph and
        // both nodes are in the main graph then do not create the shortcut
        // and instead keep the real edge.
        if ((src.node.include === InclusionType.INCLUDE ||
            sink.node.include === InclusionType.INCLUDE) &&
            src.node.include !== InclusionType.EXCLUDE &&
            sink.node.include !== InclusionType.EXCLUDE) {
            return;
        }
        // Add each annotation.
        addOutAnnotation(src, sink.node, sink, edge, AnnotationType.SHORTCUT);
        addInAnnotation(sink, src.node, src, edge, AnnotationType.SHORTCUT);
        // Remove the edge from the core graph.
        graph.removeEdge(v, w);
    }
    /**
     * Remove edges from a node, and set its isOutExtract property to true,
     * and remove the node and move it to isolatedOutExtract.
     *
     * If detachAllEdgesForHighDegree or forceDetach is true, extract all of its
     * edges. Otherwise, only extract all in-edges.
     */
    function makeOutExtract(renderNode, n, forceDetach) {
        let graph = renderNode.coreGraph;
        let child = graph.node(n);
        child.isOutExtract = true;
        lodash.each(graph.predecessors(n), (p, index) => {
            createShortcut(graph, p, n);
        });
        {
            lodash.each(graph.successors(n), (s, index) => {
                createShortcut(graph, n, s);
            });
        }
        // Remove the node from the core graph if it no longer has neighbors.
        if (graph.neighbors(n).length === 0) {
            child.node.include = InclusionType.EXCLUDE;
            renderNode.isolatedOutExtract.push(child);
            graph.removeNode(n);
        }
    }
    /**
     * Remove edges from a node, set its isInExtract property to true,
     * and remove the node and move it to isolatedInExtract.
     *
     * If detachAllEdgesForHighDegree or forceDetach is true, extract all of its
     * edges. Otherwise, only remove all out-edges.
     */
    function makeInExtract(renderNode, n, forceDetach) {
        let graph = renderNode.coreGraph;
        let child = graph.node(n);
        child.isInExtract = true;
        lodash.each(graph.successors(n), (s, index) => {
            createShortcut(graph, n, s);
        });
        {
            lodash.each(graph.predecessors(n), (p, index) => {
                createShortcut(graph, p, n);
            });
        }
        // Remove the node from the core graph if it no longer has neighbors.
        if (graph.neighbors(n).length === 0) {
            child.node.include = InclusionType.EXCLUDE;
            renderNode.isolatedInExtract.push(child);
            graph.removeNode(n);
        }
    }
    /**
     * Check whether the node's type is a member of the given list of types.
     *
     * @param node Node.
     * @param types List of type to match.
     */
    function hasTypeIn(node, types) {
        if (node.type === NodeType.OP) {
            for (let i = 0; i < types.length; i++) {
                if (node.op === types[i]) {
                    return true;
                }
            }
        }
        else if (node.type === NodeType.META) {
            let rootOpNode = node.getRootOp();
            if (rootOpNode) {
                for (let i = 0; i < types.length; i++) {
                    if (rootOpNode.op === types[i]) {
                        return true;
                    }
                }
            }
        }
        return false;
    }
    /** Move nodes that are specified to be excluded out of the core graph. */
    function extractSpecifiedNodes(renderNode) {
        let graph = renderNode.coreGraph;
        lodash.each(graph.nodes(), (n) => {
            let renderInfo = graph.node(n);
            if (renderInfo.node.include === InclusionType.EXCLUDE &&
                !n.startsWith(FUNCTION_LIBRARY_NODE_PREFIX)) {
                // Move the node if the node is excluded and not part of the library
                // function scene group, which contains nodes that do not represent ops in
                // the graph and should thus never have its nodes added to the core graph.
                if (renderNode.coreGraph.outEdges(n).length >
                    renderNode.coreGraph.inEdges(n).length) {
                    makeOutExtract(renderNode, n);
                }
                else {
                    makeInExtract(renderNode, n);
                }
            }
        });
    }
    /** Remove edges from pre-defined out-extract patterns */
    function extractPredefinedSink(renderNode) {
        let graph = renderNode.coreGraph;
        lodash.each(graph.nodes(), (n) => {
            let renderInfo = graph.node(n);
            if (renderInfo.node.include !== InclusionType.UNSPECIFIED) {
                return;
            }
            if (hasTypeIn(renderInfo.node, PARAMS.outExtractTypes)) {
                makeOutExtract(renderNode, n);
            }
        });
    }
    /** Remove edges from pre-defined in-extract patterns */
    function extractPredefinedSource(renderNode) {
        let graph = renderNode.coreGraph;
        lodash.each(graph.nodes(), (n) => {
            let renderInfo = graph.node(n);
            if (renderInfo.node.include !== InclusionType.UNSPECIFIED) {
                return;
            }
            if (hasTypeIn(renderInfo.node, PARAMS.inExtractTypes)) {
                makeInExtract(renderNode, n);
            }
        });
    }
    /** Extract nodes deemed to have either high in-degree or high out-degree. */
    function extractHighInOrOutDegree(renderNode) {
        let graph = renderNode.coreGraph;
        // Create mappings from node to in and out degrees. Count the number of valid
        // nodes along the way.
        let nodeToInDegree = {};
        let nodeToOutDegree = {};
        let validNodeCount = 0;
        lodash.each(graph.nodes(), (currentNode) => {
            if (graph.node(currentNode).node.include !== InclusionType.UNSPECIFIED) {
                // This node is not included in the first place.
                return;
            }
            // Count the in and out degrees based on only regular edges, unless there
            // are no regular edges, in which case use the number of control edges.
            // This is done so that control edges don't affect if nodes are extracted
            // from the core graph, unless the node is only used for control.
            let inDegree = lodash.reduce(graph.predecessors(currentNode), (inDegree, pred) => {
                let metaedge = graph.edge(pred, currentNode).metaedge;
                return inDegree + (metaedge.numRegularEdges ? 1 : 0);
            }, 0);
            if (inDegree === 0 && graph.predecessors(currentNode).length > 0) {
                inDegree = graph.predecessors(currentNode).length;
            }
            let outDegree = lodash.reduce(graph.successors(currentNode), (outDegree, succ) => {
                let metaedge = graph.edge(currentNode, succ).metaedge;
                return outDegree + (metaedge.numRegularEdges ? 1 : 0);
            }, 0);
            if (outDegree === 0 && graph.successors(currentNode).length > 0) {
                outDegree = graph.successors(currentNode).length;
            }
            // Store the in and out degrees of this node to avoid recomputing.
            nodeToInDegree[currentNode] = inDegree;
            nodeToOutDegree[currentNode] = outDegree;
            validNodeCount++;
        });
        if (validNodeCount < PARAMS.minNodeCountForExtraction) {
            // This graph has few nodes. Do not extract any nodes.
            return;
        }
        // We only extract if the node has a min in or out degree greater than this.
        let minUpperBound = PARAMS.minDegreeForExtraction - 1;
        // Mark for extraction nodes with in-degree > Q3 + (Q3 - Q1).
        let q3Index = Math.round(validNodeCount * 0.75);
        let q1Index = Math.round(validNodeCount * 0.25);
        let sortedByInDegree = Object.keys(nodeToInDegree).sort((node0, node1) => {
            return nodeToInDegree[node0] - nodeToInDegree[node1];
        });
        let inDegreeQ3 = nodeToInDegree[sortedByInDegree[q3Index]];
        let inDegreeQ1 = nodeToInDegree[sortedByInDegree[q1Index]];
        let inDegreeUpperBound = inDegreeQ3 + inDegreeQ3 - inDegreeQ1;
        // Only extract if the upper bound is high enough.
        inDegreeUpperBound = Math.max(inDegreeUpperBound, minUpperBound);
        for (let i = validNodeCount - 1; nodeToInDegree[sortedByInDegree[i]] > inDegreeUpperBound; i--) {
            // Extract a high in-degree node.
            makeInExtract(renderNode, sortedByInDegree[i]);
        }
        // Mark for extraction nodes with out-degree > Q3 + (Q3 - Q1) * 4.
        let sortedByOutDegree = Object.keys(nodeToOutDegree).sort((node0, node1) => {
            return nodeToOutDegree[node0] - nodeToOutDegree[node1];
        });
        let outDegreeQ3 = nodeToOutDegree[sortedByOutDegree[q3Index]];
        let outDegreeQ1 = nodeToOutDegree[sortedByOutDegree[q1Index]];
        // The upper bound for extracting out-degree nodes is higher than that for
        // extracting in-degree ones (Note the "* 4") because, in practice, some
        // graphs look worse with a smaller out-degree bound. For instance, a smaller
        // out-degree bound removes the convolution nodes from cifar 10 train's graph.
        let outDegreeUpperBound = outDegreeQ3 + (outDegreeQ3 - outDegreeQ1) * 4;
        // Only extract if the upper bound is high enough.
        outDegreeUpperBound = Math.max(outDegreeUpperBound, minUpperBound);
        for (let i = validNodeCount - 1; nodeToOutDegree[sortedByOutDegree[i]] > outDegreeUpperBound; i--) {
            let node = graph.node(sortedByOutDegree[i]);
            if (!node || node.isInExtract) {
                // This node has already been extracted due to high in-degree. It might
                // have been removed from the graph in general (during in-degree
                // extraction) due to a lack of neighbors. Do not extract this node twice.
                continue;
            }
            // Extract a high out-degree node that has not already been extracted.
            makeOutExtract(renderNode, sortedByOutDegree[i]);
        }
    }
    /** Remove control edges from nodes that have too many control edges */
    function removeControlEdges(renderNode) {
        let graph = renderNode.coreGraph;
        // Collect control edges into a map by node name.
        let map = {};
        lodash.each(graph.edges(), (e) => {
            if (!graph.edge(e).metaedge.numRegularEdges) {
                (map[e.v] = map[e.v] || []).push(e);
                (map[e.w] = map[e.w] || []).push(e);
            }
        });
        // For each node with too many control edges, turn them into annotations.
        lodash.each(map, (edges, nodeName) => {
            if (edges.length > PARAMS.maxControlDegree) {
                lodash.each(edges, (e) => createShortcut(graph, e.v, e.w));
            }
        });
    }
    /**
     * Remove edges and add to annotation instead.
     *
     * For root node, consider predefined types for source and sink.
     * We do not extract predefined type from non-root so that Variables and the
     * sgd node (op type = 'NoOp') do not get extract from inside own group.
     *
     * The order of extraction is important here as swapping the order can totally
     * screw up the graph layout.
     *
     * @param {Render.Node} renderNode Node to manipulate.
     */
    function extractHighDegrees(renderNode) {
        extractSpecifiedNodes(renderNode);
        if (PARAMS.outExtractTypes) {
            extractPredefinedSink(renderNode);
        }
        // This has to come before extract high in-degree to protect the core part
        // that takes many variables.
        if (PARAMS.inExtractTypes) {
            extractPredefinedSource(renderNode);
        }
        extractHighInOrOutDegree(renderNode);
        {
            removeControlEdges(renderNode);
        }
        // Extract isolated nodes, which can be
        // (1) source-like and sink-like nodes that are not originally isolated but
        //     become isolated after further removal.
        // (2) isolated nodes with annotations on one-side.  These might be either
        //     - nodes that originally have high out-degree but because we remove
        //       high in-degree nodes first, they no longer have high in-degree when
        //       we check.  (Detecting all high-degree before removing also leads to
        //       another problem.)
        //     - nodes that do not have high degree, but their neighbors are all
        //       extracted, so it might make sense to extract them too.
        let graph = renderNode.coreGraph;
        lodash.each(graph.nodes(), (n) => {
            let child = graph.node(n);
            let degree = graph.neighbors(n).length;
            if (child.node.include !== InclusionType.UNSPECIFIED) {
                return;
            }
            if (degree === 0) {
                let hasOutAnnotations = child.outAnnotations.list.length > 0;
                let hasInAnnotations = child.inAnnotations.list.length > 0;
                if (child.isInExtract) {
                    // Is source-like.
                    // This case only happens if detachAllEdgesForHighDegree is false.
                    // (Otherwise all source-like nodes are all isolated already.)
                    renderNode.isolatedInExtract.push(child);
                    child.node.include = InclusionType.EXCLUDE;
                    graph.removeNode(n);
                }
                else if (child.isOutExtract) {
                    // Is sink-like.
                    // This case only happens if detachAllEdgesForHighDegree is false.
                    // // (Otherwise all sink-like nodes are all isolated already.)
                    renderNode.isolatedOutExtract.push(child);
                    child.node.include = InclusionType.EXCLUDE;
                    graph.removeNode(n);
                }
                else {
                    if (hasOutAnnotations && !hasInAnnotations) {
                        child.isInExtract = true; // for ones with high out-annotations
                        renderNode.isolatedInExtract.push(child);
                        child.node.include = InclusionType.EXCLUDE;
                        graph.removeNode(n);
                    }
                    else if (hasInAnnotations && !hasOutAnnotations) {
                        child.isOutExtract = true; // for ones with high in-annotations
                        renderNode.isolatedOutExtract.push(child);
                        child.node.include = InclusionType.EXCLUDE;
                        graph.removeNode(n);
                    }
                    else ;
                }
            }
        });
    }
    /**
     * Expands nodes in the graph until the desired node is visible.
     *
     * @param scene The scene polymer component.
     * @param renderHierarchy The render hierarchy.
     * @param tensorName The name of a tensor.
     * @return A string that is the name of the node representing the given tensor.
     *     Note that the original tensor name might differ from this returned node
     *     name. Specifically, for instance, the tensor name usually ends with an
     *     output slot index (such as :0), while the node name lacks that suffix.
     */
    function expandUntilNodeIsShown(scene, renderHierarchy, tensorName) {
        const splitTensorName = tensorName.split('/');
        // Graph names do not take into account the output slot. Strip it.
        const lastNodeNameMatch = splitTensorName[splitTensorName.length - 1].match(/(.*):\w+/);
        if (lastNodeNameMatch.length === 2) {
            splitTensorName[splitTensorName.length - 1] = lastNodeNameMatch[1];
        }
        let nodeName = splitTensorName[0];
        let renderNode = renderHierarchy.getRenderNodeByName(nodeName);
        for (let i = 1; i < splitTensorName.length; i++) {
            // Op nodes are not expandable.
            if (renderNode.node.type === NodeType.OP) {
                break;
            }
            renderHierarchy.buildSubhierarchy(nodeName);
            renderNode.expanded = true;
            scene.setNodeExpanded(renderNode);
            nodeName += '/' + splitTensorName[i];
            renderNode = renderHierarchy.getRenderNodeByName(nodeName);
        }
        return renderNode.node.name;
    }

    /* Copyright 2015 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the 'License');
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an 'AS IS' BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    const PARAMS$1 = {
        animation: {
            /** Default duration for graph animations in ms. */
            duration: 250,
        },
        graph: {
            /** Graph parameter for metanode. */
            meta: {
                /**
                 * Dagre's nodesep param - number of pixels that
                 * separate nodes horizontally in the layout.
                 *
                 * See https://github.com/cpettitt/dagre/wiki#configuring-the-layout
                 */
                nodeSep: 5,
                /**
                 * Dagre's ranksep param - number of pixels
                 * between each rank in the layout.
                 *
                 * See https://github.com/cpettitt/dagre/wiki#configuring-the-layout
                 */
                rankSep: 25,
                /**
                 * Dagre's edgesep param - number of pixels that separate
                 * edges horizontally in the layout.
                 */
                edgeSep: 5,
            },
            /** Graph parameter for metanode. */
            series: {
                /**
                 * Dagre's nodesep param - number of pixels that
                 * separate nodes horizontally in the layout.
                 *
                 * See https://github.com/cpettitt/dagre/wiki#configuring-the-layout
                 */
                nodeSep: 5,
                /**
                 * Dagre's ranksep param - number of pixels
                 * between each rank in the layout.
                 *
                 * See https://github.com/cpettitt/dagre/wiki#configuring-the-layout
                 */
                rankSep: 25,
                /**
                 * Dagre's edgesep param - number of pixels that separate
                 * edges horizontally in the layout.
                 */
                edgeSep: 5,
            },
            /**
             * Padding is used to correctly position the graph SVG inside of its parent
             * element. The padding amounts are applied using an SVG transform of X and
             * Y coordinates.
             */
            padding: { paddingTop: 40, paddingLeft: 20 },
        },
        subscene: {
            meta: {
                paddingTop: 10,
                paddingBottom: 10,
                paddingLeft: 10,
                paddingRight: 10,
                /**
                 * Used to leave room for the label on top of the highest node in
                 * the core graph.
                 */
                labelHeight: 20,
                /** X-space between each extracted node and the core graph. */
                extractXOffset: 15,
                /** Y-space between each extracted node. */
                extractYOffset: 20,
            },
            series: {
                paddingTop: 10,
                paddingBottom: 10,
                paddingLeft: 10,
                paddingRight: 10,
                labelHeight: 10,
            },
        },
        nodeSize: {
            /** Size of meta nodes. */
            meta: {
                radius: 5,
                width: 60,
                maxLabelWidth: 52,
                /** A scale for the node's height based on number of nodes inside */
                // Hack - set this as an any type to avoid issues in exporting a type
                // from an external module.
                height: linear$2()
                    .domain([1, 200])
                    .range([15, 60])
                    .clamp(true),
                /** The radius of the circle denoting the expand button. */
                expandButtonRadius: 3,
            },
            /** Size of op nodes. */
            op: {
                width: 15,
                height: 6,
                radius: 3,
                labelOffset: -8,
                maxLabelWidth: 30,
            },
            /** Size of series nodes. */
            series: {
                expanded: {
                    // For expanded series nodes, width and height will be
                    // computed to account for the subscene.
                    radius: 10,
                    labelOffset: 0,
                },
                vertical: {
                    // When unexpanded, series whose underlying metagraphs contain
                    // one or more non-control edges will show as a vertical stack
                    // of ellipses.
                    width: 16,
                    height: 13,
                    labelOffset: -13,
                },
                horizontal: {
                    // When unexpanded, series whose underlying metagraphs contain
                    // no non-control edges will show as a horizontal stack of
                    // ellipses.
                    width: 24,
                    height: 8,
                    radius: 10,
                    labelOffset: -10,
                },
            },
            /** Size of bridge nodes. */
            bridge: {
                // NOTE: bridge nodes will normally be invisible, but they must
                // take up some space so that the layout step leaves room for
                // their edges.
                width: 20,
                height: 20,
                radius: 2,
                labelOffset: 0,
            },
        },
        shortcutSize: {
            /** Size of shortcuts for op nodes */
            op: { width: 10, height: 4 },
            /** Size of shortcuts for meta nodes */
            meta: { width: 12, height: 4, radius: 1 },
            /** Size of shortcuts for series nodes */
            series: {
                width: 14,
                height: 4,
            },
        },
        annotations: {
            /** Maximum possible width of the bounding box for in annotations */
            inboxWidth: 50,
            /** Maximum possible width of the bounding box for out annotations */
            outboxWidth: 50,
            /** X-space between the shape and each annotation-node. */
            xOffset: 10,
            /** Y-space between each annotation-node. */
            yOffset: 3,
            /** X-space between each annotation-node and its label. */
            labelOffset: 2,
            /** Defines the max width for annotation label */
            maxLabelWidth: 120,
        },
        constant: { size: { width: 4, height: 4 } },
        series: {
            /** Maximum number of repeated item for unexpanded series node. */
            maxStackCount: 3,
            /**
             * Positioning offset ratio for collapsed stack
             * of parallel series (series without edges between its members).
             */
            parallelStackOffsetRatio: 0.2,
            /**
             * Positioning offset ratio for collapsed stack
             * of tower series (series with edges between its members).
             */
            towerStackOffsetRatio: 0.5,
        },
        minimap: {
            /** The maximum width/height the minimap can have. */
            size: 150,
        },
    };
    /**
     * The minimum width we confer upon the auxiliary nodes section if functions
     * also appear. Without enforcing this minimum, metanodes in the function
     * library section could jut into the auxiliary nodes section because the
     * title "Auxiliary Nodes" is longer than the width of the auxiliary nodes
     * section itself.
     */
    const MIN_AUX_WIDTH = 140;
    /** Calculate layout for a scene of a group node. */
    function layoutScene(renderNodeInfo) {
        // Update layout, size, and annotations of its children nodes and edges.
        if (renderNodeInfo.node.isGroupNode) {
            layoutChildren(renderNodeInfo);
        }
        // Update position of its children nodes and edges
        if (renderNodeInfo.node.type === NodeType.META) {
            layoutMetanode(renderNodeInfo);
        }
        else if (renderNodeInfo.node.type === NodeType.SERIES) {
            layoutSeriesNode(renderNodeInfo);
        }
    }
    /**
     * Updates the total width of an unexpanded node which includes the size of its
     * in and out annotations.
     */
    function updateTotalWidthOfNode(renderInfo) {
        renderInfo.inboxWidth =
            renderInfo.inAnnotations.list.length > 0
                ? PARAMS$1.annotations.inboxWidth
                : 0;
        renderInfo.outboxWidth =
            renderInfo.outAnnotations.list.length > 0
                ? PARAMS$1.annotations.outboxWidth
                : 0;
        // Assign the width of the core box (the main shape of the node).
        renderInfo.coreBox.width = renderInfo.width;
        renderInfo.coreBox.height = renderInfo.height;
        // TODO: Account for font width rather than using a magic number.
        let labelLength = renderInfo.displayName.length;
        let charWidth = 3; // 3 pixels per character.
        // Compute the total width of the node.
        renderInfo.width = Math.max(renderInfo.coreBox.width + renderInfo.inboxWidth + renderInfo.outboxWidth, labelLength * charWidth);
    }
    /**
     * Update layout, size, and annotations of its children nodes and edges.
     */
    function layoutChildren(renderNodeInfo) {
        let children = renderNodeInfo.coreGraph
            .nodes()
            .map((n) => {
            return renderNodeInfo.coreGraph.node(n);
        })
            .concat(renderNodeInfo.isolatedInExtract, renderNodeInfo.isolatedOutExtract, renderNodeInfo.libraryFunctionsExtract);
        lodash.each(children, (childNodeInfo) => {
            // Set size of each child
            switch (childNodeInfo.node.type) {
                case NodeType.OP:
                    lodash.extend(childNodeInfo, PARAMS$1.nodeSize.op);
                    break;
                case NodeType.BRIDGE:
                    lodash.extend(childNodeInfo, PARAMS$1.nodeSize.bridge);
                    break;
                case NodeType.META:
                    if (!childNodeInfo.expanded) {
                        // Set fixed width and scalable height based on cardinality
                        lodash.extend(childNodeInfo, PARAMS$1.nodeSize.meta);
                        childNodeInfo.height = PARAMS$1.nodeSize.meta.height(childNodeInfo.node.cardinality);
                    }
                    else {
                        let childGroupNodeInfo = childNodeInfo;
                        layoutScene(childGroupNodeInfo); // Recursively layout its subscene.
                    }
                    break;
                case NodeType.SERIES:
                    if (childNodeInfo.expanded) {
                        lodash.extend(childNodeInfo, PARAMS$1.nodeSize.series.expanded);
                        let childGroupNodeInfo = childNodeInfo;
                        layoutScene(childGroupNodeInfo); // Recursively layout its subscene.
                    }
                    else {
                        let childGroupNodeInfo = childNodeInfo;
                        let seriesParams = childGroupNodeInfo.node.hasNonControlEdges
                            ? PARAMS$1.nodeSize.series.vertical
                            : PARAMS$1.nodeSize.series.horizontal;
                        lodash.extend(childNodeInfo, seriesParams);
                    }
                    break;
                default:
                    throw Error('Unrecognized node type: ' + childNodeInfo.node.type);
            }
            // Compute total width of un-expanded nodes. Width of expanded nodes
            // has already been computed.
            if (!childNodeInfo.expanded) {
                updateTotalWidthOfNode(childNodeInfo);
            }
            // Layout each child's annotations
            layoutAnnotation(childNodeInfo);
        });
    }
    /**
     * Calculate layout for a graph using dagre
     * @param graph the graph to be laid out
     * @param params layout parameters
     * @return width and height of the core graph
     */
    function dagreLayout(graph, params) {
        lodash.extend(graph.graph(), {
            nodesep: params.nodeSep,
            ranksep: params.rankSep,
            edgesep: params.edgeSep,
        });
        let nonBridgeNodeNames = [];
        // Split out nodes into bridge and non-bridge nodes, and calculate the total
        // width we should use for bridge nodes.
        lodash.each(graph.nodes(), (nodeName) => {
            let nodeInfo = graph.node(nodeName);
            if (nodeInfo.node.type === NodeType.BRIDGE) ;
            else {
                nonBridgeNodeNames.push(nodeName);
            }
        });
        // If there are no non-bridge nodes, then the graph has zero size.
        if (!nonBridgeNodeNames.length) {
            return {
                width: 0,
                height: 0,
            };
        }
        dagre.layout(graph);
        // Calculate the true bounding box of the graph by iterating over nodes and
        // edges rather than accepting dagre's word for it. In particular, we should
        // ignore the extra-wide bridge nodes and bridge edges, and allow for
        // annotation boxes and labels.
        let minX = Infinity;
        let minY = Infinity;
        let maxX = -Infinity;
        let maxY = -Infinity;
        lodash.each(nonBridgeNodeNames, (nodeName) => {
            let nodeInfo = graph.node(nodeName);
            let w = 0.5 * nodeInfo.width;
            let x1 = nodeInfo.x - w;
            let x2 = nodeInfo.x + w;
            minX = x1 < minX ? x1 : minX;
            maxX = x2 > maxX ? x2 : maxX;
            // TODO: Account for the height of labels above op nodes here.
            let h = 0.5 * nodeInfo.height;
            let y1 = nodeInfo.y - h;
            let y2 = nodeInfo.y + h;
            minY = y1 < minY ? y1 : minY;
            maxY = y2 > maxY ? y2 : maxY;
        });
        lodash.each(graph.edges(), (edgeObj) => {
            let edgeInfo = graph.edge(edgeObj);
            if (edgeInfo.structural) {
                return; // Skip structural edges from min/max calculations.
            }
            // Since the node size passed to dagre includes the in and out
            // annotations, the endpoints of the edge produced by dagre may not
            // point to the actual node shape (rectangle, ellipse). We correct the
            // end-points by finding the intersection of a line between the
            // next-to-last (next-to-first) point and the destination (source)
            // rectangle.
            let sourceNode = graph.node(edgeInfo.metaedge.v);
            let destNode = graph.node(edgeInfo.metaedge.w);
            // Straight 3-points edges are special case, since they are curved after
            // our default correction. To keep them straight, we remove the mid point
            // and correct the first and the last point to be the center of the
            // source and destination node respectively.
            if (edgeInfo.points.length === 3 && isStraightLine(edgeInfo.points)) {
                if (sourceNode != null) {
                    let cxSource = sourceNode.expanded
                        ? sourceNode.x
                        : computeCXPositionOfNodeShape(sourceNode);
                    edgeInfo.points[0].x = cxSource;
                }
                if (destNode != null) {
                    let cxDest = destNode.expanded
                        ? destNode.x
                        : computeCXPositionOfNodeShape(destNode);
                    edgeInfo.points[2].x = cxDest;
                }
                // Remove the middle point so the edge doesn't curve.
                edgeInfo.points = [edgeInfo.points[0], edgeInfo.points[1]];
            }
            // Correct the destination endpoint of the edge.
            let nextToLastPoint = edgeInfo.points[edgeInfo.points.length - 2];
            // The destination node might be null if this is a bridge edge.
            if (destNode != null) {
                edgeInfo.points[edgeInfo.points.length - 1] = intersectPointAndNode(nextToLastPoint, destNode);
            }
            // Correct the source endpoint of the edge.
            let secondPoint = edgeInfo.points[1];
            // The source might be null if this is a bridge edge.
            if (sourceNode != null) {
                edgeInfo.points[0] = intersectPointAndNode(secondPoint, sourceNode);
            }
            lodash.each(edgeInfo.points, (point) => {
                minX = point.x < minX ? point.x : minX;
                maxX = point.x > maxX ? point.x : maxX;
                minY = point.y < minY ? point.y : minY;
                maxY = point.y > maxY ? point.y : maxY;
            });
        });
        // Shift all nodes and edge points to account for the left-padding amount,
        // and the invisible bridge nodes.
        lodash.each(graph.nodes(), (nodeName) => {
            let nodeInfo = graph.node(nodeName);
            nodeInfo.x -= minX;
            nodeInfo.y -= minY;
        });
        lodash.each(graph.edges(), (edgeObj) => {
            lodash.each(graph.edge(edgeObj).points, (point) => {
                point.x -= minX;
                point.y -= minY;
            });
        });
        return {
            width: maxX - minX,
            height: maxY - minY,
        };
    }
    /** Layout a metanode. Only called for an expanded node. */
    function layoutMetanode(renderNodeInfo) {
        // First, copy params specific to meta nodes onto this render info object.
        let params = PARAMS$1.subscene.meta;
        lodash.extend(renderNodeInfo, params);
        // Invoke dagre.layout() on the core graph and record the bounding box
        // dimensions.
        lodash.extend(renderNodeInfo.coreBox, dagreLayout(renderNodeInfo.coreGraph, PARAMS$1.graph.meta));
        // Calculate the position of nodes in isolatedInExtract relative to the
        // top-left corner of inExtractBox (the bounding box for all inExtract nodes)
        // and calculate the size of the inExtractBox.
        let maxInExtractWidth = renderNodeInfo.isolatedInExtract.length
            ? lodash.maxBy(renderNodeInfo.isolatedInExtract, (renderNode) => renderNode.width).width
            : null;
        renderNodeInfo.inExtractBox.width =
            maxInExtractWidth != null ? maxInExtractWidth : 0;
        renderNodeInfo.inExtractBox.height = lodash.reduce(renderNodeInfo.isolatedInExtract, (height, child, i) => {
            let yOffset = i > 0 ? params.extractYOffset : 0;
            // use width/height here to avoid overlaps between extracts
            child.x = 0;
            child.y = height + yOffset + child.height / 2;
            return height + yOffset + child.height;
        }, 0);
        // Calculate the position of nodes in isolatedOutExtract relative to the
        // top-left corner of outExtractBox (the bounding box for all outExtract
        // nodes) and calculate the size of the outExtractBox.
        let maxOutExtractWidth = renderNodeInfo.isolatedOutExtract.length
            ? lodash.maxBy(renderNodeInfo.isolatedOutExtract, (renderNode) => renderNode.width).width
            : null;
        renderNodeInfo.outExtractBox.width =
            maxOutExtractWidth != null ? maxOutExtractWidth : 0;
        renderNodeInfo.outExtractBox.height = lodash.reduce(renderNodeInfo.isolatedOutExtract, (height, child, i) => {
            let yOffset = i > 0 ? params.extractYOffset : 0;
            // use width/height here to avoid overlaps between extracts
            child.x = 0;
            child.y = height + yOffset + child.height / 2;
            return height + yOffset + child.height;
        }, 0);
        // Calculate the position of nodes in libraryFunctionsExtract relative to the
        // top-left corner of libraryFunctionsBox (the bounding box for all library
        // function nodes) and calculate the size of the libraryFunctionsBox.
        let maxLibraryFunctionsWidth = renderNodeInfo.libraryFunctionsExtract.length
            ? lodash.maxBy(renderNodeInfo.libraryFunctionsExtract, (renderNode) => renderNode.width).width
            : null;
        renderNodeInfo.libraryFunctionsBox.width =
            maxLibraryFunctionsWidth != null ? maxLibraryFunctionsWidth : 0;
        renderNodeInfo.libraryFunctionsBox.height = lodash.reduce(renderNodeInfo.libraryFunctionsExtract, (height, child, i) => {
            let yOffset = i > 0 ? params.extractYOffset : 0;
            // use width/height here to avoid overlaps between extracts
            child.x = 0;
            child.y = height + yOffset + child.height / 2;
            return height + yOffset + child.height;
        }, 0);
        // Compute the total padding between the core graph, in-extract and
        // out-extract boxes.
        let numParts = 0;
        if (renderNodeInfo.isolatedInExtract.length > 0) {
            numParts++;
        }
        if (renderNodeInfo.isolatedOutExtract.length > 0) {
            numParts++;
        }
        if (renderNodeInfo.libraryFunctionsExtract.length > 0) {
            numParts++;
        }
        if (renderNodeInfo.coreGraph.nodeCount() > 0) {
            numParts++;
        }
        let offset = PARAMS$1.subscene.meta.extractXOffset;
        let padding = numParts <= 1 ? 0 : numParts * offset;
        // Add the in-extract and out-extract width to the core box width. Do not let
        // the auxiliary width be too small, lest it be smaller than the title.
        const auxWidth = Math.max(MIN_AUX_WIDTH, renderNodeInfo.inExtractBox.width + renderNodeInfo.outExtractBox.width);
        renderNodeInfo.coreBox.width +=
            auxWidth + padding + renderNodeInfo.libraryFunctionsBox.width + padding;
        renderNodeInfo.coreBox.height =
            params.labelHeight +
                Math.max(renderNodeInfo.inExtractBox.height, renderNodeInfo.coreBox.height, renderNodeInfo.libraryFunctionsBox.height, renderNodeInfo.outExtractBox.height);
        // Determine the whole metanode's width (from left to right).
        renderNodeInfo.width =
            renderNodeInfo.coreBox.width + params.paddingLeft + params.paddingRight;
        // Determine the whole metanode's height (from top to bottom).
        renderNodeInfo.height =
            renderNodeInfo.paddingTop +
                renderNodeInfo.coreBox.height +
                renderNodeInfo.paddingBottom;
    }
    /**
     * Calculate layout for series node's core graph. Only called for an expanded
     * series.
     */
    function layoutSeriesNode(node) {
        let graph = node.coreGraph;
        let params = PARAMS$1.subscene.series;
        lodash.extend(node, params);
        // Layout the core.
        lodash.extend(node.coreBox, dagreLayout(node.coreGraph, PARAMS$1.graph.series));
        lodash.each(graph.nodes(), (nodeName) => {
            graph.node(nodeName).excluded = false;
        });
        // Series do not have in/outExtractBox so no need to include them here.
        node.width = node.coreBox.width + params.paddingLeft + params.paddingRight;
        node.height = node.coreBox.height + params.paddingTop + params.paddingBottom;
    }
    /**
     * Calculate layout for annotations of a given node.
     * This will modify positions of the given node and its annotations.
     *
     * @see tf.graph.render.Node and tf.graph.render.Annotation
     * for description of each property of each render node.
     *
     */
    function layoutAnnotation(renderNodeInfo) {
        // If the render node is an expanded metanode, then its annotations will not
        // be visible and we should skip the annotation calculations.
        if (renderNodeInfo.expanded) {
            return;
        }
        let inAnnotations = renderNodeInfo.inAnnotations.list;
        let outAnnotations = renderNodeInfo.outAnnotations.list;
        // Calculate size for in-annotations
        lodash.each(inAnnotations, (a) => sizeAnnotation(a));
        // Calculate size for out-annotations
        lodash.each(outAnnotations, (a) => sizeAnnotation(a));
        let params = PARAMS$1.annotations;
        // Calculate annotation node position (a.dx, a.dy)
        // and total height for in-annotations
        // After this chunk of code:
        // inboxHeight = sum of annotation heights+ (annotation.length - 1 * yOffset)
        let inboxHeight = lodash.reduce(inAnnotations, (height, a, i) => {
            let yOffset = i > 0 ? params.yOffset : 0;
            a.dx = -(renderNodeInfo.coreBox.width + a.width) / 2 - params.xOffset;
            a.dy = height + yOffset + a.height / 2;
            return height + yOffset + a.height;
        }, 0);
        lodash.each(inAnnotations, (a) => {
            a.dy -= inboxHeight / 2;
            a.labelOffset = params.labelOffset;
        });
        // Calculate annotation node position (a.dx, a.dy)
        // and total height for out-annotations
        // After this chunk of code:
        // outboxHeight = sum of annotation heights +
        //                (annotation.length - 1 * yOffset)
        let outboxHeight = lodash.reduce(outAnnotations, (height, a, i) => {
            let yOffset = i > 0 ? params.yOffset : 0;
            a.dx = (renderNodeInfo.coreBox.width + a.width) / 2 + params.xOffset;
            a.dy = height + yOffset + a.height / 2;
            return height + yOffset + a.height;
        }, 0);
        lodash.each(outAnnotations, (a) => {
            // adjust by (half of ) the total height
            // so dy is relative to the host node's center.
            a.dy -= outboxHeight / 2;
            a.labelOffset = params.labelOffset;
        });
        // Creating scales for touch point between the in-annotation edges
        // and their hosts.
        let inTouchHeight = Math.min(renderNodeInfo.height / 2 - renderNodeInfo.radius, inboxHeight / 2);
        inTouchHeight = inTouchHeight < 0 ? 0 : inTouchHeight;
        let inY = linear$2()
            .domain([0, inAnnotations.length - 1])
            .range([-inTouchHeight, inTouchHeight]);
        // Calculate annotation edge position
        lodash.each(inAnnotations, (a, i) => {
            a.points = [
                // The annotation node end
                {
                    dx: a.dx + a.width / 2,
                    dy: a.dy,
                },
                // The host node end
                {
                    dx: -renderNodeInfo.coreBox.width / 2,
                    // only use scale if there are more than one,
                    // otherwise center it vertically
                    dy: inAnnotations.length > 1 ? inY(i) : 0,
                },
            ];
        });
        // Creating scales for touch point between the out-annotation edges
        // and their hosts.
        let outTouchHeight = Math.min(renderNodeInfo.height / 2 - renderNodeInfo.radius, outboxHeight / 2);
        outTouchHeight = outTouchHeight < 0 ? 0 : outTouchHeight;
        let outY = linear$2()
            .domain([0, outAnnotations.length - 1])
            .range([-outTouchHeight, outTouchHeight]);
        lodash.each(outAnnotations, (a, i) => {
            // Add point from the border of the annotation node
            a.points = [
                // The host node end
                {
                    dx: renderNodeInfo.coreBox.width / 2,
                    // only use scale if there are more than one,
                    // otherwise center it vertically
                    dy: outAnnotations.length > 1 ? outY(i) : 0,
                },
                // The annotation node end
                {
                    dx: a.dx - a.width / 2,
                    dy: a.dy,
                },
            ];
        });
        renderNodeInfo.height = Math.max(renderNodeInfo.height, inboxHeight, outboxHeight);
    }
    /**
     * Set size of an annotation node.
     */
    function sizeAnnotation(a) {
        switch (a.annotationType) {
            case AnnotationType.CONSTANT:
                lodash.extend(a, PARAMS$1.constant.size);
                break;
            case AnnotationType.SHORTCUT:
                if (a.node.type === NodeType.OP) {
                    lodash.extend(a, PARAMS$1.shortcutSize.op);
                }
                else if (a.node.type === NodeType.META) {
                    lodash.extend(a, PARAMS$1.shortcutSize.meta);
                }
                else if (a.node.type === NodeType.SERIES) {
                    lodash.extend(a, PARAMS$1.shortcutSize.series);
                }
                else {
                    throw Error('Invalid node type: ' + a.node.type);
                }
                break;
            case AnnotationType.SUMMARY:
                lodash.extend(a, PARAMS$1.constant.size);
                break;
        }
    }
    /**
     * Determines the center position of the node's shape. The position depends
     * on if the node has in and out-annotations.
     */
    function computeCXPositionOfNodeShape(renderInfo) {
        if (renderInfo.expanded) {
            return renderInfo.x;
        }
        let dx = renderInfo.inAnnotations.list.length ? renderInfo.inboxWidth : 0;
        return (renderInfo.x - renderInfo.width / 2 + dx + renderInfo.coreBox.width / 2);
    }
    /** Returns the angle (in degrees) between two points. */
    function angleBetweenTwoPoints(a, b) {
        let dx = b.x - a.x;
        let dy = b.y - a.y;
        return (180 * Math.atan(dy / dx)) / Math.PI;
    }
    /**
     * Returns if a line going through the specified points is a straight line.
     */
    function isStraightLine(points) {
        let angle = angleBetweenTwoPoints(points[0], points[1]);
        for (let i = 1; i < points.length - 1; i++) {
            let newAngle = angleBetweenTwoPoints(points[i], points[i + 1]);
            // Have a tolerance of 1 degree.
            if (Math.abs(newAngle - angle) > 1) {
                return false;
            }
            angle = newAngle;
        }
        return true;
    }
    /**
     * Returns the intersection of a line between the provided point
     * and the provided rectangle.
     */
    function intersectPointAndNode(point, node) {
        // cx and cy are the center of the rectangle.
        let cx = node.expanded ? node.x : computeCXPositionOfNodeShape(node);
        let cy = node.y;
        // Calculate the slope
        let dx = point.x - cx;
        let dy = point.y - cy;
        let w = node.expanded ? node.width : node.coreBox.width;
        let h = node.expanded ? node.height : node.coreBox.height;
        let deltaX, deltaY;
        if ((Math.abs(dy) * w) / 2 > (Math.abs(dx) * h) / 2) {
            // The intersection is above or below the rectangle.
            if (dy < 0) {
                h = -h;
            }
            deltaX = dy === 0 ? 0 : ((h / 2) * dx) / dy;
            deltaY = h / 2;
        }
        else {
            // The intersection is left or right of the rectangle.
            if (dx < 0) {
                w = -w;
            }
            deltaX = w / 2;
            deltaY = dx === 0 ? 0 : ((w / 2) * dy) / dx;
        }
        return { x: cx + deltaX, y: cy + deltaY };
    }

    /* Copyright 2015 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the 'License');
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an 'AS IS' BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    const selectChild$1 = selectChild;
    const Class$2 = Class$1;
    /**
     * The dimensions of the minimap including padding and margin.
     */
    const MINIMAP_BOX_WIDTH = 320;
    const MINIMAP_BOX_HEIGHT = 150;
    let healthPillEntries = [
        {
            background_color: '#CC2F2C',
            label: 'NaN',
        },
        {
            background_color: '#FF8D00',
            label: '-∞',
        },
        {
            background_color: '#EAEAEA',
            label: '-',
        },
        {
            background_color: '#A5A5A5',
            label: '0',
        },
        {
            background_color: '#262626',
            label: '+',
        },
        {
            background_color: '#003ED4',
            label: '+∞',
        },
    ];
    /**
     * Helper method for fitting the graph in the svg view.
     *
     * @param svg The main svg.
     * @param zoomG The svg group used for panning and zooming.
     * @param d3zoom The zoom behavior.
     * @param callback Called when the fitting is done.
     */
    function fit$2(svg, zoomG, d3zoom, callback) {
        let svgRect = svg.getBoundingClientRect();
        let sceneSize = null;
        try {
            sceneSize = zoomG.getBBox();
            if (sceneSize.width === 0) {
                // There is no scene anymore. We have been detached from the dom.
                return;
            }
        }
        catch (e) {
            // Firefox produced NS_ERROR_FAILURE if we have been
            // detached from the dom.
            return;
        }
        let scale = 0.9 *
            Math.min(svgRect.width / sceneSize.width, svgRect.height / sceneSize.height, 2);
        let params = PARAMS$1.graph;
        const transform = identity$9
            .scale(scale)
            .translate(params.padding.paddingLeft, params.padding.paddingTop);
        select(svg)
            .transition()
            .duration(500)
            .call(d3zoom.transform, transform)
            .on('end.fitted', () => {
            // Remove the listener for the zoomend event,
            // so we don't get called at the end of regular zoom events,
            // just those that fit the graph to screen.
            d3zoom.on('end.fitted', null);
            callback();
        });
    }
    /**
     * Helper method for panning the graph to center on the provided node,
     * if the node is currently off-screen.
     *
     * @param nodeName The node to center the graph on
     * @param svg The root SVG element for the graph
     * @param zoomG The svg group used for panning and zooming.
     * @param d3zoom The zoom behavior.
     * @return True if the graph had to be panned to display the
     *            provided node.
     */
    function panToNode(nodeName, svg, zoomG, d3zoom) {
        const node = (select(svg).select(`[data-name="${nodeName}"]`).node());
        if (!node) {
            console.warn(`panToNode() failed for node name "${nodeName}"`);
            return false;
        }
        // Check if the selected node is off-screen in either
        // X or Y dimension in either direction.
        let nodeBox = node.getBBox();
        let nodeCtm = node.getScreenCTM();
        let pointTL = svg.createSVGPoint();
        let pointBR = svg.createSVGPoint();
        pointTL.x = nodeBox.x;
        pointTL.y = nodeBox.y;
        pointBR.x = nodeBox.x + nodeBox.width;
        pointBR.y = nodeBox.y + nodeBox.height;
        pointTL = pointTL.matrixTransform(nodeCtm);
        pointBR = pointBR.matrixTransform(nodeCtm);
        let isOutsideOfBounds = (start, end, lowerBound, upperBound) => {
            // Return if even a part of the interval is out of bounds.
            return !(start > lowerBound && end < upperBound);
        };
        let svgRect = svg.getBoundingClientRect();
        // Subtract to make sure that the node is not hidden behind the minimap.
        const horizontalBound = svgRect.left + svgRect.width - MINIMAP_BOX_WIDTH;
        const verticalBound = svgRect.top + svgRect.height - MINIMAP_BOX_HEIGHT;
        if (isOutsideOfBounds(pointTL.x, pointBR.x, svgRect.left, horizontalBound) ||
            isOutsideOfBounds(pointTL.y, pointBR.y, svgRect.top, verticalBound)) {
            // Determine the amount to translate the graph in both X and Y dimensions in
            // order to center the selected node. This takes into account the position
            // of the node, the size of the svg scene, the amount the scene has been
            // scaled by through zooming, and any previous transforms already performed
            // by this logic.
            let centerX = (pointTL.x + pointBR.x) / 2;
            let centerY = (pointTL.y + pointBR.y) / 2;
            let dx = svgRect.left + svgRect.width / 2 - centerX;
            let dy = svgRect.top + svgRect.height / 2 - centerY;
            // We translate by this amount. We divide the X and Y translations by the
            // scale to undo how translateBy scales the translations (in d3 v4).
            const svgTransform = transform$1(svg);
            select(svg)
                .transition()
                .duration(500)
                .call(d3zoom.translateBy, dx / svgTransform.k, dy / svgTransform.k);
            return true;
        }
        return false;
    }
    /**
     * Given a scene's svg group, set  g.in-extract, g.coreGraph, g.out-extract svg
     * groups' position relative to the scene.
     *
     * @param sceneGroup
     * @param renderNode render node of a metanode or series node.
     */
    function position$1(sceneGroup, renderNode) {
        // Translate scenes down by the label height so that when showing graphs in
        // expanded metanodes, the graphs are below the labels.  Do not shift them
        // down for series nodes as series nodes don't have labels inside of their
        // bounding boxes.
        let yTranslate = renderNode.node.type === NodeType.SERIES
            ? 0
            : PARAMS$1.subscene.meta.labelHeight;
        // core
        translate$1(selectChild$1(sceneGroup, 'g', Class$2.Scene.CORE), 0, yTranslate);
        // in-extract
        let hasInExtract = renderNode.isolatedInExtract.length > 0;
        let hasOutExtract = renderNode.isolatedOutExtract.length > 0;
        let hasLibraryFunctions = renderNode.libraryFunctionsExtract.length > 0;
        let offset = PARAMS$1.subscene.meta.extractXOffset;
        let auxWidth = 0;
        if (hasInExtract) {
            auxWidth += renderNode.outExtractBox.width;
        }
        if (hasOutExtract) {
            auxWidth += renderNode.outExtractBox.width;
        }
        if (hasInExtract) {
            let inExtractX = renderNode.coreBox.width;
            if (auxWidth < MIN_AUX_WIDTH) {
                inExtractX =
                    inExtractX - MIN_AUX_WIDTH + renderNode.inExtractBox.width / 2;
            }
            else {
                inExtractX =
                    inExtractX -
                        renderNode.inExtractBox.width / 2 -
                        renderNode.outExtractBox.width -
                        (hasOutExtract ? offset : 0);
            }
            inExtractX =
                inExtractX -
                    renderNode.libraryFunctionsBox.width -
                    (hasLibraryFunctions ? offset : 0);
            translate$1(selectChild$1(sceneGroup, 'g', Class$2.Scene.INEXTRACT), inExtractX, yTranslate);
        }
        // out-extract
        if (hasOutExtract) {
            let outExtractX = renderNode.coreBox.width;
            if (auxWidth < MIN_AUX_WIDTH) {
                outExtractX =
                    outExtractX - MIN_AUX_WIDTH + renderNode.outExtractBox.width / 2;
            }
            else {
                outExtractX -= renderNode.outExtractBox.width / 2;
            }
            outExtractX =
                outExtractX -
                    renderNode.libraryFunctionsBox.width -
                    (hasLibraryFunctions ? offset : 0);
            translate$1(selectChild$1(sceneGroup, 'g', Class$2.Scene.OUTEXTRACT), outExtractX, yTranslate);
        }
        if (hasLibraryFunctions) {
            let libraryFunctionsExtractX = renderNode.coreBox.width - renderNode.libraryFunctionsBox.width / 2;
            translate$1(selectChild$1(sceneGroup, 'g', Class$2.Scene.FUNCTION_LIBRARY), libraryFunctionsExtractX, yTranslate);
        }
    }
    /** Adds a click listener to a group that fires a graph-select event */
    function addGraphClickListener(graphGroup, sceneElement) {
        select(graphGroup).on('click', () => {
            sceneElement.fire('graph-select');
        });
    }
    /** Helper for adding transform: translate(x0, y0) */
    function translate$1(selection, x0, y0) {
        // If it is already placed on the screen, make it a transition.
        if (selection.attr('transform') != null) {
            selection = selection.transition('position');
        }
        selection.attr('transform', 'translate(' + x0 + ',' + y0 + ')');
    }
    /**
     * Helper for setting position of a svg rect
     * @param rect A d3 selection of rect(s) to set position of.
     * @param cx Center x.
     * @param cy Center x.
     * @param width Width to set.
     * @param height Height to set.
     */
    function positionRect(rect, cx, cy, width, height) {
        rect
            .transition()
            .attr('x', cx - width / 2)
            .attr('y', cy - height / 2)
            .attr('width', width)
            .attr('height', height);
    }
    /**
     * Positions a triangle and sizes it.
     * @param polygon polygon to set position of.
     * @param cx Center x.
     * @param cy Center y.
     * @param width Width of bounding box for triangle.
     * @param height Height of bounding box for triangle.
     */
    function positionTriangle(polygon, cx, cy, width, height) {
        const halfHeight = height / 2;
        const halfWidth = width / 2;
        const points = [
            [cx, cy - halfHeight],
            [cx + halfWidth, cy + halfHeight],
            [cx - halfWidth, cy + halfHeight],
        ];
        polygon
            .transition()
            .attr('points', points.map((point) => point.join(',')).join(' '));
    }
    /**
     * Helper for setting position of a svg expand/collapse button
     * @param button container group
     * @param renderNode the render node of the group node to position
     *        the button on.
     */
    function positionButton(button, renderNode) {
        let cx = computeCXPositionOfNodeShape(renderNode);
        // Position the button in the top-right corner of the group node,
        // with space given the draw the button inside of the corner.
        let width = renderNode.expanded ? renderNode.width : renderNode.coreBox.width;
        let height = renderNode.expanded
            ? renderNode.height
            : renderNode.coreBox.height;
        let x = cx + width / 2 - 6;
        let y = renderNode.y - height / 2 + 6;
        // For unexpanded series nodes, the button has special placement due
        // to the unique visuals of this group node.
        if (renderNode.node.type === NodeType.SERIES && !renderNode.expanded) {
            x += 10;
            y -= 2;
        }
        let translateStr = 'translate(' + x + ',' + y + ')';
        button.selectAll('path').transition().attr('transform', translateStr);
        button
            .select('circle')
            .transition()
            .attr({ cx: x, cy: y, r: PARAMS$1.nodeSize.meta.expandButtonRadius });
    }
    /**
     * Helper for setting position of a svg ellipse
     * @param ellipse ellipse to set position of.
     * @param cx Center x.
     * @param cy Center x.
     * @param width Width to set.
     * @param height Height to set.
     */
    function positionEllipse(ellipse, cx, cy, width, height) {
        ellipse
            .transition()
            .attr('cx', cx)
            .attr('cy', cy)
            .attr('rx', width / 2)
            .attr('ry', height / 2);
    }
    /**
     * @param {number} stat A stat for a health pill (such as mean or variance).
     * @param {boolean} shouldRoundOnesDigit Whether to round this number to the
     *     ones digit. Useful for say int, uint, and bool output types.
     * @return {string} A human-friendly string representation of that stat.
     */
    function humanizeHealthPillStat(stat, shouldRoundOnesDigit) {
        if (shouldRoundOnesDigit) {
            return stat.toFixed(0);
        }
        if (Math.abs(stat) >= 1) {
            return stat.toFixed(1);
        }
        return stat.toExponential(1);
    }
    /**
     * Get text content describing a health pill.
     */
    function _getHealthPillTextContent(healthPill, totalCount, elementsBreakdown, numericStats) {
        let text = 'Device: ' + healthPill.device_name + '\n';
        text += 'dtype: ' + healthPill.dtype + '\n';
        let shapeStr = '(scalar)';
        if (healthPill.shape.length > 0) {
            shapeStr = '(' + healthPill.shape.join(',') + ')';
        }
        text += '\nshape: ' + shapeStr + '\n\n';
        text += '#(elements): ' + totalCount + '\n';
        const breakdownItems = [];
        for (let i = 0; i < elementsBreakdown.length; i++) {
            if (elementsBreakdown[i] > 0) {
                breakdownItems.push('#(' + healthPillEntries[i].label + '): ' + elementsBreakdown[i]);
            }
        }
        text += breakdownItems.join(', ') + '\n\n';
        // In some cases (e.g., size-0 tensors; all elements are nan or inf) the
        // min/max and mean/stddev stats are meaningless.
        if (numericStats.max >= numericStats.min) {
            text += 'min: ' + numericStats.min + ', max: ' + numericStats.max + '\n';
            text += 'mean: ' + numericStats.mean + ', stddev: ' + numericStats.stddev;
        }
        return text;
    }
    /**
     * Renders a health pill for an op atop a node.
     * nodeGroupElement: The SVG element in which to render.
     * healthPill: A list of backend.HealthPill objects.
     * nodeInfo: Info on the associated node.
     * healthPillId: A unique numeric ID assigned to this health pill.
     * healthPillWidth: Optional width of the health pill.
     * healthPillHeight: Optional height of the health pill.
     * healthPillYOffset: Optional y-offset of the health pill (that is, the
     *   color-coded region).
     * textOffset: Optional value for the x-offset of the top text label
     *   relative to the left edge of the health pill. If not provided, will
     *   default to `healthPillWidth / 2`.
     */
    function addHealthPill(nodeGroupElement, healthPill, nodeInfo, healthPillId, healthPillWidth = 60, healthPillHeight = 10, healthPillYOffset = 0, textXOffset) {
        // Check if text already exists at location.
        select(nodeGroupElement.parentNode)
            .selectAll('.health-pill')
            .remove();
        if (!healthPill) {
            return;
        }
        const lastHealthPillData = healthPill.value;
        // For now, we only visualize the 6 values that summarize counts of tensor
        // elements of various categories: -Inf, negative, 0, positive, Inf, and NaN.
        const lastHealthPillElementsBreakdown = lastHealthPillData.slice(2, 8);
        const nanCount = lastHealthPillElementsBreakdown[0];
        const negInfCount = lastHealthPillElementsBreakdown[1];
        const posInfCount = lastHealthPillElementsBreakdown[5];
        let totalCount = lastHealthPillData[1];
        const numericStats = {
            min: lastHealthPillData[8],
            max: lastHealthPillData[9],
            mean: lastHealthPillData[10],
            stddev: Math.sqrt(lastHealthPillData[11]),
        };
        if (healthPillWidth == null) {
            healthPillWidth = 60;
        }
        if (healthPillHeight == null) {
            healthPillHeight = 10;
        }
        if (healthPillYOffset == null) {
            healthPillYOffset = 0;
        }
        if (nodeInfo != null && nodeInfo.node.type === NodeType.OP) {
            // Use a smaller health pill for op nodes (rendered as smaller ellipses).
            healthPillWidth /= 2;
            healthPillHeight /= 2;
        }
        let healthPillGroup = document.createElementNS(SVG_NAMESPACE, 'g');
        healthPillGroup.classList.add('health-pill');
        // Define the gradient for the health pill.
        let healthPillDefs = document.createElementNS(SVG_NAMESPACE, 'defs');
        healthPillGroup.appendChild(healthPillDefs);
        let healthPillGradient = document.createElementNS(SVG_NAMESPACE, 'linearGradient');
        // Every element in a web page must have a unique ID.
        const healthPillGradientId = 'health-pill-gradient-' + healthPillId;
        healthPillGradient.setAttribute('id', healthPillGradientId);
        let cumulativeCount = 0;
        let previousOffset = '0%';
        for (let i = 0; i < lastHealthPillElementsBreakdown.length; i++) {
            if (!lastHealthPillElementsBreakdown[i]) {
                // Exclude empty categories.
                continue;
            }
            cumulativeCount += lastHealthPillElementsBreakdown[i];
            // Create a color interval using 2 stop elements.
            let stopElement0 = document.createElementNS(SVG_NAMESPACE, 'stop');
            stopElement0.setAttribute('offset', previousOffset);
            stopElement0.setAttribute('stop-color', healthPillEntries[i].background_color);
            healthPillGradient.appendChild(stopElement0);
            let stopElement1 = document.createElementNS(SVG_NAMESPACE, 'stop');
            let percent = (cumulativeCount * 100) / totalCount + '%';
            stopElement1.setAttribute('offset', percent);
            stopElement1.setAttribute('stop-color', healthPillEntries[i].background_color);
            healthPillGradient.appendChild(stopElement1);
            previousOffset = percent;
        }
        healthPillDefs.appendChild(healthPillGradient);
        // Create the rectangle for the health pill.
        let rect = document.createElementNS(SVG_NAMESPACE, 'rect');
        rect.setAttribute('fill', 'url(#' + healthPillGradientId + ')');
        rect.setAttribute('width', String(healthPillWidth));
        rect.setAttribute('height', String(healthPillHeight));
        rect.setAttribute('y', String(healthPillYOffset));
        healthPillGroup.appendChild(rect);
        // Show a title with specific counts on hover.
        let titleSvg = document.createElementNS(SVG_NAMESPACE, 'title');
        titleSvg.textContent = _getHealthPillTextContent(healthPill, totalCount, lastHealthPillElementsBreakdown, numericStats);
        healthPillGroup.appendChild(titleSvg);
        // Center this health pill just right above the node for the op.
        let shouldRoundOnesDigit = false;
        if (nodeInfo != null) {
            let healthPillX = nodeInfo.x - healthPillWidth / 2;
            let healthPillY = nodeInfo.y - healthPillHeight - nodeInfo.height / 2 - 2;
            if (nodeInfo.labelOffset < 0) {
                // The label is positioned above the node. Do not occlude the label.
                healthPillY += nodeInfo.labelOffset;
            }
            healthPillGroup.setAttribute('transform', 'translate(' + healthPillX + ', ' + healthPillY + ')');
            if (lastHealthPillElementsBreakdown[2] ||
                lastHealthPillElementsBreakdown[3] ||
                lastHealthPillElementsBreakdown[4]) {
                // At least 1 "non-Inf and non-NaN" value exists (a -, 0, or + value). Show
                // stats on tensor values.
                // Determine if we should display the output range as integers.
                let node = nodeInfo.node;
                let attributes = node.attr;
                if (attributes && attributes.length) {
                    // Find the attribute for output type if there is one.
                    for (let i = 0; i < attributes.length; i++) {
                        if (attributes[i].key === 'T') {
                            // Note whether the output type is an integer.
                            let outputType = attributes[i].value['type'];
                            shouldRoundOnesDigit =
                                outputType && /^DT_(BOOL|INT|UINT)/.test(outputType);
                            break;
                        }
                    }
                }
            }
        }
        let statsSvg = document.createElementNS(SVG_NAMESPACE, 'text');
        if (Number.isFinite(numericStats.min) && Number.isFinite(numericStats.max)) {
            const minString = humanizeHealthPillStat(numericStats.min, shouldRoundOnesDigit);
            const maxString = humanizeHealthPillStat(numericStats.max, shouldRoundOnesDigit);
            if (totalCount > 1) {
                statsSvg.textContent = minString + ' ~ ' + maxString;
            }
            else {
                statsSvg.textContent = minString;
            }
            if (nanCount > 0 || negInfCount > 0 || posInfCount > 0) {
                statsSvg.textContent += ' (';
                const badValueStrings = [];
                if (nanCount > 0) {
                    badValueStrings.push(`NaN×${nanCount}`);
                }
                if (negInfCount > 0) {
                    badValueStrings.push(`-∞×${negInfCount}`);
                }
                if (posInfCount > 0) {
                    badValueStrings.push(`+∞×${posInfCount}`);
                }
                statsSvg.textContent += badValueStrings.join('; ') + ')';
            }
        }
        else {
            statsSvg.textContent = '(No finite elements)';
        }
        statsSvg.classList.add('health-pill-stats');
        if (textXOffset == null) {
            textXOffset = healthPillWidth / 2;
        }
        statsSvg.setAttribute('x', String(textXOffset));
        statsSvg.setAttribute('y', String(healthPillYOffset - 2));
        healthPillGroup.appendChild(statsSvg);
        dom(nodeGroupElement.parentNode).appendChild(healthPillGroup);
    }
    /**
     * Adds health pills (which visualize tensor summaries) to a graph group.
     * @param svgRoot The root SVG element of the graph to add heath pills to.
     * @param nodeNamesToHealthPills An object mapping node name to health pill.
     * @param colors A list of colors to use.
     */
    function addHealthPills(svgRoot, nodeNamesToHealthPills, healthPillStepIndex) {
        if (!nodeNamesToHealthPills) {
            // No health pill information available.
            return;
        }
        // We generate a unique ID for each health pill because the ID of each element
        // in a web page must be unique, and each health pill generates a gradient
        // that its code later refers to.
        let healthPillId = 1;
        let svgRootSelection = select(svgRoot);
        svgRootSelection
            .selectAll('g.nodeshape')
            .each(function (nodeInfo) {
            // Only show health pill data for this node if it is available.
            const healthPills = nodeNamesToHealthPills[nodeInfo.node.name];
            const healthPill = healthPills ? healthPills[healthPillStepIndex] : null;
            addHealthPill(this, healthPill, nodeInfo, healthPillId++);
        });
    }

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    let TfGraphDebuggerDataCard = class TfGraphDebuggerDataCard extends LegacyElementMixin(PolymerElement) {
        constructor() {
            super(...arguments);
            // Only relevant if we are in all steps mode, in which case the user may want to view health
            // pills for a specific step.
            this.specificHealthPillStep = 0;
            this.healthPillEntries = healthPillEntries;
        }
        ready() {
            super.ready();
            var mainContainer = document.getElementById('mainContainer');
            var scrollbarContainer = document.querySelector('tf-dashboard-layout .scrollbar');
            if (mainContainer && scrollbarContainer) {
                // If this component is being used inside of TensorBoard's dashboard layout, it may easily
                // cause the dashboard layout element to overflow, giving the user 2 scroll bars. Prevent
                // that by hiding whatever content overflows - the user will have to expand the viewport to
                // use this debugging card.
                mainContainer.style.overflow = 'hidden';
                scrollbarContainer.style.overflow = 'hidden';
            }
        }
        _healthPillsAvailable(debuggerDataEnabled, nodeNamesToHealthPills) {
            // So long as there is a mapping (even if empty) from node name to health pills, show the
            // legend and slider. We do that because, even if no health pills exist at the current step,
            // the user may desire to change steps, and the slider must show for the user to do that.
            return debuggerDataEnabled && nodeNamesToHealthPills;
        }
        _computeTensorCountString(healthPillValuesForSelectedNode, valueIndex) {
            if (!healthPillValuesForSelectedNode) {
                // No health pill data is available.
                return '';
            }
            return healthPillValuesForSelectedNode[valueIndex].toFixed(0);
        }
        get healthPillValuesForSelectedNode() {
            var nodeNamesToHealthPills = this.nodeNamesToHealthPills;
            var healthPillStepIndex = this.healthPillStepIndex;
            var selectedNode = this.selectedNode;
            var allStepsModeEnabled = this.allStepsModeEnabled;
            var areHealthPillsLoading = this.areHealthPillsLoading;
            if (areHealthPillsLoading) {
                // Health pills are loading. Do not render data that is out of date.
                return null;
            }
            if (!selectedNode) {
                // No node is selected.
                return null;
            }
            const healthPills = nodeNamesToHealthPills[selectedNode];
            if (!healthPills) {
                // This node lacks a health pill.
                return null;
            }
            // If all steps mode is enabled, we use the first health pill in the list because the JSON
            // response from the server is a mapping between node name and a list of 1 health pill.
            const healthPill = healthPills[allStepsModeEnabled ? 0 : healthPillStepIndex];
            if (!healthPill) {
                // This node lacks a health pill at the current step.
                return null;
            }
            // The health pill count values start at 2. Each health pill contains 6 values.
            return healthPill.value.slice(2, 8);
        }
        get _currentStepDisplayValue() {
            var nodeNamesToHealthPills = this.nodeNamesToHealthPills;
            var healthPillStepIndex = this.healthPillStepIndex;
            var allStepsModeEnabled = this.allStepsModeEnabled;
            var specificHealthPillStep = this.specificHealthPillStep;
            var areHealthPillsLoading = this.areHealthPillsLoading;
            if (allStepsModeEnabled) {
                // The user seeks health pills for specific step from the server.
                return specificHealthPillStep.toFixed(0);
            }
            if (areHealthPillsLoading) {
                // The current step is undefined.
                return 0;
            }
            for (let nodeName in nodeNamesToHealthPills) {
                // All nodes have the same number of steps stored, so only examine 1 node. We cannot
                // directly index into the nodeNamesToHealthPills object because we do not have a key.
                // If all steps mode is enabled, we only have 1 step to show.
                return nodeNamesToHealthPills[nodeName][healthPillStepIndex].step.toFixed(0);
            }
            // The current step could not be computed.
            return 0;
        }
        // The biggest step value ever seen. Used to determine what steps of health pills to let the
        // user fetch in all steps mode.
        get _biggestStepEverSeen() {
            var nodeNamesToHealthPills = this.nodeNamesToHealthPills;
            for (let nodeName in nodeNamesToHealthPills) {
                // All nodes have the same number of steps stored, so only examine 1 node.
                // The index is 1 less than the count. Tensorboard backend logic guarantees that the length
                // of the array will be greater than 1.
                var healthPills = nodeNamesToHealthPills[nodeName];
                return Math.max(this._biggestStepEverSeen, healthPills[healthPills.length - 1].step);
            }
            // No steps seen so far. Default to 0.
            return this._biggestStepEverSeen || 0;
        }
        get _maxStepIndex() {
            var nodeNamesToHealthPills = this.nodeNamesToHealthPills;
            for (let nodeName in nodeNamesToHealthPills) {
                // All nodes have the same number of steps stored, so only examine 1 node.
                // The index is 1 less than the count. Tensorboard backend logic guarantees that the length
                // of the array will be greater than 1.
                return nodeNamesToHealthPills[nodeName].length - 1;
            }
            // Return a falsy value. The slider should be hidden.
            return 0;
        }
        _hasDebuggerNumericAlerts(debuggerNumericAlerts) {
            return debuggerNumericAlerts && debuggerNumericAlerts.length;
        }
        _updateAlertsList() {
            var debuggerNumericAlerts = this.debuggerNumericAlerts;
            var alertBody = this.$$('#numeric-alerts-body');
            if (!alertBody) {
                return;
            }
            alertBody.innerText = '';
            for (var i = 0; i < debuggerNumericAlerts.length; i++) {
                var alert = debuggerNumericAlerts[i];
                var tableRow = document.createElement('tr');
                var timestampTd = document.createElement('td');
                timestampTd.innerText = computeHumanFriendlyTime(alert.first_timestamp);
                timestampTd.classList.add('first-offense-td');
                tableRow.appendChild(timestampTd);
                var tensorDeviceTd = document.createElement('td');
                tensorDeviceTd.classList.add('tensor-device-td');
                var tensorSection = document.createElement('div');
                tensorSection.classList.add('tensor-section-within-table');
                tensorSection.innerText = alert.tensor_name;
                this._addOpExpansionListener(tensorSection, alert.tensor_name);
                tensorDeviceTd.appendChild(tensorSection);
                var deviceSection = document.createElement('div');
                deviceSection.classList.add('device-section-within-table');
                deviceSection.innerText = '(' + alert.device_name + ')';
                tensorDeviceTd.appendChild(deviceSection);
                tableRow.appendChild(tensorDeviceTd);
                var miniHealthPill = document.createElement('div');
                miniHealthPill.classList.add('mini-health-pill');
                var miniHealthPillTd = document.createElement('td');
                miniHealthPillTd.classList.add('mini-health-pill-td');
                miniHealthPillTd.appendChild(miniHealthPill);
                tableRow.appendChild(miniHealthPillTd);
                if (alert.neg_inf_event_count) {
                    var negativeInfCountSection = document.createElement('div');
                    negativeInfCountSection.classList.add('negative-inf-mini-health-pill-section');
                    negativeInfCountSection.innerText = alert.neg_inf_event_count;
                    negativeInfCountSection.setAttribute('title', alert.neg_inf_event_count + ' events with -\u221E');
                    miniHealthPill.appendChild(negativeInfCountSection);
                }
                if (alert.pos_inf_event_count) {
                    var positiveInfCountSection = document.createElement('div');
                    positiveInfCountSection.classList.add('positive-inf-mini-health-pill-section');
                    positiveInfCountSection.innerText = alert.pos_inf_event_count;
                    positiveInfCountSection.setAttribute('title', alert.pos_inf_event_count + ' events with +\u221E');
                    miniHealthPill.appendChild(positiveInfCountSection);
                }
                if (alert.nan_event_count) {
                    var nanCountSection = document.createElement('div');
                    nanCountSection.classList.add('nan-mini-health-pill-section');
                    nanCountSection.innerText = alert.nan_event_count;
                    nanCountSection.setAttribute('title', alert.nan_event_count + ' events with NaN');
                    miniHealthPill.appendChild(nanCountSection);
                }
                dom(alertBody).appendChild(tableRow);
            }
        }
        // Adds a listener to an element, so that when that element is clicked, the tensor with
        // tensorName expands.
        _addOpExpansionListener(clickableElement, tensorName) {
            clickableElement.addEventListener('click', () => {
                // When the user clicks on a tensor name, expand all nodes until the user can see the
                // associated node.
                var nameOfNodeToSelect = expandUntilNodeIsShown(document.getElementById('scene'), this.renderHierarchy, tensorName);
                // Store the current scroll of the graph info card. Node selection alters that scroll, and
                // we restore the scroll later.
                var previousScrollFromBottom;
                var graphInfoCard = document.querySelector('tf-graph-info#graph-info');
                if (graphInfoCard) {
                    previousScrollFromBottom =
                        graphInfoCard.scrollHeight - graphInfoCard.scrollTop;
                }
                // Update the selected node within graph logic.
                var previousSelectedNode = this.selectedNode;
                this.set('selectedNode', nameOfNodeToSelect);
                // Scroll the graph info card back down if necessary so that user can see the alerts section
                // again. Selecting the node causes the info card to scroll to the top, which may mean the
                // user no longer sees the list of alerts.
                var scrollToOriginalLocation = () => {
                    graphInfoCard.scrollTop =
                        graphInfoCard.scrollHeight - previousScrollFromBottom;
                };
                if (graphInfoCard) {
                    // This component is used within an info card. Restore the original scroll.
                    if (previousSelectedNode) {
                        // The card for the selected node has already opened. Immediately restore the scroll.
                        scrollToOriginalLocation();
                    }
                    else {
                        // Give some time for the DOM of the info card to be created before scrolling down.
                        window.setTimeout(scrollToOriginalLocation, 20);
                    }
                }
            });
        }
    };
    TfGraphDebuggerDataCard.template = html `
    <style>
      :host {
        font-size: 12px;
        margin: 0;
        padding: 0;
        display: block;
      }

      h2 {
        padding: 0;
        text-align: center;
        margin: 0;
      }

      .health-pill-legend {
        padding: 15px;
      }

      .health-pill-legend h2 {
        text-align: left;
      }

      .health-pill-entry {
        margin: 10px 10px 10px 0;
      }

      .health-pill-entry .color-preview {
        width: 26px;
        height: 26px;
        border-radius: 3px;
        display: inline-block;
        margin: 0 10px 0 0;
      }

      .health-pill-entry .color-label,
      .health-pill-entry .tensor-count {
        color: #777;
        display: inline-block;
        height: 26px;
        font-size: 22px;
        line-height: 26px;
        vertical-align: top;
      }

      .health-pill-entry .tensor-count {
        float: right;
      }

      #health-pill-step-slider {
        width: 100%;
        margin: 0 0 0 -15px;
        /* 31 comes from adding a padding of 15px from both sides of the paper-slider, subtracting
   * 1px so that the slider width aligns with the image (the last slider marker takes up 1px),
   * and adding 2px to account for a border of 1px on both sides of the image. 30 - 1 + 2.
   * Apparently, the paper-slider lacks a mixin for those padding values. */
        width: calc(100% + 31px);
      }

      #health-pills-loading-spinner {
        width: 20px;
        height: 20px;
        vertical-align: top;
      }

      #health-pill-step-number-input {
        text-align: center;
        vertical-align: top;
      }

      #numeric-alerts-table-container {
        max-height: 400px;
        overflow-x: hidden;
        overflow-y: auto;
      }

      #numeric-alerts-table {
        text-align: left;
      }

      #numeric-alerts-table td {
        vertical-align: top;
      }

      #numeric-alerts-table .first-offense-td {
        display: inline-block;
      }

      .first-offense-td {
        width: 80px;
      }

      .tensor-device-td {
        max-width: 140px;
        word-wrap: break-word;
      }

      .tensor-section-within-table {
        color: #266236;
        cursor: pointer;
        opacity: 0.8;
        text-decoration: underline;
      }

      .tensor-section-within-table:hover {
        opacity: 1;
      }

      .device-section-within-table {
        color: #666;
      }

      .mini-health-pill {
        width: 130px;
      }

      .mini-health-pill > div {
        height: 100%;
        width: 60px;
        border-radius: 3px;
      }

      #event-counts-th {
        padding: 0 0 0 10px;
      }

      .negative-inf-mini-health-pill-section {
        background: rgb(255, 141, 0);
        width: 20px;
      }

      .positive-inf-mini-health-pill-section {
        background: rgb(0, 62, 212);
        width: 20px;
      }

      .nan-mini-health-pill-section {
        background: rgb(204, 47, 44);
        width: 20px;
      }

      .negative-inf-mini-health-pill-section,
      .positive-inf-mini-health-pill-section,
      .nan-mini-health-pill-section {
        color: #fff;
        display: inline-block;
        height: 100%;
        line-height: 20px;
        margin: 0 0 0 10px;
        text-align: center;
      }

      .no-numeric-alerts-notification {
        margin: 0;
      }
    </style>
    <paper-material elevation="1" class="card health-pill-legend">
      <div class="title">
        Enable all (not just sampled) steps. Requires slow disk read.
      </div>
      <paper-toggle-button
        id="enableAllStepsModeToggle"
        checked="{{allStepsModeEnabled}}"
      >
      </paper-toggle-button>
      <h2>
        Step of Health Pills:
        <template is="dom-if" if="[[allStepsModeEnabled]]">
          <input
            type="number"
            id="health-pill-step-number-input"
            min="0"
            max="[[_biggestStepEverSeen]]"
            value="{{specificHealthPillStep::input}}"
          />
        </template>
        <template is="dom-if" if="[[!allStepsModeEnabled]]">
          [[_currentStepDisplayValue]]
        </template>
        <paper-spinner-lite
          active
          hidden$="[[!areHealthPillsLoading]]"
          id="health-pills-loading-spinner"
        ></paper-spinner-lite>
      </h2>
      <template is="dom-if" if="[[allStepsModeEnabled]]">
        <paper-slider
          id="health-pill-step-slider"
          immediate-value="{{specificHealthPillStep}}"
          max="[[_biggestStepEverSeen]]"
          snaps
          step="1"
          value="{{specificHealthPillStep}}"
        ></paper-slider>
      </template>
      <template is="dom-if" if="[[!allStepsModeEnabled]]">
        <template is="dom-if" if="[[_maxStepIndex]]">
          <paper-slider
            id="health-pill-step-slider"
            immediate-value="{{healthPillStepIndex}}"
            max="[[_maxStepIndex]]"
            snaps
            step="1"
            value="{{healthPillStepIndex}}"
          ></paper-slider>
        </template>
      </template>
      <h2>
        Health Pill
        <template is="dom-if" if="[[healthPillValuesForSelectedNode]]">
          Counts for Selected Node
        </template>
        <template is="dom-if" if="[[!healthPillValuesForSelectedNode]]">
          Legend
        </template>
      </h2>
      <template is="dom-repeat" items="[[healthPillEntries]]">
        <div class="health-pill-entry">
          <div
            class="color-preview"
            style="background:[[item.background_color]]"
          ></div>
          <div class="color-label">[[item.label]]</div>
          <div class="tensor-count">
            [[_computeTensorCountString(healthPillValuesForSelectedNode,
            index)]]
          </div>
        </div>
      </template>
      <div hidden$="[[!_hasDebuggerNumericAlerts(debuggerNumericAlerts)]]">
        <h2 id="numeric-alerts-header">Numeric Alerts</h2>
        <p>Alerts are sorted from top to bottom by increasing timestamp.</p>
        <div id="numeric-alerts-table-container">
          <table id="numeric-alerts-table">
            <thead>
              <tr>
                <th>First Offense</th>
                <th>Tensor (Device)</th>
                <th id="event-counts-th">Event Counts</th>
              </tr>
            </thead>
            <tbody id="numeric-alerts-body"></tbody>
          </table>
        </div>
      </div>
      <template
        is="dom-if"
        if="[[!_hasDebuggerNumericAlerts(debuggerNumericAlerts)]]"
      >
        <p class="no-numeric-alerts-notification">
          No numeric alerts so far. That is likely good. Alerts indicate the
          presence of NaN or (+/-) Infinity values, which may be concerning.
        </p>
      </template>
    </paper-material>
  `;
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraphDebuggerDataCard.prototype, "renderHierarchy", void 0);
    __decorate([
        property({
            type: Array,
            notify: true,
        }),
        __metadata("design:type", Object)
    ], TfGraphDebuggerDataCard.prototype, "debuggerNumericAlerts", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraphDebuggerDataCard.prototype, "nodeNamesToHealthPills", void 0);
    __decorate([
        property({
            type: Number,
            notify: true,
        }),
        __metadata("design:type", Object)
    ], TfGraphDebuggerDataCard.prototype, "healthPillStepIndex", void 0);
    __decorate([
        property({
            type: Number,
            notify: true,
        }),
        __metadata("design:type", Number)
    ], TfGraphDebuggerDataCard.prototype, "specificHealthPillStep", void 0);
    __decorate([
        property({
            type: String,
            notify: true,
        }),
        __metadata("design:type", Object)
    ], TfGraphDebuggerDataCard.prototype, "selectedNode", void 0);
    __decorate([
        property({
            type: String,
            notify: true,
        }),
        __metadata("design:type", Object)
    ], TfGraphDebuggerDataCard.prototype, "highlightedNode", void 0);
    __decorate([
        property({
            type: Number,
            notify: true,
        }),
        __metadata("design:type", Object)
    ], TfGraphDebuggerDataCard.prototype, "selectedNodeInclude", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Object)
    ], TfGraphDebuggerDataCard.prototype, "areHealthPillsLoading", void 0);
    __decorate([
        property({
            type: Array,
        }),
        __metadata("design:type", Array)
    ], TfGraphDebuggerDataCard.prototype, "healthPillEntries", void 0);
    __decorate([
        property({
            type: Boolean,
            notify: true,
        }),
        __metadata("design:type", Object)
    ], TfGraphDebuggerDataCard.prototype, "allStepsModeEnabled", void 0);
    __decorate([
        computed('nodeNamesToHealthPills', 'healthPillStepIndex', 'selectedNode', 'allStepsModeEnabled', 'areHealthPillsLoading'),
        __metadata("design:type", Array),
        __metadata("design:paramtypes", [])
    ], TfGraphDebuggerDataCard.prototype, "healthPillValuesForSelectedNode", null);
    __decorate([
        computed('nodeNamesToHealthPills', 'healthPillStepIndex', 'allStepsModeEnabled', 'specificHealthPillStep', 'areHealthPillsLoading'),
        __metadata("design:type", Object),
        __metadata("design:paramtypes", [])
    ], TfGraphDebuggerDataCard.prototype, "_currentStepDisplayValue", null);
    __decorate([
        computed('nodeNamesToHealthPills'),
        __metadata("design:type", Number),
        __metadata("design:paramtypes", [])
    ], TfGraphDebuggerDataCard.prototype, "_biggestStepEverSeen", null);
    __decorate([
        computed('nodeNamesToHealthPills'),
        __metadata("design:type", Number),
        __metadata("design:paramtypes", [])
    ], TfGraphDebuggerDataCard.prototype, "_maxStepIndex", null);
    __decorate([
        observe('debuggerNumericAlerts'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfGraphDebuggerDataCard.prototype, "_updateAlertsList", null);
    TfGraphDebuggerDataCard = __decorate([
        customElement('tf-graph-debugger-data-card')
    ], TfGraphDebuggerDataCard);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    var GraphIconType;
    (function (GraphIconType) {
        GraphIconType["CONST"] = "CONST";
        GraphIconType["META"] = "META";
        GraphIconType["OP"] = "OP";
        GraphIconType["SERIES"] = "SERIES";
        GraphIconType["SUMMARY"] = "SUMMARY";
    })(GraphIconType || (GraphIconType = {}));
    let TfGraphIcon = class TfGraphIcon extends LegacyElementMixin(PolymerElement) {
        constructor() {
            super(...arguments);
            this.vertical = false;
            this.fillOverride = null;
            this.strokeOverride = null;
            this.height = 20;
            this.faded = false;
        }
        getSvgDefinableElement() {
            return this.$.svgDefs;
        }
        get _fill() {
            var type = this.type;
            var fillOverride = this.fillOverride;
            if (fillOverride != null)
                return fillOverride;
            switch (type) {
                case GraphIconType.META:
                    return MetanodeColors.DEFAULT_FILL;
                case GraphIconType.SERIES:
                    return SeriesNodeColors.DEFAULT_FILL;
                default:
                    return OpNodeColors.DEFAULT_FILL;
            }
        }
        get _stroke() {
            var type = this.type;
            var strokeOverride = this.strokeOverride;
            if (strokeOverride != null)
                return strokeOverride;
            switch (type) {
                case GraphIconType.META:
                    return MetanodeColors.DEFAULT_STROKE;
                case GraphIconType.SERIES:
                    return SeriesNodeColors.DEFAULT_STROKE;
                default:
                    return OpNodeColors.DEFAULT_STROKE;
            }
        }
        /**
         * Test whether the specified node's type, or the literal type string,
         * match a particular other type.
         */
        _isType(type, targetType) {
            return type === targetType;
        }
        _fadedClass(faded, shape) {
            return faded ? 'faded-' + shape : '';
        }
    };
    TfGraphIcon.template = html `
    <style>
      :host {
        font-size: 0;
      }

      .faded-rect {
        fill: url(#rectHatch);
      }

      .faded-ellipse {
        fill: url(#ellipseHatch);
      }

      .faded-rect,
      .faded-ellipse,
      .faded-series {
        stroke: var(--tb-graph-faded) !important;
      }
      #rectHatch line,
      #ellipseHatch line {
        color: #e0d4b3 !important;
        fill: white;
        stroke: #e0d4b3 !important;
      }
    </style>
    <!-- SVG for definitions -->
    <svg height="0" width="0" id="svgDefs">
      <defs>
        <!-- Hatch patterns for faded out nodes. -->
        <pattern
          id="rectHatch"
          patternTransform="rotate(45 0 0)"
          width="5"
          height="5"
          patternUnits="userSpaceOnUse"
        >
          <line x1="0" y1="0" x2="0" y2="5" style="stroke-width: 1"></line>
        </pattern>
        <pattern
          id="ellipseHatch"
          patternTransform="rotate(45 0 0)"
          width="2"
          height="2"
          patternUnits="userSpaceOnUse"
        >
          <line x1="0" y1="0" x2="0" y2="2" style="stroke-width: 1"></line>
        </pattern>
        <!-- Template for an Op node ellipse. -->
        <ellipse
          id="op-node-stamp"
          rx="7.5"
          ry="3"
          stroke="inherit"
          fill="inherit"
        ></ellipse>
        <!-- Template for an Op node annotation ellipse (smaller). -->
        <ellipse
          id="op-node-annotation-stamp"
          rx="5"
          ry="2"
          stroke="inherit"
          fill="inherit"
        ></ellipse>
        <!-- Vertically stacked series of Op nodes when unexpanded. -->
        <g id="op-series-vertical-stamp">
          <use xlink:href="#op-node-stamp" x="8" y="9"></use>
          <use xlink:href="#op-node-stamp" x="8" y="6"></use>
          <use xlink:href="#op-node-stamp" x="8" y="3"></use>
        </g>
        <g id="op-series-horizontal-stamp">
          <use xlink:href="#op-node-stamp" x="16" y="4"></use>
          <use xlink:href="#op-node-stamp" x="12" y="4"></use>
          <use xlink:href="#op-node-stamp" x="8" y="4"></use>
        </g>
        <g
          id="summary-icon"
          fill="#848484"
          height="12"
          viewBox="0 0 24 24"
          width="12"
        >
          <path
            d="M19 3H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zM9 17H7v-7h2v7zm4 0h-2V7h2v10zm4 0h-2v-4h2v4z"
          ></path>
        </g>
      </defs>
    </svg>
    <template is="dom-if" if="[[_isType(type, 'CONST')]]">
      <svg
        height$="[[height]]"
        preserveAspectRatio="xMinYMid meet"
        viewBox="0 0 10 10"
      >
        <circle
          cx="5"
          cy="5"
          r="3"
          fill$="[[_fill]]"
          stroke$="[[_stroke]]"
        ></circle>
      </svg>
    </template>
    <template is="dom-if" if="[[_isType(type, 'SUMMARY')]]">
      <svg
        width$="[[height]]"
        height$="[[height]]"
        viewBox="0 0 24 24"
        fill="#848484"
      >
        <path
          d="M19 3H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zM9 17H7v-7h2v7zm4 0h-2V7h2v10zm4 0h-2v-4h2v4z"
        ></path>
      </svg>
    </template>
    <template is="dom-if" if="[[_isType(type, 'OP')]]">
      <svg
        height$="[[height]]"
        preserveAspectRatio="xMinYMid meet"
        viewBox="0 0 16 8"
      >
        <use
          xmlns:xlink="http://www.w3.org/1999/xlink"
          xlink:href="#op-node-stamp"
          fill$="[[_fill]]"
          stroke$="[[_stroke]]"
          class$="{{_fadedClass(faded, 'ellipse')}}"
          x="8"
          y="4"
        ></use>
      </svg>
    </template>
    <template is="dom-if" if="[[_isType(type, 'META')]]">
      <svg
        height$="[[height]]"
        preserveAspectRatio="xMinYMid meet"
        viewBox="0 0 37 16"
      >
        <rect
          x="1"
          y="1"
          fill$="[[_fill]]"
          stroke$="[[_stroke]]"
          class$="{{_fadedClass(faded, 'rect')}}"
          stroke-width="2px"
          height="14"
          width="35"
          rx="5"
          ry="5"
        ></rect>
      </svg>
    </template>
    <template is="dom-if" if="[[_isType(type, 'SERIES')]]">
      <template is="dom-if" if="[[vertical]]">
        <svg
          height$="[[height]]"
          preserveAspectRatio="xMinYMid meet"
          viewBox="0 0 16 15"
        >
          <use
            xmlns:xlink="http://www.w3.org/1999/xlink"
            xlink:href="#op-series-vertical-stamp"
            fill$="[[_fill]]"
            stroke$="[[_stroke]]"
            class$="{{_fadedClass(faded, 'series')}}"
            x="0"
            y="2"
          ></use>
        </svg>
      </template>
      <template is="dom-if" if="[[!vertical]]">
        <svg
          height$="[[height]]"
          preserveAspectRatio="xMinYMid meet"
          viewBox="0 0 24 10"
        >
          <use
            xmlns:xlink="http://www.w3.org/1999/xlink"
            xlink:href="#op-series-horizontal-stamp"
            fill$="[[_fill]]"
            stroke$="[[_stroke]]"
            class$="{{_fadedClass(faded, 'series')}}"
            x="0"
            y="1"
          ></use>
        </svg>
      </template>
    </template>
  `;
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfGraphIcon.prototype, "type", void 0);
    __decorate([
        property({
            type: Boolean,
        }),
        __metadata("design:type", Boolean)
    ], TfGraphIcon.prototype, "vertical", void 0);
    __decorate([
        property({
            type: String,
        }),
        __metadata("design:type", String)
    ], TfGraphIcon.prototype, "fillOverride", void 0);
    __decorate([
        property({
            type: String,
        }),
        __metadata("design:type", String)
    ], TfGraphIcon.prototype, "strokeOverride", void 0);
    __decorate([
        property({
            type: Number,
        }),
        __metadata("design:type", Number)
    ], TfGraphIcon.prototype, "height", void 0);
    __decorate([
        property({
            type: Boolean,
        }),
        __metadata("design:type", Boolean)
    ], TfGraphIcon.prototype, "faded", void 0);
    __decorate([
        computed('type', 'fillOverride'),
        __metadata("design:type", String),
        __metadata("design:paramtypes", [])
    ], TfGraphIcon.prototype, "_fill", null);
    __decorate([
        computed('type', 'strokeOverride'),
        __metadata("design:type", String),
        __metadata("design:paramtypes", [])
    ], TfGraphIcon.prototype, "_stroke", null);
    TfGraphIcon = __decorate([
        customElement('tf-graph-icon')
    ], TfGraphIcon);

    var tf_graph_icon = /*#__PURE__*/Object.freeze({
        __proto__: null,
        get GraphIconType () { return GraphIconType; }
    });

    /* Copyright 2015 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the 'License');
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an 'AS IS' BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * Returns the top and left distance of the scene element from the top left
     * corner of the screen.
     */
    function getOffset(sceneElement) {
        let leftDistance = 0;
        let topDistance = 0;
        let currentElement = sceneElement;
        while (currentElement &&
            currentElement.offsetLeft >= 0 &&
            currentElement.offsetTop >= 0) {
            leftDistance += currentElement.offsetLeft - currentElement.scrollLeft;
            topDistance += currentElement.offsetTop - currentElement.scrollTop;
            currentElement = currentElement.offsetParent;
        }
        return {
            left: leftDistance,
            top: topDistance,
        };
    }
    /**
     * Returns the event listener, which can be used as an argument for the d3
     * selection.on function. Renders the context menu that is to be displayed
     * in response to the event.
     */
    function getMenu(sceneElement, menu) {
        const menuNode = sceneElement.getContextMenu();
        const menuSelection = select(sceneElement.getContextMenu());
        // Function called to populate the context menu.
        return function (data, index) {
            // Position and display the menu.
            let event$1 = event;
            const sceneOffset = getOffset(sceneElement);
            menuSelection
                .style('display', 'block')
                .style('left', event$1.clientX - sceneOffset.left + 1 + 'px')
                .style('top', event$1.clientY - sceneOffset.top + 1 + 'px');
            // Stop the event from propagating further.
            event$1.preventDefault();
            event$1.stopPropagation();
            function maybeCloseMenu(event) {
                if (event && event.composedPath().includes(menuNode)) {
                    return;
                }
                menuSelection.style('display', 'none');
                document.body.removeEventListener('mousedown', maybeCloseMenu, {
                    capture: true,
                });
            }
            // Dismiss and remove the click listener as soon as there is a mousedown
            // on the document. We use capture listener so no component can stop
            // context menu from dismissing due to stopped propagation.
            document.body.addEventListener('mousedown', maybeCloseMenu, {
                capture: true,
            });
            // Add provided items to the context menu.
            menuSelection.html('');
            let list = menuSelection.append('ul');
            list
                .selectAll('li')
                .data(menu)
                .enter()
                .append('li')
                .on('click', (d, i) => {
                d.action(this, data, index);
                maybeCloseMenu();
            })
                .html(function (d) {
                return d.title(data);
            });
        };
    }

    /* Copyright 2015 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the 'License');
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an 'AS IS' BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /** Delimiter between dimensions when showing sizes of tensors. */
    const TENSOR_SHAPE_DELIM = '\u00D7';
    let arrowheadMap = quantize$1()
        .domain([MIN_EDGE_WIDTH, MAX_EDGE_WIDTH])
        .range(['small', 'medium', 'large', 'xlarge']);
    function getEdgeKey(edgeObj) {
        return edgeObj.v + EDGE_KEY_DELIM$1 + edgeObj.w;
    }
    /**
     * Select or Create a 'g.edges' group to a given sceneGroup
     * and builds a number of 'g.edge' groups inside the group.
     *
     * Structure Pattern:
     *
     * <g class='edges'>
     *   <g class='edge'>
     *     <path class='edgeline'/>
     *   </g>
     *   ...
     * </g>
     *
     *
     * @param sceneGroup container
     * @param graph
     * @param sceneElement <tf-graph-scene> polymer element.
     * @return selection of the created nodeGroups
     */
    function buildGroup(sceneGroup, graph, sceneElement) {
        const sceneComponent = sceneElement;
        let edges = [];
        edges = lodash.reduce(graph.edges(), (edges, edgeObj) => {
            let edgeLabel = graph.edge(edgeObj);
            edges.push({
                v: edgeObj.v,
                w: edgeObj.w,
                label: edgeLabel,
            });
            return edges;
        }, edges);
        let container = selectOrCreateChild(sceneGroup, 'g', Class$1.Edge.CONTAINER);
        // Select all children and join with data.
        // (Note that all children of g.edges are g.edge)
        let edgeGroups = container
            .selectAll(function () {
            return this.childNodes;
        })
            .data(edges, getEdgeKey);
        // Make edges a group to support rendering multiple lines for metaedge
        edgeGroups
            .enter()
            .append('g')
            .attr('class', Class$1.Edge.GROUP)
            .attr('data-edge', getEdgeKey)
            .each(function (d) {
            let edgeGroup = select(this);
            d.label.edgeGroup = edgeGroup;
            // index node group for quick highlighting
            sceneComponent._edgeGroupIndex[getEdgeKey(d)] = edgeGroup;
            if (sceneComponent.handleEdgeSelected) {
                // The user or some higher-level component has opted to make edges selectable.
                edgeGroup.on('click', (d) => {
                    // Stop this event's propagation so that it isn't also considered
                    // a graph-select.
                    event.stopPropagation();
                    sceneComponent.fire('edge-select', {
                        edgeData: d,
                        edgeGroup: edgeGroup,
                    });
                });
            }
            // Add line during enter because we're assuming that type of line
            // normally does not change.
            appendEdge(edgeGroup, d, sceneComponent);
        })
            .merge(edgeGroups)
            .each(function () {
            position$2(sceneElement, this);
        })
            .each(function (d) {
            stylize(select(this), d);
        });
        edgeGroups
            .exit()
            .each((d) => {
            delete sceneComponent._edgeGroupIndex[getEdgeKey(d)];
        })
            .remove();
        return edgeGroups;
    }
    /**
     * Returns the label for the given base edge.
     * The label is the shape of the underlying tensor.
     */
    function getLabelForBaseEdge(baseEdge, renderInfo) {
        let node = renderInfo.getNodeByName(baseEdge.v);
        if (node.outputShapes == null || lodash.isEmpty(node.outputShapes)) {
            return null;
        }
        let shape = node.outputShapes[baseEdge.outputTensorKey];
        if (shape == null) {
            return null;
        }
        if (shape.length === 0) {
            return 'scalar';
        }
        return shape
            .map((size) => {
            return size === -1 ? '?' : size;
        })
            .join(TENSOR_SHAPE_DELIM);
    }
    /**
     * Creates the label for the given metaedge. If the metaedge consists
     * of only 1 tensor, and it's shape is known, the label will contain that
     * shape. Otherwise, the label will say the number of tensors in the metaedge.
     */
    function getLabelForEdge(metaedge, renderInfo) {
        if (renderInfo.edgeLabelFunction) {
            // The user has specified a means of computing the label.
            return renderInfo.edgeLabelFunction(metaedge, renderInfo);
        }
        // Compute the label based on either tensor count or size.
        let isMultiEdge = metaedge.baseEdgeList.length > 1;
        return isMultiEdge
            ? metaedge.baseEdgeList.length + ' tensors'
            : getLabelForBaseEdge(metaedge.baseEdgeList[0], renderInfo);
    }
    /**
     * Computes the index into a set of points that constitute a path for which the
     * distance along the path from the initial point is as large as possible
     * without exceeding the length. This function was introduced after the
     * native getPathSegAtLength method got deprecated by SVG 2.
     * @param points Array of path control points. A point has x and y properties.
     *   Must be of length at least 2.
     * @param length The length (float).
     * @param lineFunc A function that takes points and returns the "d" attribute
     *   of a path made from connecting the points.
     * @return The index into the points array.
     */
    function getPathSegmentIndexAtLength(points, length, lineFunc) {
        const path = document.createElementNS(SVG_NAMESPACE, 'path');
        for (let i = 1; i < points.length; i++) {
            path.setAttribute('d', lineFunc(points.slice(0, i)));
            if (path.getTotalLength() > length) {
                // This many points has already exceeded the length.
                return i - 1;
            }
        }
        // The entire path is shorter than the specified length.
        return points.length - 1;
    }
    /**
     * Shortens the path enought such that the tip of the start/end marker will
     * point to the start/end of the path. The marker can be of arbitrary size.
     *
     * @param points Array of path control points.
     * @param marker D3 selection of the <marker> svg element.
     * @param isStart Is the marker a `start-marker`. If false, the marker is
     *     an `end-marker`.
     * @return The new array of control points.
     */
    function adjustPathPointsForMarker(points, marker, isStart) {
        let lineFunc = line()
            .x((d) => d.x)
            .y((d) => d.y);
        let path = select(document.createElementNS('http://www.w3.org/2000/svg', 'path'))
            .attr('d', lineFunc(points));
        let markerWidth = +marker.attr('markerWidth');
        let viewBox = marker.attr('viewBox').split(' ').map(Number);
        let viewBoxWidth = viewBox[2] - viewBox[0];
        let refX = +marker.attr('refX');
        let pathNode = path.node();
        if (isStart) {
            // The edge flows downwards. Do not make the edge go the whole way, lest we
            // clobber the arrowhead.
            const fractionStickingOut = 1 - refX / viewBoxWidth;
            const length = markerWidth * fractionStickingOut;
            const point = pathNode.getPointAtLength(length);
            // Figure out how many segments of the path we need to remove in order
            // to shorten the path.
            const segIndex = getPathSegmentIndexAtLength(points, length, lineFunc);
            // Update the very first segment.
            points[segIndex - 1] = { x: point.x, y: point.y };
            // Ignore every point before segIndex - 1.
            return points.slice(segIndex - 1);
        }
        else {
            // The edge flows upwards. Do not make the edge go the whole way, lest we
            // clobber the arrowhead.
            const fractionStickingOut = 1 - refX / viewBoxWidth;
            const length = pathNode.getTotalLength() - markerWidth * fractionStickingOut;
            const point = pathNode.getPointAtLength(length);
            // Figure out how many segments of the path we need to remove in order
            // to shorten the path.
            const segIndex = getPathSegmentIndexAtLength(points, length, lineFunc);
            // Update the very last segment.
            points[segIndex] = { x: point.x, y: point.y };
            // Ignore every point after segIndex.
            return points.slice(0, segIndex + 1);
        }
    }
    /**
     * For a given d3 selection and data object, create a path to represent the
     * edge described in d.label.
     *
     * If d.label is defined, it will be a RenderMetaedgeInfo instance. It
     * will sometimes be undefined, for example for some Annotation edges for which
     * there is no underlying Metaedge in the hierarchical graph.
     */
    function appendEdge(edgeGroup, d, sceneElement, edgeClass) {
        edgeClass = edgeClass || Class$1.Edge.LINE; // set default type
        if (d.label && d.label.structural) {
            edgeClass += ' ' + Class$1.Edge.STRUCTURAL;
        }
        if (d.label && d.label.metaedge && d.label.metaedge.numRefEdges) {
            edgeClass += ' ' + Class$1.Edge.REFERENCE_EDGE;
        }
        if (sceneElement.handleEdgeSelected) {
            // The user has opted to make edges selectable.
            edgeClass += ' ' + Class$1.Edge.SELECTABLE;
        }
        // Give the path a unique id, which will be used to link
        // the textPath (edge label) to this path.
        let pathId = 'path_' + getEdgeKey(d);
        let strokeWidth;
        if (sceneElement.renderHierarchy.edgeWidthFunction) {
            // Compute edge thickness based on the user-specified method.
            strokeWidth = sceneElement.renderHierarchy.edgeWidthFunction(d, edgeClass);
        }
        else {
            // Encode tensor size within edge thickness.
            let size = 1;
            if (d.label != null && d.label.metaedge != null) {
                // There is an underlying Metaedge.
                size = d.label.metaedge.totalSize;
            }
            strokeWidth = sceneElement.renderHierarchy.edgeWidthSizedBasedScale(size);
        }
        let path = edgeGroup
            .append('path')
            .attr('id', pathId)
            .attr('class', edgeClass)
            .style('stroke-width', strokeWidth + 'px');
        // Check if there is a reference edge and add an arrowhead of the right size.
        if (d.label && d.label.metaedge) {
            if (d.label.metaedge.numRefEdges) {
                // We have a reference edge.
                const markerId = `reference-arrowhead-${arrowheadMap(strokeWidth)}`;
                path.style('marker-start', `url(#${markerId})`);
                d.label.startMarkerId = markerId;
            }
            else {
                // We have a dataflow edge.
                const markerId = `dataflow-arrowhead-${arrowheadMap(strokeWidth)}`;
                path.style('marker-end', `url(#${markerId})`);
                d.label.endMarkerId = markerId;
            }
        }
        if (d.label == null || d.label.metaedge == null) {
            // There is no associated metaedge, thus no text.
            // This happens for annotation edges.
            return;
        }
        let labelForEdge = getLabelForEdge(d.label.metaedge, sceneElement.renderHierarchy);
        if (labelForEdge == null) {
            // We have no information to show on this edge.
            return;
        }
        edgeGroup
            .append('text')
            .append('textPath')
            .attr('xlink:href', '#' + pathId)
            .attr('startOffset', '50%')
            .attr('text-anchor', 'middle')
            .attr('dominant-baseline', 'central')
            .text(labelForEdge);
    }
    let interpolate$7 = line()
        .curve(basis$2)
        .x((d) => {
        return d.x;
    })
        .y((d) => {
        return d.y;
    });
    /**
     * Returns a tween interpolator for the endpoint of an edge path.
     */
    function getEdgePathInterpolator(component, renderPath, d, i, a) {
        let renderMetaedgeInfo = d.label;
        let adjoiningMetaedge = renderMetaedgeInfo.adjoiningMetaedge;
        let points = renderMetaedgeInfo.points;
        // Adjust the path so that start/end markers point to the end
        // of the path.
        const { shadowRoot } = component;
        if (d.label.startMarkerId) {
            points = adjustPathPointsForMarker(points, select(shadowRoot.querySelector('#' + d.label.startMarkerId)), true);
        }
        if (d.label.endMarkerId) {
            points = adjustPathPointsForMarker(points, select(shadowRoot.querySelector('#' + d.label.endMarkerId)), false);
        }
        if (!adjoiningMetaedge) {
            return interpolateValue(a, interpolate$7(points));
        }
        // Get the adjoining path that matches the adjoining metaedge.
        let adjoiningPath = (adjoiningMetaedge.edgeGroup.node().firstChild);
        // Find the desired SVGPoint along the adjoining path, then convert those
        // coordinates into the space of the renderPath using its Current
        // Transformation Matrix (CTM).
        let inbound = renderMetaedgeInfo.metaedge.inbound;
        return function (t) {
            let adjoiningPoint = adjoiningPath
                .getPointAtLength(inbound ? adjoiningPath.getTotalLength() : 0)
                .matrixTransform(adjoiningPath.getCTM())
                .matrixTransform(renderPath.getCTM().inverse());
            // Update the relevant point in the renderMetaedgeInfo's points list, then
            // re-interpolate the path.
            let index = inbound ? 0 : points.length - 1;
            points[index].x = adjoiningPoint.x;
            points[index].y = adjoiningPoint.y;
            let dPath = interpolate$7(points);
            return dPath;
        };
    }
    function position$2(component, edgeGroup) {
        select(edgeGroup)
            .select('path.' + Class$1.Edge.LINE)
            .transition()
            .attrTween('d', function (d, i, a) {
            return getEdgePathInterpolator(component, this, d, i, a);
        });
    }
    /**
     * For a given d3 selection and data object, mark the edge as a control
     * dependency if it contains only control edges.
     *
     * d's label property will be a RenderMetaedgeInfo object.
     */
    function stylize(edgeGroup, d, stylize) {
        edgeGroup.classed('faded', d.label.isFadedOut);
        let metaedge = d.label.metaedge;
        edgeGroup
            .select('path.' + Class$1.Edge.LINE)
            .classed('control-dep', metaedge && !metaedge.numRegularEdges);
    }

    /* Copyright 2015 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the 'License');
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an 'AS IS' BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * Select or Create a 'g.nodes' group to a given sceneGroup
     * and builds a number of 'g.node' groups inside the group.
     *
     * Structure Pattern:
     *
     * <g class='nodes'>
     *   <g class='node'>
     *     <g class='in-annotations'>
     *       ...
     *     </g>
     *     <g class='out-annotations'>
     *       ...
     *     </g>
     *     <g class='nodeshape'>
     *      <!--
     *      Content of the node shape should be for the node itself. For example a
     *      Metanode would have a <rect> with rounded edges, an op would have an
     *      <ellipse>. More complex nodes like series may contain multiple
     *      elements which are conditionally visible based on whether the node is
     *      expanded.
     *      -->
     *     </g>
     *     <text class='label'>node name</text>
     *     <g class='subscene'>
     *       <!--
     *       Content of  the subscene (only for metanode and series node).
     *
     *       Subscene is a svg group that contains content of the
     *       metanode's metagraph that is recursively generated by Scene.build().
     *
     *       When the graph is expanded multiple times, a subscene can contain
     *       nested subscenes inside.
     *       -->
     *     </g>
     *   </g>
     *   ...
     * </g>
     *
     *
     * @param sceneGroup selection of the container
     * @param nodeData array of render node information to map
     * @param sceneElement <tf-graph-scene> polymer element
     * @return selection of the created nodeGroups
     */
    function buildGroup$1(sceneGroup, nodeData, sceneElement) {
        let container = selectOrCreateChild(sceneGroup, 'g', Class$1.Node.CONTAINER);
        // Select all children and join with data.
        // (Note that all children of g.nodes are g.node)
        let nodeGroups = container
            .selectAll(function () {
            return this.childNodes;
        })
            .data(nodeData, (d) => {
            // make sure that we don't have to swap shape type
            return d.node.name + ':' + d.node.type;
        });
        // ENTER
        nodeGroups
            .enter()
            .append('g')
            .attr('data-name', (d) => {
            return d.node.name;
        })
            .each(function (d) {
            let nodeGroup = select(this);
            // index node group for quick stylizing
            sceneElement.addNodeGroup(d.node.name, nodeGroup);
        })
            .merge(nodeGroups)
            // ENTER + UPDATE
            .attr('class', (d) => {
            return Class$1.Node.GROUP + ' ' + nodeClass(d);
        })
            .each(function (d) {
            let nodeGroup = select(this);
            // Add g.in-annotations (always add -- to keep layer order
            // consistent.)
            let inAnnotationBox = selectOrCreateChild(nodeGroup, 'g', Class$1.Annotation.INBOX);
            buildGroupForAnnotation(inAnnotationBox, d.inAnnotations, d, sceneElement);
            // Add g.out-annotations  (always add -- to keep layer order
            // consistent.)
            let outAnnotationBox = selectOrCreateChild(nodeGroup, 'g', Class$1.Annotation.OUTBOX);
            buildGroupForAnnotation(outAnnotationBox, d.outAnnotations, d, sceneElement);
            // Build .shape first (background of the node).
            let shape = buildShape(nodeGroup, d, Class$1.Node.SHAPE);
            if (d.node.isGroupNode) {
                addButton(shape, d, sceneElement);
            }
            addInteraction(shape, d, sceneElement);
            // Build subscene on the top.
            subsceneBuild(nodeGroup, d, sceneElement);
            // Build label last. Should be on top of everything else.
            let label = labelBuild(nodeGroup, d, sceneElement);
            // Do not add interaction to metanode labels as they live inside the
            // metanode shape which already has the same interactions.
            addInteraction(label, d, sceneElement, d.node.type === NodeType.META);
            stylize$1(nodeGroup, d, sceneElement);
            position$3(nodeGroup, d);
        });
        // EXIT
        nodeGroups
            .exit()
            .each(function (d) {
            // remove all indices on remove
            sceneElement.removeNodeGroup(d.node.name);
            let nodeGroup = select(this);
            if (d.inAnnotations.list.length > 0) {
                nodeGroup
                    .select('.' + Class$1.Annotation.INBOX)
                    .selectAll('.' + Class$1.Annotation.GROUP)
                    .each((a) => {
                    sceneElement.removeAnnotationGroup(a, d);
                });
            }
            if (d.outAnnotations.list.length > 0) {
                nodeGroup
                    .select('.' + Class$1.Annotation.OUTBOX)
                    .selectAll('.' + Class$1.Annotation.GROUP)
                    .each((a) => {
                    sceneElement.removeAnnotationGroup(a, d);
                });
            }
        })
            .remove();
        return nodeGroups;
    }
    /**
     * Update or remove the subscene of a render group node depending on whether it
     * is a expanded. If the node is not a group node, this method has no effect.
     *
     * @param nodeGroup selection of the container
     * @param renderNodeInfo the render information for the node.
     * @param sceneElement <tf-graph-scene> polymer element.
     * @return Selection of the subscene group, or null if node group does not have
     *        a subscene. Op nodes, bridge nodes and unexpanded group nodes will
     *        not have a subscene.
     */
    function subsceneBuild(nodeGroup, renderNodeInfo, sceneElement) {
        if (renderNodeInfo.node.isGroupNode) {
            if (renderNodeInfo.expanded) {
                // Recursively build the subscene.
                return buildGroupForScene(nodeGroup, renderNodeInfo, sceneElement, Class$1.Subscene.GROUP);
            }
            // Clean out existing subscene if the node is not expanded.
            selectChild$1(nodeGroup, 'g', Class$1.Subscene.GROUP).remove();
        }
        return null;
    }
    /**
     * Translate the subscene of the given node group
     */
    function subscenePosition(nodeGroup, d) {
        let x0 = d.x - d.width / 2 + d.paddingLeft;
        let y0 = d.y - d.height / 2 + d.paddingTop;
        let subscene = selectChild$1(nodeGroup, 'g', Class$1.Subscene.GROUP);
        translate$1(subscene, x0, y0);
    }
    /**
     * Add an expand/collapse button to a group node
     *
     * @param selection The group node selection.
     * @param d Info about the node being rendered.
     * @param sceneElement <tf-graph-scene> polymer element.
     */
    function addButton(selection, d, sceneElement) {
        let group = selectOrCreateChild(selection, 'g', Class$1.Node.BUTTON_CONTAINER);
        selectOrCreateChild(group, 'circle', Class$1.Node.BUTTON_CIRCLE);
        selectOrCreateChild(group, 'path', Class$1.Node.EXPAND_BUTTON)
            .attr('d', 'M0,-2.2 V2.2 M-2.2,0 H2.2');
        selectOrCreateChild(group, 'path', Class$1.Node.COLLAPSE_BUTTON)
            .attr('d', 'M-2.2,0 H2.2');
        group.on('click', (d) => {
            // Stop this event's propagation so that it isn't also considered a
            // node-select.
            event.stopPropagation();
            sceneElement.fire('node-toggle-expand', { name: d.node.name });
        });
        positionButton(group, d);
    }
    /**
     * Fire node-* events when the selection is interacted.
     *
     * @param disableInteraction When true, have the provided selection
     * ignore all pointer events. Used for text labels inside of metanodes, which
     * don't need interaction as their surrounding shape has interaction, and if
     * given interaction would cause conflicts with the expand/collapse button.
     */
    function addInteraction(selection, d, sceneElement, disableInteraction) {
        if (disableInteraction) {
            selection.attr('pointer-events', 'none');
            return;
        }
        let contextMenuFunction = getMenu(sceneElement, getContextMenu(d.node, sceneElement));
        selection
            .on('dblclick', (d) => {
            sceneElement.fire('node-toggle-expand', { name: d.node.name });
        })
            .on('mouseover', (d) => {
            // don't send mouseover over expanded group,
            // otherwise it is causing too much glitches
            if (sceneElement.isNodeExpanded(d)) {
                return;
            }
            sceneElement.fire('node-highlight', { name: d.node.name });
        })
            .on('mouseout', (d) => {
            // don't send mouseover over expanded group,
            // otherwise it is causing too much glitches
            if (sceneElement.isNodeExpanded(d)) {
                return;
            }
            sceneElement.fire('node-unhighlight', { name: d.node.name });
        })
            .on('click', (d) => {
            // Stop this event's propagation so that it isn't also considered
            // a graph-select.
            event.stopPropagation();
            sceneElement.fire('node-select', { name: d.node.name });
        })
            .on('contextmenu', (d, i) => {
            sceneElement.fire('node-select', { name: d.node.name });
            contextMenuFunction.call(d, i);
        });
    }
    /**
     * Returns the d3 context menu specification for the provided node.
     */
    function getContextMenu(node, sceneElement) {
        let menu = [
            {
                title: (d) => {
                    return getIncludeNodeButtonString(node.include);
                },
                action: (elm, d, i) => {
                    sceneElement.fire('node-toggle-extract', { name: node.name });
                },
            },
        ];
        if (sceneElement.nodeContextMenuItems) {
            // Add these additional context menu items.
            menu = menu.concat(sceneElement.nodeContextMenuItems);
        }
        if (canBeInSeries(node)) {
            menu.push({
                title: (d) => {
                    return getGroupSettingLabel(node);
                },
                action: (elm, d, i) => {
                    sceneElement.fire('node-toggle-seriesgroup', {
                        name: getSeriesName(node),
                    });
                },
            });
        }
        return menu;
    }
    /** Returns if a node can be part of a grouped series */
    function canBeInSeries(node) {
        return getSeriesName(node) !== null;
    }
    /**
     * Returns the name of the possible grouped series containing this node.
     * Returns null if the node cannot be part of a grouped series of nodes.
     */
    function getSeriesName(node) {
        if (!node) {
            return null;
        }
        if (node.type === NodeType.SERIES) {
            return node.name;
        }
        if (node.type === NodeType.OP) {
            let op = node;
            return op.owningSeries;
        }
        return null;
    }
    /**
     * Returns the SeriesNode that represents the series that the provided node
     * is contained in (or itself if the provided node is itself a SeriesNode).
     * Returns null if the node is not rendered as part of a series.
     */
    function getContainingSeries(node) {
        let s = null;
        if (!node) {
            return null;
        }
        else if (node.type === NodeType.SERIES) {
            s = node;
        }
        else if (node.parentNode && node.parentNode.type === NodeType.SERIES) {
            s = node.parentNode;
        }
        return s;
    }
    /**
     * Returns the label for a button to toggle the group setting of the provided
     * node.
     */
    function getGroupSettingLabel(node) {
        return getGroupSeriesNodeButtonString(getContainingSeries(node) !== null
            ? SeriesGroupingType.GROUP
            : SeriesGroupingType.UNGROUP);
    }
    /**
     * Append svg text for label and assign data.
     * @param nodeGroup
     * @param renderNodeInfo The render node information for the label.
     * @param sceneElement <tf-graph-scene> polymer element.
     */
    function labelBuild(nodeGroup, renderNodeInfo, sceneElement) {
        let text = renderNodeInfo.displayName;
        // Truncate long labels for unexpanded Metanodes.
        let useFontScale = renderNodeInfo.node.type === NodeType.META && !renderNodeInfo.expanded;
        let label = selectOrCreateChild(nodeGroup, 'text', Class$1.Node.LABEL);
        // Make sure the label is visually on top among its siblings.
        let labelNode = label.node();
        labelNode.parentNode.appendChild(labelNode);
        label.attr('dy', '.35em').attr('text-anchor', 'middle');
        if (useFontScale) {
            if (text.length > sceneElement.maxMetanodeLabelLength) {
                text = text.substr(0, sceneElement.maxMetanodeLabelLength - 2) + '...';
            }
            let scale = getLabelFontScale(sceneElement);
            label.attr('font-size', scale(text.length) + 'px');
        }
        let txtElement = label.text(text);
        enforceLabelWidth(txtElement, renderNodeInfo.node.type, renderNodeInfo);
        return label;
    }
    /**
     * This function shortens text which would exceed the maximum pixel width of
     * a label.
     *
     * @param txtElementSelection The text element containing the label's text as d3
     * selection.
     * @param nodeType The type of the node the label belongs to. If the node is
     * an annotation, the value is -1. Label widths are defined in
     * layout.PARAMS.nodeSize.{meta|op|...}.maxLabelWidth for nodes and
     * layout.PARAMS.annotations.labelWidth for annotations.
     * @param renderNodeInfo The render information about the node, required to
     * determine whether META nodes are collapsed or expanded.
     */
    function enforceLabelWidth(txtElementSelection, nodeType, renderNodeInfo) {
        // Get text element itself and its on-screen width.
        let txtNode = txtElementSelection.node();
        let computedTxtLength = txtNode.getComputedTextLength();
        let labelContent = txtNode.textContent;
        // Get maximum length from settings.
        let maxLength = null;
        switch (nodeType) {
            case NodeType.META:
                if (renderNodeInfo && !renderNodeInfo.expanded) {
                    // Only trim text if
                    // node expanded.
                    maxLength = PARAMS$1.nodeSize.meta.maxLabelWidth;
                }
                break;
            case NodeType.OP:
                maxLength = PARAMS$1.nodeSize.op.maxLabelWidth;
                break;
            case -1:
                maxLength = PARAMS$1.annotations.maxLabelWidth;
                break;
        }
        // Return if no max length provided for node type, or current label length is
        // less than or equal to the provided length limit.
        if (maxLength === null || computedTxtLength <= maxLength) {
            return;
        }
        // Find the index of the character which exceeds the width.
        // getSubStringLength performs far better than getComputedTextLength, and
        // results in a 3x speed-up on average.
        let index = 1;
        while (txtNode.getSubStringLength(0, index) < maxLength) {
            index++;
        }
        // Shorten the label starting at the string length known to be one
        // character above max pixel length.
        // When shortened the original label's substring is concatenated with
        // '...', baseText contains the substring not including the '...'.
        let baseText = txtNode.textContent.substr(0, index);
        do {
            baseText = baseText.substr(0, baseText.length - 1);
            // Recompute text length.
            txtNode.textContent = baseText + '...';
            computedTxtLength = txtNode.getComputedTextLength();
        } while (computedTxtLength > maxLength && baseText.length > 0);
        // Add tooltip with full name and return.
        return txtElementSelection.append('title').text(labelContent);
    }
    /**
     * d3 scale used for sizing font of labels, used by labelBuild,
     * initialized once by getLabelFontScale.
     */
    let fontScale = null;
    function getLabelFontScale(sceneElement) {
        if (!fontScale) {
            fontScale = linear$2()
                .domain([
                sceneElement.maxMetanodeLabelLengthLargeFont,
                sceneElement.maxMetanodeLabelLength,
            ])
                .range([
                sceneElement.maxMetanodeLabelLengthFontSize,
                sceneElement.minMetanodeLabelLengthFontSize,
            ])
                .clamp(true);
        }
        return fontScale;
    }
    /**
     * Set label position of a given node group
     */
    function labelPosition(nodeGroup, cx, cy, yOffset) {
        selectChild$1(nodeGroup, 'text', Class$1.Node.LABEL)
            .transition()
            .attr('x', cx)
            .attr('y', cy + yOffset);
    }
    /**
     * Select or append/insert shape for a node and assign renderNode
     * as the shape's data.
     *
     * @param nodeGroup
     * @param d Render node information.
     * @param nodeClass class for the element.
     * @return Selection of the shape.
     */
    function buildShape(nodeGroup, d, nodeClass) {
        // Create a group to house the underlying visual elements.
        let shapeGroup = selectOrCreateChild(nodeGroup, 'g', nodeClass);
        // TODO: DOM structure should be templated in HTML somewhere, not JS.
        switch (d.node.type) {
            case NodeType.OP:
                const opNode = d.node;
                if (lodash.isNumber(opNode.functionInputIndex) ||
                    lodash.isNumber(opNode.functionOutputIndex)) {
                    // This is input or output arg for a TensorFlow function. Use a special
                    // shape (a triangle) for them.
                    selectOrCreateChild(shapeGroup, 'polygon', Class$1.Node.COLOR_TARGET);
                    break;
                }
                selectOrCreateChild(shapeGroup, 'ellipse', Class$1.Node.COLOR_TARGET);
                break;
            case NodeType.SERIES:
                // Choose the correct stamp to use to represent this series.
                let stampType = 'annotation';
                let groupNodeInfo = d;
                if (groupNodeInfo.coreGraph) {
                    stampType = groupNodeInfo.node.hasNonControlEdges
                        ? 'vertical'
                        : 'horizontal';
                }
                let classList = [Class$1.Node.COLOR_TARGET];
                if (groupNodeInfo.isFadedOut) {
                    classList.push('faded-ellipse');
                }
                selectOrCreateChild(shapeGroup, 'use', classList)
                    .attr('xlink:href', '#op-series-' + stampType + '-stamp');
                selectOrCreateChild(shapeGroup, 'rect', Class$1.Node.COLOR_TARGET)
                    .attr('rx', d.radius)
                    .attr('ry', d.radius);
                break;
            case NodeType.BRIDGE:
                selectOrCreateChild(shapeGroup, 'rect', Class$1.Node.COLOR_TARGET)
                    .attr('rx', d.radius)
                    .attr('ry', d.radius);
                break;
            case NodeType.META:
                selectOrCreateChild(shapeGroup, 'rect', Class$1.Node.COLOR_TARGET)
                    .attr('rx', d.radius)
                    .attr('ry', d.radius);
                break;
            default:
                throw Error('Unrecognized node type: ' + d.node.type);
        }
        return shapeGroup;
    }
    function nodeClass(d) {
        switch (d.node.type) {
            case NodeType.OP:
                return Class$1.OPNODE;
            case NodeType.META:
                return Class$1.METANODE;
            case NodeType.SERIES:
                return Class$1.SERIESNODE;
            case NodeType.BRIDGE:
                return Class$1.BRIDGENODE;
            case NodeType.ELLIPSIS:
                return Class$1.ELLIPSISNODE;
        }
        throw Error('Unrecognized node type: ' + d.node.type);
    }
    /** Modify node and its subscene and its label's positional attributes */
    function position$3(nodeGroup, d) {
        let shapeGroup = selectChild$1(nodeGroup, 'g', Class$1.Node.SHAPE);
        let cx = computeCXPositionOfNodeShape(d);
        switch (d.node.type) {
            case NodeType.OP: {
                // position shape
                const opNode = d.node;
                if (lodash.isNumber(opNode.functionInputIndex) ||
                    lodash.isNumber(opNode.functionOutputIndex)) {
                    // This shape represents the input into or output out of a TensorFlow
                    // function.
                    let shape = selectChild$1(shapeGroup, 'polygon');
                    positionTriangle(shape, d.x, d.y, d.coreBox.width, d.coreBox.height);
                }
                else {
                    let shape = selectChild$1(shapeGroup, 'ellipse');
                    positionEllipse(shape, cx, d.y, d.coreBox.width, d.coreBox.height);
                }
                labelPosition(nodeGroup, cx, d.y, d.labelOffset);
                break;
            }
            case NodeType.META: {
                // position shape
                let shapes = shapeGroup.selectAll('rect');
                if (d.expanded) {
                    positionRect(shapes, d.x, d.y, d.width, d.height);
                    subscenePosition(nodeGroup, d);
                    // Put the label on top.
                    labelPosition(nodeGroup, cx, d.y, -d.height / 2 + d.labelHeight / 2);
                }
                else {
                    positionRect(shapes, cx, d.y, d.coreBox.width, d.coreBox.height);
                    // Place the label in the middle.
                    labelPosition(nodeGroup, cx, d.y, 0);
                }
                break;
            }
            case NodeType.SERIES: {
                let shape = selectChild$1(shapeGroup, 'use');
                if (d.expanded) {
                    positionRect(shape, d.x, d.y, d.width, d.height);
                    subscenePosition(nodeGroup, d);
                    // put label on top
                    labelPosition(nodeGroup, cx, d.y, -d.height / 2 + d.labelHeight / 2);
                }
                else {
                    positionRect(shape, cx, d.y, d.coreBox.width, d.coreBox.height);
                    labelPosition(nodeGroup, cx, d.y, d.labelOffset);
                }
                break;
            }
            case NodeType.BRIDGE: {
                // position shape
                // NOTE: In reality, these will not be visible, but it helps to put them
                // in the correct position for debugging purposes.
                let shape = selectChild$1(shapeGroup, 'rect');
                positionRect(shape, d.x, d.y, d.width, d.height);
                break;
            }
            default: {
                throw Error('Unrecognized node type: ' + d.node.type);
            }
        }
    }
    /** Enum specifying the options to color nodes by */
    var ColorBy;
    (function (ColorBy) {
        ColorBy[ColorBy["STRUCTURE"] = 0] = "STRUCTURE";
        ColorBy[ColorBy["DEVICE"] = 1] = "DEVICE";
        ColorBy[ColorBy["XLA_CLUSTER"] = 2] = "XLA_CLUSTER";
        ColorBy[ColorBy["COMPUTE_TIME"] = 3] = "COMPUTE_TIME";
        ColorBy[ColorBy["MEMORY"] = 4] = "MEMORY";
        ColorBy[ColorBy["OP_COMPATIBILITY"] = 5] = "OP_COMPATIBILITY";
    })(ColorBy || (ColorBy = {}));
    function getGradient(id, colors, svgRoot) {
        let escapedId = escapeQuerySelector(id);
        if (!svgRoot)
            return `url(#${escapedId})`;
        let $svgRoot = select(svgRoot);
        let gradientDefs = $svgRoot.select('defs#_graph-gradients');
        if (gradientDefs.empty()) {
            gradientDefs = $svgRoot.append('defs').attr('id', '_graph-gradients');
        }
        let linearGradient = gradientDefs.select('linearGradient#' + escapedId);
        // If the linear gradient is not there yet, create it.
        if (linearGradient.empty()) {
            linearGradient = gradientDefs.append('linearGradient').attr('id', id);
            // Re-create the stops of the linear gradient.
            linearGradient.selectAll('*').remove();
            let cumulativeProportion = 0;
            // For each color, create a stop using the proportion of that device.
            lodash.each(colors, (d) => {
                let color = d.color;
                linearGradient
                    .append('stop')
                    .attr('offset', cumulativeProportion)
                    .attr('stop-color', color);
                linearGradient
                    .append('stop')
                    .attr('offset', cumulativeProportion + d.proportion)
                    .attr('stop-color', color);
                cumulativeProportion += d.proportion;
            });
        }
        return `url(#${escapedId})`;
    }
    function removeGradientDefinitions(svgRoot) {
        select(svgRoot).select('defs#_graph-gradients').remove();
    }
    /**
     * Returns the fill color for the node given its state and the 'color by'
     * option.
     * Takes in optional svgRoot, when passed, that populates SVG definitions
     * for the fill inside the svgRoot when necessary.
     */
    function getFillForNode(templateIndex, colorBy, renderInfo, isExpanded, svgRoot) {
        let colorParams = MetanodeColors;
        switch (colorBy) {
            case ColorBy.STRUCTURE:
                if (renderInfo.node.type === NodeType.META) {
                    let tid = renderInfo.node.templateId;
                    return tid === null
                        ? colorParams.UNKNOWN
                        : colorParams.STRUCTURE_PALETTE(templateIndex(tid), isExpanded);
                }
                else if (renderInfo.node.type === NodeType.SERIES) {
                    // If expanded, we're showing the background rect, which we want to
                    // appear gray. Otherwise we're showing a stack of ellipses which we
                    // want to show white.
                    return isExpanded ? colorParams.EXPANDED_COLOR : 'white';
                }
                else if (renderInfo.node.type === NodeType.BRIDGE) {
                    return renderInfo.structural
                        ? '#f0e'
                        : renderInfo.node.inbound
                            ? '#0ef'
                            : '#fe0';
                }
                else if (lodash.isNumber(renderInfo.node.functionInputIndex)) {
                    // This is an input of a TensorFlow function.
                    return '#795548';
                }
                else if (lodash.isNumber(renderInfo.node.functionOutputIndex)) {
                    // This is an output of a TensorFlow function.
                    return '#009688';
                }
                else {
                    // Op nodes are white.
                    return 'white';
                }
            case ColorBy.DEVICE:
                if (renderInfo.deviceColors == null) {
                    // Return the hue for unknown device.
                    return colorParams.UNKNOWN;
                }
                return isExpanded
                    ? colorParams.EXPANDED_COLOR
                    : getGradient('device-' + renderInfo.node.name, renderInfo.deviceColors, svgRoot);
            case ColorBy.XLA_CLUSTER:
                if (renderInfo.xlaClusterColors == null) {
                    // Return the hue for unknown xlaCluster.
                    return colorParams.UNKNOWN;
                }
                return isExpanded
                    ? colorParams.EXPANDED_COLOR
                    : getGradient('xla-' + renderInfo.node.name, renderInfo.xlaClusterColors, svgRoot);
            case ColorBy.COMPUTE_TIME:
                return isExpanded
                    ? colorParams.EXPANDED_COLOR
                    : renderInfo.computeTimeColor || colorParams.UNKNOWN;
            case ColorBy.MEMORY:
                return isExpanded
                    ? colorParams.EXPANDED_COLOR
                    : renderInfo.memoryColor || colorParams.UNKNOWN;
            case ColorBy.OP_COMPATIBILITY:
                if (renderInfo.compatibilityColors == null) {
                    // Return the hue for unknown compatibility info.
                    return colorParams.UNKNOWN;
                }
                return isExpanded
                    ? colorParams.EXPANDED_COLOR
                    : getGradient('op-compat-' + renderInfo.node.name, renderInfo.compatibilityColors, svgRoot);
            default:
                throw new Error('Unknown case to color nodes by');
        }
    }
    /**
     * Modify node style by toggling class and assign attributes (only for things
     * that can't be done in css).
     */
    function stylize$1(nodeGroup, renderInfo, sceneElement, nodeClass) {
        nodeClass = nodeClass || Class$1.Node.SHAPE;
        let isHighlighted = sceneElement.isNodeHighlighted(renderInfo.node.name);
        let isSelected = sceneElement.isNodeSelected(renderInfo.node.name);
        let isExtract = renderInfo.isInExtract ||
            renderInfo.isOutExtract ||
            renderInfo.isLibraryFunction;
        let isExpanded = renderInfo.expanded && nodeClass !== Class$1.Annotation.NODE;
        let isFadedOut = renderInfo.isFadedOut;
        nodeGroup.classed('highlighted', isHighlighted);
        nodeGroup.classed('selected', isSelected);
        nodeGroup.classed('extract', isExtract);
        nodeGroup.classed('expanded', isExpanded);
        nodeGroup.classed('faded', isFadedOut);
        // Main node always exists here and it will be reached before subscene,
        // so d3 selection is fine here.
        let node = nodeGroup.select('.' + nodeClass + ' .' + Class$1.Node.COLOR_TARGET);
        let fillColor = getFillForNode(sceneElement.templateIndex, ColorBy[sceneElement.colorBy.toUpperCase()], renderInfo, isExpanded, sceneElement.getGraphSvgRoot());
        node.style('fill', fillColor);
        // Choose outline to be darker version of node color if the node is a single
        // color and is not selected.
        node.style('stroke', isSelected ? null : getStrokeForFill(fillColor));
    }
    /**
     * Given a node's fill color/gradient, determine the stroke for the node.
     */
    function getStrokeForFill(fill) {
        // If node is colored by a gradient, then use a dark gray outline.
        return fill.substring(0, 3) === 'url'
            ? MetanodeColors.GRADIENT_OUTLINE
            : rgb(fill).darker().toString();
    }
    /**
     * Finds selected node and highlights all nodes which are providing direct
     * or indirect input to the node and all edges connecting these nodes
     * together and to the selected node.
     */
    function updateInputTrace(svgRoot, renderGraphInfo, selectedNodeName, traceInputs) {
        // Reset all styling.
        const svg = select(svgRoot);
        svg.selectAll('.input-highlight').classed('input-highlight', false);
        svg.selectAll('.non-input').classed('non-input', false);
        svg.selectAll('.input-parent').classed('input-parent', false);
        svg.selectAll('.input-child').classed('input-child', false);
        svg.selectAll('.input-edge-highlight').classed('input-edge-highlight', false);
        svg
            .selectAll('.non-input-edge-highlight')
            .classed('non-input-edge-highlight', false);
        svg
            .selectAll('.input-highlight-selected')
            .classed('input-highlight-selected', false);
        // Extract currently selected node. Return if input tracing disabled or no
        // node is selected.
        if (!renderGraphInfo || !traceInputs || !selectedNodeName) {
            return;
        }
        let opNodes = _getAllContainedOpNodes(selectedNodeName, renderGraphInfo);
        let allTracedNodes = {};
        lodash.each(opNodes, function (nodeInstance) {
            allTracedNodes = traceAllInputsOfOpNode(svgRoot, renderGraphInfo, nodeInstance, allTracedNodes);
        });
        // Highlight all parent nodes of each OpNode as input parent to allow
        // specific highlighting.
        let highlightedNodes = Object.keys(allTracedNodes);
        let visibleNodes = _findVisibleParentsFromOpNodes(renderGraphInfo, highlightedNodes);
        _markParentsOfNodes(svgRoot, visibleNodes);
        // Attach class to all non-input nodes and edges for styling.
        svg
            .selectAll('g.node:not(.selected):not(.input-highlight)' +
            ':not(.input-parent):not(.input-children)')
            .classed('non-input', true)
            .each(function (d) {
            // Mark all nodes with the specified name as non-inputs. This
            // results in Annotation nodes which are attached to inputs to be
            // tagged as well.
            let nodeName = d.node.name;
            svg.selectAll(`[data-name="${nodeName}"]`).classed('non-input', true);
        });
        svg
            .selectAll('g.edge:not(.input-edge-highlight)')
            .classed('non-input-edge-highlight', true);
    }
    /**
     * Recursively find all op nodes contained by the node identified by the
     * provided name.
     * @param nodeName The meta or op node of which the OpNode instances are
     * required.
     * @param renderGraphInfo The rendered graph information object.
     * @returns {Array} An array of OpNodeImpl instances.
     */
    function _getAllContainedOpNodes(nodeName, renderGraphInfo) {
        let opNodes = [];
        // Get current node.
        let node = renderGraphInfo.getNodeByName(nodeName);
        // If node is already OpNode then return the node plus its input embeddings.
        if (node instanceof OpNodeImpl) {
            return [node].concat(node.inEmbeddings);
        }
        // Otherwise, make recursive call for each node contained by the GroupNode.
        let childNodeNames = node.metagraph.nodes();
        lodash.each(childNodeNames, function (childNodeName) {
            opNodes = opNodes.concat(_getAllContainedOpNodes(childNodeName, renderGraphInfo));
        });
        return opNodes;
    }
    function traceAllInputsOfOpNode(svgRoot, renderGraphInfo, startNode, allTracedNodes) {
        // To prevent infinite loops due to cyclical relationships and improving
        // performance by tracing OpNode which is input to 2+ nodes only once.
        if (allTracedNodes[startNode.name]) {
            return allTracedNodes;
        }
        else {
            allTracedNodes[startNode.name] = true;
        }
        // Extract the inputs.
        let inputs = startNode.inputs;
        // Get visible parent.
        let currentVisibleParent = getVisibleParent(renderGraphInfo, startNode);
        // Mark as input node.
        select(svgRoot)
            .select(`.node[data-name="${currentVisibleParent.name}"]`)
            .classed('input-highlight', true);
        // Find the visible parent of each input.
        let visibleInputs = {};
        lodash.each(inputs, function (nodeInstance) {
            let resolvedNode = renderGraphInfo.getNodeByName(nodeInstance.name);
            if (resolvedNode === undefined) {
                // Node could not be found in rendered Hierarchy, which happens when
                // tracing inputs of a SummaryNode.
                return;
            }
            // Ensure node is resolved to OpNode if name collision with Metanode exists.
            if (resolvedNode instanceof MetanodeImpl) {
                let resolvedNodeName = getStrictName(resolvedNode.name);
                resolvedNode = renderGraphInfo.getNodeByName(resolvedNodeName);
            }
            let visibleParent = getVisibleParent(renderGraphInfo, resolvedNode);
            // Append OpNode to visible parent entry.
            let visibleInputsEntry = visibleInputs[visibleParent.name];
            if (visibleInputsEntry) {
                visibleInputsEntry.opNodes.push(resolvedNode);
            }
            else {
                // Create new entry.
                visibleInputs[visibleParent.name] = {
                    visibleParent: visibleParent,
                    opNodes: [resolvedNode],
                };
            }
        });
        // Find all parents of the start node.
        let startNodeParents = {};
        let indexedStartNodeParents = [currentVisibleParent];
        startNodeParents[currentVisibleParent.name] = {
            traced: false,
            index: 0,
            connectionEndpoints: [],
        };
        let currentNode = currentVisibleParent;
        for (let index = 1; currentNode.name !== ROOT_NAME; index++) {
            currentNode = currentNode.parentNode;
            startNodeParents[currentNode.name] = {
                traced: false,
                index: index,
                connectionEndpoints: [],
            };
            indexedStartNodeParents[index] = currentNode;
        }
        // Find first mutual parent of each input node and highlight connection.
        lodash.forOwn(visibleInputs, function (visibleParentInfo, key) {
            let nodeInstance = visibleParentInfo.visibleParent;
            // Make recursive call for each input-OpNode contained by the visible
            // parent.
            lodash.each(visibleParentInfo.opNodes, function (opNode) {
                allTracedNodes = traceAllInputsOfOpNode(svgRoot, renderGraphInfo, opNode, allTracedNodes);
            });
            if (nodeInstance.name !== currentVisibleParent.name) {
                _createVisibleTrace(svgRoot, nodeInstance, startNodeParents, indexedStartNodeParents);
            }
        });
        return allTracedNodes;
    }
    /**
     * Colors the edges to connect the passed node to the start node. This is
     * done by:
     *
     * a) Finding the first (visible) common parent in the rendered
     * hierarchy.
     * NB: There are 2 types of connections:
     * 1) Direct connections between node A
     * and B, marked below as II,
     * 2) Connections from any node A to its parent, A'. Marked below as I and III.
     * For type 2 connection you need to know the inner-nested node, the
     * direct parent, and the ultimate destination of the connection.
     *
     *  A_parent      B_parent
     * +--------+    +---------+
     * |        |    |         |
     * |  +--+ I| II |III+--+  |
     * |  |A +---------->+B |  |
     * |  +--+  |    |   +--+  |
     * |        |    |         |
     * +--------+    +---------+
     *
     *
     * b) Highlighting the direct connection between the parents of A and B,
     * called A_parent and B_parent, s.t. A_parent and B_parent are children of the
     * mutual parent of A and B found in a), marked above as II.
     *
     * c) Highlighting the connection from A to A_parent and B to B_parent
     * (through all layers of parents between A and A_parent and B and B_parent,
     * respectively). Marked above as I and III.
     *
     * @param nodeInstance The instance of the node to use as destination node, B.
     * @param startNodeParents Map of startNodeParent names to information objects
     * about the parent.
     * @param indexedStartNodeParents An array of all parents of the start node.
     * This is required to find the child of the mutual parent which is a parent
     * of the start node.
     * @private
     */
    function _createVisibleTrace(svgRoot, nodeInstance, startNodeParents, indexedStartNodeParents) {
        let currentNode = nodeInstance;
        let previousNode = nodeInstance;
        // Ascend through parents until a mutual parent is found with the start
        // node.
        let destinationParentPairs = [];
        while (!startNodeParents[currentNode.name]) {
            if (previousNode.name !== currentNode.name) {
                destinationParentPairs.push([previousNode, currentNode]);
            }
            previousNode = currentNode;
            currentNode = currentNode.parentNode;
        }
        // Connection between nodes is drawn between the parents of each
        // respective node, both of which share the mutual parent.
        let startNodeIndex = startNodeParents[currentNode.name].index;
        let startNodeName = indexedStartNodeParents[Math.max(startNodeIndex - 1, 0)].name;
        let startNodeTopParentName = startNodeName;
        let targetNodeTopParentName = previousNode.name;
        let endNodeName = previousNode.name;
        const svg = select(svgRoot);
        svg
            .selectAll(`[data-edge="${endNodeName}--${startNodeName}"]`)
            .classed('input-edge-highlight', true);
        // Trace up the parents of the input.
        lodash.each(destinationParentPairs, function (value) {
            let inner = value[0];
            let outer = value[1];
            let edgeSelector = `[data-edge="${inner.name}--${startNodeTopParentName}` +
                `~~${outer.name}~~OUT"]`;
            svg.selectAll(edgeSelector).classed('input-edge-highlight', true);
        });
        // Trace up the parents of the start node.
        for (let index = 1; index < startNodeIndex; index++) {
            let inner = indexedStartNodeParents[index - 1];
            let outer = indexedStartNodeParents[index];
            let edgeSelector = `[data-edge="${targetNodeTopParentName}~~${outer.name}` +
                `~~IN--${inner.name}"]`;
            svg.selectAll(edgeSelector).classed('input-edge-highlight', true);
        }
    }
    /**
     * Creates map { [name: string] -> Node } of all visible / rendered parents
     * of the nodes identified by the node names passed in.
     *
     * @param renderGraphInfo The information on the rendered graph.
     * @param nodeNames String array of node names.
     * @returns {[nodeName: string]: Node}
     * @private
     */
    function _findVisibleParentsFromOpNodes(renderGraphInfo, nodeNames) {
        let visibleParents = {};
        lodash.each(nodeNames, function (nodeName) {
            let currentNode = renderGraphInfo.getNodeByName(nodeName);
            let visibleParent = getVisibleParent(renderGraphInfo, currentNode);
            visibleParents[visibleParent.name] = visibleParent;
        });
        return visibleParents;
    }
    /**
     * Traverse through the parents of all nodes in the list and mark each
     * encountered node as input-parent.
     * @param visibleNodes Map of input nodes, have to be visible/rendered when
     * called.
     * @private
     */
    function _markParentsOfNodes(svgRoot, visibleNodes) {
        lodash.forOwn(visibleNodes, function (nodeInstance) {
            // Mark all parents of the node as input-parents.
            let currentNode = nodeInstance;
            while (currentNode.name !== ROOT_NAME) {
                const renderedElementSelection = select(svgRoot)
                    .select(`.node[data-name="${currentNode.name}"]`);
                // Only mark the element as a parent node to an input if it is not
                // marked as input node itself.
                if (renderedElementSelection.nodes().length &&
                    !renderedElementSelection.classed('input-highlight') &&
                    !renderedElementSelection.classed('selected') &&
                    // OpNode only parent if start node is embedded node, in which case
                    // the OpNode should be faded as well.
                    !renderedElementSelection.classed('op')) {
                    renderedElementSelection.classed('input-parent', true);
                }
                currentNode = currentNode.parentNode;
            }
        });
    }
    /**
     * Find the parent of the passed in op node which is expanded. This is done
     * by going through all parents until the parent's parent is expanded, thus
     * finding the first unexpanded parent which is rendered on the screen.
     * @param renderGraphInfo The graph info object used to gain access to the
     * render info of the parents.
     * @param currentNode The node whose parent is to be found.
     * @returns Node
     */
    function getVisibleParent(renderGraphInfo, currentNode) {
        let found = false;
        let currentParent = currentNode;
        while (!found) {
            // Get parent element, to extract name.
            currentNode = currentParent;
            currentParent = currentNode.parentNode;
            if (currentParent === undefined) {
                found = true;
            }
            else {
                let renderNode = renderGraphInfo.getRenderNodeByName(currentParent.name);
                // Found if node is rendered on the screen (renderNode truthy), and
                // the parent is either expanded (i.e. it is a metanode or seriesnode)
                // or the parent is an OpNode in which case currentNode is an embedded
                // node which has another OpNode as parent.
                if (renderNode &&
                    (renderNode.expanded || currentParent instanceof OpNodeImpl)) {
                    found = true;
                }
            }
        } // Close while loop.
        return currentNode;
    }
    /**
     * Annotations.
     */
    function buildGroupForAnnotation(container, annotationData, d, sceneElement) {
        // Select all children and join with data.
        let annotationGroups = container
            .selectAll(function () {
            // using d3's selector function
            // See https://github.com/mbostock/d3/releases/tag/v2.0.0
            // (It's not listed in the d3 wiki.)
            return this.childNodes;
        })
            .data(annotationData.list, (d) => {
            return d.node.name;
        });
        annotationGroups
            .enter()
            .append('g')
            .attr('data-name', (a) => {
            return a.node.name;
        })
            .each(function (a) {
            let aGroup = select(this);
            // Add annotation to the index in the scene
            sceneElement.addAnnotationGroup(a, d, aGroup);
            // Append annotation edge
            let edgeType = Class$1.Annotation.EDGE;
            let metaedge = a.renderMetaedgeInfo && a.renderMetaedgeInfo.metaedge;
            if (metaedge && !metaedge.numRegularEdges) {
                edgeType += ' ' + Class$1.Annotation.CONTROL_EDGE;
            }
            // If any edges are reference edges, add the reference edge class.
            if (metaedge && metaedge.numRefEdges) {
                edgeType += ' ' + Class$1.Edge.REF_LINE;
            }
            appendEdge(aGroup, a, sceneElement, edgeType);
            if (a.annotationType !== AnnotationType.ELLIPSIS) {
                addAnnotationLabelFromNode(aGroup, a);
                buildShapeForAnnotation(aGroup, a);
            }
            else {
                addAnnotationLabel(aGroup, a.node.name, a, Class$1.Annotation.ELLIPSIS);
            }
        })
            .merge(annotationGroups)
            .attr('class', (a) => {
            return (Class$1.Annotation.GROUP +
                ' ' +
                annotationToClassName(a.annotationType) +
                ' ' +
                nodeClass(a));
        })
            .each(function (a) {
            let aGroup = select(this);
            update(aGroup, d, a, sceneElement);
            if (a.annotationType !== AnnotationType.ELLIPSIS) {
                addInteractionForAnnotation(aGroup, d, a, sceneElement);
            }
        });
        annotationGroups
            .exit()
            .each(function (a) {
            let aGroup = select(this);
            // Remove annotation from the index in the scene
            sceneElement.removeAnnotationGroup(a, d, aGroup);
        })
            .remove();
        return annotationGroups;
    }
    /**
     * Maps an annotation enum to a class name used in css rules.
     */
    function annotationToClassName(annotationType) {
        return (AnnotationType[annotationType] || '').toLowerCase() || null;
    }
    function buildShapeForAnnotation(aGroup, a) {
        if (a.annotationType === AnnotationType.SUMMARY) {
            let summary = selectOrCreateChild(aGroup, 'use');
            summary
                .attr('class', 'summary')
                .attr('xlink:href', '#summary-icon')
                .attr('cursor', 'pointer');
        }
        else {
            let shape = buildShape(aGroup, a, Class$1.Annotation.NODE);
            // add title tag to get native tooltips
            selectOrCreateChild(shape, 'title').text(a.node.name);
        }
    }
    function addAnnotationLabelFromNode(aGroup, a) {
        let namePath = a.node.name.split('/');
        let text = namePath[namePath.length - 1];
        return addAnnotationLabel(aGroup, text, a, null);
    }
    function addAnnotationLabel(aGroup, label, a, additionalClassNames) {
        let classNames = Class$1.Annotation.LABEL;
        if (additionalClassNames) {
            classNames += ' ' + additionalClassNames;
        }
        let txtElement = aGroup
            .append('text')
            .attr('class', classNames)
            .attr('dy', '.35em')
            .attr('text-anchor', a.isIn ? 'end' : 'start')
            .text(label);
        return enforceLabelWidth(txtElement, -1);
    }
    function addInteractionForAnnotation(selection, d, annotation, sceneElement) {
        selection
            .on('mouseover', (a) => {
            sceneElement.fire('annotation-highlight', {
                name: a.node.name,
                hostName: d.node.name,
            });
        })
            .on('mouseout', (a) => {
            sceneElement.fire('annotation-unhighlight', {
                name: a.node.name,
                hostName: d.node.name,
            });
        })
            .on('click', (a) => {
            // Stop this event's propagation so that it isn't also considered a
            // graph-select.
            event.stopPropagation();
            sceneElement.fire('annotation-select', {
                name: a.node.name,
                hostName: d.node.name,
            });
        });
        if (annotation.annotationType !== AnnotationType.SUMMARY &&
            annotation.annotationType !== AnnotationType.CONSTANT) {
            selection.on('contextmenu', getMenu(sceneElement, getContextMenu(annotation.node, sceneElement)));
        }
    }
    /**
     * Adjust annotation's position.
     *
     * @param aGroup selection of a 'g.annotation' element.
     * @param d Host node data.
     * @param a annotation node data.
     * @param sceneElement <tf-graph-scene> polymer element.
     */
    function update(aGroup, d, a, sceneElement) {
        let cx = computeCXPositionOfNodeShape(d);
        // Annotations that point to embedded nodes (constants,summary)
        // don't have a render information attached so we don't stylize these.
        // Also we don't stylize ellipsis annotations (the string '... and X more').
        if (a.renderNodeInfo && a.annotationType !== AnnotationType.ELLIPSIS) {
            stylize$1(aGroup, a.renderNodeInfo, sceneElement, Class$1.Annotation.NODE);
        }
        if (a.annotationType === AnnotationType.SUMMARY) {
            // Update the width of the annotation to give space for the image.
            a.width += 10;
        }
        // label position
        aGroup
            .select('text.' + Class$1.Annotation.LABEL)
            .transition()
            .attr('x', cx + a.dx + (a.isIn ? -1 : 1) * (a.width / 2 + a.labelOffset))
            .attr('y', d.y + a.dy);
        // Some annotations (such as summary) are represented using a 12x12 image tag.
        // Purposely omitted units (e.g. pixels) since the images are vector graphics.
        // If there is an image, we adjust the location of the image to be vertically
        // centered with the node and horizontally centered between the arrow and the
        // text label.
        aGroup
            .select('use.summary')
            .transition()
            .attr('x', cx + a.dx - 3)
            .attr('y', d.y + a.dy - 6);
        // Node position (only one of the shape selection will be non-empty.)
        positionEllipse(aGroup.select('.' + Class$1.Annotation.NODE + ' ellipse'), cx + a.dx, d.y + a.dy, a.width, a.height);
        positionRect(aGroup.select('.' + Class$1.Annotation.NODE + ' rect'), cx + a.dx, d.y + a.dy, a.width, a.height);
        positionRect(aGroup.select('.' + Class$1.Annotation.NODE + ' use'), cx + a.dx, d.y + a.dy, a.width, a.height);
        // Edge position
        aGroup
            .select('path.' + Class$1.Annotation.EDGE)
            .transition()
            .attr('d', (a) => {
            // map relative position to absolute position
            let points = a.points.map((p) => {
                return { x: p.dx + cx, y: p.dy + d.y };
            });
            return interpolate$7(points);
        });
    }
    /**
     * Scene.
     */
    /**
     * Select or create a sceneGroup and build/update its nodes and edges.
     *
     * Structure Pattern:
     *
     * <g class='scene'>
     *   <g class='core'>
     *     <g class='edges'>
     *       ... stuff from tf.graph.scene.edges.build ...
     *     </g>
     *     <g class='nodes'>
     *       ... stuff from tf.graph.scene.nodes.build ...
     *     </g>
     *   </g>
     *   <g class='in-extract'>
     *     <g class='nodes'>
     *       ... stuff from tf.graph.scene.nodes.build ...
     *     </g>
     *   </g>
     *   <g class='out-extract'>
     *     <g class='nodes'>
     *       ... stuff from tf.graph.scene.nodes.build ...
     *     </g>
     *   </g>
     * </g>
     *
     * @param container D3 selection of the parent.
     * @param renderNode render node of a metanode or series node.
     * @param sceneElement <tf-graph-scene> polymer element.
     * @param sceneClass class attribute of the scene (default='scene').
     */
    function buildGroupForScene(container, renderNode, sceneElement, sceneClass) {
        sceneClass = sceneClass || Class$1.Scene.GROUP;
        let isNewSceneGroup = selectChild(container, 'g', sceneClass).empty();
        let sceneGroup = selectOrCreateChild(container, 'g', sceneClass);
        // core
        let coreGroup = selectOrCreateChild(sceneGroup, 'g', Class$1.Scene.CORE);
        let coreNodes = lodash.reduce(renderNode.coreGraph.nodes(), (nodes, name) => {
            let node = renderNode.coreGraph.node(name);
            if (!node.excluded) {
                nodes.push(node);
            }
            return nodes;
        }, []);
        if (renderNode.node.type === NodeType.SERIES) {
            // For series, we want the first item on top, so reverse the array so
            // the first item in the series becomes last item in the top, and thus
            // is rendered on the top.
            coreNodes.reverse();
        }
        // Create the layer of edges for this scene (paths).
        buildGroup(coreGroup, renderNode.coreGraph, sceneElement);
        // Create the layer of nodes for this scene (ellipses, rects etc).
        buildGroup$1(coreGroup, coreNodes, sceneElement);
        // In-extract
        if (renderNode.isolatedInExtract.length > 0) {
            let inExtractGroup = selectOrCreateChild(sceneGroup, 'g', Class$1.Scene.INEXTRACT);
            buildGroup$1(inExtractGroup, renderNode.isolatedInExtract, sceneElement);
        }
        else {
            selectChild(sceneGroup, 'g', Class$1.Scene.INEXTRACT).remove();
        }
        // Out-extract
        if (renderNode.isolatedOutExtract.length > 0) {
            let outExtractGroup = selectOrCreateChild(sceneGroup, 'g', Class$1.Scene.OUTEXTRACT);
            buildGroup$1(outExtractGroup, renderNode.isolatedOutExtract, sceneElement);
        }
        else {
            selectChild(sceneGroup, 'g', Class$1.Scene.OUTEXTRACT).remove();
        }
        // Library functions
        if (renderNode.libraryFunctionsExtract.length > 0) {
            let outExtractGroup = selectOrCreateChild(sceneGroup, 'g', Class$1.Scene.FUNCTION_LIBRARY);
            buildGroup$1(outExtractGroup, renderNode.libraryFunctionsExtract, sceneElement);
        }
        else {
            selectChild(sceneGroup, 'g', Class$1.Scene.FUNCTION_LIBRARY).remove();
        }
        position$1(sceneGroup, renderNode);
        // Fade in the scene group if it didn't already exist.
        if (isNewSceneGroup) {
            sceneGroup.attr('opacity', 0).transition().attr('opacity', 1);
        }
        return sceneGroup;
    }

    var tf_graph_scene_node = /*#__PURE__*/Object.freeze({
        __proto__: null,
        buildGroup: buildGroup$1,
        getContextMenu: getContextMenu,
        canBeInSeries: canBeInSeries,
        getSeriesName: getSeriesName,
        getGroupSettingLabel: getGroupSettingLabel,
        enforceLabelWidth: enforceLabelWidth,
        buildShape: buildShape,
        nodeClass: nodeClass,
        get ColorBy () { return ColorBy; },
        removeGradientDefinitions: removeGradientDefinitions,
        getFillForNode: getFillForNode,
        stylize: stylize$1,
        getStrokeForFill: getStrokeForFill,
        updateInputTrace: updateInputTrace,
        getVisibleParent: getVisibleParent,
        buildGroupForAnnotation: buildGroupForAnnotation,
        buildGroupForScene: buildGroupForScene
    });

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    let TfNodeIcon = class TfNodeIcon extends LegacyElementMixin(PolymerElement) {
        constructor() {
            super(...arguments);
            /**
             * Node to represent with an icon. Optional, but if specified, its
             * properties override those defined in the type, vertical, const and
             * summary properties.
             * This property is a tf.graph.Node.
             */
            this.node = null;
            /**
             * Render node information associated with this node. Optional. If
             * specified, this is only used when computing the fill of the icon
             * element.
             * This property is a tf.graph.render.RenderNodeInfo.
             */
            this.renderInfo = null;
            /**
             * String indicating the type of coloring to use for this node, used
             * only for determining the fill.
             */
            this.colorBy = 'structural';
            /**
             * Function used by structural coloring algorithm to determine which
             * color to use based on the template ID of the node. Optional.
             */
            this.templateIndex = null;
            /** Type of node to draw (ignored if node is set). */
            this.type = null;
            /** Direction for series (ignored for other types). */
            this.vertical = false;
            /** Whether the op is Const (ignored for non-ops). */
            this.const = false;
            /** Whether the op is a Summary (ignored for non-ops). */
            this.summary = false;
            /**
             * Fill for the icon, optional. If fill is specified and node is not
             * specified, then this value will override the default for the
             * element. However, if node is specified, this value will be ignored.
             */
            this.fill = null;
            /** Height of the SVG element in pixels, used for scaling. */
            this.height = 20;
        }
        /**
         * Returns fill value based on node and configuration. If any of those
         * configurations or node value missing, it returns `fill` property.
         * Note that if this evaluates to null, it will be chosen based on
         * the type of the node.
         */
        _computeFillOverride(inputNode, inputRenderInfo, inputColorBy, inputTemplateIndex, inputFill) {
            if (inputNode && inputRenderInfo && inputColorBy && inputTemplateIndex) {
                var ns = tf_graph_scene_node;
                var colorBy = ns.ColorBy[inputColorBy.toUpperCase()];
                return ns.getFillForNode(inputTemplateIndex, colorBy, inputRenderInfo, false);
            }
            return inputFill;
        }
        _getStrokeOverride(fillOverride) {
            return fillOverride
                ? getStrokeForFill(fillOverride)
                : null;
        }
        /**
         * Returns graph-icon type from input, type, and summary.
         */
        _getType(inputNode, isSummary, isConst, inputType) {
            const { GraphIconType } = tf_graph_icon;
            if (inputNode) {
                switch (inputNode.type) {
                    case NodeType.OP: {
                        const opName = inputNode.op;
                        // TODO(tensorboarad-team): `op` should have a predictable type.
                        // Remove the type check.
                        if (typeof opName !== 'string')
                            return GraphIconType.OP;
                        if (opName === 'Const' || isConst)
                            return GraphIconType.CONST;
                        if (opName.endsWith('Summary') || isSummary) {
                            return GraphIconType.SUMMARY;
                        }
                        return GraphIconType.OP;
                    }
                    case NodeType.META:
                        return GraphIconType.META;
                    case NodeType.SERIES:
                        return GraphIconType.SERIES;
                }
            }
            return inputType;
        }
        /**
         * Test whether the specified node should be represented as a vertical
         * series. Defaults to the value of the vertical property if node is
         * not specified.
         */
        _isVertical(inputNode, inputVertical) {
            if (inputNode) {
                return inputNode.hasNonControlEdges;
            }
            return !!inputVertical;
        }
        _getFaded(itemRenderInfo) {
            return itemRenderInfo && itemRenderInfo.isFadedOut;
        }
        _onFillOverrideChanged(newFill, oldFill) {
            const { node, renderInfo, colorBy, templateIndex } = this;
            const ns = tf_graph_scene_node;
            if (newFill !== oldFill) {
                ns.removeGradientDefinitions(this.$.icon.getSvgDefinableElement());
            }
            if (node && renderInfo && colorBy && templateIndex) {
                const nsColorBy = ns.ColorBy[colorBy.toUpperCase()];
                ns.getFillForNode(templateIndex, nsColorBy, renderInfo, false, this.$.icon.getSvgDefinableElement());
            }
        }
    };
    TfNodeIcon.template = html `
    <style>
      tf-graph-icon {
        --tb-graph-faded: var(--tb-graph-faded);
      }
    </style>
    <tf-graph-icon
      id="icon"
      type="[[_getType(node, summary, const, type)]]"
      height="[[height]]"
      fill-override="[[_fillOverride]]"
      stroke-override="[[_getStrokeOverride(_fillOverride)]]"
      faded="[[_getFaded(renderInfo)]]"
      vertical="[[_isVertical(node, vertical)]]"
    ></tf-graph-icon>
  `;
    __decorate([
        property({
            type: Object,
        }),
        __metadata("design:type", Object)
    ], TfNodeIcon.prototype, "node", void 0);
    __decorate([
        property({
            type: Object,
        }),
        __metadata("design:type", Object)
    ], TfNodeIcon.prototype, "renderInfo", void 0);
    __decorate([
        property({
            type: Object,
        }),
        __metadata("design:type", Object)
    ], TfNodeIcon.prototype, "colorBy", void 0);
    __decorate([
        property({
            type: Object,
        }),
        __metadata("design:type", Object)
    ], TfNodeIcon.prototype, "templateIndex", void 0);
    __decorate([
        property({
            type: String,
        }),
        __metadata("design:type", String)
    ], TfNodeIcon.prototype, "type", void 0);
    __decorate([
        property({
            type: Boolean,
        }),
        __metadata("design:type", Boolean)
    ], TfNodeIcon.prototype, "vertical", void 0);
    __decorate([
        property({
            type: Boolean,
        }),
        __metadata("design:type", Boolean)
    ], TfNodeIcon.prototype, "const", void 0);
    __decorate([
        property({
            type: Boolean,
        }),
        __metadata("design:type", Boolean)
    ], TfNodeIcon.prototype, "summary", void 0);
    __decorate([
        property({
            type: String,
        }),
        __metadata("design:type", String)
    ], TfNodeIcon.prototype, "fill", void 0);
    __decorate([
        property({
            type: Number,
        }),
        __metadata("design:type", Number)
    ], TfNodeIcon.prototype, "height", void 0);
    __decorate([
        property({
            type: String,
            computed: '_computeFillOverride(node, renderInfo, colorBy, templateIndex, fill)',
            observer: '_onFillOverrideChanged',
        }),
        __metadata("design:type", String)
    ], TfNodeIcon.prototype, "_fillOverride", void 0);
    TfNodeIcon = __decorate([
        customElement('tf-node-icon')
    ], TfNodeIcon);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    let TfGraphOpCompatListItem = class TfGraphOpCompatListItem extends LegacyElementMixin(PolymerElement) {
        _itemTypeChanged() {
            if (this.itemType !== 'subnode') {
                this.$['list-item'].classList.add('clickable');
            }
            else {
                this.$['list-item'].classList.remove('clickable');
            }
        }
        _nodeListener(event) {
            // fire node.click/mouseover/mouseout
            this.fire('node-list-item-' + event.type, {
                nodeName: this.name,
                type: this.itemType,
            });
        }
        _fadedClass(itemRenderInfo) {
            return itemRenderInfo && itemRenderInfo.isFadedOut ? 'faded' : '';
        }
    };
    TfGraphOpCompatListItem.template = html `
    <style>
      #list-item {
        width: 100%;
        color: #565656;
        font-size: 11pt;
        font-weight: 400;
        position: relative;
        display: inline-block;
      }

      #list-item:hover {
        background-color: var(--google-yellow-100);
      }

      .clickable {
        cursor: pointer;
      }

      #list-item span {
        margin-left: 40px;
      }

      #list-item.excluded span {
        color: #999;
      }

      #list-item span.edge-label {
        float: right;
        font-size: 10px;
        margin-left: 3px;
        margin-right: 5px;
      }

      .node-icon {
        position: absolute;
        top: 1px;
        left: 2px;
      }

      .faded span {
        color: var(--tb-graph-faded);
      }
    </style>

    <div
      id="list-item"
      on-mouseover="_nodeListener"
      on-mouseout="_nodeListener"
      on-click="_nodeListener"
    >
      <div class$="{{_fadedClass(itemRenderInfo)}}">
        <tf-node-icon
          class="node-icon"
          height="12"
          color-by="[[colorBy]]"
          color-by-params="[[colorByParams]]"
          node="[[itemNode]]"
          render-info="[[itemRenderInfo]]"
          template-index="[[templateIndex]]"
        >
        </tf-node-icon>
        <span title$="[[name]]">[[name]]</span>
      </div>
    </div>
  `;
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraphOpCompatListItem.prototype, "cardNode", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraphOpCompatListItem.prototype, "itemNode", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfGraphOpCompatListItem.prototype, "edgeLabel", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraphOpCompatListItem.prototype, "itemRenderInfo", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfGraphOpCompatListItem.prototype, "name", void 0);
    __decorate([
        property({
            type: String,
            observer: '_itemTypeChanged',
        }),
        __metadata("design:type", String)
    ], TfGraphOpCompatListItem.prototype, "itemType", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfGraphOpCompatListItem.prototype, "colorBy", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraphOpCompatListItem.prototype, "colorByParams", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraphOpCompatListItem.prototype, "templateIndex", void 0);
    TfGraphOpCompatListItem = __decorate([
        customElement('tf-graph-op-compat-list-item')
    ], TfGraphOpCompatListItem);

    function detect(h, verifyTemplate) {
        // In any particular subgraph, there are either
        // - leaf nodes (which do not have subgraph)
        // - metanode nodes - some of them have only one member (singular metanode)
        //                    and some have multiple members (non-singular metanode)
        // First, generate a nearest neighbor hash of metanode nodes.
        let nnGroups = clusterSimilarSubgraphs(h);
        // For each metanode, compare its subgraph (starting from shallower groups)
        // and assign template id.
        let templates = groupTemplateAndAssignId(nnGroups, verifyTemplate);
        // Sort the templates by minimum level in the graph at which they appear,
        // as this leads to optimal setting of the colors of each template for
        // maximum differentiation.
        return Object.keys(templates)
            .sort((key) => templates[key].level)
            .reduce((obj, key) => {
            obj[key] = templates[key];
            return obj;
        }, {});
    }
    /**
     * @return Unique string for a metanode based on depth, |V|, |E| and
     * op type histogram.
     */
    function getSignature(metanode) {
        // depth=<number> |V|=<number> |E|=<number>
        let props = lodash.map({
            depth: metanode.depth,
            '|V|': metanode.metagraph.nodes().length,
            '|E|': metanode.metagraph.edges().length,
        }, function (v, k) {
            return k + '=' + v;
        }).join(' ');
        // optype1=count1,optype2=count2
        let ops = lodash.map(metanode.opHistogram, function (count, op) {
            return op + '=' + count;
        }).join(',');
        return props + ' [ops] ' + ops;
    }
    /**
     * Generate a nearest neighbor hash of metanodes
     * based on depth, |V|, |E|, and opHistogram of their subgraph
     * (excluding leaf nodes and singular metanodes).
     * @param graph The graph
     * @return Array of pairs of [signature,
     *   Object with min level of the template and an Array of tf.graph.Group]
     *   sort by ascending order of minimum depth at which metanode appears.
     */
    function clusterSimilarSubgraphs(h) {
        /** a dict from metanode.signature() => Array of tf.graph.Groups */
        const map = h.getNodeMap();
        let hashDict = Object.keys(map).reduce((reduced, name) => {
            const node = map[name];
            if (node.type !== NodeType.META) {
                return reduced;
            }
            let levelOfMetaNode = name.split('/').length - 1;
            let signature = getSignature(node);
            let templateInfo = reduced[signature] || {
                nodes: [],
                level: levelOfMetaNode,
            };
            reduced[signature] = templateInfo;
            templateInfo.nodes.push(node);
            if (templateInfo.level > levelOfMetaNode) {
                templateInfo.level = levelOfMetaNode;
            }
            return reduced;
        }, {});
        return Object.keys(hashDict)
            .map((key) => [key, hashDict[key]])
            .filter(([_, subGraph]) => {
            const { nodes } = subGraph;
            if (nodes.length > 1) {
                // There is more than 1 node with this template. It is worth assigning
                // a unique color to this template.
                return true;
            }
            // If there is only 1 node with this template, only make a template for
            // it if it represents a function. In that case, the graph explorer may
            // add more nodes with the template later.
            const node = nodes[0];
            return (node.type === NodeType.META && node.associatedFunction);
        })
            .sort(([_, subGraph]) => {
            // sort by depth
            // (all members in the same nnGroup has equal depth)
            return subGraph.nodes[0].depth;
        });
    }
    function groupTemplateAndAssignId(nnGroups, verifyTemplate) {
        // For each metanode, compare its subgraph (starting from shallower groups)
        // and assign template id.
        let result = {};
        return lodash.reduce(nnGroups, function (templates, nnGroupPair) {
            let signature = nnGroupPair[0], nnGroup = nnGroupPair[1].nodes, clusters = [];
            nnGroup.forEach(function (metanode) {
                // check with each existing cluster
                for (let i = 0; i < clusters.length; i++) {
                    let similar = !verifyTemplate ||
                        isSimilarSubgraph(clusters[i].metanode.metagraph, metanode.metagraph);
                    // if similar, just add this metanode to the cluster
                    if (similar) {
                        // get template from the first one
                        metanode.templateId = clusters[i].metanode.templateId;
                        clusters[i].members.push(metanode.name);
                        return;
                    }
                }
                // otherwise create a new cluster with id 'signature [count] '
                metanode.templateId = signature + '[' + clusters.length + ']';
                clusters.push({
                    metanode: metanode,
                    members: [metanode.name],
                });
            });
            clusters.forEach(function (c) {
                templates[c.metanode.templateId] = {
                    level: nnGroupPair[1].level,
                    nodes: c.members,
                };
            });
            return templates;
        }, result);
    }
    function sortNodes(names, graph, prefix) {
        return lodash.sortBy(names, [
            (name) => graph.node(name).op,
            (name) => graph.node(name).templateId,
            (name) => graph.neighbors(name).length,
            (name) => graph.predecessors(name).length,
            (name) => graph.successors(name).length,
            (name) => name.substr(prefix.length),
        ]);
    }
    function isSimilarSubgraph(g1, g2) {
        if (!hasSimilarDegreeSequence(g1, g2)) {
            return false;
        }
        // if we want to skip, just return true here.
        // return true;
        // Verify sequence by running DFS
        let g1prefix = g1.graph().name;
        let g2prefix = g2.graph().name;
        let visited1 = {};
        let visited2 = {};
        let stack = [];
        /**
         * push sources or successors into the stack
         * if the visiting pattern has been similar.
         */
        function stackPushIfNotDifferent(n1, n2) {
            let sub1 = n1.substr(g1prefix.length), sub2 = n2.substr(g2prefix.length);
            /* tslint:disable */
            if (visited1[sub1] ^ visited2[sub2]) {
                console.warn('different visit pattern', '[' + g1prefix + ']', sub1, '[' + g2prefix + ']', sub2);
                return true;
            }
            /* tslint:enable */
            if (!visited1[sub1]) {
                // implied && !visited2[sub2]
                visited1[sub1] = visited2[sub2] = true;
                stack.push({ n1: n1, n2: n2 });
            }
            return false;
        }
        // check if have same # of sources then sort and push
        let sources1 = g1.sources();
        let sources2 = g2.sources();
        if (sources1.length !== sources2.length) {
            /* tslint:disable */
            console.log('different source length');
            /* tslint:enable */
            return false;
        }
        sources1 = sortNodes(sources1, g1, g1prefix);
        sources2 = sortNodes(sources2, g2, g2prefix);
        for (let i = 0; i < sources1.length; i++) {
            let different = stackPushIfNotDifferent(sources1[i], sources2[i]);
            if (different) {
                return false;
            }
        }
        while (stack.length > 0) {
            let cur = stack.pop();
            // check node
            let similar = isSimilarNode(g1.node(cur.n1), g2.node(cur.n2));
            if (!similar) {
                return false;
            }
            // check if have same # of successors then sort and push
            let succ1 = g1.successors(cur.n1), succ2 = g2.successors(cur.n2);
            if (succ1.length !== succ2.length) {
                /* tslint:disable */
                console.log('# of successors mismatch', succ1, succ2);
                /* tslint:enable */
                return false;
            }
            succ1 = sortNodes(succ1, g1, g1prefix);
            succ2 = sortNodes(succ2, g2, g2prefix);
            for (let j = 0; j < succ1.length; j++) {
                let different = stackPushIfNotDifferent(succ1[j], succ2[j]);
                if (different) {
                    return false;
                }
            }
        }
        return true;
    }
    /**
     * Returns if two nodes have identical structure.
     */
    function isSimilarNode(n1, n2) {
        if (n1.type === NodeType.META) {
            // compare metanode
            let metanode1 = n1;
            let metanode2 = n2;
            return (metanode1.templateId &&
                metanode2.templateId &&
                metanode1.templateId === metanode2.templateId);
        }
        else if (n1.type === NodeType.OP && n2.type === NodeType.OP) {
            // compare leaf node
            return n1.op === n2.op;
        }
        else if (n1.type === NodeType.SERIES && n2.type === NodeType.SERIES) {
            // compare series node sizes and operations
            // (only need to check one op as all op nodes are identical in series)
            let sn1 = n1;
            let sn2 = n2;
            let seriesnode1Count = sn1.metagraph.nodeCount();
            return (seriesnode1Count === sn2.metagraph.nodeCount() &&
                (seriesnode1Count === 0 ||
                    sn1.metagraph.node(sn1.metagraph.nodes()[0]).op ===
                        sn2.metagraph.node(sn2.metagraph.nodes()[0]).op));
        }
        return false;
    }

    /* Copyright 2015 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the 'License');
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an 'AS IS' BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * Class for the Graph Hierarchy for TensorFlow graph.
     */
    class HierarchyImpl {
        /**
         * Constructs a hierarchy.
         * @param graphOptions Options passed to dagre for creating the graph. Note
         *   that the `compound` argument will be overridden to true.
         */
        constructor(graphOptions) {
            this.hasShapeInfo = false;
            this.maxMetaEdgeSize = 1;
            this.graphOptions = graphOptions || {};
            this.graphOptions.compound = true;
            this.root = createMetanode(ROOT_NAME, this.graphOptions);
            this.libraryFunctions = {};
            this.templates = null;
            this.devices = null;
            this.xlaClusters = null;
            /**
             * @type {Object} Dictionary object that maps node name to the node
             * (could be op-node, metanode, or series-node)
             */
            this.index = {};
            this.index[ROOT_NAME] = this.root;
            this.orderings = {};
        }
        getNodeMap() {
            return this.index;
        }
        node(name) {
            return this.index[name];
        }
        setNode(name, node) {
            this.index[name] = node;
        }
        /**
         * Given the name of a node in this hierarchy, get its bridgegraph, creating
         * it on the fly if necessary. If the node is not a GroupNode, then this
         * method returns null. If the provided name does not map to a node in the
         * hierarchy, an error will be thrown.
         */
        getBridgegraph(nodeName) {
            let node = this.index[nodeName];
            if (!node) {
                throw Error('Could not find node in hierarchy: ' + nodeName);
            }
            if (!('metagraph' in node)) {
                return null;
            }
            let groupNode = node;
            if (groupNode.bridgegraph) {
                return groupNode.bridgegraph;
            }
            let bridgegraph = (groupNode.bridgegraph = createGraph('BRIDGEGRAPH', GraphType.BRIDGE, this.graphOptions));
            if (!node.parentNode || !('metagraph' in node.parentNode)) {
                return bridgegraph;
            }
            let parentNode = node.parentNode;
            let parentMetagraph = parentNode.metagraph;
            let parentBridgegraph = this.getBridgegraph(parentNode.name);
            // For each of the parent node's two Metaedge containing graphs, process
            // each Metaedge involving this node.
            lodash.each([parentMetagraph, parentBridgegraph], (parentGraph) => {
                parentGraph
                    .edges()
                    .filter((e) => e.v === nodeName || e.w === nodeName)
                    .forEach((parentEdgeObj) => {
                    let inbound = parentEdgeObj.w === nodeName;
                    let parentMetaedge = parentGraph.edge(parentEdgeObj);
                    // The parent's Metaedge represents some number of underlying
                    // BaseEdges from the original full graph. For each of those, we need
                    // to determine which immediate child is involved and make sure
                    // there's a Metaedge in the bridgegraph that covers it.
                    lodash.each(parentMetaedge.baseEdgeList, (baseEdge) => {
                        // Based on the direction, figure out which is the descendant node
                        // and which is the 'other' node (sibling of parent or ancestor).
                        let [descendantName, otherName] = inbound
                            ? [baseEdge.w, parentEdgeObj.v]
                            : [baseEdge.v, parentEdgeObj.w];
                        // Determine the immediate child containing this descendant node.
                        let childName = this.getChildName(nodeName, descendantName);
                        // Look for an existing Metaedge in the bridgegraph (or create a
                        // new one) that covers the relationship between child and other.
                        let bridgeEdgeObj = {
                            v: inbound ? otherName : childName,
                            w: inbound ? childName : otherName,
                        };
                        let bridgeMetaedge = bridgegraph.edge(bridgeEdgeObj);
                        if (!bridgeMetaedge) {
                            bridgeMetaedge = createMetaedge(bridgeEdgeObj.v, bridgeEdgeObj.w);
                            bridgeMetaedge.inbound = inbound;
                            bridgegraph.setEdge(bridgeEdgeObj.v, bridgeEdgeObj.w, bridgeMetaedge);
                        }
                        // Copy the BaseEdge from the parent's Metaedge into this
                        // bridgegraph Metaedge.
                        bridgeMetaedge.addBaseEdge(baseEdge, this);
                    });
                });
            });
            return bridgegraph;
        }
        /**
         * Utility function for determining the name of the immediate child under a
         * node for a given descendant path. If the descendant corresponds to no
         * immediate child, an error is thrown.
         */
        getChildName(nodeName, descendantName) {
            // Walk up the hierarchy from the descendant to find the child.
            let currentNode = this.index[descendantName];
            while (currentNode) {
                if (currentNode.parentNode && currentNode.parentNode.name === nodeName) {
                    return currentNode.name;
                }
                currentNode = currentNode.parentNode;
            }
            throw Error('Could not find immediate child for descendant: ' + descendantName);
        }
        /** Given the name of a node, return its incoming metaedges. */
        getPredecessors(nodeName) {
            let node = this.index[nodeName];
            if (!node) {
                throw Error('Could not find node with name: ' + nodeName);
            }
            let predecessors = this.getOneWayEdges(node, true);
            // Add embedded predecessors, such as constants.
            if (!node.isGroupNode) {
                lodash.each(node.inEmbeddings, (embeddedNode) => {
                    lodash.each(node.inputs, (input) => {
                        if (input.name === embeddedNode.name) {
                            // Make a new metaedge holding the edge between the
                            // node and the in-embedding.
                            let metaedge = new MetaedgeImpl(embeddedNode.name, nodeName);
                            metaedge.addBaseEdge({
                                isControlDependency: input.isControlDependency,
                                outputTensorKey: input.outputTensorKey,
                                isReferenceEdge: false,
                                v: embeddedNode.name,
                                w: nodeName,
                            }, this);
                            predecessors.regular.push(metaedge);
                        }
                    });
                });
            }
            return predecessors;
        }
        /**
         * Given the name of a node, return its outgoing metaedges.
         *
         * This is the inverse of getPredecessors(). See that method's documentation
         * for an in-depth example.
         */
        getSuccessors(nodeName) {
            let node = this.index[nodeName];
            if (!node) {
                throw Error('Could not find node with name: ' + nodeName);
            }
            let successors = this.getOneWayEdges(node, false);
            // Add embedded successors, such as summaries.
            if (!node.isGroupNode) {
                lodash.each(node.outEmbeddings, (embeddedNode) => {
                    lodash.each(embeddedNode.inputs, (input) => {
                        if (input.name === nodeName) {
                            // Make a new metaedge holding the edge between the
                            // node and the out-embedding.
                            let metaedge = new MetaedgeImpl(nodeName, embeddedNode.name);
                            metaedge.addBaseEdge({
                                isControlDependency: input.isControlDependency,
                                outputTensorKey: input.outputTensorKey,
                                isReferenceEdge: false,
                                v: nodeName,
                                w: embeddedNode.name,
                            }, this);
                            successors.regular.push(metaedge);
                        }
                    });
                });
            }
            return successors;
        }
        /** Helper method for getPredecessors and getSuccessors */
        getOneWayEdges(node, inEdges) {
            let edges = { control: [], regular: [] };
            // A node with no parent cannot have any edges.
            if (!node.parentNode || !node.parentNode.isGroupNode) {
                return edges;
            }
            let parentNode = node.parentNode;
            let metagraph = parentNode.metagraph;
            let bridgegraph = this.getBridgegraph(parentNode.name);
            findEdgeTargetsInGraph(metagraph, node, inEdges, edges);
            findEdgeTargetsInGraph(bridgegraph, node, inEdges, edges);
            return edges;
        }
        /**
         * For a given GroupNode, get or calculate an object which describes a
         * topological ordering of child nodes within that GroupNode's metagraph.
         *
         * This ordering is used when rendering bridge control edges which are
         * sometimes backwards relative to the dataflow.
         *
         * For example, say we have a graph with two edges A->B and A->C, and we're
         * interested in the ordering under ROOT. In this case, any of the following
         * would be legitimate return values:
         *
         *  - { 'A': 0, 'B': 1, 'C': 2 } -- most likely
         *  - { 'A': 0, 'B': 2, 'C': 1 } -- less likely
         *  - { 'A': 12, 'B': 100, 'C': 99 } -- unlikely, but still OK
         *
         * The algorithm does not guarantee that all numbers from 0-N (where N is
         * the number of nodes) appear exactly once. Rather it guarantees that if
         * there is a path between two nodes, the earlier one will have a lower
         * number in the ordering hash.
         *
         * When generating the ordering, we ignore control Metaedges (those which
         * represent only BaseEdges that have isControlDependency set to true).
         *
         * If there is no node with the specified name, an error is thrown. If the
         * node with the specified name is not a group node, null is returned.
         */
        getTopologicalOrdering(nodeName) {
            let node = this.index[nodeName];
            if (!node) {
                throw Error('Could not find node with name: ' + nodeName);
            }
            if (!node.isGroupNode) {
                return null;
            }
            if (nodeName in this.orderings) {
                return this.orderings[nodeName];
            }
            // Mapping of a child node names to lists of their successors.
            let successors = {};
            // Set of node names which have appeared as a destination.
            let destinations = {};
            let metagraph = node.metagraph;
            lodash.each(metagraph.edges(), (e) => {
                if (!metagraph.edge(e).numRegularEdges) {
                    return; // Skip control edges.
                }
                // Keep track of successors and destinations.
                if (!(e.v in successors)) {
                    successors[e.v] = [];
                }
                successors[e.v].push(e.w);
                destinations[e.w] = true;
            });
            // Seed the queue with true sources (those that are not destinations).
            let queue = lodash.difference(lodash.keys(successors), lodash.keys(destinations));
            // Produce an ordering by traversing the graph breadth first.
            let ordering = (this.orderings[nodeName] = {});
            let index = 0;
            while (queue.length) {
                let childName = queue.shift();
                ordering[childName] = index++;
                lodash.each(successors[childName], (succName) => queue.push(succName));
                delete successors[childName]; // Prevent cycles from infinite looping.
            }
            return ordering;
        }
        /**
         * Returns a d3 Ordinal function that can be used to look up the index of
         * a node based on its template id.
         */
        getTemplateIndex() {
            let templateNames = keys(this.templates);
            let templateIndex = ordinal()
                .domain(templateNames)
                .range(sequence(0, templateNames.length));
            return (templateId) => templateIndex(templateId);
        }
    }
    /**
     * Internal utility function - given a graph (should be either a metagraph or a
     * bridgegraph) and a node which is known to be in that graph, determine
     * the other ends of edges that involve that node in the direction specified
     * by whether it's inbound.
     *
     * For example if you wanted to find the predecessors of a node, you'd call
     * this method for the parent's metagraph and bridgegraph, specifying inbound
     * as true (look at the source of inbound edges to the specified node).
     *
     * Discovered target names are appended to the targets array.
     */
    function findEdgeTargetsInGraph(graph, node, inbound, targets) {
        let edges = inbound ? graph.inEdges(node.name) : graph.outEdges(node.name);
        lodash.each(edges, (e) => {
            let metaedge = graph.edge(e);
            let targetList = metaedge.numRegularEdges
                ? targets.regular
                : targets.control;
            targetList.push(metaedge);
        });
    }
    const DefaultHierarchyParams = {
        verifyTemplate: true,
        seriesNodeMinSize: 5,
        seriesMap: {},
        rankDirection: 'BT',
        useGeneralizedSeriesPatterns: false,
    };
    /**
     * @param graph The raw graph.
     * @param params Parameters used when building a hierarchy.
     */
    function build$1(graph, params, tracker) {
        let h = new HierarchyImpl({ rankdir: params.rankDirection });
        let seriesNames = {};
        return runAsyncTask('Adding nodes', 20, () => {
            // Get all the possible device and XLA cluster names.
            let deviceNames = {};
            let xlaClusterNames = {};
            lodash.each(graph.nodes, (node, nodeName) => {
                if (node.device) {
                    deviceNames[node.device] = true;
                }
                if (node.xlaCluster) {
                    xlaClusterNames[node.xlaCluster] = true;
                }
            });
            h.devices = lodash.keys(deviceNames);
            h.xlaClusters = lodash.keys(xlaClusterNames);
            addNodes(h, graph);
        }, tracker)
            .then(() => {
            return runAsyncTask('Detect series', 20, () => {
                if (params.seriesNodeMinSize > 0) {
                    groupSeries(h.root, h, seriesNames, params.seriesNodeMinSize, params.seriesMap, params.useGeneralizedSeriesPatterns);
                }
            }, tracker);
        })
            .then(() => {
            return runAsyncTask('Adding edges', 30, () => {
                addEdges(h, graph);
            }, tracker);
        })
            .then(() => {
            return runAsyncTask('Finding similar subgraphs', 30, () => {
                h.templates = detect(h, params.verifyTemplate);
            }, tracker);
        })
            .then(() => {
            return h;
        });
    }
    function joinAndAggregateStats(h, stats) {
        // Get all the possible device and XLA cluster names.
        let deviceNames = {};
        let xlaClusterNames = {};
        lodash.each(h.root.leaves(), (nodeName) => {
            let leaf = h.node(nodeName);
            if (leaf.device != null) {
                deviceNames[leaf.device] = true;
            }
            if (leaf.xlaCluster != null) {
                xlaClusterNames[leaf.xlaCluster] = true;
            }
        });
        h.devices = lodash.keys(deviceNames);
        h.xlaClusters = lodash.keys(xlaClusterNames);
        // Reset stats for each group node.
        lodash.each(h.getNodeMap(), (node, nodeName) => {
            if (node.isGroupNode) {
                node.stats = new NodeStats(null);
                node.deviceHistogram = {};
            }
        });
        // Bubble-up the stats and device distribution from leaves to parents.
        lodash.each(h.root.leaves(), (nodeName) => {
            let leaf = h.node(nodeName);
            let node = leaf;
            while (node.parentNode != null) {
                if (leaf.device != null) {
                    let deviceHistogram = node.parentNode.deviceHistogram;
                    deviceHistogram[leaf.device] = (deviceHistogram[leaf.device] || 0) + 1;
                }
                if (leaf.xlaCluster != null) {
                    let xlaClusterHistogram = node.parentNode
                        .xlaClusterHistogram;
                    xlaClusterHistogram[leaf.xlaCluster] =
                        (xlaClusterHistogram[leaf.xlaCluster] || 0) + 1;
                }
                if (leaf.stats != null) {
                    node.parentNode.stats.combine(leaf.stats);
                }
                node = node.parentNode;
            }
        });
    }
    function getIncompatibleOps(hierarchy, hierarchyParams) {
        let nodes = [];
        let addedSeriesNodes = {};
        lodash.each(hierarchy.root.leaves(), (leaf) => {
            let node = hierarchy.node(leaf);
            if (node.type == NodeType.OP) {
                let opNode = node;
                if (!opNode.compatible) {
                    if (opNode.owningSeries) {
                        if (hierarchyParams &&
                            hierarchyParams.seriesMap[opNode.owningSeries] ===
                                SeriesGroupingType.UNGROUP) {
                            // For un-grouped series node, add each node individually
                            nodes.push(opNode);
                        }
                        else {
                            if (!addedSeriesNodes[opNode.owningSeries]) {
                                let series = hierarchy.node(opNode.owningSeries);
                                if (series) {
                                    addedSeriesNodes[opNode.owningSeries] = series;
                                    nodes.push(series);
                                }
                            }
                        }
                    }
                    else {
                        nodes.push(opNode);
                    }
                }
                // Check the embeddings for invalid operations
                lodash.each(opNode.inEmbeddings, (inNode) => {
                    if (!inNode.compatible) {
                        nodes.push(inNode);
                    }
                });
                lodash.each(opNode.outEmbeddings, (outNode) => {
                    if (!outNode.compatible) {
                        nodes.push(outNode);
                    }
                });
            }
        });
        return nodes;
    }
    /**
     * Creates the metanodes in the hierarchical graph and assigns parent-child
     * relationship between them. Also assigns relationships between library
     * functions and their usages throughout the graph.
     */
    function addNodes(h, graph) {
        // Maps the op of a node to names of nodes that have the op. Used to populate
        // the libraryFunctions field of the hierarchy.
        const opToNode = {};
        lodash.each(graph.nodes, (node, nodeName) => {
            let path = getHierarchicalPath(node.name);
            let parent = h.root;
            parent.depth = Math.max(path.length, parent.depth);
            // Track which nodes are associated with which ops.
            if (!opToNode[node.op]) {
                opToNode[node.op] = [];
            }
            opToNode[node.op].push(node);
            // Create parent metanodes for each depth. For example if the node name
            // is 'a/b/c', then create metanodes 'a' and 'a/b', where 'a/b' is a child
            // of a.
            for (let i = 0; i < path.length; i++) {
                parent.depth = Math.max(parent.depth, path.length - i);
                parent.cardinality += node.cardinality;
                parent.opHistogram[node.op] = (parent.opHistogram[node.op] || 0) + 1;
                if (node.device != null) {
                    parent.deviceHistogram[node.device] =
                        (parent.deviceHistogram[node.device] || 0) + 1;
                }
                if (node.xlaCluster != null) {
                    parent.xlaClusterHistogram[node.xlaCluster] =
                        (parent.xlaClusterHistogram[node.xlaCluster] || 0) + 1;
                }
                // Increment parents appropriate compatibility count
                if (node.compatible) {
                    parent.compatibilityHistogram.compatible =
                        (parent.compatibilityHistogram.compatible || 0) + 1;
                }
                else {
                    parent.compatibilityHistogram.incompatible =
                        (parent.compatibilityHistogram.incompatible || 0) + 1;
                }
                // Increment capability counts for in and out embeddings
                lodash.each(node.inEmbeddings, (inNode) => {
                    if (inNode.compatible) {
                        parent.compatibilityHistogram.compatible =
                            (parent.compatibilityHistogram.compatible || 0) + 1;
                    }
                    else {
                        parent.compatibilityHistogram.incompatible =
                            (parent.compatibilityHistogram.incompatible || 0) + 1;
                    }
                });
                lodash.each(node.outEmbeddings, (outNode) => {
                    if (outNode.compatible) {
                        parent.compatibilityHistogram.compatible =
                            (parent.compatibilityHistogram.compatible || 0) + 1;
                    }
                    else {
                        parent.compatibilityHistogram.incompatible =
                            (parent.compatibilityHistogram.incompatible || 0) + 1;
                    }
                });
                if (i === path.length - 1) {
                    break;
                }
                let name = path[i];
                let child = h.node(name);
                if (!child) {
                    child = createMetanode(name, h.graphOptions);
                    child.parentNode = parent;
                    h.setNode(name, child);
                    parent.metagraph.setNode(name, child);
                    if (name.indexOf(FUNCTION_LIBRARY_NODE_PREFIX) === 0 &&
                        parent.name === ROOT_NAME) {
                        // This metanode represents a function in the Library. We later copy
                        // its contents to dynamically inject function data into the graph
                        // when the subhierarchy of a metanode is built (upon its expansion).
                        const functionName = name.substring(FUNCTION_LIBRARY_NODE_PREFIX.length);
                        // For now, remember the metanode that represents the function with
                        // this name.
                        if (!opToNode[functionName]) {
                            opToNode[functionName] = [];
                        }
                        h.libraryFunctions[functionName] = {
                            node: child,
                            usages: opToNode[functionName],
                        };
                        child.associatedFunction = functionName;
                    }
                }
                parent = child;
            }
            // Assuming node name is 'a/b/c', assign the OpNode as a child of the
            // metanode 'a/b'.
            h.setNode(node.name, node);
            node.parentNode = parent;
            parent.metagraph.setNode(node.name, node);
            // Add each of the in-embeddings and out-embeddings in the hierarchy.
            lodash.each(node.inEmbeddings, function (embedding) {
                h.setNode(embedding.name, embedding);
                embedding.parentNode = node;
            });
            lodash.each(node.outEmbeddings, function (embedding) {
                h.setNode(embedding.name, embedding);
                embedding.parentNode = node;
            });
        });
    }
    /**
     * For each metanode in the hierarchical graph, this method adds:
     * the edges in the metagraph. These are edges between nodes
     * that share the same parent.
     */
    function addEdges(h, graph, seriesNames) {
        let nodeIndex = h.getNodeMap();
        // Ancestor paths for the source and destination nodes of an edge. These are
        // reused for each edge rather than allocating new ones. It's about 10% faster
        // than allocating new ones on each pass through the loop.
        let sourcePath = [];
        let destPath = [];
        // Insert the ancestor path for a node into the provided array, including the
        // node itself. Return the index of the last node inserted (always ROOT).
        let getPath = (node, path) => {
            let i = 0;
            while (node) {
                path[i++] = node.name;
                node = node.parentNode;
            }
            return i - 1;
        };
        lodash.each(graph.edges, (baseEdge) => {
            // Get the hierarchical paths for the source and destination of the edge.
            let sourceAncestorIndex = getPath(graph.nodes[baseEdge.v], sourcePath);
            let destAncestorIndex = getPath(graph.nodes[baseEdge.w], destPath);
            // If the hierarchical path cannot be found for either endpoint, then we
            // cannot create the edge. This happens for example when a node has a
            // control dependency on a summary node, which are embedded.
            if (sourceAncestorIndex === -1 || destAncestorIndex === -1) {
                return;
            }
            // Find the lowest shared ancestor between source and dest by looking for
            // the highest nodes that differ between their ancestor paths.
            while (sourcePath[sourceAncestorIndex] === destPath[destAncestorIndex]) {
                sourceAncestorIndex--;
                destAncestorIndex--;
                if (sourceAncestorIndex < 0 || destAncestorIndex < 0) {
                    // This would only occur if the two nodes were the same (a cycle in the
                    // graph), or if one endpoint was a strict ancestor of the other. The
                    // latter shouldn't happen because we rename nodes which are both
                    // metanodes and op nodes. E.g. 'A/B' becomes 'A/B/(B)'.
                    throw Error('No difference found between ancestor paths.');
                }
            }
            let sharedAncestorNode = (nodeIndex[sourcePath[sourceAncestorIndex + 1]]);
            let sourceAncestorName = sourcePath[sourceAncestorIndex];
            let destAncestorName = destPath[destAncestorIndex];
            // Find or create the Metaedge which should contain this BaseEdge inside
            // the shared ancestor.
            let metaedge = sharedAncestorNode.metagraph.edge(sourceAncestorName, destAncestorName);
            if (!metaedge) {
                metaedge = createMetaedge(sourceAncestorName, destAncestorName);
                sharedAncestorNode.metagraph.setEdge(sourceAncestorName, destAncestorName, metaedge);
            }
            if (!sharedAncestorNode.hasNonControlEdges &&
                !baseEdge.isControlDependency) {
                sharedAncestorNode.hasNonControlEdges = true;
            }
            metaedge.addBaseEdge(baseEdge, h);
        });
    }
    /**
     * Using the hierarchy template information, detect series in the provided
     * metanode.  For each detected series, create a new SeriesNode
     * and remove series members from the metanode's metagraph and move them to
     * the new series node's metagraph.
     *
     * @param metanode
     * @param hierarchy
     * @param seriesNames Map of node names to their series they are contained in.
     *     This should be provided empty and is populated by this method.
     * @param threshold If the series has this many nodes or more, then group them
     *     into a series.
     * @param map Map of series names to their series grouping type, if one has
     *     been set.
     * @param useGeneralizedSeriesPatterns Whether to use find patterns for series
     *     nodes using any parts of names of nodes. If false, only uses patterns
     *     discovered within numeric suffixes of nodes names.
     * @return A dictionary from node name to series node name that contains the
     *     node.
     */
    function groupSeries(metanode, hierarchy, seriesNames, threshold, map, useGeneralizedSeriesPatterns) {
        let metagraph = metanode.metagraph;
        lodash.each(metagraph.nodes(), (n) => {
            let child = metagraph.node(n);
            if (child.type === NodeType.META) {
                groupSeries(child, hierarchy, seriesNames, threshold, map, useGeneralizedSeriesPatterns);
            }
        });
        let clusters = clusterNodes(metagraph);
        const detectSeriesMethod = useGeneralizedSeriesPatterns
            ? detectSeriesAnywhereInNodeName
            : detectSeriesUsingNumericSuffixes;
        let seriesDict = detectSeriesMethod(clusters, metagraph, hierarchy.graphOptions);
        // Add each series node to the graph and add its grouped children to its own
        // metagraph.
        lodash.each(seriesDict, function (seriesNode, seriesName) {
            let nodeMemberNames = seriesNode.metagraph.nodes();
            lodash.each(nodeMemberNames, (n) => {
                let child = metagraph.node(n);
                if (!child.owningSeries) {
                    child.owningSeries = seriesName;
                }
            });
            // If the series contains less than the threshold number of nodes and
            // this series has not been adding to the series map, then set this
            // series to be shown ungrouped in the map.
            if (nodeMemberNames.length < threshold && !(seriesNode.name in map)) {
                map[seriesNode.name] = SeriesGroupingType.UNGROUP;
            }
            // If the series is in the map as ungrouped then do not group the series.
            if (seriesNode.name in map &&
                map[seriesNode.name] === SeriesGroupingType.UNGROUP) {
                return;
            }
            hierarchy.setNode(seriesName, seriesNode); // add to the index
            metagraph.setNode(seriesName, seriesNode);
            lodash.each(nodeMemberNames, (n) => {
                let child = metagraph.node(n);
                seriesNode.metagraph.setNode(n, child);
                seriesNode.parentNode = child.parentNode;
                seriesNode.cardinality++;
                if (child.device != null) {
                    seriesNode.deviceHistogram[child.device] =
                        (seriesNode.deviceHistogram[child.device] || 0) + 1;
                }
                if (child.xlaCluster != null) {
                    seriesNode.xlaClusterHistogram[child.xlaCluster] =
                        (seriesNode.xlaClusterHistogram[child.xlaCluster] || 0) + 1;
                }
                // Increment parents appropriate compatibility count
                if (child.compatible) {
                    seriesNode.compatibilityHistogram.compatible =
                        (seriesNode.compatibilityHistogram.compatible || 0) + 1;
                }
                else {
                    seriesNode.compatibilityHistogram.incompatible =
                        (seriesNode.compatibilityHistogram.incompatible || 0) + 1;
                }
                // Increment capability counts for in and out embeddings
                lodash.each(child.inEmbeddings, (inNode) => {
                    if (inNode.compatible) {
                        seriesNode.compatibilityHistogram.compatible =
                            (seriesNode.compatibilityHistogram.compatible || 0) + 1;
                    }
                    else {
                        seriesNode.compatibilityHistogram.incompatible =
                            (seriesNode.compatibilityHistogram.incompatible || 0) + 1;
                    }
                });
                lodash.each(child.outEmbeddings, (outNode) => {
                    if (outNode.compatible) {
                        seriesNode.compatibilityHistogram.compatible =
                            (seriesNode.compatibilityHistogram.compatible || 0) + 1;
                    }
                    else {
                        seriesNode.compatibilityHistogram.incompatible =
                            (seriesNode.compatibilityHistogram.incompatible || 0) + 1;
                    }
                });
                child.parentNode = seriesNode;
                seriesNames[n] = seriesName;
                // Remove now-grouped node from its original parent's metagraph.
                metagraph.removeNode(n);
            });
        });
    }
    /** cluster op-nodes with similar op */
    function clusterNodes(metagraph) {
        let result = {};
        return lodash.reduce(metagraph.nodes(), (clusters, n) => {
            let child = metagraph.node(n);
            if (child.type === NodeType.META) {
                // skip metanodes
                return clusters;
            }
            let template = child.op;
            if (template) {
                clusters[template] = clusters[template] || [];
                clusters[template].push(child.name);
            }
            return clusters;
        }, result);
    }
    /**
     * For each cluster of op-nodes based op type, try to detect groupings.
     * Infer series name using by trying to find pattern '<number>' towards the end
     * of node names.
     *
     * @param clusters Dictionary output from clusterNodes().
     * @param metagraph
     * @return A dictionary from series name => seriesNode
     */
    function detectSeriesUsingNumericSuffixes(clusters, metagraph, graphOptions) {
        let seriesDict = {};
        lodash.each(clusters, function (members, clusterId) {
            if (members.length <= 1) {
                return;
            } // isolated clusters can't make series
            /** @type {Object}  A dictionary mapping seriesName to seriesInfoArray,
             * which is an array that contains objects with name, id, prefix, suffix,
             * and parent properties.
             */
            let candidatesDict = {};
            // Group all nodes that have the same name, with the exception of a
            // number at the end of the name after an underscore, which is allowed to
            // vary.
            lodash.each(members, function (name) {
                let isGroup = name.charAt(name.length - 1) === '*';
                let namepath = name.split('/');
                let leaf = namepath[namepath.length - 1];
                let parent = namepath.slice(0, namepath.length - 1).join('/');
                let matches = leaf.match(/^(\D*)_(\d+)$/);
                let prefix;
                let id;
                let suffix = '';
                if (matches) {
                    // if found '<number>' in the name, assign id.
                    prefix = matches[1]; // the front non-numeric characters
                    id = matches[2]; // the digits
                }
                else {
                    // for node without '_<number>', make them zero-th items.
                    prefix = isGroup ? leaf.substr(0, leaf.length - 1) : leaf;
                    id = 0;
                    suffix = isGroup ? '*' : '';
                }
                let seriesName = getSeriesNodeName(prefix, suffix, parent);
                candidatesDict[seriesName] = candidatesDict[seriesName] || [];
                let seriesNode = createSeriesNode(prefix, suffix, parent, +id, name, graphOptions);
                candidatesDict[seriesName].push(seriesNode);
            });
            // In each group of nodes, group nodes in bunches that have monotonically
            // increasing numbers in their names.  Each of these bunches is a series.
            lodash.each(candidatesDict, function (seriesInfoArray, seriesName) {
                if (seriesInfoArray.length < 2) {
                    return;
                }
                seriesInfoArray.sort(function (a, b) {
                    return +a.clusterId - +b.clusterId;
                });
                // Loop through the nodes sorted by its detected series number, grouping
                // all nodes with monotonically-increasing series numbers.
                let seriesNodes = [seriesInfoArray[0]];
                for (let index = 1; index < seriesInfoArray.length; index++) {
                    let nextNode = seriesInfoArray[index];
                    if (nextNode.clusterId ===
                        seriesNodes[seriesNodes.length - 1].clusterId + 1) {
                        seriesNodes.push(nextNode);
                        continue;
                    }
                    addSeriesToDict(seriesNodes, seriesDict, +clusterId, metagraph, graphOptions);
                    seriesNodes = [nextNode];
                }
                addSeriesToDict(seriesNodes, seriesDict, +clusterId, metagraph, graphOptions);
            });
        });
        return seriesDict;
    }
    /**
     * For each cluster of op-nodes based op type, try to detect groupings.
     * Infer series name using by trying to find a pattern of numbers
     * anywhere within node names.
     *
     * @param clusters Dictionary output from clusterNodes().
     * @param metagraph
     * @return A dictionary from series name => seriesNode
     */
    function detectSeriesAnywhereInNodeName(clusters, metagraph, graphOptions) {
        let seriesDict = {};
        lodash.each(clusters, function (members, clusterId) {
            if (members.length <= 1) {
                return;
            } // isolated clusters can't make series
            /**
             * @type {Object}  A dictionary mapping a series name to a SeriesNode.
             */
            let forwardDict = {};
            /**
             * @type {Object}  A dictionary mapping member name to an array of series
             * names this member could potentially be grouped under and the
             * corresponding ids.
             */
            let reverseDict = {};
            // Group all nodes that have the same name, with the exception of a
            // number at the end of the name after an underscore, which is allowed to
            // vary.
            lodash.each(members, function (name) {
                let isGroup = name.charAt(name.length - 1) === '*';
                let namepath = name.split('/');
                let leaf = namepath[namepath.length - 1];
                let parent = namepath.slice(0, namepath.length - 1).join('/');
                const numRegex = /(\d+)/g;
                let matchResult;
                let prefix;
                let id;
                let suffix;
                let seriesName;
                let matched = 0;
                // Scan over the entire leaf name and match any possible numbers,
                // and put the results into corresponding dictionaries.
                while ((matchResult = numRegex.exec(leaf))) {
                    ++matched;
                    prefix = leaf.slice(0, matchResult.index);
                    id = matchResult[0];
                    suffix = leaf.slice(matchResult.index + matchResult[0].length);
                    seriesName = getSeriesNodeName(prefix, suffix, parent);
                    forwardDict[seriesName] = forwardDict[seriesName];
                    if (!forwardDict[seriesName]) {
                        forwardDict[seriesName] = createSeriesNode(prefix, suffix, parent, +id, name, graphOptions);
                    }
                    forwardDict[seriesName].ids.push(id);
                    reverseDict[name] = reverseDict[name] || [];
                    reverseDict[name].push([seriesName, id]);
                }
                if (matched < 1) {
                    prefix = isGroup ? leaf.substr(0, leaf.length - 1) : leaf;
                    id = 0;
                    suffix = isGroup ? '*' : '';
                    seriesName = getSeriesNodeName(prefix, suffix, parent);
                    forwardDict[seriesName] = forwardDict[seriesName];
                    if (!forwardDict[seriesName]) {
                        forwardDict[seriesName] = createSeriesNode(prefix, suffix, parent, +id, name, graphOptions);
                    }
                    forwardDict[seriesName].ids.push(id);
                    reverseDict[name] = reverseDict[name] || [];
                    reverseDict[name].push([seriesName, id]);
                }
            });
            /** @type {Object}  A dictionary mapping seriesName to seriesInfoArray,
             * which is an array that contains objects with name, id, prefix, suffix,
             * and parent properties.
             */
            var candidatesDict = {};
            // For each of the member, put it into the maximum possible series,
            // and create candidatesDict accordingly.
            lodash.each(reverseDict, function (seriesNameIdArray, name) {
                seriesNameIdArray.sort(function (a, b) {
                    return forwardDict[b[0]].ids.length - forwardDict[a[0]].ids.length;
                });
                var seriesName = seriesNameIdArray[0][0];
                var id = seriesNameIdArray[0][1];
                candidatesDict[seriesName] = candidatesDict[seriesName] || [];
                const namepath = name.split('/');
                const leaf = namepath[namepath.length - 1];
                const parent = namepath.slice(0, namepath.length - 1).join('/');
                var seriesNode = createSeriesNode(forwardDict[seriesName].prefix, forwardDict[seriesName].suffix, parent, +id, name, graphOptions);
                candidatesDict[seriesName].push(seriesNode);
            });
            // In each group of nodes, group nodes in bunches that have monotonically
            // increasing numbers in their names.  Each of these bunches is a series.
            lodash.each(candidatesDict, function (seriesInfoArray, seriesName) {
                if (seriesInfoArray.length < 2) {
                    return;
                }
                seriesInfoArray.sort(function (a, b) {
                    return +a.clusterId - +b.clusterId;
                });
                // Loop through the nodes sorted by its detected series number, grouping
                // all nodes with monotonically-increasing series numbers.
                let seriesNodes = [seriesInfoArray[0]];
                for (let index = 1; index < seriesInfoArray.length; index++) {
                    let nextNode = seriesInfoArray[index];
                    if (nextNode.clusterId ===
                        seriesNodes[seriesNodes.length - 1].clusterId + 1) {
                        seriesNodes.push(nextNode);
                        continue;
                    }
                    addSeriesToDict(seriesNodes, seriesDict, +clusterId, metagraph, graphOptions);
                    seriesNodes = [nextNode];
                }
                addSeriesToDict(seriesNodes, seriesDict, +clusterId, metagraph, graphOptions);
            });
        });
        return seriesDict;
    }
    /**
     * Add a series to the provided dictionary mapping series names to series.
     *
     * @param seriesNodes the nodes in the series. Contains
     *     name, id, prefix, suffix and parent properties of the node.
     * @param seriesDict the dictionary of series
     * @param clusterId ID of the template of the nodes of the series
     * @param metagraph
     * @param graphOptions
     */
    function addSeriesToDict(seriesNodes, seriesDict, clusterId, metagraph, graphOptions) {
        if (seriesNodes.length > 1) {
            let curSeriesName = getSeriesNodeName(seriesNodes[0].prefix, seriesNodes[0].suffix, seriesNodes[0].parent, seriesNodes[0].clusterId, seriesNodes[seriesNodes.length - 1].clusterId);
            let curSeriesNode = createSeriesNode(seriesNodes[0].prefix, seriesNodes[0].suffix, seriesNodes[0].parent, clusterId, curSeriesName, graphOptions);
            lodash.each(seriesNodes, function (node) {
                curSeriesNode.ids.push(node.clusterId);
                curSeriesNode.metagraph.setNode(node.name, metagraph.node(node.name));
            });
            seriesDict[curSeriesName] = curSeriesNode;
        }
    }

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    let TfGraphOpCompatCard = class TfGraphOpCompatCard extends LegacyElementMixin(PolymerElement) {
        constructor() {
            super(...arguments);
            this._expanded = true;
            this._opCompatColor = OpNodeColors.COMPATIBLE;
            this._opIncompatColor = OpNodeColors.INCOMPATIBLE;
        }
        get _templateIndex() {
            var graphHierarchy = this.graphHierarchy;
            return graphHierarchy.getTemplateIndex();
        }
        _getNode(nodeName, graphHierarchy) {
            return graphHierarchy.node(nodeName);
        }
        _getRenderInfo(nodeName, renderHierarchy) {
            return this.renderHierarchy.getOrCreateRenderNodeByName(nodeName);
        }
        _toggleExpanded() {
            this._expanded = !this._expanded;
        }
        _getToggleIcon(expanded) {
            return expanded ? 'expand-less' : 'expand-more';
        }
        _resizeList(selector) {
            var list = document.querySelector(selector);
            if (list) {
                list.fire('iron-resize');
            }
        }
        get _incompatibleOpNodes() {
            var graphHierarchy = this.graphHierarchy;
            var hierarchyParams = this.hierarchyParams;
            if (graphHierarchy && graphHierarchy.root) {
                this.async(this._resizeList.bind(this, '#incompatibleOpsList'));
                return getIncompatibleOps(graphHierarchy, hierarchyParams);
            }
        }
        get _opCompatScore() {
            var graphHierarchy = this.graphHierarchy;
            if (graphHierarchy && graphHierarchy.root) {
                var root = graphHierarchy.root;
                var numCompat = root.compatibilityHistogram.compatible;
                var numIncompat = root.compatibilityHistogram.incompatible;
                if (numCompat == 0 && numIncompat == 0)
                    return 0;
                var numTotal = numCompat + numIncompat;
                // Round the ratio towards negative infinity.
                return Math.floor((100 * numCompat) / numTotal) / 100;
            }
            return 0;
        }
        get _opCompatScoreLabel() {
            var opCompatScore = this._opCompatScore;
            return format('.0%')(opCompatScore);
        }
        get _totalIncompatOps() {
            var graphHierarchy = this.graphHierarchy;
            if (graphHierarchy && graphHierarchy.root) {
                return graphHierarchy.root.compatibilityHistogram.incompatible;
            }
            return 0;
        }
    };
    TfGraphOpCompatCard.template = html `
    <style>
      :host {
        max-height: 500px;
      }

      .incompatible-ops-list {
        height: 350px;
        max-height: 400px;
        overflow-y: scroll;
        display: flex;
        flex-direction: column;
      }

      iron-list {
        flex: 1 1 auto;
      }

      paper-item {
        padding: 0;
        background: #e9e9e9;
      }

      paper-item-body[two-line] {
        min-height: 0;
        padding: 8px 12px 4px;
      }

      .expandedInfo {
        padding: 8px 12px;
        font-weight: 500;
        font-size: 12pt;
        width: 100%;
      }

      .node-name {
        white-space: normal;
        word-wrap: break-word;
        font-size: 14pt;
        font-weight: 500;
      }

      .subtitle {
        font-size: 12pt;
        color: #5e5e5e;
      }

      .toggle-button {
        float: right;
        max-height: 20px;
        max-width: 20px;
        padding: 0;
      }

      .non-control-list-item {
        padding-left: 10px;
      }

      div.op-compat-display {
        margin-top: 10px;
        display: inline-block;
      }

      svg.op-compat {
        width: 250px;
        height: 25px;
        float: left;
      }

      div.op-compat-value {
        float: right;
        height: 100%;
        font-size: 14px;
        color: black;
        margin-left: 10px;
      }
    </style>

    <paper-item>
      <paper-item-body two-line>
        <div>
          <paper-icon-button
            icon="{{_getToggleIcon(_expanded)}}"
            on-click="_toggleExpanded"
            class="toggle-button"
          >
          </paper-icon-button>
          <div class="node-name" id="nodetitle">[[nodeTitle]]</div>
        </div>
        <div secondary>
          <div class="subtitle">
            <div class="op-compat-display">
              <svg
                class="op-compat"
                preserveAspectRatio="xMinYMid meet"
                viewBox="0 0 250 25"
              >
                <defs>
                  <linearGradient id="op-compat-fill">
                    <stop offset="0" stop-color$="[[_opCompatColor]]"></stop>
                    <stop
                      offset$="[[_opCompatScore]]"
                      stop-color$="[[_opCompatColor]]"
                    ></stop>
                    <stop
                      offset$="[[_opCompatScore]]"
                      stop-color$="[[_opIncompatColor]]"
                    ></stop>
                    <stop offset="1" stop-color$="[[_opIncompatColor ]]"></stop>
                  </linearGradient>
                </defs>
                <rect
                  height="25"
                  width="250"
                  rx="5"
                  ry="5"
                  style="fill: url('#op-compat-fill');"
                ></rect>
              </svg>
              <div class="op-compat-value">[[_opCompatScoreLabel]]</div>
            </div>
          </div>
        </div>
      </paper-item-body>
    </paper-item>

    <iron-collapse opened="{{_expanded}}">
      <template is="dom-if" if="{{_expanded}}" restamp="true">
        <div class="expandedInfo">
          Incompatible Operations: (<span>[[_totalIncompatOps]]</span>)
          <iron-list
            class="incompatible-ops-list"
            id="incompatibleOpsList"
            items="[[_incompatibleOpNodes]]"
          >
            <template>
              <tf-graph-op-compat-list-item
                class="non-control-list-item"
                item-node="[[item]]"
                item-render-info="[[_getRenderInfo(item.name, renderHierarchy)]]"
                name="[[item.name]]"
                template-index="[[_templateIndex]]"
                color-by="[[colorBy]]"
                item-type="incompatible-ops"
              >
              </tf-graph-op-compat-list-item>
            </template>
          </iron-list>
        </div>
      </template>
    </iron-collapse>
  `;
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraphOpCompatCard.prototype, "graphHierarchy", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraphOpCompatCard.prototype, "hierarchyParams", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraphOpCompatCard.prototype, "renderHierarchy", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfGraphOpCompatCard.prototype, "nodeTitle", void 0);
    __decorate([
        property({
            type: Boolean,
        }),
        __metadata("design:type", Boolean)
    ], TfGraphOpCompatCard.prototype, "_expanded", void 0);
    __decorate([
        property({
            type: String,
        }),
        __metadata("design:type", String)
    ], TfGraphOpCompatCard.prototype, "_opCompatColor", void 0);
    __decorate([
        property({
            type: String,
        }),
        __metadata("design:type", String)
    ], TfGraphOpCompatCard.prototype, "_opIncompatColor", void 0);
    __decorate([
        computed('graphHierarchy'),
        __metadata("design:type", Object),
        __metadata("design:paramtypes", [])
    ], TfGraphOpCompatCard.prototype, "_templateIndex", null);
    __decorate([
        computed('graphHierarchy', 'hierarchyParams'),
        __metadata("design:type", Object),
        __metadata("design:paramtypes", [])
    ], TfGraphOpCompatCard.prototype, "_incompatibleOpNodes", null);
    __decorate([
        computed('graphHierarchy'),
        __metadata("design:type", Number),
        __metadata("design:paramtypes", [])
    ], TfGraphOpCompatCard.prototype, "_opCompatScore", null);
    __decorate([
        computed('_opCompatScore'),
        __metadata("design:type", String),
        __metadata("design:paramtypes", [])
    ], TfGraphOpCompatCard.prototype, "_opCompatScoreLabel", null);
    __decorate([
        computed('graphHierarchy'),
        __metadata("design:type", Number),
        __metadata("design:paramtypes", [])
    ], TfGraphOpCompatCard.prototype, "_totalIncompatOps", null);
    TfGraphOpCompatCard = __decorate([
        customElement('tf-graph-op-compat-card')
    ], TfGraphOpCompatCard);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    let TfNodeListItem = class TfNodeListItem extends LegacyElementMixin(PolymerElement) {
        _itemTypeChanged() {
            if (this.itemType !== 'subnode') {
                this.$['list-item'].classList.add('clickable');
            }
            else {
                this.$['list-item'].classList.remove('clickable');
            }
        }
        _nodeListener(event) {
            // fire node.click/mouseover/mouseout
            this.fire('node-list-item-' + event.type, {
                cardNode: this.cardNode.name,
                nodeName: this.name,
                type: this.itemType,
            });
        }
        _fadedClass(itemRenderInfo) {
            return itemRenderInfo && itemRenderInfo.isFadedOut ? 'faded' : '';
        }
    };
    TfNodeListItem.template = html `
    <style>
      #list-item {
        width: 100%;
        color: #565656;
        font-size: 11pt;
        font-weight: 400;
        position: relative;
        display: inline-block;
      }

      #list-item:hover {
        background-color: var(--google-yellow-100);
      }

      .clickable {
        cursor: pointer;
      }

      #list-item span {
        margin-left: 40px;
      }

      #list-item.excluded span {
        color: #999;
      }

      #list-item span.edge-label {
        float: right;
        font-size: 10px;
        margin-left: 3px;
        margin-right: 5px;
      }

      .node-icon {
        position: absolute;
        top: 1px;
        left: 2px;
      }

      .faded span {
        color: var(--tb-graph-faded);
      }
    </style>
    <div
      id="list-item"
      on-mouseover="_nodeListener"
      on-mouseout="_nodeListener"
      on-click="_nodeListener"
    >
      <div class$="{{_fadedClass(itemRenderInfo)}}">
        <tf-node-icon
          class="node-icon"
          height="12"
          color-by="[[colorBy]]"
          color-by-params="[[colorByParams]]"
          node="[[itemNode]]"
          render-info="[[itemRenderInfo]]"
          template-index="[[templateIndex]]"
        ></tf-node-icon>
        <span title$="[[name]]">[[name]]</span>
        <span class="edge-label">[[edgeLabel]]</span>
      </div>
    </div>
  `;
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfNodeListItem.prototype, "cardNode", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfNodeListItem.prototype, "itemNode", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfNodeListItem.prototype, "edgeLabel", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfNodeListItem.prototype, "itemRenderInfo", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfNodeListItem.prototype, "name", void 0);
    __decorate([
        property({
            type: String,
            observer: '_itemTypeChanged',
        }),
        __metadata("design:type", String)
    ], TfNodeListItem.prototype, "itemType", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfNodeListItem.prototype, "colorBy", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfNodeListItem.prototype, "colorByParams", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfNodeListItem.prototype, "templateIndex", void 0);
    TfNodeListItem = __decorate([
        customElement('tf-node-list-item')
    ], TfNodeListItem);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    let TfNodeInfo = class TfNodeInfo extends LegacyElementMixin(PolymerElement) {
        constructor() {
            super(...arguments);
            this._expanded = true;
            this._openedControlPred = false;
            this._openedControlSucc = false;
        }
        expandNode() {
            this.fire('_node.expand', this.node);
        }
        get _templateIndex() {
            var graphHierarchy = this.graphHierarchy;
            return graphHierarchy.getTemplateIndex();
        }
        _getNode(graphNodeName, graphHierarchy) {
            return graphHierarchy.node(graphNodeName);
        }
        _getNodeStats(graphNodeName, graphHierarchy) {
            var node = this._getNode(graphNodeName, graphHierarchy);
            if (node) {
                return node.stats;
            }
            return null;
        }
        _getTotalMicros(stats) {
            return stats ? stats.getTotalMicros() : 0;
        }
        get _hasDisplayableNodeStats() {
            var stats = this._nodeStats;
            return hasDisplayableNodeStats(stats);
        }
        get _nodeStatsFormattedBytes() {
            var stats = this._nodeStats;
            if (!stats || !stats.totalBytes) {
                return;
            }
            return convertUnitsToHumanReadable(stats.totalBytes, MEMORY_UNITS);
        }
        get _nodeStatsFormattedComputeTime() {
            var stats = this._nodeStats;
            if (!stats || !stats.getTotalMicros()) {
                return;
            }
            return convertUnitsToHumanReadable(stats.getTotalMicros(), TIME_UNITS);
        }
        get _nodeStatsFormattedOutputSizes() {
            var stats = this._nodeStats;
            if (!stats || !stats.outputSize || !stats.outputSize.length) {
                return;
            }
            return lodash.map(stats.outputSize, function (shape) {
                if (shape.length === 0) {
                    return 'scalar';
                }
                return '[' + shape.join(', ') + ']';
            });
        }
        _getRenderInfo(graphNodeName, renderHierarchy) {
            return this.renderHierarchy.getOrCreateRenderNodeByName(graphNodeName);
        }
        get _attributes() {
            var node = this._node;
            this.async(this._resizeList.bind(this, '#attributesList'));
            if (!node || !node.attr) {
                return [];
            }
            var attrs = [];
            lodash.each(node.attr, function (entry) {
                // Unpack the "too large" attributes into separate attributes
                // in the info card, with values "too large to show".
                if (entry.key === LARGE_ATTRS_KEY) {
                    attrs = attrs.concat(entry.value.list.s.map(function (key) {
                        return { key: key, value: 'Too large to show...' };
                    }));
                }
                else {
                    attrs.push({
                        key: entry.key,
                        value: JSON.stringify(entry.value),
                    });
                }
            });
            return attrs;
        }
        get _device() {
            var node = this._node;
            return node ? node.device : null;
        }
        get _successors() {
            var node = this._node;
            var hierarchy = this.graphHierarchy;
            this._refreshNodeItemList('inputsList');
            if (!node) {
                return { regular: [], control: [] };
            }
            return this._convertEdgeListToEdgeInfoList(hierarchy.getSuccessors(node.name), false, node.isGroupNode);
        }
        get _predecessors() {
            var node = this._node;
            var hierarchy = this.graphHierarchy;
            this._refreshNodeItemList('outputsList');
            if (!node) {
                return { regular: [], control: [] };
            }
            return this._convertEdgeListToEdgeInfoList(hierarchy.getPredecessors(node.name), true, node.isGroupNode);
        }
        // Only relevant if this is a library function. A list of nodes that
        // represent where the function is used.
        get _functionUsages() {
            var node = this._node;
            var hierarchy = this.graphHierarchy;
            this._refreshNodeItemList('functionUsagesList');
            if (!node || node.type !== NodeType.META) {
                // Functions must be represented by metanodes.
                return [];
            }
            const libraryFunctionData = hierarchy.libraryFunctions[node.associatedFunction];
            if (!libraryFunctionData) {
                // This is no function.
                return [];
            }
            // Return where the function is used.
            return libraryFunctionData.usages;
        }
        // The iron lists that enumerate ops must be asynchronously updated when
        // the data they render changes. This function triggers that update.
        _refreshNodeItemList(nodeListId) {
            this.async(this._resizeList.bind(this, `#${nodeListId}`));
        }
        _convertEdgeListToEdgeInfoList(list, isPredecessor, isGroupNode) {
            /**
             * Unpacks the metaedge into a list of base edge information
             * that can be rendered.
             */
            var unpackMetaedge = (metaedge) => {
                return lodash.map(metaedge.baseEdgeList, (baseEdge) => {
                    var name = isPredecessor ? baseEdge.v : baseEdge.w;
                    return {
                        name: name,
                        node: this._getNode(name, this.graphHierarchy),
                        edgeLabel: getLabelForBaseEdge(baseEdge, this.renderHierarchy),
                        renderInfo: this._getRenderInfo(name, this.renderHierarchy),
                    };
                });
            };
            /**
             * Converts a list of metaedges to a list of edge information
             * that can be rendered.
             */
            var toEdgeInfoList = function (edges) {
                var edgeInfoList = [];
                lodash.each(edges, (metaedge) => {
                    var name = isPredecessor ? metaedge.v : metaedge.w;
                    // Enumerate all the base edges if the node is an OpNode, or the
                    // metaedge has only 1 edge in it.
                    if (!isGroupNode || metaedge.baseEdgeList.length == 1) {
                        edgeInfoList = edgeInfoList.concat(unpackMetaedge(metaedge));
                    }
                    else {
                        edgeInfoList.push({
                            name: name,
                            node: this._getNode(name, this.graphHierarchy),
                            edgeLabel: getLabelForEdge(metaedge, this.renderHierarchy),
                            renderInfo: this._getRenderInfo(name, this.renderHierarchy),
                        });
                    }
                });
                return edgeInfoList;
            }.bind(this);
            return {
                regular: toEdgeInfoList(list.regular),
                control: toEdgeInfoList(list.control),
            };
        }
        get _subnodes() {
            var node = this._node;
            return node && node.metagraph ? node.metagraph.nodes() : null;
        }
        get _totalPredecessors() {
            var predecessors = this._predecessors;
            return predecessors.regular.length + predecessors.control.length;
        }
        get _totalSuccessors() {
            var successors = this._successors;
            return successors.regular.length + successors.control.length;
        }
        _toggleControlPred() {
            this._openedControlPred = !this._openedControlPred;
        }
        _toggleControlSucc() {
            this._openedControlSucc = !this._openedControlSucc;
        }
        _toggleExpanded() {
            this._expanded = !this._expanded;
        }
        _getToggleIcon(expanded) {
            return expanded ? 'expand-less' : 'expand-more';
        }
        _resetState() {
            this._openedControlPred = false;
            this._openedControlSucc = false;
            this.set('_groupButtonText', getGroupSettingLabel(this._node));
        }
        _resizeList(selector) {
            var list = document.querySelector(selector);
            if (list) {
                list.fire('iron-resize');
            }
        }
        _toggleInclude() {
            this.fire('node-toggle-inclusion', { name: this.graphNodeName });
        }
        _nodeIncludeStateChanged(include, oldInclude) {
            this.set('_auxButtonText', getIncludeNodeButtonString(include));
        }
        _toggleGroup() {
            var seriesName = getSeriesName(this._node);
            this.fire('node-toggle-seriesgroup', { name: seriesName });
        }
        _isLibraryFunction(node) {
            // If the node name starts with this string, the node is either a
            // library function or a node within it. Those nodes should never be
            // extracted into the auxiliary scene group because they represent
            // templates for function call nodes, not ops in the graph themselves.
            return node && node.name.startsWith(FUNCTION_LIBRARY_NODE_PREFIX);
        }
        _isInSeries(node) {
            return canBeInSeries(node);
        }
    };
    TfNodeInfo.template = html `
    <style>
      .sub-list-group {
        font-weight: 500;
        font-size: 12pt;
        padding-bottom: 8px;
        width: 100%;
      }

      .sub-list {
        max-height: 300px;
        overflow-y: scroll;
      }

      .attr-left {
        float: left;
        width: 30%;
        word-wrap: break-word;
        color: #565656;
        font-size: 11pt;
        font-weight: 400;
      }

      .attr-right {
        margin-left: 30%;
        word-wrap: break-word;
        color: #565656;
        font-weight: 400;
      }

      .sub-list-table {
        display: table;
        width: 100%;
      }

      .sub-list-table-row {
        display: table-row;
      }

      .sub-list-table-row .sub-list-table-cell:last-child {
        text-align: right;
      }

      .sub-list-table-cell {
        color: #565656;
        display: table-cell;
        font-size: 11pt;
        font-weight: 400;
        max-width: 200px;
        padding: 0 4px;
      }

      paper-item {
        padding: 0;
        background: #e9e9e9;
      }

      paper-item-body[two-line] {
        min-height: 0;
        padding: 8px 12px 4px;
      }

      .expandedInfo {
        padding: 8px 12px;
      }

      .controlDeps {
        padding: 0 0 0 8px;
      }

      .node-name {
        white-space: normal;
        word-wrap: break-word;
        font-size: 14pt;
        font-weight: 500;
      }

      .node-icon {
        float: right;
      }

      .subtitle {
        font-size: 12pt;
        color: #5e5e5e;
      }

      .controlLine {
        font-size: 11pt;
        font-weight: 400;
      }

      .toggle-button {
        float: right;
        max-height: 20px;
        max-width: 20px;
        padding: 0;
      }

      .control-toggle-button {
        float: left;
        max-height: 20px;
        max-width: 20px;
        padding: 0;
      }

      .toggle-include-group {
        padding-top: 4px;
      }

      .toggle-include {
        margin: 5px 6px;
        text-transform: none;
        padding: 4px 6px;
        font-size: 10pt;
        background-color: #fafafa;
        color: #666;
      }

      .toggle-include:hover {
        background-color: var(--google-yellow-100);
      }

      .non-control-list-item {
        padding-left: 10px;
      }
    </style>
    <paper-item>
      <paper-item-body two-line>
        <div>
          <paper-icon-button
            icon="{{_getToggleIcon(_expanded)}}"
            on-click="_toggleExpanded"
            class="toggle-button"
          >
          </paper-icon-button>
          <div class="node-name">
            <tf-wbr-string value="[[_node.name]]" delimiter-pattern="/">
            </tf-wbr-string>
          </div>
        </div>
        <div secondary>
          <tf-node-icon
            class="node-icon"
            node="[[_node]]"
            render-info="[[_getRenderInfo(graphNodeName, renderHierarchy)]]"
            color-by="[[colorBy]]"
            template-index="[[_templateIndex]]"
          ></tf-node-icon>
          <template is="dom-if" if="{{_node.op}}">
            <div class="subtitle">
              Operation:
              <span>[[_node.op]]</span>
            </div>
          </template>
          <template is="dom-if" if="{{_node.metagraph}}">
            <div class="subtitle">
              Subgraph:
              <span>[[_node.cardinality]]</span> nodes
            </div>
          </template>
        </div>
      </paper-item-body>
    </paper-item>
    <iron-collapse opened="{{_expanded}}">
      <template is="dom-if" if="{{_expanded}}" restamp="true">
        <div class="expandedInfo">
          <div class="sub-list-group attributes">
            Attributes (<span>[[_attributes.length]]</span>)
            <iron-list
              class="sub-list"
              id="attributesList"
              items="[[_attributes]]"
            >
              <template>
                <div>
                  <div class="attr-left">[[item.key]]</div>
                  <div class="attr-right">[[item.value]]</div>
                </div>
              </template>
            </iron-list>
          </div>

          <template is="dom-if" if="{{_device}}">
            <div class="sub-list-group device">
              <div class="attr-left">Device</div>
              <div class="attr-right">[[_device]]</div>
            </div>
          </template>

          <div class="sub-list-group predecessors">
            Inputs (<span>[[_totalPredecessors]]</span>)
            <iron-list
              class="sub-list"
              id="inputsList"
              items="[[_predecessors.regular]]"
            >
              <template>
                <tf-node-list-item
                  class="non-control-list-item"
                  card-node="[[_node]]"
                  item-node="[[item.node]]"
                  edge-label="[[item.edgeLabel]]"
                  item-render-info="[[item.renderInfo]]"
                  name="[[item.name]]"
                  item-type="predecessors"
                  color-by="[[colorBy]]"
                  template-index="[[_templateIndex]]"
                >
                </tf-node-list-item>
              </template>
            </iron-list>
            <template is="dom-if" if="[[_predecessors.control.length]]">
              <div class="controlDeps">
                <div class="controlLine">
                  <paper-icon-button
                    icon="{{_getToggleIcon(_openedControlPred)}}"
                    on-click="_toggleControlPred"
                    class="control-toggle-button"
                  >
                  </paper-icon-button>
                  Control dependencies
                </div>
                <iron-collapse opened="{{_openedControlPred}}" no-animation>
                  <template
                    is="dom-if"
                    if="{{_openedControlPred}}"
                    restamp="true"
                  >
                    <iron-list
                      class="sub-list"
                      items="[[_predecessors.control]]"
                    >
                      <template>
                        <tf-node-list-item
                          card-node="[[_node]]"
                          item-node="[[item.node]]"
                          item-render-info="[[item.renderInfo]]"
                          name="[[item.name]]"
                          item-type="predecessors"
                          color-by="[[colorBy]]"
                          template-index="[[_templateIndex]]"
                        >
                        </tf-node-list-item>
                      </template>
                    </iron-list>
                  </template>
                </iron-collapse>
              </div>
            </template>
          </div>

          <div class="sub-list-group successors">
            Outputs (<span>[[_totalSuccessors]]</span>)
            <iron-list
              class="sub-list"
              id="outputsList"
              items="[[_successors.regular]]"
            >
              <template>
                <tf-node-list-item
                  class="non-control-list-item"
                  card-node="[[_node]]"
                  item-node="[[item.node]]"
                  edge-label="[[item.edgeLabel]]"
                  item-render-info="[[item.renderInfo]]"
                  name="[[item.name]]"
                  item-type="successor"
                  color-by="[[colorBy]]"
                  template-index="[[_templateIndex]]"
                >
                </tf-node-list-item>
              </template>
            </iron-list>
            <template is="dom-if" if="[[_successors.control.length]]">
              <div class="controlDeps">
                <div class="controlLine">
                  <paper-icon-button
                    icon="{{_getToggleIcon(_openedControlSucc)}}"
                    on-click="_toggleControlSucc"
                    class="control-toggle-button"
                  >
                  </paper-icon-button>
                  Control dependencies
                </div>
                <iron-collapse opened="{{_openedControlSucc}}" no-animation>
                  <template
                    is="dom-if"
                    if="{{_openedControlSucc}}"
                    restamp="true"
                  >
                    <iron-list class="sub-list" items="[[_successors.control]]">
                      <template>
                        <tf-node-list-item
                          card-node="[[_node]]"
                          item-node="[[item.node]]"
                          item-render-info="[[item.renderInfo]]"
                          name="[[item.name]]"
                          item-type="successors"
                          color-by="[[colorBy]]"
                          template-index="[[_templateIndex]]"
                        >
                        </tf-node-list-item>
                      </template>
                    </iron-list>
                  </template>
                </iron-collapse>
              </div>
            </template>
          </div>
          <template is="dom-if" if="{{_hasDisplayableNodeStats}}">
            <div class="sub-list-group node-stats">
              Node Stats
              <div class="sub-list-table">
                <template is="dom-if" if="{{_nodeStats.totalBytes}}">
                  <div class="sub-list-table-row">
                    <div class="sub-list-table-cell">Memory</div>
                    <div class="sub-list-table-cell">
                      [[_nodeStatsFormattedBytes]]
                    </div>
                  </div>
                </template>
                <template is="dom-if" if="{{_getTotalMicros(_nodeStats)}}">
                  <div class="sub-list-table-row">
                    <div class="sub-list-table-cell">Compute Time</div>
                    <div class="sub-list-table-cell">
                      [[_nodeStatsFormattedComputeTime]]
                    </div>
                  </div>
                </template>
                <template is="dom-if" if="{{_nodeStats.outputSize}}">
                  <div class="sub-list-table-row">
                    <div class="sub-list-table-cell">Tensor Output Sizes</div>
                    <div class="sub-list-table-cell">
                      <template
                        is="dom-repeat"
                        items="{{_nodeStatsFormattedOutputSizes}}"
                      >
                        [[item]] <br />
                      </template>
                    </div>
                  </div>
                </template>
              </div>
            </div>
          </template>

          <template is="dom-if" if="[[_functionUsages.length]]">
            <div class="sub-list-group predecessors">
              Usages of the Function (<span>[[_functionUsages.length]]</span>)
              <iron-list
                class="sub-list"
                id="functionUsagesList"
                items="[[_functionUsages]]"
              >
                <template>
                  <tf-node-list-item
                    class="non-control-list-item"
                    card-node="[[_node]]"
                    item-node="[[item]]"
                    name="[[item.name]]"
                    item-type="functionUsages"
                    color-by="[[colorBy]]"
                    template-index="[[_templateIndex]]"
                  >
                  </tf-node-list-item>
                </template>
              </iron-list>
            </div>
          </template>

          <template is="dom-if" if="[[!_isLibraryFunction(_node)]]">
            <div class="toggle-include-group">
              <paper-button
                raised
                class="toggle-include"
                on-click="_toggleInclude"
              >
                <span>[[_auxButtonText]]</span>
              </paper-button>
            </div>
          </template>

          <template is="dom-if" if="{{_isInSeries(_node)}}">
            <div class="toggle-include-group">
              <paper-button
                raised
                class="toggle-include"
                on-click="_toggleGroup"
              >
                <span>[[_groupButtonText]]</span>
              </paper-button>
            </div>
          </template>
        </div>
      </template>
    </iron-collapse>
  `;
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfNodeInfo.prototype, "graphNodeName", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfNodeInfo.prototype, "graphHierarchy", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfNodeInfo.prototype, "renderHierarchy", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfNodeInfo.prototype, "colorBy", void 0);
    __decorate([
        property({
            type: Object,
            computed: '_getNode(graphNodeName, graphHierarchy)',
            observer: '_resetState',
        }),
        __metadata("design:type", Object)
    ], TfNodeInfo.prototype, "_node", void 0);
    __decorate([
        property({
            type: Object,
            computed: '_getNodeStats(graphNodeName, graphHierarchy)',
            observer: '_resetState',
        }),
        __metadata("design:type", Object)
    ], TfNodeInfo.prototype, "_nodeStats", void 0);
    __decorate([
        property({
            type: Number,
            observer: '_nodeIncludeStateChanged',
        }),
        __metadata("design:type", Number)
    ], TfNodeInfo.prototype, "nodeInclude", void 0);
    __decorate([
        property({
            type: Boolean,
        }),
        __metadata("design:type", Boolean)
    ], TfNodeInfo.prototype, "_expanded", void 0);
    __decorate([
        property({
            type: Boolean,
        }),
        __metadata("design:type", Boolean)
    ], TfNodeInfo.prototype, "_openedControlPred", void 0);
    __decorate([
        property({
            type: Boolean,
        }),
        __metadata("design:type", Boolean)
    ], TfNodeInfo.prototype, "_openedControlSucc", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfNodeInfo.prototype, "_auxButtonText", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfNodeInfo.prototype, "_groupButtonText", void 0);
    __decorate([
        computed('graphHierarchy'),
        __metadata("design:type", Object),
        __metadata("design:paramtypes", [])
    ], TfNodeInfo.prototype, "_templateIndex", null);
    __decorate([
        computed('_nodeStats'),
        __metadata("design:type", Boolean),
        __metadata("design:paramtypes", [])
    ], TfNodeInfo.prototype, "_hasDisplayableNodeStats", null);
    __decorate([
        computed('_nodeStats'),
        __metadata("design:type", String),
        __metadata("design:paramtypes", [])
    ], TfNodeInfo.prototype, "_nodeStatsFormattedBytes", null);
    __decorate([
        computed('_nodeStats'),
        __metadata("design:type", String),
        __metadata("design:paramtypes", [])
    ], TfNodeInfo.prototype, "_nodeStatsFormattedComputeTime", null);
    __decorate([
        computed('_nodeStats'),
        __metadata("design:type", Array),
        __metadata("design:paramtypes", [])
    ], TfNodeInfo.prototype, "_nodeStatsFormattedOutputSizes", null);
    __decorate([
        computed('_node'),
        __metadata("design:type", Array),
        __metadata("design:paramtypes", [])
    ], TfNodeInfo.prototype, "_attributes", null);
    __decorate([
        computed('_node'),
        __metadata("design:type", String),
        __metadata("design:paramtypes", [])
    ], TfNodeInfo.prototype, "_device", null);
    __decorate([
        computed('_node', 'graphHierarchy'),
        __metadata("design:type", Object),
        __metadata("design:paramtypes", [])
    ], TfNodeInfo.prototype, "_successors", null);
    __decorate([
        computed('_node', 'graphHierarchy'),
        __metadata("design:type", Object),
        __metadata("design:paramtypes", [])
    ], TfNodeInfo.prototype, "_predecessors", null);
    __decorate([
        computed('_node', 'graphHierarchy'),
        __metadata("design:type", Array),
        __metadata("design:paramtypes", [])
    ], TfNodeInfo.prototype, "_functionUsages", null);
    __decorate([
        computed('_node'),
        __metadata("design:type", Array),
        __metadata("design:paramtypes", [])
    ], TfNodeInfo.prototype, "_subnodes", null);
    __decorate([
        computed('_predecessors'),
        __metadata("design:type", Number),
        __metadata("design:paramtypes", [])
    ], TfNodeInfo.prototype, "_totalPredecessors", null);
    __decorate([
        computed('_successors'),
        __metadata("design:type", Number),
        __metadata("design:paramtypes", [])
    ], TfNodeInfo.prototype, "_totalSuccessors", null);
    TfNodeInfo = __decorate([
        customElement('tf-node-info')
    ], TfNodeInfo);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    let TfGraphInfo = class TfGraphInfo extends LegacyElementMixin(PolymerElement) {
        ready() {
            super.ready();
            this.addEventListener('node-list-item-click', this._nodeListItemClicked.bind(this));
            this.addEventListener('node-list-item-mouseover', this._nodeListItemMouseover.bind(this));
            this.addEventListener('node-list-item-mouseout', this._nodeListItemMouseout.bind(this));
        }
        _nodeListItemClicked(event) {
            this.selectedNode = event.detail.nodeName;
        }
        _nodeListItemMouseover(event) {
            this.highlightedNode = event.detail.nodeName;
        }
        _nodeListItemMouseout() {
            this.highlightedNode = null;
        }
        _healthPillsAvailable(debuggerDataEnabled, nodeNamesToHealthPills) {
            // So long as there is a mapping (even if empty) from node name to health pills, show the
            // legend and slider. We do that because, even if no health pills exist at the current step,
            // the user may desire to change steps, and the slider must show for the user to do that.
            return (debuggerDataEnabled &&
                nodeNamesToHealthPills &&
                Object.keys(nodeNamesToHealthPills).length > 0);
        }
        _equals(a, b) {
            return a === b;
        }
    };
    TfGraphInfo.template = html `
    <style>
      :host {
        font-size: 12px;
        margin: 0;
        padding: 0;
        display: block;
        max-height: 650px;
        overflow-x: hidden;
        overflow-y: auto;
      }

      h2 {
        padding: 0;
        text-align: center;
        margin: 0;
      }
    </style>
    <template is="dom-if" if="{{selectedNode}}">
      <paper-material elevation="1" class="card">
        <tf-node-info
          graph-hierarchy="[[graphHierarchy]]"
          render-hierarchy="[[renderHierarchy]]"
          flat-graph="[[graph]]"
          graph-node-name="[[selectedNode]]"
          node-include="[[selectedNodeInclude]]"
          highlighted-node="{{highlightedNode}}"
          color-by="[[colorBy]]"
        >
        </tf-node-info>
      </paper-material>
    </template>
    <template is="dom-if" if="[[_equals(colorBy, 'op_compatibility')]]">
      <tf-graph-op-compat-card
        graph-hierarchy="[[graphHierarchy]]"
        hierarchy-params="[[hierarchyParams]]"
        render-hierarchy="[[renderHierarchy]]"
        color-by="[[colorBy]]"
        node-title="[[compatNodeTitle]]"
      >
      </tf-graph-op-compat-card>
    </template>
    <template
      is="dom-if"
      if="[[_healthPillsAvailable(debuggerDataEnabled, nodeNamesToHealthPills)]]"
    >
      <tf-graph-debugger-data-card
        render-hierarchy="[[renderHierarchy]]"
        debugger-numeric-alerts="[[debuggerNumericAlerts]]"
        node-names-to-health-pills="[[nodeNamesToHealthPills]]"
        selected-node="{{selectedNode}}"
        highlighted-node="{{highlightedNode}}"
        are-health-pills-loading="[[areHealthPillsLoading]]"
        all-steps-mode-enabled="{{allStepsModeEnabled}}"
        specific-health-pill-step="{{specificHealthPillStep}}"
        health-pill-step-index="{{healthPillStepIndex}}"
      >
      </tf-graph-debugger-data-card>
    </template>
  `;
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfGraphInfo.prototype, "title", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraphInfo.prototype, "graphHierarchy", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraphInfo.prototype, "graph", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraphInfo.prototype, "renderHierarchy", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraphInfo.prototype, "nodeNamesToHealthPills", void 0);
    __decorate([
        property({
            type: Number,
            notify: true,
        }),
        __metadata("design:type", Number)
    ], TfGraphInfo.prototype, "healthPillStepIndex", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfGraphInfo.prototype, "colorBy", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfGraphInfo.prototype, "compatNodeTitle", void 0);
    __decorate([
        property({
            type: String,
            notify: true,
        }),
        __metadata("design:type", String)
    ], TfGraphInfo.prototype, "selectedNode", void 0);
    __decorate([
        property({
            type: String,
            notify: true,
        }),
        __metadata("design:type", String)
    ], TfGraphInfo.prototype, "highlightedNode", void 0);
    __decorate([
        property({
            type: Number,
            notify: true,
        }),
        __metadata("design:type", Number)
    ], TfGraphInfo.prototype, "selectedNodeInclude", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfGraphInfo.prototype, "debuggerDataEnabled", void 0);
    TfGraphInfo = __decorate([
        customElement('tf-graph-info')
    ], TfGraphInfo);

    /* Copyright 2015 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the 'License');
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an 'AS IS' BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    const FRAC_VIEWPOINT_AREA = 0.8;
    class Minimap {
        /**
         * Constructs a new minimap.
         *
         * @param svg The main svg element.
         * @param zoomG The svg group used for panning and zooming the main svg.
         * @param mainZoom The main zoom behavior.
         * @param minimap The minimap container.
         * @param maxWandH The maximum width/height for the minimap.
         * @param labelPadding Padding in pixels due to the main graph labels.
         */
        constructor(svg, zoomG, mainZoom, minimap, maxWandH, labelPadding) {
            this.svg = svg;
            this.labelPadding = labelPadding;
            this.zoomG = zoomG;
            this.mainZoom = mainZoom;
            this.maxWandH = maxWandH;
            let $shadowRoot = select(minimap.shadowRoot);
            // The minimap will have 2 main components: the canvas showing the content
            // and an svg showing a rectangle of the currently zoomed/panned viewpoint.
            let $minimapSvg = $shadowRoot.select('svg');
            // Make the viewpoint rectangle draggable.
            let $viewpoint = $minimapSvg.select('rect');
            let dragmove = (d) => {
                this.viewpointCoord.x = event.x;
                this.viewpointCoord.y = event.y;
                this.updateViewpoint();
            };
            this.viewpointCoord = { x: 0, y: 0 };
            let drag$1 = drag().subject(Object).on('drag', dragmove);
            $viewpoint.datum(this.viewpointCoord).call(drag$1);
            // Make the minimap clickable.
            $minimapSvg.on('click', () => {
                if (event.defaultPrevented) {
                    // This click was part of a drag event, so suppress it.
                    return;
                }
                // Update the coordinates of the viewpoint.
                let width = Number($viewpoint.attr('width'));
                let height = Number($viewpoint.attr('height'));
                let clickCoords = mouse($minimapSvg.node());
                this.viewpointCoord.x = clickCoords[0] - width / 2;
                this.viewpointCoord.y = clickCoords[1] - height / 2;
                this.updateViewpoint();
            });
            this.viewpoint = $viewpoint.node();
            this.minimapSvg = $minimapSvg.node();
            this.minimap = minimap;
            this.canvas = $shadowRoot.select('canvas.first').node();
            this.canvasBuffer = ($shadowRoot.select('canvas.second').node());
            this.downloadCanvas = ($shadowRoot.select('canvas.download').node());
            select(this.downloadCanvas).style('display', 'none');
            this.update();
        }
        /**
         * Updates the position and the size of the viewpoint rectangle.
         * It also notifies the main svg about the new panned position.
         */
        updateViewpoint() {
            // Update the coordinates of the viewpoint rectangle.
            select(this.viewpoint)
                .attr('x', this.viewpointCoord.x)
                .attr('y', this.viewpointCoord.y);
            // Update the translation vector of the main svg to reflect the
            // new viewpoint.
            let mainX = (-this.viewpointCoord.x * this.scaleMain) / this.scaleMinimap;
            let mainY = (-this.viewpointCoord.y * this.scaleMain) / this.scaleMinimap;
            select(this.svg).call(this.mainZoom.transform, identity$9.translate(mainX, mainY).scale(this.scaleMain));
        }
        /**
         * Redraws the minimap. Should be called whenever the main svg
         * was updated (e.g. when a node was expanded).
         */
        update() {
            let sceneSize = null;
            try {
                // Get the size of the entire scene.
                sceneSize = this.zoomG.getBBox();
                if (sceneSize.width === 0) {
                    // There is no scene anymore. We have been detached from the dom.
                    return;
                }
            }
            catch (e) {
                // Firefox produced NS_ERROR_FAILURE if we have been
                // detached from the dom.
                return;
            }
            let $download = select('#graphdownload');
            this.download = $download.node();
            $download.on('click', (d) => {
                // Revoke the old URL, if any. Then, generate a new URL.
                URL.revokeObjectURL(this.download.href);
                // We can't use the `HTMLCanvasElement.toBlob` API because it does
                // not have a synchronous variant, and we need to update this href
                // synchronously. Instead, we create a blob manually from the data
                // URL.
                const dataUrl = this.downloadCanvas.toDataURL('image/png');
                const prefix = dataUrl.slice(0, dataUrl.indexOf(','));
                if (!prefix.endsWith(';base64')) {
                    console.warn(`non-base64 data URL (${prefix}); cannot use blob download`);
                    this.download.href = dataUrl;
                    return;
                }
                const data = atob(dataUrl.slice(dataUrl.indexOf(',') + 1));
                const bytes = new Uint8Array(data.length).map((_, i) => data.charCodeAt(i));
                const blob = new Blob([bytes], { type: 'image/png' });
                this.download.href = URL.createObjectURL(blob);
            });
            let $svg = select(this.svg);
            // Read all the style rules in the document and embed them into the svg.
            // The svg needs to be self contained, i.e. all the style rules need to be
            // embedded so the canvas output matches the origin.
            let stylesText = '';
            const anySvg = this.svg;
            // MSEdge does not have `getRootNode`. In that case, manually get the root
            // node. This is more brittle than the getRootNode as changing DOM structure
            // will break this.
            const rootNode = anySvg.getRootNode
                ? anySvg.getRootNode()
                : this.svg.parentNode;
            const styleSheets = rootNode.styleSheets;
            for (let k = 0; k < styleSheets.length; k++) {
                try {
                    let cssRules = styleSheets[k].cssRules || styleSheets[k].rules;
                    if (cssRules == null) {
                        continue;
                    }
                    for (let i = 0; i < cssRules.length; i++) {
                        // Remove tf-* selectors from the styles.
                        stylesText +=
                            cssRules[i].cssText.replace(/ ?tf-[\w-]+ ?/g, '') + '\n';
                    }
                }
                catch (e) {
                    if (e.name !== 'SecurityError') {
                        throw e;
                    }
                }
            }
            // Temporarily add the css rules to the main svg.
            let svgStyle = $svg.append('style');
            svgStyle.text(stylesText);
            // Temporarily remove the zoom/pan transform from the main svg since we
            // want the minimap to show a zoomed-out and centered view.
            let $zoomG = select(this.zoomG);
            let zoomTransform = $zoomG.attr('transform');
            $zoomG.attr('transform', null);
            // https://github.com/tensorflow/tensorboard/issues/1598
            // Account for SVG content shift. SVGGraphicsElement.getBBox().width returns
            // width in pixel value of very tight bounding box of non-empty content.
            // Since we want to measure the sceneSize from the origin to the right most
            // edge of the right most node, we need to account for distance from the
            // origin to the left edge of the bounding box.
            sceneSize.height += sceneSize.y;
            sceneSize.width += sceneSize.x;
            // Since we add padding, account for that here.
            sceneSize.height += this.labelPadding * 2;
            sceneSize.width += this.labelPadding * 2;
            // Temporarily assign an explicit width/height to the main svg, since
            // it doesn't have one (uses flex-box), but we need it for the canvas
            // to work.
            $svg.attr('width', sceneSize.width).attr('height', sceneSize.height);
            // Since the content inside the svg changed (e.g. a node was expanded),
            // the aspect ratio have also changed. Thus, we need to update the scale
            // factor of the minimap. The scale factor is determined such that both
            // the width and height of the minimap are <= maximum specified w/h.
            this.scaleMinimap =
                this.maxWandH / Math.max(sceneSize.width, sceneSize.height);
            this.minimapSize = {
                width: sceneSize.width * this.scaleMinimap,
                height: sceneSize.height * this.scaleMinimap,
            };
            // Update the size of the minimap's svg, the buffer canvas and the
            // viewpoint rect.
            select(this.minimapSvg).attr(this.minimapSize);
            select(this.canvasBuffer).attr(this.minimapSize);
            // Download canvas width and height are multiples of the style width and
            // height in order to increase pixel density of the PNG for clarity.
            const downloadCanvasSelection = select(this.downloadCanvas);
            downloadCanvasSelection.style('width', sceneSize.width);
            downloadCanvasSelection.style('height', sceneSize.height);
            downloadCanvasSelection.attr('width', 3 * sceneSize.width);
            downloadCanvasSelection.attr('height', 3 * sceneSize.height);
            if (this.translate != null && this.zoom != null) {
                // Update the viewpoint rectangle shape since the aspect ratio of the
                // map has changed.
                requestAnimationFrame(() => this.zoom());
            }
            // TODO(stephanwlee): Consider not mutating the original DOM then read it --
            // this may cause reflow.
            // Serialize the main svg to a string which will be used as the rendering
            // content for the canvas.
            let svgXml = new XMLSerializer().serializeToString(this.svg);
            // Now that the svg is serialized for rendering, remove the temporarily
            // assigned styles, explicit width and height and bring back the pan/zoom
            // transform.
            svgStyle.remove();
            $svg.attr('width', null).attr('height', null);
            $zoomG.attr('transform', zoomTransform);
            let image = new Image();
            image.onload = () => {
                // Draw the svg content onto the buffer canvas.
                let context = this.canvasBuffer.getContext('2d');
                context.clearRect(0, 0, this.canvasBuffer.width, this.canvasBuffer.height);
                context.drawImage(image, 0, 0, this.minimapSize.width, this.minimapSize.height);
                requestAnimationFrame(() => {
                    // Hide the old canvas and show the new buffer canvas.
                    select(this.canvasBuffer).style('display', null);
                    select(this.canvas).style('display', 'none');
                    // Swap the two canvases.
                    [this.canvas, this.canvasBuffer] = [this.canvasBuffer, this.canvas];
                });
                let downloadContext = this.downloadCanvas.getContext('2d');
                downloadContext.clearRect(0, 0, this.downloadCanvas.width, this.downloadCanvas.height);
                downloadContext.drawImage(image, 0, 0, this.downloadCanvas.width, this.downloadCanvas.height);
            };
            image.onerror = () => {
                let blob = new Blob([svgXml], { type: 'image/svg+xml;charset=utf-8' });
                image.src = URL.createObjectURL(blob);
            };
            image.src =
                'data:image/svg+xml;charset=utf-8,' + encodeURIComponent(svgXml);
        }
        /**
         * Handles changes in zooming/panning. Should be called from the main svg
         * to notify that a zoom/pan was performed and this minimap will update it's
         * viewpoint rectangle.
         *
         * @param translate The translate vector, or none to use the last used one.
         * @param scale The scaling factor, or none to use the last used one.
         */
        zoom(transform) {
            if (this.scaleMinimap == null) {
                // Scene is not ready yet.
                return;
            }
            // Update the new translate and scale params, only if specified.
            if (transform) {
                this.translate = [transform.x, transform.y];
                this.scaleMain = transform.k;
            }
            // Update the location of the viewpoint rectangle.
            let svgRect = this.svg.getBoundingClientRect();
            let $viewpoint = select(this.viewpoint);
            this.viewpointCoord.x =
                (-this.translate[0] * this.scaleMinimap) / this.scaleMain;
            this.viewpointCoord.y =
                (-this.translate[1] * this.scaleMinimap) / this.scaleMain;
            let viewpointWidth = (svgRect.width * this.scaleMinimap) / this.scaleMain;
            let viewpointHeight = (svgRect.height * this.scaleMinimap) / this.scaleMain;
            $viewpoint
                .attr('x', this.viewpointCoord.x)
                .attr('y', this.viewpointCoord.y)
                .attr('width', viewpointWidth)
                .attr('height', viewpointHeight);
            // Show/hide the minimap depending on the viewpoint area as fraction of the
            // whole minimap.
            let mapWidth = this.minimapSize.width;
            let mapHeight = this.minimapSize.height;
            let x = this.viewpointCoord.x;
            let y = this.viewpointCoord.y;
            let w = Math.min(Math.max(0, x + viewpointWidth), mapWidth) -
                Math.min(Math.max(0, x), mapWidth);
            let h = Math.min(Math.max(0, y + viewpointHeight), mapHeight) -
                Math.min(Math.max(0, y), mapHeight);
            let fracIntersect = (w * h) / (mapWidth * mapHeight);
            if (fracIntersect < FRAC_VIEWPOINT_AREA) {
                this.minimap.classList.remove('hidden');
            }
            else {
                this.minimap.classList.add('hidden');
            }
        }
    }

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    let TfGraphMinimap = class TfGraphMinimap extends PolymerElement {
        /**
         * Initializes the minimap and returns a minimap object to notify when
         * things update.
         *
         * @param svg The main svg element.
         * @param zoomG The svg group used for panning and zooming the main svg.
         * @param mainZoom The main zoom behavior.
         * @param maxWAndH The maximum width/height for the minimap.
         * @param labelPadding Padding in pixels due to the main graph labels.
         */
        init(svg, zoomG, mainZoom, maxWAndH, labelPadding) {
            return new Minimap(svg, zoomG, mainZoom, this, maxWAndH, labelPadding);
        }
    };
    TfGraphMinimap.template = html `
    <style>
      :host {
        background-color: white;
        transition: opacity 0.3s linear;
        pointer-events: auto;
      }

      :host(.hidden) {
        opacity: 0;
        pointer-events: none;
      }

      canvas {
        border: 1px solid #999;
      }

      rect {
        fill: white;
        stroke: #111111;
        stroke-width: 1px;
        fill-opacity: 0;
        filter: url(#minimapDropShadow);
        cursor: move;
      }

      svg {
        position: absolute;
      }
    </style>
    <svg>
      <defs>
        <filter
          id="minimapDropShadow"
          x="-20%"
          y="-20%"
          width="150%"
          height="150%"
        >
          <feOffset result="offOut" in="SourceGraphic" dx="1" dy="1"></feOffset>
          <feColorMatrix
            result="matrixOut"
            in="offOut"
            type="matrix"
            values="0.1 0 0 0 0 0 0.1 0 0 0 0 0 0.1 0 0 0 0 0 0.5 0"
          ></feColorMatrix>
          <feGaussianBlur
            result="blurOut"
            in="matrixOut"
            stdDeviation="2"
          ></feGaussianBlur>
          <feBlend in="SourceGraphic" in2="blurOut" mode="normal"></feBlend>
        </filter>
      </defs>
      <rect></rect>
    </svg>
    <canvas class="first"></canvas>
    <!-- Additional canvas to use as buffer to avoid flickering between updates -->
    <canvas class="second"></canvas>
    <canvas class="download"></canvas>
  `;
    TfGraphMinimap = __decorate([
        customElement('tf-graph-minimap')
    ], TfGraphMinimap);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    const template$j = html `
  <style>
    :host {
      display: flex;
      font-size: 20px;
      height: 100%;
      width: 100%;
    }

    #svg {
      flex: 1;
      font-family: Roboto, sans-serif;
      height: 100%;
      overflow: hidden;
      width: 100%;
    }

    #hidden {
      position: fixed;
      top: 0px;
      visibility: hidden;
    }

    /* --- Node and annotation-node for Metanode --- */

    .meta > .nodeshape > rect,
    .meta > .annotation-node > rect {
      cursor: pointer;
      fill: hsl(0, 0%, 70%);
    }
    .node.meta.highlighted > .nodeshape > rect,
    .node.meta.highlighted > .annotation-node > rect {
      stroke-width: 2;
    }
    .annotation.meta.highlighted > .nodeshape > rect,
    .annotation.meta.highlighted > .annotation-node > rect {
      stroke-width: 1;
    }
    .meta.selected > .nodeshape > rect,
    .meta.selected > .annotation-node > rect {
      stroke: red;
      stroke-width: 2;
    }
    .node.meta.selected.expanded > .nodeshape > rect,
    .node.meta.selected.expanded > .annotation-node > rect {
      stroke: red;
      stroke-width: 3;
    }
    .annotation.meta.selected > .nodeshape > rect,
    .annotation.meta.selected > .annotation-node > rect {
      stroke: red;
      stroke-width: 2;
    }
    .node.meta.selected.expanded.highlighted > .nodeshape > rect,
    .node.meta.selected.expanded.highlighted > .annotation-node > rect {
      stroke: red;
      stroke-width: 4;
    }

    .faded,
    .faded rect,
    .faded ellipse,
    .faded path,
    .faded use,
    #rectHatch line,
    #ellipseHatch line {
      color: #e0d4b3 !important;
      fill: white;
      stroke: #e0d4b3 !important;
    }

    .faded path {
      stroke-width: 1px !important;
    }

    .faded rect {
      fill: url(#rectHatch) !important;
    }

    .faded ellipse,
    .faded use {
      fill: url(#ellipseHatch) !important;
    }

    .faded text {
      opacity: 0;
    }

    /* Rules used for input-tracing. */
    .input-highlight > * > rect,
    .input-highlight > * > ellipse,
    .input-highlight > * > use {
      fill: white;
      stroke: #ff9800 !important;
    }

    /*  - Faded non-input styling */
    .non-input > * > rect,
.non-input > * > ellipse,
.non-input > * > use,
/* For Const nodes. */
.non-input > * > .constant:not([class*="input-highlight"]) >
  .annotation-node > ellipse,
/* For styling of annotation nodes of non-input nodes. */
.non-input > g > .annotation > .annotation-node > rect {
      stroke: #e0d4b3 !important;
      stroke-width: inherit;
      stroke-dasharray: inherit;
    }

    .non-input path {
      visibility: hidden;
    }

    .non-input > .nodeshape > rect,
.non-input > .annotation-node > rect,
/* For styling of annotation nodes of non-input nodes. */
.non-input > g > .annotation > .annotation-node > rect {
      fill: url(#rectHatch) !important;
    }

    .non-input ellipse,
    .non-input use {
      fill: url(#ellipseHatch) !important;
    }

    .non-input > text {
      opacity: 0;
    }

    .non-input .annotation > .annotation-edge {
      marker-end: url(#annotation-arrowhead-faded);
    }

    .non-input .annotation > .annotation-edge.refline {
      marker-start: url(#ref-annotation-arrowhead-faded);
    }

    /* Input edges. */
    .input-edge-highlight > text {
      fill: black !important;
    }
    .input-highlight > .in-annotations > .annotation > .annotation-edge,
    .input-highlight-selected
      > .in-annotations
      > .annotation
      > .annotation-edge {
      stroke: #999 !important;
    }

    /* Non-input edges. */
    .non-input-edge-highlight,
.non-input > g > .annotation > path,
/* Annotation styles (label and edges respectively). */
.non-input > g >
.annotation:not(.input-highlight):not(.input-highlight-selected) >
.annotation-label
/*.annotation-edge*/ {
      visibility: hidden;
    }

    /* --- Op Node --- */

    .op > .nodeshape > .nodecolortarget,
    .op > .annotation-node > .nodecolortarget {
      cursor: pointer;
      fill: #fff;
      stroke: #ccc;
    }

    .op.selected > .nodeshape > .nodecolortarget,
    .op.selected > .annotation-node > .nodecolortarget {
      stroke: red;
      stroke-width: 2;
    }

    .op.highlighted > .nodeshape > .nodecolortarget,
    .op.highlighted > .annotation-node > .nodecolortarget {
      stroke-width: 2;
    }

    /* --- Series Node --- */

    /* By default, don't show the series background <rect>. */
    .series > .nodeshape > rect {
      fill: hsl(0, 0%, 70%);
      fill-opacity: 0;
      stroke-dasharray: 5, 5;
      stroke-opacity: 0;
      cursor: pointer;
    }

    /* Once expanded, show the series background <rect> and hide the <use>. */
    .series.expanded > .nodeshape > rect {
      fill-opacity: 0.15;
      stroke: hsl(0, 0%, 70%);
      stroke-opacity: 1;
    }
    .series.expanded > .nodeshape > use {
      visibility: hidden;
    }

    /**
 * TODO: Simplify this by applying a stable class name to all <g>
 * elements that currently have either the nodeshape or annotation-node classes.
 */
    .series > .nodeshape > use,
    .series > .annotation-node > use {
      stroke: #ccc;
    }
    .series.highlighted > .nodeshape > use,
    .series.highlighted > .annotation-node > use {
      stroke-width: 2;
    }
    .series.selected > .nodeshape > use,
    .series.selected > .annotation-node > use {
      stroke: red;
      stroke-width: 2;
    }

    .series.selected > .nodeshape > rect {
      stroke: red;
      stroke-width: 2;
    }

    .annotation.series.selected > .annotation-node > use {
      stroke: red;
      stroke-width: 2;
    }

    /* --- Bridge Node --- */
    .bridge > .nodeshape > rect {
      stroke: #f0f;
      opacity: 0.2;
      display: none;
    }

    /* --- Structural Elements --- */
    .edge > path.edgeline.structural {
      stroke: #f0f;
      opacity: 0.2;
      display: none;
    }

    /* Reference Edge */
    .edge > path.edgeline.referenceedge {
      stroke: #ffb74d;
      opacity: 1;
    }

    /* --- Series Nodes --- */

    /* Hide the rect for a series' annotation. */
    .series > .annotation-node > rect {
      display: none;
    }

    /* --- Node label --- */

    .node > text.nodelabel {
      cursor: pointer;
      fill: #444;
    }

    .meta.expanded > text.nodelabel {
      font-size: 9px;
    }

    .series > text.nodelabel {
      font-size: 8px;
    }

    .op > text.nodelabel {
      font-size: 6px;
    }

    .bridge > text.nodelabel {
      display: none;
    }

    .node.meta.expanded > text.nodelabel {
      cursor: normal;
    }

    .annotation.meta.highlighted > text.annotation-label {
      fill: #50a3f7;
    }

    .annotation.meta.selected > text.annotation-label {
      fill: #4285f4;
    }

    /* --- Annotation --- */

    /* only applied for annotations that are not summary or constant.
(.summary, .constant gets overridden below) */
    .annotation > .annotation-node > * {
      stroke-width: 0.5;
      stroke-dasharray: 1, 1;
    }

    .annotation.summary > .annotation-node > *,
    .annotation.constant > .annotation-node > * {
      stroke-width: 1;
      stroke-dasharray: none;
    }

    .annotation > .annotation-edge {
      fill: none;
      stroke: #aaa;
      stroke-width: 0.5;
      marker-end: url(#annotation-arrowhead);
    }

    .faded .annotation > .annotation-edge {
      marker-end: url(#annotation-arrowhead-faded);
    }

    .annotation > .annotation-edge.refline {
      marker-start: url(#ref-annotation-arrowhead);
    }

    .faded .annotation > .annotation-edge.refline {
      marker-start: url(#ref-annotation-arrowhead-faded);
    }

    .annotation > .annotation-control-edge {
      stroke-dasharray: 1, 1;
    }

    #annotation-arrowhead {
      fill: #aaa;
    }

    #annotation-arrowhead-faded {
      fill: #e0d4b3;
    }

    #ref-annotation-arrowhead {
      fill: #aaa;
    }

    #ref-annotation-arrowhead-faded {
      fill: #e0d4b3;
    }

    .annotation > .annotation-label {
      font-size: 5px;
      cursor: pointer;
    }
    .annotation > .annotation-label.annotation-ellipsis {
      cursor: default;
    }

    /* Hide annotations on expanded meta nodes since they're redundant. */
    .expanded > .in-annotations,
    .expanded > .out-annotations {
      display: none;
    }

    /* --- Annotation: Constant --- */

    .constant > .annotation-node > ellipse {
      cursor: pointer;
      fill: white;
      stroke: #848484;
    }

    .constant.selected > .annotation-node > ellipse {
      fill: white;
      stroke: red;
    }

    .constant.highlighted > .annotation-node > ellipse {
      stroke-width: 1.5;
    }

    /* --- Annotation: Summary --- */

    .summary > .annotation-node > ellipse {
      cursor: pointer;
      fill: #db4437;
      stroke: #db4437;
    }

    .summary.selected > .annotation-node > ellipse {
      fill: #a52714;
      stroke: #a52714;
    }

    .summary.highlighted > .annotation-node > ellipse {
      stroke-width: 1.5;
    }

    /* --- Edge --- */

    .edge > path.edgeline {
      fill: none;
      stroke: #bbb;
      stroke-linecap: round;
      stroke-width: 0.75;
    }

    .edge .selectableedge {
      cursor: pointer;
    }

    .selectededge > path.edgeline {
      cursor: default;
      stroke: #f00;
    }

    .edge.selectededge text {
      fill: #000;
    }

    /* Labels showing tensor shapes on edges */
    .edge > text {
      font-size: 3.5px;
      fill: #666;
    }

    .dataflow-arrowhead {
      fill: #bbb;
    }

    .reference-arrowhead {
      fill: #ffb74d;
    }

    .selected-arrowhead {
      fill: #f00;
    }

    .edge .control-dep {
      stroke-dasharray: 2, 2;
    }

    /* --- Group node expand/collapse button --- */

    /* Hides expand/collapse buttons when a node isn't expanded or highlighted. Using
   incredibly small opacity so that the bounding box of the <g> parent still takes
   this container into account even when it isn't visible */
    .node:not(.highlighted):not(.expanded) > .nodeshape > .buttoncontainer {
      opacity: 0.01;
    }
    .node.highlighted > .nodeshape > .buttoncontainer {
      cursor: pointer;
    }
    .buttoncircle {
      fill: #e7811d;
    }
    .buttoncircle:hover {
      fill: #b96717;
    }
    .expandbutton,
    .collapsebutton {
      stroke: white;
    }
    /* Do not let the path elements in the button take pointer focus */
    .node > .nodeshape > .buttoncontainer > .expandbutton,
    .node > .nodeshape > .buttoncontainer > .collapsebutton {
      pointer-events: none;
    }
    /* Only show the expand button when a node is collapsed and only show the
   collapse button when a node is expanded. */
    .node.expanded > .nodeshape > .buttoncontainer > .expandbutton {
      display: none;
    }
    .node:not(.expanded) > .nodeshape > .buttoncontainer > .collapsebutton {
      display: none;
    }

    .health-pill-stats {
      font-size: 4px;
      text-anchor: middle;
    }

    .health-pill rect {
      filter: url(#health-pill-shadow);
      rx: 3;
      ry: 3;
    }

    .titleContainer {
      position: relative;
      top: 20px;
    }

    .title,
    .auxTitle,
    .functionLibraryTitle {
      position: absolute;
    }

    #minimap {
      position: absolute;
      right: 20px;
      bottom: 20px;
    }

    .context-menu {
      position: absolute;
      display: none;
      background-color: #e2e2e2;
      border-radius: 2px;
      font-size: 14px;
      min-width: 150px;
      border: 1px solid #d4d4d4;
    }

    .context-menu ul {
      list-style-type: none;
      margin: 0;
      padding: 0;
      cursor: default;
    }

    .context-menu ul li {
      padding: 4px 16px;
    }

    .context-menu ul li:hover {
      background-color: #f3913e;
      color: white;
    }
  </style>
  <div class="titleContainer">
    <div id="title" class="title">Main Graph</div>
    <div id="auxTitle" class="auxTitle">Auxiliary Nodes</div>
    <div id="functionLibraryTitle" class="functionLibraryTitle">Functions</div>
  </div>
  <svg id="svg">
    <defs>
      <!-- Arrow heads for reference edge paths of different predefined sizes per color. -->
      <path
        id="reference-arrowhead-path"
        d="M 0,0 L 10,5 L 0,10 C 3,7 3,3 0,0"
      ></path>
      <marker
        class="reference-arrowhead"
        id="reference-arrowhead-small"
        viewBox="0 0 10 10"
        markerWidth="5"
        markerHeight="5"
        refX="2"
        refY="5"
        orient="auto-start-reverse"
        markerUnits="userSpaceOnUse"
      >
        <use xlink:href="#reference-arrowhead-path"></use>
      </marker>
      <marker
        class="reference-arrowhead"
        id="reference-arrowhead-medium"
        viewBox="0 0 10 10"
        markerWidth="13"
        markerHeight="13"
        refX="2"
        refY="5"
        orient="auto-start-reverse"
        markerUnits="userSpaceOnUse"
      >
        <use xlink:href="#reference-arrowhead-path"></use>
      </marker>
      <marker
        class="reference-arrowhead"
        id="reference-arrowhead-large"
        viewBox="0 0 10 10"
        markerWidth="16"
        markerHeight="16"
        refX="2"
        refY="5"
        orient="auto-start-reverse"
        markerUnits="userSpaceOnUse"
      >
        <use xlink:href="#reference-arrowhead-path"></use>
      </marker>
      <marker
        class="reference-arrowhead"
        id="reference-arrowhead-xlarge"
        viewBox="0 0 10 10"
        markerWidth="20"
        markerHeight="20"
        refX="2"
        refY="5"
        orient="auto-start-reverse"
        markerUnits="userSpaceOnUse"
      >
        <use xlink:href="#reference-arrowhead-path"></use>
      </marker>

      <!-- Arrow heads for dataflow edge paths of different predefined sizes per color. -->
      <path
        id="dataflow-arrowhead-path"
        d="M 0,0 L 10,5 L 0,10 C 3,7 3,3 0,0"
      ></path>
      <marker
        class="dataflow-arrowhead"
        id="dataflow-arrowhead-small"
        viewBox="0 0 10 10"
        markerWidth="5"
        markerHeight="5"
        refX="2"
        refY="5"
        orient="auto-start-reverse"
        markerUnits="userSpaceOnUse"
      >
        <use xlink:href="#dataflow-arrowhead-path"></use>
      </marker>
      <marker
        class="dataflow-arrowhead"
        id="dataflow-arrowhead-medium"
        viewBox="0 0 10 10"
        markerWidth="13"
        markerHeight="13"
        refX="2"
        refY="5"
        orient="auto-start-reverse"
        markerUnits="userSpaceOnUse"
      >
        <use xlink:href="#dataflow-arrowhead-path"></use>
      </marker>
      <marker
        class="dataflow-arrowhead"
        id="dataflow-arrowhead-large"
        viewBox="0 0 10 10"
        markerWidth="16"
        markerHeight="16"
        refX="2"
        refY="5"
        orient="auto-start-reverse"
        markerUnits="userSpaceOnUse"
      >
        <use xlink:href="#dataflow-arrowhead-path"></use>
      </marker>
      <marker
        class="dataflow-arrowhead"
        id="dataflow-arrowhead-xlarge"
        viewBox="0 0 10 10"
        markerWidth="20"
        markerHeight="20"
        refX="2"
        refY="5"
        orient="auto-start-reverse"
        markerUnits="userSpaceOnUse"
      >
        <use xlink:href="#dataflow-arrowhead-path"></use>
      </marker>

      <!-- Arrow head for annotation edge paths. -->
      <marker
        id="annotation-arrowhead"
        markerWidth="5"
        markerHeight="5"
        refX="5"
        refY="2.5"
        orient="auto"
      >
        <path d="M 0,0 L 5,2.5 L 0,5 L 0,0"></path>
      </marker>
      <marker
        id="annotation-arrowhead-faded"
        markerWidth="5"
        markerHeight="5"
        refX="5"
        refY="2.5"
        orient="auto"
      >
        <path d="M 0,0 L 5,2.5 L 0,5 L 0,0"></path>
      </marker>
      <marker
        id="ref-annotation-arrowhead"
        markerWidth="5"
        markerHeight="5"
        refX="0"
        refY="2.5"
        orient="auto"
      >
        <path d="M 5,0 L 0,2.5 L 5,5 L 5,0"></path>
      </marker>
      <marker
        id="ref-annotation-arrowhead-faded"
        markerWidth="5"
        markerHeight="5"
        refX="0"
        refY="2.5"
        orient="auto"
      >
        <path d="M 5,0 L 0,2.5 L 5,5 L 5,0"></path>
      </marker>
      <!-- Template for an Op node ellipse. -->
      <ellipse
        id="op-node-stamp"
        rx="7.5"
        ry="3"
        stroke="inherit"
        fill="inherit"
      ></ellipse>
      <!-- Template for an Op node annotation ellipse (smaller). -->
      <ellipse
        id="op-node-annotation-stamp"
        rx="5"
        ry="2"
        stroke="inherit"
        fill="inherit"
      ></ellipse>
      <!-- Vertically stacked series of Op nodes when unexpanded. -->
      <g id="op-series-vertical-stamp">
        <use xlink:href="#op-node-stamp" x="8" y="9"></use>
        <use xlink:href="#op-node-stamp" x="8" y="6"></use>
        <use xlink:href="#op-node-stamp" x="8" y="3"></use>
      </g>
      <!-- Horizontally stacked series of Op nodes when unexpanded. -->
      <g id="op-series-horizontal-stamp">
        <use xlink:href="#op-node-stamp" x="16" y="4"></use>
        <use xlink:href="#op-node-stamp" x="12" y="4"></use>
        <use xlink:href="#op-node-stamp" x="8" y="4"></use>
      </g>
      <!-- Horizontally stacked series of Op nodes for annotation. -->
      <g id="op-series-annotation-stamp">
        <use xlink:href="#op-node-annotation-stamp" x="9" y="2"></use>
        <use xlink:href="#op-node-annotation-stamp" x="7" y="2"></use>
        <use xlink:href="#op-node-annotation-stamp" x="5" y="2"></use>
      </g>
      <svg
        id="summary-icon"
        fill="#848484"
        height="12"
        viewBox="0 0 24 24"
        width="12"
      >
        <path
          d="M19 3H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zM9 17H7v-7h2v7zm4 0h-2V7h2v10zm4 0h-2v-4h2v4z"
        ></path>
      </svg>

      <!-- Hatch patterns for faded out nodes. -->
      <pattern
        id="rectHatch"
        patternTransform="rotate(45 0 0)"
        width="5"
        height="5"
        patternUnits="userSpaceOnUse"
      >
        <line x1="0" y1="0" x2="0" y2="5" style="stroke-width: 1"></line>
      </pattern>
      <pattern
        id="ellipseHatch"
        patternTransform="rotate(45 0 0)"
        width="2"
        height="2"
        patternUnits="userSpaceOnUse"
      >
        <line x1="0" y1="0" x2="0" y2="2" style="stroke-width: 1"></line>
      </pattern>

      <!-- A shadow for health pills. -->
      <filter
        id="health-pill-shadow"
        x="-40%"
        y="-40%"
        width="180%"
        height="180%"
      >
        <feGaussianBlur in="SourceAlpha" stdDeviation="0.8"></feGaussianBlur>
        <feOffset dx="0" dy="0" result="offsetblur"></feOffset>
        <feFlood flood-color="#000000"></feFlood>
        <feComposite in2="offsetblur" operator="in"></feComposite>
        <feMerge>
          <feMergeNode></feMergeNode>
          <feMergeNode in="SourceGraphic"></feMergeNode>
        </feMerge>
      </filter>
    </defs>
    <!-- Make a large rectangle that fills the svg space so that
  zoom events get captured on safari -->
    <rect fill="white" width="10000" height="10000"></rect>
    <g id="root"></g>
  </svg>
  <tf-graph-minimap id="minimap"></tf-graph-minimap>
  <div id="contextMenu" class="context-menu"></div>
`;

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    let TfGraphScene2 = class TfGraphScene2 extends LegacyElementMixin(PolymerElement) {
        constructor() {
            super(...arguments);
            /** Keeps track of if the graph has been zoomed/panned since loading */
            this._zoomed = false;
            /**
             * Keeps track of the starting coordinates of a graph zoom/pan.
             *
             * @private {{x: number, y: number}?}
             */
            this._zoomStartCoords = null;
            /**
             * Keeps track of the current coordinates of a graph zoom/pan
             *
             * @private {{x: number, y: number}?}
             */
            this._zoomTransform = null;
            /** Maximum distance of a zoom event for it to be interpreted as a click */
            this._maxZoomDistanceForClick = 20;
            /*
             * Dictionary for easily stylizing nodes when state changes.
             * _nodeGroupIndex[nodeName] = d3_selection of the nodeGroup
             */
            this._nodeGroupIndex = {};
            /*
             * Dictionary for easily stylizing annotation nodes when state changes.
             * _annotationGroupIndex[nodeName][hostNodeName] =
             *   d3_selection of the annotationGroup
             */
            this._annotationGroupIndex = {};
            /*
             * Dictionary for easily stylizing edges when state changes.
             * _edgeGroupIndex[edgeName] = d3_selection of the edgeGroup
             */
            this._edgeGroupIndex = {};
            /**
             * Max font size for metanode label strings.
             */
            this.maxMetanodeLabelLengthFontSize = 9;
            /**
             * Min font size for metanode label strings.
             */
            this.minMetanodeLabelLengthFontSize = 6;
            /**
             * Metanode label strings longer than this are given smaller fonts.
             */
            this.maxMetanodeLabelLengthLargeFont = 11;
            /**
             * Metanode label strings longer than this are truncated with ellipses.
             */
            this.maxMetanodeLabelLength = 18;
        }
        getNode(nodeName) {
            return this.renderHierarchy.getRenderNodeByName(nodeName);
        }
        isNodeExpanded(node) {
            return node.expanded;
        }
        setNodeExpanded(renderNode) {
            this._build(this.renderHierarchy);
            this._updateLabels(!this._zoomed);
        }
        /**
         * Pans to a node. Assumes that the node exists.
         * @param nodeName {string} The name of the node to pan to.
         */
        panToNode(nodeName) {
            const zoomed = panToNode(nodeName, this.$.svg, this.$.root, this._zoom);
            if (zoomed) {
                this._zoomed = true;
            }
        }
        /**
         * Returns the outer-most SVG that renders the graph.
         */
        getGraphSvgRoot() {
            return this.$.svg;
        }
        /**
         * @returns {!HTMLElement}
         */
        getContextMenu() {
            return this.$.contextMenu;
        }
        /**
         * Resets the state of the component. Called whenever the whole graph
         * (dataset) changes.
         */
        _resetState() {
            // Reset the state of the component.
            this._nodeGroupIndex = {};
            this._annotationGroupIndex = {};
            this._edgeGroupIndex = {};
            this._updateLabels(false);
            // Remove all svg elements under the 'root' svg group.
            select(this.$.svg).select('#root').selectAll('*').remove();
            // And the defs.
            removeGradientDefinitions(this.$.svg);
        }
        /** Main method for building the scene */
        _build(renderHierarchy) {
            this.templateIndex = renderHierarchy.hierarchy.getTemplateIndex();
            time$3('tf-graph-scene (layout):', function () {
                // layout the scene for this meta / series node
                layoutScene(renderHierarchy.root);
            }.bind(this));
            time$3('tf-graph-scene (build scene):', function () {
                buildGroupForScene(select(this.$.root), renderHierarchy.root, this);
                addGraphClickListener(this.$.svg, this);
                this._updateInputTrace();
            }.bind(this));
            // Update the minimap again when the graph is done animating.
            setTimeout(function () {
                this._updateHealthPills(this.nodeNamesToHealthPills, this.healthPillStepIndex);
                this.minimap.update();
            }.bind(this), PARAMS$1.animation.duration);
        }
        ready() {
            super.ready();
            this._zoom = zoom()
                .on('end', function () {
                if (this._zoomStartCoords) {
                    // Calculate the total distance dragged during the zoom event.
                    // If it is sufficiently small, then fire an event indicating
                    // that zooming has ended. Otherwise wait to fire the zoom end
                    // event, so that a mouse click registered as part of this zooming
                    // is ignored (as this mouse click was part of a zooming, and should
                    // not be used to indicate an actual click on the graph).
                    var dragDistance = Math.sqrt(Math.pow(this._zoomStartCoords.x - this._zoomTransform.x, 2) +
                        Math.pow(this._zoomStartCoords.y - this._zoomTransform.y, 2));
                    if (dragDistance < this._maxZoomDistanceForClick) {
                        this._fireEnableClick();
                    }
                    else {
                        setTimeout(this._fireEnableClick.bind(this), 50);
                    }
                }
                this._zoomStartCoords = null;
            }.bind(this))
                .on('zoom', function () {
                // Store the coordinates of the zoom event.
                this._zoomTransform = event.transform;
                // If this is the first zoom event after a zoom-end, then
                // store the coordinates as the start coordinates as well,
                // and fire an event to indicate that zooming has started.
                // This doesn't use the zoomstart event, as d3 sends this
                // event on mouse-down, even if there has been no dragging
                // done to translate the graph around.
                if (!this._zoomStartCoords) {
                    this._zoomStartCoords = this._zoomTransform;
                    this.fire('disable-click');
                }
                this._zoomed = true;
                select(this.$.root).attr('transform', event.transform);
                // Notify the minimap.
                this.minimap.zoom(event.transform);
            }.bind(this));
            select(this.$.svg)
                .call(this._zoom)
                .on('dblclick.zoom', null);
            select(window).on('resize', function () {
                // Notify the minimap that the user's window was resized.
                // The minimap will figure out the new dimensions of the main svg
                // and will use the existing translate and scale params.
                this.minimap.zoom();
            }.bind(this));
            // Initialize the minimap.
            this.minimap = this.$.minimap.init(this.$.svg, this.$.root, this._zoom, PARAMS$1.minimap.size, PARAMS$1.subscene.meta.labelHeight);
        }
        attached() {
            this.set('_isAttached', true);
        }
        detached() {
            this.set('_isAttached', false);
        }
        _renderHierarchyChanged() {
            var renderHierarchy = this.renderHierarchy;
            this._hasRenderHierarchyBeenFitOnce = false;
            this._resetState();
            this._build(renderHierarchy);
        }
        // Animation and fitting must come after the observer for the hierarchy changing because we must
        // first build the render hierarchy.
        _animateAndFit() {
            var isAttached = this._isAttached;
            var renderHierarchy = this.renderHierarchy;
            if (this._hasRenderHierarchyBeenFitOnce || !isAttached) {
                // Do not animate and fit if the scene has already fitted this render hierarchy once. Or if
                // the graph dashboard is not attached (in which case the scene lacks DOM info for fitting).
                return;
            }
            // Fit to screen after the graph is done animating.
            setTimeout(this.fit.bind(this), PARAMS$1.animation.duration);
        }
        _updateLabels(showLabels) {
            var mainGraphTitleElement = this.$$('.title');
            var titleStyle = mainGraphTitleElement.style;
            var auxTitleElement = this.$$('.auxTitle');
            var auxTitleStyle = auxTitleElement.style;
            var functionLibraryTitleStyle = this.$$('.functionLibraryTitle').style;
            const root = select(this.$.svg);
            var core = root
                .select('.' +
                Class$2.Scene.GROUP +
                '>.' +
                Class$2.Scene.CORE)
                .node();
            // Only show labels if the graph is fully loaded.
            if (showLabels && core && this.progress && this.progress.value === 100) {
                var aux = root
                    .select('.' +
                    Class$2.Scene.GROUP +
                    '>.' +
                    Class$2.Scene.INEXTRACT)
                    .node() ||
                    root
                        .select('.' +
                        Class$2.Scene.GROUP +
                        '>.' +
                        Class$2.Scene.OUTEXTRACT)
                        .node();
                var coreX = core.getCTM().e;
                var auxX = aux ? aux.getCTM().e : null;
                titleStyle.display = 'inline';
                titleStyle.left = coreX + 'px';
                if (auxX !== null && auxX !== coreX) {
                    auxTitleStyle.display = 'inline';
                    // Make sure that the aux title is positioned rightwards enough so as to
                    // prevent overlap with the main graph title.
                    auxX = Math.max(coreX + mainGraphTitleElement.getBoundingClientRect().width, auxX);
                    auxTitleStyle.left = auxX + 'px';
                }
                else {
                    auxTitleStyle.display = 'none';
                }
                let functionLibrary = root
                    .select('.' +
                    Class$2.Scene.GROUP +
                    '>.' +
                    Class$2.Scene.FUNCTION_LIBRARY)
                    .node();
                let functionLibraryX = functionLibrary
                    ? functionLibrary.getCTM().e
                    : null;
                if (functionLibraryX !== null && functionLibraryX !== auxX) {
                    functionLibraryTitleStyle.display = 'inline';
                    // Make sure that the function library title is positioned rightwards
                    // enough so as to prevent overlap with other content.
                    functionLibraryX = Math.max(auxX + auxTitleElement.getBoundingClientRect().width, functionLibraryX);
                    functionLibraryTitleStyle.left = functionLibraryX + 'px';
                }
                else {
                    functionLibraryTitleStyle.display = 'none';
                }
            }
            else {
                titleStyle.display = 'none';
                auxTitleStyle.display = 'none';
                functionLibraryTitleStyle.display = 'none';
            }
        }
        /**
         * Called whenever the user changed the 'color by' option in the
         * UI controls.
         */
        _colorByChanged() {
            if (this.renderHierarchy != null) {
                // We iterate through each svg node and update its state.
                lodash.each(this._nodeGroupIndex, (nodeGroup, nodeName) => {
                    this._updateNodeState(nodeName);
                });
                // Notify also the minimap.
                this.minimap.update();
            }
        }
        fit() {
            this._hasRenderHierarchyBeenFitOnce = true;
            fit$2(this.$.svg, this.$.root, this._zoom, function () {
                this._zoomed = false;
            }.bind(this));
        }
        isNodeSelected(n) {
            return n === this.selectedNode;
        }
        isNodeHighlighted(n) {
            return n === this.highlightedNode;
        }
        addAnnotationGroup(a, d, selection) {
            var an = a.node.name;
            this._annotationGroupIndex[an] = this._annotationGroupIndex[an] || {};
            this._annotationGroupIndex[an][d.node.name] = selection;
        }
        getAnnotationGroupsIndex(a) {
            return this._annotationGroupIndex[a];
        }
        removeAnnotationGroup(a, d) {
            delete this._annotationGroupIndex[a.node.name][d.node.name];
        }
        addNodeGroup(n, selection) {
            this._nodeGroupIndex[n] = selection;
        }
        getNodeGroup(n) {
            return this._nodeGroupIndex[n];
        }
        removeNodeGroup(n) {
            delete this._nodeGroupIndex[n];
        }
        addEdgeGroup(n, selection) {
            this._edgeGroupIndex[n] = selection;
        }
        getEdgeGroup(e) {
            return this._edgeGroupIndex[e];
        }
        _updateHealthPills() {
            var nodeNamesToHealthPills = this.nodeNamesToHealthPills;
            var healthPillStepIndex = this.healthPillStepIndex;
            addHealthPills(this.$.svg, nodeNamesToHealthPills, healthPillStepIndex);
        }
        /**
         * Update node and annotation node of the given name.
         * @param  {String} n node name
         */
        _updateNodeState(n) {
            var node = this.getNode(n);
            var nodeGroup = this.getNodeGroup(n);
            if (nodeGroup) {
                stylize$1(nodeGroup, node, this);
            }
            if (node.node.type === NodeType.META &&
                node.node.associatedFunction &&
                !node.isLibraryFunction) {
                // The node is that of a function call. Also link the node within the
                // function library. This clarifies to the user that the library function
                // is being used.
                var libraryFunctionNodeName = FUNCTION_LIBRARY_NODE_PREFIX +
                    node.node.associatedFunction;
                var functionGroup = select('.' +
                    Class$2.Scene.GROUP +
                    '>.' +
                    Class$2.Scene.FUNCTION_LIBRARY +
                    ' g[data-name="' +
                    libraryFunctionNodeName +
                    '"]');
                stylize$1(functionGroup, node, this);
            }
            var annotationGroupIndex = this.getAnnotationGroupsIndex(n);
            lodash.each(annotationGroupIndex, (aGroup, hostName) => {
                stylize$1(aGroup, node, this, Class$2.Annotation.NODE);
            });
        }
        /**
         * Handles new node selection. 1) Updates the selected-state of each node,
         * 2) triggers input tracing.
         * @param selectedNode {string} The name of the newly selected node.
         * @param oldSelectedNode {string} The name of the previously selected node.
         * @private
         */
        _selectedNodeChanged(selectedNode, oldSelectedNode) {
            if (selectedNode === oldSelectedNode) {
                return;
            }
            if (oldSelectedNode) {
                this._updateNodeState(oldSelectedNode);
            }
            if (!selectedNode) {
                return;
            }
            // Update the minimap to reflect the highlighted (selected) node.
            this.minimap.update();
            var node = this.renderHierarchy.hierarchy.node(selectedNode);
            var nodeParents = [];
            // Create list of all metanode parents of the selected node.
            while (node.parentNode != null &&
                node.parentNode.name != ROOT_NAME) {
                node = node.parentNode;
                nodeParents.push(node.name);
            }
            // Ensure each parent metanode is built and expanded.
            var topParentNodeToBeExpanded;
            lodash.forEachRight(nodeParents, (parentName) => {
                this.renderHierarchy.buildSubhierarchy(parentName);
                var renderNode = this.renderHierarchy.getRenderNodeByName(parentName);
                if (renderNode.node.isGroupNode && !renderNode.expanded) {
                    renderNode.expanded = true;
                    if (!topParentNodeToBeExpanded) {
                        topParentNodeToBeExpanded = renderNode;
                    }
                }
            });
            // If any expansion was needed to display this selected node, then
            // inform the scene of the top-most expansion.
            if (topParentNodeToBeExpanded) {
                this.setNodeExpanded(topParentNodeToBeExpanded);
                this._zoomed = true;
            }
            if (selectedNode) {
                this._updateNodeState(selectedNode);
            }
            // Give time for any expanding to finish before panning to a node.
            // Otherwise, the pan will be computed from incorrect measurements.
            setTimeout(() => {
                this.panToNode(selectedNode);
            }, PARAMS$1.animation.duration);
        }
        _highlightedNodeChanged(highlightedNode, oldHighlightedNode) {
            if (highlightedNode === oldHighlightedNode) {
                return;
            }
            if (highlightedNode) {
                this._updateNodeState(highlightedNode);
            }
            if (oldHighlightedNode) {
                this._updateNodeState(oldHighlightedNode);
            }
        }
        _onZoomChanged() {
            this._updateLabels(!this._zoomed);
        }
        _fireEnableClick() {
            this.fire('enable-click');
        }
        // When renderHierarchy changes, we need to first build the new SVG based
        // on the new hierarchy (and it is asynchronous). We will let that observer
        // update the input trace.
        _updateInputTrace() {
            updateInputTrace(this.getGraphSvgRoot(), this.renderHierarchy, this.selectedNode, this.traceInputs);
        }
    };
    TfGraphScene2.template = template$j;
    __decorate([
        property({ type: Object }),
        __metadata("design:type", RenderGraphInfo)
    ], TfGraphScene2.prototype, "renderHierarchy", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfGraphScene2.prototype, "name", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfGraphScene2.prototype, "colorBy", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfGraphScene2.prototype, "traceInputs", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfGraphScene2.prototype, "_hasRenderHierarchyBeenFitOnce", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfGraphScene2.prototype, "_isAttached", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraphScene2.prototype, "_zoom", void 0);
    __decorate([
        property({
            type: String,
            observer: '_highlightedNodeChanged',
        }),
        __metadata("design:type", String)
    ], TfGraphScene2.prototype, "highlightedNode", void 0);
    __decorate([
        property({
            type: String,
            observer: '_selectedNodeChanged',
        }),
        __metadata("design:type", String)
    ], TfGraphScene2.prototype, "selectedNode", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraphScene2.prototype, "handleEdgeSelected", void 0);
    __decorate([
        property({
            type: Boolean,
            observer: '_onZoomChanged',
        }),
        __metadata("design:type", Boolean)
    ], TfGraphScene2.prototype, "_zoomed", void 0);
    __decorate([
        property({
            type: Object,
        }),
        __metadata("design:type", Object)
    ], TfGraphScene2.prototype, "_zoomStartCoords", void 0);
    __decorate([
        property({
            type: Object,
        }),
        __metadata("design:type", Object)
    ], TfGraphScene2.prototype, "_zoomTransform", void 0);
    __decorate([
        property({
            type: Number,
        }),
        __metadata("design:type", Number)
    ], TfGraphScene2.prototype, "_maxZoomDistanceForClick", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Function)
    ], TfGraphScene2.prototype, "templateIndex", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraphScene2.prototype, "minimap", void 0);
    __decorate([
        property({
            type: Object,
        }),
        __metadata("design:type", Object)
    ], TfGraphScene2.prototype, "_nodeGroupIndex", void 0);
    __decorate([
        property({
            type: Object,
        }),
        __metadata("design:type", Object)
    ], TfGraphScene2.prototype, "_annotationGroupIndex", void 0);
    __decorate([
        property({
            type: Object,
        }),
        __metadata("design:type", Object)
    ], TfGraphScene2.prototype, "_edgeGroupIndex", void 0);
    __decorate([
        property({
            type: Number,
        }),
        __metadata("design:type", Number)
    ], TfGraphScene2.prototype, "maxMetanodeLabelLengthFontSize", void 0);
    __decorate([
        property({
            type: Number,
        }),
        __metadata("design:type", Number)
    ], TfGraphScene2.prototype, "minMetanodeLabelLengthFontSize", void 0);
    __decorate([
        property({
            type: Number,
        }),
        __metadata("design:type", Number)
    ], TfGraphScene2.prototype, "maxMetanodeLabelLengthLargeFont", void 0);
    __decorate([
        property({
            type: Number,
        }),
        __metadata("design:type", Number)
    ], TfGraphScene2.prototype, "maxMetanodeLabelLength", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraphScene2.prototype, "progress", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfGraphScene2.prototype, "nodeContextMenuItems", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraphScene2.prototype, "nodeNamesToHealthPills", void 0);
    __decorate([
        property({ type: Number }),
        __metadata("design:type", Number)
    ], TfGraphScene2.prototype, "healthPillStepIndex", void 0);
    __decorate([
        observe('renderHierarchy'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfGraphScene2.prototype, "_renderHierarchyChanged", null);
    __decorate([
        observe('_isAttached', 'renderHierarchy'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfGraphScene2.prototype, "_animateAndFit", null);
    __decorate([
        observe('colorBy'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfGraphScene2.prototype, "_colorByChanged", null);
    __decorate([
        observe('nodeNamesToHealthPills', 'healthPillStepIndex'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfGraphScene2.prototype, "_updateHealthPills", null);
    __decorate([
        observe('traceInputs', 'selectedNode'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfGraphScene2.prototype, "_updateInputTrace", null);
    TfGraphScene2 = __decorate([
        customElement('tf-graph-scene')
    ], TfGraphScene2);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    let TfGraph = class TfGraph extends LegacyElementMixin(PolymerElement) {
        constructor() {
            super(...arguments);
            this._renderDepth = 1;
            this._allowGraphSelect = true;
            this.edgeWidthFunction = '';
            this.handleNodeSelected = '';
            this.edgeLabelFunction = '';
            this.handleEdgeSelected = '';
        }
        /**
         * Pans to a node. Assumes that the node exists.
         * @param nodeName {string} The name of the node to pan to.
         */
        panToNode(nodeName) {
            this.$$('tf-graph-scene').panToNode(nodeName);
        }
        _buildNewRenderHierarchy() {
            var graphHierarchy = this.graphHierarchy;
            if (!graphHierarchy)
                return;
            this._buildRenderHierarchy(graphHierarchy);
        }
        _statsChanged() {
            var stats = this.stats;
            var devicesForStats = this.devicesForStats;
            if (this.graphHierarchy) {
                if (stats && devicesForStats) {
                    joinStatsInfoWithGraph(this.basicGraph, stats, devicesForStats);
                    joinAndAggregateStats(this.graphHierarchy);
                }
                // Recompute the rendering information.
                this._buildRenderHierarchy(this.graphHierarchy);
            }
        }
        ready() {
            super.ready();
            this.addEventListener('graph-select', this._graphSelected.bind(this));
            this.addEventListener('disable-click', this._disableClick.bind(this));
            this.addEventListener('enable-click', this._enableClick.bind(this));
            // Nodes
            this.addEventListener('node-toggle-expand', this._nodeToggleExpand.bind(this));
            this.addEventListener('node-select', this._nodeSelected.bind(this));
            this.addEventListener('node-highlight', this._nodeHighlighted.bind(this));
            this.addEventListener('node-unhighlight', this._nodeUnhighlighted.bind(this));
            this.addEventListener('node-toggle-extract', this._nodeToggleExtract.bind(this));
            this.addEventListener('node-toggle-seriesgroup', this._nodeToggleSeriesGroup.bind(this));
            // Edges
            this.addEventListener('edge-select', this._edgeSelected.bind(this));
            // Annotations
            /* Note: currently highlighting/selecting annotation node has the same
             * behavior as highlighting/selecting actual node so we point to the same
             * set of event listeners.  However, we might redesign this to be a bit
             * different.
             */
            this.addEventListener('annotation-select', this._nodeSelected.bind(this));
            this.addEventListener('annotation-highlight', this._nodeHighlighted.bind(this));
            this.addEventListener('annotation-unhighlight', this._nodeUnhighlighted.bind(this));
        }
        _buildRenderHierarchy(graphHierarchy) {
            time$3('new tf_graph_render.Hierarchy', function () {
                if (graphHierarchy.root.type !== NodeType.META) {
                    // root must be metanode but sometimes Polymer's dom-if has not
                    // remove tf-graph element yet in <tf-node-info>
                    // and thus mistakenly pass non-metanode to this module.
                    return;
                }
                var renderGraph = new RenderGraphInfo(graphHierarchy, !!this.stats /** displayingStats */);
                renderGraph.edgeLabelFunction = this.edgeLabelFunction;
                renderGraph.edgeWidthFunction = this.edgeWidthFunction;
                // Producing the 'color by' parameters to be consumed
                // by the tf-graph-controls panel. It contains information about the
                // min and max values and their respective colors, as well as list
                // of devices with their respective colors.
                function getColorParamsFromScale(scale) {
                    return {
                        minValue: scale.domain()[0],
                        maxValue: scale.domain()[1],
                        startColor: scale.range()[0],
                        endColor: scale.range()[1],
                    };
                }
                this._setColorByParams({
                    compute_time: getColorParamsFromScale(renderGraph.computeTimeScale),
                    memory: getColorParamsFromScale(renderGraph.memoryUsageScale),
                    device: lodash.map(renderGraph.deviceColorMap.domain(), function (deviceName) {
                        return {
                            device: deviceName,
                            color: renderGraph.deviceColorMap(deviceName),
                        };
                    }),
                    xla_cluster: lodash.map(renderGraph.xlaClusterColorMap.domain(), function (xlaClusterName) {
                        return {
                            xla_cluster: xlaClusterName,
                            color: renderGraph.xlaClusterColorMap(xlaClusterName),
                        };
                    }),
                });
                this._setRenderHierarchy(renderGraph);
                this.async(function () {
                    this.fire('rendered');
                });
            }.bind(this));
        }
        _getVisible(name) {
            if (!name) {
                return name;
            }
            return this.renderHierarchy.getNearestVisibleAncestor(name);
        }
        fit() {
            this.$.scene.fit();
        }
        _graphChanged() {
            // When a new graph is loaded, fire this event so that there is no
            // info-card being displayed for the previously-loaded graph.
            this.fire('graph-select');
        }
        _graphSelected(event) {
            // Graph selection is not allowed during an active zoom event, as the
            // click seen during a zoom/pan is part of the zooming and does not
            // indicate a user desire to click on a specific section of the graph.
            if (this._allowGraphSelect) {
                this.set('selectedNode', null);
                this.set('selectedEdge', null);
            }
            // Reset this variable as a bug in d3 zoom behavior can cause zoomend
            // callback not to be called if a right-click happens during a zoom event.
            this._allowGraphSelect = true;
        }
        _disableClick(event) {
            this._allowGraphSelect = false;
        }
        _enableClick(event) {
            this._allowGraphSelect = true;
        }
        // Called when the selected node changes, ie there is a new selected node or
        // the current one is unselected.
        _selectedNodeChanged() {
            var selectedNode = this.selectedNode;
            if (this.handleNodeSelected) {
                // A higher-level component provided a callback. Run it.
                this.handleNodeSelected(selectedNode);
            }
        }
        // Called when the selected edge changes, ie there is a new selected edge or
        // the current one is unselected.
        _selectedEdgeChanged() {
            var selectedEdge = this.selectedEdge;
            this._deselectPreviousEdge();
            // Visually mark this new edge as selected.
            if (selectedEdge) {
                this._lastSelectedEdgeGroup.classed(Class$2.Edge.SELECTED, true);
                // Update the color of the marker too if the edge has one.
                this._updateMarkerOfSelectedEdge(selectedEdge);
            }
            if (this.handleEdgeSelected) {
                // A higher-level component provided a callback. Run it.
                this.handleEdgeSelected(selectedEdge);
            }
        }
        // Called only when a new (non-null) node is selected.
        _nodeSelected(event) {
            if (this._allowGraphSelect) {
                this.set('selectedNode', event.detail.name);
            }
            // Reset this variable as a bug in d3 zoom behavior can cause zoomend
            // callback not to be called if a right-click happens during a zoom event.
            this._allowGraphSelect = true;
        }
        _edgeSelected(event) {
            if (this._allowGraphSelect) {
                this.set('_lastSelectedEdgeGroup', event.detail.edgeGroup);
                this.set('selectedEdge', event.detail.edgeData);
            }
            // Reset this variable as a bug in d3 zoom behavior can cause zoomend
            // callback not to be called if a right-click happens during a zoom event.
            this._allowGraphSelect = true;
        }
        _nodeHighlighted(event) {
            this.set('highlightedNode', event.detail.name);
        }
        _nodeUnhighlighted(event) {
            this.set('highlightedNode', null);
        }
        _nodeToggleExpand(event) {
            // Immediately select the node that is about to be expanded.
            this._nodeSelected(event);
            // Compute the sub-hierarchy scene.
            var nodeName = event.detail.name;
            var renderNode = this.renderHierarchy.getRenderNodeByName(nodeName);
            // Op nodes are not expandable.
            if (renderNode.node.type === NodeType.OP) {
                return;
            }
            this.renderHierarchy.buildSubhierarchy(nodeName);
            renderNode.expanded = !renderNode.expanded;
            // Expand the node with some delay so that the user can immediately see
            // the visual effect of selecting that node, before the expansion is
            // done.
            this.async(function () {
                this.$.scene.setNodeExpanded(renderNode);
            }, 75);
        }
        _nodeToggleExtract(event) {
            // Toggle the include setting of the specified node appropriately.
            var nodeName = event.detail.name;
            this.nodeToggleExtract(nodeName);
        }
        nodeToggleExtract(nodeName) {
            var renderNode = this.renderHierarchy.getRenderNodeByName(nodeName);
            if (renderNode.node.include == InclusionType.INCLUDE) {
                renderNode.node.include = InclusionType.EXCLUDE;
            }
            else if (renderNode.node.include == InclusionType.EXCLUDE) {
                renderNode.node.include = InclusionType.INCLUDE;
            }
            else {
                renderNode.node.include = this.renderHierarchy.isNodeAuxiliary(renderNode)
                    ? InclusionType.INCLUDE
                    : InclusionType.EXCLUDE;
            }
            // Rebuild the render hierarchy.
            this._buildRenderHierarchy(this.graphHierarchy);
        }
        _nodeToggleSeriesGroup(event) {
            // Toggle the group setting of the specified node appropriately.
            var nodeName = event.detail.name;
            this.nodeToggleSeriesGroup(nodeName);
        }
        nodeToggleSeriesGroup(nodeName) {
            // Toggle the group setting of the specified node appropriately.
            toggleNodeSeriesGroup(this.hierarchyParams.seriesMap, nodeName);
            // Rebuild the render hierarchy with the updated series grouping map.
            this.set('progress', {
                value: 0,
                msg: '',
            });
            var tracker = getTracker(this);
            var hierarchyTracker = getSubtaskTracker(tracker, 100, 'Namespace hierarchy');
            build$1(this.basicGraph, this.hierarchyParams, hierarchyTracker)
                .then(function (graphHierarchy) {
                this.set('graphHierarchy', graphHierarchy);
                this._buildRenderHierarchy(this.graphHierarchy);
            }.bind(this));
        }
        _deselectPreviousEdge() {
            const selectedSelector = '.' + Class$2.Edge.SELECTED;
            // Visually mark the previously selected edge (if any) as deselected.
            select(selectedSelector)
                .classed(Class$2.Edge.SELECTED, false)
                .each((d, i) => {
                // Reset its marker.
                if (d.label) {
                    const paths = select(this).selectAll('path.edgeline');
                    if (d.label.startMarkerId) {
                        paths.style('marker-start', `url(#${d.label.startMarkerId})`);
                    }
                    if (d.label.endMarkerId) {
                        paths.style('marker-end', `url(#${d.label.endMarkerId})`);
                    }
                }
            });
        }
        _updateMarkerOfSelectedEdge(selectedEdge) {
            if (selectedEdge.label) {
                // The marker will vary based on the direction of the edge.
                const markerId = selectedEdge.label.startMarkerId || selectedEdge.label.endMarkerId;
                if (markerId) {
                    // Find the corresponding marker for a selected edge.
                    const selectedMarkerId = markerId.replace('dataflow-', 'selected-');
                    let selectedMarker = this.$$('#' + selectedMarkerId);
                    if (!selectedMarker) {
                        // The marker for a selected edge of this size does not exist yet. Create it.
                        const originalMarker = this.$.scene.querySelector('#' + markerId);
                        selectedMarker = originalMarker.cloneNode(true);
                        selectedMarker.setAttribute('id', selectedMarkerId);
                        selectedMarker.classList.add('selected-arrowhead');
                        originalMarker.parentNode.appendChild(selectedMarker);
                    }
                    // Make the path use this new marker while it is selected.
                    const markerAttribute = selectedEdge.label.startMarkerId
                        ? 'marker-start'
                        : 'marker-end';
                    this._lastSelectedEdgeGroup
                        .selectAll('path.edgeline')
                        .style(markerAttribute, `url(#${selectedMarkerId})`);
                }
            }
        }
        not(x) {
            return !x;
        }
    };
    TfGraph.template = html `
    <style>
      .container {
        width: 100%;
        height: 100%;
        background: white;
        box-shadow: 0 1px 5px rgba(0, 0, 0, 0.2);
      }

      .vertical {
        width: 100%;
        height: 100%;
        @apply --layout-vertical;
      }

      .auto {
        @apply --layout-flex-auto;
        @apply --layout-vertical;
      }

      h2 {
        text-align: center;
      }

      paper-button {
        text-transform: none;
      }
    </style>
    <div class="container">
      <div class="vertical">
        <template is="dom-if" if="[[title]]">
          <h2>[[title]]</h2>
        </template>
        <tf-graph-scene
          id="scene"
          class="auto"
          render-hierarchy="[[renderHierarchy]]"
          highlighted-node="[[_getVisible(highlightedNode)]]"
          selected-node="{{selectedNode}}"
          selected-edge="{{selectedEdge}}"
          color-by="[[colorBy]]"
          progress="[[progress]]"
          node-context-menu-items="[[nodeContextMenuItems]]"
          node-names-to-health-pills="[[nodeNamesToHealthPills]]"
          health-pill-step-index="{{healthPillStepIndex}}"
          handle-edge-selected="[[handleEdgeSelected]]"
          trace-inputs="[[traceInputs]]"
        ></tf-graph-scene>
      </div>
    </div>
  `;
    __decorate([
        property({
            type: Object,
            notify: true,
            observer: '_graphChanged',
        }),
        __metadata("design:type", Object)
    ], TfGraph.prototype, "graphHierarchy", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraph.prototype, "basicGraph", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraph.prototype, "stats", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraph.prototype, "devicesForStats", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraph.prototype, "hierarchyParams", void 0);
    __decorate([
        property({
            type: Object,
            notify: true,
        }),
        __metadata("design:type", Object)
    ], TfGraph.prototype, "progress", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfGraph.prototype, "title", void 0);
    __decorate([
        property({
            type: String,
            notify: true,
        }),
        __metadata("design:type", String)
    ], TfGraph.prototype, "selectedNode", void 0);
    __decorate([
        property({
            type: Object,
            notify: true,
        }),
        __metadata("design:type", Object)
    ], TfGraph.prototype, "selectedEdge", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraph.prototype, "_lastSelectedEdgeGroup", void 0);
    __decorate([
        property({
            type: String,
            notify: true,
        }),
        __metadata("design:type", String)
    ], TfGraph.prototype, "highlightedNode", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfGraph.prototype, "colorBy", void 0);
    __decorate([
        property({
            type: Object,
            notify: true,
            readOnly: true,
        }),
        __metadata("design:type", Object)
    ], TfGraph.prototype, "colorByParams", void 0);
    __decorate([
        property({
            type: Object,
            readOnly: true,
            notify: true,
        }),
        __metadata("design:type", Object)
    ], TfGraph.prototype, "renderHierarchy", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfGraph.prototype, "traceInputs", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfGraph.prototype, "nodeContextMenuItems", void 0);
    __decorate([
        property({
            type: Number,
        }),
        __metadata("design:type", Number)
    ], TfGraph.prototype, "_renderDepth", void 0);
    __decorate([
        property({
            type: Boolean,
        }),
        __metadata("design:type", Boolean)
    ], TfGraph.prototype, "_allowGraphSelect", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraph.prototype, "nodeNamesToHealthPills", void 0);
    __decorate([
        property({ type: Number }),
        __metadata("design:type", Number)
    ], TfGraph.prototype, "healthPillStepIndex", void 0);
    __decorate([
        property({
            type: Object,
        }),
        __metadata("design:type", Object)
    ], TfGraph.prototype, "edgeWidthFunction", void 0);
    __decorate([
        property({
            type: Object,
        }),
        __metadata("design:type", Object)
    ], TfGraph.prototype, "handleNodeSelected", void 0);
    __decorate([
        property({
            type: Object,
        }),
        __metadata("design:type", Object)
    ], TfGraph.prototype, "edgeLabelFunction", void 0);
    __decorate([
        property({
            type: Object,
        }),
        __metadata("design:type", Object)
    ], TfGraph.prototype, "handleEdgeSelected", void 0);
    __decorate([
        observe('graphHierarchy', 'edgeWidthFunction', 'handleNodeSelected', 'edgeLabelFunction', 'handleEdgeSelected'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfGraph.prototype, "_buildNewRenderHierarchy", null);
    __decorate([
        observe('stats', 'devicesForStats'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfGraph.prototype, "_statsChanged", null);
    __decorate([
        observe('selectedNode'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfGraph.prototype, "_selectedNodeChanged", null);
    __decorate([
        observe('selectedEdge'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfGraph.prototype, "_selectedEdgeChanged", null);
    TfGraph = __decorate([
        customElement('tf-graph')
    ], TfGraph);

    /**
     * Element for putting tf-graph and tf-graph-info side by side.
     *
     * Example
     * <tf-graph-board graph=[[graph]]></tf-graph-board>
     */
    let TfGraphBoard = class TfGraphBoard extends LegacyElementMixin(PolymerElement) {
        constructor() {
            super(...arguments);
            // Whether the user can request health pills for individual steps from the server. This can be
            // slow compared the default of showing sampled health pills.
            this.allStepsModeEnabled = false;
            // Relevant if allStepsModeEnabled. The specific step for which to fetch health pills from the
            // server for.
            this.specificHealthPillStep = 0;
            this.compatNodeTitle = 'TPU Compatibility';
        }
        fit() {
            this.$.graph.fit();
        }
        /** True if the progress is not complete yet (< 100 %). */
        _isNotComplete(progress) {
            return progress.value < 100;
        }
        _getContainerClass(progress) {
            var result = 'container';
            if (progress.error) {
                result += ' error';
            }
            if (this._isNotComplete(progress)) {
                result += ' loading';
            }
            return result;
        }
        _onNodeInclusionToggled(event) {
            this.$.graph.nodeToggleExtract(event.detail.name);
        }
        _onNodeSeriesGroupToggled(event) {
            this.$.graph.nodeToggleSeriesGroup(event.detail.name);
        }
        _updateNodeInclude() {
            const node = !this.renderHierarchy
                ? null
                : this.renderHierarchy.getNodeByName(this.selectedNode);
            this._selectedNodeInclude = node
                ? node.include
                : InclusionType.UNSPECIFIED;
        }
    };
    TfGraphBoard.template = html `
    <style>
      ::host {
        display: block;
      }

      /deep/ .close {
        position: absolute;
        cursor: pointer;
        left: 15px;
        bottom: 15px;
      }

      .container {
        width: 100%;
        height: 100%;
        opacity: 1;
      }

      .container.loading {
        cursor: progress;
        opacity: 0.1;
      }

      .container.loading.error {
        cursor: auto;
      }

      #info {
        position: absolute;
        right: 5px;
        top: 5px;
        padding: 0px;
        max-width: 380px;
        min-width: 320px;
        background-color: rgba(255, 255, 255, 0.9);
        @apply --shadow-elevation-2dp;
      }

      #main {
        width: 100%;
        height: 100%;
      }

      #progress-bar {
        display: flex;
        flex-direction: column;
        align-items: center;
        justify-content: center;
        width: 100%;
        position: absolute;
        top: 40px;
        left: 0;
        font-size: 13px;
      }

      #progress-msg {
        margin-bottom: 5px;
        white-space: pre-wrap;
        width: 400px;
      }

      paper-progress {
        width: 400px;
        --paper-progress-height: 6px;
        --paper-progress-active-color: #f3913e;
      }

      .context-menu {
        position: absolute;
        display: none;
        background-color: #e2e2e2;
        border-radius: 2px;
        font-size: 14px;
        min-width: 150px;
        border: 1px solid #d4d4d4;
      }

      /deep/ .context-menu ul {
        list-style-type: none;
        margin: 0;
        padding: 0;
        cursor: default;
      }

      /deep/ .context-menu ul li {
        padding: 4px 16px;
      }

      /deep/ .context-menu ul li:hover {
        background-color: #f3913e;
        color: white;
      }
    </style>
    <template is="dom-if" if="[[_isNotComplete(progress)]]">
      <div id="progress-bar">
        <div id="progress-msg">[[progress.msg]]</div>
        <paper-progress value="[[progress.value]]"></paper-progress>
      </div>
    </template>
    <div class$="[[_getContainerClass(progress)]]">
      <div id="main">
        <tf-graph
          id="graph"
          graph-hierarchy="{{graphHierarchy}}"
          basic-graph="[[graph]]"
          hierarchy-params="[[hierarchyParams]]"
          render-hierarchy="{{renderHierarchy}}"
          devices-for-stats="[[devicesForStats]]"
          stats="[[stats]]"
          selected-node="{{selectedNode}}"
          highlighted-node="{{_highlightedNode}}"
          color-by="[[colorBy]]"
          color-by-params="{{colorByParams}}"
          progress="{{progress}}"
          edge-label-function="[[edgeLabelFunction]]"
          edge-width-function="[[edgeWidthFunction]]"
          node-names-to-health-pills="[[nodeNamesToHealthPills]]"
          health-pill-step-index="[[healthPillStepIndex]]"
          handle-node-selected="[[handleNodeSelected]]"
          handle-edge-selected="[[handleEdgeSelected]]"
          trace-inputs="[[traceInputs]]"
        ></tf-graph>
      </div>
      <div id="info">
        <tf-graph-info
          id="graph-info"
          title="selected"
          graph-hierarchy="[[graphHierarchy]]"
          hierarchy-params="[[hierarchyParams]]"
          render-hierarchy="[[renderHierarchy]]"
          graph="[[graph]]"
          selected-node="{{selectedNode}}"
          selected-node-include="{{_selectedNodeInclude}}"
          highlighted-node="{{_highlightedNode}}"
          color-by="[[colorBy]]"
          color-by-params="[[colorByParams]]"
          debugger-data-enabled="[[debuggerDataEnabled]]"
          are-health-pills-loading="[[areHealthPillsLoading]]"
          debugger-numeric-alerts="[[debuggerNumericAlerts]]"
          node-names-to-health-pills="[[nodeNamesToHealthPills]]"
          all-steps-mode-enabled="{{allStepsModeEnabled}}"
          specific-health-pill-step="{{specificHealthPillStep}}"
          health-pill-step-index="{{healthPillStepIndex}}"
          compat-node-title="[[compatNodeTitle]]"
          on-node-toggle-inclusion="_onNodeInclusionToggled"
          on-node-toggle-seriesgroup="_onNodeSeriesGroupToggled"
        ></tf-graph-info>
      </div>
    </div>
  `;
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraphBoard.prototype, "graphHierarchy", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraphBoard.prototype, "graph", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraphBoard.prototype, "stats", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraphBoard.prototype, "progress", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfGraphBoard.prototype, "traceInputs", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfGraphBoard.prototype, "colorBy", void 0);
    __decorate([
        property({
            type: Object,
            notify: true,
        }),
        __metadata("design:type", Object)
    ], TfGraphBoard.prototype, "colorByParams", void 0);
    __decorate([
        property({
            type: Object,
            notify: true,
        }),
        __metadata("design:type", Object)
    ], TfGraphBoard.prototype, "renderHierarchy", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfGraphBoard.prototype, "debuggerDataEnabled", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfGraphBoard.prototype, "areHealthPillsLoading", void 0);
    __decorate([
        property({
            type: Array,
            notify: true,
        }),
        __metadata("design:type", Array)
    ], TfGraphBoard.prototype, "debuggerNumericAlerts", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraphBoard.prototype, "nodeNamesToHealthPills", void 0);
    __decorate([
        property({
            type: Boolean,
            notify: true,
        }),
        __metadata("design:type", Boolean)
    ], TfGraphBoard.prototype, "allStepsModeEnabled", void 0);
    __decorate([
        property({
            type: Number,
            notify: true,
        }),
        __metadata("design:type", Number)
    ], TfGraphBoard.prototype, "specificHealthPillStep", void 0);
    __decorate([
        property({ type: Number }),
        __metadata("design:type", Number)
    ], TfGraphBoard.prototype, "healthPillStepIndex", void 0);
    __decorate([
        property({
            type: String,
            notify: true,
        }),
        __metadata("design:type", String)
    ], TfGraphBoard.prototype, "selectedNode", void 0);
    __decorate([
        property({
            type: String,
        }),
        __metadata("design:type", String)
    ], TfGraphBoard.prototype, "compatNodeTitle", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraphBoard.prototype, "edgeWidthFunction", void 0);
    __decorate([
        property({ type: Number }),
        __metadata("design:type", Number)
    ], TfGraphBoard.prototype, "_selectedNodeInclude", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfGraphBoard.prototype, "_highlightedNode", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraphBoard.prototype, "handleNodeSelected", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraphBoard.prototype, "edgeLabelFunction", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraphBoard.prototype, "handleEdgeSelected", void 0);
    __decorate([
        observe('selectedNode', 'renderHierarchy'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfGraphBoard.prototype, "_updateNodeInclude", null);
    TfGraphBoard = __decorate([
        customElement('tf-graph-board')
    ], TfGraphBoard);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    let TfGraphNodeSearch = class TfGraphNodeSearch extends LegacyElementMixin(PolymerElement) {
        constructor() {
            super(...arguments);
            this._rawRegexInput = '';
            // The value of the regex input for the last search.
            this._previousRegexInput = '';
            this._searchTimeoutDelay = 150;
            this._maxRegexResults = 42;
        }
        // This is the cleaned input.
        get _regexInput() {
            var renderHierarchy = this.renderHierarchy;
            var rawRegexInput = this._rawRegexInput;
            return rawRegexInput.trim();
        }
        _regexInputChanged() {
            var regexInput = this._regexInput;
            this._requestSearch();
        }
        _clearSearchResults() {
            this.set('_regexMatches', []);
        }
        _requestSearch() {
            if (this._searchPending) {
                return;
            }
            if (this._regexInput === this._previousRegexInput) {
                // No new search is needed.
                this._searchPending = false;
                return;
            }
            this._searchPending = true;
            this._executeSearch();
            // After some time, perhaps execute another search.
            this.async(() => {
                this._searchPending = false;
                this._requestSearch();
            }, this._searchTimeoutDelay);
        }
        _executeSearch() {
            this._previousRegexInput = this._regexInput;
            if (!this._regexInput) {
                this._clearSearchResults();
                return;
            }
            try {
                var regex = new RegExp(this._regexInput);
            }
            catch (e) {
                // The regular expression is invalid.
                this._clearSearchResults();
                return;
            }
            const matchedNodes = [];
            const nodeMap = this.renderHierarchy.hierarchy.getNodeMap();
            lodash.each(nodeMap, (_, nodeName) => {
                if (matchedNodes.length >= this._maxRegexResults) {
                    // Terminate.
                    return false;
                }
                if (!regex.test(nodeName)) {
                    return;
                }
                matchedNodes.push(nodeName);
            });
            this.set('_regexMatches', matchedNodes);
        }
        _matchClicked(e) {
            const node = e.model.item;
            this.set('selectedNode', node);
        }
    };
    TfGraphNodeSearch.template = html `
    <div id="search-container">
      <paper-input
        id="runs-regex"
        label="Search nodes. Regexes supported."
        value="{{_rawRegexInput}}"
      >
      </paper-input>
      <div id="search-results-anchor">
        <div id="search-results">
          <template is="dom-repeat" items="[[_regexMatches]]">
            <div id="search-match" on-click="_matchClicked">[[item]]</div>
          </template>
        </div>
      </div>
    </div>
    <style>
      #search-container {
        width: 100%;
        overflow: visible;
      }

      #runs-regex {
        width: 100%;
      }

      #search-results-anchor {
        position: relative;
      }

      #search-results {
        color: #fff;
        position: absolute;
        max-height: 200px;
        overflow-x: hidden;
        overflow-y: auto;
        text-align: right;
        max-width: 100%;
        box-sizing: border-box;
      }

      #search-match {
        background: var(--tb-orange-strong);
        padding: 3px;
        float: right;
        width: 100%;
        box-sizing: border-box;
        direction: rtl;
      }

      #search-match:hover {
        background: var(--tb-orange-weak);
        cursor: pointer;
      }
    </style>
  `;
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraphNodeSearch.prototype, "renderHierarchy", void 0);
    __decorate([
        property({
            type: String,
            notify: true,
        }),
        __metadata("design:type", String)
    ], TfGraphNodeSearch.prototype, "selectedNode", void 0);
    __decorate([
        property({
            type: String,
        }),
        __metadata("design:type", String)
    ], TfGraphNodeSearch.prototype, "_rawRegexInput", void 0);
    __decorate([
        property({
            type: String,
        }),
        __metadata("design:type", String)
    ], TfGraphNodeSearch.prototype, "_previousRegexInput", void 0);
    __decorate([
        property({
            type: Number,
        }),
        __metadata("design:type", Number)
    ], TfGraphNodeSearch.prototype, "_searchTimeoutDelay", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfGraphNodeSearch.prototype, "_searchPending", void 0);
    __decorate([
        property({
            type: Number,
        }),
        __metadata("design:type", Number)
    ], TfGraphNodeSearch.prototype, "_maxRegexResults", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfGraphNodeSearch.prototype, "_regexMatches", void 0);
    __decorate([
        computed('renderHierarchy', '_rawRegexInput'),
        __metadata("design:type", String),
        __metadata("design:paramtypes", [])
    ], TfGraphNodeSearch.prototype, "_regexInput", null);
    __decorate([
        observe('_regexInput'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfGraphNodeSearch.prototype, "_regexInputChanged", null);
    TfGraphNodeSearch = __decorate([
        customElement('tf-graph-node-search')
    ], TfGraphNodeSearch);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    const DEVICE_NAME_REGEX = /device:([^:]+:[0-9]+)$/;
    /**
     * Display only devices matching one of the following regex.
     */
    const DEVICE_NAMES_INCLUDE = [
        {
            // Don't include GPU stream, memcpy, etc. devices
            regex: DEVICE_NAME_REGEX,
        },
    ];
    /**
     * Stats from device names that match these regexes will be disabled by default.
     * The user can still turn on a device by selecting the checkbox in the device list.
     */
    const DEVICE_STATS_DEFAULT_OFF = [];
    var ColorBy$1;
    (function (ColorBy) {
        ColorBy["COMPUTE_TIME"] = "compute_time";
        ColorBy["MEMORY"] = "memory";
        ColorBy["STRUCTURE"] = "structure";
        ColorBy["XLA_CLUSTER"] = "xla_cluster";
        ColorBy["OP_COMPATIBILITY"] = "op_compatibility";
    })(ColorBy$1 || (ColorBy$1 = {}));
    const GRADIENT_COMPATIBLE_COLOR_BY = new Set([
        ColorBy$1.COMPUTE_TIME,
        ColorBy$1.MEMORY,
    ]);
    let TfGraphControls = class TfGraphControls extends LegacyElementMixin(PolymerElement) {
        constructor() {
            super(...arguments);
            // Public API.
            /**
             * @type {?tf_graph_proto.StepStats}
             */
            this.stats = null;
            /**
             * @type {?Object<string, boolean>}
             */
            this.devicesForStats = null;
            /**
             * @type {!ColorBy}
             */
            this.colorBy = ColorBy$1.STRUCTURE;
            this.datasets = [];
            this._selectedRunIndex = 0;
            this.traceInputs = false;
            this._selectedTagIndex = 0;
            /**
             * @type {tf_graph_common.SelectionType}
             */
            this._selectedGraphType = SelectionType.OP_GRAPH;
            this.showSessionRunsDropdown = true;
            this.showUploadButton = true;
            this._legendOpened = true;
        }
        _xlaClustersProvided(renderHierarchy) {
            return (renderHierarchy &&
                renderHierarchy.hierarchy &&
                renderHierarchy.hierarchy.xlaClusters.length > 0);
        }
        _statsChanged(stats) {
            if (stats == null) {
                return;
            }
            var devicesForStats = {};
            var devices = lodash.each(stats.dev_stats, function (d) {
                // Only considered included devices.
                var include = lodash.some(DEVICE_NAMES_INCLUDE, function (rule) {
                    return rule.regex.test(d.device);
                });
                // Exclude device names that are ignored by default.
                var exclude = lodash.some(DEVICE_STATS_DEFAULT_OFF, function (rule) {
                    return rule.regex.test(d.device);
                });
                if (include && !exclude) {
                    devicesForStats[d.device] = true;
                }
            });
            this.set('devicesForStats', devicesForStats);
        }
        get _currentDevices() {
            var devicesForStats = this.devicesForStats;
            const stats = this.stats;
            const devStats = stats ? stats.dev_stats : [];
            const allDevices = devStats.map((d) => d.device);
            const devices = allDevices.filter((deviceName) => {
                return DEVICE_NAMES_INCLUDE.some((rule) => {
                    return rule.regex.test(deviceName);
                });
            });
            // Devices names can be long so we remove the longest common prefix
            // before showing the devices in a list.
            const suffixes = removeCommonPrefix(devices);
            if (suffixes.length == 1) {
                const found = suffixes[0].match(DEVICE_NAME_REGEX);
                if (found) {
                    suffixes[0] = found[1];
                }
            }
            return devices.map((device, i) => {
                let ignoredMsg = null;
                // TODO(stephanwlee): this should probably bail on the first match or
                // do something useful with multiple rule.msgs.
                DEVICE_STATS_DEFAULT_OFF.forEach((rule) => {
                    if (rule.regex.test(device)) {
                        ignoredMsg = rule.msg;
                    }
                });
                return {
                    device: device,
                    suffix: suffixes[i],
                    used: devicesForStats[device],
                    ignoredMsg: ignoredMsg,
                };
            });
        }
        _deviceCheckboxClicked(event) {
            // Update the device map.
            const input = event.target;
            const devicesForStats = Object.assign({}, this.devicesForStats);
            const device = input.value;
            if (input.checked) {
                devicesForStats[device] = true;
            }
            else {
                delete devicesForStats[device];
            }
            this.set('devicesForStats', devicesForStats);
        }
        _numTags(datasets, _selectedRunIndex) {
            return this._getTags(datasets, _selectedRunIndex).length;
        }
        _getTags(datasets, _selectedRunIndex) {
            if (!datasets || !datasets[_selectedRunIndex]) {
                return [];
            }
            return datasets[_selectedRunIndex].tags;
        }
        _fit() {
            this.fire('fit-tap');
        }
        _isGradientColoring(stats, colorBy) {
            return GRADIENT_COMPATIBLE_COLOR_BY.has(colorBy) && stats != null;
        }
        _equals(a, b) {
            return a === b;
        }
        get _currentDeviceParams() {
            var colorByParams = this.colorByParams;
            const deviceParams = colorByParams.device.filter((param) => {
                return DEVICE_NAMES_INCLUDE.some((rule) => {
                    return rule.regex.test(param.device);
                });
            });
            // Remove common prefix and merge back corresponding color. If
            // there is only one device then remove everything up to "/device:".
            const suffixes = removeCommonPrefix(deviceParams.map((d) => d.device));
            if (suffixes.length == 1) {
                var found = suffixes[0].match(DEVICE_NAME_REGEX);
                if (found) {
                    suffixes[0] = found[1];
                }
            }
            return deviceParams.map((d, i) => {
                return { device: suffixes[i], color: d.color };
            });
        }
        get _currentXlaClusterParams() {
            var colorByParams = this.colorByParams;
            return colorByParams.xla_cluster;
        }
        get _currentGradientParams() {
            var colorByParams = this.colorByParams;
            var colorBy = this.colorBy;
            if (!this._isGradientColoring(this.stats, colorBy)) {
                return;
            }
            const params = colorByParams[colorBy];
            let minValue = params.minValue;
            let maxValue = params.maxValue;
            if (colorBy === ColorBy$1.MEMORY) {
                minValue = convertUnitsToHumanReadable(minValue, MEMORY_UNITS);
                maxValue = convertUnitsToHumanReadable(maxValue, MEMORY_UNITS);
            }
            else if (colorBy === ColorBy$1.COMPUTE_TIME) {
                minValue = convertUnitsToHumanReadable(minValue, TIME_UNITS);
                maxValue = convertUnitsToHumanReadable(maxValue, TIME_UNITS);
            }
            return {
                minValue,
                maxValue,
                startColor: params.startColor,
                endColor: params.endColor,
            };
        }
        download() {
            this.$.graphdownload.click();
        }
        _updateFileInput(e) {
            const file = e.target.files[0];
            if (!file)
                return;
            // Strip off everything before the last "/" and strip off the file
            // extension in order to get the name of the PNG for the graph.
            let filePath = file.name;
            const dotIndex = filePath.lastIndexOf('.');
            if (dotIndex >= 0) {
                filePath = filePath.substring(0, dotIndex);
            }
            const lastSlashIndex = filePath.lastIndexOf('/');
            if (lastSlashIndex >= 0) {
                filePath = filePath.substring(lastSlashIndex + 1);
            }
            this._setDownloadFilename(filePath);
            this.set('selectedFile', e);
        }
        _datasetsChanged(newDatasets, oldDatasets) {
            if (oldDatasets != null) {
                // Select the first dataset by default.
                this._selectedRunIndex = 0;
            }
        }
        _computeSelection(datasets, _selectedRunIndex, _selectedTagIndex, _selectedGraphType) {
            if (!datasets[_selectedRunIndex] ||
                !datasets[_selectedRunIndex].tags[_selectedTagIndex]) {
                return null;
            }
            return {
                run: datasets[_selectedRunIndex].name,
                tag: datasets[_selectedRunIndex].tags[_selectedTagIndex].tag,
                type: _selectedGraphType,
            };
        }
        _selectedRunIndexChanged(runIndex) {
            if (!this.datasets)
                return;
            // Reset the states when user pick a different run.
            this.colorBy = ColorBy$1.STRUCTURE;
            this._selectedTagIndex = 0;
            this._selectedGraphType = this._getDefaultSelectionType();
            this.traceInputs = false; // Set trace input to off-state.
            this._setDownloadFilename(this.datasets[runIndex] ? this.datasets[runIndex].name : '');
        }
        _selectedTagIndexChanged() {
            this._selectedGraphType = this._getDefaultSelectionType();
        }
        _getDefaultSelectionType() {
            const { datasets, _selectedRunIndex: run, _selectedTagIndex: tag } = this;
            if (!datasets ||
                !datasets[run] ||
                !datasets[run].tags[tag] ||
                datasets[run].tags[tag].opGraph) {
                return SelectionType.OP_GRAPH;
            }
            const datasetForRun = datasets[run];
            if (datasetForRun.tags[tag].profile) {
                return SelectionType.PROFILE;
            }
            if (datasetForRun.tags[tag].conceptualGraph) {
                return SelectionType.CONCEPTUAL_GRAPH;
            }
            return SelectionType.OP_GRAPH;
        }
        _getFile() {
            this.$$('#file').click();
        }
        _setDownloadFilename(name) {
            this.$.graphdownload.setAttribute('download', name + '.png');
        }
        _statsNotNull(stats) {
            return stats !== null;
        }
        _toggleLegendOpen() {
            this.set('_legendOpened', !this._legendOpened);
        }
        _getToggleText(legendOpened) {
            return legendOpened ? 'Close legend.' : 'Expand legend.';
        }
        _getToggleLegendIcon(legendOpened) {
            // This seems counter-intuitive, but actually makes sense because the
            // expand-more button points downwards, and the expand-less button points
            // upwards. For most collapsibles, this works because the collapsibles
            // expand in the downwards direction. This collapsible expands upwards
            // though, so we reverse the icons.
            return legendOpened ? 'expand-more' : 'expand-less';
        }
        _getSelectionOpGraphDisabled(datasets, _selectedRunIndex, _selectedTagIndex) {
            return (!datasets[_selectedRunIndex] ||
                !datasets[_selectedRunIndex].tags[_selectedTagIndex] ||
                !datasets[_selectedRunIndex].tags[_selectedTagIndex].opGraph);
        }
        _getSelectionProfileDisabled(datasets, _selectedRunIndex, _selectedTagIndex) {
            return (!datasets[_selectedRunIndex] ||
                !datasets[_selectedRunIndex].tags[_selectedTagIndex] ||
                !datasets[_selectedRunIndex].tags[_selectedTagIndex].profile);
        }
        _getSelectionConceptualGraphDisabled(datasets, _selectedRunIndex, _selectedTagIndex) {
            return (!datasets[_selectedRunIndex] ||
                !datasets[_selectedRunIndex].tags[_selectedTagIndex] ||
                !datasets[_selectedRunIndex].tags[_selectedTagIndex].conceptualGraph);
        }
    };
    TfGraphControls.template = html `
    <style>
      :host {
        color: gray;
        display: flex;
        flex-direction: column;
        font-size: 12px;
        width: 100%;
      }

      paper-dropdown-menu {
        --paper-dropdown-menu-input: {
          padding: 0;
          color: gray;
        }
        --iron-icon-width: 15px;
        --iron-icon-height: 15px;
        --primary-text-color: gray;
        --paper-item-min-height: 30px;
      }

      paper-button[raised].keyboard-focus {
        font-weight: normal;
      }

      .run-dropdown {
        --paper-input-container: {
          padding: 8px 0 8px 10px;
        }
      }

      .color-dropdown {
        --paper-input-container: {
          padding: 9px 0 0 13px;
        }
      }

      table {
        border-collapse: collapse;
        border-spacing: 0;
      }

      table td {
        padding: 0;
        margin: 0;
      }

      .allcontrols {
        padding: 0 20px 20px;
        flex-grow: 1;
        overflow-y: auto;
      }

      .legend-holder {
        background: #e9e9e9;
        border-top: 1px solid #ccc;
        box-sizing: border-box;
        color: #555;
        padding: 15px 20px;
        width: 100%;
      }

      .toggle-legend-button {
        max-height: 20px;
        max-width: 20px;
        padding: 0;
      }

      .toggle-legend-text {
        vertical-align: middle;
      }

      paper-radio-button {
        display: block;
        padding: 5px;
      }
      svg.icon,
      tf-graph-icon {
        width: 60px;
        height: 18px;
      }
      .domainValues {
        margin-bottom: 10px;
        width: 165px;
      }
      .domainStart {
        float: left;
      }
      .domainEnd {
        float: right;
      }
      .colorBox {
        width: 20px;
      }

      .image-icon {
        width: 24px;
        height: 24px;
      }

      .help-icon {
        height: 15px;
        margin: 0;
        padding: 0;
      }

      .gray {
        color: #666;
      }

      .title {
        font-size: 16px;
        margin: 8px 5px 8px 0;
        color: black;
      }
      .title small {
        font-weight: normal;
      }
      .deviceList,
      .xlaClusterList {
        max-height: 200px;
        overflow-y: auto;
      }

      #file {
        padding: 8px 0;
      }

      .color-legend-row {
        align-items: center;
        clear: both;
        display: flex;
        height: 20px;
        margin-top: 5px;
      }

      .color-legend-row .label,
      .color-legend-row svg,
      .color-legend-row tf-graph-icon {
        flex: 0 0 40px;
        margin-right: 20px;
      }

      .devices-checkbox input {
        text-align: left;
        vertical-align: middle;
      }

      .control-holder .icon-button {
        font-size: 14px;
        margin: 0 -5px;
        padding: 5px;
      }

      .button-text {
        padding-left: 20px;
        text-transform: none;
      }

      .upload-button {
        width: 165px;
        height: 25px;
        text-transform: none;
        margin-top: 4px;
      }

      .button-icon {
        width: 26px;
        height: 26px;
        color: var(--paper-orange-500);
      }

      .hidden-input {
        height: 0px;
        width: 0px;
        overflow: hidden;
      }

      .allcontrols .control-holder {
        clear: both;
        display: flex;
        justify-content: space-between;
      }

      .allcontrols .control-holder paper-radio-group {
        margin-top: 5px;
      }

      span.counter {
        font-size: 13px;
        color: gray;
      }

      .runs paper-item {
        --paper-item: {
          white-space: nowrap;
        }
      }

      table.control-holder {
        border: 0;
        border-collapse: collapse;
      }

      table.tf-graph-controls td.input-element-table-data {
        padding: 0 0 0 20px;
      }

      .spacer {
        flex-grow: 1;
      }

      .color-text {
        overflow: hidden;
      }

      /** Override inline styles that suppress pointer events for disabled buttons. Otherwise, the */
      /*  tooltips do not appear. */
      paper-radio-group paper-radio-button {
        pointer-events: auto !important;
      }

      .legend-clarifier {
        color: #266236;
        cursor: help;
        display: inline-block;
        text-decoration: underline;
      }

      .legend-clarifier paper-tooltip {
        width: 150px;
      }

      /** Otherwise, polymer UI controls appear atop node search. */
      tf-graph-node-search {
        z-index: 1;
        width: 100%;
      }

      paper-dropdown-menu {
        flex-grow: 1;
      }
    </style>

    <div class="allcontrols">
      <div class="control-holder">
        <tf-graph-node-search
          selected-node="{{selectedNode}}"
          render-hierarchy="[[renderHierarchy]]"
        ></tf-graph-node-search>
      </div>
      <div class="control-holder">
        <paper-button class="icon-button" on-tap="_fit" alt="Fit to screen">
          <iron-icon icon="aspect-ratio" class="button-icon"></iron-icon>
          <span class="button-text">Fit to Screen</span>
        </paper-button>
      </div>
      <div class="control-holder">
        <paper-button
          class="icon-button"
          on-click="download"
          alt="Download PNG"
        >
          <iron-icon icon="file-download" class="button-icon"></iron-icon>
          <span class="button-text">Download PNG</span>
        </paper-button>
        <a href="#" id="graphdownload" class="title" download="graph.png"></a>
      </div>
      <div class="control-holder runs">
        <div class="title">
          Run <span class="counter">([[datasets.length]])</span>
        </div>
        <paper-dropdown-menu
          no-label-float
          no-animations
          noink
          horizontal-align="left"
          class="run-dropdown"
        >
          <paper-listbox
            class="dropdown-content"
            selected="{{_selectedRunIndex}}"
            slot="dropdown-content"
          >
            <template is="dom-repeat" items="[[datasets]]">
              <paper-item>[[item.name]]</paper-item>
            </template>
          </paper-listbox>
        </paper-dropdown-menu>
      </div>
      <template is="dom-if" if="[[showSessionRunsDropdown]]">
        <div class="control-holder">
          <div class="title">
            Tag
            <span class="counter"
              >([[_numTags(datasets, _selectedRunIndex)]])</span
            >
          </div>
          <paper-dropdown-menu
            no-label-float
            no-animations
            horizontal-align="left"
            noink
            class="run-dropdown"
          >
            <paper-listbox
              class="dropdown-content"
              selected="{{_selectedTagIndex}}"
              slot="dropdown-content"
            >
              <template
                is="dom-repeat"
                items="[[_getTags(datasets, _selectedRunIndex)]]"
              >
                <paper-item>[[item.displayName]]</paper-item>
              </template>
            </paper-listbox>
          </paper-dropdown-menu>
        </div>
      </template>
      <template is="dom-if" if="[[showUploadButton]]">
        <div class="control-holder">
          <div class="title">Upload</div>
          <paper-button
            raised
            class="upload-button"
            on-click="_getFile"
            title="Upload a graph pbtxt file to view the graph"
          >
            Choose File
          </paper-button>
          <div class="hidden-input">
            <input
              type="file"
              id="file"
              name="file"
              on-change="_updateFileInput"
              accept=".pbtxt"
            />
          </div>
        </div>
      </template>
      <div class="control-holder">
        <paper-radio-group selected="{{_selectedGraphType}}">
          <!-- Note that the name has to match that of tf_graph_common.SelectionType. -->
          <paper-radio-button
            name="op_graph"
            disabled="[[_getSelectionOpGraphDisabled(datasets, _selectedRunIndex, _selectedTagIndex)]]"
            >Graph</paper-radio-button
          >
          <paper-radio-button
            name="conceptual_graph"
            disabled="[[_getSelectionConceptualGraphDisabled(datasets, _selectedRunIndex, _selectedTagIndex)]]"
            >Conceptual Graph</paper-radio-button
          >
          <paper-radio-button
            name="profile"
            disabled="[[_getSelectionProfileDisabled(datasets, _selectedRunIndex, _selectedTagIndex)]]"
            >Profile</paper-radio-button
          >
        </paper-radio-group>
      </div>
      <div class="control-holder">
        <div>
          <paper-toggle-button checked="{{traceInputs}}" class="title">
            Trace inputs
          </paper-toggle-button>
        </div>
      </div>
      <template is="dom-if" if="[[healthPillsFeatureEnabled]]">
        <div class="control-holder">
          <paper-toggle-button checked="{{healthPillsToggledOn}}" class="title"
            >Show health pills</paper-toggle-button
          >
        </div>
      </template>
      <div class="control-holder">
        <div class="title">Color</div>
        <paper-radio-group selected="{{colorBy}}">
          <paper-radio-button name="structure">Structure</paper-radio-button>

          <paper-radio-button name="device">Device</paper-radio-button>

          <paper-radio-button
            id="xla-cluster-radio-button"
            name="xla_cluster"
            disabled="[[!_xlaClustersProvided(renderHierarchy)]]"
          >
            XLA Cluster
          </paper-radio-button>
          <paper-tooltip
            animation-delay="0"
            for="xla-cluster-radio-button"
            position="right"
            offset="0"
          >
            Coloring by XLA cluster is only enabled if at least 1 op specifies
            an XLA cluster.
          </paper-tooltip>

          <paper-radio-button
            id="compute-time-radio-button"
            name="compute_time"
            disabled="[[!stats]]"
          >
            Compute time
          </paper-radio-button>
          <paper-tooltip
            animation-delay="0"
            for="compute-time-radio-button"
            position="right"
            offset="0"
          >
            Coloring by compute time is only enabled if the RunMetadata proto is
            passed to the FileWriter when a specific session is run.
          </paper-tooltip>

          <paper-radio-button
            id="memory-radio-button"
            name="memory"
            disabled="[[!stats]]"
          >
            Memory
          </paper-radio-button>
          <paper-tooltip
            animation-delay="0"
            for="memory-radio-button"
            position="right"
            offset="0"
          >
            Coloring by memory is only enabled if the RunMetadata proto is
            passed to the FileWriter when a specific session is run.
          </paper-tooltip>

          <paper-radio-button
            id="tpu-compatibility-radio-button"
            name="op_compatibility"
          >
            TPU Compatibility
          </paper-radio-button>
          <paper-tooltip
            animation-delay="0"
            for="tpu-compatibility-radio-button"
            position="right"
            offset="0"
          >
            Coloring by whether an operation is compatible for the TPU device.
          </paper-tooltip>
        </paper-radio-group>
        <span class="spacer"></span>
      </div>
      <div>
        <template is="dom-if" if="[[_isGradientColoring(stats, colorBy)]]">
          <svg width="140" height="20" style="margin: 0 5px" class="color-text">
            <defs>
              <linearGradient
                id="linearGradient"
                x1="0%"
                y1="0%"
                x2="100%"
                y2="0%"
              >
                <stop
                  class="start"
                  offset="0%"
                  stop-color$="[[_currentGradientParams.startColor]]"
                ></stop>
                <stop
                  class="end"
                  offset="100%"
                  stop-color$="[[_currentGradientParams.endColor]]"
                ></stop>
              </linearGradient>
            </defs>
            <rect
              x="0"
              y="0"
              width="135"
              height="20"
              fill="url(#linearGradient)"
              stroke="black"
            ></rect>
          </svg>
          <div class="domainValues color-text">
            <div class="domainStart">[[_currentGradientParams.minValue]]</div>
            <div class="domainEnd">[[_currentGradientParams.maxValue]]</div>
          </div>
          <br style="clear: both" />
          <div>Devices included in stats:</div>
          <div class="deviceList">
            <template is="dom-repeat" items="[[_currentDevices]]">
              <div class="color-legend-row devices-checkbox">
                <span
                  ><input
                    type="checkbox"
                    value$="[[item.device]]"
                    checked$="[[item.used]]"
                    on-click="_deviceCheckboxClicked"
                /></span>
                <span>[[item.suffix]]</span>
                <template is="dom-if" if="[[item.ignoredMsg]]">
                  <paper-icon-button
                    icon="help"
                    class="help-icon"
                  ></paper-icon-button>
                  <paper-tooltip position="right" offset="0" animation-delay="0"
                    >[[item.ignoredMsg]]</paper-tooltip
                  >
                </template>
              </div>
            </template>
          </div>
        </template>
        <template is="dom-if" if="[[_equals(colorBy, 'structure')]]">
          <div class="color-text">
            <div class="color-legend-row">
              <span class="label"> colors </span>
              <span class="color-legend-value">same substructure</span>
            </div>
            <div class="color-legend-row">
              <tf-graph-icon
                type="META"
                height="16"
                fill-override="#eee"
                stroke-override="#a6a6a6"
              ></tf-graph-icon>
              <span class="color-legend-value">unique substructure</span>
            </div>
          </div>
        </template>
        <template is="dom-if" if="[[_equals(colorBy, 'device')]]">
          <div>
            <template is="dom-repeat" items="[[_currentDeviceParams]]">
              <div class="color-legend-row">
                <tf-graph-icon
                  type="META"
                  height="16"
                  fill-override="[[item.color]]"
                  stroke-override="#a6a6a6"
                ></tf-graph-icon>
                <span class="color-legend-value">[[item.device]]</span>
              </div>
            </template>
            <div class="color-legend-row">
              <tf-graph-icon
                type="META"
                height="16"
                fill-override="#eee"
                stroke-override="#a6a6a6"
              ></tf-graph-icon>
              <span class="color-legend-value">unknown device</span>
            </div>
          </div>
        </template>
        <template is="dom-if" if="[[_equals(colorBy, 'xla_cluster')]]">
          <div>
            <template is="dom-repeat" items="[[_currentXlaClusterParams]]">
              <div class="color-legend-row">
                <svg>
                  <use
                    xmlns:xlink="http://www.w3.org/1999/xlink"
                    xlink:href="#unfilled-rect"
                    x="0"
                    y="0"
                    style="fill:[[item.color]]"
                  ></use>
                </svg>
                <span class="color-legend-value">[[item.xla_cluster]]</span>
              </div>
            </template>
            <div class="color-legend-row">
              <svg>
                <use
                  xmlns:xlink="http://www.w3.org/1999/xlink"
                  xlink:href="#grey-rect"
                  x="0"
                  y="0"
                ></use>
              </svg>
              <span class="color-legend-value">unknown XLA cluster</span>
            </div>
          </div>
        </template>
        <template is="dom-if" if="[[_equals(colorBy, 'op_compatibility')]]">
          <div class="color-text">
            <div class="color-legend-row">
              <tf-graph-icon
                type="OP"
                height="16"
                fill-override="#0f9d58"
                stroke-override="#ccc"
              ></tf-graph-icon>
              <span class="color-legend-value">Valid Op</span>
            </div>
            <div class="color-legend-row">
              <tf-graph-icon
                type="OP"
                height="16"
                fill-override="#db4437"
                stroke-override="#ccc"
              ></tf-graph-icon>
              <span class="color-legend-value">Invalid Op</span>
            </div>
          </div>
        </template>
        <template is="dom-if" if="[[_statsNotNull(stats)]]">
          <div class="color-legend-row">
            <tf-graph-icon type="META" height="16" faded></tf-graph-icon>
            <span class="color-legend-value">unused substructure</span>
          </div>
        </template>
      </div>
    </div>
    <div class="legend-holder">
      <paper-icon-button
        icon="[[_getToggleLegendIcon(_legendOpened)]]"
        on-click="_toggleLegendOpen"
        class="toggle-legend-button"
      >
      </paper-icon-button>
      <span class="toggle-legend-text">
        [[_getToggleText(_legendOpened)]]
      </span>
      <iron-collapse opened="[[_legendOpened]]">
        <div>
          <table>
            <tbody>
              <tr>
                <td><div class="title">Graph</div></td>
                <td>(* = expandable)</td>
              </tr>
              <tr>
                <td>
                  <tf-graph-icon
                    type="META"
                    height="16"
                    fill-override="#d9d9d9"
                    stroke-override="#ccc"
                  ></tf-graph-icon>
                </td>
                <td>
                  Namespace<span class="gray">*</span>
                  <div class="legend-clarifier">
                    <span>?</span>
                    <paper-tooltip
                      animation-delay="0"
                      position="right"
                      offset="0"
                    >
                      Encapsulates a set of nodes. Namespace is hierarchical and
                      based on scope.
                    </paper-tooltip>
                  </div>
                </td>
              </tr>
              <tr>
                <td>
                  <tf-graph-icon type="OP" height="16"></tf-graph-icon>
                </td>
                <td>
                  OpNode
                  <div class="legend-clarifier">
                    <span>?</span>
                    <paper-tooltip
                      animation-delay="0"
                      position="right"
                      offset="0"
                    >
                      Node that performs an operation. These nodes cannot
                      expand.
                    </paper-tooltip>
                  </div>
                </td>
              </tr>
              <tr>
                <td>
                  <tf-graph-icon type="SERIES" height="16"></tf-graph-icon>
                </td>
                <td>
                  Unconnected series<span class="gray">*</span>
                  <div class="legend-clarifier">
                    <span>?</span>
                    <paper-tooltip
                      animation-delay="0"
                      position="right"
                      offset="0"
                    >
                      Sequence of numbered nodes that are not connected to each
                      other.
                    </paper-tooltip>
                  </div>
                </td>
              </tr>
              <tr>
                <td>
                  <tf-graph-icon
                    type="SERIES"
                    height="16"
                    vertical
                  ></tf-graph-icon>
                </td>
                <td>
                  Connected series<span class="gray">*</span>
                  <div class="legend-clarifier">
                    <span>?</span>
                    <paper-tooltip
                      animation-delay="0"
                      position="right"
                      offset="0"
                    >
                      Sequence of numbered nodes that are connected to each
                      other.
                    </paper-tooltip>
                  </div>
                </td>
              </tr>
              <tr>
                <td>
                  <svg class="icon">
                    <circle
                      fill="white"
                      stroke="#848484"
                      cx="10"
                      cy="10"
                      r="5"
                    ></circle>
                  </svg>
                </td>
                <td>
                  Constant
                  <div class="legend-clarifier">
                    <span>?</span>
                    <paper-tooltip
                      animation-delay="0"
                      position="right"
                      offset="0"
                    >
                      Node that outputs a constant value.
                    </paper-tooltip>
                  </div>
                </td>
              </tr>
              <tr>
                <td>
                  <tf-graph-icon type="SUMMARY" height="20"></tf-graph-icon>
                </td>
                <td>
                  Summary
                  <div class="legend-clarifier">
                    <span>?</span>
                    <paper-tooltip
                      animation-delay="0"
                      position="right"
                      offset="0"
                    >
                      Node that collects data for visualization within
                      TensorBoard.
                    </paper-tooltip>
                  </div>
                </td>
              </tr>
              <tr>
                <td>
                  <svg
                    class="icon"
                    height="15px"
                    preserveAspectRatio="xMinYMid meet"
                    viewBox="0 0 15 15"
                  >
                    <defs>
                      <marker
                        id="dataflow-arrowhead-legend"
                        fill="#bbb"
                        markerWidth="10"
                        markerHeight="10"
                        refX="9"
                        refY="5"
                        orient="auto-start-reverse"
                      >
                        <path d="M 0,0 L 10,5 L 0,10 C 3,7 3,3 0,0"></path>
                      </marker>
                    </defs>
                    <path
                      marker-end="url(#dataflow-arrowhead-legend)"
                      stroke="#bbb"
                      d="M2 9 l 29 0"
                      stroke-linecap="round"
                    ></path>
                  </svg>
                </td>
                <td>
                  Dataflow edge
                  <div class="legend-clarifier">
                    <span>?</span>
                    <paper-tooltip
                      animation-delay="0"
                      position="right"
                      offset="0"
                    >
                      Edge showing the data flow between operations. Edges flow
                      upwards unless arrowheads specify otherwise.
                    </paper-tooltip>
                  </div>
                </td>
              </tr>
              <tr>
                <td>
                  <svg
                    class="icon"
                    height="15px"
                    preserveAspectRatio="xMinYMid meet"
                    viewBox="0 0 15 15"
                  >
                    <path
                      stroke="#bbb"
                      d="M2 9 l 29 0"
                      stroke-linecap="round"
                      stroke-dasharray="2, 2"
                    ></path>
                  </svg>
                </td>
                <td>
                  Control dependency edge
                  <div class="legend-clarifier">
                    <span>?</span>
                    <paper-tooltip
                      animation-delay="0"
                      position="right"
                      offset="0"
                    >
                      Edge showing the control dependency between operations.
                    </paper-tooltip>
                  </div>
                </td>
              </tr>
              <tr>
                <td>
                  <svg
                    class="icon"
                    height="15px"
                    preserveAspectRatio="xMinYMid meet"
                    viewBox="0 0 15 15"
                  >
                    <defs>
                      <marker
                        id="reference-arrowhead-legend"
                        fill="#FFB74D"
                        markerWidth="10"
                        markerHeight="10"
                        refX="9"
                        refY="5"
                        orient="auto-start-reverse"
                      >
                        <path d="M 0,0 L 10,5 L 0,10 C 3,7 3,3 0,0"></path>
                      </marker>
                    </defs>
                    <path
                      marker-end="url(#reference-arrowhead-legend)"
                      stroke="#FFB74D"
                      d="M2 9 l 29 0"
                      stroke-linecap="round"
                    ></path>
                  </svg>
                </td>
                <td>
                  Reference edge
                  <div class="legend-clarifier">
                    <span>?</span>
                    <paper-tooltip
                      animation-delay="0"
                      position="right"
                      offset="0"
                    >
                      Edge showing that the outgoing operation node can mutate
                      the incoming tensor.
                    </paper-tooltip>
                  </div>
                </td>
              </tr>
            </tbody>
          </table>
        </div>
      </iron-collapse>
    </div>
  `;
    __decorate([
        property({
            type: Object,
            observer: '_statsChanged',
        }),
        __metadata("design:type", Object)
    ], TfGraphControls.prototype, "stats", void 0);
    __decorate([
        property({
            type: Object,
            notify: true,
        }),
        __metadata("design:type", Object)
    ], TfGraphControls.prototype, "devicesForStats", void 0);
    __decorate([
        property({
            type: String,
            notify: true,
        }),
        __metadata("design:type", String)
    ], TfGraphControls.prototype, "colorBy", void 0);
    __decorate([
        property({
            type: Object,
            notify: true,
        }),
        __metadata("design:type", Object)
    ], TfGraphControls.prototype, "colorByParams", void 0);
    __decorate([
        property({
            type: Array,
            observer: '_datasetsChanged',
        }),
        __metadata("design:type", Object)
    ], TfGraphControls.prototype, "datasets", void 0);
    __decorate([
        property({
            type: Object,
        }),
        __metadata("design:type", Object)
    ], TfGraphControls.prototype, "renderHierarchy", void 0);
    __decorate([
        property({
            type: Object,
            notify: true,
            readOnly: true,
            computed: '_computeSelection(datasets, _selectedRunIndex, _selectedTagIndex, _selectedGraphType)',
        }),
        __metadata("design:type", Object)
    ], TfGraphControls.prototype, "selection", void 0);
    __decorate([
        property({
            type: Object,
            notify: true,
        }),
        __metadata("design:type", Object)
    ], TfGraphControls.prototype, "selectedFile", void 0);
    __decorate([
        property({
            type: Number,
            observer: '_selectedRunIndexChanged',
        }),
        __metadata("design:type", Number)
    ], TfGraphControls.prototype, "_selectedRunIndex", void 0);
    __decorate([
        property({
            type: Boolean,
            notify: true,
        }),
        __metadata("design:type", Boolean)
    ], TfGraphControls.prototype, "traceInputs", void 0);
    __decorate([
        property({
            type: Number,
            observer: '_selectedTagIndexChanged',
        }),
        __metadata("design:type", Number)
    ], TfGraphControls.prototype, "_selectedTagIndex", void 0);
    __decorate([
        property({
            type: String,
        }),
        __metadata("design:type", String)
    ], TfGraphControls.prototype, "_selectedGraphType", void 0);
    __decorate([
        property({
            type: String,
            notify: true,
        }),
        __metadata("design:type", String)
    ], TfGraphControls.prototype, "selectedNode", void 0);
    __decorate([
        property({
            type: Boolean,
        }),
        __metadata("design:type", Boolean)
    ], TfGraphControls.prototype, "showSessionRunsDropdown", void 0);
    __decorate([
        property({
            type: Boolean,
        }),
        __metadata("design:type", Boolean)
    ], TfGraphControls.prototype, "showUploadButton", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfGraphControls.prototype, "healthPillsFeatureEnabled", void 0);
    __decorate([
        property({
            type: Boolean,
            notify: true,
        }),
        __metadata("design:type", Boolean)
    ], TfGraphControls.prototype, "healthPillsToggledOn", void 0);
    __decorate([
        property({
            type: Boolean,
        }),
        __metadata("design:type", Boolean)
    ], TfGraphControls.prototype, "_legendOpened", void 0);
    __decorate([
        computed('devicesForStats'),
        __metadata("design:type", Array),
        __metadata("design:paramtypes", [])
    ], TfGraphControls.prototype, "_currentDevices", null);
    __decorate([
        computed('colorByParams'),
        __metadata("design:type", Array),
        __metadata("design:paramtypes", [])
    ], TfGraphControls.prototype, "_currentDeviceParams", null);
    __decorate([
        computed('colorByParams'),
        __metadata("design:type", Array),
        __metadata("design:paramtypes", [])
    ], TfGraphControls.prototype, "_currentXlaClusterParams", null);
    __decorate([
        computed('colorByParams', 'colorBy'),
        __metadata("design:type", Object),
        __metadata("design:paramtypes", [])
    ], TfGraphControls.prototype, "_currentGradientParams", null);
    TfGraphControls = __decorate([
        customElement('tf-graph-controls')
    ], TfGraphControls);

    /* Copyright 2017 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the 'License');
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an 'AS IS' BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    class TpuCompatibilityProvider {
        /**
         * Returns true if the node's inferred device is not the TPU.
         * Note that this is only a best-effort check.
         */
        isNotTpuOp(opDevice) {
            if (opDevice.toLowerCase().search('cpu:') != -1) {
                return true;
            }
            if (opDevice.toLowerCase().search('gpu:') != -1) {
                return true;
            }
            return opDevice.toLowerCase().search('tpu') == -1;
        }
        opValid(opNode) {
            // Function library nodes are generally for internal use.
            if (opNode.name.search(FUNCTION_LIBRARY_NODE_PREFIX) == 0) {
                return true;
            }
            // Nodes that lack op types should be ignored.
            if (!opNode.op) {
                return true;
            }
            // If assigned a device that is not TPU-related assume op is valid.
            if (opNode.device && this.isNotTpuOp(opNode.device)) {
                return true;
            }
            // If assigned to the TPU_SYSTEM device, assume op is valid.
            if (opNode.device && opNode.device.search('TPU_SYSTEM') != -1) {
                return true;
            }
            return lodash.includes(TpuCompatibilityProvider.WHITELIST, opNode.op);
        }
    }
    /**
     * Allowed list of current Tensorflow ops valid on the TPU.
     * Note that some data types may be unsupported.
     */
    TpuCompatibilityProvider.WHITELIST = [
        'Abs',
        'Acos',
        'Acosh',
        'Add',
        'AddN',
        'AddV2',
        'AdjustContrastv2',
        'AdjustHue',
        'AdjustSaturation',
        'All',
        'AllToAll',
        'Angle',
        'Any',
        'ApproximateEqual',
        'ArgMax',
        'ArgMin',
        'Asin',
        'Asinh',
        'Assert',
        'AssignAddVariableOp',
        'AssignSubVariableOp',
        'AssignVariableOp',
        'Atan',
        'Atan2',
        'Atanh',
        'AvgPool',
        'AvgPool3D',
        'AvgPool3DGrad',
        'AvgPoolGrad',
        'BatchMatMul',
        'BatchMatMulV2',
        'BatchToSpace',
        'BatchToSpaceND',
        'BesselI0e',
        'BesselI1e',
        'Betainc',
        'BiasAdd',
        'BiasAddGrad',
        'BiasAddV1',
        'Bitcast',
        'BitwiseAnd',
        'BitwiseOr',
        'BitwiseXor',
        'BroadcastArgs',
        'BroadcastGradientArgs',
        'BroadcastTo',
        'Bucketize',
        'Case',
        'Cast',
        'Ceil',
        'CheckNumerics',
        'Cholesky',
        'ClipByValue',
        'CollectivePermute',
        'Complex',
        'ComplexAbs',
        'Concat',
        'ConcatOffset',
        'ConcatV2',
        'Conj',
        'ConjugateTranspose',
        'Const',
        'ControlTrigger',
        'Conv2D',
        'Conv2DBackpropFilter',
        'Conv2DBackpropInput',
        'Conv3D',
        'Conv3DBackpropFilterV2',
        'Conv3DBackpropInputV2',
        'Cos',
        'Cosh',
        'Cross',
        'CrossReplicaSum',
        'Cumprod',
        'Cumsum',
        'DataFormatDimMap',
        'DataFormatVecPermute',
        'DataFormatVecPermute',
        'DepthToSpace',
        'DepthwiseConv2dNative',
        'DepthwiseConv2dNativeBackpropFilter',
        'DepthwiseConv2dNativeBackpropInput',
        'Dequantize',
        'DeviceIndex',
        'Diag',
        'DiagPart',
        'Digamma',
        'Div',
        'DivNoNan',
        'DynamicStitch',
        'Einsum',
        'Elu',
        'EluGrad',
        'Empty',
        'EmptyTensorList',
        'EnsureShape',
        'Equal',
        'Erf',
        'Erfc',
        'Erfinv',
        'Exp',
        'ExpandDims',
        'Expm1',
        'ExtractImagePatches',
        'FFT',
        'FFT2D',
        'FFT3D',
        'FakeParam',
        'FakeQuantWithMinMaxArgs',
        'FakeQuantWithMinMaxArgsGradient',
        'FakeQuantWithMinMaxVars',
        'FakeQuantWithMinMaxVarsGradient',
        'Fill',
        'Floor',
        'FloorDiv',
        'FloorMod',
        'FusedBatchNorm',
        'FusedBatchNormGrad',
        'FusedBatchNormGradV2',
        'FusedBatchNormGradV3',
        'FusedBatchNormV2',
        'FusedBatchNormV3',
        'Gather',
        'GatherNd',
        'GatherV2',
        'GetItem',
        'Greater',
        'GreaterEqual',
        'HSVToRGB',
        'IFFT',
        'IFFT2D',
        'IFFT3D',
        'IRFFT',
        'IRFFT2D',
        'IRFFT3D',
        'Identity',
        'IdentityN',
        'If',
        'Igamma',
        'IgammaGradA',
        'Igammac',
        'Imag',
        'InTopKV2',
        'InfeedDequeue',
        'InfeedDequeueTuple',
        'InplaceAdd',
        'InplaceUpdate',
        'Inv',
        'Invert',
        'InvertPermutation',
        'IsFinite',
        'IsInf',
        'IsNan',
        'KthOrderStatistic',
        'L2Loss',
        'LRN',
        'LRNGrad',
        'LeakyRelu',
        'LeakyReluGrad',
        'LeftShift',
        'Less',
        'LessEqual',
        'Lgamma',
        'LinSpace',
        'ListDiff',
        'Log',
        'Log1p',
        'LogSoftmax',
        'LogicalAnd',
        'LogicalNot',
        'LogicalOr',
        'LowerBound',
        'MakeUnique',
        'MatMul',
        'MatrixBandPart',
        'MatrixDiag',
        'MatrixDiagPart',
        'MatrixDiagPartV2',
        'MatrixDiagPartV3',
        'MatrixDiagV2',
        'MatrixDiagV3',
        'MatrixInverse',
        'MatrixSetDiag',
        'MatrixSetDiagV2',
        'MatrixSetDiagV3',
        'MatrixSolve',
        'MatrixTriangularSolve',
        'Max',
        'MaxPool',
        'MaxPool3D',
        'MaxPool3DGrad',
        'MaxPool3DGradGrad',
        'MaxPoolGrad',
        'MaxPoolGradGrad',
        'MaxPoolGradGradV2',
        'MaxPoolGradV2',
        'MaxPoolV2',
        'Maximum',
        'Mean',
        'Min',
        'Minimum',
        'MirrorPad',
        'Mod',
        'Mul',
        'MulNoNan',
        'Multinomial',
        'Ndtri',
        'Neg',
        'NextAfter',
        'NoOp',
        'NonMaxSuppressionV4',
        'NotEqual',
        'OneHot',
        'OnesLike',
        'OutfeedEnqueue',
        'OutfeedEnqueueTuple',
        'Pack',
        'Pad',
        'PadV2',
        'ParallelDynamicStitch',
        'ParameterizedTruncatedNormal',
        'PartitionedCall',
        'PlaceholderWithDefault',
        'PopulationCount',
        'Pow',
        'PreventGradient',
        'Prod',
        'Qr',
        'QuantizeAndDequantizeV2',
        'QuantizeAndDequantizeV3',
        'RFFT',
        'RFFT2D',
        'RFFT3D',
        'RGBToHSV',
        'RandomGammaGrad',
        'RandomShuffle',
        'RandomStandardNormal',
        'RandomUniform',
        'RandomUniformInt',
        'Range',
        'Rank',
        'ReadVariableOp',
        'Real',
        'RealDiv',
        'Reciprocal',
        'ReciprocalGrad',
        'Relu',
        'Relu6',
        'Relu6Grad',
        'ReluGrad',
        'Reshape',
        'ResizeBilinear',
        'ResizeBilinearGrad',
        'ResizeNearestNeighbor',
        'ResizeNearestNeighborGrad',
        'ResourceApplyAdaMax',
        'ResourceApplyAdadelta',
        'ResourceApplyAdagrad',
        'ResourceApplyAdagradDA',
        'ResourceApplyAdagradV2',
        'ResourceApplyAdam',
        'ResourceApplyAddSign',
        'ResourceApplyCenteredRMSProp',
        'ResourceApplyFtrl',
        'ResourceApplyFtrlV2',
        'ResourceApplyGradientDescent',
        'ResourceApplyKerasMomentum',
        'ResourceApplyMomentum',
        'ResourceApplyPowerSign',
        'ResourceApplyProximalAdagrad',
        'ResourceApplyProximalGradientDescent',
        'ResourceApplyRMSProp',
        'ResourceGather',
        'ResourceScatterAdd',
        'ResourceScatterDiv',
        'ResourceScatterMax',
        'ResourceScatterMin',
        'ResourceScatterMul',
        'ResourceScatterNdAdd',
        'ResourceScatterNdSub',
        'ResourceScatterNdUpdate',
        'ResourceScatterSub',
        'ResourceScatterUpdate',
        'ResourceStridedSliceAssign',
        'Reverse',
        'ReverseSequence',
        'ReverseV2',
        'RightShift',
        'Rint',
        'Roll',
        'Round',
        'Rsqrt',
        'RsqrtGrad',
        'ScatterNd',
        'Select',
        'SelectV2',
        'SelfAdjointEigV2',
        'Selu',
        'SeluGrad',
        'Shape',
        'ShapeN',
        'Sigmoid',
        'SigmoidGrad',
        'Sign',
        'Sin',
        'Sinh',
        'Size',
        'Slice',
        'Snapshot',
        'Softmax',
        'SoftmaxCrossEntropyWithLogits',
        'Softplus',
        'SoftplusGrad',
        'Softsign',
        'SoftsignGrad',
        'SpaceToBatch',
        'SpaceToBatchND',
        'SpaceToDepth',
        'SparseMatMul',
        'SparseSoftmaxCrossEntropyWithLogits',
        'SparseToDense',
        'Split',
        'SplitV',
        'Sqrt',
        'SqrtGrad',
        'Square',
        'SquaredDifference',
        'Squeeze',
        'StackCloseV2',
        'StackPopV2',
        'StackPushV2',
        'StackV2',
        'StatefulPartitionedCall',
        'StatefulStandardNormalV2',
        'StatefulTruncatedNormal',
        'StatefulUniform',
        'StatefulUniformFullInt',
        'StatefulUniformInt',
        'StatelessCase',
        'StatelessIf',
        'StatelessMultinomial',
        'StatelessRandomNormal',
        'StatelessRandomUniform',
        'StatelessRandomUniformFullInt',
        'StatelessRandomUniformInt',
        'StatelessTruncatedNormal',
        'StatelessWhile',
        'StopGradient',
        'StridedSlice',
        'StridedSliceGrad',
        'Sub',
        'Sum',
        'Svd',
        'SymbolicGradient',
        'TPUEmbeddingActivations',
        'Tan',
        'Tanh',
        'TanhGrad',
        'TensorArrayCloseV3',
        'TensorArrayConcatV3',
        'TensorArrayGatherV3',
        'TensorArrayGradV3',
        'TensorArrayReadV3',
        'TensorArrayScatterV3',
        'TensorArraySizeV3',
        'TensorArraySplitV3',
        'TensorArrayV3',
        'TensorArrayWriteV3',
        'TensorListConcatV2',
        'TensorListElementShape',
        'TensorListFromTensor',
        'TensorListGather',
        'TensorListGetItem',
        'TensorListLength',
        'TensorListPopBack',
        'TensorListPushBack',
        'TensorListReserve',
        'TensorListSetItem',
        'TensorListSplit',
        'TensorListStack',
        'TensorScatterAdd',
        'TensorScatterSub',
        'TensorScatterUpdate',
        'Tile',
        'TopKUnique',
        'TopKV2',
        'TopKWithUnique',
        'Transpose',
        'TridiagonalSolve',
        'TruncateDiv',
        'TruncateMod',
        'TruncatedNormal',
        'Unpack',
        'UnsortedSegmentMax',
        'UnsortedSegmentMin',
        'UnsortedSegmentProd',
        'UnsortedSegmentSum',
        'UpperBound',
        'VarIsInitializedOp',
        'VariableShape',
        'Where',
        'While',
        'Xdivy',
        'XlaBroadcastHelper',
        'XlaConv',
        'XlaDequantize',
        'XlaDot',
        'XlaDynamicSlice',
        'XlaDynamicUpdateSlice',
        'XlaEinsum',
        'XlaGather',
        'XlaHostCompute',
        'XlaIf',
        'XlaKeyValueSort',
        'XlaPad',
        'XlaRecv',
        'XlaRecvFromHost',
        'XlaReduce',
        'XlaReduceWindow',
        'XlaReplicaId',
        'XlaScatter',
        'XlaSelectAndScatter',
        'XlaSelfAdjointEig',
        'XlaSend',
        'XlaSendToHost',
        'XlaSharding',
        'XlaSort',
        'XlaSpmdFullToShardShape',
        'XlaSpmdShardToFullShape',
        'XlaSvd',
        'XlaWhile',
        'Xlog1py',
        'Xlogy',
        'ZerosLike',
        // Ops below are manually whitelisted and should not be evaluated for
        // compatibility for various reasons.
        // Control flow ops.
        'Enter',
        'Exit',
        'LoopCond',
        'Merge',
        'NextIteration',
        'Switch',
        // Ops below are inserted by the compiler.
        '_Arg',
        '_ArrayToList',
        '_FusedBatchNormEx',
        '_ListToArray',
        '_ParallelConcatUpdate',
        '_RecvTPUEmbeddingActivations',
        '_RecvTPUEmbeddingDeduplicationData',
        '_Retval',
        '_SendTPUEmbeddingGradients',
        '_TPUCompile',
        '_TPUExecute',
        '_UnaryOpsComposition',
        // Distributed TPU ops.
        'TPUCompilationResult',
        'TPUReplicatedInput',
        'TPUReplicatedOutput',
        'TPUReplicateMetadata',
        // Checkpointing ops.
        'MergeV2Checkpoints',
        'RestoreV2',
        'SaveV2',
        // Miscellaneous CPU ops.
        'Abort',
        'Assert',
        'Assign',
        'Placeholder',
        'PlaceholderV2',
        'ShardedFilename',
        'StringJoin',
        'Variable',
        'VariableV2',
        'VarHandleOp',
        // Summary ops.
        'AudioSummary',
        'AudioSummaryV2',
        'DebugNumericSummary',
        'HistogramSummary',
        'ImageSummary',
        'MergeSummary',
        'ScalarSummary',
        'StatsAggregatorSummary',
    ];
    function checkOpsForCompatibility(graph, provider) {
        if (provider === null) {
            throw new Error('Compatibility provider required, but got: ' + provider);
        }
        lodash.each(graph.nodes, (node) => {
            node.compatible = provider.opValid(node);
            lodash.each(node.inEmbeddings, (node) => {
                node.compatible = provider.opValid(node);
            });
            lodash.each(node.outEmbeddings, (node) => {
                node.compatible = provider.opValid(node);
            });
        });
    }

    function parseValue(value) {
        if (value === 'true') {
            return true;
        }
        if (value === 'false') {
            return false;
        }
        let firstChar = value[0];
        if (firstChar === '"') {
            return value.substring(1, value.length - 1);
        }
        let num = parseFloat(value);
        return isNaN(num) ? value : num;
    }
    /**
     * Fetches a text file and returns a promise of the result.
     */
    function fetchPbTxt(filepath) {
        return new Promise((resolve, reject) => {
            fetch(filepath).then((res) => {
                // Fetch does not reject for 400+.
                if (res.ok) {
                    res.arrayBuffer().then(resolve, reject);
                }
                else {
                    res.text().then(reject, reject);
                }
            });
        });
    }
    /**
     * Fetches the metadata file, parses it and returns a promise of the result.
     */
    function fetchAndParseMetadata(path, tracker) {
        return runTask('Reading metadata pbtxt', 40, () => {
            if (path == null) {
                return Promise.resolve(null);
            }
            return fetchPbTxt(path);
        }, tracker)
            .then((arrayBuffer) => {
            return runAsyncPromiseTask('Parsing metadata.pbtxt', 60, () => {
                return arrayBuffer != null
                    ? parseStatsPbTxt(arrayBuffer)
                    : Promise.resolve(null);
            }, tracker);
        });
    }
    /**
     * Fetches the graph file, parses it and returns a promise of the result. The
     * result will be undefined if the graph is empty.
     */
    function fetchAndParseGraphData(path, pbTxtFile, tracker) {
        return runAsyncPromiseTask('Reading graph pbtxt', 40, () => {
            if (pbTxtFile) {
                return new Promise(function (resolve, reject) {
                    let fileReader = new FileReader();
                    fileReader.onload = () => resolve(fileReader.result);
                    fileReader.onerror = () => reject(fileReader.error);
                    fileReader.readAsArrayBuffer(pbTxtFile);
                });
            }
            else {
                return fetchPbTxt(path);
            }
        }, tracker)
            .then((arrayBuffer) => {
            return runAsyncPromiseTask('Parsing graph.pbtxt', 60, () => {
                return parseGraphPbTxt(arrayBuffer);
            }, tracker);
        });
    }
    /**
     * Parse a file object in a streaming fashion line by line (or custom delim).
     * Can handle very large files.
     * @param input The file object as an array buffer.
     * @param callback The callback called on each line
     * @param chunkSize The size of each read chunk. (optional)
     * @param delim The delimiter used to split a line. (optional)
     * @returns Promise that resolves with true when it is finished.
     */
    function streamParse(arrayBuffer, callback, chunkSize = 1000000, delim = '\n') {
        return new Promise(function (resolve, reject) {
            function readChunk(oldData, newData, offset) {
                const doneReading = offset >= arrayBuffer.byteLength;
                const parts = newData.split(delim);
                parts[0] = oldData + parts[0];
                // The last part may be part of a longer string that got cut off
                // due to the chunking.
                const remainder = doneReading ? '' : parts.pop();
                for (let part of parts) {
                    try {
                        callback(part);
                    }
                    catch (e) {
                        reject(e);
                        return;
                    }
                }
                if (doneReading) {
                    resolve(true);
                    return;
                }
                const nextChunk = new Blob([
                    arrayBuffer.slice(offset, offset + chunkSize),
                ]);
                const file = new FileReader();
                file.onload = function (e) {
                    readChunk(remainder, e.target.result, offset + chunkSize);
                };
                file.readAsText(nextChunk);
            }
            readChunk('', '', 0);
        });
    }
    /**
     * Since proto-txt doesn't explicitly say whether an attribute is repeated
     * (an array) or not, we keep a hard-coded list of attributes that are known
     * to be repeated. This list is used in parsing time to convert repeated
     * attributes into arrays even when the attribute only shows up once in the
     * object.
     * Repeated fields have to be in sync with graph.proto and all of its
     * dependencies.
     * See https://github.com/tensorflow/tensorflow/blob/master/tensorflow/core/framework/graph.proto
     */
    const GRAPH_REPEATED_FIELDS = {
        'library.function': true,
        'library.function.node_def': true,
        'library.function.node_def.input': true,
        'library.function.node_def.attr': true,
        'library.function.node_def.attr.value.list.b': true,
        'library.function.node_def.attr.value.list.f': true,
        'library.function.node_def.attr.value.list.func': true,
        'library.function.node_def.attr.value.list.i': true,
        'library.function.node_def.attr.value.list.s': true,
        'library.function.node_def.attr.value.list.shape': true,
        'library.function.node_def.attr.value.list.shape.dim': true,
        'library.function.node_def.attr.value.list.tensor': true,
        'library.function.node_def.attr.value.list.type': true,
        'library.function.node_def.attr.value.shape.dim': true,
        'library.function.node_def.attr.value.tensor.string_val': true,
        'library.function.node_def.attr.value.tensor.tensor_shape.dim': true,
        'library.function.signature.input_arg': true,
        'library.function.signature.output_arg': true,
        'library.versions': true,
        node: true,
        'node.input': true,
        'node.attr': true,
        'node.attr.value.list.b': true,
        'node.attr.value.list.f': true,
        'node.attr.value.list.func': true,
        'node.attr.value.list.i': true,
        'node.attr.value.list.s': true,
        'node.attr.value.list.shape': true,
        'node.attr.value.list.shape.dim': true,
        'node.attr.value.list.tensor': true,
        'node.attr.value.list.type': true,
        'node.attr.value.shape.dim': true,
        'node.attr.value.tensor.string_val': true,
        'node.attr.value.tensor.tensor_shape.dim': true,
    };
    const METADATA_REPEATED_FIELDS = {
        'step_stats.dev_stats': true,
        'step_stats.dev_stats.node_stats': true,
        'step_stats.dev_stats.node_stats.output': true,
        'step_stats.dev_stats.node_stats.memory': true,
        'step_stats.dev_stats.node_stats.output.tensor_description.shape.dim': true,
    };
    /**
     * Parses an ArrayBuffer of a proto txt file into a raw Graph object.
     */
    function parseGraphPbTxt(input) {
        return parsePbtxtFile(input, GRAPH_REPEATED_FIELDS);
    }
    /**
     * Parses an ArrayBuffer of a proto txt file into a StepStats object.
     */
    function parseStatsPbTxt(input) {
        return parsePbtxtFile(input, METADATA_REPEATED_FIELDS).then((obj) => obj['step_stats']);
    }
    /**
     * Parses a ArrayBuffer of a proto txt file into javascript object.
     *
     * @param input The ArrayBuffer or file object implementing slice.
     * @param repeatedFields Map (Set) of all the repeated fields, since you can't
     *   tell directly from the pbtxt if a field is repeated or not.
     * @returns The parsed object.
     */
    function parsePbtxtFile(input, repeatedFields) {
        let output = {};
        let stack = [];
        let path = [];
        let current = output;
        function splitNameAndValueInAttribute(line) {
            let colonIndex = line.indexOf(':');
            let name = line.substring(0, colonIndex).trim();
            let value = parseValue(line.substring(colonIndex + 2).trim());
            return {
                name: name,
                value: value,
            };
        }
        /**
         * Adds a value, given the attribute name and the host object. If the
         * attribute already exists, but is not an array, it will convert it to an
         * array of values.
         *
         * @param obj The host object that holds the attribute.
         * @param name The attribute name (key).
         * @param value The attribute value.
         * @param path A path that identifies the attribute. Used to check if
         *     an attribute is an array or not.
         */
        function addAttribute(obj, name, value, path) {
            // We treat 'node' specially since it is done so often.
            let existingValue = obj[name];
            if (existingValue == null) {
                obj[name] = path.join('.') in repeatedFields ? [value] : value;
            }
            else if (Array.isArray(existingValue)) {
                existingValue.push(value);
            }
            else {
                obj[name] = [existingValue, value];
            }
        }
        // Run through the file a line at a time.
        return streamParse(input, function (line) {
            if (!line) {
                return;
            }
            line = line.trim();
            switch (line[line.length - 1]) {
                case '{': // create new object
                    let name = line.substring(0, line.length - 2).trim();
                    let newValue = {};
                    stack.push(current);
                    path.push(name);
                    addAttribute(current, name, newValue, path);
                    current = newValue;
                    break;
                case '}':
                    current = stack.pop();
                    path.pop();
                    break;
                default:
                    let x = splitNameAndValueInAttribute(line);
                    addAttribute(current, x.name, x.value, path.concat(x.name));
                    break;
            }
        }).then(function () {
            return output;
        });
    }

    function fetchAndConstructHierarchicalGraph(tracker, remotePath, pbTxtFile, compatibilityProvider = new TpuCompatibilityProvider(), hierarchyParams = DefaultHierarchyParams) {
        const dataTracker = getSubtaskTracker(tracker, 30, 'Data');
        const graphTracker = getSubtaskTracker(tracker, 20, 'Graph');
        const hierarchyTracker = getSubtaskTracker(tracker, 50, 'Namespace hierarchy');
        return fetchAndParseGraphData(remotePath, pbTxtFile, dataTracker)
            .then(function (graph) {
            if (!graph.node) {
                throw new Error('The graph is empty. This can happen when ' +
                    'TensorFlow could not trace any graph. Please refer to ' +
                    'https://github.com/tensorflow/tensorboard/issues/1961 for more ' +
                    'information.');
            }
            return build(graph, DefaultBuildParams, graphTracker);
        }, () => {
            throw new Error('Malformed GraphDef. This can sometimes be caused by ' +
                'a bad network connection or difficulty reconciling multiple ' +
                'GraphDefs; for the latter case, please refer to ' +
                'https://github.com/tensorflow/tensorboard/issues/1929.');
        })
            .then((graph) => __awaiter(this, void 0, void 0, function* () {
            // Populate compatibile field of OpNode based on whitelist
            checkOpsForCompatibility(graph, compatibilityProvider);
            const graphHierarchy = yield build$1(graph, hierarchyParams, hierarchyTracker);
            return { graph, graphHierarchy };
        }))
            .catch((e) => {
            // Generic error catch, for errors that happened outside
            // asynchronous tasks.
            const msg = `Graph visualization failed.\n\n${e}`;
            tracker.reportError(msg, e);
            // Don't swallow the error.
            throw e;
        });
    }

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * Data loader for tf-graph-dashboard.
     *
     * The loader loads op graph, conceptual graphs, and RunMetadata associated with
     * an op graph which is the major difference from the tf-graph-loader which is
     * only capable of loading an op graph. Another difference is that the loader
     * takes `selection` from the tf-graph-controls as an input as opposed to URL
     * path of an data endpoint.
     */
    let TfGraphDashboardLoader = class TfGraphDashboardLoader extends LegacyElementMixin(PolymerElement) {
        constructor() {
            super(...arguments);
            this.compatibilityProvider = new TpuCompatibilityProvider();
            this.hierarchyParams = DefaultHierarchyParams;
            this._template = null;
        }
        _selectionChanged() {
            // selection can change a lot within a microtask.
            // Don't fetch too much too fast and introduce race condition.
            this.debounce('selectionchange', () => {
                this._load(this.selection);
            });
        }
        _load(selection) {
            const { run, tag, type: selectionType } = selection;
            switch (selectionType) {
                case SelectionType.OP_GRAPH:
                case SelectionType.CONCEPTUAL_GRAPH: {
                    // Clear stats about the previous graph.
                    (function () {
                        this._setOutStats(null);
                    }.bind(this)());
                    const params = new URLSearchParams();
                    params.set('run', run);
                    params.set('conceptual', String(selectionType === SelectionType.CONCEPTUAL_GRAPH));
                    if (tag)
                        params.set('tag', tag);
                    const graphPath = getRouter().pluginRoute('graphs', '/graph', params);
                    return this._fetchAndConstructHierarchicalGraph(graphPath).then(() => {
                        this._graphRunTag = { run, tag };
                    });
                }
                case SelectionType.PROFILE: {
                    const { tags } = this.datasets.find(({ name }) => name === run);
                    const tagMeta = tags.find((t) => t.tag === tag);
                    // In case current tag misses opGraph but has profile information,
                    // we fallback to the v1 behavior of fetching the run graph.
                    const requiredOpGraphTag = tagMeta.opGraph ? tag : null;
                    console.assert(tags.find((t) => t.tag === requiredOpGraphTag), `Required tag (${requiredOpGraphTag}) is missing.`);
                    const shouldFetchGraph = !this._graphRunTag ||
                        this._graphRunTag.run !== run ||
                        this._graphRunTag.tag !== requiredOpGraphTag;
                    const maybeFetchGraphPromise = shouldFetchGraph
                        ? this._load({
                            run,
                            tag: requiredOpGraphTag,
                            type: SelectionType.OP_GRAPH,
                        })
                        : Promise.resolve();
                    const params = new URLSearchParams();
                    params.set('tag', tag);
                    params.set('run', run);
                    const metadataPath = getRouter().pluginRoute('graphs', '/run_metadata', params);
                    return maybeFetchGraphPromise.then(() => this._readAndParseMetadata(metadataPath));
                }
                default:
                    return Promise.reject(new Error(`Unknown selection type: ${selectionType}`));
            }
        }
        _readAndParseMetadata(path) {
            // Reset the progress bar to 0.
            this.set('progress', {
                value: 0,
                msg: '',
            });
            var tracker = getTracker(this);
            fetchAndParseMetadata(path, tracker).then(function (stats) {
                this._setOutStats(stats);
            }.bind(this));
        }
        _fetchAndConstructHierarchicalGraph(path, pbTxtFile) {
            // Reset the progress bar to 0.
            this.set('progress', {
                value: 0,
                msg: '',
            });
            const tracker = getTracker(this);
            return fetchAndConstructHierarchicalGraph(tracker, path, pbTxtFile, this.compatibilityProvider, this.hierarchyParams)
                .then(function ({ graph, graphHierarchy }) {
                this._setOutGraph(graph);
                this._setOutGraphHierarchy(graphHierarchy);
            }.bind(this));
        }
        _selectedFileChanged() {
            var e = this.selectedFile;
            if (!e) {
                return;
            }
            const target = e.target;
            const file = target.files[0];
            if (!file) {
                return;
            }
            // Clear out the value of the file chooser. This ensures that if the user
            // selects the same file, we'll re-read it.
            target.value = '';
            this._fetchAndConstructHierarchicalGraph(null, file);
        }
    };
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfGraphDashboardLoader.prototype, "datasets", void 0);
    __decorate([
        property({
            type: Object,
            notify: true,
        }),
        __metadata("design:type", Object)
    ], TfGraphDashboardLoader.prototype, "progress", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraphDashboardLoader.prototype, "selection", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraphDashboardLoader.prototype, "selectedFile", void 0);
    __decorate([
        property({
            type: Object,
        }),
        __metadata("design:type", Object)
    ], TfGraphDashboardLoader.prototype, "compatibilityProvider", void 0);
    __decorate([
        property({
            type: Object,
        }),
        __metadata("design:type", Object)
    ], TfGraphDashboardLoader.prototype, "hierarchyParams", void 0);
    __decorate([
        property({
            type: Object,
            readOnly: true,
            notify: true,
        }),
        __metadata("design:type", Object)
    ], TfGraphDashboardLoader.prototype, "outGraphHierarchy", void 0);
    __decorate([
        property({
            type: Object,
            readOnly: true,
            notify: true,
        }),
        __metadata("design:type", Object)
    ], TfGraphDashboardLoader.prototype, "outGraph", void 0);
    __decorate([
        property({
            type: Object,
            readOnly: true,
            notify: true,
        }),
        __metadata("design:type", Object)
    ], TfGraphDashboardLoader.prototype, "outStats", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraphDashboardLoader.prototype, "_graphRunTag", void 0);
    __decorate([
        observe('selection', 'compatibilityProvider'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfGraphDashboardLoader.prototype, "_selectionChanged", null);
    __decorate([
        observe('selectedFile', 'compatibilityProvider'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfGraphDashboardLoader.prototype, "_selectedFileChanged", null);
    TfGraphDashboardLoader = __decorate([
        customElement('tf-graph-dashboard-loader')
    ], TfGraphDashboardLoader);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * The (string) name for the run of the selected dataset in the graph dashboard.
     */
    const RUN_STORAGE_KEY = 'run';
    /**
     * tf-graph-dashboard displays a graph from a TensorFlow run.
     *
     * It has simple behavior: Creates a url-generator and run-generator
     * to talk to the backend, and then passes the runsWithGraph (list of runs with
     * associated graphs) along with the url generator into tf-graph-board for display.
     *
     * If there are multiple runs with graphs, the first run's graph is shown
     * by default. The user can select a different run from a dropdown menu.
     */
    let TfGraphDashboard = class TfGraphDashboard extends LegacyElementMixin(PolymerElement) {
        constructor() {
            super(...arguments);
            /**
             * @type {!Array<!RunItem>}
             */
            this._datasets = [];
            this._datasetsFetched = false;
            this._selectedDataset = 0;
            this._requestManager = new RequestManager();
            this._canceller = new Canceller();
            this.specificHealthPillStep = 0;
            this.healthPillsToggledOn = false;
            // An array of alerts (in chronological order) provided by debugging libraries on when bad
            // values (NaN, +/- Inf) appear.
            this._debuggerNumericAlerts = [];
            // Maps the names of nodes to an array of health pills (HealthPillDatums).
            this._nodeNamesToHealthPills = {};
            // A strictly increasing ID. Each request for health pills has a unique ID. This helps us
            // identify stale requests.
            this._healthPillRequestId = 1;
            // The request for health pills at a specific step (as opposed to all sampled health pills) may
            // involve slow disk reads. Hence, we throttle to 1 of those requests every this many ms.
            this._healthPillStepRequestTimerDelay = 500;
            this.run = getStringInitializer(RUN_STORAGE_KEY, {
                defaultValue: '',
                useLocalStorage: false,
            })
                .call(this);
            this._runObserver = getStringObserver(RUN_STORAGE_KEY, {
                defaultValue: '',
                polymerProperty: 'run',
                useLocalStorage: false,
            });
        }
        attached() {
            this.set('_isAttached', true);
        }
        detached() {
            this.set('_isAttached', false);
        }
        ready() {
            super.ready();
            this.addEventListener('node-toggle-expand', this._handleNodeToggleExpand.bind(this));
        }
        reload() {
            if (!this._debuggerDataEnabled) {
                // Check if the debugger plugin is enabled now.
                this._requestManager.request(getRouter().pluginsListing()).then(this._canceller.cancellable((result) => {
                    if (result.cancelled) {
                        return;
                    }
                    if (result.value['debugger']) {
                        // The debugger plugin is enabled. Request debugger-related
                        // data. Perhaps the debugger plugin had been disabled
                        // beforehand because no bad values (NaN, -/+ Inf) had been
                        // found and muted_if_healthy had been on.
                        this.set('_debuggerDataEnabled', true);
                    }
                }));
            }
            this._maybeFetchHealthPills();
        }
        _fit() {
            this.$$('#graphboard').fit();
        }
        _fetchDataset() {
            return this._requestManager.request(getRouter().pluginRoute('graphs', '/info'));
        }
        /*
         * See also _maybeFetchHealthPills, _initiateNetworkRequestForHealthPills.
         * This function returns a promise with the raw health pill data.
         */
        _fetchHealthPills(nodeNames, step) {
            const postData = {
                node_names: JSON.stringify(nodeNames),
                // Events files with debugger data fall under this special run.
                run: '__debugger_data__',
            };
            if (step !== undefined) {
                // The user requested health pills for a specific step. This request
                // might be slow since the backend reads events sequentially from disk.
                postData['step'] = step;
            }
            const url = getRouter().pluginRoute('debugger', '/health_pills');
            return this._requestManager.request(url, postData);
        }
        _fetchDebuggerNumericsAlerts() {
            return this._requestManager.request(getRouter().pluginRoute('debugger', '/numerics_alert_report'));
        }
        _graphUrl(run, limitAttrSize, largeAttrsKey) {
            return getRouter().pluginRoute('graphs', '/graph', new URLSearchParams({
                run: run,
                limit_attr_size: limitAttrSize,
                large_attrs_key: largeAttrsKey,
            }));
        }
        _shouldRequestHealthPills() {
            // Do not load debugger data if the feature is disabled, if the user toggled off the feature,
            // or if the graph itself has not loaded yet. We need the graph to load so that we know which
            // nodes to request health pills for.
            return (this._debuggerDataEnabled &&
                this.healthPillsToggledOn &&
                this._renderHierarchy &&
                this._datasetsState(this._datasetsFetched, this._datasets, 'PRESENT'));
        }
        _maybeInitializeDashboard() {
            var isAttached = this._isAttached;
            if (this._initialized || !isAttached) {
                // Either this dashboard is already initialized ... or we are not yet ready to initialize.
                return;
            }
            this.set('_compatibilityProvider', new TpuCompatibilityProvider());
            // Set this to true so we only initialize once.
            this._initialized = true;
            this._fetchDataset().then((dataset) => {
                const runNames = Object.keys(dataset);
                // Transform raw data into UI friendly data.
                this._datasets = runNames
                    .sort(compareTagNames)
                    .map((runName) => {
                    const runData = dataset[runName];
                    const tagNames = Object.keys(runData.tags).sort(compareTagNames);
                    const tags = tagNames
                        .map((name) => runData.tags[name])
                        .map(({ tag, conceptual_graph, op_graph, profile }) => ({
                        tag,
                        displayName: tag,
                        conceptualGraph: conceptual_graph,
                        opGraph: op_graph,
                        profile,
                    }));
                    // Translate a run-wide GraphDef into specially named (without a tag) op graph
                    // to abstract the difference between run_graph vs. op_graph from other
                    // components.
                    const tagsWithV1Graph = runData.run_graph
                        ? [
                            {
                                tag: null,
                                displayName: 'Default',
                                conceptualGraph: false,
                                opGraph: true,
                                profile: false,
                            },
                            ...tags,
                        ]
                        : tags;
                    return { name: runName, tags: tagsWithV1Graph };
                });
                this._datasetsFetched = true;
            });
        }
        _determineSelectedDataset() {
            var datasetsFetched = this._datasetsFetched;
            var datasets = this._datasets;
            var run = this.run;
            // By default, load the first dataset.
            if (!run) {
                // By default, load the first dataset.
                this.set('_selectedDataset', 0);
                return;
            }
            // If the URL specifies a dataset, load it.
            const dataset = datasets.findIndex((d) => d.name === run);
            if (dataset === -1) {
                if (datasetsFetched) {
                    // Tell the user if the dataset cannot be found to avoid misleading
                    // the user.
                    const dialog = this.$$('#error-dialog');
                    dialog.textContent = `No dataset named "${run}" could be found.`;
                    dialog.open();
                }
                return;
            }
            this.set('_selectedDataset', dataset);
        }
        _updateSelectedDatasetName() {
            var datasetsFetched = this._datasetsFetched;
            var datasets = this._datasets;
            var selectedDataset = this._selectedDataset;
            if (!datasetsFetched)
                return;
            // Cannot update `run` to update the hash in case datasets for graph is empty.
            if (datasets.length <= selectedDataset)
                return;
            this.set('run', datasets[selectedDataset].name);
        }
        _requestHealthPills() {
            this.set('_areHealthPillsLoading', true);
            var requestId = ++this._healthPillRequestId;
            if (this._healthPillStepRequestTimerId !== null) {
                // A request for health pills is already scheduled to be initiated. Clear it, and schedule a
                // new request.
                window.clearTimeout(this._healthPillStepRequestTimerId);
                this._healthPillStepRequestTimerId = null;
            }
            if (this.allStepsModeEnabled) {
                // This path may be slow. Schedule network requests to start some time later. If another
                // request is scheduled in the mean time, drop this current request.
                this._healthPillStepRequestTimerId = setTimeout(function () {
                    this._healthPillStepRequestTimerId = null;
                    this._initiateNetworkRequestForHealthPills(requestId);
                }.bind(this), this._healthPillStepRequestTimerDelay);
            }
            else {
                // The user is fetching sampled steps. This path is fast, so no need to throttle. Directly
                // fetch the health pills across the network.
                this._initiateNetworkRequestForHealthPills(requestId);
            }
        }
        // Initiates the network request for health pills. Do not directly call this method - network
        // requests may be throttled. Instead, call _requestHealthPills, which uses this method.
        _initiateNetworkRequestForHealthPills(requestId) {
            if (this._healthPillRequestId !== requestId) {
                // This possibly scheduled request was outdated before it was even sent across the network. Do
                // not bother initiating it.
                return;
            }
            const specificStep = this.allStepsModeEnabled
                ? this.specificHealthPillStep
                : undefined;
            const healthPillsPromise = this._fetchHealthPills(this._renderHierarchy.getNamesOfRenderedOps(), specificStep);
            const alertsPromise = this._fetchDebuggerNumericsAlerts();
            Promise.all([healthPillsPromise, alertsPromise]).then(function (result) {
                var healthPillsResult = result[0];
                var alertsResult = result[1];
                if (!this.healthPillsToggledOn) {
                    // The user has opted to hide health pills via the toggle button.
                    return;
                }
                if (requestId !== this._healthPillRequestId) {
                    // This response is no longer relevant.
                    return;
                }
                // Set the index for which step to show for the health pills. By default, show the last step.
                // A precondition we assume (that Tensorboard's reservoir sampling guarantees) is that all
                // node names should be mapped to the same number of steps.
                for (var nodeName in healthPillsResult) {
                    this.set('_healthPillStepIndex', healthPillsResult[nodeName].length - 1);
                    break;
                }
                this.set('_debuggerNumericAlerts', alertsResult);
                this.set('_nodeNamesToHealthPills', healthPillsResult);
                this.set('_areHealthPillsLoading', false);
                this.set('_healthPillStepRequestTimerId', null);
            }.bind(this));
        }
        _datasetsState(datasetsFetched, datasets, state) {
            if (!datasetsFetched)
                return state === 'NOT_LOADED';
            if (!datasets || !datasets.length)
                return state === 'EMPTY';
            return state === 'PRESENT';
        }
        _renderHierarchyChanged(renderHierarchy) {
            // Reload any data on the graph when the render hierarchy (which determines which nodes are
            // rendered) changes.
            this.reload();
        }
        _handleNodeToggleExpand() {
            // Nodes were toggled. We may need to request health pills for more nodes.
            this._maybeFetchHealthPills();
        }
        _healthPillsToggledOnChanged(healthPillsToggledOn) {
            if (healthPillsToggledOn) {
                // Load health pills.
                this.reload();
            }
            else {
                // Remove all health pills by setting an empty mapping.
                this.set('_nodeNamesToHealthPills', {});
            }
        }
        // Fetch health pills for a specific step if applicable.
        _maybeFetchHealthPills() {
            if (!this._shouldRequestHealthPills()) {
                return;
            }
            this._requestHealthPills();
        }
    };
    TfGraphDashboard.template = html `
    <paper-dialog id="error-dialog" with-backdrop></paper-dialog>
    <template
      is="dom-if"
      if="[[_datasetsState(_datasetsFetched, _datasets, 'EMPTY')]]"
    >
      <div style="max-width: 540px; margin: 80px auto 0 auto;">
        <h3>No graph definition files were found.</h3>
        <p>
          To store a graph, create a
          <code>tf.summary.FileWriter</code>
          and pass the graph either via the constructor, or by calling its
          <code>add_graph()</code> method. You may want to check out the
          <a href="https://www.tensorflow.org/get_started/graph_viz"
            >graph visualizer tutorial</a
          >.
        </p>

        <p>
          If you’re new to using TensorBoard, and want to find out how to add
          data and set up your event files, check out the
          <a
            href="https://github.com/tensorflow/tensorboard/blob/master/README.md"
            >README</a
          >
          and perhaps the
          <a
            href="https://www.tensorflow.org/get_started/summaries_and_tensorboard"
            >TensorBoard tutorial</a
          >.
        </p>

        <p>
          If you think TensorBoard is configured properly, please see
          <a
            href="https://github.com/tensorflow/tensorboard/blob/master/README.md#my-tensorboard-isnt-showing-any-data-whats-wrong"
            >the section of the README devoted to missing data problems</a
          >
          and consider filing an issue on GitHub.
        </p>
      </div>
    </template>
    <template
      is="dom-if"
      if="[[_datasetsState(_datasetsFetched, _datasets, 'PRESENT')]]"
    >
      <tf-dashboard-layout>
        <tf-graph-controls
          id="controls"
          class="sidebar"
          slot="sidebar"
          devices-for-stats="{{_devicesForStats}}"
          color-by-params="[[_colorByParams]]"
          stats="[[_stats]]"
          color-by="{{_colorBy}}"
          datasets="[[_datasets]]"
          render-hierarchy="[[_renderHierarchy]]"
          selection="{{_selection}}"
          selected-file="{{_selectedFile}}"
          selected-node="{{_selectedNode}}"
          health-pills-feature-enabled="[[_debuggerDataEnabled]]"
          health-pills-toggled-on="{{healthPillsToggledOn}}"
          on-fit-tap="_fit"
          trace-inputs="{{_traceInputs}}"
        ></tf-graph-controls>
        <div class="center" slot="center">
          <tf-graph-dashboard-loader
            id="loader"
            datasets="[[_datasets]]"
            selection="[[_selection]]"
            selected-file="[[_selectedFile]]"
            out-graph-hierarchy="{{_graphHierarchy}}"
            out-graph="{{_graph}}"
            out-stats="{{_stats}}"
            progress="{{_progress}}"
            hierarchy-params="[[_hierarchyParams]]"
            compatibility-provider="[[_compatibilityProvider]]"
          ></tf-graph-dashboard-loader>
          <tf-graph-board
            id="graphboard"
            devices-for-stats="[[_devicesForStats]]"
            color-by="[[_colorBy]]"
            color-by-params="{{_colorByParams}}"
            graph-hierarchy="[[_graphHierarchy]]"
            graph="[[_graph]]"
            hierarchy-params="[[_hierarchyParams]]"
            progress="[[_progress]]"
            debugger-data-enabled="[[_debuggerDataEnabled]]"
            are-health-pills-loading="[[_areHealthPillsLoading]]"
            debugger-numeric-alerts="[[_debuggerNumericAlerts]]"
            node-names-to-health-pills="[[_nodeNamesToHealthPills]]"
            all-steps-mode-enabled="{{allStepsModeEnabled}}"
            specific-health-pill-step="{{specificHealthPillStep}}"
            health-pill-step-index="[[_healthPillStepIndex]]"
            render-hierarchy="{{_renderHierarchy}}"
            selected-node="{{_selectedNode}}"
            stats="[[_stats]]"
            trace-inputs="[[_traceInputs]]"
          ></tf-graph-board>
        </div>
      </tf-dashboard-layout>
    </template>
    <style>
      :host /deep/ {
        font-family: 'Roboto', sans-serif;
      }

      .sidebar {
        display: flex;
        height: 100%;
      }

      .center {
        position: relative;
        height: 100%;
      }

      paper-dialog {
        padding: 20px;
      }
    </style>
  `;
    __decorate([
        property({
            type: Array,
        }),
        __metadata("design:type", Array)
    ], TfGraphDashboard.prototype, "_datasets", void 0);
    __decorate([
        property({
            type: Boolean,
        }),
        __metadata("design:type", Boolean)
    ], TfGraphDashboard.prototype, "_datasetsFetched", void 0);
    __decorate([
        property({
            type: Number,
        }),
        __metadata("design:type", Number)
    ], TfGraphDashboard.prototype, "_selectedDataset", void 0);
    __decorate([
        property({ type: Object, observer: '_renderHierarchyChanged' }),
        __metadata("design:type", Object)
    ], TfGraphDashboard.prototype, "_renderHierarchy", void 0);
    __decorate([
        property({
            type: Object,
        }),
        __metadata("design:type", RequestManager)
    ], TfGraphDashboard.prototype, "_requestManager", void 0);
    __decorate([
        property({
            type: Object,
        }),
        __metadata("design:type", Canceller)
    ], TfGraphDashboard.prototype, "_canceller", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfGraphDashboard.prototype, "_debuggerDataEnabled", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfGraphDashboard.prototype, "allStepsModeEnabled", void 0);
    __decorate([
        property({ type: Number }),
        __metadata("design:type", Number)
    ], TfGraphDashboard.prototype, "specificHealthPillStep", void 0);
    __decorate([
        property({
            type: Boolean,
            observer: '_healthPillsToggledOnChanged',
        }),
        __metadata("design:type", Boolean)
    ], TfGraphDashboard.prototype, "healthPillsToggledOn", void 0);
    __decorate([
        property({
            type: String,
            notify: true,
        }),
        __metadata("design:type", String)
    ], TfGraphDashboard.prototype, "selectedNode", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfGraphDashboard.prototype, "_isAttached", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfGraphDashboard.prototype, "_initialized", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfGraphDashboard.prototype, "_areHealthPillsLoading", void 0);
    __decorate([
        property({
            type: Array,
            notify: true,
        }),
        __metadata("design:type", Array)
    ], TfGraphDashboard.prototype, "_debuggerNumericAlerts", void 0);
    __decorate([
        property({
            type: Object,
        }),
        __metadata("design:type", Object)
    ], TfGraphDashboard.prototype, "_nodeNamesToHealthPills", void 0);
    __decorate([
        property({ type: Number }),
        __metadata("design:type", Number)
    ], TfGraphDashboard.prototype, "_healthPillStepIndex", void 0);
    __decorate([
        property({ type: Number }),
        __metadata("design:type", Number)
    ], TfGraphDashboard.prototype, "_healthPillRequestId", void 0);
    __decorate([
        property({ type: Number }),
        __metadata("design:type", Number)
    ], TfGraphDashboard.prototype, "_healthPillStepRequestTimerId", void 0);
    __decorate([
        property({
            type: Number,
        }),
        __metadata("design:type", Number)
    ], TfGraphDashboard.prototype, "_healthPillStepRequestTimerDelay", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfGraphDashboard.prototype, "runs", void 0);
    __decorate([
        property({
            type: String,
            notify: true,
            observer: '_runObserver',
        }),
        __metadata("design:type", String)
    ], TfGraphDashboard.prototype, "run", void 0);
    __decorate([
        property({
            type: Object,
        }),
        __metadata("design:type", Object)
    ], TfGraphDashboard.prototype, "_selection", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfGraphDashboard.prototype, "_compatibilityProvider", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfGraphDashboard.prototype, "_traceInputs", void 0);
    __decorate([
        observe('_isAttached'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfGraphDashboard.prototype, "_maybeInitializeDashboard", null);
    __decorate([
        observe('_datasetsFetched', '_datasets', 'run'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfGraphDashboard.prototype, "_determineSelectedDataset", null);
    __decorate([
        observe('_datasetsFetched', '_datasets', '_selectedDataset'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfGraphDashboard.prototype, "_updateSelectedDatasetName", null);
    TfGraphDashboard = __decorate([
        customElement('tf-graph-dashboard')
    ], TfGraphDashboard);

    /* Copyright 2016 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    // TypeScript can't deal with d3's style of overloading and
    // polymorphism, and constantly fails to select the correct overload.
    // This module was converted from working non-TypeScript code, so we
    // grandfather it in untyped.
    const d3$1 = d3Typed;
    let _VzHistogramTimeseries = class _VzHistogramTimeseries extends LegacyElementMixin(PolymerElement) {
        constructor() {
            super(...arguments);
            this.mode = 'offset';
            this.timeProperty = 'step';
            this.bins = 'bins';
            this.x = 'x';
            this.dx = 'dx';
            this.y = 'y';
            this.colorScale = d3$1.scaleOrdinal(d3$1.schemeCategory10);
            this.modeTransitionDuration = 500;
            this._name = null;
            this._data = null;
        }
        ready() {
            super.ready();
            // Polymer's way of scoping styles on nodes that d3 created
            this.scopeSubtree(this.$.svg, true);
        }
        attached() {
            this._attached = true;
        }
        detached() {
            this._attached = false;
        }
        setSeriesData(name, data) {
            this._name = name;
            this._data = data;
            this.redraw();
        }
        _redrawOnChange() {
            this.redraw();
        }
        /**
         * Redraws the chart. This is only called if the chart is attached to the
         * screen and if the chart has data.
         */
        redraw() {
            this._draw(0);
        }
        _modeRedraw() {
            this._draw(this.modeTransitionDuration);
        }
        _draw(duration) {
            if (!this._attached || !this._data) {
                return;
            }
            //
            // Data verification
            //
            if (duration === undefined)
                throw new Error('vz-histogram-timeseries _draw needs duration');
            if (this._data.length <= 0)
                throw new Error('Not enough steps in the data');
            if (!this._data[0].hasOwnProperty(this.bins))
                throw new Error("No bins property of '" + this.bins + "' in data");
            if (this._data[0][this.bins].length <= 0)
                throw new Error('Must have at least one bin in bins in data');
            if (!this._data[0][this.bins][0].hasOwnProperty(this.x))
                throw new Error("No x property '" + this.x + "' on bins data");
            if (!this._data[0][this.bins][0].hasOwnProperty(this.dx))
                throw new Error("No dx property '" + this.dx + "' on bins data");
            if (!this._data[0][this.bins][0].hasOwnProperty(this.y))
                throw new Error("No y property '" + this.y + "' on bins data");
            //
            // Initialization
            //
            var timeProp = this.timeProperty;
            var xProp = this.x;
            var binsProp = this.bins;
            var dxProp = this.dx;
            var yProp = this.y;
            var data = this._data;
            var name = this._name;
            var mode = this.mode;
            var color = d3$1.hcl(this.colorScale(name));
            var tooltip = d3$1.select(this.$.tooltip);
            var xAccessor = function (d) {
                return d[xProp];
            };
            var yAccessor = function (d) {
                return d[yProp];
            };
            var xRightAccessor = function (d) {
                return d[xProp] + d[dxProp];
            };
            var timeAccessor = function (d) {
                return d[timeProp];
            };
            if (timeProp === 'relative') {
                timeAccessor = function (d) {
                    return d.wall_time - data[0].wall_time;
                };
            }
            var brect = this.$.svg.getBoundingClientRect();
            var outerWidth = brect.width, outerHeight = brect.height;
            var sliceHeight, margin = { top: 5, right: 60, bottom: 20, left: 24 };
            if (mode === 'offset') {
                sliceHeight = outerHeight / 2.5;
                margin.top = sliceHeight + 5;
            }
            else {
                sliceHeight = outerHeight - margin.top - margin.bottom;
            }
            var width = outerWidth - margin.left - margin.right, height = outerHeight - margin.top - margin.bottom;
            var leftMin = d3$1.min(data, xAccessor), rightMax = d3$1.max(data, xRightAccessor);
            //
            // Text formatters
            //
            var format = d3$1.format('.3n');
            var yAxisFormat = d3$1.format('.0f');
            if (timeProp === 'wall_time') {
                yAxisFormat = d3$1.timeFormat('%m/%d %X');
            }
            else if (timeProp === 'relative') {
                yAxisFormat = function (d) {
                    return d3$1.format('.1r')(d / 3.6e6) + 'h'; // Convert to hours.
                };
            }
            //
            // Calculate the extents
            //
            var xExtents = data.map(function (d, i) {
                return [
                    d3$1.min(d[binsProp], xAccessor),
                    d3$1.max(d[binsProp], xRightAccessor),
                ];
            });
            var yExtents = data.map(function (d) {
                return d3$1.extent(d[binsProp], yAccessor);
            });
            //
            // Scales and axis
            //
            var outlineCanvasSize = 500;
            var extent = d3$1.extent(data, timeAccessor);
            var yScale = (timeProp === 'wall_time' ? d3$1.scaleTime() : d3$1.scaleLinear())
                .domain(extent)
                .range([0, mode === 'offset' ? height : 0]);
            var ySliceScale = d3$1
                .scaleLinear()
                .domain([
                0,
                d3$1.max(data, function (d, i) {
                    return yExtents[i][1];
                }),
            ])
                .range([sliceHeight, 0]);
            var yLineScale = d3$1
                .scaleLinear()
                .domain(ySliceScale.domain())
                .range([outlineCanvasSize, 0]);
            var xScale = d3$1
                .scaleLinear()
                .domain([
                d3$1.min(data, function (d, i) {
                    return xExtents[i][0];
                }),
                d3$1.max(data, function (d, i) {
                    return xExtents[i][1];
                }),
            ])
                .nice()
                .range([0, width]);
            var xLineScale = d3$1
                .scaleLinear()
                .domain(xScale.domain())
                .range([0, outlineCanvasSize]);
            var outlineColor = d3$1
                .scaleLinear()
                .domain(d3$1.extent(data, timeAccessor))
                .range([color.darker(), color.brighter()])
                .interpolate(d3$1.interpolateHcl);
            var xAxis = d3$1.axisBottom(xScale).ticks(Math.max(2, width / 20));
            var yAxis = d3$1
                .axisRight(yScale)
                .ticks(Math.max(2, height / 15))
                .tickFormat(yAxisFormat);
            var ySliceAxis = d3$1
                .axisRight(ySliceScale)
                .ticks(Math.max(2, height / 15))
                .tickSize(width + 5)
                .tickFormat(format);
            var xBinCentroid = function (d) {
                return d[xProp] + d[dxProp] / 2;
            };
            var linePath = d3$1
                .line()
                .x(function (d) {
                return xLineScale(xBinCentroid(d));
            })
                .y(function (d) {
                return yLineScale(d[yProp]);
            });
            var path = function (d) {
                // Draw a line from 0 to the first point and from the last point to 0.
                return ('M' +
                    xLineScale(xBinCentroid(d[0])) +
                    ',' +
                    yLineScale(0) +
                    'L' +
                    linePath(d).slice(1) +
                    'L' +
                    xLineScale(xBinCentroid(d[d.length - 1])) +
                    ',' +
                    yLineScale(0));
            };
            //
            // Render
            //
            var svgNode = this.$.svg;
            var svg = d3$1.select(svgNode);
            var svgTransition = svg.transition().duration(duration);
            var g = svg
                .select('g')
                .classed('small', function () {
                return width > 0 && width <= 150;
            })
                .classed('medium', function () {
                return width > 150 && width <= 300;
            })
                .classed('large', function () {
                return width > 300;
            });
            var gTransition = svgTransition
                .select('g')
                .attr('transform', 'translate(' + margin.left + ',' + margin.top + ')');
            var bisect = d3$1.bisector(xRightAccessor).left;
            var stage = g
                .select('.stage')
                .on('mouseover', function () {
                hoverUpdate.style('opacity', 1);
                xAxisHoverUpdate.style('opacity', 1);
                yAxisHoverUpdate.style('opacity', 1);
                ySliceAxisHoverUpdate.style('opacity', 1);
                tooltip.style('opacity', 1);
            })
                .on('mouseout', function () {
                hoverUpdate.style('opacity', 0);
                xAxisHoverUpdate.style('opacity', 0);
                yAxisHoverUpdate.style('opacity', 0);
                ySliceAxisHoverUpdate.style('opacity', 0);
                hoverUpdate.classed('hover-closest', false);
                outlineUpdate.classed('outline-hover', false);
                tooltip.style('opacity', 0);
            })
                .on('mousemove', onMouseMove);
            var background = stage
                .select('.background')
                .attr('transform', 'translate(' + -margin.left + ',' + -margin.top + ')')
                .attr('width', outerWidth)
                .attr('height', outerHeight);
            var histogram = stage.selectAll('.histogram').data(data), histogramExit = histogram.exit().remove(), histogramEnter = histogram.enter().append('g').attr('class', 'histogram'), histogramUpdate = histogramEnter.merge(histogram).sort(function (a, b) {
                return timeAccessor(a) - timeAccessor(b);
            }), histogramTransition = gTransition
                .selectAll('.histogram')
                .attr('transform', function (d) {
                return ('translate(0, ' +
                    (mode === 'offset' ? yScale(timeAccessor(d)) - sliceHeight : 0) +
                    ')');
            });
            var baselineEnter = histogramEnter.append('line').attr('class', 'baseline'), baselineUpdate = histogramTransition
                .select('.baseline')
                .style('stroke-opacity', function (d) {
                return mode === 'offset' ? 0.1 : 0;
            })
                .attr('y1', sliceHeight)
                .attr('y2', sliceHeight)
                .attr('x2', width);
            var outlineEnter = histogramEnter.append('path').attr('class', 'outline'), outlineUpdate = histogramUpdate
                .select('.outline')
                .attr('vector-effect', 'non-scaling-stroke')
                .attr('d', function (d) {
                return path(d[binsProp]);
            })
                .style('stroke-width', 1), outlineTransition = histogramTransition
                .select('.outline')
                .attr('transform', 'scale(' +
                width / outlineCanvasSize +
                ', ' +
                sliceHeight / outlineCanvasSize +
                ')')
                .style('stroke', function (d) {
                return mode === 'offset' ? 'white' : outlineColor(timeAccessor(d));
            })
                .style('fill-opacity', function (d) {
                return mode === 'offset' ? 1 : 0;
            })
                .style('fill', function (d) {
                return outlineColor(timeAccessor(d));
            });
            var hoverEnter = histogramEnter.append('g').attr('class', 'hover');
            var hoverUpdate = histogramUpdate
                .select('.hover')
                .style('fill', function (d) {
                return outlineColor(timeAccessor(d));
            });
            hoverEnter.append('circle').attr('r', 2);
            hoverEnter.append('text').style('display', 'none').attr('dx', 4);
            var xAxisHover = g.select('.x-axis-hover').selectAll('.label').data(['x']), xAxisHoverEnter = xAxisHover.enter().append('g').attr('class', 'label'), xAxisHoverUpdate = xAxisHover.merge(xAxisHoverEnter);
            xAxisHoverEnter
                .append('rect')
                .attr('x', -20)
                .attr('y', 6)
                .attr('width', 40)
                .attr('height', 14);
            xAxisHoverEnter
                .append('line')
                .attr('x1', 0)
                .attr('x2', 0)
                .attr('y1', 0)
                .attr('y2', 6);
            xAxisHoverEnter.append('text').attr('dy', 18);
            var yAxisHover = g.select('.y-axis-hover').selectAll('.label').data(['y']), yAxisHoverEnter = yAxisHover.enter().append('g').attr('class', 'label'), yAxisHoverUpdate = yAxisHover.merge(yAxisHoverEnter);
            yAxisHoverEnter
                .append('rect')
                .attr('x', 8)
                .attr('y', -6)
                .attr('width', 40)
                .attr('height', 14);
            yAxisHoverEnter
                .append('line')
                .attr('x1', 0)
                .attr('x2', 6)
                .attr('y1', 0)
                .attr('y2', 0);
            yAxisHoverEnter.append('text').attr('dx', 8).attr('dy', 4);
            var ySliceAxisHover = g
                .select('.y-slice-axis-hover')
                .selectAll('.label')
                .data(['y']), ySliceAxisHoverEnter = ySliceAxisHover
                .enter()
                .append('g')
                .attr('class', 'label'), ySliceAxisHoverUpdate = ySliceAxisHover.merge(ySliceAxisHoverEnter);
            ySliceAxisHoverEnter
                .append('rect')
                .attr('x', 8)
                .attr('y', -6)
                .attr('width', 40)
                .attr('height', 14);
            ySliceAxisHoverEnter
                .append('line')
                .attr('x1', 0)
                .attr('x2', 6)
                .attr('y1', 0)
                .attr('y2', 0);
            ySliceAxisHoverEnter.append('text').attr('dx', 8).attr('dy', 4);
            gTransition
                .select('.y.axis.slice')
                .style('opacity', mode === 'offset' ? 0 : 1)
                .attr('transform', 'translate(0, ' + (mode === 'offset' ? -sliceHeight : 0) + ')')
                .call(ySliceAxis);
            gTransition
                .select('.x.axis')
                .attr('transform', 'translate(0, ' + height + ')')
                .call(xAxis);
            gTransition
                .select('.y.axis')
                .style('opacity', mode === 'offset' ? 1 : 0)
                .attr('transform', 'translate(' + width + ', ' + (mode === 'offset' ? 0 : height) + ')')
                .call(yAxis);
            gTransition.selectAll('.tick text').attr('fill', '#aaa');
            gTransition.selectAll('.axis path.domain').attr('stroke', 'none');
            function onMouseMove() {
                var m = d3$1.mouse(this), v = xScale.invert(m[0]), t = yScale.invert(m[1]);
                function hoverXIndex(d) {
                    return Math.min(d[binsProp].length - 1, bisect(d[binsProp], v));
                }
                var closestSliceData;
                var closestSliceDistance = Infinity;
                var lastSliceData;
                hoverUpdate.attr('transform', function (d, i) {
                    var index = hoverXIndex(d);
                    lastSliceData = d;
                    var x = xScale(d[binsProp][index][xProp] + d[binsProp][index][dxProp] / 2);
                    var y = ySliceScale(d[binsProp][index][yProp]);
                    var globalY = mode === 'offset' ? yScale(timeAccessor(d)) - (sliceHeight - y) : y;
                    var dist = Math.abs(m[1] - globalY);
                    if (dist < closestSliceDistance) {
                        closestSliceDistance = dist;
                        closestSliceData = d;
                    }
                    return 'translate(' + x + ',' + y + ')';
                });
                hoverUpdate.select('text').text(function (d) {
                    var index = hoverXIndex(d);
                    return d[binsProp][index][yProp];
                });
                hoverUpdate.classed('hover-closest', function (d) {
                    return d === closestSliceData;
                });
                outlineUpdate.classed('outline-hover', function (d) {
                    return d === closestSliceData;
                });
                var index = hoverXIndex(lastSliceData);
                xAxisHoverUpdate
                    .attr('transform', function (d) {
                    return ('translate(' +
                        xScale(lastSliceData[binsProp][index][xProp] +
                            lastSliceData[binsProp][index][dxProp] / 2) +
                        ', ' +
                        height +
                        ')');
                })
                    .select('text')
                    .text(function (d) {
                    return format(lastSliceData[binsProp][index][xProp] +
                        lastSliceData[binsProp][index][dxProp] / 2);
                });
                var fy = yAxis.tickFormat();
                yAxisHoverUpdate
                    .attr('transform', function (d) {
                    return ('translate(' +
                        width +
                        ', ' +
                        (mode === 'offset' ? yScale(timeAccessor(closestSliceData)) : 0) +
                        ')');
                })
                    .style('display', mode === 'offset' ? '' : 'none')
                    .select('text')
                    .text(function (d) {
                    return fy(timeAccessor(closestSliceData));
                });
                var fsy = ySliceAxis.tickFormat();
                ySliceAxisHoverUpdate
                    .attr('transform', function (d) {
                    return ('translate(' +
                        width +
                        ', ' +
                        (mode === 'offset'
                            ? 0
                            : ySliceScale(closestSliceData[binsProp][index][yProp])) +
                        ')');
                })
                    .style('display', mode === 'offset' ? 'none' : '')
                    .select('text')
                    .text(function (d) {
                    return fsy(closestSliceData[binsProp][index][yProp]);
                });
                var svgMouse = d3$1.mouse(svgNode);
                tooltip
                    .style('transform', 'translate(' + (svgMouse[0] + 15) + 'px,' + (svgMouse[1] - 15) + 'px)')
                    .select('span')
                    .text(mode === 'offset'
                    ? fsy(closestSliceData[binsProp][index][yProp])
                    : (timeProp === 'step' ? 'step ' : '') +
                        fy(timeAccessor(closestSliceData)));
            }
        }
    };
    _VzHistogramTimeseries.template = html `
    <div id="tooltip"><span></span></div>
    <svg id="svg">
      <g>
        <g class="axis x"></g>
        <g class="axis y"></g>
        <g class="axis y slice"></g>
        <g class="stage">
          <rect class="background"></rect>
        </g>
        <g class="x-axis-hover"></g>
        <g class="y-axis-hover"></g>
        <g class="y-slice-axis-hover"></g>
      </g>
    </svg>

    <style>
      :host {
        display: flex;
        flex-direction: column;
        flex-grow: 1;
        flex-shrink: 1;
        position: relative;
      }

      svg {
        font-family: roboto, sans-serif;
        overflow: visible;
        display: block;
        width: 100%;
        flex-grow: 1;
        flex-shrink: 1;
      }

      #tooltip {
        position: absolute;
        display: block;
        opacity: 0;
        font-weight: bold;
        font-size: 11px;
      }

      .background {
        fill-opacity: 0;
        fill: red;
      }

      .histogram {
        pointer-events: none;
      }

      .hover {
        font-size: 9px;
        dominant-baseline: middle;
        opacity: 0;
      }

      .hover circle {
        stroke: white;
        stroke-opacity: 0.5;
        stroke-width: 1px;
      }

      .hover text {
        fill: black;
        opacity: 0;
      }

      .hover.hover-closest circle {
        fill: black !important;
      }

      .hover.hover-closest text {
        opacity: 1;
      }

      .baseline {
        stroke: black;
        stroke-opacity: 0.1;
      }

      .outline {
        fill: none;
        stroke: white;
        stroke-opacity: 0.5;
      }

      .outline.outline-hover {
        stroke: black !important;
        stroke-opacity: 1;
      }

      .x-axis-hover,
      .y-axis-hover,
      .y-slice-axis-hover {
        pointer-events: none;
      }

      .x-axis-hover .label,
      .y-axis-hover .label,
      .y-slice-axis-hover .label {
        opacity: 0;
        font-weight: bold;
        font-size: 11px;
        text-anchor: end;
      }

      .x-axis-hover text {
        text-anchor: middle;
      }

      .y-axis-hover text,
      .y-slice-axis-hover text {
        text-anchor: start;
      }

      .x-axis-hover line,
      .y-axis-hover line,
      .y-slice-axis-hover line {
        stroke: black;
      }

      .x-axis-hover rect,
      .y-axis-hover rect,
      .y-slice-axis-hover rect {
        fill: white;
      }

      .axis {
        font-size: 11px;
      }

      .axis path.domain {
        fill: none;
      }

      .axis .tick line {
        stroke: #ddd;
      }

      .axis.slice {
        opacity: 0;
      }

      .axis.slice .tick line {
        stroke-dasharray: 2;
      }

      .small .axis text {
        display: none;
      }
      .small .axis .tick:first-of-type text {
        display: block;
      }
      .small .axis .tick:last-of-type text {
        display: block;
      }
      .medium .axis text {
        display: none;
      }
      .medium .axis .tick:nth-child(2n + 1) text {
        display: block;
      }
      .large .axis text {
        display: none;
      }
      .large .axis .tick:nth-child(2n + 1) text {
        display: block;
      }
    </style>
  `;
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], _VzHistogramTimeseries.prototype, "mode", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], _VzHistogramTimeseries.prototype, "timeProperty", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], _VzHistogramTimeseries.prototype, "bins", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], _VzHistogramTimeseries.prototype, "x", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], _VzHistogramTimeseries.prototype, "dx", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], _VzHistogramTimeseries.prototype, "y", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], _VzHistogramTimeseries.prototype, "colorScale", void 0);
    __decorate([
        property({ type: Number }),
        __metadata("design:type", Number)
    ], _VzHistogramTimeseries.prototype, "modeTransitionDuration", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], _VzHistogramTimeseries.prototype, "_attached", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], _VzHistogramTimeseries.prototype, "_name", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], _VzHistogramTimeseries.prototype, "_data", void 0);
    __decorate([
        observe('timeProperty', 'colorScale', '_attached'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], _VzHistogramTimeseries.prototype, "_redrawOnChange", null);
    __decorate([
        observe('mode'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], _VzHistogramTimeseries.prototype, "_modeRedraw", null);
    _VzHistogramTimeseries = __decorate([
        customElement('vz-histogram-timeseries')
    ], _VzHistogramTimeseries);

    /* Copyright 2015 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    function backendToIntermediate(histogram) {
        const [wall_time, step, buckets] = histogram;
        return {
            wall_time,
            step,
            min: min(buckets.map(([left, ,]) => left)),
            max: max(buckets.map(([, right]) => right)),
            buckets: buckets.map(([left, right, count]) => ({ left, right, count })),
        };
    }
    /**
     * Convert histogram data to the standard D3 format to make it more
     * compatible and easier to visualize. When rendering histograms, having
     * access to the left edge and width of each bin makes things quite a
     * bit easier, so we include these in the result. We also convert the
     * bins to have a uniform width, which makes the visualization easier to
     * understand.
     *
     * @param histogram
     * @param min The leftmost edge. The binning will start on it.
     * @param max The rightmost edge. The binning will end on it.
     * @param numBins The number of bins of the converted data. The default
     * of 30 is sensible: if you use more, you start to get artifacts
     * because the event data is stored in buckets, and you start being able
     * to see the aliased borders between each bucket.
     *
     * @return A list of histogram bins. Each bin has an `x` (left
     *     edge), a `dx` (width), and a `y` (count). If the given
     *     right edges are inclusive, then these left edges (`x`) are
     *     exclusive.
     */
    function intermediateToD3(histogram, min, max, numBins = 30) {
        if (max === min) {
            // Create bins even if all the data has a single value.
            max = min * 1.1 + 1;
            min = min / 1.1 - 1;
        }
        // Terminology note: _buckets_ are the input to this function,
        // while _bins_ are our output.
        const binWidth = (max - min) / numBins;
        let bucketIndex = 0;
        return sequence(min, max, binWidth).map((binLeft) => {
            const binRight = binLeft + binWidth;
            // Take the count of each existing bucket, multiply it by the
            // proportion of overlap with the new bin, then sum and store as the
            // count for the new bin. If no overlap, will add to zero; if 100%
            // overlap, will include the full count into new bin.
            let binY = 0;
            while (bucketIndex < histogram.buckets.length) {
                // Clip the right edge because right-most edge can be
                // infinite-sized.
                const bucketRight = Math.min(max, histogram.buckets[bucketIndex].right);
                const bucketLeft = Math.max(min, histogram.buckets[bucketIndex].left);
                const intersect = Math.min(bucketRight, binRight) - Math.max(bucketLeft, binLeft);
                const count = (intersect / (bucketRight - bucketLeft)) *
                    histogram.buckets[bucketIndex].count;
                binY += intersect > 0 ? count : 0;
                // If `bucketRight` is bigger than `binRight`, then this bin is
                // finished and there is data for the next bin, so don't increment
                // `bucketIndex`.
                if (bucketRight > binRight) {
                    break;
                }
                bucketIndex++;
            }
            return { x: binLeft, dx: binWidth, y: binY };
        });
    }
    function backendToVz(histograms) {
        const intermediateHistograms = histograms.map(backendToIntermediate);
        const minmin = min(intermediateHistograms, (h) => h.min);
        const maxmax = max(intermediateHistograms, (h) => h.max);
        return intermediateHistograms.map((h) => ({
            wall_time: h.wall_time,
            step: h.step,
            bins: intermediateToD3(h, minmin, maxmax),
        }));
    }

    /* Copyright 2016 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    let _TfHistogramLoader = class _TfHistogramLoader extends DataLoaderBehavior(LegacyElementMixin(PolymerElement)) {
        constructor() {
            super(...arguments);
            this.getDataLoadName = ({ run }) => run;
            this.requestData = (items, onLoad, onFinish) => {
                const router = getRouter();
                const baseUrl = router.pluginRoute('histograms', '/histograms');
                Promise.all(items.map((item) => {
                    const url = addParams(baseUrl, { tag: item.tag, run: item.run });
                    return this.requestManager
                        .request(url)
                        .then((data) => void onLoad({ item, data }));
                })).finally(() => void onFinish());
            };
            this.loadDataCallback = (_, datum, data) => {
                const d3Data = backendToVz(data);
                const name = this.getDataLoadName(datum);
                this.$.chart.setSeriesData(name, d3Data);
            };
            this._colorScaleFunction = runsColorScale;
            this._expanded = false;
        }
        _reloadOnRunTagRequestManagerChange() {
            this.reload();
        }
        _updateDataToLoad() {
            var run = this.run;
            var tag = this.tag;
            this.dataToLoad = [{ run, tag }];
        }
        get _runColor() {
            var run = this.run;
            return this._colorScaleFunction(run);
        }
        redraw() {
            this.$.chart.redraw();
        }
        _toggleExpanded(e) {
            this.set('_expanded', !this._expanded);
            this.redraw();
        }
    };
    _TfHistogramLoader.template = html `
    <tf-card-heading
      tag="[[tag]]"
      run="[[run]]"
      display-name="[[tagMetadata.displayName]]"
      description="[[tagMetadata.description]]"
      color="[[_runColor]]"
    ></tf-card-heading>
    <!--
      The main histogram that we render. Data is set directly with
      \`setSeriesData\`, not with a bound property.
    -->
    <vz-histogram-timeseries
      id="chart"
      time-property="[[timeProperty]]"
      mode="[[histogramMode]]"
      color-scale="[[_colorScaleFunction]]"
    ></vz-histogram-timeseries>
    <div style="display: flex; flex-direction: row;">
      <paper-icon-button
        selected$="[[_expanded]]"
        icon="fullscreen"
        on-tap="_toggleExpanded"
      ></paper-icon-button>
    </div>
    <style>
      :host {
        display: flex;
        flex-direction: column;
        width: 330px;
        height: 235px;
        margin-right: 10px;
        margin-bottom: 15px;
      }
      :host([_expanded]) {
        width: 700px;
        height: 500px;
      }

      vz-histogram-timeseries {
        -moz-user-select: none;
        -webkit-user-select: none;
        will-change: transform;
      }

      paper-icon-button {
        color: #2196f3;
        border-radius: 100%;
        width: 32px;
        height: 32px;
        padding: 4px;
      }

      paper-icon-button[selected] {
        background: var(--tb-ui-light-accent);
      }

      tf-card-heading {
        margin-bottom: 10px;
        width: 90%;
      }
    </style>
  `;
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], _TfHistogramLoader.prototype, "run", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], _TfHistogramLoader.prototype, "tag", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], _TfHistogramLoader.prototype, "getDataLoadName", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", RequestManager)
    ], _TfHistogramLoader.prototype, "requestManager", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], _TfHistogramLoader.prototype, "loadDataCallback", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], _TfHistogramLoader.prototype, "tagMetadata", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], _TfHistogramLoader.prototype, "timeProperty", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], _TfHistogramLoader.prototype, "histogramMode", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Function)
    ], _TfHistogramLoader.prototype, "_colorScaleFunction", void 0);
    __decorate([
        property({ type: Boolean, reflectToAttribute: true }),
        __metadata("design:type", Boolean)
    ], _TfHistogramLoader.prototype, "_expanded", void 0);
    __decorate([
        observe('run', 'tag', 'requestManager'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], _TfHistogramLoader.prototype, "_reloadOnRunTagRequestManagerChange", null);
    __decorate([
        observe('run', 'tag'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], _TfHistogramLoader.prototype, "_updateDataToLoad", null);
    __decorate([
        computed('run'),
        __metadata("design:type", String),
        __metadata("design:paramtypes", [])
    ], _TfHistogramLoader.prototype, "_runColor", null);
    _TfHistogramLoader = __decorate([
        customElement('tf-histogram-loader')
    ], _TfHistogramLoader);

    /* Copyright 2016 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    let TfHistogramDashboard = class TfHistogramDashboard extends LegacyElementMixin(PolymerElement) {
        constructor() {
            super(...arguments);
            this.reloadOnReady = true;
            this._histogramMode = 'offset';
            this._timeProperty = 'step';
            this._restamp = false;
            this._requestManager = new RequestManager();
        }
        _redrawCategoryPane(event, val) {
            if (!val)
                return;
            event.target
                .querySelectorAll('tf-histogram-loader')
                .forEach((histogram) => histogram.redraw());
        }
        ready() {
            super.ready();
            if (this.reloadOnReady)
                this.reload();
        }
        reload() {
            this._fetchTags().then(() => {
                this._reloadHistograms();
            });
        }
        _fetchTags() {
            const url = getRouter().pluginRoute('histograms', '/tags');
            return this._requestManager.request(url).then((runToTagInfo) => {
                if (lodash.isEqual(runToTagInfo, this._runToTagInfo)) {
                    // No need to update anything if there are no changes.
                    return;
                }
                const runToTag = lodash.mapValues(runToTagInfo, (x) => Object.keys(x));
                const tags = getTags(runToTag);
                this.set('_dataNotFound', tags.length === 0);
                this.set('_runToTag', runToTag);
                this.set('_runToTagInfo', runToTagInfo);
                this.async(() => {
                    // See the comment above `_categoriesDomReady`.
                    this.set('_categoriesDomReady', true);
                });
            });
        }
        _reloadHistograms() {
            this.root.querySelectorAll('tf-histogram-loader').forEach((histogram) => {
                histogram.reload();
            });
        }
        _shouldOpen(index) {
            return index <= 2;
        }
        get _categories() {
            var runToTag = this._runToTag;
            var selectedRuns = this._selectedRuns;
            var tagFilter = this._tagFilter;
            var categoriesDomReady = this._categoriesDomReady;
            return categorizeRunTagCombinations(runToTag, selectedRuns, tagFilter);
        }
        _tagMetadata(runToTagInfo, run, tag) {
            return runToTagInfo[run][tag];
        }
    };
    TfHistogramDashboard.template = html `
    <tf-dashboard-layout>
      <div slot="sidebar">
        <div class="settings">
          <div class="sidebar-section">
            <tf-option-selector
              id="histogramModeSelector"
              name="Histogram mode"
              selected-id="{{_histogramMode}}"
            >
              <paper-button id="overlay">overlay</paper-button>
              <paper-button id="offset">offset</paper-button>
            </tf-option-selector>
          </div>
          <div class="sidebar-section">
            <tf-option-selector
              id="timePropertySelector"
              name="Offset time axis"
              selected-id="{{_timeProperty}}"
            >
              <paper-button id="step">step</paper-button>
              <paper-button id="relative">relative</paper-button>
              <paper-button id="wall_time">wall</paper-button>
            </tf-option-selector>
          </div>
        </div>
        <div class="sidebar-section runs-selector">
          <tf-runs-selector selected-runs="{{_selectedRuns}}">
          </tf-runs-selector>
        </div>
      </div>
      <div slot="center">
        <template is="dom-if" if="[[_dataNotFound]]">
          <div class="no-data-warning">
            <h3>No histogram data was found.</h3>
            <p>Probable causes:</p>
            <ul>
              <li>
                You haven’t written any histogram data to your event files.
              </li>
              <li>TensorBoard can’t find your event files.</li>
            </ul>

            <p>
              If you’re new to using TensorBoard, and want to find out how to
              add data and set up your event files, check out the
              <a
                href="https://github.com/tensorflow/tensorboard/blob/master/README.md"
                >README</a
              >
              and perhaps the
              <a
                href="https://www.tensorflow.org/get_started/summaries_and_tensorboard"
                >TensorBoard tutorial</a
              >.
            </p>

            <p>
              If you think TensorBoard is configured properly, please see
              <a
                href="https://github.com/tensorflow/tensorboard/blob/master/README.md#my-tensorboard-isnt-showing-any-data-whats-wrong"
                >the section of the README devoted to missing data problems</a
              >
              and consider filing an issue on GitHub.
            </p>
          </div>
        </template>
        <template is="dom-if" if="[[!_dataNotFound]]">
          <tf-tag-filterer tag-filter="{{_tagFilter}}"></tf-tag-filterer>
          <template is="dom-repeat" items="[[_categories]]" as="category">
            <tf-category-paginated-view
              category="[[category]]"
              initial-opened="[[_shouldOpen(index)]]"
            >
              <template>
                <tf-histogram-loader
                  run="[[item.run]]"
                  tag="[[item.tag]]"
                  active="[[active]]"
                  tag-metadata="[[_tagMetadata(_runToTagInfo, item.run, item.tag)]]"
                  time-property="[[_timeProperty]]"
                  histogram-mode="[[_histogramMode]]"
                  request-manager="[[_requestManager]]"
                ></tf-histogram-loader>
              </template>
            </tf-category-paginated-view>
          </template>
        </template>
      </div>
    </tf-dashboard-layout>

    <style include="dashboard-style"></style>
    <style>
      .no-data-warning {
        max-width: 540px;
        margin: 80px auto 0 auto;
      }
    </style>
  `;
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfHistogramDashboard.prototype, "reloadOnReady", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfHistogramDashboard.prototype, "_histogramMode", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfHistogramDashboard.prototype, "_timeProperty", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfHistogramDashboard.prototype, "_selectedRuns", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfHistogramDashboard.prototype, "_runToTag", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfHistogramDashboard.prototype, "_runToTagInfo", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfHistogramDashboard.prototype, "_dataNotFound", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfHistogramDashboard.prototype, "_tagFilter", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfHistogramDashboard.prototype, "_restamp", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfHistogramDashboard.prototype, "_categoriesDomReady", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", RequestManager)
    ], TfHistogramDashboard.prototype, "_requestManager", void 0);
    __decorate([
        computed('_runToTag', '_selectedRuns', '_tagFilter', '_categoriesDomReady'),
        __metadata("design:type", Array),
        __metadata("design:paramtypes", [])
    ], TfHistogramDashboard.prototype, "_categories", null);
    TfHistogramDashboard = __decorate([
        customElement('tf-histogram-dashboard')
    ], TfHistogramDashboard);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    // -----------------------------------------------------------------------
    // Functions for dealing with HParams, Metrics, Columns,
    // SessionGroups and Schema objects.
    // In the following routines we use the following naming conventions for
    // parameters:
    //
    // hparamInfo, metricInfo: A JavaScript object representation of the
    //     HParamInfo proto defined in api.proto.
    // schema: The object storing metadata on the metrics and hparams used
    //     in the experiment. See the documentation in
    //     'tf-hparams-query-pane.html'.
    // visibleSchema: The object storing metadata on just the visible metrics
    //     and hparams in 'schema'. This is the object stored in the property
    //     'visibleSchema' of 'configuration' (see the documentation in
    //     tf-hparams-query-pane.html' for more details). This object and the
    //     utility functions below that accept it are deprecated and the
    //     intention is to eventually remove them.
    //     New code should instead filter the entries in the schema object
    //     directly to find all the visible columns and only use the functions
    //     below that take a 'schema' object.
    // sessionGroup: A JavaScript object representation of the SessionGroup
    //     proto defined in api.proto.
    // sessionGroups: An array of 'sessionGroup' objects.
    // columnIndex: An index of a 'column' of a sessionGroup with respect
    //     to a schema or visibleSchema object. We regard a
    //     sessionGroup as a row in a table where the columns represent
    //     hparams and metric values for the session group. The columns are
    //     ordered by listing the hparam columns first in the same order as
    //     the corresponding elements of schema.hparamColumns or
    //     visibleSchema.hparamInfos, followed by the
    //     metric columns in the same order as the corresponding elements of
    //     schema.metricColumns or visibleSchema.metricInfos. Whether the
    //     index is given with respect to a schema or a visibleScheme depends
    //     on the type of schema parameter the utility function below accepts
    //     in addition to the columnIndex parameter.
    // -----------------------------------------------------------------------
    // Computes the name to display for the given 'hparamInfo' object.
    function hparamName(hparamInfo) {
        if (hparamInfo.displayName !== '' && hparamInfo.displayName !== undefined) {
            return hparamInfo.displayName;
        }
        return hparamInfo.name;
    }
    // Computes the name to display for the given metricInfo object.
    function metricName(metricInfo) {
        if (metricInfo.displayName !== '' && metricInfo.displayName !== undefined) {
            return metricInfo.displayName;
        }
        let group = metricInfo.name.group;
        let tag = metricInfo.name.tag;
        if (group === undefined) {
            group = '';
        }
        if (tag === undefined) {
            tag = '';
        }
        if (group === '') {
            return tag;
        }
        return group + '.' + tag;
    }
    function schemaColumnName(schema, columnIndex) {
        if (columnIndex < schema.hparamColumns.length) {
            return hparamName(schema.hparamColumns[columnIndex].hparamInfo);
        }
        const metricIndex = columnIndex - schema.hparamColumns.length;
        return metricName(schema.metricColumns[metricIndex].metricInfo);
    }
    // Returns the number of hparams in schema (visible and invisible).
    function numHParams(schema) {
        return schema.hparamColumns.length;
    }
    // Returns the number of metrics in schema (visible and invisible).
    function numMetrics(schema) {
        return schema.metricColumns.length;
    }
    // Returns the number of columns in schema (visible and invisible).
    function numColumns(schema) {
        return numHParams(schema) + numMetrics(schema);
    }
    // Returns hparamValues[hparamName]. To be used in a Polymer databinding
    // annotation (as Polymer doesn't have an annotation for looking up a
    // property in an JS object).
    function hparamValueByName(hparamValues, hparamName) {
        return hparamValues[hparamName];
    }
    // Given an array 'metricValues' of (javascript object representation) of
    // tensorboard.hparams.MetricValue's protocol buffers, returns the first
    // element whose metric name is 'metricName' or undefined if no such
    // element exists.
    function metricValueByName(metricValues, metricName) {
        return metricValues.find((mv) => lodash.isEqual(mv.name, metricName));
    }
    // Returns sessionGroup's metric value of the metric with index
    // 'metricIndex' in schema.metricColumns.
    function hparamValueByIndex(schema, sessionGroup, hparamIndex) {
        return sessionGroup.hparams[schema.hparamColumns[hparamIndex].hparamInfo.name];
    }
    // Returns sessionGroup's metric value of the metric with index
    // 'metricIndex' in schema.metricColumns.
    function metricValueByIndex(schema, sessionGroup, metricIndex) {
        const metricName = schema.metricColumns[metricIndex].metricInfo.name;
        const metricValue = metricValueByName(sessionGroup.metricValues, metricName);
        return metricValue === undefined ? undefined : metricValue.value;
    }
    // Returns sessionGroup's column value of the column with index
    // 'columnIndex' in schema.
    function columnValueByIndex(schema, sessionGroup, columnIndex) {
        if (columnIndex < schema.hparamColumns.length) {
            return hparamValueByIndex(schema, sessionGroup, columnIndex);
        }
        return metricValueByIndex(schema, sessionGroup, columnIndex - schema.hparamColumns.length);
    }
    // Returns an array [min, max] representing the minimum and maximum
    // value of the given column in the sessionGroups array.
    // Ignores session groups with missing values for the column.
    function numericColumnExtent(schema, sessionGroups, columnIndex) {
        return extent(sessionGroups, (sg) => columnValueByIndex(schema, sg, columnIndex));
    }
    // Converts a visibleSchema columnIndex to a schema columnIndex.
    // Returns the schema-relative columnIndex for the visible column with
    // visibleSchema-releative columnIndex given by 'visibleColumnIndex'.
    function getAbsoluteColumnIndex(schema, visibleSchema, visibleColumnIndex) {
        let result;
        if (visibleColumnIndex < visibleSchema.hparamInfos.length) {
            result = schema.hparamColumns.findIndex((c) => c.hparamInfo.name === visibleSchema.hparamInfos[visibleColumnIndex].name);
        }
        else {
            const metricIndex = visibleColumnIndex - visibleSchema.hparamInfos.length;
            const metricName = visibleSchema.metricInfos[metricIndex].name;
            result =
                schema.hparamColumns.length +
                    schema.metricColumns.findIndex((c) => c.metricInfo.name === metricName);
        }
        console.assert(result !== -1);
        return result;
    }
    // DEPRECATED. Use schemaColumnName instead (with an "absolute"
    // columnIndex).
    // Computes the name to display for the given visible column index.
    function schemaVisibleColumnName(visibleSchema, columnIndex) {
        if (columnIndex < visibleSchema.hparamInfos.length) {
            return hparamName(visibleSchema.hparamInfos[columnIndex]);
        }
        const metricIndex = columnIndex - visibleSchema.hparamInfos.length;
        return metricName(visibleSchema.metricInfos[metricIndex]);
    }
    // DEPRECATED. Use numDisplayedHParams instead.
    // Returns the number of hparams in visibleSchema. This is the same
    // value as numDisplayedHParams(schema) with schema being the "containing"
    // schema of visibleSchema.
    function numVisibleHParams(visibleSchema) {
        return visibleSchema.hparamInfos.length;
    }
    // DEPRECATED. Use numDisplayedMetrics instead.
    // Returns the number of hparams in visibleSchema. This is the same
    // value as numDisplayedMetrics(schema) with schema being the "containing"
    // schema of visibleSchema.
    function numVisibleMetrics(visibleSchema) {
        return visibleSchema.metricInfos.length;
    }
    // DEPRECATED.
    // Returns the number of visible columns (having 'displayed' true).
    // This is the same value as numDisplayedColumns(schema) with schema
    // being the "containing" schema of visibleSchema.
    function numVisibleColumns(visibleSchema) {
        return numVisibleHParams(visibleSchema) + numVisibleMetrics(visibleSchema);
    }
    // DEPRECATED. Use numericColumnExtent with a schema columnIndex instead.
    // Returns an array [min, max] representing the minimum and maximum
    // value of the given visible column in the sessionGroups array.
    // Ignores session groups with missing values for the column.
    function visibleNumericColumnExtent(visibleSchema, sessionGroups, columnIndex) {
        return extent(sessionGroups, (sg) => columnValueByVisibleIndex(visibleSchema, sg, columnIndex));
    }
    // Returns a string representation of hparamValues[hparamName] suitable
    // for display.
    function prettyPrintHParamValueByName(hparamValues, hparamName) {
        return prettyPrint(hparamValueByName(hparamValues, hparamName));
    }
    // Returns a string representation of metricValueByName suitable for
    // display.
    function prettyPrintMetricValueByName(metricValues, metricName) {
        return prettyPrint(metricValueByName(metricValues, metricName));
    }
    // Returns the session group with name 'name' in sessionGroups or
    // undefined of no such element exist.
    function sessionGroupWithName(sessionGroups, name) {
        return sessionGroups.find((sg) => sg.name === name);
    }
    // DEPRECATED. Use hparamValueByIndex with a schema hparamIndex instead.
    // Returns sessionGroup's hparam value of the visible hparam with index
    // 'hparamIndex' in visibleSchema.hparamInfos.
    function hparamValueByVisibleIndex(visibleSchema, sessionGroup, hparamIndex) {
        return sessionGroup.hparams[visibleSchema.hparamInfos[hparamIndex].name];
    }
    // DEPRECATED. Use metricValueByIndex with a schema metricIndex instead.
    // Returns sessionGroup's metric value of the visible metric with index
    // 'metricIndex' in visibleSchema.metricInfos.
    function metricValueByVisibleIndex(visibleSchema, sessionGroup, visibleMetricIndex) {
        const metricName = visibleSchema.metricInfos[visibleMetricIndex].name;
        const metricValue = metricValueByName(sessionGroup.metricValues, metricName);
        return metricValue === undefined ? undefined : metricValue.value;
    }
    // DEPRECATED. Use columnValueByIndex with a schema columnIndex instead.
    // Returns sessionGroup's column value of the visible column with index
    // 'columnIndex' in visibleSchema.
    function columnValueByVisibleIndex(visibleSchema, sessionGroup, columnIndex) {
        if (columnIndex < visibleSchema.hparamInfos.length) {
            return hparamValueByVisibleIndex(visibleSchema, sessionGroup, columnIndex);
        }
        return metricValueByVisibleIndex(visibleSchema, sessionGroup, columnIndex - visibleSchema.hparamInfos.length);
    }
    // ---- Misc functions ---------------------------------------------------
    // Returns a string representation of 'value' suitable for display.
    function prettyPrint(value) {
        if (lodash.isNumber(value)) {
            // TODO(erez):Make the precision user-configurable.
            return value.toPrecision(5);
        }
        if (value === undefined) {
            return '';
        }
        return value.toString();
    }
    // Returns the square of the L2-norm of (x, y).
    function l2NormSquared(x, y) {
        return x * x + y * y;
    }
    // Returns the euclidean distance between (x0, y0) and (x1, y1).
    function euclideanDist(x0, y0, x1, y1) {
        return Math.sqrt(l2NormSquared(x0 - x1, y0 - y1));
    }
    // Returns the (euclidean) distance between the point (x, y) and the
    // rectangle [x0, x1) x [y0, y1).
    function pointToRectangleDist(x, y, x0, y0, x1, y1) {
        if (x < x0 && y < y0) {
            return euclideanDist(x, y, x0, y0);
        }
        else if (x0 <= x && x < x1 && y < y0) {
            return y0 - y;
        }
        else if (x1 <= x && y < y0) {
            return euclideanDist(x, y, x1, y0);
        }
        else if (x < x0 && y0 <= y && y < y1) {
            return x0 - x;
        }
        else if (x0 <= x && x < x1 && y0 <= y && y < y1) {
            return 0;
        }
        else if (x1 <= x && y0 <= y && y < y1) {
            return x - x1;
        }
        else if (x < x0 && y1 <= y) {
            return euclideanDist(x, y, x0, y1);
        }
        else if (x0 <= x && x < x1 && y1 <= y) {
            return y - y1;
        }
        else if (x1 <= x && y1 <= y) {
            return euclideanDist(x, y, x1, y1);
        }
        else {
            throw 'Point (x,y) must be in one of the regions defined above.';
        }
    }
    // SVG elements such as <g> can optionally have a "transform" attribute
    // the alters the way the element and its children are drawn.
    // The following function helps generate a "translate function" value
    // for this attribute.
    // See
    // https://developer.mozilla.org/en-US/docs/Web/SVG/Attribute/transform
    // for more details.
    function translateStr(x, opt_y) {
        if (opt_y === undefined) {
            return 'translate(' + x + ')';
        }
        return 'translate(' + x + ',' + opt_y + ')';
    }
    // SVG elements such as <g> can optionally have a "transform" attribute
    // the alters the way the element and its children are drawn.
    // The following function helps generate a "rotate function" value
    // for this attribute.
    // See
    // https://developer.mozilla.org/en-US/docs/Web/SVG/Attribute/transform
    // for more details.
    function rotateStr(angle, x, y) {
        let result = 'rotate(' + angle;
        if (x !== undefined && y !== undefined) {
            result = result + ',' + x + ',' + y;
        }
        result = result + ')';
        return result;
    }
    function isNullOrUndefined(x) {
        return x === null || x === undefined;
    }
    // Given a d3.quadTree object, visits all the points in it
    // that lie inside the rectangle [x0, x1) x [y0, y1).
    // For each such point calls the given callback, passing the
    // point's quadtree data.
    function quadTreeVisitPointsInRect(quadTree, x0, y0, x1, y1, callback) {
        quadTree.visit((node, nx0, ny0, nx1, ny1) => {
            // Represents the set of points [nx0, nx1) x [ny0, ny1).
            if (node.length === undefined) {
                do {
                    const x = quadTree.x()(node.data);
                    const y = quadTree.y()(node.data);
                    if (x0 <= x && x < x1 && y0 <= y && y < y1) {
                        callback(node.data);
                    }
                } while ((node = node.next));
                return true;
            }
            // Skip this node if Intersection([nx0, nx1) x [ny0, ny1),
            //  [x0, x1) x [y0, y1)) is empty, or equivalently, if
            // either Intersection([nx0, nx1), [x0, x1)) or
            // Intersection([ny0, ny1), [y0, y1)) is empty.
            return nx0 >= x1 || nx1 <= x0 || ny0 >= y1 || ny1 <= y0;
        });
    }
    // Given a d3.quadTree object, visits all the points in it
    // that lie inside the closed disk centered at (centerX, centerY)
    // with the given radius.
    // For each such point calls the given callback, passing the
    // point's quadtree data and the distance from the point to the center
    // of the disk.
    function quadTreeVisitPointsInDisk(quadTree, centerX, centerY, radius, callback) {
        quadTree.visit((node, x0, y0, x1, y1) => {
            // Represents the set of points [x0, x1) x [y0, y1).
            if (node.length === undefined) {
                do {
                    const x = quadTree.x()(node.data);
                    const y = quadTree.y()(node.data);
                    const centerDist = euclideanDist(centerX, centerY, x, y);
                    if (centerDist <= radius) {
                        callback(node.data, centerDist);
                    }
                } while ((node = node.next));
                return true;
            }
            // Skip nodes that represent a rectangle that does not intersect the
            // disk. Equivalently, skip nodes that represent a rectangle whose
            // distance to (centerX, centerY) is larger than radius.
            return pointToRectangleDist(centerX, centerY, x0, y0, x1, y1) > radius;
        });
    }
    // Returns a Set consisting of all elements in 'set' for which
    // predicateFn evaluates to a truthy value.
    function filterSet(set, predicateFn) {
        const result = new Set();
        set.forEach((val) => {
            if (predicateFn(val)) {
                result.add(val);
            }
        });
        return result;
    }
    // Sets the array property of polymerElement to 'newArray' in
    // a Polymer-"observable" manner, so that other elements that have
    // the array as a property (like a dom-repeat) would update correctly.
    // Args:
    //   polymerElement: the polymer element whose array property we want
    //                   to change.
    //   pathToArray: a polymer dot-separated path string to the array
    //   newArray: the new array to set.
    function setArrayObservably(polymerElement, pathToArray, newArray) {
        const currentArray = polymerElement.get(pathToArray, polymerElement);
        // If the current value is not an array, then we use
        // 'polymerElement.set' to replace it with newArray.
        if (!Array.isArray(currentArray)) {
            polymerElement.set(pathToArray, newArray);
            return;
        }
        // Call Polymer.Base.splice() removing the old elements and inserting
        // the new ones.
        // We need to call polymerElement.splice with 'apply' since splice
        // receives a variable argument-list and we want to pass it an array
        // (newArray).
        polymerElement.splice.apply(polymerElement, [pathToArray, 0, currentArray.length].concat(newArray));
    }
    // Computes a simple not-secure 32-bit integer hash value for a string.
    function hashOfString(str) {
        let result = 0;
        for (let i = 0; i < str.length; ++i) {
            result = (result * 31 + str.charCodeAt(i)) & 4294967295;
        }
        // Bitwise operations in JavaScript convert operands to 32-bit 2's
        // complement representation in the range [-2**31,(2**31)-1]. Shift it
        // to the range [0, (2**32)-1].
        return result + Math.pow(2, 31);
    }

    var tf_hparams_utils = /*#__PURE__*/Object.freeze({
        __proto__: null,
        hparamName: hparamName,
        metricName: metricName,
        schemaColumnName: schemaColumnName,
        numHParams: numHParams,
        numMetrics: numMetrics,
        numColumns: numColumns,
        hparamValueByName: hparamValueByName,
        metricValueByName: metricValueByName,
        hparamValueByIndex: hparamValueByIndex,
        metricValueByIndex: metricValueByIndex,
        columnValueByIndex: columnValueByIndex,
        numericColumnExtent: numericColumnExtent,
        getAbsoluteColumnIndex: getAbsoluteColumnIndex,
        schemaVisibleColumnName: schemaVisibleColumnName,
        numVisibleHParams: numVisibleHParams,
        numVisibleMetrics: numVisibleMetrics,
        numVisibleColumns: numVisibleColumns,
        visibleNumericColumnExtent: visibleNumericColumnExtent,
        prettyPrintHParamValueByName: prettyPrintHParamValueByName,
        prettyPrintMetricValueByName: prettyPrintMetricValueByName,
        sessionGroupWithName: sessionGroupWithName,
        hparamValueByVisibleIndex: hparamValueByVisibleIndex,
        metricValueByVisibleIndex: metricValueByVisibleIndex,
        columnValueByVisibleIndex: columnValueByVisibleIndex,
        prettyPrint: prettyPrint,
        l2NormSquared: l2NormSquared,
        euclideanDist: euclideanDist,
        pointToRectangleDist: pointToRectangleDist,
        translateStr: translateStr,
        rotateStr: rotateStr,
        isNullOrUndefined: isNullOrUndefined,
        quadTreeVisitPointsInRect: quadTreeVisitPointsInRect,
        quadTreeVisitPointsInDisk: quadTreeVisitPointsInDisk,
        filterSet: filterSet,
        setArrayObservably: setArrayObservably,
        hashOfString: hashOfString
    });

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    let HparamsSplitLayout = class HparamsSplitLayout extends PolymerElement {
        constructor() {
            super(...arguments);
            this.orientation = 'horizontal';
        }
    };
    HparamsSplitLayout.template = html `
    <slot name="content"></slot>

    <style>
      :host {
        display: block;
      }

      :host slot {
        display: flex;
        height: 100%;
        width: 100%;
      }

      :host ::slotted(*) {
        flex: 0 0 auto;
      }

      :host([orientation='horizontal']) slot {
        flex-direction: row;
        overflow-x: auto;
      }

      :host([orientation='vertical']) slot {
        flex-direction: column;
        overflow-y: auto;
      }

      :host ::slotted(*:not(:last-child)) {
        border: 0 solid var(--divider-color, #ccc);
      }

      :host([orientation='vertical']) ::slotted(*:not(:last-child)) {
        border-bottom-width: 5px;
      }

      :host([orientation='horizontal']) ::slotted(*:not(:last-child)) {
        border-right-width: 5px;
      }
    </style>
  `;
    __decorate([
        property({ type: String, reflectToAttribute: true }),
        __metadata("design:type", String)
    ], HparamsSplitLayout.prototype, "orientation", void 0);
    HparamsSplitLayout = __decorate([
        customElement('hparams-split-layout')
    ], HparamsSplitLayout);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * The tf-hparams-query-pane element implements controls for querying the
     * server for a list of session groups. It provides filtering, and
     * sorting controls.
     *
     * TODO(erez): Add aggregation controls for repeated sessions.
     */
    let TfHparamsQueryPane = class TfHparamsQueryPane extends LegacyElementMixin(PolymerElement) {
        constructor() {
            super(...arguments);
            // Contains the schema and columns visibility status.
            // We use a single object we call 'configuration' to hold both of
            // these properties, since Polymer (v1) doesn't allow atomic
            // changes to multiple properties (that is, sending the notification
            // to downstream consumers only after all properties have been updated).
            //
            // Properties:
            // -----------
            // schema.hparamColumns[i].hparamInfo contains the HParamInfo protocol
            // buffer representing the ith hparam.
            // schema.metricColumns[i].metricInfo contains the MetricInfo protocol
            // buffer representing the ith metric.
            //
            // columnsVisibility[].
            // A boolean array whose ith entry is true if
            // the ith column is visible (selected to be displayed by the user).
            // Columns are indexed by listing the hyperparameters first in
            // the order they are represented by schema.hparamColumns, followed
            // by the metrics in the order they are represented by
            // schema.metricColumns.
            //
            // visibleSchema.
            // DEPRECATED. New code should use the schema and columnVisibile
            // properties instead.
            // TODO(): Remove when all consumers are migrated to use the
            // schema and columnVisibile fields.
            // Contains arrays of HParamInfo and MetricInfo protocol buffers
            // consisting of only the visible hyperparameters and metrics,
            // respectively.
            /**
             * @type {{
             *    schema: {
             *      hparamColumns: Array<{hparamInfo: Object}>,
             *      metricColumns: Array<{metricInfo: Object}>,
             *    },
             *    columnsVisibility: Array<Boolean>,
             *    visibleSchema: {
             *      hparamInfos: Array<Object>,
             *      metricInfos: Array<Object>,
             *    }
             * }}
             */
            this.configuration = {
                schema: {
                    hparamColumns: [],
                    metricColumns: [],
                },
                columnsVisibility: [],
                visibleSchema: {
                    hparamInfos: [],
                    metricInfos: [],
                },
            };
            // The latest list of session groups received from the server.
            // See the comments in the _buildListSessionGroupsRequest() methods for
            // more details.
            this.sessionGroups = [];
            // We track both "data found" and "data not found", because this
            // makes it expedient to display nothing until the initial fetch is
            // complete.  In the absence of this "undefined" case, we would end up
            // briefly displaying the "no data" message before the data arrives.
            // Note that, in the context of a dom-if condition, we cannot easily
            // distinguish between false and undefined.
            this.dataLoadedWithNonEmptyHparams = false;
            // Note that `dataLoadedWithEmptyHparams = false` means that we don't
            // know yet whether data will be found or not (i.e., the fetch has not
            // yet returned).  `dataLoadedWithEmptyHparams = true` means the fetch
            // completed, and there really is no data.
            this.dataLoadedWithEmptyHparams = false;
            // An array of objects each representing information about a session
            // status and whether it is currently allowed by the user.
            this._statuses = [
                { value: 'STATUS_UNKNOWN', displayName: 'Unknown', allowed: true },
                { value: 'STATUS_SUCCESS', displayName: 'Success', allowed: true },
                { value: 'STATUS_FAILURE', displayName: 'Failure', allowed: true },
                { value: 'STATUS_RUNNING', displayName: 'Running', allowed: true },
            ];
            // A promise that resolves after the initial getExperiment network RPC
            // resolves. Used for unit-test to allow testing to run after the
            // element has initialized.
            this._getExperimentResolved = new Promise((resolve) => {
                this._resolveGetExperiment = resolve;
            });
            // A tf_backend.canceller used to keep track of pending
            // ListSessionGroups requests and cancel their resulting UI updates
            // when a new ListSessionGroups request is made.
            this._listSessionGroupsCanceller = new Canceller();
            // The value and invalid properties of the 'page-size' input box.
            this._pageSizeInput = { value: '100', invalid: false };
            // The value and invalid properties of the 'page-number' input box.
            this._pageNumberInput = { value: '1', invalid: false };
            // The string displaying the total number of pages or '?' if unknown.
            this._pageCountStr = '?';
            this._hparamName = hparamName;
            this._metricName = metricName;
            this._prettyPrint = prettyPrint;
        }
        reload() {
            this._queryServer();
        }
        _csvUrl(request, configuration) {
            return this._downloadDataUrl(request, configuration, 'csv');
        }
        _jsonUrl(request, configuration) {
            return this._downloadDataUrl(request, configuration, 'json');
        }
        _latexUrl(request, configuration) {
            return this._downloadDataUrl(request, configuration, 'latex');
        }
        _downloadDataUrl(request, configuration, format) {
            const visibility = configuration.columnsVisibility;
            return this.backend.getDownloadUrl(format, request, visibility);
        }
        // Sends a request for experiment to the server, and updates
        // the state of this element accordingly. Currently, only called
        // once on element initialization.
        _computeExperimentAndRelatedProps() {
            const utils = tf_hparams_utils;
            if (utils.isNullOrUndefined(this.backend) ||
                utils.isNullOrUndefined(this.experimentName)) {
                return;
            }
            const experimentRequest = {
                experimentName: this.experimentName,
            };
            this.backend
                .getExperiment(experimentRequest)
                .then((experiment) => {
                if (lodash.isEqual(experiment, this._experiment)) {
                    // No need to update anything if there are no changes.
                    return;
                }
                this.set('_experiment', experiment);
                this._computeHParams();
                this._computeMetrics();
                this._queryServer();
                this._resolveGetExperiment();
            })
                .finally(() => {
                this._computeDataFound();
            });
        }
        _computeDataFound() {
            const result = Boolean(this._experiment &&
                this._experiment.hparamInfos &&
                this._experiment.hparamInfos.length > 0 &&
                this._experiment.metricInfos &&
                this._experiment.metricInfos.length > 0);
            this.set('dataLoadedWithNonEmptyHparams', result);
            this.set('dataLoadedWithEmptyHparams', !result);
        }
        // Updates the _hparams property from the _experiment property.
        _computeHParams() {
            const result = [];
            const kNumHParamsToDisplayByDefault = 5;
            this._experiment.hparamInfos.forEach((anInfo, index) => {
                const hparam = {
                    info: anInfo,
                    displayed: index < kNumHParamsToDisplayByDefault,
                    filter: {},
                };
                if (hparam.info.hasOwnProperty('domainDiscrete')) {
                    hparam.filter.domainDiscrete = [];
                    hparam.info.domainDiscrete.forEach((val) => {
                        hparam.filter.domainDiscrete.push({
                            value: val,
                            checked: true,
                        });
                    });
                }
                else if (hparam.info.type === 'DATA_TYPE_BOOL') {
                    hparam.filter.domainDiscrete = [
                        {
                            value: false,
                            checked: true,
                        },
                        {
                            value: true,
                            checked: true,
                        },
                    ];
                }
                else if (hparam.info.type === 'DATA_TYPE_FLOAT64') {
                    hparam.filter.interval = {
                        min: {
                            value: '',
                            invalid: false,
                        },
                        max: {
                            value: '',
                            invalid: false,
                        },
                    };
                }
                else if (hparam.info.type === 'DATA_TYPE_STRING') {
                    hparam.filter.regexp = '';
                }
                else {
                    console.warn('unknown hparam.info.type: %s', hparam.info.type);
                }
                result.push(hparam);
            });
            this.set('_hparams', result);
        }
        // Updates the _metrics property from the _experiment property.
        _computeMetrics() {
            const result = [];
            // By default we display the first kNumMetricsToDisplayByDefault metrics
            // and not the rest.
            const kNumMetricsToDisplayByDefault = 5;
            this._experiment.metricInfos.forEach((info, index) => {
                const metric = {
                    info: info,
                    filter: {
                        interval: {
                            min: {
                                value: '',
                                invalid: false,
                            },
                            max: {
                                value: '',
                                invalid: false,
                            },
                        },
                    },
                    displayed: index < kNumMetricsToDisplayByDefault,
                };
                result.push(metric);
            });
            this.set('_metrics', result);
        }
        _computeSchema() {
            if (!this._hparams || !this._metrics) {
                return { hparamColumns: [], metricColumns: [] };
            }
            return {
                hparamColumns: this._hparams.map((hparam) => ({
                    hparamInfo: hparam.info,
                })),
                metricColumns: this._metrics.map((metric) => ({
                    metricInfo: metric.info,
                })),
            };
        }
        _updateConfiguration() {
            this.debounce('_updateConfiguration', () => {
                this.configuration = {
                    schema: this._computeSchema(),
                    columnsVisibility: this._computeColumnsVisibility(),
                    visibleSchema: this._computeVisibleSchema(),
                };
            });
        }
        _computeColumnsVisibility() {
            if (!this._hparams || !this._metrics)
                return [];
            return this._hparams
                .map((hparam) => hparam.displayed)
                .concat(this._metrics.map((metric) => metric.displayed));
        }
        _computeVisibleSchema() {
            if (!this._hparams || !this._metrics) {
                return { hparamInfos: [], metricInfos: [] };
            }
            const newHParamInfos = this._hparams
                .filter((hparam) => hparam.displayed)
                .map((hparam) => hparam.info);
            const newMetricInfos = this._metrics
                .filter((metric) => metric.displayed)
                .map((metric) => metric.info);
            return {
                hparamInfos: newHParamInfos,
                metricInfos: newMetricInfos,
            };
        }
        // Sends a query to the server for the list of session groups.
        // Asynchronously updates the sessionGroups property with the response.
        _queryServer() {
            this.debounce('queryServer', () => this._queryServerNoDebounce(), 100);
        }
        // Directly queries the server without a debounce, called in queryServer()
        // above. Returns a promise, for use in unit-tests, that resolves (with
        // 'undefined') when the ListSessionGroups RPC returns.
        _queryServerNoDebounce() {
            if (!this._hparams || !this._metrics) {
                return;
            }
            return this._sendListSessionGroupsRequest().then(this._listSessionGroupsCanceller.cancellable(({ value, cancelled }) => {
                if (cancelled) {
                    return;
                }
                // The server may not support a "totalSize" field in
                // which case this field would be negative, and we
                // populate the page count with a "?".
                if (value.totalSize >= 0) {
                    const pageSize = +this._pageSizeInput.value;
                    this.set('_pageCountStr', String(Math.ceil(value.totalSize / pageSize)));
                    this.set('_totalSessionGroupsCountStr', value.totalSize);
                }
                else {
                    this.set('_pageCountStr', '?');
                    this.set('_totalSessionGroupsCountStr', 'Unknown');
                }
                setArrayObservably(this, 'sessionGroups', value.sessionGroups);
            }));
        }
        _sendListSessionGroupsRequest() {
            const request = this._buildListSessionGroupsRequest();
            if (request === null) {
                // If query configuration is not valid don't send a request.
                return;
            }
            this.set('_sessionGroupsRequest', request);
            // Note that the responses to the RPCs sent
            // to the backend may return in a different order than the order in
            // which they were sent. If we just update sessionGroups in the
            // order of responses received, we can end up
            // with a sessionGroups object that doesn't match the filtering and
            // sorting configuration. For example, if the last response received
            // was for a query belonging to the penultimate filtering and
            // sorting configuration. To solve this, we use tf_backend.canceller
            // to cancel all the pending promises from previous queries for which
            // we haven't received a response yet.
            this._listSessionGroupsCanceller.cancelAll();
            return this.backend.listSessionGroups(request);
        }
        // Reads the values of the properties bound to the query elements
        // and builds the corresponding ListSessionGroupsRequest.
        // If the user settings are invalid (e.g. an input-box doesn't containt
        // a parsable number), marks the corresponding elements (if they exist,
        // see next paragraph) as invalid so that they will display an
        // error message and returns null.
        //
        // Note, that this method can be called before the elements have
        // been rendered (when the tf-hparams-query is first created), in which
        // case it will not try to mark the elements as invalid.
        _buildListSessionGroupsRequest() {
            const _this = this;
            // Will be set to false if we encounter any invalid inputs.
            let queryValid = true;
            // Parses an inputInterval object of the form:
            // {min: {value: string, invalid: boolean},
            //  max: {value: string  invalid: boolean}}. Returns an object
            // of the form: {minValue: number, maxValue: number} suitable to
            // be included as an interval in ListSessionGroupsRequest. If the
            // min or max values cannot be parsed, returns null. In both cases
            // sets the 'invalid' fields to denote whether the inputs is invalid.
            //
            // The inputIntervalPath should be a Polymer path for the inputInterval
            // object. We work with paths, rather than the object, so that we
            // can make observable changes.
            function parseInputInterval(inputIntervalPath) {
                const minValueStr = _this.get(inputIntervalPath + '.min.value');
                console.assert(minValueStr !== undefined);
                // The protobuffer JSON mapping maps the strings "-Infinity" and
                // "Infinity" to the floating-point infinity and -infinity values.
                const minValue = minValueStr === '' ? '-Infinity' : +minValueStr;
                _this.set(inputIntervalPath + '.min.invalid', isNaN(minValue));
                queryValid = queryValid && !isNaN(minValue);
                const maxValueStr = _this.get(inputIntervalPath + '.max.value');
                console.assert(maxValueStr !== undefined);
                const maxValue = maxValueStr === '' ? 'Infinity' : +maxValueStr;
                _this.set(inputIntervalPath + '.max.invalid', isNaN(maxValue));
                queryValid = queryValid && !isNaN(maxValue);
                if (isNaN(minValue) || isNaN(maxValue)) {
                    return null;
                }
                return { minValue: minValue, maxValue: maxValue };
            }
            // Parses the value of an input object of the form:
            // {value: string, invalid: boolean} as a positive integer and returns
            // the integer or null if the parsing fails.
            // If parsing fails, also sets the 'invalid' property of the input
            // object to false.
            //
            // The inputPath should be a Polymer path to input object. We work
            // with paths, rather than the object, so that we can make observable
            // changes.
            function parseInputAsPositiveInt(inputPath) {
                const inputValueStr = _this.get(inputPath + '.value');
                console.assert(inputValueStr !== undefined);
                const result = +inputValueStr;
                const validResult = Number.isInteger(result) && result > 0;
                _this.set(inputPath + '.invalid', !validResult);
                queryValid = queryValid && validResult;
                if (validResult) {
                    return result;
                }
                return null;
            }
            // Build the allowed status filters.
            const allowedStatuses = this._statuses
                .filter((s) => s.allowed)
                .map((s) => s.value);
            let colParams = [];
            // Build the hparams filters in the request.
            this._hparams.forEach((hparam, index) => {
                let colParam = { hparam: hparam.info.name };
                if (hparam.filter.domainDiscrete) {
                    colParam.filterDiscrete = [];
                    hparam.filter.domainDiscrete.forEach((filterVal) => {
                        if (filterVal.checked) {
                            colParam.filterDiscrete.push(filterVal.value);
                        }
                    });
                }
                else if (hparam.filter.interval) {
                    colParam.filterInterval = parseInputInterval('_hparams.' + index + '.filter.interval');
                }
                else if (hparam.filter.regexp) {
                    colParam.filterRegexp = hparam.filter.regexp;
                }
                else {
                    console.error('hparam.filter with no domainDiscrete, interval or regexp' +
                        ' properties set: %s', hparam);
                    return null;
                }
                colParams.push(colParam);
            });
            // Build the metric filters in the request.
            this._metrics.forEach((metric, index) => {
                let colParam = {
                    metric: metric.info.name,
                    filterInterval: parseInputInterval('_metrics.' + index + '.filter.interval'),
                };
                colParams.push(colParam);
            });
            // Sorting.
            // TODO(erez): Support sorting by multiple columns.
            if (this._sortByIndex !== undefined && this._sortDirection !== undefined) {
                if (!(this._sortByIndex in colParams)) {
                    console.error('No column in colParams with index sortByIndex: %s', this._sortByIndex);
                    return null;
                }
                colParams[this._sortByIndex].order =
                    this._sortDirection === 0 ? 'ORDER_ASC' : 'ORDER_DESC';
            }
            // Paging.
            const pageNum = parseInputAsPositiveInt('_pageNumberInput');
            const pageSize = parseInputAsPositiveInt('_pageSizeInput');
            if (!queryValid) {
                return null;
            }
            const startIndex = pageSize * (pageNum - 1);
            return {
                experimentName: this.experimentName,
                allowedStatuses: allowedStatuses,
                colParams: colParams,
                startIndex: startIndex,
                sliceSize: pageSize,
            };
        }
        // We allow sorting by any column of a session group. A column is either a
        // metric or a hyperparameter.
        // For the purpose of sorting, we identify each such column by its index
        // in the list formed by concatenating the metric in _metrics after the
        // hparams in _hparams. This method computes this sorting index of the
        // metric column represented by _metrics[metricIndex].
        _metricSortByIndex(metricIndex) {
            return metricIndex + this._hparams.length;
        }
    };
    TfHparamsQueryPane.template = html `
    <hparams-split-layout orientation="vertical">
      <div slot="content" class="section hyperparameters">
        <div class="section-title">Hyperparameters</div>
        <template is="dom-repeat" items="{{_hparams}}" as="hparam">
          <div class="hparam">
            <paper-checkbox
              checked="{{hparam.displayed}}"
              class="hparam-checkbox"
            >
              [[_hparamName(hparam.info)]]
            </paper-checkbox>
            <!-- Precisely one of the templates below will be stamped.-->
            <!-- 1. A list of checkboxes -->
            <template is="dom-if" if="[[hparam.filter.domainDiscrete]]">
              <template
                is="dom-repeat"
                items="[[hparam.filter.domainDiscrete]]"
              >
                <paper-checkbox
                  checked="{{item.checked}}"
                  class="discrete-value-checkbox"
                  on-change="_queryServer"
                >
                  [[_prettyPrint(item.value)]]
                </paper-checkbox>
              </template>
            </template>
            <!-- 2. A numeric interval -->
            <template is="dom-if" if="[[hparam.filter.interval]]">
              <paper-input
                label="Min"
                value="{{hparam.filter.interval.min.value}}"
                allowed_pattern="[0-9.e\\-]"
                on-value-changed="_queryServer"
                error-message="Invalid input"
                invalid="[[hparam.filter.interval.min.invalid]]"
                placeholder="-infinity"
              >
              </paper-input>
              <paper-input
                label="Max"
                value="{{hparam.filter.interval.max.value}}"
                allowed_pattern="[0-9.e\\-]"
                on-value-changed="_queryServer"
                error-message="Invalid input"
                invalid="[[hparam.filter.interval.max.invalid]]"
                placeholder="+infinity"
              >
              </paper-input>
            </template>
            <!-- 3. A regexp -->
            <template is="dom-if" if="[[hparam.filter.regexp]]">
              <paper-input
                label="Regular expression"
                value="{{hparam.filter.regexp}}"
                on-value-changed="_queryServer"
              >
              </paper-input>
            </template>
          </div>
        </template>
      </div>
      <div slot="content" class="section metrics">
        <div class="section-title">Metrics</div>
        <template is="dom-repeat" items="{{_metrics}}" as="metric">
          <div class="metric">
            <!-- TODO(erez): Make it easier to handle a large number of
                  metrics:
                  1. Add an 'isolator' radio-button to select just one
                  metric and
                  hide all the rest
                  2. Add a 'toggle-all' button that will hide/unhide
                    all the
                  metrics.
                  Use similar logic/appearance to the run-selector of
                  scalars.-->
            <paper-checkbox
              checked="{{metric.displayed}}"
              class="metric-checkbox"
            >
              [[_metricName(metric.info)]]
            </paper-checkbox>
            <div class="inline-element">
              <paper-input
                label="Min"
                value="{{metric.filter.interval.min.value}}"
                allowed-pattern="[0-9.e\\-]"
                on-value-changed="_queryServer"
                error-message="Invalid input"
                invalid="{{metric.filter.interval.min.invalid}}"
                placeholder="-infinity"
              >
              </paper-input>
            </div>
            <div class="inline-element">
              <paper-input
                label="Max"
                allowed-pattern="[0-9.e\\-]"
                value="{{metric.filter.interval.max.value}}"
                on-value-changed="_queryServer"
                error-message="Invalid input"
                invalid="{{metric.filter.interval.max.invalid}}"
                placeholder="+infinity"
              >
              </paper-input>
            </div>
          </div>
        </template>
      </div>
      <div slot="content" class="section status">
        <div class="section-title">Status</div>
        <template is="dom-repeat" items="[[_statuses]]" as="status">
          <paper-checkbox checked="{{status.allowed}}" on-change="_queryServer">
            [[status.displayName]]
          </paper-checkbox>
        </template>
      </div>
      <div slot="content" class="section sorting">
        <div class="section-title">Sorting</div>
        <paper-dropdown-menu
          label="Sort by"
          on-selected-item-changed="_queryServer"
          horizontal-align="left"
        >
          <paper-listbox
            class="dropdown-content"
            slot="dropdown-content"
            selected="{{_sortByIndex}}"
            on-selected-item-changed="_queryServer"
          >
            <template is="dom-repeat" items="[[_hparams]]" as="hparam">
              <paper-item> [[_hparamName(hparam.info)]] </paper-item>
            </template>
            <template is="dom-repeat" items="[[_metrics]]" as="metric">
              <paper-item> [[_metricName(metric.info)]] </paper-item>
            </template>
          </paper-listbox>
        </paper-dropdown-menu>
        <paper-dropdown-menu
          label="Direction"
          on-selected-item-changed="_queryServer"
          horizontal-align="left"
        >
          <paper-listbox
            class="dropdown-content"
            slot="dropdown-content"
            selected="{{_sortDirection}}"
          >
            <paper-item>Ascending</paper-item>
            <paper-item>Descending</paper-item>
          </paper-listbox>
        </paper-dropdown-menu>
      </div>
      <div slot="content" class="section paging">
        <div class="section-title">Paging</div>
        <div>
          Number of matching session groups: [[_totalSessionGroupsCountStr]]
        </div>
        <div class="inline-element page-number-input">
          <paper-input
            label="Page #"
            value="{{_pageNumberInput.value}}"
            allowed-pattern="[0-9]"
            error-message="Invalid input"
            invalid="[[_pageNumberInput.invalid]]"
            on-value-changed="_queryServer"
          >
            <div slot="suffix" class="page-suffix">/ [[_pageCountStr]]</div>
          </paper-input>
        </div>
        <div class="inline-element page-size-input">
          <paper-input
            label="Max # of session groups per page:"
            value="{{_pageSizeInput.value}}"
            allowed-pattern="[0-9]"
            error-message="Invalid input"
            invalid="[[_pageSizeInput.invalid]]"
            on-value-changed="_queryServer"
          >
          </paper-input>
        </div>
      </div>
      <div slot="content" class="section download">
        <template is="dom-if" if="[[_sessionGroupsRequest]]">
          Download data as
          <span>
            <a
              id="csvLink"
              download="hparams_table.csv"
              href="[[_csvUrl(_sessionGroupsRequest, configuration)]]"
              >CSV</a
            >
            <a
              id="jsonLink"
              download="hparams_table.json"
              href="[[_jsonUrl(_sessionGroupsRequest, configuration)]]"
              >JSON</a
            >
            <a
              id="latexLink"
              download="hparams_table.tex"
              href="[[_latexUrl(_sessionGroupsRequest, configuration)]]"
              >LaTeX</a
            >
          </span>
        </template>
      </div>
    </hparams-split-layout>
    <style>
      .section {
        padding: 10px;
      }
      .section-title {
        display: block;
        font-weight: bold;
        text-decoration: underline;
        margin-bottom: 7px;
      }
      .discrete-value-checkbox,
      .metric-checkbox,
      .hparam-checkbox {
        display: block;
      }
      .discrete-value-checkbox {
        margin-left: 20px;
      }
      .hparam,
      .metric {
        display: block;
      }
      .inline-element {
        display: inline-block;
        width: 40%;
        margin-left: 10px;
      }
      .page-number-input {
        width: 20%;
      }
      .page-size-input {
        width: 60%;
      }
      vaadin-split-layout {
        height: 100%;
      }
      paper-listbox {
        max-height: 15em;
      }
      .page-suffix {
        white-space: nowrap;
      }
    </style>
  `;
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfHparamsQueryPane.prototype, "experimentName", void 0);
    __decorate([
        property({
            type: Object,
            notify: true,
        }),
        __metadata("design:type", Object)
    ], TfHparamsQueryPane.prototype, "configuration", void 0);
    __decorate([
        property({
            type: Array,
            notify: true,
        }),
        __metadata("design:type", Object)
    ], TfHparamsQueryPane.prototype, "sessionGroups", void 0);
    __decorate([
        property({
            type: Boolean,
            notify: true,
        }),
        __metadata("design:type", Boolean)
    ], TfHparamsQueryPane.prototype, "dataLoadedWithNonEmptyHparams", void 0);
    __decorate([
        property({
            type: Boolean,
            notify: true,
        }),
        __metadata("design:type", Boolean)
    ], TfHparamsQueryPane.prototype, "dataLoadedWithEmptyHparams", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfHparamsQueryPane.prototype, "_experiment", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfHparamsQueryPane.prototype, "_hparams", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfHparamsQueryPane.prototype, "_metrics", void 0);
    __decorate([
        property({
            type: Array,
        }),
        __metadata("design:type", Object)
    ], TfHparamsQueryPane.prototype, "_statuses", void 0);
    __decorate([
        property({
            type: Object,
        }),
        __metadata("design:type", Object)
    ], TfHparamsQueryPane.prototype, "_getExperimentResolved", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Function)
    ], TfHparamsQueryPane.prototype, "_resolveGetExperiment", void 0);
    __decorate([
        property({
            type: Object,
        }),
        __metadata("design:type", Object)
    ], TfHparamsQueryPane.prototype, "_listSessionGroupsCanceller", void 0);
    __decorate([
        property({ type: Number }),
        __metadata("design:type", Number)
    ], TfHparamsQueryPane.prototype, "_sortByIndex", void 0);
    __decorate([
        property({ type: Number }),
        __metadata("design:type", Number)
    ], TfHparamsQueryPane.prototype, "_sortDirection", void 0);
    __decorate([
        property({
            type: Object,
        }),
        __metadata("design:type", Object)
    ], TfHparamsQueryPane.prototype, "_pageSizeInput", void 0);
    __decorate([
        property({
            type: Object,
        }),
        __metadata("design:type", Object)
    ], TfHparamsQueryPane.prototype, "_pageNumberInput", void 0);
    __decorate([
        property({
            type: String,
        }),
        __metadata("design:type", String)
    ], TfHparamsQueryPane.prototype, "_pageCountStr", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfHparamsQueryPane.prototype, "_totalSessionGroupsCountStr", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfHparamsQueryPane.prototype, "_sessionGroupsRequest", void 0);
    __decorate([
        observe('backend', 'experimentName'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfHparamsQueryPane.prototype, "_computeExperimentAndRelatedProps", null);
    __decorate([
        observe('_hparams.*', '_metrics.*'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfHparamsQueryPane.prototype, "_updateConfiguration", null);
    TfHparamsQueryPane = __decorate([
        customElement('tf-hparams-query-pane')
    ], TfHparamsQueryPane);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    let TfHparamsScaleAndColorControls = class TfHparamsScaleAndColorControls extends PolymerElement {
        constructor() {
            super(...arguments);
            this.options = null;
        }
        /* Private methods */
        _configurationChanged() {
            // Populate options.columns with a linear scale for each column (
            // hparam or metric).
            const visibleSchema = this.configuration.visibleSchema;
            const schema = this.configuration.schema;
            const newHParamColumn = (info, index) => {
                return {
                    name: hparamName(info),
                    index: index,
                    absoluteIndex: getAbsoluteColumnIndex(schema, visibleSchema, index),
                    scale: this._isNumericColumn(index) ? 'LINEAR' : 'NON_NUMERIC',
                };
            };
            const newMetricColumn = (info, index) => {
                const colIndex = index + visibleSchema.hparamInfos.length;
                return {
                    scale: 'LINEAR',
                    name: metricName(info),
                    index: colIndex,
                    absoluteIndex: getAbsoluteColumnIndex(schema, visibleSchema, colIndex),
                };
            };
            const options = {
                columns: visibleSchema.hparamInfos
                    .map(newHParamColumn)
                    .concat(visibleSchema.metricInfos.map(newMetricColumn)),
                minColor: '#0000FF',
                maxColor: '#FF0000',
                configuration: this.configuration,
            };
            // Set the colorByColumnIndex property.
            // If we set options.colorByColumnIndex at the same time as we
            // set the other options, Polymer first updates the drop-down menu
            // selected label and only then updates the list box with the new items.
            // As a result the selected label gets an erroneous value (based on
            // the old elements in the list).
            // To overcome this, we first set the selected item to "undefined",
            // call flush to synchronously update the drop-down menu list with
            // the new items, and then reset the selected item index so that
            // Polymer will update the label based on the new list.
            // See also: https://github.com/PolymerElements/paper-dropdown-menu/issues/197#issuecomment-249927371, and http://jsbin.com/fuqoye/edit?html,output.
            this.set('options', options); // set the bound selected item to
            // undefined.
            flush();
            this.set('options.colorByColumnIndex', this._defaultColorByColumnIndex());
        }
        _unselectDisabledLogScales() {
            if (this.options === null) {
                // We may be called before the options are constructed by
                // _configurationChanged(). In this case we need not worry
                // about selected disabled log scales.
                return;
            }
            this.options.columns.forEach((col) => {
                const colPath = 'options.columns.' + col.index;
                if (!this._allowLogScale(col) && col.scale === 'LOG') {
                    // We need to use Polymer paths to make the change observable.
                    this.set(colPath + '.scale', 'LINEAR');
                }
            });
        }
        _allowLogScale(column) {
            if (!this._isNumericColumn(column.index) || !this.sessionGroups) {
                return false;
            }
            const [min, max] = visibleNumericColumnExtent(this.configuration.visibleSchema, this.sessionGroups, column.index);
            // Log scale is only defined when the domain does not include 0.
            return min > 0 || max < 0;
        }
        // Returns true if the scale is numeric.
        // Used to prevent non-numeric columns from having a scale-selection
        // radio group.
        _isNumericColumn(colIndex) {
            return (colIndex >= this.configuration.visibleSchema.hparamInfos.length ||
                this.configuration.visibleSchema.hparamInfos[colIndex].type ===
                    'DATA_TYPE_FLOAT64');
        }
        // Use the first metric if there are metrics, or otherwise the first
        // numeric hparam if there are hparams. If there are no numeric columns
        // return undefined.
        _defaultColorByColumnIndex() {
            if (this.configuration.visibleSchema.metricInfos.length > 0) {
                return this.configuration.visibleSchema.hparamInfos.length;
            }
            const i = this.configuration.visibleSchema.hparamInfos.findIndex((info) => info.type === 'DATA_TYPE_FLOAT64');
            if (i !== -1) {
                return i;
            }
            return undefined;
        }
    };
    TfHparamsScaleAndColorControls.template = html `
    <div class="control-panel">
      <!-- 'Color by' drop down menu -->
      <paper-dropdown-menu
        label="Color by"
        id="colorByDropDownMenu"
        horizontal-align="left"
      >
        <paper-listbox
          class="dropdown-content"
          slot="dropdown-content"
          selected="{{options.colorByColumnIndex}}"
          id="colorByListBox"
        >
          <template
            is="dom-repeat"
            items="[[options.columns]]"
            as="column"
            id="colorByColumnTemplate"
          >
            <paper-item disabled="[[!_isNumericColumn(column.index)]]">
              [[column.name]]
            </paper-item>
          </template>
        </paper-listbox>
      </paper-dropdown-menu>

      <!-- Columns scales -->
      <div class="columns-container">
        <!-- Scale options for each numeric feature -->
        <template is="dom-repeat" items="{{options.columns}}" as="column">
          <template is="dom-if" if="[[_isNumericColumn(column.index)]]">
            <div class="column">
              <div class="column-title">[[column.name]]</div>
              <div>
                <paper-radio-group
                  class="scale-radio-group"
                  selected="{{column.scale}}"
                >
                  <paper-radio-button name="LINEAR">
                    Linear
                  </paper-radio-button>
                  <!-- The id here is used to access this button in unit
                       tests.-->
                  <paper-radio-button
                    id="logScaleButton_[[column.name]]"
                    name="LOG"
                    disabled="[[!_allowLogScale(column, sessionGroups.*)]]"
                  >
                    Logarithmic
                  </paper-radio-button>
                  <paper-radio-button name="QUANTILE">
                    Quantile
                  </paper-radio-button>
                </paper-radio-group>
              </div>
            </div>
          </template>
        </template>
      </div>
    </div>

    <style>
      :host {
        display: block;
      }
      .control-panel {
        overflow: auto;
      }
      .column {
        flex-grow: 1;
        flex-shrink: 1;
        margin-right: 5px;
        border: solid 1px darkgray;
        padding: 3px;
      }
      .column-title {
        /* Fit every title in one line so the radio boxes align vertically. */
        white-space: nowrap;
        text-decoration: underline;
      }
      .columns-container {
        display: flex;
        flex-direction: row;
      }
      .scale-radio-group paper-radio-button {
        padding: 2px;
        display: block;
      }
      paper-listbox {
        max-height: 15em;
      }
    </style>
  `;
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfHparamsScaleAndColorControls.prototype, "configuration", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfHparamsScaleAndColorControls.prototype, "sessionGroups", void 0);
    __decorate([
        property({
            type: Object,
            notify: true,
        }),
        __metadata("design:type", Object)
    ], TfHparamsScaleAndColorControls.prototype, "options", void 0);
    __decorate([
        observe('configuration.*'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfHparamsScaleAndColorControls.prototype, "_configurationChanged", null);
    __decorate([
        observe('sessionGroups.*'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfHparamsScaleAndColorControls.prototype, "_unselectDisabledLogScales", null);
    TfHparamsScaleAndColorControls = __decorate([
        customElement('tf-hparams-scale-and-color-controls')
    ], TfHparamsScaleAndColorControls);

    class Lumo extends HTMLElement {
      static get version() {
        return '1.6.0';
      }
    }

    customElements.define('vaadin-lumo-styles', Lumo);

    const $_documentContainer$5 = document.createElement('template');

    $_documentContainer$5.innerHTML = `<custom-style>
  <style>
    html {
      /* Base (background) */
      --lumo-base-color: #FFF;

      /* Tint */
      --lumo-tint-5pct: hsla(0, 0%, 100%, 0.3);
      --lumo-tint-10pct: hsla(0, 0%, 100%, 0.37);
      --lumo-tint-20pct: hsla(0, 0%, 100%, 0.44);
      --lumo-tint-30pct: hsla(0, 0%, 100%, 0.5);
      --lumo-tint-40pct: hsla(0, 0%, 100%, 0.57);
      --lumo-tint-50pct: hsla(0, 0%, 100%, 0.64);
      --lumo-tint-60pct: hsla(0, 0%, 100%, 0.7);
      --lumo-tint-70pct: hsla(0, 0%, 100%, 0.77);
      --lumo-tint-80pct: hsla(0, 0%, 100%, 0.84);
      --lumo-tint-90pct: hsla(0, 0%, 100%, 0.9);
      --lumo-tint: #FFF;

      /* Shade */
      --lumo-shade-5pct: hsla(214, 61%, 25%, 0.05);
      --lumo-shade-10pct: hsla(214, 57%, 24%, 0.1);
      --lumo-shade-20pct: hsla(214, 53%, 23%, 0.16);
      --lumo-shade-30pct: hsla(214, 50%, 22%, 0.26);
      --lumo-shade-40pct: hsla(214, 47%, 21%, 0.38);
      --lumo-shade-50pct: hsla(214, 45%, 20%, 0.5);
      --lumo-shade-60pct: hsla(214, 43%, 19%, 0.61);
      --lumo-shade-70pct: hsla(214, 42%, 18%, 0.72);
      --lumo-shade-80pct: hsla(214, 41%, 17%, 0.83);
      --lumo-shade-90pct: hsla(214, 40%, 16%, 0.94);
      --lumo-shade: hsl(214, 35%, 15%);

      /* Contrast */
      --lumo-contrast-5pct: var(--lumo-shade-5pct);
      --lumo-contrast-10pct: var(--lumo-shade-10pct);
      --lumo-contrast-20pct: var(--lumo-shade-20pct);
      --lumo-contrast-30pct: var(--lumo-shade-30pct);
      --lumo-contrast-40pct: var(--lumo-shade-40pct);
      --lumo-contrast-50pct: var(--lumo-shade-50pct);
      --lumo-contrast-60pct: var(--lumo-shade-60pct);
      --lumo-contrast-70pct: var(--lumo-shade-70pct);
      --lumo-contrast-80pct: var(--lumo-shade-80pct);
      --lumo-contrast-90pct: var(--lumo-shade-90pct);
      --lumo-contrast: var(--lumo-shade);

      /* Text */
      --lumo-header-text-color: var(--lumo-contrast);
      --lumo-body-text-color: var(--lumo-contrast-90pct);
      --lumo-secondary-text-color: var(--lumo-contrast-70pct);
      --lumo-tertiary-text-color: var(--lumo-contrast-50pct);
      --lumo-disabled-text-color: var(--lumo-contrast-30pct);

      /* Primary */
      --lumo-primary-color: hsl(214, 90%, 52%);
      --lumo-primary-color-50pct: hsla(214, 90%, 52%, 0.5);
      --lumo-primary-color-10pct: hsla(214, 90%, 52%, 0.1);
      --lumo-primary-text-color: var(--lumo-primary-color);
      --lumo-primary-contrast-color: #FFF;

      /* Error */
      --lumo-error-color: hsl(3, 100%, 61%);
      --lumo-error-color-50pct: hsla(3, 100%, 60%, 0.5);
      --lumo-error-color-10pct: hsla(3, 100%, 60%, 0.1);
      --lumo-error-text-color: hsl(3, 92%, 53%);
      --lumo-error-contrast-color: #FFF;

      /* Success */
      --lumo-success-color: hsl(145, 80%, 42%); /* hsl(144,82%,37%); */
      --lumo-success-color-50pct: hsla(145, 76%, 44%, 0.55);
      --lumo-success-color-10pct: hsla(145, 76%, 44%, 0.12);
      --lumo-success-text-color: hsl(145, 100%, 32%);
      --lumo-success-contrast-color: #FFF;
    }
  </style>
</custom-style><dom-module id="lumo-color">
  <template>
    <style>
      [theme~="dark"] {
        /* Base (background) */
        --lumo-base-color: hsl(214, 35%, 21%);

        /* Tint */
        --lumo-tint-5pct: hsla(214, 65%, 85%, 0.06);
        --lumo-tint-10pct: hsla(214, 60%, 80%, 0.14);
        --lumo-tint-20pct: hsla(214, 64%, 82%, 0.23);
        --lumo-tint-30pct: hsla(214, 69%, 84%, 0.32);
        --lumo-tint-40pct: hsla(214, 73%, 86%, 0.41);
        --lumo-tint-50pct: hsla(214, 78%, 88%, 0.5);
        --lumo-tint-60pct: hsla(214, 82%, 90%, 0.6);
        --lumo-tint-70pct: hsla(214, 87%, 92%, 0.7);
        --lumo-tint-80pct: hsla(214, 91%, 94%, 0.8);
        --lumo-tint-90pct: hsla(214, 96%, 96%, 0.9);
        --lumo-tint: hsl(214, 100%, 98%);

        /* Shade */
        --lumo-shade-5pct: hsla(214, 0%, 0%, 0.07);
        --lumo-shade-10pct: hsla(214, 4%, 2%, 0.15);
        --lumo-shade-20pct: hsla(214, 8%, 4%, 0.23);
        --lumo-shade-30pct: hsla(214, 12%, 6%, 0.32);
        --lumo-shade-40pct: hsla(214, 16%, 8%, 0.41);
        --lumo-shade-50pct: hsla(214, 20%, 10%, 0.5);
        --lumo-shade-60pct: hsla(214, 24%, 12%, 0.6);
        --lumo-shade-70pct: hsla(214, 28%, 13%, 0.7);
        --lumo-shade-80pct: hsla(214, 32%, 13%, 0.8);
        --lumo-shade-90pct: hsla(214, 33%, 13%, 0.9);
        --lumo-shade: hsl(214, 33%, 13%);

        /* Contrast */
        --lumo-contrast-5pct: var(--lumo-tint-5pct);
        --lumo-contrast-10pct: var(--lumo-tint-10pct);
        --lumo-contrast-20pct: var(--lumo-tint-20pct);
        --lumo-contrast-30pct: var(--lumo-tint-30pct);
        --lumo-contrast-40pct: var(--lumo-tint-40pct);
        --lumo-contrast-50pct: var(--lumo-tint-50pct);
        --lumo-contrast-60pct: var(--lumo-tint-60pct);
        --lumo-contrast-70pct: var(--lumo-tint-70pct);
        --lumo-contrast-80pct: var(--lumo-tint-80pct);
        --lumo-contrast-90pct: var(--lumo-tint-90pct);
        --lumo-contrast: var(--lumo-tint);

        /* Text */
        --lumo-header-text-color: var(--lumo-contrast);
        --lumo-body-text-color: var(--lumo-contrast-90pct);
        --lumo-secondary-text-color: var(--lumo-contrast-70pct);
        --lumo-tertiary-text-color: var(--lumo-contrast-50pct);
        --lumo-disabled-text-color: var(--lumo-contrast-30pct);

        /* Primary */
        --lumo-primary-color: hsl(214, 86%, 55%);
        --lumo-primary-color-50pct: hsla(214, 86%, 55%, 0.5);
        --lumo-primary-color-10pct: hsla(214, 90%, 63%, 0.1);
        --lumo-primary-text-color: hsl(214, 100%, 70%);
        --lumo-primary-contrast-color: #FFF;

        /* Error */
        --lumo-error-color: hsl(3, 90%, 63%);
        --lumo-error-color-50pct: hsla(3, 90%, 63%, 0.5);
        --lumo-error-color-10pct: hsla(3, 90%, 63%, 0.1);
        --lumo-error-text-color: hsl(3, 100%, 67%);

        /* Success */
        --lumo-success-color: hsl(145, 65%, 42%);
        --lumo-success-color-50pct: hsla(145, 65%, 42%, 0.5);
        --lumo-success-color-10pct: hsla(145, 65%, 42%, 0.1);
        --lumo-success-text-color: hsl(145, 85%, 47%);
      }

      html {
        color: var(--lumo-body-text-color);
        background-color: var(--lumo-base-color);
      }

      [theme~="dark"] {
        color: var(--lumo-body-text-color);
        background-color: var(--lumo-base-color);
      }

      h1,
      h2,
      h3,
      h4,
      h5,
      h6 {
        color: var(--lumo-header-text-color);
      }

      a {
        color: var(--lumo-primary-text-color);
      }

      blockquote {
        color: var(--lumo-secondary-text-color);
      }

      code,
      pre {
        background-color: var(--lumo-contrast-10pct);
        border-radius: var(--lumo-border-radius-m);
      }
    </style>
  </template>
</dom-module><dom-module id="lumo-color-legacy">
  <template>
    <style include="lumo-color">
      :host {
        color: var(--lumo-body-text-color) !important;
        background-color: var(--lumo-base-color) !important;
      }
    </style>
  </template>
</dom-module>`;

    document.head.appendChild($_documentContainer$5.content);

    const $_documentContainer$6 = document.createElement('template');

    $_documentContainer$6.innerHTML = `<custom-style>
  <style>
    @font-face {
      font-family: 'lumo-icons';
      src: url(data:application/font-woff;charset=utf-8;base64,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) format('woff');
      font-weight: normal;
      font-style: normal;
    }

    html {
      --lumo-icons-align-center: "\\ea01";
      --lumo-icons-align-left: "\\ea02";
      --lumo-icons-align-right: "\\ea03";
      --lumo-icons-angle-down: "\\ea04";
      --lumo-icons-angle-left: "\\ea05";
      --lumo-icons-angle-right: "\\ea06";
      --lumo-icons-angle-up: "\\ea07";
      --lumo-icons-arrow-down: "\\ea08";
      --lumo-icons-arrow-left: "\\ea09";
      --lumo-icons-arrow-right: "\\ea0a";
      --lumo-icons-arrow-up: "\\ea0b";
      --lumo-icons-bar-chart: "\\ea0c";
      --lumo-icons-bell: "\\ea0d";
      --lumo-icons-calendar: "\\ea0e";
      --lumo-icons-checkmark: "\\ea0f";
      --lumo-icons-chevron-down: "\\ea10";
      --lumo-icons-chevron-left: "\\ea11";
      --lumo-icons-chevron-right: "\\ea12";
      --lumo-icons-chevron-up: "\\ea13";
      --lumo-icons-clock: "\\ea14";
      --lumo-icons-cog: "\\ea15";
      --lumo-icons-cross: "\\ea16";
      --lumo-icons-download: "\\ea17";
      --lumo-icons-dropdown: "\\ea18";
      --lumo-icons-edit: "\\ea19";
      --lumo-icons-error: "\\ea1a";
      --lumo-icons-eye: "\\ea1b";
      --lumo-icons-eye-disabled: "\\ea1c";
      --lumo-icons-menu: "\\ea1d";
      --lumo-icons-minus: "\\ea1e";
      --lumo-icons-ordered-list: "\\ea1f";
      --lumo-icons-phone: "\\ea20";
      --lumo-icons-photo: "\\ea21";
      --lumo-icons-play: "\\ea22";
      --lumo-icons-plus: "\\ea23";
      --lumo-icons-redo: "\\ea24";
      --lumo-icons-reload: "\\ea25";
      --lumo-icons-search: "\\ea26";
      --lumo-icons-undo: "\\ea27";
      --lumo-icons-unordered-list: "\\ea28";
      --lumo-icons-upload: "\\ea29";
      --lumo-icons-user: "\\ea2a";
    }
  </style>
</custom-style>`;

    document.head.appendChild($_documentContainer$6.content);

    const $_documentContainer$7 = document.createElement('template');

    $_documentContainer$7.innerHTML = `<custom-style>
  <style>
    html {
      --lumo-size-xs: 1.625rem;
      --lumo-size-s: 1.875rem;
      --lumo-size-m: 2.25rem;
      --lumo-size-l: 2.75rem;
      --lumo-size-xl: 3.5rem;

      /* Icons */
      --lumo-icon-size-s: 1.25em;
      --lumo-icon-size-m: 1.5em;
      --lumo-icon-size-l: 2.25em;
      /* For backwards compatibility */
      --lumo-icon-size: var(--lumo-icon-size-m);
    }
  </style>
</custom-style>`;

    document.head.appendChild($_documentContainer$7.content);

    const $_documentContainer$8 = document.createElement('template');

    $_documentContainer$8.innerHTML = `<custom-style>
  <style>
    html {
      /* Square */
      --lumo-space-xs: 0.25rem;
      --lumo-space-s: 0.5rem;
      --lumo-space-m: 1rem;
      --lumo-space-l: 1.5rem;
      --lumo-space-xl: 2.5rem;

      /* Wide */
      --lumo-space-wide-xs: calc(var(--lumo-space-xs) / 2) var(--lumo-space-xs);
      --lumo-space-wide-s: calc(var(--lumo-space-s) / 2) var(--lumo-space-s);
      --lumo-space-wide-m: calc(var(--lumo-space-m) / 2) var(--lumo-space-m);
      --lumo-space-wide-l: calc(var(--lumo-space-l) / 2) var(--lumo-space-l);
      --lumo-space-wide-xl: calc(var(--lumo-space-xl) / 2) var(--lumo-space-xl);

      /* Tall */
      --lumo-space-tall-xs: var(--lumo-space-xs) calc(var(--lumo-space-xs) / 2);
      --lumo-space-tall-s: var(--lumo-space-s) calc(var(--lumo-space-s) / 2);
      --lumo-space-tall-m: var(--lumo-space-m) calc(var(--lumo-space-m) / 2);
      --lumo-space-tall-l: var(--lumo-space-l) calc(var(--lumo-space-l) / 2);
      --lumo-space-tall-xl: var(--lumo-space-xl) calc(var(--lumo-space-xl) / 2);
    }
  </style>
</custom-style>`;

    document.head.appendChild($_documentContainer$8.content);

    const $_documentContainer$9 = document.createElement('template');

    $_documentContainer$9.innerHTML = `<custom-style>
  <style>
    html {
      /* Border radius */
      --lumo-border-radius-s: 0.25em; /* Checkbox, badge, date-picker year indicator, etc */
      --lumo-border-radius-m: var(--lumo-border-radius, 0.25em); /* Button, text field, menu overlay, etc */
      --lumo-border-radius-l: 0.5em; /* Dialog, notification, etc */
      --lumo-border-radius: 0.25em; /* Deprecated */

      /* Shadow */
      --lumo-box-shadow-xs: 0 1px 4px -1px var(--lumo-shade-50pct);
      --lumo-box-shadow-s: 0 2px 4px -1px var(--lumo-shade-20pct), 0 3px 12px -1px var(--lumo-shade-30pct);
      --lumo-box-shadow-m: 0 2px 6px -1px var(--lumo-shade-20pct), 0 8px 24px -4px var(--lumo-shade-40pct);
      --lumo-box-shadow-l: 0 3px 18px -2px var(--lumo-shade-20pct), 0 12px 48px -6px var(--lumo-shade-40pct);
      --lumo-box-shadow-xl: 0 4px 24px -3px var(--lumo-shade-20pct), 0 18px 64px -8px var(--lumo-shade-40pct);

      /* Clickable element cursor */
      --lumo-clickable-cursor: default;
    }
  </style>
</custom-style>`;

    document.head.appendChild($_documentContainer$9.content);

    const $_documentContainer$a = document.createElement('template');

    $_documentContainer$a.innerHTML = `<custom-style>
  <style>
    html {
      /* Font families */
      --lumo-font-family: -apple-system, BlinkMacSystemFont, "Roboto", "Segoe UI", Helvetica, Arial, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol";

      /* Font sizes */
      --lumo-font-size-xxs: .75rem;
      --lumo-font-size-xs: .8125rem;
      --lumo-font-size-s: .875rem;
      --lumo-font-size-m: 1rem;
      --lumo-font-size-l: 1.125rem;
      --lumo-font-size-xl: 1.375rem;
      --lumo-font-size-xxl: 1.75rem;
      --lumo-font-size-xxxl: 2.5rem;

      /* Line heights */
      --lumo-line-height-xs: 1.25;
      --lumo-line-height-s: 1.375;
      --lumo-line-height-m: 1.625;
    }

  </style>
</custom-style><dom-module id="lumo-typography">
  <template>
    <style>
      html {
        font-family: var(--lumo-font-family);
        font-size: var(--lumo-font-size, var(--lumo-font-size-m));
        line-height: var(--lumo-line-height-m);
        -webkit-text-size-adjust: 100%;
        -webkit-font-smoothing: antialiased;
        -moz-osx-font-smoothing: grayscale;
      }

      /* Can’t combine with the above selector because that doesn’t work in browsers without native shadow dom */
      :host {
        font-family: var(--lumo-font-family);
        font-size: var(--lumo-font-size, var(--lumo-font-size-m));
        line-height: var(--lumo-line-height-m);
        -webkit-text-size-adjust: 100%;
        -webkit-font-smoothing: antialiased;
        -moz-osx-font-smoothing: grayscale;
      }

      small,
      [theme~="font-size-s"] {
        font-size: var(--lumo-font-size-s);
        line-height: var(--lumo-line-height-s);
      }

      [theme~="font-size-xs"] {
        font-size: var(--lumo-font-size-xs);
        line-height: var(--lumo-line-height-xs);
      }

      h1,
      h2,
      h3,
      h4,
      h5,
      h6 {
        font-weight: 600;
        line-height: var(--lumo-line-height-xs);
        margin-top: 1.25em;
      }

      h1 {
        font-size: var(--lumo-font-size-xxxl);
        margin-bottom: 0.75em;
      }

      h2 {
        font-size: var(--lumo-font-size-xxl);
        margin-bottom: 0.5em;
      }

      h3 {
        font-size: var(--lumo-font-size-xl);
        margin-bottom: 0.5em;
      }

      h4 {
        font-size: var(--lumo-font-size-l);
        margin-bottom: 0.5em;
      }

      h5 {
        font-size: var(--lumo-font-size-m);
        margin-bottom: 0.25em;
      }

      h6 {
        font-size: var(--lumo-font-size-xs);
        margin-bottom: 0;
        text-transform: uppercase;
        letter-spacing: 0.03em;
      }

      p,
      blockquote {
        margin-top: 0.5em;
        margin-bottom: 0.75em;
      }

      a {
        text-decoration: none;
      }

      a:hover {
        text-decoration: underline;
      }

      hr {
        display: block;
        align-self: stretch;
        height: 1px;
        border: 0;
        padding: 0;
        margin: var(--lumo-space-s) calc(var(--lumo-border-radius-m) / 2);
        background-color: var(--lumo-contrast-10pct);
      }

      blockquote {
        border-left: 2px solid var(--lumo-contrast-30pct);
      }

      b,
      strong {
        font-weight: 600;
      }

      /* RTL specific styles */

      blockquote[dir="rtl"] {
        border-left: none;
        border-right: 2px solid var(--lumo-contrast-30pct);
      }

    </style>
  </template>
</dom-module>`;

    document.head.appendChild($_documentContainer$a.content);

    const $_documentContainer$b = html`<dom-module id="lumo-checkbox" theme-for="vaadin-checkbox">
  <template>
    <style include="lumo-checkbox-style lumo-checkbox-effects">
      /* IE11 only */
      ::-ms-backdrop,
      [part="checkbox"] {
        line-height: 1;
      }
    </style>
  </template>
</dom-module><dom-module id="lumo-checkbox-style">
  <template>
    <style>
      :host {
        -webkit-tap-highlight-color: transparent;
        -webkit-user-select: none;
        -moz-user-select: none;
        -ms-user-select: none;
        user-select: none;
        cursor: default;
        outline: none;
      }

      [part="label"]:not([empty]) {
        margin: 0.1875em 0.875em 0.1875em 0.375em;
      }

      [part="checkbox"] {
        width: calc(1em + 2px);
        height: calc(1em + 2px);
        margin: 0.1875em;
        position: relative;
        border-radius: var(--lumo-border-radius-s);
        background-color: var(--lumo-contrast-20pct);
        transition: transform 0.2s cubic-bezier(.12, .32, .54, 2), background-color 0.15s;
        pointer-events: none;
        line-height: 1.2;
      }

      :host([indeterminate]) [part="checkbox"],
      :host([checked]) [part="checkbox"] {
        background-color: var(--lumo-primary-color);
      }

      /* Needed to align the checkbox nicely on the baseline */
      [part="checkbox"]::before {
        content: "\\2003";
      }

      /* Checkmark */
      [part="checkbox"]::after {
        content: "";
        display: inline-block;
        width: 0;
        height: 0;
        border: 0 solid var(--lumo-primary-contrast-color);
        border-width: 0.1875em 0 0 0.1875em;
        box-sizing: border-box;
        transform-origin: 0 0;
        position: absolute;
        top: 0.8125em;
        left: 0.5em;
        transform: scale(0.55) rotate(-135deg);
        opacity: 0;
      }

      :host([checked]) [part="checkbox"]::after {
        opacity: 1;
        width: 0.625em;
        height: 1.0625em;
      }

      /* Indeterminate checkmark */

      :host([indeterminate]) [part="checkbox"]::after {
        transform: none;
        opacity: 1;
        top: 45%;
        height: 10%;
        left: 22%;
        right: 22%;
        width: auto;
        border: 0;
        background-color: var(--lumo-primary-contrast-color);
        transition: opacity 0.25s;
      }

      /* Focus ring */

      :host([focus-ring]) [part="checkbox"] {
        box-shadow: 0 0 0 3px var(--lumo-primary-color-50pct);
      }

      /* Disabled */

      :host([disabled]) {
        pointer-events: none;
        color: var(--lumo-disabled-text-color);
      }

      :host([disabled]) [part="label"] ::slotted(*) {
        color: inherit;
      }

      :host([disabled]) [part="checkbox"] {
        background-color: var(--lumo-contrast-10pct);
      }

      :host([disabled]) [part="checkbox"]::after {
        border-color: var(--lumo-contrast-30pct);
      }

      :host([indeterminate][disabled]) [part="checkbox"]::after {
        background-color: var(--lumo-contrast-30pct);
      }

      /* RTL specific styles */

      :host([dir="rtl"]) [part="label"]:not([empty]) {
        margin: 0.1875em 0.375em 0.1875em 0.875em;
      }
    </style>
  </template>
</dom-module><dom-module id="lumo-checkbox-effects">
  <template>
    <style>
      /* Transition the checkmark if activated with the mouse (disabled for grid select-all this way) */
      :host(:hover) [part="checkbox"]::after {
        transition: width 0.1s, height 0.25s;
      }

      /* Used for activation "halo" */
      [part="checkbox"]::before {
        color: transparent;
        display: inline-block;
        width: 100%;
        height: 100%;
        border-radius: inherit;
        background-color: inherit;
        transform: scale(1.4);
        opacity: 0;
        transition: transform 0.1s, opacity 0.8s;
      }

      /* Hover */

      :host(:not([checked]):not([indeterminate]):not([disabled]):hover) [part="checkbox"] {
        background-color: var(--lumo-contrast-30pct);
      }

      /* Disable hover for touch devices */
      @media (pointer: coarse) {
        :host(:not([checked]):not([indeterminate]):not([disabled]):hover) [part="checkbox"] {
          background-color: var(--lumo-contrast-20pct);
        }
      }

      /* Active */

      :host([active]) [part="checkbox"] {
        transform: scale(0.9);
        transition-duration: 0.05s;
      }

      :host([active][checked]) [part="checkbox"] {
        transform: scale(1.1);
      }

      :host([active]:not([checked])) [part="checkbox"]::before {
        transition-duration: 0.01s, 0.01s;
        transform: scale(0);
        opacity: 0.4;
      }
    </style>
  </template>
</dom-module>`;

    document.head.appendChild($_documentContainer$b.content);

    /**
     * @polymerMixin
     */
    const ThemePropertyMixin = superClass => class VaadinThemePropertyMixin extends superClass {
      static get properties() {
        return {
          /**
           * Helper property with theme attribute value facilitating propagation
           * in shadow DOM.
           *
           * Enables the component implementation to propagate the `theme`
           * attribute value to the subcomponents in Shadow DOM by binding
           * the subcomponent’s "theme" attribute to the `theme` property of
           * the host.
           *
           * **NOTE:** Extending the mixin only provides the property for binding,
           * and does not make the propagation alone.
           *
           * See [Theme Attribute and Subcomponents](https://github.com/vaadin/vaadin-themable-mixin/wiki/5.-Theme-Attribute-and-Subcomponents).
           * page for more information.
           *
           * @protected
           */
          theme: {
            type: String,
            readOnly: true
          }
        };
      }

      /** @protected */
      attributeChangedCallback(name, oldValue, newValue) {
        super.attributeChangedCallback(name, oldValue, newValue);

        if (name === 'theme') {
          this._setTheme(newValue);
        }
      }
    };

    /**
     * @polymerMixin
     * @mixes ThemePropertyMixin
     */
    const ThemableMixin = superClass => class VaadinThemableMixin extends ThemePropertyMixin(superClass) {

      /** @protected */
      static finalize() {
        super.finalize();

        const template = this.prototype._template;

        const hasOwnTemplate = this.template && this.template.parentElement && this.template.parentElement.id === this.is;
        const inheritedTemplate = Object.getPrototypeOf(this.prototype)._template;
        if (inheritedTemplate && !hasOwnTemplate) {
          // The element doesn't define its own template -> include the theme modules from the inherited template
          Array.from(inheritedTemplate.content.querySelectorAll('style[include]')).forEach(s => {
            this._includeStyle(s.getAttribute('include'), template);
          });
        }

        this._includeMatchingThemes(template);
      }

      /** @private */
      static _includeMatchingThemes(template) {
        const domModule = DomModule;
        const modules = domModule.prototype.modules;

        let hasThemes = false;
        const defaultModuleName = this.is + '-default-theme';

        Object.keys(modules)
          .sort((moduleNameA, moduleNameB) => {
            const vaadinA = moduleNameA.indexOf('vaadin-') === 0;
            const vaadinB = moduleNameB.indexOf('vaadin-') === 0;

            const vaadinThemePrefixes = ['lumo-', 'material-'];
            const vaadinThemeA = vaadinThemePrefixes.filter(prefix => moduleNameA.indexOf(prefix) === 0).length > 0;
            const vaadinThemeB = vaadinThemePrefixes.filter(prefix => moduleNameB.indexOf(prefix) === 0).length > 0;

            if (vaadinA !== vaadinB) {
              // Include vaadin core styles first
              return vaadinA ? -1 : 1;
            } else if (vaadinThemeA !== vaadinThemeB) {
              // Include vaadin theme styles after that
              return vaadinThemeA ? -1 : 1;
            } else {
              // Lastly include custom styles so they override all vaadin styles
              return 0;
            }
          })
          .forEach(moduleName => {
            if (moduleName !== defaultModuleName) {
              const themeFor = modules[moduleName].getAttribute('theme-for');
              if (themeFor) {
                themeFor.split(' ').forEach(themeForToken => {
                  if (new RegExp('^' + themeForToken.split('*').join('.*') + '$').test(this.is)) {
                    hasThemes = true;
                    this._includeStyle(moduleName, template);
                  }
                });
              }
            }
          });

        if (!hasThemes && modules[defaultModuleName]) {
          // No theme modules found, include the default module if it exists
          this._includeStyle(defaultModuleName, template);
        }
      }

      /** @private */
      static _includeStyle(moduleName, template) {
        if (template && !template.content.querySelector(`style[include="${moduleName}"]`)) {
          const styleEl = document.createElement('style');
          styleEl.setAttribute('include', moduleName);
          template.content.appendChild(styleEl);
        }
      }

    };

    /**
    @license
    Copyright (c) 2017 Vaadin Ltd.
    This program is available under Apache License Version 2.0, available at https://vaadin.com/license/
    */
    /**
     * A private mixin to avoid problems with dynamic properties and Polymer Analyzer.
     * No need to expose these properties in the API docs.
     * @polymerMixin
     * @private
     */
    const TabIndexMixin = superClass => class VaadinTabIndexMixin extends superClass {
      static get properties() {
        var properties = {
          /**
           * Internal property needed to listen to `tabindex` attribute changes.
           *
           * For changing the tabindex of this component use the native `tabIndex` property.
           * @private
           */
          tabindex: {
            type: Number,
            value: 0,
            reflectToAttribute: true,
            observer: '_tabindexChanged'
          }
        };

        if (window.ShadyDOM) {
          // ShadyDOM browsers need the `tabIndex` in order to notify when the user changes it programmatically.
          properties['tabIndex'] = properties.tabindex;
        }

        return properties;
      }
    };

    /**
     * Polymer.IronControlState is not a proper 2.0 class, also, its tabindex
     * implementation fails in the shadow dom, so we have this for vaadin elements.
     * @polymerMixin
     */
    const ControlStateMixin = superClass => class VaadinControlStateMixin extends TabIndexMixin(superClass) {
      static get properties() {
        return {
          /**
           * Specify that this control should have input focus when the page loads.
           */
          autofocus: {
            type: Boolean
          },

          /**
           * Stores the previous value of tabindex attribute of the disabled element
           * @private
           */
          _previousTabIndex: {
            type: Number
          },

          /**
           * If true, the user cannot interact with this element.
           */
          disabled: {
            type: Boolean,
            observer: '_disabledChanged',
            reflectToAttribute: true
          },

          /**
           * @private
           */
          _isShiftTabbing: {
            type: Boolean
          }
        };
      }

      /**
       * @protected
       */
      ready() {
        this.addEventListener('focusin', e => {
          if (e.composedPath()[0] === this) {
            // Only focus if the focus is received from somewhere outside
            if (!this.contains(e.relatedTarget)) {
              this._focus();
            }
          } else if (e.composedPath().indexOf(this.focusElement) !== -1 && !this.disabled) {
            this._setFocused(true);
          }
        });
        this.addEventListener('focusout', e => this._setFocused(false));

        // In super.ready() other 'focusin' and 'focusout' listeners might be
        // added, so we call it after our own ones to ensure they execute first.
        // Issue to watch out: when incorrect, <vaadin-combo-box> refocuses the
        // input field on iOS after “Done” is pressed.
        super.ready();

        // This fixes the bug in Firefox 61 (https://bugzilla.mozilla.org/show_bug.cgi?id=1472887)
        // where focusout event does not go out of shady DOM because composed property in the event is not true
        const ensureEventComposed = e => {
          if (!e.composed) {
            e.target.dispatchEvent(new CustomEvent(e.type, {
              bubbles: true,
              composed: true,
              cancelable: false
            }));
          }
        };
        this.shadowRoot.addEventListener('focusin', ensureEventComposed);
        this.shadowRoot.addEventListener('focusout', ensureEventComposed);

        this.addEventListener('keydown', e => {
          if (!e.defaultPrevented && e.keyCode === 9) {
            if (e.shiftKey) {
              // Flag is checked in _focus event handler.
              this._isShiftTabbing = true;
              HTMLElement.prototype.focus.apply(this);
              this._setFocused(false);
              // Event handling in IE is asynchronous and the flag is removed asynchronously as well
              setTimeout(() => this._isShiftTabbing = false, 0);
            } else {
              // Workaround for FF63-65 bug that causes the focus to get lost when
              // blurring a slotted component with focusable shadow root content
              // https://bugzilla.mozilla.org/show_bug.cgi?id=1528686
              // TODO: Remove when safe
              const firefox = window.navigator.userAgent.match(/Firefox\/(\d\d\.\d)/);
              if (firefox
                && parseFloat(firefox[1]) >= 63
                && parseFloat(firefox[1]) < 66
                && this.parentNode
                && this.nextSibling) {
                const fakeTarget = document.createElement('input');
                fakeTarget.style.position = 'absolute';
                fakeTarget.style.opacity = '0';
                fakeTarget.tabIndex = this.tabIndex;

                this.parentNode.insertBefore(fakeTarget, this.nextSibling);
                fakeTarget.focus();
                fakeTarget.addEventListener('focusout', () => this.parentNode.removeChild(fakeTarget));
              }
            }

          }
        });

        if (this.autofocus && !this.disabled) {
          window.requestAnimationFrame(() => {
            this._focus();
            this._setFocused(true);
            this.setAttribute('focus-ring', '');
          });
        }

        this._boundKeydownListener = this._bodyKeydownListener.bind(this);
        this._boundKeyupListener = this._bodyKeyupListener.bind(this);
      }

      /**
       * @protected
       */
      connectedCallback() {
        super.connectedCallback();

        document.body.addEventListener('keydown', this._boundKeydownListener, true);
        document.body.addEventListener('keyup', this._boundKeyupListener, true);
      }

      /**
       * @protected
       */
      disconnectedCallback() {
        super.disconnectedCallback();

        document.body.removeEventListener('keydown', this._boundKeydownListener, true);
        document.body.removeEventListener('keyup', this._boundKeyupListener, true);

        // in non-Chrome browsers, blur does not fire on the element when it is disconnected.
        // reproducible in `<vaadin-date-picker>` when closing on `Cancel` or `Today` click.
        if (this.hasAttribute('focused')) {
          this._setFocused(false);
        }
      }

      /**
       * @param {boolean} focused
       * @protected
       */
      _setFocused(focused) {
        if (focused) {
          this.setAttribute('focused', '');
        } else {
          this.removeAttribute('focused');
        }

        // focus-ring is true when the element was focused from the keyboard.
        // Focus Ring [A11ycasts]: https://youtu.be/ilj2P5-5CjI
        if (focused && this._tabPressed) {
          this.setAttribute('focus-ring', '');
        } else {
          this.removeAttribute('focus-ring');
        }
      }

      /**
       * @param {KeyboardEvent} e
       * @private
       */
      _bodyKeydownListener(e) {
        this._tabPressed = e.keyCode === 9;
      }

      /**
       * @private
       */
      _bodyKeyupListener() {
        this._tabPressed = false;
      }

      /**
       * Any element extending this mixin is required to implement this getter.
       * It returns the actual focusable element in the component.
       * @return {Element | null | undefined}
       */
      get focusElement() {
        window.console.warn(`Please implement the 'focusElement' property in <${this.localName}>`);
        return this;
      }

      /**
       * @protected
       */
      _focus() {
        if (!this.focusElement || this._isShiftTabbing) {
          return;
        }

        this.focusElement.focus();
        this._setFocused(true);
      }

      /**
       * Moving the focus from the host element causes firing of the blur event what leads to problems in IE.
       * @private
       */
      focus() {
        if (!this.focusElement || this.disabled) {
          return;
        }

        this.focusElement.focus();
        this._setFocused(true);
      }

      /**
       * Native bluring in the host element does nothing because it does not have the focus.
       * In chrome it works, but not in FF.
       * @private
       */
      blur() {
        if (!this.focusElement) {
          return;
        }
        this.focusElement.blur();
        this._setFocused(false);
      }

      /**
       * @param {boolean} disabled
       * @private
       */
      _disabledChanged(disabled) {
        this.focusElement.disabled = disabled;
        if (disabled) {
          this.blur();
          this._previousTabIndex = this.tabindex;
          this.tabindex = -1;
          this.setAttribute('aria-disabled', 'true');
        } else {
          if (typeof this._previousTabIndex !== 'undefined') {
            this.tabindex = this._previousTabIndex;
          }
          this.removeAttribute('aria-disabled');
        }
      }

      /**
       * @param {number | null | undefined} tabindex
       * @private
       */
      _tabindexChanged(tabindex) {
        if (tabindex !== undefined) {
          this.focusElement.tabIndex = tabindex;
        }

        if (this.disabled && this.tabindex) {
          // If tabindex attribute was changed while checkbox was disabled
          if (this.tabindex !== -1) {
            this._previousTabIndex = this.tabindex;
          }
          this.tabindex = tabindex = undefined;
        }

        if (window.ShadyDOM) {
          this.setProperties({tabIndex: tabindex, tabindex: tabindex});
        }
      }

      /**
       * @protected
       */
      click() {
        if (!this.disabled) {
          super.click();
        }
      }
    };

    /**
    @license
    Copyright (c) 2020 Vaadin Ltd.
    This program is available under Apache License Version 2.0, available at https://vaadin.com/license/
    */
    /**
     * Helper that provides a set of functions for RTL.
     */
    class DirHelper {
      /**
       * Get the scroll type in the current browser view.
       *
       * @return {string} the scroll type. Possible values are `default|reverse|negative`
       */
      static detectScrollType() {
        const dummy = document.createElement('div');
        dummy.textContent = 'ABCD';
        dummy.dir = 'rtl';
        dummy.style.fontSize = '14px';
        dummy.style.width = '4px';
        dummy.style.height = '1px';
        dummy.style.position = 'absolute';
        dummy.style.top = '-1000px';
        dummy.style.overflow = 'scroll';
        document.body.appendChild(dummy);

        let cachedType = 'reverse';
        if (dummy.scrollLeft > 0) {
          cachedType = 'default';
        } else {
          dummy.scrollLeft = 1;
          if (dummy.scrollLeft === 0) {
            cachedType = 'negative';
          }
        }
        document.body.removeChild(dummy);
        return cachedType;
      }

      /**
       * Get the scrollLeft value of the element relative to the direction
       *
       * @param {string} scrollType type of the scroll detected with `detectScrollType`
       * @param {string} direction current direction of the element
       * @param {Element} element
       * @return {number} the scrollLeft value.
      */
      static getNormalizedScrollLeft(scrollType, direction, element) {
        const {scrollLeft} = element;
        if (direction !== 'rtl' || !scrollType) {
          return scrollLeft;
        }

        switch (scrollType) {
          case 'negative':
            return element.scrollWidth - element.clientWidth + scrollLeft;
          case 'reverse':
            return element.scrollWidth - element.clientWidth - scrollLeft;
        }
        return scrollLeft;
      }

      /**
       * Set the scrollLeft value of the element relative to the direction
       *
       * @param {string} scrollType type of the scroll detected with `detectScrollType`
       * @param {string} direction current direction of the element
       * @param {Element} element
       * @param {number} scrollLeft the scrollLeft value to be set
       */
      static setNormalizedScrollLeft(scrollType, direction, element, scrollLeft) {
        if (direction !== 'rtl' || !scrollType) {
          element.scrollLeft = scrollLeft;
          return;
        }

        switch (scrollType) {
          case 'negative':
            element.scrollLeft = element.clientWidth - element.scrollWidth + scrollLeft;
            break;
          case 'reverse':
            element.scrollLeft = element.scrollWidth - element.clientWidth - scrollLeft;
            break;
          default:
            element.scrollLeft = scrollLeft;
            break;
        }
      }
    }

    /**
     * Array of Vaadin custom element classes that have been subscribed to the dir changes.
     */
    const directionSubscribers = [];
    const directionUpdater = function() {
      const documentDir = getDocumentDir();
      directionSubscribers.forEach(element => {
        alignDirs(element, documentDir);
      });
    };

    let scrollType;

    const directionObserver = new MutationObserver(directionUpdater);
    directionObserver.observe(document.documentElement, {attributes: true, attributeFilter: ['dir']});

    const alignDirs = function(element, documentDir) {
      if (documentDir) {
        element.setAttribute('dir', documentDir);
      } else {
        element.removeAttribute('dir');
      }
    };

    const getDocumentDir = function() {
      return document.documentElement.getAttribute('dir');
    };

    /**
     * @polymerMixin
     */
    const DirMixin$1 = superClass => class VaadinDirMixin extends superClass {
      static get properties() {
        return {
          /**
           * @protected
           */
          dir: {
            type: String,
            readOnly: true
          }
        };
      }

      /** @protected */
      static finalize() {
        super.finalize();

        if (!scrollType) {
          scrollType = DirHelper.detectScrollType();
        }
      }

      /** @protected */
      connectedCallback() {
        super.connectedCallback();

        if (!this.hasAttribute('dir')) {
          this.__subscribe();
          alignDirs(this, getDocumentDir());
        }
      }

      /** @protected */
      attributeChangedCallback(name, oldValue, newValue) {
        super.attributeChangedCallback(name, oldValue, newValue);
        if (name !== 'dir') {
          return;
        }

        // New value equals to the document direction and the element is not subscribed to the changes
        const newValueEqlDocDir = newValue === getDocumentDir() && directionSubscribers.indexOf(this) === -1;
        // Value was emptied and the element is not subscribed to the changes
        const newValueEmptied = !newValue && oldValue && directionSubscribers.indexOf(this) === -1;
        // New value is different and the old equals to document direction and the element is not subscribed to the changes
        const newDiffValue = newValue !== getDocumentDir() && oldValue === getDocumentDir();

        if (newValueEqlDocDir || newValueEmptied) {
          this.__subscribe();
          alignDirs(this, getDocumentDir());
        } else if (newDiffValue) {
          this.__subscribe(false);
        }
      }

      /** @protected */
      disconnectedCallback() {
        super.disconnectedCallback();
        this.__subscribe(false);
        this.removeAttribute('dir');
      }

      /** @private */
      __subscribe(push = true) {
        if (push) {
          directionSubscribers.indexOf(this) === -1 &&
            directionSubscribers.push(this);
        } else {
          directionSubscribers.indexOf(this) > -1 &&
            directionSubscribers.splice(directionSubscribers.indexOf(this), 1);
        }
      }

      /**
       * @param {Element} element
       * @return {number}
       * @protected
       */
      __getNormalizedScrollLeft(element) {
        return DirHelper.getNormalizedScrollLeft(scrollType, this.getAttribute('dir') || 'ltr', element);
      }

      /**
       * @param {Element} element
       * @param {number} scrollLeft
       * @protected
       */
      __setNormalizedScrollLeft(element, scrollLeft) {
        return DirHelper.setNormalizedScrollLeft(scrollType, this.getAttribute('dir') || 'ltr', element, scrollLeft);
      }
    };

    const DEV_MODE_CODE_REGEXP =
      /\/\*\*\s+vaadin-dev-mode:start([\s\S]*)vaadin-dev-mode:end\s+\*\*\//i;

    const FlowClients = window.Vaadin && window.Vaadin.Flow && window.Vaadin.Flow.clients;

    function isMinified() {
      function test() {
        /** vaadin-dev-mode:start
        return false;
        vaadin-dev-mode:end **/
        return true;
      }
      return uncommentAndRun(test);
    }

    function isDevelopmentMode() {
      try {
        if (isForcedDevelopmentMode()) {
          return true;
        }

        if (!isLocalhost()) {
          return false;
        }

        if (FlowClients) {
          return !isFlowProductionMode();
        }

        return !isMinified();
      } catch (e) {
        // Some error in this code, assume production so no further actions will be taken
        return false;
      }
    }

    function isForcedDevelopmentMode() {
      return localStorage.getItem("vaadin.developmentmode.force");
    }

    function isLocalhost() {
      return (["localhost","127.0.0.1"].indexOf(window.location.hostname) >= 0);
    }

    function isFlowProductionMode() {
      if (FlowClients) {
        const productionModeApps = Object.keys(FlowClients)
          .map(key => FlowClients[key])
          .filter(client => client.productionMode);
        if (productionModeApps.length > 0) {
          return true;
        }
      }
      return false;
    }

    function uncommentAndRun(callback, args) {
      if (typeof callback !== 'function') {
        return;
      }

      const match = DEV_MODE_CODE_REGEXP.exec(callback.toString());
      if (match) {
        try {
          // requires CSP: script-src 'unsafe-eval'
          callback = new Function(match[1]);
        } catch (e) {
          // eat the exception
          console.log('vaadin-development-mode-detector: uncommentAndRun() failed', e);
        }
      }

      return callback(args);
    }

    // A guard against polymer-modulizer removing the window.Vaadin
    // initialization above.
    window['Vaadin'] = window['Vaadin'] || {};

    /**
     * Inspects the source code of the given `callback` function for
     * specially-marked _commented_ code. If such commented code is found in the
     * callback source, uncomments and runs that code instead of the callback
     * itself. Otherwise runs the callback as is.
     *
     * The optional arguments are passed into the callback / uncommented code,
     * the result is returned.
     *
     * See the `isMinified()` function source code in this file for an example.
     *
     */
    const runIfDevelopmentMode = function(callback, args) {
      if (window.Vaadin.developmentMode) {
        return uncommentAndRun(callback, args);
      }
    };

    if (window.Vaadin.developmentMode === undefined) {
      window.Vaadin.developmentMode = isDevelopmentMode();
    }

    /* This file is autogenerated from src/vaadin-usage-statistics.tpl.html */

    function maybeGatherAndSendStats() {
      /** vaadin-dev-mode:start
      (function () {
    'use strict';

    var _typeof = typeof Symbol === "function" && typeof Symbol.iterator === "symbol" ? function (obj) {
      return typeof obj;
    } : function (obj) {
      return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj;
    };

    var classCallCheck = function (instance, Constructor) {
      if (!(instance instanceof Constructor)) {
        throw new TypeError("Cannot call a class as a function");
      }
    };

    var createClass = function () {
      function defineProperties(target, props) {
        for (var i = 0; i < props.length; i++) {
          var descriptor = props[i];
          descriptor.enumerable = descriptor.enumerable || false;
          descriptor.configurable = true;
          if ("value" in descriptor) descriptor.writable = true;
          Object.defineProperty(target, descriptor.key, descriptor);
        }
      }

      return function (Constructor, protoProps, staticProps) {
        if (protoProps) defineProperties(Constructor.prototype, protoProps);
        if (staticProps) defineProperties(Constructor, staticProps);
        return Constructor;
      };
    }();

    var getPolymerVersion = function getPolymerVersion() {
      return window.Polymer && window.Polymer.version;
    };

    var StatisticsGatherer = function () {
      function StatisticsGatherer(logger) {
        classCallCheck(this, StatisticsGatherer);

        this.now = new Date().getTime();
        this.logger = logger;
      }

      createClass(StatisticsGatherer, [{
        key: 'frameworkVersionDetectors',
        value: function frameworkVersionDetectors() {
          return {
            'Flow': function Flow() {
              if (window.Vaadin && window.Vaadin.Flow && window.Vaadin.Flow.clients) {
                var flowVersions = Object.keys(window.Vaadin.Flow.clients).map(function (key) {
                  return window.Vaadin.Flow.clients[key];
                }).filter(function (client) {
                  return client.getVersionInfo;
                }).map(function (client) {
                  return client.getVersionInfo().flow;
                });
                if (flowVersions.length > 0) {
                  return flowVersions[0];
                }
              }
            },
            'Vaadin Framework': function VaadinFramework() {
              if (window.vaadin && window.vaadin.clients) {
                var frameworkVersions = Object.values(window.vaadin.clients).filter(function (client) {
                  return client.getVersionInfo;
                }).map(function (client) {
                  return client.getVersionInfo().vaadinVersion;
                });
                if (frameworkVersions.length > 0) {
                  return frameworkVersions[0];
                }
              }
            },
            'AngularJs': function AngularJs() {
              if (window.angular && window.angular.version && window.angular.version) {
                return window.angular.version.full;
              }
            },
            'Angular': function Angular() {
              if (window.ng) {
                var tags = document.querySelectorAll("[ng-version]");
                if (tags.length > 0) {
                  return tags[0].getAttribute("ng-version");
                }
                return "Unknown";
              }
            },
            'Backbone.js': function BackboneJs() {
              if (window.Backbone) {
                return window.Backbone.VERSION;
              }
            },
            'React': function React() {
              var reactSelector = '[data-reactroot], [data-reactid]';
              if (!!document.querySelector(reactSelector)) {
                // React does not publish the version by default
                return "unknown";
              }
            },
            'Ember': function Ember() {
              if (window.Em && window.Em.VERSION) {
                return window.Em.VERSION;
              } else if (window.Ember && window.Ember.VERSION) {
                return window.Ember.VERSION;
              }
            },
            'jQuery': function (_jQuery) {
              function jQuery() {
                return _jQuery.apply(this, arguments);
              }

              jQuery.toString = function () {
                return _jQuery.toString();
              };

              return jQuery;
            }(function () {
              if (typeof jQuery === 'function' && jQuery.prototype.jquery !== undefined) {
                return jQuery.prototype.jquery;
              }
            }),
            'Polymer': function Polymer() {
              var version = getPolymerVersion();
              if (version) {
                return version;
              }
            },
            'LitElement': function LitElement() {
              var version = window.litElementVersions && window.litElementVersions[0];
              if (version) {
                return version;
              }
            },
            'LitHtml': function LitHtml() {
              var version = window.litHtmlVersions && window.litHtmlVersions[0];
              if (version) {
                return version;
              }
            },
            'Vue.js': function VueJs() {
              if (window.Vue) {
                return window.Vue.version;
              }
            }
          };
        }
      }, {
        key: 'getUsedVaadinElements',
        value: function getUsedVaadinElements(elements) {
          var version = getPolymerVersion();
          var elementClasses = void 0;
          // NOTE: In case you edit the code here, YOU MUST UPDATE any statistics reporting code in Flow.
          // Check all locations calling the method getEntries() in
          // https://github.com/vaadin/flow/blob/master/flow-server/src/main/java/com/vaadin/flow/internal/UsageStatistics.java#L106
          // Currently it is only used by BootstrapHandler.
          if (version && version.indexOf('2') === 0) {
            // Polymer 2: components classes are stored in window.Vaadin
            elementClasses = Object.keys(window.Vaadin).map(function (c) {
              return window.Vaadin[c];
            }).filter(function (c) {
              return c.is;
            });
          } else {
            // Polymer 3: components classes are stored in window.Vaadin.registrations
            elementClasses = window.Vaadin.registrations || [];
          }
          elementClasses.forEach(function (klass) {
            var version = klass.version ? klass.version : "0.0.0";
            elements[klass.is] = { version: version };
          });
        }
      }, {
        key: 'getUsedVaadinThemes',
        value: function getUsedVaadinThemes(themes) {
          ['Lumo', 'Material'].forEach(function (themeName) {
            var theme;
            var version = getPolymerVersion();
            if (version && version.indexOf('2') === 0) {
              // Polymer 2: themes are stored in window.Vaadin
              theme = window.Vaadin[themeName];
            } else {
              // Polymer 3: themes are stored in custom element registry
              theme = customElements.get('vaadin-' + themeName.toLowerCase() + '-styles');
            }
            if (theme && theme.version) {
              themes[themeName] = { version: theme.version };
            }
          });
        }
      }, {
        key: 'getFrameworks',
        value: function getFrameworks(frameworks) {
          var detectors = this.frameworkVersionDetectors();
          Object.keys(detectors).forEach(function (framework) {
            var detector = detectors[framework];
            try {
              var version = detector();
              if (version) {
                frameworks[framework] = { version: version };
              }
            } catch (e) {}
          });
        }
      }, {
        key: 'gather',
        value: function gather(storage) {
          var storedStats = storage.read();
          var gatheredStats = {};
          var types = ["elements", "frameworks", "themes"];

          types.forEach(function (type) {
            gatheredStats[type] = {};
            if (!storedStats[type]) {
              storedStats[type] = {};
            }
          });

          var previousStats = JSON.stringify(storedStats);

          this.getUsedVaadinElements(gatheredStats.elements);
          this.getFrameworks(gatheredStats.frameworks);
          this.getUsedVaadinThemes(gatheredStats.themes);

          var now = this.now;
          types.forEach(function (type) {
            var keys = Object.keys(gatheredStats[type]);
            keys.forEach(function (key) {
              if (!storedStats[type][key] || _typeof(storedStats[type][key]) != _typeof({})) {
                storedStats[type][key] = { firstUsed: now };
              }
              // Discards any previously logged version number
              storedStats[type][key].version = gatheredStats[type][key].version;
              storedStats[type][key].lastUsed = now;
            });
          });

          var newStats = JSON.stringify(storedStats);
          storage.write(newStats);
          if (newStats != previousStats && Object.keys(storedStats).length > 0) {
            this.logger.debug("New stats: " + newStats);
          }
        }
      }]);
      return StatisticsGatherer;
    }();

    var StatisticsStorage = function () {
      function StatisticsStorage(key) {
        classCallCheck(this, StatisticsStorage);

        this.key = key;
      }

      createClass(StatisticsStorage, [{
        key: 'read',
        value: function read() {
          var localStorageStatsString = localStorage.getItem(this.key);
          try {
            return JSON.parse(localStorageStatsString ? localStorageStatsString : '{}');
          } catch (e) {
            return {};
          }
        }
      }, {
        key: 'write',
        value: function write(data) {
          localStorage.setItem(this.key, data);
        }
      }, {
        key: 'clear',
        value: function clear() {
          localStorage.removeItem(this.key);
        }
      }, {
        key: 'isEmpty',
        value: function isEmpty() {
          var storedStats = this.read();
          var empty = true;
          Object.keys(storedStats).forEach(function (key) {
            if (Object.keys(storedStats[key]).length > 0) {
              empty = false;
            }
          });

          return empty;
        }
      }]);
      return StatisticsStorage;
    }();

    var StatisticsSender = function () {
      function StatisticsSender(url, logger) {
        classCallCheck(this, StatisticsSender);

        this.url = url;
        this.logger = logger;
      }

      createClass(StatisticsSender, [{
        key: 'send',
        value: function send(data, errorHandler) {
          var logger = this.logger;

          if (navigator.onLine === false) {
            logger.debug("Offline, can't send");
            errorHandler();
            return;
          }
          logger.debug("Sending data to " + this.url);

          var req = new XMLHttpRequest();
          req.withCredentials = true;
          req.addEventListener("load", function () {
            // Stats sent, nothing more to do
            logger.debug("Response: " + req.responseText);
          });
          req.addEventListener("error", function () {
            logger.debug("Send failed");
            errorHandler();
          });
          req.addEventListener("abort", function () {
            logger.debug("Send aborted");
            errorHandler();
          });
          req.open("POST", this.url);
          req.setRequestHeader("Content-Type", "application/json");
          req.send(data);
        }
      }]);
      return StatisticsSender;
    }();

    var StatisticsLogger = function () {
      function StatisticsLogger(id) {
        classCallCheck(this, StatisticsLogger);

        this.id = id;
      }

      createClass(StatisticsLogger, [{
        key: '_isDebug',
        value: function _isDebug() {
          return localStorage.getItem("vaadin." + this.id + ".debug");
        }
      }, {
        key: 'debug',
        value: function debug(msg) {
          if (this._isDebug()) {
            console.info(this.id + ": " + msg);
          }
        }
      }]);
      return StatisticsLogger;
    }();

    var UsageStatistics = function () {
      function UsageStatistics() {
        classCallCheck(this, UsageStatistics);

        this.now = new Date();
        this.timeNow = this.now.getTime();
        this.gatherDelay = 10; // Delay between loading this file and gathering stats
        this.initialDelay = 24 * 60 * 60;

        this.logger = new StatisticsLogger("statistics");
        this.storage = new StatisticsStorage("vaadin.statistics.basket");
        this.gatherer = new StatisticsGatherer(this.logger);
        this.sender = new StatisticsSender("https://tools.vaadin.com/usage-stats/submit", this.logger);
      }

      createClass(UsageStatistics, [{
        key: 'maybeGatherAndSend',
        value: function maybeGatherAndSend() {
          var _this = this;

          if (localStorage.getItem(UsageStatistics.optOutKey)) {
            return;
          }
          this.gatherer.gather(this.storage);
          setTimeout(function () {
            _this.maybeSend();
          }, this.gatherDelay * 1000);
        }
      }, {
        key: 'lottery',
        value: function lottery() {
          return true;
        }
      }, {
        key: 'currentMonth',
        value: function currentMonth() {
          return this.now.getYear() * 12 + this.now.getMonth();
        }
      }, {
        key: 'maybeSend',
        value: function maybeSend() {
          var firstUse = Number(localStorage.getItem(UsageStatistics.firstUseKey));
          var monthProcessed = Number(localStorage.getItem(UsageStatistics.monthProcessedKey));

          if (!firstUse) {
            // Use a grace period to avoid interfering with tests, incognito mode etc
            firstUse = this.timeNow;
            localStorage.setItem(UsageStatistics.firstUseKey, firstUse);
          }

          if (this.timeNow < firstUse + this.initialDelay * 1000) {
            this.logger.debug("No statistics will be sent until the initial delay of " + this.initialDelay + "s has passed");
            return;
          }
          if (this.currentMonth() <= monthProcessed) {
            this.logger.debug("This month has already been processed");
            return;
          }
          localStorage.setItem(UsageStatistics.monthProcessedKey, this.currentMonth());
          // Use random sampling
          if (this.lottery()) {
            this.logger.debug("Congratulations, we have a winner!");
          } else {
            this.logger.debug("Sorry, no stats from you this time");
            return;
          }

          this.send();
        }
      }, {
        key: 'send',
        value: function send() {
          // Ensure we have the latest data
          this.gatherer.gather(this.storage);

          // Read, send and clean up
          var data = this.storage.read();
          data["firstUse"] = Number(localStorage.getItem(UsageStatistics.firstUseKey));
          data["usageStatisticsVersion"] = UsageStatistics.version;
          var info = 'This request contains usage statistics gathered from the application running in development mode. \n\nStatistics gathering is automatically disabled and excluded from production builds.\n\nFor details and to opt-out, see https://github.com/vaadin/vaadin-usage-statistics.\n\n\n\n';
          var self = this;
          this.sender.send(info + JSON.stringify(data), function () {
            // Revert the 'month processed' flag
            localStorage.setItem(UsageStatistics.monthProcessedKey, self.currentMonth() - 1);
          });
        }
      }], [{
        key: 'version',
        get: function get$1() {
          return '2.1.0';
        }
      }, {
        key: 'firstUseKey',
        get: function get$1() {
          return 'vaadin.statistics.firstuse';
        }
      }, {
        key: 'monthProcessedKey',
        get: function get$1() {
          return 'vaadin.statistics.monthProcessed';
        }
      }, {
        key: 'optOutKey',
        get: function get$1() {
          return 'vaadin.statistics.optout';
        }
      }]);
      return UsageStatistics;
    }();

    try {
      window.Vaadin = window.Vaadin || {};
      window.Vaadin.usageStatsChecker = window.Vaadin.usageStatsChecker || new UsageStatistics();
      window.Vaadin.usageStatsChecker.maybeGatherAndSend();
    } catch (e) {
      // Intentionally ignored as this is not a problem in the app being developed
    }

    }());

      vaadin-dev-mode:end **/
    }

    const usageStatistics = function() {
      if (typeof runIfDevelopmentMode === 'function') {
        return runIfDevelopmentMode(maybeGatherAndSendStats);
      }
    };

    if (!window.Vaadin) {
      window['Vaadin'] = {};
    }

    /**
     * Array of Vaadin custom element classes that have been finalized.
     */
    window['Vaadin'].registrations = window.Vaadin.registrations || [];

    // Use the hack to prevent polymer-modulizer from converting to exports
    window['Vaadin'].developmentModeCallback = window.Vaadin.developmentModeCallback || {};
    window['Vaadin'].developmentModeCallback['vaadin-usage-statistics'] = function() {
      if (usageStatistics) {
        usageStatistics();
      }
    };

    let statsJob;

    const registered = new Set();

    /**
     * @polymerMixin
     * @mixes DirMixin
     */
    const ElementMixin$1 = superClass => class VaadinElementMixin extends DirMixin$1(superClass) {
      /** @protected */
      static finalize() {
        super.finalize();

        const {is} = this;

        // Registers a class prototype for telemetry purposes.
        if (is && !registered.has(is)) {
          window.Vaadin.registrations.push(this);
          registered.add(is);

          if (window.Vaadin.developmentModeCallback) {
            statsJob = Debouncer.debounce(statsJob,
              idlePeriod, () => {
                window.Vaadin.developmentModeCallback['vaadin-usage-statistics']();
              }
            );
            enqueueDebouncer(statsJob);
          }
        }
      }

      constructor() {
        super();
        if (document.doctype === null) {
          console.warn(
            'Vaadin components require the "standards mode" declaration. Please add <!DOCTYPE html> to the HTML document.'
          );
        }
      }
    };

    /**
    @license
    Copyright (c) 2017 Vaadin Ltd.
    This program is available under Apache License Version 2.0, available at https://vaadin.com/license/
    */
    /**
     * `<vaadin-checkbox>` is a Web Component for customized checkboxes.
     *
     * ```html
     * <vaadin-checkbox>
     *   Make my profile visible
     * </vaadin-checkbox>
     * ```
     *
     * ### Styling
     *
     * The following shadow DOM parts are available for styling:
     *
     * Part name         | Description
     * ------------------|----------------
     * `checkbox`        | The wrapper element for the native <input type="checkbox">
     * `label`           | The wrapper element in which the component's children, namely the label, is slotted
     *
     * The following state attributes are available for styling:
     *
     * Attribute    | Description | Part name
     * -------------|-------------|--------------
     * `active`     | Set when the checkbox is pressed down, either with mouse, touch or the keyboard. | `:host`
     * `disabled`   | Set when the checkbox is disabled. | `:host`
     * `focus-ring` | Set when the checkbox is focused using the keyboard. | `:host`
     * `focused`    | Set when the checkbox is focused. | `:host`
     * `indeterminate` | Set when the checkbox is in indeterminate mode. | `:host`
     * `checked` | Set when the checkbox is checked. | `:host`
     * `empty` | Set when there is no label provided. | `label`
     *
     * See [ThemableMixin – how to apply styles for shadow parts](https://github.com/vaadin/vaadin-themable-mixin/wiki)
     *
     * @extends PolymerElement
     * @mixes ElementMixin
     * @mixes ControlStateMixin
     * @mixes ThemableMixin
     * @mixes GestureEventListeners
     * @demo demo/index.html
     */
    class CheckboxElement extends
      ElementMixin$1(
        ControlStateMixin(
          ThemableMixin(
            GestureEventListeners(PolymerElement)))) {
      static get template() {
        return html`
    <style>
      :host {
        display: inline-block;
      }

      :host([hidden]) {
        display: none !important;
      }

      label {
        display: inline-flex;
        align-items: baseline;
        outline: none;
      }

      [part="checkbox"] {
        position: relative;
        display: inline-block;
        flex: none;
      }

      input[type="checkbox"] {
        position: absolute;
        top: 0;
        left: 0;
        right: 0;
        width: 100%;
        height: 100%;
        opacity: 0;
        cursor: inherit;
        margin: 0;
      }

      :host([disabled]) {
        -webkit-tap-highlight-color: transparent;
      }
    </style>

    <label>
      <span part="checkbox">
        <input type="checkbox" checked="{{checked::change}}" disabled\$="[[disabled]]" indeterminate="{{indeterminate::change}}" role="presentation" tabindex="-1">
      </span>

      <span part="label">
        <slot></slot>
      </span>
    </label>
`;
      }

      static get is() {
        return 'vaadin-checkbox';
      }

      static get version() {
        return '2.3.0';
      }

      static get properties() {
        return {
          /**
           * True if the checkbox is checked.
           */
          checked: {
            type: Boolean,
            value: false,
            notify: true,
            observer: '_checkedChanged',
            reflectToAttribute: true
          },

          /**
           * Indeterminate state of the checkbox when it's neither checked nor unchecked, but undetermined.
           * https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input/checkbox#Indeterminate_state_checkboxes
           */
          indeterminate: {
            type: Boolean,
            notify: true,
            observer: '_indeterminateChanged',
            reflectToAttribute: true,
            value: false
          },

          /**
           * The value given to the data submitted with the checkbox's name to the server when the control is inside a form.
           */
          value: {
            type: String,
            value: 'on'
          },

          _nativeCheckbox: {
            type: Object
          }
        };
      }

      constructor() {
        super();
        /**
         * @type {string}
         * Name of the element.
         */
        this.name;
      }

      get name() {
        return this.checked ? this._storedName : '';
      }

      set name(name) {
        this._storedName = name;
      }

      ready() {
        super.ready();

        this.setAttribute('role', 'checkbox');

        this._nativeCheckbox = this.shadowRoot.querySelector('input[type="checkbox"]');

        this.addEventListener('click', this._handleClick.bind(this));

        this._addActiveListeners();

        const attrName = this.getAttribute('name');
        if (attrName) {
          this.name = attrName;
        }

        this.shadowRoot.querySelector('[part~="label"]').querySelector('slot')
          .addEventListener('slotchange', this._updateLabelAttribute.bind(this));

        this._updateLabelAttribute();
      }

      _updateLabelAttribute() {
        const label = this.shadowRoot.querySelector('[part~="label"]');
        const assignedNodes = label.firstElementChild.assignedNodes();
        if (this._isAssignedNodesEmpty(assignedNodes)) {
          label.setAttribute('empty', '');
        } else {
          label.removeAttribute('empty');
        }
      }

      _isAssignedNodesEmpty(nodes) {
        // The assigned nodes considered to be empty if there is no slotted content or only one empty text node
        return nodes.length === 0 ||
              (nodes.length == 1
              && nodes[0].nodeType == Node.TEXT_NODE
              && nodes[0].textContent.trim() === '');
      }

      _checkedChanged(checked) {
        if (this.indeterminate) {
          this.setAttribute('aria-checked', 'mixed');
        } else {
          this.setAttribute('aria-checked', Boolean(checked));
        }
      }

      _indeterminateChanged(indeterminate) {
        if (indeterminate) {
          this.setAttribute('aria-checked', 'mixed');
        } else {
          this.setAttribute('aria-checked', this.checked);
        }
      }

      _addActiveListeners() {
        // DOWN
        this._addEventListenerToNode(this, 'down', (e) => {
          if (this.__interactionsAllowed(e)) {
            this.setAttribute('active', '');
          }
        });

        // UP
        this._addEventListenerToNode(this, 'up', () => this.removeAttribute('active'));

        // KEYDOWN
        this.addEventListener('keydown', e => {
          if (this.__interactionsAllowed(e) && e.keyCode === 32) {
            e.preventDefault();
            this.setAttribute('active', '');
          }
        });

        // KEYUP
        this.addEventListener('keyup', e => {
          if (this.__interactionsAllowed(e) && e.keyCode === 32) {
            e.preventDefault();
            this._toggleChecked();
            this.removeAttribute('active');

            if (this.indeterminate) {
              this.indeterminate = false;
            }
          }
        });
      }

      /** @protected */
      get focusElement() {
        return this.shadowRoot.querySelector('input');
      }

      /**
       * True if users' interactions (mouse or keyboard)
       * should toggle the checkbox
       */
      __interactionsAllowed(e) {
        if (this.disabled) {
          return false;
        }

        // https://github.com/vaadin/vaadin-checkbox/issues/63
        if (e.target.localName === 'a') {
          return false;
        }

        return true;
      }

      _handleClick(e) {
        if (this.__interactionsAllowed(e)) {
          if (!this.indeterminate) {
            if (e.composedPath()[0] !== this._nativeCheckbox) {
              e.preventDefault();
              this._toggleChecked();
            }
          } else {
            /*
             * Required for IE 11 and Edge.
             * See issue here: https://developer.microsoft.com/en-us/microsoft-edge/platform/issues/7344418/
             */
            this.indeterminate = false;
            e.preventDefault();
            this._toggleChecked();
          }
        }
      }

      _toggleChecked() {
        this.checked = !this.checked;
        this.dispatchEvent(new CustomEvent('change', {composed: false, bubbles: true}));
      }

      /**
       * Fired when the user commits a value change.
       *
       * @event change
       */
    }

    customElements.define(CheckboxElement.is, CheckboxElement);

    const $_documentContainer$c = html`<dom-module id="lumo-grid" theme-for="vaadin-grid">
  <template>
    <style>
      :host {
        font-family: var(--lumo-font-family);
        font-size: var(--lumo-font-size-m);
        line-height: var(--lumo-line-height-s);
        color: var(--lumo-body-text-color);
        background-color: var(--lumo-base-color);
        box-sizing: border-box;
        -webkit-text-size-adjust: 100%;
        -webkit-tap-highlight-color: transparent;
        -webkit-font-smoothing: antialiased;
        -moz-osx-font-smoothing: grayscale;

        /* For internal use only */
        --_lumo-grid-border-color: var(--lumo-contrast-20pct);
        --_lumo-grid-secondary-border-color: var(--lumo-contrast-10pct);
        --_lumo-grid-border-width: 1px;
        --_lumo-grid-selected-row-color: var(--lumo-primary-color-10pct);
      }

      /* No (outer) border */

      :host(:not([theme~="no-border"])) {
        border: var(--_lumo-grid-border-width) solid var(--_lumo-grid-border-color);
      }

      /* Cell styles */

      [part~="cell"] {
        min-height: var(--lumo-size-m);
        background-color: var(--lumo-base-color);
      }

      [part~="cell"] ::slotted(vaadin-grid-cell-content) {
        cursor: default;
        padding: var(--lumo-space-xs) var(--lumo-space-m);
      }

      /* Apply row borders by default and introduce the "no-row-borders" variant */
      :host(:not([theme~="no-row-borders"])) [part~="cell"]:not([part~="details-cell"]) {
        border-top: var(--_lumo-grid-border-width) solid var(--_lumo-grid-secondary-border-color);
      }

      /* Hide first body row top border */
      :host(:not([theme~="no-row-borders"])) [part="row"][first] [part~="cell"]:not([part~="details-cell"]) {
        border-top: 0;
        min-height: calc(var(--lumo-size-m) - var(--_lumo-grid-border-width));
      }

      /* Focus-ring */

      [part~="cell"]:focus {
        outline: none;
      }

      :host([navigating]) [part~="cell"]:focus::before {
        content: "";
        position: absolute;
        top: 0;
        right: 0;
        bottom: 0;
        left: 0;
        pointer-events: none;
        box-shadow: inset 0 0 0 2px var(--lumo-primary-color-50pct);
      }

      /* Drag and Drop styles */
      :host([dragover])::after {
        content: "";
        position: absolute;
        z-index: 100;
        top: 0;
        right: 0;
        bottom: 0;
        left: 0;
        pointer-events: none;
        box-shadow: inset 0 0 0 2px var(--lumo-primary-color-50pct);
      }

      [part~="row"][dragover] {
        z-index: 100 !important;
      }

      [part~="row"][dragover] [part~="cell"] {
        overflow: visible;
      }

      [part~="row"][dragover] [part~="cell"]::after {
        content: "";
        position: absolute;
        top: 0;
        right: 0;
        bottom: 0;
        left: 0;
        height: calc(var(--_lumo-grid-border-width) + 2px);
        pointer-events: none;
        background: var(--lumo-primary-color-50pct);
      }

      :host([theme~="no-row-borders"]) [dragover] [part~="cell"]::after {
        height: 2px;
      }

      [part~="row"][dragover="below"] [part~="cell"]::after {
        top: 100%;
        bottom: auto;
        margin-top: -1px;
      }

      [part~="row"][dragover="above"] [part~="cell"]::after {
        top: auto;
        bottom: 100%;
        margin-bottom: -1px;
      }

      [part~="row"][details-opened][dragover="below"] [part~="cell"]:not([part~="details-cell"])::after,
      [part~="row"][details-opened][dragover="above"] [part~="details-cell"]::after {
        display: none;
      }

      [part~="row"][dragover][dragover="on-top"] [part~="cell"]::after {
        height: 100%;
      }

      [part~="row"][dragstart] {
        /* Add bottom-space to the row so the drag number doesn't get clipped. Needed for IE/Edge */
        border-bottom: 100px solid transparent;
        z-index: 100 !important;
        opacity: 0.9;
      }

      [part~="row"][dragstart] [part~="cell"] {
        border: none !important;
        box-shadow: none !important;
      }

      [part~="row"][dragstart] [part~="cell"][last-column] {
        border-radius: 0 var(--lumo-border-radius-s) var(--lumo-border-radius-s) 0;
      }

      [part~="row"][dragstart] [part~="cell"][first-column] {
        border-radius: var(--lumo-border-radius-s) 0 0 var(--lumo-border-radius-s);
      }

      [ios] [part~="row"][dragstart] [part~="cell"] {
        background: var(--lumo-primary-color-50pct);
      }

      #scroller:not([ios]) [part~="row"][dragstart]:not([dragstart=""])::after {
        display: block;
        position: absolute;
        left: var(--_grid-drag-start-x);
        top: var(--_grid-drag-start-y);
        z-index: 100;
        content: attr(dragstart);
        align-items: center;
        justify-content: center;
        box-sizing: border-box;
        padding: calc(var(--lumo-space-xs) * 0.8);
        color: var(--lumo-error-contrast-color);
        background-color: var(--lumo-error-color);
        border-radius: var(--lumo-border-radius-m);
        font-family: var(--lumo-font-family);
        font-size: var(--lumo-font-size-xxs);
        line-height: 1;
        font-weight: 500;
        text-transform: initial;
        letter-spacing: initial;
        min-width: calc(var(--lumo-size-s) * 0.7);
        text-align: center;
      }

      /* Headers and footers */

      [part~="header-cell"] ::slotted(vaadin-grid-cell-content),
      [part~="footer-cell"] ::slotted(vaadin-grid-cell-content),
      [part~="reorder-ghost"] {
        font-size: var(--lumo-font-size-s);
        font-weight: 500;
      }

      [part~="footer-cell"] ::slotted(vaadin-grid-cell-content) {
        font-weight: 400;
      }

      [part="row"]:only-child [part~="header-cell"] {
        min-height: var(--lumo-size-xl);
      }

      /* Header borders */

      /* Hide first header row top border */
      :host(:not([theme~="no-row-borders"])) [part="row"]:first-child [part~="header-cell"] {
        border-top: 0;
      }

      [part="row"]:last-child [part~="header-cell"] {
        border-bottom: var(--_lumo-grid-border-width) solid transparent;
      }

      :host(:not([theme~="no-row-borders"])) [part="row"]:last-child [part~="header-cell"] {
        border-bottom-color: var(--_lumo-grid-secondary-border-color);
      }

      /* Overflow uses a stronger border color */
      :host([overflow~="top"]) [part="row"]:last-child [part~="header-cell"] {
        border-bottom-color: var(--_lumo-grid-border-color);
      }

      /* Footer borders */

      [part="row"]:first-child [part~="footer-cell"] {
        border-top: var(--_lumo-grid-border-width) solid transparent;
      }

      :host(:not([theme~="no-row-borders"])) [part="row"]:first-child [part~="footer-cell"] {
        border-top-color: var(--_lumo-grid-secondary-border-color);
      }

      /* Overflow uses a stronger border color */
      :host([overflow~="bottom"]) [part="row"]:first-child [part~="footer-cell"] {
        border-top-color: var(--_lumo-grid-border-color);
      }

      /* Column reordering */

      :host([reordering]) [part~="cell"] {
        background: linear-gradient(var(--lumo-shade-20pct), var(--lumo-shade-20pct)) var(--lumo-base-color);
      }

      :host([reordering]) [part~="cell"][reorder-status="allowed"] {
        background: var(--lumo-base-color);
      }

      :host([reordering]) [part~="cell"][reorder-status="dragging"] {
        background: linear-gradient(var(--lumo-contrast-5pct), var(--lumo-contrast-5pct)) var(--lumo-base-color);
      }

      [part~="reorder-ghost"] {
        opacity: 0.85;
        box-shadow: var(--lumo-box-shadow-s);
        /* TODO Use the same styles as for the cell element (reorder-ghost copies styles from the cell element) */
        padding: var(--lumo-space-s) var(--lumo-space-m) !important;
      }

      /* Column resizing */

      [part="resize-handle"] {
        width: 3px;
        background-color: var(--lumo-primary-color-50pct);
        opacity: 0;
        transition: opacity 0.2s;
      }

      :host(:not([reordering])) *:not([column-resizing]) [part~="cell"]:hover [part="resize-handle"],
      [part="resize-handle"]:active {
        opacity: 1;
        transition-delay: 0.15s;
      }

      /* Column borders */

      :host([theme~="column-borders"]) [part~="cell"]:not([last-column]):not([part~="details-cell"]) {
        border-right: var(--_lumo-grid-border-width) solid var(--_lumo-grid-secondary-border-color);
      }

      /* Frozen columns */

      [last-frozen] {
        border-right: var(--_lumo-grid-border-width) solid transparent;
        overflow: hidden;
      }

      :host([overflow~="left"]) [part~="cell"][last-frozen]:not([part~="details-cell"]) {
        border-right-color: var(--_lumo-grid-border-color);
      }

      /* Row stripes */

      :host([theme~="row-stripes"]) [part~="row"]:not([odd]) [part~="body-cell"],
      :host([theme~="row-stripes"]) [part~="row"]:not([odd]) [part~="details-cell"] {
        background-image: linear-gradient(var(--lumo-contrast-5pct), var(--lumo-contrast-5pct));
        background-repeat: repeat-x;
      }

      /* Selected row */

      /* Raise the selected rows above unselected rows (so that box-shadow can cover unselected rows) */
      :host(:not([reordering])) [part~="row"][selected] {
        z-index: 1;
      }

      :host(:not([reordering])) [part~="row"][selected] [part~="body-cell"]:not([part~="details-cell"]) {
        background-image: linear-gradient(var(--_lumo-grid-selected-row-color), var(--_lumo-grid-selected-row-color));
        background-repeat: repeat;
      }

      /* Cover the border of an unselected row */
      :host(:not([theme~="no-row-borders"])) [part~="row"][selected] [part~="cell"]:not([part~="details-cell"]) {
        box-shadow: 0 var(--_lumo-grid-border-width) 0 0 var(--_lumo-grid-selected-row-color);
      }

      /* Compact */

      :host([theme~="compact"]) [part="row"]:only-child [part~="header-cell"] {
        min-height: var(--lumo-size-m);
      }

      :host([theme~="compact"]) [part~="cell"] {
        min-height: var(--lumo-size-s);
      }

      :host([theme~="compact"]) [part="row"][first] [part~="cell"]:not([part~="details-cell"]) {
        min-height: calc(var(--lumo-size-s) - var(--_lumo-grid-border-width));
      }

      :host([theme~="compact"]) [part~="cell"] ::slotted(vaadin-grid-cell-content) {
        padding: var(--lumo-space-xs) var(--lumo-space-s);
      }

      /* Wrap cell contents */

      :host([theme~="wrap-cell-content"]) [part~="cell"] ::slotted(vaadin-grid-cell-content) {
        white-space: normal;
      }

      /* RTL specific styles */

      :host([dir="rtl"]) [part~="row"][dragstart] [part~="cell"][last-column] {
        border-radius: var(--lumo-border-radius-s) 0 0 var(--lumo-border-radius-s);
      }

      :host([dir="rtl"]) [part~="row"][dragstart] [part~="cell"][first-column] {
        border-radius: 0 var(--lumo-border-radius-s) var(--lumo-border-radius-s) 0;
      }

      :host([dir="rtl"][theme~="column-borders"]) [part~="cell"]:not([last-column]):not([part~="details-cell"]) {
        border-right: none;
        border-left: var(--_lumo-grid-border-width) solid var(--_lumo-grid-secondary-border-color);
      }

      :host([dir="rtl"]) [last-frozen] {
        border-right: none;
        border-left: var(--_lumo-grid-border-width) solid transparent;
      }

      :host([dir="rtl"][overflow~="right"]) [part~="cell"][last-frozen]:not([part~="details-cell"]) {
        border-left-color: var(--_lumo-grid-border-color);
      }
    </style>
  </template>
</dom-module><dom-module theme-for="vaadin-checkbox" id="vaadin-grid-select-all-checkbox-lumo">
  <template>
    <style>
      :host(.vaadin-grid-select-all-checkbox) {
        font-size: var(--lumo-font-size-m);
      }
   </style>
  </template>
</dom-module>`;

    document.head.appendChild($_documentContainer$c.content);

    /**
    @license
    Copyright (c) 2016 The Polymer Project Authors. All rights reserved.
    This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt
    The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt
    The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt
    Code distributed by Google as part of the polymer project is also
    subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt
    */
    var IOS$1 = navigator.userAgent.match(/iP(?:hone|ad;(?: U;)? CPU) OS (\d+)/);
    var IOS_TOUCH_SCROLLING$1 = IOS$1 && IOS$1[1] >= 8;
    var DEFAULT_PHYSICAL_COUNT$1 = 3;
    var ANIMATION_FRAME = animationFrame;
    var IDLE_TIME = idlePeriod;
    var MICRO_TASK = microTask;

    const PolymerIronList = Class({

      behaviors: [
        IronResizableBehavior,
        IronScrollTargetBehavior
      ],

      /**
       * The ratio of hidden tiles that should remain in the scroll direction.
       * Recommended value ~0.5, so it will distribute tiles evenly in both directions.
       */
      _ratio: 0.5,

      /**
       * The padding-top value for the list.
       */
      _scrollerPaddingTop: 0,

      /**
       * This value is the same as `scrollTop`.
       */
      _scrollPosition: 0,

      /**
       * The sum of the heights of all the tiles in the DOM.
       */
      _physicalSize: 0,

      /**
       * The average `offsetHeight` of the tiles observed till now.
       */
      _physicalAverage: 0,

      /**
       * The number of tiles which `offsetHeight` > 0 observed until now.
       */
      _physicalAverageCount: 0,

      /**
       * The Y position of the item rendered in the `_physicalStart`
       * tile relative to the scrolling list.
       */
      _physicalTop: 0,

      /**
       * The number of items in the list.
       */
      _virtualCount: 0,

      /**
       * The estimated scroll height based on `_physicalAverage`
       */
      _estScrollHeight: 0,

      /**
       * The scroll height of the dom node
       */
      _scrollHeight: 0,

      /**
       * The height of the list. This is referred as the viewport in the context of list.
       */
      _viewportHeight: 0,

      /**
       * The width of the list. This is referred as the viewport in the context of list.
       */
      _viewportWidth: 0,

      /**
       * An array of DOM nodes that are currently in the tree
       * @type {?Array<!TemplatizerNode>}
       */
      _physicalItems: null,

      /**
       * An array of heights for each item in `_physicalItems`
       * @type {?Array<number>}
       */
      _physicalSizes: null,

      /**
       * A cached value for the first visible index.
       * See `firstVisibleIndex`
       * @type {?number}
       */
      _firstVisibleIndexVal: null,

      /**
       * A Polymer collection for the items.
       * @type {?Polymer.Collection}
       */
      _collection: null,

      /**
       * A cached value for the last visible index.
       * See `lastVisibleIndex`
       * @type {?number}
       */
      _lastVisibleIndexVal: null,

      /**
       * The max number of pages to render. One page is equivalent to the height of the list.
       */
      _maxPages: 2,

      /**
       * The virtual index of the focused item.
       */
      _focusedVirtualIndex: -1,

      /**
       * The maximum items per row
       */
      _itemsPerRow: 1,

      /**
       * The height of the row in grid layout.
       */
      _rowHeight: 0,

      /**
       * The cost of stamping a template in ms.
       */
      _templateCost: 0,

      /**
       * The bottom of the physical content.
       */
      get _physicalBottom() {
        return this._physicalTop + this._physicalSize;
      },

      /**
       * The bottom of the scroll.
       */
      get _scrollBottom() {
        return this._scrollPosition + this._viewportHeight;
      },

      /**
       * The n-th item rendered in the last physical item.
       */
      get _virtualEnd() {
        return this._virtualStart + this._physicalCount - 1;
      },

      /**
       * The height of the physical content that isn't on the screen.
       */
      get _hiddenContentSize() {
        var size = this.grid ? this._physicalRows * this._rowHeight : this._physicalSize;
        return size - this._viewportHeight;
      },

      /**
       * The maximum scroll top value.
       */
      get _maxScrollTop() {
        return this._estScrollHeight - this._viewportHeight + this._scrollOffset;
      },

      /**
       * The largest n-th value for an item such that it can be rendered in `_physicalStart`.
       */
      get _maxVirtualStart() {
        var virtualCount = this._convertIndexToCompleteRow(this._virtualCount);
        return Math.max(0, virtualCount - this._physicalCount);
      },

      set _virtualStart(val) {
        val = this._clamp(val, 0, this._maxVirtualStart);
        if (this.grid) {
          val = val - (val % this._itemsPerRow);
        }
        this._virtualStartVal = val;
      },

      get _virtualStart() {
        return this._virtualStartVal || 0;
      },

      /**
       * The k-th tile that is at the top of the scrolling list.
       */
      set _physicalStart(val) {
        val = val % this._physicalCount;
        if (val < 0) {
          val = this._physicalCount + val;
        }
        if (this.grid) {
          val = val - (val % this._itemsPerRow);
        }
        this._physicalStartVal = val;
      },

      get _physicalStart() {
        return this._physicalStartVal || 0;
      },

      /**
       * The k-th tile that is at the bottom of the scrolling list.
       */
      get _physicalEnd() {
        return (this._physicalStart + this._physicalCount - 1) % this._physicalCount;
      },

      set _physicalCount(val) {
        this._physicalCountVal = val;
      },

      get _physicalCount() {
        return this._physicalCountVal || 0;
      },

      /**
       * An optimal physical size such that we will have enough physical items
       * to fill up the viewport and recycle when the user scrolls.
       *
       * This default value assumes that we will at least have the equivalent
       * to a viewport of physical items above and below the user's viewport.
       */
      get _optPhysicalSize() {
        return this._viewportHeight === 0 ? Infinity : this._viewportHeight * this._maxPages;
      },

      /**
       * True if the current list is visible.
       */
      get _isVisible() {
        return Boolean(this.offsetWidth || this.offsetHeight);
      },

      /**
       * Gets the index of the first visible item in the viewport.
       *
       * @type {number}
       */
      get firstVisibleIndex() {
        var idx = this._firstVisibleIndexVal;
        if (idx == null) {
          var physicalOffset = this._physicalTop + this._scrollOffset;

          idx = this._iterateItems(function(pidx, vidx) {
            physicalOffset += this._getPhysicalSizeIncrement(pidx);

            if (physicalOffset > this._scrollPosition) {
              return this.grid ? vidx - (vidx % this._itemsPerRow) : vidx;
            }
            // Handle a partially rendered final row in grid mode
            if (this.grid && this._virtualCount - 1 === vidx) {
              return vidx - (vidx % this._itemsPerRow);
            }
          }) || 0;
          this._firstVisibleIndexVal = idx;
        }
        return idx;
      },

      /**
       * Gets the index of the last visible item in the viewport.
       *
       * @type {number}
       */
      get lastVisibleIndex() {
        var idx = this._lastVisibleIndexVal;
        if (idx == null) {
          if (this.grid) {
            idx = Math.min(this._virtualCount, this.firstVisibleIndex + this._estRowsInView * this._itemsPerRow - 1);
          } else {
            var physicalOffset = this._physicalTop + this._scrollOffset;
            this._iterateItems(function(pidx, vidx) {
              if (physicalOffset < this._scrollBottom) {
                idx = vidx;
              }
              physicalOffset += this._getPhysicalSizeIncrement(pidx);
            });
          }
          this._lastVisibleIndexVal = idx;
        }
        return idx;
      },

      get _scrollOffset() {
        return this._scrollerPaddingTop;
      },

      attached: function() {
        this._debounce('_render', this._render, ANIMATION_FRAME);
        // `iron-resize` is fired when the list is attached if the event is added
        // before attached causing unnecessary work.
        this.listen(this, 'iron-resize', '_resizeHandler');
      },

      detached: function() {
        this.unlisten(this, 'iron-resize', '_resizeHandler');
      },

      /**
       * Invoke this method if you dynamically update the viewport's
       * size or CSS padding.
       *
       * @method updateViewportBoundaries
       */
      updateViewportBoundaries: function() {
        var styles = window.getComputedStyle(this);
        this._scrollerPaddingTop = this.scrollTarget === this ? 0 : parseInt(styles['padding-top'], 10);
        this._isRTL = Boolean(styles.direction === 'rtl');
        this._viewportWidth = this.$.items.offsetWidth;
        this._viewportHeight = this._scrollTargetHeight;
        this.grid && this._updateGridMetrics();
      },

      /**
       * Recycles the physical items when needed.
       */
      _scrollHandler: function() {
        var scrollTop = Math.max(0, Math.min(this._maxScrollTop, this._scrollTop));
        var delta = scrollTop - this._scrollPosition;
        var isScrollingDown = delta >= 0;
        // Track the current scroll position.
        this._scrollPosition = scrollTop;
        // Clear indexes for first and last visible indexes.
        this._firstVisibleIndexVal = null;
        this._lastVisibleIndexVal = null;
        // Random access.
        if (Math.abs(delta) > this._physicalSize && this._physicalSize > 0) {
          delta = delta - this._scrollOffset;
          var idxAdjustment = Math.round(delta / this._physicalAverage) * this._itemsPerRow;
          this._virtualStart = this._virtualStart + idxAdjustment;
          this._physicalStart = this._physicalStart + idxAdjustment;
          // Estimate new physical offset.
          this._physicalTop = Math.floor(this._virtualStart / this._itemsPerRow) * this._physicalAverage;
          this._update();
        } else if (this._physicalCount > 0) {
          var reusables = this._getReusables(isScrollingDown);
          if (isScrollingDown) {
            this._physicalTop = reusables.physicalTop;
            this._virtualStart = this._virtualStart + reusables.indexes.length;
            this._physicalStart = this._physicalStart + reusables.indexes.length;
          } else {
            this._virtualStart = this._virtualStart - reusables.indexes.length;
            this._physicalStart = this._physicalStart - reusables.indexes.length;
          }
          this._update(reusables.indexes, isScrollingDown ? null : reusables.indexes);
          this._debounce('_increasePoolIfNeeded', this._increasePoolIfNeeded.bind(this, 0), MICRO_TASK);
        }
      },

      /**
       * Returns an object that contains the indexes of the physical items
       * that might be reused and the physicalTop.
       *
       * @param {boolean} fromTop If the potential reusable items are above the scrolling region.
       */
      _getReusables: function(fromTop) {
        var ith, offsetContent, physicalItemHeight;
        var idxs = [];
        var protectedOffsetContent = this._hiddenContentSize * this._ratio;
        var virtualStart = this._virtualStart;
        var virtualEnd = this._virtualEnd;
        var physicalCount = this._physicalCount;
        var top = this._physicalTop + this._scrollOffset;
        var bottom = this._physicalBottom + this._scrollOffset;
        var scrollTop = this._scrollTop;
        var scrollBottom = this._scrollBottom;

        if (fromTop) {
          ith = this._physicalStart;
          offsetContent = scrollTop - top;
        } else {
          ith = this._physicalEnd;
          offsetContent = bottom - scrollBottom;
        }
        // eslint-disable-next-line no-constant-condition
        while (true) {
          physicalItemHeight = this._getPhysicalSizeIncrement(ith);
          offsetContent = offsetContent - physicalItemHeight;
          if (idxs.length >= physicalCount || offsetContent <= protectedOffsetContent) {
            break;
          }
          if (fromTop) {
            // Check that index is within the valid range.
            if (virtualEnd + idxs.length + 1 >= this._virtualCount) {
              break;
            }
            // Check that the index is not visible.
            if (top + physicalItemHeight >= scrollTop - this._scrollOffset) {
              break;
            }
            idxs.push(ith);
            top = top + physicalItemHeight;
            ith = (ith + 1) % physicalCount;
          } else {
            // Check that index is within the valid range.
            if (virtualStart - idxs.length <= 0) {
              break;
            }
            // Check that the index is not visible.
            if (top + this._physicalSize - physicalItemHeight <= scrollBottom) {
              break;
            }
            idxs.push(ith);
            top = top - physicalItemHeight;
            ith = (ith === 0) ? physicalCount - 1 : ith - 1;
          }
        }
        return {indexes: idxs, physicalTop: top - this._scrollOffset};
      },

      /**
       * Update the list of items, starting from the `_virtualStart` item.
       * @param {!Array<number>=} itemSet
       * @param {!Array<number>=} movingUp
       */
      _update: function(itemSet, movingUp) {
        if ((itemSet && itemSet.length === 0) || this._physicalCount === 0) {
          return;
        }
        this._manageFocus();
        this._assignModels(itemSet);
        this._updateMetrics(itemSet);
        // Adjust offset after measuring.
        if (movingUp) {
          while (movingUp.length) {
            var idx = movingUp.pop();
            this._physicalTop -= this._getPhysicalSizeIncrement(idx);
          }
        }
        this._positionItems();
        this._updateScrollerSize();
      },

      _isClientFull: function() {
        return this._scrollBottom != 0 && this._physicalBottom - 1 >= this._scrollBottom &&
            this._physicalTop <= this._scrollPosition;
      },

      /**
       * Increases the pool size.
       */
      _increasePoolIfNeeded: function(count) {
        var nextPhysicalCount = this._clamp(this._physicalCount + count,
          DEFAULT_PHYSICAL_COUNT$1, this._virtualCount - this._virtualStart);
        nextPhysicalCount = this._convertIndexToCompleteRow(nextPhysicalCount);
        var delta = nextPhysicalCount - this._physicalCount;
        var nextIncrease = Math.round(this._physicalCount * 0.5);

        if (delta < 0) {
          return;
        }
        if (delta > 0) {
          var ts = window.performance.now();
          // Concat arrays in place.
          [].push.apply(this._physicalItems, this._createPool(delta));
          // Push 0s into physicalSizes. Can't use Array.fill because IE11 doesn't support it.
          for (var i = 0; i < delta; i++) {
            this._physicalSizes.push(0);
          }
          this._physicalCount = this._physicalCount + delta;
          // Update the physical start if it needs to preserve the model of the focused item.
          // In this situation, the focused item is currently rendered and its model would
          // have changed after increasing the pool if the physical start remained unchanged.
          if (this._physicalStart > this._physicalEnd &&
              this._isIndexRendered(this._focusedVirtualIndex) &&
              this._getPhysicalIndex(this._focusedVirtualIndex) < this._physicalEnd) {
            this._physicalStart = this._physicalStart + delta;
          }
          this._update();
          this._templateCost = (window.performance.now() - ts) / delta;
          nextIncrease = Math.round(this._physicalCount * 0.5);
        }
        // The upper bounds is not fixed when dealing with a grid that doesn't
        // fill it's last row with the exact number of items per row.
        if (this._virtualEnd >= this._virtualCount - 1 || nextIncrease === 0) ; else if (!this._isClientFull()) {
          this._debounce(
            '_increasePoolIfNeeded',
            this._increasePoolIfNeeded.bind(
              this,
              nextIncrease
            ), MICRO_TASK);
        } else if (this._physicalSize < this._optPhysicalSize) {
          // Yield and increase the pool during idle time until the physical size is optimal.
          this._debounce(
            '_increasePoolIfNeeded',
            this._increasePoolIfNeeded.bind(
              this,
              this._clamp(Math.round(50 / this._templateCost), 1, nextIncrease)
            ), IDLE_TIME);
        }
      },

      /**
       * Renders the a new list.
       */
      _render: function() {
        if (!this.isAttached || !this._isVisible) {
          return;
        }
        if (this._physicalCount !== 0) {
          var reusables = this._getReusables(true);
          this._physicalTop = reusables.physicalTop;
          this._virtualStart = this._virtualStart + reusables.indexes.length;
          this._physicalStart = this._physicalStart + reusables.indexes.length;
          this._update(reusables.indexes);
          this._update();
          this._increasePoolIfNeeded(0);
        } else if (this._virtualCount > 0) {
          // Initial render
          this.updateViewportBoundaries();
          this._increasePoolIfNeeded(DEFAULT_PHYSICAL_COUNT$1);
        }
      },

      /**
       * Called when the items have changed. That is, reassignments
       * to `items`, splices or updates to a single item.
       */
      _itemsChanged: function(change) {
        if (change.path === 'items') {
          this._virtualStart = 0;
          this._physicalTop = 0;
          this._virtualCount = this.items ? this.items.length : 0;
          this._collection = this.items && undefined ?
            undefined.get(this.items) : null;
          this._physicalIndexForKey = {};
          this._firstVisibleIndexVal = null;
          this._lastVisibleIndexVal = null;
          this._physicalCount = this._physicalCount || 0;
          this._physicalItems = this._physicalItems || [];
          this._physicalSizes = this._physicalSizes || [];
          this._physicalStart = 0;
          if (this._scrollTop > this._scrollOffset) {
            this._resetScrollPosition(0);
          }
          this._removeFocusedItem();
          this._debounce('_render', this._render, ANIMATION_FRAME);
        }
      },

      /**
       * Executes a provided function per every physical index in `itemSet`
       * `itemSet` default value is equivalent to the entire set of physical indexes.
       *
       * @param {!function(number, number)} fn
       * @param {!Array<number>=} itemSet
       */
      _iterateItems: function(fn, itemSet) {
        var pidx, vidx, rtn, i;

        if (arguments.length === 2 && itemSet) {
          for (i = 0; i < itemSet.length; i++) {
            pidx = itemSet[i];
            vidx = this._computeVidx(pidx);
            if ((rtn = fn.call(this, pidx, vidx)) != null) {
              return rtn;
            }
          }
        } else {
          pidx = this._physicalStart;
          vidx = this._virtualStart;
          for (; pidx < this._physicalCount; pidx++, vidx++) {
            if ((rtn = fn.call(this, pidx, vidx)) != null) {
              return rtn;
            }
          }
          for (pidx = 0; pidx < this._physicalStart; pidx++, vidx++) {
            if ((rtn = fn.call(this, pidx, vidx)) != null) {
              return rtn;
            }
          }
        }
      },

      /**
       * Returns the virtual index for a given physical index
       *
       * @param {number} pidx Physical index
       * @return {number}
       */
      _computeVidx: function(pidx) {
        if (pidx >= this._physicalStart) {
          return this._virtualStart + (pidx - this._physicalStart);
        }
        return this._virtualStart + (this._physicalCount - this._physicalStart) + pidx;
      },

      /**
       * Updates the height for a given set of items.
       *
       * @param {!Array<number>=} itemSet
       */
      _updateMetrics: function(itemSet) {
        // Make sure we distributed all the physical items
        // so we can measure them.
        flush ? flush() : flush();

        var newPhysicalSize = 0;
        var oldPhysicalSize = 0;
        var prevAvgCount = this._physicalAverageCount;
        var prevPhysicalAvg = this._physicalAverage;

        this._iterateItems(function(pidx, vidx) {
          oldPhysicalSize += this._physicalSizes[pidx];
          this._physicalSizes[pidx] = this._physicalItems[pidx].offsetHeight;
          newPhysicalSize += this._physicalSizes[pidx];
          this._physicalAverageCount += this._physicalSizes[pidx] ? 1 : 0;
        }, itemSet);

        if (this.grid) {
          this._updateGridMetrics();
          this._physicalSize = Math.ceil(this._physicalCount / this._itemsPerRow) * this._rowHeight;
        } else {
          oldPhysicalSize = (this._itemsPerRow === 1) ?
            oldPhysicalSize :
            Math.ceil(this._physicalCount / this._itemsPerRow) * this._rowHeight;
          this._physicalSize = this._physicalSize + newPhysicalSize - oldPhysicalSize;
          this._itemsPerRow = 1;
        }
        // Update the average if it measured something.
        if (this._physicalAverageCount !== prevAvgCount) {
          this._physicalAverage = Math.round(
            ((prevPhysicalAvg * prevAvgCount) + newPhysicalSize) /
            this._physicalAverageCount);
        }
      },

      /**
       * Updates the position of the physical items.
       */
      _positionItems: function() {
        this._adjustScrollPosition();

        var y = this._physicalTop;

        this._iterateItems(function(pidx, vidx) {
          this.translate3d(0, y + 'px', 0, this._physicalItems[pidx]);
          y += this._physicalSizes[pidx];
        });
      },

      _getPhysicalSizeIncrement: function(pidx) {
        if (!this.grid) {
          return this._physicalSizes[pidx];
        }
        if (this._computeVidx(pidx) % this._itemsPerRow !== this._itemsPerRow - 1) {
          return 0;
        }
        return this._rowHeight;
      },

      /**
       * Adjusts the scroll position when it was overestimated.
       */
      _adjustScrollPosition: function() {
        var deltaHeight = this._virtualStart === 0 ? this._physicalTop : Math.min(this._scrollPosition + this._physicalTop, 0);
        // Note: the delta can be positive or negative.
        if (deltaHeight !== 0) {
          this._physicalTop = this._physicalTop - deltaHeight;
          var scrollTop = this._scrollTop;
          // juking scroll position during interial scrolling on iOS is no bueno
          if (!IOS_TOUCH_SCROLLING$1 && scrollTop > 0) {
            this._resetScrollPosition(scrollTop - deltaHeight);
          }
        }
      },

      /**
       * Sets the position of the scroll.
       */
      _resetScrollPosition: function(pos) {
        if (this.scrollTarget && pos >= 0) {
          this._scrollTop = pos;
          this._scrollPosition = this._scrollTop;
        }
      },

      /**
       * Sets the scroll height, that's the height of the content,
       *
       * @param {boolean=} forceUpdate If true, updates the height no matter what.
       */
      _updateScrollerSize: function(forceUpdate) {
        if (this.grid) {
          this._estScrollHeight = this._virtualRowCount * this._rowHeight;
        } else {
          this._estScrollHeight = (this._physicalBottom +
              Math.max(this._virtualCount - this._physicalCount - this._virtualStart, 0) * this._physicalAverage);
        }
        forceUpdate = forceUpdate || this._scrollHeight === 0;
        forceUpdate = forceUpdate || this._scrollPosition >= this._estScrollHeight - this._physicalSize;
        forceUpdate = forceUpdate || this.grid && this.$.items.style.height < this._estScrollHeight;
        // Amortize height adjustment, so it won't trigger large repaints too often.
        if (forceUpdate || Math.abs(this._estScrollHeight - this._scrollHeight) >= this._viewportHeight) {
          this.$.items.style.height = this._estScrollHeight + 'px';
          this._scrollHeight = this._estScrollHeight;
        }
      },

      /**
       * Scroll to a specific index in the virtual list regardless
       * of the physical items in the DOM tree.
       *
       * @method scrollToIndex
       * @param {number} idx The index of the item
       */
      scrollToIndex: function(idx) {
        if (typeof idx !== 'number' || idx < 0 || idx > this.items.length - 1) {
          return;
        }
        flush ? flush() : flush();
        // Items should have been rendered prior scrolling to an index.
        if (this._physicalCount === 0) {
          return;
        }
        idx = this._clamp(idx, 0, this._virtualCount - 1);
        // Update the virtual start only when needed.
        if (!this._isIndexRendered(idx) || idx >= this._maxVirtualStart) {
          this._virtualStart = this.grid ? (idx - this._itemsPerRow * 2) : (idx - 1);
        }
        this._manageFocus();
        this._assignModels();
        this._updateMetrics();
        // Estimate new physical offset.
        this._physicalTop = Math.floor(this._virtualStart / this._itemsPerRow) * this._physicalAverage;

        var currentTopItem = this._physicalStart;
        var currentVirtualItem = this._virtualStart;
        var targetOffsetTop = 0;
        var hiddenContentSize = this._hiddenContentSize;
        // scroll to the item as much as we can.
        while (currentVirtualItem < idx && targetOffsetTop <= hiddenContentSize) {
          targetOffsetTop = targetOffsetTop + this._getPhysicalSizeIncrement(currentTopItem);
          currentTopItem = (currentTopItem + 1) % this._physicalCount;
          currentVirtualItem++;
        }
        this._updateScrollerSize(true);
        this._positionItems();
        this._resetScrollPosition(this._physicalTop + this._scrollOffset + targetOffsetTop);
        this._increasePoolIfNeeded(0);
        // clear cached visible index.
        this._firstVisibleIndexVal = null;
        this._lastVisibleIndexVal = null;
      },

      /**
       * Reset the physical average and the average count.
       */
      _resetAverage: function() {
        this._physicalAverage = 0;
        this._physicalAverageCount = 0;
      },

      /**
       * A handler for the `iron-resize` event triggered by `IronResizableBehavior`
       * when the element is resized.
       */
      _resizeHandler: function() {
        this._debounce('_render', function() {
          // clear cached visible index.
          this._firstVisibleIndexVal = null;
          this._lastVisibleIndexVal = null;
          // Skip the resize event on touch devices when the address bar slides up.
          this.updateViewportBoundaries();
          if (this._isVisible) {
            // Reinstall the scroll event listener.
            this.toggleScrollListener(true);
            this._resetAverage();
            this._render();
          } else {
            // Uninstall the scroll event listener.
            this.toggleScrollListener(false);
          }
        }, ANIMATION_FRAME);
      },

      /**
       * Converts a random index to the index of the item that completes it's row.
       * Allows for better order and fill computation when grid == true.
       */
      _convertIndexToCompleteRow: function(idx) {
        // when grid == false _itemPerRow can be unset.
        this._itemsPerRow = this._itemsPerRow || 1;
        return this.grid ? Math.ceil(idx / this._itemsPerRow) * this._itemsPerRow : idx;
      },

      _isIndexRendered: function(idx) {
        return idx >= this._virtualStart && idx <= this._virtualEnd;
      },

      _getPhysicalIndex: function(vidx) {
        return (this._physicalStart + (vidx - this._virtualStart)) % this._physicalCount;
      },

      _clamp: function(v, min, max) {
        return Math.min(max, Math.max(min, v));
      },

      _debounce: function(name, cb, asyncModule) {
        this._debouncers = this._debouncers || {};
        this._debouncers[name] = Debouncer.debounce(
          this._debouncers[name],
          asyncModule,
          cb.bind(this));
        enqueueDebouncer(this._debouncers[name]);
      }

    });

    /**
    @license
    Copyright (c) 2017 Vaadin Ltd.
    This program is available under Apache License Version 2.0, available at https://vaadin.com/license/
    */
    /**
     * This Element is used internally by vaadin-grid.
     *
     * @private
     */
    class GridScrollerElement extends PolymerIronList {

      static get is() {
        return 'vaadin-grid-scroller';
      }

      static get properties() {
        return {
          size: {
            type: Number,
            notify: true
          },
          _vidxOffset: {
            value: 0
          }
        };
      }

      static get observers() {
        return [
          '_effectiveSizeChanged(_effectiveSize)'
        ];
      }

      connectedCallback() {
        super.connectedCallback();
        this._scrollHandler();
      }

      /**
      * @protected
      */
      _updateScrollerItem(item, index) {}
      /**
      * @protected
      */
      _afterScroll() {}
      /**
      * @protected
      */
      _getRowTarget() {}
      /**
      * @protected
      */
      _createScrollerRows() {}
      /**
      * @protected
      */
      _canPopulate() {}

      /**
      * @private
      */
      scrollToIndex(index) {
        this._warnPrivateAPIAccess('scrollToIndex');

        if (index > 0) {
          this._pendingScrollToIndex = null;
        }
        if (!parseInt(this.$.items.style.borderTopWidth) && index > 0) {
          // Schedule another scroll to be invoked once init is complete
          this._pendingScrollToIndex = index;
        }

        this._scrollingToIndex = true;
        index = Math.min(Math.max(index, 0), this._effectiveSize - 1);
        this.$.table.scrollTop = index / this._effectiveSize * (this.$.table.scrollHeight - this.$.table.offsetHeight);
        this._scrollHandler();

        if (this._accessIronListAPI(() => this._maxScrollTop) && this._virtualCount < this._effectiveSize) {
          this._adjustVirtualIndexOffset(1000000);
        }

        this._accessIronListAPI(() => super.scrollToIndex(index - this._vidxOffset));
        this._scrollHandler();

        // Ensure scroll position
        const row = Array.from(this.$.items.children).filter(child => child.index === index)[0];
        if (row) {
          const headerOffset = row.getBoundingClientRect().top - this.$.header.getBoundingClientRect().bottom;
          if (Math.abs(headerOffset) > 1) {
            this.$.table.scrollTop += headerOffset;
            this._scrollHandler();
          }
        }

        this._scrollingToIndex = false;
      }

      _effectiveSizeChanged(size) {
        let fvi; // first visible (adjusted) index
        let fviOffset = 0;
        this._iterateItems((pidx, vidx) => {
          if (vidx === this._firstVisibleIndex) {
            const row = this._physicalItems[pidx];
            fvi = row.index;
            fviOffset = row.getBoundingClientRect().top;
          }
        });

        if (this.items && size < this.items.length) {
          // Size was reduced, scroll to 0 first
          this._scrollTop = 0;
        }
        if (!Array.isArray(this.items)) {
          // Edge/IE seems to have the lowest maximum
          const maxVirtualItems = this._edge || this._ie ? 30000 : 100000;
          this.items = {length: Math.min(size, maxVirtualItems)};
        }

        this._accessIronListAPI(() => super._itemsChanged({path: 'items'}));

        this._virtualCount = Math.min(this.items.length, size) || 0;

        if (this._scrollTop === 0) {
          this._accessIronListAPI(() => this._scrollToIndex(Math.min(size - 1, fvi)));
          this._iterateItems((pidx, vidx) => {
            const row = this._physicalItems[pidx];
            if (row.index === fvi) {
              this.$.table.scrollTop += Math.round(row.getBoundingClientRect().top - fviOffset);
            }
            // Restore keyboard focus to the right cell
            if (row.index === this._focusedItemIndex && this._itemsFocusable && this.$.items.contains(this.shadowRoot.activeElement)) {
              const cellIndex = Array.from(this._itemsFocusable.parentElement.children).indexOf(this._itemsFocusable);
              row.children[cellIndex].focus();
            }
          });
        }
        this._assignModels();
        requestAnimationFrame(() => this._update());
      }

      _positionItems() {
        this._adjustScrollPosition();

        let rePosition;
        if (isNaN(this._physicalTop)) {
          rePosition = true;
          this._physicalTop = 0;
        }

        let y = this._physicalTop;
        this._iterateItems((pidx, vidx) => {
          this._physicalItems[pidx].style.transform = `translateY(${y}px)`;
          y += this._physicalSizes[pidx];
        });

        if (rePosition) {
          this._scrollToIndex(0);
        }
      }

      _increasePoolIfNeeded(count) {
        if ((count === 0 && this._scrollingToIndex) || !this._canPopulate() || !this._effectiveSize) {
          return;
        }

        if (!this._initialPoolCreated) {
          this._initialPoolCreated = true;
          super._increasePoolIfNeeded(25);
        } else if (this._optPhysicalSize !== Infinity) {
          this._debounceIncreasePool = Debouncer.debounce(
            this._debounceIncreasePool,
            animationFrame,
            () => {
              this._updateMetrics();
              const remainingPhysicalSize = this._optPhysicalSize - this._physicalSize;
              let estimatedMissingRowCount = Math.ceil(remainingPhysicalSize / this._physicalAverage);

              if (this._physicalCount + estimatedMissingRowCount > this._effectiveSize) {
                // Do not increase the physical item count above the this._effectiveSize
                estimatedMissingRowCount = Math.max(0, this._effectiveSize - this._physicalCount);
              }

              if (this._physicalSize && estimatedMissingRowCount > 0 && this._optPhysicalSize !== Infinity) {
                super._increasePoolIfNeeded(estimatedMissingRowCount);
                // Ensure the rows are in order after increasing pool
                this.__reorderChildNodes();
              }
            });
        }
      }

      __reorderChildNodes() {
        const childNodes = Array.from(this.$.items.childNodes);
        const rowsInOrder = !!childNodes.reduce((inOrder, current, currentIndex, array) => {
          if (currentIndex === 0 || array[currentIndex - 1].index === current.index - 1) {
            return inOrder;
          }
        }, true);

        if (!rowsInOrder) {
          childNodes.sort((row1, row2) => {
            return row1.index - row2.index;
          }).forEach(row => this.$.items.appendChild(row));
        }
      }

      _createPool(size) {
        const fragment = document.createDocumentFragment();
        const physicalItems = this._createScrollerRows(size);

        physicalItems.forEach(inst => fragment.appendChild(inst));
        this._getRowTarget().appendChild(fragment);

        // Weird hack needed to get Safari to actually distribute slots
        const content = this.querySelector('[slot]');
        if (content) {
          const slot = content.getAttribute('slot');
          content.setAttribute('slot', 'foo-bar');
          content.setAttribute('slot', slot);
        }

        this._updateHeaderFooterMetrics();

        afterNextRender(this, () => this.notifyResize());
        return physicalItems;
      }

      /**
       * Assigns the data models to a given set of items.
       * @param {!Array<number>=} itemSet
       */
      _assignModels(itemSet) {
        this._iterateItems((pidx, vidx) => {
          const el = this._physicalItems[pidx];
          this._toggleAttribute('hidden', vidx >= this._effectiveSize, el);
          this._updateScrollerItem(el, vidx + (this._vidxOffset || 0));
        }, itemSet);
      }

      _scrollHandler() {
        const delta = this.$.table.scrollTop - this._scrollPosition;
        this._accessIronListAPI(super._scrollHandler);
        const oldOffset = this._vidxOffset;
        if (this._accessIronListAPI(() => this._maxScrollTop) && this._virtualCount < this._effectiveSize) {
          this._adjustVirtualIndexOffset(delta);
        }
        if (this._vidxOffset !== oldOffset) {
          this._update();
        }
        this._afterScroll();
      }

      _adjustVirtualIndexOffset(delta) {
        if (Math.abs(delta) > 10000) {
          if (this._noScale) {
            this._noScale = false;
            return;
          }
          const scale = this.$.table.scrollTop / (this.$.table.scrollHeight - this.$.table.offsetHeight);
          const offset = scale * this._effectiveSize;
          this._vidxOffset = Math.round(offset - scale * this._virtualCount);
        } else {
          // Make sure user can always swipe/wheel scroll to the start and end
          const oldOffset = this._vidxOffset || 0;
          const threshold = 1000;
          const maxShift = 100;
          // At start
          if (this._scrollTop === 0) {
            this._vidxOffset = 0;
            if (oldOffset !== this._vidxOffset) {
              super.scrollToIndex(0);
            }
          } else if (this.firstVisibleIndex < threshold && this._vidxOffset > 0) {
            this._vidxOffset -= Math.min(this._vidxOffset, maxShift);
            if (oldOffset !== this._vidxOffset) {
              super.scrollToIndex(this.firstVisibleIndex + (oldOffset - this._vidxOffset));
            }
            this._noScale = true;
          }
          // At end
          const maxOffset = this._effectiveSize - this._virtualCount;
          if (this._scrollTop >= this._maxScrollTop && this._maxScrollTop > 0) {
            this._vidxOffset = maxOffset;
            if (oldOffset !== this._vidxOffset) {
              super.scrollToIndex(this._virtualCount);
            }
          } else if (this.firstVisibleIndex > this._virtualCount - threshold && this._vidxOffset < maxOffset) {
            this._vidxOffset += Math.min(maxOffset - this._vidxOffset, maxShift);
            if (oldOffset !== this._vidxOffset) {
              super.scrollToIndex(this.firstVisibleIndex - (this._vidxOffset - oldOffset));
            }
            this._noScale = true;
          }
        }
      }

      _accessIronListAPI(cb) {
        this._warnPrivateAPIAccessAsyncEnabled = false;
        const returnValue = cb.apply(this);
        this._debouncerWarnPrivateAPIAccess = Debouncer.debounce(
          this._debouncerWarnPrivateAPIAccess,
          animationFrame,
          () => this._warnPrivateAPIAccessAsyncEnabled = true
        );
        return returnValue;
      }

      /* Allow iron-list to access its APIs from debounced callbacks without warns */
      _debounceRender(cb, asyncModule) {
        super._debounceRender(() => this._accessIronListAPI(cb), asyncModule);
      }

      /* Warn when iron-list APIs are being accessed directly */
      _warnPrivateAPIAccess(apiName) {
        if (this._warnPrivateAPIAccessAsyncEnabled) {
          console.warn(`Accessing private API (${apiName})!`);
        }
      }

      _render() {
        this._accessIronListAPI(super._render);
      }

      _createFocusBackfillItem() { /* Ignore */ }
      _multiSelectionChanged() { /* Ignore */ }
      clearSelection() { /* Ignore */ }
      _itemsChanged() { /* Ignore */ }
      _manageFocus() { /* Ignore */ }
      _removeFocusedItem() { /* Ignore */ }

      get _firstVisibleIndex() {
        return this._accessIronListAPI(() => super.firstVisibleIndex);
      }
      get _lastVisibleIndex() {
        return this._accessIronListAPI(() => super.lastVisibleIndex);
      }
      _scrollToIndex(index) {
        this._accessIronListAPI(() => this.scrollToIndex(index));
      }
      get firstVisibleIndex() {
        this._warnPrivateAPIAccess('firstVisibleIndex'); return super.firstVisibleIndex;
      }
      set firstVisibleIndex(value) {
        this._warnPrivateAPIAccess('firstVisibleIndex'); super.firstVisibleIndex = value;
      }
      get lastVisibleIndex() {
        this._warnPrivateAPIAccess('lastVisibleIndex'); return super.lastVisibleIndex;
      }
      set lastVisibleIndex(value) {
        this._warnPrivateAPIAccess('lastVisibleIndex'); super.lastVisibleIndex = value;
      }
      updateViewportBoundaries() {
        this._warnPrivateAPIAccess('updateViewportBoundaries'); super.updateViewportBoundaries.apply(this, arguments);
      }
      _resizeHandler() {
        super._resizeHandler();
        flush();
      }
    }

    customElements.define(GridScrollerElement.is, GridScrollerElement);

    /**
    @license
    Copyright (c) 2017 Vaadin Ltd.
    This program is available under Apache License Version 2.0, available at https://vaadin.com/license/
    */
    /**
     * @polymerMixin
     */
    const A11yMixin = superClass => class A11yMixin extends superClass {
      static get observers() {
        return [
          '_a11yUpdateGridSize(size, _columnTree, _columnTree.*)'
        ];
      }

      _a11yGetHeaderRowCount(_columnTree) {
        return _columnTree.filter(level => level.some(col => col._headerTemplate || col.headerRenderer || col.path || col.header)).length;
      }

      _a11yGetFooterRowCount(_columnTree) {
        return _columnTree.filter(level => level.some(col => col._headerTemplate || col.headerRenderer)).length;
      }

      _a11yUpdateGridSize(size, _columnTree) {
        if (size === undefined || _columnTree === undefined) {
          return;
        }

        const bodyColumns = _columnTree[_columnTree.length - 1];
        this.$.table.setAttribute(
          'aria-rowcount',
          size + this._a11yGetHeaderRowCount(_columnTree) + this._a11yGetFooterRowCount(_columnTree)
        );
        this.$.table.setAttribute('aria-colcount', bodyColumns && bodyColumns.length || 0);

        this._a11yUpdateHeaderRows();
        this._a11yUpdateFooterRows();
      }

      _a11yUpdateHeaderRows() {
        Array.from(this.$.header.children).forEach((headerRow, index) =>
          headerRow.setAttribute('aria-rowindex', index + 1)
        );
      }

      _a11yUpdateFooterRows() {
        Array.from(this.$.footer.children).forEach((footerRow, index) =>
          footerRow.setAttribute(
            'aria-rowindex',
            this._a11yGetHeaderRowCount(this._columnTree) + this.size + index + 1
          )
        );
      }

      _a11yUpdateRowRowindex(row, index) {
        row.setAttribute('aria-rowindex', index + this._a11yGetHeaderRowCount(this._columnTree) + 1);
      }

      _a11yUpdateRowSelected(row, selected) {
        // Jaws reads selection only for rows, NVDA only for cells
        row.setAttribute('aria-selected', Boolean(selected));
        Array.from(row.children).forEach(cell =>
          cell.setAttribute('aria-selected', Boolean(selected))
        );
      }

      _a11yUpdateRowLevel(row, level) {
        row.setAttribute('aria-level', level + 1);
      }

      _a11yUpdateRowDetailsOpened(row, detailsOpened) {
        Array.from(row.children).forEach(cell => {
          if (typeof detailsOpened === 'boolean') {
            cell.setAttribute('aria-expanded', detailsOpened);
          } else {
            if (cell.hasAttribute('aria-expanded')) {
              cell.removeAttribute('aria-expanded');
            }
          }
        });
      }

      _a11ySetRowDetailsCell(row, detailsCell) {
        Array.from(row.children).forEach(cell => {
          if (cell !== detailsCell) {
            cell.setAttribute('aria-controls', detailsCell.id);
          }
        });
      }

      _a11yUpdateCellColspan(cell, colspan) {
        cell.setAttribute('aria-colspan', Number(colspan));
      }

      _a11yUpdateSorters() {
        Array.from(this.querySelectorAll('vaadin-grid-sorter')).forEach(sorter => {
          let cellContent = sorter.parentNode;
          while (cellContent && cellContent.localName !== 'vaadin-grid-cell-content') {
            cellContent = cellContent.parentNode;
          }
          if (cellContent && cellContent.assignedSlot) {
            const cell = cellContent.assignedSlot.parentNode;
            cell.setAttribute('aria-sort', {
              'asc': 'ascending',
              'desc': 'descending'
            }[String(sorter.direction)] || 'none');
          }
        });
      }
    };

    /**
    @license
    Copyright (c) 2017 Vaadin Ltd.
    This program is available under Apache License Version 2.0, available at https://vaadin.com/license/
    */
    /**
     * @polymerMixin
     */
    const ActiveItemMixin = superClass => class ActiveItemMixin extends superClass {

      static get properties() {
        return {
          /**
           * The item user has last interacted with. Turns to `null` after user deactivates
           * the item by re-interacting with the currently active item.
           */
          activeItem: {
            type: Object,
            notify: true,
            value: null
          }
        };
      }

      ready() {
        super.ready();

        this.$.scroller.addEventListener('click', this._onClick.bind(this));
        this.addEventListener('cell-activate', this._activateItem.bind(this));
      }

      _activateItem(e) {
        const model = e.detail.model;
        const clickedItem = model ? model.item : null;

        if (clickedItem) {
          this.activeItem = !this._itemsEqual(this.activeItem, clickedItem) ? clickedItem : null;
        }
      }

      // we need to listen to click instead of tap because on mobile safari, the
      // document.activeElement has not been updated (focus has not been shifted)
      // yet at the point when tap event is being executed.
      _onClick(e) {
        if (e.defaultPrevented) {
          // Something has handled this click already, e. g., <vaadin-grid-sorter>
          return;
        }

        const path = e.composedPath();
        const cell = path[path.indexOf(this.$.table) - 3];
        if (!cell || cell.getAttribute('part').indexOf('details-cell') > -1) {
          return;
        }
        const cellContent = cell._content;

        const activeElement = this.getRootNode().activeElement;
        const cellContentHasFocus = cellContent.contains(activeElement) &&
          // MSIE bug: flex children receive focus. Make type & attributes check.
          (!this._ie || this._isFocusable(activeElement));
        if (!cellContentHasFocus && !this._isFocusable(e.target)) {
          this.dispatchEvent(new CustomEvent('cell-activate', {detail: {
            model: this.__getRowModel(cell.parentElement)
          }}));
        }
      }

      _isFocusable(target) {
        if (!target.parentNode) {
          return false;
        }
        const focusables = Array.from(target.parentNode
          .querySelectorAll('[tabindex], button, input, select, textarea, object, iframe, label, a[href], area[href]'))
          .filter(element => element.getAttribute('part') !== 'cell body-cell');

        const isFocusableElement = focusables.indexOf(target) !== -1;
        return !target.disabled && isFocusableElement;
      }
    };

    /**
    @license
    Copyright (c) 2017 Vaadin Ltd.
    This program is available under Apache License Version 2.0, available at https://vaadin.com/license/
    */

    /**
     * @polymerMixin
     */
    const ArrayDataProviderMixin = superClass => class ArrayDataProviderMixin extends superClass {

      static get properties() {

        return {
          /**
           * An array containing the items which will be stamped to the column template
           * instances.
           */
          items: Array

        };
      }

      static get observers() {
        return [
          '_itemsChanged(items, items.*, isAttached)'
        ];
      }

      _itemsChanged(items, splices, isAttached) {
        if (!isAttached) {
          return;
        }
        if (!Array.isArray(items)) {
          if (items === undefined || items === null) {
            this.size = 0;
          }
          if (this.dataProvider === this._arrayDataProvider) {
            this.dataProvider = undefined;
          }
          return;
        }

        this.size = items.length;
        this.dataProvider = this.dataProvider || this._arrayDataProvider;
        this.clearCache();
        this._ensureFirstPageLoaded();
      }

      _arrayDataProvider(opts, cb) {
        let items = (Array.isArray(this.items) ? this.items : []).slice(0);

        if (this._filters && this._checkPaths(this._filters, 'filtering', items)) {
          items = this._filter(items);
        }

        this.size = items.length;

        if (opts.sortOrders.length && this._checkPaths(this._sorters, 'sorting', items)) {
          items = items.sort(this._multiSort.bind(this));
        }

        const start = opts.page * opts.pageSize;
        const end = start + opts.pageSize;
        const slice = items.slice(start, end);
        cb(slice, items.length);
      }

      /**
       * Check array of filters/sorters for paths validity, console.warn invalid items
       * @param {Array}  arrayToCheck The array of filters/sorters to check
       * @param {string} action       The name of action to include in warning (filtering, sorting)
       * @param {Array}  items
       */
      _checkPaths(arrayToCheck, action, items) {
        if (!items.length) {
          return false;
        }

        let result = true;

        for (var i in arrayToCheck) {
          const path = arrayToCheck[i].path;

          // skip simple paths
          if (!path || path.indexOf('.') === -1) {
            continue;
          }

          const parentProperty = path.replace(/\.[^\.]*$/, ''); // a.b.c -> a.b
          if (Base.get(parentProperty, items[0]) === undefined) {
            console.warn(`Path "${path}" used for ${action} does not exist in all of the items, ${action} is disabled.`);
            result = false;
          }
        }

        return result;
      }

      _multiSort(a, b) {
        return this._sorters.map(sort => {
          if (sort.direction === 'asc') {
            return this._compare(Base.get(sort.path, a), Base.get(sort.path, b));
          } else if (sort.direction === 'desc') {
            return this._compare(Base.get(sort.path, b), Base.get(sort.path, a));
          }
          return 0;
        }).reduce((p, n) => {
          return p ? p : n;
        }, 0);
      }

      _normalizeEmptyValue(value) {
        if ([undefined, null].indexOf(value) >= 0) {
          return '';
        } else if (isNaN(value)) {
          return value.toString();
        } else {
          return value;
        }
      }

      _compare(a, b) {
        a = this._normalizeEmptyValue(a);
        b = this._normalizeEmptyValue(b);

        if (a < b) {
          return -1;
        }
        if (a > b) {
          return 1;
        }
        return 0;
      }

      _filter(items) {
        return items.filter((item, index) => {
          return this._filters.filter(filter => {
            const value = this._normalizeEmptyValue(Base.get(filter.path, item));
            const filterValueLowercase = this._normalizeEmptyValue(filter.value).toString().toLowerCase();
            return value.toString().toLowerCase().indexOf(filterValueLowercase) === -1;
          }).length === 0;
        });
      }

    };

    /**
    @license
    Copyright (c) 2017 Vaadin Ltd.
    This program is available under Apache License Version 2.0, available at https://vaadin.com/license/
    */

    /**
     * @polymerMixin
     */
    const ColumnResizingMixin = superClass => class ColumnResizingMixin extends GestureEventListeners(superClass) {

      ready() {
        super.ready();
        const scroller = this.$.scroller;
        addListener(scroller, 'track', this._onHeaderTrack.bind(this));

        // Disallow scrolling while resizing
        scroller.addEventListener('touchmove', e => scroller.hasAttribute('column-resizing') && e.preventDefault());

        // Disable contextmenu on any resize separator.
        scroller.addEventListener('contextmenu', e => e.target.getAttribute('part') == 'resize-handle' && e.preventDefault());

        // Disable native cell focus when resizing
        scroller.addEventListener('mousedown', e => e.target.getAttribute('part') === 'resize-handle' && e.preventDefault());
      }

      _onHeaderTrack(e) {
        const handle = e.target;
        if (handle.getAttribute('part') === 'resize-handle') {
          const cell = handle.parentElement;
          let column = cell._column;

          this._toggleAttribute('column-resizing', true, this.$.scroller);

          // Get the target column to resize
          while (column.localName === 'vaadin-grid-column-group') {
            column = Array.prototype.slice.call(column._childColumns, 0)
              .sort(function(a, b) {
                return a._order - b._order;
              })
              .filter(function(column) {
                return !column.hidden;
              }).pop();
          }

          const columnRowCells = Array.from(this.$.header.querySelectorAll('[part~="row"]:last-child [part~="cell"]'));
          var targetCell = columnRowCells.filter(cell => cell._column === column)[0];
          // Resize the target column
          if (targetCell.offsetWidth) {
            var style = window.getComputedStyle(targetCell);
            var minWidth = 10 + parseInt(style.paddingLeft) + parseInt(style.paddingRight) + parseInt(style.borderLeftWidth)
              + parseInt(style.borderRightWidth) + parseInt(style.marginLeft) + parseInt(style.marginRight);
            const maxWidth = targetCell.offsetWidth + (this.__isRTL ? targetCell.getBoundingClientRect().left - e.detail.x :
              e.detail.x - targetCell.getBoundingClientRect().right);
            column.width = Math.max(minWidth, maxWidth) + 'px';
            column.flexGrow = 0;
          }
          // Fix width and flex-grow for all preceding columns
          columnRowCells
            .sort(
              function(a, b) {
                return a._column._order - b._column._order;
              })
            .forEach(function(cell, index, array) {
              if (index < array.indexOf(targetCell)) {
                cell._column.width = cell.offsetWidth + 'px';
                cell._column.flexGrow = 0;
              }
            });

          if (e.detail.state === 'end') {
            this._toggleAttribute('column-resizing', false, this.$.scroller);
            this.dispatchEvent(new CustomEvent('column-resize', {
              detail: {resizedColumn: column}
            }));
          }

          // Notify resize
          this._resizeHandler();
        }
      }

      /**
      * Fired when a column in the grid is resized by the user.
      *
      * @event column-resize
      * @param {Object} detail
      * @param {Object} detail.resizedColumn the column that was resized
      */
    };

    /**
    @license
    Copyright (c) 2017 Vaadin Ltd.
    This program is available under Apache License Version 2.0, available at https://vaadin.com/license/
    */

    const ItemCache = class ItemCache {

      constructor(grid, parentCache, parentItem) {
        this.grid = grid;
        this.parentCache = parentCache;
        this.parentItem = parentItem;
        this.itemCaches = {};
        this.items = {};
        this.effectiveSize = 0;
        this.size = 0;
        this.pendingRequests = {};
      }

      isLoading() {
        return Object.keys(this.pendingRequests).length || Object.keys(this.itemCaches).filter(index => {
          return this.itemCaches[index].isLoading();
        })[0];
      }

      getItemForIndex(index) {
        const {cache, scaledIndex} = this.getCacheAndIndex(index);
        return cache.items[scaledIndex];
      }

      updateSize() {
        this.effectiveSize = (!this.parentItem || this.grid._isExpanded(this.parentItem)) ?
          this.size + Object.keys(this.itemCaches).reduce((prev, curr) => {
            const subCache = this.itemCaches[curr];
            subCache.updateSize();
            return prev + subCache.effectiveSize;
          }, 0) :
          0;
      }

      ensureSubCacheForScaledIndex(scaledIndex) {
        if (!this.itemCaches[scaledIndex]) {
          const subCache = new ItemCache(this.grid, this, this.items[scaledIndex]);
          this.itemCaches[scaledIndex] = subCache;
          this.grid._loadPage(0, subCache);
        }
      }

      getCacheAndIndex(index) {
        let thisLevelIndex = index;
        const keys = Object.keys(this.itemCaches);
        for (var i = 0; i < keys.length; i++) {
          const expandedIndex = Number(keys[i]);
          const subCache = this.itemCaches[expandedIndex];
          if (thisLevelIndex <= expandedIndex) {
            return {cache: this, scaledIndex: thisLevelIndex};
          } else if (thisLevelIndex <= expandedIndex + subCache.effectiveSize) {
            return subCache.getCacheAndIndex(thisLevelIndex - expandedIndex - 1);
          }
          thisLevelIndex -= subCache.effectiveSize;
        }
        return {cache: this, scaledIndex: thisLevelIndex};
      }

    };

    /**
     * @polymerMixin
     */
    const DataProviderMixin = superClass => class DataProviderMixin extends superClass {

      static get properties() {
        return {

          /**
           * Number of items fetched at a time from the dataprovider.
           */
          pageSize: {
            type: Number,
            value: 50,
            observer: '_pageSizeChanged'
          },

          /**
           * Function that provides items lazily. Receives arguments `params`, `callback`
           *
           * `params.page` Requested page index
           *
           * `params.pageSize` Current page size
           *
           * `params.filters` Currently applied filters
           *
           * `params.sortOrders` Currently applied sorting orders
           *
           * `params.parentItem` When tree is used, and sublevel items
           * are requested, reference to parent item of the requested sublevel.
           * Otherwise `undefined`.
           *
           * `callback(items, size)` Callback function with arguments:
           *   - `items` Current page of items
           *   - `size` Total number of items. When tree sublevel items
           *     are requested, total number of items in the requested sublevel.
           *     Optional when tree is not used, required for tree.
           */
          dataProvider: {
            type: Object,
            notify: true,
            observer: '_dataProviderChanged'
          },

          /**
           * `true` while data is being requested from the data provider.
           */
          loading: {
            type: Boolean,
            notify: true,
            readOnly: true,
            reflectToAttribute: true
          },

          _cache: {
            type: Object,
            value: function() {
              const cache = new ItemCache(this);
              return cache;
            }
          },

          /**
           * Path to an item sub-property that identifies the item.
           */
          itemIdPath: {
            type: String,
            value: null
          },

          /**
           * An array that contains the expanded items.
           */
          expandedItems: {
            type: Object,
            notify: true,
            value: () => []
          }

        };
      }

      static get observers() {
        return [
          '_sizeChanged(size)',
          '_itemIdPathChanged(itemIdPath)',
          '_expandedItemsChanged(expandedItems.*)'
        ];
      }

      _sizeChanged(size) {
        const delta = size - this._cache.size;
        this._cache.size += delta;
        this._cache.effectiveSize += delta;
        this._effectiveSize = this._cache.effectiveSize;
      }

      _getItem(index, el) {
        if (index >= this._effectiveSize) {
          return;
        }

        el.index = index;
        const {cache, scaledIndex} = this._cache.getCacheAndIndex(index);
        const item = cache.items[scaledIndex];
        if (item) {
          this._toggleAttribute('loading', false, el);
          this._updateItem(el, item);
          if (this._isExpanded(item)) {
            cache.ensureSubCacheForScaledIndex(scaledIndex);
          }
        } else {
          this._toggleAttribute('loading', true, el);
          this._loadPage(this._getPageForIndex(scaledIndex), cache);
        }

      }

      _expandedInstanceChangedCallback(inst, value) {
        if (inst.item === undefined) {
          return;
        }
        if (value) {
          this.expandItem(inst.item);
        } else {
          this.collapseItem(inst.item);
        }
      }

      /**
       * Returns a value that identifies the item. Uses `itemIdPath` if available.
       * Can be customized by overriding.
       */
      getItemId(item) {
        return this.itemIdPath ? this.get(this.itemIdPath, item) : item;
      }

      _isExpanded(item) {
        return this.__expandedKeys.has(this.getItemId(item));
      }

      _expandedItemsChanged(e) {
        this.__cacheExpandedKeys();
        this._cache.updateSize();
        this._effectiveSize = this._cache.effectiveSize;
        this._assignModels();
      }

      _itemIdPathChanged(e) {
        this.__cacheExpandedKeys();
      }

      __cacheExpandedKeys() {
        if (this.expandedItems) {
          this.__expandedKeys = new Set();
          this.expandedItems.forEach(item => {
            this.__expandedKeys.add(this.getItemId(item));
          });
        }
      }

      /**
       * Expands the given item tree.
       */
      expandItem(item) {
        if (!this._isExpanded(item)) {
          this.push('expandedItems', item);
        }
      }

      /**
       * Collapses the given item tree.
       */
      collapseItem(item) {
        if (this._isExpanded(item)) {
          this.splice('expandedItems', this._getItemIndexInArray(item, this.expandedItems), 1);
        }
      }

      _getIndexLevel(index) {
        let {cache} = this._cache.getCacheAndIndex(index);
        let level = 0;
        while (cache.parentCache) {
          cache = cache.parentCache;
          level++;
        }
        return level;
      }

      _canPopulate() {
        return this._hasData && this._columnTree;
      }

      _loadPage(page, cache) {
        // make sure same page isn't requested multiple times.
        if (!cache.pendingRequests[page] && this.dataProvider) {
          this._setLoading(true);
          cache.pendingRequests[page] = true;
          const params = {
            page,
            pageSize: this.pageSize,
            sortOrders: this._mapSorters(),
            filters: this._mapFilters(),
            parentItem: cache.parentItem
          };

          this.dataProvider(params, (items, size) => {
            if (size !== undefined) {
              cache.size = size;
            } else {
              if (params.parentItem) {
                cache.size = items.length;
              }
            }

            const currentItems = Array.from(this.$.items.children).map(row => row._item);

            // Populate the cache with new items
            items.forEach((item, itemsIndex) => {
              const itemIndex = page * this.pageSize + itemsIndex;
              cache.items[itemIndex] = item;
              if (this._isExpanded(item) && currentItems.indexOf(item) > -1) {
                // Force synchronous data request for expanded item sub-cache
                cache.ensureSubCacheForScaledIndex(itemIndex);
              }
            });

            this._hasData = true;

            delete cache.pendingRequests[page];

            this._setLoading(false);
            this._cache.updateSize();
            this._effectiveSize = this._cache.effectiveSize;

            Array.from(this.$.items.children)
              .filter(row => !row.hidden)
              .forEach(row => {
                const cachedItem = this._cache.getItemForIndex(row.index);
                if (cachedItem) {
                  this._toggleAttribute('loading', false, row);
                  this._updateItem(row, cachedItem);
                }
              });

            this._increasePoolIfNeeded(0);

            this.__itemsReceived();
          });
        }
      }

      _getPageForIndex(index) {
        return Math.floor(index / this.pageSize);
      }

      /**
       * Clears the cached pages and reloads data from dataprovider when needed.
       */
      clearCache() {
        this._cache = new ItemCache(this);
        Array.from(this.$.items.children).forEach(row => {
          Array.from(row.children).forEach(cell => {
            // Force data system to pick up subproperty changes
            cell._instance && cell._instance._setPendingProperty('item', {}, false);
          });
        });
        this._cache.size = this.size || 0;
        this._cache.updateSize();
        this._hasData = false;
        this._assignModels();

        if (!this._effectiveSize) {
          this._loadPage(0, this._cache);
        }
      }

      _pageSizeChanged(pageSize, oldPageSize) {
        if (oldPageSize !== undefined && pageSize !== oldPageSize) {
          this.clearCache();
        }
      }

      _checkSize() {
        if (this.size === undefined && this._effectiveSize === 0) {
          console.warn(
            'The <vaadin-grid> needs the total number of items' +
            ' in order to display rows. Set the total number of items' +
            ' to the `size` property, or provide the total number of items' +
            ' in the second argument of the `dataProvider`’s `callback` call.'
          );
        }
      }

      _dataProviderChanged(dataProvider, oldDataProvider) {
        if (oldDataProvider !== undefined) {
          this.clearCache();
        }

        if (dataProvider && this.items && this.items.length) {
          // Fixes possibly invalid cached lastVisibleIndex value in <iron-list>
          this._scrollToIndex(this._firstVisibleIndex);
        }

        this._ensureFirstPageLoaded();

        this._debouncerCheckSize = Debouncer.debounce(
          this._debouncerCheckSize,
          timeOut.after(2000),
          this._checkSize.bind(this));

        this._scrollHandler();
      }

      _ensureFirstPageLoaded() {
        if (!this._hasData) {
          // load data before adding rows to make sure they have content when
          // rendered for the first time.
          this._loadPage(0, this._cache, () => {
            const hadData = this._hasData;
            this._hasData = true;
            if (!hadData) {
              this.notifyResize();
            }
          });
        }
      }

      _itemsEqual(item1, item2) {
        return this.getItemId(item1) === this.getItemId(item2);
      }

      _getItemIndexInArray(item, array) {
        let result = -1;
        array.forEach((i, idx) => {
          if (this._itemsEqual(i, item)) {
            result = idx;
          }
        });
        return result;
      }

    };

    /**
    @license
    Copyright (c) 2017 Vaadin Ltd.
    This program is available under Apache License Version 2.0, available at https://vaadin.com/license/
    */

    /**
     * @polymerMixin
     */
    const DynamicColumnsMixin = superClass => class DynamicColumnsMixin extends superClass {

      ready() {
        super.ready();
        this._addNodeObserver();
      }

      _hasColumnGroups(columns) {
        for (let i = 0; i < columns.length; i++) {
          if (columns[i].localName === 'vaadin-grid-column-group') {
            return true;
          }
        }

        return false;
      }

      _getChildColumns(el) {
        return FlattenedNodesObserver.getFlattenedNodes(el).filter(this._isColumnElement);
      }

      _flattenColumnGroups(columns) {
        return columns.map(col => {
          if (col.localName === 'vaadin-grid-column-group') {
            return this._getChildColumns(col);
          } else {
            return [col];
          }
        }).reduce((prev, curr) => {
          return prev.concat(curr);
        }, []);
      }

      _getColumnTree() {
        var rootColumns = FlattenedNodesObserver.getFlattenedNodes(this).filter(this._isColumnElement);
        var _columnTree = [];

        for (var c = rootColumns; ;) {
          _columnTree.push(c);
          if (!this._hasColumnGroups(c)) {
            break;
          }
          c = this._flattenColumnGroups(c);
        }

        return _columnTree;
      }

      _updateColumnTree() {
        var columnTree = this._getColumnTree();
        if (!this._arrayEquals(columnTree, this._columnTree)) {
          this._columnTree = columnTree;
        }
      }

      _addNodeObserver() {
        this._observer = new FlattenedNodesObserver(this, info => {

          const rowDetailsTemplate = info.addedNodes.filter(n => n.localName === 'template' && n.classList.contains('row-details'))[0];
          if (rowDetailsTemplate && this._rowDetailsTemplate !== rowDetailsTemplate) {
            this._rowDetailsTemplate = rowDetailsTemplate;
          }

          if (info.addedNodes.filter(this._isColumnElement).length > 0 ||
            info.removedNodes.filter(this._isColumnElement).length > 0) {
            this._updateColumnTree();
          }

          this._debouncerCheckImports = Debouncer.debounce(
            this._debouncerCheckImports,
            timeOut.after(2000),
            this._checkImports.bind(this));

          this._ensureFirstPageLoaded();
        });
      }

      _arrayEquals(arr1, arr2) {
        if (!arr1 || !arr2 || arr1.length != arr2.length) {
          return false;
        }

        for (var i = 0, l = arr1.length; i < l; i++) {
          // Check if we have nested arrays
          if (arr1[i] instanceof Array && arr2[i] instanceof Array) {
            // recurse into the nested arrays
            if (!this._arrayEquals(arr1[i], arr2[i])) {
              return false;
            }
          } else if (arr1[i] != arr2[i]) {
            return false;
          }
        }
        return true;
      }

      _checkImports() {
        [
          'vaadin-grid-column-group',
          'vaadin-grid-filter',
          'vaadin-grid-filter-column',
          'vaadin-grid-tree-toggle',
          'vaadin-grid-selection-column',
          'vaadin-grid-sort-column',
          'vaadin-grid-sorter'
        ].forEach(elementName => {
          var element = this.querySelector(elementName);
          if (element && !(element instanceof PolymerElement)) {
            console.warn(`Make sure you have imported the required module for <${elementName}> element.`);
          }
        });
      }

      _updateFirstAndLastColumn() {
        Array.from(this.shadowRoot.querySelectorAll('tr')).forEach(row => this._updateFirstAndLastColumnForRow(row));
      }

      _updateFirstAndLastColumnForRow(row) {
        Array.from(row.querySelectorAll('[part~="cell"]:not([part~="details-cell"])'))
          .sort((a, b) => {
            return a._column._order - b._column._order;
          }).forEach((cell, cellIndex, children) => {
            this._toggleAttribute('first-column', cellIndex === 0, cell);
            this._toggleAttribute('last-column', cellIndex === children.length - 1, cell);
          });
      }

      _isColumnElement(node) {
        return node.nodeType === Node.ELEMENT_NODE && /\bcolumn\b/.test(node.localName);
      }
    };

    /**
    @license
    Copyright (c) 2018 Vaadin Ltd.
    This program is available under Apache License Version 2.0, available at https://vaadin.com/license/
    */
    /**
     * @polymerMixin
     */
    const EventContextMixin = superClass => class EventContextMixin extends superClass {

      /**
       * Returns an object with context information about the event target:
       * - `item`: the data object corresponding to the targeted row (not specified when targeting header or footer)
       * - `column`: the column element corresponding to the targeted cell (not specified when targeting row details)
       * - `section`: whether the event targeted the body, header, footer or details of the grid
       *
       * These additional properties are included when `item` is specified:
       * - `index`: the index of the item
       * - `selected`: the selected state of the item
       * - `detailsOpened`: whether the row details are open for the item
       * - `expanded`: the expanded state of the tree toggle
       * - `level`: the tree hierarchy level
       *
       * The returned object is populated only when a grid cell, header, footer or row details is found in `event.composedPath()`.
       * This means mostly mouse and keyboard events. If such a grid part is not found in the path, an empty object is returned.
       * This may be the case eg. if the event is fired on the `<vaadin-grid>` element and not any deeper in the DOM, or if
       * the event targets the empty part of the grid body.
       */
      getEventContext(event) {

        const context = {};

        const path = event.composedPath();
        const cell = path[path.indexOf(this.$.table) - 3];

        if (!cell) {
          return context;
        }

        context.section = ['body', 'header', 'footer', 'details']
          .filter(section => cell.getAttribute('part').indexOf(section) > -1)[0];

        if (cell._column) {
          context.column = cell._column;
        }

        if (context.section === 'body' || context.section === 'details') {
          Object.assign(context, this.__getRowModel(cell.parentElement));
        }

        return context;
      }

    };

    /**
    @license
    Copyright (c) 2017 Vaadin Ltd.
    This program is available under Apache License Version 2.0, available at https://vaadin.com/license/
    */
    /**
     * @polymerMixin
     */
    const FilterMixin = superClass => class FilterMixin extends superClass {

      static get properties() {
        return {
          _filters: {
            type: Array,
            value: function() {
              return [];
            }
          }
        };
      }

      ready() {
        super.ready();
        this.addEventListener('filter-changed', this._filterChanged.bind(this));
      }

      _filterChanged(e) {
        if (this._filters.indexOf(e.target) === -1) {
          this._filters.push(e.target);
        }

        e.stopPropagation();

        if (this.dataProvider) {
          this.clearCache();
        }
      }

      _mapFilters() {
        return this._filters.map(filter => {
          return {
            path: filter.path,
            value: filter.value
          };
        });
      }
    };

    /**
    @license
    Copyright (c) 2017 Vaadin Ltd.
    This program is available under Apache License Version 2.0, available at https://vaadin.com/license/
    */
    /**
     * `vaadin-grid-templatizer` is a helper element for the `vaadin-grid` that is preparing and
     * stamping instances of cells and columns templates
     *
     * @extends PolymerElement
     * @private
     */
    class GridTemplatizer extends (class extends PolymerElement {}) {
      static get is() {
        return 'vaadin-grid-templatizer';
      }

      static get properties() {
        return {
          dataHost: Object,

          template: Object,

          _templateInstances: {
            type: Array,
            value: function() {
              return [];
            }
          },

          _parentPathValues: {
            value: function() {
              return {};
            }
          },

          _grid: Object
        };
      }

      static get observers() {
        return [
          '_templateInstancesChanged(_templateInstances.*, _parentPathValues.*)'
        ];
      }

      constructor() {
        super();

        this._instanceProps = {
          detailsOpened: true,
          index: true,
          item: true,
          selected: true,
          expanded: true,
          level: true
        };
      }

      createInstance() {
        this._ensureTemplatized();
        const instance = new this._TemplateClass({});
        this.addInstance(instance);

        return instance;
      }

      addInstance(instance) {
        if (this._templateInstances.indexOf(instance) === -1) {
          this._templateInstances.push(instance);
          requestAnimationFrame(() => this.notifyPath('_templateInstances.*', this._templateInstances));
        }
      }

      removeInstance(instance) {
        const index = this._templateInstances.indexOf(instance);
        this.splice('_templateInstances', index, 1);
      }

      _ensureTemplatized() {
        if (!this._TemplateClass) {
          this._TemplateClass = templatize(this.template, this, {
            instanceProps: this._instanceProps,
            parentModel: true,

            forwardHostProp: function(prop, value) {
              this._forwardParentProp(prop, value);

              if (this._templateInstances) {
                this._templateInstances.forEach(inst => inst.notifyPath(prop, value));
              }
            },

            notifyInstanceProp: function(inst, prop, value) {
              if (prop === 'index' || prop === 'item') {
                // We don’t need a change notification for these.
                return;
              }

              const originalProp = `__${prop}__`;

              // Notify for only user-action changes, not for scrolling updates. E. g.,
              // if `detailsOpened` is different from `__detailsOpened__`, which was set during render.
              if (inst[originalProp] === value) {
                return;
              }
              inst[originalProp] = value;

              const row = Array.from(this._grid.$.items.children).filter(row => this._grid._itemsEqual(row._item, inst.item))[0];
              if (row) {
                Array.from(row.children).forEach(cell => {
                  if (cell._instance) {
                    cell._instance[originalProp] = value;
                    cell._instance.notifyPath(prop, value);
                  }
                });
              }

              const itemPrefix = 'item.';
              if (Array.isArray(this._grid.items) && prop.indexOf(itemPrefix) === 0) {
                const itemsIndex = this._grid.items.indexOf(inst.item);
                const path = prop.slice(itemPrefix.length);
                this._grid.notifyPath(`items.${itemsIndex}.${path}`, value);
              }

              const gridCallback = `_${prop}InstanceChangedCallback`;
              if (this._grid && this._grid[gridCallback]) {
                this._grid[gridCallback](inst, value);
              }
            }

          });
        }
      }

      _forwardParentProp(prop, value) {
        this._parentPathValues[prop] = value;
        this._templateInstances.forEach(inst => inst.notifyPath(prop, value));
      }

      _templateInstancesChanged(t, p) {
        let index, count;
        if (t.path === '_templateInstances') {
          // Iterate all instances
          index = 0;
          count = this._templateInstances.length;
        } else if (t.path === '_templateInstances.splices') {
          // Iterate only new instances
          index = t.value.index;
          count = t.value.addedCount;
        } else {
          return;
        }
        Object.keys(this._parentPathValues || {}).forEach(keyName => {
          for (var i = index; i < index + count; i++) {
            this._templateInstances[i].set(keyName, this._parentPathValues[keyName]);
          }
        });
      }

    }

    customElements.define(GridTemplatizer.is, GridTemplatizer);

    /**
    @license
    Copyright (c) 2017 Vaadin Ltd.
    This program is available under Apache License Version 2.0, available at https://vaadin.com/license/
    */

    /**
     * @polymerMixin
     */
    const RowDetailsMixin = superClass => class RowDetailsMixin extends superClass {
      static get properties() {
        return {
          /**
           * An array containing references to items with open row details.
           */
          detailsOpenedItems: {
            type: Array,
            value: function() {
              return [];
            }
          },

          _rowDetailsTemplate: Object,

          /**
           * Custom function for rendering the content of the row details.
           * Receives three arguments:
           *
           * - `root` The row details content DOM element. Append your content to it.
           * - `grid` The `<vaadin-grid>` element.
           * - `rowData` The object with the properties related with
           *   the rendered item, contains:
           *   - `rowData.index` The index of the item.
           *   - `rowData.item` The item.
           */
          rowDetailsRenderer: Function,

          _detailsCells: {
            type: Array,
          }
        };
      }

      static get observers() {
        return [
          '_detailsOpenedItemsChanged(detailsOpenedItems.*, _rowDetailsTemplate, rowDetailsRenderer)',
          '_rowDetailsTemplateOrRendererChanged(_rowDetailsTemplate, rowDetailsRenderer)'
        ];
      }

      _rowDetailsTemplateOrRendererChanged(rowDetailsTemplate, rowDetailsRenderer) {
        if (rowDetailsTemplate && rowDetailsRenderer) {
          throw new Error('You should only use either a renderer or a template for row details');
        }
        if (rowDetailsTemplate || rowDetailsRenderer) {
          if (rowDetailsTemplate && !rowDetailsTemplate.templatizer) {
            var templatizer = new GridTemplatizer();
            templatizer._grid = this;
            templatizer.dataHost = this.dataHost;
            templatizer.template = rowDetailsTemplate;
            rowDetailsTemplate.templatizer = templatizer;
          }

          if (this._columnTree) {
            // Only update the rows if the column tree has already been initialized
            Array.from(this.$.items.children).forEach(row => {
              if (!row.querySelector('[part~=details-cell]')) {
                this._updateRow(row, this._columnTree[this._columnTree.length - 1]);
                this._a11yUpdateRowDetailsOpened(row, false);
              }
              // Clear any old template instances
              delete row.querySelector('[part~=details-cell]')._instance;
            });
          }

          if (this.detailsOpenedItems.length) {
            Array.from(this.$.items.children).forEach(this._toggleDetailsCell, this);
            this._update();
          }
        }
      }

      _detailsOpenedItemsChanged(changeRecord, rowDetailsTemplate, rowDetailsRenderer) {
        if (changeRecord.path === 'detailsOpenedItems.length' || !changeRecord.value) {
          // Let’s avoid duplicate work of both “.splices” and “.length” updates.
          return;
        }
        Array.from(this.$.items.children).forEach(row => {
          this._toggleDetailsCell(row, row._item);
          this._a11yUpdateRowDetailsOpened(row, this._isDetailsOpened(row._item));
          this._toggleAttribute('details-opened', this._isDetailsOpened(row._item), row);
        });
      }

      _configureDetailsCell(cell) {
        cell.setAttribute('part', 'cell details-cell');
        // Freeze the details cell, so that it does not scroll horizontally
        // with the normal cells. This way it looks less weird.
        this._toggleAttribute('frozen', true, cell);
      }

      _toggleDetailsCell(row, item) {
        const cell = row.querySelector('[part~="details-cell"]');
        if (!cell) {
          return;
        }
        const detailsHidden = !this._isDetailsOpened(item);
        const hiddenChanged = !!cell.hidden !== detailsHidden;

        if (!cell._instance && !cell._renderer || cell.hidden !== detailsHidden) {
          cell.hidden = detailsHidden;
          if (detailsHidden) {
            row.style.removeProperty('padding-bottom');
          } else {
            if (this.rowDetailsRenderer) {
              cell._renderer = this.rowDetailsRenderer;
              cell._renderer.call(this, cell._content, this, {index: row.index, item: item});
            } else if (this._rowDetailsTemplate && !cell._instance) {
              // Stamp the template
              cell._instance = this._rowDetailsTemplate.templatizer.createInstance();
              cell._content.innerHTML = '';
              cell._content.appendChild(cell._instance.root);
              this._updateItem(row, item);
            }

            flush();
            row.style.setProperty('padding-bottom', `${cell.offsetHeight}px`);

            requestAnimationFrame(() => this.notifyResize());
          }
        }
        if (hiddenChanged) {
          this._updateMetrics();
          this._positionItems();
        }
      }

      _updateDetailsCellHeights() {
        Array.from(this.$.items.querySelectorAll('[part~="details-cell"]:not([hidden])')).forEach(cell => {
          cell.parentElement.style.setProperty('padding-bottom', `${cell.offsetHeight}px`);
        });
      }

      _isDetailsOpened(item) {
        return this.detailsOpenedItems && this._getItemIndexInArray(item, this.detailsOpenedItems) !== -1;
      }

      /**
       * Open the details row of a given item.
       */
      openItemDetails(item) {
        if (!this._isDetailsOpened(item)) {
          this.push('detailsOpenedItems', item);
        }
      }

      /**
       * Close the details row of a given item.
       */
      closeItemDetails(item) {
        if (this._isDetailsOpened(item)) {
          this.splice('detailsOpenedItems', this._getItemIndexInArray(item, this.detailsOpenedItems), 1);
        }
      }

      _detailsOpenedInstanceChangedCallback(instance, value) {
        if (super._detailsOpenedInstanceChangedCallback) {
          super._detailsOpenedInstanceChangedCallback(instance, value);
        }
        if (value) {
          this.openItemDetails(instance.item);
        } else {
          this.closeItemDetails(instance.item);
        }
      }
    };

    /**
    @license
    Copyright (c) 2017 Vaadin Ltd.
    This program is available under Apache License Version 2.0, available at https://vaadin.com/license/
    */

    /**
     * @polymerMixin
     */
    const ScrollMixin = superClass => class ScrollMixin extends superClass {

      get _timeouts() {
        return {
          SCROLL_PERIOD: 1000,
          WHEEL_SCROLLING: 200,
          SCROLLING: 100,
          IGNORE_WHEEL: 500
        };
      }

      static get properties() {
        return {

          // Cached array of frozen cells
          _frozenCells: {
            type: Array,
            value: function() {
              return [];
            },
          },

          _scrollbarWidth: {
            type: Number,
            value: function() {
              // Create the measurement node
              var scrollDiv = document.createElement('div');
              scrollDiv.style.width = '100px';
              scrollDiv.style.height = '100px';
              scrollDiv.style.overflow = 'scroll';
              scrollDiv.style.position = 'absolute';
              scrollDiv.style.top = '-9999px';
              document.body.appendChild(scrollDiv);
              // Get the scrollbar width
              var scrollbarWidth = scrollDiv.offsetWidth - scrollDiv.clientWidth;
              // Delete the DIV
              document.body.removeChild(scrollDiv);
              return scrollbarWidth;
            }
          },

          _rowWithFocusedElement: Element,

          _deltaYAcc: {
            type: Number,
            value: 0
          }

        };
      }

      static get observers() {
        return [
          '_scrollHeightUpdated(_estScrollHeight)',
          '_scrollViewportHeightUpdated(_viewportHeight)'
        ];
      }

      // Override (from iron-scroll-target-behavior) to avoid document scroll
      set _scrollTop(top) {
        this.$.table.scrollTop = top;
      }

      get _scrollTop() {
        return this.$.table.scrollTop;
      }

      constructor() {
        super();
        this._scrollLineHeight = this._getScrollLineHeight();
      }

      /**
       * @returns {Number|undefined} - The browser's default font-size in pixels
       */
      _getScrollLineHeight() {
        const el = document.createElement('div');
        el.style.fontSize = 'initial';
        el.style.display = 'none';
        document.body.appendChild(el);
        const fontSize = window.getComputedStyle(el).fontSize;
        document.body.removeChild(el);
        return fontSize ? window.parseInt(fontSize) : undefined;
      }

      _scrollViewportHeightUpdated(_viewportHeight) {
        this._scrollPageHeight = _viewportHeight - this.$.header.clientHeight - this.$.footer.clientHeight - this._scrollLineHeight;
      }

      ready() {
        super.ready();
        this.scrollTarget = this.$.table;

        this.addEventListener('wheel', e => {
          this._wheelScrolling = true;
          this._debouncerWheelScrolling = Debouncer.debounce(
            this._debouncerWheelScrolling,
            timeOut.after(this._timeouts.WHEEL_SCROLLING),
            () => this._wheelScrolling = false
          );
          this._onWheel(e);
        });

        this.$.table.addEventListener('scroll', e => {
          if (this.$.outerscroller.outerScrolling) {
            e.stopImmediatePropagation();
          }
        }, true);

        this.$.items.addEventListener('focusin', (e) => {
          const itemsIndex = e.composedPath().indexOf(this.$.items);
          this._rowWithFocusedElement = e.composedPath()[itemsIndex - 1];
        });
        this.$.items.addEventListener('focusout', () => this._rowWithFocusedElement = undefined);
      }

      /**
       * Scroll to a specific row index in the virtual list. Note that the row index is
       * not always the same for any particular item. For example, sorting/filtering/expanding
       * or collapsing hierarchical items can affect the row index related to an item.
       *
       * @param {number} index Row index to scroll to
       */
      scrollToIndex(index) {
        this._accessIronListAPI(() => super.scrollToIndex(index));
      }

      _onWheel(e) {
        if (e.ctrlKey || this._hasScrolledAncestor(e.target, e.deltaX, e.deltaY)) {
          return;
        }

        const table = this.$.table;

        let deltaY = e.deltaY;
        if (e.deltaMode === WheelEvent.DOM_DELTA_LINE) {
          // Scrolling by "lines of text" instead of pixels
          deltaY *= this._scrollLineHeight;
        } else if (e.deltaMode === WheelEvent.DOM_DELTA_PAGE) {
          // Scrolling by "pages" instead of pixels
          deltaY *= this._scrollPageHeight;
        }

        if (this._wheelAnimationFrame) {
          // Skip new wheel events while one is being processed
          this._deltaYAcc += deltaY;
          e.preventDefault();
          return;
        }

        deltaY += this._deltaYAcc;
        this._deltaYAcc = 0;

        this._wheelAnimationFrame = true;
        this._debouncerWheelAnimationFrame = Debouncer.debounce(
          this._debouncerWheelAnimationFrame,
          animationFrame,
          () => this._wheelAnimationFrame = false
        );

        var momentum = Math.abs(e.deltaX) + Math.abs(deltaY);

        if (this._canScroll(table, e.deltaX, deltaY)) {
          e.preventDefault();
          table.scrollTop += deltaY;
          table.scrollLeft += e.deltaX;
          this._scrollHandler();
          this._hasResidualMomentum = true;

          this._ignoreNewWheel = true;
          this._debouncerIgnoreNewWheel = Debouncer.debounce(
            this._debouncerIgnoreNewWheel,
            timeOut.after(this._timeouts.IGNORE_WHEEL),
            () => this._ignoreNewWheel = false
          );
        } else if (this._hasResidualMomentum && momentum <= this._previousMomentum || this._ignoreNewWheel) {
          e.preventDefault();
        } else if (momentum > this._previousMomentum) {
          this._hasResidualMomentum = false;
        }
        this._previousMomentum = momentum;
      }

      /**
       * Determines if the element has an ancestor prior to this
       * cell content that handles the scroll delta
       */
      _hasScrolledAncestor(el, deltaX, deltaY) {
        if (el.localName === 'vaadin-grid-cell-content') {
          return false;
        } else if (this._canScroll(el, deltaX, deltaY)
          && ['auto', 'scroll'].indexOf(getComputedStyle(el).overflow) !== -1) {
          return true;
        } else if (el !== this && el.parentElement) {
          return this._hasScrolledAncestor(el.parentElement, deltaX, deltaY);
        }
      }

      /**
       * Determines if the the given scroll deltas can be applied to the element
       * (fully or partially)
       */
      _canScroll(el, deltaX, deltaY) {
        return (deltaY > 0 && el.scrollTop < el.scrollHeight - el.offsetHeight) ||
        (deltaY < 0 && el.scrollTop > 0) ||
        (deltaX > 0 && el.scrollLeft < el.scrollWidth - el.offsetWidth) ||
        (deltaX < 0 && el.scrollLeft > 0);
      }

      _scheduleScrolling() {
        if (!this._scrollingFrame) {
          // Defer setting state attributes to avoid Edge hiccups
          this._scrollingFrame = requestAnimationFrame(() => this._toggleAttribute('scrolling', true, this.$.scroller));
        }
        this._debounceScrolling = Debouncer.debounce(
          this._debounceScrolling,
          timeOut.after(this._timeouts.SCROLLING),
          () => {
            cancelAnimationFrame(this._scrollingFrame);
            delete this._scrollingFrame;
            this._toggleAttribute('scrolling', false, this.$.scroller);
            if (!this.$.outerscroller.outerScrolling) {
              this._reorderRows();
            }
          }
        );

        if (!this._scrollPeriodFrame) {
          this._scrollPeriodFrame = requestAnimationFrame(() => this._toggleAttribute('scroll-period', true, this.$.scroller));
        }
        this._debounceScrollPeriod = Debouncer.debounce(
          this._debounceScrollPeriod,
          timeOut.after(this._timeouts.SCROLL_PERIOD),
          () => {
            cancelAnimationFrame(this._scrollPeriodFrame);
            delete this._scrollPeriodFrame;
            this._toggleAttribute('scroll-period', false, this.$.scroller);
          }
        );
      }

      _afterScroll() {
        this._translateStationaryElements();

        if (!this.hasAttribute('reordering')) {
          this._scheduleScrolling();
        }

        const os = this.$.outerscroller;
        if (!this._ios && (this._wheelScrolling || os.passthrough)) {
          os.syncOuterScroller();
        }

        if (this._ios) {
          // Enable vertical rubberband feedback
          const overScroll = Math.max(-os.scrollTop, 0) ||
            Math.min(0, os.scrollHeight - os.scrollTop - os.offsetHeight);
          this.$.items.style.transform = `translateY(${overScroll}px)`;
        }

        this._updateOverflow();
      }

      _updateOverflow() {
        // Set overflow styling attributes
        let overflow = '';
        const table = this.$.table;
        if (table.scrollTop < table.scrollHeight - table.clientHeight) {
          overflow += ' bottom';
        }

        if (table.scrollTop > 0) {
          overflow += ' top';
        }

        if (table.scrollLeft < table.scrollWidth - table.clientWidth) {
          overflow += ' right';
        }

        if (table.scrollLeft > 0) {
          overflow += ' left';
        }

        this._debounceOverflow = Debouncer.debounce(
          this._debounceOverflow,
          animationFrame,
          () => {
            const value = overflow.trim();
            if (value.length > 0 && this.getAttribute('overflow') !== value) {
              this.setAttribute('overflow', value);
            } else if (value.length == 0 && this.hasAttribute('overflow')) {
              this.removeAttribute('overflow');
            }
          }
        );
      }

      // correct order needed for preserving correct tab order between cell contents.
      _reorderRows() {
        const body = this.$.items;
        const items = body.querySelectorAll('tr');
        if (!items.length) {
          return;
        }

        const adjustedVirtualStart = this._virtualStart + this._vidxOffset;

        // Which row to use as a target?
        const targetRow = this._rowWithFocusedElement || Array.from(items).filter(row => !row.hidden)[0];
        if (!targetRow) {
          // All rows are hidden, don't reorder
          return;
        }

        // Where the target row should be?
        const targetPhysicalIndex = targetRow.index - adjustedVirtualStart;

        // Reodrer the DOM elements to keep the target row at the target physical index
        const delta = Array.from(items).indexOf(targetRow) - targetPhysicalIndex;
        if (delta > 0) {
          for (let i = 0; i < delta; i++) {
            body.appendChild(items[i]);
          }
        } else if (delta < 0) {
          for (let i = items.length + delta; i < items.length; i++) {
            body.insertBefore(items[i], items[0]);
          }
        }
      }

      _frozenCellsChanged() {
        this._debouncerCacheElements = Debouncer.debounce(
          this._debouncerCacheElements,
          microTask,
          () => {
            Array.from(this.root.querySelectorAll('[part~="cell"]')).forEach(function(cell) {
              cell.style.transform = '';
            });
            this._frozenCells = Array.prototype.slice.call(this.$.table.querySelectorAll('[frozen]'));
            this._updateScrollerMeasurements();
            this._translateStationaryElements();
          }
        );
        this._updateLastFrozen();
      }

      _updateScrollerMeasurements() {
        if (this._frozenCells.length > 0 && this.__isRTL) {
          this.__scrollerMetrics = {
            scrollWidth: this.$.outerscroller.scrollWidth,
            clientWidth: this.$.outerscroller.clientWidth
          };
        }
      }

      _updateLastFrozen() {
        if (!this._columnTree) {
          return;
        }

        const columnsRow = this._columnTree[this._columnTree.length - 1].slice(0);
        columnsRow.sort((a, b) => {
          return a._order - b._order;
        });
        const lastFrozen = columnsRow.reduce((prev, col, index) => {
          col._lastFrozen = false;
          return col.frozen && !col.hidden ? index : prev;
        }, undefined);
        if (lastFrozen !== undefined) {
          columnsRow[lastFrozen]._lastFrozen = true;
        }
      }

      _translateStationaryElements() {
        if (this._edge && !this._scrollbarWidth) {
          // Fixed mode (Tablet Edge)
          this.$.items.style.transform =
          this._getTranslate(-this._scrollLeft || 0, -this._scrollTop || 0);

          this.$.footer.style.transform = this.$.header.style.transform =
          this._getTranslate(-this._scrollLeft || 0, 0);
        } else {
          this.$.footer.style.transform = this.$.header.style.transform = this._getTranslate(0, this._scrollTop);
        }

        if (this._frozenCells.length > 0) {
          const x = this.__isRTL ?
            this.__getNormalizedScrollLeft(this.$.table) + this.__scrollerMetrics.clientWidth -
            this.__scrollerMetrics.scrollWidth
            : this._scrollLeft;
          var frozenCellTransform = this._getTranslate(x, 0);
          for (var i = 0; i < this._frozenCells.length; i++) {
            this._frozenCells[i].style.transform = frozenCellTransform;
          }
        }
      }

      _getTranslate(x, y) {
        return 'translate(' + x + 'px,' + y + 'px)';
      }

      _scrollHeightUpdated(_estScrollHeight) {
        this.$.outersizer.style.top = this.$.fixedsizer.style.top = _estScrollHeight + 'px';
      }

    };

    /**
    @license
    Copyright (c) 2017 Vaadin Ltd.
    This program is available under Apache License Version 2.0, available at https://vaadin.com/license/
    */
    /**
     * @polymerMixin
     */
    const SelectionMixin = superClass => class SelectionMixin extends superClass {

      static get properties() {
        return {
          /**
           * An array that contains the selected items.
           */
          selectedItems: {
            type: Object,
            notify: true,
            value: () => []
          }
        };
      }

      static get observers() {
        return [
          '_selectedItemsChanged(selectedItems.*)'
        ];
      }

      _isSelected(item) {
        return this.selectedItems && this._getItemIndexInArray(item, this.selectedItems) > -1;
      }

      /**
       * Selects the given item.
       *
       * @method selectItem
       * @param {Object} item The item object
       */
      selectItem(item) {
        if (!this._isSelected(item)) {
          this.push('selectedItems', item);
        }
      }

      /**
       * Deselects the given item if it is already selected.
       *
       * @method deselect
       * @param {Object} item The item object
       */
      deselectItem(item) {
        const index = this._getItemIndexInArray(item, this.selectedItems);
        if (index > -1) {
          this.splice('selectedItems', index, 1);
        }
      }

      /**
       * Toggles the selected state of the given item.
       *
       * @method toggle
       * @param {Object} item The item object
       */
      _toggleItem(item) {
        const index = this._getItemIndexInArray(item, this.selectedItems);
        if (index === -1) {
          this.selectItem(item);
        } else {
          this.deselectItem(item);
        }
      }

      _selectedItemsChanged(e) {
        if (this.$.items.children.length && (e.path === 'selectedItems' || e.path === 'selectedItems.splices')) {
          Array.from(this.$.items.children).forEach(row => {
            this._updateItem(row, row._item);
          });
        }
      }

      _selectedInstanceChangedCallback(instance, value) {
        if (super._selectedInstanceChangedCallback) {
          super._selectedInstanceChangedCallback(instance, value);
        }
        if (value) {
          this.selectItem(instance.item);
        } else {
          this.deselectItem(instance.item);
        }
      }
    };

    /**
    @license
    Copyright (c) 2017 Vaadin Ltd.
    This program is available under Apache License Version 2.0, available at https://vaadin.com/license/
    */

    /**
     * @polymerMixin
     */
    const SortMixin = superClass => class SortMixin extends superClass {
      static get properties() {
        return {
          /**
           * When `true`, all `<vaadin-grid-sorter>` are applied for sorting.
           */
          multiSort: {
            type: Boolean,
            value: false
          },

          _sorters: {
            type: Array,
            value: function() {
              return [];
            }
          },

          _previousSorters: {
            type: Array,
            value: function() {
              return [];
            }
          }
        };
      }

      ready() {
        super.ready();
        this.addEventListener('sorter-changed', this._onSorterChanged);

        // With Polymer 2 & shady the 'sorter-changed' listener isn't guaranteed to be registered
        // before child <vaadin-grid-sorter>'s upgrade and fire the events. The following
        // makes sure that 'sorter-changed' is fired for all <vaadin-grid-sorter> elements
        // after this (<vaadin-grid>) is ready (and the listeners are active).
        if (window.ShadyDOM) {
          microTask.run(() => {
            const sorters = this.querySelectorAll('vaadin-grid-sorter');
            Array.from(sorters).forEach((sorter) => {
              // Don't try to fire if the sorter hasn't been upgraded yet
              if (sorter instanceof PolymerElement) {
                sorter.dispatchEvent(new CustomEvent('sorter-changed', {bubbles: true, composed: true}));
              }
            });
          });
        }
      }

      _onSorterChanged(e) {
        const sorter = e.target;

        this._removeArrayItem(this._sorters, sorter);
        sorter._order = null;

        if (this.multiSort) {
          if (sorter.direction) {
            this._sorters.unshift(sorter);
          }

          this._sorters.forEach((sorter, index) => sorter._order = this._sorters.length > 1 ? index : null, this);
        } else {
          if (sorter.direction) {
            this._sorters.forEach(sorter => {
              sorter._order = null;
              sorter.direction = null;
            });
            this._sorters = [sorter];
          }
        }

        e.stopPropagation();

        if (this.dataProvider &&
          // No need to clear cache if sorters didn't change
          JSON.stringify(this._previousSorters) !== JSON.stringify(this._mapSorters())) {
          this.clearCache();
        }

        this._a11yUpdateSorters();

        this._previousSorters = this._mapSorters();
      }

      _mapSorters() {
        return this._sorters.map(sorter => {
          return {
            path: sorter.path,
            direction: sorter.direction
          };
        });
      }

      _removeArrayItem(array, item) {
        const index = array.indexOf(item);
        if (index > -1) {
          array.splice(index, 1);
        }
      }
    };

    /**
    @license
    Copyright (c) 2018 Vaadin Ltd.
    This program is available under Apache License Version 2.0, available at https://vaadin.com/license/
    */
    /**
     * @polymerMixin
     */
    const StylingMixin = superClass => class StylingMixin extends superClass {

      static get properties() {
        return {
          /**
           * A function that allows generating CSS class names for grid cells
           * based on their row and column. The return value should be the generated
           * class name as a string, or multiple class names separated by whitespace
           * characters.
           *
           * Receives two arguments:
           * - `column` The `<vaadin-grid-column>` element (`undefined` for details-cell).
           * - `rowData` The object with the properties related with
           *   the rendered item, contains:
           *   - `rowData.index` The index of the item.
           *   - `rowData.item` The item.
           *   - `rowData.expanded` Sublevel toggle state.
           *   - `rowData.level` Level of the tree represented with a horizontal offset of the toggle button.
           *   - `rowData.selected` Selected state.
           */
          cellClassNameGenerator: Function
        };
      }

      static get observers() {
        return [
          '__cellClassNameGeneratorChanged(cellClassNameGenerator)'
        ];
      }

      __cellClassNameGeneratorChanged(cellClassGenerator) {
        this.generateCellClassNames();
      }

      /**
       * Runs the `cellClassNameGenerator` for the visible cells.
       * If the generator depends on varying conditions, you need to
       * call this function manually in order to update the styles when
       * the conditions change.
       */
      generateCellClassNames() {
        Array.from(this.$.items.children).filter(row => !row.hidden).forEach(
          row => this._generateCellClassNames(row, this.__getRowModel(row)));
      }

      _generateCellClassNames(row, rowData) {
        Array.from(row.children).forEach(cell => {
          if (cell.__generatedClasses) {
            cell.__generatedClasses.forEach(className => cell.classList.remove(className));
          }
          if (this.cellClassNameGenerator) {
            const result = this.cellClassNameGenerator(cell._column, rowData);
            cell.__generatedClasses = result && result.split(' ').filter(className => className.length > 0);
            if (cell.__generatedClasses) {
              cell.__generatedClasses.forEach(className => cell.classList.add(className));
            }
          }
        });
      }
    };

    /**
    @license
    Copyright (c) 2019 Vaadin Ltd.
    This program is available under Apache License Version 2.0, available at https://vaadin.com/license/
    */
    const DropMode = {
      BETWEEN: 'between',
      ON_TOP: 'on-top',
      ON_TOP_OR_BETWEEN: 'on-top-or-between',
      ON_GRID: 'on-grid'
    };

    const DropLocation = {
      ON_TOP: 'on-top',
      ABOVE: 'above',
      BELOW: 'below',
      EMPTY: 'empty'
    };

    /**
     * @polymerMixin
     */
    const DragAndDropMixin = superClass => class DragAndDropMixin extends superClass {

      static get properties() {
        return {
          /**
           * Defines the locations within the Grid row where an element can be dropped.
           *
           * Possible values are:
           * - `between`: The drop event can happen between Grid rows.
           * - `on-top`: The drop event can happen on top of Grid rows.
           * - `on-top-or-between`: The drop event can happen either on top of or between Grid rows.
           * - `on-grid`: The drop event will not happen on any specific row, it will show the drop target outline around the whole grid.
           */
          dropMode: String,

          /**
           * Marks the grid's rows to be available for dragging.
           */
          rowsDraggable: Boolean,

          /**
           * A function that filters dragging of specific grid rows. The return value should be false
           * if dragging of the row should be disabled.
           *
           * Receives one argument:
           * - `rowData` The object with the properties related with
           *   the rendered item, contains:
           *   - `rowData.index` The index of the item.
           *   - `rowData.item` The item.
           *   - `rowData.expanded` Sublevel toggle state.
           *   - `rowData.level` Level of the tree represented with a horizontal offset of the toggle button.
           *   - `rowData.selected` Selected state.
           */
          dragFilter: Function,

          /**
           * A function that filters dropping on specific grid rows. The return value should be false
           * if dropping on the row should be disabled.
           *
           * Receives one argument:
           * - `rowData` The object with the properties related with
           *   the rendered item, contains:
           *   - `rowData.index` The index of the item.
           *   - `rowData.item` The item.
           *   - `rowData.expanded` Sublevel toggle state.
           *   - `rowData.level` Level of the tree represented with a horizontal offset of the toggle button.
           *   - `rowData.selected` Selected state.
           */
          dropFilter: Function,

          __dndAutoScrollThreshold: {
            value: 50
          }

        };
      }

      static get observers() {
        return [
          '_dragDropAccessChanged(rowsDraggable, dropMode, dragFilter, dropFilter)'
        ];
      }

      ready() {
        super.ready();
        this.$.table.addEventListener('dragstart', this._onDragStart.bind(this));
        this.$.table.addEventListener('dragend', this._onDragEnd.bind(this));
        this.$.table.addEventListener('dragover', this._onDragOver.bind(this));
        this.$.table.addEventListener('dragleave', this._onDragLeave.bind(this));
        this.$.table.addEventListener('drop', this._onDrop.bind(this));
        this.$.table.addEventListener('dragenter', e => {
          if (this.dropMode) {
            e.preventDefault();
            e.stopPropagation();
          }
        });

      }

      _onDragStart(e) {
        if (this.rowsDraggable) {
          let row = e.target;
          if (row.localName === 'vaadin-grid-cell-content') {
            // The draggable node is the cell content element on browsers that support native shadow
            row = row.assignedSlot.parentNode.parentNode;
          }
          if (row.parentNode !== this.$.items) {
            return;
          }

          e.stopPropagation();
          this._toggleAttribute('dragging-rows', true, this);

          if (this._safari) {
            // Safari doesn't get proper drag images from transformed
            // elements so we need to switch to top temporarily
            const transform = row.style.transform;
            row.style.top = /translateY\((.*)\)/.exec(transform)[1];
            row.style.transform = 'none';
            requestAnimationFrame(() => {
              row.style.top = '';
              row.style.transform = transform;
            });
          }

          const rowRect = row.getBoundingClientRect();
          if (!window.ShadyDOM) {
            if (this._ios) {
              e.dataTransfer.setDragImage(row);
            } else {
              e.dataTransfer.setDragImage(row, e.clientX - rowRect.left, e.clientY - rowRect.top);
            }

          }

          let rows = [row];
          if (this._isSelected(row._item)) {
            rows = this.__getViewportRows()
              .filter(row => this._isSelected(row._item))
              .filter(row => !this.dragFilter || this.dragFilter(this.__getRowModel(row)));
          }

          // Set the default transfer data
          e.dataTransfer.setData('text', this.__formatDefaultTransferData(rows));

          row.setAttribute('dragstart', rows.length > 1 ? rows.length : '');
          this.updateStyles({
            '--_grid-drag-start-x': `${e.clientX - rowRect.left + 20}px`,
            '--_grid-drag-start-y': `${e.clientY - rowRect.top + 10}px`
          });

          requestAnimationFrame(() => {
            row.removeAttribute('dragstart');
            this.updateStyles({'--_grid-drag-start-x': '', '--_grid-drag-start-y': ''});
          });

          const event = new CustomEvent('grid-dragstart', {
            detail: {
              draggedItems: rows.map(row => row._item),
              setDragData: (type, data) => e.dataTransfer.setData(type, data),
              setDraggedItemsCount: count => row.setAttribute('dragstart', count)
            }
          });
          event.originalEvent = e;
          this.dispatchEvent(event);
        }
      }

      _onDragEnd(e) {
        this._toggleAttribute('dragging-rows', false, this);
        e.stopPropagation();
        const event = new CustomEvent('grid-dragend');
        event.originalEvent = e;
        this.dispatchEvent(event);
      }

      _onDragLeave(e) {
        e.stopPropagation();
        this._clearDragStyles();
      }

      _onDragOver(e) {

        if (this.dropMode) {
          this._dropLocation = undefined;
          this._dragOverItem = undefined;

          if (this.__dndAutoScroll(e.clientY)) {
            this._clearDragStyles();
            return;
          }

          let row = e.composedPath().filter(node => node.localName === 'tr')[0];

          if (!this._effectiveSize || this.dropMode === DropMode.ON_GRID) {
            // The grid is empty or "on-grid" drop mode was used, always default to "empty"
            this._dropLocation = DropLocation.EMPTY;
          } else if (!row || row.parentNode !== this.$.items) {
            // The dragover didn't occur on a body row but the grid has items
            if (row) {
              // The dragover occurred over a header/footer row
              return;
            } else if (this.dropMode === DropMode.BETWEEN || this.dropMode === DropMode.ON_TOP_OR_BETWEEN) {
              // The drop mode allows setting the last row as the drag over item
              row = Array.from(this.$.items.children).filter(row => !row.hidden).pop();
              this._dropLocation = DropLocation.BELOW;
            } else {
              // Drop mode on-top used but the dragover didn't occur over one of the existing rows
              return;
            }
          } else {
            // The dragover occurred on a body row, determine the drop location from coordinates
            const rowRect = row.getBoundingClientRect();

            this._dropLocation = DropLocation.ON_TOP;

            if (this.dropMode === DropMode.BETWEEN) {
              const dropAbove = e.clientY - rowRect.top < rowRect.bottom - e.clientY;
              this._dropLocation = dropAbove ? DropLocation.ABOVE : DropLocation.BELOW;

            } else if (this.dropMode === DropMode.ON_TOP_OR_BETWEEN) {
              if (e.clientY - rowRect.top < rowRect.height / 3) {
                this._dropLocation = DropLocation.ABOVE;
              } else if (e.clientY - rowRect.top > (rowRect.height / 3) * 2) {
                this._dropLocation = DropLocation.BELOW;
              }
            }
          }

          if (row && row.hasAttribute('drop-disabled')) {
            this._dropLocation = undefined;
            return;
          }

          e.stopPropagation();
          e.preventDefault();

          if (this._dropLocation === DropLocation.EMPTY) {
            this._toggleAttribute('dragover', true, this);
          } else if (row) {
            this._dragOverItem = row._item;
            if (row.getAttribute('dragover') !== this._dropLocation) {
              row.setAttribute('dragover', this._dropLocation);
            }
          } else {
            this._clearDragStyles();
          }
        }
      }

      __dndAutoScroll(clientY) {
        if (this.__dndAutoScrolling) {
          return true;
        }

        const headerBottom = this.$.header.getBoundingClientRect().bottom;
        const footerTop = this.$.footer.getBoundingClientRect().top;
        const topDiff = headerBottom - clientY + this.__dndAutoScrollThreshold;
        const bottomDiff = clientY - footerTop + this.__dndAutoScrollThreshold;
        let scrollTopDelta = 0;

        if (bottomDiff > 0) {
          scrollTopDelta = bottomDiff * 2;
        } else if (topDiff > 0) {
          scrollTopDelta = -topDiff * 2;
        }

        if (scrollTopDelta) {
          const scrollTop = this.$.table.scrollTop;
          this.$.table.scrollTop += scrollTopDelta;
          const scrollTopChanged = scrollTop !== this.$.table.scrollTop;
          if (scrollTopChanged) {
            this.__dndAutoScrolling = true;
            // Disallow more auto-scrolls within 20ms
            setTimeout(() => this.__dndAutoScrolling = false, 20);
            this._scrollHandler();
            return true;
          }
        }
      }

      __getViewportRows() {
        const headerBottom = this.$.header.getBoundingClientRect().bottom;
        const footerTop = this.$.footer.getBoundingClientRect().top;
        return Array.from(this.$.items.children)
          .filter(row => {
            const rowRect = row.getBoundingClientRect();
            return rowRect.bottom > headerBottom && rowRect.top < footerTop;
          });
      }

      _clearDragStyles() {
        this.removeAttribute('dragover');
        Array.from(this.$.items.children).forEach(row => row.removeAttribute('dragover'));
      }

      _onDrop(e) {
        if (this.dropMode) {
          e.stopPropagation();
          e.preventDefault();

          const dragData = e.dataTransfer.types && Array.from(e.dataTransfer.types).map(type => {
            return {
              type,
              data: e.dataTransfer.getData(type)
            };
          });

          this._clearDragStyles();

          const event = new CustomEvent('grid-drop', {
            bubbles: e.bubbles,
            cancelable: e.cancelable,
            detail: {
              dropTargetItem: this._dragOverItem,
              dropLocation: this._dropLocation,
              dragData
            }
          });
          event.originalEvent = e;
          this.dispatchEvent(event);
        }
      }

      __formatDefaultTransferData(rows) {
        return rows
          .map(row => {
            return Array.from(row.children)
              .filter(cell => !cell.hidden && cell.getAttribute('part').indexOf('details-cell') === -1)
              .sort((a, b) => {
                return a._column._order > b._column._order ? 1 : -1;
              })
              .map(cell => cell._content.textContent.trim())
              .filter(content => content)
              .join('\t');
          })
          .join('\n');
      }

      _dragDropAccessChanged(rowsDraggable, dropMode, dragFilter, dropFilter) {
        this.filterDragAndDrop();
      }

      /**
       * Runs the `dragFilter` and `dropFilter` hooks for the visible cells.
       * If the filter depends on varying conditions, you may need to
       * call this function manually in order to update the draggability when
       * the conditions change.
       */
      filterDragAndDrop() {
        Array.from(this.$.items.children).filter(row => !row.hidden).forEach(
          row => {
            this._filterDragAndDrop(row, this.__getRowModel(row));
          }
        );
      }

      _filterDragAndDrop(row, rowData) {
        const dragDisabled = !this.rowsDraggable || (this.dragFilter && !this.dragFilter(rowData));
        const dropDisabled = !this.dropMode || (this.dropFilter && !this.dropFilter(rowData));

        const draggableElements = window.ShadyDOM
          ? [row]
          : Array.from(row.children).map(cell => cell._content);

        draggableElements.forEach(e => {
          if (dragDisabled) {
            e.removeAttribute('draggable');
          } else {
            e.setAttribute('draggable', true);
          }
        });

        this._toggleAttribute('drag-disabled', dragDisabled, row);
        this._toggleAttribute('drop-disabled', dropDisabled, row);
      }

      /**
       * Fired when starting to drag grid rows.
       *
       * @event grid-dragstart
       * @param {Object} originalEvent The native dragstart event
       * @param {Object} detail
       * @param {Object} detail.draggedItems the items in the visible viewport that are dragged
       * @param {Function} detail.setDraggedItemsCount Overrides the default number shown in the drag image on multi row drag.
       * Parameter is of type number.
       * @param {Function} detail.setDragData Sets dataTransfer data for the drag operation.
       * Note that "text" is the only data type supported by all the browsers the grid currently supports (including IE11).
       * The function takes two parameters:
       * - type:string The type of the data
       * - data:string The data
       */

      /**
       * Fired when the dragging of the rows ends.
       *
       * @event grid-dragend
       * @param {Object} originalEvent The native dragend event
       */

      /**
       * Fired when a drop occurs on top of the grid.
       *
       * @event grid-drop
       * @param {Object} originalEvent The native drop event
       * @param {Object} detail
       * @param {Object} detail.dropTargetItem The item of the grid row on which the drop occurred.
       * @param {string} detail.dropLocation The position at which the drop event took place relative to a row.
       * Depending on the dropMode value, the drop location can be one of the following
       * - `on-top`: when the drop occurred on top of the row
       * - `above`: when the drop occurred above the row
       * - `below`: when the drop occurred below the row
       * - `empty`: when the drop occurred over the grid, not relative to any specific row
       * @param {string} detail.dragData An array of items with the payload as a string representation as the
       * `data` property and the type of the data as `type` property.
       */
    };

    /**
    @license
    Copyright (c) 2017 Vaadin Ltd.
    This program is available under Apache License Version 2.0, available at https://vaadin.com/license/
    */
    /**
     * @polymerMixin
     */
    const KeyboardNavigationMixin = superClass => class KeyboardNavigationMixin extends superClass {
      static get properties() {
        return {
          _headerFocusable: {
            type: Object,
            observer: '_focusableChanged'
          },
          _itemsFocusable: {
            type: Object,
            observer: '_focusableChanged'
          },
          _footerFocusable: {
            type: Object,
            observer: '_focusableChanged'
          },
          _navigatingIsHidden: Boolean,
          _focusedItemIndex: {
            type: Number,
            value: 0
          },
          _focusedColumnOrder: Number
        };
      }

      ready() {
        super.ready();

        if (this._ios || this._android) {
          // Disable keyboard navigation on mobile devices
          return;
        }

        this.addEventListener('keydown', this._onKeyDown);
        this.addEventListener('keyup', this._onKeyUp);

        this.addEventListener('focusin', this._onFocusIn);
        this.addEventListener('focusout', this._onFocusOut);

        // When focus goes from cell to another cell, focusin/focusout events do
        // not escape the grid’s shadowRoot, thus listening inside the shadowRoot.
        this.$.table.addEventListener('focusin', this._onCellFocusIn.bind(this));
        this.$.table.addEventListener('focusout', this._onCellFocusOut.bind(this));

        this.addEventListener('mousedown', () => {
          this._toggleAttribute('navigating', false, this);
          this._isMousedown = true;
        });
        this.addEventListener('mouseup', () => this._isMousedown = false);
      }

      _focusableChanged(focusable, oldFocusable) {
        if (oldFocusable) {
          oldFocusable.setAttribute('tabindex', '-1');
        }
        if (focusable) {
          focusable.setAttribute('tabindex', '0');
        }
      }

      _onKeyDown(e) {
        // Ensure standard key value, unified across browsers
        let key = e.key;
        if (key === 'Up' || key === 'Down' || key === 'Left' || key === 'Right') {
          // MSIE & Edge
          key = 'Arrow' + key;
        }
        if (key === 'Esc') {
          // MSIE & Edge
          key = 'Escape';
        }
        if (key === 'Spacebar') {
          // MSIE
          key = ' ';
        }

        let keyGroup;
        switch (key) {
          case 'ArrowUp':
          case 'ArrowDown':
          case 'ArrowLeft':
          case 'ArrowRight':
          case 'PageUp':
          case 'PageDown':
          case 'Home':
          case 'End':
            keyGroup = 'Navigation';
            break;
          case 'Enter':
          case 'Escape':
          case 'F2':
            keyGroup = 'Interaction';
            break;
          case 'Tab':
            keyGroup = 'Tab';
            break;
          case ' ':
            keyGroup = 'Space';
            break;
        }

        this._detectInteracting(e);
        if (this.hasAttribute('interacting') && keyGroup !== 'Interaction') {
          // When in the interacting mode, only the “Interaction” keys are handled.
          keyGroup = undefined;
        }

        if (keyGroup) {
          this[`_on${keyGroup}KeyDown`](e, key);
        }
      }

      _ensureScrolledToIndex(index) {
        const targetRowInDom = Array.from(this.$.items.children).filter(child => child.index === index)[0];
        if (!targetRowInDom) {
          this._scrollToIndex(index);
        }
      }

      _onNavigationKeyDown(e, key) {
        e.preventDefault();

        function indexOfChildElement(el) {
          return Array.prototype.indexOf.call(el.parentNode.children, el);
        }

        const visibleItemsCount = this._lastVisibleIndex - this._firstVisibleIndex - 1;

        let dx = 0, dy = 0;
        switch (key) {
          case 'ArrowRight':
            dx = this.__isRTL ? -1 : 1;
            break;
          case 'ArrowLeft':
            dx = this.__isRTL ? 1 : -1;
            break;
          case 'Home':
            dx = -Infinity;
            e.ctrlKey && (dy = -Infinity);
            break;
          case 'End':
            dx = Infinity;
            e.ctrlKey && (dy = Infinity);
            break;
          case 'ArrowDown':
            dy = 1;
            break;
          case 'ArrowUp':
            dy = -1;
            break;
          case 'PageDown':
            dy = visibleItemsCount;
            break;
          case 'PageUp':
            dy = -visibleItemsCount;
            break;
        }

        const activeCell = e.composedPath()[0];
        const columnIndex = indexOfChildElement(activeCell);
        const isRowDetails = this._elementMatches(activeCell, '[part~="details-cell"]');

        const activeRow = activeCell.parentNode;

        const activeRowGroup = activeRow.parentNode;
        const maxRowIndex = (activeRowGroup === this.$.items ? this._effectiveSize : activeRowGroup.children.length) - 1;

        // Body rows have index property, otherwise DOM child index of the row is used.
        const rowIndex = (activeRowGroup === this.$.items) ?
          (this._focusedItemIndex !== undefined ? this._focusedItemIndex : activeRow.index) :
          indexOfChildElement(activeRow);

        // Index of the destination row
        let dstRowIndex = Math.max(0, Math.min(rowIndex + dy, maxRowIndex));

        // Row details navigation logic
        let dstIsRowDetails = false;
        if (activeRowGroup === this.$.items) {
          const item = activeRow._item;
          const dstItem = this._cache.getItemForIndex(dstRowIndex);
          // Should we navigate to row details?
          if (isRowDetails) {
            dstIsRowDetails = dy === 0;
          } else {
            dstIsRowDetails = dy === 1 && this._isDetailsOpened(item) ||
              dy === -1 && dstRowIndex !== rowIndex && this._isDetailsOpened(dstItem);
          }
          // Should we navigate between details and regular cells of the same row?
          if (dstIsRowDetails !== isRowDetails &&
              (dy === 1 && dstIsRowDetails || dy === -1 && !dstIsRowDetails)) {
            dstRowIndex = rowIndex;
          }
        }

        // Header and footer could have hidden rows, e. g., if none of the columns
        // or groups on the given column tree level define template. Skip them
        // in vertical keyboard navigation.
        if (activeRowGroup !== this.$.items) {
          if (dstRowIndex > rowIndex) {
            while (
              dstRowIndex < maxRowIndex &&
              activeRowGroup.children[dstRowIndex].hidden
            ) {
              dstRowIndex++;
            }
          } else if (dstRowIndex < rowIndex) {
            while (
              dstRowIndex > 0 &&
              activeRowGroup.children[dstRowIndex].hidden
            ) {
              dstRowIndex--;
            }
          }
        }

        // _focusedColumnOrder is memoized — this is to ensure predictable
        // navigation when entering and leaving detail and column group cells.
        if (this._focusedColumnOrder === undefined) {
          if (isRowDetails) {
            this._focusedColumnOrder = 0;
          } else {
            this._focusedColumnOrder = this._getColumns(activeRowGroup, rowIndex).filter(c => !c.hidden)[columnIndex]._order;
          }
        }

        // Find orderedColumnIndex — the index of order closest matching the
        // original _focusedColumnOrder in the sorted array of orders
        // of the visible columns on the destination row.
        const dstColumns = this._getColumns(activeRowGroup, dstRowIndex).filter(c => !c.hidden);
        const dstSortedColumnOrders = dstColumns.map(c => c._order)
          .sort((b, a) => (b - a));
        const maxOrderedColumnIndex = dstSortedColumnOrders.length - 1;
        const orderedColumnIndex = dstSortedColumnOrders.indexOf(
          dstSortedColumnOrders.slice(0).sort((b, a) =>
            Math.abs(b - this._focusedColumnOrder) - Math.abs(a - this._focusedColumnOrder)
          )[0]
        );

        // Index of the destination column order
        const dstOrderedColumnIndex = (dy === 0 && isRowDetails) ? orderedColumnIndex :
          Math.max(0, Math.min(orderedColumnIndex + dx, maxOrderedColumnIndex));

        if (dstOrderedColumnIndex !== orderedColumnIndex) {
          // Horizontal movement invalidates stored _focusedColumnOrder
          this._focusedColumnOrder = undefined;
        }

        // Ensure correct vertical scroll position, destination row is visible
        if (activeRowGroup === this.$.items) {
          this._ensureScrolledToIndex(dstRowIndex);
        }

        // This has to be set after scrolling, otherwise it can be removed by
        // `_preventScrollerRotatingCellFocus(item, index)` during scrolling.
        this._toggleAttribute('navigating', true, this);

        const columnIndexByOrder = dstColumns.reduce((acc, col, i) => (acc[col._order] = i, acc), {});
        const dstColumnIndex = columnIndexByOrder[dstSortedColumnOrders[dstOrderedColumnIndex]];

        // For body rows, use index property to find destination row, otherwise use DOM child index
        const dstRow = activeRowGroup === this.$.items ?
          Array.from(activeRowGroup.children).filter(el => el.index === dstRowIndex)[0] :
          activeRowGroup.children[dstRowIndex];
        if (!dstRow) {
          return;
        }

        // Here we go!
        const dstCell = dstIsRowDetails ?
          Array.from(dstRow.children)
            .filter(el => this._elementMatches(el, '[part~="details-cell"]'))[0] :
          dstRow.children[dstColumnIndex];
        this._scrollHorizontallyToCell(dstCell);
        if (activeRowGroup === this.$.items) {
          // When scrolling with repeated keydown, sometimes FocusEvent listeners
          // are too late to update _focusedItemIndex. Ensure next keydown
          // listener invocation gets updated _focusedItemIndex value.
          this._focusedItemIndex = dstRowIndex;
        }

        if (activeRowGroup === this.$.items) {
          const dstRect = dstCell.getBoundingClientRect();
          const footerTop = this.$.footer.getBoundingClientRect().top;
          const headerBottom = this.$.header.getBoundingClientRect().bottom;
          if (dstRect.bottom > footerTop) {
            this.$.table.scrollTop += dstRect.bottom - footerTop;
            this._scrollHandler();
          } else if (dstRect.top < headerBottom) {
            this.$.table.scrollTop -= headerBottom - dstRect.top;
            this._scrollHandler();
          }
        }

        dstCell.focus();
      }

      _parseEventPath(path) {
        const tableIndex = path.indexOf(this.$.table);
        return {
          rowGroup: path[tableIndex - 1],
          row: path[tableIndex - 2],
          cell: path[tableIndex - 3]
        };
      }

      _onInteractionKeyDown(e, key) {
        const localTarget = e.composedPath()[0];
        const localTargetIsTextInput = localTarget.localName === 'input' &&
          !/^(button|checkbox|color|file|image|radio|range|reset|submit)$/i.test(localTarget.type);

        let wantInteracting;
        switch (key) {
          case 'Enter':
            wantInteracting = this.hasAttribute('interacting') ? !localTargetIsTextInput : true;
            break;
          case 'Escape':
            wantInteracting = false;
            break;
          case 'F2':
            wantInteracting = !this.hasAttribute('interacting');
            break;
        }

        const {cell} = this._parseEventPath(e.composedPath());

        if (this.hasAttribute('interacting') !== wantInteracting) {
          if (wantInteracting) {
            const focusTarget = cell._content.querySelector('[focus-target]') ||
              cell._content.firstElementChild;
            if (focusTarget) {
              e.preventDefault();
              focusTarget.focus();
              this._toggleAttribute('interacting', true, this);
              this._toggleAttribute('navigating', false, this);
            }
          } else {
            e.preventDefault();
            this._focusedColumnOrder = undefined;
            cell.focus();
            this._toggleAttribute('interacting', false, this);
            this._toggleAttribute('navigating', true, this);
          }
        }
      }

      _predictFocusStepTarget(srcElement, step) {
        const tabOrder = [
          this.$.table,
          this._headerFocusable,
          this._itemsFocusable,
          this._footerFocusable,
          this.$.focusexit
        ];

        let index = tabOrder.indexOf(srcElement);

        index += step;
        while (index >= 0 && index <= tabOrder.length - 1 &&
            (!tabOrder[index] || tabOrder[index].parentNode.hidden)) {
          index += step;
        }

        return tabOrder[index];
      }

      _onTabKeyDown(e) {
        const focusTarget = this._predictFocusStepTarget(e.composedPath()[0], e.shiftKey ? -1 : 1);

        if (focusTarget === this.$.table) {
          // The focus is about to exit the grid to the top.
          this.$.table.focus();
        } else if (focusTarget === this.$.focusexit) {
          // The focus is about to exit the grid to the bottom.
          this.$.focusexit.focus();
        } else if (focusTarget === this._itemsFocusable) {
          let itemsFocusTarget = focusTarget;
          const targetRow = this._itemsFocusable.parentNode;
          this._ensureScrolledToIndex(this._focusedItemIndex);
          if (targetRow.index !== this._focusedItemIndex) {
            // The target row, which is about to be focused next, has been
            // assigned with a new index since last focus, probably because of
            // scrolling. Focus the row for the stored focused item index instead.
            const columnIndex = Array.from(targetRow.children).indexOf(this._itemsFocusable);
            const focusedItemRow = Array.from(this.$.items.children)
              .filter(row => row.index === this._focusedItemIndex)[0];
            if (focusedItemRow) {
              itemsFocusTarget = focusedItemRow.children[columnIndex];
            }
          }
          e.preventDefault();
          itemsFocusTarget.focus();
        } else {
          e.preventDefault();
          focusTarget.focus();
        }

        this._toggleAttribute('navigating', true, this);
      }

      _onSpaceKeyDown(e) {
        e.preventDefault();

        const cell = e.composedPath()[0];
        if (!cell._content || !cell._content.firstElementChild) {
          this.dispatchEvent(new CustomEvent('cell-activate', {detail: {
            model: this.__getRowModel(cell.parentElement)
          }}));
        }
      }

      /** @private */
      _onKeyUp(e) {
        if (!/^( |SpaceBar)$/.test(e.key)) {
          return;
        }

        e.preventDefault();

        const cell = e.composedPath()[0];
        if (cell._content && cell._content.firstElementChild) {
          const wasNavigating = this.hasAttribute('navigating');
          cell._content.firstElementChild.click();
          this._toggleAttribute('navigating', wasNavigating, this);
        }
      }

      _onFocusIn(e) {
        if (!this._isMousedown) {
          this._toggleAttribute('navigating', true, this);
        }

        const rootTarget = e.composedPath()[0];

        if (rootTarget === this.$.table ||
            rootTarget === this.$.focusexit) {
          // The focus enters the top (bottom) of the grid, meaning that user has
          // tabbed (shift-tabbed) into the grid. Move the focus to
          // the first (the last) focusable.
          this._predictFocusStepTarget(
            rootTarget,
            rootTarget === this.$.table ? 1 : -1
          ).focus();
          this._toggleAttribute('interacting', false, this);
        } else {
          this._detectInteracting(e);
        }
      }

      _onFocusOut(e) {
        this._toggleAttribute('navigating', false, this);
        this._detectInteracting(e);
      }

      _onCellFocusIn(e) {
        this._detectInteracting(e);

        if (e.composedPath().indexOf(this.$.table) === 3) {
          const cell = e.composedPath()[0];
          this._activeRowGroup = cell.parentNode.parentNode;
          if (this._activeRowGroup === this.$.header) {
            this._headerFocusable = cell;
          } else if (this._activeRowGroup === this.$.items) {
            this._itemsFocusable = cell;
          } else if (this._activeRowGroup === this.$.footer) {
            this._footerFocusable = cell;
          }
          // Inform cell content of the focus (used in <vaadin-grid-sorter>)
          cell._content.dispatchEvent(new CustomEvent('cell-focusin', {bubbles: false}));
        }

        this._detectFocusedItemIndex(e);
      }

      _onCellFocusOut(e) {
        if (e.composedPath().indexOf(this.$.table) === 3) {
          const cell = e.composedPath()[0];
          // Inform cell content of the focus (used in <vaadin-grid-sorter>)
          cell._content.dispatchEvent(new CustomEvent('cell-focusout', {bubbles: false}));
        }
      }

      _detectInteracting(e) {
        this._toggleAttribute('interacting',
          e.composedPath().some(el => el.localName === 'vaadin-grid-cell-content'),
          this);
      }

      _detectFocusedItemIndex(e) {
        const {rowGroup, row} = this._parseEventPath(e.composedPath());
        if (rowGroup === this.$.items) {
          this._focusedItemIndex = row.index;
        }
      }

      _preventScrollerRotatingCellFocus(item, index) {
        if (item.index === this._focusedItemIndex && this.hasAttribute('navigating') && this._activeRowGroup === this.$.items) {
          // Focused item has went, hide navigation mode
          this._navigatingIsHidden = true;
          this._toggleAttribute('navigating', false, this);
        }
        if (index === this._focusedItemIndex && this._navigatingIsHidden) {
          // Focused item is back, restore navigation mode
          this._navigatingIsHidden = false;
          this._toggleAttribute('navigating', true, this);
        }
      }

      _getColumns(rowGroup, rowIndex) {
        let columnTreeLevel = this._columnTree.length - 1;
        if (rowGroup === this.$.header) {
          columnTreeLevel = rowIndex;
        } else if (rowGroup === this.$.footer) {
          columnTreeLevel = this._columnTree.length - 1 - rowIndex;
        }
        return this._columnTree[columnTreeLevel];
      }

      _resetKeyboardNavigation() {
        if (this.$.header.firstElementChild) {
          this._headerFocusable = Array.from(this.$.header.firstElementChild.children).filter(el => !el.hidden)[0];
        }

        if (this.$.items.firstElementChild) {
          const firstVisibleIndexRow = this._iterateItems((pidx, vidx) => {
            if (this._firstVisibleIndex === vidx) {
              return this.$.items.children[pidx];
            }
          });
          if (firstVisibleIndexRow) {
            this._itemsFocusable = Array.from(firstVisibleIndexRow.children).filter(el => !el.hidden)[0];
          }
        }

        if (this.$.footer.firstElementChild) {
          this._footerFocusable = Array.from(this.$.footer.firstElementChild.children).filter(el => !el.hidden)[0];
        }
      }

      _scrollHorizontallyToCell(dstCell) {
        if (dstCell.hasAttribute('frozen') || this._elementMatches(dstCell, '[part~="details-cell"]')) {
          // These cells are, by design, always visible, no need to scroll.
          return;
        }

        const dstCellRect = dstCell.getBoundingClientRect();
        const dstRow = dstCell.parentNode;
        const dstCellIndex = Array.from(dstRow.children).indexOf(dstCell);
        const tableRect = this.$.table.getBoundingClientRect();
        let leftBoundary = tableRect.left, rightBoundary = tableRect.right;
        for (let i = dstCellIndex - 1; i >= 0; i--) {
          const cell = dstRow.children[i];
          if (cell.hasAttribute('hidden') ||
              this._elementMatches(cell, '[part~="details-cell"]')) {
            continue;
          }
          if (cell.hasAttribute('frozen')) {
            leftBoundary = cell.getBoundingClientRect().right;
            break;
          }
        }
        for (let i = dstCellIndex + 1; i < dstRow.children.length; i++) {
          const cell = dstRow.children[i];
          if (cell.hasAttribute('hidden') ||
              this._elementMatches(cell, '[part~="details-cell"]')) {
            continue;
          }
          if (cell.hasAttribute('frozen')) {
            rightBoundary = cell.getBoundingClientRect().left;
            break;
          }
        }

        if (dstCellRect.left < leftBoundary) {
          this.$.table.scrollLeft += Math.round(dstCellRect.left - leftBoundary);
        }
        if (dstCellRect.right > rightBoundary) {
          this.$.table.scrollLeft += Math.round(dstCellRect.right - rightBoundary);
        }
      }

      _elementMatches(el, query) {
        return el.matches ? el.matches(query) :
          Array.from(el.parentNode.querySelectorAll(query)).indexOf(el) !== -1;
      }
    };

    /**
    @license
    Copyright (c) 2017 Vaadin Ltd.
    This program is available under Apache License Version 2.0, available at https://vaadin.com/license/
    */

    /**
     * @polymerMixin
     */
    const ColumnReorderingMixin = superClass => class ColumnReorderingMixin extends GestureEventListeners(superClass) {

      static get properties() {
        return {
          /**
           * Set to true to allow column reordering.
           */
          columnReorderingAllowed: {
            type: Boolean,
            value: false
          },

          _orderBaseScope: {
            type: Number,
            value: 10000000
          }
        };
      }

      static get observers() {
        return [
          '_updateOrders(_columnTree, _columnTree.*)'
        ];
      }

      ready() {
        super.ready();
        addListener(this, 'track', this._onTrackEvent);
        this._reorderGhost = this.shadowRoot.querySelector('[part="reorder-ghost"]');

        this.addEventListener('touchstart', this._onTouchStart.bind(this));
        this.addEventListener('touchmove', this._onTouchMove.bind(this));
        this.addEventListener('touchend', this._onTouchEnd.bind(this));
        this.addEventListener('contextmenu', this._onContextMenu.bind(this));
      }

      _onContextMenu(e) {
        if (this.hasAttribute('reordering')) {
          e.preventDefault();
        }
      }

      _onTouchStart(e) {
        // Touch event, delay activation by 100ms
        this._startTouchReorderTimeout = setTimeout(() => {
          this._onTrackStart({
            detail: {
              x: e.touches[0].clientX,
              y: e.touches[0].clientY
            }
          });
        }, 100);
      }

      _onTouchMove(e) {
        if (this._draggedColumn) {
          e.preventDefault();
        }
        clearTimeout(this._startTouchReorderTimeout);
      }

      _onTouchEnd() {
        clearTimeout(this._startTouchReorderTimeout);
        this._onTrackEnd();
      }

      _onTrackEvent(e) {
        if (e.detail.state === 'start') {
          const path = e.composedPath();
          const headerCell = path[path.indexOf(this.$.header) - 2];
          if (!headerCell || !headerCell._content) {
            // Not a header column
            return;
          }

          const activeElement = this.getRootNode().activeElement;
          if (headerCell._content.contains(this.getRootNode().activeElement)
              && (!this._ie || !this._isFocusable(activeElement))) {
            // Something was focused inside the cell
            return;
          }

          if (this.$.scroller.hasAttribute('column-resizing')) {
            // Resizing is in progress
            return;
          }

          if (!this._touchDevice) {
            // Not a touch device
            this._onTrackStart(e);
          }
        } else if (e.detail.state === 'track') {
          this._onTrack(e);
        } else if (e.detail.state === 'end') {
          this._onTrackEnd(e);
        }
      }

      _onTrackStart(e) {
        if (!this.columnReorderingAllowed) {
          return;
        }

        // Cancel reordering if there are draggable nodes on the event path
        const path = e.path || dom(e).path;
        if (path && path.filter(node => node.hasAttribute && node.hasAttribute('draggable'))[0]) {
          return;
        }

        const headerCell = this._cellFromPoint(e.detail.x, e.detail.y);
        if (!headerCell || headerCell.getAttribute('part').indexOf('header-cell') === -1) {
          return;
        }

        this._toggleAttribute('reordering', true, this);
        this._draggedColumn = headerCell._column;
        while (this._draggedColumn.parentElement.childElementCount === 1) {
          // This is the only column in the group, drag the whole group instead
          this._draggedColumn = this._draggedColumn.parentElement;
        }
        this._setSiblingsReorderStatus(this._draggedColumn, 'allowed');
        this._draggedColumn._reorderStatus = 'dragging';

        this._updateGhost(headerCell);
        this._reorderGhost.style.visibility = 'visible';
        this._updateGhostPosition(e.detail.x, this._touchDevice ? e.detail.y - 50 : e.detail.y);
        this._autoScroller();
      }

      _onTrack(e) {
        if (!this._draggedColumn) {
          // Reordering didn’t start. Skip this event.
          return;
        }

        const targetCell = this._cellFromPoint(e.detail.x, e.detail.y);
        if (!targetCell) {
          return;
        }

        const targetColumn = this._getTargetColumn(targetCell, this._draggedColumn);
        if (this._isSwapAllowed(this._draggedColumn, targetColumn) &&
          this._isSwappableByPosition(targetColumn, e.detail.x)) {
          this._swapColumnOrders(this._draggedColumn, targetColumn);
        }

        this._updateGhostPosition(e.detail.x, this._touchDevice ? e.detail.y - 50 : e.detail.y);
        this._lastDragClientX = e.detail.x;
      }

      _onTrackEnd() {
        if (!this._draggedColumn) {
          // Reordering didn’t start. Skip this event.
          return;
        }

        this._toggleAttribute('reordering', false, this);
        this._draggedColumn._reorderStatus = '';
        this._setSiblingsReorderStatus(this._draggedColumn, '');
        this._draggedColumn = null;
        this._lastDragClientX = null;
        this._reorderGhost.style.visibility = 'hidden';

        this.dispatchEvent(new CustomEvent('column-reorder', {detail: {
          columns: this._getColumnsInOrder()
        }}));
      }

      _getColumnsInOrder() {
        return this._columnTree.slice(0).pop()
          .filter(c => !c.hidden)
          .sort((b, a) => (b._order - a._order));
      }

      _cellFromPoint(x, y) {
        x = x || 0;
        y = y || 0;
        if (!this._draggedColumn) {
          this._toggleAttribute('no-content-pointer-events', true, this.$.scroller);
        }
        let cell;
        if (useShadow) {
          cell = this.shadowRoot.elementFromPoint(x, y);
        } else {
          cell = document.elementFromPoint(x, y);

          // Workaround a FF58 bug
          if (cell.localName === 'vaadin-grid-cell-content') {
            cell = cell.assignedSlot.parentNode;
          }

        }
        this._toggleAttribute('no-content-pointer-events', false, this.$.scroller);

        // Make sure the element is actually a cell
        if (cell && cell._column) {
          return cell;
        }
      }

      _updateGhostPosition(eventClientX, eventClientY) {
        const ghostRect = this._reorderGhost.getBoundingClientRect();
        // // This is where we want to position the ghost
        const targetLeft = eventClientX - ghostRect.width / 2;
        const targetTop = eventClientY - ghostRect.height / 2;
        // Current position
        const _left = parseInt(this._reorderGhost._left || 0);
        const _top = parseInt(this._reorderGhost._top || 0);
        // Reposition the ghost
        this._reorderGhost._left = _left - (ghostRect.left - targetLeft);
        this._reorderGhost._top = _top - (ghostRect.top - targetTop);
        this._reorderGhost.style.transform = `translate(${this._reorderGhost._left}px, ${this._reorderGhost._top}px)`;
      }

      _getInnerText(e) {
        if (e.localName) {
          // Custom implementation needed since IE11 doesn't respect the spec in case of hidden elements
          if (getComputedStyle(e).display === 'none') {
            return '';
          } else {
            return Array.from(e.childNodes).map(n => this._getInnerText(n)).join('');
          }
        } else {
          return e.textContent;
        }
      }

      _updateGhost(cell) {
        const ghost = this._reorderGhost;
        ghost.textContent = this._getInnerText(cell._content);
        const style = window.getComputedStyle(cell);
        ['boxSizing', 'display', 'width', 'height', 'background', 'alignItems', 'padding', 'border', 'flex-direction', 'overflow']
          .forEach(propertyName => ghost.style[propertyName] = style[propertyName]);
        return ghost;
      }

      _updateOrders(columnTree, splices) {
        if (columnTree === undefined || splices === undefined) {
          return;
        }

        // Reset all column orders
        columnTree[0].forEach((column, index) => column._order = 0);
        // Set order numbers to top-level columns
        columnTree[0].forEach((column, index) => column._order = (index + 1) * this._orderBaseScope);
      }

      _setSiblingsReorderStatus(column, status) {
        Array.from(column.parentNode.children)
          .filter(child => /column/.test(child.localName) && this._isSwapAllowed(child, column))
          .forEach(sibling => sibling._reorderStatus = status);
      }

      _autoScroller() {
        if (this._lastDragClientX) {
          const rightDiff = this._lastDragClientX - this.getBoundingClientRect().right + 50;
          const leftDiff = this.getBoundingClientRect().left - this._lastDragClientX + 50;

          if (rightDiff > 0) {
            this.$.table.scrollLeft += rightDiff / 10;
          } else if (leftDiff > 0) {
            this.$.table.scrollLeft -= leftDiff / 10;
          }
          this._scrollHandler();
        }

        if (this._draggedColumn) {
          this.async(this._autoScroller, 10);
        }
      }

      _isSwapAllowed(column1, column2) {
        if (column1 && column2) {
          const differentColumns = column1 !== column2;
          const sameParent = column1.parentElement === column2.parentElement;
          const sameFrozen = column1.frozen === column2.frozen;
          return differentColumns && sameParent && sameFrozen;
        }
      }

      _isSwappableByPosition(targetColumn, clientX) {
        const targetCell =
          Array.from(this.$.header.querySelectorAll('tr:not([hidden]) [part~="cell"]')).filter(cell => targetColumn.contains(cell._column))[0];
        const sourceCellRect = this.$.header.querySelector('tr:not([hidden]) [reorder-status=dragging]').getBoundingClientRect();
        const targetRect = targetCell.getBoundingClientRect();
        if (targetRect.left > sourceCellRect.left) {
          return clientX > targetRect.right - sourceCellRect.width;
        } else {
          return clientX < targetRect.left + sourceCellRect.width;
        }
      }

      _swapColumnOrders(column1, column2) {
        const _order = column1._order;
        column1._order = column2._order;
        column2._order = _order;
        this._updateLastFrozen();
        this._updateFirstAndLastColumn();
      }

      _getTargetColumn(targetCell, draggedColumn) {
        if (targetCell && draggedColumn) {
          let candidate = targetCell._column;
          while (candidate.parentElement !== draggedColumn.parentElement && candidate !== this) {
            candidate = candidate.parentElement;
          }
          if (candidate.parentElement === draggedColumn.parentElement) {
            return candidate;
          } else {
            return targetCell._column;
          }
        }
      }

      /**
       * Fired when the columns in the grid are reordered.
       *
       * @event column-reorder
       * @param {Object} detail
       * @param {Object} detail.columns the columns in the new order
       */
    };

    /**
    @license
    Copyright (c) 2018 Vaadin Ltd.
    This program is available under Apache License Version 2.0, available at https://vaadin.com/license/
    */

    /**
     * @polymerMixin
     */
    const ColumnBaseMixin = superClass => class ColumnBaseMixin extends superClass {
      static get properties() {
        return {
          /**
           * When set to true, the column is user-resizable.
           * @default false
           */
          resizable: {
            type: Boolean,
            value: function() {
              if (this.localName === 'vaadin-grid-column-group') {
                return;
              }

              const parent = this.parentNode;
              if (parent && parent.localName === 'vaadin-grid-column-group') {
                return parent.resizable || false;
              } else {
                return false;
              }
            }
          },

          _headerTemplate: {
            type: Object
          },

          _footerTemplate: {
            type: Object
          },

          /**
           * When true, the column is frozen. When a column inside of a column group is frozen,
           * all of the sibling columns inside the group will get frozen also.
           */
          frozen: {
            type: Boolean,
            value: false
          },

          /**
           * When set to true, the cells for this column are hidden.
           */
          hidden: {
            type: Boolean
          },

          /**
           * Text content to display in the header cell of the column.
           */
          header: {
            type: String
          },

          /**
           * Aligns the columns cell content horizontally.
           * Supported values: "start", "center" and "end".
           */
          textAlign: {
            type: String
          },

          _lastFrozen: {
            type: Boolean,
            value: false
          },

          _order: Number,

          _reorderStatus: Boolean,

          _emptyCells: Array,

          _headerCell: Object,

          _footerCell: Object,

          _grid: Object,

          /**
           * Custom function for rendering the header content.
           * Receives two arguments:
           *
           * - `root` The header cell content DOM element. Append your content to it.
           * - `column` The `<vaadin-grid-column>` element.
           */
          headerRenderer: Function,

          /**
           * Custom function for rendering the footer content.
           * Receives two arguments:
           *
           * - `root` The footer cell content DOM element. Append your content to it.
           * - `column` The `<vaadin-grid-column>` element.
           */
          footerRenderer: Function
        };
      }

      static get observers() {
        return [
          '_widthChanged(width, _headerCell, _footerCell, _cells.*)',
          '_frozenChanged(frozen, _headerCell, _footerCell, _cells.*)',
          '_flexGrowChanged(flexGrow, _headerCell, _footerCell, _cells.*)',
          '_pathOrHeaderChanged(path, header, _headerCell, _footerCell, _cells.*, renderer, headerRenderer, _bodyTemplate, _headerTemplate)',
          '_textAlignChanged(textAlign, _cells.*, _headerCell, _footerCell)',
          '_orderChanged(_order, _headerCell, _footerCell, _cells.*)',
          '_lastFrozenChanged(_lastFrozen)',
          '_setBodyTemplateOrRenderer(_bodyTemplate, renderer, _cells, _cells.*)',
          '_setHeaderTemplateOrRenderer(_headerTemplate, headerRenderer, _headerCell)',
          '_setFooterTemplateOrRenderer(_footerTemplate, footerRenderer, _footerCell)',
          '_resizableChanged(resizable, _headerCell)',
          '_reorderStatusChanged(_reorderStatus, _headerCell, _footerCell, _cells.*)',
          '_hiddenChanged(hidden, _headerCell, _footerCell, _cells.*)'
        ];
      }

      /** @protected */
      connectedCallback() {
        super.connectedCallback();

        this._bodyTemplate && (this._bodyTemplate.templatizer._grid = this._grid);
        this._headerTemplate && (this._headerTemplate.templatizer._grid = this._grid);
        this._footerTemplate && (this._footerTemplate.templatizer._grid = this._grid);

        this._templateObserver.flush();
        if (!this._bodyTemplate) {
          // The observer might not have triggered if the tag is empty. Run manually.
          this._templateObserver.callback();
        }

        requestAnimationFrame(() => {
          this._allCells.forEach(cell => {
            if (!cell._content.parentNode) {
              this._grid && this._grid.appendChild(cell._content);
            }
          });
        });
      }

      /** @protected */
      disconnectedCallback() {
        super.disconnectedCallback();

        requestAnimationFrame(() => {
          if (!this._findHostGrid()) {
            this._allCells.forEach(cell => {
              if (cell._content.parentNode) {
                cell._content.parentNode.removeChild(cell._content);
              }
            });
          }
        });

        this._gridValue = undefined;
      }

      _findHostGrid() {
        let el = this;
        // Custom elements extending grid must have a specific localName
        while (el && !/^vaadin.*grid(-pro)?$/.test(el.localName)) {
          el = el.assignedSlot ? el.assignedSlot.parentNode : el.parentNode;
        }
        return el || undefined;
      }

      get _grid() {
        if (!this._gridValue) {
          this._gridValue = this._findHostGrid();
        }
        return this._gridValue;
      }

      get _allCells() {
        return []
          .concat(this._cells || [])
          .concat(this._emptyCells || [])
          .concat(this._headerCell)
          .concat(this._footerCell)
          .filter(cell => cell);
      }

      constructor() {
        super();

        this._templateObserver = new FlattenedNodesObserver(this, info => {
          this._headerTemplate = this._prepareHeaderTemplate();
          this._footerTemplate = this._prepareFooterTemplate();
          this._bodyTemplate = this._prepareBodyTemplate();
        });
      }

      _prepareHeaderTemplate() {
        return this._prepareTemplatizer(this._findTemplate(true) || null, {});
      }

      _prepareFooterTemplate() {
        return this._prepareTemplatizer(this._findTemplate(false, true) || null, {});
      }

      _prepareBodyTemplate() {
        return this._prepareTemplatizer(this._findTemplate() || null);
      }

      _prepareTemplatizer(template, instanceProps) {
        if (template && !template.templatizer) {
          const templatizer = new GridTemplatizer();
          templatizer._grid = this._grid;
          templatizer.dataHost = this.dataHost;
          templatizer._instanceProps = instanceProps || templatizer._instanceProps;
          templatizer.template = template;
          template.templatizer = templatizer;
        }

        return template;
      }

      _renderHeaderAndFooter() {
        if (this.headerRenderer && this._headerCell) {
          this.__runRenderer(this.headerRenderer, this._headerCell);
        }
        if (this.footerRenderer && this._footerCell) {
          this.__runRenderer(this.footerRenderer, this._footerCell);
        }
      }

      __runRenderer(renderer, cell, rowData) {
        const args = [cell._content, this];
        if (rowData && rowData.item) {
          args.push(rowData);
        }
        renderer.apply(this, args);
      }

      __setColumnTemplateOrRenderer(template, renderer, cells) {
        if (template && renderer) {
          throw new Error('You should only use either a renderer or a template');
        }

        cells.forEach(cell => {
          const model = this._grid.__getRowModel(cell.parentElement);

          if (renderer) {
            cell._renderer = renderer;

            if (model.item || renderer === this.headerRenderer || renderer === this.footerRenderer) {
              this.__runRenderer(renderer, cell, model);
            }
          } else if (cell._template !== template) {
            cell._template = template;

            cell._content.innerHTML = '';
            template.templatizer._grid = template.templatizer._grid || this._grid;
            const inst = template.templatizer.createInstance();
            cell._content.appendChild(inst.root);
            cell._instance = inst;
            if (model.item) {
              cell._instance.setProperties(model);
            }
          }
        });
      }

      _setBodyTemplateOrRenderer(template, renderer, cells, splices) {
        if ((template || renderer) && cells) {
          this.__setColumnTemplateOrRenderer(template, renderer, cells);
        }
      }

      _setHeaderTemplateOrRenderer(headerTemplate, headerRenderer, headerCell) {
        if ((headerTemplate || headerRenderer) && headerCell) {
          this.__setColumnTemplateOrRenderer(headerTemplate, headerRenderer, [headerCell]);
        }
      }

      _setFooterTemplateOrRenderer(footerTemplate, footerRenderer, footerCell) {
        if ((footerTemplate || footerRenderer) && footerCell) {
          this.__setColumnTemplateOrRenderer(footerTemplate, footerRenderer, [footerCell]);
          this._grid.__updateHeaderFooterRowVisibility(footerCell.parentElement);
        }
      }

      _selectFirstTemplate(header = false, footer = false) {
        return FlattenedNodesObserver.getFlattenedNodes(this)
          .filter(node =>
            node.localName === 'template'
            && node.classList.contains('header') === header
            && node.classList.contains('footer') === footer
          )[0];
      }

      _findTemplate(header, footer) {
        const template = this._selectFirstTemplate(header, footer);
        if (template) {
          if (this.dataHost) {
            // set dataHost to the context where template has been defined
            template._rootDataHost = this.dataHost._rootDataHost || this.dataHost;
          }
        }
        return template;
      }

      _flexGrowChanged(flexGrow, headerCell, footerCell, cells) {
        if (this.parentElement && this.parentElement._columnPropChanged) {
          this.parentElement._columnPropChanged('flexGrow');
        }

        this._allCells.forEach(cell => cell.style.flexGrow = flexGrow);
      }

      _orderChanged(order, headerCell, footerCell, cells) {
        this._allCells.forEach(cell => cell.style.order = order);
      }

      _widthChanged(width, headerCell, footerCell, cells) {
        if (this.parentElement && this.parentElement._columnPropChanged) {
          this.parentElement._columnPropChanged('width');
        }

        this._allCells.forEach(cell => cell.style.width = width);

        // Force a reflow to workaround browser issues causing double scrollbars to grid
        // https://github.com/vaadin/vaadin-grid/issues/1586
        if (this._grid && this._grid.__forceReflow) {
          this._grid.__forceReflow();
        }
      }

      _frozenChanged(frozen, headerCell, footerCell, cells) {
        if (this.parentElement && this.parentElement._columnPropChanged) {
          this.parentElement._columnPropChanged('frozen', frozen);
        }

        this._allCells.forEach(cell => this._toggleAttribute('frozen', frozen, cell));

        this._grid && this._grid._frozenCellsChanged && this._grid._frozenCellsChanged();
      }

      _lastFrozenChanged(lastFrozen) {
        this._allCells.forEach(cell => this._toggleAttribute('last-frozen', lastFrozen, cell));

        if (this.parentElement && this.parentElement._columnPropChanged) {
          this.parentElement._lastFrozen = lastFrozen;
        }
      }

      _pathOrHeaderChanged(path, header, headerCell, footerCell, cells, renderer, headerRenderer, bodyTemplate, headerTemplate) {
        const hasHeaderText = header !== undefined;
        if (!headerRenderer && !headerTemplate && hasHeaderText && headerCell) {
          this.__setTextContent(headerCell._content, header);
        }

        if (path && cells.value) {
          if (!renderer && !bodyTemplate) {
            const pathRenderer = (root, owner, {item}) => this.__setTextContent(root, this.get(path, item));
            this.__setColumnTemplateOrRenderer(undefined, pathRenderer, cells.value);
          }

          if (!headerRenderer && !headerTemplate && !hasHeaderText && headerCell && header !== null) {
            this.__setTextContent(headerCell._content, this._generateHeader(path));
          }
        }

        if (headerCell) {
          this._grid.__updateHeaderFooterRowVisibility(headerCell.parentElement);
        }
      }

      __setTextContent(node, textContent) {
        node.textContent !== textContent && (node.textContent = textContent);
      }

      _generateHeader(path) {
        return path
          .substr(path.lastIndexOf('.') + 1)
          .replace(/([A-Z])/g, '-$1').toLowerCase()
          .replace(/-/g, ' ')
          .replace(/^./, match => match.toUpperCase());
      }

      _toggleAttribute(name, bool, node) {
        if (node.hasAttribute(name) === !bool) {
          if (bool) {
            node.setAttribute(name, '');
          } else {
            node.removeAttribute(name);
          }
        }
      }

      _reorderStatusChanged(reorderStatus, headerCell, footerCell, cells) {
        this._allCells.forEach(cell => cell.setAttribute('reorder-status', reorderStatus));
      }

      _resizableChanged(resizable, headerCell) {
        if (resizable === undefined || headerCell === undefined) {
          return;
        }

        if (headerCell) {
          [headerCell].concat(this._emptyCells).forEach(cell => {
            if (cell) {
              const existingHandle = cell.querySelector('[part~="resize-handle"]');
              if (existingHandle) {
                cell.removeChild(existingHandle);
              }

              if (resizable) {
                const handle = document.createElement('div');
                handle.setAttribute('part', 'resize-handle');
                cell.appendChild(handle);
              }
            }
          });
        }
      }

      _textAlignChanged(textAlign, _cells, _headerCell, _footerCell) {
        if (textAlign === undefined) {
          return;
        }
        if (['start', 'end', 'center'].indexOf(textAlign) === -1) {
          console.warn('textAlign can only be set as "start", "end" or "center"');
          return;
        }

        let textAlignFallback;
        if (getComputedStyle(this._grid).direction === 'ltr') {
          if (textAlign === 'start') {
            textAlignFallback = 'left';
          } else if (textAlign === 'end') {
            textAlignFallback = 'right';
          }
        } else {
          if (textAlign === 'start') {
            textAlignFallback = 'right';
          } else if (textAlign === 'end') {
            textAlignFallback = 'left';
          }
        }

        this._allCells.forEach(cell => {
          cell._content.style.textAlign = textAlign;
          if (getComputedStyle(cell._content).textAlign !== textAlign) {
            cell._content.style.textAlign = textAlignFallback;
          }
        });
      }

      _hiddenChanged(hidden, headerCell, footerCell, cells) {
        if (this.parentElement && this.parentElement._columnPropChanged) {
          this.parentElement._columnPropChanged('hidden', hidden);
        }

        if (!!hidden !== !!this._previousHidden && this._grid) {
          if (hidden === true) {
            this._allCells.forEach(cell => {
              if (cell._content.parentNode) {
                cell._content.parentNode.removeChild(cell._content);
              }
            });
          }
          this._grid._debouncerHiddenChanged = Debouncer.debounce(
            this._grid._debouncerHiddenChanged,
            animationFrame,
            () => {
              if (this._grid && this._grid._renderColumnTree) {
                this._grid._renderColumnTree(this._grid._columnTree);
              }
            }
          );

          this._grid._updateLastFrozen && this._grid._updateLastFrozen();
          this._grid.notifyResize && this._grid.notifyResize();
          this._grid._resetKeyboardNavigation && this._grid._resetKeyboardNavigation();
        }
        this._previousHidden = hidden;
      }

    };

    /**
     * A `<vaadin-grid-column>` is used to configure how a column in `<vaadin-grid>`
     * should look like.
     *
     * See `<vaadin-grid>` documentation and demos for instructions and examples on how
     * to configure the `<vaadin-grid-column>`.
     * ```
     *
     * @extends PolymerElement
     * @mixes Grid.ColumnBaseMixin
     */
    class GridColumnElement extends ColumnBaseMixin(DirMixin$1(PolymerElement)) {
      static get is() {
        return 'vaadin-grid-column';
      }

      static get properties() {
        return {
          /**
           * Width of the cells for this column.
           */
          width: {
            type: String,
            value: '100px'
          },

          /**
           * Flex grow ratio for the cell widths. When set to 0, cell width is fixed.
           */
          flexGrow: {
            type: Number,
            value: 1
          },

          /**
           * Custom function for rendering the cell content.
           * Receives three arguments:
           *
           * - `root` The cell content DOM element. Append your content to it.
           * - `column` The `<vaadin-grid-column>` element.
           * - `rowData` The object with the properties related with
           *   the rendered item, contains:
           *   - `rowData.index` The index of the item.
           *   - `rowData.item` The item.
           *   - `rowData.expanded` Sublevel toggle state.
           *   - `rowData.level` Level of the tree represented with a horizontal offset of the toggle button.
           *   - `rowData.selected` Selected state.
           */
          renderer: Function,

          /**
           * Path to an item sub-property whose value gets displayed in the column body cells.
           * The property name is also shown in the column header if an explicit header or renderer isn't defined.
           */
          path: {
            type: String
          },

          /**
           * Automatically sets the width of the column based on the column contents when this is set to `true`.
           *
           * For performance reasons the column width is calculated automatically only once when the grid items
           * are rendered for the first time and the calculation only considers the rows which are currently
           * rendered in DOM (a bit more than what is currently visible). If the grid is scrolled, or the cell
           * content changes, the column width might not match the contents anymore.
           *
           * Hidden columns are ignored in the calculation and their widths are not automatically updated when
           * you show a column that was initially hidden.
           *
           * You can manually trigger the auto sizing behavior again by calling `grid.recalculateColumnWidths()`.
           *
           * The column width may still grow larger when `flexGrow` is not 0.
           */
          autoWidth: {
            type: Boolean,
            value: false
          },

          _bodyTemplate: {
            type: Object
          },

          _cells: Array

        };
      }

    }

    customElements.define(GridColumnElement.is, GridColumnElement);

    /**
    @license
    Copyright (c) 2017 Vaadin Ltd.
    This program is available under Apache License Version 2.0, available at https://vaadin.com/license/
    */
    /**
     * This Element is used internally by vaadin-grid.
     *
     * @private
     */
    class GridOuterScrollerElement extends (class extends PolymerElement {}) {
      static get template() {
        return html`
    <style>
      :host {
        display: block;
        height: 100%;
        width: 100%;
        position: absolute;
        top: 0;
        box-sizing: border-box;
        overflow: auto;
      }

      :host([passthrough]) {
        pointer-events: none;
      }
    </style>

    <slot></slot>
`;
      }

      static get is() {
        return 'vaadin-grid-outer-scroller';
      }

      static get properties() {
        return {

          scrollTarget: {
            type: Object
          },

          scrollHandler: {
            type: Object
          },

          passthrough: {
            type: Boolean,
            reflectToAttribute: true,
            value: true
          },

          outerScrolling: Boolean,

          noScrollbars: Boolean,

          _touchDevice: Boolean
        };
      }

      ready() {
        super.ready();
        this.addEventListener('scroll', () => this._syncScrollTarget());
        this.parentElement.addEventListener('mousemove', this._onMouseMove.bind(this));

        // for some reason scroll bars are hidden in iOS if this style is
        // added in stylesheets or before attaching.
        this.style.webkitOverflowScrolling = 'touch';

        this.addEventListener('mousedown', _ => this.outerScrolling = true);
        this.addEventListener('mouseup', _ => {
          this.outerScrolling = false;
          this.scrollHandler._scrollHandler();
        });
      }

      _onMouseMove(e) {
        // Ignore mousemove events on touch devices
        if (!this._touchDevice) {
          if (this.noScrollbars && this.parentElement.hasAttribute('scroll-period')) {
            this.passthrough = e.offsetY <= this.clientHeight - 20 && e.offsetX <= this.clientWidth - 20;
          } else {
            this.passthrough = e.offsetY <= this.clientHeight && e.offsetX <= this.clientWidth;
          }
        }
      }

      syncOuterScroller() {
        this.scrollTop = this.scrollTarget.scrollTop;
        this.scrollLeft = this.scrollTarget.scrollLeft;
      }

      _syncScrollTarget() {
        requestAnimationFrame(() => {
          this.scrollTarget.scrollTop = this.scrollTop;
          this.scrollTarget.scrollLeft = this.scrollLeft;
          this.scrollHandler._scrollHandler();
        });

      }
    }

    customElements.define(GridOuterScrollerElement.is, GridOuterScrollerElement);

    /**
    @license
    Copyright (c) 2017 Vaadin Ltd.
    This program is available under Apache License Version 2.0, available at https://vaadin.com/license/
    */
    const VaadinGridStyles = document.createElement('dom-module');

    // NOTE(web-padawan): https://github.com/vaadin/vaadin-grid/issues/1514
    VaadinGridStyles.appendChild(
      html`
  <style>
    @keyframes vaadin-grid-appear {
      to {
        opacity: 1;
      }
    }

    :host {
      display: block;
      animation: 1ms vaadin-grid-appear;
      height: 400px;
      flex: 1 1 auto;
      align-self: stretch;
      position: relative;
    }

    :host([hidden]) {
      display: none !important;
    }

    #scroller {
      display: block;
      transform: translateY(0);
      width: auto;
      height: auto;
      position: absolute;
      top: 0;
      right: 0;
      bottom: 0;
      left: 0;
    }

    :host([height-by-rows]) {
      height: auto;
      align-self: flex-start;
      flex-grow: 0;
      width: 100%;
    }

    :host([height-by-rows]) #scroller {
      width: 100%;
      height: 100%;
      position: relative;
    }

    #table {
      display: block;
      width: 100%;
      height: 100%;
      overflow: auto;
      z-index: -2;
      position: relative;
      outline: none;
    }

    #header {
      display: block;
      position: absolute;
      top: 0;
      width: 100%;
    }

    th {
      text-align: inherit;
    }

    /* Safari doesn't work with "inherit" */
    [safari] th {
      text-align: initial;
    }

    #footer {
      display: block;
      position: absolute;
      bottom: 0;
      width: 100%;
    }

    #items {
      display: block;
      width: 100%;
      position: relative;
      z-index: -1;
    }

    #items,
    #outersizer,
    #fixedsizer {
      border-top: 0 solid transparent;
      border-bottom: 0 solid transparent;
    }

    [part~="row"] {
      display: flex;
      width: 100%;
      box-sizing: border-box;
      margin: 0;
    }

    [part~="row"][loading] [part~="body-cell"] ::slotted(vaadin-grid-cell-content) {
      opacity: 0;
    }

    #items [part~="row"] {
      position: absolute;
    }

    #items [part~="row"]:empty {
      height: 1em;
    }

    [part~="cell"]:not([part~="details-cell"]) {
      flex-shrink: 0;
      flex-grow: 1;
      box-sizing: border-box;
      display: flex;
      width: 100%;
      position: relative;
      align-items: center;
      padding: 0;
      white-space: nowrap;
    }

    [part~="details-cell"] {
      position: absolute;
      bottom: 0;
      width: 100%;
      box-sizing: border-box;
      padding: 0;
    }

    [part~="cell"] ::slotted(vaadin-grid-cell-content) {
      display: block;
      width: 100%;
      box-sizing: border-box;
      overflow: hidden;
      text-overflow: ellipsis;
    }

    [hidden] {
      display: none !important;
    }

    [frozen] {
      z-index: 2;
      will-change: transform;
    }

    #outerscroller {
      /* Needed (at least) for Android Chrome */
      z-index: 0;
    }

    #scroller:not([safari]) #outerscroller {
      /* Needed for Android Chrome (#1020). Can't be applied to Safari
      since it would re-introduce the sub-pixel overflow bug (#853) */
      will-change: transform;
    }

    [no-scrollbars]:not([safari]):not([firefox]) #outerscroller,
    [no-scrollbars][safari] #table,
    [no-scrollbars][firefox] #table {
      overflow: hidden;
    }

    [no-scrollbars]:not([safari]):not([firefox]) #outerscroller {
      pointer-events: none;
    }

    /* Reordering styles */
    :host([reordering]) [part~="cell"] ::slotted(vaadin-grid-cell-content),
    :host([reordering]) [part~="resize-handle"],
    #scroller[no-content-pointer-events] [part~="cell"] ::slotted(vaadin-grid-cell-content) {
      pointer-events: none;
    }

    [part~="reorder-ghost"] {
      visibility: hidden;
      position: fixed;
      pointer-events: none;
      opacity: 0.5;

      /* Prevent overflowing the grid in Firefox */
      top: 0;
      left: 0;
    }

    :host([reordering]) {
      -moz-user-select: none;
      -webkit-user-select: none;
      user-select: none;
    }

    #scroller[ie][column-reordering-allowed] [part~="header-cell"] {
      -ms-user-select: none;
    }

    :host([reordering]) #outerscroller {
      -webkit-overflow-scrolling: auto !important;
    }

    /* Resizing styles */
    [part~="resize-handle"] {
      position: absolute;
      top: 0;
      right: 0;
      height: 100%;
      cursor: col-resize;
      z-index: 1;
    }

    [part~="resize-handle"]::before {
      position: absolute;
      content: "";
      height: 100%;
      width: 35px;
      transform: translateX(-50%);
    }

    [last-column] [part~="resize-handle"]::before,
    [last-frozen] [part~="resize-handle"]::before {
      width: 18px;
      transform: none;
      right: 0;
    }

    #scroller[column-resizing] {
      -ms-user-select: none;
      -moz-user-select: none;
      -webkit-user-select: none;
      user-select: none;
    }

    /* Sizer styles */
    .sizer {
      display: flex;
      position: relative;
      width: 100%;
      visibility: hidden;
    }

    .sizer [part~="details-cell"] {
      display: none !important;
    }

    .sizer [part~="cell"][hidden] {
      display: none !important;
    }

    .sizer [part~="cell"] {
      display: block;
      flex-shrink: 0;
      line-height: 0;
      margin-top: -1em;
      height: 0 !important;
      min-height: 0 !important;
      max-height: 0 !important;
      padding: 0 !important;
    }

    .sizer [part~="cell"]::before {
      content: "-";
    }

    .sizer [part~="cell"] ::slotted(vaadin-grid-cell-content) {
      display: none !important;
    }

    /* Fixed mode (Tablet Edge) */
    #fixedsizer {
      position: absolute;
    }

    :not([edge][no-scrollbars]) #fixedsizer {
      display: none;
    }

    [edge][no-scrollbars] {
      /* Any value other than ‘none’ for the transform results in the creation of both a stacking context and
      a containing block. The object acts as a containing block for fixed positioned descendants. */
      transform: translateZ(0);
      overflow: hidden;
    }

    [edge][no-scrollbars] #header,
    [edge][no-scrollbars] #footer {
      position: fixed;
    }

    [edge][no-scrollbars] #items {
      position: fixed;
      width: 100%;
      will-change: transform;
    }

    /* RTL specific styles */

    :host([dir="rtl"]) [part~="reorder-ghost"] {
      left: auto;
      right: 0;
    }

    :host([dir="rtl"]) [part~="resize-handle"] {
      left: 0;
      right: auto;
    }

    :host([dir="rtl"]) [part~="resize-handle"]::before {
      transform: translateX(50%);
    }

    :host([dir="rtl"]) [last-column] [part~="resize-handle"]::before,
    :host([dir="rtl"]) [last-frozen] [part~="resize-handle"]::before {
      left: 0;
      right: auto;
    }
  </style>
`);

    const safari = /^((?!chrome|android).)*safari/i.test(navigator.userAgent);
    const firefox = navigator.userAgent.toLowerCase().indexOf('firefox') > -1;

    if (safari || firefox) {
      const scrollingStyles = document.createElement('style');
      scrollingStyles.textContent = `
    [scrolling][safari] #outerscroller,
    [scrolling][firefox] #outerscroller {
      pointer-events: auto;
    }

    [ios] #outerscroller {
      pointer-events: auto;
      z-index: -3;
    }

    [ios][scrolling] #outerscroller {
      z-index: 0;
    }

    [ios] [frozen] {
      will-change: auto;
    }
  `;
      VaadinGridStyles.querySelector('template').content.appendChild(scrollingStyles);
    }

    VaadinGridStyles.register('vaadin-grid-styles');

    /**
    @license
    Copyright (c) 2017 Vaadin Ltd.
    This program is available under Apache License Version 2.0, available at https://vaadin.com/license/
    */

    const TOUCH_DEVICE = (() => {
      try {
        document.createEvent('TouchEvent');
        return true;
      } catch (e) {
        return false;
      }
    })();

    /**
     *
     * `<vaadin-grid>` is a free, high quality data grid / data table Web Component. The content of the
     * the grid can be populated in two ways: imperatively by using renderer callback function and
     * declaratively by using Polymer's Templates.
     *
     * ### Quick Start
     *
     * Start with an assigning an array to the [`items`](#/elements/vaadin-grid#property-items) property to visualize your data.
     *
     * Use the [`<vaadin-grid-column>`](#/elements/vaadin-grid-column) element to configure the grid columns. Set `path` and `header`
     * shorthand properties for the columns to define what gets rendered in the cells of the column.
     *
     * #### Example:
     * ```html
     * <vaadin-grid>
     *   <vaadin-grid-column path="name.first" header="First name"></vaadin-grid-column>
     *   <vaadin-grid-column path="name.last" header="Last name"></vaadin-grid-column>
     *   <vaadin-grid-column path="email"></vaadin-grid-column>
     * </vaadin-grid>
     * ```
     *
     * For custom content `vaadin-grid-column` element provides you with three types of `renderer` callback functions: `headerRenderer`,
     * `renderer` and `footerRenderer`.
     *
     * Each of those renderer functions provides `root`, `column`, `rowData` arguments when applicable.
     * Generate DOM content, append it to the `root` element and control the state
     * of the host element by accessing `column`. Before generating new content,
     * users are able to check if there is already content in `root` for reusing it.
     *
     * Renderers are called on initialization of new column cells and each time the
     * related row data is updated. DOM generated during the renderer call can be reused
     * in the next renderer call and will be provided with the `root` argument.
     * On first call it will be empty.
     *
     * #### Example:
     * ```html
     * <vaadin-grid>
     *   <vaadin-grid-column></vaadin-grid-column>
     *   <vaadin-grid-column></vaadin-grid-column>
     *   <vaadin-grid-column></vaadin-grid-column>
     * </vaadin-grid>
     * ```
     * ```js
     * const grid = document.querySelector('vaadin-grid');
     * grid.items = [{'name': 'John', 'surname': 'Lennon', 'role': 'singer'},
     *               {'name': 'Ringo', 'surname': 'Starr', 'role': 'drums'}];
     *
     * const columns = grid.querySelectorAll('vaadin-grid-column');
     *
     * columns[0].headerRenderer = function(root) {
     *   root.textContent = 'Name';
     * };
     * columns[0].renderer = function(root, column, rowData) {
     *   root.textContent = rowData.item.name;
     * };
     *
     * columns[1].headerRenderer = function(root) {
     *   root.textContent = 'Surname';
     * };
     * columns[1].renderer = function(root, column, rowData) {
     *   root.textContent = rowData.item.surname;
     * };
     *
     * columns[2].headerRenderer = function(root) {
     *   root.textContent = 'Role';
     * };
     * columns[2].renderer = function(root, column, rowData) {
     *   root.textContent = rowData.item.role;
     * };
     * ```
     *
     * Alternatively, the content can be provided with Polymer's Templates:
     *
     * #### Example:
     * ```html
     * <vaadin-grid items='[{"name": "John", "surname": "Lennon", "role": "singer"},
     * {"name": "Ringo", "surname": "Starr", "role": "drums"}]'>
     *   <vaadin-grid-column>
     *     <template class="header">Name</template>
     *     <template>[[item.name]]</template>
     *   </vaadin-grid-column>
     *   <vaadin-grid-column>
     *     <template class="header">Surname</template>
     *     <template>[[item.surname]]</template>
     *   </vaadin-grid-column>
     *   <vaadin-grid-column>
     *     <template class="header">Role</template>
     *     <template>[[item.role]]</template>
     *   </vaadin-grid-column>
     * </vaadin-grid>
     * ```
     *
     * The following helper elements can be used for further customization:
     * - [`<vaadin-grid-column-group>`](#/elements/vaadin-grid-column-group)
     * - [`<vaadin-grid-filter>`](#/elements/vaadin-grid-filter)
     * - [`<vaadin-grid-sorter>`](#/elements/vaadin-grid-sorter)
     * - [`<vaadin-grid-selection-column>`](#/elements/vaadin-grid-selection-column)
     * - [`<vaadin-grid-tree-toggle>`](#/elements/vaadin-grid-tree-toggle)
     *
     * __Note that the helper elements must be explicitly imported.__
     * If you want to import everything at once you can use the `all-imports.html` bundle.
     *
     * A column template can be decorated with one the following class names to specify its purpose
     * - `header`: Marks a header template
     * - `footer`: Marks a footer template
     * - `row-details`: Marks a row details template
     *
     * The following built-in template variables can be bound to inside the column templates:
     * - `[[index]]`: Number representing the row index
     * - `[[item]]` and it's sub-properties: Data object (provided by a data provider / items array)
     * - `{{selected}}`: True if the item is selected (can be two-way bound)
     * - `{{detailsOpened}}`: True if the item has row details opened (can be two-way bound)
     * - `{{expanded}}`: True if the item has tree sublevel expanded (can be two-way bound)
     * - `[[level]]`: Number of the tree sublevel of the item, first level-items have 0
     *
     * ### Lazy Loading with Function Data Provider
     *
     * In addition to assigning an array to the items property, you can alternatively
     * provide the `<vaadin-grid>` data through the
     * [`dataProvider`](#/elements/vaadin-grid#property-dataProvider) function property.
     * The `<vaadin-grid>` calls this function lazily, only when it needs more data
     * to be displayed.
     *
     * See the [`dataProvider`](#/elements/vaadin-grid#property-dataProvider) in
     * the API reference below for the detailed data provider arguments description,
     * and the “Assigning Data” page in the demos.
     *
     * __Note that expanding the tree grid's item will trigger a call to the `dataProvider`.__
     *
     * __Also, note that when using function data providers, the total number of items
     * needs to be set manually. The total number of items can be returned
     * in the second argument of the data provider callback:__
     *
     * ```javascript
     * grid.dataProvider = function(params, callback) {
     *   var url = 'https://api.example/data' +
     *       '?page=' + params.page +        // the requested page index
     *       '&per_page=' + params.pageSize; // number of items on the page
     *   var xhr = new XMLHttpRequest();
     *   xhr.onload = function() {
     *     var response = JSON.parse(xhr.responseText);
     *     callback(
     *       response.employees, // requested page of items
     *       response.totalSize  // total number of items
     *     );
     *   };
     *   xhr.open('GET', url, true);
     *   xhr.send();
     * };
     * ```
     *
     * __Alternatively, you can use the `size` property to set the total number of items:__
     *
     * ```javascript
     * grid.size = 200; // The total number of items
     * grid.dataProvider = function(params, callback) {
     *   var url = 'https://api.example/data' +
     *       '?page=' + params.page +        // the requested page index
     *       '&per_page=' + params.pageSize; // number of items on the page
     *   var xhr = new XMLHttpRequest();
     *   xhr.onload = function() {
     *     var response = JSON.parse(xhr.responseText);
     *     callback(response.employees);
     *   };
     *   xhr.open('GET', url, true);
     *   xhr.send();
     * };
     * ```
     *
     * ### Styling
     *
     * The following shadow DOM parts are available for styling:
     *
     * Part name | Description
     * ----------------|----------------
     * `row` | Row in the internal table
     * `cell` | Cell in the internal table
     * `header-cell` | Header cell in the internal table
     * `body-cell` | Body cell in the internal table
     * `footer-cell` | Footer cell in the internal table
     * `details-cell` | Row details cell in the internal table
     * `resize-handle` | Handle for resizing the columns
     * `reorder-ghost` | Ghost element of the header cell being dragged
     *
     * The following state attributes are available for styling:
     *
     * Attribute    | Description | Part name
     * -------------|-------------|------------
     * `loading` | Set when the grid is loading data from data provider | :host
     * `interacting` | Keyboard navigation in interaction mode | :host
     * `navigating` | Keyboard navigation in navigation mode | :host
     * `overflow` | Set when rows are overflowing the grid viewport. Possible values: `top`, `bottom`, `left`, `right` | :host
     * `reordering` | Set when the grid's columns are being reordered | :host
     * `dragover` | Set when the grid (not a specific row) is dragged over | :host
     * `dragging-rows` : Set when grid rows are dragged  | :host
     * `reorder-status` | Reflects the status of a cell while columns are being reordered | cell
     * `frozen` | Frozen cell | cell
     * `last-frozen` | Last frozen cell | cell
    * * `first-column` | First visible cell on a row | cell
     * `last-column` | Last visible cell on a row | cell
     * `selected` | Selected row | row
     * `expanded` | Expanded row | row
     * `details-opened` | Row with details open | row
     * `loading` | Row that is waiting for data from data provider | row
     * `odd` | Odd row | row
     * `first` | The first body row | row
     * `dragstart` | Set for one frame when drag of a row is starting. The value is a number when multiple rows are dragged | row
     * `dragover` | Set when the row is dragged over | row
     * `drag-disabled` | Set to a row that isn't available for dragging | row
     * `drop-disabled` | Set to a row that can't be dropped on top of | row
     *
     * See [ThemableMixin – how to apply styles for shadow parts](https://github.com/vaadin/vaadin-themable-mixin/wiki)
     *
     * @extends PolymerElement
     * @mixes ThemableMixin
     * @mixes Grid.A11yMixin
     * @mixes Grid.ActiveItemMixin
     * @mixes Grid.ArrayDataProviderMixin
     * @mixes Grid.ColumnResizingMixin
     * @mixes Grid.DataProviderMixin
     * @mixes Grid.DynamicColumnsMixin
     * @mixes Grid.FilterMixin
     * @mixes Grid.RowDetailsMixin
     * @mixes Grid.ScrollMixin
     * @mixes Grid.SelectionMixin
     * @mixes Grid.SortMixin
     * @mixes Grid.KeyboardNavigationMixin
     * @mixes Grid.ColumnReorderingMixin
     * @mixes Grid.EventContextMixin
     * @mixes Grid.StylingMixin
     * @mixes Grid.DragAndDropMixin
     * @demo demo/index.html
     */
    class GridElement extends
      ElementMixin$1(
        ThemableMixin(
          DataProviderMixin(
            ArrayDataProviderMixin(
              DynamicColumnsMixin(
                ActiveItemMixin(
                  ScrollMixin(
                    SelectionMixin(
                      SortMixin(
                        RowDetailsMixin(
                          KeyboardNavigationMixin(
                            A11yMixin(
                              FilterMixin(
                                ColumnReorderingMixin(
                                  ColumnResizingMixin(
                                    EventContextMixin(
                                      DragAndDropMixin(
                                        StylingMixin(
                                          GridScrollerElement)))))))))))))))))) {
      static get template() {
        return html`
    <style include="vaadin-grid-styles"></style>

    <div id="scroller" no-scrollbars\$="[[!_scrollbarWidth]]" wheel-scrolling\$="[[_wheelScrolling]]" safari\$="[[_safari]]" ios\$="[[_ios]]" loading\$="[[loading]]" edge\$="[[_edge]]" firefox\$="[[_firefox]]" ie\$="[[_ie]]" column-reordering-allowed\$="[[columnReorderingAllowed]]">

      <table id="table" role="grid" aria-multiselectable="true" tabindex="0">
        <caption id="fixedsizer" class="sizer" part="row"></caption>
        <thead id="header" role="rowgroup"></thead>
        <tbody id="items" role="rowgroup"></tbody>
        <tfoot id="footer" role="rowgroup"></tfoot>
      </table>

      <div part="reorder-ghost"></div>
      <vaadin-grid-outer-scroller id="outerscroller" _touch-device="[[_touchDevice]]" scroll-target="[[scrollTarget]]" scroll-handler="[[_this]]" no-scrollbars="[[!_scrollbarWidth]]">
        <div id="outersizer" class="sizer" part="row"></div>
      </vaadin-grid-outer-scroller>
    </div>

    <!-- The template needs at least one slot or else shady doesn't distribute -->
    <slot name="nodistribute"></slot>

    <div id="focusexit" tabindex="0"></div>
`;
      }

      static get is() {
        return 'vaadin-grid';
      }

      static get version() {
        return '5.6.6';
      }

      static get observers() {
        return [
          '_columnTreeChanged(_columnTree, _columnTree.*)'
        ];
      }

      static get properties() {
        return {

          _this: {
            type: Object,
            value: function() {
              return this;
            }
          },

          _safari: {
            type: Boolean,
            value: /^((?!chrome|android).)*safari/i.test(navigator.userAgent)
          },

          _ios: {
            type: Boolean,
            value: (/iPad|iPhone|iPod/.test(navigator.userAgent) && !window.MSStream)
              || (navigator.platform === 'MacIntel' && navigator.maxTouchPoints > 1)
          },

          _edge: {
            type: Boolean,
            value: typeof CSS !== 'undefined' && CSS.supports('(-ms-ime-align:auto)')
          },

          _ie: {
            type: Boolean,
            value: !!(navigator.userAgent.match(/Trident/) && !navigator.userAgent.match(/MSIE/))
          },

          _firefox: {
            type: Boolean,
            value: navigator.userAgent.toLowerCase().indexOf('firefox') > -1
          },

          _android: {
            type: Boolean,
            value: /android/i.test(navigator.userAgent)
          },

          _touchDevice: {
            type: Boolean,
            value: TOUCH_DEVICE
          },

          /**
           * If true, the grid's height is defined by its rows.
           *
           * Effectively, this disables the grid's virtual scrolling so that all the rows are rendered in the DOM at once.
           * If the grid has a large number of items, using the feature is discouraged to avoid performance issues.
           */
          heightByRows: {
            type: Boolean,
            value: false,
            reflectToAttribute: true,
            observer: '_heightByRowsChanged'
          },
          _recalculateColumnWidthOnceLoadingFinished: {
            type: Boolean,
            value: true
          }
        };
      }

      constructor() {
        super();
        this.addEventListener('animationend', this._onAnimationEnd);
      }

      connectedCallback() {
        super.connectedCallback();
        this.recalculateColumnWidths();
      }

      attributeChangedCallback(name, oldValue, newValue) {
        super.attributeChangedCallback(name, oldValue, newValue);
        if (name === 'dir') {
          this.__isRTL = newValue === 'rtl';
          this._updateScrollerMeasurements();
        }
      }

      __hasRowsWithClientHeight() {
        return !!Array.from(this.$.items.children).filter(row => row.clientHeight).length;
      }

      __itemsReceived() {
        if (this._recalculateColumnWidthOnceLoadingFinished
          && !this._cache.isLoading()
          && this.__hasRowsWithClientHeight()) {
          this._recalculateColumnWidthOnceLoadingFinished = false;
          this.recalculateColumnWidths();
        }
      }

      /**
       * @param {Array<Vaadin.GridColumnElement>} cols the columns to auto size based on their content width
       */
      _recalculateColumnWidths(cols) {
        // Note: The `cols.forEach()` loops below could be implemented as a single loop but this has been
        // split for performance reasons to batch these similar actions [write/read] together to avoid
        // unnecessary layout trashing.

        // [write] Set automatic width for all cells (breaks column alignment)
        cols.forEach(col => {
          col.width = 'auto';
          col._origFlexGrow = col.flexGrow;
          col.flexGrow = 0;
        });
        // [read] Measure max cell width in each column
        cols.forEach(col => {
          col._currentWidth = 0;
          // Note: _allCells only contains cells which are currently rendered in DOM
          col._allCells.forEach(c => {
            // Add 1px buffer to the offset width to avoid too narrow columns (sub-pixel rendering)
            const cellWidth = c.offsetWidth + 1;
            col._currentWidth = Math.max(col._currentWidth, cellWidth);
          });
        });
        // [write] Set column widths to fit widest measured content
        cols.forEach(col => {
          col.width = `${col._currentWidth}px`;
          col.flexGrow = col._origFlexGrow;
          col._currentWidth = undefined;
          col._origFlexGrow = undefined;
        });
      }

      /**
       * Updates the `width` of all columns which have `autoWidth` set to `true`.
       */
      recalculateColumnWidths() {
        if (!this._columnTree) {
          return; // No columns
        }
        if (this._cache.isLoading()) {
          this._recalculateColumnWidthOnceLoadingFinished = true;
        } else {
          const cols = this._getColumns().filter(col => !col.hidden && col.autoWidth);
          this._recalculateColumnWidths(cols);

        }
      }

      _createScrollerRows(count) {
        const rows = [];
        for (var i = 0; i < count; i++) {
          const row = document.createElement('tr');
          row.setAttribute('part', 'row');
          row.setAttribute('role', 'row');
          if (this._columnTree) {
            this._updateRow(row, this._columnTree[this._columnTree.length - 1], 'body', false, true);
          }
          rows.push(row);
        }

        if (this._columnTree) {
          this._columnTree[this._columnTree.length - 1].forEach(c => c.notifyPath && c.notifyPath('_cells.*', c._cells));
        }

        beforeNextRender(this, () => {
          this._updateFirstAndLastColumn();
          this._resetKeyboardNavigation();
        });
        return rows;
      }

      _getRowTarget() {
        return this.$.items;
      }

      _createCell(tagName) {
        const contentId = this._contentIndex = this._contentIndex + 1 || 0;
        const slotName = 'vaadin-grid-cell-content-' + contentId;

        const cellContent = document.createElement('vaadin-grid-cell-content');
        cellContent.setAttribute('slot', slotName);

        const cell = document.createElement(tagName);
        cell.id = slotName.replace('-content-', '-');
        cell.setAttribute('tabindex', '-1');
        cell.setAttribute('role', tagName === 'td' ? 'gridcell' : 'columnheader');

        const slot = document.createElement('slot');
        slot.setAttribute('name', slotName);

        cell.appendChild(slot);

        cell._content = cellContent;

        // With native Shadow DOM, mousedown on slotted element does not focus
        // focusable slot wrapper, that is why cells are not focused with
        // mousedown. Workaround: listen for mousedown and focus manually.
        cellContent.addEventListener('mousedown', () => {
          if (window.chrome) {
            // Chrome bug: focusing before mouseup prevents text selection, see http://crbug.com/771903
            const mouseUpListener = () => {
              if (!cellContent.contains(this.getRootNode().activeElement)) {
                cell.focus();
              }
              // If focus is in the cell content — respect it, do not change.
              document.removeEventListener('mouseup', mouseUpListener, true);
            };
            document.addEventListener('mouseup', mouseUpListener, true);
          } else {
            // Focus on mouseup, on the other hand, removes selection on Safari.
            // Watch out sync focus removal issue, only async focus works here.
            setTimeout(() => {
              if (!cellContent.contains(this.getRootNode().activeElement)) {
                cell.focus();
              }
            });
          }
        });

        return cell;
      }

      _updateRow(row, columns, section, isColumnRow, noNotify) {
        section = section || 'body';

        const contentsFragment = document.createDocumentFragment();

        Array.from(row.children).forEach(cell => cell._vacant = true);
        row.innerHTML = '';
        if (row.id !== 'outersizer' && row.id !== 'fixedsizer') {
          row.hidden = true;
        }
        columns
          .filter(column => !column.hidden)
          .forEach((column, index, cols) => {
            let cell;

            if (section === 'body') {
            // Body
              column._cells = column._cells || [];
              cell = column._cells.filter(cell => cell._vacant)[0];
              if (!cell) {
                cell = this._createCell('td');
                column._cells.push(cell);
              }
              cell.setAttribute('part', 'cell body-cell');
              row.appendChild(cell);

              if (index === cols.length - 1 && (this._rowDetailsTemplate || this.rowDetailsRenderer)) {
              // Add details cell as last cell to body rows
                this._detailsCells = this._detailsCells || [];
                const detailsCell = this._detailsCells.filter(cell => cell._vacant)[0] || this._createCell('td');
                if (this._detailsCells.indexOf(detailsCell) === -1) {
                  this._detailsCells.push(detailsCell);
                }
                if (!detailsCell._content.parentElement) {
                  contentsFragment.appendChild(detailsCell._content);
                }
                this._configureDetailsCell(detailsCell);
                row.appendChild(detailsCell);
                this._a11ySetRowDetailsCell(row, detailsCell);
                detailsCell._vacant = false;
              }

              if (column.notifyPath && !noNotify) {
                column.notifyPath('_cells.*', column._cells);
              }
            } else {
            // Header & footer
              const tagName = section === 'header' ? 'th' : 'td';
              if (isColumnRow || column.localName === 'vaadin-grid-column-group') {
                cell = column[`_${section}Cell`] || this._createCell(tagName);
                cell._column = column;
                row.appendChild(cell);
                column[`_${section}Cell`] = cell;
              } else {
                column._emptyCells = column._emptyCells || [];
                cell = column._emptyCells.filter(cell => cell._vacant)[0] || this._createCell(tagName);
                cell._column = column;
                row.appendChild(cell);
                if (column._emptyCells.indexOf(cell) === -1) {
                  column._emptyCells.push(cell);
                }
              }
              cell.setAttribute('part', `cell ${section}-cell`);
              this.__updateHeaderFooterRowVisibility(row);
            }

            if (!cell._content.parentElement) {
              contentsFragment.appendChild(cell._content);
            }
            cell._vacant = false;
            cell._column = column;
          });

        // Might be empty if only cache was used
        this.appendChild(contentsFragment);

        this._frozenCellsChanged();
        this._updateFirstAndLastColumnForRow(row);
      }

      __updateHeaderFooterRowVisibility(row) {
        if (!row) {
          return;
        }

        const visibleRowCells = Array.from(row.children).filter(cell => {
          const column = cell._column;
          if (column._emptyCells && column._emptyCells.indexOf(cell) > -1) {
            // The cell is an "empty cell"  -> doesn't block hiding the row
            return false;
          }
          if (row.parentElement === this.$.header) {
            if (column.headerRenderer || column._headerTemplate) {
              // The cell is the header cell of a column that has a header renderer
              // or a header template -> row should be visible
              return true;
            }
            if (column.header === null) {
              // The column header is explicilty set to null -> doesn't block hiding the row
              return false;
            }
            if (column.path || column.header !== undefined) {
              // The column has an explicit non-null header or a path that generates a header
              // -> row should be visible
              return true;
            }
          } else {
            if (column.footerRenderer || column._footerTemplate) {
              // The cell is the footer cell of a column that has a footer renderer
              // or a footer template -> row should be visible
              return true;
            }
          }
        });

        if (row.hidden !== !visibleRowCells.length) {
          row.hidden = !visibleRowCells.length;
          this.notifyResize();
        }
      }

      _updateScrollerItem(row, index) {
        this._preventScrollerRotatingCellFocus(row, index);

        if (!this._columnTree) {
          return;
        }

        this._toggleAttribute('first', index === 0, row);
        this._toggleAttribute('odd', index % 2, row);
        this._a11yUpdateRowRowindex(row, index);
        this._getItem(index, row);
      }

      _columnTreeChanged(columnTree, splices) {
        this._renderColumnTree(columnTree);
        this.recalculateColumnWidths();
      }

      _renderColumnTree(columnTree) {
        Array.from(this.$.items.children).forEach((row) => this._updateRow(row, columnTree[columnTree.length - 1], null, false, true));

        while (this.$.header.children.length < columnTree.length) {
          const headerRow = document.createElement('tr');
          headerRow.setAttribute('part', 'row');
          headerRow.setAttribute('role', 'row');
          this.$.header.appendChild(headerRow);

          const footerRow = document.createElement('tr');
          footerRow.setAttribute('part', 'row');
          footerRow.setAttribute('role', 'row');
          this.$.footer.appendChild(footerRow);
        }
        while (this.$.header.children.length > columnTree.length) {
          this.$.header.removeChild(this.$.header.firstElementChild);
          this.$.footer.removeChild(this.$.footer.firstElementChild);
        }

        Array.from(this.$.header.children)
          .forEach((headerRow, index) => this._updateRow(headerRow, columnTree[index], 'header', index === columnTree.length - 1));

        Array.from(this.$.footer.children)
          .forEach((footerRow, index) => this._updateRow(footerRow, columnTree[columnTree.length - 1 - index], 'footer', index === 0));

        // Sizer rows
        this._updateRow(this.$.outersizer, columnTree[columnTree.length - 1], null, false, true);
        this._updateRow(this.$.fixedsizer, columnTree[columnTree.length - 1]);

        this._resizeHandler();
        this._frozenCellsChanged();
        this._updateFirstAndLastColumn();
        this._resetKeyboardNavigation();
        this._a11yUpdateHeaderRows();
        this._a11yUpdateFooterRows();
      }

      _updateItem(row, item) {
        row._item = item;
        const model = this.__getRowModel(row);

        this._toggleAttribute('selected', model.selected, row);
        this._a11yUpdateRowSelected(row, model.selected);
        this._a11yUpdateRowLevel(row, model.level);
        this._toggleAttribute('expanded', model.expanded, row);
        if (this._rowDetailsTemplate || this.rowDetailsRenderer) {
          this._toggleDetailsCell(row, item);
        }
        this._generateCellClassNames(row, model);
        this._filterDragAndDrop(row, model);

        Array.from(row.children).forEach(cell => {
          if (cell._renderer) {
            const owner = cell._column || this;
            cell._renderer.call(owner, cell._content, owner, model);
          } else if (cell._instance) {
            cell._instance.__detailsOpened__ = model.detailsOpened;
            cell._instance.__selected__ = model.selected;
            cell._instance.__level__ = model.level;
            cell._instance.__expanded__ = model.expanded;
            cell._instance.setProperties(model);
          }
        });

        this._debouncerUpdateHeights = Debouncer.debounce(this._debouncerUpdateHeights,
          timeOut.after(1), () => {
            this._updateMetrics();
            this._positionItems();
            this._updateScrollerSize();
          }
        );
      }

      _resizeHandler() {
        this._updateDetailsCellHeights();
        this._accessIronListAPI(super._resizeHandler, true);
        this._updateScrollerMeasurements();
        this._updateHeaderFooterMetrics();
      }

      _updateHeaderFooterMetrics() {
        const headerHeight = this.$.header.clientHeight + 'px';
        const footerHeight = this.$.footer.clientHeight + 'px';
        [this.$.outersizer, this.$.fixedsizer, this.$.items].forEach(element => {
          element.style.borderTopWidth = headerHeight;
          element.style.borderBottomWidth = footerHeight;
        });

        afterNextRender(this.$.header, () => {
          if (this._pendingScrollToIndex) {
            this._scrollToIndex(this._pendingScrollToIndex);
          }
        });
      }

      _onAnimationEnd(e) {
        // ShadyCSS applies scoping suffixes to animation names
        if (e.animationName.indexOf('vaadin-grid-appear') === 0) {
          this._render();
          this._updateHeaderFooterMetrics();
          e.stopPropagation();
          this.notifyResize();
          this.__itemsReceived();
        }
      }

      _toggleAttribute(name, bool, node) {
        if (node.hasAttribute(name) === !bool) {
          if (bool) {
            node.setAttribute(name, '');
          } else {
            node.removeAttribute(name);
          }
        }
      }

      __getRowModel(row) {
        return {
          index: row.index,
          item: row._item,
          level: this._getIndexLevel(row.index),
          expanded: this._isExpanded(row._item),
          selected: this._isSelected(row._item),
          detailsOpened:
            !!(this._rowDetailsTemplate || this.rowDetailsRenderer) && this._isDetailsOpened(row._item)
        };
      }

      /**
       * Manually invoke existing renderers for all the columns
       * (header, footer and body cells) and opened row details.
       */
      render() {
        if (this._columnTree) {
          // header and footer renderers
          this._columnTree.forEach(level => {
            level.forEach(column => column._renderHeaderAndFooter());
          });

          // body and row details renderers
          this._update();
        }
      }

      /**
       * Updates the computed metrics and positioning of internal grid parts
       * (row/details cell positioning etc). Needs to be invoked whenever the sizing of grid
       * content changes asynchronously to ensure consistent appearance (e.g. when a
       * contained image whose bounds aren't known beforehand finishes loading).
       */
      notifyResize() {
        super.notifyResize();
      }

      _heightByRowsChanged(value, oldValue) {
        if (value || oldValue) {
          this.notifyResize();
        }
      }

      __forceReflow() {
        this._debouncerForceReflow = Debouncer.debounce(this._debouncerForceReflow,
          animationFrame, () => {
            this.$.scroller.style.overflow = 'hidden';
            setTimeout(() => this.$.scroller.style.overflow = '');
          }
        );
      }
    }

    customElements.define(GridElement.is, GridElement);

    /* Copyright 2016 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    let TfDownloader = class TfDownloader extends PolymerElement {
        constructor() {
            super(...arguments);
            this._run = '';
        }
        _csvUrl(tag, run, urlFn) {
            if (!run)
                return '';
            return addParams(urlFn(tag, run), { format: 'csv' });
        }
        _jsonUrl(tag, run, urlFn) {
            if (!run)
                return '';
            return urlFn(tag, run);
        }
        _csvName(tag, run) {
            if (!run)
                return '';
            return `run-${run}-tag-${tag}.csv`;
        }
        _jsonName(tag, run) {
            if (!run)
                return '';
            return `run-${run}-tag-${tag}.json`;
        }
    };
    TfDownloader.template = html `
    <paper-dropdown-menu
      no-label-float="true"
      label="run to download"
      selected-item-label="{{_run}}"
    >
      <paper-listbox slot="dropdown-content">
        <template is="dom-repeat" items="[[runs]]">
          <paper-item no-label-float="true">[[item]]</paper-item>
        </template>
      </paper-listbox>
    </paper-dropdown-menu>
    <template is="dom-if" if="[[_run]]">
      <a download="[[_csvName(tag, _run)]]" href="[[_csvUrl(tag, _run, urlFn)]]"
        >CSV</a
      ><!--
      --><a
        download="[[_jsonName(tag, _run)]]"
        href="[[_jsonUrl(tag, _run, urlFn)]]"
        >JSON</a
      >
    </template>
    <style>
      :host {
        display: flex;
        align-items: center;
        height: 32px;
      }
      paper-dropdown-menu {
        width: 100px;
        --paper-input-container-label: {
          font-size: 10px;
        }
        --paper-input-container-input: {
          font-size: 10px;
        }
      }
      a {
        font-size: 10px;
        margin: 0 0.2em;
      }
      paper-input {
        font-size: 22px;
      }
    </style>
  `;
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfDownloader.prototype, "_run", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfDownloader.prototype, "runs", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfDownloader.prototype, "tag", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfDownloader.prototype, "urlFn", void 0);
    TfDownloader = __decorate([
        customElement('tf-downloader')
    ], TfDownloader);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * Save the initial URL query params, before the AppRoutingEffects initialize.
     */
    const initialURLSearchParams$1 = new URLSearchParams(window.location.search);
    /**
     * A card that handles loading data (at the right times), rendering a scalar
     * chart, and providing UI affordances (such as buttons) for scalar data.
     */
    let TfScalarCard = 
    // tslint:disable-next-line:no-unused-variable
    class TfScalarCard extends PolymerElement {
        constructor() {
            super(...arguments);
            // If specified uses the given colorScale; otherwise, uses a built-in
            // default. See _getColorScale below for more details.
            this.colorScale = null;
            // This function is called when data is received from the backend.
            this._loadDataCallback = (scalarChart, item, maybeData) => {
                if (maybeData == null) {
                    console.error('Failed to load data for:', item);
                    return;
                }
                const formattedData = maybeData.map((datum) => ({
                    wall_time: new Date(datum[0] * 1000),
                    step: datum[1],
                    scalar: datum[2],
                }));
                const name = this._getSeriesNameFromDatum(item);
                scalarChart.setSeriesMetadata(name, item);
                scalarChart.setSeriesData(name, formattedData);
            };
            this.getDataLoadUrl = ({ tag, run }) => {
                return getRouter().pluginRoute('scalars', '/scalars', new URLSearchParams({ tag, run }));
            };
            // To be provided as the `url-fn` property to `tf-downloader`.
            this._downloadUrlFn = (tag, run) => this.getDataLoadUrl({ tag, run });
            // A function called to fetch the scalars data from the backend.
            // Should receive a {tag, run, experiment} object and return
            // a promise resolving with the fetched scalars. The default
            // implementation of this function executes:
            // this.requestManager.request(
            //      this.getDataLoadUrl({tag, run, experiment})
            this.requestData = (items, onLoad, onFinish) => {
                // Google-internal Colab doesn't support HTTP POST requests, so we fall
                // back to HTTP GET (even though public Colab supports POST).
                // See b/126387106.
                const inColab = initialURLSearchParams$1.get('tensorboardColab') === 'true';
                if (inColab) {
                    return this._requestDataGet(items, onLoad, onFinish);
                }
                else {
                    return this._requestDataPost(items, onLoad, onFinish);
                }
            };
            this._requestDataGet = (items, onLoad, onFinish) => {
                const router = getRouter();
                const baseUrl = router.pluginRoute('scalars', '/scalars');
                Promise.all(items.map((item) => {
                    const url = addParams(baseUrl, { tag: item.tag, run: item.run });
                    return this.requestManager
                        .request(url)
                        .then((data) => void onLoad({ item, data }));
                })).finally(() => void onFinish());
            };
            this._requestDataPost = (items, onLoad, onFinish) => {
                const router = getRouter();
                const url = router.pluginRoute('scalars', '/scalars_multirun');
                const runsByTag = new Map();
                for (const { tag, run } of items) {
                    let runs = runsByTag.get(tag);
                    if (runs == null) {
                        runsByTag.set(tag, (runs = []));
                    }
                    runs.push(run);
                }
                // Request at most this many runs at once.
                //
                // Back-of-the-envelope math: each scalar datum JSON value contains
                // two floats and a small-ish integer. Floats are about 18 bytes,
                // since f64s have -log_10(2^-53) ~= 16 digits of precision plus
                // decimal point and leading zero. Small-ish integers (steps) are
                // about 5 bytes. Add JSON overhead `[,,],` and you're looking at
                // about 48 bytes per datum. With standard downsampling of
                // 1000 points per time series, expect ~50 KB of response payload
                // per requested time series.
                //
                // Requesting 64 time series warrants a ~3 MB response, which seems
                // reasonable.
                const BATCH_SIZE = 64;
                const requestGroups = [];
                for (const [tag, runs] of runsByTag) {
                    for (let i = 0; i < runs.length; i += BATCH_SIZE) {
                        requestGroups.push({ tag, runs: runs.slice(i, i + BATCH_SIZE) });
                    }
                }
                Promise.all(requestGroups.map(({ tag, runs }) => {
                    return this.requestManager.request(url, { tag, runs }).then((allData) => {
                        for (const run of runs) {
                            const item = { tag, run };
                            if (Object.prototype.hasOwnProperty.call(allData, run)) {
                                onLoad({ item, data: allData[run] });
                            }
                            else {
                                onLoad({ item, data: null });
                            }
                        }
                    });
                })).finally(() => void onFinish());
            };
            this._getDataLoadName = (datum) => this._getSeriesNameFromDatum(datum);
            this._expanded = false;
            this._tooltipColumns = (() => {
                const columns = DEFAULT_TOOLTIP_COLUMNS.slice();
                const ind = columns.findIndex((c) => c.title == 'Name');
                columns.splice(ind, 1, {
                    title: 'Name',
                    evaluate: (d) => {
                        const datum = d.dataset.metadata().meta;
                        return this._getSeriesDisplayNameFromDatum(datum);
                    },
                });
                return columns;
            })();
        }
        _getChartDataLoader() {
            // tslint:disable-next-line:no-unnecessary-type-assertion
            return this.shadowRoot.querySelector('tf-line-chart-data-loader'); // TfLineChartDataLoader
        }
        reload() {
            this._getChartDataLoader().reload();
        }
        redraw() {
            this._getChartDataLoader().redraw();
        }
        _toggleExpanded(e) {
            this.set('_expanded', !this._expanded);
            this.redraw();
        }
        _toggleLogScale() {
            this.set('_logScaleActive', !this._logScaleActive);
        }
        _resetDomain() {
            const chart = this._getChartDataLoader();
            if (chart) {
                chart.resetDomain();
            }
        }
        _updateDownloadLink() {
            const svgStr = this._getChartDataLoader().exportAsSvgString();
            // The SVG code string may include hash characters, such as an
            // attribute `clipPath="url(#foo)"`. Thus, we base64-encode the
            // data so that such a hash is not interpreted as a fragment
            // specifier, truncating the SVG. (See issue #1874.)
            // tslint:disable-next-line:no-unnecessary-type-assertion
            const svgLink = this.shadowRoot.querySelector('#svgLink');
            svgLink.href = `data:image/svg+xml;base64,${btoa(svgStr)}`;
        }
        _runsFromData(data) {
            return data.map((datum) => datum.run);
        }
        _getDataSeries() {
            return this.dataToLoad.map((d) => this._getSeriesNameFromDatum(d));
        }
        // name is a stable identifier for a series.
        _getSeriesNameFromDatum({ run, experiment = { name: '_default' } }) {
            return JSON.stringify([experiment.name, run]);
        }
        // title is a visible string of a series for the UI.
        _getSeriesDisplayNameFromDatum(datum) {
            return datum.run;
        }
        _getColorScale() {
            if (this.colorScale !== null) {
                return this.colorScale;
            }
            // If 'colorScale' isn't explicitly specified, use the ones
            // defined in tf_color_scale.
            return {
                scale: (name) => {
                    const [, run] = JSON.parse(name);
                    return runsColorScale(run);
                },
            };
        }
    };
    TfScalarCard.template = html `
    <tf-card-heading
      tag="[[tag]]"
      display-name="[[tagMetadata.displayName]]"
      description="[[tagMetadata.description]]"
    ></tf-card-heading>
    <div id="tf-line-chart-data-loader-container">
      <tf-line-chart-data-loader
        active="[[active]]"
        color-scale="[[_getColorScale(colorScale)]]"
        data-series="[[_getDataSeries(dataToLoad.*)]]"
        data-to-load="[[dataToLoad]]"
        get-data-load-name="[[_getDataLoadName]]"
        get-data-load-url="[[getDataLoadUrl]]"
        request-data="[[requestData]]"
        ignore-y-outliers="[[ignoreYOutliers]]"
        load-data-callback="[[_loadDataCallback]]"
        load-key="[[tag]]"
        log-scale-active="[[_logScaleActive]]"
        request-manager="[[requestManager]]"
        smoothing-enabled="[[smoothingEnabled]]"
        smoothing-weight="[[smoothingWeight]]"
        tag-metadata="[[tagMetadata]]"
        tooltip-columns="[[_tooltipColumns]]"
        tooltip-position="auto"
        tooltip-sorting-method="[[tooltipSortingMethod]]"
        x-type="[[xType]]"
      >
      </tf-line-chart-data-loader>
    </div>
    <div id="buttons">
      <paper-icon-button
        selected$="[[_expanded]]"
        icon="fullscreen"
        on-tap="_toggleExpanded"
      ></paper-icon-button>
      <paper-icon-button
        selected$="[[_logScaleActive]]"
        icon="line-weight"
        on-tap="_toggleLogScale"
        title="Toggle y-axis log scale"
      ></paper-icon-button>
      <paper-icon-button
        icon="settings-overscan"
        on-tap="_resetDomain"
        title="Fit domain to data"
      ></paper-icon-button>
      <template is="dom-if" if="[[showDownloadLinks]]">
        <paper-menu-button on-paper-dropdown-open="_updateDownloadLink">
          <paper-icon-button
            class="dropdown-trigger"
            slot="dropdown-trigger"
            icon="file-download"
          ></paper-icon-button>
          <paper-listbox class="dropdown-content" slot="dropdown-content">
            <paper-item>
              <a id="svgLink" download="[[tag]].svg">
                Download Current Chart as SVG
              </a>
            </paper-item>
          </paper-listbox>
        </paper-menu-button>
      </template>
      <span style="flex-grow: 1"></span>
      <template is="dom-if" if="[[showDownloadLinks]]">
        <div class="download-links">
          <tf-downloader
            runs="[[_runsFromData(dataToLoad)]]"
            tag="[[tag]]"
            url-fn="[[_downloadUrlFn]]"
          ></tf-downloader>
        </div>
      </template>
    </div>
    <style>
      :host {
        margin: 5px;
        display: block;
        width: 330px;
      }

      :host([_expanded]) {
        width: 100%;
      }

      :host([_expanded]) #tf-line-chart-data-loader-container {
        height: 400px;
      }

      #tf-line-chart-data-loader-container {
        height: 200px;
        width: 100%;
      }

      tf-card-heading {
        display: block;
        margin-bottom: 10px;
      }

      #buttons {
        display: flex;
        flex-direction: row;
      }

      paper-icon-button {
        color: #2196f3;
        border-radius: 100%;
        width: 32px;
        height: 32px;
        padding: 4px;
      }

      paper-icon-button[selected] {
        background: var(--tb-ui-light-accent);
      }

      .download-links {
        display: flex;
        height: 32px;
      }

      .download-links a {
        align-self: center;
        font-size: 10px;
        margin: 2px;
      }

      .download-links paper-dropdown-menu {
        width: 100px;
        --paper-input-container-label: {
          font-size: 10px;
        }
        --paper-input-container-input: {
          font-size: 10px;
        }
      }

      paper-menu-button {
        padding: 0;
      }
      paper-item a {
        color: inherit;
        text-decoration: none;
        white-space: nowrap;
      }
    </style>
  `;
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfScalarCard.prototype, "tag", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfScalarCard.prototype, "dataToLoad", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfScalarCard.prototype, "xType", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfScalarCard.prototype, "active", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfScalarCard.prototype, "ignoreYOutliers", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", RequestManager)
    ], TfScalarCard.prototype, "requestManager", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfScalarCard.prototype, "showDownLinks", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfScalarCard.prototype, "smoothingEnabled", void 0);
    __decorate([
        property({ type: Number }),
        __metadata("design:type", Number)
    ], TfScalarCard.prototype, "smoothingWeight", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfScalarCard.prototype, "tagMetadata", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfScalarCard.prototype, "colorScale", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfScalarCard.prototype, "tooltipSortingMethod", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfScalarCard.prototype, "_loadDataCallback", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Function)
    ], TfScalarCard.prototype, "getDataLoadUrl", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfScalarCard.prototype, "_downloadUrlFn", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Function)
    ], TfScalarCard.prototype, "requestData", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfScalarCard.prototype, "_getDataLoadName", void 0);
    __decorate([
        property({
            type: Boolean,
            reflectToAttribute: true,
        }),
        __metadata("design:type", Boolean)
    ], TfScalarCard.prototype, "_expanded", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfScalarCard.prototype, "_logScaleActive", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfScalarCard.prototype, "_tooltipColumns", void 0);
    TfScalarCard = __decorate([
        customElement('tf-scalar-card')
        // tslint:disable-next-line:no-unused-variable
    ], TfScalarCard);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * Shows a session group in more detail. Specifically, shows graphs of the
     * metrics for the session in the group as a function of the training step.
     * This element is shown one the user "drills-in" to a session group, for
     * example when she clicks on a row in table-view or a curve in parallel-coords
     * view.
     */
    let TfHparamsSessionGroupDetails = class TfHparamsSessionGroupDetails extends mixinBehaviors([IronResizableBehavior$1], PolymerElement) {
        constructor() {
            super(...arguments);
            // The session group object to display. Matches the SessionGroup
            // protocol buffer defined in api.proto.
            this.sessionGroup = null;
            this._xType = XType.STEP;
            this._noMultiExperiments = false;
            this._requestData = (items, onLoad, onFinish) => {
                Promise.all(items.map((item) => {
                    const request = {
                        experimentName: this.experimentName,
                        sessionName: item.run,
                        metricName: item.tag,
                    };
                    return this.backend
                        .listMetricEvals(request)
                        .then((data) => void onLoad({ item, data }));
                })).finally(() => void onFinish());
            };
            this._colorScale = {
                scale: (seriesName) => {
                    // The tf-scalar-card API sends a seriesName as a JSON
                    // representation of the 2-element array [experiment, run],
                    // where 'run' is the run we return in the 'requestData'
                    // function--in our case that is the session name, and
                    // 'experiment' is not used here tf-scalar-card's
                    // multi-experiment attribute is turned off here).
                    // The color of a session is determined by choosing a
                    // starting point in 'palette' based on the sessionGroup name
                    // and walking cyclicly 'sessionIndex' steps in the array, where
                    // sesionIndex is the index of the session in the
                    // group. We do this instead of just hashing the session name,
                    // to guarantee that we don't repeat a color if the number
                    // of sessions is at most the size of the palette.
                    // Note that this method is called every time the user moves
                    // over the metric plot, so it needs to be reasonably fast.
                    const sessionName = JSON.parse(seriesName)[1];
                    const sessionIndex = this._indexOfSession.get(sessionName);
                    const palette = standard;
                    return palette[(this._sessionGroupNameHash + sessionIndex) % palette.length];
                },
            };
        }
        // Whenever the backend sends a new 'sessionGroups' array reply to the
        // ListSessionGroups RPC, we recreate the tf-session-group-detail
        // elements that correspond to session groups whose name is in the new
        // sessionGroups array. As such this element can be constructed when one
        // of its parents container is invisible: for example, if sessionGroups
        // was replaced when the user was on the parallel-coordinates view tab
        // and the new sessionGroups contains a session group object
        // whose metrics were open in the table view tab (i.e. the old
        // sessionGroups array contained a distinct JavaScript sessionGroup
        // object with same name).
        // If this happens, the tf-scalar-card child elements of this element
        // would measure a 0 size of their bounding boxes and would render
        // incorrectly. To avoid that, we listen for the 'iron-resize' event
        // that is fired by 'iron-pages' whenever the active page changes and
        // redraw each tf-scalar-card child.
        connectedCallback() {
            super.connectedCallback();
            this.addEventListener('iron-resize', this.redraw.bind(this));
        }
        redraw() {
            this.shadowRoot
                .querySelectorAll('tf-scalar-card')
                .forEach((c) => c.redraw());
        }
        _sessionGroupChanged() {
            if (!this.sessionGroup || Object.keys(this.sessionGroup).length == 0) {
                this._indexOfSession = new Map();
                this._sessionGroupNameHash = 0;
            }
            else {
                this._indexOfSession = new Map(this.sessionGroup.sessions.map((session, index) => [
                    session.name,
                    index,
                ]));
                this._sessionGroupNameHash = hashOfString(this.sessionGroup.name);
            }
            // Reset each scalar-card by prodding 'tag'.
            // We do this so the card will reset its domain on the next load.
            this.shadowRoot.querySelectorAll('tf-scalar-card').forEach((c) => {
                const anyCard = c;
                const tag = anyCard.get('tag');
                anyCard.set('tag', '');
                anyCard.set('tag', tag);
            });
        }
        _haveMetrics() {
            return (this.visibleSchema &&
                Array.isArray(this.visibleSchema.metricInfos) &&
                this.visibleSchema.metricInfos.length > 0);
        }
        _haveMetricsAndSessionGroup() {
            return this.sessionGroup && this._haveMetrics();
        }
        // Returns an array of the tensorboard 'runs' and 'tags' corresponding to
        // the given metric in each of the sessions in the given session group.
        _computeSeriesForSessionGroupMetric(sessionGroup, metricInfo) {
            if (sessionGroup === null ||
                Object.keys(sessionGroup).length == 0 ||
                metricInfo === null) {
                return [];
            }
            return sessionGroup.sessions
                .filter(
            // If this session does not currently have a value for the given
            // metric then we don't want to include its run.
            (session) => metricValueByName(session.metricValues, metricInfo.name) !== undefined)
                .map(
            // Since the 'tag' and 'run' fields are essentially opaque
            // for the tf-scalar-card element, and passed as is to the
            // requestData function, we use them to encode the fields we
            // need to supply to the eval_metrics API (called in our
            // implementation of requestData above).
            (session) => ({ tag: metricInfo.name, run: session.name }));
        }
        // Computes the tag-metadata attribute to send to the
        // tf-scalar-card element.
        _computeTagMetadata(metricInfo) {
            return {
                displayName: metricName(metricInfo),
                description: metricInfo.description || '',
            };
        }
    };
    TfHparamsSessionGroupDetails.template = html `
    <template is="dom-if" if="[[!sessionGroup]]">
      <div>
        <h3>No session group selected</h3>
        <p>Please select a session group to see its metric-graphs here.</p>
      </div>
    </template>
    <template is="dom-if" if="[[!_haveMetrics(visibleSchema.*)]]">
      <div>
        <h3>No metrics are enabled</h3>
        <p>Please enable some metrics to see content here.</p>
      </div>
    </template>
    <div class="layout horizontal wrap session-group-details">
      <template
        is="dom-if"
        if="[[_haveMetricsAndSessionGroup(visibleSchema.*, sessionGroup)]]"
      >
        <template
          is="dom-repeat"
          items="[[visibleSchema.metricInfos]]"
          as="metricInfo"
        >
          <!-- Note that we do not provide a request-manager attribute since
               we provide a function in request-data for calling the backend
               to get the metrics data.
            -->
          <tf-scalar-card
            class="scalar-card"
            color-scale="[[_colorScale]]"
            data-to-load="[[_computeSeriesForSessionGroupMetric(sessionGroup, metricInfo)]]"
            tag="[[metricInfo.name.tag]]"
            tag-metadata="[[_computeTagMetadata(metricInfo)]]"
            x-type="[[_xType]]"
            multi-experiments="[[_noMultiExperiments]]"
            request-data="[[_requestData]]"
            active
          >
          </tf-scalar-card>
        </template>
      </template>
    </div>
    <!-- "iron-flex" is needed to use the layout classes in the div above -->
    <style include="iron-flex">
      :host {
        display: block;
      }
    </style>
  `;
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfHparamsSessionGroupDetails.prototype, "backend", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfHparamsSessionGroupDetails.prototype, "experimentName", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfHparamsSessionGroupDetails.prototype, "visibleSchema", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfHparamsSessionGroupDetails.prototype, "sessionGroup", void 0);
    __decorate([
        property({
            type: String,
        }),
        __metadata("design:type", String)
    ], TfHparamsSessionGroupDetails.prototype, "_xType", void 0);
    __decorate([
        property({
            type: Boolean,
        }),
        __metadata("design:type", Boolean)
    ], TfHparamsSessionGroupDetails.prototype, "_noMultiExperiments", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfHparamsSessionGroupDetails.prototype, "_indexOfSession", void 0);
    __decorate([
        property({ type: Number }),
        __metadata("design:type", Number)
    ], TfHparamsSessionGroupDetails.prototype, "_sessionGroupNameHash", void 0);
    __decorate([
        property({
            type: Object,
        }),
        __metadata("design:type", Function)
    ], TfHparamsSessionGroupDetails.prototype, "_requestData", void 0);
    __decorate([
        property({
            type: Object,
        }),
        __metadata("design:type", Object)
    ], TfHparamsSessionGroupDetails.prototype, "_colorScale", void 0);
    __decorate([
        observe('sessionGroup.*'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfHparamsSessionGroupDetails.prototype, "_sessionGroupChanged", null);
    TfHparamsSessionGroupDetails = __decorate([
        customElement('tf-hparams-session-group-details')
    ], TfHparamsSessionGroupDetails);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    let TfHparamsTableView = class TfHparamsTableView extends LegacyElementMixin(PolymerElement) {
        constructor() {
            super(...arguments);
            this._hparamName = hparamName;
            this._metricName = metricName;
        }
        _visibleSchemaOrSessionGroupsChanged() {
            // From vaadin-grid 3.0.0:
            // Vaadin-grid removes 'row-details' if the visibleSchema changes
            // and doesn't update 'expandedItems'. So we first close the
            // expanded items, call Polymer.dom.flush() to update the grid,
            // and then re-open the groups that were open before.
            //
            // Since then, we have upgraded to vaadin-grid 5.6.6, where the comment
            // may be stale. Notably, Vaadin renamed 'expandedItems' to
            // 'detailsOpenedItems'.
            const expandedItems = this.$.sessionGroupsTable.get('detailsOpenedItems');
            this.$.sessionGroupsTable.set('detailsOpenedItems', []);
            flush();
            // Index sessionGroups by name.
            const sessionGroupsByName = new Map();
            this.sessionGroups.forEach((sg) => {
                sessionGroupsByName.set(sg.name, sg);
            });
            this.$.sessionGroupsTable.set('detailsOpenedItems', expandedItems
                .map((sg) => sessionGroupsByName.get(sg.name))
                .filter(Boolean));
        }
        _sessionGroupHParam(sessionGroup, hparam) {
            if (sessionGroup == null ||
                Object.keys(sessionGroup).length == 0 ||
                !Object.prototype.hasOwnProperty.call(sessionGroup.hparams, hparam)) {
                return '';
            }
            return prettyPrint(sessionGroup.hparams[hparam]);
        }
        // Returns the metric value of the given sessionGroup. The value is
        // returned as a string suitable for display.
        _sessionGroupMetric(sessionGroup, metricName) {
            if (sessionGroup == null || Object.keys(sessionGroup).length == 0) {
                return '';
            }
            for (let i = 0; i < sessionGroup.metricValues.length; ++i) {
                let metricVal = sessionGroup.metricValues[i];
                if (metricVal.name.group === metricName.group &&
                    metricVal.name.tag == metricName.tag) {
                    return prettyPrint(metricVal.value);
                }
            }
            return '';
        }
        _rowNumber(rowIndex) {
            return rowIndex + 1;
        }
    };
    TfHparamsTableView.template = html `
    <vaadin-grid
      class="session-group-table"
      id="sessionGroupsTable"
      column-reordering-allowed=""
      items="[[sessionGroups]]"
    >
      <vaadin-grid-column flex-grow="0" width="10em" resizable="">
        <template class="header">
          <div class="table-header table-cell">Trial ID</div>
        </template>
        <template>
          <div class="table-cell">[[item.name]]</div>
        </template>
      </vaadin-grid-column>
      <template is="dom-if" if="[[enableShowMetrics]]">
        <vaadin-grid-column flex-grow="0" autoWidth="" resizable="">
          <template class="header">
            <div class="table-header table-cell">Show Metrics</div>
          </template>
          <template>
            <paper-checkbox class="table-cell" checked="{{detailsOpened}}">
            </paper-checkbox>
          </template>
        </vaadin-grid-column>
      </template>
      <template
        is="dom-repeat"
        items="[[visibleSchema.hparamInfos]]"
        as="hparamInfo"
        index-as="hparamIndex"
      >
        <vaadin-grid-column flex-grow="2" width="10em" resizable="">
          <template class="header">
            <div class="table-header table-cell">
              [[_hparamName(hparamInfo)]]
            </div>
          </template>
          <template>
            <div class="table-cell">
              [[_sessionGroupHParam(item, hparamInfo.name)]]
            </div>
          </template>
        </vaadin-grid-column>
      </template>
      <template
        is="dom-repeat"
        items="{{visibleSchema.metricInfos}}"
        as="metricInfo"
        index-as="metricIndex"
      >
        <vaadin-grid-column flex-grow="2" width="10em" resizable="">
          <template class="header">
            <div class="table-header table-cell">
              [[_metricName(metricInfo)]]
            </div>
          </template>
          <template>
            <div class="table-cell">
              [[_sessionGroupMetric(item, metricInfo.name)]]
            </div>
          </template>
        </vaadin-grid-column>
      </template>
      <template class="row-details">
        <tf-hparams-session-group-details
          backend="[[backend]]"
          experiment-name="[[experimentName]]"
          session-group="[[item]]"
          visible-schema="[[visibleSchema]]"
          class="session-group-details"
        >
        </tf-hparams-session-group-details>
      </template>
    </vaadin-grid>

    <style>
      :host {
        display: inline;
      }
      .table-cell {
        white-space: nowrap;
        text-overflow: ellipsis;
        overflow: hidden;
      }
      .table-header {
        /* line-break overflowing column headers */
        white-space: normal;
        overflow-wrap: break-word;
      }
      .session-group-table {
        height: 100%;
      }
      .session-group-details {
        height: 360px;
        overflow-y: auto;
      }
    </style>
  `;
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfHparamsTableView.prototype, "visibleSchema", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfHparamsTableView.prototype, "sessionGroups", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfHparamsTableView.prototype, "enableShowMetrics", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfHparamsTableView.prototype, "backend", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfHparamsTableView.prototype, "experimentName", void 0);
    __decorate([
        observe('visibleSchema.*', 'sessionGroups.*'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfHparamsTableView.prototype, "_visibleSchemaOrSessionGroupsChanged", null);
    TfHparamsTableView = __decorate([
        customElement('tf-hparams-table-view')
    ], TfHparamsTableView);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    let TfHparamsSessionGroupValues = class TfHparamsSessionGroupValues extends PolymerElement {
        constructor() {
            super(...arguments);
            // The session group whose values will be displayed.
            // A JavaScript representation of the tensorboard.hparams.SessionGroup
            // protocol buffer.
            this.sessionGroup = null;
            // The visibleSchema defining the columns to use.
            // Only the values of columns in the visibleSchema will be displayed.
            // See tf-hparams-query-pane for the definition of this object.
            this.visibleSchema = null;
        }
        _propertiesArePopulated(visibleSchema, sessionGroup) {
            return (visibleSchema !== undefined &&
                visibleSchema !== null &&
                sessionGroup !== undefined &&
                sessionGroup !== null);
        }
        _singletonSessionGroups(sessionGroup) {
            if (sessionGroup === null || sessionGroup === undefined) {
                return [];
            }
            return [sessionGroup];
        }
    };
    TfHparamsSessionGroupValues.template = html `
    <!-- If sessionGroup or visibleSchema are not populated, do not display
         anything.
      -->
    <template
      is="dom-if"
      if="[[_propertiesArePopulated(visibleSchema, sessionGroup)]]"
    >
      <!-- Display one row without a "show-metrics" column -->
      <tf-hparams-table-view
        visible-schema="[[visibleSchema]]"
        session-groups="[[_singletonSessionGroups(sessionGroup)]]"
      >
      </tf-hparams-table-view>
    </template>
    <template
      is="dom-if"
      if="[[!_propertiesArePopulated(visibleSchema, sessionGroup)]]"
    >
      <div>Click or hover over a session group to display its values here.</div>
    </template>

    <style>
      :host {
        display: block;
      }
    </style>
  `;
    __decorate([
        property({
            type: Object,
        }),
        __metadata("design:type", Object)
    ], TfHparamsSessionGroupValues.prototype, "sessionGroup", void 0);
    __decorate([
        property({
            type: Object,
        }),
        __metadata("design:type", Object)
    ], TfHparamsSessionGroupValues.prototype, "visibleSchema", void 0);
    TfHparamsSessionGroupValues = __decorate([
        customElement('tf-hparams-session-group-values')
    ], TfHparamsSessionGroupValues);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    // Finds the "closest" path to a given 'target' point from a given array
    // of 'paths'.
    // Each element in 'paths' represents a path and should be an object with
    // a 'controlPoints' property containing a d-element array of the form
    // [[x1,y1],...,[x_d,y_d]] containing the path's control points
    // in ascending 'x' coordinate order, where d is the number of axes.
    // 'axesPos' should be a d-element array containng the x coordinate of
    // each axis.
    // 'target' should be a 2-d array of the form [x, y] defining the target
    // point.
    // 'threshold' denotes a threshold where if the closest path has distance
    // (see below) of more than 'threshold', the return value is null.
    //
    // We define "Closest path" with respect to the following definition of
    // distance. The distance from a point 'target' to a path 'p' is
    // the minimum Euclidean distance from 'target' to a point in the
    // straight-line segment connecting the two control points of 'p'
    // "associated" with 'target'. The control points associated with target
    // are those corresponding to the 2 axes closest (under the usual
    // Euclidean meaning) to 'target' (ties are broken by preferring axes
    // with smaller coordinates).
    //
    // The return value of this method is a closest member of 'paths' to
    // 'target' if the closest distance is at most 'threshold' or null
    // otherwise.
    function findClosestPath(paths, axesPos, target, threshold) {
        if (axesPos.length < 2) {
            console.error('Less than two axes in parallel coordinates plot.');
            return null;
        }
        const cx = target[0];
        const cy = target[1];
        if (cx <= axesPos[0] || cx >= axesPos[axesPos.length - 1]) {
            return null;
        }
        // Find the indices a, b of the two axes closest to 'target'.
        const b = lodash.sortedIndex(axesPos, cx);
        console.assert(b > 0);
        console.assert(b < axesPos.length);
        const a = b - 1;
        // Computes the square of the minimum Euclidean distance from
        // (cx, cy) to a point on the line segment connecting
        // (ax,ay) and (bx,by).
        //
        // Method: Let A=(ax,ay), B=(bx,by) and C=(cx, cy):
        // A parametric from of the line passing through A and B is:
        // B+t(A-B); t real.
        // The line segment is the part with 0<=t<=1.
        // Let 'tp' be the 't' corresponding to the projection of C onto the
        // line. Clearly, 'tp' satisfies:
        // <C-(B+tp(A-B)),A-B> = 0, where <> denotes the dot product. Solving
        // for tp, one gets: tp = <A-B,C-B> / <A-B,A-B>.
        // Since the point on the line segment closest to C is the point on
        // the segment closest to the projection of C onto the line, we get:
        // if tp<=0, the closest point on the line segment is B.
        // if tp>=1, the closest point on the segment is A.
        // Otherwise, the closest point is B+tp(A-B).
        function distFn(ax, ay, bx, by) {
            const abx = ax - bx;
            const aby = ay - by;
            const cbx = cx - bx;
            const cby = cy - by;
            const tp = (abx * cbx + aby * cby) / (abx * abx + aby * aby);
            if (tp <= 0) {
                return l2NormSquared(cbx, cby);
            }
            if (tp >= 1) {
                const cax = ax - cx;
                const cay = ay - cy;
                return l2NormSquared(cax, cay);
            }
            return l2NormSquared(cbx - tp * abx, cby - tp * aby);
        }
        let minDist = null;
        let closestPath = null;
        paths.forEach((p) => {
            const dist = distFn(p.controlPoints[a][0], p.controlPoints[a][1], p.controlPoints[b][0], p.controlPoints[b][1]);
            if (dist > threshold) {
                return;
            }
            if (minDist === null || dist < minDist) {
                minDist = dist;
                closestPath = p;
            }
        });
        return closestPath;
    }
    // Computes the inverse image (in the mathematical sense) of the
    // real-line interval [a,b] under the given point scale mapping.
    // Returns the array consisting of all elements x such that scale(x)
    // is in the real-line interval [a,b].
    function pointScaleInverseImage(scale, a, b) {
        return scale.domain().filter((x) => {
            const y = scale(x);
            return a <= y && y <= b;
        });
    }
    // Computes the inverse image (in the mathematical sense) of the
    // real-line interval [a,b] under the given quantile scale mapping;
    // precisely: {x : scale(x) in [a,b]}
    // Note that for a D3 quantile scale this set is a real-line half-open
    // interval of the form [c,d).
    // This function returns that interval as a 2-element array [c, d].
    function quantileScaleInverseImage(scale, a, b) {
        const range = scale.range();
        const domains = range
            .filter((y) => a <= y && y <= b)
            .map((y) => {
            const domain = scale.invertExtent(y);
            // Find the half open interval of real numbers mapping to y.
            // This is typically returned by scale.invertExtent(y).
            // However, if 'y' is the last value in the range, that
            // interval has the form [d,+infinity), whereas the upper
            // bound returned by scale.invertExtent is the largest element
            // in scale.domain(). Since we return a half-open interval, if
            // we use that value, we will drop session groups whose
            // column value exactly equals it. So we test for this special
            // case here and adjust domain[1] to compensate.
            return y === range[range.length - 1]
                ? [domain[0], domain[1] + 1]
                : domain;
        });
        if (domains.length == 0) {
            return [0, 0]; // Return an empty interval [0,0).
        }
        // Since our source set is a contiguous interval [a,b], the union of
        // domains is a single half-open interval.
        return extent(merge(domains));
    }
    // Computes the inverse image (in the mathematical sense) of the
    // real-line interval [a,b] under the given continuous scale mapping;
    // precisely: {x : scale(x) in [a,b]}
    // Note that for a D3 continuous scale this set is a real-line closed
    // interval. This function returns that interval as a 2-element array.
    function continuousScaleInverseImage(scale, a, b) {
        // D3 continuous scales are monotonic continuous functions; hence to
        // get the inverse image interval we just need to invert the
        // end-points of the source interval. We sort the resulting end-points,
        // to handle the case where the scale is decreasing.
        return [scale.invert(a), scale.invert(b)].sort((x, y) => x - y);
    }
    // Creates the d3-scale to use for the axis with given domain values
    // and height and scale-type. This function may permute the given
    // 'domainValues' array.
    function createAxisScale(domainValues, axisHeight, scaleType) {
        function computeNumericExtent() {
            if (domainValues.length === 0) {
                // If there are no values, there won't be any session groups.
                // We choose an arbitrary numeric domain that doesn't contain 0
                // (to allow a log scale).
                return [1, 2];
            }
            const [min, max] = extent(domainValues);
            if (min !== max) {
                return [min, max];
            }
            // We shift the min and max of the extent here since
            // d3.scaleLinear() and d3.scaleLog() both expect the domain
            // to be an interval with more than one point. Since in this case
            // all session groups would pass through a single point in the axis,
            // the axis resolution doesn't matter much so the amount we shift by
            // doesn't matter much as well. Since for log scales we don't allow
            // the domain to contain 0, we need to be careful not to make the
            // domain contain 0 if it hadn't before.
            if (min > 0) {
                return [min * 0.5, min * 1.5];
            }
            if (min < 0) {
                return [min * 1.5, min * 0.5];
            }
            return [-1, 1];
        }
        if (scaleType === 'LINEAR') {
            return linear$2()
                .domain(computeNumericExtent())
                .range([axisHeight, 0]);
        }
        else if (scaleType === 'LOG') {
            const extent = computeNumericExtent();
            if (extent[0] <= 0 && extent[1] >= 0) {
                // We can't have a log scale for data whose extent contains 0.
                // Use a linear scale instead.
                // TODO(erez): Create a symlog scale similar to Matplotlib's
                // symlog. See also d3 issue here:
                // https://github.com/d3/d3-scale/issues/105
                // and b/111755540
                return createAxisScale(domainValues, axisHeight, 'LINEAR');
            }
            return log$1()
                .domain(extent)
                .range([axisHeight, 0]);
        }
        else if (scaleType === 'QUANTILE') {
            // Compute kNumQuantiles quantiles.
            const kNumQuantiles = 20;
            // Compute the scale's range to be the array:
            // [axisHeight,
            //  axisHeight-1*axisHeight/(kNumQuantiles-1),
            //  axisHeight-2*axisHeight/(kNumQuantiles-1), ...,
            //  0].
            // Unfortunatley,
            // d3.range(axisHeight, -axisHeight/(kNumQuantiles-1),
            //          -axisHeight/(kNumQuantiles-1))
            // has numerical issues and may produce an extra member, so we use a
            // different procedure:
            const scaleRange = sequence(kNumQuantiles)
                .map((i) => axisHeight - (i * axisHeight) / (kNumQuantiles - 1));
            if (domainValues.length === 0) {
                // No session groups in this case. We use a dummy value since
                // d3.scaleQuantile() requires a non-empty domain.
                domainValues = [1];
            }
            return quantile().domain(lodash.uniq(domainValues)).range(scaleRange);
        }
        else if (scaleType === 'NON_NUMERIC') {
            return point$1()
                .domain(
            // We sort the domain values to make the order
            // stable across 'ListSessionGroups' RPCs
            lodash.uniq(domainValues.sort()))
                .range([axisHeight, 0])
                .padding(0.1);
        }
        else
            throw RangeError('Unknown scale: ' + scaleType);
    }

    /* Copyright 2019 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the 'License');
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an 'AS IS' BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    var ScaleType;
    (function (ScaleType) {
        ScaleType["LINEAR"] = "LINEAR";
        ScaleType["LOG"] = "LOG";
        ScaleType["QUANTILE"] = "QUANTILE";
        ScaleType["NON_NUMERIC"] = "NON_NUMERIC";
    })(ScaleType || (ScaleType = {}));
    /**
     * An AlwaysPassingBrushFilter returns 'true' for any value. It is used
     * to represent the case when an Axis does not have an active brush selection.
     */
    class AlwaysPassingBrushFilter {
        isPassing(value) {
            return true;
        }
    }
    /**
     * An IntervalBrushFilter returns 'true' if the given (numeric) value lies
     * in a given interval specified on construction . It's used to represent
     * brush filters for Axis with linear, logarithmic or quantile scales.
     */
    class IntervalBrushFilter {
        /** Constructs the filter. The interval used is defined by lower, and upper.
         * If lowerOpen (resp. upperOpen) is true, the interval will be open in
         * its lower (resp. upper) end, otherwise it will be closed.
         */
        constructor(lower, upper, lowerOpen, upperOpen) {
            this._lower = lower;
            this._upper = upper;
            this._lowerOpen = lowerOpen;
            this._upperOpen = upperOpen;
        }
        isPassing(value) {
            const numValue = value;
            return (this._before(this._lower, numValue, !this._lowerOpen) &&
                this._before(numValue, this._upper, !this._upperOpen));
        }
        _before(a, b, inclusive) {
            if (inclusive) {
                return a <= b;
            }
            return a < b;
        }
    }
    /**
     * A SetBrushFilter returns 'true' if the value is in a given set specified
     * in construction.
     */
    class SetBrushFilter {
        constructor(domainSet) {
            this._domainSet = domainSet;
        }
        isPassing(value) {
            return this._domainSet.findIndex((element) => element === value) !== -1;
        }
    }
    /**
     * Represents a single Axis. An axis does not know its horizontal location in
     * the SVG; instead the axes locations are managed by the AxesCollection class.
     * An axis represents a single column (metric or haparam). It stores a scale
     * type and a D3 scale that maps values in the axis domain (column values)
     * to y-coordinates in the SVG. Additionally, an axis stores a
     * D3-brush-selection which is a 2-element numeric array of the form
     * [lower, upper] containing the upper and lower y-coordinates of the current
     * brush selection. If no brush selection exists, the brush selection stored is
     * null.
     * Finally, an axis can be visible (displayed) or invisible (which will be
     * set based on the user's settings for the corresponding column). An invisible
     * axis need not have its scale or scale-type populated.
     */
    class Axis {
        /**
         * Constructs an axis representing the column indexed by 'colIndex' with
         * respect to 'schema'. Needs an InteractionManager instance so that it can
         * call its event handlers upon receiving events from the DOM.
         */
        constructor(svgProps, schema, interactionManager, colIndex) {
            this._svgProps = svgProps;
            this._schema = schema;
            this._interactionManager = interactionManager;
            this._colIndex = colIndex;
            this._isDisplayed = false;
            this._yScale = null;
            this._scaleType = null;
            this.setBrushSelection(null);
        }
        colIndex() {
            return this._colIndex;
        }
        yScale() {
            return this._yScale;
        }
        scaleType() {
            return this._scaleType;
        }
        brushSelection() {
            return this._brushSelection;
        }
        isDisplayed() {
            return this._isDisplayed;
        }
        setBrushSelection(brushSelection) {
            this._brushSelection = brushSelection;
            this._brushFilter = this._buildBrushFilter(this.brushSelection(), this.scaleType(), this.yScale());
        }
        /**
         * Sets the domain and scale type for the axis. The current brush selection
         * is preserved.
         */
        setDomainAndScale(domainValues, scaleType) {
            this._scaleType = scaleType;
            this._yScale = createAxisScale(
            // Pass a copy since createAxisScale may permute the domainValues array.
            domainValues.slice(), this._svgProps.height, this.scaleType());
            // TODO(erez): Try to modify the brush selection so that it selects
            // the same subset of the axis domain which was selected before
            // this method was called.
            // This can't always be done (e.g. if we switched from a linear to a
            // quantile scale, or if the domain values changed significantly) but in
            // the cases when it is possible, it will be more convenient to the user.
            // Currently, we keep the same brush selection and recalculate the filter.
            // Note that this function will be called every time data is reloaded
            // (e.g. every 30 seconds by default in Tensorboard), so we have to make
            // sure not to change the selection if the data hasn't changed, as that
            // would be very annoying to the end user.
            this._brushFilter = this._buildBrushFilter(this.brushSelection(), this.scaleType(), this.yScale());
        }
        brushFilter() {
            return this._brushFilter;
        }
        /**
         * Renders the axis as child elements of 'axisParent'. Removes any preexisting
         * children of axisParent. 'axisParent' is expected to be a <g> element.
         */
        updateDOM(axisParent /* HTMLElement */) {
            let d3Axis = axisLeft(this.yScale());
            if (this.scaleType() === ScaleType.QUANTILE) {
                // The default tickValues of a quantile scale is just the
                // scale domain, which produces overlapping labels if the
                // number of elements in the domain is greater than the
                // number of quantiles (since then the scale maps more than
                // one domain value to the same quantile).
                d3Axis = d3Axis
                    .tickValues(this.yScale().quantiles())
                    .tickFormat(format('-.6g'));
            }
            const axisParentSel = select(axisParent);
            axisParentSel.selectAll('g').remove();
            axisParentSel
                .append('g')
                .classed('axis', true)
                .call(d3Axis)
                // Add axis title.
                .append('text')
                .classed('axis-title', true)
                .style('cursor', 'move')
                .style('text-anchor', 'middle')
                .attr('y', -9)
                .text((colIndex) => schemaColumnName(this._schema, colIndex));
            // Add dragging event handlers.
            axisParentSel.call(drag()
                .on('start', () => {
                // We set an attribute on the axis that signals
                // that it is being dragged. This allows
                // integration tests to know when dragging is done.
                axisParent.setAttribute('is-dragging', '');
                this._interactionManager.onDragStart(this.colIndex());
            })
                .on('drag', () => this._interactionManager.onDrag(event.x))
                .on('end', () => {
                this._interactionManager.onDragEnd();
                axisParent.removeAttribute('is-dragging');
            }));
            // Add the brush.
            const d3Brush = brushY()
                .extent([
                [-8, 0],
                [8, this._svgProps.height + 1],
            ])
                /* Define the brush event handlers. D3 will call these both when
                       the user moves the brush selection and when we change the brush
                       selection programmatically using d3Brush.move(). We'd like to
                       avoid calling the interactionManager in the latter case; thus,
                       we call _isInteractiveD3Event() to find out if the event was fired
                       due to a programmetic change of the brush selection , and if so,
                       ignore the event. */
                .on('start', () => {
                if (!_isInteractiveD3Event(event)) {
                    return;
                }
                // We set the 'is-brushing' attribute on the containing
                // 'axis-parent'-classed <g> element to notify integration tests
                // that the axis is busy brushing.
                axisParent.setAttribute('is-brushing', '');
                this._interactionManager.onBrushChanged(this.colIndex(), event.selection);
            })
                .on('brush', () => {
                if (!_isInteractiveD3Event(event)) {
                    return;
                }
                this._interactionManager.onBrushChanged(this.colIndex(), event.selection);
            })
                .on('end', () => {
                if (!_isInteractiveD3Event(event)) {
                    return;
                }
                this._interactionManager.onBrushChanged(this.colIndex(), event.selection);
                axisParent.removeAttribute('is-brushing');
            });
            const brushG = select(axisParent)
                .append('g')
                .classed('brush', true);
            brushG.call(d3Brush);
            // Set the brush selection programmatically.
            // We need to cast brushG to 'any' here since TypeScript doesn't realize
            // the brushG is a <g> selection and complains.
            d3Brush.move(brushG, this.brushSelection());
        }
        setDisplayed(value) {
            this._isDisplayed = value;
        }
        /**
         * @return the brush filter for the given selection using the current
         * scale.
         */
        _buildBrushFilter(brushSelection, scaleType, yScale /* D3 scale */) {
            if (brushSelection === null) {
                return new AlwaysPassingBrushFilter();
            }
            if (scaleType === null) {
                console.error("Scale type is null, but brushSelection isn't: ", brushSelection);
                return new AlwaysPassingBrushFilter();
            }
            switch (scaleType) {
                case ScaleType.LINEAR:
                case ScaleType.LOG: {
                    /* Fall Through */
                    const [lower, upper,] = continuousScaleInverseImage(yScale, brushSelection[0], brushSelection[1]);
                    return new IntervalBrushFilter(lower, upper, 
                    /*lowerOpen=*/ false, 
                    /*upperOpen=*/ false);
                }
                case ScaleType.QUANTILE: {
                    const [lower, upper,] = quantileScaleInverseImage(yScale, brushSelection[0], brushSelection[1]);
                    return new IntervalBrushFilter(lower, upper, 
                    /*lowerOpen=*/ false, 
                    /*upperOpen=*/ true);
                }
                case ScaleType.NON_NUMERIC:
                    return new SetBrushFilter(pointScaleInverseImage(yScale, brushSelection[0], brushSelection[1]));
            }
            console.error('Unknown scale type: ', scaleType);
            return new AlwaysPassingBrushFilter();
        }
    }
    /**
     * Manages the collection of axes shown in the plot. Has methods that handle
     * dragging an axis and contains the logic for re-ordering the axes
     * during dragging.
     */
    class AxesCollection {
        constructor(svgProps, schema, interactionManager) {
            this._svgProps = svgProps;
            this._schema = schema;
            this._axes = this._createAxes(interactionManager);
            this._stationaryAxesPositions = point$1()
                .range([1, this._svgProps.width - 1])
                .padding(0.5);
            this._draggedAxis = null;
            this._svgProps.svgG.selectAll('g.axis-parent').remove();
            this._parentsSel = this._svgProps.svgG.selectAll('.axis-parent');
        }
        /**
         * Updates all axes based on the given 'options' (see the comments in
         * tf-hparams-scale-and-color-controls.html) and sessionGroups. SessionGroups
         * are used to update the domain (and thus scale) of the axes. The 'options'
         * object control which axes are visible.
         */
        updateAxes(options, sessionGroups) {
            console.assert(!this.isAxisDragging());
            // Traverse options.columns, and update each corresponding axis.
            const visibleColIndices = new Set();
            options.columns.forEach((column) => {
                const colIndex = column.absoluteIndex;
                let axis = this._axes[colIndex];
                axis.setDisplayed(true);
                const domainValues = sessionGroups.map((sg) => columnValueByIndex(this._schema, sg, colIndex));
                axis.setDomainAndScale(domainValues, column.scale);
                visibleColIndices.add(colIndex);
            });
            // Set the visibility of the remaining axes to false.
            this._axes.forEach((axis) => {
                if (!visibleColIndices.has(axis.colIndex())) {
                    axis.setDisplayed(false);
                }
            });
            this._updateStationaryAxesPositions(visibleColIndices);
            // Update the DOM.
            this._parentsSel = this._parentsSel.data(Array.from(visibleColIndices), 
            /*key=*/ (colIndex) => colIndex);
            this._parentsSel.exit().remove();
            this._parentsSel = this._parentsSel
                .enter()
                .append('g')
                .classed('axis-parent', true)
                .merge(this._parentsSel);
            const _this = this;
            this._parentsSel
                .call((sel) => this._updateAxesPositionsInDOM(sel))
                .each(function (colIndex) {
                /* Here 'this' is the 'axis-parent'-classed <g> element,
                     and '_this' is the AxesCollection element. */
                _this._axes[colIndex].updateDOM(this);
            });
        }
        /**
         * Executes mapFunction on each visible axis. Returns an array containing the
         * result from each invocation. The function is invoked on the axes ordered
         * by their increasing xPosition.
         */
        mapVisibleAxes(mapFunction) {
            return this._stationaryAxesPositions
                .domain()
                .map((colIndex) => mapFunction(this.getAxisPosition(colIndex), this._axes[colIndex]));
        }
        /**
         * @return true if the given predicate returns true on every visible axis,
         *     false otherwise. Note that the predicate will only be evaluated until
         *     the first time it returns false.
         */
        allVisibleAxesSatisfy(predicate) {
            return this._stationaryAxesPositions
                .domain()
                .every((colIndex) => predicate(this.getAxisPosition(colIndex), this._axes[colIndex]));
        }
        getAxisForColIndex(colIndex) {
            return this._axes[colIndex];
        }
        /* Axis dragging.
         * To drag an axis, call: dragStart(), followed by one or more drag() calls
         * followed by a single call to dragEnd().
         * At most one axis can be dragged at any given time.
         * Each axis (whether dragged or not) has an associated "stationary"
         * position which is its (x-coordinate) position when it is not being dragged.
         * The actual position of an axis is either its associated stationary
         * position if its not dragged or its currently dragged position. This class
         * maintains the invariant that the axes' stationary positions match the order
         * of their actual position by re-assigning stationary positions to axes when
         * dragging an axis causes it to "pass" another axes.
         */
        dragStart(colIndex) {
            console.assert(!this.isAxisDragging());
            console.assert(this._axes[colIndex].isDisplayed());
            this._draggedAxis = this._axes[colIndex];
            this._draggedAxisPosition = this._stationaryAxesPositions(colIndex);
        }
        drag(newX) {
            newX = Math.min(Math.max(newX, 0), this._svgProps.width);
            this._draggedAxisPosition = newX;
            let visibleColIndices = this._stationaryAxesPositions.domain();
            visibleColIndices.sort((ci1, ci2) => this.getAxisPosition(ci1) - this.getAxisPosition(ci2));
            this._stationaryAxesPositions.domain(visibleColIndices);
            this._updateAxesPositionsInDOM(this._parentsSel);
        }
        dragEnd(duration) {
            console.assert(this.isAxisDragging());
            this._draggedAxisPosition = null;
            this._draggedAxis = null;
            this._updateAxesPositionsInDOM(this._parentsSel.transition().duration(duration));
        }
        isAxisDragging() {
            return this._draggedAxis !== null;
        }
        getAxisPosition(colIndex) {
            return this._draggedAxis !== null &&
                this._draggedAxis.colIndex() === colIndex
                ? this._draggedAxisPosition
                : this._stationaryAxesPositions(colIndex);
        }
        /**
         * Sets the domain of 'stationaryAxesPositions' to be precisely the given
         * visibleColIndices, but preserves the order of the column indices that
         * are already in the domain. Essentially, this method removes indices in
         * the domain that are not in visibleColIndices and then appends indices in
         * visibleColIndices that are not currently in the domain. Thus, indices in
         * visibleColIndices that are already in stationaryAxesPositions
         * will maintain their order in stationaryAxesPositions and will precede
         * the new elements.
         *
         * This reassigns stationary positions to axes so that the only visible
         * axes are the ones with column indices in 'visibleColIndices', but preserves
         * the order of axes indexed by visibleColIndices that are already visible.
         */
        _updateStationaryAxesPositions(visibleColIndices) {
            const visibleDomain = this._stationaryAxesPositions
                .domain()
                .filter((colIndex) => visibleColIndices.has(colIndex));
            const newDomain = Array.from(new Set([...visibleDomain, ...Array.from(visibleColIndices)]));
            this._stationaryAxesPositions.domain(newDomain);
        }
        _updateAxesPositionsInDOM(selectionOrTransition) {
            selectionOrTransition.attr('transform', (colIndex) => translateStr(this.getAxisPosition(colIndex)));
        }
        _createAxes(interactionManager) {
            return sequence(numColumns(this._schema))
                .map((colIndex) => new Axis(this._svgProps, this._schema, interactionManager, colIndex));
        }
    }
    function _isInteractiveD3Event(d3Event) {
        return d3Event.sourceEvent !== null;
    }

    /* Copyright 2019 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the 'License');
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an 'AS IS' BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    var LineType;
    (function (LineType) {
        LineType[LineType["FOREGROUND"] = 0] = "FOREGROUND";
        LineType[LineType["BACKGROUND"] = 1] = "BACKGROUND";
    })(LineType || (LineType = {}));
    /**
     * A handle to a representation of a session group in the 'LinesCollection'
     * class below. The handle can also be "null" -- meaning it references no
     * session group (similar to a "null pointer"), in which case the
     * 'sessionGroup()' method returns null.
     *
     * Note: we use this class rather than a simple SessionGroup object so that we
     * won't need to search the DOM for the <path> representing the session group;
     * instead the (foreground) <path> element is stored in the handle.
     */
    class SessionGroupHandle {
        /**
         * Constructs a session group handle from a D3 selection of the path
         * element representing the sessionGroup. This should only be called by the
         * 'LinesCollection' class below. If sessionGroupSel is empty or undefined, a
         * "null" handle will be constructed.
         */
        constructor(sessionGroupSel) {
            if (sessionGroupSel === undefined) {
                sessionGroupSel = selectAll(null);
            }
            console.assert(sessionGroupSel.size() <= 1);
            this._sessionGroupSel = sessionGroupSel;
        }
        /**
         * @return the sessionGroup object this handle references or null if
         * this is a "null" reference.
         */
        sessionGroup() {
            return this._sessionGroupSel.size() === 1
                ? this._sessionGroupSel.datum()
                : null;
        }
        isNull() {
            return this.sessionGroup() === null;
        }
        /**
         * Should only be called by the 'LinesCollection' class below.
         * @return the d3-selection given on construction.
         */
        selection() {
            return this._sessionGroupSel;
        }
        /**
         * Compares this handle to 'otherHandle' and returns true if both are null
         * or both are not null and they reference equal session groups.
         * Session group equality is determined by their names.
         */
        equalsTo(otherHandle) {
            if (this.isNull()) {
                return otherHandle.isNull();
            }
            if (otherHandle.isNull()) {
                return false;
            }
            return otherHandle.sessionGroup().name == this.sessionGroup().name;
        }
    }
    /**
     * Manages the lines representing the session groups. Each session group is
     * represented by two <path> elements with the same control points: a foreground
     * and a background line, with the foreground line always rendered after the
     * background line (it succeeds it in document order). The foreground line is
     * colored based on the value  of the 'color-by' column (specified in options),
     * whereas the background line is gray. The foreground line is displayed only
     * if the session group passes the current axes brush filters. This way, only
     * session groups that pass the current brush filters will be represented by
     * colored lines, and the other session groups will be represented by gray
     * lines.
     *
     * Note that we could have represented each session group with a single line
     * with a color that depends on whether the session group passed the brush
     * filters. However, this would have required us to re-order the DOM to avoid
     * rendering gray lines on top of colored lines (which would look bad,
     * especially if the density of lines is large).
     *
     * This class also stores two SessionGroupHandles: peakedSessionGroupHandle and
     * selectedSessionGroupHandle, and provides methods for updating them. These
     * are used to store the peaked session group: the session group whose line
     * is closest to the mouse point. The peaked session group can be selected
     * by clicking the mouse pointer. The currently selected session group is
     * referenced by the selectedSessionGroupHandle.
     */
    class LinesCollection {
        constructor(svgProps, schema, axesCollection) {
            this._svgProps = svgProps;
            this._schema = schema;
            this._axesCollection = axesCollection;
            this._sessionGroups = [];
            this._svgProps.svgG.selectAll('g.background').remove();
            this._svgProps.svgG.selectAll('g.foreground').remove();
            this._bgPathsSel = this._svgProps.svgG
                .append('g')
                .classed('background', true)
                .selectAll('path');
            this._fgPathsSel = this._svgProps.svgG
                .append('g')
                .classed('foreground', true)
                .selectAll('path');
            this._updateVisibleFgPathsSel();
            this._peakedSessionGroupHandle = new SessionGroupHandle();
            this._selectedSessionGroupHandle = new SessionGroupHandle();
            this._d3line = line().curve(curveLinear);
        }
        /**
         * @return a SessionGroupHandle referencing the given sessionGroup. If the
         * given sessionGroup is null or undefined returns a "null" handle.
         */
        getSessionGroupHandle(sessionGroup) {
            if (sessionGroup === null || sessionGroup === undefined) {
                return new SessionGroupHandle();
            }
            return new SessionGroupHandle(this._fgPathsSel.filter((sg) => sg.name === sessionGroup.name));
        }
        hideBackgroundLines() {
            this._bgPathsSel.attr('visibility', 'hidden');
        }
        showBackgroundLines() {
            this._bgPathsSel.attr('visibility', null);
        }
        peakedSessionGroupHandle() {
            return this._peakedSessionGroupHandle;
        }
        selectedSessionGroupHandle() {
            return this._selectedSessionGroupHandle;
        }
        /**
         * Recomputes the control points of the lines with the given 'type'
         * (foreground or background) to correspond to the current state of the
         * axesCollection.
         *
         * @param lineType - The type of lines to update.
         * @param transitionDuration - The lines will be transitioned (animated) to
         *     their new state. This specifies the duration of that transition. 0
         *     means no animation.
         */
        recomputeControlPoints(lineType, transitionDuration = 0) {
            const pathSel = lineType === LineType.FOREGROUND ? this._fgPathsSel : this._bgPathsSel;
            pathSel
                .transition()
                .duration(transitionDuration)
                .attr('d', (sessionGroup) => this._pathDAttribute(sessionGroup));
            if (lineType === LineType.FOREGROUND) {
                // Update the control points property, if we're updating the foreground
                // lines.
                window.setTimeout(() => {
                    const _this = this;
                    this._fgPathsSel.each(function (sessionGroup) {
                        // '_this' refers to the 'LinesCollection' instance.
                        _this._setControlPointsProperty(this, sessionGroup);
                    });
                });
            }
        }
        /**
         * Rerenders the foreground lines so that their visibility matches the
         * current brush filters.
         */
        recomputeForegroundLinesVisibility() {
            this._fgPathsSel.classed('invisible-path', (sessionGroup) => !this._axesCollection.allVisibleAxesSatisfy((xPosition, axis) => axis
                .brushFilter()
                .isPassing(columnValueByIndex(this._schema, sessionGroup, axis.colIndex()))));
            this._updateVisibleFgPathsSel();
        }
        /**
         * Sets the coloring scheme of the (visible) foreground lines.
         * A foreground line will be colored by a color corresponding to the value
         * of the column indexed by colorByColumnIndex in the session group
         * represented by the line. The color corresponding to a value is
         * interpolated between minColor and maxColor.
         */
        setForegroundLinesColor(colorByColumnIndex, minColor, maxColor) {
            const lineColorFunction = this._createLineColorFunction(colorByColumnIndex, minColor, maxColor);
            this._fgPathsSel.attr('stroke', lineColorFunction);
        }
        /**
         * Updates the sessionGroups, colorByColumnIndex, minColor and maxColor and
         * redraws the lines.
         */
        redraw(sessionGroups, colorByColumnIndex, minColor, maxColor) {
            const peakedSG = this._peakedSessionGroupHandle.sessionGroup();
            const selectedSG = this._selectedSessionGroupHandle.sessionGroup();
            this._sessionGroups = sessionGroups;
            this._fgPathsSel = this._recomputePathSelection(this._fgPathsSel);
            this._bgPathsSel = this._recomputePathSelection(this._bgPathsSel);
            this._peakedSessionGroupHandle = this.getSessionGroupHandle(peakedSG);
            this._selectedSessionGroupHandle = this.getSessionGroupHandle(selectedSG);
            this.recomputeControlPoints(LineType.FOREGROUND);
            this.recomputeControlPoints(LineType.BACKGROUND);
            this.recomputeForegroundLinesVisibility();
            this.setForegroundLinesColor(colorByColumnIndex, minColor, maxColor);
        }
        updatePeakedSessionGroup(newHandle) {
            this._peakedSessionGroupHandle.selection().classed('peaked-path', false);
            this._peakedSessionGroupHandle = newHandle;
            this._peakedSessionGroupHandle.selection().classed('peaked-path', true);
        }
        clearPeakedSessionGroup() {
            this.updatePeakedSessionGroup(new SessionGroupHandle());
        }
        updateSelectedSessionGroup(newHandle) {
            this._selectedSessionGroupHandle
                .selection()
                .classed('selected-path', false);
            this._selectedSessionGroupHandle = newHandle;
            this._selectedSessionGroupHandle.selection().classed('selected-path', true);
        }
        /**
         * Returns a handle referencing the closest session group to a given point
         * in the SVG or a null handle if the closest session group is
         * too far away.
         */
        findClosestSessionGroup(x, y) {
            const axesPositions = this._axesCollection.mapVisibleAxes((xPosition, axis) => xPosition);
            const closestFgPath = findClosestPath(this._visibleFgPathsSel.nodes(), axesPositions, [x, y], 
            /* threshold */ 100);
            if (closestFgPath === null) {
                return new SessionGroupHandle();
            }
            return new SessionGroupHandle(select(closestFgPath));
        }
        _createLineColorFunction(colorByColumnIndex, minColor, maxColor) {
            if (colorByColumnIndex === null) {
                /* Use a default color if no color-by column is selected. */
                return () => 'red';
            }
            const colorScale = linear$2()
                .domain(numericColumnExtent(this._schema, this._sessionGroups, colorByColumnIndex))
                .range([minColor, maxColor])
                .interpolate(lab$1);
            return (sessionGroup) => colorScale(columnValueByIndex(this._schema, sessionGroup, colorByColumnIndex));
        }
        _recomputePathSelection(currentPathSel /* d3 selection */) {
            currentPathSel = currentPathSel.data(this._sessionGroups, 
            /*key=*/ (sessionGroup) => sessionGroup.name);
            currentPathSel.exit().remove();
            return currentPathSel.enter().append('path').merge(currentPathSel);
        }
        /** Sets the controlPoints property of 'pathElement' to the control-points
         * array of the given sessionGroup with respect to the current state of
         * the axesCollection.
         */
        _setControlPointsProperty(pathElement, sessionGroup) {
            pathElement.controlPoints = this._computeControlPoints(sessionGroup);
        }
        /**
         * @return an array of 2-tuples--each representing a control point for
         * a line representing the given 'sessionGroup'. The control points are
         * computed with respect to the current state of the axesCollection.
         */
        _computeControlPoints(sessionGroup) {
            return this._axesCollection.mapVisibleAxes((xPosition, axis) => [
                xPosition,
                axis.yScale()(columnValueByIndex(this._schema, sessionGroup, axis.colIndex())),
            ]);
        }
        _pathDAttribute(sessionGroup) {
            return this._d3line(this._computeControlPoints(sessionGroup));
        }
        _updateVisibleFgPathsSel() {
            this._visibleFgPathsSel = this._fgPathsSel.filter(':not(.invisible-path)');
        }
    }

    /* Copyright 2019 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the 'License');
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an 'AS IS' BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * Stores some global properties such as width and height of the SVG element
     * used for rendering the parallel coordinates plot. Also contains the top-level
     * DOM <g> element underwhich the plot will be rendered.
     */
    class SVGProperties {
        /**
         * Computes the width and height of the SVG element based on the number of
         * columns in the schema. Adds some margins and adds a top-level <g> element
         * underwhich the plot should be rendered.
         */
        constructor(svg, numColumns) {
            // We use the following algorithm for laying out our SVG:
            // We compute a minimum size for the SVG based on the number columns
            // and some margins. We set the svg "width" and "height" styles
            // to "100%" so that it takes up the full area of its parent, but use
            // "min-width" and "min-height", so that if the parent is too small
            // the svg won't shrink down (it will overflow with scroll bars).
            // If the parent is larger than the minimum size, we use its
            // preserveAspectRatio attr to scale the contents to fit the larger size.
            this.svg = select(svg);
            const margin = { top: 30, right: 10, bottom: 10, left: 10 };
            const COL_WIDTH = 100;
            const COL_HEIGHT = 200;
            const totalWidth = numColumns * COL_WIDTH + margin.left + margin.right;
            const totalHeight = COL_HEIGHT + margin.top + margin.bottom;
            this.svg.attr('viewBox', `0 0 ${totalWidth} ${totalHeight}`);
            this.svg.attr('preserveAspectRatio', 'xMidYMid');
            // Set a minimum width so scale factor want be less than 1
            // (but if size of '#container' is larger then we'll scale up
            // our svg).
            this.svg.style('min-width', totalWidth + 'px');
            this.svg.style('min-height', totalHeight + 'px');
            // 'width' and 'height' store the width of the svg without our margins.
            this.width = totalWidth - margin.left - margin.right;
            this.height = totalHeight - margin.top - margin.bottom;
            this.svgG = this.svg
                .append('g')
                .attr('transform', translateStr(margin.left, margin.top));
        }
    }
    /**
     * Orchastrates the behavior of the parallel coordinates plot. This is the
     * class that consumers of this module should use.
     * Usage example:
     *    manager = new InteractionManager(svgProps, schema, peakedSessionChangedCB,
     *                                     selectedSessionChangedCB);
     *    ...
     *    // Notify manager of new options or session groups:
     *    manager.onOptionsOrSessionGroupsChanged(newOptions, newSessionGroups)
     *    ...
     *    // This will be called when peaked session changed:
     *    function peakedSessionChangedCB(newSessionGroup) {
     *      // Do something with the newSessionGroup.
     *    }
     */
    class InteractionManager {
        constructor(svgProps, schema, peakedSessionGroupChangedCallback, selectedSessionChangedCallback) {
            this._svgProps = svgProps;
            this._schema = schema;
            this._peakedSessionGroupChangedCB = peakedSessionGroupChangedCallback;
            this._selectedSessionGroupChangedCB = selectedSessionChangedCallback;
            this._axesCollection = new AxesCollection(svgProps, schema, 
            /*interactionManager=*/ this);
            this._linesCollection = new LinesCollection(svgProps, schema, this._axesCollection);
            this._svgProps.svg
                .on('click', () => this.onClick())
                .on('mousemove mouseenter', () => {
                const [x, y] = mouse(this._svgProps.svgG.node());
                this.onMouseMoved(x, y);
            })
                .on('mouseleave', () => this.onMouseLeave());
        }
        onDragStart(colIndex) {
            this._axesCollection.dragStart(colIndex);
            this._linesCollection.hideBackgroundLines();
        }
        onDrag(newX) {
            this._axesCollection.drag(newX);
            this._linesCollection.recomputeControlPoints(LineType.FOREGROUND);
        }
        onDragEnd() {
            this._axesCollection.dragEnd(/*transitionDuration=*/ 500);
            this._linesCollection.recomputeControlPoints(LineType.FOREGROUND, 
            /* transitionDuration=*/ 500);
            window.setTimeout(() => {
                this._linesCollection.recomputeControlPoints(LineType.BACKGROUND);
                this._linesCollection.showBackgroundLines();
            }, 500);
        }
        onBrushChanged(colIndex, newBrushSelection) {
            this._axesCollection
                .getAxisForColIndex(colIndex)
                .setBrushSelection(newBrushSelection);
            this._linesCollection.recomputeForegroundLinesVisibility();
        }
        onMouseMoved(newX, newY) {
            this._linesCollection.updatePeakedSessionGroup(this._linesCollection.findClosestSessionGroup(newX, newY));
            this._peakedSessionGroupChangedCB(this._linesCollection.peakedSessionGroupHandle().sessionGroup());
        }
        onMouseLeave() {
            if (!this._linesCollection.peakedSessionGroupHandle().isNull()) {
                this._linesCollection.clearPeakedSessionGroup();
                this._peakedSessionGroupChangedCB(null);
            }
        }
        onClick() {
            if (this._linesCollection.peakedSessionGroupHandle().sessionGroup() ===
                this._linesCollection.selectedSessionGroupHandle().sessionGroup()) {
                /* If the selected session group is the same as the "peaked" one,
                       clear the selection. */
                this._linesCollection.updateSelectedSessionGroup(new SessionGroupHandle());
            }
            else {
                this._linesCollection.updateSelectedSessionGroup(this._linesCollection.peakedSessionGroupHandle());
            }
            this._selectedSessionGroupChangedCB(this._linesCollection.selectedSessionGroupHandle().sessionGroup());
        }
        onOptionsOrSessionGroupsChanged(newOptions, newSessionGroups) {
            this._axesCollection.updateAxes(newOptions, newSessionGroups);
            const oldPeakedSessionGroupHandle = this._linesCollection.peakedSessionGroupHandle();
            const oldSelectedSessionGroupHandle = this._linesCollection.selectedSessionGroupHandle();
            this._linesCollection.redraw(newSessionGroups, newOptions.colorByColumnIndex !== undefined
                ? newOptions.columns[newOptions.colorByColumnIndex].absoluteIndex
                : null, newOptions.minColor, newOptions.maxColor);
            // A redraw may change the selected / peaked session group. So call the
            // appropriate callbacks if needed.
            if (!oldPeakedSessionGroupHandle.equalsTo(this._linesCollection.peakedSessionGroupHandle())) {
                this._peakedSessionGroupChangedCB(this._linesCollection.peakedSessionGroupHandle().sessionGroup());
            }
            if (!oldSelectedSessionGroupHandle.equalsTo(this._linesCollection.selectedSessionGroupHandle())) {
                this._selectedSessionGroupChangedCB(this._linesCollection.selectedSessionGroupHandle().sessionGroup());
            }
        }
        schema() {
            return this._schema;
        }
    }

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    let TfHparamsParallelCoordsPlot = class TfHparamsParallelCoordsPlot extends LegacyElementMixin(PolymerElement) {
        constructor() {
            super(...arguments);
            // The last session group that was clicked on or null if no
            // session group was clicked on yet.
            /**
             * @type {?Object}
             */
            this.selectedSessionGroup = null;
            // The session group represented by the curve "closest" to the mouse
            // pointer (the corresponding path element will have the 'peaked-path'
            // class). If the closest session group distance is larger than a
            // threshold, this property will be null.
            /**
             * @type {?Object}
             */
            this.closestSessionGroup = null;
            // Counts the number of times the element has been redrawn since it
            // was created. This is incremented at the end of a redraw and allows
            // integration tests to wait for the element to finish redrawing.
            this.redrawCount = 0;
        }
        _optionsOrSessionGroupsChanged() {
            if (!this.options) {
                return;
            }
            const configuration = this.options.configuration;
            // See if we need to redraw from scratch. We redraw from scratch if
            // this is initialization or if configuration.schema has changed.
            if (this._interactionManager === undefined ||
                !lodash.isEqual(this._interactionManager.schema(), configuration.schema)) {
                // Remove any pre-existing DOM children of our SVG.
                select(this.$.svg)
                    .selectAll('*')
                    .remove();
                const svgProps = new SVGProperties(this.$.svg, numColumns(configuration.schema));
                // Listen to DOM changes underneath this.$.svg, and apply local CSS
                // scoping rules so that our rules in the <style> section above
                // would apply.
                this.scopeSubtree(this.$.svg, true);
                this._interactionManager = new InteractionManager(svgProps, configuration.schema, (sessionGroup) => this.closestSessionGroupChanged(sessionGroup), (sessionGroup) => this.selectedSessionGroupChanged(sessionGroup));
            }
            this._computeValidSessionGroups();
            this._interactionManager.onOptionsOrSessionGroupsChanged(this.options, this._validSessionGroups);
            this.redrawCount++;
        }
        closestSessionGroupChanged(sessionGroup) {
            this.closestSessionGroup = sessionGroup;
        }
        selectedSessionGroupChanged(sessionGroup) {
            this.selectedSessionGroup = sessionGroup;
        }
        // computes validSessionGroups: Filters out the session groups in the
        // sessionGroups that have one or more of their column values undefined.
        // If sessionGroups is undefined sets validSessionGroups to be
        // undefined as well. (This can happen during testing when we don't set
        // the sessionGroups property).
        _computeValidSessionGroups() {
            const utils = tf_hparams_utils;
            if (this.sessionGroups === undefined) {
                this._validSessionGroups = undefined;
                return;
            }
            const schema = this.options.configuration.schema;
            this._validSessionGroups = this.sessionGroups.filter((sg) => {
                for (let colIndex = 0; colIndex < utils.numColumns(schema); ++colIndex) {
                    if (!this.options.configuration.columnsVisibility[colIndex]) {
                        continue;
                    }
                    if (utils.columnValueByIndex(schema, sg, colIndex) === undefined) {
                        return false;
                    }
                }
                return true;
            });
        }
    };
    TfHparamsParallelCoordsPlot.template = html `
    <div id="container">
      <svg id="svg"></svg>
    </div>
    <style>
      :host {
        display: block;
      }
      svg {
        font: 10px sans-serif;
      }

      .background path {
        fill: none;
        stroke: #ddd;
        shape-rendering: crispEdges;
      }

      .foreground path {
        fill: none;
        stroke-opacity: 0.7;
        stroke-width: 1;
      }

      /* Will be set on foreground paths that are not "contained" in the current
         axes brushes. If no brushes are set, no path will have this class. */
      .foreground .invisible-path {
        display: none;
      }

      /* Style for the path closest to the mouse pointer (typically will become
      the selected path when the user clicks). */
      .foreground .peaked-path {
        stroke-width: 3;
      }

      /* The currently selected path class. We use !important to override the
         inline style that sets the regular color of a path. */
      .foreground .selected-path {
        stroke-width: 3 !important;
        stroke: #0f0 !important;
      }

      #container {
        height: 100%;
        width: 100%;
      }

      svg {
        width: 100%;
        height: 100%;
      }

      .axis text {
        text-shadow: 0 1px 0 #fff, 1px 0 0 #fff, 0 -1px 0 #fff, -1px 0 0 #fff;
        fill: #000;
        cursor: move;
      }
    </style>
  `;
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfHparamsParallelCoordsPlot.prototype, "sessionGroups", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfHparamsParallelCoordsPlot.prototype, "options", void 0);
    __decorate([
        property({
            type: Object,
            notify: true,
        }),
        __metadata("design:type", Object)
    ], TfHparamsParallelCoordsPlot.prototype, "selectedSessionGroup", void 0);
    __decorate([
        property({
            type: Object,
            notify: true,
        }),
        __metadata("design:type", Object)
    ], TfHparamsParallelCoordsPlot.prototype, "closestSessionGroup", void 0);
    __decorate([
        property({
            type: Number,
        }),
        __metadata("design:type", Number)
    ], TfHparamsParallelCoordsPlot.prototype, "redrawCount", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfHparamsParallelCoordsPlot.prototype, "_validSessionGroups", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfHparamsParallelCoordsPlot.prototype, "_interactionManager", void 0);
    __decorate([
        observe('options.*', 'sessionGroups.*'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfHparamsParallelCoordsPlot.prototype, "_optionsOrSessionGroupsChanged", null);
    TfHparamsParallelCoordsPlot = __decorate([
        customElement('tf-hparams-parallel-coords-plot')
    ], TfHparamsParallelCoordsPlot);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * There are 3 elements involved in the parallel coordinates visualization:
     *
     * 1. <tf-hparams-parallel-coords-plot>
     *   Renders the actual parallel coordinate plot. See the comments in the
     *   code for that element for more details.
     * 2. <tf-hparams-scale-and-color-controls>
     *   A control panel for configuring the behavior of the plot (e.g. scale
     *   of each axis, colormap, etc.)
     * 3. <tf-hparams-parallel-coords-view>
     *   The container element for the above 2 elements.
     */
    let TfHparamsParallelCoordsView = class TfHparamsParallelCoordsView extends PolymerElement {
        _closestOrSelected(closestSessionGroup, selectedSessionGroup) {
            if (closestSessionGroup !== null) {
                return closestSessionGroup;
            }
            return selectedSessionGroup;
        }
    };
    TfHparamsParallelCoordsView.template = html `
    <!-- Controls behavior of parallel coordinates plot
         outputs set options to the _options property.
      -->
    <hparams-split-layout orientation="vertical">
      <!-- The scale and color controls. -->
      <tf-hparams-scale-and-color-controls
        id="controls"
        slot="content"
        class="section"
        configuration="[[configuration]]"
        session-groups="[[sessionGroups]]"
        options="{{_options}}"
      >
      </tf-hparams-scale-and-color-controls>
      <!-- The actual parallel coordinates plot -->
      <tf-hparams-parallel-coords-plot
        id="plot"
        slot="content"
        class="section"
        session-groups="[[sessionGroups]]"
        selected-session-group="{{_selectedGroup}}"
        closest-session-group="{{_closestGroup}}"
        options="[[_options]]"
      >
      </tf-hparams-parallel-coords-plot>
      <tf-hparams-session-group-values
        id="values"
        slot="content"
        class="section"
        visible-schema="[[configuration.visibleSchema]]"
        session-group="[[_closestOrSelected(
                             _closestGroup, _selectedGroup)]]"
      >
      </tf-hparams-session-group-values>
      <tf-hparams-session-group-details
        id="details"
        slot="content"
        class="section"
        backend="[[backend]]"
        experiment-name="[[experimentName]]"
        session-group="[[_selectedGroup]]"
        visible-schema="[[configuration.visibleSchema]]"
      >
      </tf-hparams-session-group-details>
    </hparams-split-layout>

    <style>
      .section {
        padding: 10px;
      }
      #values {
        height: 115px;
      }
      #details {
        flex-grow: 1;
        max-height: fit-content;
      }
    </style>
  `;
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfHparamsParallelCoordsView.prototype, "backend", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfHparamsParallelCoordsView.prototype, "experimentName", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfHparamsParallelCoordsView.prototype, "configuration", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfHparamsParallelCoordsView.prototype, "sessionGroups", void 0);
    TfHparamsParallelCoordsView = __decorate([
        customElement('tf-hparams-parallel-coords-view')
    ], TfHparamsParallelCoordsView);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * A d3-based scatter plot matrix visualization component.
     * This component renders the actual plots; the "controls" part of the
     * visualization are rendered by tf-hparams-scale-and-color-controls.
     *
     * TODO(erez): The logic for computing the number of ticks so that tick labels
     * do not overlap is ignored, for some reason, when we set compile=True in
     * the vulcanization build step. Figure out why.
     */
    let TfHparamsScatterPlotMatrixPlot = class TfHparamsScatterPlotMatrixPlot extends LegacyElementMixin(PolymerElement) {
        constructor() {
            super(...arguments);
            // The last session group that was clicked on or null if no
            // session group was clicked on yet.
            this.selectedSessionGroup = null;
            // The session group represented by the marker "closest" to the mouse
            // pointer. If the closest session group distance is larger than a
            // threshold, this property will be null.
            this.closestSessionGroup = null;
            // The <div> element with id "container".
            this._container = null;
            // A D3 selection containing just the root <svg> element.
            this._svg = null;
            this.width = 0;
            this.height = 0;
            // The index of the cell containing a brush selection as an array
            // of the form [col, metric] or null if no cell has an active brush
            // selection.
            this._brushedCellIndex = null;
            // The the active brush selection in the form
            // [[x0,y0],[x1,y1]] where the coordinates are relative to the cell
            // indexed by _brushedCellIndex. Set to null, if there is no active
            // brush.
            this._brushSelection = null;
        }
        ready() {
            super.ready();
            this._container = this.$['container'];
            this._svg = select(this.$['svg']);
            this._redraw();
        }
        _sessionGroupsChanged() {
            if (this.selectedSessionGroup !== null) {
                // Try to keep the selected session group: if the new sessionGroups
                // array has a sessionGroup with the same name as the one that was
                // selected before, select it.
                this.selectedSessionGroup =
                    sessionGroupWithName(this.sessionGroups, this.selectedSessionGroup.name) || null;
            }
            this._redraw();
        }
        _visibleSchemaChanged() {
            this._brushedCellIndex = null;
            this._brushSelection = null;
            this._redraw();
        }
        // Redraws the plot.
        _redraw() {
            this.debounce('_redraw', () => {
                const utils = tf_hparams_utils;
                const PLOT_MIN_WIDTH = 1200;
                const PLOT_MIN_HEIGHT = 0.4 * PLOT_MIN_WIDTH;
                const CELL_MIN_WIDTH = 150;
                const CELL_MIN_HEIGHT = 0.75 * CELL_MIN_WIDTH;
                this.width = Math.max(CELL_MIN_WIDTH * utils.numVisibleColumns(this.visibleSchema), PLOT_MIN_WIDTH);
                this.height = Math.max(CELL_MIN_HEIGHT * utils.numVisibleMetrics(this.visibleSchema), PLOT_MIN_HEIGHT);
                this._container.style.width = this.width + 'px';
                this._container.style.height = this.height + 'px';
                this._svg.attr('width', this.width).attr('height', this.height);
                // Delete all elements in the svg subtree
                this._svg.selectAll('g').remove();
                // Draw.
                this._draw();
            }, 100);
        }
        // Creates the DOM elements comprising the scatter-plot-matrix plot and
        // registers event handlers to handle user actions.
        _draw() {
            const utils = tf_hparams_utils;
            const _this = this;
            if (!this.sessionGroups ||
                this.sessionGroups.length == 0 ||
                !this.visibleSchema ||
                this.visibleSchema.metricInfos.length == 0) {
                // If there's no metrics or session groups. There's nothing to draw.
                return;
            }
            // An array containing the visibleSchema-columns (hparams followed by
            // metrics) indices. These index the columns of the scatter plot matrix.
            const cols = sequence(utils.numVisibleColumns(_this.visibleSchema));
            // An array containing the metric indices. These index the rows of the
            // scatter plot matrix.
            const metrics = sequence(utils.numVisibleMetrics(_this.visibleSchema));
            // The margin in pixels from the left to leave for the y-axis text
            // (tick values and x-axis label).
            const yAxisTextMargin = 80;
            // The margin in pixels from the bottom to leave for the x-axis text
            // (tick values and x-axis label).
            const xAxisTextMargin = 50;
            // Each cell in the scatter plot matrix has a rectangular frame.
            // The margin in pixels to use between the cell boundary and its frame.
            const frameMargin = 5;
            // cellX(col), cellY(metric) are the svg coordinates of the upper
            // left corner of the boundary of the cell indexed by (col, metric).
            const cellX = band()
                .domain(cols)
                .range([yAxisTextMargin + frameMargin, this.width - 1 - frameMargin])
                .paddingInner(0.1);
            const cellY = band()
                .domain(metrics)
                .range([this.height - 1 - frameMargin - xAxisTextMargin, frameMargin])
                .paddingInner(0.1);
            const cellWidth = cellX.bandwidth();
            const cellHeight = cellY.bandwidth();
            // xCoords[col](colValue), yCoords[metric](metricValue) are the
            // coordinates of the marker representing the data
            // (colValue, metricValue) in the cell indexed by (col, metric).
            // The coordinates are relative to the cell boundary's upper
            // left corner.
            const xCoords = cols.map((c) => _this._cellScale(c, [0, cellWidth - 1]));
            const yCoords = metrics.map((m) => _this._cellScale(m + utils.numVisibleHParams(_this.visibleSchema), [
                cellHeight - 1,
                0,
            ]));
            // ---------------------------------------------------------------------
            // Draw axes.
            // ---------------------------------------------------------------------
            // X-Axes and labels.
            const xAxesG = this._svg
                .selectAll('.x-axis')
                .data(cols)
                .enter()
                .append('g')
                .classed('x-axis', true)
                .attr('transform', (col) => utils.translateStr(cellX(col), 0));
            function xAxisClipPathId(col) {
                return 'x-axis-clip-path-' + col;
            }
            function xLabelClipPathId(col) {
                return 'x-label-clip-path-' + col;
            }
            xAxesG
                .append('clipPath')
                .attr('id', xAxisClipPathId)
                .append('rect')
                .attr('x', -frameMargin)
                .attr('y', 0)
                .attr('width', cellWidth + 2 * frameMargin)
                .attr('height', _this.height - xAxisTextMargin / 2);
            xAxesG
                .append('clipPath')
                .attr('id', xLabelClipPathId)
                .append('rect')
                .attr('x', 0)
                .attr('y', _this.height - xAxisTextMargin / 2)
                .attr('width', cellWidth)
                .attr('height', xAxisTextMargin / 2);
            xAxesG
                .append('g')
                .attr('clip-path', (col) => 'url(#' + xAxisClipPathId(col) + ')')
                .each(function (col) {
                select(this).call(drawAxis, axisBottom(xCoords[col])
                    .tickSize(_this.height - xAxisTextMargin), cellWidth, 
                /* minLabelSize */ 40, _this.options.columns[col].scale);
            });
            // Draw a label for each axis.
            xAxesG
                .append('g')
                .classed('x-axis-label', true)
                .attr('clip-path', (col) => 'url(#' + xLabelClipPathId(col) + ')')
                .append('text')
                .attr('text-anchor', 'middle')
                .attr('x', cellWidth / 2)
                .attr('y', _this.height - 1 - xAxisTextMargin / 4)
                .text((col) => utils.schemaVisibleColumnName(_this.visibleSchema, col))
                .append('title') // Show full name as a tooltip.
                .text((col) => utils.schemaVisibleColumnName(_this.visibleSchema, col));
            // Y-Axes and labels.
            const yAxesG = this._svg
                .selectAll('.y-axis')
                .data(metrics)
                .enter()
                .append('g')
                .classed('y-axis', true)
                .attr('transform', (metric) => utils.translateStr(_this.width - 1, cellY(metric)));
            function yAxisClipPathId(metric) {
                return 'y-axis-clip-path-' + metric;
            }
            function yLabelClipPathId(metric) {
                return 'y-label-clip-path-' + metric;
            }
            yAxesG
                .append('clipPath')
                .attr('id', yAxisClipPathId)
                .append('rect')
                .attr('x', -(_this.width - yAxisTextMargin / 2 - 1))
                .attr('y', -frameMargin)
                .attr('width', _this.width - yAxisTextMargin / 2)
                .attr('height', cellHeight + 2 * frameMargin);
            yAxesG
                .append('clipPath')
                .attr('id', yLabelClipPathId)
                .append('rect')
                .attr('x', -(_this.width - 1))
                .attr('y', 0)
                .attr('width', yAxisTextMargin / 2)
                .attr('height', cellHeight);
            yAxesG
                .append('g')
                .attr('clip-path', (metric) => 'url(#' + yAxisClipPathId(metric) + ')')
                .each(function (metric) {
                select(this).call(drawAxis, axisLeft(yCoords[metric])
                    .tickSize(_this.width - yAxisTextMargin), cellHeight, 
                /* minLabelSize */ 20, _this.options.columns[metric + utils.numVisibleHParams(_this.visibleSchema)].scale);
            });
            // Append a label for each axis.
            yAxesG
                .append('g')
                .classed('y-axis-label', true)
                .attr('clip-path', (metric) => 'url(#' + yLabelClipPathId(metric) + ')')
                .append('text')
                .attr('text-anchor', 'middle')
                .attr('x', -(_this.width - yAxisTextMargin / 4 - 1))
                .attr('y', cellHeight / 2)
                .attr('transform', utils.rotateStr(90, -(_this.width - yAxisTextMargin / 4 - 1), cellHeight / 2))
                .text((metric) => utils.metricName(_this.visibleSchema.metricInfos[metric]))
                .append('title') // Show full name as a tooltip.
                .text((metric) => utils.metricName(_this.visibleSchema.metricInfos[metric]));
            function drawAxis(g, axisGen, axisLength, minLabelSize, scaleType) {
                // We compute the number of ticks to display based on the estimate
                // of the minimum size to allow for a label.
                const numTicks = Math.floor(axisLength / minLabelSize);
                const scale = axisGen.scale();
                if (scaleType === 'QUANTILE') {
                    // The default tickValues of a quantile scale is just the scale
                    // domain, which produces overlapping labels if the number of
                    // elements in the domain is greater than the number of
                    // quantiles.
                    let quantiles = scale.quantiles();
                    const step = Math.ceil(quantiles.length / numTicks);
                    quantiles = sequence(0, quantiles.length, step)
                        .map((i) => quantiles[i]);
                    axisGen.tickValues(quantiles).tickFormat(format('-.2g'));
                }
                if (scaleType === 'LINEAR' || scaleType === 'LOG') {
                    // The following is equivalent to: axisGen.ticks(numTicks). We
                    // use the form below, since otherwise the closure compiler
                    // erroneously drops the parameter 'numTicks' from the call. It does
                    // this, since d3 defines the variadic 'ticks' method as
                    // function(), which closure regards as a function that takes no
                    // parameters.
                    axisGen['ticks'](numTicks);
                }
                g.call(axisGen);
                // Remove the actual axis line, and grey out the tick lines.
                g.selectAll('.domain').remove();
                g.selectAll('.tick line').attr('stroke', '#ddd');
            }
            // ---------------------------------------------------------------------
            // Draw cell frames.
            // ---------------------------------------------------------------------
            const cells = this._svg
                .selectAll('.cell')
                .data(cross(cols, metrics))
                .enter()
                .append('g')
                .classed('cell', true)
                .attr('transform', ([col, metric]) => utils.translateStr(cellX(col), cellY(metric)));
            const frames = cells
                .append('g')
                .classed('frame', true)
                .append('rect')
                .attr('x', -frameMargin)
                .attr('y', -frameMargin)
                .attr('width', cellWidth + 2 * frameMargin)
                .attr('height', cellHeight + 2 * frameMargin)
                .attr('stroke', '#000')
                .attr('fill', 'none')
                .attr('shape-rendering', 'crispEdges');
            // ---------------------------------------------------------------------
            // Draw data point markers.
            // ---------------------------------------------------------------------
            let colorScale = null;
            if (_this.options.colorByColumnIndex !== undefined) {
                colorScale = linear$2()
                    .domain(this._colExtent(this.options.colorByColumnIndex))
                    .range([this.options.minColor, this.options.maxColor])
                    .interpolate(lab$1);
            }
            // A function mapping a sessionGroup to its marker's color.
            const markerColorFn = _this.options.colorByColumnIndex === undefined
                ? /* Use default color if no color-by column is selected. */
                    () => 'red'
                : ({ sessionGroup }) => colorScale(this._colValue(sessionGroup, _this.options.colorByColumnIndex));
            // Returns the x coordinate for the marker representing sessionGroup
            // in a cell in the scatter plot matrix column indexed by 'col'.
            function markerX(sessionGroup, col) {
                return xCoords[col](_this._colValue(sessionGroup, col));
            }
            // Returns the y coordinate for the marker representing sessionGroup
            // in a cell in the scatter plot matrix row indexed by 'metric'.
            function markerY(sessionGroup, metric) {
                return yCoords[metric](_this._metricValue(sessionGroup, metric));
            }
            // A function that gets a selection of <g> elements--each should be
            // a child node of a cell <g> element (a memeber of the 'cells'
            // selection) and draws markers representing the data points in each
            // cell. The parameter 'fill' is either a constant specifying the
            // 'fill' attribute of each marker or a function taking a session
            // group that returns the fill attribute that should be set for the
            // marker representing the given session group. The function returns
            // a 3-tuple of [markers, cellMarkers, sessionGroupMarkersMap],
            // where: markers is the d3-selection of the markers,
            // cellMarkers is a 2-D array whose [col][metric] entry has a
            // d3-selection containing the markers in the [col, metric] cell,
            // and sessionGroupMarkersMap is a Map mapping a sessionGroup to the
            // array of marker HTML elements representing that sessionGroup.
            function addMarkers(cellsGSelection, fill) {
                const markers = cellsGSelection
                    .selectAll('.data-marker')
                    .data(([col, metric]) => 
                // Filter out session groups that don't have a metric-value
                // or a column-value for the current cell.
                _this.sessionGroups
                    .filter((sessionGroup) => _this._colValue(sessionGroup, col) !== undefined &&
                    _this._metricValue(sessionGroup, metric) !== undefined)
                    .map((sessionGroup) => ({
                    col: col,
                    metric: metric,
                    sessionGroup: sessionGroup,
                    x: markerX(sessionGroup, col),
                    y: markerY(sessionGroup, metric),
                    // This will be populated by the code below with
                    // a Set of all the markers representing this session
                    // group.
                    sessionGroupMarkers: null,
                })))
                    .enter()
                    .append('circle')
                    .classed('data-marker', true)
                    .attr('cx', ({ x }) => x)
                    .attr('cy', ({ y }) => y)
                    .attr('r', 2)
                    .attr('fill', fill);
                const sessionGroupMarkersMap = new Map();
                _this.sessionGroups.forEach((sessionGroup) => {
                    sessionGroupMarkersMap.set(sessionGroup, []);
                });
                markers.each(function (d) {
                    sessionGroupMarkersMap.get(d.sessionGroup).push(this);
                });
                markers.each((d) => {
                    const sessionGroupMarkers = sessionGroupMarkersMap.get(d.sessionGroup);
                    d.sessionGroupMarkers = new Set(sessionGroupMarkers);
                });
                const cellMarkers = cols.map((col) => metrics.map((metric) => markers.filter((d) => d.col == col && d.metric == metric)));
                return [markers, cellMarkers, sessionGroupMarkersMap];
            }
            const [markers, cellMarkers, sessionGroupMarkersMap] = addMarkers(cells.append('g'), 
            /* fill */ markerColorFn);
            // ---------------------------------------------------------------------
            // Create a brush for each cell. Brushing a cell makes "visible" only
            // the markers associated with session groups whose markers
            // in the brushed cell lie within the brush selection. By "visibile"
            // here, we man colored according to color-by column. Markers that
            // are not "visibile" will be shown as grayed out.
            // ---------------------------------------------------------------------
            // For each cell, we index the markers in a quad-tree to quickly
            // find the intersection of the markers with the (brush) selection.
            // The following function creates this quad-tree for the cell indexed by
            // (col, metric). Each quad tree datum is the corresponding marker's
            // element.
            function createCellQuadTree(col, metric) {
                const data = [];
                cellMarkers[col][metric].each(function () {
                    data.push(this);
                });
                return quadtree()
                    .x((elem) => select(elem).datum().x)
                    .y((elem) => select(elem).datum().y)
                    .addAll(data);
            }
            const quadTrees = cols.map((col) => metrics.map((metric) => createCellQuadTree(col, metric)));
            // A d3-selection of the cell in 'cells' that has the active
            // brush selection, or null if the brush is not active.
            let brushedCellG = null;
            if (isBrushActive()) {
                brushedCellG = cells.filter((cellIndex) => lodash.isEqual(cellIndex, _this._brushedCellIndex));
                console.assert(brushedCellG.size() == 1, brushedCellG);
            }
            // The set of markers (in all cells) that are visible. We keep this
            // set around so that when the brush selection changes we can change
            // the "fill" attribute of only the markers we need to. This reduces
            // the browser's rendering time and makes brushing smoother.
            let visibleMarkers = new Set(markers.nodes());
            updateVisibleMarkers();
            function updateVisibleMarkers() {
                // We regard an empty (or inactive) brush selection as selecting
                // all markers.
                let newVisibleMarkers = new Set(markers.nodes());
                if (!isBrushSelectionEmpty()) {
                    newVisibleMarkers = findMarkersInSelection(_this._brushedCellIndex, _this._brushSelection);
                }
                // Highlight the new visible markers.
                selectAll(Array.from(utils.filterSet(newVisibleMarkers, (elem) => !visibleMarkers.has(elem)))).attr('fill', markerColorFn);
                // Gray-out the no-longer visible markers.
                selectAll(Array.from(utils.filterSet(visibleMarkers, (elem) => !newVisibleMarkers.has(elem)))).attr('fill', '#ddd');
                visibleMarkers = newVisibleMarkers;
            }
            // Returns a Set of all marker elements that are in the
            // rectangle 'selection' given in coordinates relative to the
            // cell indexed by cellIndex .
            function findMarkersInSelection(cellIndex, selection) {
                console.assert(cellIndex !== null);
                console.assert(selection !== null);
                const [col, metric] = cellIndex;
                const result = new Set();
                utils.quadTreeVisitPointsInRect(quadTrees[col][metric], selection[0][0], selection[0][1], selection[1][0], selection[1][1], (elem) => {
                    const data = select(elem).datum();
                    data.sessionGroupMarkers.forEach((sg_elem) => {
                        result.add(sg_elem);
                    });
                });
                return result;
            }
            const brush$1 = brush()
                .extent([
                [-frameMargin + 1, -frameMargin + 1],
                [cellWidth - 1 + frameMargin - 1, cellHeight - 1 + frameMargin - 1],
            ])
                .on('start', function () {
                if (isBrushActive() && brushedCellG.node() != this) {
                    // The brush is active in a different cell.
                    // Clear the selection first.
                    // This will recursively call the 'start', 'brush', and
                    // 'end' event listeners for the cell with the selection
                    // and will update the markers. The 'if' above
                    // prevents infinite recursion.
                    brush$1.move(brushedCellG, null);
                }
                brushChanged(this);
            })
                .on('brush', function () {
                brushChanged(this);
            })
                .on('end', function () {
                brushChanged(this);
            });
            // Updates the internal state in response to a brush event in
            // the cell whose <g> element (in 'cells') is given by cellGNode
            function brushChanged(cellGNode) {
                // For some reason the closure compiler drops the argument when we
                // write the call below as 'd3.brushSelection(cellGNode)'.
                const brushSelection$1 = brushSelection(cellGNode);
                if ((!isBrushActive() && brushSelection$1 === null) ||
                    (isBrushActive() &&
                        cellGNode === brushedCellG.node() &&
                        lodash.isEqual(brushSelection$1, _this._brushSelection))) {
                    // Nothing to do if selection hasn't changed.
                    return;
                }
                _this._brushSelection = brushSelection$1;
                if (brushSelection$1 !== null) {
                    brushedCellG = select(cellGNode);
                    _this._brushedCellIndex = brushedCellG.datum();
                }
                else {
                    brushedCellG = null;
                    _this._brushedCellIndex = null;
                }
                updateVisibleMarkers();
            }
            function isBrushActive() {
                return _this._brushedCellIndex !== null && _this._brushSelection !== null;
            }
            function isBrushSelectionEmpty() {
                return (!isBrushActive() ||
                    _this._brushSelection[0][0] === _this._brushSelection[1][0] ||
                    _this._brushSelection[0][1] === _this._brushSelection[1][1]);
            }
            // Render the brush elements on each cell.
            cells.call(brush$1);
            if (isBrushActive()) {
                // Set the internal brush selection to what it was before
                // the 'redraw()'.
                brush$1.move(brushedCellG, _this._brushSelection);
            }
            // ---------------------------------------------------------------------
            // Add event listeners for highlighting the session group whose markers
            // are closest to the mouse pointer (only "visible" session groups
            // are considered -- see brushing above). Also, add event listeners
            // for making the highlighted session group the currently-selected
            // group by clicking.
            // ---------------------------------------------------------------------
            // A d3-selection containing the nodes in markers representing the
            // SessionGroup with a marker closest to the mouse pointer or null
            // if the distance to the closest session group is greater than a
            // threshold. This won't get set until the first mouse movement over
            // a cell.
            let closestMarkers = null;
            // A d3-selection containing the nodes in markers representing the
            // markers of the currently selected session group or null if no
            // session group is selected.
            let selectedMarkers = null;
            if (this.selectedSessionGroup !== null) {
                selectedMarkers = selectAll(sessionGroupMarkersMap.get(this.selectedSessionGroup))
                    .classed('selected-marker', true);
            }
            cells
                .on('click', function () {
                const newSelectedMarkers = closestMarkers === selectedMarkers ? null : closestMarkers;
                if (newSelectedMarkers === selectedMarkers) {
                    return;
                }
                if (selectedMarkers !== null) {
                    selectedMarkers.classed('selected-marker', false);
                }
                selectedMarkers = newSelectedMarkers;
                if (selectedMarkers !== null) {
                    selectedMarkers.classed('selected-marker', true);
                }
                const newSessionGroup = selectedMarkers === null
                    ? null
                    : // All elements in selectedMarkers should have the same
                        // sessionGroup.
                        selectedMarkers.datum().sessionGroup;
                _this.selectedSessionGroup = newSessionGroup;
            })
                .on('mousemove mouseenter', function ([col, metric]) {
                const [x, y] = mouse(this);
                const newClosestMarkers = findClosestMarkers(col, metric, x, y, 
                /* threshold */ 20);
                if (closestMarkers === newClosestMarkers) {
                    return;
                }
                if (closestMarkers !== null) {
                    closestMarkers.classed('closest-marker', false);
                }
                closestMarkers = newClosestMarkers;
                if (closestMarkers !== null) {
                    closestMarkers.classed('closest-marker', true);
                    // All elements in closestMarkers should have the same
                    // sessionGroup.
                    _this.closestSessionGroup = closestMarkers.datum().sessionGroup;
                }
                else {
                    _this.closestSessionGroup = null;
                }
            })
                .on('mouseleave', function ([col, metric]) {
                if (closestMarkers !== null) {
                    closestMarkers.classed('closest-marker', false);
                    closestMarkers = null;
                    _this.closestSessionGroup = null;
                }
            });
            // Finds a closest visible marker in the [col,metric] cell to the point
            // with cell-relative coordinates (x,y). If that point's distance
            // to the point at (x,y) is larger than threshold, returns null;
            // otherwise returns the d3-selection consisting of the markers
            // representing the session group of that closest marker.
            function findClosestMarkers(metric, col, x, y, threshold) {
                let minDist = Infinity;
                let minSessionGroup = null;
                utils.quadTreeVisitPointsInDisk(quadTrees[metric][col], x, y, threshold, (elem, distanceToCenter) => {
                    if (visibleMarkers.has(elem) && distanceToCenter < minDist) {
                        const data = select(elem).datum();
                        minDist = distanceToCenter;
                        minSessionGroup = data.sessionGroup;
                    }
                });
                if (minSessionGroup === null) {
                    return null;
                }
                return selectAll(sessionGroupMarkersMap.get(minSessionGroup));
            }
            // ---------------------------------------------------------------------
            // Polymer adds an extra ".tf-hparams-scatter-plot-matrix-plot" class to
            // each rule selector in the <style> section written above. When
            // polymer stamps a template it adds this class to every element
            // stamped; since we're injecting our own elements here, we add this
            // class to each element so that the style rules defined above will
            // apply.
            this._svg
                .selectAll('*')
                .classed('tf-hparams-scatter-plot-matrix-plot', true);
        }
        // Returns a d3 scale mapping either a metric or an hparam value
        // to the cell's coordinate system. 'range' should be an interval in the
        // cell's coordinate system representing the range of the scale. The
        // metric or hparam to be mapped should be specified as the
        // visibleSchema-column indexed by colIndex.
        _cellScale(colIndex, range) {
            const extent = this._colExtent(colIndex);
            const linearScale = linear$2().domain(extent).range(range);
            if (this.options.columns[colIndex].scale === 'LINEAR') {
                return linearScale;
            }
            else if (this.options.columns[colIndex].scale === 'LOG') {
                if (extent[0] <= 0 && extent[1] >= 0) {
                    // We can't have a log scale for data whose extent contains 0.
                    // Use a linear scale instead.
                    // TODO(erez): Create a symlog scale similar to Matplotlib's
                    // symlog. See also d3 issue here:
                    // https://github.com/d3/d3-scale/issues/105
                    // and b/111755540
                    return linearScale;
                }
                return log$1().domain(extent).range(range);
            }
            else if (this.options.columns[colIndex].scale === 'QUANTILE') {
                // Compute 20-quantiles:
                const step = (range[1] - range[0]) / 19;
                // Compute the scale's range to be:
                // {range[0], range[0]+step, ..., range[1]}.
                // d3.range(range[0], range[1]+step, step) has numerical
                // issues and may produce an extra member, so we use a
                // different procedure:
                const scaleRange = sequence(20).map((i) => range[0] + step * i);
                return quantile()
                    .domain(lodash.uniq(this.sessionGroups.map((sg) => this._colValue(sg, colIndex))))
                    .range(scaleRange);
            }
            else if (this.options.columns[colIndex].scale === 'NON_NUMERIC') {
                return point$1()
                    .domain(
                // We sort the session groups to make the order
                // stable across 'ListSessionGroups' RPCs
                lodash.uniq(this.sessionGroups.map((sg) => this._colValue(sg, colIndex)).sort()))
                    .range(range)
                    .padding(0.1);
            }
            else {
                throw ('Unknown scale for column: ' + colIndex + '. options: ' + this.options);
            }
        }
        _colValue(sessionGroup, colIndex) {
            return columnValueByVisibleIndex(this.visibleSchema, sessionGroup, colIndex);
        }
        _metricValue(sessionGroup, metricIndex) {
            return metricValueByVisibleIndex(this.visibleSchema, sessionGroup, metricIndex);
        }
        _colExtent(colIndex) {
            return visibleNumericColumnExtent(this.visibleSchema, this.sessionGroups, colIndex);
        }
    };
    TfHparamsScatterPlotMatrixPlot.template = html `
    <div id="container">
      <svg id="svg"></svg>
    </div>

    <style>
      :host {
        display: block;
      }
      svg {
        font: 10px sans-serif;
      }

      /* The closest data point marker to the mouse pointer. We use !important
         to override the inline style that sets the regular style of a marker.
      */
      .closest-marker {
        r: 6 !important;
      }

      /* The currently selected data point marker. We use !important to
         override the inline style that sets the regular style of a marker. */
      .selected-marker {
        r: 6 !important;
        fill: #0f0 !important;
      }
    </style>
  `;
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfHparamsScatterPlotMatrixPlot.prototype, "visibleSchema", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfHparamsScatterPlotMatrixPlot.prototype, "sessionGroups", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfHparamsScatterPlotMatrixPlot.prototype, "options", void 0);
    __decorate([
        property({
            type: Object,
            notify: true,
        }),
        __metadata("design:type", Object)
    ], TfHparamsScatterPlotMatrixPlot.prototype, "selectedSessionGroup", void 0);
    __decorate([
        property({
            type: Object,
            notify: true,
        }),
        __metadata("design:type", Object)
    ], TfHparamsScatterPlotMatrixPlot.prototype, "closestSessionGroup", void 0);
    __decorate([
        property({
            type: Object,
        }),
        __metadata("design:type", HTMLElement)
    ], TfHparamsScatterPlotMatrixPlot.prototype, "_container", void 0);
    __decorate([
        property({
            type: Object,
        }),
        __metadata("design:type", Object)
    ], TfHparamsScatterPlotMatrixPlot.prototype, "_svg", void 0);
    __decorate([
        property({
            type: Number,
        }),
        __metadata("design:type", Number)
    ], TfHparamsScatterPlotMatrixPlot.prototype, "width", void 0);
    __decorate([
        property({
            type: Number,
        }),
        __metadata("design:type", Number)
    ], TfHparamsScatterPlotMatrixPlot.prototype, "height", void 0);
    __decorate([
        property({
            type: Object,
        }),
        __metadata("design:type", Object)
    ], TfHparamsScatterPlotMatrixPlot.prototype, "_brushedCellIndex", void 0);
    __decorate([
        property({
            type: Object,
        }),
        __metadata("design:type", Object)
    ], TfHparamsScatterPlotMatrixPlot.prototype, "_brushSelection", void 0);
    __decorate([
        observe('sessionGroups.*'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfHparamsScatterPlotMatrixPlot.prototype, "_sessionGroupsChanged", null);
    __decorate([
        observe('visibleSchema.*'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfHparamsScatterPlotMatrixPlot.prototype, "_visibleSchemaChanged", null);
    __decorate([
        observe('options.*'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfHparamsScatterPlotMatrixPlot.prototype, "_redraw", null);
    TfHparamsScatterPlotMatrixPlot = __decorate([
        customElement('tf-hparams-scatter-plot-matrix-plot')
    ], TfHparamsScatterPlotMatrixPlot);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * A D3-based implementation of a scatter plot matrix of the sessions
     * groups.
     *
     * For more details on a scatter plot matrix visualization see for example:
     * https://www.itl.nist.gov/div898/handbook/eda/section3/eda33qb.htm
     *
     * There are 3 elements involved in the scatter plot visualization:
     *
     * 1. <tf-hparams-scatter-plot-matrix-plot>
     *   Renders the actual scatter plot matrix
     * 2. <tf-hparams-scale-and-color-controls>
     *   A control panel for configuring the behavior of the plot (e.g. scale
     *   of each axis, colormap, etc.)
     * 3. <tf-hparams-scatter-plot-matrix-view>
     *   The container element for the above 2 elements.
     */
    let TfHparamsScatterPlotMatrixView = class TfHparamsScatterPlotMatrixView extends PolymerElement {
        _closestOrSelected(closestSessionGroup, selectedSessionGroup) {
            if (closestSessionGroup !== null) {
                return closestSessionGroup;
            }
            return selectedSessionGroup;
        }
    };
    TfHparamsScatterPlotMatrixView.template = html `
    <hparams-split-layout orientation="vertical">
      <!-- Controls behavior of the scatter plot matrix
             outputs the configured options to the _options property. -->
      <tf-hparams-scale-and-color-controls
        slot="content"
        class="section"
        id="controls"
        configuration="[[configuration]]"
        session-groups="[[sessionGroups]]"
        options="{{_options}}"
      >
      </tf-hparams-scale-and-color-controls>
      <!-- The actual scatter plot matrix -->
      <tf-hparams-scatter-plot-matrix-plot
        slot="content"
        class="section"
        id="plot"
        visible-schema="[[configuration.visibleSchema]]"
        session-groups="[[sessionGroups]]"
        selected-session-group="{{_selectedGroup}}"
        closest-session-group="{{_closestGroup}}"
        options="[[_options]]"
      >
      </tf-hparams-scatter-plot-matrix-plot>
      <tf-hparams-session-group-values
        slot="content"
        class="section"
        id="values"
        visible-schema="[[configuration.visibleSchema]]"
        session-group="[[_closestOrSelected(
                                 _closestGroup, _selectedGroup)]]"
      >
      </tf-hparams-session-group-values>
      <!-- Shows session group details for the clicked marker. -->
      <tf-hparams-session-group-details
        slot="content"
        class="section"
        id="details"
        backend="[[backend]]"
        experiment-name="[[experimentName]]"
        session-group="[[_selectedGroup]]"
        visible-schema="[[configuration.visibleSchema]]"
      >
      </tf-hparams-session-group-details>
    </hparams-split-layout>
    <style>
      .section {
        padding: 10px;
      }
      #controls {
        flex-grow: 0;
        flex-shrink: 0;
        flex-basis: auto;
        height: auto;
        overflow-y: auto;
        max-height: fit-content;
      }
      #plot {
        flex-grow: 1;
        flex-shrink: 1;
        flex-basis: auto;
        height: auto;
        overflow-y: auto;
        max-height: fit-content;
      }
      #values {
        flex-grow: 0;
        flex-shrink: 0;
        flex-basis: auto;
        height: 115px;
        overflow-y: auto;
        max-height: fit-content;
      }
      #details {
        flex-grow: 0;
        flex-shrink: 1;
        flex-basis: auto;
        height: auto;
        overflow-y: auto;
        max-height: fit-content;
      }
      vaadin-split-layout {
        height: 100%;
      }
    </style>
  `;
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfHparamsScatterPlotMatrixView.prototype, "backend", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfHparamsScatterPlotMatrixView.prototype, "experimentName", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfHparamsScatterPlotMatrixView.prototype, "configuration", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfHparamsScatterPlotMatrixView.prototype, "sessionGroups", void 0);
    TfHparamsScatterPlotMatrixView = __decorate([
        customElement('tf-hparams-scatter-plot-matrix-view')
    ], TfHparamsScatterPlotMatrixView);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * The tf-hparams-session-pane element implements a tabbed view of the
     * session groups. Each tab is implemented by its own element and shows
     * a specific view of the session groups list. Example of views are a table-view
     * and parallel-coordinates (https://en.wikipedia.org/wiki/Parallel_coordinates)
     * view.
     */
    let TfHparamsSessionsPane = class TfHparamsSessionsPane extends PolymerElement {
        constructor() {
            super(...arguments);
            this._selectedTab = 0;
        }
    };
    TfHparamsSessionsPane.template = html `
    <paper-header-panel>
      <paper-toolbar slot="header" class="tab-bar">
        <paper-tabs selected="{{_selectedTab}}" slot="top">
          <!-- view-id can be used by integration tests to locate a tab.
               It should be the name of the root element implementing the view
               without the 'tf-hparams-' prefix. -->
          <paper-tab view-id="table-view"> TABLE VIEW </paper-tab>
          <paper-tab view-id="parallel-coords-view">
            PARALLEL COORDINATES VIEW
          </paper-tab>
          <paper-tab view-id="scatter-plot-matrix-view">
            SCATTER PLOT MATRIX VIEW
          </paper-tab>
          <div class="help-and-feedback">
            <template is="dom-if" if="[[bugReportUrl]]">
              <a
                href$="[[bugReportUrl]]"
                target="_blank"
                rel="noopener noreferrer"
              >
                <paper-button
                  id="bug-report"
                  raised
                  title="Send a bug report or feature request"
                >
                  Bug Report / Feature Request
                </paper-button>
              </a>
            </template>
            <template is="dom-if" if="[[helpUrl]]">
              <a href$="[[helpUrl]]" target="_blank" rel="noopener noreferrer">
                <paper-icon-button
                  icon="help-outline"
                  title="View documentation"
                >
                </paper-icon-button>
              </a>
            </template>
          </div>
        </paper-tabs>
      </paper-toolbar>
      <iron-pages selected="[[_selectedTab]]" class="fit tab-view">
        <div id="0" class="tab">
          <tf-hparams-table-view
            backend="[[backend]]"
            experiment-name="[[experimentName]]"
            visible-schema="[[configuration.visibleSchema]]"
            session-groups="[[sessionGroups]]"
            enable-show-metrics
          >
          </tf-hparams-table-view>
        </div>
        <div id="1" class="tab">
          <tf-hparams-parallel-coords-view
            backend="[[backend]]"
            experiment-name="[[experimentName]]"
            configuration="[[configuration]]"
            session-groups="[[sessionGroups]]"
          >
          </tf-hparams-parallel-coords-view>
        </div>
        <div id="2" class="tab">
          <tf-hparams-scatter-plot-matrix-view
            backend="[[backend]]"
            experiment-name="[[experimentName]]"
            configuration="[[configuration]]"
            session-groups="[[sessionGroups]]"
          >
          </tf-hparams-scatter-plot-matrix-view>
        </div>
      </iron-pages>
    </paper-header-panel>

    <style>
      .tab-view {
        height: 100%;
      }
      .tab-bar {
        overflow-y: auto;
        color: white;
        background-color: var(
          --tb-toolbar-background-color,
          var(--tb-orange-strong)
        );
      }
      .tab {
        height: 100%;
      }
      paper-tabs {
        flex-grow: 1;
        width: 100%;
        height: 100%;
        --paper-tabs-selection-bar-color: white;
        --paper-tabs-content: {
          -webkit-font-smoothing: antialiased;
        }
      }
      tf-hparams-table-view {
        width: 100%;
        height: 100%;
      }
      .help-and-feedback {
        display: inline-flex; /* Ensure that icons stay aligned */
        justify-content: flex-end;
        align-items: center;
        text-align: right;
        color: white;
      }
      #bug-report {
        border: solid black;
        background: red;
        white-space: normal;
        word-break: break-words;
        font-size: 12px;
        max-width: 150px;
        text-align: left;
      }
      .help-and-feedback a {
        color: white;
        text-decoration: none;
      }
    </style>
  `;
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfHparamsSessionsPane.prototype, "backend", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfHparamsSessionsPane.prototype, "helpUrl", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfHparamsSessionsPane.prototype, "bugReportUrl", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfHparamsSessionsPane.prototype, "experimentName", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfHparamsSessionsPane.prototype, "configuration", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfHparamsSessionsPane.prototype, "sessionGroups", void 0);
    __decorate([
        property({
            type: Number,
        }),
        __metadata("design:type", Number)
    ], TfHparamsSessionsPane.prototype, "_selectedTab", void 0);
    TfHparamsSessionsPane = __decorate([
        customElement('tf-hparams-sessions-pane')
    ], TfHparamsSessionsPane);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /* A 'tf-hparams-backend' encapsulates sending HParams API requests to the
       backend.
       Any implementation with the same public interface can be passed to
       tf-hparams-main as the 'backend' property.
    */
    class Backend {
        // Constructs a backend that uses the given tf_backend.requestManager to
        // send requests. The apiUrl parameter should denote the base
        // url to use. If useHttpGet is true uses HTTP GET to send a request
        // otherwise uses HTTP POST. See tensorboard/plugins/hparams/http_api.md
        // for details on how the requests and responses are encoded in each
        // scheme (GET or POST).
        constructor(apiUrl, requestManager, useHttpGet = true) {
            this._apiUrl = apiUrl;
            this._requestManager = requestManager;
            this._useHttpGet = useHttpGet;
        }
        // In the API methods below, 'request' should be a JSON translated request
        // protocol buffer and the response is a JSON translated response protocol
        // buffer. See api.proto for the details.
        getExperiment(experimentRequest) {
            return this._sendRequest('experiment', experimentRequest);
        }
        getDownloadUrl(format, listSessionGroupsRequest, columnsVisibility) {
            return (this._apiUrl +
                '/download_data?' +
                new URLSearchParams({
                    format: format,
                    columnsVisibility: JSON.stringify(columnsVisibility),
                    request: JSON.stringify(listSessionGroupsRequest),
                }));
        }
        listSessionGroups(listSessionGroupsRequest) {
            return this._sendRequest('session_groups', listSessionGroupsRequest);
        }
        listMetricEvals(listMetricEvalsRequest) {
            return this._sendRequest('metric_evals', listMetricEvalsRequest);
        }
        // ---- Private methods below -------------------------------------------
        _sendRequest(methodName, request_proto) {
            if (this._useHttpGet) {
                const encodedRequest = encodeURIComponent(JSON.stringify(request_proto));
                const url = this._apiUrl + '/' + methodName + '?request=' + encodedRequest;
                return this._requestManager.request(url);
            }
            /* Use POST */
            const requestOptions = new RequestOptions();
            requestOptions.withCredentials = true;
            requestOptions.methodType = HttpMethodType.POST;
            requestOptions.contentType = 'text/plain';
            requestOptions.body = JSON.stringify(request_proto);
            const url = this._apiUrl + '/' + methodName;
            return this._requestManager.requestWithOptions(url, requestOptions);
        }
    }

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * The entry point into the hparams plugin frontend.
     * A container for two sub-elements:
     * 1. The query-pane encapsulated by the tf-hparams-query-pane element. This
     * element issues queries to the backend to get a list of session groups. It has
     * controls to allow the user to specify filters and sorting options in the
     * issued-query. The actual filtering and sorting is done by the backend.
     * 2. The session-pane encapsulated by the tf-hparams-session-pane element. This
     * element displays multiple tabs--each providing a "view" of the list of
     * session groups. Example of views are a table-view and a parallel-coordinate
     * view.
     *
     * TODO(erez): Add aggregation of repeated trials.
     */
    let TfHparamsMain = class TfHparamsMain extends LegacyElementMixin(PolymerElement) {
        // This can be called to refresh the plugin.
        reload() {
            this.$['query-pane'].reload();
        }
    };
    TfHparamsMain.template = html `
    <hparams-split-layout>
      <div slot="content" class="sidebar">
        <tf-hparams-query-pane
          id="query-pane"
          backend="[[backend]]"
          experiment-name="[[experimentName]]"
          configuration="{{_configuration}}"
          session-groups="{{_sessionGroups}}"
          data-loaded-with-non-empty-hparams="{{_dataLoadedWithNonEmptyHparams}}"
          data-loaded-with-empty-hparams="{{_dataLoadedWithEmptyHparams}}"
        >
        </tf-hparams-query-pane>
      </div>
      <div slot="content" class="center">
        <template is="dom-if" if="[[_dataLoadedWithEmptyHparams]]">
          <div class="no-data-warning">
            <h3>No hparams data was found.</h3>
            <p>Probable causes:</p>
            <ul>
              <li>You haven’t written any hparams data to your event files.</li>
              <li>
                Event files are still being loaded (try reloading this page).
              </li>
              <li>TensorBoard can’t find your event files.</li>
            </ul>

            <p>
              If you’re new to using TensorBoard, and want to find out how to
              add data and set up your event files, check out the
              <a
                href="https://github.com/tensorflow/tensorboard/blob/master/README.md"
                >README</a
              >
              and perhaps the
              <a
                href="https://www.tensorflow.org/get_started/summaries_and_tensorboard"
                >TensorBoard tutorial</a
              >.
            </p>

            <p>
              If you think TensorBoard is configured properly, please see
              <a
                href="https://github.com/tensorflow/tensorboard/blob/master/README.md#my-tensorboard-isnt-showing-any-data-whats-wrong"
                >the section of the README devoted to missing data problems</a
              >
              and consider filing an issue on GitHub.
            </p>
          </div>
        </template>

        <template is="dom-if" if="[[_dataLoadedWithNonEmptyHparams]]">
          <tf-hparams-sessions-pane
            id="sessions-pane"
            backend="[[backend]]"
            help-url="[[helpUrl]]"
            bug-report-url="[[bugReportUrl]]"
            experiment-name="[[experimentName]]"
            configuration="[[_configuration]]"
            session-groups="[[_sessionGroups]]"
          >
          </tf-hparams-sessions-pane>
        </template>
      </div>
    </hparams-split-layout>
    <style>
      hparams-split-layout {
        width: 100%;
      }

      .sidebar {
        width: 20%;
        height: 100%;
        overflow: auto;
        flex-grow: 0;
        flex-shrink: 0;
        min-width: 10%;
      }

      .center {
        height: 100%;
        overflow-y: auto;
        flex-grow: 1;
        flex-shrink: 1;
        width: 80%;
      }

      :host {
        display: flex;
        flex-direction: row;
        height: 100%;
        width: 100%;
      }

      .no-data-warning {
        max-width: 540px;
        margin: 80px auto 0 auto;
      }
    </style>
  `;
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Backend)
    ], TfHparamsMain.prototype, "backend", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfHparamsMain.prototype, "experimentName", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfHparamsMain.prototype, "helpUrl", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfHparamsMain.prototype, "bugReportUrl", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfHparamsMain.prototype, "_configuration", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfHparamsMain.prototype, "_sessionGroups", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfHparamsMain.prototype, "_dataLoadedWithNonEmptyHparams", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfHparamsMain.prototype, "_dataLoadedWithEmptyHparams", void 0);
    TfHparamsMain = __decorate([
        customElement('tf-hparams-main')
    ], TfHparamsMain);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    // Read URL at module import time, before AppRoutingEffects stomps it.
    const inColab = new URLSearchParams(window.location.search).get('tensorboardColab') ===
        'true';
    const PLUGIN_NAME = 'hparams';
    let TfHparamsDashboard = class TfHparamsDashboard extends LegacyElementMixin(PolymerElement) {
        constructor() {
            super(...arguments);
            this._backend = new Backend(
            /* apiUrl= */ getRouter()
                .pluginRoute(/* pluginName= */ PLUGIN_NAME, /* route= */ ''), new RequestManager(), 
            /* Use GETs if we're running in colab (due to b/126387106).
                  Otherwise use POSTs. */
            /* useHttpGet= */ inColab);
        }
        // This is called by the tensorboard web framework to refresh the plugin.
        reload() {
            this.$['hparams-main'].reload();
        }
    };
    TfHparamsDashboard.template = html `
    <!-- TensorBoard does not specify an experimentName. Currently it only
         supports one experiment per invocation. -->
    <tf-hparams-main
      id="hparams-main"
      backend="[[_backend]]"
      experiment-name=""
    >
    </tf-hparams-main>
  `;
    __decorate([
        property({
            type: Object,
        }),
        __metadata("design:type", Object)
    ], TfHparamsDashboard.prototype, "_backend", void 0);
    TfHparamsDashboard = __decorate([
        customElement('tf-hparams-dashboard')
    ], TfHparamsDashboard);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    let TfImageLoader = class TfImageLoader extends LegacyElementMixin(PolymerElement) {
        constructor() {
            super(...arguments);
            this.actualSize = false;
            this.brightnessAdjustment = 0.5;
            this.contrastPercentage = 0;
            this._metadataCanceller = new Canceller();
            this._imageCanceller = new Canceller();
            this._steps = [];
            this._isImageLoading = false;
        }
        get _runColor() {
            var run = this.run;
            return runsColorScale(run);
        }
        get _hasAtLeastOneStep() {
            var steps = this._steps;
            return !!steps && steps.length > 0;
        }
        get _hasMultipleSteps() {
            var steps = this._steps;
            return !!steps && steps.length > 1;
        }
        get _currentStep() {
            var steps = this._steps;
            var stepIndex = this._stepIndex;
            return steps[stepIndex] || null;
        }
        get _stepValue() {
            var currentStep = this._currentStep;
            if (!currentStep)
                return 0;
            return currentStep.step;
        }
        get _currentWallTime() {
            var currentStep = this._currentStep;
            if (!currentStep)
                return '';
            return formatDate$1(currentStep.wall_time);
        }
        get _maxStepIndex() {
            var steps = this._steps;
            return steps.length - 1;
        }
        get _sampleText() {
            var sample = this.sample;
            return `${sample + 1}`;
        }
        get _hasMultipleSamples() {
            var ofSamples = this.ofSamples;
            return ofSamples > 1;
        }
        _getAriaExpanded() {
            return this.actualSize ? 'true' : 'false';
        }
        attached() {
            this.reload();
        }
        reload() {
            if (!this.isAttached) {
                return;
            }
            this._metadataCanceller.cancelAll();
            const router = getRouter();
            const url = addParams(router.pluginRoute('images', '/images'), {
                tag: this.tag,
                run: this.run,
                sample: this.sample,
            });
            const updateSteps = this._metadataCanceller.cancellable((result) => {
                if (result.cancelled) {
                    return;
                }
                const data = result.value;
                const steps = data.map(this._createStepDatum.bind(this));
                this.set('_steps', steps);
                this.set('_stepIndex', steps.length - 1);
            });
            this.requestManager.request(url).then(updateSteps);
        }
        _createStepDatum(imageMetadata) {
            let url = getRouter().pluginRoute('images', '/individualImage');
            // Include wall_time just to disambiguate the URL and force
            // the browser to reload the image when the URL changes. The
            // backend doesn't care about the value.
            url = addParams(url, { ts: imageMetadata.wall_time });
            url += '&' + imageMetadata.query;
            return {
                // The wall time within the metadata is in seconds. The Date
                // constructor accepts a time in milliseconds, so we multiply by 1000.
                wall_time: new Date(imageMetadata.wall_time * 1000),
                step: imageMetadata.step,
                url,
            };
        }
        _updateImageUrl() {
            var currentStep = this._currentStep;
            var brightnessAdjustment = this.brightnessAdjustment;
            var contrastPercentage = this.contrastPercentage;
            // We manually change the image URL (instead of binding to the
            // image's src attribute) because we would like to manage what
            // happens when the image starts and stops loading.
            if (!currentStep)
                return;
            const img = new Image();
            this._imageCanceller.cancelAll();
            img.onload = img.onerror = this._imageCanceller
                .cancellable((result) => {
                if (result.cancelled) {
                    return;
                }
                const mainImageContainer = this.$$('#main-image-container');
                mainImageContainer.textContent = '';
                dom(mainImageContainer).appendChild(img);
                this.set('_isImageLoading', false);
            })
                .bind(this);
            img.style.filter = `contrast(${contrastPercentage}%) `;
            img.style.filter += `brightness(${brightnessAdjustment})`;
            // Load the new image.
            this.set('_isImageLoading', true);
            img.src = currentStep.url;
        }
        _handleTap(e) {
            this.set('actualSize', !this.actualSize);
        }
        _toLocaleString(number) {
            // Shows commas (or locale-appropriate punctuation) for large numbers.
            return number.toLocaleString();
        }
    };
    TfImageLoader.template = html `
    <tf-card-heading
      tag="[[tag]]"
      run="[[run]]"
      display-name="[[tagMetadata.displayName]]"
      description="[[tagMetadata.description]]"
      color="[[_runColor]]"
    >
      <template is="dom-if" if="[[_hasMultipleSamples]]">
        <div>sample: [[_sampleText]] of [[ofSamples]]</div>
      </template>
      <template is="dom-if" if="[[_hasAtLeastOneStep]]">
        <div class="heading-row">
          <div class="heading-label">
            step
            <span style="font-weight: bold"
              >[[_toLocaleString(_stepValue)]]</span
            >
          </div>
          <div class="heading-label heading-right datetime">
            <template is="dom-if" if="[[_currentWallTime]]">
              [[_currentWallTime]]
            </template>
          </div>
          <div class="label right">
            <paper-spinner-lite active hidden$="[[!_isImageLoading]]">
            </paper-spinner-lite>
          </div>
        </div>
      </template>
      <template is="dom-if" if="[[_hasMultipleSteps]]">
        <div>
          <paper-slider
            id="steps"
            immediate-value="{{_stepIndex}}"
            max="[[_maxStepIndex]]"
            max-markers="[[_maxStepIndex]]"
            snaps
            step="1"
            value="{{_stepIndex}}"
          ></paper-slider>
        </div>
      </template>
    </tf-card-heading>

    <!-- Semantically a button but <img> inside a <button> disallows user to do
    an interesting operation like "Copy Image" in non-Chromium browsers. -->
    <a
      id="main-image-container"
      role="button"
      aria-label="Toggle actual size"
      aria-expanded$="[[_getAriaExpanded(actualSize)]]"
      on-tap="_handleTap"
    ></a>

    <style include="tf-card-heading-style">
      /** Make button a div. */
      button {
        width: 100%;
        display: block;
        background: none;
        border: 0;
        padding: 0;
      }

      /** Firefox: Get rid of dotted line inside button. */
      button::-moz-focus-inner {
        border: 0;
        padding: 0;
      }

      /** Firefox: Simulate Chrome's outer glow on button when focused. */
      button:-moz-focusring {
        outline: none;
        box-shadow: 0px 0px 1px 2px Highlight;
      }

      :host {
        display: block;
        width: 350px;
        height: auto;
        position: relative;
        margin: 0 15px 40px 0;
        overflow-x: auto;
      }

      /** When actual size shown is on, use the actual image width. */
      :host([actual-size]) {
        max-width: 100%;
        width: auto;
      }

      :host([actual-size]) #main-image-container {
        max-height: none;
        width: auto;
      }

      :host([actual-size]) #main-image-container img {
        width: auto;
      }

      paper-spinner-lite {
        width: 14px;
        height: 14px;
        vertical-align: text-bottom;
        --paper-spinner-color: var(--tb-orange-strong);
      }

      #steps {
        height: 15px;
        margin: 0 0 0 -15px;
        /*
         * 31 comes from adding a padding of 15px from both sides of the
         * paper-slider, subtracting 1px so that the slider width aligns
         * with the image (the last slider marker takes up 1px), and
         * adding 2px to account for a border of 1px on both sides of
         * the image. 30 - 1 + 2.
         */
        width: calc(100% + 31px);
        --paper-slider-active-color: var(--tb-orange-strong);
        --paper-slider-knob-color: var(--tb-orange-strong);
        --paper-slider-knob-start-border-color: var(--tb-orange-strong);
        --paper-slider-knob-start-color: var(--tb-orange-strong);
        --paper-slider-markers-color: var(--tb-orange-strong);
        --paper-slider-pin-color: var(--tb-orange-strong);
        --paper-slider-pin-start-color: var(--tb-orange-strong);
      }

      #main-image-container {
        max-height: 1024px;
        overflow: auto;
      }

      #main-image-container img {
        cursor: pointer;
        display: block;
        image-rendering: -moz-crisp-edges;
        image-rendering: pixelated;
        width: 100%;
        height: auto;
      }

      paper-icon-button {
        color: #2196f3;
        border-radius: 100%;
        width: 32px;
        height: 32px;
        padding: 4px;
      }
      paper-icon-button[selected] {
        background: var(--tb-ui-light-accent);
      }
      [hidden] {
        display: none;
      }
    </style>
  `;
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfImageLoader.prototype, "run", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfImageLoader.prototype, "tag", void 0);
    __decorate([
        property({ type: Number }),
        __metadata("design:type", Number)
    ], TfImageLoader.prototype, "sample", void 0);
    __decorate([
        property({ type: Number }),
        __metadata("design:type", Number)
    ], TfImageLoader.prototype, "ofSamples", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfImageLoader.prototype, "tagMetadata", void 0);
    __decorate([
        property({
            type: Boolean,
            reflectToAttribute: true,
        }),
        __metadata("design:type", Boolean)
    ], TfImageLoader.prototype, "actualSize", void 0);
    __decorate([
        property({
            type: Number,
        }),
        __metadata("design:type", Number)
    ], TfImageLoader.prototype, "brightnessAdjustment", void 0);
    __decorate([
        property({
            type: Number,
        }),
        __metadata("design:type", Number)
    ], TfImageLoader.prototype, "contrastPercentage", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", RequestManager)
    ], TfImageLoader.prototype, "requestManager", void 0);
    __decorate([
        property({
            type: Object,
        }),
        __metadata("design:type", Object)
    ], TfImageLoader.prototype, "_metadataCanceller", void 0);
    __decorate([
        property({
            type: Object,
        }),
        __metadata("design:type", Object)
    ], TfImageLoader.prototype, "_imageCanceller", void 0);
    __decorate([
        property({
            type: Array,
            notify: true,
        }),
        __metadata("design:type", Array)
    ], TfImageLoader.prototype, "_steps", void 0);
    __decorate([
        property({
            type: Number,
            notify: true,
        }),
        __metadata("design:type", Number)
    ], TfImageLoader.prototype, "_stepIndex", void 0);
    __decorate([
        property({
            type: Boolean,
        }),
        __metadata("design:type", Boolean)
    ], TfImageLoader.prototype, "_isImageLoading", void 0);
    __decorate([
        computed('run'),
        __metadata("design:type", String),
        __metadata("design:paramtypes", [])
    ], TfImageLoader.prototype, "_runColor", null);
    __decorate([
        computed('_steps'),
        __metadata("design:type", Boolean),
        __metadata("design:paramtypes", [])
    ], TfImageLoader.prototype, "_hasAtLeastOneStep", null);
    __decorate([
        computed('_steps'),
        __metadata("design:type", Boolean),
        __metadata("design:paramtypes", [])
    ], TfImageLoader.prototype, "_hasMultipleSteps", null);
    __decorate([
        computed('_steps', '_stepIndex'),
        __metadata("design:type", Object),
        __metadata("design:paramtypes", [])
    ], TfImageLoader.prototype, "_currentStep", null);
    __decorate([
        computed('_currentStep'),
        __metadata("design:type", Number),
        __metadata("design:paramtypes", [])
    ], TfImageLoader.prototype, "_stepValue", null);
    __decorate([
        computed('_currentStep'),
        __metadata("design:type", String),
        __metadata("design:paramtypes", [])
    ], TfImageLoader.prototype, "_currentWallTime", null);
    __decorate([
        computed('_steps'),
        __metadata("design:type", Number),
        __metadata("design:paramtypes", [])
    ], TfImageLoader.prototype, "_maxStepIndex", null);
    __decorate([
        computed('sample'),
        __metadata("design:type", String),
        __metadata("design:paramtypes", [])
    ], TfImageLoader.prototype, "_sampleText", null);
    __decorate([
        computed('ofSamples'),
        __metadata("design:type", Boolean),
        __metadata("design:paramtypes", [])
    ], TfImageLoader.prototype, "_hasMultipleSamples", null);
    __decorate([
        observe('run', 'tag'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfImageLoader.prototype, "reload", null);
    __decorate([
        observe('_currentStep', 'brightnessAdjustment', 'contrastPercentage'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfImageLoader.prototype, "_updateImageUrl", null);
    TfImageLoader = __decorate([
        customElement('tf-image-loader')
    ], TfImageLoader);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    let TfImageDashboard = class TfImageDashboard extends LegacyElementMixin(PolymerElement) {
        constructor() {
            super(...arguments);
            this.reloadOnReady = true;
            this._defaultBrightnessAdjustment = 1;
            this._defaultContrastPercentage = 100;
            this._brightnessAdjustment = 1;
            this._contrastPercentage = 100;
            this._requestManager = new RequestManager();
        }
        ready() {
            super.ready();
            if (this.reloadOnReady)
                this.reload();
        }
        reload() {
            this._fetchTags().then(() => {
                this._reloadImages();
            });
        }
        _fetchTags() {
            const url = getRouter().pluginRoute('images', '/tags');
            return this._requestManager.request(url).then((runToTagInfo) => {
                if (lodash.isEqual(runToTagInfo, this._runToTagInfo)) {
                    // No need to update anything if there are no changes.
                    return;
                }
                const runToTag = lodash.mapValues(runToTagInfo, (x) => Object.keys(x));
                const tags = getTags(runToTag);
                this.set('_dataNotFound', tags.length === 0);
                this.set('_runToTagInfo', runToTagInfo);
                this.async(() => {
                    // See the comment above `_categoriesDomReady`.
                    this.set('_categoriesDomReady', true);
                });
            });
        }
        _reloadImages() {
            this.root.querySelectorAll('tf-image-loader').forEach((image) => {
                image.reload();
            });
        }
        _shouldOpen(index) {
            return index <= 2;
        }
        _resetBrightness() {
            this._brightnessAdjustment = this._defaultBrightnessAdjustment;
        }
        _resetContrast() {
            this._contrastPercentage = this._defaultContrastPercentage;
        }
        get _brightnessIsDefault() {
            var brightnessAdjustment = this._brightnessAdjustment;
            return brightnessAdjustment === this._defaultBrightnessAdjustment;
        }
        get _contrastIsDefault() {
            var contrastPercentage = this._contrastPercentage;
            return contrastPercentage === this._defaultContrastPercentage;
        }
        get _categories() {
            var runToTagInfo = this._runToTagInfo;
            var selectedRuns = this._selectedRuns;
            var tagFilter = this._tagFilter;
            var categoriesDomReady = this._categoriesDomReady;
            const runToTag = lodash.mapValues(runToTagInfo, (x) => Object.keys(x));
            const baseCategories = categorizeRunTagCombinations(runToTag, selectedRuns, tagFilter);
            function explodeItem(item) {
                const samples = runToTagInfo[item.run][item.tag].samples;
                return lodash.range(samples).map((i) => Object.assign({}, item, {
                    sample: i,
                    ofSamples: samples,
                }));
            }
            const withSamples = baseCategories.map((category) => Object.assign({}, category, {
                items: [].concat.apply([], category.items.map(explodeItem)),
            }));
            return withSamples;
        }
        _tagMetadata(runToTagInfo, run, tag) {
            return runToTagInfo[run][tag];
        }
    };
    TfImageDashboard.template = html `
    <tf-dashboard-layout>
      <div class="sidebar" slot="sidebar">
        <div class="settings">
          <div class="sidebar-section">
            <div class="line-item">
              <paper-checkbox checked="{{_actualSize}}"
                >Show actual image size</paper-checkbox
              >
            </div>
          </div>
          <div class="sidebar-section">
            <h3 class="tooltip-container">Brightness adjustment</h3>
            <div class="resettable-slider-container">
              <paper-slider
                min="0"
                max="2"
                snaps
                pin
                step="0.01"
                value="{{_brightnessAdjustment}}"
                immediate-value="{{_brightnessAdjustment}}"
              ></paper-slider>
              <paper-button
                class="x-button"
                on-tap="_resetBrightness"
                disabled="[[_brightnessIsDefault]]"
                >Reset</paper-button
              >
            </div>
          </div>
          <div class="sidebar-section">
            <h3 class="tooltip-container">Contrast adjustment</h3>
            <div class="resettable-slider-container">
              <paper-slider
                min="0"
                max="500"
                snaps
                pin
                step="1"
                value="{{_contrastPercentage}}"
                immediate-value="{{_contrastPercentage}}"
              ></paper-slider>
              <paper-button
                class="x-button"
                on-tap="_resetContrast"
                disabled="[[_contrastIsDefault]]"
                >Reset</paper-button
              >
            </div>
          </div>
        </div>
        <div class="sidebar-section runs-selector">
          <tf-runs-selector
            id="runs-selector"
            selected-runs="{{_selectedRuns}}"
          ></tf-runs-selector>
        </div>
      </div>
      <div class="center" slot="center">
        <template is="dom-if" if="[[_dataNotFound]]">
          <div class="no-data-warning">
            <h3>No image data was found.</h3>
            <p>Probable causes:</p>
            <ul>
              <li>You haven’t written any image data to your event files.</li>
              <li>TensorBoard can’t find your event files.</li>
            </ul>

            <p>
              If you’re new to using TensorBoard, and want to find out how to
              add data and set up your event files, check out the
              <a
                href="https://github.com/tensorflow/tensorboard/blob/master/README.md"
                >README</a
              >
              and perhaps the
              <a
                href="https://www.tensorflow.org/get_started/summaries_and_tensorboard"
                >TensorBoard tutorial</a
              >.
            </p>

            <p>
              If you think TensorBoard is configured properly, please see
              <a
                href="https://github.com/tensorflow/tensorboard/blob/master/README.md#my-tensorboard-isnt-showing-any-data-whats-wrong"
                >the section of the README devoted to missing data problems</a
              >
              and consider filing an issue on GitHub.
            </p>
          </div>
        </template>
        <template is="dom-if" if="[[!_dataNotFound]]">
          <tf-tag-filterer tag-filter="{{_tagFilter}}"></tf-tag-filterer>
          <template is="dom-repeat" items="[[_categories]]" as="category">
            <tf-category-paginated-view
              category="[[category]]"
              initial-opened="[[_shouldOpen(index)]]"
            >
              <template>
                <tf-image-loader
                  active="[[active]]"
                  run="[[item.run]]"
                  tag="[[item.tag]]"
                  sample="[[item.sample]]"
                  of-samples="[[item.ofSamples]]"
                  tag-metadata="[[_tagMetadata(_runToTagInfo, item.run, item.tag)]]"
                  request-manager="[[_requestManager]]"
                  actual-size="[[_actualSize]]"
                  brightness-adjustment="[[_brightnessAdjustment]]"
                  contrast-percentage="[[_contrastPercentage]]"
                ></tf-image-loader>
              </template>
            </tf-category-paginated-view>
          </template>
        </template>
      </div>
    </tf-dashboard-layout>
    <style include="dashboard-style"></style>
    <style>
      .resettable-slider-container {
        display: flex;
      }
      .resettable-slider-container paper-slider {
        flex-grow: 1;
      }
      .resettable-slider-container paper-button {
        flex-grow: 0;
      }
      .resettable-slider-container paper-button[disabled] {
        background-color: unset;
      }
      .x-button {
        font-size: 13px;
        background-color: var(--tb-ui-light-accent);
        color: var(--tb-ui-dark-accent);
      }
      .no-data-warning {
        max-width: 540px;
        margin: 80px auto 0 auto;
      }
      paper-slider {
        --paper-slider-active-color: var(--tb-orange-strong);
        --paper-slider-knob-color: var(--tb-orange-strong);
        --paper-slider-knob-start-border-color: var(--tb-orange-strong);
        --paper-slider-knob-start-color: var(--tb-orange-strong);
        --paper-slider-markers-color: var(--tb-orange-strong);
        --paper-slider-pin-color: var(--tb-orange-strong);
        --paper-slider-pin-start-color: var(--tb-orange-strong);
      }
    </style>
  `;
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfImageDashboard.prototype, "reloadOnReady", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfImageDashboard.prototype, "_selectedRuns", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfImageDashboard.prototype, "_runToTagInfo", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfImageDashboard.prototype, "_dataNotFound", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfImageDashboard.prototype, "_actualSize", void 0);
    __decorate([
        property({ type: Number }),
        __metadata("design:type", Number)
    ], TfImageDashboard.prototype, "_defaultBrightnessAdjustment", void 0);
    __decorate([
        property({ type: Number }),
        __metadata("design:type", Number)
    ], TfImageDashboard.prototype, "_defaultContrastPercentage", void 0);
    __decorate([
        property({ type: Number }),
        __metadata("design:type", Number)
    ], TfImageDashboard.prototype, "_brightnessAdjustment", void 0);
    __decorate([
        property({ type: Number }),
        __metadata("design:type", Number)
    ], TfImageDashboard.prototype, "_contrastPercentage", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfImageDashboard.prototype, "_tagFilter", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfImageDashboard.prototype, "_categoriesDomReady", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfImageDashboard.prototype, "_requestManager", void 0);
    __decorate([
        computed('_brightnessAdjustment'),
        __metadata("design:type", Boolean),
        __metadata("design:paramtypes", [])
    ], TfImageDashboard.prototype, "_brightnessIsDefault", null);
    __decorate([
        computed('_contrastPercentage'),
        __metadata("design:type", Boolean),
        __metadata("design:paramtypes", [])
    ], TfImageDashboard.prototype, "_contrastIsDefault", null);
    __decorate([
        computed('_runToTagInfo', '_selectedRuns', '_tagFilter', '_categoriesDomReady'),
        __metadata("design:type", Array),
        __metadata("design:paramtypes", [])
    ], TfImageDashboard.prototype, "_categories", null);
    TfImageDashboard = __decorate([
        customElement('tf-image-dashboard')
    ], TfImageDashboard);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * Types of errors during network data roundtrip.
     * @enum {number}
     */
    var ErrorCodes;
    (function (ErrorCodes) {
        ErrorCodes[ErrorCodes["CANCELLED"] = 1] = "CANCELLED";
    })(ErrorCodes || (ErrorCodes = {}));
    /**
     * Types of content displayed by the plugin.
     * @enum {number}
     */
    var ContentType;
    (function (ContentType) {
        ContentType[ContentType["VERTEX"] = 1] = "VERTEX";
        ContentType[ContentType["FACE"] = 2] = "FACE";
        ContentType[ContentType["COLOR"] = 3] = "COLOR";
    })(ContentType || (ContentType = {}));
    /**
     * Types of content displayed by the plugin mapped to underlying data types.
     * @enum {string}
     */
    var ContentTypeToItemType;
    (function (ContentTypeToItemType) {
        ContentTypeToItemType["VERTEX"] = "float32";
        ContentTypeToItemType["FACE"] = "int32";
        ContentTypeToItemType["COLOR"] = "uint8";
    })(ContentTypeToItemType || (ContentTypeToItemType = {}));
    class ArrayBufferDataProvider {
        /**
         * ArrayBufferDataProvider constructor, initializes everything needed for
         * future requests to the server.
         * @param requestManager Request manager to communicate with the
         *  server.
         */
        constructor(requestManager) {
            this._canceller = new Canceller();
            this._requestManager = requestManager;
        }
        /**
         * Requests new data from the server.
         */
        reload(run, tag, sample) {
            this._canceller.cancelAll();
            return this._fetchMetadata(run, tag, sample);
        }
        /**
           * Requests new data of some particular type from the server.
           * @param {string} run Name of the run to get data for.
           * @param {string} tag Name of the tag to get data for.
           * @param {string} content_type Type of the content to retrieve.
           * @param {!array} metadata List of metadata to complete with data from the
           *  server.
           * @param {number} sample Sample index from a batch of data.
           * @param {number} step Step value, representing a point in the time when the
              event occurred.
           * @param {!Object} meshData Map to populate with mesh data.
           * @return {!Object} Promise object representing server request.
           * @private
           */
        _fetchDataByStep(run, tag, content_type, sample, step, meshData) {
            const url = getRouter().pluginRoute('mesh', '/data', new URLSearchParams({
                tag,
                run,
                content_type,
                sample: String(sample),
                step: String(step),
            }));
            const reshapeTo1xNx3 = function (data) {
                const channelsCount = 3;
                let items = [];
                for (let i = 0; i < data.length / channelsCount; i++) {
                    let dataEntry = [];
                    for (let j = 0; j < channelsCount; j++) {
                        dataEntry.push(data[i * channelsCount + j]);
                    }
                    items.push(dataEntry);
                }
                return items;
            };
            const processData = this._canceller.cancellable((response) => {
                if (response.cancelled) {
                    return Promise.reject({
                        code: ErrorCodes.CANCELLED,
                        message: 'Response was invalidated.',
                    });
                }
                let buffer = response.value;
                switch (content_type) {
                    case 'VERTEX':
                        meshData.vertices = reshapeTo1xNx3(new Float32Array(buffer));
                        break;
                    case 'FACE':
                        meshData.faces = reshapeTo1xNx3(new Int32Array(buffer));
                        break;
                    case 'COLOR':
                        meshData.colors = reshapeTo1xNx3(new Uint8Array(buffer));
                        break;
                }
                return meshData;
            });
            return this._requestManager
                .fetch(url, {
                method: 'GET',
                headers: {
                    responseType: 'arraybuffer',
                    contentType: ContentTypeToItemType[content_type],
                },
            })
                .then((response) => response.arrayBuffer())
                .then(processData);
        }
        /**
         * Requests new data for each type of metadata from the server.
         * Metadata consists of wall_time, step, tensor shape, content type and other
         * info, but not tensor data itself.
         * @param {!Object} stepDatum Dictionary with mesh data for a current step.
         * @param {string} run Name of the run to get data for.
         * @param {string} tag Name of the tug to get data for.
         * @param {number} sample Sample index from a batch of data.
         * @return {!Object} Joint promise for all requests being sent.
         * @private
         */
        fetchData(stepDatum, run, tag, sample) {
            let promises = [];
            // Map to populate with mesh data, i.e. vertices, faces, etc.
            let meshData = new Map();
            Object.keys(ContentType).forEach((contentType) => {
                const component = 1 << ContentType[contentType];
                if (stepDatum.components & component) {
                    promises.push(this._fetchDataByStep(run, tag, contentType, sample, stepDatum.step, meshData));
                }
            });
            return Promise.all(promises);
        }
        /**
         * Requests new metadata from the server
         * @param {string} run Name of the run to get data for.
         * @param {string} tag Name of the tug to get data for.
         * @param {number} sample Sample index from a batch of data.
         *  completion.
         * @return {!Object} Promise for requested metadata.
         * @private
         */
        _fetchMetadata(run, tag, sample) {
            this._canceller.cancelAll();
            const url = getRouter().pluginRoute('mesh', '/meshes', new URLSearchParams({ tag, run, sample }));
            const requestData = this._canceller.cancellable((response) => {
                if (response.cancelled) {
                    return Promise.reject({
                        code: ErrorCodes.CANCELLED,
                        message: 'Response was invalidated.',
                    });
                }
                return response.value;
            });
            return this._requestManager
                .fetch(url)
                .then((response) => response.json())
                .then(requestData)
                .then(this._processMetadata.bind(this));
        }
        /**
         * Process server raw data into frontend friendly format.
         * @param {!Array|undefined} data list of raw server records.
         * @return {!Array} list of step datums.
         * @private
         */
        _processMetadata(data) {
            if (!data)
                return;
            const stepToData = new Map();
            for (let i = 0; i < data.length; i++) {
                let dataEntry = data[i];
                if (!stepToData.has(dataEntry.step)) {
                    stepToData.set(dataEntry.step, []);
                }
                stepToData.get(dataEntry.step).push(dataEntry);
            }
            let datums = [];
            stepToData.forEach((data) => {
                let datum = this._createStepDatum(data[0]);
                datums.push(datum);
            });
            return datums;
        }
        /**
         * Process single row of server-side data and puts it in more structured form.
         * @param {!Object} metadata Object describing step summary.
         * @private
         * @return {!Object} with wall_time, step number and data for the step.
         */
        _createStepDatum(metadata) {
            return {
                // The wall time within the metadata is in seconds. The Date
                // constructor accepts a time in milliseconds, so we multiply by 1000.
                wall_time: new Date(metadata.wall_time * 1000),
                step: metadata.step,
                config: metadata.config,
                content_type: metadata.content_type,
                components: metadata.components,
            };
        }
    }

    // Polyfills

    if ( Number.EPSILON === undefined ) {

    	Number.EPSILON = Math.pow( 2, - 52 );

    }

    if ( Number.isInteger === undefined ) {

    	// Missing in IE
    	// https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Number/isInteger

    	Number.isInteger = function ( value ) {

    		return typeof value === 'number' && isFinite( value ) && Math.floor( value ) === value;

    	};

    }

    //

    if ( Math.sign === undefined ) {

    	// https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Math/sign

    	Math.sign = function ( x ) {

    		return ( x < 0 ) ? - 1 : ( x > 0 ) ? 1 : + x;

    	};

    }

    if ( 'name' in Function.prototype === false ) {

    	// Missing in IE
    	// https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Function/name

    	Object.defineProperty( Function.prototype, 'name', {

    		get: function () {

    			return this.toString().match( /^\s*function\s*([^\(\s]*)/ )[ 1 ];

    		}

    	} );

    }

    if ( Object.assign === undefined ) {

    	// Missing in IE
    	// https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Object/assign

    	Object.assign = function ( target ) {

    		if ( target === undefined || target === null ) {

    			throw new TypeError( 'Cannot convert undefined or null to object' );

    		}

    		var output = Object( target );

    		for ( var index = 1; index < arguments.length; index ++ ) {

    			var source = arguments[ index ];

    			if ( source !== undefined && source !== null ) {

    				for ( var nextKey in source ) {

    					if ( Object.prototype.hasOwnProperty.call( source, nextKey ) ) {

    						output[ nextKey ] = source[ nextKey ];

    					}

    				}

    			}

    		}

    		return output;

    	};

    }

    /**
     * https://github.com/mrdoob/eventdispatcher.js/
     */

    function EventDispatcher() {}

    Object.assign( EventDispatcher.prototype, {

    	addEventListener: function ( type, listener ) {

    		if ( this._listeners === undefined ) this._listeners = {};

    		var listeners = this._listeners;

    		if ( listeners[ type ] === undefined ) {

    			listeners[ type ] = [];

    		}

    		if ( listeners[ type ].indexOf( listener ) === - 1 ) {

    			listeners[ type ].push( listener );

    		}

    	},

    	hasEventListener: function ( type, listener ) {

    		if ( this._listeners === undefined ) return false;

    		var listeners = this._listeners;

    		return listeners[ type ] !== undefined && listeners[ type ].indexOf( listener ) !== - 1;

    	},

    	removeEventListener: function ( type, listener ) {

    		if ( this._listeners === undefined ) return;

    		var listeners = this._listeners;
    		var listenerArray = listeners[ type ];

    		if ( listenerArray !== undefined ) {

    			var index = listenerArray.indexOf( listener );

    			if ( index !== - 1 ) {

    				listenerArray.splice( index, 1 );

    			}

    		}

    	},

    	dispatchEvent: function ( event ) {

    		if ( this._listeners === undefined ) return;

    		var listeners = this._listeners;
    		var listenerArray = listeners[ event.type ];

    		if ( listenerArray !== undefined ) {

    			event.target = this;

    			var array = listenerArray.slice( 0 );

    			for ( var i = 0, l = array.length; i < l; i ++ ) {

    				array[ i ].call( this, event );

    			}

    		}

    	}

    } );

    var REVISION = '108';
    var MOUSE = { LEFT: 0, MIDDLE: 1, RIGHT: 2, ROTATE: 0, DOLLY: 1, PAN: 2 };
    var TOUCH = { ROTATE: 0, PAN: 1, DOLLY_PAN: 2, DOLLY_ROTATE: 3 };
    var CullFaceNone = 0;
    var CullFaceBack = 1;
    var CullFaceFront = 2;
    var CullFaceFrontBack = 3;
    var FrontFaceDirectionCW = 0;
    var FrontFaceDirectionCCW = 1;
    var BasicShadowMap = 0;
    var PCFShadowMap = 1;
    var PCFSoftShadowMap = 2;
    var VSMShadowMap = 3;
    var FrontSide = 0;
    var BackSide = 1;
    var DoubleSide = 2;
    var FlatShading = 1;
    var SmoothShading = 2;
    var NoColors = 0;
    var FaceColors = 1;
    var VertexColors = 2;
    var NoBlending = 0;
    var NormalBlending = 1;
    var AdditiveBlending = 2;
    var SubtractiveBlending = 3;
    var MultiplyBlending = 4;
    var CustomBlending = 5;
    var AddEquation = 100;
    var SubtractEquation = 101;
    var ReverseSubtractEquation = 102;
    var MinEquation = 103;
    var MaxEquation = 104;
    var ZeroFactor = 200;
    var OneFactor = 201;
    var SrcColorFactor = 202;
    var OneMinusSrcColorFactor = 203;
    var SrcAlphaFactor = 204;
    var OneMinusSrcAlphaFactor = 205;
    var DstAlphaFactor = 206;
    var OneMinusDstAlphaFactor = 207;
    var DstColorFactor = 208;
    var OneMinusDstColorFactor = 209;
    var SrcAlphaSaturateFactor = 210;
    var NeverDepth = 0;
    var AlwaysDepth = 1;
    var LessDepth = 2;
    var LessEqualDepth = 3;
    var EqualDepth = 4;
    var GreaterEqualDepth = 5;
    var GreaterDepth = 6;
    var NotEqualDepth = 7;
    var MultiplyOperation = 0;
    var MixOperation = 1;
    var AddOperation = 2;
    var NoToneMapping = 0;
    var LinearToneMapping = 1;
    var ReinhardToneMapping = 2;
    var Uncharted2ToneMapping = 3;
    var CineonToneMapping = 4;
    var ACESFilmicToneMapping = 5;

    var UVMapping = 300;
    var CubeReflectionMapping = 301;
    var CubeRefractionMapping = 302;
    var EquirectangularReflectionMapping = 303;
    var EquirectangularRefractionMapping = 304;
    var SphericalReflectionMapping = 305;
    var CubeUVReflectionMapping = 306;
    var CubeUVRefractionMapping = 307;
    var RepeatWrapping = 1000;
    var ClampToEdgeWrapping = 1001;
    var MirroredRepeatWrapping = 1002;
    var NearestFilter = 1003;
    var NearestMipmapNearestFilter = 1004;
    var NearestMipMapNearestFilter = 1004;
    var NearestMipmapLinearFilter = 1005;
    var NearestMipMapLinearFilter = 1005;
    var LinearFilter = 1006;
    var LinearMipmapNearestFilter = 1007;
    var LinearMipMapNearestFilter = 1007;
    var LinearMipmapLinearFilter = 1008;
    var LinearMipMapLinearFilter = 1008;
    var UnsignedByteType = 1009;
    var ByteType = 1010;
    var ShortType = 1011;
    var UnsignedShortType = 1012;
    var IntType = 1013;
    var UnsignedIntType = 1014;
    var FloatType = 1015;
    var HalfFloatType = 1016;
    var UnsignedShort4444Type = 1017;
    var UnsignedShort5551Type = 1018;
    var UnsignedShort565Type = 1019;
    var UnsignedInt248Type = 1020;
    var AlphaFormat = 1021;
    var RGBFormat = 1022;
    var RGBAFormat = 1023;
    var LuminanceFormat = 1024;
    var LuminanceAlphaFormat = 1025;
    var RGBEFormat = RGBAFormat;
    var DepthFormat = 1026;
    var DepthStencilFormat = 1027;
    var RedFormat = 1028;
    var RGB_S3TC_DXT1_Format = 33776;
    var RGBA_S3TC_DXT1_Format = 33777;
    var RGBA_S3TC_DXT3_Format = 33778;
    var RGBA_S3TC_DXT5_Format = 33779;
    var RGB_PVRTC_4BPPV1_Format = 35840;
    var RGB_PVRTC_2BPPV1_Format = 35841;
    var RGBA_PVRTC_4BPPV1_Format = 35842;
    var RGBA_PVRTC_2BPPV1_Format = 35843;
    var RGB_ETC1_Format = 36196;
    var RGBA_ASTC_4x4_Format = 37808;
    var RGBA_ASTC_5x4_Format = 37809;
    var RGBA_ASTC_5x5_Format = 37810;
    var RGBA_ASTC_6x5_Format = 37811;
    var RGBA_ASTC_6x6_Format = 37812;
    var RGBA_ASTC_8x5_Format = 37813;
    var RGBA_ASTC_8x6_Format = 37814;
    var RGBA_ASTC_8x8_Format = 37815;
    var RGBA_ASTC_10x5_Format = 37816;
    var RGBA_ASTC_10x6_Format = 37817;
    var RGBA_ASTC_10x8_Format = 37818;
    var RGBA_ASTC_10x10_Format = 37819;
    var RGBA_ASTC_12x10_Format = 37820;
    var RGBA_ASTC_12x12_Format = 37821;
    var LoopOnce = 2200;
    var LoopRepeat = 2201;
    var LoopPingPong = 2202;
    var InterpolateDiscrete = 2300;
    var InterpolateLinear = 2301;
    var InterpolateSmooth = 2302;
    var ZeroCurvatureEnding = 2400;
    var ZeroSlopeEnding = 2401;
    var WrapAroundEnding = 2402;
    var TrianglesDrawMode = 0;
    var TriangleStripDrawMode = 1;
    var TriangleFanDrawMode = 2;
    var LinearEncoding = 3000;
    var sRGBEncoding = 3001;
    var GammaEncoding = 3007;
    var RGBEEncoding = 3002;
    var LogLuvEncoding = 3003;
    var RGBM7Encoding = 3004;
    var RGBM16Encoding = 3005;
    var RGBDEncoding = 3006;
    var BasicDepthPacking = 3200;
    var RGBADepthPacking = 3201;
    var TangentSpaceNormalMap = 0;
    var ObjectSpaceNormalMap = 1;

    var ZeroStencilOp = 0;
    var KeepStencilOp = 7680;
    var ReplaceStencilOp = 7681;
    var IncrementStencilOp = 7682;
    var DecrementStencilOp = 7683;
    var IncrementWrapStencilOp = 34055;
    var DecrementWrapStencilOp = 34056;
    var InvertStencilOp = 5386;

    var NeverStencilFunc = 512;
    var LessStencilFunc = 513;
    var EqualStencilFunc = 514;
    var LessEqualStencilFunc = 515;
    var GreaterStencilFunc = 516;
    var NotEqualStencilFunc = 517;
    var GreaterEqualStencilFunc = 518;
    var AlwaysStencilFunc = 519;

    /**
     * @author alteredq / http://alteredqualia.com/
     * @author mrdoob / http://mrdoob.com/
     */

    var _lut = [];

    for ( var i = 0; i < 256; i ++ ) {

    	_lut[ i ] = ( i < 16 ? '0' : '' ) + ( i ).toString( 16 );

    }

    var _Math = {

    	DEG2RAD: Math.PI / 180,
    	RAD2DEG: 180 / Math.PI,

    	generateUUID: function () {

    		// http://stackoverflow.com/questions/105034/how-to-create-a-guid-uuid-in-javascript/21963136#21963136

    		var d0 = Math.random() * 0xffffffff | 0;
    		var d1 = Math.random() * 0xffffffff | 0;
    		var d2 = Math.random() * 0xffffffff | 0;
    		var d3 = Math.random() * 0xffffffff | 0;
    		var uuid = _lut[ d0 & 0xff ] + _lut[ d0 >> 8 & 0xff ] + _lut[ d0 >> 16 & 0xff ] + _lut[ d0 >> 24 & 0xff ] + '-' +
    			_lut[ d1 & 0xff ] + _lut[ d1 >> 8 & 0xff ] + '-' + _lut[ d1 >> 16 & 0x0f | 0x40 ] + _lut[ d1 >> 24 & 0xff ] + '-' +
    			_lut[ d2 & 0x3f | 0x80 ] + _lut[ d2 >> 8 & 0xff ] + '-' + _lut[ d2 >> 16 & 0xff ] + _lut[ d2 >> 24 & 0xff ] +
    			_lut[ d3 & 0xff ] + _lut[ d3 >> 8 & 0xff ] + _lut[ d3 >> 16 & 0xff ] + _lut[ d3 >> 24 & 0xff ];

    		// .toUpperCase() here flattens concatenated strings to save heap memory space.
    		return uuid.toUpperCase();

    	},

    	clamp: function ( value, min, max ) {

    		return Math.max( min, Math.min( max, value ) );

    	},

    	// compute euclidian modulo of m % n
    	// https://en.wikipedia.org/wiki/Modulo_operation

    	euclideanModulo: function ( n, m ) {

    		return ( ( n % m ) + m ) % m;

    	},

    	// Linear mapping from range <a1, a2> to range <b1, b2>

    	mapLinear: function ( x, a1, a2, b1, b2 ) {

    		return b1 + ( x - a1 ) * ( b2 - b1 ) / ( a2 - a1 );

    	},

    	// https://en.wikipedia.org/wiki/Linear_interpolation

    	lerp: function ( x, y, t ) {

    		return ( 1 - t ) * x + t * y;

    	},

    	// http://en.wikipedia.org/wiki/Smoothstep

    	smoothstep: function ( x, min, max ) {

    		if ( x <= min ) return 0;
    		if ( x >= max ) return 1;

    		x = ( x - min ) / ( max - min );

    		return x * x * ( 3 - 2 * x );

    	},

    	smootherstep: function ( x, min, max ) {

    		if ( x <= min ) return 0;
    		if ( x >= max ) return 1;

    		x = ( x - min ) / ( max - min );

    		return x * x * x * ( x * ( x * 6 - 15 ) + 10 );

    	},

    	// Random integer from <low, high> interval

    	randInt: function ( low, high ) {

    		return low + Math.floor( Math.random() * ( high - low + 1 ) );

    	},

    	// Random float from <low, high> interval

    	randFloat: function ( low, high ) {

    		return low + Math.random() * ( high - low );

    	},

    	// Random float from <-range/2, range/2> interval

    	randFloatSpread: function ( range ) {

    		return range * ( 0.5 - Math.random() );

    	},

    	degToRad: function ( degrees ) {

    		return degrees * _Math.DEG2RAD;

    	},

    	radToDeg: function ( radians ) {

    		return radians * _Math.RAD2DEG;

    	},

    	isPowerOfTwo: function ( value ) {

    		return ( value & ( value - 1 ) ) === 0 && value !== 0;

    	},

    	ceilPowerOfTwo: function ( value ) {

    		return Math.pow( 2, Math.ceil( Math.log( value ) / Math.LN2 ) );

    	},

    	floorPowerOfTwo: function ( value ) {

    		return Math.pow( 2, Math.floor( Math.log( value ) / Math.LN2 ) );

    	}

    };

    /**
     * @author mrdoob / http://mrdoob.com/
     * @author philogb / http://blog.thejit.org/
     * @author egraether / http://egraether.com/
     * @author zz85 / http://www.lab4games.net/zz85/blog
     */

    function Vector2( x, y ) {

    	this.x = x || 0;
    	this.y = y || 0;

    }

    Object.defineProperties( Vector2.prototype, {

    	"width": {

    		get: function () {

    			return this.x;

    		},

    		set: function ( value ) {

    			this.x = value;

    		}

    	},

    	"height": {

    		get: function () {

    			return this.y;

    		},

    		set: function ( value ) {

    			this.y = value;

    		}

    	}

    } );

    Object.assign( Vector2.prototype, {

    	isVector2: true,

    	set: function ( x, y ) {

    		this.x = x;
    		this.y = y;

    		return this;

    	},

    	setScalar: function ( scalar ) {

    		this.x = scalar;
    		this.y = scalar;

    		return this;

    	},

    	setX: function ( x ) {

    		this.x = x;

    		return this;

    	},

    	setY: function ( y ) {

    		this.y = y;

    		return this;

    	},

    	setComponent: function ( index, value ) {

    		switch ( index ) {

    			case 0: this.x = value; break;
    			case 1: this.y = value; break;
    			default: throw new Error( 'index is out of range: ' + index );

    		}

    		return this;

    	},

    	getComponent: function ( index ) {

    		switch ( index ) {

    			case 0: return this.x;
    			case 1: return this.y;
    			default: throw new Error( 'index is out of range: ' + index );

    		}

    	},

    	clone: function () {

    		return new this.constructor( this.x, this.y );

    	},

    	copy: function ( v ) {

    		this.x = v.x;
    		this.y = v.y;

    		return this;

    	},

    	add: function ( v, w ) {

    		if ( w !== undefined ) {

    			console.warn( 'THREE.Vector2: .add() now only accepts one argument. Use .addVectors( a, b ) instead.' );
    			return this.addVectors( v, w );

    		}

    		this.x += v.x;
    		this.y += v.y;

    		return this;

    	},

    	addScalar: function ( s ) {

    		this.x += s;
    		this.y += s;

    		return this;

    	},

    	addVectors: function ( a, b ) {

    		this.x = a.x + b.x;
    		this.y = a.y + b.y;

    		return this;

    	},

    	addScaledVector: function ( v, s ) {

    		this.x += v.x * s;
    		this.y += v.y * s;

    		return this;

    	},

    	sub: function ( v, w ) {

    		if ( w !== undefined ) {

    			console.warn( 'THREE.Vector2: .sub() now only accepts one argument. Use .subVectors( a, b ) instead.' );
    			return this.subVectors( v, w );

    		}

    		this.x -= v.x;
    		this.y -= v.y;

    		return this;

    	},

    	subScalar: function ( s ) {

    		this.x -= s;
    		this.y -= s;

    		return this;

    	},

    	subVectors: function ( a, b ) {

    		this.x = a.x - b.x;
    		this.y = a.y - b.y;

    		return this;

    	},

    	multiply: function ( v ) {

    		this.x *= v.x;
    		this.y *= v.y;

    		return this;

    	},

    	multiplyScalar: function ( scalar ) {

    		this.x *= scalar;
    		this.y *= scalar;

    		return this;

    	},

    	divide: function ( v ) {

    		this.x /= v.x;
    		this.y /= v.y;

    		return this;

    	},

    	divideScalar: function ( scalar ) {

    		return this.multiplyScalar( 1 / scalar );

    	},

    	applyMatrix3: function ( m ) {

    		var x = this.x, y = this.y;
    		var e = m.elements;

    		this.x = e[ 0 ] * x + e[ 3 ] * y + e[ 6 ];
    		this.y = e[ 1 ] * x + e[ 4 ] * y + e[ 7 ];

    		return this;

    	},

    	min: function ( v ) {

    		this.x = Math.min( this.x, v.x );
    		this.y = Math.min( this.y, v.y );

    		return this;

    	},

    	max: function ( v ) {

    		this.x = Math.max( this.x, v.x );
    		this.y = Math.max( this.y, v.y );

    		return this;

    	},

    	clamp: function ( min, max ) {

    		// assumes min < max, componentwise

    		this.x = Math.max( min.x, Math.min( max.x, this.x ) );
    		this.y = Math.max( min.y, Math.min( max.y, this.y ) );

    		return this;

    	},

    	clampScalar: function ( minVal, maxVal ) {

    		this.x = Math.max( minVal, Math.min( maxVal, this.x ) );
    		this.y = Math.max( minVal, Math.min( maxVal, this.y ) );

    		return this;

    	},

    	clampLength: function ( min, max ) {

    		var length = this.length();

    		return this.divideScalar( length || 1 ).multiplyScalar( Math.max( min, Math.min( max, length ) ) );

    	},

    	floor: function () {

    		this.x = Math.floor( this.x );
    		this.y = Math.floor( this.y );

    		return this;

    	},

    	ceil: function () {

    		this.x = Math.ceil( this.x );
    		this.y = Math.ceil( this.y );

    		return this;

    	},

    	round: function () {

    		this.x = Math.round( this.x );
    		this.y = Math.round( this.y );

    		return this;

    	},

    	roundToZero: function () {

    		this.x = ( this.x < 0 ) ? Math.ceil( this.x ) : Math.floor( this.x );
    		this.y = ( this.y < 0 ) ? Math.ceil( this.y ) : Math.floor( this.y );

    		return this;

    	},

    	negate: function () {

    		this.x = - this.x;
    		this.y = - this.y;

    		return this;

    	},

    	dot: function ( v ) {

    		return this.x * v.x + this.y * v.y;

    	},

    	cross: function ( v ) {

    		return this.x * v.y - this.y * v.x;

    	},

    	lengthSq: function () {

    		return this.x * this.x + this.y * this.y;

    	},

    	length: function () {

    		return Math.sqrt( this.x * this.x + this.y * this.y );

    	},

    	manhattanLength: function () {

    		return Math.abs( this.x ) + Math.abs( this.y );

    	},

    	normalize: function () {

    		return this.divideScalar( this.length() || 1 );

    	},

    	angle: function () {

    		// computes the angle in radians with respect to the positive x-axis

    		var angle = Math.atan2( this.y, this.x );

    		if ( angle < 0 ) angle += 2 * Math.PI;

    		return angle;

    	},

    	distanceTo: function ( v ) {

    		return Math.sqrt( this.distanceToSquared( v ) );

    	},

    	distanceToSquared: function ( v ) {

    		var dx = this.x - v.x, dy = this.y - v.y;
    		return dx * dx + dy * dy;

    	},

    	manhattanDistanceTo: function ( v ) {

    		return Math.abs( this.x - v.x ) + Math.abs( this.y - v.y );

    	},

    	setLength: function ( length ) {

    		return this.normalize().multiplyScalar( length );

    	},

    	lerp: function ( v, alpha ) {

    		this.x += ( v.x - this.x ) * alpha;
    		this.y += ( v.y - this.y ) * alpha;

    		return this;

    	},

    	lerpVectors: function ( v1, v2, alpha ) {

    		return this.subVectors( v2, v1 ).multiplyScalar( alpha ).add( v1 );

    	},

    	equals: function ( v ) {

    		return ( ( v.x === this.x ) && ( v.y === this.y ) );

    	},

    	fromArray: function ( array, offset ) {

    		if ( offset === undefined ) offset = 0;

    		this.x = array[ offset ];
    		this.y = array[ offset + 1 ];

    		return this;

    	},

    	toArray: function ( array, offset ) {

    		if ( array === undefined ) array = [];
    		if ( offset === undefined ) offset = 0;

    		array[ offset ] = this.x;
    		array[ offset + 1 ] = this.y;

    		return array;

    	},

    	fromBufferAttribute: function ( attribute, index, offset ) {

    		if ( offset !== undefined ) {

    			console.warn( 'THREE.Vector2: offset has been removed from .fromBufferAttribute().' );

    		}

    		this.x = attribute.getX( index );
    		this.y = attribute.getY( index );

    		return this;

    	},

    	rotateAround: function ( center, angle ) {

    		var c = Math.cos( angle ), s = Math.sin( angle );

    		var x = this.x - center.x;
    		var y = this.y - center.y;

    		this.x = x * c - y * s + center.x;
    		this.y = x * s + y * c + center.y;

    		return this;

    	}

    } );

    /**
     * @author mikael emtinger / http://gomo.se/
     * @author alteredq / http://alteredqualia.com/
     * @author WestLangley / http://github.com/WestLangley
     * @author bhouston / http://clara.io
     */

    function Quaternion( x, y, z, w ) {

    	this._x = x || 0;
    	this._y = y || 0;
    	this._z = z || 0;
    	this._w = ( w !== undefined ) ? w : 1;

    }

    Object.assign( Quaternion, {

    	slerp: function ( qa, qb, qm, t ) {

    		return qm.copy( qa ).slerp( qb, t );

    	},

    	slerpFlat: function ( dst, dstOffset, src0, srcOffset0, src1, srcOffset1, t ) {

    		// fuzz-free, array-based Quaternion SLERP operation

    		var x0 = src0[ srcOffset0 + 0 ],
    			y0 = src0[ srcOffset0 + 1 ],
    			z0 = src0[ srcOffset0 + 2 ],
    			w0 = src0[ srcOffset0 + 3 ],

    			x1 = src1[ srcOffset1 + 0 ],
    			y1 = src1[ srcOffset1 + 1 ],
    			z1 = src1[ srcOffset1 + 2 ],
    			w1 = src1[ srcOffset1 + 3 ];

    		if ( w0 !== w1 || x0 !== x1 || y0 !== y1 || z0 !== z1 ) {

    			var s = 1 - t,

    				cos = x0 * x1 + y0 * y1 + z0 * z1 + w0 * w1,

    				dir = ( cos >= 0 ? 1 : - 1 ),
    				sqrSin = 1 - cos * cos;

    			// Skip the Slerp for tiny steps to avoid numeric problems:
    			if ( sqrSin > Number.EPSILON ) {

    				var sin = Math.sqrt( sqrSin ),
    					len = Math.atan2( sin, cos * dir );

    				s = Math.sin( s * len ) / sin;
    				t = Math.sin( t * len ) / sin;

    			}

    			var tDir = t * dir;

    			x0 = x0 * s + x1 * tDir;
    			y0 = y0 * s + y1 * tDir;
    			z0 = z0 * s + z1 * tDir;
    			w0 = w0 * s + w1 * tDir;

    			// Normalize in case we just did a lerp:
    			if ( s === 1 - t ) {

    				var f = 1 / Math.sqrt( x0 * x0 + y0 * y0 + z0 * z0 + w0 * w0 );

    				x0 *= f;
    				y0 *= f;
    				z0 *= f;
    				w0 *= f;

    			}

    		}

    		dst[ dstOffset ] = x0;
    		dst[ dstOffset + 1 ] = y0;
    		dst[ dstOffset + 2 ] = z0;
    		dst[ dstOffset + 3 ] = w0;

    	}

    } );

    Object.defineProperties( Quaternion.prototype, {

    	x: {

    		get: function () {

    			return this._x;

    		},

    		set: function ( value ) {

    			this._x = value;
    			this._onChangeCallback();

    		}

    	},

    	y: {

    		get: function () {

    			return this._y;

    		},

    		set: function ( value ) {

    			this._y = value;
    			this._onChangeCallback();

    		}

    	},

    	z: {

    		get: function () {

    			return this._z;

    		},

    		set: function ( value ) {

    			this._z = value;
    			this._onChangeCallback();

    		}

    	},

    	w: {

    		get: function () {

    			return this._w;

    		},

    		set: function ( value ) {

    			this._w = value;
    			this._onChangeCallback();

    		}

    	}

    } );

    Object.assign( Quaternion.prototype, {

    	isQuaternion: true,

    	set: function ( x, y, z, w ) {

    		this._x = x;
    		this._y = y;
    		this._z = z;
    		this._w = w;

    		this._onChangeCallback();

    		return this;

    	},

    	clone: function () {

    		return new this.constructor( this._x, this._y, this._z, this._w );

    	},

    	copy: function ( quaternion ) {

    		this._x = quaternion.x;
    		this._y = quaternion.y;
    		this._z = quaternion.z;
    		this._w = quaternion.w;

    		this._onChangeCallback();

    		return this;

    	},

    	setFromEuler: function ( euler, update ) {

    		if ( ! ( euler && euler.isEuler ) ) {

    			throw new Error( 'THREE.Quaternion: .setFromEuler() now expects an Euler rotation rather than a Vector3 and order.' );

    		}

    		var x = euler._x, y = euler._y, z = euler._z, order = euler.order;

    		// http://www.mathworks.com/matlabcentral/fileexchange/
    		// 	20696-function-to-convert-between-dcm-euler-angles-quaternions-and-euler-vectors/
    		//	content/SpinCalc.m

    		var cos = Math.cos;
    		var sin = Math.sin;

    		var c1 = cos( x / 2 );
    		var c2 = cos( y / 2 );
    		var c3 = cos( z / 2 );

    		var s1 = sin( x / 2 );
    		var s2 = sin( y / 2 );
    		var s3 = sin( z / 2 );

    		if ( order === 'XYZ' ) {

    			this._x = s1 * c2 * c3 + c1 * s2 * s3;
    			this._y = c1 * s2 * c3 - s1 * c2 * s3;
    			this._z = c1 * c2 * s3 + s1 * s2 * c3;
    			this._w = c1 * c2 * c3 - s1 * s2 * s3;

    		} else if ( order === 'YXZ' ) {

    			this._x = s1 * c2 * c3 + c1 * s2 * s3;
    			this._y = c1 * s2 * c3 - s1 * c2 * s3;
    			this._z = c1 * c2 * s3 - s1 * s2 * c3;
    			this._w = c1 * c2 * c3 + s1 * s2 * s3;

    		} else if ( order === 'ZXY' ) {

    			this._x = s1 * c2 * c3 - c1 * s2 * s3;
    			this._y = c1 * s2 * c3 + s1 * c2 * s3;
    			this._z = c1 * c2 * s3 + s1 * s2 * c3;
    			this._w = c1 * c2 * c3 - s1 * s2 * s3;

    		} else if ( order === 'ZYX' ) {

    			this._x = s1 * c2 * c3 - c1 * s2 * s3;
    			this._y = c1 * s2 * c3 + s1 * c2 * s3;
    			this._z = c1 * c2 * s3 - s1 * s2 * c3;
    			this._w = c1 * c2 * c3 + s1 * s2 * s3;

    		} else if ( order === 'YZX' ) {

    			this._x = s1 * c2 * c3 + c1 * s2 * s3;
    			this._y = c1 * s2 * c3 + s1 * c2 * s3;
    			this._z = c1 * c2 * s3 - s1 * s2 * c3;
    			this._w = c1 * c2 * c3 - s1 * s2 * s3;

    		} else if ( order === 'XZY' ) {

    			this._x = s1 * c2 * c3 - c1 * s2 * s3;
    			this._y = c1 * s2 * c3 - s1 * c2 * s3;
    			this._z = c1 * c2 * s3 + s1 * s2 * c3;
    			this._w = c1 * c2 * c3 + s1 * s2 * s3;

    		}

    		if ( update !== false ) this._onChangeCallback();

    		return this;

    	},

    	setFromAxisAngle: function ( axis, angle ) {

    		// http://www.euclideanspace.com/maths/geometry/rotations/conversions/angleToQuaternion/index.htm

    		// assumes axis is normalized

    		var halfAngle = angle / 2, s = Math.sin( halfAngle );

    		this._x = axis.x * s;
    		this._y = axis.y * s;
    		this._z = axis.z * s;
    		this._w = Math.cos( halfAngle );

    		this._onChangeCallback();

    		return this;

    	},

    	setFromRotationMatrix: function ( m ) {

    		// http://www.euclideanspace.com/maths/geometry/rotations/conversions/matrixToQuaternion/index.htm

    		// assumes the upper 3x3 of m is a pure rotation matrix (i.e, unscaled)

    		var te = m.elements,

    			m11 = te[ 0 ], m12 = te[ 4 ], m13 = te[ 8 ],
    			m21 = te[ 1 ], m22 = te[ 5 ], m23 = te[ 9 ],
    			m31 = te[ 2 ], m32 = te[ 6 ], m33 = te[ 10 ],

    			trace = m11 + m22 + m33,
    			s;

    		if ( trace > 0 ) {

    			s = 0.5 / Math.sqrt( trace + 1.0 );

    			this._w = 0.25 / s;
    			this._x = ( m32 - m23 ) * s;
    			this._y = ( m13 - m31 ) * s;
    			this._z = ( m21 - m12 ) * s;

    		} else if ( m11 > m22 && m11 > m33 ) {

    			s = 2.0 * Math.sqrt( 1.0 + m11 - m22 - m33 );

    			this._w = ( m32 - m23 ) / s;
    			this._x = 0.25 * s;
    			this._y = ( m12 + m21 ) / s;
    			this._z = ( m13 + m31 ) / s;

    		} else if ( m22 > m33 ) {

    			s = 2.0 * Math.sqrt( 1.0 + m22 - m11 - m33 );

    			this._w = ( m13 - m31 ) / s;
    			this._x = ( m12 + m21 ) / s;
    			this._y = 0.25 * s;
    			this._z = ( m23 + m32 ) / s;

    		} else {

    			s = 2.0 * Math.sqrt( 1.0 + m33 - m11 - m22 );

    			this._w = ( m21 - m12 ) / s;
    			this._x = ( m13 + m31 ) / s;
    			this._y = ( m23 + m32 ) / s;
    			this._z = 0.25 * s;

    		}

    		this._onChangeCallback();

    		return this;

    	},

    	setFromUnitVectors: function ( vFrom, vTo ) {

    		// assumes direction vectors vFrom and vTo are normalized

    		var EPS = 0.000001;

    		var r = vFrom.dot( vTo ) + 1;

    		if ( r < EPS ) {

    			r = 0;

    			if ( Math.abs( vFrom.x ) > Math.abs( vFrom.z ) ) {

    				this._x = - vFrom.y;
    				this._y = vFrom.x;
    				this._z = 0;
    				this._w = r;

    			} else {

    				this._x = 0;
    				this._y = - vFrom.z;
    				this._z = vFrom.y;
    				this._w = r;

    			}

    		} else {

    			// crossVectors( vFrom, vTo ); // inlined to avoid cyclic dependency on Vector3

    			this._x = vFrom.y * vTo.z - vFrom.z * vTo.y;
    			this._y = vFrom.z * vTo.x - vFrom.x * vTo.z;
    			this._z = vFrom.x * vTo.y - vFrom.y * vTo.x;
    			this._w = r;

    		}

    		return this.normalize();

    	},

    	angleTo: function ( q ) {

    		return 2 * Math.acos( Math.abs( _Math.clamp( this.dot( q ), - 1, 1 ) ) );

    	},

    	rotateTowards: function ( q, step ) {

    		var angle = this.angleTo( q );

    		if ( angle === 0 ) return this;

    		var t = Math.min( 1, step / angle );

    		this.slerp( q, t );

    		return this;

    	},

    	inverse: function () {

    		// quaternion is assumed to have unit length

    		return this.conjugate();

    	},

    	conjugate: function () {

    		this._x *= - 1;
    		this._y *= - 1;
    		this._z *= - 1;

    		this._onChangeCallback();

    		return this;

    	},

    	dot: function ( v ) {

    		return this._x * v._x + this._y * v._y + this._z * v._z + this._w * v._w;

    	},

    	lengthSq: function () {

    		return this._x * this._x + this._y * this._y + this._z * this._z + this._w * this._w;

    	},

    	length: function () {

    		return Math.sqrt( this._x * this._x + this._y * this._y + this._z * this._z + this._w * this._w );

    	},

    	normalize: function () {

    		var l = this.length();

    		if ( l === 0 ) {

    			this._x = 0;
    			this._y = 0;
    			this._z = 0;
    			this._w = 1;

    		} else {

    			l = 1 / l;

    			this._x = this._x * l;
    			this._y = this._y * l;
    			this._z = this._z * l;
    			this._w = this._w * l;

    		}

    		this._onChangeCallback();

    		return this;

    	},

    	multiply: function ( q, p ) {

    		if ( p !== undefined ) {

    			console.warn( 'THREE.Quaternion: .multiply() now only accepts one argument. Use .multiplyQuaternions( a, b ) instead.' );
    			return this.multiplyQuaternions( q, p );

    		}

    		return this.multiplyQuaternions( this, q );

    	},

    	premultiply: function ( q ) {

    		return this.multiplyQuaternions( q, this );

    	},

    	multiplyQuaternions: function ( a, b ) {

    		// from http://www.euclideanspace.com/maths/algebra/realNormedAlgebra/quaternions/code/index.htm

    		var qax = a._x, qay = a._y, qaz = a._z, qaw = a._w;
    		var qbx = b._x, qby = b._y, qbz = b._z, qbw = b._w;

    		this._x = qax * qbw + qaw * qbx + qay * qbz - qaz * qby;
    		this._y = qay * qbw + qaw * qby + qaz * qbx - qax * qbz;
    		this._z = qaz * qbw + qaw * qbz + qax * qby - qay * qbx;
    		this._w = qaw * qbw - qax * qbx - qay * qby - qaz * qbz;

    		this._onChangeCallback();

    		return this;

    	},

    	slerp: function ( qb, t ) {

    		if ( t === 0 ) return this;
    		if ( t === 1 ) return this.copy( qb );

    		var x = this._x, y = this._y, z = this._z, w = this._w;

    		// http://www.euclideanspace.com/maths/algebra/realNormedAlgebra/quaternions/slerp/

    		var cosHalfTheta = w * qb._w + x * qb._x + y * qb._y + z * qb._z;

    		if ( cosHalfTheta < 0 ) {

    			this._w = - qb._w;
    			this._x = - qb._x;
    			this._y = - qb._y;
    			this._z = - qb._z;

    			cosHalfTheta = - cosHalfTheta;

    		} else {

    			this.copy( qb );

    		}

    		if ( cosHalfTheta >= 1.0 ) {

    			this._w = w;
    			this._x = x;
    			this._y = y;
    			this._z = z;

    			return this;

    		}

    		var sqrSinHalfTheta = 1.0 - cosHalfTheta * cosHalfTheta;

    		if ( sqrSinHalfTheta <= Number.EPSILON ) {

    			var s = 1 - t;
    			this._w = s * w + t * this._w;
    			this._x = s * x + t * this._x;
    			this._y = s * y + t * this._y;
    			this._z = s * z + t * this._z;

    			this.normalize();
    			this._onChangeCallback();

    			return this;

    		}

    		var sinHalfTheta = Math.sqrt( sqrSinHalfTheta );
    		var halfTheta = Math.atan2( sinHalfTheta, cosHalfTheta );
    		var ratioA = Math.sin( ( 1 - t ) * halfTheta ) / sinHalfTheta,
    			ratioB = Math.sin( t * halfTheta ) / sinHalfTheta;

    		this._w = ( w * ratioA + this._w * ratioB );
    		this._x = ( x * ratioA + this._x * ratioB );
    		this._y = ( y * ratioA + this._y * ratioB );
    		this._z = ( z * ratioA + this._z * ratioB );

    		this._onChangeCallback();

    		return this;

    	},

    	equals: function ( quaternion ) {

    		return ( quaternion._x === this._x ) && ( quaternion._y === this._y ) && ( quaternion._z === this._z ) && ( quaternion._w === this._w );

    	},

    	fromArray: function ( array, offset ) {

    		if ( offset === undefined ) offset = 0;

    		this._x = array[ offset ];
    		this._y = array[ offset + 1 ];
    		this._z = array[ offset + 2 ];
    		this._w = array[ offset + 3 ];

    		this._onChangeCallback();

    		return this;

    	},

    	toArray: function ( array, offset ) {

    		if ( array === undefined ) array = [];
    		if ( offset === undefined ) offset = 0;

    		array[ offset ] = this._x;
    		array[ offset + 1 ] = this._y;
    		array[ offset + 2 ] = this._z;
    		array[ offset + 3 ] = this._w;

    		return array;

    	},

    	_onChange: function ( callback ) {

    		this._onChangeCallback = callback;

    		return this;

    	},

    	_onChangeCallback: function () {}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     * @author kile / http://kile.stravaganza.org/
     * @author philogb / http://blog.thejit.org/
     * @author mikael emtinger / http://gomo.se/
     * @author egraether / http://egraether.com/
     * @author WestLangley / http://github.com/WestLangley
     */

    var _vector = new Vector3();
    var _quaternion = new Quaternion();

    function Vector3( x, y, z ) {

    	this.x = x || 0;
    	this.y = y || 0;
    	this.z = z || 0;

    }

    Object.assign( Vector3.prototype, {

    	isVector3: true,

    	set: function ( x, y, z ) {

    		this.x = x;
    		this.y = y;
    		this.z = z;

    		return this;

    	},

    	setScalar: function ( scalar ) {

    		this.x = scalar;
    		this.y = scalar;
    		this.z = scalar;

    		return this;

    	},

    	setX: function ( x ) {

    		this.x = x;

    		return this;

    	},

    	setY: function ( y ) {

    		this.y = y;

    		return this;

    	},

    	setZ: function ( z ) {

    		this.z = z;

    		return this;

    	},

    	setComponent: function ( index, value ) {

    		switch ( index ) {

    			case 0: this.x = value; break;
    			case 1: this.y = value; break;
    			case 2: this.z = value; break;
    			default: throw new Error( 'index is out of range: ' + index );

    		}

    		return this;

    	},

    	getComponent: function ( index ) {

    		switch ( index ) {

    			case 0: return this.x;
    			case 1: return this.y;
    			case 2: return this.z;
    			default: throw new Error( 'index is out of range: ' + index );

    		}

    	},

    	clone: function () {

    		return new this.constructor( this.x, this.y, this.z );

    	},

    	copy: function ( v ) {

    		this.x = v.x;
    		this.y = v.y;
    		this.z = v.z;

    		return this;

    	},

    	add: function ( v, w ) {

    		if ( w !== undefined ) {

    			console.warn( 'THREE.Vector3: .add() now only accepts one argument. Use .addVectors( a, b ) instead.' );
    			return this.addVectors( v, w );

    		}

    		this.x += v.x;
    		this.y += v.y;
    		this.z += v.z;

    		return this;

    	},

    	addScalar: function ( s ) {

    		this.x += s;
    		this.y += s;
    		this.z += s;

    		return this;

    	},

    	addVectors: function ( a, b ) {

    		this.x = a.x + b.x;
    		this.y = a.y + b.y;
    		this.z = a.z + b.z;

    		return this;

    	},

    	addScaledVector: function ( v, s ) {

    		this.x += v.x * s;
    		this.y += v.y * s;
    		this.z += v.z * s;

    		return this;

    	},

    	sub: function ( v, w ) {

    		if ( w !== undefined ) {

    			console.warn( 'THREE.Vector3: .sub() now only accepts one argument. Use .subVectors( a, b ) instead.' );
    			return this.subVectors( v, w );

    		}

    		this.x -= v.x;
    		this.y -= v.y;
    		this.z -= v.z;

    		return this;

    	},

    	subScalar: function ( s ) {

    		this.x -= s;
    		this.y -= s;
    		this.z -= s;

    		return this;

    	},

    	subVectors: function ( a, b ) {

    		this.x = a.x - b.x;
    		this.y = a.y - b.y;
    		this.z = a.z - b.z;

    		return this;

    	},

    	multiply: function ( v, w ) {

    		if ( w !== undefined ) {

    			console.warn( 'THREE.Vector3: .multiply() now only accepts one argument. Use .multiplyVectors( a, b ) instead.' );
    			return this.multiplyVectors( v, w );

    		}

    		this.x *= v.x;
    		this.y *= v.y;
    		this.z *= v.z;

    		return this;

    	},

    	multiplyScalar: function ( scalar ) {

    		this.x *= scalar;
    		this.y *= scalar;
    		this.z *= scalar;

    		return this;

    	},

    	multiplyVectors: function ( a, b ) {

    		this.x = a.x * b.x;
    		this.y = a.y * b.y;
    		this.z = a.z * b.z;

    		return this;

    	},

    	applyEuler: function ( euler ) {

    		if ( ! ( euler && euler.isEuler ) ) {

    			console.error( 'THREE.Vector3: .applyEuler() now expects an Euler rotation rather than a Vector3 and order.' );

    		}

    		return this.applyQuaternion( _quaternion.setFromEuler( euler ) );

    	},

    	applyAxisAngle: function ( axis, angle ) {

    		return this.applyQuaternion( _quaternion.setFromAxisAngle( axis, angle ) );

    	},

    	applyMatrix3: function ( m ) {

    		var x = this.x, y = this.y, z = this.z;
    		var e = m.elements;

    		this.x = e[ 0 ] * x + e[ 3 ] * y + e[ 6 ] * z;
    		this.y = e[ 1 ] * x + e[ 4 ] * y + e[ 7 ] * z;
    		this.z = e[ 2 ] * x + e[ 5 ] * y + e[ 8 ] * z;

    		return this;

    	},

    	applyMatrix4: function ( m ) {

    		var x = this.x, y = this.y, z = this.z;
    		var e = m.elements;

    		var w = 1 / ( e[ 3 ] * x + e[ 7 ] * y + e[ 11 ] * z + e[ 15 ] );

    		this.x = ( e[ 0 ] * x + e[ 4 ] * y + e[ 8 ] * z + e[ 12 ] ) * w;
    		this.y = ( e[ 1 ] * x + e[ 5 ] * y + e[ 9 ] * z + e[ 13 ] ) * w;
    		this.z = ( e[ 2 ] * x + e[ 6 ] * y + e[ 10 ] * z + e[ 14 ] ) * w;

    		return this;

    	},

    	applyQuaternion: function ( q ) {

    		var x = this.x, y = this.y, z = this.z;
    		var qx = q.x, qy = q.y, qz = q.z, qw = q.w;

    		// calculate quat * vector

    		var ix = qw * x + qy * z - qz * y;
    		var iy = qw * y + qz * x - qx * z;
    		var iz = qw * z + qx * y - qy * x;
    		var iw = - qx * x - qy * y - qz * z;

    		// calculate result * inverse quat

    		this.x = ix * qw + iw * - qx + iy * - qz - iz * - qy;
    		this.y = iy * qw + iw * - qy + iz * - qx - ix * - qz;
    		this.z = iz * qw + iw * - qz + ix * - qy - iy * - qx;

    		return this;

    	},

    	project: function ( camera ) {

    		return this.applyMatrix4( camera.matrixWorldInverse ).applyMatrix4( camera.projectionMatrix );

    	},

    	unproject: function ( camera ) {

    		return this.applyMatrix4( camera.projectionMatrixInverse ).applyMatrix4( camera.matrixWorld );

    	},

    	transformDirection: function ( m ) {

    		// input: THREE.Matrix4 affine matrix
    		// vector interpreted as a direction

    		var x = this.x, y = this.y, z = this.z;
    		var e = m.elements;

    		this.x = e[ 0 ] * x + e[ 4 ] * y + e[ 8 ] * z;
    		this.y = e[ 1 ] * x + e[ 5 ] * y + e[ 9 ] * z;
    		this.z = e[ 2 ] * x + e[ 6 ] * y + e[ 10 ] * z;

    		return this.normalize();

    	},

    	divide: function ( v ) {

    		this.x /= v.x;
    		this.y /= v.y;
    		this.z /= v.z;

    		return this;

    	},

    	divideScalar: function ( scalar ) {

    		return this.multiplyScalar( 1 / scalar );

    	},

    	min: function ( v ) {

    		this.x = Math.min( this.x, v.x );
    		this.y = Math.min( this.y, v.y );
    		this.z = Math.min( this.z, v.z );

    		return this;

    	},

    	max: function ( v ) {

    		this.x = Math.max( this.x, v.x );
    		this.y = Math.max( this.y, v.y );
    		this.z = Math.max( this.z, v.z );

    		return this;

    	},

    	clamp: function ( min, max ) {

    		// assumes min < max, componentwise

    		this.x = Math.max( min.x, Math.min( max.x, this.x ) );
    		this.y = Math.max( min.y, Math.min( max.y, this.y ) );
    		this.z = Math.max( min.z, Math.min( max.z, this.z ) );

    		return this;

    	},

    	clampScalar: function ( minVal, maxVal ) {

    		this.x = Math.max( minVal, Math.min( maxVal, this.x ) );
    		this.y = Math.max( minVal, Math.min( maxVal, this.y ) );
    		this.z = Math.max( minVal, Math.min( maxVal, this.z ) );

    		return this;

    	},

    	clampLength: function ( min, max ) {

    		var length = this.length();

    		return this.divideScalar( length || 1 ).multiplyScalar( Math.max( min, Math.min( max, length ) ) );

    	},

    	floor: function () {

    		this.x = Math.floor( this.x );
    		this.y = Math.floor( this.y );
    		this.z = Math.floor( this.z );

    		return this;

    	},

    	ceil: function () {

    		this.x = Math.ceil( this.x );
    		this.y = Math.ceil( this.y );
    		this.z = Math.ceil( this.z );

    		return this;

    	},

    	round: function () {

    		this.x = Math.round( this.x );
    		this.y = Math.round( this.y );
    		this.z = Math.round( this.z );

    		return this;

    	},

    	roundToZero: function () {

    		this.x = ( this.x < 0 ) ? Math.ceil( this.x ) : Math.floor( this.x );
    		this.y = ( this.y < 0 ) ? Math.ceil( this.y ) : Math.floor( this.y );
    		this.z = ( this.z < 0 ) ? Math.ceil( this.z ) : Math.floor( this.z );

    		return this;

    	},

    	negate: function () {

    		this.x = - this.x;
    		this.y = - this.y;
    		this.z = - this.z;

    		return this;

    	},

    	dot: function ( v ) {

    		return this.x * v.x + this.y * v.y + this.z * v.z;

    	},

    	// TODO lengthSquared?

    	lengthSq: function () {

    		return this.x * this.x + this.y * this.y + this.z * this.z;

    	},

    	length: function () {

    		return Math.sqrt( this.x * this.x + this.y * this.y + this.z * this.z );

    	},

    	manhattanLength: function () {

    		return Math.abs( this.x ) + Math.abs( this.y ) + Math.abs( this.z );

    	},

    	normalize: function () {

    		return this.divideScalar( this.length() || 1 );

    	},

    	setLength: function ( length ) {

    		return this.normalize().multiplyScalar( length );

    	},

    	lerp: function ( v, alpha ) {

    		this.x += ( v.x - this.x ) * alpha;
    		this.y += ( v.y - this.y ) * alpha;
    		this.z += ( v.z - this.z ) * alpha;

    		return this;

    	},

    	lerpVectors: function ( v1, v2, alpha ) {

    		return this.subVectors( v2, v1 ).multiplyScalar( alpha ).add( v1 );

    	},

    	cross: function ( v, w ) {

    		if ( w !== undefined ) {

    			console.warn( 'THREE.Vector3: .cross() now only accepts one argument. Use .crossVectors( a, b ) instead.' );
    			return this.crossVectors( v, w );

    		}

    		return this.crossVectors( this, v );

    	},

    	crossVectors: function ( a, b ) {

    		var ax = a.x, ay = a.y, az = a.z;
    		var bx = b.x, by = b.y, bz = b.z;

    		this.x = ay * bz - az * by;
    		this.y = az * bx - ax * bz;
    		this.z = ax * by - ay * bx;

    		return this;

    	},

    	projectOnVector: function ( vector ) {

    		var scalar = vector.dot( this ) / vector.lengthSq();

    		return this.copy( vector ).multiplyScalar( scalar );

    	},

    	projectOnPlane: function ( planeNormal ) {

    		_vector.copy( this ).projectOnVector( planeNormal );

    		return this.sub( _vector );

    	},

    	reflect: function ( normal ) {

    		// reflect incident vector off plane orthogonal to normal
    		// normal is assumed to have unit length

    		return this.sub( _vector.copy( normal ).multiplyScalar( 2 * this.dot( normal ) ) );

    	},

    	angleTo: function ( v ) {

    		var theta = this.dot( v ) / ( Math.sqrt( this.lengthSq() * v.lengthSq() ) );

    		// clamp, to handle numerical problems

    		return Math.acos( _Math.clamp( theta, - 1, 1 ) );

    	},

    	distanceTo: function ( v ) {

    		return Math.sqrt( this.distanceToSquared( v ) );

    	},

    	distanceToSquared: function ( v ) {

    		var dx = this.x - v.x, dy = this.y - v.y, dz = this.z - v.z;

    		return dx * dx + dy * dy + dz * dz;

    	},

    	manhattanDistanceTo: function ( v ) {

    		return Math.abs( this.x - v.x ) + Math.abs( this.y - v.y ) + Math.abs( this.z - v.z );

    	},

    	setFromSpherical: function ( s ) {

    		return this.setFromSphericalCoords( s.radius, s.phi, s.theta );

    	},

    	setFromSphericalCoords: function ( radius, phi, theta ) {

    		var sinPhiRadius = Math.sin( phi ) * radius;

    		this.x = sinPhiRadius * Math.sin( theta );
    		this.y = Math.cos( phi ) * radius;
    		this.z = sinPhiRadius * Math.cos( theta );

    		return this;

    	},

    	setFromCylindrical: function ( c ) {

    		return this.setFromCylindricalCoords( c.radius, c.theta, c.y );

    	},

    	setFromCylindricalCoords: function ( radius, theta, y ) {

    		this.x = radius * Math.sin( theta );
    		this.y = y;
    		this.z = radius * Math.cos( theta );

    		return this;

    	},

    	setFromMatrixPosition: function ( m ) {

    		var e = m.elements;

    		this.x = e[ 12 ];
    		this.y = e[ 13 ];
    		this.z = e[ 14 ];

    		return this;

    	},

    	setFromMatrixScale: function ( m ) {

    		var sx = this.setFromMatrixColumn( m, 0 ).length();
    		var sy = this.setFromMatrixColumn( m, 1 ).length();
    		var sz = this.setFromMatrixColumn( m, 2 ).length();

    		this.x = sx;
    		this.y = sy;
    		this.z = sz;

    		return this;

    	},

    	setFromMatrixColumn: function ( m, index ) {

    		return this.fromArray( m.elements, index * 4 );

    	},

    	equals: function ( v ) {

    		return ( ( v.x === this.x ) && ( v.y === this.y ) && ( v.z === this.z ) );

    	},

    	fromArray: function ( array, offset ) {

    		if ( offset === undefined ) offset = 0;

    		this.x = array[ offset ];
    		this.y = array[ offset + 1 ];
    		this.z = array[ offset + 2 ];

    		return this;

    	},

    	toArray: function ( array, offset ) {

    		if ( array === undefined ) array = [];
    		if ( offset === undefined ) offset = 0;

    		array[ offset ] = this.x;
    		array[ offset + 1 ] = this.y;
    		array[ offset + 2 ] = this.z;

    		return array;

    	},

    	fromBufferAttribute: function ( attribute, index, offset ) {

    		if ( offset !== undefined ) {

    			console.warn( 'THREE.Vector3: offset has been removed from .fromBufferAttribute().' );

    		}

    		this.x = attribute.getX( index );
    		this.y = attribute.getY( index );
    		this.z = attribute.getZ( index );

    		return this;

    	}

    } );

    /**
     * @author alteredq / http://alteredqualia.com/
     * @author WestLangley / http://github.com/WestLangley
     * @author bhouston / http://clara.io
     * @author tschw
     */

    var _vector$1 = new Vector3();

    function Matrix3() {

    	this.elements = [

    		1, 0, 0,
    		0, 1, 0,
    		0, 0, 1

    	];

    	if ( arguments.length > 0 ) {

    		console.error( 'THREE.Matrix3: the constructor no longer reads arguments. use .set() instead.' );

    	}

    }

    Object.assign( Matrix3.prototype, {

    	isMatrix3: true,

    	set: function ( n11, n12, n13, n21, n22, n23, n31, n32, n33 ) {

    		var te = this.elements;

    		te[ 0 ] = n11; te[ 1 ] = n21; te[ 2 ] = n31;
    		te[ 3 ] = n12; te[ 4 ] = n22; te[ 5 ] = n32;
    		te[ 6 ] = n13; te[ 7 ] = n23; te[ 8 ] = n33;

    		return this;

    	},

    	identity: function () {

    		this.set(

    			1, 0, 0,
    			0, 1, 0,
    			0, 0, 1

    		);

    		return this;

    	},

    	clone: function () {

    		return new this.constructor().fromArray( this.elements );

    	},

    	copy: function ( m ) {

    		var te = this.elements;
    		var me = m.elements;

    		te[ 0 ] = me[ 0 ]; te[ 1 ] = me[ 1 ]; te[ 2 ] = me[ 2 ];
    		te[ 3 ] = me[ 3 ]; te[ 4 ] = me[ 4 ]; te[ 5 ] = me[ 5 ];
    		te[ 6 ] = me[ 6 ]; te[ 7 ] = me[ 7 ]; te[ 8 ] = me[ 8 ];

    		return this;

    	},

    	setFromMatrix4: function ( m ) {

    		var me = m.elements;

    		this.set(

    			me[ 0 ], me[ 4 ], me[ 8 ],
    			me[ 1 ], me[ 5 ], me[ 9 ],
    			me[ 2 ], me[ 6 ], me[ 10 ]

    		);

    		return this;

    	},

    	applyToBufferAttribute: function ( attribute ) {

    		for ( var i = 0, l = attribute.count; i < l; i ++ ) {

    			_vector$1.x = attribute.getX( i );
    			_vector$1.y = attribute.getY( i );
    			_vector$1.z = attribute.getZ( i );

    			_vector$1.applyMatrix3( this );

    			attribute.setXYZ( i, _vector$1.x, _vector$1.y, _vector$1.z );

    		}

    		return attribute;

    	},

    	multiply: function ( m ) {

    		return this.multiplyMatrices( this, m );

    	},

    	premultiply: function ( m ) {

    		return this.multiplyMatrices( m, this );

    	},

    	multiplyMatrices: function ( a, b ) {

    		var ae = a.elements;
    		var be = b.elements;
    		var te = this.elements;

    		var a11 = ae[ 0 ], a12 = ae[ 3 ], a13 = ae[ 6 ];
    		var a21 = ae[ 1 ], a22 = ae[ 4 ], a23 = ae[ 7 ];
    		var a31 = ae[ 2 ], a32 = ae[ 5 ], a33 = ae[ 8 ];

    		var b11 = be[ 0 ], b12 = be[ 3 ], b13 = be[ 6 ];
    		var b21 = be[ 1 ], b22 = be[ 4 ], b23 = be[ 7 ];
    		var b31 = be[ 2 ], b32 = be[ 5 ], b33 = be[ 8 ];

    		te[ 0 ] = a11 * b11 + a12 * b21 + a13 * b31;
    		te[ 3 ] = a11 * b12 + a12 * b22 + a13 * b32;
    		te[ 6 ] = a11 * b13 + a12 * b23 + a13 * b33;

    		te[ 1 ] = a21 * b11 + a22 * b21 + a23 * b31;
    		te[ 4 ] = a21 * b12 + a22 * b22 + a23 * b32;
    		te[ 7 ] = a21 * b13 + a22 * b23 + a23 * b33;

    		te[ 2 ] = a31 * b11 + a32 * b21 + a33 * b31;
    		te[ 5 ] = a31 * b12 + a32 * b22 + a33 * b32;
    		te[ 8 ] = a31 * b13 + a32 * b23 + a33 * b33;

    		return this;

    	},

    	multiplyScalar: function ( s ) {

    		var te = this.elements;

    		te[ 0 ] *= s; te[ 3 ] *= s; te[ 6 ] *= s;
    		te[ 1 ] *= s; te[ 4 ] *= s; te[ 7 ] *= s;
    		te[ 2 ] *= s; te[ 5 ] *= s; te[ 8 ] *= s;

    		return this;

    	},

    	determinant: function () {

    		var te = this.elements;

    		var a = te[ 0 ], b = te[ 1 ], c = te[ 2 ],
    			d = te[ 3 ], e = te[ 4 ], f = te[ 5 ],
    			g = te[ 6 ], h = te[ 7 ], i = te[ 8 ];

    		return a * e * i - a * f * h - b * d * i + b * f * g + c * d * h - c * e * g;

    	},

    	getInverse: function ( matrix, throwOnDegenerate ) {

    		if ( matrix && matrix.isMatrix4 ) {

    			console.error( "THREE.Matrix3: .getInverse() no longer takes a Matrix4 argument." );

    		}

    		var me = matrix.elements,
    			te = this.elements,

    			n11 = me[ 0 ], n21 = me[ 1 ], n31 = me[ 2 ],
    			n12 = me[ 3 ], n22 = me[ 4 ], n32 = me[ 5 ],
    			n13 = me[ 6 ], n23 = me[ 7 ], n33 = me[ 8 ],

    			t11 = n33 * n22 - n32 * n23,
    			t12 = n32 * n13 - n33 * n12,
    			t13 = n23 * n12 - n22 * n13,

    			det = n11 * t11 + n21 * t12 + n31 * t13;

    		if ( det === 0 ) {

    			var msg = "THREE.Matrix3: .getInverse() can't invert matrix, determinant is 0";

    			if ( throwOnDegenerate === true ) {

    				throw new Error( msg );

    			} else {

    				console.warn( msg );

    			}

    			return this.identity();

    		}

    		var detInv = 1 / det;

    		te[ 0 ] = t11 * detInv;
    		te[ 1 ] = ( n31 * n23 - n33 * n21 ) * detInv;
    		te[ 2 ] = ( n32 * n21 - n31 * n22 ) * detInv;

    		te[ 3 ] = t12 * detInv;
    		te[ 4 ] = ( n33 * n11 - n31 * n13 ) * detInv;
    		te[ 5 ] = ( n31 * n12 - n32 * n11 ) * detInv;

    		te[ 6 ] = t13 * detInv;
    		te[ 7 ] = ( n21 * n13 - n23 * n11 ) * detInv;
    		te[ 8 ] = ( n22 * n11 - n21 * n12 ) * detInv;

    		return this;

    	},

    	transpose: function () {

    		var tmp, m = this.elements;

    		tmp = m[ 1 ]; m[ 1 ] = m[ 3 ]; m[ 3 ] = tmp;
    		tmp = m[ 2 ]; m[ 2 ] = m[ 6 ]; m[ 6 ] = tmp;
    		tmp = m[ 5 ]; m[ 5 ] = m[ 7 ]; m[ 7 ] = tmp;

    		return this;

    	},

    	getNormalMatrix: function ( matrix4 ) {

    		return this.setFromMatrix4( matrix4 ).getInverse( this ).transpose();

    	},

    	transposeIntoArray: function ( r ) {

    		var m = this.elements;

    		r[ 0 ] = m[ 0 ];
    		r[ 1 ] = m[ 3 ];
    		r[ 2 ] = m[ 6 ];
    		r[ 3 ] = m[ 1 ];
    		r[ 4 ] = m[ 4 ];
    		r[ 5 ] = m[ 7 ];
    		r[ 6 ] = m[ 2 ];
    		r[ 7 ] = m[ 5 ];
    		r[ 8 ] = m[ 8 ];

    		return this;

    	},

    	setUvTransform: function ( tx, ty, sx, sy, rotation, cx, cy ) {

    		var c = Math.cos( rotation );
    		var s = Math.sin( rotation );

    		this.set(
    			sx * c, sx * s, - sx * ( c * cx + s * cy ) + cx + tx,
    			- sy * s, sy * c, - sy * ( - s * cx + c * cy ) + cy + ty,
    			0, 0, 1
    		);

    	},

    	scale: function ( sx, sy ) {

    		var te = this.elements;

    		te[ 0 ] *= sx; te[ 3 ] *= sx; te[ 6 ] *= sx;
    		te[ 1 ] *= sy; te[ 4 ] *= sy; te[ 7 ] *= sy;

    		return this;

    	},

    	rotate: function ( theta ) {

    		var c = Math.cos( theta );
    		var s = Math.sin( theta );

    		var te = this.elements;

    		var a11 = te[ 0 ], a12 = te[ 3 ], a13 = te[ 6 ];
    		var a21 = te[ 1 ], a22 = te[ 4 ], a23 = te[ 7 ];

    		te[ 0 ] = c * a11 + s * a21;
    		te[ 3 ] = c * a12 + s * a22;
    		te[ 6 ] = c * a13 + s * a23;

    		te[ 1 ] = - s * a11 + c * a21;
    		te[ 4 ] = - s * a12 + c * a22;
    		te[ 7 ] = - s * a13 + c * a23;

    		return this;

    	},

    	translate: function ( tx, ty ) {

    		var te = this.elements;

    		te[ 0 ] += tx * te[ 2 ]; te[ 3 ] += tx * te[ 5 ]; te[ 6 ] += tx * te[ 8 ];
    		te[ 1 ] += ty * te[ 2 ]; te[ 4 ] += ty * te[ 5 ]; te[ 7 ] += ty * te[ 8 ];

    		return this;

    	},

    	equals: function ( matrix ) {

    		var te = this.elements;
    		var me = matrix.elements;

    		for ( var i = 0; i < 9; i ++ ) {

    			if ( te[ i ] !== me[ i ] ) return false;

    		}

    		return true;

    	},

    	fromArray: function ( array, offset ) {

    		if ( offset === undefined ) offset = 0;

    		for ( var i = 0; i < 9; i ++ ) {

    			this.elements[ i ] = array[ i + offset ];

    		}

    		return this;

    	},

    	toArray: function ( array, offset ) {

    		if ( array === undefined ) array = [];
    		if ( offset === undefined ) offset = 0;

    		var te = this.elements;

    		array[ offset ] = te[ 0 ];
    		array[ offset + 1 ] = te[ 1 ];
    		array[ offset + 2 ] = te[ 2 ];

    		array[ offset + 3 ] = te[ 3 ];
    		array[ offset + 4 ] = te[ 4 ];
    		array[ offset + 5 ] = te[ 5 ];

    		array[ offset + 6 ] = te[ 6 ];
    		array[ offset + 7 ] = te[ 7 ];
    		array[ offset + 8 ] = te[ 8 ];

    		return array;

    	}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     * @author alteredq / http://alteredqualia.com/
     * @author szimek / https://github.com/szimek/
     */

    var _canvas;

    var ImageUtils = {

    	getDataURL: function ( image ) {

    		var canvas;

    		if ( typeof HTMLCanvasElement == 'undefined' ) {

    			return image.src;

    		} else if ( image instanceof HTMLCanvasElement ) {

    			canvas = image;

    		} else {

    			if ( _canvas === undefined ) _canvas = document.createElementNS( 'http://www.w3.org/1999/xhtml', 'canvas' );

    			_canvas.width = image.width;
    			_canvas.height = image.height;

    			var context = _canvas.getContext( '2d' );

    			if ( image instanceof ImageData ) {

    				context.putImageData( image, 0, 0 );

    			} else {

    				context.drawImage( image, 0, 0, image.width, image.height );

    			}

    			canvas = _canvas;

    		}

    		if ( canvas.width > 2048 || canvas.height > 2048 ) {

    			return canvas.toDataURL( 'image/jpeg', 0.6 );

    		} else {

    			return canvas.toDataURL( 'image/png' );

    		}

    	}

    };

    /**
     * @author mrdoob / http://mrdoob.com/
     * @author alteredq / http://alteredqualia.com/
     * @author szimek / https://github.com/szimek/
     */

    var textureId = 0;

    function Texture( image, mapping, wrapS, wrapT, magFilter, minFilter, format, type, anisotropy, encoding ) {

    	Object.defineProperty( this, 'id', { value: textureId ++ } );

    	this.uuid = _Math.generateUUID();

    	this.name = '';

    	this.image = image !== undefined ? image : Texture.DEFAULT_IMAGE;
    	this.mipmaps = [];

    	this.mapping = mapping !== undefined ? mapping : Texture.DEFAULT_MAPPING;

    	this.wrapS = wrapS !== undefined ? wrapS : ClampToEdgeWrapping;
    	this.wrapT = wrapT !== undefined ? wrapT : ClampToEdgeWrapping;

    	this.magFilter = magFilter !== undefined ? magFilter : LinearFilter;
    	this.minFilter = minFilter !== undefined ? minFilter : LinearMipmapLinearFilter;

    	this.anisotropy = anisotropy !== undefined ? anisotropy : 1;

    	this.format = format !== undefined ? format : RGBAFormat;
    	this.type = type !== undefined ? type : UnsignedByteType;

    	this.offset = new Vector2( 0, 0 );
    	this.repeat = new Vector2( 1, 1 );
    	this.center = new Vector2( 0, 0 );
    	this.rotation = 0;

    	this.matrixAutoUpdate = true;
    	this.matrix = new Matrix3();

    	this.generateMipmaps = true;
    	this.premultiplyAlpha = false;
    	this.flipY = true;
    	this.unpackAlignment = 4;	// valid values: 1, 2, 4, 8 (see http://www.khronos.org/opengles/sdk/docs/man/xhtml/glPixelStorei.xml)

    	// Values of encoding !== THREE.LinearEncoding only supported on map, envMap and emissiveMap.
    	//
    	// Also changing the encoding after already used by a Material will not automatically make the Material
    	// update. You need to explicitly call Material.needsUpdate to trigger it to recompile.
    	this.encoding = encoding !== undefined ? encoding : LinearEncoding;

    	this.version = 0;
    	this.onUpdate = null;

    }

    Texture.DEFAULT_IMAGE = undefined;
    Texture.DEFAULT_MAPPING = UVMapping;

    Texture.prototype = Object.assign( Object.create( EventDispatcher.prototype ), {

    	constructor: Texture,

    	isTexture: true,

    	updateMatrix: function () {

    		this.matrix.setUvTransform( this.offset.x, this.offset.y, this.repeat.x, this.repeat.y, this.rotation, this.center.x, this.center.y );

    	},

    	clone: function () {

    		return new this.constructor().copy( this );

    	},

    	copy: function ( source ) {

    		this.name = source.name;

    		this.image = source.image;
    		this.mipmaps = source.mipmaps.slice( 0 );

    		this.mapping = source.mapping;

    		this.wrapS = source.wrapS;
    		this.wrapT = source.wrapT;

    		this.magFilter = source.magFilter;
    		this.minFilter = source.minFilter;

    		this.anisotropy = source.anisotropy;

    		this.format = source.format;
    		this.type = source.type;

    		this.offset.copy( source.offset );
    		this.repeat.copy( source.repeat );
    		this.center.copy( source.center );
    		this.rotation = source.rotation;

    		this.matrixAutoUpdate = source.matrixAutoUpdate;
    		this.matrix.copy( source.matrix );

    		this.generateMipmaps = source.generateMipmaps;
    		this.premultiplyAlpha = source.premultiplyAlpha;
    		this.flipY = source.flipY;
    		this.unpackAlignment = source.unpackAlignment;
    		this.encoding = source.encoding;

    		return this;

    	},

    	toJSON: function ( meta ) {

    		var isRootObject = ( meta === undefined || typeof meta === 'string' );

    		if ( ! isRootObject && meta.textures[ this.uuid ] !== undefined ) {

    			return meta.textures[ this.uuid ];

    		}

    		var output = {

    			metadata: {
    				version: 4.5,
    				type: 'Texture',
    				generator: 'Texture.toJSON'
    			},

    			uuid: this.uuid,
    			name: this.name,

    			mapping: this.mapping,

    			repeat: [ this.repeat.x, this.repeat.y ],
    			offset: [ this.offset.x, this.offset.y ],
    			center: [ this.center.x, this.center.y ],
    			rotation: this.rotation,

    			wrap: [ this.wrapS, this.wrapT ],

    			format: this.format,
    			type: this.type,
    			encoding: this.encoding,

    			minFilter: this.minFilter,
    			magFilter: this.magFilter,
    			anisotropy: this.anisotropy,

    			flipY: this.flipY,

    			premultiplyAlpha: this.premultiplyAlpha,
    			unpackAlignment: this.unpackAlignment

    		};

    		if ( this.image !== undefined ) {

    			// TODO: Move to THREE.Image

    			var image = this.image;

    			if ( image.uuid === undefined ) {

    				image.uuid = _Math.generateUUID(); // UGH

    			}

    			if ( ! isRootObject && meta.images[ image.uuid ] === undefined ) {

    				var url;

    				if ( Array.isArray( image ) ) {

    					// process array of images e.g. CubeTexture

    					url = [];

    					for ( var i = 0, l = image.length; i < l; i ++ ) {

    						url.push( ImageUtils.getDataURL( image[ i ] ) );

    					}

    				} else {

    					// process single image

    					url = ImageUtils.getDataURL( image );

    				}

    				meta.images[ image.uuid ] = {
    					uuid: image.uuid,
    					url: url
    				};

    			}

    			output.image = image.uuid;

    		}

    		if ( ! isRootObject ) {

    			meta.textures[ this.uuid ] = output;

    		}

    		return output;

    	},

    	dispose: function () {

    		this.dispatchEvent( { type: 'dispose' } );

    	},

    	transformUv: function ( uv ) {

    		if ( this.mapping !== UVMapping ) return uv;

    		uv.applyMatrix3( this.matrix );

    		if ( uv.x < 0 || uv.x > 1 ) {

    			switch ( this.wrapS ) {

    				case RepeatWrapping:

    					uv.x = uv.x - Math.floor( uv.x );
    					break;

    				case ClampToEdgeWrapping:

    					uv.x = uv.x < 0 ? 0 : 1;
    					break;

    				case MirroredRepeatWrapping:

    					if ( Math.abs( Math.floor( uv.x ) % 2 ) === 1 ) {

    						uv.x = Math.ceil( uv.x ) - uv.x;

    					} else {

    						uv.x = uv.x - Math.floor( uv.x );

    					}
    					break;

    			}

    		}

    		if ( uv.y < 0 || uv.y > 1 ) {

    			switch ( this.wrapT ) {

    				case RepeatWrapping:

    					uv.y = uv.y - Math.floor( uv.y );
    					break;

    				case ClampToEdgeWrapping:

    					uv.y = uv.y < 0 ? 0 : 1;
    					break;

    				case MirroredRepeatWrapping:

    					if ( Math.abs( Math.floor( uv.y ) % 2 ) === 1 ) {

    						uv.y = Math.ceil( uv.y ) - uv.y;

    					} else {

    						uv.y = uv.y - Math.floor( uv.y );

    					}
    					break;

    			}

    		}

    		if ( this.flipY ) {

    			uv.y = 1 - uv.y;

    		}

    		return uv;

    	}

    } );

    Object.defineProperty( Texture.prototype, "needsUpdate", {

    	set: function ( value ) {

    		if ( value === true ) this.version ++;

    	}

    } );

    /**
     * @author supereggbert / http://www.paulbrunt.co.uk/
     * @author philogb / http://blog.thejit.org/
     * @author mikael emtinger / http://gomo.se/
     * @author egraether / http://egraether.com/
     * @author WestLangley / http://github.com/WestLangley
     */

    function Vector4( x, y, z, w ) {

    	this.x = x || 0;
    	this.y = y || 0;
    	this.z = z || 0;
    	this.w = ( w !== undefined ) ? w : 1;

    }

    Object.defineProperties( Vector4.prototype, {

    	"width": {

    		get: function () {

    			return this.z;

    		},

    		set: function ( value ) {

    			this.z = value;

    		}

    	},

    	"height": {

    		get: function () {

    			return this.w;

    		},

    		set: function ( value ) {

    			this.w = value;

    		}

    	}

    } );

    Object.assign( Vector4.prototype, {

    	isVector4: true,

    	set: function ( x, y, z, w ) {

    		this.x = x;
    		this.y = y;
    		this.z = z;
    		this.w = w;

    		return this;

    	},

    	setScalar: function ( scalar ) {

    		this.x = scalar;
    		this.y = scalar;
    		this.z = scalar;
    		this.w = scalar;

    		return this;

    	},

    	setX: function ( x ) {

    		this.x = x;

    		return this;

    	},

    	setY: function ( y ) {

    		this.y = y;

    		return this;

    	},

    	setZ: function ( z ) {

    		this.z = z;

    		return this;

    	},

    	setW: function ( w ) {

    		this.w = w;

    		return this;

    	},

    	setComponent: function ( index, value ) {

    		switch ( index ) {

    			case 0: this.x = value; break;
    			case 1: this.y = value; break;
    			case 2: this.z = value; break;
    			case 3: this.w = value; break;
    			default: throw new Error( 'index is out of range: ' + index );

    		}

    		return this;

    	},

    	getComponent: function ( index ) {

    		switch ( index ) {

    			case 0: return this.x;
    			case 1: return this.y;
    			case 2: return this.z;
    			case 3: return this.w;
    			default: throw new Error( 'index is out of range: ' + index );

    		}

    	},

    	clone: function () {

    		return new this.constructor( this.x, this.y, this.z, this.w );

    	},

    	copy: function ( v ) {

    		this.x = v.x;
    		this.y = v.y;
    		this.z = v.z;
    		this.w = ( v.w !== undefined ) ? v.w : 1;

    		return this;

    	},

    	add: function ( v, w ) {

    		if ( w !== undefined ) {

    			console.warn( 'THREE.Vector4: .add() now only accepts one argument. Use .addVectors( a, b ) instead.' );
    			return this.addVectors( v, w );

    		}

    		this.x += v.x;
    		this.y += v.y;
    		this.z += v.z;
    		this.w += v.w;

    		return this;

    	},

    	addScalar: function ( s ) {

    		this.x += s;
    		this.y += s;
    		this.z += s;
    		this.w += s;

    		return this;

    	},

    	addVectors: function ( a, b ) {

    		this.x = a.x + b.x;
    		this.y = a.y + b.y;
    		this.z = a.z + b.z;
    		this.w = a.w + b.w;

    		return this;

    	},

    	addScaledVector: function ( v, s ) {

    		this.x += v.x * s;
    		this.y += v.y * s;
    		this.z += v.z * s;
    		this.w += v.w * s;

    		return this;

    	},

    	sub: function ( v, w ) {

    		if ( w !== undefined ) {

    			console.warn( 'THREE.Vector4: .sub() now only accepts one argument. Use .subVectors( a, b ) instead.' );
    			return this.subVectors( v, w );

    		}

    		this.x -= v.x;
    		this.y -= v.y;
    		this.z -= v.z;
    		this.w -= v.w;

    		return this;

    	},

    	subScalar: function ( s ) {

    		this.x -= s;
    		this.y -= s;
    		this.z -= s;
    		this.w -= s;

    		return this;

    	},

    	subVectors: function ( a, b ) {

    		this.x = a.x - b.x;
    		this.y = a.y - b.y;
    		this.z = a.z - b.z;
    		this.w = a.w - b.w;

    		return this;

    	},

    	multiplyScalar: function ( scalar ) {

    		this.x *= scalar;
    		this.y *= scalar;
    		this.z *= scalar;
    		this.w *= scalar;

    		return this;

    	},

    	applyMatrix4: function ( m ) {

    		var x = this.x, y = this.y, z = this.z, w = this.w;
    		var e = m.elements;

    		this.x = e[ 0 ] * x + e[ 4 ] * y + e[ 8 ] * z + e[ 12 ] * w;
    		this.y = e[ 1 ] * x + e[ 5 ] * y + e[ 9 ] * z + e[ 13 ] * w;
    		this.z = e[ 2 ] * x + e[ 6 ] * y + e[ 10 ] * z + e[ 14 ] * w;
    		this.w = e[ 3 ] * x + e[ 7 ] * y + e[ 11 ] * z + e[ 15 ] * w;

    		return this;

    	},

    	divideScalar: function ( scalar ) {

    		return this.multiplyScalar( 1 / scalar );

    	},

    	setAxisAngleFromQuaternion: function ( q ) {

    		// http://www.euclideanspace.com/maths/geometry/rotations/conversions/quaternionToAngle/index.htm

    		// q is assumed to be normalized

    		this.w = 2 * Math.acos( q.w );

    		var s = Math.sqrt( 1 - q.w * q.w );

    		if ( s < 0.0001 ) {

    			this.x = 1;
    			this.y = 0;
    			this.z = 0;

    		} else {

    			this.x = q.x / s;
    			this.y = q.y / s;
    			this.z = q.z / s;

    		}

    		return this;

    	},

    	setAxisAngleFromRotationMatrix: function ( m ) {

    		// http://www.euclideanspace.com/maths/geometry/rotations/conversions/matrixToAngle/index.htm

    		// assumes the upper 3x3 of m is a pure rotation matrix (i.e, unscaled)

    		var angle, x, y, z,		// variables for result
    			epsilon = 0.01,		// margin to allow for rounding errors
    			epsilon2 = 0.1,		// margin to distinguish between 0 and 180 degrees

    			te = m.elements,

    			m11 = te[ 0 ], m12 = te[ 4 ], m13 = te[ 8 ],
    			m21 = te[ 1 ], m22 = te[ 5 ], m23 = te[ 9 ],
    			m31 = te[ 2 ], m32 = te[ 6 ], m33 = te[ 10 ];

    		if ( ( Math.abs( m12 - m21 ) < epsilon ) &&
    		     ( Math.abs( m13 - m31 ) < epsilon ) &&
    		     ( Math.abs( m23 - m32 ) < epsilon ) ) {

    			// singularity found
    			// first check for identity matrix which must have +1 for all terms
    			// in leading diagonal and zero in other terms

    			if ( ( Math.abs( m12 + m21 ) < epsilon2 ) &&
    			     ( Math.abs( m13 + m31 ) < epsilon2 ) &&
    			     ( Math.abs( m23 + m32 ) < epsilon2 ) &&
    			     ( Math.abs( m11 + m22 + m33 - 3 ) < epsilon2 ) ) {

    				// this singularity is identity matrix so angle = 0

    				this.set( 1, 0, 0, 0 );

    				return this; // zero angle, arbitrary axis

    			}

    			// otherwise this singularity is angle = 180

    			angle = Math.PI;

    			var xx = ( m11 + 1 ) / 2;
    			var yy = ( m22 + 1 ) / 2;
    			var zz = ( m33 + 1 ) / 2;
    			var xy = ( m12 + m21 ) / 4;
    			var xz = ( m13 + m31 ) / 4;
    			var yz = ( m23 + m32 ) / 4;

    			if ( ( xx > yy ) && ( xx > zz ) ) {

    				// m11 is the largest diagonal term

    				if ( xx < epsilon ) {

    					x = 0;
    					y = 0.707106781;
    					z = 0.707106781;

    				} else {

    					x = Math.sqrt( xx );
    					y = xy / x;
    					z = xz / x;

    				}

    			} else if ( yy > zz ) {

    				// m22 is the largest diagonal term

    				if ( yy < epsilon ) {

    					x = 0.707106781;
    					y = 0;
    					z = 0.707106781;

    				} else {

    					y = Math.sqrt( yy );
    					x = xy / y;
    					z = yz / y;

    				}

    			} else {

    				// m33 is the largest diagonal term so base result on this

    				if ( zz < epsilon ) {

    					x = 0.707106781;
    					y = 0.707106781;
    					z = 0;

    				} else {

    					z = Math.sqrt( zz );
    					x = xz / z;
    					y = yz / z;

    				}

    			}

    			this.set( x, y, z, angle );

    			return this; // return 180 deg rotation

    		}

    		// as we have reached here there are no singularities so we can handle normally

    		var s = Math.sqrt( ( m32 - m23 ) * ( m32 - m23 ) +
    		                   ( m13 - m31 ) * ( m13 - m31 ) +
    		                   ( m21 - m12 ) * ( m21 - m12 ) ); // used to normalize

    		if ( Math.abs( s ) < 0.001 ) s = 1;

    		// prevent divide by zero, should not happen if matrix is orthogonal and should be
    		// caught by singularity test above, but I've left it in just in case

    		this.x = ( m32 - m23 ) / s;
    		this.y = ( m13 - m31 ) / s;
    		this.z = ( m21 - m12 ) / s;
    		this.w = Math.acos( ( m11 + m22 + m33 - 1 ) / 2 );

    		return this;

    	},

    	min: function ( v ) {

    		this.x = Math.min( this.x, v.x );
    		this.y = Math.min( this.y, v.y );
    		this.z = Math.min( this.z, v.z );
    		this.w = Math.min( this.w, v.w );

    		return this;

    	},

    	max: function ( v ) {

    		this.x = Math.max( this.x, v.x );
    		this.y = Math.max( this.y, v.y );
    		this.z = Math.max( this.z, v.z );
    		this.w = Math.max( this.w, v.w );

    		return this;

    	},

    	clamp: function ( min, max ) {

    		// assumes min < max, componentwise

    		this.x = Math.max( min.x, Math.min( max.x, this.x ) );
    		this.y = Math.max( min.y, Math.min( max.y, this.y ) );
    		this.z = Math.max( min.z, Math.min( max.z, this.z ) );
    		this.w = Math.max( min.w, Math.min( max.w, this.w ) );

    		return this;

    	},

    	clampScalar: function ( minVal, maxVal ) {

    		this.x = Math.max( minVal, Math.min( maxVal, this.x ) );
    		this.y = Math.max( minVal, Math.min( maxVal, this.y ) );
    		this.z = Math.max( minVal, Math.min( maxVal, this.z ) );
    		this.w = Math.max( minVal, Math.min( maxVal, this.w ) );

    		return this;

    	},

    	clampLength: function ( min, max ) {

    		var length = this.length();

    		return this.divideScalar( length || 1 ).multiplyScalar( Math.max( min, Math.min( max, length ) ) );

    	},

    	floor: function () {

    		this.x = Math.floor( this.x );
    		this.y = Math.floor( this.y );
    		this.z = Math.floor( this.z );
    		this.w = Math.floor( this.w );

    		return this;

    	},

    	ceil: function () {

    		this.x = Math.ceil( this.x );
    		this.y = Math.ceil( this.y );
    		this.z = Math.ceil( this.z );
    		this.w = Math.ceil( this.w );

    		return this;

    	},

    	round: function () {

    		this.x = Math.round( this.x );
    		this.y = Math.round( this.y );
    		this.z = Math.round( this.z );
    		this.w = Math.round( this.w );

    		return this;

    	},

    	roundToZero: function () {

    		this.x = ( this.x < 0 ) ? Math.ceil( this.x ) : Math.floor( this.x );
    		this.y = ( this.y < 0 ) ? Math.ceil( this.y ) : Math.floor( this.y );
    		this.z = ( this.z < 0 ) ? Math.ceil( this.z ) : Math.floor( this.z );
    		this.w = ( this.w < 0 ) ? Math.ceil( this.w ) : Math.floor( this.w );

    		return this;

    	},

    	negate: function () {

    		this.x = - this.x;
    		this.y = - this.y;
    		this.z = - this.z;
    		this.w = - this.w;

    		return this;

    	},

    	dot: function ( v ) {

    		return this.x * v.x + this.y * v.y + this.z * v.z + this.w * v.w;

    	},

    	lengthSq: function () {

    		return this.x * this.x + this.y * this.y + this.z * this.z + this.w * this.w;

    	},

    	length: function () {

    		return Math.sqrt( this.x * this.x + this.y * this.y + this.z * this.z + this.w * this.w );

    	},

    	manhattanLength: function () {

    		return Math.abs( this.x ) + Math.abs( this.y ) + Math.abs( this.z ) + Math.abs( this.w );

    	},

    	normalize: function () {

    		return this.divideScalar( this.length() || 1 );

    	},

    	setLength: function ( length ) {

    		return this.normalize().multiplyScalar( length );

    	},

    	lerp: function ( v, alpha ) {

    		this.x += ( v.x - this.x ) * alpha;
    		this.y += ( v.y - this.y ) * alpha;
    		this.z += ( v.z - this.z ) * alpha;
    		this.w += ( v.w - this.w ) * alpha;

    		return this;

    	},

    	lerpVectors: function ( v1, v2, alpha ) {

    		return this.subVectors( v2, v1 ).multiplyScalar( alpha ).add( v1 );

    	},

    	equals: function ( v ) {

    		return ( ( v.x === this.x ) && ( v.y === this.y ) && ( v.z === this.z ) && ( v.w === this.w ) );

    	},

    	fromArray: function ( array, offset ) {

    		if ( offset === undefined ) offset = 0;

    		this.x = array[ offset ];
    		this.y = array[ offset + 1 ];
    		this.z = array[ offset + 2 ];
    		this.w = array[ offset + 3 ];

    		return this;

    	},

    	toArray: function ( array, offset ) {

    		if ( array === undefined ) array = [];
    		if ( offset === undefined ) offset = 0;

    		array[ offset ] = this.x;
    		array[ offset + 1 ] = this.y;
    		array[ offset + 2 ] = this.z;
    		array[ offset + 3 ] = this.w;

    		return array;

    	},

    	fromBufferAttribute: function ( attribute, index, offset ) {

    		if ( offset !== undefined ) {

    			console.warn( 'THREE.Vector4: offset has been removed from .fromBufferAttribute().' );

    		}

    		this.x = attribute.getX( index );
    		this.y = attribute.getY( index );
    		this.z = attribute.getZ( index );
    		this.w = attribute.getW( index );

    		return this;

    	}

    } );

    /**
     * @author szimek / https://github.com/szimek/
     * @author alteredq / http://alteredqualia.com/
     * @author Marius Kintel / https://github.com/kintel
     */

    /*
     In options, we can specify:
     * Texture parameters for an auto-generated target texture
     * depthBuffer/stencilBuffer: Booleans to indicate if we should generate these buffers
    */
    function WebGLRenderTarget( width, height, options ) {

    	this.width = width;
    	this.height = height;

    	this.scissor = new Vector4( 0, 0, width, height );
    	this.scissorTest = false;

    	this.viewport = new Vector4( 0, 0, width, height );

    	options = options || {};

    	this.texture = new Texture( undefined, undefined, options.wrapS, options.wrapT, options.magFilter, options.minFilter, options.format, options.type, options.anisotropy, options.encoding );

    	this.texture.image = {};
    	this.texture.image.width = width;
    	this.texture.image.height = height;

    	this.texture.generateMipmaps = options.generateMipmaps !== undefined ? options.generateMipmaps : false;
    	this.texture.minFilter = options.minFilter !== undefined ? options.minFilter : LinearFilter;

    	this.depthBuffer = options.depthBuffer !== undefined ? options.depthBuffer : true;
    	this.stencilBuffer = options.stencilBuffer !== undefined ? options.stencilBuffer : true;
    	this.depthTexture = options.depthTexture !== undefined ? options.depthTexture : null;

    }

    WebGLRenderTarget.prototype = Object.assign( Object.create( EventDispatcher.prototype ), {

    	constructor: WebGLRenderTarget,

    	isWebGLRenderTarget: true,

    	setSize: function ( width, height ) {

    		if ( this.width !== width || this.height !== height ) {

    			this.width = width;
    			this.height = height;

    			this.texture.image.width = width;
    			this.texture.image.height = height;

    			this.dispose();

    		}

    		this.viewport.set( 0, 0, width, height );
    		this.scissor.set( 0, 0, width, height );

    	},

    	clone: function () {

    		return new this.constructor().copy( this );

    	},

    	copy: function ( source ) {

    		this.width = source.width;
    		this.height = source.height;

    		this.viewport.copy( source.viewport );

    		this.texture = source.texture.clone();

    		this.depthBuffer = source.depthBuffer;
    		this.stencilBuffer = source.stencilBuffer;
    		this.depthTexture = source.depthTexture;

    		return this;

    	},

    	dispose: function () {

    		this.dispatchEvent( { type: 'dispose' } );

    	}

    } );

    /**
     * @author Mugen87 / https://github.com/Mugen87
     * @author Matt DesLauriers / @mattdesl
     */

    function WebGLMultisampleRenderTarget( width, height, options ) {

    	WebGLRenderTarget.call( this, width, height, options );

    	this.samples = 4;

    }

    WebGLMultisampleRenderTarget.prototype = Object.assign( Object.create( WebGLRenderTarget.prototype ), {

    	constructor: WebGLMultisampleRenderTarget,

    	isWebGLMultisampleRenderTarget: true,

    	copy: function ( source ) {

    		WebGLRenderTarget.prototype.copy.call( this, source );

    		this.samples = source.samples;

    		return this;

    	}

    } );

    var _v1 = new Vector3();
    var _m1 = new Matrix4();
    var _zero = new Vector3( 0, 0, 0 );
    var _one = new Vector3( 1, 1, 1 );
    var _x = new Vector3();
    var _y = new Vector3();
    var _z = new Vector3();

    /**
     * @author mrdoob / http://mrdoob.com/
     * @author supereggbert / http://www.paulbrunt.co.uk/
     * @author philogb / http://blog.thejit.org/
     * @author jordi_ros / http://plattsoft.com
     * @author D1plo1d / http://github.com/D1plo1d
     * @author alteredq / http://alteredqualia.com/
     * @author mikael emtinger / http://gomo.se/
     * @author timknip / http://www.floorplanner.com/
     * @author bhouston / http://clara.io
     * @author WestLangley / http://github.com/WestLangley
     */

    function Matrix4() {

    	this.elements = [

    		1, 0, 0, 0,
    		0, 1, 0, 0,
    		0, 0, 1, 0,
    		0, 0, 0, 1

    	];

    	if ( arguments.length > 0 ) {

    		console.error( 'THREE.Matrix4: the constructor no longer reads arguments. use .set() instead.' );

    	}

    }

    Object.assign( Matrix4.prototype, {

    	isMatrix4: true,

    	set: function ( n11, n12, n13, n14, n21, n22, n23, n24, n31, n32, n33, n34, n41, n42, n43, n44 ) {

    		var te = this.elements;

    		te[ 0 ] = n11; te[ 4 ] = n12; te[ 8 ] = n13; te[ 12 ] = n14;
    		te[ 1 ] = n21; te[ 5 ] = n22; te[ 9 ] = n23; te[ 13 ] = n24;
    		te[ 2 ] = n31; te[ 6 ] = n32; te[ 10 ] = n33; te[ 14 ] = n34;
    		te[ 3 ] = n41; te[ 7 ] = n42; te[ 11 ] = n43; te[ 15 ] = n44;

    		return this;

    	},

    	identity: function () {

    		this.set(

    			1, 0, 0, 0,
    			0, 1, 0, 0,
    			0, 0, 1, 0,
    			0, 0, 0, 1

    		);

    		return this;

    	},

    	clone: function () {

    		return new Matrix4().fromArray( this.elements );

    	},

    	copy: function ( m ) {

    		var te = this.elements;
    		var me = m.elements;

    		te[ 0 ] = me[ 0 ]; te[ 1 ] = me[ 1 ]; te[ 2 ] = me[ 2 ]; te[ 3 ] = me[ 3 ];
    		te[ 4 ] = me[ 4 ]; te[ 5 ] = me[ 5 ]; te[ 6 ] = me[ 6 ]; te[ 7 ] = me[ 7 ];
    		te[ 8 ] = me[ 8 ]; te[ 9 ] = me[ 9 ]; te[ 10 ] = me[ 10 ]; te[ 11 ] = me[ 11 ];
    		te[ 12 ] = me[ 12 ]; te[ 13 ] = me[ 13 ]; te[ 14 ] = me[ 14 ]; te[ 15 ] = me[ 15 ];

    		return this;

    	},

    	copyPosition: function ( m ) {

    		var te = this.elements, me = m.elements;

    		te[ 12 ] = me[ 12 ];
    		te[ 13 ] = me[ 13 ];
    		te[ 14 ] = me[ 14 ];

    		return this;

    	},

    	extractBasis: function ( xAxis, yAxis, zAxis ) {

    		xAxis.setFromMatrixColumn( this, 0 );
    		yAxis.setFromMatrixColumn( this, 1 );
    		zAxis.setFromMatrixColumn( this, 2 );

    		return this;

    	},

    	makeBasis: function ( xAxis, yAxis, zAxis ) {

    		this.set(
    			xAxis.x, yAxis.x, zAxis.x, 0,
    			xAxis.y, yAxis.y, zAxis.y, 0,
    			xAxis.z, yAxis.z, zAxis.z, 0,
    			0, 0, 0, 1
    		);

    		return this;

    	},

    	extractRotation: function ( m ) {

    		// this method does not support reflection matrices

    		var te = this.elements;
    		var me = m.elements;

    		var scaleX = 1 / _v1.setFromMatrixColumn( m, 0 ).length();
    		var scaleY = 1 / _v1.setFromMatrixColumn( m, 1 ).length();
    		var scaleZ = 1 / _v1.setFromMatrixColumn( m, 2 ).length();

    		te[ 0 ] = me[ 0 ] * scaleX;
    		te[ 1 ] = me[ 1 ] * scaleX;
    		te[ 2 ] = me[ 2 ] * scaleX;
    		te[ 3 ] = 0;

    		te[ 4 ] = me[ 4 ] * scaleY;
    		te[ 5 ] = me[ 5 ] * scaleY;
    		te[ 6 ] = me[ 6 ] * scaleY;
    		te[ 7 ] = 0;

    		te[ 8 ] = me[ 8 ] * scaleZ;
    		te[ 9 ] = me[ 9 ] * scaleZ;
    		te[ 10 ] = me[ 10 ] * scaleZ;
    		te[ 11 ] = 0;

    		te[ 12 ] = 0;
    		te[ 13 ] = 0;
    		te[ 14 ] = 0;
    		te[ 15 ] = 1;

    		return this;

    	},

    	makeRotationFromEuler: function ( euler ) {

    		if ( ! ( euler && euler.isEuler ) ) {

    			console.error( 'THREE.Matrix4: .makeRotationFromEuler() now expects a Euler rotation rather than a Vector3 and order.' );

    		}

    		var te = this.elements;

    		var x = euler.x, y = euler.y, z = euler.z;
    		var a = Math.cos( x ), b = Math.sin( x );
    		var c = Math.cos( y ), d = Math.sin( y );
    		var e = Math.cos( z ), f = Math.sin( z );

    		if ( euler.order === 'XYZ' ) {

    			var ae = a * e, af = a * f, be = b * e, bf = b * f;

    			te[ 0 ] = c * e;
    			te[ 4 ] = - c * f;
    			te[ 8 ] = d;

    			te[ 1 ] = af + be * d;
    			te[ 5 ] = ae - bf * d;
    			te[ 9 ] = - b * c;

    			te[ 2 ] = bf - ae * d;
    			te[ 6 ] = be + af * d;
    			te[ 10 ] = a * c;

    		} else if ( euler.order === 'YXZ' ) {

    			var ce = c * e, cf = c * f, de = d * e, df = d * f;

    			te[ 0 ] = ce + df * b;
    			te[ 4 ] = de * b - cf;
    			te[ 8 ] = a * d;

    			te[ 1 ] = a * f;
    			te[ 5 ] = a * e;
    			te[ 9 ] = - b;

    			te[ 2 ] = cf * b - de;
    			te[ 6 ] = df + ce * b;
    			te[ 10 ] = a * c;

    		} else if ( euler.order === 'ZXY' ) {

    			var ce = c * e, cf = c * f, de = d * e, df = d * f;

    			te[ 0 ] = ce - df * b;
    			te[ 4 ] = - a * f;
    			te[ 8 ] = de + cf * b;

    			te[ 1 ] = cf + de * b;
    			te[ 5 ] = a * e;
    			te[ 9 ] = df - ce * b;

    			te[ 2 ] = - a * d;
    			te[ 6 ] = b;
    			te[ 10 ] = a * c;

    		} else if ( euler.order === 'ZYX' ) {

    			var ae = a * e, af = a * f, be = b * e, bf = b * f;

    			te[ 0 ] = c * e;
    			te[ 4 ] = be * d - af;
    			te[ 8 ] = ae * d + bf;

    			te[ 1 ] = c * f;
    			te[ 5 ] = bf * d + ae;
    			te[ 9 ] = af * d - be;

    			te[ 2 ] = - d;
    			te[ 6 ] = b * c;
    			te[ 10 ] = a * c;

    		} else if ( euler.order === 'YZX' ) {

    			var ac = a * c, ad = a * d, bc = b * c, bd = b * d;

    			te[ 0 ] = c * e;
    			te[ 4 ] = bd - ac * f;
    			te[ 8 ] = bc * f + ad;

    			te[ 1 ] = f;
    			te[ 5 ] = a * e;
    			te[ 9 ] = - b * e;

    			te[ 2 ] = - d * e;
    			te[ 6 ] = ad * f + bc;
    			te[ 10 ] = ac - bd * f;

    		} else if ( euler.order === 'XZY' ) {

    			var ac = a * c, ad = a * d, bc = b * c, bd = b * d;

    			te[ 0 ] = c * e;
    			te[ 4 ] = - f;
    			te[ 8 ] = d * e;

    			te[ 1 ] = ac * f + bd;
    			te[ 5 ] = a * e;
    			te[ 9 ] = ad * f - bc;

    			te[ 2 ] = bc * f - ad;
    			te[ 6 ] = b * e;
    			te[ 10 ] = bd * f + ac;

    		}

    		// bottom row
    		te[ 3 ] = 0;
    		te[ 7 ] = 0;
    		te[ 11 ] = 0;

    		// last column
    		te[ 12 ] = 0;
    		te[ 13 ] = 0;
    		te[ 14 ] = 0;
    		te[ 15 ] = 1;

    		return this;

    	},

    	makeRotationFromQuaternion: function ( q ) {

    		return this.compose( _zero, q, _one );

    	},

    	lookAt: function ( eye, target, up ) {

    		var te = this.elements;

    		_z.subVectors( eye, target );

    		if ( _z.lengthSq() === 0 ) {

    			// eye and target are in the same position

    			_z.z = 1;

    		}

    		_z.normalize();
    		_x.crossVectors( up, _z );

    		if ( _x.lengthSq() === 0 ) {

    			// up and z are parallel

    			if ( Math.abs( up.z ) === 1 ) {

    				_z.x += 0.0001;

    			} else {

    				_z.z += 0.0001;

    			}

    			_z.normalize();
    			_x.crossVectors( up, _z );

    		}

    		_x.normalize();
    		_y.crossVectors( _z, _x );

    		te[ 0 ] = _x.x; te[ 4 ] = _y.x; te[ 8 ] = _z.x;
    		te[ 1 ] = _x.y; te[ 5 ] = _y.y; te[ 9 ] = _z.y;
    		te[ 2 ] = _x.z; te[ 6 ] = _y.z; te[ 10 ] = _z.z;

    		return this;

    	},

    	multiply: function ( m, n ) {

    		if ( n !== undefined ) {

    			console.warn( 'THREE.Matrix4: .multiply() now only accepts one argument. Use .multiplyMatrices( a, b ) instead.' );
    			return this.multiplyMatrices( m, n );

    		}

    		return this.multiplyMatrices( this, m );

    	},

    	premultiply: function ( m ) {

    		return this.multiplyMatrices( m, this );

    	},

    	multiplyMatrices: function ( a, b ) {

    		var ae = a.elements;
    		var be = b.elements;
    		var te = this.elements;

    		var a11 = ae[ 0 ], a12 = ae[ 4 ], a13 = ae[ 8 ], a14 = ae[ 12 ];
    		var a21 = ae[ 1 ], a22 = ae[ 5 ], a23 = ae[ 9 ], a24 = ae[ 13 ];
    		var a31 = ae[ 2 ], a32 = ae[ 6 ], a33 = ae[ 10 ], a34 = ae[ 14 ];
    		var a41 = ae[ 3 ], a42 = ae[ 7 ], a43 = ae[ 11 ], a44 = ae[ 15 ];

    		var b11 = be[ 0 ], b12 = be[ 4 ], b13 = be[ 8 ], b14 = be[ 12 ];
    		var b21 = be[ 1 ], b22 = be[ 5 ], b23 = be[ 9 ], b24 = be[ 13 ];
    		var b31 = be[ 2 ], b32 = be[ 6 ], b33 = be[ 10 ], b34 = be[ 14 ];
    		var b41 = be[ 3 ], b42 = be[ 7 ], b43 = be[ 11 ], b44 = be[ 15 ];

    		te[ 0 ] = a11 * b11 + a12 * b21 + a13 * b31 + a14 * b41;
    		te[ 4 ] = a11 * b12 + a12 * b22 + a13 * b32 + a14 * b42;
    		te[ 8 ] = a11 * b13 + a12 * b23 + a13 * b33 + a14 * b43;
    		te[ 12 ] = a11 * b14 + a12 * b24 + a13 * b34 + a14 * b44;

    		te[ 1 ] = a21 * b11 + a22 * b21 + a23 * b31 + a24 * b41;
    		te[ 5 ] = a21 * b12 + a22 * b22 + a23 * b32 + a24 * b42;
    		te[ 9 ] = a21 * b13 + a22 * b23 + a23 * b33 + a24 * b43;
    		te[ 13 ] = a21 * b14 + a22 * b24 + a23 * b34 + a24 * b44;

    		te[ 2 ] = a31 * b11 + a32 * b21 + a33 * b31 + a34 * b41;
    		te[ 6 ] = a31 * b12 + a32 * b22 + a33 * b32 + a34 * b42;
    		te[ 10 ] = a31 * b13 + a32 * b23 + a33 * b33 + a34 * b43;
    		te[ 14 ] = a31 * b14 + a32 * b24 + a33 * b34 + a34 * b44;

    		te[ 3 ] = a41 * b11 + a42 * b21 + a43 * b31 + a44 * b41;
    		te[ 7 ] = a41 * b12 + a42 * b22 + a43 * b32 + a44 * b42;
    		te[ 11 ] = a41 * b13 + a42 * b23 + a43 * b33 + a44 * b43;
    		te[ 15 ] = a41 * b14 + a42 * b24 + a43 * b34 + a44 * b44;

    		return this;

    	},

    	multiplyScalar: function ( s ) {

    		var te = this.elements;

    		te[ 0 ] *= s; te[ 4 ] *= s; te[ 8 ] *= s; te[ 12 ] *= s;
    		te[ 1 ] *= s; te[ 5 ] *= s; te[ 9 ] *= s; te[ 13 ] *= s;
    		te[ 2 ] *= s; te[ 6 ] *= s; te[ 10 ] *= s; te[ 14 ] *= s;
    		te[ 3 ] *= s; te[ 7 ] *= s; te[ 11 ] *= s; te[ 15 ] *= s;

    		return this;

    	},

    	applyToBufferAttribute: function ( attribute ) {

    		for ( var i = 0, l = attribute.count; i < l; i ++ ) {

    			_v1.x = attribute.getX( i );
    			_v1.y = attribute.getY( i );
    			_v1.z = attribute.getZ( i );

    			_v1.applyMatrix4( this );

    			attribute.setXYZ( i, _v1.x, _v1.y, _v1.z );

    		}

    		return attribute;

    	},

    	determinant: function () {

    		var te = this.elements;

    		var n11 = te[ 0 ], n12 = te[ 4 ], n13 = te[ 8 ], n14 = te[ 12 ];
    		var n21 = te[ 1 ], n22 = te[ 5 ], n23 = te[ 9 ], n24 = te[ 13 ];
    		var n31 = te[ 2 ], n32 = te[ 6 ], n33 = te[ 10 ], n34 = te[ 14 ];
    		var n41 = te[ 3 ], n42 = te[ 7 ], n43 = te[ 11 ], n44 = te[ 15 ];

    		//TODO: make this more efficient
    		//( based on http://www.euclideanspace.com/maths/algebra/matrix/functions/inverse/fourD/index.htm )

    		return (
    			n41 * (
    				+ n14 * n23 * n32
    				 - n13 * n24 * n32
    				 - n14 * n22 * n33
    				 + n12 * n24 * n33
    				 + n13 * n22 * n34
    				 - n12 * n23 * n34
    			) +
    			n42 * (
    				+ n11 * n23 * n34
    				 - n11 * n24 * n33
    				 + n14 * n21 * n33
    				 - n13 * n21 * n34
    				 + n13 * n24 * n31
    				 - n14 * n23 * n31
    			) +
    			n43 * (
    				+ n11 * n24 * n32
    				 - n11 * n22 * n34
    				 - n14 * n21 * n32
    				 + n12 * n21 * n34
    				 + n14 * n22 * n31
    				 - n12 * n24 * n31
    			) +
    			n44 * (
    				- n13 * n22 * n31
    				 - n11 * n23 * n32
    				 + n11 * n22 * n33
    				 + n13 * n21 * n32
    				 - n12 * n21 * n33
    				 + n12 * n23 * n31
    			)

    		);

    	},

    	transpose: function () {

    		var te = this.elements;
    		var tmp;

    		tmp = te[ 1 ]; te[ 1 ] = te[ 4 ]; te[ 4 ] = tmp;
    		tmp = te[ 2 ]; te[ 2 ] = te[ 8 ]; te[ 8 ] = tmp;
    		tmp = te[ 6 ]; te[ 6 ] = te[ 9 ]; te[ 9 ] = tmp;

    		tmp = te[ 3 ]; te[ 3 ] = te[ 12 ]; te[ 12 ] = tmp;
    		tmp = te[ 7 ]; te[ 7 ] = te[ 13 ]; te[ 13 ] = tmp;
    		tmp = te[ 11 ]; te[ 11 ] = te[ 14 ]; te[ 14 ] = tmp;

    		return this;

    	},

    	setPosition: function ( x, y, z ) {

    		var te = this.elements;

    		if ( x.isVector3 ) {

    			te[ 12 ] = x.x;
    			te[ 13 ] = x.y;
    			te[ 14 ] = x.z;

    		} else {

    			te[ 12 ] = x;
    			te[ 13 ] = y;
    			te[ 14 ] = z;

    		}

    		return this;

    	},

    	getInverse: function ( m, throwOnDegenerate ) {

    		// based on http://www.euclideanspace.com/maths/algebra/matrix/functions/inverse/fourD/index.htm
    		var te = this.elements,
    			me = m.elements,

    			n11 = me[ 0 ], n21 = me[ 1 ], n31 = me[ 2 ], n41 = me[ 3 ],
    			n12 = me[ 4 ], n22 = me[ 5 ], n32 = me[ 6 ], n42 = me[ 7 ],
    			n13 = me[ 8 ], n23 = me[ 9 ], n33 = me[ 10 ], n43 = me[ 11 ],
    			n14 = me[ 12 ], n24 = me[ 13 ], n34 = me[ 14 ], n44 = me[ 15 ],

    			t11 = n23 * n34 * n42 - n24 * n33 * n42 + n24 * n32 * n43 - n22 * n34 * n43 - n23 * n32 * n44 + n22 * n33 * n44,
    			t12 = n14 * n33 * n42 - n13 * n34 * n42 - n14 * n32 * n43 + n12 * n34 * n43 + n13 * n32 * n44 - n12 * n33 * n44,
    			t13 = n13 * n24 * n42 - n14 * n23 * n42 + n14 * n22 * n43 - n12 * n24 * n43 - n13 * n22 * n44 + n12 * n23 * n44,
    			t14 = n14 * n23 * n32 - n13 * n24 * n32 - n14 * n22 * n33 + n12 * n24 * n33 + n13 * n22 * n34 - n12 * n23 * n34;

    		var det = n11 * t11 + n21 * t12 + n31 * t13 + n41 * t14;

    		if ( det === 0 ) {

    			var msg = "THREE.Matrix4: .getInverse() can't invert matrix, determinant is 0";

    			if ( throwOnDegenerate === true ) {

    				throw new Error( msg );

    			} else {

    				console.warn( msg );

    			}

    			return this.identity();

    		}

    		var detInv = 1 / det;

    		te[ 0 ] = t11 * detInv;
    		te[ 1 ] = ( n24 * n33 * n41 - n23 * n34 * n41 - n24 * n31 * n43 + n21 * n34 * n43 + n23 * n31 * n44 - n21 * n33 * n44 ) * detInv;
    		te[ 2 ] = ( n22 * n34 * n41 - n24 * n32 * n41 + n24 * n31 * n42 - n21 * n34 * n42 - n22 * n31 * n44 + n21 * n32 * n44 ) * detInv;
    		te[ 3 ] = ( n23 * n32 * n41 - n22 * n33 * n41 - n23 * n31 * n42 + n21 * n33 * n42 + n22 * n31 * n43 - n21 * n32 * n43 ) * detInv;

    		te[ 4 ] = t12 * detInv;
    		te[ 5 ] = ( n13 * n34 * n41 - n14 * n33 * n41 + n14 * n31 * n43 - n11 * n34 * n43 - n13 * n31 * n44 + n11 * n33 * n44 ) * detInv;
    		te[ 6 ] = ( n14 * n32 * n41 - n12 * n34 * n41 - n14 * n31 * n42 + n11 * n34 * n42 + n12 * n31 * n44 - n11 * n32 * n44 ) * detInv;
    		te[ 7 ] = ( n12 * n33 * n41 - n13 * n32 * n41 + n13 * n31 * n42 - n11 * n33 * n42 - n12 * n31 * n43 + n11 * n32 * n43 ) * detInv;

    		te[ 8 ] = t13 * detInv;
    		te[ 9 ] = ( n14 * n23 * n41 - n13 * n24 * n41 - n14 * n21 * n43 + n11 * n24 * n43 + n13 * n21 * n44 - n11 * n23 * n44 ) * detInv;
    		te[ 10 ] = ( n12 * n24 * n41 - n14 * n22 * n41 + n14 * n21 * n42 - n11 * n24 * n42 - n12 * n21 * n44 + n11 * n22 * n44 ) * detInv;
    		te[ 11 ] = ( n13 * n22 * n41 - n12 * n23 * n41 - n13 * n21 * n42 + n11 * n23 * n42 + n12 * n21 * n43 - n11 * n22 * n43 ) * detInv;

    		te[ 12 ] = t14 * detInv;
    		te[ 13 ] = ( n13 * n24 * n31 - n14 * n23 * n31 + n14 * n21 * n33 - n11 * n24 * n33 - n13 * n21 * n34 + n11 * n23 * n34 ) * detInv;
    		te[ 14 ] = ( n14 * n22 * n31 - n12 * n24 * n31 - n14 * n21 * n32 + n11 * n24 * n32 + n12 * n21 * n34 - n11 * n22 * n34 ) * detInv;
    		te[ 15 ] = ( n12 * n23 * n31 - n13 * n22 * n31 + n13 * n21 * n32 - n11 * n23 * n32 - n12 * n21 * n33 + n11 * n22 * n33 ) * detInv;

    		return this;

    	},

    	scale: function ( v ) {

    		var te = this.elements;
    		var x = v.x, y = v.y, z = v.z;

    		te[ 0 ] *= x; te[ 4 ] *= y; te[ 8 ] *= z;
    		te[ 1 ] *= x; te[ 5 ] *= y; te[ 9 ] *= z;
    		te[ 2 ] *= x; te[ 6 ] *= y; te[ 10 ] *= z;
    		te[ 3 ] *= x; te[ 7 ] *= y; te[ 11 ] *= z;

    		return this;

    	},

    	getMaxScaleOnAxis: function () {

    		var te = this.elements;

    		var scaleXSq = te[ 0 ] * te[ 0 ] + te[ 1 ] * te[ 1 ] + te[ 2 ] * te[ 2 ];
    		var scaleYSq = te[ 4 ] * te[ 4 ] + te[ 5 ] * te[ 5 ] + te[ 6 ] * te[ 6 ];
    		var scaleZSq = te[ 8 ] * te[ 8 ] + te[ 9 ] * te[ 9 ] + te[ 10 ] * te[ 10 ];

    		return Math.sqrt( Math.max( scaleXSq, scaleYSq, scaleZSq ) );

    	},

    	makeTranslation: function ( x, y, z ) {

    		this.set(

    			1, 0, 0, x,
    			0, 1, 0, y,
    			0, 0, 1, z,
    			0, 0, 0, 1

    		);

    		return this;

    	},

    	makeRotationX: function ( theta ) {

    		var c = Math.cos( theta ), s = Math.sin( theta );

    		this.set(

    			1, 0, 0, 0,
    			0, c, - s, 0,
    			0, s, c, 0,
    			0, 0, 0, 1

    		);

    		return this;

    	},

    	makeRotationY: function ( theta ) {

    		var c = Math.cos( theta ), s = Math.sin( theta );

    		this.set(

    			 c, 0, s, 0,
    			 0, 1, 0, 0,
    			- s, 0, c, 0,
    			 0, 0, 0, 1

    		);

    		return this;

    	},

    	makeRotationZ: function ( theta ) {

    		var c = Math.cos( theta ), s = Math.sin( theta );

    		this.set(

    			c, - s, 0, 0,
    			s, c, 0, 0,
    			0, 0, 1, 0,
    			0, 0, 0, 1

    		);

    		return this;

    	},

    	makeRotationAxis: function ( axis, angle ) {

    		// Based on http://www.gamedev.net/reference/articles/article1199.asp

    		var c = Math.cos( angle );
    		var s = Math.sin( angle );
    		var t = 1 - c;
    		var x = axis.x, y = axis.y, z = axis.z;
    		var tx = t * x, ty = t * y;

    		this.set(

    			tx * x + c, tx * y - s * z, tx * z + s * y, 0,
    			tx * y + s * z, ty * y + c, ty * z - s * x, 0,
    			tx * z - s * y, ty * z + s * x, t * z * z + c, 0,
    			0, 0, 0, 1

    		);

    		 return this;

    	},

    	makeScale: function ( x, y, z ) {

    		this.set(

    			x, 0, 0, 0,
    			0, y, 0, 0,
    			0, 0, z, 0,
    			0, 0, 0, 1

    		);

    		return this;

    	},

    	makeShear: function ( x, y, z ) {

    		this.set(

    			1, y, z, 0,
    			x, 1, z, 0,
    			x, y, 1, 0,
    			0, 0, 0, 1

    		);

    		return this;

    	},

    	compose: function ( position, quaternion, scale ) {

    		var te = this.elements;

    		var x = quaternion._x, y = quaternion._y, z = quaternion._z, w = quaternion._w;
    		var x2 = x + x,	y2 = y + y, z2 = z + z;
    		var xx = x * x2, xy = x * y2, xz = x * z2;
    		var yy = y * y2, yz = y * z2, zz = z * z2;
    		var wx = w * x2, wy = w * y2, wz = w * z2;

    		var sx = scale.x, sy = scale.y, sz = scale.z;

    		te[ 0 ] = ( 1 - ( yy + zz ) ) * sx;
    		te[ 1 ] = ( xy + wz ) * sx;
    		te[ 2 ] = ( xz - wy ) * sx;
    		te[ 3 ] = 0;

    		te[ 4 ] = ( xy - wz ) * sy;
    		te[ 5 ] = ( 1 - ( xx + zz ) ) * sy;
    		te[ 6 ] = ( yz + wx ) * sy;
    		te[ 7 ] = 0;

    		te[ 8 ] = ( xz + wy ) * sz;
    		te[ 9 ] = ( yz - wx ) * sz;
    		te[ 10 ] = ( 1 - ( xx + yy ) ) * sz;
    		te[ 11 ] = 0;

    		te[ 12 ] = position.x;
    		te[ 13 ] = position.y;
    		te[ 14 ] = position.z;
    		te[ 15 ] = 1;

    		return this;

    	},

    	decompose: function ( position, quaternion, scale ) {

    		var te = this.elements;

    		var sx = _v1.set( te[ 0 ], te[ 1 ], te[ 2 ] ).length();
    		var sy = _v1.set( te[ 4 ], te[ 5 ], te[ 6 ] ).length();
    		var sz = _v1.set( te[ 8 ], te[ 9 ], te[ 10 ] ).length();

    		// if determine is negative, we need to invert one scale
    		var det = this.determinant();
    		if ( det < 0 ) sx = - sx;

    		position.x = te[ 12 ];
    		position.y = te[ 13 ];
    		position.z = te[ 14 ];

    		// scale the rotation part
    		_m1.copy( this );

    		var invSX = 1 / sx;
    		var invSY = 1 / sy;
    		var invSZ = 1 / sz;

    		_m1.elements[ 0 ] *= invSX;
    		_m1.elements[ 1 ] *= invSX;
    		_m1.elements[ 2 ] *= invSX;

    		_m1.elements[ 4 ] *= invSY;
    		_m1.elements[ 5 ] *= invSY;
    		_m1.elements[ 6 ] *= invSY;

    		_m1.elements[ 8 ] *= invSZ;
    		_m1.elements[ 9 ] *= invSZ;
    		_m1.elements[ 10 ] *= invSZ;

    		quaternion.setFromRotationMatrix( _m1 );

    		scale.x = sx;
    		scale.y = sy;
    		scale.z = sz;

    		return this;

    	},

    	makePerspective: function ( left, right, top, bottom, near, far ) {

    		if ( far === undefined ) {

    			console.warn( 'THREE.Matrix4: .makePerspective() has been redefined and has a new signature. Please check the docs.' );

    		}

    		var te = this.elements;
    		var x = 2 * near / ( right - left );
    		var y = 2 * near / ( top - bottom );

    		var a = ( right + left ) / ( right - left );
    		var b = ( top + bottom ) / ( top - bottom );
    		var c = - ( far + near ) / ( far - near );
    		var d = - 2 * far * near / ( far - near );

    		te[ 0 ] = x;	te[ 4 ] = 0;	te[ 8 ] = a;	te[ 12 ] = 0;
    		te[ 1 ] = 0;	te[ 5 ] = y;	te[ 9 ] = b;	te[ 13 ] = 0;
    		te[ 2 ] = 0;	te[ 6 ] = 0;	te[ 10 ] = c;	te[ 14 ] = d;
    		te[ 3 ] = 0;	te[ 7 ] = 0;	te[ 11 ] = - 1;	te[ 15 ] = 0;

    		return this;

    	},

    	makeOrthographic: function ( left, right, top, bottom, near, far ) {

    		var te = this.elements;
    		var w = 1.0 / ( right - left );
    		var h = 1.0 / ( top - bottom );
    		var p = 1.0 / ( far - near );

    		var x = ( right + left ) * w;
    		var y = ( top + bottom ) * h;
    		var z = ( far + near ) * p;

    		te[ 0 ] = 2 * w;	te[ 4 ] = 0;	te[ 8 ] = 0;	te[ 12 ] = - x;
    		te[ 1 ] = 0;	te[ 5 ] = 2 * h;	te[ 9 ] = 0;	te[ 13 ] = - y;
    		te[ 2 ] = 0;	te[ 6 ] = 0;	te[ 10 ] = - 2 * p;	te[ 14 ] = - z;
    		te[ 3 ] = 0;	te[ 7 ] = 0;	te[ 11 ] = 0;	te[ 15 ] = 1;

    		return this;

    	},

    	equals: function ( matrix ) {

    		var te = this.elements;
    		var me = matrix.elements;

    		for ( var i = 0; i < 16; i ++ ) {

    			if ( te[ i ] !== me[ i ] ) return false;

    		}

    		return true;

    	},

    	fromArray: function ( array, offset ) {

    		if ( offset === undefined ) offset = 0;

    		for ( var i = 0; i < 16; i ++ ) {

    			this.elements[ i ] = array[ i + offset ];

    		}

    		return this;

    	},

    	toArray: function ( array, offset ) {

    		if ( array === undefined ) array = [];
    		if ( offset === undefined ) offset = 0;

    		var te = this.elements;

    		array[ offset ] = te[ 0 ];
    		array[ offset + 1 ] = te[ 1 ];
    		array[ offset + 2 ] = te[ 2 ];
    		array[ offset + 3 ] = te[ 3 ];

    		array[ offset + 4 ] = te[ 4 ];
    		array[ offset + 5 ] = te[ 5 ];
    		array[ offset + 6 ] = te[ 6 ];
    		array[ offset + 7 ] = te[ 7 ];

    		array[ offset + 8 ] = te[ 8 ];
    		array[ offset + 9 ] = te[ 9 ];
    		array[ offset + 10 ] = te[ 10 ];
    		array[ offset + 11 ] = te[ 11 ];

    		array[ offset + 12 ] = te[ 12 ];
    		array[ offset + 13 ] = te[ 13 ];
    		array[ offset + 14 ] = te[ 14 ];
    		array[ offset + 15 ] = te[ 15 ];

    		return array;

    	}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     * @author WestLangley / http://github.com/WestLangley
     * @author bhouston / http://clara.io
     */

    var _matrix = new Matrix4();
    var _quaternion$1 = new Quaternion();

    function Euler( x, y, z, order ) {

    	this._x = x || 0;
    	this._y = y || 0;
    	this._z = z || 0;
    	this._order = order || Euler.DefaultOrder;

    }

    Euler.RotationOrders = [ 'XYZ', 'YZX', 'ZXY', 'XZY', 'YXZ', 'ZYX' ];

    Euler.DefaultOrder = 'XYZ';

    Object.defineProperties( Euler.prototype, {

    	x: {

    		get: function () {

    			return this._x;

    		},

    		set: function ( value ) {

    			this._x = value;
    			this._onChangeCallback();

    		}

    	},

    	y: {

    		get: function () {

    			return this._y;

    		},

    		set: function ( value ) {

    			this._y = value;
    			this._onChangeCallback();

    		}

    	},

    	z: {

    		get: function () {

    			return this._z;

    		},

    		set: function ( value ) {

    			this._z = value;
    			this._onChangeCallback();

    		}

    	},

    	order: {

    		get: function () {

    			return this._order;

    		},

    		set: function ( value ) {

    			this._order = value;
    			this._onChangeCallback();

    		}

    	}

    } );

    Object.assign( Euler.prototype, {

    	isEuler: true,

    	set: function ( x, y, z, order ) {

    		this._x = x;
    		this._y = y;
    		this._z = z;
    		this._order = order || this._order;

    		this._onChangeCallback();

    		return this;

    	},

    	clone: function () {

    		return new this.constructor( this._x, this._y, this._z, this._order );

    	},

    	copy: function ( euler ) {

    		this._x = euler._x;
    		this._y = euler._y;
    		this._z = euler._z;
    		this._order = euler._order;

    		this._onChangeCallback();

    		return this;

    	},

    	setFromRotationMatrix: function ( m, order, update ) {

    		var clamp = _Math.clamp;

    		// assumes the upper 3x3 of m is a pure rotation matrix (i.e, unscaled)

    		var te = m.elements;
    		var m11 = te[ 0 ], m12 = te[ 4 ], m13 = te[ 8 ];
    		var m21 = te[ 1 ], m22 = te[ 5 ], m23 = te[ 9 ];
    		var m31 = te[ 2 ], m32 = te[ 6 ], m33 = te[ 10 ];

    		order = order || this._order;

    		if ( order === 'XYZ' ) {

    			this._y = Math.asin( clamp( m13, - 1, 1 ) );

    			if ( Math.abs( m13 ) < 0.9999999 ) {

    				this._x = Math.atan2( - m23, m33 );
    				this._z = Math.atan2( - m12, m11 );

    			} else {

    				this._x = Math.atan2( m32, m22 );
    				this._z = 0;

    			}

    		} else if ( order === 'YXZ' ) {

    			this._x = Math.asin( - clamp( m23, - 1, 1 ) );

    			if ( Math.abs( m23 ) < 0.9999999 ) {

    				this._y = Math.atan2( m13, m33 );
    				this._z = Math.atan2( m21, m22 );

    			} else {

    				this._y = Math.atan2( - m31, m11 );
    				this._z = 0;

    			}

    		} else if ( order === 'ZXY' ) {

    			this._x = Math.asin( clamp( m32, - 1, 1 ) );

    			if ( Math.abs( m32 ) < 0.9999999 ) {

    				this._y = Math.atan2( - m31, m33 );
    				this._z = Math.atan2( - m12, m22 );

    			} else {

    				this._y = 0;
    				this._z = Math.atan2( m21, m11 );

    			}

    		} else if ( order === 'ZYX' ) {

    			this._y = Math.asin( - clamp( m31, - 1, 1 ) );

    			if ( Math.abs( m31 ) < 0.9999999 ) {

    				this._x = Math.atan2( m32, m33 );
    				this._z = Math.atan2( m21, m11 );

    			} else {

    				this._x = 0;
    				this._z = Math.atan2( - m12, m22 );

    			}

    		} else if ( order === 'YZX' ) {

    			this._z = Math.asin( clamp( m21, - 1, 1 ) );

    			if ( Math.abs( m21 ) < 0.9999999 ) {

    				this._x = Math.atan2( - m23, m22 );
    				this._y = Math.atan2( - m31, m11 );

    			} else {

    				this._x = 0;
    				this._y = Math.atan2( m13, m33 );

    			}

    		} else if ( order === 'XZY' ) {

    			this._z = Math.asin( - clamp( m12, - 1, 1 ) );

    			if ( Math.abs( m12 ) < 0.9999999 ) {

    				this._x = Math.atan2( m32, m22 );
    				this._y = Math.atan2( m13, m11 );

    			} else {

    				this._x = Math.atan2( - m23, m33 );
    				this._y = 0;

    			}

    		} else {

    			console.warn( 'THREE.Euler: .setFromRotationMatrix() given unsupported order: ' + order );

    		}

    		this._order = order;

    		if ( update !== false ) this._onChangeCallback();

    		return this;

    	},

    	setFromQuaternion: function ( q, order, update ) {

    		_matrix.makeRotationFromQuaternion( q );

    		return this.setFromRotationMatrix( _matrix, order, update );

    	},

    	setFromVector3: function ( v, order ) {

    		return this.set( v.x, v.y, v.z, order || this._order );

    	},

    	reorder: function ( newOrder ) {

    		// WARNING: this discards revolution information -bhouston

    		_quaternion$1.setFromEuler( this );

    		return this.setFromQuaternion( _quaternion$1, newOrder );

    	},

    	equals: function ( euler ) {

    		return ( euler._x === this._x ) && ( euler._y === this._y ) && ( euler._z === this._z ) && ( euler._order === this._order );

    	},

    	fromArray: function ( array ) {

    		this._x = array[ 0 ];
    		this._y = array[ 1 ];
    		this._z = array[ 2 ];
    		if ( array[ 3 ] !== undefined ) this._order = array[ 3 ];

    		this._onChangeCallback();

    		return this;

    	},

    	toArray: function ( array, offset ) {

    		if ( array === undefined ) array = [];
    		if ( offset === undefined ) offset = 0;

    		array[ offset ] = this._x;
    		array[ offset + 1 ] = this._y;
    		array[ offset + 2 ] = this._z;
    		array[ offset + 3 ] = this._order;

    		return array;

    	},

    	toVector3: function ( optionalResult ) {

    		if ( optionalResult ) {

    			return optionalResult.set( this._x, this._y, this._z );

    		} else {

    			return new Vector3( this._x, this._y, this._z );

    		}

    	},

    	_onChange: function ( callback ) {

    		this._onChangeCallback = callback;

    		return this;

    	},

    	_onChangeCallback: function () {}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function Layers() {

    	this.mask = 1 | 0;

    }

    Object.assign( Layers.prototype, {

    	set: function ( channel ) {

    		this.mask = 1 << channel | 0;

    	},

    	enable: function ( channel ) {

    		this.mask |= 1 << channel | 0;

    	},

    	enableAll: function () {

    		this.mask = 0xffffffff | 0;

    	},

    	toggle: function ( channel ) {

    		this.mask ^= 1 << channel | 0;

    	},

    	disable: function ( channel ) {

    		this.mask &= ~ ( 1 << channel | 0 );

    	},

    	disableAll: function () {

    		this.mask = 0;

    	},

    	test: function ( layers ) {

    		return ( this.mask & layers.mask ) !== 0;

    	}

    } );

    var _object3DId = 0;

    var _v1$1 = new Vector3();
    var _q1 = new Quaternion();
    var _m1$1 = new Matrix4();
    var _target = new Vector3();

    var _position = new Vector3();
    var _scale = new Vector3();
    var _quaternion$2 = new Quaternion();

    var _xAxis = new Vector3( 1, 0, 0 );
    var _yAxis = new Vector3( 0, 1, 0 );
    var _zAxis = new Vector3( 0, 0, 1 );

    var _addedEvent = { type: 'added' };
    var _removedEvent = { type: 'removed' };

    /**
     * @author mrdoob / http://mrdoob.com/
     * @author mikael emtinger / http://gomo.se/
     * @author alteredq / http://alteredqualia.com/
     * @author WestLangley / http://github.com/WestLangley
     * @author elephantatwork / www.elephantatwork.ch
     */

    function Object3D() {

    	Object.defineProperty( this, 'id', { value: _object3DId ++ } );

    	this.uuid = _Math.generateUUID();

    	this.name = '';
    	this.type = 'Object3D';

    	this.parent = null;
    	this.children = [];

    	this.up = Object3D.DefaultUp.clone();

    	var position = new Vector3();
    	var rotation = new Euler();
    	var quaternion = new Quaternion();
    	var scale = new Vector3( 1, 1, 1 );

    	function onRotationChange() {

    		quaternion.setFromEuler( rotation, false );

    	}

    	function onQuaternionChange() {

    		rotation.setFromQuaternion( quaternion, undefined, false );

    	}

    	rotation._onChange( onRotationChange );
    	quaternion._onChange( onQuaternionChange );

    	Object.defineProperties( this, {
    		position: {
    			configurable: true,
    			enumerable: true,
    			value: position
    		},
    		rotation: {
    			configurable: true,
    			enumerable: true,
    			value: rotation
    		},
    		quaternion: {
    			configurable: true,
    			enumerable: true,
    			value: quaternion
    		},
    		scale: {
    			configurable: true,
    			enumerable: true,
    			value: scale
    		},
    		modelViewMatrix: {
    			value: new Matrix4()
    		},
    		normalMatrix: {
    			value: new Matrix3()
    		}
    	} );

    	this.matrix = new Matrix4();
    	this.matrixWorld = new Matrix4();

    	this.matrixAutoUpdate = Object3D.DefaultMatrixAutoUpdate;
    	this.matrixWorldNeedsUpdate = false;

    	this.layers = new Layers();
    	this.visible = true;

    	this.castShadow = false;
    	this.receiveShadow = false;

    	this.frustumCulled = true;
    	this.renderOrder = 0;

    	this.userData = {};

    }

    Object3D.DefaultUp = new Vector3( 0, 1, 0 );
    Object3D.DefaultMatrixAutoUpdate = true;

    Object3D.prototype = Object.assign( Object.create( EventDispatcher.prototype ), {

    	constructor: Object3D,

    	isObject3D: true,

    	onBeforeRender: function () {},
    	onAfterRender: function () {},

    	applyMatrix: function ( matrix ) {

    		if ( this.matrixAutoUpdate ) this.updateMatrix();

    		this.matrix.premultiply( matrix );

    		this.matrix.decompose( this.position, this.quaternion, this.scale );

    	},

    	applyQuaternion: function ( q ) {

    		this.quaternion.premultiply( q );

    		return this;

    	},

    	setRotationFromAxisAngle: function ( axis, angle ) {

    		// assumes axis is normalized

    		this.quaternion.setFromAxisAngle( axis, angle );

    	},

    	setRotationFromEuler: function ( euler ) {

    		this.quaternion.setFromEuler( euler, true );

    	},

    	setRotationFromMatrix: function ( m ) {

    		// assumes the upper 3x3 of m is a pure rotation matrix (i.e, unscaled)

    		this.quaternion.setFromRotationMatrix( m );

    	},

    	setRotationFromQuaternion: function ( q ) {

    		// assumes q is normalized

    		this.quaternion.copy( q );

    	},

    	rotateOnAxis: function ( axis, angle ) {

    		// rotate object on axis in object space
    		// axis is assumed to be normalized

    		_q1.setFromAxisAngle( axis, angle );

    		this.quaternion.multiply( _q1 );

    		return this;

    	},

    	rotateOnWorldAxis: function ( axis, angle ) {

    		// rotate object on axis in world space
    		// axis is assumed to be normalized
    		// method assumes no rotated parent

    		_q1.setFromAxisAngle( axis, angle );

    		this.quaternion.premultiply( _q1 );

    		return this;

    	},

    	rotateX: function ( angle ) {

    		return this.rotateOnAxis( _xAxis, angle );

    	},

    	rotateY: function ( angle ) {

    		return this.rotateOnAxis( _yAxis, angle );

    	},

    	rotateZ: function ( angle ) {

    		return this.rotateOnAxis( _zAxis, angle );

    	},

    	translateOnAxis: function ( axis, distance ) {

    		// translate object by distance along axis in object space
    		// axis is assumed to be normalized

    		_v1$1.copy( axis ).applyQuaternion( this.quaternion );

    		this.position.add( _v1$1.multiplyScalar( distance ) );

    		return this;

    	},

    	translateX: function ( distance ) {

    		return this.translateOnAxis( _xAxis, distance );

    	},

    	translateY: function ( distance ) {

    		return this.translateOnAxis( _yAxis, distance );

    	},

    	translateZ: function ( distance ) {

    		return this.translateOnAxis( _zAxis, distance );

    	},

    	localToWorld: function ( vector ) {

    		return vector.applyMatrix4( this.matrixWorld );

    	},

    	worldToLocal: function ( vector ) {

    		return vector.applyMatrix4( _m1$1.getInverse( this.matrixWorld ) );

    	},

    	lookAt: function ( x, y, z ) {

    		// This method does not support objects having non-uniformly-scaled parent(s)

    		if ( x.isVector3 ) {

    			_target.copy( x );

    		} else {

    			_target.set( x, y, z );

    		}

    		var parent = this.parent;

    		this.updateWorldMatrix( true, false );

    		_position.setFromMatrixPosition( this.matrixWorld );

    		if ( this.isCamera || this.isLight ) {

    			_m1$1.lookAt( _position, _target, this.up );

    		} else {

    			_m1$1.lookAt( _target, _position, this.up );

    		}

    		this.quaternion.setFromRotationMatrix( _m1$1 );

    		if ( parent ) {

    			_m1$1.extractRotation( parent.matrixWorld );
    			_q1.setFromRotationMatrix( _m1$1 );
    			this.quaternion.premultiply( _q1.inverse() );

    		}

    	},

    	add: function ( object ) {

    		if ( arguments.length > 1 ) {

    			for ( var i = 0; i < arguments.length; i ++ ) {

    				this.add( arguments[ i ] );

    			}

    			return this;

    		}

    		if ( object === this ) {

    			console.error( "THREE.Object3D.add: object can't be added as a child of itself.", object );
    			return this;

    		}

    		if ( ( object && object.isObject3D ) ) {

    			if ( object.parent !== null ) {

    				object.parent.remove( object );

    			}

    			object.parent = this;
    			this.children.push( object );

    			object.dispatchEvent( _addedEvent );

    		} else {

    			console.error( "THREE.Object3D.add: object not an instance of THREE.Object3D.", object );

    		}

    		return this;

    	},

    	remove: function ( object ) {

    		if ( arguments.length > 1 ) {

    			for ( var i = 0; i < arguments.length; i ++ ) {

    				this.remove( arguments[ i ] );

    			}

    			return this;

    		}

    		var index = this.children.indexOf( object );

    		if ( index !== - 1 ) {

    			object.parent = null;
    			this.children.splice( index, 1 );

    			object.dispatchEvent( _removedEvent );

    		}

    		return this;

    	},

    	attach: function ( object ) {

    		// adds object as a child of this, while maintaining the object's world transform

    		this.updateWorldMatrix( true, false );

    		_m1$1.getInverse( this.matrixWorld );

    		if ( object.parent !== null ) {

    			object.parent.updateWorldMatrix( true, false );

    			_m1$1.multiply( object.parent.matrixWorld );

    		}

    		object.applyMatrix( _m1$1 );

    		object.updateWorldMatrix( false, false );

    		this.add( object );

    		return this;

    	},

    	getObjectById: function ( id ) {

    		return this.getObjectByProperty( 'id', id );

    	},

    	getObjectByName: function ( name ) {

    		return this.getObjectByProperty( 'name', name );

    	},

    	getObjectByProperty: function ( name, value ) {

    		if ( this[ name ] === value ) return this;

    		for ( var i = 0, l = this.children.length; i < l; i ++ ) {

    			var child = this.children[ i ];
    			var object = child.getObjectByProperty( name, value );

    			if ( object !== undefined ) {

    				return object;

    			}

    		}

    		return undefined;

    	},

    	getWorldPosition: function ( target ) {

    		if ( target === undefined ) {

    			console.warn( 'THREE.Object3D: .getWorldPosition() target is now required' );
    			target = new Vector3();

    		}

    		this.updateMatrixWorld( true );

    		return target.setFromMatrixPosition( this.matrixWorld );

    	},

    	getWorldQuaternion: function ( target ) {

    		if ( target === undefined ) {

    			console.warn( 'THREE.Object3D: .getWorldQuaternion() target is now required' );
    			target = new Quaternion();

    		}

    		this.updateMatrixWorld( true );

    		this.matrixWorld.decompose( _position, target, _scale );

    		return target;

    	},

    	getWorldScale: function ( target ) {

    		if ( target === undefined ) {

    			console.warn( 'THREE.Object3D: .getWorldScale() target is now required' );
    			target = new Vector3();

    		}

    		this.updateMatrixWorld( true );

    		this.matrixWorld.decompose( _position, _quaternion$2, target );

    		return target;

    	},

    	getWorldDirection: function ( target ) {

    		if ( target === undefined ) {

    			console.warn( 'THREE.Object3D: .getWorldDirection() target is now required' );
    			target = new Vector3();

    		}

    		this.updateMatrixWorld( true );

    		var e = this.matrixWorld.elements;

    		return target.set( e[ 8 ], e[ 9 ], e[ 10 ] ).normalize();

    	},

    	raycast: function () {},

    	traverse: function ( callback ) {

    		callback( this );

    		var children = this.children;

    		for ( var i = 0, l = children.length; i < l; i ++ ) {

    			children[ i ].traverse( callback );

    		}

    	},

    	traverseVisible: function ( callback ) {

    		if ( this.visible === false ) return;

    		callback( this );

    		var children = this.children;

    		for ( var i = 0, l = children.length; i < l; i ++ ) {

    			children[ i ].traverseVisible( callback );

    		}

    	},

    	traverseAncestors: function ( callback ) {

    		var parent = this.parent;

    		if ( parent !== null ) {

    			callback( parent );

    			parent.traverseAncestors( callback );

    		}

    	},

    	updateMatrix: function () {

    		this.matrix.compose( this.position, this.quaternion, this.scale );

    		this.matrixWorldNeedsUpdate = true;

    	},

    	updateMatrixWorld: function ( force ) {

    		if ( this.matrixAutoUpdate ) this.updateMatrix();

    		if ( this.matrixWorldNeedsUpdate || force ) {

    			if ( this.parent === null ) {

    				this.matrixWorld.copy( this.matrix );

    			} else {

    				this.matrixWorld.multiplyMatrices( this.parent.matrixWorld, this.matrix );

    			}

    			this.matrixWorldNeedsUpdate = false;

    			force = true;

    		}

    		// update children

    		var children = this.children;

    		for ( var i = 0, l = children.length; i < l; i ++ ) {

    			children[ i ].updateMatrixWorld( force );

    		}

    	},

    	updateWorldMatrix: function ( updateParents, updateChildren ) {

    		var parent = this.parent;

    		if ( updateParents === true && parent !== null ) {

    			parent.updateWorldMatrix( true, false );

    		}

    		if ( this.matrixAutoUpdate ) this.updateMatrix();

    		if ( this.parent === null ) {

    			this.matrixWorld.copy( this.matrix );

    		} else {

    			this.matrixWorld.multiplyMatrices( this.parent.matrixWorld, this.matrix );

    		}

    		// update children

    		if ( updateChildren === true ) {

    			var children = this.children;

    			for ( var i = 0, l = children.length; i < l; i ++ ) {

    				children[ i ].updateWorldMatrix( false, true );

    			}

    		}

    	},

    	toJSON: function ( meta ) {

    		// meta is a string when called from JSON.stringify
    		var isRootObject = ( meta === undefined || typeof meta === 'string' );

    		var output = {};

    		// meta is a hash used to collect geometries, materials.
    		// not providing it implies that this is the root object
    		// being serialized.
    		if ( isRootObject ) {

    			// initialize meta obj
    			meta = {
    				geometries: {},
    				materials: {},
    				textures: {},
    				images: {},
    				shapes: {}
    			};

    			output.metadata = {
    				version: 4.5,
    				type: 'Object',
    				generator: 'Object3D.toJSON'
    			};

    		}

    		// standard Object3D serialization

    		var object = {};

    		object.uuid = this.uuid;
    		object.type = this.type;

    		if ( this.name !== '' ) object.name = this.name;
    		if ( this.castShadow === true ) object.castShadow = true;
    		if ( this.receiveShadow === true ) object.receiveShadow = true;
    		if ( this.visible === false ) object.visible = false;
    		if ( this.frustumCulled === false ) object.frustumCulled = false;
    		if ( this.renderOrder !== 0 ) object.renderOrder = this.renderOrder;
    		if ( JSON.stringify( this.userData ) !== '{}' ) object.userData = this.userData;

    		object.layers = this.layers.mask;
    		object.matrix = this.matrix.toArray();

    		if ( this.matrixAutoUpdate === false ) object.matrixAutoUpdate = false;

    		// object specific properties

    		if ( this.isMesh && this.drawMode !== TrianglesDrawMode ) object.drawMode = this.drawMode;

    		//

    		function serialize( library, element ) {

    			if ( library[ element.uuid ] === undefined ) {

    				library[ element.uuid ] = element.toJSON( meta );

    			}

    			return element.uuid;

    		}

    		if ( this.isMesh || this.isLine || this.isPoints ) {

    			object.geometry = serialize( meta.geometries, this.geometry );

    			var parameters = this.geometry.parameters;

    			if ( parameters !== undefined && parameters.shapes !== undefined ) {

    				var shapes = parameters.shapes;

    				if ( Array.isArray( shapes ) ) {

    					for ( var i = 0, l = shapes.length; i < l; i ++ ) {

    						var shape = shapes[ i ];

    						serialize( meta.shapes, shape );

    					}

    				} else {

    					serialize( meta.shapes, shapes );

    				}

    			}

    		}

    		if ( this.material !== undefined ) {

    			if ( Array.isArray( this.material ) ) {

    				var uuids = [];

    				for ( var i = 0, l = this.material.length; i < l; i ++ ) {

    					uuids.push( serialize( meta.materials, this.material[ i ] ) );

    				}

    				object.material = uuids;

    			} else {

    				object.material = serialize( meta.materials, this.material );

    			}

    		}

    		//

    		if ( this.children.length > 0 ) {

    			object.children = [];

    			for ( var i = 0; i < this.children.length; i ++ ) {

    				object.children.push( this.children[ i ].toJSON( meta ).object );

    			}

    		}

    		if ( isRootObject ) {

    			var geometries = extractFromCache( meta.geometries );
    			var materials = extractFromCache( meta.materials );
    			var textures = extractFromCache( meta.textures );
    			var images = extractFromCache( meta.images );
    			var shapes = extractFromCache( meta.shapes );

    			if ( geometries.length > 0 ) output.geometries = geometries;
    			if ( materials.length > 0 ) output.materials = materials;
    			if ( textures.length > 0 ) output.textures = textures;
    			if ( images.length > 0 ) output.images = images;
    			if ( shapes.length > 0 ) output.shapes = shapes;

    		}

    		output.object = object;

    		return output;

    		// extract data from the cache hash
    		// remove metadata on each item
    		// and return as array
    		function extractFromCache( cache ) {

    			var values = [];
    			for ( var key in cache ) {

    				var data = cache[ key ];
    				delete data.metadata;
    				values.push( data );

    			}
    			return values;

    		}

    	},

    	clone: function ( recursive ) {

    		return new this.constructor().copy( this, recursive );

    	},

    	copy: function ( source, recursive ) {

    		if ( recursive === undefined ) recursive = true;

    		this.name = source.name;

    		this.up.copy( source.up );

    		this.position.copy( source.position );
    		this.quaternion.copy( source.quaternion );
    		this.scale.copy( source.scale );

    		this.matrix.copy( source.matrix );
    		this.matrixWorld.copy( source.matrixWorld );

    		this.matrixAutoUpdate = source.matrixAutoUpdate;
    		this.matrixWorldNeedsUpdate = source.matrixWorldNeedsUpdate;

    		this.layers.mask = source.layers.mask;
    		this.visible = source.visible;

    		this.castShadow = source.castShadow;
    		this.receiveShadow = source.receiveShadow;

    		this.frustumCulled = source.frustumCulled;
    		this.renderOrder = source.renderOrder;

    		this.userData = JSON.parse( JSON.stringify( source.userData ) );

    		if ( recursive === true ) {

    			for ( var i = 0; i < source.children.length; i ++ ) {

    				var child = source.children[ i ];
    				this.add( child.clone() );

    			}

    		}

    		return this;

    	}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function Scene() {

    	Object3D.call( this );

    	this.type = 'Scene';

    	this.background = null;
    	this.fog = null;
    	this.overrideMaterial = null;

    	this.autoUpdate = true; // checked by the renderer

    	if ( typeof __THREE_DEVTOOLS__ !== 'undefined' ) {

    		__THREE_DEVTOOLS__.dispatchEvent( new CustomEvent( 'observe', { detail: this } ) ); // eslint-disable-line no-undef

    	}

    }

    Scene.prototype = Object.assign( Object.create( Object3D.prototype ), {

    	constructor: Scene,

    	isScene: true,

    	copy: function ( source, recursive ) {

    		Object3D.prototype.copy.call( this, source, recursive );

    		if ( source.background !== null ) this.background = source.background.clone();
    		if ( source.fog !== null ) this.fog = source.fog.clone();
    		if ( source.overrideMaterial !== null ) this.overrideMaterial = source.overrideMaterial.clone();

    		this.autoUpdate = source.autoUpdate;
    		this.matrixAutoUpdate = source.matrixAutoUpdate;

    		return this;

    	},

    	toJSON: function ( meta ) {

    		var data = Object3D.prototype.toJSON.call( this, meta );

    		if ( this.background !== null ) data.object.background = this.background.toJSON( meta );
    		if ( this.fog !== null ) data.object.fog = this.fog.toJSON();

    		return data;

    	},

    	dispose: function () {

    		this.dispatchEvent( { type: 'dispose' } );

    	}

    } );

    var _points = [
    	new Vector3(),
    	new Vector3(),
    	new Vector3(),
    	new Vector3(),
    	new Vector3(),
    	new Vector3(),
    	new Vector3(),
    	new Vector3()
    ];
    var _vector$2 = new Vector3();

    // triangle centered vertices

    var _v0 = new Vector3();
    var _v1$2 = new Vector3();
    var _v2 = new Vector3();

    // triangle edge vectors

    var _f0 = new Vector3();
    var _f1 = new Vector3();
    var _f2 = new Vector3();

    var _center = new Vector3();
    var _extents = new Vector3();
    var _triangleNormal = new Vector3();
    var _testAxis = new Vector3();

    /**
     * @author bhouston / http://clara.io
     * @author WestLangley / http://github.com/WestLangley
     */

    function Box3( min, max ) {

    	this.min = ( min !== undefined ) ? min : new Vector3( + Infinity, + Infinity, + Infinity );
    	this.max = ( max !== undefined ) ? max : new Vector3( - Infinity, - Infinity, - Infinity );

    }

    Object.assign( Box3.prototype, {

    	isBox3: true,

    	set: function ( min, max ) {

    		this.min.copy( min );
    		this.max.copy( max );

    		return this;

    	},

    	setFromArray: function ( array ) {

    		var minX = + Infinity;
    		var minY = + Infinity;
    		var minZ = + Infinity;

    		var maxX = - Infinity;
    		var maxY = - Infinity;
    		var maxZ = - Infinity;

    		for ( var i = 0, l = array.length; i < l; i += 3 ) {

    			var x = array[ i ];
    			var y = array[ i + 1 ];
    			var z = array[ i + 2 ];

    			if ( x < minX ) minX = x;
    			if ( y < minY ) minY = y;
    			if ( z < minZ ) minZ = z;

    			if ( x > maxX ) maxX = x;
    			if ( y > maxY ) maxY = y;
    			if ( z > maxZ ) maxZ = z;

    		}

    		this.min.set( minX, minY, minZ );
    		this.max.set( maxX, maxY, maxZ );

    		return this;

    	},

    	setFromBufferAttribute: function ( attribute ) {

    		var minX = + Infinity;
    		var minY = + Infinity;
    		var minZ = + Infinity;

    		var maxX = - Infinity;
    		var maxY = - Infinity;
    		var maxZ = - Infinity;

    		for ( var i = 0, l = attribute.count; i < l; i ++ ) {

    			var x = attribute.getX( i );
    			var y = attribute.getY( i );
    			var z = attribute.getZ( i );

    			if ( x < minX ) minX = x;
    			if ( y < minY ) minY = y;
    			if ( z < minZ ) minZ = z;

    			if ( x > maxX ) maxX = x;
    			if ( y > maxY ) maxY = y;
    			if ( z > maxZ ) maxZ = z;

    		}

    		this.min.set( minX, minY, minZ );
    		this.max.set( maxX, maxY, maxZ );

    		return this;

    	},

    	setFromPoints: function ( points ) {

    		this.makeEmpty();

    		for ( var i = 0, il = points.length; i < il; i ++ ) {

    			this.expandByPoint( points[ i ] );

    		}

    		return this;

    	},

    	setFromCenterAndSize: function ( center, size ) {

    		var halfSize = _vector$2.copy( size ).multiplyScalar( 0.5 );

    		this.min.copy( center ).sub( halfSize );
    		this.max.copy( center ).add( halfSize );

    		return this;

    	},

    	setFromObject: function ( object ) {

    		this.makeEmpty();

    		return this.expandByObject( object );

    	},

    	clone: function () {

    		return new this.constructor().copy( this );

    	},

    	copy: function ( box ) {

    		this.min.copy( box.min );
    		this.max.copy( box.max );

    		return this;

    	},

    	makeEmpty: function () {

    		this.min.x = this.min.y = this.min.z = + Infinity;
    		this.max.x = this.max.y = this.max.z = - Infinity;

    		return this;

    	},

    	isEmpty: function () {

    		// this is a more robust check for empty than ( volume <= 0 ) because volume can get positive with two negative axes

    		return ( this.max.x < this.min.x ) || ( this.max.y < this.min.y ) || ( this.max.z < this.min.z );

    	},

    	getCenter: function ( target ) {

    		if ( target === undefined ) {

    			console.warn( 'THREE.Box3: .getCenter() target is now required' );
    			target = new Vector3();

    		}

    		return this.isEmpty() ? target.set( 0, 0, 0 ) : target.addVectors( this.min, this.max ).multiplyScalar( 0.5 );

    	},

    	getSize: function ( target ) {

    		if ( target === undefined ) {

    			console.warn( 'THREE.Box3: .getSize() target is now required' );
    			target = new Vector3();

    		}

    		return this.isEmpty() ? target.set( 0, 0, 0 ) : target.subVectors( this.max, this.min );

    	},

    	expandByPoint: function ( point ) {

    		this.min.min( point );
    		this.max.max( point );

    		return this;

    	},

    	expandByVector: function ( vector ) {

    		this.min.sub( vector );
    		this.max.add( vector );

    		return this;

    	},

    	expandByScalar: function ( scalar ) {

    		this.min.addScalar( - scalar );
    		this.max.addScalar( scalar );

    		return this;

    	},

    	expandByObject: function ( object ) {

    		var i, l;

    		// Computes the world-axis-aligned bounding box of an object (including its children),
    		// accounting for both the object's, and children's, world transforms

    		object.updateWorldMatrix( false, false );

    		var geometry = object.geometry;

    		if ( geometry !== undefined ) {

    			if ( geometry.isGeometry ) {

    				var vertices = geometry.vertices;

    				for ( i = 0, l = vertices.length; i < l; i ++ ) {

    					_vector$2.copy( vertices[ i ] );
    					_vector$2.applyMatrix4( object.matrixWorld );

    					this.expandByPoint( _vector$2 );

    				}

    			} else if ( geometry.isBufferGeometry ) {

    				var attribute = geometry.attributes.position;

    				if ( attribute !== undefined ) {

    					for ( i = 0, l = attribute.count; i < l; i ++ ) {

    						_vector$2.fromBufferAttribute( attribute, i ).applyMatrix4( object.matrixWorld );

    						this.expandByPoint( _vector$2 );

    					}

    				}

    			}

    		}

    		//

    		var children = object.children;

    		for ( i = 0, l = children.length; i < l; i ++ ) {

    			this.expandByObject( children[ i ] );

    		}

    		return this;

    	},

    	containsPoint: function ( point ) {

    		return point.x < this.min.x || point.x > this.max.x ||
    			point.y < this.min.y || point.y > this.max.y ||
    			point.z < this.min.z || point.z > this.max.z ? false : true;

    	},

    	containsBox: function ( box ) {

    		return this.min.x <= box.min.x && box.max.x <= this.max.x &&
    			this.min.y <= box.min.y && box.max.y <= this.max.y &&
    			this.min.z <= box.min.z && box.max.z <= this.max.z;

    	},

    	getParameter: function ( point, target ) {

    		// This can potentially have a divide by zero if the box
    		// has a size dimension of 0.

    		if ( target === undefined ) {

    			console.warn( 'THREE.Box3: .getParameter() target is now required' );
    			target = new Vector3();

    		}

    		return target.set(
    			( point.x - this.min.x ) / ( this.max.x - this.min.x ),
    			( point.y - this.min.y ) / ( this.max.y - this.min.y ),
    			( point.z - this.min.z ) / ( this.max.z - this.min.z )
    		);

    	},

    	intersectsBox: function ( box ) {

    		// using 6 splitting planes to rule out intersections.
    		return box.max.x < this.min.x || box.min.x > this.max.x ||
    			box.max.y < this.min.y || box.min.y > this.max.y ||
    			box.max.z < this.min.z || box.min.z > this.max.z ? false : true;

    	},

    	intersectsSphere: function ( sphere ) {

    		// Find the point on the AABB closest to the sphere center.
    		this.clampPoint( sphere.center, _vector$2 );

    		// If that point is inside the sphere, the AABB and sphere intersect.
    		return _vector$2.distanceToSquared( sphere.center ) <= ( sphere.radius * sphere.radius );

    	},

    	intersectsPlane: function ( plane ) {

    		// We compute the minimum and maximum dot product values. If those values
    		// are on the same side (back or front) of the plane, then there is no intersection.

    		var min, max;

    		if ( plane.normal.x > 0 ) {

    			min = plane.normal.x * this.min.x;
    			max = plane.normal.x * this.max.x;

    		} else {

    			min = plane.normal.x * this.max.x;
    			max = plane.normal.x * this.min.x;

    		}

    		if ( plane.normal.y > 0 ) {

    			min += plane.normal.y * this.min.y;
    			max += plane.normal.y * this.max.y;

    		} else {

    			min += plane.normal.y * this.max.y;
    			max += plane.normal.y * this.min.y;

    		}

    		if ( plane.normal.z > 0 ) {

    			min += plane.normal.z * this.min.z;
    			max += plane.normal.z * this.max.z;

    		} else {

    			min += plane.normal.z * this.max.z;
    			max += plane.normal.z * this.min.z;

    		}

    		return ( min <= - plane.constant && max >= - plane.constant );

    	},

    	intersectsTriangle: function ( triangle ) {

    		if ( this.isEmpty() ) {

    			return false;

    		}

    		// compute box center and extents
    		this.getCenter( _center );
    		_extents.subVectors( this.max, _center );

    		// translate triangle to aabb origin
    		_v0.subVectors( triangle.a, _center );
    		_v1$2.subVectors( triangle.b, _center );
    		_v2.subVectors( triangle.c, _center );

    		// compute edge vectors for triangle
    		_f0.subVectors( _v1$2, _v0 );
    		_f1.subVectors( _v2, _v1$2 );
    		_f2.subVectors( _v0, _v2 );

    		// test against axes that are given by cross product combinations of the edges of the triangle and the edges of the aabb
    		// make an axis testing of each of the 3 sides of the aabb against each of the 3 sides of the triangle = 9 axis of separation
    		// axis_ij = u_i x f_j (u0, u1, u2 = face normals of aabb = x,y,z axes vectors since aabb is axis aligned)
    		var axes = [
    			0, - _f0.z, _f0.y, 0, - _f1.z, _f1.y, 0, - _f2.z, _f2.y,
    			_f0.z, 0, - _f0.x, _f1.z, 0, - _f1.x, _f2.z, 0, - _f2.x,
    			- _f0.y, _f0.x, 0, - _f1.y, _f1.x, 0, - _f2.y, _f2.x, 0
    		];
    		if ( ! satForAxes( axes, _v0, _v1$2, _v2, _extents ) ) {

    			return false;

    		}

    		// test 3 face normals from the aabb
    		axes = [ 1, 0, 0, 0, 1, 0, 0, 0, 1 ];
    		if ( ! satForAxes( axes, _v0, _v1$2, _v2, _extents ) ) {

    			return false;

    		}

    		// finally testing the face normal of the triangle
    		// use already existing triangle edge vectors here
    		_triangleNormal.crossVectors( _f0, _f1 );
    		axes = [ _triangleNormal.x, _triangleNormal.y, _triangleNormal.z ];

    		return satForAxes( axes, _v0, _v1$2, _v2, _extents );

    	},

    	clampPoint: function ( point, target ) {

    		if ( target === undefined ) {

    			console.warn( 'THREE.Box3: .clampPoint() target is now required' );
    			target = new Vector3();

    		}

    		return target.copy( point ).clamp( this.min, this.max );

    	},

    	distanceToPoint: function ( point ) {

    		var clampedPoint = _vector$2.copy( point ).clamp( this.min, this.max );

    		return clampedPoint.sub( point ).length();

    	},

    	getBoundingSphere: function ( target ) {

    		if ( target === undefined ) {

    			console.error( 'THREE.Box3: .getBoundingSphere() target is now required' );
    			//target = new Sphere(); // removed to avoid cyclic dependency

    		}

    		this.getCenter( target.center );

    		target.radius = this.getSize( _vector$2 ).length() * 0.5;

    		return target;

    	},

    	intersect: function ( box ) {

    		this.min.max( box.min );
    		this.max.min( box.max );

    		// ensure that if there is no overlap, the result is fully empty, not slightly empty with non-inf/+inf values that will cause subsequence intersects to erroneously return valid values.
    		if ( this.isEmpty() ) this.makeEmpty();

    		return this;

    	},

    	union: function ( box ) {

    		this.min.min( box.min );
    		this.max.max( box.max );

    		return this;

    	},

    	applyMatrix4: function ( matrix ) {

    		// transform of empty box is an empty box.
    		if ( this.isEmpty() ) return this;

    		// NOTE: I am using a binary pattern to specify all 2^3 combinations below
    		_points[ 0 ].set( this.min.x, this.min.y, this.min.z ).applyMatrix4( matrix ); // 000
    		_points[ 1 ].set( this.min.x, this.min.y, this.max.z ).applyMatrix4( matrix ); // 001
    		_points[ 2 ].set( this.min.x, this.max.y, this.min.z ).applyMatrix4( matrix ); // 010
    		_points[ 3 ].set( this.min.x, this.max.y, this.max.z ).applyMatrix4( matrix ); // 011
    		_points[ 4 ].set( this.max.x, this.min.y, this.min.z ).applyMatrix4( matrix ); // 100
    		_points[ 5 ].set( this.max.x, this.min.y, this.max.z ).applyMatrix4( matrix ); // 101
    		_points[ 6 ].set( this.max.x, this.max.y, this.min.z ).applyMatrix4( matrix ); // 110
    		_points[ 7 ].set( this.max.x, this.max.y, this.max.z ).applyMatrix4( matrix ); // 111

    		this.setFromPoints( _points );

    		return this;

    	},

    	translate: function ( offset ) {

    		this.min.add( offset );
    		this.max.add( offset );

    		return this;

    	},

    	equals: function ( box ) {

    		return box.min.equals( this.min ) && box.max.equals( this.max );

    	}

    } );

    function satForAxes( axes, v0, v1, v2, extents ) {

    	var i, j;

    	for ( i = 0, j = axes.length - 3; i <= j; i += 3 ) {

    		_testAxis.fromArray( axes, i );
    		// project the aabb onto the seperating axis
    		var r = extents.x * Math.abs( _testAxis.x ) + extents.y * Math.abs( _testAxis.y ) + extents.z * Math.abs( _testAxis.z );
    		// project all 3 vertices of the triangle onto the seperating axis
    		var p0 = v0.dot( _testAxis );
    		var p1 = v1.dot( _testAxis );
    		var p2 = v2.dot( _testAxis );
    		// actual test, basically see if either of the most extreme of the triangle points intersects r
    		if ( Math.max( - Math.max( p0, p1, p2 ), Math.min( p0, p1, p2 ) ) > r ) {

    			// points of the projected triangle are outside the projected half-length of the aabb
    			// the axis is seperating and we can exit
    			return false;

    		}

    	}

    	return true;

    }

    var _box = new Box3();

    /**
     * @author bhouston / http://clara.io
     * @author mrdoob / http://mrdoob.com/
     */

    function Sphere( center, radius ) {

    	this.center = ( center !== undefined ) ? center : new Vector3();
    	this.radius = ( radius !== undefined ) ? radius : 0;

    }

    Object.assign( Sphere.prototype, {

    	set: function ( center, radius ) {

    		this.center.copy( center );
    		this.radius = radius;

    		return this;

    	},

    	setFromPoints: function ( points, optionalCenter ) {

    		var center = this.center;

    		if ( optionalCenter !== undefined ) {

    			center.copy( optionalCenter );

    		} else {

    			_box.setFromPoints( points ).getCenter( center );

    		}

    		var maxRadiusSq = 0;

    		for ( var i = 0, il = points.length; i < il; i ++ ) {

    			maxRadiusSq = Math.max( maxRadiusSq, center.distanceToSquared( points[ i ] ) );

    		}

    		this.radius = Math.sqrt( maxRadiusSq );

    		return this;

    	},

    	clone: function () {

    		return new this.constructor().copy( this );

    	},

    	copy: function ( sphere ) {

    		this.center.copy( sphere.center );
    		this.radius = sphere.radius;

    		return this;

    	},

    	empty: function () {

    		return ( this.radius <= 0 );

    	},

    	containsPoint: function ( point ) {

    		return ( point.distanceToSquared( this.center ) <= ( this.radius * this.radius ) );

    	},

    	distanceToPoint: function ( point ) {

    		return ( point.distanceTo( this.center ) - this.radius );

    	},

    	intersectsSphere: function ( sphere ) {

    		var radiusSum = this.radius + sphere.radius;

    		return sphere.center.distanceToSquared( this.center ) <= ( radiusSum * radiusSum );

    	},

    	intersectsBox: function ( box ) {

    		return box.intersectsSphere( this );

    	},

    	intersectsPlane: function ( plane ) {

    		return Math.abs( plane.distanceToPoint( this.center ) ) <= this.radius;

    	},

    	clampPoint: function ( point, target ) {

    		var deltaLengthSq = this.center.distanceToSquared( point );

    		if ( target === undefined ) {

    			console.warn( 'THREE.Sphere: .clampPoint() target is now required' );
    			target = new Vector3();

    		}

    		target.copy( point );

    		if ( deltaLengthSq > ( this.radius * this.radius ) ) {

    			target.sub( this.center ).normalize();
    			target.multiplyScalar( this.radius ).add( this.center );

    		}

    		return target;

    	},

    	getBoundingBox: function ( target ) {

    		if ( target === undefined ) {

    			console.warn( 'THREE.Sphere: .getBoundingBox() target is now required' );
    			target = new Box3();

    		}

    		target.set( this.center, this.center );
    		target.expandByScalar( this.radius );

    		return target;

    	},

    	applyMatrix4: function ( matrix ) {

    		this.center.applyMatrix4( matrix );
    		this.radius = this.radius * matrix.getMaxScaleOnAxis();

    		return this;

    	},

    	translate: function ( offset ) {

    		this.center.add( offset );

    		return this;

    	},

    	equals: function ( sphere ) {

    		return sphere.center.equals( this.center ) && ( sphere.radius === this.radius );

    	}

    } );

    var _vector$3 = new Vector3();
    var _segCenter = new Vector3();
    var _segDir = new Vector3();
    var _diff = new Vector3();

    var _edge1 = new Vector3();
    var _edge2 = new Vector3();
    var _normal = new Vector3();

    /**
     * @author bhouston / http://clara.io
     */

    function Ray( origin, direction ) {

    	this.origin = ( origin !== undefined ) ? origin : new Vector3();
    	this.direction = ( direction !== undefined ) ? direction : new Vector3();

    }

    Object.assign( Ray.prototype, {

    	set: function ( origin, direction ) {

    		this.origin.copy( origin );
    		this.direction.copy( direction );

    		return this;

    	},

    	clone: function () {

    		return new this.constructor().copy( this );

    	},

    	copy: function ( ray ) {

    		this.origin.copy( ray.origin );
    		this.direction.copy( ray.direction );

    		return this;

    	},

    	at: function ( t, target ) {

    		if ( target === undefined ) {

    			console.warn( 'THREE.Ray: .at() target is now required' );
    			target = new Vector3();

    		}

    		return target.copy( this.direction ).multiplyScalar( t ).add( this.origin );

    	},

    	lookAt: function ( v ) {

    		this.direction.copy( v ).sub( this.origin ).normalize();

    		return this;

    	},

    	recast: function ( t ) {

    		this.origin.copy( this.at( t, _vector$3 ) );

    		return this;

    	},

    	closestPointToPoint: function ( point, target ) {

    		if ( target === undefined ) {

    			console.warn( 'THREE.Ray: .closestPointToPoint() target is now required' );
    			target = new Vector3();

    		}

    		target.subVectors( point, this.origin );

    		var directionDistance = target.dot( this.direction );

    		if ( directionDistance < 0 ) {

    			return target.copy( this.origin );

    		}

    		return target.copy( this.direction ).multiplyScalar( directionDistance ).add( this.origin );

    	},

    	distanceToPoint: function ( point ) {

    		return Math.sqrt( this.distanceSqToPoint( point ) );

    	},

    	distanceSqToPoint: function ( point ) {

    		var directionDistance = _vector$3.subVectors( point, this.origin ).dot( this.direction );

    		// point behind the ray

    		if ( directionDistance < 0 ) {

    			return this.origin.distanceToSquared( point );

    		}

    		_vector$3.copy( this.direction ).multiplyScalar( directionDistance ).add( this.origin );

    		return _vector$3.distanceToSquared( point );

    	},

    	distanceSqToSegment: function ( v0, v1, optionalPointOnRay, optionalPointOnSegment ) {

    		// from http://www.geometrictools.com/GTEngine/Include/Mathematics/GteDistRaySegment.h
    		// It returns the min distance between the ray and the segment
    		// defined by v0 and v1
    		// It can also set two optional targets :
    		// - The closest point on the ray
    		// - The closest point on the segment

    		_segCenter.copy( v0 ).add( v1 ).multiplyScalar( 0.5 );
    		_segDir.copy( v1 ).sub( v0 ).normalize();
    		_diff.copy( this.origin ).sub( _segCenter );

    		var segExtent = v0.distanceTo( v1 ) * 0.5;
    		var a01 = - this.direction.dot( _segDir );
    		var b0 = _diff.dot( this.direction );
    		var b1 = - _diff.dot( _segDir );
    		var c = _diff.lengthSq();
    		var det = Math.abs( 1 - a01 * a01 );
    		var s0, s1, sqrDist, extDet;

    		if ( det > 0 ) {

    			// The ray and segment are not parallel.

    			s0 = a01 * b1 - b0;
    			s1 = a01 * b0 - b1;
    			extDet = segExtent * det;

    			if ( s0 >= 0 ) {

    				if ( s1 >= - extDet ) {

    					if ( s1 <= extDet ) {

    						// region 0
    						// Minimum at interior points of ray and segment.

    						var invDet = 1 / det;
    						s0 *= invDet;
    						s1 *= invDet;
    						sqrDist = s0 * ( s0 + a01 * s1 + 2 * b0 ) + s1 * ( a01 * s0 + s1 + 2 * b1 ) + c;

    					} else {

    						// region 1

    						s1 = segExtent;
    						s0 = Math.max( 0, - ( a01 * s1 + b0 ) );
    						sqrDist = - s0 * s0 + s1 * ( s1 + 2 * b1 ) + c;

    					}

    				} else {

    					// region 5

    					s1 = - segExtent;
    					s0 = Math.max( 0, - ( a01 * s1 + b0 ) );
    					sqrDist = - s0 * s0 + s1 * ( s1 + 2 * b1 ) + c;

    				}

    			} else {

    				if ( s1 <= - extDet ) {

    					// region 4

    					s0 = Math.max( 0, - ( - a01 * segExtent + b0 ) );
    					s1 = ( s0 > 0 ) ? - segExtent : Math.min( Math.max( - segExtent, - b1 ), segExtent );
    					sqrDist = - s0 * s0 + s1 * ( s1 + 2 * b1 ) + c;

    				} else if ( s1 <= extDet ) {

    					// region 3

    					s0 = 0;
    					s1 = Math.min( Math.max( - segExtent, - b1 ), segExtent );
    					sqrDist = s1 * ( s1 + 2 * b1 ) + c;

    				} else {

    					// region 2

    					s0 = Math.max( 0, - ( a01 * segExtent + b0 ) );
    					s1 = ( s0 > 0 ) ? segExtent : Math.min( Math.max( - segExtent, - b1 ), segExtent );
    					sqrDist = - s0 * s0 + s1 * ( s1 + 2 * b1 ) + c;

    				}

    			}

    		} else {

    			// Ray and segment are parallel.

    			s1 = ( a01 > 0 ) ? - segExtent : segExtent;
    			s0 = Math.max( 0, - ( a01 * s1 + b0 ) );
    			sqrDist = - s0 * s0 + s1 * ( s1 + 2 * b1 ) + c;

    		}

    		if ( optionalPointOnRay ) {

    			optionalPointOnRay.copy( this.direction ).multiplyScalar( s0 ).add( this.origin );

    		}

    		if ( optionalPointOnSegment ) {

    			optionalPointOnSegment.copy( _segDir ).multiplyScalar( s1 ).add( _segCenter );

    		}

    		return sqrDist;

    	},

    	intersectSphere: function ( sphere, target ) {

    		_vector$3.subVectors( sphere.center, this.origin );
    		var tca = _vector$3.dot( this.direction );
    		var d2 = _vector$3.dot( _vector$3 ) - tca * tca;
    		var radius2 = sphere.radius * sphere.radius;

    		if ( d2 > radius2 ) return null;

    		var thc = Math.sqrt( radius2 - d2 );

    		// t0 = first intersect point - entrance on front of sphere
    		var t0 = tca - thc;

    		// t1 = second intersect point - exit point on back of sphere
    		var t1 = tca + thc;

    		// test to see if both t0 and t1 are behind the ray - if so, return null
    		if ( t0 < 0 && t1 < 0 ) return null;

    		// test to see if t0 is behind the ray:
    		// if it is, the ray is inside the sphere, so return the second exit point scaled by t1,
    		// in order to always return an intersect point that is in front of the ray.
    		if ( t0 < 0 ) return this.at( t1, target );

    		// else t0 is in front of the ray, so return the first collision point scaled by t0
    		return this.at( t0, target );

    	},

    	intersectsSphere: function ( sphere ) {

    		return this.distanceSqToPoint( sphere.center ) <= ( sphere.radius * sphere.radius );

    	},

    	distanceToPlane: function ( plane ) {

    		var denominator = plane.normal.dot( this.direction );

    		if ( denominator === 0 ) {

    			// line is coplanar, return origin
    			if ( plane.distanceToPoint( this.origin ) === 0 ) {

    				return 0;

    			}

    			// Null is preferable to undefined since undefined means.... it is undefined

    			return null;

    		}

    		var t = - ( this.origin.dot( plane.normal ) + plane.constant ) / denominator;

    		// Return if the ray never intersects the plane

    		return t >= 0 ? t : null;

    	},

    	intersectPlane: function ( plane, target ) {

    		var t = this.distanceToPlane( plane );

    		if ( t === null ) {

    			return null;

    		}

    		return this.at( t, target );

    	},

    	intersectsPlane: function ( plane ) {

    		// check if the ray lies on the plane first

    		var distToPoint = plane.distanceToPoint( this.origin );

    		if ( distToPoint === 0 ) {

    			return true;

    		}

    		var denominator = plane.normal.dot( this.direction );

    		if ( denominator * distToPoint < 0 ) {

    			return true;

    		}

    		// ray origin is behind the plane (and is pointing behind it)

    		return false;

    	},

    	intersectBox: function ( box, target ) {

    		var tmin, tmax, tymin, tymax, tzmin, tzmax;

    		var invdirx = 1 / this.direction.x,
    			invdiry = 1 / this.direction.y,
    			invdirz = 1 / this.direction.z;

    		var origin = this.origin;

    		if ( invdirx >= 0 ) {

    			tmin = ( box.min.x - origin.x ) * invdirx;
    			tmax = ( box.max.x - origin.x ) * invdirx;

    		} else {

    			tmin = ( box.max.x - origin.x ) * invdirx;
    			tmax = ( box.min.x - origin.x ) * invdirx;

    		}

    		if ( invdiry >= 0 ) {

    			tymin = ( box.min.y - origin.y ) * invdiry;
    			tymax = ( box.max.y - origin.y ) * invdiry;

    		} else {

    			tymin = ( box.max.y - origin.y ) * invdiry;
    			tymax = ( box.min.y - origin.y ) * invdiry;

    		}

    		if ( ( tmin > tymax ) || ( tymin > tmax ) ) return null;

    		// These lines also handle the case where tmin or tmax is NaN
    		// (result of 0 * Infinity). x !== x returns true if x is NaN

    		if ( tymin > tmin || tmin !== tmin ) tmin = tymin;

    		if ( tymax < tmax || tmax !== tmax ) tmax = tymax;

    		if ( invdirz >= 0 ) {

    			tzmin = ( box.min.z - origin.z ) * invdirz;
    			tzmax = ( box.max.z - origin.z ) * invdirz;

    		} else {

    			tzmin = ( box.max.z - origin.z ) * invdirz;
    			tzmax = ( box.min.z - origin.z ) * invdirz;

    		}

    		if ( ( tmin > tzmax ) || ( tzmin > tmax ) ) return null;

    		if ( tzmin > tmin || tmin !== tmin ) tmin = tzmin;

    		if ( tzmax < tmax || tmax !== tmax ) tmax = tzmax;

    		//return point closest to the ray (positive side)

    		if ( tmax < 0 ) return null;

    		return this.at( tmin >= 0 ? tmin : tmax, target );

    	},

    	intersectsBox: function ( box ) {

    		return this.intersectBox( box, _vector$3 ) !== null;

    	},

    	intersectTriangle: function ( a, b, c, backfaceCulling, target ) {

    		// Compute the offset origin, edges, and normal.

    		// from http://www.geometrictools.com/GTEngine/Include/Mathematics/GteIntrRay3Triangle3.h

    		_edge1.subVectors( b, a );
    		_edge2.subVectors( c, a );
    		_normal.crossVectors( _edge1, _edge2 );

    		// Solve Q + t*D = b1*E1 + b2*E2 (Q = kDiff, D = ray direction,
    		// E1 = kEdge1, E2 = kEdge2, N = Cross(E1,E2)) by
    		//   |Dot(D,N)|*b1 = sign(Dot(D,N))*Dot(D,Cross(Q,E2))
    		//   |Dot(D,N)|*b2 = sign(Dot(D,N))*Dot(D,Cross(E1,Q))
    		//   |Dot(D,N)|*t = -sign(Dot(D,N))*Dot(Q,N)
    		var DdN = this.direction.dot( _normal );
    		var sign;

    		if ( DdN > 0 ) {

    			if ( backfaceCulling ) return null;
    			sign = 1;

    		} else if ( DdN < 0 ) {

    			sign = - 1;
    			DdN = - DdN;

    		} else {

    			return null;

    		}

    		_diff.subVectors( this.origin, a );
    		var DdQxE2 = sign * this.direction.dot( _edge2.crossVectors( _diff, _edge2 ) );

    		// b1 < 0, no intersection
    		if ( DdQxE2 < 0 ) {

    			return null;

    		}

    		var DdE1xQ = sign * this.direction.dot( _edge1.cross( _diff ) );

    		// b2 < 0, no intersection
    		if ( DdE1xQ < 0 ) {

    			return null;

    		}

    		// b1+b2 > 1, no intersection
    		if ( DdQxE2 + DdE1xQ > DdN ) {

    			return null;

    		}

    		// Line intersects triangle, check if ray does.
    		var QdN = - sign * _diff.dot( _normal );

    		// t < 0, no intersection
    		if ( QdN < 0 ) {

    			return null;

    		}

    		// Ray intersects triangle.
    		return this.at( QdN / DdN, target );

    	},

    	applyMatrix4: function ( matrix4 ) {

    		this.origin.applyMatrix4( matrix4 );
    		this.direction.transformDirection( matrix4 );

    		return this;

    	},

    	equals: function ( ray ) {

    		return ray.origin.equals( this.origin ) && ray.direction.equals( this.direction );

    	}

    } );

    /**
     * @author bhouston / http://clara.io
     * @author mrdoob / http://mrdoob.com/
     */

    var _v0$1 = new Vector3();
    var _v1$3 = new Vector3();
    var _v2$1 = new Vector3();
    var _v3 = new Vector3();

    var _vab = new Vector3();
    var _vac = new Vector3();
    var _vbc = new Vector3();
    var _vap = new Vector3();
    var _vbp = new Vector3();
    var _vcp = new Vector3();

    function Triangle( a, b, c ) {

    	this.a = ( a !== undefined ) ? a : new Vector3();
    	this.b = ( b !== undefined ) ? b : new Vector3();
    	this.c = ( c !== undefined ) ? c : new Vector3();

    }

    Object.assign( Triangle, {

    	getNormal: function ( a, b, c, target ) {

    		if ( target === undefined ) {

    			console.warn( 'THREE.Triangle: .getNormal() target is now required' );
    			target = new Vector3();

    		}

    		target.subVectors( c, b );
    		_v0$1.subVectors( a, b );
    		target.cross( _v0$1 );

    		var targetLengthSq = target.lengthSq();
    		if ( targetLengthSq > 0 ) {

    			return target.multiplyScalar( 1 / Math.sqrt( targetLengthSq ) );

    		}

    		return target.set( 0, 0, 0 );

    	},

    	// static/instance method to calculate barycentric coordinates
    	// based on: http://www.blackpawn.com/texts/pointinpoly/default.html
    	getBarycoord: function ( point, a, b, c, target ) {

    		_v0$1.subVectors( c, a );
    		_v1$3.subVectors( b, a );
    		_v2$1.subVectors( point, a );

    		var dot00 = _v0$1.dot( _v0$1 );
    		var dot01 = _v0$1.dot( _v1$3 );
    		var dot02 = _v0$1.dot( _v2$1 );
    		var dot11 = _v1$3.dot( _v1$3 );
    		var dot12 = _v1$3.dot( _v2$1 );

    		var denom = ( dot00 * dot11 - dot01 * dot01 );

    		if ( target === undefined ) {

    			console.warn( 'THREE.Triangle: .getBarycoord() target is now required' );
    			target = new Vector3();

    		}

    		// collinear or singular triangle
    		if ( denom === 0 ) {

    			// arbitrary location outside of triangle?
    			// not sure if this is the best idea, maybe should be returning undefined
    			return target.set( - 2, - 1, - 1 );

    		}

    		var invDenom = 1 / denom;
    		var u = ( dot11 * dot02 - dot01 * dot12 ) * invDenom;
    		var v = ( dot00 * dot12 - dot01 * dot02 ) * invDenom;

    		// barycentric coordinates must always sum to 1
    		return target.set( 1 - u - v, v, u );

    	},

    	containsPoint: function ( point, a, b, c ) {

    		Triangle.getBarycoord( point, a, b, c, _v3 );

    		return ( _v3.x >= 0 ) && ( _v3.y >= 0 ) && ( ( _v3.x + _v3.y ) <= 1 );

    	},

    	getUV: function ( point, p1, p2, p3, uv1, uv2, uv3, target ) {

    		this.getBarycoord( point, p1, p2, p3, _v3 );

    		target.set( 0, 0 );
    		target.addScaledVector( uv1, _v3.x );
    		target.addScaledVector( uv2, _v3.y );
    		target.addScaledVector( uv3, _v3.z );

    		return target;

    	},

    	isFrontFacing: function ( a, b, c, direction ) {

    		_v0$1.subVectors( c, b );
    		_v1$3.subVectors( a, b );

    		// strictly front facing
    		return ( _v0$1.cross( _v1$3 ).dot( direction ) < 0 ) ? true : false;

    	}

    } );

    Object.assign( Triangle.prototype, {

    	set: function ( a, b, c ) {

    		this.a.copy( a );
    		this.b.copy( b );
    		this.c.copy( c );

    		return this;

    	},

    	setFromPointsAndIndices: function ( points, i0, i1, i2 ) {

    		this.a.copy( points[ i0 ] );
    		this.b.copy( points[ i1 ] );
    		this.c.copy( points[ i2 ] );

    		return this;

    	},

    	clone: function () {

    		return new this.constructor().copy( this );

    	},

    	copy: function ( triangle ) {

    		this.a.copy( triangle.a );
    		this.b.copy( triangle.b );
    		this.c.copy( triangle.c );

    		return this;

    	},

    	getArea: function () {

    		_v0$1.subVectors( this.c, this.b );
    		_v1$3.subVectors( this.a, this.b );

    		return _v0$1.cross( _v1$3 ).length() * 0.5;

    	},

    	getMidpoint: function ( target ) {

    		if ( target === undefined ) {

    			console.warn( 'THREE.Triangle: .getMidpoint() target is now required' );
    			target = new Vector3();

    		}

    		return target.addVectors( this.a, this.b ).add( this.c ).multiplyScalar( 1 / 3 );

    	},

    	getNormal: function ( target ) {

    		return Triangle.getNormal( this.a, this.b, this.c, target );

    	},

    	getPlane: function ( target ) {

    		if ( target === undefined ) {

    			console.warn( 'THREE.Triangle: .getPlane() target is now required' );
    			target = new Vector3();

    		}

    		return target.setFromCoplanarPoints( this.a, this.b, this.c );

    	},

    	getBarycoord: function ( point, target ) {

    		return Triangle.getBarycoord( point, this.a, this.b, this.c, target );

    	},

    	getUV: function ( point, uv1, uv2, uv3, target ) {

    		return Triangle.getUV( point, this.a, this.b, this.c, uv1, uv2, uv3, target );

    	},

    	containsPoint: function ( point ) {

    		return Triangle.containsPoint( point, this.a, this.b, this.c );

    	},

    	isFrontFacing: function ( direction ) {

    		return Triangle.isFrontFacing( this.a, this.b, this.c, direction );

    	},

    	intersectsBox: function ( box ) {

    		return box.intersectsTriangle( this );

    	},

    	closestPointToPoint: function ( p, target ) {

    		if ( target === undefined ) {

    			console.warn( 'THREE.Triangle: .closestPointToPoint() target is now required' );
    			target = new Vector3();

    		}

    		var a = this.a, b = this.b, c = this.c;
    		var v, w;

    		// algorithm thanks to Real-Time Collision Detection by Christer Ericson,
    		// published by Morgan Kaufmann Publishers, (c) 2005 Elsevier Inc.,
    		// under the accompanying license; see chapter 5.1.5 for detailed explanation.
    		// basically, we're distinguishing which of the voronoi regions of the triangle
    		// the point lies in with the minimum amount of redundant computation.

    		_vab.subVectors( b, a );
    		_vac.subVectors( c, a );
    		_vap.subVectors( p, a );
    		var d1 = _vab.dot( _vap );
    		var d2 = _vac.dot( _vap );
    		if ( d1 <= 0 && d2 <= 0 ) {

    			// vertex region of A; barycentric coords (1, 0, 0)
    			return target.copy( a );

    		}

    		_vbp.subVectors( p, b );
    		var d3 = _vab.dot( _vbp );
    		var d4 = _vac.dot( _vbp );
    		if ( d3 >= 0 && d4 <= d3 ) {

    			// vertex region of B; barycentric coords (0, 1, 0)
    			return target.copy( b );

    		}

    		var vc = d1 * d4 - d3 * d2;
    		if ( vc <= 0 && d1 >= 0 && d3 <= 0 ) {

    			v = d1 / ( d1 - d3 );
    			// edge region of AB; barycentric coords (1-v, v, 0)
    			return target.copy( a ).addScaledVector( _vab, v );

    		}

    		_vcp.subVectors( p, c );
    		var d5 = _vab.dot( _vcp );
    		var d6 = _vac.dot( _vcp );
    		if ( d6 >= 0 && d5 <= d6 ) {

    			// vertex region of C; barycentric coords (0, 0, 1)
    			return target.copy( c );

    		}

    		var vb = d5 * d2 - d1 * d6;
    		if ( vb <= 0 && d2 >= 0 && d6 <= 0 ) {

    			w = d2 / ( d2 - d6 );
    			// edge region of AC; barycentric coords (1-w, 0, w)
    			return target.copy( a ).addScaledVector( _vac, w );

    		}

    		var va = d3 * d6 - d5 * d4;
    		if ( va <= 0 && ( d4 - d3 ) >= 0 && ( d5 - d6 ) >= 0 ) {

    			_vbc.subVectors( c, b );
    			w = ( d4 - d3 ) / ( ( d4 - d3 ) + ( d5 - d6 ) );
    			// edge region of BC; barycentric coords (0, 1-w, w)
    			return target.copy( b ).addScaledVector( _vbc, w ); // edge region of BC

    		}

    		// face region
    		var denom = 1 / ( va + vb + vc );
    		// u = va * denom
    		v = vb * denom;
    		w = vc * denom;

    		return target.copy( a ).addScaledVector( _vab, v ).addScaledVector( _vac, w );

    	},

    	equals: function ( triangle ) {

    		return triangle.a.equals( this.a ) && triangle.b.equals( this.b ) && triangle.c.equals( this.c );

    	}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    var _colorKeywords = { 'aliceblue': 0xF0F8FF, 'antiquewhite': 0xFAEBD7, 'aqua': 0x00FFFF, 'aquamarine': 0x7FFFD4, 'azure': 0xF0FFFF,
    	'beige': 0xF5F5DC, 'bisque': 0xFFE4C4, 'black': 0x000000, 'blanchedalmond': 0xFFEBCD, 'blue': 0x0000FF, 'blueviolet': 0x8A2BE2,
    	'brown': 0xA52A2A, 'burlywood': 0xDEB887, 'cadetblue': 0x5F9EA0, 'chartreuse': 0x7FFF00, 'chocolate': 0xD2691E, 'coral': 0xFF7F50,
    	'cornflowerblue': 0x6495ED, 'cornsilk': 0xFFF8DC, 'crimson': 0xDC143C, 'cyan': 0x00FFFF, 'darkblue': 0x00008B, 'darkcyan': 0x008B8B,
    	'darkgoldenrod': 0xB8860B, 'darkgray': 0xA9A9A9, 'darkgreen': 0x006400, 'darkgrey': 0xA9A9A9, 'darkkhaki': 0xBDB76B, 'darkmagenta': 0x8B008B,
    	'darkolivegreen': 0x556B2F, 'darkorange': 0xFF8C00, 'darkorchid': 0x9932CC, 'darkred': 0x8B0000, 'darksalmon': 0xE9967A, 'darkseagreen': 0x8FBC8F,
    	'darkslateblue': 0x483D8B, 'darkslategray': 0x2F4F4F, 'darkslategrey': 0x2F4F4F, 'darkturquoise': 0x00CED1, 'darkviolet': 0x9400D3,
    	'deeppink': 0xFF1493, 'deepskyblue': 0x00BFFF, 'dimgray': 0x696969, 'dimgrey': 0x696969, 'dodgerblue': 0x1E90FF, 'firebrick': 0xB22222,
    	'floralwhite': 0xFFFAF0, 'forestgreen': 0x228B22, 'fuchsia': 0xFF00FF, 'gainsboro': 0xDCDCDC, 'ghostwhite': 0xF8F8FF, 'gold': 0xFFD700,
    	'goldenrod': 0xDAA520, 'gray': 0x808080, 'green': 0x008000, 'greenyellow': 0xADFF2F, 'grey': 0x808080, 'honeydew': 0xF0FFF0, 'hotpink': 0xFF69B4,
    	'indianred': 0xCD5C5C, 'indigo': 0x4B0082, 'ivory': 0xFFFFF0, 'khaki': 0xF0E68C, 'lavender': 0xE6E6FA, 'lavenderblush': 0xFFF0F5, 'lawngreen': 0x7CFC00,
    	'lemonchiffon': 0xFFFACD, 'lightblue': 0xADD8E6, 'lightcoral': 0xF08080, 'lightcyan': 0xE0FFFF, 'lightgoldenrodyellow': 0xFAFAD2, 'lightgray': 0xD3D3D3,
    	'lightgreen': 0x90EE90, 'lightgrey': 0xD3D3D3, 'lightpink': 0xFFB6C1, 'lightsalmon': 0xFFA07A, 'lightseagreen': 0x20B2AA, 'lightskyblue': 0x87CEFA,
    	'lightslategray': 0x778899, 'lightslategrey': 0x778899, 'lightsteelblue': 0xB0C4DE, 'lightyellow': 0xFFFFE0, 'lime': 0x00FF00, 'limegreen': 0x32CD32,
    	'linen': 0xFAF0E6, 'magenta': 0xFF00FF, 'maroon': 0x800000, 'mediumaquamarine': 0x66CDAA, 'mediumblue': 0x0000CD, 'mediumorchid': 0xBA55D3,
    	'mediumpurple': 0x9370DB, 'mediumseagreen': 0x3CB371, 'mediumslateblue': 0x7B68EE, 'mediumspringgreen': 0x00FA9A, 'mediumturquoise': 0x48D1CC,
    	'mediumvioletred': 0xC71585, 'midnightblue': 0x191970, 'mintcream': 0xF5FFFA, 'mistyrose': 0xFFE4E1, 'moccasin': 0xFFE4B5, 'navajowhite': 0xFFDEAD,
    	'navy': 0x000080, 'oldlace': 0xFDF5E6, 'olive': 0x808000, 'olivedrab': 0x6B8E23, 'orange': 0xFFA500, 'orangered': 0xFF4500, 'orchid': 0xDA70D6,
    	'palegoldenrod': 0xEEE8AA, 'palegreen': 0x98FB98, 'paleturquoise': 0xAFEEEE, 'palevioletred': 0xDB7093, 'papayawhip': 0xFFEFD5, 'peachpuff': 0xFFDAB9,
    	'peru': 0xCD853F, 'pink': 0xFFC0CB, 'plum': 0xDDA0DD, 'powderblue': 0xB0E0E6, 'purple': 0x800080, 'rebeccapurple': 0x663399, 'red': 0xFF0000, 'rosybrown': 0xBC8F8F,
    	'royalblue': 0x4169E1, 'saddlebrown': 0x8B4513, 'salmon': 0xFA8072, 'sandybrown': 0xF4A460, 'seagreen': 0x2E8B57, 'seashell': 0xFFF5EE,
    	'sienna': 0xA0522D, 'silver': 0xC0C0C0, 'skyblue': 0x87CEEB, 'slateblue': 0x6A5ACD, 'slategray': 0x708090, 'slategrey': 0x708090, 'snow': 0xFFFAFA,
    	'springgreen': 0x00FF7F, 'steelblue': 0x4682B4, 'tan': 0xD2B48C, 'teal': 0x008080, 'thistle': 0xD8BFD8, 'tomato': 0xFF6347, 'turquoise': 0x40E0D0,
    	'violet': 0xEE82EE, 'wheat': 0xF5DEB3, 'white': 0xFFFFFF, 'whitesmoke': 0xF5F5F5, 'yellow': 0xFFFF00, 'yellowgreen': 0x9ACD32 };

    var _hslA = { h: 0, s: 0, l: 0 };
    var _hslB = { h: 0, s: 0, l: 0 };

    function Color$7( r, g, b ) {

    	if ( g === undefined && b === undefined ) {

    		// r is THREE.Color, hex or string
    		return this.set( r );

    	}

    	return this.setRGB( r, g, b );

    }

    function hue2rgb( p, q, t ) {

    	if ( t < 0 ) t += 1;
    	if ( t > 1 ) t -= 1;
    	if ( t < 1 / 6 ) return p + ( q - p ) * 6 * t;
    	if ( t < 1 / 2 ) return q;
    	if ( t < 2 / 3 ) return p + ( q - p ) * 6 * ( 2 / 3 - t );
    	return p;

    }

    function SRGBToLinear( c ) {

    	return ( c < 0.04045 ) ? c * 0.0773993808 : Math.pow( c * 0.9478672986 + 0.0521327014, 2.4 );

    }

    function LinearToSRGB( c ) {

    	return ( c < 0.0031308 ) ? c * 12.92 : 1.055 * ( Math.pow( c, 0.41666 ) ) - 0.055;

    }

    Object.assign( Color$7.prototype, {

    	isColor: true,

    	r: 1, g: 1, b: 1,

    	set: function ( value ) {

    		if ( value && value.isColor ) {

    			this.copy( value );

    		} else if ( typeof value === 'number' ) {

    			this.setHex( value );

    		} else if ( typeof value === 'string' ) {

    			this.setStyle( value );

    		}

    		return this;

    	},

    	setScalar: function ( scalar ) {

    		this.r = scalar;
    		this.g = scalar;
    		this.b = scalar;

    		return this;

    	},

    	setHex: function ( hex ) {

    		hex = Math.floor( hex );

    		this.r = ( hex >> 16 & 255 ) / 255;
    		this.g = ( hex >> 8 & 255 ) / 255;
    		this.b = ( hex & 255 ) / 255;

    		return this;

    	},

    	setRGB: function ( r, g, b ) {

    		this.r = r;
    		this.g = g;
    		this.b = b;

    		return this;

    	},

    	setHSL: function ( h, s, l ) {

    		// h,s,l ranges are in 0.0 - 1.0
    		h = _Math.euclideanModulo( h, 1 );
    		s = _Math.clamp( s, 0, 1 );
    		l = _Math.clamp( l, 0, 1 );

    		if ( s === 0 ) {

    			this.r = this.g = this.b = l;

    		} else {

    			var p = l <= 0.5 ? l * ( 1 + s ) : l + s - ( l * s );
    			var q = ( 2 * l ) - p;

    			this.r = hue2rgb( q, p, h + 1 / 3 );
    			this.g = hue2rgb( q, p, h );
    			this.b = hue2rgb( q, p, h - 1 / 3 );

    		}

    		return this;

    	},

    	setStyle: function ( style ) {

    		function handleAlpha( string ) {

    			if ( string === undefined ) return;

    			if ( parseFloat( string ) < 1 ) {

    				console.warn( 'THREE.Color: Alpha component of ' + style + ' will be ignored.' );

    			}

    		}


    		var m;

    		if ( m = /^((?:rgb|hsl)a?)\(\s*([^\)]*)\)/.exec( style ) ) {

    			// rgb / hsl

    			var color;
    			var name = m[ 1 ];
    			var components = m[ 2 ];

    			switch ( name ) {

    				case 'rgb':
    				case 'rgba':

    					if ( color = /^(\d+)\s*,\s*(\d+)\s*,\s*(\d+)\s*(,\s*([0-9]*\.?[0-9]+)\s*)?$/.exec( components ) ) {

    						// rgb(255,0,0) rgba(255,0,0,0.5)
    						this.r = Math.min( 255, parseInt( color[ 1 ], 10 ) ) / 255;
    						this.g = Math.min( 255, parseInt( color[ 2 ], 10 ) ) / 255;
    						this.b = Math.min( 255, parseInt( color[ 3 ], 10 ) ) / 255;

    						handleAlpha( color[ 5 ] );

    						return this;

    					}

    					if ( color = /^(\d+)\%\s*,\s*(\d+)\%\s*,\s*(\d+)\%\s*(,\s*([0-9]*\.?[0-9]+)\s*)?$/.exec( components ) ) {

    						// rgb(100%,0%,0%) rgba(100%,0%,0%,0.5)
    						this.r = Math.min( 100, parseInt( color[ 1 ], 10 ) ) / 100;
    						this.g = Math.min( 100, parseInt( color[ 2 ], 10 ) ) / 100;
    						this.b = Math.min( 100, parseInt( color[ 3 ], 10 ) ) / 100;

    						handleAlpha( color[ 5 ] );

    						return this;

    					}

    					break;

    				case 'hsl':
    				case 'hsla':

    					if ( color = /^([0-9]*\.?[0-9]+)\s*,\s*(\d+)\%\s*,\s*(\d+)\%\s*(,\s*([0-9]*\.?[0-9]+)\s*)?$/.exec( components ) ) {

    						// hsl(120,50%,50%) hsla(120,50%,50%,0.5)
    						var h = parseFloat( color[ 1 ] ) / 360;
    						var s = parseInt( color[ 2 ], 10 ) / 100;
    						var l = parseInt( color[ 3 ], 10 ) / 100;

    						handleAlpha( color[ 5 ] );

    						return this.setHSL( h, s, l );

    					}

    					break;

    			}

    		} else if ( m = /^\#([A-Fa-f0-9]+)$/.exec( style ) ) {

    			// hex color

    			var hex = m[ 1 ];
    			var size = hex.length;

    			if ( size === 3 ) {

    				// #ff0
    				this.r = parseInt( hex.charAt( 0 ) + hex.charAt( 0 ), 16 ) / 255;
    				this.g = parseInt( hex.charAt( 1 ) + hex.charAt( 1 ), 16 ) / 255;
    				this.b = parseInt( hex.charAt( 2 ) + hex.charAt( 2 ), 16 ) / 255;

    				return this;

    			} else if ( size === 6 ) {

    				// #ff0000
    				this.r = parseInt( hex.charAt( 0 ) + hex.charAt( 1 ), 16 ) / 255;
    				this.g = parseInt( hex.charAt( 2 ) + hex.charAt( 3 ), 16 ) / 255;
    				this.b = parseInt( hex.charAt( 4 ) + hex.charAt( 5 ), 16 ) / 255;

    				return this;

    			}

    		}

    		if ( style && style.length > 0 ) {

    			// color keywords
    			var hex = _colorKeywords[ style ];

    			if ( hex !== undefined ) {

    				// red
    				this.setHex( hex );

    			} else {

    				// unknown color
    				console.warn( 'THREE.Color: Unknown color ' + style );

    			}

    		}

    		return this;

    	},

    	clone: function () {

    		return new this.constructor( this.r, this.g, this.b );

    	},

    	copy: function ( color ) {

    		this.r = color.r;
    		this.g = color.g;
    		this.b = color.b;

    		return this;

    	},

    	copyGammaToLinear: function ( color, gammaFactor ) {

    		if ( gammaFactor === undefined ) gammaFactor = 2.0;

    		this.r = Math.pow( color.r, gammaFactor );
    		this.g = Math.pow( color.g, gammaFactor );
    		this.b = Math.pow( color.b, gammaFactor );

    		return this;

    	},

    	copyLinearToGamma: function ( color, gammaFactor ) {

    		if ( gammaFactor === undefined ) gammaFactor = 2.0;

    		var safeInverse = ( gammaFactor > 0 ) ? ( 1.0 / gammaFactor ) : 1.0;

    		this.r = Math.pow( color.r, safeInverse );
    		this.g = Math.pow( color.g, safeInverse );
    		this.b = Math.pow( color.b, safeInverse );

    		return this;

    	},

    	convertGammaToLinear: function ( gammaFactor ) {

    		this.copyGammaToLinear( this, gammaFactor );

    		return this;

    	},

    	convertLinearToGamma: function ( gammaFactor ) {

    		this.copyLinearToGamma( this, gammaFactor );

    		return this;

    	},

    	copySRGBToLinear: function ( color ) {

    		this.r = SRGBToLinear( color.r );
    		this.g = SRGBToLinear( color.g );
    		this.b = SRGBToLinear( color.b );

    		return this;

    	},

    	copyLinearToSRGB: function ( color ) {

    		this.r = LinearToSRGB( color.r );
    		this.g = LinearToSRGB( color.g );
    		this.b = LinearToSRGB( color.b );

    		return this;

    	},

    	convertSRGBToLinear: function () {

    		this.copySRGBToLinear( this );

    		return this;

    	},

    	convertLinearToSRGB: function () {

    		this.copyLinearToSRGB( this );

    		return this;

    	},

    	getHex: function () {

    		return ( this.r * 255 ) << 16 ^ ( this.g * 255 ) << 8 ^ ( this.b * 255 ) << 0;

    	},

    	getHexString: function () {

    		return ( '000000' + this.getHex().toString( 16 ) ).slice( - 6 );

    	},

    	getHSL: function ( target ) {

    		// h,s,l ranges are in 0.0 - 1.0

    		if ( target === undefined ) {

    			console.warn( 'THREE.Color: .getHSL() target is now required' );
    			target = { h: 0, s: 0, l: 0 };

    		}

    		var r = this.r, g = this.g, b = this.b;

    		var max = Math.max( r, g, b );
    		var min = Math.min( r, g, b );

    		var hue, saturation;
    		var lightness = ( min + max ) / 2.0;

    		if ( min === max ) {

    			hue = 0;
    			saturation = 0;

    		} else {

    			var delta = max - min;

    			saturation = lightness <= 0.5 ? delta / ( max + min ) : delta / ( 2 - max - min );

    			switch ( max ) {

    				case r: hue = ( g - b ) / delta + ( g < b ? 6 : 0 ); break;
    				case g: hue = ( b - r ) / delta + 2; break;
    				case b: hue = ( r - g ) / delta + 4; break;

    			}

    			hue /= 6;

    		}

    		target.h = hue;
    		target.s = saturation;
    		target.l = lightness;

    		return target;

    	},

    	getStyle: function () {

    		return 'rgb(' + ( ( this.r * 255 ) | 0 ) + ',' + ( ( this.g * 255 ) | 0 ) + ',' + ( ( this.b * 255 ) | 0 ) + ')';

    	},

    	offsetHSL: function ( h, s, l ) {

    		this.getHSL( _hslA );

    		_hslA.h += h; _hslA.s += s; _hslA.l += l;

    		this.setHSL( _hslA.h, _hslA.s, _hslA.l );

    		return this;

    	},

    	add: function ( color ) {

    		this.r += color.r;
    		this.g += color.g;
    		this.b += color.b;

    		return this;

    	},

    	addColors: function ( color1, color2 ) {

    		this.r = color1.r + color2.r;
    		this.g = color1.g + color2.g;
    		this.b = color1.b + color2.b;

    		return this;

    	},

    	addScalar: function ( s ) {

    		this.r += s;
    		this.g += s;
    		this.b += s;

    		return this;

    	},

    	sub: function ( color ) {

    		this.r = Math.max( 0, this.r - color.r );
    		this.g = Math.max( 0, this.g - color.g );
    		this.b = Math.max( 0, this.b - color.b );

    		return this;

    	},

    	multiply: function ( color ) {

    		this.r *= color.r;
    		this.g *= color.g;
    		this.b *= color.b;

    		return this;

    	},

    	multiplyScalar: function ( s ) {

    		this.r *= s;
    		this.g *= s;
    		this.b *= s;

    		return this;

    	},

    	lerp: function ( color, alpha ) {

    		this.r += ( color.r - this.r ) * alpha;
    		this.g += ( color.g - this.g ) * alpha;
    		this.b += ( color.b - this.b ) * alpha;

    		return this;

    	},

    	lerpHSL: function ( color, alpha ) {

    		this.getHSL( _hslA );
    		color.getHSL( _hslB );

    		var h = _Math.lerp( _hslA.h, _hslB.h, alpha );
    		var s = _Math.lerp( _hslA.s, _hslB.s, alpha );
    		var l = _Math.lerp( _hslA.l, _hslB.l, alpha );

    		this.setHSL( h, s, l );

    		return this;

    	},

    	equals: function ( c ) {

    		return ( c.r === this.r ) && ( c.g === this.g ) && ( c.b === this.b );

    	},

    	fromArray: function ( array, offset ) {

    		if ( offset === undefined ) offset = 0;

    		this.r = array[ offset ];
    		this.g = array[ offset + 1 ];
    		this.b = array[ offset + 2 ];

    		return this;

    	},

    	toArray: function ( array, offset ) {

    		if ( array === undefined ) array = [];
    		if ( offset === undefined ) offset = 0;

    		array[ offset ] = this.r;
    		array[ offset + 1 ] = this.g;
    		array[ offset + 2 ] = this.b;

    		return array;

    	},

    	toJSON: function () {

    		return this.getHex();

    	}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     * @author alteredq / http://alteredqualia.com/
     */

    function Face3( a, b, c, normal, color, materialIndex ) {

    	this.a = a;
    	this.b = b;
    	this.c = c;

    	this.normal = ( normal && normal.isVector3 ) ? normal : new Vector3();
    	this.vertexNormals = Array.isArray( normal ) ? normal : [];

    	this.color = ( color && color.isColor ) ? color : new Color$7();
    	this.vertexColors = Array.isArray( color ) ? color : [];

    	this.materialIndex = materialIndex !== undefined ? materialIndex : 0;

    }

    Object.assign( Face3.prototype, {

    	clone: function () {

    		return new this.constructor().copy( this );

    	},

    	copy: function ( source ) {

    		this.a = source.a;
    		this.b = source.b;
    		this.c = source.c;

    		this.normal.copy( source.normal );
    		this.color.copy( source.color );

    		this.materialIndex = source.materialIndex;

    		for ( var i = 0, il = source.vertexNormals.length; i < il; i ++ ) {

    			this.vertexNormals[ i ] = source.vertexNormals[ i ].clone();

    		}

    		for ( var i = 0, il = source.vertexColors.length; i < il; i ++ ) {

    			this.vertexColors[ i ] = source.vertexColors[ i ].clone();

    		}

    		return this;

    	}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     * @author alteredq / http://alteredqualia.com/
     */

    var materialId = 0;

    function Material() {

    	Object.defineProperty( this, 'id', { value: materialId ++ } );

    	this.uuid = _Math.generateUUID();

    	this.name = '';
    	this.type = 'Material';

    	this.fog = true;
    	this.lights = true;

    	this.blending = NormalBlending;
    	this.side = FrontSide;
    	this.flatShading = false;
    	this.vertexTangents = false;
    	this.vertexColors = NoColors; // THREE.NoColors, THREE.VertexColors, THREE.FaceColors

    	this.opacity = 1;
    	this.transparent = false;

    	this.blendSrc = SrcAlphaFactor;
    	this.blendDst = OneMinusSrcAlphaFactor;
    	this.blendEquation = AddEquation;
    	this.blendSrcAlpha = null;
    	this.blendDstAlpha = null;
    	this.blendEquationAlpha = null;

    	this.depthFunc = LessEqualDepth;
    	this.depthTest = true;
    	this.depthWrite = true;

    	this.stencilFunc = AlwaysStencilFunc;
    	this.stencilRef = 0;
    	this.stencilMask = 0xff;
    	this.stencilFail = KeepStencilOp;
    	this.stencilZFail = KeepStencilOp;
    	this.stencilZPass = KeepStencilOp;
    	this.stencilWrite = false;

    	this.clippingPlanes = null;
    	this.clipIntersection = false;
    	this.clipShadows = false;

    	this.shadowSide = null;

    	this.colorWrite = true;

    	this.precision = null; // override the renderer's default precision for this material

    	this.polygonOffset = false;
    	this.polygonOffsetFactor = 0;
    	this.polygonOffsetUnits = 0;

    	this.dithering = false;

    	this.alphaTest = 0;
    	this.premultipliedAlpha = false;

    	this.visible = true;

    	this.toneMapped = true;

    	this.userData = {};

    	this.needsUpdate = true;

    }

    Material.prototype = Object.assign( Object.create( EventDispatcher.prototype ), {

    	constructor: Material,

    	isMaterial: true,

    	onBeforeCompile: function () {},

    	setValues: function ( values ) {

    		if ( values === undefined ) return;

    		for ( var key in values ) {

    			var newValue = values[ key ];

    			if ( newValue === undefined ) {

    				console.warn( "THREE.Material: '" + key + "' parameter is undefined." );
    				continue;

    			}

    			// for backward compatability if shading is set in the constructor
    			if ( key === 'shading' ) {

    				console.warn( 'THREE.' + this.type + ': .shading has been removed. Use the boolean .flatShading instead.' );
    				this.flatShading = ( newValue === FlatShading ) ? true : false;
    				continue;

    			}

    			var currentValue = this[ key ];

    			if ( currentValue === undefined ) {

    				console.warn( "THREE." + this.type + ": '" + key + "' is not a property of this material." );
    				continue;

    			}

    			if ( currentValue && currentValue.isColor ) {

    				currentValue.set( newValue );

    			} else if ( ( currentValue && currentValue.isVector3 ) && ( newValue && newValue.isVector3 ) ) {

    				currentValue.copy( newValue );

    			} else {

    				this[ key ] = newValue;

    			}

    		}

    	},

    	toJSON: function ( meta ) {

    		var isRoot = ( meta === undefined || typeof meta === 'string' );

    		if ( isRoot ) {

    			meta = {
    				textures: {},
    				images: {}
    			};

    		}

    		var data = {
    			metadata: {
    				version: 4.5,
    				type: 'Material',
    				generator: 'Material.toJSON'
    			}
    		};

    		// standard Material serialization
    		data.uuid = this.uuid;
    		data.type = this.type;

    		if ( this.name !== '' ) data.name = this.name;

    		if ( this.color && this.color.isColor ) data.color = this.color.getHex();

    		if ( this.roughness !== undefined ) data.roughness = this.roughness;
    		if ( this.metalness !== undefined ) data.metalness = this.metalness;

    		if ( this.emissive && this.emissive.isColor ) data.emissive = this.emissive.getHex();
    		if ( this.emissiveIntensity && this.emissiveIntensity !== 1 ) data.emissiveIntensity = this.emissiveIntensity;

    		if ( this.specular && this.specular.isColor ) data.specular = this.specular.getHex();
    		if ( this.shininess !== undefined ) data.shininess = this.shininess;
    		if ( this.clearcoat !== undefined ) data.clearcoat = this.clearcoat;
    		if ( this.clearcoatRoughness !== undefined ) data.clearcoatRoughness = this.clearcoatRoughness;

    		if ( this.clearcoatNormalMap && this.clearcoatNormalMap.isTexture ) {

    			data.clearcoatNormalMap = this.clearcoatNormalMap.toJSON( meta ).uuid;
    			data.clearcoatNormalScale = this.clearcoatNormalScale.toArray();

    		}

    		if ( this.map && this.map.isTexture ) data.map = this.map.toJSON( meta ).uuid;
    		if ( this.matcap && this.matcap.isTexture ) data.matcap = this.matcap.toJSON( meta ).uuid;
    		if ( this.alphaMap && this.alphaMap.isTexture ) data.alphaMap = this.alphaMap.toJSON( meta ).uuid;
    		if ( this.lightMap && this.lightMap.isTexture ) data.lightMap = this.lightMap.toJSON( meta ).uuid;

    		if ( this.aoMap && this.aoMap.isTexture ) {

    			data.aoMap = this.aoMap.toJSON( meta ).uuid;
    			data.aoMapIntensity = this.aoMapIntensity;

    		}

    		if ( this.bumpMap && this.bumpMap.isTexture ) {

    			data.bumpMap = this.bumpMap.toJSON( meta ).uuid;
    			data.bumpScale = this.bumpScale;

    		}

    		if ( this.normalMap && this.normalMap.isTexture ) {

    			data.normalMap = this.normalMap.toJSON( meta ).uuid;
    			data.normalMapType = this.normalMapType;
    			data.normalScale = this.normalScale.toArray();

    		}

    		if ( this.displacementMap && this.displacementMap.isTexture ) {

    			data.displacementMap = this.displacementMap.toJSON( meta ).uuid;
    			data.displacementScale = this.displacementScale;
    			data.displacementBias = this.displacementBias;

    		}

    		if ( this.roughnessMap && this.roughnessMap.isTexture ) data.roughnessMap = this.roughnessMap.toJSON( meta ).uuid;
    		if ( this.metalnessMap && this.metalnessMap.isTexture ) data.metalnessMap = this.metalnessMap.toJSON( meta ).uuid;

    		if ( this.emissiveMap && this.emissiveMap.isTexture ) data.emissiveMap = this.emissiveMap.toJSON( meta ).uuid;
    		if ( this.specularMap && this.specularMap.isTexture ) data.specularMap = this.specularMap.toJSON( meta ).uuid;

    		if ( this.envMap && this.envMap.isTexture ) {

    			data.envMap = this.envMap.toJSON( meta ).uuid;
    			data.reflectivity = this.reflectivity; // Scale behind envMap
    			data.refractionRatio = this.refractionRatio;

    			if ( this.combine !== undefined ) data.combine = this.combine;
    			if ( this.envMapIntensity !== undefined ) data.envMapIntensity = this.envMapIntensity;

    		}

    		if ( this.gradientMap && this.gradientMap.isTexture ) {

    			data.gradientMap = this.gradientMap.toJSON( meta ).uuid;

    		}

    		if ( this.size !== undefined ) data.size = this.size;
    		if ( this.sizeAttenuation !== undefined ) data.sizeAttenuation = this.sizeAttenuation;

    		if ( this.blending !== NormalBlending ) data.blending = this.blending;
    		if ( this.flatShading === true ) data.flatShading = this.flatShading;
    		if ( this.side !== FrontSide ) data.side = this.side;
    		if ( this.vertexColors !== NoColors ) data.vertexColors = this.vertexColors;

    		if ( this.opacity < 1 ) data.opacity = this.opacity;
    		if ( this.transparent === true ) data.transparent = this.transparent;

    		data.depthFunc = this.depthFunc;
    		data.depthTest = this.depthTest;
    		data.depthWrite = this.depthWrite;

    		data.stencilWrite = this.stencilWrite;
    		data.stencilFunc = this.stencilFunc;
    		data.stencilRef = this.stencilRef;
    		data.stencilMask = this.stencilMask;
    		data.stencilFail = this.stencilFail;
    		data.stencilZFail = this.stencilZFail;
    		data.stencilZPass = this.stencilZPass;

    		// rotation (SpriteMaterial)
    		if ( this.rotation && this.rotation !== 0 ) data.rotation = this.rotation;

    		if ( this.polygonOffset === true ) data.polygonOffset = true;
    		if ( this.polygonOffsetFactor !== 0 ) data.polygonOffsetFactor = this.polygonOffsetFactor;
    		if ( this.polygonOffsetUnits !== 0 ) data.polygonOffsetUnits = this.polygonOffsetUnits;

    		if ( this.linewidth && this.linewidth !== 1 ) data.linewidth = this.linewidth;
    		if ( this.dashSize !== undefined ) data.dashSize = this.dashSize;
    		if ( this.gapSize !== undefined ) data.gapSize = this.gapSize;
    		if ( this.scale !== undefined ) data.scale = this.scale;

    		if ( this.dithering === true ) data.dithering = true;

    		if ( this.alphaTest > 0 ) data.alphaTest = this.alphaTest;
    		if ( this.premultipliedAlpha === true ) data.premultipliedAlpha = this.premultipliedAlpha;

    		if ( this.wireframe === true ) data.wireframe = this.wireframe;
    		if ( this.wireframeLinewidth > 1 ) data.wireframeLinewidth = this.wireframeLinewidth;
    		if ( this.wireframeLinecap !== 'round' ) data.wireframeLinecap = this.wireframeLinecap;
    		if ( this.wireframeLinejoin !== 'round' ) data.wireframeLinejoin = this.wireframeLinejoin;

    		if ( this.morphTargets === true ) data.morphTargets = true;
    		if ( this.morphNormals === true ) data.morphNormals = true;
    		if ( this.skinning === true ) data.skinning = true;

    		if ( this.visible === false ) data.visible = false;

    		if ( this.toneMapped === false ) data.toneMapped = false;

    		if ( JSON.stringify( this.userData ) !== '{}' ) data.userData = this.userData;

    		// TODO: Copied from Object3D.toJSON

    		function extractFromCache( cache ) {

    			var values = [];

    			for ( var key in cache ) {

    				var data = cache[ key ];
    				delete data.metadata;
    				values.push( data );

    			}

    			return values;

    		}

    		if ( isRoot ) {

    			var textures = extractFromCache( meta.textures );
    			var images = extractFromCache( meta.images );

    			if ( textures.length > 0 ) data.textures = textures;
    			if ( images.length > 0 ) data.images = images;

    		}

    		return data;

    	},

    	clone: function () {

    		return new this.constructor().copy( this );

    	},

    	copy: function ( source ) {

    		this.name = source.name;

    		this.fog = source.fog;
    		this.lights = source.lights;

    		this.blending = source.blending;
    		this.side = source.side;
    		this.flatShading = source.flatShading;
    		this.vertexColors = source.vertexColors;

    		this.opacity = source.opacity;
    		this.transparent = source.transparent;

    		this.blendSrc = source.blendSrc;
    		this.blendDst = source.blendDst;
    		this.blendEquation = source.blendEquation;
    		this.blendSrcAlpha = source.blendSrcAlpha;
    		this.blendDstAlpha = source.blendDstAlpha;
    		this.blendEquationAlpha = source.blendEquationAlpha;

    		this.depthFunc = source.depthFunc;
    		this.depthTest = source.depthTest;
    		this.depthWrite = source.depthWrite;

    		this.stencilWrite = source.stencilWrite;
    		this.stencilFunc = source.stencilFunc;
    		this.stencilRef = source.stencilRef;
    		this.stencilMask = source.stencilMask;
    		this.stencilFail = source.stencilFail;
    		this.stencilZFail = source.stencilZFail;
    		this.stencilZPass = source.stencilZPass;

    		this.colorWrite = source.colorWrite;

    		this.precision = source.precision;

    		this.polygonOffset = source.polygonOffset;
    		this.polygonOffsetFactor = source.polygonOffsetFactor;
    		this.polygonOffsetUnits = source.polygonOffsetUnits;

    		this.dithering = source.dithering;

    		this.alphaTest = source.alphaTest;
    		this.premultipliedAlpha = source.premultipliedAlpha;

    		this.visible = source.visible;

    		this.toneMapped = source.toneMapped;

    		this.userData = JSON.parse( JSON.stringify( source.userData ) );

    		this.clipShadows = source.clipShadows;
    		this.clipIntersection = source.clipIntersection;

    		var srcPlanes = source.clippingPlanes,
    			dstPlanes = null;

    		if ( srcPlanes !== null ) {

    			var n = srcPlanes.length;
    			dstPlanes = new Array( n );

    			for ( var i = 0; i !== n; ++ i )
    				dstPlanes[ i ] = srcPlanes[ i ].clone();

    		}

    		this.clippingPlanes = dstPlanes;

    		this.shadowSide = source.shadowSide;

    		return this;

    	},

    	dispose: function () {

    		this.dispatchEvent( { type: 'dispose' } );

    	}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     * @author alteredq / http://alteredqualia.com/
     *
     * parameters = {
     *  color: <hex>,
     *  opacity: <float>,
     *  map: new THREE.Texture( <Image> ),
     *
     *  lightMap: new THREE.Texture( <Image> ),
     *  lightMapIntensity: <float>
     *
     *  aoMap: new THREE.Texture( <Image> ),
     *  aoMapIntensity: <float>
     *
     *  specularMap: new THREE.Texture( <Image> ),
     *
     *  alphaMap: new THREE.Texture( <Image> ),
     *
     *  envMap: new THREE.CubeTexture( [posx, negx, posy, negy, posz, negz] ),
     *  combine: THREE.Multiply,
     *  reflectivity: <float>,
     *  refractionRatio: <float>,
     *
     *  depthTest: <bool>,
     *  depthWrite: <bool>,
     *
     *  wireframe: <boolean>,
     *  wireframeLinewidth: <float>,
     *
     *  skinning: <bool>,
     *  morphTargets: <bool>
     * }
     */

    function MeshBasicMaterial( parameters ) {

    	Material.call( this );

    	this.type = 'MeshBasicMaterial';

    	this.color = new Color$7( 0xffffff ); // emissive

    	this.map = null;

    	this.lightMap = null;
    	this.lightMapIntensity = 1.0;

    	this.aoMap = null;
    	this.aoMapIntensity = 1.0;

    	this.specularMap = null;

    	this.alphaMap = null;

    	this.envMap = null;
    	this.combine = MultiplyOperation;
    	this.reflectivity = 1;
    	this.refractionRatio = 0.98;

    	this.wireframe = false;
    	this.wireframeLinewidth = 1;
    	this.wireframeLinecap = 'round';
    	this.wireframeLinejoin = 'round';

    	this.skinning = false;
    	this.morphTargets = false;

    	this.lights = false;

    	this.setValues( parameters );

    }

    MeshBasicMaterial.prototype = Object.create( Material.prototype );
    MeshBasicMaterial.prototype.constructor = MeshBasicMaterial;

    MeshBasicMaterial.prototype.isMeshBasicMaterial = true;

    MeshBasicMaterial.prototype.copy = function ( source ) {

    	Material.prototype.copy.call( this, source );

    	this.color.copy( source.color );

    	this.map = source.map;

    	this.lightMap = source.lightMap;
    	this.lightMapIntensity = source.lightMapIntensity;

    	this.aoMap = source.aoMap;
    	this.aoMapIntensity = source.aoMapIntensity;

    	this.specularMap = source.specularMap;

    	this.alphaMap = source.alphaMap;

    	this.envMap = source.envMap;
    	this.combine = source.combine;
    	this.reflectivity = source.reflectivity;
    	this.refractionRatio = source.refractionRatio;

    	this.wireframe = source.wireframe;
    	this.wireframeLinewidth = source.wireframeLinewidth;
    	this.wireframeLinecap = source.wireframeLinecap;
    	this.wireframeLinejoin = source.wireframeLinejoin;

    	this.skinning = source.skinning;
    	this.morphTargets = source.morphTargets;

    	return this;

    };

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function BufferAttribute( array, itemSize, normalized ) {

    	if ( Array.isArray( array ) ) {

    		throw new TypeError( 'THREE.BufferAttribute: array should be a Typed Array.' );

    	}

    	this.name = '';

    	this.array = array;
    	this.itemSize = itemSize;
    	this.count = array !== undefined ? array.length / itemSize : 0;
    	this.normalized = normalized === true;

    	this.dynamic = false;
    	this.updateRange = { offset: 0, count: - 1 };

    	this.version = 0;

    }

    Object.defineProperty( BufferAttribute.prototype, 'needsUpdate', {

    	set: function ( value ) {

    		if ( value === true ) this.version ++;

    	}

    } );

    Object.assign( BufferAttribute.prototype, {

    	isBufferAttribute: true,

    	onUploadCallback: function () {},

    	setArray: function ( array ) {

    		if ( Array.isArray( array ) ) {

    			throw new TypeError( 'THREE.BufferAttribute: array should be a Typed Array.' );

    		}

    		this.count = array !== undefined ? array.length / this.itemSize : 0;
    		this.array = array;

    		return this;

    	},

    	setDynamic: function ( value ) {

    		this.dynamic = value;

    		return this;

    	},

    	copy: function ( source ) {

    		this.name = source.name;
    		this.array = new source.array.constructor( source.array );
    		this.itemSize = source.itemSize;
    		this.count = source.count;
    		this.normalized = source.normalized;

    		this.dynamic = source.dynamic;

    		return this;

    	},

    	copyAt: function ( index1, attribute, index2 ) {

    		index1 *= this.itemSize;
    		index2 *= attribute.itemSize;

    		for ( var i = 0, l = this.itemSize; i < l; i ++ ) {

    			this.array[ index1 + i ] = attribute.array[ index2 + i ];

    		}

    		return this;

    	},

    	copyArray: function ( array ) {

    		this.array.set( array );

    		return this;

    	},

    	copyColorsArray: function ( colors ) {

    		var array = this.array, offset = 0;

    		for ( var i = 0, l = colors.length; i < l; i ++ ) {

    			var color = colors[ i ];

    			if ( color === undefined ) {

    				console.warn( 'THREE.BufferAttribute.copyColorsArray(): color is undefined', i );
    				color = new Color$7();

    			}

    			array[ offset ++ ] = color.r;
    			array[ offset ++ ] = color.g;
    			array[ offset ++ ] = color.b;

    		}

    		return this;

    	},

    	copyVector2sArray: function ( vectors ) {

    		var array = this.array, offset = 0;

    		for ( var i = 0, l = vectors.length; i < l; i ++ ) {

    			var vector = vectors[ i ];

    			if ( vector === undefined ) {

    				console.warn( 'THREE.BufferAttribute.copyVector2sArray(): vector is undefined', i );
    				vector = new Vector2();

    			}

    			array[ offset ++ ] = vector.x;
    			array[ offset ++ ] = vector.y;

    		}

    		return this;

    	},

    	copyVector3sArray: function ( vectors ) {

    		var array = this.array, offset = 0;

    		for ( var i = 0, l = vectors.length; i < l; i ++ ) {

    			var vector = vectors[ i ];

    			if ( vector === undefined ) {

    				console.warn( 'THREE.BufferAttribute.copyVector3sArray(): vector is undefined', i );
    				vector = new Vector3();

    			}

    			array[ offset ++ ] = vector.x;
    			array[ offset ++ ] = vector.y;
    			array[ offset ++ ] = vector.z;

    		}

    		return this;

    	},

    	copyVector4sArray: function ( vectors ) {

    		var array = this.array, offset = 0;

    		for ( var i = 0, l = vectors.length; i < l; i ++ ) {

    			var vector = vectors[ i ];

    			if ( vector === undefined ) {

    				console.warn( 'THREE.BufferAttribute.copyVector4sArray(): vector is undefined', i );
    				vector = new Vector4();

    			}

    			array[ offset ++ ] = vector.x;
    			array[ offset ++ ] = vector.y;
    			array[ offset ++ ] = vector.z;
    			array[ offset ++ ] = vector.w;

    		}

    		return this;

    	},

    	set: function ( value, offset ) {

    		if ( offset === undefined ) offset = 0;

    		this.array.set( value, offset );

    		return this;

    	},

    	getX: function ( index ) {

    		return this.array[ index * this.itemSize ];

    	},

    	setX: function ( index, x ) {

    		this.array[ index * this.itemSize ] = x;

    		return this;

    	},

    	getY: function ( index ) {

    		return this.array[ index * this.itemSize + 1 ];

    	},

    	setY: function ( index, y ) {

    		this.array[ index * this.itemSize + 1 ] = y;

    		return this;

    	},

    	getZ: function ( index ) {

    		return this.array[ index * this.itemSize + 2 ];

    	},

    	setZ: function ( index, z ) {

    		this.array[ index * this.itemSize + 2 ] = z;

    		return this;

    	},

    	getW: function ( index ) {

    		return this.array[ index * this.itemSize + 3 ];

    	},

    	setW: function ( index, w ) {

    		this.array[ index * this.itemSize + 3 ] = w;

    		return this;

    	},

    	setXY: function ( index, x, y ) {

    		index *= this.itemSize;

    		this.array[ index + 0 ] = x;
    		this.array[ index + 1 ] = y;

    		return this;

    	},

    	setXYZ: function ( index, x, y, z ) {

    		index *= this.itemSize;

    		this.array[ index + 0 ] = x;
    		this.array[ index + 1 ] = y;
    		this.array[ index + 2 ] = z;

    		return this;

    	},

    	setXYZW: function ( index, x, y, z, w ) {

    		index *= this.itemSize;

    		this.array[ index + 0 ] = x;
    		this.array[ index + 1 ] = y;
    		this.array[ index + 2 ] = z;
    		this.array[ index + 3 ] = w;

    		return this;

    	},

    	onUpload: function ( callback ) {

    		this.onUploadCallback = callback;

    		return this;

    	},

    	clone: function () {

    		return new this.constructor( this.array, this.itemSize ).copy( this );

    	},

    	toJSON: function () {

    		return {
    			itemSize: this.itemSize,
    			type: this.array.constructor.name,
    			array: Array.prototype.slice.call( this.array ),
    			normalized: this.normalized
    		};

    	}

    } );

    //

    function Int8BufferAttribute( array, itemSize, normalized ) {

    	BufferAttribute.call( this, new Int8Array( array ), itemSize, normalized );

    }

    Int8BufferAttribute.prototype = Object.create( BufferAttribute.prototype );
    Int8BufferAttribute.prototype.constructor = Int8BufferAttribute;


    function Uint8BufferAttribute( array, itemSize, normalized ) {

    	BufferAttribute.call( this, new Uint8Array( array ), itemSize, normalized );

    }

    Uint8BufferAttribute.prototype = Object.create( BufferAttribute.prototype );
    Uint8BufferAttribute.prototype.constructor = Uint8BufferAttribute;


    function Uint8ClampedBufferAttribute( array, itemSize, normalized ) {

    	BufferAttribute.call( this, new Uint8ClampedArray( array ), itemSize, normalized );

    }

    Uint8ClampedBufferAttribute.prototype = Object.create( BufferAttribute.prototype );
    Uint8ClampedBufferAttribute.prototype.constructor = Uint8ClampedBufferAttribute;


    function Int16BufferAttribute( array, itemSize, normalized ) {

    	BufferAttribute.call( this, new Int16Array( array ), itemSize, normalized );

    }

    Int16BufferAttribute.prototype = Object.create( BufferAttribute.prototype );
    Int16BufferAttribute.prototype.constructor = Int16BufferAttribute;


    function Uint16BufferAttribute( array, itemSize, normalized ) {

    	BufferAttribute.call( this, new Uint16Array( array ), itemSize, normalized );

    }

    Uint16BufferAttribute.prototype = Object.create( BufferAttribute.prototype );
    Uint16BufferAttribute.prototype.constructor = Uint16BufferAttribute;


    function Int32BufferAttribute( array, itemSize, normalized ) {

    	BufferAttribute.call( this, new Int32Array( array ), itemSize, normalized );

    }

    Int32BufferAttribute.prototype = Object.create( BufferAttribute.prototype );
    Int32BufferAttribute.prototype.constructor = Int32BufferAttribute;


    function Uint32BufferAttribute( array, itemSize, normalized ) {

    	BufferAttribute.call( this, new Uint32Array( array ), itemSize, normalized );

    }

    Uint32BufferAttribute.prototype = Object.create( BufferAttribute.prototype );
    Uint32BufferAttribute.prototype.constructor = Uint32BufferAttribute;


    function Float32BufferAttribute( array, itemSize, normalized ) {

    	BufferAttribute.call( this, new Float32Array( array ), itemSize, normalized );

    }

    Float32BufferAttribute.prototype = Object.create( BufferAttribute.prototype );
    Float32BufferAttribute.prototype.constructor = Float32BufferAttribute;


    function Float64BufferAttribute( array, itemSize, normalized ) {

    	BufferAttribute.call( this, new Float64Array( array ), itemSize, normalized );

    }

    Float64BufferAttribute.prototype = Object.create( BufferAttribute.prototype );
    Float64BufferAttribute.prototype.constructor = Float64BufferAttribute;

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function DirectGeometry() {

    	this.vertices = [];
    	this.normals = [];
    	this.colors = [];
    	this.uvs = [];
    	this.uvs2 = [];

    	this.groups = [];

    	this.morphTargets = {};

    	this.skinWeights = [];
    	this.skinIndices = [];

    	// this.lineDistances = [];

    	this.boundingBox = null;
    	this.boundingSphere = null;

    	// update flags

    	this.verticesNeedUpdate = false;
    	this.normalsNeedUpdate = false;
    	this.colorsNeedUpdate = false;
    	this.uvsNeedUpdate = false;
    	this.groupsNeedUpdate = false;

    }

    Object.assign( DirectGeometry.prototype, {

    	computeGroups: function ( geometry ) {

    		var group;
    		var groups = [];
    		var materialIndex = undefined;

    		var faces = geometry.faces;

    		for ( var i = 0; i < faces.length; i ++ ) {

    			var face = faces[ i ];

    			// materials

    			if ( face.materialIndex !== materialIndex ) {

    				materialIndex = face.materialIndex;

    				if ( group !== undefined ) {

    					group.count = ( i * 3 ) - group.start;
    					groups.push( group );

    				}

    				group = {
    					start: i * 3,
    					materialIndex: materialIndex
    				};

    			}

    		}

    		if ( group !== undefined ) {

    			group.count = ( i * 3 ) - group.start;
    			groups.push( group );

    		}

    		this.groups = groups;

    	},

    	fromGeometry: function ( geometry ) {

    		var faces = geometry.faces;
    		var vertices = geometry.vertices;
    		var faceVertexUvs = geometry.faceVertexUvs;

    		var hasFaceVertexUv = faceVertexUvs[ 0 ] && faceVertexUvs[ 0 ].length > 0;
    		var hasFaceVertexUv2 = faceVertexUvs[ 1 ] && faceVertexUvs[ 1 ].length > 0;

    		// morphs

    		var morphTargets = geometry.morphTargets;
    		var morphTargetsLength = morphTargets.length;

    		var morphTargetsPosition;

    		if ( morphTargetsLength > 0 ) {

    			morphTargetsPosition = [];

    			for ( var i = 0; i < morphTargetsLength; i ++ ) {

    				morphTargetsPosition[ i ] = {
    					name: morphTargets[ i ].name,
    				 	data: []
    				};

    			}

    			this.morphTargets.position = morphTargetsPosition;

    		}

    		var morphNormals = geometry.morphNormals;
    		var morphNormalsLength = morphNormals.length;

    		var morphTargetsNormal;

    		if ( morphNormalsLength > 0 ) {

    			morphTargetsNormal = [];

    			for ( var i = 0; i < morphNormalsLength; i ++ ) {

    				morphTargetsNormal[ i ] = {
    					name: morphNormals[ i ].name,
    				 	data: []
    				};

    			}

    			this.morphTargets.normal = morphTargetsNormal;

    		}

    		// skins

    		var skinIndices = geometry.skinIndices;
    		var skinWeights = geometry.skinWeights;

    		var hasSkinIndices = skinIndices.length === vertices.length;
    		var hasSkinWeights = skinWeights.length === vertices.length;

    		//

    		if ( vertices.length > 0 && faces.length === 0 ) {

    			console.error( 'THREE.DirectGeometry: Faceless geometries are not supported.' );

    		}

    		for ( var i = 0; i < faces.length; i ++ ) {

    			var face = faces[ i ];

    			this.vertices.push( vertices[ face.a ], vertices[ face.b ], vertices[ face.c ] );

    			var vertexNormals = face.vertexNormals;

    			if ( vertexNormals.length === 3 ) {

    				this.normals.push( vertexNormals[ 0 ], vertexNormals[ 1 ], vertexNormals[ 2 ] );

    			} else {

    				var normal = face.normal;

    				this.normals.push( normal, normal, normal );

    			}

    			var vertexColors = face.vertexColors;

    			if ( vertexColors.length === 3 ) {

    				this.colors.push( vertexColors[ 0 ], vertexColors[ 1 ], vertexColors[ 2 ] );

    			} else {

    				var color = face.color;

    				this.colors.push( color, color, color );

    			}

    			if ( hasFaceVertexUv === true ) {

    				var vertexUvs = faceVertexUvs[ 0 ][ i ];

    				if ( vertexUvs !== undefined ) {

    					this.uvs.push( vertexUvs[ 0 ], vertexUvs[ 1 ], vertexUvs[ 2 ] );

    				} else {

    					console.warn( 'THREE.DirectGeometry.fromGeometry(): Undefined vertexUv ', i );

    					this.uvs.push( new Vector2(), new Vector2(), new Vector2() );

    				}

    			}

    			if ( hasFaceVertexUv2 === true ) {

    				var vertexUvs = faceVertexUvs[ 1 ][ i ];

    				if ( vertexUvs !== undefined ) {

    					this.uvs2.push( vertexUvs[ 0 ], vertexUvs[ 1 ], vertexUvs[ 2 ] );

    				} else {

    					console.warn( 'THREE.DirectGeometry.fromGeometry(): Undefined vertexUv2 ', i );

    					this.uvs2.push( new Vector2(), new Vector2(), new Vector2() );

    				}

    			}

    			// morphs

    			for ( var j = 0; j < morphTargetsLength; j ++ ) {

    				var morphTarget = morphTargets[ j ].vertices;

    				morphTargetsPosition[ j ].data.push( morphTarget[ face.a ], morphTarget[ face.b ], morphTarget[ face.c ] );

    			}

    			for ( var j = 0; j < morphNormalsLength; j ++ ) {

    				var morphNormal = morphNormals[ j ].vertexNormals[ i ];

    				morphTargetsNormal[ j ].data.push( morphNormal.a, morphNormal.b, morphNormal.c );

    			}

    			// skins

    			if ( hasSkinIndices ) {

    				this.skinIndices.push( skinIndices[ face.a ], skinIndices[ face.b ], skinIndices[ face.c ] );

    			}

    			if ( hasSkinWeights ) {

    				this.skinWeights.push( skinWeights[ face.a ], skinWeights[ face.b ], skinWeights[ face.c ] );

    			}

    		}

    		this.computeGroups( geometry );

    		this.verticesNeedUpdate = geometry.verticesNeedUpdate;
    		this.normalsNeedUpdate = geometry.normalsNeedUpdate;
    		this.colorsNeedUpdate = geometry.colorsNeedUpdate;
    		this.uvsNeedUpdate = geometry.uvsNeedUpdate;
    		this.groupsNeedUpdate = geometry.groupsNeedUpdate;

    		if ( geometry.boundingSphere !== null ) {

    			this.boundingSphere = geometry.boundingSphere.clone();

    		}

    		if ( geometry.boundingBox !== null ) {

    			this.boundingBox = geometry.boundingBox.clone();

    		}

    		return this;

    	}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function arrayMax( array ) {

    	if ( array.length === 0 ) return - Infinity;

    	var max = array[ 0 ];

    	for ( var i = 1, l = array.length; i < l; ++ i ) {

    		if ( array[ i ] > max ) max = array[ i ];

    	}

    	return max;

    }

    /**
     * @author alteredq / http://alteredqualia.com/
     * @author mrdoob / http://mrdoob.com/
     */

    var _bufferGeometryId = 1; // BufferGeometry uses odd numbers as Id

    var _m1$2 = new Matrix4();
    var _obj = new Object3D();
    var _offset = new Vector3();
    var _box$1 = new Box3();
    var _boxMorphTargets = new Box3();
    var _vector$4 = new Vector3();

    function BufferGeometry() {

    	Object.defineProperty( this, 'id', { value: _bufferGeometryId += 2 } );

    	this.uuid = _Math.generateUUID();

    	this.name = '';
    	this.type = 'BufferGeometry';

    	this.index = null;
    	this.attributes = {};

    	this.morphAttributes = {};

    	this.groups = [];

    	this.boundingBox = null;
    	this.boundingSphere = null;

    	this.drawRange = { start: 0, count: Infinity };

    	this.userData = {};

    }

    BufferGeometry.prototype = Object.assign( Object.create( EventDispatcher.prototype ), {

    	constructor: BufferGeometry,

    	isBufferGeometry: true,

    	getIndex: function () {

    		return this.index;

    	},

    	setIndex: function ( index ) {

    		if ( Array.isArray( index ) ) {

    			this.index = new ( arrayMax( index ) > 65535 ? Uint32BufferAttribute : Uint16BufferAttribute )( index, 1 );

    		} else {

    			this.index = index;

    		}

    	},

    	addAttribute: function ( name, attribute ) {

    		if ( ! ( attribute && attribute.isBufferAttribute ) && ! ( attribute && attribute.isInterleavedBufferAttribute ) ) {

    			console.warn( 'THREE.BufferGeometry: .addAttribute() now expects ( name, attribute ).' );

    			return this.addAttribute( name, new BufferAttribute( arguments[ 1 ], arguments[ 2 ] ) );

    		}

    		if ( name === 'index' ) {

    			console.warn( 'THREE.BufferGeometry.addAttribute: Use .setIndex() for index attribute.' );
    			this.setIndex( attribute );

    			return this;

    		}

    		this.attributes[ name ] = attribute;

    		return this;

    	},

    	getAttribute: function ( name ) {

    		return this.attributes[ name ];

    	},

    	removeAttribute: function ( name ) {

    		delete this.attributes[ name ];

    		return this;

    	},

    	addGroup: function ( start, count, materialIndex ) {

    		this.groups.push( {

    			start: start,
    			count: count,
    			materialIndex: materialIndex !== undefined ? materialIndex : 0

    		} );

    	},

    	clearGroups: function () {

    		this.groups = [];

    	},

    	setDrawRange: function ( start, count ) {

    		this.drawRange.start = start;
    		this.drawRange.count = count;

    	},

    	applyMatrix: function ( matrix ) {

    		var position = this.attributes.position;

    		if ( position !== undefined ) {

    			matrix.applyToBufferAttribute( position );
    			position.needsUpdate = true;

    		}

    		var normal = this.attributes.normal;

    		if ( normal !== undefined ) {

    			var normalMatrix = new Matrix3().getNormalMatrix( matrix );

    			normalMatrix.applyToBufferAttribute( normal );
    			normal.needsUpdate = true;

    		}

    		var tangent = this.attributes.tangent;

    		if ( tangent !== undefined ) {

    			var normalMatrix = new Matrix3().getNormalMatrix( matrix );

    			// Tangent is vec4, but the '.w' component is a sign value (+1/-1).
    			normalMatrix.applyToBufferAttribute( tangent );
    			tangent.needsUpdate = true;

    		}

    		if ( this.boundingBox !== null ) {

    			this.computeBoundingBox();

    		}

    		if ( this.boundingSphere !== null ) {

    			this.computeBoundingSphere();

    		}

    		return this;

    	},

    	rotateX: function ( angle ) {

    		// rotate geometry around world x-axis

    		_m1$2.makeRotationX( angle );

    		this.applyMatrix( _m1$2 );

    		return this;

    	},

    	rotateY: function ( angle ) {

    		// rotate geometry around world y-axis

    		_m1$2.makeRotationY( angle );

    		this.applyMatrix( _m1$2 );

    		return this;

    	},

    	rotateZ: function ( angle ) {

    		// rotate geometry around world z-axis

    		_m1$2.makeRotationZ( angle );

    		this.applyMatrix( _m1$2 );

    		return this;

    	},

    	translate: function ( x, y, z ) {

    		// translate geometry

    		_m1$2.makeTranslation( x, y, z );

    		this.applyMatrix( _m1$2 );

    		return this;

    	},

    	scale: function ( x, y, z ) {

    		// scale geometry

    		_m1$2.makeScale( x, y, z );

    		this.applyMatrix( _m1$2 );

    		return this;

    	},

    	lookAt: function ( vector ) {

    		_obj.lookAt( vector );

    		_obj.updateMatrix();

    		this.applyMatrix( _obj.matrix );

    		return this;

    	},

    	center: function () {

    		this.computeBoundingBox();

    		this.boundingBox.getCenter( _offset ).negate();

    		this.translate( _offset.x, _offset.y, _offset.z );

    		return this;

    	},

    	setFromObject: function ( object ) {

    		// console.log( 'THREE.BufferGeometry.setFromObject(). Converting', object, this );

    		var geometry = object.geometry;

    		if ( object.isPoints || object.isLine ) {

    			var positions = new Float32BufferAttribute( geometry.vertices.length * 3, 3 );
    			var colors = new Float32BufferAttribute( geometry.colors.length * 3, 3 );

    			this.addAttribute( 'position', positions.copyVector3sArray( geometry.vertices ) );
    			this.addAttribute( 'color', colors.copyColorsArray( geometry.colors ) );

    			if ( geometry.lineDistances && geometry.lineDistances.length === geometry.vertices.length ) {

    				var lineDistances = new Float32BufferAttribute( geometry.lineDistances.length, 1 );

    				this.addAttribute( 'lineDistance', lineDistances.copyArray( geometry.lineDistances ) );

    			}

    			if ( geometry.boundingSphere !== null ) {

    				this.boundingSphere = geometry.boundingSphere.clone();

    			}

    			if ( geometry.boundingBox !== null ) {

    				this.boundingBox = geometry.boundingBox.clone();

    			}

    		} else if ( object.isMesh ) {

    			if ( geometry && geometry.isGeometry ) {

    				this.fromGeometry( geometry );

    			}

    		}

    		return this;

    	},

    	setFromPoints: function ( points ) {

    		var position = [];

    		for ( var i = 0, l = points.length; i < l; i ++ ) {

    			var point = points[ i ];
    			position.push( point.x, point.y, point.z || 0 );

    		}

    		this.addAttribute( 'position', new Float32BufferAttribute( position, 3 ) );

    		return this;

    	},

    	updateFromObject: function ( object ) {

    		var geometry = object.geometry;

    		if ( object.isMesh ) {

    			var direct = geometry.__directGeometry;

    			if ( geometry.elementsNeedUpdate === true ) {

    				direct = undefined;
    				geometry.elementsNeedUpdate = false;

    			}

    			if ( direct === undefined ) {

    				return this.fromGeometry( geometry );

    			}

    			direct.verticesNeedUpdate = geometry.verticesNeedUpdate;
    			direct.normalsNeedUpdate = geometry.normalsNeedUpdate;
    			direct.colorsNeedUpdate = geometry.colorsNeedUpdate;
    			direct.uvsNeedUpdate = geometry.uvsNeedUpdate;
    			direct.groupsNeedUpdate = geometry.groupsNeedUpdate;

    			geometry.verticesNeedUpdate = false;
    			geometry.normalsNeedUpdate = false;
    			geometry.colorsNeedUpdate = false;
    			geometry.uvsNeedUpdate = false;
    			geometry.groupsNeedUpdate = false;

    			geometry = direct;

    		}

    		var attribute;

    		if ( geometry.verticesNeedUpdate === true ) {

    			attribute = this.attributes.position;

    			if ( attribute !== undefined ) {

    				attribute.copyVector3sArray( geometry.vertices );
    				attribute.needsUpdate = true;

    			}

    			geometry.verticesNeedUpdate = false;

    		}

    		if ( geometry.normalsNeedUpdate === true ) {

    			attribute = this.attributes.normal;

    			if ( attribute !== undefined ) {

    				attribute.copyVector3sArray( geometry.normals );
    				attribute.needsUpdate = true;

    			}

    			geometry.normalsNeedUpdate = false;

    		}

    		if ( geometry.colorsNeedUpdate === true ) {

    			attribute = this.attributes.color;

    			if ( attribute !== undefined ) {

    				attribute.copyColorsArray( geometry.colors );
    				attribute.needsUpdate = true;

    			}

    			geometry.colorsNeedUpdate = false;

    		}

    		if ( geometry.uvsNeedUpdate ) {

    			attribute = this.attributes.uv;

    			if ( attribute !== undefined ) {

    				attribute.copyVector2sArray( geometry.uvs );
    				attribute.needsUpdate = true;

    			}

    			geometry.uvsNeedUpdate = false;

    		}

    		if ( geometry.lineDistancesNeedUpdate ) {

    			attribute = this.attributes.lineDistance;

    			if ( attribute !== undefined ) {

    				attribute.copyArray( geometry.lineDistances );
    				attribute.needsUpdate = true;

    			}

    			geometry.lineDistancesNeedUpdate = false;

    		}

    		if ( geometry.groupsNeedUpdate ) {

    			geometry.computeGroups( object.geometry );
    			this.groups = geometry.groups;

    			geometry.groupsNeedUpdate = false;

    		}

    		return this;

    	},

    	fromGeometry: function ( geometry ) {

    		geometry.__directGeometry = new DirectGeometry().fromGeometry( geometry );

    		return this.fromDirectGeometry( geometry.__directGeometry );

    	},

    	fromDirectGeometry: function ( geometry ) {

    		var positions = new Float32Array( geometry.vertices.length * 3 );
    		this.addAttribute( 'position', new BufferAttribute( positions, 3 ).copyVector3sArray( geometry.vertices ) );

    		if ( geometry.normals.length > 0 ) {

    			var normals = new Float32Array( geometry.normals.length * 3 );
    			this.addAttribute( 'normal', new BufferAttribute( normals, 3 ).copyVector3sArray( geometry.normals ) );

    		}

    		if ( geometry.colors.length > 0 ) {

    			var colors = new Float32Array( geometry.colors.length * 3 );
    			this.addAttribute( 'color', new BufferAttribute( colors, 3 ).copyColorsArray( geometry.colors ) );

    		}

    		if ( geometry.uvs.length > 0 ) {

    			var uvs = new Float32Array( geometry.uvs.length * 2 );
    			this.addAttribute( 'uv', new BufferAttribute( uvs, 2 ).copyVector2sArray( geometry.uvs ) );

    		}

    		if ( geometry.uvs2.length > 0 ) {

    			var uvs2 = new Float32Array( geometry.uvs2.length * 2 );
    			this.addAttribute( 'uv2', new BufferAttribute( uvs2, 2 ).copyVector2sArray( geometry.uvs2 ) );

    		}

    		// groups

    		this.groups = geometry.groups;

    		// morphs

    		for ( var name in geometry.morphTargets ) {

    			var array = [];
    			var morphTargets = geometry.morphTargets[ name ];

    			for ( var i = 0, l = morphTargets.length; i < l; i ++ ) {

    				var morphTarget = morphTargets[ i ];

    				var attribute = new Float32BufferAttribute( morphTarget.data.length * 3, 3 );
    				attribute.name = morphTarget.name;

    				array.push( attribute.copyVector3sArray( morphTarget.data ) );

    			}

    			this.morphAttributes[ name ] = array;

    		}

    		// skinning

    		if ( geometry.skinIndices.length > 0 ) {

    			var skinIndices = new Float32BufferAttribute( geometry.skinIndices.length * 4, 4 );
    			this.addAttribute( 'skinIndex', skinIndices.copyVector4sArray( geometry.skinIndices ) );

    		}

    		if ( geometry.skinWeights.length > 0 ) {

    			var skinWeights = new Float32BufferAttribute( geometry.skinWeights.length * 4, 4 );
    			this.addAttribute( 'skinWeight', skinWeights.copyVector4sArray( geometry.skinWeights ) );

    		}

    		//

    		if ( geometry.boundingSphere !== null ) {

    			this.boundingSphere = geometry.boundingSphere.clone();

    		}

    		if ( geometry.boundingBox !== null ) {

    			this.boundingBox = geometry.boundingBox.clone();

    		}

    		return this;

    	},

    	computeBoundingBox: function () {

    		if ( this.boundingBox === null ) {

    			this.boundingBox = new Box3();

    		}

    		var position = this.attributes.position;
    		var morphAttributesPosition = this.morphAttributes.position;

    		if ( position !== undefined ) {

    			this.boundingBox.setFromBufferAttribute( position );

    			// process morph attributes if present

    			if ( morphAttributesPosition ) {

    				for ( var i = 0, il = morphAttributesPosition.length; i < il; i ++ ) {

    					var morphAttribute = morphAttributesPosition[ i ];
    					_box$1.setFromBufferAttribute( morphAttribute );

    					this.boundingBox.expandByPoint( _box$1.min );
    					this.boundingBox.expandByPoint( _box$1.max );

    				}

    			}

    		} else {

    			this.boundingBox.makeEmpty();

    		}

    		if ( isNaN( this.boundingBox.min.x ) || isNaN( this.boundingBox.min.y ) || isNaN( this.boundingBox.min.z ) ) {

    			console.error( 'THREE.BufferGeometry.computeBoundingBox: Computed min/max have NaN values. The "position" attribute is likely to have NaN values.', this );

    		}

    	},

    	computeBoundingSphere: function () {

    		if ( this.boundingSphere === null ) {

    			this.boundingSphere = new Sphere();

    		}

    		var position = this.attributes.position;
    		var morphAttributesPosition = this.morphAttributes.position;

    		if ( position ) {

    			// first, find the center of the bounding sphere

    			var center = this.boundingSphere.center;

    			_box$1.setFromBufferAttribute( position );

    			// process morph attributes if present

    			if ( morphAttributesPosition ) {

    				for ( var i = 0, il = morphAttributesPosition.length; i < il; i ++ ) {

    					var morphAttribute = morphAttributesPosition[ i ];
    					_boxMorphTargets.setFromBufferAttribute( morphAttribute );

    					_box$1.expandByPoint( _boxMorphTargets.min );
    					_box$1.expandByPoint( _boxMorphTargets.max );

    				}

    			}

    			_box$1.getCenter( center );

    			// second, try to find a boundingSphere with a radius smaller than the
    			// boundingSphere of the boundingBox: sqrt(3) smaller in the best case

    			var maxRadiusSq = 0;

    			for ( var i = 0, il = position.count; i < il; i ++ ) {

    				_vector$4.fromBufferAttribute( position, i );

    				maxRadiusSq = Math.max( maxRadiusSq, center.distanceToSquared( _vector$4 ) );

    			}

    			// process morph attributes if present

    			if ( morphAttributesPosition ) {

    				for ( var i = 0, il = morphAttributesPosition.length; i < il; i ++ ) {

    					var morphAttribute = morphAttributesPosition[ i ];

    					for ( var j = 0, jl = morphAttribute.count; j < jl; j ++ ) {

    						_vector$4.fromBufferAttribute( morphAttribute, j );

    						maxRadiusSq = Math.max( maxRadiusSq, center.distanceToSquared( _vector$4 ) );

    					}

    				}

    			}

    			this.boundingSphere.radius = Math.sqrt( maxRadiusSq );

    			if ( isNaN( this.boundingSphere.radius ) ) {

    				console.error( 'THREE.BufferGeometry.computeBoundingSphere(): Computed radius is NaN. The "position" attribute is likely to have NaN values.', this );

    			}

    		}

    	},

    	computeFaceNormals: function () {

    		// backwards compatibility

    	},

    	computeVertexNormals: function () {

    		var index = this.index;
    		var attributes = this.attributes;

    		if ( attributes.position ) {

    			var positions = attributes.position.array;

    			if ( attributes.normal === undefined ) {

    				this.addAttribute( 'normal', new BufferAttribute( new Float32Array( positions.length ), 3 ) );

    			} else {

    				// reset existing normals to zero

    				var array = attributes.normal.array;

    				for ( var i = 0, il = array.length; i < il; i ++ ) {

    					array[ i ] = 0;

    				}

    			}

    			var normals = attributes.normal.array;

    			var vA, vB, vC;
    			var pA = new Vector3(), pB = new Vector3(), pC = new Vector3();
    			var cb = new Vector3(), ab = new Vector3();

    			// indexed elements

    			if ( index ) {

    				var indices = index.array;

    				for ( var i = 0, il = index.count; i < il; i += 3 ) {

    					vA = indices[ i + 0 ] * 3;
    					vB = indices[ i + 1 ] * 3;
    					vC = indices[ i + 2 ] * 3;

    					pA.fromArray( positions, vA );
    					pB.fromArray( positions, vB );
    					pC.fromArray( positions, vC );

    					cb.subVectors( pC, pB );
    					ab.subVectors( pA, pB );
    					cb.cross( ab );

    					normals[ vA ] += cb.x;
    					normals[ vA + 1 ] += cb.y;
    					normals[ vA + 2 ] += cb.z;

    					normals[ vB ] += cb.x;
    					normals[ vB + 1 ] += cb.y;
    					normals[ vB + 2 ] += cb.z;

    					normals[ vC ] += cb.x;
    					normals[ vC + 1 ] += cb.y;
    					normals[ vC + 2 ] += cb.z;

    				}

    			} else {

    				// non-indexed elements (unconnected triangle soup)

    				for ( var i = 0, il = positions.length; i < il; i += 9 ) {

    					pA.fromArray( positions, i );
    					pB.fromArray( positions, i + 3 );
    					pC.fromArray( positions, i + 6 );

    					cb.subVectors( pC, pB );
    					ab.subVectors( pA, pB );
    					cb.cross( ab );

    					normals[ i ] = cb.x;
    					normals[ i + 1 ] = cb.y;
    					normals[ i + 2 ] = cb.z;

    					normals[ i + 3 ] = cb.x;
    					normals[ i + 4 ] = cb.y;
    					normals[ i + 5 ] = cb.z;

    					normals[ i + 6 ] = cb.x;
    					normals[ i + 7 ] = cb.y;
    					normals[ i + 8 ] = cb.z;

    				}

    			}

    			this.normalizeNormals();

    			attributes.normal.needsUpdate = true;

    		}

    	},

    	merge: function ( geometry, offset ) {

    		if ( ! ( geometry && geometry.isBufferGeometry ) ) {

    			console.error( 'THREE.BufferGeometry.merge(): geometry not an instance of THREE.BufferGeometry.', geometry );
    			return;

    		}

    		if ( offset === undefined ) {

    			offset = 0;

    			console.warn(
    				'THREE.BufferGeometry.merge(): Overwriting original geometry, starting at offset=0. '
    				+ 'Use BufferGeometryUtils.mergeBufferGeometries() for lossless merge.'
    			);

    		}

    		var attributes = this.attributes;

    		for ( var key in attributes ) {

    			if ( geometry.attributes[ key ] === undefined ) continue;

    			var attribute1 = attributes[ key ];
    			var attributeArray1 = attribute1.array;

    			var attribute2 = geometry.attributes[ key ];
    			var attributeArray2 = attribute2.array;

    			var attributeOffset = attribute2.itemSize * offset;
    			var length = Math.min( attributeArray2.length, attributeArray1.length - attributeOffset );

    			for ( var i = 0, j = attributeOffset; i < length; i ++, j ++ ) {

    				attributeArray1[ j ] = attributeArray2[ i ];

    			}

    		}

    		return this;

    	},

    	normalizeNormals: function () {

    		var normals = this.attributes.normal;

    		for ( var i = 0, il = normals.count; i < il; i ++ ) {

    			_vector$4.x = normals.getX( i );
    			_vector$4.y = normals.getY( i );
    			_vector$4.z = normals.getZ( i );

    			_vector$4.normalize();

    			normals.setXYZ( i, _vector$4.x, _vector$4.y, _vector$4.z );

    		}

    	},

    	toNonIndexed: function () {

    		function convertBufferAttribute( attribute, indices ) {

    			var array = attribute.array;
    			var itemSize = attribute.itemSize;

    			var array2 = new array.constructor( indices.length * itemSize );

    			var index = 0, index2 = 0;

    			for ( var i = 0, l = indices.length; i < l; i ++ ) {

    				index = indices[ i ] * itemSize;

    				for ( var j = 0; j < itemSize; j ++ ) {

    					array2[ index2 ++ ] = array[ index ++ ];

    				}

    			}

    			return new BufferAttribute( array2, itemSize );

    		}

    		//

    		if ( this.index === null ) {

    			console.warn( 'THREE.BufferGeometry.toNonIndexed(): Geometry is already non-indexed.' );
    			return this;

    		}

    		var geometry2 = new BufferGeometry();

    		var indices = this.index.array;
    		var attributes = this.attributes;

    		// attributes

    		for ( var name in attributes ) {

    			var attribute = attributes[ name ];

    			var newAttribute = convertBufferAttribute( attribute, indices );

    			geometry2.addAttribute( name, newAttribute );

    		}

    		// morph attributes

    		var morphAttributes = this.morphAttributes;

    		for ( name in morphAttributes ) {

    			var morphArray = [];
    			var morphAttribute = morphAttributes[ name ]; // morphAttribute: array of Float32BufferAttributes

    			for ( var i = 0, il = morphAttribute.length; i < il; i ++ ) {

    				var attribute = morphAttribute[ i ];

    				var newAttribute = convertBufferAttribute( attribute, indices );

    				morphArray.push( newAttribute );

    			}

    			geometry2.morphAttributes[ name ] = morphArray;

    		}

    		// groups

    		var groups = this.groups;

    		for ( var i = 0, l = groups.length; i < l; i ++ ) {

    			var group = groups[ i ];
    			geometry2.addGroup( group.start, group.count, group.materialIndex );

    		}

    		return geometry2;

    	},

    	toJSON: function () {

    		var data = {
    			metadata: {
    				version: 4.5,
    				type: 'BufferGeometry',
    				generator: 'BufferGeometry.toJSON'
    			}
    		};

    		// standard BufferGeometry serialization

    		data.uuid = this.uuid;
    		data.type = this.type;
    		if ( this.name !== '' ) data.name = this.name;
    		if ( Object.keys( this.userData ).length > 0 ) data.userData = this.userData;

    		if ( this.parameters !== undefined ) {

    			var parameters = this.parameters;

    			for ( var key in parameters ) {

    				if ( parameters[ key ] !== undefined ) data[ key ] = parameters[ key ];

    			}

    			return data;

    		}

    		data.data = { attributes: {} };

    		var index = this.index;

    		if ( index !== null ) {

    			data.data.index = {
    				type: index.array.constructor.name,
    				array: Array.prototype.slice.call( index.array )
    			};

    		}

    		var attributes = this.attributes;

    		for ( var key in attributes ) {

    			var attribute = attributes[ key ];

    			var attributeData = attribute.toJSON();

    			if ( attribute.name !== '' ) attributeData.name = attribute.name;

    			data.data.attributes[ key ] = attributeData;

    		}

    		var morphAttributes = {};
    		var hasMorphAttributes = false;

    		for ( var key in this.morphAttributes ) {

    			var attributeArray = this.morphAttributes[ key ];

    			var array = [];

    			for ( var i = 0, il = attributeArray.length; i < il; i ++ ) {

    				var attribute = attributeArray[ i ];

    				var attributeData = attribute.toJSON();

    				if ( attribute.name !== '' ) attributeData.name = attribute.name;

    				array.push( attributeData );

    			}

    			if ( array.length > 0 ) {

    				morphAttributes[ key ] = array;

    				hasMorphAttributes = true;

    			}

    		}

    		if ( hasMorphAttributes ) data.data.morphAttributes = morphAttributes;

    		var groups = this.groups;

    		if ( groups.length > 0 ) {

    			data.data.groups = JSON.parse( JSON.stringify( groups ) );

    		}

    		var boundingSphere = this.boundingSphere;

    		if ( boundingSphere !== null ) {

    			data.data.boundingSphere = {
    				center: boundingSphere.center.toArray(),
    				radius: boundingSphere.radius
    			};

    		}

    		return data;

    	},

    	clone: function () {

    		/*
    		 // Handle primitives

    		 var parameters = this.parameters;

    		 if ( parameters !== undefined ) {

    		 var values = [];

    		 for ( var key in parameters ) {

    		 values.push( parameters[ key ] );

    		 }

    		 var geometry = Object.create( this.constructor.prototype );
    		 this.constructor.apply( geometry, values );
    		 return geometry;

    		 }

    		 return new this.constructor().copy( this );
    		 */

    		return new BufferGeometry().copy( this );

    	},

    	copy: function ( source ) {

    		var name, i, l;

    		// reset

    		this.index = null;
    		this.attributes = {};
    		this.morphAttributes = {};
    		this.groups = [];
    		this.boundingBox = null;
    		this.boundingSphere = null;

    		// name

    		this.name = source.name;

    		// index

    		var index = source.index;

    		if ( index !== null ) {

    			this.setIndex( index.clone() );

    		}

    		// attributes

    		var attributes = source.attributes;

    		for ( name in attributes ) {

    			var attribute = attributes[ name ];
    			this.addAttribute( name, attribute.clone() );

    		}

    		// morph attributes

    		var morphAttributes = source.morphAttributes;

    		for ( name in morphAttributes ) {

    			var array = [];
    			var morphAttribute = morphAttributes[ name ]; // morphAttribute: array of Float32BufferAttributes

    			for ( i = 0, l = morphAttribute.length; i < l; i ++ ) {

    				array.push( morphAttribute[ i ].clone() );

    			}

    			this.morphAttributes[ name ] = array;

    		}

    		// groups

    		var groups = source.groups;

    		for ( i = 0, l = groups.length; i < l; i ++ ) {

    			var group = groups[ i ];
    			this.addGroup( group.start, group.count, group.materialIndex );

    		}

    		// bounding box

    		var boundingBox = source.boundingBox;

    		if ( boundingBox !== null ) {

    			this.boundingBox = boundingBox.clone();

    		}

    		// bounding sphere

    		var boundingSphere = source.boundingSphere;

    		if ( boundingSphere !== null ) {

    			this.boundingSphere = boundingSphere.clone();

    		}

    		// draw range

    		this.drawRange.start = source.drawRange.start;
    		this.drawRange.count = source.drawRange.count;

    		// user data

    		this.userData = source.userData;

    		return this;

    	},

    	dispose: function () {

    		this.dispatchEvent( { type: 'dispose' } );

    	}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     * @author alteredq / http://alteredqualia.com/
     * @author mikael emtinger / http://gomo.se/
     * @author jonobr1 / http://jonobr1.com/
     */

    var _inverseMatrix = new Matrix4();
    var _ray = new Ray();
    var _sphere = new Sphere();

    var _vA = new Vector3();
    var _vB = new Vector3();
    var _vC = new Vector3();

    var _tempA = new Vector3();
    var _tempB = new Vector3();
    var _tempC = new Vector3();

    var _morphA = new Vector3();
    var _morphB = new Vector3();
    var _morphC = new Vector3();

    var _uvA = new Vector2();
    var _uvB = new Vector2();
    var _uvC = new Vector2();

    var _intersectionPoint = new Vector3();
    var _intersectionPointWorld = new Vector3();

    function Mesh( geometry, material ) {

    	Object3D.call( this );

    	this.type = 'Mesh';

    	this.geometry = geometry !== undefined ? geometry : new BufferGeometry();
    	this.material = material !== undefined ? material : new MeshBasicMaterial( { color: Math.random() * 0xffffff } );

    	this.drawMode = TrianglesDrawMode;

    	this.updateMorphTargets();

    }

    Mesh.prototype = Object.assign( Object.create( Object3D.prototype ), {

    	constructor: Mesh,

    	isMesh: true,

    	setDrawMode: function ( value ) {

    		this.drawMode = value;

    	},

    	copy: function ( source ) {

    		Object3D.prototype.copy.call( this, source );

    		this.drawMode = source.drawMode;

    		if ( source.morphTargetInfluences !== undefined ) {

    			this.morphTargetInfluences = source.morphTargetInfluences.slice();

    		}

    		if ( source.morphTargetDictionary !== undefined ) {

    			this.morphTargetDictionary = Object.assign( {}, source.morphTargetDictionary );

    		}

    		return this;

    	},

    	updateMorphTargets: function () {

    		var geometry = this.geometry;
    		var m, ml, name;

    		if ( geometry.isBufferGeometry ) {

    			var morphAttributes = geometry.morphAttributes;
    			var keys = Object.keys( morphAttributes );

    			if ( keys.length > 0 ) {

    				var morphAttribute = morphAttributes[ keys[ 0 ] ];

    				if ( morphAttribute !== undefined ) {

    					this.morphTargetInfluences = [];
    					this.morphTargetDictionary = {};

    					for ( m = 0, ml = morphAttribute.length; m < ml; m ++ ) {

    						name = morphAttribute[ m ].name || String( m );

    						this.morphTargetInfluences.push( 0 );
    						this.morphTargetDictionary[ name ] = m;

    					}

    				}

    			}

    		} else {

    			var morphTargets = geometry.morphTargets;

    			if ( morphTargets !== undefined && morphTargets.length > 0 ) {

    				console.error( 'THREE.Mesh.updateMorphTargets() no longer supports THREE.Geometry. Use THREE.BufferGeometry instead.' );

    			}

    		}

    	},

    	raycast: function ( raycaster, intersects ) {

    		var geometry = this.geometry;
    		var material = this.material;
    		var matrixWorld = this.matrixWorld;

    		if ( material === undefined ) return;

    		// Checking boundingSphere distance to ray

    		if ( geometry.boundingSphere === null ) geometry.computeBoundingSphere();

    		_sphere.copy( geometry.boundingSphere );
    		_sphere.applyMatrix4( matrixWorld );

    		if ( raycaster.ray.intersectsSphere( _sphere ) === false ) return;

    		//

    		_inverseMatrix.getInverse( matrixWorld );
    		_ray.copy( raycaster.ray ).applyMatrix4( _inverseMatrix );

    		// Check boundingBox before continuing

    		if ( geometry.boundingBox !== null ) {

    			if ( _ray.intersectsBox( geometry.boundingBox ) === false ) return;

    		}

    		var intersection;

    		if ( geometry.isBufferGeometry ) {

    			var a, b, c;
    			var index = geometry.index;
    			var position = geometry.attributes.position;
    			var morphPosition = geometry.morphAttributes.position;
    			var uv = geometry.attributes.uv;
    			var uv2 = geometry.attributes.uv2;
    			var groups = geometry.groups;
    			var drawRange = geometry.drawRange;
    			var i, j, il, jl;
    			var group, groupMaterial;
    			var start, end;

    			if ( index !== null ) {

    				// indexed buffer geometry

    				if ( Array.isArray( material ) ) {

    					for ( i = 0, il = groups.length; i < il; i ++ ) {

    						group = groups[ i ];
    						groupMaterial = material[ group.materialIndex ];

    						start = Math.max( group.start, drawRange.start );
    						end = Math.min( ( group.start + group.count ), ( drawRange.start + drawRange.count ) );

    						for ( j = start, jl = end; j < jl; j += 3 ) {

    							a = index.getX( j );
    							b = index.getX( j + 1 );
    							c = index.getX( j + 2 );

    							intersection = checkBufferGeometryIntersection( this, groupMaterial, raycaster, _ray, position, morphPosition, uv, uv2, a, b, c );

    							if ( intersection ) {

    								intersection.faceIndex = Math.floor( j / 3 ); // triangle number in indexed buffer semantics
    								intersection.face.materialIndex = group.materialIndex;
    								intersects.push( intersection );

    							}

    						}

    					}

    				} else {

    					start = Math.max( 0, drawRange.start );
    					end = Math.min( index.count, ( drawRange.start + drawRange.count ) );

    					for ( i = start, il = end; i < il; i += 3 ) {

    						a = index.getX( i );
    						b = index.getX( i + 1 );
    						c = index.getX( i + 2 );

    						intersection = checkBufferGeometryIntersection( this, material, raycaster, _ray, position, morphPosition, uv, uv2, a, b, c );

    						if ( intersection ) {

    							intersection.faceIndex = Math.floor( i / 3 ); // triangle number in indexed buffer semantics
    							intersects.push( intersection );

    						}

    					}

    				}

    			} else if ( position !== undefined ) {

    				// non-indexed buffer geometry

    				if ( Array.isArray( material ) ) {

    					for ( i = 0, il = groups.length; i < il; i ++ ) {

    						group = groups[ i ];
    						groupMaterial = material[ group.materialIndex ];

    						start = Math.max( group.start, drawRange.start );
    						end = Math.min( ( group.start + group.count ), ( drawRange.start + drawRange.count ) );

    						for ( j = start, jl = end; j < jl; j += 3 ) {

    							a = j;
    							b = j + 1;
    							c = j + 2;

    							intersection = checkBufferGeometryIntersection( this, groupMaterial, raycaster, _ray, position, morphPosition, uv, uv2, a, b, c );

    							if ( intersection ) {

    								intersection.faceIndex = Math.floor( j / 3 ); // triangle number in non-indexed buffer semantics
    								intersection.face.materialIndex = group.materialIndex;
    								intersects.push( intersection );

    							}

    						}

    					}

    				} else {

    					start = Math.max( 0, drawRange.start );
    					end = Math.min( position.count, ( drawRange.start + drawRange.count ) );

    					for ( i = start, il = end; i < il; i += 3 ) {

    						a = i;
    						b = i + 1;
    						c = i + 2;

    						intersection = checkBufferGeometryIntersection( this, material, raycaster, _ray, position, morphPosition, uv, uv2, a, b, c );

    						if ( intersection ) {

    							intersection.faceIndex = Math.floor( i / 3 ); // triangle number in non-indexed buffer semantics
    							intersects.push( intersection );

    						}

    					}

    				}

    			}

    		} else if ( geometry.isGeometry ) {

    			var fvA, fvB, fvC;
    			var isMultiMaterial = Array.isArray( material );

    			var vertices = geometry.vertices;
    			var faces = geometry.faces;
    			var uvs;

    			var faceVertexUvs = geometry.faceVertexUvs[ 0 ];
    			if ( faceVertexUvs.length > 0 ) uvs = faceVertexUvs;

    			for ( var f = 0, fl = faces.length; f < fl; f ++ ) {

    				var face = faces[ f ];
    				var faceMaterial = isMultiMaterial ? material[ face.materialIndex ] : material;

    				if ( faceMaterial === undefined ) continue;

    				fvA = vertices[ face.a ];
    				fvB = vertices[ face.b ];
    				fvC = vertices[ face.c ];

    				intersection = checkIntersection( this, faceMaterial, raycaster, _ray, fvA, fvB, fvC, _intersectionPoint );

    				if ( intersection ) {

    					if ( uvs && uvs[ f ] ) {

    						var uvs_f = uvs[ f ];
    						_uvA.copy( uvs_f[ 0 ] );
    						_uvB.copy( uvs_f[ 1 ] );
    						_uvC.copy( uvs_f[ 2 ] );

    						intersection.uv = Triangle.getUV( _intersectionPoint, fvA, fvB, fvC, _uvA, _uvB, _uvC, new Vector2() );

    					}

    					intersection.face = face;
    					intersection.faceIndex = f;
    					intersects.push( intersection );

    				}

    			}

    		}

    	},

    	clone: function () {

    		return new this.constructor( this.geometry, this.material ).copy( this );

    	}

    } );

    function checkIntersection( object, material, raycaster, ray, pA, pB, pC, point ) {

    	var intersect;

    	if ( material.side === BackSide ) {

    		intersect = ray.intersectTriangle( pC, pB, pA, true, point );

    	} else {

    		intersect = ray.intersectTriangle( pA, pB, pC, material.side !== DoubleSide, point );

    	}

    	if ( intersect === null ) return null;

    	_intersectionPointWorld.copy( point );
    	_intersectionPointWorld.applyMatrix4( object.matrixWorld );

    	var distance = raycaster.ray.origin.distanceTo( _intersectionPointWorld );

    	if ( distance < raycaster.near || distance > raycaster.far ) return null;

    	return {
    		distance: distance,
    		point: _intersectionPointWorld.clone(),
    		object: object
    	};

    }

    function checkBufferGeometryIntersection( object, material, raycaster, ray, position, morphPosition, uv, uv2, a, b, c ) {

    	_vA.fromBufferAttribute( position, a );
    	_vB.fromBufferAttribute( position, b );
    	_vC.fromBufferAttribute( position, c );

    	var morphInfluences = object.morphTargetInfluences;

    	if ( material.morphTargets && morphPosition && morphInfluences ) {

    		_morphA.set( 0, 0, 0 );
    		_morphB.set( 0, 0, 0 );
    		_morphC.set( 0, 0, 0 );

    		for ( var i = 0, il = morphPosition.length; i < il; i ++ ) {

    			var influence = morphInfluences[ i ];
    			var morphAttribute = morphPosition[ i ];

    			if ( influence === 0 ) continue;

    			_tempA.fromBufferAttribute( morphAttribute, a );
    			_tempB.fromBufferAttribute( morphAttribute, b );
    			_tempC.fromBufferAttribute( morphAttribute, c );

    			_morphA.addScaledVector( _tempA.sub( _vA ), influence );
    			_morphB.addScaledVector( _tempB.sub( _vB ), influence );
    			_morphC.addScaledVector( _tempC.sub( _vC ), influence );

    		}

    		_vA.add( _morphA );
    		_vB.add( _morphB );
    		_vC.add( _morphC );

    	}

    	var intersection = checkIntersection( object, material, raycaster, ray, _vA, _vB, _vC, _intersectionPoint );

    	if ( intersection ) {

    		if ( uv ) {

    			_uvA.fromBufferAttribute( uv, a );
    			_uvB.fromBufferAttribute( uv, b );
    			_uvC.fromBufferAttribute( uv, c );

    			intersection.uv = Triangle.getUV( _intersectionPoint, _vA, _vB, _vC, _uvA, _uvB, _uvC, new Vector2() );

    		}

    		if ( uv2 ) {

    			_uvA.fromBufferAttribute( uv2, a );
    			_uvB.fromBufferAttribute( uv2, b );
    			_uvC.fromBufferAttribute( uv2, c );

    			intersection.uv2 = Triangle.getUV( _intersectionPoint, _vA, _vB, _vC, _uvA, _uvB, _uvC, new Vector2() );

    		}

    		var face = new Face3( a, b, c );
    		Triangle.getNormal( _vA, _vB, _vC, face.normal );

    		intersection.face = face;

    	}

    	return intersection;

    }

    /**
     * @author mrdoob / http://mrdoob.com/
     * @author kile / http://kile.stravaganza.org/
     * @author alteredq / http://alteredqualia.com/
     * @author mikael emtinger / http://gomo.se/
     * @author zz85 / http://www.lab4games.net/zz85/blog
     * @author bhouston / http://clara.io
     */

    var _geometryId = 0; // Geometry uses even numbers as Id
    var _m1$3 = new Matrix4();
    var _obj$1 = new Object3D();
    var _offset$1 = new Vector3();

    function Geometry() {

    	Object.defineProperty( this, 'id', { value: _geometryId += 2 } );

    	this.uuid = _Math.generateUUID();

    	this.name = '';
    	this.type = 'Geometry';

    	this.vertices = [];
    	this.colors = [];
    	this.faces = [];
    	this.faceVertexUvs = [[]];

    	this.morphTargets = [];
    	this.morphNormals = [];

    	this.skinWeights = [];
    	this.skinIndices = [];

    	this.lineDistances = [];

    	this.boundingBox = null;
    	this.boundingSphere = null;

    	// update flags

    	this.elementsNeedUpdate = false;
    	this.verticesNeedUpdate = false;
    	this.uvsNeedUpdate = false;
    	this.normalsNeedUpdate = false;
    	this.colorsNeedUpdate = false;
    	this.lineDistancesNeedUpdate = false;
    	this.groupsNeedUpdate = false;

    }

    Geometry.prototype = Object.assign( Object.create( EventDispatcher.prototype ), {

    	constructor: Geometry,

    	isGeometry: true,

    	applyMatrix: function ( matrix ) {

    		var normalMatrix = new Matrix3().getNormalMatrix( matrix );

    		for ( var i = 0, il = this.vertices.length; i < il; i ++ ) {

    			var vertex = this.vertices[ i ];
    			vertex.applyMatrix4( matrix );

    		}

    		for ( var i = 0, il = this.faces.length; i < il; i ++ ) {

    			var face = this.faces[ i ];
    			face.normal.applyMatrix3( normalMatrix ).normalize();

    			for ( var j = 0, jl = face.vertexNormals.length; j < jl; j ++ ) {

    				face.vertexNormals[ j ].applyMatrix3( normalMatrix ).normalize();

    			}

    		}

    		if ( this.boundingBox !== null ) {

    			this.computeBoundingBox();

    		}

    		if ( this.boundingSphere !== null ) {

    			this.computeBoundingSphere();

    		}

    		this.verticesNeedUpdate = true;
    		this.normalsNeedUpdate = true;

    		return this;

    	},

    	rotateX: function ( angle ) {

    		// rotate geometry around world x-axis

    		_m1$3.makeRotationX( angle );

    		this.applyMatrix( _m1$3 );

    		return this;

    	},

    	rotateY: function ( angle ) {

    		// rotate geometry around world y-axis

    		_m1$3.makeRotationY( angle );

    		this.applyMatrix( _m1$3 );

    		return this;

    	},

    	rotateZ: function ( angle ) {

    		// rotate geometry around world z-axis

    		_m1$3.makeRotationZ( angle );

    		this.applyMatrix( _m1$3 );

    		return this;

    	},

    	translate: function ( x, y, z ) {

    		// translate geometry

    		_m1$3.makeTranslation( x, y, z );

    		this.applyMatrix( _m1$3 );

    		return this;

    	},

    	scale: function ( x, y, z ) {

    		// scale geometry

    		_m1$3.makeScale( x, y, z );

    		this.applyMatrix( _m1$3 );

    		return this;

    	},

    	lookAt: function ( vector ) {

    		_obj$1.lookAt( vector );

    		_obj$1.updateMatrix();

    		this.applyMatrix( _obj$1.matrix );

    		return this;

    	},

    	fromBufferGeometry: function ( geometry ) {

    		var scope = this;

    		var indices = geometry.index !== null ? geometry.index.array : undefined;
    		var attributes = geometry.attributes;

    		var positions = attributes.position.array;
    		var normals = attributes.normal !== undefined ? attributes.normal.array : undefined;
    		var colors = attributes.color !== undefined ? attributes.color.array : undefined;
    		var uvs = attributes.uv !== undefined ? attributes.uv.array : undefined;
    		var uvs2 = attributes.uv2 !== undefined ? attributes.uv2.array : undefined;

    		if ( uvs2 !== undefined ) this.faceVertexUvs[ 1 ] = [];

    		for ( var i = 0; i < positions.length; i += 3 ) {

    			scope.vertices.push( new Vector3().fromArray( positions, i ) );

    			if ( colors !== undefined ) {

    				scope.colors.push( new Color$7().fromArray( colors, i ) );

    			}

    		}

    		function addFace( a, b, c, materialIndex ) {

    			var vertexColors = ( colors === undefined ) ? [] : [
    				scope.colors[ a ].clone(),
    				scope.colors[ b ].clone(),
    				scope.colors[ c ].clone() ];

    			var vertexNormals = ( normals === undefined ) ? [] : [
    				new Vector3().fromArray( normals, a * 3 ),
    				new Vector3().fromArray( normals, b * 3 ),
    				new Vector3().fromArray( normals, c * 3 )
    			];

    			var face = new Face3( a, b, c, vertexNormals, vertexColors, materialIndex );

    			scope.faces.push( face );

    			if ( uvs !== undefined ) {

    				scope.faceVertexUvs[ 0 ].push( [
    					new Vector2().fromArray( uvs, a * 2 ),
    					new Vector2().fromArray( uvs, b * 2 ),
    					new Vector2().fromArray( uvs, c * 2 )
    				] );

    			}

    			if ( uvs2 !== undefined ) {

    				scope.faceVertexUvs[ 1 ].push( [
    					new Vector2().fromArray( uvs2, a * 2 ),
    					new Vector2().fromArray( uvs2, b * 2 ),
    					new Vector2().fromArray( uvs2, c * 2 )
    				] );

    			}

    		}

    		var groups = geometry.groups;

    		if ( groups.length > 0 ) {

    			for ( var i = 0; i < groups.length; i ++ ) {

    				var group = groups[ i ];

    				var start = group.start;
    				var count = group.count;

    				for ( var j = start, jl = start + count; j < jl; j += 3 ) {

    					if ( indices !== undefined ) {

    						addFace( indices[ j ], indices[ j + 1 ], indices[ j + 2 ], group.materialIndex );

    					} else {

    						addFace( j, j + 1, j + 2, group.materialIndex );

    					}

    				}

    			}

    		} else {

    			if ( indices !== undefined ) {

    				for ( var i = 0; i < indices.length; i += 3 ) {

    					addFace( indices[ i ], indices[ i + 1 ], indices[ i + 2 ] );

    				}

    			} else {

    				for ( var i = 0; i < positions.length / 3; i += 3 ) {

    					addFace( i, i + 1, i + 2 );

    				}

    			}

    		}

    		this.computeFaceNormals();

    		if ( geometry.boundingBox !== null ) {

    			this.boundingBox = geometry.boundingBox.clone();

    		}

    		if ( geometry.boundingSphere !== null ) {

    			this.boundingSphere = geometry.boundingSphere.clone();

    		}

    		return this;

    	},

    	center: function () {

    		this.computeBoundingBox();

    		this.boundingBox.getCenter( _offset$1 ).negate();

    		this.translate( _offset$1.x, _offset$1.y, _offset$1.z );

    		return this;

    	},

    	normalize: function () {

    		this.computeBoundingSphere();

    		var center = this.boundingSphere.center;
    		var radius = this.boundingSphere.radius;

    		var s = radius === 0 ? 1 : 1.0 / radius;

    		var matrix = new Matrix4();
    		matrix.set(
    			s, 0, 0, - s * center.x,
    			0, s, 0, - s * center.y,
    			0, 0, s, - s * center.z,
    			0, 0, 0, 1
    		);

    		this.applyMatrix( matrix );

    		return this;

    	},

    	computeFaceNormals: function () {

    		var cb = new Vector3(), ab = new Vector3();

    		for ( var f = 0, fl = this.faces.length; f < fl; f ++ ) {

    			var face = this.faces[ f ];

    			var vA = this.vertices[ face.a ];
    			var vB = this.vertices[ face.b ];
    			var vC = this.vertices[ face.c ];

    			cb.subVectors( vC, vB );
    			ab.subVectors( vA, vB );
    			cb.cross( ab );

    			cb.normalize();

    			face.normal.copy( cb );

    		}

    	},

    	computeVertexNormals: function ( areaWeighted ) {

    		if ( areaWeighted === undefined ) areaWeighted = true;

    		var v, vl, f, fl, face, vertices;

    		vertices = new Array( this.vertices.length );

    		for ( v = 0, vl = this.vertices.length; v < vl; v ++ ) {

    			vertices[ v ] = new Vector3();

    		}

    		if ( areaWeighted ) {

    			// vertex normals weighted by triangle areas
    			// http://www.iquilezles.org/www/articles/normals/normals.htm

    			var vA, vB, vC;
    			var cb = new Vector3(), ab = new Vector3();

    			for ( f = 0, fl = this.faces.length; f < fl; f ++ ) {

    				face = this.faces[ f ];

    				vA = this.vertices[ face.a ];
    				vB = this.vertices[ face.b ];
    				vC = this.vertices[ face.c ];

    				cb.subVectors( vC, vB );
    				ab.subVectors( vA, vB );
    				cb.cross( ab );

    				vertices[ face.a ].add( cb );
    				vertices[ face.b ].add( cb );
    				vertices[ face.c ].add( cb );

    			}

    		} else {

    			this.computeFaceNormals();

    			for ( f = 0, fl = this.faces.length; f < fl; f ++ ) {

    				face = this.faces[ f ];

    				vertices[ face.a ].add( face.normal );
    				vertices[ face.b ].add( face.normal );
    				vertices[ face.c ].add( face.normal );

    			}

    		}

    		for ( v = 0, vl = this.vertices.length; v < vl; v ++ ) {

    			vertices[ v ].normalize();

    		}

    		for ( f = 0, fl = this.faces.length; f < fl; f ++ ) {

    			face = this.faces[ f ];

    			var vertexNormals = face.vertexNormals;

    			if ( vertexNormals.length === 3 ) {

    				vertexNormals[ 0 ].copy( vertices[ face.a ] );
    				vertexNormals[ 1 ].copy( vertices[ face.b ] );
    				vertexNormals[ 2 ].copy( vertices[ face.c ] );

    			} else {

    				vertexNormals[ 0 ] = vertices[ face.a ].clone();
    				vertexNormals[ 1 ] = vertices[ face.b ].clone();
    				vertexNormals[ 2 ] = vertices[ face.c ].clone();

    			}

    		}

    		if ( this.faces.length > 0 ) {

    			this.normalsNeedUpdate = true;

    		}

    	},

    	computeFlatVertexNormals: function () {

    		var f, fl, face;

    		this.computeFaceNormals();

    		for ( f = 0, fl = this.faces.length; f < fl; f ++ ) {

    			face = this.faces[ f ];

    			var vertexNormals = face.vertexNormals;

    			if ( vertexNormals.length === 3 ) {

    				vertexNormals[ 0 ].copy( face.normal );
    				vertexNormals[ 1 ].copy( face.normal );
    				vertexNormals[ 2 ].copy( face.normal );

    			} else {

    				vertexNormals[ 0 ] = face.normal.clone();
    				vertexNormals[ 1 ] = face.normal.clone();
    				vertexNormals[ 2 ] = face.normal.clone();

    			}

    		}

    		if ( this.faces.length > 0 ) {

    			this.normalsNeedUpdate = true;

    		}

    	},

    	computeMorphNormals: function () {

    		var i, il, f, fl, face;

    		// save original normals
    		// - create temp variables on first access
    		//   otherwise just copy (for faster repeated calls)

    		for ( f = 0, fl = this.faces.length; f < fl; f ++ ) {

    			face = this.faces[ f ];

    			if ( ! face.__originalFaceNormal ) {

    				face.__originalFaceNormal = face.normal.clone();

    			} else {

    				face.__originalFaceNormal.copy( face.normal );

    			}

    			if ( ! face.__originalVertexNormals ) face.__originalVertexNormals = [];

    			for ( i = 0, il = face.vertexNormals.length; i < il; i ++ ) {

    				if ( ! face.__originalVertexNormals[ i ] ) {

    					face.__originalVertexNormals[ i ] = face.vertexNormals[ i ].clone();

    				} else {

    					face.__originalVertexNormals[ i ].copy( face.vertexNormals[ i ] );

    				}

    			}

    		}

    		// use temp geometry to compute face and vertex normals for each morph

    		var tmpGeo = new Geometry();
    		tmpGeo.faces = this.faces;

    		for ( i = 0, il = this.morphTargets.length; i < il; i ++ ) {

    			// create on first access

    			if ( ! this.morphNormals[ i ] ) {

    				this.morphNormals[ i ] = {};
    				this.morphNormals[ i ].faceNormals = [];
    				this.morphNormals[ i ].vertexNormals = [];

    				var dstNormalsFace = this.morphNormals[ i ].faceNormals;
    				var dstNormalsVertex = this.morphNormals[ i ].vertexNormals;

    				var faceNormal, vertexNormals;

    				for ( f = 0, fl = this.faces.length; f < fl; f ++ ) {

    					faceNormal = new Vector3();
    					vertexNormals = { a: new Vector3(), b: new Vector3(), c: new Vector3() };

    					dstNormalsFace.push( faceNormal );
    					dstNormalsVertex.push( vertexNormals );

    				}

    			}

    			var morphNormals = this.morphNormals[ i ];

    			// set vertices to morph target

    			tmpGeo.vertices = this.morphTargets[ i ].vertices;

    			// compute morph normals

    			tmpGeo.computeFaceNormals();
    			tmpGeo.computeVertexNormals();

    			// store morph normals

    			var faceNormal, vertexNormals;

    			for ( f = 0, fl = this.faces.length; f < fl; f ++ ) {

    				face = this.faces[ f ];

    				faceNormal = morphNormals.faceNormals[ f ];
    				vertexNormals = morphNormals.vertexNormals[ f ];

    				faceNormal.copy( face.normal );

    				vertexNormals.a.copy( face.vertexNormals[ 0 ] );
    				vertexNormals.b.copy( face.vertexNormals[ 1 ] );
    				vertexNormals.c.copy( face.vertexNormals[ 2 ] );

    			}

    		}

    		// restore original normals

    		for ( f = 0, fl = this.faces.length; f < fl; f ++ ) {

    			face = this.faces[ f ];

    			face.normal = face.__originalFaceNormal;
    			face.vertexNormals = face.__originalVertexNormals;

    		}

    	},

    	computeBoundingBox: function () {

    		if ( this.boundingBox === null ) {

    			this.boundingBox = new Box3();

    		}

    		this.boundingBox.setFromPoints( this.vertices );

    	},

    	computeBoundingSphere: function () {

    		if ( this.boundingSphere === null ) {

    			this.boundingSphere = new Sphere();

    		}

    		this.boundingSphere.setFromPoints( this.vertices );

    	},

    	merge: function ( geometry, matrix, materialIndexOffset ) {

    		if ( ! ( geometry && geometry.isGeometry ) ) {

    			console.error( 'THREE.Geometry.merge(): geometry not an instance of THREE.Geometry.', geometry );
    			return;

    		}

    		var normalMatrix,
    			vertexOffset = this.vertices.length,
    			vertices1 = this.vertices,
    			vertices2 = geometry.vertices,
    			faces1 = this.faces,
    			faces2 = geometry.faces,
    			colors1 = this.colors,
    			colors2 = geometry.colors;

    		if ( materialIndexOffset === undefined ) materialIndexOffset = 0;

    		if ( matrix !== undefined ) {

    			normalMatrix = new Matrix3().getNormalMatrix( matrix );

    		}

    		// vertices

    		for ( var i = 0, il = vertices2.length; i < il; i ++ ) {

    			var vertex = vertices2[ i ];

    			var vertexCopy = vertex.clone();

    			if ( matrix !== undefined ) vertexCopy.applyMatrix4( matrix );

    			vertices1.push( vertexCopy );

    		}

    		// colors

    		for ( var i = 0, il = colors2.length; i < il; i ++ ) {

    			colors1.push( colors2[ i ].clone() );

    		}

    		// faces

    		for ( i = 0, il = faces2.length; i < il; i ++ ) {

    			var face = faces2[ i ], faceCopy, normal, color,
    				faceVertexNormals = face.vertexNormals,
    				faceVertexColors = face.vertexColors;

    			faceCopy = new Face3( face.a + vertexOffset, face.b + vertexOffset, face.c + vertexOffset );
    			faceCopy.normal.copy( face.normal );

    			if ( normalMatrix !== undefined ) {

    				faceCopy.normal.applyMatrix3( normalMatrix ).normalize();

    			}

    			for ( var j = 0, jl = faceVertexNormals.length; j < jl; j ++ ) {

    				normal = faceVertexNormals[ j ].clone();

    				if ( normalMatrix !== undefined ) {

    					normal.applyMatrix3( normalMatrix ).normalize();

    				}

    				faceCopy.vertexNormals.push( normal );

    			}

    			faceCopy.color.copy( face.color );

    			for ( var j = 0, jl = faceVertexColors.length; j < jl; j ++ ) {

    				color = faceVertexColors[ j ];
    				faceCopy.vertexColors.push( color.clone() );

    			}

    			faceCopy.materialIndex = face.materialIndex + materialIndexOffset;

    			faces1.push( faceCopy );

    		}

    		// uvs

    		for ( var i = 0, il = geometry.faceVertexUvs.length; i < il; i ++ ) {

    			var faceVertexUvs2 = geometry.faceVertexUvs[ i ];

    			if ( this.faceVertexUvs[ i ] === undefined ) this.faceVertexUvs[ i ] = [];

    			for ( var j = 0, jl = faceVertexUvs2.length; j < jl; j ++ ) {

    				var uvs2 = faceVertexUvs2[ j ], uvsCopy = [];

    				for ( var k = 0, kl = uvs2.length; k < kl; k ++ ) {

    					uvsCopy.push( uvs2[ k ].clone() );

    				}

    				this.faceVertexUvs[ i ].push( uvsCopy );

    			}

    		}

    	},

    	mergeMesh: function ( mesh ) {

    		if ( ! ( mesh && mesh.isMesh ) ) {

    			console.error( 'THREE.Geometry.mergeMesh(): mesh not an instance of THREE.Mesh.', mesh );
    			return;

    		}

    		if ( mesh.matrixAutoUpdate ) mesh.updateMatrix();

    		this.merge( mesh.geometry, mesh.matrix );

    	},

    	/*
    	 * Checks for duplicate vertices with hashmap.
    	 * Duplicated vertices are removed
    	 * and faces' vertices are updated.
    	 */

    	mergeVertices: function () {

    		var verticesMap = {}; // Hashmap for looking up vertices by position coordinates (and making sure they are unique)
    		var unique = [], changes = [];

    		var v, key;
    		var precisionPoints = 4; // number of decimal points, e.g. 4 for epsilon of 0.0001
    		var precision = Math.pow( 10, precisionPoints );
    		var i, il, face;
    		var indices, j, jl;

    		for ( i = 0, il = this.vertices.length; i < il; i ++ ) {

    			v = this.vertices[ i ];
    			key = Math.round( v.x * precision ) + '_' + Math.round( v.y * precision ) + '_' + Math.round( v.z * precision );

    			if ( verticesMap[ key ] === undefined ) {

    				verticesMap[ key ] = i;
    				unique.push( this.vertices[ i ] );
    				changes[ i ] = unique.length - 1;

    			} else {

    				//console.log('Duplicate vertex found. ', i, ' could be using ', verticesMap[key]);
    				changes[ i ] = changes[ verticesMap[ key ] ];

    			}

    		}


    		// if faces are completely degenerate after merging vertices, we
    		// have to remove them from the geometry.
    		var faceIndicesToRemove = [];

    		for ( i = 0, il = this.faces.length; i < il; i ++ ) {

    			face = this.faces[ i ];

    			face.a = changes[ face.a ];
    			face.b = changes[ face.b ];
    			face.c = changes[ face.c ];

    			indices = [ face.a, face.b, face.c ];

    			// if any duplicate vertices are found in a Face3
    			// we have to remove the face as nothing can be saved
    			for ( var n = 0; n < 3; n ++ ) {

    				if ( indices[ n ] === indices[ ( n + 1 ) % 3 ] ) {

    					faceIndicesToRemove.push( i );
    					break;

    				}

    			}

    		}

    		for ( i = faceIndicesToRemove.length - 1; i >= 0; i -- ) {

    			var idx = faceIndicesToRemove[ i ];

    			this.faces.splice( idx, 1 );

    			for ( j = 0, jl = this.faceVertexUvs.length; j < jl; j ++ ) {

    				this.faceVertexUvs[ j ].splice( idx, 1 );

    			}

    		}

    		// Use unique set of vertices

    		var diff = this.vertices.length - unique.length;
    		this.vertices = unique;
    		return diff;

    	},

    	setFromPoints: function ( points ) {

    		this.vertices = [];

    		for ( var i = 0, l = points.length; i < l; i ++ ) {

    			var point = points[ i ];
    			this.vertices.push( new Vector3( point.x, point.y, point.z || 0 ) );

    		}

    		return this;

    	},

    	sortFacesByMaterialIndex: function () {

    		var faces = this.faces;
    		var length = faces.length;

    		// tag faces

    		for ( var i = 0; i < length; i ++ ) {

    			faces[ i ]._id = i;

    		}

    		// sort faces

    		function materialIndexSort( a, b ) {

    			return a.materialIndex - b.materialIndex;

    		}

    		faces.sort( materialIndexSort );

    		// sort uvs

    		var uvs1 = this.faceVertexUvs[ 0 ];
    		var uvs2 = this.faceVertexUvs[ 1 ];

    		var newUvs1, newUvs2;

    		if ( uvs1 && uvs1.length === length ) newUvs1 = [];
    		if ( uvs2 && uvs2.length === length ) newUvs2 = [];

    		for ( var i = 0; i < length; i ++ ) {

    			var id = faces[ i ]._id;

    			if ( newUvs1 ) newUvs1.push( uvs1[ id ] );
    			if ( newUvs2 ) newUvs2.push( uvs2[ id ] );

    		}

    		if ( newUvs1 ) this.faceVertexUvs[ 0 ] = newUvs1;
    		if ( newUvs2 ) this.faceVertexUvs[ 1 ] = newUvs2;

    	},

    	toJSON: function () {

    		var data = {
    			metadata: {
    				version: 4.5,
    				type: 'Geometry',
    				generator: 'Geometry.toJSON'
    			}
    		};

    		// standard Geometry serialization

    		data.uuid = this.uuid;
    		data.type = this.type;
    		if ( this.name !== '' ) data.name = this.name;

    		if ( this.parameters !== undefined ) {

    			var parameters = this.parameters;

    			for ( var key in parameters ) {

    				if ( parameters[ key ] !== undefined ) data[ key ] = parameters[ key ];

    			}

    			return data;

    		}

    		var vertices = [];

    		for ( var i = 0; i < this.vertices.length; i ++ ) {

    			var vertex = this.vertices[ i ];
    			vertices.push( vertex.x, vertex.y, vertex.z );

    		}

    		var faces = [];
    		var normals = [];
    		var normalsHash = {};
    		var colors = [];
    		var colorsHash = {};
    		var uvs = [];
    		var uvsHash = {};

    		for ( var i = 0; i < this.faces.length; i ++ ) {

    			var face = this.faces[ i ];

    			var hasMaterial = true;
    			var hasFaceUv = false; // deprecated
    			var hasFaceVertexUv = this.faceVertexUvs[ 0 ][ i ] !== undefined;
    			var hasFaceNormal = face.normal.length() > 0;
    			var hasFaceVertexNormal = face.vertexNormals.length > 0;
    			var hasFaceColor = face.color.r !== 1 || face.color.g !== 1 || face.color.b !== 1;
    			var hasFaceVertexColor = face.vertexColors.length > 0;

    			var faceType = 0;

    			faceType = setBit( faceType, 0, 0 ); // isQuad
    			faceType = setBit( faceType, 1, hasMaterial );
    			faceType = setBit( faceType, 2, hasFaceUv );
    			faceType = setBit( faceType, 3, hasFaceVertexUv );
    			faceType = setBit( faceType, 4, hasFaceNormal );
    			faceType = setBit( faceType, 5, hasFaceVertexNormal );
    			faceType = setBit( faceType, 6, hasFaceColor );
    			faceType = setBit( faceType, 7, hasFaceVertexColor );

    			faces.push( faceType );
    			faces.push( face.a, face.b, face.c );
    			faces.push( face.materialIndex );

    			if ( hasFaceVertexUv ) {

    				var faceVertexUvs = this.faceVertexUvs[ 0 ][ i ];

    				faces.push(
    					getUvIndex( faceVertexUvs[ 0 ] ),
    					getUvIndex( faceVertexUvs[ 1 ] ),
    					getUvIndex( faceVertexUvs[ 2 ] )
    				);

    			}

    			if ( hasFaceNormal ) {

    				faces.push( getNormalIndex( face.normal ) );

    			}

    			if ( hasFaceVertexNormal ) {

    				var vertexNormals = face.vertexNormals;

    				faces.push(
    					getNormalIndex( vertexNormals[ 0 ] ),
    					getNormalIndex( vertexNormals[ 1 ] ),
    					getNormalIndex( vertexNormals[ 2 ] )
    				);

    			}

    			if ( hasFaceColor ) {

    				faces.push( getColorIndex( face.color ) );

    			}

    			if ( hasFaceVertexColor ) {

    				var vertexColors = face.vertexColors;

    				faces.push(
    					getColorIndex( vertexColors[ 0 ] ),
    					getColorIndex( vertexColors[ 1 ] ),
    					getColorIndex( vertexColors[ 2 ] )
    				);

    			}

    		}

    		function setBit( value, position, enabled ) {

    			return enabled ? value | ( 1 << position ) : value & ( ~ ( 1 << position ) );

    		}

    		function getNormalIndex( normal ) {

    			var hash = normal.x.toString() + normal.y.toString() + normal.z.toString();

    			if ( normalsHash[ hash ] !== undefined ) {

    				return normalsHash[ hash ];

    			}

    			normalsHash[ hash ] = normals.length / 3;
    			normals.push( normal.x, normal.y, normal.z );

    			return normalsHash[ hash ];

    		}

    		function getColorIndex( color ) {

    			var hash = color.r.toString() + color.g.toString() + color.b.toString();

    			if ( colorsHash[ hash ] !== undefined ) {

    				return colorsHash[ hash ];

    			}

    			colorsHash[ hash ] = colors.length;
    			colors.push( color.getHex() );

    			return colorsHash[ hash ];

    		}

    		function getUvIndex( uv ) {

    			var hash = uv.x.toString() + uv.y.toString();

    			if ( uvsHash[ hash ] !== undefined ) {

    				return uvsHash[ hash ];

    			}

    			uvsHash[ hash ] = uvs.length / 2;
    			uvs.push( uv.x, uv.y );

    			return uvsHash[ hash ];

    		}

    		data.data = {};

    		data.data.vertices = vertices;
    		data.data.normals = normals;
    		if ( colors.length > 0 ) data.data.colors = colors;
    		if ( uvs.length > 0 ) data.data.uvs = [ uvs ]; // temporal backward compatibility
    		data.data.faces = faces;

    		return data;

    	},

    	clone: function () {

    		/*
    		 // Handle primitives

    		 var parameters = this.parameters;

    		 if ( parameters !== undefined ) {

    		 var values = [];

    		 for ( var key in parameters ) {

    		 values.push( parameters[ key ] );

    		 }

    		 var geometry = Object.create( this.constructor.prototype );
    		 this.constructor.apply( geometry, values );
    		 return geometry;

    		 }

    		 return new this.constructor().copy( this );
    		 */

    		return new Geometry().copy( this );

    	},

    	copy: function ( source ) {

    		var i, il, j, jl, k, kl;

    		// reset

    		this.vertices = [];
    		this.colors = [];
    		this.faces = [];
    		this.faceVertexUvs = [[]];
    		this.morphTargets = [];
    		this.morphNormals = [];
    		this.skinWeights = [];
    		this.skinIndices = [];
    		this.lineDistances = [];
    		this.boundingBox = null;
    		this.boundingSphere = null;

    		// name

    		this.name = source.name;

    		// vertices

    		var vertices = source.vertices;

    		for ( i = 0, il = vertices.length; i < il; i ++ ) {

    			this.vertices.push( vertices[ i ].clone() );

    		}

    		// colors

    		var colors = source.colors;

    		for ( i = 0, il = colors.length; i < il; i ++ ) {

    			this.colors.push( colors[ i ].clone() );

    		}

    		// faces

    		var faces = source.faces;

    		for ( i = 0, il = faces.length; i < il; i ++ ) {

    			this.faces.push( faces[ i ].clone() );

    		}

    		// face vertex uvs

    		for ( i = 0, il = source.faceVertexUvs.length; i < il; i ++ ) {

    			var faceVertexUvs = source.faceVertexUvs[ i ];

    			if ( this.faceVertexUvs[ i ] === undefined ) {

    				this.faceVertexUvs[ i ] = [];

    			}

    			for ( j = 0, jl = faceVertexUvs.length; j < jl; j ++ ) {

    				var uvs = faceVertexUvs[ j ], uvsCopy = [];

    				for ( k = 0, kl = uvs.length; k < kl; k ++ ) {

    					var uv = uvs[ k ];

    					uvsCopy.push( uv.clone() );

    				}

    				this.faceVertexUvs[ i ].push( uvsCopy );

    			}

    		}

    		// morph targets

    		var morphTargets = source.morphTargets;

    		for ( i = 0, il = morphTargets.length; i < il; i ++ ) {

    			var morphTarget = {};
    			morphTarget.name = morphTargets[ i ].name;

    			// vertices

    			if ( morphTargets[ i ].vertices !== undefined ) {

    				morphTarget.vertices = [];

    				for ( j = 0, jl = morphTargets[ i ].vertices.length; j < jl; j ++ ) {

    					morphTarget.vertices.push( morphTargets[ i ].vertices[ j ].clone() );

    				}

    			}

    			// normals

    			if ( morphTargets[ i ].normals !== undefined ) {

    				morphTarget.normals = [];

    				for ( j = 0, jl = morphTargets[ i ].normals.length; j < jl; j ++ ) {

    					morphTarget.normals.push( morphTargets[ i ].normals[ j ].clone() );

    				}

    			}

    			this.morphTargets.push( morphTarget );

    		}

    		// morph normals

    		var morphNormals = source.morphNormals;

    		for ( i = 0, il = morphNormals.length; i < il; i ++ ) {

    			var morphNormal = {};

    			// vertex normals

    			if ( morphNormals[ i ].vertexNormals !== undefined ) {

    				morphNormal.vertexNormals = [];

    				for ( j = 0, jl = morphNormals[ i ].vertexNormals.length; j < jl; j ++ ) {

    					var srcVertexNormal = morphNormals[ i ].vertexNormals[ j ];
    					var destVertexNormal = {};

    					destVertexNormal.a = srcVertexNormal.a.clone();
    					destVertexNormal.b = srcVertexNormal.b.clone();
    					destVertexNormal.c = srcVertexNormal.c.clone();

    					morphNormal.vertexNormals.push( destVertexNormal );

    				}

    			}

    			// face normals

    			if ( morphNormals[ i ].faceNormals !== undefined ) {

    				morphNormal.faceNormals = [];

    				for ( j = 0, jl = morphNormals[ i ].faceNormals.length; j < jl; j ++ ) {

    					morphNormal.faceNormals.push( morphNormals[ i ].faceNormals[ j ].clone() );

    				}

    			}

    			this.morphNormals.push( morphNormal );

    		}

    		// skin weights

    		var skinWeights = source.skinWeights;

    		for ( i = 0, il = skinWeights.length; i < il; i ++ ) {

    			this.skinWeights.push( skinWeights[ i ].clone() );

    		}

    		// skin indices

    		var skinIndices = source.skinIndices;

    		for ( i = 0, il = skinIndices.length; i < il; i ++ ) {

    			this.skinIndices.push( skinIndices[ i ].clone() );

    		}

    		// line distances

    		var lineDistances = source.lineDistances;

    		for ( i = 0, il = lineDistances.length; i < il; i ++ ) {

    			this.lineDistances.push( lineDistances[ i ] );

    		}

    		// bounding box

    		var boundingBox = source.boundingBox;

    		if ( boundingBox !== null ) {

    			this.boundingBox = boundingBox.clone();

    		}

    		// bounding sphere

    		var boundingSphere = source.boundingSphere;

    		if ( boundingSphere !== null ) {

    			this.boundingSphere = boundingSphere.clone();

    		}

    		// update flags

    		this.elementsNeedUpdate = source.elementsNeedUpdate;
    		this.verticesNeedUpdate = source.verticesNeedUpdate;
    		this.uvsNeedUpdate = source.uvsNeedUpdate;
    		this.normalsNeedUpdate = source.normalsNeedUpdate;
    		this.colorsNeedUpdate = source.colorsNeedUpdate;
    		this.lineDistancesNeedUpdate = source.lineDistancesNeedUpdate;
    		this.groupsNeedUpdate = source.groupsNeedUpdate;

    		return this;

    	},

    	dispose: function () {

    		this.dispatchEvent( { type: 'dispose' } );

    	}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     * @author Mugen87 / https://github.com/Mugen87
     */

    // BoxGeometry

    function BoxGeometry( width, height, depth, widthSegments, heightSegments, depthSegments ) {

    	Geometry.call( this );

    	this.type = 'BoxGeometry';

    	this.parameters = {
    		width: width,
    		height: height,
    		depth: depth,
    		widthSegments: widthSegments,
    		heightSegments: heightSegments,
    		depthSegments: depthSegments
    	};

    	this.fromBufferGeometry( new BoxBufferGeometry( width, height, depth, widthSegments, heightSegments, depthSegments ) );
    	this.mergeVertices();

    }

    BoxGeometry.prototype = Object.create( Geometry.prototype );
    BoxGeometry.prototype.constructor = BoxGeometry;

    // BoxBufferGeometry

    function BoxBufferGeometry( width, height, depth, widthSegments, heightSegments, depthSegments ) {

    	BufferGeometry.call( this );

    	this.type = 'BoxBufferGeometry';

    	this.parameters = {
    		width: width,
    		height: height,
    		depth: depth,
    		widthSegments: widthSegments,
    		heightSegments: heightSegments,
    		depthSegments: depthSegments
    	};

    	var scope = this;

    	width = width || 1;
    	height = height || 1;
    	depth = depth || 1;

    	// segments

    	widthSegments = Math.floor( widthSegments ) || 1;
    	heightSegments = Math.floor( heightSegments ) || 1;
    	depthSegments = Math.floor( depthSegments ) || 1;

    	// buffers

    	var indices = [];
    	var vertices = [];
    	var normals = [];
    	var uvs = [];

    	// helper variables

    	var numberOfVertices = 0;
    	var groupStart = 0;

    	// build each side of the box geometry

    	buildPlane( 'z', 'y', 'x', - 1, - 1, depth, height, width, depthSegments, heightSegments, 0 ); // px
    	buildPlane( 'z', 'y', 'x', 1, - 1, depth, height, - width, depthSegments, heightSegments, 1 ); // nx
    	buildPlane( 'x', 'z', 'y', 1, 1, width, depth, height, widthSegments, depthSegments, 2 ); // py
    	buildPlane( 'x', 'z', 'y', 1, - 1, width, depth, - height, widthSegments, depthSegments, 3 ); // ny
    	buildPlane( 'x', 'y', 'z', 1, - 1, width, height, depth, widthSegments, heightSegments, 4 ); // pz
    	buildPlane( 'x', 'y', 'z', - 1, - 1, width, height, - depth, widthSegments, heightSegments, 5 ); // nz

    	// build geometry

    	this.setIndex( indices );
    	this.addAttribute( 'position', new Float32BufferAttribute( vertices, 3 ) );
    	this.addAttribute( 'normal', new Float32BufferAttribute( normals, 3 ) );
    	this.addAttribute( 'uv', new Float32BufferAttribute( uvs, 2 ) );

    	function buildPlane( u, v, w, udir, vdir, width, height, depth, gridX, gridY, materialIndex ) {

    		var segmentWidth = width / gridX;
    		var segmentHeight = height / gridY;

    		var widthHalf = width / 2;
    		var heightHalf = height / 2;
    		var depthHalf = depth / 2;

    		var gridX1 = gridX + 1;
    		var gridY1 = gridY + 1;

    		var vertexCounter = 0;
    		var groupCount = 0;

    		var ix, iy;

    		var vector = new Vector3();

    		// generate vertices, normals and uvs

    		for ( iy = 0; iy < gridY1; iy ++ ) {

    			var y = iy * segmentHeight - heightHalf;

    			for ( ix = 0; ix < gridX1; ix ++ ) {

    				var x = ix * segmentWidth - widthHalf;

    				// set values to correct vector component

    				vector[ u ] = x * udir;
    				vector[ v ] = y * vdir;
    				vector[ w ] = depthHalf;

    				// now apply vector to vertex buffer

    				vertices.push( vector.x, vector.y, vector.z );

    				// set values to correct vector component

    				vector[ u ] = 0;
    				vector[ v ] = 0;
    				vector[ w ] = depth > 0 ? 1 : - 1;

    				// now apply vector to normal buffer

    				normals.push( vector.x, vector.y, vector.z );

    				// uvs

    				uvs.push( ix / gridX );
    				uvs.push( 1 - ( iy / gridY ) );

    				// counters

    				vertexCounter += 1;

    			}

    		}

    		// indices

    		// 1. you need three indices to draw a single face
    		// 2. a single segment consists of two faces
    		// 3. so we need to generate six (2*3) indices per segment

    		for ( iy = 0; iy < gridY; iy ++ ) {

    			for ( ix = 0; ix < gridX; ix ++ ) {

    				var a = numberOfVertices + ix + gridX1 * iy;
    				var b = numberOfVertices + ix + gridX1 * ( iy + 1 );
    				var c = numberOfVertices + ( ix + 1 ) + gridX1 * ( iy + 1 );
    				var d = numberOfVertices + ( ix + 1 ) + gridX1 * iy;

    				// faces

    				indices.push( a, b, d );
    				indices.push( b, c, d );

    				// increase counter

    				groupCount += 6;

    			}

    		}

    		// add a group to the geometry. this will ensure multi material support

    		scope.addGroup( groupStart, groupCount, materialIndex );

    		// calculate new start value for groups

    		groupStart += groupCount;

    		// update total number of vertices

    		numberOfVertices += vertexCounter;

    	}

    }

    BoxBufferGeometry.prototype = Object.create( BufferGeometry.prototype );
    BoxBufferGeometry.prototype.constructor = BoxBufferGeometry;

    /**
     * Uniform Utilities
     */

    function cloneUniforms( src ) {

    	var dst = {};

    	for ( var u in src ) {

    		dst[ u ] = {};

    		for ( var p in src[ u ] ) {

    			var property = src[ u ][ p ];

    			if ( property && ( property.isColor ||
    				property.isMatrix3 || property.isMatrix4 ||
    				property.isVector2 || property.isVector3 || property.isVector4 ||
    				property.isTexture ) ) {

    				dst[ u ][ p ] = property.clone();

    			} else if ( Array.isArray( property ) ) {

    				dst[ u ][ p ] = property.slice();

    			} else {

    				dst[ u ][ p ] = property;

    			}

    		}

    	}

    	return dst;

    }

    function mergeUniforms( uniforms ) {

    	var merged = {};

    	for ( var u = 0; u < uniforms.length; u ++ ) {

    		var tmp = cloneUniforms( uniforms[ u ] );

    		for ( var p in tmp ) {

    			merged[ p ] = tmp[ p ];

    		}

    	}

    	return merged;

    }

    // Legacy

    var UniformsUtils = { clone: cloneUniforms, merge: mergeUniforms };

    var default_vertex = "void main() {\n\tgl_Position = projectionMatrix * modelViewMatrix * vec4( position, 1.0 );\n}";

    var default_fragment = "void main() {\n\tgl_FragColor = vec4( 1.0, 0.0, 0.0, 1.0 );\n}";

    /**
     * @author alteredq / http://alteredqualia.com/
     *
     * parameters = {
     *  defines: { "label" : "value" },
     *  uniforms: { "parameter1": { value: 1.0 }, "parameter2": { value2: 2 } },
     *
     *  fragmentShader: <string>,
     *  vertexShader: <string>,
     *
     *  wireframe: <boolean>,
     *  wireframeLinewidth: <float>,
     *
     *  lights: <bool>,
     *
     *  skinning: <bool>,
     *  morphTargets: <bool>,
     *  morphNormals: <bool>
     * }
     */

    function ShaderMaterial( parameters ) {

    	Material.call( this );

    	this.type = 'ShaderMaterial';

    	this.defines = {};
    	this.uniforms = {};

    	this.vertexShader = default_vertex;
    	this.fragmentShader = default_fragment;

    	this.linewidth = 1;

    	this.wireframe = false;
    	this.wireframeLinewidth = 1;

    	this.fog = false; // set to use scene fog
    	this.lights = false; // set to use scene lights
    	this.clipping = false; // set to use user-defined clipping planes

    	this.skinning = false; // set to use skinning attribute streams
    	this.morphTargets = false; // set to use morph targets
    	this.morphNormals = false; // set to use morph normals

    	this.extensions = {
    		derivatives: false, // set to use derivatives
    		fragDepth: false, // set to use fragment depth values
    		drawBuffers: false, // set to use draw buffers
    		shaderTextureLOD: false // set to use shader texture LOD
    	};

    	// When rendered geometry doesn't include these attributes but the material does,
    	// use these default values in WebGL. This avoids errors when buffer data is missing.
    	this.defaultAttributeValues = {
    		'color': [ 1, 1, 1 ],
    		'uv': [ 0, 0 ],
    		'uv2': [ 0, 0 ]
    	};

    	this.index0AttributeName = undefined;
    	this.uniformsNeedUpdate = false;

    	if ( parameters !== undefined ) {

    		if ( parameters.attributes !== undefined ) {

    			console.error( 'THREE.ShaderMaterial: attributes should now be defined in THREE.BufferGeometry instead.' );

    		}

    		this.setValues( parameters );

    	}

    }

    ShaderMaterial.prototype = Object.create( Material.prototype );
    ShaderMaterial.prototype.constructor = ShaderMaterial;

    ShaderMaterial.prototype.isShaderMaterial = true;

    ShaderMaterial.prototype.copy = function ( source ) {

    	Material.prototype.copy.call( this, source );

    	this.fragmentShader = source.fragmentShader;
    	this.vertexShader = source.vertexShader;

    	this.uniforms = cloneUniforms( source.uniforms );

    	this.defines = Object.assign( {}, source.defines );

    	this.wireframe = source.wireframe;
    	this.wireframeLinewidth = source.wireframeLinewidth;

    	this.lights = source.lights;
    	this.clipping = source.clipping;

    	this.skinning = source.skinning;

    	this.morphTargets = source.morphTargets;
    	this.morphNormals = source.morphNormals;

    	this.extensions = source.extensions;

    	return this;

    };

    ShaderMaterial.prototype.toJSON = function ( meta ) {

    	var data = Material.prototype.toJSON.call( this, meta );

    	data.uniforms = {};

    	for ( var name in this.uniforms ) {

    		var uniform = this.uniforms[ name ];
    		var value = uniform.value;

    		if ( value && value.isTexture ) {

    			data.uniforms[ name ] = {
    				type: 't',
    				value: value.toJSON( meta ).uuid
    			};

    		} else if ( value && value.isColor ) {

    			data.uniforms[ name ] = {
    				type: 'c',
    				value: value.getHex()
    			};

    		} else if ( value && value.isVector2 ) {

    			data.uniforms[ name ] = {
    				type: 'v2',
    				value: value.toArray()
    			};

    		} else if ( value && value.isVector3 ) {

    			data.uniforms[ name ] = {
    				type: 'v3',
    				value: value.toArray()
    			};

    		} else if ( value && value.isVector4 ) {

    			data.uniforms[ name ] = {
    				type: 'v4',
    				value: value.toArray()
    			};

    		} else if ( value && value.isMatrix3 ) {

    			data.uniforms[ name ] = {
    				type: 'm3',
    				value: value.toArray()
    			};

    		} else if ( value && value.isMatrix4 ) {

    			data.uniforms[ name ] = {
    				type: 'm4',
    				value: value.toArray()
    			};

    		} else {

    			data.uniforms[ name ] = {
    				value: value
    			};

    			// note: the array variants v2v, v3v, v4v, m4v and tv are not supported so far

    		}

    	}

    	if ( Object.keys( this.defines ).length > 0 ) data.defines = this.defines;

    	data.vertexShader = this.vertexShader;
    	data.fragmentShader = this.fragmentShader;

    	var extensions = {};

    	for ( var key in this.extensions ) {

    		if ( this.extensions[ key ] === true ) extensions[ key ] = true;

    	}

    	if ( Object.keys( extensions ).length > 0 ) data.extensions = extensions;

    	return data;

    };

    /**
     * @author mrdoob / http://mrdoob.com/
     * @author mikael emtinger / http://gomo.se/
     * @author WestLangley / http://github.com/WestLangley
    */

    function Camera() {

    	Object3D.call( this );

    	this.type = 'Camera';

    	this.matrixWorldInverse = new Matrix4();

    	this.projectionMatrix = new Matrix4();
    	this.projectionMatrixInverse = new Matrix4();

    }

    Camera.prototype = Object.assign( Object.create( Object3D.prototype ), {

    	constructor: Camera,

    	isCamera: true,

    	copy: function ( source, recursive ) {

    		Object3D.prototype.copy.call( this, source, recursive );

    		this.matrixWorldInverse.copy( source.matrixWorldInverse );

    		this.projectionMatrix.copy( source.projectionMatrix );
    		this.projectionMatrixInverse.copy( source.projectionMatrixInverse );

    		return this;

    	},

    	getWorldDirection: function ( target ) {

    		if ( target === undefined ) {

    			console.warn( 'THREE.Camera: .getWorldDirection() target is now required' );
    			target = new Vector3();

    		}

    		this.updateMatrixWorld( true );

    		var e = this.matrixWorld.elements;

    		return target.set( - e[ 8 ], - e[ 9 ], - e[ 10 ] ).normalize();

    	},

    	updateMatrixWorld: function ( force ) {

    		Object3D.prototype.updateMatrixWorld.call( this, force );

    		this.matrixWorldInverse.getInverse( this.matrixWorld );

    	},

    	clone: function () {

    		return new this.constructor().copy( this );

    	}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     * @author greggman / http://games.greggman.com/
     * @author zz85 / http://www.lab4games.net/zz85/blog
     * @author tschw
     */

    function PerspectiveCamera( fov, aspect, near, far ) {

    	Camera.call( this );

    	this.type = 'PerspectiveCamera';

    	this.fov = fov !== undefined ? fov : 50;
    	this.zoom = 1;

    	this.near = near !== undefined ? near : 0.1;
    	this.far = far !== undefined ? far : 2000;
    	this.focus = 10;

    	this.aspect = aspect !== undefined ? aspect : 1;
    	this.view = null;

    	this.filmGauge = 35;	// width of the film (default in millimeters)
    	this.filmOffset = 0;	// horizontal film offset (same unit as gauge)

    	this.updateProjectionMatrix();

    }

    PerspectiveCamera.prototype = Object.assign( Object.create( Camera.prototype ), {

    	constructor: PerspectiveCamera,

    	isPerspectiveCamera: true,

    	copy: function ( source, recursive ) {

    		Camera.prototype.copy.call( this, source, recursive );

    		this.fov = source.fov;
    		this.zoom = source.zoom;

    		this.near = source.near;
    		this.far = source.far;
    		this.focus = source.focus;

    		this.aspect = source.aspect;
    		this.view = source.view === null ? null : Object.assign( {}, source.view );

    		this.filmGauge = source.filmGauge;
    		this.filmOffset = source.filmOffset;

    		return this;

    	},

    	/**
    	 * Sets the FOV by focal length in respect to the current .filmGauge.
    	 *
    	 * The default film gauge is 35, so that the focal length can be specified for
    	 * a 35mm (full frame) camera.
    	 *
    	 * Values for focal length and film gauge must have the same unit.
    	 */
    	setFocalLength: function ( focalLength ) {

    		// see http://www.bobatkins.com/photography/technical/field_of_view.html
    		var vExtentSlope = 0.5 * this.getFilmHeight() / focalLength;

    		this.fov = _Math.RAD2DEG * 2 * Math.atan( vExtentSlope );
    		this.updateProjectionMatrix();

    	},

    	/**
    	 * Calculates the focal length from the current .fov and .filmGauge.
    	 */
    	getFocalLength: function () {

    		var vExtentSlope = Math.tan( _Math.DEG2RAD * 0.5 * this.fov );

    		return 0.5 * this.getFilmHeight() / vExtentSlope;

    	},

    	getEffectiveFOV: function () {

    		return _Math.RAD2DEG * 2 * Math.atan(
    			Math.tan( _Math.DEG2RAD * 0.5 * this.fov ) / this.zoom );

    	},

    	getFilmWidth: function () {

    		// film not completely covered in portrait format (aspect < 1)
    		return this.filmGauge * Math.min( this.aspect, 1 );

    	},

    	getFilmHeight: function () {

    		// film not completely covered in landscape format (aspect > 1)
    		return this.filmGauge / Math.max( this.aspect, 1 );

    	},

    	/**
    	 * Sets an offset in a larger frustum. This is useful for multi-window or
    	 * multi-monitor/multi-machine setups.
    	 *
    	 * For example, if you have 3x2 monitors and each monitor is 1920x1080 and
    	 * the monitors are in grid like this
    	 *
    	 *   +---+---+---+
    	 *   | A | B | C |
    	 *   +---+---+---+
    	 *   | D | E | F |
    	 *   +---+---+---+
    	 *
    	 * then for each monitor you would call it like this
    	 *
    	 *   var w = 1920;
    	 *   var h = 1080;
    	 *   var fullWidth = w * 3;
    	 *   var fullHeight = h * 2;
    	 *
    	 *   --A--
    	 *   camera.setViewOffset( fullWidth, fullHeight, w * 0, h * 0, w, h );
    	 *   --B--
    	 *   camera.setViewOffset( fullWidth, fullHeight, w * 1, h * 0, w, h );
    	 *   --C--
    	 *   camera.setViewOffset( fullWidth, fullHeight, w * 2, h * 0, w, h );
    	 *   --D--
    	 *   camera.setViewOffset( fullWidth, fullHeight, w * 0, h * 1, w, h );
    	 *   --E--
    	 *   camera.setViewOffset( fullWidth, fullHeight, w * 1, h * 1, w, h );
    	 *   --F--
    	 *   camera.setViewOffset( fullWidth, fullHeight, w * 2, h * 1, w, h );
    	 *
    	 *   Note there is no reason monitors have to be the same size or in a grid.
    	 */
    	setViewOffset: function ( fullWidth, fullHeight, x, y, width, height ) {

    		this.aspect = fullWidth / fullHeight;

    		if ( this.view === null ) {

    			this.view = {
    				enabled: true,
    				fullWidth: 1,
    				fullHeight: 1,
    				offsetX: 0,
    				offsetY: 0,
    				width: 1,
    				height: 1
    			};

    		}

    		this.view.enabled = true;
    		this.view.fullWidth = fullWidth;
    		this.view.fullHeight = fullHeight;
    		this.view.offsetX = x;
    		this.view.offsetY = y;
    		this.view.width = width;
    		this.view.height = height;

    		this.updateProjectionMatrix();

    	},

    	clearViewOffset: function () {

    		if ( this.view !== null ) {

    			this.view.enabled = false;

    		}

    		this.updateProjectionMatrix();

    	},

    	updateProjectionMatrix: function () {

    		var near = this.near,
    			top = near * Math.tan( _Math.DEG2RAD * 0.5 * this.fov ) / this.zoom,
    			height = 2 * top,
    			width = this.aspect * height,
    			left = - 0.5 * width,
    			view = this.view;

    		if ( this.view !== null && this.view.enabled ) {

    			var fullWidth = view.fullWidth,
    				fullHeight = view.fullHeight;

    			left += view.offsetX * width / fullWidth;
    			top -= view.offsetY * height / fullHeight;
    			width *= view.width / fullWidth;
    			height *= view.height / fullHeight;

    		}

    		var skew = this.filmOffset;
    		if ( skew !== 0 ) left += near * skew / this.getFilmWidth();

    		this.projectionMatrix.makePerspective( left, left + width, top, top - height, near, this.far );

    		this.projectionMatrixInverse.getInverse( this.projectionMatrix );

    	},

    	toJSON: function ( meta ) {

    		var data = Object3D.prototype.toJSON.call( this, meta );

    		data.object.fov = this.fov;
    		data.object.zoom = this.zoom;

    		data.object.near = this.near;
    		data.object.far = this.far;
    		data.object.focus = this.focus;

    		data.object.aspect = this.aspect;

    		if ( this.view !== null ) data.object.view = Object.assign( {}, this.view );

    		data.object.filmGauge = this.filmGauge;
    		data.object.filmOffset = this.filmOffset;

    		return data;

    	}

    } );

    /**
     * Camera for rendering cube maps
     *	- renders scene into axis-aligned cube
     *
     * @author alteredq / http://alteredqualia.com/
     */

    var fov = 90, aspect = 1;

    function CubeCamera( near, far, cubeResolution, options ) {

    	Object3D.call( this );

    	this.type = 'CubeCamera';

    	var cameraPX = new PerspectiveCamera( fov, aspect, near, far );
    	cameraPX.up.set( 0, - 1, 0 );
    	cameraPX.lookAt( new Vector3( 1, 0, 0 ) );
    	this.add( cameraPX );

    	var cameraNX = new PerspectiveCamera( fov, aspect, near, far );
    	cameraNX.up.set( 0, - 1, 0 );
    	cameraNX.lookAt( new Vector3( - 1, 0, 0 ) );
    	this.add( cameraNX );

    	var cameraPY = new PerspectiveCamera( fov, aspect, near, far );
    	cameraPY.up.set( 0, 0, 1 );
    	cameraPY.lookAt( new Vector3( 0, 1, 0 ) );
    	this.add( cameraPY );

    	var cameraNY = new PerspectiveCamera( fov, aspect, near, far );
    	cameraNY.up.set( 0, 0, - 1 );
    	cameraNY.lookAt( new Vector3( 0, - 1, 0 ) );
    	this.add( cameraNY );

    	var cameraPZ = new PerspectiveCamera( fov, aspect, near, far );
    	cameraPZ.up.set( 0, - 1, 0 );
    	cameraPZ.lookAt( new Vector3( 0, 0, 1 ) );
    	this.add( cameraPZ );

    	var cameraNZ = new PerspectiveCamera( fov, aspect, near, far );
    	cameraNZ.up.set( 0, - 1, 0 );
    	cameraNZ.lookAt( new Vector3( 0, 0, - 1 ) );
    	this.add( cameraNZ );

    	options = options || { format: RGBFormat, magFilter: LinearFilter, minFilter: LinearFilter };

    	this.renderTarget = new WebGLRenderTargetCube( cubeResolution, cubeResolution, options );
    	this.renderTarget.texture.name = "CubeCamera";

    	this.update = function ( renderer, scene ) {

    		if ( this.parent === null ) this.updateMatrixWorld();

    		var currentRenderTarget = renderer.getRenderTarget();

    		var renderTarget = this.renderTarget;
    		var generateMipmaps = renderTarget.texture.generateMipmaps;

    		renderTarget.texture.generateMipmaps = false;

    		renderer.setRenderTarget( renderTarget, 0 );
    		renderer.render( scene, cameraPX );

    		renderer.setRenderTarget( renderTarget, 1 );
    		renderer.render( scene, cameraNX );

    		renderer.setRenderTarget( renderTarget, 2 );
    		renderer.render( scene, cameraPY );

    		renderer.setRenderTarget( renderTarget, 3 );
    		renderer.render( scene, cameraNY );

    		renderer.setRenderTarget( renderTarget, 4 );
    		renderer.render( scene, cameraPZ );

    		renderTarget.texture.generateMipmaps = generateMipmaps;

    		renderer.setRenderTarget( renderTarget, 5 );
    		renderer.render( scene, cameraNZ );

    		renderer.setRenderTarget( currentRenderTarget );

    	};

    	this.clear = function ( renderer, color, depth, stencil ) {

    		var currentRenderTarget = renderer.getRenderTarget();

    		var renderTarget = this.renderTarget;

    		for ( var i = 0; i < 6; i ++ ) {

    			renderer.setRenderTarget( renderTarget, i );

    			renderer.clear( color, depth, stencil );

    		}

    		renderer.setRenderTarget( currentRenderTarget );

    	};

    }

    CubeCamera.prototype = Object.create( Object3D.prototype );
    CubeCamera.prototype.constructor = CubeCamera;

    /**
     * @author alteredq / http://alteredqualia.com
     * @author WestLangley / http://github.com/WestLangley
     */

    function WebGLRenderTargetCube( width, height, options ) {

    	WebGLRenderTarget.call( this, width, height, options );

    }

    WebGLRenderTargetCube.prototype = Object.create( WebGLRenderTarget.prototype );
    WebGLRenderTargetCube.prototype.constructor = WebGLRenderTargetCube;

    WebGLRenderTargetCube.prototype.isWebGLRenderTargetCube = true;

    WebGLRenderTargetCube.prototype.fromEquirectangularTexture = function ( renderer, texture ) {

    	this.texture.type = texture.type;
    	this.texture.format = texture.format;
    	this.texture.encoding = texture.encoding;

    	var scene = new Scene();

    	var shader = {

    		uniforms: {
    			tEquirect: { value: null },
    		},

    		vertexShader: [

    			"varying vec3 vWorldDirection;",

    			"vec3 transformDirection( in vec3 dir, in mat4 matrix ) {",

    			"	return normalize( ( matrix * vec4( dir, 0.0 ) ).xyz );",

    			"}",

    			"void main() {",

    			"	vWorldDirection = transformDirection( position, modelMatrix );",

    			"	#include <begin_vertex>",
    			"	#include <project_vertex>",

    			"}"

    		].join( '\n' ),

    		fragmentShader: [

    			"uniform sampler2D tEquirect;",

    			"varying vec3 vWorldDirection;",

    			"#define RECIPROCAL_PI 0.31830988618",
    			"#define RECIPROCAL_PI2 0.15915494",

    			"void main() {",

    			"	vec3 direction = normalize( vWorldDirection );",

    			"	vec2 sampleUV;",

    			"	sampleUV.y = asin( clamp( direction.y, - 1.0, 1.0 ) ) * RECIPROCAL_PI + 0.5;",

    			"	sampleUV.x = atan( direction.z, direction.x ) * RECIPROCAL_PI2 + 0.5;",

    			"	gl_FragColor = texture2D( tEquirect, sampleUV );",

    			"}"

    		].join( '\n' ),
    	};

    	var material = new ShaderMaterial( {

    		type: 'CubemapFromEquirect',

    		uniforms: cloneUniforms( shader.uniforms ),
    		vertexShader: shader.vertexShader,
    		fragmentShader: shader.fragmentShader,
    		side: BackSide,
    		blending: NoBlending

    	} );

    	material.uniforms.tEquirect.value = texture;

    	var mesh = new Mesh( new BoxBufferGeometry( 5, 5, 5 ), material );

    	scene.add( mesh );

    	var camera = new CubeCamera( 1, 10, 1 );

    	camera.renderTarget = this;
    	camera.renderTarget.texture.name = 'CubeCameraTexture';

    	camera.update( renderer, scene );

    	mesh.geometry.dispose();
    	mesh.material.dispose();

    	return this;

    };

    /**
     * @author alteredq / http://alteredqualia.com/
     */

    function DataTexture( data, width, height, format, type, mapping, wrapS, wrapT, magFilter, minFilter, anisotropy, encoding ) {

    	Texture.call( this, null, mapping, wrapS, wrapT, magFilter, minFilter, format, type, anisotropy, encoding );

    	this.image = { data: data, width: width, height: height };

    	this.magFilter = magFilter !== undefined ? magFilter : NearestFilter;
    	this.minFilter = minFilter !== undefined ? minFilter : NearestFilter;

    	this.generateMipmaps = false;
    	this.flipY = false;
    	this.unpackAlignment = 1;

    }

    DataTexture.prototype = Object.create( Texture.prototype );
    DataTexture.prototype.constructor = DataTexture;

    DataTexture.prototype.isDataTexture = true;

    /**
     * @author bhouston / http://clara.io
     */

    var _vector1 = new Vector3();
    var _vector2 = new Vector3();
    var _normalMatrix = new Matrix3();

    function Plane( normal, constant ) {

    	// normal is assumed to be normalized

    	this.normal = ( normal !== undefined ) ? normal : new Vector3( 1, 0, 0 );
    	this.constant = ( constant !== undefined ) ? constant : 0;

    }

    Object.assign( Plane.prototype, {

    	isPlane: true,

    	set: function ( normal, constant ) {

    		this.normal.copy( normal );
    		this.constant = constant;

    		return this;

    	},

    	setComponents: function ( x, y, z, w ) {

    		this.normal.set( x, y, z );
    		this.constant = w;

    		return this;

    	},

    	setFromNormalAndCoplanarPoint: function ( normal, point ) {

    		this.normal.copy( normal );
    		this.constant = - point.dot( this.normal );

    		return this;

    	},

    	setFromCoplanarPoints: function ( a, b, c ) {

    		var normal = _vector1.subVectors( c, b ).cross( _vector2.subVectors( a, b ) ).normalize();

    		// Q: should an error be thrown if normal is zero (e.g. degenerate plane)?

    		this.setFromNormalAndCoplanarPoint( normal, a );

    		return this;

    	},

    	clone: function () {

    		return new this.constructor().copy( this );

    	},

    	copy: function ( plane ) {

    		this.normal.copy( plane.normal );
    		this.constant = plane.constant;

    		return this;

    	},

    	normalize: function () {

    		// Note: will lead to a divide by zero if the plane is invalid.

    		var inverseNormalLength = 1.0 / this.normal.length();
    		this.normal.multiplyScalar( inverseNormalLength );
    		this.constant *= inverseNormalLength;

    		return this;

    	},

    	negate: function () {

    		this.constant *= - 1;
    		this.normal.negate();

    		return this;

    	},

    	distanceToPoint: function ( point ) {

    		return this.normal.dot( point ) + this.constant;

    	},

    	distanceToSphere: function ( sphere ) {

    		return this.distanceToPoint( sphere.center ) - sphere.radius;

    	},

    	projectPoint: function ( point, target ) {

    		if ( target === undefined ) {

    			console.warn( 'THREE.Plane: .projectPoint() target is now required' );
    			target = new Vector3();

    		}

    		return target.copy( this.normal ).multiplyScalar( - this.distanceToPoint( point ) ).add( point );

    	},

    	intersectLine: function ( line, target ) {

    		if ( target === undefined ) {

    			console.warn( 'THREE.Plane: .intersectLine() target is now required' );
    			target = new Vector3();

    		}

    		var direction = line.delta( _vector1 );

    		var denominator = this.normal.dot( direction );

    		if ( denominator === 0 ) {

    			// line is coplanar, return origin
    			if ( this.distanceToPoint( line.start ) === 0 ) {

    				return target.copy( line.start );

    			}

    			// Unsure if this is the correct method to handle this case.
    			return undefined;

    		}

    		var t = - ( line.start.dot( this.normal ) + this.constant ) / denominator;

    		if ( t < 0 || t > 1 ) {

    			return undefined;

    		}

    		return target.copy( direction ).multiplyScalar( t ).add( line.start );

    	},

    	intersectsLine: function ( line ) {

    		// Note: this tests if a line intersects the plane, not whether it (or its end-points) are coplanar with it.

    		var startSign = this.distanceToPoint( line.start );
    		var endSign = this.distanceToPoint( line.end );

    		return ( startSign < 0 && endSign > 0 ) || ( endSign < 0 && startSign > 0 );

    	},

    	intersectsBox: function ( box ) {

    		return box.intersectsPlane( this );

    	},

    	intersectsSphere: function ( sphere ) {

    		return sphere.intersectsPlane( this );

    	},

    	coplanarPoint: function ( target ) {

    		if ( target === undefined ) {

    			console.warn( 'THREE.Plane: .coplanarPoint() target is now required' );
    			target = new Vector3();

    		}

    		return target.copy( this.normal ).multiplyScalar( - this.constant );

    	},

    	applyMatrix4: function ( matrix, optionalNormalMatrix ) {

    		var normalMatrix = optionalNormalMatrix || _normalMatrix.getNormalMatrix( matrix );

    		var referencePoint = this.coplanarPoint( _vector1 ).applyMatrix4( matrix );

    		var normal = this.normal.applyMatrix3( normalMatrix ).normalize();

    		this.constant = - referencePoint.dot( normal );

    		return this;

    	},

    	translate: function ( offset ) {

    		this.constant -= offset.dot( this.normal );

    		return this;

    	},

    	equals: function ( plane ) {

    		return plane.normal.equals( this.normal ) && ( plane.constant === this.constant );

    	}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     * @author alteredq / http://alteredqualia.com/
     * @author bhouston / http://clara.io
     */

    var _sphere$1 = new Sphere();
    var _vector$5 = new Vector3();

    function Frustum( p0, p1, p2, p3, p4, p5 ) {

    	this.planes = [

    		( p0 !== undefined ) ? p0 : new Plane(),
    		( p1 !== undefined ) ? p1 : new Plane(),
    		( p2 !== undefined ) ? p2 : new Plane(),
    		( p3 !== undefined ) ? p3 : new Plane(),
    		( p4 !== undefined ) ? p4 : new Plane(),
    		( p5 !== undefined ) ? p5 : new Plane()

    	];

    }

    Object.assign( Frustum.prototype, {

    	set: function ( p0, p1, p2, p3, p4, p5 ) {

    		var planes = this.planes;

    		planes[ 0 ].copy( p0 );
    		planes[ 1 ].copy( p1 );
    		planes[ 2 ].copy( p2 );
    		planes[ 3 ].copy( p3 );
    		planes[ 4 ].copy( p4 );
    		planes[ 5 ].copy( p5 );

    		return this;

    	},

    	clone: function () {

    		return new this.constructor().copy( this );

    	},

    	copy: function ( frustum ) {

    		var planes = this.planes;

    		for ( var i = 0; i < 6; i ++ ) {

    			planes[ i ].copy( frustum.planes[ i ] );

    		}

    		return this;

    	},

    	setFromMatrix: function ( m ) {

    		var planes = this.planes;
    		var me = m.elements;
    		var me0 = me[ 0 ], me1 = me[ 1 ], me2 = me[ 2 ], me3 = me[ 3 ];
    		var me4 = me[ 4 ], me5 = me[ 5 ], me6 = me[ 6 ], me7 = me[ 7 ];
    		var me8 = me[ 8 ], me9 = me[ 9 ], me10 = me[ 10 ], me11 = me[ 11 ];
    		var me12 = me[ 12 ], me13 = me[ 13 ], me14 = me[ 14 ], me15 = me[ 15 ];

    		planes[ 0 ].setComponents( me3 - me0, me7 - me4, me11 - me8, me15 - me12 ).normalize();
    		planes[ 1 ].setComponents( me3 + me0, me7 + me4, me11 + me8, me15 + me12 ).normalize();
    		planes[ 2 ].setComponents( me3 + me1, me7 + me5, me11 + me9, me15 + me13 ).normalize();
    		planes[ 3 ].setComponents( me3 - me1, me7 - me5, me11 - me9, me15 - me13 ).normalize();
    		planes[ 4 ].setComponents( me3 - me2, me7 - me6, me11 - me10, me15 - me14 ).normalize();
    		planes[ 5 ].setComponents( me3 + me2, me7 + me6, me11 + me10, me15 + me14 ).normalize();

    		return this;

    	},

    	intersectsObject: function ( object ) {

    		var geometry = object.geometry;

    		if ( geometry.boundingSphere === null ) geometry.computeBoundingSphere();

    		_sphere$1.copy( geometry.boundingSphere ).applyMatrix4( object.matrixWorld );

    		return this.intersectsSphere( _sphere$1 );

    	},

    	intersectsSprite: function ( sprite ) {

    		_sphere$1.center.set( 0, 0, 0 );
    		_sphere$1.radius = 0.7071067811865476;
    		_sphere$1.applyMatrix4( sprite.matrixWorld );

    		return this.intersectsSphere( _sphere$1 );

    	},

    	intersectsSphere: function ( sphere ) {

    		var planes = this.planes;
    		var center = sphere.center;
    		var negRadius = - sphere.radius;

    		for ( var i = 0; i < 6; i ++ ) {

    			var distance = planes[ i ].distanceToPoint( center );

    			if ( distance < negRadius ) {

    				return false;

    			}

    		}

    		return true;

    	},

    	intersectsBox: function ( box ) {

    		var planes = this.planes;

    		for ( var i = 0; i < 6; i ++ ) {

    			var plane = planes[ i ];

    			// corner at max distance

    			_vector$5.x = plane.normal.x > 0 ? box.max.x : box.min.x;
    			_vector$5.y = plane.normal.y > 0 ? box.max.y : box.min.y;
    			_vector$5.z = plane.normal.z > 0 ? box.max.z : box.min.z;

    			if ( plane.distanceToPoint( _vector$5 ) < 0 ) {

    				return false;

    			}

    		}

    		return true;

    	},

    	containsPoint: function ( point ) {

    		var planes = this.planes;

    		for ( var i = 0; i < 6; i ++ ) {

    			if ( planes[ i ].distanceToPoint( point ) < 0 ) {

    				return false;

    			}

    		}

    		return true;

    	}

    } );

    var alphamap_fragment = "#ifdef USE_ALPHAMAP\n\tdiffuseColor.a *= texture2D( alphaMap, vUv ).g;\n#endif";

    var alphamap_pars_fragment = "#ifdef USE_ALPHAMAP\n\tuniform sampler2D alphaMap;\n#endif";

    var alphatest_fragment = "#ifdef ALPHATEST\n\tif ( diffuseColor.a < ALPHATEST ) discard;\n#endif";

    var aomap_fragment = "#ifdef USE_AOMAP\n\tfloat ambientOcclusion = ( texture2D( aoMap, vUv2 ).r - 1.0 ) * aoMapIntensity + 1.0;\n\treflectedLight.indirectDiffuse *= ambientOcclusion;\n\t#if defined( USE_ENVMAP ) && defined( STANDARD )\n\t\tfloat dotNV = saturate( dot( geometry.normal, geometry.viewDir ) );\n\t\treflectedLight.indirectSpecular *= computeSpecularOcclusion( dotNV, ambientOcclusion, material.specularRoughness );\n\t#endif\n#endif";

    var aomap_pars_fragment = "#ifdef USE_AOMAP\n\tuniform sampler2D aoMap;\n\tuniform float aoMapIntensity;\n#endif";

    var begin_vertex = "vec3 transformed = vec3( position );";

    var beginnormal_vertex = "vec3 objectNormal = vec3( normal );\n#ifdef USE_TANGENT\n\tvec3 objectTangent = vec3( tangent.xyz );\n#endif";

    var bsdfs = "vec2 integrateSpecularBRDF( const in float dotNV, const in float roughness ) {\n\tconst vec4 c0 = vec4( - 1, - 0.0275, - 0.572, 0.022 );\n\tconst vec4 c1 = vec4( 1, 0.0425, 1.04, - 0.04 );\n\tvec4 r = roughness * c0 + c1;\n\tfloat a004 = min( r.x * r.x, exp2( - 9.28 * dotNV ) ) * r.x + r.y;\n\treturn vec2( -1.04, 1.04 ) * a004 + r.zw;\n}\nfloat punctualLightIntensityToIrradianceFactor( const in float lightDistance, const in float cutoffDistance, const in float decayExponent ) {\n#if defined ( PHYSICALLY_CORRECT_LIGHTS )\n\tfloat distanceFalloff = 1.0 / max( pow( lightDistance, decayExponent ), 0.01 );\n\tif( cutoffDistance > 0.0 ) {\n\t\tdistanceFalloff *= pow2( saturate( 1.0 - pow4( lightDistance / cutoffDistance ) ) );\n\t}\n\treturn distanceFalloff;\n#else\n\tif( cutoffDistance > 0.0 && decayExponent > 0.0 ) {\n\t\treturn pow( saturate( -lightDistance / cutoffDistance + 1.0 ), decayExponent );\n\t}\n\treturn 1.0;\n#endif\n}\nvec3 BRDF_Diffuse_Lambert( const in vec3 diffuseColor ) {\n\treturn RECIPROCAL_PI * diffuseColor;\n}\nvec3 F_Schlick( const in vec3 specularColor, const in float dotLH ) {\n\tfloat fresnel = exp2( ( -5.55473 * dotLH - 6.98316 ) * dotLH );\n\treturn ( 1.0 - specularColor ) * fresnel + specularColor;\n}\nvec3 F_Schlick_RoughnessDependent( const in vec3 F0, const in float dotNV, const in float roughness ) {\n\tfloat fresnel = exp2( ( -5.55473 * dotNV - 6.98316 ) * dotNV );\n\tvec3 Fr = max( vec3( 1.0 - roughness ), F0 ) - F0;\n\treturn Fr * fresnel + F0;\n}\nfloat G_GGX_Smith( const in float alpha, const in float dotNL, const in float dotNV ) {\n\tfloat a2 = pow2( alpha );\n\tfloat gl = dotNL + sqrt( a2 + ( 1.0 - a2 ) * pow2( dotNL ) );\n\tfloat gv = dotNV + sqrt( a2 + ( 1.0 - a2 ) * pow2( dotNV ) );\n\treturn 1.0 / ( gl * gv );\n}\nfloat G_GGX_SmithCorrelated( const in float alpha, const in float dotNL, const in float dotNV ) {\n\tfloat a2 = pow2( alpha );\n\tfloat gv = dotNL * sqrt( a2 + ( 1.0 - a2 ) * pow2( dotNV ) );\n\tfloat gl = dotNV * sqrt( a2 + ( 1.0 - a2 ) * pow2( dotNL ) );\n\treturn 0.5 / max( gv + gl, EPSILON );\n}\nfloat D_GGX( const in float alpha, const in float dotNH ) {\n\tfloat a2 = pow2( alpha );\n\tfloat denom = pow2( dotNH ) * ( a2 - 1.0 ) + 1.0;\n\treturn RECIPROCAL_PI * a2 / pow2( denom );\n}\nvec3 BRDF_Specular_GGX( const in IncidentLight incidentLight, const in vec3 viewDir, const in vec3 normal, const in vec3 specularColor, const in float roughness ) {\n\tfloat alpha = pow2( roughness );\n\tvec3 halfDir = normalize( incidentLight.direction + viewDir );\n\tfloat dotNL = saturate( dot( normal, incidentLight.direction ) );\n\tfloat dotNV = saturate( dot( normal, viewDir ) );\n\tfloat dotNH = saturate( dot( normal, halfDir ) );\n\tfloat dotLH = saturate( dot( incidentLight.direction, halfDir ) );\n\tvec3 F = F_Schlick( specularColor, dotLH );\n\tfloat G = G_GGX_SmithCorrelated( alpha, dotNL, dotNV );\n\tfloat D = D_GGX( alpha, dotNH );\n\treturn F * ( G * D );\n}\nvec2 LTC_Uv( const in vec3 N, const in vec3 V, const in float roughness ) {\n\tconst float LUT_SIZE  = 64.0;\n\tconst float LUT_SCALE = ( LUT_SIZE - 1.0 ) / LUT_SIZE;\n\tconst float LUT_BIAS  = 0.5 / LUT_SIZE;\n\tfloat dotNV = saturate( dot( N, V ) );\n\tvec2 uv = vec2( roughness, sqrt( 1.0 - dotNV ) );\n\tuv = uv * LUT_SCALE + LUT_BIAS;\n\treturn uv;\n}\nfloat LTC_ClippedSphereFormFactor( const in vec3 f ) {\n\tfloat l = length( f );\n\treturn max( ( l * l + f.z ) / ( l + 1.0 ), 0.0 );\n}\nvec3 LTC_EdgeVectorFormFactor( const in vec3 v1, const in vec3 v2 ) {\n\tfloat x = dot( v1, v2 );\n\tfloat y = abs( x );\n\tfloat a = 0.8543985 + ( 0.4965155 + 0.0145206 * y ) * y;\n\tfloat b = 3.4175940 + ( 4.1616724 + y ) * y;\n\tfloat v = a / b;\n\tfloat theta_sintheta = ( x > 0.0 ) ? v : 0.5 * inversesqrt( max( 1.0 - x * x, 1e-7 ) ) - v;\n\treturn cross( v1, v2 ) * theta_sintheta;\n}\nvec3 LTC_Evaluate( const in vec3 N, const in vec3 V, const in vec3 P, const in mat3 mInv, const in vec3 rectCoords[ 4 ] ) {\n\tvec3 v1 = rectCoords[ 1 ] - rectCoords[ 0 ];\n\tvec3 v2 = rectCoords[ 3 ] - rectCoords[ 0 ];\n\tvec3 lightNormal = cross( v1, v2 );\n\tif( dot( lightNormal, P - rectCoords[ 0 ] ) < 0.0 ) return vec3( 0.0 );\n\tvec3 T1, T2;\n\tT1 = normalize( V - N * dot( V, N ) );\n\tT2 = - cross( N, T1 );\n\tmat3 mat = mInv * transposeMat3( mat3( T1, T2, N ) );\n\tvec3 coords[ 4 ];\n\tcoords[ 0 ] = mat * ( rectCoords[ 0 ] - P );\n\tcoords[ 1 ] = mat * ( rectCoords[ 1 ] - P );\n\tcoords[ 2 ] = mat * ( rectCoords[ 2 ] - P );\n\tcoords[ 3 ] = mat * ( rectCoords[ 3 ] - P );\n\tcoords[ 0 ] = normalize( coords[ 0 ] );\n\tcoords[ 1 ] = normalize( coords[ 1 ] );\n\tcoords[ 2 ] = normalize( coords[ 2 ] );\n\tcoords[ 3 ] = normalize( coords[ 3 ] );\n\tvec3 vectorFormFactor = vec3( 0.0 );\n\tvectorFormFactor += LTC_EdgeVectorFormFactor( coords[ 0 ], coords[ 1 ] );\n\tvectorFormFactor += LTC_EdgeVectorFormFactor( coords[ 1 ], coords[ 2 ] );\n\tvectorFormFactor += LTC_EdgeVectorFormFactor( coords[ 2 ], coords[ 3 ] );\n\tvectorFormFactor += LTC_EdgeVectorFormFactor( coords[ 3 ], coords[ 0 ] );\n\tfloat result = LTC_ClippedSphereFormFactor( vectorFormFactor );\n\treturn vec3( result );\n}\nvec3 BRDF_Specular_GGX_Environment( const in vec3 viewDir, const in vec3 normal, const in vec3 specularColor, const in float roughness ) {\n\tfloat dotNV = saturate( dot( normal, viewDir ) );\n\tvec2 brdf = integrateSpecularBRDF( dotNV, roughness );\n\treturn specularColor * brdf.x + brdf.y;\n}\nvoid BRDF_Specular_Multiscattering_Environment( const in GeometricContext geometry, const in vec3 specularColor, const in float roughness, inout vec3 singleScatter, inout vec3 multiScatter ) {\n\tfloat dotNV = saturate( dot( geometry.normal, geometry.viewDir ) );\n\tvec3 F = F_Schlick_RoughnessDependent( specularColor, dotNV, roughness );\n\tvec2 brdf = integrateSpecularBRDF( dotNV, roughness );\n\tvec3 FssEss = F * brdf.x + brdf.y;\n\tfloat Ess = brdf.x + brdf.y;\n\tfloat Ems = 1.0 - Ess;\n\tvec3 Favg = specularColor + ( 1.0 - specularColor ) * 0.047619;\tvec3 Fms = FssEss * Favg / ( 1.0 - Ems * Favg );\n\tsingleScatter += FssEss;\n\tmultiScatter += Fms * Ems;\n}\nfloat G_BlinnPhong_Implicit( ) {\n\treturn 0.25;\n}\nfloat D_BlinnPhong( const in float shininess, const in float dotNH ) {\n\treturn RECIPROCAL_PI * ( shininess * 0.5 + 1.0 ) * pow( dotNH, shininess );\n}\nvec3 BRDF_Specular_BlinnPhong( const in IncidentLight incidentLight, const in GeometricContext geometry, const in vec3 specularColor, const in float shininess ) {\n\tvec3 halfDir = normalize( incidentLight.direction + geometry.viewDir );\n\tfloat dotNH = saturate( dot( geometry.normal, halfDir ) );\n\tfloat dotLH = saturate( dot( incidentLight.direction, halfDir ) );\n\tvec3 F = F_Schlick( specularColor, dotLH );\n\tfloat G = G_BlinnPhong_Implicit( );\n\tfloat D = D_BlinnPhong( shininess, dotNH );\n\treturn F * ( G * D );\n}\nfloat GGXRoughnessToBlinnExponent( const in float ggxRoughness ) {\n\treturn ( 2.0 / pow2( ggxRoughness + 0.0001 ) - 2.0 );\n}\nfloat BlinnExponentToGGXRoughness( const in float blinnExponent ) {\n\treturn sqrt( 2.0 / ( blinnExponent + 2.0 ) );\n}\n#if defined( USE_SHEEN )\nfloat D_Charlie(float roughness, float NoH) {\n\tfloat invAlpha  = 1.0 / roughness;\n\tfloat cos2h = NoH * NoH;\n\tfloat sin2h = max(1.0 - cos2h, 0.0078125);\treturn (2.0 + invAlpha) * pow(sin2h, invAlpha * 0.5) / (2.0 * PI);\n}\nfloat V_Neubelt(float NoV, float NoL) {\n\treturn saturate(1.0 / (4.0 * (NoL + NoV - NoL * NoV)));\n}\nvec3 BRDF_Specular_Sheen( const in float roughness, const in vec3 L, const in GeometricContext geometry, vec3 specularColor ) {\n\tvec3 N = geometry.normal;\n\tvec3 V = geometry.viewDir;\n\tvec3 H = normalize( V + L );\n\tfloat dotNH = saturate( dot( N, H ) );\n\treturn specularColor * D_Charlie( roughness, dotNH ) * V_Neubelt( dot(N, V), dot(N, L) );\n}\n#endif";

    var bumpmap_pars_fragment = "#ifdef USE_BUMPMAP\n\tuniform sampler2D bumpMap;\n\tuniform float bumpScale;\n\tvec2 dHdxy_fwd() {\n\t\tvec2 dSTdx = dFdx( vUv );\n\t\tvec2 dSTdy = dFdy( vUv );\n\t\tfloat Hll = bumpScale * texture2D( bumpMap, vUv ).x;\n\t\tfloat dBx = bumpScale * texture2D( bumpMap, vUv + dSTdx ).x - Hll;\n\t\tfloat dBy = bumpScale * texture2D( bumpMap, vUv + dSTdy ).x - Hll;\n\t\treturn vec2( dBx, dBy );\n\t}\n\tvec3 perturbNormalArb( vec3 surf_pos, vec3 surf_norm, vec2 dHdxy ) {\n\t\tvec3 vSigmaX = vec3( dFdx( surf_pos.x ), dFdx( surf_pos.y ), dFdx( surf_pos.z ) );\n\t\tvec3 vSigmaY = vec3( dFdy( surf_pos.x ), dFdy( surf_pos.y ), dFdy( surf_pos.z ) );\n\t\tvec3 vN = surf_norm;\n\t\tvec3 R1 = cross( vSigmaY, vN );\n\t\tvec3 R2 = cross( vN, vSigmaX );\n\t\tfloat fDet = dot( vSigmaX, R1 );\n\t\tfDet *= ( float( gl_FrontFacing ) * 2.0 - 1.0 );\n\t\tvec3 vGrad = sign( fDet ) * ( dHdxy.x * R1 + dHdxy.y * R2 );\n\t\treturn normalize( abs( fDet ) * surf_norm - vGrad );\n\t}\n#endif";

    var clipping_planes_fragment = "#if NUM_CLIPPING_PLANES > 0\n\tvec4 plane;\n\t#pragma unroll_loop\n\tfor ( int i = 0; i < UNION_CLIPPING_PLANES; i ++ ) {\n\t\tplane = clippingPlanes[ i ];\n\t\tif ( dot( vViewPosition, plane.xyz ) > plane.w ) discard;\n\t}\n\t#if UNION_CLIPPING_PLANES < NUM_CLIPPING_PLANES\n\t\tbool clipped = true;\n\t\t#pragma unroll_loop\n\t\tfor ( int i = UNION_CLIPPING_PLANES; i < NUM_CLIPPING_PLANES; i ++ ) {\n\t\t\tplane = clippingPlanes[ i ];\n\t\t\tclipped = ( dot( vViewPosition, plane.xyz ) > plane.w ) && clipped;\n\t\t}\n\t\tif ( clipped ) discard;\n\t#endif\n#endif";

    var clipping_planes_pars_fragment = "#if NUM_CLIPPING_PLANES > 0\n\t#if ! defined( STANDARD ) && ! defined( PHONG ) && ! defined( MATCAP )\n\t\tvarying vec3 vViewPosition;\n\t#endif\n\tuniform vec4 clippingPlanes[ NUM_CLIPPING_PLANES ];\n#endif";

    var clipping_planes_pars_vertex = "#if NUM_CLIPPING_PLANES > 0 && ! defined( STANDARD ) && ! defined( PHONG ) && ! defined( MATCAP )\n\tvarying vec3 vViewPosition;\n#endif";

    var clipping_planes_vertex = "#if NUM_CLIPPING_PLANES > 0 && ! defined( STANDARD ) && ! defined( PHONG ) && ! defined( MATCAP )\n\tvViewPosition = - mvPosition.xyz;\n#endif";

    var color_fragment = "#ifdef USE_COLOR\n\tdiffuseColor.rgb *= vColor;\n#endif";

    var color_pars_fragment = "#ifdef USE_COLOR\n\tvarying vec3 vColor;\n#endif";

    var color_pars_vertex = "#ifdef USE_COLOR\n\tvarying vec3 vColor;\n#endif";

    var color_vertex = "#ifdef USE_COLOR\n\tvColor.xyz = color.xyz;\n#endif";

    var common = "#define PI 3.14159265359\n#define PI2 6.28318530718\n#define PI_HALF 1.5707963267949\n#define RECIPROCAL_PI 0.31830988618\n#define RECIPROCAL_PI2 0.15915494\n#define LOG2 1.442695\n#define EPSILON 1e-6\n#define saturate(a) clamp( a, 0.0, 1.0 )\n#define whiteComplement(a) ( 1.0 - saturate( a ) )\nfloat pow2( const in float x ) { return x*x; }\nfloat pow3( const in float x ) { return x*x*x; }\nfloat pow4( const in float x ) { float x2 = x*x; return x2*x2; }\nfloat average( const in vec3 color ) { return dot( color, vec3( 0.3333 ) ); }\nhighp float rand( const in vec2 uv ) {\n\tconst highp float a = 12.9898, b = 78.233, c = 43758.5453;\n\thighp float dt = dot( uv.xy, vec2( a,b ) ), sn = mod( dt, PI );\n\treturn fract(sin(sn) * c);\n}\n#ifdef HIGH_PRECISION\n\tfloat precisionSafeLength( vec3 v ) { return length( v ); }\n#else\n\tfloat max3( vec3 v ) { return max( max( v.x, v.y ), v.z ); }\n\tfloat precisionSafeLength( vec3 v ) {\n\t\tfloat maxComponent = max3( abs( v ) );\n\t\treturn length( v / maxComponent ) * maxComponent;\n\t}\n#endif\nstruct IncidentLight {\n\tvec3 color;\n\tvec3 direction;\n\tbool visible;\n};\nstruct ReflectedLight {\n\tvec3 directDiffuse;\n\tvec3 directSpecular;\n\tvec3 indirectDiffuse;\n\tvec3 indirectSpecular;\n};\nstruct GeometricContext {\n\tvec3 position;\n\tvec3 normal;\n\tvec3 viewDir;\n#ifdef CLEARCOAT\n\tvec3 clearcoatNormal;\n#endif\n};\nvec3 transformDirection( in vec3 dir, in mat4 matrix ) {\n\treturn normalize( ( matrix * vec4( dir, 0.0 ) ).xyz );\n}\nvec3 inverseTransformDirection( in vec3 dir, in mat4 matrix ) {\n\treturn normalize( ( vec4( dir, 0.0 ) * matrix ).xyz );\n}\nvec3 projectOnPlane(in vec3 point, in vec3 pointOnPlane, in vec3 planeNormal ) {\n\tfloat distance = dot( planeNormal, point - pointOnPlane );\n\treturn - distance * planeNormal + point;\n}\nfloat sideOfPlane( in vec3 point, in vec3 pointOnPlane, in vec3 planeNormal ) {\n\treturn sign( dot( point - pointOnPlane, planeNormal ) );\n}\nvec3 linePlaneIntersect( in vec3 pointOnLine, in vec3 lineDirection, in vec3 pointOnPlane, in vec3 planeNormal ) {\n\treturn lineDirection * ( dot( planeNormal, pointOnPlane - pointOnLine ) / dot( planeNormal, lineDirection ) ) + pointOnLine;\n}\nmat3 transposeMat3( const in mat3 m ) {\n\tmat3 tmp;\n\ttmp[ 0 ] = vec3( m[ 0 ].x, m[ 1 ].x, m[ 2 ].x );\n\ttmp[ 1 ] = vec3( m[ 0 ].y, m[ 1 ].y, m[ 2 ].y );\n\ttmp[ 2 ] = vec3( m[ 0 ].z, m[ 1 ].z, m[ 2 ].z );\n\treturn tmp;\n}\nfloat linearToRelativeLuminance( const in vec3 color ) {\n\tvec3 weights = vec3( 0.2126, 0.7152, 0.0722 );\n\treturn dot( weights, color.rgb );\n}";

    var cube_uv_reflection_fragment = "#ifdef ENVMAP_TYPE_CUBE_UV\n#define cubeUV_textureSize (1024.0)\nint getFaceFromDirection(vec3 direction) {\n\tvec3 absDirection = abs(direction);\n\tint face = -1;\n\tif( absDirection.x > absDirection.z ) {\n\t\tif(absDirection.x > absDirection.y )\n\t\t\tface = direction.x > 0.0 ? 0 : 3;\n\t\telse\n\t\t\tface = direction.y > 0.0 ? 1 : 4;\n\t}\n\telse {\n\t\tif(absDirection.z > absDirection.y )\n\t\t\tface = direction.z > 0.0 ? 2 : 5;\n\t\telse\n\t\t\tface = direction.y > 0.0 ? 1 : 4;\n\t}\n\treturn face;\n}\n#define cubeUV_maxLods1  (log2(cubeUV_textureSize*0.25) - 1.0)\n#define cubeUV_rangeClamp (exp2((6.0 - 1.0) * 2.0))\nvec2 MipLevelInfo( vec3 vec, float roughnessLevel, float roughness ) {\n\tfloat scale = exp2(cubeUV_maxLods1 - roughnessLevel);\n\tfloat dxRoughness = dFdx(roughness);\n\tfloat dyRoughness = dFdy(roughness);\n\tvec3 dx = dFdx( vec * scale * dxRoughness );\n\tvec3 dy = dFdy( vec * scale * dyRoughness );\n\tfloat d = max( dot( dx, dx ), dot( dy, dy ) );\n\td = clamp(d, 1.0, cubeUV_rangeClamp);\n\tfloat mipLevel = 0.5 * log2(d);\n\treturn vec2(floor(mipLevel), fract(mipLevel));\n}\n#define cubeUV_maxLods2 (log2(cubeUV_textureSize*0.25) - 2.0)\n#define cubeUV_rcpTextureSize (1.0 / cubeUV_textureSize)\nvec2 getCubeUV(vec3 direction, float roughnessLevel, float mipLevel) {\n\tmipLevel = roughnessLevel > cubeUV_maxLods2 - 3.0 ? 0.0 : mipLevel;\n\tfloat a = 16.0 * cubeUV_rcpTextureSize;\n\tvec2 exp2_packed = exp2( vec2( roughnessLevel, mipLevel ) );\n\tvec2 rcp_exp2_packed = vec2( 1.0 ) / exp2_packed;\n\tfloat powScale = exp2_packed.x * exp2_packed.y;\n\tfloat scale = rcp_exp2_packed.x * rcp_exp2_packed.y * 0.25;\n\tfloat mipOffset = 0.75*(1.0 - rcp_exp2_packed.y) * rcp_exp2_packed.x;\n\tbool bRes = mipLevel == 0.0;\n\tscale =  bRes && (scale < a) ? a : scale;\n\tvec3 r;\n\tvec2 offset;\n\tint face = getFaceFromDirection(direction);\n\tfloat rcpPowScale = 1.0 / powScale;\n\tif( face == 0) {\n\t\tr = vec3(direction.x, -direction.z, direction.y);\n\t\toffset = vec2(0.0+mipOffset,0.75 * rcpPowScale);\n\t\toffset.y = bRes && (offset.y < 2.0*a) ? a : offset.y;\n\t}\n\telse if( face == 1) {\n\t\tr = vec3(direction.y, direction.x, direction.z);\n\t\toffset = vec2(scale+mipOffset, 0.75 * rcpPowScale);\n\t\toffset.y = bRes && (offset.y < 2.0*a) ? a : offset.y;\n\t}\n\telse if( face == 2) {\n\t\tr = vec3(direction.z, direction.x, direction.y);\n\t\toffset = vec2(2.0*scale+mipOffset, 0.75 * rcpPowScale);\n\t\toffset.y = bRes && (offset.y < 2.0*a) ? a : offset.y;\n\t}\n\telse if( face == 3) {\n\t\tr = vec3(direction.x, direction.z, direction.y);\n\t\toffset = vec2(0.0+mipOffset,0.5 * rcpPowScale);\n\t\toffset.y = bRes && (offset.y < 2.0*a) ? 0.0 : offset.y;\n\t}\n\telse if( face == 4) {\n\t\tr = vec3(direction.y, direction.x, -direction.z);\n\t\toffset = vec2(scale+mipOffset, 0.5 * rcpPowScale);\n\t\toffset.y = bRes && (offset.y < 2.0*a) ? 0.0 : offset.y;\n\t}\n\telse {\n\t\tr = vec3(direction.z, -direction.x, direction.y);\n\t\toffset = vec2(2.0*scale+mipOffset, 0.5 * rcpPowScale);\n\t\toffset.y = bRes && (offset.y < 2.0*a) ? 0.0 : offset.y;\n\t}\n\tr = normalize(r);\n\tfloat texelOffset = 0.5 * cubeUV_rcpTextureSize;\n\tvec2 s = ( r.yz / abs( r.x ) + vec2( 1.0 ) ) * 0.5;\n\tvec2 base = offset + vec2( texelOffset );\n\treturn base + s * ( scale - 2.0 * texelOffset );\n}\n#define cubeUV_maxLods3 (log2(cubeUV_textureSize*0.25) - 3.0)\nvec4 textureCubeUV( sampler2D envMap, vec3 reflectedDirection, float roughness ) {\n\tfloat roughnessVal = roughness* cubeUV_maxLods3;\n\tfloat r1 = floor(roughnessVal);\n\tfloat r2 = r1 + 1.0;\n\tfloat t = fract(roughnessVal);\n\tvec2 mipInfo = MipLevelInfo(reflectedDirection, r1, roughness);\n\tfloat s = mipInfo.y;\n\tfloat level0 = mipInfo.x;\n\tfloat level1 = level0 + 1.0;\n\tlevel1 = level1 > 5.0 ? 5.0 : level1;\n\tlevel0 += min( floor( s + 0.5 ), 5.0 );\n\tvec2 uv_10 = getCubeUV(reflectedDirection, r1, level0);\n\tvec4 color10 = envMapTexelToLinear(texture2D(envMap, uv_10));\n\tvec2 uv_20 = getCubeUV(reflectedDirection, r2, level0);\n\tvec4 color20 = envMapTexelToLinear(texture2D(envMap, uv_20));\n\tvec4 result = mix(color10, color20, t);\n\treturn vec4(result.rgb, 1.0);\n}\n#endif";

    var defaultnormal_vertex = "vec3 transformedNormal = normalMatrix * objectNormal;\n#ifdef FLIP_SIDED\n\ttransformedNormal = - transformedNormal;\n#endif\n#ifdef USE_TANGENT\n\tvec3 transformedTangent = normalMatrix * objectTangent;\n\t#ifdef FLIP_SIDED\n\t\ttransformedTangent = - transformedTangent;\n\t#endif\n#endif";

    var displacementmap_pars_vertex = "#ifdef USE_DISPLACEMENTMAP\n\tuniform sampler2D displacementMap;\n\tuniform float displacementScale;\n\tuniform float displacementBias;\n#endif";

    var displacementmap_vertex = "#ifdef USE_DISPLACEMENTMAP\n\ttransformed += normalize( objectNormal ) * ( texture2D( displacementMap, uv ).x * displacementScale + displacementBias );\n#endif";

    var emissivemap_fragment = "#ifdef USE_EMISSIVEMAP\n\tvec4 emissiveColor = texture2D( emissiveMap, vUv );\n\temissiveColor.rgb = emissiveMapTexelToLinear( emissiveColor ).rgb;\n\ttotalEmissiveRadiance *= emissiveColor.rgb;\n#endif";

    var emissivemap_pars_fragment = "#ifdef USE_EMISSIVEMAP\n\tuniform sampler2D emissiveMap;\n#endif";

    var encodings_fragment = "gl_FragColor = linearToOutputTexel( gl_FragColor );";

    var encodings_pars_fragment = "\nvec4 LinearToLinear( in vec4 value ) {\n\treturn value;\n}\nvec4 GammaToLinear( in vec4 value, in float gammaFactor ) {\n\treturn vec4( pow( value.rgb, vec3( gammaFactor ) ), value.a );\n}\nvec4 LinearToGamma( in vec4 value, in float gammaFactor ) {\n\treturn vec4( pow( value.rgb, vec3( 1.0 / gammaFactor ) ), value.a );\n}\nvec4 sRGBToLinear( in vec4 value ) {\n\treturn vec4( mix( pow( value.rgb * 0.9478672986 + vec3( 0.0521327014 ), vec3( 2.4 ) ), value.rgb * 0.0773993808, vec3( lessThanEqual( value.rgb, vec3( 0.04045 ) ) ) ), value.a );\n}\nvec4 LinearTosRGB( in vec4 value ) {\n\treturn vec4( mix( pow( value.rgb, vec3( 0.41666 ) ) * 1.055 - vec3( 0.055 ), value.rgb * 12.92, vec3( lessThanEqual( value.rgb, vec3( 0.0031308 ) ) ) ), value.a );\n}\nvec4 RGBEToLinear( in vec4 value ) {\n\treturn vec4( value.rgb * exp2( value.a * 255.0 - 128.0 ), 1.0 );\n}\nvec4 LinearToRGBE( in vec4 value ) {\n\tfloat maxComponent = max( max( value.r, value.g ), value.b );\n\tfloat fExp = clamp( ceil( log2( maxComponent ) ), -128.0, 127.0 );\n\treturn vec4( value.rgb / exp2( fExp ), ( fExp + 128.0 ) / 255.0 );\n}\nvec4 RGBMToLinear( in vec4 value, in float maxRange ) {\n\treturn vec4( value.rgb * value.a * maxRange, 1.0 );\n}\nvec4 LinearToRGBM( in vec4 value, in float maxRange ) {\n\tfloat maxRGB = max( value.r, max( value.g, value.b ) );\n\tfloat M = clamp( maxRGB / maxRange, 0.0, 1.0 );\n\tM = ceil( M * 255.0 ) / 255.0;\n\treturn vec4( value.rgb / ( M * maxRange ), M );\n}\nvec4 RGBDToLinear( in vec4 value, in float maxRange ) {\n\treturn vec4( value.rgb * ( ( maxRange / 255.0 ) / value.a ), 1.0 );\n}\nvec4 LinearToRGBD( in vec4 value, in float maxRange ) {\n\tfloat maxRGB = max( value.r, max( value.g, value.b ) );\n\tfloat D = max( maxRange / maxRGB, 1.0 );\n\tD = min( floor( D ) / 255.0, 1.0 );\n\treturn vec4( value.rgb * ( D * ( 255.0 / maxRange ) ), D );\n}\nconst mat3 cLogLuvM = mat3( 0.2209, 0.3390, 0.4184, 0.1138, 0.6780, 0.7319, 0.0102, 0.1130, 0.2969 );\nvec4 LinearToLogLuv( in vec4 value )  {\n\tvec3 Xp_Y_XYZp = cLogLuvM * value.rgb;\n\tXp_Y_XYZp = max( Xp_Y_XYZp, vec3( 1e-6, 1e-6, 1e-6 ) );\n\tvec4 vResult;\n\tvResult.xy = Xp_Y_XYZp.xy / Xp_Y_XYZp.z;\n\tfloat Le = 2.0 * log2(Xp_Y_XYZp.y) + 127.0;\n\tvResult.w = fract( Le );\n\tvResult.z = ( Le - ( floor( vResult.w * 255.0 ) ) / 255.0 ) / 255.0;\n\treturn vResult;\n}\nconst mat3 cLogLuvInverseM = mat3( 6.0014, -2.7008, -1.7996, -1.3320, 3.1029, -5.7721, 0.3008, -1.0882, 5.6268 );\nvec4 LogLuvToLinear( in vec4 value ) {\n\tfloat Le = value.z * 255.0 + value.w;\n\tvec3 Xp_Y_XYZp;\n\tXp_Y_XYZp.y = exp2( ( Le - 127.0 ) / 2.0 );\n\tXp_Y_XYZp.z = Xp_Y_XYZp.y / value.y;\n\tXp_Y_XYZp.x = value.x * Xp_Y_XYZp.z;\n\tvec3 vRGB = cLogLuvInverseM * Xp_Y_XYZp.rgb;\n\treturn vec4( max( vRGB, 0.0 ), 1.0 );\n}";

    var envmap_fragment = "#ifdef USE_ENVMAP\n\t#ifdef ENV_WORLDPOS\n\t\tvec3 cameraToVertex = normalize( vWorldPosition - cameraPosition );\n\t\tvec3 worldNormal = inverseTransformDirection( normal, viewMatrix );\n\t\t#ifdef ENVMAP_MODE_REFLECTION\n\t\t\tvec3 reflectVec = reflect( cameraToVertex, worldNormal );\n\t\t#else\n\t\t\tvec3 reflectVec = refract( cameraToVertex, worldNormal, refractionRatio );\n\t\t#endif\n\t#else\n\t\tvec3 reflectVec = vReflect;\n\t#endif\n\t#ifdef ENVMAP_TYPE_CUBE\n\t\tvec4 envColor = textureCube( envMap, vec3( flipEnvMap * reflectVec.x, reflectVec.yz ) );\n\t#elif defined( ENVMAP_TYPE_EQUIREC )\n\t\tvec2 sampleUV;\n\t\treflectVec = normalize( reflectVec );\n\t\tsampleUV.y = asin( clamp( reflectVec.y, - 1.0, 1.0 ) ) * RECIPROCAL_PI + 0.5;\n\t\tsampleUV.x = atan( reflectVec.z, reflectVec.x ) * RECIPROCAL_PI2 + 0.5;\n\t\tvec4 envColor = texture2D( envMap, sampleUV );\n\t#elif defined( ENVMAP_TYPE_SPHERE )\n\t\treflectVec = normalize( reflectVec );\n\t\tvec3 reflectView = normalize( ( viewMatrix * vec4( reflectVec, 0.0 ) ).xyz + vec3( 0.0, 0.0, 1.0 ) );\n\t\tvec4 envColor = texture2D( envMap, reflectView.xy * 0.5 + 0.5 );\n\t#else\n\t\tvec4 envColor = vec4( 0.0 );\n\t#endif\n\tenvColor = envMapTexelToLinear( envColor );\n\t#ifdef ENVMAP_BLENDING_MULTIPLY\n\t\toutgoingLight = mix( outgoingLight, outgoingLight * envColor.xyz, specularStrength * reflectivity );\n\t#elif defined( ENVMAP_BLENDING_MIX )\n\t\toutgoingLight = mix( outgoingLight, envColor.xyz, specularStrength * reflectivity );\n\t#elif defined( ENVMAP_BLENDING_ADD )\n\t\toutgoingLight += envColor.xyz * specularStrength * reflectivity;\n\t#endif\n#endif";

    var envmap_common_pars_fragment = "#ifdef USE_ENVMAP\n\tuniform float envMapIntensity;\n\tuniform float flipEnvMap;\n\tuniform int maxMipLevel;\n\t#ifdef ENVMAP_TYPE_CUBE\n\t\tuniform samplerCube envMap;\n\t#else\n\t\tuniform sampler2D envMap;\n\t#endif\n\t\n#endif";

    var envmap_pars_fragment = "#ifdef USE_ENVMAP\n\tuniform float reflectivity;\n\t#if defined( USE_BUMPMAP ) || defined( USE_NORMALMAP ) || defined( PHONG )\n\t\t#define ENV_WORLDPOS\n\t#endif\n\t#ifdef ENV_WORLDPOS\n\t\tvarying vec3 vWorldPosition;\n\t\tuniform float refractionRatio;\n\t#else\n\t\tvarying vec3 vReflect;\n\t#endif\n#endif";

    var envmap_pars_vertex = "#ifdef USE_ENVMAP\n\t#if defined( USE_BUMPMAP ) || defined( USE_NORMALMAP ) ||defined( PHONG )\n\t\t#define ENV_WORLDPOS\n\t#endif\n\t#ifdef ENV_WORLDPOS\n\t\t\n\t\tvarying vec3 vWorldPosition;\n\t#else\n\t\tvarying vec3 vReflect;\n\t\tuniform float refractionRatio;\n\t#endif\n#endif";

    var envmap_vertex = "#ifdef USE_ENVMAP\n\t#ifdef ENV_WORLDPOS\n\t\tvWorldPosition = worldPosition.xyz;\n\t#else\n\t\tvec3 cameraToVertex = normalize( worldPosition.xyz - cameraPosition );\n\t\tvec3 worldNormal = inverseTransformDirection( transformedNormal, viewMatrix );\n\t\t#ifdef ENVMAP_MODE_REFLECTION\n\t\t\tvReflect = reflect( cameraToVertex, worldNormal );\n\t\t#else\n\t\t\tvReflect = refract( cameraToVertex, worldNormal, refractionRatio );\n\t\t#endif\n\t#endif\n#endif";

    var fog_vertex = "#ifdef USE_FOG\n\tfogDepth = -mvPosition.z;\n#endif";

    var fog_pars_vertex = "#ifdef USE_FOG\n\tvarying float fogDepth;\n#endif";

    var fog_fragment = "#ifdef USE_FOG\n\t#ifdef FOG_EXP2\n\t\tfloat fogFactor = 1.0 - exp( - fogDensity * fogDensity * fogDepth * fogDepth );\n\t#else\n\t\tfloat fogFactor = smoothstep( fogNear, fogFar, fogDepth );\n\t#endif\n\tgl_FragColor.rgb = mix( gl_FragColor.rgb, fogColor, fogFactor );\n#endif";

    var fog_pars_fragment = "#ifdef USE_FOG\n\tuniform vec3 fogColor;\n\tvarying float fogDepth;\n\t#ifdef FOG_EXP2\n\t\tuniform float fogDensity;\n\t#else\n\t\tuniform float fogNear;\n\t\tuniform float fogFar;\n\t#endif\n#endif";

    var gradientmap_pars_fragment = "#ifdef TOON\n\tuniform sampler2D gradientMap;\n\tvec3 getGradientIrradiance( vec3 normal, vec3 lightDirection ) {\n\t\tfloat dotNL = dot( normal, lightDirection );\n\t\tvec2 coord = vec2( dotNL * 0.5 + 0.5, 0.0 );\n\t\t#ifdef USE_GRADIENTMAP\n\t\t\treturn texture2D( gradientMap, coord ).rgb;\n\t\t#else\n\t\t\treturn ( coord.x < 0.7 ) ? vec3( 0.7 ) : vec3( 1.0 );\n\t\t#endif\n\t}\n#endif";

    var lightmap_fragment = "#ifdef USE_LIGHTMAP\n\treflectedLight.indirectDiffuse += PI * texture2D( lightMap, vUv2 ).xyz * lightMapIntensity;\n#endif";

    var lightmap_pars_fragment = "#ifdef USE_LIGHTMAP\n\tuniform sampler2D lightMap;\n\tuniform float lightMapIntensity;\n#endif";

    var lights_lambert_vertex = "vec3 diffuse = vec3( 1.0 );\nGeometricContext geometry;\ngeometry.position = mvPosition.xyz;\ngeometry.normal = normalize( transformedNormal );\ngeometry.viewDir = normalize( -mvPosition.xyz );\nGeometricContext backGeometry;\nbackGeometry.position = geometry.position;\nbackGeometry.normal = -geometry.normal;\nbackGeometry.viewDir = geometry.viewDir;\nvLightFront = vec3( 0.0 );\nvIndirectFront = vec3( 0.0 );\n#ifdef DOUBLE_SIDED\n\tvLightBack = vec3( 0.0 );\n\tvIndirectBack = vec3( 0.0 );\n#endif\nIncidentLight directLight;\nfloat dotNL;\nvec3 directLightColor_Diffuse;\n#if NUM_POINT_LIGHTS > 0\n\t#pragma unroll_loop\n\tfor ( int i = 0; i < NUM_POINT_LIGHTS; i ++ ) {\n\t\tgetPointDirectLightIrradiance( pointLights[ i ], geometry, directLight );\n\t\tdotNL = dot( geometry.normal, directLight.direction );\n\t\tdirectLightColor_Diffuse = PI * directLight.color;\n\t\tvLightFront += saturate( dotNL ) * directLightColor_Diffuse;\n\t\t#ifdef DOUBLE_SIDED\n\t\t\tvLightBack += saturate( -dotNL ) * directLightColor_Diffuse;\n\t\t#endif\n\t}\n#endif\n#if NUM_SPOT_LIGHTS > 0\n\t#pragma unroll_loop\n\tfor ( int i = 0; i < NUM_SPOT_LIGHTS; i ++ ) {\n\t\tgetSpotDirectLightIrradiance( spotLights[ i ], geometry, directLight );\n\t\tdotNL = dot( geometry.normal, directLight.direction );\n\t\tdirectLightColor_Diffuse = PI * directLight.color;\n\t\tvLightFront += saturate( dotNL ) * directLightColor_Diffuse;\n\t\t#ifdef DOUBLE_SIDED\n\t\t\tvLightBack += saturate( -dotNL ) * directLightColor_Diffuse;\n\t\t#endif\n\t}\n#endif\n#if NUM_DIR_LIGHTS > 0\n\t#pragma unroll_loop\n\tfor ( int i = 0; i < NUM_DIR_LIGHTS; i ++ ) {\n\t\tgetDirectionalDirectLightIrradiance( directionalLights[ i ], geometry, directLight );\n\t\tdotNL = dot( geometry.normal, directLight.direction );\n\t\tdirectLightColor_Diffuse = PI * directLight.color;\n\t\tvLightFront += saturate( dotNL ) * directLightColor_Diffuse;\n\t\t#ifdef DOUBLE_SIDED\n\t\t\tvLightBack += saturate( -dotNL ) * directLightColor_Diffuse;\n\t\t#endif\n\t}\n#endif\n#if NUM_HEMI_LIGHTS > 0\n\t#pragma unroll_loop\n\tfor ( int i = 0; i < NUM_HEMI_LIGHTS; i ++ ) {\n\t\tvIndirectFront += getHemisphereLightIrradiance( hemisphereLights[ i ], geometry );\n\t\t#ifdef DOUBLE_SIDED\n\t\t\tvIndirectBack += getHemisphereLightIrradiance( hemisphereLights[ i ], backGeometry );\n\t\t#endif\n\t}\n#endif";

    var lights_pars_begin = "uniform vec3 ambientLightColor;\nuniform vec3 lightProbe[ 9 ];\nvec3 shGetIrradianceAt( in vec3 normal, in vec3 shCoefficients[ 9 ] ) {\n\tfloat x = normal.x, y = normal.y, z = normal.z;\n\tvec3 result = shCoefficients[ 0 ] * 0.886227;\n\tresult += shCoefficients[ 1 ] * 2.0 * 0.511664 * y;\n\tresult += shCoefficients[ 2 ] * 2.0 * 0.511664 * z;\n\tresult += shCoefficients[ 3 ] * 2.0 * 0.511664 * x;\n\tresult += shCoefficients[ 4 ] * 2.0 * 0.429043 * x * y;\n\tresult += shCoefficients[ 5 ] * 2.0 * 0.429043 * y * z;\n\tresult += shCoefficients[ 6 ] * ( 0.743125 * z * z - 0.247708 );\n\tresult += shCoefficients[ 7 ] * 2.0 * 0.429043 * x * z;\n\tresult += shCoefficients[ 8 ] * 0.429043 * ( x * x - y * y );\n\treturn result;\n}\nvec3 getLightProbeIrradiance( const in vec3 lightProbe[ 9 ], const in GeometricContext geometry ) {\n\tvec3 worldNormal = inverseTransformDirection( geometry.normal, viewMatrix );\n\tvec3 irradiance = shGetIrradianceAt( worldNormal, lightProbe );\n\treturn irradiance;\n}\nvec3 getAmbientLightIrradiance( const in vec3 ambientLightColor ) {\n\tvec3 irradiance = ambientLightColor;\n\t#ifndef PHYSICALLY_CORRECT_LIGHTS\n\t\tirradiance *= PI;\n\t#endif\n\treturn irradiance;\n}\n#if NUM_DIR_LIGHTS > 0\n\tstruct DirectionalLight {\n\t\tvec3 direction;\n\t\tvec3 color;\n\t\tint shadow;\n\t\tfloat shadowBias;\n\t\tfloat shadowRadius;\n\t\tvec2 shadowMapSize;\n\t};\n\tuniform DirectionalLight directionalLights[ NUM_DIR_LIGHTS ];\n\tvoid getDirectionalDirectLightIrradiance( const in DirectionalLight directionalLight, const in GeometricContext geometry, out IncidentLight directLight ) {\n\t\tdirectLight.color = directionalLight.color;\n\t\tdirectLight.direction = directionalLight.direction;\n\t\tdirectLight.visible = true;\n\t}\n#endif\n#if NUM_POINT_LIGHTS > 0\n\tstruct PointLight {\n\t\tvec3 position;\n\t\tvec3 color;\n\t\tfloat distance;\n\t\tfloat decay;\n\t\tint shadow;\n\t\tfloat shadowBias;\n\t\tfloat shadowRadius;\n\t\tvec2 shadowMapSize;\n\t\tfloat shadowCameraNear;\n\t\tfloat shadowCameraFar;\n\t};\n\tuniform PointLight pointLights[ NUM_POINT_LIGHTS ];\n\tvoid getPointDirectLightIrradiance( const in PointLight pointLight, const in GeometricContext geometry, out IncidentLight directLight ) {\n\t\tvec3 lVector = pointLight.position - geometry.position;\n\t\tdirectLight.direction = normalize( lVector );\n\t\tfloat lightDistance = length( lVector );\n\t\tdirectLight.color = pointLight.color;\n\t\tdirectLight.color *= punctualLightIntensityToIrradianceFactor( lightDistance, pointLight.distance, pointLight.decay );\n\t\tdirectLight.visible = ( directLight.color != vec3( 0.0 ) );\n\t}\n#endif\n#if NUM_SPOT_LIGHTS > 0\n\tstruct SpotLight {\n\t\tvec3 position;\n\t\tvec3 direction;\n\t\tvec3 color;\n\t\tfloat distance;\n\t\tfloat decay;\n\t\tfloat coneCos;\n\t\tfloat penumbraCos;\n\t\tint shadow;\n\t\tfloat shadowBias;\n\t\tfloat shadowRadius;\n\t\tvec2 shadowMapSize;\n\t};\n\tuniform SpotLight spotLights[ NUM_SPOT_LIGHTS ];\n\tvoid getSpotDirectLightIrradiance( const in SpotLight spotLight, const in GeometricContext geometry, out IncidentLight directLight  ) {\n\t\tvec3 lVector = spotLight.position - geometry.position;\n\t\tdirectLight.direction = normalize( lVector );\n\t\tfloat lightDistance = length( lVector );\n\t\tfloat angleCos = dot( directLight.direction, spotLight.direction );\n\t\tif ( angleCos > spotLight.coneCos ) {\n\t\t\tfloat spotEffect = smoothstep( spotLight.coneCos, spotLight.penumbraCos, angleCos );\n\t\t\tdirectLight.color = spotLight.color;\n\t\t\tdirectLight.color *= spotEffect * punctualLightIntensityToIrradianceFactor( lightDistance, spotLight.distance, spotLight.decay );\n\t\t\tdirectLight.visible = true;\n\t\t} else {\n\t\t\tdirectLight.color = vec3( 0.0 );\n\t\t\tdirectLight.visible = false;\n\t\t}\n\t}\n#endif\n#if NUM_RECT_AREA_LIGHTS > 0\n\tstruct RectAreaLight {\n\t\tvec3 color;\n\t\tvec3 position;\n\t\tvec3 halfWidth;\n\t\tvec3 halfHeight;\n\t};\n\tuniform sampler2D ltc_1;\tuniform sampler2D ltc_2;\n\tuniform RectAreaLight rectAreaLights[ NUM_RECT_AREA_LIGHTS ];\n#endif\n#if NUM_HEMI_LIGHTS > 0\n\tstruct HemisphereLight {\n\t\tvec3 direction;\n\t\tvec3 skyColor;\n\t\tvec3 groundColor;\n\t};\n\tuniform HemisphereLight hemisphereLights[ NUM_HEMI_LIGHTS ];\n\tvec3 getHemisphereLightIrradiance( const in HemisphereLight hemiLight, const in GeometricContext geometry ) {\n\t\tfloat dotNL = dot( geometry.normal, hemiLight.direction );\n\t\tfloat hemiDiffuseWeight = 0.5 * dotNL + 0.5;\n\t\tvec3 irradiance = mix( hemiLight.groundColor, hemiLight.skyColor, hemiDiffuseWeight );\n\t\t#ifndef PHYSICALLY_CORRECT_LIGHTS\n\t\t\tirradiance *= PI;\n\t\t#endif\n\t\treturn irradiance;\n\t}\n#endif";

    var envmap_physical_pars_fragment = "#if defined( USE_ENVMAP )\n\t#ifdef ENVMAP_MODE_REFRACTION\n\t\tuniform float refractionRatio;\n\t#endif\n\tvec3 getLightProbeIndirectIrradiance( const in GeometricContext geometry, const in int maxMIPLevel ) {\n\t\tvec3 worldNormal = inverseTransformDirection( geometry.normal, viewMatrix );\n\t\t#ifdef ENVMAP_TYPE_CUBE\n\t\t\tvec3 queryVec = vec3( flipEnvMap * worldNormal.x, worldNormal.yz );\n\t\t\t#ifdef TEXTURE_LOD_EXT\n\t\t\t\tvec4 envMapColor = textureCubeLodEXT( envMap, queryVec, float( maxMIPLevel ) );\n\t\t\t#else\n\t\t\t\tvec4 envMapColor = textureCube( envMap, queryVec, float( maxMIPLevel ) );\n\t\t\t#endif\n\t\t\tenvMapColor.rgb = envMapTexelToLinear( envMapColor ).rgb;\n\t\t#elif defined( ENVMAP_TYPE_CUBE_UV )\n\t\t\tvec3 queryVec = vec3( flipEnvMap * worldNormal.x, worldNormal.yz );\n\t\t\tvec4 envMapColor = textureCubeUV( envMap, queryVec, 1.0 );\n\t\t#else\n\t\t\tvec4 envMapColor = vec4( 0.0 );\n\t\t#endif\n\t\treturn PI * envMapColor.rgb * envMapIntensity;\n\t}\n\tfloat getSpecularMIPLevel( const in float roughness, const in int maxMIPLevel ) {\n\t\tfloat maxMIPLevelScalar = float( maxMIPLevel );\n\t\tfloat sigma = PI * roughness * roughness / ( 1.0 + roughness );\n\t\tfloat desiredMIPLevel = maxMIPLevelScalar + log2( sigma );\n\t\treturn clamp( desiredMIPLevel, 0.0, maxMIPLevelScalar );\n\t}\n\tvec3 getLightProbeIndirectRadiance( const in vec3 viewDir, const in vec3 normal, const in float roughness, const in int maxMIPLevel ) {\n\t\t#ifdef ENVMAP_MODE_REFLECTION\n\t\t  vec3 reflectVec = reflect( -viewDir, normal );\n\t\t  reflectVec = normalize( mix( reflectVec, normal, roughness * roughness) );\n\t\t#else\n\t\t  vec3 reflectVec = refract( -viewDir, normal, refractionRatio );\n\t\t#endif\n\t\treflectVec = inverseTransformDirection( reflectVec, viewMatrix );\n\t\tfloat specularMIPLevel = getSpecularMIPLevel( roughness, maxMIPLevel );\n\t\t#ifdef ENVMAP_TYPE_CUBE\n\t\t\tvec3 queryReflectVec = vec3( flipEnvMap * reflectVec.x, reflectVec.yz );\n\t\t\t#ifdef TEXTURE_LOD_EXT\n\t\t\t\tvec4 envMapColor = textureCubeLodEXT( envMap, queryReflectVec, specularMIPLevel );\n\t\t\t#else\n\t\t\t\tvec4 envMapColor = textureCube( envMap, queryReflectVec, specularMIPLevel );\n\t\t\t#endif\n\t\t\tenvMapColor.rgb = envMapTexelToLinear( envMapColor ).rgb;\n\t\t#elif defined( ENVMAP_TYPE_CUBE_UV )\n\t\t\tvec3 queryReflectVec = vec3( flipEnvMap * reflectVec.x, reflectVec.yz );\n\t\t\tvec4 envMapColor = textureCubeUV( envMap, queryReflectVec, roughness );\n\t\t#elif defined( ENVMAP_TYPE_EQUIREC )\n\t\t\tvec2 sampleUV;\n\t\t\tsampleUV.y = asin( clamp( reflectVec.y, - 1.0, 1.0 ) ) * RECIPROCAL_PI + 0.5;\n\t\t\tsampleUV.x = atan( reflectVec.z, reflectVec.x ) * RECIPROCAL_PI2 + 0.5;\n\t\t\t#ifdef TEXTURE_LOD_EXT\n\t\t\t\tvec4 envMapColor = texture2DLodEXT( envMap, sampleUV, specularMIPLevel );\n\t\t\t#else\n\t\t\t\tvec4 envMapColor = texture2D( envMap, sampleUV, specularMIPLevel );\n\t\t\t#endif\n\t\t\tenvMapColor.rgb = envMapTexelToLinear( envMapColor ).rgb;\n\t\t#elif defined( ENVMAP_TYPE_SPHERE )\n\t\t\tvec3 reflectView = normalize( ( viewMatrix * vec4( reflectVec, 0.0 ) ).xyz + vec3( 0.0,0.0,1.0 ) );\n\t\t\t#ifdef TEXTURE_LOD_EXT\n\t\t\t\tvec4 envMapColor = texture2DLodEXT( envMap, reflectView.xy * 0.5 + 0.5, specularMIPLevel );\n\t\t\t#else\n\t\t\t\tvec4 envMapColor = texture2D( envMap, reflectView.xy * 0.5 + 0.5, specularMIPLevel );\n\t\t\t#endif\n\t\t\tenvMapColor.rgb = envMapTexelToLinear( envMapColor ).rgb;\n\t\t#endif\n\t\treturn envMapColor.rgb * envMapIntensity;\n\t}\n#endif";

    var lights_phong_fragment = "BlinnPhongMaterial material;\nmaterial.diffuseColor = diffuseColor.rgb;\nmaterial.specularColor = specular;\nmaterial.specularShininess = shininess;\nmaterial.specularStrength = specularStrength;";

    var lights_phong_pars_fragment = "varying vec3 vViewPosition;\n#ifndef FLAT_SHADED\n\tvarying vec3 vNormal;\n#endif\nstruct BlinnPhongMaterial {\n\tvec3\tdiffuseColor;\n\tvec3\tspecularColor;\n\tfloat\tspecularShininess;\n\tfloat\tspecularStrength;\n};\nvoid RE_Direct_BlinnPhong( const in IncidentLight directLight, const in GeometricContext geometry, const in BlinnPhongMaterial material, inout ReflectedLight reflectedLight ) {\n\t#ifdef TOON\n\t\tvec3 irradiance = getGradientIrradiance( geometry.normal, directLight.direction ) * directLight.color;\n\t#else\n\t\tfloat dotNL = saturate( dot( geometry.normal, directLight.direction ) );\n\t\tvec3 irradiance = dotNL * directLight.color;\n\t#endif\n\t#ifndef PHYSICALLY_CORRECT_LIGHTS\n\t\tirradiance *= PI;\n\t#endif\n\treflectedLight.directDiffuse += irradiance * BRDF_Diffuse_Lambert( material.diffuseColor );\n\treflectedLight.directSpecular += irradiance * BRDF_Specular_BlinnPhong( directLight, geometry, material.specularColor, material.specularShininess ) * material.specularStrength;\n}\nvoid RE_IndirectDiffuse_BlinnPhong( const in vec3 irradiance, const in GeometricContext geometry, const in BlinnPhongMaterial material, inout ReflectedLight reflectedLight ) {\n\treflectedLight.indirectDiffuse += irradiance * BRDF_Diffuse_Lambert( material.diffuseColor );\n}\n#define RE_Direct\t\t\t\tRE_Direct_BlinnPhong\n#define RE_IndirectDiffuse\t\tRE_IndirectDiffuse_BlinnPhong\n#define Material_LightProbeLOD( material )\t(0)";

    var lights_physical_fragment = "PhysicalMaterial material;\nmaterial.diffuseColor = diffuseColor.rgb * ( 1.0 - metalnessFactor );\nmaterial.specularRoughness = clamp( roughnessFactor, 0.04, 1.0 );\n#ifdef REFLECTIVITY\n\tmaterial.specularColor = mix( vec3( MAXIMUM_SPECULAR_COEFFICIENT * pow2( reflectivity ) ), diffuseColor.rgb, metalnessFactor );\n#else\n\tmaterial.specularColor = mix( vec3( DEFAULT_SPECULAR_COEFFICIENT ), diffuseColor.rgb, metalnessFactor );\n#endif\n#ifdef CLEARCOAT\n\tmaterial.clearcoat = saturate( clearcoat );\tmaterial.clearcoatRoughness = clamp( clearcoatRoughness, 0.04, 1.0 );\n#endif\n#ifdef USE_SHEEN\n\tmaterial.sheenColor = sheen;\n#endif";

    var lights_physical_pars_fragment = "struct PhysicalMaterial {\n\tvec3\tdiffuseColor;\n\tfloat\tspecularRoughness;\n\tvec3\tspecularColor;\n#ifdef CLEARCOAT\n\tfloat clearcoat;\n\tfloat clearcoatRoughness;\n#endif\n#ifdef USE_SHEEN\n\tvec3 sheenColor;\n#endif\n};\n#define MAXIMUM_SPECULAR_COEFFICIENT 0.16\n#define DEFAULT_SPECULAR_COEFFICIENT 0.04\nfloat clearcoatDHRApprox( const in float roughness, const in float dotNL ) {\n\treturn DEFAULT_SPECULAR_COEFFICIENT + ( 1.0 - DEFAULT_SPECULAR_COEFFICIENT ) * ( pow( 1.0 - dotNL, 5.0 ) * pow( 1.0 - roughness, 2.0 ) );\n}\n#if NUM_RECT_AREA_LIGHTS > 0\n\tvoid RE_Direct_RectArea_Physical( const in RectAreaLight rectAreaLight, const in GeometricContext geometry, const in PhysicalMaterial material, inout ReflectedLight reflectedLight ) {\n\t\tvec3 normal = geometry.normal;\n\t\tvec3 viewDir = geometry.viewDir;\n\t\tvec3 position = geometry.position;\n\t\tvec3 lightPos = rectAreaLight.position;\n\t\tvec3 halfWidth = rectAreaLight.halfWidth;\n\t\tvec3 halfHeight = rectAreaLight.halfHeight;\n\t\tvec3 lightColor = rectAreaLight.color;\n\t\tfloat roughness = material.specularRoughness;\n\t\tvec3 rectCoords[ 4 ];\n\t\trectCoords[ 0 ] = lightPos + halfWidth - halfHeight;\t\trectCoords[ 1 ] = lightPos - halfWidth - halfHeight;\n\t\trectCoords[ 2 ] = lightPos - halfWidth + halfHeight;\n\t\trectCoords[ 3 ] = lightPos + halfWidth + halfHeight;\n\t\tvec2 uv = LTC_Uv( normal, viewDir, roughness );\n\t\tvec4 t1 = texture2D( ltc_1, uv );\n\t\tvec4 t2 = texture2D( ltc_2, uv );\n\t\tmat3 mInv = mat3(\n\t\t\tvec3( t1.x, 0, t1.y ),\n\t\t\tvec3(    0, 1,    0 ),\n\t\t\tvec3( t1.z, 0, t1.w )\n\t\t);\n\t\tvec3 fresnel = ( material.specularColor * t2.x + ( vec3( 1.0 ) - material.specularColor ) * t2.y );\n\t\treflectedLight.directSpecular += lightColor * fresnel * LTC_Evaluate( normal, viewDir, position, mInv, rectCoords );\n\t\treflectedLight.directDiffuse += lightColor * material.diffuseColor * LTC_Evaluate( normal, viewDir, position, mat3( 1.0 ), rectCoords );\n\t}\n#endif\nvoid RE_Direct_Physical( const in IncidentLight directLight, const in GeometricContext geometry, const in PhysicalMaterial material, inout ReflectedLight reflectedLight ) {\n\tfloat dotNL = saturate( dot( geometry.normal, directLight.direction ) );\n\tvec3 irradiance = dotNL * directLight.color;\n\t#ifndef PHYSICALLY_CORRECT_LIGHTS\n\t\tirradiance *= PI;\n\t#endif\n\t#ifdef CLEARCOAT\n\t\tfloat ccDotNL = saturate( dot( geometry.clearcoatNormal, directLight.direction ) );\n\t\tvec3 ccIrradiance = ccDotNL * directLight.color;\n\t\t#ifndef PHYSICALLY_CORRECT_LIGHTS\n\t\t\tccIrradiance *= PI;\n\t\t#endif\n\t\tfloat clearcoatDHR = material.clearcoat * clearcoatDHRApprox( material.clearcoatRoughness, ccDotNL );\n\t\treflectedLight.directSpecular += ccIrradiance * material.clearcoat * BRDF_Specular_GGX( directLight, geometry.viewDir, geometry.clearcoatNormal, vec3( DEFAULT_SPECULAR_COEFFICIENT ), material.clearcoatRoughness );\n\t#else\n\t\tfloat clearcoatDHR = 0.0;\n\t#endif\n\t#ifdef USE_SHEEN\n\t\treflectedLight.directSpecular += ( 1.0 - clearcoatDHR ) * irradiance * BRDF_Specular_Sheen(\n\t\t\tmaterial.specularRoughness,\n\t\t\tdirectLight.direction,\n\t\t\tgeometry,\n\t\t\tmaterial.sheenColor\n\t\t);\n\t#else\n\t\treflectedLight.directSpecular += ( 1.0 - clearcoatDHR ) * irradiance * BRDF_Specular_GGX( directLight, geometry.viewDir, geometry.normal, material.specularColor, material.specularRoughness);\n\t#endif\n\treflectedLight.directDiffuse += ( 1.0 - clearcoatDHR ) * irradiance * BRDF_Diffuse_Lambert( material.diffuseColor );\n}\nvoid RE_IndirectDiffuse_Physical( const in vec3 irradiance, const in GeometricContext geometry, const in PhysicalMaterial material, inout ReflectedLight reflectedLight ) {\n\treflectedLight.indirectDiffuse += irradiance * BRDF_Diffuse_Lambert( material.diffuseColor );\n}\nvoid RE_IndirectSpecular_Physical( const in vec3 radiance, const in vec3 irradiance, const in vec3 clearcoatRadiance, const in GeometricContext geometry, const in PhysicalMaterial material, inout ReflectedLight reflectedLight) {\n\t#ifdef CLEARCOAT\n\t\tfloat ccDotNV = saturate( dot( geometry.clearcoatNormal, geometry.viewDir ) );\n\t\treflectedLight.indirectSpecular += clearcoatRadiance * material.clearcoat * BRDF_Specular_GGX_Environment( geometry.viewDir, geometry.clearcoatNormal, vec3( DEFAULT_SPECULAR_COEFFICIENT ), material.clearcoatRoughness );\n\t\tfloat ccDotNL = ccDotNV;\n\t\tfloat clearcoatDHR = material.clearcoat * clearcoatDHRApprox( material.clearcoatRoughness, ccDotNL );\n\t#else\n\t\tfloat clearcoatDHR = 0.0;\n\t#endif\n\tfloat clearcoatInv = 1.0 - clearcoatDHR;\n\tvec3 singleScattering = vec3( 0.0 );\n\tvec3 multiScattering = vec3( 0.0 );\n\tvec3 cosineWeightedIrradiance = irradiance * RECIPROCAL_PI;\n\tBRDF_Specular_Multiscattering_Environment( geometry, material.specularColor, material.specularRoughness, singleScattering, multiScattering );\n\tvec3 diffuse = material.diffuseColor * ( 1.0 - ( singleScattering + multiScattering ) );\n\treflectedLight.indirectSpecular += clearcoatInv * radiance * singleScattering;\n\treflectedLight.indirectDiffuse += multiScattering * cosineWeightedIrradiance;\n\treflectedLight.indirectDiffuse += diffuse * cosineWeightedIrradiance;\n}\n#define RE_Direct\t\t\t\tRE_Direct_Physical\n#define RE_Direct_RectArea\t\tRE_Direct_RectArea_Physical\n#define RE_IndirectDiffuse\t\tRE_IndirectDiffuse_Physical\n#define RE_IndirectSpecular\t\tRE_IndirectSpecular_Physical\nfloat computeSpecularOcclusion( const in float dotNV, const in float ambientOcclusion, const in float roughness ) {\n\treturn saturate( pow( dotNV + ambientOcclusion, exp2( - 16.0 * roughness - 1.0 ) ) - 1.0 + ambientOcclusion );\n}";

    var lights_fragment_begin = "\nGeometricContext geometry;\ngeometry.position = - vViewPosition;\ngeometry.normal = normal;\ngeometry.viewDir = normalize( vViewPosition );\n#ifdef CLEARCOAT\n\tgeometry.clearcoatNormal = clearcoatNormal;\n#endif\nIncidentLight directLight;\n#if ( NUM_POINT_LIGHTS > 0 ) && defined( RE_Direct )\n\tPointLight pointLight;\n\t#pragma unroll_loop\n\tfor ( int i = 0; i < NUM_POINT_LIGHTS; i ++ ) {\n\t\tpointLight = pointLights[ i ];\n\t\tgetPointDirectLightIrradiance( pointLight, geometry, directLight );\n\t\t#if defined( USE_SHADOWMAP ) && ( UNROLLED_LOOP_INDEX < NUM_POINT_LIGHT_SHADOWS )\n\t\tdirectLight.color *= all( bvec2( pointLight.shadow, directLight.visible ) ) ? getPointShadow( pointShadowMap[ i ], pointLight.shadowMapSize, pointLight.shadowBias, pointLight.shadowRadius, vPointShadowCoord[ i ], pointLight.shadowCameraNear, pointLight.shadowCameraFar ) : 1.0;\n\t\t#endif\n\t\tRE_Direct( directLight, geometry, material, reflectedLight );\n\t}\n#endif\n#if ( NUM_SPOT_LIGHTS > 0 ) && defined( RE_Direct )\n\tSpotLight spotLight;\n\t#pragma unroll_loop\n\tfor ( int i = 0; i < NUM_SPOT_LIGHTS; i ++ ) {\n\t\tspotLight = spotLights[ i ];\n\t\tgetSpotDirectLightIrradiance( spotLight, geometry, directLight );\n\t\t#if defined( USE_SHADOWMAP ) && ( UNROLLED_LOOP_INDEX < NUM_SPOT_LIGHT_SHADOWS )\n\t\tdirectLight.color *= all( bvec2( spotLight.shadow, directLight.visible ) ) ? getShadow( spotShadowMap[ i ], spotLight.shadowMapSize, spotLight.shadowBias, spotLight.shadowRadius, vSpotShadowCoord[ i ] ) : 1.0;\n\t\t#endif\n\t\tRE_Direct( directLight, geometry, material, reflectedLight );\n\t}\n#endif\n#if ( NUM_DIR_LIGHTS > 0 ) && defined( RE_Direct )\n\tDirectionalLight directionalLight;\n\t#pragma unroll_loop\n\tfor ( int i = 0; i < NUM_DIR_LIGHTS; i ++ ) {\n\t\tdirectionalLight = directionalLights[ i ];\n\t\tgetDirectionalDirectLightIrradiance( directionalLight, geometry, directLight );\n\t\t#if defined( USE_SHADOWMAP ) && ( UNROLLED_LOOP_INDEX < NUM_DIR_LIGHT_SHADOWS )\n\t\tdirectLight.color *= all( bvec2( directionalLight.shadow, directLight.visible ) ) ? getShadow( directionalShadowMap[ i ], directionalLight.shadowMapSize, directionalLight.shadowBias, directionalLight.shadowRadius, vDirectionalShadowCoord[ i ] ) : 1.0;\n\t\t#endif\n\t\tRE_Direct( directLight, geometry, material, reflectedLight );\n\t}\n#endif\n#if ( NUM_RECT_AREA_LIGHTS > 0 ) && defined( RE_Direct_RectArea )\n\tRectAreaLight rectAreaLight;\n\t#pragma unroll_loop\n\tfor ( int i = 0; i < NUM_RECT_AREA_LIGHTS; i ++ ) {\n\t\trectAreaLight = rectAreaLights[ i ];\n\t\tRE_Direct_RectArea( rectAreaLight, geometry, material, reflectedLight );\n\t}\n#endif\n#if defined( RE_IndirectDiffuse )\n\tvec3 iblIrradiance = vec3( 0.0 );\n\tvec3 irradiance = getAmbientLightIrradiance( ambientLightColor );\n\tirradiance += getLightProbeIrradiance( lightProbe, geometry );\n\t#if ( NUM_HEMI_LIGHTS > 0 )\n\t\t#pragma unroll_loop\n\t\tfor ( int i = 0; i < NUM_HEMI_LIGHTS; i ++ ) {\n\t\t\tirradiance += getHemisphereLightIrradiance( hemisphereLights[ i ], geometry );\n\t\t}\n\t#endif\n#endif\n#if defined( RE_IndirectSpecular )\n\tvec3 radiance = vec3( 0.0 );\n\tvec3 clearcoatRadiance = vec3( 0.0 );\n#endif";

    var lights_fragment_maps = "#if defined( RE_IndirectDiffuse )\n\t#ifdef USE_LIGHTMAP\n\t\tvec3 lightMapIrradiance = texture2D( lightMap, vUv2 ).xyz * lightMapIntensity;\n\t\t#ifndef PHYSICALLY_CORRECT_LIGHTS\n\t\t\tlightMapIrradiance *= PI;\n\t\t#endif\n\t\tirradiance += lightMapIrradiance;\n\t#endif\n\t#if defined( USE_ENVMAP ) && defined( STANDARD ) && defined( ENVMAP_TYPE_CUBE_UV )\n\t\tiblIrradiance += getLightProbeIndirectIrradiance( geometry, maxMipLevel );\n\t#endif\n#endif\n#if defined( USE_ENVMAP ) && defined( RE_IndirectSpecular )\n\tradiance += getLightProbeIndirectRadiance( geometry.viewDir, geometry.normal, material.specularRoughness, maxMipLevel );\n\t#ifdef CLEARCOAT\n\t\tclearcoatRadiance += getLightProbeIndirectRadiance( geometry.viewDir, geometry.clearcoatNormal, material.clearcoatRoughness, maxMipLevel );\n\t#endif\n#endif";

    var lights_fragment_end = "#if defined( RE_IndirectDiffuse )\n\tRE_IndirectDiffuse( irradiance, geometry, material, reflectedLight );\n#endif\n#if defined( RE_IndirectSpecular )\n\tRE_IndirectSpecular( radiance, iblIrradiance, clearcoatRadiance, geometry, material, reflectedLight );\n#endif";

    var logdepthbuf_fragment = "#if defined( USE_LOGDEPTHBUF ) && defined( USE_LOGDEPTHBUF_EXT )\n\tgl_FragDepthEXT = log2( vFragDepth ) * logDepthBufFC * 0.5;\n#endif";

    var logdepthbuf_pars_fragment = "#if defined( USE_LOGDEPTHBUF ) && defined( USE_LOGDEPTHBUF_EXT )\n\tuniform float logDepthBufFC;\n\tvarying float vFragDepth;\n#endif";

    var logdepthbuf_pars_vertex = "#ifdef USE_LOGDEPTHBUF\n\t#ifdef USE_LOGDEPTHBUF_EXT\n\t\tvarying float vFragDepth;\n\t#else\n\t\tuniform float logDepthBufFC;\n\t#endif\n#endif";

    var logdepthbuf_vertex = "#ifdef USE_LOGDEPTHBUF\n\t#ifdef USE_LOGDEPTHBUF_EXT\n\t\tvFragDepth = 1.0 + gl_Position.w;\n\t#else\n\t\tgl_Position.z = log2( max( EPSILON, gl_Position.w + 1.0 ) ) * logDepthBufFC - 1.0;\n\t\tgl_Position.z *= gl_Position.w;\n\t#endif\n#endif";

    var map_fragment = "#ifdef USE_MAP\n\tvec4 texelColor = texture2D( map, vUv );\n\ttexelColor = mapTexelToLinear( texelColor );\n\tdiffuseColor *= texelColor;\n#endif";

    var map_pars_fragment = "#ifdef USE_MAP\n\tuniform sampler2D map;\n#endif";

    var map_particle_fragment = "#ifdef USE_MAP\n\tvec2 uv = ( uvTransform * vec3( gl_PointCoord.x, 1.0 - gl_PointCoord.y, 1 ) ).xy;\n\tvec4 mapTexel = texture2D( map, uv );\n\tdiffuseColor *= mapTexelToLinear( mapTexel );\n#endif";

    var map_particle_pars_fragment = "#ifdef USE_MAP\n\tuniform mat3 uvTransform;\n\tuniform sampler2D map;\n#endif";

    var metalnessmap_fragment = "float metalnessFactor = metalness;\n#ifdef USE_METALNESSMAP\n\tvec4 texelMetalness = texture2D( metalnessMap, vUv );\n\tmetalnessFactor *= texelMetalness.b;\n#endif";

    var metalnessmap_pars_fragment = "#ifdef USE_METALNESSMAP\n\tuniform sampler2D metalnessMap;\n#endif";

    var morphnormal_vertex = "#ifdef USE_MORPHNORMALS\n\tobjectNormal += ( morphNormal0 - normal ) * morphTargetInfluences[ 0 ];\n\tobjectNormal += ( morphNormal1 - normal ) * morphTargetInfluences[ 1 ];\n\tobjectNormal += ( morphNormal2 - normal ) * morphTargetInfluences[ 2 ];\n\tobjectNormal += ( morphNormal3 - normal ) * morphTargetInfluences[ 3 ];\n#endif";

    var morphtarget_pars_vertex = "#ifdef USE_MORPHTARGETS\n\t#ifndef USE_MORPHNORMALS\n\tuniform float morphTargetInfluences[ 8 ];\n\t#else\n\tuniform float morphTargetInfluences[ 4 ];\n\t#endif\n#endif";

    var morphtarget_vertex = "#ifdef USE_MORPHTARGETS\n\ttransformed += ( morphTarget0 - position ) * morphTargetInfluences[ 0 ];\n\ttransformed += ( morphTarget1 - position ) * morphTargetInfluences[ 1 ];\n\ttransformed += ( morphTarget2 - position ) * morphTargetInfluences[ 2 ];\n\ttransformed += ( morphTarget3 - position ) * morphTargetInfluences[ 3 ];\n\t#ifndef USE_MORPHNORMALS\n\ttransformed += ( morphTarget4 - position ) * morphTargetInfluences[ 4 ];\n\ttransformed += ( morphTarget5 - position ) * morphTargetInfluences[ 5 ];\n\ttransformed += ( morphTarget6 - position ) * morphTargetInfluences[ 6 ];\n\ttransformed += ( morphTarget7 - position ) * morphTargetInfluences[ 7 ];\n\t#endif\n#endif";

    var normal_fragment_begin = "#ifdef FLAT_SHADED\n\tvec3 fdx = vec3( dFdx( vViewPosition.x ), dFdx( vViewPosition.y ), dFdx( vViewPosition.z ) );\n\tvec3 fdy = vec3( dFdy( vViewPosition.x ), dFdy( vViewPosition.y ), dFdy( vViewPosition.z ) );\n\tvec3 normal = normalize( cross( fdx, fdy ) );\n#else\n\tvec3 normal = normalize( vNormal );\n\t#ifdef DOUBLE_SIDED\n\t\tnormal = normal * ( float( gl_FrontFacing ) * 2.0 - 1.0 );\n\t#endif\n\t#ifdef USE_TANGENT\n\t\tvec3 tangent = normalize( vTangent );\n\t\tvec3 bitangent = normalize( vBitangent );\n\t\t#ifdef DOUBLE_SIDED\n\t\t\ttangent = tangent * ( float( gl_FrontFacing ) * 2.0 - 1.0 );\n\t\t\tbitangent = bitangent * ( float( gl_FrontFacing ) * 2.0 - 1.0 );\n\t\t#endif\n\t#endif\n#endif\nvec3 geometryNormal = normal;";

    var normal_fragment_maps = "#ifdef OBJECTSPACE_NORMALMAP\n\tnormal = texture2D( normalMap, vUv ).xyz * 2.0 - 1.0;\n\t#ifdef FLIP_SIDED\n\t\tnormal = - normal;\n\t#endif\n\t#ifdef DOUBLE_SIDED\n\t\tnormal = normal * ( float( gl_FrontFacing ) * 2.0 - 1.0 );\n\t#endif\n\tnormal = normalize( normalMatrix * normal );\n#elif defined( TANGENTSPACE_NORMALMAP )\n\t#ifdef USE_TANGENT\n\t\tmat3 vTBN = mat3( tangent, bitangent, normal );\n\t\tvec3 mapN = texture2D( normalMap, vUv ).xyz * 2.0 - 1.0;\n\t\tmapN.xy = normalScale * mapN.xy;\n\t\tnormal = normalize( vTBN * mapN );\n\t#else\n\t\tnormal = perturbNormal2Arb( -vViewPosition, normal, normalScale, normalMap );\n\t#endif\n#elif defined( USE_BUMPMAP )\n\tnormal = perturbNormalArb( -vViewPosition, normal, dHdxy_fwd() );\n#endif";

    var normalmap_pars_fragment = "#ifdef USE_NORMALMAP\n\tuniform sampler2D normalMap;\n\tuniform vec2 normalScale;\n#endif\n#ifdef OBJECTSPACE_NORMALMAP\n\tuniform mat3 normalMatrix;\n#endif\n#if ! defined ( USE_TANGENT ) && ( defined ( TANGENTSPACE_NORMALMAP ) || defined ( USE_CLEARCOAT_NORMALMAP ) )\n\tvec3 perturbNormal2Arb( vec3 eye_pos, vec3 surf_norm, vec2 normalScale, in sampler2D normalMap ) {\n\t\tvec3 q0 = vec3( dFdx( eye_pos.x ), dFdx( eye_pos.y ), dFdx( eye_pos.z ) );\n\t\tvec3 q1 = vec3( dFdy( eye_pos.x ), dFdy( eye_pos.y ), dFdy( eye_pos.z ) );\n\t\tvec2 st0 = dFdx( vUv.st );\n\t\tvec2 st1 = dFdy( vUv.st );\n\t\tfloat scale = sign( st1.t * st0.s - st0.t * st1.s );\n\t\tvec3 S = normalize( ( q0 * st1.t - q1 * st0.t ) * scale );\n\t\tvec3 T = normalize( ( - q0 * st1.s + q1 * st0.s ) * scale );\n\t\tvec3 N = normalize( surf_norm );\n\t\tvec3 mapN = texture2D( normalMap, vUv ).xyz * 2.0 - 1.0;\n\t\tmapN.xy *= normalScale;\n\t\t#ifdef DOUBLE_SIDED\n\t\t\tvec3 NfromST = cross( S, T );\n\t\t\tif( dot( NfromST, N ) > 0.0 ) {\n\t\t\t\tS *= -1.0;\n\t\t\t\tT *= -1.0;\n\t\t\t}\n\t\t#else\n\t\t\tmapN.xy *= ( float( gl_FrontFacing ) * 2.0 - 1.0 );\n\t\t#endif\n\t\tmat3 tsn = mat3( S, T, N );\n\t\treturn normalize( tsn * mapN );\n\t}\n#endif";

    var clearcoat_normal_fragment_begin = "#ifdef CLEARCOAT\n\tvec3 clearcoatNormal = geometryNormal;\n#endif";

    var clearcoat_normal_fragment_maps = "#ifdef USE_CLEARCOAT_NORMALMAP\n\t#ifdef USE_TANGENT\n\t\tmat3 vTBN = mat3( tangent, bitangent, clearcoatNormal );\n\t\tvec3 mapN = texture2D( normalMap, vUv ).xyz * 2.0 - 1.0;\n\t\tmapN.xy = clearcoatNormalScale * mapN.xy;\n\t\tclearcoatNormal = normalize( vTBN * mapN );\n\t#else\n\t\tclearcoatNormal = perturbNormal2Arb( - vViewPosition, clearcoatNormal, clearcoatNormalScale, clearcoatNormalMap );\n\t#endif\n#endif";

    var clearcoat_normalmap_pars_fragment = "#ifdef USE_CLEARCOAT_NORMALMAP\n\tuniform sampler2D clearcoatNormalMap;\n\tuniform vec2 clearcoatNormalScale;\n#endif";

    var packing = "vec3 packNormalToRGB( const in vec3 normal ) {\n\treturn normalize( normal ) * 0.5 + 0.5;\n}\nvec3 unpackRGBToNormal( const in vec3 rgb ) {\n\treturn 2.0 * rgb.xyz - 1.0;\n}\nconst float PackUpscale = 256. / 255.;const float UnpackDownscale = 255. / 256.;\nconst vec3 PackFactors = vec3( 256. * 256. * 256., 256. * 256.,  256. );\nconst vec4 UnpackFactors = UnpackDownscale / vec4( PackFactors, 1. );\nconst float ShiftRight8 = 1. / 256.;\nvec4 packDepthToRGBA( const in float v ) {\n\tvec4 r = vec4( fract( v * PackFactors ), v );\n\tr.yzw -= r.xyz * ShiftRight8;\treturn r * PackUpscale;\n}\nfloat unpackRGBAToDepth( const in vec4 v ) {\n\treturn dot( v, UnpackFactors );\n}\nvec4 encodeHalfRGBA ( vec2 v ) {\n\tvec4 encoded = vec4( 0.0 );\n\tconst vec2 offset = vec2( 1.0 / 255.0, 0.0 );\n\tencoded.xy = vec2( v.x, fract( v.x * 255.0 ) );\n\tencoded.xy = encoded.xy - ( encoded.yy * offset );\n\tencoded.zw = vec2( v.y, fract( v.y * 255.0 ) );\n\tencoded.zw = encoded.zw - ( encoded.ww * offset );\n\treturn encoded;\n}\nvec2 decodeHalfRGBA( vec4 v ) {\n\treturn vec2( v.x + ( v.y / 255.0 ), v.z + ( v.w / 255.0 ) );\n}\nfloat viewZToOrthographicDepth( const in float viewZ, const in float near, const in float far ) {\n\treturn ( viewZ + near ) / ( near - far );\n}\nfloat orthographicDepthToViewZ( const in float linearClipZ, const in float near, const in float far ) {\n\treturn linearClipZ * ( near - far ) - near;\n}\nfloat viewZToPerspectiveDepth( const in float viewZ, const in float near, const in float far ) {\n\treturn (( near + viewZ ) * far ) / (( far - near ) * viewZ );\n}\nfloat perspectiveDepthToViewZ( const in float invClipZ, const in float near, const in float far ) {\n\treturn ( near * far ) / ( ( far - near ) * invClipZ - far );\n}";

    var premultiplied_alpha_fragment = "#ifdef PREMULTIPLIED_ALPHA\n\tgl_FragColor.rgb *= gl_FragColor.a;\n#endif";

    var project_vertex = "vec4 mvPosition = modelViewMatrix * vec4( transformed, 1.0 );\ngl_Position = projectionMatrix * mvPosition;";

    var dithering_fragment = "#ifdef DITHERING\n\tgl_FragColor.rgb = dithering( gl_FragColor.rgb );\n#endif";

    var dithering_pars_fragment = "#ifdef DITHERING\n\tvec3 dithering( vec3 color ) {\n\t\tfloat grid_position = rand( gl_FragCoord.xy );\n\t\tvec3 dither_shift_RGB = vec3( 0.25 / 255.0, -0.25 / 255.0, 0.25 / 255.0 );\n\t\tdither_shift_RGB = mix( 2.0 * dither_shift_RGB, -2.0 * dither_shift_RGB, grid_position );\n\t\treturn color + dither_shift_RGB;\n\t}\n#endif";

    var roughnessmap_fragment = "float roughnessFactor = roughness;\n#ifdef USE_ROUGHNESSMAP\n\tvec4 texelRoughness = texture2D( roughnessMap, vUv );\n\troughnessFactor *= texelRoughness.g;\n#endif";

    var roughnessmap_pars_fragment = "#ifdef USE_ROUGHNESSMAP\n\tuniform sampler2D roughnessMap;\n#endif";

    var shadowmap_pars_fragment = "#ifdef USE_SHADOWMAP\n\t#if NUM_DIR_LIGHT_SHADOWS > 0\n\t\tuniform sampler2D directionalShadowMap[ NUM_DIR_LIGHT_SHADOWS ];\n\t\tvarying vec4 vDirectionalShadowCoord[ NUM_DIR_LIGHT_SHADOWS ];\n\t#endif\n\t#if NUM_SPOT_LIGHT_SHADOWS > 0\n\t\tuniform sampler2D spotShadowMap[ NUM_SPOT_LIGHT_SHADOWS ];\n\t\tvarying vec4 vSpotShadowCoord[ NUM_SPOT_LIGHT_SHADOWS ];\n\t#endif\n\t#if NUM_POINT_LIGHT_SHADOWS > 0\n\t\tuniform sampler2D pointShadowMap[ NUM_POINT_LIGHT_SHADOWS ];\n\t\tvarying vec4 vPointShadowCoord[ NUM_POINT_LIGHT_SHADOWS ];\n\t#endif\n\tfloat texture2DCompare( sampler2D depths, vec2 uv, float compare ) {\n\t\treturn step( compare, unpackRGBAToDepth( texture2D( depths, uv ) ) );\n\t}\n\tvec2 texture2DDistribution( sampler2D shadow, vec2 uv ) {\n\t\treturn decodeHalfRGBA( texture2D( shadow, uv ) );\n\t}\n\tfloat VSMShadow (sampler2D shadow, vec2 uv, float compare ){\n\t\tfloat occlusion = 1.0;\n\t\tvec2 distribution = texture2DDistribution( shadow, uv );\n\t\tfloat hard_shadow = step( compare , distribution.x );\n\t\tif (hard_shadow != 1.0 ) {\n\t\t\tfloat distance = compare - distribution.x ;\n\t\t\tfloat variance = max( 0.00000, distribution.y * distribution.y );\n\t\t\tfloat softness_probability = variance / (variance + distance * distance );\t\t\tsoftness_probability = clamp( ( softness_probability - 0.3 ) / ( 0.95 - 0.3 ), 0.0, 1.0 );\t\t\tocclusion = clamp( max( hard_shadow, softness_probability ), 0.0, 1.0 );\n\t\t}\n\t\treturn occlusion;\n\t}\n\tfloat texture2DShadowLerp( sampler2D depths, vec2 size, vec2 uv, float compare ) {\n\t\tconst vec2 offset = vec2( 0.0, 1.0 );\n\t\tvec2 texelSize = vec2( 1.0 ) / size;\n\t\tvec2 centroidUV = ( floor( uv * size - 0.5 ) + 0.5 ) * texelSize;\n\t\tfloat lb = texture2DCompare( depths, centroidUV + texelSize * offset.xx, compare );\n\t\tfloat lt = texture2DCompare( depths, centroidUV + texelSize * offset.xy, compare );\n\t\tfloat rb = texture2DCompare( depths, centroidUV + texelSize * offset.yx, compare );\n\t\tfloat rt = texture2DCompare( depths, centroidUV + texelSize * offset.yy, compare );\n\t\tvec2 f = fract( uv * size + 0.5 );\n\t\tfloat a = mix( lb, lt, f.y );\n\t\tfloat b = mix( rb, rt, f.y );\n\t\tfloat c = mix( a, b, f.x );\n\t\treturn c;\n\t}\n\tfloat getShadow( sampler2D shadowMap, vec2 shadowMapSize, float shadowBias, float shadowRadius, vec4 shadowCoord ) {\n\t\tfloat shadow = 1.0;\n\t\tshadowCoord.xyz /= shadowCoord.w;\n\t\tshadowCoord.z += shadowBias;\n\t\tbvec4 inFrustumVec = bvec4 ( shadowCoord.x >= 0.0, shadowCoord.x <= 1.0, shadowCoord.y >= 0.0, shadowCoord.y <= 1.0 );\n\t\tbool inFrustum = all( inFrustumVec );\n\t\tbvec2 frustumTestVec = bvec2( inFrustum, shadowCoord.z <= 1.0 );\n\t\tbool frustumTest = all( frustumTestVec );\n\t\tif ( frustumTest ) {\n\t\t#if defined( SHADOWMAP_TYPE_PCF )\n\t\t\tvec2 texelSize = vec2( 1.0 ) / shadowMapSize;\n\t\t\tfloat dx0 = - texelSize.x * shadowRadius;\n\t\t\tfloat dy0 = - texelSize.y * shadowRadius;\n\t\t\tfloat dx1 = + texelSize.x * shadowRadius;\n\t\t\tfloat dy1 = + texelSize.y * shadowRadius;\n\t\t\tfloat dx2 = dx0 / 2.0;\n\t\t\tfloat dy2 = dy0 / 2.0;\n\t\t\tfloat dx3 = dx1 / 2.0;\n\t\t\tfloat dy3 = dy1 / 2.0;\n\t\t\tshadow = (\n\t\t\t\ttexture2DCompare( shadowMap, shadowCoord.xy + vec2( dx0, dy0 ), shadowCoord.z ) +\n\t\t\t\ttexture2DCompare( shadowMap, shadowCoord.xy + vec2( 0.0, dy0 ), shadowCoord.z ) +\n\t\t\t\ttexture2DCompare( shadowMap, shadowCoord.xy + vec2( dx1, dy0 ), shadowCoord.z ) +\n\t\t\t\ttexture2DCompare( shadowMap, shadowCoord.xy + vec2( dx2, dy2 ), shadowCoord.z ) +\n\t\t\t\ttexture2DCompare( shadowMap, shadowCoord.xy + vec2( 0.0, dy2 ), shadowCoord.z ) +\n\t\t\t\ttexture2DCompare( shadowMap, shadowCoord.xy + vec2( dx3, dy2 ), shadowCoord.z ) +\n\t\t\t\ttexture2DCompare( shadowMap, shadowCoord.xy + vec2( dx0, 0.0 ), shadowCoord.z ) +\n\t\t\t\ttexture2DCompare( shadowMap, shadowCoord.xy + vec2( dx2, 0.0 ), shadowCoord.z ) +\n\t\t\t\ttexture2DCompare( shadowMap, shadowCoord.xy, shadowCoord.z ) +\n\t\t\t\ttexture2DCompare( shadowMap, shadowCoord.xy + vec2( dx3, 0.0 ), shadowCoord.z ) +\n\t\t\t\ttexture2DCompare( shadowMap, shadowCoord.xy + vec2( dx1, 0.0 ), shadowCoord.z ) +\n\t\t\t\ttexture2DCompare( shadowMap, shadowCoord.xy + vec2( dx2, dy3 ), shadowCoord.z ) +\n\t\t\t\ttexture2DCompare( shadowMap, shadowCoord.xy + vec2( 0.0, dy3 ), shadowCoord.z ) +\n\t\t\t\ttexture2DCompare( shadowMap, shadowCoord.xy + vec2( dx3, dy3 ), shadowCoord.z ) +\n\t\t\t\ttexture2DCompare( shadowMap, shadowCoord.xy + vec2( dx0, dy1 ), shadowCoord.z ) +\n\t\t\t\ttexture2DCompare( shadowMap, shadowCoord.xy + vec2( 0.0, dy1 ), shadowCoord.z ) +\n\t\t\t\ttexture2DCompare( shadowMap, shadowCoord.xy + vec2( dx1, dy1 ), shadowCoord.z )\n\t\t\t) * ( 1.0 / 17.0 );\n\t\t#elif defined( SHADOWMAP_TYPE_PCF_SOFT )\n\t\t\tvec2 texelSize = vec2( 1.0 ) / shadowMapSize;\n\t\t\tfloat dx0 = - texelSize.x * shadowRadius;\n\t\t\tfloat dy0 = - texelSize.y * shadowRadius;\n\t\t\tfloat dx1 = + texelSize.x * shadowRadius;\n\t\t\tfloat dy1 = + texelSize.y * shadowRadius;\n\t\t\tshadow = (\n\t\t\t\ttexture2DShadowLerp( shadowMap, shadowMapSize, shadowCoord.xy + vec2( dx0, dy0 ), shadowCoord.z ) +\n\t\t\t\ttexture2DShadowLerp( shadowMap, shadowMapSize, shadowCoord.xy + vec2( 0.0, dy0 ), shadowCoord.z ) +\n\t\t\t\ttexture2DShadowLerp( shadowMap, shadowMapSize, shadowCoord.xy + vec2( dx1, dy0 ), shadowCoord.z ) +\n\t\t\t\ttexture2DShadowLerp( shadowMap, shadowMapSize, shadowCoord.xy + vec2( dx0, 0.0 ), shadowCoord.z ) +\n\t\t\t\ttexture2DShadowLerp( shadowMap, shadowMapSize, shadowCoord.xy, shadowCoord.z ) +\n\t\t\t\ttexture2DShadowLerp( shadowMap, shadowMapSize, shadowCoord.xy + vec2( dx1, 0.0 ), shadowCoord.z ) +\n\t\t\t\ttexture2DShadowLerp( shadowMap, shadowMapSize, shadowCoord.xy + vec2( dx0, dy1 ), shadowCoord.z ) +\n\t\t\t\ttexture2DShadowLerp( shadowMap, shadowMapSize, shadowCoord.xy + vec2( 0.0, dy1 ), shadowCoord.z ) +\n\t\t\t\ttexture2DShadowLerp( shadowMap, shadowMapSize, shadowCoord.xy + vec2( dx1, dy1 ), shadowCoord.z )\n\t\t\t) * ( 1.0 / 9.0 );\n\t\t#elif defined( SHADOWMAP_TYPE_VSM )\n\t\t\tshadow = VSMShadow( shadowMap, shadowCoord.xy, shadowCoord.z );\n\t\t#else\n\t\t\tshadow = texture2DCompare( shadowMap, shadowCoord.xy, shadowCoord.z );\n\t\t#endif\n\t\t}\n\t\treturn shadow;\n\t}\n\tvec2 cubeToUV( vec3 v, float texelSizeY ) {\n\t\tvec3 absV = abs( v );\n\t\tfloat scaleToCube = 1.0 / max( absV.x, max( absV.y, absV.z ) );\n\t\tabsV *= scaleToCube;\n\t\tv *= scaleToCube * ( 1.0 - 2.0 * texelSizeY );\n\t\tvec2 planar = v.xy;\n\t\tfloat almostATexel = 1.5 * texelSizeY;\n\t\tfloat almostOne = 1.0 - almostATexel;\n\t\tif ( absV.z >= almostOne ) {\n\t\t\tif ( v.z > 0.0 )\n\t\t\t\tplanar.x = 4.0 - v.x;\n\t\t} else if ( absV.x >= almostOne ) {\n\t\t\tfloat signX = sign( v.x );\n\t\t\tplanar.x = v.z * signX + 2.0 * signX;\n\t\t} else if ( absV.y >= almostOne ) {\n\t\t\tfloat signY = sign( v.y );\n\t\t\tplanar.x = v.x + 2.0 * signY + 2.0;\n\t\t\tplanar.y = v.z * signY - 2.0;\n\t\t}\n\t\treturn vec2( 0.125, 0.25 ) * planar + vec2( 0.375, 0.75 );\n\t}\n\tfloat getPointShadow( sampler2D shadowMap, vec2 shadowMapSize, float shadowBias, float shadowRadius, vec4 shadowCoord, float shadowCameraNear, float shadowCameraFar ) {\n\t\tvec2 texelSize = vec2( 1.0 ) / ( shadowMapSize * vec2( 4.0, 2.0 ) );\n\t\tvec3 lightToPosition = shadowCoord.xyz;\n\t\tfloat dp = ( length( lightToPosition ) - shadowCameraNear ) / ( shadowCameraFar - shadowCameraNear );\t\tdp += shadowBias;\n\t\tvec3 bd3D = normalize( lightToPosition );\n\t\t#if defined( SHADOWMAP_TYPE_PCF ) || defined( SHADOWMAP_TYPE_PCF_SOFT ) || defined( SHADOWMAP_TYPE_VSM )\n\t\t\tvec2 offset = vec2( - 1, 1 ) * shadowRadius * texelSize.y;\n\t\t\treturn (\n\t\t\t\ttexture2DCompare( shadowMap, cubeToUV( bd3D + offset.xyy, texelSize.y ), dp ) +\n\t\t\t\ttexture2DCompare( shadowMap, cubeToUV( bd3D + offset.yyy, texelSize.y ), dp ) +\n\t\t\t\ttexture2DCompare( shadowMap, cubeToUV( bd3D + offset.xyx, texelSize.y ), dp ) +\n\t\t\t\ttexture2DCompare( shadowMap, cubeToUV( bd3D + offset.yyx, texelSize.y ), dp ) +\n\t\t\t\ttexture2DCompare( shadowMap, cubeToUV( bd3D, texelSize.y ), dp ) +\n\t\t\t\ttexture2DCompare( shadowMap, cubeToUV( bd3D + offset.xxy, texelSize.y ), dp ) +\n\t\t\t\ttexture2DCompare( shadowMap, cubeToUV( bd3D + offset.yxy, texelSize.y ), dp ) +\n\t\t\t\ttexture2DCompare( shadowMap, cubeToUV( bd3D + offset.xxx, texelSize.y ), dp ) +\n\t\t\t\ttexture2DCompare( shadowMap, cubeToUV( bd3D + offset.yxx, texelSize.y ), dp )\n\t\t\t) * ( 1.0 / 9.0 );\n\t\t#else\n\t\t\treturn texture2DCompare( shadowMap, cubeToUV( bd3D, texelSize.y ), dp );\n\t\t#endif\n\t}\n#endif";

    var shadowmap_pars_vertex = "#ifdef USE_SHADOWMAP\n\t#if NUM_DIR_LIGHT_SHADOWS > 0\n\t\tuniform mat4 directionalShadowMatrix[ NUM_DIR_LIGHT_SHADOWS ];\n\t\tvarying vec4 vDirectionalShadowCoord[ NUM_DIR_LIGHT_SHADOWS ];\n\t#endif\n\t#if NUM_SPOT_LIGHT_SHADOWS > 0\n\t\tuniform mat4 spotShadowMatrix[ NUM_SPOT_LIGHT_SHADOWS ];\n\t\tvarying vec4 vSpotShadowCoord[ NUM_SPOT_LIGHT_SHADOWS ];\n\t#endif\n\t#if NUM_POINT_LIGHT_SHADOWS > 0\n\t\tuniform mat4 pointShadowMatrix[ NUM_POINT_LIGHT_SHADOWS ];\n\t\tvarying vec4 vPointShadowCoord[ NUM_POINT_LIGHT_SHADOWS ];\n\t#endif\n#endif";

    var shadowmap_vertex = "#ifdef USE_SHADOWMAP\n\t#if NUM_DIR_LIGHT_SHADOWS > 0\n\t#pragma unroll_loop\n\tfor ( int i = 0; i < NUM_DIR_LIGHT_SHADOWS; i ++ ) {\n\t\tvDirectionalShadowCoord[ i ] = directionalShadowMatrix[ i ] * worldPosition;\n\t}\n\t#endif\n\t#if NUM_SPOT_LIGHT_SHADOWS > 0\n\t#pragma unroll_loop\n\tfor ( int i = 0; i < NUM_SPOT_LIGHT_SHADOWS; i ++ ) {\n\t\tvSpotShadowCoord[ i ] = spotShadowMatrix[ i ] * worldPosition;\n\t}\n\t#endif\n\t#if NUM_POINT_LIGHT_SHADOWS > 0\n\t#pragma unroll_loop\n\tfor ( int i = 0; i < NUM_POINT_LIGHT_SHADOWS; i ++ ) {\n\t\tvPointShadowCoord[ i ] = pointShadowMatrix[ i ] * worldPosition;\n\t}\n\t#endif\n#endif";

    var shadowmask_pars_fragment = "float getShadowMask() {\n\tfloat shadow = 1.0;\n\t#ifdef USE_SHADOWMAP\n\t#if NUM_DIR_LIGHT_SHADOWS > 0\n\tDirectionalLight directionalLight;\n\t#pragma unroll_loop\n\tfor ( int i = 0; i < NUM_DIR_LIGHT_SHADOWS; i ++ ) {\n\t\tdirectionalLight = directionalLights[ i ];\n\t\tshadow *= bool( directionalLight.shadow ) ? getShadow( directionalShadowMap[ i ], directionalLight.shadowMapSize, directionalLight.shadowBias, directionalLight.shadowRadius, vDirectionalShadowCoord[ i ] ) : 1.0;\n\t}\n\t#endif\n\t#if NUM_SPOT_LIGHT_SHADOWS > 0\n\tSpotLight spotLight;\n\t#pragma unroll_loop\n\tfor ( int i = 0; i < NUM_SPOT_LIGHT_SHADOWS; i ++ ) {\n\t\tspotLight = spotLights[ i ];\n\t\tshadow *= bool( spotLight.shadow ) ? getShadow( spotShadowMap[ i ], spotLight.shadowMapSize, spotLight.shadowBias, spotLight.shadowRadius, vSpotShadowCoord[ i ] ) : 1.0;\n\t}\n\t#endif\n\t#if NUM_POINT_LIGHT_SHADOWS > 0\n\tPointLight pointLight;\n\t#pragma unroll_loop\n\tfor ( int i = 0; i < NUM_POINT_LIGHT_SHADOWS; i ++ ) {\n\t\tpointLight = pointLights[ i ];\n\t\tshadow *= bool( pointLight.shadow ) ? getPointShadow( pointShadowMap[ i ], pointLight.shadowMapSize, pointLight.shadowBias, pointLight.shadowRadius, vPointShadowCoord[ i ], pointLight.shadowCameraNear, pointLight.shadowCameraFar ) : 1.0;\n\t}\n\t#endif\n\t#endif\n\treturn shadow;\n}";

    var skinbase_vertex = "#ifdef USE_SKINNING\n\tmat4 boneMatX = getBoneMatrix( skinIndex.x );\n\tmat4 boneMatY = getBoneMatrix( skinIndex.y );\n\tmat4 boneMatZ = getBoneMatrix( skinIndex.z );\n\tmat4 boneMatW = getBoneMatrix( skinIndex.w );\n#endif";

    var skinning_pars_vertex = "#ifdef USE_SKINNING\n\tuniform mat4 bindMatrix;\n\tuniform mat4 bindMatrixInverse;\n\t#ifdef BONE_TEXTURE\n\t\tuniform highp sampler2D boneTexture;\n\t\tuniform int boneTextureSize;\n\t\tmat4 getBoneMatrix( const in float i ) {\n\t\t\tfloat j = i * 4.0;\n\t\t\tfloat x = mod( j, float( boneTextureSize ) );\n\t\t\tfloat y = floor( j / float( boneTextureSize ) );\n\t\t\tfloat dx = 1.0 / float( boneTextureSize );\n\t\t\tfloat dy = 1.0 / float( boneTextureSize );\n\t\t\ty = dy * ( y + 0.5 );\n\t\t\tvec4 v1 = texture2D( boneTexture, vec2( dx * ( x + 0.5 ), y ) );\n\t\t\tvec4 v2 = texture2D( boneTexture, vec2( dx * ( x + 1.5 ), y ) );\n\t\t\tvec4 v3 = texture2D( boneTexture, vec2( dx * ( x + 2.5 ), y ) );\n\t\t\tvec4 v4 = texture2D( boneTexture, vec2( dx * ( x + 3.5 ), y ) );\n\t\t\tmat4 bone = mat4( v1, v2, v3, v4 );\n\t\t\treturn bone;\n\t\t}\n\t#else\n\t\tuniform mat4 boneMatrices[ MAX_BONES ];\n\t\tmat4 getBoneMatrix( const in float i ) {\n\t\t\tmat4 bone = boneMatrices[ int(i) ];\n\t\t\treturn bone;\n\t\t}\n\t#endif\n#endif";

    var skinning_vertex = "#ifdef USE_SKINNING\n\tvec4 skinVertex = bindMatrix * vec4( transformed, 1.0 );\n\tvec4 skinned = vec4( 0.0 );\n\tskinned += boneMatX * skinVertex * skinWeight.x;\n\tskinned += boneMatY * skinVertex * skinWeight.y;\n\tskinned += boneMatZ * skinVertex * skinWeight.z;\n\tskinned += boneMatW * skinVertex * skinWeight.w;\n\ttransformed = ( bindMatrixInverse * skinned ).xyz;\n#endif";

    var skinnormal_vertex = "#ifdef USE_SKINNING\n\tmat4 skinMatrix = mat4( 0.0 );\n\tskinMatrix += skinWeight.x * boneMatX;\n\tskinMatrix += skinWeight.y * boneMatY;\n\tskinMatrix += skinWeight.z * boneMatZ;\n\tskinMatrix += skinWeight.w * boneMatW;\n\tskinMatrix  = bindMatrixInverse * skinMatrix * bindMatrix;\n\tobjectNormal = vec4( skinMatrix * vec4( objectNormal, 0.0 ) ).xyz;\n\t#ifdef USE_TANGENT\n\t\tobjectTangent = vec4( skinMatrix * vec4( objectTangent, 0.0 ) ).xyz;\n\t#endif\n#endif";

    var specularmap_fragment = "float specularStrength;\n#ifdef USE_SPECULARMAP\n\tvec4 texelSpecular = texture2D( specularMap, vUv );\n\tspecularStrength = texelSpecular.r;\n#else\n\tspecularStrength = 1.0;\n#endif";

    var specularmap_pars_fragment = "#ifdef USE_SPECULARMAP\n\tuniform sampler2D specularMap;\n#endif";

    var tonemapping_fragment = "#if defined( TONE_MAPPING )\n\tgl_FragColor.rgb = toneMapping( gl_FragColor.rgb );\n#endif";

    var tonemapping_pars_fragment = "#ifndef saturate\n\t#define saturate(a) clamp( a, 0.0, 1.0 )\n#endif\nuniform float toneMappingExposure;\nuniform float toneMappingWhitePoint;\nvec3 LinearToneMapping( vec3 color ) {\n\treturn toneMappingExposure * color;\n}\nvec3 ReinhardToneMapping( vec3 color ) {\n\tcolor *= toneMappingExposure;\n\treturn saturate( color / ( vec3( 1.0 ) + color ) );\n}\n#define Uncharted2Helper( x ) max( ( ( x * ( 0.15 * x + 0.10 * 0.50 ) + 0.20 * 0.02 ) / ( x * ( 0.15 * x + 0.50 ) + 0.20 * 0.30 ) ) - 0.02 / 0.30, vec3( 0.0 ) )\nvec3 Uncharted2ToneMapping( vec3 color ) {\n\tcolor *= toneMappingExposure;\n\treturn saturate( Uncharted2Helper( color ) / Uncharted2Helper( vec3( toneMappingWhitePoint ) ) );\n}\nvec3 OptimizedCineonToneMapping( vec3 color ) {\n\tcolor *= toneMappingExposure;\n\tcolor = max( vec3( 0.0 ), color - 0.004 );\n\treturn pow( ( color * ( 6.2 * color + 0.5 ) ) / ( color * ( 6.2 * color + 1.7 ) + 0.06 ), vec3( 2.2 ) );\n}\nvec3 ACESFilmicToneMapping( vec3 color ) {\n\tcolor *= toneMappingExposure;\n\treturn saturate( ( color * ( 2.51 * color + 0.03 ) ) / ( color * ( 2.43 * color + 0.59 ) + 0.14 ) );\n}";

    var uv_pars_fragment = "#ifdef USE_UV\n\tvarying vec2 vUv;\n#endif";

    var uv_pars_vertex = "#ifdef USE_UV\n\tvarying vec2 vUv;\n\tuniform mat3 uvTransform;\n#endif";

    var uv_vertex = "#ifdef USE_UV\n\tvUv = ( uvTransform * vec3( uv, 1 ) ).xy;\n#endif";

    var uv2_pars_fragment = "#if defined( USE_LIGHTMAP ) || defined( USE_AOMAP )\n\tvarying vec2 vUv2;\n#endif";

    var uv2_pars_vertex = "#if defined( USE_LIGHTMAP ) || defined( USE_AOMAP )\n\tattribute vec2 uv2;\n\tvarying vec2 vUv2;\n#endif";

    var uv2_vertex = "#if defined( USE_LIGHTMAP ) || defined( USE_AOMAP )\n\tvUv2 = uv2;\n#endif";

    var worldpos_vertex = "#if defined( USE_ENVMAP ) || defined( DISTANCE ) || defined ( USE_SHADOWMAP )\n\tvec4 worldPosition = modelMatrix * vec4( transformed, 1.0 );\n#endif";

    var background_frag = "uniform sampler2D t2D;\nvarying vec2 vUv;\nvoid main() {\n\tvec4 texColor = texture2D( t2D, vUv );\n\tgl_FragColor = mapTexelToLinear( texColor );\n\t#include <tonemapping_fragment>\n\t#include <encodings_fragment>\n}";

    var background_vert = "varying vec2 vUv;\nuniform mat3 uvTransform;\nvoid main() {\n\tvUv = ( uvTransform * vec3( uv, 1 ) ).xy;\n\tgl_Position = vec4( position.xy, 1.0, 1.0 );\n}";

    var cube_frag = "uniform samplerCube tCube;\nuniform float tFlip;\nuniform float opacity;\nvarying vec3 vWorldDirection;\nvoid main() {\n\tvec4 texColor = textureCube( tCube, vec3( tFlip * vWorldDirection.x, vWorldDirection.yz ) );\n\tgl_FragColor = mapTexelToLinear( texColor );\n\tgl_FragColor.a *= opacity;\n\t#include <tonemapping_fragment>\n\t#include <encodings_fragment>\n}";

    var cube_vert = "varying vec3 vWorldDirection;\n#include <common>\nvoid main() {\n\tvWorldDirection = transformDirection( position, modelMatrix );\n\t#include <begin_vertex>\n\t#include <project_vertex>\n\tgl_Position.z = gl_Position.w;\n}";

    var depth_frag = "#if DEPTH_PACKING == 3200\n\tuniform float opacity;\n#endif\n#include <common>\n#include <packing>\n#include <uv_pars_fragment>\n#include <map_pars_fragment>\n#include <alphamap_pars_fragment>\n#include <logdepthbuf_pars_fragment>\n#include <clipping_planes_pars_fragment>\nvoid main() {\n\t#include <clipping_planes_fragment>\n\tvec4 diffuseColor = vec4( 1.0 );\n\t#if DEPTH_PACKING == 3200\n\t\tdiffuseColor.a = opacity;\n\t#endif\n\t#include <map_fragment>\n\t#include <alphamap_fragment>\n\t#include <alphatest_fragment>\n\t#include <logdepthbuf_fragment>\n\t#if DEPTH_PACKING == 3200\n\t\tgl_FragColor = vec4( vec3( 1.0 - gl_FragCoord.z ), opacity );\n\t#elif DEPTH_PACKING == 3201\n\t\tgl_FragColor = packDepthToRGBA( gl_FragCoord.z );\n\t#endif\n}";

    var depth_vert = "#include <common>\n#include <uv_pars_vertex>\n#include <displacementmap_pars_vertex>\n#include <morphtarget_pars_vertex>\n#include <skinning_pars_vertex>\n#include <logdepthbuf_pars_vertex>\n#include <clipping_planes_pars_vertex>\nvoid main() {\n\t#include <uv_vertex>\n\t#include <skinbase_vertex>\n\t#ifdef USE_DISPLACEMENTMAP\n\t\t#include <beginnormal_vertex>\n\t\t#include <morphnormal_vertex>\n\t\t#include <skinnormal_vertex>\n\t#endif\n\t#include <begin_vertex>\n\t#include <morphtarget_vertex>\n\t#include <skinning_vertex>\n\t#include <displacementmap_vertex>\n\t#include <project_vertex>\n\t#include <logdepthbuf_vertex>\n\t#include <clipping_planes_vertex>\n}";

    var distanceRGBA_frag = "#define DISTANCE\nuniform vec3 referencePosition;\nuniform float nearDistance;\nuniform float farDistance;\nvarying vec3 vWorldPosition;\n#include <common>\n#include <packing>\n#include <uv_pars_fragment>\n#include <map_pars_fragment>\n#include <alphamap_pars_fragment>\n#include <clipping_planes_pars_fragment>\nvoid main () {\n\t#include <clipping_planes_fragment>\n\tvec4 diffuseColor = vec4( 1.0 );\n\t#include <map_fragment>\n\t#include <alphamap_fragment>\n\t#include <alphatest_fragment>\n\tfloat dist = length( vWorldPosition - referencePosition );\n\tdist = ( dist - nearDistance ) / ( farDistance - nearDistance );\n\tdist = saturate( dist );\n\tgl_FragColor = packDepthToRGBA( dist );\n}";

    var distanceRGBA_vert = "#define DISTANCE\nvarying vec3 vWorldPosition;\n#include <common>\n#include <uv_pars_vertex>\n#include <displacementmap_pars_vertex>\n#include <morphtarget_pars_vertex>\n#include <skinning_pars_vertex>\n#include <clipping_planes_pars_vertex>\nvoid main() {\n\t#include <uv_vertex>\n\t#include <skinbase_vertex>\n\t#ifdef USE_DISPLACEMENTMAP\n\t\t#include <beginnormal_vertex>\n\t\t#include <morphnormal_vertex>\n\t\t#include <skinnormal_vertex>\n\t#endif\n\t#include <begin_vertex>\n\t#include <morphtarget_vertex>\n\t#include <skinning_vertex>\n\t#include <displacementmap_vertex>\n\t#include <project_vertex>\n\t#include <worldpos_vertex>\n\t#include <clipping_planes_vertex>\n\tvWorldPosition = worldPosition.xyz;\n}";

    var equirect_frag = "uniform sampler2D tEquirect;\nvarying vec3 vWorldDirection;\n#include <common>\nvoid main() {\n\tvec3 direction = normalize( vWorldDirection );\n\tvec2 sampleUV;\n\tsampleUV.y = asin( clamp( direction.y, - 1.0, 1.0 ) ) * RECIPROCAL_PI + 0.5;\n\tsampleUV.x = atan( direction.z, direction.x ) * RECIPROCAL_PI2 + 0.5;\n\tvec4 texColor = texture2D( tEquirect, sampleUV );\n\tgl_FragColor = mapTexelToLinear( texColor );\n\t#include <tonemapping_fragment>\n\t#include <encodings_fragment>\n}";

    var equirect_vert = "varying vec3 vWorldDirection;\n#include <common>\nvoid main() {\n\tvWorldDirection = transformDirection( position, modelMatrix );\n\t#include <begin_vertex>\n\t#include <project_vertex>\n}";

    var linedashed_frag = "uniform vec3 diffuse;\nuniform float opacity;\nuniform float dashSize;\nuniform float totalSize;\nvarying float vLineDistance;\n#include <common>\n#include <color_pars_fragment>\n#include <fog_pars_fragment>\n#include <logdepthbuf_pars_fragment>\n#include <clipping_planes_pars_fragment>\nvoid main() {\n\t#include <clipping_planes_fragment>\n\tif ( mod( vLineDistance, totalSize ) > dashSize ) {\n\t\tdiscard;\n\t}\n\tvec3 outgoingLight = vec3( 0.0 );\n\tvec4 diffuseColor = vec4( diffuse, opacity );\n\t#include <logdepthbuf_fragment>\n\t#include <color_fragment>\n\toutgoingLight = diffuseColor.rgb;\n\tgl_FragColor = vec4( outgoingLight, diffuseColor.a );\n\t#include <premultiplied_alpha_fragment>\n\t#include <tonemapping_fragment>\n\t#include <encodings_fragment>\n\t#include <fog_fragment>\n}";

    var linedashed_vert = "uniform float scale;\nattribute float lineDistance;\nvarying float vLineDistance;\n#include <common>\n#include <color_pars_vertex>\n#include <fog_pars_vertex>\n#include <logdepthbuf_pars_vertex>\n#include <clipping_planes_pars_vertex>\nvoid main() {\n\t#include <color_vertex>\n\tvLineDistance = scale * lineDistance;\n\tvec4 mvPosition = modelViewMatrix * vec4( position, 1.0 );\n\tgl_Position = projectionMatrix * mvPosition;\n\t#include <logdepthbuf_vertex>\n\t#include <clipping_planes_vertex>\n\t#include <fog_vertex>\n}";

    var meshbasic_frag = "uniform vec3 diffuse;\nuniform float opacity;\n#ifndef FLAT_SHADED\n\tvarying vec3 vNormal;\n#endif\n#include <common>\n#include <color_pars_fragment>\n#include <uv_pars_fragment>\n#include <uv2_pars_fragment>\n#include <map_pars_fragment>\n#include <alphamap_pars_fragment>\n#include <aomap_pars_fragment>\n#include <lightmap_pars_fragment>\n#include <envmap_common_pars_fragment>\n#include <envmap_pars_fragment>\n#include <fog_pars_fragment>\n#include <specularmap_pars_fragment>\n#include <logdepthbuf_pars_fragment>\n#include <clipping_planes_pars_fragment>\nvoid main() {\n\t#include <clipping_planes_fragment>\n\tvec4 diffuseColor = vec4( diffuse, opacity );\n\t#include <logdepthbuf_fragment>\n\t#include <map_fragment>\n\t#include <color_fragment>\n\t#include <alphamap_fragment>\n\t#include <alphatest_fragment>\n\t#include <specularmap_fragment>\n\tReflectedLight reflectedLight = ReflectedLight( vec3( 0.0 ), vec3( 0.0 ), vec3( 0.0 ), vec3( 0.0 ) );\n\t#ifdef USE_LIGHTMAP\n\t\treflectedLight.indirectDiffuse += texture2D( lightMap, vUv2 ).xyz * lightMapIntensity;\n\t#else\n\t\treflectedLight.indirectDiffuse += vec3( 1.0 );\n\t#endif\n\t#include <aomap_fragment>\n\treflectedLight.indirectDiffuse *= diffuseColor.rgb;\n\tvec3 outgoingLight = reflectedLight.indirectDiffuse;\n\t#include <envmap_fragment>\n\tgl_FragColor = vec4( outgoingLight, diffuseColor.a );\n\t#include <premultiplied_alpha_fragment>\n\t#include <tonemapping_fragment>\n\t#include <encodings_fragment>\n\t#include <fog_fragment>\n}";

    var meshbasic_vert = "#include <common>\n#include <uv_pars_vertex>\n#include <uv2_pars_vertex>\n#include <envmap_pars_vertex>\n#include <color_pars_vertex>\n#include <fog_pars_vertex>\n#include <morphtarget_pars_vertex>\n#include <skinning_pars_vertex>\n#include <logdepthbuf_pars_vertex>\n#include <clipping_planes_pars_vertex>\nvoid main() {\n\t#include <uv_vertex>\n\t#include <uv2_vertex>\n\t#include <color_vertex>\n\t#include <skinbase_vertex>\n\t#ifdef USE_ENVMAP\n\t#include <beginnormal_vertex>\n\t#include <morphnormal_vertex>\n\t#include <skinnormal_vertex>\n\t#include <defaultnormal_vertex>\n\t#endif\n\t#include <begin_vertex>\n\t#include <morphtarget_vertex>\n\t#include <skinning_vertex>\n\t#include <project_vertex>\n\t#include <logdepthbuf_vertex>\n\t#include <worldpos_vertex>\n\t#include <clipping_planes_vertex>\n\t#include <envmap_vertex>\n\t#include <fog_vertex>\n}";

    var meshlambert_frag = "uniform vec3 diffuse;\nuniform vec3 emissive;\nuniform float opacity;\nvarying vec3 vLightFront;\nvarying vec3 vIndirectFront;\n#ifdef DOUBLE_SIDED\n\tvarying vec3 vLightBack;\n\tvarying vec3 vIndirectBack;\n#endif\n#include <common>\n#include <packing>\n#include <dithering_pars_fragment>\n#include <color_pars_fragment>\n#include <uv_pars_fragment>\n#include <uv2_pars_fragment>\n#include <map_pars_fragment>\n#include <alphamap_pars_fragment>\n#include <aomap_pars_fragment>\n#include <lightmap_pars_fragment>\n#include <emissivemap_pars_fragment>\n#include <envmap_common_pars_fragment>\n#include <envmap_pars_fragment>\n#include <bsdfs>\n#include <lights_pars_begin>\n#include <fog_pars_fragment>\n#include <shadowmap_pars_fragment>\n#include <shadowmask_pars_fragment>\n#include <specularmap_pars_fragment>\n#include <logdepthbuf_pars_fragment>\n#include <clipping_planes_pars_fragment>\nvoid main() {\n\t#include <clipping_planes_fragment>\n\tvec4 diffuseColor = vec4( diffuse, opacity );\n\tReflectedLight reflectedLight = ReflectedLight( vec3( 0.0 ), vec3( 0.0 ), vec3( 0.0 ), vec3( 0.0 ) );\n\tvec3 totalEmissiveRadiance = emissive;\n\t#include <logdepthbuf_fragment>\n\t#include <map_fragment>\n\t#include <color_fragment>\n\t#include <alphamap_fragment>\n\t#include <alphatest_fragment>\n\t#include <specularmap_fragment>\n\t#include <emissivemap_fragment>\n\treflectedLight.indirectDiffuse = getAmbientLightIrradiance( ambientLightColor );\n\t#ifdef DOUBLE_SIDED\n\t\treflectedLight.indirectDiffuse += ( gl_FrontFacing ) ? vIndirectFront : vIndirectBack;\n\t#else\n\t\treflectedLight.indirectDiffuse += vIndirectFront;\n\t#endif\n\t#include <lightmap_fragment>\n\treflectedLight.indirectDiffuse *= BRDF_Diffuse_Lambert( diffuseColor.rgb );\n\t#ifdef DOUBLE_SIDED\n\t\treflectedLight.directDiffuse = ( gl_FrontFacing ) ? vLightFront : vLightBack;\n\t#else\n\t\treflectedLight.directDiffuse = vLightFront;\n\t#endif\n\treflectedLight.directDiffuse *= BRDF_Diffuse_Lambert( diffuseColor.rgb ) * getShadowMask();\n\t#include <aomap_fragment>\n\tvec3 outgoingLight = reflectedLight.directDiffuse + reflectedLight.indirectDiffuse + totalEmissiveRadiance;\n\t#include <envmap_fragment>\n\tgl_FragColor = vec4( outgoingLight, diffuseColor.a );\n\t#include <tonemapping_fragment>\n\t#include <encodings_fragment>\n\t#include <fog_fragment>\n\t#include <premultiplied_alpha_fragment>\n\t#include <dithering_fragment>\n}";

    var meshlambert_vert = "#define LAMBERT\nvarying vec3 vLightFront;\nvarying vec3 vIndirectFront;\n#ifdef DOUBLE_SIDED\n\tvarying vec3 vLightBack;\n\tvarying vec3 vIndirectBack;\n#endif\n#include <common>\n#include <uv_pars_vertex>\n#include <uv2_pars_vertex>\n#include <envmap_pars_vertex>\n#include <bsdfs>\n#include <lights_pars_begin>\n#include <color_pars_vertex>\n#include <fog_pars_vertex>\n#include <morphtarget_pars_vertex>\n#include <skinning_pars_vertex>\n#include <shadowmap_pars_vertex>\n#include <logdepthbuf_pars_vertex>\n#include <clipping_planes_pars_vertex>\nvoid main() {\n\t#include <uv_vertex>\n\t#include <uv2_vertex>\n\t#include <color_vertex>\n\t#include <beginnormal_vertex>\n\t#include <morphnormal_vertex>\n\t#include <skinbase_vertex>\n\t#include <skinnormal_vertex>\n\t#include <defaultnormal_vertex>\n\t#include <begin_vertex>\n\t#include <morphtarget_vertex>\n\t#include <skinning_vertex>\n\t#include <project_vertex>\n\t#include <logdepthbuf_vertex>\n\t#include <clipping_planes_vertex>\n\t#include <worldpos_vertex>\n\t#include <envmap_vertex>\n\t#include <lights_lambert_vertex>\n\t#include <shadowmap_vertex>\n\t#include <fog_vertex>\n}";

    var meshmatcap_frag = "#define MATCAP\nuniform vec3 diffuse;\nuniform float opacity;\nuniform sampler2D matcap;\nvarying vec3 vViewPosition;\n#ifndef FLAT_SHADED\n\tvarying vec3 vNormal;\n#endif\n#include <common>\n#include <uv_pars_fragment>\n#include <map_pars_fragment>\n#include <alphamap_pars_fragment>\n#include <fog_pars_fragment>\n#include <bumpmap_pars_fragment>\n#include <normalmap_pars_fragment>\n#include <logdepthbuf_pars_fragment>\n#include <clipping_planes_pars_fragment>\nvoid main() {\n\t#include <clipping_planes_fragment>\n\tvec4 diffuseColor = vec4( diffuse, opacity );\n\t#include <logdepthbuf_fragment>\n\t#include <map_fragment>\n\t#include <alphamap_fragment>\n\t#include <alphatest_fragment>\n\t#include <normal_fragment_begin>\n\t#include <normal_fragment_maps>\n\tvec3 viewDir = normalize( vViewPosition );\n\tvec3 x = normalize( vec3( viewDir.z, 0.0, - viewDir.x ) );\n\tvec3 y = cross( viewDir, x );\n\tvec2 uv = vec2( dot( x, normal ), dot( y, normal ) ) * 0.495 + 0.5;\n\t#ifdef USE_MATCAP\n\t\tvec4 matcapColor = texture2D( matcap, uv );\n\t\tmatcapColor = matcapTexelToLinear( matcapColor );\n\t#else\n\t\tvec4 matcapColor = vec4( 1.0 );\n\t#endif\n\tvec3 outgoingLight = diffuseColor.rgb * matcapColor.rgb;\n\tgl_FragColor = vec4( outgoingLight, diffuseColor.a );\n\t#include <premultiplied_alpha_fragment>\n\t#include <tonemapping_fragment>\n\t#include <encodings_fragment>\n\t#include <fog_fragment>\n}";

    var meshmatcap_vert = "#define MATCAP\nvarying vec3 vViewPosition;\n#ifndef FLAT_SHADED\n\tvarying vec3 vNormal;\n#endif\n#include <common>\n#include <uv_pars_vertex>\n#include <displacementmap_pars_vertex>\n#include <fog_pars_vertex>\n#include <morphtarget_pars_vertex>\n#include <skinning_pars_vertex>\n#include <logdepthbuf_pars_vertex>\n#include <clipping_planes_pars_vertex>\nvoid main() {\n\t#include <uv_vertex>\n\t#include <beginnormal_vertex>\n\t#include <morphnormal_vertex>\n\t#include <skinbase_vertex>\n\t#include <skinnormal_vertex>\n\t#include <defaultnormal_vertex>\n\t#ifndef FLAT_SHADED\n\t\tvNormal = normalize( transformedNormal );\n\t#endif\n\t#include <begin_vertex>\n\t#include <morphtarget_vertex>\n\t#include <skinning_vertex>\n\t#include <displacementmap_vertex>\n\t#include <project_vertex>\n\t#include <logdepthbuf_vertex>\n\t#include <clipping_planes_vertex>\n\t#include <fog_vertex>\n\tvViewPosition = - mvPosition.xyz;\n}";

    var meshphong_frag = "#define PHONG\nuniform vec3 diffuse;\nuniform vec3 emissive;\nuniform vec3 specular;\nuniform float shininess;\nuniform float opacity;\n#include <common>\n#include <packing>\n#include <dithering_pars_fragment>\n#include <color_pars_fragment>\n#include <uv_pars_fragment>\n#include <uv2_pars_fragment>\n#include <map_pars_fragment>\n#include <alphamap_pars_fragment>\n#include <aomap_pars_fragment>\n#include <lightmap_pars_fragment>\n#include <emissivemap_pars_fragment>\n#include <envmap_common_pars_fragment>\n#include <envmap_pars_fragment>\n#include <gradientmap_pars_fragment>\n#include <fog_pars_fragment>\n#include <bsdfs>\n#include <lights_pars_begin>\n#include <lights_phong_pars_fragment>\n#include <shadowmap_pars_fragment>\n#include <bumpmap_pars_fragment>\n#include <normalmap_pars_fragment>\n#include <specularmap_pars_fragment>\n#include <logdepthbuf_pars_fragment>\n#include <clipping_planes_pars_fragment>\nvoid main() {\n\t#include <clipping_planes_fragment>\n\tvec4 diffuseColor = vec4( diffuse, opacity );\n\tReflectedLight reflectedLight = ReflectedLight( vec3( 0.0 ), vec3( 0.0 ), vec3( 0.0 ), vec3( 0.0 ) );\n\tvec3 totalEmissiveRadiance = emissive;\n\t#include <logdepthbuf_fragment>\n\t#include <map_fragment>\n\t#include <color_fragment>\n\t#include <alphamap_fragment>\n\t#include <alphatest_fragment>\n\t#include <specularmap_fragment>\n\t#include <normal_fragment_begin>\n\t#include <normal_fragment_maps>\n\t#include <emissivemap_fragment>\n\t#include <lights_phong_fragment>\n\t#include <lights_fragment_begin>\n\t#include <lights_fragment_maps>\n\t#include <lights_fragment_end>\n\t#include <aomap_fragment>\n\tvec3 outgoingLight = reflectedLight.directDiffuse + reflectedLight.indirectDiffuse + reflectedLight.directSpecular + reflectedLight.indirectSpecular + totalEmissiveRadiance;\n\t#include <envmap_fragment>\n\tgl_FragColor = vec4( outgoingLight, diffuseColor.a );\n\t#include <tonemapping_fragment>\n\t#include <encodings_fragment>\n\t#include <fog_fragment>\n\t#include <premultiplied_alpha_fragment>\n\t#include <dithering_fragment>\n}";

    var meshphong_vert = "#define PHONG\nvarying vec3 vViewPosition;\n#ifndef FLAT_SHADED\n\tvarying vec3 vNormal;\n#endif\n#include <common>\n#include <uv_pars_vertex>\n#include <uv2_pars_vertex>\n#include <displacementmap_pars_vertex>\n#include <envmap_pars_vertex>\n#include <color_pars_vertex>\n#include <fog_pars_vertex>\n#include <morphtarget_pars_vertex>\n#include <skinning_pars_vertex>\n#include <shadowmap_pars_vertex>\n#include <logdepthbuf_pars_vertex>\n#include <clipping_planes_pars_vertex>\nvoid main() {\n\t#include <uv_vertex>\n\t#include <uv2_vertex>\n\t#include <color_vertex>\n\t#include <beginnormal_vertex>\n\t#include <morphnormal_vertex>\n\t#include <skinbase_vertex>\n\t#include <skinnormal_vertex>\n\t#include <defaultnormal_vertex>\n#ifndef FLAT_SHADED\n\tvNormal = normalize( transformedNormal );\n#endif\n\t#include <begin_vertex>\n\t#include <morphtarget_vertex>\n\t#include <skinning_vertex>\n\t#include <displacementmap_vertex>\n\t#include <project_vertex>\n\t#include <logdepthbuf_vertex>\n\t#include <clipping_planes_vertex>\n\tvViewPosition = - mvPosition.xyz;\n\t#include <worldpos_vertex>\n\t#include <envmap_vertex>\n\t#include <shadowmap_vertex>\n\t#include <fog_vertex>\n}";

    var meshphysical_frag = "#define STANDARD\n#ifdef PHYSICAL\n\t#define REFLECTIVITY\n\t#define CLEARCOAT\n\t#define TRANSPARENCY\n#endif\nuniform vec3 diffuse;\nuniform vec3 emissive;\nuniform float roughness;\nuniform float metalness;\nuniform float opacity;\n#ifdef TRANSPARENCY\n\tuniform float transparency;\n#endif\n#ifdef REFLECTIVITY\n\tuniform float reflectivity;\n#endif\n#ifdef CLEARCOAT\n\tuniform float clearcoat;\n\tuniform float clearcoatRoughness;\n#endif\n#ifdef USE_SHEEN\n\tuniform vec3 sheen;\n#endif\nvarying vec3 vViewPosition;\n#ifndef FLAT_SHADED\n\tvarying vec3 vNormal;\n\t#ifdef USE_TANGENT\n\t\tvarying vec3 vTangent;\n\t\tvarying vec3 vBitangent;\n\t#endif\n#endif\n#include <common>\n#include <packing>\n#include <dithering_pars_fragment>\n#include <color_pars_fragment>\n#include <uv_pars_fragment>\n#include <uv2_pars_fragment>\n#include <map_pars_fragment>\n#include <alphamap_pars_fragment>\n#include <aomap_pars_fragment>\n#include <lightmap_pars_fragment>\n#include <emissivemap_pars_fragment>\n#include <bsdfs>\n#include <cube_uv_reflection_fragment>\n#include <envmap_common_pars_fragment>\n#include <envmap_physical_pars_fragment>\n#include <fog_pars_fragment>\n#include <lights_pars_begin>\n#include <lights_physical_pars_fragment>\n#include <shadowmap_pars_fragment>\n#include <bumpmap_pars_fragment>\n#include <normalmap_pars_fragment>\n#include <clearcoat_normalmap_pars_fragment>\n#include <roughnessmap_pars_fragment>\n#include <metalnessmap_pars_fragment>\n#include <logdepthbuf_pars_fragment>\n#include <clipping_planes_pars_fragment>\nvoid main() {\n\t#include <clipping_planes_fragment>\n\tvec4 diffuseColor = vec4( diffuse, opacity );\n\tReflectedLight reflectedLight = ReflectedLight( vec3( 0.0 ), vec3( 0.0 ), vec3( 0.0 ), vec3( 0.0 ) );\n\tvec3 totalEmissiveRadiance = emissive;\n\t#include <logdepthbuf_fragment>\n\t#include <map_fragment>\n\t#include <color_fragment>\n\t#include <alphamap_fragment>\n\t#include <alphatest_fragment>\n\t#include <roughnessmap_fragment>\n\t#include <metalnessmap_fragment>\n\t#include <normal_fragment_begin>\n\t#include <normal_fragment_maps>\n\t#include <clearcoat_normal_fragment_begin>\n\t#include <clearcoat_normal_fragment_maps>\n\t#include <emissivemap_fragment>\n\t#include <lights_physical_fragment>\n\t#include <lights_fragment_begin>\n\t#include <lights_fragment_maps>\n\t#include <lights_fragment_end>\n\t#include <aomap_fragment>\n\tvec3 outgoingLight = reflectedLight.directDiffuse + reflectedLight.indirectDiffuse + reflectedLight.directSpecular + reflectedLight.indirectSpecular + totalEmissiveRadiance;\n\t#ifdef TRANSPARENCY\n\t\tdiffuseColor.a *= saturate( 1. - transparency + linearToRelativeLuminance( reflectedLight.directSpecular + reflectedLight.indirectSpecular ) );\n\t#endif\n\tgl_FragColor = vec4( outgoingLight, diffuseColor.a );\n\t#include <tonemapping_fragment>\n\t#include <encodings_fragment>\n\t#include <fog_fragment>\n\t#include <premultiplied_alpha_fragment>\n\t#include <dithering_fragment>\n}";

    var meshphysical_vert = "#define STANDARD\nvarying vec3 vViewPosition;\n#ifndef FLAT_SHADED\n\tvarying vec3 vNormal;\n\t#ifdef USE_TANGENT\n\t\tvarying vec3 vTangent;\n\t\tvarying vec3 vBitangent;\n\t#endif\n#endif\n#include <common>\n#include <uv_pars_vertex>\n#include <uv2_pars_vertex>\n#include <displacementmap_pars_vertex>\n#include <color_pars_vertex>\n#include <fog_pars_vertex>\n#include <morphtarget_pars_vertex>\n#include <skinning_pars_vertex>\n#include <shadowmap_pars_vertex>\n#include <logdepthbuf_pars_vertex>\n#include <clipping_planes_pars_vertex>\nvoid main() {\n\t#include <uv_vertex>\n\t#include <uv2_vertex>\n\t#include <color_vertex>\n\t#include <beginnormal_vertex>\n\t#include <morphnormal_vertex>\n\t#include <skinbase_vertex>\n\t#include <skinnormal_vertex>\n\t#include <defaultnormal_vertex>\n#ifndef FLAT_SHADED\n\tvNormal = normalize( transformedNormal );\n\t#ifdef USE_TANGENT\n\t\tvTangent = normalize( transformedTangent );\n\t\tvBitangent = normalize( cross( vNormal, vTangent ) * tangent.w );\n\t#endif\n#endif\n\t#include <begin_vertex>\n\t#include <morphtarget_vertex>\n\t#include <skinning_vertex>\n\t#include <displacementmap_vertex>\n\t#include <project_vertex>\n\t#include <logdepthbuf_vertex>\n\t#include <clipping_planes_vertex>\n\tvViewPosition = - mvPosition.xyz;\n\t#include <worldpos_vertex>\n\t#include <shadowmap_vertex>\n\t#include <fog_vertex>\n}";

    var normal_frag = "#define NORMAL\nuniform float opacity;\n#if defined( FLAT_SHADED ) || defined( USE_BUMPMAP ) || defined( TANGENTSPACE_NORMALMAP )\n\tvarying vec3 vViewPosition;\n#endif\n#ifndef FLAT_SHADED\n\tvarying vec3 vNormal;\n\t#ifdef USE_TANGENT\n\t\tvarying vec3 vTangent;\n\t\tvarying vec3 vBitangent;\n\t#endif\n#endif\n#include <packing>\n#include <uv_pars_fragment>\n#include <bumpmap_pars_fragment>\n#include <normalmap_pars_fragment>\n#include <logdepthbuf_pars_fragment>\n#include <clipping_planes_pars_fragment>\nvoid main() {\n\t#include <clipping_planes_fragment>\n\t#include <logdepthbuf_fragment>\n\t#include <normal_fragment_begin>\n\t#include <normal_fragment_maps>\n\tgl_FragColor = vec4( packNormalToRGB( normal ), opacity );\n}";

    var normal_vert = "#define NORMAL\n#if defined( FLAT_SHADED ) || defined( USE_BUMPMAP ) || defined( TANGENTSPACE_NORMALMAP )\n\tvarying vec3 vViewPosition;\n#endif\n#ifndef FLAT_SHADED\n\tvarying vec3 vNormal;\n\t#ifdef USE_TANGENT\n\t\tvarying vec3 vTangent;\n\t\tvarying vec3 vBitangent;\n\t#endif\n#endif\n#include <uv_pars_vertex>\n#include <displacementmap_pars_vertex>\n#include <morphtarget_pars_vertex>\n#include <skinning_pars_vertex>\n#include <logdepthbuf_pars_vertex>\n#include <clipping_planes_pars_vertex>\nvoid main() {\n\t#include <uv_vertex>\n\t#include <beginnormal_vertex>\n\t#include <morphnormal_vertex>\n\t#include <skinbase_vertex>\n\t#include <skinnormal_vertex>\n\t#include <defaultnormal_vertex>\n#ifndef FLAT_SHADED\n\tvNormal = normalize( transformedNormal );\n\t#ifdef USE_TANGENT\n\t\tvTangent = normalize( transformedTangent );\n\t\tvBitangent = normalize( cross( vNormal, vTangent ) * tangent.w );\n\t#endif\n#endif\n\t#include <begin_vertex>\n\t#include <morphtarget_vertex>\n\t#include <skinning_vertex>\n\t#include <displacementmap_vertex>\n\t#include <project_vertex>\n\t#include <logdepthbuf_vertex>\n\t#include <clipping_planes_vertex>\n#if defined( FLAT_SHADED ) || defined( USE_BUMPMAP ) || defined( TANGENTSPACE_NORMALMAP )\n\tvViewPosition = - mvPosition.xyz;\n#endif\n}";

    var points_frag = "uniform vec3 diffuse;\nuniform float opacity;\n#include <common>\n#include <color_pars_fragment>\n#include <map_particle_pars_fragment>\n#include <fog_pars_fragment>\n#include <logdepthbuf_pars_fragment>\n#include <clipping_planes_pars_fragment>\nvoid main() {\n\t#include <clipping_planes_fragment>\n\tvec3 outgoingLight = vec3( 0.0 );\n\tvec4 diffuseColor = vec4( diffuse, opacity );\n\t#include <logdepthbuf_fragment>\n\t#include <map_particle_fragment>\n\t#include <color_fragment>\n\t#include <alphatest_fragment>\n\toutgoingLight = diffuseColor.rgb;\n\tgl_FragColor = vec4( outgoingLight, diffuseColor.a );\n\t#include <premultiplied_alpha_fragment>\n\t#include <tonemapping_fragment>\n\t#include <encodings_fragment>\n\t#include <fog_fragment>\n}";

    var points_vert = "uniform float size;\nuniform float scale;\n#include <common>\n#include <color_pars_vertex>\n#include <fog_pars_vertex>\n#include <morphtarget_pars_vertex>\n#include <logdepthbuf_pars_vertex>\n#include <clipping_planes_pars_vertex>\nvoid main() {\n\t#include <color_vertex>\n\t#include <begin_vertex>\n\t#include <morphtarget_vertex>\n\t#include <project_vertex>\n\tgl_PointSize = size;\n\t#ifdef USE_SIZEATTENUATION\n\t\tbool isPerspective = ( projectionMatrix[ 2 ][ 3 ] == - 1.0 );\n\t\tif ( isPerspective ) gl_PointSize *= ( scale / - mvPosition.z );\n\t#endif\n\t#include <logdepthbuf_vertex>\n\t#include <clipping_planes_vertex>\n\t#include <worldpos_vertex>\n\t#include <fog_vertex>\n}";

    var shadow_frag = "uniform vec3 color;\nuniform float opacity;\n#include <common>\n#include <packing>\n#include <fog_pars_fragment>\n#include <bsdfs>\n#include <lights_pars_begin>\n#include <shadowmap_pars_fragment>\n#include <shadowmask_pars_fragment>\nvoid main() {\n\tgl_FragColor = vec4( color, opacity * ( 1.0 - getShadowMask() ) );\n\t#include <fog_fragment>\n}";

    var shadow_vert = "#include <fog_pars_vertex>\n#include <shadowmap_pars_vertex>\nvoid main() {\n\t#include <begin_vertex>\n\t#include <project_vertex>\n\t#include <worldpos_vertex>\n\t#include <shadowmap_vertex>\n\t#include <fog_vertex>\n}";

    var sprite_frag = "uniform vec3 diffuse;\nuniform float opacity;\n#include <common>\n#include <uv_pars_fragment>\n#include <map_pars_fragment>\n#include <fog_pars_fragment>\n#include <logdepthbuf_pars_fragment>\n#include <clipping_planes_pars_fragment>\nvoid main() {\n\t#include <clipping_planes_fragment>\n\tvec3 outgoingLight = vec3( 0.0 );\n\tvec4 diffuseColor = vec4( diffuse, opacity );\n\t#include <logdepthbuf_fragment>\n\t#include <map_fragment>\n\t#include <alphatest_fragment>\n\toutgoingLight = diffuseColor.rgb;\n\tgl_FragColor = vec4( outgoingLight, diffuseColor.a );\n\t#include <tonemapping_fragment>\n\t#include <encodings_fragment>\n\t#include <fog_fragment>\n}";

    var sprite_vert = "uniform float rotation;\nuniform vec2 center;\n#include <common>\n#include <uv_pars_vertex>\n#include <fog_pars_vertex>\n#include <logdepthbuf_pars_vertex>\n#include <clipping_planes_pars_vertex>\nvoid main() {\n\t#include <uv_vertex>\n\tvec4 mvPosition = modelViewMatrix * vec4( 0.0, 0.0, 0.0, 1.0 );\n\tvec2 scale;\n\tscale.x = length( vec3( modelMatrix[ 0 ].x, modelMatrix[ 0 ].y, modelMatrix[ 0 ].z ) );\n\tscale.y = length( vec3( modelMatrix[ 1 ].x, modelMatrix[ 1 ].y, modelMatrix[ 1 ].z ) );\n\t#ifndef USE_SIZEATTENUATION\n\t\tbool isPerspective = ( projectionMatrix[ 2 ][ 3 ] == - 1.0 );\n\t\tif ( isPerspective ) scale *= - mvPosition.z;\n\t#endif\n\tvec2 alignedPosition = ( position.xy - ( center - vec2( 0.5 ) ) ) * scale;\n\tvec2 rotatedPosition;\n\trotatedPosition.x = cos( rotation ) * alignedPosition.x - sin( rotation ) * alignedPosition.y;\n\trotatedPosition.y = sin( rotation ) * alignedPosition.x + cos( rotation ) * alignedPosition.y;\n\tmvPosition.xy += rotatedPosition;\n\tgl_Position = projectionMatrix * mvPosition;\n\t#include <logdepthbuf_vertex>\n\t#include <clipping_planes_vertex>\n\t#include <fog_vertex>\n}";

    var ShaderChunk = {
    	alphamap_fragment: alphamap_fragment,
    	alphamap_pars_fragment: alphamap_pars_fragment,
    	alphatest_fragment: alphatest_fragment,
    	aomap_fragment: aomap_fragment,
    	aomap_pars_fragment: aomap_pars_fragment,
    	begin_vertex: begin_vertex,
    	beginnormal_vertex: beginnormal_vertex,
    	bsdfs: bsdfs,
    	bumpmap_pars_fragment: bumpmap_pars_fragment,
    	clipping_planes_fragment: clipping_planes_fragment,
    	clipping_planes_pars_fragment: clipping_planes_pars_fragment,
    	clipping_planes_pars_vertex: clipping_planes_pars_vertex,
    	clipping_planes_vertex: clipping_planes_vertex,
    	color_fragment: color_fragment,
    	color_pars_fragment: color_pars_fragment,
    	color_pars_vertex: color_pars_vertex,
    	color_vertex: color_vertex,
    	common: common,
    	cube_uv_reflection_fragment: cube_uv_reflection_fragment,
    	defaultnormal_vertex: defaultnormal_vertex,
    	displacementmap_pars_vertex: displacementmap_pars_vertex,
    	displacementmap_vertex: displacementmap_vertex,
    	emissivemap_fragment: emissivemap_fragment,
    	emissivemap_pars_fragment: emissivemap_pars_fragment,
    	encodings_fragment: encodings_fragment,
    	encodings_pars_fragment: encodings_pars_fragment,
    	envmap_fragment: envmap_fragment,
    	envmap_common_pars_fragment: envmap_common_pars_fragment,
    	envmap_pars_fragment: envmap_pars_fragment,
    	envmap_pars_vertex: envmap_pars_vertex,
    	envmap_physical_pars_fragment: envmap_physical_pars_fragment,
    	envmap_vertex: envmap_vertex,
    	fog_vertex: fog_vertex,
    	fog_pars_vertex: fog_pars_vertex,
    	fog_fragment: fog_fragment,
    	fog_pars_fragment: fog_pars_fragment,
    	gradientmap_pars_fragment: gradientmap_pars_fragment,
    	lightmap_fragment: lightmap_fragment,
    	lightmap_pars_fragment: lightmap_pars_fragment,
    	lights_lambert_vertex: lights_lambert_vertex,
    	lights_pars_begin: lights_pars_begin,
    	lights_phong_fragment: lights_phong_fragment,
    	lights_phong_pars_fragment: lights_phong_pars_fragment,
    	lights_physical_fragment: lights_physical_fragment,
    	lights_physical_pars_fragment: lights_physical_pars_fragment,
    	lights_fragment_begin: lights_fragment_begin,
    	lights_fragment_maps: lights_fragment_maps,
    	lights_fragment_end: lights_fragment_end,
    	logdepthbuf_fragment: logdepthbuf_fragment,
    	logdepthbuf_pars_fragment: logdepthbuf_pars_fragment,
    	logdepthbuf_pars_vertex: logdepthbuf_pars_vertex,
    	logdepthbuf_vertex: logdepthbuf_vertex,
    	map_fragment: map_fragment,
    	map_pars_fragment: map_pars_fragment,
    	map_particle_fragment: map_particle_fragment,
    	map_particle_pars_fragment: map_particle_pars_fragment,
    	metalnessmap_fragment: metalnessmap_fragment,
    	metalnessmap_pars_fragment: metalnessmap_pars_fragment,
    	morphnormal_vertex: morphnormal_vertex,
    	morphtarget_pars_vertex: morphtarget_pars_vertex,
    	morphtarget_vertex: morphtarget_vertex,
    	normal_fragment_begin: normal_fragment_begin,
    	normal_fragment_maps: normal_fragment_maps,
    	normalmap_pars_fragment: normalmap_pars_fragment,
    	clearcoat_normal_fragment_begin: clearcoat_normal_fragment_begin,
    	clearcoat_normal_fragment_maps: clearcoat_normal_fragment_maps,
    	clearcoat_normalmap_pars_fragment: clearcoat_normalmap_pars_fragment,
    	packing: packing,
    	premultiplied_alpha_fragment: premultiplied_alpha_fragment,
    	project_vertex: project_vertex,
    	dithering_fragment: dithering_fragment,
    	dithering_pars_fragment: dithering_pars_fragment,
    	roughnessmap_fragment: roughnessmap_fragment,
    	roughnessmap_pars_fragment: roughnessmap_pars_fragment,
    	shadowmap_pars_fragment: shadowmap_pars_fragment,
    	shadowmap_pars_vertex: shadowmap_pars_vertex,
    	shadowmap_vertex: shadowmap_vertex,
    	shadowmask_pars_fragment: shadowmask_pars_fragment,
    	skinbase_vertex: skinbase_vertex,
    	skinning_pars_vertex: skinning_pars_vertex,
    	skinning_vertex: skinning_vertex,
    	skinnormal_vertex: skinnormal_vertex,
    	specularmap_fragment: specularmap_fragment,
    	specularmap_pars_fragment: specularmap_pars_fragment,
    	tonemapping_fragment: tonemapping_fragment,
    	tonemapping_pars_fragment: tonemapping_pars_fragment,
    	uv_pars_fragment: uv_pars_fragment,
    	uv_pars_vertex: uv_pars_vertex,
    	uv_vertex: uv_vertex,
    	uv2_pars_fragment: uv2_pars_fragment,
    	uv2_pars_vertex: uv2_pars_vertex,
    	uv2_vertex: uv2_vertex,
    	worldpos_vertex: worldpos_vertex,

    	background_frag: background_frag,
    	background_vert: background_vert,
    	cube_frag: cube_frag,
    	cube_vert: cube_vert,
    	depth_frag: depth_frag,
    	depth_vert: depth_vert,
    	distanceRGBA_frag: distanceRGBA_frag,
    	distanceRGBA_vert: distanceRGBA_vert,
    	equirect_frag: equirect_frag,
    	equirect_vert: equirect_vert,
    	linedashed_frag: linedashed_frag,
    	linedashed_vert: linedashed_vert,
    	meshbasic_frag: meshbasic_frag,
    	meshbasic_vert: meshbasic_vert,
    	meshlambert_frag: meshlambert_frag,
    	meshlambert_vert: meshlambert_vert,
    	meshmatcap_frag: meshmatcap_frag,
    	meshmatcap_vert: meshmatcap_vert,
    	meshphong_frag: meshphong_frag,
    	meshphong_vert: meshphong_vert,
    	meshphysical_frag: meshphysical_frag,
    	meshphysical_vert: meshphysical_vert,
    	normal_frag: normal_frag,
    	normal_vert: normal_vert,
    	points_frag: points_frag,
    	points_vert: points_vert,
    	shadow_frag: shadow_frag,
    	shadow_vert: shadow_vert,
    	sprite_frag: sprite_frag,
    	sprite_vert: sprite_vert
    };

    /**
     * Uniforms library for shared webgl shaders
     */

    var UniformsLib = {

    	common: {

    		diffuse: { value: new Color$7( 0xeeeeee ) },
    		opacity: { value: 1.0 },

    		map: { value: null },
    		uvTransform: { value: new Matrix3() },

    		alphaMap: { value: null },

    	},

    	specularmap: {

    		specularMap: { value: null },

    	},

    	envmap: {

    		envMap: { value: null },
    		flipEnvMap: { value: - 1 },
    		reflectivity: { value: 1.0 },
    		refractionRatio: { value: 0.98 },
    		maxMipLevel: { value: 0 }

    	},

    	aomap: {

    		aoMap: { value: null },
    		aoMapIntensity: { value: 1 }

    	},

    	lightmap: {

    		lightMap: { value: null },
    		lightMapIntensity: { value: 1 }

    	},

    	emissivemap: {

    		emissiveMap: { value: null }

    	},

    	bumpmap: {

    		bumpMap: { value: null },
    		bumpScale: { value: 1 }

    	},

    	normalmap: {

    		normalMap: { value: null },
    		normalScale: { value: new Vector2( 1, 1 ) }

    	},

    	displacementmap: {

    		displacementMap: { value: null },
    		displacementScale: { value: 1 },
    		displacementBias: { value: 0 }

    	},

    	roughnessmap: {

    		roughnessMap: { value: null }

    	},

    	metalnessmap: {

    		metalnessMap: { value: null }

    	},

    	gradientmap: {

    		gradientMap: { value: null }

    	},

    	fog: {

    		fogDensity: { value: 0.00025 },
    		fogNear: { value: 1 },
    		fogFar: { value: 2000 },
    		fogColor: { value: new Color$7( 0xffffff ) }

    	},

    	lights: {

    		ambientLightColor: { value: [] },

    		lightProbe: { value: [] },

    		directionalLights: { value: [], properties: {
    			direction: {},
    			color: {},

    			shadow: {},
    			shadowBias: {},
    			shadowRadius: {},
    			shadowMapSize: {}
    		} },

    		directionalShadowMap: { value: [] },
    		directionalShadowMatrix: { value: [] },

    		spotLights: { value: [], properties: {
    			color: {},
    			position: {},
    			direction: {},
    			distance: {},
    			coneCos: {},
    			penumbraCos: {},
    			decay: {},

    			shadow: {},
    			shadowBias: {},
    			shadowRadius: {},
    			shadowMapSize: {}
    		} },

    		spotShadowMap: { value: [] },
    		spotShadowMatrix: { value: [] },

    		pointLights: { value: [], properties: {
    			color: {},
    			position: {},
    			decay: {},
    			distance: {},

    			shadow: {},
    			shadowBias: {},
    			shadowRadius: {},
    			shadowMapSize: {},
    			shadowCameraNear: {},
    			shadowCameraFar: {}
    		} },

    		pointShadowMap: { value: [] },
    		pointShadowMatrix: { value: [] },

    		hemisphereLights: { value: [], properties: {
    			direction: {},
    			skyColor: {},
    			groundColor: {}
    		} },

    		// TODO (abelnation): RectAreaLight BRDF data needs to be moved from example to main src
    		rectAreaLights: { value: [], properties: {
    			color: {},
    			position: {},
    			width: {},
    			height: {}
    		} }

    	},

    	points: {

    		diffuse: { value: new Color$7( 0xeeeeee ) },
    		opacity: { value: 1.0 },
    		size: { value: 1.0 },
    		scale: { value: 1.0 },
    		map: { value: null },
    		uvTransform: { value: new Matrix3() }

    	},

    	sprite: {

    		diffuse: { value: new Color$7( 0xeeeeee ) },
    		opacity: { value: 1.0 },
    		center: { value: new Vector2( 0.5, 0.5 ) },
    		rotation: { value: 0.0 },
    		map: { value: null },
    		uvTransform: { value: new Matrix3() }

    	}

    };

    /**
     * @author alteredq / http://alteredqualia.com/
     * @author mrdoob / http://mrdoob.com/
     * @author mikael emtinger / http://gomo.se/
     */

    var ShaderLib = {

    	basic: {

    		uniforms: mergeUniforms( [
    			UniformsLib.common,
    			UniformsLib.specularmap,
    			UniformsLib.envmap,
    			UniformsLib.aomap,
    			UniformsLib.lightmap,
    			UniformsLib.fog
    		] ),

    		vertexShader: ShaderChunk.meshbasic_vert,
    		fragmentShader: ShaderChunk.meshbasic_frag

    	},

    	lambert: {

    		uniforms: mergeUniforms( [
    			UniformsLib.common,
    			UniformsLib.specularmap,
    			UniformsLib.envmap,
    			UniformsLib.aomap,
    			UniformsLib.lightmap,
    			UniformsLib.emissivemap,
    			UniformsLib.fog,
    			UniformsLib.lights,
    			{
    				emissive: { value: new Color$7( 0x000000 ) }
    			}
    		] ),

    		vertexShader: ShaderChunk.meshlambert_vert,
    		fragmentShader: ShaderChunk.meshlambert_frag

    	},

    	phong: {

    		uniforms: mergeUniforms( [
    			UniformsLib.common,
    			UniformsLib.specularmap,
    			UniformsLib.envmap,
    			UniformsLib.aomap,
    			UniformsLib.lightmap,
    			UniformsLib.emissivemap,
    			UniformsLib.bumpmap,
    			UniformsLib.normalmap,
    			UniformsLib.displacementmap,
    			UniformsLib.gradientmap,
    			UniformsLib.fog,
    			UniformsLib.lights,
    			{
    				emissive: { value: new Color$7( 0x000000 ) },
    				specular: { value: new Color$7( 0x111111 ) },
    				shininess: { value: 30 }
    			}
    		] ),

    		vertexShader: ShaderChunk.meshphong_vert,
    		fragmentShader: ShaderChunk.meshphong_frag

    	},

    	standard: {

    		uniforms: mergeUniforms( [
    			UniformsLib.common,
    			UniformsLib.envmap,
    			UniformsLib.aomap,
    			UniformsLib.lightmap,
    			UniformsLib.emissivemap,
    			UniformsLib.bumpmap,
    			UniformsLib.normalmap,
    			UniformsLib.displacementmap,
    			UniformsLib.roughnessmap,
    			UniformsLib.metalnessmap,
    			UniformsLib.fog,
    			UniformsLib.lights,
    			{
    				emissive: { value: new Color$7( 0x000000 ) },
    				roughness: { value: 0.5 },
    				metalness: { value: 0.5 },
    				envMapIntensity: { value: 1 } // temporary
    			}
    		] ),

    		vertexShader: ShaderChunk.meshphysical_vert,
    		fragmentShader: ShaderChunk.meshphysical_frag

    	},

    	matcap: {

    		uniforms: mergeUniforms( [
    			UniformsLib.common,
    			UniformsLib.bumpmap,
    			UniformsLib.normalmap,
    			UniformsLib.displacementmap,
    			UniformsLib.fog,
    			{
    				matcap: { value: null }
    			}
    		] ),

    		vertexShader: ShaderChunk.meshmatcap_vert,
    		fragmentShader: ShaderChunk.meshmatcap_frag

    	},

    	points: {

    		uniforms: mergeUniforms( [
    			UniformsLib.points,
    			UniformsLib.fog
    		] ),

    		vertexShader: ShaderChunk.points_vert,
    		fragmentShader: ShaderChunk.points_frag

    	},

    	dashed: {

    		uniforms: mergeUniforms( [
    			UniformsLib.common,
    			UniformsLib.fog,
    			{
    				scale: { value: 1 },
    				dashSize: { value: 1 },
    				totalSize: { value: 2 }
    			}
    		] ),

    		vertexShader: ShaderChunk.linedashed_vert,
    		fragmentShader: ShaderChunk.linedashed_frag

    	},

    	depth: {

    		uniforms: mergeUniforms( [
    			UniformsLib.common,
    			UniformsLib.displacementmap
    		] ),

    		vertexShader: ShaderChunk.depth_vert,
    		fragmentShader: ShaderChunk.depth_frag

    	},

    	normal: {

    		uniforms: mergeUniforms( [
    			UniformsLib.common,
    			UniformsLib.bumpmap,
    			UniformsLib.normalmap,
    			UniformsLib.displacementmap,
    			{
    				opacity: { value: 1.0 }
    			}
    		] ),

    		vertexShader: ShaderChunk.normal_vert,
    		fragmentShader: ShaderChunk.normal_frag

    	},

    	sprite: {

    		uniforms: mergeUniforms( [
    			UniformsLib.sprite,
    			UniformsLib.fog
    		] ),

    		vertexShader: ShaderChunk.sprite_vert,
    		fragmentShader: ShaderChunk.sprite_frag

    	},

    	background: {

    		uniforms: {
    			uvTransform: { value: new Matrix3() },
    			t2D: { value: null },
    		},

    		vertexShader: ShaderChunk.background_vert,
    		fragmentShader: ShaderChunk.background_frag

    	},
    	/* -------------------------------------------------------------------------
    	//	Cube map shader
    	 ------------------------------------------------------------------------- */

    	cube: {

    		uniforms: {
    			tCube: { value: null },
    			tFlip: { value: - 1 },
    			opacity: { value: 1.0 }
    		},

    		vertexShader: ShaderChunk.cube_vert,
    		fragmentShader: ShaderChunk.cube_frag

    	},

    	equirect: {

    		uniforms: {
    			tEquirect: { value: null },
    		},

    		vertexShader: ShaderChunk.equirect_vert,
    		fragmentShader: ShaderChunk.equirect_frag

    	},

    	distanceRGBA: {

    		uniforms: mergeUniforms( [
    			UniformsLib.common,
    			UniformsLib.displacementmap,
    			{
    				referencePosition: { value: new Vector3() },
    				nearDistance: { value: 1 },
    				farDistance: { value: 1000 }
    			}
    		] ),

    		vertexShader: ShaderChunk.distanceRGBA_vert,
    		fragmentShader: ShaderChunk.distanceRGBA_frag

    	},

    	shadow: {

    		uniforms: mergeUniforms( [
    			UniformsLib.lights,
    			UniformsLib.fog,
    			{
    				color: { value: new Color$7( 0x00000 ) },
    				opacity: { value: 1.0 }
    			},
    		] ),

    		vertexShader: ShaderChunk.shadow_vert,
    		fragmentShader: ShaderChunk.shadow_frag

    	}

    };

    ShaderLib.physical = {

    	uniforms: mergeUniforms( [
    		ShaderLib.standard.uniforms,
    		{
    			transparency: { value: 0 },
    			clearcoat: { value: 0 },
    			clearcoatRoughness: { value: 0 },
    			sheen: { value: new Color$7( 0x000000 ) },
    			clearcoatNormalScale: { value: new Vector2( 1, 1 ) },
    			clearcoatNormalMap: { value: null },
    		}
    	] ),

    	vertexShader: ShaderChunk.meshphysical_vert,
    	fragmentShader: ShaderChunk.meshphysical_frag

    };

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function WebGLAnimation() {

    	var context = null;
    	var isAnimating = false;
    	var animationLoop = null;

    	function onAnimationFrame( time, frame ) {

    		if ( isAnimating === false ) return;

    		animationLoop( time, frame );

    		context.requestAnimationFrame( onAnimationFrame );

    	}

    	return {

    		start: function () {

    			if ( isAnimating === true ) return;
    			if ( animationLoop === null ) return;

    			context.requestAnimationFrame( onAnimationFrame );

    			isAnimating = true;

    		},

    		stop: function () {

    			isAnimating = false;

    		},

    		setAnimationLoop: function ( callback ) {

    			animationLoop = callback;

    		},

    		setContext: function ( value ) {

    			context = value;

    		}

    	};

    }

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function WebGLAttributes( gl ) {

    	var buffers = new WeakMap();

    	function createBuffer( attribute, bufferType ) {

    		var array = attribute.array;
    		var usage = attribute.dynamic ? 35048 : 35044;

    		var buffer = gl.createBuffer();

    		gl.bindBuffer( bufferType, buffer );
    		gl.bufferData( bufferType, array, usage );

    		attribute.onUploadCallback();

    		var type = 5126;

    		if ( array instanceof Float32Array ) {

    			type = 5126;

    		} else if ( array instanceof Float64Array ) {

    			console.warn( 'THREE.WebGLAttributes: Unsupported data buffer format: Float64Array.' );

    		} else if ( array instanceof Uint16Array ) {

    			type = 5123;

    		} else if ( array instanceof Int16Array ) {

    			type = 5122;

    		} else if ( array instanceof Uint32Array ) {

    			type = 5125;

    		} else if ( array instanceof Int32Array ) {

    			type = 5124;

    		} else if ( array instanceof Int8Array ) {

    			type = 5120;

    		} else if ( array instanceof Uint8Array ) {

    			type = 5121;

    		}

    		return {
    			buffer: buffer,
    			type: type,
    			bytesPerElement: array.BYTES_PER_ELEMENT,
    			version: attribute.version
    		};

    	}

    	function updateBuffer( buffer, attribute, bufferType ) {

    		var array = attribute.array;
    		var updateRange = attribute.updateRange;

    		gl.bindBuffer( bufferType, buffer );

    		if ( attribute.dynamic === false ) {

    			gl.bufferData( bufferType, array, 35044 );

    		} else if ( updateRange.count === - 1 ) {

    			// Not using update ranges

    			gl.bufferSubData( bufferType, 0, array );

    		} else if ( updateRange.count === 0 ) {

    			console.error( 'THREE.WebGLObjects.updateBuffer: dynamic THREE.BufferAttribute marked as needsUpdate but updateRange.count is 0, ensure you are using set methods or updating manually.' );

    		} else {

    			gl.bufferSubData( bufferType, updateRange.offset * array.BYTES_PER_ELEMENT,
    				array.subarray( updateRange.offset, updateRange.offset + updateRange.count ) );

    			updateRange.count = - 1; // reset range

    		}

    	}

    	//

    	function get( attribute ) {

    		if ( attribute.isInterleavedBufferAttribute ) attribute = attribute.data;

    		return buffers.get( attribute );

    	}

    	function remove( attribute ) {

    		if ( attribute.isInterleavedBufferAttribute ) attribute = attribute.data;

    		var data = buffers.get( attribute );

    		if ( data ) {

    			gl.deleteBuffer( data.buffer );

    			buffers.delete( attribute );

    		}

    	}

    	function update( attribute, bufferType ) {

    		if ( attribute.isInterleavedBufferAttribute ) attribute = attribute.data;

    		var data = buffers.get( attribute );

    		if ( data === undefined ) {

    			buffers.set( attribute, createBuffer( attribute, bufferType ) );

    		} else if ( data.version < attribute.version ) {

    			updateBuffer( data.buffer, attribute, bufferType );

    			data.version = attribute.version;

    		}

    	}

    	return {

    		get: get,
    		remove: remove,
    		update: update

    	};

    }

    /**
     * @author mrdoob / http://mrdoob.com/
     * @author Mugen87 / https://github.com/Mugen87
     */

    // PlaneGeometry

    function PlaneGeometry( width, height, widthSegments, heightSegments ) {

    	Geometry.call( this );

    	this.type = 'PlaneGeometry';

    	this.parameters = {
    		width: width,
    		height: height,
    		widthSegments: widthSegments,
    		heightSegments: heightSegments
    	};

    	this.fromBufferGeometry( new PlaneBufferGeometry( width, height, widthSegments, heightSegments ) );
    	this.mergeVertices();

    }

    PlaneGeometry.prototype = Object.create( Geometry.prototype );
    PlaneGeometry.prototype.constructor = PlaneGeometry;

    // PlaneBufferGeometry

    function PlaneBufferGeometry( width, height, widthSegments, heightSegments ) {

    	BufferGeometry.call( this );

    	this.type = 'PlaneBufferGeometry';

    	this.parameters = {
    		width: width,
    		height: height,
    		widthSegments: widthSegments,
    		heightSegments: heightSegments
    	};

    	width = width || 1;
    	height = height || 1;

    	var width_half = width / 2;
    	var height_half = height / 2;

    	var gridX = Math.floor( widthSegments ) || 1;
    	var gridY = Math.floor( heightSegments ) || 1;

    	var gridX1 = gridX + 1;
    	var gridY1 = gridY + 1;

    	var segment_width = width / gridX;
    	var segment_height = height / gridY;

    	var ix, iy;

    	// buffers

    	var indices = [];
    	var vertices = [];
    	var normals = [];
    	var uvs = [];

    	// generate vertices, normals and uvs

    	for ( iy = 0; iy < gridY1; iy ++ ) {

    		var y = iy * segment_height - height_half;

    		for ( ix = 0; ix < gridX1; ix ++ ) {

    			var x = ix * segment_width - width_half;

    			vertices.push( x, - y, 0 );

    			normals.push( 0, 0, 1 );

    			uvs.push( ix / gridX );
    			uvs.push( 1 - ( iy / gridY ) );

    		}

    	}

    	// indices

    	for ( iy = 0; iy < gridY; iy ++ ) {

    		for ( ix = 0; ix < gridX; ix ++ ) {

    			var a = ix + gridX1 * iy;
    			var b = ix + gridX1 * ( iy + 1 );
    			var c = ( ix + 1 ) + gridX1 * ( iy + 1 );
    			var d = ( ix + 1 ) + gridX1 * iy;

    			// faces

    			indices.push( a, b, d );
    			indices.push( b, c, d );

    		}

    	}

    	// build geometry

    	this.setIndex( indices );
    	this.addAttribute( 'position', new Float32BufferAttribute( vertices, 3 ) );
    	this.addAttribute( 'normal', new Float32BufferAttribute( normals, 3 ) );
    	this.addAttribute( 'uv', new Float32BufferAttribute( uvs, 2 ) );

    }

    PlaneBufferGeometry.prototype = Object.create( BufferGeometry.prototype );
    PlaneBufferGeometry.prototype.constructor = PlaneBufferGeometry;

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function WebGLBackground( renderer, state, objects, premultipliedAlpha ) {

    	var clearColor = new Color$7( 0x000000 );
    	var clearAlpha = 0;

    	var planeMesh;
    	var boxMesh;
    	// Store the current background texture and its `version`
    	// so we can recompile the material accordingly.
    	var currentBackground = null;
    	var currentBackgroundVersion = 0;

    	function render( renderList, scene, camera, forceClear ) {

    		var background = scene.background;

    		// Ignore background in AR
    		// TODO: Reconsider this.

    		var vr = renderer.vr;
    		var session = vr.getSession && vr.getSession();

    		if ( session && session.environmentBlendMode === 'additive' ) {

    			background = null;

    		}

    		if ( background === null ) {

    			setClear( clearColor, clearAlpha );
    			currentBackground = null;
    			currentBackgroundVersion = 0;

    		} else if ( background && background.isColor ) {

    			setClear( background, 1 );
    			forceClear = true;
    			currentBackground = null;
    			currentBackgroundVersion = 0;

    		}

    		if ( renderer.autoClear || forceClear ) {

    			renderer.clear( renderer.autoClearColor, renderer.autoClearDepth, renderer.autoClearStencil );

    		}

    		if ( background && ( background.isCubeTexture || background.isWebGLRenderTargetCube ) ) {

    			if ( boxMesh === undefined ) {

    				boxMesh = new Mesh(
    					new BoxBufferGeometry( 1, 1, 1 ),
    					new ShaderMaterial( {
    						type: 'BackgroundCubeMaterial',
    						uniforms: cloneUniforms( ShaderLib.cube.uniforms ),
    						vertexShader: ShaderLib.cube.vertexShader,
    						fragmentShader: ShaderLib.cube.fragmentShader,
    						side: BackSide,
    						depthTest: false,
    						depthWrite: false,
    						fog: false
    					} )
    				);

    				boxMesh.geometry.removeAttribute( 'normal' );
    				boxMesh.geometry.removeAttribute( 'uv' );

    				boxMesh.onBeforeRender = function ( renderer, scene, camera ) {

    					this.matrixWorld.copyPosition( camera.matrixWorld );

    				};

    				// enable code injection for non-built-in material
    				Object.defineProperty( boxMesh.material, 'map', {

    					get: function () {

    						return this.uniforms.tCube.value;

    					}

    				} );

    				objects.update( boxMesh );

    			}

    			var texture = background.isWebGLRenderTargetCube ? background.texture : background;
    			boxMesh.material.uniforms.tCube.value = texture;
    			boxMesh.material.uniforms.tFlip.value = ( background.isWebGLRenderTargetCube ) ? 1 : - 1;

    			if ( currentBackground !== background ||
    			     currentBackgroundVersion !== texture.version ) {

    				boxMesh.material.needsUpdate = true;

    				currentBackground = background;
    				currentBackgroundVersion = texture.version;

    			}

    			// push to the pre-sorted opaque render list
    			renderList.unshift( boxMesh, boxMesh.geometry, boxMesh.material, 0, 0, null );

    		} else if ( background && background.isTexture ) {

    			if ( planeMesh === undefined ) {

    				planeMesh = new Mesh(
    					new PlaneBufferGeometry( 2, 2 ),
    					new ShaderMaterial( {
    						type: 'BackgroundMaterial',
    						uniforms: cloneUniforms( ShaderLib.background.uniforms ),
    						vertexShader: ShaderLib.background.vertexShader,
    						fragmentShader: ShaderLib.background.fragmentShader,
    						side: FrontSide,
    						depthTest: false,
    						depthWrite: false,
    						fog: false
    					} )
    				);

    				planeMesh.geometry.removeAttribute( 'normal' );

    				// enable code injection for non-built-in material
    				Object.defineProperty( planeMesh.material, 'map', {

    					get: function () {

    						return this.uniforms.t2D.value;

    					}

    				} );

    				objects.update( planeMesh );

    			}

    			planeMesh.material.uniforms.t2D.value = background;

    			if ( background.matrixAutoUpdate === true ) {

    				background.updateMatrix();

    			}

    			planeMesh.material.uniforms.uvTransform.value.copy( background.matrix );

    			if ( currentBackground !== background ||
    				   currentBackgroundVersion !== background.version ) {

    				planeMesh.material.needsUpdate = true;

    				currentBackground = background;
    				currentBackgroundVersion = background.version;

    			}


    			// push to the pre-sorted opaque render list
    			renderList.unshift( planeMesh, planeMesh.geometry, planeMesh.material, 0, 0, null );

    		}

    	}

    	function setClear( color, alpha ) {

    		state.buffers.color.setClear( color.r, color.g, color.b, alpha, premultipliedAlpha );

    	}

    	return {

    		getClearColor: function () {

    			return clearColor;

    		},
    		setClearColor: function ( color, alpha ) {

    			clearColor.set( color );
    			clearAlpha = alpha !== undefined ? alpha : 1;
    			setClear( clearColor, clearAlpha );

    		},
    		getClearAlpha: function () {

    			return clearAlpha;

    		},
    		setClearAlpha: function ( alpha ) {

    			clearAlpha = alpha;
    			setClear( clearColor, clearAlpha );

    		},
    		render: render

    	};

    }

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function WebGLBufferRenderer( gl, extensions, info, capabilities ) {

    	var mode;

    	function setMode( value ) {

    		mode = value;

    	}

    	function render( start, count ) {

    		gl.drawArrays( mode, start, count );

    		info.update( count, mode );

    	}

    	function renderInstances( geometry, start, count ) {

    		var extension, methodName;

    		if ( capabilities.isWebGL2 ) {

    			extension = gl;
    			methodName = 'drawArraysInstanced';

    		} else {

    			extension = extensions.get( 'ANGLE_instanced_arrays' );
    			methodName = 'drawArraysInstancedANGLE';

    			if ( extension === null ) {

    				console.error( 'THREE.WebGLBufferRenderer: using THREE.InstancedBufferGeometry but hardware does not support extension ANGLE_instanced_arrays.' );
    				return;

    			}

    		}

    		extension[ methodName ]( mode, start, count, geometry.maxInstancedCount );

    		info.update( count, mode, geometry.maxInstancedCount );

    	}

    	//

    	this.setMode = setMode;
    	this.render = render;
    	this.renderInstances = renderInstances;

    }

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function WebGLCapabilities( gl, extensions, parameters ) {

    	var maxAnisotropy;

    	function getMaxAnisotropy() {

    		if ( maxAnisotropy !== undefined ) return maxAnisotropy;

    		var extension = extensions.get( 'EXT_texture_filter_anisotropic' );

    		if ( extension !== null ) {

    			maxAnisotropy = gl.getParameter( extension.MAX_TEXTURE_MAX_ANISOTROPY_EXT );

    		} else {

    			maxAnisotropy = 0;

    		}

    		return maxAnisotropy;

    	}

    	function getMaxPrecision( precision ) {

    		if ( precision === 'highp' ) {

    			if ( gl.getShaderPrecisionFormat( 35633, 36338 ).precision > 0 &&
    			     gl.getShaderPrecisionFormat( 35632, 36338 ).precision > 0 ) {

    				return 'highp';

    			}

    			precision = 'mediump';

    		}

    		if ( precision === 'mediump' ) {

    			if ( gl.getShaderPrecisionFormat( 35633, 36337 ).precision > 0 &&
    			     gl.getShaderPrecisionFormat( 35632, 36337 ).precision > 0 ) {

    				return 'mediump';

    			}

    		}

    		return 'lowp';

    	}

    	var isWebGL2 = typeof WebGL2RenderingContext !== 'undefined' && gl instanceof WebGL2RenderingContext;

    	var precision = parameters.precision !== undefined ? parameters.precision : 'highp';
    	var maxPrecision = getMaxPrecision( precision );

    	if ( maxPrecision !== precision ) {

    		console.warn( 'THREE.WebGLRenderer:', precision, 'not supported, using', maxPrecision, 'instead.' );
    		precision = maxPrecision;

    	}

    	var logarithmicDepthBuffer = parameters.logarithmicDepthBuffer === true;

    	var maxTextures = gl.getParameter( 34930 );
    	var maxVertexTextures = gl.getParameter( 35660 );
    	var maxTextureSize = gl.getParameter( 3379 );
    	var maxCubemapSize = gl.getParameter( 34076 );

    	var maxAttributes = gl.getParameter( 34921 );
    	var maxVertexUniforms = gl.getParameter( 36347 );
    	var maxVaryings = gl.getParameter( 36348 );
    	var maxFragmentUniforms = gl.getParameter( 36349 );

    	var vertexTextures = maxVertexTextures > 0;
    	var floatFragmentTextures = isWebGL2 || !! extensions.get( 'OES_texture_float' );
    	var floatVertexTextures = vertexTextures && floatFragmentTextures;

    	var maxSamples = isWebGL2 ? gl.getParameter( 36183 ) : 0;

    	return {

    		isWebGL2: isWebGL2,

    		getMaxAnisotropy: getMaxAnisotropy,
    		getMaxPrecision: getMaxPrecision,

    		precision: precision,
    		logarithmicDepthBuffer: logarithmicDepthBuffer,

    		maxTextures: maxTextures,
    		maxVertexTextures: maxVertexTextures,
    		maxTextureSize: maxTextureSize,
    		maxCubemapSize: maxCubemapSize,

    		maxAttributes: maxAttributes,
    		maxVertexUniforms: maxVertexUniforms,
    		maxVaryings: maxVaryings,
    		maxFragmentUniforms: maxFragmentUniforms,

    		vertexTextures: vertexTextures,
    		floatFragmentTextures: floatFragmentTextures,
    		floatVertexTextures: floatVertexTextures,

    		maxSamples: maxSamples

    	};

    }

    /**
     * @author tschw
     */

    function WebGLClipping() {

    	var scope = this,

    		globalState = null,
    		numGlobalPlanes = 0,
    		localClippingEnabled = false,
    		renderingShadows = false,

    		plane = new Plane(),
    		viewNormalMatrix = new Matrix3(),

    		uniform = { value: null, needsUpdate: false };

    	this.uniform = uniform;
    	this.numPlanes = 0;
    	this.numIntersection = 0;

    	this.init = function ( planes, enableLocalClipping, camera ) {

    		var enabled =
    			planes.length !== 0 ||
    			enableLocalClipping ||
    			// enable state of previous frame - the clipping code has to
    			// run another frame in order to reset the state:
    			numGlobalPlanes !== 0 ||
    			localClippingEnabled;

    		localClippingEnabled = enableLocalClipping;

    		globalState = projectPlanes( planes, camera, 0 );
    		numGlobalPlanes = planes.length;

    		return enabled;

    	};

    	this.beginShadows = function () {

    		renderingShadows = true;
    		projectPlanes( null );

    	};

    	this.endShadows = function () {

    		renderingShadows = false;
    		resetGlobalState();

    	};

    	this.setState = function ( planes, clipIntersection, clipShadows, camera, cache, fromCache ) {

    		if ( ! localClippingEnabled || planes === null || planes.length === 0 || renderingShadows && ! clipShadows ) {

    			// there's no local clipping

    			if ( renderingShadows ) {

    				// there's no global clipping

    				projectPlanes( null );

    			} else {

    				resetGlobalState();

    			}

    		} else {

    			var nGlobal = renderingShadows ? 0 : numGlobalPlanes,
    				lGlobal = nGlobal * 4,

    				dstArray = cache.clippingState || null;

    			uniform.value = dstArray; // ensure unique state

    			dstArray = projectPlanes( planes, camera, lGlobal, fromCache );

    			for ( var i = 0; i !== lGlobal; ++ i ) {

    				dstArray[ i ] = globalState[ i ];

    			}

    			cache.clippingState = dstArray;
    			this.numIntersection = clipIntersection ? this.numPlanes : 0;
    			this.numPlanes += nGlobal;

    		}


    	};

    	function resetGlobalState() {

    		if ( uniform.value !== globalState ) {

    			uniform.value = globalState;
    			uniform.needsUpdate = numGlobalPlanes > 0;

    		}

    		scope.numPlanes = numGlobalPlanes;
    		scope.numIntersection = 0;

    	}

    	function projectPlanes( planes, camera, dstOffset, skipTransform ) {

    		var nPlanes = planes !== null ? planes.length : 0,
    			dstArray = null;

    		if ( nPlanes !== 0 ) {

    			dstArray = uniform.value;

    			if ( skipTransform !== true || dstArray === null ) {

    				var flatSize = dstOffset + nPlanes * 4,
    					viewMatrix = camera.matrixWorldInverse;

    				viewNormalMatrix.getNormalMatrix( viewMatrix );

    				if ( dstArray === null || dstArray.length < flatSize ) {

    					dstArray = new Float32Array( flatSize );

    				}

    				for ( var i = 0, i4 = dstOffset; i !== nPlanes; ++ i, i4 += 4 ) {

    					plane.copy( planes[ i ] ).applyMatrix4( viewMatrix, viewNormalMatrix );

    					plane.normal.toArray( dstArray, i4 );
    					dstArray[ i4 + 3 ] = plane.constant;

    				}

    			}

    			uniform.value = dstArray;
    			uniform.needsUpdate = true;

    		}

    		scope.numPlanes = nPlanes;

    		return dstArray;

    	}

    }

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function WebGLExtensions( gl ) {

    	var extensions = {};

    	return {

    		get: function ( name ) {

    			if ( extensions[ name ] !== undefined ) {

    				return extensions[ name ];

    			}

    			var extension;

    			switch ( name ) {

    				case 'WEBGL_depth_texture':
    					extension = gl.getExtension( 'WEBGL_depth_texture' ) || gl.getExtension( 'MOZ_WEBGL_depth_texture' ) || gl.getExtension( 'WEBKIT_WEBGL_depth_texture' );
    					break;

    				case 'EXT_texture_filter_anisotropic':
    					extension = gl.getExtension( 'EXT_texture_filter_anisotropic' ) || gl.getExtension( 'MOZ_EXT_texture_filter_anisotropic' ) || gl.getExtension( 'WEBKIT_EXT_texture_filter_anisotropic' );
    					break;

    				case 'WEBGL_compressed_texture_s3tc':
    					extension = gl.getExtension( 'WEBGL_compressed_texture_s3tc' ) || gl.getExtension( 'MOZ_WEBGL_compressed_texture_s3tc' ) || gl.getExtension( 'WEBKIT_WEBGL_compressed_texture_s3tc' );
    					break;

    				case 'WEBGL_compressed_texture_pvrtc':
    					extension = gl.getExtension( 'WEBGL_compressed_texture_pvrtc' ) || gl.getExtension( 'WEBKIT_WEBGL_compressed_texture_pvrtc' );
    					break;

    				default:
    					extension = gl.getExtension( name );

    			}

    			if ( extension === null ) {

    				console.warn( 'THREE.WebGLRenderer: ' + name + ' extension not supported.' );

    			}

    			extensions[ name ] = extension;

    			return extension;

    		}

    	};

    }

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function WebGLGeometries( gl, attributes, info ) {

    	var geometries = new WeakMap();
    	var wireframeAttributes = new WeakMap();

    	function onGeometryDispose( event ) {

    		var geometry = event.target;
    		var buffergeometry = geometries.get( geometry );

    		if ( buffergeometry.index !== null ) {

    			attributes.remove( buffergeometry.index );

    		}

    		for ( var name in buffergeometry.attributes ) {

    			attributes.remove( buffergeometry.attributes[ name ] );

    		}

    		geometry.removeEventListener( 'dispose', onGeometryDispose );

    		geometries.delete( geometry );

    		var attribute = wireframeAttributes.get( buffergeometry );

    		if ( attribute ) {

    			attributes.remove( attribute );
    			wireframeAttributes.delete( buffergeometry );

    		}

    		//

    		info.memory.geometries --;

    	}

    	function get( object, geometry ) {

    		var buffergeometry = geometries.get( geometry );

    		if ( buffergeometry ) return buffergeometry;

    		geometry.addEventListener( 'dispose', onGeometryDispose );

    		if ( geometry.isBufferGeometry ) {

    			buffergeometry = geometry;

    		} else if ( geometry.isGeometry ) {

    			if ( geometry._bufferGeometry === undefined ) {

    				geometry._bufferGeometry = new BufferGeometry().setFromObject( object );

    			}

    			buffergeometry = geometry._bufferGeometry;

    		}

    		geometries.set( geometry, buffergeometry );

    		info.memory.geometries ++;

    		return buffergeometry;

    	}

    	function update( geometry ) {

    		var index = geometry.index;
    		var geometryAttributes = geometry.attributes;

    		if ( index !== null ) {

    			attributes.update( index, 34963 );

    		}

    		for ( var name in geometryAttributes ) {

    			attributes.update( geometryAttributes[ name ], 34962 );

    		}

    		// morph targets

    		var morphAttributes = geometry.morphAttributes;

    		for ( var name in morphAttributes ) {

    			var array = morphAttributes[ name ];

    			for ( var i = 0, l = array.length; i < l; i ++ ) {

    				attributes.update( array[ i ], 34962 );

    			}

    		}

    	}

    	function updateWireframeAttribute( geometry ) {

    		var indices = [];

    		var geometryIndex = geometry.index;
    		var geometryPosition = geometry.attributes.position;
    		var version = 0;

    		if ( geometryIndex !== null ) {

    			var array = geometryIndex.array;
    			version = geometryIndex.version;

    			for ( var i = 0, l = array.length; i < l; i += 3 ) {

    				var a = array[ i + 0 ];
    				var b = array[ i + 1 ];
    				var c = array[ i + 2 ];

    				indices.push( a, b, b, c, c, a );

    			}

    		} else {

    			var array = geometryPosition.array;
    			version = geometryPosition.version;

    			for ( var i = 0, l = ( array.length / 3 ) - 1; i < l; i += 3 ) {

    				var a = i + 0;
    				var b = i + 1;
    				var c = i + 2;

    				indices.push( a, b, b, c, c, a );

    			}

    		}

    		var attribute = new ( arrayMax( indices ) > 65535 ? Uint32BufferAttribute : Uint16BufferAttribute )( indices, 1 );
    		attribute.version = version;

    		attributes.update( attribute, 34963 );

    		//

    		var previousAttribute = wireframeAttributes.get( geometry );

    		if ( previousAttribute ) attributes.remove( previousAttribute );

    		//

    		wireframeAttributes.set( geometry, attribute );

    	}

    	function getWireframeAttribute( geometry ) {

    		var currentAttribute = wireframeAttributes.get( geometry );

    		if ( currentAttribute ) {

    			var geometryIndex = geometry.index;

    			if ( geometryIndex !== null ) {

    				// if the attribute is obsolete, create a new one

    				if ( currentAttribute.version < geometryIndex.version ) {

    					updateWireframeAttribute( geometry );

    				}

    			}

    		} else {

    			updateWireframeAttribute( geometry );

    		}

    		return wireframeAttributes.get( geometry );

    	}

    	return {

    		get: get,
    		update: update,

    		getWireframeAttribute: getWireframeAttribute

    	};

    }

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function WebGLIndexedBufferRenderer( gl, extensions, info, capabilities ) {

    	var mode;

    	function setMode( value ) {

    		mode = value;

    	}

    	var type, bytesPerElement;

    	function setIndex( value ) {

    		type = value.type;
    		bytesPerElement = value.bytesPerElement;

    	}

    	function render( start, count ) {

    		gl.drawElements( mode, count, type, start * bytesPerElement );

    		info.update( count, mode );

    	}

    	function renderInstances( geometry, start, count ) {

    		var extension, methodName;

    		if ( capabilities.isWebGL2 ) {

    			extension = gl;
    			methodName = 'drawElementsInstanced';

    		} else {

    			extension = extensions.get( 'ANGLE_instanced_arrays' );
    			methodName = 'drawElementsInstancedANGLE';

    			if ( extension === null ) {

    				console.error( 'THREE.WebGLIndexedBufferRenderer: using THREE.InstancedBufferGeometry but hardware does not support extension ANGLE_instanced_arrays.' );
    				return;

    			}

    		}

    		extension[ methodName ]( mode, count, type, start * bytesPerElement, geometry.maxInstancedCount );

    		info.update( count, mode, geometry.maxInstancedCount );

    	}

    	//

    	this.setMode = setMode;
    	this.setIndex = setIndex;
    	this.render = render;
    	this.renderInstances = renderInstances;

    }

    /**
     * @author Mugen87 / https://github.com/Mugen87
     */

    function WebGLInfo( gl ) {

    	var memory = {
    		geometries: 0,
    		textures: 0
    	};

    	var render = {
    		frame: 0,
    		calls: 0,
    		triangles: 0,
    		points: 0,
    		lines: 0
    	};

    	function update( count, mode, instanceCount ) {

    		instanceCount = instanceCount || 1;

    		render.calls ++;

    		switch ( mode ) {

    			case 4:
    				render.triangles += instanceCount * ( count / 3 );
    				break;

    			case 5:
    			case 6:
    				render.triangles += instanceCount * ( count - 2 );
    				break;

    			case 1:
    				render.lines += instanceCount * ( count / 2 );
    				break;

    			case 3:
    				render.lines += instanceCount * ( count - 1 );
    				break;

    			case 2:
    				render.lines += instanceCount * count;
    				break;

    			case 0:
    				render.points += instanceCount * count;
    				break;

    			default:
    				console.error( 'THREE.WebGLInfo: Unknown draw mode:', mode );
    				break;

    		}

    	}

    	function reset() {

    		render.frame ++;
    		render.calls = 0;
    		render.triangles = 0;
    		render.points = 0;
    		render.lines = 0;

    	}

    	return {
    		memory: memory,
    		render: render,
    		programs: null,
    		autoReset: true,
    		reset: reset,
    		update: update
    	};

    }

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function absNumericalSort( a, b ) {

    	return Math.abs( b[ 1 ] ) - Math.abs( a[ 1 ] );

    }

    function WebGLMorphtargets( gl ) {

    	var influencesList = {};
    	var morphInfluences = new Float32Array( 8 );

    	function update( object, geometry, material, program ) {

    		var objectInfluences = object.morphTargetInfluences;

    		var length = objectInfluences.length;

    		var influences = influencesList[ geometry.id ];

    		if ( influences === undefined ) {

    			// initialise list

    			influences = [];

    			for ( var i = 0; i < length; i ++ ) {

    				influences[ i ] = [ i, 0 ];

    			}

    			influencesList[ geometry.id ] = influences;

    		}

    		var morphTargets = material.morphTargets && geometry.morphAttributes.position;
    		var morphNormals = material.morphNormals && geometry.morphAttributes.normal;

    		// Remove current morphAttributes

    		for ( var i = 0; i < length; i ++ ) {

    			var influence = influences[ i ];

    			if ( influence[ 1 ] !== 0 ) {

    				if ( morphTargets ) geometry.removeAttribute( 'morphTarget' + i );
    				if ( morphNormals ) geometry.removeAttribute( 'morphNormal' + i );

    			}

    		}

    		// Collect influences

    		for ( var i = 0; i < length; i ++ ) {

    			var influence = influences[ i ];

    			influence[ 0 ] = i;
    			influence[ 1 ] = objectInfluences[ i ];

    		}

    		influences.sort( absNumericalSort );

    		// Add morphAttributes

    		for ( var i = 0; i < 8; i ++ ) {

    			var influence = influences[ i ];

    			if ( influence ) {

    				var index = influence[ 0 ];
    				var value = influence[ 1 ];

    				if ( value ) {

    					if ( morphTargets ) geometry.addAttribute( 'morphTarget' + i, morphTargets[ index ] );
    					if ( morphNormals ) geometry.addAttribute( 'morphNormal' + i, morphNormals[ index ] );

    					morphInfluences[ i ] = value;
    					continue;

    				}

    			}

    			morphInfluences[ i ] = 0;

    		}

    		program.getUniforms().setValue( gl, 'morphTargetInfluences', morphInfluences );

    	}

    	return {

    		update: update

    	};

    }

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function WebGLObjects( geometries, info ) {

    	var updateList = {};

    	function update( object ) {

    		var frame = info.render.frame;

    		var geometry = object.geometry;
    		var buffergeometry = geometries.get( object, geometry );

    		// Update once per frame

    		if ( updateList[ buffergeometry.id ] !== frame ) {

    			if ( geometry.isGeometry ) {

    				buffergeometry.updateFromObject( object );

    			}

    			geometries.update( buffergeometry );

    			updateList[ buffergeometry.id ] = frame;

    		}

    		return buffergeometry;

    	}

    	function dispose() {

    		updateList = {};

    	}

    	return {

    		update: update,
    		dispose: dispose

    	};

    }

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function CubeTexture( images, mapping, wrapS, wrapT, magFilter, minFilter, format, type, anisotropy, encoding ) {

    	images = images !== undefined ? images : [];
    	mapping = mapping !== undefined ? mapping : CubeReflectionMapping;
    	format = format !== undefined ? format : RGBFormat;

    	Texture.call( this, images, mapping, wrapS, wrapT, magFilter, minFilter, format, type, anisotropy, encoding );

    	this.flipY = false;

    }

    CubeTexture.prototype = Object.create( Texture.prototype );
    CubeTexture.prototype.constructor = CubeTexture;

    CubeTexture.prototype.isCubeTexture = true;

    Object.defineProperty( CubeTexture.prototype, 'images', {

    	get: function () {

    		return this.image;

    	},

    	set: function ( value ) {

    		this.image = value;

    	}

    } );

    /**
     * @author Takahiro https://github.com/takahirox
     */

    function DataTexture2DArray( data, width, height, depth ) {

    	Texture.call( this, null );

    	this.image = { data: data, width: width, height: height, depth: depth };

    	this.magFilter = NearestFilter;
    	this.minFilter = NearestFilter;

    	this.wrapR = ClampToEdgeWrapping;

    	this.generateMipmaps = false;
    	this.flipY = false;

    }

    DataTexture2DArray.prototype = Object.create( Texture.prototype );
    DataTexture2DArray.prototype.constructor = DataTexture2DArray;
    DataTexture2DArray.prototype.isDataTexture2DArray = true;

    /**
     * @author Artur Trzesiok
     */

    function DataTexture3D( data, width, height, depth ) {

    	// We're going to add .setXXX() methods for setting properties later.
    	// Users can still set in DataTexture3D directly.
    	//
    	//	var texture = new THREE.DataTexture3D( data, width, height, depth );
    	// 	texture.anisotropy = 16;
    	//
    	// See #14839

    	Texture.call( this, null );

    	this.image = { data: data, width: width, height: height, depth: depth };

    	this.magFilter = NearestFilter;
    	this.minFilter = NearestFilter;

    	this.wrapR = ClampToEdgeWrapping;

    	this.generateMipmaps = false;
    	this.flipY = false;

    }

    DataTexture3D.prototype = Object.create( Texture.prototype );
    DataTexture3D.prototype.constructor = DataTexture3D;
    DataTexture3D.prototype.isDataTexture3D = true;

    /**
     * @author tschw
     * @author Mugen87 / https://github.com/Mugen87
     * @author mrdoob / http://mrdoob.com/
     *
     * Uniforms of a program.
     * Those form a tree structure with a special top-level container for the root,
     * which you get by calling 'new WebGLUniforms( gl, program )'.
     *
     *
     * Properties of inner nodes including the top-level container:
     *
     * .seq - array of nested uniforms
     * .map - nested uniforms by name
     *
     *
     * Methods of all nodes except the top-level container:
     *
     * .setValue( gl, value, [textures] )
     *
     * 		uploads a uniform value(s)
     *  	the 'textures' parameter is needed for sampler uniforms
     *
     *
     * Static methods of the top-level container (textures factorizations):
     *
     * .upload( gl, seq, values, textures )
     *
     * 		sets uniforms in 'seq' to 'values[id].value'
     *
     * .seqWithValue( seq, values ) : filteredSeq
     *
     * 		filters 'seq' entries with corresponding entry in values
     *
     *
     * Methods of the top-level container (textures factorizations):
     *
     * .setValue( gl, name, value, textures )
     *
     * 		sets uniform with  name 'name' to 'value'
     *
     * .setOptional( gl, obj, prop )
     *
     * 		like .set for an optional property of the object
     *
     */

    var emptyTexture = new Texture();
    var emptyTexture2dArray = new DataTexture2DArray();
    var emptyTexture3d = new DataTexture3D();
    var emptyCubeTexture = new CubeTexture();

    // --- Utilities ---

    // Array Caches (provide typed arrays for temporary by size)

    var arrayCacheF32 = [];
    var arrayCacheI32 = [];

    // Float32Array caches used for uploading Matrix uniforms

    var mat4array = new Float32Array( 16 );
    var mat3array = new Float32Array( 9 );
    var mat2array = new Float32Array( 4 );

    // Flattening for arrays of vectors and matrices

    function flatten$1( array, nBlocks, blockSize ) {

    	var firstElem = array[ 0 ];

    	if ( firstElem <= 0 || firstElem > 0 ) return array;
    	// unoptimized: ! isNaN( firstElem )
    	// see http://jacksondunstan.com/articles/983

    	var n = nBlocks * blockSize,
    		r = arrayCacheF32[ n ];

    	if ( r === undefined ) {

    		r = new Float32Array( n );
    		arrayCacheF32[ n ] = r;

    	}

    	if ( nBlocks !== 0 ) {

    		firstElem.toArray( r, 0 );

    		for ( var i = 1, offset = 0; i !== nBlocks; ++ i ) {

    			offset += blockSize;
    			array[ i ].toArray( r, offset );

    		}

    	}

    	return r;

    }

    function arraysEqual( a, b ) {

    	if ( a.length !== b.length ) return false;

    	for ( var i = 0, l = a.length; i < l; i ++ ) {

    		if ( a[ i ] !== b[ i ] ) return false;

    	}

    	return true;

    }

    function copyArray$1( a, b ) {

    	for ( var i = 0, l = b.length; i < l; i ++ ) {

    		a[ i ] = b[ i ];

    	}

    }

    // Texture unit allocation

    function allocTexUnits( textures, n ) {

    	var r = arrayCacheI32[ n ];

    	if ( r === undefined ) {

    		r = new Int32Array( n );
    		arrayCacheI32[ n ] = r;

    	}

    	for ( var i = 0; i !== n; ++ i )
    		r[ i ] = textures.allocateTextureUnit();

    	return r;

    }

    // --- Setters ---

    // Note: Defining these methods externally, because they come in a bunch
    // and this way their names minify.

    // Single scalar

    function setValueV1f( gl, v ) {

    	var cache = this.cache;

    	if ( cache[ 0 ] === v ) return;

    	gl.uniform1f( this.addr, v );

    	cache[ 0 ] = v;

    }

    // Single float vector (from flat array or THREE.VectorN)

    function setValueV2f( gl, v ) {

    	var cache = this.cache;

    	if ( v.x !== undefined ) {

    		if ( cache[ 0 ] !== v.x || cache[ 1 ] !== v.y ) {

    			gl.uniform2f( this.addr, v.x, v.y );

    			cache[ 0 ] = v.x;
    			cache[ 1 ] = v.y;

    		}

    	} else {

    		if ( arraysEqual( cache, v ) ) return;

    		gl.uniform2fv( this.addr, v );

    		copyArray$1( cache, v );

    	}

    }

    function setValueV3f( gl, v ) {

    	var cache = this.cache;

    	if ( v.x !== undefined ) {

    		if ( cache[ 0 ] !== v.x || cache[ 1 ] !== v.y || cache[ 2 ] !== v.z ) {

    			gl.uniform3f( this.addr, v.x, v.y, v.z );

    			cache[ 0 ] = v.x;
    			cache[ 1 ] = v.y;
    			cache[ 2 ] = v.z;

    		}

    	} else if ( v.r !== undefined ) {

    		if ( cache[ 0 ] !== v.r || cache[ 1 ] !== v.g || cache[ 2 ] !== v.b ) {

    			gl.uniform3f( this.addr, v.r, v.g, v.b );

    			cache[ 0 ] = v.r;
    			cache[ 1 ] = v.g;
    			cache[ 2 ] = v.b;

    		}

    	} else {

    		if ( arraysEqual( cache, v ) ) return;

    		gl.uniform3fv( this.addr, v );

    		copyArray$1( cache, v );

    	}

    }

    function setValueV4f( gl, v ) {

    	var cache = this.cache;

    	if ( v.x !== undefined ) {

    		if ( cache[ 0 ] !== v.x || cache[ 1 ] !== v.y || cache[ 2 ] !== v.z || cache[ 3 ] !== v.w ) {

    			gl.uniform4f( this.addr, v.x, v.y, v.z, v.w );

    			cache[ 0 ] = v.x;
    			cache[ 1 ] = v.y;
    			cache[ 2 ] = v.z;
    			cache[ 3 ] = v.w;

    		}

    	} else {

    		if ( arraysEqual( cache, v ) ) return;

    		gl.uniform4fv( this.addr, v );

    		copyArray$1( cache, v );

    	}

    }

    // Single matrix (from flat array or MatrixN)

    function setValueM2( gl, v ) {

    	var cache = this.cache;
    	var elements = v.elements;

    	if ( elements === undefined ) {

    		if ( arraysEqual( cache, v ) ) return;

    		gl.uniformMatrix2fv( this.addr, false, v );

    		copyArray$1( cache, v );

    	} else {

    		if ( arraysEqual( cache, elements ) ) return;

    		mat2array.set( elements );

    		gl.uniformMatrix2fv( this.addr, false, mat2array );

    		copyArray$1( cache, elements );

    	}

    }

    function setValueM3( gl, v ) {

    	var cache = this.cache;
    	var elements = v.elements;

    	if ( elements === undefined ) {

    		if ( arraysEqual( cache, v ) ) return;

    		gl.uniformMatrix3fv( this.addr, false, v );

    		copyArray$1( cache, v );

    	} else {

    		if ( arraysEqual( cache, elements ) ) return;

    		mat3array.set( elements );

    		gl.uniformMatrix3fv( this.addr, false, mat3array );

    		copyArray$1( cache, elements );

    	}

    }

    function setValueM4( gl, v ) {

    	var cache = this.cache;
    	var elements = v.elements;

    	if ( elements === undefined ) {

    		if ( arraysEqual( cache, v ) ) return;

    		gl.uniformMatrix4fv( this.addr, false, v );

    		copyArray$1( cache, v );

    	} else {

    		if ( arraysEqual( cache, elements ) ) return;

    		mat4array.set( elements );

    		gl.uniformMatrix4fv( this.addr, false, mat4array );

    		copyArray$1( cache, elements );

    	}

    }

    // Single texture (2D / Cube)

    function setValueT1( gl, v, textures ) {

    	var cache = this.cache;
    	var unit = textures.allocateTextureUnit();

    	if ( cache[ 0 ] !== unit ) {

    		gl.uniform1i( this.addr, unit );
    		cache[ 0 ] = unit;

    	}

    	textures.safeSetTexture2D( v || emptyTexture, unit );

    }

    function setValueT2DArray1( gl, v, textures ) {

    	var cache = this.cache;
    	var unit = textures.allocateTextureUnit();

    	if ( cache[ 0 ] !== unit ) {

    		gl.uniform1i( this.addr, unit );
    		cache[ 0 ] = unit;

    	}

    	textures.setTexture2DArray( v || emptyTexture2dArray, unit );

    }

    function setValueT3D1( gl, v, textures ) {

    	var cache = this.cache;
    	var unit = textures.allocateTextureUnit();

    	if ( cache[ 0 ] !== unit ) {

    		gl.uniform1i( this.addr, unit );
    		cache[ 0 ] = unit;

    	}

    	textures.setTexture3D( v || emptyTexture3d, unit );

    }

    function setValueT6( gl, v, textures ) {

    	var cache = this.cache;
    	var unit = textures.allocateTextureUnit();

    	if ( cache[ 0 ] !== unit ) {

    		gl.uniform1i( this.addr, unit );
    		cache[ 0 ] = unit;

    	}

    	textures.safeSetTextureCube( v || emptyCubeTexture, unit );

    }

    // Integer / Boolean vectors or arrays thereof (always flat arrays)

    function setValueV1i( gl, v ) {

    	var cache = this.cache;

    	if ( cache[ 0 ] === v ) return;

    	gl.uniform1i( this.addr, v );

    	cache[ 0 ] = v;

    }

    function setValueV2i( gl, v ) {

    	var cache = this.cache;

    	if ( arraysEqual( cache, v ) ) return;

    	gl.uniform2iv( this.addr, v );

    	copyArray$1( cache, v );

    }

    function setValueV3i( gl, v ) {

    	var cache = this.cache;

    	if ( arraysEqual( cache, v ) ) return;

    	gl.uniform3iv( this.addr, v );

    	copyArray$1( cache, v );

    }

    function setValueV4i( gl, v ) {

    	var cache = this.cache;

    	if ( arraysEqual( cache, v ) ) return;

    	gl.uniform4iv( this.addr, v );

    	copyArray$1( cache, v );

    }

    // Helper to pick the right setter for the singular case

    function getSingularSetter( type ) {

    	switch ( type ) {

    		case 0x1406: return setValueV1f; // FLOAT
    		case 0x8b50: return setValueV2f; // _VEC2
    		case 0x8b51: return setValueV3f; // _VEC3
    		case 0x8b52: return setValueV4f; // _VEC4

    		case 0x8b5a: return setValueM2; // _MAT2
    		case 0x8b5b: return setValueM3; // _MAT3
    		case 0x8b5c: return setValueM4; // _MAT4

    		case 0x8b5e: case 0x8d66: return setValueT1; // SAMPLER_2D, SAMPLER_EXTERNAL_OES
    		case 0x8b5f: return setValueT3D1; // SAMPLER_3D
    		case 0x8b60: return setValueT6; // SAMPLER_CUBE
    		case 0x8DC1: return setValueT2DArray1; // SAMPLER_2D_ARRAY

    		case 0x1404: case 0x8b56: return setValueV1i; // INT, BOOL
    		case 0x8b53: case 0x8b57: return setValueV2i; // _VEC2
    		case 0x8b54: case 0x8b58: return setValueV3i; // _VEC3
    		case 0x8b55: case 0x8b59: return setValueV4i; // _VEC4

    	}

    }

    // Array of scalars
    function setValueV1fArray( gl, v ) {

    	gl.uniform1fv( this.addr, v );

    }

    // Integer / Boolean vectors or arrays thereof (always flat arrays)
    function setValueV1iArray( gl, v ) {

    	gl.uniform1iv( this.addr, v );

    }

    function setValueV2iArray( gl, v ) {

    	gl.uniform2iv( this.addr, v );

    }

    function setValueV3iArray( gl, v ) {

    	gl.uniform3iv( this.addr, v );

    }

    function setValueV4iArray( gl, v ) {

    	gl.uniform4iv( this.addr, v );

    }


    // Array of vectors (flat or from THREE classes)

    function setValueV2fArray( gl, v ) {

    	var data = flatten$1( v, this.size, 2 );

    	gl.uniform2fv( this.addr, data );

    }

    function setValueV3fArray( gl, v ) {

    	var data = flatten$1( v, this.size, 3 );

    	gl.uniform3fv( this.addr, data );

    }

    function setValueV4fArray( gl, v ) {

    	var data = flatten$1( v, this.size, 4 );

    	gl.uniform4fv( this.addr, data );

    }

    // Array of matrices (flat or from THREE clases)

    function setValueM2Array( gl, v ) {

    	var data = flatten$1( v, this.size, 4 );

    	gl.uniformMatrix2fv( this.addr, false, data );

    }

    function setValueM3Array( gl, v ) {

    	var data = flatten$1( v, this.size, 9 );

    	gl.uniformMatrix3fv( this.addr, false, data );

    }

    function setValueM4Array( gl, v ) {

    	var data = flatten$1( v, this.size, 16 );

    	gl.uniformMatrix4fv( this.addr, false, data );

    }

    // Array of textures (2D / Cube)

    function setValueT1Array( gl, v, textures ) {

    	var n = v.length;

    	var units = allocTexUnits( textures, n );

    	gl.uniform1iv( this.addr, units );

    	for ( var i = 0; i !== n; ++ i ) {

    		textures.safeSetTexture2D( v[ i ] || emptyTexture, units[ i ] );

    	}

    }

    function setValueT6Array( gl, v, textures ) {

    	var n = v.length;

    	var units = allocTexUnits( textures, n );

    	gl.uniform1iv( this.addr, units );

    	for ( var i = 0; i !== n; ++ i ) {

    		textures.safeSetTextureCube( v[ i ] || emptyCubeTexture, units[ i ] );

    	}

    }

    // Helper to pick the right setter for a pure (bottom-level) array

    function getPureArraySetter( type ) {

    	switch ( type ) {

    		case 0x1406: return setValueV1fArray; // FLOAT
    		case 0x8b50: return setValueV2fArray; // _VEC2
    		case 0x8b51: return setValueV3fArray; // _VEC3
    		case 0x8b52: return setValueV4fArray; // _VEC4

    		case 0x8b5a: return setValueM2Array; // _MAT2
    		case 0x8b5b: return setValueM3Array; // _MAT3
    		case 0x8b5c: return setValueM4Array; // _MAT4

    		case 0x8b5e: return setValueT1Array; // SAMPLER_2D
    		case 0x8b60: return setValueT6Array; // SAMPLER_CUBE

    		case 0x1404: case 0x8b56: return setValueV1iArray; // INT, BOOL
    		case 0x8b53: case 0x8b57: return setValueV2iArray; // _VEC2
    		case 0x8b54: case 0x8b58: return setValueV3iArray; // _VEC3
    		case 0x8b55: case 0x8b59: return setValueV4iArray; // _VEC4

    	}

    }

    // --- Uniform Classes ---

    function SingleUniform( id, activeInfo, addr ) {

    	this.id = id;
    	this.addr = addr;
    	this.cache = [];
    	this.setValue = getSingularSetter( activeInfo.type );

    	// this.path = activeInfo.name; // DEBUG

    }

    function PureArrayUniform( id, activeInfo, addr ) {

    	this.id = id;
    	this.addr = addr;
    	this.cache = [];
    	this.size = activeInfo.size;
    	this.setValue = getPureArraySetter( activeInfo.type );

    	// this.path = activeInfo.name; // DEBUG

    }

    PureArrayUniform.prototype.updateCache = function ( data ) {

    	var cache = this.cache;

    	if ( data instanceof Float32Array && cache.length !== data.length ) {

    		this.cache = new Float32Array( data.length );

    	}

    	copyArray$1( cache, data );

    };

    function StructuredUniform( id ) {

    	this.id = id;

    	this.seq = [];
    	this.map = {};

    }

    StructuredUniform.prototype.setValue = function ( gl, value, textures ) {

    	var seq = this.seq;

    	for ( var i = 0, n = seq.length; i !== n; ++ i ) {

    		var u = seq[ i ];
    		u.setValue( gl, value[ u.id ], textures );

    	}

    };

    // --- Top-level ---

    // Parser - builds up the property tree from the path strings

    var RePathPart = /([\w\d_]+)(\])?(\[|\.)?/g;

    // extracts
    // 	- the identifier (member name or array index)
    //  - followed by an optional right bracket (found when array index)
    //  - followed by an optional left bracket or dot (type of subscript)
    //
    // Note: These portions can be read in a non-overlapping fashion and
    // allow straightforward parsing of the hierarchy that WebGL encodes
    // in the uniform names.

    function addUniform( container, uniformObject ) {

    	container.seq.push( uniformObject );
    	container.map[ uniformObject.id ] = uniformObject;

    }

    function parseUniform( activeInfo, addr, container ) {

    	var path = activeInfo.name,
    		pathLength = path.length;

    	// reset RegExp object, because of the early exit of a previous run
    	RePathPart.lastIndex = 0;

    	while ( true ) {

    		var match = RePathPart.exec( path ),
    			matchEnd = RePathPart.lastIndex,

    			id = match[ 1 ],
    			idIsIndex = match[ 2 ] === ']',
    			subscript = match[ 3 ];

    		if ( idIsIndex ) id = id | 0; // convert to integer

    		if ( subscript === undefined || subscript === '[' && matchEnd + 2 === pathLength ) {

    			// bare name or "pure" bottom-level array "[0]" suffix

    			addUniform( container, subscript === undefined ?
    				new SingleUniform( id, activeInfo, addr ) :
    				new PureArrayUniform( id, activeInfo, addr ) );

    			break;

    		} else {

    			// step into inner node / create it in case it doesn't exist

    			var map = container.map, next = map[ id ];

    			if ( next === undefined ) {

    				next = new StructuredUniform( id );
    				addUniform( container, next );

    			}

    			container = next;

    		}

    	}

    }

    // Root Container

    function WebGLUniforms( gl, program ) {

    	this.seq = [];
    	this.map = {};

    	var n = gl.getProgramParameter( program, 35718 );

    	for ( var i = 0; i < n; ++ i ) {

    		var info = gl.getActiveUniform( program, i ),
    			addr = gl.getUniformLocation( program, info.name );

    		parseUniform( info, addr, this );

    	}

    }

    WebGLUniforms.prototype.setValue = function ( gl, name, value, textures ) {

    	var u = this.map[ name ];

    	if ( u !== undefined ) u.setValue( gl, value, textures );

    };

    WebGLUniforms.prototype.setOptional = function ( gl, object, name ) {

    	var v = object[ name ];

    	if ( v !== undefined ) this.setValue( gl, name, v );

    };


    // Static interface

    WebGLUniforms.upload = function ( gl, seq, values, textures ) {

    	for ( var i = 0, n = seq.length; i !== n; ++ i ) {

    		var u = seq[ i ],
    			v = values[ u.id ];

    		if ( v.needsUpdate !== false ) {

    			// note: always updating when .needsUpdate is undefined
    			u.setValue( gl, v.value, textures );

    		}

    	}

    };

    WebGLUniforms.seqWithValue = function ( seq, values ) {

    	var r = [];

    	for ( var i = 0, n = seq.length; i !== n; ++ i ) {

    		var u = seq[ i ];
    		if ( u.id in values ) r.push( u );

    	}

    	return r;

    };

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function WebGLShader( gl, type, string ) {

    	var shader = gl.createShader( type );

    	gl.shaderSource( shader, string );
    	gl.compileShader( shader );

    	return shader;

    }

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    var programIdCount = 0;

    function addLineNumbers( string ) {

    	var lines = string.split( '\n' );

    	for ( var i = 0; i < lines.length; i ++ ) {

    		lines[ i ] = ( i + 1 ) + ': ' + lines[ i ];

    	}

    	return lines.join( '\n' );

    }

    function getEncodingComponents( encoding ) {

    	switch ( encoding ) {

    		case LinearEncoding:
    			return [ 'Linear', '( value )' ];
    		case sRGBEncoding:
    			return [ 'sRGB', '( value )' ];
    		case RGBEEncoding:
    			return [ 'RGBE', '( value )' ];
    		case RGBM7Encoding:
    			return [ 'RGBM', '( value, 7.0 )' ];
    		case RGBM16Encoding:
    			return [ 'RGBM', '( value, 16.0 )' ];
    		case RGBDEncoding:
    			return [ 'RGBD', '( value, 256.0 )' ];
    		case GammaEncoding:
    			return [ 'Gamma', '( value, float( GAMMA_FACTOR ) )' ];
    		case LogLuvEncoding:
    			return [ 'LogLuv', '( value )' ];
    		default:
    			throw new Error( 'unsupported encoding: ' + encoding );

    	}

    }

    function getShaderErrors( gl, shader, type ) {

    	var status = gl.getShaderParameter( shader, 35713 );
    	var log = gl.getShaderInfoLog( shader ).trim();

    	if ( status && log === '' ) return '';

    	// --enable-privileged-webgl-extension
    	// console.log( '**' + type + '**', gl.getExtension( 'WEBGL_debug_shaders' ).getTranslatedShaderSource( shader ) );

    	var source = gl.getShaderSource( shader );

    	return 'THREE.WebGLShader: gl.getShaderInfoLog() ' + type + '\n' + log + addLineNumbers( source );

    }

    function getTexelDecodingFunction( functionName, encoding ) {

    	var components = getEncodingComponents( encoding );
    	return 'vec4 ' + functionName + '( vec4 value ) { return ' + components[ 0 ] + 'ToLinear' + components[ 1 ] + '; }';

    }

    function getTexelEncodingFunction( functionName, encoding ) {

    	var components = getEncodingComponents( encoding );
    	return 'vec4 ' + functionName + '( vec4 value ) { return LinearTo' + components[ 0 ] + components[ 1 ] + '; }';

    }

    function getToneMappingFunction( functionName, toneMapping ) {

    	var toneMappingName;

    	switch ( toneMapping ) {

    		case LinearToneMapping:
    			toneMappingName = 'Linear';
    			break;

    		case ReinhardToneMapping:
    			toneMappingName = 'Reinhard';
    			break;

    		case Uncharted2ToneMapping:
    			toneMappingName = 'Uncharted2';
    			break;

    		case CineonToneMapping:
    			toneMappingName = 'OptimizedCineon';
    			break;

    		case ACESFilmicToneMapping:
    			toneMappingName = 'ACESFilmic';
    			break;

    		default:
    			throw new Error( 'unsupported toneMapping: ' + toneMapping );

    	}

    	return 'vec3 ' + functionName + '( vec3 color ) { return ' + toneMappingName + 'ToneMapping( color ); }';

    }

    function generateExtensions( extensions, parameters, rendererExtensions ) {

    	extensions = extensions || {};

    	var chunks = [
    		( extensions.derivatives || parameters.envMapCubeUV || parameters.bumpMap || parameters.tangentSpaceNormalMap || parameters.clearcoatNormalMap || parameters.flatShading ) ? '#extension GL_OES_standard_derivatives : enable' : '',
    		( extensions.fragDepth || parameters.logarithmicDepthBuffer ) && rendererExtensions.get( 'EXT_frag_depth' ) ? '#extension GL_EXT_frag_depth : enable' : '',
    		( extensions.drawBuffers ) && rendererExtensions.get( 'WEBGL_draw_buffers' ) ? '#extension GL_EXT_draw_buffers : require' : '',
    		( extensions.shaderTextureLOD || parameters.envMap ) && rendererExtensions.get( 'EXT_shader_texture_lod' ) ? '#extension GL_EXT_shader_texture_lod : enable' : ''
    	];

    	return chunks.filter( filterEmptyLine ).join( '\n' );

    }

    function generateDefines( defines ) {

    	var chunks = [];

    	for ( var name in defines ) {

    		var value = defines[ name ];

    		if ( value === false ) continue;

    		chunks.push( '#define ' + name + ' ' + value );

    	}

    	return chunks.join( '\n' );

    }

    function fetchAttributeLocations( gl, program ) {

    	var attributes = {};

    	var n = gl.getProgramParameter( program, 35721 );

    	for ( var i = 0; i < n; i ++ ) {

    		var info = gl.getActiveAttrib( program, i );
    		var name = info.name;

    		// console.log( 'THREE.WebGLProgram: ACTIVE VERTEX ATTRIBUTE:', name, i );

    		attributes[ name ] = gl.getAttribLocation( program, name );

    	}

    	return attributes;

    }

    function filterEmptyLine( string ) {

    	return string !== '';

    }

    function replaceLightNums( string, parameters ) {

    	return string
    		.replace( /NUM_DIR_LIGHTS/g, parameters.numDirLights )
    		.replace( /NUM_SPOT_LIGHTS/g, parameters.numSpotLights )
    		.replace( /NUM_RECT_AREA_LIGHTS/g, parameters.numRectAreaLights )
    		.replace( /NUM_POINT_LIGHTS/g, parameters.numPointLights )
    		.replace( /NUM_HEMI_LIGHTS/g, parameters.numHemiLights )
    		.replace( /NUM_DIR_LIGHT_SHADOWS/g, parameters.numDirLightShadows )
    		.replace( /NUM_SPOT_LIGHT_SHADOWS/g, parameters.numSpotLightShadows )
    		.replace( /NUM_POINT_LIGHT_SHADOWS/g, parameters.numPointLightShadows );

    }

    function replaceClippingPlaneNums( string, parameters ) {

    	return string
    		.replace( /NUM_CLIPPING_PLANES/g, parameters.numClippingPlanes )
    		.replace( /UNION_CLIPPING_PLANES/g, ( parameters.numClippingPlanes - parameters.numClipIntersection ) );

    }

    function parseIncludes( string ) {

    	var pattern = /^[ \t]*#include +<([\w\d./]+)>/gm;

    	function replace( match, include ) {

    		var replace = ShaderChunk[ include ];

    		if ( replace === undefined ) {

    			throw new Error( 'Can not resolve #include <' + include + '>' );

    		}

    		return parseIncludes( replace );

    	}

    	return string.replace( pattern, replace );

    }

    function unrollLoops( string ) {

    	var pattern = /#pragma unroll_loop[\s]+?for \( int i \= (\d+)\; i < (\d+)\; i \+\+ \) \{([\s\S]+?)(?=\})\}/g;

    	function replace( match, start, end, snippet ) {

    		var unroll = '';

    		for ( var i = parseInt( start ); i < parseInt( end ); i ++ ) {

    			unroll += snippet
    				.replace( /\[ i \]/g, '[ ' + i + ' ]' )
    				.replace( /UNROLLED_LOOP_INDEX/g, i );

    		}

    		return unroll;

    	}

    	return string.replace( pattern, replace );

    }

    function WebGLProgram( renderer, extensions, code, material, shader, parameters, capabilities ) {

    	var gl = renderer.getContext();

    	var defines = material.defines;

    	var vertexShader = shader.vertexShader;
    	var fragmentShader = shader.fragmentShader;

    	var shadowMapTypeDefine = 'SHADOWMAP_TYPE_BASIC';

    	if ( parameters.shadowMapType === PCFShadowMap ) {

    		shadowMapTypeDefine = 'SHADOWMAP_TYPE_PCF';

    	} else if ( parameters.shadowMapType === PCFSoftShadowMap ) {

    		shadowMapTypeDefine = 'SHADOWMAP_TYPE_PCF_SOFT';

    	} else if ( parameters.shadowMapType === VSMShadowMap ) {

    		shadowMapTypeDefine = 'SHADOWMAP_TYPE_VSM';

    	}

    	var envMapTypeDefine = 'ENVMAP_TYPE_CUBE';
    	var envMapModeDefine = 'ENVMAP_MODE_REFLECTION';
    	var envMapBlendingDefine = 'ENVMAP_BLENDING_MULTIPLY';

    	if ( parameters.envMap ) {

    		switch ( material.envMap.mapping ) {

    			case CubeReflectionMapping:
    			case CubeRefractionMapping:
    				envMapTypeDefine = 'ENVMAP_TYPE_CUBE';
    				break;

    			case CubeUVReflectionMapping:
    			case CubeUVRefractionMapping:
    				envMapTypeDefine = 'ENVMAP_TYPE_CUBE_UV';
    				break;

    			case EquirectangularReflectionMapping:
    			case EquirectangularRefractionMapping:
    				envMapTypeDefine = 'ENVMAP_TYPE_EQUIREC';
    				break;

    			case SphericalReflectionMapping:
    				envMapTypeDefine = 'ENVMAP_TYPE_SPHERE';
    				break;

    		}

    		switch ( material.envMap.mapping ) {

    			case CubeRefractionMapping:
    			case EquirectangularRefractionMapping:
    				envMapModeDefine = 'ENVMAP_MODE_REFRACTION';
    				break;

    		}

    		switch ( material.combine ) {

    			case MultiplyOperation:
    				envMapBlendingDefine = 'ENVMAP_BLENDING_MULTIPLY';
    				break;

    			case MixOperation:
    				envMapBlendingDefine = 'ENVMAP_BLENDING_MIX';
    				break;

    			case AddOperation:
    				envMapBlendingDefine = 'ENVMAP_BLENDING_ADD';
    				break;

    		}

    	}

    	var gammaFactorDefine = ( renderer.gammaFactor > 0 ) ? renderer.gammaFactor : 1.0;

    	// console.log( 'building new program ' );

    	//

    	var customExtensions = capabilities.isWebGL2 ? '' : generateExtensions( material.extensions, parameters, extensions );

    	var customDefines = generateDefines( defines );

    	//

    	var program = gl.createProgram();

    	var prefixVertex, prefixFragment;

    	if ( material.isRawShaderMaterial ) {

    		prefixVertex = [

    			customDefines

    		].filter( filterEmptyLine ).join( '\n' );

    		if ( prefixVertex.length > 0 ) {

    			prefixVertex += '\n';

    		}

    		prefixFragment = [

    			customExtensions,
    			customDefines

    		].filter( filterEmptyLine ).join( '\n' );

    		if ( prefixFragment.length > 0 ) {

    			prefixFragment += '\n';

    		}

    	} else {

    		prefixVertex = [

    			'precision ' + parameters.precision + ' float;',
    			'precision ' + parameters.precision + ' int;',

    			( parameters.precision === 'highp' ) ? '#define HIGH_PRECISION' : '',

    			'#define SHADER_NAME ' + shader.name,

    			customDefines,

    			parameters.supportsVertexTextures ? '#define VERTEX_TEXTURES' : '',

    			'#define GAMMA_FACTOR ' + gammaFactorDefine,

    			'#define MAX_BONES ' + parameters.maxBones,
    			( parameters.useFog && parameters.fog ) ? '#define USE_FOG' : '',
    			( parameters.useFog && parameters.fogExp2 ) ? '#define FOG_EXP2' : '',

    			parameters.map ? '#define USE_MAP' : '',
    			parameters.envMap ? '#define USE_ENVMAP' : '',
    			parameters.envMap ? '#define ' + envMapModeDefine : '',
    			parameters.lightMap ? '#define USE_LIGHTMAP' : '',
    			parameters.aoMap ? '#define USE_AOMAP' : '',
    			parameters.emissiveMap ? '#define USE_EMISSIVEMAP' : '',
    			parameters.bumpMap ? '#define USE_BUMPMAP' : '',
    			parameters.normalMap ? '#define USE_NORMALMAP' : '',
    			( parameters.normalMap && parameters.objectSpaceNormalMap ) ? '#define OBJECTSPACE_NORMALMAP' : '',
    			( parameters.normalMap && parameters.tangentSpaceNormalMap ) ? '#define TANGENTSPACE_NORMALMAP' : '',

    			parameters.clearcoatNormalMap ? '#define USE_CLEARCOAT_NORMALMAP' : '',
    			parameters.displacementMap && parameters.supportsVertexTextures ? '#define USE_DISPLACEMENTMAP' : '',
    			parameters.specularMap ? '#define USE_SPECULARMAP' : '',
    			parameters.roughnessMap ? '#define USE_ROUGHNESSMAP' : '',
    			parameters.metalnessMap ? '#define USE_METALNESSMAP' : '',
    			parameters.alphaMap ? '#define USE_ALPHAMAP' : '',

    			parameters.vertexTangents ? '#define USE_TANGENT' : '',
    			parameters.vertexColors ? '#define USE_COLOR' : '',
    			parameters.vertexUvs ? '#define USE_UV' : '',

    			parameters.flatShading ? '#define FLAT_SHADED' : '',

    			parameters.skinning ? '#define USE_SKINNING' : '',
    			parameters.useVertexTexture ? '#define BONE_TEXTURE' : '',

    			parameters.morphTargets ? '#define USE_MORPHTARGETS' : '',
    			parameters.morphNormals && parameters.flatShading === false ? '#define USE_MORPHNORMALS' : '',
    			parameters.doubleSided ? '#define DOUBLE_SIDED' : '',
    			parameters.flipSided ? '#define FLIP_SIDED' : '',

    			parameters.shadowMapEnabled ? '#define USE_SHADOWMAP' : '',
    			parameters.shadowMapEnabled ? '#define ' + shadowMapTypeDefine : '',

    			parameters.sizeAttenuation ? '#define USE_SIZEATTENUATION' : '',

    			parameters.logarithmicDepthBuffer ? '#define USE_LOGDEPTHBUF' : '',
    			parameters.logarithmicDepthBuffer && ( capabilities.isWebGL2 || extensions.get( 'EXT_frag_depth' ) ) ? '#define USE_LOGDEPTHBUF_EXT' : '',

    			'uniform mat4 modelMatrix;',
    			'uniform mat4 modelViewMatrix;',
    			'uniform mat4 projectionMatrix;',
    			'uniform mat4 viewMatrix;',
    			'uniform mat3 normalMatrix;',
    			'uniform vec3 cameraPosition;',

    			'attribute vec3 position;',
    			'attribute vec3 normal;',
    			'attribute vec2 uv;',

    			'#ifdef USE_TANGENT',

    			'	attribute vec4 tangent;',

    			'#endif',

    			'#ifdef USE_COLOR',

    			'	attribute vec3 color;',

    			'#endif',

    			'#ifdef USE_MORPHTARGETS',

    			'	attribute vec3 morphTarget0;',
    			'	attribute vec3 morphTarget1;',
    			'	attribute vec3 morphTarget2;',
    			'	attribute vec3 morphTarget3;',

    			'	#ifdef USE_MORPHNORMALS',

    			'		attribute vec3 morphNormal0;',
    			'		attribute vec3 morphNormal1;',
    			'		attribute vec3 morphNormal2;',
    			'		attribute vec3 morphNormal3;',

    			'	#else',

    			'		attribute vec3 morphTarget4;',
    			'		attribute vec3 morphTarget5;',
    			'		attribute vec3 morphTarget6;',
    			'		attribute vec3 morphTarget7;',

    			'	#endif',

    			'#endif',

    			'#ifdef USE_SKINNING',

    			'	attribute vec4 skinIndex;',
    			'	attribute vec4 skinWeight;',

    			'#endif',

    			'\n'

    		].filter( filterEmptyLine ).join( '\n' );

    		prefixFragment = [

    			customExtensions,

    			'precision ' + parameters.precision + ' float;',
    			'precision ' + parameters.precision + ' int;',

    			( parameters.precision === 'highp' ) ? '#define HIGH_PRECISION' : '',

    			'#define SHADER_NAME ' + shader.name,

    			customDefines,

    			parameters.alphaTest ? '#define ALPHATEST ' + parameters.alphaTest + ( parameters.alphaTest % 1 ? '' : '.0' ) : '', // add '.0' if integer

    			'#define GAMMA_FACTOR ' + gammaFactorDefine,

    			( parameters.useFog && parameters.fog ) ? '#define USE_FOG' : '',
    			( parameters.useFog && parameters.fogExp2 ) ? '#define FOG_EXP2' : '',

    			parameters.map ? '#define USE_MAP' : '',
    			parameters.matcap ? '#define USE_MATCAP' : '',
    			parameters.envMap ? '#define USE_ENVMAP' : '',
    			parameters.envMap ? '#define ' + envMapTypeDefine : '',
    			parameters.envMap ? '#define ' + envMapModeDefine : '',
    			parameters.envMap ? '#define ' + envMapBlendingDefine : '',
    			parameters.lightMap ? '#define USE_LIGHTMAP' : '',
    			parameters.aoMap ? '#define USE_AOMAP' : '',
    			parameters.emissiveMap ? '#define USE_EMISSIVEMAP' : '',
    			parameters.bumpMap ? '#define USE_BUMPMAP' : '',
    			parameters.normalMap ? '#define USE_NORMALMAP' : '',
    			( parameters.normalMap && parameters.objectSpaceNormalMap ) ? '#define OBJECTSPACE_NORMALMAP' : '',
    			( parameters.normalMap && parameters.tangentSpaceNormalMap ) ? '#define TANGENTSPACE_NORMALMAP' : '',
    			parameters.clearcoatNormalMap ? '#define USE_CLEARCOAT_NORMALMAP' : '',
    			parameters.specularMap ? '#define USE_SPECULARMAP' : '',
    			parameters.roughnessMap ? '#define USE_ROUGHNESSMAP' : '',
    			parameters.metalnessMap ? '#define USE_METALNESSMAP' : '',
    			parameters.alphaMap ? '#define USE_ALPHAMAP' : '',

    			parameters.sheen ? '#define USE_SHEEN' : '',

    			parameters.vertexTangents ? '#define USE_TANGENT' : '',
    			parameters.vertexColors ? '#define USE_COLOR' : '',
    			parameters.vertexUvs ? '#define USE_UV' : '',

    			parameters.gradientMap ? '#define USE_GRADIENTMAP' : '',

    			parameters.flatShading ? '#define FLAT_SHADED' : '',

    			parameters.doubleSided ? '#define DOUBLE_SIDED' : '',
    			parameters.flipSided ? '#define FLIP_SIDED' : '',

    			parameters.shadowMapEnabled ? '#define USE_SHADOWMAP' : '',
    			parameters.shadowMapEnabled ? '#define ' + shadowMapTypeDefine : '',

    			parameters.premultipliedAlpha ? '#define PREMULTIPLIED_ALPHA' : '',

    			parameters.physicallyCorrectLights ? '#define PHYSICALLY_CORRECT_LIGHTS' : '',

    			parameters.logarithmicDepthBuffer ? '#define USE_LOGDEPTHBUF' : '',
    			parameters.logarithmicDepthBuffer && ( capabilities.isWebGL2 || extensions.get( 'EXT_frag_depth' ) ) ? '#define USE_LOGDEPTHBUF_EXT' : '',

    			( ( material.extensions ? material.extensions.shaderTextureLOD : false ) || parameters.envMap ) && ( capabilities.isWebGL2 || extensions.get( 'EXT_shader_texture_lod' ) ) ? '#define TEXTURE_LOD_EXT' : '',

    			'uniform mat4 viewMatrix;',
    			'uniform vec3 cameraPosition;',

    			( parameters.toneMapping !== NoToneMapping ) ? '#define TONE_MAPPING' : '',
    			( parameters.toneMapping !== NoToneMapping ) ? ShaderChunk[ 'tonemapping_pars_fragment' ] : '', // this code is required here because it is used by the toneMapping() function defined below
    			( parameters.toneMapping !== NoToneMapping ) ? getToneMappingFunction( 'toneMapping', parameters.toneMapping ) : '',

    			parameters.dithering ? '#define DITHERING' : '',

    			( parameters.outputEncoding || parameters.mapEncoding || parameters.matcapEncoding || parameters.envMapEncoding || parameters.emissiveMapEncoding ) ?
    				ShaderChunk[ 'encodings_pars_fragment' ] : '', // this code is required here because it is used by the various encoding/decoding function defined below
    			parameters.mapEncoding ? getTexelDecodingFunction( 'mapTexelToLinear', parameters.mapEncoding ) : '',
    			parameters.matcapEncoding ? getTexelDecodingFunction( 'matcapTexelToLinear', parameters.matcapEncoding ) : '',
    			parameters.envMapEncoding ? getTexelDecodingFunction( 'envMapTexelToLinear', parameters.envMapEncoding ) : '',
    			parameters.emissiveMapEncoding ? getTexelDecodingFunction( 'emissiveMapTexelToLinear', parameters.emissiveMapEncoding ) : '',
    			parameters.outputEncoding ? getTexelEncodingFunction( 'linearToOutputTexel', parameters.outputEncoding ) : '',

    			parameters.depthPacking ? '#define DEPTH_PACKING ' + material.depthPacking : '',

    			'\n'

    		].filter( filterEmptyLine ).join( '\n' );

    	}

    	vertexShader = parseIncludes( vertexShader );
    	vertexShader = replaceLightNums( vertexShader, parameters );
    	vertexShader = replaceClippingPlaneNums( vertexShader, parameters );

    	fragmentShader = parseIncludes( fragmentShader );
    	fragmentShader = replaceLightNums( fragmentShader, parameters );
    	fragmentShader = replaceClippingPlaneNums( fragmentShader, parameters );

    	vertexShader = unrollLoops( vertexShader );
    	fragmentShader = unrollLoops( fragmentShader );

    	if ( capabilities.isWebGL2 && ! material.isRawShaderMaterial ) {

    		var isGLSL3ShaderMaterial = false;

    		var versionRegex = /^\s*#version\s+300\s+es\s*\n/;

    		if ( material.isShaderMaterial &&
    			vertexShader.match( versionRegex ) !== null &&
    			fragmentShader.match( versionRegex ) !== null ) {

    			isGLSL3ShaderMaterial = true;

    			vertexShader = vertexShader.replace( versionRegex, '' );
    			fragmentShader = fragmentShader.replace( versionRegex, '' );

    		}

    		// GLSL 3.0 conversion
    		prefixVertex = [
    			'#version 300 es\n',
    			'#define attribute in',
    			'#define varying out',
    			'#define texture2D texture'
    		].join( '\n' ) + '\n' + prefixVertex;

    		prefixFragment = [
    			'#version 300 es\n',
    			'#define varying in',
    			isGLSL3ShaderMaterial ? '' : 'out highp vec4 pc_fragColor;',
    			isGLSL3ShaderMaterial ? '' : '#define gl_FragColor pc_fragColor',
    			'#define gl_FragDepthEXT gl_FragDepth',
    			'#define texture2D texture',
    			'#define textureCube texture',
    			'#define texture2DProj textureProj',
    			'#define texture2DLodEXT textureLod',
    			'#define texture2DProjLodEXT textureProjLod',
    			'#define textureCubeLodEXT textureLod',
    			'#define texture2DGradEXT textureGrad',
    			'#define texture2DProjGradEXT textureProjGrad',
    			'#define textureCubeGradEXT textureGrad'
    		].join( '\n' ) + '\n' + prefixFragment;

    	}

    	var vertexGlsl = prefixVertex + vertexShader;
    	var fragmentGlsl = prefixFragment + fragmentShader;

    	// console.log( '*VERTEX*', vertexGlsl );
    	// console.log( '*FRAGMENT*', fragmentGlsl );

    	var glVertexShader = WebGLShader( gl, 35633, vertexGlsl );
    	var glFragmentShader = WebGLShader( gl, 35632, fragmentGlsl );

    	gl.attachShader( program, glVertexShader );
    	gl.attachShader( program, glFragmentShader );

    	// Force a particular attribute to index 0.

    	if ( material.index0AttributeName !== undefined ) {

    		gl.bindAttribLocation( program, 0, material.index0AttributeName );

    	} else if ( parameters.morphTargets === true ) {

    		// programs with morphTargets displace position out of attribute 0
    		gl.bindAttribLocation( program, 0, 'position' );

    	}

    	gl.linkProgram( program );

    	// check for link errors
    	if ( renderer.debug.checkShaderErrors ) {

    		var programLog = gl.getProgramInfoLog( program ).trim();
    		var vertexLog = gl.getShaderInfoLog( glVertexShader ).trim();
    		var fragmentLog = gl.getShaderInfoLog( glFragmentShader ).trim();

    		var runnable = true;
    		var haveDiagnostics = true;

    		if ( gl.getProgramParameter( program, 35714 ) === false ) {

    			runnable = false;

    			var vertexErrors = getShaderErrors( gl, glVertexShader, 'vertex' );
    			var fragmentErrors = getShaderErrors( gl, glFragmentShader, 'fragment' );

    			console.error( 'THREE.WebGLProgram: shader error: ', gl.getError(), '35715', gl.getProgramParameter( program, 35715 ), 'gl.getProgramInfoLog', programLog, vertexErrors, fragmentErrors );

    		} else if ( programLog !== '' ) {

    			console.warn( 'THREE.WebGLProgram: gl.getProgramInfoLog()', programLog );

    		} else if ( vertexLog === '' || fragmentLog === '' ) {

    			haveDiagnostics = false;

    		}

    		if ( haveDiagnostics ) {

    			this.diagnostics = {

    				runnable: runnable,
    				material: material,

    				programLog: programLog,

    				vertexShader: {

    					log: vertexLog,
    					prefix: prefixVertex

    				},

    				fragmentShader: {

    					log: fragmentLog,
    					prefix: prefixFragment

    				}

    			};

    		}

    	}

    	// clean up

    	gl.deleteShader( glVertexShader );
    	gl.deleteShader( glFragmentShader );

    	// set up caching for uniform locations

    	var cachedUniforms;

    	this.getUniforms = function () {

    		if ( cachedUniforms === undefined ) {

    			cachedUniforms = new WebGLUniforms( gl, program );

    		}

    		return cachedUniforms;

    	};

    	// set up caching for attribute locations

    	var cachedAttributes;

    	this.getAttributes = function () {

    		if ( cachedAttributes === undefined ) {

    			cachedAttributes = fetchAttributeLocations( gl, program );

    		}

    		return cachedAttributes;

    	};

    	// free resource

    	this.destroy = function () {

    		gl.deleteProgram( program );
    		this.program = undefined;

    	};

    	//

    	this.name = shader.name;
    	this.id = programIdCount ++;
    	this.code = code;
    	this.usedTimes = 1;
    	this.program = program;
    	this.vertexShader = glVertexShader;
    	this.fragmentShader = glFragmentShader;

    	return this;

    }

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function WebGLPrograms( renderer, extensions, capabilities ) {

    	var programs = [];

    	var shaderIDs = {
    		MeshDepthMaterial: 'depth',
    		MeshDistanceMaterial: 'distanceRGBA',
    		MeshNormalMaterial: 'normal',
    		MeshBasicMaterial: 'basic',
    		MeshLambertMaterial: 'lambert',
    		MeshPhongMaterial: 'phong',
    		MeshToonMaterial: 'phong',
    		MeshStandardMaterial: 'physical',
    		MeshPhysicalMaterial: 'physical',
    		MeshMatcapMaterial: 'matcap',
    		LineBasicMaterial: 'basic',
    		LineDashedMaterial: 'dashed',
    		PointsMaterial: 'points',
    		ShadowMaterial: 'shadow',
    		SpriteMaterial: 'sprite'
    	};

    	var parameterNames = [
    		"precision", "supportsVertexTextures", "map", "mapEncoding", "matcap", "matcapEncoding", "envMap", "envMapMode", "envMapEncoding",
    		"lightMap", "aoMap", "emissiveMap", "emissiveMapEncoding", "bumpMap", "normalMap", "objectSpaceNormalMap", "tangentSpaceNormalMap", "clearcoatNormalMap", "displacementMap", "specularMap",
    		"roughnessMap", "metalnessMap", "gradientMap",
    		"alphaMap", "combine", "vertexColors", "vertexTangents", "fog", "useFog", "fogExp2",
    		"flatShading", "sizeAttenuation", "logarithmicDepthBuffer", "skinning",
    		"maxBones", "useVertexTexture", "morphTargets", "morphNormals",
    		"maxMorphTargets", "maxMorphNormals", "premultipliedAlpha",
    		"numDirLights", "numPointLights", "numSpotLights", "numHemiLights", "numRectAreaLights",
    		"shadowMapEnabled", "shadowMapType", "toneMapping", 'physicallyCorrectLights',
    		"alphaTest", "doubleSided", "flipSided", "numClippingPlanes", "numClipIntersection", "depthPacking", "dithering",
    		"sheen"
    	];


    	function allocateBones( object ) {

    		var skeleton = object.skeleton;
    		var bones = skeleton.bones;

    		if ( capabilities.floatVertexTextures ) {

    			return 1024;

    		} else {

    			// default for when object is not specified
    			// ( for example when prebuilding shader to be used with multiple objects )
    			//
    			//  - leave some extra space for other uniforms
    			//  - limit here is ANGLE's 254 max uniform vectors
    			//    (up to 54 should be safe)

    			var nVertexUniforms = capabilities.maxVertexUniforms;
    			var nVertexMatrices = Math.floor( ( nVertexUniforms - 20 ) / 4 );

    			var maxBones = Math.min( nVertexMatrices, bones.length );

    			if ( maxBones < bones.length ) {

    				console.warn( 'THREE.WebGLRenderer: Skeleton has ' + bones.length + ' bones. This GPU supports ' + maxBones + '.' );
    				return 0;

    			}

    			return maxBones;

    		}

    	}

    	function getTextureEncodingFromMap( map, gammaOverrideLinear ) {

    		var encoding;

    		if ( ! map ) {

    			encoding = LinearEncoding;

    		} else if ( map.isTexture ) {

    			encoding = map.encoding;

    		} else if ( map.isWebGLRenderTarget ) {

    			console.warn( "THREE.WebGLPrograms.getTextureEncodingFromMap: don't use render targets as textures. Use their .texture property instead." );
    			encoding = map.texture.encoding;

    		}

    		// add backwards compatibility for WebGLRenderer.gammaInput/gammaOutput parameter, should probably be removed at some point.
    		if ( encoding === LinearEncoding && gammaOverrideLinear ) {

    			encoding = GammaEncoding;

    		}

    		return encoding;

    	}

    	this.getParameters = function ( material, lights, shadows, fog, nClipPlanes, nClipIntersection, object ) {

    		var shaderID = shaderIDs[ material.type ];

    		// heuristics to create shader parameters according to lights in the scene
    		// (not to blow over maxLights budget)

    		var maxBones = object.isSkinnedMesh ? allocateBones( object ) : 0;
    		var precision = capabilities.precision;

    		if ( material.precision !== null ) {

    			precision = capabilities.getMaxPrecision( material.precision );

    			if ( precision !== material.precision ) {

    				console.warn( 'THREE.WebGLProgram.getParameters:', material.precision, 'not supported, using', precision, 'instead.' );

    			}

    		}

    		var currentRenderTarget = renderer.getRenderTarget();

    		var parameters = {

    			shaderID: shaderID,

    			precision: precision,
    			supportsVertexTextures: capabilities.vertexTextures,
    			outputEncoding: getTextureEncodingFromMap( ( ! currentRenderTarget ) ? null : currentRenderTarget.texture, renderer.gammaOutput ),
    			map: !! material.map,
    			mapEncoding: getTextureEncodingFromMap( material.map, renderer.gammaInput ),
    			matcap: !! material.matcap,
    			matcapEncoding: getTextureEncodingFromMap( material.matcap, renderer.gammaInput ),
    			envMap: !! material.envMap,
    			envMapMode: material.envMap && material.envMap.mapping,
    			envMapEncoding: getTextureEncodingFromMap( material.envMap, renderer.gammaInput ),
    			envMapCubeUV: ( !! material.envMap ) && ( ( material.envMap.mapping === CubeUVReflectionMapping ) || ( material.envMap.mapping === CubeUVRefractionMapping ) ),
    			lightMap: !! material.lightMap,
    			aoMap: !! material.aoMap,
    			emissiveMap: !! material.emissiveMap,
    			emissiveMapEncoding: getTextureEncodingFromMap( material.emissiveMap, renderer.gammaInput ),
    			bumpMap: !! material.bumpMap,
    			normalMap: !! material.normalMap,
    			objectSpaceNormalMap: material.normalMapType === ObjectSpaceNormalMap,
    			tangentSpaceNormalMap: material.normalMapType === TangentSpaceNormalMap,
    			clearcoatNormalMap: !! material.clearcoatNormalMap,
    			displacementMap: !! material.displacementMap,
    			roughnessMap: !! material.roughnessMap,
    			metalnessMap: !! material.metalnessMap,
    			specularMap: !! material.specularMap,
    			alphaMap: !! material.alphaMap,

    			gradientMap: !! material.gradientMap,

    			sheen: !! material.sheen,

    			combine: material.combine,

    			vertexTangents: ( material.normalMap && material.vertexTangents ),
    			vertexColors: material.vertexColors,
    			vertexUvs: !! material.map || !! material.bumpMap || !! material.normalMap || !! material.specularMap || !! material.alphaMap || !! material.emissiveMap || !! material.roughnessMap || !! material.metalnessMap || !! material.clearcoatNormalMap,

    			fog: !! fog,
    			useFog: material.fog,
    			fogExp2: ( fog && fog.isFogExp2 ),

    			flatShading: material.flatShading,

    			sizeAttenuation: material.sizeAttenuation,
    			logarithmicDepthBuffer: capabilities.logarithmicDepthBuffer,

    			skinning: material.skinning && maxBones > 0,
    			maxBones: maxBones,
    			useVertexTexture: capabilities.floatVertexTextures,

    			morphTargets: material.morphTargets,
    			morphNormals: material.morphNormals,
    			maxMorphTargets: renderer.maxMorphTargets,
    			maxMorphNormals: renderer.maxMorphNormals,

    			numDirLights: lights.directional.length,
    			numPointLights: lights.point.length,
    			numSpotLights: lights.spot.length,
    			numRectAreaLights: lights.rectArea.length,
    			numHemiLights: lights.hemi.length,

    			numDirLightShadows: lights.directionalShadowMap.length,
    			numPointLightShadows: lights.pointShadowMap.length,
    			numSpotLightShadows: lights.spotShadowMap.length,

    			numClippingPlanes: nClipPlanes,
    			numClipIntersection: nClipIntersection,

    			dithering: material.dithering,

    			shadowMapEnabled: renderer.shadowMap.enabled && object.receiveShadow && shadows.length > 0,
    			shadowMapType: renderer.shadowMap.type,

    			toneMapping: material.toneMapped ? renderer.toneMapping : NoToneMapping,
    			physicallyCorrectLights: renderer.physicallyCorrectLights,

    			premultipliedAlpha: material.premultipliedAlpha,

    			alphaTest: material.alphaTest,
    			doubleSided: material.side === DoubleSide,
    			flipSided: material.side === BackSide,

    			depthPacking: ( material.depthPacking !== undefined ) ? material.depthPacking : false

    		};

    		return parameters;

    	};

    	this.getProgramCode = function ( material, parameters ) {

    		var array = [];

    		if ( parameters.shaderID ) {

    			array.push( parameters.shaderID );

    		} else {

    			array.push( material.fragmentShader );
    			array.push( material.vertexShader );

    		}

    		if ( material.defines !== undefined ) {

    			for ( var name in material.defines ) {

    				array.push( name );
    				array.push( material.defines[ name ] );

    			}

    		}

    		for ( var i = 0; i < parameterNames.length; i ++ ) {

    			array.push( parameters[ parameterNames[ i ] ] );

    		}

    		array.push( material.onBeforeCompile.toString() );

    		array.push( renderer.gammaOutput );

    		array.push( renderer.gammaFactor );

    		return array.join();

    	};

    	this.acquireProgram = function ( material, shader, parameters, code ) {

    		var program;

    		// Check if code has been already compiled
    		for ( var p = 0, pl = programs.length; p < pl; p ++ ) {

    			var programInfo = programs[ p ];

    			if ( programInfo.code === code ) {

    				program = programInfo;
    				++ program.usedTimes;

    				break;

    			}

    		}

    		if ( program === undefined ) {

    			program = new WebGLProgram( renderer, extensions, code, material, shader, parameters, capabilities );
    			programs.push( program );

    		}

    		return program;

    	};

    	this.releaseProgram = function ( program ) {

    		if ( -- program.usedTimes === 0 ) {

    			// Remove from unordered set
    			var i = programs.indexOf( program );
    			programs[ i ] = programs[ programs.length - 1 ];
    			programs.pop();

    			// Free WebGL resources
    			program.destroy();

    		}

    	};

    	// Exposed for resource monitoring & error feedback via renderer.info:
    	this.programs = programs;

    }

    /**
     * @author fordacious / fordacious.github.io
     */

    function WebGLProperties() {

    	var properties = new WeakMap();

    	function get( object ) {

    		var map = properties.get( object );

    		if ( map === undefined ) {

    			map = {};
    			properties.set( object, map );

    		}

    		return map;

    	}

    	function remove( object ) {

    		properties.delete( object );

    	}

    	function update( object, key, value ) {

    		properties.get( object )[ key ] = value;

    	}

    	function dispose() {

    		properties = new WeakMap();

    	}

    	return {
    		get: get,
    		remove: remove,
    		update: update,
    		dispose: dispose
    	};

    }

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function painterSortStable( a, b ) {

    	if ( a.groupOrder !== b.groupOrder ) {

    		return a.groupOrder - b.groupOrder;

    	} else if ( a.renderOrder !== b.renderOrder ) {

    		return a.renderOrder - b.renderOrder;

    	} else if ( a.program !== b.program ) {

    		return a.program.id - b.program.id;

    	} else if ( a.material.id !== b.material.id ) {

    		return a.material.id - b.material.id;

    	} else if ( a.z !== b.z ) {

    		return a.z - b.z;

    	} else {

    		return a.id - b.id;

    	}

    }

    function reversePainterSortStable( a, b ) {

    	if ( a.groupOrder !== b.groupOrder ) {

    		return a.groupOrder - b.groupOrder;

    	} else if ( a.renderOrder !== b.renderOrder ) {

    		return a.renderOrder - b.renderOrder;

    	} else if ( a.z !== b.z ) {

    		return b.z - a.z;

    	} else {

    		return a.id - b.id;

    	}

    }


    function WebGLRenderList() {

    	var renderItems = [];
    	var renderItemsIndex = 0;

    	var opaque = [];
    	var transparent = [];

    	var defaultProgram = { id: - 1 };

    	function init() {

    		renderItemsIndex = 0;

    		opaque.length = 0;
    		transparent.length = 0;

    	}

    	function getNextRenderItem( object, geometry, material, groupOrder, z, group ) {

    		var renderItem = renderItems[ renderItemsIndex ];

    		if ( renderItem === undefined ) {

    			renderItem = {
    				id: object.id,
    				object: object,
    				geometry: geometry,
    				material: material,
    				program: material.program || defaultProgram,
    				groupOrder: groupOrder,
    				renderOrder: object.renderOrder,
    				z: z,
    				group: group
    			};

    			renderItems[ renderItemsIndex ] = renderItem;

    		} else {

    			renderItem.id = object.id;
    			renderItem.object = object;
    			renderItem.geometry = geometry;
    			renderItem.material = material;
    			renderItem.program = material.program || defaultProgram;
    			renderItem.groupOrder = groupOrder;
    			renderItem.renderOrder = object.renderOrder;
    			renderItem.z = z;
    			renderItem.group = group;

    		}

    		renderItemsIndex ++;

    		return renderItem;

    	}

    	function push( object, geometry, material, groupOrder, z, group ) {

    		var renderItem = getNextRenderItem( object, geometry, material, groupOrder, z, group );

    		( material.transparent === true ? transparent : opaque ).push( renderItem );

    	}

    	function unshift( object, geometry, material, groupOrder, z, group ) {

    		var renderItem = getNextRenderItem( object, geometry, material, groupOrder, z, group );

    		( material.transparent === true ? transparent : opaque ).unshift( renderItem );

    	}

    	function sort() {

    		if ( opaque.length > 1 ) opaque.sort( painterSortStable );
    		if ( transparent.length > 1 ) transparent.sort( reversePainterSortStable );

    	}

    	return {
    		opaque: opaque,
    		transparent: transparent,

    		init: init,
    		push: push,
    		unshift: unshift,

    		sort: sort
    	};

    }

    function WebGLRenderLists() {

    	var lists = new WeakMap();

    	function onSceneDispose( event ) {

    		var scene = event.target;

    		scene.removeEventListener( 'dispose', onSceneDispose );

    		lists.delete( scene );

    	}

    	function get( scene, camera ) {

    		var cameras = lists.get( scene );
    		var list;
    		if ( cameras === undefined ) {

    			list = new WebGLRenderList();
    			lists.set( scene, new WeakMap() );
    			lists.get( scene ).set( camera, list );

    			scene.addEventListener( 'dispose', onSceneDispose );

    		} else {

    			list = cameras.get( camera );
    			if ( list === undefined ) {

    				list = new WebGLRenderList();
    				cameras.set( camera, list );

    			}

    		}

    		return list;

    	}

    	function dispose() {

    		lists = new WeakMap();

    	}

    	return {
    		get: get,
    		dispose: dispose
    	};

    }

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function UniformsCache() {

    	var lights = {};

    	return {

    		get: function ( light ) {

    			if ( lights[ light.id ] !== undefined ) {

    				return lights[ light.id ];

    			}

    			var uniforms;

    			switch ( light.type ) {

    				case 'DirectionalLight':
    					uniforms = {
    						direction: new Vector3(),
    						color: new Color$7(),

    						shadow: false,
    						shadowBias: 0,
    						shadowRadius: 1,
    						shadowMapSize: new Vector2()
    					};
    					break;

    				case 'SpotLight':
    					uniforms = {
    						position: new Vector3(),
    						direction: new Vector3(),
    						color: new Color$7(),
    						distance: 0,
    						coneCos: 0,
    						penumbraCos: 0,
    						decay: 0,

    						shadow: false,
    						shadowBias: 0,
    						shadowRadius: 1,
    						shadowMapSize: new Vector2()
    					};
    					break;

    				case 'PointLight':
    					uniforms = {
    						position: new Vector3(),
    						color: new Color$7(),
    						distance: 0,
    						decay: 0,

    						shadow: false,
    						shadowBias: 0,
    						shadowRadius: 1,
    						shadowMapSize: new Vector2(),
    						shadowCameraNear: 1,
    						shadowCameraFar: 1000
    					};
    					break;

    				case 'HemisphereLight':
    					uniforms = {
    						direction: new Vector3(),
    						skyColor: new Color$7(),
    						groundColor: new Color$7()
    					};
    					break;

    				case 'RectAreaLight':
    					uniforms = {
    						color: new Color$7(),
    						position: new Vector3(),
    						halfWidth: new Vector3(),
    						halfHeight: new Vector3()
    						// TODO (abelnation): set RectAreaLight shadow uniforms
    					};
    					break;

    			}

    			lights[ light.id ] = uniforms;

    			return uniforms;

    		}

    	};

    }

    var nextVersion = 0;

    function shadowCastingLightsFirst( lightA, lightB ) {

    	return ( lightB.castShadow ? 1 : 0 ) - ( lightA.castShadow ? 1 : 0 );

    }

    function WebGLLights() {

    	var cache = new UniformsCache();

    	var state = {

    		version: 0,

    		hash: {
    			directionalLength: - 1,
    			pointLength: - 1,
    			spotLength: - 1,
    			rectAreaLength: - 1,
    			hemiLength: - 1,

    			numDirectionalShadows: - 1,
    			numPointShadows: - 1,
    			numSpotShadows: - 1,
    		},

    		ambient: [ 0, 0, 0 ],
    		probe: [],
    		directional: [],
    		directionalShadowMap: [],
    		directionalShadowMatrix: [],
    		spot: [],
    		spotShadowMap: [],
    		spotShadowMatrix: [],
    		rectArea: [],
    		point: [],
    		pointShadowMap: [],
    		pointShadowMatrix: [],
    		hemi: [],

    		numDirectionalShadows: - 1,
    		numPointShadows: - 1,
    		numSpotShadows: - 1

    	};

    	for ( var i = 0; i < 9; i ++ ) state.probe.push( new Vector3() );

    	var vector3 = new Vector3();
    	var matrix4 = new Matrix4();
    	var matrix42 = new Matrix4();

    	function setup( lights, shadows, camera ) {

    		var r = 0, g = 0, b = 0;

    		for ( var i = 0; i < 9; i ++ ) state.probe[ i ].set( 0, 0, 0 );

    		var directionalLength = 0;
    		var pointLength = 0;
    		var spotLength = 0;
    		var rectAreaLength = 0;
    		var hemiLength = 0;

    		var numDirectionalShadows = 0;
    		var numPointShadows = 0;
    		var numSpotShadows = 0;

    		var viewMatrix = camera.matrixWorldInverse;

    		lights.sort( shadowCastingLightsFirst );

    		for ( var i = 0, l = lights.length; i < l; i ++ ) {

    			var light = lights[ i ];

    			var color = light.color;
    			var intensity = light.intensity;
    			var distance = light.distance;

    			var shadowMap = ( light.shadow && light.shadow.map ) ? light.shadow.map.texture : null;

    			if ( light.isAmbientLight ) {

    				r += color.r * intensity;
    				g += color.g * intensity;
    				b += color.b * intensity;

    			} else if ( light.isLightProbe ) {

    				for ( var j = 0; j < 9; j ++ ) {

    					state.probe[ j ].addScaledVector( light.sh.coefficients[ j ], intensity );

    				}

    			} else if ( light.isDirectionalLight ) {

    				var uniforms = cache.get( light );

    				uniforms.color.copy( light.color ).multiplyScalar( light.intensity );
    				uniforms.direction.setFromMatrixPosition( light.matrixWorld );
    				vector3.setFromMatrixPosition( light.target.matrixWorld );
    				uniforms.direction.sub( vector3 );
    				uniforms.direction.transformDirection( viewMatrix );

    				uniforms.shadow = light.castShadow;

    				if ( light.castShadow ) {

    					var shadow = light.shadow;

    					uniforms.shadowBias = shadow.bias;
    					uniforms.shadowRadius = shadow.radius;
    					uniforms.shadowMapSize = shadow.mapSize;

    					state.directionalShadowMap[ directionalLength ] = shadowMap;
    					state.directionalShadowMatrix[ directionalLength ] = light.shadow.matrix;

    					numDirectionalShadows ++;

    				}

    				state.directional[ directionalLength ] = uniforms;

    				directionalLength ++;

    			} else if ( light.isSpotLight ) {

    				var uniforms = cache.get( light );

    				uniforms.position.setFromMatrixPosition( light.matrixWorld );
    				uniforms.position.applyMatrix4( viewMatrix );

    				uniforms.color.copy( color ).multiplyScalar( intensity );
    				uniforms.distance = distance;

    				uniforms.direction.setFromMatrixPosition( light.matrixWorld );
    				vector3.setFromMatrixPosition( light.target.matrixWorld );
    				uniforms.direction.sub( vector3 );
    				uniforms.direction.transformDirection( viewMatrix );

    				uniforms.coneCos = Math.cos( light.angle );
    				uniforms.penumbraCos = Math.cos( light.angle * ( 1 - light.penumbra ) );
    				uniforms.decay = light.decay;

    				uniforms.shadow = light.castShadow;

    				if ( light.castShadow ) {

    					var shadow = light.shadow;

    					uniforms.shadowBias = shadow.bias;
    					uniforms.shadowRadius = shadow.radius;
    					uniforms.shadowMapSize = shadow.mapSize;

    					state.spotShadowMap[ spotLength ] = shadowMap;
    					state.spotShadowMatrix[ spotLength ] = light.shadow.matrix;

    					numSpotShadows ++;

    				}

    				state.spot[ spotLength ] = uniforms;

    				spotLength ++;

    			} else if ( light.isRectAreaLight ) {

    				var uniforms = cache.get( light );

    				// (a) intensity is the total visible light emitted
    				//uniforms.color.copy( color ).multiplyScalar( intensity / ( light.width * light.height * Math.PI ) );

    				// (b) intensity is the brightness of the light
    				uniforms.color.copy( color ).multiplyScalar( intensity );

    				uniforms.position.setFromMatrixPosition( light.matrixWorld );
    				uniforms.position.applyMatrix4( viewMatrix );

    				// extract local rotation of light to derive width/height half vectors
    				matrix42.identity();
    				matrix4.copy( light.matrixWorld );
    				matrix4.premultiply( viewMatrix );
    				matrix42.extractRotation( matrix4 );

    				uniforms.halfWidth.set( light.width * 0.5, 0.0, 0.0 );
    				uniforms.halfHeight.set( 0.0, light.height * 0.5, 0.0 );

    				uniforms.halfWidth.applyMatrix4( matrix42 );
    				uniforms.halfHeight.applyMatrix4( matrix42 );

    				// TODO (abelnation): RectAreaLight distance?
    				// uniforms.distance = distance;

    				state.rectArea[ rectAreaLength ] = uniforms;

    				rectAreaLength ++;

    			} else if ( light.isPointLight ) {

    				var uniforms = cache.get( light );

    				uniforms.position.setFromMatrixPosition( light.matrixWorld );
    				uniforms.position.applyMatrix4( viewMatrix );

    				uniforms.color.copy( light.color ).multiplyScalar( light.intensity );
    				uniforms.distance = light.distance;
    				uniforms.decay = light.decay;

    				uniforms.shadow = light.castShadow;

    				if ( light.castShadow ) {

    					var shadow = light.shadow;

    					uniforms.shadowBias = shadow.bias;
    					uniforms.shadowRadius = shadow.radius;
    					uniforms.shadowMapSize = shadow.mapSize;
    					uniforms.shadowCameraNear = shadow.camera.near;
    					uniforms.shadowCameraFar = shadow.camera.far;

    					state.pointShadowMap[ pointLength ] = shadowMap;
    					state.pointShadowMatrix[ pointLength ] = light.shadow.matrix;

    					numPointShadows ++;

    				}

    				state.point[ pointLength ] = uniforms;

    				pointLength ++;

    			} else if ( light.isHemisphereLight ) {

    				var uniforms = cache.get( light );

    				uniforms.direction.setFromMatrixPosition( light.matrixWorld );
    				uniforms.direction.transformDirection( viewMatrix );
    				uniforms.direction.normalize();

    				uniforms.skyColor.copy( light.color ).multiplyScalar( intensity );
    				uniforms.groundColor.copy( light.groundColor ).multiplyScalar( intensity );

    				state.hemi[ hemiLength ] = uniforms;

    				hemiLength ++;

    			}

    		}

    		state.ambient[ 0 ] = r;
    		state.ambient[ 1 ] = g;
    		state.ambient[ 2 ] = b;

    		var hash = state.hash;

    		if ( hash.directionalLength !== directionalLength ||
    			hash.pointLength !== pointLength ||
    			hash.spotLength !== spotLength ||
    			hash.rectAreaLength !== rectAreaLength ||
    			hash.hemiLength !== hemiLength ||
    			hash.numDirectionalShadows !== numDirectionalShadows ||
    			hash.numPointShadows !== numPointShadows ||
    			hash.numSpotShadows !== numSpotShadows ) {

    			state.directional.length = directionalLength;
    			state.spot.length = spotLength;
    			state.rectArea.length = rectAreaLength;
    			state.point.length = pointLength;
    			state.hemi.length = hemiLength;

    			state.directionalShadowMap.length = numDirectionalShadows;
    			state.pointShadowMap.length = numPointShadows;
    			state.spotShadowMap.length = numSpotShadows;
    			state.directionalShadowMatrix.length = numDirectionalShadows;
    			state.pointShadowMatrix.length = numPointShadows;
    			state.spotShadowMatrix.length = numSpotShadows;

    			hash.directionalLength = directionalLength;
    			hash.pointLength = pointLength;
    			hash.spotLength = spotLength;
    			hash.rectAreaLength = rectAreaLength;
    			hash.hemiLength = hemiLength;

    			hash.numDirectionalShadows = numDirectionalShadows;
    			hash.numPointShadows = numPointShadows;
    			hash.numSpotShadows = numSpotShadows;

    			state.version = nextVersion ++;

    		}

    	}

    	return {
    		setup: setup,
    		state: state
    	};

    }

    /**
     * @author Mugen87 / https://github.com/Mugen87
     */

    function WebGLRenderState() {

    	var lights = new WebGLLights();

    	var lightsArray = [];
    	var shadowsArray = [];

    	function init() {

    		lightsArray.length = 0;
    		shadowsArray.length = 0;

    	}

    	function pushLight( light ) {

    		lightsArray.push( light );

    	}

    	function pushShadow( shadowLight ) {

    		shadowsArray.push( shadowLight );

    	}

    	function setupLights( camera ) {

    		lights.setup( lightsArray, shadowsArray, camera );

    	}

    	var state = {
    		lightsArray: lightsArray,
    		shadowsArray: shadowsArray,

    		lights: lights
    	};

    	return {
    		init: init,
    		state: state,
    		setupLights: setupLights,

    		pushLight: pushLight,
    		pushShadow: pushShadow
    	};

    }

    function WebGLRenderStates() {

    	var renderStates = new WeakMap();

    	function onSceneDispose( event ) {

    		var scene = event.target;

    		scene.removeEventListener( 'dispose', onSceneDispose );

    		renderStates.delete( scene );

    	}

    	function get( scene, camera ) {

    		var renderState;

    		if ( renderStates.has( scene ) === false ) {

    			renderState = new WebGLRenderState();
    			renderStates.set( scene, new WeakMap() );
    			renderStates.get( scene ).set( camera, renderState );

    			scene.addEventListener( 'dispose', onSceneDispose );

    		} else {

    			if ( renderStates.get( scene ).has( camera ) === false ) {

    				renderState = new WebGLRenderState();
    				renderStates.get( scene ).set( camera, renderState );

    			} else {

    				renderState = renderStates.get( scene ).get( camera );

    			}

    		}

    		return renderState;

    	}

    	function dispose() {

    		renderStates = new WeakMap();

    	}

    	return {
    		get: get,
    		dispose: dispose
    	};

    }

    /**
     * @author mrdoob / http://mrdoob.com/
     * @author alteredq / http://alteredqualia.com/
     * @author bhouston / https://clara.io
     * @author WestLangley / http://github.com/WestLangley
     *
     * parameters = {
     *
     *  opacity: <float>,
     *
     *  map: new THREE.Texture( <Image> ),
     *
     *  alphaMap: new THREE.Texture( <Image> ),
     *
     *  displacementMap: new THREE.Texture( <Image> ),
     *  displacementScale: <float>,
     *  displacementBias: <float>,
     *
     *  wireframe: <boolean>,
     *  wireframeLinewidth: <float>
     * }
     */

    function MeshDepthMaterial( parameters ) {

    	Material.call( this );

    	this.type = 'MeshDepthMaterial';

    	this.depthPacking = BasicDepthPacking;

    	this.skinning = false;
    	this.morphTargets = false;

    	this.map = null;

    	this.alphaMap = null;

    	this.displacementMap = null;
    	this.displacementScale = 1;
    	this.displacementBias = 0;

    	this.wireframe = false;
    	this.wireframeLinewidth = 1;

    	this.fog = false;
    	this.lights = false;

    	this.setValues( parameters );

    }

    MeshDepthMaterial.prototype = Object.create( Material.prototype );
    MeshDepthMaterial.prototype.constructor = MeshDepthMaterial;

    MeshDepthMaterial.prototype.isMeshDepthMaterial = true;

    MeshDepthMaterial.prototype.copy = function ( source ) {

    	Material.prototype.copy.call( this, source );

    	this.depthPacking = source.depthPacking;

    	this.skinning = source.skinning;
    	this.morphTargets = source.morphTargets;

    	this.map = source.map;

    	this.alphaMap = source.alphaMap;

    	this.displacementMap = source.displacementMap;
    	this.displacementScale = source.displacementScale;
    	this.displacementBias = source.displacementBias;

    	this.wireframe = source.wireframe;
    	this.wireframeLinewidth = source.wireframeLinewidth;

    	return this;

    };

    /**
     * @author WestLangley / http://github.com/WestLangley
     *
     * parameters = {
     *
     *  referencePosition: <float>,
     *  nearDistance: <float>,
     *  farDistance: <float>,
     *
     *  skinning: <bool>,
     *  morphTargets: <bool>,
     *
     *  map: new THREE.Texture( <Image> ),
     *
     *  alphaMap: new THREE.Texture( <Image> ),
     *
     *  displacementMap: new THREE.Texture( <Image> ),
     *  displacementScale: <float>,
     *  displacementBias: <float>
     *
     * }
     */

    function MeshDistanceMaterial( parameters ) {

    	Material.call( this );

    	this.type = 'MeshDistanceMaterial';

    	this.referencePosition = new Vector3();
    	this.nearDistance = 1;
    	this.farDistance = 1000;

    	this.skinning = false;
    	this.morphTargets = false;

    	this.map = null;

    	this.alphaMap = null;

    	this.displacementMap = null;
    	this.displacementScale = 1;
    	this.displacementBias = 0;

    	this.fog = false;
    	this.lights = false;

    	this.setValues( parameters );

    }

    MeshDistanceMaterial.prototype = Object.create( Material.prototype );
    MeshDistanceMaterial.prototype.constructor = MeshDistanceMaterial;

    MeshDistanceMaterial.prototype.isMeshDistanceMaterial = true;

    MeshDistanceMaterial.prototype.copy = function ( source ) {

    	Material.prototype.copy.call( this, source );

    	this.referencePosition.copy( source.referencePosition );
    	this.nearDistance = source.nearDistance;
    	this.farDistance = source.farDistance;

    	this.skinning = source.skinning;
    	this.morphTargets = source.morphTargets;

    	this.map = source.map;

    	this.alphaMap = source.alphaMap;

    	this.displacementMap = source.displacementMap;
    	this.displacementScale = source.displacementScale;
    	this.displacementBias = source.displacementBias;

    	return this;

    };

    var vsm_frag = "uniform sampler2D shadow_pass;\nuniform vec2 resolution;\nuniform float radius;\n#include <packing>\nvoid main() {\n  float mean = 0.0;\n  float squared_mean = 0.0;\n  \n\tfloat depth = unpackRGBAToDepth( texture2D( shadow_pass, ( gl_FragCoord.xy  ) / resolution ) );\n  for ( float i = -1.0; i < 1.0 ; i += SAMPLE_RATE) {\n    #ifdef HORIZONAL_PASS\n      vec2 distribution = decodeHalfRGBA ( texture2D( shadow_pass, ( gl_FragCoord.xy + vec2( i, 0.0 ) * radius ) / resolution ) );\n      mean += distribution.x;\n      squared_mean += distribution.y * distribution.y + distribution.x * distribution.x;\n    #else\n      float depth = unpackRGBAToDepth( texture2D( shadow_pass, ( gl_FragCoord.xy + vec2( 0.0,  i )  * radius ) / resolution ) );\n      mean += depth;\n      squared_mean += depth * depth;\n    #endif\n  }\n  mean = mean * HALF_SAMPLE_RATE;\n  squared_mean = squared_mean * HALF_SAMPLE_RATE;\n  float std_dev = pow( squared_mean - mean * mean, 0.5 );\n  gl_FragColor = encodeHalfRGBA( vec2( mean, std_dev ) );\n}";

    var vsm_vert = "void main() {\n\tgl_Position = vec4( position, 1.0 );\n}";

    /**
     * @author alteredq / http://alteredqualia.com/
     * @author mrdoob / http://mrdoob.com/
     */

    function WebGLShadowMap( _renderer, _objects, maxTextureSize ) {

    	var _frustum = new Frustum(),

    		_shadowMapSize = new Vector2(),
    		_viewportSize = new Vector2(),

    		_viewport = new Vector4(),

    		_MorphingFlag = 1,
    		_SkinningFlag = 2,

    		_NumberOfMaterialVariants = ( _MorphingFlag | _SkinningFlag ) + 1,

    		_depthMaterials = new Array( _NumberOfMaterialVariants ),
    		_distanceMaterials = new Array( _NumberOfMaterialVariants ),

    		_materialCache = {};

    	var shadowSide = { 0: BackSide, 1: FrontSide, 2: DoubleSide };

    	var shadowMaterialVertical = new ShaderMaterial( {

    		defines: {
    			SAMPLE_RATE: 2.0 / 8.0,
    			HALF_SAMPLE_RATE: 1.0 / 8.0
    		},

    		uniforms: {
    			shadow_pass: { value: null },
    			resolution: { value: new Vector2() },
    			radius: { value: 4.0 }
    		},

    		vertexShader: vsm_vert,

    		fragmentShader: vsm_frag

    	} );

    	var shadowMaterialHorizonal = shadowMaterialVertical.clone();
    	shadowMaterialHorizonal.defines.HORIZONAL_PASS = 1;

    	var fullScreenTri = new BufferGeometry();
    	fullScreenTri.addAttribute(
    		"position",
    		new BufferAttribute(
    			new Float32Array( [ - 1, - 1, 0.5, 3, - 1, 0.5, - 1, 3, 0.5 ] ),
    			3
    		)
    	);

    	var fullScreenMesh = new Mesh( fullScreenTri, shadowMaterialVertical );

    	// init

    	for ( var i = 0; i !== _NumberOfMaterialVariants; ++ i ) {

    		var useMorphing = ( i & _MorphingFlag ) !== 0;
    		var useSkinning = ( i & _SkinningFlag ) !== 0;

    		var depthMaterial = new MeshDepthMaterial( {

    			depthPacking: RGBADepthPacking,

    			morphTargets: useMorphing,
    			skinning: useSkinning

    		} );

    		_depthMaterials[ i ] = depthMaterial;

    		var distanceMaterial = new MeshDistanceMaterial( {

    			morphTargets: useMorphing,
    			skinning: useSkinning

    		} );

    		_distanceMaterials[ i ] = distanceMaterial;

    	}

    	var scope = this;

    	this.enabled = false;

    	this.autoUpdate = true;
    	this.needsUpdate = false;

    	this.type = PCFShadowMap;

    	this.render = function ( lights, scene, camera ) {

    		if ( scope.enabled === false ) return;
    		if ( scope.autoUpdate === false && scope.needsUpdate === false ) return;

    		if ( lights.length === 0 ) return;

    		var currentRenderTarget = _renderer.getRenderTarget();
    		var activeCubeFace = _renderer.getActiveCubeFace();
    		var activeMipmapLevel = _renderer.getActiveMipmapLevel();

    		var _state = _renderer.state;

    		// Set GL state for depth map.
    		_state.setBlending( NoBlending );
    		_state.buffers.color.setClear( 1, 1, 1, 1 );
    		_state.buffers.depth.setTest( true );
    		_state.setScissorTest( false );

    		// render depth map

    		for ( var i = 0, il = lights.length; i < il; i ++ ) {

    			var light = lights[ i ];
    			var shadow = light.shadow;

    			if ( shadow === undefined ) {

    				console.warn( 'THREE.WebGLShadowMap:', light, 'has no shadow.' );
    				continue;

    			}

    			_shadowMapSize.copy( shadow.mapSize );

    			var shadowFrameExtents = shadow.getFrameExtents();

    			_shadowMapSize.multiply( shadowFrameExtents );

    			_viewportSize.copy( shadow.mapSize );

    			if ( _shadowMapSize.x > maxTextureSize || _shadowMapSize.y > maxTextureSize ) {

    				console.warn( 'THREE.WebGLShadowMap:', light, 'has shadow exceeding max texture size, reducing' );

    				if ( _shadowMapSize.x > maxTextureSize ) {

    					_viewportSize.x = Math.floor( maxTextureSize / shadowFrameExtents.x );
    					_shadowMapSize.x = _viewportSize.x * shadowFrameExtents.x;
    					shadow.mapSize.x = _viewportSize.x;

    				}

    				if ( _shadowMapSize.y > maxTextureSize ) {

    					_viewportSize.y = Math.floor( maxTextureSize / shadowFrameExtents.y );
    					_shadowMapSize.y = _viewportSize.y * shadowFrameExtents.y;
    					shadow.mapSize.y = _viewportSize.y;

    				}

    			}

    			if ( shadow.map === null && ! shadow.isPointLightShadow && this.type === VSMShadowMap ) {

    				var pars = { minFilter: LinearFilter, magFilter: LinearFilter, format: RGBAFormat };

    				shadow.map = new WebGLRenderTarget( _shadowMapSize.x, _shadowMapSize.y, pars );
    				shadow.map.texture.name = light.name + ".shadowMap";

    				shadow.mapPass = new WebGLRenderTarget( _shadowMapSize.x, _shadowMapSize.y, pars );

    				shadow.camera.updateProjectionMatrix();

    			}

    			if ( shadow.map === null ) {

    				var pars = { minFilter: NearestFilter, magFilter: NearestFilter, format: RGBAFormat };

    				shadow.map = new WebGLRenderTarget( _shadowMapSize.x, _shadowMapSize.y, pars );
    				shadow.map.texture.name = light.name + ".shadowMap";

    				shadow.camera.updateProjectionMatrix();

    			}

    			_renderer.setRenderTarget( shadow.map );
    			_renderer.clear();

    			var viewportCount = shadow.getViewportCount();

    			for ( var vp = 0; vp < viewportCount; vp ++ ) {

    				var viewport = shadow.getViewport( vp );

    				_viewport.set(
    					_viewportSize.x * viewport.x,
    					_viewportSize.y * viewport.y,
    					_viewportSize.x * viewport.z,
    					_viewportSize.y * viewport.w
    				);

    				_state.viewport( _viewport );

    				shadow.updateMatrices( light, camera, vp );

    				_frustum = shadow.getFrustum();

    				renderObject( scene, camera, shadow.camera, light, this.type );

    			}

    			// do blur pass for VSM

    			if ( ! shadow.isPointLightShadow && this.type === VSMShadowMap ) {

    				VSMPass( shadow, camera );

    			}

    		}

    		scope.needsUpdate = false;

    		_renderer.setRenderTarget( currentRenderTarget, activeCubeFace, activeMipmapLevel );

    	};

    	function VSMPass( shadow, camera ) {

    		var geometry = _objects.update( fullScreenMesh );

    		// vertical pass

    		shadowMaterialVertical.uniforms.shadow_pass.value = shadow.map.texture;
    		shadowMaterialVertical.uniforms.resolution.value = shadow.mapSize;
    		shadowMaterialVertical.uniforms.radius.value = shadow.radius;
    		_renderer.setRenderTarget( shadow.mapPass );
    		_renderer.clear();
    		_renderer.renderBufferDirect( camera, null, geometry, shadowMaterialVertical, fullScreenMesh, null );

    		// horizonal pass

    		shadowMaterialHorizonal.uniforms.shadow_pass.value = shadow.mapPass.texture;
    		shadowMaterialHorizonal.uniforms.resolution.value = shadow.mapSize;
    		shadowMaterialHorizonal.uniforms.radius.value = shadow.radius;
    		_renderer.setRenderTarget( shadow.map );
    		_renderer.clear();
    		_renderer.renderBufferDirect( camera, null, geometry, shadowMaterialHorizonal, fullScreenMesh, null );

    	}

    	function getDepthMaterial( object, material, light, shadowCameraNear, shadowCameraFar, type ) {

    		var geometry = object.geometry;

    		var result = null;

    		var materialVariants = _depthMaterials;
    		var customMaterial = object.customDepthMaterial;

    		if ( light.isPointLight ) {

    			materialVariants = _distanceMaterials;
    			customMaterial = object.customDistanceMaterial;

    		}

    		if ( ! customMaterial ) {

    			var useMorphing = false;

    			if ( material.morphTargets ) {

    				if ( geometry && geometry.isBufferGeometry ) {

    					useMorphing = geometry.morphAttributes && geometry.morphAttributes.position && geometry.morphAttributes.position.length > 0;

    				} else if ( geometry && geometry.isGeometry ) {

    					useMorphing = geometry.morphTargets && geometry.morphTargets.length > 0;

    				}

    			}

    			if ( object.isSkinnedMesh && material.skinning === false ) {

    				console.warn( 'THREE.WebGLShadowMap: THREE.SkinnedMesh with material.skinning set to false:', object );

    			}

    			var useSkinning = object.isSkinnedMesh && material.skinning;

    			var variantIndex = 0;

    			if ( useMorphing ) variantIndex |= _MorphingFlag;
    			if ( useSkinning ) variantIndex |= _SkinningFlag;

    			result = materialVariants[ variantIndex ];

    		} else {

    			result = customMaterial;

    		}

    		if ( _renderer.localClippingEnabled &&
    				material.clipShadows === true &&
    				material.clippingPlanes.length !== 0 ) {

    			// in this case we need a unique material instance reflecting the
    			// appropriate state

    			var keyA = result.uuid, keyB = material.uuid;

    			var materialsForVariant = _materialCache[ keyA ];

    			if ( materialsForVariant === undefined ) {

    				materialsForVariant = {};
    				_materialCache[ keyA ] = materialsForVariant;

    			}

    			var cachedMaterial = materialsForVariant[ keyB ];

    			if ( cachedMaterial === undefined ) {

    				cachedMaterial = result.clone();
    				materialsForVariant[ keyB ] = cachedMaterial;

    			}

    			result = cachedMaterial;

    		}

    		result.visible = material.visible;
    		result.wireframe = material.wireframe;

    		if ( type === VSMShadowMap ) {

    			result.side = ( material.shadowSide != null ) ? material.shadowSide : material.side;

    		} else {

    			result.side = ( material.shadowSide != null ) ? material.shadowSide : shadowSide[ material.side ];

    		}

    		result.clipShadows = material.clipShadows;
    		result.clippingPlanes = material.clippingPlanes;
    		result.clipIntersection = material.clipIntersection;

    		result.wireframeLinewidth = material.wireframeLinewidth;
    		result.linewidth = material.linewidth;

    		if ( light.isPointLight && result.isMeshDistanceMaterial ) {

    			result.referencePosition.setFromMatrixPosition( light.matrixWorld );
    			result.nearDistance = shadowCameraNear;
    			result.farDistance = shadowCameraFar;

    		}

    		return result;

    	}

    	function renderObject( object, camera, shadowCamera, light, type ) {

    		if ( object.visible === false ) return;

    		var visible = object.layers.test( camera.layers );

    		if ( visible && ( object.isMesh || object.isLine || object.isPoints ) ) {

    			if ( ( object.castShadow || ( object.receiveShadow && type === VSMShadowMap ) ) && ( ! object.frustumCulled || _frustum.intersectsObject( object ) ) ) {

    				object.modelViewMatrix.multiplyMatrices( shadowCamera.matrixWorldInverse, object.matrixWorld );

    				var geometry = _objects.update( object );
    				var material = object.material;

    				if ( Array.isArray( material ) ) {

    					var groups = geometry.groups;

    					for ( var k = 0, kl = groups.length; k < kl; k ++ ) {

    						var group = groups[ k ];
    						var groupMaterial = material[ group.materialIndex ];

    						if ( groupMaterial && groupMaterial.visible ) {

    							var depthMaterial = getDepthMaterial( object, groupMaterial, light, shadowCamera.near, shadowCamera.far, type );

    							_renderer.renderBufferDirect( shadowCamera, null, geometry, depthMaterial, object, group );

    						}

    					}

    				} else if ( material.visible ) {

    					var depthMaterial = getDepthMaterial( object, material, light, shadowCamera.near, shadowCamera.far, type );

    					_renderer.renderBufferDirect( shadowCamera, null, geometry, depthMaterial, object, null );

    				}

    			}

    		}

    		var children = object.children;

    		for ( var i = 0, l = children.length; i < l; i ++ ) {

    			renderObject( children[ i ], camera, shadowCamera, light, type );

    		}

    	}

    }

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function WebGLState( gl, extensions, utils, capabilities ) {

    	function ColorBuffer() {

    		var locked = false;

    		var color = new Vector4();
    		var currentColorMask = null;
    		var currentColorClear = new Vector4( 0, 0, 0, 0 );

    		return {

    			setMask: function ( colorMask ) {

    				if ( currentColorMask !== colorMask && ! locked ) {

    					gl.colorMask( colorMask, colorMask, colorMask, colorMask );
    					currentColorMask = colorMask;

    				}

    			},

    			setLocked: function ( lock ) {

    				locked = lock;

    			},

    			setClear: function ( r, g, b, a, premultipliedAlpha ) {

    				if ( premultipliedAlpha === true ) {

    					r *= a; g *= a; b *= a;

    				}

    				color.set( r, g, b, a );

    				if ( currentColorClear.equals( color ) === false ) {

    					gl.clearColor( r, g, b, a );
    					currentColorClear.copy( color );

    				}

    			},

    			reset: function () {

    				locked = false;

    				currentColorMask = null;
    				currentColorClear.set( - 1, 0, 0, 0 ); // set to invalid state

    			}

    		};

    	}

    	function DepthBuffer() {

    		var locked = false;

    		var currentDepthMask = null;
    		var currentDepthFunc = null;
    		var currentDepthClear = null;

    		return {

    			setTest: function ( depthTest ) {

    				if ( depthTest ) {

    					enable( 2929 );

    				} else {

    					disable( 2929 );

    				}

    			},

    			setMask: function ( depthMask ) {

    				if ( currentDepthMask !== depthMask && ! locked ) {

    					gl.depthMask( depthMask );
    					currentDepthMask = depthMask;

    				}

    			},

    			setFunc: function ( depthFunc ) {

    				if ( currentDepthFunc !== depthFunc ) {

    					if ( depthFunc ) {

    						switch ( depthFunc ) {

    							case NeverDepth:

    								gl.depthFunc( 512 );
    								break;

    							case AlwaysDepth:

    								gl.depthFunc( 519 );
    								break;

    							case LessDepth:

    								gl.depthFunc( 513 );
    								break;

    							case LessEqualDepth:

    								gl.depthFunc( 515 );
    								break;

    							case EqualDepth:

    								gl.depthFunc( 514 );
    								break;

    							case GreaterEqualDepth:

    								gl.depthFunc( 518 );
    								break;

    							case GreaterDepth:

    								gl.depthFunc( 516 );
    								break;

    							case NotEqualDepth:

    								gl.depthFunc( 517 );
    								break;

    							default:

    								gl.depthFunc( 515 );

    						}

    					} else {

    						gl.depthFunc( 515 );

    					}

    					currentDepthFunc = depthFunc;

    				}

    			},

    			setLocked: function ( lock ) {

    				locked = lock;

    			},

    			setClear: function ( depth ) {

    				if ( currentDepthClear !== depth ) {

    					gl.clearDepth( depth );
    					currentDepthClear = depth;

    				}

    			},

    			reset: function () {

    				locked = false;

    				currentDepthMask = null;
    				currentDepthFunc = null;
    				currentDepthClear = null;

    			}

    		};

    	}

    	function StencilBuffer() {

    		var locked = false;

    		var currentStencilMask = null;
    		var currentStencilFunc = null;
    		var currentStencilRef = null;
    		var currentStencilFuncMask = null;
    		var currentStencilFail = null;
    		var currentStencilZFail = null;
    		var currentStencilZPass = null;
    		var currentStencilClear = null;

    		return {

    			setTest: function ( stencilTest ) {

    				if ( ! locked ) {

    					if ( stencilTest ) {

    						enable( 2960 );

    					} else {

    						disable( 2960 );

    					}

    				}

    			},

    			setMask: function ( stencilMask ) {

    				if ( currentStencilMask !== stencilMask && ! locked ) {

    					gl.stencilMask( stencilMask );
    					currentStencilMask = stencilMask;

    				}

    			},

    			setFunc: function ( stencilFunc, stencilRef, stencilMask ) {

    				if ( currentStencilFunc !== stencilFunc ||
    				     currentStencilRef 	!== stencilRef 	||
    				     currentStencilFuncMask !== stencilMask ) {

    					gl.stencilFunc( stencilFunc, stencilRef, stencilMask );

    					currentStencilFunc = stencilFunc;
    					currentStencilRef = stencilRef;
    					currentStencilFuncMask = stencilMask;

    				}

    			},

    			setOp: function ( stencilFail, stencilZFail, stencilZPass ) {

    				if ( currentStencilFail	 !== stencilFail 	||
    				     currentStencilZFail !== stencilZFail ||
    				     currentStencilZPass !== stencilZPass ) {

    					gl.stencilOp( stencilFail, stencilZFail, stencilZPass );

    					currentStencilFail = stencilFail;
    					currentStencilZFail = stencilZFail;
    					currentStencilZPass = stencilZPass;

    				}

    			},

    			setLocked: function ( lock ) {

    				locked = lock;

    			},

    			setClear: function ( stencil ) {

    				if ( currentStencilClear !== stencil ) {

    					gl.clearStencil( stencil );
    					currentStencilClear = stencil;

    				}

    			},

    			reset: function () {

    				locked = false;

    				currentStencilMask = null;
    				currentStencilFunc = null;
    				currentStencilRef = null;
    				currentStencilFuncMask = null;
    				currentStencilFail = null;
    				currentStencilZFail = null;
    				currentStencilZPass = null;
    				currentStencilClear = null;

    			}

    		};

    	}

    	//

    	var colorBuffer = new ColorBuffer();
    	var depthBuffer = new DepthBuffer();
    	var stencilBuffer = new StencilBuffer();

    	var maxVertexAttributes = gl.getParameter( 34921 );
    	var newAttributes = new Uint8Array( maxVertexAttributes );
    	var enabledAttributes = new Uint8Array( maxVertexAttributes );
    	var attributeDivisors = new Uint8Array( maxVertexAttributes );

    	var enabledCapabilities = {};

    	var compressedTextureFormats = null;

    	var currentProgram = null;

    	var currentBlendingEnabled = null;
    	var currentBlending = null;
    	var currentBlendEquation = null;
    	var currentBlendSrc = null;
    	var currentBlendDst = null;
    	var currentBlendEquationAlpha = null;
    	var currentBlendSrcAlpha = null;
    	var currentBlendDstAlpha = null;
    	var currentPremultipledAlpha = false;

    	var currentFlipSided = null;
    	var currentCullFace = null;

    	var currentLineWidth = null;

    	var currentPolygonOffsetFactor = null;
    	var currentPolygonOffsetUnits = null;

    	var maxTextures = gl.getParameter( 35661 );

    	var lineWidthAvailable = false;
    	var version = 0;
    	var glVersion = gl.getParameter( 7938 );

    	if ( glVersion.indexOf( 'WebGL' ) !== - 1 ) {

    		version = parseFloat( /^WebGL\ ([0-9])/.exec( glVersion )[ 1 ] );
    		lineWidthAvailable = ( version >= 1.0 );

    	} else if ( glVersion.indexOf( 'OpenGL ES' ) !== - 1 ) {

    		version = parseFloat( /^OpenGL\ ES\ ([0-9])/.exec( glVersion )[ 1 ] );
    		lineWidthAvailable = ( version >= 2.0 );

    	}

    	var currentTextureSlot = null;
    	var currentBoundTextures = {};

    	var currentScissor = new Vector4();
    	var currentViewport = new Vector4();

    	function createTexture( type, target, count ) {

    		var data = new Uint8Array( 4 ); // 4 is required to match default unpack alignment of 4.
    		var texture = gl.createTexture();

    		gl.bindTexture( type, texture );
    		gl.texParameteri( type, 10241, 9728 );
    		gl.texParameteri( type, 10240, 9728 );

    		for ( var i = 0; i < count; i ++ ) {

    			gl.texImage2D( target + i, 0, 6408, 1, 1, 0, 6408, 5121, data );

    		}

    		return texture;

    	}

    	var emptyTextures = {};
    	emptyTextures[ 3553 ] = createTexture( 3553, 3553, 1 );
    	emptyTextures[ 34067 ] = createTexture( 34067, 34069, 6 );

    	// init

    	colorBuffer.setClear( 0, 0, 0, 1 );
    	depthBuffer.setClear( 1 );
    	stencilBuffer.setClear( 0 );

    	enable( 2929 );
    	depthBuffer.setFunc( LessEqualDepth );

    	setFlipSided( false );
    	setCullFace( CullFaceBack );
    	enable( 2884 );

    	setBlending( NoBlending );

    	//

    	function initAttributes() {

    		for ( var i = 0, l = newAttributes.length; i < l; i ++ ) {

    			newAttributes[ i ] = 0;

    		}

    	}

    	function enableAttribute( attribute ) {

    		enableAttributeAndDivisor( attribute, 0 );

    	}

    	function enableAttributeAndDivisor( attribute, meshPerAttribute ) {

    		newAttributes[ attribute ] = 1;

    		if ( enabledAttributes[ attribute ] === 0 ) {

    			gl.enableVertexAttribArray( attribute );
    			enabledAttributes[ attribute ] = 1;

    		}

    		if ( attributeDivisors[ attribute ] !== meshPerAttribute ) {

    			var extension = capabilities.isWebGL2 ? gl : extensions.get( 'ANGLE_instanced_arrays' );

    			extension[ capabilities.isWebGL2 ? 'vertexAttribDivisor' : 'vertexAttribDivisorANGLE' ]( attribute, meshPerAttribute );
    			attributeDivisors[ attribute ] = meshPerAttribute;

    		}

    	}

    	function disableUnusedAttributes() {

    		for ( var i = 0, l = enabledAttributes.length; i !== l; ++ i ) {

    			if ( enabledAttributes[ i ] !== newAttributes[ i ] ) {

    				gl.disableVertexAttribArray( i );
    				enabledAttributes[ i ] = 0;

    			}

    		}

    	}

    	function enable( id ) {

    		if ( enabledCapabilities[ id ] !== true ) {

    			gl.enable( id );
    			enabledCapabilities[ id ] = true;

    		}

    	}

    	function disable( id ) {

    		if ( enabledCapabilities[ id ] !== false ) {

    			gl.disable( id );
    			enabledCapabilities[ id ] = false;

    		}

    	}

    	function getCompressedTextureFormats() {

    		if ( compressedTextureFormats === null ) {

    			compressedTextureFormats = [];

    			if ( extensions.get( 'WEBGL_compressed_texture_pvrtc' ) ||
    			     extensions.get( 'WEBGL_compressed_texture_s3tc' ) ||
    			     extensions.get( 'WEBGL_compressed_texture_etc1' ) ||
    			     extensions.get( 'WEBGL_compressed_texture_astc' ) ) {

    				var formats = gl.getParameter( 34467 );

    				for ( var i = 0; i < formats.length; i ++ ) {

    					compressedTextureFormats.push( formats[ i ] );

    				}

    			}

    		}

    		return compressedTextureFormats;

    	}

    	function useProgram( program ) {

    		if ( currentProgram !== program ) {

    			gl.useProgram( program );

    			currentProgram = program;

    			return true;

    		}

    		return false;

    	}

    	function setBlending( blending, blendEquation, blendSrc, blendDst, blendEquationAlpha, blendSrcAlpha, blendDstAlpha, premultipliedAlpha ) {

    		if ( blending === NoBlending ) {

    			if ( currentBlendingEnabled ) {

    				disable( 3042 );
    				currentBlendingEnabled = false;

    			}

    			return;

    		}

    		if ( ! currentBlendingEnabled ) {

    			enable( 3042 );
    			currentBlendingEnabled = true;

    		}

    		if ( blending !== CustomBlending ) {

    			if ( blending !== currentBlending || premultipliedAlpha !== currentPremultipledAlpha ) {

    				if ( currentBlendEquation !== AddEquation || currentBlendEquationAlpha !== AddEquation ) {

    					gl.blendEquation( 32774 );

    					currentBlendEquation = AddEquation;
    					currentBlendEquationAlpha = AddEquation;

    				}

    				if ( premultipliedAlpha ) {

    					switch ( blending ) {

    						case NormalBlending:
    							gl.blendFuncSeparate( 1, 771, 1, 771 );
    							break;

    						case AdditiveBlending:
    							gl.blendFunc( 1, 1 );
    							break;

    						case SubtractiveBlending:
    							gl.blendFuncSeparate( 0, 0, 769, 771 );
    							break;

    						case MultiplyBlending:
    							gl.blendFuncSeparate( 0, 768, 0, 770 );
    							break;

    						default:
    							console.error( 'THREE.WebGLState: Invalid blending: ', blending );
    							break;

    					}

    				} else {

    					switch ( blending ) {

    						case NormalBlending:
    							gl.blendFuncSeparate( 770, 771, 1, 771 );
    							break;

    						case AdditiveBlending:
    							gl.blendFunc( 770, 1 );
    							break;

    						case SubtractiveBlending:
    							gl.blendFunc( 0, 769 );
    							break;

    						case MultiplyBlending:
    							gl.blendFunc( 0, 768 );
    							break;

    						default:
    							console.error( 'THREE.WebGLState: Invalid blending: ', blending );
    							break;

    					}

    				}

    				currentBlendSrc = null;
    				currentBlendDst = null;
    				currentBlendSrcAlpha = null;
    				currentBlendDstAlpha = null;

    				currentBlending = blending;
    				currentPremultipledAlpha = premultipliedAlpha;

    			}

    			return;

    		}

    		// custom blending

    		blendEquationAlpha = blendEquationAlpha || blendEquation;
    		blendSrcAlpha = blendSrcAlpha || blendSrc;
    		blendDstAlpha = blendDstAlpha || blendDst;

    		if ( blendEquation !== currentBlendEquation || blendEquationAlpha !== currentBlendEquationAlpha ) {

    			gl.blendEquationSeparate( utils.convert( blendEquation ), utils.convert( blendEquationAlpha ) );

    			currentBlendEquation = blendEquation;
    			currentBlendEquationAlpha = blendEquationAlpha;

    		}

    		if ( blendSrc !== currentBlendSrc || blendDst !== currentBlendDst || blendSrcAlpha !== currentBlendSrcAlpha || blendDstAlpha !== currentBlendDstAlpha ) {

    			gl.blendFuncSeparate( utils.convert( blendSrc ), utils.convert( blendDst ), utils.convert( blendSrcAlpha ), utils.convert( blendDstAlpha ) );

    			currentBlendSrc = blendSrc;
    			currentBlendDst = blendDst;
    			currentBlendSrcAlpha = blendSrcAlpha;
    			currentBlendDstAlpha = blendDstAlpha;

    		}

    		currentBlending = blending;
    		currentPremultipledAlpha = null;

    	}

    	function setMaterial( material, frontFaceCW ) {

    		material.side === DoubleSide
    			? disable( 2884 )
    			: enable( 2884 );

    		var flipSided = ( material.side === BackSide );
    		if ( frontFaceCW ) flipSided = ! flipSided;

    		setFlipSided( flipSided );

    		( material.blending === NormalBlending && material.transparent === false )
    			? setBlending( NoBlending )
    			: setBlending( material.blending, material.blendEquation, material.blendSrc, material.blendDst, material.blendEquationAlpha, material.blendSrcAlpha, material.blendDstAlpha, material.premultipliedAlpha );

    		depthBuffer.setFunc( material.depthFunc );
    		depthBuffer.setTest( material.depthTest );
    		depthBuffer.setMask( material.depthWrite );
    		colorBuffer.setMask( material.colorWrite );

    		var stencilWrite = material.stencilWrite;
    		stencilBuffer.setTest( stencilWrite );
    		if ( stencilWrite ) {

    			stencilBuffer.setFunc( material.stencilFunc, material.stencilRef, material.stencilMask );
    			stencilBuffer.setOp( material.stencilFail, material.stencilZFail, material.stencilZPass );

    		}

    		setPolygonOffset( material.polygonOffset, material.polygonOffsetFactor, material.polygonOffsetUnits );

    	}

    	//

    	function setFlipSided( flipSided ) {

    		if ( currentFlipSided !== flipSided ) {

    			if ( flipSided ) {

    				gl.frontFace( 2304 );

    			} else {

    				gl.frontFace( 2305 );

    			}

    			currentFlipSided = flipSided;

    		}

    	}

    	function setCullFace( cullFace ) {

    		if ( cullFace !== CullFaceNone ) {

    			enable( 2884 );

    			if ( cullFace !== currentCullFace ) {

    				if ( cullFace === CullFaceBack ) {

    					gl.cullFace( 1029 );

    				} else if ( cullFace === CullFaceFront ) {

    					gl.cullFace( 1028 );

    				} else {

    					gl.cullFace( 1032 );

    				}

    			}

    		} else {

    			disable( 2884 );

    		}

    		currentCullFace = cullFace;

    	}

    	function setLineWidth( width ) {

    		if ( width !== currentLineWidth ) {

    			if ( lineWidthAvailable ) gl.lineWidth( width );

    			currentLineWidth = width;

    		}

    	}

    	function setPolygonOffset( polygonOffset, factor, units ) {

    		if ( polygonOffset ) {

    			enable( 32823 );

    			if ( currentPolygonOffsetFactor !== factor || currentPolygonOffsetUnits !== units ) {

    				gl.polygonOffset( factor, units );

    				currentPolygonOffsetFactor = factor;
    				currentPolygonOffsetUnits = units;

    			}

    		} else {

    			disable( 32823 );

    		}

    	}

    	function setScissorTest( scissorTest ) {

    		if ( scissorTest ) {

    			enable( 3089 );

    		} else {

    			disable( 3089 );

    		}

    	}

    	// texture

    	function activeTexture( webglSlot ) {

    		if ( webglSlot === undefined ) webglSlot = 33984 + maxTextures - 1;

    		if ( currentTextureSlot !== webglSlot ) {

    			gl.activeTexture( webglSlot );
    			currentTextureSlot = webglSlot;

    		}

    	}

    	function bindTexture( webglType, webglTexture ) {

    		if ( currentTextureSlot === null ) {

    			activeTexture();

    		}

    		var boundTexture = currentBoundTextures[ currentTextureSlot ];

    		if ( boundTexture === undefined ) {

    			boundTexture = { type: undefined, texture: undefined };
    			currentBoundTextures[ currentTextureSlot ] = boundTexture;

    		}

    		if ( boundTexture.type !== webglType || boundTexture.texture !== webglTexture ) {

    			gl.bindTexture( webglType, webglTexture || emptyTextures[ webglType ] );

    			boundTexture.type = webglType;
    			boundTexture.texture = webglTexture;

    		}

    	}

    	function compressedTexImage2D() {

    		try {

    			gl.compressedTexImage2D.apply( gl, arguments );

    		} catch ( error ) {

    			console.error( 'THREE.WebGLState:', error );

    		}

    	}

    	function texImage2D() {

    		try {

    			gl.texImage2D.apply( gl, arguments );

    		} catch ( error ) {

    			console.error( 'THREE.WebGLState:', error );

    		}

    	}

    	function texImage3D() {

    		try {

    			gl.texImage3D.apply( gl, arguments );

    		} catch ( error ) {

    			console.error( 'THREE.WebGLState:', error );

    		}

    	}

    	//

    	function scissor( scissor ) {

    		if ( currentScissor.equals( scissor ) === false ) {

    			gl.scissor( scissor.x, scissor.y, scissor.z, scissor.w );
    			currentScissor.copy( scissor );

    		}

    	}

    	function viewport( viewport ) {

    		if ( currentViewport.equals( viewport ) === false ) {

    			gl.viewport( viewport.x, viewport.y, viewport.z, viewport.w );
    			currentViewport.copy( viewport );

    		}

    	}

    	//

    	function reset() {

    		for ( var i = 0; i < enabledAttributes.length; i ++ ) {

    			if ( enabledAttributes[ i ] === 1 ) {

    				gl.disableVertexAttribArray( i );
    				enabledAttributes[ i ] = 0;

    			}

    		}

    		enabledCapabilities = {};

    		compressedTextureFormats = null;

    		currentTextureSlot = null;
    		currentBoundTextures = {};

    		currentProgram = null;

    		currentBlending = null;

    		currentFlipSided = null;
    		currentCullFace = null;

    		colorBuffer.reset();
    		depthBuffer.reset();
    		stencilBuffer.reset();

    	}

    	return {

    		buffers: {
    			color: colorBuffer,
    			depth: depthBuffer,
    			stencil: stencilBuffer
    		},

    		initAttributes: initAttributes,
    		enableAttribute: enableAttribute,
    		enableAttributeAndDivisor: enableAttributeAndDivisor,
    		disableUnusedAttributes: disableUnusedAttributes,
    		enable: enable,
    		disable: disable,
    		getCompressedTextureFormats: getCompressedTextureFormats,

    		useProgram: useProgram,

    		setBlending: setBlending,
    		setMaterial: setMaterial,

    		setFlipSided: setFlipSided,
    		setCullFace: setCullFace,

    		setLineWidth: setLineWidth,
    		setPolygonOffset: setPolygonOffset,

    		setScissorTest: setScissorTest,

    		activeTexture: activeTexture,
    		bindTexture: bindTexture,
    		compressedTexImage2D: compressedTexImage2D,
    		texImage2D: texImage2D,
    		texImage3D: texImage3D,

    		scissor: scissor,
    		viewport: viewport,

    		reset: reset

    	};

    }

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function WebGLTextures( _gl, extensions, state, properties, capabilities, utils, info ) {

    	var _videoTextures = new WeakMap();
    	var _canvas;

    	//

    	var useOffscreenCanvas = typeof OffscreenCanvas !== 'undefined';

    	function createCanvas( width, height ) {

    		// Use OffscreenCanvas when available. Specially needed in web workers

    		return useOffscreenCanvas ?
    			new OffscreenCanvas( width, height ) :
    			document.createElementNS( 'http://www.w3.org/1999/xhtml', 'canvas' );

    	}

    	function resizeImage( image, needsPowerOfTwo, needsNewCanvas, maxSize ) {

    		var scale = 1;

    		// handle case if texture exceeds max size

    		if ( image.width > maxSize || image.height > maxSize ) {

    			scale = maxSize / Math.max( image.width, image.height );

    		}

    		// only perform resize if necessary

    		if ( scale < 1 || needsPowerOfTwo === true ) {

    			// only perform resize for certain image types

    			if ( ( typeof HTMLImageElement !== 'undefined' && image instanceof HTMLImageElement ) ||
    				( typeof HTMLCanvasElement !== 'undefined' && image instanceof HTMLCanvasElement ) ||
    				( typeof ImageBitmap !== 'undefined' && image instanceof ImageBitmap ) ) {

    				var floor = needsPowerOfTwo ? _Math.floorPowerOfTwo : Math.floor;

    				var width = floor( scale * image.width );
    				var height = floor( scale * image.height );

    				if ( _canvas === undefined ) _canvas = createCanvas( width, height );

    				// cube textures can't reuse the same canvas

    				var canvas = needsNewCanvas ? createCanvas( width, height ) : _canvas;

    				canvas.width = width;
    				canvas.height = height;

    				var context = canvas.getContext( '2d' );
    				context.drawImage( image, 0, 0, width, height );

    				console.warn( 'THREE.WebGLRenderer: Texture has been resized from (' + image.width + 'x' + image.height + ') to (' + width + 'x' + height + ').' );

    				return canvas;

    			} else {

    				if ( 'data' in image ) {

    					console.warn( 'THREE.WebGLRenderer: Image in DataTexture is too big (' + image.width + 'x' + image.height + ').' );

    				}

    				return image;

    			}

    		}

    		return image;

    	}

    	function isPowerOfTwo( image ) {

    		return _Math.isPowerOfTwo( image.width ) && _Math.isPowerOfTwo( image.height );

    	}

    	function textureNeedsPowerOfTwo( texture ) {

    		if ( capabilities.isWebGL2 ) return false;

    		return ( texture.wrapS !== ClampToEdgeWrapping || texture.wrapT !== ClampToEdgeWrapping ) ||
    			( texture.minFilter !== NearestFilter && texture.minFilter !== LinearFilter );

    	}

    	function textureNeedsGenerateMipmaps( texture, supportsMips ) {

    		return texture.generateMipmaps && supportsMips &&
    			texture.minFilter !== NearestFilter && texture.minFilter !== LinearFilter;

    	}

    	function generateMipmap( target, texture, width, height ) {

    		_gl.generateMipmap( target );

    		var textureProperties = properties.get( texture );

    		// Note: Math.log( x ) * Math.LOG2E used instead of Math.log2( x ) which is not supported by IE11
    		textureProperties.__maxMipLevel = Math.log( Math.max( width, height ) ) * Math.LOG2E;

    	}

    	function getInternalFormat( glFormat, glType ) {

    		if ( ! capabilities.isWebGL2 ) return glFormat;

    		var internalFormat = glFormat;

    		if ( glFormat === 6403 ) {

    			if ( glType === 5126 ) internalFormat = 33326;
    			if ( glType === 5131 ) internalFormat = 33325;
    			if ( glType === 5121 ) internalFormat = 33321;

    		}

    		if ( glFormat === 6407 ) {

    			if ( glType === 5126 ) internalFormat = 34837;
    			if ( glType === 5131 ) internalFormat = 34843;
    			if ( glType === 5121 ) internalFormat = 32849;

    		}

    		if ( glFormat === 6408 ) {

    			if ( glType === 5126 ) internalFormat = 34836;
    			if ( glType === 5131 ) internalFormat = 34842;
    			if ( glType === 5121 ) internalFormat = 32856;

    		}

    		if ( internalFormat === 33325 || internalFormat === 33326 ||
    			internalFormat === 34842 || internalFormat === 34836 ) {

    			extensions.get( 'EXT_color_buffer_float' );

    		} else if ( internalFormat === 34843 || internalFormat === 34837 ) {

    			console.warn( 'THREE.WebGLRenderer: Floating point textures with RGB format not supported. Please use RGBA instead.' );

    		}

    		return internalFormat;

    	}

    	// Fallback filters for non-power-of-2 textures

    	function filterFallback( f ) {

    		if ( f === NearestFilter || f === NearestMipmapNearestFilter || f === NearestMipmapLinearFilter ) {

    			return 9728;

    		}

    		return 9729;

    	}

    	//

    	function onTextureDispose( event ) {

    		var texture = event.target;

    		texture.removeEventListener( 'dispose', onTextureDispose );

    		deallocateTexture( texture );

    		if ( texture.isVideoTexture ) {

    			_videoTextures.delete( texture );

    		}

    		info.memory.textures --;

    	}

    	function onRenderTargetDispose( event ) {

    		var renderTarget = event.target;

    		renderTarget.removeEventListener( 'dispose', onRenderTargetDispose );

    		deallocateRenderTarget( renderTarget );

    		info.memory.textures --;

    	}

    	//

    	function deallocateTexture( texture ) {

    		var textureProperties = properties.get( texture );

    		if ( textureProperties.__webglInit === undefined ) return;

    		_gl.deleteTexture( textureProperties.__webglTexture );

    		properties.remove( texture );

    	}

    	function deallocateRenderTarget( renderTarget ) {

    		var renderTargetProperties = properties.get( renderTarget );
    		var textureProperties = properties.get( renderTarget.texture );

    		if ( ! renderTarget ) return;

    		if ( textureProperties.__webglTexture !== undefined ) {

    			_gl.deleteTexture( textureProperties.__webglTexture );

    		}

    		if ( renderTarget.depthTexture ) {

    			renderTarget.depthTexture.dispose();

    		}

    		if ( renderTarget.isWebGLRenderTargetCube ) {

    			for ( var i = 0; i < 6; i ++ ) {

    				_gl.deleteFramebuffer( renderTargetProperties.__webglFramebuffer[ i ] );
    				if ( renderTargetProperties.__webglDepthbuffer ) _gl.deleteRenderbuffer( renderTargetProperties.__webglDepthbuffer[ i ] );

    			}

    		} else {

    			_gl.deleteFramebuffer( renderTargetProperties.__webglFramebuffer );
    			if ( renderTargetProperties.__webglDepthbuffer ) _gl.deleteRenderbuffer( renderTargetProperties.__webglDepthbuffer );

    		}

    		properties.remove( renderTarget.texture );
    		properties.remove( renderTarget );

    	}

    	//

    	var textureUnits = 0;

    	function resetTextureUnits() {

    		textureUnits = 0;

    	}

    	function allocateTextureUnit() {

    		var textureUnit = textureUnits;

    		if ( textureUnit >= capabilities.maxTextures ) {

    			console.warn( 'THREE.WebGLTextures: Trying to use ' + textureUnit + ' texture units while this GPU supports only ' + capabilities.maxTextures );

    		}

    		textureUnits += 1;

    		return textureUnit;

    	}

    	//

    	function setTexture2D( texture, slot ) {

    		var textureProperties = properties.get( texture );

    		if ( texture.isVideoTexture ) updateVideoTexture( texture );

    		if ( texture.version > 0 && textureProperties.__version !== texture.version ) {

    			var image = texture.image;

    			if ( image === undefined ) {

    				console.warn( 'THREE.WebGLRenderer: Texture marked for update but image is undefined' );

    			} else if ( image.complete === false ) {

    				console.warn( 'THREE.WebGLRenderer: Texture marked for update but image is incomplete' );

    			} else {

    				uploadTexture( textureProperties, texture, slot );
    				return;

    			}

    		}

    		state.activeTexture( 33984 + slot );
    		state.bindTexture( 3553, textureProperties.__webglTexture );

    	}

    	function setTexture2DArray( texture, slot ) {

    		var textureProperties = properties.get( texture );

    		if ( texture.version > 0 && textureProperties.__version !== texture.version ) {

    			uploadTexture( textureProperties, texture, slot );
    			return;

    		}

    		state.activeTexture( 33984 + slot );
    		state.bindTexture( 35866, textureProperties.__webglTexture );

    	}

    	function setTexture3D( texture, slot ) {

    		var textureProperties = properties.get( texture );

    		if ( texture.version > 0 && textureProperties.__version !== texture.version ) {

    			uploadTexture( textureProperties, texture, slot );
    			return;

    		}

    		state.activeTexture( 33984 + slot );
    		state.bindTexture( 32879, textureProperties.__webglTexture );

    	}

    	function setTextureCube( texture, slot ) {

    		if ( texture.image.length !== 6 ) return;

    		var textureProperties = properties.get( texture );

    		if ( texture.version > 0 && textureProperties.__version !== texture.version ) {

    			initTexture( textureProperties, texture );

    			state.activeTexture( 33984 + slot );
    			state.bindTexture( 34067, textureProperties.__webglTexture );

    			_gl.pixelStorei( 37440, texture.flipY );

    			var isCompressed = ( texture && texture.isCompressedTexture );
    			var isDataTexture = ( texture.image[ 0 ] && texture.image[ 0 ].isDataTexture );

    			var cubeImage = [];

    			for ( var i = 0; i < 6; i ++ ) {

    				if ( ! isCompressed && ! isDataTexture ) {

    					cubeImage[ i ] = resizeImage( texture.image[ i ], false, true, capabilities.maxCubemapSize );

    				} else {

    					cubeImage[ i ] = isDataTexture ? texture.image[ i ].image : texture.image[ i ];

    				}

    			}

    			var image = cubeImage[ 0 ],
    				supportsMips = isPowerOfTwo( image ) || capabilities.isWebGL2,
    				glFormat = utils.convert( texture.format ),
    				glType = utils.convert( texture.type ),
    				glInternalFormat = getInternalFormat( glFormat, glType );

    			setTextureParameters( 34067, texture, supportsMips );

    			var mipmaps;

    			if ( isCompressed ) {

    				for ( var i = 0; i < 6; i ++ ) {

    					mipmaps = cubeImage[ i ].mipmaps;

    					for ( var j = 0; j < mipmaps.length; j ++ ) {

    						var mipmap = mipmaps[ j ];

    						if ( texture.format !== RGBAFormat && texture.format !== RGBFormat ) {

    							if ( state.getCompressedTextureFormats().indexOf( glFormat ) > - 1 ) {

    								state.compressedTexImage2D( 34069 + i, j, glInternalFormat, mipmap.width, mipmap.height, 0, mipmap.data );

    							} else {

    								console.warn( 'THREE.WebGLRenderer: Attempt to load unsupported compressed texture format in .setTextureCube()' );

    							}

    						} else {

    							state.texImage2D( 34069 + i, j, glInternalFormat, mipmap.width, mipmap.height, 0, glFormat, glType, mipmap.data );

    						}

    					}

    				}

    				textureProperties.__maxMipLevel = mipmaps.length - 1;

    			} else {

    				mipmaps = texture.mipmaps;

    				for ( var i = 0; i < 6; i ++ ) {

    					if ( isDataTexture ) {

    						state.texImage2D( 34069 + i, 0, glInternalFormat, cubeImage[ i ].width, cubeImage[ i ].height, 0, glFormat, glType, cubeImage[ i ].data );

    						for ( var j = 0; j < mipmaps.length; j ++ ) {

    							var mipmap = mipmaps[ j ];
    							var mipmapImage = mipmap.image[ i ].image;

    							state.texImage2D( 34069 + i, j + 1, glInternalFormat, mipmapImage.width, mipmapImage.height, 0, glFormat, glType, mipmapImage.data );

    						}

    					} else {

    						state.texImage2D( 34069 + i, 0, glInternalFormat, glFormat, glType, cubeImage[ i ] );

    						for ( var j = 0; j < mipmaps.length; j ++ ) {

    							var mipmap = mipmaps[ j ];

    							state.texImage2D( 34069 + i, j + 1, glInternalFormat, glFormat, glType, mipmap.image[ i ] );

    						}

    					}

    				}

    				textureProperties.__maxMipLevel = mipmaps.length;

    			}

    			if ( textureNeedsGenerateMipmaps( texture, supportsMips ) ) {

    				// We assume images for cube map have the same size.
    				generateMipmap( 34067, texture, image.width, image.height );

    			}

    			textureProperties.__version = texture.version;

    			if ( texture.onUpdate ) texture.onUpdate( texture );

    		} else {

    			state.activeTexture( 33984 + slot );
    			state.bindTexture( 34067, textureProperties.__webglTexture );

    		}

    	}

    	function setTextureCubeDynamic( texture, slot ) {

    		state.activeTexture( 33984 + slot );
    		state.bindTexture( 34067, properties.get( texture ).__webglTexture );

    	}

    	function setTextureParameters( textureType, texture, supportsMips ) {

    		var extension;

    		if ( supportsMips ) {

    			_gl.texParameteri( textureType, 10242, utils.convert( texture.wrapS ) );
    			_gl.texParameteri( textureType, 10243, utils.convert( texture.wrapT ) );

    			if ( textureType === 32879 || textureType === 35866 ) {

    				_gl.texParameteri( textureType, 32882, utils.convert( texture.wrapR ) );

    			}

    			_gl.texParameteri( textureType, 10240, utils.convert( texture.magFilter ) );
    			_gl.texParameteri( textureType, 10241, utils.convert( texture.minFilter ) );

    		} else {

    			_gl.texParameteri( textureType, 10242, 33071 );
    			_gl.texParameteri( textureType, 10243, 33071 );

    			if ( textureType === 32879 || textureType === 35866 ) {

    				_gl.texParameteri( textureType, 32882, 33071 );

    			}

    			if ( texture.wrapS !== ClampToEdgeWrapping || texture.wrapT !== ClampToEdgeWrapping ) {

    				console.warn( 'THREE.WebGLRenderer: Texture is not power of two. Texture.wrapS and Texture.wrapT should be set to THREE.ClampToEdgeWrapping.' );

    			}

    			_gl.texParameteri( textureType, 10240, filterFallback( texture.magFilter ) );
    			_gl.texParameteri( textureType, 10241, filterFallback( texture.minFilter ) );

    			if ( texture.minFilter !== NearestFilter && texture.minFilter !== LinearFilter ) {

    				console.warn( 'THREE.WebGLRenderer: Texture is not power of two. Texture.minFilter should be set to THREE.NearestFilter or THREE.LinearFilter.' );

    			}

    		}

    		extension = extensions.get( 'EXT_texture_filter_anisotropic' );

    		if ( extension ) {

    			if ( texture.type === FloatType && extensions.get( 'OES_texture_float_linear' ) === null ) return;
    			if ( texture.type === HalfFloatType && ( capabilities.isWebGL2 || extensions.get( 'OES_texture_half_float_linear' ) ) === null ) return;

    			if ( texture.anisotropy > 1 || properties.get( texture ).__currentAnisotropy ) {

    				_gl.texParameterf( textureType, extension.TEXTURE_MAX_ANISOTROPY_EXT, Math.min( texture.anisotropy, capabilities.getMaxAnisotropy() ) );
    				properties.get( texture ).__currentAnisotropy = texture.anisotropy;

    			}

    		}

    	}

    	function initTexture( textureProperties, texture ) {

    		if ( textureProperties.__webglInit === undefined ) {

    			textureProperties.__webglInit = true;

    			texture.addEventListener( 'dispose', onTextureDispose );

    			textureProperties.__webglTexture = _gl.createTexture();

    			info.memory.textures ++;

    		}

    	}

    	function uploadTexture( textureProperties, texture, slot ) {

    		var textureType = 3553;

    		if ( texture.isDataTexture2DArray ) textureType = 35866;
    		if ( texture.isDataTexture3D ) textureType = 32879;

    		initTexture( textureProperties, texture );

    		state.activeTexture( 33984 + slot );
    		state.bindTexture( textureType, textureProperties.__webglTexture );

    		_gl.pixelStorei( 37440, texture.flipY );
    		_gl.pixelStorei( 37441, texture.premultiplyAlpha );
    		_gl.pixelStorei( 3317, texture.unpackAlignment );

    		var needsPowerOfTwo = textureNeedsPowerOfTwo( texture ) && isPowerOfTwo( texture.image ) === false;
    		var image = resizeImage( texture.image, needsPowerOfTwo, false, capabilities.maxTextureSize );

    		var supportsMips = isPowerOfTwo( image ) || capabilities.isWebGL2,
    			glFormat = utils.convert( texture.format ),
    			glType = utils.convert( texture.type ),
    			glInternalFormat = getInternalFormat( glFormat, glType );

    		setTextureParameters( textureType, texture, supportsMips );

    		var mipmap, mipmaps = texture.mipmaps;

    		if ( texture.isDepthTexture ) {

    			// populate depth texture with dummy data

    			glInternalFormat = 6402;

    			if ( texture.type === FloatType ) {

    				if ( ! capabilities.isWebGL2 ) throw new Error( 'Float Depth Texture only supported in WebGL2.0' );
    				glInternalFormat = 36012;

    			} else if ( capabilities.isWebGL2 ) {

    				// WebGL 2.0 requires signed internalformat for glTexImage2D
    				glInternalFormat = 33189;

    			}

    			if ( texture.format === DepthFormat && glInternalFormat === 6402 ) {

    				// The error INVALID_OPERATION is generated by texImage2D if format and internalformat are
    				// DEPTH_COMPONENT and type is not UNSIGNED_SHORT or UNSIGNED_INT
    				// (https://www.khronos.org/registry/webgl/extensions/WEBGL_depth_texture/)
    				if ( texture.type !== UnsignedShortType && texture.type !== UnsignedIntType ) {

    					console.warn( 'THREE.WebGLRenderer: Use UnsignedShortType or UnsignedIntType for DepthFormat DepthTexture.' );

    					texture.type = UnsignedShortType;
    					glType = utils.convert( texture.type );

    				}

    			}

    			// Depth stencil textures need the DEPTH_STENCIL internal format
    			// (https://www.khronos.org/registry/webgl/extensions/WEBGL_depth_texture/)
    			if ( texture.format === DepthStencilFormat ) {

    				glInternalFormat = 34041;

    				// The error INVALID_OPERATION is generated by texImage2D if format and internalformat are
    				// DEPTH_STENCIL and type is not UNSIGNED_INT_24_8_WEBGL.
    				// (https://www.khronos.org/registry/webgl/extensions/WEBGL_depth_texture/)
    				if ( texture.type !== UnsignedInt248Type ) {

    					console.warn( 'THREE.WebGLRenderer: Use UnsignedInt248Type for DepthStencilFormat DepthTexture.' );

    					texture.type = UnsignedInt248Type;
    					glType = utils.convert( texture.type );

    				}

    			}

    			state.texImage2D( 3553, 0, glInternalFormat, image.width, image.height, 0, glFormat, glType, null );

    		} else if ( texture.isDataTexture ) {

    			// use manually created mipmaps if available
    			// if there are no manual mipmaps
    			// set 0 level mipmap and then use GL to generate other mipmap levels

    			if ( mipmaps.length > 0 && supportsMips ) {

    				for ( var i = 0, il = mipmaps.length; i < il; i ++ ) {

    					mipmap = mipmaps[ i ];
    					state.texImage2D( 3553, i, glInternalFormat, mipmap.width, mipmap.height, 0, glFormat, glType, mipmap.data );

    				}

    				texture.generateMipmaps = false;
    				textureProperties.__maxMipLevel = mipmaps.length - 1;

    			} else {

    				state.texImage2D( 3553, 0, glInternalFormat, image.width, image.height, 0, glFormat, glType, image.data );
    				textureProperties.__maxMipLevel = 0;

    			}

    		} else if ( texture.isCompressedTexture ) {

    			for ( var i = 0, il = mipmaps.length; i < il; i ++ ) {

    				mipmap = mipmaps[ i ];

    				if ( texture.format !== RGBAFormat && texture.format !== RGBFormat ) {

    					if ( state.getCompressedTextureFormats().indexOf( glFormat ) > - 1 ) {

    						state.compressedTexImage2D( 3553, i, glInternalFormat, mipmap.width, mipmap.height, 0, mipmap.data );

    					} else {

    						console.warn( 'THREE.WebGLRenderer: Attempt to load unsupported compressed texture format in .uploadTexture()' );

    					}

    				} else {

    					state.texImage2D( 3553, i, glInternalFormat, mipmap.width, mipmap.height, 0, glFormat, glType, mipmap.data );

    				}

    			}

    			textureProperties.__maxMipLevel = mipmaps.length - 1;

    		} else if ( texture.isDataTexture2DArray ) {

    			state.texImage3D( 35866, 0, glInternalFormat, image.width, image.height, image.depth, 0, glFormat, glType, image.data );
    			textureProperties.__maxMipLevel = 0;

    		} else if ( texture.isDataTexture3D ) {

    			state.texImage3D( 32879, 0, glInternalFormat, image.width, image.height, image.depth, 0, glFormat, glType, image.data );
    			textureProperties.__maxMipLevel = 0;

    		} else {

    			// regular Texture (image, video, canvas)

    			// use manually created mipmaps if available
    			// if there are no manual mipmaps
    			// set 0 level mipmap and then use GL to generate other mipmap levels

    			if ( mipmaps.length > 0 && supportsMips ) {

    				for ( var i = 0, il = mipmaps.length; i < il; i ++ ) {

    					mipmap = mipmaps[ i ];
    					state.texImage2D( 3553, i, glInternalFormat, glFormat, glType, mipmap );

    				}

    				texture.generateMipmaps = false;
    				textureProperties.__maxMipLevel = mipmaps.length - 1;

    			} else {

    				state.texImage2D( 3553, 0, glInternalFormat, glFormat, glType, image );
    				textureProperties.__maxMipLevel = 0;

    			}

    		}

    		if ( textureNeedsGenerateMipmaps( texture, supportsMips ) ) {

    			generateMipmap( 3553, texture, image.width, image.height );

    		}

    		textureProperties.__version = texture.version;

    		if ( texture.onUpdate ) texture.onUpdate( texture );

    	}

    	// Render targets

    	// Setup storage for target texture and bind it to correct framebuffer
    	function setupFrameBufferTexture( framebuffer, renderTarget, attachment, textureTarget ) {

    		var glFormat = utils.convert( renderTarget.texture.format );
    		var glType = utils.convert( renderTarget.texture.type );
    		var glInternalFormat = getInternalFormat( glFormat, glType );
    		state.texImage2D( textureTarget, 0, glInternalFormat, renderTarget.width, renderTarget.height, 0, glFormat, glType, null );
    		_gl.bindFramebuffer( 36160, framebuffer );
    		_gl.framebufferTexture2D( 36160, attachment, textureTarget, properties.get( renderTarget.texture ).__webglTexture, 0 );
    		_gl.bindFramebuffer( 36160, null );

    	}

    	// Setup storage for internal depth/stencil buffers and bind to correct framebuffer
    	function setupRenderBufferStorage( renderbuffer, renderTarget, isMultisample ) {

    		_gl.bindRenderbuffer( 36161, renderbuffer );

    		if ( renderTarget.depthBuffer && ! renderTarget.stencilBuffer ) {

    			if ( isMultisample ) {

    				var samples = getRenderTargetSamples( renderTarget );

    				_gl.renderbufferStorageMultisample( 36161, samples, 33189, renderTarget.width, renderTarget.height );

    			} else {

    				_gl.renderbufferStorage( 36161, 33189, renderTarget.width, renderTarget.height );

    			}

    			_gl.framebufferRenderbuffer( 36160, 36096, 36161, renderbuffer );

    		} else if ( renderTarget.depthBuffer && renderTarget.stencilBuffer ) {

    			if ( isMultisample ) {

    				var samples = getRenderTargetSamples( renderTarget );

    				_gl.renderbufferStorageMultisample( 36161, samples, 35056, renderTarget.width, renderTarget.height );

    			} else {

    				_gl.renderbufferStorage( 36161, 34041, renderTarget.width, renderTarget.height );

    			}


    			_gl.framebufferRenderbuffer( 36160, 33306, 36161, renderbuffer );

    		} else {

    			var glFormat = utils.convert( renderTarget.texture.format );
    			var glType = utils.convert( renderTarget.texture.type );
    			var glInternalFormat = getInternalFormat( glFormat, glType );

    			if ( isMultisample ) {

    				var samples = getRenderTargetSamples( renderTarget );

    				_gl.renderbufferStorageMultisample( 36161, samples, glInternalFormat, renderTarget.width, renderTarget.height );

    			} else {

    				_gl.renderbufferStorage( 36161, glInternalFormat, renderTarget.width, renderTarget.height );

    			}

    		}

    		_gl.bindRenderbuffer( 36161, null );

    	}

    	// Setup resources for a Depth Texture for a FBO (needs an extension)
    	function setupDepthTexture( framebuffer, renderTarget ) {

    		var isCube = ( renderTarget && renderTarget.isWebGLRenderTargetCube );
    		if ( isCube ) throw new Error( 'Depth Texture with cube render targets is not supported' );

    		_gl.bindFramebuffer( 36160, framebuffer );

    		if ( ! ( renderTarget.depthTexture && renderTarget.depthTexture.isDepthTexture ) ) {

    			throw new Error( 'renderTarget.depthTexture must be an instance of THREE.DepthTexture' );

    		}

    		// upload an empty depth texture with framebuffer size
    		if ( ! properties.get( renderTarget.depthTexture ).__webglTexture ||
    				renderTarget.depthTexture.image.width !== renderTarget.width ||
    				renderTarget.depthTexture.image.height !== renderTarget.height ) {

    			renderTarget.depthTexture.image.width = renderTarget.width;
    			renderTarget.depthTexture.image.height = renderTarget.height;
    			renderTarget.depthTexture.needsUpdate = true;

    		}

    		setTexture2D( renderTarget.depthTexture, 0 );

    		var webglDepthTexture = properties.get( renderTarget.depthTexture ).__webglTexture;

    		if ( renderTarget.depthTexture.format === DepthFormat ) {

    			_gl.framebufferTexture2D( 36160, 36096, 3553, webglDepthTexture, 0 );

    		} else if ( renderTarget.depthTexture.format === DepthStencilFormat ) {

    			_gl.framebufferTexture2D( 36160, 33306, 3553, webglDepthTexture, 0 );

    		} else {

    			throw new Error( 'Unknown depthTexture format' );

    		}

    	}

    	// Setup GL resources for a non-texture depth buffer
    	function setupDepthRenderbuffer( renderTarget ) {

    		var renderTargetProperties = properties.get( renderTarget );

    		var isCube = ( renderTarget.isWebGLRenderTargetCube === true );

    		if ( renderTarget.depthTexture ) {

    			if ( isCube ) throw new Error( 'target.depthTexture not supported in Cube render targets' );

    			setupDepthTexture( renderTargetProperties.__webglFramebuffer, renderTarget );

    		} else {

    			if ( isCube ) {

    				renderTargetProperties.__webglDepthbuffer = [];

    				for ( var i = 0; i < 6; i ++ ) {

    					_gl.bindFramebuffer( 36160, renderTargetProperties.__webglFramebuffer[ i ] );
    					renderTargetProperties.__webglDepthbuffer[ i ] = _gl.createRenderbuffer();
    					setupRenderBufferStorage( renderTargetProperties.__webglDepthbuffer[ i ], renderTarget );

    				}

    			} else {

    				_gl.bindFramebuffer( 36160, renderTargetProperties.__webglFramebuffer );
    				renderTargetProperties.__webglDepthbuffer = _gl.createRenderbuffer();
    				setupRenderBufferStorage( renderTargetProperties.__webglDepthbuffer, renderTarget );

    			}

    		}

    		_gl.bindFramebuffer( 36160, null );

    	}

    	// Set up GL resources for the render target
    	function setupRenderTarget( renderTarget ) {

    		var renderTargetProperties = properties.get( renderTarget );
    		var textureProperties = properties.get( renderTarget.texture );

    		renderTarget.addEventListener( 'dispose', onRenderTargetDispose );

    		textureProperties.__webglTexture = _gl.createTexture();

    		info.memory.textures ++;

    		var isCube = ( renderTarget.isWebGLRenderTargetCube === true );
    		var isMultisample = ( renderTarget.isWebGLMultisampleRenderTarget === true );
    		var supportsMips = isPowerOfTwo( renderTarget ) || capabilities.isWebGL2;

    		// Setup framebuffer

    		if ( isCube ) {

    			renderTargetProperties.__webglFramebuffer = [];

    			for ( var i = 0; i < 6; i ++ ) {

    				renderTargetProperties.__webglFramebuffer[ i ] = _gl.createFramebuffer();

    			}

    		} else {

    			renderTargetProperties.__webglFramebuffer = _gl.createFramebuffer();

    			if ( isMultisample ) {

    				if ( capabilities.isWebGL2 ) {

    					renderTargetProperties.__webglMultisampledFramebuffer = _gl.createFramebuffer();
    					renderTargetProperties.__webglColorRenderbuffer = _gl.createRenderbuffer();

    					_gl.bindRenderbuffer( 36161, renderTargetProperties.__webglColorRenderbuffer );
    					var glFormat = utils.convert( renderTarget.texture.format );
    					var glType = utils.convert( renderTarget.texture.type );
    					var glInternalFormat = getInternalFormat( glFormat, glType );
    					var samples = getRenderTargetSamples( renderTarget );
    					_gl.renderbufferStorageMultisample( 36161, samples, glInternalFormat, renderTarget.width, renderTarget.height );

    					_gl.bindFramebuffer( 36160, renderTargetProperties.__webglMultisampledFramebuffer );
    					_gl.framebufferRenderbuffer( 36160, 36064, 36161, renderTargetProperties.__webglColorRenderbuffer );
    					_gl.bindRenderbuffer( 36161, null );

    					if ( renderTarget.depthBuffer ) {

    						renderTargetProperties.__webglDepthRenderbuffer = _gl.createRenderbuffer();
    						setupRenderBufferStorage( renderTargetProperties.__webglDepthRenderbuffer, renderTarget, true );

    					}

    					_gl.bindFramebuffer( 36160, null );


    				} else {

    					console.warn( 'THREE.WebGLRenderer: WebGLMultisampleRenderTarget can only be used with WebGL2.' );

    				}

    			}

    		}

    		// Setup color buffer

    		if ( isCube ) {

    			state.bindTexture( 34067, textureProperties.__webglTexture );
    			setTextureParameters( 34067, renderTarget.texture, supportsMips );

    			for ( var i = 0; i < 6; i ++ ) {

    				setupFrameBufferTexture( renderTargetProperties.__webglFramebuffer[ i ], renderTarget, 36064, 34069 + i );

    			}

    			if ( textureNeedsGenerateMipmaps( renderTarget.texture, supportsMips ) ) {

    				generateMipmap( 34067, renderTarget.texture, renderTarget.width, renderTarget.height );

    			}

    			state.bindTexture( 34067, null );

    		} else {

    			state.bindTexture( 3553, textureProperties.__webglTexture );
    			setTextureParameters( 3553, renderTarget.texture, supportsMips );
    			setupFrameBufferTexture( renderTargetProperties.__webglFramebuffer, renderTarget, 36064, 3553 );

    			if ( textureNeedsGenerateMipmaps( renderTarget.texture, supportsMips ) ) {

    				generateMipmap( 3553, renderTarget.texture, renderTarget.width, renderTarget.height );

    			}

    			state.bindTexture( 3553, null );

    		}

    		// Setup depth and stencil buffers

    		if ( renderTarget.depthBuffer ) {

    			setupDepthRenderbuffer( renderTarget );

    		}

    	}

    	function updateRenderTargetMipmap( renderTarget ) {

    		var texture = renderTarget.texture;
    		var supportsMips = isPowerOfTwo( renderTarget ) || capabilities.isWebGL2;

    		if ( textureNeedsGenerateMipmaps( texture, supportsMips ) ) {

    			var target = renderTarget.isWebGLRenderTargetCube ? 34067 : 3553;
    			var webglTexture = properties.get( texture ).__webglTexture;

    			state.bindTexture( target, webglTexture );
    			generateMipmap( target, texture, renderTarget.width, renderTarget.height );
    			state.bindTexture( target, null );

    		}

    	}

    	function updateMultisampleRenderTarget( renderTarget ) {

    		if ( renderTarget.isWebGLMultisampleRenderTarget ) {

    			if ( capabilities.isWebGL2 ) {

    				var renderTargetProperties = properties.get( renderTarget );

    				_gl.bindFramebuffer( 36008, renderTargetProperties.__webglMultisampledFramebuffer );
    				_gl.bindFramebuffer( 36009, renderTargetProperties.__webglFramebuffer );

    				var width = renderTarget.width;
    				var height = renderTarget.height;
    				var mask = 16384;

    				if ( renderTarget.depthBuffer ) mask |= 256;
    				if ( renderTarget.stencilBuffer ) mask |= 1024;

    				_gl.blitFramebuffer( 0, 0, width, height, 0, 0, width, height, mask, 9728 );

    			} else {

    				console.warn( 'THREE.WebGLRenderer: WebGLMultisampleRenderTarget can only be used with WebGL2.' );

    			}

    		}

    	}

    	function getRenderTargetSamples( renderTarget ) {

    		return ( capabilities.isWebGL2 && renderTarget.isWebGLMultisampleRenderTarget ) ?
    			Math.min( capabilities.maxSamples, renderTarget.samples ) : 0;

    	}

    	function updateVideoTexture( texture ) {

    		var frame = info.render.frame;

    		// Check the last frame we updated the VideoTexture

    		if ( _videoTextures.get( texture ) !== frame ) {

    			_videoTextures.set( texture, frame );
    			texture.update();

    		}

    	}

    	// backwards compatibility

    	var warnedTexture2D = false;
    	var warnedTextureCube = false;

    	function safeSetTexture2D( texture, slot ) {

    		if ( texture && texture.isWebGLRenderTarget ) {

    			if ( warnedTexture2D === false ) {

    				console.warn( "THREE.WebGLTextures.safeSetTexture2D: don't use render targets as textures. Use their .texture property instead." );
    				warnedTexture2D = true;

    			}

    			texture = texture.texture;

    		}

    		setTexture2D( texture, slot );

    	}

    	function safeSetTextureCube( texture, slot ) {

    		if ( texture && texture.isWebGLRenderTargetCube ) {

    			if ( warnedTextureCube === false ) {

    				console.warn( "THREE.WebGLTextures.safeSetTextureCube: don't use cube render targets as textures. Use their .texture property instead." );
    				warnedTextureCube = true;

    			}

    			texture = texture.texture;

    		}

    		// currently relying on the fact that WebGLRenderTargetCube.texture is a Texture and NOT a CubeTexture
    		// TODO: unify these code paths
    		if ( ( texture && texture.isCubeTexture ) ||
    			( Array.isArray( texture.image ) && texture.image.length === 6 ) ) {

    			// CompressedTexture can have Array in image :/

    			// this function alone should take care of cube textures
    			setTextureCube( texture, slot );

    		} else {

    			// assumed: texture property of THREE.WebGLRenderTargetCube
    			setTextureCubeDynamic( texture, slot );

    		}

    	}

    	//

    	this.allocateTextureUnit = allocateTextureUnit;
    	this.resetTextureUnits = resetTextureUnits;

    	this.setTexture2D = setTexture2D;
    	this.setTexture2DArray = setTexture2DArray;
    	this.setTexture3D = setTexture3D;
    	this.setTextureCube = setTextureCube;
    	this.setTextureCubeDynamic = setTextureCubeDynamic;
    	this.setupRenderTarget = setupRenderTarget;
    	this.updateRenderTargetMipmap = updateRenderTargetMipmap;
    	this.updateMultisampleRenderTarget = updateMultisampleRenderTarget;

    	this.safeSetTexture2D = safeSetTexture2D;
    	this.safeSetTextureCube = safeSetTextureCube;

    }

    /**
     * @author thespite / http://www.twitter.com/thespite
     */

    function WebGLUtils( gl, extensions, capabilities ) {

    	function convert( p ) {

    		var extension;

    		if ( p === RepeatWrapping ) return 10497;
    		if ( p === ClampToEdgeWrapping ) return 33071;
    		if ( p === MirroredRepeatWrapping ) return 33648;

    		if ( p === NearestFilter ) return 9728;
    		if ( p === NearestMipmapNearestFilter ) return 9984;
    		if ( p === NearestMipmapLinearFilter ) return 9986;

    		if ( p === LinearFilter ) return 9729;
    		if ( p === LinearMipmapNearestFilter ) return 9985;
    		if ( p === LinearMipmapLinearFilter ) return 9987;

    		if ( p === UnsignedByteType ) return 5121;
    		if ( p === UnsignedShort4444Type ) return 32819;
    		if ( p === UnsignedShort5551Type ) return 32820;
    		if ( p === UnsignedShort565Type ) return 33635;

    		if ( p === ByteType ) return 5120;
    		if ( p === ShortType ) return 5122;
    		if ( p === UnsignedShortType ) return 5123;
    		if ( p === IntType ) return 5124;
    		if ( p === UnsignedIntType ) return 5125;
    		if ( p === FloatType ) return 5126;

    		if ( p === HalfFloatType ) {

    			if ( capabilities.isWebGL2 ) return 5131;

    			extension = extensions.get( 'OES_texture_half_float' );

    			if ( extension !== null ) return extension.HALF_FLOAT_OES;

    		}

    		if ( p === AlphaFormat ) return 6406;
    		if ( p === RGBFormat ) return 6407;
    		if ( p === RGBAFormat ) return 6408;
    		if ( p === LuminanceFormat ) return 6409;
    		if ( p === LuminanceAlphaFormat ) return 6410;
    		if ( p === DepthFormat ) return 6402;
    		if ( p === DepthStencilFormat ) return 34041;
    		if ( p === RedFormat ) return 6403;

    		if ( p === AddEquation ) return 32774;
    		if ( p === SubtractEquation ) return 32778;
    		if ( p === ReverseSubtractEquation ) return 32779;

    		if ( p === ZeroFactor ) return 0;
    		if ( p === OneFactor ) return 1;
    		if ( p === SrcColorFactor ) return 768;
    		if ( p === OneMinusSrcColorFactor ) return 769;
    		if ( p === SrcAlphaFactor ) return 770;
    		if ( p === OneMinusSrcAlphaFactor ) return 771;
    		if ( p === DstAlphaFactor ) return 772;
    		if ( p === OneMinusDstAlphaFactor ) return 773;

    		if ( p === DstColorFactor ) return 774;
    		if ( p === OneMinusDstColorFactor ) return 775;
    		if ( p === SrcAlphaSaturateFactor ) return 776;

    		if ( p === RGB_S3TC_DXT1_Format || p === RGBA_S3TC_DXT1_Format ||
    			p === RGBA_S3TC_DXT3_Format || p === RGBA_S3TC_DXT5_Format ) {

    			extension = extensions.get( 'WEBGL_compressed_texture_s3tc' );

    			if ( extension !== null ) {

    				if ( p === RGB_S3TC_DXT1_Format ) return extension.COMPRESSED_RGB_S3TC_DXT1_EXT;
    				if ( p === RGBA_S3TC_DXT1_Format ) return extension.COMPRESSED_RGBA_S3TC_DXT1_EXT;
    				if ( p === RGBA_S3TC_DXT3_Format ) return extension.COMPRESSED_RGBA_S3TC_DXT3_EXT;
    				if ( p === RGBA_S3TC_DXT5_Format ) return extension.COMPRESSED_RGBA_S3TC_DXT5_EXT;

    			}

    		}

    		if ( p === RGB_PVRTC_4BPPV1_Format || p === RGB_PVRTC_2BPPV1_Format ||
    			p === RGBA_PVRTC_4BPPV1_Format || p === RGBA_PVRTC_2BPPV1_Format ) {

    			extension = extensions.get( 'WEBGL_compressed_texture_pvrtc' );

    			if ( extension !== null ) {

    				if ( p === RGB_PVRTC_4BPPV1_Format ) return extension.COMPRESSED_RGB_PVRTC_4BPPV1_IMG;
    				if ( p === RGB_PVRTC_2BPPV1_Format ) return extension.COMPRESSED_RGB_PVRTC_2BPPV1_IMG;
    				if ( p === RGBA_PVRTC_4BPPV1_Format ) return extension.COMPRESSED_RGBA_PVRTC_4BPPV1_IMG;
    				if ( p === RGBA_PVRTC_2BPPV1_Format ) return extension.COMPRESSED_RGBA_PVRTC_2BPPV1_IMG;

    			}

    		}

    		if ( p === RGB_ETC1_Format ) {

    			extension = extensions.get( 'WEBGL_compressed_texture_etc1' );

    			if ( extension !== null ) return extension.COMPRESSED_RGB_ETC1_WEBGL;

    		}

    		if ( p === RGBA_ASTC_4x4_Format || p === RGBA_ASTC_5x4_Format || p === RGBA_ASTC_5x5_Format ||
    			p === RGBA_ASTC_6x5_Format || p === RGBA_ASTC_6x6_Format || p === RGBA_ASTC_8x5_Format ||
    			p === RGBA_ASTC_8x6_Format || p === RGBA_ASTC_8x8_Format || p === RGBA_ASTC_10x5_Format ||
    			p === RGBA_ASTC_10x6_Format || p === RGBA_ASTC_10x8_Format || p === RGBA_ASTC_10x10_Format ||
    			p === RGBA_ASTC_12x10_Format || p === RGBA_ASTC_12x12_Format ) {

    			extension = extensions.get( 'WEBGL_compressed_texture_astc' );

    			if ( extension !== null ) {

    				return p;

    			}

    		}

    		if ( p === MinEquation || p === MaxEquation ) {

    			if ( capabilities.isWebGL2 ) {

    				if ( p === MinEquation ) return 32775;
    				if ( p === MaxEquation ) return 32776;

    			}

    			extension = extensions.get( 'EXT_blend_minmax' );

    			if ( extension !== null ) {

    				if ( p === MinEquation ) return extension.MIN_EXT;
    				if ( p === MaxEquation ) return extension.MAX_EXT;

    			}

    		}

    		if ( p === UnsignedInt248Type ) {

    			if ( capabilities.isWebGL2 ) return 34042;

    			extension = extensions.get( 'WEBGL_depth_texture' );

    			if ( extension !== null ) return extension.UNSIGNED_INT_24_8_WEBGL;

    		}

    		return 0;

    	}

    	return { convert: convert };

    }

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function Group() {

    	Object3D.call( this );

    	this.type = 'Group';

    }

    Group.prototype = Object.assign( Object.create( Object3D.prototype ), {

    	constructor: Group,

    	isGroup: true

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function ArrayCamera( array ) {

    	PerspectiveCamera.call( this );

    	this.cameras = array || [];

    }

    ArrayCamera.prototype = Object.assign( Object.create( PerspectiveCamera.prototype ), {

    	constructor: ArrayCamera,

    	isArrayCamera: true

    } );

    /**
     * @author jsantell / https://www.jsantell.com/
     * @author mrdoob / http://mrdoob.com/
     */

    var cameraLPos = new Vector3();
    var cameraRPos = new Vector3();

    /**
     * Assumes 2 cameras that are parallel and share an X-axis, and that
     * the cameras' projection and world matrices have already been set.
     * And that near and far planes are identical for both cameras.
     * Visualization of this technique: https://computergraphics.stackexchange.com/a/4765
     */
    function setProjectionFromUnion( camera, cameraL, cameraR ) {

    	cameraLPos.setFromMatrixPosition( cameraL.matrixWorld );
    	cameraRPos.setFromMatrixPosition( cameraR.matrixWorld );

    	var ipd = cameraLPos.distanceTo( cameraRPos );

    	var projL = cameraL.projectionMatrix.elements;
    	var projR = cameraR.projectionMatrix.elements;

    	// VR systems will have identical far and near planes, and
    	// most likely identical top and bottom frustum extents.
    	// Use the left camera for these values.
    	var near = projL[ 14 ] / ( projL[ 10 ] - 1 );
    	var far = projL[ 14 ] / ( projL[ 10 ] + 1 );
    	var topFov = ( projL[ 9 ] + 1 ) / projL[ 5 ];
    	var bottomFov = ( projL[ 9 ] - 1 ) / projL[ 5 ];

    	var leftFov = ( projL[ 8 ] - 1 ) / projL[ 0 ];
    	var rightFov = ( projR[ 8 ] + 1 ) / projR[ 0 ];
    	var left = near * leftFov;
    	var right = near * rightFov;

    	// Calculate the new camera's position offset from the
    	// left camera. xOffset should be roughly half `ipd`.
    	var zOffset = ipd / ( - leftFov + rightFov );
    	var xOffset = zOffset * - leftFov;

    	// TODO: Better way to apply this offset?
    	cameraL.matrixWorld.decompose( camera.position, camera.quaternion, camera.scale );
    	camera.translateX( xOffset );
    	camera.translateZ( zOffset );
    	camera.matrixWorld.compose( camera.position, camera.quaternion, camera.scale );
    	camera.matrixWorldInverse.getInverse( camera.matrixWorld );

    	// Find the union of the frustum values of the cameras and scale
    	// the values so that the near plane's position does not change in world space,
    	// although must now be relative to the new union camera.
    	var near2 = near + zOffset;
    	var far2 = far + zOffset;
    	var left2 = left - xOffset;
    	var right2 = right + ( ipd - xOffset );
    	var top2 = topFov * far / far2 * near2;
    	var bottom2 = bottomFov * far / far2 * near2;

    	camera.projectionMatrix.makePerspective( left2, right2, top2, bottom2, near2, far2 );

    }

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function WebVRManager( renderer ) {

    	var renderWidth, renderHeight;
    	var scope = this;

    	var device = null;
    	var frameData = null;

    	var poseTarget = null;

    	var controllers = [];
    	var standingMatrix = new Matrix4();
    	var standingMatrixInverse = new Matrix4();

    	var framebufferScaleFactor = 1.0;

    	var referenceSpaceType = 'local-floor';

    	if ( typeof window !== 'undefined' && 'VRFrameData' in window ) {

    		frameData = new window.VRFrameData();
    		window.addEventListener( 'vrdisplaypresentchange', onVRDisplayPresentChange, false );

    	}

    	var matrixWorldInverse = new Matrix4();
    	var tempQuaternion = new Quaternion();
    	var tempPosition = new Vector3();

    	var cameraL = new PerspectiveCamera();
    	cameraL.viewport = new Vector4();
    	cameraL.layers.enable( 1 );

    	var cameraR = new PerspectiveCamera();
    	cameraR.viewport = new Vector4();
    	cameraR.layers.enable( 2 );

    	var cameraVR = new ArrayCamera( [ cameraL, cameraR ] );
    	cameraVR.layers.enable( 1 );
    	cameraVR.layers.enable( 2 );

    	//

    	function isPresenting() {

    		return device !== null && device.isPresenting === true;

    	}

    	var currentSize = new Vector2(), currentPixelRatio;

    	function onVRDisplayPresentChange() {

    		if ( isPresenting() ) {

    			var eyeParameters = device.getEyeParameters( 'left' );
    			renderWidth = 2 * eyeParameters.renderWidth * framebufferScaleFactor;
    			renderHeight = eyeParameters.renderHeight * framebufferScaleFactor;

    			currentPixelRatio = renderer.getPixelRatio();
    			renderer.getSize( currentSize );

    			renderer.setDrawingBufferSize( renderWidth, renderHeight, 1 );

    			cameraL.viewport.set( 0, 0, renderWidth / 2, renderHeight );
    			cameraR.viewport.set( renderWidth / 2, 0, renderWidth / 2, renderHeight );

    			animation.start();

    			scope.dispatchEvent( { type: 'sessionstart' } );

    		} else {

    			if ( scope.enabled ) {

    				renderer.setDrawingBufferSize( currentSize.width, currentSize.height, currentPixelRatio );

    			}

    			animation.stop();

    			scope.dispatchEvent( { type: 'sessionend' } );

    		}

    	}

    	//

    	var triggers = [];

    	function findGamepad( id ) {

    		var gamepads = navigator.getGamepads && navigator.getGamepads();

    		for ( var i = 0, j = 0, l = gamepads.length; i < l; i ++ ) {

    			var gamepad = gamepads[ i ];

    			if ( gamepad && ( gamepad.id === 'Daydream Controller' ||
    				gamepad.id === 'Gear VR Controller' || gamepad.id === 'Oculus Go Controller' ||
    				gamepad.id === 'OpenVR Gamepad' || gamepad.id.startsWith( 'Oculus Touch' ) ||
    				gamepad.id.startsWith( 'HTC Vive Focus' ) ||
    				gamepad.id.startsWith( 'Spatial Controller' ) ) ) {

    				if ( j === id ) return gamepad;

    				j ++;

    			}

    		}

    	}

    	function updateControllers() {

    		for ( var i = 0; i < controllers.length; i ++ ) {

    			var controller = controllers[ i ];

    			var gamepad = findGamepad( i );

    			if ( gamepad !== undefined && gamepad.pose !== undefined ) {

    				if ( gamepad.pose === null ) return;

    				// Pose

    				var pose = gamepad.pose;

    				if ( pose.hasPosition === false ) controller.position.set( 0.2, - 0.6, - 0.05 );

    				if ( pose.position !== null ) controller.position.fromArray( pose.position );
    				if ( pose.orientation !== null ) controller.quaternion.fromArray( pose.orientation );
    				controller.matrix.compose( controller.position, controller.quaternion, controller.scale );
    				controller.matrix.premultiply( standingMatrix );
    				controller.matrix.decompose( controller.position, controller.quaternion, controller.scale );
    				controller.matrixWorldNeedsUpdate = true;
    				controller.visible = true;

    				// Trigger

    				var buttonId = gamepad.id === 'Daydream Controller' ? 0 : 1;

    				if ( triggers[ i ] === undefined ) triggers[ i ] = false;

    				if ( triggers[ i ] !== gamepad.buttons[ buttonId ].pressed ) {

    					triggers[ i ] = gamepad.buttons[ buttonId ].pressed;

    					if ( triggers[ i ] === true ) {

    						controller.dispatchEvent( { type: 'selectstart' } );

    					} else {

    						controller.dispatchEvent( { type: 'selectend' } );
    						controller.dispatchEvent( { type: 'select' } );

    					}

    				}

    			} else {

    				controller.visible = false;

    			}

    		}

    	}

    	function updateViewportFromBounds( viewport, bounds ) {

    		if ( bounds !== null && bounds.length === 4 ) {

    			viewport.set( bounds[ 0 ] * renderWidth, bounds[ 1 ] * renderHeight, bounds[ 2 ] * renderWidth, bounds[ 3 ] * renderHeight );

    		}

    	}

    	//

    	this.enabled = false;

    	this.getController = function ( id ) {

    		var controller = controllers[ id ];

    		if ( controller === undefined ) {

    			controller = new Group();
    			controller.matrixAutoUpdate = false;
    			controller.visible = false;

    			controllers[ id ] = controller;

    		}

    		return controller;

    	};

    	this.getDevice = function () {

    		return device;

    	};

    	this.setDevice = function ( value ) {

    		if ( value !== undefined ) device = value;

    		animation.setContext( value );

    	};

    	this.setFramebufferScaleFactor = function ( value ) {

    		framebufferScaleFactor = value;

    	};

    	this.setReferenceSpaceType = function ( value ) {

    		referenceSpaceType = value;

    	};

    	this.setPoseTarget = function ( object ) {

    		if ( object !== undefined ) poseTarget = object;

    	};

    	this.getCamera = function ( camera ) {

    		var userHeight = referenceSpaceType === 'local-floor' ? 1.6 : 0;

    		if ( isPresenting() === false ) {

    			camera.position.set( 0, userHeight, 0 );
    			camera.rotation.set( 0, 0, 0 );

    			return camera;

    		}

    		device.depthNear = camera.near;
    		device.depthFar = camera.far;

    		device.getFrameData( frameData );

    		//

    		if ( referenceSpaceType === 'local-floor' ) {

    			var stageParameters = device.stageParameters;

    			if ( stageParameters ) {

    				standingMatrix.fromArray( stageParameters.sittingToStandingTransform );

    			} else {

    				standingMatrix.makeTranslation( 0, userHeight, 0 );

    			}

    		}


    		var pose = frameData.pose;
    		var poseObject = poseTarget !== null ? poseTarget : camera;

    		// We want to manipulate poseObject by its position and quaternion components since users may rely on them.
    		poseObject.matrix.copy( standingMatrix );
    		poseObject.matrix.decompose( poseObject.position, poseObject.quaternion, poseObject.scale );

    		if ( pose.orientation !== null ) {

    			tempQuaternion.fromArray( pose.orientation );
    			poseObject.quaternion.multiply( tempQuaternion );

    		}

    		if ( pose.position !== null ) {

    			tempQuaternion.setFromRotationMatrix( standingMatrix );
    			tempPosition.fromArray( pose.position );
    			tempPosition.applyQuaternion( tempQuaternion );
    			poseObject.position.add( tempPosition );

    		}

    		poseObject.updateMatrixWorld();

    		//

    		cameraL.near = camera.near;
    		cameraR.near = camera.near;

    		cameraL.far = camera.far;
    		cameraR.far = camera.far;

    		cameraL.matrixWorldInverse.fromArray( frameData.leftViewMatrix );
    		cameraR.matrixWorldInverse.fromArray( frameData.rightViewMatrix );

    		// TODO (mrdoob) Double check this code

    		standingMatrixInverse.getInverse( standingMatrix );

    		if ( referenceSpaceType === 'local-floor' ) {

    			cameraL.matrixWorldInverse.multiply( standingMatrixInverse );
    			cameraR.matrixWorldInverse.multiply( standingMatrixInverse );

    		}

    		var parent = poseObject.parent;

    		if ( parent !== null ) {

    			matrixWorldInverse.getInverse( parent.matrixWorld );

    			cameraL.matrixWorldInverse.multiply( matrixWorldInverse );
    			cameraR.matrixWorldInverse.multiply( matrixWorldInverse );

    		}

    		// envMap and Mirror needs camera.matrixWorld

    		cameraL.matrixWorld.getInverse( cameraL.matrixWorldInverse );
    		cameraR.matrixWorld.getInverse( cameraR.matrixWorldInverse );

    		cameraL.projectionMatrix.fromArray( frameData.leftProjectionMatrix );
    		cameraR.projectionMatrix.fromArray( frameData.rightProjectionMatrix );

    		setProjectionFromUnion( cameraVR, cameraL, cameraR );

    		//

    		var layers = device.getLayers();

    		if ( layers.length ) {

    			var layer = layers[ 0 ];

    			updateViewportFromBounds( cameraL.viewport, layer.leftBounds );
    			updateViewportFromBounds( cameraR.viewport, layer.rightBounds );

    		}

    		updateControllers();

    		return cameraVR;

    	};

    	this.getStandingMatrix = function () {

    		return standingMatrix;

    	};

    	this.isPresenting = isPresenting;

    	// Animation Loop

    	var animation = new WebGLAnimation();

    	this.setAnimationLoop = function ( callback ) {

    		animation.setAnimationLoop( callback );

    		if ( isPresenting() ) animation.start();

    	};

    	this.submitFrame = function () {

    		if ( isPresenting() ) device.submitFrame();

    	};

    	this.dispose = function () {

    		if ( typeof window !== 'undefined' ) {

    			window.removeEventListener( 'vrdisplaypresentchange', onVRDisplayPresentChange );

    		}

    	};

    	// DEPRECATED

    	this.setFrameOfReferenceType = function () {

    		console.warn( 'THREE.WebVRManager: setFrameOfReferenceType() has been deprecated.' );

    	};

    }

    Object.assign( WebVRManager.prototype, EventDispatcher.prototype );

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function WebXRManager( renderer, gl ) {

    	var scope = this;

    	var session = null;

    	var referenceSpace = null;
    	var referenceSpaceType = 'local-floor';

    	var pose = null;

    	var controllers = [];
    	var inputSources = [];

    	function isPresenting() {

    		return session !== null && referenceSpace !== null;

    	}

    	//

    	var cameraL = new PerspectiveCamera();
    	cameraL.layers.enable( 1 );
    	cameraL.viewport = new Vector4();

    	var cameraR = new PerspectiveCamera();
    	cameraR.layers.enable( 2 );
    	cameraR.viewport = new Vector4();

    	var cameraVR = new ArrayCamera( [ cameraL, cameraR ] );
    	cameraVR.layers.enable( 1 );
    	cameraVR.layers.enable( 2 );

    	//

    	this.enabled = false;

    	this.getController = function ( id ) {

    		var controller = controllers[ id ];

    		if ( controller === undefined ) {

    			controller = new Group();
    			controller.matrixAutoUpdate = false;
    			controller.visible = false;

    			controllers[ id ] = controller;

    		}

    		return controller;

    	};

    	//

    	function onSessionEvent( event ) {

    		for ( var i = 0; i < controllers.length; i ++ ) {

    			if ( inputSources[ i ] === event.inputSource ) {

    				controllers[ i ].dispatchEvent( { type: event.type } );

    			}

    		}

    	}

    	function onSessionEnd() {

    		renderer.setFramebuffer( null );
    		renderer.setRenderTarget( renderer.getRenderTarget() ); // Hack #15830
    		animation.stop();

    		scope.dispatchEvent( { type: 'sessionend' } );

    	}

    	function onRequestReferenceSpace( value ) {

    		referenceSpace = value;

    		animation.setContext( session );
    		animation.start();

    		scope.dispatchEvent( { type: 'sessionstart' } );

    	}

    	this.setFramebufferScaleFactor = function ( value ) {

    	};

    	this.setReferenceSpaceType = function ( value ) {

    		referenceSpaceType = value;

    	};

    	this.getSession = function () {

    		return session;

    	};

    	this.setSession = function ( value ) {

    		session = value;

    		if ( session !== null ) {

    			session.addEventListener( 'select', onSessionEvent );
    			session.addEventListener( 'selectstart', onSessionEvent );
    			session.addEventListener( 'selectend', onSessionEvent );
    			session.addEventListener( 'end', onSessionEnd );

    			session.updateRenderState( { baseLayer: new XRWebGLLayer( session, gl ) } );

    			session.requestReferenceSpace( referenceSpaceType ).then( onRequestReferenceSpace );

    			//

    			inputSources = session.inputSources;

    			session.addEventListener( 'inputsourceschange', function () {

    				inputSources = session.inputSources;
    				console.log( inputSources );

    				for ( var i = 0; i < controllers.length; i ++ ) {

    					var controller = controllers[ i ];
    					controller.userData.inputSource = inputSources[ i ];

    				}

    			} );

    		}

    	};

    	function updateCamera( camera, parent ) {

    		if ( parent === null ) {

    			camera.matrixWorld.copy( camera.matrix );

    		} else {

    			camera.matrixWorld.multiplyMatrices( parent.matrixWorld, camera.matrix );

    		}

    		camera.matrixWorldInverse.getInverse( camera.matrixWorld );

    	}

    	this.getCamera = function ( camera ) {

    		if ( isPresenting() ) {

    			var parent = camera.parent;
    			var cameras = cameraVR.cameras;

    			updateCamera( cameraVR, parent );

    			for ( var i = 0; i < cameras.length; i ++ ) {

    				updateCamera( cameras[ i ], parent );

    			}

    			// update camera and its children

    			camera.matrixWorld.copy( cameraVR.matrixWorld );

    			var children = camera.children;

    			for ( var i = 0, l = children.length; i < l; i ++ ) {

    				children[ i ].updateMatrixWorld( true );

    			}

    			setProjectionFromUnion( cameraVR, cameraL, cameraR );

    			return cameraVR;

    		}

    		return camera;

    	};

    	this.isPresenting = isPresenting;

    	// Animation Loop

    	var onAnimationFrameCallback = null;

    	function onAnimationFrame( time, frame ) {

    		pose = frame.getViewerPose( referenceSpace );

    		if ( pose !== null ) {

    			var views = pose.views;
    			var baseLayer = session.renderState.baseLayer;

    			renderer.setFramebuffer( baseLayer.framebuffer );

    			for ( var i = 0; i < views.length; i ++ ) {

    				var view = views[ i ];
    				var viewport = baseLayer.getViewport( view );
    				var viewMatrix = view.transform.inverse.matrix;

    				var camera = cameraVR.cameras[ i ];
    				camera.matrix.fromArray( viewMatrix ).getInverse( camera.matrix );
    				camera.projectionMatrix.fromArray( view.projectionMatrix );
    				camera.viewport.set( viewport.x, viewport.y, viewport.width, viewport.height );

    				if ( i === 0 ) {

    					cameraVR.matrix.copy( camera.matrix );

    				}

    			}

    		}

    		//

    		for ( var i = 0; i < controllers.length; i ++ ) {

    			var controller = controllers[ i ];

    			var inputSource = inputSources[ i ];

    			if ( inputSource ) {

    				var inputPose = frame.getPose( inputSource.targetRaySpace, referenceSpace );

    				if ( inputPose !== null ) {

    					controller.matrix.fromArray( inputPose.transform.matrix );
    					controller.matrix.decompose( controller.position, controller.rotation, controller.scale );
    					controller.visible = true;

    					continue;

    				}

    			}

    			controller.visible = false;

    		}

    		if ( onAnimationFrameCallback ) onAnimationFrameCallback( time );

    	}

    	var animation = new WebGLAnimation();
    	animation.setAnimationLoop( onAnimationFrame );

    	this.setAnimationLoop = function ( callback ) {

    		onAnimationFrameCallback = callback;

    	};

    	this.dispose = function () {};

    	// DEPRECATED

    	this.getStandingMatrix = function () {

    		console.warn( 'THREE.WebXRManager: getStandingMatrix() is no longer needed.' );
    		return new Matrix4();

    	};

    	this.getDevice = function () {

    		console.warn( 'THREE.WebXRManager: getDevice() has been deprecated.' );

    	};

    	this.setDevice = function () {

    		console.warn( 'THREE.WebXRManager: setDevice() has been deprecated.' );

    	};

    	this.setFrameOfReferenceType = function () {

    		console.warn( 'THREE.WebXRManager: setFrameOfReferenceType() has been deprecated.' );

    	};

    	this.submitFrame = function () {};

    }

    Object.assign( WebXRManager.prototype, EventDispatcher.prototype );

    /**
     * @author supereggbert / http://www.paulbrunt.co.uk/
     * @author mrdoob / http://mrdoob.com/
     * @author alteredq / http://alteredqualia.com/
     * @author szimek / https://github.com/szimek/
     * @author tschw
     */

    function WebGLRenderer( parameters ) {

    	parameters = parameters || {};

    	var _canvas = parameters.canvas !== undefined ? parameters.canvas : document.createElementNS( 'http://www.w3.org/1999/xhtml', 'canvas' ),
    		_context = parameters.context !== undefined ? parameters.context : null,

    		_alpha = parameters.alpha !== undefined ? parameters.alpha : false,
    		_depth = parameters.depth !== undefined ? parameters.depth : true,
    		_stencil = parameters.stencil !== undefined ? parameters.stencil : true,
    		_antialias = parameters.antialias !== undefined ? parameters.antialias : false,
    		_premultipliedAlpha = parameters.premultipliedAlpha !== undefined ? parameters.premultipliedAlpha : true,
    		_preserveDrawingBuffer = parameters.preserveDrawingBuffer !== undefined ? parameters.preserveDrawingBuffer : false,
    		_powerPreference = parameters.powerPreference !== undefined ? parameters.powerPreference : 'default',
    		_failIfMajorPerformanceCaveat = parameters.failIfMajorPerformanceCaveat !== undefined ? parameters.failIfMajorPerformanceCaveat : false;

    	var currentRenderList = null;
    	var currentRenderState = null;

    	// public properties

    	this.domElement = _canvas;

    	// Debug configuration container
    	this.debug = {

    		/**
    		 * Enables error checking and reporting when shader programs are being compiled
    		 * @type {boolean}
    		 */
    		checkShaderErrors: true
    	};

    	// clearing

    	this.autoClear = true;
    	this.autoClearColor = true;
    	this.autoClearDepth = true;
    	this.autoClearStencil = true;

    	// scene graph

    	this.sortObjects = true;

    	// user-defined clipping

    	this.clippingPlanes = [];
    	this.localClippingEnabled = false;

    	// physically based shading

    	this.gammaFactor = 2.0;	// for backwards compatibility
    	this.gammaInput = false;
    	this.gammaOutput = false;

    	// physical lights

    	this.physicallyCorrectLights = false;

    	// tone mapping

    	this.toneMapping = LinearToneMapping;
    	this.toneMappingExposure = 1.0;
    	this.toneMappingWhitePoint = 1.0;

    	// morphs

    	this.maxMorphTargets = 8;
    	this.maxMorphNormals = 4;

    	// internal properties

    	var _this = this,

    		_isContextLost = false,

    		// internal state cache

    		_framebuffer = null,

    		_currentActiveCubeFace = 0,
    		_currentActiveMipmapLevel = 0,
    		_currentRenderTarget = null,
    		_currentFramebuffer = null,
    		_currentMaterialId = - 1,

    		// geometry and program caching

    		_currentGeometryProgram = {
    			geometry: null,
    			program: null,
    			wireframe: false
    		},

    		_currentCamera = null,
    		_currentArrayCamera = null,

    		_currentViewport = new Vector4(),
    		_currentScissor = new Vector4(),
    		_currentScissorTest = null,

    		//

    		_width = _canvas.width,
    		_height = _canvas.height,

    		_pixelRatio = 1,

    		_viewport = new Vector4( 0, 0, _width, _height ),
    		_scissor = new Vector4( 0, 0, _width, _height ),
    		_scissorTest = false,

    		// frustum

    		_frustum = new Frustum(),

    		// clipping

    		_clipping = new WebGLClipping(),
    		_clippingEnabled = false,
    		_localClippingEnabled = false,

    		// camera matrices cache

    		_projScreenMatrix = new Matrix4(),

    		_vector3 = new Vector3();

    	function getTargetPixelRatio() {

    		return _currentRenderTarget === null ? _pixelRatio : 1;

    	}

    	// initialize

    	var _gl;

    	try {

    		var contextAttributes = {
    			alpha: _alpha,
    			depth: _depth,
    			stencil: _stencil,
    			antialias: _antialias,
    			premultipliedAlpha: _premultipliedAlpha,
    			preserveDrawingBuffer: _preserveDrawingBuffer,
    			powerPreference: _powerPreference,
    			failIfMajorPerformanceCaveat: _failIfMajorPerformanceCaveat,
    			xrCompatible: true
    		};

    		// event listeners must be registered before WebGL context is created, see #12753

    		_canvas.addEventListener( 'webglcontextlost', onContextLost, false );
    		_canvas.addEventListener( 'webglcontextrestored', onContextRestore, false );

    		_gl = _context || _canvas.getContext( 'webgl', contextAttributes ) || _canvas.getContext( 'experimental-webgl', contextAttributes );

    		if ( _gl === null ) {

    			if ( _canvas.getContext( 'webgl' ) !== null ) {

    				throw new Error( 'Error creating WebGL context with your selected attributes.' );

    			} else {

    				throw new Error( 'Error creating WebGL context.' );

    			}

    		}

    		// Some experimental-webgl implementations do not have getShaderPrecisionFormat

    		if ( _gl.getShaderPrecisionFormat === undefined ) {

    			_gl.getShaderPrecisionFormat = function () {

    				return { 'rangeMin': 1, 'rangeMax': 1, 'precision': 1 };

    			};

    		}

    	} catch ( error ) {

    		console.error( 'THREE.WebGLRenderer: ' + error.message );
    		throw error;

    	}

    	var extensions, capabilities, state, info;
    	var properties, textures, attributes, geometries, objects;
    	var programCache, renderLists, renderStates;

    	var background, morphtargets, bufferRenderer, indexedBufferRenderer;

    	var utils;

    	function initGLContext() {

    		extensions = new WebGLExtensions( _gl );

    		capabilities = new WebGLCapabilities( _gl, extensions, parameters );

    		if ( ! capabilities.isWebGL2 ) {

    			extensions.get( 'WEBGL_depth_texture' );
    			extensions.get( 'OES_texture_float' );
    			extensions.get( 'OES_texture_half_float' );
    			extensions.get( 'OES_texture_half_float_linear' );
    			extensions.get( 'OES_standard_derivatives' );
    			extensions.get( 'OES_element_index_uint' );
    			extensions.get( 'ANGLE_instanced_arrays' );

    		}

    		extensions.get( 'OES_texture_float_linear' );

    		utils = new WebGLUtils( _gl, extensions, capabilities );

    		state = new WebGLState( _gl, extensions, utils, capabilities );
    		state.scissor( _currentScissor.copy( _scissor ).multiplyScalar( _pixelRatio ).floor() );
    		state.viewport( _currentViewport.copy( _viewport ).multiplyScalar( _pixelRatio ).floor() );

    		info = new WebGLInfo( _gl );
    		properties = new WebGLProperties();
    		textures = new WebGLTextures( _gl, extensions, state, properties, capabilities, utils, info );
    		attributes = new WebGLAttributes( _gl );
    		geometries = new WebGLGeometries( _gl, attributes, info );
    		objects = new WebGLObjects( geometries, info );
    		morphtargets = new WebGLMorphtargets( _gl );
    		programCache = new WebGLPrograms( _this, extensions, capabilities );
    		renderLists = new WebGLRenderLists();
    		renderStates = new WebGLRenderStates();

    		background = new WebGLBackground( _this, state, objects, _premultipliedAlpha );

    		bufferRenderer = new WebGLBufferRenderer( _gl, extensions, info, capabilities );
    		indexedBufferRenderer = new WebGLIndexedBufferRenderer( _gl, extensions, info, capabilities );

    		info.programs = programCache.programs;

    		_this.capabilities = capabilities;
    		_this.extensions = extensions;
    		_this.properties = properties;
    		_this.renderLists = renderLists;
    		_this.state = state;
    		_this.info = info;

    	}

    	initGLContext();

    	// vr

    	var vr = ( typeof navigator !== 'undefined' && 'xr' in navigator && 'supportsSession' in navigator.xr ) ? new WebXRManager( _this, _gl ) : new WebVRManager( _this );

    	this.vr = vr;

    	// shadow map

    	var shadowMap = new WebGLShadowMap( _this, objects, capabilities.maxTextureSize );

    	this.shadowMap = shadowMap;

    	// API

    	this.getContext = function () {

    		return _gl;

    	};

    	this.getContextAttributes = function () {

    		return _gl.getContextAttributes();

    	};

    	this.forceContextLoss = function () {

    		var extension = extensions.get( 'WEBGL_lose_context' );
    		if ( extension ) extension.loseContext();

    	};

    	this.forceContextRestore = function () {

    		var extension = extensions.get( 'WEBGL_lose_context' );
    		if ( extension ) extension.restoreContext();

    	};

    	this.getPixelRatio = function () {

    		return _pixelRatio;

    	};

    	this.setPixelRatio = function ( value ) {

    		if ( value === undefined ) return;

    		_pixelRatio = value;

    		this.setSize( _width, _height, false );

    	};

    	this.getSize = function ( target ) {

    		if ( target === undefined ) {

    			console.warn( 'WebGLRenderer: .getsize() now requires a Vector2 as an argument' );

    			target = new Vector2();

    		}

    		return target.set( _width, _height );

    	};

    	this.setSize = function ( width, height, updateStyle ) {

    		if ( vr.isPresenting() ) {

    			console.warn( 'THREE.WebGLRenderer: Can\'t change size while VR device is presenting.' );
    			return;

    		}

    		_width = width;
    		_height = height;

    		_canvas.width = Math.floor( width * _pixelRatio );
    		_canvas.height = Math.floor( height * _pixelRatio );

    		if ( updateStyle !== false ) {

    			_canvas.style.width = width + 'px';
    			_canvas.style.height = height + 'px';

    		}

    		this.setViewport( 0, 0, width, height );

    	};

    	this.getDrawingBufferSize = function ( target ) {

    		if ( target === undefined ) {

    			console.warn( 'WebGLRenderer: .getdrawingBufferSize() now requires a Vector2 as an argument' );

    			target = new Vector2();

    		}

    		return target.set( _width * _pixelRatio, _height * _pixelRatio ).floor();

    	};

    	this.setDrawingBufferSize = function ( width, height, pixelRatio ) {

    		_width = width;
    		_height = height;

    		_pixelRatio = pixelRatio;

    		_canvas.width = Math.floor( width * pixelRatio );
    		_canvas.height = Math.floor( height * pixelRatio );

    		this.setViewport( 0, 0, width, height );

    	};

    	this.getCurrentViewport = function ( target ) {

    		if ( target === undefined ) {

    			console.warn( 'WebGLRenderer: .getCurrentViewport() now requires a Vector4 as an argument' );

    			target = new Vector4();

    		}

    		return target.copy( _currentViewport );

    	};

    	this.getViewport = function ( target ) {

    		return target.copy( _viewport );

    	};

    	this.setViewport = function ( x, y, width, height ) {

    		if ( x.isVector4 ) {

    			_viewport.set( x.x, x.y, x.z, x.w );

    		} else {

    			_viewport.set( x, y, width, height );

    		}

    		state.viewport( _currentViewport.copy( _viewport ).multiplyScalar( _pixelRatio ).floor() );

    	};

    	this.getScissor = function ( target ) {

    		return target.copy( _scissor );

    	};

    	this.setScissor = function ( x, y, width, height ) {

    		if ( x.isVector4 ) {

    			_scissor.set( x.x, x.y, x.z, x.w );

    		} else {

    			_scissor.set( x, y, width, height );

    		}

    		state.scissor( _currentScissor.copy( _scissor ).multiplyScalar( _pixelRatio ).floor() );

    	};

    	this.getScissorTest = function () {

    		return _scissorTest;

    	};

    	this.setScissorTest = function ( boolean ) {

    		state.setScissorTest( _scissorTest = boolean );

    	};

    	// Clearing

    	this.getClearColor = function () {

    		return background.getClearColor();

    	};

    	this.setClearColor = function () {

    		background.setClearColor.apply( background, arguments );

    	};

    	this.getClearAlpha = function () {

    		return background.getClearAlpha();

    	};

    	this.setClearAlpha = function () {

    		background.setClearAlpha.apply( background, arguments );

    	};

    	this.clear = function ( color, depth, stencil ) {

    		var bits = 0;

    		if ( color === undefined || color ) bits |= 16384;
    		if ( depth === undefined || depth ) bits |= 256;
    		if ( stencil === undefined || stencil ) bits |= 1024;

    		_gl.clear( bits );

    	};

    	this.clearColor = function () {

    		this.clear( true, false, false );

    	};

    	this.clearDepth = function () {

    		this.clear( false, true, false );

    	};

    	this.clearStencil = function () {

    		this.clear( false, false, true );

    	};

    	//

    	this.dispose = function () {

    		_canvas.removeEventListener( 'webglcontextlost', onContextLost, false );
    		_canvas.removeEventListener( 'webglcontextrestored', onContextRestore, false );

    		renderLists.dispose();
    		renderStates.dispose();
    		properties.dispose();
    		objects.dispose();

    		vr.dispose();

    		animation.stop();

    	};

    	// Events

    	function onContextLost( event ) {

    		event.preventDefault();

    		console.log( 'THREE.WebGLRenderer: Context Lost.' );

    		_isContextLost = true;

    	}

    	function onContextRestore( /* event */ ) {

    		console.log( 'THREE.WebGLRenderer: Context Restored.' );

    		_isContextLost = false;

    		initGLContext();

    	}

    	function onMaterialDispose( event ) {

    		var material = event.target;

    		material.removeEventListener( 'dispose', onMaterialDispose );

    		deallocateMaterial( material );

    	}

    	// Buffer deallocation

    	function deallocateMaterial( material ) {

    		releaseMaterialProgramReference( material );

    		properties.remove( material );

    	}


    	function releaseMaterialProgramReference( material ) {

    		var programInfo = properties.get( material ).program;

    		material.program = undefined;

    		if ( programInfo !== undefined ) {

    			programCache.releaseProgram( programInfo );

    		}

    	}

    	// Buffer rendering

    	function renderObjectImmediate( object, program ) {

    		object.render( function ( object ) {

    			_this.renderBufferImmediate( object, program );

    		} );

    	}

    	this.renderBufferImmediate = function ( object, program ) {

    		state.initAttributes();

    		var buffers = properties.get( object );

    		if ( object.hasPositions && ! buffers.position ) buffers.position = _gl.createBuffer();
    		if ( object.hasNormals && ! buffers.normal ) buffers.normal = _gl.createBuffer();
    		if ( object.hasUvs && ! buffers.uv ) buffers.uv = _gl.createBuffer();
    		if ( object.hasColors && ! buffers.color ) buffers.color = _gl.createBuffer();

    		var programAttributes = program.getAttributes();

    		if ( object.hasPositions ) {

    			_gl.bindBuffer( 34962, buffers.position );
    			_gl.bufferData( 34962, object.positionArray, 35048 );

    			state.enableAttribute( programAttributes.position );
    			_gl.vertexAttribPointer( programAttributes.position, 3, 5126, false, 0, 0 );

    		}

    		if ( object.hasNormals ) {

    			_gl.bindBuffer( 34962, buffers.normal );
    			_gl.bufferData( 34962, object.normalArray, 35048 );

    			state.enableAttribute( programAttributes.normal );
    			_gl.vertexAttribPointer( programAttributes.normal, 3, 5126, false, 0, 0 );

    		}

    		if ( object.hasUvs ) {

    			_gl.bindBuffer( 34962, buffers.uv );
    			_gl.bufferData( 34962, object.uvArray, 35048 );

    			state.enableAttribute( programAttributes.uv );
    			_gl.vertexAttribPointer( programAttributes.uv, 2, 5126, false, 0, 0 );

    		}

    		if ( object.hasColors ) {

    			_gl.bindBuffer( 34962, buffers.color );
    			_gl.bufferData( 34962, object.colorArray, 35048 );

    			state.enableAttribute( programAttributes.color );
    			_gl.vertexAttribPointer( programAttributes.color, 3, 5126, false, 0, 0 );

    		}

    		state.disableUnusedAttributes();

    		_gl.drawArrays( 4, 0, object.count );

    		object.count = 0;

    	};

    	this.renderBufferDirect = function ( camera, fog, geometry, material, object, group ) {

    		var frontFaceCW = ( object.isMesh && object.matrixWorld.determinant() < 0 );

    		state.setMaterial( material, frontFaceCW );

    		var program = setProgram( camera, fog, material, object );

    		var updateBuffers = false;

    		if ( _currentGeometryProgram.geometry !== geometry.id ||
    			_currentGeometryProgram.program !== program.id ||
    			_currentGeometryProgram.wireframe !== ( material.wireframe === true ) ) {

    			_currentGeometryProgram.geometry = geometry.id;
    			_currentGeometryProgram.program = program.id;
    			_currentGeometryProgram.wireframe = material.wireframe === true;
    			updateBuffers = true;

    		}

    		if ( object.morphTargetInfluences ) {

    			morphtargets.update( object, geometry, material, program );

    			updateBuffers = true;

    		}

    		//

    		var index = geometry.index;
    		var position = geometry.attributes.position;
    		var rangeFactor = 1;

    		if ( material.wireframe === true ) {

    			index = geometries.getWireframeAttribute( geometry );
    			rangeFactor = 2;

    		}

    		var attribute;
    		var renderer = bufferRenderer;

    		if ( index !== null ) {

    			attribute = attributes.get( index );

    			renderer = indexedBufferRenderer;
    			renderer.setIndex( attribute );

    		}

    		if ( updateBuffers ) {

    			setupVertexAttributes( material, program, geometry );

    			if ( index !== null ) {

    				_gl.bindBuffer( 34963, attribute.buffer );

    			}

    		}

    		//

    		var dataCount = Infinity;

    		if ( index !== null ) {

    			dataCount = index.count;

    		} else if ( position !== undefined ) {

    			dataCount = position.count;

    		}

    		var rangeStart = geometry.drawRange.start * rangeFactor;
    		var rangeCount = geometry.drawRange.count * rangeFactor;

    		var groupStart = group !== null ? group.start * rangeFactor : 0;
    		var groupCount = group !== null ? group.count * rangeFactor : Infinity;

    		var drawStart = Math.max( rangeStart, groupStart );
    		var drawEnd = Math.min( dataCount, rangeStart + rangeCount, groupStart + groupCount ) - 1;

    		var drawCount = Math.max( 0, drawEnd - drawStart + 1 );

    		if ( drawCount === 0 ) return;

    		//

    		if ( object.isMesh ) {

    			if ( material.wireframe === true ) {

    				state.setLineWidth( material.wireframeLinewidth * getTargetPixelRatio() );
    				renderer.setMode( 1 );

    			} else {

    				switch ( object.drawMode ) {

    					case TrianglesDrawMode:
    						renderer.setMode( 4 );
    						break;

    					case TriangleStripDrawMode:
    						renderer.setMode( 5 );
    						break;

    					case TriangleFanDrawMode:
    						renderer.setMode( 6 );
    						break;

    				}

    			}


    		} else if ( object.isLine ) {

    			var lineWidth = material.linewidth;

    			if ( lineWidth === undefined ) lineWidth = 1; // Not using Line*Material

    			state.setLineWidth( lineWidth * getTargetPixelRatio() );

    			if ( object.isLineSegments ) {

    				renderer.setMode( 1 );

    			} else if ( object.isLineLoop ) {

    				renderer.setMode( 2 );

    			} else {

    				renderer.setMode( 3 );

    			}

    		} else if ( object.isPoints ) {

    			renderer.setMode( 0 );

    		} else if ( object.isSprite ) {

    			renderer.setMode( 4 );

    		}

    		if ( geometry && geometry.isInstancedBufferGeometry ) {

    			if ( geometry.maxInstancedCount > 0 ) {

    				renderer.renderInstances( geometry, drawStart, drawCount );

    			}

    		} else {

    			renderer.render( drawStart, drawCount );

    		}

    	};

    	function setupVertexAttributes( material, program, geometry ) {

    		if ( geometry && geometry.isInstancedBufferGeometry && ! capabilities.isWebGL2 ) {

    			if ( extensions.get( 'ANGLE_instanced_arrays' ) === null ) {

    				console.error( 'THREE.WebGLRenderer.setupVertexAttributes: using THREE.InstancedBufferGeometry but hardware does not support extension ANGLE_instanced_arrays.' );
    				return;

    			}

    		}

    		state.initAttributes();

    		var geometryAttributes = geometry.attributes;

    		var programAttributes = program.getAttributes();

    		var materialDefaultAttributeValues = material.defaultAttributeValues;

    		for ( var name in programAttributes ) {

    			var programAttribute = programAttributes[ name ];

    			if ( programAttribute >= 0 ) {

    				var geometryAttribute = geometryAttributes[ name ];

    				if ( geometryAttribute !== undefined ) {

    					var normalized = geometryAttribute.normalized;
    					var size = geometryAttribute.itemSize;

    					var attribute = attributes.get( geometryAttribute );

    					// TODO Attribute may not be available on context restore

    					if ( attribute === undefined ) continue;

    					var buffer = attribute.buffer;
    					var type = attribute.type;
    					var bytesPerElement = attribute.bytesPerElement;

    					if ( geometryAttribute.isInterleavedBufferAttribute ) {

    						var data = geometryAttribute.data;
    						var stride = data.stride;
    						var offset = geometryAttribute.offset;

    						if ( data && data.isInstancedInterleavedBuffer ) {

    							state.enableAttributeAndDivisor( programAttribute, data.meshPerAttribute );

    							if ( geometry.maxInstancedCount === undefined ) {

    								geometry.maxInstancedCount = data.meshPerAttribute * data.count;

    							}

    						} else {

    							state.enableAttribute( programAttribute );

    						}

    						_gl.bindBuffer( 34962, buffer );
    						_gl.vertexAttribPointer( programAttribute, size, type, normalized, stride * bytesPerElement, offset * bytesPerElement );

    					} else {

    						if ( geometryAttribute.isInstancedBufferAttribute ) {

    							state.enableAttributeAndDivisor( programAttribute, geometryAttribute.meshPerAttribute );

    							if ( geometry.maxInstancedCount === undefined ) {

    								geometry.maxInstancedCount = geometryAttribute.meshPerAttribute * geometryAttribute.count;

    							}

    						} else {

    							state.enableAttribute( programAttribute );

    						}

    						_gl.bindBuffer( 34962, buffer );
    						_gl.vertexAttribPointer( programAttribute, size, type, normalized, 0, 0 );

    					}

    				} else if ( materialDefaultAttributeValues !== undefined ) {

    					var value = materialDefaultAttributeValues[ name ];

    					if ( value !== undefined ) {

    						switch ( value.length ) {

    							case 2:
    								_gl.vertexAttrib2fv( programAttribute, value );
    								break;

    							case 3:
    								_gl.vertexAttrib3fv( programAttribute, value );
    								break;

    							case 4:
    								_gl.vertexAttrib4fv( programAttribute, value );
    								break;

    							default:
    								_gl.vertexAttrib1fv( programAttribute, value );

    						}

    					}

    				}

    			}

    		}

    		state.disableUnusedAttributes();

    	}

    	// Compile

    	this.compile = function ( scene, camera ) {

    		currentRenderState = renderStates.get( scene, camera );
    		currentRenderState.init();

    		scene.traverse( function ( object ) {

    			if ( object.isLight ) {

    				currentRenderState.pushLight( object );

    				if ( object.castShadow ) {

    					currentRenderState.pushShadow( object );

    				}

    			}

    		} );

    		currentRenderState.setupLights( camera );

    		scene.traverse( function ( object ) {

    			if ( object.material ) {

    				if ( Array.isArray( object.material ) ) {

    					for ( var i = 0; i < object.material.length; i ++ ) {

    						initMaterial( object.material[ i ], scene.fog, object );

    					}

    				} else {

    					initMaterial( object.material, scene.fog, object );

    				}

    			}

    		} );

    	};

    	// Animation Loop

    	var onAnimationFrameCallback = null;

    	function onAnimationFrame( time ) {

    		if ( vr.isPresenting() ) return;
    		if ( onAnimationFrameCallback ) onAnimationFrameCallback( time );

    	}

    	var animation = new WebGLAnimation();
    	animation.setAnimationLoop( onAnimationFrame );

    	if ( typeof window !== 'undefined' ) animation.setContext( window );

    	this.setAnimationLoop = function ( callback ) {

    		onAnimationFrameCallback = callback;
    		vr.setAnimationLoop( callback );

    		animation.start();

    	};

    	// Rendering

    	this.render = function ( scene, camera ) {

    		var renderTarget, forceClear;

    		if ( arguments[ 2 ] !== undefined ) {

    			console.warn( 'THREE.WebGLRenderer.render(): the renderTarget argument has been removed. Use .setRenderTarget() instead.' );
    			renderTarget = arguments[ 2 ];

    		}

    		if ( arguments[ 3 ] !== undefined ) {

    			console.warn( 'THREE.WebGLRenderer.render(): the forceClear argument has been removed. Use .clear() instead.' );
    			forceClear = arguments[ 3 ];

    		}

    		if ( ! ( camera && camera.isCamera ) ) {

    			console.error( 'THREE.WebGLRenderer.render: camera is not an instance of THREE.Camera.' );
    			return;

    		}

    		if ( _isContextLost ) return;

    		// reset caching for this frame

    		_currentGeometryProgram.geometry = null;
    		_currentGeometryProgram.program = null;
    		_currentGeometryProgram.wireframe = false;
    		_currentMaterialId = - 1;
    		_currentCamera = null;

    		// update scene graph

    		if ( scene.autoUpdate === true ) scene.updateMatrixWorld();

    		// update camera matrices and frustum

    		if ( camera.parent === null ) camera.updateMatrixWorld();

    		if ( vr.enabled ) {

    			camera = vr.getCamera( camera );

    		}

    		//

    		currentRenderState = renderStates.get( scene, camera );
    		currentRenderState.init();

    		scene.onBeforeRender( _this, scene, camera, renderTarget || _currentRenderTarget );

    		_projScreenMatrix.multiplyMatrices( camera.projectionMatrix, camera.matrixWorldInverse );
    		_frustum.setFromMatrix( _projScreenMatrix );

    		_localClippingEnabled = this.localClippingEnabled;
    		_clippingEnabled = _clipping.init( this.clippingPlanes, _localClippingEnabled, camera );

    		currentRenderList = renderLists.get( scene, camera );
    		currentRenderList.init();

    		projectObject( scene, camera, 0, _this.sortObjects );

    		if ( _this.sortObjects === true ) {

    			currentRenderList.sort();

    		}

    		//

    		if ( _clippingEnabled ) _clipping.beginShadows();

    		var shadowsArray = currentRenderState.state.shadowsArray;

    		shadowMap.render( shadowsArray, scene, camera );

    		currentRenderState.setupLights( camera );

    		if ( _clippingEnabled ) _clipping.endShadows();

    		//

    		if ( this.info.autoReset ) this.info.reset();

    		if ( renderTarget !== undefined ) {

    			this.setRenderTarget( renderTarget );

    		}

    		//

    		background.render( currentRenderList, scene, camera, forceClear );

    		// render scene

    		var opaqueObjects = currentRenderList.opaque;
    		var transparentObjects = currentRenderList.transparent;

    		if ( scene.overrideMaterial ) {

    			var overrideMaterial = scene.overrideMaterial;

    			if ( opaqueObjects.length ) renderObjects( opaqueObjects, scene, camera, overrideMaterial );
    			if ( transparentObjects.length ) renderObjects( transparentObjects, scene, camera, overrideMaterial );

    		} else {

    			// opaque pass (front-to-back order)

    			if ( opaqueObjects.length ) renderObjects( opaqueObjects, scene, camera );

    			// transparent pass (back-to-front order)

    			if ( transparentObjects.length ) renderObjects( transparentObjects, scene, camera );

    		}

    		//

    		scene.onAfterRender( _this, scene, camera );

    		//

    		if ( _currentRenderTarget !== null ) {

    			// Generate mipmap if we're using any kind of mipmap filtering

    			textures.updateRenderTargetMipmap( _currentRenderTarget );

    			// resolve multisample renderbuffers to a single-sample texture if necessary

    			textures.updateMultisampleRenderTarget( _currentRenderTarget );

    		}

    		// Ensure depth buffer writing is enabled so it can be cleared on next render

    		state.buffers.depth.setTest( true );
    		state.buffers.depth.setMask( true );
    		state.buffers.color.setMask( true );

    		state.setPolygonOffset( false );

    		if ( vr.enabled ) {

    			vr.submitFrame();

    		}

    		// _gl.finish();

    		currentRenderList = null;
    		currentRenderState = null;

    	};

    	function projectObject( object, camera, groupOrder, sortObjects ) {

    		if ( object.visible === false ) return;

    		var visible = object.layers.test( camera.layers );

    		if ( visible ) {

    			if ( object.isGroup ) {

    				groupOrder = object.renderOrder;

    			} else if ( object.isLOD ) {

    				if ( object.autoUpdate === true ) object.update( camera );

    			} else if ( object.isLight ) {

    				currentRenderState.pushLight( object );

    				if ( object.castShadow ) {

    					currentRenderState.pushShadow( object );

    				}

    			} else if ( object.isSprite ) {

    				if ( ! object.frustumCulled || _frustum.intersectsSprite( object ) ) {

    					if ( sortObjects ) {

    						_vector3.setFromMatrixPosition( object.matrixWorld )
    							.applyMatrix4( _projScreenMatrix );

    					}

    					var geometry = objects.update( object );
    					var material = object.material;

    					if ( material.visible ) {

    						currentRenderList.push( object, geometry, material, groupOrder, _vector3.z, null );

    					}

    				}

    			} else if ( object.isImmediateRenderObject ) {

    				if ( sortObjects ) {

    					_vector3.setFromMatrixPosition( object.matrixWorld )
    						.applyMatrix4( _projScreenMatrix );

    				}

    				currentRenderList.push( object, null, object.material, groupOrder, _vector3.z, null );

    			} else if ( object.isMesh || object.isLine || object.isPoints ) {

    				if ( object.isSkinnedMesh ) {

    					object.skeleton.update();

    				}

    				if ( ! object.frustumCulled || _frustum.intersectsObject( object ) ) {

    					if ( sortObjects ) {

    						_vector3.setFromMatrixPosition( object.matrixWorld )
    							.applyMatrix4( _projScreenMatrix );

    					}

    					var geometry = objects.update( object );
    					var material = object.material;

    					if ( Array.isArray( material ) ) {

    						var groups = geometry.groups;

    						for ( var i = 0, l = groups.length; i < l; i ++ ) {

    							var group = groups[ i ];
    							var groupMaterial = material[ group.materialIndex ];

    							if ( groupMaterial && groupMaterial.visible ) {

    								currentRenderList.push( object, geometry, groupMaterial, groupOrder, _vector3.z, group );

    							}

    						}

    					} else if ( material.visible ) {

    						currentRenderList.push( object, geometry, material, groupOrder, _vector3.z, null );

    					}

    				}

    			}

    		}

    		var children = object.children;

    		for ( var i = 0, l = children.length; i < l; i ++ ) {

    			projectObject( children[ i ], camera, groupOrder, sortObjects );

    		}

    	}

    	function renderObjects( renderList, scene, camera, overrideMaterial ) {

    		for ( var i = 0, l = renderList.length; i < l; i ++ ) {

    			var renderItem = renderList[ i ];

    			var object = renderItem.object;
    			var geometry = renderItem.geometry;
    			var material = overrideMaterial === undefined ? renderItem.material : overrideMaterial;
    			var group = renderItem.group;

    			if ( camera.isArrayCamera ) {

    				_currentArrayCamera = camera;

    				var cameras = camera.cameras;

    				for ( var j = 0, jl = cameras.length; j < jl; j ++ ) {

    					var camera2 = cameras[ j ];

    					if ( object.layers.test( camera2.layers ) ) {

    						state.viewport( _currentViewport.copy( camera2.viewport ) );

    						currentRenderState.setupLights( camera2 );

    						renderObject( object, scene, camera2, geometry, material, group );

    					}

    				}

    			} else {

    				_currentArrayCamera = null;

    				renderObject( object, scene, camera, geometry, material, group );

    			}

    		}

    	}

    	function renderObject( object, scene, camera, geometry, material, group ) {

    		object.onBeforeRender( _this, scene, camera, geometry, material, group );
    		currentRenderState = renderStates.get( scene, _currentArrayCamera || camera );

    		object.modelViewMatrix.multiplyMatrices( camera.matrixWorldInverse, object.matrixWorld );
    		object.normalMatrix.getNormalMatrix( object.modelViewMatrix );

    		if ( object.isImmediateRenderObject ) {

    			state.setMaterial( material );

    			var program = setProgram( camera, scene.fog, material, object );

    			_currentGeometryProgram.geometry = null;
    			_currentGeometryProgram.program = null;
    			_currentGeometryProgram.wireframe = false;

    			renderObjectImmediate( object, program );

    		} else {

    			_this.renderBufferDirect( camera, scene.fog, geometry, material, object, group );

    		}

    		object.onAfterRender( _this, scene, camera, geometry, material, group );
    		currentRenderState = renderStates.get( scene, _currentArrayCamera || camera );

    	}

    	function initMaterial( material, fog, object ) {

    		var materialProperties = properties.get( material );

    		var lights = currentRenderState.state.lights;
    		var shadowsArray = currentRenderState.state.shadowsArray;

    		var lightsStateVersion = lights.state.version;

    		var parameters = programCache.getParameters(
    			material, lights.state, shadowsArray, fog, _clipping.numPlanes, _clipping.numIntersection, object );

    		var code = programCache.getProgramCode( material, parameters );

    		var program = materialProperties.program;
    		var programChange = true;

    		if ( program === undefined ) {

    			// new material
    			material.addEventListener( 'dispose', onMaterialDispose );

    		} else if ( program.code !== code ) {

    			// changed glsl or parameters
    			releaseMaterialProgramReference( material );

    		} else if ( materialProperties.lightsStateVersion !== lightsStateVersion ) {

    			materialProperties.lightsStateVersion = lightsStateVersion;

    			programChange = false;

    		} else if ( parameters.shaderID !== undefined ) {

    			// same glsl and uniform list
    			return;

    		} else {

    			// only rebuild uniform list
    			programChange = false;

    		}

    		if ( programChange ) {

    			if ( parameters.shaderID ) {

    				var shader = ShaderLib[ parameters.shaderID ];

    				materialProperties.shader = {
    					name: material.type,
    					uniforms: cloneUniforms( shader.uniforms ),
    					vertexShader: shader.vertexShader,
    					fragmentShader: shader.fragmentShader
    				};

    			} else {

    				materialProperties.shader = {
    					name: material.type,
    					uniforms: material.uniforms,
    					vertexShader: material.vertexShader,
    					fragmentShader: material.fragmentShader
    				};

    			}

    			material.onBeforeCompile( materialProperties.shader, _this );

    			// Computing code again as onBeforeCompile may have changed the shaders
    			code = programCache.getProgramCode( material, parameters );

    			program = programCache.acquireProgram( material, materialProperties.shader, parameters, code );

    			materialProperties.program = program;
    			material.program = program;

    		}

    		var programAttributes = program.getAttributes();

    		if ( material.morphTargets ) {

    			material.numSupportedMorphTargets = 0;

    			for ( var i = 0; i < _this.maxMorphTargets; i ++ ) {

    				if ( programAttributes[ 'morphTarget' + i ] >= 0 ) {

    					material.numSupportedMorphTargets ++;

    				}

    			}

    		}

    		if ( material.morphNormals ) {

    			material.numSupportedMorphNormals = 0;

    			for ( var i = 0; i < _this.maxMorphNormals; i ++ ) {

    				if ( programAttributes[ 'morphNormal' + i ] >= 0 ) {

    					material.numSupportedMorphNormals ++;

    				}

    			}

    		}

    		var uniforms = materialProperties.shader.uniforms;

    		if ( ! material.isShaderMaterial &&
    			! material.isRawShaderMaterial ||
    			material.clipping === true ) {

    			materialProperties.numClippingPlanes = _clipping.numPlanes;
    			materialProperties.numIntersection = _clipping.numIntersection;
    			uniforms.clippingPlanes = _clipping.uniform;

    		}

    		materialProperties.fog = fog;

    		// store the light setup it was created for

    		materialProperties.lightsStateVersion = lightsStateVersion;

    		if ( material.lights ) {

    			// wire up the material to this renderer's lighting state

    			uniforms.ambientLightColor.value = lights.state.ambient;
    			uniforms.lightProbe.value = lights.state.probe;
    			uniforms.directionalLights.value = lights.state.directional;
    			uniforms.spotLights.value = lights.state.spot;
    			uniforms.rectAreaLights.value = lights.state.rectArea;
    			uniforms.pointLights.value = lights.state.point;
    			uniforms.hemisphereLights.value = lights.state.hemi;

    			uniforms.directionalShadowMap.value = lights.state.directionalShadowMap;
    			uniforms.directionalShadowMatrix.value = lights.state.directionalShadowMatrix;
    			uniforms.spotShadowMap.value = lights.state.spotShadowMap;
    			uniforms.spotShadowMatrix.value = lights.state.spotShadowMatrix;
    			uniforms.pointShadowMap.value = lights.state.pointShadowMap;
    			uniforms.pointShadowMatrix.value = lights.state.pointShadowMatrix;
    			// TODO (abelnation): add area lights shadow info to uniforms

    		}

    		var progUniforms = materialProperties.program.getUniforms(),
    			uniformsList =
    				WebGLUniforms.seqWithValue( progUniforms.seq, uniforms );

    		materialProperties.uniformsList = uniformsList;

    	}

    	function setProgram( camera, fog, material, object ) {

    		textures.resetTextureUnits();

    		var materialProperties = properties.get( material );
    		var lights = currentRenderState.state.lights;

    		if ( _clippingEnabled ) {

    			if ( _localClippingEnabled || camera !== _currentCamera ) {

    				var useCache =
    					camera === _currentCamera &&
    					material.id === _currentMaterialId;

    				// we might want to call this function with some ClippingGroup
    				// object instead of the material, once it becomes feasible
    				// (#8465, #8379)
    				_clipping.setState(
    					material.clippingPlanes, material.clipIntersection, material.clipShadows,
    					camera, materialProperties, useCache );

    			}

    		}

    		if ( material.needsUpdate === false ) {

    			if ( materialProperties.program === undefined ) {

    				material.needsUpdate = true;

    			} else if ( material.fog && materialProperties.fog !== fog ) {

    				material.needsUpdate = true;

    			} else if ( material.lights && materialProperties.lightsStateVersion !== lights.state.version ) {

    				material.needsUpdate = true;

    			} else if ( materialProperties.numClippingPlanes !== undefined &&
    				( materialProperties.numClippingPlanes !== _clipping.numPlanes ||
    				materialProperties.numIntersection !== _clipping.numIntersection ) ) {

    				material.needsUpdate = true;

    			}

    		}

    		if ( material.needsUpdate ) {

    			initMaterial( material, fog, object );
    			material.needsUpdate = false;

    		}

    		var refreshProgram = false;
    		var refreshMaterial = false;
    		var refreshLights = false;

    		var program = materialProperties.program,
    			p_uniforms = program.getUniforms(),
    			m_uniforms = materialProperties.shader.uniforms;

    		if ( state.useProgram( program.program ) ) {

    			refreshProgram = true;
    			refreshMaterial = true;
    			refreshLights = true;

    		}

    		if ( material.id !== _currentMaterialId ) {

    			_currentMaterialId = material.id;

    			refreshMaterial = true;

    		}

    		if ( refreshProgram || _currentCamera !== camera ) {

    			p_uniforms.setValue( _gl, 'projectionMatrix', camera.projectionMatrix );

    			if ( capabilities.logarithmicDepthBuffer ) {

    				p_uniforms.setValue( _gl, 'logDepthBufFC',
    					2.0 / ( Math.log( camera.far + 1.0 ) / Math.LN2 ) );

    			}

    			if ( _currentCamera !== camera ) {

    				_currentCamera = camera;

    				// lighting uniforms depend on the camera so enforce an update
    				// now, in case this material supports lights - or later, when
    				// the next material that does gets activated:

    				refreshMaterial = true;		// set to true on material change
    				refreshLights = true;		// remains set until update done

    			}

    			// load material specific uniforms
    			// (shader material also gets them for the sake of genericity)

    			if ( material.isShaderMaterial ||
    				material.isMeshPhongMaterial ||
    				material.isMeshStandardMaterial ||
    				material.envMap ) {

    				var uCamPos = p_uniforms.map.cameraPosition;

    				if ( uCamPos !== undefined ) {

    					uCamPos.setValue( _gl,
    						_vector3.setFromMatrixPosition( camera.matrixWorld ) );

    				}

    			}

    			if ( material.isMeshPhongMaterial ||
    				material.isMeshLambertMaterial ||
    				material.isMeshBasicMaterial ||
    				material.isMeshStandardMaterial ||
    				material.isShaderMaterial ||
    				material.skinning ) {

    				p_uniforms.setValue( _gl, 'viewMatrix', camera.matrixWorldInverse );

    			}

    		}

    		// skinning uniforms must be set even if material didn't change
    		// auto-setting of texture unit for bone texture must go before other textures
    		// not sure why, but otherwise weird things happen

    		if ( material.skinning ) {

    			p_uniforms.setOptional( _gl, object, 'bindMatrix' );
    			p_uniforms.setOptional( _gl, object, 'bindMatrixInverse' );

    			var skeleton = object.skeleton;

    			if ( skeleton ) {

    				var bones = skeleton.bones;

    				if ( capabilities.floatVertexTextures ) {

    					if ( skeleton.boneTexture === undefined ) {

    						// layout (1 matrix = 4 pixels)
    						//      RGBA RGBA RGBA RGBA (=> column1, column2, column3, column4)
    						//  with  8x8  pixel texture max   16 bones * 4 pixels =  (8 * 8)
    						//       16x16 pixel texture max   64 bones * 4 pixels = (16 * 16)
    						//       32x32 pixel texture max  256 bones * 4 pixels = (32 * 32)
    						//       64x64 pixel texture max 1024 bones * 4 pixels = (64 * 64)


    						var size = Math.sqrt( bones.length * 4 ); // 4 pixels needed for 1 matrix
    						size = _Math.ceilPowerOfTwo( size );
    						size = Math.max( size, 4 );

    						var boneMatrices = new Float32Array( size * size * 4 ); // 4 floats per RGBA pixel
    						boneMatrices.set( skeleton.boneMatrices ); // copy current values

    						var boneTexture = new DataTexture( boneMatrices, size, size, RGBAFormat, FloatType );
    						boneTexture.needsUpdate = true;

    						skeleton.boneMatrices = boneMatrices;
    						skeleton.boneTexture = boneTexture;
    						skeleton.boneTextureSize = size;

    					}

    					p_uniforms.setValue( _gl, 'boneTexture', skeleton.boneTexture, textures );
    					p_uniforms.setValue( _gl, 'boneTextureSize', skeleton.boneTextureSize );

    				} else {

    					p_uniforms.setOptional( _gl, skeleton, 'boneMatrices' );

    				}

    			}

    		}

    		if ( refreshMaterial ) {

    			p_uniforms.setValue( _gl, 'toneMappingExposure', _this.toneMappingExposure );
    			p_uniforms.setValue( _gl, 'toneMappingWhitePoint', _this.toneMappingWhitePoint );

    			if ( material.lights ) {

    				// the current material requires lighting info

    				// note: all lighting uniforms are always set correctly
    				// they simply reference the renderer's state for their
    				// values
    				//
    				// use the current material's .needsUpdate flags to set
    				// the GL state when required

    				markUniformsLightsNeedsUpdate( m_uniforms, refreshLights );

    			}

    			// refresh uniforms common to several materials

    			if ( fog && material.fog ) {

    				refreshUniformsFog( m_uniforms, fog );

    			}

    			if ( material.isMeshBasicMaterial ) {

    				refreshUniformsCommon( m_uniforms, material );

    			} else if ( material.isMeshLambertMaterial ) {

    				refreshUniformsCommon( m_uniforms, material );
    				refreshUniformsLambert( m_uniforms, material );

    			} else if ( material.isMeshPhongMaterial ) {

    				refreshUniformsCommon( m_uniforms, material );

    				if ( material.isMeshToonMaterial ) {

    					refreshUniformsToon( m_uniforms, material );

    				} else {

    					refreshUniformsPhong( m_uniforms, material );

    				}

    			} else if ( material.isMeshStandardMaterial ) {

    				refreshUniformsCommon( m_uniforms, material );

    				if ( material.isMeshPhysicalMaterial ) {

    					refreshUniformsPhysical( m_uniforms, material );

    				} else {

    					refreshUniformsStandard( m_uniforms, material );

    				}

    			} else if ( material.isMeshMatcapMaterial ) {

    				refreshUniformsCommon( m_uniforms, material );

    				refreshUniformsMatcap( m_uniforms, material );

    			} else if ( material.isMeshDepthMaterial ) {

    				refreshUniformsCommon( m_uniforms, material );
    				refreshUniformsDepth( m_uniforms, material );

    			} else if ( material.isMeshDistanceMaterial ) {

    				refreshUniformsCommon( m_uniforms, material );
    				refreshUniformsDistance( m_uniforms, material );

    			} else if ( material.isMeshNormalMaterial ) {

    				refreshUniformsCommon( m_uniforms, material );
    				refreshUniformsNormal( m_uniforms, material );

    			} else if ( material.isLineBasicMaterial ) {

    				refreshUniformsLine( m_uniforms, material );

    				if ( material.isLineDashedMaterial ) {

    					refreshUniformsDash( m_uniforms, material );

    				}

    			} else if ( material.isPointsMaterial ) {

    				refreshUniformsPoints( m_uniforms, material );

    			} else if ( material.isSpriteMaterial ) {

    				refreshUniformsSprites( m_uniforms, material );

    			} else if ( material.isShadowMaterial ) {

    				m_uniforms.color.value.copy( material.color );
    				m_uniforms.opacity.value = material.opacity;

    			}

    			// RectAreaLight Texture
    			// TODO (mrdoob): Find a nicer implementation

    			if ( m_uniforms.ltc_1 !== undefined ) m_uniforms.ltc_1.value = UniformsLib.LTC_1;
    			if ( m_uniforms.ltc_2 !== undefined ) m_uniforms.ltc_2.value = UniformsLib.LTC_2;

    			WebGLUniforms.upload( _gl, materialProperties.uniformsList, m_uniforms, textures );

    		}

    		if ( material.isShaderMaterial && material.uniformsNeedUpdate === true ) {

    			WebGLUniforms.upload( _gl, materialProperties.uniformsList, m_uniforms, textures );
    			material.uniformsNeedUpdate = false;

    		}

    		if ( material.isSpriteMaterial ) {

    			p_uniforms.setValue( _gl, 'center', object.center );

    		}

    		// common matrices

    		p_uniforms.setValue( _gl, 'modelViewMatrix', object.modelViewMatrix );
    		p_uniforms.setValue( _gl, 'normalMatrix', object.normalMatrix );
    		p_uniforms.setValue( _gl, 'modelMatrix', object.matrixWorld );

    		return program;

    	}

    	// Uniforms (refresh uniforms objects)

    	function refreshUniformsCommon( uniforms, material ) {

    		uniforms.opacity.value = material.opacity;

    		if ( material.color ) {

    			uniforms.diffuse.value.copy( material.color );

    		}

    		if ( material.emissive ) {

    			uniforms.emissive.value.copy( material.emissive ).multiplyScalar( material.emissiveIntensity );

    		}

    		if ( material.map ) {

    			uniforms.map.value = material.map;

    		}

    		if ( material.alphaMap ) {

    			uniforms.alphaMap.value = material.alphaMap;

    		}

    		if ( material.specularMap ) {

    			uniforms.specularMap.value = material.specularMap;

    		}

    		if ( material.envMap ) {

    			uniforms.envMap.value = material.envMap;

    			// don't flip CubeTexture envMaps, flip everything else:
    			//  WebGLRenderTargetCube will be flipped for backwards compatibility
    			//  WebGLRenderTargetCube.texture will be flipped because it's a Texture and NOT a CubeTexture
    			// this check must be handled differently, or removed entirely, if WebGLRenderTargetCube uses a CubeTexture in the future
    			uniforms.flipEnvMap.value = material.envMap.isCubeTexture ? - 1 : 1;

    			uniforms.reflectivity.value = material.reflectivity;
    			uniforms.refractionRatio.value = material.refractionRatio;

    			uniforms.maxMipLevel.value = properties.get( material.envMap ).__maxMipLevel;

    		}

    		if ( material.lightMap ) {

    			uniforms.lightMap.value = material.lightMap;
    			uniforms.lightMapIntensity.value = material.lightMapIntensity;

    		}

    		if ( material.aoMap ) {

    			uniforms.aoMap.value = material.aoMap;
    			uniforms.aoMapIntensity.value = material.aoMapIntensity;

    		}

    		// uv repeat and offset setting priorities
    		// 1. color map
    		// 2. specular map
    		// 3. normal map
    		// 4. bump map
    		// 5. alpha map
    		// 6. emissive map

    		var uvScaleMap;

    		if ( material.map ) {

    			uvScaleMap = material.map;

    		} else if ( material.specularMap ) {

    			uvScaleMap = material.specularMap;

    		} else if ( material.displacementMap ) {

    			uvScaleMap = material.displacementMap;

    		} else if ( material.normalMap ) {

    			uvScaleMap = material.normalMap;

    		} else if ( material.bumpMap ) {

    			uvScaleMap = material.bumpMap;

    		} else if ( material.roughnessMap ) {

    			uvScaleMap = material.roughnessMap;

    		} else if ( material.metalnessMap ) {

    			uvScaleMap = material.metalnessMap;

    		} else if ( material.alphaMap ) {

    			uvScaleMap = material.alphaMap;

    		} else if ( material.emissiveMap ) {

    			uvScaleMap = material.emissiveMap;

    		}

    		if ( uvScaleMap !== undefined ) {

    			// backwards compatibility
    			if ( uvScaleMap.isWebGLRenderTarget ) {

    				uvScaleMap = uvScaleMap.texture;

    			}

    			if ( uvScaleMap.matrixAutoUpdate === true ) {

    				uvScaleMap.updateMatrix();

    			}

    			uniforms.uvTransform.value.copy( uvScaleMap.matrix );

    		}

    	}

    	function refreshUniformsLine( uniforms, material ) {

    		uniforms.diffuse.value.copy( material.color );
    		uniforms.opacity.value = material.opacity;

    	}

    	function refreshUniformsDash( uniforms, material ) {

    		uniforms.dashSize.value = material.dashSize;
    		uniforms.totalSize.value = material.dashSize + material.gapSize;
    		uniforms.scale.value = material.scale;

    	}

    	function refreshUniformsPoints( uniforms, material ) {

    		uniforms.diffuse.value.copy( material.color );
    		uniforms.opacity.value = material.opacity;
    		uniforms.size.value = material.size * _pixelRatio;
    		uniforms.scale.value = _height * 0.5;

    		uniforms.map.value = material.map;

    		if ( material.map !== null ) {

    			if ( material.map.matrixAutoUpdate === true ) {

    				material.map.updateMatrix();

    			}

    			uniforms.uvTransform.value.copy( material.map.matrix );

    		}

    	}

    	function refreshUniformsSprites( uniforms, material ) {

    		uniforms.diffuse.value.copy( material.color );
    		uniforms.opacity.value = material.opacity;
    		uniforms.rotation.value = material.rotation;
    		uniforms.map.value = material.map;

    		if ( material.map !== null ) {

    			if ( material.map.matrixAutoUpdate === true ) {

    				material.map.updateMatrix();

    			}

    			uniforms.uvTransform.value.copy( material.map.matrix );

    		}

    	}

    	function refreshUniformsFog( uniforms, fog ) {

    		uniforms.fogColor.value.copy( fog.color );

    		if ( fog.isFog ) {

    			uniforms.fogNear.value = fog.near;
    			uniforms.fogFar.value = fog.far;

    		} else if ( fog.isFogExp2 ) {

    			uniforms.fogDensity.value = fog.density;

    		}

    	}

    	function refreshUniformsLambert( uniforms, material ) {

    		if ( material.emissiveMap ) {

    			uniforms.emissiveMap.value = material.emissiveMap;

    		}

    	}

    	function refreshUniformsPhong( uniforms, material ) {

    		uniforms.specular.value.copy( material.specular );
    		uniforms.shininess.value = Math.max( material.shininess, 1e-4 ); // to prevent pow( 0.0, 0.0 )

    		if ( material.emissiveMap ) {

    			uniforms.emissiveMap.value = material.emissiveMap;

    		}

    		if ( material.bumpMap ) {

    			uniforms.bumpMap.value = material.bumpMap;
    			uniforms.bumpScale.value = material.bumpScale;
    			if ( material.side === BackSide ) uniforms.bumpScale.value *= - 1;

    		}

    		if ( material.normalMap ) {

    			uniforms.normalMap.value = material.normalMap;
    			uniforms.normalScale.value.copy( material.normalScale );
    			if ( material.side === BackSide ) uniforms.normalScale.value.negate();

    		}

    		if ( material.displacementMap ) {

    			uniforms.displacementMap.value = material.displacementMap;
    			uniforms.displacementScale.value = material.displacementScale;
    			uniforms.displacementBias.value = material.displacementBias;

    		}

    	}

    	function refreshUniformsToon( uniforms, material ) {

    		refreshUniformsPhong( uniforms, material );

    		if ( material.gradientMap ) {

    			uniforms.gradientMap.value = material.gradientMap;

    		}

    	}

    	function refreshUniformsStandard( uniforms, material ) {

    		uniforms.roughness.value = material.roughness;
    		uniforms.metalness.value = material.metalness;

    		if ( material.roughnessMap ) {

    			uniforms.roughnessMap.value = material.roughnessMap;

    		}

    		if ( material.metalnessMap ) {

    			uniforms.metalnessMap.value = material.metalnessMap;

    		}

    		if ( material.emissiveMap ) {

    			uniforms.emissiveMap.value = material.emissiveMap;

    		}

    		if ( material.bumpMap ) {

    			uniforms.bumpMap.value = material.bumpMap;
    			uniforms.bumpScale.value = material.bumpScale;
    			if ( material.side === BackSide ) uniforms.bumpScale.value *= - 1;

    		}

    		if ( material.normalMap ) {

    			uniforms.normalMap.value = material.normalMap;
    			uniforms.normalScale.value.copy( material.normalScale );
    			if ( material.side === BackSide ) uniforms.normalScale.value.negate();

    		}

    		if ( material.displacementMap ) {

    			uniforms.displacementMap.value = material.displacementMap;
    			uniforms.displacementScale.value = material.displacementScale;
    			uniforms.displacementBias.value = material.displacementBias;

    		}

    		if ( material.envMap ) {

    			//uniforms.envMap.value = material.envMap; // part of uniforms common
    			uniforms.envMapIntensity.value = material.envMapIntensity;

    		}

    	}

    	function refreshUniformsPhysical( uniforms, material ) {

    		refreshUniformsStandard( uniforms, material );

    		uniforms.reflectivity.value = material.reflectivity; // also part of uniforms common

    		uniforms.clearcoat.value = material.clearcoat;
    		uniforms.clearcoatRoughness.value = material.clearcoatRoughness;
    		if ( material.sheen ) uniforms.sheen.value.copy( material.sheen );

    		if ( material.clearcoatNormalMap ) {

    			uniforms.clearcoatNormalScale.value.copy( material.clearcoatNormalScale );
    			uniforms.clearcoatNormalMap.value = material.clearcoatNormalMap;

    			if ( material.side === BackSide ) {

    				uniforms.clearcoatNormalScale.value.negate();

    			}

    		}

    		uniforms.transparency.value = material.transparency;

    	}

    	function refreshUniformsMatcap( uniforms, material ) {

    		if ( material.matcap ) {

    			uniforms.matcap.value = material.matcap;

    		}

    		if ( material.bumpMap ) {

    			uniforms.bumpMap.value = material.bumpMap;
    			uniforms.bumpScale.value = material.bumpScale;
    			if ( material.side === BackSide ) uniforms.bumpScale.value *= - 1;

    		}

    		if ( material.normalMap ) {

    			uniforms.normalMap.value = material.normalMap;
    			uniforms.normalScale.value.copy( material.normalScale );
    			if ( material.side === BackSide ) uniforms.normalScale.value.negate();

    		}

    		if ( material.displacementMap ) {

    			uniforms.displacementMap.value = material.displacementMap;
    			uniforms.displacementScale.value = material.displacementScale;
    			uniforms.displacementBias.value = material.displacementBias;

    		}

    	}

    	function refreshUniformsDepth( uniforms, material ) {

    		if ( material.displacementMap ) {

    			uniforms.displacementMap.value = material.displacementMap;
    			uniforms.displacementScale.value = material.displacementScale;
    			uniforms.displacementBias.value = material.displacementBias;

    		}

    	}

    	function refreshUniformsDistance( uniforms, material ) {

    		if ( material.displacementMap ) {

    			uniforms.displacementMap.value = material.displacementMap;
    			uniforms.displacementScale.value = material.displacementScale;
    			uniforms.displacementBias.value = material.displacementBias;

    		}

    		uniforms.referencePosition.value.copy( material.referencePosition );
    		uniforms.nearDistance.value = material.nearDistance;
    		uniforms.farDistance.value = material.farDistance;

    	}

    	function refreshUniformsNormal( uniforms, material ) {

    		if ( material.bumpMap ) {

    			uniforms.bumpMap.value = material.bumpMap;
    			uniforms.bumpScale.value = material.bumpScale;
    			if ( material.side === BackSide ) uniforms.bumpScale.value *= - 1;

    		}

    		if ( material.normalMap ) {

    			uniforms.normalMap.value = material.normalMap;
    			uniforms.normalScale.value.copy( material.normalScale );
    			if ( material.side === BackSide ) uniforms.normalScale.value.negate();

    		}

    		if ( material.displacementMap ) {

    			uniforms.displacementMap.value = material.displacementMap;
    			uniforms.displacementScale.value = material.displacementScale;
    			uniforms.displacementBias.value = material.displacementBias;

    		}

    	}

    	// If uniforms are marked as clean, they don't need to be loaded to the GPU.

    	function markUniformsLightsNeedsUpdate( uniforms, value ) {

    		uniforms.ambientLightColor.needsUpdate = value;
    		uniforms.lightProbe.needsUpdate = value;

    		uniforms.directionalLights.needsUpdate = value;
    		uniforms.pointLights.needsUpdate = value;
    		uniforms.spotLights.needsUpdate = value;
    		uniforms.rectAreaLights.needsUpdate = value;
    		uniforms.hemisphereLights.needsUpdate = value;

    	}

    	//
    	this.setFramebuffer = function ( value ) {

    		if ( _framebuffer !== value ) _gl.bindFramebuffer( 36160, value );

    		_framebuffer = value;

    	};

    	this.getActiveCubeFace = function () {

    		return _currentActiveCubeFace;

    	};

    	this.getActiveMipmapLevel = function () {

    		return _currentActiveMipmapLevel;

    	};

    	this.getRenderTarget = function () {

    		return _currentRenderTarget;

    	};

    	this.setRenderTarget = function ( renderTarget, activeCubeFace, activeMipmapLevel ) {

    		_currentRenderTarget = renderTarget;
    		_currentActiveCubeFace = activeCubeFace;
    		_currentActiveMipmapLevel = activeMipmapLevel;

    		if ( renderTarget && properties.get( renderTarget ).__webglFramebuffer === undefined ) {

    			textures.setupRenderTarget( renderTarget );

    		}

    		var framebuffer = _framebuffer;
    		var isCube = false;

    		if ( renderTarget ) {

    			var __webglFramebuffer = properties.get( renderTarget ).__webglFramebuffer;

    			if ( renderTarget.isWebGLRenderTargetCube ) {

    				framebuffer = __webglFramebuffer[ activeCubeFace || 0 ];
    				isCube = true;

    			} else if ( renderTarget.isWebGLMultisampleRenderTarget ) {

    				framebuffer = properties.get( renderTarget ).__webglMultisampledFramebuffer;

    			} else {

    				framebuffer = __webglFramebuffer;

    			}

    			_currentViewport.copy( renderTarget.viewport );
    			_currentScissor.copy( renderTarget.scissor );
    			_currentScissorTest = renderTarget.scissorTest;

    		} else {

    			_currentViewport.copy( _viewport ).multiplyScalar( _pixelRatio ).floor();
    			_currentScissor.copy( _scissor ).multiplyScalar( _pixelRatio ).floor();
    			_currentScissorTest = _scissorTest;

    		}

    		if ( _currentFramebuffer !== framebuffer ) {

    			_gl.bindFramebuffer( 36160, framebuffer );
    			_currentFramebuffer = framebuffer;

    		}

    		state.viewport( _currentViewport );
    		state.scissor( _currentScissor );
    		state.setScissorTest( _currentScissorTest );

    		if ( isCube ) {

    			var textureProperties = properties.get( renderTarget.texture );
    			_gl.framebufferTexture2D( 36160, 36064, 34069 + ( activeCubeFace || 0 ), textureProperties.__webglTexture, activeMipmapLevel || 0 );

    		}

    	};

    	this.readRenderTargetPixels = function ( renderTarget, x, y, width, height, buffer, activeCubeFaceIndex ) {

    		if ( ! ( renderTarget && renderTarget.isWebGLRenderTarget ) ) {

    			console.error( 'THREE.WebGLRenderer.readRenderTargetPixels: renderTarget is not THREE.WebGLRenderTarget.' );
    			return;

    		}

    		var framebuffer = properties.get( renderTarget ).__webglFramebuffer;

    		if ( renderTarget.isWebGLRenderTargetCube && activeCubeFaceIndex !== undefined ) {

    			framebuffer = framebuffer[ activeCubeFaceIndex ];

    		}

    		if ( framebuffer ) {

    			var restore = false;

    			if ( framebuffer !== _currentFramebuffer ) {

    				_gl.bindFramebuffer( 36160, framebuffer );

    				restore = true;

    			}

    			try {

    				var texture = renderTarget.texture;
    				var textureFormat = texture.format;
    				var textureType = texture.type;

    				if ( textureFormat !== RGBAFormat && utils.convert( textureFormat ) !== _gl.getParameter( 35739 ) ) {

    					console.error( 'THREE.WebGLRenderer.readRenderTargetPixels: renderTarget is not in RGBA or implementation defined format.' );
    					return;

    				}

    				if ( textureType !== UnsignedByteType && utils.convert( textureType ) !== _gl.getParameter( 35738 ) && // IE11, Edge and Chrome Mac < 52 (#9513)
    					! ( textureType === FloatType && ( capabilities.isWebGL2 || extensions.get( 'OES_texture_float' ) || extensions.get( 'WEBGL_color_buffer_float' ) ) ) && // Chrome Mac >= 52 and Firefox
    					! ( textureType === HalfFloatType && ( capabilities.isWebGL2 ? extensions.get( 'EXT_color_buffer_float' ) : extensions.get( 'EXT_color_buffer_half_float' ) ) ) ) {

    					console.error( 'THREE.WebGLRenderer.readRenderTargetPixels: renderTarget is not in UnsignedByteType or implementation defined type.' );
    					return;

    				}

    				if ( _gl.checkFramebufferStatus( 36160 ) === 36053 ) {

    					// the following if statement ensures valid read requests (no out-of-bounds pixels, see #8604)

    					if ( ( x >= 0 && x <= ( renderTarget.width - width ) ) && ( y >= 0 && y <= ( renderTarget.height - height ) ) ) {

    						_gl.readPixels( x, y, width, height, utils.convert( textureFormat ), utils.convert( textureType ), buffer );

    					}

    				} else {

    					console.error( 'THREE.WebGLRenderer.readRenderTargetPixels: readPixels from renderTarget failed. Framebuffer not complete.' );

    				}

    			} finally {

    				if ( restore ) {

    					_gl.bindFramebuffer( 36160, _currentFramebuffer );

    				}

    			}

    		}

    	};

    	this.copyFramebufferToTexture = function ( position, texture, level ) {

    		var width = texture.image.width;
    		var height = texture.image.height;
    		var glFormat = utils.convert( texture.format );

    		textures.setTexture2D( texture, 0 );

    		_gl.copyTexImage2D( 3553, level || 0, glFormat, position.x, position.y, width, height, 0 );

    	};

    	this.copyTextureToTexture = function ( position, srcTexture, dstTexture, level ) {

    		var width = srcTexture.image.width;
    		var height = srcTexture.image.height;
    		var glFormat = utils.convert( dstTexture.format );
    		var glType = utils.convert( dstTexture.type );

    		textures.setTexture2D( dstTexture, 0 );

    		if ( srcTexture.isDataTexture ) {

    			_gl.texSubImage2D( 3553, level || 0, position.x, position.y, width, height, glFormat, glType, srcTexture.image.data );

    		} else {

    			_gl.texSubImage2D( 3553, level || 0, position.x, position.y, glFormat, glType, srcTexture.image );

    		}

    	};

    	if ( typeof __THREE_DEVTOOLS__ !== 'undefined' ) {

    		__THREE_DEVTOOLS__.dispatchEvent( new CustomEvent( 'observe', { detail: this } ) ); // eslint-disable-line no-undef

    	}

    }

    /**
     * @author mrdoob / http://mrdoob.com/
     * @author alteredq / http://alteredqualia.com/
     */

    function FogExp2( color, density ) {

    	this.name = '';

    	this.color = new Color$7( color );
    	this.density = ( density !== undefined ) ? density : 0.00025;

    }

    Object.assign( FogExp2.prototype, {

    	isFogExp2: true,

    	clone: function () {

    		return new FogExp2( this.color, this.density );

    	},

    	toJSON: function ( /* meta */ ) {

    		return {
    			type: 'FogExp2',
    			color: this.color.getHex(),
    			density: this.density
    		};

    	}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     * @author alteredq / http://alteredqualia.com/
     */

    function Fog( color, near, far ) {

    	this.name = '';

    	this.color = new Color$7( color );

    	this.near = ( near !== undefined ) ? near : 1;
    	this.far = ( far !== undefined ) ? far : 1000;

    }

    Object.assign( Fog.prototype, {

    	isFog: true,

    	clone: function () {

    		return new Fog( this.color, this.near, this.far );

    	},

    	toJSON: function ( /* meta */ ) {

    		return {
    			type: 'Fog',
    			color: this.color.getHex(),
    			near: this.near,
    			far: this.far
    		};

    	}

    } );

    /**
     * @author benaadams / https://twitter.com/ben_a_adams
     */

    function InterleavedBuffer( array, stride ) {

    	this.array = array;
    	this.stride = stride;
    	this.count = array !== undefined ? array.length / stride : 0;

    	this.dynamic = false;
    	this.updateRange = { offset: 0, count: - 1 };

    	this.version = 0;

    }

    Object.defineProperty( InterleavedBuffer.prototype, 'needsUpdate', {

    	set: function ( value ) {

    		if ( value === true ) this.version ++;

    	}

    } );

    Object.assign( InterleavedBuffer.prototype, {

    	isInterleavedBuffer: true,

    	onUploadCallback: function () {},

    	setArray: function ( array ) {

    		if ( Array.isArray( array ) ) {

    			throw new TypeError( 'THREE.BufferAttribute: array should be a Typed Array.' );

    		}

    		this.count = array !== undefined ? array.length / this.stride : 0;
    		this.array = array;

    		return this;

    	},

    	setDynamic: function ( value ) {

    		this.dynamic = value;

    		return this;

    	},

    	copy: function ( source ) {

    		this.array = new source.array.constructor( source.array );
    		this.count = source.count;
    		this.stride = source.stride;
    		this.dynamic = source.dynamic;

    		return this;

    	},

    	copyAt: function ( index1, attribute, index2 ) {

    		index1 *= this.stride;
    		index2 *= attribute.stride;

    		for ( var i = 0, l = this.stride; i < l; i ++ ) {

    			this.array[ index1 + i ] = attribute.array[ index2 + i ];

    		}

    		return this;

    	},

    	set: function ( value, offset ) {

    		if ( offset === undefined ) offset = 0;

    		this.array.set( value, offset );

    		return this;

    	},

    	clone: function () {

    		return new this.constructor().copy( this );

    	},

    	onUpload: function ( callback ) {

    		this.onUploadCallback = callback;

    		return this;

    	}

    } );

    /**
     * @author benaadams / https://twitter.com/ben_a_adams
     */

    function InterleavedBufferAttribute( interleavedBuffer, itemSize, offset, normalized ) {

    	this.data = interleavedBuffer;
    	this.itemSize = itemSize;
    	this.offset = offset;

    	this.normalized = normalized === true;

    }

    Object.defineProperties( InterleavedBufferAttribute.prototype, {

    	count: {

    		get: function () {

    			return this.data.count;

    		}

    	},

    	array: {

    		get: function () {

    			return this.data.array;

    		}

    	}

    } );

    Object.assign( InterleavedBufferAttribute.prototype, {

    	isInterleavedBufferAttribute: true,

    	setX: function ( index, x ) {

    		this.data.array[ index * this.data.stride + this.offset ] = x;

    		return this;

    	},

    	setY: function ( index, y ) {

    		this.data.array[ index * this.data.stride + this.offset + 1 ] = y;

    		return this;

    	},

    	setZ: function ( index, z ) {

    		this.data.array[ index * this.data.stride + this.offset + 2 ] = z;

    		return this;

    	},

    	setW: function ( index, w ) {

    		this.data.array[ index * this.data.stride + this.offset + 3 ] = w;

    		return this;

    	},

    	getX: function ( index ) {

    		return this.data.array[ index * this.data.stride + this.offset ];

    	},

    	getY: function ( index ) {

    		return this.data.array[ index * this.data.stride + this.offset + 1 ];

    	},

    	getZ: function ( index ) {

    		return this.data.array[ index * this.data.stride + this.offset + 2 ];

    	},

    	getW: function ( index ) {

    		return this.data.array[ index * this.data.stride + this.offset + 3 ];

    	},

    	setXY: function ( index, x, y ) {

    		index = index * this.data.stride + this.offset;

    		this.data.array[ index + 0 ] = x;
    		this.data.array[ index + 1 ] = y;

    		return this;

    	},

    	setXYZ: function ( index, x, y, z ) {

    		index = index * this.data.stride + this.offset;

    		this.data.array[ index + 0 ] = x;
    		this.data.array[ index + 1 ] = y;
    		this.data.array[ index + 2 ] = z;

    		return this;

    	},

    	setXYZW: function ( index, x, y, z, w ) {

    		index = index * this.data.stride + this.offset;

    		this.data.array[ index + 0 ] = x;
    		this.data.array[ index + 1 ] = y;
    		this.data.array[ index + 2 ] = z;
    		this.data.array[ index + 3 ] = w;

    		return this;

    	}

    } );

    /**
     * @author alteredq / http://alteredqualia.com/
     *
     * parameters = {
     *  color: <hex>,
     *  map: new THREE.Texture( <Image> ),
     *  rotation: <float>,
     *  sizeAttenuation: <bool>
     * }
     */

    function SpriteMaterial( parameters ) {

    	Material.call( this );

    	this.type = 'SpriteMaterial';

    	this.color = new Color$7( 0xffffff );
    	this.map = null;

    	this.rotation = 0;

    	this.sizeAttenuation = true;

    	this.lights = false;
    	this.transparent = true;

    	this.setValues( parameters );

    }

    SpriteMaterial.prototype = Object.create( Material.prototype );
    SpriteMaterial.prototype.constructor = SpriteMaterial;
    SpriteMaterial.prototype.isSpriteMaterial = true;

    SpriteMaterial.prototype.copy = function ( source ) {

    	Material.prototype.copy.call( this, source );

    	this.color.copy( source.color );
    	this.map = source.map;

    	this.rotation = source.rotation;

    	this.sizeAttenuation = source.sizeAttenuation;

    	return this;

    };

    /**
     * @author mikael emtinger / http://gomo.se/
     * @author alteredq / http://alteredqualia.com/
     */

    var _geometry;

    var _intersectPoint = new Vector3();
    var _worldScale = new Vector3();
    var _mvPosition = new Vector3();

    var _alignedPosition = new Vector2();
    var _rotatedPosition = new Vector2();
    var _viewWorldMatrix = new Matrix4();

    var _vA$1 = new Vector3();
    var _vB$1 = new Vector3();
    var _vC$1 = new Vector3();

    var _uvA$1 = new Vector2();
    var _uvB$1 = new Vector2();
    var _uvC$1 = new Vector2();

    function Sprite( material ) {

    	Object3D.call( this );

    	this.type = 'Sprite';

    	if ( _geometry === undefined ) {

    		_geometry = new BufferGeometry();

    		var float32Array = new Float32Array( [
    			- 0.5, - 0.5, 0, 0, 0,
    			0.5, - 0.5, 0, 1, 0,
    			0.5, 0.5, 0, 1, 1,
    			- 0.5, 0.5, 0, 0, 1
    		] );

    		var interleavedBuffer = new InterleavedBuffer( float32Array, 5 );

    		_geometry.setIndex( [ 0, 1, 2,	0, 2, 3 ] );
    		_geometry.addAttribute( 'position', new InterleavedBufferAttribute( interleavedBuffer, 3, 0, false ) );
    		_geometry.addAttribute( 'uv', new InterleavedBufferAttribute( interleavedBuffer, 2, 3, false ) );

    	}

    	this.geometry = _geometry;
    	this.material = ( material !== undefined ) ? material : new SpriteMaterial();

    	this.center = new Vector2( 0.5, 0.5 );

    }

    Sprite.prototype = Object.assign( Object.create( Object3D.prototype ), {

    	constructor: Sprite,

    	isSprite: true,

    	raycast: function ( raycaster, intersects ) {

    		if ( raycaster.camera === null ) {

    			console.error( 'THREE.Sprite: "Raycaster.camera" needs to be set in order to raycast against sprites.' );

    		}

    		_worldScale.setFromMatrixScale( this.matrixWorld );

    		_viewWorldMatrix.copy( raycaster.camera.matrixWorld );
    		this.modelViewMatrix.multiplyMatrices( raycaster.camera.matrixWorldInverse, this.matrixWorld );

    		_mvPosition.setFromMatrixPosition( this.modelViewMatrix );

    		if ( raycaster.camera.isPerspectiveCamera && this.material.sizeAttenuation === false ) {

    			_worldScale.multiplyScalar( - _mvPosition.z );

    		}

    		var rotation = this.material.rotation;
    		var sin, cos;
    		if ( rotation !== 0 ) {

    			cos = Math.cos( rotation );
    			sin = Math.sin( rotation );

    		}

    		var center = this.center;

    		transformVertex( _vA$1.set( - 0.5, - 0.5, 0 ), _mvPosition, center, _worldScale, sin, cos );
    		transformVertex( _vB$1.set( 0.5, - 0.5, 0 ), _mvPosition, center, _worldScale, sin, cos );
    		transformVertex( _vC$1.set( 0.5, 0.5, 0 ), _mvPosition, center, _worldScale, sin, cos );

    		_uvA$1.set( 0, 0 );
    		_uvB$1.set( 1, 0 );
    		_uvC$1.set( 1, 1 );

    		// check first triangle
    		var intersect = raycaster.ray.intersectTriangle( _vA$1, _vB$1, _vC$1, false, _intersectPoint );

    		if ( intersect === null ) {

    			// check second triangle
    			transformVertex( _vB$1.set( - 0.5, 0.5, 0 ), _mvPosition, center, _worldScale, sin, cos );
    			_uvB$1.set( 0, 1 );

    			intersect = raycaster.ray.intersectTriangle( _vA$1, _vC$1, _vB$1, false, _intersectPoint );
    			if ( intersect === null ) {

    				return;

    			}

    		}

    		var distance = raycaster.ray.origin.distanceTo( _intersectPoint );

    		if ( distance < raycaster.near || distance > raycaster.far ) return;

    		intersects.push( {

    			distance: distance,
    			point: _intersectPoint.clone(),
    			uv: Triangle.getUV( _intersectPoint, _vA$1, _vB$1, _vC$1, _uvA$1, _uvB$1, _uvC$1, new Vector2() ),
    			face: null,
    			object: this

    		} );

    	},

    	clone: function () {

    		return new this.constructor( this.material ).copy( this );

    	},

    	copy: function ( source ) {

    		Object3D.prototype.copy.call( this, source );

    		if ( source.center !== undefined ) this.center.copy( source.center );

    		return this;

    	}


    } );

    function transformVertex( vertexPosition, mvPosition, center, scale, sin, cos ) {

    	// compute position in camera space
    	_alignedPosition.subVectors( vertexPosition, center ).addScalar( 0.5 ).multiply( scale );

    	// to check if rotation is not zero
    	if ( sin !== undefined ) {

    		_rotatedPosition.x = ( cos * _alignedPosition.x ) - ( sin * _alignedPosition.y );
    		_rotatedPosition.y = ( sin * _alignedPosition.x ) + ( cos * _alignedPosition.y );

    	} else {

    		_rotatedPosition.copy( _alignedPosition );

    	}


    	vertexPosition.copy( mvPosition );
    	vertexPosition.x += _rotatedPosition.x;
    	vertexPosition.y += _rotatedPosition.y;

    	// transform to world space
    	vertexPosition.applyMatrix4( _viewWorldMatrix );

    }

    /**
     * @author mikael emtinger / http://gomo.se/
     * @author alteredq / http://alteredqualia.com/
     * @author mrdoob / http://mrdoob.com/
     */

    var _v1$4 = new Vector3();
    var _v2$2 = new Vector3();

    function LOD() {

    	Object3D.call( this );

    	this.type = 'LOD';

    	Object.defineProperties( this, {
    		levels: {
    			enumerable: true,
    			value: []
    		}
    	} );

    	this.autoUpdate = true;

    }

    LOD.prototype = Object.assign( Object.create( Object3D.prototype ), {

    	constructor: LOD,

    	isLOD: true,

    	copy: function ( source ) {

    		Object3D.prototype.copy.call( this, source, false );

    		var levels = source.levels;

    		for ( var i = 0, l = levels.length; i < l; i ++ ) {

    			var level = levels[ i ];

    			this.addLevel( level.object.clone(), level.distance );

    		}

    		return this;

    	},

    	addLevel: function ( object, distance ) {

    		if ( distance === undefined ) distance = 0;

    		distance = Math.abs( distance );

    		var levels = this.levels;

    		for ( var l = 0; l < levels.length; l ++ ) {

    			if ( distance < levels[ l ].distance ) {

    				break;

    			}

    		}

    		levels.splice( l, 0, { distance: distance, object: object } );

    		this.add( object );

    		return this;

    	},

    	getObjectForDistance: function ( distance ) {

    		var levels = this.levels;

    		for ( var i = 1, l = levels.length; i < l; i ++ ) {

    			if ( distance < levels[ i ].distance ) {

    				break;

    			}

    		}

    		return levels[ i - 1 ].object;

    	},

    	raycast: function ( raycaster, intersects ) {

    		_v1$4.setFromMatrixPosition( this.matrixWorld );

    		var distance = raycaster.ray.origin.distanceTo( _v1$4 );

    		this.getObjectForDistance( distance ).raycast( raycaster, intersects );

    	},

    	update: function ( camera ) {

    		var levels = this.levels;

    		if ( levels.length > 1 ) {

    			_v1$4.setFromMatrixPosition( camera.matrixWorld );
    			_v2$2.setFromMatrixPosition( this.matrixWorld );

    			var distance = _v1$4.distanceTo( _v2$2 );

    			levels[ 0 ].object.visible = true;

    			for ( var i = 1, l = levels.length; i < l; i ++ ) {

    				if ( distance >= levels[ i ].distance ) {

    					levels[ i - 1 ].object.visible = false;
    					levels[ i ].object.visible = true;

    				} else {

    					break;

    				}

    			}

    			for ( ; i < l; i ++ ) {

    				levels[ i ].object.visible = false;

    			}

    		}

    	},

    	toJSON: function ( meta ) {

    		var data = Object3D.prototype.toJSON.call( this, meta );

    		data.object.levels = [];

    		var levels = this.levels;

    		for ( var i = 0, l = levels.length; i < l; i ++ ) {

    			var level = levels[ i ];

    			data.object.levels.push( {
    				object: level.object.uuid,
    				distance: level.distance
    			} );

    		}

    		return data;

    	}

    } );

    /**
     * @author mikael emtinger / http://gomo.se/
     * @author alteredq / http://alteredqualia.com/
     * @author ikerr / http://verold.com
     */

    function SkinnedMesh( geometry, material ) {

    	if ( geometry && geometry.isGeometry ) {

    		console.error( 'THREE.SkinnedMesh no longer supports THREE.Geometry. Use THREE.BufferGeometry instead.' );

    	}

    	Mesh.call( this, geometry, material );

    	this.type = 'SkinnedMesh';

    	this.bindMode = 'attached';
    	this.bindMatrix = new Matrix4();
    	this.bindMatrixInverse = new Matrix4();

    }

    SkinnedMesh.prototype = Object.assign( Object.create( Mesh.prototype ), {

    	constructor: SkinnedMesh,

    	isSkinnedMesh: true,

    	bind: function ( skeleton, bindMatrix ) {

    		this.skeleton = skeleton;

    		if ( bindMatrix === undefined ) {

    			this.updateMatrixWorld( true );

    			this.skeleton.calculateInverses();

    			bindMatrix = this.matrixWorld;

    		}

    		this.bindMatrix.copy( bindMatrix );
    		this.bindMatrixInverse.getInverse( bindMatrix );

    	},

    	pose: function () {

    		this.skeleton.pose();

    	},

    	normalizeSkinWeights: function () {

    		var vector = new Vector4();

    		var skinWeight = this.geometry.attributes.skinWeight;

    		for ( var i = 0, l = skinWeight.count; i < l; i ++ ) {

    			vector.x = skinWeight.getX( i );
    			vector.y = skinWeight.getY( i );
    			vector.z = skinWeight.getZ( i );
    			vector.w = skinWeight.getW( i );

    			var scale = 1.0 / vector.manhattanLength();

    			if ( scale !== Infinity ) {

    				vector.multiplyScalar( scale );

    			} else {

    				vector.set( 1, 0, 0, 0 ); // do something reasonable

    			}

    			skinWeight.setXYZW( i, vector.x, vector.y, vector.z, vector.w );

    		}

    	},

    	updateMatrixWorld: function ( force ) {

    		Mesh.prototype.updateMatrixWorld.call( this, force );

    		if ( this.bindMode === 'attached' ) {

    			this.bindMatrixInverse.getInverse( this.matrixWorld );

    		} else if ( this.bindMode === 'detached' ) {

    			this.bindMatrixInverse.getInverse( this.bindMatrix );

    		} else {

    			console.warn( 'THREE.SkinnedMesh: Unrecognized bindMode: ' + this.bindMode );

    		}

    	},

    	clone: function () {

    		return new this.constructor( this.geometry, this.material ).copy( this );

    	}

    } );

    /**
     * @author mikael emtinger / http://gomo.se/
     * @author alteredq / http://alteredqualia.com/
     * @author michael guerrero / http://realitymeltdown.com
     * @author ikerr / http://verold.com
     */

    var _offsetMatrix = new Matrix4();
    var _identityMatrix = new Matrix4();

    function Skeleton( bones, boneInverses ) {

    	// copy the bone array

    	bones = bones || [];

    	this.bones = bones.slice( 0 );
    	this.boneMatrices = new Float32Array( this.bones.length * 16 );

    	// use the supplied bone inverses or calculate the inverses

    	if ( boneInverses === undefined ) {

    		this.calculateInverses();

    	} else {

    		if ( this.bones.length === boneInverses.length ) {

    			this.boneInverses = boneInverses.slice( 0 );

    		} else {

    			console.warn( 'THREE.Skeleton boneInverses is the wrong length.' );

    			this.boneInverses = [];

    			for ( var i = 0, il = this.bones.length; i < il; i ++ ) {

    				this.boneInverses.push( new Matrix4() );

    			}

    		}

    	}

    }

    Object.assign( Skeleton.prototype, {

    	calculateInverses: function () {

    		this.boneInverses = [];

    		for ( var i = 0, il = this.bones.length; i < il; i ++ ) {

    			var inverse = new Matrix4();

    			if ( this.bones[ i ] ) {

    				inverse.getInverse( this.bones[ i ].matrixWorld );

    			}

    			this.boneInverses.push( inverse );

    		}

    	},

    	pose: function () {

    		var bone, i, il;

    		// recover the bind-time world matrices

    		for ( i = 0, il = this.bones.length; i < il; i ++ ) {

    			bone = this.bones[ i ];

    			if ( bone ) {

    				bone.matrixWorld.getInverse( this.boneInverses[ i ] );

    			}

    		}

    		// compute the local matrices, positions, rotations and scales

    		for ( i = 0, il = this.bones.length; i < il; i ++ ) {

    			bone = this.bones[ i ];

    			if ( bone ) {

    				if ( bone.parent && bone.parent.isBone ) {

    					bone.matrix.getInverse( bone.parent.matrixWorld );
    					bone.matrix.multiply( bone.matrixWorld );

    				} else {

    					bone.matrix.copy( bone.matrixWorld );

    				}

    				bone.matrix.decompose( bone.position, bone.quaternion, bone.scale );

    			}

    		}

    	},

    	update: function () {

    		var bones = this.bones;
    		var boneInverses = this.boneInverses;
    		var boneMatrices = this.boneMatrices;
    		var boneTexture = this.boneTexture;

    		// flatten bone matrices to array

    		for ( var i = 0, il = bones.length; i < il; i ++ ) {

    			// compute the offset between the current and the original transform

    			var matrix = bones[ i ] ? bones[ i ].matrixWorld : _identityMatrix;

    			_offsetMatrix.multiplyMatrices( matrix, boneInverses[ i ] );
    			_offsetMatrix.toArray( boneMatrices, i * 16 );

    		}

    		if ( boneTexture !== undefined ) {

    			boneTexture.needsUpdate = true;

    		}

    	},

    	clone: function () {

    		return new Skeleton( this.bones, this.boneInverses );

    	},

    	getBoneByName: function ( name ) {

    		for ( var i = 0, il = this.bones.length; i < il; i ++ ) {

    			var bone = this.bones[ i ];

    			if ( bone.name === name ) {

    				return bone;

    			}

    		}

    		return undefined;

    	}

    } );

    /**
     * @author mikael emtinger / http://gomo.se/
     * @author alteredq / http://alteredqualia.com/
     * @author ikerr / http://verold.com
     */

    function Bone() {

    	Object3D.call( this );

    	this.type = 'Bone';

    }

    Bone.prototype = Object.assign( Object.create( Object3D.prototype ), {

    	constructor: Bone,

    	isBone: true

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     * @author alteredq / http://alteredqualia.com/
     *
     * parameters = {
     *  color: <hex>,
     *  opacity: <float>,
     *
     *  linewidth: <float>,
     *  linecap: "round",
     *  linejoin: "round"
     * }
     */

    function LineBasicMaterial( parameters ) {

    	Material.call( this );

    	this.type = 'LineBasicMaterial';

    	this.color = new Color$7( 0xffffff );

    	this.linewidth = 1;
    	this.linecap = 'round';
    	this.linejoin = 'round';

    	this.lights = false;

    	this.setValues( parameters );

    }

    LineBasicMaterial.prototype = Object.create( Material.prototype );
    LineBasicMaterial.prototype.constructor = LineBasicMaterial;

    LineBasicMaterial.prototype.isLineBasicMaterial = true;

    LineBasicMaterial.prototype.copy = function ( source ) {

    	Material.prototype.copy.call( this, source );

    	this.color.copy( source.color );

    	this.linewidth = source.linewidth;
    	this.linecap = source.linecap;
    	this.linejoin = source.linejoin;

    	return this;

    };

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    var _start = new Vector3();
    var _end = new Vector3();
    var _inverseMatrix$1 = new Matrix4();
    var _ray$1 = new Ray();
    var _sphere$2 = new Sphere();

    function Line( geometry, material, mode ) {

    	if ( mode === 1 ) {

    		console.error( 'THREE.Line: parameter THREE.LinePieces no longer supported. Use THREE.LineSegments instead.' );

    	}

    	Object3D.call( this );

    	this.type = 'Line';

    	this.geometry = geometry !== undefined ? geometry : new BufferGeometry();
    	this.material = material !== undefined ? material : new LineBasicMaterial( { color: Math.random() * 0xffffff } );

    }

    Line.prototype = Object.assign( Object.create( Object3D.prototype ), {

    	constructor: Line,

    	isLine: true,

    	computeLineDistances: function () {

    		var geometry = this.geometry;

    		if ( geometry.isBufferGeometry ) {

    			// we assume non-indexed geometry

    			if ( geometry.index === null ) {

    				var positionAttribute = geometry.attributes.position;
    				var lineDistances = [ 0 ];

    				for ( var i = 1, l = positionAttribute.count; i < l; i ++ ) {

    					_start.fromBufferAttribute( positionAttribute, i - 1 );
    					_end.fromBufferAttribute( positionAttribute, i );

    					lineDistances[ i ] = lineDistances[ i - 1 ];
    					lineDistances[ i ] += _start.distanceTo( _end );

    				}

    				geometry.addAttribute( 'lineDistance', new Float32BufferAttribute( lineDistances, 1 ) );

    			} else {

    				console.warn( 'THREE.Line.computeLineDistances(): Computation only possible with non-indexed BufferGeometry.' );

    			}

    		} else if ( geometry.isGeometry ) {

    			var vertices = geometry.vertices;
    			var lineDistances = geometry.lineDistances;

    			lineDistances[ 0 ] = 0;

    			for ( var i = 1, l = vertices.length; i < l; i ++ ) {

    				lineDistances[ i ] = lineDistances[ i - 1 ];
    				lineDistances[ i ] += vertices[ i - 1 ].distanceTo( vertices[ i ] );

    			}

    		}

    		return this;

    	},

    	raycast: function ( raycaster, intersects ) {

    		var precision = raycaster.linePrecision;

    		var geometry = this.geometry;
    		var matrixWorld = this.matrixWorld;

    		// Checking boundingSphere distance to ray

    		if ( geometry.boundingSphere === null ) geometry.computeBoundingSphere();

    		_sphere$2.copy( geometry.boundingSphere );
    		_sphere$2.applyMatrix4( matrixWorld );
    		_sphere$2.radius += precision;

    		if ( raycaster.ray.intersectsSphere( _sphere$2 ) === false ) return;

    		//

    		_inverseMatrix$1.getInverse( matrixWorld );
    		_ray$1.copy( raycaster.ray ).applyMatrix4( _inverseMatrix$1 );

    		var localPrecision = precision / ( ( this.scale.x + this.scale.y + this.scale.z ) / 3 );
    		var localPrecisionSq = localPrecision * localPrecision;

    		var vStart = new Vector3();
    		var vEnd = new Vector3();
    		var interSegment = new Vector3();
    		var interRay = new Vector3();
    		var step = ( this && this.isLineSegments ) ? 2 : 1;

    		if ( geometry.isBufferGeometry ) {

    			var index = geometry.index;
    			var attributes = geometry.attributes;
    			var positions = attributes.position.array;

    			if ( index !== null ) {

    				var indices = index.array;

    				for ( var i = 0, l = indices.length - 1; i < l; i += step ) {

    					var a = indices[ i ];
    					var b = indices[ i + 1 ];

    					vStart.fromArray( positions, a * 3 );
    					vEnd.fromArray( positions, b * 3 );

    					var distSq = _ray$1.distanceSqToSegment( vStart, vEnd, interRay, interSegment );

    					if ( distSq > localPrecisionSq ) continue;

    					interRay.applyMatrix4( this.matrixWorld ); //Move back to world space for distance calculation

    					var distance = raycaster.ray.origin.distanceTo( interRay );

    					if ( distance < raycaster.near || distance > raycaster.far ) continue;

    					intersects.push( {

    						distance: distance,
    						// What do we want? intersection point on the ray or on the segment??
    						// point: raycaster.ray.at( distance ),
    						point: interSegment.clone().applyMatrix4( this.matrixWorld ),
    						index: i,
    						face: null,
    						faceIndex: null,
    						object: this

    					} );

    				}

    			} else {

    				for ( var i = 0, l = positions.length / 3 - 1; i < l; i += step ) {

    					vStart.fromArray( positions, 3 * i );
    					vEnd.fromArray( positions, 3 * i + 3 );

    					var distSq = _ray$1.distanceSqToSegment( vStart, vEnd, interRay, interSegment );

    					if ( distSq > localPrecisionSq ) continue;

    					interRay.applyMatrix4( this.matrixWorld ); //Move back to world space for distance calculation

    					var distance = raycaster.ray.origin.distanceTo( interRay );

    					if ( distance < raycaster.near || distance > raycaster.far ) continue;

    					intersects.push( {

    						distance: distance,
    						// What do we want? intersection point on the ray or on the segment??
    						// point: raycaster.ray.at( distance ),
    						point: interSegment.clone().applyMatrix4( this.matrixWorld ),
    						index: i,
    						face: null,
    						faceIndex: null,
    						object: this

    					} );

    				}

    			}

    		} else if ( geometry.isGeometry ) {

    			var vertices = geometry.vertices;
    			var nbVertices = vertices.length;

    			for ( var i = 0; i < nbVertices - 1; i += step ) {

    				var distSq = _ray$1.distanceSqToSegment( vertices[ i ], vertices[ i + 1 ], interRay, interSegment );

    				if ( distSq > localPrecisionSq ) continue;

    				interRay.applyMatrix4( this.matrixWorld ); //Move back to world space for distance calculation

    				var distance = raycaster.ray.origin.distanceTo( interRay );

    				if ( distance < raycaster.near || distance > raycaster.far ) continue;

    				intersects.push( {

    					distance: distance,
    					// What do we want? intersection point on the ray or on the segment??
    					// point: raycaster.ray.at( distance ),
    					point: interSegment.clone().applyMatrix4( this.matrixWorld ),
    					index: i,
    					face: null,
    					faceIndex: null,
    					object: this

    				} );

    			}

    		}

    	},

    	clone: function () {

    		return new this.constructor( this.geometry, this.material ).copy( this );

    	}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    var _start$1 = new Vector3();
    var _end$1 = new Vector3();

    function LineSegments( geometry, material ) {

    	Line.call( this, geometry, material );

    	this.type = 'LineSegments';

    }

    LineSegments.prototype = Object.assign( Object.create( Line.prototype ), {

    	constructor: LineSegments,

    	isLineSegments: true,

    	computeLineDistances: function () {

    		var geometry = this.geometry;

    		if ( geometry.isBufferGeometry ) {

    			// we assume non-indexed geometry

    			if ( geometry.index === null ) {

    				var positionAttribute = geometry.attributes.position;
    				var lineDistances = [];

    				for ( var i = 0, l = positionAttribute.count; i < l; i += 2 ) {

    					_start$1.fromBufferAttribute( positionAttribute, i );
    					_end$1.fromBufferAttribute( positionAttribute, i + 1 );

    					lineDistances[ i ] = ( i === 0 ) ? 0 : lineDistances[ i - 1 ];
    					lineDistances[ i + 1 ] = lineDistances[ i ] + _start$1.distanceTo( _end$1 );

    				}

    				geometry.addAttribute( 'lineDistance', new Float32BufferAttribute( lineDistances, 1 ) );

    			} else {

    				console.warn( 'THREE.LineSegments.computeLineDistances(): Computation only possible with non-indexed BufferGeometry.' );

    			}

    		} else if ( geometry.isGeometry ) {

    			var vertices = geometry.vertices;
    			var lineDistances = geometry.lineDistances;

    			for ( var i = 0, l = vertices.length; i < l; i += 2 ) {

    				_start$1.copy( vertices[ i ] );
    				_end$1.copy( vertices[ i + 1 ] );

    				lineDistances[ i ] = ( i === 0 ) ? 0 : lineDistances[ i - 1 ];
    				lineDistances[ i + 1 ] = lineDistances[ i ] + _start$1.distanceTo( _end$1 );

    			}

    		}

    		return this;

    	}

    } );

    /**
     * @author mgreter / http://github.com/mgreter
     */

    function LineLoop( geometry, material ) {

    	Line.call( this, geometry, material );

    	this.type = 'LineLoop';

    }

    LineLoop.prototype = Object.assign( Object.create( Line.prototype ), {

    	constructor: LineLoop,

    	isLineLoop: true,

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     * @author alteredq / http://alteredqualia.com/
     *
     * parameters = {
     *  color: <hex>,
     *  opacity: <float>,
     *  map: new THREE.Texture( <Image> ),
     *
     *  size: <float>,
     *  sizeAttenuation: <bool>
     *
     *  morphTargets: <bool>
     * }
     */

    function PointsMaterial( parameters ) {

    	Material.call( this );

    	this.type = 'PointsMaterial';

    	this.color = new Color$7( 0xffffff );

    	this.map = null;

    	this.size = 1;
    	this.sizeAttenuation = true;

    	this.morphTargets = false;

    	this.lights = false;

    	this.setValues( parameters );

    }

    PointsMaterial.prototype = Object.create( Material.prototype );
    PointsMaterial.prototype.constructor = PointsMaterial;

    PointsMaterial.prototype.isPointsMaterial = true;

    PointsMaterial.prototype.copy = function ( source ) {

    	Material.prototype.copy.call( this, source );

    	this.color.copy( source.color );

    	this.map = source.map;

    	this.size = source.size;
    	this.sizeAttenuation = source.sizeAttenuation;

    	this.morphTargets = source.morphTargets;

    	return this;

    };

    /**
     * @author alteredq / http://alteredqualia.com/
     */

    var _inverseMatrix$2 = new Matrix4();
    var _ray$2 = new Ray();
    var _sphere$3 = new Sphere();
    var _position$1 = new Vector3();

    function Points( geometry, material ) {

    	Object3D.call( this );

    	this.type = 'Points';

    	this.geometry = geometry !== undefined ? geometry : new BufferGeometry();
    	this.material = material !== undefined ? material : new PointsMaterial( { color: Math.random() * 0xffffff } );

    	this.updateMorphTargets();

    }

    Points.prototype = Object.assign( Object.create( Object3D.prototype ), {

    	constructor: Points,

    	isPoints: true,

    	raycast: function ( raycaster, intersects ) {

    		var geometry = this.geometry;
    		var matrixWorld = this.matrixWorld;
    		var threshold = raycaster.params.Points.threshold;

    		// Checking boundingSphere distance to ray

    		if ( geometry.boundingSphere === null ) geometry.computeBoundingSphere();

    		_sphere$3.copy( geometry.boundingSphere );
    		_sphere$3.applyMatrix4( matrixWorld );
    		_sphere$3.radius += threshold;

    		if ( raycaster.ray.intersectsSphere( _sphere$3 ) === false ) return;

    		//

    		_inverseMatrix$2.getInverse( matrixWorld );
    		_ray$2.copy( raycaster.ray ).applyMatrix4( _inverseMatrix$2 );

    		var localThreshold = threshold / ( ( this.scale.x + this.scale.y + this.scale.z ) / 3 );
    		var localThresholdSq = localThreshold * localThreshold;

    		if ( geometry.isBufferGeometry ) {

    			var index = geometry.index;
    			var attributes = geometry.attributes;
    			var positions = attributes.position.array;

    			if ( index !== null ) {

    				var indices = index.array;

    				for ( var i = 0, il = indices.length; i < il; i ++ ) {

    					var a = indices[ i ];

    					_position$1.fromArray( positions, a * 3 );

    					testPoint( _position$1, a, localThresholdSq, matrixWorld, raycaster, intersects, this );

    				}

    			} else {

    				for ( var i = 0, l = positions.length / 3; i < l; i ++ ) {

    					_position$1.fromArray( positions, i * 3 );

    					testPoint( _position$1, i, localThresholdSq, matrixWorld, raycaster, intersects, this );

    				}

    			}

    		} else {

    			var vertices = geometry.vertices;

    			for ( var i = 0, l = vertices.length; i < l; i ++ ) {

    				testPoint( vertices[ i ], i, localThresholdSq, matrixWorld, raycaster, intersects, this );

    			}

    		}

    	},

    	updateMorphTargets: function () {

    		var geometry = this.geometry;
    		var m, ml, name;

    		if ( geometry.isBufferGeometry ) {

    			var morphAttributes = geometry.morphAttributes;
    			var keys = Object.keys( morphAttributes );

    			if ( keys.length > 0 ) {

    				var morphAttribute = morphAttributes[ keys[ 0 ] ];

    				if ( morphAttribute !== undefined ) {

    					this.morphTargetInfluences = [];
    					this.morphTargetDictionary = {};

    					for ( m = 0, ml = morphAttribute.length; m < ml; m ++ ) {

    						name = morphAttribute[ m ].name || String( m );

    						this.morphTargetInfluences.push( 0 );
    						this.morphTargetDictionary[ name ] = m;

    					}

    				}

    			}

    		} else {

    			var morphTargets = geometry.morphTargets;

    			if ( morphTargets !== undefined && morphTargets.length > 0 ) {

    				console.error( 'THREE.Points.updateMorphTargets() does not support THREE.Geometry. Use THREE.BufferGeometry instead.' );

    			}

    		}

    	},

    	clone: function () {

    		return new this.constructor( this.geometry, this.material ).copy( this );

    	}

    } );

    function testPoint( point, index, localThresholdSq, matrixWorld, raycaster, intersects, object ) {

    	var rayPointDistanceSq = _ray$2.distanceSqToPoint( point );

    	if ( rayPointDistanceSq < localThresholdSq ) {

    		var intersectPoint = new Vector3();

    		_ray$2.closestPointToPoint( point, intersectPoint );
    		intersectPoint.applyMatrix4( matrixWorld );

    		var distance = raycaster.ray.origin.distanceTo( intersectPoint );

    		if ( distance < raycaster.near || distance > raycaster.far ) return;

    		intersects.push( {

    			distance: distance,
    			distanceToRay: Math.sqrt( rayPointDistanceSq ),
    			point: intersectPoint,
    			index: index,
    			face: null,
    			object: object

    		} );

    	}

    }

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function VideoTexture( video, mapping, wrapS, wrapT, magFilter, minFilter, format, type, anisotropy ) {

    	Texture.call( this, video, mapping, wrapS, wrapT, magFilter, minFilter, format, type, anisotropy );

    	this.format = format !== undefined ? format : RGBFormat;

    	this.minFilter = minFilter !== undefined ? minFilter : LinearFilter;
    	this.magFilter = magFilter !== undefined ? magFilter : LinearFilter;

    	this.generateMipmaps = false;

    }

    VideoTexture.prototype = Object.assign( Object.create( Texture.prototype ), {

    	constructor: VideoTexture,

    	isVideoTexture: true,

    	update: function () {

    		var video = this.image;

    		if ( video.readyState >= video.HAVE_CURRENT_DATA ) {

    			this.needsUpdate = true;

    		}

    	}

    } );

    /**
     * @author alteredq / http://alteredqualia.com/
     */

    function CompressedTexture( mipmaps, width, height, format, type, mapping, wrapS, wrapT, magFilter, minFilter, anisotropy, encoding ) {

    	Texture.call( this, null, mapping, wrapS, wrapT, magFilter, minFilter, format, type, anisotropy, encoding );

    	this.image = { width: width, height: height };
    	this.mipmaps = mipmaps;

    	// no flipping for cube textures
    	// (also flipping doesn't work for compressed textures )

    	this.flipY = false;

    	// can't generate mipmaps for compressed textures
    	// mips must be embedded in DDS files

    	this.generateMipmaps = false;

    }

    CompressedTexture.prototype = Object.create( Texture.prototype );
    CompressedTexture.prototype.constructor = CompressedTexture;

    CompressedTexture.prototype.isCompressedTexture = true;

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function CanvasTexture( canvas, mapping, wrapS, wrapT, magFilter, minFilter, format, type, anisotropy ) {

    	Texture.call( this, canvas, mapping, wrapS, wrapT, magFilter, minFilter, format, type, anisotropy );

    	this.needsUpdate = true;

    }

    CanvasTexture.prototype = Object.create( Texture.prototype );
    CanvasTexture.prototype.constructor = CanvasTexture;
    CanvasTexture.prototype.isCanvasTexture = true;

    /**
     * @author Matt DesLauriers / @mattdesl
     * @author atix / arthursilber.de
     */

    function DepthTexture( width, height, type, mapping, wrapS, wrapT, magFilter, minFilter, anisotropy, format ) {

    	format = format !== undefined ? format : DepthFormat;

    	if ( format !== DepthFormat && format !== DepthStencilFormat ) {

    		throw new Error( 'DepthTexture format must be either THREE.DepthFormat or THREE.DepthStencilFormat' );

    	}

    	if ( type === undefined && format === DepthFormat ) type = UnsignedShortType;
    	if ( type === undefined && format === DepthStencilFormat ) type = UnsignedInt248Type;

    	Texture.call( this, null, mapping, wrapS, wrapT, magFilter, minFilter, format, type, anisotropy );

    	this.image = { width: width, height: height };

    	this.magFilter = magFilter !== undefined ? magFilter : NearestFilter;
    	this.minFilter = minFilter !== undefined ? minFilter : NearestFilter;

    	this.flipY = false;
    	this.generateMipmaps	= false;

    }

    DepthTexture.prototype = Object.create( Texture.prototype );
    DepthTexture.prototype.constructor = DepthTexture;
    DepthTexture.prototype.isDepthTexture = true;

    /**
     * @author mrdoob / http://mrdoob.com/
     * @author Mugen87 / https://github.com/Mugen87
     */

    function WireframeGeometry( geometry ) {

    	BufferGeometry.call( this );

    	this.type = 'WireframeGeometry';

    	// buffer

    	var vertices = [];

    	// helper variables

    	var i, j, l, o, ol;
    	var edge = [ 0, 0 ], edges = {}, e, edge1, edge2;
    	var key, keys = [ 'a', 'b', 'c' ];
    	var vertex;

    	// different logic for Geometry and BufferGeometry

    	if ( geometry && geometry.isGeometry ) {

    		// create a data structure that contains all edges without duplicates

    		var faces = geometry.faces;

    		for ( i = 0, l = faces.length; i < l; i ++ ) {

    			var face = faces[ i ];

    			for ( j = 0; j < 3; j ++ ) {

    				edge1 = face[ keys[ j ] ];
    				edge2 = face[ keys[ ( j + 1 ) % 3 ] ];
    				edge[ 0 ] = Math.min( edge1, edge2 ); // sorting prevents duplicates
    				edge[ 1 ] = Math.max( edge1, edge2 );

    				key = edge[ 0 ] + ',' + edge[ 1 ];

    				if ( edges[ key ] === undefined ) {

    					edges[ key ] = { index1: edge[ 0 ], index2: edge[ 1 ] };

    				}

    			}

    		}

    		// generate vertices

    		for ( key in edges ) {

    			e = edges[ key ];

    			vertex = geometry.vertices[ e.index1 ];
    			vertices.push( vertex.x, vertex.y, vertex.z );

    			vertex = geometry.vertices[ e.index2 ];
    			vertices.push( vertex.x, vertex.y, vertex.z );

    		}

    	} else if ( geometry && geometry.isBufferGeometry ) {

    		var position, indices, groups;
    		var group, start, count;
    		var index1, index2;

    		vertex = new Vector3();

    		if ( geometry.index !== null ) {

    			// indexed BufferGeometry

    			position = geometry.attributes.position;
    			indices = geometry.index;
    			groups = geometry.groups;

    			if ( groups.length === 0 ) {

    				groups = [ { start: 0, count: indices.count, materialIndex: 0 } ];

    			}

    			// create a data structure that contains all eges without duplicates

    			for ( o = 0, ol = groups.length; o < ol; ++ o ) {

    				group = groups[ o ];

    				start = group.start;
    				count = group.count;

    				for ( i = start, l = ( start + count ); i < l; i += 3 ) {

    					for ( j = 0; j < 3; j ++ ) {

    						edge1 = indices.getX( i + j );
    						edge2 = indices.getX( i + ( j + 1 ) % 3 );
    						edge[ 0 ] = Math.min( edge1, edge2 ); // sorting prevents duplicates
    						edge[ 1 ] = Math.max( edge1, edge2 );

    						key = edge[ 0 ] + ',' + edge[ 1 ];

    						if ( edges[ key ] === undefined ) {

    							edges[ key ] = { index1: edge[ 0 ], index2: edge[ 1 ] };

    						}

    					}

    				}

    			}

    			// generate vertices

    			for ( key in edges ) {

    				e = edges[ key ];

    				vertex.fromBufferAttribute( position, e.index1 );
    				vertices.push( vertex.x, vertex.y, vertex.z );

    				vertex.fromBufferAttribute( position, e.index2 );
    				vertices.push( vertex.x, vertex.y, vertex.z );

    			}

    		} else {

    			// non-indexed BufferGeometry

    			position = geometry.attributes.position;

    			for ( i = 0, l = ( position.count / 3 ); i < l; i ++ ) {

    				for ( j = 0; j < 3; j ++ ) {

    					// three edges per triangle, an edge is represented as (index1, index2)
    					// e.g. the first triangle has the following edges: (0,1),(1,2),(2,0)

    					index1 = 3 * i + j;
    					vertex.fromBufferAttribute( position, index1 );
    					vertices.push( vertex.x, vertex.y, vertex.z );

    					index2 = 3 * i + ( ( j + 1 ) % 3 );
    					vertex.fromBufferAttribute( position, index2 );
    					vertices.push( vertex.x, vertex.y, vertex.z );

    				}

    			}

    		}

    	}

    	// build geometry

    	this.addAttribute( 'position', new Float32BufferAttribute( vertices, 3 ) );

    }

    WireframeGeometry.prototype = Object.create( BufferGeometry.prototype );
    WireframeGeometry.prototype.constructor = WireframeGeometry;

    /**
     * @author zz85 / https://github.com/zz85
     * @author Mugen87 / https://github.com/Mugen87
     *
     * Parametric Surfaces Geometry
     * based on the brilliant article by @prideout http://prideout.net/blog/?p=44
     */

    // ParametricGeometry

    function ParametricGeometry( func, slices, stacks ) {

    	Geometry.call( this );

    	this.type = 'ParametricGeometry';

    	this.parameters = {
    		func: func,
    		slices: slices,
    		stacks: stacks
    	};

    	this.fromBufferGeometry( new ParametricBufferGeometry( func, slices, stacks ) );
    	this.mergeVertices();

    }

    ParametricGeometry.prototype = Object.create( Geometry.prototype );
    ParametricGeometry.prototype.constructor = ParametricGeometry;

    // ParametricBufferGeometry

    function ParametricBufferGeometry( func, slices, stacks ) {

    	BufferGeometry.call( this );

    	this.type = 'ParametricBufferGeometry';

    	this.parameters = {
    		func: func,
    		slices: slices,
    		stacks: stacks
    	};

    	// buffers

    	var indices = [];
    	var vertices = [];
    	var normals = [];
    	var uvs = [];

    	var EPS = 0.00001;

    	var normal = new Vector3();

    	var p0 = new Vector3(), p1 = new Vector3();
    	var pu = new Vector3(), pv = new Vector3();

    	var i, j;

    	if ( func.length < 3 ) {

    		console.error( 'THREE.ParametricGeometry: Function must now modify a Vector3 as third parameter.' );

    	}

    	// generate vertices, normals and uvs

    	var sliceCount = slices + 1;

    	for ( i = 0; i <= stacks; i ++ ) {

    		var v = i / stacks;

    		for ( j = 0; j <= slices; j ++ ) {

    			var u = j / slices;

    			// vertex

    			func( u, v, p0 );
    			vertices.push( p0.x, p0.y, p0.z );

    			// normal

    			// approximate tangent vectors via finite differences

    			if ( u - EPS >= 0 ) {

    				func( u - EPS, v, p1 );
    				pu.subVectors( p0, p1 );

    			} else {

    				func( u + EPS, v, p1 );
    				pu.subVectors( p1, p0 );

    			}

    			if ( v - EPS >= 0 ) {

    				func( u, v - EPS, p1 );
    				pv.subVectors( p0, p1 );

    			} else {

    				func( u, v + EPS, p1 );
    				pv.subVectors( p1, p0 );

    			}

    			// cross product of tangent vectors returns surface normal

    			normal.crossVectors( pu, pv ).normalize();
    			normals.push( normal.x, normal.y, normal.z );

    			// uv

    			uvs.push( u, v );

    		}

    	}

    	// generate indices

    	for ( i = 0; i < stacks; i ++ ) {

    		for ( j = 0; j < slices; j ++ ) {

    			var a = i * sliceCount + j;
    			var b = i * sliceCount + j + 1;
    			var c = ( i + 1 ) * sliceCount + j + 1;
    			var d = ( i + 1 ) * sliceCount + j;

    			// faces one and two

    			indices.push( a, b, d );
    			indices.push( b, c, d );

    		}

    	}

    	// build geometry

    	this.setIndex( indices );
    	this.addAttribute( 'position', new Float32BufferAttribute( vertices, 3 ) );
    	this.addAttribute( 'normal', new Float32BufferAttribute( normals, 3 ) );
    	this.addAttribute( 'uv', new Float32BufferAttribute( uvs, 2 ) );

    }

    ParametricBufferGeometry.prototype = Object.create( BufferGeometry.prototype );
    ParametricBufferGeometry.prototype.constructor = ParametricBufferGeometry;

    /**
     * @author clockworkgeek / https://github.com/clockworkgeek
     * @author timothypratley / https://github.com/timothypratley
     * @author WestLangley / http://github.com/WestLangley
     * @author Mugen87 / https://github.com/Mugen87
     */

    // PolyhedronGeometry

    function PolyhedronGeometry( vertices, indices, radius, detail ) {

    	Geometry.call( this );

    	this.type = 'PolyhedronGeometry';

    	this.parameters = {
    		vertices: vertices,
    		indices: indices,
    		radius: radius,
    		detail: detail
    	};

    	this.fromBufferGeometry( new PolyhedronBufferGeometry( vertices, indices, radius, detail ) );
    	this.mergeVertices();

    }

    PolyhedronGeometry.prototype = Object.create( Geometry.prototype );
    PolyhedronGeometry.prototype.constructor = PolyhedronGeometry;

    // PolyhedronBufferGeometry

    function PolyhedronBufferGeometry( vertices, indices, radius, detail ) {

    	BufferGeometry.call( this );

    	this.type = 'PolyhedronBufferGeometry';

    	this.parameters = {
    		vertices: vertices,
    		indices: indices,
    		radius: radius,
    		detail: detail
    	};

    	radius = radius || 1;
    	detail = detail || 0;

    	// default buffer data

    	var vertexBuffer = [];
    	var uvBuffer = [];

    	// the subdivision creates the vertex buffer data

    	subdivide( detail );

    	// all vertices should lie on a conceptual sphere with a given radius

    	applyRadius( radius );

    	// finally, create the uv data

    	generateUVs();

    	// build non-indexed geometry

    	this.addAttribute( 'position', new Float32BufferAttribute( vertexBuffer, 3 ) );
    	this.addAttribute( 'normal', new Float32BufferAttribute( vertexBuffer.slice(), 3 ) );
    	this.addAttribute( 'uv', new Float32BufferAttribute( uvBuffer, 2 ) );

    	if ( detail === 0 ) {

    		this.computeVertexNormals(); // flat normals

    	} else {

    		this.normalizeNormals(); // smooth normals

    	}

    	// helper functions

    	function subdivide( detail ) {

    		var a = new Vector3();
    		var b = new Vector3();
    		var c = new Vector3();

    		// iterate over all faces and apply a subdivison with the given detail value

    		for ( var i = 0; i < indices.length; i += 3 ) {

    			// get the vertices of the face

    			getVertexByIndex( indices[ i + 0 ], a );
    			getVertexByIndex( indices[ i + 1 ], b );
    			getVertexByIndex( indices[ i + 2 ], c );

    			// perform subdivision

    			subdivideFace( a, b, c, detail );

    		}

    	}

    	function subdivideFace( a, b, c, detail ) {

    		var cols = Math.pow( 2, detail );

    		// we use this multidimensional array as a data structure for creating the subdivision

    		var v = [];

    		var i, j;

    		// construct all of the vertices for this subdivision

    		for ( i = 0; i <= cols; i ++ ) {

    			v[ i ] = [];

    			var aj = a.clone().lerp( c, i / cols );
    			var bj = b.clone().lerp( c, i / cols );

    			var rows = cols - i;

    			for ( j = 0; j <= rows; j ++ ) {

    				if ( j === 0 && i === cols ) {

    					v[ i ][ j ] = aj;

    				} else {

    					v[ i ][ j ] = aj.clone().lerp( bj, j / rows );

    				}

    			}

    		}

    		// construct all of the faces

    		for ( i = 0; i < cols; i ++ ) {

    			for ( j = 0; j < 2 * ( cols - i ) - 1; j ++ ) {

    				var k = Math.floor( j / 2 );

    				if ( j % 2 === 0 ) {

    					pushVertex( v[ i ][ k + 1 ] );
    					pushVertex( v[ i + 1 ][ k ] );
    					pushVertex( v[ i ][ k ] );

    				} else {

    					pushVertex( v[ i ][ k + 1 ] );
    					pushVertex( v[ i + 1 ][ k + 1 ] );
    					pushVertex( v[ i + 1 ][ k ] );

    				}

    			}

    		}

    	}

    	function applyRadius( radius ) {

    		var vertex = new Vector3();

    		// iterate over the entire buffer and apply the radius to each vertex

    		for ( var i = 0; i < vertexBuffer.length; i += 3 ) {

    			vertex.x = vertexBuffer[ i + 0 ];
    			vertex.y = vertexBuffer[ i + 1 ];
    			vertex.z = vertexBuffer[ i + 2 ];

    			vertex.normalize().multiplyScalar( radius );

    			vertexBuffer[ i + 0 ] = vertex.x;
    			vertexBuffer[ i + 1 ] = vertex.y;
    			vertexBuffer[ i + 2 ] = vertex.z;

    		}

    	}

    	function generateUVs() {

    		var vertex = new Vector3();

    		for ( var i = 0; i < vertexBuffer.length; i += 3 ) {

    			vertex.x = vertexBuffer[ i + 0 ];
    			vertex.y = vertexBuffer[ i + 1 ];
    			vertex.z = vertexBuffer[ i + 2 ];

    			var u = azimuth( vertex ) / 2 / Math.PI + 0.5;
    			var v = inclination( vertex ) / Math.PI + 0.5;
    			uvBuffer.push( u, 1 - v );

    		}

    		correctUVs();

    		correctSeam();

    	}

    	function correctSeam() {

    		// handle case when face straddles the seam, see #3269

    		for ( var i = 0; i < uvBuffer.length; i += 6 ) {

    			// uv data of a single face

    			var x0 = uvBuffer[ i + 0 ];
    			var x1 = uvBuffer[ i + 2 ];
    			var x2 = uvBuffer[ i + 4 ];

    			var max = Math.max( x0, x1, x2 );
    			var min = Math.min( x0, x1, x2 );

    			// 0.9 is somewhat arbitrary

    			if ( max > 0.9 && min < 0.1 ) {

    				if ( x0 < 0.2 ) uvBuffer[ i + 0 ] += 1;
    				if ( x1 < 0.2 ) uvBuffer[ i + 2 ] += 1;
    				if ( x2 < 0.2 ) uvBuffer[ i + 4 ] += 1;

    			}

    		}

    	}

    	function pushVertex( vertex ) {

    		vertexBuffer.push( vertex.x, vertex.y, vertex.z );

    	}

    	function getVertexByIndex( index, vertex ) {

    		var stride = index * 3;

    		vertex.x = vertices[ stride + 0 ];
    		vertex.y = vertices[ stride + 1 ];
    		vertex.z = vertices[ stride + 2 ];

    	}

    	function correctUVs() {

    		var a = new Vector3();
    		var b = new Vector3();
    		var c = new Vector3();

    		var centroid = new Vector3();

    		var uvA = new Vector2();
    		var uvB = new Vector2();
    		var uvC = new Vector2();

    		for ( var i = 0, j = 0; i < vertexBuffer.length; i += 9, j += 6 ) {

    			a.set( vertexBuffer[ i + 0 ], vertexBuffer[ i + 1 ], vertexBuffer[ i + 2 ] );
    			b.set( vertexBuffer[ i + 3 ], vertexBuffer[ i + 4 ], vertexBuffer[ i + 5 ] );
    			c.set( vertexBuffer[ i + 6 ], vertexBuffer[ i + 7 ], vertexBuffer[ i + 8 ] );

    			uvA.set( uvBuffer[ j + 0 ], uvBuffer[ j + 1 ] );
    			uvB.set( uvBuffer[ j + 2 ], uvBuffer[ j + 3 ] );
    			uvC.set( uvBuffer[ j + 4 ], uvBuffer[ j + 5 ] );

    			centroid.copy( a ).add( b ).add( c ).divideScalar( 3 );

    			var azi = azimuth( centroid );

    			correctUV( uvA, j + 0, a, azi );
    			correctUV( uvB, j + 2, b, azi );
    			correctUV( uvC, j + 4, c, azi );

    		}

    	}

    	function correctUV( uv, stride, vector, azimuth ) {

    		if ( ( azimuth < 0 ) && ( uv.x === 1 ) ) {

    			uvBuffer[ stride ] = uv.x - 1;

    		}

    		if ( ( vector.x === 0 ) && ( vector.z === 0 ) ) {

    			uvBuffer[ stride ] = azimuth / 2 / Math.PI + 0.5;

    		}

    	}

    	// Angle around the Y axis, counter-clockwise when looking from above.

    	function azimuth( vector ) {

    		return Math.atan2( vector.z, - vector.x );

    	}


    	// Angle above the XZ plane.

    	function inclination( vector ) {

    		return Math.atan2( - vector.y, Math.sqrt( ( vector.x * vector.x ) + ( vector.z * vector.z ) ) );

    	}

    }

    PolyhedronBufferGeometry.prototype = Object.create( BufferGeometry.prototype );
    PolyhedronBufferGeometry.prototype.constructor = PolyhedronBufferGeometry;

    /**
     * @author timothypratley / https://github.com/timothypratley
     * @author Mugen87 / https://github.com/Mugen87
     */

    // TetrahedronGeometry

    function TetrahedronGeometry( radius, detail ) {

    	Geometry.call( this );

    	this.type = 'TetrahedronGeometry';

    	this.parameters = {
    		radius: radius,
    		detail: detail
    	};

    	this.fromBufferGeometry( new TetrahedronBufferGeometry( radius, detail ) );
    	this.mergeVertices();

    }

    TetrahedronGeometry.prototype = Object.create( Geometry.prototype );
    TetrahedronGeometry.prototype.constructor = TetrahedronGeometry;

    // TetrahedronBufferGeometry

    function TetrahedronBufferGeometry( radius, detail ) {

    	var vertices = [
    		1, 1, 1, 	- 1, - 1, 1, 	- 1, 1, - 1, 	1, - 1, - 1
    	];

    	var indices = [
    		2, 1, 0, 	0, 3, 2,	1, 3, 0,	2, 3, 1
    	];

    	PolyhedronBufferGeometry.call( this, vertices, indices, radius, detail );

    	this.type = 'TetrahedronBufferGeometry';

    	this.parameters = {
    		radius: radius,
    		detail: detail
    	};

    }

    TetrahedronBufferGeometry.prototype = Object.create( PolyhedronBufferGeometry.prototype );
    TetrahedronBufferGeometry.prototype.constructor = TetrahedronBufferGeometry;

    /**
     * @author timothypratley / https://github.com/timothypratley
     * @author Mugen87 / https://github.com/Mugen87
     */

    // OctahedronGeometry

    function OctahedronGeometry( radius, detail ) {

    	Geometry.call( this );

    	this.type = 'OctahedronGeometry';

    	this.parameters = {
    		radius: radius,
    		detail: detail
    	};

    	this.fromBufferGeometry( new OctahedronBufferGeometry( radius, detail ) );
    	this.mergeVertices();

    }

    OctahedronGeometry.prototype = Object.create( Geometry.prototype );
    OctahedronGeometry.prototype.constructor = OctahedronGeometry;

    // OctahedronBufferGeometry

    function OctahedronBufferGeometry( radius, detail ) {

    	var vertices = [
    		1, 0, 0, 	- 1, 0, 0,	0, 1, 0,
    		0, - 1, 0, 	0, 0, 1,	0, 0, - 1
    	];

    	var indices = [
    		0, 2, 4,	0, 4, 3,	0, 3, 5,
    		0, 5, 2,	1, 2, 5,	1, 5, 3,
    		1, 3, 4,	1, 4, 2
    	];

    	PolyhedronBufferGeometry.call( this, vertices, indices, radius, detail );

    	this.type = 'OctahedronBufferGeometry';

    	this.parameters = {
    		radius: radius,
    		detail: detail
    	};

    }

    OctahedronBufferGeometry.prototype = Object.create( PolyhedronBufferGeometry.prototype );
    OctahedronBufferGeometry.prototype.constructor = OctahedronBufferGeometry;

    /**
     * @author timothypratley / https://github.com/timothypratley
     * @author Mugen87 / https://github.com/Mugen87
     */

    // IcosahedronGeometry

    function IcosahedronGeometry( radius, detail ) {

    	Geometry.call( this );

    	this.type = 'IcosahedronGeometry';

    	this.parameters = {
    		radius: radius,
    		detail: detail
    	};

    	this.fromBufferGeometry( new IcosahedronBufferGeometry( radius, detail ) );
    	this.mergeVertices();

    }

    IcosahedronGeometry.prototype = Object.create( Geometry.prototype );
    IcosahedronGeometry.prototype.constructor = IcosahedronGeometry;

    // IcosahedronBufferGeometry

    function IcosahedronBufferGeometry( radius, detail ) {

    	var t = ( 1 + Math.sqrt( 5 ) ) / 2;

    	var vertices = [
    		- 1, t, 0, 	1, t, 0, 	- 1, - t, 0, 	1, - t, 0,
    		 0, - 1, t, 	0, 1, t,	0, - 1, - t, 	0, 1, - t,
    		 t, 0, - 1, 	t, 0, 1, 	- t, 0, - 1, 	- t, 0, 1
    	];

    	var indices = [
    		 0, 11, 5, 	0, 5, 1, 	0, 1, 7, 	0, 7, 10, 	0, 10, 11,
    		 1, 5, 9, 	5, 11, 4,	11, 10, 2,	10, 7, 6,	7, 1, 8,
    		 3, 9, 4, 	3, 4, 2,	3, 2, 6,	3, 6, 8,	3, 8, 9,
    		 4, 9, 5, 	2, 4, 11,	6, 2, 10,	8, 6, 7,	9, 8, 1
    	];

    	PolyhedronBufferGeometry.call( this, vertices, indices, radius, detail );

    	this.type = 'IcosahedronBufferGeometry';

    	this.parameters = {
    		radius: radius,
    		detail: detail
    	};

    }

    IcosahedronBufferGeometry.prototype = Object.create( PolyhedronBufferGeometry.prototype );
    IcosahedronBufferGeometry.prototype.constructor = IcosahedronBufferGeometry;

    /**
     * @author Abe Pazos / https://hamoid.com
     * @author Mugen87 / https://github.com/Mugen87
     */

    // DodecahedronGeometry

    function DodecahedronGeometry( radius, detail ) {

    	Geometry.call( this );

    	this.type = 'DodecahedronGeometry';

    	this.parameters = {
    		radius: radius,
    		detail: detail
    	};

    	this.fromBufferGeometry( new DodecahedronBufferGeometry( radius, detail ) );
    	this.mergeVertices();

    }

    DodecahedronGeometry.prototype = Object.create( Geometry.prototype );
    DodecahedronGeometry.prototype.constructor = DodecahedronGeometry;

    // DodecahedronBufferGeometry

    function DodecahedronBufferGeometry( radius, detail ) {

    	var t = ( 1 + Math.sqrt( 5 ) ) / 2;
    	var r = 1 / t;

    	var vertices = [

    		// (±1, ±1, ±1)
    		- 1, - 1, - 1,	- 1, - 1, 1,
    		- 1, 1, - 1, - 1, 1, 1,
    		1, - 1, - 1, 1, - 1, 1,
    		1, 1, - 1, 1, 1, 1,

    		// (0, ±1/φ, ±φ)
    		 0, - r, - t, 0, - r, t,
    		 0, r, - t, 0, r, t,

    		// (±1/φ, ±φ, 0)
    		- r, - t, 0, - r, t, 0,
    		 r, - t, 0, r, t, 0,

    		// (±φ, 0, ±1/φ)
    		- t, 0, - r, t, 0, - r,
    		- t, 0, r, t, 0, r
    	];

    	var indices = [
    		3, 11, 7, 	3, 7, 15, 	3, 15, 13,
    		7, 19, 17, 	7, 17, 6, 	7, 6, 15,
    		17, 4, 8, 	17, 8, 10, 	17, 10, 6,
    		8, 0, 16, 	8, 16, 2, 	8, 2, 10,
    		0, 12, 1, 	0, 1, 18, 	0, 18, 16,
    		6, 10, 2, 	6, 2, 13, 	6, 13, 15,
    		2, 16, 18, 	2, 18, 3, 	2, 3, 13,
    		18, 1, 9, 	18, 9, 11, 	18, 11, 3,
    		4, 14, 12, 	4, 12, 0, 	4, 0, 8,
    		11, 9, 5, 	11, 5, 19, 	11, 19, 7,
    		19, 5, 14, 	19, 14, 4, 	19, 4, 17,
    		1, 12, 14, 	1, 14, 5, 	1, 5, 9
    	];

    	PolyhedronBufferGeometry.call( this, vertices, indices, radius, detail );

    	this.type = 'DodecahedronBufferGeometry';

    	this.parameters = {
    		radius: radius,
    		detail: detail
    	};

    }

    DodecahedronBufferGeometry.prototype = Object.create( PolyhedronBufferGeometry.prototype );
    DodecahedronBufferGeometry.prototype.constructor = DodecahedronBufferGeometry;

    /**
     * @author oosmoxiecode / https://github.com/oosmoxiecode
     * @author WestLangley / https://github.com/WestLangley
     * @author zz85 / https://github.com/zz85
     * @author miningold / https://github.com/miningold
     * @author jonobr1 / https://github.com/jonobr1
     * @author Mugen87 / https://github.com/Mugen87
     *
     */

    // TubeGeometry

    function TubeGeometry( path, tubularSegments, radius, radialSegments, closed, taper ) {

    	Geometry.call( this );

    	this.type = 'TubeGeometry';

    	this.parameters = {
    		path: path,
    		tubularSegments: tubularSegments,
    		radius: radius,
    		radialSegments: radialSegments,
    		closed: closed
    	};

    	if ( taper !== undefined ) console.warn( 'THREE.TubeGeometry: taper has been removed.' );

    	var bufferGeometry = new TubeBufferGeometry( path, tubularSegments, radius, radialSegments, closed );

    	// expose internals

    	this.tangents = bufferGeometry.tangents;
    	this.normals = bufferGeometry.normals;
    	this.binormals = bufferGeometry.binormals;

    	// create geometry

    	this.fromBufferGeometry( bufferGeometry );
    	this.mergeVertices();

    }

    TubeGeometry.prototype = Object.create( Geometry.prototype );
    TubeGeometry.prototype.constructor = TubeGeometry;

    // TubeBufferGeometry

    function TubeBufferGeometry( path, tubularSegments, radius, radialSegments, closed ) {

    	BufferGeometry.call( this );

    	this.type = 'TubeBufferGeometry';

    	this.parameters = {
    		path: path,
    		tubularSegments: tubularSegments,
    		radius: radius,
    		radialSegments: radialSegments,
    		closed: closed
    	};

    	tubularSegments = tubularSegments || 64;
    	radius = radius || 1;
    	radialSegments = radialSegments || 8;
    	closed = closed || false;

    	var frames = path.computeFrenetFrames( tubularSegments, closed );

    	// expose internals

    	this.tangents = frames.tangents;
    	this.normals = frames.normals;
    	this.binormals = frames.binormals;

    	// helper variables

    	var vertex = new Vector3();
    	var normal = new Vector3();
    	var uv = new Vector2();
    	var P = new Vector3();

    	var i, j;

    	// buffer

    	var vertices = [];
    	var normals = [];
    	var uvs = [];
    	var indices = [];

    	// create buffer data

    	generateBufferData();

    	// build geometry

    	this.setIndex( indices );
    	this.addAttribute( 'position', new Float32BufferAttribute( vertices, 3 ) );
    	this.addAttribute( 'normal', new Float32BufferAttribute( normals, 3 ) );
    	this.addAttribute( 'uv', new Float32BufferAttribute( uvs, 2 ) );

    	// functions

    	function generateBufferData() {

    		for ( i = 0; i < tubularSegments; i ++ ) {

    			generateSegment( i );

    		}

    		// if the geometry is not closed, generate the last row of vertices and normals
    		// at the regular position on the given path
    		//
    		// if the geometry is closed, duplicate the first row of vertices and normals (uvs will differ)

    		generateSegment( ( closed === false ) ? tubularSegments : 0 );

    		// uvs are generated in a separate function.
    		// this makes it easy compute correct values for closed geometries

    		generateUVs();

    		// finally create faces

    		generateIndices();

    	}

    	function generateSegment( i ) {

    		// we use getPointAt to sample evenly distributed points from the given path

    		P = path.getPointAt( i / tubularSegments, P );

    		// retrieve corresponding normal and binormal

    		var N = frames.normals[ i ];
    		var B = frames.binormals[ i ];

    		// generate normals and vertices for the current segment

    		for ( j = 0; j <= radialSegments; j ++ ) {

    			var v = j / radialSegments * Math.PI * 2;

    			var sin = Math.sin( v );
    			var cos = - Math.cos( v );

    			// normal

    			normal.x = ( cos * N.x + sin * B.x );
    			normal.y = ( cos * N.y + sin * B.y );
    			normal.z = ( cos * N.z + sin * B.z );
    			normal.normalize();

    			normals.push( normal.x, normal.y, normal.z );

    			// vertex

    			vertex.x = P.x + radius * normal.x;
    			vertex.y = P.y + radius * normal.y;
    			vertex.z = P.z + radius * normal.z;

    			vertices.push( vertex.x, vertex.y, vertex.z );

    		}

    	}

    	function generateIndices() {

    		for ( j = 1; j <= tubularSegments; j ++ ) {

    			for ( i = 1; i <= radialSegments; i ++ ) {

    				var a = ( radialSegments + 1 ) * ( j - 1 ) + ( i - 1 );
    				var b = ( radialSegments + 1 ) * j + ( i - 1 );
    				var c = ( radialSegments + 1 ) * j + i;
    				var d = ( radialSegments + 1 ) * ( j - 1 ) + i;

    				// faces

    				indices.push( a, b, d );
    				indices.push( b, c, d );

    			}

    		}

    	}

    	function generateUVs() {

    		for ( i = 0; i <= tubularSegments; i ++ ) {

    			for ( j = 0; j <= radialSegments; j ++ ) {

    				uv.x = i / tubularSegments;
    				uv.y = j / radialSegments;

    				uvs.push( uv.x, uv.y );

    			}

    		}

    	}

    }

    TubeBufferGeometry.prototype = Object.create( BufferGeometry.prototype );
    TubeBufferGeometry.prototype.constructor = TubeBufferGeometry;

    TubeBufferGeometry.prototype.toJSON = function () {

    	var data = BufferGeometry.prototype.toJSON.call( this );

    	data.path = this.parameters.path.toJSON();

    	return data;

    };

    /**
     * @author oosmoxiecode
     * @author Mugen87 / https://github.com/Mugen87
     *
     * based on http://www.blackpawn.com/texts/pqtorus/
     */

    // TorusKnotGeometry

    function TorusKnotGeometry( radius, tube, tubularSegments, radialSegments, p, q, heightScale ) {

    	Geometry.call( this );

    	this.type = 'TorusKnotGeometry';

    	this.parameters = {
    		radius: radius,
    		tube: tube,
    		tubularSegments: tubularSegments,
    		radialSegments: radialSegments,
    		p: p,
    		q: q
    	};

    	if ( heightScale !== undefined ) console.warn( 'THREE.TorusKnotGeometry: heightScale has been deprecated. Use .scale( x, y, z ) instead.' );

    	this.fromBufferGeometry( new TorusKnotBufferGeometry( radius, tube, tubularSegments, radialSegments, p, q ) );
    	this.mergeVertices();

    }

    TorusKnotGeometry.prototype = Object.create( Geometry.prototype );
    TorusKnotGeometry.prototype.constructor = TorusKnotGeometry;

    // TorusKnotBufferGeometry

    function TorusKnotBufferGeometry( radius, tube, tubularSegments, radialSegments, p, q ) {

    	BufferGeometry.call( this );

    	this.type = 'TorusKnotBufferGeometry';

    	this.parameters = {
    		radius: radius,
    		tube: tube,
    		tubularSegments: tubularSegments,
    		radialSegments: radialSegments,
    		p: p,
    		q: q
    	};

    	radius = radius || 1;
    	tube = tube || 0.4;
    	tubularSegments = Math.floor( tubularSegments ) || 64;
    	radialSegments = Math.floor( radialSegments ) || 8;
    	p = p || 2;
    	q = q || 3;

    	// buffers

    	var indices = [];
    	var vertices = [];
    	var normals = [];
    	var uvs = [];

    	// helper variables

    	var i, j;

    	var vertex = new Vector3();
    	var normal = new Vector3();

    	var P1 = new Vector3();
    	var P2 = new Vector3();

    	var B = new Vector3();
    	var T = new Vector3();
    	var N = new Vector3();

    	// generate vertices, normals and uvs

    	for ( i = 0; i <= tubularSegments; ++ i ) {

    		// the radian "u" is used to calculate the position on the torus curve of the current tubular segement

    		var u = i / tubularSegments * p * Math.PI * 2;

    		// now we calculate two points. P1 is our current position on the curve, P2 is a little farther ahead.
    		// these points are used to create a special "coordinate space", which is necessary to calculate the correct vertex positions

    		calculatePositionOnCurve( u, p, q, radius, P1 );
    		calculatePositionOnCurve( u + 0.01, p, q, radius, P2 );

    		// calculate orthonormal basis

    		T.subVectors( P2, P1 );
    		N.addVectors( P2, P1 );
    		B.crossVectors( T, N );
    		N.crossVectors( B, T );

    		// normalize B, N. T can be ignored, we don't use it

    		B.normalize();
    		N.normalize();

    		for ( j = 0; j <= radialSegments; ++ j ) {

    			// now calculate the vertices. they are nothing more than an extrusion of the torus curve.
    			// because we extrude a shape in the xy-plane, there is no need to calculate a z-value.

    			var v = j / radialSegments * Math.PI * 2;
    			var cx = - tube * Math.cos( v );
    			var cy = tube * Math.sin( v );

    			// now calculate the final vertex position.
    			// first we orient the extrusion with our basis vectos, then we add it to the current position on the curve

    			vertex.x = P1.x + ( cx * N.x + cy * B.x );
    			vertex.y = P1.y + ( cx * N.y + cy * B.y );
    			vertex.z = P1.z + ( cx * N.z + cy * B.z );

    			vertices.push( vertex.x, vertex.y, vertex.z );

    			// normal (P1 is always the center/origin of the extrusion, thus we can use it to calculate the normal)

    			normal.subVectors( vertex, P1 ).normalize();

    			normals.push( normal.x, normal.y, normal.z );

    			// uv

    			uvs.push( i / tubularSegments );
    			uvs.push( j / radialSegments );

    		}

    	}

    	// generate indices

    	for ( j = 1; j <= tubularSegments; j ++ ) {

    		for ( i = 1; i <= radialSegments; i ++ ) {

    			// indices

    			var a = ( radialSegments + 1 ) * ( j - 1 ) + ( i - 1 );
    			var b = ( radialSegments + 1 ) * j + ( i - 1 );
    			var c = ( radialSegments + 1 ) * j + i;
    			var d = ( radialSegments + 1 ) * ( j - 1 ) + i;

    			// faces

    			indices.push( a, b, d );
    			indices.push( b, c, d );

    		}

    	}

    	// build geometry

    	this.setIndex( indices );
    	this.addAttribute( 'position', new Float32BufferAttribute( vertices, 3 ) );
    	this.addAttribute( 'normal', new Float32BufferAttribute( normals, 3 ) );
    	this.addAttribute( 'uv', new Float32BufferAttribute( uvs, 2 ) );

    	// this function calculates the current position on the torus curve

    	function calculatePositionOnCurve( u, p, q, radius, position ) {

    		var cu = Math.cos( u );
    		var su = Math.sin( u );
    		var quOverP = q / p * u;
    		var cs = Math.cos( quOverP );

    		position.x = radius * ( 2 + cs ) * 0.5 * cu;
    		position.y = radius * ( 2 + cs ) * su * 0.5;
    		position.z = radius * Math.sin( quOverP ) * 0.5;

    	}

    }

    TorusKnotBufferGeometry.prototype = Object.create( BufferGeometry.prototype );
    TorusKnotBufferGeometry.prototype.constructor = TorusKnotBufferGeometry;

    /**
     * @author oosmoxiecode
     * @author mrdoob / http://mrdoob.com/
     * @author Mugen87 / https://github.com/Mugen87
     */

    // TorusGeometry

    function TorusGeometry( radius, tube, radialSegments, tubularSegments, arc ) {

    	Geometry.call( this );

    	this.type = 'TorusGeometry';

    	this.parameters = {
    		radius: radius,
    		tube: tube,
    		radialSegments: radialSegments,
    		tubularSegments: tubularSegments,
    		arc: arc
    	};

    	this.fromBufferGeometry( new TorusBufferGeometry( radius, tube, radialSegments, tubularSegments, arc ) );
    	this.mergeVertices();

    }

    TorusGeometry.prototype = Object.create( Geometry.prototype );
    TorusGeometry.prototype.constructor = TorusGeometry;

    // TorusBufferGeometry

    function TorusBufferGeometry( radius, tube, radialSegments, tubularSegments, arc ) {

    	BufferGeometry.call( this );

    	this.type = 'TorusBufferGeometry';

    	this.parameters = {
    		radius: radius,
    		tube: tube,
    		radialSegments: radialSegments,
    		tubularSegments: tubularSegments,
    		arc: arc
    	};

    	radius = radius || 1;
    	tube = tube || 0.4;
    	radialSegments = Math.floor( radialSegments ) || 8;
    	tubularSegments = Math.floor( tubularSegments ) || 6;
    	arc = arc || Math.PI * 2;

    	// buffers

    	var indices = [];
    	var vertices = [];
    	var normals = [];
    	var uvs = [];

    	// helper variables

    	var center = new Vector3();
    	var vertex = new Vector3();
    	var normal = new Vector3();

    	var j, i;

    	// generate vertices, normals and uvs

    	for ( j = 0; j <= radialSegments; j ++ ) {

    		for ( i = 0; i <= tubularSegments; i ++ ) {

    			var u = i / tubularSegments * arc;
    			var v = j / radialSegments * Math.PI * 2;

    			// vertex

    			vertex.x = ( radius + tube * Math.cos( v ) ) * Math.cos( u );
    			vertex.y = ( radius + tube * Math.cos( v ) ) * Math.sin( u );
    			vertex.z = tube * Math.sin( v );

    			vertices.push( vertex.x, vertex.y, vertex.z );

    			// normal

    			center.x = radius * Math.cos( u );
    			center.y = radius * Math.sin( u );
    			normal.subVectors( vertex, center ).normalize();

    			normals.push( normal.x, normal.y, normal.z );

    			// uv

    			uvs.push( i / tubularSegments );
    			uvs.push( j / radialSegments );

    		}

    	}

    	// generate indices

    	for ( j = 1; j <= radialSegments; j ++ ) {

    		for ( i = 1; i <= tubularSegments; i ++ ) {

    			// indices

    			var a = ( tubularSegments + 1 ) * j + i - 1;
    			var b = ( tubularSegments + 1 ) * ( j - 1 ) + i - 1;
    			var c = ( tubularSegments + 1 ) * ( j - 1 ) + i;
    			var d = ( tubularSegments + 1 ) * j + i;

    			// faces

    			indices.push( a, b, d );
    			indices.push( b, c, d );

    		}

    	}

    	// build geometry

    	this.setIndex( indices );
    	this.addAttribute( 'position', new Float32BufferAttribute( vertices, 3 ) );
    	this.addAttribute( 'normal', new Float32BufferAttribute( normals, 3 ) );
    	this.addAttribute( 'uv', new Float32BufferAttribute( uvs, 2 ) );

    }

    TorusBufferGeometry.prototype = Object.create( BufferGeometry.prototype );
    TorusBufferGeometry.prototype.constructor = TorusBufferGeometry;

    /**
     * @author Mugen87 / https://github.com/Mugen87
     * Port from https://github.com/mapbox/earcut (v2.1.5)
     */

    var Earcut = {

    	triangulate: function ( data, holeIndices, dim ) {

    		dim = dim || 2;

    		var hasHoles = holeIndices && holeIndices.length,
    			outerLen = hasHoles ? holeIndices[ 0 ] * dim : data.length,
    			outerNode = linkedList( data, 0, outerLen, dim, true ),
    			triangles = [];

    		if ( ! outerNode || outerNode.next === outerNode.prev ) return triangles;

    		var minX, minY, maxX, maxY, x, y, invSize;

    		if ( hasHoles ) outerNode = eliminateHoles( data, holeIndices, outerNode, dim );

    		// if the shape is not too simple, we'll use z-order curve hash later; calculate polygon bbox
    		if ( data.length > 80 * dim ) {

    			minX = maxX = data[ 0 ];
    			minY = maxY = data[ 1 ];

    			for ( var i = dim; i < outerLen; i += dim ) {

    				x = data[ i ];
    				y = data[ i + 1 ];
    				if ( x < minX ) minX = x;
    				if ( y < minY ) minY = y;
    				if ( x > maxX ) maxX = x;
    				if ( y > maxY ) maxY = y;

    			}

    			// minX, minY and invSize are later used to transform coords into integers for z-order calculation
    			invSize = Math.max( maxX - minX, maxY - minY );
    			invSize = invSize !== 0 ? 1 / invSize : 0;

    		}

    		earcutLinked( outerNode, triangles, dim, minX, minY, invSize );

    		return triangles;

    	}

    };

    // create a circular doubly linked list from polygon points in the specified winding order
    function linkedList( data, start, end, dim, clockwise ) {

    	var i, last;

    	if ( clockwise === ( signedArea( data, start, end, dim ) > 0 ) ) {

    		for ( i = start; i < end; i += dim ) last = insertNode( i, data[ i ], data[ i + 1 ], last );

    	} else {

    		for ( i = end - dim; i >= start; i -= dim ) last = insertNode( i, data[ i ], data[ i + 1 ], last );

    	}

    	if ( last && equals$1( last, last.next ) ) {

    		removeNode$1( last );
    		last = last.next;

    	}

    	return last;

    }

    // eliminate colinear or duplicate points
    function filterPoints( start, end ) {

    	if ( ! start ) return start;
    	if ( ! end ) end = start;

    	var p = start,
    		again;
    	do {

    		again = false;

    		if ( ! p.steiner && ( equals$1( p, p.next ) || area$7( p.prev, p, p.next ) === 0 ) ) {

    			removeNode$1( p );
    			p = end = p.prev;
    			if ( p === p.next ) break;
    			again = true;

    		} else {

    			p = p.next;

    		}

    	} while ( again || p !== end );

    	return end;

    }

    // main ear slicing loop which triangulates a polygon (given as a linked list)
    function earcutLinked( ear, triangles, dim, minX, minY, invSize, pass ) {

    	if ( ! ear ) return;

    	// interlink polygon nodes in z-order
    	if ( ! pass && invSize ) indexCurve( ear, minX, minY, invSize );

    	var stop = ear,
    		prev, next;

    	// iterate through ears, slicing them one by one
    	while ( ear.prev !== ear.next ) {

    		prev = ear.prev;
    		next = ear.next;

    		if ( invSize ? isEarHashed( ear, minX, minY, invSize ) : isEar( ear ) ) {

    			// cut off the triangle
    			triangles.push( prev.i / dim );
    			triangles.push( ear.i / dim );
    			triangles.push( next.i / dim );

    			removeNode$1( ear );

    			// skipping the next vertex leads to less sliver triangles
    			ear = next.next;
    			stop = next.next;

    			continue;

    		}

    		ear = next;

    		// if we looped through the whole remaining polygon and can't find any more ears
    		if ( ear === stop ) {

    			// try filtering points and slicing again
    			if ( ! pass ) {

    				earcutLinked( filterPoints( ear ), triangles, dim, minX, minY, invSize, 1 );

    				// if this didn't work, try curing all small self-intersections locally

    			} else if ( pass === 1 ) {

    				ear = cureLocalIntersections( ear, triangles, dim );
    				earcutLinked( ear, triangles, dim, minX, minY, invSize, 2 );

    				// as a last resort, try splitting the remaining polygon into two

    			} else if ( pass === 2 ) {

    				splitEarcut( ear, triangles, dim, minX, minY, invSize );

    			}

    			break;

    		}

    	}

    }

    // check whether a polygon node forms a valid ear with adjacent nodes
    function isEar( ear ) {

    	var a = ear.prev,
    		b = ear,
    		c = ear.next;

    	if ( area$7( a, b, c ) >= 0 ) return false; // reflex, can't be an ear

    	// now make sure we don't have other points inside the potential ear
    	var p = ear.next.next;

    	while ( p !== ear.prev ) {

    		if ( pointInTriangle( a.x, a.y, b.x, b.y, c.x, c.y, p.x, p.y ) &&
    			area$7( p.prev, p, p.next ) >= 0 ) return false;
    		p = p.next;

    	}

    	return true;

    }

    function isEarHashed( ear, minX, minY, invSize ) {

    	var a = ear.prev,
    		b = ear,
    		c = ear.next;

    	if ( area$7( a, b, c ) >= 0 ) return false; // reflex, can't be an ear

    	// triangle bbox; min & max are calculated like this for speed
    	var minTX = a.x < b.x ? ( a.x < c.x ? a.x : c.x ) : ( b.x < c.x ? b.x : c.x ),
    		minTY = a.y < b.y ? ( a.y < c.y ? a.y : c.y ) : ( b.y < c.y ? b.y : c.y ),
    		maxTX = a.x > b.x ? ( a.x > c.x ? a.x : c.x ) : ( b.x > c.x ? b.x : c.x ),
    		maxTY = a.y > b.y ? ( a.y > c.y ? a.y : c.y ) : ( b.y > c.y ? b.y : c.y );

    	// z-order range for the current triangle bbox;
    	var minZ = zOrder( minTX, minTY, minX, minY, invSize ),
    		maxZ = zOrder( maxTX, maxTY, minX, minY, invSize );

    	var p = ear.prevZ,
    		n = ear.nextZ;

    	// look for points inside the triangle in both directions
    	while ( p && p.z >= minZ && n && n.z <= maxZ ) {

    		if ( p !== ear.prev && p !== ear.next &&
    			pointInTriangle( a.x, a.y, b.x, b.y, c.x, c.y, p.x, p.y ) &&
    			area$7( p.prev, p, p.next ) >= 0 ) return false;
    		p = p.prevZ;

    		if ( n !== ear.prev && n !== ear.next &&
    			pointInTriangle( a.x, a.y, b.x, b.y, c.x, c.y, n.x, n.y ) &&
    			area$7( n.prev, n, n.next ) >= 0 ) return false;
    		n = n.nextZ;

    	}

    	// look for remaining points in decreasing z-order
    	while ( p && p.z >= minZ ) {

    		if ( p !== ear.prev && p !== ear.next &&
    			pointInTriangle( a.x, a.y, b.x, b.y, c.x, c.y, p.x, p.y ) &&
    			area$7( p.prev, p, p.next ) >= 0 ) return false;
    		p = p.prevZ;

    	}

    	// look for remaining points in increasing z-order
    	while ( n && n.z <= maxZ ) {

    		if ( n !== ear.prev && n !== ear.next &&
    			pointInTriangle( a.x, a.y, b.x, b.y, c.x, c.y, n.x, n.y ) &&
    			area$7( n.prev, n, n.next ) >= 0 ) return false;
    		n = n.nextZ;

    	}

    	return true;

    }

    // go through all polygon nodes and cure small local self-intersections
    function cureLocalIntersections( start, triangles, dim ) {

    	var p = start;
    	do {

    		var a = p.prev,
    			b = p.next.next;

    		if ( ! equals$1( a, b ) && intersects$2( a, p, p.next, b ) && locallyInside( a, b ) && locallyInside( b, a ) ) {

    			triangles.push( a.i / dim );
    			triangles.push( p.i / dim );
    			triangles.push( b.i / dim );

    			// remove two nodes involved
    			removeNode$1( p );
    			removeNode$1( p.next );

    			p = start = b;

    		}

    		p = p.next;

    	} while ( p !== start );

    	return p;

    }

    // try splitting polygon into two and triangulate them independently
    function splitEarcut( start, triangles, dim, minX, minY, invSize ) {

    	// look for a valid diagonal that divides the polygon into two
    	var a = start;
    	do {

    		var b = a.next.next;
    		while ( b !== a.prev ) {

    			if ( a.i !== b.i && isValidDiagonal( a, b ) ) {

    				// split the polygon in two by the diagonal
    				var c = splitPolygon( a, b );

    				// filter colinear points around the cuts
    				a = filterPoints( a, a.next );
    				c = filterPoints( c, c.next );

    				// run earcut on each half
    				earcutLinked( a, triangles, dim, minX, minY, invSize );
    				earcutLinked( c, triangles, dim, minX, minY, invSize );
    				return;

    			}

    			b = b.next;

    		}

    		a = a.next;

    	} while ( a !== start );

    }

    // link every hole into the outer loop, producing a single-ring polygon without holes
    function eliminateHoles( data, holeIndices, outerNode, dim ) {

    	var queue = [],
    		i, len, start, end, list;

    	for ( i = 0, len = holeIndices.length; i < len; i ++ ) {

    		start = holeIndices[ i ] * dim;
    		end = i < len - 1 ? holeIndices[ i + 1 ] * dim : data.length;
    		list = linkedList( data, start, end, dim, false );
    		if ( list === list.next ) list.steiner = true;
    		queue.push( getLeftmost( list ) );

    	}

    	queue.sort( compareX );

    	// process holes from left to right
    	for ( i = 0; i < queue.length; i ++ ) {

    		eliminateHole( queue[ i ], outerNode );
    		outerNode = filterPoints( outerNode, outerNode.next );

    	}

    	return outerNode;

    }

    function compareX( a, b ) {

    	return a.x - b.x;

    }

    // find a bridge between vertices that connects hole with an outer ring and and link it
    function eliminateHole( hole, outerNode ) {

    	outerNode = findHoleBridge( hole, outerNode );
    	if ( outerNode ) {

    		var b = splitPolygon( outerNode, hole );
    		filterPoints( b, b.next );

    	}

    }

    // David Eberly's algorithm for finding a bridge between hole and outer polygon
    function findHoleBridge( hole, outerNode ) {

    	var p = outerNode,
    		hx = hole.x,
    		hy = hole.y,
    		qx = - Infinity,
    		m;

    	// find a segment intersected by a ray from the hole's leftmost point to the left;
    	// segment's endpoint with lesser x will be potential connection point
    	do {

    		if ( hy <= p.y && hy >= p.next.y && p.next.y !== p.y ) {

    			var x = p.x + ( hy - p.y ) * ( p.next.x - p.x ) / ( p.next.y - p.y );
    			if ( x <= hx && x > qx ) {

    				qx = x;
    				if ( x === hx ) {

    					if ( hy === p.y ) return p;
    					if ( hy === p.next.y ) return p.next;

    				}

    				m = p.x < p.next.x ? p : p.next;

    			}

    		}

    		p = p.next;

    	} while ( p !== outerNode );

    	if ( ! m ) return null;

    	if ( hx === qx ) return m.prev; // hole touches outer segment; pick lower endpoint

    	// look for points inside the triangle of hole point, segment intersection and endpoint;
    	// if there are no points found, we have a valid connection;
    	// otherwise choose the point of the minimum angle with the ray as connection point

    	var stop = m,
    		mx = m.x,
    		my = m.y,
    		tanMin = Infinity,
    		tan;

    	p = m.next;

    	while ( p !== stop ) {

    		if ( hx >= p.x && p.x >= mx && hx !== p.x &&
    				pointInTriangle( hy < my ? hx : qx, hy, mx, my, hy < my ? qx : hx, hy, p.x, p.y ) ) {

    			tan = Math.abs( hy - p.y ) / ( hx - p.x ); // tangential

    			if ( ( tan < tanMin || ( tan === tanMin && p.x > m.x ) ) && locallyInside( p, hole ) ) {

    				m = p;
    				tanMin = tan;

    			}

    		}

    		p = p.next;

    	}

    	return m;

    }

    // interlink polygon nodes in z-order
    function indexCurve( start, minX, minY, invSize ) {

    	var p = start;
    	do {

    		if ( p.z === null ) p.z = zOrder( p.x, p.y, minX, minY, invSize );
    		p.prevZ = p.prev;
    		p.nextZ = p.next;
    		p = p.next;

    	} while ( p !== start );

    	p.prevZ.nextZ = null;
    	p.prevZ = null;

    	sortLinked( p );

    }

    // Simon Tatham's linked list merge sort algorithm
    // http://www.chiark.greenend.org.uk/~sgtatham/algorithms/listsort.html
    function sortLinked( list ) {

    	var i, p, q, e, tail, numMerges, pSize, qSize,
    		inSize = 1;

    	do {

    		p = list;
    		list = null;
    		tail = null;
    		numMerges = 0;

    		while ( p ) {

    			numMerges ++;
    			q = p;
    			pSize = 0;
    			for ( i = 0; i < inSize; i ++ ) {

    				pSize ++;
    				q = q.nextZ;
    				if ( ! q ) break;

    			}

    			qSize = inSize;

    			while ( pSize > 0 || ( qSize > 0 && q ) ) {

    				if ( pSize !== 0 && ( qSize === 0 || ! q || p.z <= q.z ) ) {

    					e = p;
    					p = p.nextZ;
    					pSize --;

    				} else {

    					e = q;
    					q = q.nextZ;
    					qSize --;

    				}

    				if ( tail ) tail.nextZ = e;
    				else list = e;

    				e.prevZ = tail;
    				tail = e;

    			}

    			p = q;

    		}

    		tail.nextZ = null;
    		inSize *= 2;

    	} while ( numMerges > 1 );

    	return list;

    }

    // z-order of a point given coords and inverse of the longer side of data bbox
    function zOrder( x, y, minX, minY, invSize ) {

    	// coords are transformed into non-negative 15-bit integer range
    	x = 32767 * ( x - minX ) * invSize;
    	y = 32767 * ( y - minY ) * invSize;

    	x = ( x | ( x << 8 ) ) & 0x00FF00FF;
    	x = ( x | ( x << 4 ) ) & 0x0F0F0F0F;
    	x = ( x | ( x << 2 ) ) & 0x33333333;
    	x = ( x | ( x << 1 ) ) & 0x55555555;

    	y = ( y | ( y << 8 ) ) & 0x00FF00FF;
    	y = ( y | ( y << 4 ) ) & 0x0F0F0F0F;
    	y = ( y | ( y << 2 ) ) & 0x33333333;
    	y = ( y | ( y << 1 ) ) & 0x55555555;

    	return x | ( y << 1 );

    }

    // find the leftmost node of a polygon ring
    function getLeftmost( start ) {

    	var p = start,
    		leftmost = start;
    	do {

    		if ( p.x < leftmost.x || ( p.x === leftmost.x && p.y < leftmost.y ) ) leftmost = p;
    		p = p.next;

    	} while ( p !== start );

    	return leftmost;

    }

    // check if a point lies within a convex triangle
    function pointInTriangle( ax, ay, bx, by, cx, cy, px, py ) {

    	return ( cx - px ) * ( ay - py ) - ( ax - px ) * ( cy - py ) >= 0 &&
    		   ( ax - px ) * ( by - py ) - ( bx - px ) * ( ay - py ) >= 0 &&
    		   ( bx - px ) * ( cy - py ) - ( cx - px ) * ( by - py ) >= 0;

    }

    // check if a diagonal between two polygon nodes is valid (lies in polygon interior)
    function isValidDiagonal( a, b ) {

    	return a.next.i !== b.i && a.prev.i !== b.i && ! intersectsPolygon( a, b ) &&
    		   locallyInside( a, b ) && locallyInside( b, a ) && middleInside( a, b );

    }

    // signed area of a triangle
    function area$7( p, q, r ) {

    	return ( q.y - p.y ) * ( r.x - q.x ) - ( q.x - p.x ) * ( r.y - q.y );

    }

    // check if two points are equal
    function equals$1( p1, p2 ) {

    	return p1.x === p2.x && p1.y === p2.y;

    }

    // check if two segments intersect
    function intersects$2( p1, q1, p2, q2 ) {

    	if ( ( equals$1( p1, p2 ) && equals$1( q1, q2 ) ) ||
    		( equals$1( p1, q2 ) && equals$1( p2, q1 ) ) ) return true;
    	return area$7( p1, q1, p2 ) > 0 !== area$7( p1, q1, q2 ) > 0 &&
    		   area$7( p2, q2, p1 ) > 0 !== area$7( p2, q2, q1 ) > 0;

    }

    // check if a polygon diagonal intersects any polygon segments
    function intersectsPolygon( a, b ) {

    	var p = a;
    	do {

    		if ( p.i !== a.i && p.next.i !== a.i && p.i !== b.i && p.next.i !== b.i &&
    				intersects$2( p, p.next, a, b ) ) return true;
    		p = p.next;

    	} while ( p !== a );

    	return false;

    }

    // check if a polygon diagonal is locally inside the polygon
    function locallyInside( a, b ) {

    	return area$7( a.prev, a, a.next ) < 0 ?
    		area$7( a, b, a.next ) >= 0 && area$7( a, a.prev, b ) >= 0 :
    		area$7( a, b, a.prev ) < 0 || area$7( a, a.next, b ) < 0;

    }

    // check if the middle point of a polygon diagonal is inside the polygon
    function middleInside( a, b ) {

    	var p = a,
    		inside = false,
    		px = ( a.x + b.x ) / 2,
    		py = ( a.y + b.y ) / 2;
    	do {

    		if ( ( ( p.y > py ) !== ( p.next.y > py ) ) && p.next.y !== p.y &&
    				( px < ( p.next.x - p.x ) * ( py - p.y ) / ( p.next.y - p.y ) + p.x ) )
    			inside = ! inside;
    		p = p.next;

    	} while ( p !== a );

    	return inside;

    }

    // link two polygon vertices with a bridge; if the vertices belong to the same ring, it splits polygon into two;
    // if one belongs to the outer ring and another to a hole, it merges it into a single ring
    function splitPolygon( a, b ) {

    	var a2 = new Node$5( a.i, a.x, a.y ),
    		b2 = new Node$5( b.i, b.x, b.y ),
    		an = a.next,
    		bp = b.prev;

    	a.next = b;
    	b.prev = a;

    	a2.next = an;
    	an.prev = a2;

    	b2.next = a2;
    	a2.prev = b2;

    	bp.next = b2;
    	b2.prev = bp;

    	return b2;

    }

    // create a node and optionally link it with previous one (in a circular doubly linked list)
    function insertNode( i, x, y, last ) {

    	var p = new Node$5( i, x, y );

    	if ( ! last ) {

    		p.prev = p;
    		p.next = p;

    	} else {

    		p.next = last.next;
    		p.prev = last;
    		last.next.prev = p;
    		last.next = p;

    	}

    	return p;

    }

    function removeNode$1( p ) {

    	p.next.prev = p.prev;
    	p.prev.next = p.next;

    	if ( p.prevZ ) p.prevZ.nextZ = p.nextZ;
    	if ( p.nextZ ) p.nextZ.prevZ = p.prevZ;

    }

    function Node$5( i, x, y ) {

    	// vertex index in coordinates array
    	this.i = i;

    	// vertex coordinates
    	this.x = x;
    	this.y = y;

    	// previous and next vertex nodes in a polygon ring
    	this.prev = null;
    	this.next = null;

    	// z-order curve value
    	this.z = null;

    	// previous and next nodes in z-order
    	this.prevZ = null;
    	this.nextZ = null;

    	// indicates whether this is a steiner point
    	this.steiner = false;

    }

    function signedArea( data, start, end, dim ) {

    	var sum = 0;
    	for ( var i = start, j = end - dim; i < end; i += dim ) {

    		sum += ( data[ j ] - data[ i ] ) * ( data[ i + 1 ] + data[ j + 1 ] );
    		j = i;

    	}

    	return sum;

    }

    /**
     * @author zz85 / http://www.lab4games.net/zz85/blog
     */

    var ShapeUtils = {

    	// calculate area of the contour polygon

    	area: function ( contour ) {

    		var n = contour.length;
    		var a = 0.0;

    		for ( var p = n - 1, q = 0; q < n; p = q ++ ) {

    			a += contour[ p ].x * contour[ q ].y - contour[ q ].x * contour[ p ].y;

    		}

    		return a * 0.5;

    	},

    	isClockWise: function ( pts ) {

    		return ShapeUtils.area( pts ) < 0;

    	},

    	triangulateShape: function ( contour, holes ) {

    		var vertices = []; // flat array of vertices like [ x0,y0, x1,y1, x2,y2, ... ]
    		var holeIndices = []; // array of hole indices
    		var faces = []; // final array of vertex indices like [ [ a,b,d ], [ b,c,d ] ]

    		removeDupEndPts( contour );
    		addContour( vertices, contour );

    		//

    		var holeIndex = contour.length;

    		holes.forEach( removeDupEndPts );

    		for ( var i = 0; i < holes.length; i ++ ) {

    			holeIndices.push( holeIndex );
    			holeIndex += holes[ i ].length;
    			addContour( vertices, holes[ i ] );

    		}

    		//

    		var triangles = Earcut.triangulate( vertices, holeIndices );

    		//

    		for ( var i = 0; i < triangles.length; i += 3 ) {

    			faces.push( triangles.slice( i, i + 3 ) );

    		}

    		return faces;

    	}

    };

    function removeDupEndPts( points ) {

    	var l = points.length;

    	if ( l > 2 && points[ l - 1 ].equals( points[ 0 ] ) ) {

    		points.pop();

    	}

    }

    function addContour( vertices, contour ) {

    	for ( var i = 0; i < contour.length; i ++ ) {

    		vertices.push( contour[ i ].x );
    		vertices.push( contour[ i ].y );

    	}

    }

    /**
     * @author zz85 / http://www.lab4games.net/zz85/blog
     *
     * Creates extruded geometry from a path shape.
     *
     * parameters = {
     *
     *  curveSegments: <int>, // number of points on the curves
     *  steps: <int>, // number of points for z-side extrusions / used for subdividing segments of extrude spline too
     *  depth: <float>, // Depth to extrude the shape
     *
     *  bevelEnabled: <bool>, // turn on bevel
     *  bevelThickness: <float>, // how deep into the original shape bevel goes
     *  bevelSize: <float>, // how far from shape outline (including bevelOffset) is bevel
     *  bevelOffset: <float>, // how far from shape outline does bevel start
     *  bevelSegments: <int>, // number of bevel layers
     *
     *  extrudePath: <THREE.Curve> // curve to extrude shape along
     *
     *  UVGenerator: <Object> // object that provides UV generator functions
     *
     * }
     */

    // ExtrudeGeometry

    function ExtrudeGeometry( shapes, options ) {

    	Geometry.call( this );

    	this.type = 'ExtrudeGeometry';

    	this.parameters = {
    		shapes: shapes,
    		options: options
    	};

    	this.fromBufferGeometry( new ExtrudeBufferGeometry( shapes, options ) );
    	this.mergeVertices();

    }

    ExtrudeGeometry.prototype = Object.create( Geometry.prototype );
    ExtrudeGeometry.prototype.constructor = ExtrudeGeometry;

    ExtrudeGeometry.prototype.toJSON = function () {

    	var data = Geometry.prototype.toJSON.call( this );

    	var shapes = this.parameters.shapes;
    	var options = this.parameters.options;

    	return toJSON( shapes, options, data );

    };

    // ExtrudeBufferGeometry

    function ExtrudeBufferGeometry( shapes, options ) {

    	BufferGeometry.call( this );

    	this.type = 'ExtrudeBufferGeometry';

    	this.parameters = {
    		shapes: shapes,
    		options: options
    	};

    	shapes = Array.isArray( shapes ) ? shapes : [ shapes ];

    	var scope = this;

    	var verticesArray = [];
    	var uvArray = [];

    	for ( var i = 0, l = shapes.length; i < l; i ++ ) {

    		var shape = shapes[ i ];
    		addShape( shape );

    	}

    	// build geometry

    	this.addAttribute( 'position', new Float32BufferAttribute( verticesArray, 3 ) );
    	this.addAttribute( 'uv', new Float32BufferAttribute( uvArray, 2 ) );

    	this.computeVertexNormals();

    	// functions

    	function addShape( shape ) {

    		var placeholder = [];

    		// options

    		var curveSegments = options.curveSegments !== undefined ? options.curveSegments : 12;
    		var steps = options.steps !== undefined ? options.steps : 1;
    		var depth = options.depth !== undefined ? options.depth : 100;

    		var bevelEnabled = options.bevelEnabled !== undefined ? options.bevelEnabled : true;
    		var bevelThickness = options.bevelThickness !== undefined ? options.bevelThickness : 6;
    		var bevelSize = options.bevelSize !== undefined ? options.bevelSize : bevelThickness - 2;
    		var bevelOffset = options.bevelOffset !== undefined ? options.bevelOffset : 0;
    		var bevelSegments = options.bevelSegments !== undefined ? options.bevelSegments : 3;

    		var extrudePath = options.extrudePath;

    		var uvgen = options.UVGenerator !== undefined ? options.UVGenerator : WorldUVGenerator;

    		// deprecated options

    		if ( options.amount !== undefined ) {

    			console.warn( 'THREE.ExtrudeBufferGeometry: amount has been renamed to depth.' );
    			depth = options.amount;

    		}

    		//

    		var extrudePts, extrudeByPath = false;
    		var splineTube, binormal, normal, position2;

    		if ( extrudePath ) {

    			extrudePts = extrudePath.getSpacedPoints( steps );

    			extrudeByPath = true;
    			bevelEnabled = false; // bevels not supported for path extrusion

    			// SETUP TNB variables

    			// TODO1 - have a .isClosed in spline?

    			splineTube = extrudePath.computeFrenetFrames( steps, false );

    			// console.log(splineTube, 'splineTube', splineTube.normals.length, 'steps', steps, 'extrudePts', extrudePts.length);

    			binormal = new Vector3();
    			normal = new Vector3();
    			position2 = new Vector3();

    		}

    		// Safeguards if bevels are not enabled

    		if ( ! bevelEnabled ) {

    			bevelSegments = 0;
    			bevelThickness = 0;
    			bevelSize = 0;
    			bevelOffset = 0;

    		}

    		// Variables initialization

    		var ahole, h, hl; // looping of holes

    		var shapePoints = shape.extractPoints( curveSegments );

    		var vertices = shapePoints.shape;
    		var holes = shapePoints.holes;

    		var reverse = ! ShapeUtils.isClockWise( vertices );

    		if ( reverse ) {

    			vertices = vertices.reverse();

    			// Maybe we should also check if holes are in the opposite direction, just to be safe ...

    			for ( h = 0, hl = holes.length; h < hl; h ++ ) {

    				ahole = holes[ h ];

    				if ( ShapeUtils.isClockWise( ahole ) ) {

    					holes[ h ] = ahole.reverse();

    				}

    			}

    		}


    		var faces = ShapeUtils.triangulateShape( vertices, holes );

    		/* Vertices */

    		var contour = vertices; // vertices has all points but contour has only points of circumference

    		for ( h = 0, hl = holes.length; h < hl; h ++ ) {

    			ahole = holes[ h ];

    			vertices = vertices.concat( ahole );

    		}


    		function scalePt2( pt, vec, size ) {

    			if ( ! vec ) console.error( "THREE.ExtrudeGeometry: vec does not exist" );

    			return vec.clone().multiplyScalar( size ).add( pt );

    		}

    		var b, bs, t, z,
    			vert, vlen = vertices.length,
    			face, flen = faces.length;


    		// Find directions for point movement


    		function getBevelVec( inPt, inPrev, inNext ) {

    			// computes for inPt the corresponding point inPt' on a new contour
    			//   shifted by 1 unit (length of normalized vector) to the left
    			// if we walk along contour clockwise, this new contour is outside the old one
    			//
    			// inPt' is the intersection of the two lines parallel to the two
    			//  adjacent edges of inPt at a distance of 1 unit on the left side.

    			var v_trans_x, v_trans_y, shrink_by; // resulting translation vector for inPt

    			// good reading for geometry algorithms (here: line-line intersection)
    			// http://geomalgorithms.com/a05-_intersect-1.html

    			var v_prev_x = inPt.x - inPrev.x,
    				v_prev_y = inPt.y - inPrev.y;
    			var v_next_x = inNext.x - inPt.x,
    				v_next_y = inNext.y - inPt.y;

    			var v_prev_lensq = ( v_prev_x * v_prev_x + v_prev_y * v_prev_y );

    			// check for collinear edges
    			var collinear0 = ( v_prev_x * v_next_y - v_prev_y * v_next_x );

    			if ( Math.abs( collinear0 ) > Number.EPSILON ) {

    				// not collinear

    				// length of vectors for normalizing

    				var v_prev_len = Math.sqrt( v_prev_lensq );
    				var v_next_len = Math.sqrt( v_next_x * v_next_x + v_next_y * v_next_y );

    				// shift adjacent points by unit vectors to the left

    				var ptPrevShift_x = ( inPrev.x - v_prev_y / v_prev_len );
    				var ptPrevShift_y = ( inPrev.y + v_prev_x / v_prev_len );

    				var ptNextShift_x = ( inNext.x - v_next_y / v_next_len );
    				var ptNextShift_y = ( inNext.y + v_next_x / v_next_len );

    				// scaling factor for v_prev to intersection point

    				var sf = ( ( ptNextShift_x - ptPrevShift_x ) * v_next_y -
    						( ptNextShift_y - ptPrevShift_y ) * v_next_x ) /
    					( v_prev_x * v_next_y - v_prev_y * v_next_x );

    				// vector from inPt to intersection point

    				v_trans_x = ( ptPrevShift_x + v_prev_x * sf - inPt.x );
    				v_trans_y = ( ptPrevShift_y + v_prev_y * sf - inPt.y );

    				// Don't normalize!, otherwise sharp corners become ugly
    				//  but prevent crazy spikes
    				var v_trans_lensq = ( v_trans_x * v_trans_x + v_trans_y * v_trans_y );
    				if ( v_trans_lensq <= 2 ) {

    					return new Vector2( v_trans_x, v_trans_y );

    				} else {

    					shrink_by = Math.sqrt( v_trans_lensq / 2 );

    				}

    			} else {

    				// handle special case of collinear edges

    				var direction_eq = false; // assumes: opposite
    				if ( v_prev_x > Number.EPSILON ) {

    					if ( v_next_x > Number.EPSILON ) {

    						direction_eq = true;

    					}

    				} else {

    					if ( v_prev_x < - Number.EPSILON ) {

    						if ( v_next_x < - Number.EPSILON ) {

    							direction_eq = true;

    						}

    					} else {

    						if ( Math.sign( v_prev_y ) === Math.sign( v_next_y ) ) {

    							direction_eq = true;

    						}

    					}

    				}

    				if ( direction_eq ) {

    					// console.log("Warning: lines are a straight sequence");
    					v_trans_x = - v_prev_y;
    					v_trans_y = v_prev_x;
    					shrink_by = Math.sqrt( v_prev_lensq );

    				} else {

    					// console.log("Warning: lines are a straight spike");
    					v_trans_x = v_prev_x;
    					v_trans_y = v_prev_y;
    					shrink_by = Math.sqrt( v_prev_lensq / 2 );

    				}

    			}

    			return new Vector2( v_trans_x / shrink_by, v_trans_y / shrink_by );

    		}


    		var contourMovements = [];

    		for ( var i = 0, il = contour.length, j = il - 1, k = i + 1; i < il; i ++, j ++, k ++ ) {

    			if ( j === il ) j = 0;
    			if ( k === il ) k = 0;

    			//  (j)---(i)---(k)
    			// console.log('i,j,k', i, j , k)

    			contourMovements[ i ] = getBevelVec( contour[ i ], contour[ j ], contour[ k ] );

    		}

    		var holesMovements = [],
    			oneHoleMovements, verticesMovements = contourMovements.concat();

    		for ( h = 0, hl = holes.length; h < hl; h ++ ) {

    			ahole = holes[ h ];

    			oneHoleMovements = [];

    			for ( i = 0, il = ahole.length, j = il - 1, k = i + 1; i < il; i ++, j ++, k ++ ) {

    				if ( j === il ) j = 0;
    				if ( k === il ) k = 0;

    				//  (j)---(i)---(k)
    				oneHoleMovements[ i ] = getBevelVec( ahole[ i ], ahole[ j ], ahole[ k ] );

    			}

    			holesMovements.push( oneHoleMovements );
    			verticesMovements = verticesMovements.concat( oneHoleMovements );

    		}


    		// Loop bevelSegments, 1 for the front, 1 for the back

    		for ( b = 0; b < bevelSegments; b ++ ) {

    			//for ( b = bevelSegments; b > 0; b -- ) {

    			t = b / bevelSegments;
    			z = bevelThickness * Math.cos( t * Math.PI / 2 );
    			bs = bevelSize * Math.sin( t * Math.PI / 2 ) + bevelOffset;

    			// contract shape

    			for ( i = 0, il = contour.length; i < il; i ++ ) {

    				vert = scalePt2( contour[ i ], contourMovements[ i ], bs );

    				v( vert.x, vert.y, - z );

    			}

    			// expand holes

    			for ( h = 0, hl = holes.length; h < hl; h ++ ) {

    				ahole = holes[ h ];
    				oneHoleMovements = holesMovements[ h ];

    				for ( i = 0, il = ahole.length; i < il; i ++ ) {

    					vert = scalePt2( ahole[ i ], oneHoleMovements[ i ], bs );

    					v( vert.x, vert.y, - z );

    				}

    			}

    		}

    		bs = bevelSize + bevelOffset;

    		// Back facing vertices

    		for ( i = 0; i < vlen; i ++ ) {

    			vert = bevelEnabled ? scalePt2( vertices[ i ], verticesMovements[ i ], bs ) : vertices[ i ];

    			if ( ! extrudeByPath ) {

    				v( vert.x, vert.y, 0 );

    			} else {

    				// v( vert.x, vert.y + extrudePts[ 0 ].y, extrudePts[ 0 ].x );

    				normal.copy( splineTube.normals[ 0 ] ).multiplyScalar( vert.x );
    				binormal.copy( splineTube.binormals[ 0 ] ).multiplyScalar( vert.y );

    				position2.copy( extrudePts[ 0 ] ).add( normal ).add( binormal );

    				v( position2.x, position2.y, position2.z );

    			}

    		}

    		// Add stepped vertices...
    		// Including front facing vertices

    		var s;

    		for ( s = 1; s <= steps; s ++ ) {

    			for ( i = 0; i < vlen; i ++ ) {

    				vert = bevelEnabled ? scalePt2( vertices[ i ], verticesMovements[ i ], bs ) : vertices[ i ];

    				if ( ! extrudeByPath ) {

    					v( vert.x, vert.y, depth / steps * s );

    				} else {

    					// v( vert.x, vert.y + extrudePts[ s - 1 ].y, extrudePts[ s - 1 ].x );

    					normal.copy( splineTube.normals[ s ] ).multiplyScalar( vert.x );
    					binormal.copy( splineTube.binormals[ s ] ).multiplyScalar( vert.y );

    					position2.copy( extrudePts[ s ] ).add( normal ).add( binormal );

    					v( position2.x, position2.y, position2.z );

    				}

    			}

    		}


    		// Add bevel segments planes

    		//for ( b = 1; b <= bevelSegments; b ++ ) {
    		for ( b = bevelSegments - 1; b >= 0; b -- ) {

    			t = b / bevelSegments;
    			z = bevelThickness * Math.cos( t * Math.PI / 2 );
    			bs = bevelSize * Math.sin( t * Math.PI / 2 ) + bevelOffset;

    			// contract shape

    			for ( i = 0, il = contour.length; i < il; i ++ ) {

    				vert = scalePt2( contour[ i ], contourMovements[ i ], bs );
    				v( vert.x, vert.y, depth + z );

    			}

    			// expand holes

    			for ( h = 0, hl = holes.length; h < hl; h ++ ) {

    				ahole = holes[ h ];
    				oneHoleMovements = holesMovements[ h ];

    				for ( i = 0, il = ahole.length; i < il; i ++ ) {

    					vert = scalePt2( ahole[ i ], oneHoleMovements[ i ], bs );

    					if ( ! extrudeByPath ) {

    						v( vert.x, vert.y, depth + z );

    					} else {

    						v( vert.x, vert.y + extrudePts[ steps - 1 ].y, extrudePts[ steps - 1 ].x + z );

    					}

    				}

    			}

    		}

    		/* Faces */

    		// Top and bottom faces

    		buildLidFaces();

    		// Sides faces

    		buildSideFaces();


    		/////  Internal functions

    		function buildLidFaces() {

    			var start = verticesArray.length / 3;

    			if ( bevelEnabled ) {

    				var layer = 0; // steps + 1
    				var offset = vlen * layer;

    				// Bottom faces

    				for ( i = 0; i < flen; i ++ ) {

    					face = faces[ i ];
    					f3( face[ 2 ] + offset, face[ 1 ] + offset, face[ 0 ] + offset );

    				}

    				layer = steps + bevelSegments * 2;
    				offset = vlen * layer;

    				// Top faces

    				for ( i = 0; i < flen; i ++ ) {

    					face = faces[ i ];
    					f3( face[ 0 ] + offset, face[ 1 ] + offset, face[ 2 ] + offset );

    				}

    			} else {

    				// Bottom faces

    				for ( i = 0; i < flen; i ++ ) {

    					face = faces[ i ];
    					f3( face[ 2 ], face[ 1 ], face[ 0 ] );

    				}

    				// Top faces

    				for ( i = 0; i < flen; i ++ ) {

    					face = faces[ i ];
    					f3( face[ 0 ] + vlen * steps, face[ 1 ] + vlen * steps, face[ 2 ] + vlen * steps );

    				}

    			}

    			scope.addGroup( start, verticesArray.length / 3 - start, 0 );

    		}

    		// Create faces for the z-sides of the shape

    		function buildSideFaces() {

    			var start = verticesArray.length / 3;
    			var layeroffset = 0;
    			sidewalls( contour, layeroffset );
    			layeroffset += contour.length;

    			for ( h = 0, hl = holes.length; h < hl; h ++ ) {

    				ahole = holes[ h ];
    				sidewalls( ahole, layeroffset );

    				//, true
    				layeroffset += ahole.length;

    			}


    			scope.addGroup( start, verticesArray.length / 3 - start, 1 );


    		}

    		function sidewalls( contour, layeroffset ) {

    			var j, k;
    			i = contour.length;

    			while ( -- i >= 0 ) {

    				j = i;
    				k = i - 1;
    				if ( k < 0 ) k = contour.length - 1;

    				//console.log('b', i,j, i-1, k,vertices.length);

    				var s = 0,
    					sl = steps + bevelSegments * 2;

    				for ( s = 0; s < sl; s ++ ) {

    					var slen1 = vlen * s;
    					var slen2 = vlen * ( s + 1 );

    					var a = layeroffset + j + slen1,
    						b = layeroffset + k + slen1,
    						c = layeroffset + k + slen2,
    						d = layeroffset + j + slen2;

    					f4( a, b, c, d );

    				}

    			}

    		}

    		function v( x, y, z ) {

    			placeholder.push( x );
    			placeholder.push( y );
    			placeholder.push( z );

    		}


    		function f3( a, b, c ) {

    			addVertex( a );
    			addVertex( b );
    			addVertex( c );

    			var nextIndex = verticesArray.length / 3;
    			var uvs = uvgen.generateTopUV( scope, verticesArray, nextIndex - 3, nextIndex - 2, nextIndex - 1 );

    			addUV( uvs[ 0 ] );
    			addUV( uvs[ 1 ] );
    			addUV( uvs[ 2 ] );

    		}

    		function f4( a, b, c, d ) {

    			addVertex( a );
    			addVertex( b );
    			addVertex( d );

    			addVertex( b );
    			addVertex( c );
    			addVertex( d );


    			var nextIndex = verticesArray.length / 3;
    			var uvs = uvgen.generateSideWallUV( scope, verticesArray, nextIndex - 6, nextIndex - 3, nextIndex - 2, nextIndex - 1 );

    			addUV( uvs[ 0 ] );
    			addUV( uvs[ 1 ] );
    			addUV( uvs[ 3 ] );

    			addUV( uvs[ 1 ] );
    			addUV( uvs[ 2 ] );
    			addUV( uvs[ 3 ] );

    		}

    		function addVertex( index ) {

    			verticesArray.push( placeholder[ index * 3 + 0 ] );
    			verticesArray.push( placeholder[ index * 3 + 1 ] );
    			verticesArray.push( placeholder[ index * 3 + 2 ] );

    		}


    		function addUV( vector2 ) {

    			uvArray.push( vector2.x );
    			uvArray.push( vector2.y );

    		}

    	}

    }

    ExtrudeBufferGeometry.prototype = Object.create( BufferGeometry.prototype );
    ExtrudeBufferGeometry.prototype.constructor = ExtrudeBufferGeometry;

    ExtrudeBufferGeometry.prototype.toJSON = function () {

    	var data = BufferGeometry.prototype.toJSON.call( this );

    	var shapes = this.parameters.shapes;
    	var options = this.parameters.options;

    	return toJSON( shapes, options, data );

    };

    //

    var WorldUVGenerator = {

    	generateTopUV: function ( geometry, vertices, indexA, indexB, indexC ) {

    		var a_x = vertices[ indexA * 3 ];
    		var a_y = vertices[ indexA * 3 + 1 ];
    		var b_x = vertices[ indexB * 3 ];
    		var b_y = vertices[ indexB * 3 + 1 ];
    		var c_x = vertices[ indexC * 3 ];
    		var c_y = vertices[ indexC * 3 + 1 ];

    		return [
    			new Vector2( a_x, a_y ),
    			new Vector2( b_x, b_y ),
    			new Vector2( c_x, c_y )
    		];

    	},

    	generateSideWallUV: function ( geometry, vertices, indexA, indexB, indexC, indexD ) {

    		var a_x = vertices[ indexA * 3 ];
    		var a_y = vertices[ indexA * 3 + 1 ];
    		var a_z = vertices[ indexA * 3 + 2 ];
    		var b_x = vertices[ indexB * 3 ];
    		var b_y = vertices[ indexB * 3 + 1 ];
    		var b_z = vertices[ indexB * 3 + 2 ];
    		var c_x = vertices[ indexC * 3 ];
    		var c_y = vertices[ indexC * 3 + 1 ];
    		var c_z = vertices[ indexC * 3 + 2 ];
    		var d_x = vertices[ indexD * 3 ];
    		var d_y = vertices[ indexD * 3 + 1 ];
    		var d_z = vertices[ indexD * 3 + 2 ];

    		if ( Math.abs( a_y - b_y ) < 0.01 ) {

    			return [
    				new Vector2( a_x, 1 - a_z ),
    				new Vector2( b_x, 1 - b_z ),
    				new Vector2( c_x, 1 - c_z ),
    				new Vector2( d_x, 1 - d_z )
    			];

    		} else {

    			return [
    				new Vector2( a_y, 1 - a_z ),
    				new Vector2( b_y, 1 - b_z ),
    				new Vector2( c_y, 1 - c_z ),
    				new Vector2( d_y, 1 - d_z )
    			];

    		}

    	}
    };

    function toJSON( shapes, options, data ) {

    	//

    	data.shapes = [];

    	if ( Array.isArray( shapes ) ) {

    		for ( var i = 0, l = shapes.length; i < l; i ++ ) {

    			var shape = shapes[ i ];

    			data.shapes.push( shape.uuid );

    		}

    	} else {

    		data.shapes.push( shapes.uuid );

    	}

    	//

    	if ( options.extrudePath !== undefined ) data.options.extrudePath = options.extrudePath.toJSON();

    	return data;

    }

    /**
     * @author zz85 / http://www.lab4games.net/zz85/blog
     * @author alteredq / http://alteredqualia.com/
     *
     * Text = 3D Text
     *
     * parameters = {
     *  font: <THREE.Font>, // font
     *
     *  size: <float>, // size of the text
     *  height: <float>, // thickness to extrude text
     *  curveSegments: <int>, // number of points on the curves
     *
     *  bevelEnabled: <bool>, // turn on bevel
     *  bevelThickness: <float>, // how deep into text bevel goes
     *  bevelSize: <float>, // how far from text outline (including bevelOffset) is bevel
     *  bevelOffset: <float> // how far from text outline does bevel start
     * }
     */

    // TextGeometry

    function TextGeometry( text, parameters ) {

    	Geometry.call( this );

    	this.type = 'TextGeometry';

    	this.parameters = {
    		text: text,
    		parameters: parameters
    	};

    	this.fromBufferGeometry( new TextBufferGeometry( text, parameters ) );
    	this.mergeVertices();

    }

    TextGeometry.prototype = Object.create( Geometry.prototype );
    TextGeometry.prototype.constructor = TextGeometry;

    // TextBufferGeometry

    function TextBufferGeometry( text, parameters ) {

    	parameters = parameters || {};

    	var font = parameters.font;

    	if ( ! ( font && font.isFont ) ) {

    		console.error( 'THREE.TextGeometry: font parameter is not an instance of THREE.Font.' );
    		return new Geometry();

    	}

    	var shapes = font.generateShapes( text, parameters.size );

    	// translate parameters to ExtrudeGeometry API

    	parameters.depth = parameters.height !== undefined ? parameters.height : 50;

    	// defaults

    	if ( parameters.bevelThickness === undefined ) parameters.bevelThickness = 10;
    	if ( parameters.bevelSize === undefined ) parameters.bevelSize = 8;
    	if ( parameters.bevelEnabled === undefined ) parameters.bevelEnabled = false;

    	ExtrudeBufferGeometry.call( this, shapes, parameters );

    	this.type = 'TextBufferGeometry';

    }

    TextBufferGeometry.prototype = Object.create( ExtrudeBufferGeometry.prototype );
    TextBufferGeometry.prototype.constructor = TextBufferGeometry;

    /**
     * @author mrdoob / http://mrdoob.com/
     * @author benaadams / https://twitter.com/ben_a_adams
     * @author Mugen87 / https://github.com/Mugen87
     */

    // SphereGeometry

    function SphereGeometry( radius, widthSegments, heightSegments, phiStart, phiLength, thetaStart, thetaLength ) {

    	Geometry.call( this );

    	this.type = 'SphereGeometry';

    	this.parameters = {
    		radius: radius,
    		widthSegments: widthSegments,
    		heightSegments: heightSegments,
    		phiStart: phiStart,
    		phiLength: phiLength,
    		thetaStart: thetaStart,
    		thetaLength: thetaLength
    	};

    	this.fromBufferGeometry( new SphereBufferGeometry( radius, widthSegments, heightSegments, phiStart, phiLength, thetaStart, thetaLength ) );
    	this.mergeVertices();

    }

    SphereGeometry.prototype = Object.create( Geometry.prototype );
    SphereGeometry.prototype.constructor = SphereGeometry;

    // SphereBufferGeometry

    function SphereBufferGeometry( radius, widthSegments, heightSegments, phiStart, phiLength, thetaStart, thetaLength ) {

    	BufferGeometry.call( this );

    	this.type = 'SphereBufferGeometry';

    	this.parameters = {
    		radius: radius,
    		widthSegments: widthSegments,
    		heightSegments: heightSegments,
    		phiStart: phiStart,
    		phiLength: phiLength,
    		thetaStart: thetaStart,
    		thetaLength: thetaLength
    	};

    	radius = radius || 1;

    	widthSegments = Math.max( 3, Math.floor( widthSegments ) || 8 );
    	heightSegments = Math.max( 2, Math.floor( heightSegments ) || 6 );

    	phiStart = phiStart !== undefined ? phiStart : 0;
    	phiLength = phiLength !== undefined ? phiLength : Math.PI * 2;

    	thetaStart = thetaStart !== undefined ? thetaStart : 0;
    	thetaLength = thetaLength !== undefined ? thetaLength : Math.PI;

    	var thetaEnd = Math.min( thetaStart + thetaLength, Math.PI );

    	var ix, iy;

    	var index = 0;
    	var grid = [];

    	var vertex = new Vector3();
    	var normal = new Vector3();

    	// buffers

    	var indices = [];
    	var vertices = [];
    	var normals = [];
    	var uvs = [];

    	// generate vertices, normals and uvs

    	for ( iy = 0; iy <= heightSegments; iy ++ ) {

    		var verticesRow = [];

    		var v = iy / heightSegments;

    		// special case for the poles

    		var uOffset = 0;

    		if ( iy == 0 && thetaStart == 0 ) {

    			uOffset = 0.5 / widthSegments;

    		} else if ( iy == heightSegments && thetaEnd == Math.PI ) {

    			uOffset = - 0.5 / widthSegments;

    		}

    		for ( ix = 0; ix <= widthSegments; ix ++ ) {

    			var u = ix / widthSegments;

    			// vertex

    			vertex.x = - radius * Math.cos( phiStart + u * phiLength ) * Math.sin( thetaStart + v * thetaLength );
    			vertex.y = radius * Math.cos( thetaStart + v * thetaLength );
    			vertex.z = radius * Math.sin( phiStart + u * phiLength ) * Math.sin( thetaStart + v * thetaLength );

    			vertices.push( vertex.x, vertex.y, vertex.z );

    			// normal

    			normal.copy( vertex ).normalize();
    			normals.push( normal.x, normal.y, normal.z );

    			// uv

    			uvs.push( u + uOffset, 1 - v );

    			verticesRow.push( index ++ );

    		}

    		grid.push( verticesRow );

    	}

    	// indices

    	for ( iy = 0; iy < heightSegments; iy ++ ) {

    		for ( ix = 0; ix < widthSegments; ix ++ ) {

    			var a = grid[ iy ][ ix + 1 ];
    			var b = grid[ iy ][ ix ];
    			var c = grid[ iy + 1 ][ ix ];
    			var d = grid[ iy + 1 ][ ix + 1 ];

    			if ( iy !== 0 || thetaStart > 0 ) indices.push( a, b, d );
    			if ( iy !== heightSegments - 1 || thetaEnd < Math.PI ) indices.push( b, c, d );

    		}

    	}

    	// build geometry

    	this.setIndex( indices );
    	this.addAttribute( 'position', new Float32BufferAttribute( vertices, 3 ) );
    	this.addAttribute( 'normal', new Float32BufferAttribute( normals, 3 ) );
    	this.addAttribute( 'uv', new Float32BufferAttribute( uvs, 2 ) );

    }

    SphereBufferGeometry.prototype = Object.create( BufferGeometry.prototype );
    SphereBufferGeometry.prototype.constructor = SphereBufferGeometry;

    /**
     * @author Kaleb Murphy
     * @author Mugen87 / https://github.com/Mugen87
     */

    // RingGeometry

    function RingGeometry( innerRadius, outerRadius, thetaSegments, phiSegments, thetaStart, thetaLength ) {

    	Geometry.call( this );

    	this.type = 'RingGeometry';

    	this.parameters = {
    		innerRadius: innerRadius,
    		outerRadius: outerRadius,
    		thetaSegments: thetaSegments,
    		phiSegments: phiSegments,
    		thetaStart: thetaStart,
    		thetaLength: thetaLength
    	};

    	this.fromBufferGeometry( new RingBufferGeometry( innerRadius, outerRadius, thetaSegments, phiSegments, thetaStart, thetaLength ) );
    	this.mergeVertices();

    }

    RingGeometry.prototype = Object.create( Geometry.prototype );
    RingGeometry.prototype.constructor = RingGeometry;

    // RingBufferGeometry

    function RingBufferGeometry( innerRadius, outerRadius, thetaSegments, phiSegments, thetaStart, thetaLength ) {

    	BufferGeometry.call( this );

    	this.type = 'RingBufferGeometry';

    	this.parameters = {
    		innerRadius: innerRadius,
    		outerRadius: outerRadius,
    		thetaSegments: thetaSegments,
    		phiSegments: phiSegments,
    		thetaStart: thetaStart,
    		thetaLength: thetaLength
    	};

    	innerRadius = innerRadius || 0.5;
    	outerRadius = outerRadius || 1;

    	thetaStart = thetaStart !== undefined ? thetaStart : 0;
    	thetaLength = thetaLength !== undefined ? thetaLength : Math.PI * 2;

    	thetaSegments = thetaSegments !== undefined ? Math.max( 3, thetaSegments ) : 8;
    	phiSegments = phiSegments !== undefined ? Math.max( 1, phiSegments ) : 1;

    	// buffers

    	var indices = [];
    	var vertices = [];
    	var normals = [];
    	var uvs = [];

    	// some helper variables

    	var segment;
    	var radius = innerRadius;
    	var radiusStep = ( ( outerRadius - innerRadius ) / phiSegments );
    	var vertex = new Vector3();
    	var uv = new Vector2();
    	var j, i;

    	// generate vertices, normals and uvs

    	for ( j = 0; j <= phiSegments; j ++ ) {

    		for ( i = 0; i <= thetaSegments; i ++ ) {

    			// values are generate from the inside of the ring to the outside

    			segment = thetaStart + i / thetaSegments * thetaLength;

    			// vertex

    			vertex.x = radius * Math.cos( segment );
    			vertex.y = radius * Math.sin( segment );

    			vertices.push( vertex.x, vertex.y, vertex.z );

    			// normal

    			normals.push( 0, 0, 1 );

    			// uv

    			uv.x = ( vertex.x / outerRadius + 1 ) / 2;
    			uv.y = ( vertex.y / outerRadius + 1 ) / 2;

    			uvs.push( uv.x, uv.y );

    		}

    		// increase the radius for next row of vertices

    		radius += radiusStep;

    	}

    	// indices

    	for ( j = 0; j < phiSegments; j ++ ) {

    		var thetaSegmentLevel = j * ( thetaSegments + 1 );

    		for ( i = 0; i < thetaSegments; i ++ ) {

    			segment = i + thetaSegmentLevel;

    			var a = segment;
    			var b = segment + thetaSegments + 1;
    			var c = segment + thetaSegments + 2;
    			var d = segment + 1;

    			// faces

    			indices.push( a, b, d );
    			indices.push( b, c, d );

    		}

    	}

    	// build geometry

    	this.setIndex( indices );
    	this.addAttribute( 'position', new Float32BufferAttribute( vertices, 3 ) );
    	this.addAttribute( 'normal', new Float32BufferAttribute( normals, 3 ) );
    	this.addAttribute( 'uv', new Float32BufferAttribute( uvs, 2 ) );

    }

    RingBufferGeometry.prototype = Object.create( BufferGeometry.prototype );
    RingBufferGeometry.prototype.constructor = RingBufferGeometry;

    /**
     * @author zz85 / https://github.com/zz85
     * @author bhouston / http://clara.io
     * @author Mugen87 / https://github.com/Mugen87
     */

    // LatheGeometry

    function LatheGeometry( points, segments, phiStart, phiLength ) {

    	Geometry.call( this );

    	this.type = 'LatheGeometry';

    	this.parameters = {
    		points: points,
    		segments: segments,
    		phiStart: phiStart,
    		phiLength: phiLength
    	};

    	this.fromBufferGeometry( new LatheBufferGeometry( points, segments, phiStart, phiLength ) );
    	this.mergeVertices();

    }

    LatheGeometry.prototype = Object.create( Geometry.prototype );
    LatheGeometry.prototype.constructor = LatheGeometry;

    // LatheBufferGeometry

    function LatheBufferGeometry( points, segments, phiStart, phiLength ) {

    	BufferGeometry.call( this );

    	this.type = 'LatheBufferGeometry';

    	this.parameters = {
    		points: points,
    		segments: segments,
    		phiStart: phiStart,
    		phiLength: phiLength
    	};

    	segments = Math.floor( segments ) || 12;
    	phiStart = phiStart || 0;
    	phiLength = phiLength || Math.PI * 2;

    	// clamp phiLength so it's in range of [ 0, 2PI ]

    	phiLength = _Math.clamp( phiLength, 0, Math.PI * 2 );


    	// buffers

    	var indices = [];
    	var vertices = [];
    	var uvs = [];

    	// helper variables

    	var base;
    	var inverseSegments = 1.0 / segments;
    	var vertex = new Vector3();
    	var uv = new Vector2();
    	var i, j;

    	// generate vertices and uvs

    	for ( i = 0; i <= segments; i ++ ) {

    		var phi = phiStart + i * inverseSegments * phiLength;

    		var sin = Math.sin( phi );
    		var cos = Math.cos( phi );

    		for ( j = 0; j <= ( points.length - 1 ); j ++ ) {

    			// vertex

    			vertex.x = points[ j ].x * sin;
    			vertex.y = points[ j ].y;
    			vertex.z = points[ j ].x * cos;

    			vertices.push( vertex.x, vertex.y, vertex.z );

    			// uv

    			uv.x = i / segments;
    			uv.y = j / ( points.length - 1 );

    			uvs.push( uv.x, uv.y );


    		}

    	}

    	// indices

    	for ( i = 0; i < segments; i ++ ) {

    		for ( j = 0; j < ( points.length - 1 ); j ++ ) {

    			base = j + i * points.length;

    			var a = base;
    			var b = base + points.length;
    			var c = base + points.length + 1;
    			var d = base + 1;

    			// faces

    			indices.push( a, b, d );
    			indices.push( b, c, d );

    		}

    	}

    	// build geometry

    	this.setIndex( indices );
    	this.addAttribute( 'position', new Float32BufferAttribute( vertices, 3 ) );
    	this.addAttribute( 'uv', new Float32BufferAttribute( uvs, 2 ) );

    	// generate normals

    	this.computeVertexNormals();

    	// if the geometry is closed, we need to average the normals along the seam.
    	// because the corresponding vertices are identical (but still have different UVs).

    	if ( phiLength === Math.PI * 2 ) {

    		var normals = this.attributes.normal.array;
    		var n1 = new Vector3();
    		var n2 = new Vector3();
    		var n = new Vector3();

    		// this is the buffer offset for the last line of vertices

    		base = segments * points.length * 3;

    		for ( i = 0, j = 0; i < points.length; i ++, j += 3 ) {

    			// select the normal of the vertex in the first line

    			n1.x = normals[ j + 0 ];
    			n1.y = normals[ j + 1 ];
    			n1.z = normals[ j + 2 ];

    			// select the normal of the vertex in the last line

    			n2.x = normals[ base + j + 0 ];
    			n2.y = normals[ base + j + 1 ];
    			n2.z = normals[ base + j + 2 ];

    			// average normals

    			n.addVectors( n1, n2 ).normalize();

    			// assign the new values to both normals

    			normals[ j + 0 ] = normals[ base + j + 0 ] = n.x;
    			normals[ j + 1 ] = normals[ base + j + 1 ] = n.y;
    			normals[ j + 2 ] = normals[ base + j + 2 ] = n.z;

    		}

    	}

    }

    LatheBufferGeometry.prototype = Object.create( BufferGeometry.prototype );
    LatheBufferGeometry.prototype.constructor = LatheBufferGeometry;

    /**
     * @author jonobr1 / http://jonobr1.com
     * @author Mugen87 / https://github.com/Mugen87
     */

    // ShapeGeometry

    function ShapeGeometry( shapes, curveSegments ) {

    	Geometry.call( this );

    	this.type = 'ShapeGeometry';

    	if ( typeof curveSegments === 'object' ) {

    		console.warn( 'THREE.ShapeGeometry: Options parameter has been removed.' );

    		curveSegments = curveSegments.curveSegments;

    	}

    	this.parameters = {
    		shapes: shapes,
    		curveSegments: curveSegments
    	};

    	this.fromBufferGeometry( new ShapeBufferGeometry( shapes, curveSegments ) );
    	this.mergeVertices();

    }

    ShapeGeometry.prototype = Object.create( Geometry.prototype );
    ShapeGeometry.prototype.constructor = ShapeGeometry;

    ShapeGeometry.prototype.toJSON = function () {

    	var data = Geometry.prototype.toJSON.call( this );

    	var shapes = this.parameters.shapes;

    	return toJSON$1( shapes, data );

    };

    // ShapeBufferGeometry

    function ShapeBufferGeometry( shapes, curveSegments ) {

    	BufferGeometry.call( this );

    	this.type = 'ShapeBufferGeometry';

    	this.parameters = {
    		shapes: shapes,
    		curveSegments: curveSegments
    	};

    	curveSegments = curveSegments || 12;

    	// buffers

    	var indices = [];
    	var vertices = [];
    	var normals = [];
    	var uvs = [];

    	// helper variables

    	var groupStart = 0;
    	var groupCount = 0;

    	// allow single and array values for "shapes" parameter

    	if ( Array.isArray( shapes ) === false ) {

    		addShape( shapes );

    	} else {

    		for ( var i = 0; i < shapes.length; i ++ ) {

    			addShape( shapes[ i ] );

    			this.addGroup( groupStart, groupCount, i ); // enables MultiMaterial support

    			groupStart += groupCount;
    			groupCount = 0;

    		}

    	}

    	// build geometry

    	this.setIndex( indices );
    	this.addAttribute( 'position', new Float32BufferAttribute( vertices, 3 ) );
    	this.addAttribute( 'normal', new Float32BufferAttribute( normals, 3 ) );
    	this.addAttribute( 'uv', new Float32BufferAttribute( uvs, 2 ) );


    	// helper functions

    	function addShape( shape ) {

    		var i, l, shapeHole;

    		var indexOffset = vertices.length / 3;
    		var points = shape.extractPoints( curveSegments );

    		var shapeVertices = points.shape;
    		var shapeHoles = points.holes;

    		// check direction of vertices

    		if ( ShapeUtils.isClockWise( shapeVertices ) === false ) {

    			shapeVertices = shapeVertices.reverse();

    		}

    		for ( i = 0, l = shapeHoles.length; i < l; i ++ ) {

    			shapeHole = shapeHoles[ i ];

    			if ( ShapeUtils.isClockWise( shapeHole ) === true ) {

    				shapeHoles[ i ] = shapeHole.reverse();

    			}

    		}

    		var faces = ShapeUtils.triangulateShape( shapeVertices, shapeHoles );

    		// join vertices of inner and outer paths to a single array

    		for ( i = 0, l = shapeHoles.length; i < l; i ++ ) {

    			shapeHole = shapeHoles[ i ];
    			shapeVertices = shapeVertices.concat( shapeHole );

    		}

    		// vertices, normals, uvs

    		for ( i = 0, l = shapeVertices.length; i < l; i ++ ) {

    			var vertex = shapeVertices[ i ];

    			vertices.push( vertex.x, vertex.y, 0 );
    			normals.push( 0, 0, 1 );
    			uvs.push( vertex.x, vertex.y ); // world uvs

    		}

    		// incides

    		for ( i = 0, l = faces.length; i < l; i ++ ) {

    			var face = faces[ i ];

    			var a = face[ 0 ] + indexOffset;
    			var b = face[ 1 ] + indexOffset;
    			var c = face[ 2 ] + indexOffset;

    			indices.push( a, b, c );
    			groupCount += 3;

    		}

    	}

    }

    ShapeBufferGeometry.prototype = Object.create( BufferGeometry.prototype );
    ShapeBufferGeometry.prototype.constructor = ShapeBufferGeometry;

    ShapeBufferGeometry.prototype.toJSON = function () {

    	var data = BufferGeometry.prototype.toJSON.call( this );

    	var shapes = this.parameters.shapes;

    	return toJSON$1( shapes, data );

    };

    //

    function toJSON$1( shapes, data ) {

    	data.shapes = [];

    	if ( Array.isArray( shapes ) ) {

    		for ( var i = 0, l = shapes.length; i < l; i ++ ) {

    			var shape = shapes[ i ];

    			data.shapes.push( shape.uuid );

    		}

    	} else {

    		data.shapes.push( shapes.uuid );

    	}

    	return data;

    }

    /**
     * @author WestLangley / http://github.com/WestLangley
     * @author Mugen87 / https://github.com/Mugen87
     */

    function EdgesGeometry( geometry, thresholdAngle ) {

    	BufferGeometry.call( this );

    	this.type = 'EdgesGeometry';

    	this.parameters = {
    		thresholdAngle: thresholdAngle
    	};

    	thresholdAngle = ( thresholdAngle !== undefined ) ? thresholdAngle : 1;

    	// buffer

    	var vertices = [];

    	// helper variables

    	var thresholdDot = Math.cos( _Math.DEG2RAD * thresholdAngle );
    	var edge = [ 0, 0 ], edges = {}, edge1, edge2;
    	var key, keys = [ 'a', 'b', 'c' ];

    	// prepare source geometry

    	var geometry2;

    	if ( geometry.isBufferGeometry ) {

    		geometry2 = new Geometry();
    		geometry2.fromBufferGeometry( geometry );

    	} else {

    		geometry2 = geometry.clone();

    	}

    	geometry2.mergeVertices();
    	geometry2.computeFaceNormals();

    	var sourceVertices = geometry2.vertices;
    	var faces = geometry2.faces;

    	// now create a data structure where each entry represents an edge with its adjoining faces

    	for ( var i = 0, l = faces.length; i < l; i ++ ) {

    		var face = faces[ i ];

    		for ( var j = 0; j < 3; j ++ ) {

    			edge1 = face[ keys[ j ] ];
    			edge2 = face[ keys[ ( j + 1 ) % 3 ] ];
    			edge[ 0 ] = Math.min( edge1, edge2 );
    			edge[ 1 ] = Math.max( edge1, edge2 );

    			key = edge[ 0 ] + ',' + edge[ 1 ];

    			if ( edges[ key ] === undefined ) {

    				edges[ key ] = { index1: edge[ 0 ], index2: edge[ 1 ], face1: i, face2: undefined };

    			} else {

    				edges[ key ].face2 = i;

    			}

    		}

    	}

    	// generate vertices

    	for ( key in edges ) {

    		var e = edges[ key ];

    		// an edge is only rendered if the angle (in degrees) between the face normals of the adjoining faces exceeds this value. default = 1 degree.

    		if ( e.face2 === undefined || faces[ e.face1 ].normal.dot( faces[ e.face2 ].normal ) <= thresholdDot ) {

    			var vertex = sourceVertices[ e.index1 ];
    			vertices.push( vertex.x, vertex.y, vertex.z );

    			vertex = sourceVertices[ e.index2 ];
    			vertices.push( vertex.x, vertex.y, vertex.z );

    		}

    	}

    	// build geometry

    	this.addAttribute( 'position', new Float32BufferAttribute( vertices, 3 ) );

    }

    EdgesGeometry.prototype = Object.create( BufferGeometry.prototype );
    EdgesGeometry.prototype.constructor = EdgesGeometry;

    /**
     * @author mrdoob / http://mrdoob.com/
     * @author Mugen87 / https://github.com/Mugen87
     */

    // CylinderGeometry

    function CylinderGeometry( radiusTop, radiusBottom, height, radialSegments, heightSegments, openEnded, thetaStart, thetaLength ) {

    	Geometry.call( this );

    	this.type = 'CylinderGeometry';

    	this.parameters = {
    		radiusTop: radiusTop,
    		radiusBottom: radiusBottom,
    		height: height,
    		radialSegments: radialSegments,
    		heightSegments: heightSegments,
    		openEnded: openEnded,
    		thetaStart: thetaStart,
    		thetaLength: thetaLength
    	};

    	this.fromBufferGeometry( new CylinderBufferGeometry( radiusTop, radiusBottom, height, radialSegments, heightSegments, openEnded, thetaStart, thetaLength ) );
    	this.mergeVertices();

    }

    CylinderGeometry.prototype = Object.create( Geometry.prototype );
    CylinderGeometry.prototype.constructor = CylinderGeometry;

    // CylinderBufferGeometry

    function CylinderBufferGeometry( radiusTop, radiusBottom, height, radialSegments, heightSegments, openEnded, thetaStart, thetaLength ) {

    	BufferGeometry.call( this );

    	this.type = 'CylinderBufferGeometry';

    	this.parameters = {
    		radiusTop: radiusTop,
    		radiusBottom: radiusBottom,
    		height: height,
    		radialSegments: radialSegments,
    		heightSegments: heightSegments,
    		openEnded: openEnded,
    		thetaStart: thetaStart,
    		thetaLength: thetaLength
    	};

    	var scope = this;

    	radiusTop = radiusTop !== undefined ? radiusTop : 1;
    	radiusBottom = radiusBottom !== undefined ? radiusBottom : 1;
    	height = height || 1;

    	radialSegments = Math.floor( radialSegments ) || 8;
    	heightSegments = Math.floor( heightSegments ) || 1;

    	openEnded = openEnded !== undefined ? openEnded : false;
    	thetaStart = thetaStart !== undefined ? thetaStart : 0.0;
    	thetaLength = thetaLength !== undefined ? thetaLength : Math.PI * 2;

    	// buffers

    	var indices = [];
    	var vertices = [];
    	var normals = [];
    	var uvs = [];

    	// helper variables

    	var index = 0;
    	var indexArray = [];
    	var halfHeight = height / 2;
    	var groupStart = 0;

    	// generate geometry

    	generateTorso();

    	if ( openEnded === false ) {

    		if ( radiusTop > 0 ) generateCap( true );
    		if ( radiusBottom > 0 ) generateCap( false );

    	}

    	// build geometry

    	this.setIndex( indices );
    	this.addAttribute( 'position', new Float32BufferAttribute( vertices, 3 ) );
    	this.addAttribute( 'normal', new Float32BufferAttribute( normals, 3 ) );
    	this.addAttribute( 'uv', new Float32BufferAttribute( uvs, 2 ) );

    	function generateTorso() {

    		var x, y;
    		var normal = new Vector3();
    		var vertex = new Vector3();

    		var groupCount = 0;

    		// this will be used to calculate the normal
    		var slope = ( radiusBottom - radiusTop ) / height;

    		// generate vertices, normals and uvs

    		for ( y = 0; y <= heightSegments; y ++ ) {

    			var indexRow = [];

    			var v = y / heightSegments;

    			// calculate the radius of the current row

    			var radius = v * ( radiusBottom - radiusTop ) + radiusTop;

    			for ( x = 0; x <= radialSegments; x ++ ) {

    				var u = x / radialSegments;

    				var theta = u * thetaLength + thetaStart;

    				var sinTheta = Math.sin( theta );
    				var cosTheta = Math.cos( theta );

    				// vertex

    				vertex.x = radius * sinTheta;
    				vertex.y = - v * height + halfHeight;
    				vertex.z = radius * cosTheta;
    				vertices.push( vertex.x, vertex.y, vertex.z );

    				// normal

    				normal.set( sinTheta, slope, cosTheta ).normalize();
    				normals.push( normal.x, normal.y, normal.z );

    				// uv

    				uvs.push( u, 1 - v );

    				// save index of vertex in respective row

    				indexRow.push( index ++ );

    			}

    			// now save vertices of the row in our index array

    			indexArray.push( indexRow );

    		}

    		// generate indices

    		for ( x = 0; x < radialSegments; x ++ ) {

    			for ( y = 0; y < heightSegments; y ++ ) {

    				// we use the index array to access the correct indices

    				var a = indexArray[ y ][ x ];
    				var b = indexArray[ y + 1 ][ x ];
    				var c = indexArray[ y + 1 ][ x + 1 ];
    				var d = indexArray[ y ][ x + 1 ];

    				// faces

    				indices.push( a, b, d );
    				indices.push( b, c, d );

    				// update group counter

    				groupCount += 6;

    			}

    		}

    		// add a group to the geometry. this will ensure multi material support

    		scope.addGroup( groupStart, groupCount, 0 );

    		// calculate new start value for groups

    		groupStart += groupCount;

    	}

    	function generateCap( top ) {

    		var x, centerIndexStart, centerIndexEnd;

    		var uv = new Vector2();
    		var vertex = new Vector3();

    		var groupCount = 0;

    		var radius = ( top === true ) ? radiusTop : radiusBottom;
    		var sign = ( top === true ) ? 1 : - 1;

    		// save the index of the first center vertex
    		centerIndexStart = index;

    		// first we generate the center vertex data of the cap.
    		// because the geometry needs one set of uvs per face,
    		// we must generate a center vertex per face/segment

    		for ( x = 1; x <= radialSegments; x ++ ) {

    			// vertex

    			vertices.push( 0, halfHeight * sign, 0 );

    			// normal

    			normals.push( 0, sign, 0 );

    			// uv

    			uvs.push( 0.5, 0.5 );

    			// increase index

    			index ++;

    		}

    		// save the index of the last center vertex

    		centerIndexEnd = index;

    		// now we generate the surrounding vertices, normals and uvs

    		for ( x = 0; x <= radialSegments; x ++ ) {

    			var u = x / radialSegments;
    			var theta = u * thetaLength + thetaStart;

    			var cosTheta = Math.cos( theta );
    			var sinTheta = Math.sin( theta );

    			// vertex

    			vertex.x = radius * sinTheta;
    			vertex.y = halfHeight * sign;
    			vertex.z = radius * cosTheta;
    			vertices.push( vertex.x, vertex.y, vertex.z );

    			// normal

    			normals.push( 0, sign, 0 );

    			// uv

    			uv.x = ( cosTheta * 0.5 ) + 0.5;
    			uv.y = ( sinTheta * 0.5 * sign ) + 0.5;
    			uvs.push( uv.x, uv.y );

    			// increase index

    			index ++;

    		}

    		// generate indices

    		for ( x = 0; x < radialSegments; x ++ ) {

    			var c = centerIndexStart + x;
    			var i = centerIndexEnd + x;

    			if ( top === true ) {

    				// face top

    				indices.push( i, i + 1, c );

    			} else {

    				// face bottom

    				indices.push( i + 1, i, c );

    			}

    			groupCount += 3;

    		}

    		// add a group to the geometry. this will ensure multi material support

    		scope.addGroup( groupStart, groupCount, top === true ? 1 : 2 );

    		// calculate new start value for groups

    		groupStart += groupCount;

    	}

    }

    CylinderBufferGeometry.prototype = Object.create( BufferGeometry.prototype );
    CylinderBufferGeometry.prototype.constructor = CylinderBufferGeometry;

    /**
     * @author abelnation / http://github.com/abelnation
     */

    // ConeGeometry

    function ConeGeometry( radius, height, radialSegments, heightSegments, openEnded, thetaStart, thetaLength ) {

    	CylinderGeometry.call( this, 0, radius, height, radialSegments, heightSegments, openEnded, thetaStart, thetaLength );

    	this.type = 'ConeGeometry';

    	this.parameters = {
    		radius: radius,
    		height: height,
    		radialSegments: radialSegments,
    		heightSegments: heightSegments,
    		openEnded: openEnded,
    		thetaStart: thetaStart,
    		thetaLength: thetaLength
    	};

    }

    ConeGeometry.prototype = Object.create( CylinderGeometry.prototype );
    ConeGeometry.prototype.constructor = ConeGeometry;

    // ConeBufferGeometry

    function ConeBufferGeometry( radius, height, radialSegments, heightSegments, openEnded, thetaStart, thetaLength ) {

    	CylinderBufferGeometry.call( this, 0, radius, height, radialSegments, heightSegments, openEnded, thetaStart, thetaLength );

    	this.type = 'ConeBufferGeometry';

    	this.parameters = {
    		radius: radius,
    		height: height,
    		radialSegments: radialSegments,
    		heightSegments: heightSegments,
    		openEnded: openEnded,
    		thetaStart: thetaStart,
    		thetaLength: thetaLength
    	};

    }

    ConeBufferGeometry.prototype = Object.create( CylinderBufferGeometry.prototype );
    ConeBufferGeometry.prototype.constructor = ConeBufferGeometry;

    /**
     * @author benaadams / https://twitter.com/ben_a_adams
     * @author Mugen87 / https://github.com/Mugen87
     * @author hughes
     */

    // CircleGeometry

    function CircleGeometry( radius, segments, thetaStart, thetaLength ) {

    	Geometry.call( this );

    	this.type = 'CircleGeometry';

    	this.parameters = {
    		radius: radius,
    		segments: segments,
    		thetaStart: thetaStart,
    		thetaLength: thetaLength
    	};

    	this.fromBufferGeometry( new CircleBufferGeometry( radius, segments, thetaStart, thetaLength ) );
    	this.mergeVertices();

    }

    CircleGeometry.prototype = Object.create( Geometry.prototype );
    CircleGeometry.prototype.constructor = CircleGeometry;

    // CircleBufferGeometry

    function CircleBufferGeometry( radius, segments, thetaStart, thetaLength ) {

    	BufferGeometry.call( this );

    	this.type = 'CircleBufferGeometry';

    	this.parameters = {
    		radius: radius,
    		segments: segments,
    		thetaStart: thetaStart,
    		thetaLength: thetaLength
    	};

    	radius = radius || 1;
    	segments = segments !== undefined ? Math.max( 3, segments ) : 8;

    	thetaStart = thetaStart !== undefined ? thetaStart : 0;
    	thetaLength = thetaLength !== undefined ? thetaLength : Math.PI * 2;

    	// buffers

    	var indices = [];
    	var vertices = [];
    	var normals = [];
    	var uvs = [];

    	// helper variables

    	var i, s;
    	var vertex = new Vector3();
    	var uv = new Vector2();

    	// center point

    	vertices.push( 0, 0, 0 );
    	normals.push( 0, 0, 1 );
    	uvs.push( 0.5, 0.5 );

    	for ( s = 0, i = 3; s <= segments; s ++, i += 3 ) {

    		var segment = thetaStart + s / segments * thetaLength;

    		// vertex

    		vertex.x = radius * Math.cos( segment );
    		vertex.y = radius * Math.sin( segment );

    		vertices.push( vertex.x, vertex.y, vertex.z );

    		// normal

    		normals.push( 0, 0, 1 );

    		// uvs

    		uv.x = ( vertices[ i ] / radius + 1 ) / 2;
    		uv.y = ( vertices[ i + 1 ] / radius + 1 ) / 2;

    		uvs.push( uv.x, uv.y );

    	}

    	// indices

    	for ( i = 1; i <= segments; i ++ ) {

    		indices.push( i, i + 1, 0 );

    	}

    	// build geometry

    	this.setIndex( indices );
    	this.addAttribute( 'position', new Float32BufferAttribute( vertices, 3 ) );
    	this.addAttribute( 'normal', new Float32BufferAttribute( normals, 3 ) );
    	this.addAttribute( 'uv', new Float32BufferAttribute( uvs, 2 ) );

    }

    CircleBufferGeometry.prototype = Object.create( BufferGeometry.prototype );
    CircleBufferGeometry.prototype.constructor = CircleBufferGeometry;



    var Geometries = /*#__PURE__*/Object.freeze({
    	WireframeGeometry: WireframeGeometry,
    	ParametricGeometry: ParametricGeometry,
    	ParametricBufferGeometry: ParametricBufferGeometry,
    	TetrahedronGeometry: TetrahedronGeometry,
    	TetrahedronBufferGeometry: TetrahedronBufferGeometry,
    	OctahedronGeometry: OctahedronGeometry,
    	OctahedronBufferGeometry: OctahedronBufferGeometry,
    	IcosahedronGeometry: IcosahedronGeometry,
    	IcosahedronBufferGeometry: IcosahedronBufferGeometry,
    	DodecahedronGeometry: DodecahedronGeometry,
    	DodecahedronBufferGeometry: DodecahedronBufferGeometry,
    	PolyhedronGeometry: PolyhedronGeometry,
    	PolyhedronBufferGeometry: PolyhedronBufferGeometry,
    	TubeGeometry: TubeGeometry,
    	TubeBufferGeometry: TubeBufferGeometry,
    	TorusKnotGeometry: TorusKnotGeometry,
    	TorusKnotBufferGeometry: TorusKnotBufferGeometry,
    	TorusGeometry: TorusGeometry,
    	TorusBufferGeometry: TorusBufferGeometry,
    	TextGeometry: TextGeometry,
    	TextBufferGeometry: TextBufferGeometry,
    	SphereGeometry: SphereGeometry,
    	SphereBufferGeometry: SphereBufferGeometry,
    	RingGeometry: RingGeometry,
    	RingBufferGeometry: RingBufferGeometry,
    	PlaneGeometry: PlaneGeometry,
    	PlaneBufferGeometry: PlaneBufferGeometry,
    	LatheGeometry: LatheGeometry,
    	LatheBufferGeometry: LatheBufferGeometry,
    	ShapeGeometry: ShapeGeometry,
    	ShapeBufferGeometry: ShapeBufferGeometry,
    	ExtrudeGeometry: ExtrudeGeometry,
    	ExtrudeBufferGeometry: ExtrudeBufferGeometry,
    	EdgesGeometry: EdgesGeometry,
    	ConeGeometry: ConeGeometry,
    	ConeBufferGeometry: ConeBufferGeometry,
    	CylinderGeometry: CylinderGeometry,
    	CylinderBufferGeometry: CylinderBufferGeometry,
    	CircleGeometry: CircleGeometry,
    	CircleBufferGeometry: CircleBufferGeometry,
    	BoxGeometry: BoxGeometry,
    	BoxBufferGeometry: BoxBufferGeometry
    });

    /**
     * @author mrdoob / http://mrdoob.com/
     *
     * parameters = {
     *  color: <THREE.Color>
     * }
     */

    function ShadowMaterial( parameters ) {

    	Material.call( this );

    	this.type = 'ShadowMaterial';

    	this.color = new Color$7( 0x000000 );
    	this.transparent = true;

    	this.setValues( parameters );

    }

    ShadowMaterial.prototype = Object.create( Material.prototype );
    ShadowMaterial.prototype.constructor = ShadowMaterial;

    ShadowMaterial.prototype.isShadowMaterial = true;

    ShadowMaterial.prototype.copy = function ( source ) {

    	Material.prototype.copy.call( this, source );

    	this.color.copy( source.color );

    	return this;

    };

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function RawShaderMaterial( parameters ) {

    	ShaderMaterial.call( this, parameters );

    	this.type = 'RawShaderMaterial';

    }

    RawShaderMaterial.prototype = Object.create( ShaderMaterial.prototype );
    RawShaderMaterial.prototype.constructor = RawShaderMaterial;

    RawShaderMaterial.prototype.isRawShaderMaterial = true;

    /**
     * @author WestLangley / http://github.com/WestLangley
     *
     * parameters = {
     *  color: <hex>,
     *  roughness: <float>,
     *  metalness: <float>,
     *  opacity: <float>,
     *
     *  map: new THREE.Texture( <Image> ),
     *
     *  lightMap: new THREE.Texture( <Image> ),
     *  lightMapIntensity: <float>
     *
     *  aoMap: new THREE.Texture( <Image> ),
     *  aoMapIntensity: <float>
     *
     *  emissive: <hex>,
     *  emissiveIntensity: <float>
     *  emissiveMap: new THREE.Texture( <Image> ),
     *
     *  bumpMap: new THREE.Texture( <Image> ),
     *  bumpScale: <float>,
     *
     *  normalMap: new THREE.Texture( <Image> ),
     *  normalMapType: THREE.TangentSpaceNormalMap,
     *  normalScale: <Vector2>,
     *
     *  displacementMap: new THREE.Texture( <Image> ),
     *  displacementScale: <float>,
     *  displacementBias: <float>,
     *
     *  roughnessMap: new THREE.Texture( <Image> ),
     *
     *  metalnessMap: new THREE.Texture( <Image> ),
     *
     *  alphaMap: new THREE.Texture( <Image> ),
     *
     *  envMap: new THREE.CubeTexture( [posx, negx, posy, negy, posz, negz] ),
     *  envMapIntensity: <float>
     *
     *  refractionRatio: <float>,
     *
     *  wireframe: <boolean>,
     *  wireframeLinewidth: <float>,
     *
     *  skinning: <bool>,
     *  morphTargets: <bool>,
     *  morphNormals: <bool>
     * }
     */

    function MeshStandardMaterial( parameters ) {

    	Material.call( this );

    	this.defines = { 'STANDARD': '' };

    	this.type = 'MeshStandardMaterial';

    	this.color = new Color$7( 0xffffff ); // diffuse
    	this.roughness = 0.5;
    	this.metalness = 0.5;

    	this.map = null;

    	this.lightMap = null;
    	this.lightMapIntensity = 1.0;

    	this.aoMap = null;
    	this.aoMapIntensity = 1.0;

    	this.emissive = new Color$7( 0x000000 );
    	this.emissiveIntensity = 1.0;
    	this.emissiveMap = null;

    	this.bumpMap = null;
    	this.bumpScale = 1;

    	this.normalMap = null;
    	this.normalMapType = TangentSpaceNormalMap;
    	this.normalScale = new Vector2( 1, 1 );

    	this.displacementMap = null;
    	this.displacementScale = 1;
    	this.displacementBias = 0;

    	this.roughnessMap = null;

    	this.metalnessMap = null;

    	this.alphaMap = null;

    	this.envMap = null;
    	this.envMapIntensity = 1.0;

    	this.refractionRatio = 0.98;

    	this.wireframe = false;
    	this.wireframeLinewidth = 1;
    	this.wireframeLinecap = 'round';
    	this.wireframeLinejoin = 'round';

    	this.skinning = false;
    	this.morphTargets = false;
    	this.morphNormals = false;

    	this.setValues( parameters );

    }

    MeshStandardMaterial.prototype = Object.create( Material.prototype );
    MeshStandardMaterial.prototype.constructor = MeshStandardMaterial;

    MeshStandardMaterial.prototype.isMeshStandardMaterial = true;

    MeshStandardMaterial.prototype.copy = function ( source ) {

    	Material.prototype.copy.call( this, source );

    	this.defines = { 'STANDARD': '' };

    	this.color.copy( source.color );
    	this.roughness = source.roughness;
    	this.metalness = source.metalness;

    	this.map = source.map;

    	this.lightMap = source.lightMap;
    	this.lightMapIntensity = source.lightMapIntensity;

    	this.aoMap = source.aoMap;
    	this.aoMapIntensity = source.aoMapIntensity;

    	this.emissive.copy( source.emissive );
    	this.emissiveMap = source.emissiveMap;
    	this.emissiveIntensity = source.emissiveIntensity;

    	this.bumpMap = source.bumpMap;
    	this.bumpScale = source.bumpScale;

    	this.normalMap = source.normalMap;
    	this.normalMapType = source.normalMapType;
    	this.normalScale.copy( source.normalScale );

    	this.displacementMap = source.displacementMap;
    	this.displacementScale = source.displacementScale;
    	this.displacementBias = source.displacementBias;

    	this.roughnessMap = source.roughnessMap;

    	this.metalnessMap = source.metalnessMap;

    	this.alphaMap = source.alphaMap;

    	this.envMap = source.envMap;
    	this.envMapIntensity = source.envMapIntensity;

    	this.refractionRatio = source.refractionRatio;

    	this.wireframe = source.wireframe;
    	this.wireframeLinewidth = source.wireframeLinewidth;
    	this.wireframeLinecap = source.wireframeLinecap;
    	this.wireframeLinejoin = source.wireframeLinejoin;

    	this.skinning = source.skinning;
    	this.morphTargets = source.morphTargets;
    	this.morphNormals = source.morphNormals;

    	return this;

    };

    /**
     * @author WestLangley / http://github.com/WestLangley
     *
     * parameters = {
     *  reflectivity: <float>
     *  clearcoat: <float>
     *  clearcoatRoughness: <float>
     *
     *  sheen: <Color>
     *
     *  clearcoatNormalScale: <Vector2>,
     *  clearcoatNormalMap: new THREE.Texture( <Image> ),
     * }
     */

    function MeshPhysicalMaterial( parameters ) {

    	MeshStandardMaterial.call( this );

    	this.defines = {

    		'STANDARD': '',
    		'PHYSICAL': ''

    	};

    	this.type = 'MeshPhysicalMaterial';

    	this.reflectivity = 0.5; // maps to F0 = 0.04

    	this.clearcoat = 0.0;
    	this.clearcoatRoughness = 0.0;

    	this.sheen = null; // null will disable sheen bsdf

    	this.clearcoatNormalScale = new Vector2( 1, 1 );
    	this.clearcoatNormalMap = null;

    	this.transparency = 0.0;

    	this.setValues( parameters );

    }

    MeshPhysicalMaterial.prototype = Object.create( MeshStandardMaterial.prototype );
    MeshPhysicalMaterial.prototype.constructor = MeshPhysicalMaterial;

    MeshPhysicalMaterial.prototype.isMeshPhysicalMaterial = true;

    MeshPhysicalMaterial.prototype.copy = function ( source ) {

    	MeshStandardMaterial.prototype.copy.call( this, source );

    	this.defines = {

    		'STANDARD': '',
    		'PHYSICAL': ''

    	};

    	this.reflectivity = source.reflectivity;

    	this.clearcoat = source.clearcoat;
    	this.clearcoatRoughness = source.clearcoatRoughness;

    	if ( source.sheen ) this.sheen = ( this.sheen || new Color$7() ).copy( source.sheen );
    	else this.sheen = null;

    	this.clearcoatNormalMap = source.clearcoatNormalMap;
    	this.clearcoatNormalScale.copy( source.clearcoatNormalScale );

    	this.transparency = source.transparency;

    	return this;

    };

    /**
     * @author mrdoob / http://mrdoob.com/
     * @author alteredq / http://alteredqualia.com/
     *
     * parameters = {
     *  color: <hex>,
     *  specular: <hex>,
     *  shininess: <float>,
     *  opacity: <float>,
     *
     *  map: new THREE.Texture( <Image> ),
     *
     *  lightMap: new THREE.Texture( <Image> ),
     *  lightMapIntensity: <float>
     *
     *  aoMap: new THREE.Texture( <Image> ),
     *  aoMapIntensity: <float>
     *
     *  emissive: <hex>,
     *  emissiveIntensity: <float>
     *  emissiveMap: new THREE.Texture( <Image> ),
     *
     *  bumpMap: new THREE.Texture( <Image> ),
     *  bumpScale: <float>,
     *
     *  normalMap: new THREE.Texture( <Image> ),
     *  normalMapType: THREE.TangentSpaceNormalMap,
     *  normalScale: <Vector2>,
     *
     *  displacementMap: new THREE.Texture( <Image> ),
     *  displacementScale: <float>,
     *  displacementBias: <float>,
     *
     *  specularMap: new THREE.Texture( <Image> ),
     *
     *  alphaMap: new THREE.Texture( <Image> ),
     *
     *  envMap: new THREE.CubeTexture( [posx, negx, posy, negy, posz, negz] ),
     *  combine: THREE.Multiply,
     *  reflectivity: <float>,
     *  refractionRatio: <float>,
     *
     *  wireframe: <boolean>,
     *  wireframeLinewidth: <float>,
     *
     *  skinning: <bool>,
     *  morphTargets: <bool>,
     *  morphNormals: <bool>
     * }
     */

    function MeshPhongMaterial( parameters ) {

    	Material.call( this );

    	this.type = 'MeshPhongMaterial';

    	this.color = new Color$7( 0xffffff ); // diffuse
    	this.specular = new Color$7( 0x111111 );
    	this.shininess = 30;

    	this.map = null;

    	this.lightMap = null;
    	this.lightMapIntensity = 1.0;

    	this.aoMap = null;
    	this.aoMapIntensity = 1.0;

    	this.emissive = new Color$7( 0x000000 );
    	this.emissiveIntensity = 1.0;
    	this.emissiveMap = null;

    	this.bumpMap = null;
    	this.bumpScale = 1;

    	this.normalMap = null;
    	this.normalMapType = TangentSpaceNormalMap;
    	this.normalScale = new Vector2( 1, 1 );

    	this.displacementMap = null;
    	this.displacementScale = 1;
    	this.displacementBias = 0;

    	this.specularMap = null;

    	this.alphaMap = null;

    	this.envMap = null;
    	this.combine = MultiplyOperation;
    	this.reflectivity = 1;
    	this.refractionRatio = 0.98;

    	this.wireframe = false;
    	this.wireframeLinewidth = 1;
    	this.wireframeLinecap = 'round';
    	this.wireframeLinejoin = 'round';

    	this.skinning = false;
    	this.morphTargets = false;
    	this.morphNormals = false;

    	this.setValues( parameters );

    }

    MeshPhongMaterial.prototype = Object.create( Material.prototype );
    MeshPhongMaterial.prototype.constructor = MeshPhongMaterial;

    MeshPhongMaterial.prototype.isMeshPhongMaterial = true;

    MeshPhongMaterial.prototype.copy = function ( source ) {

    	Material.prototype.copy.call( this, source );

    	this.color.copy( source.color );
    	this.specular.copy( source.specular );
    	this.shininess = source.shininess;

    	this.map = source.map;

    	this.lightMap = source.lightMap;
    	this.lightMapIntensity = source.lightMapIntensity;

    	this.aoMap = source.aoMap;
    	this.aoMapIntensity = source.aoMapIntensity;

    	this.emissive.copy( source.emissive );
    	this.emissiveMap = source.emissiveMap;
    	this.emissiveIntensity = source.emissiveIntensity;

    	this.bumpMap = source.bumpMap;
    	this.bumpScale = source.bumpScale;

    	this.normalMap = source.normalMap;
    	this.normalMapType = source.normalMapType;
    	this.normalScale.copy( source.normalScale );

    	this.displacementMap = source.displacementMap;
    	this.displacementScale = source.displacementScale;
    	this.displacementBias = source.displacementBias;

    	this.specularMap = source.specularMap;

    	this.alphaMap = source.alphaMap;

    	this.envMap = source.envMap;
    	this.combine = source.combine;
    	this.reflectivity = source.reflectivity;
    	this.refractionRatio = source.refractionRatio;

    	this.wireframe = source.wireframe;
    	this.wireframeLinewidth = source.wireframeLinewidth;
    	this.wireframeLinecap = source.wireframeLinecap;
    	this.wireframeLinejoin = source.wireframeLinejoin;

    	this.skinning = source.skinning;
    	this.morphTargets = source.morphTargets;
    	this.morphNormals = source.morphNormals;

    	return this;

    };

    /**
     * @author takahirox / http://github.com/takahirox
     *
     * parameters = {
     *  gradientMap: new THREE.Texture( <Image> )
     * }
     */

    function MeshToonMaterial( parameters ) {

    	MeshPhongMaterial.call( this );

    	this.defines = { 'TOON': '' };

    	this.type = 'MeshToonMaterial';

    	this.gradientMap = null;

    	this.setValues( parameters );

    }

    MeshToonMaterial.prototype = Object.create( MeshPhongMaterial.prototype );
    MeshToonMaterial.prototype.constructor = MeshToonMaterial;

    MeshToonMaterial.prototype.isMeshToonMaterial = true;

    MeshToonMaterial.prototype.copy = function ( source ) {

    	MeshPhongMaterial.prototype.copy.call( this, source );

    	this.gradientMap = source.gradientMap;

    	return this;

    };

    /**
     * @author mrdoob / http://mrdoob.com/
     * @author WestLangley / http://github.com/WestLangley
     *
     * parameters = {
     *  opacity: <float>,
     *
     *  bumpMap: new THREE.Texture( <Image> ),
     *  bumpScale: <float>,
     *
     *  normalMap: new THREE.Texture( <Image> ),
     *  normalMapType: THREE.TangentSpaceNormalMap,
     *  normalScale: <Vector2>,
     *
     *  displacementMap: new THREE.Texture( <Image> ),
     *  displacementScale: <float>,
     *  displacementBias: <float>,
     *
     *  wireframe: <boolean>,
     *  wireframeLinewidth: <float>
     *
     *  skinning: <bool>,
     *  morphTargets: <bool>,
     *  morphNormals: <bool>
     * }
     */

    function MeshNormalMaterial( parameters ) {

    	Material.call( this );

    	this.type = 'MeshNormalMaterial';

    	this.bumpMap = null;
    	this.bumpScale = 1;

    	this.normalMap = null;
    	this.normalMapType = TangentSpaceNormalMap;
    	this.normalScale = new Vector2( 1, 1 );

    	this.displacementMap = null;
    	this.displacementScale = 1;
    	this.displacementBias = 0;

    	this.wireframe = false;
    	this.wireframeLinewidth = 1;

    	this.fog = false;
    	this.lights = false;

    	this.skinning = false;
    	this.morphTargets = false;
    	this.morphNormals = false;

    	this.setValues( parameters );

    }

    MeshNormalMaterial.prototype = Object.create( Material.prototype );
    MeshNormalMaterial.prototype.constructor = MeshNormalMaterial;

    MeshNormalMaterial.prototype.isMeshNormalMaterial = true;

    MeshNormalMaterial.prototype.copy = function ( source ) {

    	Material.prototype.copy.call( this, source );

    	this.bumpMap = source.bumpMap;
    	this.bumpScale = source.bumpScale;

    	this.normalMap = source.normalMap;
    	this.normalMapType = source.normalMapType;
    	this.normalScale.copy( source.normalScale );

    	this.displacementMap = source.displacementMap;
    	this.displacementScale = source.displacementScale;
    	this.displacementBias = source.displacementBias;

    	this.wireframe = source.wireframe;
    	this.wireframeLinewidth = source.wireframeLinewidth;

    	this.skinning = source.skinning;
    	this.morphTargets = source.morphTargets;
    	this.morphNormals = source.morphNormals;

    	return this;

    };

    /**
     * @author mrdoob / http://mrdoob.com/
     * @author alteredq / http://alteredqualia.com/
     *
     * parameters = {
     *  color: <hex>,
     *  opacity: <float>,
     *
     *  map: new THREE.Texture( <Image> ),
     *
     *  lightMap: new THREE.Texture( <Image> ),
     *  lightMapIntensity: <float>
     *
     *  aoMap: new THREE.Texture( <Image> ),
     *  aoMapIntensity: <float>
     *
     *  emissive: <hex>,
     *  emissiveIntensity: <float>
     *  emissiveMap: new THREE.Texture( <Image> ),
     *
     *  specularMap: new THREE.Texture( <Image> ),
     *
     *  alphaMap: new THREE.Texture( <Image> ),
     *
     *  envMap: new THREE.CubeTexture( [posx, negx, posy, negy, posz, negz] ),
     *  combine: THREE.Multiply,
     *  reflectivity: <float>,
     *  refractionRatio: <float>,
     *
     *  wireframe: <boolean>,
     *  wireframeLinewidth: <float>,
     *
     *  skinning: <bool>,
     *  morphTargets: <bool>,
     *  morphNormals: <bool>
     * }
     */

    function MeshLambertMaterial( parameters ) {

    	Material.call( this );

    	this.type = 'MeshLambertMaterial';

    	this.color = new Color$7( 0xffffff ); // diffuse

    	this.map = null;

    	this.lightMap = null;
    	this.lightMapIntensity = 1.0;

    	this.aoMap = null;
    	this.aoMapIntensity = 1.0;

    	this.emissive = new Color$7( 0x000000 );
    	this.emissiveIntensity = 1.0;
    	this.emissiveMap = null;

    	this.specularMap = null;

    	this.alphaMap = null;

    	this.envMap = null;
    	this.combine = MultiplyOperation;
    	this.reflectivity = 1;
    	this.refractionRatio = 0.98;

    	this.wireframe = false;
    	this.wireframeLinewidth = 1;
    	this.wireframeLinecap = 'round';
    	this.wireframeLinejoin = 'round';

    	this.skinning = false;
    	this.morphTargets = false;
    	this.morphNormals = false;

    	this.setValues( parameters );

    }

    MeshLambertMaterial.prototype = Object.create( Material.prototype );
    MeshLambertMaterial.prototype.constructor = MeshLambertMaterial;

    MeshLambertMaterial.prototype.isMeshLambertMaterial = true;

    MeshLambertMaterial.prototype.copy = function ( source ) {

    	Material.prototype.copy.call( this, source );

    	this.color.copy( source.color );

    	this.map = source.map;

    	this.lightMap = source.lightMap;
    	this.lightMapIntensity = source.lightMapIntensity;

    	this.aoMap = source.aoMap;
    	this.aoMapIntensity = source.aoMapIntensity;

    	this.emissive.copy( source.emissive );
    	this.emissiveMap = source.emissiveMap;
    	this.emissiveIntensity = source.emissiveIntensity;

    	this.specularMap = source.specularMap;

    	this.alphaMap = source.alphaMap;

    	this.envMap = source.envMap;
    	this.combine = source.combine;
    	this.reflectivity = source.reflectivity;
    	this.refractionRatio = source.refractionRatio;

    	this.wireframe = source.wireframe;
    	this.wireframeLinewidth = source.wireframeLinewidth;
    	this.wireframeLinecap = source.wireframeLinecap;
    	this.wireframeLinejoin = source.wireframeLinejoin;

    	this.skinning = source.skinning;
    	this.morphTargets = source.morphTargets;
    	this.morphNormals = source.morphNormals;

    	return this;

    };

    /**
     * @author WestLangley / http://github.com/WestLangley
     *
     * parameters = {
     *  color: <hex>,
     *  opacity: <float>,
     *
     *  matcap: new THREE.Texture( <Image> ),
     *
     *  map: new THREE.Texture( <Image> ),
     *
     *  bumpMap: new THREE.Texture( <Image> ),
     *  bumpScale: <float>,
     *
     *  normalMap: new THREE.Texture( <Image> ),
     *  normalMapType: THREE.TangentSpaceNormalMap,
     *  normalScale: <Vector2>,
     *
     *  displacementMap: new THREE.Texture( <Image> ),
     *  displacementScale: <float>,
     *  displacementBias: <float>,
     *
     *  alphaMap: new THREE.Texture( <Image> ),
     *
     *  skinning: <bool>,
     *  morphTargets: <bool>,
     *  morphNormals: <bool>
     * }
     */

    function MeshMatcapMaterial( parameters ) {

    	Material.call( this );

    	this.defines = { 'MATCAP': '' };

    	this.type = 'MeshMatcapMaterial';

    	this.color = new Color$7( 0xffffff ); // diffuse

    	this.matcap = null;

    	this.map = null;

    	this.bumpMap = null;
    	this.bumpScale = 1;

    	this.normalMap = null;
    	this.normalMapType = TangentSpaceNormalMap;
    	this.normalScale = new Vector2( 1, 1 );

    	this.displacementMap = null;
    	this.displacementScale = 1;
    	this.displacementBias = 0;

    	this.alphaMap = null;

    	this.skinning = false;
    	this.morphTargets = false;
    	this.morphNormals = false;

    	this.lights = false;

    	this.setValues( parameters );

    }

    MeshMatcapMaterial.prototype = Object.create( Material.prototype );
    MeshMatcapMaterial.prototype.constructor = MeshMatcapMaterial;

    MeshMatcapMaterial.prototype.isMeshMatcapMaterial = true;

    MeshMatcapMaterial.prototype.copy = function ( source ) {

    	Material.prototype.copy.call( this, source );

    	this.defines = { 'MATCAP': '' };

    	this.color.copy( source.color );

    	this.matcap = source.matcap;

    	this.map = source.map;

    	this.bumpMap = source.bumpMap;
    	this.bumpScale = source.bumpScale;

    	this.normalMap = source.normalMap;
    	this.normalMapType = source.normalMapType;
    	this.normalScale.copy( source.normalScale );

    	this.displacementMap = source.displacementMap;
    	this.displacementScale = source.displacementScale;
    	this.displacementBias = source.displacementBias;

    	this.alphaMap = source.alphaMap;

    	this.skinning = source.skinning;
    	this.morphTargets = source.morphTargets;
    	this.morphNormals = source.morphNormals;

    	return this;

    };

    /**
     * @author alteredq / http://alteredqualia.com/
     *
     * parameters = {
     *  color: <hex>,
     *  opacity: <float>,
     *
     *  linewidth: <float>,
     *
     *  scale: <float>,
     *  dashSize: <float>,
     *  gapSize: <float>
     * }
     */

    function LineDashedMaterial( parameters ) {

    	LineBasicMaterial.call( this );

    	this.type = 'LineDashedMaterial';

    	this.scale = 1;
    	this.dashSize = 3;
    	this.gapSize = 1;

    	this.setValues( parameters );

    }

    LineDashedMaterial.prototype = Object.create( LineBasicMaterial.prototype );
    LineDashedMaterial.prototype.constructor = LineDashedMaterial;

    LineDashedMaterial.prototype.isLineDashedMaterial = true;

    LineDashedMaterial.prototype.copy = function ( source ) {

    	LineBasicMaterial.prototype.copy.call( this, source );

    	this.scale = source.scale;
    	this.dashSize = source.dashSize;
    	this.gapSize = source.gapSize;

    	return this;

    };



    var Materials = /*#__PURE__*/Object.freeze({
    	ShadowMaterial: ShadowMaterial,
    	SpriteMaterial: SpriteMaterial,
    	RawShaderMaterial: RawShaderMaterial,
    	ShaderMaterial: ShaderMaterial,
    	PointsMaterial: PointsMaterial,
    	MeshPhysicalMaterial: MeshPhysicalMaterial,
    	MeshStandardMaterial: MeshStandardMaterial,
    	MeshPhongMaterial: MeshPhongMaterial,
    	MeshToonMaterial: MeshToonMaterial,
    	MeshNormalMaterial: MeshNormalMaterial,
    	MeshLambertMaterial: MeshLambertMaterial,
    	MeshDepthMaterial: MeshDepthMaterial,
    	MeshDistanceMaterial: MeshDistanceMaterial,
    	MeshBasicMaterial: MeshBasicMaterial,
    	MeshMatcapMaterial: MeshMatcapMaterial,
    	LineDashedMaterial: LineDashedMaterial,
    	LineBasicMaterial: LineBasicMaterial,
    	Material: Material
    });

    /**
     * @author tschw
     * @author Ben Houston / http://clara.io/
     * @author David Sarno / http://lighthaus.us/
     */

    var AnimationUtils = {

    	// same as Array.prototype.slice, but also works on typed arrays
    	arraySlice: function ( array, from, to ) {

    		if ( AnimationUtils.isTypedArray( array ) ) {

    			// in ios9 array.subarray(from, undefined) will return empty array
    			// but array.subarray(from) or array.subarray(from, len) is correct
    			return new array.constructor( array.subarray( from, to !== undefined ? to : array.length ) );

    		}

    		return array.slice( from, to );

    	},

    	// converts an array to a specific type
    	convertArray: function ( array, type, forceClone ) {

    		if ( ! array || // let 'undefined' and 'null' pass
    			! forceClone && array.constructor === type ) return array;

    		if ( typeof type.BYTES_PER_ELEMENT === 'number' ) {

    			return new type( array ); // create typed array

    		}

    		return Array.prototype.slice.call( array ); // create Array

    	},

    	isTypedArray: function ( object ) {

    		return ArrayBuffer.isView( object ) &&
    			! ( object instanceof DataView );

    	},

    	// returns an array by which times and values can be sorted
    	getKeyframeOrder: function ( times ) {

    		function compareTime( i, j ) {

    			return times[ i ] - times[ j ];

    		}

    		var n = times.length;
    		var result = new Array( n );
    		for ( var i = 0; i !== n; ++ i ) result[ i ] = i;

    		result.sort( compareTime );

    		return result;

    	},

    	// uses the array previously returned by 'getKeyframeOrder' to sort data
    	sortedArray: function ( values, stride, order ) {

    		var nValues = values.length;
    		var result = new values.constructor( nValues );

    		for ( var i = 0, dstOffset = 0; dstOffset !== nValues; ++ i ) {

    			var srcOffset = order[ i ] * stride;

    			for ( var j = 0; j !== stride; ++ j ) {

    				result[ dstOffset ++ ] = values[ srcOffset + j ];

    			}

    		}

    		return result;

    	},

    	// function for parsing AOS keyframe formats
    	flattenJSON: function ( jsonKeys, times, values, valuePropertyName ) {

    		var i = 1, key = jsonKeys[ 0 ];

    		while ( key !== undefined && key[ valuePropertyName ] === undefined ) {

    			key = jsonKeys[ i ++ ];

    		}

    		if ( key === undefined ) return; // no data

    		var value = key[ valuePropertyName ];
    		if ( value === undefined ) return; // no data

    		if ( Array.isArray( value ) ) {

    			do {

    				value = key[ valuePropertyName ];

    				if ( value !== undefined ) {

    					times.push( key.time );
    					values.push.apply( values, value ); // push all elements

    				}

    				key = jsonKeys[ i ++ ];

    			} while ( key !== undefined );

    		} else if ( value.toArray !== undefined ) {

    			// ...assume THREE.Math-ish

    			do {

    				value = key[ valuePropertyName ];

    				if ( value !== undefined ) {

    					times.push( key.time );
    					value.toArray( values, values.length );

    				}

    				key = jsonKeys[ i ++ ];

    			} while ( key !== undefined );

    		} else {

    			// otherwise push as-is

    			do {

    				value = key[ valuePropertyName ];

    				if ( value !== undefined ) {

    					times.push( key.time );
    					values.push( value );

    				}

    				key = jsonKeys[ i ++ ];

    			} while ( key !== undefined );

    		}

    	}

    };

    /**
     * Abstract base class of interpolants over parametric samples.
     *
     * The parameter domain is one dimensional, typically the time or a path
     * along a curve defined by the data.
     *
     * The sample values can have any dimensionality and derived classes may
     * apply special interpretations to the data.
     *
     * This class provides the interval seek in a Template Method, deferring
     * the actual interpolation to derived classes.
     *
     * Time complexity is O(1) for linear access crossing at most two points
     * and O(log N) for random access, where N is the number of positions.
     *
     * References:
     *
     * 		http://www.oodesign.com/template-method-pattern.html
     *
     * @author tschw
     */

    function Interpolant( parameterPositions, sampleValues, sampleSize, resultBuffer ) {

    	this.parameterPositions = parameterPositions;
    	this._cachedIndex = 0;

    	this.resultBuffer = resultBuffer !== undefined ?
    		resultBuffer : new sampleValues.constructor( sampleSize );
    	this.sampleValues = sampleValues;
    	this.valueSize = sampleSize;

    }

    Object.assign( Interpolant.prototype, {

    	evaluate: function ( t ) {

    		var pp = this.parameterPositions,
    			i1 = this._cachedIndex,

    			t1 = pp[ i1 ],
    			t0 = pp[ i1 - 1 ];

    		validate_interval: {

    			seek: {

    				var right;

    				linear_scan: {

    					//- See http://jsperf.com/comparison-to-undefined/3
    					//- slower code:
    					//-
    					//- 				if ( t >= t1 || t1 === undefined ) {
    					forward_scan: if ( ! ( t < t1 ) ) {

    						for ( var giveUpAt = i1 + 2; ; ) {

    							if ( t1 === undefined ) {

    								if ( t < t0 ) break forward_scan;

    								// after end

    								i1 = pp.length;
    								this._cachedIndex = i1;
    								return this.afterEnd_( i1 - 1, t, t0 );

    							}

    							if ( i1 === giveUpAt ) break; // this loop

    							t0 = t1;
    							t1 = pp[ ++ i1 ];

    							if ( t < t1 ) {

    								// we have arrived at the sought interval
    								break seek;

    							}

    						}

    						// prepare binary search on the right side of the index
    						right = pp.length;
    						break linear_scan;

    					}

    					//- slower code:
    					//-					if ( t < t0 || t0 === undefined ) {
    					if ( ! ( t >= t0 ) ) {

    						// looping?

    						var t1global = pp[ 1 ];

    						if ( t < t1global ) {

    							i1 = 2; // + 1, using the scan for the details
    							t0 = t1global;

    						}

    						// linear reverse scan

    						for ( var giveUpAt = i1 - 2; ; ) {

    							if ( t0 === undefined ) {

    								// before start

    								this._cachedIndex = 0;
    								return this.beforeStart_( 0, t, t1 );

    							}

    							if ( i1 === giveUpAt ) break; // this loop

    							t1 = t0;
    							t0 = pp[ -- i1 - 1 ];

    							if ( t >= t0 ) {

    								// we have arrived at the sought interval
    								break seek;

    							}

    						}

    						// prepare binary search on the left side of the index
    						right = i1;
    						i1 = 0;
    						break linear_scan;

    					}

    					// the interval is valid

    					break validate_interval;

    				} // linear scan

    				// binary search

    				while ( i1 < right ) {

    					var mid = ( i1 + right ) >>> 1;

    					if ( t < pp[ mid ] ) {

    						right = mid;

    					} else {

    						i1 = mid + 1;

    					}

    				}

    				t1 = pp[ i1 ];
    				t0 = pp[ i1 - 1 ];

    				// check boundary cases, again

    				if ( t0 === undefined ) {

    					this._cachedIndex = 0;
    					return this.beforeStart_( 0, t, t1 );

    				}

    				if ( t1 === undefined ) {

    					i1 = pp.length;
    					this._cachedIndex = i1;
    					return this.afterEnd_( i1 - 1, t0, t );

    				}

    			} // seek

    			this._cachedIndex = i1;

    			this.intervalChanged_( i1, t0, t1 );

    		} // validate_interval

    		return this.interpolate_( i1, t0, t, t1 );

    	},

    	settings: null, // optional, subclass-specific settings structure
    	// Note: The indirection allows central control of many interpolants.

    	// --- Protected interface

    	DefaultSettings_: {},

    	getSettings_: function () {

    		return this.settings || this.DefaultSettings_;

    	},

    	copySampleValue_: function ( index ) {

    		// copies a sample value to the result buffer

    		var result = this.resultBuffer,
    			values = this.sampleValues,
    			stride = this.valueSize,
    			offset = index * stride;

    		for ( var i = 0; i !== stride; ++ i ) {

    			result[ i ] = values[ offset + i ];

    		}

    		return result;

    	},

    	// Template methods for derived classes:

    	interpolate_: function ( /* i1, t0, t, t1 */ ) {

    		throw new Error( 'call to abstract method' );
    		// implementations shall return this.resultBuffer

    	},

    	intervalChanged_: function ( /* i1, t0, t1 */ ) {

    		// empty

    	}

    } );

    //!\ DECLARE ALIAS AFTER assign prototype !
    Object.assign( Interpolant.prototype, {

    	//( 0, t, t0 ), returns this.resultBuffer
    	beforeStart_: Interpolant.prototype.copySampleValue_,

    	//( N-1, tN-1, t ), returns this.resultBuffer
    	afterEnd_: Interpolant.prototype.copySampleValue_,

    } );

    /**
     * Fast and simple cubic spline interpolant.
     *
     * It was derived from a Hermitian construction setting the first derivative
     * at each sample position to the linear slope between neighboring positions
     * over their parameter interval.
     *
     * @author tschw
     */

    function CubicInterpolant( parameterPositions, sampleValues, sampleSize, resultBuffer ) {

    	Interpolant.call( this, parameterPositions, sampleValues, sampleSize, resultBuffer );

    	this._weightPrev = - 0;
    	this._offsetPrev = - 0;
    	this._weightNext = - 0;
    	this._offsetNext = - 0;

    }

    CubicInterpolant.prototype = Object.assign( Object.create( Interpolant.prototype ), {

    	constructor: CubicInterpolant,

    	DefaultSettings_: {

    		endingStart: ZeroCurvatureEnding,
    		endingEnd: ZeroCurvatureEnding

    	},

    	intervalChanged_: function ( i1, t0, t1 ) {

    		var pp = this.parameterPositions,
    			iPrev = i1 - 2,
    			iNext = i1 + 1,

    			tPrev = pp[ iPrev ],
    			tNext = pp[ iNext ];

    		if ( tPrev === undefined ) {

    			switch ( this.getSettings_().endingStart ) {

    				case ZeroSlopeEnding:

    					// f'(t0) = 0
    					iPrev = i1;
    					tPrev = 2 * t0 - t1;

    					break;

    				case WrapAroundEnding:

    					// use the other end of the curve
    					iPrev = pp.length - 2;
    					tPrev = t0 + pp[ iPrev ] - pp[ iPrev + 1 ];

    					break;

    				default: // ZeroCurvatureEnding

    					// f''(t0) = 0 a.k.a. Natural Spline
    					iPrev = i1;
    					tPrev = t1;

    			}

    		}

    		if ( tNext === undefined ) {

    			switch ( this.getSettings_().endingEnd ) {

    				case ZeroSlopeEnding:

    					// f'(tN) = 0
    					iNext = i1;
    					tNext = 2 * t1 - t0;

    					break;

    				case WrapAroundEnding:

    					// use the other end of the curve
    					iNext = 1;
    					tNext = t1 + pp[ 1 ] - pp[ 0 ];

    					break;

    				default: // ZeroCurvatureEnding

    					// f''(tN) = 0, a.k.a. Natural Spline
    					iNext = i1 - 1;
    					tNext = t0;

    			}

    		}

    		var halfDt = ( t1 - t0 ) * 0.5,
    			stride = this.valueSize;

    		this._weightPrev = halfDt / ( t0 - tPrev );
    		this._weightNext = halfDt / ( tNext - t1 );
    		this._offsetPrev = iPrev * stride;
    		this._offsetNext = iNext * stride;

    	},

    	interpolate_: function ( i1, t0, t, t1 ) {

    		var result = this.resultBuffer,
    			values = this.sampleValues,
    			stride = this.valueSize,

    			o1 = i1 * stride,		o0 = o1 - stride,
    			oP = this._offsetPrev, 	oN = this._offsetNext,
    			wP = this._weightPrev,	wN = this._weightNext,

    			p = ( t - t0 ) / ( t1 - t0 ),
    			pp = p * p,
    			ppp = pp * p;

    		// evaluate polynomials

    		var sP = - wP * ppp + 2 * wP * pp - wP * p;
    		var s0 = ( 1 + wP ) * ppp + ( - 1.5 - 2 * wP ) * pp + ( - 0.5 + wP ) * p + 1;
    		var s1 = ( - 1 - wN ) * ppp + ( 1.5 + wN ) * pp + 0.5 * p;
    		var sN = wN * ppp - wN * pp;

    		// combine data linearly

    		for ( var i = 0; i !== stride; ++ i ) {

    			result[ i ] =
    					sP * values[ oP + i ] +
    					s0 * values[ o0 + i ] +
    					s1 * values[ o1 + i ] +
    					sN * values[ oN + i ];

    		}

    		return result;

    	}

    } );

    /**
     * @author tschw
     */

    function LinearInterpolant( parameterPositions, sampleValues, sampleSize, resultBuffer ) {

    	Interpolant.call( this, parameterPositions, sampleValues, sampleSize, resultBuffer );

    }

    LinearInterpolant.prototype = Object.assign( Object.create( Interpolant.prototype ), {

    	constructor: LinearInterpolant,

    	interpolate_: function ( i1, t0, t, t1 ) {

    		var result = this.resultBuffer,
    			values = this.sampleValues,
    			stride = this.valueSize,

    			offset1 = i1 * stride,
    			offset0 = offset1 - stride,

    			weight1 = ( t - t0 ) / ( t1 - t0 ),
    			weight0 = 1 - weight1;

    		for ( var i = 0; i !== stride; ++ i ) {

    			result[ i ] =
    					values[ offset0 + i ] * weight0 +
    					values[ offset1 + i ] * weight1;

    		}

    		return result;

    	}

    } );

    /**
     *
     * Interpolant that evaluates to the sample value at the position preceeding
     * the parameter.
     *
     * @author tschw
     */

    function DiscreteInterpolant( parameterPositions, sampleValues, sampleSize, resultBuffer ) {

    	Interpolant.call( this, parameterPositions, sampleValues, sampleSize, resultBuffer );

    }

    DiscreteInterpolant.prototype = Object.assign( Object.create( Interpolant.prototype ), {

    	constructor: DiscreteInterpolant,

    	interpolate_: function ( i1 /*, t0, t, t1 */ ) {

    		return this.copySampleValue_( i1 - 1 );

    	}

    } );

    /**
     *
     * A timed sequence of keyframes for a specific property.
     *
     *
     * @author Ben Houston / http://clara.io/
     * @author David Sarno / http://lighthaus.us/
     * @author tschw
     */

    function KeyframeTrack( name, times, values, interpolation ) {

    	if ( name === undefined ) throw new Error( 'THREE.KeyframeTrack: track name is undefined' );
    	if ( times === undefined || times.length === 0 ) throw new Error( 'THREE.KeyframeTrack: no keyframes in track named ' + name );

    	this.name = name;

    	this.times = AnimationUtils.convertArray( times, this.TimeBufferType );
    	this.values = AnimationUtils.convertArray( values, this.ValueBufferType );

    	this.setInterpolation( interpolation || this.DefaultInterpolation );

    }

    // Static methods

    Object.assign( KeyframeTrack, {

    	// Serialization (in static context, because of constructor invocation
    	// and automatic invocation of .toJSON):

    	toJSON: function ( track ) {

    		var trackType = track.constructor;

    		var json;

    		// derived classes can define a static toJSON method
    		if ( trackType.toJSON !== undefined ) {

    			json = trackType.toJSON( track );

    		} else {

    			// by default, we assume the data can be serialized as-is
    			json = {

    				'name': track.name,
    				'times': AnimationUtils.convertArray( track.times, Array ),
    				'values': AnimationUtils.convertArray( track.values, Array )

    			};

    			var interpolation = track.getInterpolation();

    			if ( interpolation !== track.DefaultInterpolation ) {

    				json.interpolation = interpolation;

    			}

    		}

    		json.type = track.ValueTypeName; // mandatory

    		return json;

    	}

    } );

    Object.assign( KeyframeTrack.prototype, {

    	constructor: KeyframeTrack,

    	TimeBufferType: Float32Array,

    	ValueBufferType: Float32Array,

    	DefaultInterpolation: InterpolateLinear,

    	InterpolantFactoryMethodDiscrete: function ( result ) {

    		return new DiscreteInterpolant( this.times, this.values, this.getValueSize(), result );

    	},

    	InterpolantFactoryMethodLinear: function ( result ) {

    		return new LinearInterpolant( this.times, this.values, this.getValueSize(), result );

    	},

    	InterpolantFactoryMethodSmooth: function ( result ) {

    		return new CubicInterpolant( this.times, this.values, this.getValueSize(), result );

    	},

    	setInterpolation: function ( interpolation ) {

    		var factoryMethod;

    		switch ( interpolation ) {

    			case InterpolateDiscrete:

    				factoryMethod = this.InterpolantFactoryMethodDiscrete;

    				break;

    			case InterpolateLinear:

    				factoryMethod = this.InterpolantFactoryMethodLinear;

    				break;

    			case InterpolateSmooth:

    				factoryMethod = this.InterpolantFactoryMethodSmooth;

    				break;

    		}

    		if ( factoryMethod === undefined ) {

    			var message = "unsupported interpolation for " +
    				this.ValueTypeName + " keyframe track named " + this.name;

    			if ( this.createInterpolant === undefined ) {

    				// fall back to default, unless the default itself is messed up
    				if ( interpolation !== this.DefaultInterpolation ) {

    					this.setInterpolation( this.DefaultInterpolation );

    				} else {

    					throw new Error( message ); // fatal, in this case

    				}

    			}

    			console.warn( 'THREE.KeyframeTrack:', message );
    			return this;

    		}

    		this.createInterpolant = factoryMethod;

    		return this;

    	},

    	getInterpolation: function () {

    		switch ( this.createInterpolant ) {

    			case this.InterpolantFactoryMethodDiscrete:

    				return InterpolateDiscrete;

    			case this.InterpolantFactoryMethodLinear:

    				return InterpolateLinear;

    			case this.InterpolantFactoryMethodSmooth:

    				return InterpolateSmooth;

    		}

    	},

    	getValueSize: function () {

    		return this.values.length / this.times.length;

    	},

    	// move all keyframes either forwards or backwards in time
    	shift: function ( timeOffset ) {

    		if ( timeOffset !== 0.0 ) {

    			var times = this.times;

    			for ( var i = 0, n = times.length; i !== n; ++ i ) {

    				times[ i ] += timeOffset;

    			}

    		}

    		return this;

    	},

    	// scale all keyframe times by a factor (useful for frame <-> seconds conversions)
    	scale: function ( timeScale ) {

    		if ( timeScale !== 1.0 ) {

    			var times = this.times;

    			for ( var i = 0, n = times.length; i !== n; ++ i ) {

    				times[ i ] *= timeScale;

    			}

    		}

    		return this;

    	},

    	// removes keyframes before and after animation without changing any values within the range [startTime, endTime].
    	// IMPORTANT: We do not shift around keys to the start of the track time, because for interpolated keys this will change their values
    	trim: function ( startTime, endTime ) {

    		var times = this.times,
    			nKeys = times.length,
    			from = 0,
    			to = nKeys - 1;

    		while ( from !== nKeys && times[ from ] < startTime ) {

    			++ from;

    		}

    		while ( to !== - 1 && times[ to ] > endTime ) {

    			-- to;

    		}

    		++ to; // inclusive -> exclusive bound

    		if ( from !== 0 || to !== nKeys ) {

    			// empty tracks are forbidden, so keep at least one keyframe
    			if ( from >= to ) to = Math.max( to, 1 ), from = to - 1;

    			var stride = this.getValueSize();
    			this.times = AnimationUtils.arraySlice( times, from, to );
    			this.values = AnimationUtils.arraySlice( this.values, from * stride, to * stride );

    		}

    		return this;

    	},

    	// ensure we do not get a GarbageInGarbageOut situation, make sure tracks are at least minimally viable
    	validate: function () {

    		var valid = true;

    		var valueSize = this.getValueSize();
    		if ( valueSize - Math.floor( valueSize ) !== 0 ) {

    			console.error( 'THREE.KeyframeTrack: Invalid value size in track.', this );
    			valid = false;

    		}

    		var times = this.times,
    			values = this.values,

    			nKeys = times.length;

    		if ( nKeys === 0 ) {

    			console.error( 'THREE.KeyframeTrack: Track is empty.', this );
    			valid = false;

    		}

    		var prevTime = null;

    		for ( var i = 0; i !== nKeys; i ++ ) {

    			var currTime = times[ i ];

    			if ( typeof currTime === 'number' && isNaN( currTime ) ) {

    				console.error( 'THREE.KeyframeTrack: Time is not a valid number.', this, i, currTime );
    				valid = false;
    				break;

    			}

    			if ( prevTime !== null && prevTime > currTime ) {

    				console.error( 'THREE.KeyframeTrack: Out of order keys.', this, i, currTime, prevTime );
    				valid = false;
    				break;

    			}

    			prevTime = currTime;

    		}

    		if ( values !== undefined ) {

    			if ( AnimationUtils.isTypedArray( values ) ) {

    				for ( var i = 0, n = values.length; i !== n; ++ i ) {

    					var value = values[ i ];

    					if ( isNaN( value ) ) {

    						console.error( 'THREE.KeyframeTrack: Value is not a valid number.', this, i, value );
    						valid = false;
    						break;

    					}

    				}

    			}

    		}

    		return valid;

    	},

    	// removes equivalent sequential keys as common in morph target sequences
    	// (0,0,0,0,1,1,1,0,0,0,0,0,0,0) --> (0,0,1,1,0,0)
    	optimize: function () {

    		var times = this.times,
    			values = this.values,
    			stride = this.getValueSize(),

    			smoothInterpolation = this.getInterpolation() === InterpolateSmooth,

    			writeIndex = 1,
    			lastIndex = times.length - 1;

    		for ( var i = 1; i < lastIndex; ++ i ) {

    			var keep = false;

    			var time = times[ i ];
    			var timeNext = times[ i + 1 ];

    			// remove adjacent keyframes scheduled at the same time

    			if ( time !== timeNext && ( i !== 1 || time !== time[ 0 ] ) ) {

    				if ( ! smoothInterpolation ) {

    					// remove unnecessary keyframes same as their neighbors

    					var offset = i * stride,
    						offsetP = offset - stride,
    						offsetN = offset + stride;

    					for ( var j = 0; j !== stride; ++ j ) {

    						var value = values[ offset + j ];

    						if ( value !== values[ offsetP + j ] ||
    							value !== values[ offsetN + j ] ) {

    							keep = true;
    							break;

    						}

    					}

    				} else {

    					keep = true;

    				}

    			}

    			// in-place compaction

    			if ( keep ) {

    				if ( i !== writeIndex ) {

    					times[ writeIndex ] = times[ i ];

    					var readOffset = i * stride,
    						writeOffset = writeIndex * stride;

    					for ( var j = 0; j !== stride; ++ j ) {

    						values[ writeOffset + j ] = values[ readOffset + j ];

    					}

    				}

    				++ writeIndex;

    			}

    		}

    		// flush last keyframe (compaction looks ahead)

    		if ( lastIndex > 0 ) {

    			times[ writeIndex ] = times[ lastIndex ];

    			for ( var readOffset = lastIndex * stride, writeOffset = writeIndex * stride, j = 0; j !== stride; ++ j ) {

    				values[ writeOffset + j ] = values[ readOffset + j ];

    			}

    			++ writeIndex;

    		}

    		if ( writeIndex !== times.length ) {

    			this.times = AnimationUtils.arraySlice( times, 0, writeIndex );
    			this.values = AnimationUtils.arraySlice( values, 0, writeIndex * stride );

    		}

    		return this;

    	},

    	clone: function () {

    		var times = AnimationUtils.arraySlice( this.times, 0 );
    		var values = AnimationUtils.arraySlice( this.values, 0 );

    		var TypedKeyframeTrack = this.constructor;
    		var track = new TypedKeyframeTrack( this.name, times, values );

    		// Interpolant argument to constructor is not saved, so copy the factory method directly.
    		track.createInterpolant = this.createInterpolant;

    		return track;

    	}

    } );

    /**
     *
     * A Track of Boolean keyframe values.
     *
     *
     * @author Ben Houston / http://clara.io/
     * @author David Sarno / http://lighthaus.us/
     * @author tschw
     */

    function BooleanKeyframeTrack( name, times, values ) {

    	KeyframeTrack.call( this, name, times, values );

    }

    BooleanKeyframeTrack.prototype = Object.assign( Object.create( KeyframeTrack.prototype ), {

    	constructor: BooleanKeyframeTrack,

    	ValueTypeName: 'bool',
    	ValueBufferType: Array,

    	DefaultInterpolation: InterpolateDiscrete,

    	InterpolantFactoryMethodLinear: undefined,
    	InterpolantFactoryMethodSmooth: undefined

    	// Note: Actually this track could have a optimized / compressed
    	// representation of a single value and a custom interpolant that
    	// computes "firstValue ^ isOdd( index )".

    } );

    /**
     *
     * A Track of keyframe values that represent color.
     *
     *
     * @author Ben Houston / http://clara.io/
     * @author David Sarno / http://lighthaus.us/
     * @author tschw
     */

    function ColorKeyframeTrack( name, times, values, interpolation ) {

    	KeyframeTrack.call( this, name, times, values, interpolation );

    }

    ColorKeyframeTrack.prototype = Object.assign( Object.create( KeyframeTrack.prototype ), {

    	constructor: ColorKeyframeTrack,

    	ValueTypeName: 'color'

    	// ValueBufferType is inherited

    	// DefaultInterpolation is inherited

    	// Note: Very basic implementation and nothing special yet.
    	// However, this is the place for color space parameterization.

    } );

    /**
     *
     * A Track of numeric keyframe values.
     *
     * @author Ben Houston / http://clara.io/
     * @author David Sarno / http://lighthaus.us/
     * @author tschw
     */

    function NumberKeyframeTrack( name, times, values, interpolation ) {

    	KeyframeTrack.call( this, name, times, values, interpolation );

    }

    NumberKeyframeTrack.prototype = Object.assign( Object.create( KeyframeTrack.prototype ), {

    	constructor: NumberKeyframeTrack,

    	ValueTypeName: 'number'

    	// ValueBufferType is inherited

    	// DefaultInterpolation is inherited

    } );

    /**
     * Spherical linear unit quaternion interpolant.
     *
     * @author tschw
     */

    function QuaternionLinearInterpolant( parameterPositions, sampleValues, sampleSize, resultBuffer ) {

    	Interpolant.call( this, parameterPositions, sampleValues, sampleSize, resultBuffer );

    }

    QuaternionLinearInterpolant.prototype = Object.assign( Object.create( Interpolant.prototype ), {

    	constructor: QuaternionLinearInterpolant,

    	interpolate_: function ( i1, t0, t, t1 ) {

    		var result = this.resultBuffer,
    			values = this.sampleValues,
    			stride = this.valueSize,

    			offset = i1 * stride,

    			alpha = ( t - t0 ) / ( t1 - t0 );

    		for ( var end = offset + stride; offset !== end; offset += 4 ) {

    			Quaternion.slerpFlat( result, 0, values, offset - stride, values, offset, alpha );

    		}

    		return result;

    	}

    } );

    /**
     *
     * A Track of quaternion keyframe values.
     *
     * @author Ben Houston / http://clara.io/
     * @author David Sarno / http://lighthaus.us/
     * @author tschw
     */

    function QuaternionKeyframeTrack( name, times, values, interpolation ) {

    	KeyframeTrack.call( this, name, times, values, interpolation );

    }

    QuaternionKeyframeTrack.prototype = Object.assign( Object.create( KeyframeTrack.prototype ), {

    	constructor: QuaternionKeyframeTrack,

    	ValueTypeName: 'quaternion',

    	// ValueBufferType is inherited

    	DefaultInterpolation: InterpolateLinear,

    	InterpolantFactoryMethodLinear: function ( result ) {

    		return new QuaternionLinearInterpolant( this.times, this.values, this.getValueSize(), result );

    	},

    	InterpolantFactoryMethodSmooth: undefined // not yet implemented

    } );

    /**
     *
     * A Track that interpolates Strings
     *
     *
     * @author Ben Houston / http://clara.io/
     * @author David Sarno / http://lighthaus.us/
     * @author tschw
     */

    function StringKeyframeTrack( name, times, values, interpolation ) {

    	KeyframeTrack.call( this, name, times, values, interpolation );

    }

    StringKeyframeTrack.prototype = Object.assign( Object.create( KeyframeTrack.prototype ), {

    	constructor: StringKeyframeTrack,

    	ValueTypeName: 'string',
    	ValueBufferType: Array,

    	DefaultInterpolation: InterpolateDiscrete,

    	InterpolantFactoryMethodLinear: undefined,

    	InterpolantFactoryMethodSmooth: undefined

    } );

    /**
     *
     * A Track of vectored keyframe values.
     *
     *
     * @author Ben Houston / http://clara.io/
     * @author David Sarno / http://lighthaus.us/
     * @author tschw
     */

    function VectorKeyframeTrack( name, times, values, interpolation ) {

    	KeyframeTrack.call( this, name, times, values, interpolation );

    }

    VectorKeyframeTrack.prototype = Object.assign( Object.create( KeyframeTrack.prototype ), {

    	constructor: VectorKeyframeTrack,

    	ValueTypeName: 'vector'

    	// ValueBufferType is inherited

    	// DefaultInterpolation is inherited

    } );

    /**
     *
     * Reusable set of Tracks that represent an animation.
     *
     * @author Ben Houston / http://clara.io/
     * @author David Sarno / http://lighthaus.us/
     */

    function AnimationClip( name, duration, tracks ) {

    	this.name = name;
    	this.tracks = tracks;
    	this.duration = ( duration !== undefined ) ? duration : - 1;

    	this.uuid = _Math.generateUUID();

    	// this means it should figure out its duration by scanning the tracks
    	if ( this.duration < 0 ) {

    		this.resetDuration();

    	}

    }

    function getTrackTypeForValueTypeName( typeName ) {

    	switch ( typeName.toLowerCase() ) {

    		case 'scalar':
    		case 'double':
    		case 'float':
    		case 'number':
    		case 'integer':

    			return NumberKeyframeTrack;

    		case 'vector':
    		case 'vector2':
    		case 'vector3':
    		case 'vector4':

    			return VectorKeyframeTrack;

    		case 'color':

    			return ColorKeyframeTrack;

    		case 'quaternion':

    			return QuaternionKeyframeTrack;

    		case 'bool':
    		case 'boolean':

    			return BooleanKeyframeTrack;

    		case 'string':

    			return StringKeyframeTrack;

    	}

    	throw new Error( 'THREE.KeyframeTrack: Unsupported typeName: ' + typeName );

    }

    function parseKeyframeTrack( json ) {

    	if ( json.type === undefined ) {

    		throw new Error( 'THREE.KeyframeTrack: track type undefined, can not parse' );

    	}

    	var trackType = getTrackTypeForValueTypeName( json.type );

    	if ( json.times === undefined ) {

    		var times = [], values = [];

    		AnimationUtils.flattenJSON( json.keys, times, values, 'value' );

    		json.times = times;
    		json.values = values;

    	}

    	// derived classes can define a static parse method
    	if ( trackType.parse !== undefined ) {

    		return trackType.parse( json );

    	} else {

    		// by default, we assume a constructor compatible with the base
    		return new trackType( json.name, json.times, json.values, json.interpolation );

    	}

    }

    Object.assign( AnimationClip, {

    	parse: function ( json ) {

    		var tracks = [],
    			jsonTracks = json.tracks,
    			frameTime = 1.0 / ( json.fps || 1.0 );

    		for ( var i = 0, n = jsonTracks.length; i !== n; ++ i ) {

    			tracks.push( parseKeyframeTrack( jsonTracks[ i ] ).scale( frameTime ) );

    		}

    		return new AnimationClip( json.name, json.duration, tracks );

    	},

    	toJSON: function ( clip ) {

    		var tracks = [],
    			clipTracks = clip.tracks;

    		var json = {

    			'name': clip.name,
    			'duration': clip.duration,
    			'tracks': tracks,
    			'uuid': clip.uuid

    		};

    		for ( var i = 0, n = clipTracks.length; i !== n; ++ i ) {

    			tracks.push( KeyframeTrack.toJSON( clipTracks[ i ] ) );

    		}

    		return json;

    	},

    	CreateFromMorphTargetSequence: function ( name, morphTargetSequence, fps, noLoop ) {

    		var numMorphTargets = morphTargetSequence.length;
    		var tracks = [];

    		for ( var i = 0; i < numMorphTargets; i ++ ) {

    			var times = [];
    			var values = [];

    			times.push(
    				( i + numMorphTargets - 1 ) % numMorphTargets,
    				i,
    				( i + 1 ) % numMorphTargets );

    			values.push( 0, 1, 0 );

    			var order = AnimationUtils.getKeyframeOrder( times );
    			times = AnimationUtils.sortedArray( times, 1, order );
    			values = AnimationUtils.sortedArray( values, 1, order );

    			// if there is a key at the first frame, duplicate it as the
    			// last frame as well for perfect loop.
    			if ( ! noLoop && times[ 0 ] === 0 ) {

    				times.push( numMorphTargets );
    				values.push( values[ 0 ] );

    			}

    			tracks.push(
    				new NumberKeyframeTrack(
    					'.morphTargetInfluences[' + morphTargetSequence[ i ].name + ']',
    					times, values
    				).scale( 1.0 / fps ) );

    		}

    		return new AnimationClip( name, - 1, tracks );

    	},

    	findByName: function ( objectOrClipArray, name ) {

    		var clipArray = objectOrClipArray;

    		if ( ! Array.isArray( objectOrClipArray ) ) {

    			var o = objectOrClipArray;
    			clipArray = o.geometry && o.geometry.animations || o.animations;

    		}

    		for ( var i = 0; i < clipArray.length; i ++ ) {

    			if ( clipArray[ i ].name === name ) {

    				return clipArray[ i ];

    			}

    		}

    		return null;

    	},

    	CreateClipsFromMorphTargetSequences: function ( morphTargets, fps, noLoop ) {

    		var animationToMorphTargets = {};

    		// tested with https://regex101.com/ on trick sequences
    		// such flamingo_flyA_003, flamingo_run1_003, crdeath0059
    		var pattern = /^([\w-]*?)([\d]+)$/;

    		// sort morph target names into animation groups based
    		// patterns like Walk_001, Walk_002, Run_001, Run_002
    		for ( var i = 0, il = morphTargets.length; i < il; i ++ ) {

    			var morphTarget = morphTargets[ i ];
    			var parts = morphTarget.name.match( pattern );

    			if ( parts && parts.length > 1 ) {

    				var name = parts[ 1 ];

    				var animationMorphTargets = animationToMorphTargets[ name ];
    				if ( ! animationMorphTargets ) {

    					animationToMorphTargets[ name ] = animationMorphTargets = [];

    				}

    				animationMorphTargets.push( morphTarget );

    			}

    		}

    		var clips = [];

    		for ( var name in animationToMorphTargets ) {

    			clips.push( AnimationClip.CreateFromMorphTargetSequence( name, animationToMorphTargets[ name ], fps, noLoop ) );

    		}

    		return clips;

    	},

    	// parse the animation.hierarchy format
    	parseAnimation: function ( animation, bones ) {

    		if ( ! animation ) {

    			console.error( 'THREE.AnimationClip: No animation in JSONLoader data.' );
    			return null;

    		}

    		var addNonemptyTrack = function ( trackType, trackName, animationKeys, propertyName, destTracks ) {

    			// only return track if there are actually keys.
    			if ( animationKeys.length !== 0 ) {

    				var times = [];
    				var values = [];

    				AnimationUtils.flattenJSON( animationKeys, times, values, propertyName );

    				// empty keys are filtered out, so check again
    				if ( times.length !== 0 ) {

    					destTracks.push( new trackType( trackName, times, values ) );

    				}

    			}

    		};

    		var tracks = [];

    		var clipName = animation.name || 'default';
    		// automatic length determination in AnimationClip.
    		var duration = animation.length || - 1;
    		var fps = animation.fps || 30;

    		var hierarchyTracks = animation.hierarchy || [];

    		for ( var h = 0; h < hierarchyTracks.length; h ++ ) {

    			var animationKeys = hierarchyTracks[ h ].keys;

    			// skip empty tracks
    			if ( ! animationKeys || animationKeys.length === 0 ) continue;

    			// process morph targets
    			if ( animationKeys[ 0 ].morphTargets ) {

    				// figure out all morph targets used in this track
    				var morphTargetNames = {};

    				for ( var k = 0; k < animationKeys.length; k ++ ) {

    					if ( animationKeys[ k ].morphTargets ) {

    						for ( var m = 0; m < animationKeys[ k ].morphTargets.length; m ++ ) {

    							morphTargetNames[ animationKeys[ k ].morphTargets[ m ] ] = - 1;

    						}

    					}

    				}

    				// create a track for each morph target with all zero
    				// morphTargetInfluences except for the keys in which
    				// the morphTarget is named.
    				for ( var morphTargetName in morphTargetNames ) {

    					var times = [];
    					var values = [];

    					for ( var m = 0; m !== animationKeys[ k ].morphTargets.length; ++ m ) {

    						var animationKey = animationKeys[ k ];

    						times.push( animationKey.time );
    						values.push( ( animationKey.morphTarget === morphTargetName ) ? 1 : 0 );

    					}

    					tracks.push( new NumberKeyframeTrack( '.morphTargetInfluence[' + morphTargetName + ']', times, values ) );

    				}

    				duration = morphTargetNames.length * ( fps || 1.0 );

    			} else {

    				// ...assume skeletal animation

    				var boneName = '.bones[' + bones[ h ].name + ']';

    				addNonemptyTrack(
    					VectorKeyframeTrack, boneName + '.position',
    					animationKeys, 'pos', tracks );

    				addNonemptyTrack(
    					QuaternionKeyframeTrack, boneName + '.quaternion',
    					animationKeys, 'rot', tracks );

    				addNonemptyTrack(
    					VectorKeyframeTrack, boneName + '.scale',
    					animationKeys, 'scl', tracks );

    			}

    		}

    		if ( tracks.length === 0 ) {

    			return null;

    		}

    		var clip = new AnimationClip( clipName, duration, tracks );

    		return clip;

    	}

    } );

    Object.assign( AnimationClip.prototype, {

    	resetDuration: function () {

    		var tracks = this.tracks, duration = 0;

    		for ( var i = 0, n = tracks.length; i !== n; ++ i ) {

    			var track = this.tracks[ i ];

    			duration = Math.max( duration, track.times[ track.times.length - 1 ] );

    		}

    		this.duration = duration;

    		return this;

    	},

    	trim: function () {

    		for ( var i = 0; i < this.tracks.length; i ++ ) {

    			this.tracks[ i ].trim( 0, this.duration );

    		}

    		return this;

    	},

    	validate: function () {

    		var valid = true;

    		for ( var i = 0; i < this.tracks.length; i ++ ) {

    			valid = valid && this.tracks[ i ].validate();

    		}

    		return valid;

    	},

    	optimize: function () {

    		for ( var i = 0; i < this.tracks.length; i ++ ) {

    			this.tracks[ i ].optimize();

    		}

    		return this;

    	},


    	clone: function () {

    		var tracks = [];

    		for ( var i = 0; i < this.tracks.length; i ++ ) {

    			tracks.push( this.tracks[ i ].clone() );

    		}

    		return new AnimationClip( this.name, this.duration, tracks );

    	}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    var Cache = {

    	enabled: false,

    	files: {},

    	add: function ( key, file ) {

    		if ( this.enabled === false ) return;

    		// console.log( 'THREE.Cache', 'Adding key:', key );

    		this.files[ key ] = file;

    	},

    	get: function ( key ) {

    		if ( this.enabled === false ) return;

    		// console.log( 'THREE.Cache', 'Checking key:', key );

    		return this.files[ key ];

    	},

    	remove: function ( key ) {

    		delete this.files[ key ];

    	},

    	clear: function () {

    		this.files = {};

    	}

    };

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function LoadingManager( onLoad, onProgress, onError ) {

    	var scope = this;

    	var isLoading = false;
    	var itemsLoaded = 0;
    	var itemsTotal = 0;
    	var urlModifier = undefined;

    	// Refer to #5689 for the reason why we don't set .onStart
    	// in the constructor

    	this.onStart = undefined;
    	this.onLoad = onLoad;
    	this.onProgress = onProgress;
    	this.onError = onError;

    	this.itemStart = function ( url ) {

    		itemsTotal ++;

    		if ( isLoading === false ) {

    			if ( scope.onStart !== undefined ) {

    				scope.onStart( url, itemsLoaded, itemsTotal );

    			}

    		}

    		isLoading = true;

    	};

    	this.itemEnd = function ( url ) {

    		itemsLoaded ++;

    		if ( scope.onProgress !== undefined ) {

    			scope.onProgress( url, itemsLoaded, itemsTotal );

    		}

    		if ( itemsLoaded === itemsTotal ) {

    			isLoading = false;

    			if ( scope.onLoad !== undefined ) {

    				scope.onLoad();

    			}

    		}

    	};

    	this.itemError = function ( url ) {

    		if ( scope.onError !== undefined ) {

    			scope.onError( url );

    		}

    	};

    	this.resolveURL = function ( url ) {

    		if ( urlModifier ) {

    			return urlModifier( url );

    		}

    		return url;

    	};

    	this.setURLModifier = function ( transform ) {

    		urlModifier = transform;
    		return this;

    	};

    }

    var DefaultLoadingManager = new LoadingManager();

    /**
     * @author alteredq / http://alteredqualia.com/
     */

    function Loader( manager ) {

    	this.manager = ( manager !== undefined ) ? manager : DefaultLoadingManager;

    	this.crossOrigin = 'anonymous';
    	this.path = '';
    	this.resourcePath = '';

    }

    Object.assign( Loader.prototype, {

    	load: function ( /* url, onLoad, onProgress, onError */ ) {},

    	parse: function ( /* data */ ) {},

    	setCrossOrigin: function ( crossOrigin ) {

    		this.crossOrigin = crossOrigin;
    		return this;

    	},

    	setPath: function ( path ) {

    		this.path = path;
    		return this;

    	},

    	setResourcePath: function ( resourcePath ) {

    		this.resourcePath = resourcePath;
    		return this;

    	}

    } );

    //

    Loader.Handlers = {

    	handlers: [],

    	add: function ( regex, loader ) {

    		this.handlers.push( regex, loader );

    	},

    	get: function ( file ) {

    		var handlers = this.handlers;

    		for ( var i = 0, l = handlers.length; i < l; i += 2 ) {

    			var regex = handlers[ i ];
    			var loader = handlers[ i + 1 ];

    			if ( regex.test( file ) ) {

    				return loader;

    			}

    		}

    		return null;

    	}

    };

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    var loading = {};

    function FileLoader( manager ) {

    	Loader.call( this, manager );

    }

    FileLoader.prototype = Object.assign( Object.create( Loader.prototype ), {

    	constructor: FileLoader,

    	load: function ( url, onLoad, onProgress, onError ) {

    		if ( url === undefined ) url = '';

    		if ( this.path !== undefined ) url = this.path + url;

    		url = this.manager.resolveURL( url );

    		var scope = this;

    		var cached = Cache.get( url );

    		if ( cached !== undefined ) {

    			scope.manager.itemStart( url );

    			setTimeout( function () {

    				if ( onLoad ) onLoad( cached );

    				scope.manager.itemEnd( url );

    			}, 0 );

    			return cached;

    		}

    		// Check if request is duplicate

    		if ( loading[ url ] !== undefined ) {

    			loading[ url ].push( {

    				onLoad: onLoad,
    				onProgress: onProgress,
    				onError: onError

    			} );

    			return;

    		}

    		// Check for data: URI
    		var dataUriRegex = /^data:(.*?)(;base64)?,(.*)$/;
    		var dataUriRegexResult = url.match( dataUriRegex );

    		// Safari can not handle Data URIs through XMLHttpRequest so process manually
    		if ( dataUriRegexResult ) {

    			var mimeType = dataUriRegexResult[ 1 ];
    			var isBase64 = !! dataUriRegexResult[ 2 ];
    			var data = dataUriRegexResult[ 3 ];

    			data = decodeURIComponent( data );

    			if ( isBase64 ) data = atob( data );

    			try {

    				var response;
    				var responseType = ( this.responseType || '' ).toLowerCase();

    				switch ( responseType ) {

    					case 'arraybuffer':
    					case 'blob':

    						var view = new Uint8Array( data.length );

    						for ( var i = 0; i < data.length; i ++ ) {

    							view[ i ] = data.charCodeAt( i );

    						}

    						if ( responseType === 'blob' ) {

    							response = new Blob( [ view.buffer ], { type: mimeType } );

    						} else {

    							response = view.buffer;

    						}

    						break;

    					case 'document':

    						var parser = new DOMParser();
    						response = parser.parseFromString( data, mimeType );

    						break;

    					case 'json':

    						response = JSON.parse( data );

    						break;

    					default: // 'text' or other

    						response = data;

    						break;

    				}

    				// Wait for next browser tick like standard XMLHttpRequest event dispatching does
    				setTimeout( function () {

    					if ( onLoad ) onLoad( response );

    					scope.manager.itemEnd( url );

    				}, 0 );

    			} catch ( error ) {

    				// Wait for next browser tick like standard XMLHttpRequest event dispatching does
    				setTimeout( function () {

    					if ( onError ) onError( error );

    					scope.manager.itemError( url );
    					scope.manager.itemEnd( url );

    				}, 0 );

    			}

    		} else {

    			// Initialise array for duplicate requests

    			loading[ url ] = [];

    			loading[ url ].push( {

    				onLoad: onLoad,
    				onProgress: onProgress,
    				onError: onError

    			} );

    			var request = new XMLHttpRequest();

    			request.open( 'GET', url, true );

    			request.addEventListener( 'load', function ( event ) {

    				var response = this.response;

    				Cache.add( url, response );

    				var callbacks = loading[ url ];

    				delete loading[ url ];

    				if ( this.status === 200 || this.status === 0 ) {

    					// Some browsers return HTTP Status 0 when using non-http protocol
    					// e.g. 'file://' or 'data://'. Handle as success.

    					if ( this.status === 0 ) console.warn( 'THREE.FileLoader: HTTP Status 0 received.' );

    					for ( var i = 0, il = callbacks.length; i < il; i ++ ) {

    						var callback = callbacks[ i ];
    						if ( callback.onLoad ) callback.onLoad( response );

    					}

    					scope.manager.itemEnd( url );

    				} else {

    					for ( var i = 0, il = callbacks.length; i < il; i ++ ) {

    						var callback = callbacks[ i ];
    						if ( callback.onError ) callback.onError( event );

    					}

    					scope.manager.itemError( url );
    					scope.manager.itemEnd( url );

    				}

    			}, false );

    			request.addEventListener( 'progress', function ( event ) {

    				var callbacks = loading[ url ];

    				for ( var i = 0, il = callbacks.length; i < il; i ++ ) {

    					var callback = callbacks[ i ];
    					if ( callback.onProgress ) callback.onProgress( event );

    				}

    			}, false );

    			request.addEventListener( 'error', function ( event ) {

    				var callbacks = loading[ url ];

    				delete loading[ url ];

    				for ( var i = 0, il = callbacks.length; i < il; i ++ ) {

    					var callback = callbacks[ i ];
    					if ( callback.onError ) callback.onError( event );

    				}

    				scope.manager.itemError( url );
    				scope.manager.itemEnd( url );

    			}, false );

    			request.addEventListener( 'abort', function ( event ) {

    				var callbacks = loading[ url ];

    				delete loading[ url ];

    				for ( var i = 0, il = callbacks.length; i < il; i ++ ) {

    					var callback = callbacks[ i ];
    					if ( callback.onError ) callback.onError( event );

    				}

    				scope.manager.itemError( url );
    				scope.manager.itemEnd( url );

    			}, false );

    			if ( this.responseType !== undefined ) request.responseType = this.responseType;
    			if ( this.withCredentials !== undefined ) request.withCredentials = this.withCredentials;

    			if ( request.overrideMimeType ) request.overrideMimeType( this.mimeType !== undefined ? this.mimeType : 'text/plain' );

    			for ( var header in this.requestHeader ) {

    				request.setRequestHeader( header, this.requestHeader[ header ] );

    			}

    			request.send( null );

    		}

    		scope.manager.itemStart( url );

    		return request;

    	},

    	setResponseType: function ( value ) {

    		this.responseType = value;
    		return this;

    	},

    	setWithCredentials: function ( value ) {

    		this.withCredentials = value;
    		return this;

    	},

    	setMimeType: function ( value ) {

    		this.mimeType = value;
    		return this;

    	},

    	setRequestHeader: function ( value ) {

    		this.requestHeader = value;
    		return this;

    	}

    } );

    /**
     * @author bhouston / http://clara.io/
     */

    function AnimationLoader( manager ) {

    	Loader.call( this, manager );

    }

    AnimationLoader.prototype = Object.assign( Object.create( Loader.prototype ), {

    	constructor: AnimationLoader,

    	load: function ( url, onLoad, onProgress, onError ) {

    		var scope = this;

    		var loader = new FileLoader( scope.manager );
    		loader.setPath( scope.path );
    		loader.load( url, function ( text ) {

    			onLoad( scope.parse( JSON.parse( text ) ) );

    		}, onProgress, onError );

    	},

    	parse: function ( json ) {

    		var animations = [];

    		for ( var i = 0; i < json.length; i ++ ) {

    			var clip = AnimationClip.parse( json[ i ] );

    			animations.push( clip );

    		}

    		return animations;

    	}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     *
     * Abstract Base class to block based textures loader (dds, pvr, ...)
     */

    function CompressedTextureLoader( manager ) {

    	Loader.call( this, manager );

    	// override in sub classes
    	this._parser = null;

    }

    CompressedTextureLoader.prototype = Object.assign( Object.create( Loader.prototype ), {

    	constructor: CompressedTextureLoader,

    	load: function ( url, onLoad, onProgress, onError ) {

    		var scope = this;

    		var images = [];

    		var texture = new CompressedTexture();
    		texture.image = images;

    		var loader = new FileLoader( this.manager );
    		loader.setPath( this.path );
    		loader.setResponseType( 'arraybuffer' );

    		function loadTexture( i ) {

    			loader.load( url[ i ], function ( buffer ) {

    				var texDatas = scope._parser( buffer, true );

    				images[ i ] = {
    					width: texDatas.width,
    					height: texDatas.height,
    					format: texDatas.format,
    					mipmaps: texDatas.mipmaps
    				};

    				loaded += 1;

    				if ( loaded === 6 ) {

    					if ( texDatas.mipmapCount === 1 )
    						texture.minFilter = LinearFilter;

    					texture.format = texDatas.format;
    					texture.needsUpdate = true;

    					if ( onLoad ) onLoad( texture );

    				}

    			}, onProgress, onError );

    		}

    		if ( Array.isArray( url ) ) {

    			var loaded = 0;

    			for ( var i = 0, il = url.length; i < il; ++ i ) {

    				loadTexture( i );

    			}

    		} else {

    			// compressed cubemap texture stored in a single DDS file

    			loader.load( url, function ( buffer ) {

    				var texDatas = scope._parser( buffer, true );

    				if ( texDatas.isCubemap ) {

    					var faces = texDatas.mipmaps.length / texDatas.mipmapCount;

    					for ( var f = 0; f < faces; f ++ ) {

    						images[ f ] = { mipmaps: [] };

    						for ( var i = 0; i < texDatas.mipmapCount; i ++ ) {

    							images[ f ].mipmaps.push( texDatas.mipmaps[ f * texDatas.mipmapCount + i ] );
    							images[ f ].format = texDatas.format;
    							images[ f ].width = texDatas.width;
    							images[ f ].height = texDatas.height;

    						}

    					}

    				} else {

    					texture.image.width = texDatas.width;
    					texture.image.height = texDatas.height;
    					texture.mipmaps = texDatas.mipmaps;

    				}

    				if ( texDatas.mipmapCount === 1 ) {

    					texture.minFilter = LinearFilter;

    				}

    				texture.format = texDatas.format;
    				texture.needsUpdate = true;

    				if ( onLoad ) onLoad( texture );

    			}, onProgress, onError );

    		}

    		return texture;

    	}

    } );

    /**
     * @author Nikos M. / https://github.com/foo123/
     *
     * Abstract Base class to load generic binary textures formats (rgbe, hdr, ...)
     */

    function DataTextureLoader( manager ) {

    	Loader.call( this, manager );

    	// override in sub classes
    	this._parser = null;

    }

    DataTextureLoader.prototype = Object.assign( Object.create( Loader.prototype ), {

    	constructor: DataTextureLoader,

    	load: function ( url, onLoad, onProgress, onError ) {

    		var scope = this;

    		var texture = new DataTexture();

    		var loader = new FileLoader( this.manager );
    		loader.setResponseType( 'arraybuffer' );
    		loader.setPath( this.path );
    		loader.load( url, function ( buffer ) {

    			var texData = scope._parser( buffer );

    			if ( ! texData ) return;

    			if ( texData.image !== undefined ) {

    				texture.image = texData.image;

    			} else if ( texData.data !== undefined ) {

    				texture.image.width = texData.width;
    				texture.image.height = texData.height;
    				texture.image.data = texData.data;

    			}

    			texture.wrapS = texData.wrapS !== undefined ? texData.wrapS : ClampToEdgeWrapping;
    			texture.wrapT = texData.wrapT !== undefined ? texData.wrapT : ClampToEdgeWrapping;

    			texture.magFilter = texData.magFilter !== undefined ? texData.magFilter : LinearFilter;
    			texture.minFilter = texData.minFilter !== undefined ? texData.minFilter : LinearMipmapLinearFilter;

    			texture.anisotropy = texData.anisotropy !== undefined ? texData.anisotropy : 1;

    			if ( texData.format !== undefined ) {

    				texture.format = texData.format;

    			}
    			if ( texData.type !== undefined ) {

    				texture.type = texData.type;

    			}

    			if ( texData.mipmaps !== undefined ) {

    				texture.mipmaps = texData.mipmaps;

    			}

    			if ( texData.mipmapCount === 1 ) {

    				texture.minFilter = LinearFilter;

    			}

    			texture.needsUpdate = true;

    			if ( onLoad ) onLoad( texture, texData );

    		}, onProgress, onError );


    		return texture;

    	}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function ImageLoader( manager ) {

    	Loader.call( this, manager );

    }

    ImageLoader.prototype = Object.assign( Object.create( Loader.prototype ), {

    	constructor: ImageLoader,

    	load: function ( url, onLoad, onProgress, onError ) {

    		if ( this.path !== undefined ) url = this.path + url;

    		url = this.manager.resolveURL( url );

    		var scope = this;

    		var cached = Cache.get( url );

    		if ( cached !== undefined ) {

    			scope.manager.itemStart( url );

    			setTimeout( function () {

    				if ( onLoad ) onLoad( cached );

    				scope.manager.itemEnd( url );

    			}, 0 );

    			return cached;

    		}

    		var image = document.createElementNS( 'http://www.w3.org/1999/xhtml', 'img' );

    		function onImageLoad() {

    			image.removeEventListener( 'load', onImageLoad, false );
    			image.removeEventListener( 'error', onImageError, false );

    			Cache.add( url, this );

    			if ( onLoad ) onLoad( this );

    			scope.manager.itemEnd( url );

    		}

    		function onImageError( event ) {

    			image.removeEventListener( 'load', onImageLoad, false );
    			image.removeEventListener( 'error', onImageError, false );

    			if ( onError ) onError( event );

    			scope.manager.itemError( url );
    			scope.manager.itemEnd( url );

    		}

    		image.addEventListener( 'load', onImageLoad, false );
    		image.addEventListener( 'error', onImageError, false );

    		if ( url.substr( 0, 5 ) !== 'data:' ) {

    			if ( this.crossOrigin !== undefined ) image.crossOrigin = this.crossOrigin;

    		}

    		scope.manager.itemStart( url );

    		image.src = url;

    		return image;

    	}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     */


    function CubeTextureLoader( manager ) {

    	Loader.call( this, manager );

    }

    CubeTextureLoader.prototype = Object.assign( Object.create( Loader.prototype ), {

    	constructor: CubeTextureLoader,

    	load: function ( urls, onLoad, onProgress, onError ) {

    		var texture = new CubeTexture();

    		var loader = new ImageLoader( this.manager );
    		loader.setCrossOrigin( this.crossOrigin );
    		loader.setPath( this.path );

    		var loaded = 0;

    		function loadTexture( i ) {

    			loader.load( urls[ i ], function ( image ) {

    				texture.images[ i ] = image;

    				loaded ++;

    				if ( loaded === 6 ) {

    					texture.needsUpdate = true;

    					if ( onLoad ) onLoad( texture );

    				}

    			}, undefined, onError );

    		}

    		for ( var i = 0; i < urls.length; ++ i ) {

    			loadTexture( i );

    		}

    		return texture;

    	}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function TextureLoader( manager ) {

    	Loader.call( this, manager );

    }

    TextureLoader.prototype = Object.assign( Object.create( Loader.prototype ), {

    	constructor: TextureLoader,

    	load: function ( url, onLoad, onProgress, onError ) {

    		var texture = new Texture();

    		var loader = new ImageLoader( this.manager );
    		loader.setCrossOrigin( this.crossOrigin );
    		loader.setPath( this.path );

    		loader.load( url, function ( image ) {

    			texture.image = image;

    			// JPEGs can't have an alpha channel, so memory can be saved by storing them as RGB.
    			var isJPEG = url.search( /\.jpe?g($|\?)/i ) > 0 || url.search( /^data\:image\/jpeg/ ) === 0;

    			texture.format = isJPEG ? RGBFormat : RGBAFormat;
    			texture.needsUpdate = true;

    			if ( onLoad !== undefined ) {

    				onLoad( texture );

    			}

    		}, onProgress, onError );

    		return texture;

    	}

    } );

    /**
     * @author zz85 / http://www.lab4games.net/zz85/blog
     * Extensible curve object
     *
     * Some common of curve methods:
     * .getPoint( t, optionalTarget ), .getTangent( t )
     * .getPointAt( u, optionalTarget ), .getTangentAt( u )
     * .getPoints(), .getSpacedPoints()
     * .getLength()
     * .updateArcLengths()
     *
     * This following curves inherit from THREE.Curve:
     *
     * -- 2D curves --
     * THREE.ArcCurve
     * THREE.CubicBezierCurve
     * THREE.EllipseCurve
     * THREE.LineCurve
     * THREE.QuadraticBezierCurve
     * THREE.SplineCurve
     *
     * -- 3D curves --
     * THREE.CatmullRomCurve3
     * THREE.CubicBezierCurve3
     * THREE.LineCurve3
     * THREE.QuadraticBezierCurve3
     *
     * A series of curves can be represented as a THREE.CurvePath.
     *
     **/

    /**************************************************************
     *	Abstract Curve base class
     **************************************************************/

    function Curve() {

    	this.type = 'Curve';

    	this.arcLengthDivisions = 200;

    }

    Object.assign( Curve.prototype, {

    	// Virtual base class method to overwrite and implement in subclasses
    	//	- t [0 .. 1]

    	getPoint: function ( /* t, optionalTarget */ ) {

    		console.warn( 'THREE.Curve: .getPoint() not implemented.' );
    		return null;

    	},

    	// Get point at relative position in curve according to arc length
    	// - u [0 .. 1]

    	getPointAt: function ( u, optionalTarget ) {

    		var t = this.getUtoTmapping( u );
    		return this.getPoint( t, optionalTarget );

    	},

    	// Get sequence of points using getPoint( t )

    	getPoints: function ( divisions ) {

    		if ( divisions === undefined ) divisions = 5;

    		var points = [];

    		for ( var d = 0; d <= divisions; d ++ ) {

    			points.push( this.getPoint( d / divisions ) );

    		}

    		return points;

    	},

    	// Get sequence of points using getPointAt( u )

    	getSpacedPoints: function ( divisions ) {

    		if ( divisions === undefined ) divisions = 5;

    		var points = [];

    		for ( var d = 0; d <= divisions; d ++ ) {

    			points.push( this.getPointAt( d / divisions ) );

    		}

    		return points;

    	},

    	// Get total curve arc length

    	getLength: function () {

    		var lengths = this.getLengths();
    		return lengths[ lengths.length - 1 ];

    	},

    	// Get list of cumulative segment lengths

    	getLengths: function ( divisions ) {

    		if ( divisions === undefined ) divisions = this.arcLengthDivisions;

    		if ( this.cacheArcLengths &&
    			( this.cacheArcLengths.length === divisions + 1 ) &&
    			! this.needsUpdate ) {

    			return this.cacheArcLengths;

    		}

    		this.needsUpdate = false;

    		var cache = [];
    		var current, last = this.getPoint( 0 );
    		var p, sum = 0;

    		cache.push( 0 );

    		for ( p = 1; p <= divisions; p ++ ) {

    			current = this.getPoint( p / divisions );
    			sum += current.distanceTo( last );
    			cache.push( sum );
    			last = current;

    		}

    		this.cacheArcLengths = cache;

    		return cache; // { sums: cache, sum: sum }; Sum is in the last element.

    	},

    	updateArcLengths: function () {

    		this.needsUpdate = true;
    		this.getLengths();

    	},

    	// Given u ( 0 .. 1 ), get a t to find p. This gives you points which are equidistant

    	getUtoTmapping: function ( u, distance ) {

    		var arcLengths = this.getLengths();

    		var i = 0, il = arcLengths.length;

    		var targetArcLength; // The targeted u distance value to get

    		if ( distance ) {

    			targetArcLength = distance;

    		} else {

    			targetArcLength = u * arcLengths[ il - 1 ];

    		}

    		// binary search for the index with largest value smaller than target u distance

    		var low = 0, high = il - 1, comparison;

    		while ( low <= high ) {

    			i = Math.floor( low + ( high - low ) / 2 ); // less likely to overflow, though probably not issue here, JS doesn't really have integers, all numbers are floats

    			comparison = arcLengths[ i ] - targetArcLength;

    			if ( comparison < 0 ) {

    				low = i + 1;

    			} else if ( comparison > 0 ) {

    				high = i - 1;

    			} else {

    				high = i;
    				break;

    				// DONE

    			}

    		}

    		i = high;

    		if ( arcLengths[ i ] === targetArcLength ) {

    			return i / ( il - 1 );

    		}

    		// we could get finer grain at lengths, or use simple interpolation between two points

    		var lengthBefore = arcLengths[ i ];
    		var lengthAfter = arcLengths[ i + 1 ];

    		var segmentLength = lengthAfter - lengthBefore;

    		// determine where we are between the 'before' and 'after' points

    		var segmentFraction = ( targetArcLength - lengthBefore ) / segmentLength;

    		// add that fractional amount to t

    		var t = ( i + segmentFraction ) / ( il - 1 );

    		return t;

    	},

    	// Returns a unit vector tangent at t
    	// In case any sub curve does not implement its tangent derivation,
    	// 2 points a small delta apart will be used to find its gradient
    	// which seems to give a reasonable approximation

    	getTangent: function ( t ) {

    		var delta = 0.0001;
    		var t1 = t - delta;
    		var t2 = t + delta;

    		// Capping in case of danger

    		if ( t1 < 0 ) t1 = 0;
    		if ( t2 > 1 ) t2 = 1;

    		var pt1 = this.getPoint( t1 );
    		var pt2 = this.getPoint( t2 );

    		var vec = pt2.clone().sub( pt1 );
    		return vec.normalize();

    	},

    	getTangentAt: function ( u ) {

    		var t = this.getUtoTmapping( u );
    		return this.getTangent( t );

    	},

    	computeFrenetFrames: function ( segments, closed ) {

    		// see http://www.cs.indiana.edu/pub/techreports/TR425.pdf

    		var normal = new Vector3();

    		var tangents = [];
    		var normals = [];
    		var binormals = [];

    		var vec = new Vector3();
    		var mat = new Matrix4();

    		var i, u, theta;

    		// compute the tangent vectors for each segment on the curve

    		for ( i = 0; i <= segments; i ++ ) {

    			u = i / segments;

    			tangents[ i ] = this.getTangentAt( u );
    			tangents[ i ].normalize();

    		}

    		// select an initial normal vector perpendicular to the first tangent vector,
    		// and in the direction of the minimum tangent xyz component

    		normals[ 0 ] = new Vector3();
    		binormals[ 0 ] = new Vector3();
    		var min = Number.MAX_VALUE;
    		var tx = Math.abs( tangents[ 0 ].x );
    		var ty = Math.abs( tangents[ 0 ].y );
    		var tz = Math.abs( tangents[ 0 ].z );

    		if ( tx <= min ) {

    			min = tx;
    			normal.set( 1, 0, 0 );

    		}

    		if ( ty <= min ) {

    			min = ty;
    			normal.set( 0, 1, 0 );

    		}

    		if ( tz <= min ) {

    			normal.set( 0, 0, 1 );

    		}

    		vec.crossVectors( tangents[ 0 ], normal ).normalize();

    		normals[ 0 ].crossVectors( tangents[ 0 ], vec );
    		binormals[ 0 ].crossVectors( tangents[ 0 ], normals[ 0 ] );


    		// compute the slowly-varying normal and binormal vectors for each segment on the curve

    		for ( i = 1; i <= segments; i ++ ) {

    			normals[ i ] = normals[ i - 1 ].clone();

    			binormals[ i ] = binormals[ i - 1 ].clone();

    			vec.crossVectors( tangents[ i - 1 ], tangents[ i ] );

    			if ( vec.length() > Number.EPSILON ) {

    				vec.normalize();

    				theta = Math.acos( _Math.clamp( tangents[ i - 1 ].dot( tangents[ i ] ), - 1, 1 ) ); // clamp for floating pt errors

    				normals[ i ].applyMatrix4( mat.makeRotationAxis( vec, theta ) );

    			}

    			binormals[ i ].crossVectors( tangents[ i ], normals[ i ] );

    		}

    		// if the curve is closed, postprocess the vectors so the first and last normal vectors are the same

    		if ( closed === true ) {

    			theta = Math.acos( _Math.clamp( normals[ 0 ].dot( normals[ segments ] ), - 1, 1 ) );
    			theta /= segments;

    			if ( tangents[ 0 ].dot( vec.crossVectors( normals[ 0 ], normals[ segments ] ) ) > 0 ) {

    				theta = - theta;

    			}

    			for ( i = 1; i <= segments; i ++ ) {

    				// twist a little...
    				normals[ i ].applyMatrix4( mat.makeRotationAxis( tangents[ i ], theta * i ) );
    				binormals[ i ].crossVectors( tangents[ i ], normals[ i ] );

    			}

    		}

    		return {
    			tangents: tangents,
    			normals: normals,
    			binormals: binormals
    		};

    	},

    	clone: function () {

    		return new this.constructor().copy( this );

    	},

    	copy: function ( source ) {

    		this.arcLengthDivisions = source.arcLengthDivisions;

    		return this;

    	},

    	toJSON: function () {

    		var data = {
    			metadata: {
    				version: 4.5,
    				type: 'Curve',
    				generator: 'Curve.toJSON'
    			}
    		};

    		data.arcLengthDivisions = this.arcLengthDivisions;
    		data.type = this.type;

    		return data;

    	},

    	fromJSON: function ( json ) {

    		this.arcLengthDivisions = json.arcLengthDivisions;

    		return this;

    	}

    } );

    function EllipseCurve( aX, aY, xRadius, yRadius, aStartAngle, aEndAngle, aClockwise, aRotation ) {

    	Curve.call( this );

    	this.type = 'EllipseCurve';

    	this.aX = aX || 0;
    	this.aY = aY || 0;

    	this.xRadius = xRadius || 1;
    	this.yRadius = yRadius || 1;

    	this.aStartAngle = aStartAngle || 0;
    	this.aEndAngle = aEndAngle || 2 * Math.PI;

    	this.aClockwise = aClockwise || false;

    	this.aRotation = aRotation || 0;

    }

    EllipseCurve.prototype = Object.create( Curve.prototype );
    EllipseCurve.prototype.constructor = EllipseCurve;

    EllipseCurve.prototype.isEllipseCurve = true;

    EllipseCurve.prototype.getPoint = function ( t, optionalTarget ) {

    	var point = optionalTarget || new Vector2();

    	var twoPi = Math.PI * 2;
    	var deltaAngle = this.aEndAngle - this.aStartAngle;
    	var samePoints = Math.abs( deltaAngle ) < Number.EPSILON;

    	// ensures that deltaAngle is 0 .. 2 PI
    	while ( deltaAngle < 0 ) deltaAngle += twoPi;
    	while ( deltaAngle > twoPi ) deltaAngle -= twoPi;

    	if ( deltaAngle < Number.EPSILON ) {

    		if ( samePoints ) {

    			deltaAngle = 0;

    		} else {

    			deltaAngle = twoPi;

    		}

    	}

    	if ( this.aClockwise === true && ! samePoints ) {

    		if ( deltaAngle === twoPi ) {

    			deltaAngle = - twoPi;

    		} else {

    			deltaAngle = deltaAngle - twoPi;

    		}

    	}

    	var angle = this.aStartAngle + t * deltaAngle;
    	var x = this.aX + this.xRadius * Math.cos( angle );
    	var y = this.aY + this.yRadius * Math.sin( angle );

    	if ( this.aRotation !== 0 ) {

    		var cos = Math.cos( this.aRotation );
    		var sin = Math.sin( this.aRotation );

    		var tx = x - this.aX;
    		var ty = y - this.aY;

    		// Rotate the point about the center of the ellipse.
    		x = tx * cos - ty * sin + this.aX;
    		y = tx * sin + ty * cos + this.aY;

    	}

    	return point.set( x, y );

    };

    EllipseCurve.prototype.copy = function ( source ) {

    	Curve.prototype.copy.call( this, source );

    	this.aX = source.aX;
    	this.aY = source.aY;

    	this.xRadius = source.xRadius;
    	this.yRadius = source.yRadius;

    	this.aStartAngle = source.aStartAngle;
    	this.aEndAngle = source.aEndAngle;

    	this.aClockwise = source.aClockwise;

    	this.aRotation = source.aRotation;

    	return this;

    };


    EllipseCurve.prototype.toJSON = function () {

    	var data = Curve.prototype.toJSON.call( this );

    	data.aX = this.aX;
    	data.aY = this.aY;

    	data.xRadius = this.xRadius;
    	data.yRadius = this.yRadius;

    	data.aStartAngle = this.aStartAngle;
    	data.aEndAngle = this.aEndAngle;

    	data.aClockwise = this.aClockwise;

    	data.aRotation = this.aRotation;

    	return data;

    };

    EllipseCurve.prototype.fromJSON = function ( json ) {

    	Curve.prototype.fromJSON.call( this, json );

    	this.aX = json.aX;
    	this.aY = json.aY;

    	this.xRadius = json.xRadius;
    	this.yRadius = json.yRadius;

    	this.aStartAngle = json.aStartAngle;
    	this.aEndAngle = json.aEndAngle;

    	this.aClockwise = json.aClockwise;

    	this.aRotation = json.aRotation;

    	return this;

    };

    function ArcCurve( aX, aY, aRadius, aStartAngle, aEndAngle, aClockwise ) {

    	EllipseCurve.call( this, aX, aY, aRadius, aRadius, aStartAngle, aEndAngle, aClockwise );

    	this.type = 'ArcCurve';

    }

    ArcCurve.prototype = Object.create( EllipseCurve.prototype );
    ArcCurve.prototype.constructor = ArcCurve;

    ArcCurve.prototype.isArcCurve = true;

    /**
     * @author zz85 https://github.com/zz85
     *
     * Centripetal CatmullRom Curve - which is useful for avoiding
     * cusps and self-intersections in non-uniform catmull rom curves.
     * http://www.cemyuksel.com/research/catmullrom_param/catmullrom.pdf
     *
     * curve.type accepts centripetal(default), chordal and catmullrom
     * curve.tension is used for catmullrom which defaults to 0.5
     */


    /*
    Based on an optimized c++ solution in
     - http://stackoverflow.com/questions/9489736/catmull-rom-curve-with-no-cusps-and-no-self-intersections/
     - http://ideone.com/NoEbVM

    This CubicPoly class could be used for reusing some variables and calculations,
    but for three.js curve use, it could be possible inlined and flatten into a single function call
    which can be placed in CurveUtils.
    */

    function CubicPoly() {

    	var c0 = 0, c1 = 0, c2 = 0, c3 = 0;

    	/*
    	 * Compute coefficients for a cubic polynomial
    	 *   p(s) = c0 + c1*s + c2*s^2 + c3*s^3
    	 * such that
    	 *   p(0) = x0, p(1) = x1
    	 *  and
    	 *   p'(0) = t0, p'(1) = t1.
    	 */
    	function init( x0, x1, t0, t1 ) {

    		c0 = x0;
    		c1 = t0;
    		c2 = - 3 * x0 + 3 * x1 - 2 * t0 - t1;
    		c3 = 2 * x0 - 2 * x1 + t0 + t1;

    	}

    	return {

    		initCatmullRom: function ( x0, x1, x2, x3, tension ) {

    			init( x1, x2, tension * ( x2 - x0 ), tension * ( x3 - x1 ) );

    		},

    		initNonuniformCatmullRom: function ( x0, x1, x2, x3, dt0, dt1, dt2 ) {

    			// compute tangents when parameterized in [t1,t2]
    			var t1 = ( x1 - x0 ) / dt0 - ( x2 - x0 ) / ( dt0 + dt1 ) + ( x2 - x1 ) / dt1;
    			var t2 = ( x2 - x1 ) / dt1 - ( x3 - x1 ) / ( dt1 + dt2 ) + ( x3 - x2 ) / dt2;

    			// rescale tangents for parametrization in [0,1]
    			t1 *= dt1;
    			t2 *= dt1;

    			init( x1, x2, t1, t2 );

    		},

    		calc: function ( t ) {

    			var t2 = t * t;
    			var t3 = t2 * t;
    			return c0 + c1 * t + c2 * t2 + c3 * t3;

    		}

    	};

    }

    //

    var tmp = new Vector3();
    var px = new CubicPoly(), py = new CubicPoly(), pz = new CubicPoly();

    function CatmullRomCurve3( points, closed, curveType, tension ) {

    	Curve.call( this );

    	this.type = 'CatmullRomCurve3';

    	this.points = points || [];
    	this.closed = closed || false;
    	this.curveType = curveType || 'centripetal';
    	this.tension = tension || 0.5;

    }

    CatmullRomCurve3.prototype = Object.create( Curve.prototype );
    CatmullRomCurve3.prototype.constructor = CatmullRomCurve3;

    CatmullRomCurve3.prototype.isCatmullRomCurve3 = true;

    CatmullRomCurve3.prototype.getPoint = function ( t, optionalTarget ) {

    	var point = optionalTarget || new Vector3();

    	var points = this.points;
    	var l = points.length;

    	var p = ( l - ( this.closed ? 0 : 1 ) ) * t;
    	var intPoint = Math.floor( p );
    	var weight = p - intPoint;

    	if ( this.closed ) {

    		intPoint += intPoint > 0 ? 0 : ( Math.floor( Math.abs( intPoint ) / l ) + 1 ) * l;

    	} else if ( weight === 0 && intPoint === l - 1 ) {

    		intPoint = l - 2;
    		weight = 1;

    	}

    	var p0, p1, p2, p3; // 4 points

    	if ( this.closed || intPoint > 0 ) {

    		p0 = points[ ( intPoint - 1 ) % l ];

    	} else {

    		// extrapolate first point
    		tmp.subVectors( points[ 0 ], points[ 1 ] ).add( points[ 0 ] );
    		p0 = tmp;

    	}

    	p1 = points[ intPoint % l ];
    	p2 = points[ ( intPoint + 1 ) % l ];

    	if ( this.closed || intPoint + 2 < l ) {

    		p3 = points[ ( intPoint + 2 ) % l ];

    	} else {

    		// extrapolate last point
    		tmp.subVectors( points[ l - 1 ], points[ l - 2 ] ).add( points[ l - 1 ] );
    		p3 = tmp;

    	}

    	if ( this.curveType === 'centripetal' || this.curveType === 'chordal' ) {

    		// init Centripetal / Chordal Catmull-Rom
    		var pow = this.curveType === 'chordal' ? 0.5 : 0.25;
    		var dt0 = Math.pow( p0.distanceToSquared( p1 ), pow );
    		var dt1 = Math.pow( p1.distanceToSquared( p2 ), pow );
    		var dt2 = Math.pow( p2.distanceToSquared( p3 ), pow );

    		// safety check for repeated points
    		if ( dt1 < 1e-4 ) dt1 = 1.0;
    		if ( dt0 < 1e-4 ) dt0 = dt1;
    		if ( dt2 < 1e-4 ) dt2 = dt1;

    		px.initNonuniformCatmullRom( p0.x, p1.x, p2.x, p3.x, dt0, dt1, dt2 );
    		py.initNonuniformCatmullRom( p0.y, p1.y, p2.y, p3.y, dt0, dt1, dt2 );
    		pz.initNonuniformCatmullRom( p0.z, p1.z, p2.z, p3.z, dt0, dt1, dt2 );

    	} else if ( this.curveType === 'catmullrom' ) {

    		px.initCatmullRom( p0.x, p1.x, p2.x, p3.x, this.tension );
    		py.initCatmullRom( p0.y, p1.y, p2.y, p3.y, this.tension );
    		pz.initCatmullRom( p0.z, p1.z, p2.z, p3.z, this.tension );

    	}

    	point.set(
    		px.calc( weight ),
    		py.calc( weight ),
    		pz.calc( weight )
    	);

    	return point;

    };

    CatmullRomCurve3.prototype.copy = function ( source ) {

    	Curve.prototype.copy.call( this, source );

    	this.points = [];

    	for ( var i = 0, l = source.points.length; i < l; i ++ ) {

    		var point = source.points[ i ];

    		this.points.push( point.clone() );

    	}

    	this.closed = source.closed;
    	this.curveType = source.curveType;
    	this.tension = source.tension;

    	return this;

    };

    CatmullRomCurve3.prototype.toJSON = function () {

    	var data = Curve.prototype.toJSON.call( this );

    	data.points = [];

    	for ( var i = 0, l = this.points.length; i < l; i ++ ) {

    		var point = this.points[ i ];
    		data.points.push( point.toArray() );

    	}

    	data.closed = this.closed;
    	data.curveType = this.curveType;
    	data.tension = this.tension;

    	return data;

    };

    CatmullRomCurve3.prototype.fromJSON = function ( json ) {

    	Curve.prototype.fromJSON.call( this, json );

    	this.points = [];

    	for ( var i = 0, l = json.points.length; i < l; i ++ ) {

    		var point = json.points[ i ];
    		this.points.push( new Vector3().fromArray( point ) );

    	}

    	this.closed = json.closed;
    	this.curveType = json.curveType;
    	this.tension = json.tension;

    	return this;

    };

    /**
     * @author zz85 / http://www.lab4games.net/zz85/blog
     *
     * Bezier Curves formulas obtained from
     * http://en.wikipedia.org/wiki/Bézier_curve
     */

    function CatmullRom$2( t, p0, p1, p2, p3 ) {

    	var v0 = ( p2 - p0 ) * 0.5;
    	var v1 = ( p3 - p1 ) * 0.5;
    	var t2 = t * t;
    	var t3 = t * t2;
    	return ( 2 * p1 - 2 * p2 + v0 + v1 ) * t3 + ( - 3 * p1 + 3 * p2 - 2 * v0 - v1 ) * t2 + v0 * t + p1;

    }

    //

    function QuadraticBezierP0( t, p ) {

    	var k = 1 - t;
    	return k * k * p;

    }

    function QuadraticBezierP1( t, p ) {

    	return 2 * ( 1 - t ) * t * p;

    }

    function QuadraticBezierP2( t, p ) {

    	return t * t * p;

    }

    function QuadraticBezier( t, p0, p1, p2 ) {

    	return QuadraticBezierP0( t, p0 ) + QuadraticBezierP1( t, p1 ) +
    		QuadraticBezierP2( t, p2 );

    }

    //

    function CubicBezierP0( t, p ) {

    	var k = 1 - t;
    	return k * k * k * p;

    }

    function CubicBezierP1( t, p ) {

    	var k = 1 - t;
    	return 3 * k * k * t * p;

    }

    function CubicBezierP2( t, p ) {

    	return 3 * ( 1 - t ) * t * t * p;

    }

    function CubicBezierP3( t, p ) {

    	return t * t * t * p;

    }

    function CubicBezier( t, p0, p1, p2, p3 ) {

    	return CubicBezierP0( t, p0 ) + CubicBezierP1( t, p1 ) + CubicBezierP2( t, p2 ) +
    		CubicBezierP3( t, p3 );

    }

    function CubicBezierCurve( v0, v1, v2, v3 ) {

    	Curve.call( this );

    	this.type = 'CubicBezierCurve';

    	this.v0 = v0 || new Vector2();
    	this.v1 = v1 || new Vector2();
    	this.v2 = v2 || new Vector2();
    	this.v3 = v3 || new Vector2();

    }

    CubicBezierCurve.prototype = Object.create( Curve.prototype );
    CubicBezierCurve.prototype.constructor = CubicBezierCurve;

    CubicBezierCurve.prototype.isCubicBezierCurve = true;

    CubicBezierCurve.prototype.getPoint = function ( t, optionalTarget ) {

    	var point = optionalTarget || new Vector2();

    	var v0 = this.v0, v1 = this.v1, v2 = this.v2, v3 = this.v3;

    	point.set(
    		CubicBezier( t, v0.x, v1.x, v2.x, v3.x ),
    		CubicBezier( t, v0.y, v1.y, v2.y, v3.y )
    	);

    	return point;

    };

    CubicBezierCurve.prototype.copy = function ( source ) {

    	Curve.prototype.copy.call( this, source );

    	this.v0.copy( source.v0 );
    	this.v1.copy( source.v1 );
    	this.v2.copy( source.v2 );
    	this.v3.copy( source.v3 );

    	return this;

    };

    CubicBezierCurve.prototype.toJSON = function () {

    	var data = Curve.prototype.toJSON.call( this );

    	data.v0 = this.v0.toArray();
    	data.v1 = this.v1.toArray();
    	data.v2 = this.v2.toArray();
    	data.v3 = this.v3.toArray();

    	return data;

    };

    CubicBezierCurve.prototype.fromJSON = function ( json ) {

    	Curve.prototype.fromJSON.call( this, json );

    	this.v0.fromArray( json.v0 );
    	this.v1.fromArray( json.v1 );
    	this.v2.fromArray( json.v2 );
    	this.v3.fromArray( json.v3 );

    	return this;

    };

    function CubicBezierCurve3( v0, v1, v2, v3 ) {

    	Curve.call( this );

    	this.type = 'CubicBezierCurve3';

    	this.v0 = v0 || new Vector3();
    	this.v1 = v1 || new Vector3();
    	this.v2 = v2 || new Vector3();
    	this.v3 = v3 || new Vector3();

    }

    CubicBezierCurve3.prototype = Object.create( Curve.prototype );
    CubicBezierCurve3.prototype.constructor = CubicBezierCurve3;

    CubicBezierCurve3.prototype.isCubicBezierCurve3 = true;

    CubicBezierCurve3.prototype.getPoint = function ( t, optionalTarget ) {

    	var point = optionalTarget || new Vector3();

    	var v0 = this.v0, v1 = this.v1, v2 = this.v2, v3 = this.v3;

    	point.set(
    		CubicBezier( t, v0.x, v1.x, v2.x, v3.x ),
    		CubicBezier( t, v0.y, v1.y, v2.y, v3.y ),
    		CubicBezier( t, v0.z, v1.z, v2.z, v3.z )
    	);

    	return point;

    };

    CubicBezierCurve3.prototype.copy = function ( source ) {

    	Curve.prototype.copy.call( this, source );

    	this.v0.copy( source.v0 );
    	this.v1.copy( source.v1 );
    	this.v2.copy( source.v2 );
    	this.v3.copy( source.v3 );

    	return this;

    };

    CubicBezierCurve3.prototype.toJSON = function () {

    	var data = Curve.prototype.toJSON.call( this );

    	data.v0 = this.v0.toArray();
    	data.v1 = this.v1.toArray();
    	data.v2 = this.v2.toArray();
    	data.v3 = this.v3.toArray();

    	return data;

    };

    CubicBezierCurve3.prototype.fromJSON = function ( json ) {

    	Curve.prototype.fromJSON.call( this, json );

    	this.v0.fromArray( json.v0 );
    	this.v1.fromArray( json.v1 );
    	this.v2.fromArray( json.v2 );
    	this.v3.fromArray( json.v3 );

    	return this;

    };

    function LineCurve( v1, v2 ) {

    	Curve.call( this );

    	this.type = 'LineCurve';

    	this.v1 = v1 || new Vector2();
    	this.v2 = v2 || new Vector2();

    }

    LineCurve.prototype = Object.create( Curve.prototype );
    LineCurve.prototype.constructor = LineCurve;

    LineCurve.prototype.isLineCurve = true;

    LineCurve.prototype.getPoint = function ( t, optionalTarget ) {

    	var point = optionalTarget || new Vector2();

    	if ( t === 1 ) {

    		point.copy( this.v2 );

    	} else {

    		point.copy( this.v2 ).sub( this.v1 );
    		point.multiplyScalar( t ).add( this.v1 );

    	}

    	return point;

    };

    // Line curve is linear, so we can overwrite default getPointAt

    LineCurve.prototype.getPointAt = function ( u, optionalTarget ) {

    	return this.getPoint( u, optionalTarget );

    };

    LineCurve.prototype.getTangent = function ( /* t */ ) {

    	var tangent = this.v2.clone().sub( this.v1 );

    	return tangent.normalize();

    };

    LineCurve.prototype.copy = function ( source ) {

    	Curve.prototype.copy.call( this, source );

    	this.v1.copy( source.v1 );
    	this.v2.copy( source.v2 );

    	return this;

    };

    LineCurve.prototype.toJSON = function () {

    	var data = Curve.prototype.toJSON.call( this );

    	data.v1 = this.v1.toArray();
    	data.v2 = this.v2.toArray();

    	return data;

    };

    LineCurve.prototype.fromJSON = function ( json ) {

    	Curve.prototype.fromJSON.call( this, json );

    	this.v1.fromArray( json.v1 );
    	this.v2.fromArray( json.v2 );

    	return this;

    };

    function LineCurve3( v1, v2 ) {

    	Curve.call( this );

    	this.type = 'LineCurve3';

    	this.v1 = v1 || new Vector3();
    	this.v2 = v2 || new Vector3();

    }

    LineCurve3.prototype = Object.create( Curve.prototype );
    LineCurve3.prototype.constructor = LineCurve3;

    LineCurve3.prototype.isLineCurve3 = true;

    LineCurve3.prototype.getPoint = function ( t, optionalTarget ) {

    	var point = optionalTarget || new Vector3();

    	if ( t === 1 ) {

    		point.copy( this.v2 );

    	} else {

    		point.copy( this.v2 ).sub( this.v1 );
    		point.multiplyScalar( t ).add( this.v1 );

    	}

    	return point;

    };

    // Line curve is linear, so we can overwrite default getPointAt

    LineCurve3.prototype.getPointAt = function ( u, optionalTarget ) {

    	return this.getPoint( u, optionalTarget );

    };

    LineCurve3.prototype.copy = function ( source ) {

    	Curve.prototype.copy.call( this, source );

    	this.v1.copy( source.v1 );
    	this.v2.copy( source.v2 );

    	return this;

    };

    LineCurve3.prototype.toJSON = function () {

    	var data = Curve.prototype.toJSON.call( this );

    	data.v1 = this.v1.toArray();
    	data.v2 = this.v2.toArray();

    	return data;

    };

    LineCurve3.prototype.fromJSON = function ( json ) {

    	Curve.prototype.fromJSON.call( this, json );

    	this.v1.fromArray( json.v1 );
    	this.v2.fromArray( json.v2 );

    	return this;

    };

    function QuadraticBezierCurve( v0, v1, v2 ) {

    	Curve.call( this );

    	this.type = 'QuadraticBezierCurve';

    	this.v0 = v0 || new Vector2();
    	this.v1 = v1 || new Vector2();
    	this.v2 = v2 || new Vector2();

    }

    QuadraticBezierCurve.prototype = Object.create( Curve.prototype );
    QuadraticBezierCurve.prototype.constructor = QuadraticBezierCurve;

    QuadraticBezierCurve.prototype.isQuadraticBezierCurve = true;

    QuadraticBezierCurve.prototype.getPoint = function ( t, optionalTarget ) {

    	var point = optionalTarget || new Vector2();

    	var v0 = this.v0, v1 = this.v1, v2 = this.v2;

    	point.set(
    		QuadraticBezier( t, v0.x, v1.x, v2.x ),
    		QuadraticBezier( t, v0.y, v1.y, v2.y )
    	);

    	return point;

    };

    QuadraticBezierCurve.prototype.copy = function ( source ) {

    	Curve.prototype.copy.call( this, source );

    	this.v0.copy( source.v0 );
    	this.v1.copy( source.v1 );
    	this.v2.copy( source.v2 );

    	return this;

    };

    QuadraticBezierCurve.prototype.toJSON = function () {

    	var data = Curve.prototype.toJSON.call( this );

    	data.v0 = this.v0.toArray();
    	data.v1 = this.v1.toArray();
    	data.v2 = this.v2.toArray();

    	return data;

    };

    QuadraticBezierCurve.prototype.fromJSON = function ( json ) {

    	Curve.prototype.fromJSON.call( this, json );

    	this.v0.fromArray( json.v0 );
    	this.v1.fromArray( json.v1 );
    	this.v2.fromArray( json.v2 );

    	return this;

    };

    function QuadraticBezierCurve3( v0, v1, v2 ) {

    	Curve.call( this );

    	this.type = 'QuadraticBezierCurve3';

    	this.v0 = v0 || new Vector3();
    	this.v1 = v1 || new Vector3();
    	this.v2 = v2 || new Vector3();

    }

    QuadraticBezierCurve3.prototype = Object.create( Curve.prototype );
    QuadraticBezierCurve3.prototype.constructor = QuadraticBezierCurve3;

    QuadraticBezierCurve3.prototype.isQuadraticBezierCurve3 = true;

    QuadraticBezierCurve3.prototype.getPoint = function ( t, optionalTarget ) {

    	var point = optionalTarget || new Vector3();

    	var v0 = this.v0, v1 = this.v1, v2 = this.v2;

    	point.set(
    		QuadraticBezier( t, v0.x, v1.x, v2.x ),
    		QuadraticBezier( t, v0.y, v1.y, v2.y ),
    		QuadraticBezier( t, v0.z, v1.z, v2.z )
    	);

    	return point;

    };

    QuadraticBezierCurve3.prototype.copy = function ( source ) {

    	Curve.prototype.copy.call( this, source );

    	this.v0.copy( source.v0 );
    	this.v1.copy( source.v1 );
    	this.v2.copy( source.v2 );

    	return this;

    };

    QuadraticBezierCurve3.prototype.toJSON = function () {

    	var data = Curve.prototype.toJSON.call( this );

    	data.v0 = this.v0.toArray();
    	data.v1 = this.v1.toArray();
    	data.v2 = this.v2.toArray();

    	return data;

    };

    QuadraticBezierCurve3.prototype.fromJSON = function ( json ) {

    	Curve.prototype.fromJSON.call( this, json );

    	this.v0.fromArray( json.v0 );
    	this.v1.fromArray( json.v1 );
    	this.v2.fromArray( json.v2 );

    	return this;

    };

    function SplineCurve( points /* array of Vector2 */ ) {

    	Curve.call( this );

    	this.type = 'SplineCurve';

    	this.points = points || [];

    }

    SplineCurve.prototype = Object.create( Curve.prototype );
    SplineCurve.prototype.constructor = SplineCurve;

    SplineCurve.prototype.isSplineCurve = true;

    SplineCurve.prototype.getPoint = function ( t, optionalTarget ) {

    	var point = optionalTarget || new Vector2();

    	var points = this.points;
    	var p = ( points.length - 1 ) * t;

    	var intPoint = Math.floor( p );
    	var weight = p - intPoint;

    	var p0 = points[ intPoint === 0 ? intPoint : intPoint - 1 ];
    	var p1 = points[ intPoint ];
    	var p2 = points[ intPoint > points.length - 2 ? points.length - 1 : intPoint + 1 ];
    	var p3 = points[ intPoint > points.length - 3 ? points.length - 1 : intPoint + 2 ];

    	point.set(
    		CatmullRom$2( weight, p0.x, p1.x, p2.x, p3.x ),
    		CatmullRom$2( weight, p0.y, p1.y, p2.y, p3.y )
    	);

    	return point;

    };

    SplineCurve.prototype.copy = function ( source ) {

    	Curve.prototype.copy.call( this, source );

    	this.points = [];

    	for ( var i = 0, l = source.points.length; i < l; i ++ ) {

    		var point = source.points[ i ];

    		this.points.push( point.clone() );

    	}

    	return this;

    };

    SplineCurve.prototype.toJSON = function () {

    	var data = Curve.prototype.toJSON.call( this );

    	data.points = [];

    	for ( var i = 0, l = this.points.length; i < l; i ++ ) {

    		var point = this.points[ i ];
    		data.points.push( point.toArray() );

    	}

    	return data;

    };

    SplineCurve.prototype.fromJSON = function ( json ) {

    	Curve.prototype.fromJSON.call( this, json );

    	this.points = [];

    	for ( var i = 0, l = json.points.length; i < l; i ++ ) {

    		var point = json.points[ i ];
    		this.points.push( new Vector2().fromArray( point ) );

    	}

    	return this;

    };



    var Curves = /*#__PURE__*/Object.freeze({
    	ArcCurve: ArcCurve,
    	CatmullRomCurve3: CatmullRomCurve3,
    	CubicBezierCurve: CubicBezierCurve,
    	CubicBezierCurve3: CubicBezierCurve3,
    	EllipseCurve: EllipseCurve,
    	LineCurve: LineCurve,
    	LineCurve3: LineCurve3,
    	QuadraticBezierCurve: QuadraticBezierCurve,
    	QuadraticBezierCurve3: QuadraticBezierCurve3,
    	SplineCurve: SplineCurve
    });

    /**
     * @author zz85 / http://www.lab4games.net/zz85/blog
     *
     **/

    /**************************************************************
     *	Curved Path - a curve path is simply a array of connected
     *  curves, but retains the api of a curve
     **************************************************************/

    function CurvePath() {

    	Curve.call( this );

    	this.type = 'CurvePath';

    	this.curves = [];
    	this.autoClose = false; // Automatically closes the path

    }

    CurvePath.prototype = Object.assign( Object.create( Curve.prototype ), {

    	constructor: CurvePath,

    	add: function ( curve ) {

    		this.curves.push( curve );

    	},

    	closePath: function () {

    		// Add a line curve if start and end of lines are not connected
    		var startPoint = this.curves[ 0 ].getPoint( 0 );
    		var endPoint = this.curves[ this.curves.length - 1 ].getPoint( 1 );

    		if ( ! startPoint.equals( endPoint ) ) {

    			this.curves.push( new LineCurve( endPoint, startPoint ) );

    		}

    	},

    	// To get accurate point with reference to
    	// entire path distance at time t,
    	// following has to be done:

    	// 1. Length of each sub path have to be known
    	// 2. Locate and identify type of curve
    	// 3. Get t for the curve
    	// 4. Return curve.getPointAt(t')

    	getPoint: function ( t ) {

    		var d = t * this.getLength();
    		var curveLengths = this.getCurveLengths();
    		var i = 0;

    		// To think about boundaries points.

    		while ( i < curveLengths.length ) {

    			if ( curveLengths[ i ] >= d ) {

    				var diff = curveLengths[ i ] - d;
    				var curve = this.curves[ i ];

    				var segmentLength = curve.getLength();
    				var u = segmentLength === 0 ? 0 : 1 - diff / segmentLength;

    				return curve.getPointAt( u );

    			}

    			i ++;

    		}

    		return null;

    		// loop where sum != 0, sum > d , sum+1 <d

    	},

    	// We cannot use the default THREE.Curve getPoint() with getLength() because in
    	// THREE.Curve, getLength() depends on getPoint() but in THREE.CurvePath
    	// getPoint() depends on getLength

    	getLength: function () {

    		var lens = this.getCurveLengths();
    		return lens[ lens.length - 1 ];

    	},

    	// cacheLengths must be recalculated.
    	updateArcLengths: function () {

    		this.needsUpdate = true;
    		this.cacheLengths = null;
    		this.getCurveLengths();

    	},

    	// Compute lengths and cache them
    	// We cannot overwrite getLengths() because UtoT mapping uses it.

    	getCurveLengths: function () {

    		// We use cache values if curves and cache array are same length

    		if ( this.cacheLengths && this.cacheLengths.length === this.curves.length ) {

    			return this.cacheLengths;

    		}

    		// Get length of sub-curve
    		// Push sums into cached array

    		var lengths = [], sums = 0;

    		for ( var i = 0, l = this.curves.length; i < l; i ++ ) {

    			sums += this.curves[ i ].getLength();
    			lengths.push( sums );

    		}

    		this.cacheLengths = lengths;

    		return lengths;

    	},

    	getSpacedPoints: function ( divisions ) {

    		if ( divisions === undefined ) divisions = 40;

    		var points = [];

    		for ( var i = 0; i <= divisions; i ++ ) {

    			points.push( this.getPoint( i / divisions ) );

    		}

    		if ( this.autoClose ) {

    			points.push( points[ 0 ] );

    		}

    		return points;

    	},

    	getPoints: function ( divisions ) {

    		divisions = divisions || 12;

    		var points = [], last;

    		for ( var i = 0, curves = this.curves; i < curves.length; i ++ ) {

    			var curve = curves[ i ];
    			var resolution = ( curve && curve.isEllipseCurve ) ? divisions * 2
    				: ( curve && ( curve.isLineCurve || curve.isLineCurve3 ) ) ? 1
    					: ( curve && curve.isSplineCurve ) ? divisions * curve.points.length
    						: divisions;

    			var pts = curve.getPoints( resolution );

    			for ( var j = 0; j < pts.length; j ++ ) {

    				var point = pts[ j ];

    				if ( last && last.equals( point ) ) continue; // ensures no consecutive points are duplicates

    				points.push( point );
    				last = point;

    			}

    		}

    		if ( this.autoClose && points.length > 1 && ! points[ points.length - 1 ].equals( points[ 0 ] ) ) {

    			points.push( points[ 0 ] );

    		}

    		return points;

    	},

    	copy: function ( source ) {

    		Curve.prototype.copy.call( this, source );

    		this.curves = [];

    		for ( var i = 0, l = source.curves.length; i < l; i ++ ) {

    			var curve = source.curves[ i ];

    			this.curves.push( curve.clone() );

    		}

    		this.autoClose = source.autoClose;

    		return this;

    	},

    	toJSON: function () {

    		var data = Curve.prototype.toJSON.call( this );

    		data.autoClose = this.autoClose;
    		data.curves = [];

    		for ( var i = 0, l = this.curves.length; i < l; i ++ ) {

    			var curve = this.curves[ i ];
    			data.curves.push( curve.toJSON() );

    		}

    		return data;

    	},

    	fromJSON: function ( json ) {

    		Curve.prototype.fromJSON.call( this, json );

    		this.autoClose = json.autoClose;
    		this.curves = [];

    		for ( var i = 0, l = json.curves.length; i < l; i ++ ) {

    			var curve = json.curves[ i ];
    			this.curves.push( new Curves[ curve.type ]().fromJSON( curve ) );

    		}

    		return this;

    	}

    } );

    /**
     * @author zz85 / http://www.lab4games.net/zz85/blog
     * Creates free form 2d path using series of points, lines or curves.
     **/

    function Path$4( points ) {

    	CurvePath.call( this );

    	this.type = 'Path';

    	this.currentPoint = new Vector2();

    	if ( points ) {

    		this.setFromPoints( points );

    	}

    }

    Path$4.prototype = Object.assign( Object.create( CurvePath.prototype ), {

    	constructor: Path$4,

    	setFromPoints: function ( points ) {

    		this.moveTo( points[ 0 ].x, points[ 0 ].y );

    		for ( var i = 1, l = points.length; i < l; i ++ ) {

    			this.lineTo( points[ i ].x, points[ i ].y );

    		}

    	},

    	moveTo: function ( x, y ) {

    		this.currentPoint.set( x, y ); // TODO consider referencing vectors instead of copying?

    	},

    	lineTo: function ( x, y ) {

    		var curve = new LineCurve( this.currentPoint.clone(), new Vector2( x, y ) );
    		this.curves.push( curve );

    		this.currentPoint.set( x, y );

    	},

    	quadraticCurveTo: function ( aCPx, aCPy, aX, aY ) {

    		var curve = new QuadraticBezierCurve(
    			this.currentPoint.clone(),
    			new Vector2( aCPx, aCPy ),
    			new Vector2( aX, aY )
    		);

    		this.curves.push( curve );

    		this.currentPoint.set( aX, aY );

    	},

    	bezierCurveTo: function ( aCP1x, aCP1y, aCP2x, aCP2y, aX, aY ) {

    		var curve = new CubicBezierCurve(
    			this.currentPoint.clone(),
    			new Vector2( aCP1x, aCP1y ),
    			new Vector2( aCP2x, aCP2y ),
    			new Vector2( aX, aY )
    		);

    		this.curves.push( curve );

    		this.currentPoint.set( aX, aY );

    	},

    	splineThru: function ( pts /*Array of Vector*/ ) {

    		var npts = [ this.currentPoint.clone() ].concat( pts );

    		var curve = new SplineCurve( npts );
    		this.curves.push( curve );

    		this.currentPoint.copy( pts[ pts.length - 1 ] );

    	},

    	arc: function ( aX, aY, aRadius, aStartAngle, aEndAngle, aClockwise ) {

    		var x0 = this.currentPoint.x;
    		var y0 = this.currentPoint.y;

    		this.absarc( aX + x0, aY + y0, aRadius,
    			aStartAngle, aEndAngle, aClockwise );

    	},

    	absarc: function ( aX, aY, aRadius, aStartAngle, aEndAngle, aClockwise ) {

    		this.absellipse( aX, aY, aRadius, aRadius, aStartAngle, aEndAngle, aClockwise );

    	},

    	ellipse: function ( aX, aY, xRadius, yRadius, aStartAngle, aEndAngle, aClockwise, aRotation ) {

    		var x0 = this.currentPoint.x;
    		var y0 = this.currentPoint.y;

    		this.absellipse( aX + x0, aY + y0, xRadius, yRadius, aStartAngle, aEndAngle, aClockwise, aRotation );

    	},

    	absellipse: function ( aX, aY, xRadius, yRadius, aStartAngle, aEndAngle, aClockwise, aRotation ) {

    		var curve = new EllipseCurve( aX, aY, xRadius, yRadius, aStartAngle, aEndAngle, aClockwise, aRotation );

    		if ( this.curves.length > 0 ) {

    			// if a previous curve is present, attempt to join
    			var firstPoint = curve.getPoint( 0 );

    			if ( ! firstPoint.equals( this.currentPoint ) ) {

    				this.lineTo( firstPoint.x, firstPoint.y );

    			}

    		}

    		this.curves.push( curve );

    		var lastPoint = curve.getPoint( 1 );
    		this.currentPoint.copy( lastPoint );

    	},

    	copy: function ( source ) {

    		CurvePath.prototype.copy.call( this, source );

    		this.currentPoint.copy( source.currentPoint );

    		return this;

    	},

    	toJSON: function () {

    		var data = CurvePath.prototype.toJSON.call( this );

    		data.currentPoint = this.currentPoint.toArray();

    		return data;

    	},

    	fromJSON: function ( json ) {

    		CurvePath.prototype.fromJSON.call( this, json );

    		this.currentPoint.fromArray( json.currentPoint );

    		return this;

    	}

    } );

    /**
     * @author zz85 / http://www.lab4games.net/zz85/blog
     * Defines a 2d shape plane using paths.
     **/

    // STEP 1 Create a path.
    // STEP 2 Turn path into shape.
    // STEP 3 ExtrudeGeometry takes in Shape/Shapes
    // STEP 3a - Extract points from each shape, turn to vertices
    // STEP 3b - Triangulate each shape, add faces.

    function Shape( points ) {

    	Path$4.call( this, points );

    	this.uuid = _Math.generateUUID();

    	this.type = 'Shape';

    	this.holes = [];

    }

    Shape.prototype = Object.assign( Object.create( Path$4.prototype ), {

    	constructor: Shape,

    	getPointsHoles: function ( divisions ) {

    		var holesPts = [];

    		for ( var i = 0, l = this.holes.length; i < l; i ++ ) {

    			holesPts[ i ] = this.holes[ i ].getPoints( divisions );

    		}

    		return holesPts;

    	},

    	// get points of shape and holes (keypoints based on segments parameter)

    	extractPoints: function ( divisions ) {

    		return {

    			shape: this.getPoints( divisions ),
    			holes: this.getPointsHoles( divisions )

    		};

    	},

    	copy: function ( source ) {

    		Path$4.prototype.copy.call( this, source );

    		this.holes = [];

    		for ( var i = 0, l = source.holes.length; i < l; i ++ ) {

    			var hole = source.holes[ i ];

    			this.holes.push( hole.clone() );

    		}

    		return this;

    	},

    	toJSON: function () {

    		var data = Path$4.prototype.toJSON.call( this );

    		data.uuid = this.uuid;
    		data.holes = [];

    		for ( var i = 0, l = this.holes.length; i < l; i ++ ) {

    			var hole = this.holes[ i ];
    			data.holes.push( hole.toJSON() );

    		}

    		return data;

    	},

    	fromJSON: function ( json ) {

    		Path$4.prototype.fromJSON.call( this, json );

    		this.uuid = json.uuid;
    		this.holes = [];

    		for ( var i = 0, l = json.holes.length; i < l; i ++ ) {

    			var hole = json.holes[ i ];
    			this.holes.push( new Path$4().fromJSON( hole ) );

    		}

    		return this;

    	}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     * @author alteredq / http://alteredqualia.com/
     */

    function Light( color, intensity ) {

    	Object3D.call( this );

    	this.type = 'Light';

    	this.color = new Color$7( color );
    	this.intensity = intensity !== undefined ? intensity : 1;

    	this.receiveShadow = undefined;

    }

    Light.prototype = Object.assign( Object.create( Object3D.prototype ), {

    	constructor: Light,

    	isLight: true,

    	copy: function ( source ) {

    		Object3D.prototype.copy.call( this, source );

    		this.color.copy( source.color );
    		this.intensity = source.intensity;

    		return this;

    	},

    	toJSON: function ( meta ) {

    		var data = Object3D.prototype.toJSON.call( this, meta );

    		data.object.color = this.color.getHex();
    		data.object.intensity = this.intensity;

    		if ( this.groundColor !== undefined ) data.object.groundColor = this.groundColor.getHex();

    		if ( this.distance !== undefined ) data.object.distance = this.distance;
    		if ( this.angle !== undefined ) data.object.angle = this.angle;
    		if ( this.decay !== undefined ) data.object.decay = this.decay;
    		if ( this.penumbra !== undefined ) data.object.penumbra = this.penumbra;

    		if ( this.shadow !== undefined ) data.object.shadow = this.shadow.toJSON();

    		return data;

    	}

    } );

    /**
     * @author alteredq / http://alteredqualia.com/
     */

    function HemisphereLight( skyColor, groundColor, intensity ) {

    	Light.call( this, skyColor, intensity );

    	this.type = 'HemisphereLight';

    	this.castShadow = undefined;

    	this.position.copy( Object3D.DefaultUp );
    	this.updateMatrix();

    	this.groundColor = new Color$7( groundColor );

    }

    HemisphereLight.prototype = Object.assign( Object.create( Light.prototype ), {

    	constructor: HemisphereLight,

    	isHemisphereLight: true,

    	copy: function ( source ) {

    		Light.prototype.copy.call( this, source );

    		this.groundColor.copy( source.groundColor );

    		return this;

    	}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function LightShadow( camera ) {

    	this.camera = camera;

    	this.bias = 0;
    	this.radius = 1;

    	this.mapSize = new Vector2( 512, 512 );

    	this.map = null;
    	this.mapPass = null;
    	this.matrix = new Matrix4();

    	this._frustum = new Frustum();
    	this._frameExtents = new Vector2( 1, 1 );

    	this._viewportCount = 1;

    	this._viewports = [

    		new Vector4( 0, 0, 1, 1 )

    	];

    }

    Object.assign( LightShadow.prototype, {

    	_projScreenMatrix: new Matrix4(),

    	_lightPositionWorld: new Vector3(),

    	_lookTarget: new Vector3(),

    	getViewportCount: function () {

    		return this._viewportCount;

    	},

    	getFrustum: function () {

    		return this._frustum;

    	},

    	updateMatrices: function ( light ) {

    		var shadowCamera = this.camera,
    			shadowMatrix = this.matrix,
    			projScreenMatrix = this._projScreenMatrix,
    			lookTarget = this._lookTarget,
    			lightPositionWorld = this._lightPositionWorld;

    		lightPositionWorld.setFromMatrixPosition( light.matrixWorld );
    		shadowCamera.position.copy( lightPositionWorld );

    		lookTarget.setFromMatrixPosition( light.target.matrixWorld );
    		shadowCamera.lookAt( lookTarget );
    		shadowCamera.updateMatrixWorld();

    		projScreenMatrix.multiplyMatrices( shadowCamera.projectionMatrix, shadowCamera.matrixWorldInverse );
    		this._frustum.setFromMatrix( projScreenMatrix );

    		shadowMatrix.set(
    			0.5, 0.0, 0.0, 0.5,
    			0.0, 0.5, 0.0, 0.5,
    			0.0, 0.0, 0.5, 0.5,
    			0.0, 0.0, 0.0, 1.0
    		);

    		shadowMatrix.multiply( shadowCamera.projectionMatrix );
    		shadowMatrix.multiply( shadowCamera.matrixWorldInverse );

    	},

    	getViewport: function ( viewportIndex ) {

    		return this._viewports[ viewportIndex ];

    	},

    	getFrameExtents: function () {

    		return this._frameExtents;

    	},

    	copy: function ( source ) {

    		this.camera = source.camera.clone();

    		this.bias = source.bias;
    		this.radius = source.radius;

    		this.mapSize.copy( source.mapSize );

    		return this;

    	},

    	clone: function () {

    		return new this.constructor().copy( this );

    	},

    	toJSON: function () {

    		var object = {};

    		if ( this.bias !== 0 ) object.bias = this.bias;
    		if ( this.radius !== 1 ) object.radius = this.radius;
    		if ( this.mapSize.x !== 512 || this.mapSize.y !== 512 ) object.mapSize = this.mapSize.toArray();

    		object.camera = this.camera.toJSON( false ).object;
    		delete object.camera.matrix;

    		return object;

    	}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function SpotLightShadow() {

    	LightShadow.call( this, new PerspectiveCamera( 50, 1, 0.5, 500 ) );

    }

    SpotLightShadow.prototype = Object.assign( Object.create( LightShadow.prototype ), {

    	constructor: SpotLightShadow,

    	isSpotLightShadow: true,

    	updateMatrices: function ( light, viewCamera, viewportIndex ) {

    		var camera = this.camera;

    		var fov = _Math.RAD2DEG * 2 * light.angle;
    		var aspect = this.mapSize.width / this.mapSize.height;
    		var far = light.distance || camera.far;

    		if ( fov !== camera.fov || aspect !== camera.aspect || far !== camera.far ) {

    			camera.fov = fov;
    			camera.aspect = aspect;
    			camera.far = far;
    			camera.updateProjectionMatrix();

    		}

    		LightShadow.prototype.updateMatrices.call( this, light, viewCamera, viewportIndex );

    	}

    } );

    /**
     * @author alteredq / http://alteredqualia.com/
     */

    function SpotLight( color, intensity, distance, angle, penumbra, decay ) {

    	Light.call( this, color, intensity );

    	this.type = 'SpotLight';

    	this.position.copy( Object3D.DefaultUp );
    	this.updateMatrix();

    	this.target = new Object3D();

    	Object.defineProperty( this, 'power', {
    		get: function () {

    			// intensity = power per solid angle.
    			// ref: equation (17) from https://seblagarde.files.wordpress.com/2015/07/course_notes_moving_frostbite_to_pbr_v32.pdf
    			return this.intensity * Math.PI;

    		},
    		set: function ( power ) {

    			// intensity = power per solid angle.
    			// ref: equation (17) from https://seblagarde.files.wordpress.com/2015/07/course_notes_moving_frostbite_to_pbr_v32.pdf
    			this.intensity = power / Math.PI;

    		}
    	} );

    	this.distance = ( distance !== undefined ) ? distance : 0;
    	this.angle = ( angle !== undefined ) ? angle : Math.PI / 3;
    	this.penumbra = ( penumbra !== undefined ) ? penumbra : 0;
    	this.decay = ( decay !== undefined ) ? decay : 1;	// for physically correct lights, should be 2.

    	this.shadow = new SpotLightShadow();

    }

    SpotLight.prototype = Object.assign( Object.create( Light.prototype ), {

    	constructor: SpotLight,

    	isSpotLight: true,

    	copy: function ( source ) {

    		Light.prototype.copy.call( this, source );

    		this.distance = source.distance;
    		this.angle = source.angle;
    		this.penumbra = source.penumbra;
    		this.decay = source.decay;

    		this.target = source.target.clone();

    		this.shadow = source.shadow.clone();

    		return this;

    	}

    } );

    function PointLightShadow() {

    	LightShadow.call( this, new PerspectiveCamera( 90, 1, 0.5, 500 ) );

    	this._frameExtents = new Vector2( 4, 2 );

    	this._viewportCount = 6;

    	this._viewports = [
    		// These viewports map a cube-map onto a 2D texture with the
    		// following orientation:
    		//
    		//  xzXZ
    		//   y Y
    		//
    		// X - Positive x direction
    		// x - Negative x direction
    		// Y - Positive y direction
    		// y - Negative y direction
    		// Z - Positive z direction
    		// z - Negative z direction

    		// positive X
    		new Vector4( 2, 1, 1, 1 ),
    		// negative X
    		new Vector4( 0, 1, 1, 1 ),
    		// positive Z
    		new Vector4( 3, 1, 1, 1 ),
    		// negative Z
    		new Vector4( 1, 1, 1, 1 ),
    		// positive Y
    		new Vector4( 3, 0, 1, 1 ),
    		// negative Y
    		new Vector4( 1, 0, 1, 1 )
    	];

    	this._cubeDirections = [
    		new Vector3( 1, 0, 0 ), new Vector3( - 1, 0, 0 ), new Vector3( 0, 0, 1 ),
    		new Vector3( 0, 0, - 1 ), new Vector3( 0, 1, 0 ), new Vector3( 0, - 1, 0 )
    	];

    	this._cubeUps = [
    		new Vector3( 0, 1, 0 ), new Vector3( 0, 1, 0 ), new Vector3( 0, 1, 0 ),
    		new Vector3( 0, 1, 0 ), new Vector3( 0, 0, 1 ),	new Vector3( 0, 0, - 1 )
    	];

    }

    PointLightShadow.prototype = Object.assign( Object.create( LightShadow.prototype ), {

    	constructor: PointLightShadow,

    	isPointLightShadow: true,

    	updateMatrices: function ( light, viewCamera, viewportIndex ) {

    		var camera = this.camera,
    			shadowMatrix = this.matrix,
    			lightPositionWorld = this._lightPositionWorld,
    			lookTarget = this._lookTarget,
    			projScreenMatrix = this._projScreenMatrix;

    		lightPositionWorld.setFromMatrixPosition( light.matrixWorld );
    		camera.position.copy( lightPositionWorld );

    		lookTarget.copy( camera.position );
    		lookTarget.add( this._cubeDirections[ viewportIndex ] );
    		camera.up.copy( this._cubeUps[ viewportIndex ] );
    		camera.lookAt( lookTarget );
    		camera.updateMatrixWorld();

    		shadowMatrix.makeTranslation( - lightPositionWorld.x, - lightPositionWorld.y, - lightPositionWorld.z );

    		projScreenMatrix.multiplyMatrices( camera.projectionMatrix, camera.matrixWorldInverse );
    		this._frustum.setFromMatrix( projScreenMatrix );

    	}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     */


    function PointLight( color, intensity, distance, decay ) {

    	Light.call( this, color, intensity );

    	this.type = 'PointLight';

    	Object.defineProperty( this, 'power', {
    		get: function () {

    			// intensity = power per solid angle.
    			// ref: equation (15) from https://seblagarde.files.wordpress.com/2015/07/course_notes_moving_frostbite_to_pbr_v32.pdf
    			return this.intensity * 4 * Math.PI;

    		},
    		set: function ( power ) {

    			// intensity = power per solid angle.
    			// ref: equation (15) from https://seblagarde.files.wordpress.com/2015/07/course_notes_moving_frostbite_to_pbr_v32.pdf
    			this.intensity = power / ( 4 * Math.PI );

    		}
    	} );

    	this.distance = ( distance !== undefined ) ? distance : 0;
    	this.decay = ( decay !== undefined ) ? decay : 1;	// for physically correct lights, should be 2.

    	this.shadow = new PointLightShadow();

    }

    PointLight.prototype = Object.assign( Object.create( Light.prototype ), {

    	constructor: PointLight,

    	isPointLight: true,

    	copy: function ( source ) {

    		Light.prototype.copy.call( this, source );

    		this.distance = source.distance;
    		this.decay = source.decay;

    		this.shadow = source.shadow.clone();

    		return this;

    	}

    } );

    /**
     * @author alteredq / http://alteredqualia.com/
     * @author arose / http://github.com/arose
     */

    function OrthographicCamera( left, right, top, bottom, near, far ) {

    	Camera.call( this );

    	this.type = 'OrthographicCamera';

    	this.zoom = 1;
    	this.view = null;

    	this.left = ( left !== undefined ) ? left : - 1;
    	this.right = ( right !== undefined ) ? right : 1;
    	this.top = ( top !== undefined ) ? top : 1;
    	this.bottom = ( bottom !== undefined ) ? bottom : - 1;

    	this.near = ( near !== undefined ) ? near : 0.1;
    	this.far = ( far !== undefined ) ? far : 2000;

    	this.updateProjectionMatrix();

    }

    OrthographicCamera.prototype = Object.assign( Object.create( Camera.prototype ), {

    	constructor: OrthographicCamera,

    	isOrthographicCamera: true,

    	copy: function ( source, recursive ) {

    		Camera.prototype.copy.call( this, source, recursive );

    		this.left = source.left;
    		this.right = source.right;
    		this.top = source.top;
    		this.bottom = source.bottom;
    		this.near = source.near;
    		this.far = source.far;

    		this.zoom = source.zoom;
    		this.view = source.view === null ? null : Object.assign( {}, source.view );

    		return this;

    	},

    	setViewOffset: function ( fullWidth, fullHeight, x, y, width, height ) {

    		if ( this.view === null ) {

    			this.view = {
    				enabled: true,
    				fullWidth: 1,
    				fullHeight: 1,
    				offsetX: 0,
    				offsetY: 0,
    				width: 1,
    				height: 1
    			};

    		}

    		this.view.enabled = true;
    		this.view.fullWidth = fullWidth;
    		this.view.fullHeight = fullHeight;
    		this.view.offsetX = x;
    		this.view.offsetY = y;
    		this.view.width = width;
    		this.view.height = height;

    		this.updateProjectionMatrix();

    	},

    	clearViewOffset: function () {

    		if ( this.view !== null ) {

    			this.view.enabled = false;

    		}

    		this.updateProjectionMatrix();

    	},

    	updateProjectionMatrix: function () {

    		var dx = ( this.right - this.left ) / ( 2 * this.zoom );
    		var dy = ( this.top - this.bottom ) / ( 2 * this.zoom );
    		var cx = ( this.right + this.left ) / 2;
    		var cy = ( this.top + this.bottom ) / 2;

    		var left = cx - dx;
    		var right = cx + dx;
    		var top = cy + dy;
    		var bottom = cy - dy;

    		if ( this.view !== null && this.view.enabled ) {

    			var zoomW = this.zoom / ( this.view.width / this.view.fullWidth );
    			var zoomH = this.zoom / ( this.view.height / this.view.fullHeight );
    			var scaleW = ( this.right - this.left ) / this.view.width;
    			var scaleH = ( this.top - this.bottom ) / this.view.height;

    			left += scaleW * ( this.view.offsetX / zoomW );
    			right = left + scaleW * ( this.view.width / zoomW );
    			top -= scaleH * ( this.view.offsetY / zoomH );
    			bottom = top - scaleH * ( this.view.height / zoomH );

    		}

    		this.projectionMatrix.makeOrthographic( left, right, top, bottom, this.near, this.far );

    		this.projectionMatrixInverse.getInverse( this.projectionMatrix );

    	},

    	toJSON: function ( meta ) {

    		var data = Object3D.prototype.toJSON.call( this, meta );

    		data.object.zoom = this.zoom;
    		data.object.left = this.left;
    		data.object.right = this.right;
    		data.object.top = this.top;
    		data.object.bottom = this.bottom;
    		data.object.near = this.near;
    		data.object.far = this.far;

    		if ( this.view !== null ) data.object.view = Object.assign( {}, this.view );

    		return data;

    	}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function DirectionalLightShadow() {

    	LightShadow.call( this, new OrthographicCamera( - 5, 5, 5, - 5, 0.5, 500 ) );

    }

    DirectionalLightShadow.prototype = Object.assign( Object.create( LightShadow.prototype ), {

    	constructor: DirectionalLightShadow,

    	isDirectionalLightShadow: true,

    	updateMatrices: function ( light, viewCamera, viewportIndex ) {

    		LightShadow.prototype.updateMatrices.call( this, light, viewCamera, viewportIndex );

    	}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     * @author alteredq / http://alteredqualia.com/
     */

    function DirectionalLight( color, intensity ) {

    	Light.call( this, color, intensity );

    	this.type = 'DirectionalLight';

    	this.position.copy( Object3D.DefaultUp );
    	this.updateMatrix();

    	this.target = new Object3D();

    	this.shadow = new DirectionalLightShadow();

    }

    DirectionalLight.prototype = Object.assign( Object.create( Light.prototype ), {

    	constructor: DirectionalLight,

    	isDirectionalLight: true,

    	copy: function ( source ) {

    		Light.prototype.copy.call( this, source );

    		this.target = source.target.clone();

    		this.shadow = source.shadow.clone();

    		return this;

    	}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function AmbientLight( color, intensity ) {

    	Light.call( this, color, intensity );

    	this.type = 'AmbientLight';

    	this.castShadow = undefined;

    }

    AmbientLight.prototype = Object.assign( Object.create( Light.prototype ), {

    	constructor: AmbientLight,

    	isAmbientLight: true

    } );

    /**
     * @author abelnation / http://github.com/abelnation
     */

    function RectAreaLight( color, intensity, width, height ) {

    	Light.call( this, color, intensity );

    	this.type = 'RectAreaLight';

    	this.width = ( width !== undefined ) ? width : 10;
    	this.height = ( height !== undefined ) ? height : 10;

    }

    RectAreaLight.prototype = Object.assign( Object.create( Light.prototype ), {

    	constructor: RectAreaLight,

    	isRectAreaLight: true,

    	copy: function ( source ) {

    		Light.prototype.copy.call( this, source );

    		this.width = source.width;
    		this.height = source.height;

    		return this;

    	},

    	toJSON: function ( meta ) {

    		var data = Light.prototype.toJSON.call( this, meta );

    		data.object.width = this.width;
    		data.object.height = this.height;

    		return data;

    	}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function MaterialLoader( manager ) {

    	Loader.call( this, manager );

    	this.textures = {};

    }

    MaterialLoader.prototype = Object.assign( Object.create( Loader.prototype ), {

    	constructor: MaterialLoader,

    	load: function ( url, onLoad, onProgress, onError ) {

    		var scope = this;

    		var loader = new FileLoader( scope.manager );
    		loader.setPath( scope.path );
    		loader.load( url, function ( text ) {

    			onLoad( scope.parse( JSON.parse( text ) ) );

    		}, onProgress, onError );

    	},

    	parse: function ( json ) {

    		var textures = this.textures;

    		function getTexture( name ) {

    			if ( textures[ name ] === undefined ) {

    				console.warn( 'THREE.MaterialLoader: Undefined texture', name );

    			}

    			return textures[ name ];

    		}

    		var material = new Materials[ json.type ]();

    		if ( json.uuid !== undefined ) material.uuid = json.uuid;
    		if ( json.name !== undefined ) material.name = json.name;
    		if ( json.color !== undefined ) material.color.setHex( json.color );
    		if ( json.roughness !== undefined ) material.roughness = json.roughness;
    		if ( json.metalness !== undefined ) material.metalness = json.metalness;
    		if ( json.emissive !== undefined ) material.emissive.setHex( json.emissive );
    		if ( json.specular !== undefined ) material.specular.setHex( json.specular );
    		if ( json.shininess !== undefined ) material.shininess = json.shininess;
    		if ( json.clearcoat !== undefined ) material.clearcoat = json.clearcoat;
    		if ( json.clearcoatRoughness !== undefined ) material.clearcoatRoughness = json.clearcoatRoughness;
    		if ( json.vertexColors !== undefined ) material.vertexColors = json.vertexColors;
    		if ( json.fog !== undefined ) material.fog = json.fog;
    		if ( json.flatShading !== undefined ) material.flatShading = json.flatShading;
    		if ( json.blending !== undefined ) material.blending = json.blending;
    		if ( json.combine !== undefined ) material.combine = json.combine;
    		if ( json.side !== undefined ) material.side = json.side;
    		if ( json.opacity !== undefined ) material.opacity = json.opacity;
    		if ( json.transparent !== undefined ) material.transparent = json.transparent;
    		if ( json.alphaTest !== undefined ) material.alphaTest = json.alphaTest;
    		if ( json.depthTest !== undefined ) material.depthTest = json.depthTest;
    		if ( json.depthWrite !== undefined ) material.depthWrite = json.depthWrite;
    		if ( json.colorWrite !== undefined ) material.colorWrite = json.colorWrite;
    		if ( json.wireframe !== undefined ) material.wireframe = json.wireframe;
    		if ( json.wireframeLinewidth !== undefined ) material.wireframeLinewidth = json.wireframeLinewidth;
    		if ( json.wireframeLinecap !== undefined ) material.wireframeLinecap = json.wireframeLinecap;
    		if ( json.wireframeLinejoin !== undefined ) material.wireframeLinejoin = json.wireframeLinejoin;

    		if ( json.rotation !== undefined ) material.rotation = json.rotation;

    		if ( json.linewidth !== 1 ) material.linewidth = json.linewidth;
    		if ( json.dashSize !== undefined ) material.dashSize = json.dashSize;
    		if ( json.gapSize !== undefined ) material.gapSize = json.gapSize;
    		if ( json.scale !== undefined ) material.scale = json.scale;

    		if ( json.polygonOffset !== undefined ) material.polygonOffset = json.polygonOffset;
    		if ( json.polygonOffsetFactor !== undefined ) material.polygonOffsetFactor = json.polygonOffsetFactor;
    		if ( json.polygonOffsetUnits !== undefined ) material.polygonOffsetUnits = json.polygonOffsetUnits;

    		if ( json.skinning !== undefined ) material.skinning = json.skinning;
    		if ( json.morphTargets !== undefined ) material.morphTargets = json.morphTargets;
    		if ( json.morphNormals !== undefined ) material.morphNormals = json.morphNormals;
    		if ( json.dithering !== undefined ) material.dithering = json.dithering;

    		if ( json.visible !== undefined ) material.visible = json.visible;

    		if ( json.toneMapped !== undefined ) material.toneMapped = json.toneMapped;

    		if ( json.userData !== undefined ) material.userData = json.userData;

    		// Shader Material

    		if ( json.uniforms !== undefined ) {

    			for ( var name in json.uniforms ) {

    				var uniform = json.uniforms[ name ];

    				material.uniforms[ name ] = {};

    				switch ( uniform.type ) {

    					case 't':
    						material.uniforms[ name ].value = getTexture( uniform.value );
    						break;

    					case 'c':
    						material.uniforms[ name ].value = new Color$7().setHex( uniform.value );
    						break;

    					case 'v2':
    						material.uniforms[ name ].value = new Vector2().fromArray( uniform.value );
    						break;

    					case 'v3':
    						material.uniforms[ name ].value = new Vector3().fromArray( uniform.value );
    						break;

    					case 'v4':
    						material.uniforms[ name ].value = new Vector4().fromArray( uniform.value );
    						break;

    					case 'm3':
    						material.uniforms[ name ].value = new Matrix3().fromArray( uniform.value );

    					case 'm4':
    						material.uniforms[ name ].value = new Matrix4().fromArray( uniform.value );
    						break;

    					default:
    						material.uniforms[ name ].value = uniform.value;

    				}

    			}

    		}

    		if ( json.defines !== undefined ) material.defines = json.defines;
    		if ( json.vertexShader !== undefined ) material.vertexShader = json.vertexShader;
    		if ( json.fragmentShader !== undefined ) material.fragmentShader = json.fragmentShader;

    		if ( json.extensions !== undefined ) {

    			for ( var key in json.extensions ) {

    				material.extensions[ key ] = json.extensions[ key ];

    			}

    		}

    		// Deprecated

    		if ( json.shading !== undefined ) material.flatShading = json.shading === 1; // THREE.FlatShading

    		// for PointsMaterial

    		if ( json.size !== undefined ) material.size = json.size;
    		if ( json.sizeAttenuation !== undefined ) material.sizeAttenuation = json.sizeAttenuation;

    		// maps

    		if ( json.map !== undefined ) material.map = getTexture( json.map );
    		if ( json.matcap !== undefined ) material.matcap = getTexture( json.matcap );

    		if ( json.alphaMap !== undefined ) {

    			material.alphaMap = getTexture( json.alphaMap );
    			material.transparent = true;

    		}

    		if ( json.bumpMap !== undefined ) material.bumpMap = getTexture( json.bumpMap );
    		if ( json.bumpScale !== undefined ) material.bumpScale = json.bumpScale;

    		if ( json.normalMap !== undefined ) material.normalMap = getTexture( json.normalMap );
    		if ( json.normalMapType !== undefined ) material.normalMapType = json.normalMapType;
    		if ( json.normalScale !== undefined ) {

    			var normalScale = json.normalScale;

    			if ( Array.isArray( normalScale ) === false ) {

    				// Blender exporter used to export a scalar. See #7459

    				normalScale = [ normalScale, normalScale ];

    			}

    			material.normalScale = new Vector2().fromArray( normalScale );

    		}

    		if ( json.displacementMap !== undefined ) material.displacementMap = getTexture( json.displacementMap );
    		if ( json.displacementScale !== undefined ) material.displacementScale = json.displacementScale;
    		if ( json.displacementBias !== undefined ) material.displacementBias = json.displacementBias;

    		if ( json.roughnessMap !== undefined ) material.roughnessMap = getTexture( json.roughnessMap );
    		if ( json.metalnessMap !== undefined ) material.metalnessMap = getTexture( json.metalnessMap );

    		if ( json.emissiveMap !== undefined ) material.emissiveMap = getTexture( json.emissiveMap );
    		if ( json.emissiveIntensity !== undefined ) material.emissiveIntensity = json.emissiveIntensity;

    		if ( json.specularMap !== undefined ) material.specularMap = getTexture( json.specularMap );

    		if ( json.envMap !== undefined ) material.envMap = getTexture( json.envMap );
    		if ( json.envMapIntensity !== undefined ) material.envMapIntensity = json.envMapIntensity;

    		if ( json.reflectivity !== undefined ) material.reflectivity = json.reflectivity;
    		if ( json.refractionRatio !== undefined ) material.refractionRatio = json.refractionRatio;

    		if ( json.lightMap !== undefined ) material.lightMap = getTexture( json.lightMap );
    		if ( json.lightMapIntensity !== undefined ) material.lightMapIntensity = json.lightMapIntensity;

    		if ( json.aoMap !== undefined ) material.aoMap = getTexture( json.aoMap );
    		if ( json.aoMapIntensity !== undefined ) material.aoMapIntensity = json.aoMapIntensity;

    		if ( json.gradientMap !== undefined ) material.gradientMap = getTexture( json.gradientMap );

    		if ( json.clearcoatNormalMap !== undefined ) material.clearcoatNormalMap = getTexture( json.clearcoatNormalMap );
    		if ( json.clearcoatNormalScale !== undefined ) material.clearcoatNormalScale = new Vector2().fromArray( json.clearcoatNormalScale );

    		return material;

    	},

    	setTextures: function ( value ) {

    		this.textures = value;
    		return this;

    	}

    } );

    /**
     * @author Don McCurdy / https://www.donmccurdy.com
     */

    var LoaderUtils = {

    	decodeText: function ( array ) {

    		if ( typeof TextDecoder !== 'undefined' ) {

    			return new TextDecoder().decode( array );

    		}

    		// Avoid the String.fromCharCode.apply(null, array) shortcut, which
    		// throws a "maximum call stack size exceeded" error for large arrays.

    		var s = '';

    		for ( var i = 0, il = array.length; i < il; i ++ ) {

    			// Implicitly assumes little-endian.
    			s += String.fromCharCode( array[ i ] );

    		}

    		try {

    			// merges multi-byte utf-8 characters.

    			return decodeURIComponent( escape( s ) );

    		} catch ( e ) { // see #16358

    			return s;

    		}

    	},

    	extractUrlBase: function ( url ) {

    		var index = url.lastIndexOf( '/' );

    		if ( index === - 1 ) return './';

    		return url.substr( 0, index + 1 );

    	}

    };

    /**
     * @author benaadams / https://twitter.com/ben_a_adams
     */

    function InstancedBufferGeometry() {

    	BufferGeometry.call( this );

    	this.type = 'InstancedBufferGeometry';
    	this.maxInstancedCount = undefined;

    }

    InstancedBufferGeometry.prototype = Object.assign( Object.create( BufferGeometry.prototype ), {

    	constructor: InstancedBufferGeometry,

    	isInstancedBufferGeometry: true,

    	copy: function ( source ) {

    		BufferGeometry.prototype.copy.call( this, source );

    		this.maxInstancedCount = source.maxInstancedCount;

    		return this;

    	},

    	clone: function () {

    		return new this.constructor().copy( this );

    	},

    	toJSON: function () {

    		var data = BufferGeometry.prototype.toJSON.call( this );

    		data.maxInstancedCount = this.maxInstancedCount;

    		data.isInstancedBufferGeometry = true;

    		return data;

    	}

    } );

    /**
     * @author benaadams / https://twitter.com/ben_a_adams
     */

    function InstancedBufferAttribute( array, itemSize, normalized, meshPerAttribute ) {

    	if ( typeof ( normalized ) === 'number' ) {

    		meshPerAttribute = normalized;

    		normalized = false;

    		console.error( 'THREE.InstancedBufferAttribute: The constructor now expects normalized as the third argument.' );

    	}

    	BufferAttribute.call( this, array, itemSize, normalized );

    	this.meshPerAttribute = meshPerAttribute || 1;

    }

    InstancedBufferAttribute.prototype = Object.assign( Object.create( BufferAttribute.prototype ), {

    	constructor: InstancedBufferAttribute,

    	isInstancedBufferAttribute: true,

    	copy: function ( source ) {

    		BufferAttribute.prototype.copy.call( this, source );

    		this.meshPerAttribute = source.meshPerAttribute;

    		return this;

    	},

    	toJSON: function ()	{

    		var data = BufferAttribute.prototype.toJSON.call( this );

    		data.meshPerAttribute = this.meshPerAttribute;

    		data.isInstancedBufferAttribute = true;

    		return data;

    	}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function BufferGeometryLoader( manager ) {

    	Loader.call( this, manager );

    }

    BufferGeometryLoader.prototype = Object.assign( Object.create( Loader.prototype ), {

    	constructor: BufferGeometryLoader,

    	load: function ( url, onLoad, onProgress, onError ) {

    		var scope = this;

    		var loader = new FileLoader( scope.manager );
    		loader.setPath( scope.path );
    		loader.load( url, function ( text ) {

    			onLoad( scope.parse( JSON.parse( text ) ) );

    		}, onProgress, onError );

    	},

    	parse: function ( json ) {

    		var geometry = json.isInstancedBufferGeometry ? new InstancedBufferGeometry() : new BufferGeometry();

    		var index = json.data.index;

    		if ( index !== undefined ) {

    			var typedArray = new TYPED_ARRAYS[ index.type ]( index.array );
    			geometry.setIndex( new BufferAttribute( typedArray, 1 ) );

    		}

    		var attributes = json.data.attributes;

    		for ( var key in attributes ) {

    			var attribute = attributes[ key ];
    			var typedArray = new TYPED_ARRAYS[ attribute.type ]( attribute.array );
    			var bufferAttributeConstr = attribute.isInstancedBufferAttribute ? InstancedBufferAttribute : BufferAttribute;
    			var bufferAttribute = new bufferAttributeConstr( typedArray, attribute.itemSize, attribute.normalized );
    			if ( attribute.name !== undefined ) bufferAttribute.name = attribute.name;
    			geometry.addAttribute( key, bufferAttribute );

    		}

    		var morphAttributes = json.data.morphAttributes;

    		if ( morphAttributes ) {

    			for ( var key in morphAttributes ) {

    				var attributeArray = morphAttributes[ key ];

    				var array = [];

    				for ( var i = 0, il = attributeArray.length; i < il; i ++ ) {

    					var attribute = attributeArray[ i ];
    					var typedArray = new TYPED_ARRAYS[ attribute.type ]( attribute.array );

    					var bufferAttribute = new BufferAttribute( typedArray, attribute.itemSize, attribute.normalized );
    					if ( attribute.name !== undefined ) bufferAttribute.name = attribute.name;
    					array.push( bufferAttribute );

    				}

    				geometry.morphAttributes[ key ] = array;

    			}

    		}

    		var groups = json.data.groups || json.data.drawcalls || json.data.offsets;

    		if ( groups !== undefined ) {

    			for ( var i = 0, n = groups.length; i !== n; ++ i ) {

    				var group = groups[ i ];

    				geometry.addGroup( group.start, group.count, group.materialIndex );

    			}

    		}

    		var boundingSphere = json.data.boundingSphere;

    		if ( boundingSphere !== undefined ) {

    			var center = new Vector3();

    			if ( boundingSphere.center !== undefined ) {

    				center.fromArray( boundingSphere.center );

    			}

    			geometry.boundingSphere = new Sphere( center, boundingSphere.radius );

    		}

    		if ( json.name ) geometry.name = json.name;
    		if ( json.userData ) geometry.userData = json.userData;

    		return geometry;

    	}

    } );

    var TYPED_ARRAYS = {
    	Int8Array: Int8Array,
    	Uint8Array: Uint8Array,
    	// Workaround for IE11 pre KB2929437. See #11440
    	Uint8ClampedArray: typeof Uint8ClampedArray !== 'undefined' ? Uint8ClampedArray : Uint8Array,
    	Int16Array: Int16Array,
    	Uint16Array: Uint16Array,
    	Int32Array: Int32Array,
    	Uint32Array: Uint32Array,
    	Float32Array: Float32Array,
    	Float64Array: Float64Array
    };

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function ObjectLoader( manager ) {

    	Loader.call( this, manager );

    }

    ObjectLoader.prototype = Object.assign( Object.create( Loader.prototype ), {

    	constructor: ObjectLoader,

    	load: function ( url, onLoad, onProgress, onError ) {

    		var scope = this;

    		var path = ( this.path === '' ) ? LoaderUtils.extractUrlBase( url ) : this.path;
    		this.resourcePath = this.resourcePath || path;

    		var loader = new FileLoader( scope.manager );
    		loader.setPath( this.path );
    		loader.load( url, function ( text ) {

    			var json = null;

    			try {

    				json = JSON.parse( text );

    			} catch ( error ) {

    				if ( onError !== undefined ) onError( error );

    				console.error( 'THREE:ObjectLoader: Can\'t parse ' + url + '.', error.message );

    				return;

    			}

    			var metadata = json.metadata;

    			if ( metadata === undefined || metadata.type === undefined || metadata.type.toLowerCase() === 'geometry' ) {

    				console.error( 'THREE.ObjectLoader: Can\'t load ' + url );
    				return;

    			}

    			scope.parse( json, onLoad );

    		}, onProgress, onError );

    	},

    	parse: function ( json, onLoad ) {

    		var shapes = this.parseShape( json.shapes );
    		var geometries = this.parseGeometries( json.geometries, shapes );

    		var images = this.parseImages( json.images, function () {

    			if ( onLoad !== undefined ) onLoad( object );

    		} );

    		var textures = this.parseTextures( json.textures, images );
    		var materials = this.parseMaterials( json.materials, textures );

    		var object = this.parseObject( json.object, geometries, materials );

    		if ( json.animations ) {

    			object.animations = this.parseAnimations( json.animations );

    		}

    		if ( json.images === undefined || json.images.length === 0 ) {

    			if ( onLoad !== undefined ) onLoad( object );

    		}

    		return object;

    	},

    	parseShape: function ( json ) {

    		var shapes = {};

    		if ( json !== undefined ) {

    			for ( var i = 0, l = json.length; i < l; i ++ ) {

    				var shape = new Shape().fromJSON( json[ i ] );

    				shapes[ shape.uuid ] = shape;

    			}

    		}

    		return shapes;

    	},

    	parseGeometries: function ( json, shapes ) {

    		var geometries = {};

    		if ( json !== undefined ) {

    			var bufferGeometryLoader = new BufferGeometryLoader();

    			for ( var i = 0, l = json.length; i < l; i ++ ) {

    				var geometry;
    				var data = json[ i ];

    				switch ( data.type ) {

    					case 'PlaneGeometry':
    					case 'PlaneBufferGeometry':

    						geometry = new Geometries[ data.type ](
    							data.width,
    							data.height,
    							data.widthSegments,
    							data.heightSegments
    						);

    						break;

    					case 'BoxGeometry':
    					case 'BoxBufferGeometry':
    					case 'CubeGeometry': // backwards compatible

    						geometry = new Geometries[ data.type ](
    							data.width,
    							data.height,
    							data.depth,
    							data.widthSegments,
    							data.heightSegments,
    							data.depthSegments
    						);

    						break;

    					case 'CircleGeometry':
    					case 'CircleBufferGeometry':

    						geometry = new Geometries[ data.type ](
    							data.radius,
    							data.segments,
    							data.thetaStart,
    							data.thetaLength
    						);

    						break;

    					case 'CylinderGeometry':
    					case 'CylinderBufferGeometry':

    						geometry = new Geometries[ data.type ](
    							data.radiusTop,
    							data.radiusBottom,
    							data.height,
    							data.radialSegments,
    							data.heightSegments,
    							data.openEnded,
    							data.thetaStart,
    							data.thetaLength
    						);

    						break;

    					case 'ConeGeometry':
    					case 'ConeBufferGeometry':

    						geometry = new Geometries[ data.type ](
    							data.radius,
    							data.height,
    							data.radialSegments,
    							data.heightSegments,
    							data.openEnded,
    							data.thetaStart,
    							data.thetaLength
    						);

    						break;

    					case 'SphereGeometry':
    					case 'SphereBufferGeometry':

    						geometry = new Geometries[ data.type ](
    							data.radius,
    							data.widthSegments,
    							data.heightSegments,
    							data.phiStart,
    							data.phiLength,
    							data.thetaStart,
    							data.thetaLength
    						);

    						break;

    					case 'DodecahedronGeometry':
    					case 'DodecahedronBufferGeometry':
    					case 'IcosahedronGeometry':
    					case 'IcosahedronBufferGeometry':
    					case 'OctahedronGeometry':
    					case 'OctahedronBufferGeometry':
    					case 'TetrahedronGeometry':
    					case 'TetrahedronBufferGeometry':

    						geometry = new Geometries[ data.type ](
    							data.radius,
    							data.detail
    						);

    						break;

    					case 'RingGeometry':
    					case 'RingBufferGeometry':

    						geometry = new Geometries[ data.type ](
    							data.innerRadius,
    							data.outerRadius,
    							data.thetaSegments,
    							data.phiSegments,
    							data.thetaStart,
    							data.thetaLength
    						);

    						break;

    					case 'TorusGeometry':
    					case 'TorusBufferGeometry':

    						geometry = new Geometries[ data.type ](
    							data.radius,
    							data.tube,
    							data.radialSegments,
    							data.tubularSegments,
    							data.arc
    						);

    						break;

    					case 'TorusKnotGeometry':
    					case 'TorusKnotBufferGeometry':

    						geometry = new Geometries[ data.type ](
    							data.radius,
    							data.tube,
    							data.tubularSegments,
    							data.radialSegments,
    							data.p,
    							data.q
    						);

    						break;

    					case 'TubeGeometry':
    					case 'TubeBufferGeometry':

    						// This only works for built-in curves (e.g. CatmullRomCurve3).
    						// User defined curves or instances of CurvePath will not be deserialized.
    						geometry = new Geometries[ data.type ](
    							new Curves[ data.path.type ]().fromJSON( data.path ),
    							data.tubularSegments,
    							data.radius,
    							data.radialSegments,
    							data.closed
    						);

    						break;

    					case 'LatheGeometry':
    					case 'LatheBufferGeometry':

    						geometry = new Geometries[ data.type ](
    							data.points,
    							data.segments,
    							data.phiStart,
    							data.phiLength
    						);

    						break;

    					case 'PolyhedronGeometry':
    					case 'PolyhedronBufferGeometry':

    						geometry = new Geometries[ data.type ](
    							data.vertices,
    							data.indices,
    							data.radius,
    							data.details
    						);

    						break;

    					case 'ShapeGeometry':
    					case 'ShapeBufferGeometry':

    						var geometryShapes = [];

    						for ( var j = 0, jl = data.shapes.length; j < jl; j ++ ) {

    							var shape = shapes[ data.shapes[ j ] ];

    							geometryShapes.push( shape );

    						}

    						geometry = new Geometries[ data.type ](
    							geometryShapes,
    							data.curveSegments
    						);

    						break;


    					case 'ExtrudeGeometry':
    					case 'ExtrudeBufferGeometry':

    						var geometryShapes = [];

    						for ( var j = 0, jl = data.shapes.length; j < jl; j ++ ) {

    							var shape = shapes[ data.shapes[ j ] ];

    							geometryShapes.push( shape );

    						}

    						var extrudePath = data.options.extrudePath;

    						if ( extrudePath !== undefined ) {

    							data.options.extrudePath = new Curves[ extrudePath.type ]().fromJSON( extrudePath );

    						}

    						geometry = new Geometries[ data.type ](
    							geometryShapes,
    							data.options
    						);

    						break;

    					case 'BufferGeometry':
    					case 'InstancedBufferGeometry':

    						geometry = bufferGeometryLoader.parse( data );

    						break;

    					case 'Geometry':

    						if ( 'THREE' in window && 'LegacyJSONLoader' in THREE ) {

    							var geometryLoader = new THREE.LegacyJSONLoader();
    							geometry = geometryLoader.parse( data, this.resourcePath ).geometry;


    						} else {

    							console.error( 'THREE.ObjectLoader: You have to import LegacyJSONLoader in order load geometry data of type "Geometry".' );

    						}

    						break;

    					default:

    						console.warn( 'THREE.ObjectLoader: Unsupported geometry type "' + data.type + '"' );

    						continue;

    				}

    				geometry.uuid = data.uuid;

    				if ( data.name !== undefined ) geometry.name = data.name;
    				if ( geometry.isBufferGeometry === true && data.userData !== undefined ) geometry.userData = data.userData;

    				geometries[ data.uuid ] = geometry;

    			}

    		}

    		return geometries;

    	},

    	parseMaterials: function ( json, textures ) {

    		var cache = {}; // MultiMaterial
    		var materials = {};

    		if ( json !== undefined ) {

    			var loader = new MaterialLoader();
    			loader.setTextures( textures );

    			for ( var i = 0, l = json.length; i < l; i ++ ) {

    				var data = json[ i ];

    				if ( data.type === 'MultiMaterial' ) {

    					// Deprecated

    					var array = [];

    					for ( var j = 0; j < data.materials.length; j ++ ) {

    						var material = data.materials[ j ];

    						if ( cache[ material.uuid ] === undefined ) {

    							cache[ material.uuid ] = loader.parse( material );

    						}

    						array.push( cache[ material.uuid ] );

    					}

    					materials[ data.uuid ] = array;

    				} else {

    					if ( cache[ data.uuid ] === undefined ) {

    						cache[ data.uuid ] = loader.parse( data );

    					}

    					materials[ data.uuid ] = cache[ data.uuid ];

    				}

    			}

    		}

    		return materials;

    	},

    	parseAnimations: function ( json ) {

    		var animations = [];

    		for ( var i = 0; i < json.length; i ++ ) {

    			var data = json[ i ];

    			var clip = AnimationClip.parse( data );

    			if ( data.uuid !== undefined ) clip.uuid = data.uuid;

    			animations.push( clip );

    		}

    		return animations;

    	},

    	parseImages: function ( json, onLoad ) {

    		var scope = this;
    		var images = {};

    		function loadImage( url ) {

    			scope.manager.itemStart( url );

    			return loader.load( url, function () {

    				scope.manager.itemEnd( url );

    			}, undefined, function () {

    				scope.manager.itemError( url );
    				scope.manager.itemEnd( url );

    			} );

    		}

    		if ( json !== undefined && json.length > 0 ) {

    			var manager = new LoadingManager( onLoad );

    			var loader = new ImageLoader( manager );
    			loader.setCrossOrigin( this.crossOrigin );

    			for ( var i = 0, il = json.length; i < il; i ++ ) {

    				var image = json[ i ];
    				var url = image.url;

    				if ( Array.isArray( url ) ) {

    					// load array of images e.g CubeTexture

    					images[ image.uuid ] = [];

    					for ( var j = 0, jl = url.length; j < jl; j ++ ) {

    						var currentUrl = url[ j ];

    						var path = /^(\/\/)|([a-z]+:(\/\/)?)/i.test( currentUrl ) ? currentUrl : scope.resourcePath + currentUrl;

    						images[ image.uuid ].push( loadImage( path ) );

    					}

    				} else {

    					// load single image

    					var path = /^(\/\/)|([a-z]+:(\/\/)?)/i.test( image.url ) ? image.url : scope.resourcePath + image.url;

    					images[ image.uuid ] = loadImage( path );

    				}

    			}

    		}

    		return images;

    	},

    	parseTextures: function ( json, images ) {

    		function parseConstant( value, type ) {

    			if ( typeof value === 'number' ) return value;

    			console.warn( 'THREE.ObjectLoader.parseTexture: Constant should be in numeric form.', value );

    			return type[ value ];

    		}

    		var textures = {};

    		if ( json !== undefined ) {

    			for ( var i = 0, l = json.length; i < l; i ++ ) {

    				var data = json[ i ];

    				if ( data.image === undefined ) {

    					console.warn( 'THREE.ObjectLoader: No "image" specified for', data.uuid );

    				}

    				if ( images[ data.image ] === undefined ) {

    					console.warn( 'THREE.ObjectLoader: Undefined image', data.image );

    				}

    				var texture;

    				if ( Array.isArray( images[ data.image ] ) ) {

    					texture = new CubeTexture( images[ data.image ] );

    				} else {

    					texture = new Texture( images[ data.image ] );

    				}

    				texture.needsUpdate = true;

    				texture.uuid = data.uuid;

    				if ( data.name !== undefined ) texture.name = data.name;

    				if ( data.mapping !== undefined ) texture.mapping = parseConstant( data.mapping, TEXTURE_MAPPING );

    				if ( data.offset !== undefined ) texture.offset.fromArray( data.offset );
    				if ( data.repeat !== undefined ) texture.repeat.fromArray( data.repeat );
    				if ( data.center !== undefined ) texture.center.fromArray( data.center );
    				if ( data.rotation !== undefined ) texture.rotation = data.rotation;

    				if ( data.wrap !== undefined ) {

    					texture.wrapS = parseConstant( data.wrap[ 0 ], TEXTURE_WRAPPING );
    					texture.wrapT = parseConstant( data.wrap[ 1 ], TEXTURE_WRAPPING );

    				}

    				if ( data.format !== undefined ) texture.format = data.format;
    				if ( data.type !== undefined ) texture.type = data.type;
    				if ( data.encoding !== undefined ) texture.encoding = data.encoding;

    				if ( data.minFilter !== undefined ) texture.minFilter = parseConstant( data.minFilter, TEXTURE_FILTER );
    				if ( data.magFilter !== undefined ) texture.magFilter = parseConstant( data.magFilter, TEXTURE_FILTER );
    				if ( data.anisotropy !== undefined ) texture.anisotropy = data.anisotropy;

    				if ( data.flipY !== undefined ) texture.flipY = data.flipY;

    				if ( data.premultiplyAlpha !== undefined ) texture.premultiplyAlpha = data.premultiplyAlpha;
    				if ( data.unpackAlignment !== undefined ) texture.unpackAlignment = data.unpackAlignment;

    				textures[ data.uuid ] = texture;

    			}

    		}

    		return textures;

    	},

    	parseObject: function ( data, geometries, materials ) {

    		var object;

    		function getGeometry( name ) {

    			if ( geometries[ name ] === undefined ) {

    				console.warn( 'THREE.ObjectLoader: Undefined geometry', name );

    			}

    			return geometries[ name ];

    		}

    		function getMaterial( name ) {

    			if ( name === undefined ) return undefined;

    			if ( Array.isArray( name ) ) {

    				var array = [];

    				for ( var i = 0, l = name.length; i < l; i ++ ) {

    					var uuid = name[ i ];

    					if ( materials[ uuid ] === undefined ) {

    						console.warn( 'THREE.ObjectLoader: Undefined material', uuid );

    					}

    					array.push( materials[ uuid ] );

    				}

    				return array;

    			}

    			if ( materials[ name ] === undefined ) {

    				console.warn( 'THREE.ObjectLoader: Undefined material', name );

    			}

    			return materials[ name ];

    		}

    		switch ( data.type ) {

    			case 'Scene':

    				object = new Scene();

    				if ( data.background !== undefined ) {

    					if ( Number.isInteger( data.background ) ) {

    						object.background = new Color$7( data.background );

    					}

    				}

    				if ( data.fog !== undefined ) {

    					if ( data.fog.type === 'Fog' ) {

    						object.fog = new Fog( data.fog.color, data.fog.near, data.fog.far );

    					} else if ( data.fog.type === 'FogExp2' ) {

    						object.fog = new FogExp2( data.fog.color, data.fog.density );

    					}

    				}

    				break;

    			case 'PerspectiveCamera':

    				object = new PerspectiveCamera( data.fov, data.aspect, data.near, data.far );

    				if ( data.focus !== undefined ) object.focus = data.focus;
    				if ( data.zoom !== undefined ) object.zoom = data.zoom;
    				if ( data.filmGauge !== undefined ) object.filmGauge = data.filmGauge;
    				if ( data.filmOffset !== undefined ) object.filmOffset = data.filmOffset;
    				if ( data.view !== undefined ) object.view = Object.assign( {}, data.view );

    				break;

    			case 'OrthographicCamera':

    				object = new OrthographicCamera( data.left, data.right, data.top, data.bottom, data.near, data.far );

    				if ( data.zoom !== undefined ) object.zoom = data.zoom;
    				if ( data.view !== undefined ) object.view = Object.assign( {}, data.view );

    				break;

    			case 'AmbientLight':

    				object = new AmbientLight( data.color, data.intensity );

    				break;

    			case 'DirectionalLight':

    				object = new DirectionalLight( data.color, data.intensity );

    				break;

    			case 'PointLight':

    				object = new PointLight( data.color, data.intensity, data.distance, data.decay );

    				break;

    			case 'RectAreaLight':

    				object = new RectAreaLight( data.color, data.intensity, data.width, data.height );

    				break;

    			case 'SpotLight':

    				object = new SpotLight( data.color, data.intensity, data.distance, data.angle, data.penumbra, data.decay );

    				break;

    			case 'HemisphereLight':

    				object = new HemisphereLight( data.color, data.groundColor, data.intensity );

    				break;

    			case 'SkinnedMesh':

    				console.warn( 'THREE.ObjectLoader.parseObject() does not support SkinnedMesh yet.' );

    			case 'Mesh':

    				var geometry = getGeometry( data.geometry );
    				var material = getMaterial( data.material );

    				if ( geometry.bones && geometry.bones.length > 0 ) {

    					object = new SkinnedMesh( geometry, material );

    				} else {

    					object = new Mesh( geometry, material );

    				}

    				if ( data.drawMode !== undefined ) object.setDrawMode( data.drawMode );

    				break;

    			case 'LOD':

    				object = new LOD();

    				break;

    			case 'Line':

    				object = new Line( getGeometry( data.geometry ), getMaterial( data.material ), data.mode );

    				break;

    			case 'LineLoop':

    				object = new LineLoop( getGeometry( data.geometry ), getMaterial( data.material ) );

    				break;

    			case 'LineSegments':

    				object = new LineSegments( getGeometry( data.geometry ), getMaterial( data.material ) );

    				break;

    			case 'PointCloud':
    			case 'Points':

    				object = new Points( getGeometry( data.geometry ), getMaterial( data.material ) );

    				break;

    			case 'Sprite':

    				object = new Sprite( getMaterial( data.material ) );

    				break;

    			case 'Group':

    				object = new Group();

    				break;

    			default:

    				object = new Object3D();

    		}

    		object.uuid = data.uuid;

    		if ( data.name !== undefined ) object.name = data.name;

    		if ( data.matrix !== undefined ) {

    			object.matrix.fromArray( data.matrix );

    			if ( data.matrixAutoUpdate !== undefined ) object.matrixAutoUpdate = data.matrixAutoUpdate;
    			if ( object.matrixAutoUpdate ) object.matrix.decompose( object.position, object.quaternion, object.scale );

    		} else {

    			if ( data.position !== undefined ) object.position.fromArray( data.position );
    			if ( data.rotation !== undefined ) object.rotation.fromArray( data.rotation );
    			if ( data.quaternion !== undefined ) object.quaternion.fromArray( data.quaternion );
    			if ( data.scale !== undefined ) object.scale.fromArray( data.scale );

    		}

    		if ( data.castShadow !== undefined ) object.castShadow = data.castShadow;
    		if ( data.receiveShadow !== undefined ) object.receiveShadow = data.receiveShadow;

    		if ( data.shadow ) {

    			if ( data.shadow.bias !== undefined ) object.shadow.bias = data.shadow.bias;
    			if ( data.shadow.radius !== undefined ) object.shadow.radius = data.shadow.radius;
    			if ( data.shadow.mapSize !== undefined ) object.shadow.mapSize.fromArray( data.shadow.mapSize );
    			if ( data.shadow.camera !== undefined ) object.shadow.camera = this.parseObject( data.shadow.camera );

    		}

    		if ( data.visible !== undefined ) object.visible = data.visible;
    		if ( data.frustumCulled !== undefined ) object.frustumCulled = data.frustumCulled;
    		if ( data.renderOrder !== undefined ) object.renderOrder = data.renderOrder;
    		if ( data.userData !== undefined ) object.userData = data.userData;
    		if ( data.layers !== undefined ) object.layers.mask = data.layers;

    		if ( data.children !== undefined ) {

    			var children = data.children;

    			for ( var i = 0; i < children.length; i ++ ) {

    				object.add( this.parseObject( children[ i ], geometries, materials ) );

    			}

    		}

    		if ( data.type === 'LOD' ) {

    			var levels = data.levels;

    			for ( var l = 0; l < levels.length; l ++ ) {

    				var level = levels[ l ];
    				var child = object.getObjectByProperty( 'uuid', level.object );

    				if ( child !== undefined ) {

    					object.addLevel( child, level.distance );

    				}

    			}

    		}

    		return object;

    	}

    } );

    var TEXTURE_MAPPING = {
    	UVMapping: UVMapping,
    	CubeReflectionMapping: CubeReflectionMapping,
    	CubeRefractionMapping: CubeRefractionMapping,
    	EquirectangularReflectionMapping: EquirectangularReflectionMapping,
    	EquirectangularRefractionMapping: EquirectangularRefractionMapping,
    	SphericalReflectionMapping: SphericalReflectionMapping,
    	CubeUVReflectionMapping: CubeUVReflectionMapping,
    	CubeUVRefractionMapping: CubeUVRefractionMapping
    };

    var TEXTURE_WRAPPING = {
    	RepeatWrapping: RepeatWrapping,
    	ClampToEdgeWrapping: ClampToEdgeWrapping,
    	MirroredRepeatWrapping: MirroredRepeatWrapping
    };

    var TEXTURE_FILTER = {
    	NearestFilter: NearestFilter,
    	NearestMipmapNearestFilter: NearestMipmapNearestFilter,
    	NearestMipmapLinearFilter: NearestMipmapLinearFilter,
    	LinearFilter: LinearFilter,
    	LinearMipmapNearestFilter: LinearMipmapNearestFilter,
    	LinearMipmapLinearFilter: LinearMipmapLinearFilter
    };

    /**
     * @author thespite / http://clicktorelease.com/
     */


    function ImageBitmapLoader( manager ) {

    	if ( typeof createImageBitmap === 'undefined' ) {

    		console.warn( 'THREE.ImageBitmapLoader: createImageBitmap() not supported.' );

    	}

    	if ( typeof fetch === 'undefined' ) {

    		console.warn( 'THREE.ImageBitmapLoader: fetch() not supported.' );

    	}

    	Loader.call( this, manager );

    	this.options = undefined;

    }

    ImageBitmapLoader.prototype = Object.assign( Object.create( Loader.prototype ), {

    	constructor: ImageBitmapLoader,

    	setOptions: function setOptions( options ) {

    		this.options = options;

    		return this;

    	},

    	load: function ( url, onLoad, onProgress, onError ) {

    		if ( url === undefined ) url = '';

    		if ( this.path !== undefined ) url = this.path + url;

    		url = this.manager.resolveURL( url );

    		var scope = this;

    		var cached = Cache.get( url );

    		if ( cached !== undefined ) {

    			scope.manager.itemStart( url );

    			setTimeout( function () {

    				if ( onLoad ) onLoad( cached );

    				scope.manager.itemEnd( url );

    			}, 0 );

    			return cached;

    		}

    		fetch( url ).then( function ( res ) {

    			return res.blob();

    		} ).then( function ( blob ) {

    			if ( scope.options === undefined ) {

    				// Workaround for FireFox. It causes an error if you pass options.
    				return createImageBitmap( blob );

    			} else {

    				return createImageBitmap( blob, scope.options );

    			}

    		} ).then( function ( imageBitmap ) {

    			Cache.add( url, imageBitmap );

    			if ( onLoad ) onLoad( imageBitmap );

    			scope.manager.itemEnd( url );

    		} ).catch( function ( e ) {

    			if ( onError ) onError( e );

    			scope.manager.itemError( url );
    			scope.manager.itemEnd( url );

    		} );

    		scope.manager.itemStart( url );

    	}

    } );

    /**
     * @author zz85 / http://www.lab4games.net/zz85/blog
     * minimal class for proxing functions to Path. Replaces old "extractSubpaths()"
     **/

    function ShapePath() {

    	this.type = 'ShapePath';

    	this.color = new Color$7();

    	this.subPaths = [];
    	this.currentPath = null;

    }

    Object.assign( ShapePath.prototype, {

    	moveTo: function ( x, y ) {

    		this.currentPath = new Path$4();
    		this.subPaths.push( this.currentPath );
    		this.currentPath.moveTo( x, y );

    	},

    	lineTo: function ( x, y ) {

    		this.currentPath.lineTo( x, y );

    	},

    	quadraticCurveTo: function ( aCPx, aCPy, aX, aY ) {

    		this.currentPath.quadraticCurveTo( aCPx, aCPy, aX, aY );

    	},

    	bezierCurveTo: function ( aCP1x, aCP1y, aCP2x, aCP2y, aX, aY ) {

    		this.currentPath.bezierCurveTo( aCP1x, aCP1y, aCP2x, aCP2y, aX, aY );

    	},

    	splineThru: function ( pts ) {

    		this.currentPath.splineThru( pts );

    	},

    	toShapes: function ( isCCW, noHoles ) {

    		function toShapesNoHoles( inSubpaths ) {

    			var shapes = [];

    			for ( var i = 0, l = inSubpaths.length; i < l; i ++ ) {

    				var tmpPath = inSubpaths[ i ];

    				var tmpShape = new Shape();
    				tmpShape.curves = tmpPath.curves;

    				shapes.push( tmpShape );

    			}

    			return shapes;

    		}

    		function isPointInsidePolygon( inPt, inPolygon ) {

    			var polyLen = inPolygon.length;

    			// inPt on polygon contour => immediate success    or
    			// toggling of inside/outside at every single! intersection point of an edge
    			//  with the horizontal line through inPt, left of inPt
    			//  not counting lowerY endpoints of edges and whole edges on that line
    			var inside = false;
    			for ( var p = polyLen - 1, q = 0; q < polyLen; p = q ++ ) {

    				var edgeLowPt = inPolygon[ p ];
    				var edgeHighPt = inPolygon[ q ];

    				var edgeDx = edgeHighPt.x - edgeLowPt.x;
    				var edgeDy = edgeHighPt.y - edgeLowPt.y;

    				if ( Math.abs( edgeDy ) > Number.EPSILON ) {

    					// not parallel
    					if ( edgeDy < 0 ) {

    						edgeLowPt = inPolygon[ q ]; edgeDx = - edgeDx;
    						edgeHighPt = inPolygon[ p ]; edgeDy = - edgeDy;

    					}
    					if ( ( inPt.y < edgeLowPt.y ) || ( inPt.y > edgeHighPt.y ) ) 		continue;

    					if ( inPt.y === edgeLowPt.y ) {

    						if ( inPt.x === edgeLowPt.x )		return	true;		// inPt is on contour ?
    						// continue;				// no intersection or edgeLowPt => doesn't count !!!

    					} else {

    						var perpEdge = edgeDy * ( inPt.x - edgeLowPt.x ) - edgeDx * ( inPt.y - edgeLowPt.y );
    						if ( perpEdge === 0 )				return	true;		// inPt is on contour ?
    						if ( perpEdge < 0 ) 				continue;
    						inside = ! inside;		// true intersection left of inPt

    					}

    				} else {

    					// parallel or collinear
    					if ( inPt.y !== edgeLowPt.y ) 		continue;			// parallel
    					// edge lies on the same horizontal line as inPt
    					if ( ( ( edgeHighPt.x <= inPt.x ) && ( inPt.x <= edgeLowPt.x ) ) ||
    						 ( ( edgeLowPt.x <= inPt.x ) && ( inPt.x <= edgeHighPt.x ) ) )		return	true;	// inPt: Point on contour !
    					// continue;

    				}

    			}

    			return	inside;

    		}

    		var isClockWise = ShapeUtils.isClockWise;

    		var subPaths = this.subPaths;
    		if ( subPaths.length === 0 ) return [];

    		if ( noHoles === true )	return	toShapesNoHoles( subPaths );


    		var solid, tmpPath, tmpShape, shapes = [];

    		if ( subPaths.length === 1 ) {

    			tmpPath = subPaths[ 0 ];
    			tmpShape = new Shape();
    			tmpShape.curves = tmpPath.curves;
    			shapes.push( tmpShape );
    			return shapes;

    		}

    		var holesFirst = ! isClockWise( subPaths[ 0 ].getPoints() );
    		holesFirst = isCCW ? ! holesFirst : holesFirst;

    		// console.log("Holes first", holesFirst);

    		var betterShapeHoles = [];
    		var newShapes = [];
    		var newShapeHoles = [];
    		var mainIdx = 0;
    		var tmpPoints;

    		newShapes[ mainIdx ] = undefined;
    		newShapeHoles[ mainIdx ] = [];

    		for ( var i = 0, l = subPaths.length; i < l; i ++ ) {

    			tmpPath = subPaths[ i ];
    			tmpPoints = tmpPath.getPoints();
    			solid = isClockWise( tmpPoints );
    			solid = isCCW ? ! solid : solid;

    			if ( solid ) {

    				if ( ( ! holesFirst ) && ( newShapes[ mainIdx ] ) )	mainIdx ++;

    				newShapes[ mainIdx ] = { s: new Shape(), p: tmpPoints };
    				newShapes[ mainIdx ].s.curves = tmpPath.curves;

    				if ( holesFirst )	mainIdx ++;
    				newShapeHoles[ mainIdx ] = [];

    				//console.log('cw', i);

    			} else {

    				newShapeHoles[ mainIdx ].push( { h: tmpPath, p: tmpPoints[ 0 ] } );

    				//console.log('ccw', i);

    			}

    		}

    		// only Holes? -> probably all Shapes with wrong orientation
    		if ( ! newShapes[ 0 ] )	return	toShapesNoHoles( subPaths );


    		if ( newShapes.length > 1 ) {

    			var ambiguous = false;
    			var toChange = [];

    			for ( var sIdx = 0, sLen = newShapes.length; sIdx < sLen; sIdx ++ ) {

    				betterShapeHoles[ sIdx ] = [];

    			}

    			for ( var sIdx = 0, sLen = newShapes.length; sIdx < sLen; sIdx ++ ) {

    				var sho = newShapeHoles[ sIdx ];

    				for ( var hIdx = 0; hIdx < sho.length; hIdx ++ ) {

    					var ho = sho[ hIdx ];
    					var hole_unassigned = true;

    					for ( var s2Idx = 0; s2Idx < newShapes.length; s2Idx ++ ) {

    						if ( isPointInsidePolygon( ho.p, newShapes[ s2Idx ].p ) ) {

    							if ( sIdx !== s2Idx )	toChange.push( { froms: sIdx, tos: s2Idx, hole: hIdx } );
    							if ( hole_unassigned ) {

    								hole_unassigned = false;
    								betterShapeHoles[ s2Idx ].push( ho );

    							} else {

    								ambiguous = true;

    							}

    						}

    					}
    					if ( hole_unassigned ) {

    						betterShapeHoles[ sIdx ].push( ho );

    					}

    				}

    			}
    			// console.log("ambiguous: ", ambiguous);
    			if ( toChange.length > 0 ) {

    				// console.log("to change: ", toChange);
    				if ( ! ambiguous )	newShapeHoles = betterShapeHoles;

    			}

    		}

    		var tmpHoles;

    		for ( var i = 0, il = newShapes.length; i < il; i ++ ) {

    			tmpShape = newShapes[ i ].s;
    			shapes.push( tmpShape );
    			tmpHoles = newShapeHoles[ i ];

    			for ( var j = 0, jl = tmpHoles.length; j < jl; j ++ ) {

    				tmpShape.holes.push( tmpHoles[ j ].h );

    			}

    		}

    		//console.log("shape", shapes);

    		return shapes;

    	}

    } );

    /**
     * @author zz85 / http://www.lab4games.net/zz85/blog
     * @author mrdoob / http://mrdoob.com/
     */


    function Font( data ) {

    	this.type = 'Font';

    	this.data = data;

    }

    Object.assign( Font.prototype, {

    	isFont: true,

    	generateShapes: function ( text, size ) {

    		if ( size === undefined ) size = 100;

    		var shapes = [];
    		var paths = createPaths( text, size, this.data );

    		for ( var p = 0, pl = paths.length; p < pl; p ++ ) {

    			Array.prototype.push.apply( shapes, paths[ p ].toShapes() );

    		}

    		return shapes;

    	}

    } );

    function createPaths( text, size, data ) {

    	var chars = Array.from ? Array.from( text ) : String( text ).split( '' ); // see #13988
    	var scale = size / data.resolution;
    	var line_height = ( data.boundingBox.yMax - data.boundingBox.yMin + data.underlineThickness ) * scale;

    	var paths = [];

    	var offsetX = 0, offsetY = 0;

    	for ( var i = 0; i < chars.length; i ++ ) {

    		var char = chars[ i ];

    		if ( char === '\n' ) {

    			offsetX = 0;
    			offsetY -= line_height;

    		} else {

    			var ret = createPath( char, scale, offsetX, offsetY, data );
    			offsetX += ret.offsetX;
    			paths.push( ret.path );

    		}

    	}

    	return paths;

    }

    function createPath( char, scale, offsetX, offsetY, data ) {

    	var glyph = data.glyphs[ char ] || data.glyphs[ '?' ];

    	if ( ! glyph ) {

    		console.error( 'THREE.Font: character "' + char + '" does not exists in font family ' + data.familyName + '.' );

    		return;

    	}

    	var path = new ShapePath();

    	var x, y, cpx, cpy, cpx1, cpy1, cpx2, cpy2;

    	if ( glyph.o ) {

    		var outline = glyph._cachedOutline || ( glyph._cachedOutline = glyph.o.split( ' ' ) );

    		for ( var i = 0, l = outline.length; i < l; ) {

    			var action = outline[ i ++ ];

    			switch ( action ) {

    				case 'm': // moveTo

    					x = outline[ i ++ ] * scale + offsetX;
    					y = outline[ i ++ ] * scale + offsetY;

    					path.moveTo( x, y );

    					break;

    				case 'l': // lineTo

    					x = outline[ i ++ ] * scale + offsetX;
    					y = outline[ i ++ ] * scale + offsetY;

    					path.lineTo( x, y );

    					break;

    				case 'q': // quadraticCurveTo

    					cpx = outline[ i ++ ] * scale + offsetX;
    					cpy = outline[ i ++ ] * scale + offsetY;
    					cpx1 = outline[ i ++ ] * scale + offsetX;
    					cpy1 = outline[ i ++ ] * scale + offsetY;

    					path.quadraticCurveTo( cpx1, cpy1, cpx, cpy );

    					break;

    				case 'b': // bezierCurveTo

    					cpx = outline[ i ++ ] * scale + offsetX;
    					cpy = outline[ i ++ ] * scale + offsetY;
    					cpx1 = outline[ i ++ ] * scale + offsetX;
    					cpy1 = outline[ i ++ ] * scale + offsetY;
    					cpx2 = outline[ i ++ ] * scale + offsetX;
    					cpy2 = outline[ i ++ ] * scale + offsetY;

    					path.bezierCurveTo( cpx1, cpy1, cpx2, cpy2, cpx, cpy );

    					break;

    			}

    		}

    	}

    	return { offsetX: glyph.ha * scale, path: path };

    }

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function FontLoader( manager ) {

    	Loader.call( this, manager );

    }

    FontLoader.prototype = Object.assign( Object.create( Loader.prototype ), {

    	constructor: FontLoader,

    	load: function ( url, onLoad, onProgress, onError ) {

    		var scope = this;

    		var loader = new FileLoader( this.manager );
    		loader.setPath( this.path );
    		loader.load( url, function ( text ) {

    			var json;

    			try {

    				json = JSON.parse( text );

    			} catch ( e ) {

    				console.warn( 'THREE.FontLoader: typeface.js support is being deprecated. Use typeface.json instead.' );
    				json = JSON.parse( text.substring( 65, text.length - 2 ) );

    			}

    			var font = scope.parse( json );

    			if ( onLoad ) onLoad( font );

    		}, onProgress, onError );

    	},

    	parse: function ( json ) {

    		return new Font( json );

    	}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    var _context;

    var AudioContext = {

    	getContext: function () {

    		if ( _context === undefined ) {

    			_context = new ( window.AudioContext || window.webkitAudioContext )();

    		}

    		return _context;

    	},

    	setContext: function ( value ) {

    		_context = value;

    	}

    };

    /**
     * @author Reece Aaron Lecrivain / http://reecenotes.com/
     */

    function AudioLoader( manager ) {

    	Loader.call( this, manager );

    }

    AudioLoader.prototype = Object.assign( Object.create( Loader.prototype ), {

    	constructor: AudioLoader,

    	load: function ( url, onLoad, onProgress, onError ) {

    		var loader = new FileLoader( this.manager );
    		loader.setResponseType( 'arraybuffer' );
    		loader.setPath( this.path );
    		loader.load( url, function ( buffer ) {

    			// Create a copy of the buffer. The `decodeAudioData` method
    			// detaches the buffer when complete, preventing reuse.
    			var bufferCopy = buffer.slice( 0 );

    			var context = AudioContext.getContext();
    			context.decodeAudioData( bufferCopy, function ( audioBuffer ) {

    				onLoad( audioBuffer );

    			} );

    		}, onProgress, onError );

    	}

    } );

    /**
     * @author bhouston / http://clara.io
     * @author WestLangley / http://github.com/WestLangley
     *
     * Primary reference:
     *   https://graphics.stanford.edu/papers/envmap/envmap.pdf
     *
     * Secondary reference:
     *   https://www.ppsloan.org/publications/StupidSH36.pdf
     */

    // 3-band SH defined by 9 coefficients

    function SphericalHarmonics3() {

    	this.coefficients = [];

    	for ( var i = 0; i < 9; i ++ ) {

    		this.coefficients.push( new Vector3() );

    	}

    }

    Object.assign( SphericalHarmonics3.prototype, {

    	isSphericalHarmonics3: true,

    	set: function ( coefficients ) {

    		for ( var i = 0; i < 9; i ++ ) {

    			this.coefficients[ i ].copy( coefficients[ i ] );

    		}

    		return this;

    	},

    	zero: function () {

    		for ( var i = 0; i < 9; i ++ ) {

    			this.coefficients[ i ].set( 0, 0, 0 );

    		}

    		return this;

    	},

    	// get the radiance in the direction of the normal
    	// target is a Vector3
    	getAt: function ( normal, target ) {

    		// normal is assumed to be unit length

    		var x = normal.x, y = normal.y, z = normal.z;

    		var coeff = this.coefficients;

    		// band 0
    		target.copy( coeff[ 0 ] ).multiplyScalar( 0.282095 );

    		// band 1
    		target.addScale( coeff[ 1 ], 0.488603 * y );
    		target.addScale( coeff[ 2 ], 0.488603 * z );
    		target.addScale( coeff[ 3 ], 0.488603 * x );

    		// band 2
    		target.addScale( coeff[ 4 ], 1.092548 * ( x * y ) );
    		target.addScale( coeff[ 5 ], 1.092548 * ( y * z ) );
    		target.addScale( coeff[ 6 ], 0.315392 * ( 3.0 * z * z - 1.0 ) );
    		target.addScale( coeff[ 7 ], 1.092548 * ( x * z ) );
    		target.addScale( coeff[ 8 ], 0.546274 * ( x * x - y * y ) );

    		return target;

    	},

    	// get the irradiance (radiance convolved with cosine lobe) in the direction of the normal
    	// target is a Vector3
    	// https://graphics.stanford.edu/papers/envmap/envmap.pdf
    	getIrradianceAt: function ( normal, target ) {

    		// normal is assumed to be unit length

    		var x = normal.x, y = normal.y, z = normal.z;

    		var coeff = this.coefficients;

    		// band 0
    		target.copy( coeff[ 0 ] ).multiplyScalar( 0.886227 ); // π * 0.282095

    		// band 1
    		target.addScale( coeff[ 1 ], 2.0 * 0.511664 * y ); // ( 2 * π / 3 ) * 0.488603
    		target.addScale( coeff[ 2 ], 2.0 * 0.511664 * z );
    		target.addScale( coeff[ 3 ], 2.0 * 0.511664 * x );

    		// band 2
    		target.addScale( coeff[ 4 ], 2.0 * 0.429043 * x * y ); // ( π / 4 ) * 1.092548
    		target.addScale( coeff[ 5 ], 2.0 * 0.429043 * y * z );
    		target.addScale( coeff[ 6 ], 0.743125 * z * z - 0.247708 ); // ( π / 4 ) * 0.315392 * 3
    		target.addScale( coeff[ 7 ], 2.0 * 0.429043 * x * z );
    		target.addScale( coeff[ 8 ], 0.429043 * ( x * x - y * y ) ); // ( π / 4 ) * 0.546274

    		return target;

    	},

    	add: function ( sh ) {

    		for ( var i = 0; i < 9; i ++ ) {

    			this.coefficients[ i ].add( sh.coefficients[ i ] );

    		}

    		return this;

    	},


    	scale: function ( s ) {

    		for ( var i = 0; i < 9; i ++ ) {

    			this.coefficients[ i ].multiplyScalar( s );

    		}

    		return this;

    	},

    	lerp: function ( sh, alpha ) {

    		for ( var i = 0; i < 9; i ++ ) {

    			this.coefficients[ i ].lerp( sh.coefficients[ i ], alpha );

    		}

    		return this;

    	},

    	equals: function ( sh ) {

    		for ( var i = 0; i < 9; i ++ ) {

    			if ( ! this.coefficients[ i ].equals( sh.coefficients[ i ] ) ) {

    				return false;

    			}

    		}

    		return true;

    	},

    	copy: function ( sh ) {

    		return this.set( sh.coefficients );

    	},

    	clone: function () {

    		return new this.constructor().copy( this );

    	},

    	fromArray: function ( array, offset ) {

    		if ( offset === undefined ) offset = 0;

    		var coefficients = this.coefficients;

    		for ( var i = 0; i < 9; i ++ ) {

    			coefficients[ i ].fromArray( array, offset + ( i * 3 ) );

    		}

    		return this;

    	},

    	toArray: function ( array, offset ) {

    		if ( array === undefined ) array = [];
    		if ( offset === undefined ) offset = 0;

    		var coefficients = this.coefficients;

    		for ( var i = 0; i < 9; i ++ ) {

    			coefficients[ i ].toArray( array, offset + ( i * 3 ) );

    		}

    		return array;

    	}

    } );

    Object.assign( SphericalHarmonics3, {

    	// evaluate the basis functions
    	// shBasis is an Array[ 9 ]
    	getBasisAt: function ( normal, shBasis ) {

    		// normal is assumed to be unit length

    		var x = normal.x, y = normal.y, z = normal.z;

    		// band 0
    		shBasis[ 0 ] = 0.282095;

    		// band 1
    		shBasis[ 1 ] = 0.488603 * y;
    		shBasis[ 2 ] = 0.488603 * z;
    		shBasis[ 3 ] = 0.488603 * x;

    		// band 2
    		shBasis[ 4 ] = 1.092548 * x * y;
    		shBasis[ 5 ] = 1.092548 * y * z;
    		shBasis[ 6 ] = 0.315392 * ( 3 * z * z - 1 );
    		shBasis[ 7 ] = 1.092548 * x * z;
    		shBasis[ 8 ] = 0.546274 * ( x * x - y * y );

    	}

    } );

    /**
     * @author WestLangley / http://github.com/WestLangley
     *
     * A LightProbe is a source of indirect-diffuse light
     */

    function LightProbe( sh, intensity ) {

    	Light.call( this, undefined, intensity );

    	this.sh = ( sh !== undefined ) ? sh : new SphericalHarmonics3();

    }

    LightProbe.prototype = Object.assign( Object.create( Light.prototype ), {

    	constructor: LightProbe,

    	isLightProbe: true,

    	copy: function ( source ) {

    		Light.prototype.copy.call( this, source );

    		this.sh.copy( source.sh );
    		this.intensity = source.intensity;

    		return this;

    	},

    	toJSON: function ( meta ) {

    		var data = Light.prototype.toJSON.call( this, meta );

    		// data.sh = this.sh.toArray(); // todo

    		return data;

    	}

    } );

    /**
     * @author WestLangley / http://github.com/WestLangley
     */

    function HemisphereLightProbe( skyColor, groundColor, intensity ) {

    	LightProbe.call( this, undefined, intensity );

    	var color1 = new Color$7().set( skyColor );
    	var color2 = new Color$7().set( groundColor );

    	var sky = new Vector3( color1.r, color1.g, color1.b );
    	var ground = new Vector3( color2.r, color2.g, color2.b );

    	// without extra factor of PI in the shader, should = 1 / Math.sqrt( Math.PI );
    	var c0 = Math.sqrt( Math.PI );
    	var c1 = c0 * Math.sqrt( 0.75 );

    	this.sh.coefficients[ 0 ].copy( sky ).add( ground ).multiplyScalar( c0 );
    	this.sh.coefficients[ 1 ].copy( sky ).sub( ground ).multiplyScalar( c1 );

    }

    HemisphereLightProbe.prototype = Object.assign( Object.create( LightProbe.prototype ), {

    	constructor: HemisphereLightProbe,

    	isHemisphereLightProbe: true,

    	copy: function ( source ) { // modifying colors not currently supported

    		LightProbe.prototype.copy.call( this, source );

    		return this;

    	},

    	toJSON: function ( meta ) {

    		var data = LightProbe.prototype.toJSON.call( this, meta );

    		// data.sh = this.sh.toArray(); // todo

    		return data;

    	}

    } );

    /**
     * @author WestLangley / http://github.com/WestLangley
     */

    function AmbientLightProbe( color, intensity ) {

    	LightProbe.call( this, undefined, intensity );

    	var color1 = new Color$7().set( color );

    	// without extra factor of PI in the shader, would be 2 / Math.sqrt( Math.PI );
    	this.sh.coefficients[ 0 ].set( color1.r, color1.g, color1.b ).multiplyScalar( 2 * Math.sqrt( Math.PI ) );

    }

    AmbientLightProbe.prototype = Object.assign( Object.create( LightProbe.prototype ), {

    	constructor: AmbientLightProbe,

    	isAmbientLightProbe: true,

    	copy: function ( source ) { // modifying color not currently supported

    		LightProbe.prototype.copy.call( this, source );

    		return this;

    	},

    	toJSON: function ( meta ) {

    		var data = LightProbe.prototype.toJSON.call( this, meta );

    		// data.sh = this.sh.toArray(); // todo

    		return data;

    	}

    } );

    var _eyeRight = new Matrix4();
    var _eyeLeft = new Matrix4();

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function StereoCamera() {

    	this.type = 'StereoCamera';

    	this.aspect = 1;

    	this.eyeSep = 0.064;

    	this.cameraL = new PerspectiveCamera();
    	this.cameraL.layers.enable( 1 );
    	this.cameraL.matrixAutoUpdate = false;

    	this.cameraR = new PerspectiveCamera();
    	this.cameraR.layers.enable( 2 );
    	this.cameraR.matrixAutoUpdate = false;

    	this._cache = {
    		focus: null,
    		fov: null,
    		aspect: null,
    		near: null,
    		far: null,
    		zoom: null,
    		eyeSep: null
    	};

    }

    Object.assign( StereoCamera.prototype, {

    	update: function ( camera ) {

    		var cache = this._cache;

    		var needsUpdate = cache.focus !== camera.focus || cache.fov !== camera.fov ||
    			cache.aspect !== camera.aspect * this.aspect || cache.near !== camera.near ||
    			cache.far !== camera.far || cache.zoom !== camera.zoom || cache.eyeSep !== this.eyeSep;

    		if ( needsUpdate ) {

    			cache.focus = camera.focus;
    			cache.fov = camera.fov;
    			cache.aspect = camera.aspect * this.aspect;
    			cache.near = camera.near;
    			cache.far = camera.far;
    			cache.zoom = camera.zoom;
    			cache.eyeSep = this.eyeSep;

    			// Off-axis stereoscopic effect based on
    			// http://paulbourke.net/stereographics/stereorender/

    			var projectionMatrix = camera.projectionMatrix.clone();
    			var eyeSepHalf = cache.eyeSep / 2;
    			var eyeSepOnProjection = eyeSepHalf * cache.near / cache.focus;
    			var ymax = ( cache.near * Math.tan( _Math.DEG2RAD * cache.fov * 0.5 ) ) / cache.zoom;
    			var xmin, xmax;

    			// translate xOffset

    			_eyeLeft.elements[ 12 ] = - eyeSepHalf;
    			_eyeRight.elements[ 12 ] = eyeSepHalf;

    			// for left eye

    			xmin = - ymax * cache.aspect + eyeSepOnProjection;
    			xmax = ymax * cache.aspect + eyeSepOnProjection;

    			projectionMatrix.elements[ 0 ] = 2 * cache.near / ( xmax - xmin );
    			projectionMatrix.elements[ 8 ] = ( xmax + xmin ) / ( xmax - xmin );

    			this.cameraL.projectionMatrix.copy( projectionMatrix );

    			// for right eye

    			xmin = - ymax * cache.aspect - eyeSepOnProjection;
    			xmax = ymax * cache.aspect - eyeSepOnProjection;

    			projectionMatrix.elements[ 0 ] = 2 * cache.near / ( xmax - xmin );
    			projectionMatrix.elements[ 8 ] = ( xmax + xmin ) / ( xmax - xmin );

    			this.cameraR.projectionMatrix.copy( projectionMatrix );

    		}

    		this.cameraL.matrixWorld.copy( camera.matrixWorld ).multiply( _eyeLeft );
    		this.cameraR.matrixWorld.copy( camera.matrixWorld ).multiply( _eyeRight );

    	}

    } );

    /**
     * @author alteredq / http://alteredqualia.com/
     */

    function Clock( autoStart ) {

    	this.autoStart = ( autoStart !== undefined ) ? autoStart : true;

    	this.startTime = 0;
    	this.oldTime = 0;
    	this.elapsedTime = 0;

    	this.running = false;

    }

    Object.assign( Clock.prototype, {

    	start: function () {

    		this.startTime = ( typeof performance === 'undefined' ? Date : performance ).now(); // see #10732

    		this.oldTime = this.startTime;
    		this.elapsedTime = 0;
    		this.running = true;

    	},

    	stop: function () {

    		this.getElapsedTime();
    		this.running = false;
    		this.autoStart = false;

    	},

    	getElapsedTime: function () {

    		this.getDelta();
    		return this.elapsedTime;

    	},

    	getDelta: function () {

    		var diff = 0;

    		if ( this.autoStart && ! this.running ) {

    			this.start();
    			return 0;

    		}

    		if ( this.running ) {

    			var newTime = ( typeof performance === 'undefined' ? Date : performance ).now();

    			diff = ( newTime - this.oldTime ) / 1000;
    			this.oldTime = newTime;

    			this.elapsedTime += diff;

    		}

    		return diff;

    	}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    var _position$2 = new Vector3();
    var _quaternion$3 = new Quaternion();
    var _scale$1 = new Vector3();
    var _orientation = new Vector3();

    function AudioListener() {

    	Object3D.call( this );

    	this.type = 'AudioListener';

    	this.context = AudioContext.getContext();

    	this.gain = this.context.createGain();
    	this.gain.connect( this.context.destination );

    	this.filter = null;

    	this.timeDelta = 0;

    	// private

    	this._clock = new Clock();

    }

    AudioListener.prototype = Object.assign( Object.create( Object3D.prototype ), {

    	constructor: AudioListener,

    	getInput: function () {

    		return this.gain;

    	},

    	removeFilter: function ( ) {

    		if ( this.filter !== null ) {

    			this.gain.disconnect( this.filter );
    			this.filter.disconnect( this.context.destination );
    			this.gain.connect( this.context.destination );
    			this.filter = null;

    		}

    		return this;

    	},

    	getFilter: function () {

    		return this.filter;

    	},

    	setFilter: function ( value ) {

    		if ( this.filter !== null ) {

    			this.gain.disconnect( this.filter );
    			this.filter.disconnect( this.context.destination );

    		} else {

    			this.gain.disconnect( this.context.destination );

    		}

    		this.filter = value;
    		this.gain.connect( this.filter );
    		this.filter.connect( this.context.destination );

    		return this;

    	},

    	getMasterVolume: function () {

    		return this.gain.gain.value;

    	},

    	setMasterVolume: function ( value ) {

    		this.gain.gain.setTargetAtTime( value, this.context.currentTime, 0.01 );

    		return this;

    	},

    	updateMatrixWorld: function ( force ) {

    		Object3D.prototype.updateMatrixWorld.call( this, force );

    		var listener = this.context.listener;
    		var up = this.up;

    		this.timeDelta = this._clock.getDelta();

    		this.matrixWorld.decompose( _position$2, _quaternion$3, _scale$1 );

    		_orientation.set( 0, 0, - 1 ).applyQuaternion( _quaternion$3 );

    		if ( listener.positionX ) {

    			// code path for Chrome (see #14393)

    			var endTime = this.context.currentTime + this.timeDelta;

    			listener.positionX.linearRampToValueAtTime( _position$2.x, endTime );
    			listener.positionY.linearRampToValueAtTime( _position$2.y, endTime );
    			listener.positionZ.linearRampToValueAtTime( _position$2.z, endTime );
    			listener.forwardX.linearRampToValueAtTime( _orientation.x, endTime );
    			listener.forwardY.linearRampToValueAtTime( _orientation.y, endTime );
    			listener.forwardZ.linearRampToValueAtTime( _orientation.z, endTime );
    			listener.upX.linearRampToValueAtTime( up.x, endTime );
    			listener.upY.linearRampToValueAtTime( up.y, endTime );
    			listener.upZ.linearRampToValueAtTime( up.z, endTime );

    		} else {

    			listener.setPosition( _position$2.x, _position$2.y, _position$2.z );
    			listener.setOrientation( _orientation.x, _orientation.y, _orientation.z, up.x, up.y, up.z );

    		}

    	}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     * @author Reece Aaron Lecrivain / http://reecenotes.com/
     */

    function Audio( listener ) {

    	Object3D.call( this );

    	this.type = 'Audio';

    	this.listener = listener;
    	this.context = listener.context;

    	this.gain = this.context.createGain();
    	this.gain.connect( listener.getInput() );

    	this.autoplay = false;

    	this.buffer = null;
    	this.detune = 0;
    	this.loop = false;
    	this.startTime = 0;
    	this.offset = 0;
    	this.duration = undefined;
    	this.playbackRate = 1;
    	this.isPlaying = false;
    	this.hasPlaybackControl = true;
    	this.sourceType = 'empty';

    	this.filters = [];

    }

    Audio.prototype = Object.assign( Object.create( Object3D.prototype ), {

    	constructor: Audio,

    	getOutput: function () {

    		return this.gain;

    	},

    	setNodeSource: function ( audioNode ) {

    		this.hasPlaybackControl = false;
    		this.sourceType = 'audioNode';
    		this.source = audioNode;
    		this.connect();

    		return this;

    	},

    	setMediaElementSource: function ( mediaElement ) {

    		this.hasPlaybackControl = false;
    		this.sourceType = 'mediaNode';
    		this.source = this.context.createMediaElementSource( mediaElement );
    		this.connect();

    		return this;

    	},

    	setBuffer: function ( audioBuffer ) {

    		this.buffer = audioBuffer;
    		this.sourceType = 'buffer';

    		if ( this.autoplay ) this.play();

    		return this;

    	},

    	play: function () {

    		if ( this.isPlaying === true ) {

    			console.warn( 'THREE.Audio: Audio is already playing.' );
    			return;

    		}

    		if ( this.hasPlaybackControl === false ) {

    			console.warn( 'THREE.Audio: this Audio has no playback control.' );
    			return;

    		}

    		var source = this.context.createBufferSource();

    		source.buffer = this.buffer;
    		source.loop = this.loop;
    		source.onended = this.onEnded.bind( this );
    		this.startTime = this.context.currentTime;
    		source.start( this.startTime, this.offset, this.duration );

    		this.isPlaying = true;

    		this.source = source;

    		this.setDetune( this.detune );
    		this.setPlaybackRate( this.playbackRate );

    		return this.connect();

    	},

    	pause: function () {

    		if ( this.hasPlaybackControl === false ) {

    			console.warn( 'THREE.Audio: this Audio has no playback control.' );
    			return;

    		}

    		if ( this.isPlaying === true ) {

    			this.source.stop();
    			this.source.onended = null;
    			this.offset += ( this.context.currentTime - this.startTime ) * this.playbackRate;
    			this.isPlaying = false;

    		}

    		return this;

    	},

    	stop: function () {

    		if ( this.hasPlaybackControl === false ) {

    			console.warn( 'THREE.Audio: this Audio has no playback control.' );
    			return;

    		}

    		this.source.stop();
    		this.source.onended = null;
    		this.offset = 0;
    		this.isPlaying = false;

    		return this;

    	},

    	connect: function () {

    		if ( this.filters.length > 0 ) {

    			this.source.connect( this.filters[ 0 ] );

    			for ( var i = 1, l = this.filters.length; i < l; i ++ ) {

    				this.filters[ i - 1 ].connect( this.filters[ i ] );

    			}

    			this.filters[ this.filters.length - 1 ].connect( this.getOutput() );

    		} else {

    			this.source.connect( this.getOutput() );

    		}

    		return this;

    	},

    	disconnect: function () {

    		if ( this.filters.length > 0 ) {

    			this.source.disconnect( this.filters[ 0 ] );

    			for ( var i = 1, l = this.filters.length; i < l; i ++ ) {

    				this.filters[ i - 1 ].disconnect( this.filters[ i ] );

    			}

    			this.filters[ this.filters.length - 1 ].disconnect( this.getOutput() );

    		} else {

    			this.source.disconnect( this.getOutput() );

    		}

    		return this;

    	},

    	getFilters: function () {

    		return this.filters;

    	},

    	setFilters: function ( value ) {

    		if ( ! value ) value = [];

    		if ( this.isPlaying === true ) {

    			this.disconnect();
    			this.filters = value;
    			this.connect();

    		} else {

    			this.filters = value;

    		}

    		return this;

    	},

    	setDetune: function ( value ) {

    		this.detune = value;

    		if ( this.source.detune === undefined ) return; // only set detune when available

    		if ( this.isPlaying === true ) {

    			this.source.detune.setTargetAtTime( this.detune, this.context.currentTime, 0.01 );

    		}

    		return this;

    	},

    	getDetune: function () {

    		return this.detune;

    	},

    	getFilter: function () {

    		return this.getFilters()[ 0 ];

    	},

    	setFilter: function ( filter ) {

    		return this.setFilters( filter ? [ filter ] : [] );

    	},

    	setPlaybackRate: function ( value ) {

    		if ( this.hasPlaybackControl === false ) {

    			console.warn( 'THREE.Audio: this Audio has no playback control.' );
    			return;

    		}

    		this.playbackRate = value;

    		if ( this.isPlaying === true ) {

    			this.source.playbackRate.setTargetAtTime( this.playbackRate, this.context.currentTime, 0.01 );

    		}

    		return this;

    	},

    	getPlaybackRate: function () {

    		return this.playbackRate;

    	},

    	onEnded: function () {

    		this.isPlaying = false;

    	},

    	getLoop: function () {

    		if ( this.hasPlaybackControl === false ) {

    			console.warn( 'THREE.Audio: this Audio has no playback control.' );
    			return false;

    		}

    		return this.loop;

    	},

    	setLoop: function ( value ) {

    		if ( this.hasPlaybackControl === false ) {

    			console.warn( 'THREE.Audio: this Audio has no playback control.' );
    			return;

    		}

    		this.loop = value;

    		if ( this.isPlaying === true ) {

    			this.source.loop = this.loop;

    		}

    		return this;

    	},

    	getVolume: function () {

    		return this.gain.gain.value;

    	},

    	setVolume: function ( value ) {

    		this.gain.gain.setTargetAtTime( value, this.context.currentTime, 0.01 );

    		return this;

    	}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    var _position$3 = new Vector3();
    var _quaternion$4 = new Quaternion();
    var _scale$2 = new Vector3();
    var _orientation$1 = new Vector3();

    function PositionalAudio( listener ) {

    	Audio.call( this, listener );

    	this.panner = this.context.createPanner();
    	this.panner.panningModel = 'HRTF';
    	this.panner.connect( this.gain );

    }

    PositionalAudio.prototype = Object.assign( Object.create( Audio.prototype ), {

    	constructor: PositionalAudio,

    	getOutput: function () {

    		return this.panner;

    	},

    	getRefDistance: function () {

    		return this.panner.refDistance;

    	},

    	setRefDistance: function ( value ) {

    		this.panner.refDistance = value;

    		return this;

    	},

    	getRolloffFactor: function () {

    		return this.panner.rolloffFactor;

    	},

    	setRolloffFactor: function ( value ) {

    		this.panner.rolloffFactor = value;

    		return this;

    	},

    	getDistanceModel: function () {

    		return this.panner.distanceModel;

    	},

    	setDistanceModel: function ( value ) {

    		this.panner.distanceModel = value;

    		return this;

    	},

    	getMaxDistance: function () {

    		return this.panner.maxDistance;

    	},

    	setMaxDistance: function ( value ) {

    		this.panner.maxDistance = value;

    		return this;

    	},

    	setDirectionalCone: function ( coneInnerAngle, coneOuterAngle, coneOuterGain ) {

    		this.panner.coneInnerAngle = coneInnerAngle;
    		this.panner.coneOuterAngle = coneOuterAngle;
    		this.panner.coneOuterGain = coneOuterGain;

    		return this;

    	},

    	updateMatrixWorld: function ( force ) {

    		Object3D.prototype.updateMatrixWorld.call( this, force );

    		if ( this.hasPlaybackControl === true && this.isPlaying === false ) return;

    		this.matrixWorld.decompose( _position$3, _quaternion$4, _scale$2 );

    		_orientation$1.set( 0, 0, 1 ).applyQuaternion( _quaternion$4 );

    		var panner = this.panner;

    		if ( panner.positionX ) {

    			// code path for Chrome and Firefox (see #14393)

    			var endTime = this.context.currentTime + this.listener.timeDelta;

    			panner.positionX.linearRampToValueAtTime( _position$3.x, endTime );
    			panner.positionY.linearRampToValueAtTime( _position$3.y, endTime );
    			panner.positionZ.linearRampToValueAtTime( _position$3.z, endTime );
    			panner.orientationX.linearRampToValueAtTime( _orientation$1.x, endTime );
    			panner.orientationY.linearRampToValueAtTime( _orientation$1.y, endTime );
    			panner.orientationZ.linearRampToValueAtTime( _orientation$1.z, endTime );

    		} else {

    			panner.setPosition( _position$3.x, _position$3.y, _position$3.z );
    			panner.setOrientation( _orientation$1.x, _orientation$1.y, _orientation$1.z );

    		}

    	}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function AudioAnalyser( audio, fftSize ) {

    	this.analyser = audio.context.createAnalyser();
    	this.analyser.fftSize = fftSize !== undefined ? fftSize : 2048;

    	this.data = new Uint8Array( this.analyser.frequencyBinCount );

    	audio.getOutput().connect( this.analyser );

    }

    Object.assign( AudioAnalyser.prototype, {

    	getFrequencyData: function () {

    		this.analyser.getByteFrequencyData( this.data );

    		return this.data;

    	},

    	getAverageFrequency: function () {

    		var value = 0, data = this.getFrequencyData();

    		for ( var i = 0; i < data.length; i ++ ) {

    			value += data[ i ];

    		}

    		return value / data.length;

    	}

    } );

    /**
     *
     * Buffered scene graph property that allows weighted accumulation.
     *
     *
     * @author Ben Houston / http://clara.io/
     * @author David Sarno / http://lighthaus.us/
     * @author tschw
     */

    function PropertyMixer( binding, typeName, valueSize ) {

    	this.binding = binding;
    	this.valueSize = valueSize;

    	var bufferType = Float64Array,
    		mixFunction;

    	switch ( typeName ) {

    		case 'quaternion':
    			mixFunction = this._slerp;
    			break;

    		case 'string':
    		case 'bool':
    			bufferType = Array;
    			mixFunction = this._select;
    			break;

    		default:
    			mixFunction = this._lerp;

    	}

    	this.buffer = new bufferType( valueSize * 4 );
    	// layout: [ incoming | accu0 | accu1 | orig ]
    	//
    	// interpolators can use .buffer as their .result
    	// the data then goes to 'incoming'
    	//
    	// 'accu0' and 'accu1' are used frame-interleaved for
    	// the cumulative result and are compared to detect
    	// changes
    	//
    	// 'orig' stores the original state of the property

    	this._mixBufferRegion = mixFunction;

    	this.cumulativeWeight = 0;

    	this.useCount = 0;
    	this.referenceCount = 0;

    }

    Object.assign( PropertyMixer.prototype, {

    	// accumulate data in the 'incoming' region into 'accu<i>'
    	accumulate: function ( accuIndex, weight ) {

    		// note: happily accumulating nothing when weight = 0, the caller knows
    		// the weight and shouldn't have made the call in the first place

    		var buffer = this.buffer,
    			stride = this.valueSize,
    			offset = accuIndex * stride + stride,

    			currentWeight = this.cumulativeWeight;

    		if ( currentWeight === 0 ) {

    			// accuN := incoming * weight

    			for ( var i = 0; i !== stride; ++ i ) {

    				buffer[ offset + i ] = buffer[ i ];

    			}

    			currentWeight = weight;

    		} else {

    			// accuN := accuN + incoming * weight

    			currentWeight += weight;
    			var mix = weight / currentWeight;
    			this._mixBufferRegion( buffer, offset, 0, mix, stride );

    		}

    		this.cumulativeWeight = currentWeight;

    	},

    	// apply the state of 'accu<i>' to the binding when accus differ
    	apply: function ( accuIndex ) {

    		var stride = this.valueSize,
    			buffer = this.buffer,
    			offset = accuIndex * stride + stride,

    			weight = this.cumulativeWeight,

    			binding = this.binding;

    		this.cumulativeWeight = 0;

    		if ( weight < 1 ) {

    			// accuN := accuN + original * ( 1 - cumulativeWeight )

    			var originalValueOffset = stride * 3;

    			this._mixBufferRegion(
    				buffer, offset, originalValueOffset, 1 - weight, stride );

    		}

    		for ( var i = stride, e = stride + stride; i !== e; ++ i ) {

    			if ( buffer[ i ] !== buffer[ i + stride ] ) {

    				// value has changed -> update scene graph

    				binding.setValue( buffer, offset );
    				break;

    			}

    		}

    	},

    	// remember the state of the bound property and copy it to both accus
    	saveOriginalState: function () {

    		var binding = this.binding;

    		var buffer = this.buffer,
    			stride = this.valueSize,

    			originalValueOffset = stride * 3;

    		binding.getValue( buffer, originalValueOffset );

    		// accu[0..1] := orig -- initially detect changes against the original
    		for ( var i = stride, e = originalValueOffset; i !== e; ++ i ) {

    			buffer[ i ] = buffer[ originalValueOffset + ( i % stride ) ];

    		}

    		this.cumulativeWeight = 0;

    	},

    	// apply the state previously taken via 'saveOriginalState' to the binding
    	restoreOriginalState: function () {

    		var originalValueOffset = this.valueSize * 3;
    		this.binding.setValue( this.buffer, originalValueOffset );

    	},


    	// mix functions

    	_select: function ( buffer, dstOffset, srcOffset, t, stride ) {

    		if ( t >= 0.5 ) {

    			for ( var i = 0; i !== stride; ++ i ) {

    				buffer[ dstOffset + i ] = buffer[ srcOffset + i ];

    			}

    		}

    	},

    	_slerp: function ( buffer, dstOffset, srcOffset, t ) {

    		Quaternion.slerpFlat( buffer, dstOffset, buffer, dstOffset, buffer, srcOffset, t );

    	},

    	_lerp: function ( buffer, dstOffset, srcOffset, t, stride ) {

    		var s = 1 - t;

    		for ( var i = 0; i !== stride; ++ i ) {

    			var j = dstOffset + i;

    			buffer[ j ] = buffer[ j ] * s + buffer[ srcOffset + i ] * t;

    		}

    	}

    } );

    /**
     *
     * A reference to a real property in the scene graph.
     *
     *
     * @author Ben Houston / http://clara.io/
     * @author David Sarno / http://lighthaus.us/
     * @author tschw
     */

    // Characters [].:/ are reserved for track binding syntax.
    var _RESERVED_CHARS_RE = '\\[\\]\\.:\\/';
    var _reservedRe = new RegExp( '[' + _RESERVED_CHARS_RE + ']', 'g' );

    // Attempts to allow node names from any language. ES5's `\w` regexp matches
    // only latin characters, and the unicode \p{L} is not yet supported. So
    // instead, we exclude reserved characters and match everything else.
    var _wordChar = '[^' + _RESERVED_CHARS_RE + ']';
    var _wordCharOrDot = '[^' + _RESERVED_CHARS_RE.replace( '\\.', '' ) + ']';

    // Parent directories, delimited by '/' or ':'. Currently unused, but must
    // be matched to parse the rest of the track name.
    var _directoryRe = /((?:WC+[\/:])*)/.source.replace( 'WC', _wordChar );

    // Target node. May contain word characters (a-zA-Z0-9_) and '.' or '-'.
    var _nodeRe = /(WCOD+)?/.source.replace( 'WCOD', _wordCharOrDot );

    // Object on target node, and accessor. May not contain reserved
    // characters. Accessor may contain any character except closing bracket.
    var _objectRe = /(?:\.(WC+)(?:\[(.+)\])?)?/.source.replace( 'WC', _wordChar );

    // Property and accessor. May not contain reserved characters. Accessor may
    // contain any non-bracket characters.
    var _propertyRe = /\.(WC+)(?:\[(.+)\])?/.source.replace( 'WC', _wordChar );

    var _trackRe = new RegExp( ''
    	+ '^'
    	+ _directoryRe
    	+ _nodeRe
    	+ _objectRe
    	+ _propertyRe
    	+ '$'
    );

    var _supportedObjectNames = [ 'material', 'materials', 'bones' ];

    function Composite( targetGroup, path, optionalParsedPath ) {

    	var parsedPath = optionalParsedPath || PropertyBinding.parseTrackName( path );

    	this._targetGroup = targetGroup;
    	this._bindings = targetGroup.subscribe_( path, parsedPath );

    }

    Object.assign( Composite.prototype, {

    	getValue: function ( array, offset ) {

    		this.bind(); // bind all binding

    		var firstValidIndex = this._targetGroup.nCachedObjects_,
    			binding = this._bindings[ firstValidIndex ];

    		// and only call .getValue on the first
    		if ( binding !== undefined ) binding.getValue( array, offset );

    	},

    	setValue: function ( array, offset ) {

    		var bindings = this._bindings;

    		for ( var i = this._targetGroup.nCachedObjects_, n = bindings.length; i !== n; ++ i ) {

    			bindings[ i ].setValue( array, offset );

    		}

    	},

    	bind: function () {

    		var bindings = this._bindings;

    		for ( var i = this._targetGroup.nCachedObjects_, n = bindings.length; i !== n; ++ i ) {

    			bindings[ i ].bind();

    		}

    	},

    	unbind: function () {

    		var bindings = this._bindings;

    		for ( var i = this._targetGroup.nCachedObjects_, n = bindings.length; i !== n; ++ i ) {

    			bindings[ i ].unbind();

    		}

    	}

    } );


    function PropertyBinding( rootNode, path, parsedPath ) {

    	this.path = path;
    	this.parsedPath = parsedPath || PropertyBinding.parseTrackName( path );

    	this.node = PropertyBinding.findNode( rootNode, this.parsedPath.nodeName ) || rootNode;

    	this.rootNode = rootNode;

    }

    Object.assign( PropertyBinding, {

    	Composite: Composite,

    	create: function ( root, path, parsedPath ) {

    		if ( ! ( root && root.isAnimationObjectGroup ) ) {

    			return new PropertyBinding( root, path, parsedPath );

    		} else {

    			return new PropertyBinding.Composite( root, path, parsedPath );

    		}

    	},

    	/**
    	 * Replaces spaces with underscores and removes unsupported characters from
    	 * node names, to ensure compatibility with parseTrackName().
    	 *
    	 * @param {string} name Node name to be sanitized.
    	 * @return {string}
    	 */
    	sanitizeNodeName: function ( name ) {

    		return name.replace( /\s/g, '_' ).replace( _reservedRe, '' );

    	},

    	parseTrackName: function ( trackName ) {

    		var matches = _trackRe.exec( trackName );

    		if ( ! matches ) {

    			throw new Error( 'PropertyBinding: Cannot parse trackName: ' + trackName );

    		}

    		var results = {
    			// directoryName: matches[ 1 ], // (tschw) currently unused
    			nodeName: matches[ 2 ],
    			objectName: matches[ 3 ],
    			objectIndex: matches[ 4 ],
    			propertyName: matches[ 5 ], // required
    			propertyIndex: matches[ 6 ]
    		};

    		var lastDot = results.nodeName && results.nodeName.lastIndexOf( '.' );

    		if ( lastDot !== undefined && lastDot !== - 1 ) {

    			var objectName = results.nodeName.substring( lastDot + 1 );

    			// Object names must be checked against a whitelist. Otherwise, there
    			// is no way to parse 'foo.bar.baz': 'baz' must be a property, but
    			// 'bar' could be the objectName, or part of a nodeName (which can
    			// include '.' characters).
    			if ( _supportedObjectNames.indexOf( objectName ) !== - 1 ) {

    				results.nodeName = results.nodeName.substring( 0, lastDot );
    				results.objectName = objectName;

    			}

    		}

    		if ( results.propertyName === null || results.propertyName.length === 0 ) {

    			throw new Error( 'PropertyBinding: can not parse propertyName from trackName: ' + trackName );

    		}

    		return results;

    	},

    	findNode: function ( root, nodeName ) {

    		if ( ! nodeName || nodeName === "" || nodeName === "root" || nodeName === "." || nodeName === - 1 || nodeName === root.name || nodeName === root.uuid ) {

    			return root;

    		}

    		// search into skeleton bones.
    		if ( root.skeleton ) {

    			var bone = root.skeleton.getBoneByName( nodeName );

    			if ( bone !== undefined ) {

    				return bone;

    			}

    		}

    		// search into node subtree.
    		if ( root.children ) {

    			var searchNodeSubtree = function ( children ) {

    				for ( var i = 0; i < children.length; i ++ ) {

    					var childNode = children[ i ];

    					if ( childNode.name === nodeName || childNode.uuid === nodeName ) {

    						return childNode;

    					}

    					var result = searchNodeSubtree( childNode.children );

    					if ( result ) return result;

    				}

    				return null;

    			};

    			var subTreeNode = searchNodeSubtree( root.children );

    			if ( subTreeNode ) {

    				return subTreeNode;

    			}

    		}

    		return null;

    	}

    } );

    Object.assign( PropertyBinding.prototype, { // prototype, continued

    	// these are used to "bind" a nonexistent property
    	_getValue_unavailable: function () {},
    	_setValue_unavailable: function () {},

    	BindingType: {
    		Direct: 0,
    		EntireArray: 1,
    		ArrayElement: 2,
    		HasFromToArray: 3
    	},

    	Versioning: {
    		None: 0,
    		NeedsUpdate: 1,
    		MatrixWorldNeedsUpdate: 2
    	},

    	GetterByBindingType: [

    		function getValue_direct( buffer, offset ) {

    			buffer[ offset ] = this.node[ this.propertyName ];

    		},

    		function getValue_array( buffer, offset ) {

    			var source = this.resolvedProperty;

    			for ( var i = 0, n = source.length; i !== n; ++ i ) {

    				buffer[ offset ++ ] = source[ i ];

    			}

    		},

    		function getValue_arrayElement( buffer, offset ) {

    			buffer[ offset ] = this.resolvedProperty[ this.propertyIndex ];

    		},

    		function getValue_toArray( buffer, offset ) {

    			this.resolvedProperty.toArray( buffer, offset );

    		}

    	],

    	SetterByBindingTypeAndVersioning: [

    		[
    			// Direct

    			function setValue_direct( buffer, offset ) {

    				this.targetObject[ this.propertyName ] = buffer[ offset ];

    			},

    			function setValue_direct_setNeedsUpdate( buffer, offset ) {

    				this.targetObject[ this.propertyName ] = buffer[ offset ];
    				this.targetObject.needsUpdate = true;

    			},

    			function setValue_direct_setMatrixWorldNeedsUpdate( buffer, offset ) {

    				this.targetObject[ this.propertyName ] = buffer[ offset ];
    				this.targetObject.matrixWorldNeedsUpdate = true;

    			}

    		], [

    			// EntireArray

    			function setValue_array( buffer, offset ) {

    				var dest = this.resolvedProperty;

    				for ( var i = 0, n = dest.length; i !== n; ++ i ) {

    					dest[ i ] = buffer[ offset ++ ];

    				}

    			},

    			function setValue_array_setNeedsUpdate( buffer, offset ) {

    				var dest = this.resolvedProperty;

    				for ( var i = 0, n = dest.length; i !== n; ++ i ) {

    					dest[ i ] = buffer[ offset ++ ];

    				}

    				this.targetObject.needsUpdate = true;

    			},

    			function setValue_array_setMatrixWorldNeedsUpdate( buffer, offset ) {

    				var dest = this.resolvedProperty;

    				for ( var i = 0, n = dest.length; i !== n; ++ i ) {

    					dest[ i ] = buffer[ offset ++ ];

    				}

    				this.targetObject.matrixWorldNeedsUpdate = true;

    			}

    		], [

    			// ArrayElement

    			function setValue_arrayElement( buffer, offset ) {

    				this.resolvedProperty[ this.propertyIndex ] = buffer[ offset ];

    			},

    			function setValue_arrayElement_setNeedsUpdate( buffer, offset ) {

    				this.resolvedProperty[ this.propertyIndex ] = buffer[ offset ];
    				this.targetObject.needsUpdate = true;

    			},

    			function setValue_arrayElement_setMatrixWorldNeedsUpdate( buffer, offset ) {

    				this.resolvedProperty[ this.propertyIndex ] = buffer[ offset ];
    				this.targetObject.matrixWorldNeedsUpdate = true;

    			}

    		], [

    			// HasToFromArray

    			function setValue_fromArray( buffer, offset ) {

    				this.resolvedProperty.fromArray( buffer, offset );

    			},

    			function setValue_fromArray_setNeedsUpdate( buffer, offset ) {

    				this.resolvedProperty.fromArray( buffer, offset );
    				this.targetObject.needsUpdate = true;

    			},

    			function setValue_fromArray_setMatrixWorldNeedsUpdate( buffer, offset ) {

    				this.resolvedProperty.fromArray( buffer, offset );
    				this.targetObject.matrixWorldNeedsUpdate = true;

    			}

    		]

    	],

    	getValue: function getValue_unbound( targetArray, offset ) {

    		this.bind();
    		this.getValue( targetArray, offset );

    		// Note: This class uses a State pattern on a per-method basis:
    		// 'bind' sets 'this.getValue' / 'setValue' and shadows the
    		// prototype version of these methods with one that represents
    		// the bound state. When the property is not found, the methods
    		// become no-ops.

    	},

    	setValue: function getValue_unbound( sourceArray, offset ) {

    		this.bind();
    		this.setValue( sourceArray, offset );

    	},

    	// create getter / setter pair for a property in the scene graph
    	bind: function () {

    		var targetObject = this.node,
    			parsedPath = this.parsedPath,

    			objectName = parsedPath.objectName,
    			propertyName = parsedPath.propertyName,
    			propertyIndex = parsedPath.propertyIndex;

    		if ( ! targetObject ) {

    			targetObject = PropertyBinding.findNode( this.rootNode, parsedPath.nodeName ) || this.rootNode;

    			this.node = targetObject;

    		}

    		// set fail state so we can just 'return' on error
    		this.getValue = this._getValue_unavailable;
    		this.setValue = this._setValue_unavailable;

    		// ensure there is a value node
    		if ( ! targetObject ) {

    			console.error( 'THREE.PropertyBinding: Trying to update node for track: ' + this.path + ' but it wasn\'t found.' );
    			return;

    		}

    		if ( objectName ) {

    			var objectIndex = parsedPath.objectIndex;

    			// special cases were we need to reach deeper into the hierarchy to get the face materials....
    			switch ( objectName ) {

    				case 'materials':

    					if ( ! targetObject.material ) {

    						console.error( 'THREE.PropertyBinding: Can not bind to material as node does not have a material.', this );
    						return;

    					}

    					if ( ! targetObject.material.materials ) {

    						console.error( 'THREE.PropertyBinding: Can not bind to material.materials as node.material does not have a materials array.', this );
    						return;

    					}

    					targetObject = targetObject.material.materials;

    					break;

    				case 'bones':

    					if ( ! targetObject.skeleton ) {

    						console.error( 'THREE.PropertyBinding: Can not bind to bones as node does not have a skeleton.', this );
    						return;

    					}

    					// potential future optimization: skip this if propertyIndex is already an integer
    					// and convert the integer string to a true integer.

    					targetObject = targetObject.skeleton.bones;

    					// support resolving morphTarget names into indices.
    					for ( var i = 0; i < targetObject.length; i ++ ) {

    						if ( targetObject[ i ].name === objectIndex ) {

    							objectIndex = i;
    							break;

    						}

    					}

    					break;

    				default:

    					if ( targetObject[ objectName ] === undefined ) {

    						console.error( 'THREE.PropertyBinding: Can not bind to objectName of node undefined.', this );
    						return;

    					}

    					targetObject = targetObject[ objectName ];

    			}


    			if ( objectIndex !== undefined ) {

    				if ( targetObject[ objectIndex ] === undefined ) {

    					console.error( 'THREE.PropertyBinding: Trying to bind to objectIndex of objectName, but is undefined.', this, targetObject );
    					return;

    				}

    				targetObject = targetObject[ objectIndex ];

    			}

    		}

    		// resolve property
    		var nodeProperty = targetObject[ propertyName ];

    		if ( nodeProperty === undefined ) {

    			var nodeName = parsedPath.nodeName;

    			console.error( 'THREE.PropertyBinding: Trying to update property for track: ' + nodeName +
    				'.' + propertyName + ' but it wasn\'t found.', targetObject );
    			return;

    		}

    		// determine versioning scheme
    		var versioning = this.Versioning.None;

    		this.targetObject = targetObject;

    		if ( targetObject.needsUpdate !== undefined ) { // material

    			versioning = this.Versioning.NeedsUpdate;

    		} else if ( targetObject.matrixWorldNeedsUpdate !== undefined ) { // node transform

    			versioning = this.Versioning.MatrixWorldNeedsUpdate;

    		}

    		// determine how the property gets bound
    		var bindingType = this.BindingType.Direct;

    		if ( propertyIndex !== undefined ) {

    			// access a sub element of the property array (only primitives are supported right now)

    			if ( propertyName === "morphTargetInfluences" ) {

    				// potential optimization, skip this if propertyIndex is already an integer, and convert the integer string to a true integer.

    				// support resolving morphTarget names into indices.
    				if ( ! targetObject.geometry ) {

    					console.error( 'THREE.PropertyBinding: Can not bind to morphTargetInfluences because node does not have a geometry.', this );
    					return;

    				}

    				if ( targetObject.geometry.isBufferGeometry ) {

    					if ( ! targetObject.geometry.morphAttributes ) {

    						console.error( 'THREE.PropertyBinding: Can not bind to morphTargetInfluences because node does not have a geometry.morphAttributes.', this );
    						return;

    					}

    					for ( var i = 0; i < this.node.geometry.morphAttributes.position.length; i ++ ) {

    						if ( targetObject.geometry.morphAttributes.position[ i ].name === propertyIndex ) {

    							propertyIndex = i;
    							break;

    						}

    					}


    				} else {

    					if ( ! targetObject.geometry.morphTargets ) {

    						console.error( 'THREE.PropertyBinding: Can not bind to morphTargetInfluences because node does not have a geometry.morphTargets.', this );
    						return;

    					}

    					for ( var i = 0; i < this.node.geometry.morphTargets.length; i ++ ) {

    						if ( targetObject.geometry.morphTargets[ i ].name === propertyIndex ) {

    							propertyIndex = i;
    							break;

    						}

    					}

    				}

    			}

    			bindingType = this.BindingType.ArrayElement;

    			this.resolvedProperty = nodeProperty;
    			this.propertyIndex = propertyIndex;

    		} else if ( nodeProperty.fromArray !== undefined && nodeProperty.toArray !== undefined ) {

    			// must use copy for Object3D.Euler/Quaternion

    			bindingType = this.BindingType.HasFromToArray;

    			this.resolvedProperty = nodeProperty;

    		} else if ( Array.isArray( nodeProperty ) ) {

    			bindingType = this.BindingType.EntireArray;

    			this.resolvedProperty = nodeProperty;

    		} else {

    			this.propertyName = propertyName;

    		}

    		// select getter / setter
    		this.getValue = this.GetterByBindingType[ bindingType ];
    		this.setValue = this.SetterByBindingTypeAndVersioning[ bindingType ][ versioning ];

    	},

    	unbind: function () {

    		this.node = null;

    		// back to the prototype version of getValue / setValue
    		// note: avoiding to mutate the shape of 'this' via 'delete'
    		this.getValue = this._getValue_unbound;
    		this.setValue = this._setValue_unbound;

    	}

    } );

    //!\ DECLARE ALIAS AFTER assign prototype !
    Object.assign( PropertyBinding.prototype, {

    	// initial state of these methods that calls 'bind'
    	_getValue_unbound: PropertyBinding.prototype.getValue,
    	_setValue_unbound: PropertyBinding.prototype.setValue,

    } );

    /**
     *
     * A group of objects that receives a shared animation state.
     *
     * Usage:
     *
     *  - Add objects you would otherwise pass as 'root' to the
     *    constructor or the .clipAction method of AnimationMixer.
     *
     *  - Instead pass this object as 'root'.
     *
     *  - You can also add and remove objects later when the mixer
     *    is running.
     *
     * Note:
     *
     *    Objects of this class appear as one object to the mixer,
     *    so cache control of the individual objects must be done
     *    on the group.
     *
     * Limitation:
     *
     *  - The animated properties must be compatible among the
     *    all objects in the group.
     *
     *  - A single property can either be controlled through a
     *    target group or directly, but not both.
     *
     * @author tschw
     */

    function AnimationObjectGroup() {

    	this.uuid = _Math.generateUUID();

    	// cached objects followed by the active ones
    	this._objects = Array.prototype.slice.call( arguments );

    	this.nCachedObjects_ = 0; // threshold
    	// note: read by PropertyBinding.Composite

    	var indices = {};
    	this._indicesByUUID = indices; // for bookkeeping

    	for ( var i = 0, n = arguments.length; i !== n; ++ i ) {

    		indices[ arguments[ i ].uuid ] = i;

    	}

    	this._paths = []; // inside: string
    	this._parsedPaths = []; // inside: { we don't care, here }
    	this._bindings = []; // inside: Array< PropertyBinding >
    	this._bindingsIndicesByPath = {}; // inside: indices in these arrays

    	var scope = this;

    	this.stats = {

    		objects: {
    			get total() {

    				return scope._objects.length;

    			},
    			get inUse() {

    				return this.total - scope.nCachedObjects_;

    			}
    		},
    		get bindingsPerObject() {

    			return scope._bindings.length;

    		}

    	};

    }

    Object.assign( AnimationObjectGroup.prototype, {

    	isAnimationObjectGroup: true,

    	add: function () {

    		var objects = this._objects,
    			nObjects = objects.length,
    			nCachedObjects = this.nCachedObjects_,
    			indicesByUUID = this._indicesByUUID,
    			paths = this._paths,
    			parsedPaths = this._parsedPaths,
    			bindings = this._bindings,
    			nBindings = bindings.length,
    			knownObject = undefined;

    		for ( var i = 0, n = arguments.length; i !== n; ++ i ) {

    			var object = arguments[ i ],
    				uuid = object.uuid,
    				index = indicesByUUID[ uuid ];

    			if ( index === undefined ) {

    				// unknown object -> add it to the ACTIVE region

    				index = nObjects ++;
    				indicesByUUID[ uuid ] = index;
    				objects.push( object );

    				// accounting is done, now do the same for all bindings

    				for ( var j = 0, m = nBindings; j !== m; ++ j ) {

    					bindings[ j ].push( new PropertyBinding( object, paths[ j ], parsedPaths[ j ] ) );

    				}

    			} else if ( index < nCachedObjects ) {

    				knownObject = objects[ index ];

    				// move existing object to the ACTIVE region

    				var firstActiveIndex = -- nCachedObjects,
    					lastCachedObject = objects[ firstActiveIndex ];

    				indicesByUUID[ lastCachedObject.uuid ] = index;
    				objects[ index ] = lastCachedObject;

    				indicesByUUID[ uuid ] = firstActiveIndex;
    				objects[ firstActiveIndex ] = object;

    				// accounting is done, now do the same for all bindings

    				for ( var j = 0, m = nBindings; j !== m; ++ j ) {

    					var bindingsForPath = bindings[ j ],
    						lastCached = bindingsForPath[ firstActiveIndex ],
    						binding = bindingsForPath[ index ];

    					bindingsForPath[ index ] = lastCached;

    					if ( binding === undefined ) {

    						// since we do not bother to create new bindings
    						// for objects that are cached, the binding may
    						// or may not exist

    						binding = new PropertyBinding( object, paths[ j ], parsedPaths[ j ] );

    					}

    					bindingsForPath[ firstActiveIndex ] = binding;

    				}

    			} else if ( objects[ index ] !== knownObject ) {

    				console.error( 'THREE.AnimationObjectGroup: Different objects with the same UUID ' +
    					'detected. Clean the caches or recreate your infrastructure when reloading scenes.' );

    			} // else the object is already where we want it to be

    		} // for arguments

    		this.nCachedObjects_ = nCachedObjects;

    	},

    	remove: function () {

    		var objects = this._objects,
    			nCachedObjects = this.nCachedObjects_,
    			indicesByUUID = this._indicesByUUID,
    			bindings = this._bindings,
    			nBindings = bindings.length;

    		for ( var i = 0, n = arguments.length; i !== n; ++ i ) {

    			var object = arguments[ i ],
    				uuid = object.uuid,
    				index = indicesByUUID[ uuid ];

    			if ( index !== undefined && index >= nCachedObjects ) {

    				// move existing object into the CACHED region

    				var lastCachedIndex = nCachedObjects ++,
    					firstActiveObject = objects[ lastCachedIndex ];

    				indicesByUUID[ firstActiveObject.uuid ] = index;
    				objects[ index ] = firstActiveObject;

    				indicesByUUID[ uuid ] = lastCachedIndex;
    				objects[ lastCachedIndex ] = object;

    				// accounting is done, now do the same for all bindings

    				for ( var j = 0, m = nBindings; j !== m; ++ j ) {

    					var bindingsForPath = bindings[ j ],
    						firstActive = bindingsForPath[ lastCachedIndex ],
    						binding = bindingsForPath[ index ];

    					bindingsForPath[ index ] = firstActive;
    					bindingsForPath[ lastCachedIndex ] = binding;

    				}

    			}

    		} // for arguments

    		this.nCachedObjects_ = nCachedObjects;

    	},

    	// remove & forget
    	uncache: function () {

    		var objects = this._objects,
    			nObjects = objects.length,
    			nCachedObjects = this.nCachedObjects_,
    			indicesByUUID = this._indicesByUUID,
    			bindings = this._bindings,
    			nBindings = bindings.length;

    		for ( var i = 0, n = arguments.length; i !== n; ++ i ) {

    			var object = arguments[ i ],
    				uuid = object.uuid,
    				index = indicesByUUID[ uuid ];

    			if ( index !== undefined ) {

    				delete indicesByUUID[ uuid ];

    				if ( index < nCachedObjects ) {

    					// object is cached, shrink the CACHED region

    					var firstActiveIndex = -- nCachedObjects,
    						lastCachedObject = objects[ firstActiveIndex ],
    						lastIndex = -- nObjects,
    						lastObject = objects[ lastIndex ];

    					// last cached object takes this object's place
    					indicesByUUID[ lastCachedObject.uuid ] = index;
    					objects[ index ] = lastCachedObject;

    					// last object goes to the activated slot and pop
    					indicesByUUID[ lastObject.uuid ] = firstActiveIndex;
    					objects[ firstActiveIndex ] = lastObject;
    					objects.pop();

    					// accounting is done, now do the same for all bindings

    					for ( var j = 0, m = nBindings; j !== m; ++ j ) {

    						var bindingsForPath = bindings[ j ],
    							lastCached = bindingsForPath[ firstActiveIndex ],
    							last = bindingsForPath[ lastIndex ];

    						bindingsForPath[ index ] = lastCached;
    						bindingsForPath[ firstActiveIndex ] = last;
    						bindingsForPath.pop();

    					}

    				} else {

    					// object is active, just swap with the last and pop

    					var lastIndex = -- nObjects,
    						lastObject = objects[ lastIndex ];

    					indicesByUUID[ lastObject.uuid ] = index;
    					objects[ index ] = lastObject;
    					objects.pop();

    					// accounting is done, now do the same for all bindings

    					for ( var j = 0, m = nBindings; j !== m; ++ j ) {

    						var bindingsForPath = bindings[ j ];

    						bindingsForPath[ index ] = bindingsForPath[ lastIndex ];
    						bindingsForPath.pop();

    					}

    				} // cached or active

    			} // if object is known

    		} // for arguments

    		this.nCachedObjects_ = nCachedObjects;

    	},

    	// Internal interface used by befriended PropertyBinding.Composite:

    	subscribe_: function ( path, parsedPath ) {

    		// returns an array of bindings for the given path that is changed
    		// according to the contained objects in the group

    		var indicesByPath = this._bindingsIndicesByPath,
    			index = indicesByPath[ path ],
    			bindings = this._bindings;

    		if ( index !== undefined ) return bindings[ index ];

    		var paths = this._paths,
    			parsedPaths = this._parsedPaths,
    			objects = this._objects,
    			nObjects = objects.length,
    			nCachedObjects = this.nCachedObjects_,
    			bindingsForPath = new Array( nObjects );

    		index = bindings.length;

    		indicesByPath[ path ] = index;

    		paths.push( path );
    		parsedPaths.push( parsedPath );
    		bindings.push( bindingsForPath );

    		for ( var i = nCachedObjects, n = objects.length; i !== n; ++ i ) {

    			var object = objects[ i ];
    			bindingsForPath[ i ] = new PropertyBinding( object, path, parsedPath );

    		}

    		return bindingsForPath;

    	},

    	unsubscribe_: function ( path ) {

    		// tells the group to forget about a property path and no longer
    		// update the array previously obtained with 'subscribe_'

    		var indicesByPath = this._bindingsIndicesByPath,
    			index = indicesByPath[ path ];

    		if ( index !== undefined ) {

    			var paths = this._paths,
    				parsedPaths = this._parsedPaths,
    				bindings = this._bindings,
    				lastBindingsIndex = bindings.length - 1,
    				lastBindings = bindings[ lastBindingsIndex ],
    				lastBindingsPath = path[ lastBindingsIndex ];

    			indicesByPath[ lastBindingsPath ] = index;

    			bindings[ index ] = lastBindings;
    			bindings.pop();

    			parsedPaths[ index ] = parsedPaths[ lastBindingsIndex ];
    			parsedPaths.pop();

    			paths[ index ] = paths[ lastBindingsIndex ];
    			paths.pop();

    		}

    	}

    } );

    /**
     *
     * Action provided by AnimationMixer for scheduling clip playback on specific
     * objects.
     *
     * @author Ben Houston / http://clara.io/
     * @author David Sarno / http://lighthaus.us/
     * @author tschw
     *
     */

    function AnimationAction( mixer, clip, localRoot ) {

    	this._mixer = mixer;
    	this._clip = clip;
    	this._localRoot = localRoot || null;

    	var tracks = clip.tracks,
    		nTracks = tracks.length,
    		interpolants = new Array( nTracks );

    	var interpolantSettings = {
    		endingStart: ZeroCurvatureEnding,
    		endingEnd: ZeroCurvatureEnding
    	};

    	for ( var i = 0; i !== nTracks; ++ i ) {

    		var interpolant = tracks[ i ].createInterpolant( null );
    		interpolants[ i ] = interpolant;
    		interpolant.settings = interpolantSettings;

    	}

    	this._interpolantSettings = interpolantSettings;

    	this._interpolants = interpolants; // bound by the mixer

    	// inside: PropertyMixer (managed by the mixer)
    	this._propertyBindings = new Array( nTracks );

    	this._cacheIndex = null; // for the memory manager
    	this._byClipCacheIndex = null; // for the memory manager

    	this._timeScaleInterpolant = null;
    	this._weightInterpolant = null;

    	this.loop = LoopRepeat;
    	this._loopCount = - 1;

    	// global mixer time when the action is to be started
    	// it's set back to 'null' upon start of the action
    	this._startTime = null;

    	// scaled local time of the action
    	// gets clamped or wrapped to 0..clip.duration according to loop
    	this.time = 0;

    	this.timeScale = 1;
    	this._effectiveTimeScale = 1;

    	this.weight = 1;
    	this._effectiveWeight = 1;

    	this.repetitions = Infinity; // no. of repetitions when looping

    	this.paused = false; // true -> zero effective time scale
    	this.enabled = true; // false -> zero effective weight

    	this.clampWhenFinished = false;// keep feeding the last frame?

    	this.zeroSlopeAtStart = true;// for smooth interpolation w/o separate
    	this.zeroSlopeAtEnd = true;// clips for start, loop and end

    }

    Object.assign( AnimationAction.prototype, {

    	// State & Scheduling

    	play: function () {

    		this._mixer._activateAction( this );

    		return this;

    	},

    	stop: function () {

    		this._mixer._deactivateAction( this );

    		return this.reset();

    	},

    	reset: function () {

    		this.paused = false;
    		this.enabled = true;

    		this.time = 0; // restart clip
    		this._loopCount = - 1;// forget previous loops
    		this._startTime = null;// forget scheduling

    		return this.stopFading().stopWarping();

    	},

    	isRunning: function () {

    		return this.enabled && ! this.paused && this.timeScale !== 0 &&
    			this._startTime === null && this._mixer._isActiveAction( this );

    	},

    	// return true when play has been called
    	isScheduled: function () {

    		return this._mixer._isActiveAction( this );

    	},

    	startAt: function ( time ) {

    		this._startTime = time;

    		return this;

    	},

    	setLoop: function ( mode, repetitions ) {

    		this.loop = mode;
    		this.repetitions = repetitions;

    		return this;

    	},

    	// Weight

    	// set the weight stopping any scheduled fading
    	// although .enabled = false yields an effective weight of zero, this
    	// method does *not* change .enabled, because it would be confusing
    	setEffectiveWeight: function ( weight ) {

    		this.weight = weight;

    		// note: same logic as when updated at runtime
    		this._effectiveWeight = this.enabled ? weight : 0;

    		return this.stopFading();

    	},

    	// return the weight considering fading and .enabled
    	getEffectiveWeight: function () {

    		return this._effectiveWeight;

    	},

    	fadeIn: function ( duration ) {

    		return this._scheduleFading( duration, 0, 1 );

    	},

    	fadeOut: function ( duration ) {

    		return this._scheduleFading( duration, 1, 0 );

    	},

    	crossFadeFrom: function ( fadeOutAction, duration, warp ) {

    		fadeOutAction.fadeOut( duration );
    		this.fadeIn( duration );

    		if ( warp ) {

    			var fadeInDuration = this._clip.duration,
    				fadeOutDuration = fadeOutAction._clip.duration,

    				startEndRatio = fadeOutDuration / fadeInDuration,
    				endStartRatio = fadeInDuration / fadeOutDuration;

    			fadeOutAction.warp( 1.0, startEndRatio, duration );
    			this.warp( endStartRatio, 1.0, duration );

    		}

    		return this;

    	},

    	crossFadeTo: function ( fadeInAction, duration, warp ) {

    		return fadeInAction.crossFadeFrom( this, duration, warp );

    	},

    	stopFading: function () {

    		var weightInterpolant = this._weightInterpolant;

    		if ( weightInterpolant !== null ) {

    			this._weightInterpolant = null;
    			this._mixer._takeBackControlInterpolant( weightInterpolant );

    		}

    		return this;

    	},

    	// Time Scale Control

    	// set the time scale stopping any scheduled warping
    	// although .paused = true yields an effective time scale of zero, this
    	// method does *not* change .paused, because it would be confusing
    	setEffectiveTimeScale: function ( timeScale ) {

    		this.timeScale = timeScale;
    		this._effectiveTimeScale = this.paused ? 0 : timeScale;

    		return this.stopWarping();

    	},

    	// return the time scale considering warping and .paused
    	getEffectiveTimeScale: function () {

    		return this._effectiveTimeScale;

    	},

    	setDuration: function ( duration ) {

    		this.timeScale = this._clip.duration / duration;

    		return this.stopWarping();

    	},

    	syncWith: function ( action ) {

    		this.time = action.time;
    		this.timeScale = action.timeScale;

    		return this.stopWarping();

    	},

    	halt: function ( duration ) {

    		return this.warp( this._effectiveTimeScale, 0, duration );

    	},

    	warp: function ( startTimeScale, endTimeScale, duration ) {

    		var mixer = this._mixer, now = mixer.time,
    			interpolant = this._timeScaleInterpolant,

    			timeScale = this.timeScale;

    		if ( interpolant === null ) {

    			interpolant = mixer._lendControlInterpolant();
    			this._timeScaleInterpolant = interpolant;

    		}

    		var times = interpolant.parameterPositions,
    			values = interpolant.sampleValues;

    		times[ 0 ] = now;
    		times[ 1 ] = now + duration;

    		values[ 0 ] = startTimeScale / timeScale;
    		values[ 1 ] = endTimeScale / timeScale;

    		return this;

    	},

    	stopWarping: function () {

    		var timeScaleInterpolant = this._timeScaleInterpolant;

    		if ( timeScaleInterpolant !== null ) {

    			this._timeScaleInterpolant = null;
    			this._mixer._takeBackControlInterpolant( timeScaleInterpolant );

    		}

    		return this;

    	},

    	// Object Accessors

    	getMixer: function () {

    		return this._mixer;

    	},

    	getClip: function () {

    		return this._clip;

    	},

    	getRoot: function () {

    		return this._localRoot || this._mixer._root;

    	},

    	// Interna

    	_update: function ( time, deltaTime, timeDirection, accuIndex ) {

    		// called by the mixer

    		if ( ! this.enabled ) {

    			// call ._updateWeight() to update ._effectiveWeight

    			this._updateWeight( time );
    			return;

    		}

    		var startTime = this._startTime;

    		if ( startTime !== null ) {

    			// check for scheduled start of action

    			var timeRunning = ( time - startTime ) * timeDirection;
    			if ( timeRunning < 0 || timeDirection === 0 ) {

    				return; // yet to come / don't decide when delta = 0

    			}

    			// start

    			this._startTime = null; // unschedule
    			deltaTime = timeDirection * timeRunning;

    		}

    		// apply time scale and advance time

    		deltaTime *= this._updateTimeScale( time );
    		var clipTime = this._updateTime( deltaTime );

    		// note: _updateTime may disable the action resulting in
    		// an effective weight of 0

    		var weight = this._updateWeight( time );

    		if ( weight > 0 ) {

    			var interpolants = this._interpolants;
    			var propertyMixers = this._propertyBindings;

    			for ( var j = 0, m = interpolants.length; j !== m; ++ j ) {

    				interpolants[ j ].evaluate( clipTime );
    				propertyMixers[ j ].accumulate( accuIndex, weight );

    			}

    		}

    	},

    	_updateWeight: function ( time ) {

    		var weight = 0;

    		if ( this.enabled ) {

    			weight = this.weight;
    			var interpolant = this._weightInterpolant;

    			if ( interpolant !== null ) {

    				var interpolantValue = interpolant.evaluate( time )[ 0 ];

    				weight *= interpolantValue;

    				if ( time > interpolant.parameterPositions[ 1 ] ) {

    					this.stopFading();

    					if ( interpolantValue === 0 ) {

    						// faded out, disable
    						this.enabled = false;

    					}

    				}

    			}

    		}

    		this._effectiveWeight = weight;
    		return weight;

    	},

    	_updateTimeScale: function ( time ) {

    		var timeScale = 0;

    		if ( ! this.paused ) {

    			timeScale = this.timeScale;

    			var interpolant = this._timeScaleInterpolant;

    			if ( interpolant !== null ) {

    				var interpolantValue = interpolant.evaluate( time )[ 0 ];

    				timeScale *= interpolantValue;

    				if ( time > interpolant.parameterPositions[ 1 ] ) {

    					this.stopWarping();

    					if ( timeScale === 0 ) {

    						// motion has halted, pause
    						this.paused = true;

    					} else {

    						// warp done - apply final time scale
    						this.timeScale = timeScale;

    					}

    				}

    			}

    		}

    		this._effectiveTimeScale = timeScale;
    		return timeScale;

    	},

    	_updateTime: function ( deltaTime ) {

    		var time = this.time + deltaTime;
    		var duration = this._clip.duration;
    		var loop = this.loop;
    		var loopCount = this._loopCount;

    		var pingPong = ( loop === LoopPingPong );

    		if ( deltaTime === 0 ) {

    			if ( loopCount === - 1 ) return time;

    			return ( pingPong && ( loopCount & 1 ) === 1 ) ? duration - time : time;

    		}

    		if ( loop === LoopOnce ) {

    			if ( loopCount === - 1 ) {

    				// just started

    				this._loopCount = 0;
    				this._setEndings( true, true, false );

    			}

    			handle_stop: {

    				if ( time >= duration ) {

    					time = duration;

    				} else if ( time < 0 ) {

    					time = 0;

    				} else {

    					this.time = time;

    					break handle_stop;

    				}

    				if ( this.clampWhenFinished ) this.paused = true;
    				else this.enabled = false;

    				this.time = time;

    				this._mixer.dispatchEvent( {
    					type: 'finished', action: this,
    					direction: deltaTime < 0 ? - 1 : 1
    				} );

    			}

    		} else { // repetitive Repeat or PingPong

    			if ( loopCount === - 1 ) {

    				// just started

    				if ( deltaTime >= 0 ) {

    					loopCount = 0;

    					this._setEndings( true, this.repetitions === 0, pingPong );

    				} else {

    					// when looping in reverse direction, the initial
    					// transition through zero counts as a repetition,
    					// so leave loopCount at -1

    					this._setEndings( this.repetitions === 0, true, pingPong );

    				}

    			}

    			if ( time >= duration || time < 0 ) {

    				// wrap around

    				var loopDelta = Math.floor( time / duration ); // signed
    				time -= duration * loopDelta;

    				loopCount += Math.abs( loopDelta );

    				var pending = this.repetitions - loopCount;

    				if ( pending <= 0 ) {

    					// have to stop (switch state, clamp time, fire event)

    					if ( this.clampWhenFinished ) this.paused = true;
    					else this.enabled = false;

    					time = deltaTime > 0 ? duration : 0;

    					this.time = time;

    					this._mixer.dispatchEvent( {
    						type: 'finished', action: this,
    						direction: deltaTime > 0 ? 1 : - 1
    					} );

    				} else {

    					// keep running

    					if ( pending === 1 ) {

    						// entering the last round

    						var atStart = deltaTime < 0;
    						this._setEndings( atStart, ! atStart, pingPong );

    					} else {

    						this._setEndings( false, false, pingPong );

    					}

    					this._loopCount = loopCount;

    					this.time = time;

    					this._mixer.dispatchEvent( {
    						type: 'loop', action: this, loopDelta: loopDelta
    					} );

    				}

    			} else {

    				this.time = time;

    			}

    			if ( pingPong && ( loopCount & 1 ) === 1 ) {

    				// invert time for the "pong round"

    				return duration - time;

    			}

    		}

    		return time;

    	},

    	_setEndings: function ( atStart, atEnd, pingPong ) {

    		var settings = this._interpolantSettings;

    		if ( pingPong ) {

    			settings.endingStart = ZeroSlopeEnding;
    			settings.endingEnd = ZeroSlopeEnding;

    		} else {

    			// assuming for LoopOnce atStart == atEnd == true

    			if ( atStart ) {

    				settings.endingStart = this.zeroSlopeAtStart ? ZeroSlopeEnding : ZeroCurvatureEnding;

    			} else {

    				settings.endingStart = WrapAroundEnding;

    			}

    			if ( atEnd ) {

    				settings.endingEnd = this.zeroSlopeAtEnd ? ZeroSlopeEnding : ZeroCurvatureEnding;

    			} else {

    				settings.endingEnd 	 = WrapAroundEnding;

    			}

    		}

    	},

    	_scheduleFading: function ( duration, weightNow, weightThen ) {

    		var mixer = this._mixer, now = mixer.time,
    			interpolant = this._weightInterpolant;

    		if ( interpolant === null ) {

    			interpolant = mixer._lendControlInterpolant();
    			this._weightInterpolant = interpolant;

    		}

    		var times = interpolant.parameterPositions,
    			values = interpolant.sampleValues;

    		times[ 0 ] = now;
    		values[ 0 ] = weightNow;
    		times[ 1 ] = now + duration;
    		values[ 1 ] = weightThen;

    		return this;

    	}

    } );

    /**
     *
     * Player for AnimationClips.
     *
     *
     * @author Ben Houston / http://clara.io/
     * @author David Sarno / http://lighthaus.us/
     * @author tschw
     */

    function AnimationMixer( root ) {

    	this._root = root;
    	this._initMemoryManager();
    	this._accuIndex = 0;

    	this.time = 0;

    	this.timeScale = 1.0;

    }

    AnimationMixer.prototype = Object.assign( Object.create( EventDispatcher.prototype ), {

    	constructor: AnimationMixer,

    	_bindAction: function ( action, prototypeAction ) {

    		var root = action._localRoot || this._root,
    			tracks = action._clip.tracks,
    			nTracks = tracks.length,
    			bindings = action._propertyBindings,
    			interpolants = action._interpolants,
    			rootUuid = root.uuid,
    			bindingsByRoot = this._bindingsByRootAndName,
    			bindingsByName = bindingsByRoot[ rootUuid ];

    		if ( bindingsByName === undefined ) {

    			bindingsByName = {};
    			bindingsByRoot[ rootUuid ] = bindingsByName;

    		}

    		for ( var i = 0; i !== nTracks; ++ i ) {

    			var track = tracks[ i ],
    				trackName = track.name,
    				binding = bindingsByName[ trackName ];

    			if ( binding !== undefined ) {

    				bindings[ i ] = binding;

    			} else {

    				binding = bindings[ i ];

    				if ( binding !== undefined ) {

    					// existing binding, make sure the cache knows

    					if ( binding._cacheIndex === null ) {

    						++ binding.referenceCount;
    						this._addInactiveBinding( binding, rootUuid, trackName );

    					}

    					continue;

    				}

    				var path = prototypeAction && prototypeAction.
    					_propertyBindings[ i ].binding.parsedPath;

    				binding = new PropertyMixer(
    					PropertyBinding.create( root, trackName, path ),
    					track.ValueTypeName, track.getValueSize() );

    				++ binding.referenceCount;
    				this._addInactiveBinding( binding, rootUuid, trackName );

    				bindings[ i ] = binding;

    			}

    			interpolants[ i ].resultBuffer = binding.buffer;

    		}

    	},

    	_activateAction: function ( action ) {

    		if ( ! this._isActiveAction( action ) ) {

    			if ( action._cacheIndex === null ) {

    				// this action has been forgotten by the cache, but the user
    				// appears to be still using it -> rebind

    				var rootUuid = ( action._localRoot || this._root ).uuid,
    					clipUuid = action._clip.uuid,
    					actionsForClip = this._actionsByClip[ clipUuid ];

    				this._bindAction( action,
    					actionsForClip && actionsForClip.knownActions[ 0 ] );

    				this._addInactiveAction( action, clipUuid, rootUuid );

    			}

    			var bindings = action._propertyBindings;

    			// increment reference counts / sort out state
    			for ( var i = 0, n = bindings.length; i !== n; ++ i ) {

    				var binding = bindings[ i ];

    				if ( binding.useCount ++ === 0 ) {

    					this._lendBinding( binding );
    					binding.saveOriginalState();

    				}

    			}

    			this._lendAction( action );

    		}

    	},

    	_deactivateAction: function ( action ) {

    		if ( this._isActiveAction( action ) ) {

    			var bindings = action._propertyBindings;

    			// decrement reference counts / sort out state
    			for ( var i = 0, n = bindings.length; i !== n; ++ i ) {

    				var binding = bindings[ i ];

    				if ( -- binding.useCount === 0 ) {

    					binding.restoreOriginalState();
    					this._takeBackBinding( binding );

    				}

    			}

    			this._takeBackAction( action );

    		}

    	},

    	// Memory manager

    	_initMemoryManager: function () {

    		this._actions = []; // 'nActiveActions' followed by inactive ones
    		this._nActiveActions = 0;

    		this._actionsByClip = {};
    		// inside:
    		// {
    		// 	knownActions: Array< AnimationAction > - used as prototypes
    		// 	actionByRoot: AnimationAction - lookup
    		// }


    		this._bindings = []; // 'nActiveBindings' followed by inactive ones
    		this._nActiveBindings = 0;

    		this._bindingsByRootAndName = {}; // inside: Map< name, PropertyMixer >


    		this._controlInterpolants = []; // same game as above
    		this._nActiveControlInterpolants = 0;

    		var scope = this;

    		this.stats = {

    			actions: {
    				get total() {

    					return scope._actions.length;

    				},
    				get inUse() {

    					return scope._nActiveActions;

    				}
    			},
    			bindings: {
    				get total() {

    					return scope._bindings.length;

    				},
    				get inUse() {

    					return scope._nActiveBindings;

    				}
    			},
    			controlInterpolants: {
    				get total() {

    					return scope._controlInterpolants.length;

    				},
    				get inUse() {

    					return scope._nActiveControlInterpolants;

    				}
    			}

    		};

    	},

    	// Memory management for AnimationAction objects

    	_isActiveAction: function ( action ) {

    		var index = action._cacheIndex;
    		return index !== null && index < this._nActiveActions;

    	},

    	_addInactiveAction: function ( action, clipUuid, rootUuid ) {

    		var actions = this._actions,
    			actionsByClip = this._actionsByClip,
    			actionsForClip = actionsByClip[ clipUuid ];

    		if ( actionsForClip === undefined ) {

    			actionsForClip = {

    				knownActions: [ action ],
    				actionByRoot: {}

    			};

    			action._byClipCacheIndex = 0;

    			actionsByClip[ clipUuid ] = actionsForClip;

    		} else {

    			var knownActions = actionsForClip.knownActions;

    			action._byClipCacheIndex = knownActions.length;
    			knownActions.push( action );

    		}

    		action._cacheIndex = actions.length;
    		actions.push( action );

    		actionsForClip.actionByRoot[ rootUuid ] = action;

    	},

    	_removeInactiveAction: function ( action ) {

    		var actions = this._actions,
    			lastInactiveAction = actions[ actions.length - 1 ],
    			cacheIndex = action._cacheIndex;

    		lastInactiveAction._cacheIndex = cacheIndex;
    		actions[ cacheIndex ] = lastInactiveAction;
    		actions.pop();

    		action._cacheIndex = null;


    		var clipUuid = action._clip.uuid,
    			actionsByClip = this._actionsByClip,
    			actionsForClip = actionsByClip[ clipUuid ],
    			knownActionsForClip = actionsForClip.knownActions,

    			lastKnownAction =
    				knownActionsForClip[ knownActionsForClip.length - 1 ],

    			byClipCacheIndex = action._byClipCacheIndex;

    		lastKnownAction._byClipCacheIndex = byClipCacheIndex;
    		knownActionsForClip[ byClipCacheIndex ] = lastKnownAction;
    		knownActionsForClip.pop();

    		action._byClipCacheIndex = null;


    		var actionByRoot = actionsForClip.actionByRoot,
    			rootUuid = ( action._localRoot || this._root ).uuid;

    		delete actionByRoot[ rootUuid ];

    		if ( knownActionsForClip.length === 0 ) {

    			delete actionsByClip[ clipUuid ];

    		}

    		this._removeInactiveBindingsForAction( action );

    	},

    	_removeInactiveBindingsForAction: function ( action ) {

    		var bindings = action._propertyBindings;
    		for ( var i = 0, n = bindings.length; i !== n; ++ i ) {

    			var binding = bindings[ i ];

    			if ( -- binding.referenceCount === 0 ) {

    				this._removeInactiveBinding( binding );

    			}

    		}

    	},

    	_lendAction: function ( action ) {

    		// [ active actions |  inactive actions  ]
    		// [  active actions >| inactive actions ]
    		//                 s        a
    		//                  <-swap->
    		//                 a        s

    		var actions = this._actions,
    			prevIndex = action._cacheIndex,

    			lastActiveIndex = this._nActiveActions ++,

    			firstInactiveAction = actions[ lastActiveIndex ];

    		action._cacheIndex = lastActiveIndex;
    		actions[ lastActiveIndex ] = action;

    		firstInactiveAction._cacheIndex = prevIndex;
    		actions[ prevIndex ] = firstInactiveAction;

    	},

    	_takeBackAction: function ( action ) {

    		// [  active actions  | inactive actions ]
    		// [ active actions |< inactive actions  ]
    		//        a        s
    		//         <-swap->
    		//        s        a

    		var actions = this._actions,
    			prevIndex = action._cacheIndex,

    			firstInactiveIndex = -- this._nActiveActions,

    			lastActiveAction = actions[ firstInactiveIndex ];

    		action._cacheIndex = firstInactiveIndex;
    		actions[ firstInactiveIndex ] = action;

    		lastActiveAction._cacheIndex = prevIndex;
    		actions[ prevIndex ] = lastActiveAction;

    	},

    	// Memory management for PropertyMixer objects

    	_addInactiveBinding: function ( binding, rootUuid, trackName ) {

    		var bindingsByRoot = this._bindingsByRootAndName,
    			bindingByName = bindingsByRoot[ rootUuid ],

    			bindings = this._bindings;

    		if ( bindingByName === undefined ) {

    			bindingByName = {};
    			bindingsByRoot[ rootUuid ] = bindingByName;

    		}

    		bindingByName[ trackName ] = binding;

    		binding._cacheIndex = bindings.length;
    		bindings.push( binding );

    	},

    	_removeInactiveBinding: function ( binding ) {

    		var bindings = this._bindings,
    			propBinding = binding.binding,
    			rootUuid = propBinding.rootNode.uuid,
    			trackName = propBinding.path,
    			bindingsByRoot = this._bindingsByRootAndName,
    			bindingByName = bindingsByRoot[ rootUuid ],

    			lastInactiveBinding = bindings[ bindings.length - 1 ],
    			cacheIndex = binding._cacheIndex;

    		lastInactiveBinding._cacheIndex = cacheIndex;
    		bindings[ cacheIndex ] = lastInactiveBinding;
    		bindings.pop();

    		delete bindingByName[ trackName ];

    		if ( Object.keys( bindingByName ).length === 0 ) {

    			delete bindingsByRoot[ rootUuid ];

    		}

    	},

    	_lendBinding: function ( binding ) {

    		var bindings = this._bindings,
    			prevIndex = binding._cacheIndex,

    			lastActiveIndex = this._nActiveBindings ++,

    			firstInactiveBinding = bindings[ lastActiveIndex ];

    		binding._cacheIndex = lastActiveIndex;
    		bindings[ lastActiveIndex ] = binding;

    		firstInactiveBinding._cacheIndex = prevIndex;
    		bindings[ prevIndex ] = firstInactiveBinding;

    	},

    	_takeBackBinding: function ( binding ) {

    		var bindings = this._bindings,
    			prevIndex = binding._cacheIndex,

    			firstInactiveIndex = -- this._nActiveBindings,

    			lastActiveBinding = bindings[ firstInactiveIndex ];

    		binding._cacheIndex = firstInactiveIndex;
    		bindings[ firstInactiveIndex ] = binding;

    		lastActiveBinding._cacheIndex = prevIndex;
    		bindings[ prevIndex ] = lastActiveBinding;

    	},


    	// Memory management of Interpolants for weight and time scale

    	_lendControlInterpolant: function () {

    		var interpolants = this._controlInterpolants,
    			lastActiveIndex = this._nActiveControlInterpolants ++,
    			interpolant = interpolants[ lastActiveIndex ];

    		if ( interpolant === undefined ) {

    			interpolant = new LinearInterpolant(
    				new Float32Array( 2 ), new Float32Array( 2 ),
    				1, this._controlInterpolantsResultBuffer );

    			interpolant.__cacheIndex = lastActiveIndex;
    			interpolants[ lastActiveIndex ] = interpolant;

    		}

    		return interpolant;

    	},

    	_takeBackControlInterpolant: function ( interpolant ) {

    		var interpolants = this._controlInterpolants,
    			prevIndex = interpolant.__cacheIndex,

    			firstInactiveIndex = -- this._nActiveControlInterpolants,

    			lastActiveInterpolant = interpolants[ firstInactiveIndex ];

    		interpolant.__cacheIndex = firstInactiveIndex;
    		interpolants[ firstInactiveIndex ] = interpolant;

    		lastActiveInterpolant.__cacheIndex = prevIndex;
    		interpolants[ prevIndex ] = lastActiveInterpolant;

    	},

    	_controlInterpolantsResultBuffer: new Float32Array( 1 ),

    	// return an action for a clip optionally using a custom root target
    	// object (this method allocates a lot of dynamic memory in case a
    	// previously unknown clip/root combination is specified)
    	clipAction: function ( clip, optionalRoot ) {

    		var root = optionalRoot || this._root,
    			rootUuid = root.uuid,

    			clipObject = typeof clip === 'string' ?
    				AnimationClip.findByName( root, clip ) : clip,

    			clipUuid = clipObject !== null ? clipObject.uuid : clip,

    			actionsForClip = this._actionsByClip[ clipUuid ],
    			prototypeAction = null;

    		if ( actionsForClip !== undefined ) {

    			var existingAction =
    					actionsForClip.actionByRoot[ rootUuid ];

    			if ( existingAction !== undefined ) {

    				return existingAction;

    			}

    			// we know the clip, so we don't have to parse all
    			// the bindings again but can just copy
    			prototypeAction = actionsForClip.knownActions[ 0 ];

    			// also, take the clip from the prototype action
    			if ( clipObject === null )
    				clipObject = prototypeAction._clip;

    		}

    		// clip must be known when specified via string
    		if ( clipObject === null ) return null;

    		// allocate all resources required to run it
    		var newAction = new AnimationAction( this, clipObject, optionalRoot );

    		this._bindAction( newAction, prototypeAction );

    		// and make the action known to the memory manager
    		this._addInactiveAction( newAction, clipUuid, rootUuid );

    		return newAction;

    	},

    	// get an existing action
    	existingAction: function ( clip, optionalRoot ) {

    		var root = optionalRoot || this._root,
    			rootUuid = root.uuid,

    			clipObject = typeof clip === 'string' ?
    				AnimationClip.findByName( root, clip ) : clip,

    			clipUuid = clipObject ? clipObject.uuid : clip,

    			actionsForClip = this._actionsByClip[ clipUuid ];

    		if ( actionsForClip !== undefined ) {

    			return actionsForClip.actionByRoot[ rootUuid ] || null;

    		}

    		return null;

    	},

    	// deactivates all previously scheduled actions
    	stopAllAction: function () {

    		var actions = this._actions,
    			nActions = this._nActiveActions,
    			bindings = this._bindings,
    			nBindings = this._nActiveBindings;

    		this._nActiveActions = 0;
    		this._nActiveBindings = 0;

    		for ( var i = 0; i !== nActions; ++ i ) {

    			actions[ i ].reset();

    		}

    		for ( var i = 0; i !== nBindings; ++ i ) {

    			bindings[ i ].useCount = 0;

    		}

    		return this;

    	},

    	// advance the time and update apply the animation
    	update: function ( deltaTime ) {

    		deltaTime *= this.timeScale;

    		var actions = this._actions,
    			nActions = this._nActiveActions,

    			time = this.time += deltaTime,
    			timeDirection = Math.sign( deltaTime ),

    			accuIndex = this._accuIndex ^= 1;

    		// run active actions

    		for ( var i = 0; i !== nActions; ++ i ) {

    			var action = actions[ i ];

    			action._update( time, deltaTime, timeDirection, accuIndex );

    		}

    		// update scene graph

    		var bindings = this._bindings,
    			nBindings = this._nActiveBindings;

    		for ( var i = 0; i !== nBindings; ++ i ) {

    			bindings[ i ].apply( accuIndex );

    		}

    		return this;

    	},

    	// return this mixer's root target object
    	getRoot: function () {

    		return this._root;

    	},

    	// free all resources specific to a particular clip
    	uncacheClip: function ( clip ) {

    		var actions = this._actions,
    			clipUuid = clip.uuid,
    			actionsByClip = this._actionsByClip,
    			actionsForClip = actionsByClip[ clipUuid ];

    		if ( actionsForClip !== undefined ) {

    			// note: just calling _removeInactiveAction would mess up the
    			// iteration state and also require updating the state we can
    			// just throw away

    			var actionsToRemove = actionsForClip.knownActions;

    			for ( var i = 0, n = actionsToRemove.length; i !== n; ++ i ) {

    				var action = actionsToRemove[ i ];

    				this._deactivateAction( action );

    				var cacheIndex = action._cacheIndex,
    					lastInactiveAction = actions[ actions.length - 1 ];

    				action._cacheIndex = null;
    				action._byClipCacheIndex = null;

    				lastInactiveAction._cacheIndex = cacheIndex;
    				actions[ cacheIndex ] = lastInactiveAction;
    				actions.pop();

    				this._removeInactiveBindingsForAction( action );

    			}

    			delete actionsByClip[ clipUuid ];

    		}

    	},

    	// free all resources specific to a particular root target object
    	uncacheRoot: function ( root ) {

    		var rootUuid = root.uuid,
    			actionsByClip = this._actionsByClip;

    		for ( var clipUuid in actionsByClip ) {

    			var actionByRoot = actionsByClip[ clipUuid ].actionByRoot,
    				action = actionByRoot[ rootUuid ];

    			if ( action !== undefined ) {

    				this._deactivateAction( action );
    				this._removeInactiveAction( action );

    			}

    		}

    		var bindingsByRoot = this._bindingsByRootAndName,
    			bindingByName = bindingsByRoot[ rootUuid ];

    		if ( bindingByName !== undefined ) {

    			for ( var trackName in bindingByName ) {

    				var binding = bindingByName[ trackName ];
    				binding.restoreOriginalState();
    				this._removeInactiveBinding( binding );

    			}

    		}

    	},

    	// remove a targeted clip from the cache
    	uncacheAction: function ( clip, optionalRoot ) {

    		var action = this.existingAction( clip, optionalRoot );

    		if ( action !== null ) {

    			this._deactivateAction( action );
    			this._removeInactiveAction( action );

    		}

    	}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function Uniform( value ) {

    	if ( typeof value === 'string' ) {

    		console.warn( 'THREE.Uniform: Type parameter is no longer needed.' );
    		value = arguments[ 1 ];

    	}

    	this.value = value;

    }

    Uniform.prototype.clone = function () {

    	return new Uniform( this.value.clone === undefined ? this.value : this.value.clone() );

    };

    /**
     * @author benaadams / https://twitter.com/ben_a_adams
     */

    function InstancedInterleavedBuffer( array, stride, meshPerAttribute ) {

    	InterleavedBuffer.call( this, array, stride );

    	this.meshPerAttribute = meshPerAttribute || 1;

    }

    InstancedInterleavedBuffer.prototype = Object.assign( Object.create( InterleavedBuffer.prototype ), {

    	constructor: InstancedInterleavedBuffer,

    	isInstancedInterleavedBuffer: true,

    	copy: function ( source ) {

    		InterleavedBuffer.prototype.copy.call( this, source );

    		this.meshPerAttribute = source.meshPerAttribute;

    		return this;

    	}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     * @author bhouston / http://clara.io/
     * @author stephomi / http://stephaneginier.com/
     */

    function Raycaster( origin, direction, near, far ) {

    	this.ray = new Ray( origin, direction );
    	// direction is assumed to be normalized (for accurate distance calculations)

    	this.near = near || 0;
    	this.far = far || Infinity;
    	this.camera = null;

    	this.params = {
    		Mesh: {},
    		Line: {},
    		LOD: {},
    		Points: { threshold: 1 },
    		Sprite: {}
    	};

    	Object.defineProperties( this.params, {
    		PointCloud: {
    			get: function () {

    				console.warn( 'THREE.Raycaster: params.PointCloud has been renamed to params.Points.' );
    				return this.Points;

    			}
    		}
    	} );

    }

    function ascSort( a, b ) {

    	return a.distance - b.distance;

    }

    function intersectObject( object, raycaster, intersects, recursive ) {

    	if ( object.visible === false ) return;

    	object.raycast( raycaster, intersects );

    	if ( recursive === true ) {

    		var children = object.children;

    		for ( var i = 0, l = children.length; i < l; i ++ ) {

    			intersectObject( children[ i ], raycaster, intersects, true );

    		}

    	}

    }

    Object.assign( Raycaster.prototype, {

    	linePrecision: 1,

    	set: function ( origin, direction ) {

    		// direction is assumed to be normalized (for accurate distance calculations)

    		this.ray.set( origin, direction );

    	},

    	setFromCamera: function ( coords, camera ) {

    		if ( ( camera && camera.isPerspectiveCamera ) ) {

    			this.ray.origin.setFromMatrixPosition( camera.matrixWorld );
    			this.ray.direction.set( coords.x, coords.y, 0.5 ).unproject( camera ).sub( this.ray.origin ).normalize();
    			this.camera = camera;

    		} else if ( ( camera && camera.isOrthographicCamera ) ) {

    			this.ray.origin.set( coords.x, coords.y, ( camera.near + camera.far ) / ( camera.near - camera.far ) ).unproject( camera ); // set origin in plane of camera
    			this.ray.direction.set( 0, 0, - 1 ).transformDirection( camera.matrixWorld );
    			this.camera = camera;

    		} else {

    			console.error( 'THREE.Raycaster: Unsupported camera type.' );

    		}

    	},

    	intersectObject: function ( object, recursive, optionalTarget ) {

    		var intersects = optionalTarget || [];

    		intersectObject( object, this, intersects, recursive );

    		intersects.sort( ascSort );

    		return intersects;

    	},

    	intersectObjects: function ( objects, recursive, optionalTarget ) {

    		var intersects = optionalTarget || [];

    		if ( Array.isArray( objects ) === false ) {

    			console.warn( 'THREE.Raycaster.intersectObjects: objects is not an Array.' );
    			return intersects;

    		}

    		for ( var i = 0, l = objects.length; i < l; i ++ ) {

    			intersectObject( objects[ i ], this, intersects, recursive );

    		}

    		intersects.sort( ascSort );

    		return intersects;

    	}

    } );

    /**
     * @author bhouston / http://clara.io
     * @author WestLangley / http://github.com/WestLangley
     *
     * Ref: https://en.wikipedia.org/wiki/Spherical_coordinate_system
     *
     * The polar angle (phi) is measured from the positive y-axis. The positive y-axis is up.
     * The azimuthal angle (theta) is measured from the positive z-axiz.
     */

    function Spherical( radius, phi, theta ) {

    	this.radius = ( radius !== undefined ) ? radius : 1.0;
    	this.phi = ( phi !== undefined ) ? phi : 0; // polar angle
    	this.theta = ( theta !== undefined ) ? theta : 0; // azimuthal angle

    	return this;

    }

    Object.assign( Spherical.prototype, {

    	set: function ( radius, phi, theta ) {

    		this.radius = radius;
    		this.phi = phi;
    		this.theta = theta;

    		return this;

    	},

    	clone: function () {

    		return new this.constructor().copy( this );

    	},

    	copy: function ( other ) {

    		this.radius = other.radius;
    		this.phi = other.phi;
    		this.theta = other.theta;

    		return this;

    	},

    	// restrict phi to be betwee EPS and PI-EPS
    	makeSafe: function () {

    		var EPS = 0.000001;
    		this.phi = Math.max( EPS, Math.min( Math.PI - EPS, this.phi ) );

    		return this;

    	},

    	setFromVector3: function ( v ) {

    		return this.setFromCartesianCoords( v.x, v.y, v.z );

    	},

    	setFromCartesianCoords: function ( x, y, z ) {

    		this.radius = Math.sqrt( x * x + y * y + z * z );

    		if ( this.radius === 0 ) {

    			this.theta = 0;
    			this.phi = 0;

    		} else {

    			this.theta = Math.atan2( x, z );
    			this.phi = Math.acos( _Math.clamp( y / this.radius, - 1, 1 ) );

    		}

    		return this;

    	}

    } );

    /**
     * @author Mugen87 / https://github.com/Mugen87
     *
     * Ref: https://en.wikipedia.org/wiki/Cylindrical_coordinate_system
     *
     */

    function Cylindrical( radius, theta, y ) {

    	this.radius = ( radius !== undefined ) ? radius : 1.0; // distance from the origin to a point in the x-z plane
    	this.theta = ( theta !== undefined ) ? theta : 0; // counterclockwise angle in the x-z plane measured in radians from the positive z-axis
    	this.y = ( y !== undefined ) ? y : 0; // height above the x-z plane

    	return this;

    }

    Object.assign( Cylindrical.prototype, {

    	set: function ( radius, theta, y ) {

    		this.radius = radius;
    		this.theta = theta;
    		this.y = y;

    		return this;

    	},

    	clone: function () {

    		return new this.constructor().copy( this );

    	},

    	copy: function ( other ) {

    		this.radius = other.radius;
    		this.theta = other.theta;
    		this.y = other.y;

    		return this;

    	},

    	setFromVector3: function ( v ) {

    		return this.setFromCartesianCoords( v.x, v.y, v.z );

    	},

    	setFromCartesianCoords: function ( x, y, z ) {

    		this.radius = Math.sqrt( x * x + z * z );
    		this.theta = Math.atan2( x, z );
    		this.y = y;

    		return this;

    	}

    } );

    /**
     * @author bhouston / http://clara.io
     */

    var _vector$6 = new Vector2();

    function Box2( min, max ) {

    	this.min = ( min !== undefined ) ? min : new Vector2( + Infinity, + Infinity );
    	this.max = ( max !== undefined ) ? max : new Vector2( - Infinity, - Infinity );

    }

    Object.assign( Box2.prototype, {

    	set: function ( min, max ) {

    		this.min.copy( min );
    		this.max.copy( max );

    		return this;

    	},

    	setFromPoints: function ( points ) {

    		this.makeEmpty();

    		for ( var i = 0, il = points.length; i < il; i ++ ) {

    			this.expandByPoint( points[ i ] );

    		}

    		return this;

    	},

    	setFromCenterAndSize: function ( center, size ) {

    		var halfSize = _vector$6.copy( size ).multiplyScalar( 0.5 );
    		this.min.copy( center ).sub( halfSize );
    		this.max.copy( center ).add( halfSize );

    		return this;

    	},

    	clone: function () {

    		return new this.constructor().copy( this );

    	},

    	copy: function ( box ) {

    		this.min.copy( box.min );
    		this.max.copy( box.max );

    		return this;

    	},

    	makeEmpty: function () {

    		this.min.x = this.min.y = + Infinity;
    		this.max.x = this.max.y = - Infinity;

    		return this;

    	},

    	isEmpty: function () {

    		// this is a more robust check for empty than ( volume <= 0 ) because volume can get positive with two negative axes

    		return ( this.max.x < this.min.x ) || ( this.max.y < this.min.y );

    	},

    	getCenter: function ( target ) {

    		if ( target === undefined ) {

    			console.warn( 'THREE.Box2: .getCenter() target is now required' );
    			target = new Vector2();

    		}

    		return this.isEmpty() ? target.set( 0, 0 ) : target.addVectors( this.min, this.max ).multiplyScalar( 0.5 );

    	},

    	getSize: function ( target ) {

    		if ( target === undefined ) {

    			console.warn( 'THREE.Box2: .getSize() target is now required' );
    			target = new Vector2();

    		}

    		return this.isEmpty() ? target.set( 0, 0 ) : target.subVectors( this.max, this.min );

    	},

    	expandByPoint: function ( point ) {

    		this.min.min( point );
    		this.max.max( point );

    		return this;

    	},

    	expandByVector: function ( vector ) {

    		this.min.sub( vector );
    		this.max.add( vector );

    		return this;

    	},

    	expandByScalar: function ( scalar ) {

    		this.min.addScalar( - scalar );
    		this.max.addScalar( scalar );

    		return this;

    	},

    	containsPoint: function ( point ) {

    		return point.x < this.min.x || point.x > this.max.x ||
    			point.y < this.min.y || point.y > this.max.y ? false : true;

    	},

    	containsBox: function ( box ) {

    		return this.min.x <= box.min.x && box.max.x <= this.max.x &&
    			this.min.y <= box.min.y && box.max.y <= this.max.y;

    	},

    	getParameter: function ( point, target ) {

    		// This can potentially have a divide by zero if the box
    		// has a size dimension of 0.

    		if ( target === undefined ) {

    			console.warn( 'THREE.Box2: .getParameter() target is now required' );
    			target = new Vector2();

    		}

    		return target.set(
    			( point.x - this.min.x ) / ( this.max.x - this.min.x ),
    			( point.y - this.min.y ) / ( this.max.y - this.min.y )
    		);

    	},

    	intersectsBox: function ( box ) {

    		// using 4 splitting planes to rule out intersections

    		return box.max.x < this.min.x || box.min.x > this.max.x ||
    			box.max.y < this.min.y || box.min.y > this.max.y ? false : true;

    	},

    	clampPoint: function ( point, target ) {

    		if ( target === undefined ) {

    			console.warn( 'THREE.Box2: .clampPoint() target is now required' );
    			target = new Vector2();

    		}

    		return target.copy( point ).clamp( this.min, this.max );

    	},

    	distanceToPoint: function ( point ) {

    		var clampedPoint = _vector$6.copy( point ).clamp( this.min, this.max );
    		return clampedPoint.sub( point ).length();

    	},

    	intersect: function ( box ) {

    		this.min.max( box.min );
    		this.max.min( box.max );

    		return this;

    	},

    	union: function ( box ) {

    		this.min.min( box.min );
    		this.max.max( box.max );

    		return this;

    	},

    	translate: function ( offset ) {

    		this.min.add( offset );
    		this.max.add( offset );

    		return this;

    	},

    	equals: function ( box ) {

    		return box.min.equals( this.min ) && box.max.equals( this.max );

    	}

    } );

    /**
     * @author bhouston / http://clara.io
     */

    var _startP = new Vector3();
    var _startEnd = new Vector3();

    function Line3( start, end ) {

    	this.start = ( start !== undefined ) ? start : new Vector3();
    	this.end = ( end !== undefined ) ? end : new Vector3();

    }

    Object.assign( Line3.prototype, {

    	set: function ( start, end ) {

    		this.start.copy( start );
    		this.end.copy( end );

    		return this;

    	},

    	clone: function () {

    		return new this.constructor().copy( this );

    	},

    	copy: function ( line ) {

    		this.start.copy( line.start );
    		this.end.copy( line.end );

    		return this;

    	},

    	getCenter: function ( target ) {

    		if ( target === undefined ) {

    			console.warn( 'THREE.Line3: .getCenter() target is now required' );
    			target = new Vector3();

    		}

    		return target.addVectors( this.start, this.end ).multiplyScalar( 0.5 );

    	},

    	delta: function ( target ) {

    		if ( target === undefined ) {

    			console.warn( 'THREE.Line3: .delta() target is now required' );
    			target = new Vector3();

    		}

    		return target.subVectors( this.end, this.start );

    	},

    	distanceSq: function () {

    		return this.start.distanceToSquared( this.end );

    	},

    	distance: function () {

    		return this.start.distanceTo( this.end );

    	},

    	at: function ( t, target ) {

    		if ( target === undefined ) {

    			console.warn( 'THREE.Line3: .at() target is now required' );
    			target = new Vector3();

    		}

    		return this.delta( target ).multiplyScalar( t ).add( this.start );

    	},

    	closestPointToPointParameter: function ( point, clampToLine ) {

    		_startP.subVectors( point, this.start );
    		_startEnd.subVectors( this.end, this.start );

    		var startEnd2 = _startEnd.dot( _startEnd );
    		var startEnd_startP = _startEnd.dot( _startP );

    		var t = startEnd_startP / startEnd2;

    		if ( clampToLine ) {

    			t = _Math.clamp( t, 0, 1 );

    		}

    		return t;

    	},

    	closestPointToPoint: function ( point, clampToLine, target ) {

    		var t = this.closestPointToPointParameter( point, clampToLine );

    		if ( target === undefined ) {

    			console.warn( 'THREE.Line3: .closestPointToPoint() target is now required' );
    			target = new Vector3();

    		}

    		return this.delta( target ).multiplyScalar( t ).add( this.start );

    	},

    	applyMatrix4: function ( matrix ) {

    		this.start.applyMatrix4( matrix );
    		this.end.applyMatrix4( matrix );

    		return this;

    	},

    	equals: function ( line ) {

    		return line.start.equals( this.start ) && line.end.equals( this.end );

    	}

    } );

    /**
     * @author alteredq / http://alteredqualia.com/
     */

    function ImmediateRenderObject( material ) {

    	Object3D.call( this );

    	this.material = material;
    	this.render = function ( /* renderCallback */ ) {};

    }

    ImmediateRenderObject.prototype = Object.create( Object3D.prototype );
    ImmediateRenderObject.prototype.constructor = ImmediateRenderObject;

    ImmediateRenderObject.prototype.isImmediateRenderObject = true;

    /**
     * @author mrdoob / http://mrdoob.com/
     * @author WestLangley / http://github.com/WestLangley
     */

    var _v1$5 = new Vector3();
    var _v2$3 = new Vector3();
    var _normalMatrix$1 = new Matrix3();
    var _keys = [ 'a', 'b', 'c' ];

    function VertexNormalsHelper( object, size, hex, linewidth ) {

    	this.object = object;

    	this.size = ( size !== undefined ) ? size : 1;

    	var color = ( hex !== undefined ) ? hex : 0xff0000;

    	var width = ( linewidth !== undefined ) ? linewidth : 1;

    	//

    	var nNormals = 0;

    	var objGeometry = this.object.geometry;

    	if ( objGeometry && objGeometry.isGeometry ) {

    		nNormals = objGeometry.faces.length * 3;

    	} else if ( objGeometry && objGeometry.isBufferGeometry ) {

    		nNormals = objGeometry.attributes.normal.count;

    	}

    	//

    	var geometry = new BufferGeometry();

    	var positions = new Float32BufferAttribute( nNormals * 2 * 3, 3 );

    	geometry.addAttribute( 'position', positions );

    	LineSegments.call( this, geometry, new LineBasicMaterial( { color: color, linewidth: width } ) );

    	//

    	this.matrixAutoUpdate = false;

    	this.update();

    }

    VertexNormalsHelper.prototype = Object.create( LineSegments.prototype );
    VertexNormalsHelper.prototype.constructor = VertexNormalsHelper;

    VertexNormalsHelper.prototype.update = function () {

    	this.object.updateMatrixWorld( true );

    	_normalMatrix$1.getNormalMatrix( this.object.matrixWorld );

    	var matrixWorld = this.object.matrixWorld;

    	var position = this.geometry.attributes.position;

    	//

    	var objGeometry = this.object.geometry;

    	if ( objGeometry && objGeometry.isGeometry ) {

    		var vertices = objGeometry.vertices;

    		var faces = objGeometry.faces;

    		var idx = 0;

    		for ( var i = 0, l = faces.length; i < l; i ++ ) {

    			var face = faces[ i ];

    			for ( var j = 0, jl = face.vertexNormals.length; j < jl; j ++ ) {

    				var vertex = vertices[ face[ _keys[ j ] ] ];

    				var normal = face.vertexNormals[ j ];

    				_v1$5.copy( vertex ).applyMatrix4( matrixWorld );

    				_v2$3.copy( normal ).applyMatrix3( _normalMatrix$1 ).normalize().multiplyScalar( this.size ).add( _v1$5 );

    				position.setXYZ( idx, _v1$5.x, _v1$5.y, _v1$5.z );

    				idx = idx + 1;

    				position.setXYZ( idx, _v2$3.x, _v2$3.y, _v2$3.z );

    				idx = idx + 1;

    			}

    		}

    	} else if ( objGeometry && objGeometry.isBufferGeometry ) {

    		var objPos = objGeometry.attributes.position;

    		var objNorm = objGeometry.attributes.normal;

    		var idx = 0;

    		// for simplicity, ignore index and drawcalls, and render every normal

    		for ( var j = 0, jl = objPos.count; j < jl; j ++ ) {

    			_v1$5.set( objPos.getX( j ), objPos.getY( j ), objPos.getZ( j ) ).applyMatrix4( matrixWorld );

    			_v2$3.set( objNorm.getX( j ), objNorm.getY( j ), objNorm.getZ( j ) );

    			_v2$3.applyMatrix3( _normalMatrix$1 ).normalize().multiplyScalar( this.size ).add( _v1$5 );

    			position.setXYZ( idx, _v1$5.x, _v1$5.y, _v1$5.z );

    			idx = idx + 1;

    			position.setXYZ( idx, _v2$3.x, _v2$3.y, _v2$3.z );

    			idx = idx + 1;

    		}

    	}

    	position.needsUpdate = true;

    };

    /**
     * @author alteredq / http://alteredqualia.com/
     * @author mrdoob / http://mrdoob.com/
     * @author WestLangley / http://github.com/WestLangley
     */

    var _vector$7 = new Vector3();

    function SpotLightHelper( light, color ) {

    	Object3D.call( this );

    	this.light = light;
    	this.light.updateMatrixWorld();

    	this.matrix = light.matrixWorld;
    	this.matrixAutoUpdate = false;

    	this.color = color;

    	var geometry = new BufferGeometry();

    	var positions = [
    		0, 0, 0, 	0, 0, 1,
    		0, 0, 0, 	1, 0, 1,
    		0, 0, 0,	- 1, 0, 1,
    		0, 0, 0, 	0, 1, 1,
    		0, 0, 0, 	0, - 1, 1
    	];

    	for ( var i = 0, j = 1, l = 32; i < l; i ++, j ++ ) {

    		var p1 = ( i / l ) * Math.PI * 2;
    		var p2 = ( j / l ) * Math.PI * 2;

    		positions.push(
    			Math.cos( p1 ), Math.sin( p1 ), 1,
    			Math.cos( p2 ), Math.sin( p2 ), 1
    		);

    	}

    	geometry.addAttribute( 'position', new Float32BufferAttribute( positions, 3 ) );

    	var material = new LineBasicMaterial( { fog: false } );

    	this.cone = new LineSegments( geometry, material );
    	this.add( this.cone );

    	this.update();

    }

    SpotLightHelper.prototype = Object.create( Object3D.prototype );
    SpotLightHelper.prototype.constructor = SpotLightHelper;

    SpotLightHelper.prototype.dispose = function () {

    	this.cone.geometry.dispose();
    	this.cone.material.dispose();

    };

    SpotLightHelper.prototype.update = function () {

    	this.light.updateMatrixWorld();

    	var coneLength = this.light.distance ? this.light.distance : 1000;
    	var coneWidth = coneLength * Math.tan( this.light.angle );

    	this.cone.scale.set( coneWidth, coneWidth, coneLength );

    	_vector$7.setFromMatrixPosition( this.light.target.matrixWorld );

    	this.cone.lookAt( _vector$7 );

    	if ( this.color !== undefined ) {

    		this.cone.material.color.set( this.color );

    	} else {

    		this.cone.material.color.copy( this.light.color );

    	}

    };

    /**
     * @author Sean Griffin / http://twitter.com/sgrif
     * @author Michael Guerrero / http://realitymeltdown.com
     * @author mrdoob / http://mrdoob.com/
     * @author ikerr / http://verold.com
     * @author Mugen87 / https://github.com/Mugen87
     */

    var _vector$8 = new Vector3();
    var _boneMatrix = new Matrix4();
    var _matrixWorldInv = new Matrix4();

    function getBoneList( object ) {

    	var boneList = [];

    	if ( object && object.isBone ) {

    		boneList.push( object );

    	}

    	for ( var i = 0; i < object.children.length; i ++ ) {

    		boneList.push.apply( boneList, getBoneList( object.children[ i ] ) );

    	}

    	return boneList;

    }

    function SkeletonHelper( object ) {

    	var bones = getBoneList( object );

    	var geometry = new BufferGeometry();

    	var vertices = [];
    	var colors = [];

    	var color1 = new Color$7( 0, 0, 1 );
    	var color2 = new Color$7( 0, 1, 0 );

    	for ( var i = 0; i < bones.length; i ++ ) {

    		var bone = bones[ i ];

    		if ( bone.parent && bone.parent.isBone ) {

    			vertices.push( 0, 0, 0 );
    			vertices.push( 0, 0, 0 );
    			colors.push( color1.r, color1.g, color1.b );
    			colors.push( color2.r, color2.g, color2.b );

    		}

    	}

    	geometry.addAttribute( 'position', new Float32BufferAttribute( vertices, 3 ) );
    	geometry.addAttribute( 'color', new Float32BufferAttribute( colors, 3 ) );

    	var material = new LineBasicMaterial( { vertexColors: VertexColors, depthTest: false, depthWrite: false, transparent: true } );

    	LineSegments.call( this, geometry, material );

    	this.root = object;
    	this.bones = bones;

    	this.matrix = object.matrixWorld;
    	this.matrixAutoUpdate = false;

    }

    SkeletonHelper.prototype = Object.create( LineSegments.prototype );
    SkeletonHelper.prototype.constructor = SkeletonHelper;

    SkeletonHelper.prototype.updateMatrixWorld = function ( force ) {

    	var bones = this.bones;

    	var geometry = this.geometry;
    	var position = geometry.getAttribute( 'position' );

    	_matrixWorldInv.getInverse( this.root.matrixWorld );

    	for ( var i = 0, j = 0; i < bones.length; i ++ ) {

    		var bone = bones[ i ];

    		if ( bone.parent && bone.parent.isBone ) {

    			_boneMatrix.multiplyMatrices( _matrixWorldInv, bone.matrixWorld );
    			_vector$8.setFromMatrixPosition( _boneMatrix );
    			position.setXYZ( j, _vector$8.x, _vector$8.y, _vector$8.z );

    			_boneMatrix.multiplyMatrices( _matrixWorldInv, bone.parent.matrixWorld );
    			_vector$8.setFromMatrixPosition( _boneMatrix );
    			position.setXYZ( j + 1, _vector$8.x, _vector$8.y, _vector$8.z );

    			j += 2;

    		}

    	}

    	geometry.getAttribute( 'position' ).needsUpdate = true;

    	Object3D.prototype.updateMatrixWorld.call( this, force );

    };

    /**
     * @author alteredq / http://alteredqualia.com/
     * @author mrdoob / http://mrdoob.com/
     */

    function PointLightHelper( light, sphereSize, color ) {

    	this.light = light;
    	this.light.updateMatrixWorld();

    	this.color = color;

    	var geometry = new SphereBufferGeometry( sphereSize, 4, 2 );
    	var material = new MeshBasicMaterial( { wireframe: true, fog: false } );

    	Mesh.call( this, geometry, material );

    	this.matrix = this.light.matrixWorld;
    	this.matrixAutoUpdate = false;

    	this.update();


    	/*
    	var distanceGeometry = new THREE.IcosahedronBufferGeometry( 1, 2 );
    	var distanceMaterial = new THREE.MeshBasicMaterial( { color: hexColor, fog: false, wireframe: true, opacity: 0.1, transparent: true } );

    	this.lightSphere = new THREE.Mesh( bulbGeometry, bulbMaterial );
    	this.lightDistance = new THREE.Mesh( distanceGeometry, distanceMaterial );

    	var d = light.distance;

    	if ( d === 0.0 ) {

    		this.lightDistance.visible = false;

    	} else {

    		this.lightDistance.scale.set( d, d, d );

    	}

    	this.add( this.lightDistance );
    	*/

    }

    PointLightHelper.prototype = Object.create( Mesh.prototype );
    PointLightHelper.prototype.constructor = PointLightHelper;

    PointLightHelper.prototype.dispose = function () {

    	this.geometry.dispose();
    	this.material.dispose();

    };

    PointLightHelper.prototype.update = function () {

    	if ( this.color !== undefined ) {

    		this.material.color.set( this.color );

    	} else {

    		this.material.color.copy( this.light.color );

    	}

    	/*
    	var d = this.light.distance;

    	if ( d === 0.0 ) {

    		this.lightDistance.visible = false;

    	} else {

    		this.lightDistance.visible = true;
    		this.lightDistance.scale.set( d, d, d );

    	}
    	*/

    };

    /**
     * @author abelnation / http://github.com/abelnation
     * @author Mugen87 / http://github.com/Mugen87
     * @author WestLangley / http://github.com/WestLangley
     *
     *  This helper must be added as a child of the light
     */

    function RectAreaLightHelper( light, color ) {

    	this.type = 'RectAreaLightHelper';

    	this.light = light;

    	this.color = color; // optional hardwired color for the helper

    	var positions = [ 1, 1, 0, - 1, 1, 0, - 1, - 1, 0, 1, - 1, 0, 1, 1, 0 ];

    	var geometry = new BufferGeometry();
    	geometry.addAttribute( 'position', new Float32BufferAttribute( positions, 3 ) );
    	geometry.computeBoundingSphere();

    	var material = new LineBasicMaterial( { fog: false } );

    	Line.call( this, geometry, material );

    	//

    	var positions2 = [ 1, 1, 0, - 1, 1, 0, - 1, - 1, 0, 1, 1, 0, - 1, - 1, 0, 1, - 1, 0 ];

    	var geometry2 = new BufferGeometry();
    	geometry2.addAttribute( 'position', new Float32BufferAttribute( positions2, 3 ) );
    	geometry2.computeBoundingSphere();

    	this.add( new Mesh( geometry2, new MeshBasicMaterial( { side: BackSide, fog: false } ) ) );

    	this.update();

    }

    RectAreaLightHelper.prototype = Object.create( Line.prototype );
    RectAreaLightHelper.prototype.constructor = RectAreaLightHelper;

    RectAreaLightHelper.prototype.update = function () {

    	this.scale.set( 0.5 * this.light.width, 0.5 * this.light.height, 1 );

    	if ( this.color !== undefined ) {

    		this.material.color.set( this.color );
    		this.children[ 0 ].material.color.set( this.color );

    	} else {

    		this.material.color.copy( this.light.color ).multiplyScalar( this.light.intensity );

    		// prevent hue shift
    		var c = this.material.color;
    		var max = Math.max( c.r, c.g, c.b );
    		if ( max > 1 ) c.multiplyScalar( 1 / max );

    		this.children[ 0 ].material.color.copy( this.material.color );

    	}

    };

    RectAreaLightHelper.prototype.dispose = function () {

    	this.geometry.dispose();
    	this.material.dispose();
    	this.children[ 0 ].geometry.dispose();
    	this.children[ 0 ].material.dispose();

    };

    /**
     * @author alteredq / http://alteredqualia.com/
     * @author mrdoob / http://mrdoob.com/
     * @author Mugen87 / https://github.com/Mugen87
     */

    var _vector$9 = new Vector3();
    var _color1 = new Color$7();
    var _color2 = new Color$7();

    function HemisphereLightHelper( light, size, color ) {

    	Object3D.call( this );

    	this.light = light;
    	this.light.updateMatrixWorld();

    	this.matrix = light.matrixWorld;
    	this.matrixAutoUpdate = false;

    	this.color = color;

    	var geometry = new OctahedronBufferGeometry( size );
    	geometry.rotateY( Math.PI * 0.5 );

    	this.material = new MeshBasicMaterial( { wireframe: true, fog: false } );
    	if ( this.color === undefined ) this.material.vertexColors = VertexColors;

    	var position = geometry.getAttribute( 'position' );
    	var colors = new Float32Array( position.count * 3 );

    	geometry.addAttribute( 'color', new BufferAttribute( colors, 3 ) );

    	this.add( new Mesh( geometry, this.material ) );

    	this.update();

    }

    HemisphereLightHelper.prototype = Object.create( Object3D.prototype );
    HemisphereLightHelper.prototype.constructor = HemisphereLightHelper;

    HemisphereLightHelper.prototype.dispose = function () {

    	this.children[ 0 ].geometry.dispose();
    	this.children[ 0 ].material.dispose();

    };

    HemisphereLightHelper.prototype.update = function () {

    	var mesh = this.children[ 0 ];

    	if ( this.color !== undefined ) {

    		this.material.color.set( this.color );

    	} else {

    		var colors = mesh.geometry.getAttribute( 'color' );

    		_color1.copy( this.light.color );
    		_color2.copy( this.light.groundColor );

    		for ( var i = 0, l = colors.count; i < l; i ++ ) {

    			var color = ( i < ( l / 2 ) ) ? _color1 : _color2;

    			colors.setXYZ( i, color.r, color.g, color.b );

    		}

    		colors.needsUpdate = true;

    	}

    	mesh.lookAt( _vector$9.setFromMatrixPosition( this.light.matrixWorld ).negate() );

    };

    /**
     * @author WestLangley / http://github.com/WestLangley
     */

    function LightProbeHelper( lightProbe, size ) {

    	this.lightProbe = lightProbe;

    	this.size = size;

    	var defines = {};
    	defines[ 'GAMMA_OUTPUT' ] = "";

    	// material
    	var material = new ShaderMaterial( {

    		defines: defines,

    		uniforms: {

    			sh: { value: this.lightProbe.sh.coefficients }, // by reference

    			intensity: { value: this.lightProbe.intensity }

    		},

    		vertexShader: [

    			'varying vec3 vNormal;',

    			'void main() {',

    			'	vNormal = normalize( normalMatrix * normal );',

    			'	gl_Position = projectionMatrix * modelViewMatrix * vec4( position, 1.0 );',

    			'}',

    		].join( '\n' ),

    		fragmentShader: [

    			'#define RECIPROCAL_PI 0.318309886',

    			'vec3 inverseTransformDirection( in vec3 normal, in mat4 matrix ) {',

    			'	// matrix is assumed to be orthogonal',

    			'	return normalize( ( vec4( normal, 0.0 ) * matrix ).xyz );',

    			'}',

    			'vec3 linearToOutput( in vec3 a ) {',

    			'	#ifdef GAMMA_OUTPUT',

    			'		return pow( a, vec3( 1.0 / float( GAMMA_FACTOR ) ) );',

    			'	#else',

    			'		return a;',

    			'	#endif',

    			'}',

    			'// source: https://graphics.stanford.edu/papers/envmap/envmap.pdf',
    			'vec3 shGetIrradianceAt( in vec3 normal, in vec3 shCoefficients[ 9 ] ) {',

    			'	// normal is assumed to have unit length',

    			'	float x = normal.x, y = normal.y, z = normal.z;',

    			'	// band 0',
    			'	vec3 result = shCoefficients[ 0 ] * 0.886227;',

    			'	// band 1',
    			'	result += shCoefficients[ 1 ] * 2.0 * 0.511664 * y;',
    			'	result += shCoefficients[ 2 ] * 2.0 * 0.511664 * z;',
    			'	result += shCoefficients[ 3 ] * 2.0 * 0.511664 * x;',

    			'	// band 2',
    			'	result += shCoefficients[ 4 ] * 2.0 * 0.429043 * x * y;',
    			'	result += shCoefficients[ 5 ] * 2.0 * 0.429043 * y * z;',
    			'	result += shCoefficients[ 6 ] * ( 0.743125 * z * z - 0.247708 );',
    			'	result += shCoefficients[ 7 ] * 2.0 * 0.429043 * x * z;',
    			'	result += shCoefficients[ 8 ] * 0.429043 * ( x * x - y * y );',

    			'	return result;',

    			'}',

    			'uniform vec3 sh[ 9 ]; // sh coefficients',

    			'uniform float intensity; // light probe intensity',

    			'varying vec3 vNormal;',

    			'void main() {',

    			'	vec3 normal = normalize( vNormal );',

    			'	vec3 worldNormal = inverseTransformDirection( normal, viewMatrix );',

    			'	vec3 irradiance = shGetIrradianceAt( worldNormal, sh );',

    			'	vec3 outgoingLight = RECIPROCAL_PI * irradiance * intensity;',

    			'	outgoingLight = linearToOutput( outgoingLight );',

    			'	gl_FragColor = vec4( outgoingLight, 1.0 );',

    			'}'

    		].join( '\n' )

    	} );

    	var geometry = new SphereBufferGeometry( 1, 32, 16 );

    	Mesh.call( this, geometry, material );

    	this.onBeforeRender();

    }

    LightProbeHelper.prototype = Object.create( Mesh.prototype );
    LightProbeHelper.prototype.constructor = LightProbeHelper;

    LightProbeHelper.prototype.dispose = function () {

    	this.geometry.dispose();
    	this.material.dispose();

    };

    LightProbeHelper.prototype.onBeforeRender = function () {

    	this.position.copy( this.lightProbe.position );

    	this.scale.set( 1, 1, 1 ).multiplyScalar( this.size );

    	this.material.uniforms.intensity.value = this.lightProbe.intensity;

    };

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function GridHelper( size, divisions, color1, color2 ) {

    	size = size || 10;
    	divisions = divisions || 10;
    	color1 = new Color$7( color1 !== undefined ? color1 : 0x444444 );
    	color2 = new Color$7( color2 !== undefined ? color2 : 0x888888 );

    	var center = divisions / 2;
    	var step = size / divisions;
    	var halfSize = size / 2;

    	var vertices = [], colors = [];

    	for ( var i = 0, j = 0, k = - halfSize; i <= divisions; i ++, k += step ) {

    		vertices.push( - halfSize, 0, k, halfSize, 0, k );
    		vertices.push( k, 0, - halfSize, k, 0, halfSize );

    		var color = i === center ? color1 : color2;

    		color.toArray( colors, j ); j += 3;
    		color.toArray( colors, j ); j += 3;
    		color.toArray( colors, j ); j += 3;
    		color.toArray( colors, j ); j += 3;

    	}

    	var geometry = new BufferGeometry();
    	geometry.addAttribute( 'position', new Float32BufferAttribute( vertices, 3 ) );
    	geometry.addAttribute( 'color', new Float32BufferAttribute( colors, 3 ) );

    	var material = new LineBasicMaterial( { vertexColors: VertexColors } );

    	LineSegments.call( this, geometry, material );

    }

    GridHelper.prototype = Object.assign( Object.create( LineSegments.prototype ), {

    	constructor: GridHelper,

    	copy: function ( source ) {

    		LineSegments.prototype.copy.call( this, source );

    		this.geometry.copy( source.geometry );
    		this.material.copy( source.material );

    		return this;

    	},

    	clone: function () {

    		return new this.constructor().copy( this );

    	}

    } );

    /**
     * @author mrdoob / http://mrdoob.com/
     * @author Mugen87 / http://github.com/Mugen87
     * @author Hectate / http://www.github.com/Hectate
     */

    function PolarGridHelper( radius, radials, circles, divisions, color1, color2 ) {

    	radius = radius || 10;
    	radials = radials || 16;
    	circles = circles || 8;
    	divisions = divisions || 64;
    	color1 = new Color$7( color1 !== undefined ? color1 : 0x444444 );
    	color2 = new Color$7( color2 !== undefined ? color2 : 0x888888 );

    	var vertices = [];
    	var colors = [];

    	var x, z;
    	var v, i, j, r, color;

    	// create the radials

    	for ( i = 0; i <= radials; i ++ ) {

    		v = ( i / radials ) * ( Math.PI * 2 );

    		x = Math.sin( v ) * radius;
    		z = Math.cos( v ) * radius;

    		vertices.push( 0, 0, 0 );
    		vertices.push( x, 0, z );

    		color = ( i & 1 ) ? color1 : color2;

    		colors.push( color.r, color.g, color.b );
    		colors.push( color.r, color.g, color.b );

    	}

    	// create the circles

    	for ( i = 0; i <= circles; i ++ ) {

    		color = ( i & 1 ) ? color1 : color2;

    		r = radius - ( radius / circles * i );

    		for ( j = 0; j < divisions; j ++ ) {

    			// first vertex

    			v = ( j / divisions ) * ( Math.PI * 2 );

    			x = Math.sin( v ) * r;
    			z = Math.cos( v ) * r;

    			vertices.push( x, 0, z );
    			colors.push( color.r, color.g, color.b );

    			// second vertex

    			v = ( ( j + 1 ) / divisions ) * ( Math.PI * 2 );

    			x = Math.sin( v ) * r;
    			z = Math.cos( v ) * r;

    			vertices.push( x, 0, z );
    			colors.push( color.r, color.g, color.b );

    		}

    	}

    	var geometry = new BufferGeometry();
    	geometry.addAttribute( 'position', new Float32BufferAttribute( vertices, 3 ) );
    	geometry.addAttribute( 'color', new Float32BufferAttribute( colors, 3 ) );

    	var material = new LineBasicMaterial( { vertexColors: VertexColors } );

    	LineSegments.call( this, geometry, material );

    }

    PolarGridHelper.prototype = Object.create( LineSegments.prototype );
    PolarGridHelper.prototype.constructor = PolarGridHelper;

    /**
     * @author Mugen87 / http://github.com/Mugen87
     */

    function PositionalAudioHelper( audio, range, divisionsInnerAngle, divisionsOuterAngle ) {

    	this.audio = audio;
    	this.range = range || 1;
    	this.divisionsInnerAngle = divisionsInnerAngle || 16;
    	this.divisionsOuterAngle = divisionsOuterAngle || 2;

    	var geometry = new BufferGeometry();
    	var divisions = this.divisionsInnerAngle + this.divisionsOuterAngle * 2;
    	var positions = new Float32Array( ( divisions * 3 + 3 ) * 3 );
    	geometry.addAttribute( 'position', new BufferAttribute( positions, 3 ) );

    	var materialInnerAngle = new LineBasicMaterial( { color: 0x00ff00 } );
    	var materialOuterAngle = new LineBasicMaterial( { color: 0xffff00 } );

    	Line.call( this, geometry, [ materialOuterAngle, materialInnerAngle ] );

    	this.update();

    }

    PositionalAudioHelper.prototype = Object.create( Line.prototype );
    PositionalAudioHelper.prototype.constructor = PositionalAudioHelper;

    PositionalAudioHelper.prototype.update = function () {

    	var audio = this.audio;
    	var range = this.range;
    	var divisionsInnerAngle = this.divisionsInnerAngle;
    	var divisionsOuterAngle = this.divisionsOuterAngle;

    	var coneInnerAngle = _Math.degToRad( audio.panner.coneInnerAngle );
    	var coneOuterAngle = _Math.degToRad( audio.panner.coneOuterAngle );

    	var halfConeInnerAngle = coneInnerAngle / 2;
    	var halfConeOuterAngle = coneOuterAngle / 2;

    	var start = 0;
    	var count = 0;
    	var i, stride;

    	var geometry = this.geometry;
    	var positionAttribute = geometry.attributes.position;

    	geometry.clearGroups();

    	//

    	function generateSegment( from, to, divisions, materialIndex ) {

    		var step = ( to - from ) / divisions;

    		positionAttribute.setXYZ( start, 0, 0, 0 );
    		count ++;

    		for ( i = from; i < to; i += step ) {

    			stride = start + count;

    			positionAttribute.setXYZ( stride, Math.sin( i ) * range, 0, Math.cos( i ) * range );
    			positionAttribute.setXYZ( stride + 1, Math.sin( Math.min( i + step, to ) ) * range, 0, Math.cos( Math.min( i + step, to ) ) * range );
    			positionAttribute.setXYZ( stride + 2, 0, 0, 0 );

    			count += 3;

    		}

    		geometry.addGroup( start, count, materialIndex );

    		start += count;
    		count = 0;

    	}

    	//

    	generateSegment( - halfConeOuterAngle, - halfConeInnerAngle, divisionsOuterAngle, 0 );
    	generateSegment( - halfConeInnerAngle, halfConeInnerAngle, divisionsInnerAngle, 1 );
    	generateSegment( halfConeInnerAngle, halfConeOuterAngle, divisionsOuterAngle, 0 );

    	//

    	positionAttribute.needsUpdate = true;

    	if ( coneInnerAngle === coneOuterAngle ) this.material[ 0 ].visible = false;

    };

    PositionalAudioHelper.prototype.dispose = function () {

    	this.geometry.dispose();
    	this.material[ 0 ].dispose();
    	this.material[ 1 ].dispose();

    };

    /**
     * @author mrdoob / http://mrdoob.com/
     * @author WestLangley / http://github.com/WestLangley
     */

    var _v1$6 = new Vector3();
    var _v2$4 = new Vector3();
    var _normalMatrix$2 = new Matrix3();

    function FaceNormalsHelper( object, size, hex, linewidth ) {

    	// FaceNormalsHelper only supports THREE.Geometry

    	this.object = object;

    	this.size = ( size !== undefined ) ? size : 1;

    	var color = ( hex !== undefined ) ? hex : 0xffff00;

    	var width = ( linewidth !== undefined ) ? linewidth : 1;

    	//

    	var nNormals = 0;

    	var objGeometry = this.object.geometry;

    	if ( objGeometry && objGeometry.isGeometry ) {

    		nNormals = objGeometry.faces.length;

    	} else {

    		console.warn( 'THREE.FaceNormalsHelper: only THREE.Geometry is supported. Use THREE.VertexNormalsHelper, instead.' );

    	}

    	//

    	var geometry = new BufferGeometry();

    	var positions = new Float32BufferAttribute( nNormals * 2 * 3, 3 );

    	geometry.addAttribute( 'position', positions );

    	LineSegments.call( this, geometry, new LineBasicMaterial( { color: color, linewidth: width } ) );

    	//

    	this.matrixAutoUpdate = false;
    	this.update();

    }

    FaceNormalsHelper.prototype = Object.create( LineSegments.prototype );
    FaceNormalsHelper.prototype.constructor = FaceNormalsHelper;

    FaceNormalsHelper.prototype.update = function () {

    	this.object.updateMatrixWorld( true );

    	_normalMatrix$2.getNormalMatrix( this.object.matrixWorld );

    	var matrixWorld = this.object.matrixWorld;

    	var position = this.geometry.attributes.position;

    	//

    	var objGeometry = this.object.geometry;

    	var vertices = objGeometry.vertices;

    	var faces = objGeometry.faces;

    	var idx = 0;

    	for ( var i = 0, l = faces.length; i < l; i ++ ) {

    		var face = faces[ i ];

    		var normal = face.normal;

    		_v1$6.copy( vertices[ face.a ] )
    			.add( vertices[ face.b ] )
    			.add( vertices[ face.c ] )
    			.divideScalar( 3 )
    			.applyMatrix4( matrixWorld );

    		_v2$4.copy( normal ).applyMatrix3( _normalMatrix$2 ).normalize().multiplyScalar( this.size ).add( _v1$6 );

    		position.setXYZ( idx, _v1$6.x, _v1$6.y, _v1$6.z );

    		idx = idx + 1;

    		position.setXYZ( idx, _v2$4.x, _v2$4.y, _v2$4.z );

    		idx = idx + 1;

    	}

    	position.needsUpdate = true;

    };

    /**
     * @author alteredq / http://alteredqualia.com/
     * @author mrdoob / http://mrdoob.com/
     * @author WestLangley / http://github.com/WestLangley
     */

    var _v1$7 = new Vector3();
    var _v2$5 = new Vector3();
    var _v3$1 = new Vector3();

    function DirectionalLightHelper( light, size, color ) {

    	Object3D.call( this );

    	this.light = light;
    	this.light.updateMatrixWorld();

    	this.matrix = light.matrixWorld;
    	this.matrixAutoUpdate = false;

    	this.color = color;

    	if ( size === undefined ) size = 1;

    	var geometry = new BufferGeometry();
    	geometry.addAttribute( 'position', new Float32BufferAttribute( [
    		- size, size, 0,
    		size, size, 0,
    		size, - size, 0,
    		- size, - size, 0,
    		- size, size, 0
    	], 3 ) );

    	var material = new LineBasicMaterial( { fog: false } );

    	this.lightPlane = new Line( geometry, material );
    	this.add( this.lightPlane );

    	geometry = new BufferGeometry();
    	geometry.addAttribute( 'position', new Float32BufferAttribute( [ 0, 0, 0, 0, 0, 1 ], 3 ) );

    	this.targetLine = new Line( geometry, material );
    	this.add( this.targetLine );

    	this.update();

    }

    DirectionalLightHelper.prototype = Object.create( Object3D.prototype );
    DirectionalLightHelper.prototype.constructor = DirectionalLightHelper;

    DirectionalLightHelper.prototype.dispose = function () {

    	this.lightPlane.geometry.dispose();
    	this.lightPlane.material.dispose();
    	this.targetLine.geometry.dispose();
    	this.targetLine.material.dispose();

    };

    DirectionalLightHelper.prototype.update = function () {

    	_v1$7.setFromMatrixPosition( this.light.matrixWorld );
    	_v2$5.setFromMatrixPosition( this.light.target.matrixWorld );
    	_v3$1.subVectors( _v2$5, _v1$7 );

    	this.lightPlane.lookAt( _v2$5 );

    	if ( this.color !== undefined ) {

    		this.lightPlane.material.color.set( this.color );
    		this.targetLine.material.color.set( this.color );

    	} else {

    		this.lightPlane.material.color.copy( this.light.color );
    		this.targetLine.material.color.copy( this.light.color );

    	}

    	this.targetLine.lookAt( _v2$5 );
    	this.targetLine.scale.z = _v3$1.length();

    };

    /**
     * @author alteredq / http://alteredqualia.com/
     * @author Mugen87 / https://github.com/Mugen87
     *
     *	- shows frustum, line of sight and up of the camera
     *	- suitable for fast updates
     * 	- based on frustum visualization in lightgl.js shadowmap example
     *		http://evanw.github.com/lightgl.js/tests/shadowmap.html
     */

    var _vector$a = new Vector3();
    var _camera = new Camera();

    function CameraHelper( camera ) {

    	var geometry = new BufferGeometry();
    	var material = new LineBasicMaterial( { color: 0xffffff, vertexColors: FaceColors } );

    	var vertices = [];
    	var colors = [];

    	var pointMap = {};

    	// colors

    	var colorFrustum = new Color$7( 0xffaa00 );
    	var colorCone = new Color$7( 0xff0000 );
    	var colorUp = new Color$7( 0x00aaff );
    	var colorTarget = new Color$7( 0xffffff );
    	var colorCross = new Color$7( 0x333333 );

    	// near

    	addLine( 'n1', 'n2', colorFrustum );
    	addLine( 'n2', 'n4', colorFrustum );
    	addLine( 'n4', 'n3', colorFrustum );
    	addLine( 'n3', 'n1', colorFrustum );

    	// far

    	addLine( 'f1', 'f2', colorFrustum );
    	addLine( 'f2', 'f4', colorFrustum );
    	addLine( 'f4', 'f3', colorFrustum );
    	addLine( 'f3', 'f1', colorFrustum );

    	// sides

    	addLine( 'n1', 'f1', colorFrustum );
    	addLine( 'n2', 'f2', colorFrustum );
    	addLine( 'n3', 'f3', colorFrustum );
    	addLine( 'n4', 'f4', colorFrustum );

    	// cone

    	addLine( 'p', 'n1', colorCone );
    	addLine( 'p', 'n2', colorCone );
    	addLine( 'p', 'n3', colorCone );
    	addLine( 'p', 'n4', colorCone );

    	// up

    	addLine( 'u1', 'u2', colorUp );
    	addLine( 'u2', 'u3', colorUp );
    	addLine( 'u3', 'u1', colorUp );

    	// target

    	addLine( 'c', 't', colorTarget );
    	addLine( 'p', 'c', colorCross );

    	// cross

    	addLine( 'cn1', 'cn2', colorCross );
    	addLine( 'cn3', 'cn4', colorCross );

    	addLine( 'cf1', 'cf2', colorCross );
    	addLine( 'cf3', 'cf4', colorCross );

    	function addLine( a, b, color ) {

    		addPoint( a, color );
    		addPoint( b, color );

    	}

    	function addPoint( id, color ) {

    		vertices.push( 0, 0, 0 );
    		colors.push( color.r, color.g, color.b );

    		if ( pointMap[ id ] === undefined ) {

    			pointMap[ id ] = [];

    		}

    		pointMap[ id ].push( ( vertices.length / 3 ) - 1 );

    	}

    	geometry.addAttribute( 'position', new Float32BufferAttribute( vertices, 3 ) );
    	geometry.addAttribute( 'color', new Float32BufferAttribute( colors, 3 ) );

    	LineSegments.call( this, geometry, material );

    	this.camera = camera;
    	if ( this.camera.updateProjectionMatrix ) this.camera.updateProjectionMatrix();

    	this.matrix = camera.matrixWorld;
    	this.matrixAutoUpdate = false;

    	this.pointMap = pointMap;

    	this.update();

    }

    CameraHelper.prototype = Object.create( LineSegments.prototype );
    CameraHelper.prototype.constructor = CameraHelper;

    CameraHelper.prototype.update = function () {

    	var geometry = this.geometry;
    	var pointMap = this.pointMap;

    	var w = 1, h = 1;

    	// we need just camera projection matrix inverse
    	// world matrix must be identity

    	_camera.projectionMatrixInverse.copy( this.camera.projectionMatrixInverse );

    	// center / target

    	setPoint( 'c', pointMap, geometry, _camera, 0, 0, - 1 );
    	setPoint( 't', pointMap, geometry, _camera, 0, 0, 1 );

    	// near

    	setPoint( 'n1', pointMap, geometry, _camera, - w, - h, - 1 );
    	setPoint( 'n2', pointMap, geometry, _camera, w, - h, - 1 );
    	setPoint( 'n3', pointMap, geometry, _camera, - w, h, - 1 );
    	setPoint( 'n4', pointMap, geometry, _camera, w, h, - 1 );

    	// far

    	setPoint( 'f1', pointMap, geometry, _camera, - w, - h, 1 );
    	setPoint( 'f2', pointMap, geometry, _camera, w, - h, 1 );
    	setPoint( 'f3', pointMap, geometry, _camera, - w, h, 1 );
    	setPoint( 'f4', pointMap, geometry, _camera, w, h, 1 );

    	// up

    	setPoint( 'u1', pointMap, geometry, _camera, w * 0.7, h * 1.1, - 1 );
    	setPoint( 'u2', pointMap, geometry, _camera, - w * 0.7, h * 1.1, - 1 );
    	setPoint( 'u3', pointMap, geometry, _camera, 0, h * 2, - 1 );

    	// cross

    	setPoint( 'cf1', pointMap, geometry, _camera, - w, 0, 1 );
    	setPoint( 'cf2', pointMap, geometry, _camera, w, 0, 1 );
    	setPoint( 'cf3', pointMap, geometry, _camera, 0, - h, 1 );
    	setPoint( 'cf4', pointMap, geometry, _camera, 0, h, 1 );

    	setPoint( 'cn1', pointMap, geometry, _camera, - w, 0, - 1 );
    	setPoint( 'cn2', pointMap, geometry, _camera, w, 0, - 1 );
    	setPoint( 'cn3', pointMap, geometry, _camera, 0, - h, - 1 );
    	setPoint( 'cn4', pointMap, geometry, _camera, 0, h, - 1 );

    	geometry.getAttribute( 'position' ).needsUpdate = true;

    };

    function setPoint( point, pointMap, geometry, camera, x, y, z ) {

    	_vector$a.set( x, y, z ).unproject( camera );

    	var points = pointMap[ point ];

    	if ( points !== undefined ) {

    		var position = geometry.getAttribute( 'position' );

    		for ( var i = 0, l = points.length; i < l; i ++ ) {

    			position.setXYZ( points[ i ], _vector$a.x, _vector$a.y, _vector$a.z );

    		}

    	}

    }

    /**
     * @author mrdoob / http://mrdoob.com/
     * @author Mugen87 / http://github.com/Mugen87
     */

    var _box$2 = new Box3();

    function BoxHelper( object, color ) {

    	this.object = object;

    	if ( color === undefined ) color = 0xffff00;

    	var indices = new Uint16Array( [ 0, 1, 1, 2, 2, 3, 3, 0, 4, 5, 5, 6, 6, 7, 7, 4, 0, 4, 1, 5, 2, 6, 3, 7 ] );
    	var positions = new Float32Array( 8 * 3 );

    	var geometry = new BufferGeometry();
    	geometry.setIndex( new BufferAttribute( indices, 1 ) );
    	geometry.addAttribute( 'position', new BufferAttribute( positions, 3 ) );

    	LineSegments.call( this, geometry, new LineBasicMaterial( { color: color } ) );

    	this.matrixAutoUpdate = false;

    	this.update();

    }

    BoxHelper.prototype = Object.create( LineSegments.prototype );
    BoxHelper.prototype.constructor = BoxHelper;

    BoxHelper.prototype.update = function ( object ) {

    	if ( object !== undefined ) {

    		console.warn( 'THREE.BoxHelper: .update() has no longer arguments.' );

    	}

    	if ( this.object !== undefined ) {

    		_box$2.setFromObject( this.object );

    	}

    	if ( _box$2.isEmpty() ) return;

    	var min = _box$2.min;
    	var max = _box$2.max;

    	/*
    	  5____4
    	1/___0/|
    	| 6__|_7
    	2/___3/

    	0: max.x, max.y, max.z
    	1: min.x, max.y, max.z
    	2: min.x, min.y, max.z
    	3: max.x, min.y, max.z
    	4: max.x, max.y, min.z
    	5: min.x, max.y, min.z
    	6: min.x, min.y, min.z
    	7: max.x, min.y, min.z
    	*/

    	var position = this.geometry.attributes.position;
    	var array = position.array;

    	array[ 0 ] = max.x; array[ 1 ] = max.y; array[ 2 ] = max.z;
    	array[ 3 ] = min.x; array[ 4 ] = max.y; array[ 5 ] = max.z;
    	array[ 6 ] = min.x; array[ 7 ] = min.y; array[ 8 ] = max.z;
    	array[ 9 ] = max.x; array[ 10 ] = min.y; array[ 11 ] = max.z;
    	array[ 12 ] = max.x; array[ 13 ] = max.y; array[ 14 ] = min.z;
    	array[ 15 ] = min.x; array[ 16 ] = max.y; array[ 17 ] = min.z;
    	array[ 18 ] = min.x; array[ 19 ] = min.y; array[ 20 ] = min.z;
    	array[ 21 ] = max.x; array[ 22 ] = min.y; array[ 23 ] = min.z;

    	position.needsUpdate = true;

    	this.geometry.computeBoundingSphere();


    };

    BoxHelper.prototype.setFromObject = function ( object ) {

    	this.object = object;
    	this.update();

    	return this;

    };

    BoxHelper.prototype.copy = function ( source ) {

    	LineSegments.prototype.copy.call( this, source );

    	this.object = source.object;

    	return this;

    };

    BoxHelper.prototype.clone = function () {

    	return new this.constructor().copy( this );

    };

    /**
     * @author WestLangley / http://github.com/WestLangley
     */

    function Box3Helper( box, color ) {

    	this.type = 'Box3Helper';

    	this.box = box;

    	color = color || 0xffff00;

    	var indices = new Uint16Array( [ 0, 1, 1, 2, 2, 3, 3, 0, 4, 5, 5, 6, 6, 7, 7, 4, 0, 4, 1, 5, 2, 6, 3, 7 ] );

    	var positions = [ 1, 1, 1, - 1, 1, 1, - 1, - 1, 1, 1, - 1, 1, 1, 1, - 1, - 1, 1, - 1, - 1, - 1, - 1, 1, - 1, - 1 ];

    	var geometry = new BufferGeometry();

    	geometry.setIndex( new BufferAttribute( indices, 1 ) );

    	geometry.addAttribute( 'position', new Float32BufferAttribute( positions, 3 ) );

    	LineSegments.call( this, geometry, new LineBasicMaterial( { color: color } ) );

    	this.geometry.computeBoundingSphere();

    }

    Box3Helper.prototype = Object.create( LineSegments.prototype );
    Box3Helper.prototype.constructor = Box3Helper;

    Box3Helper.prototype.updateMatrixWorld = function ( force ) {

    	var box = this.box;

    	if ( box.isEmpty() ) return;

    	box.getCenter( this.position );

    	box.getSize( this.scale );

    	this.scale.multiplyScalar( 0.5 );

    	Object3D.prototype.updateMatrixWorld.call( this, force );

    };

    /**
     * @author WestLangley / http://github.com/WestLangley
     */

    function PlaneHelper( plane, size, hex ) {

    	this.type = 'PlaneHelper';

    	this.plane = plane;

    	this.size = ( size === undefined ) ? 1 : size;

    	var color = ( hex !== undefined ) ? hex : 0xffff00;

    	var positions = [ 1, - 1, 1, - 1, 1, 1, - 1, - 1, 1, 1, 1, 1, - 1, 1, 1, - 1, - 1, 1, 1, - 1, 1, 1, 1, 1, 0, 0, 1, 0, 0, 0 ];

    	var geometry = new BufferGeometry();
    	geometry.addAttribute( 'position', new Float32BufferAttribute( positions, 3 ) );
    	geometry.computeBoundingSphere();

    	Line.call( this, geometry, new LineBasicMaterial( { color: color } ) );

    	//

    	var positions2 = [ 1, 1, 1, - 1, 1, 1, - 1, - 1, 1, 1, 1, 1, - 1, - 1, 1, 1, - 1, 1 ];

    	var geometry2 = new BufferGeometry();
    	geometry2.addAttribute( 'position', new Float32BufferAttribute( positions2, 3 ) );
    	geometry2.computeBoundingSphere();

    	this.add( new Mesh( geometry2, new MeshBasicMaterial( { color: color, opacity: 0.2, transparent: true, depthWrite: false } ) ) );

    }

    PlaneHelper.prototype = Object.create( Line.prototype );
    PlaneHelper.prototype.constructor = PlaneHelper;

    PlaneHelper.prototype.updateMatrixWorld = function ( force ) {

    	var scale = - this.plane.constant;

    	if ( Math.abs( scale ) < 1e-8 ) scale = 1e-8; // sign does not matter

    	this.scale.set( 0.5 * this.size, 0.5 * this.size, scale );

    	this.children[ 0 ].material.side = ( scale < 0 ) ? BackSide : FrontSide; // renderer flips side when determinant < 0; flipping not wanted here

    	this.lookAt( this.plane.normal );

    	Object3D.prototype.updateMatrixWorld.call( this, force );

    };

    /**
     * @author WestLangley / http://github.com/WestLangley
     * @author zz85 / http://github.com/zz85
     * @author bhouston / http://clara.io
     *
     * Creates an arrow for visualizing directions
     *
     * Parameters:
     *  dir - Vector3
     *  origin - Vector3
     *  length - Number
     *  color - color in hex value
     *  headLength - Number
     *  headWidth - Number
     */

    var _axis = new Vector3();
    var _lineGeometry, _coneGeometry;

    function ArrowHelper( dir, origin, length, color, headLength, headWidth ) {

    	// dir is assumed to be normalized

    	Object3D.call( this );

    	if ( dir === undefined ) dir = new Vector3( 0, 0, 1 );
    	if ( origin === undefined ) origin = new Vector3( 0, 0, 0 );
    	if ( length === undefined ) length = 1;
    	if ( color === undefined ) color = 0xffff00;
    	if ( headLength === undefined ) headLength = 0.2 * length;
    	if ( headWidth === undefined ) headWidth = 0.2 * headLength;

    	if ( _lineGeometry === undefined ) {

    		_lineGeometry = new BufferGeometry();
    		_lineGeometry.addAttribute( 'position', new Float32BufferAttribute( [ 0, 0, 0, 0, 1, 0 ], 3 ) );

    		_coneGeometry = new CylinderBufferGeometry( 0, 0.5, 1, 5, 1 );
    		_coneGeometry.translate( 0, - 0.5, 0 );

    	}

    	this.position.copy( origin );

    	this.line = new Line( _lineGeometry, new LineBasicMaterial( { color: color } ) );
    	this.line.matrixAutoUpdate = false;
    	this.add( this.line );

    	this.cone = new Mesh( _coneGeometry, new MeshBasicMaterial( { color: color } ) );
    	this.cone.matrixAutoUpdate = false;
    	this.add( this.cone );

    	this.setDirection( dir );
    	this.setLength( length, headLength, headWidth );

    }

    ArrowHelper.prototype = Object.create( Object3D.prototype );
    ArrowHelper.prototype.constructor = ArrowHelper;

    ArrowHelper.prototype.setDirection = function ( dir ) {

    	// dir is assumed to be normalized

    	if ( dir.y > 0.99999 ) {

    		this.quaternion.set( 0, 0, 0, 1 );

    	} else if ( dir.y < - 0.99999 ) {

    		this.quaternion.set( 1, 0, 0, 0 );

    	} else {

    		_axis.set( dir.z, 0, - dir.x ).normalize();

    		var radians = Math.acos( dir.y );

    		this.quaternion.setFromAxisAngle( _axis, radians );

    	}

    };

    ArrowHelper.prototype.setLength = function ( length, headLength, headWidth ) {

    	if ( headLength === undefined ) headLength = 0.2 * length;
    	if ( headWidth === undefined ) headWidth = 0.2 * headLength;

    	this.line.scale.set( 1, Math.max( 0, length - headLength ), 1 );
    	this.line.updateMatrix();

    	this.cone.scale.set( headWidth, headLength, headWidth );
    	this.cone.position.y = length;
    	this.cone.updateMatrix();

    };

    ArrowHelper.prototype.setColor = function ( color ) {

    	this.line.material.color.set( color );
    	this.cone.material.color.set( color );

    };

    ArrowHelper.prototype.copy = function ( source ) {

    	Object3D.prototype.copy.call( this, source, false );

    	this.line.copy( source.line );
    	this.cone.copy( source.cone );

    	return this;

    };

    ArrowHelper.prototype.clone = function () {

    	return new this.constructor().copy( this );

    };

    /**
     * @author sroucheray / http://sroucheray.org/
     * @author mrdoob / http://mrdoob.com/
     */

    function AxesHelper( size ) {

    	size = size || 1;

    	var vertices = [
    		0, 0, 0,	size, 0, 0,
    		0, 0, 0,	0, size, 0,
    		0, 0, 0,	0, 0, size
    	];

    	var colors = [
    		1, 0, 0,	1, 0.6, 0,
    		0, 1, 0,	0.6, 1, 0,
    		0, 0, 1,	0, 0.6, 1
    	];

    	var geometry = new BufferGeometry();
    	geometry.addAttribute( 'position', new Float32BufferAttribute( vertices, 3 ) );
    	geometry.addAttribute( 'color', new Float32BufferAttribute( colors, 3 ) );

    	var material = new LineBasicMaterial( { vertexColors: VertexColors } );

    	LineSegments.call( this, geometry, material );

    }

    AxesHelper.prototype = Object.create( LineSegments.prototype );
    AxesHelper.prototype.constructor = AxesHelper;

    /**
     * @author mrdoob / http://mrdoob.com/
     */

    function Face4( a, b, c, d, normal, color, materialIndex ) {

    	console.warn( 'THREE.Face4 has been removed. A THREE.Face3 will be created instead.' );
    	return new Face3( a, b, c, normal, color, materialIndex );

    }

    var LineStrip = 0;

    var LinePieces = 1;

    function MeshFaceMaterial( materials ) {

    	console.warn( 'THREE.MeshFaceMaterial has been removed. Use an Array instead.' );
    	return materials;

    }

    function MultiMaterial( materials ) {

    	if ( materials === undefined ) materials = [];

    	console.warn( 'THREE.MultiMaterial has been removed. Use an Array instead.' );
    	materials.isMultiMaterial = true;
    	materials.materials = materials;
    	materials.clone = function () {

    		return materials.slice();

    	};
    	return materials;

    }

    function PointCloud( geometry, material ) {

    	console.warn( 'THREE.PointCloud has been renamed to THREE.Points.' );
    	return new Points( geometry, material );

    }

    function Particle( material ) {

    	console.warn( 'THREE.Particle has been renamed to THREE.Sprite.' );
    	return new Sprite( material );

    }

    function ParticleSystem( geometry, material ) {

    	console.warn( 'THREE.ParticleSystem has been renamed to THREE.Points.' );
    	return new Points( geometry, material );

    }

    function PointCloudMaterial( parameters ) {

    	console.warn( 'THREE.PointCloudMaterial has been renamed to THREE.PointsMaterial.' );
    	return new PointsMaterial( parameters );

    }

    function ParticleBasicMaterial( parameters ) {

    	console.warn( 'THREE.ParticleBasicMaterial has been renamed to THREE.PointsMaterial.' );
    	return new PointsMaterial( parameters );

    }

    function ParticleSystemMaterial( parameters ) {

    	console.warn( 'THREE.ParticleSystemMaterial has been renamed to THREE.PointsMaterial.' );
    	return new PointsMaterial( parameters );

    }

    function Vertex( x, y, z ) {

    	console.warn( 'THREE.Vertex has been removed. Use THREE.Vector3 instead.' );
    	return new Vector3( x, y, z );

    }

    //

    function DynamicBufferAttribute( array, itemSize ) {

    	console.warn( 'THREE.DynamicBufferAttribute has been removed. Use new THREE.BufferAttribute().setDynamic( true ) instead.' );
    	return new BufferAttribute( array, itemSize ).setDynamic( true );

    }

    function Int8Attribute( array, itemSize ) {

    	console.warn( 'THREE.Int8Attribute has been removed. Use new THREE.Int8BufferAttribute() instead.' );
    	return new Int8BufferAttribute( array, itemSize );

    }

    function Uint8Attribute( array, itemSize ) {

    	console.warn( 'THREE.Uint8Attribute has been removed. Use new THREE.Uint8BufferAttribute() instead.' );
    	return new Uint8BufferAttribute( array, itemSize );

    }

    function Uint8ClampedAttribute( array, itemSize ) {

    	console.warn( 'THREE.Uint8ClampedAttribute has been removed. Use new THREE.Uint8ClampedBufferAttribute() instead.' );
    	return new Uint8ClampedBufferAttribute( array, itemSize );

    }

    function Int16Attribute( array, itemSize ) {

    	console.warn( 'THREE.Int16Attribute has been removed. Use new THREE.Int16BufferAttribute() instead.' );
    	return new Int16BufferAttribute( array, itemSize );

    }

    function Uint16Attribute( array, itemSize ) {

    	console.warn( 'THREE.Uint16Attribute has been removed. Use new THREE.Uint16BufferAttribute() instead.' );
    	return new Uint16BufferAttribute( array, itemSize );

    }

    function Int32Attribute( array, itemSize ) {

    	console.warn( 'THREE.Int32Attribute has been removed. Use new THREE.Int32BufferAttribute() instead.' );
    	return new Int32BufferAttribute( array, itemSize );

    }

    function Uint32Attribute( array, itemSize ) {

    	console.warn( 'THREE.Uint32Attribute has been removed. Use new THREE.Uint32BufferAttribute() instead.' );
    	return new Uint32BufferAttribute( array, itemSize );

    }

    function Float32Attribute( array, itemSize ) {

    	console.warn( 'THREE.Float32Attribute has been removed. Use new THREE.Float32BufferAttribute() instead.' );
    	return new Float32BufferAttribute( array, itemSize );

    }

    function Float64Attribute( array, itemSize ) {

    	console.warn( 'THREE.Float64Attribute has been removed. Use new THREE.Float64BufferAttribute() instead.' );
    	return new Float64BufferAttribute( array, itemSize );

    }

    //

    Curve.create = function ( construct, getPoint ) {

    	console.log( 'THREE.Curve.create() has been deprecated' );

    	construct.prototype = Object.create( Curve.prototype );
    	construct.prototype.constructor = construct;
    	construct.prototype.getPoint = getPoint;

    	return construct;

    };

    //

    Object.assign( CurvePath.prototype, {

    	createPointsGeometry: function ( divisions ) {

    		console.warn( 'THREE.CurvePath: .createPointsGeometry() has been removed. Use new THREE.Geometry().setFromPoints( points ) instead.' );

    		// generate geometry from path points (for Line or Points objects)

    		var pts = this.getPoints( divisions );
    		return this.createGeometry( pts );

    	},

    	createSpacedPointsGeometry: function ( divisions ) {

    		console.warn( 'THREE.CurvePath: .createSpacedPointsGeometry() has been removed. Use new THREE.Geometry().setFromPoints( points ) instead.' );

    		// generate geometry from equidistant sampling along the path

    		var pts = this.getSpacedPoints( divisions );
    		return this.createGeometry( pts );

    	},

    	createGeometry: function ( points ) {

    		console.warn( 'THREE.CurvePath: .createGeometry() has been removed. Use new THREE.Geometry().setFromPoints( points ) instead.' );

    		var geometry = new Geometry();

    		for ( var i = 0, l = points.length; i < l; i ++ ) {

    			var point = points[ i ];
    			geometry.vertices.push( new Vector3( point.x, point.y, point.z || 0 ) );

    		}

    		return geometry;

    	}

    } );

    //

    Object.assign( Path$4.prototype, {

    	fromPoints: function ( points ) {

    		console.warn( 'THREE.Path: .fromPoints() has been renamed to .setFromPoints().' );
    		this.setFromPoints( points );

    	}

    } );

    //

    function ClosedSplineCurve3( points ) {

    	console.warn( 'THREE.ClosedSplineCurve3 has been deprecated. Use THREE.CatmullRomCurve3 instead.' );

    	CatmullRomCurve3.call( this, points );
    	this.type = 'catmullrom';
    	this.closed = true;

    }

    ClosedSplineCurve3.prototype = Object.create( CatmullRomCurve3.prototype );

    //

    function SplineCurve3( points ) {

    	console.warn( 'THREE.SplineCurve3 has been deprecated. Use THREE.CatmullRomCurve3 instead.' );

    	CatmullRomCurve3.call( this, points );
    	this.type = 'catmullrom';

    }

    SplineCurve3.prototype = Object.create( CatmullRomCurve3.prototype );

    //

    function Spline( points ) {

    	console.warn( 'THREE.Spline has been removed. Use THREE.CatmullRomCurve3 instead.' );

    	CatmullRomCurve3.call( this, points );
    	this.type = 'catmullrom';

    }

    Spline.prototype = Object.create( CatmullRomCurve3.prototype );

    Object.assign( Spline.prototype, {

    	initFromArray: function ( /* a */ ) {

    		console.error( 'THREE.Spline: .initFromArray() has been removed.' );

    	},
    	getControlPointsArray: function ( /* optionalTarget */ ) {

    		console.error( 'THREE.Spline: .getControlPointsArray() has been removed.' );

    	},
    	reparametrizeByArcLength: function ( /* samplingCoef */ ) {

    		console.error( 'THREE.Spline: .reparametrizeByArcLength() has been removed.' );

    	}

    } );

    //

    function AxisHelper( size ) {

    	console.warn( 'THREE.AxisHelper has been renamed to THREE.AxesHelper.' );
    	return new AxesHelper( size );

    }

    function BoundingBoxHelper( object, color ) {

    	console.warn( 'THREE.BoundingBoxHelper has been deprecated. Creating a THREE.BoxHelper instead.' );
    	return new BoxHelper( object, color );

    }

    function EdgesHelper( object, hex ) {

    	console.warn( 'THREE.EdgesHelper has been removed. Use THREE.EdgesGeometry instead.' );
    	return new LineSegments( new EdgesGeometry( object.geometry ), new LineBasicMaterial( { color: hex !== undefined ? hex : 0xffffff } ) );

    }

    GridHelper.prototype.setColors = function () {

    	console.error( 'THREE.GridHelper: setColors() has been deprecated, pass them in the constructor instead.' );

    };

    SkeletonHelper.prototype.update = function () {

    	console.error( 'THREE.SkeletonHelper: update() no longer needs to be called.' );

    };

    function WireframeHelper( object, hex ) {

    	console.warn( 'THREE.WireframeHelper has been removed. Use THREE.WireframeGeometry instead.' );
    	return new LineSegments( new WireframeGeometry( object.geometry ), new LineBasicMaterial( { color: hex !== undefined ? hex : 0xffffff } ) );

    }

    //

    Object.assign( Loader.prototype, {

    	extractUrlBase: function ( url ) {

    		console.warn( 'THREE.Loader: .extractUrlBase() has been deprecated. Use THREE.LoaderUtils.extractUrlBase() instead.' );
    		return LoaderUtils.extractUrlBase( url );

    	}

    } );

    function XHRLoader( manager ) {

    	console.warn( 'THREE.XHRLoader has been renamed to THREE.FileLoader.' );
    	return new FileLoader( manager );

    }

    function BinaryTextureLoader( manager ) {

    	console.warn( 'THREE.BinaryTextureLoader has been renamed to THREE.DataTextureLoader.' );
    	return new DataTextureLoader( manager );

    }

    Object.assign( ObjectLoader.prototype, {

    	setTexturePath: function ( value ) {

    		console.warn( 'THREE.ObjectLoader: .setTexturePath() has been renamed to .setResourcePath().' );
    		return this.setResourcePath( value );

    	}

    } );

    //

    Object.assign( Box2.prototype, {

    	center: function ( optionalTarget ) {

    		console.warn( 'THREE.Box2: .center() has been renamed to .getCenter().' );
    		return this.getCenter( optionalTarget );

    	},
    	empty: function () {

    		console.warn( 'THREE.Box2: .empty() has been renamed to .isEmpty().' );
    		return this.isEmpty();

    	},
    	isIntersectionBox: function ( box ) {

    		console.warn( 'THREE.Box2: .isIntersectionBox() has been renamed to .intersectsBox().' );
    		return this.intersectsBox( box );

    	},
    	size: function ( optionalTarget ) {

    		console.warn( 'THREE.Box2: .size() has been renamed to .getSize().' );
    		return this.getSize( optionalTarget );

    	}
    } );

    Object.assign( Box3.prototype, {

    	center: function ( optionalTarget ) {

    		console.warn( 'THREE.Box3: .center() has been renamed to .getCenter().' );
    		return this.getCenter( optionalTarget );

    	},
    	empty: function () {

    		console.warn( 'THREE.Box3: .empty() has been renamed to .isEmpty().' );
    		return this.isEmpty();

    	},
    	isIntersectionBox: function ( box ) {

    		console.warn( 'THREE.Box3: .isIntersectionBox() has been renamed to .intersectsBox().' );
    		return this.intersectsBox( box );

    	},
    	isIntersectionSphere: function ( sphere ) {

    		console.warn( 'THREE.Box3: .isIntersectionSphere() has been renamed to .intersectsSphere().' );
    		return this.intersectsSphere( sphere );

    	},
    	size: function ( optionalTarget ) {

    		console.warn( 'THREE.Box3: .size() has been renamed to .getSize().' );
    		return this.getSize( optionalTarget );

    	}
    } );

    Line3.prototype.center = function ( optionalTarget ) {

    	console.warn( 'THREE.Line3: .center() has been renamed to .getCenter().' );
    	return this.getCenter( optionalTarget );

    };

    Object.assign( _Math, {

    	random16: function () {

    		console.warn( 'THREE.Math: .random16() has been deprecated. Use Math.random() instead.' );
    		return Math.random();

    	},

    	nearestPowerOfTwo: function ( value ) {

    		console.warn( 'THREE.Math: .nearestPowerOfTwo() has been renamed to .floorPowerOfTwo().' );
    		return _Math.floorPowerOfTwo( value );

    	},

    	nextPowerOfTwo: function ( value ) {

    		console.warn( 'THREE.Math: .nextPowerOfTwo() has been renamed to .ceilPowerOfTwo().' );
    		return _Math.ceilPowerOfTwo( value );

    	}

    } );

    Object.assign( Matrix3.prototype, {

    	flattenToArrayOffset: function ( array, offset ) {

    		console.warn( "THREE.Matrix3: .flattenToArrayOffset() has been deprecated. Use .toArray() instead." );
    		return this.toArray( array, offset );

    	},
    	multiplyVector3: function ( vector ) {

    		console.warn( 'THREE.Matrix3: .multiplyVector3() has been removed. Use vector.applyMatrix3( matrix ) instead.' );
    		return vector.applyMatrix3( this );

    	},
    	multiplyVector3Array: function ( /* a */ ) {

    		console.error( 'THREE.Matrix3: .multiplyVector3Array() has been removed.' );

    	},
    	applyToBuffer: function ( buffer /*, offset, length */ ) {

    		console.warn( 'THREE.Matrix3: .applyToBuffer() has been removed. Use matrix.applyToBufferAttribute( attribute ) instead.' );
    		return this.applyToBufferAttribute( buffer );

    	},
    	applyToVector3Array: function ( /* array, offset, length */ ) {

    		console.error( 'THREE.Matrix3: .applyToVector3Array() has been removed.' );

    	}

    } );

    Object.assign( Matrix4.prototype, {

    	extractPosition: function ( m ) {

    		console.warn( 'THREE.Matrix4: .extractPosition() has been renamed to .copyPosition().' );
    		return this.copyPosition( m );

    	},
    	flattenToArrayOffset: function ( array, offset ) {

    		console.warn( "THREE.Matrix4: .flattenToArrayOffset() has been deprecated. Use .toArray() instead." );
    		return this.toArray( array, offset );

    	},
    	getPosition: function () {

    		console.warn( 'THREE.Matrix4: .getPosition() has been removed. Use Vector3.setFromMatrixPosition( matrix ) instead.' );
    		return new Vector3().setFromMatrixColumn( this, 3 );

    	},
    	setRotationFromQuaternion: function ( q ) {

    		console.warn( 'THREE.Matrix4: .setRotationFromQuaternion() has been renamed to .makeRotationFromQuaternion().' );
    		return this.makeRotationFromQuaternion( q );

    	},
    	multiplyToArray: function () {

    		console.warn( 'THREE.Matrix4: .multiplyToArray() has been removed.' );

    	},
    	multiplyVector3: function ( vector ) {

    		console.warn( 'THREE.Matrix4: .multiplyVector3() has been removed. Use vector.applyMatrix4( matrix ) instead.' );
    		return vector.applyMatrix4( this );

    	},
    	multiplyVector4: function ( vector ) {

    		console.warn( 'THREE.Matrix4: .multiplyVector4() has been removed. Use vector.applyMatrix4( matrix ) instead.' );
    		return vector.applyMatrix4( this );

    	},
    	multiplyVector3Array: function ( /* a */ ) {

    		console.error( 'THREE.Matrix4: .multiplyVector3Array() has been removed.' );

    	},
    	rotateAxis: function ( v ) {

    		console.warn( 'THREE.Matrix4: .rotateAxis() has been removed. Use Vector3.transformDirection( matrix ) instead.' );
    		v.transformDirection( this );

    	},
    	crossVector: function ( vector ) {

    		console.warn( 'THREE.Matrix4: .crossVector() has been removed. Use vector.applyMatrix4( matrix ) instead.' );
    		return vector.applyMatrix4( this );

    	},
    	translate: function () {

    		console.error( 'THREE.Matrix4: .translate() has been removed.' );

    	},
    	rotateX: function () {

    		console.error( 'THREE.Matrix4: .rotateX() has been removed.' );

    	},
    	rotateY: function () {

    		console.error( 'THREE.Matrix4: .rotateY() has been removed.' );

    	},
    	rotateZ: function () {

    		console.error( 'THREE.Matrix4: .rotateZ() has been removed.' );

    	},
    	rotateByAxis: function () {

    		console.error( 'THREE.Matrix4: .rotateByAxis() has been removed.' );

    	},
    	applyToBuffer: function ( buffer /*, offset, length */ ) {

    		console.warn( 'THREE.Matrix4: .applyToBuffer() has been removed. Use matrix.applyToBufferAttribute( attribute ) instead.' );
    		return this.applyToBufferAttribute( buffer );

    	},
    	applyToVector3Array: function ( /* array, offset, length */ ) {

    		console.error( 'THREE.Matrix4: .applyToVector3Array() has been removed.' );

    	},
    	makeFrustum: function ( left, right, bottom, top, near, far ) {

    		console.warn( 'THREE.Matrix4: .makeFrustum() has been removed. Use .makePerspective( left, right, top, bottom, near, far ) instead.' );
    		return this.makePerspective( left, right, top, bottom, near, far );

    	}

    } );

    Plane.prototype.isIntersectionLine = function ( line ) {

    	console.warn( 'THREE.Plane: .isIntersectionLine() has been renamed to .intersectsLine().' );
    	return this.intersectsLine( line );

    };

    Quaternion.prototype.multiplyVector3 = function ( vector ) {

    	console.warn( 'THREE.Quaternion: .multiplyVector3() has been removed. Use is now vector.applyQuaternion( quaternion ) instead.' );
    	return vector.applyQuaternion( this );

    };

    Object.assign( Ray.prototype, {

    	isIntersectionBox: function ( box ) {

    		console.warn( 'THREE.Ray: .isIntersectionBox() has been renamed to .intersectsBox().' );
    		return this.intersectsBox( box );

    	},
    	isIntersectionPlane: function ( plane ) {

    		console.warn( 'THREE.Ray: .isIntersectionPlane() has been renamed to .intersectsPlane().' );
    		return this.intersectsPlane( plane );

    	},
    	isIntersectionSphere: function ( sphere ) {

    		console.warn( 'THREE.Ray: .isIntersectionSphere() has been renamed to .intersectsSphere().' );
    		return this.intersectsSphere( sphere );

    	}

    } );

    Object.assign( Triangle.prototype, {

    	area: function () {

    		console.warn( 'THREE.Triangle: .area() has been renamed to .getArea().' );
    		return this.getArea();

    	},
    	barycoordFromPoint: function ( point, target ) {

    		console.warn( 'THREE.Triangle: .barycoordFromPoint() has been renamed to .getBarycoord().' );
    		return this.getBarycoord( point, target );

    	},
    	midpoint: function ( target ) {

    		console.warn( 'THREE.Triangle: .midpoint() has been renamed to .getMidpoint().' );
    		return this.getMidpoint( target );

    	},
    	normal: function ( target ) {

    		console.warn( 'THREE.Triangle: .normal() has been renamed to .getNormal().' );
    		return this.getNormal( target );

    	},
    	plane: function ( target ) {

    		console.warn( 'THREE.Triangle: .plane() has been renamed to .getPlane().' );
    		return this.getPlane( target );

    	}

    } );

    Object.assign( Triangle, {

    	barycoordFromPoint: function ( point, a, b, c, target ) {

    		console.warn( 'THREE.Triangle: .barycoordFromPoint() has been renamed to .getBarycoord().' );
    		return Triangle.getBarycoord( point, a, b, c, target );

    	},
    	normal: function ( a, b, c, target ) {

    		console.warn( 'THREE.Triangle: .normal() has been renamed to .getNormal().' );
    		return Triangle.getNormal( a, b, c, target );

    	}

    } );

    Object.assign( Shape.prototype, {

    	extractAllPoints: function ( divisions ) {

    		console.warn( 'THREE.Shape: .extractAllPoints() has been removed. Use .extractPoints() instead.' );
    		return this.extractPoints( divisions );

    	},
    	extrude: function ( options ) {

    		console.warn( 'THREE.Shape: .extrude() has been removed. Use ExtrudeGeometry() instead.' );
    		return new ExtrudeGeometry( this, options );

    	},
    	makeGeometry: function ( options ) {

    		console.warn( 'THREE.Shape: .makeGeometry() has been removed. Use ShapeGeometry() instead.' );
    		return new ShapeGeometry( this, options );

    	}

    } );

    Object.assign( Vector2.prototype, {

    	fromAttribute: function ( attribute, index, offset ) {

    		console.warn( 'THREE.Vector2: .fromAttribute() has been renamed to .fromBufferAttribute().' );
    		return this.fromBufferAttribute( attribute, index, offset );

    	},
    	distanceToManhattan: function ( v ) {

    		console.warn( 'THREE.Vector2: .distanceToManhattan() has been renamed to .manhattanDistanceTo().' );
    		return this.manhattanDistanceTo( v );

    	},
    	lengthManhattan: function () {

    		console.warn( 'THREE.Vector2: .lengthManhattan() has been renamed to .manhattanLength().' );
    		return this.manhattanLength();

    	}

    } );

    Object.assign( Vector3.prototype, {

    	setEulerFromRotationMatrix: function () {

    		console.error( 'THREE.Vector3: .setEulerFromRotationMatrix() has been removed. Use Euler.setFromRotationMatrix() instead.' );

    	},
    	setEulerFromQuaternion: function () {

    		console.error( 'THREE.Vector3: .setEulerFromQuaternion() has been removed. Use Euler.setFromQuaternion() instead.' );

    	},
    	getPositionFromMatrix: function ( m ) {

    		console.warn( 'THREE.Vector3: .getPositionFromMatrix() has been renamed to .setFromMatrixPosition().' );
    		return this.setFromMatrixPosition( m );

    	},
    	getScaleFromMatrix: function ( m ) {

    		console.warn( 'THREE.Vector3: .getScaleFromMatrix() has been renamed to .setFromMatrixScale().' );
    		return this.setFromMatrixScale( m );

    	},
    	getColumnFromMatrix: function ( index, matrix ) {

    		console.warn( 'THREE.Vector3: .getColumnFromMatrix() has been renamed to .setFromMatrixColumn().' );
    		return this.setFromMatrixColumn( matrix, index );

    	},
    	applyProjection: function ( m ) {

    		console.warn( 'THREE.Vector3: .applyProjection() has been removed. Use .applyMatrix4( m ) instead.' );
    		return this.applyMatrix4( m );

    	},
    	fromAttribute: function ( attribute, index, offset ) {

    		console.warn( 'THREE.Vector3: .fromAttribute() has been renamed to .fromBufferAttribute().' );
    		return this.fromBufferAttribute( attribute, index, offset );

    	},
    	distanceToManhattan: function ( v ) {

    		console.warn( 'THREE.Vector3: .distanceToManhattan() has been renamed to .manhattanDistanceTo().' );
    		return this.manhattanDistanceTo( v );

    	},
    	lengthManhattan: function () {

    		console.warn( 'THREE.Vector3: .lengthManhattan() has been renamed to .manhattanLength().' );
    		return this.manhattanLength();

    	}

    } );

    Object.assign( Vector4.prototype, {

    	fromAttribute: function ( attribute, index, offset ) {

    		console.warn( 'THREE.Vector4: .fromAttribute() has been renamed to .fromBufferAttribute().' );
    		return this.fromBufferAttribute( attribute, index, offset );

    	},
    	lengthManhattan: function () {

    		console.warn( 'THREE.Vector4: .lengthManhattan() has been renamed to .manhattanLength().' );
    		return this.manhattanLength();

    	}

    } );

    //

    Object.assign( Geometry.prototype, {

    	computeTangents: function () {

    		console.error( 'THREE.Geometry: .computeTangents() has been removed.' );

    	},
    	computeLineDistances: function () {

    		console.error( 'THREE.Geometry: .computeLineDistances() has been removed. Use THREE.Line.computeLineDistances() instead.' );

    	}

    } );

    Object.assign( Object3D.prototype, {

    	getChildByName: function ( name ) {

    		console.warn( 'THREE.Object3D: .getChildByName() has been renamed to .getObjectByName().' );
    		return this.getObjectByName( name );

    	},
    	renderDepth: function () {

    		console.warn( 'THREE.Object3D: .renderDepth has been removed. Use .renderOrder, instead.' );

    	},
    	translate: function ( distance, axis ) {

    		console.warn( 'THREE.Object3D: .translate() has been removed. Use .translateOnAxis( axis, distance ) instead.' );
    		return this.translateOnAxis( axis, distance );

    	},
    	getWorldRotation: function () {

    		console.error( 'THREE.Object3D: .getWorldRotation() has been removed. Use THREE.Object3D.getWorldQuaternion( target ) instead.' );

    	}

    } );

    Object.defineProperties( Object3D.prototype, {

    	eulerOrder: {
    		get: function () {

    			console.warn( 'THREE.Object3D: .eulerOrder is now .rotation.order.' );
    			return this.rotation.order;

    		},
    		set: function ( value ) {

    			console.warn( 'THREE.Object3D: .eulerOrder is now .rotation.order.' );
    			this.rotation.order = value;

    		}
    	},
    	useQuaternion: {
    		get: function () {

    			console.warn( 'THREE.Object3D: .useQuaternion has been removed. The library now uses quaternions by default.' );

    		},
    		set: function () {

    			console.warn( 'THREE.Object3D: .useQuaternion has been removed. The library now uses quaternions by default.' );

    		}
    	}

    } );

    Object.defineProperties( LOD.prototype, {

    	objects: {
    		get: function () {

    			console.warn( 'THREE.LOD: .objects has been renamed to .levels.' );
    			return this.levels;

    		}
    	}

    } );

    Object.defineProperty( Skeleton.prototype, 'useVertexTexture', {

    	get: function () {

    		console.warn( 'THREE.Skeleton: useVertexTexture has been removed.' );

    	},
    	set: function () {

    		console.warn( 'THREE.Skeleton: useVertexTexture has been removed.' );

    	}

    } );

    SkinnedMesh.prototype.initBones = function () {

    	console.error( 'THREE.SkinnedMesh: initBones() has been removed.' );

    };

    Object.defineProperty( Curve.prototype, '__arcLengthDivisions', {

    	get: function () {

    		console.warn( 'THREE.Curve: .__arcLengthDivisions is now .arcLengthDivisions.' );
    		return this.arcLengthDivisions;

    	},
    	set: function ( value ) {

    		console.warn( 'THREE.Curve: .__arcLengthDivisions is now .arcLengthDivisions.' );
    		this.arcLengthDivisions = value;

    	}

    } );

    //

    PerspectiveCamera.prototype.setLens = function ( focalLength, filmGauge ) {

    	console.warn( "THREE.PerspectiveCamera.setLens is deprecated. " +
    			"Use .setFocalLength and .filmGauge for a photographic setup." );

    	if ( filmGauge !== undefined ) this.filmGauge = filmGauge;
    	this.setFocalLength( focalLength );

    };

    //

    Object.defineProperties( Light.prototype, {
    	onlyShadow: {
    		set: function () {

    			console.warn( 'THREE.Light: .onlyShadow has been removed.' );

    		}
    	},
    	shadowCameraFov: {
    		set: function ( value ) {

    			console.warn( 'THREE.Light: .shadowCameraFov is now .shadow.camera.fov.' );
    			this.shadow.camera.fov = value;

    		}
    	},
    	shadowCameraLeft: {
    		set: function ( value ) {

    			console.warn( 'THREE.Light: .shadowCameraLeft is now .shadow.camera.left.' );
    			this.shadow.camera.left = value;

    		}
    	},
    	shadowCameraRight: {
    		set: function ( value ) {

    			console.warn( 'THREE.Light: .shadowCameraRight is now .shadow.camera.right.' );
    			this.shadow.camera.right = value;

    		}
    	},
    	shadowCameraTop: {
    		set: function ( value ) {

    			console.warn( 'THREE.Light: .shadowCameraTop is now .shadow.camera.top.' );
    			this.shadow.camera.top = value;

    		}
    	},
    	shadowCameraBottom: {
    		set: function ( value ) {

    			console.warn( 'THREE.Light: .shadowCameraBottom is now .shadow.camera.bottom.' );
    			this.shadow.camera.bottom = value;

    		}
    	},
    	shadowCameraNear: {
    		set: function ( value ) {

    			console.warn( 'THREE.Light: .shadowCameraNear is now .shadow.camera.near.' );
    			this.shadow.camera.near = value;

    		}
    	},
    	shadowCameraFar: {
    		set: function ( value ) {

    			console.warn( 'THREE.Light: .shadowCameraFar is now .shadow.camera.far.' );
    			this.shadow.camera.far = value;

    		}
    	},
    	shadowCameraVisible: {
    		set: function () {

    			console.warn( 'THREE.Light: .shadowCameraVisible has been removed. Use new THREE.CameraHelper( light.shadow.camera ) instead.' );

    		}
    	},
    	shadowBias: {
    		set: function ( value ) {

    			console.warn( 'THREE.Light: .shadowBias is now .shadow.bias.' );
    			this.shadow.bias = value;

    		}
    	},
    	shadowDarkness: {
    		set: function () {

    			console.warn( 'THREE.Light: .shadowDarkness has been removed.' );

    		}
    	},
    	shadowMapWidth: {
    		set: function ( value ) {

    			console.warn( 'THREE.Light: .shadowMapWidth is now .shadow.mapSize.width.' );
    			this.shadow.mapSize.width = value;

    		}
    	},
    	shadowMapHeight: {
    		set: function ( value ) {

    			console.warn( 'THREE.Light: .shadowMapHeight is now .shadow.mapSize.height.' );
    			this.shadow.mapSize.height = value;

    		}
    	}
    } );

    //

    Object.defineProperties( BufferAttribute.prototype, {

    	length: {
    		get: function () {

    			console.warn( 'THREE.BufferAttribute: .length has been deprecated. Use .count instead.' );
    			return this.array.length;

    		}
    	},
    	copyIndicesArray: function ( /* indices */ ) {

    		console.error( 'THREE.BufferAttribute: .copyIndicesArray() has been removed.' );

    	}

    } );

    Object.assign( BufferGeometry.prototype, {

    	addIndex: function ( index ) {

    		console.warn( 'THREE.BufferGeometry: .addIndex() has been renamed to .setIndex().' );
    		this.setIndex( index );

    	},
    	addDrawCall: function ( start, count, indexOffset ) {

    		if ( indexOffset !== undefined ) {

    			console.warn( 'THREE.BufferGeometry: .addDrawCall() no longer supports indexOffset.' );

    		}
    		console.warn( 'THREE.BufferGeometry: .addDrawCall() is now .addGroup().' );
    		this.addGroup( start, count );

    	},
    	clearDrawCalls: function () {

    		console.warn( 'THREE.BufferGeometry: .clearDrawCalls() is now .clearGroups().' );
    		this.clearGroups();

    	},
    	computeTangents: function () {

    		console.warn( 'THREE.BufferGeometry: .computeTangents() has been removed.' );

    	},
    	computeOffsets: function () {

    		console.warn( 'THREE.BufferGeometry: .computeOffsets() has been removed.' );

    	}

    } );

    Object.defineProperties( BufferGeometry.prototype, {

    	drawcalls: {
    		get: function () {

    			console.error( 'THREE.BufferGeometry: .drawcalls has been renamed to .groups.' );
    			return this.groups;

    		}
    	},
    	offsets: {
    		get: function () {

    			console.warn( 'THREE.BufferGeometry: .offsets has been renamed to .groups.' );
    			return this.groups;

    		}
    	}

    } );

    //

    Object.assign( ExtrudeBufferGeometry.prototype, {

    	getArrays: function () {

    		console.error( 'THREE.ExtrudeBufferGeometry: .getArrays() has been removed.' );

    	},

    	addShapeList: function () {

    		console.error( 'THREE.ExtrudeBufferGeometry: .addShapeList() has been removed.' );

    	},

    	addShape: function () {

    		console.error( 'THREE.ExtrudeBufferGeometry: .addShape() has been removed.' );

    	}

    } );

    //

    Object.defineProperties( Uniform.prototype, {

    	dynamic: {
    		set: function () {

    			console.warn( 'THREE.Uniform: .dynamic has been removed. Use object.onBeforeRender() instead.' );

    		}
    	},
    	onUpdate: {
    		value: function () {

    			console.warn( 'THREE.Uniform: .onUpdate() has been removed. Use object.onBeforeRender() instead.' );
    			return this;

    		}
    	}

    } );

    //

    Object.defineProperties( Material.prototype, {

    	wrapAround: {
    		get: function () {

    			console.warn( 'THREE.Material: .wrapAround has been removed.' );

    		},
    		set: function () {

    			console.warn( 'THREE.Material: .wrapAround has been removed.' );

    		}
    	},

    	overdraw: {
    		get: function () {

    			console.warn( 'THREE.Material: .overdraw has been removed.' );

    		},
    		set: function () {

    			console.warn( 'THREE.Material: .overdraw has been removed.' );

    		}
    	},

    	wrapRGB: {
    		get: function () {

    			console.warn( 'THREE.Material: .wrapRGB has been removed.' );
    			return new Color$7();

    		}
    	},

    	shading: {
    		get: function () {

    			console.error( 'THREE.' + this.type + ': .shading has been removed. Use the boolean .flatShading instead.' );

    		},
    		set: function ( value ) {

    			console.warn( 'THREE.' + this.type + ': .shading has been removed. Use the boolean .flatShading instead.' );
    			this.flatShading = ( value === FlatShading );

    		}
    	}

    } );

    Object.defineProperties( MeshPhongMaterial.prototype, {

    	metal: {
    		get: function () {

    			console.warn( 'THREE.MeshPhongMaterial: .metal has been removed. Use THREE.MeshStandardMaterial instead.' );
    			return false;

    		},
    		set: function () {

    			console.warn( 'THREE.MeshPhongMaterial: .metal has been removed. Use THREE.MeshStandardMaterial instead' );

    		}
    	}

    } );

    Object.defineProperties( ShaderMaterial.prototype, {

    	derivatives: {
    		get: function () {

    			console.warn( 'THREE.ShaderMaterial: .derivatives has been moved to .extensions.derivatives.' );
    			return this.extensions.derivatives;

    		},
    		set: function ( value ) {

    			console.warn( 'THREE. ShaderMaterial: .derivatives has been moved to .extensions.derivatives.' );
    			this.extensions.derivatives = value;

    		}
    	}

    } );

    //

    Object.assign( WebGLRenderer.prototype, {

    	clearTarget: function ( renderTarget, color, depth, stencil ) {

    		console.warn( 'THREE.WebGLRenderer: .clearTarget() has been deprecated. Use .setRenderTarget() and .clear() instead.' );
    		this.setRenderTarget( renderTarget );
    		this.clear( color, depth, stencil );

    	},
    	animate: function ( callback ) {

    		console.warn( 'THREE.WebGLRenderer: .animate() is now .setAnimationLoop().' );
    		this.setAnimationLoop( callback );

    	},
    	getCurrentRenderTarget: function () {

    		console.warn( 'THREE.WebGLRenderer: .getCurrentRenderTarget() is now .getRenderTarget().' );
    		return this.getRenderTarget();

    	},
    	getMaxAnisotropy: function () {

    		console.warn( 'THREE.WebGLRenderer: .getMaxAnisotropy() is now .capabilities.getMaxAnisotropy().' );
    		return this.capabilities.getMaxAnisotropy();

    	},
    	getPrecision: function () {

    		console.warn( 'THREE.WebGLRenderer: .getPrecision() is now .capabilities.precision.' );
    		return this.capabilities.precision;

    	},
    	resetGLState: function () {

    		console.warn( 'THREE.WebGLRenderer: .resetGLState() is now .state.reset().' );
    		return this.state.reset();

    	},
    	supportsFloatTextures: function () {

    		console.warn( 'THREE.WebGLRenderer: .supportsFloatTextures() is now .extensions.get( \'OES_texture_float\' ).' );
    		return this.extensions.get( 'OES_texture_float' );

    	},
    	supportsHalfFloatTextures: function () {

    		console.warn( 'THREE.WebGLRenderer: .supportsHalfFloatTextures() is now .extensions.get( \'OES_texture_half_float\' ).' );
    		return this.extensions.get( 'OES_texture_half_float' );

    	},
    	supportsStandardDerivatives: function () {

    		console.warn( 'THREE.WebGLRenderer: .supportsStandardDerivatives() is now .extensions.get( \'OES_standard_derivatives\' ).' );
    		return this.extensions.get( 'OES_standard_derivatives' );

    	},
    	supportsCompressedTextureS3TC: function () {

    		console.warn( 'THREE.WebGLRenderer: .supportsCompressedTextureS3TC() is now .extensions.get( \'WEBGL_compressed_texture_s3tc\' ).' );
    		return this.extensions.get( 'WEBGL_compressed_texture_s3tc' );

    	},
    	supportsCompressedTexturePVRTC: function () {

    		console.warn( 'THREE.WebGLRenderer: .supportsCompressedTexturePVRTC() is now .extensions.get( \'WEBGL_compressed_texture_pvrtc\' ).' );
    		return this.extensions.get( 'WEBGL_compressed_texture_pvrtc' );

    	},
    	supportsBlendMinMax: function () {

    		console.warn( 'THREE.WebGLRenderer: .supportsBlendMinMax() is now .extensions.get( \'EXT_blend_minmax\' ).' );
    		return this.extensions.get( 'EXT_blend_minmax' );

    	},
    	supportsVertexTextures: function () {

    		console.warn( 'THREE.WebGLRenderer: .supportsVertexTextures() is now .capabilities.vertexTextures.' );
    		return this.capabilities.vertexTextures;

    	},
    	supportsInstancedArrays: function () {

    		console.warn( 'THREE.WebGLRenderer: .supportsInstancedArrays() is now .extensions.get( \'ANGLE_instanced_arrays\' ).' );
    		return this.extensions.get( 'ANGLE_instanced_arrays' );

    	},
    	enableScissorTest: function ( boolean ) {

    		console.warn( 'THREE.WebGLRenderer: .enableScissorTest() is now .setScissorTest().' );
    		this.setScissorTest( boolean );

    	},
    	initMaterial: function () {

    		console.warn( 'THREE.WebGLRenderer: .initMaterial() has been removed.' );

    	},
    	addPrePlugin: function () {

    		console.warn( 'THREE.WebGLRenderer: .addPrePlugin() has been removed.' );

    	},
    	addPostPlugin: function () {

    		console.warn( 'THREE.WebGLRenderer: .addPostPlugin() has been removed.' );

    	},
    	updateShadowMap: function () {

    		console.warn( 'THREE.WebGLRenderer: .updateShadowMap() has been removed.' );

    	},
    	setFaceCulling: function () {

    		console.warn( 'THREE.WebGLRenderer: .setFaceCulling() has been removed.' );

    	},
    	allocTextureUnit: function () {

    		console.warn( 'THREE.WebGLRenderer: .allocTextureUnit() has been removed.' );

    	},
    	setTexture: function () {

    		console.warn( 'THREE.WebGLRenderer: .setTexture() has been removed.' );

    	},
    	setTexture2D: function () {

    		console.warn( 'THREE.WebGLRenderer: .setTexture2D() has been removed.' );

    	},
    	setTextureCube: function () {

    		console.warn( 'THREE.WebGLRenderer: .setTextureCube() has been removed.' );

    	},
    	getActiveMipMapLevel: function () {

    		console.warn( 'THREE.WebGLRenderer: .getActiveMipMapLevel() is now .getActiveMipmapLevel().' );
    		return this.getActiveMipmapLevel();

    	}

    } );

    Object.defineProperties( WebGLRenderer.prototype, {

    	shadowMapEnabled: {
    		get: function () {

    			return this.shadowMap.enabled;

    		},
    		set: function ( value ) {

    			console.warn( 'THREE.WebGLRenderer: .shadowMapEnabled is now .shadowMap.enabled.' );
    			this.shadowMap.enabled = value;

    		}
    	},
    	shadowMapType: {
    		get: function () {

    			return this.shadowMap.type;

    		},
    		set: function ( value ) {

    			console.warn( 'THREE.WebGLRenderer: .shadowMapType is now .shadowMap.type.' );
    			this.shadowMap.type = value;

    		}
    	},
    	shadowMapCullFace: {
    		get: function () {

    			console.warn( 'THREE.WebGLRenderer: .shadowMapCullFace has been removed. Set Material.shadowSide instead.' );
    			return undefined;

    		},
    		set: function ( /* value */ ) {

    			console.warn( 'THREE.WebGLRenderer: .shadowMapCullFace has been removed. Set Material.shadowSide instead.' );

    		}
    	},
    	context: {
    		get: function () {

    			console.warn( 'THREE.WebGLRenderer: .context has been removed. Use .getContext() instead.' );
    			return this.getContext();

    		}
    	}

    } );

    Object.defineProperties( WebGLShadowMap.prototype, {

    	cullFace: {
    		get: function () {

    			console.warn( 'THREE.WebGLRenderer: .shadowMap.cullFace has been removed. Set Material.shadowSide instead.' );
    			return undefined;

    		},
    		set: function ( /* cullFace */ ) {

    			console.warn( 'THREE.WebGLRenderer: .shadowMap.cullFace has been removed. Set Material.shadowSide instead.' );

    		}
    	},
    	renderReverseSided: {
    		get: function () {

    			console.warn( 'THREE.WebGLRenderer: .shadowMap.renderReverseSided has been removed. Set Material.shadowSide instead.' );
    			return undefined;

    		},
    		set: function () {

    			console.warn( 'THREE.WebGLRenderer: .shadowMap.renderReverseSided has been removed. Set Material.shadowSide instead.' );

    		}
    	},
    	renderSingleSided: {
    		get: function () {

    			console.warn( 'THREE.WebGLRenderer: .shadowMap.renderSingleSided has been removed. Set Material.shadowSide instead.' );
    			return undefined;

    		},
    		set: function () {

    			console.warn( 'THREE.WebGLRenderer: .shadowMap.renderSingleSided has been removed. Set Material.shadowSide instead.' );

    		}
    	}

    } );

    //

    Object.defineProperties( WebGLRenderTargetCube.prototype, {

    	activeCubeFace: {
    		set: function ( /* value */ ) {

    			console.warn( 'THREE.WebGLRenderTargetCube: .activeCubeFace has been removed. It is now the second parameter of WebGLRenderer.setRenderTarget().' );

    		}
    	},
    	activeMipMapLevel: {
    		set: function ( /* value */ ) {

    			console.warn( 'THREE.WebGLRenderTargetCube: .activeMipMapLevel has been removed. It is now the third parameter of WebGLRenderer.setRenderTarget().' );

    		}
    	}

    } );

    //

    Object.defineProperties( WebGLRenderTarget.prototype, {

    	wrapS: {
    		get: function () {

    			console.warn( 'THREE.WebGLRenderTarget: .wrapS is now .texture.wrapS.' );
    			return this.texture.wrapS;

    		},
    		set: function ( value ) {

    			console.warn( 'THREE.WebGLRenderTarget: .wrapS is now .texture.wrapS.' );
    			this.texture.wrapS = value;

    		}
    	},
    	wrapT: {
    		get: function () {

    			console.warn( 'THREE.WebGLRenderTarget: .wrapT is now .texture.wrapT.' );
    			return this.texture.wrapT;

    		},
    		set: function ( value ) {

    			console.warn( 'THREE.WebGLRenderTarget: .wrapT is now .texture.wrapT.' );
    			this.texture.wrapT = value;

    		}
    	},
    	magFilter: {
    		get: function () {

    			console.warn( 'THREE.WebGLRenderTarget: .magFilter is now .texture.magFilter.' );
    			return this.texture.magFilter;

    		},
    		set: function ( value ) {

    			console.warn( 'THREE.WebGLRenderTarget: .magFilter is now .texture.magFilter.' );
    			this.texture.magFilter = value;

    		}
    	},
    	minFilter: {
    		get: function () {

    			console.warn( 'THREE.WebGLRenderTarget: .minFilter is now .texture.minFilter.' );
    			return this.texture.minFilter;

    		},
    		set: function ( value ) {

    			console.warn( 'THREE.WebGLRenderTarget: .minFilter is now .texture.minFilter.' );
    			this.texture.minFilter = value;

    		}
    	},
    	anisotropy: {
    		get: function () {

    			console.warn( 'THREE.WebGLRenderTarget: .anisotropy is now .texture.anisotropy.' );
    			return this.texture.anisotropy;

    		},
    		set: function ( value ) {

    			console.warn( 'THREE.WebGLRenderTarget: .anisotropy is now .texture.anisotropy.' );
    			this.texture.anisotropy = value;

    		}
    	},
    	offset: {
    		get: function () {

    			console.warn( 'THREE.WebGLRenderTarget: .offset is now .texture.offset.' );
    			return this.texture.offset;

    		},
    		set: function ( value ) {

    			console.warn( 'THREE.WebGLRenderTarget: .offset is now .texture.offset.' );
    			this.texture.offset = value;

    		}
    	},
    	repeat: {
    		get: function () {

    			console.warn( 'THREE.WebGLRenderTarget: .repeat is now .texture.repeat.' );
    			return this.texture.repeat;

    		},
    		set: function ( value ) {

    			console.warn( 'THREE.WebGLRenderTarget: .repeat is now .texture.repeat.' );
    			this.texture.repeat = value;

    		}
    	},
    	format: {
    		get: function () {

    			console.warn( 'THREE.WebGLRenderTarget: .format is now .texture.format.' );
    			return this.texture.format;

    		},
    		set: function ( value ) {

    			console.warn( 'THREE.WebGLRenderTarget: .format is now .texture.format.' );
    			this.texture.format = value;

    		}
    	},
    	type: {
    		get: function () {

    			console.warn( 'THREE.WebGLRenderTarget: .type is now .texture.type.' );
    			return this.texture.type;

    		},
    		set: function ( value ) {

    			console.warn( 'THREE.WebGLRenderTarget: .type is now .texture.type.' );
    			this.texture.type = value;

    		}
    	},
    	generateMipmaps: {
    		get: function () {

    			console.warn( 'THREE.WebGLRenderTarget: .generateMipmaps is now .texture.generateMipmaps.' );
    			return this.texture.generateMipmaps;

    		},
    		set: function ( value ) {

    			console.warn( 'THREE.WebGLRenderTarget: .generateMipmaps is now .texture.generateMipmaps.' );
    			this.texture.generateMipmaps = value;

    		}
    	}

    } );

    //

    Object.defineProperties( WebVRManager.prototype, {

    	standing: {
    		set: function ( /* value */ ) {

    			console.warn( 'THREE.WebVRManager: .standing has been removed.' );

    		}
    	},
    	userHeight: {
    		set: function ( /* value */ ) {

    			console.warn( 'THREE.WebVRManager: .userHeight has been removed.' );

    		}
    	}

    } );

    //

    Audio.prototype.load = function ( file ) {

    	console.warn( 'THREE.Audio: .load has been deprecated. Use THREE.AudioLoader instead.' );
    	var scope = this;
    	var audioLoader = new AudioLoader();
    	audioLoader.load( file, function ( buffer ) {

    		scope.setBuffer( buffer );

    	} );
    	return this;

    };

    AudioAnalyser.prototype.getData = function () {

    	console.warn( 'THREE.AudioAnalyser: .getData() is now .getFrequencyData().' );
    	return this.getFrequencyData();

    };

    //

    CubeCamera.prototype.updateCubeMap = function ( renderer, scene ) {

    	console.warn( 'THREE.CubeCamera: .updateCubeMap() is now .update().' );
    	return this.update( renderer, scene );

    };

    //

    var GeometryUtils = {

    	merge: function ( geometry1, geometry2, materialIndexOffset ) {

    		console.warn( 'THREE.GeometryUtils: .merge() has been moved to Geometry. Use geometry.merge( geometry2, matrix, materialIndexOffset ) instead.' );
    		var matrix;

    		if ( geometry2.isMesh ) {

    			geometry2.matrixAutoUpdate && geometry2.updateMatrix();

    			matrix = geometry2.matrix;
    			geometry2 = geometry2.geometry;

    		}

    		geometry1.merge( geometry2, matrix, materialIndexOffset );

    	},

    	center: function ( geometry ) {

    		console.warn( 'THREE.GeometryUtils: .center() has been moved to Geometry. Use geometry.center() instead.' );
    		return geometry.center();

    	}

    };

    ImageUtils.crossOrigin = undefined;

    ImageUtils.loadTexture = function ( url, mapping, onLoad, onError ) {

    	console.warn( 'THREE.ImageUtils.loadTexture has been deprecated. Use THREE.TextureLoader() instead.' );

    	var loader = new TextureLoader();
    	loader.setCrossOrigin( this.crossOrigin );

    	var texture = loader.load( url, onLoad, undefined, onError );

    	if ( mapping ) texture.mapping = mapping;

    	return texture;

    };

    ImageUtils.loadTextureCube = function ( urls, mapping, onLoad, onError ) {

    	console.warn( 'THREE.ImageUtils.loadTextureCube has been deprecated. Use THREE.CubeTextureLoader() instead.' );

    	var loader = new CubeTextureLoader();
    	loader.setCrossOrigin( this.crossOrigin );

    	var texture = loader.load( urls, onLoad, undefined, onError );

    	if ( mapping ) texture.mapping = mapping;

    	return texture;

    };

    ImageUtils.loadCompressedTexture = function () {

    	console.error( 'THREE.ImageUtils.loadCompressedTexture has been removed. Use THREE.DDSLoader instead.' );

    };

    ImageUtils.loadCompressedTextureCube = function () {

    	console.error( 'THREE.ImageUtils.loadCompressedTextureCube has been removed. Use THREE.DDSLoader instead.' );

    };

    //

    function CanvasRenderer() {

    	console.error( 'THREE.CanvasRenderer has been removed' );

    }

    //

    function JSONLoader() {

    	console.error( 'THREE.JSONLoader has been removed.' );

    }

    //

    var SceneUtils = {

    	createMultiMaterialObject: function ( /* geometry, materials */ ) {

    		console.error( 'THREE.SceneUtils has been moved to /examples/js/utils/SceneUtils.js' );

    	},

    	detach: function ( /* child, parent, scene */ ) {

    		console.error( 'THREE.SceneUtils has been moved to /examples/js/utils/SceneUtils.js' );

    	},

    	attach: function ( /* child, scene, parent */ ) {

    		console.error( 'THREE.SceneUtils has been moved to /examples/js/utils/SceneUtils.js' );

    	}

    };

    //

    function LensFlare() {

    	console.error( 'THREE.LensFlare has been moved to /examples/js/objects/Lensflare.js' );

    }

    var THREE$1 = /*#__PURE__*/Object.freeze({
        __proto__: null,
        ACESFilmicToneMapping: ACESFilmicToneMapping,
        AddEquation: AddEquation,
        AddOperation: AddOperation,
        AdditiveBlending: AdditiveBlending,
        AlphaFormat: AlphaFormat,
        AlwaysDepth: AlwaysDepth,
        AlwaysStencilFunc: AlwaysStencilFunc,
        AmbientLight: AmbientLight,
        AmbientLightProbe: AmbientLightProbe,
        AnimationClip: AnimationClip,
        AnimationLoader: AnimationLoader,
        AnimationMixer: AnimationMixer,
        AnimationObjectGroup: AnimationObjectGroup,
        AnimationUtils: AnimationUtils,
        ArcCurve: ArcCurve,
        ArrayCamera: ArrayCamera,
        ArrowHelper: ArrowHelper,
        Audio: Audio,
        AudioAnalyser: AudioAnalyser,
        AudioContext: AudioContext,
        AudioListener: AudioListener,
        AudioLoader: AudioLoader,
        AxesHelper: AxesHelper,
        AxisHelper: AxisHelper,
        BackSide: BackSide,
        BasicDepthPacking: BasicDepthPacking,
        BasicShadowMap: BasicShadowMap,
        BinaryTextureLoader: BinaryTextureLoader,
        Bone: Bone,
        BooleanKeyframeTrack: BooleanKeyframeTrack,
        BoundingBoxHelper: BoundingBoxHelper,
        Box2: Box2,
        Box3: Box3,
        Box3Helper: Box3Helper,
        BoxBufferGeometry: BoxBufferGeometry,
        BoxGeometry: BoxGeometry,
        BoxHelper: BoxHelper,
        BufferAttribute: BufferAttribute,
        BufferGeometry: BufferGeometry,
        BufferGeometryLoader: BufferGeometryLoader,
        ByteType: ByteType,
        Cache: Cache,
        Camera: Camera,
        CameraHelper: CameraHelper,
        CanvasRenderer: CanvasRenderer,
        CanvasTexture: CanvasTexture,
        CatmullRomCurve3: CatmullRomCurve3,
        CineonToneMapping: CineonToneMapping,
        CircleBufferGeometry: CircleBufferGeometry,
        CircleGeometry: CircleGeometry,
        ClampToEdgeWrapping: ClampToEdgeWrapping,
        Clock: Clock,
        ClosedSplineCurve3: ClosedSplineCurve3,
        Color: Color$7,
        ColorKeyframeTrack: ColorKeyframeTrack,
        CompressedTexture: CompressedTexture,
        CompressedTextureLoader: CompressedTextureLoader,
        ConeBufferGeometry: ConeBufferGeometry,
        ConeGeometry: ConeGeometry,
        CubeCamera: CubeCamera,
        CubeGeometry: BoxGeometry,
        CubeReflectionMapping: CubeReflectionMapping,
        CubeRefractionMapping: CubeRefractionMapping,
        CubeTexture: CubeTexture,
        CubeTextureLoader: CubeTextureLoader,
        CubeUVReflectionMapping: CubeUVReflectionMapping,
        CubeUVRefractionMapping: CubeUVRefractionMapping,
        CubicBezierCurve: CubicBezierCurve,
        CubicBezierCurve3: CubicBezierCurve3,
        CubicInterpolant: CubicInterpolant,
        CullFaceBack: CullFaceBack,
        CullFaceFront: CullFaceFront,
        CullFaceFrontBack: CullFaceFrontBack,
        CullFaceNone: CullFaceNone,
        Curve: Curve,
        CurvePath: CurvePath,
        CustomBlending: CustomBlending,
        CylinderBufferGeometry: CylinderBufferGeometry,
        CylinderGeometry: CylinderGeometry,
        Cylindrical: Cylindrical,
        DataTexture: DataTexture,
        DataTexture2DArray: DataTexture2DArray,
        DataTexture3D: DataTexture3D,
        DataTextureLoader: DataTextureLoader,
        DecrementStencilOp: DecrementStencilOp,
        DecrementWrapStencilOp: DecrementWrapStencilOp,
        DefaultLoadingManager: DefaultLoadingManager,
        DepthFormat: DepthFormat,
        DepthStencilFormat: DepthStencilFormat,
        DepthTexture: DepthTexture,
        DirectionalLight: DirectionalLight,
        DirectionalLightHelper: DirectionalLightHelper,
        DirectionalLightShadow: DirectionalLightShadow,
        DiscreteInterpolant: DiscreteInterpolant,
        DodecahedronBufferGeometry: DodecahedronBufferGeometry,
        DodecahedronGeometry: DodecahedronGeometry,
        DoubleSide: DoubleSide,
        DstAlphaFactor: DstAlphaFactor,
        DstColorFactor: DstColorFactor,
        DynamicBufferAttribute: DynamicBufferAttribute,
        EdgesGeometry: EdgesGeometry,
        EdgesHelper: EdgesHelper,
        EllipseCurve: EllipseCurve,
        EqualDepth: EqualDepth,
        EqualStencilFunc: EqualStencilFunc,
        EquirectangularReflectionMapping: EquirectangularReflectionMapping,
        EquirectangularRefractionMapping: EquirectangularRefractionMapping,
        Euler: Euler,
        EventDispatcher: EventDispatcher,
        ExtrudeBufferGeometry: ExtrudeBufferGeometry,
        ExtrudeGeometry: ExtrudeGeometry,
        Face3: Face3,
        Face4: Face4,
        FaceColors: FaceColors,
        FaceNormalsHelper: FaceNormalsHelper,
        FileLoader: FileLoader,
        FlatShading: FlatShading,
        Float32Attribute: Float32Attribute,
        Float32BufferAttribute: Float32BufferAttribute,
        Float64Attribute: Float64Attribute,
        Float64BufferAttribute: Float64BufferAttribute,
        FloatType: FloatType,
        Fog: Fog,
        FogExp2: FogExp2,
        Font: Font,
        FontLoader: FontLoader,
        FrontFaceDirectionCCW: FrontFaceDirectionCCW,
        FrontFaceDirectionCW: FrontFaceDirectionCW,
        FrontSide: FrontSide,
        Frustum: Frustum,
        GammaEncoding: GammaEncoding,
        Geometry: Geometry,
        GeometryUtils: GeometryUtils,
        GreaterDepth: GreaterDepth,
        GreaterEqualDepth: GreaterEqualDepth,
        GreaterEqualStencilFunc: GreaterEqualStencilFunc,
        GreaterStencilFunc: GreaterStencilFunc,
        GridHelper: GridHelper,
        Group: Group,
        HalfFloatType: HalfFloatType,
        HemisphereLight: HemisphereLight,
        HemisphereLightHelper: HemisphereLightHelper,
        HemisphereLightProbe: HemisphereLightProbe,
        IcosahedronBufferGeometry: IcosahedronBufferGeometry,
        IcosahedronGeometry: IcosahedronGeometry,
        ImageBitmapLoader: ImageBitmapLoader,
        ImageLoader: ImageLoader,
        ImageUtils: ImageUtils,
        ImmediateRenderObject: ImmediateRenderObject,
        IncrementStencilOp: IncrementStencilOp,
        IncrementWrapStencilOp: IncrementWrapStencilOp,
        InstancedBufferAttribute: InstancedBufferAttribute,
        InstancedBufferGeometry: InstancedBufferGeometry,
        InstancedInterleavedBuffer: InstancedInterleavedBuffer,
        Int16Attribute: Int16Attribute,
        Int16BufferAttribute: Int16BufferAttribute,
        Int32Attribute: Int32Attribute,
        Int32BufferAttribute: Int32BufferAttribute,
        Int8Attribute: Int8Attribute,
        Int8BufferAttribute: Int8BufferAttribute,
        IntType: IntType,
        InterleavedBuffer: InterleavedBuffer,
        InterleavedBufferAttribute: InterleavedBufferAttribute,
        Interpolant: Interpolant,
        InterpolateDiscrete: InterpolateDiscrete,
        InterpolateLinear: InterpolateLinear,
        InterpolateSmooth: InterpolateSmooth,
        InvertStencilOp: InvertStencilOp,
        JSONLoader: JSONLoader,
        KeepStencilOp: KeepStencilOp,
        KeyframeTrack: KeyframeTrack,
        LOD: LOD,
        LatheBufferGeometry: LatheBufferGeometry,
        LatheGeometry: LatheGeometry,
        Layers: Layers,
        LensFlare: LensFlare,
        LessDepth: LessDepth,
        LessEqualDepth: LessEqualDepth,
        LessEqualStencilFunc: LessEqualStencilFunc,
        LessStencilFunc: LessStencilFunc,
        Light: Light,
        LightProbe: LightProbe,
        LightProbeHelper: LightProbeHelper,
        LightShadow: LightShadow,
        Line: Line,
        Line3: Line3,
        LineBasicMaterial: LineBasicMaterial,
        LineCurve: LineCurve,
        LineCurve3: LineCurve3,
        LineDashedMaterial: LineDashedMaterial,
        LineLoop: LineLoop,
        LinePieces: LinePieces,
        LineSegments: LineSegments,
        LineStrip: LineStrip,
        LinearEncoding: LinearEncoding,
        LinearFilter: LinearFilter,
        LinearInterpolant: LinearInterpolant,
        LinearMipMapLinearFilter: LinearMipMapLinearFilter,
        LinearMipMapNearestFilter: LinearMipMapNearestFilter,
        LinearMipmapLinearFilter: LinearMipmapLinearFilter,
        LinearMipmapNearestFilter: LinearMipmapNearestFilter,
        LinearToneMapping: LinearToneMapping,
        Loader: Loader,
        LoaderUtils: LoaderUtils,
        LoadingManager: LoadingManager,
        LogLuvEncoding: LogLuvEncoding,
        LoopOnce: LoopOnce,
        LoopPingPong: LoopPingPong,
        LoopRepeat: LoopRepeat,
        LuminanceAlphaFormat: LuminanceAlphaFormat,
        LuminanceFormat: LuminanceFormat,
        MOUSE: MOUSE,
        Material: Material,
        MaterialLoader: MaterialLoader,
        Math: _Math,
        Matrix3: Matrix3,
        Matrix4: Matrix4,
        MaxEquation: MaxEquation,
        Mesh: Mesh,
        MeshBasicMaterial: MeshBasicMaterial,
        MeshDepthMaterial: MeshDepthMaterial,
        MeshDistanceMaterial: MeshDistanceMaterial,
        MeshFaceMaterial: MeshFaceMaterial,
        MeshLambertMaterial: MeshLambertMaterial,
        MeshMatcapMaterial: MeshMatcapMaterial,
        MeshNormalMaterial: MeshNormalMaterial,
        MeshPhongMaterial: MeshPhongMaterial,
        MeshPhysicalMaterial: MeshPhysicalMaterial,
        MeshStandardMaterial: MeshStandardMaterial,
        MeshToonMaterial: MeshToonMaterial,
        MinEquation: MinEquation,
        MirroredRepeatWrapping: MirroredRepeatWrapping,
        MixOperation: MixOperation,
        MultiMaterial: MultiMaterial,
        MultiplyBlending: MultiplyBlending,
        MultiplyOperation: MultiplyOperation,
        NearestFilter: NearestFilter,
        NearestMipMapLinearFilter: NearestMipMapLinearFilter,
        NearestMipMapNearestFilter: NearestMipMapNearestFilter,
        NearestMipmapLinearFilter: NearestMipmapLinearFilter,
        NearestMipmapNearestFilter: NearestMipmapNearestFilter,
        NeverDepth: NeverDepth,
        NeverStencilFunc: NeverStencilFunc,
        NoBlending: NoBlending,
        NoColors: NoColors,
        NoToneMapping: NoToneMapping,
        NormalBlending: NormalBlending,
        NotEqualDepth: NotEqualDepth,
        NotEqualStencilFunc: NotEqualStencilFunc,
        NumberKeyframeTrack: NumberKeyframeTrack,
        Object3D: Object3D,
        ObjectLoader: ObjectLoader,
        ObjectSpaceNormalMap: ObjectSpaceNormalMap,
        OctahedronBufferGeometry: OctahedronBufferGeometry,
        OctahedronGeometry: OctahedronGeometry,
        OneFactor: OneFactor,
        OneMinusDstAlphaFactor: OneMinusDstAlphaFactor,
        OneMinusDstColorFactor: OneMinusDstColorFactor,
        OneMinusSrcAlphaFactor: OneMinusSrcAlphaFactor,
        OneMinusSrcColorFactor: OneMinusSrcColorFactor,
        OrthographicCamera: OrthographicCamera,
        PCFShadowMap: PCFShadowMap,
        PCFSoftShadowMap: PCFSoftShadowMap,
        ParametricBufferGeometry: ParametricBufferGeometry,
        ParametricGeometry: ParametricGeometry,
        Particle: Particle,
        ParticleBasicMaterial: ParticleBasicMaterial,
        ParticleSystem: ParticleSystem,
        ParticleSystemMaterial: ParticleSystemMaterial,
        Path: Path$4,
        PerspectiveCamera: PerspectiveCamera,
        Plane: Plane,
        PlaneBufferGeometry: PlaneBufferGeometry,
        PlaneGeometry: PlaneGeometry,
        PlaneHelper: PlaneHelper,
        PointCloud: PointCloud,
        PointCloudMaterial: PointCloudMaterial,
        PointLight: PointLight,
        PointLightHelper: PointLightHelper,
        Points: Points,
        PointsMaterial: PointsMaterial,
        PolarGridHelper: PolarGridHelper,
        PolyhedronBufferGeometry: PolyhedronBufferGeometry,
        PolyhedronGeometry: PolyhedronGeometry,
        PositionalAudio: PositionalAudio,
        PositionalAudioHelper: PositionalAudioHelper,
        PropertyBinding: PropertyBinding,
        PropertyMixer: PropertyMixer,
        QuadraticBezierCurve: QuadraticBezierCurve,
        QuadraticBezierCurve3: QuadraticBezierCurve3,
        Quaternion: Quaternion,
        QuaternionKeyframeTrack: QuaternionKeyframeTrack,
        QuaternionLinearInterpolant: QuaternionLinearInterpolant,
        REVISION: REVISION,
        RGBADepthPacking: RGBADepthPacking,
        RGBAFormat: RGBAFormat,
        RGBA_ASTC_10x10_Format: RGBA_ASTC_10x10_Format,
        RGBA_ASTC_10x5_Format: RGBA_ASTC_10x5_Format,
        RGBA_ASTC_10x6_Format: RGBA_ASTC_10x6_Format,
        RGBA_ASTC_10x8_Format: RGBA_ASTC_10x8_Format,
        RGBA_ASTC_12x10_Format: RGBA_ASTC_12x10_Format,
        RGBA_ASTC_12x12_Format: RGBA_ASTC_12x12_Format,
        RGBA_ASTC_4x4_Format: RGBA_ASTC_4x4_Format,
        RGBA_ASTC_5x4_Format: RGBA_ASTC_5x4_Format,
        RGBA_ASTC_5x5_Format: RGBA_ASTC_5x5_Format,
        RGBA_ASTC_6x5_Format: RGBA_ASTC_6x5_Format,
        RGBA_ASTC_6x6_Format: RGBA_ASTC_6x6_Format,
        RGBA_ASTC_8x5_Format: RGBA_ASTC_8x5_Format,
        RGBA_ASTC_8x6_Format: RGBA_ASTC_8x6_Format,
        RGBA_ASTC_8x8_Format: RGBA_ASTC_8x8_Format,
        RGBA_PVRTC_2BPPV1_Format: RGBA_PVRTC_2BPPV1_Format,
        RGBA_PVRTC_4BPPV1_Format: RGBA_PVRTC_4BPPV1_Format,
        RGBA_S3TC_DXT1_Format: RGBA_S3TC_DXT1_Format,
        RGBA_S3TC_DXT3_Format: RGBA_S3TC_DXT3_Format,
        RGBA_S3TC_DXT5_Format: RGBA_S3TC_DXT5_Format,
        RGBDEncoding: RGBDEncoding,
        RGBEEncoding: RGBEEncoding,
        RGBEFormat: RGBEFormat,
        RGBFormat: RGBFormat,
        RGBM16Encoding: RGBM16Encoding,
        RGBM7Encoding: RGBM7Encoding,
        RGB_ETC1_Format: RGB_ETC1_Format,
        RGB_PVRTC_2BPPV1_Format: RGB_PVRTC_2BPPV1_Format,
        RGB_PVRTC_4BPPV1_Format: RGB_PVRTC_4BPPV1_Format,
        RGB_S3TC_DXT1_Format: RGB_S3TC_DXT1_Format,
        RawShaderMaterial: RawShaderMaterial,
        Ray: Ray,
        Raycaster: Raycaster,
        RectAreaLight: RectAreaLight,
        RectAreaLightHelper: RectAreaLightHelper,
        RedFormat: RedFormat,
        ReinhardToneMapping: ReinhardToneMapping,
        RepeatWrapping: RepeatWrapping,
        ReplaceStencilOp: ReplaceStencilOp,
        ReverseSubtractEquation: ReverseSubtractEquation,
        RingBufferGeometry: RingBufferGeometry,
        RingGeometry: RingGeometry,
        Scene: Scene,
        SceneUtils: SceneUtils,
        ShaderChunk: ShaderChunk,
        ShaderLib: ShaderLib,
        ShaderMaterial: ShaderMaterial,
        ShadowMaterial: ShadowMaterial,
        Shape: Shape,
        ShapeBufferGeometry: ShapeBufferGeometry,
        ShapeGeometry: ShapeGeometry,
        ShapePath: ShapePath,
        ShapeUtils: ShapeUtils,
        ShortType: ShortType,
        Skeleton: Skeleton,
        SkeletonHelper: SkeletonHelper,
        SkinnedMesh: SkinnedMesh,
        SmoothShading: SmoothShading,
        Sphere: Sphere,
        SphereBufferGeometry: SphereBufferGeometry,
        SphereGeometry: SphereGeometry,
        Spherical: Spherical,
        SphericalHarmonics3: SphericalHarmonics3,
        SphericalReflectionMapping: SphericalReflectionMapping,
        Spline: Spline,
        SplineCurve: SplineCurve,
        SplineCurve3: SplineCurve3,
        SpotLight: SpotLight,
        SpotLightHelper: SpotLightHelper,
        SpotLightShadow: SpotLightShadow,
        Sprite: Sprite,
        SpriteMaterial: SpriteMaterial,
        SrcAlphaFactor: SrcAlphaFactor,
        SrcAlphaSaturateFactor: SrcAlphaSaturateFactor,
        SrcColorFactor: SrcColorFactor,
        StereoCamera: StereoCamera,
        StringKeyframeTrack: StringKeyframeTrack,
        SubtractEquation: SubtractEquation,
        SubtractiveBlending: SubtractiveBlending,
        TOUCH: TOUCH,
        TangentSpaceNormalMap: TangentSpaceNormalMap,
        TetrahedronBufferGeometry: TetrahedronBufferGeometry,
        TetrahedronGeometry: TetrahedronGeometry,
        TextBufferGeometry: TextBufferGeometry,
        TextGeometry: TextGeometry,
        Texture: Texture,
        TextureLoader: TextureLoader,
        TorusBufferGeometry: TorusBufferGeometry,
        TorusGeometry: TorusGeometry,
        TorusKnotBufferGeometry: TorusKnotBufferGeometry,
        TorusKnotGeometry: TorusKnotGeometry,
        Triangle: Triangle,
        TriangleFanDrawMode: TriangleFanDrawMode,
        TriangleStripDrawMode: TriangleStripDrawMode,
        TrianglesDrawMode: TrianglesDrawMode,
        TubeBufferGeometry: TubeBufferGeometry,
        TubeGeometry: TubeGeometry,
        UVMapping: UVMapping,
        Uint16Attribute: Uint16Attribute,
        Uint16BufferAttribute: Uint16BufferAttribute,
        Uint32Attribute: Uint32Attribute,
        Uint32BufferAttribute: Uint32BufferAttribute,
        Uint8Attribute: Uint8Attribute,
        Uint8BufferAttribute: Uint8BufferAttribute,
        Uint8ClampedAttribute: Uint8ClampedAttribute,
        Uint8ClampedBufferAttribute: Uint8ClampedBufferAttribute,
        Uncharted2ToneMapping: Uncharted2ToneMapping,
        Uniform: Uniform,
        UniformsLib: UniformsLib,
        UniformsUtils: UniformsUtils,
        UnsignedByteType: UnsignedByteType,
        UnsignedInt248Type: UnsignedInt248Type,
        UnsignedIntType: UnsignedIntType,
        UnsignedShort4444Type: UnsignedShort4444Type,
        UnsignedShort5551Type: UnsignedShort5551Type,
        UnsignedShort565Type: UnsignedShort565Type,
        UnsignedShortType: UnsignedShortType,
        VSMShadowMap: VSMShadowMap,
        Vector2: Vector2,
        Vector3: Vector3,
        Vector4: Vector4,
        VectorKeyframeTrack: VectorKeyframeTrack,
        Vertex: Vertex,
        VertexColors: VertexColors,
        VertexNormalsHelper: VertexNormalsHelper,
        VideoTexture: VideoTexture,
        WebGLMultisampleRenderTarget: WebGLMultisampleRenderTarget,
        WebGLRenderTarget: WebGLRenderTarget,
        WebGLRenderTargetCube: WebGLRenderTargetCube,
        WebGLRenderer: WebGLRenderer,
        WebGLUtils: WebGLUtils,
        WireframeGeometry: WireframeGeometry,
        WireframeHelper: WireframeHelper,
        WrapAroundEnding: WrapAroundEnding,
        XHRLoader: XHRLoader,
        ZeroCurvatureEnding: ZeroCurvatureEnding,
        ZeroFactor: ZeroFactor,
        ZeroSlopeEnding: ZeroSlopeEnding,
        ZeroStencilOp: ZeroStencilOp,
        sRGBEncoding: sRGBEncoding
    });

    /**
     * @author qiao / https://github.com/qiao
     * @author mrdoob / http://mrdoob.com
     * @author alteredq / http://alteredqualia.com/
     * @author WestLangley / http://github.com/WestLangley
     * @author erich666 / http://erichaines.com
     * @author ScieCode / http://github.com/sciecode
     */

    // This set of controls performs orbiting, dollying (zooming), and panning.
    // Unlike TrackballControls, it maintains the "up" direction object.up (+Y by default).
    //
    //    Orbit - left mouse / touch: one-finger move
    //    Zoom - middle mouse, or mousewheel / touch: two-finger spread or squish
    //    Pan - right mouse, or left mouse + ctrl/meta/shiftKey, or arrow keys / touch: two-finger move

    var OrbitControls = function ( object, domElement ) {

    	this.object = object;

    	this.domElement = ( domElement !== undefined ) ? domElement : document;

    	// Set to false to disable this control
    	this.enabled = true;

    	// "target" sets the location of focus, where the object orbits around
    	this.target = new Vector3();

    	// How far you can dolly in and out ( PerspectiveCamera only )
    	this.minDistance = 0;
    	this.maxDistance = Infinity;

    	// How far you can zoom in and out ( OrthographicCamera only )
    	this.minZoom = 0;
    	this.maxZoom = Infinity;

    	// How far you can orbit vertically, upper and lower limits.
    	// Range is 0 to Math.PI radians.
    	this.minPolarAngle = 0; // radians
    	this.maxPolarAngle = Math.PI; // radians

    	// How far you can orbit horizontally, upper and lower limits.
    	// If set, must be a sub-interval of the interval [ - Math.PI, Math.PI ].
    	this.minAzimuthAngle = - Infinity; // radians
    	this.maxAzimuthAngle = Infinity; // radians

    	// Set to true to enable damping (inertia)
    	// If damping is enabled, you must call controls.update() in your animation loop
    	this.enableDamping = false;
    	this.dampingFactor = 0.05;

    	// This option actually enables dollying in and out; left as "zoom" for backwards compatibility.
    	// Set to false to disable zooming
    	this.enableZoom = true;
    	this.zoomSpeed = 1.0;

    	// Set to false to disable rotating
    	this.enableRotate = true;
    	this.rotateSpeed = 1.0;

    	// Set to false to disable panning
    	this.enablePan = true;
    	this.panSpeed = 1.0;
    	this.screenSpacePanning = false; // if true, pan in screen-space
    	this.keyPanSpeed = 7.0;	// pixels moved per arrow key push

    	// Set to true to automatically rotate around the target
    	// If auto-rotate is enabled, you must call controls.update() in your animation loop
    	this.autoRotate = false;
    	this.autoRotateSpeed = 2.0; // 30 seconds per round when fps is 60

    	// Set to false to disable use of the keys
    	this.enableKeys = true;

    	// The four arrow keys
    	this.keys = { LEFT: 37, UP: 38, RIGHT: 39, BOTTOM: 40 };

    	// Mouse buttons
    	this.mouseButtons = { LEFT: MOUSE.ROTATE, MIDDLE: MOUSE.DOLLY, RIGHT: MOUSE.PAN };

    	// Touch fingers
    	this.touches = { ONE: TOUCH.ROTATE, TWO: TOUCH.DOLLY_PAN };

    	// for reset
    	this.target0 = this.target.clone();
    	this.position0 = this.object.position.clone();
    	this.zoom0 = this.object.zoom;

    	//
    	// public methods
    	//

    	this.getPolarAngle = function () {

    		return spherical.phi;

    	};

    	this.getAzimuthalAngle = function () {

    		return spherical.theta;

    	};

    	this.saveState = function () {

    		scope.target0.copy( scope.target );
    		scope.position0.copy( scope.object.position );
    		scope.zoom0 = scope.object.zoom;

    	};

    	this.reset = function () {

    		scope.target.copy( scope.target0 );
    		scope.object.position.copy( scope.position0 );
    		scope.object.zoom = scope.zoom0;

    		scope.object.updateProjectionMatrix();
    		scope.dispatchEvent( changeEvent );

    		scope.update();

    		state = STATE.NONE;

    	};

    	// this method is exposed, but perhaps it would be better if we can make it private...
    	this.update = function () {

    		var offset = new Vector3();

    		// so camera.up is the orbit axis
    		var quat = new Quaternion().setFromUnitVectors( object.up, new Vector3( 0, 1, 0 ) );
    		var quatInverse = quat.clone().inverse();

    		var lastPosition = new Vector3();
    		var lastQuaternion = new Quaternion();

    		return function update() {

    			var position = scope.object.position;

    			offset.copy( position ).sub( scope.target );

    			// rotate offset to "y-axis-is-up" space
    			offset.applyQuaternion( quat );

    			// angle from z-axis around y-axis
    			spherical.setFromVector3( offset );

    			if ( scope.autoRotate && state === STATE.NONE ) {

    				rotateLeft( getAutoRotationAngle() );

    			}

    			if ( scope.enableDamping ) {

    				spherical.theta += sphericalDelta.theta * scope.dampingFactor;
    				spherical.phi += sphericalDelta.phi * scope.dampingFactor;

    			} else {

    				spherical.theta += sphericalDelta.theta;
    				spherical.phi += sphericalDelta.phi;

    			}

    			// restrict theta to be between desired limits
    			spherical.theta = Math.max( scope.minAzimuthAngle, Math.min( scope.maxAzimuthAngle, spherical.theta ) );

    			// restrict phi to be between desired limits
    			spherical.phi = Math.max( scope.minPolarAngle, Math.min( scope.maxPolarAngle, spherical.phi ) );

    			spherical.makeSafe();


    			spherical.radius *= scale;

    			// restrict radius to be between desired limits
    			spherical.radius = Math.max( scope.minDistance, Math.min( scope.maxDistance, spherical.radius ) );

    			// move target to panned location

    			if ( scope.enableDamping === true ) {

    				scope.target.addScaledVector( panOffset, scope.dampingFactor );

    			} else {

    				scope.target.add( panOffset );

    			}

    			offset.setFromSpherical( spherical );

    			// rotate offset back to "camera-up-vector-is-up" space
    			offset.applyQuaternion( quatInverse );

    			position.copy( scope.target ).add( offset );

    			scope.object.lookAt( scope.target );

    			if ( scope.enableDamping === true ) {

    				sphericalDelta.theta *= ( 1 - scope.dampingFactor );
    				sphericalDelta.phi *= ( 1 - scope.dampingFactor );

    				panOffset.multiplyScalar( 1 - scope.dampingFactor );

    			} else {

    				sphericalDelta.set( 0, 0, 0 );

    				panOffset.set( 0, 0, 0 );

    			}

    			scale = 1;

    			// update condition is:
    			// min(camera displacement, camera rotation in radians)^2 > EPS
    			// using small-angle approximation cos(x/2) = 1 - x^2 / 8

    			if ( zoomChanged ||
    				lastPosition.distanceToSquared( scope.object.position ) > EPS ||
    				8 * ( 1 - lastQuaternion.dot( scope.object.quaternion ) ) > EPS ) {

    				scope.dispatchEvent( changeEvent );

    				lastPosition.copy( scope.object.position );
    				lastQuaternion.copy( scope.object.quaternion );
    				zoomChanged = false;

    				return true;

    			}

    			return false;

    		};

    	}();

    	this.dispose = function () {

    		scope.domElement.removeEventListener( 'contextmenu', onContextMenu, false );
    		scope.domElement.removeEventListener( 'mousedown', onMouseDown, false );
    		scope.domElement.removeEventListener( 'wheel', onMouseWheel, false );

    		scope.domElement.removeEventListener( 'touchstart', onTouchStart, false );
    		scope.domElement.removeEventListener( 'touchend', onTouchEnd, false );
    		scope.domElement.removeEventListener( 'touchmove', onTouchMove, false );

    		document.removeEventListener( 'mousemove', onMouseMove, false );
    		document.removeEventListener( 'mouseup', onMouseUp, false );

    		window.removeEventListener( 'keydown', onKeyDown, false );

    		//scope.dispatchEvent( { type: 'dispose' } ); // should this be added here?

    	};

    	//
    	// internals
    	//

    	var scope = this;

    	var changeEvent = { type: 'change' };
    	var startEvent = { type: 'start' };
    	var endEvent = { type: 'end' };

    	var STATE = {
    		NONE: - 1,
    		ROTATE: 0,
    		DOLLY: 1,
    		PAN: 2,
    		TOUCH_ROTATE: 3,
    		TOUCH_PAN: 4,
    		TOUCH_DOLLY_PAN: 5,
    		TOUCH_DOLLY_ROTATE: 6
    	};

    	var state = STATE.NONE;

    	var EPS = 0.000001;

    	// current position in spherical coordinates
    	var spherical = new Spherical();
    	var sphericalDelta = new Spherical();

    	var scale = 1;
    	var panOffset = new Vector3();
    	var zoomChanged = false;

    	var rotateStart = new Vector2();
    	var rotateEnd = new Vector2();
    	var rotateDelta = new Vector2();

    	var panStart = new Vector2();
    	var panEnd = new Vector2();
    	var panDelta = new Vector2();

    	var dollyStart = new Vector2();
    	var dollyEnd = new Vector2();
    	var dollyDelta = new Vector2();

    	function getAutoRotationAngle() {

    		return 2 * Math.PI / 60 / 60 * scope.autoRotateSpeed;

    	}

    	function getZoomScale() {

    		return Math.pow( 0.95, scope.zoomSpeed );

    	}

    	function rotateLeft( angle ) {

    		sphericalDelta.theta -= angle;

    	}

    	function rotateUp( angle ) {

    		sphericalDelta.phi -= angle;

    	}

    	var panLeft = function () {

    		var v = new Vector3();

    		return function panLeft( distance, objectMatrix ) {

    			v.setFromMatrixColumn( objectMatrix, 0 ); // get X column of objectMatrix
    			v.multiplyScalar( - distance );

    			panOffset.add( v );

    		};

    	}();

    	var panUp = function () {

    		var v = new Vector3();

    		return function panUp( distance, objectMatrix ) {

    			if ( scope.screenSpacePanning === true ) {

    				v.setFromMatrixColumn( objectMatrix, 1 );

    			} else {

    				v.setFromMatrixColumn( objectMatrix, 0 );
    				v.crossVectors( scope.object.up, v );

    			}

    			v.multiplyScalar( distance );

    			panOffset.add( v );

    		};

    	}();

    	// deltaX and deltaY are in pixels; right and down are positive
    	var pan = function () {

    		var offset = new Vector3();

    		return function pan( deltaX, deltaY ) {

    			var element = scope.domElement === document ? scope.domElement.body : scope.domElement;

    			if ( scope.object.isPerspectiveCamera ) {

    				// perspective
    				var position = scope.object.position;
    				offset.copy( position ).sub( scope.target );
    				var targetDistance = offset.length();

    				// half of the fov is center to top of screen
    				targetDistance *= Math.tan( ( scope.object.fov / 2 ) * Math.PI / 180.0 );

    				// we use only clientHeight here so aspect ratio does not distort speed
    				panLeft( 2 * deltaX * targetDistance / element.clientHeight, scope.object.matrix );
    				panUp( 2 * deltaY * targetDistance / element.clientHeight, scope.object.matrix );

    			} else if ( scope.object.isOrthographicCamera ) {

    				// orthographic
    				panLeft( deltaX * ( scope.object.right - scope.object.left ) / scope.object.zoom / element.clientWidth, scope.object.matrix );
    				panUp( deltaY * ( scope.object.top - scope.object.bottom ) / scope.object.zoom / element.clientHeight, scope.object.matrix );

    			} else {

    				// camera neither orthographic nor perspective
    				console.warn( 'WARNING: OrbitControls.js encountered an unknown camera type - pan disabled.' );
    				scope.enablePan = false;

    			}

    		};

    	}();

    	function dollyIn( dollyScale ) {

    		if ( scope.object.isPerspectiveCamera ) {

    			scale /= dollyScale;

    		} else if ( scope.object.isOrthographicCamera ) {

    			scope.object.zoom = Math.max( scope.minZoom, Math.min( scope.maxZoom, scope.object.zoom * dollyScale ) );
    			scope.object.updateProjectionMatrix();
    			zoomChanged = true;

    		} else {

    			console.warn( 'WARNING: OrbitControls.js encountered an unknown camera type - dolly/zoom disabled.' );
    			scope.enableZoom = false;

    		}

    	}

    	function dollyOut( dollyScale ) {

    		if ( scope.object.isPerspectiveCamera ) {

    			scale *= dollyScale;

    		} else if ( scope.object.isOrthographicCamera ) {

    			scope.object.zoom = Math.max( scope.minZoom, Math.min( scope.maxZoom, scope.object.zoom / dollyScale ) );
    			scope.object.updateProjectionMatrix();
    			zoomChanged = true;

    		} else {

    			console.warn( 'WARNING: OrbitControls.js encountered an unknown camera type - dolly/zoom disabled.' );
    			scope.enableZoom = false;

    		}

    	}

    	//
    	// event callbacks - update the object state
    	//

    	function handleMouseDownRotate( event ) {

    		rotateStart.set( event.clientX, event.clientY );

    	}

    	function handleMouseDownDolly( event ) {

    		dollyStart.set( event.clientX, event.clientY );

    	}

    	function handleMouseDownPan( event ) {

    		panStart.set( event.clientX, event.clientY );

    	}

    	function handleMouseMoveRotate( event ) {

    		rotateEnd.set( event.clientX, event.clientY );

    		rotateDelta.subVectors( rotateEnd, rotateStart ).multiplyScalar( scope.rotateSpeed );

    		var element = scope.domElement === document ? scope.domElement.body : scope.domElement;

    		rotateLeft( 2 * Math.PI * rotateDelta.x / element.clientHeight ); // yes, height

    		rotateUp( 2 * Math.PI * rotateDelta.y / element.clientHeight );

    		rotateStart.copy( rotateEnd );

    		scope.update();

    	}

    	function handleMouseMoveDolly( event ) {

    		dollyEnd.set( event.clientX, event.clientY );

    		dollyDelta.subVectors( dollyEnd, dollyStart );

    		if ( dollyDelta.y > 0 ) {

    			dollyIn( getZoomScale() );

    		} else if ( dollyDelta.y < 0 ) {

    			dollyOut( getZoomScale() );

    		}

    		dollyStart.copy( dollyEnd );

    		scope.update();

    	}

    	function handleMouseMovePan( event ) {

    		panEnd.set( event.clientX, event.clientY );

    		panDelta.subVectors( panEnd, panStart ).multiplyScalar( scope.panSpeed );

    		pan( panDelta.x, panDelta.y );

    		panStart.copy( panEnd );

    		scope.update();

    	}

    	function handleMouseWheel( event ) {

    		if ( event.deltaY < 0 ) {

    			dollyOut( getZoomScale() );

    		} else if ( event.deltaY > 0 ) {

    			dollyIn( getZoomScale() );

    		}

    		scope.update();

    	}

    	function handleKeyDown( event ) {

    		var needsUpdate = false;

    		switch ( event.keyCode ) {

    			case scope.keys.UP:
    				pan( 0, scope.keyPanSpeed );
    				needsUpdate = true;
    				break;

    			case scope.keys.BOTTOM:
    				pan( 0, - scope.keyPanSpeed );
    				needsUpdate = true;
    				break;

    			case scope.keys.LEFT:
    				pan( scope.keyPanSpeed, 0 );
    				needsUpdate = true;
    				break;

    			case scope.keys.RIGHT:
    				pan( - scope.keyPanSpeed, 0 );
    				needsUpdate = true;
    				break;

    		}

    		if ( needsUpdate ) {

    			// prevent the browser from scrolling on cursor keys
    			event.preventDefault();

    			scope.update();

    		}


    	}

    	function handleTouchStartRotate( event ) {

    		if ( event.touches.length == 1 ) {

    			rotateStart.set( event.touches[ 0 ].pageX, event.touches[ 0 ].pageY );

    		} else {

    			var x = 0.5 * ( event.touches[ 0 ].pageX + event.touches[ 1 ].pageX );
    			var y = 0.5 * ( event.touches[ 0 ].pageY + event.touches[ 1 ].pageY );

    			rotateStart.set( x, y );

    		}

    	}

    	function handleTouchStartPan( event ) {

    		if ( event.touches.length == 1 ) {

    			panStart.set( event.touches[ 0 ].pageX, event.touches[ 0 ].pageY );

    		} else {

    			var x = 0.5 * ( event.touches[ 0 ].pageX + event.touches[ 1 ].pageX );
    			var y = 0.5 * ( event.touches[ 0 ].pageY + event.touches[ 1 ].pageY );

    			panStart.set( x, y );

    		}

    	}

    	function handleTouchStartDolly( event ) {

    		var dx = event.touches[ 0 ].pageX - event.touches[ 1 ].pageX;
    		var dy = event.touches[ 0 ].pageY - event.touches[ 1 ].pageY;

    		var distance = Math.sqrt( dx * dx + dy * dy );

    		dollyStart.set( 0, distance );

    	}

    	function handleTouchStartDollyPan( event ) {

    		if ( scope.enableZoom ) handleTouchStartDolly( event );

    		if ( scope.enablePan ) handleTouchStartPan( event );

    	}

    	function handleTouchStartDollyRotate( event ) {

    		if ( scope.enableZoom ) handleTouchStartDolly( event );

    		if ( scope.enableRotate ) handleTouchStartRotate( event );

    	}

    	function handleTouchMoveRotate( event ) {

    		if ( event.touches.length == 1 ) {

    			rotateEnd.set( event.touches[ 0 ].pageX, event.touches[ 0 ].pageY );

    		} else {

    			var x = 0.5 * ( event.touches[ 0 ].pageX + event.touches[ 1 ].pageX );
    			var y = 0.5 * ( event.touches[ 0 ].pageY + event.touches[ 1 ].pageY );

    			rotateEnd.set( x, y );

    		}

    		rotateDelta.subVectors( rotateEnd, rotateStart ).multiplyScalar( scope.rotateSpeed );

    		var element = scope.domElement === document ? scope.domElement.body : scope.domElement;

    		rotateLeft( 2 * Math.PI * rotateDelta.x / element.clientHeight ); // yes, height

    		rotateUp( 2 * Math.PI * rotateDelta.y / element.clientHeight );

    		rotateStart.copy( rotateEnd );

    	}

    	function handleTouchMovePan( event ) {

    		if ( event.touches.length == 1 ) {

    			panEnd.set( event.touches[ 0 ].pageX, event.touches[ 0 ].pageY );

    		} else {

    			var x = 0.5 * ( event.touches[ 0 ].pageX + event.touches[ 1 ].pageX );
    			var y = 0.5 * ( event.touches[ 0 ].pageY + event.touches[ 1 ].pageY );

    			panEnd.set( x, y );

    		}

    		panDelta.subVectors( panEnd, panStart ).multiplyScalar( scope.panSpeed );

    		pan( panDelta.x, panDelta.y );

    		panStart.copy( panEnd );

    	}

    	function handleTouchMoveDolly( event ) {

    		var dx = event.touches[ 0 ].pageX - event.touches[ 1 ].pageX;
    		var dy = event.touches[ 0 ].pageY - event.touches[ 1 ].pageY;

    		var distance = Math.sqrt( dx * dx + dy * dy );

    		dollyEnd.set( 0, distance );

    		dollyDelta.set( 0, Math.pow( dollyEnd.y / dollyStart.y, scope.zoomSpeed ) );

    		dollyIn( dollyDelta.y );

    		dollyStart.copy( dollyEnd );

    	}

    	function handleTouchMoveDollyPan( event ) {

    		if ( scope.enableZoom ) handleTouchMoveDolly( event );

    		if ( scope.enablePan ) handleTouchMovePan( event );

    	}

    	function handleTouchMoveDollyRotate( event ) {

    		if ( scope.enableZoom ) handleTouchMoveDolly( event );

    		if ( scope.enableRotate ) handleTouchMoveRotate( event );

    	}

    	//
    	// event handlers - FSM: listen for events and reset state
    	//

    	function onMouseDown( event ) {

    		if ( scope.enabled === false ) return;

    		// Prevent the browser from scrolling.

    		event.preventDefault();

    		// Manually set the focus since calling preventDefault above
    		// prevents the browser from setting it automatically.

    		scope.domElement.focus ? scope.domElement.focus() : window.focus();

    		switch ( event.button ) {

    			case 0:

    				switch ( scope.mouseButtons.LEFT ) {

    					case MOUSE.ROTATE:

    						if ( event.ctrlKey || event.metaKey || event.shiftKey ) {

    							if ( scope.enablePan === false ) return;

    							handleMouseDownPan( event );

    							state = STATE.PAN;

    						} else {

    							if ( scope.enableRotate === false ) return;

    							handleMouseDownRotate( event );

    							state = STATE.ROTATE;

    						}

    						break;

    					case MOUSE.PAN:

    						if ( event.ctrlKey || event.metaKey || event.shiftKey ) {

    							if ( scope.enableRotate === false ) return;

    							handleMouseDownRotate( event );

    							state = STATE.ROTATE;

    						} else {

    							if ( scope.enablePan === false ) return;

    							handleMouseDownPan( event );

    							state = STATE.PAN;

    						}

    						break;

    					default:

    						state = STATE.NONE;

    				}

    				break;


    			case 1:

    				switch ( scope.mouseButtons.MIDDLE ) {

    					case MOUSE.DOLLY:

    						if ( scope.enableZoom === false ) return;

    						handleMouseDownDolly( event );

    						state = STATE.DOLLY;

    						break;


    					default:

    						state = STATE.NONE;

    				}

    				break;

    			case 2:

    				switch ( scope.mouseButtons.RIGHT ) {

    					case MOUSE.ROTATE:

    						if ( scope.enableRotate === false ) return;

    						handleMouseDownRotate( event );

    						state = STATE.ROTATE;

    						break;

    					case MOUSE.PAN:

    						if ( scope.enablePan === false ) return;

    						handleMouseDownPan( event );

    						state = STATE.PAN;

    						break;

    					default:

    						state = STATE.NONE;

    				}

    				break;

    		}

    		if ( state !== STATE.NONE ) {

    			document.addEventListener( 'mousemove', onMouseMove, false );
    			document.addEventListener( 'mouseup', onMouseUp, false );

    			scope.dispatchEvent( startEvent );

    		}

    	}

    	function onMouseMove( event ) {

    		if ( scope.enabled === false ) return;

    		event.preventDefault();

    		switch ( state ) {

    			case STATE.ROTATE:

    				if ( scope.enableRotate === false ) return;

    				handleMouseMoveRotate( event );

    				break;

    			case STATE.DOLLY:

    				if ( scope.enableZoom === false ) return;

    				handleMouseMoveDolly( event );

    				break;

    			case STATE.PAN:

    				if ( scope.enablePan === false ) return;

    				handleMouseMovePan( event );

    				break;

    		}

    	}

    	function onMouseUp( event ) {

    		if ( scope.enabled === false ) return;

    		document.removeEventListener( 'mousemove', onMouseMove, false );
    		document.removeEventListener( 'mouseup', onMouseUp, false );

    		scope.dispatchEvent( endEvent );

    		state = STATE.NONE;

    	}

    	function onMouseWheel( event ) {

    		if ( scope.enabled === false || scope.enableZoom === false || ( state !== STATE.NONE && state !== STATE.ROTATE ) ) return;

    		event.preventDefault();
    		event.stopPropagation();

    		scope.dispatchEvent( startEvent );

    		handleMouseWheel( event );

    		scope.dispatchEvent( endEvent );

    	}

    	function onKeyDown( event ) {

    		if ( scope.enabled === false || scope.enableKeys === false || scope.enablePan === false ) return;

    		handleKeyDown( event );

    	}

    	function onTouchStart( event ) {

    		if ( scope.enabled === false ) return;

    		event.preventDefault();

    		switch ( event.touches.length ) {

    			case 1:

    				switch ( scope.touches.ONE ) {

    					case TOUCH.ROTATE:

    						if ( scope.enableRotate === false ) return;

    						handleTouchStartRotate( event );

    						state = STATE.TOUCH_ROTATE;

    						break;

    					case TOUCH.PAN:

    						if ( scope.enablePan === false ) return;

    						handleTouchStartPan( event );

    						state = STATE.TOUCH_PAN;

    						break;

    					default:

    						state = STATE.NONE;

    				}

    				break;

    			case 2:

    				switch ( scope.touches.TWO ) {

    					case TOUCH.DOLLY_PAN:

    						if ( scope.enableZoom === false && scope.enablePan === false ) return;

    						handleTouchStartDollyPan( event );

    						state = STATE.TOUCH_DOLLY_PAN;

    						break;

    					case TOUCH.DOLLY_ROTATE:

    						if ( scope.enableZoom === false && scope.enableRotate === false ) return;

    						handleTouchStartDollyRotate( event );

    						state = STATE.TOUCH_DOLLY_ROTATE;

    						break;

    					default:

    						state = STATE.NONE;

    				}

    				break;

    			default:

    				state = STATE.NONE;

    		}

    		if ( state !== STATE.NONE ) {

    			scope.dispatchEvent( startEvent );

    		}

    	}

    	function onTouchMove( event ) {

    		if ( scope.enabled === false ) return;

    		event.preventDefault();
    		event.stopPropagation();

    		switch ( state ) {

    			case STATE.TOUCH_ROTATE:

    				if ( scope.enableRotate === false ) return;

    				handleTouchMoveRotate( event );

    				scope.update();

    				break;

    			case STATE.TOUCH_PAN:

    				if ( scope.enablePan === false ) return;

    				handleTouchMovePan( event );

    				scope.update();

    				break;

    			case STATE.TOUCH_DOLLY_PAN:

    				if ( scope.enableZoom === false && scope.enablePan === false ) return;

    				handleTouchMoveDollyPan( event );

    				scope.update();

    				break;

    			case STATE.TOUCH_DOLLY_ROTATE:

    				if ( scope.enableZoom === false && scope.enableRotate === false ) return;

    				handleTouchMoveDollyRotate( event );

    				scope.update();

    				break;

    			default:

    				state = STATE.NONE;

    		}

    	}

    	function onTouchEnd( event ) {

    		if ( scope.enabled === false ) return;

    		scope.dispatchEvent( endEvent );

    		state = STATE.NONE;

    	}

    	function onContextMenu( event ) {

    		if ( scope.enabled === false ) return;

    		event.preventDefault();

    	}

    	//

    	scope.domElement.addEventListener( 'contextmenu', onContextMenu, false );

    	scope.domElement.addEventListener( 'mousedown', onMouseDown, false );
    	scope.domElement.addEventListener( 'wheel', onMouseWheel, false );

    	scope.domElement.addEventListener( 'touchstart', onTouchStart, false );
    	scope.domElement.addEventListener( 'touchend', onTouchEnd, false );
    	scope.domElement.addEventListener( 'touchmove', onTouchMove, false );

    	window.addEventListener( 'keydown', onKeyDown, false );

    	// force an update at start

    	this.update();

    };

    OrbitControls.prototype = Object.create( EventDispatcher.prototype );
    OrbitControls.prototype.constructor = OrbitControls;


    // This set of controls performs orbiting, dollying (zooming), and panning.
    // Unlike TrackballControls, it maintains the "up" direction object.up (+Y by default).
    // This is very similar to OrbitControls, another set of touch behavior
    //
    //    Orbit - right mouse, or left mouse + ctrl/meta/shiftKey / touch: two-finger rotate
    //    Zoom - middle mouse, or mousewheel / touch: two-finger spread or squish
    //    Pan - left mouse, or arrow keys / touch: one-finger move

    var MapControls = function ( object, domElement ) {

    	OrbitControls.call( this, object, domElement );

    	this.mouseButtons.LEFT = MOUSE.PAN;
    	this.mouseButtons.RIGHT = MOUSE.ROTATE;

    	this.touches.ONE = TOUCH.PAN;
    	this.touches.TWO = TOUCH.DOLLY_ROTATE;

    };

    MapControls.prototype = Object.create( EventDispatcher.prototype );
    MapControls.prototype.constructor = MapControls;

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    class MeshViewer extends EventDispatcher {
        /**
         * MeshViewer constructor. Initializes the component and underlying objects.
         * @param {string} runColor Run color to use in case when colors are absent.
         */
        constructor(runColor) {
            super();
            this._lastMesh = null;
            this._clock = new Clock();
            /** @type {!Object} Contains width and height of the canvas. */
            this._canvasSize = null;
            /** @type {!Object} Describes what layers must be rendered in addition
             to a mesh or a point cloud layers. */
            this._layersConfig = null;
            this._runColor = runColor;
        }
        // TODO(b/130030314) replace with some thirdparty library call.
        /**
         * Returns true if the specified value is an object.
         * @param {?} val Variable to test.
         * @private
         * @return {boolean} Whether variable is an object.
         */
        _isObject(val) {
            var type = typeof val;
            // We're interested in objects representing dictionaries only. Everything
            // else is "not mergeable", so we consider it as primitive types.
            return type == 'object' && val != null && !Array.isArray(val);
        }
        /**
         * Merges two configs together.
         * @param {!Object} userConfig User configuration has higher priority.
         * @param {!Object} defaultConfig Default configuration has lower priority and
         *   will be overridden by any conflicting keys from userConfig.
         * @private
         * @return {!Object} Merged dictionary from two configuration dictionaries.
         */
        _applyDefaults(userConfig, defaultConfig) {
            let mergedConfig = {};
            const configs = [userConfig, defaultConfig];
            for (let i = 0; i < configs.length; i++) {
                const config = configs[i];
                for (let key in config) {
                    const is_key_present = key in mergedConfig;
                    if (this._isObject(config[key])) {
                        mergedConfig[key] = this._applyDefaults(mergedConfig[key] || {}, config[key]);
                    }
                    else if (!is_key_present) {
                        mergedConfig[key] = config[key];
                    }
                }
            }
            return mergedConfig;
        }
        /**
         * Creates additional layers to render on top of a mesh or a point cloud
         * layers.
         * @private
         */
        _createLayers() {
            if (!this._layersConfig || !this._scene || !this._lastMesh)
                return;
            if (this._layersConfig.showBoundingBox) {
                var box = new BoxHelper(this._lastMesh, new Color$7('rgb(0, 0, 255)'));
                this._scene.add(box);
            }
            if (this._layersConfig.showAxes) {
                var axesHelper = new AxesHelper(5);
                this._scene.add(axesHelper);
            }
        }
        /**
         * Sets layers config.
         * @param {!Object} layersConfig Config object describing what layers should
         *  be rendered.
         */
        setLayersConfig(layersConfig) {
            this._layersConfig = this._applyDefaults(layersConfig, this._layersConfig || {});
        }
        /**
         * Creates scene, camera and renderer.
         * @param {!Object} config Scene rendering configuration.
         * @param {!HTMLDOMElement} domElement The HTML element used for event listeners.
         * @private
         */
        _createWorld(config, domElement) {
            if (this.isReady()) {
                // keep world objects as singleton objects.
                return;
            }
            this._scene = new Scene();
            var camera = new THREE$1[config.camera.cls](config.camera.fov, this._canvasSize.width / this._canvasSize.height, config.camera.near, config.camera.far);
            this._camera = camera;
            var camControls = new OrbitControls(camera, domElement);
            const anyCamControls = camControls;
            // TODO(tensorboard-team): check whether these are depreacted; they no longer exist
            // in API docs.
            anyCamControls.lookSpeed = 0.4;
            anyCamControls.movementSpeed = 20;
            anyCamControls.noFly = true;
            anyCamControls.lookVertical = true;
            anyCamControls.constrainVertical = true;
            anyCamControls.verticalMin = 1;
            anyCamControls.verticalMax = 2;
            anyCamControls.addEventListener('change', this._onCameraPositionChange.bind(this));
            this._cameraControls = camControls;
            this._renderer = new WebGLRenderer({ antialias: true });
            this._renderer.setPixelRatio(window.devicePixelRatio);
            this._renderer.setSize(this._canvasSize.width, this._canvasSize.height);
            this._renderer.setClearColor(0xffffff, 1);
        }
        /**
         * Clears scene from any 3D geometry.
         */
        _clearScene() {
            while (this._scene.children.length > 0) {
                this._scene.remove(this._scene.children[0]);
            }
        }
        /**
         * Returns underlying renderer.
         * @public
         */
        getRenderer() {
            return this._renderer;
        }
        /**
         * Returns underlying camera controls.
         * @public
         */
        getCameraControls() {
            return this._cameraControls;
        }
        /**
         * Returns true when all underlying components were initialized.
         * @public
         */
        isReady() {
            return !!this._camera && !!this._cameraControls;
        }
        /**
         * Returns current camera position.
         * @public
         */
        getCameraPosition() {
            return {
                far: this._camera.far,
                position: this._camera.position.clone(),
                target: this._cameraControls.target.clone(),
            };
        }
        /**
         * Sets new canvas size.
         * @param {!Object} canvasSize Contains current canvas width and height.
         * @public
         */
        setCanvasSize(canvasSize) {
            this._canvasSize = canvasSize;
        }
        /**
         * Renders component into the browser.
         * @public
         */
        draw() {
            // Cancel any previous requests to perform redraw.
            if (this._animationFrameIndex) {
                cancelAnimationFrame(this._animationFrameIndex);
            }
            this._camera.aspect = this._canvasSize.width / this._canvasSize.height;
            this._camera.updateProjectionMatrix();
            this._renderer.setSize(this._canvasSize.width, this._canvasSize.height);
            const animate = function () {
                var delta = this._clock.getDelta();
                this._cameraControls.update(delta);
                this._animationFrameIndex = requestAnimationFrame(animate);
                this._renderer.render(this._scene, this._camera);
            }.bind(this);
            animate();
        }
        /**
         * Updates the scene.
         * @param {!Object} currentStep Step datum.
         * @param {!HTMLDOMElement} domElement The HTML element used for event listeners.
         * @public
         */
        updateScene(currentStep, domElement) {
            let config = {};
            if ('config' in currentStep && currentStep.config) {
                config = JSON.parse(currentStep.config);
            }
            // This event is an opportunity for UI-responsible component (parent) to set
            // proper canvas size.
            this.dispatchEvent({ type: 'beforeUpdateScene' });
            const default_config = {
                camera: { cls: 'PerspectiveCamera', fov: 75, near: 0.1, far: 1000 },
                lights: [
                    { cls: 'AmbientLight', color: '#ffffff', intensity: 0.75 },
                    {
                        cls: 'DirectionalLight',
                        color: '#ffffff',
                        intensity: 0.75,
                        position: [0, -1, 2],
                    },
                ],
            };
            config = this._applyDefaults(config, default_config);
            this._createWorld(config, domElement);
            this._clearScene();
            this._createLights(this._scene, config);
            this._createGeometry(currentStep, config);
            this._createLayers();
            this.draw();
        }
        /**
         * Sets camera to default position and zoom.
         * @param {?THREE.Mesh} mesh Mesh to fit into viewport.
         * @public
         */
        resetView(mesh) {
            if (!this.isReady())
                return;
            this._cameraControls.reset();
            let nextMesh;
            if (!mesh && this._lastMesh) {
                nextMesh = this._lastMesh;
            }
            if (nextMesh) {
                this._fitObjectToViewport(nextMesh);
                // Store last mesh in case of resetView method called due to some events.
                this._lastMesh = nextMesh;
            }
            this._cameraControls.update();
        }
        /**
         * Creates geometry for current step data.
         * @param {!Object} currentStep Step datum.
         * @param {!Object} config Scene rendering configuration.
         * @private
         */
        _createGeometry(currentStep, config) {
            const mesh = currentStep.mesh;
            if (mesh.vertices && mesh.faces && mesh.faces.length) {
                this._createMesh(mesh, config);
            }
            else {
                this._createPointCloud(mesh, config);
            }
        }
        /**
         * Creates point cloud geometry for current step data.
         * @param {!Object} pointCloudData Object with point cloud data.
         * @param {!Object} config Scene rendering configuration.
         * @private
         */
        _createPointCloud(pointCloudData, config) {
            const points = pointCloudData.vertices;
            const colors = pointCloudData.colors;
            let defaultConfig = {
                material: {
                    cls: 'PointsMaterial',
                    size: 0.005,
                },
            };
            // Determine what colors will be used.
            if (colors && colors.length == points.length) {
                defaultConfig.material['vertexColors'] = VertexColors;
            }
            else {
                defaultConfig.material['color'] = this._runColor;
            }
            const pc_config = this._applyDefaults(config, defaultConfig);
            var geometry = new Geometry();
            points.forEach(function (point) {
                var p = new Vector3(point[0], point[1], point[2]);
                const scale = 1;
                p.x = point[0] * scale;
                p.y = point[1] * scale;
                p.z = point[2] * scale;
                geometry.vertices.push(p);
            });
            if (colors && colors.length == points.length) {
                colors.forEach(function (color) {
                    const c = new Color$7(color[0] / 255, color[1] / 255, color[2] / 255);
                    geometry.colors.push(c);
                });
            }
            var material = new THREE$1[pc_config.material.cls](pc_config.material);
            var mesh = new Points(geometry, material);
            this._scene.add(mesh);
            this._lastMesh = mesh;
        }
        /**
         * Creates mesh geometry for current step data.
         * @param {!THREE.Vector3} position Position of the camera.
         * @param {number} far Camera frustum far plane.
         * @param {!THREE.Vector3} target Point in space for camera to look at.
         * @public
         */
        setCameraViewpoint(position, far, target) {
            this._silent = true;
            this._camera.far = far;
            this._camera.position.set(position.x, position.y, position.z);
            this._camera.lookAt(target.clone());
            this._camera.updateProjectionMatrix();
            this._cameraControls.target = target.clone();
            this._cameraControls.update();
            this._silent = false;
        }
        /**
         * Triggered when camera position changed.
         * @private
         */
        _onCameraPositionChange(event) {
            if (this._silent)
                return;
            this.dispatchEvent({ type: 'cameraPositionChange', event: event });
        }
        /**
         * Positions camera on such distance from the object that the whole object is
         * visible.
         * @param {!THREE.Mesh} mesh Mesh to fit into viewport.
         * @private
         */
        _fitObjectToViewport(mesh) {
            // Small offset multiplicator to avoid edges of mesh touching edges of
            // viewport.
            const offset = 1.25;
            const boundingBox = new Box3();
            const center = new Vector3();
            const size = new Vector3();
            boundingBox.setFromObject(mesh);
            boundingBox.getCenter(center);
            boundingBox.getSize(size);
            const max_dim = Math.max(size.x, size.y, size.z);
            const fov = this._camera.fov * (Math.PI / 180);
            let camera_z = Math.abs(max_dim / (2 * Math.tan(fov / 2))) * offset;
            const min_z = boundingBox.min.z;
            // Make sure that even after arbitrary rotation mesh won't be clipped.
            const camera_to_far_edge = min_z < 0 ? -min_z + camera_z : camera_z - min_z;
            // Set camera position and orientation.
            this.setCameraViewpoint({ x: center.x, y: center.y, z: camera_z }, camera_to_far_edge * 3, center);
        }
        /**
         * Creates mesh geometry for current step data.
         * @param {!Object} meshData Object with mesh data.
         * @param {!Object} config Scene rendering configuration.
         * @private
         */
        _createMesh(meshData, config) {
            const vertices = meshData.vertices;
            const faces = meshData.faces;
            const colors = meshData.colors;
            const mesh_config = this._applyDefaults(config, {
                material: {
                    cls: 'MeshStandardMaterial',
                    color: '#a0a0a0',
                    roughness: 1,
                    metalness: 0,
                },
            });
            let geometry = new Geometry();
            vertices.forEach(function (point) {
                let p = new Vector3(point[0], point[1], point[2]);
                const scale = 1;
                p.x = point[0] * scale;
                p.y = point[1] * scale;
                p.z = point[2] * scale;
                geometry.vertices.push(p);
            });
            faces.forEach(function (face_indices) {
                let face = new Face3(face_indices[0], face_indices[1], face_indices[2]);
                if (colors && colors.length) {
                    const face_colors = [
                        colors[face_indices[0]],
                        colors[face_indices[1]],
                        colors[face_indices[2]],
                    ];
                    for (let i = 0; i < face_colors.length; i++) {
                        const vertex_color = face_colors[i];
                        let color = new Color$7(vertex_color[0] / 255, vertex_color[1] / 255, vertex_color[2] / 255);
                        face.vertexColors.push(color);
                    }
                }
                geometry.faces.push(face);
            });
            if (colors && colors.length) {
                mesh_config.material = mesh_config.material || {};
                mesh_config.material.vertexColors = VertexColors;
            }
            geometry.center();
            geometry.computeBoundingSphere();
            geometry.computeVertexNormals();
            let material = new THREE$1[mesh_config.material.cls](mesh_config.material);
            let mesh = new Mesh(geometry, material);
            mesh.castShadow = true;
            mesh.receiveShadow = true;
            this._scene.add(mesh);
            this._lastMesh = mesh;
        }
        /**
         * Creates lights for a given scene based on passed configuration.
         * @param {!Scene} scene Scene object to add lights to.
         * @param {!Object} config Scene rendering configuration.
         * @private
         */
        _createLights(scene, config) {
            for (let i = 0; i < config.lights.length; i++) {
                const light_config = config.lights[i];
                let light = new THREE$1[light_config.cls](light_config.color, light_config.intensity);
                if (light_config.position) {
                    light.position.set(light_config.position[0], light_config.position[1], light_config.position[2]);
                }
                scene.add(light);
            }
        }
    }

    let TfMeshLoaderImpl = class TfMeshLoaderImpl extends LegacyElementMixin(PolymerElement) {
        constructor() {
            super(...arguments);
            this.selectedView = 'all';
            this.active = false;
            this._colorScaleFunction = runsColorScale;
            this._steps = [];
            this._meshViewerAttached = false;
            this._cameraPositionInitialized = false;
            this._isMeshLoading = false;
        }
        get _runColor() {
            var run = this.run;
            return this._colorScaleFunction(run);
        }
        /**
         * Called by parent when component attached to DOM.
         * @public
         */
        connectedCallback() {
            super.connectedCallback();
            // Defer reloading until after we're attached, because that ensures that
            // the requestManager has been set from above. (Polymer is tricky
            // sometimes)
            this._dataProvider = new ArrayBufferDataProvider(this.requestManager);
            const meshViewer = new MeshViewer(this._runColor);
            meshViewer.addEventListener('beforeUpdateScene', this._updateCanvasSize.bind(this));
            meshViewer.addEventListener('cameraPositionChange', this._onCameraPositionChange.bind(this));
            this._meshViewer = meshViewer;
        }
        /**
         * Function to call when component must be reloaded.
         */
        reload() {
            if (!this.active || !this._dataProvider) {
                return;
            }
            this._isMeshLoading = true;
            this._dataProvider
                .reload(this.run, this.tag, this.sample)
                .then((steps) => {
                if (!steps)
                    return; // Happens when request was cancelled at some point.
                this._steps = steps;
                this._stepIndex = steps.length - 1;
            })
                .catch((error) => {
                if (!error || !error.code || error.code != ErrorCodes.CANCELLED) {
                    error = error || 'Response processing failed.';
                    throw new Error(error);
                }
            });
        }
        /**
         * Updates the scene.
         * @private
         */
        _updateScene() {
            const currentStep = this._currentStep;
            // Mesh data is not fetched yet. Please see `_maybeFetchMesh`.
            if (!currentStep || !currentStep.mesh)
                return;
            this._meshViewer.updateScene(currentStep, this);
            if (!this._cameraPositionInitialized) {
                this._meshViewer.resetView();
                this._cameraPositionInitialized = true;
            }
            if (!this._meshViewerAttached) {
                // Mesh viewer should be added to the dom once.
                this.shadowRoot.appendChild(this._meshViewer.getRenderer().domElement);
                this._meshViewerAttached = true;
            }
        }
        _debouncedFetchMesh() {
            this.debounce('fetchMesh', () => this._maybeFetchMesh(), 100);
        }
        _maybeFetchMesh() {
            return __awaiter(this, void 0, void 0, function* () {
                const currentStep = this._currentStep;
                if (!currentStep || currentStep.mesh || currentStep.meshFetching)
                    return;
                currentStep.meshFetching = true;
                this._isMeshLoading = true;
                try {
                    const meshData = yield this._dataProvider.fetchData(currentStep, this.run, this.tag, this.sample);
                    currentStep.mesh = meshData[0];
                    this.notifyPath('_currentStep.mesh');
                }
                catch (error) {
                    if (!error || !error.code || error.code != ErrorCodes.CANCELLED) {
                        error = error || 'Response processing failed.';
                        throw new Error(error);
                    }
                }
                finally {
                    this._isMeshLoading = false;
                    currentStep.meshFetching = false;
                }
            });
        }
        /**
         * Propagates camera position change event outside of the loader.
         * @private
         */
        _onCameraPositionChange() {
            if (!this._meshViewer.isReady())
                return;
            const event = new CustomEvent('camera-position-change', {
                detail: this._meshViewer.getCameraPosition(),
            });
            this.dispatchEvent(event);
        }
        /**
         * Creates mesh geometry for current step data.
         * @param position Position of the camera.
         * @param far Camera frustum far plane.
         * @param target Point in space for camera to look at.
         */
        setCameraViewpoint(position, far, target) {
            this._meshViewer.setCameraViewpoint(position, far, target);
        }
        /**
         * Updates size of the canvas.
         * @private
         */
        _updateCanvasSize() {
            const width = this.offsetWidth;
            const height = width; // Keep the whole mesh viewer square at all times.
            const headerHeight = this.$$('.tf-mesh-loader-header')
                .offsetHeight;
            const canvasSize = {
                width: width,
                height: height - headerHeight,
            };
            this._meshViewer.setCanvasSize(canvasSize);
        }
        /**
         * Re-renders component in the browser.
         */
        redraw() {
            this._updateCanvasSize();
            // Do not render if not in the DOM.
            if (!this.isConnected)
                return;
            this._meshViewer.draw();
        }
        _hasAtLeastOneStep(steps) {
            return !!steps && steps.length > 0;
        }
        _hasMultipleSteps(steps) {
            return !!steps && steps.length > 1;
        }
        get _currentStep() {
            var steps = this._steps;
            var stepIndex = this._stepIndex;
            return steps[stepIndex] || null;
        }
        get _stepValue() {
            const currentStep = this._currentStep;
            if (!currentStep)
                return 0;
            return currentStep.step;
        }
        get _currentWallTime() {
            const currentStep = this._currentStep;
            if (!currentStep)
                return '';
            return formatDate$1(currentStep.wall_time);
        }
        _getMaxStepIndex(steps) {
            return steps.length - 1;
        }
        _getSampleText(sample) {
            return String(sample + 1);
        }
        _hasMultipleSamples(ofSamples) {
            return ofSamples > 1;
        }
        _updateView() {
            var selectedView = this.selectedView;
            if (this._meshViewer && selectedView == 'all') {
                this._meshViewer.resetView();
            }
        }
        toLocaleString_(number) {
            // Shows commas (or locale-appropriate punctuation) for large numbers.
            return number.toLocaleString();
        }
    };
    TfMeshLoaderImpl.template = html `
    <tf-card-heading color="[[_runColor]]" class="tf-mesh-loader-header">
      <template is="dom-if" if="[[_hasMultipleSamples(ofSamples)]]">
        <div>sample: [[_getSampleText(sample)]] of [[ofSamples]]</div>
      </template>
      <template is="dom-if" if="[[_hasAtLeastOneStep(_steps)]]">
        <div class="heading-row">
          <div class="heading-label">
            step
            <span style="font-weight: bold"
              >[[toLocaleString_(_stepValue)]]</span
            >
          </div>
          <div class="heading-label heading-right">
            <template is="dom-if" if="[[_currentWallTime]]">
              [[_currentWallTime]]
            </template>
          </div>
          <div class="label right">
            <paper-spinner-lite active hidden$="[[!_isMeshLoading]]">
            </paper-spinner-lite>
          </div>
        </div>
      </template>
      <template is="dom-if" if="[[_hasMultipleSteps(_steps)]]">
        <div>
          <paper-slider
            id="steps"
            immediate-value="{{_stepIndex}}"
            max="[[_getMaxStepIndex(_steps)]]"
            max-markers="[[_getMaxStepIndex(_steps)]]"
            snaps
            step="1"
            value="{{_stepIndex}}"
          ></paper-slider>
        </div>
      </template>
    </tf-card-heading>
    <style>
      paper-slider {
        width: 100%;
        margin-left: 1px;
        margin-right: 1px;
      }
      .tf-mesh-loader-header {
        display: block;
        height: 105px;
      }
      [hidden] {
        display: none;
      }
    </style>
  `;
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfMeshLoaderImpl.prototype, "run", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfMeshLoaderImpl.prototype, "tag", void 0);
    __decorate([
        property({ type: Number }),
        __metadata("design:type", Number)
    ], TfMeshLoaderImpl.prototype, "sample", void 0);
    __decorate([
        property({ type: Number }),
        __metadata("design:type", Number)
    ], TfMeshLoaderImpl.prototype, "ofSamples", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfMeshLoaderImpl.prototype, "selectedView", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfMeshLoaderImpl.prototype, "active", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", RequestManager)
    ], TfMeshLoaderImpl.prototype, "requestManager", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", MeshViewer)
    ], TfMeshLoaderImpl.prototype, "_meshViewer", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", ArrayBufferDataProvider)
    ], TfMeshLoaderImpl.prototype, "_dataProvider", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfMeshLoaderImpl.prototype, "_colorScaleFunction", void 0);
    __decorate([
        property({
            type: Array,
            notify: true,
        }),
        __metadata("design:type", Array)
    ], TfMeshLoaderImpl.prototype, "_steps", void 0);
    __decorate([
        property({
            type: Number,
            notify: true,
        }),
        __metadata("design:type", Number)
    ], TfMeshLoaderImpl.prototype, "_stepIndex", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfMeshLoaderImpl.prototype, "_meshViewerAttached", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfMeshLoaderImpl.prototype, "_cameraPositionInitialized", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfMeshLoaderImpl.prototype, "_isMeshLoading", void 0);
    __decorate([
        computed('run'),
        __metadata("design:type", String),
        __metadata("design:paramtypes", [])
    ], TfMeshLoaderImpl.prototype, "_runColor", null);
    __decorate([
        observe('run', 'tag', 'active', '_dataProvider', '_meshViewer'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfMeshLoaderImpl.prototype, "reload", null);
    __decorate([
        observe('_currentStep.*', '_meshViewer'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfMeshLoaderImpl.prototype, "_updateScene", null);
    __decorate([
        observe('_currentStep'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfMeshLoaderImpl.prototype, "_debouncedFetchMesh", null);
    __decorate([
        computed('_steps', '_stepIndex'),
        __metadata("design:type", Object),
        __metadata("design:paramtypes", [])
    ], TfMeshLoaderImpl.prototype, "_currentStep", null);
    __decorate([
        computed('_currentStep'),
        __metadata("design:type", Number),
        __metadata("design:paramtypes", [])
    ], TfMeshLoaderImpl.prototype, "_stepValue", null);
    __decorate([
        computed('_currentStep'),
        __metadata("design:type", String),
        __metadata("design:paramtypes", [])
    ], TfMeshLoaderImpl.prototype, "_currentWallTime", null);
    __decorate([
        observe('selectedView'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfMeshLoaderImpl.prototype, "_updateView", null);
    TfMeshLoaderImpl = __decorate([
        customElement('tf-mesh-loader')
    ], TfMeshLoaderImpl);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    let MeshDashboard = class MeshDashboard extends PolymerElement {
        constructor() {
            super();
            this.reloadOnReady = true;
            this._tagFilter = '.*';
            this._selectedView = 'all';
            this._requestManager = new RequestManager();
            window.addEventListener('resize', () => {
                this._handleWindowResize();
            }, false);
            if (this.reloadOnReady)
                this.reload();
        }
        _getAllChildren() {
            return Array.from(this.shadowRoot.querySelectorAll('tf-mesh-loader'));
        }
        _onCameraPositionChanged(event) {
            if (this._selectedView == 'share') {
                this._getAllChildren().forEach((g) => {
                    if (event.target == g)
                        return; // Do not update trigger camera.
                    g.setCameraViewpoint(event.detail.position, event.detail.far, event.detail.target);
                });
            }
        }
        _shouldOpen(index) {
            return index <= 2;
        }
        reload() {
            this._fetchTags().then(this._reloadMeshes.bind(this));
        }
        _handleWindowResize() {
            this._getAllChildren().forEach((g) => {
                g.redraw();
            });
        }
        _fetchTags() {
            const url = getRouter().pluginRoute('mesh', '/tags');
            return this._requestManager.request(url).then((runToTagInfo) => {
                if (lodash.isEqual(runToTagInfo, this._runToTagInfo)) {
                    // No need to update anything if there are no changes.
                    return;
                }
                const runToTag = lodash.mapValues(runToTagInfo, (x) => Object.keys(x));
                const tags = getTags(runToTag);
                this._dataNotFound = tags.length === 0;
                this._runToTagInfo = runToTagInfo;
            });
        }
        _reloadMeshes() {
            this._getAllChildren().forEach((g) => {
                g.reload();
            });
        }
        get _categories() {
            var runToTagInfo = this._runToTagInfo;
            var selectedRuns = this._selectedRuns;
            var tagFilter = this._tagFilter;
            const runToTag = lodash.mapValues(runToTagInfo, (x) => Object.keys(x));
            const baseCategories = categorizeRunTagCombinations(runToTag, selectedRuns, tagFilter);
            function explodeItem(item) {
                const samples = runToTagInfo[item.run][item.tag].samples;
                return lodash.range(samples).map((i) => Object.assign({}, item, {
                    sample: i,
                    ofSamples: samples,
                }));
            }
            const withSamples = baseCategories.map((category) => Object.assign({}, category, {
                items: [].concat.apply([], category.items.map(explodeItem)),
            }));
            return withSamples;
        }
    };
    MeshDashboard.template = html `
    <tf-dashboard-layout>
      <div slot="sidebar" class="all-controls">
        <div class="settings">
          <div class="sidebar-section view-control">
            <h3 class="title">Point of view</h3>
            <div>
              <paper-radio-group
                id="view-radio-group"
                selected="{{_selectedView}}"
              >
                <paper-radio-button id="all-radio-button" name="all">
                  Display all points
                </paper-radio-button>
                <paper-tooltip
                  animation-delay="0"
                  for="all-radio-button"
                  position="right"
                  offset="0"
                >
                  Zoom and center camera to display all points at once. Note,
                  that some points could be too far (i.e. too small) to be
                  visible.
                </paper-tooltip>
                <paper-radio-button id="user-radio-button" name="user">
                  Current view
                </paper-radio-button>
                <paper-tooltip
                  animation-delay="0"
                  for="user-radio-button"
                  position="right"
                  offset="0"
                >
                  Keep current camera position and zoom level.
                </paper-tooltip>
                <paper-radio-button id="share-radio-button" name="share">
                  Share viewpoint
                </paper-radio-button>
                <paper-tooltip
                  animation-delay="0"
                  for="share-radio-button"
                  position="right"
                  offset="0"
                >
                  Share viewpoint among all cameras.
                </paper-tooltip>
              </paper-radio-group>
            </div>
          </div>
        </div>
        <div class="sidebar-section runs-selector">
          <tf-runs-selector selected-runs="{{_selectedRuns}}">
          </tf-runs-selector>
        </div>
      </div>
      <div slot="center">
        <template is="dom-if" if="[[_dataNotFound]]">
          <div class="no-data-warning">
            <h3>No point cloud data was found.</h3>
            <p>Probable causes:</p>
            <ul>
              <li>
                You haven’t written any point cloud data to your event files.
              </li>
              <li>TensorBoard can’t find your event files.</li>
            </ul>

            <p>
              If you’re new to using TensorBoard, and want to find out how to
              add data and set up your event files, check out the
              <a
                href="https://github.com/tensorflow/tensorboard/blob/master/README.md"
                >README</a
              >
              and perhaps the
              <a
                href="https://www.tensorflow.org/get_started/summaries_and_tensorboard"
                >TensorBoard tutorial</a
              >.
            </p>

            <p>
              If you think TensorBoard is configured properly, please see
              <a
                href="https://github.com/tensorflow/tensorboard/blob/master/README.md#my-tensorboard-isnt-showing-any-data-whats-wrong"
                >the section of the README devoted to missing data problems</a
              >
              and consider filing an issue on GitHub.
            </p>
          </div>
        </template>
        <template is="dom-if" if="[[!_dataNotFound]]">
          <tf-tag-filterer tag-filter="{{_tagFilter}}"></tf-tag-filterer>
          <template is="dom-repeat" items="[[_categories]]" as="category">
            <tf-category-paginated-view
              category="[[category]]"
              initial-opened="[[_shouldOpen(index)]]"
            >
              <template>
                <tf-mesh-loader
                  active="[[active]]"
                  selected-view="[[_selectedView]]"
                  run="[[item.run]]"
                  tag="[[item.tag]]"
                  sample="[[item.sample]]"
                  of-samples="[[item.ofSamples]]"
                  request-manager="[[_requestManager]]"
                  class="tf-mesh-loader-container"
                  on-camera-position-change="_onCameraPositionChanged"
                >
                </tf-mesh-loader>
              </template>
            </tf-category-paginated-view>
          </template>
        </template>
      </div>
    </tf-dashboard-layout>

    <style include="dashboard-style"></style>
    <style>
      .no-data-warning {
        max-width: 540px;
        margin: 80px auto 0 auto;
      }
      paper-radio-button {
        display: block;
        padding: 5px;
      }
      .sidebar-section h3.title {
        color: var(--paper-grey-800);
        margin: 0;
        font-weight: normal;
        font-size: 14px;
        margin-bottom: 5px;
      }

      .runs-selector {
        flex-grow: 1;
      }

      tf-runs-selector {
        display: flex;
      }

      .view-control {
        display: block !important;
      }

      .view-control h3.title {
        padding-top: 16px;
        padding-bottom: 16px;
      }

      .allcontrols .view-control paper-radio-group {
        margin-top: 5px;
      }
      /* Layout must be horizontal, i.e. items arranged in a row. If items cannot fit in a row,
       * they should be moved to next line. All items must be square at all times. Minimum size of
       * the item is 480px. This means that maximum size of the item must be 480px + 479px = 959px.
       * */
      .horizontal {
        display: flex;
        flex-direction: row;
        flex-wrap: wrap;
      }
      tf-mesh-loader {
        width: 480px;
        flex-basis: 480px;
        flex-grow: 1;
        display: block;
      }
    </style>
  `;
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], MeshDashboard.prototype, "reloadOnReady", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], MeshDashboard.prototype, "_selectedRuns", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], MeshDashboard.prototype, "_runToTagInfo", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], MeshDashboard.prototype, "_dataNotFound", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], MeshDashboard.prototype, "_tagFilter", void 0);
    __decorate([
        property({
            type: String,
            notify: true,
        }),
        __metadata("design:type", String)
    ], MeshDashboard.prototype, "_selectedView", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], MeshDashboard.prototype, "_requestManager", void 0);
    __decorate([
        computed('_runToTagInfo', '_selectedRuns', '_tagFilter'),
        __metadata("design:type", Array),
        __metadata("design:paramtypes", [])
    ], MeshDashboard.prototype, "_categories", null);
    MeshDashboard = __decorate([
        customElement('mesh-dashboard'),
        __metadata("design:paramtypes", [])
    ], MeshDashboard);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    let TfPrCurveCard = class TfPrCurveCard extends PolymerElement {
        constructor() {
            super(...arguments);
            this._expanded = false;
            this._runToPrCurveEntry = {};
            this._previousRunToPrCurveEntry = {};
            this._colorScaleFunction = { scale: runsColorScale };
            this._canceller = new Canceller();
            this._xComponentsCreationMethod = () => {
                const scale = new src$2.Scales.Linear();
                return {
                    scale: scale,
                    axis: new src$2.Axes.Numeric(scale, 'bottom'),
                    accessor: (d) => d.recall,
                };
            };
            this._yValueAccessor = (d) => d.precision;
            this._tooltipColumns = (() => {
                const valueFormatter = multiscaleFormatter(Y_TOOLTIP_FORMATTER_PRECISION);
                const formatValueOrNaN = (x) => (isNaN(x) ? 'NaN' : valueFormatter(x));
                return [
                    {
                        title: 'Run',
                        evaluate: (d) => d.dataset.metadata().name,
                    },
                    {
                        title: 'Threshold',
                        evaluate: (d) => formatValueOrNaN(d.datum.thresholds),
                    },
                    {
                        title: 'Precision',
                        evaluate: (d) => formatValueOrNaN(d.datum.precision),
                    },
                    {
                        title: 'Recall',
                        evaluate: (d) => formatValueOrNaN(d.datum.recall),
                    },
                    {
                        title: 'TP',
                        evaluate: (d) => d.datum.true_positives,
                    },
                    {
                        title: 'FP',
                        evaluate: (d) => d.datum.false_positives,
                    },
                    {
                        title: 'TN',
                        evaluate: (d) => d.datum.true_negatives,
                    },
                    {
                        title: 'FN',
                        evaluate: (d) => d.datum.false_negatives,
                    },
                ];
            })();
            this._seriesDataFields = [
                'thresholds',
                'precision',
                'recall',
                'true_positives',
                'false_positives',
                'true_negatives',
                'false_negatives',
            ];
            this._defaultXRange = [-0.05, 1.05];
            this._defaultYRange = [-0.05, 1.05];
            this._requestData = (items, onLoad, onFinish) => {
                const router = getRouter();
                const baseUrl = router.pluginRoute('pr_curves', '/pr_curves');
                Promise.all(items.map((item) => {
                    const run = item;
                    const tag = this.tag;
                    const url = addParams(baseUrl, { tag, run });
                    return this.requestManager
                        .request(url)
                        .then((data) => void onLoad({ item, data }));
                })).finally(() => void onFinish());
            };
            this._smoothingEnabled = false;
        }
        _createProcessDataFunction() {
            // This function is called when data is received from the backend.
            return (chart, run, data) => {
                // The data maps a single run to a series of data. We merge that data
                // with the data already fetched.
                this.set('_runToDataOverTime', Object.assign({}, this._runToDataOverTime, data));
            };
        }
        _computeRunColor(run) {
            return runsColorScale(run);
        }
        connectedCallback() {
            super.connectedCallback();
            // Defer reloading until after we're attached, because that ensures that
            // the requestManager has been set from above. (Polymer is tricky
            // sometimes)
            this._attached = true;
            this.reload();
        }
        _getChartDataLoader() {
            // tslint:disable-next-line:no-unnecessary-type-assertion
            return this.shadowRoot.querySelector('tf-line-chart-data-loader'); // TfLineChartDataLoader
        }
        reload() {
            if (!this._attached) {
                return;
            }
            if (this.runs.length === 0) {
                // There are no selected runs.
                this.set('_runToDataOverTime', {});
                return;
            }
            this._getChartDataLoader().reload();
        }
        _setChartData() {
            var runToPrCurveEntry = this._runToPrCurveEntry;
            var previousRunToPrCurveEntry = this._previousRunToPrCurveEntry;
            var setOfRelevantRuns = this._setOfRelevantRuns;
            lodash.forOwn(runToPrCurveEntry, (entry, run) => {
                const previousEntry = previousRunToPrCurveEntry[run];
                if (previousEntry && runToPrCurveEntry[run].step === previousEntry.step) {
                    // The PR curve for this run does not need to be updated.
                    return;
                }
                if (!setOfRelevantRuns[run]) {
                    // Clear this dataset - the user has unselected it.
                    this._clearSeriesData(run);
                    return;
                }
                this._updateSeriesDataForRun(run, entry);
            });
        }
        _updateSeriesDataForRun(run, entryForOneStep) {
            // Reverse the values so they are plotted in order. The logic within
            // the line chart for associating information to show in the tooltip
            // with points in the chart assumes that the series data is ordered
            // by the variable on the X axis. If the values are not in order,
            // tooltips will not work because the tooltip will always be stuck on
            // one side of the chart.
            const fieldsToData = lodash.reduce(this._seriesDataFields, (result, field) => {
                result[field] = entryForOneStep[field].slice().reverse();
                return result;
            }, {});
            // The number of series data is equal to the number of entries in any of
            // the fields. We just use the first field to gauge the length.
            const seriesData = new Array(fieldsToData[this._seriesDataFields[0]].length);
            // Create a list of data for visualization.
            for (let i = 0; i < seriesData.length; i++) {
                seriesData[i] = lodash.mapValues(fieldsToData, (values) => values[i]);
            }
            const loader = this._getChartDataLoader();
            loader.setSeriesData(run, seriesData);
            loader.commitChanges();
        }
        _clearSeriesData(run) {
            // Clears data for a run in the chart.
            const loader = this._getChartDataLoader();
            loader.setSeriesData(run, []);
            loader.commitChanges();
        }
        _updateRunToPrCurveEntry() {
            var runToDataOverTime = this._runToDataOverTime;
            var runToStepCap = this.runToStepCap;
            const runToEntry = {};
            lodash.forOwn(runToDataOverTime, (entries, run) => {
                if (!entries || !entries.length) {
                    return;
                }
                runToEntry[run] = this._computeEntryClosestOrEqualToStepCap(runToStepCap[run], entries);
            });
            // Set the previous PR curve entry so we can later compare and only
            // redraw for runs that changed in step.
            this.set('_previousRunToPrCurveEntry', this._runToPrCurveEntry);
            this.set('_runToPrCurveEntry', runToEntry);
        }
        _notifyDataChange() {
            var runToDataOverTime = this._runToDataOverTime;
            if (this.onDataChange) {
                this.onDataChange(runToDataOverTime);
            }
        }
        _computeEntryClosestOrEqualToStepCap(stepCap, entries) {
            const entryIndex = Math.min(lodash.sortedIndex(entries.map((entry) => entry.step), stepCap), entries.length - 1);
            return entries[entryIndex];
        }
        get _runsWithStepAvailable() {
            var runs = this.runs;
            var actualPrCurveEntryPerRun = this._runToPrCurveEntry;
            return lodash.filter(runs, (run) => actualPrCurveEntryPerRun[run]).sort();
        }
        get _setOfRelevantRuns() {
            var runsWithStepAvailable = this._runsWithStepAvailable;
            const setOfRelevantRuns = {};
            lodash.forEach(runsWithStepAvailable, (run) => {
                setOfRelevantRuns[run] = true;
            });
            return setOfRelevantRuns;
        }
        _computeCurrentStepForRun(runToPrCurveEntry, run) {
            // If there is no data for the run, then the run is not being shown, so
            // the return value is not used. We return null as a reasonable value.
            const entry = runToPrCurveEntry[run];
            return entry ? entry.step : null;
        }
        _computeCurrentWallTimeForRun(runToPrCurveEntry, run) {
            const entry = runToPrCurveEntry[run];
            // If there is no data for the run, then the run is not being shown, so
            // the return value is not used. We return null as a reasonable value.
            if (!entry) {
                return null;
            }
            return new Date(entry.wall_time * 1000).toString();
        }
        _toggleExpanded(e) {
            this.set('_expanded', !this._expanded);
            this.redraw();
        }
        _resetDomain() {
            this._getChartDataLoader().resetDomain();
        }
        redraw() {
            this._getChartDataLoader().redraw();
        }
    };
    TfPrCurveCard.template = html `
    <tf-card-heading
      tag="[[tag]]"
      display-name="[[tagMetadata.displayName]]"
      description="[[tagMetadata.description]]"
    ></tf-card-heading>

    <tf-line-chart-data-loader
      x-components-creation-method="[[_xComponentsCreationMethod]]"
      y-value-accessor="[[_yValueAccessor]]"
      tooltip-columns="[[_tooltipColumns]]"
      color-scale="[[_colorScaleFunction]]"
      default-x-range="[[_defaultXRange]]"
      default-y-range="[[_defaultYRange]]"
      smoothing-enabled="[[_smoothingEnabled]]"
      request-manager="[[requestManager]]"
      data-to-load="[[runs]]"
      data-series="[[runs]]"
      load-key="[[tag]]"
      request-data="[[_requestData]]"
      load-data-callback="[[_createProcessDataFunction()]]"
      active="[[active]]"
    ></tf-line-chart-data-loader>

    <div id="buttons-row">
      <paper-icon-button
        selected$="[[_expanded]]"
        icon="fullscreen"
        on-tap="_toggleExpanded"
      ></paper-icon-button>
      <paper-icon-button
        icon="settings-overscan"
        on-tap="_resetDomain"
        title="Reset axes to [0, 1]."
      ></paper-icon-button>
    </div>

    <div id="step-legend">
      <template is="dom-repeat" items="[[_runsWithStepAvailable]]" as="run">
        <div class="legend-row">
          <div
            class="color-box"
            style="background: [[_computeRunColor(run)]];"
          ></div>
          [[run]] is at
          <span class="step-label-text">
            step [[_computeCurrentStepForRun(_runToPrCurveEntry, run)]] </span
          ><br />
          <span class="wall-time-label-text">
            ([[_computeCurrentWallTimeForRun(_runToPrCurveEntry, run)]])
          </span>
        </div>
      </template>
    </div>

    <style>
      :host {
        display: flex;
        flex-direction: column;
        width: 500px;
        margin-right: 10px;
        margin-bottom: 25px;
      }
      :host([_expanded]) {
        width: 100%;
      }
      tf-line-chart-data-loader {
        height: 300px;
        position: relative;
      }
      :host([_expanded]) tf-line-chart-data-loader {
        height: 600px;
      }
      #buttons-row {
        display: flex;
        flex-direction: row;
      }
      #buttons-row paper-icon-button {
        color: #2196f3;
        border-radius: 100%;
        width: 32px;
        height: 32px;
        padding: 4px;
      }
      #buttons-row paper-icon-button[selected] {
        background: var(--tb-ui-light-accent);
      }
      #step-legend {
        box-sizing: border-box;
        font-size: 0.8em;
        max-height: 200px;
        overflow-y: auto;
        padding: 0 0 0 10px;
        width: 100%;
      }
      .legend-row {
        margin: 5px 0 5px 0;
        width: 100%;
      }
      .color-box {
        display: inline-block;
        border-radius: 1px;
        width: 10px;
        height: 10px;
      }
      .step-label-text {
        font-weight: bold;
      }
      .wall-time-label-text {
        color: #888;
        font-size: 0.8em;
      }
    </style>
  `;
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfPrCurveCard.prototype, "runs", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfPrCurveCard.prototype, "tag", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfPrCurveCard.prototype, "tagMetadata", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfPrCurveCard.prototype, "runToStepCap", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", RequestManager)
    ], TfPrCurveCard.prototype, "requestManager", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfPrCurveCard.prototype, "active", void 0);
    __decorate([
        property({
            type: Boolean,
            reflectToAttribute: true,
        }),
        __metadata("design:type", Boolean)
    ], TfPrCurveCard.prototype, "_expanded", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfPrCurveCard.prototype, "_runToPrCurveEntry", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfPrCurveCard.prototype, "_previousRunToPrCurveEntry", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfPrCurveCard.prototype, "_runToDataOverTime", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Function)
    ], TfPrCurveCard.prototype, "onDataChange", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfPrCurveCard.prototype, "_colorScaleFunction", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Canceller)
    ], TfPrCurveCard.prototype, "_canceller", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfPrCurveCard.prototype, "_attached", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfPrCurveCard.prototype, "_xComponentsCreationMethod", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfPrCurveCard.prototype, "_yValueAccessor", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfPrCurveCard.prototype, "_tooltipColumns", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfPrCurveCard.prototype, "_seriesDataFields", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfPrCurveCard.prototype, "_defaultXRange", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfPrCurveCard.prototype, "_defaultYRange", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Function)
    ], TfPrCurveCard.prototype, "_requestData", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfPrCurveCard.prototype, "_smoothingEnabled", void 0);
    __decorate([
        observe('runs', 'tag'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfPrCurveCard.prototype, "reload", null);
    __decorate([
        observe('_runToPrCurveEntry', '_previousRunToPrCurveEntry', '_setOfRelevantRuns'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfPrCurveCard.prototype, "_setChartData", null);
    __decorate([
        observe('_runToDataOverTime', 'runToStepCap'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfPrCurveCard.prototype, "_updateRunToPrCurveEntry", null);
    __decorate([
        observe('_runToDataOverTime'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfPrCurveCard.prototype, "_notifyDataChange", null);
    __decorate([
        computed('runs', '_runToPrCurveEntry'),
        __metadata("design:type", Array),
        __metadata("design:paramtypes", [])
    ], TfPrCurveCard.prototype, "_runsWithStepAvailable", null);
    __decorate([
        computed('_runsWithStepAvailable'),
        __metadata("design:type", Object),
        __metadata("design:paramtypes", [])
    ], TfPrCurveCard.prototype, "_setOfRelevantRuns", null);
    TfPrCurveCard = __decorate([
        customElement('tf-pr-curve-card')
    ], TfPrCurveCard);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    let TfPrCurveStepsSelector = 
    // tslint:disable-next-line:no-unused-variable
    class TfPrCurveStepsSelector extends PolymerElement {
        constructor() {
            super(...arguments);
            this._runToStepIndex = {};
        }
        _computeColorForRun(run) {
            return runsColorScale(run);
        }
        _computeTimeTextForRun(runToAvailableTimeEntries, runToStepIndex, run, timeDisplayType) {
            const stepIndex = runToStepIndex[run];
            if (!lodash.isNumber(stepIndex)) {
                // The step is not known yet.
                return '';
            }
            const entries = runToAvailableTimeEntries[run];
            if (!entries) {
                // No PR curve data has been received from the server yet.
                return '';
            }
            const value = entries[stepIndex][timeDisplayType];
            if (timeDisplayType === 'step') {
                return `step ${value}`;
            }
            else if (timeDisplayType === 'relative') {
                // Return the time using the units that are most apt.
                if (value < 1) {
                    return `${(value * 1000).toFixed(2)} ms`;
                }
                return `${value.toFixed(2)} s`;
            }
            else if (timeDisplayType === 'wall_time') {
                return new Date(value * 1000).toString();
            }
            throw new Error(`The display type of ${timeDisplayType} is not recognized.`);
        }
        _sliderValueChanged(event) {
            const run = event.target.dataset.run;
            const val = event.target.immediateValue;
            const newRunToStepIndex = Object.assign({}, this._runToStepIndex);
            // Slider emits a changed event before it goes away with an
            // immediateValue of NaN.
            if (isNaN(val)) {
                delete newRunToStepIndex[run];
            }
            else {
                newRunToStepIndex[run] = event.target.immediateValue;
            }
            this._runToStepIndex = newRunToStepIndex;
        }
        _computeMaxStepIndexForRun(runToAvailableTimeEntries, run) {
            const entries = runToAvailableTimeEntries[run];
            return entries && entries.length ? entries.length - 1 : 0;
        }
        _updateStepsForNewRuns() {
            var runToAvailableTimeEntries = this.runToAvailableTimeEntries;
            // The mapping from run to available time entries just changed.
            const newRunToStepIndex = Object.assign({}, this._runToStepIndex);
            lodash.forOwn(runToAvailableTimeEntries, (entries, run) => {
                if (!lodash.isNumber(newRunToStepIndex[run])) {
                    // This run had previously lacked a slider. Initially set the slider
                    // for the run to point to the last step.
                    newRunToStepIndex[run] = entries.length - 1;
                }
            });
            this._runToStepIndex = newRunToStepIndex;
        }
        _getStep(runToStepIndex, run) {
            if (!this._runToStepIndex)
                return 0;
            return this._runToStepIndex[run];
        }
        _computeRunToStep(runToAvailableTimeEntries, runToStepIndex) {
            const runToStep = {};
            lodash.forOwn(runToStepIndex, (index, run) => {
                const entries = runToAvailableTimeEntries[run];
                if (!entries) {
                    return;
                }
                runToStep[run] = entries[index].step;
            });
            return runToStep;
        }
        get _runsWithSliders() {
            var runs = this.runs;
            var runToAvailableTimeEntries = this.runToAvailableTimeEntries;
            // Only create sliders for selected runs with steps available.
            return runs.filter((r) => runToAvailableTimeEntries[r]);
        }
    };
    TfPrCurveStepsSelector.template = html `
    <template is="dom-repeat" items="[[_runsWithSliders]]" as="run">
      <div class="run-widget">
        <div class="run-display-container">
          <div
            class="run-color-box"
            style="background:[[_computeColorForRun(run)]];"
          ></div>
          <div class="run-text">[[run]]</div>
        </div>
        <div class="step-display-container">
          [[_computeTimeTextForRun(runToAvailableTimeEntries, _runToStepIndex,
          run, timeDisplayType)]]
        </div>
        <paper-slider
          data-run$="[[run]]"
          step="1"
          type="number"
          min="0"
          max="[[_computeMaxStepIndexForRun(runToAvailableTimeEntries, run)]]"
          value="[[_getStep(_runToStepIndex, run)]]"
          on-immediate-value-changed="_sliderValueChanged"
        ></paper-slider>
      </div>
    </template>
    <style>
      .run-widget {
        margin: 10px 0 0 0;
      }
      paper-slider {
        margin: -8px 0 0 -15px;
        width: 100%;
      }
      .step-display-container {
        font-size: 0.9em;
        margin: 0 15px 0 0;
      }
      .run-text {
        display: inline-block;
      }
      .run-color-box {
        width: 12px;
        height: 12px;
        border-radius: 3px;
        display: inline-block;
      }
    </style>
  `;
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfPrCurveStepsSelector.prototype, "runs", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfPrCurveStepsSelector.prototype, "runToAvailableTimeEntries", void 0);
    __decorate([
        property({
            type: Object,
            notify: true,
            computed: '_computeRunToStep(runToAvailableTimeEntries, _runToStepIndex)',
        }),
        __metadata("design:type", Object)
    ], TfPrCurveStepsSelector.prototype, "runToStep", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfPrCurveStepsSelector.prototype, "timeDisplayType", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfPrCurveStepsSelector.prototype, "_runToStepIndex", void 0);
    __decorate([
        observe('runToAvailableTimeEntries'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfPrCurveStepsSelector.prototype, "_updateStepsForNewRuns", null);
    __decorate([
        computed('runs', 'runToAvailableTimeEntries'),
        __metadata("design:type", Array),
        __metadata("design:paramtypes", [])
    ], TfPrCurveStepsSelector.prototype, "_runsWithSliders", null);
    TfPrCurveStepsSelector = __decorate([
        customElement('tf-pr-curve-steps-selector')
        // tslint:disable-next-line:no-unused-variable
    ], TfPrCurveStepsSelector);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    let TfPrCurveDashboard = 
    // tslint:disable-next-line:no-unused-variable
    class TfPrCurveDashboard extends LegacyElementMixin(PolymerElement) {
        constructor() {
            super(...arguments);
            this.reloadOnReady = true;
            this._timeDisplayType = 'step';
            this._selectedRuns = [];
            this._runToTagInfo = {};
            this._tagToRunToData = {};
            this._getCategoryItemKey = (item) => item.tag;
            this._requestManager = new RequestManager();
            this._step = 0;
        }
        ready() {
            super.ready();
            if (this.reloadOnReady)
                this.reload();
        }
        reload() {
            Promise.all([this._fetchTags()]).then(() => {
                this._reloadCards();
            });
        }
        _shouldOpen(index) {
            return index <= 2;
        }
        _fetchTags() {
            const url = getRouter().pluginRoute('pr_curves', '/tags');
            return this._requestManager.request(url).then((runToTagInfo) => {
                if (lodash.isEqual(runToTagInfo, this._runToTagInfo)) {
                    // No need to update anything if there are no changes.
                    return;
                }
                const runToTag = lodash.mapValues(runToTagInfo, (o) => lodash.keys(o));
                const tags = getTags(runToTag);
                this.set('_dataNotFound', tags.length === 0);
                this.set('_runToTagInfo', runToTagInfo);
                this.async(() => {
                    // See the comment above `_categoriesDomReady`.
                    this.set('_categoriesDomReady', true);
                });
            });
        }
        _reloadCards() {
            lodash.forEach(this.root.querySelectorAll('tf-pr-curve-card'), (card) => {
                card.reload();
            });
        }
        get _categories() {
            var runToTagInfo = this._runToTagInfo;
            var selectedRuns = this._selectedRuns;
            var tagFilter = this._tagFilter;
            const runToTag = lodash.mapValues(runToTagInfo, (x) => Object.keys(x));
            return categorizeTags(runToTag, selectedRuns, tagFilter);
        }
        get _relevantSelectedRuns() {
            var selectedRuns = this._selectedRuns;
            var runToTagInfo = this._runToTagInfo;
            return selectedRuns.filter((run) => runToTagInfo[run]);
        }
        _tagMetadata(runToTagsInfo, runs, tag) {
            const runToTagInfo = {};
            runs.forEach((run) => {
                runToTagInfo[run] = runToTagsInfo[run][tag];
            });
            // All PR curve tags include the `/pr_curves` suffix. We can trim
            // that from the display name.
            const defaultDisplayName = tag.replace(/\/pr_curves$/, '');
            return aggregateTagInfo(runToTagInfo, defaultDisplayName);
        }
        _createDataChangeCallback(tag) {
            return (runToData) => {
                this.set('_tagToRunToData', Object.assign(Object.assign({}, this._tagToRunToData), { [tag]: runToData }));
            };
        }
        get _runToAvailableTimeEntries() {
            var tagToRunToData = this._tagToRunToData;
            const canonicalTag = {}; // map from run to canonical tag name
            for (const [tag, runToData] of Object.entries(tagToRunToData)) {
                for (const [run] of Object.entries(runToData)) {
                    // arbitrary stable ordering: smallest tag name is canonical
                    if (canonicalTag[run] == null || tag < canonicalTag[run]) {
                        canonicalTag[run] = tag;
                    }
                }
            }
            const result = {};
            for (const [run, tag] of Object.entries(canonicalTag)) {
                const data = tagToRunToData[tag][run];
                result[run] = data.map((d) => ({
                    step: d.step,
                    wall_time: d.wall_time,
                    relative: d.wall_time - data[0].wall_time,
                }));
            }
            return result;
        }
    };
    TfPrCurveDashboard.template = html `
    <tf-dashboard-layout>
      <div class="sidebar" slot="sidebar">
        <div class="settings">
          <div class="sidebar-section">
            <tf-option-selector
              id="time-type-selector"
              name="Time Display Type"
              selected-id="{{_timeDisplayType}}"
            >
              <paper-button id="step">step</paper-button>
              <!--
            -->
              <paper-button id="relative">relative</paper-button>
              <!--
            -->
              <paper-button id="wall_time">wall</paper-button>
            </tf-option-selector>
          </div>
          <template is="dom-if" if="[[_runToAvailableTimeEntries]]">
            <div class="sidebar-section" id="steps-selector-container">
              <tf-pr-curve-steps-selector
                runs="[[_relevantSelectedRuns]]"
                run-to-step="{{_runToStep}}"
                run-to-available-time-entries="[[_runToAvailableTimeEntries]]"
                time-display-type="[[_timeDisplayType]]"
              >
              </tf-pr-curve-steps-selector>
            </div>
          </template>
        </div>
        <div class="sidebar-section runs-selector">
          <tf-runs-selector selected-runs="{{_selectedRuns}}">
          </tf-runs-selector>
        </div>
      </div>
      <div class="center" slot="center">
        <template is="dom-if" if="[[_dataNotFound]]">
          <div class="no-data-warning">
            <h3>No precision–recall curve data was found.</h3>
            <p>Probable causes:</p>
            <ul>
              <li>
                You haven’t written any precision–recall data to your event
                files.
              </li>
              <li>TensorBoard can’t find your event files.</li>
            </ul>
            <p>
              If you’re new to using TensorBoard, and want to find out how to
              add data and set up your event files, check out the
              <a
                href="https://github.com/tensorflow/tensorboard/blob/master/README.md"
                >README</a
              >
              and perhaps the
              <a
                href="https://www.tensorflow.org/get_started/summaries_and_tensorboard"
                >TensorBoard tutorial</a
              >.
            </p>

            <p>
              If you think TensorBoard is configured properly, please see
              <a
                href="https://github.com/tensorflow/tensorboard/blob/master/README.md#my-tensorboard-isnt-showing-any-data-whats-wrong"
                >the section of the README devoted to missing data problems</a
              >
              and consider filing an issue on GitHub.
            </p>
          </div>
        </template>
        <template is="dom-if" if="[[!_dataNotFound]]">
          <tf-tag-filterer tag-filter="{{_tagFilter}}"></tf-tag-filterer>
          <template is="dom-repeat" items="[[_categories]]" as="category">
            <tf-category-paginated-view
              category="[[category]]"
              initial-opened="[[_shouldOpen(index)]]"
              get-category-item-key="[[_getCategoryItemKey]]"
            >
              <template>
                <tf-pr-curve-card
                  active="[[active]]"
                  runs="[[item.runs]]"
                  tag="[[item.tag]]"
                  tag-metadata="[[_tagMetadata(_runToTagInfo, item.runs, item.tag)]]"
                  request-manager="[[_requestManager]]"
                  run-to-step-cap="[[_runToStep]]"
                  on-data-change="[[_createDataChangeCallback(item.tag)]]"
                ></tf-pr-curve-card>
              </template>
            </tf-category-paginated-view>
          </template>
        </template>
      </div>
    </tf-dashboard-layout>

    <style include="dashboard-style"></style>
    <style>
      .no-data-warning {
        max-width: 540px;
        margin: 80px auto 0 auto;
      }

      /** Do not let the steps selector occlude the run selector. */
      #steps-selector-container {
        max-height: 60%;
        overflow-y: auto;
      }
    </style>
  `;
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfPrCurveDashboard.prototype, "reloadOnReady", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfPrCurveDashboard.prototype, "_timeDisplayType", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfPrCurveDashboard.prototype, "_selectedRuns", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfPrCurveDashboard.prototype, "_runToTagInfo", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfPrCurveDashboard.prototype, "_tagToRunToData", void 0);
    __decorate([
        property({
            type: Object,
            notify: true,
        }),
        __metadata("design:type", Object)
    ], TfPrCurveDashboard.prototype, "_runToStep", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfPrCurveDashboard.prototype, "_dataNotFound", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfPrCurveDashboard.prototype, "_tagFilter", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfPrCurveDashboard.prototype, "_categoriesDomReady", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfPrCurveDashboard.prototype, "_getCategoryItemKey", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", RequestManager)
    ], TfPrCurveDashboard.prototype, "_requestManager", void 0);
    __decorate([
        property({
            type: Number,
            notify: true,
        }),
        __metadata("design:type", Number)
    ], TfPrCurveDashboard.prototype, "_step", void 0);
    __decorate([
        computed('_runToTagInfo', '_selectedRuns', '_tagFilter', '_categoriesDomReady'),
        __metadata("design:type", Array),
        __metadata("design:paramtypes", [])
    ], TfPrCurveDashboard.prototype, "_categories", null);
    __decorate([
        computed('_selectedRuns', '_runToTagInfo'),
        __metadata("design:type", Array),
        __metadata("design:paramtypes", [])
    ], TfPrCurveDashboard.prototype, "_relevantSelectedRuns", null);
    __decorate([
        computed('_tagToRunToData'),
        __metadata("design:type", Object),
        __metadata("design:paramtypes", [])
    ], TfPrCurveDashboard.prototype, "_runToAvailableTimeEntries", null);
    TfPrCurveDashboard = __decorate([
        customElement('tf-pr-curve-dashboard')
        // tslint:disable-next-line:no-unused-variable
    ], TfPrCurveDashboard);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * A frontend that directs users to upgrade to the new version of the profile
     * plugin, now distributed separately from TensorBoard.
     */
    let TfProfileRedirectDashboard = class TfProfileRedirectDashboard extends LegacyElementMixin(PolymerElement) {
        constructor() {
            super(...arguments);
            this._installCommand = 'pip install -U tensorboard-plugin-profile';
        }
        _copyInstallCommand() {
            return __awaiter(this, void 0, void 0, function* () {
                const doCopy = () => __awaiter(this, void 0, void 0, function* () {
                    const textarea = this.$.commandTextarea;
                    textarea.select();
                    try {
                        yield navigator.clipboard.writeText(this._installCommand);
                    }
                    catch (error) {
                        // Fallback approach.
                        if (!document.execCommand('copy')) {
                            return Promise.reject();
                        }
                    }
                });
                const copiedMessageElement = this.$.copiedMessage;
                try {
                    yield doCopy();
                    copiedMessageElement.innerText = 'Copied.';
                }
                catch (e) {
                    copiedMessageElement.innerText = 'Failed to copy to clipboard.';
                }
            });
        }
        _removeCopiedMessage() {
            const copiedMessageElement = this.$.copiedMessage;
            copiedMessageElement.innerText = '';
        }
    };
    TfProfileRedirectDashboard.template = html `
    <div class="message">
      <h3>The profile plugin has moved.</h3>
      <p>
        Please install the new version of the profile plugin from PyPI by
        running the following command from the machine running TensorBoard:
      </p>
      <textarea
        id="commandTextarea"
        readonly=""
        rows="1"
        on-blur="_removeCopiedMessage"
      >
[[_installCommand]]</textarea
      >
      <div id="copyContainer">
        <span id="copiedMessage"></span>
        <paper-button raised="" on-tap="_copyInstallCommand"
          >Copy to clipboard</paper-button
        >
      </div>
    </div>

    <style>
      .message {
        margin: 80px auto 0 auto;
        max-width: 540px;
      }
      #commandTextarea {
        margin-top: 1ex;
        padding: 1ex 1em;
        resize: vertical;
        width: 100%;
      }
      #copyContainer {
        display: flex;
      }
      #copiedMessage {
        align-self: center;
        flex-grow: 1;
        font-style: italic;
        padding-right: 1em;
        text-align: right;
      }
    </style>
  `;
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfProfileRedirectDashboard.prototype, "_installCommand", void 0);
    TfProfileRedirectDashboard = __decorate([
        customElement('tf-profile-redirect-dashboard')
    ], TfProfileRedirectDashboard);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * A frontend that displays a set of tf-scalar-charts, each of which
     * represents the time series for a particular tag. This dashboard
     * provides a categorizer, run selector, and abcissa selector, by which
     * the user can customize how data is organized and displayed.
     */
    let TfScalarDashboard = 
    // tslint:disable-next-line:no-unused-variable
    class TfScalarDashboard extends LegacyElementMixin(ArrayUpdateHelper) {
        constructor() {
            super(...arguments);
            this.reloadOnReady = true;
            this._showDownloadLinks = getBooleanInitializer('_showDownloadLinks', {
                defaultValue: false,
                useLocalStorage: true,
            })
                .call(this);
            this._smoothingWeight = getNumberInitializer('_smoothingWeight', {
                defaultValue: 0.6,
            })
                .call(this);
            this._ignoreYOutliers = getBooleanInitializer('_ignoreYOutliers', {
                defaultValue: true,
                useLocalStorage: true,
            })
                .call(this);
            this._xType = XType.STEP;
            this._selectedRuns = [];
            this._tagFilter = '';
            this._categories = [];
            // Items show multiple runs, so exclude runs from category item keys for
            // efficient template reuse.
            this._getCategoryItemKey = (item) => item.tag;
            this._requestManager = new RequestManager(50);
            this._showDownloadLinksObserver = getBooleanObserver('_showDownloadLinks', { defaultValue: false, useLocalStorage: true });
            this._smoothingWeightObserver = getNumberObserver('_smoothingWeight', {
                defaultValue: 0.6,
            });
            this._ignoreYOutliersObserver = getBooleanObserver('_ignoreYOutliers', {
                defaultValue: true,
                useLocalStorage: true,
            });
        }
        get _smoothingEnabled() {
            var _smoothingWeight = this._smoothingWeight;
            return _smoothingWeight > 0;
        }
        _getCategoryKey(category) {
            return category.metadata.type ==
                CategoryType.SEARCH_RESULTS
                ? ''
                : category.name;
        }
        _shouldOpen(index) {
            return index <= 2;
        }
        ready() {
            super.ready();
            if (this.reloadOnReady)
                this.reload();
        }
        reload() {
            this._fetchTags().then(() => {
                this._reloadCharts();
            });
        }
        _fetchTags() {
            const url = getRouter().pluginRoute('scalars', '/tags');
            return this._requestManager.request(url).then((runToTagInfo) => {
                if (lodash.isEqual(runToTagInfo, this._runToTagInfo)) {
                    // No need to update anything if there are no changes.
                    return;
                }
                const runToTag = lodash.mapValues(runToTagInfo, (x) => Object.keys(x));
                const tags = getTags(runToTag);
                this.set('_dataNotFound', tags.length === 0);
                this.set('_runToTagInfo', runToTagInfo);
                this.async(() => {
                    // See the comment above `_categoriesDomReady`.
                    this.set('_categoriesDomReady', true);
                });
            });
        }
        _reloadCharts() {
            this.root.querySelectorAll('tf-scalar-card').forEach((chart) => {
                chart.reload();
            });
        }
        _updateCategories() {
            var runToTagInfo = this._runToTagInfo;
            var selectedRuns = this._selectedRuns;
            var tagFilter = this._tagFilter;
            let categories;
            let query = tagFilter;
            const runToTag = lodash.mapValues(runToTagInfo, (x) => Object.keys(x));
            categories = categorizeTags(runToTag, selectedRuns, query);
            categories.forEach((category) => {
                category.items = category.items.map((item) => ({
                    tag: item.tag,
                    series: item.runs.map((run) => ({ run, tag: item.tag })),
                }));
            });
            this.updateArrayProp('_categories', categories, this._getCategoryKey);
        }
        _tagMetadata(category, runToTagsInfo, item) {
            const categoryName = category.name;
            const tag = item.tag;
            const runToTagInfo = {};
            item.series.forEach(({ run }) => {
                runToTagInfo[run] = runToTagsInfo[run][tag];
            });
            // All new-style scalar tags include the `/scalar_summary`
            // suffix. We can trim that from the display name.
            const defaultDisplayName = tag.replace(/\/scalar_summary$/, '');
            let { description, displayName } = aggregateTagInfo(runToTagInfo, defaultDisplayName);
            // If category name is a prefix group, strip the prefix from the name
            // of the scalar-card if name != prefix.
            if (category.metadata.type ==
                CategoryType.PREFIX_GROUP &&
                displayName.startsWith(categoryName + '/')) {
                // + 1 to strip off the separator.
                displayName = displayName.slice(categoryName.length + 1);
            }
            return { description, displayName };
        }
    };
    TfScalarDashboard.template = html `
    <tf-dashboard-layout>
      <div class="sidebar" slot="sidebar">
        <div class="settings">
          <div class="sidebar-section">
            <div class="line-item">
              <paper-checkbox
                id="show-download-links"
                checked="{{_showDownloadLinks}}"
                >Show data download links</paper-checkbox
              >
            </div>
            <div class="line-item">
              <paper-checkbox
                id="ignore-y-outlier"
                checked="{{_ignoreYOutliers}}"
                >Ignore outliers in chart scaling</paper-checkbox
              >
            </div>
            <div id="tooltip-sorting">
              <div>Tooltip sorting method:</div>
              <paper-dropdown-menu
                no-label-float
                selected-item-label="{{_tooltipSortingMethod}}"
              >
                <paper-listbox
                  class="dropdown-content"
                  selected="0"
                  slot="dropdown-content"
                >
                  <paper-item>default</paper-item>
                  <paper-item>descending</paper-item>
                  <paper-item>ascending</paper-item>
                  <paper-item>nearest</paper-item>
                </paper-listbox>
              </paper-dropdown-menu>
            </div>
          </div>
          <div class="sidebar-section">
            <tf-smoothing-input
              weight="{{_smoothingWeight}}"
              step="0.001"
              min="0"
              max="0.999"
            ></tf-smoothing-input>
          </div>
          <div class="sidebar-section">
            <tf-option-selector
              id="x-type-selector"
              name="Horizontal Axis"
              selected-id="{{_xType}}"
            >
              <paper-button id="step">step</paper-button
              ><!--
            --><paper-button id="relative">relative</paper-button
              ><!--
            --><paper-button id="wall_time">wall</paper-button>
            </tf-option-selector>
          </div>
        </div>
        <div class="sidebar-section runs-selector">
          <tf-runs-selector selected-runs="{{_selectedRuns}}">
          </tf-runs-selector>
        </div>
      </div>
      <div class="center" slot="center">
        <template is="dom-if" if="[[_dataNotFound]]">
          <div class="no-data-warning">
            <h3>No scalar data was found.</h3>
            <p>Probable causes:</p>
            <ul>
              <li>You haven’t written any scalar data to your event files.</li>
              <li>TensorBoard can’t find your event files.</li>
            </ul>

            <p>
              If you’re new to using TensorBoard, and want to find out how to
              add data and set up your event files, check out the
              <a
                href="https://github.com/tensorflow/tensorboard/blob/master/README.md"
                >README</a
              >
              and perhaps the
              <a
                href="https://www.tensorflow.org/get_started/summaries_and_tensorboard"
                >TensorBoard tutorial</a
              >.
            </p>

            <p>
              If you think TensorBoard is configured properly, please see
              <a
                href="https://github.com/tensorflow/tensorboard/blob/master/README.md#my-tensorboard-isnt-showing-any-data-whats-wrong"
                >the section of the README devoted to missing data problems</a
              >
              and consider filing an issue on GitHub.
            </p>
          </div>
        </template>
        <template is="dom-if" if="[[!_dataNotFound]]">
          <tf-tag-filterer tag-filter="{{_tagFilter}}"></tf-tag-filterer>
          <template is="dom-repeat" items="[[_categories]]" as="category">
            <tf-category-paginated-view
              category="[[category]]"
              initial-opened="[[_shouldOpen(index)]]"
              get-category-item-key="[[_getCategoryItemKey]]"
            >
              <template>
                <tf-scalar-card
                  active="[[active]]"
                  data-to-load="[[item.series]]"
                  ignore-y-outliers="[[_ignoreYOutliers]]"
                  multi-experiments="[[_getMultiExperiments(dataSelection)]]"
                  request-manager="[[_requestManager]]"
                  show-download-links="[[_showDownloadLinks]]"
                  smoothing-enabled="[[_smoothingEnabled]]"
                  smoothing-weight="[[_smoothingWeight]]"
                  tag-metadata="[[_tagMetadata(category, _runToTagInfo, item)]]"
                  tag="[[item.tag]]"
                  tooltip-sorting-method="[[_tooltipSortingMethod]]"
                  x-type="[[_xType]]"
                ></tf-scalar-card>
              </template>
            </tf-category-paginated-view>
          </template>
        </template>
      </div>
    </tf-dashboard-layout>

    <style include="dashboard-style"></style>
    <style>
      #tooltip-sorting {
        align-items: center;
        display: flex;
        font-size: 14px;
        margin-top: 15px;
      }

      #tooltip-sorting paper-dropdown-menu {
        margin-left: 10px;
        --paper-input-container-focus-color: var(--tb-orange-strong);
        width: 105px;
      }

      .line-item {
        display: block;
        padding-top: 5px;
      }
      .no-data-warning {
        max-width: 540px;
        margin: 80px auto 0 auto;
      }
      .center {
        overflow-x: hidden;
      }
    </style>
  `;
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfScalarDashboard.prototype, "reloadOnReady", void 0);
    __decorate([
        property({
            type: Boolean,
            notify: true,
            observer: '_showDownloadLinksObserver',
        }),
        __metadata("design:type", Boolean)
    ], TfScalarDashboard.prototype, "_showDownloadLinks", void 0);
    __decorate([
        property({
            type: Number,
            notify: true,
            observer: '_smoothingWeightObserver',
        }),
        __metadata("design:type", Number)
    ], TfScalarDashboard.prototype, "_smoothingWeight", void 0);
    __decorate([
        property({
            type: Boolean,
            observer: '_ignoreYOutliersObserver',
        }),
        __metadata("design:type", Boolean)
    ], TfScalarDashboard.prototype, "_ignoreYOutliers", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfScalarDashboard.prototype, "_xType", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfScalarDashboard.prototype, "_selectedRuns", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfScalarDashboard.prototype, "_runToTagInfo", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfScalarDashboard.prototype, "_dataNotFound", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfScalarDashboard.prototype, "_tagFilter", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfScalarDashboard.prototype, "_categoriesDomReady", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfScalarDashboard.prototype, "_categories", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfScalarDashboard.prototype, "_getCategoryItemKey", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", RequestManager)
    ], TfScalarDashboard.prototype, "_requestManager", void 0);
    __decorate([
        computed('_smoothingWeight'),
        __metadata("design:type", Boolean),
        __metadata("design:paramtypes", [])
    ], TfScalarDashboard.prototype, "_smoothingEnabled", null);
    __decorate([
        observe('_runToTagInfo', '_selectedRuns', '_tagFilter', '_categoriesDomReady'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfScalarDashboard.prototype, "_updateCategories", null);
    TfScalarDashboard = __decorate([
        customElement('tf-scalar-dashboard')
        // tslint:disable-next-line:no-unused-variable
    ], TfScalarDashboard);

    /* Copyright 2017 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    // tf-text-loader displays markdown text data from the Text plugin.
    let TfTextLoader = class TfTextLoader extends LegacyElementMixin(PolymerElement) {
        constructor() {
            super(...arguments);
            // Ordered from newest to oldest.
            this._texts = [];
            this._canceller = new Canceller();
        }
        get _runColor() {
            var run = this.run;
            return runsColorScale(run);
        }
        _changeRunColor() {
            var runColor = this._runColor;
            this.updateStyles({
                '--tb-text-loader-outline': runColor,
            });
        }
        attached() {
            this.reload();
        }
        reload() {
            if (!this.isAttached) {
                return;
            }
            this._canceller.cancelAll();
            const router = getRouter();
            const url = addParams(router.pluginRoute('text', '/text'), {
                tag: this.tag,
                run: this.run,
            });
            const updateTexts = this._canceller.cancellable((result) => {
                if (result.cancelled) {
                    return;
                }
                const data = result.value.map((datum) => ({
                    wall_time: new Date(datum.wall_time * 1000),
                    step: datum.step,
                    text: datum.text,
                }));
                this.set('_texts', data.slice().reverse());
            });
            this.requestManager.request(url).then(updateTexts);
        }
        _formatStep(n) {
            return format(',')(n);
        }
    };
    TfTextLoader.template = html `
    <tf-card-heading run="[[run]]" tag="[[tag]]" color="[[_runColor]]">
    </tf-card-heading>
    <paper-material
      elevation="1"
      id="steps-container"
      class="container scrollbar"
    >
      <template is="dom-repeat" items="[[_texts]]">
        <paper-material elevation="1" class="step-container">
          step <span class="step-value">[[_formatStep(item.step)]]</span>
        </paper-material>
        <paper-material elevation="1" class="text">
          <tf-markdown-view html="[[item.text]]"></tf-markdown-view>
        </paper-material>
      </template>
    </paper-material>
    <style include="scrollbar-style"></style>
    <style>
      :host {
        display: flex;
        flex-direction: column;
        width: 100%;
        height: auto;
        margin-right: 10px;
        margin-bottom: 15px;
      }
      .scrollbar {
        will-change: transform;
      }
      #steps-container {
        border-radius: 3px;
        border: 2px solid /* color computed and set as inline style */;
        display: block;
        max-height: 500px;
        overflow: auto;
        padding: 10px;
        border-color: var(--tb-text-loader-outline);
      }
      .text {
        background-color: white;
        border-radius: 0 3px 3px 3px;
        padding: 5px;
        word-break: break-word;
      }
      .step-container {
        background-color: var(--tb-ui-light-accent);
        border-bottom: none;
        border-radius: 3px 3px 0 0;
        border: 1px solid #ccc;
        display: inline-block;
        font-size: 12px;
        font-style: italic;
        margin-left: -1px; /* to correct for border */
        padding: 3px;
      }
      .step-container:not(:first-child) {
        margin-top: 15px;
      }

      tf-card-heading {
        margin-bottom: 10px;
      }
    </style>
  `;
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfTextLoader.prototype, "run", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfTextLoader.prototype, "tag", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfTextLoader.prototype, "_texts", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", RequestManager)
    ], TfTextLoader.prototype, "requestManager", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Canceller)
    ], TfTextLoader.prototype, "_canceller", void 0);
    __decorate([
        computed('run'),
        __metadata("design:type", String),
        __metadata("design:paramtypes", [])
    ], TfTextLoader.prototype, "_runColor", null);
    __decorate([
        observe('_runColor'),
        __metadata("design:type", Function),
        __metadata("design:paramtypes", []),
        __metadata("design:returntype", void 0)
    ], TfTextLoader.prototype, "_changeRunColor", null);
    TfTextLoader = __decorate([
        customElement('tf-text-loader')
    ], TfTextLoader);

    /* Copyright 2017 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    let TfTextDashboard = class TfTextDashboard extends LegacyElementMixin(PolymerElement) {
        constructor() {
            super(...arguments);
            this.reloadOnReady = true;
            this._requestManager = new RequestManager();
        }
        ready() {
            super.ready();
            if (this.reloadOnReady)
                this.reload();
        }
        reload() {
            this._fetchTags().then(() => {
                this._reloadTexts();
            });
        }
        _shouldOpen(index) {
            return index <= 2;
        }
        _fetchTags() {
            const url = getRouter().pluginRoute('text', '/tags');
            return this._requestManager.request(url).then((runToTag) => {
                if (lodash.isEqual(runToTag, this._runToTag)) {
                    // No need to update anything if there are no changes.
                    return;
                }
                const tags = getTags(runToTag);
                this.set('_dataNotFound', tags.length === 0);
                this.set('_runToTag', runToTag);
                this.async(() => {
                    // See the comment above `_categoriesDomReady`.
                    this.set('_categoriesDomReady', true);
                });
            });
        }
        _reloadTexts() {
            this.root.querySelectorAll('tf-text-loader').forEach((textLoader) => {
                textLoader.reload();
            });
        }
        get _categories() {
            var runToTag = this._runToTag;
            var selectedRuns = this._selectedRuns;
            var tagFilter = this._tagFilter;
            return categorizeRunTagCombinations(runToTag, selectedRuns, tagFilter);
        }
    };
    TfTextDashboard.template = html `
    <tf-dashboard-layout>
      <div class="sidebar" slot="sidebar">
        <div class="sidebar-section runs-selector">
          <tf-runs-selector selected-runs="{{_selectedRuns}}">
          </tf-runs-selector>
        </div>
      </div>
      <div class="center" slot="center">
        <template is="dom-if" if="[[_dataNotFound]]">
          <div class="no-data-warning">
            <h3>No text data was found.</h3>
            <p>Probable causes:</p>
            <ul>
              <li>You haven’t written any text data to your event files.</li>
              <li>TensorBoard can’t find your event files.</li>
            </ul>

            <p>
              If you’re new to using TensorBoard, and want to find out how to
              add data and set up your event files, check out the
              <a
                href="https://github.com/tensorflow/tensorboard/blob/master/README.md"
                >README</a
              >
              and perhaps the
              <a
                href="https://www.tensorflow.org/get_started/summaries_and_tensorboard"
                >TensorBoard tutorial</a
              >.
            </p>

            <p>
              If you think TensorBoard is configured properly, please see
              <a
                href="https://github.com/tensorflow/tensorboard/blob/master/README.md#my-tensorboard-isnt-showing-any-data-whats-wrong"
                >the section of the README devoted to missing data problems</a
              >
              and consider filing an issue on GitHub.
            </p>
          </div>
        </template>
        <template is="dom-if" if="[[!_dataNotFound]]">
          <tf-tag-filterer tag-filter="{{_tagFilter}}"></tf-tag-filterer>
          <template is="dom-repeat" items="[[_categories]]" as="category">
            <tf-category-paginated-view
              category="[[category]]"
              initial-opened="[[_shouldOpen(index)]]"
            >
              <template>
                <tf-text-loader
                  active="[[active]]"
                  tag="[[item.tag]]"
                  run="[[item.run]]"
                  request-manager="[[_requestManager]]"
                ></tf-text-loader>
              </template>
            </tf-category-paginated-view>
          </template>
        </template>
      </div>
    </tf-dashboard-layout>
    <style include="dashboard-style"></style>
    <style>
      .no-data-warning {
        max-width: 540px;
        margin: 80px auto 0 auto;
      }
    </style>
  `;
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfTextDashboard.prototype, "reloadOnReady", void 0);
    __decorate([
        property({ type: Array }),
        __metadata("design:type", Array)
    ], TfTextDashboard.prototype, "_selectedRuns", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfTextDashboard.prototype, "_runToTag", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfTextDashboard.prototype, "_dataNotFound", void 0);
    __decorate([
        property({ type: String }),
        __metadata("design:type", String)
    ], TfTextDashboard.prototype, "_tagFilter", void 0);
    __decorate([
        property({ type: Boolean }),
        __metadata("design:type", Boolean)
    ], TfTextDashboard.prototype, "_categoriesDomReady", void 0);
    __decorate([
        property({ type: Object }),
        __metadata("design:type", Object)
    ], TfTextDashboard.prototype, "_requestManager", void 0);
    __decorate([
        computed('_runToTag', '_selectedRuns', '_tagFilter', '_categoriesDomReady'),
        __metadata("design:type", Array),
        __metadata("design:paramtypes", [])
    ], TfTextDashboard.prototype, "_categories", null);
    TfTextDashboard = __decorate([
        customElement('tf-text-dashboard')
    ], TfTextDashboard);

    class IPC {
        constructor(port) {
            this.port = port;
            this.id = 0;
            this.responseWaits = new Map();
            this.listeners = new Map();
            this.port.addEventListener('message', (event) => this.onMessage(event));
        }
        listen(type, callback) {
            this.listeners.set(type, callback);
        }
        unlisten(type) {
            this.listeners.delete(type);
        }
        onMessage(event) {
            return __awaiter(this, void 0, void 0, function* () {
                const message = JSON.parse(event.data);
                // Please see [1] for reason why we use string to access the property.
                const type = message['type'];
                const id = message['id'];
                const payload = message['payload'];
                const error = message['error'];
                const isReply = message['isReply'];
                if (isReply) {
                    if (!this.responseWaits.has(id))
                        return;
                    const { resolve, reject } = this.responseWaits.get(id);
                    this.responseWaits.delete(id);
                    if (error) {
                        reject(new Error(error));
                    }
                    else {
                        resolve(payload);
                    }
                    return;
                }
                let replyPayload = null;
                let replyError = null;
                if (this.listeners.has(type)) {
                    const callback = this.listeners.get(type);
                    try {
                        const result = yield callback(payload);
                        replyPayload = result;
                    }
                    catch (e) {
                        replyError = e;
                    }
                }
                // Please see [1] for reason why we use string to access the property.
                const replyMessage = {
                    ['type']: type,
                    ['id']: id,
                    ['payload']: replyPayload,
                    ['error']: replyError,
                    ['isReply']: true,
                };
                this.postMessage(replyMessage);
            });
        }
        postMessage(message) {
            this.port.postMessage(JSON.stringify(message));
        }
        sendMessage(type, payload) {
            const id = this.id++;
            const message = { type, id, payload, error: null, isReply: false };
            this.postMessage(message);
            return new Promise((resolve, reject) => {
                this.responseWaits.set(id, { resolve, reject });
            });
        }
    }

    /* Copyright 2019 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * Registers metadata associated with a plugin iframe upon creation. Plugins
     * registered with this do not necessarily use IPC.
     */
    function registerPluginIframe(frame, pluginName) {
        pluginMetadata.set(frame, { pluginName });
    }
    const pluginMetadata = new WeakMap();
    const portIPCs = new Set();
    const VERSION = 'experimental';
    const listeners$1 = new Map();
    // TODO(@psybuzz): replace this and the port cleanup logic in broadcast() with
    // a MutationObserver to notify us when iframes disconnect.
    const ipcToFrame = new Map();
    // The initial Window-level listener is needed to bootstrap only.
    // All further communication is done over MessagePorts.
    window.addEventListener('message', (event) => {
        if (event.data !== `${VERSION}.bootstrap`)
            return;
        const port = event.ports[0];
        if (!port)
            return;
        const frame = event.source ? event.source.frameElement : null;
        if (!frame)
            return;
        onBootstrap(port, frame);
    });
    function onBootstrap(port, frame) {
        const portIPC = new IPC(port);
        portIPCs.add(portIPC);
        ipcToFrame.set(portIPC, frame);
        port.start();
        for (const [type, callback] of listeners$1) {
            const callbackWithContext = wrapCallbackWithContext(callback, portIPC);
            portIPC.listen(type, callbackWithContext);
        }
    }
    /**
     * Provides context data from the IPC to the callback.
     */
    function wrapCallbackWithContext(callback, ipc) {
        return (payload) => {
            const frame = ipcToFrame.get(ipc);
            const context = pluginMetadata.get(frame) || null;
            return callback(context, payload);
        };
    }
    /**
     * Sends a message to all frames. Individual frames decide whether or not to
     * listen.
     * @return Promise that resolves with a list of payloads from each plugin's
     *         response (or null) to the message.
     *
     * @example
     * const someList = await broadcast('v1.some.type.guest.understands');
     * // do fun things with someList.
     */
    function broadcast(type, payload) {
        for (const ipc of portIPCs) {
            if (!ipcToFrame.get(ipc).isConnected) {
                portIPCs.delete(ipc);
                ipcToFrame.delete(ipc);
            }
        }
        const promises = [...portIPCs].map((ipc) => ipc.sendMessage(type, payload));
        return Promise.all(promises);
    }
    /**
     * Subscribes to messages of a type specified for all frames.
     */
    function listen(type, callback) {
        listeners$1.set(type, callback);
        for (const ipc of portIPCs) {
            const callbackWithContext = wrapCallbackWithContext(callback, ipc);
            ipc.listen(type, callbackWithContext);
        }
    }

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    listen('experimental.GetURLPluginData', (context) => {
        if (!context) {
            return;
        }
        const prefix = `p.${context.pluginName}.`;
        const result = {};
        const urlDict = getUrlDict();
        for (let key in urlDict) {
            if (key.startsWith(prefix)) {
                const pluginKey = key.substring(prefix.length);
                result[pluginKey] = urlDict[key];
            }
        }
        return result;
    });

    /* Copyright 2017 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    listen('experimental.GetRuns', () => {
        return runsStore.getRuns();
    });
    runsStore.addListener(() => {
        return broadcast('experimental.RunsChanged', runsStore.getRuns());
    });

    // HACK: this Polymer component allows the experimental plugin host APIs
    // to be accessible across bundle binary.
    let TfExperimentalPluginHostLib = class TfExperimentalPluginHostLib extends PolymerElement {
        constructor() {
            super(...arguments);
            this._template = null;
            this.registerPluginIframe = registerPluginIframe;
        }
    };
    TfExperimentalPluginHostLib = __decorate([
        customElement('tf-experimental-plugin-host-lib')
    ], TfExperimentalPluginHostLib);

    // HACK: this Polymer component allows stores to be accessible from
    // tf-ng-tensorboard by exposing otherwise mangled smybols.
    let TfBackend = class TfBackend extends PolymerElement {
        constructor() {
            super(...arguments);
            this._template = null;
            this.tf_backend = tf_backend;
        }
    };
    TfBackend = __decorate([
        customElement('tf-backend')
    ], TfBackend);

    let TfGlobals = class TfGlobals extends PolymerElement {
        constructor() {
            super(...arguments);
            this._template = null;
            this.tf_globals = tf_globals;
        }
    };
    TfGlobals = __decorate([
        customElement('tf-globals')
    ], TfGlobals);

    let TfStorage = class TfStorage extends PolymerElement {
        constructor() {
            super(...arguments);
            this._template = null;
            this.tf_storage = tf_storage;
        }
    };
    TfStorage = __decorate([
        customElement('tf-storage')
    ], TfStorage);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/

    var tf_paginated_view = /*#__PURE__*/Object.freeze({
        __proto__: null,
        addLimitListener: addLimitListener,
        removeLimitListener: removeLimitListener,
        getLimit: getLimit,
        setLimit: setLimit,
        TfDomRepeat: TfDomRepeat
    });

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    let TfPaginatedViewStore = class TfPaginatedViewStore extends PolymerElement {
        constructor() {
            super(...arguments);
            this._template = null;
            this.tf_paginated_view = tf_paginated_view;
        }
    };
    TfPaginatedViewStore = __decorate([
        customElement('tf-paginated-view-store')
    ], TfPaginatedViewStore);

    // HACK: this Polymer component allows stores to be accessible from
    // tf-ng-tensorboard by exposing otherwise mangled smybols.
    let TfColorScale = class TfColorScale extends PolymerElement {
        constructor() {
            super(...arguments);
            this._template = null;
            this.runsColorScale = runsColorScale;
        }
    };
    TfColorScale = __decorate([
        customElement('tf-color-scale')
    ], TfColorScale);

}());



(function () {

    function isFunction(x) {
        return typeof x === 'function';
    }

    let _enable_super_gross_mode_that_will_cause_bad_things = false;
    const config = {
        Promise: undefined,
        set useDeprecatedSynchronousErrorHandling(value) {
            if (value) {
                const error = new Error();
                console.warn('DEPRECATED! RxJS was set to use deprecated synchronous error handling behavior by code at: \n' + error.stack);
            }
            else if (_enable_super_gross_mode_that_will_cause_bad_things) {
                console.log('RxJS: Back to a better error behavior. Thank you. <3');
            }
            _enable_super_gross_mode_that_will_cause_bad_things = value;
        },
        get useDeprecatedSynchronousErrorHandling() {
            return _enable_super_gross_mode_that_will_cause_bad_things;
        },
    };

    function hostReportError(err) {
        setTimeout(() => { throw err; }, 0);
    }

    const empty = {
        closed: true,
        next(value) { },
        error(err) {
            if (config.useDeprecatedSynchronousErrorHandling) {
                throw err;
            }
            else {
                hostReportError(err);
            }
        },
        complete() { }
    };

    const isArray = (() => Array.isArray || ((x) => x && typeof x.length === 'number'))();

    function isObject(x) {
        return x !== null && typeof x === 'object';
    }

    const UnsubscriptionErrorImpl = (() => {
        function UnsubscriptionErrorImpl(errors) {
            Error.call(this);
            this.message = errors ?
                `${errors.length} errors occurred during unsubscription:
${errors.map((err, i) => `${i + 1}) ${err.toString()}`).join('\n  ')}` : '';
            this.name = 'UnsubscriptionError';
            this.errors = errors;
            return this;
        }
        UnsubscriptionErrorImpl.prototype = Object.create(Error.prototype);
        return UnsubscriptionErrorImpl;
    })();
    const UnsubscriptionError = UnsubscriptionErrorImpl;

    class Subscription {
        constructor(unsubscribe) {
            this.closed = false;
            this._parentOrParents = null;
            this._subscriptions = null;
            if (unsubscribe) {
                this._unsubscribe = unsubscribe;
            }
        }
        unsubscribe() {
            let errors;
            if (this.closed) {
                return;
            }
            let { _parentOrParents, _unsubscribe, _subscriptions } = this;
            this.closed = true;
            this._parentOrParents = null;
            this._subscriptions = null;
            if (_parentOrParents instanceof Subscription) {
                _parentOrParents.remove(this);
            }
            else if (_parentOrParents !== null) {
                for (let index = 0; index < _parentOrParents.length; ++index) {
                    const parent = _parentOrParents[index];
                    parent.remove(this);
                }
            }
            if (isFunction(_unsubscribe)) {
                try {
                    _unsubscribe.call(this);
                }
                catch (e) {
                    errors = e instanceof UnsubscriptionError ? flattenUnsubscriptionErrors(e.errors) : [e];
                }
            }
            if (isArray(_subscriptions)) {
                let index = -1;
                let len = _subscriptions.length;
                while (++index < len) {
                    const sub = _subscriptions[index];
                    if (isObject(sub)) {
                        try {
                            sub.unsubscribe();
                        }
                        catch (e) {
                            errors = errors || [];
                            if (e instanceof UnsubscriptionError) {
                                errors = errors.concat(flattenUnsubscriptionErrors(e.errors));
                            }
                            else {
                                errors.push(e);
                            }
                        }
                    }
                }
            }
            if (errors) {
                throw new UnsubscriptionError(errors);
            }
        }
        add(teardown) {
            let subscription = teardown;
            if (!teardown) {
                return Subscription.EMPTY;
            }
            switch (typeof teardown) {
                case 'function':
                    subscription = new Subscription(teardown);
                case 'object':
                    if (subscription === this || subscription.closed || typeof subscription.unsubscribe !== 'function') {
                        return subscription;
                    }
                    else if (this.closed) {
                        subscription.unsubscribe();
                        return subscription;
                    }
                    else if (!(subscription instanceof Subscription)) {
                        const tmp = subscription;
                        subscription = new Subscription();
                        subscription._subscriptions = [tmp];
                    }
                    break;
                default: {
                    throw new Error('unrecognized teardown ' + teardown + ' added to Subscription.');
                }
            }
            let { _parentOrParents } = subscription;
            if (_parentOrParents === null) {
                subscription._parentOrParents = this;
            }
            else if (_parentOrParents instanceof Subscription) {
                if (_parentOrParents === this) {
                    return subscription;
                }
                subscription._parentOrParents = [_parentOrParents, this];
            }
            else if (_parentOrParents.indexOf(this) === -1) {
                _parentOrParents.push(this);
            }
            else {
                return subscription;
            }
            const subscriptions = this._subscriptions;
            if (subscriptions === null) {
                this._subscriptions = [subscription];
            }
            else {
                subscriptions.push(subscription);
            }
            return subscription;
        }
        remove(subscription) {
            const subscriptions = this._subscriptions;
            if (subscriptions) {
                const subscriptionIndex = subscriptions.indexOf(subscription);
                if (subscriptionIndex !== -1) {
                    subscriptions.splice(subscriptionIndex, 1);
                }
            }
        }
    }
    Subscription.EMPTY = (function (empty) {
        empty.closed = true;
        return empty;
    }(new Subscription()));
    function flattenUnsubscriptionErrors(errors) {
        return errors.reduce((errs, err) => errs.concat((err instanceof UnsubscriptionError) ? err.errors : err), []);
    }

    const rxSubscriber = (() => typeof Symbol === 'function'
        ? Symbol('rxSubscriber')
        : '@@rxSubscriber_' + Math.random())();

    class Subscriber extends Subscription {
        constructor(destinationOrNext, error, complete) {
            super();
            this.syncErrorValue = null;
            this.syncErrorThrown = false;
            this.syncErrorThrowable = false;
            this.isStopped = false;
            switch (arguments.length) {
                case 0:
                    this.destination = empty;
                    break;
                case 1:
                    if (!destinationOrNext) {
                        this.destination = empty;
                        break;
                    }
                    if (typeof destinationOrNext === 'object') {
                        if (destinationOrNext instanceof Subscriber) {
                            this.syncErrorThrowable = destinationOrNext.syncErrorThrowable;
                            this.destination = destinationOrNext;
                            destinationOrNext.add(this);
                        }
                        else {
                            this.syncErrorThrowable = true;
                            this.destination = new SafeSubscriber(this, destinationOrNext);
                        }
                        break;
                    }
                default:
                    this.syncErrorThrowable = true;
                    this.destination = new SafeSubscriber(this, destinationOrNext, error, complete);
                    break;
            }
        }
        [rxSubscriber]() { return this; }
        static create(next, error, complete) {
            const subscriber = new Subscriber(next, error, complete);
            subscriber.syncErrorThrowable = false;
            return subscriber;
        }
        next(value) {
            if (!this.isStopped) {
                this._next(value);
            }
        }
        error(err) {
            if (!this.isStopped) {
                this.isStopped = true;
                this._error(err);
            }
        }
        complete() {
            if (!this.isStopped) {
                this.isStopped = true;
                this._complete();
            }
        }
        unsubscribe() {
            if (this.closed) {
                return;
            }
            this.isStopped = true;
            super.unsubscribe();
        }
        _next(value) {
            this.destination.next(value);
        }
        _error(err) {
            this.destination.error(err);
            this.unsubscribe();
        }
        _complete() {
            this.destination.complete();
            this.unsubscribe();
        }
        _unsubscribeAndRecycle() {
            const { _parentOrParents } = this;
            this._parentOrParents = null;
            this.unsubscribe();
            this.closed = false;
            this.isStopped = false;
            this._parentOrParents = _parentOrParents;
            return this;
        }
    }
    class SafeSubscriber extends Subscriber {
        constructor(_parentSubscriber, observerOrNext, error, complete) {
            super();
            this._parentSubscriber = _parentSubscriber;
            let next;
            let context = this;
            if (isFunction(observerOrNext)) {
                next = observerOrNext;
            }
            else if (observerOrNext) {
                next = observerOrNext.next;
                error = observerOrNext.error;
                complete = observerOrNext.complete;
                if (observerOrNext !== empty) {
                    context = Object.create(observerOrNext);
                    if (isFunction(context.unsubscribe)) {
                        this.add(context.unsubscribe.bind(context));
                    }
                    context.unsubscribe = this.unsubscribe.bind(this);
                }
            }
            this._context = context;
            this._next = next;
            this._error = error;
            this._complete = complete;
        }
        next(value) {
            if (!this.isStopped && this._next) {
                const { _parentSubscriber } = this;
                if (!config.useDeprecatedSynchronousErrorHandling || !_parentSubscriber.syncErrorThrowable) {
                    this.__tryOrUnsub(this._next, value);
                }
                else if (this.__tryOrSetError(_parentSubscriber, this._next, value)) {
                    this.unsubscribe();
                }
            }
        }
        error(err) {
            if (!this.isStopped) {
                const { _parentSubscriber } = this;
                const { useDeprecatedSynchronousErrorHandling } = config;
                if (this._error) {
                    if (!useDeprecatedSynchronousErrorHandling || !_parentSubscriber.syncErrorThrowable) {
                        this.__tryOrUnsub(this._error, err);
                        this.unsubscribe();
                    }
                    else {
                        this.__tryOrSetError(_parentSubscriber, this._error, err);
                        this.unsubscribe();
                    }
                }
                else if (!_parentSubscriber.syncErrorThrowable) {
                    this.unsubscribe();
                    if (useDeprecatedSynchronousErrorHandling) {
                        throw err;
                    }
                    hostReportError(err);
                }
                else {
                    if (useDeprecatedSynchronousErrorHandling) {
                        _parentSubscriber.syncErrorValue = err;
                        _parentSubscriber.syncErrorThrown = true;
                    }
                    else {
                        hostReportError(err);
                    }
                    this.unsubscribe();
                }
            }
        }
        complete() {
            if (!this.isStopped) {
                const { _parentSubscriber } = this;
                if (this._complete) {
                    const wrappedComplete = () => this._complete.call(this._context);
                    if (!config.useDeprecatedSynchronousErrorHandling || !_parentSubscriber.syncErrorThrowable) {
                        this.__tryOrUnsub(wrappedComplete);
                        this.unsubscribe();
                    }
                    else {
                        this.__tryOrSetError(_parentSubscriber, wrappedComplete);
                        this.unsubscribe();
                    }
                }
                else {
                    this.unsubscribe();
                }
            }
        }
        __tryOrUnsub(fn, value) {
            try {
                fn.call(this._context, value);
            }
            catch (err) {
                this.unsubscribe();
                if (config.useDeprecatedSynchronousErrorHandling) {
                    throw err;
                }
                else {
                    hostReportError(err);
                }
            }
        }
        __tryOrSetError(parent, fn, value) {
            if (!config.useDeprecatedSynchronousErrorHandling) {
                throw new Error('bad call');
            }
            try {
                fn.call(this._context, value);
            }
            catch (err) {
                if (config.useDeprecatedSynchronousErrorHandling) {
                    parent.syncErrorValue = err;
                    parent.syncErrorThrown = true;
                    return true;
                }
                else {
                    hostReportError(err);
                    return true;
                }
            }
            return false;
        }
        _unsubscribe() {
            const { _parentSubscriber } = this;
            this._context = null;
            this._parentSubscriber = null;
            _parentSubscriber.unsubscribe();
        }
    }

    function canReportError(observer) {
        while (observer) {
            const { closed, destination, isStopped } = observer;
            if (closed || isStopped) {
                return false;
            }
            else if (destination && destination instanceof Subscriber) {
                observer = destination;
            }
            else {
                observer = null;
            }
        }
        return true;
    }

    function toSubscriber(nextOrObserver, error, complete) {
        if (nextOrObserver) {
            if (nextOrObserver instanceof Subscriber) {
                return nextOrObserver;
            }
            if (nextOrObserver[rxSubscriber]) {
                return nextOrObserver[rxSubscriber]();
            }
        }
        if (!nextOrObserver && !error && !complete) {
            return new Subscriber(empty);
        }
        return new Subscriber(nextOrObserver, error, complete);
    }

    const observable = (() => typeof Symbol === 'function' && Symbol.observable || '@@observable')();

    function identity(x) {
        return x;
    }

    function pipeFromArray(fns) {
        if (fns.length === 0) {
            return identity;
        }
        if (fns.length === 1) {
            return fns[0];
        }
        return function piped(input) {
            return fns.reduce((prev, fn) => fn(prev), input);
        };
    }

    /*! *****************************************************************************
    Copyright (c) Microsoft Corporation.

    Permission to use, copy, modify, and/or distribute this software for any
    purpose with or without fee is hereby granted.

    THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH
    REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
    AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,
    INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
    LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR
    OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
    PERFORMANCE OF THIS SOFTWARE.
    ***************************************************************************** */

    function __rest(s, e) {
        var t = {};
        for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p) && e.indexOf(p) < 0)
            t[p] = s[p];
        if (s != null && typeof Object.getOwnPropertySymbols === "function")
            for (var i = 0, p = Object.getOwnPropertySymbols(s); i < p.length; i++) {
                if (e.indexOf(p[i]) < 0 && Object.prototype.propertyIsEnumerable.call(s, p[i]))
                    t[p[i]] = s[p[i]];
            }
        return t;
    }

    function __decorate(decorators, target, key, desc) {
        var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
        if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
        else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
        return c > 3 && r && Object.defineProperty(target, key, r), r;
    }

    function __awaiter(thisArg, _arguments, P, generator) {
        function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
        return new (P || (P = Promise))(function (resolve, reject) {
            function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
            function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
            function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
            step((generator = generator.apply(thisArg, _arguments || [])).next());
        });
    }

    function __values(o) {
        var s = typeof Symbol === "function" && Symbol.iterator, m = s && o[s], i = 0;
        if (m) return m.call(o);
        if (o && typeof o.length === "number") return {
            next: function () {
                if (o && i >= o.length) o = void 0;
                return { value: o && o[i++], done: !o };
            }
        };
        throw new TypeError(s ? "Object is not iterable." : "Symbol.iterator is not defined.");
    }

    function __await(v) {
        return this instanceof __await ? (this.v = v, this) : new __await(v);
    }

    function __asyncGenerator(thisArg, _arguments, generator) {
        if (!Symbol.asyncIterator) throw new TypeError("Symbol.asyncIterator is not defined.");
        var g = generator.apply(thisArg, _arguments || []), i, q = [];
        return i = {}, verb("next"), verb("throw"), verb("return"), i[Symbol.asyncIterator] = function () { return this; }, i;
        function verb(n) { if (g[n]) i[n] = function (v) { return new Promise(function (a, b) { q.push([n, v, a, b]) > 1 || resume(n, v); }); }; }
        function resume(n, v) { try { step(g[n](v)); } catch (e) { settle(q[0][3], e); } }
        function step(r) { r.value instanceof __await ? Promise.resolve(r.value.v).then(fulfill, reject) : settle(q[0][2], r); }
        function fulfill(value) { resume("next", value); }
        function reject(value) { resume("throw", value); }
        function settle(f, v) { if (f(v), q.shift(), q.length) resume(q[0][0], q[0][1]); }
    }

    function __asyncValues(o) {
        if (!Symbol.asyncIterator) throw new TypeError("Symbol.asyncIterator is not defined.");
        var m = o[Symbol.asyncIterator], i;
        return m ? m.call(o) : (o = typeof __values === "function" ? __values(o) : o[Symbol.iterator](), i = {}, verb("next"), verb("throw"), verb("return"), i[Symbol.asyncIterator] = function () { return this; }, i);
        function verb(n) { i[n] = o[n] && function (v) { return new Promise(function (resolve, reject) { v = o[n](v), settle(resolve, reject, v.done, v.value); }); }; }
        function settle(resolve, reject, d, v) { Promise.resolve(v).then(function(v) { resolve({ value: v, done: d }); }, reject); }
    }

    class Deferred {
        constructor() {
            this.resolve = null;
            this.reject = null;
            this.promise = new Promise((a, b) => {
                this.resolve = a;
                this.reject = b;
            });
        }
    }

    function asyncIteratorFrom(source) {
        return coroutine(source);
    }
    function coroutine(source) {
        return __asyncGenerator(this, arguments, function* coroutine_1() {
            const deferreds = [];
            const values = [];
            let hasError = false;
            let error = null;
            let completed = false;
            const subs = source.subscribe({
                next: value => {
                    if (deferreds.length > 0) {
                        deferreds.shift().resolve({ value, done: false });
                    }
                    else {
                        values.push(value);
                    }
                },
                error: err => {
                    hasError = true;
                    error = err;
                    while (deferreds.length > 0) {
                        deferreds.shift().reject(err);
                    }
                },
                complete: () => {
                    completed = true;
                    while (deferreds.length > 0) {
                        deferreds.shift().resolve({ value: undefined, done: true });
                    }
                },
            });
            try {
                while (true) {
                    if (values.length > 0) {
                        yield yield __await(values.shift());
                    }
                    else if (completed) {
                        return yield __await(void 0);
                    }
                    else if (hasError) {
                        throw error;
                    }
                    else {
                        const d = new Deferred();
                        deferreds.push(d);
                        const result = yield __await(d.promise);
                        if (result.done) {
                            return yield __await(void 0);
                        }
                        else {
                            yield yield __await(result.value);
                        }
                    }
                }
            }
            catch (err) {
                throw err;
            }
            finally {
                subs.unsubscribe();
            }
        });
    }

    class Observable {
        constructor(subscribe) {
            this._isScalar = false;
            if (subscribe) {
                this._subscribe = subscribe;
            }
        }
        lift(operator) {
            const observable = new Observable();
            observable.source = this;
            observable.operator = operator;
            return observable;
        }
        subscribe(observerOrNext, error, complete) {
            const { operator } = this;
            const sink = toSubscriber(observerOrNext, error, complete);
            if (operator) {
                sink.add(operator.call(sink, this.source));
            }
            else {
                sink.add(this.source || (config.useDeprecatedSynchronousErrorHandling && !sink.syncErrorThrowable) ?
                    this._subscribe(sink) :
                    this._trySubscribe(sink));
            }
            if (config.useDeprecatedSynchronousErrorHandling) {
                if (sink.syncErrorThrowable) {
                    sink.syncErrorThrowable = false;
                    if (sink.syncErrorThrown) {
                        throw sink.syncErrorValue;
                    }
                }
            }
            return sink;
        }
        _trySubscribe(sink) {
            try {
                return this._subscribe(sink);
            }
            catch (err) {
                if (config.useDeprecatedSynchronousErrorHandling) {
                    sink.syncErrorThrown = true;
                    sink.syncErrorValue = err;
                }
                if (canReportError(sink)) {
                    sink.error(err);
                }
                else {
                    console.warn(err);
                }
            }
        }
        forEach(next, promiseCtor) {
            promiseCtor = getPromiseCtor(promiseCtor);
            return new promiseCtor((resolve, reject) => {
                let subscription;
                subscription = this.subscribe((value) => {
                    try {
                        next(value);
                    }
                    catch (err) {
                        reject(err);
                        if (subscription) {
                            subscription.unsubscribe();
                        }
                    }
                }, reject, resolve);
            });
        }
        _subscribe(subscriber) {
            const { source } = this;
            return source && source.subscribe(subscriber);
        }
        [observable]() {
            return this;
        }
        pipe(...operations) {
            if (operations.length === 0) {
                return this;
            }
            return pipeFromArray(operations)(this);
        }
        toPromise(promiseCtor) {
            promiseCtor = getPromiseCtor(promiseCtor);
            return new promiseCtor((resolve, reject) => {
                let value;
                this.subscribe((x) => value = x, (err) => reject(err), () => resolve(value));
            });
        }
    }
    Observable.create = (subscribe) => {
        return new Observable(subscribe);
    };
    function getPromiseCtor(promiseCtor) {
        if (!promiseCtor) {
            promiseCtor =  Promise;
        }
        if (!promiseCtor) {
            throw new Error('no Promise impl found');
        }
        return promiseCtor;
    }
    (function () {
        if (Symbol && Symbol.asyncIterator) {
            Observable.prototype[Symbol.asyncIterator] = function () {
                return asyncIteratorFrom(this);
            };
        }
    })();

    const ObjectUnsubscribedErrorImpl = (() => {
        function ObjectUnsubscribedErrorImpl() {
            Error.call(this);
            this.message = 'object unsubscribed';
            this.name = 'ObjectUnsubscribedError';
            return this;
        }
        ObjectUnsubscribedErrorImpl.prototype = Object.create(Error.prototype);
        return ObjectUnsubscribedErrorImpl;
    })();
    const ObjectUnsubscribedError = ObjectUnsubscribedErrorImpl;

    class SubjectSubscription extends Subscription {
        constructor(subject, subscriber) {
            super();
            this.subject = subject;
            this.subscriber = subscriber;
            this.closed = false;
        }
        unsubscribe() {
            if (this.closed) {
                return;
            }
            this.closed = true;
            const subject = this.subject;
            const observers = subject.observers;
            this.subject = null;
            if (!observers || observers.length === 0 || subject.isStopped || subject.closed) {
                return;
            }
            const subscriberIndex = observers.indexOf(this.subscriber);
            if (subscriberIndex !== -1) {
                observers.splice(subscriberIndex, 1);
            }
        }
    }

    class SubjectSubscriber extends Subscriber {
        constructor(destination) {
            super(destination);
            this.destination = destination;
        }
    }
    class Subject extends Observable {
        constructor() {
            super();
            this.observers = [];
            this.closed = false;
            this.isStopped = false;
            this.hasError = false;
            this.thrownError = null;
        }
        [rxSubscriber]() {
            return new SubjectSubscriber(this);
        }
        lift(operator) {
            const subject = new AnonymousSubject(this, this);
            subject.operator = operator;
            return subject;
        }
        next(value) {
            if (this.closed) {
                throw new ObjectUnsubscribedError();
            }
            if (!this.isStopped) {
                const { observers } = this;
                const len = observers.length;
                const copy = observers.slice();
                for (let i = 0; i < len; i++) {
                    copy[i].next(value);
                }
            }
        }
        error(err) {
            if (this.closed) {
                throw new ObjectUnsubscribedError();
            }
            this.hasError = true;
            this.thrownError = err;
            this.isStopped = true;
            const { observers } = this;
            const len = observers.length;
            const copy = observers.slice();
            for (let i = 0; i < len; i++) {
                copy[i].error(err);
            }
            this.observers.length = 0;
        }
        complete() {
            if (this.closed) {
                throw new ObjectUnsubscribedError();
            }
            this.isStopped = true;
            const { observers } = this;
            const len = observers.length;
            const copy = observers.slice();
            for (let i = 0; i < len; i++) {
                copy[i].complete();
            }
            this.observers.length = 0;
        }
        unsubscribe() {
            this.isStopped = true;
            this.closed = true;
            this.observers = null;
        }
        _trySubscribe(subscriber) {
            if (this.closed) {
                throw new ObjectUnsubscribedError();
            }
            else {
                return super._trySubscribe(subscriber);
            }
        }
        _subscribe(subscriber) {
            if (this.closed) {
                throw new ObjectUnsubscribedError();
            }
            else if (this.hasError) {
                subscriber.error(this.thrownError);
                return Subscription.EMPTY;
            }
            else if (this.isStopped) {
                subscriber.complete();
                return Subscription.EMPTY;
            }
            else {
                this.observers.push(subscriber);
                return new SubjectSubscription(this, subscriber);
            }
        }
        asObservable() {
            const observable = new Observable();
            observable.source = this;
            return observable;
        }
    }
    Subject.create = (destination, source) => {
        return new AnonymousSubject(destination, source);
    };
    class AnonymousSubject extends Subject {
        constructor(destination, source) {
            super();
            this.destination = destination;
            this.source = source;
        }
        next(value) {
            const { destination } = this;
            if (destination && destination.next) {
                destination.next(value);
            }
        }
        error(err) {
            const { destination } = this;
            if (destination && destination.error) {
                this.destination.error(err);
            }
        }
        complete() {
            const { destination } = this;
            if (destination && destination.complete) {
                this.destination.complete();
            }
        }
        _subscribe(subscriber) {
            const { source } = this;
            if (source) {
                return this.source.subscribe(subscriber);
            }
            else {
                return Subscription.EMPTY;
            }
        }
    }

    function refCount() {
        return function refCountOperatorFunction(source) {
            return source.lift(new RefCountOperator(source));
        };
    }
    class RefCountOperator {
        constructor(connectable) {
            this.connectable = connectable;
        }
        call(subscriber, source) {
            const { connectable } = this;
            connectable._refCount++;
            const refCounter = new RefCountSubscriber(subscriber, connectable);
            const subscription = source.subscribe(refCounter);
            if (!refCounter.closed) {
                refCounter.connection = connectable.connect();
            }
            return subscription;
        }
    }
    class RefCountSubscriber extends Subscriber {
        constructor(destination, connectable) {
            super(destination);
            this.connectable = connectable;
            this.connection = null;
        }
        _unsubscribe() {
            const { connectable } = this;
            if (!connectable) {
                this.connection = null;
                return;
            }
            this.connectable = null;
            const refCount = connectable._refCount;
            if (refCount <= 0) {
                this.connection = null;
                return;
            }
            connectable._refCount = refCount - 1;
            if (refCount > 1) {
                this.connection = null;
                return;
            }
            const { connection } = this;
            const sharedConnection = connectable._connection;
            this.connection = null;
            if (sharedConnection && (!connection || sharedConnection === connection)) {
                sharedConnection.unsubscribe();
            }
        }
    }

    class ConnectableObservable extends Observable {
        constructor(source, subjectFactory) {
            super();
            this.source = source;
            this.subjectFactory = subjectFactory;
            this._refCount = 0;
            this._isComplete = false;
        }
        _subscribe(subscriber) {
            return this.getSubject().subscribe(subscriber);
        }
        getSubject() {
            const subject = this._subject;
            if (!subject || subject.isStopped) {
                this._subject = this.subjectFactory();
            }
            return this._subject;
        }
        connect() {
            let connection = this._connection;
            if (!connection) {
                this._isComplete = false;
                connection = this._connection = new Subscription();
                connection.add(this.source
                    .subscribe(new ConnectableSubscriber(this.getSubject(), this)));
                if (connection.closed) {
                    this._connection = null;
                    connection = Subscription.EMPTY;
                }
            }
            return connection;
        }
        refCount() {
            return refCount()(this);
        }
    }
    const connectableObservableDescriptor = (() => {
        const connectableProto = ConnectableObservable.prototype;
        return {
            operator: { value: null },
            _refCount: { value: 0, writable: true },
            _subject: { value: null, writable: true },
            _connection: { value: null, writable: true },
            _subscribe: { value: connectableProto._subscribe },
            _isComplete: { value: connectableProto._isComplete, writable: true },
            getSubject: { value: connectableProto.getSubject },
            connect: { value: connectableProto.connect },
            refCount: { value: connectableProto.refCount }
        };
    })();
    class ConnectableSubscriber extends SubjectSubscriber {
        constructor(destination, connectable) {
            super(destination);
            this.connectable = connectable;
        }
        _error(err) {
            this._unsubscribe();
            super._error(err);
        }
        _complete() {
            this.connectable._isComplete = true;
            this._unsubscribe();
            super._complete();
        }
        _unsubscribe() {
            const connectable = this.connectable;
            if (connectable) {
                this.connectable = null;
                const connection = connectable._connection;
                connectable._refCount = 0;
                connectable._subject = null;
                connectable._connection = null;
                if (connection) {
                    connection.unsubscribe();
                }
            }
        }
    }

    function groupBy(keySelector, elementSelector, durationSelector, subjectSelector) {
        return (source) => source.lift(new GroupByOperator(keySelector, elementSelector, durationSelector, subjectSelector));
    }
    class GroupByOperator {
        constructor(keySelector, elementSelector, durationSelector, subjectSelector) {
            this.keySelector = keySelector;
            this.elementSelector = elementSelector;
            this.durationSelector = durationSelector;
            this.subjectSelector = subjectSelector;
        }
        call(subscriber, source) {
            return source.subscribe(new GroupBySubscriber(subscriber, this.keySelector, this.elementSelector, this.durationSelector, this.subjectSelector));
        }
    }
    class GroupBySubscriber extends Subscriber {
        constructor(destination, keySelector, elementSelector, durationSelector, subjectSelector) {
            super(destination);
            this.keySelector = keySelector;
            this.elementSelector = elementSelector;
            this.durationSelector = durationSelector;
            this.subjectSelector = subjectSelector;
            this.groups = null;
            this.attemptedToUnsubscribe = false;
            this.count = 0;
        }
        _next(value) {
            let key;
            try {
                key = this.keySelector(value);
            }
            catch (err) {
                this.error(err);
                return;
            }
            this._group(value, key);
        }
        _group(value, key) {
            let groups = this.groups;
            if (!groups) {
                groups = this.groups = new Map();
            }
            let group = groups.get(key);
            let element;
            if (this.elementSelector) {
                try {
                    element = this.elementSelector(value);
                }
                catch (err) {
                    this.error(err);
                }
            }
            else {
                element = value;
            }
            if (!group) {
                group = (this.subjectSelector ? this.subjectSelector() : new Subject());
                groups.set(key, group);
                const groupedObservable = new GroupedObservable(key, group, this);
                this.destination.next(groupedObservable);
                if (this.durationSelector) {
                    let duration;
                    try {
                        duration = this.durationSelector(new GroupedObservable(key, group));
                    }
                    catch (err) {
                        this.error(err);
                        return;
                    }
                    this.add(duration.subscribe(new GroupDurationSubscriber(key, group, this)));
                }
            }
            if (!group.closed) {
                group.next(element);
            }
        }
        _error(err) {
            const groups = this.groups;
            if (groups) {
                groups.forEach((group, key) => {
                    group.error(err);
                });
                groups.clear();
            }
            this.destination.error(err);
        }
        _complete() {
            const groups = this.groups;
            if (groups) {
                groups.forEach((group, key) => {
                    group.complete();
                });
                groups.clear();
            }
            this.destination.complete();
        }
        removeGroup(key) {
            this.groups.delete(key);
        }
        unsubscribe() {
            if (!this.closed) {
                this.attemptedToUnsubscribe = true;
                if (this.count === 0) {
                    super.unsubscribe();
                }
            }
        }
    }
    class GroupDurationSubscriber extends Subscriber {
        constructor(key, group, parent) {
            super(group);
            this.key = key;
            this.group = group;
            this.parent = parent;
        }
        _next(value) {
            this.complete();
        }
        _unsubscribe() {
            const { parent, key } = this;
            this.key = this.parent = null;
            if (parent) {
                parent.removeGroup(key);
            }
        }
    }
    class GroupedObservable extends Observable {
        constructor(key, groupSubject, refCountSubscription) {
            super();
            this.key = key;
            this.groupSubject = groupSubject;
            this.refCountSubscription = refCountSubscription;
        }
        _subscribe(subscriber) {
            const subscription = new Subscription();
            const { refCountSubscription, groupSubject } = this;
            if (refCountSubscription && !refCountSubscription.closed) {
                subscription.add(new InnerRefCountSubscription(refCountSubscription));
            }
            subscription.add(groupSubject.subscribe(subscriber));
            return subscription;
        }
    }
    class InnerRefCountSubscription extends Subscription {
        constructor(parent) {
            super();
            this.parent = parent;
            parent.count++;
        }
        unsubscribe() {
            const parent = this.parent;
            if (!parent.closed && !this.closed) {
                super.unsubscribe();
                parent.count -= 1;
                if (parent.count === 0 && parent.attemptedToUnsubscribe) {
                    parent.unsubscribe();
                }
            }
        }
    }

    class BehaviorSubject extends Subject {
        constructor(_value) {
            super();
            this._value = _value;
        }
        get value() {
            return this.getValue();
        }
        _subscribe(subscriber) {
            const subscription = super._subscribe(subscriber);
            if (subscription && !subscription.closed) {
                subscriber.next(this._value);
            }
            return subscription;
        }
        getValue() {
            if (this.hasError) {
                throw this.thrownError;
            }
            else if (this.closed) {
                throw new ObjectUnsubscribedError();
            }
            else {
                return this._value;
            }
        }
        next(value) {
            super.next(this._value = value);
        }
    }

    class Action extends Subscription {
        constructor(scheduler, work) {
            super();
        }
        schedule(state, delay = 0) {
            return this;
        }
    }

    class AsyncAction extends Action {
        constructor(scheduler, work) {
            super(scheduler, work);
            this.scheduler = scheduler;
            this.work = work;
            this.pending = false;
        }
        schedule(state, delay = 0) {
            if (this.closed) {
                return this;
            }
            this.state = state;
            const id = this.id;
            const scheduler = this.scheduler;
            if (id != null) {
                this.id = this.recycleAsyncId(scheduler, id, delay);
            }
            this.pending = true;
            this.delay = delay;
            this.id = this.id || this.requestAsyncId(scheduler, this.id, delay);
            return this;
        }
        requestAsyncId(scheduler, id, delay = 0) {
            return setInterval(scheduler.flush.bind(scheduler, this), delay);
        }
        recycleAsyncId(scheduler, id, delay = 0) {
            if (delay !== null && this.delay === delay && this.pending === false) {
                return id;
            }
            clearInterval(id);
            return undefined;
        }
        execute(state, delay) {
            if (this.closed) {
                return new Error('executing a cancelled action');
            }
            this.pending = false;
            const error = this._execute(state, delay);
            if (error) {
                return error;
            }
            else if (this.pending === false && this.id != null) {
                this.id = this.recycleAsyncId(this.scheduler, this.id, null);
            }
        }
        _execute(state, delay) {
            let errored = false;
            let errorValue = undefined;
            try {
                this.work(state);
            }
            catch (e) {
                errored = true;
                errorValue = !!e && e || new Error(e);
            }
            if (errored) {
                this.unsubscribe();
                return errorValue;
            }
        }
        _unsubscribe() {
            const id = this.id;
            const scheduler = this.scheduler;
            const actions = scheduler.actions;
            const index = actions.indexOf(this);
            this.work = null;
            this.state = null;
            this.pending = false;
            this.scheduler = null;
            if (index !== -1) {
                actions.splice(index, 1);
            }
            if (id != null) {
                this.id = this.recycleAsyncId(scheduler, id, null);
            }
            this.delay = null;
        }
    }

    class QueueAction extends AsyncAction {
        constructor(scheduler, work) {
            super(scheduler, work);
            this.scheduler = scheduler;
            this.work = work;
        }
        schedule(state, delay = 0) {
            if (delay > 0) {
                return super.schedule(state, delay);
            }
            this.delay = delay;
            this.state = state;
            this.scheduler.flush(this);
            return this;
        }
        execute(state, delay) {
            return (delay > 0 || this.closed) ?
                super.execute(state, delay) :
                this._execute(state, delay);
        }
        requestAsyncId(scheduler, id, delay = 0) {
            if ((delay !== null && delay > 0) || (delay === null && this.delay > 0)) {
                return super.requestAsyncId(scheduler, id, delay);
            }
            return scheduler.flush(this);
        }
    }

    class Scheduler {
        constructor(SchedulerAction, now = Scheduler.now) {
            this.SchedulerAction = SchedulerAction;
            this.now = now;
        }
        schedule(work, delay = 0, state) {
            return new this.SchedulerAction(this, work).schedule(state, delay);
        }
    }
    Scheduler.now = () => Date.now();

    class AsyncScheduler extends Scheduler {
        constructor(SchedulerAction, now = Scheduler.now) {
            super(SchedulerAction, () => {
                if (AsyncScheduler.delegate && AsyncScheduler.delegate !== this) {
                    return AsyncScheduler.delegate.now();
                }
                else {
                    return now();
                }
            });
            this.actions = [];
            this.active = false;
            this.scheduled = undefined;
        }
        schedule(work, delay = 0, state) {
            if (AsyncScheduler.delegate && AsyncScheduler.delegate !== this) {
                return AsyncScheduler.delegate.schedule(work, delay, state);
            }
            else {
                return super.schedule(work, delay, state);
            }
        }
        flush(action) {
            const { actions } = this;
            if (this.active) {
                actions.push(action);
                return;
            }
            let error;
            this.active = true;
            do {
                if (error = action.execute(action.state, action.delay)) {
                    break;
                }
            } while (action = actions.shift());
            this.active = false;
            if (error) {
                while (action = actions.shift()) {
                    action.unsubscribe();
                }
                throw error;
            }
        }
    }

    class QueueScheduler extends AsyncScheduler {
    }

    const queue = new QueueScheduler(QueueAction);

    const EMPTY = new Observable(subscriber => subscriber.complete());

    function isScheduler(value) {
        return value && typeof value.schedule === 'function';
    }

    const subscribeToArray = (array) => (subscriber) => {
        for (let i = 0, len = array.length; i < len && !subscriber.closed; i++) {
            subscriber.next(array[i]);
        }
        subscriber.complete();
    };

    function scheduleArray(input, scheduler) {
        return new Observable(subscriber => {
            const sub = new Subscription();
            let i = 0;
            sub.add(scheduler.schedule(function () {
                if (i === input.length) {
                    subscriber.complete();
                    return;
                }
                subscriber.next(input[i++]);
                if (!subscriber.closed) {
                    sub.add(this.schedule());
                }
            }));
            return sub;
        });
    }

    function fromArray(input, scheduler) {
        if (!scheduler) {
            return new Observable(subscribeToArray(input));
        }
        else {
            return scheduleArray(input, scheduler);
        }
    }

    function of(...args) {
        let scheduler = args[args.length - 1];
        if (isScheduler(scheduler)) {
            args.pop();
            return scheduleArray(args, scheduler);
        }
        else {
            return fromArray(args);
        }
    }

    function throwError(error, scheduler) {
        if (!scheduler) {
            return new Observable(subscriber => subscriber.error(error));
        }
        else {
            return new Observable(subscriber => scheduler.schedule(dispatch, 0, { error, subscriber }));
        }
    }
    function dispatch({ error, subscriber }) {
        subscriber.error(error);
    }

    var NotificationKind;
    (function (NotificationKind) {
        NotificationKind["NEXT"] = "N";
        NotificationKind["ERROR"] = "E";
        NotificationKind["COMPLETE"] = "C";
    })(NotificationKind || (NotificationKind = {}));
    class Notification {
        constructor(kind, value, error) {
            this.kind = kind;
            this.value = value;
            this.error = error;
            this.hasValue = kind === 'N';
        }
        observe(observer) {
            switch (this.kind) {
                case 'N':
                    return observer.next && observer.next(this.value);
                case 'E':
                    return observer.error && observer.error(this.error);
                case 'C':
                    return observer.complete && observer.complete();
            }
        }
        do(next, error, complete) {
            const kind = this.kind;
            switch (kind) {
                case 'N':
                    return next && next(this.value);
                case 'E':
                    return error && error(this.error);
                case 'C':
                    return complete && complete();
            }
        }
        accept(nextOrObserver, error, complete) {
            if (nextOrObserver && typeof nextOrObserver.next === 'function') {
                return this.observe(nextOrObserver);
            }
            else {
                return this.do(nextOrObserver, error, complete);
            }
        }
        toObservable() {
            const kind = this.kind;
            switch (kind) {
                case 'N':
                    return of(this.value);
                case 'E':
                    return throwError(this.error);
                case 'C':
                    return EMPTY;
            }
            throw new Error('unexpected notification kind value');
        }
        static createNext(value) {
            if (typeof value !== 'undefined') {
                return new Notification('N', value);
            }
            return Notification.undefinedValueNotification;
        }
        static createError(err) {
            return new Notification('E', undefined, err);
        }
        static createComplete() {
            return Notification.completeNotification;
        }
    }
    Notification.completeNotification = new Notification('C');
    Notification.undefinedValueNotification = new Notification('N', undefined);

    function observeOn(scheduler, delay = 0) {
        return function observeOnOperatorFunction(source) {
            return source.lift(new ObserveOnOperator(scheduler, delay));
        };
    }
    class ObserveOnOperator {
        constructor(scheduler, delay = 0) {
            this.scheduler = scheduler;
            this.delay = delay;
        }
        call(subscriber, source) {
            return source.subscribe(new ObserveOnSubscriber(subscriber, this.scheduler, this.delay));
        }
    }
    class ObserveOnSubscriber extends Subscriber {
        constructor(destination, scheduler, delay = 0) {
            super(destination);
            this.scheduler = scheduler;
            this.delay = delay;
        }
        static dispatch(arg) {
            const { notification, destination } = arg;
            notification.observe(destination);
            this.unsubscribe();
        }
        scheduleMessage(notification) {
            const destination = this.destination;
            destination.add(this.scheduler.schedule(ObserveOnSubscriber.dispatch, this.delay, new ObserveOnMessage(notification, this.destination)));
        }
        _next(value) {
            this.scheduleMessage(Notification.createNext(value));
        }
        _error(err) {
            this.scheduleMessage(Notification.createError(err));
            this.unsubscribe();
        }
        _complete() {
            this.scheduleMessage(Notification.createComplete());
            this.unsubscribe();
        }
    }
    class ObserveOnMessage {
        constructor(notification, destination) {
            this.notification = notification;
            this.destination = destination;
        }
    }

    class ReplaySubject extends Subject {
        constructor(bufferSize = Number.POSITIVE_INFINITY, windowTime = Number.POSITIVE_INFINITY, scheduler) {
            super();
            this.scheduler = scheduler;
            this._events = [];
            this._infiniteTimeWindow = false;
            this._bufferSize = bufferSize < 1 ? 1 : bufferSize;
            this._windowTime = windowTime < 1 ? 1 : windowTime;
            if (windowTime === Number.POSITIVE_INFINITY) {
                this._infiniteTimeWindow = true;
                this.next = this.nextInfiniteTimeWindow;
            }
            else {
                this.next = this.nextTimeWindow;
            }
        }
        nextInfiniteTimeWindow(value) {
            const _events = this._events;
            _events.push(value);
            if (_events.length > this._bufferSize) {
                _events.shift();
            }
            super.next(value);
        }
        nextTimeWindow(value) {
            this._events.push(new ReplayEvent(this._getNow(), value));
            this._trimBufferThenGetEvents();
            super.next(value);
        }
        _subscribe(subscriber) {
            const _infiniteTimeWindow = this._infiniteTimeWindow;
            const _events = _infiniteTimeWindow ? this._events : this._trimBufferThenGetEvents();
            const scheduler = this.scheduler;
            const len = _events.length;
            let subscription;
            if (this.closed) {
                throw new ObjectUnsubscribedError();
            }
            else if (this.isStopped || this.hasError) {
                subscription = Subscription.EMPTY;
            }
            else {
                this.observers.push(subscriber);
                subscription = new SubjectSubscription(this, subscriber);
            }
            if (scheduler) {
                subscriber.add(subscriber = new ObserveOnSubscriber(subscriber, scheduler));
            }
            if (_infiniteTimeWindow) {
                for (let i = 0; i < len && !subscriber.closed; i++) {
                    subscriber.next(_events[i]);
                }
            }
            else {
                for (let i = 0; i < len && !subscriber.closed; i++) {
                    subscriber.next(_events[i].value);
                }
            }
            if (this.hasError) {
                subscriber.error(this.thrownError);
            }
            else if (this.isStopped) {
                subscriber.complete();
            }
            return subscription;
        }
        _getNow() {
            return (this.scheduler || queue).now();
        }
        _trimBufferThenGetEvents() {
            const now = this._getNow();
            const _bufferSize = this._bufferSize;
            const _windowTime = this._windowTime;
            const _events = this._events;
            const eventsCount = _events.length;
            let spliceCount = 0;
            while (spliceCount < eventsCount) {
                if ((now - _events[spliceCount].time) < _windowTime) {
                    break;
                }
                spliceCount++;
            }
            if (eventsCount > _bufferSize) {
                spliceCount = Math.max(spliceCount, eventsCount - _bufferSize);
            }
            if (spliceCount > 0) {
                _events.splice(0, spliceCount);
            }
            return _events;
        }
    }
    class ReplayEvent {
        constructor(time, value) {
            this.time = time;
            this.value = value;
        }
    }

    let nextHandle = 1;
    let resolved;
    const activeHandles = {};
    function findAndClearHandle(handle) {
        if (handle in activeHandles) {
            delete activeHandles[handle];
            return true;
        }
        return false;
    }
    const Immediate = {
        setImmediate(cb) {
            const handle = nextHandle++;
            activeHandles[handle] = true;
            if (!resolved) {
                resolved = Promise.resolve();
            }
            resolved.then(() => findAndClearHandle(handle) && cb());
            return handle;
        },
        clearImmediate(handle) {
            findAndClearHandle(handle);
        },
    };

    class AsapAction extends AsyncAction {
        constructor(scheduler, work) {
            super(scheduler, work);
            this.scheduler = scheduler;
            this.work = work;
        }
        requestAsyncId(scheduler, id, delay = 0) {
            if (delay !== null && delay > 0) {
                return super.requestAsyncId(scheduler, id, delay);
            }
            scheduler.actions.push(this);
            return scheduler.scheduled || (scheduler.scheduled = Immediate.setImmediate(scheduler.flush.bind(scheduler, undefined)));
        }
        recycleAsyncId(scheduler, id, delay = 0) {
            if ((delay !== null && delay > 0) || (delay === null && this.delay > 0)) {
                return super.recycleAsyncId(scheduler, id, delay);
            }
            if (scheduler.actions.length === 0) {
                Immediate.clearImmediate(id);
                scheduler.scheduled = undefined;
            }
            return undefined;
        }
    }

    class AsapScheduler extends AsyncScheduler {
        flush(action) {
            this.active = true;
            this.scheduled = undefined;
            const { actions } = this;
            let error;
            let index = -1;
            let count = actions.length;
            action = action || actions.shift();
            do {
                if (error = action.execute(action.state, action.delay)) {
                    break;
                }
            } while (++index < count && (action = actions.shift()));
            this.active = false;
            if (error) {
                while (++index < count && (action = actions.shift())) {
                    action.unsubscribe();
                }
                throw error;
            }
        }
    }

    const asap = new AsapScheduler(AsapAction);

    const async = new AsyncScheduler(AsyncAction);

    class AnimationFrameAction extends AsyncAction {
        constructor(scheduler, work) {
            super(scheduler, work);
            this.scheduler = scheduler;
            this.work = work;
        }
        requestAsyncId(scheduler, id, delay = 0) {
            if (delay !== null && delay > 0) {
                return super.requestAsyncId(scheduler, id, delay);
            }
            scheduler.actions.push(this);
            return scheduler.scheduled || (scheduler.scheduled = requestAnimationFrame(() => scheduler.flush(undefined)));
        }
        recycleAsyncId(scheduler, id, delay = 0) {
            if ((delay !== null && delay > 0) || (delay === null && this.delay > 0)) {
                return super.recycleAsyncId(scheduler, id, delay);
            }
            if (scheduler.actions.length === 0) {
                cancelAnimationFrame(id);
                scheduler.scheduled = undefined;
            }
            return undefined;
        }
    }

    class AnimationFrameScheduler extends AsyncScheduler {
        flush(action) {
            this.active = true;
            this.scheduled = undefined;
            const { actions } = this;
            let error;
            let index = -1;
            let count = actions.length;
            action = action || actions.shift();
            do {
                if (error = action.execute(action.state, action.delay)) {
                    break;
                }
            } while (++index < count && (action = actions.shift()));
            this.active = false;
            if (error) {
                while (++index < count && (action = actions.shift())) {
                    action.unsubscribe();
                }
                throw error;
            }
        }
    }

    const animationFrame = new AnimationFrameScheduler(AnimationFrameAction);

    function noop() { }

    function isObservable(obj) {
        return !!obj && (obj instanceof Observable || (typeof obj.lift === 'function' && typeof obj.subscribe === 'function'));
    }

    const EmptyErrorImpl = (() => {
        function EmptyErrorImpl() {
            Error.call(this);
            this.message = 'no elements in sequence';
            this.name = 'EmptyError';
            return this;
        }
        EmptyErrorImpl.prototype = Object.create(Error.prototype);
        return EmptyErrorImpl;
    })();
    const EmptyError = EmptyErrorImpl;

    const ArgumentOutOfRangeErrorImpl = (() => {
        function ArgumentOutOfRangeErrorImpl() {
            Error.call(this);
            this.message = 'argument out of range';
            this.name = 'ArgumentOutOfRangeError';
            return this;
        }
        ArgumentOutOfRangeErrorImpl.prototype = Object.create(Error.prototype);
        return ArgumentOutOfRangeErrorImpl;
    })();
    const ArgumentOutOfRangeError = ArgumentOutOfRangeErrorImpl;

    function map(project, thisArg) {
        return function mapOperation(source) {
            if (typeof project !== 'function') {
                throw new TypeError('argument is not a function. Are you looking for `mapTo()`?');
            }
            return source.lift(new MapOperator(project, thisArg));
        };
    }
    class MapOperator {
        constructor(project, thisArg) {
            this.project = project;
            this.thisArg = thisArg;
        }
        call(subscriber, source) {
            return source.subscribe(new MapSubscriber(subscriber, this.project, this.thisArg));
        }
    }
    class MapSubscriber extends Subscriber {
        constructor(destination, project, thisArg) {
            super(destination);
            this.project = project;
            this.count = 0;
            this.thisArg = thisArg || this;
        }
        _next(value) {
            let result;
            try {
                result = this.project.call(this.thisArg, value, this.count++);
            }
            catch (err) {
                this.destination.error(err);
                return;
            }
            this.destination.next(result);
        }
    }

    class OuterSubscriber extends Subscriber {
        notifyNext(outerValue, innerValue, outerIndex, innerIndex, innerSub) {
            this.destination.next(innerValue);
        }
        notifyError(error, innerSub) {
            this.destination.error(error);
        }
        notifyComplete(innerSub) {
            this.destination.complete();
        }
    }

    class InnerSubscriber extends Subscriber {
        constructor(parent, outerValue, outerIndex) {
            super();
            this.parent = parent;
            this.outerValue = outerValue;
            this.outerIndex = outerIndex;
            this.index = 0;
        }
        _next(value) {
            this.parent.notifyNext(this.outerValue, value, this.outerIndex, this.index++, this);
        }
        _error(error) {
            this.parent.notifyError(error, this);
            this.unsubscribe();
        }
        _complete() {
            this.parent.notifyComplete(this);
            this.unsubscribe();
        }
    }

    const subscribeToPromise = (promise) => (subscriber) => {
        promise.then((value) => {
            if (!subscriber.closed) {
                subscriber.next(value);
                subscriber.complete();
            }
        }, (err) => subscriber.error(err))
            .then(null, hostReportError);
        return subscriber;
    };

    function getSymbolIterator() {
        if (typeof Symbol !== 'function' || !Symbol.iterator) {
            return '@@iterator';
        }
        return Symbol.iterator;
    }
    const iterator = getSymbolIterator();

    const subscribeToIterable = (iterable) => (subscriber) => {
        const iterator$1 = iterable[iterator]();
        do {
            const item = iterator$1.next();
            if (item.done) {
                subscriber.complete();
                break;
            }
            subscriber.next(item.value);
            if (subscriber.closed) {
                break;
            }
        } while (true);
        if (typeof iterator$1.return === 'function') {
            subscriber.add(() => {
                if (iterator$1.return) {
                    iterator$1.return();
                }
            });
        }
        return subscriber;
    };

    const subscribeToObservable = (obj) => (subscriber) => {
        const obs = obj[observable]();
        if (typeof obs.subscribe !== 'function') {
            throw new TypeError('Provided object does not correctly implement Symbol.observable');
        }
        else {
            return obs.subscribe(subscriber);
        }
    };

    const isArrayLike = ((x) => x && typeof x.length === 'number' && typeof x !== 'function');

    function isPromise(value) {
        return !!value && typeof value.subscribe !== 'function' && typeof value.then === 'function';
    }

    function subscribeToAsyncIterable(asyncIterable) {
        return (subscriber) => {
            process$1(asyncIterable, subscriber).catch(err => subscriber.error(err));
        };
    }
    function process$1(asyncIterable, subscriber) {
        var asyncIterable_1, asyncIterable_1_1;
        var e_1, _a;
        return __awaiter(this, void 0, void 0, function* () {
            try {
                for (asyncIterable_1 = __asyncValues(asyncIterable); asyncIterable_1_1 = yield asyncIterable_1.next(), !asyncIterable_1_1.done;) {
                    const value = asyncIterable_1_1.value;
                    subscriber.next(value);
                }
            }
            catch (e_1_1) { e_1 = { error: e_1_1 }; }
            finally {
                try {
                    if (asyncIterable_1_1 && !asyncIterable_1_1.done && (_a = asyncIterable_1.return)) yield _a.call(asyncIterable_1);
                }
                finally { if (e_1) throw e_1.error; }
            }
            subscriber.complete();
        });
    }

    const subscribeTo = (result) => {
        if (!!result && typeof result[observable] === 'function') {
            return subscribeToObservable(result);
        }
        else if (isArrayLike(result)) {
            return subscribeToArray(result);
        }
        else if (isPromise(result)) {
            return subscribeToPromise(result);
        }
        else if (!!result && typeof result[iterator] === 'function') {
            return subscribeToIterable(result);
        }
        else if (Symbol && Symbol.asyncIterator &&
            !!result && typeof result[Symbol.asyncIterator] === 'function') {
            return subscribeToAsyncIterable(result);
        }
        else {
            const value = isObject(result) ? 'an invalid object' : `'${result}'`;
            const msg = `You provided ${value} where a stream was expected.`
                + ' You can provide an Observable, Promise, Array, or Iterable.';
            throw new TypeError(msg);
        }
    };

    function subscribeToResult(outerSubscriber, result, outerValue, outerIndex, innerSubscriber = new InnerSubscriber(outerSubscriber, outerValue, outerIndex)) {
        if (innerSubscriber.closed) {
            return undefined;
        }
        if (result instanceof Observable) {
            return result.subscribe(innerSubscriber);
        }
        return subscribeTo(result)(innerSubscriber);
    }

    const NONE = {};
    function combineLatest(...observables) {
        let resultSelector = undefined;
        let scheduler = undefined;
        if (isScheduler(observables[observables.length - 1])) {
            scheduler = observables.pop();
        }
        if (typeof observables[observables.length - 1] === 'function') {
            resultSelector = observables.pop();
        }
        if (observables.length === 1 && isArray(observables[0])) {
            observables = observables[0];
        }
        return fromArray(observables, scheduler).lift(new CombineLatestOperator(resultSelector));
    }
    class CombineLatestOperator {
        constructor(resultSelector) {
            this.resultSelector = resultSelector;
        }
        call(subscriber, source) {
            return source.subscribe(new CombineLatestSubscriber(subscriber, this.resultSelector));
        }
    }
    class CombineLatestSubscriber extends OuterSubscriber {
        constructor(destination, resultSelector) {
            super(destination);
            this.resultSelector = resultSelector;
            this.active = 0;
            this.values = [];
            this.observables = [];
        }
        _next(observable) {
            this.values.push(NONE);
            this.observables.push(observable);
        }
        _complete() {
            const observables = this.observables;
            const len = observables.length;
            if (len === 0) {
                this.destination.complete();
            }
            else {
                this.active = len;
                this.toRespond = len;
                for (let i = 0; i < len; i++) {
                    const observable = observables[i];
                    this.add(subscribeToResult(this, observable, observable, i));
                }
            }
        }
        notifyComplete(unused) {
            if ((this.active -= 1) === 0) {
                this.destination.complete();
            }
        }
        notifyNext(outerValue, innerValue, outerIndex, innerIndex, innerSub) {
            const values = this.values;
            const oldVal = values[outerIndex];
            const toRespond = !this.toRespond
                ? 0
                : oldVal === NONE ? --this.toRespond : this.toRespond;
            values[outerIndex] = innerValue;
            if (toRespond === 0) {
                if (this.resultSelector) {
                    this._tryResultSelector(values);
                }
                else {
                    this.destination.next(values.slice());
                }
            }
        }
        _tryResultSelector(values) {
            let result;
            try {
                result = this.resultSelector.apply(this, values);
            }
            catch (err) {
                this.destination.error(err);
                return;
            }
            this.destination.next(result);
        }
    }

    function scheduleObservable(input, scheduler) {
        return new Observable(subscriber => {
            const sub = new Subscription();
            sub.add(scheduler.schedule(() => {
                const observable$1 = input[observable]();
                sub.add(observable$1.subscribe({
                    next(value) { sub.add(scheduler.schedule(() => subscriber.next(value))); },
                    error(err) { sub.add(scheduler.schedule(() => subscriber.error(err))); },
                    complete() { sub.add(scheduler.schedule(() => subscriber.complete())); },
                }));
            }));
            return sub;
        });
    }

    function schedulePromise(input, scheduler) {
        return new Observable(subscriber => {
            const sub = new Subscription();
            sub.add(scheduler.schedule(() => input.then(value => {
                sub.add(scheduler.schedule(() => {
                    subscriber.next(value);
                    sub.add(scheduler.schedule(() => subscriber.complete()));
                }));
            }, err => {
                sub.add(scheduler.schedule(() => subscriber.error(err)));
            })));
            return sub;
        });
    }

    function scheduleIterable(input, scheduler) {
        if (!input) {
            throw new Error('Iterable cannot be null');
        }
        return new Observable(subscriber => {
            const sub = new Subscription();
            let iterator$1;
            sub.add(() => {
                if (iterator$1 && typeof iterator$1.return === 'function') {
                    iterator$1.return();
                }
            });
            sub.add(scheduler.schedule(() => {
                iterator$1 = input[iterator]();
                sub.add(scheduler.schedule(function () {
                    if (subscriber.closed) {
                        return;
                    }
                    let value;
                    let done;
                    try {
                        const result = iterator$1.next();
                        value = result.value;
                        done = result.done;
                    }
                    catch (err) {
                        subscriber.error(err);
                        return;
                    }
                    if (done) {
                        subscriber.complete();
                    }
                    else {
                        subscriber.next(value);
                        this.schedule();
                    }
                }));
            }));
            return sub;
        });
    }

    function isInteropObservable(input) {
        return input && typeof input[observable] === 'function';
    }

    function isIterable(input) {
        return input && typeof input[iterator] === 'function';
    }

    function scheduleAsyncIterable(input, scheduler) {
        if (!input) {
            throw new Error('Iterable cannot be null');
        }
        return new Observable(subscriber => {
            const sub = new Subscription();
            sub.add(scheduler.schedule(() => {
                const iterator = input[Symbol.asyncIterator]();
                sub.add(scheduler.schedule(function () {
                    iterator.next().then(result => {
                        if (result.done) {
                            subscriber.complete();
                        }
                        else {
                            subscriber.next(result.value);
                            this.schedule();
                        }
                    });
                }));
            }));
            return sub;
        });
    }

    function scheduled(input, scheduler) {
        if (input != null) {
            if (isInteropObservable(input)) {
                return scheduleObservable(input, scheduler);
            }
            else if (isPromise(input)) {
                return schedulePromise(input, scheduler);
            }
            else if (isArrayLike(input)) {
                return scheduleArray(input, scheduler);
            }
            else if (isIterable(input) || typeof input === 'string') {
                return scheduleIterable(input, scheduler);
            }
            else if (Symbol && Symbol.asyncIterator && typeof input[Symbol.asyncIterator] === 'function') {
                return scheduleAsyncIterable(input, scheduler);
            }
        }
        throw new TypeError((input !== null && typeof input || input) + ' is not observable');
    }

    function from(input, scheduler) {
        if (!scheduler) {
            if (input instanceof Observable) {
                return input;
            }
            return new Observable(subscribeTo(input));
        }
        else {
            return scheduled(input, scheduler);
        }
    }

    function mergeMap(project, resultSelector, concurrent = Number.POSITIVE_INFINITY) {
        if (typeof resultSelector === 'function') {
            return (source) => source.pipe(mergeMap((a, i) => from(project(a, i)).pipe(map((b, ii) => resultSelector(a, b, i, ii))), concurrent));
        }
        else if (typeof resultSelector === 'number') {
            concurrent = resultSelector;
        }
        return (source) => source.lift(new MergeMapOperator(project, concurrent));
    }
    class MergeMapOperator {
        constructor(project, concurrent = Number.POSITIVE_INFINITY) {
            this.project = project;
            this.concurrent = concurrent;
        }
        call(observer, source) {
            return source.subscribe(new MergeMapSubscriber(observer, this.project, this.concurrent));
        }
    }
    class MergeMapSubscriber extends OuterSubscriber {
        constructor(destination, project, concurrent = Number.POSITIVE_INFINITY) {
            super(destination);
            this.project = project;
            this.concurrent = concurrent;
            this.hasCompleted = false;
            this.buffer = [];
            this.active = 0;
            this.index = 0;
        }
        _next(value) {
            if (this.active < this.concurrent) {
                this._tryNext(value);
            }
            else {
                this.buffer.push(value);
            }
        }
        _tryNext(value) {
            let result;
            const index = this.index++;
            try {
                result = this.project(value, index);
            }
            catch (err) {
                this.destination.error(err);
                return;
            }
            this.active++;
            this._innerSub(result, value, index);
        }
        _innerSub(ish, value, index) {
            const innerSubscriber = new InnerSubscriber(this, value, index);
            const destination = this.destination;
            destination.add(innerSubscriber);
            const innerSubscription = subscribeToResult(this, ish, undefined, undefined, innerSubscriber);
            if (innerSubscription !== innerSubscriber) {
                destination.add(innerSubscription);
            }
        }
        _complete() {
            this.hasCompleted = true;
            if (this.active === 0 && this.buffer.length === 0) {
                this.destination.complete();
            }
            this.unsubscribe();
        }
        notifyNext(outerValue, innerValue, outerIndex, innerIndex, innerSub) {
            this.destination.next(innerValue);
        }
        notifyComplete(innerSub) {
            const buffer = this.buffer;
            this.remove(innerSub);
            this.active--;
            if (buffer.length > 0) {
                this._next(buffer.shift());
            }
            else if (this.active === 0 && this.hasCompleted) {
                this.destination.complete();
            }
        }
    }

    function mergeAll(concurrent = Number.POSITIVE_INFINITY) {
        return mergeMap(identity, concurrent);
    }

    function concatAll() {
        return mergeAll(1);
    }

    function concat(...observables) {
        return concatAll()(of(...observables));
    }

    function defer(observableFactory) {
        return new Observable(subscriber => {
            let input;
            try {
                input = observableFactory();
            }
            catch (err) {
                subscriber.error(err);
                return undefined;
            }
            const source = input ? from(input) : EMPTY;
            return source.subscribe(subscriber);
        });
    }

    function forkJoin(...sources) {
        if (sources.length === 1) {
            const first = sources[0];
            if (isArray(first)) {
                return forkJoinInternal(first, null);
            }
            if (isObject(first) && Object.getPrototypeOf(first) === Object.prototype) {
                const keys = Object.keys(first);
                return forkJoinInternal(keys.map(key => first[key]), keys);
            }
        }
        if (typeof sources[sources.length - 1] === 'function') {
            const resultSelector = sources.pop();
            sources = (sources.length === 1 && isArray(sources[0])) ? sources[0] : sources;
            return forkJoinInternal(sources, null).pipe(map((args) => resultSelector(...args)));
        }
        return forkJoinInternal(sources, null);
    }
    function forkJoinInternal(sources, keys) {
        return new Observable(subscriber => {
            const len = sources.length;
            if (len === 0) {
                subscriber.complete();
                return;
            }
            const values = new Array(len);
            let completed = 0;
            let emitted = 0;
            for (let i = 0; i < len; i++) {
                const source = from(sources[i]);
                let hasValue = false;
                subscriber.add(source.subscribe({
                    next: value => {
                        if (!hasValue) {
                            hasValue = true;
                            emitted++;
                        }
                        values[i] = value;
                    },
                    error: err => subscriber.error(err),
                    complete: () => {
                        completed++;
                        if (completed === len || !hasValue) {
                            if (emitted === len) {
                                subscriber.next(keys ?
                                    keys.reduce((result, key, i) => (result[key] = values[i], result), {}) :
                                    values);
                            }
                            subscriber.complete();
                        }
                    }
                }));
            }
        });
    }

    function fromEvent(target, eventName, options, resultSelector) {
        if (isFunction(options)) {
            resultSelector = options;
            options = undefined;
        }
        if (resultSelector) {
            return fromEvent(target, eventName, options).pipe(map(args => isArray(args) ? resultSelector(...args) : resultSelector(args)));
        }
        return new Observable(subscriber => {
            function handler(e) {
                if (arguments.length > 1) {
                    subscriber.next(Array.prototype.slice.call(arguments));
                }
                else {
                    subscriber.next(e);
                }
            }
            setupSubscription(target, eventName, handler, subscriber, options);
        });
    }
    function setupSubscription(sourceObj, eventName, handler, subscriber, options) {
        let unsubscribe;
        if (isEventTarget(sourceObj)) {
            const source = sourceObj;
            sourceObj.addEventListener(eventName, handler, options);
            unsubscribe = () => source.removeEventListener(eventName, handler, options);
        }
        else if (isJQueryStyleEventEmitter(sourceObj)) {
            const source = sourceObj;
            sourceObj.on(eventName, handler);
            unsubscribe = () => source.off(eventName, handler);
        }
        else if (isNodeStyleEventEmitter(sourceObj)) {
            const source = sourceObj;
            sourceObj.addListener(eventName, handler);
            unsubscribe = () => source.removeListener(eventName, handler);
        }
        else if (sourceObj && sourceObj.length) {
            for (let i = 0, len = sourceObj.length; i < len; i++) {
                setupSubscription(sourceObj[i], eventName, handler, subscriber, options);
            }
        }
        else {
            throw new TypeError('Invalid event target');
        }
        subscriber.add(unsubscribe);
    }
    function isNodeStyleEventEmitter(sourceObj) {
        return sourceObj && typeof sourceObj.addListener === 'function' && typeof sourceObj.removeListener === 'function';
    }
    function isJQueryStyleEventEmitter(sourceObj) {
        return sourceObj && typeof sourceObj.on === 'function' && typeof sourceObj.off === 'function';
    }
    function isEventTarget(sourceObj) {
        return sourceObj && typeof sourceObj.addEventListener === 'function' && typeof sourceObj.removeEventListener === 'function';
    }

    function isNumeric(val) {
        return !isArray(val) && (val - parseFloat(val) + 1) >= 0;
    }

    function interval(period = 0, scheduler = async) {
        if (!isNumeric(period) || period < 0) {
            period = 0;
        }
        if (!scheduler || typeof scheduler.schedule !== 'function') {
            scheduler = async;
        }
        return new Observable(subscriber => {
            subscriber.add(scheduler.schedule(dispatch$1, period, { subscriber, counter: 0, period }));
            return subscriber;
        });
    }
    function dispatch$1(state) {
        const { subscriber, counter, period } = state;
        subscriber.next(counter);
        this.schedule({ subscriber, counter: counter + 1, period }, period);
    }

    function merge(...observables) {
        let concurrent = Number.POSITIVE_INFINITY;
        let scheduler = undefined;
        let last = observables[observables.length - 1];
        if (isScheduler(last)) {
            scheduler = observables.pop();
            if (observables.length > 1 && typeof observables[observables.length - 1] === 'number') {
                concurrent = observables.pop();
            }
        }
        else if (typeof last === 'number') {
            concurrent = observables.pop();
        }
        if (!scheduler && observables.length === 1 && observables[0] instanceof Observable) {
            return observables[0];
        }
        return mergeAll(concurrent)(fromArray(observables, scheduler));
    }

    function filter(predicate, thisArg) {
        return function filterOperatorFunction(source) {
            return source.lift(new FilterOperator(predicate, thisArg));
        };
    }
    class FilterOperator {
        constructor(predicate, thisArg) {
            this.predicate = predicate;
            this.thisArg = thisArg;
        }
        call(subscriber, source) {
            return source.subscribe(new FilterSubscriber(subscriber, this.predicate, this.thisArg));
        }
    }
    class FilterSubscriber extends Subscriber {
        constructor(destination, predicate, thisArg) {
            super(destination);
            this.predicate = predicate;
            this.thisArg = thisArg;
            this.count = 0;
        }
        _next(value) {
            let result;
            try {
                result = this.predicate.call(this.thisArg, value, this.count++);
            }
            catch (err) {
                this.destination.error(err);
                return;
            }
            if (result) {
                this.destination.next(value);
            }
        }
    }

    function timer(dueTime = 0, periodOrScheduler, scheduler) {
        let period = -1;
        if (isNumeric(periodOrScheduler)) {
            period = Number(periodOrScheduler) < 1 && 1 || Number(periodOrScheduler);
        }
        else if (isScheduler(periodOrScheduler)) {
            scheduler = periodOrScheduler;
        }
        if (!isScheduler(scheduler)) {
            scheduler = async;
        }
        return new Observable(subscriber => {
            const due = isNumeric(dueTime)
                ? dueTime
                : (+dueTime - scheduler.now());
            return scheduler.schedule(dispatch$2, due, {
                index: 0, period, subscriber
            });
        });
    }
    function dispatch$2(state) {
        const { index, period, subscriber } = state;
        subscriber.next(index);
        if (subscriber.closed) {
            return;
        }
        else if (period === -1) {
            return subscriber.complete();
        }
        state.index = index + 1;
        this.schedule(state, period);
    }

    function zip(...observables) {
        const last = observables[observables.length - 1];
        let resultSelector = undefined;
        if (typeof last === 'function') {
            resultSelector = observables.pop();
        }
        return fromArray(observables, undefined).lift(new ZipOperator(resultSelector));
    }
    class ZipOperator {
        constructor(resultSelector) {
            this.resultSelector = resultSelector;
        }
        call(subscriber, source) {
            return source.subscribe(new ZipSubscriber(subscriber, this.resultSelector));
        }
    }
    class ZipSubscriber extends Subscriber {
        constructor(destination, resultSelector, values = Object.create(null)) {
            super(destination);
            this.iterators = [];
            this.active = 0;
            this.resultSelector = resultSelector;
            this.values = values;
        }
        _next(value) {
            const iterators = this.iterators;
            if (isArray(value)) {
                iterators.push(new StaticArrayIterator(value));
            }
            else if (typeof value[iterator] === 'function') {
                iterators.push(new StaticIterator(value[iterator]()));
            }
            else {
                iterators.push(new ZipBufferIterator(this.destination, this, value));
            }
        }
        _complete() {
            const iterators = this.iterators;
            const len = iterators.length;
            this.unsubscribe();
            if (len === 0) {
                this.destination.complete();
                return;
            }
            this.active = len;
            for (let i = 0; i < len; i++) {
                let iterator = iterators[i];
                if (iterator.stillUnsubscribed) {
                    const destination = this.destination;
                    destination.add(iterator.subscribe(iterator, i));
                }
                else {
                    this.active--;
                }
            }
        }
        notifyInactive() {
            this.active--;
            if (this.active === 0) {
                this.destination.complete();
            }
        }
        checkIterators() {
            const iterators = this.iterators;
            const len = iterators.length;
            const destination = this.destination;
            for (let i = 0; i < len; i++) {
                let iterator = iterators[i];
                if (typeof iterator.hasValue === 'function' && !iterator.hasValue()) {
                    return;
                }
            }
            let shouldComplete = false;
            const args = [];
            for (let i = 0; i < len; i++) {
                let iterator = iterators[i];
                let result = iterator.next();
                if (iterator.hasCompleted()) {
                    shouldComplete = true;
                }
                if (result.done) {
                    destination.complete();
                    return;
                }
                args.push(result.value);
            }
            if (this.resultSelector) {
                this._tryresultSelector(args);
            }
            else {
                destination.next(args);
            }
            if (shouldComplete) {
                destination.complete();
            }
        }
        _tryresultSelector(args) {
            let result;
            try {
                result = this.resultSelector.apply(this, args);
            }
            catch (err) {
                this.destination.error(err);
                return;
            }
            this.destination.next(result);
        }
    }
    class StaticIterator {
        constructor(iterator) {
            this.iterator = iterator;
            this.nextResult = iterator.next();
        }
        hasValue() {
            return true;
        }
        next() {
            const result = this.nextResult;
            this.nextResult = this.iterator.next();
            return result;
        }
        hasCompleted() {
            const nextResult = this.nextResult;
            return nextResult && !!nextResult.done;
        }
    }
    class StaticArrayIterator {
        constructor(array) {
            this.array = array;
            this.index = 0;
            this.length = 0;
            this.length = array.length;
        }
        [iterator]() {
            return this;
        }
        next(value) {
            const i = this.index++;
            const array = this.array;
            return i < this.length ? { value: array[i], done: false } : { value: null, done: true };
        }
        hasValue() {
            return this.array.length > this.index;
        }
        hasCompleted() {
            return this.array.length === this.index;
        }
    }
    class ZipBufferIterator extends OuterSubscriber {
        constructor(destination, parent, observable) {
            super(destination);
            this.parent = parent;
            this.observable = observable;
            this.stillUnsubscribed = true;
            this.buffer = [];
            this.isComplete = false;
        }
        [iterator]() {
            return this;
        }
        next() {
            const buffer = this.buffer;
            if (buffer.length === 0 && this.isComplete) {
                return { value: null, done: true };
            }
            else {
                return { value: buffer.shift(), done: false };
            }
        }
        hasValue() {
            return this.buffer.length > 0;
        }
        hasCompleted() {
            return this.buffer.length === 0 && this.isComplete;
        }
        notifyComplete() {
            if (this.buffer.length > 0) {
                this.isComplete = true;
                this.parent.notifyInactive();
            }
            else {
                this.destination.complete();
            }
        }
        notifyNext(outerValue, innerValue, outerIndex, innerIndex, innerSub) {
            this.buffer.push(innerValue);
            this.parent.checkIterators();
        }
        subscribe(value, index) {
            return subscribeToResult(this, this.observable, this, index);
        }
    }

    function audit(durationSelector) {
        return function auditOperatorFunction(source) {
            return source.lift(new AuditOperator(durationSelector));
        };
    }
    class AuditOperator {
        constructor(durationSelector) {
            this.durationSelector = durationSelector;
        }
        call(subscriber, source) {
            return source.subscribe(new AuditSubscriber(subscriber, this.durationSelector));
        }
    }
    class AuditSubscriber extends OuterSubscriber {
        constructor(destination, durationSelector) {
            super(destination);
            this.durationSelector = durationSelector;
            this.value = null;
            this.hasValue = false;
            this.throttled = null;
        }
        _next(value) {
            this.value = value;
            this.hasValue = true;
            if (!this.throttled) {
                let duration;
                try {
                    const { durationSelector } = this;
                    duration = durationSelector(value);
                }
                catch (err) {
                    return this.destination.error(err);
                }
                const innerSubscription = subscribeToResult(this, duration);
                if (!innerSubscription || innerSubscription.closed) {
                    this.clearThrottle();
                }
                else {
                    this.add(this.throttled = innerSubscription);
                }
            }
        }
        clearThrottle() {
            const { value, hasValue, throttled } = this;
            if (throttled) {
                this.remove(throttled);
                this.throttled = null;
                throttled.unsubscribe();
            }
            if (hasValue) {
                this.value = null;
                this.hasValue = false;
                this.destination.next(value);
            }
        }
        notifyNext(outerValue, innerValue, outerIndex, innerIndex) {
            this.clearThrottle();
        }
        notifyComplete() {
            this.clearThrottle();
        }
    }

    function auditTime(duration, scheduler = async) {
        return audit(() => timer(duration, scheduler));
    }

    function catchError(selector) {
        return function catchErrorOperatorFunction(source) {
            const operator = new CatchOperator(selector);
            const caught = source.lift(operator);
            return (operator.caught = caught);
        };
    }
    class CatchOperator {
        constructor(selector) {
            this.selector = selector;
        }
        call(subscriber, source) {
            return source.subscribe(new CatchSubscriber(subscriber, this.selector, this.caught));
        }
    }
    class CatchSubscriber extends OuterSubscriber {
        constructor(destination, selector, caught) {
            super(destination);
            this.selector = selector;
            this.caught = caught;
        }
        error(err) {
            if (!this.isStopped) {
                let result;
                try {
                    result = this.selector(err, this.caught);
                }
                catch (err2) {
                    super.error(err2);
                    return;
                }
                this._unsubscribeAndRecycle();
                const innerSubscriber = new InnerSubscriber(this, undefined, undefined);
                this.add(innerSubscriber);
                const innerSubscription = subscribeToResult(this, result, undefined, undefined, innerSubscriber);
                if (innerSubscription !== innerSubscriber) {
                    this.add(innerSubscription);
                }
            }
        }
    }

    function combineLatest$1(...observables) {
        let project = undefined;
        if (typeof observables[observables.length - 1] === 'function') {
            project = observables.pop();
        }
        if (observables.length === 1 && isArray(observables[0])) {
            observables = observables[0].slice();
        }
        return (source) => source.lift.call(from([source, ...observables]), new CombineLatestOperator(project));
    }
    function combineLatestWith(...otherSources) {
        return combineLatest$1(...otherSources);
    }

    function concatMap(project, resultSelector) {
        if (typeof resultSelector === 'function') {
            return mergeMap(project, resultSelector, 1);
        }
        return mergeMap(project, 1);
    }

    function debounce(durationSelector) {
        return (source) => source.lift(new DebounceOperator(durationSelector));
    }
    class DebounceOperator {
        constructor(durationSelector) {
            this.durationSelector = durationSelector;
        }
        call(subscriber, source) {
            return source.subscribe(new DebounceSubscriber(subscriber, this.durationSelector));
        }
    }
    class DebounceSubscriber extends OuterSubscriber {
        constructor(destination, durationSelector) {
            super(destination);
            this.durationSelector = durationSelector;
            this.value = null;
            this.hasValue = false;
            this.durationSubscription = null;
        }
        _next(value) {
            try {
                const result = this.durationSelector.call(this, value);
                if (result) {
                    this._tryNext(value, result);
                }
            }
            catch (err) {
                this.destination.error(err);
            }
        }
        _complete() {
            this.emitValue();
            this.destination.complete();
        }
        _tryNext(value, duration) {
            let subscription = this.durationSubscription;
            this.value = value;
            this.hasValue = true;
            if (subscription) {
                subscription.unsubscribe();
                this.remove(subscription);
            }
            subscription = subscribeToResult(this, duration);
            if (subscription && !subscription.closed) {
                this.add(this.durationSubscription = subscription);
            }
        }
        notifyNext(outerValue, innerValue, outerIndex, innerIndex, innerSub) {
            this.emitValue();
        }
        notifyComplete() {
            this.emitValue();
        }
        emitValue() {
            if (this.hasValue) {
                const value = this.value;
                const subscription = this.durationSubscription;
                if (subscription) {
                    this.durationSubscription = null;
                    subscription.unsubscribe();
                    this.remove(subscription);
                }
                this.value = null;
                this.hasValue = false;
                super._next(value);
            }
        }
    }

    function debounceTime(dueTime, scheduler = async) {
        return (source) => source.lift(new DebounceTimeOperator(dueTime, scheduler));
    }
    class DebounceTimeOperator {
        constructor(dueTime, scheduler) {
            this.dueTime = dueTime;
            this.scheduler = scheduler;
        }
        call(subscriber, source) {
            return source.subscribe(new DebounceTimeSubscriber(subscriber, this.dueTime, this.scheduler));
        }
    }
    class DebounceTimeSubscriber extends Subscriber {
        constructor(destination, dueTime, scheduler) {
            super(destination);
            this.dueTime = dueTime;
            this.scheduler = scheduler;
            this.debouncedSubscription = null;
            this.lastValue = null;
            this.hasValue = false;
        }
        _next(value) {
            this.clearDebounce();
            this.lastValue = value;
            this.hasValue = true;
            this.add(this.debouncedSubscription = this.scheduler.schedule(dispatchNext, this.dueTime, this));
        }
        _complete() {
            this.debouncedNext();
            this.destination.complete();
        }
        debouncedNext() {
            this.clearDebounce();
            if (this.hasValue) {
                const { lastValue } = this;
                this.lastValue = null;
                this.hasValue = false;
                this.destination.next(lastValue);
            }
        }
        clearDebounce() {
            const debouncedSubscription = this.debouncedSubscription;
            if (debouncedSubscription !== null) {
                this.remove(debouncedSubscription);
                debouncedSubscription.unsubscribe();
                this.debouncedSubscription = null;
            }
        }
    }
    function dispatchNext(subscriber) {
        subscriber.debouncedNext();
    }

    function defaultIfEmpty(defaultValue = null) {
        return (source) => source.lift(new DefaultIfEmptyOperator(defaultValue));
    }
    class DefaultIfEmptyOperator {
        constructor(defaultValue) {
            this.defaultValue = defaultValue;
        }
        call(subscriber, source) {
            return source.subscribe(new DefaultIfEmptySubscriber(subscriber, this.defaultValue));
        }
    }
    class DefaultIfEmptySubscriber extends Subscriber {
        constructor(destination, defaultValue) {
            super(destination);
            this.defaultValue = defaultValue;
            this.isEmpty = true;
        }
        _next(value) {
            this.isEmpty = false;
            this.destination.next(value);
        }
        _complete() {
            if (this.isEmpty) {
                this.destination.next(this.defaultValue);
            }
            this.destination.complete();
        }
    }

    function isDate(value) {
        return value instanceof Date && !isNaN(+value);
    }

    function delay(delay, scheduler = async) {
        const absoluteDelay = isDate(delay);
        const delayFor = absoluteDelay ? (+delay - scheduler.now()) : Math.abs(delay);
        return (source) => source.lift(new DelayOperator(delayFor, scheduler));
    }
    class DelayOperator {
        constructor(delay, scheduler) {
            this.delay = delay;
            this.scheduler = scheduler;
        }
        call(subscriber, source) {
            return source.subscribe(new DelaySubscriber(subscriber, this.delay, this.scheduler));
        }
    }
    class DelaySubscriber extends Subscriber {
        constructor(destination, delay, scheduler) {
            super(destination);
            this.delay = delay;
            this.scheduler = scheduler;
            this.queue = [];
            this.active = false;
            this.errored = false;
        }
        static dispatch(state) {
            const source = state.source;
            const queue = source.queue;
            const scheduler = state.scheduler;
            const destination = state.destination;
            while (queue.length > 0 && (queue[0].time - scheduler.now()) <= 0) {
                queue.shift().notification.observe(destination);
            }
            if (queue.length > 0) {
                const delay = Math.max(0, queue[0].time - scheduler.now());
                this.schedule(state, delay);
            }
            else if (source.isStopped) {
                source.destination.complete();
                source.active = false;
            }
            else {
                this.unsubscribe();
                source.active = false;
            }
        }
        _schedule(scheduler) {
            this.active = true;
            const destination = this.destination;
            destination.add(scheduler.schedule(DelaySubscriber.dispatch, this.delay, {
                source: this, destination: this.destination, scheduler: scheduler
            }));
        }
        scheduleNotification(notification) {
            if (this.errored === true) {
                return;
            }
            const scheduler = this.scheduler;
            const message = new DelayMessage(scheduler.now() + this.delay, notification);
            this.queue.push(message);
            if (this.active === false) {
                this._schedule(scheduler);
            }
        }
        _next(value) {
            this.scheduleNotification(Notification.createNext(value));
        }
        _error(err) {
            this.errored = true;
            this.queue = [];
            this.destination.error(err);
            this.unsubscribe();
        }
        _complete() {
            if (this.queue.length === 0) {
                this.destination.complete();
            }
            this.unsubscribe();
        }
    }
    class DelayMessage {
        constructor(time, notification) {
            this.time = time;
            this.notification = notification;
        }
    }

    function delayWhen(delayDurationSelector, subscriptionDelay) {
        if (subscriptionDelay) {
            return (source) => new SubscriptionDelayObservable(source, subscriptionDelay)
                .lift(new DelayWhenOperator(delayDurationSelector));
        }
        return (source) => source.lift(new DelayWhenOperator(delayDurationSelector));
    }
    class DelayWhenOperator {
        constructor(delayDurationSelector) {
            this.delayDurationSelector = delayDurationSelector;
        }
        call(subscriber, source) {
            return source.subscribe(new DelayWhenSubscriber(subscriber, this.delayDurationSelector));
        }
    }
    class DelayWhenSubscriber extends OuterSubscriber {
        constructor(destination, delayDurationSelector) {
            super(destination);
            this.delayDurationSelector = delayDurationSelector;
            this.completed = false;
            this.delayNotifierSubscriptions = [];
            this.index = 0;
        }
        notifyNext(outerValue, innerValue, outerIndex, innerIndex, innerSub) {
            this.destination.next(outerValue);
            this.removeSubscription(innerSub);
            this.tryComplete();
        }
        notifyError(error, innerSub) {
            this._error(error);
        }
        notifyComplete(innerSub) {
            const value = this.removeSubscription(innerSub);
            if (value) {
                this.destination.next(value);
            }
            this.tryComplete();
        }
        _next(value) {
            const index = this.index++;
            try {
                const delayNotifier = this.delayDurationSelector(value, index);
                if (delayNotifier) {
                    this.tryDelay(delayNotifier, value);
                }
            }
            catch (err) {
                this.destination.error(err);
            }
        }
        _complete() {
            this.completed = true;
            this.tryComplete();
            this.unsubscribe();
        }
        removeSubscription(subscription) {
            subscription.unsubscribe();
            const subscriptionIdx = this.delayNotifierSubscriptions.indexOf(subscription);
            if (subscriptionIdx !== -1) {
                this.delayNotifierSubscriptions.splice(subscriptionIdx, 1);
            }
            return subscription.outerValue;
        }
        tryDelay(delayNotifier, value) {
            const notifierSubscription = subscribeToResult(this, delayNotifier, value);
            if (notifierSubscription && !notifierSubscription.closed) {
                const destination = this.destination;
                destination.add(notifierSubscription);
                this.delayNotifierSubscriptions.push(notifierSubscription);
            }
        }
        tryComplete() {
            if (this.completed && this.delayNotifierSubscriptions.length === 0) {
                this.destination.complete();
            }
        }
    }
    class SubscriptionDelayObservable extends Observable {
        constructor(source, subscriptionDelay) {
            super();
            this.source = source;
            this.subscriptionDelay = subscriptionDelay;
        }
        _subscribe(subscriber) {
            this.subscriptionDelay.subscribe(new SubscriptionDelaySubscriber(subscriber, this.source));
        }
    }
    class SubscriptionDelaySubscriber extends Subscriber {
        constructor(parent, source) {
            super();
            this.parent = parent;
            this.source = source;
            this.sourceSubscribed = false;
        }
        _next(unused) {
            this.subscribeToSource();
        }
        _error(err) {
            this.unsubscribe();
            this.parent.error(err);
        }
        _complete() {
            this.unsubscribe();
            this.subscribeToSource();
        }
        subscribeToSource() {
            if (!this.sourceSubscribed) {
                this.sourceSubscribed = true;
                this.unsubscribe();
                this.source.subscribe(this.parent);
            }
        }
    }

    function dematerialize() {
        return function dematerializeOperatorFunction(source) {
            return source.lift(new DeMaterializeOperator());
        };
    }
    class DeMaterializeOperator {
        call(subscriber, source) {
            return source.subscribe(new DeMaterializeSubscriber(subscriber));
        }
    }
    class DeMaterializeSubscriber extends Subscriber {
        constructor(destination) {
            super(destination);
        }
        _next(value) {
            value.observe(this.destination);
        }
    }

    function distinctUntilChanged(compare, keySelector) {
        return (source) => source.lift(new DistinctUntilChangedOperator(compare, keySelector));
    }
    class DistinctUntilChangedOperator {
        constructor(compare, keySelector) {
            this.compare = compare;
            this.keySelector = keySelector;
        }
        call(subscriber, source) {
            return source.subscribe(new DistinctUntilChangedSubscriber(subscriber, this.compare, this.keySelector));
        }
    }
    class DistinctUntilChangedSubscriber extends Subscriber {
        constructor(destination, compare, keySelector) {
            super(destination);
            this.keySelector = keySelector;
            this.hasKey = false;
            if (typeof compare === 'function') {
                this.compare = compare;
            }
        }
        compare(x, y) {
            return x === y;
        }
        _next(value) {
            let key;
            try {
                const { keySelector } = this;
                key = keySelector ? keySelector(value) : value;
            }
            catch (err) {
                return this.destination.error(err);
            }
            let result = false;
            if (this.hasKey) {
                try {
                    const { compare } = this;
                    result = compare(this.key, key);
                }
                catch (err) {
                    return this.destination.error(err);
                }
            }
            else {
                this.hasKey = true;
            }
            if (!result) {
                this.key = key;
                this.destination.next(value);
            }
        }
    }

    function throwIfEmpty(errorFactory = defaultErrorFactory) {
        return (source) => {
            return source.lift(new ThrowIfEmptyOperator(errorFactory));
        };
    }
    class ThrowIfEmptyOperator {
        constructor(errorFactory) {
            this.errorFactory = errorFactory;
        }
        call(subscriber, source) {
            return source.subscribe(new ThrowIfEmptySubscriber(subscriber, this.errorFactory));
        }
    }
    class ThrowIfEmptySubscriber extends Subscriber {
        constructor(destination, errorFactory) {
            super(destination);
            this.errorFactory = errorFactory;
            this.hasValue = false;
        }
        _next(value) {
            this.hasValue = true;
            this.destination.next(value);
        }
        _complete() {
            if (!this.hasValue) {
                let err;
                try {
                    err = this.errorFactory();
                }
                catch (e) {
                    err = e;
                }
                this.destination.error(err);
            }
            else {
                return this.destination.complete();
            }
        }
    }
    function defaultErrorFactory() {
        return new EmptyError();
    }

    function take(count) {
        return (source) => {
            if (count === 0) {
                return EMPTY;
            }
            else {
                return source.lift(new TakeOperator(count));
            }
        };
    }
    class TakeOperator {
        constructor(total) {
            this.total = total;
            if (this.total < 0) {
                throw new ArgumentOutOfRangeError;
            }
        }
        call(subscriber, source) {
            return source.subscribe(new TakeSubscriber(subscriber, this.total));
        }
    }
    class TakeSubscriber extends Subscriber {
        constructor(destination, total) {
            super(destination);
            this.total = total;
            this.count = 0;
        }
        _next(value) {
            const total = this.total;
            const count = ++this.count;
            if (count <= total) {
                this.destination.next(value);
                if (count === total) {
                    this.destination.complete();
                    this.unsubscribe();
                }
            }
        }
    }

    function exhaustMap(project, resultSelector) {
        if (resultSelector) {
            return (source) => source.pipe(exhaustMap((a, i) => from(project(a, i)).pipe(map((b, ii) => resultSelector(a, b, i, ii)))));
        }
        return (source) => source.lift(new ExhaustMapOperator(project));
    }
    class ExhaustMapOperator {
        constructor(project) {
            this.project = project;
        }
        call(subscriber, source) {
            return source.subscribe(new ExhaustMapSubscriber(subscriber, this.project));
        }
    }
    class ExhaustMapSubscriber extends OuterSubscriber {
        constructor(destination, project) {
            super(destination);
            this.project = project;
            this.hasSubscription = false;
            this.hasCompleted = false;
            this.index = 0;
        }
        _next(value) {
            if (!this.hasSubscription) {
                this.tryNext(value);
            }
        }
        tryNext(value) {
            let result;
            const index = this.index++;
            try {
                result = this.project(value, index);
            }
            catch (err) {
                this.destination.error(err);
                return;
            }
            this.hasSubscription = true;
            this._innerSub(result, value, index);
        }
        _innerSub(result, value, index) {
            const innerSubscriber = new InnerSubscriber(this, value, index);
            const destination = this.destination;
            destination.add(innerSubscriber);
            const innerSubscription = subscribeToResult(this, result, undefined, undefined, innerSubscriber);
            if (innerSubscription !== innerSubscriber) {
                destination.add(innerSubscription);
            }
        }
        _complete() {
            this.hasCompleted = true;
            if (!this.hasSubscription) {
                this.destination.complete();
            }
            this.unsubscribe();
        }
        notifyNext(outerValue, innerValue, outerIndex, innerIndex, innerSub) {
            this.destination.next(innerValue);
        }
        notifyError(err) {
            this.destination.error(err);
        }
        notifyComplete(innerSub) {
            const destination = this.destination;
            destination.remove(innerSub);
            this.hasSubscription = false;
            if (this.hasCompleted) {
                this.destination.complete();
            }
        }
    }

    function finalize(callback) {
        return (source) => source.lift(new FinallyOperator(callback));
    }
    class FinallyOperator {
        constructor(callback) {
            this.callback = callback;
        }
        call(subscriber, source) {
            return source.subscribe(new FinallySubscriber(subscriber, this.callback));
        }
    }
    class FinallySubscriber extends Subscriber {
        constructor(destination, callback) {
            super(destination);
            this.add(new Subscription(callback));
        }
    }

    function ignoreElements() {
        return function ignoreElementsOperatorFunction(source) {
            return source.lift(new IgnoreElementsOperator());
        };
    }
    class IgnoreElementsOperator {
        call(subscriber, source) {
            return source.subscribe(new IgnoreElementsSubscriber(subscriber));
        }
    }
    class IgnoreElementsSubscriber extends Subscriber {
        _next(unused) {
        }
    }

    function takeLast(count) {
        return function takeLastOperatorFunction(source) {
            if (count === 0) {
                return EMPTY;
            }
            else {
                return source.lift(new TakeLastOperator(count));
            }
        };
    }
    class TakeLastOperator {
        constructor(total) {
            this.total = total;
            if (this.total < 0) {
                throw new ArgumentOutOfRangeError;
            }
        }
        call(subscriber, source) {
            return source.subscribe(new TakeLastSubscriber(subscriber, this.total));
        }
    }
    class TakeLastSubscriber extends Subscriber {
        constructor(destination, total) {
            super(destination);
            this.total = total;
            this.ring = new Array();
            this.count = 0;
        }
        _next(value) {
            const ring = this.ring;
            const total = this.total;
            const count = this.count++;
            if (ring.length < total) {
                ring.push(value);
            }
            else {
                const index = count % total;
                ring[index] = value;
            }
        }
        _complete() {
            const destination = this.destination;
            let count = this.count;
            if (count > 0) {
                const total = this.count >= this.total ? this.total : this.count;
                const ring = this.ring;
                for (let i = 0; i < total; i++) {
                    const idx = (count++) % total;
                    destination.next(ring[idx]);
                }
            }
            destination.complete();
        }
    }

    function last(predicate, defaultValue) {
        const hasDefaultValue = arguments.length >= 2;
        return (source) => source.pipe(predicate ? filter((v, i) => predicate(v, i, source)) : identity, takeLast(1), hasDefaultValue ? defaultIfEmpty(defaultValue) : throwIfEmpty(() => new EmptyError()));
    }

    function materialize() {
        return function materializeOperatorFunction(source) {
            return source.lift(new MaterializeOperator());
        };
    }
    class MaterializeOperator {
        call(subscriber, source) {
            return source.subscribe(new MaterializeSubscriber(subscriber));
        }
    }
    class MaterializeSubscriber extends Subscriber {
        constructor(destination) {
            super(destination);
        }
        _next(value) {
            this.destination.next(Notification.createNext(value));
        }
        _error(err) {
            const destination = this.destination;
            destination.next(Notification.createError(err));
            destination.complete();
        }
        _complete() {
            const destination = this.destination;
            destination.next(Notification.createComplete());
            destination.complete();
        }
    }

    function scan(accumulator, seed) {
        let hasSeed = false;
        if (arguments.length >= 2) {
            hasSeed = true;
        }
        return function scanOperatorFunction(source) {
            return source.lift(new ScanOperator(accumulator, seed, hasSeed));
        };
    }
    class ScanOperator {
        constructor(accumulator, seed, hasSeed = false) {
            this.accumulator = accumulator;
            this.seed = seed;
            this.hasSeed = hasSeed;
        }
        call(subscriber, source) {
            return source.subscribe(new ScanSubscriber(subscriber, this.accumulator, this.seed, this.hasSeed));
        }
    }
    class ScanSubscriber extends Subscriber {
        constructor(destination, accumulator, _state, _hasState) {
            super(destination);
            this.accumulator = accumulator;
            this._state = _state;
            this._hasState = _hasState;
            this.index = 0;
        }
        _next(value) {
            const { destination } = this;
            if (!this._hasState) {
                this._state = value;
                this._hasState = true;
                destination.next(value);
            }
            else {
                const index = this.index++;
                let result;
                try {
                    result = this.accumulator(this._state, value, index);
                }
                catch (err) {
                    destination.error(err);
                    return;
                }
                this._state = result;
                destination.next(result);
            }
        }
    }

    function multicast(subjectOrSubjectFactory, selector) {
        return function multicastOperatorFunction(source) {
            let subjectFactory;
            if (typeof subjectOrSubjectFactory === 'function') {
                subjectFactory = subjectOrSubjectFactory;
            }
            else {
                subjectFactory = function subjectFactory() {
                    return subjectOrSubjectFactory;
                };
            }
            if (typeof selector === 'function') {
                return source.lift(new MulticastOperator(subjectFactory, selector));
            }
            const connectable = Object.create(source, connectableObservableDescriptor);
            connectable.source = source;
            connectable.subjectFactory = subjectFactory;
            return connectable;
        };
    }
    class MulticastOperator {
        constructor(subjectFactory, selector) {
            this.subjectFactory = subjectFactory;
            this.selector = selector;
        }
        call(subscriber, source) {
            const { selector } = this;
            const subject = this.subjectFactory();
            const subscription = selector(subject).subscribe(subscriber);
            subscription.add(source.subscribe(subject));
            return subscription;
        }
    }

    function pairwise() {
        return (source) => source.lift(new PairwiseOperator());
    }
    class PairwiseOperator {
        call(subscriber, source) {
            return source.subscribe(new PairwiseSubscriber(subscriber));
        }
    }
    class PairwiseSubscriber extends Subscriber {
        constructor(destination) {
            super(destination);
            this.hasPrev = false;
        }
        _next(value) {
            let pair;
            if (this.hasPrev) {
                pair = [this.prev, value];
            }
            else {
                this.hasPrev = true;
            }
            this.prev = value;
            if (pair) {
                this.destination.next(pair);
            }
        }
    }

    function pluck(...properties) {
        const length = properties.length;
        if (length === 0) {
            throw new Error('list of properties cannot be empty.');
        }
        return map((x) => {
            let currentProp = x;
            for (let i = 0; i < length; i++) {
                const p = currentProp[properties[i]];
                if (typeof p !== 'undefined') {
                    currentProp = p;
                }
                else {
                    return undefined;
                }
            }
            return currentProp;
        });
    }

    function repeatWhen(notifier) {
        return (source) => source.lift(new RepeatWhenOperator(notifier));
    }
    class RepeatWhenOperator {
        constructor(notifier) {
            this.notifier = notifier;
        }
        call(subscriber, source) {
            return source.subscribe(new RepeatWhenSubscriber(subscriber, this.notifier, source));
        }
    }
    class RepeatWhenSubscriber extends OuterSubscriber {
        constructor(destination, notifier, source) {
            super(destination);
            this.notifier = notifier;
            this.source = source;
            this.notifications = null;
            this.retries = null;
            this.retriesSubscription = null;
            this.sourceIsBeingSubscribedTo = true;
        }
        notifyNext(outerValue, innerValue, outerIndex, innerIndex, innerSub) {
            this.sourceIsBeingSubscribedTo = true;
            this.source.subscribe(this);
        }
        notifyComplete(innerSub) {
            if (this.sourceIsBeingSubscribedTo === false) {
                return super.complete();
            }
        }
        complete() {
            this.sourceIsBeingSubscribedTo = false;
            if (!this.isStopped) {
                if (!this.retries) {
                    this.subscribeToRetries();
                }
                if (!this.retriesSubscription || this.retriesSubscription.closed) {
                    return super.complete();
                }
                this._unsubscribeAndRecycle();
                this.notifications.next();
            }
        }
        _unsubscribe() {
            const { notifications, retriesSubscription } = this;
            if (notifications) {
                notifications.unsubscribe();
                this.notifications = null;
            }
            if (retriesSubscription) {
                retriesSubscription.unsubscribe();
                this.retriesSubscription = null;
            }
            this.retries = null;
        }
        _unsubscribeAndRecycle() {
            const { _unsubscribe } = this;
            this._unsubscribe = null;
            super._unsubscribeAndRecycle();
            this._unsubscribe = _unsubscribe;
            return this;
        }
        subscribeToRetries() {
            this.notifications = new Subject();
            let retries;
            try {
                const { notifier } = this;
                retries = notifier(this.notifications);
            }
            catch (e) {
                return super.complete();
            }
            this.retries = retries;
            this.retriesSubscription = subscribeToResult(this, retries);
        }
    }

    function shareSubjectFactory() {
        return new Subject();
    }
    function share() {
        return (source) => refCount()(multicast(shareSubjectFactory)(source));
    }

    function shareReplay(configOrBufferSize, windowTime, scheduler) {
        let config;
        if (configOrBufferSize && typeof configOrBufferSize === 'object') {
            config = configOrBufferSize;
        }
        else {
            config = {
                bufferSize: configOrBufferSize,
                windowTime,
                refCount: false,
                scheduler
            };
        }
        return (source) => source.lift(shareReplayOperator(config));
    }
    function shareReplayOperator({ bufferSize = Number.POSITIVE_INFINITY, windowTime = Number.POSITIVE_INFINITY, refCount: useRefCount, scheduler }) {
        let subject;
        let refCount = 0;
        let subscription;
        let hasError = false;
        let isComplete = false;
        return function shareReplayOperation(source) {
            refCount++;
            if (!subject || hasError) {
                hasError = false;
                subject = new ReplaySubject(bufferSize, windowTime, scheduler);
                subscription = source.subscribe({
                    next(value) { subject.next(value); },
                    error(err) {
                        hasError = true;
                        subject.error(err);
                    },
                    complete() {
                        isComplete = true;
                        subscription = undefined;
                        subject.complete();
                    },
                });
            }
            const innerSub = subject.subscribe(this);
            this.add(() => {
                refCount--;
                innerSub.unsubscribe();
                if (subscription && !isComplete && useRefCount && refCount === 0) {
                    subscription.unsubscribe();
                    subscription = undefined;
                    subject = undefined;
                }
            });
        };
    }

    function skip(count) {
        return (source) => source.lift(new SkipOperator(count));
    }
    class SkipOperator {
        constructor(total) {
            this.total = total;
        }
        call(subscriber, source) {
            return source.subscribe(new SkipSubscriber(subscriber, this.total));
        }
    }
    class SkipSubscriber extends Subscriber {
        constructor(destination, total) {
            super(destination);
            this.total = total;
            this.count = 0;
        }
        _next(x) {
            if (++this.count > this.total) {
                this.destination.next(x);
            }
        }
    }

    function startWith(...values) {
        const scheduler = values[values.length - 1];
        if (isScheduler(scheduler)) {
            values.pop();
            return (source) => concat(values, source, scheduler);
        }
        else {
            return (source) => concat(values, source);
        }
    }

    function switchMap(project, resultSelector) {
        if (typeof resultSelector === 'function') {
            return (source) => source.pipe(switchMap((a, i) => from(project(a, i)).pipe(map((b, ii) => resultSelector(a, b, i, ii)))));
        }
        return (source) => source.lift(new SwitchMapOperator(project));
    }
    class SwitchMapOperator {
        constructor(project) {
            this.project = project;
        }
        call(subscriber, source) {
            return source.subscribe(new SwitchMapSubscriber(subscriber, this.project));
        }
    }
    class SwitchMapSubscriber extends OuterSubscriber {
        constructor(destination, project) {
            super(destination);
            this.project = project;
            this.index = 0;
        }
        _next(value) {
            let result;
            const index = this.index++;
            try {
                result = this.project(value, index);
            }
            catch (error) {
                this.destination.error(error);
                return;
            }
            this._innerSub(result, value, index);
        }
        _innerSub(result, value, index) {
            const innerSubscription = this.innerSubscription;
            if (innerSubscription) {
                innerSubscription.unsubscribe();
            }
            const innerSubscriber = new InnerSubscriber(this, value, index);
            const destination = this.destination;
            destination.add(innerSubscriber);
            this.innerSubscription = subscribeToResult(this, result, undefined, undefined, innerSubscriber);
            if (this.innerSubscription !== innerSubscriber) {
                destination.add(this.innerSubscription);
            }
        }
        _complete() {
            const { innerSubscription } = this;
            if (!innerSubscription || innerSubscription.closed) {
                super._complete();
            }
            this.unsubscribe();
        }
        _unsubscribe() {
            this.innerSubscription = null;
        }
        notifyComplete(innerSub) {
            const destination = this.destination;
            destination.remove(innerSub);
            this.innerSubscription = null;
            if (this.isStopped) {
                super._complete();
            }
        }
        notifyNext(outerValue, innerValue, outerIndex, innerIndex, innerSub) {
            this.destination.next(innerValue);
        }
    }

    function takeUntil(notifier) {
        return (source) => source.lift(new TakeUntilOperator(notifier));
    }
    class TakeUntilOperator {
        constructor(notifier) {
            this.notifier = notifier;
        }
        call(subscriber, source) {
            const takeUntilSubscriber = new TakeUntilSubscriber(subscriber);
            const notifierSubscription = subscribeToResult(takeUntilSubscriber, this.notifier);
            if (notifierSubscription && !takeUntilSubscriber.seenValue) {
                takeUntilSubscriber.add(notifierSubscription);
                return source.subscribe(takeUntilSubscriber);
            }
            return takeUntilSubscriber;
        }
    }
    class TakeUntilSubscriber extends OuterSubscriber {
        constructor(destination) {
            super(destination);
            this.seenValue = false;
        }
        notifyNext(outerValue, innerValue, outerIndex, innerIndex, innerSub) {
            this.seenValue = true;
            this.complete();
        }
        notifyComplete() {
        }
    }

    function tap(nextOrObserver, error, complete) {
        return function tapOperatorFunction(source) {
            return source.lift(new DoOperator(nextOrObserver, error, complete));
        };
    }
    class DoOperator {
        constructor(nextOrObserver, error, complete) {
            this.nextOrObserver = nextOrObserver;
            this.error = error;
            this.complete = complete;
        }
        call(subscriber, source) {
            return source.subscribe(new TapSubscriber(subscriber, this.nextOrObserver, this.error, this.complete));
        }
    }
    class TapSubscriber extends Subscriber {
        constructor(destination, observerOrNext, error, complete) {
            super(destination);
            this._tapNext = noop;
            this._tapError = noop;
            this._tapComplete = noop;
            this._tapError = error || noop;
            this._tapComplete = complete || noop;
            if (isFunction(observerOrNext)) {
                this._context = this;
                this._tapNext = observerOrNext;
            }
            else if (observerOrNext) {
                this._context = observerOrNext;
                this._tapNext = observerOrNext.next || noop;
                this._tapError = observerOrNext.error || noop;
                this._tapComplete = observerOrNext.complete || noop;
            }
        }
        _next(value) {
            try {
                this._tapNext.call(this._context, value);
            }
            catch (err) {
                this.destination.error(err);
                return;
            }
            this.destination.next(value);
        }
        _error(err) {
            try {
                this._tapError.call(this._context, err);
            }
            catch (err) {
                this.destination.error(err);
                return;
            }
            this.destination.error(err);
        }
        _complete() {
            try {
                this._tapComplete.call(this._context);
            }
            catch (err) {
                this.destination.error(err);
                return;
            }
            return this.destination.complete();
        }
    }

    const defaultThrottleConfig = {
        leading: true,
        trailing: false
    };

    function throttleTime(duration, scheduler = async, config = defaultThrottleConfig) {
        return (source) => source.lift(new ThrottleTimeOperator(duration, scheduler, !!config.leading, !!config.trailing));
    }
    class ThrottleTimeOperator {
        constructor(duration, scheduler, leading, trailing) {
            this.duration = duration;
            this.scheduler = scheduler;
            this.leading = leading;
            this.trailing = trailing;
        }
        call(subscriber, source) {
            return source.subscribe(new ThrottleTimeSubscriber(subscriber, this.duration, this.scheduler, this.leading, this.trailing));
        }
    }
    class ThrottleTimeSubscriber extends Subscriber {
        constructor(destination, duration, scheduler, leading, trailing) {
            super(destination);
            this.duration = duration;
            this.scheduler = scheduler;
            this.leading = leading;
            this.trailing = trailing;
            this.throttled = null;
            this._hasTrailingValue = false;
            this._trailingValue = null;
        }
        _next(value) {
            if (this.throttled) {
                if (this.trailing) {
                    this._trailingValue = value;
                    this._hasTrailingValue = true;
                }
            }
            else {
                this.add(this.throttled = this.scheduler.schedule(dispatchNext$1, this.duration, { subscriber: this }));
                if (this.leading) {
                    this.destination.next(value);
                }
                else if (this.trailing) {
                    this._trailingValue = value;
                    this._hasTrailingValue = true;
                }
            }
        }
        _complete() {
            if (this._hasTrailingValue) {
                this.destination.next(this._trailingValue);
                this.destination.complete();
            }
            else {
                this.destination.complete();
            }
        }
        clearThrottle() {
            const throttled = this.throttled;
            if (throttled) {
                if (this.trailing && this._hasTrailingValue) {
                    this.destination.next(this._trailingValue);
                    this._trailingValue = null;
                    this._hasTrailingValue = false;
                }
                throttled.unsubscribe();
                this.remove(throttled);
                this.throttled = null;
            }
        }
    }
    function dispatchNext$1(arg) {
        const { subscriber } = arg;
        subscriber.clearThrottle();
    }

    function withLatestFrom(...args) {
        return (source) => {
            let project;
            if (typeof args[args.length - 1] === 'function') {
                project = args.pop();
            }
            const observables = args;
            return source.lift(new WithLatestFromOperator(observables, project));
        };
    }
    class WithLatestFromOperator {
        constructor(observables, project) {
            this.observables = observables;
            this.project = project;
        }
        call(subscriber, source) {
            return source.subscribe(new WithLatestFromSubscriber(subscriber, this.observables, this.project));
        }
    }
    class WithLatestFromSubscriber extends OuterSubscriber {
        constructor(destination, observables, project) {
            super(destination);
            this.observables = observables;
            this.project = project;
            this.toRespond = [];
            const len = observables.length;
            this.values = new Array(len);
            for (let i = 0; i < len; i++) {
                this.toRespond.push(i);
            }
            for (let i = 0; i < len; i++) {
                let observable = observables[i];
                this.add(subscribeToResult(this, observable, observable, i));
            }
        }
        notifyNext(outerValue, innerValue, outerIndex, innerIndex, innerSub) {
            this.values[outerIndex] = innerValue;
            const toRespond = this.toRespond;
            if (toRespond.length > 0) {
                const found = toRespond.indexOf(outerIndex);
                if (found !== -1) {
                    toRespond.splice(found, 1);
                }
            }
        }
        notifyComplete() {
        }
        _next(value) {
            if (this.toRespond.length === 0) {
                const args = [value, ...this.values];
                if (this.project) {
                    this._tryProject(args);
                }
                else {
                    this.destination.next(args);
                }
            }
        }
        _tryProject(args) {
            let result;
            try {
                result = this.project.apply(this, args);
            }
            catch (err) {
                this.destination.error(err);
                return;
            }
            this.destination.next(result);
        }
    }

    /**
     * @license Angular v9.1.12
     * (c) 2010-2020 Google LLC. https://angular.io/
     * License: MIT
     */

    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * Convince closure compiler that the wrapped function has no side-effects.
     *
     * Closure compiler always assumes that `toString` has no side-effects. We use this quirk to
     * allow us to execute a function but have closure compiler mark the call as no-side-effects.
     * It is important that the return value for the `noSideEffects` function be assigned
     * to something which is retained otherwise the call to `noSideEffects` will be removed by closure
     * compiler.
     */
    function noSideEffects(fn) {
        return { toString: fn }.toString();
    }

    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    const ANNOTATIONS = '__annotations__';
    const PARAMETERS = '__parameters__';
    const PROP_METADATA = '__prop__metadata__';
    /**
     * @suppress {globalThis}
     */
    function makeDecorator(name, props, parentClass, additionalProcessing, typeFn) {
        return noSideEffects(() => {
            const metaCtor = makeMetadataCtor(props);
            function DecoratorFactory(...args) {
                if (this instanceof DecoratorFactory) {
                    metaCtor.call(this, ...args);
                    return this;
                }
                const annotationInstance = new DecoratorFactory(...args);
                return function TypeDecorator(cls) {
                    if (typeFn)
                        typeFn(cls, ...args);
                    // Use of Object.defineProperty is important since it creates non-enumerable property which
                    // prevents the property is copied during subclassing.
                    const annotations = cls.hasOwnProperty(ANNOTATIONS) ?
                        cls[ANNOTATIONS] :
                        Object.defineProperty(cls, ANNOTATIONS, { value: [] })[ANNOTATIONS];
                    annotations.push(annotationInstance);
                    if (additionalProcessing)
                        additionalProcessing(cls);
                    return cls;
                };
            }
            if (parentClass) {
                DecoratorFactory.prototype = Object.create(parentClass.prototype);
            }
            DecoratorFactory.prototype.ngMetadataName = name;
            DecoratorFactory.annotationCls = DecoratorFactory;
            return DecoratorFactory;
        });
    }
    function makeMetadataCtor(props) {
        return function ctor(...args) {
            if (props) {
                const values = props(...args);
                for (const propName in values) {
                    this[propName] = values[propName];
                }
            }
        };
    }
    function makeParamDecorator(name, props, parentClass) {
        return noSideEffects(() => {
            const metaCtor = makeMetadataCtor(props);
            function ParamDecoratorFactory(...args) {
                if (this instanceof ParamDecoratorFactory) {
                    metaCtor.apply(this, args);
                    return this;
                }
                const annotationInstance = new ParamDecoratorFactory(...args);
                ParamDecorator.annotation = annotationInstance;
                return ParamDecorator;
                function ParamDecorator(cls, unusedKey, index) {
                    // Use of Object.defineProperty is important since it creates non-enumerable property which
                    // prevents the property is copied during subclassing.
                    const parameters = cls.hasOwnProperty(PARAMETERS) ?
                        cls[PARAMETERS] :
                        Object.defineProperty(cls, PARAMETERS, { value: [] })[PARAMETERS];
                    // there might be gaps if some in between parameters do not have annotations.
                    // we pad with nulls.
                    while (parameters.length <= index) {
                        parameters.push(null);
                    }
                    (parameters[index] = parameters[index] || []).push(annotationInstance);
                    return cls;
                }
            }
            if (parentClass) {
                ParamDecoratorFactory.prototype = Object.create(parentClass.prototype);
            }
            ParamDecoratorFactory.prototype.ngMetadataName = name;
            ParamDecoratorFactory.annotationCls = ParamDecoratorFactory;
            return ParamDecoratorFactory;
        });
    }
    function makePropDecorator(name, props, parentClass, additionalProcessing) {
        return noSideEffects(() => {
            const metaCtor = makeMetadataCtor(props);
            function PropDecoratorFactory(...args) {
                if (this instanceof PropDecoratorFactory) {
                    metaCtor.apply(this, args);
                    return this;
                }
                const decoratorInstance = new PropDecoratorFactory(...args);
                function PropDecorator(target, name) {
                    const constructor = target.constructor;
                    // Use of Object.defineProperty is important since it creates non-enumerable property which
                    // prevents the property is copied during subclassing.
                    const meta = constructor.hasOwnProperty(PROP_METADATA) ?
                        constructor[PROP_METADATA] :
                        Object.defineProperty(constructor, PROP_METADATA, { value: {} })[PROP_METADATA];
                    meta[name] = meta.hasOwnProperty(name) && meta[name] || [];
                    meta[name].unshift(decoratorInstance);
                    if (additionalProcessing)
                        additionalProcessing(target, name, ...args);
                }
                return PropDecorator;
            }
            if (parentClass) {
                PropDecoratorFactory.prototype = Object.create(parentClass.prototype);
            }
            PropDecoratorFactory.prototype.ngMetadataName = name;
            PropDecoratorFactory.annotationCls = PropDecoratorFactory;
            return PropDecoratorFactory;
        });
    }
    // WARNING: interface has both a type and a value, skipping emit
    const ɵ0 = /**
     * @param {?} token
     * @return {?}
     */
    (token) => ({ token });
    /**
     * Inject decorator and metadata.
     *
     * \@Annotation
     * \@publicApi
     * @type {?}
     */
    const Inject = makeParamDecorator('Inject', (ɵ0));
    // WARNING: interface has both a type and a value, skipping emit
    /**
     * Optional decorator and metadata.
     *
     * \@Annotation
     * \@publicApi
     * @type {?}
     */
    const Optional = makeParamDecorator('Optional');
    // WARNING: interface has both a type and a value, skipping emit
    /**
     * Self decorator and metadata.
     *
     * \@Annotation
     * \@publicApi
     * @type {?}
     */
    const Self = makeParamDecorator('Self');
    // WARNING: interface has both a type and a value, skipping emit
    /**
     * SkipSelf decorator and metadata.
     *
     * \@Annotation
     * \@publicApi
     * @type {?}
     */
    const SkipSelf = makeParamDecorator('SkipSelf');
    // WARNING: interface has both a type and a value, skipping emit
    /**
     * Host decorator and metadata.
     *
     * \@Annotation
     * \@publicApi
     * @type {?}
     */
    const Host = makeParamDecorator('Host');
    // WARNING: interface has both a type and a value, skipping emit
    const ɵ1 = /**
     * @param {?=} attributeName
     * @return {?}
     */
    (attributeName) => ({ attributeName });
    /**
     * Attribute decorator and metadata.
     *
     * \@Annotation
     * \@publicApi
     * @type {?}
     */
    const Attribute = makeParamDecorator('Attribute', (ɵ1));

    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * Injection flags for DI.
     *
     * @publicApi
     */
    var InjectFlags;
    (function (InjectFlags) {
        // TODO(alxhub): make this 'const' when ngc no longer writes exports of it into ngfactory files.
        /** Check self and check parent injector if needed */
        InjectFlags[InjectFlags["Default"] = 0] = "Default";
        /**
         * Specifies that an injector should retrieve a dependency from any injector until reaching the
         * host element of the current component. (Only used with Element Injector)
         */
        InjectFlags[InjectFlags["Host"] = 1] = "Host";
        /** Don't ascend to ancestors of the node requesting injection. */
        InjectFlags[InjectFlags["Self"] = 2] = "Self";
        /** Skip the node that is requesting injection. */
        InjectFlags[InjectFlags["SkipSelf"] = 4] = "SkipSelf";
        /** Inject `defaultValue` instead if token not found. */
        InjectFlags[InjectFlags["Optional"] = 8] = "Optional";
    })(InjectFlags || (InjectFlags = {}));

    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    function getClosureSafeProperty(objWithPropertyToExtract) {
        for (let key in objWithPropertyToExtract) {
            if (objWithPropertyToExtract[key] === getClosureSafeProperty) {
                return key;
            }
        }
        throw Error('Could not find renamed property on target object.');
    }
    /**
     * Sets properties on a target object from a source object, but only if
     * the property doesn't already exist on the target object.
     * @param target The target to set properties on
     * @param source The source of the property keys and values to set
     */
    function fillProperties(target, source) {
        for (const key in source) {
            if (source.hasOwnProperty(key) && !target.hasOwnProperty(key)) {
                target[key] = source[key];
            }
        }
    }

    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * Construct an `InjectableDef` which defines how a token will be constructed by the DI system, and
     * in which injectors (if any) it will be available.
     *
     * This should be assigned to a static `ɵprov` field on a type, which will then be an
     * `InjectableType`.
     *
     * Options:
     * * `providedIn` determines which injectors will include the injectable, by either associating it
     *   with an `@NgModule` or other `InjectorType`, or by specifying that this injectable should be
     *   provided in the `'root'` injector, which will be the application-level injector in most apps.
     * * `factory` gives the zero argument function which will create an instance of the injectable.
     *   The factory can call `inject` to access the `Injector` and request injection of dependencies.
     *
     * @codeGenApi
     */
    function ɵɵdefineInjectable(opts) {
        return {
            token: opts.token,
            providedIn: opts.providedIn || null,
            factory: opts.factory,
            value: undefined,
        };
    }
    /**
     * Construct an `InjectorDef` which configures an injector.
     *
     * This should be assigned to a static injector def (`ɵinj`) field on a type, which will then be an
     * `InjectorType`.
     *
     * Options:
     *
     * * `factory`: an `InjectorType` is an instantiable type, so a zero argument `factory` function to
     *   create the type must be provided. If that factory function needs to inject arguments, it can
     *   use the `inject` function.
     * * `providers`: an optional array of providers to add to the injector. Each provider must
     *   either have a factory or point to a type which has a `ɵprov` static property (the
     *   type must be an `InjectableType`).
     * * `imports`: an optional array of imports of other `InjectorType`s or `InjectorTypeWithModule`s
     *   whose providers will also be added to the injector. Locally provided types will override
     *   providers from imports.
     *
     * @publicApi
     */
    function ɵɵdefineInjector(options) {
        return {
            factory: options.factory,
            providers: options.providers || [],
            imports: options.imports || [],
        };
    }
    /**
     * Read the injectable def (`ɵprov`) for `type` in a way which is immune to accidentally reading
     * inherited value.
     *
     * @param type A type which may have its own (non-inherited) `ɵprov`.
     */
    function getInjectableDef(type) {
        return getOwnDefinition(type, type[NG_PROV_DEF]) ||
            getOwnDefinition(type, type[NG_INJECTABLE_DEF]);
    }
    /**
     * Return `def` only if it is defined directly on `type` and is not inherited from a base
     * class of `type`.
     *
     * The function `Object.hasOwnProperty` is not sufficient to distinguish this case because in older
     * browsers (e.g. IE10) static property inheritance is implemented by copying the properties.
     *
     * Instead, the definition's `token` is compared to the `type`, and if they don't match then the
     * property was not defined directly on the type itself, and was likely inherited. The definition
     * is only returned if the `type` matches the `def.token`.
     */
    function getOwnDefinition(type, def) {
        return def && def.token === type ? def : null;
    }
    /**
     * Read the injectable def (`ɵprov`) for `type` or read the `ɵprov` from one of its ancestors.
     *
     * @param type A type which may have `ɵprov`, via inheritance.
     *
     * @deprecated Will be removed in v10, where an error will occur in the scenario if we find the
     * `ɵprov` on an ancestor only.
     */
    function getInheritedInjectableDef(type) {
        // See `jit/injectable.ts#compileInjectable` for context on NG_PROV_DEF_FALLBACK.
        const def = type &&
            (type[NG_PROV_DEF] || type[NG_INJECTABLE_DEF] ||
                (type[NG_PROV_DEF_FALLBACK] && type[NG_PROV_DEF_FALLBACK]()));
        if (def) {
            const typeName = getTypeName(type);
            // TODO(FW-1307): Re-add ngDevMode when closure can handle it
            // ngDevMode &&
            console.warn(`DEPRECATED: DI is instantiating a token "${typeName}" that inherits its @Injectable decorator but does not provide one itself.\n` +
                `This will become an error in v10. Please add @Injectable() to the "${typeName}" class.`);
            return def;
        }
        else {
            return null;
        }
    }
    /** Gets the name of a type, accounting for some cross-browser differences. */
    function getTypeName(type) {
        // `Function.prototype.name` behaves differently between IE and other browsers. In most browsers
        // it'll always return the name of the function itself, no matter how many other functions it
        // inherits from. On IE the function doesn't have its own `name` property, but it takes it from
        // the lowest level in the prototype chain. E.g. if we have `class Foo extends Parent` most
        // browsers will evaluate `Foo.name` to `Foo` while IE will return `Parent`. We work around
        // the issue by converting the function to a string and parsing its name out that way via a regex.
        if (type.hasOwnProperty('name')) {
            return type.name;
        }
        const match = ('' + type).match(/^function\s*([^\s(]+)/);
        return match === null ? '' : match[1];
    }
    /**
     * Read the injector def type in a way which is immune to accidentally reading inherited value.
     *
     * @param type type which may have an injector def (`ɵinj`)
     */
    function getInjectorDef(type) {
        return type && (type.hasOwnProperty(NG_INJ_DEF) || type.hasOwnProperty(NG_INJECTOR_DEF)) ?
            type[NG_INJ_DEF] :
            null;
    }
    const NG_PROV_DEF = getClosureSafeProperty({ ɵprov: getClosureSafeProperty });
    const NG_INJ_DEF = getClosureSafeProperty({ ɵinj: getClosureSafeProperty });
    // On IE10 properties defined via `defineProperty` won't be inherited by child classes,
    // which will break inheriting the injectable definition from a grandparent through an
    // undecorated parent class. We work around it by defining a fallback method which will be
    // used to retrieve the definition. This should only be a problem in JIT mode, because in
    // AOT TypeScript seems to have a workaround for static properties. When inheriting from an
    // undecorated parent is no longer supported in v10, this can safely be removed.
    const NG_PROV_DEF_FALLBACK = getClosureSafeProperty({ ɵprovFallback: getClosureSafeProperty });
    // We need to keep these around so we can read off old defs if new defs are unavailable
    const NG_INJECTABLE_DEF = getClosureSafeProperty({ ngInjectableDef: getClosureSafeProperty });
    const NG_INJECTOR_DEF = getClosureSafeProperty({ ngInjectorDef: getClosureSafeProperty });

    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    function stringify(token) {
        if (typeof token === 'string') {
            return token;
        }
        if (Array.isArray(token)) {
            return '[' + token.map(stringify).join(', ') + ']';
        }
        if (token == null) {
            return '' + token;
        }
        if (token.overriddenName) {
            return `${token.overriddenName}`;
        }
        if (token.name) {
            return `${token.name}`;
        }
        const res = token.toString();
        if (res == null) {
            return '' + res;
        }
        const newLineIndex = res.indexOf('\n');
        return newLineIndex === -1 ? res : res.substring(0, newLineIndex);
    }
    /**
     * Concatenates two strings with separator, allocating new strings only when necessary.
     *
     * @param before before string.
     * @param separator separator string.
     * @param after after string.
     * @returns concatenated string.
     */
    function concatStringsWithSpace(before, after) {
        return (before == null || before === '') ?
            (after === null ? '' : after) :
            ((after == null || after === '') ? before : before + ' ' + after);
    }
    /** @type {?} */
    const __forward_ref__ = getClosureSafeProperty({ __forward_ref__: getClosureSafeProperty });
    /**
     * Allows to refer to references which are not yet defined.
     *
     * For instance, `forwardRef` is used when the `token` which we need to refer to for the purposes of
     * DI is declared, but not yet defined. It is also used when the `token` which we use when creating
     * a query is not yet defined.
     *
     * \@usageNotes
     * ### Example
     * {\@example core/di/ts/forward_ref/forward_ref_spec.ts region='forward_ref'}
     * \@publicApi
     * @param {?} forwardRefFn
     * @return {?}
     */
    function forwardRef(forwardRefFn) {
        ((/** @type {?} */ (forwardRefFn))).__forward_ref__ = forwardRef;
        ((/** @type {?} */ (forwardRefFn))).toString = (/**
         * @return {?}
         */
        function () {
            return stringify(this());
        });
        return ((/** @type {?} */ ((/** @type {?} */ (forwardRefFn)))));
    }
    /**
     * Lazily retrieves the reference value from a forwardRef.
     *
     * Acts as the identity function when given a non-forward-ref value.
     *
     * \@usageNotes
     * ### Example
     *
     * {\@example core/di/ts/forward_ref/forward_ref_spec.ts region='resolve_forward_ref'}
     *
     * @see `forwardRef`
     * \@publicApi
     * @template T
     * @param {?} type
     * @return {?}
     */
    function resolveForwardRef(type) {
        return isForwardRef(type) ? type() : type;
    }
    /**
     * Checks whether a function is wrapped by a `forwardRef`.
     * @param {?} fn
     * @return {?}
     */
    function isForwardRef(fn) {
        return typeof fn === 'function' && fn.hasOwnProperty(__forward_ref__) &&
            fn.__forward_ref__ === forwardRef;
    }

    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    const __globalThis = typeof globalThis !== 'undefined' && globalThis;
    const __window = typeof window !== 'undefined' && window;
    const __self = typeof self !== 'undefined' && typeof WorkerGlobalScope !== 'undefined' &&
        self instanceof WorkerGlobalScope && self;
    const __global = typeof global !== 'undefined' && global;
    // Always use __globalThis if available, which is the spec-defined global variable across all
    // environments, then fallback to __global first, because in Node tests both __global and
    // __window may be defined and _global should be __global in that case.
    const _global = __globalThis || __global || __window || __self;

    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    var R3ResolvedDependencyType;
    (function (R3ResolvedDependencyType) {
        R3ResolvedDependencyType[R3ResolvedDependencyType["Token"] = 0] = "Token";
        R3ResolvedDependencyType[R3ResolvedDependencyType["Attribute"] = 1] = "Attribute";
        R3ResolvedDependencyType[R3ResolvedDependencyType["ChangeDetectorRef"] = 2] = "ChangeDetectorRef";
        R3ResolvedDependencyType[R3ResolvedDependencyType["Invalid"] = 3] = "Invalid";
    })(R3ResolvedDependencyType || (R3ResolvedDependencyType = {}));
    var R3FactoryTarget;
    (function (R3FactoryTarget) {
        R3FactoryTarget[R3FactoryTarget["Directive"] = 0] = "Directive";
        R3FactoryTarget[R3FactoryTarget["Component"] = 1] = "Component";
        R3FactoryTarget[R3FactoryTarget["Injectable"] = 2] = "Injectable";
        R3FactoryTarget[R3FactoryTarget["Pipe"] = 3] = "Pipe";
        R3FactoryTarget[R3FactoryTarget["NgModule"] = 4] = "NgModule";
    })(R3FactoryTarget || (R3FactoryTarget = {}));
    var ViewEncapsulation;
    (function (ViewEncapsulation) {
        ViewEncapsulation[ViewEncapsulation["Emulated"] = 0] = "Emulated";
        ViewEncapsulation[ViewEncapsulation["Native"] = 1] = "Native";
        ViewEncapsulation[ViewEncapsulation["None"] = 2] = "None";
        ViewEncapsulation[ViewEncapsulation["ShadowDom"] = 3] = "ShadowDom";
    })(ViewEncapsulation || (ViewEncapsulation = {}));

    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    function getCompilerFacade() {
        const globalNg = _global['ng'];
        if (!globalNg || !globalNg.ɵcompilerFacade) {
            throw new Error(`Angular JIT compilation failed: '@angular/compiler' not loaded!\n` +
                `  - JIT compilation is discouraged for production use-cases! Consider AOT mode instead.\n` +
                `  - Did you bootstrap using '@angular/platform-browser-dynamic' or '@angular/platform-server'?\n` +
                `  - Alternatively provide the compiler with 'import "@angular/compiler";' before bootstrapping.`);
        }
        return globalNg.ɵcompilerFacade;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/fields.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const NG_COMP_DEF = getClosureSafeProperty({ ɵcmp: getClosureSafeProperty });
    /** @type {?} */
    const NG_DIR_DEF = getClosureSafeProperty({ ɵdir: getClosureSafeProperty });
    /** @type {?} */
    const NG_PIPE_DEF = getClosureSafeProperty({ ɵpipe: getClosureSafeProperty });
    /** @type {?} */
    const NG_MOD_DEF = getClosureSafeProperty({ ɵmod: getClosureSafeProperty });
    /** @type {?} */
    const NG_LOC_ID_DEF = getClosureSafeProperty({ ɵloc: getClosureSafeProperty });
    /** @type {?} */
    const NG_FACTORY_DEF = getClosureSafeProperty({ ɵfac: getClosureSafeProperty });
    // TODO(misko): This is wrong. The NG_ELEMENT_ID should never be minified.
    /**
     * If a directive is diPublic, bloomAdd sets a property on the type with this constant as
     * the key and the directive's unique ID as the value. This allows us to map directives to their
     * bloom filter bit for DI.
     * @type {?}
     */
    const NG_ELEMENT_ID = getClosureSafeProperty({ __NG_ELEMENT_ID__: getClosureSafeProperty });

    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    function ngDevModeResetPerfCounters() {
        const locationString = typeof location !== 'undefined' ? location.toString() : '';
        const newCounters = {
            namedConstructors: locationString.indexOf('ngDevMode=namedConstructors') != -1,
            firstCreatePass: 0,
            tNode: 0,
            tView: 0,
            rendererCreateTextNode: 0,
            rendererSetText: 0,
            rendererCreateElement: 0,
            rendererAddEventListener: 0,
            rendererSetAttribute: 0,
            rendererRemoveAttribute: 0,
            rendererSetProperty: 0,
            rendererSetClassName: 0,
            rendererAddClass: 0,
            rendererRemoveClass: 0,
            rendererSetStyle: 0,
            rendererRemoveStyle: 0,
            rendererDestroy: 0,
            rendererDestroyNode: 0,
            rendererMoveNode: 0,
            rendererRemoveNode: 0,
            rendererAppendChild: 0,
            rendererInsertBefore: 0,
            rendererCreateComment: 0,
        };
        // Make sure to refer to ngDevMode as ['ngDevMode'] for closure.
        const allowNgDevModeTrue = locationString.indexOf('ngDevMode=false') === -1;
        _global['ngDevMode'] = allowNgDevModeTrue && newCounters;
        return newCounters;
    }
    /**
     * This function checks to see if the `ngDevMode` has been set. If yes,
     * then we honor it, otherwise we default to dev mode with additional checks.
     *
     * The idea is that unless we are doing production build where we explicitly
     * set `ngDevMode == false` we should be helping the developer by providing
     * as much early warning and errors as possible.
     *
     * `ɵɵdefineComponent` is guaranteed to have been called before any component template functions
     * (and thus Ivy instructions), so a single initialization there is sufficient to ensure ngDevMode
     * is defined for the entire instruction set.
     *
     * When using checking `ngDevMode` on toplevel, always init it before referencing it
     * (e.g. `((typeof ngDevMode === 'undefined' || ngDevMode) && initNgDevMode())`), otherwise you can
     *  get a `ReferenceError` like in https://github.com/angular/angular/issues/31595.
     *
     * Details on possible values for `ngDevMode` can be found on its docstring.
     *
     * NOTE:
     * - changes to the `ngDevMode` name must be synced with `compiler-cli/src/tooling.ts`.
     */
    function initNgDevMode() {
        // The below checks are to ensure that calling `initNgDevMode` multiple times does not
        // reset the counters.
        // If the `ngDevMode` is not an object, then it means we have not created the perf counters
        // yet.
        if (typeof ngDevMode === 'undefined' || ngDevMode) {
            if (typeof ngDevMode !== 'object') {
                ngDevModeResetPerfCounters();
            }
            return !!ngDevMode;
        }
        return false;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/di/injection_token.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Creates a token that can be used in a DI Provider.
     *
     * Use an `InjectionToken` whenever the type you are injecting is not reified (does not have a
     * runtime representation) such as when injecting an interface, callable type, array or
     * parameterized type.
     *
     * `InjectionToken` is parameterized on `T` which is the type of object which will be returned by
     * the `Injector`. This provides additional level of type safety.
     *
     * ```
     * interface MyInterface {...}
     * var myInterface = injector.get(new InjectionToken<MyInterface>('SomeToken'));
     * // myInterface is inferred to be MyInterface.
     * ```
     *
     * When creating an `InjectionToken`, you can optionally specify a factory function which returns
     * (possibly by creating) a default value of the parameterized type `T`. This sets up the
     * `InjectionToken` using this factory as a provider as if it was defined explicitly in the
     * application's root injector. If the factory function, which takes zero arguments, needs to inject
     * dependencies, it can do so using the `inject` function. See below for an example.
     *
     * Additionally, if a `factory` is specified you can also specify the `providedIn` option, which
     * overrides the above behavior and marks the token as belonging to a particular `\@NgModule`. As
     * mentioned above, `'root'` is the default value for `providedIn`.
     *
     * \@usageNotes
     * ### Basic Example
     *
     * ### Plain InjectionToken
     *
     * {\@example core/di/ts/injector_spec.ts region='InjectionToken'}
     *
     * ### Tree-shakable InjectionToken
     *
     * {\@example core/di/ts/injector_spec.ts region='ShakableInjectionToken'}
     *
     *
     * \@publicApi
     * @template T
     */
    class InjectionToken {
        /**
         * @param {?} _desc
         * @param {?=} options
         */
        constructor(_desc, options) {
            this._desc = _desc;
            /**
             * \@internal
             */
            this.ngMetadataName = 'InjectionToken';
            this.ɵprov = undefined;
            if (typeof options == 'number') {
                // This is a special hack to assign __NG_ELEMENT_ID__ to this instance.
                // __NG_ELEMENT_ID__ is Used by Ivy to determine bloom filter id.
                // We are using it to assign `-1` which is used to identify `Injector`.
                ((/** @type {?} */ (this))).__NG_ELEMENT_ID__ = options;
            }
            else if (options !== undefined) {
                this.ɵprov = ɵɵdefineInjectable({
                    token: this,
                    providedIn: options.providedIn || 'root',
                    factory: options.factory,
                });
            }
        }
        /**
         * @return {?}
         */
        toString() {
            return `InjectionToken ${this._desc}`;
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/di/injector_compatibility.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * An InjectionToken that gets the current `Injector` for `createInjector()`-style injectors.
     *
     * Requesting this token instead of `Injector` allows `StaticInjector` to be tree-shaken from a
     * project.
     *
     * \@publicApi
     * @type {?}
     */
    const INJECTOR = new InjectionToken('INJECTOR', (/** @type {?} */ (-1)));
    /** @type {?} */
    const _THROW_IF_NOT_FOUND = {};
    /** @type {?} */
    const THROW_IF_NOT_FOUND = _THROW_IF_NOT_FOUND;
    /** @type {?} */
    const NG_TEMP_TOKEN_PATH = 'ngTempTokenPath';
    /** @type {?} */
    const NG_TOKEN_PATH = 'ngTokenPath';
    /** @type {?} */
    const NEW_LINE = /\n/gm;
    /** @type {?} */
    const NO_NEW_LINE = 'ɵ';
    /** @type {?} */
    const SOURCE = '__source';
    const ɵ0$1 = getClosureSafeProperty;
    /** @type {?} */
    const USE_VALUE = getClosureSafeProperty({ provide: String, useValue: ɵ0$1 });
    /**
     * Current injector value used by `inject`.
     * - `undefined`: it is an error to call `inject`
     * - `null`: `inject` can be called but there is no injector (limp-mode).
     * - Injector instance: Use the injector for resolution.
     * @type {?}
     */
    let _currentInjector = undefined;
    /**
     * @param {?} injector
     * @return {?}
     */
    function setCurrentInjector(injector) {
        /** @type {?} */
        const former = _currentInjector;
        _currentInjector = injector;
        return former;
    }
    /**
     * Current implementation of inject.
     *
     * By default, it is `injectInjectorOnly`, which makes it `Injector`-only aware. It can be changed
     * to `directiveInject`, which brings in the `NodeInjector` system of ivy. It is designed this
     * way for two reasons:
     *  1. `Injector` should not depend on ivy logic.
     *  2. To maintain tree shake-ability we don't want to bring in unnecessary code.
     * @type {?}
     */
    let _injectImplementation;
    /**
     * Sets the current inject implementation.
     * @param {?} impl
     * @return {?}
     */
    function setInjectImplementation(impl) {
        /** @type {?} */
        const previous = _injectImplementation;
        _injectImplementation = impl;
        return previous;
    }
    /**
     * @template T
     * @param {?} token
     * @param {?=} flags
     * @return {?}
     */
    function injectInjectorOnly(token, flags = InjectFlags.Default) {
        if (_currentInjector === undefined) {
            throw new Error(`inject() must be called from an injection context`);
        }
        else if (_currentInjector === null) {
            return injectRootLimpMode(token, undefined, flags);
        }
        else {
            return _currentInjector.get(token, flags & InjectFlags.Optional ? null : undefined, flags);
        }
    }
    /**
     * @template T
     * @param {?} token
     * @param {?=} flags
     * @return {?}
     */
    function ɵɵinject(token, flags = InjectFlags.Default) {
        return (_injectImplementation || injectInjectorOnly)(resolveForwardRef(token), flags);
    }
    /**
     * Throws an error indicating that a factory function could not be generated by the compiler for a
     * particular class.
     *
     * This instruction allows the actual error message to be optimized away when ngDevMode is turned
     * off, saving bytes of generated code while still providing a good experience in dev mode.
     *
     * The name of the class is not mentioned here, but will be in the generated factory function name
     * and thus in the stack trace.
     *
     * \@codeGenApi
     * @param {?} index
     * @return {?}
     */
    function ɵɵinvalidFactoryDep(index) {
        /** @type {?} */
        const msg = ngDevMode ?
            `This constructor is not compatible with Angular Dependency Injection because its dependency at index ${index} of the parameter list is invalid.
This can happen if the dependency type is a primitive like a string or if an ancestor of this class is missing an Angular decorator.

Please check that 1) the type for the parameter at index ${index} is correct and 2) the correct Angular decorators are defined for this class and its ancestors.` :
            'invalid';
        throw new Error(msg);
    }
    /**
     * Injects a token from the currently active injector.
     *
     * Must be used in the context of a factory function such as one defined for an
     * `InjectionToken`. Throws an error if not called from such a context.
     *
     * Within such a factory function, using this function to request injection of a dependency
     * is faster and more type-safe than providing an additional array of dependencies
     * (as has been common with `useFactory` providers).
     *
     * \@param token The injection token for the dependency to be injected.
     * \@param flags Optional flags that control how injection is executed.
     * The flags correspond to injection strategies that can be specified with
     * parameter decorators `\@Host`, `\@Self`, `\@SkipSef`, and `\@Optional`.
     * \@return True if injection is successful, null otherwise.
     *
     * \@usageNotes
     *
     * ### Example
     *
     * {\@example core/di/ts/injector_spec.ts region='ShakableInjectionToken'}
     *
     * \@publicApi
     * @type {?}
     */
    const inject = ɵɵinject;
    /**
     * Injects `root` tokens in limp mode.
     *
     * If no injector exists, we can still inject tree-shakable providers which have `providedIn` set to
     * `"root"`. This is known as the limp mode injection. In such case the value is stored in the
     * `InjectableDef`.
     * @template T
     * @param {?} token
     * @param {?} notFoundValue
     * @param {?} flags
     * @return {?}
     */
    function injectRootLimpMode(token, notFoundValue, flags) {
        /** @type {?} */
        const injectableDef = getInjectableDef(token);
        if (injectableDef && injectableDef.providedIn == 'root') {
            return injectableDef.value === undefined ? injectableDef.value = injectableDef.factory() :
                injectableDef.value;
        }
        if (flags & InjectFlags.Optional)
            return null;
        if (notFoundValue !== undefined)
            return notFoundValue;
        throw new Error(`Injector: NOT_FOUND [${stringify(token)}]`);
    }
    /**
     * @param {?} types
     * @return {?}
     */
    function injectArgs(types) {
        /** @type {?} */
        const args = [];
        for (let i = 0; i < types.length; i++) {
            /** @type {?} */
            const arg = resolveForwardRef(types[i]);
            if (Array.isArray(arg)) {
                if (arg.length === 0) {
                    throw new Error('Arguments array must have arguments.');
                }
                /** @type {?} */
                let type = undefined;
                /** @type {?} */
                let flags = InjectFlags.Default;
                for (let j = 0; j < arg.length; j++) {
                    /** @type {?} */
                    const meta = arg[j];
                    if (meta instanceof Optional || meta.ngMetadataName === 'Optional' || meta === Optional) {
                        flags |= InjectFlags.Optional;
                    }
                    else if (meta instanceof SkipSelf || meta.ngMetadataName === 'SkipSelf' || meta === SkipSelf) {
                        flags |= InjectFlags.SkipSelf;
                    }
                    else if (meta instanceof Self || meta.ngMetadataName === 'Self' || meta === Self) {
                        flags |= InjectFlags.Self;
                    }
                    else if (meta instanceof Inject || meta === Inject) {
                        type = meta.token;
                    }
                    else {
                        type = meta;
                    }
                }
                args.push(ɵɵinject((/** @type {?} */ (type)), flags));
            }
            else {
                args.push(ɵɵinject(arg));
            }
        }
        return args;
    }
    class NullInjector {
        /**
         * @param {?} token
         * @param {?=} notFoundValue
         * @return {?}
         */
        get(token, notFoundValue = THROW_IF_NOT_FOUND) {
            if (notFoundValue === THROW_IF_NOT_FOUND) {
                // Intentionally left behind: With dev tools open the debugger will stop here. There is no
                // reason why correctly written application should cause this exception.
                // TODO(misko): uncomment the next line once `ngDevMode` works with closure.
                // if (ngDevMode) debugger;
                /** @type {?} */
                const error = new Error(`NullInjectorError: No provider for ${stringify(token)}!`);
                error.name = 'NullInjectorError';
                throw error;
            }
            return notFoundValue;
        }
    }
    /**
     * @param {?} e
     * @param {?} token
     * @param {?} injectorErrorName
     * @param {?} source
     * @return {?}
     */
    function catchInjectorError(e, token, injectorErrorName, source) {
        /** @type {?} */
        const tokenPath = e[NG_TEMP_TOKEN_PATH];
        if (token[SOURCE]) {
            tokenPath.unshift(token[SOURCE]);
        }
        e.message = formatError('\n' + e.message, tokenPath, injectorErrorName, source);
        e[NG_TOKEN_PATH] = tokenPath;
        e[NG_TEMP_TOKEN_PATH] = null;
        throw e;
    }
    /**
     * @param {?} text
     * @param {?} obj
     * @param {?} injectorErrorName
     * @param {?=} source
     * @return {?}
     */
    function formatError(text, obj, injectorErrorName, source = null) {
        text = text && text.charAt(0) === '\n' && text.charAt(1) == NO_NEW_LINE ? text.substr(2) : text;
        /** @type {?} */
        let context = stringify(obj);
        if (Array.isArray(obj)) {
            context = obj.map(stringify).join(' -> ');
        }
        else if (typeof obj === 'object') {
            /** @type {?} */
            let parts = (/** @type {?} */ ([]));
            for (let key in obj) {
                if (obj.hasOwnProperty(key)) {
                    /** @type {?} */
                    let value = obj[key];
                    parts.push(key + ':' + (typeof value === 'string' ? JSON.stringify(value) : stringify(value)));
                }
            }
            context = `{${parts.join(', ')}}`;
        }
        return `${injectorErrorName}${source ? '(' + source + ')' : ''}[${context}]: ${text.replace(NEW_LINE, '\n  ')}`;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/di/jit/environment.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * A mapping of the \@angular/core API surface used in generated expressions to the actual symbols.
     *
     * This should be kept up to date with the public exports of \@angular/core.
     * @type {?}
     */
    const angularCoreDiEnv = {
        'ɵɵdefineInjectable': ɵɵdefineInjectable,
        'ɵɵdefineInjector': ɵɵdefineInjector,
        'ɵɵinject': ɵɵinject,
        'ɵɵgetFactoryOf': getFactoryOf,
        'ɵɵinvalidFactoryDep': ɵɵinvalidFactoryDep,
    };
    /**
     * @template T
     * @param {?} type
     * @return {?}
     */
    function getFactoryOf(type) {
        /** @type {?} */
        const typeAny = (/** @type {?} */ (type));
        if (isForwardRef(type)) {
            return (/** @type {?} */ (((/**
             * @return {?}
             */
            () => {
                /** @type {?} */
                const factory = getFactoryOf(resolveForwardRef(typeAny));
                return factory ? factory() : null;
            }))));
        }
        /** @type {?} */
        const def = getInjectableDef(typeAny) || getInjectorDef(typeAny);
        if (!def || def.factory === undefined) {
            return null;
        }
        return def.factory;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/linker/ng_module_factory.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * Represents an instance of an NgModule created via a {\@link NgModuleFactory}.
     *
     * `NgModuleRef` provides access to the NgModule Instance as well other objects related to this
     * NgModule Instance.
     *
     * \@publicApi
     * @abstract
     * @template T
     */
    class NgModuleRef {
    }
    /**
     * \@publicApi
     * @abstract
     * @template T
     */
    class NgModuleFactory {
    }

    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    function assertNumber(actual, msg) {
        if (!(typeof actual === 'number')) {
            throwError$1(msg, typeof actual, 'number', '===');
        }
    }
    function assertNumberInRange(actual, minInclusive, maxInclusive) {
        assertNumber(actual, 'Expected a number');
        assertLessThanOrEqual(actual, maxInclusive, 'Expected number to be less than or equal to');
        assertGreaterThanOrEqual(actual, minInclusive, 'Expected number to be greater than or equal to');
    }
    function assertString(actual, msg) {
        if (!(typeof actual === 'string')) {
            throwError$1(msg, actual === null ? 'null' : typeof actual, 'string', '===');
        }
    }
    function assertEqual(actual, expected, msg) {
        if (!(actual == expected)) {
            throwError$1(msg, actual, expected, '==');
        }
    }
    function assertNotEqual(actual, expected, msg) {
        if (!(actual != expected)) {
            throwError$1(msg, actual, expected, '!=');
        }
    }
    function assertSame(actual, expected, msg) {
        if (!(actual === expected)) {
            throwError$1(msg, actual, expected, '===');
        }
    }
    function assertNotSame(actual, expected, msg) {
        if (!(actual !== expected)) {
            throwError$1(msg, actual, expected, '!==');
        }
    }
    function assertLessThan(actual, expected, msg) {
        if (!(actual < expected)) {
            throwError$1(msg, actual, expected, '<');
        }
    }
    function assertLessThanOrEqual(actual, expected, msg) {
        if (!(actual <= expected)) {
            throwError$1(msg, actual, expected, '<=');
        }
    }
    function assertGreaterThan(actual, expected, msg) {
        if (!(actual > expected)) {
            throwError$1(msg, actual, expected, '>');
        }
    }
    function assertGreaterThanOrEqual(actual, expected, msg) {
        if (!(actual >= expected)) {
            throwError$1(msg, actual, expected, '>=');
        }
    }
    function assertDefined(actual, msg) {
        if (actual == null) {
            throwError$1(msg, actual, null, '!=');
        }
    }
    function throwError$1(msg, actual, expected, comparison) {
        throw new Error(`ASSERTION ERROR: ${msg}` +
            (comparison == null ? '' : ` [Expected=> ${expected} ${comparison} ${actual} <=Actual]`));
    }
    function assertDomNode(node) {
        // If we're in a worker, `Node` will not be defined.
        assertEqual((typeof Node !== 'undefined' && node instanceof Node) ||
            (typeof node === 'object' && node != null &&
                node.constructor.name === 'WebWorkerRenderNode'), true, `The provided value must be an instance of a DOM Node but got ${stringify(node)}`);
    }
    function assertDataInRange(arr, index) {
        const maxLen = arr ? arr.length : 0;
        assertLessThan(index, maxLen, `Index expected to be less than ${maxLen} but got ${index}`);
    }

    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * Equivalent to ES6 spread, add each item to an array.
     *
     * @param items The items to add
     * @param arr The array to which you want to add the items
     */
    function addAllToArray(items, arr) {
        for (let i = 0; i < items.length; i++) {
            arr.push(items[i]);
        }
    }
    /**
     * Flattens an array.
     */
    function flatten(list, dst) {
        if (dst === undefined)
            dst = list;
        for (let i = 0; i < list.length; i++) {
            let item = list[i];
            if (Array.isArray(item)) {
                // we need to inline it.
                if (dst === list) {
                    // Our assumption that the list was already flat was wrong and
                    // we need to clone flat since we need to write to it.
                    dst = list.slice(0, i);
                }
                flatten(item, dst);
            }
            else if (dst !== list) {
                dst.push(item);
            }
        }
        return dst;
    }
    function deepForEach(input, fn) {
        input.forEach(value => Array.isArray(value) ? deepForEach(value, fn) : fn(value));
    }
    function addToArray(arr, index, value) {
        // perf: array.push is faster than array.splice!
        if (index >= arr.length) {
            arr.push(value);
        }
        else {
            arr.splice(index, 0, value);
        }
    }
    function removeFromArray(arr, index) {
        // perf: array.pop is faster than array.splice!
        if (index >= arr.length - 1) {
            return arr.pop();
        }
        else {
            return arr.splice(index, 1)[0];
        }
    }
    function newArray(size, value) {
        const list = [];
        for (let i = 0; i < size; i++) {
            list.push(value);
        }
        return list;
    }
    /**
     * Same as `Array.splice2(index, 0, value1, value2)` but faster.
     *
     * `Array.splice()` is not fast because it has to allocate an array for the elements which were
     * removed. This causes memory pressure and slows down code when most of the time we don't
     * care about the deleted items array.
     *
     * @param array Array to splice.
     * @param index Index in array where the `value` should be added.
     * @param value1 Value to add to array.
     * @param value2 Value to add to array.
     */
    function arrayInsert2(array, index, value1, value2) {
        ngDevMode && assertLessThanOrEqual(index, array.length, 'Can\'t insert past array end.');
        let end = array.length;
        if (end == index) {
            // inserting at the end.
            array.push(value1, value2);
        }
        else if (end === 1) {
            // corner case when we have less items in array than we have items to insert.
            array.push(value2, array[0]);
            array[0] = value1;
        }
        else {
            end--;
            array.push(array[end - 1], array[end]);
            while (end > index) {
                const previousEnd = end - 2;
                array[end] = array[previousEnd];
                end--;
            }
            array[index] = value1;
            array[index + 1] = value2;
        }
    }
    /**
     * Set a `value` for a `key`.
     *
     * @param keyValueArray to modify.
     * @param key The key to locate or create.
     * @param value The value to set for a `key`.
     * @returns index (always even) of where the value vas set.
     */
    function keyValueArraySet(keyValueArray, key, value) {
        let index = keyValueArrayIndexOf(keyValueArray, key);
        if (index >= 0) {
            // if we found it set it.
            keyValueArray[index | 1] = value;
        }
        else {
            index = ~index;
            arrayInsert2(keyValueArray, index, key, value);
        }
        return index;
    }
    /**
     * Retrieve a `value` for a `key` (on `undefined` if not found.)
     *
     * @param keyValueArray to search.
     * @param key The key to locate.
     * @return The `value` stored at the `key` location or `undefined if not found.
     */
    function keyValueArrayGet(keyValueArray, key) {
        const index = keyValueArrayIndexOf(keyValueArray, key);
        if (index >= 0) {
            // if we found it retrieve it.
            return keyValueArray[index | 1];
        }
        return undefined;
    }
    /**
     * Retrieve a `key` index value in the array or `-1` if not found.
     *
     * @param keyValueArray to search.
     * @param key The key to locate.
     * @returns index of where the key is (or should have been.)
     *   - positive (even) index if key found.
     *   - negative index if key not found. (`~index` (even) to get the index where it should have
     *     been inserted.)
     */
    function keyValueArrayIndexOf(keyValueArray, key) {
        return _arrayIndexOfSorted(keyValueArray, key, 1);
    }
    /**
     * INTERNAL: Get an index of an `value` in a sorted `array` by grouping search by `shift`.
     *
     * NOTE:
     * - This uses binary search algorithm for fast removals.
     *
     * @param array A sorted array to binary search.
     * @param value The value to look for.
     * @param shift grouping shift.
     *   - `0` means look at every location
     *   - `1` means only look at every other (even) location (the odd locations are to be ignored as
     *         they are values.)
     * @returns index of the value.
     *   - positive index if value found.
     *   - negative index if value not found. (`~index` to get the value where it should have been
     * inserted)
     */
    function _arrayIndexOfSorted(array, value, shift) {
        ngDevMode && assertEqual(Array.isArray(array), true, 'Expecting an array');
        let start = 0;
        let end = array.length >> shift;
        while (end !== start) {
            const middle = start + ((end - start) >> 1); // find the middle.
            const current = array[middle << shift];
            if (value === current) {
                return (middle << shift);
            }
            else if (current > value) {
                end = middle;
            }
            else {
                start = middle + 1; // We already searched middle so make it non-inclusive by adding 1
            }
        }
        return ~(end << shift);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/change_detection/constants.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /** @enum {number} */
    const ChangeDetectionStrategy = {
        /**
         * Use the `CheckOnce` strategy, meaning that automatic change detection is deactivated
         * until reactivated by setting the strategy to `Default` (`CheckAlways`).
         * Change detection can still be explicitly invoked.
         * This strategy applies to all child directives and cannot be overridden.
         */
        OnPush: 0,
        /**
         * Use the default `CheckAlways` strategy, in which change detection is automatic until
         * explicitly deactivated.
         */
        Default: 1,
    };
    ChangeDetectionStrategy[ChangeDetectionStrategy.OnPush] = 'OnPush';
    ChangeDetectionStrategy[ChangeDetectionStrategy.Default] = 'Default';

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/metadata/view.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /** @enum {number} */
    const ViewEncapsulation$1 = {
        /**
         * Emulate `Native` scoping of styles by adding an attribute containing surrogate id to the Host
         * Element and pre-processing the style rules provided via {@link Component#styles styles} or
         * {@link Component#styleUrls styleUrls}, and adding the new Host Element attribute to all
         * selectors.
         *
         * This is the default option.
         */
        Emulated: 0,
        /**
         * @deprecated v6.1.0 - use {ViewEncapsulation.ShadowDom} instead.
         * Use the native encapsulation mechanism of the renderer.
         *
         * For the DOM this means using the deprecated [Shadow DOM
         * v0](https://w3c.github.io/webcomponents/spec/shadow/) and
         * creating a ShadowRoot for Component's Host Element.
         */
        Native: 1,
        /**
         * Don't provide any template or style encapsulation.
         */
        None: 2,
        /**
         * Use Shadow DOM to encapsulate styles.
         *
         * For the DOM this means using modern [Shadow
         * DOM](https://w3c.github.io/webcomponents/spec/shadow/) and
         * creating a ShadowRoot for Component's Host Element.
         */
        ShadowDom: 3,
    };
    ViewEncapsulation$1[ViewEncapsulation$1.Emulated] = 'Emulated';
    ViewEncapsulation$1[ViewEncapsulation$1.Native] = 'Native';
    ViewEncapsulation$1[ViewEncapsulation$1.None] = 'None';
    ViewEncapsulation$1[ViewEncapsulation$1.ShadowDom] = 'ShadowDom';

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/empty.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * This file contains reuseable "empty" symbols that can be used as default return values
     * in different parts of the rendering code. Because the same symbols are returned, this
     * allows for identity checks against these values to be consistently used by the framework
     * code.
     * @type {?}
     */
    const EMPTY_OBJ = {};
    /** @type {?} */
    const EMPTY_ARRAY = [];
    // freezing the values prevents any code from accidentally inserting new values in
    if ((typeof ngDevMode === 'undefined' || ngDevMode) && initNgDevMode()) {
        // These property accesses can be ignored because ngDevMode will be set to false
        // when optimizing code and the whole if statement will be dropped.
        // tslint:disable-next-line:no-toplevel-property-access
        Object.freeze(EMPTY_OBJ);
        // tslint:disable-next-line:no-toplevel-property-access
        Object.freeze(EMPTY_ARRAY);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/definition.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    let _renderCompCount = 0;
    /**
     * Create a component definition object.
     *
     *
     * # Example
     * ```
     * class MyDirective {
     *   // Generated by Angular Template Compiler
     *   // [Symbol] syntax will not be supported by TypeScript until v2.7
     *   static ɵcmp = defineComponent({
     *     ...
     *   });
     * }
     * ```
     * \@codeGenApi
     * @template T
     * @param {?} componentDefinition
     * @return {?}
     */
    function ɵɵdefineComponent(componentDefinition) {
        return noSideEffects((/**
         * @return {?}
         */
        () => {
            // Initialize ngDevMode. This must be the first statement in ɵɵdefineComponent.
            // See the `initNgDevMode` docstring for more information.
            (typeof ngDevMode === 'undefined' || ngDevMode) && initNgDevMode();
            /** @type {?} */
            const type = componentDefinition.type;
            /** @type {?} */
            const typePrototype = type.prototype;
            /** @type {?} */
            const declaredInputs = (/** @type {?} */ ({}));
            /** @type {?} */
            const def = {
                type: type,
                providersResolver: null,
                decls: componentDefinition.decls,
                vars: componentDefinition.vars,
                factory: null,
                template: componentDefinition.template || (/** @type {?} */ (null)),
                consts: componentDefinition.consts || null,
                ngContentSelectors: componentDefinition.ngContentSelectors,
                hostBindings: componentDefinition.hostBindings || null,
                hostVars: componentDefinition.hostVars || 0,
                hostAttrs: componentDefinition.hostAttrs || null,
                contentQueries: componentDefinition.contentQueries || null,
                declaredInputs: declaredInputs,
                inputs: (/** @type {?} */ (null)),
                // assigned in noSideEffects
                outputs: (/** @type {?} */ (null)),
                // assigned in noSideEffects
                exportAs: componentDefinition.exportAs || null,
                onChanges: null,
                onInit: typePrototype.ngOnInit || null,
                doCheck: typePrototype.ngDoCheck || null,
                afterContentInit: typePrototype.ngAfterContentInit || null,
                afterContentChecked: typePrototype.ngAfterContentChecked || null,
                afterViewInit: typePrototype.ngAfterViewInit || null,
                afterViewChecked: typePrototype.ngAfterViewChecked || null,
                onDestroy: typePrototype.ngOnDestroy || null,
                onPush: componentDefinition.changeDetection === ChangeDetectionStrategy.OnPush,
                directiveDefs: (/** @type {?} */ (null)),
                // assigned in noSideEffects
                pipeDefs: (/** @type {?} */ (null)),
                // assigned in noSideEffects
                selectors: componentDefinition.selectors || EMPTY_ARRAY,
                viewQuery: componentDefinition.viewQuery || null,
                features: (/** @type {?} */ (componentDefinition.features)) || null,
                data: componentDefinition.data || {},
                // TODO(misko): convert ViewEncapsulation into const enum so that it can be used directly in
                // the next line. Also `None` should be 0 not 2.
                encapsulation: componentDefinition.encapsulation || ViewEncapsulation$1.Emulated,
                id: 'c',
                styles: componentDefinition.styles || EMPTY_ARRAY,
                _: (/** @type {?} */ (null)),
                setInput: null,
                schemas: componentDefinition.schemas || null,
                tView: null,
            };
            /** @type {?} */
            const directiveTypes = (/** @type {?} */ (componentDefinition.directives));
            /** @type {?} */
            const feature = componentDefinition.features;
            /** @type {?} */
            const pipeTypes = (/** @type {?} */ (componentDefinition.pipes));
            def.id += _renderCompCount++;
            def.inputs = invertObject(componentDefinition.inputs, declaredInputs),
                def.outputs = invertObject(componentDefinition.outputs),
                feature && feature.forEach((/**
                 * @param {?} fn
                 * @return {?}
                 */
                (fn) => fn(def)));
            def.directiveDefs = directiveTypes ?
                (/**
                 * @return {?}
                 */
                () => (typeof directiveTypes === 'function' ? directiveTypes() : directiveTypes)
                    .map(extractDirectiveDef)) :
                null;
            def.pipeDefs = pipeTypes ?
                (/**
                 * @return {?}
                 */
                () => (typeof pipeTypes === 'function' ? pipeTypes() : pipeTypes).map(extractPipeDef)) :
                null;
            return (/** @type {?} */ (def));
        }));
    }
    /**
     * \@codeGenApi
     * @param {?} type
     * @param {?} directives
     * @param {?} pipes
     * @return {?}
     */
    function ɵɵsetComponentScope(type, directives, pipes) {
        /** @type {?} */
        const def = ((/** @type {?} */ (type.ɵcmp)));
        def.directiveDefs = (/**
         * @return {?}
         */
        () => directives.map(extractDirectiveDef));
        def.pipeDefs = (/**
         * @return {?}
         */
        () => pipes.map(extractPipeDef));
    }
    /**
     * @param {?} type
     * @return {?}
     */
    function extractDirectiveDef(type) {
        /** @type {?} */
        const def = getComponentDef(type) || getDirectiveDef(type);
        if (ngDevMode && !def) {
            throw new Error(`'${type.name}' is neither 'ComponentType' or 'DirectiveType'.`);
        }
        return (/** @type {?} */ (def));
    }
    /**
     * @param {?} type
     * @return {?}
     */
    function extractPipeDef(type) {
        /** @type {?} */
        const def = getPipeDef(type);
        if (ngDevMode && !def) {
            throw new Error(`'${type.name}' is not a 'PipeType'.`);
        }
        return (/** @type {?} */ (def));
    }
    /** @type {?} */
    const autoRegisterModuleById = {};
    /**
     * \@codeGenApi
     * @template T
     * @param {?} def
     * @return {?}
     */
    function ɵɵdefineNgModule(def) {
        /** @type {?} */
        const res = {
            type: def.type,
            bootstrap: def.bootstrap || EMPTY_ARRAY,
            declarations: def.declarations || EMPTY_ARRAY,
            imports: def.imports || EMPTY_ARRAY,
            exports: def.exports || EMPTY_ARRAY,
            transitiveCompileScopes: null,
            schemas: def.schemas || null,
            id: def.id || null,
        };
        if (def.id != null) {
            noSideEffects((/**
             * @return {?}
             */
            () => {
                autoRegisterModuleById[(/** @type {?} */ (def.id))] = (/** @type {?} */ ((/** @type {?} */ (def.type))));
            }));
        }
        return (/** @type {?} */ (res));
    }
    /**
     * Adds the module metadata that is necessary to compute the module's transitive scope to an
     * existing module definition.
     *
     * Scope metadata of modules is not used in production builds, so calls to this function can be
     * marked pure to tree-shake it from the bundle, allowing for all referenced declarations
     * to become eligible for tree-shaking as well.
     *
     * \@codeGenApi
     * @param {?} type
     * @param {?} scope
     * @return {?}
     */
    function ɵɵsetNgModuleScope(type, scope) {
        return (/** @type {?} */ (noSideEffects((/**
         * @return {?}
         */
        () => {
            /** @type {?} */
            const ngModuleDef = getNgModuleDef(type, true);
            ngModuleDef.declarations = scope.declarations || EMPTY_ARRAY;
            ngModuleDef.imports = scope.imports || EMPTY_ARRAY;
            ngModuleDef.exports = scope.exports || EMPTY_ARRAY;
        }))));
    }
    /**
     * Inverts an inputs or outputs lookup such that the keys, which were the
     * minified keys, are part of the values, and the values are parsed so that
     * the publicName of the property is the new key
     *
     * e.g. for
     *
     * ```
     * class Comp {
     * \@Input()
     *   propName1: string;
     *
     * \@Input('publicName2')
     *   declaredPropName2: number;
     * }
     * ```
     *
     * will be serialized as
     *
     * ```
     * {
     *   propName1: 'propName1',
     *   declaredPropName2: ['publicName2', 'declaredPropName2'],
     * }
     * ```
     *
     * which is than translated by the minifier as:
     *
     * ```
     * {
     *   minifiedPropName1: 'propName1',
     *   minifiedPropName2: ['publicName2', 'declaredPropName2'],
     * }
     * ```
     *
     * becomes: (public name => minifiedName)
     *
     * ```
     * {
     *  'propName1': 'minifiedPropName1',
     *  'publicName2': 'minifiedPropName2',
     * }
     * ```
     *
     * Optionally the function can take `secondary` which will result in: (public name => declared name)
     *
     * ```
     * {
     *  'propName1': 'propName1',
     *  'publicName2': 'declaredPropName2',
     * }
     * ```
     *
     * @template T
     * @param {?=} obj
     * @param {?=} secondary
     * @return {?}
     */
    function invertObject(obj, secondary) {
        if (obj == null)
            return (/** @type {?} */ (EMPTY_OBJ));
        /** @type {?} */
        const newLookup = {};
        for (const minifiedKey in obj) {
            if (obj.hasOwnProperty(minifiedKey)) {
                /** @type {?} */
                let publicName = (/** @type {?} */ (obj[minifiedKey]));
                /** @type {?} */
                let declaredName = publicName;
                if (Array.isArray(publicName)) {
                    declaredName = publicName[1];
                    publicName = publicName[0];
                }
                newLookup[publicName] = minifiedKey;
                if (secondary) {
                    (secondary[publicName] = (/** @type {?} */ (declaredName)));
                }
            }
        }
        return newLookup;
    }
    /**
     * Create a directive definition object.
     *
     * # Example
     * ```ts
     * class MyDirective {
     *   // Generated by Angular Template Compiler
     *   // [Symbol] syntax will not be supported by TypeScript until v2.7
     *   static ɵdir = ɵɵdefineDirective({
     *     ...
     *   });
     * }
     * ```
     *
     * \@codeGenApi
     * @type {?}
     */
    const ɵɵdefineDirective = (/** @type {?} */ ((/** @type {?} */ (ɵɵdefineComponent))));
    /**
     * Create a pipe definition object.
     *
     * # Example
     * ```
     * class MyPipe implements PipeTransform {
     *   // Generated by Angular Template Compiler
     *   static ɵpipe = definePipe({
     *     ...
     *   });
     * }
     * ```
     * \@codeGenApi
     * @template T
     * @param {?} pipeDef Pipe definition generated by the compiler
     *
     * @return {?}
     */
    function ɵɵdefinePipe(pipeDef) {
        return (/** @type {?} */ (((/** @type {?} */ ({
            type: pipeDef.type,
            name: pipeDef.name,
            factory: null,
            pure: pipeDef.pure !== false,
            onDestroy: pipeDef.type.prototype.ngOnDestroy || null
        })))));
    }
    /**
     * The following getter methods retrieve the definition form the type. Currently the retrieval
     * honors inheritance, but in the future we may change the rule to require that definitions are
     * explicit. This would require some sort of migration strategy.
     * @template T
     * @param {?} type
     * @return {?}
     */
    function getComponentDef(type) {
        return type[NG_COMP_DEF] || null;
    }
    /**
     * @template T
     * @param {?} type
     * @return {?}
     */
    function getDirectiveDef(type) {
        return type[NG_DIR_DEF] || null;
    }
    /**
     * @template T
     * @param {?} type
     * @return {?}
     */
    function getPipeDef(type) {
        return type[NG_PIPE_DEF] || null;
    }
    /**
     * @template T
     * @param {?} type
     * @param {?=} throwNotFound
     * @return {?}
     */
    function getFactoryDef(type, throwNotFound) {
        /** @type {?} */
        const hasFactoryDef = type.hasOwnProperty(NG_FACTORY_DEF);
        if (!hasFactoryDef && throwNotFound === true && ngDevMode) {
            throw new Error(`Type ${stringify(type)} does not have 'ɵfac' property.`);
        }
        return hasFactoryDef ? type[NG_FACTORY_DEF] : null;
    }
    /**
     * @template T
     * @param {?} type
     * @param {?=} throwNotFound
     * @return {?}
     */
    function getNgModuleDef(type, throwNotFound) {
        /** @type {?} */
        const ngModuleDef = type[NG_MOD_DEF] || null;
        if (!ngModuleDef && throwNotFound === true) {
            throw new Error(`Type ${stringify(type)} does not have 'ɵmod' property.`);
        }
        return ngModuleDef;
    }
    /**
     * @param {?} type
     * @return {?}
     */
    function getNgLocaleIdDef(type) {
        return ((/** @type {?} */ (type)))[NG_LOC_ID_DEF] || null;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/interfaces/view.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    // Below are constants for LView indices to help us look up LView members
    // without having to remember the specific indices.
    // Uglify will inline these when minifying so there shouldn't be a cost.
    /** @type {?} */
    const HOST = 0;
    /** @type {?} */
    const TVIEW = 1;
    /** @type {?} */
    const FLAGS = 2;
    /** @type {?} */
    const PARENT = 3;
    /** @type {?} */
    const NEXT = 4;
    /** @type {?} */
    const TRANSPLANTED_VIEWS_TO_REFRESH = 5;
    /** @type {?} */
    const T_HOST = 6;
    /** @type {?} */
    const CLEANUP = 7;
    /** @type {?} */
    const CONTEXT = 8;
    /** @type {?} */
    const INJECTOR$1 = 9;
    /** @type {?} */
    const RENDERER_FACTORY = 10;
    /** @type {?} */
    const RENDERER = 11;
    /** @type {?} */
    const SANITIZER = 12;
    /** @type {?} */
    const CHILD_HEAD = 13;
    /** @type {?} */
    const CHILD_TAIL = 14;
    /** @type {?} */
    const DECLARATION_VIEW = 15;
    /** @type {?} */
    const DECLARATION_COMPONENT_VIEW = 16;
    /** @type {?} */
    const DECLARATION_LCONTAINER = 17;
    /** @type {?} */
    const PREORDER_HOOK_FLAGS = 18;
    /** @type {?} */
    const QUERIES = 19;
    /**
     * Size of LView's header. Necessary to adjust for it when setting slots.
     * @type {?}
     */
    const HEADER_OFFSET = 20;

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/interfaces/container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Special location which allows easy identification of type. If we have an array which was
     * retrieved from the `LView` and that array has `true` at `TYPE` location, we know it is
     * `LContainer`.
     * @type {?}
     */
    const TYPE = 1;
    /**
     * Flag to signify that this `LContainer` may have transplanted views which need to be change
     * detected. (see: `LView[DECLARATION_COMPONENT_VIEW])`.
     *
     * This flag, once set, is never unset for the `LContainer`. This means that when unset we can skip
     * a lot of work in `refreshEmbeddedViews`. But when set we still need to verify
     * that the `MOVED_VIEWS` are transplanted and on-push.
     * @type {?}
     */
    const HAS_TRANSPLANTED_VIEWS = 2;
    // PARENT, NEXT, TRANSPLANTED_VIEWS_TO_REFRESH are indices 3, 4, and 5
    // As we already have these constants in LView, we don't need to re-create them.
    // T_HOST is index 6
    // We already have this constants in LView, we don't need to re-create it.
    /** @type {?} */
    const NATIVE = 7;
    /** @type {?} */
    const VIEW_REFS = 8;
    /** @type {?} */
    const MOVED_VIEWS = 9;
    /**
     * Size of LContainer's header. Represents the index after which all views in the
     * container will be inserted. We need to keep a record of current views so we know
     * which views are already in the DOM (and don't need to be re-added) and so we can
     * remove views from the DOM when they are no longer required.
     * @type {?}
     */
    const CONTAINER_HEADER_OFFSET = 10;

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/interfaces/type_checks.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * True if `value` is `LView`.
     * @param {?} value wrapped value of `RNode`, `LView`, `LContainer`
     * @return {?}
     */
    function isLView(value) {
        return Array.isArray(value) && typeof value[TYPE] === 'object';
    }
    /**
     * True if `value` is `LContainer`.
     * @param {?} value wrapped value of `RNode`, `LView`, `LContainer`
     * @return {?}
     */
    function isLContainer(value) {
        return Array.isArray(value) && value[TYPE] === true;
    }
    /**
     * @param {?} tNode
     * @return {?}
     */
    function isContentQueryHost(tNode) {
        return (tNode.flags & 8 /* hasContentQuery */) !== 0;
    }
    /**
     * @param {?} tNode
     * @return {?}
     */
    function isComponentHost(tNode) {
        return (tNode.flags & 2 /* isComponentHost */) === 2 /* isComponentHost */;
    }
    /**
     * @param {?} tNode
     * @return {?}
     */
    function isDirectiveHost(tNode) {
        return (tNode.flags & 1 /* isDirectiveHost */) === 1 /* isDirectiveHost */;
    }
    /**
     * @template T
     * @param {?} def
     * @return {?}
     */
    function isComponentDef(def) {
        return ((/** @type {?} */ (def))).template !== null;
    }
    /**
     * @param {?} target
     * @return {?}
     */
    function isRootView(target) {
        return (target[FLAGS] & 512 /* IsRoot */) !== 0;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/assert.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    // [Assert functions do not constraint type when they are guarded by a truthy
    // expression.](https://github.com/microsoft/TypeScript/issues/37295)
    /**
     * @param {?} tNode
     * @param {?} lView
     * @return {?}
     */
    function assertTNodeForLView(tNode, lView) {
        tNode.hasOwnProperty('tView_') &&
            assertEqual(((/** @type {?} */ ((/** @type {?} */ (tNode))))).tView_, lView[TVIEW], 'This TNode does not belong to this LView.');
    }
    /**
     * @param {?} actual
     * @param {?=} msg
     * @return {?}
     */
    function assertComponentType(actual, msg = 'Type passed in is not ComponentType, it does not have \'ɵcmp\' property.') {
        if (!getComponentDef(actual)) {
            throwError$1(msg);
        }
    }
    /**
     * @param {?} actual
     * @param {?=} msg
     * @return {?}
     */
    function assertNgModuleType(actual, msg = 'Type passed in is not NgModuleType, it does not have \'ɵmod\' property.') {
        if (!getNgModuleDef(actual)) {
            throwError$1(msg);
        }
    }
    /**
     * @param {?} tNode
     * @return {?}
     */
    function assertHasParent(tNode) {
        assertDefined(tNode, 'previousOrParentTNode should exist!');
        assertDefined((/** @type {?} */ (tNode)).parent, 'previousOrParentTNode should have a parent');
    }
    /**
     * @param {?} value
     * @return {?}
     */
    function assertLContainer(value) {
        assertDefined(value, 'LContainer must be defined');
        assertEqual(isLContainer(value), true, 'Expecting LContainer');
    }
    /**
     * @param {?} value
     * @return {?}
     */
    function assertLViewOrUndefined(value) {
        value && assertEqual(isLView(value), true, 'Expecting LView or undefined or null');
    }
    /**
     * @param {?} value
     * @return {?}
     */
    function assertLView(value) {
        assertDefined(value, 'LView must be defined');
        assertEqual(isLView(value), true, 'Expecting LView');
    }
    /**
     * @param {?} tView
     * @param {?=} errMessage
     * @return {?}
     */
    function assertFirstCreatePass(tView, errMessage) {
        assertEqual(tView.firstCreatePass, true, errMessage || 'Should only be called in first create pass.');
    }
    /**
     * @param {?} tView
     * @param {?=} errMessage
     * @return {?}
     */
    function assertFirstUpdatePass(tView, errMessage) {
        assertEqual(tView.firstUpdatePass, true, errMessage || 'Should only be called in first update pass.');
    }
    /**
     * This is a basic sanity check that an object is probably a directive def. DirectiveDef is
     * an interface, so we can't do a direct instanceof check.
     * @template T
     * @param {?} obj
     * @return {?}
     */
    function assertDirectiveDef(obj) {
        if (obj.type === undefined || obj.selectors == undefined || obj.inputs === undefined) {
            throwError$1(`Expected a DirectiveDef/ComponentDef and this object does not seem to have the expected shape.`);
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/namespaces.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /** @type {?} */
    const SVG_NAMESPACE = 'http://www.w3.org/2000/svg';
    /** @type {?} */
    const MATH_ML_NAMESPACE = 'http://www.w3.org/1998/MathML/';

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/interfaces/context.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * This property will be monkey-patched on elements, components and directives
     * @type {?}
     */
    const MONKEY_PATCH_KEY_NAME = '__ngContext__';

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/interfaces/document.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * Most of the use of `document` in Angular is from within the DI system so it is possible to simply
     * inject the `DOCUMENT` token and are done.
     *
     * Ivy is special because it does not rely upon the DI and must get hold of the document some other
     * way.
     *
     * The solution is to define `getDocument()` and `setDocument()` top-level functions for ivy.
     * Wherever ivy needs the global document, it calls `getDocument()` instead.
     *
     * When running ivy outside of a browser environment, it is necessary to call `setDocument()` to
     * tell ivy what the global `document` is.
     *
     * Angular does this for us in each of the standard platforms (`Browser`, `Server`, and `WebWorker`)
     * by calling `setDocument()` when providing the `DOCUMENT` token.
     * @type {?}
     */
    let DOCUMENT = undefined;
    /**
     * Tell ivy what the `document` is for this platform.
     *
     * It is only necessary to call this if the current platform is not a browser.
     *
     * @param {?} document The object representing the global `document` in this environment.
     * @return {?}
     */
    function setDocument(document) {
        DOCUMENT = document;
    }
    /**
     * Access the object that represents the `document` for this platform.
     *
     * Ivy calls this whenever it needs to access the `document` object.
     * For example to create the renderer or to do sanitization.
     * @return {?}
     */
    function getDocument() {
        if (DOCUMENT !== undefined) {
            return DOCUMENT;
        }
        else if (typeof document !== 'undefined') {
            return document;
        }
        // No "document" can be found. This should only happen if we are running ivy outside Angular and
        // the current platform is not a browser. Since this is not a supported scenario at the moment
        // this should not happen in Angular apps.
        // Once we support running ivy outside of Angular we will need to publish `setDocument()` as a
        // public API. Meanwhile we just return `undefined` and let the application fail.
        return (/** @type {?} */ (undefined));
    }
    /**
     * Returns whether the `renderer` is a `ProceduralRenderer3`
     * @param {?} renderer
     * @return {?}
     */
    function isProceduralRenderer(renderer) {
        return !!(((/** @type {?} */ (renderer))).listen);
    }
    const ɵ0$2 = /**
     * @param {?} hostElement
     * @param {?} rendererType
     * @return {?}
     */
    (hostElement, rendererType) => {
        return getDocument();
    };
    /** @type {?} */
    const domRendererFactory3 = {
        createRenderer: (ɵ0$2)
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/util/view_utils.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * For efficiency reasons we often put several different data types (`RNode`, `LView`, `LContainer`)
     * in same location in `LView`. This is because we don't want to pre-allocate space for it
     * because the storage is sparse. This file contains utilities for dealing with such data types.
     *
     * How do we know what is stored at a given location in `LView`.
     * - `Array.isArray(value) === false` => `RNode` (The normal storage value)
     * - `Array.isArray(value) === true` => then the `value[0]` represents the wrapped value.
     *   - `typeof value[TYPE] === 'object'` => `LView`
     *      - This happens when we have a component at a given location
     *   - `typeof value[TYPE] === true` => `LContainer`
     *      - This happens when we have `LContainer` binding at a given location.
     *
     *
     * NOTE: it is assumed that `Array.isArray` and `typeof` operations are very efficient.
     */
    /**
     * Returns `RNode`.
     * @param {?} value wrapped value of `RNode`, `LView`, `LContainer`
     * @return {?}
     */
    function unwrapRNode(value) {
        while (Array.isArray(value)) {
            value = (/** @type {?} */ (value[HOST]));
        }
        return (/** @type {?} */ (value));
    }
    /**
     * Returns `LView` or `null` if not found.
     * @param {?} value wrapped value of `RNode`, `LView`, `LContainer`
     * @return {?}
     */
    function unwrapLView(value) {
        while (Array.isArray(value)) {
            // This check is same as `isLView()` but we don't call at as we don't want to call
            // `Array.isArray()` twice and give JITer more work for inlining.
            if (typeof value[TYPE] === 'object')
                return (/** @type {?} */ (value));
            value = (/** @type {?} */ (value[HOST]));
        }
        return null;
    }
    /**
     * Retrieves an element value from the provided `viewData`, by unwrapping
     * from any containers, component views, or style contexts.
     * @param {?} index
     * @param {?} lView
     * @return {?}
     */
    function getNativeByIndex(index, lView) {
        return unwrapRNode(lView[index + HEADER_OFFSET]);
    }
    /**
     * Retrieve an `RNode` for a given `TNode` and `LView`.
     *
     * This function guarantees in dev mode to retrieve a non-null `RNode`.
     *
     * @param {?} tNode
     * @param {?} lView
     * @return {?}
     */
    function getNativeByTNode(tNode, lView) {
        ngDevMode && assertTNodeForLView(tNode, lView);
        ngDevMode && assertDataInRange(lView, tNode.index);
        /** @type {?} */
        const node = unwrapRNode(lView[tNode.index]);
        ngDevMode && !isProceduralRenderer(lView[RENDERER]) && assertDomNode(node);
        return node;
    }
    /**
     * Retrieve an `RNode` or `null` for a given `TNode` and `LView`.
     *
     * Some `TNode`s don't have associated `RNode`s. For example `Projection`
     *
     * @param {?} tNode
     * @param {?} lView
     * @return {?}
     */
    function getNativeByTNodeOrNull(tNode, lView) {
        /** @type {?} */
        const index = tNode.index;
        if (index !== -1) {
            ngDevMode && assertTNodeForLView(tNode, lView);
            /** @type {?} */
            const node = unwrapRNode(lView[index]);
            ngDevMode && node !== null && !isProceduralRenderer(lView[RENDERER]) && assertDomNode(node);
            return node;
        }
        return null;
    }
    /**
     * @param {?} tView
     * @param {?} index
     * @return {?}
     */
    function getTNode(tView, index) {
        ngDevMode && assertGreaterThan(index, -1, 'wrong index for TNode');
        ngDevMode && assertLessThan(index, tView.data.length, 'wrong index for TNode');
        return (/** @type {?} */ (tView.data[index + HEADER_OFFSET]));
    }
    /**
     * Retrieves a value from any `LView` or `TData`.
     * @template T
     * @param {?} view
     * @param {?} index
     * @return {?}
     */
    function load(view, index) {
        ngDevMode && assertDataInRange(view, index + HEADER_OFFSET);
        return view[index + HEADER_OFFSET];
    }
    /**
     * @param {?} nodeIndex
     * @param {?} hostView
     * @return {?}
     */
    function getComponentLViewByIndex(nodeIndex, hostView) {
        // Could be an LView or an LContainer. If LContainer, unwrap to find LView.
        ngDevMode && assertDataInRange(hostView, nodeIndex);
        /** @type {?} */
        const slotValue = hostView[nodeIndex];
        /** @type {?} */
        const lView = isLView(slotValue) ? slotValue : slotValue[HOST];
        return lView;
    }
    /**
     * Returns the monkey-patch value data present on the target (which could be
     * a component, directive or a DOM node).
     * @param {?} target
     * @return {?}
     */
    function readPatchedData(target) {
        ngDevMode && assertDefined(target, 'Target expected');
        return target[MONKEY_PATCH_KEY_NAME] || null;
    }
    /**
     * @param {?} target
     * @return {?}
     */
    function readPatchedLView(target) {
        /** @type {?} */
        const value = readPatchedData(target);
        if (value) {
            return Array.isArray(value) ? value : ((/** @type {?} */ (value))).lView;
        }
        return null;
    }
    /**
     * Checks whether a given view is in creation mode
     * @param {?} view
     * @return {?}
     */
    function isCreationMode(view) {
        return (view[FLAGS] & 4 /* CreationMode */) === 4 /* CreationMode */;
    }
    /**
     * Returns a boolean for whether the view is attached to the change detection tree.
     *
     * Note: This determines whether a view should be checked, not whether it's inserted
     * into a container. For that, you'll want `viewAttachedToContainer` below.
     * @param {?} view
     * @return {?}
     */
    function viewAttachedToChangeDetector(view) {
        return (view[FLAGS] & 128 /* Attached */) === 128 /* Attached */;
    }
    /**
     * Returns a boolean for whether the view is attached to a container.
     * @param {?} view
     * @return {?}
     */
    function viewAttachedToContainer(view) {
        return isLContainer(view[PARENT]);
    }
    /**
     * Returns a constant from `TConstants` instance.
     * @template T
     * @param {?} consts
     * @param {?} index
     * @return {?}
     */
    function getConstant(consts, index) {
        return consts === null || index == null ? null : (/** @type {?} */ ((/** @type {?} */ (consts[index]))));
    }
    /**
     * Resets the pre-order hook flags of the view.
     * @param {?} lView the LView on which the flags are reset
     * @return {?}
     */
    function resetPreOrderHookFlags(lView) {
        lView[PREORDER_HOOK_FLAGS] = 0;
    }
    /**
     * Updates the `TRANSPLANTED_VIEWS_TO_REFRESH` counter on the `LContainer` as well as the parents
     * whose
     *  1. counter goes from 0 to 1, indicating that there is a new child that has a view to refresh
     *  or
     *  2. counter goes from 1 to 0, indicating there are no more descendant views to refresh
     * @param {?} lContainer
     * @param {?} amount
     * @return {?}
     */
    function updateTransplantedViewCount(lContainer, amount) {
        lContainer[TRANSPLANTED_VIEWS_TO_REFRESH] += amount;
        /** @type {?} */
        let viewOrContainer = lContainer;
        /** @type {?} */
        let parent = lContainer[PARENT];
        while (parent !== null &&
            ((amount === 1 && viewOrContainer[TRANSPLANTED_VIEWS_TO_REFRESH] === 1) ||
                (amount === -1 && viewOrContainer[TRANSPLANTED_VIEWS_TO_REFRESH] === 0))) {
            parent[TRANSPLANTED_VIEWS_TO_REFRESH] += amount;
            viewOrContainer = parent;
            parent = parent[PARENT];
        }
    }
    /** @type {?} */
    const instructionState = {
        lFrame: createLFrame(null),
        bindingsEnabled: true,
        checkNoChangesMode: false,
    };
    /**
     * @return {?}
     */
    function getElementDepthCount() {
        return instructionState.lFrame.elementDepthCount;
    }
    /**
     * @return {?}
     */
    function increaseElementDepthCount() {
        instructionState.lFrame.elementDepthCount++;
    }
    /**
     * @return {?}
     */
    function decreaseElementDepthCount() {
        instructionState.lFrame.elementDepthCount--;
    }
    /**
     * @return {?}
     */
    function getBindingsEnabled() {
        return instructionState.bindingsEnabled;
    }
    /**
     * Enables directive matching on elements.
     *
     *  * Example:
     * ```
     * <my-comp my-directive>
     *   Should match component / directive.
     * </my-comp>
     * <div ngNonBindable>
     *   <!-- ɵɵdisableBindings() -->
     *   <my-comp my-directive>
     *     Should not match component / directive because we are in ngNonBindable.
     *   </my-comp>
     *   <!-- ɵɵenableBindings() -->
     * </div>
     * ```
     *
     * \@codeGenApi
     * @return {?}
     */
    function ɵɵenableBindings() {
        instructionState.bindingsEnabled = true;
    }
    /**
     * Disables directive matching on element.
     *
     *  * Example:
     * ```
     * <my-comp my-directive>
     *   Should match component / directive.
     * </my-comp>
     * <div ngNonBindable>
     *   <!-- ɵɵdisableBindings() -->
     *   <my-comp my-directive>
     *     Should not match component / directive because we are in ngNonBindable.
     *   </my-comp>
     *   <!-- ɵɵenableBindings() -->
     * </div>
     * ```
     *
     * \@codeGenApi
     * @return {?}
     */
    function ɵɵdisableBindings() {
        instructionState.bindingsEnabled = false;
    }
    /**
     * Return the current `LView`.
     * @return {?}
     */
    function getLView() {
        return instructionState.lFrame.lView;
    }
    /**
     * Return the current `TView`.
     * @return {?}
     */
    function getTView() {
        return instructionState.lFrame.tView;
    }
    /**
     * Restores `contextViewData` to the given OpaqueViewState instance.
     *
     * Used in conjunction with the getCurrentView() instruction to save a snapshot
     * of the current view and restore it when listeners are invoked. This allows
     * walking the declaration view tree in listeners to get vars from parent views.
     *
     * \@codeGenApi
     * @param {?} viewToRestore The OpaqueViewState instance to restore.
     *
     * @return {?}
     */
    function ɵɵrestoreView(viewToRestore) {
        instructionState.lFrame.contextLView = (/** @type {?} */ ((/** @type {?} */ (viewToRestore))));
    }
    /**
     * @return {?}
     */
    function getPreviousOrParentTNode() {
        return instructionState.lFrame.previousOrParentTNode;
    }
    /**
     * @param {?} tNode
     * @param {?} isParent
     * @return {?}
     */
    function setPreviousOrParentTNode(tNode, isParent) {
        instructionState.lFrame.previousOrParentTNode = tNode;
        instructionState.lFrame.isParent = isParent;
    }
    /**
     * @return {?}
     */
    function getIsParent() {
        return instructionState.lFrame.isParent;
    }
    /**
     * @return {?}
     */
    function setIsNotParent() {
        instructionState.lFrame.isParent = false;
    }
    /**
     * @return {?}
     */
    function getContextLView() {
        return instructionState.lFrame.contextLView;
    }
    /**
     * @return {?}
     */
    function getCheckNoChangesMode() {
        // TODO(misko): remove this from the LView since it is ngDevMode=true mode only.
        return instructionState.checkNoChangesMode;
    }
    /**
     * @param {?} mode
     * @return {?}
     */
    function setCheckNoChangesMode(mode) {
        instructionState.checkNoChangesMode = mode;
    }
    // top level variables should not be exported for performance reasons (PERF_NOTES.md)
    /**
     * @return {?}
     */
    function getBindingRoot() {
        /** @type {?} */
        const lFrame = instructionState.lFrame;
        /** @type {?} */
        let index = lFrame.bindingRootIndex;
        if (index === -1) {
            index = lFrame.bindingRootIndex = lFrame.tView.bindingStartIndex;
        }
        return index;
    }
    /**
     * @return {?}
     */
    function getBindingIndex() {
        return instructionState.lFrame.bindingIndex;
    }
    /**
     * @param {?} value
     * @return {?}
     */
    function setBindingIndex(value) {
        return instructionState.lFrame.bindingIndex = value;
    }
    /**
     * @return {?}
     */
    function nextBindingIndex() {
        return instructionState.lFrame.bindingIndex++;
    }
    /**
     * @param {?} count
     * @return {?}
     */
    function incrementBindingIndex(count) {
        /** @type {?} */
        const lFrame = instructionState.lFrame;
        /** @type {?} */
        const index = lFrame.bindingIndex;
        lFrame.bindingIndex = lFrame.bindingIndex + count;
        return index;
    }
    /**
     * Set a new binding root index so that host template functions can execute.
     *
     * Bindings inside the host template are 0 index. But because we don't know ahead of time
     * how many host bindings we have we can't pre-compute them. For this reason they are all
     * 0 index and we just shift the root so that they match next available location in the LView.
     *
     * @param {?} bindingRootIndex Root index for `hostBindings`
     * @param {?} currentDirectiveIndex `TData[currentDirectiveIndex]` will point to the current directive
     *        whose `hostBindings` are being processed.
     * @return {?}
     */
    function setBindingRootForHostBindings(bindingRootIndex, currentDirectiveIndex) {
        /** @type {?} */
        const lFrame = instructionState.lFrame;
        lFrame.bindingIndex = lFrame.bindingRootIndex = bindingRootIndex;
        setCurrentDirectiveIndex(currentDirectiveIndex);
    }
    /**
     * When host binding is executing this points to the directive index.
     * `TView.data[getCurrentDirectiveIndex()]` is `DirectiveDef`
     * `LView[getCurrentDirectiveIndex()]` is directive instance.
     * @return {?}
     */
    function getCurrentDirectiveIndex() {
        return instructionState.lFrame.currentDirectiveIndex;
    }
    /**
     * Sets an index of a directive whose `hostBindings` are being processed.
     *
     * @param {?} currentDirectiveIndex `TData` index where current directive instance can be found.
     * @return {?}
     */
    function setCurrentDirectiveIndex(currentDirectiveIndex) {
        instructionState.lFrame.currentDirectiveIndex = currentDirectiveIndex;
    }
    /**
     * Retrieve the current `DirectiveDef` which is active when `hostBindings` instruction is being
     * executed.
     *
     * @param {?} tData Current `TData` where the `DirectiveDef` will be looked up at.
     * @return {?}
     */
    function getCurrentDirectiveDef(tData) {
        /** @type {?} */
        const currentDirectiveIndex = instructionState.lFrame.currentDirectiveIndex;
        return currentDirectiveIndex === -1 ? null : (/** @type {?} */ (tData[currentDirectiveIndex]));
    }
    /**
     * @return {?}
     */
    function getCurrentQueryIndex() {
        return instructionState.lFrame.currentQueryIndex;
    }
    /**
     * @param {?} value
     * @return {?}
     */
    function setCurrentQueryIndex(value) {
        instructionState.lFrame.currentQueryIndex = value;
    }
    /**
     * This is a light weight version of the `enterView` which is needed by the DI system.
     * @param {?} newView
     * @param {?} tNode
     * @return {?}
     */
    function enterDI(newView, tNode) {
        ngDevMode && assertLViewOrUndefined(newView);
        /** @type {?} */
        const newLFrame = allocLFrame();
        instructionState.lFrame = newLFrame;
        newLFrame.previousOrParentTNode = (/** @type {?} */ (tNode));
        newLFrame.lView = newView;
    }
    /**
     * Swap the current lView with a new lView.
     *
     * For performance reasons we store the lView in the top level of the module.
     * This way we minimize the number of properties to read. Whenever a new view
     * is entered we have to store the lView for later, and when the view is
     * exited the state has to be restored
     *
     * @param {?} newView New lView to become active
     * @param {?} tNode Element to which the View is a child of
     * @return {?} the previously active lView;
     */
    function enterView(newView, tNode) {
        ngDevMode && assertLViewOrUndefined(newView);
        /** @type {?} */
        const newLFrame = allocLFrame();
        if (ngDevMode) {
            assertEqual(newLFrame.isParent, true, 'Expected clean LFrame');
            assertEqual(newLFrame.lView, null, 'Expected clean LFrame');
            assertEqual(newLFrame.tView, null, 'Expected clean LFrame');
            assertEqual(newLFrame.selectedIndex, 0, 'Expected clean LFrame');
            assertEqual(newLFrame.elementDepthCount, 0, 'Expected clean LFrame');
            assertEqual(newLFrame.currentDirectiveIndex, -1, 'Expected clean LFrame');
            assertEqual(newLFrame.currentNamespace, null, 'Expected clean LFrame');
            assertEqual(newLFrame.currentSanitizer, null, 'Expected clean LFrame');
            assertEqual(newLFrame.bindingRootIndex, -1, 'Expected clean LFrame');
            assertEqual(newLFrame.currentQueryIndex, 0, 'Expected clean LFrame');
        }
        /** @type {?} */
        const tView = newView[TVIEW];
        instructionState.lFrame = newLFrame;
        newLFrame.previousOrParentTNode = (/** @type {?} */ (tNode));
        newLFrame.lView = newView;
        newLFrame.tView = tView;
        newLFrame.contextLView = (/** @type {?} */ (newView));
        newLFrame.bindingIndex = tView.bindingStartIndex;
    }
    /**
     * Allocates next free LFrame. This function tries to reuse the `LFrame`s to lower memory pressure.
     * @return {?}
     */
    function allocLFrame() {
        /** @type {?} */
        const currentLFrame = instructionState.lFrame;
        /** @type {?} */
        const childLFrame = currentLFrame === null ? null : currentLFrame.child;
        /** @type {?} */
        const newLFrame = childLFrame === null ? createLFrame(currentLFrame) : childLFrame;
        return newLFrame;
    }
    /**
     * @param {?} parent
     * @return {?}
     */
    function createLFrame(parent) {
        /** @type {?} */
        const lFrame = {
            previousOrParentTNode: (/** @type {?} */ (null)),
            //
            isParent: true,
            //
            lView: (/** @type {?} */ (null)),
            //
            tView: (/** @type {?} */ (null)),
            //
            selectedIndex: 0,
            //
            contextLView: (/** @type {?} */ (null)),
            //
            elementDepthCount: 0,
            //
            currentNamespace: null,
            //
            currentSanitizer: null,
            //
            currentDirectiveIndex: -1,
            //
            bindingRootIndex: -1,
            //
            bindingIndex: -1,
            //
            currentQueryIndex: 0,
            //
            parent: (/** @type {?} */ (parent)),
            //
            child: null,
        };
        parent !== null && (parent.child = lFrame); // link the new LFrame for reuse.
        return lFrame;
    }
    /**
     * A lightweight version of leave which is used with DI.
     *
     * This function only resets `previousOrParentTNode` and `LView` as those are the only properties
     * used with DI (`enterDI()`).
     *
     * NOTE: This function is reexported as `leaveDI`. However `leaveDI` has return type of `void` where
     * as `leaveViewLight` has `LFrame`. This is so that `leaveViewLight` can be used in `leaveView`.
     * @return {?}
     */
    function leaveViewLight() {
        /** @type {?} */
        const oldLFrame = instructionState.lFrame;
        instructionState.lFrame = oldLFrame.parent;
        oldLFrame.previousOrParentTNode = (/** @type {?} */ (null));
        oldLFrame.lView = (/** @type {?} */ (null));
        return oldLFrame;
    }
    /**
     * This is a lightweight version of the `leaveView` which is needed by the DI system.
     *
     * NOTE: this function is an alias so that we can change the type of the function to have `void`
     * return type.
     * @type {?}
     */
    const leaveDI = leaveViewLight;
    /**
     * Leave the current `LView`
     *
     * This pops the `LFrame` with the associated `LView` from the stack.
     *
     * IMPORTANT: We must zero out the `LFrame` values here otherwise they will be retained. This is
     * because for performance reasons we don't release `LFrame` but rather keep it for next use.
     * @return {?}
     */
    function leaveView() {
        /** @type {?} */
        const oldLFrame = leaveViewLight();
        oldLFrame.isParent = true;
        oldLFrame.tView = (/** @type {?} */ (null));
        oldLFrame.selectedIndex = 0;
        oldLFrame.contextLView = (/** @type {?} */ (null));
        oldLFrame.elementDepthCount = 0;
        oldLFrame.currentDirectiveIndex = -1;
        oldLFrame.currentNamespace = null;
        oldLFrame.currentSanitizer = null;
        oldLFrame.bindingRootIndex = -1;
        oldLFrame.bindingIndex = -1;
        oldLFrame.currentQueryIndex = 0;
    }
    /**
     * @template T
     * @param {?} level
     * @return {?}
     */
    function nextContextImpl(level) {
        /** @type {?} */
        const contextLView = instructionState.lFrame.contextLView =
            walkUpViews(level, (/** @type {?} */ (instructionState.lFrame.contextLView)));
        return (/** @type {?} */ (contextLView[CONTEXT]));
    }
    /**
     * @param {?} nestingLevel
     * @param {?} currentView
     * @return {?}
     */
    function walkUpViews(nestingLevel, currentView) {
        while (nestingLevel > 0) {
            ngDevMode &&
                assertDefined(currentView[DECLARATION_VIEW], 'Declaration view should be defined if nesting level is greater than 0.');
            currentView = (/** @type {?} */ (currentView[DECLARATION_VIEW]));
            nestingLevel--;
        }
        return currentView;
    }
    /**
     * Gets the currently selected element index.
     *
     * Used with {\@link property} instruction (and more in the future) to identify the index in the
     * current `LView` to act on.
     * @return {?}
     */
    function getSelectedIndex() {
        return instructionState.lFrame.selectedIndex;
    }
    /**
     * Sets the most recent index passed to {\@link select}
     *
     * Used with {\@link property} instruction (and more in the future) to identify the index in the
     * current `LView` to act on.
     *
     * (Note that if an "exit function" was set earlier (via `setElementExitFn()`) then that will be
     * run if and when the provided `index` value is different from the current selected index value.)
     * @param {?} index
     * @return {?}
     */
    function setSelectedIndex(index) {
        instructionState.lFrame.selectedIndex = index;
    }
    /**
     * Gets the `tNode` that represents currently selected element.
     * @return {?}
     */
    function getSelectedTNode() {
        /** @type {?} */
        const lFrame = instructionState.lFrame;
        return getTNode(lFrame.tView, lFrame.selectedIndex);
    }
    /**
     * Sets the namespace used to create elements to `'http://www.w3.org/2000/svg'` in global state.
     *
     * \@codeGenApi
     * @return {?}
     */
    function ɵɵnamespaceSVG() {
        instructionState.lFrame.currentNamespace = SVG_NAMESPACE;
    }
    /**
     * Sets the namespace used to create elements to `'http://www.w3.org/1998/MathML/'` in global state.
     *
     * \@codeGenApi
     * @return {?}
     */
    function ɵɵnamespaceMathML() {
        instructionState.lFrame.currentNamespace = MATH_ML_NAMESPACE;
    }
    /**
     * Sets the namespace used to create elements to `null`, which forces element creation to use
     * `createElement` rather than `createElementNS`.
     *
     * \@codeGenApi
     * @return {?}
     */
    function ɵɵnamespaceHTML() {
        namespaceHTMLInternal();
    }
    /**
     * Sets the namespace used to create elements to `null`, which forces element creation to use
     * `createElement` rather than `createElementNS`.
     * @return {?}
     */
    function namespaceHTMLInternal() {
        instructionState.lFrame.currentNamespace = null;
    }
    /**
     * @return {?}
     */
    function getNamespace() {
        return instructionState.lFrame.currentNamespace;
    }
    /**
     * @param {?} sanitizer
     * @return {?}
     */
    function setCurrentStyleSanitizer(sanitizer) {
        instructionState.lFrame.currentSanitizer = sanitizer;
    }
    /**
     * @return {?}
     */
    function getCurrentStyleSanitizer() {
        // TODO(misko): This should throw when there is no LView, but it turns out we can get here from
        // `NodeStyleDebug` hence we return `null`. This should be fixed
        /** @type {?} */
        const lFrame = instructionState.lFrame;
        return lFrame === null ? null : lFrame.currentSanitizer;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/hooks.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Adds all directive lifecycle hooks from the given `DirectiveDef` to the given `TView`.
     *
     * Must be run *only* on the first template pass.
     *
     * Sets up the pre-order hooks on the provided `tView`,
     * see {\@link HookData} for details about the data structure.
     *
     * @param {?} directiveIndex The index of the directive in LView
     * @param {?} directiveDef The definition containing the hooks to setup in tView
     * @param {?} tView The current TView
     * @return {?}
     */
    function registerPreOrderHooks(directiveIndex, directiveDef, tView) {
        ngDevMode && assertFirstCreatePass(tView);
        const { onChanges, onInit, doCheck } = directiveDef;
        if (onChanges) {
            (tView.preOrderHooks || (tView.preOrderHooks = [])).push(directiveIndex, onChanges);
            (tView.preOrderCheckHooks || (tView.preOrderCheckHooks = [])).push(directiveIndex, onChanges);
        }
        if (onInit) {
            (tView.preOrderHooks || (tView.preOrderHooks = [])).push(-directiveIndex, onInit);
        }
        if (doCheck) {
            (tView.preOrderHooks || (tView.preOrderHooks = [])).push(directiveIndex, doCheck);
            (tView.preOrderCheckHooks || (tView.preOrderCheckHooks = [])).push(directiveIndex, doCheck);
        }
    }
    /**
     *
     * Loops through the directives on the provided `tNode` and queues hooks to be
     * run that are not initialization hooks.
     *
     * Should be executed during `elementEnd()` and similar to
     * preserve hook execution order. Content, view, and destroy hooks for projected
     * components and directives must be called *before* their hosts.
     *
     * Sets up the content, view, and destroy hooks on the provided `tView`,
     * see {\@link HookData} for details about the data structure.
     *
     * NOTE: This does not set up `onChanges`, `onInit` or `doCheck`, those are set up
     * separately at `elementStart`.
     *
     * @param {?} tView The current TView
     * @param {?} tNode The TNode whose directives are to be searched for hooks to queue
     * @return {?}
     */
    function registerPostOrderHooks(tView, tNode) {
        ngDevMode && assertFirstCreatePass(tView);
        // It's necessary to loop through the directives at elementEnd() (rather than processing in
        // directiveCreate) so we can preserve the current hook order. Content, view, and destroy
        // hooks for projected components and directives must be called *before* their hosts.
        for (let i = tNode.directiveStart, end = tNode.directiveEnd; i < end; i++) {
            /** @type {?} */
            const directiveDef = (/** @type {?} */ (tView.data[i]));
            if (directiveDef.afterContentInit) {
                (tView.contentHooks || (tView.contentHooks = [])).push(-i, directiveDef.afterContentInit);
            }
            if (directiveDef.afterContentChecked) {
                (tView.contentHooks || (tView.contentHooks = [])).push(i, directiveDef.afterContentChecked);
                (tView.contentCheckHooks || (tView.contentCheckHooks = []))
                    .push(i, directiveDef.afterContentChecked);
            }
            if (directiveDef.afterViewInit) {
                (tView.viewHooks || (tView.viewHooks = [])).push(-i, directiveDef.afterViewInit);
            }
            if (directiveDef.afterViewChecked) {
                (tView.viewHooks || (tView.viewHooks = [])).push(i, directiveDef.afterViewChecked);
                (tView.viewCheckHooks || (tView.viewCheckHooks = [])).push(i, directiveDef.afterViewChecked);
            }
            if (directiveDef.onDestroy != null) {
                (tView.destroyHooks || (tView.destroyHooks = [])).push(i, directiveDef.onDestroy);
            }
        }
    }
    /**
     * Executing hooks requires complex logic as we need to deal with 2 constraints.
     *
     * 1. Init hooks (ngOnInit, ngAfterContentInit, ngAfterViewInit) must all be executed once and only
     * once, across many change detection cycles. This must be true even if some hooks throw, or if
     * some recursively trigger a change detection cycle.
     * To solve that, it is required to track the state of the execution of these init hooks.
     * This is done by storing and maintaining flags in the view: the {@link InitPhaseState},
     * and the index within that phase. They can be seen as a cursor in the following structure:
     * [[onInit1, onInit2], [afterContentInit1], [afterViewInit1, afterViewInit2, afterViewInit3]]
     * They are are stored as flags in LView[FLAGS].
     *
     * 2. Pre-order hooks can be executed in batches, because of the select instruction.
     * To be able to pause and resume their execution, we also need some state about the hook's array
     * that is being processed:
     * - the index of the next hook to be executed
     * - the number of init hooks already found in the processed part of the  array
     * They are are stored as flags in LView[PREORDER_HOOK_FLAGS].
     */
    /**
     * Executes pre-order check hooks ( OnChanges, DoChanges) given a view where all the init hooks were
     * executed once. This is a light version of executeInitAndCheckPreOrderHooks where we can skip read
     * / write of the init-hooks related flags.
     * @param {?} lView The LView where hooks are defined
     * @param {?} hooks Hooks to be run
     * @param {?=} nodeIndex 3 cases depending on the value:
     * - undefined: all hooks from the array should be executed (post-order case)
     * - null: execute hooks only from the saved index until the end of the array (pre-order case, when
     * flushing the remaining hooks)
     * - number: execute hooks only from the saved index until that node index exclusive (pre-order
     * case, when executing select(number))
     * @return {?}
     */
    function executeCheckHooks(lView, hooks, nodeIndex) {
        callHooks(lView, hooks, 3 /* InitPhaseCompleted */, nodeIndex);
    }
    /**
     * Executes post-order init and check hooks (one of AfterContentInit, AfterContentChecked,
     * AfterViewInit, AfterViewChecked) given a view where there are pending init hooks to be executed.
     * @param {?} lView The LView where hooks are defined
     * @param {?} hooks Hooks to be run
     * @param {?} initPhase A phase for which hooks should be run
     * @param {?=} nodeIndex 3 cases depending on the value:
     * - undefined: all hooks from the array should be executed (post-order case)
     * - null: execute hooks only from the saved index until the end of the array (pre-order case, when
     * flushing the remaining hooks)
     * - number: execute hooks only from the saved index until that node index exclusive (pre-order
     * case, when executing select(number))
     * @return {?}
     */
    function executeInitAndCheckHooks(lView, hooks, initPhase, nodeIndex) {
        ngDevMode &&
            assertNotEqual(initPhase, 3 /* InitPhaseCompleted */, 'Init pre-order hooks should not be called more than once');
        if ((lView[FLAGS] & 3 /* InitPhaseStateMask */) === initPhase) {
            callHooks(lView, hooks, initPhase, nodeIndex);
        }
    }
    /**
     * @param {?} lView
     * @param {?} initPhase
     * @return {?}
     */
    function incrementInitPhaseFlags(lView, initPhase) {
        ngDevMode &&
            assertNotEqual(initPhase, 3 /* InitPhaseCompleted */, 'Init hooks phase should not be incremented after all init hooks have been run.');
        /** @type {?} */
        let flags = lView[FLAGS];
        if ((flags & 3 /* InitPhaseStateMask */) === initPhase) {
            flags &= 2047 /* IndexWithinInitPhaseReset */;
            flags += 1 /* InitPhaseStateIncrementer */;
            lView[FLAGS] = flags;
        }
    }
    /**
     * Calls lifecycle hooks with their contexts, skipping init hooks if it's not
     * the first LView pass
     *
     * @param {?} currentView The current view
     * @param {?} arr The array in which the hooks are found
     * @param {?} initPhase
     * @param {?} currentNodeIndex 3 cases depending on the value:
     * - undefined: all hooks from the array should be executed (post-order case)
     * - null: execute hooks only from the saved index until the end of the array (pre-order case, when
     * flushing the remaining hooks)
     * - number: execute hooks only from the saved index until that node index exclusive (pre-order
     * case, when executing select(number))
     * @return {?}
     */
    function callHooks(currentView, arr, initPhase, currentNodeIndex) {
        ngDevMode &&
            assertEqual(getCheckNoChangesMode(), false, 'Hooks should never be run in the check no changes mode.');
        /** @type {?} */
        const startIndex = currentNodeIndex !== undefined ?
            (currentView[PREORDER_HOOK_FLAGS] & 65535 /* IndexOfTheNextPreOrderHookMaskMask */) :
            0;
        /** @type {?} */
        const nodeIndexLimit = currentNodeIndex != null ? currentNodeIndex : -1;
        /** @type {?} */
        let lastNodeIndexFound = 0;
        for (let i = startIndex; i < arr.length; i++) {
            /** @type {?} */
            const hook = (/** @type {?} */ (arr[i + 1]));
            if (typeof hook === 'number') {
                lastNodeIndexFound = (/** @type {?} */ (arr[i]));
                if (currentNodeIndex != null && lastNodeIndexFound >= currentNodeIndex) {
                    break;
                }
            }
            else {
                /** @type {?} */
                const isInitHook = arr[i] < 0;
                if (isInitHook)
                    currentView[PREORDER_HOOK_FLAGS] += 65536 /* NumberOfInitHooksCalledIncrementer */;
                if (lastNodeIndexFound < nodeIndexLimit || nodeIndexLimit == -1) {
                    callHook(currentView, initPhase, arr, i);
                    currentView[PREORDER_HOOK_FLAGS] =
                        (currentView[PREORDER_HOOK_FLAGS] & 4294901760 /* NumberOfInitHooksCalledMask */) + i +
                            2;
                }
                i++;
            }
        }
    }
    /**
     * Execute one hook against the current `LView`.
     *
     * @param {?} currentView The current view
     * @param {?} initPhase
     * @param {?} arr The array in which the hooks are found
     * @param {?} i The current index within the hook data array
     * @return {?}
     */
    function callHook(currentView, initPhase, arr, i) {
        /** @type {?} */
        const isInitHook = arr[i] < 0;
        /** @type {?} */
        const hook = (/** @type {?} */ (arr[i + 1]));
        /** @type {?} */
        const directiveIndex = isInitHook ? -arr[i] : (/** @type {?} */ (arr[i]));
        /** @type {?} */
        const directive = currentView[directiveIndex];
        if (isInitHook) {
            /** @type {?} */
            const indexWithintInitPhase = currentView[FLAGS] >> 11 /* IndexWithinInitPhaseShift */;
            // The init phase state must be always checked here as it may have been recursively
            // updated
            if (indexWithintInitPhase <
                (currentView[PREORDER_HOOK_FLAGS] >> 16 /* NumberOfInitHooksCalledShift */) &&
                (currentView[FLAGS] & 3 /* InitPhaseStateMask */) === initPhase) {
                currentView[FLAGS] += 2048 /* IndexWithinInitPhaseIncrementer */;
                hook.call(directive);
            }
        }
        else {
            hook.call(directive);
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/interfaces/injector.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /** @type {?} */
    const TNODE = 8;
    /** @type {?} */
    const PARENT_INJECTOR = 8;
    /** @type {?} */
    const INJECTOR_BLOOM_PARENT_SIZE = 9;
    /** @type {?} */
    const NO_PARENT_INJECTOR = (/** @type {?} */ (-1));
    /**
     * Each injector is saved in 9 contiguous slots in `LView` and 9 contiguous slots in
     * `TView.data`. This allows us to store information about the current node's tokens (which
     * can be shared in `TView`) as well as the tokens of its ancestor nodes (which cannot be
     * shared, so they live in `LView`).
     *
     * Each of these slots (aside from the last slot) contains a bloom filter. This bloom filter
     * determines whether a directive is available on the associated node or not. This prevents us
     * from searching the directives array at this level unless it's probable the directive is in it.
     *
     * See: https://en.wikipedia.org/wiki/Bloom_filter for more about bloom filters.
     *
     * Because all injectors have been flattened into `LView` and `TViewData`, they cannot typed
     * using interfaces as they were previously. The start index of each `LInjector` and `TInjector`
     * will differ based on where it is flattened into the main array, so it's not possible to know
     * the indices ahead of time and save their types here. The interfaces are still included here
     * for documentation purposes.
     *
     * export interface LInjector extends Array<any> {
     *
     *    // Cumulative bloom for directive IDs 0-31  (IDs are % BLOOM_SIZE)
     *    [0]: number;
     *
     *    // Cumulative bloom for directive IDs 32-63
     *    [1]: number;
     *
     *    // Cumulative bloom for directive IDs 64-95
     *    [2]: number;
     *
     *    // Cumulative bloom for directive IDs 96-127
     *    [3]: number;
     *
     *    // Cumulative bloom for directive IDs 128-159
     *    [4]: number;
     *
     *    // Cumulative bloom for directive IDs 160 - 191
     *    [5]: number;
     *
     *    // Cumulative bloom for directive IDs 192 - 223
     *    [6]: number;
     *
     *    // Cumulative bloom for directive IDs 224 - 255
     *    [7]: number;
     *
     *    // We need to store a reference to the injector's parent so DI can keep looking up
     *    // the injector tree until it finds the dependency it's looking for.
     *    [PARENT_INJECTOR]: number;
     * }
     *
     * export interface TInjector extends Array<any> {
     *
     *    // Shared node bloom for directive IDs 0-31  (IDs are % BLOOM_SIZE)
     *    [0]: number;
     *
     *    // Shared node bloom for directive IDs 32-63
     *    [1]: number;
     *
     *    // Shared node bloom for directive IDs 64-95
     *    [2]: number;
     *
     *    // Shared node bloom for directive IDs 96-127
     *    [3]: number;
     *
     *    // Shared node bloom for directive IDs 128-159
     *    [4]: number;
     *
     *    // Shared node bloom for directive IDs 160 - 191
     *    [5]: number;
     *
     *    // Shared node bloom for directive IDs 192 - 223
     *    [6]: number;
     *
     *    // Shared node bloom for directive IDs 224 - 255
     *    [7]: number;
     *
     *    // Necessary to find directive indices for a particular node.
     *    [TNODE]: TElementNode|TElementContainerNode|TContainerNode;
     *  }
     */
    /**
     * Factory for creating instances of injectors in the NodeInjector.
     *
     * This factory is complicated by the fact that it can resolve `multi` factories as well.
     *
     * NOTE: Some of the fields are optional which means that this class has two hidden classes.
     * - One without `multi` support (most common)
     * - One with `multi` values, (rare).
     *
     * Since VMs can cache up to 4 inline hidden classes this is OK.
     *
     * - Single factory: Only `resolving` and `factory` is defined.
     * - `providers` factory: `componentProviders` is a number and `index = -1`.
     * - `viewProviders` factory: `componentProviders` is a number and `index` points to `providers`.
     */
    class NodeInjectorFactory {
        /**
         * @param {?} factory
         * @param {?} isViewProvider
         * @param {?} injectImplementation
         */
        constructor(factory, 
        /**
         * Set to `true` if the token is declared in `viewProviders` (or if it is component).
         */
        isViewProvider, injectImplementation) {
            this.factory = factory;
            /**
             * Marker set to true during factory invocation to see if we get into recursive loop.
             * Recursive loop causes an error to be displayed.
             */
            this.resolving = false;
            this.canSeeViewProviders = isViewProvider;
            this.injectImpl = injectImplementation;
        }
    }
    /**
     * @param {?} obj
     * @return {?}
     */
    function isFactory(obj) {
        return obj instanceof NodeInjectorFactory;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/node_assert.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @param {?} tNode
     * @param {?} type
     * @return {?}
     */
    function assertNodeType(tNode, type) {
        assertDefined(tNode, 'should be called with a TNode');
        assertEqual(tNode.type, type, `should be a ${typeName(type)}`);
    }
    /**
     * @param {?} tNode
     * @param {...?} types
     * @return {?}
     */
    function assertNodeOfPossibleTypes(tNode, ...types) {
        assertDefined(tNode, 'should be called with a TNode');
        /** @type {?} */
        const found = types.some((/**
         * @param {?} type
         * @return {?}
         */
        type => tNode.type === type));
        assertEqual(found, true, `Should be one of ${types.map(typeName).join(', ')} but got ${typeName(tNode.type)}`);
    }
    /**
     * @param {?} type
     * @return {?}
     */
    function typeName(type) {
        if (type == 1 /* Projection */)
            return 'Projection';
        if (type == 0 /* Container */)
            return 'Container';
        if (type == 5 /* IcuContainer */)
            return 'IcuContainer';
        if (type == 2 /* View */)
            return 'View';
        if (type == 3 /* Element */)
            return 'Element';
        if (type == 4 /* ElementContainer */)
            return 'ElementContainer';
        return '<unknown>';
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/util/attrs_utils.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Assigns all attribute values to the provided element via the inferred renderer.
     *
     * This function accepts two forms of attribute entries:
     *
     * default: (key, value):
     *  attrs = [key1, value1, key2, value2]
     *
     * namespaced: (NAMESPACE_MARKER, uri, name, value)
     *  attrs = [NAMESPACE_MARKER, uri, name, value, NAMESPACE_MARKER, uri, name, value]
     *
     * The `attrs` array can contain a mix of both the default and namespaced entries.
     * The "default" values are set without a marker, but if the function comes across
     * a marker value then it will attempt to set a namespaced value. If the marker is
     * not of a namespaced value then the function will quit and return the index value
     * where it stopped during the iteration of the attrs array.
     *
     * See [AttributeMarker] to understand what the namespace marker value is.
     *
     * Note that this instruction does not support assigning style and class values to
     * an element. See `elementStart` and `elementHostAttrs` to learn how styling values
     * are applied to an element.
     * @param {?} renderer The renderer to be used
     * @param {?} native The element that the attributes will be assigned to
     * @param {?} attrs The attribute array of values that will be assigned to the element
     * @return {?} the index value that was last accessed in the attributes array
     */
    function setUpAttributes(renderer, native, attrs) {
        /** @type {?} */
        const isProc = isProceduralRenderer(renderer);
        /** @type {?} */
        let i = 0;
        while (i < attrs.length) {
            /** @type {?} */
            const value = attrs[i];
            if (typeof value === 'number') {
                // only namespaces are supported. Other value types (such as style/class
                // entries) are not supported in this function.
                if (value !== 0 /* NamespaceURI */) {
                    break;
                }
                // we just landed on the marker value ... therefore
                // we should skip to the next entry
                i++;
                /** @type {?} */
                const namespaceURI = (/** @type {?} */ (attrs[i++]));
                /** @type {?} */
                const attrName = (/** @type {?} */ (attrs[i++]));
                /** @type {?} */
                const attrVal = (/** @type {?} */ (attrs[i++]));
                ngDevMode && ngDevMode.rendererSetAttribute++;
                isProc ?
                    ((/** @type {?} */ (renderer))).setAttribute(native, attrName, attrVal, namespaceURI) :
                    native.setAttributeNS(namespaceURI, attrName, attrVal);
            }
            else {
                // attrName is string;
                /** @type {?} */
                const attrName = (/** @type {?} */ (value));
                /** @type {?} */
                const attrVal = attrs[++i];
                // Standard attributes
                ngDevMode && ngDevMode.rendererSetAttribute++;
                if (isAnimationProp(attrName)) {
                    if (isProc) {
                        ((/** @type {?} */ (renderer))).setProperty(native, attrName, attrVal);
                    }
                }
                else {
                    isProc ?
                        ((/** @type {?} */ (renderer))).setAttribute(native, attrName, (/** @type {?} */ (attrVal))) :
                        native.setAttribute(attrName, (/** @type {?} */ (attrVal)));
                }
                i++;
            }
        }
        // another piece of code may iterate over the same attributes array. Therefore
        // it may be helpful to return the exact spot where the attributes array exited
        // whether by running into an unsupported marker or if all the static values were
        // iterated over.
        return i;
    }
    /**
     * Test whether the given value is a marker that indicates that the following
     * attribute values in a `TAttributes` array are only the names of attributes,
     * and not name-value pairs.
     * @param {?} marker The attribute marker to test.
     * @return {?} true if the marker is a "name-only" marker (e.g. `Bindings`, `Template` or `I18n`).
     */
    function isNameOnlyAttributeMarker(marker) {
        return marker === 3 /* Bindings */ || marker === 4 /* Template */ ||
            marker === 6 /* I18n */;
    }
    /**
     * @param {?} name
     * @return {?}
     */
    function isAnimationProp(name) {
        // Perf note: accessing charCodeAt to check for the first character of a string is faster as
        // compared to accessing a character at index 0 (ex. name[0]). The main reason for this is that
        // charCodeAt doesn't allocate memory to return a substring.
        return name.charCodeAt(0) === 64 /* AT_SIGN */;
    }
    /**
     * Merges `src` `TAttributes` into `dst` `TAttributes` removing any duplicates in the process.
     *
     * This merge function keeps the order of attrs same.
     *
     * @param {?} dst Location of where the merged `TAttributes` should end up.
     * @param {?} src `TAttributes` which should be appended to `dst`
     * @return {?}
     */
    function mergeHostAttrs(dst, src) {
        if (src === null || src.length === 0) ;
        else if (dst === null || dst.length === 0) {
            // We have source, but dst is empty, just make a copy.
            dst = src.slice();
        }
        else {
            /** @type {?} */
            let srcMarker = -1 /* ImplicitAttributes */;
            for (let i = 0; i < src.length; i++) {
                /** @type {?} */
                const item = src[i];
                if (typeof item === 'number') {
                    srcMarker = item;
                }
                else {
                    if (srcMarker === 0 /* NamespaceURI */) ;
                    else if (srcMarker === -1 /* ImplicitAttributes */ ||
                        srcMarker === 2 /* Styles */) {
                        // Case where we have to consume `key1` and `value` only.
                        mergeHostAttribute(dst, srcMarker, (/** @type {?} */ (item)), null, (/** @type {?} */ (src[++i])));
                    }
                    else {
                        // Case where we have to consume `key1` only.
                        mergeHostAttribute(dst, srcMarker, (/** @type {?} */ (item)), null, null);
                    }
                }
            }
        }
        return dst;
    }
    /**
     * Append `key`/`value` to existing `TAttributes` taking region marker and duplicates into account.
     *
     * @param {?} dst `TAttributes` to append to.
     * @param {?} marker Region where the `key`/`value` should be added.
     * @param {?} key1 Key to add to `TAttributes`
     * @param {?} key2 Key to add to `TAttributes` (in case of `AttributeMarker.NamespaceURI`)
     * @param {?} value Value to add or to overwrite to `TAttributes` Only used if `marker` is not Class.
     * @return {?}
     */
    function mergeHostAttribute(dst, marker, key1, key2, value) {
        /** @type {?} */
        let i = 0;
        // Assume that new markers will be inserted at the end.
        /** @type {?} */
        let markerInsertPosition = dst.length;
        // scan until correct type.
        if (marker === -1 /* ImplicitAttributes */) {
            markerInsertPosition = -1;
        }
        else {
            while (i < dst.length) {
                /** @type {?} */
                const dstValue = dst[i++];
                if (typeof dstValue === 'number') {
                    if (dstValue === marker) {
                        markerInsertPosition = -1;
                        break;
                    }
                    else if (dstValue > marker) {
                        // We need to save this as we want the markers to be inserted in specific order.
                        markerInsertPosition = i - 1;
                        break;
                    }
                }
            }
        }
        // search until you find place of insertion
        while (i < dst.length) {
            /** @type {?} */
            const item = dst[i];
            if (typeof item === 'number') {
                // since `i` started as the index after the marker, we did not find it if we are at the next
                // marker
                break;
            }
            else if (item === key1) {
                // We already have same token
                if (key2 === null) {
                    if (value !== null) {
                        dst[i + 1] = value;
                    }
                    return;
                }
                else if (key2 === dst[i + 1]) {
                    dst[i + 2] = (/** @type {?} */ (value));
                    return;
                }
            }
            // Increment counter.
            i++;
            if (key2 !== null)
                i++;
            if (value !== null)
                i++;
        }
        // insert at location.
        if (markerInsertPosition !== -1) {
            dst.splice(markerInsertPosition, 0, marker);
            i = markerInsertPosition + 1;
        }
        dst.splice(i++, 0, key1);
        if (key2 !== null) {
            dst.splice(i++, 0, key2);
        }
        if (value !== null) {
            dst.splice(i++, 0, value);
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/util/injector_utils.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /// Parent Injector Utils ///////////////////////////////////////////////////////////////
    /**
     * @param {?} parentLocation
     * @return {?}
     */
    function hasParentInjector(parentLocation) {
        return parentLocation !== NO_PARENT_INJECTOR;
    }
    /**
     * @param {?} parentLocation
     * @return {?}
     */
    function getParentInjectorIndex(parentLocation) {
        return ((/** @type {?} */ ((/** @type {?} */ (parentLocation))))) & 32767 /* InjectorIndexMask */;
    }
    /**
     * @param {?} parentLocation
     * @return {?}
     */
    function getParentInjectorViewOffset(parentLocation) {
        return ((/** @type {?} */ ((/** @type {?} */ (parentLocation))))) >> 16 /* ViewOffsetShift */;
    }
    /**
     * Unwraps a parent injector location number to find the view offset from the current injector,
     * then walks up the declaration view tree until the view is found that contains the parent
     * injector.
     *
     * @param {?} location The location of the parent injector, which contains the view offset
     * @param {?} startView The LView instance from which to start walking up the view tree
     * @return {?} The LView instance that contains the parent injector
     */
    function getParentInjectorView(location, startView) {
        /** @type {?} */
        let viewOffset = getParentInjectorViewOffset(location);
        /** @type {?} */
        let parentView = startView;
        // For most cases, the parent injector can be found on the host node (e.g. for component
        // or container), but we must keep the loop here to support the rarer case of deeply nested
        // <ng-template> tags or inline views, where the parent injector might live many views
        // above the child injector.
        while (viewOffset > 0) {
            parentView = (/** @type {?} */ (parentView[DECLARATION_VIEW]));
            viewOffset--;
        }
        return parentView;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/util/misc_utils.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Used for stringify render output in Ivy.
     * Important! This function is very performance-sensitive and we should
     * be extra careful not to introduce megamorphic reads in it.
     * @param {?} value
     * @return {?}
     */
    function renderStringify(value) {
        if (typeof value === 'string')
            return value;
        if (value == null)
            return '';
        return '' + value;
    }
    /**
     * Used to stringify a value so that it can be displayed in an error message.
     * Important! This function contains a megamorphic read and should only be
     * used for error messages.
     * @param {?} value
     * @return {?}
     */
    function stringifyForError(value) {
        if (typeof value === 'function')
            return value.name || value.toString();
        if (typeof value === 'object' && value != null && typeof value.type === 'function') {
            return value.type.name || value.type.toString();
        }
        return renderStringify(value);
    }
    const ɵ0$3 = /**
     * @return {?}
     */
    () => (typeof requestAnimationFrame !== 'undefined' &&
        requestAnimationFrame || // browser only
        setTimeout // everything else
    )
        .bind(_global);
    /** @type {?} */
    const defaultScheduler = ((ɵ0$3))();
    /**
     *
     * \@codeGenApi
     * @param {?} element
     * @return {?}
     */
    function ɵɵresolveWindow(element) {
        return { name: 'window', target: element.ownerDocument.defaultView };
    }
    /**
     *
     * \@codeGenApi
     * @param {?} element
     * @return {?}
     */
    function ɵɵresolveDocument(element) {
        return { name: 'document', target: element.ownerDocument };
    }
    /**
     *
     * \@codeGenApi
     * @param {?} element
     * @return {?}
     */
    function ɵɵresolveBody(element) {
        return { name: 'body', target: element.ownerDocument.body };
    }
    /**
     * The special delimiter we use to separate property names, prefixes, and suffixes
     * in property binding metadata. See storeBindingMetadata().
     *
     * We intentionally use the Unicode "REPLACEMENT CHARACTER" (U+FFFD) as a delimiter
     * because it is a very uncommon character that is unlikely to be part of a user's
     * property names or interpolation strings. If it is in fact used in a property
     * binding, DebugElement.properties will not return the correct value for that
     * binding. However, there should be no runtime effect for real applications.
     *
     * This character is typically rendered as a question mark inside of a diamond.
     * See https://en.wikipedia.org/wiki/Specials_(Unicode_block)
     *
     * @type {?}
     */
    const INTERPOLATION_DELIMITER = `�`;
    /**
     * Unwrap a value which might be behind a closure (for forward declaration reasons).
     * @template T
     * @param {?} value
     * @return {?}
     */
    function maybeUnwrapFn(value) {
        if (value instanceof Function) {
            return value();
        }
        else {
            return value;
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/di.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Defines if the call to `inject` should include `viewProviders` in its resolution.
     *
     * This is set to true when we try to instantiate a component. This value is reset in
     * `getNodeInjectable` to a value which matches the declaration location of the token about to be
     * instantiated. This is done so that if we are injecting a token which was declared outside of
     * `viewProviders` we don't accidentally pull `viewProviders` in.
     *
     * Example:
     *
     * ```
     * \@Injectable()
     * class MyService {
     *   constructor(public value: String) {}
     * }
     *
     * \@Component({
     *   providers: [
     *     MyService,
     *     {provide: String, value: 'providers' }
     *   ]
     *   viewProviders: [
     *     {provide: String, value: 'viewProviders'}
     *   ]
     * })
     * class MyComponent {
     *   constructor(myService: MyService, value: String) {
     *     // We expect that Component can see into `viewProviders`.
     *     expect(value).toEqual('viewProviders');
     *     // `MyService` was not declared in `viewProviders` hence it can't see it.
     *     expect(myService.value).toEqual('providers');
     *   }
     * }
     *
     * ```
     * @type {?}
     */
    let includeViewProviders = true;
    /**
     * @param {?} v
     * @return {?}
     */
    function setIncludeViewProviders(v) {
        /** @type {?} */
        const oldValue = includeViewProviders;
        includeViewProviders = v;
        return oldValue;
    }
    /**
     * The number of slots in each bloom filter (used by DI). The larger this number, the fewer
     * directives that will share slots, and thus, the fewer false positives when checking for
     * the existence of a directive.
     * @type {?}
     */
    const BLOOM_SIZE = 256;
    /** @type {?} */
    const BLOOM_MASK = BLOOM_SIZE - 1;
    /**
     * Counter used to generate unique IDs for directives.
     * @type {?}
     */
    let nextNgElementId = 0;
    /**
     * Registers this directive as present in its node's injector by flipping the directive's
     * corresponding bit in the injector's bloom filter.
     *
     * @param {?} injectorIndex The index of the node injector where this token should be registered
     * @param {?} tView The TView for the injector's bloom filters
     * @param {?} type The directive token to register
     * @return {?}
     */
    function bloomAdd(injectorIndex, tView, type) {
        ngDevMode && assertEqual(tView.firstCreatePass, true, 'expected firstCreatePass to be true');
        /** @type {?} */
        let id = typeof type !== 'string' ? ((/** @type {?} */ (type)))[NG_ELEMENT_ID] : type.charCodeAt(0) || 0;
        // Set a unique ID on the directive type, so if something tries to inject the directive,
        // we can easily retrieve the ID and hash it into the bloom bit that should be checked.
        if (id == null) {
            id = ((/** @type {?} */ (type)))[NG_ELEMENT_ID] = nextNgElementId++;
        }
        // We only have BLOOM_SIZE (256) slots in our bloom filter (8 buckets * 32 bits each),
        // so all unique IDs must be modulo-ed into a number from 0 - 255 to fit into the filter.
        /** @type {?} */
        const bloomBit = id & BLOOM_MASK;
        // Create a mask that targets the specific bit associated with the directive.
        // JS bit operations are 32 bits, so this will be a number between 2^0 and 2^31, corresponding
        // to bit positions 0 - 31 in a 32 bit integer.
        /** @type {?} */
        const mask = 1 << bloomBit;
        // Use the raw bloomBit number to determine which bloom filter bucket we should check
        // e.g: bf0 = [0 - 31], bf1 = [32 - 63], bf2 = [64 - 95], bf3 = [96 - 127], etc
        /** @type {?} */
        const b7 = bloomBit & 0x80;
        /** @type {?} */
        const b6 = bloomBit & 0x40;
        /** @type {?} */
        const b5 = bloomBit & 0x20;
        /** @type {?} */
        const tData = (/** @type {?} */ (tView.data));
        if (b7) {
            b6 ? (b5 ? (tData[injectorIndex + 7] |= mask) : (tData[injectorIndex + 6] |= mask)) :
                (b5 ? (tData[injectorIndex + 5] |= mask) : (tData[injectorIndex + 4] |= mask));
        }
        else {
            b6 ? (b5 ? (tData[injectorIndex + 3] |= mask) : (tData[injectorIndex + 2] |= mask)) :
                (b5 ? (tData[injectorIndex + 1] |= mask) : (tData[injectorIndex] |= mask));
        }
    }
    /**
     * Creates (or gets an existing) injector for a given element or container.
     *
     * @param {?} tNode for which an injector should be retrieved / created.
     * @param {?} hostView View where the node is stored
     * @return {?} Node injector
     */
    function getOrCreateNodeInjectorForNode(tNode, hostView) {
        /** @type {?} */
        const existingInjectorIndex = getInjectorIndex(tNode, hostView);
        if (existingInjectorIndex !== -1) {
            return existingInjectorIndex;
        }
        /** @type {?} */
        const tView = hostView[TVIEW];
        if (tView.firstCreatePass) {
            tNode.injectorIndex = hostView.length;
            insertBloom(tView.data, tNode); // foundation for node bloom
            insertBloom(hostView, null); // foundation for cumulative bloom
            insertBloom(tView.blueprint, null);
        }
        /** @type {?} */
        const parentLoc = getParentInjectorLocation(tNode, hostView);
        /** @type {?} */
        const injectorIndex = tNode.injectorIndex;
        // If a parent injector can't be found, its location is set to -1.
        // In that case, we don't need to set up a cumulative bloom
        if (hasParentInjector(parentLoc)) {
            /** @type {?} */
            const parentIndex = getParentInjectorIndex(parentLoc);
            /** @type {?} */
            const parentLView = getParentInjectorView(parentLoc, hostView);
            /** @type {?} */
            const parentData = (/** @type {?} */ (parentLView[TVIEW].data));
            // Creates a cumulative bloom filter that merges the parent's bloom filter
            // and its own cumulative bloom (which contains tokens for all ancestors)
            for (let i = 0; i < 8; i++) {
                hostView[injectorIndex + i] = parentLView[parentIndex + i] | parentData[parentIndex + i];
            }
        }
        hostView[injectorIndex + PARENT_INJECTOR] = parentLoc;
        return injectorIndex;
    }
    /**
     * @param {?} arr
     * @param {?} footer
     * @return {?}
     */
    function insertBloom(arr, footer) {
        arr.push(0, 0, 0, 0, 0, 0, 0, 0, footer);
    }
    /**
     * @param {?} tNode
     * @param {?} hostView
     * @return {?}
     */
    function getInjectorIndex(tNode, hostView) {
        if (tNode.injectorIndex === -1 ||
            // If the injector index is the same as its parent's injector index, then the index has been
            // copied down from the parent node. No injector has been created yet on this node.
            (tNode.parent && tNode.parent.injectorIndex === tNode.injectorIndex) ||
            // After the first template pass, the injector index might exist but the parent values
            // might not have been calculated yet for this instance
            hostView[tNode.injectorIndex + PARENT_INJECTOR] == null) {
            return -1;
        }
        else {
            return tNode.injectorIndex;
        }
    }
    /**
     * Finds the index of the parent injector, with a view offset if applicable. Used to set the
     * parent injector initially.
     *
     * Returns a combination of number of `ViewData` we have to go up and index in that `Viewdata`
     * @param {?} tNode
     * @param {?} view
     * @return {?}
     */
    function getParentInjectorLocation(tNode, view) {
        if (tNode.parent && tNode.parent.injectorIndex !== -1) {
            return (/** @type {?} */ (tNode.parent.injectorIndex)); // ViewOffset is 0
        }
        // For most cases, the parent injector index can be found on the host node (e.g. for component
        // or container), so this loop will be skipped, but we must keep the loop here to support
        // the rarer case of deeply nested <ng-template> tags or inline views.
        /** @type {?} */
        let hostTNode = view[T_HOST];
        /** @type {?} */
        let viewOffset = 1;
        while (hostTNode && hostTNode.injectorIndex === -1) {
            view = (/** @type {?} */ (view[DECLARATION_VIEW]));
            hostTNode = view ? view[T_HOST] : null;
            viewOffset++;
        }
        return hostTNode ?
            hostTNode.injectorIndex | (viewOffset << 16 /* ViewOffsetShift */) :
            (/** @type {?} */ (-1));
    }
    /**
     * Makes a type or an injection token public to the DI system by adding it to an
     * injector's bloom filter.
     *
     * @param {?} injectorIndex
     * @param {?} tView
     * @param {?} token The type or the injection token to be made public
     * @return {?}
     */
    function diPublicInInjector(injectorIndex, tView, token) {
        bloomAdd(injectorIndex, tView, token);
    }
    /**
     * Inject static attribute value into directive constructor.
     *
     * This method is used with `factory` functions which are generated as part of
     * `defineDirective` or `defineComponent`. The method retrieves the static value
     * of an attribute. (Dynamic attributes are not supported since they are not resolved
     *  at the time of injection and can change over time.)
     *
     * # Example
     * Given:
     * ```
     * \@Component(...)
     * class MyComponent {
     *   constructor(\@Attribute('title') title: string) { ... }
     * }
     * ```
     * When instantiated with
     * ```
     * <my-component title="Hello"></my-component>
     * ```
     *
     * Then factory method generated is:
     * ```
     * MyComponent.ɵcmp = defineComponent({
     *   factory: () => new MyComponent(injectAttribute('title'))
     *   ...
     * })
     * ```
     *
     * \@publicApi
     * @param {?} tNode
     * @param {?} attrNameToInject
     * @return {?}
     */
    function injectAttributeImpl(tNode, attrNameToInject) {
        ngDevMode &&
            assertNodeOfPossibleTypes(tNode, 0 /* Container */, 3 /* Element */, 4 /* ElementContainer */);
        ngDevMode && assertDefined(tNode, 'expecting tNode');
        if (attrNameToInject === 'class') {
            return tNode.classes;
        }
        if (attrNameToInject === 'style') {
            return tNode.styles;
        }
        /** @type {?} */
        const attrs = tNode.attrs;
        if (attrs) {
            /** @type {?} */
            const attrsLength = attrs.length;
            /** @type {?} */
            let i = 0;
            while (i < attrsLength) {
                /** @type {?} */
                const value = attrs[i];
                // If we hit a `Bindings` or `Template` marker then we are done.
                if (isNameOnlyAttributeMarker(value))
                    break;
                // Skip namespaced attributes
                if (value === 0 /* NamespaceURI */) {
                    // we skip the next two values
                    // as namespaced attributes looks like
                    // [..., AttributeMarker.NamespaceURI, 'http://someuri.com/test', 'test:exist',
                    // 'existValue', ...]
                    i = i + 2;
                }
                else if (typeof value === 'number') {
                    // Skip to the first value of the marked attribute.
                    i++;
                    while (i < attrsLength && typeof attrs[i] === 'string') {
                        i++;
                    }
                }
                else if (value === attrNameToInject) {
                    return (/** @type {?} */ (attrs[i + 1]));
                }
                else {
                    i = i + 2;
                }
            }
        }
        return null;
    }
    /**
     * Returns the value associated to the given token from the NodeInjectors => ModuleInjector.
     *
     * Look for the injector providing the token by walking up the node injector tree and then
     * the module injector tree.
     *
     * This function patches `token` with `__NG_ELEMENT_ID__` which contains the id for the bloom
     * filter. Negative values are reserved for special objects.
     *   - `-1` is reserved for injecting `Injector` (implemented by `NodeInjector`)
     *
     * @template T
     * @param {?} tNode The Node where the search for the injector should start
     * @param {?} lView The `LView` that contains the `tNode`
     * @param {?} token The token to look for
     * @param {?=} flags Injection flags
     * @param {?=} notFoundValue The value to return when the injection flags is `InjectFlags.Optional`
     * @return {?} the value from the injector, `null` when not found, or `notFoundValue` if provided
     */
    function getOrCreateInjectable(tNode, lView, token, flags = InjectFlags.Default, notFoundValue) {
        if (tNode !== null) {
            /** @type {?} */
            const bloomHash = bloomHashBitOrFactory(token);
            // If the ID stored here is a function, this is a special object like ElementRef or TemplateRef
            // so just call the factory function to create it.
            if (typeof bloomHash === 'function') {
                enterDI(lView, tNode);
                try {
                    /** @type {?} */
                    const value = bloomHash();
                    if (value == null && !(flags & InjectFlags.Optional)) {
                        throw new Error(`No provider for ${stringifyForError(token)}!`);
                    }
                    else {
                        return value;
                    }
                }
                finally {
                    leaveDI();
                }
            }
            else if (typeof bloomHash == 'number') {
                if (bloomHash === -1) {
                    // `-1` is a special value used to identify `Injector` types.
                    return (/** @type {?} */ (new NodeInjector(tNode, lView)));
                }
                // If the token has a bloom hash, then it is a token which could be in NodeInjector.
                // A reference to the previous injector TView that was found while climbing the element
                // injector tree. This is used to know if viewProviders can be accessed on the current
                // injector.
                /** @type {?} */
                let previousTView = null;
                /** @type {?} */
                let injectorIndex = getInjectorIndex(tNode, lView);
                /** @type {?} */
                let parentLocation = NO_PARENT_INJECTOR;
                /** @type {?} */
                let hostTElementNode = flags & InjectFlags.Host ? lView[DECLARATION_COMPONENT_VIEW][T_HOST] : null;
                // If we should skip this injector, or if there is no injector on this node, start by
                // searching
                // the parent injector.
                if (injectorIndex === -1 || flags & InjectFlags.SkipSelf) {
                    parentLocation = injectorIndex === -1 ? getParentInjectorLocation(tNode, lView) :
                        lView[injectorIndex + PARENT_INJECTOR];
                    if (!shouldSearchParent(flags, false)) {
                        injectorIndex = -1;
                    }
                    else {
                        previousTView = lView[TVIEW];
                        injectorIndex = getParentInjectorIndex(parentLocation);
                        lView = getParentInjectorView(parentLocation, lView);
                    }
                }
                // Traverse up the injector tree until we find a potential match or until we know there
                // *isn't* a match.
                while (injectorIndex !== -1) {
                    parentLocation = lView[injectorIndex + PARENT_INJECTOR];
                    // Check the current injector. If it matches, see if it contains token.
                    /** @type {?} */
                    const tView = lView[TVIEW];
                    if (bloomHasToken(bloomHash, injectorIndex, tView.data)) {
                        // At this point, we have an injector which *may* contain the token, so we step through
                        // the providers and directives associated with the injector's corresponding node to get
                        // the instance.
                        /** @type {?} */
                        const instance = searchTokensOnInjector(injectorIndex, lView, token, previousTView, flags, hostTElementNode);
                        if (instance !== NOT_FOUND) {
                            return instance;
                        }
                    }
                    if (shouldSearchParent(flags, lView[TVIEW].data[injectorIndex + TNODE] === hostTElementNode) &&
                        bloomHasToken(bloomHash, injectorIndex, lView)) {
                        // The def wasn't found anywhere on this node, so it was a false positive.
                        // Traverse up the tree and continue searching.
                        previousTView = tView;
                        injectorIndex = getParentInjectorIndex(parentLocation);
                        lView = getParentInjectorView(parentLocation, lView);
                    }
                    else {
                        // If we should not search parent OR If the ancestor bloom filter value does not have the
                        // bit corresponding to the directive we can give up on traversing up to find the specific
                        // injector.
                        injectorIndex = -1;
                    }
                }
            }
        }
        if (flags & InjectFlags.Optional && notFoundValue === undefined) {
            // This must be set or the NullInjector will throw for optional deps
            notFoundValue = null;
        }
        if ((flags & (InjectFlags.Self | InjectFlags.Host)) === 0) {
            /** @type {?} */
            const moduleInjector = lView[INJECTOR$1];
            // switch to `injectInjectorOnly` implementation for module injector, since module injector
            // should not have access to Component/Directive DI scope (that may happen through
            // `directiveInject` implementation)
            /** @type {?} */
            const previousInjectImplementation = setInjectImplementation(undefined);
            try {
                if (moduleInjector) {
                    return moduleInjector.get(token, notFoundValue, flags & InjectFlags.Optional);
                }
                else {
                    return injectRootLimpMode(token, notFoundValue, flags & InjectFlags.Optional);
                }
            }
            finally {
                setInjectImplementation(previousInjectImplementation);
            }
        }
        if (flags & InjectFlags.Optional) {
            return notFoundValue;
        }
        else {
            throw new Error(`NodeInjector: NOT_FOUND [${stringifyForError(token)}]`);
        }
    }
    /** @type {?} */
    const NOT_FOUND = {};
    /**
     * @template T
     * @param {?} injectorIndex
     * @param {?} lView
     * @param {?} token
     * @param {?} previousTView
     * @param {?} flags
     * @param {?} hostTElementNode
     * @return {?}
     */
    function searchTokensOnInjector(injectorIndex, lView, token, previousTView, flags, hostTElementNode) {
        /** @type {?} */
        const currentTView = lView[TVIEW];
        /** @type {?} */
        const tNode = (/** @type {?} */ (currentTView.data[injectorIndex + TNODE]));
        // First, we need to determine if view providers can be accessed by the starting element.
        // There are two possibities
        /** @type {?} */
        const canAccessViewProviders = previousTView == null ?
            // 1) This is the first invocation `previousTView == null` which means that we are at the
            // `TNode` of where injector is starting to look. In such a case the only time we are allowed
            // to look into the ViewProviders is if:
            // - we are on a component
            // - AND the injector set `includeViewProviders` to true (implying that the token can see
            // ViewProviders because it is the Component or a Service which itself was declared in
            // ViewProviders)
            (isComponentHost(tNode) && includeViewProviders) :
            // 2) `previousTView != null` which means that we are now walking across the parent nodes.
            // In such a case we are only allowed to look into the ViewProviders if:
            // - We just crossed from child View to Parent View `previousTView != currentTView`
            // - AND the parent TNode is an Element.
            // This means that we just came from the Component's View and therefore are allowed to see
            // into the ViewProviders.
            (previousTView != currentTView && (tNode.type === 3 /* Element */));
        // This special case happens when there is a @host on the inject and when we are searching
        // on the host element node.
        /** @type {?} */
        const isHostSpecialCase = (flags & InjectFlags.Host) && hostTElementNode === tNode;
        /** @type {?} */
        const injectableIdx = locateDirectiveOrProvider(tNode, currentTView, token, canAccessViewProviders, isHostSpecialCase);
        if (injectableIdx !== null) {
            return getNodeInjectable(lView, currentTView, injectableIdx, (/** @type {?} */ (tNode)));
        }
        else {
            return NOT_FOUND;
        }
    }
    /**
     * Searches for the given token among the node's directives and providers.
     *
     * @template T
     * @param {?} tNode TNode on which directives are present.
     * @param {?} tView The tView we are currently processing
     * @param {?} token Provider token or type of a directive to look for.
     * @param {?} canAccessViewProviders Whether view providers should be considered.
     * @param {?} isHostSpecialCase Whether the host special case applies.
     * @return {?} Index of a found directive or provider, or null when none found.
     */
    function locateDirectiveOrProvider(tNode, tView, token, canAccessViewProviders, isHostSpecialCase) {
        /** @type {?} */
        const nodeProviderIndexes = tNode.providerIndexes;
        /** @type {?} */
        const tInjectables = tView.data;
        /** @type {?} */
        const injectablesStart = nodeProviderIndexes & 65535 /* ProvidersStartIndexMask */;
        /** @type {?} */
        const directivesStart = tNode.directiveStart;
        /** @type {?} */
        const directiveEnd = tNode.directiveEnd;
        /** @type {?} */
        const cptViewProvidersCount = nodeProviderIndexes >> 16 /* CptViewProvidersCountShift */;
        /** @type {?} */
        const startingIndex = canAccessViewProviders ? injectablesStart : injectablesStart + cptViewProvidersCount;
        // When the host special case applies, only the viewProviders and the component are visible
        /** @type {?} */
        const endIndex = isHostSpecialCase ? injectablesStart + cptViewProvidersCount : directiveEnd;
        for (let i = startingIndex; i < endIndex; i++) {
            /** @type {?} */
            const providerTokenOrDef = (/** @type {?} */ (tInjectables[i]));
            if (i < directivesStart && token === providerTokenOrDef ||
                i >= directivesStart && ((/** @type {?} */ (providerTokenOrDef))).type === token) {
                return i;
            }
        }
        if (isHostSpecialCase) {
            /** @type {?} */
            const dirDef = (/** @type {?} */ (tInjectables[directivesStart]));
            if (dirDef && isComponentDef(dirDef) && dirDef.type === token) {
                return directivesStart;
            }
        }
        return null;
    }
    /**
     * Retrieve or instantiate the injectable from the `LView` at particular `index`.
     *
     * This function checks to see if the value has already been instantiated and if so returns the
     * cached `injectable`. Otherwise if it detects that the value is still a factory it
     * instantiates the `injectable` and caches the value.
     * @param {?} lView
     * @param {?} tView
     * @param {?} index
     * @param {?} tNode
     * @return {?}
     */
    function getNodeInjectable(lView, tView, index, tNode) {
        /** @type {?} */
        let value = lView[index];
        /** @type {?} */
        const tData = tView.data;
        if (isFactory(value)) {
            /** @type {?} */
            const factory = value;
            if (factory.resolving) {
                throw new Error(`Circular dep for ${stringifyForError(tData[index])}`);
            }
            /** @type {?} */
            const previousIncludeViewProviders = setIncludeViewProviders(factory.canSeeViewProviders);
            factory.resolving = true;
            /** @type {?} */
            let previousInjectImplementation;
            if (factory.injectImpl) {
                previousInjectImplementation = setInjectImplementation(factory.injectImpl);
            }
            enterDI(lView, tNode);
            try {
                value = lView[index] = factory.factory(undefined, tData, lView, tNode);
                // This code path is hit for both directives and providers.
                // For perf reasons, we want to avoid searching for hooks on providers.
                // It does no harm to try (the hooks just won't exist), but the extra
                // checks are unnecessary and this is a hot path. So we check to see
                // if the index of the dependency is in the directive range for this
                // tNode. If it's not, we know it's a provider and skip hook registration.
                if (tView.firstCreatePass && index >= tNode.directiveStart) {
                    ngDevMode && assertDirectiveDef(tData[index]);
                    registerPreOrderHooks(index, (/** @type {?} */ (tData[index])), tView);
                }
            }
            finally {
                if (factory.injectImpl)
                    setInjectImplementation(previousInjectImplementation);
                setIncludeViewProviders(previousIncludeViewProviders);
                factory.resolving = false;
                leaveDI();
            }
        }
        return value;
    }
    /**
     * Returns the bit in an injector's bloom filter that should be used to determine whether or not
     * the directive might be provided by the injector.
     *
     * When a directive is public, it is added to the bloom filter and given a unique ID that can be
     * retrieved on the Type. When the directive isn't public or the token is not a directive `null`
     * is returned as the node injector can not possibly provide that token.
     *
     * @param {?} token the injection token
     * @return {?} the matching bit to check in the bloom filter or `null` if the token is not known.
     *   When the returned value is negative then it represents special values such as `Injector`.
     */
    function bloomHashBitOrFactory(token) {
        ngDevMode && assertDefined(token, 'token must be defined');
        if (typeof token === 'string') {
            return token.charCodeAt(0) || 0;
        }
        /** @type {?} */
        const tokenId = ((/** @type {?} */ (token)))[NG_ELEMENT_ID];
        // Negative token IDs are used for special objects such as `Injector`
        return (typeof tokenId === 'number' && tokenId > 0) ? tokenId & BLOOM_MASK : tokenId;
    }
    /**
     * @param {?} bloomHash
     * @param {?} injectorIndex
     * @param {?} injectorView
     * @return {?}
     */
    function bloomHasToken(bloomHash, injectorIndex, injectorView) {
        // Create a mask that targets the specific bit associated with the directive we're looking for.
        // JS bit operations are 32 bits, so this will be a number between 2^0 and 2^31, corresponding
        // to bit positions 0 - 31 in a 32 bit integer.
        /** @type {?} */
        const mask = 1 << bloomHash;
        /** @type {?} */
        const b7 = bloomHash & 0x80;
        /** @type {?} */
        const b6 = bloomHash & 0x40;
        /** @type {?} */
        const b5 = bloomHash & 0x20;
        // Our bloom filter size is 256 bits, which is eight 32-bit bloom filter buckets:
        // bf0 = [0 - 31], bf1 = [32 - 63], bf2 = [64 - 95], bf3 = [96 - 127], etc.
        // Get the bloom filter value from the appropriate bucket based on the directive's bloomBit.
        /** @type {?} */
        let value;
        if (b7) {
            value = b6 ? (b5 ? injectorView[injectorIndex + 7] : injectorView[injectorIndex + 6]) :
                (b5 ? injectorView[injectorIndex + 5] : injectorView[injectorIndex + 4]);
        }
        else {
            value = b6 ? (b5 ? injectorView[injectorIndex + 3] : injectorView[injectorIndex + 2]) :
                (b5 ? injectorView[injectorIndex + 1] : injectorView[injectorIndex]);
        }
        // If the bloom filter value has the bit corresponding to the directive's bloomBit flipped on,
        // this injector is a potential match.
        return !!(value & mask);
    }
    /**
     * Returns true if flags prevent parent injector from being searched for tokens
     * @param {?} flags
     * @param {?} isFirstHostTNode
     * @return {?}
     */
    function shouldSearchParent(flags, isFirstHostTNode) {
        return !(flags & InjectFlags.Self) && !(flags & InjectFlags.Host && isFirstHostTNode);
    }
    class NodeInjector {
        /**
         * @param {?} _tNode
         * @param {?} _lView
         */
        constructor(_tNode, _lView) {
            this._tNode = _tNode;
            this._lView = _lView;
        }
        /**
         * @param {?} token
         * @param {?=} notFoundValue
         * @return {?}
         */
        get(token, notFoundValue) {
            return getOrCreateInjectable(this._tNode, this._lView, token, undefined, notFoundValue);
        }
    }
    /**
     * \@codeGenApi
     * @template T
     * @param {?} type
     * @return {?}
     */
    function ɵɵgetFactoryOf(type) {
        /** @type {?} */
        const typeAny = (/** @type {?} */ (type));
        if (isForwardRef(type)) {
            return (/** @type {?} */ (((/**
             * @return {?}
             */
            () => {
                /** @type {?} */
                const factory = ɵɵgetFactoryOf(resolveForwardRef(typeAny));
                return factory ? factory() : null;
            }))));
        }
        /** @type {?} */
        let factory = getFactoryDef(typeAny);
        if (factory === null) {
            /** @type {?} */
            const injectorDef = getInjectorDef(typeAny);
            factory = injectorDef && injectorDef.factory;
        }
        return factory || null;
    }
    /**
     * \@codeGenApi
     * @template T
     * @param {?} type
     * @return {?}
     */
    function ɵɵgetInheritedFactory(type) {
        return noSideEffects((/**
         * @return {?}
         */
        () => {
            /** @type {?} */
            const ownConstructor = type.prototype.constructor;
            /** @type {?} */
            const ownFactory = ownConstructor[NG_FACTORY_DEF] || ɵɵgetFactoryOf(ownConstructor);
            /** @type {?} */
            const objectPrototype = Object.prototype;
            /** @type {?} */
            let parent = Object.getPrototypeOf(type.prototype).constructor;
            // Go up the prototype until we hit `Object`.
            while (parent && parent !== objectPrototype) {
                /** @type {?} */
                const factory = parent[NG_FACTORY_DEF] || ɵɵgetFactoryOf(parent);
                // If we hit something that has a factory and the factory isn't the same as the type,
                // we've found the inherited factory. Note the check that the factory isn't the type's
                // own factory is redundant in most cases, but if the user has custom decorators on the
                // class, this lookup will start one level down in the prototype chain, causing us to
                // find the own factory first and potentially triggering an infinite loop downstream.
                if (factory && factory !== ownFactory) {
                    return factory;
                }
                parent = Object.getPrototypeOf(parent);
            }
            // There is no factory defined. Either this was improper usage of inheritance
            // (no Angular decorator on the superclass) or there is no constructor at all
            // in the inheritance chain. Since the two cases cannot be distinguished, the
            // latter has to be assumed.
            return (/**
             * @param {?} t
             * @return {?}
             */
            t => new t());
        }));
    }
    const ERROR_DEBUG_CONTEXT = 'ngDebugContext';
    const ERROR_ORIGINAL_ERROR = 'ngOriginalError';
    const ERROR_LOGGER = 'ngErrorLogger';
    function wrappedError(message, originalError) {
        const msg = `${message} caused by: ${originalError instanceof Error ? originalError.message : originalError}`;
        const error = Error(msg);
        error[ERROR_ORIGINAL_ERROR] = originalError;
        return error;
    }
    /**
     * @param {?} error
     * @return {?}
     */
    function getDebugContext(error) {
        return ((/** @type {?} */ (error)))[ERROR_DEBUG_CONTEXT];
    }
    /**
     * @param {?} error
     * @return {?}
     */
    function getOriginalError(error) {
        return ((/** @type {?} */ (error)))[ERROR_ORIGINAL_ERROR];
    }
    /**
     * @param {?} error
     * @return {?}
     */
    function getErrorLogger(error) {
        return ((/** @type {?} */ (error)))[ERROR_LOGGER] || defaultErrorLogger;
    }
    /**
     * @param {?} console
     * @param {...?} values
     * @return {?}
     */
    function defaultErrorLogger(console, ...values) {
        ((/** @type {?} */ (console.error)))(...values);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/error_handler.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Provides a hook for centralized exception handling.
     *
     * The default implementation of `ErrorHandler` prints error messages to the `console`. To
     * intercept error handling, write a custom exception handler that replaces this default as
     * appropriate for your app.
     *
     * \@usageNotes
     * ### Example
     *
     * ```
     * class MyErrorHandler implements ErrorHandler {
     *   handleError(error) {
     *     // do something with the exception
     *   }
     * }
     *
     * \@NgModule({
     *   providers: [{provide: ErrorHandler, useClass: MyErrorHandler}]
     * })
     * class MyModule {}
     * ```
     *
     * \@publicApi
     */
    class ErrorHandler {
        constructor() {
            /**
             * \@internal
             */
            this._console = console;
        }
        /**
         * @param {?} error
         * @return {?}
         */
        handleError(error) {
            /** @type {?} */
            const originalError = this._findOriginalError(error);
            /** @type {?} */
            const context = this._findContext(error);
            // Note: Browser consoles show the place from where console.error was called.
            // We can use this to give users additional information about the error.
            /** @type {?} */
            const errorLogger = getErrorLogger(error);
            errorLogger(this._console, `ERROR`, error);
            if (originalError) {
                errorLogger(this._console, `ORIGINAL ERROR`, originalError);
            }
            if (context) {
                errorLogger(this._console, 'ERROR CONTEXT', context);
            }
        }
        /**
         * \@internal
         * @param {?} error
         * @return {?}
         */
        _findContext(error) {
            if (error) {
                return getDebugContext(error) ? getDebugContext(error) :
                    this._findContext(getOriginalError(error));
            }
            return null;
        }
        /**
         * \@internal
         * @param {?} error
         * @return {?}
         */
        _findOriginalError(error) {
            /** @type {?} */
            let e = getOriginalError(error);
            while (e && getOriginalError(e)) {
                e = getOriginalError(e);
            }
            return e;
        }
    }
    /**
     * Defines a schema that allows an NgModule to contain the following:
     * - Non-Angular elements named with dash case (`-`).
     * - Element properties named with dash case (`-`).
     * Dash case is the naming convention for custom elements.
     *
     * \@publicApi
     * @type {?}
     */
    const CUSTOM_ELEMENTS_SCHEMA = {
        name: 'custom-elements'
    };
    /**
     * Defines a schema that allows any property on any element.
     *
     * \@publicApi
     * @type {?}
     */
    const NO_ERRORS_SCHEMA = {
        name: 'no-errors-schema'
    };
    /**
     * @abstract
     */
    class SafeValueImpl {
        /**
         * @param {?} changingThisBreaksApplicationSecurity
         */
        constructor(changingThisBreaksApplicationSecurity) {
            this.changingThisBreaksApplicationSecurity = changingThisBreaksApplicationSecurity;
        }
        /**
         * @return {?}
         */
        toString() {
            return `SafeValue must use [property]=binding: ${this.changingThisBreaksApplicationSecurity}` +
                ` (see http://g.co/ng/security#xss)`;
        }
    }
    class SafeHtmlImpl extends SafeValueImpl {
        /**
         * @return {?}
         */
        getTypeName() {
            return "HTML" /* Html */;
        }
    }
    class SafeStyleImpl extends SafeValueImpl {
        /**
         * @return {?}
         */
        getTypeName() {
            return "Style" /* Style */;
        }
    }
    class SafeScriptImpl extends SafeValueImpl {
        /**
         * @return {?}
         */
        getTypeName() {
            return "Script" /* Script */;
        }
    }
    class SafeUrlImpl extends SafeValueImpl {
        /**
         * @return {?}
         */
        getTypeName() {
            return "URL" /* Url */;
        }
    }
    class SafeResourceUrlImpl extends SafeValueImpl {
        /**
         * @return {?}
         */
        getTypeName() {
            return "ResourceURL" /* ResourceUrl */;
        }
    }
    /**
     * @template T
     * @param {?} value
     * @return {?}
     */
    function unwrapSafeValue(value) {
        return value instanceof SafeValueImpl ? (/** @type {?} */ ((/** @type {?} */ (value.changingThisBreaksApplicationSecurity)))) :
            (/** @type {?} */ ((/** @type {?} */ (value))));
    }
    /**
     * @param {?} value
     * @param {?} type
     * @return {?}
     */
    function allowSanitizationBypassAndThrow(value, type) {
        /** @type {?} */
        const actualType = getSanitizationBypassType(value);
        if (actualType != null && actualType !== type) {
            // Allow ResourceURLs in URL contexts, they are strictly more trusted.
            if (actualType === "ResourceURL" /* ResourceUrl */ && type === "URL" /* Url */)
                return true;
            throw new Error(`Required a safe ${type}, got a ${actualType} (see http://g.co/ng/security#xss)`);
        }
        return actualType === type;
    }
    /**
     * @param {?} value
     * @return {?}
     */
    function getSanitizationBypassType(value) {
        return value instanceof SafeValueImpl && (/** @type {?} */ (value.getTypeName())) || null;
    }
    /**
     * Mark `html` string as trusted.
     *
     * This function wraps the trusted string in `String` and brands it in a way which makes it
     * recognizable to {\@link htmlSanitizer} to be trusted implicitly.
     *
     * @param {?} trustedHtml `html` string which needs to be implicitly trusted.
     * @return {?} a `html` which has been branded to be implicitly trusted.
     */
    function bypassSanitizationTrustHtml(trustedHtml) {
        return new SafeHtmlImpl(trustedHtml);
    }
    /**
     * Mark `style` string as trusted.
     *
     * This function wraps the trusted string in `String` and brands it in a way which makes it
     * recognizable to {\@link styleSanitizer} to be trusted implicitly.
     *
     * @param {?} trustedStyle `style` string which needs to be implicitly trusted.
     * @return {?} a `style` hich has been branded to be implicitly trusted.
     */
    function bypassSanitizationTrustStyle(trustedStyle) {
        return new SafeStyleImpl(trustedStyle);
    }
    /**
     * Mark `script` string as trusted.
     *
     * This function wraps the trusted string in `String` and brands it in a way which makes it
     * recognizable to {\@link scriptSanitizer} to be trusted implicitly.
     *
     * @param {?} trustedScript `script` string which needs to be implicitly trusted.
     * @return {?} a `script` which has been branded to be implicitly trusted.
     */
    function bypassSanitizationTrustScript(trustedScript) {
        return new SafeScriptImpl(trustedScript);
    }
    /**
     * Mark `url` string as trusted.
     *
     * This function wraps the trusted string in `String` and brands it in a way which makes it
     * recognizable to {\@link urlSanitizer} to be trusted implicitly.
     *
     * @param {?} trustedUrl `url` string which needs to be implicitly trusted.
     * @return {?} a `url`  which has been branded to be implicitly trusted.
     */
    function bypassSanitizationTrustUrl(trustedUrl) {
        return new SafeUrlImpl(trustedUrl);
    }
    /**
     * Mark `url` string as trusted.
     *
     * This function wraps the trusted string in `String` and brands it in a way which makes it
     * recognizable to {\@link resourceUrlSanitizer} to be trusted implicitly.
     *
     * @param {?} trustedResourceUrl `url` string which needs to be implicitly trusted.
     * @return {?} a `url` which has been branded to be implicitly trusted.
     */
    function bypassSanitizationTrustResourceUrl(trustedResourceUrl) {
        return new SafeResourceUrlImpl(trustedResourceUrl);
    }

    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * This file is used to control if the default rendering pipeline should be `ViewEngine` or `Ivy`.
     *
     * For more information on how to run and debug tests with either Ivy or View Engine (legacy),
     * please see [BAZEL.md](./docs/BAZEL.md).
     */
    let _devMode = true;
    let _runModeLocked = false;
    /**
     * Returns whether Angular is in development mode. After called once,
     * the value is locked and won't change any more.
     *
     * By default, this is true, unless a user calls `enableProdMode` before calling this.
     *
     * @publicApi
     */
    function isDevMode() {
        _runModeLocked = true;
        return _devMode;
    }
    /**
     * Disable Angular's development mode, which turns off assertions and other
     * checks within the framework.
     *
     * One important assertion this disables verifies that a change detection pass
     * does not result in additional changes to any bindings (also known as
     * unidirectional data flow).
     *
     * @publicApi
     */
    function enableProdMode() {
        if (_runModeLocked) {
            throw new Error('Cannot enable prod mode after platform setup.');
        }
        _devMode = false;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/sanitization/inert_body.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * This helper class is used to get hold of an inert tree of DOM elements containing dirty HTML
     * that needs sanitizing.
     * Depending upon browser support we must use one of three strategies for doing this.
     * Support: Safari 10.x -> XHR strategy
     * Support: Firefox -> DomParser strategy
     * Default: InertDocument strategy
     */
    class InertBodyHelper {
        /**
         * @param {?} defaultDoc
         */
        constructor(defaultDoc) {
            this.defaultDoc = defaultDoc;
            this.inertDocument = this.defaultDoc.implementation.createHTMLDocument('sanitization-inert');
            /** @type {?} */
            let inertBodyElement = this.inertDocument.body;
            if (inertBodyElement == null) {
                // usually there should be only one body element in the document, but IE doesn't have any, so
                // we need to create one.
                /** @type {?} */
                const inertHtml = this.inertDocument.createElement('html');
                this.inertDocument.appendChild(inertHtml);
                inertBodyElement = this.inertDocument.createElement('body');
                inertHtml.appendChild(inertBodyElement);
            }
            inertBodyElement.innerHTML = '<svg><g onload="this.parentNode.remove()"></g></svg>';
            if (inertBodyElement.querySelector && !inertBodyElement.querySelector('svg')) {
                // We just hit the Safari 10.1 bug - which allows JS to run inside the SVG G element
                // so use the XHR strategy.
                this.getInertBodyElement = this.getInertBodyElement_XHR;
                return;
            }
            inertBodyElement.innerHTML = '<svg><p><style><img src="</style><img src=x onerror=alert(1)//">';
            if (inertBodyElement.querySelector && inertBodyElement.querySelector('svg img')) {
                // We just hit the Firefox bug - which prevents the inner img JS from being sanitized
                // so use the DOMParser strategy, if it is available.
                // If the DOMParser is not available then we are not in Firefox (Server/WebWorker?) so we
                // fall through to the default strategy below.
                if (isDOMParserAvailable()) {
                    this.getInertBodyElement = this.getInertBodyElement_DOMParser;
                    return;
                }
            }
            // None of the bugs were hit so it is safe for us to use the default InertDocument strategy
            this.getInertBodyElement = this.getInertBodyElement_InertDocument;
        }
        /**
         * Use XHR to create and fill an inert body element (on Safari 10.1)
         * See
         * https://github.com/cure53/DOMPurify/blob/a992d3a75031cb8bb032e5ea8399ba972bdf9a65/src/purify.js#L439-L449
         * @private
         * @param {?} html
         * @return {?}
         */
        getInertBodyElement_XHR(html) {
            // We add these extra elements to ensure that the rest of the content is parsed as expected
            // e.g. leading whitespace is maintained and tags like `<meta>` do not get hoisted to the
            // `<head>` tag.
            html = '<body><remove></remove>' + html + '</body>';
            try {
                html = encodeURI(html);
            }
            catch (_a) {
                return null;
            }
            /** @type {?} */
            const xhr = new XMLHttpRequest();
            xhr.responseType = 'document';
            xhr.open('GET', 'data:text/html;charset=utf-8,' + html, false);
            xhr.send(undefined);
            /** @type {?} */
            const body = xhr.response.body;
            body.removeChild((/** @type {?} */ (body.firstChild)));
            return body;
        }
        /**
         * Use DOMParser to create and fill an inert body element (on Firefox)
         * See https://github.com/cure53/DOMPurify/releases/tag/0.6.7
         *
         * @private
         * @param {?} html
         * @return {?}
         */
        getInertBodyElement_DOMParser(html) {
            // We add these extra elements to ensure that the rest of the content is parsed as expected
            // e.g. leading whitespace is maintained and tags like `<meta>` do not get hoisted to the
            // `<head>` tag.
            html = '<body><remove></remove>' + html + '</body>';
            try {
                /** @type {?} */
                const body = (/** @type {?} */ (new ((/** @type {?} */ (window))).DOMParser().parseFromString(html, 'text/html').body));
                body.removeChild((/** @type {?} */ (body.firstChild)));
                return body;
            }
            catch (_a) {
                return null;
            }
        }
        /**
         * Use an HTML5 `template` element, if supported, or an inert body element created via
         * `createHtmlDocument` to create and fill an inert DOM element.
         * This is the default sane strategy to use if the browser does not require one of the specialised
         * strategies above.
         * @private
         * @param {?} html
         * @return {?}
         */
        getInertBodyElement_InertDocument(html) {
            // Prefer using <template> element if supported.
            /** @type {?} */
            const templateEl = this.inertDocument.createElement('template');
            if ('content' in templateEl) {
                templateEl.innerHTML = html;
                return templateEl;
            }
            // Note that previously we used to do something like `this.inertDocument.body.innerHTML = html`
            // and we returned the inert `body` node. This was changed, because IE seems to treat setting
            // `innerHTML` on an inserted element differently, compared to one that hasn't been inserted
            // yet. In particular, IE appears to split some of the text into multiple text nodes rather
            // than keeping them in a single one which ends up messing with Ivy's i18n parsing further
            // down the line. This has been worked around by creating a new inert `body` and using it as
            // the root node in which we insert the HTML.
            /** @type {?} */
            const inertBody = this.inertDocument.createElement('body');
            inertBody.innerHTML = html;
            // Support: IE 9-11 only
            // strip custom-namespaced attributes on IE<=11
            if (((/** @type {?} */ (this.defaultDoc))).documentMode) {
                this.stripCustomNsAttrs(inertBody);
            }
            return inertBody;
        }
        /**
         * When IE9-11 comes across an unknown namespaced attribute e.g. 'xlink:foo' it adds 'xmlns:ns1'
         * attribute to declare ns1 namespace and prefixes the attribute with 'ns1' (e.g.
         * 'ns1:xlink:foo').
         *
         * This is undesirable since we don't want to allow any of these custom attributes. This method
         * strips them all.
         * @private
         * @param {?} el
         * @return {?}
         */
        stripCustomNsAttrs(el) {
            /** @type {?} */
            const elAttrs = el.attributes;
            // loop backwards so that we can support removals.
            for (let i = elAttrs.length - 1; 0 < i; i--) {
                /** @type {?} */
                const attrib = elAttrs.item(i);
                /** @type {?} */
                const attrName = (/** @type {?} */ (attrib)).name;
                if (attrName === 'xmlns:ns1' || attrName.indexOf('ns1:') === 0) {
                    el.removeAttribute(attrName);
                }
            }
            /** @type {?} */
            let childNode = (/** @type {?} */ (el.firstChild));
            while (childNode) {
                if (childNode.nodeType === Node.ELEMENT_NODE)
                    this.stripCustomNsAttrs((/** @type {?} */ (childNode)));
                childNode = childNode.nextSibling;
            }
        }
    }
    /**
     * We need to determine whether the DOMParser exists in the global context.
     * The try-catch is because, on some browsers, trying to access this property
     * on window can actually throw an error.
     *
     * @suppress {uselessCode}
     * @return {?}
     */
    function isDOMParserAvailable() {
        try {
            return !!((/** @type {?} */ (window))).DOMParser;
        }
        catch (_a) {
            return false;
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/sanitization/url_sanitizer.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * A pattern that recognizes a commonly useful subset of URLs that are safe.
     *
     * This regular expression matches a subset of URLs that will not cause script
     * execution if used in URL context within a HTML document. Specifically, this
     * regular expression matches if (comment from here on and regex copied from
     * Soy's EscapingConventions):
     * (1) Either an allowed protocol (http, https, mailto or ftp).
     * (2) or no protocol.  A protocol must be followed by a colon. The below
     *     allows that by allowing colons only after one of the characters [/?#].
     *     A colon after a hash (#) must be in the fragment.
     *     Otherwise, a colon after a (?) must be in a query.
     *     Otherwise, a colon after a single solidus (/) must be in a path.
     *     Otherwise, a colon after a double solidus (//) must be in the authority
     *     (before port).
     *
     * The pattern disallows &, used in HTML entity declarations before
     * one of the characters in [/?#]. This disallows HTML entities used in the
     * protocol name, which should never happen, e.g. "h&#116;tp" for "http".
     * It also disallows HTML entities in the first path part of a relative path,
     * e.g. "foo&lt;bar/baz".  Our existing escaping functions should not produce
     * that. More importantly, it disallows masking of a colon,
     * e.g. "javascript&#58;...".
     *
     * This regular expression was taken from the Closure sanitization library.
     * @type {?}
     */
    const SAFE_URL_PATTERN = /^(?:(?:https?|mailto|ftp|tel|file):|[^&:/?#]*(?:[/?#]|$))/gi;
    /**
     * A pattern that matches safe data URLs. Only matches image, video and audio types.
     * @type {?}
     */
    const DATA_URL_PATTERN = /^data:(?:image\/(?:bmp|gif|jpeg|jpg|png|tiff|webp)|video\/(?:mpeg|mp4|ogg|webm)|audio\/(?:mp3|oga|ogg|opus));base64,[a-z0-9+\/]+=*$/i;
    /**
     * @param {?} url
     * @return {?}
     */
    function _sanitizeUrl(url) {
        url = String(url);
        if (url.match(SAFE_URL_PATTERN) || url.match(DATA_URL_PATTERN))
            return url;
        if (isDevMode()) {
            console.warn(`WARNING: sanitizing unsafe URL value ${url} (see http://g.co/ng/security#xss)`);
        }
        return 'unsafe:' + url;
    }
    /**
     * @param {?} srcset
     * @return {?}
     */
    function sanitizeSrcset(srcset) {
        srcset = String(srcset);
        return srcset.split(',').map((/**
         * @param {?} srcset
         * @return {?}
         */
        (srcset) => _sanitizeUrl(srcset.trim()))).join(', ');
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/sanitization/html_sanitizer.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @param {?} tags
     * @return {?}
     */
    function tagSet(tags) {
        /** @type {?} */
        const res = {};
        for (const t of tags.split(','))
            res[t] = true;
        return res;
    }
    /**
     * @param {...?} sets
     * @return {?}
     */
    function merge$1(...sets) {
        /** @type {?} */
        const res = {};
        for (const s of sets) {
            for (const v in s) {
                if (s.hasOwnProperty(v))
                    res[v] = true;
            }
        }
        return res;
    }
    // Good source of info about elements and attributes
    // http://dev.w3.org/html5/spec/Overview.html#semantics
    // http://simon.html5.org/html-elements
    // Safe Void Elements - HTML5
    // http://dev.w3.org/html5/spec/Overview.html#void-elements
    /** @type {?} */
    const VOID_ELEMENTS = tagSet('area,br,col,hr,img,wbr');
    // Elements that you can, intentionally, leave open (and which close themselves)
    // http://dev.w3.org/html5/spec/Overview.html#optional-tags
    /** @type {?} */
    const OPTIONAL_END_TAG_BLOCK_ELEMENTS = tagSet('colgroup,dd,dt,li,p,tbody,td,tfoot,th,thead,tr');
    /** @type {?} */
    const OPTIONAL_END_TAG_INLINE_ELEMENTS = tagSet('rp,rt');
    /** @type {?} */
    const OPTIONAL_END_TAG_ELEMENTS = merge$1(OPTIONAL_END_TAG_INLINE_ELEMENTS, OPTIONAL_END_TAG_BLOCK_ELEMENTS);
    // Safe Block Elements - HTML5
    /** @type {?} */
    const BLOCK_ELEMENTS = merge$1(OPTIONAL_END_TAG_BLOCK_ELEMENTS, tagSet('address,article,' +
        'aside,blockquote,caption,center,del,details,dialog,dir,div,dl,figure,figcaption,footer,h1,h2,h3,h4,h5,' +
        'h6,header,hgroup,hr,ins,main,map,menu,nav,ol,pre,section,summary,table,ul'));
    // Inline Elements - HTML5
    /** @type {?} */
    const INLINE_ELEMENTS = merge$1(OPTIONAL_END_TAG_INLINE_ELEMENTS, tagSet('a,abbr,acronym,audio,b,' +
        'bdi,bdo,big,br,cite,code,del,dfn,em,font,i,img,ins,kbd,label,map,mark,picture,q,ruby,rp,rt,s,' +
        'samp,small,source,span,strike,strong,sub,sup,time,track,tt,u,var,video'));
    /** @type {?} */
    const VALID_ELEMENTS = merge$1(VOID_ELEMENTS, BLOCK_ELEMENTS, INLINE_ELEMENTS, OPTIONAL_END_TAG_ELEMENTS);
    // Attributes that have href and hence need to be sanitized
    /** @type {?} */
    const URI_ATTRS = tagSet('background,cite,href,itemtype,longdesc,poster,src,xlink:href');
    // Attributes that have special href set hence need to be sanitized
    /** @type {?} */
    const SRCSET_ATTRS = tagSet('srcset');
    /** @type {?} */
    const HTML_ATTRS = tagSet('abbr,accesskey,align,alt,autoplay,axis,bgcolor,border,cellpadding,cellspacing,class,clear,color,cols,colspan,' +
        'compact,controls,coords,datetime,default,dir,download,face,headers,height,hidden,hreflang,hspace,' +
        'ismap,itemscope,itemprop,kind,label,lang,language,loop,media,muted,nohref,nowrap,open,preload,rel,rev,role,rows,rowspan,rules,' +
        'scope,scrolling,shape,size,sizes,span,srclang,start,summary,tabindex,target,title,translate,type,usemap,' +
        'valign,value,vspace,width');
    // Accessibility attributes as per WAI-ARIA 1.1 (W3C Working Draft 14 December 2018)
    /** @type {?} */
    const ARIA_ATTRS = tagSet('aria-activedescendant,aria-atomic,aria-autocomplete,aria-busy,aria-checked,aria-colcount,aria-colindex,' +
        'aria-colspan,aria-controls,aria-current,aria-describedby,aria-details,aria-disabled,aria-dropeffect,' +
        'aria-errormessage,aria-expanded,aria-flowto,aria-grabbed,aria-haspopup,aria-hidden,aria-invalid,' +
        'aria-keyshortcuts,aria-label,aria-labelledby,aria-level,aria-live,aria-modal,aria-multiline,' +
        'aria-multiselectable,aria-orientation,aria-owns,aria-placeholder,aria-posinset,aria-pressed,aria-readonly,' +
        'aria-relevant,aria-required,aria-roledescription,aria-rowcount,aria-rowindex,aria-rowspan,aria-selected,' +
        'aria-setsize,aria-sort,aria-valuemax,aria-valuemin,aria-valuenow,aria-valuetext');
    // NB: This currently consciously doesn't support SVG. SVG sanitization has had several security
    // issues in the past, so it seems safer to leave it out if possible. If support for binding SVG via
    // innerHTML is required, SVG attributes should be added here.
    // NB: Sanitization does not allow <form> elements or other active elements (<button> etc). Those
    // can be sanitized, but they increase security surface area without a legitimate use case, so they
    // are left out here.
    /** @type {?} */
    const VALID_ATTRS = merge$1(URI_ATTRS, SRCSET_ATTRS, HTML_ATTRS, ARIA_ATTRS);
    // Elements whose content should not be traversed/preserved, if the elements themselves are invalid.
    //
    // Typically, `<invalid>Some content</invalid>` would traverse (and in this case preserve)
    // `Some content`, but strip `invalid-element` opening/closing tags. For some elements, though, we
    // don't want to preserve the content, if the elements themselves are going to be removed.
    /** @type {?} */
    const SKIP_TRAVERSING_CONTENT_IF_INVALID_ELEMENTS = tagSet('script,style,template');
    /**
     * SanitizingHtmlSerializer serializes a DOM fragment, stripping out any unsafe elements and unsafe
     * attributes.
     */
    class SanitizingHtmlSerializer {
        constructor() {
            // Explicitly track if something was stripped, to avoid accidentally warning of sanitization just
            // because characters were re-encoded.
            this.sanitizedSomething = false;
            this.buf = [];
        }
        /**
         * @param {?} el
         * @return {?}
         */
        sanitizeChildren(el) {
            // This cannot use a TreeWalker, as it has to run on Angular's various DOM adapters.
            // However this code never accesses properties off of `document` before deleting its contents
            // again, so it shouldn't be vulnerable to DOM clobbering.
            /** @type {?} */
            let current = (/** @type {?} */ (el.firstChild));
            /** @type {?} */
            let traverseContent = true;
            while (current) {
                if (current.nodeType === Node.ELEMENT_NODE) {
                    traverseContent = this.startElement((/** @type {?} */ (current)));
                }
                else if (current.nodeType === Node.TEXT_NODE) {
                    this.chars((/** @type {?} */ (current.nodeValue)));
                }
                else {
                    // Strip non-element, non-text nodes.
                    this.sanitizedSomething = true;
                }
                if (traverseContent && current.firstChild) {
                    current = (/** @type {?} */ (current.firstChild));
                    continue;
                }
                while (current) {
                    // Leaving the element. Walk up and to the right, closing tags as we go.
                    if (current.nodeType === Node.ELEMENT_NODE) {
                        this.endElement((/** @type {?} */ (current)));
                    }
                    /** @type {?} */
                    let next = this.checkClobberedElement(current, (/** @type {?} */ (current.nextSibling)));
                    if (next) {
                        current = next;
                        break;
                    }
                    current = this.checkClobberedElement(current, (/** @type {?} */ (current.parentNode)));
                }
            }
            return this.buf.join('');
        }
        /**
         * Sanitizes an opening element tag (if valid) and returns whether the element's contents should
         * be traversed. Element content must always be traversed (even if the element itself is not
         * valid/safe), unless the element is one of `SKIP_TRAVERSING_CONTENT_IF_INVALID_ELEMENTS`.
         *
         * @private
         * @param {?} element The element to sanitize.
         * @return {?} True if the element's contents should be traversed.
         */
        startElement(element) {
            /** @type {?} */
            const tagName = element.nodeName.toLowerCase();
            if (!VALID_ELEMENTS.hasOwnProperty(tagName)) {
                this.sanitizedSomething = true;
                return !SKIP_TRAVERSING_CONTENT_IF_INVALID_ELEMENTS.hasOwnProperty(tagName);
            }
            this.buf.push('<');
            this.buf.push(tagName);
            /** @type {?} */
            const elAttrs = element.attributes;
            for (let i = 0; i < elAttrs.length; i++) {
                /** @type {?} */
                const elAttr = elAttrs.item(i);
                /** @type {?} */
                const attrName = (/** @type {?} */ (elAttr)).name;
                /** @type {?} */
                const lower = attrName.toLowerCase();
                if (!VALID_ATTRS.hasOwnProperty(lower)) {
                    this.sanitizedSomething = true;
                    continue;
                }
                /** @type {?} */
                let value = (/** @type {?} */ (elAttr)).value;
                // TODO(martinprobst): Special case image URIs for data:image/...
                if (URI_ATTRS[lower])
                    value = _sanitizeUrl(value);
                if (SRCSET_ATTRS[lower])
                    value = sanitizeSrcset(value);
                this.buf.push(' ', attrName, '="', encodeEntities(value), '"');
            }
            this.buf.push('>');
            return true;
        }
        /**
         * @private
         * @param {?} current
         * @return {?}
         */
        endElement(current) {
            /** @type {?} */
            const tagName = current.nodeName.toLowerCase();
            if (VALID_ELEMENTS.hasOwnProperty(tagName) && !VOID_ELEMENTS.hasOwnProperty(tagName)) {
                this.buf.push('</');
                this.buf.push(tagName);
                this.buf.push('>');
            }
        }
        /**
         * @private
         * @param {?} chars
         * @return {?}
         */
        chars(chars) {
            this.buf.push(encodeEntities(chars));
        }
        /**
         * @param {?} node
         * @param {?} nextNode
         * @return {?}
         */
        checkClobberedElement(node, nextNode) {
            if (nextNode &&
                (node.compareDocumentPosition(nextNode) &
                    Node.DOCUMENT_POSITION_CONTAINED_BY) === Node.DOCUMENT_POSITION_CONTAINED_BY) {
                throw new Error(`Failed to sanitize html because the element is clobbered: ${((/** @type {?} */ (node))).outerHTML}`);
            }
            return nextNode;
        }
    }
    // Regular Expressions for parsing tags and attributes
    /** @type {?} */
    const SURROGATE_PAIR_REGEXP = /[\uD800-\uDBFF][\uDC00-\uDFFF]/g;
    // ! to ~ is the ASCII range.
    /** @type {?} */
    const NON_ALPHANUMERIC_REGEXP = /([^\#-~ |!])/g;
    /**
     * Escapes all potentially dangerous characters, so that the
     * resulting string can be safely inserted into attribute or
     * element text.
     * @param {?} value
     * @return {?}
     */
    function encodeEntities(value) {
        return value.replace(/&/g, '&amp;')
            .replace(SURROGATE_PAIR_REGEXP, (/**
         * @param {?} match
         * @return {?}
         */
        function (match) {
            /** @type {?} */
            const hi = match.charCodeAt(0);
            /** @type {?} */
            const low = match.charCodeAt(1);
            return '&#' + (((hi - 0xD800) * 0x400) + (low - 0xDC00) + 0x10000) + ';';
        }))
            .replace(NON_ALPHANUMERIC_REGEXP, (/**
         * @param {?} match
         * @return {?}
         */
        function (match) {
            return '&#' + match.charCodeAt(0) + ';';
        }))
            .replace(/</g, '&lt;')
            .replace(/>/g, '&gt;');
    }
    /** @type {?} */
    let inertBodyHelper;
    /**
     * Sanitizes the given unsafe, untrusted HTML fragment, and returns HTML text that is safe to add to
     * the DOM in a browser environment.
     * @param {?} defaultDoc
     * @param {?} unsafeHtmlInput
     * @return {?}
     */
    function _sanitizeHtml(defaultDoc, unsafeHtmlInput) {
        /** @type {?} */
        let inertBodyElement = null;
        try {
            inertBodyHelper = inertBodyHelper || new InertBodyHelper(defaultDoc);
            // Make sure unsafeHtml is actually a string (TypeScript types are not enforced at runtime).
            /** @type {?} */
            let unsafeHtml = unsafeHtmlInput ? String(unsafeHtmlInput) : '';
            inertBodyElement = inertBodyHelper.getInertBodyElement(unsafeHtml);
            // mXSS protection. Repeatedly parse the document to make sure it stabilizes, so that a browser
            // trying to auto-correct incorrect HTML cannot cause formerly inert HTML to become dangerous.
            /** @type {?} */
            let mXSSAttempts = 5;
            /** @type {?} */
            let parsedHtml = unsafeHtml;
            do {
                if (mXSSAttempts === 0) {
                    throw new Error('Failed to sanitize html because the input is unstable');
                }
                mXSSAttempts--;
                unsafeHtml = parsedHtml;
                parsedHtml = (/** @type {?} */ (inertBodyElement)).innerHTML;
                inertBodyElement = inertBodyHelper.getInertBodyElement(unsafeHtml);
            } while (unsafeHtml !== parsedHtml);
            /** @type {?} */
            const sanitizer = new SanitizingHtmlSerializer();
            /** @type {?} */
            const safeHtml = sanitizer.sanitizeChildren((/** @type {?} */ (getTemplateContent((/** @type {?} */ (inertBodyElement))))) || inertBodyElement);
            if (isDevMode() && sanitizer.sanitizedSomething) {
                console.warn('WARNING: sanitizing HTML stripped some content, see http://g.co/ng/security#xss');
            }
            return safeHtml;
        }
        finally {
            // In case anything goes wrong, clear out inertElement to reset the entire DOM structure.
            if (inertBodyElement) {
                /** @type {?} */
                const parent = getTemplateContent(inertBodyElement) || inertBodyElement;
                while (parent.firstChild) {
                    parent.removeChild(parent.firstChild);
                }
            }
        }
    }
    /**
     * @param {?} el
     * @return {?}
     */
    function getTemplateContent(el) {
        return 'content' in ((/** @type {?} */ (el))) && isTemplateElement(el) ?
            el.content :
            null;
    }
    /**
     * @param {?} el
     * @return {?}
     */
    function isTemplateElement(el) {
        return el.nodeType === Node.ELEMENT_NODE && el.nodeName === 'TEMPLATE';
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/sanitization/security.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /** @enum {number} */
    const SecurityContext = {
        NONE: 0,
        HTML: 1,
        STYLE: 2,
        SCRIPT: 3,
        URL: 4,
        RESOURCE_URL: 5,
    };
    SecurityContext[SecurityContext.NONE] = 'NONE';
    SecurityContext[SecurityContext.HTML] = 'HTML';
    SecurityContext[SecurityContext.STYLE] = 'STYLE';
    SecurityContext[SecurityContext.SCRIPT] = 'SCRIPT';
    SecurityContext[SecurityContext.URL] = 'URL';
    SecurityContext[SecurityContext.RESOURCE_URL] = 'RESOURCE_URL';

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/sanitization/style_sanitizer.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Regular expression for safe style values.
     *
     * Quotes (" and ') are allowed, but a check must be done elsewhere to ensure they're balanced.
     *
     * ',' allows multiple values to be assigned to the same property (e.g. background-attachment or
     * font-family) and hence could allow multiple values to get injected, but that should pose no risk
     * of XSS.
     *
     * The function expression checks only for XSS safety, not for CSS validity.
     *
     * This regular expression was taken from the Closure sanitization library, and augmented for
     * transformation values.
     * @type {?}
     */
    const VALUES = '[-,."\'%_!# a-zA-Z0-9]+';
    /** @type {?} */
    const TRANSFORMATION_FNS = '(?:matrix|translate|scale|rotate|skew|perspective)(?:X|Y|Z|3d)?';
    /** @type {?} */
    const COLOR_FNS = '(?:rgb|hsl)a?';
    /** @type {?} */
    const GRADIENTS = '(?:repeating-)?(?:linear|radial)-gradient';
    /** @type {?} */
    const CSS3_FNS = '(?:attr|calc|var)';
    /** @type {?} */
    const FN_ARGS = '\\([-0-9.%, #a-zA-Z]+\\)';
    /** @type {?} */
    const SAFE_STYLE_VALUE = new RegExp(`^(${VALUES}|` +
        `(?:${TRANSFORMATION_FNS}|${COLOR_FNS}|${GRADIENTS}|${CSS3_FNS})` +
        `${FN_ARGS})$`, 'g');
    /**
     * Matches a `url(...)` value with an arbitrary argument as long as it does
     * not contain parentheses.
     *
     * The URL value still needs to be sanitized separately.
     *
     * `url(...)` values are a very common use case, e.g. for `background-image`. With carefully crafted
     * CSS style rules, it is possible to construct an information leak with `url` values in CSS, e.g.
     * by observing whether scroll bars are displayed, or character ranges used by a font face
     * definition.
     *
     * Angular only allows binding CSS values (as opposed to entire CSS rules), so it is unlikely that
     * binding a URL value without further cooperation from the page will cause an information leak, and
     * if so, it is just a leak, not a full blown XSS vulnerability.
     *
     * Given the common use case, low likelihood of attack vector, and low impact of an attack, this
     * code is permissive and allows URLs that sanitize otherwise.
     * @type {?}
     */
    const URL_RE = /^url\(([^)]+)\)$/;
    /**
     * Checks that quotes (" and ') are properly balanced inside a string. Assumes
     * that neither escape (\) nor any other character that could result in
     * breaking out of a string parsing context are allowed;
     * see http://www.w3.org/TR/css3-syntax/#string-token-diagram.
     *
     * This code was taken from the Closure sanitization library.
     * @param {?} value
     * @return {?}
     */
    function hasBalancedQuotes(value) {
        /** @type {?} */
        let outsideSingle = true;
        /** @type {?} */
        let outsideDouble = true;
        for (let i = 0; i < value.length; i++) {
            /** @type {?} */
            const c = value.charAt(i);
            if (c === '\'' && outsideDouble) {
                outsideSingle = !outsideSingle;
            }
            else if (c === '"' && outsideSingle) {
                outsideDouble = !outsideDouble;
            }
        }
        return outsideSingle && outsideDouble;
    }
    /**
     * Sanitizes the given untrusted CSS style property value (i.e. not an entire object, just a single
     * value) and returns a value that is safe to use in a browser environment.
     * @param {?} value
     * @return {?}
     */
    function _sanitizeStyle(value) {
        value = String(value).trim(); // Make sure it's actually a string.
        if (!value)
            return '';
        // Single url(...) values are supported, but only for URLs that sanitize cleanly. See above for
        // reasoning behind this.
        /** @type {?} */
        const urlMatch = value.match(URL_RE);
        if ((urlMatch && _sanitizeUrl(urlMatch[1]) === urlMatch[1]) ||
            value.match(SAFE_STYLE_VALUE) && hasBalancedQuotes(value)) {
            return value; // Safe style values.
        }
        if (isDevMode()) {
            console.warn(`WARNING: sanitizing unsafe style value ${value} (see http://g.co/ng/security#xss).`);
        }
        return 'unsafe';
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/sanitization/sanitization.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * An `html` sanitizer which converts untrusted `html` **string** into trusted string by removing
     * dangerous content.
     *
     * This method parses the `html` and locates potentially dangerous content (such as urls and
     * javascript) and removes it.
     *
     * It is possible to mark a string as trusted by calling {\@link bypassSanitizationTrustHtml}.
     *
     * \@publicApi
     * @param {?} unsafeHtml untrusted `html`, typically from the user.
     * @return {?} `html` string which is safe to display to user, because all of the dangerous javascript
     * and urls have been removed.
     *
     */
    function ɵɵsanitizeHtml(unsafeHtml) {
        /** @type {?} */
        const sanitizer = getSanitizer();
        if (sanitizer) {
            return sanitizer.sanitize(SecurityContext.HTML, unsafeHtml) || '';
        }
        if (allowSanitizationBypassAndThrow(unsafeHtml, "HTML" /* Html */)) {
            return unwrapSafeValue(unsafeHtml);
        }
        return _sanitizeHtml(getDocument(), renderStringify(unsafeHtml));
    }
    /**
     * A `style` sanitizer which converts untrusted `style` **string** into trusted string by removing
     * dangerous content.
     *
     * This method parses the `style` and locates potentially dangerous content (such as urls and
     * javascript) and removes it.
     *
     * It is possible to mark a string as trusted by calling {\@link bypassSanitizationTrustStyle}.
     *
     * \@publicApi
     * @param {?} unsafeStyle untrusted `style`, typically from the user.
     * @return {?} `style` string which is safe to bind to the `style` properties, because all of the
     * dangerous javascript and urls have been removed.
     *
     */
    function ɵɵsanitizeStyle(unsafeStyle) {
        /** @type {?} */
        const sanitizer = getSanitizer();
        if (sanitizer) {
            return sanitizer.sanitize(SecurityContext.STYLE, unsafeStyle) || '';
        }
        if (allowSanitizationBypassAndThrow(unsafeStyle, "Style" /* Style */)) {
            return unwrapSafeValue(unsafeStyle);
        }
        return _sanitizeStyle(renderStringify(unsafeStyle));
    }
    /**
     * A `url` sanitizer which converts untrusted `url` **string** into trusted string by removing
     * dangerous
     * content.
     *
     * This method parses the `url` and locates potentially dangerous content (such as javascript) and
     * removes it.
     *
     * It is possible to mark a string as trusted by calling {\@link bypassSanitizationTrustUrl}.
     *
     * \@publicApi
     * @param {?} unsafeUrl untrusted `url`, typically from the user.
     * @return {?} `url` string which is safe to bind to the `src` properties such as `<img src>`, because
     * all of the dangerous javascript has been removed.
     *
     */
    function ɵɵsanitizeUrl(unsafeUrl) {
        /** @type {?} */
        const sanitizer = getSanitizer();
        if (sanitizer) {
            return sanitizer.sanitize(SecurityContext.URL, unsafeUrl) || '';
        }
        if (allowSanitizationBypassAndThrow(unsafeUrl, "URL" /* Url */)) {
            return unwrapSafeValue(unsafeUrl);
        }
        return _sanitizeUrl(renderStringify(unsafeUrl));
    }
    /**
     * A `url` sanitizer which only lets trusted `url`s through.
     *
     * This passes only `url`s marked trusted by calling {\@link bypassSanitizationTrustResourceUrl}.
     *
     * \@publicApi
     * @param {?} unsafeResourceUrl untrusted `url`, typically from the user.
     * @return {?} `url` string which is safe to bind to the `src` properties such as `<img src>`, because
     * only trusted `url`s have been allowed to pass.
     *
     */
    function ɵɵsanitizeResourceUrl(unsafeResourceUrl) {
        /** @type {?} */
        const sanitizer = getSanitizer();
        if (sanitizer) {
            return sanitizer.sanitize(SecurityContext.RESOURCE_URL, unsafeResourceUrl) || '';
        }
        if (allowSanitizationBypassAndThrow(unsafeResourceUrl, "ResourceURL" /* ResourceUrl */)) {
            return unwrapSafeValue(unsafeResourceUrl);
        }
        throw new Error('unsafe value used in a resource URL context (see http://g.co/ng/security#xss)');
    }
    /**
     * A `script` sanitizer which only lets trusted javascript through.
     *
     * This passes only `script`s marked trusted by calling {\@link
     * bypassSanitizationTrustScript}.
     *
     * \@publicApi
     * @param {?} unsafeScript untrusted `script`, typically from the user.
     * @return {?} `url` string which is safe to bind to the `<script>` element such as `<img src>`,
     * because only trusted `scripts` have been allowed to pass.
     *
     */
    function ɵɵsanitizeScript(unsafeScript) {
        /** @type {?} */
        const sanitizer = getSanitizer();
        if (sanitizer) {
            return sanitizer.sanitize(SecurityContext.SCRIPT, unsafeScript) || '';
        }
        if (allowSanitizationBypassAndThrow(unsafeScript, "Script" /* Script */)) {
            return unwrapSafeValue(unsafeScript);
        }
        throw new Error('unsafe value used in a script context');
    }
    /**
     * Detects which sanitizer to use for URL property, based on tag name and prop name.
     *
     * The rules are based on the RESOURCE_URL context config from
     * `packages/compiler/src/schema/dom_security_schema.ts`.
     * If tag and prop names don't match Resource URL schema, use URL sanitizer.
     * @param {?} tag
     * @param {?} prop
     * @return {?}
     */
    function getUrlSanitizer(tag, prop) {
        if ((prop === 'src' &&
            (tag === 'embed' || tag === 'frame' || tag === 'iframe' || tag === 'media' ||
                tag === 'script')) ||
            (prop === 'href' && (tag === 'base' || tag === 'link'))) {
            return ɵɵsanitizeResourceUrl;
        }
        return ɵɵsanitizeUrl;
    }
    /**
     * Sanitizes URL, selecting sanitizer function based on tag and property names.
     *
     * This function is used in case we can't define security context at compile time, when only prop
     * name is available. This happens when we generate host bindings for Directives/Components. The
     * host element is unknown at compile time, so we defer calculation of specific sanitizer to
     * runtime.
     *
     * \@publicApi
     * @param {?} unsafeUrl untrusted `url`, typically from the user.
     * @param {?} tag target element tag name.
     * @param {?} prop name of the property that contains the value.
     * @return {?} `url` string which is safe to bind.
     *
     */
    function ɵɵsanitizeUrlOrResourceUrl(unsafeUrl, tag, prop) {
        return getUrlSanitizer(tag, prop)(unsafeUrl);
    }
    /**
     * The default style sanitizer will handle sanitization for style properties by
     * sanitizing any CSS property that can include a `url` value (usually image-based properties)
     *
     * \@publicApi
     * @type {?}
     */
    const ɵɵdefaultStyleSanitizer = ((/** @type {?} */ ((/**
     * @param {?} prop
     * @param {?} value
     * @param {?=} mode
     * @return {?}
     */
    function (prop, value, mode) {
        if (value === undefined && mode === undefined) {
            // This is a workaround for the fact that `StyleSanitizeFn` should not exist once PR#34480
            // lands. For now the `StyleSanitizeFn` and should act like `(value: any) => string` as a
            // work around.
            return ɵɵsanitizeStyle(prop);
        }
        mode = mode || 3 /* ValidateAndSanitize */;
        /** @type {?} */
        let doSanitizeValue = true;
        if (mode & 1 /* ValidateProperty */) {
            doSanitizeValue = stylePropNeedsSanitization(prop);
        }
        if (mode & 2 /* SanitizeOnly */) {
            return doSanitizeValue ? ɵɵsanitizeStyle(value) : unwrapSafeValue(value);
        }
        else {
            return doSanitizeValue;
        }
    }))));
    /**
     * @param {?} prop
     * @return {?}
     */
    function stylePropNeedsSanitization(prop) {
        return prop === 'background-image' || prop === 'backgroundImage' || prop === 'background' ||
            prop === 'border-image' || prop === 'borderImage' || prop === 'border-image-source' ||
            prop === 'borderImageSource' || prop === 'filter' || prop === 'list-style' ||
            prop === 'listStyle' || prop === 'list-style-image' || prop === 'listStyleImage' ||
            prop === 'clip-path' || prop === 'clipPath';
    }
    /**
     * @param {?} name
     * @return {?}
     */
    function validateAgainstEventProperties(name) {
        if (name.toLowerCase().startsWith('on')) {
            /** @type {?} */
            const msg = `Binding to event property '${name}' is disallowed for security reasons, ` +
                `please use (${name.slice(2)})=...` +
                `\nIf '${name}' is a directive input, make sure the directive is imported by the` +
                ` current module.`;
            throw new Error(msg);
        }
    }
    /**
     * @param {?} name
     * @return {?}
     */
    function validateAgainstEventAttributes(name) {
        if (name.toLowerCase().startsWith('on')) {
            /** @type {?} */
            const msg = `Binding to event attribute '${name}' is disallowed for security reasons, ` +
                `please use (${name.slice(2)})=...`;
            throw new Error(msg);
        }
    }
    /**
     * @return {?}
     */
    function getSanitizer() {
        /** @type {?} */
        const lView = getLView();
        return lView && lView[SANITIZER];
    }

    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * THIS FILE CONTAINS CODE WHICH SHOULD BE TREE SHAKEN AND NEVER CALLED FROM PRODUCTION CODE!!!
     */
    /**
     * Creates an `Array` construction with a given name. This is useful when
     * looking for memory consumption to see what time of array it is.
     *
     *
     * @param name Name to give to the constructor
     * @returns A subclass of `Array` if possible. This can only be done in
     *          environments which support `class` construct.
     */
    function createNamedArrayType(name) {
        // This should never be called in prod mode, so let's verify that is the case.
        if (ngDevMode) {
            try {
                // We need to do it this way so that TypeScript does not down-level the below code.
                const FunctionConstructor = createNamedArrayType.constructor;
                return (new FunctionConstructor('Array', `return class ${name} extends Array{}`))(Array);
            }
            catch (e) {
                // If it does not work just give up and fall back to regular Array.
                return Array;
            }
        }
        else {
            throw new Error('Looks like we are in \'prod mode\', but we are creating a named Array type, which is wrong! Check your code');
        }
    }

    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    function normalizeDebugBindingName(name) {
        // Attribute names with `$` (eg `x-y$`) are valid per spec, but unsupported by some browsers
        name = camelCaseToDashCase(name.replace(/[$@]/g, '_'));
        return `ng-reflect-${name}`;
    }
    const CAMEL_CASE_REGEXP = /([A-Z])/g;
    function camelCaseToDashCase(input) {
        return input.replace(CAMEL_CASE_REGEXP, (...m) => '-' + m[1].toLowerCase());
    }
    function normalizeDebugBindingValue(value) {
        try {
            // Limit the size of the value as otherwise the DOM just gets polluted.
            return value != null ? value.toString().slice(0, 30) : value;
        }
        catch (e) {
            return '[ERROR] Exception while trying to serialize the value';
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/context_discovery.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Returns the matching `LContext` data for a given DOM node, directive or component instance.
     *
     * This function will examine the provided DOM element, component, or directive instance\'s
     * monkey-patched property to derive the `LContext` data. Once called then the monkey-patched
     * value will be that of the newly created `LContext`.
     *
     * If the monkey-patched value is the `LView` instance then the context value for that
     * target will be created and the monkey-patch reference will be updated. Therefore when this
     * function is called it may mutate the provided element\'s, component\'s or any of the associated
     * directive\'s monkey-patch values.
     *
     * If the monkey-patch value is not detected then the code will walk up the DOM until an element
     * is found which contains a monkey-patch reference. When that occurs then the provided element
     * will be updated with a new context (which is then returned). If the monkey-patch value is not
     * detected for a component/directive instance then it will throw an error (all components and
     * directives should be automatically monkey-patched by ivy).
     *
     * @param {?} target Component, Directive or DOM Node.
     * @return {?}
     */
    function getLContext(target) {
        /** @type {?} */
        let mpValue = readPatchedData(target);
        if (mpValue) {
            // only when it's an array is it considered an LView instance
            // ... otherwise it's an already constructed LContext instance
            if (Array.isArray(mpValue)) {
                /** @type {?} */
                const lView = (/** @type {?} */ (mpValue));
                /** @type {?} */
                let nodeIndex;
                /** @type {?} */
                let component = undefined;
                /** @type {?} */
                let directives = undefined;
                if (isComponentInstance(target)) {
                    nodeIndex = findViaComponent(lView, target);
                    if (nodeIndex == -1) {
                        throw new Error('The provided component was not found in the application');
                    }
                    component = target;
                }
                else if (isDirectiveInstance(target)) {
                    nodeIndex = findViaDirective(lView, target);
                    if (nodeIndex == -1) {
                        throw new Error('The provided directive was not found in the application');
                    }
                    directives = getDirectivesAtNodeIndex(nodeIndex, lView, false);
                }
                else {
                    nodeIndex = findViaNativeElement(lView, (/** @type {?} */ (target)));
                    if (nodeIndex == -1) {
                        return null;
                    }
                }
                // the goal is not to fill the entire context full of data because the lookups
                // are expensive. Instead, only the target data (the element, component, container, ICU
                // expression or directive details) are filled into the context. If called multiple times
                // with different target values then the missing target data will be filled in.
                /** @type {?} */
                const native = unwrapRNode(lView[nodeIndex]);
                /** @type {?} */
                const existingCtx = readPatchedData(native);
                /** @type {?} */
                const context = (existingCtx && !Array.isArray(existingCtx)) ?
                    existingCtx :
                    createLContext(lView, nodeIndex, native);
                // only when the component has been discovered then update the monkey-patch
                if (component && context.component === undefined) {
                    context.component = component;
                    attachPatchData(context.component, context);
                }
                // only when the directives have been discovered then update the monkey-patch
                if (directives && context.directives === undefined) {
                    context.directives = directives;
                    for (let i = 0; i < directives.length; i++) {
                        attachPatchData(directives[i], context);
                    }
                }
                attachPatchData(context.native, context);
                mpValue = context;
            }
        }
        else {
            /** @type {?} */
            const rElement = (/** @type {?} */ (target));
            ngDevMode && assertDomNode(rElement);
            // if the context is not found then we need to traverse upwards up the DOM
            // to find the nearest element that has already been monkey patched with data
            /** @type {?} */
            let parent = (/** @type {?} */ (rElement));
            while (parent = parent.parentNode) {
                /** @type {?} */
                const parentContext = readPatchedData(parent);
                if (parentContext) {
                    /** @type {?} */
                    let lView;
                    if (Array.isArray(parentContext)) {
                        lView = (/** @type {?} */ (parentContext));
                    }
                    else {
                        lView = parentContext.lView;
                    }
                    // the edge of the app was also reached here through another means
                    // (maybe because the DOM was changed manually).
                    if (!lView) {
                        return null;
                    }
                    /** @type {?} */
                    const index = findViaNativeElement(lView, rElement);
                    if (index >= 0) {
                        /** @type {?} */
                        const native = unwrapRNode(lView[index]);
                        /** @type {?} */
                        const context = createLContext(lView, index, native);
                        attachPatchData(native, context);
                        mpValue = context;
                        break;
                    }
                }
            }
        }
        return ((/** @type {?} */ (mpValue))) || null;
    }
    /**
     * Creates an empty instance of a `LContext` context
     * @param {?} lView
     * @param {?} nodeIndex
     * @param {?} native
     * @return {?}
     */
    function createLContext(lView, nodeIndex, native) {
        return {
            lView,
            nodeIndex,
            native,
            component: undefined,
            directives: undefined,
            localRefs: undefined,
        };
    }
    /**
     * Takes a component instance and returns the view for that component.
     *
     * @param {?} componentInstance
     * @return {?} The component's view
     */
    function getComponentViewByInstance(componentInstance) {
        /** @type {?} */
        let lView = readPatchedData(componentInstance);
        /** @type {?} */
        let view;
        if (Array.isArray(lView)) {
            /** @type {?} */
            const nodeIndex = findViaComponent(lView, componentInstance);
            view = getComponentLViewByIndex(nodeIndex, lView);
            /** @type {?} */
            const context = createLContext(lView, nodeIndex, (/** @type {?} */ (view[HOST])));
            context.component = componentInstance;
            attachPatchData(componentInstance, context);
            attachPatchData(context.native, context);
        }
        else {
            /** @type {?} */
            const context = (/** @type {?} */ ((/** @type {?} */ (lView))));
            view = getComponentLViewByIndex(context.nodeIndex, context.lView);
        }
        return view;
    }
    /**
     * Assigns the given data to the given target (which could be a component,
     * directive or DOM node instance) using monkey-patching.
     * @param {?} target
     * @param {?} data
     * @return {?}
     */
    function attachPatchData(target, data) {
        target[MONKEY_PATCH_KEY_NAME] = data;
    }
    /**
     * @param {?} instance
     * @return {?}
     */
    function isComponentInstance(instance) {
        return instance && instance.constructor && instance.constructor.ɵcmp;
    }
    /**
     * @param {?} instance
     * @return {?}
     */
    function isDirectiveInstance(instance) {
        return instance && instance.constructor && instance.constructor.ɵdir;
    }
    /**
     * Locates the element within the given LView and returns the matching index
     * @param {?} lView
     * @param {?} target
     * @return {?}
     */
    function findViaNativeElement(lView, target) {
        /** @type {?} */
        let tNode = lView[TVIEW].firstChild;
        while (tNode) {
            /** @type {?} */
            const native = (/** @type {?} */ (getNativeByTNodeOrNull(tNode, lView)));
            if (native === target) {
                return tNode.index;
            }
            tNode = traverseNextElement(tNode);
        }
        return -1;
    }
    /**
     * Locates the next tNode (child, sibling or parent).
     * @param {?} tNode
     * @return {?}
     */
    function traverseNextElement(tNode) {
        if (tNode.child) {
            return tNode.child;
        }
        else if (tNode.next) {
            return tNode.next;
        }
        else {
            // Let's take the following template: <div><span>text</span></div><component/>
            // After checking the text node, we need to find the next parent that has a "next" TNode,
            // in this case the parent `div`, so that we can find the component.
            while (tNode.parent && !tNode.parent.next) {
                tNode = tNode.parent;
            }
            return tNode.parent && tNode.parent.next;
        }
    }
    /**
     * Locates the component within the given LView and returns the matching index
     * @param {?} lView
     * @param {?} componentInstance
     * @return {?}
     */
    function findViaComponent(lView, componentInstance) {
        /** @type {?} */
        const componentIndices = lView[TVIEW].components;
        if (componentIndices) {
            for (let i = 0; i < componentIndices.length; i++) {
                /** @type {?} */
                const elementComponentIndex = componentIndices[i];
                /** @type {?} */
                const componentView = getComponentLViewByIndex(elementComponentIndex, lView);
                if (componentView[CONTEXT] === componentInstance) {
                    return elementComponentIndex;
                }
            }
        }
        else {
            /** @type {?} */
            const rootComponentView = getComponentLViewByIndex(HEADER_OFFSET, lView);
            /** @type {?} */
            const rootComponent = rootComponentView[CONTEXT];
            if (rootComponent === componentInstance) {
                // we are dealing with the root element here therefore we know that the
                // element is the very first element after the HEADER data in the lView
                return HEADER_OFFSET;
            }
        }
        return -1;
    }
    /**
     * Locates the directive within the given LView and returns the matching index
     * @param {?} lView
     * @param {?} directiveInstance
     * @return {?}
     */
    function findViaDirective(lView, directiveInstance) {
        // if a directive is monkey patched then it will (by default)
        // have a reference to the LView of the current view. The
        // element bound to the directive being search lives somewhere
        // in the view data. We loop through the nodes and check their
        // list of directives for the instance.
        /** @type {?} */
        let tNode = lView[TVIEW].firstChild;
        while (tNode) {
            /** @type {?} */
            const directiveIndexStart = tNode.directiveStart;
            /** @type {?} */
            const directiveIndexEnd = tNode.directiveEnd;
            for (let i = directiveIndexStart; i < directiveIndexEnd; i++) {
                if (lView[i] === directiveInstance) {
                    return tNode.index;
                }
            }
            tNode = traverseNextElement(tNode);
        }
        return -1;
    }
    /**
     * Returns a list of directives extracted from the given view based on the
     * provided list of directive index values.
     *
     * @param {?} nodeIndex The node index
     * @param {?} lView The target view data
     * @param {?} includeComponents Whether or not to include components in returned directives
     * @return {?}
     */
    function getDirectivesAtNodeIndex(nodeIndex, lView, includeComponents) {
        /** @type {?} */
        const tNode = (/** @type {?} */ (lView[TVIEW].data[nodeIndex]));
        /** @type {?} */
        let directiveStartIndex = tNode.directiveStart;
        if (directiveStartIndex == 0)
            return EMPTY_ARRAY;
        /** @type {?} */
        const directiveEndIndex = tNode.directiveEnd;
        if (!includeComponents && tNode.flags & 2 /* isComponentHost */)
            directiveStartIndex++;
        return lView.slice(directiveStartIndex, directiveEndIndex);
    }
    /**
     * @param {?} nodeIndex
     * @param {?} lView
     * @return {?}
     */
    function getComponentAtNodeIndex(nodeIndex, lView) {
        /** @type {?} */
        const tNode = (/** @type {?} */ (lView[TVIEW].data[nodeIndex]));
        /** @type {?} */
        let directiveStartIndex = tNode.directiveStart;
        return tNode.flags & 2 /* isComponentHost */ ? lView[directiveStartIndex] : null;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/errors.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Called when directives inject each other (creating a circular dependency)
     * @param {?} token
     * @return {?}
     */
    function throwCyclicDependencyError(token) {
        throw new Error(`Cannot instantiate cyclic dependency! ${token}`);
    }
    /**
     * Called when there are multiple component selectors that match a given node
     * @param {?} tNode
     * @return {?}
     */
    function throwMultipleComponentError(tNode) {
        throw new Error(`Multiple components match node with tagname ${tNode.tagName}`);
    }
    /**
     * @return {?}
     */
    function throwMixedMultiProviderError() {
        throw new Error(`Cannot mix multi providers and regular providers`);
    }
    /**
     * @param {?=} ngModuleType
     * @param {?=} providers
     * @param {?=} provider
     * @return {?}
     */
    function throwInvalidProviderError(ngModuleType, providers, provider) {
        /** @type {?} */
        let ngModuleDetail = '';
        if (ngModuleType && providers) {
            /** @type {?} */
            const providerDetail = providers.map((/**
             * @param {?} v
             * @return {?}
             */
            v => v == provider ? '?' + provider + '?' : '...'));
            ngModuleDetail =
                ` - only instances of Provider and Type are allowed, got: [${providerDetail.join(', ')}]`;
        }
        throw new Error(`Invalid provider for the NgModule '${stringify(ngModuleType)}'` + ngModuleDetail);
    }
    /**
     * Throws an ExpressionChangedAfterChecked error if checkNoChanges mode is on.
     * @param {?} creationMode
     * @param {?} oldValue
     * @param {?} currValue
     * @param {?=} propName
     * @return {?}
     */
    function throwErrorIfNoChangesMode(creationMode, oldValue, currValue, propName) {
        /** @type {?} */
        const field = propName ? ` for '${propName}'` : '';
        /** @type {?} */
        let msg = `ExpressionChangedAfterItHasBeenCheckedError: Expression has changed after it was checked. Previous value${field}: '${oldValue}'. Current value: '${currValue}'.`;
        if (creationMode) {
            msg +=
                ` It seems like the view has been created after its parent and its children have been dirty checked.` +
                    ` Has it been created in a change detection hook?`;
        }
        // TODO: include debug context, see `viewDebugError` function in
        // `packages/core/src/view/errors.ts` for reference.
        throw new Error(msg);
    }
    /**
     * @param {?} lView
     * @param {?} rootIndex
     * @param {?} expressionIndex
     * @param {?} meta
     * @param {?} changedValue
     * @return {?}
     */
    function constructDetailsForInterpolation(lView, rootIndex, expressionIndex, meta, changedValue) {
        const [propName, prefix, ...chunks] = meta.split(INTERPOLATION_DELIMITER);
        /** @type {?} */
        let oldValue = prefix;
        /** @type {?} */
        let newValue = prefix;
        for (let i = 0; i < chunks.length; i++) {
            /** @type {?} */
            const slotIdx = rootIndex + i;
            oldValue += `${lView[slotIdx]}${chunks[i]}`;
            newValue += `${slotIdx === expressionIndex ? changedValue : lView[slotIdx]}${chunks[i]}`;
        }
        return { propName, oldValue, newValue };
    }
    /**
     * Constructs an object that contains details for the ExpressionChangedAfterItHasBeenCheckedError:
     * - property name (for property bindings or interpolations)
     * - old and new values, enriched using information from metadata
     *
     * More information on the metadata storage format can be found in `storePropertyBindingMetadata`
     * function description.
     * @param {?} lView
     * @param {?} bindingIndex
     * @param {?} oldValue
     * @param {?} newValue
     * @return {?}
     */
    function getExpressionChangedErrorDetails(lView, bindingIndex, oldValue, newValue) {
        /** @type {?} */
        const tData = lView[TVIEW].data;
        /** @type {?} */
        const metadata = tData[bindingIndex];
        if (typeof metadata === 'string') {
            // metadata for property interpolation
            if (metadata.indexOf(INTERPOLATION_DELIMITER) > -1) {
                return constructDetailsForInterpolation(lView, bindingIndex, bindingIndex, metadata, newValue);
            }
            // metadata for property binding
            return { propName: metadata, oldValue, newValue };
        }
        // metadata is not available for this expression, check if this expression is a part of the
        // property interpolation by going from the current binding index left and look for a string that
        // contains INTERPOLATION_DELIMITER, the layout in tView.data for this case will look like this:
        // [..., 'id�Prefix � and � suffix', null, null, null, ...]
        if (metadata === null) {
            /** @type {?} */
            let idx = bindingIndex - 1;
            while (typeof tData[idx] !== 'string' && tData[idx + 1] === null) {
                idx--;
            }
            /** @type {?} */
            const meta = tData[idx];
            if (typeof meta === 'string') {
                /** @type {?} */
                const matches = meta.match(new RegExp(INTERPOLATION_DELIMITER, 'g'));
                // first interpolation delimiter separates property name from interpolation parts (in case of
                // property interpolations), so we subtract one from total number of found delimiters
                if (matches && (matches.length - 1) > bindingIndex - idx) {
                    return constructDetailsForInterpolation(lView, idx, bindingIndex, meta, newValue);
                }
            }
        }
        return { propName: undefined, oldValue, newValue };
    }
    /**
     * Returns `true` if the `TNode` has a directive which has `\@Input()` for `class` binding.
     *
     * ```
     * <div my-dir [class]="exp"></div>
     * ```
     * and
     * ```
     * \@Directive({
     * })
     * class MyDirective {
     * \@Input()
     *   class: string;
     * }
     * ```
     *
     * In the above case it is necessary to write the reconciled styling information into the
     * directive's input.
     *
     * @param {?} tNode
     * @return {?}
     */
    function hasClassInput(tNode) {
        return (tNode.flags & 16 /* hasClassInput */) !== 0;
    }
    /**
     * Returns `true` if the `TNode` has a directive which has `\@Input()` for `style` binding.
     *
     * ```
     * <div my-dir [style]="exp"></div>
     * ```
     * and
     * ```
     * \@Directive({
     * })
     * class MyDirective {
     * \@Input()
     *   class: string;
     * }
     * ```
     *
     * In the above case it is necessary to write the reconciled styling information into the
     * directive's input.
     *
     * @param {?} tNode
     * @return {?}
     */
    function hasStyleInput(tNode) {
        return (tNode.flags & 32 /* hasStyleInput */) !== 0;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/styling/class_differ.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Returns an index of `classToSearch` in `className` taking token boundaries into account.
     *
     * `classIndexOf('AB A', 'A', 0)` will be 3 (not 0 since `AB!==A`)
     *
     * @param {?} className A string containing classes (whitespace separated)
     * @param {?} classToSearch A class name to locate
     * @param {?} startingIndex Starting location of search
     * @return {?} an index of the located class (or -1 if not found)
     */
    function classIndexOf(className, classToSearch, startingIndex) {
        ngDevMode && assertNotEqual(classToSearch, '', 'can not look for "" string.');
        /** @type {?} */
        let end = className.length;
        while (true) {
            /** @type {?} */
            const foundIndex = className.indexOf(classToSearch, startingIndex);
            if (foundIndex === -1)
                return foundIndex;
            if (foundIndex === 0 || className.charCodeAt(foundIndex - 1) <= 32 /* SPACE */) {
                // Ensure that it has leading whitespace
                /** @type {?} */
                const length = classToSearch.length;
                if (foundIndex + length === end ||
                    className.charCodeAt(foundIndex + length) <= 32 /* SPACE */) {
                    // Ensure that it has trailing whitespace
                    return foundIndex;
                }
            }
            // False positive, keep searching from where we left off.
            startingIndex = foundIndex + 1;
        }
    }
    /** @type {?} */
    const NG_TEMPLATE_SELECTOR = 'ng-template';
    /**
     * Search the `TAttributes` to see if it contains `cssClassToMatch` (case insensitive)
     *
     * @param {?} attrs `TAttributes` to search through.
     * @param {?} cssClassToMatch class to match (lowercase)
     * @param {?} isProjectionMode Whether or not class matching should look into the attribute `class` in
     *    addition to the `AttributeMarker.Classes`.
     * @return {?}
     */
    function isCssClassMatching(attrs, cssClassToMatch, isProjectionMode) {
        // TODO(misko): The fact that this function needs to know about `isProjectionMode` seems suspect.
        // It is strange to me that sometimes the class information comes in form of `class` attribute
        // and sometimes in form of `AttributeMarker.Classes`. Some investigation is needed to determine
        // if that is the right behavior.
        ngDevMode &&
            assertEqual(cssClassToMatch, cssClassToMatch.toLowerCase(), 'Class name expected to be lowercase.');
        /** @type {?} */
        let i = 0;
        while (i < attrs.length) {
            /** @type {?} */
            let item = attrs[i++];
            if (isProjectionMode && item === 'class') {
                item = (/** @type {?} */ (attrs[i]));
                if (classIndexOf(item.toLowerCase(), cssClassToMatch, 0) !== -1) {
                    return true;
                }
            }
            else if (item === 1 /* Classes */) {
                // We found the classes section. Start searching for the class.
                while (i < attrs.length && typeof (item = attrs[i++]) == 'string') {
                    // while we have strings
                    if (item.toLowerCase() === cssClassToMatch)
                        return true;
                }
                return false;
            }
        }
        return false;
    }
    /**
     * Checks whether the `tNode` represents an inline template (e.g. `*ngFor`).
     *
     * @param {?} tNode current TNode
     * @return {?}
     */
    function isInlineTemplate(tNode) {
        return tNode.type === 0 /* Container */ && tNode.tagName !== NG_TEMPLATE_SELECTOR;
    }
    /**
     * Function that checks whether a given tNode matches tag-based selector and has a valid type.
     *
     * Matching can be performed in 2 modes: projection mode (when we project nodes) and regular
     * directive matching mode:
     * - in the "directive matching" mode we do _not_ take TContainer's tagName into account if it is
     * different from NG_TEMPLATE_SELECTOR (value different from NG_TEMPLATE_SELECTOR indicates that a
     * tag name was extracted from * syntax so we would match the same directive twice);
     * - in the "projection" mode, we use a tag name potentially extracted from the * syntax processing
     * (applicable to TNodeType.Container only).
     * @param {?} tNode
     * @param {?} currentSelector
     * @param {?} isProjectionMode
     * @return {?}
     */
    function hasTagAndTypeMatch(tNode, currentSelector, isProjectionMode) {
        /** @type {?} */
        const tagNameToCompare = tNode.type === 0 /* Container */ && !isProjectionMode ?
            NG_TEMPLATE_SELECTOR :
            tNode.tagName;
        return currentSelector === tagNameToCompare;
    }
    /**
     * A utility function to match an Ivy node static data against a simple CSS selector
     *
     * @param {?} tNode
     * @param {?} selector The selector to try matching against the node.
     * @param {?} isProjectionMode if `true` we are matching for content projection, otherwise we are doing
     * directive matching.
     * @return {?} true if node matches the selector.
     */
    function isNodeMatchingSelector(tNode, selector, isProjectionMode) {
        ngDevMode && assertDefined(selector[0], 'Selector should have a tag name');
        /** @type {?} */
        let mode = 4 /* ELEMENT */;
        /** @type {?} */
        const nodeAttrs = tNode.attrs || [];
        // Find the index of first attribute that has no value, only a name.
        /** @type {?} */
        const nameOnlyMarkerIdx = getNameOnlyMarkerIndex(nodeAttrs);
        // When processing ":not" selectors, we skip to the next ":not" if the
        // current one doesn't match
        /** @type {?} */
        let skipToNextSelector = false;
        for (let i = 0; i < selector.length; i++) {
            /** @type {?} */
            const current = selector[i];
            if (typeof current === 'number') {
                // If we finish processing a :not selector and it hasn't failed, return false
                if (!skipToNextSelector && !isPositive(mode) && !isPositive(current)) {
                    return false;
                }
                // If we are skipping to the next :not() and this mode flag is positive,
                // it's a part of the current :not() selector, and we should keep skipping
                if (skipToNextSelector && isPositive(current))
                    continue;
                skipToNextSelector = false;
                mode = ((/** @type {?} */ (current))) | (mode & 1 /* NOT */);
                continue;
            }
            if (skipToNextSelector)
                continue;
            if (mode & 4 /* ELEMENT */) {
                mode = 2 /* ATTRIBUTE */ | mode & 1 /* NOT */;
                if (current !== '' && !hasTagAndTypeMatch(tNode, current, isProjectionMode) ||
                    current === '' && selector.length === 1) {
                    if (isPositive(mode))
                        return false;
                    skipToNextSelector = true;
                }
            }
            else {
                /** @type {?} */
                const selectorAttrValue = mode & 8 /* CLASS */ ? current : selector[++i];
                // special case for matching against classes when a tNode has been instantiated with
                // class and style values as separate attribute values (e.g. ['title', CLASS, 'foo'])
                if ((mode & 8 /* CLASS */) && tNode.attrs !== null) {
                    if (!isCssClassMatching(tNode.attrs, (/** @type {?} */ (selectorAttrValue)), isProjectionMode)) {
                        if (isPositive(mode))
                            return false;
                        skipToNextSelector = true;
                    }
                    continue;
                }
                /** @type {?} */
                const attrName = (mode & 8 /* CLASS */) ? 'class' : current;
                /** @type {?} */
                const attrIndexInNode = findAttrIndexInNode(attrName, nodeAttrs, isInlineTemplate(tNode), isProjectionMode);
                if (attrIndexInNode === -1) {
                    if (isPositive(mode))
                        return false;
                    skipToNextSelector = true;
                    continue;
                }
                if (selectorAttrValue !== '') {
                    /** @type {?} */
                    let nodeAttrValue;
                    if (attrIndexInNode > nameOnlyMarkerIdx) {
                        nodeAttrValue = '';
                    }
                    else {
                        ngDevMode &&
                            assertNotEqual(nodeAttrs[attrIndexInNode], 0 /* NamespaceURI */, 'We do not match directives on namespaced attributes');
                        // we lowercase the attribute value to be able to match
                        // selectors without case-sensitivity
                        // (selectors are already in lowercase when generated)
                        nodeAttrValue = ((/** @type {?} */ (nodeAttrs[attrIndexInNode + 1]))).toLowerCase();
                    }
                    /** @type {?} */
                    const compareAgainstClassName = mode & 8 /* CLASS */ ? nodeAttrValue : null;
                    if (compareAgainstClassName &&
                        classIndexOf(compareAgainstClassName, (/** @type {?} */ (selectorAttrValue)), 0) !== -1 ||
                        mode & 2 /* ATTRIBUTE */ && selectorAttrValue !== nodeAttrValue) {
                        if (isPositive(mode))
                            return false;
                        skipToNextSelector = true;
                    }
                }
            }
        }
        return isPositive(mode) || skipToNextSelector;
    }
    /**
     * @param {?} mode
     * @return {?}
     */
    function isPositive(mode) {
        return (mode & 1 /* NOT */) === 0;
    }
    /**
     * Examines the attribute's definition array for a node to find the index of the
     * attribute that matches the given `name`.
     *
     * NOTE: This will not match namespaced attributes.
     *
     * Attribute matching depends upon `isInlineTemplate` and `isProjectionMode`.
     * The following table summarizes which types of attributes we attempt to match:
     *
     * ===========================================================================================================
     * Modes                   | Normal Attributes | Bindings Attributes | Template Attributes | I18n
     * Attributes
     * ===========================================================================================================
     * Inline + Projection     | YES               | YES                 | NO                  | YES
     * -----------------------------------------------------------------------------------------------------------
     * Inline + Directive      | NO                | NO                  | YES                 | NO
     * -----------------------------------------------------------------------------------------------------------
     * Non-inline + Projection | YES               | YES                 | NO                  | YES
     * -----------------------------------------------------------------------------------------------------------
     * Non-inline + Directive  | YES               | YES                 | NO                  | YES
     * ===========================================================================================================
     *
     * @param {?} name the name of the attribute to find
     * @param {?} attrs the attribute array to examine
     * @param {?} isInlineTemplate true if the node being matched is an inline template (e.g. `*ngFor`)
     * rather than a manually expanded template node (e.g `<ng-template>`).
     * @param {?} isProjectionMode true if we are matching against content projection otherwise we are
     * matching against directives.
     * @return {?}
     */
    function findAttrIndexInNode(name, attrs, isInlineTemplate, isProjectionMode) {
        if (attrs === null)
            return -1;
        /** @type {?} */
        let i = 0;
        if (isProjectionMode || !isInlineTemplate) {
            /** @type {?} */
            let bindingsMode = false;
            while (i < attrs.length) {
                /** @type {?} */
                const maybeAttrName = attrs[i];
                if (maybeAttrName === name) {
                    return i;
                }
                else if (maybeAttrName === 3 /* Bindings */ || maybeAttrName === 6 /* I18n */) {
                    bindingsMode = true;
                }
                else if (maybeAttrName === 1 /* Classes */ || maybeAttrName === 2 /* Styles */) {
                    /** @type {?} */
                    let value = attrs[++i];
                    // We should skip classes here because we have a separate mechanism for
                    // matching classes in projection mode.
                    while (typeof value === 'string') {
                        value = attrs[++i];
                    }
                    continue;
                }
                else if (maybeAttrName === 4 /* Template */) {
                    // We do not care about Template attributes in this scenario.
                    break;
                }
                else if (maybeAttrName === 0 /* NamespaceURI */) {
                    // Skip the whole namespaced attribute and value. This is by design.
                    i += 4;
                    continue;
                }
                // In binding mode there are only names, rather than name-value pairs.
                i += bindingsMode ? 1 : 2;
            }
            // We did not match the attribute
            return -1;
        }
        else {
            return matchTemplateAttribute(attrs, name);
        }
    }
    /**
     * @param {?} tNode
     * @param {?} selector
     * @param {?=} isProjectionMode
     * @return {?}
     */
    function isNodeMatchingSelectorList(tNode, selector, isProjectionMode = false) {
        for (let i = 0; i < selector.length; i++) {
            if (isNodeMatchingSelector(tNode, selector[i], isProjectionMode)) {
                return true;
            }
        }
        return false;
    }
    /**
     * @param {?} tNode
     * @return {?}
     */
    function getProjectAsAttrValue(tNode) {
        /** @type {?} */
        const nodeAttrs = tNode.attrs;
        if (nodeAttrs != null) {
            /** @type {?} */
            const ngProjectAsAttrIdx = nodeAttrs.indexOf(5 /* ProjectAs */);
            // only check for ngProjectAs in attribute names, don't accidentally match attribute's value
            // (attribute names are stored at even indexes)
            if ((ngProjectAsAttrIdx & 1) === 0) {
                return (/** @type {?} */ (nodeAttrs[ngProjectAsAttrIdx + 1]));
            }
        }
        return null;
    }
    /**
     * @param {?} nodeAttrs
     * @return {?}
     */
    function getNameOnlyMarkerIndex(nodeAttrs) {
        for (let i = 0; i < nodeAttrs.length; i++) {
            /** @type {?} */
            const nodeAttr = nodeAttrs[i];
            if (isNameOnlyAttributeMarker(nodeAttr)) {
                return i;
            }
        }
        return nodeAttrs.length;
    }
    /**
     * @param {?} attrs
     * @param {?} name
     * @return {?}
     */
    function matchTemplateAttribute(attrs, name) {
        /** @type {?} */
        let i = attrs.indexOf(4 /* Template */);
        if (i > -1) {
            i++;
            while (i < attrs.length) {
                /** @type {?} */
                const attr = attrs[i];
                // Return in case we checked all template attrs and are switching to the next section in the
                // attrs array (that starts with a number that represents an attribute marker).
                if (typeof attr === 'number')
                    return -1;
                if (attr === name)
                    return i;
                i++;
            }
        }
        return -1;
    }
    /**
     * Checks whether a selector is inside a CssSelectorList
     * @param {?} selector Selector to be checked.
     * @param {?} list List in which to look for the selector.
     * @return {?}
     */
    function isSelectorInSelectorList(selector, list) {
        selectorListLoop: for (let i = 0; i < list.length; i++) {
            /** @type {?} */
            const currentSelectorInList = list[i];
            if (selector.length !== currentSelectorInList.length) {
                continue;
            }
            for (let j = 0; j < selector.length; j++) {
                if (selector[j] !== currentSelectorInList[j]) {
                    continue selectorListLoop;
                }
            }
            return true;
        }
        return false;
    }
    /**
     * @param {?} isNegativeMode
     * @param {?} chunk
     * @return {?}
     */
    function maybeWrapInNotSelector(isNegativeMode, chunk) {
        return isNegativeMode ? ':not(' + chunk.trim() + ')' : chunk;
    }
    /**
     * @param {?} selector
     * @return {?}
     */
    function stringifyCSSSelector(selector) {
        /** @type {?} */
        let result = (/** @type {?} */ (selector[0]));
        /** @type {?} */
        let i = 1;
        /** @type {?} */
        let mode = 2 /* ATTRIBUTE */;
        /** @type {?} */
        let currentChunk = '';
        /** @type {?} */
        let isNegativeMode = false;
        while (i < selector.length) {
            /** @type {?} */
            let valueOrMarker = selector[i];
            if (typeof valueOrMarker === 'string') {
                if (mode & 2 /* ATTRIBUTE */) {
                    /** @type {?} */
                    const attrValue = (/** @type {?} */ (selector[++i]));
                    currentChunk +=
                        '[' + valueOrMarker + (attrValue.length > 0 ? '="' + attrValue + '"' : '') + ']';
                }
                else if (mode & 8 /* CLASS */) {
                    currentChunk += '.' + valueOrMarker;
                }
                else if (mode & 4 /* ELEMENT */) {
                    currentChunk += ' ' + valueOrMarker;
                }
            }
            else {
                //
                // Append current chunk to the final result in case we come across SelectorFlag, which
                // indicates that the previous section of a selector is over. We need to accumulate content
                // between flags to make sure we wrap the chunk later in :not() selector if needed, e.g.
                // ```
                //  ['', Flags.CLASS, '.classA', Flags.CLASS | Flags.NOT, '.classB', '.classC']
                // ```
                // should be transformed to `.classA :not(.classB .classC)`.
                //
                // Note: for negative selector part, we accumulate content between flags until we find the
                // next negative flag. This is needed to support a case where `:not()` rule contains more than
                // one chunk, e.g. the following selector:
                // ```
                //  ['', Flags.ELEMENT | Flags.NOT, 'p', Flags.CLASS, 'foo', Flags.CLASS | Flags.NOT, 'bar']
                // ```
                // should be stringified to `:not(p.foo) :not(.bar)`
                //
                if (currentChunk !== '' && !isPositive(valueOrMarker)) {
                    result += maybeWrapInNotSelector(isNegativeMode, currentChunk);
                    currentChunk = '';
                }
                mode = valueOrMarker;
                // According to CssSelector spec, once we come across `SelectorFlags.NOT` flag, the negative
                // mode is maintained for remaining chunks of a selector.
                isNegativeMode = isNegativeMode || !isPositive(mode);
            }
            i++;
        }
        if (currentChunk !== '') {
            result += maybeWrapInNotSelector(isNegativeMode, currentChunk);
        }
        return result;
    }
    /**
     * Generates string representation of CSS selector in parsed form.
     *
     * ComponentDef and DirectiveDef are generated with the selector in parsed form to avoid doing
     * additional parsing at runtime (for example, for directive matching). However in some cases (for
     * example, while bootstrapping a component), a string version of the selector is required to query
     * for the host element on the page. This function takes the parsed form of a selector and returns
     * its string representation.
     *
     * @param {?} selectorList selector in parsed form
     * @return {?} string representation of a given selector
     */
    function stringifyCSSSelectorList(selectorList) {
        return selectorList.map(stringifyCSSSelector).join(',');
    }
    /**
     * Extracts attributes and classes information from a given CSS selector.
     *
     * This function is used while creating a component dynamically. In this case, the host element
     * (that is created dynamically) should contain attributes and classes specified in component's CSS
     * selector.
     *
     * @param {?} selector CSS selector in parsed form (in a form of array)
     * @return {?} object with `attrs` and `classes` fields that contain extracted information
     */
    function extractAttrsAndClassesFromSelector(selector) {
        /** @type {?} */
        const attrs = [];
        /** @type {?} */
        const classes = [];
        /** @type {?} */
        let i = 1;
        /** @type {?} */
        let mode = 2 /* ATTRIBUTE */;
        while (i < selector.length) {
            /** @type {?} */
            let valueOrMarker = selector[i];
            if (typeof valueOrMarker === 'string') {
                if (mode === 2 /* ATTRIBUTE */) {
                    if (valueOrMarker !== '') {
                        attrs.push(valueOrMarker, (/** @type {?} */ (selector[++i])));
                    }
                }
                else if (mode === 8 /* CLASS */) {
                    classes.push(valueOrMarker);
                }
            }
            else {
                // According to CssSelector spec, once we come across `SelectorFlags.NOT` flag, the negative
                // mode is maintained for remaining chunks of a selector. Since attributes and classes are
                // extracted only for "positive" part of the selector, we can stop here.
                if (!isPositive(mode))
                    break;
                mode = valueOrMarker;
            }
            i++;
        }
        return { attrs, classes };
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/tokens.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    // WARNING: interface has both a type and a value, skipping emit
    /**
     * A special value which designates that a value has not changed.
     * @type {?}
     */
    const NO_CHANGE = (typeof ngDevMode === 'undefined' || ngDevMode) ? { __brand__: 'NO_CHANGE' } : ((/** @type {?} */ ({})));

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/util/view_traversal_utils.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Gets the parent LView of the passed LView, if the PARENT is an LContainer, will get the parent of
     * that LContainer, which is an LView
     * @param {?} lView the lView whose parent to get
     * @return {?}
     */
    function getLViewParent(lView) {
        ngDevMode && assertLView(lView);
        /** @type {?} */
        const parent = lView[PARENT];
        return isLContainer(parent) ? (/** @type {?} */ (parent[PARENT])) : parent;
    }
    /**
     * Retrieve the root view from any component or `LView` by walking the parent `LView` until
     * reaching the root `LView`.
     *
     * @param {?} componentOrLView any component or `LView`
     * @return {?}
     */
    function getRootView(componentOrLView) {
        ngDevMode && assertDefined(componentOrLView, 'component');
        /** @type {?} */
        let lView = isLView(componentOrLView) ? componentOrLView : (/** @type {?} */ (readPatchedLView(componentOrLView)));
        while (lView && !(lView[FLAGS] & 512 /* IsRoot */)) {
            lView = (/** @type {?} */ (getLViewParent(lView)));
        }
        ngDevMode && assertLView(lView);
        return lView;
    }
    /**
     * Returns the `RootContext` instance that is associated with
     * the application where the target is situated. It does this by walking the parent views until it
     * gets to the root view, then getting the context off of that.
     *
     * @param {?} viewOrComponent the `LView` or component to get the root context for.
     * @return {?}
     */
    function getRootContext(viewOrComponent) {
        /** @type {?} */
        const rootView = getRootView(viewOrComponent);
        ngDevMode &&
            assertDefined(rootView[CONTEXT], 'RootView has no context. Perhaps it is disconnected?');
        return (/** @type {?} */ (rootView[CONTEXT]));
    }
    /**
     * Gets the first `LContainer` in the LView or `null` if none exists.
     * @param {?} lView
     * @return {?}
     */
    function getFirstLContainer(lView) {
        return getNearestLContainer(lView[CHILD_HEAD]);
    }
    /**
     * Gets the next `LContainer` that is a sibling of the given container.
     * @param {?} container
     * @return {?}
     */
    function getNextLContainer(container) {
        return getNearestLContainer(container[NEXT]);
    }
    /**
     * @param {?} viewOrContainer
     * @return {?}
     */
    function getNearestLContainer(viewOrContainer) {
        while (viewOrContainer !== null && !isLContainer(viewOrContainer)) {
            viewOrContainer = viewOrContainer[NEXT];
        }
        return viewOrContainer;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/instructions/advance.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Advances to an element for later binding instructions.
     *
     * Used in conjunction with instructions like {\@link property} to act on elements with specified
     * indices, for example those created with {\@link element} or {\@link elementStart}.
     *
     * ```ts
     * (rf: RenderFlags, ctx: any) => {
     *   if (rf & 1) {
     *     text(0, 'Hello');
     *     text(1, 'Goodbye')
     *     element(2, 'div');
     *   }
     *   if (rf & 2) {
     *     advance(2); // Advance twice to the <div>.
     *     property('title', 'test');
     *   }
     *  }
     * ```
     * \@codeGenApi
     * @param {?} delta Number of elements to advance forwards by.
     *
     * @return {?}
     */
    function ɵɵadvance(delta) {
        ngDevMode && assertGreaterThan(delta, 0, 'Can only advance forward');
        selectIndexInternal(getTView(), getLView(), getSelectedIndex() + delta, getCheckNoChangesMode());
    }
    /**
     * Selects an element for later binding instructions.
     * @deprecated No longer being generated, but still used in unit tests.
     * \@codeGenApi
     * @param {?} index
     * @return {?}
     */
    function ɵɵselect(index) {
        // TODO(misko): Remove this function as it is no longer being used.
        selectIndexInternal(getTView(), getLView(), index, getCheckNoChangesMode());
    }
    /**
     * @param {?} tView
     * @param {?} lView
     * @param {?} index
     * @param {?} checkNoChangesMode
     * @return {?}
     */
    function selectIndexInternal(tView, lView, index, checkNoChangesMode) {
        ngDevMode && assertGreaterThan(index, -1, 'Invalid index');
        ngDevMode && assertDataInRange(lView, index + HEADER_OFFSET);
        // Flush the initial hooks for elements in the view that have been added up to this point.
        // PERF WARNING: do NOT extract this to a separate function without running benchmarks
        if (!checkNoChangesMode) {
            /** @type {?} */
            const hooksInitPhaseCompleted = (lView[FLAGS] & 3 /* InitPhaseStateMask */) === 3 /* InitPhaseCompleted */;
            if (hooksInitPhaseCompleted) {
                /** @type {?} */
                const preOrderCheckHooks = tView.preOrderCheckHooks;
                if (preOrderCheckHooks !== null) {
                    executeCheckHooks(lView, preOrderCheckHooks, index);
                }
            }
            else {
                /** @type {?} */
                const preOrderHooks = tView.preOrderHooks;
                if (preOrderHooks !== null) {
                    executeInitAndCheckHooks(lView, preOrderHooks, 0 /* OnInitHooksToBeRun */, index);
                }
            }
        }
        // We must set the selected index *after* running the hooks, because hooks may have side-effects
        // that cause other template functions to run, thus updating the selected index, which is global
        // state. If we run `setSelectedIndex` *before* we run the hooks, in some cases the selected index
        // will be altered by the time we leave the `ɵɵadvance` instruction.
        setSelectedIndex(index);
    }
    /**
     * Marks that the next string is for element.
     *
     * See `I18nMutateOpCodes` documentation.
     * @type {?}
     */
    const ELEMENT_MARKER = {
        marker: 'element'
    };
    // WARNING: interface has both a type and a value, skipping emit
    /**
     * Marks that the next string is for comment.
     *
     * See `I18nMutateOpCodes` documentation.
     * @type {?}
     */
    const COMMENT_MARKER = {
        marker: 'comment'
    };
    /**
     * @param {?} prev
     * @param {?} next
     * @return {?}
     */
    function toTStylingRange(prev, next) {
        ngDevMode && assertNumberInRange(prev, 0, 32767 /* UNSIGNED_MASK */);
        ngDevMode && assertNumberInRange(next, 0, 32767 /* UNSIGNED_MASK */);
        return (/** @type {?} */ ((prev << 17 /* PREV_SHIFT */ | next << 2 /* NEXT_SHIFT */)));
    }
    /**
     * @param {?} tStylingRange
     * @return {?}
     */
    function getTStylingRangePrev(tStylingRange) {
        ngDevMode && assertNumber(tStylingRange, 'expected number');
        return (((/** @type {?} */ ((/** @type {?} */ (tStylingRange))))) >> 17 /* PREV_SHIFT */) & 32767 /* UNSIGNED_MASK */;
    }
    /**
     * @param {?} tStylingRange
     * @return {?}
     */
    function getTStylingRangePrevDuplicate(tStylingRange) {
        ngDevMode && assertNumber(tStylingRange, 'expected number');
        return (((/** @type {?} */ ((/** @type {?} */ (tStylingRange))))) & 2 /* PREV_DUPLICATE */) ==
            2 /* PREV_DUPLICATE */;
    }
    /**
     * @param {?} tStylingRange
     * @param {?} previous
     * @return {?}
     */
    function setTStylingRangePrev(tStylingRange, previous) {
        ngDevMode && assertNumber(tStylingRange, 'expected number');
        ngDevMode && assertNumberInRange(previous, 0, 32767 /* UNSIGNED_MASK */);
        return (/** @type {?} */ (((((/** @type {?} */ ((/** @type {?} */ (tStylingRange))))) & ~4294836224 /* PREV_MASK */) |
            (previous << 17 /* PREV_SHIFT */))));
    }
    /**
     * @param {?} tStylingRange
     * @return {?}
     */
    function setTStylingRangePrevDuplicate(tStylingRange) {
        ngDevMode && assertNumber(tStylingRange, 'expected number');
        return (/** @type {?} */ ((((/** @type {?} */ ((/** @type {?} */ (tStylingRange))))) | 2 /* PREV_DUPLICATE */)));
    }
    /**
     * @param {?} tStylingRange
     * @return {?}
     */
    function getTStylingRangeNext(tStylingRange) {
        ngDevMode && assertNumber(tStylingRange, 'expected number');
        return (((/** @type {?} */ ((/** @type {?} */ (tStylingRange))))) & 131068 /* NEXT_MASK */) >> 2 /* NEXT_SHIFT */;
    }
    /**
     * @param {?} tStylingRange
     * @param {?} next
     * @return {?}
     */
    function setTStylingRangeNext(tStylingRange, next) {
        ngDevMode && assertNumber(tStylingRange, 'expected number');
        ngDevMode && assertNumberInRange(next, 0, 32767 /* UNSIGNED_MASK */);
        return (/** @type {?} */ (((((/** @type {?} */ ((/** @type {?} */ (tStylingRange))))) & ~131068 /* NEXT_MASK */) | //
            next << 2 /* NEXT_SHIFT */)));
    }
    /**
     * @param {?} tStylingRange
     * @return {?}
     */
    function getTStylingRangeNextDuplicate(tStylingRange) {
        ngDevMode && assertNumber(tStylingRange, 'expected number');
        return (((/** @type {?} */ ((/** @type {?} */ (tStylingRange))))) & 1 /* NEXT_DUPLICATE */) ===
            1 /* NEXT_DUPLICATE */;
    }
    /**
     * @param {?} tStylingRange
     * @return {?}
     */
    function setTStylingRangeNextDuplicate(tStylingRange) {
        ngDevMode && assertNumber(tStylingRange, 'expected number');
        return (/** @type {?} */ ((((/** @type {?} */ ((/** @type {?} */ (tStylingRange))))) | 1 /* NEXT_DUPLICATE */)));
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/util/debug_utils.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     * @param {?} obj
     * @param {?} debug
     * @return {?}
     */
    function attachDebugObject(obj, debug) {
        Object.defineProperty(obj, 'debug', { value: debug, enumerable: false });
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/instructions/lview_debug.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const NG_DEV_MODE = ((typeof ngDevMode === 'undefined' || !!ngDevMode) && initNgDevMode());
    /*
     * This file contains conditionally attached classes which provide human readable (debug) level
     * information for `LView`, `LContainer` and other internal data structures. These data structures
     * are stored internally as array which makes it very difficult during debugging to reason about the
     * current state of the system.
     *
     * Patching the array with extra property does change the array's hidden class' but it does not
     * change the cost of access, therefore this patching should not have significant if any impact in
     * `ngDevMode` mode. (see: https://jsperf.com/array-vs-monkey-patch-array)
     *
     * So instead of seeing:
     * ```
     * Array(30) [Object, 659, null, …]
     * ```
     *
     * You get to see:
     * ```
     * LViewDebug {
     *   views: [...],
     *   flags: {attached: true, ...}
     *   nodes: [
     *     {html: '<div id="123">', ..., nodes: [
     *       {html: '<span>', ..., nodes: null}
     *     ]}
     *   ]
     * }
     * ```
     */
    /** @type {?} */
    let LVIEW_COMPONENT_CACHE;
    /** @type {?} */
    let LVIEW_EMBEDDED_CACHE;
    /** @type {?} */
    let LVIEW_ROOT;
    /**
     * This function clones a blueprint and creates LView.
     *
     * Simple slice will keep the same type, and we need it to be LView
     * @param {?} tView
     * @return {?}
     */
    function cloneToLViewFromTViewBlueprint(tView) {
        /** @type {?} */
        const debugTView = (/** @type {?} */ (tView));
        /** @type {?} */
        const lView = getLViewToClone(debugTView.type, tView.template && tView.template.name);
        return (/** @type {?} */ (lView.concat(tView.blueprint)));
    }
    /**
     * @param {?} type
     * @param {?} name
     * @return {?}
     */
    function getLViewToClone(type, name) {
        switch (type) {
            case 0 /* Root */:
                if (LVIEW_ROOT === undefined)
                    LVIEW_ROOT = new (createNamedArrayType('LRootView'))();
                return LVIEW_ROOT;
            case 1 /* Component */:
                if (LVIEW_COMPONENT_CACHE === undefined)
                    LVIEW_COMPONENT_CACHE = new Map();
                /** @type {?} */
                let componentArray = LVIEW_COMPONENT_CACHE.get(name);
                if (componentArray === undefined) {
                    componentArray = new (createNamedArrayType('LComponentView' + nameSuffix(name)))();
                    LVIEW_COMPONENT_CACHE.set(name, componentArray);
                }
                return componentArray;
            case 2 /* Embedded */:
                if (LVIEW_EMBEDDED_CACHE === undefined)
                    LVIEW_EMBEDDED_CACHE = new Map();
                /** @type {?} */
                let embeddedArray = LVIEW_EMBEDDED_CACHE.get(name);
                if (embeddedArray === undefined) {
                    embeddedArray = new (createNamedArrayType('LEmbeddedView' + nameSuffix(name)))();
                    LVIEW_EMBEDDED_CACHE.set(name, embeddedArray);
                }
                return embeddedArray;
        }
        throw new Error('unreachable code');
    }
    /**
     * @param {?} text
     * @return {?}
     */
    function nameSuffix(text) {
        if (text == null)
            return '';
        /** @type {?} */
        const index = text.lastIndexOf('_Template');
        return '_' + (index === -1 ? text : text.substr(0, index));
    }
    /**
     * This class is a debug version of Object literal so that we can have constructor name show up
     * in
     * debug tools in ngDevMode.
     * @type {?}
     */
    const TViewConstructor = class TView {
        /**
         * @param {?} type
         * @param {?} id
         * @param {?} blueprint
         * @param {?} template
         * @param {?} queries
         * @param {?} viewQuery
         * @param {?} node
         * @param {?} data
         * @param {?} bindingStartIndex
         * @param {?} expandoStartIndex
         * @param {?} expandoInstructions
         * @param {?} firstCreatePass
         * @param {?} firstUpdatePass
         * @param {?} staticViewQueries
         * @param {?} staticContentQueries
         * @param {?} preOrderHooks
         * @param {?} preOrderCheckHooks
         * @param {?} contentHooks
         * @param {?} contentCheckHooks
         * @param {?} viewHooks
         * @param {?} viewCheckHooks
         * @param {?} destroyHooks
         * @param {?} cleanup
         * @param {?} contentQueries
         * @param {?} components
         * @param {?} directiveRegistry
         * @param {?} pipeRegistry
         * @param {?} firstChild
         * @param {?} schemas
         * @param {?} consts
         * @param {?} incompleteFirstPass
         */
        constructor(type, //
        id, //
        blueprint, //
        template, //
        queries, //
        viewQuery, //
        node, //
        data, //
        bindingStartIndex, //
        expandoStartIndex, //
        expandoInstructions, //
        firstCreatePass, //
        firstUpdatePass, //
        staticViewQueries, //
        staticContentQueries, //
        preOrderHooks, //
        preOrderCheckHooks, //
        contentHooks, //
        contentCheckHooks, //
        viewHooks, //
        viewCheckHooks, //
        destroyHooks, //
        cleanup, //
        contentQueries, //
        components, //
        directiveRegistry, //
        pipeRegistry, //
        firstChild, //
        schemas, //
        consts, //
        incompleteFirstPass //
        ) {
            this.type = type;
            this.id = id;
            this.blueprint = blueprint;
            this.template = template;
            this.queries = queries;
            this.viewQuery = viewQuery;
            this.node = node;
            this.data = data;
            this.bindingStartIndex = bindingStartIndex;
            this.expandoStartIndex = expandoStartIndex;
            this.expandoInstructions = expandoInstructions;
            this.firstCreatePass = firstCreatePass;
            this.firstUpdatePass = firstUpdatePass;
            this.staticViewQueries = staticViewQueries;
            this.staticContentQueries = staticContentQueries;
            this.preOrderHooks = preOrderHooks;
            this.preOrderCheckHooks = preOrderCheckHooks;
            this.contentHooks = contentHooks;
            this.contentCheckHooks = contentCheckHooks;
            this.viewHooks = viewHooks;
            this.viewCheckHooks = viewCheckHooks;
            this.destroyHooks = destroyHooks;
            this.cleanup = cleanup;
            this.contentQueries = contentQueries;
            this.components = components;
            this.directiveRegistry = directiveRegistry;
            this.pipeRegistry = pipeRegistry;
            this.firstChild = firstChild;
            this.schemas = schemas;
            this.consts = consts;
            this.incompleteFirstPass = incompleteFirstPass;
        }
        /**
         * @return {?}
         */
        get template_() {
            /** @type {?} */
            const buf = [];
            processTNodeChildren(this.firstChild, buf);
            return buf.join('');
        }
    };
    class TNode$1 {
        /**
         * @param {?} tView_
         * @param {?} type
         * @param {?} index
         * @param {?} injectorIndex
         * @param {?} directiveStart
         * @param {?} directiveEnd
         * @param {?} directiveStylingLast
         * @param {?} propertyBindings
         * @param {?} flags
         * @param {?} providerIndexes
         * @param {?} tagName
         * @param {?} attrs
         * @param {?} mergedAttrs
         * @param {?} localNames
         * @param {?} initialInputs
         * @param {?} inputs
         * @param {?} outputs
         * @param {?} tViews
         * @param {?} next
         * @param {?} projectionNext
         * @param {?} child
         * @param {?} parent
         * @param {?} projection
         * @param {?} styles
         * @param {?} stylesWithoutHost
         * @param {?} residualStyles
         * @param {?} classes
         * @param {?} classesWithoutHost
         * @param {?} residualClasses
         * @param {?} classBindings
         * @param {?} styleBindings
         */
        constructor(tView_, //
        type, //
        index, //
        injectorIndex, //
        directiveStart, //
        directiveEnd, //
        directiveStylingLast, //
        propertyBindings, //
        flags, //
        providerIndexes, //
        tagName, //
        attrs, //
        mergedAttrs, //
        localNames, //
        initialInputs, //
        inputs, //
        outputs, //
        tViews, //
        next, //
        projectionNext, //
        child, //
        parent, //
        projection, //
        styles, //
        stylesWithoutHost, //
        residualStyles, //
        classes, //
        classesWithoutHost, //
        residualClasses, //
        classBindings, //
        styleBindings) {
            this.tView_ = tView_;
            this.type = type;
            this.index = index;
            this.injectorIndex = injectorIndex;
            this.directiveStart = directiveStart;
            this.directiveEnd = directiveEnd;
            this.directiveStylingLast = directiveStylingLast;
            this.propertyBindings = propertyBindings;
            this.flags = flags;
            this.providerIndexes = providerIndexes;
            this.tagName = tagName;
            this.attrs = attrs;
            this.mergedAttrs = mergedAttrs;
            this.localNames = localNames;
            this.initialInputs = initialInputs;
            this.inputs = inputs;
            this.outputs = outputs;
            this.tViews = tViews;
            this.next = next;
            this.projectionNext = projectionNext;
            this.child = child;
            this.parent = parent;
            this.projection = projection;
            this.styles = styles;
            this.stylesWithoutHost = stylesWithoutHost;
            this.residualStyles = residualStyles;
            this.classes = classes;
            this.classesWithoutHost = classesWithoutHost;
            this.residualClasses = residualClasses;
            this.classBindings = classBindings;
            this.styleBindings = styleBindings;
        }
        /**
         * @return {?}
         */
        get type_() {
            switch (this.type) {
                case 0 /* Container */:
                    return 'TNodeType.Container';
                case 3 /* Element */:
                    return 'TNodeType.Element';
                case 4 /* ElementContainer */:
                    return 'TNodeType.ElementContainer';
                case 5 /* IcuContainer */:
                    return 'TNodeType.IcuContainer';
                case 1 /* Projection */:
                    return 'TNodeType.Projection';
                case 2 /* View */:
                    return 'TNodeType.View';
                default:
                    return 'TNodeType.???';
            }
        }
        /**
         * @return {?}
         */
        get flags_() {
            /** @type {?} */
            const flags = [];
            if (this.flags & 16 /* hasClassInput */)
                flags.push('TNodeFlags.hasClassInput');
            if (this.flags & 8 /* hasContentQuery */)
                flags.push('TNodeFlags.hasContentQuery');
            if (this.flags & 32 /* hasStyleInput */)
                flags.push('TNodeFlags.hasStyleInput');
            if (this.flags & 128 /* hasHostBindings */)
                flags.push('TNodeFlags.hasHostBindings');
            if (this.flags & 2 /* isComponentHost */)
                flags.push('TNodeFlags.isComponentHost');
            if (this.flags & 1 /* isDirectiveHost */)
                flags.push('TNodeFlags.isDirectiveHost');
            if (this.flags & 64 /* isDetached */)
                flags.push('TNodeFlags.isDetached');
            if (this.flags & 4 /* isProjected */)
                flags.push('TNodeFlags.isProjected');
            return flags.join('|');
        }
        /**
         * @return {?}
         */
        get template_() {
            /** @type {?} */
            const buf = [];
            buf.push('<', this.tagName || this.type_);
            if (this.attrs) {
                for (let i = 0; i < this.attrs.length;) {
                    /** @type {?} */
                    const attrName = this.attrs[i++];
                    if (typeof attrName == 'number') {
                        break;
                    }
                    /** @type {?} */
                    const attrValue = this.attrs[i++];
                    buf.push(' ', (/** @type {?} */ (attrName)), '="', (/** @type {?} */ (attrValue)), '"');
                }
            }
            buf.push('>');
            processTNodeChildren(this.child, buf);
            buf.push('</', this.tagName || this.type_, '>');
            return buf.join('');
        }
        /**
         * @return {?}
         */
        get styleBindings_() {
            return toDebugStyleBinding(this, false);
        }
        /**
         * @return {?}
         */
        get classBindings_() {
            return toDebugStyleBinding(this, true);
        }
    }
    /** @type {?} */
    const TNodeDebug = TNode$1;
    /**
     * @param {?} tNode
     * @param {?} isClassBased
     * @return {?}
     */
    function toDebugStyleBinding(tNode, isClassBased) {
        /** @type {?} */
        const tData = tNode.tView_.data;
        /** @type {?} */
        const bindings = (/** @type {?} */ ([]));
        /** @type {?} */
        const range = isClassBased ? tNode.classBindings : tNode.styleBindings;
        /** @type {?} */
        const prev = getTStylingRangePrev(range);
        /** @type {?} */
        const next = getTStylingRangeNext(range);
        /** @type {?} */
        let isTemplate = next !== 0;
        /** @type {?} */
        let cursor = isTemplate ? next : prev;
        while (cursor !== 0) {
            /** @type {?} */
            const itemKey = (/** @type {?} */ (tData[cursor]));
            /** @type {?} */
            const itemRange = (/** @type {?} */ (tData[cursor + 1]));
            bindings.unshift({
                key: itemKey,
                index: cursor,
                isTemplate: isTemplate,
                prevDuplicate: getTStylingRangePrevDuplicate(itemRange),
                nextDuplicate: getTStylingRangeNextDuplicate(itemRange),
                nextIndex: getTStylingRangeNext(itemRange),
                prevIndex: getTStylingRangePrev(itemRange),
            });
            if (cursor === prev)
                isTemplate = false;
            cursor = getTStylingRangePrev(itemRange);
        }
        bindings.push((isClassBased ? tNode.residualClasses : tNode.residualStyles) || null);
        return bindings;
    }
    /**
     * @param {?} tNode
     * @param {?} buf
     * @return {?}
     */
    function processTNodeChildren(tNode, buf) {
        while (tNode) {
            buf.push(((/** @type {?} */ ((/** @type {?} */ (tNode))))).template_);
            tNode = tNode.next;
        }
    }
    /** @type {?} */
    const TViewData = NG_DEV_MODE && createNamedArrayType('TViewData') || (/** @type {?} */ ((/** @type {?} */ (null))));
    /** @type {?} */
    let TVIEWDATA_EMPTY;
    // can't initialize here or it will not be tree shaken, because
    // `LView` constructor could have side-effects.
    /**
     * This function clones a blueprint and creates TData.
     *
     * Simple slice will keep the same type, and we need it to be TData
     * @param {?} list
     * @return {?}
     */
    function cloneToTViewData(list) {
        if (TVIEWDATA_EMPTY === undefined)
            TVIEWDATA_EMPTY = new TViewData();
        return (/** @type {?} */ (TVIEWDATA_EMPTY.concat(list)));
    }
    /** @type {?} */
    const LViewBlueprint = NG_DEV_MODE && createNamedArrayType('LViewBlueprint') || (/** @type {?} */ ((/** @type {?} */ (null))));
    /** @type {?} */
    const MatchesArray = NG_DEV_MODE && createNamedArrayType('MatchesArray') || (/** @type {?} */ ((/** @type {?} */ (null))));
    /** @type {?} */
    const TViewComponents = NG_DEV_MODE && createNamedArrayType('TViewComponents') || (/** @type {?} */ ((/** @type {?} */ (null))));
    /** @type {?} */
    const TNodeLocalNames = NG_DEV_MODE && createNamedArrayType('TNodeLocalNames') || (/** @type {?} */ ((/** @type {?} */ (null))));
    /** @type {?} */
    const TNodeInitialInputs = NG_DEV_MODE && createNamedArrayType('TNodeInitialInputs') || (/** @type {?} */ ((/** @type {?} */ (null))));
    /** @type {?} */
    const TNodeInitialData = NG_DEV_MODE && createNamedArrayType('TNodeInitialData') || (/** @type {?} */ ((/** @type {?} */ (null))));
    /** @type {?} */
    const LCleanup = NG_DEV_MODE && createNamedArrayType('LCleanup') || (/** @type {?} */ ((/** @type {?} */ (null))));
    /** @type {?} */
    const TCleanup = NG_DEV_MODE && createNamedArrayType('TCleanup') || (/** @type {?} */ ((/** @type {?} */ (null))));
    /**
     * @param {?} lView
     * @return {?}
     */
    function attachLViewDebug(lView) {
        attachDebugObject(lView, new LViewDebug(lView));
    }
    /**
     * @param {?} lContainer
     * @return {?}
     */
    function attachLContainerDebug(lContainer) {
        attachDebugObject(lContainer, new LContainerDebug(lContainer));
    }
    /**
     * @param {?} obj
     * @return {?}
     */
    function toDebug(obj) {
        if (obj) {
            /** @type {?} */
            const debug = ((/** @type {?} */ (obj))).debug;
            assertDefined(debug, 'Object does not have a debug representation.');
            return debug;
        }
        else {
            return obj;
        }
    }
    /**
     * Use this method to unwrap a native element in `LView` and convert it into HTML for easier
     * reading.
     *
     * @param {?} value possibly wrapped native DOM node.
     * @param {?=} includeChildren If `true` then the serialized HTML form will include child elements
     * (same
     * as `outerHTML`). If `false` then the serialized HTML form will only contain the element
     * itself
     * (will not serialize child elements).
     * @return {?}
     */
    function toHtml(value, includeChildren = false) {
        /** @type {?} */
        const node = (/** @type {?} */ (unwrapRNode(value)));
        if (node) {
            /** @type {?} */
            const isTextNode = node.nodeType === Node.TEXT_NODE;
            /** @type {?} */
            const outerHTML = (isTextNode ? node.textContent : node.outerHTML) || '';
            if (includeChildren || isTextNode) {
                return outerHTML;
            }
            else {
                /** @type {?} */
                const innerHTML = '>' + node.innerHTML + '<';
                return (outerHTML.split(innerHTML)[0]) + '>';
            }
        }
        else {
            return null;
        }
    }
    class LViewDebug {
        /**
         * @param {?} _raw_lView
         */
        constructor(_raw_lView) {
            this._raw_lView = _raw_lView;
        }
        /**
         * Flags associated with the `LView` unpacked into a more readable state.
         * @return {?}
         */
        get flags() {
            /** @type {?} */
            const flags = this._raw_lView[FLAGS];
            return {
                __raw__flags__: flags,
                initPhaseState: flags & 3 /* InitPhaseStateMask */,
                creationMode: !!(flags & 4 /* CreationMode */),
                firstViewPass: !!(flags & 8 /* FirstLViewPass */),
                checkAlways: !!(flags & 16 /* CheckAlways */),
                dirty: !!(flags & 64 /* Dirty */),
                attached: !!(flags & 128 /* Attached */),
                destroyed: !!(flags & 256 /* Destroyed */),
                isRoot: !!(flags & 512 /* IsRoot */),
                indexWithinInitPhase: flags >> 11 /* IndexWithinInitPhaseShift */,
            };
        }
        /**
         * @return {?}
         */
        get parent() {
            return toDebug(this._raw_lView[PARENT]);
        }
        /**
         * @return {?}
         */
        get host() {
            return toHtml(this._raw_lView[HOST], true);
        }
        /**
         * @return {?}
         */
        get html() {
            return (this.nodes || []).map((/**
             * @param {?} node
             * @return {?}
             */
            node => toHtml(node.native, true))).join('');
        }
        /**
         * @return {?}
         */
        get context() {
            return this._raw_lView[CONTEXT];
        }
        /**
         * The tree of nodes associated with the current `LView`. The nodes have been normalized into
         * a
         * tree structure with relevant details pulled out for readability.
         * @return {?}
         */
        get nodes() {
            /** @type {?} */
            const lView = this._raw_lView;
            /** @type {?} */
            const tNode = lView[TVIEW].firstChild;
            return toDebugNodes(tNode, lView);
        }
        /**
         * @return {?}
         */
        get tView() {
            return this._raw_lView[TVIEW];
        }
        /**
         * @return {?}
         */
        get cleanup() {
            return this._raw_lView[CLEANUP];
        }
        /**
         * @return {?}
         */
        get injector() {
            return this._raw_lView[INJECTOR$1];
        }
        /**
         * @return {?}
         */
        get rendererFactory() {
            return this._raw_lView[RENDERER_FACTORY];
        }
        /**
         * @return {?}
         */
        get renderer() {
            return this._raw_lView[RENDERER];
        }
        /**
         * @return {?}
         */
        get sanitizer() {
            return this._raw_lView[SANITIZER];
        }
        /**
         * @return {?}
         */
        get childHead() {
            return toDebug(this._raw_lView[CHILD_HEAD]);
        }
        /**
         * @return {?}
         */
        get next() {
            return toDebug(this._raw_lView[NEXT]);
        }
        /**
         * @return {?}
         */
        get childTail() {
            return toDebug(this._raw_lView[CHILD_TAIL]);
        }
        /**
         * @return {?}
         */
        get declarationView() {
            return toDebug(this._raw_lView[DECLARATION_VIEW]);
        }
        /**
         * @return {?}
         */
        get queries() {
            return this._raw_lView[QUERIES];
        }
        /**
         * @return {?}
         */
        get tHost() {
            return this._raw_lView[T_HOST];
        }
        /**
         * Normalized view of child views (and containers) attached at this location.
         * @return {?}
         */
        get childViews() {
            /** @type {?} */
            const childViews = [];
            /** @type {?} */
            let child = this.childHead;
            while (child) {
                childViews.push(child);
                child = child.next;
            }
            return childViews;
        }
    }
    /**
     * Turns a flat list of nodes into a tree by walking the associated `TNode` tree.
     *
     * @param {?} tNode
     * @param {?} lView
     * @return {?}
     */
    function toDebugNodes(tNode, lView) {
        if (tNode) {
            /** @type {?} */
            const debugNodes = [];
            /** @type {?} */
            let tNodeCursor = tNode;
            while (tNodeCursor) {
                debugNodes.push(buildDebugNode(tNodeCursor, lView, tNodeCursor.index));
                tNodeCursor = tNodeCursor.next;
            }
            return debugNodes;
        }
        else {
            return null;
        }
    }
    /**
     * @param {?} tNode
     * @param {?} lView
     * @param {?} nodeIndex
     * @return {?}
     */
    function buildDebugNode(tNode, lView, nodeIndex) {
        /** @type {?} */
        const rawValue = lView[nodeIndex];
        /** @type {?} */
        const native = unwrapRNode(rawValue);
        /** @type {?} */
        const componentLViewDebug = toDebug(readLViewValue(rawValue));
        return {
            html: toHtml(native),
            native: (/** @type {?} */ (native)),
            nodes: toDebugNodes(tNode.child, lView),
            component: componentLViewDebug,
        };
    }
    class LContainerDebug {
        /**
         * @param {?} _raw_lContainer
         */
        constructor(_raw_lContainer) {
            this._raw_lContainer = _raw_lContainer;
        }
        /**
         * @return {?}
         */
        get hasTransplantedViews() {
            return this._raw_lContainer[HAS_TRANSPLANTED_VIEWS];
        }
        /**
         * @return {?}
         */
        get views() {
            return this._raw_lContainer.slice(CONTAINER_HEADER_OFFSET)
                .map((/** @type {?} */ (toDebug)));
        }
        /**
         * @return {?}
         */
        get parent() {
            return toDebug(this._raw_lContainer[PARENT]);
        }
        /**
         * @return {?}
         */
        get movedViews() {
            return this._raw_lContainer[MOVED_VIEWS];
        }
        /**
         * @return {?}
         */
        get host() {
            return this._raw_lContainer[HOST];
        }
        /**
         * @return {?}
         */
        get native() {
            return this._raw_lContainer[NATIVE];
        }
        /**
         * @return {?}
         */
        get next() {
            return toDebug(this._raw_lContainer[NEXT]);
        }
    }
    /**
     * Return an `LView` value if found.
     *
     * @param {?} value `LView` if any
     * @return {?}
     */
    function readLViewValue(value) {
        while (Array.isArray(value)) {
            // This check is not quite right, as it does not take into account `StylingContext`
            // This is why it is in debug, not in util.ts
            if (value.length >= HEADER_OFFSET - 1)
                return (/** @type {?} */ (value));
            value = value[HOST];
        }
        return null;
    }
    class I18NDebugItem {
        /**
         * @param {?} __raw_opCode
         * @param {?} _lView
         * @param {?} nodeIndex
         * @param {?} type
         */
        constructor(__raw_opCode, _lView, nodeIndex, type) {
            this.__raw_opCode = __raw_opCode;
            this._lView = _lView;
            this.nodeIndex = nodeIndex;
            this.type = type;
        }
        /**
         * @return {?}
         */
        get tNode() {
            return getTNode(this._lView[TVIEW], this.nodeIndex);
        }
    }
    /**
     * Turns a list of "Create" & "Update" OpCodes into a human-readable list of operations for
     * debugging purposes.
     * @param {?} mutateOpCodes mutation opCodes to read
     * @param {?} updateOpCodes update opCodes to read
     * @param {?} icus list of ICU expressions
     * @param {?} lView The view the opCodes are acting on
     * @return {?}
     */
    function attachI18nOpCodesDebug(mutateOpCodes, updateOpCodes, icus, lView) {
        attachDebugObject(mutateOpCodes, new I18nMutateOpCodesDebug(mutateOpCodes, lView));
        attachDebugObject(updateOpCodes, new I18nUpdateOpCodesDebug(updateOpCodes, icus, lView));
        if (icus) {
            icus.forEach((/**
             * @param {?} icu
             * @return {?}
             */
            icu => {
                icu.create.forEach((/**
                 * @param {?} icuCase
                 * @return {?}
                 */
                icuCase => {
                    attachDebugObject(icuCase, new I18nMutateOpCodesDebug(icuCase, lView));
                }));
                icu.update.forEach((/**
                 * @param {?} icuCase
                 * @return {?}
                 */
                icuCase => {
                    attachDebugObject(icuCase, new I18nUpdateOpCodesDebug(icuCase, icus, lView));
                }));
            }));
        }
    }
    class I18nMutateOpCodesDebug {
        /**
         * @param {?} __raw_opCodes
         * @param {?} __lView
         */
        constructor(__raw_opCodes, __lView) {
            this.__raw_opCodes = __raw_opCodes;
            this.__lView = __lView;
        }
        /**
         * A list of operation information about how the OpCodes will act on the view.
         * @return {?}
         */
        get operations() {
            const { __lView, __raw_opCodes } = this;
            /** @type {?} */
            const results = [];
            for (let i = 0; i < __raw_opCodes.length; i++) {
                /** @type {?} */
                const opCode = __raw_opCodes[i];
                /** @type {?} */
                let result;
                if (typeof opCode === 'string') {
                    result = {
                        __raw_opCode: opCode,
                        type: 'Create Text Node',
                        nodeIndex: __raw_opCodes[++i],
                        text: opCode,
                    };
                }
                if (typeof opCode === 'number') {
                    switch (opCode & 7 /* MASK_OPCODE */) {
                        case 1 /* AppendChild */:
                            /** @type {?} */
                            const destinationNodeIndex = opCode >>> 17 /* SHIFT_PARENT */;
                            result = new I18NDebugItem(opCode, __lView, destinationNodeIndex, 'AppendChild');
                            break;
                        case 0 /* Select */:
                            /** @type {?} */
                            const nodeIndex = opCode >>> 3 /* SHIFT_REF */;
                            result = new I18NDebugItem(opCode, __lView, nodeIndex, 'Select');
                            break;
                        case 5 /* ElementEnd */:
                            /** @type {?} */
                            let elementIndex = opCode >>> 3 /* SHIFT_REF */;
                            result = new I18NDebugItem(opCode, __lView, elementIndex, 'ElementEnd');
                            break;
                        case 4 /* Attr */:
                            elementIndex = opCode >>> 3 /* SHIFT_REF */;
                            result = new I18NDebugItem(opCode, __lView, elementIndex, 'Attr');
                            result['attrName'] = __raw_opCodes[++i];
                            result['attrValue'] = __raw_opCodes[++i];
                            break;
                    }
                }
                if (!result) {
                    switch (opCode) {
                        case COMMENT_MARKER:
                            result = {
                                __raw_opCode: opCode,
                                type: 'COMMENT_MARKER',
                                commentValue: __raw_opCodes[++i],
                                nodeIndex: __raw_opCodes[++i],
                            };
                            break;
                        case ELEMENT_MARKER:
                            result = {
                                __raw_opCode: opCode,
                                type: 'ELEMENT_MARKER',
                            };
                            break;
                    }
                }
                if (!result) {
                    result = {
                        __raw_opCode: opCode,
                        type: 'Unknown Op Code',
                        code: opCode,
                    };
                }
                results.push(result);
            }
            return results;
        }
    }
    class I18nUpdateOpCodesDebug {
        /**
         * @param {?} __raw_opCodes
         * @param {?} icus
         * @param {?} __lView
         */
        constructor(__raw_opCodes, icus, __lView) {
            this.__raw_opCodes = __raw_opCodes;
            this.icus = icus;
            this.__lView = __lView;
        }
        /**
         * A list of operation information about how the OpCodes will act on the view.
         * @return {?}
         */
        get operations() {
            const { __lView, __raw_opCodes, icus } = this;
            /** @type {?} */
            const results = [];
            for (let i = 0; i < __raw_opCodes.length; i++) {
                // bit code to check if we should apply the next update
                /** @type {?} */
                const checkBit = (/** @type {?} */ (__raw_opCodes[i]));
                // Number of opCodes to skip until next set of update codes
                /** @type {?} */
                const skipCodes = (/** @type {?} */ (__raw_opCodes[++i]));
                /** @type {?} */
                let value = '';
                for (let j = i + 1; j <= (i + skipCodes); j++) {
                    /** @type {?} */
                    const opCode = __raw_opCodes[j];
                    if (typeof opCode === 'string') {
                        value += opCode;
                    }
                    else if (typeof opCode == 'number') {
                        if (opCode < 0) {
                            // It's a binding index whose value is negative
                            // We cannot know the value of the binding so we only show the index
                            value += `�${-opCode - 1}�`;
                        }
                        else {
                            /** @type {?} */
                            const nodeIndex = opCode >>> 2 /* SHIFT_REF */;
                            /** @type {?} */
                            let tIcuIndex;
                            /** @type {?} */
                            let tIcu;
                            switch (opCode & 3 /* MASK_OPCODE */) {
                                case 1 /* Attr */:
                                    /** @type {?} */
                                    const attrName = (/** @type {?} */ (__raw_opCodes[++j]));
                                    /** @type {?} */
                                    const sanitizeFn = __raw_opCodes[++j];
                                    results.push({
                                        __raw_opCode: opCode,
                                        checkBit,
                                        type: 'Attr',
                                        attrValue: value,
                                        attrName,
                                        sanitizeFn,
                                    });
                                    break;
                                case 0 /* Text */:
                                    results.push({
                                        __raw_opCode: opCode,
                                        checkBit,
                                        type: 'Text',
                                        nodeIndex,
                                        text: value,
                                    });
                                    break;
                                case 2 /* IcuSwitch */:
                                    tIcuIndex = (/** @type {?} */ (__raw_opCodes[++j]));
                                    tIcu = (/** @type {?} */ (icus))[tIcuIndex];
                                    /** @type {?} */
                                    let result = new I18NDebugItem(opCode, __lView, nodeIndex, 'IcuSwitch');
                                    result['tIcuIndex'] = tIcuIndex;
                                    result['checkBit'] = checkBit;
                                    result['mainBinding'] = value;
                                    result['tIcu'] = tIcu;
                                    results.push(result);
                                    break;
                                case 3 /* IcuUpdate */:
                                    tIcuIndex = (/** @type {?} */ (__raw_opCodes[++j]));
                                    tIcu = (/** @type {?} */ (icus))[tIcuIndex];
                                    result = new I18NDebugItem(opCode, __lView, nodeIndex, 'IcuUpdate');
                                    result['tIcuIndex'] = tIcuIndex;
                                    result['checkBit'] = checkBit;
                                    result['tIcu'] = tIcu;
                                    results.push(result);
                                    break;
                            }
                        }
                    }
                }
                i += skipCodes;
            }
            return results;
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/instructions/shared.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    const ɵ0$4 = /**
     * @return {?}
     */
    () => Promise.resolve(null);
    /**
     * A permanent marker promise which signifies that the current CD tree is
     * clean.
     * @type {?}
     */
    const _CLEAN_PROMISE = ((ɵ0$4))();
    /**
     * Process the `TView.expandoInstructions`. (Execute the `hostBindings`.)
     *
     * @param {?} tView `TView` containing the `expandoInstructions`
     * @param {?} lView `LView` associated with the `TView`
     * @return {?}
     */
    function setHostBindingsByExecutingExpandoInstructions(tView, lView) {
        ngDevMode && assertSame(tView, lView[TVIEW], '`LView` is not associated with the `TView`!');
        try {
            /** @type {?} */
            const expandoInstructions = tView.expandoInstructions;
            if (expandoInstructions !== null) {
                /** @type {?} */
                let bindingRootIndex = tView.expandoStartIndex;
                /** @type {?} */
                let currentDirectiveIndex = -1;
                /** @type {?} */
                let currentElementIndex = -1;
                // TODO(misko): PERF It is possible to get here with `TView.expandoInstructions` containing no
                // functions to execute. This is wasteful as there is no work to be done, but we still need
                // to iterate over the instructions.
                // In example of this is in this test: `host_binding_spec.ts`
                // `fit('should not cause problems if detectChanges is called when a property updates', ...`
                // In the above test we get here with expando [0, 0, 1] which requires a lot of processing but
                // there is no function to execute.
                for (let i = 0; i < expandoInstructions.length; i++) {
                    /** @type {?} */
                    const instruction = expandoInstructions[i];
                    if (typeof instruction === 'number') {
                        if (instruction <= 0) {
                            // Negative numbers mean that we are starting new EXPANDO block and need to update
                            // the current element and directive index.
                            // Important: In JS `-x` and `0-x` is not the same! If `x===0` then `-x` will produce
                            // `-0` which requires non standard math arithmetic and it can prevent VM optimizations.
                            // `0-0` will always produce `0` and will not cause a potential deoptimization in VM.
                            // TODO(misko): PERF This should be refactored to use `~instruction` as that does not
                            // suffer from `-0` and it is faster/more compact.
                            currentElementIndex = 0 - instruction;
                            setSelectedIndex(currentElementIndex);
                            // Injector block and providers are taken into account.
                            /** @type {?} */
                            const providerCount = ((/** @type {?} */ (expandoInstructions[++i])));
                            bindingRootIndex += INJECTOR_BLOOM_PARENT_SIZE + providerCount;
                            currentDirectiveIndex = bindingRootIndex;
                        }
                        else {
                            // This is either the injector size (so the binding root can skip over directives
                            // and get to the first set of host bindings on this node) or the host var count
                            // (to get to the next set of host bindings on this node).
                            bindingRootIndex += instruction;
                        }
                    }
                    else {
                        // If it's not a number, it's a host binding function that needs to be executed.
                        if (instruction !== null) {
                            setBindingRootForHostBindings(bindingRootIndex, currentDirectiveIndex);
                            /** @type {?} */
                            const hostCtx = lView[currentDirectiveIndex];
                            instruction(2 /* Update */, hostCtx);
                        }
                        // TODO(misko): PERF Relying on incrementing the `currentDirectiveIndex` here is
                        // sub-optimal. The implications are that if we have a lot of directives but none of them
                        // have host bindings we nevertheless need to iterate over the expando instructions to
                        // update the counter. It would be much better if we could encode the
                        // `currentDirectiveIndex` into the `expandoInstruction` array so that we only need to
                        // iterate over those directives which actually have `hostBindings`.
                        currentDirectiveIndex++;
                    }
                }
            }
        }
        finally {
            setSelectedIndex(-1);
        }
    }
    /**
     * Refreshes all content queries declared by directives in a given view
     * @param {?} tView
     * @param {?} lView
     * @return {?}
     */
    function refreshContentQueries(tView, lView) {
        /** @type {?} */
        const contentQueries = tView.contentQueries;
        if (contentQueries !== null) {
            for (let i = 0; i < contentQueries.length; i += 2) {
                /** @type {?} */
                const queryStartIdx = contentQueries[i];
                /** @type {?} */
                const directiveDefIdx = contentQueries[i + 1];
                if (directiveDefIdx !== -1) {
                    /** @type {?} */
                    const directiveDef = (/** @type {?} */ (tView.data[directiveDefIdx]));
                    ngDevMode &&
                        assertDefined(directiveDef.contentQueries, 'contentQueries function should be defined');
                    setCurrentQueryIndex(queryStartIdx);
                    (/** @type {?} */ (directiveDef.contentQueries))(2 /* Update */, lView[directiveDefIdx], directiveDefIdx);
                }
            }
        }
    }
    /**
     * Refreshes child components in the current view (update mode).
     * @param {?} hostLView
     * @param {?} components
     * @return {?}
     */
    function refreshChildComponents(hostLView, components) {
        for (let i = 0; i < components.length; i++) {
            refreshComponent(hostLView, components[i]);
        }
    }
    /**
     * Renders child components in the current view (creation mode).
     * @param {?} hostLView
     * @param {?} components
     * @return {?}
     */
    function renderChildComponents(hostLView, components) {
        for (let i = 0; i < components.length; i++) {
            renderComponent(hostLView, components[i]);
        }
    }
    /**
     * Creates a native element from a tag name, using a renderer.
     * @param {?} name the tag name
     * @param {?} renderer A renderer to use
     * @param {?} namespace
     * @return {?} the element created
     */
    function elementCreate(name, renderer, namespace) {
        if (isProceduralRenderer(renderer)) {
            return renderer.createElement(name, namespace);
        }
        else {
            return namespace === null ? renderer.createElement(name) :
                renderer.createElementNS(namespace, name);
        }
    }
    /**
     * @template T
     * @param {?} parentLView
     * @param {?} tView
     * @param {?} context
     * @param {?} flags
     * @param {?} host
     * @param {?} tHostNode
     * @param {?=} rendererFactory
     * @param {?=} renderer
     * @param {?=} sanitizer
     * @param {?=} injector
     * @return {?}
     */
    function createLView(parentLView, tView, context, flags, host, tHostNode, rendererFactory, renderer, sanitizer, injector) {
        /** @type {?} */
        const lView = ngDevMode ? cloneToLViewFromTViewBlueprint(tView) : (/** @type {?} */ (tView.blueprint.slice()));
        lView[HOST] = host;
        lView[FLAGS] = flags | 4 /* CreationMode */ | 128 /* Attached */ | 8 /* FirstLViewPass */;
        resetPreOrderHookFlags(lView);
        lView[PARENT] = lView[DECLARATION_VIEW] = parentLView;
        lView[CONTEXT] = context;
        lView[RENDERER_FACTORY] = (/** @type {?} */ ((rendererFactory || parentLView && parentLView[RENDERER_FACTORY])));
        ngDevMode && assertDefined(lView[RENDERER_FACTORY], 'RendererFactory is required');
        lView[RENDERER] = (/** @type {?} */ ((renderer || parentLView && parentLView[RENDERER])));
        ngDevMode && assertDefined(lView[RENDERER], 'Renderer is required');
        lView[SANITIZER] = sanitizer || parentLView && parentLView[SANITIZER] || (/** @type {?} */ (null));
        lView[(/** @type {?} */ (INJECTOR$1))] = injector || parentLView && parentLView[INJECTOR$1] || null;
        lView[T_HOST] = tHostNode;
        ngDevMode &&
            assertEqual(tView.type == 2 /* Embedded */ ? parentLView !== null : true, true, 'Embedded views must have parentLView');
        lView[DECLARATION_COMPONENT_VIEW] =
            tView.type == 2 /* Embedded */ ? (/** @type {?} */ (parentLView))[DECLARATION_COMPONENT_VIEW] : lView;
        ngDevMode && attachLViewDebug(lView);
        return lView;
    }
    /**
     * @param {?} tView
     * @param {?} tHostNode
     * @param {?} index
     * @param {?} type
     * @param {?} name
     * @param {?} attrs
     * @return {?}
     */
    function getOrCreateTNode(tView, tHostNode, index, type, name, attrs) {
        // Keep this function short, so that the VM will inline it.
        /** @type {?} */
        const adjustedIndex = index + HEADER_OFFSET;
        /** @type {?} */
        const tNode = (/** @type {?} */ (tView.data[adjustedIndex])) ||
            createTNodeAtIndex(tView, tHostNode, adjustedIndex, type, name, attrs);
        setPreviousOrParentTNode(tNode, true);
        return (/** @type {?} */ (tNode));
    }
    /**
     * @param {?} tView
     * @param {?} tHostNode
     * @param {?} adjustedIndex
     * @param {?} type
     * @param {?} name
     * @param {?} attrs
     * @return {?}
     */
    function createTNodeAtIndex(tView, tHostNode, adjustedIndex, type, name, attrs) {
        /** @type {?} */
        const previousOrParentTNode = getPreviousOrParentTNode();
        /** @type {?} */
        const isParent = getIsParent();
        /** @type {?} */
        const parent = isParent ? previousOrParentTNode : previousOrParentTNode && previousOrParentTNode.parent;
        // Parents cannot cross component boundaries because components will be used in multiple places,
        // so it's only set if the view is the same.
        /** @type {?} */
        const parentInSameView = parent && parent !== tHostNode;
        /** @type {?} */
        const tParentNode = parentInSameView ? (/** @type {?} */ (parent)) : null;
        /** @type {?} */
        const tNode = tView.data[adjustedIndex] =
            createTNode(tView, tParentNode, type, adjustedIndex, name, attrs);
        // Assign a pointer to the first child node of a given view. The first node is not always the one
        // at index 0, in case of i18n, index 0 can be the instruction `i18nStart` and the first node has
        // the index 1 or more, so we can't just check node index.
        if (tView.firstChild === null) {
            tView.firstChild = tNode;
        }
        if (previousOrParentTNode) {
            if (isParent && previousOrParentTNode.child == null &&
                (tNode.parent !== null || previousOrParentTNode.type === 2 /* View */)) {
                // We are in the same view, which means we are adding content node to the parent view.
                previousOrParentTNode.child = tNode;
            }
            else if (!isParent) {
                previousOrParentTNode.next = tNode;
            }
        }
        return tNode;
    }
    /**
     * @param {?} tView
     * @param {?} tParentNode
     * @param {?} index
     * @param {?} lView
     * @return {?}
     */
    function assignTViewNodeToLView(tView, tParentNode, index, lView) {
        // View nodes are not stored in data because they can be added / removed at runtime (which
        // would cause indices to change). Their TNodes are instead stored in tView.node.
        /** @type {?} */
        let tNode = tView.node;
        if (tNode == null) {
            ngDevMode && tParentNode &&
                assertNodeOfPossibleTypes(tParentNode, 3 /* Element */, 0 /* Container */);
            tView.node = tNode = (/** @type {?} */ (createTNode(tView, (/** @type {?} */ (tParentNode)), //
            2 /* View */, index, null, null)));
        }
        lView[T_HOST] = (/** @type {?} */ (tNode));
    }
    /**
     * When elements are created dynamically after a view blueprint is created (e.g. through
     * i18nApply() or ComponentFactory.create), we need to adjust the blueprint for future
     * template passes.
     *
     * @param {?} tView `TView` associated with `LView`
     * @param {?} lView
     * @param {?} numSlotsToAlloc The number of slots to alloc in the LView, should be >0
     * @return {?}
     */
    function allocExpando(tView, lView, numSlotsToAlloc) {
        ngDevMode &&
            assertGreaterThan(numSlotsToAlloc, 0, 'The number of slots to alloc should be greater than 0');
        if (numSlotsToAlloc > 0) {
            if (tView.firstCreatePass) {
                for (let i = 0; i < numSlotsToAlloc; i++) {
                    tView.blueprint.push(null);
                    tView.data.push(null);
                    lView.push(null);
                }
                // We should only increment the expando start index if there aren't already directives
                // and injectors saved in the "expando" section
                if (!tView.expandoInstructions) {
                    tView.expandoStartIndex += numSlotsToAlloc;
                }
                else {
                    // Since we're adding the dynamic nodes into the expando section, we need to let the host
                    // bindings know that they should skip x slots
                    tView.expandoInstructions.push(numSlotsToAlloc);
                }
            }
        }
    }
    //////////////////////////
    //// Render
    //////////////////////////
    /**
     * Processes a view in the creation mode. This includes a number of steps in a specific order:
     * - creating view query functions (if any);
     * - executing a template function in the creation mode;
     * - updating static queries (if any);
     * - creating child components defined in a given view.
     * @template T
     * @param {?} tView
     * @param {?} lView
     * @param {?} context
     * @return {?}
     */
    function renderView(tView, lView, context) {
        ngDevMode && assertEqual(isCreationMode(lView), true, 'Should be run in creation mode');
        enterView(lView, lView[T_HOST]);
        try {
            /** @type {?} */
            const viewQuery = tView.viewQuery;
            if (viewQuery !== null) {
                executeViewQueryFn(1 /* Create */, viewQuery, context);
            }
            // Execute a template associated with this view, if it exists. A template function might not be
            // defined for the root component views.
            /** @type {?} */
            const templateFn = tView.template;
            if (templateFn !== null) {
                executeTemplate(tView, lView, templateFn, 1 /* Create */, context);
            }
            // This needs to be set before children are processed to support recursive components.
            // This must be set to false immediately after the first creation run because in an
            // ngFor loop, all the views will be created together before update mode runs and turns
            // off firstCreatePass. If we don't set it here, instances will perform directive
            // matching, etc again and again.
            if (tView.firstCreatePass) {
                tView.firstCreatePass = false;
            }
            // We resolve content queries specifically marked as `static` in creation mode. Dynamic
            // content queries are resolved during change detection (i.e. update mode), after embedded
            // views are refreshed (see block above).
            if (tView.staticContentQueries) {
                refreshContentQueries(tView, lView);
            }
            // We must materialize query results before child components are processed
            // in case a child component has projected a container. The LContainer needs
            // to exist so the embedded views are properly attached by the container.
            if (tView.staticViewQueries) {
                executeViewQueryFn(2 /* Update */, (/** @type {?} */ (tView.viewQuery)), context);
            }
            // Render child component views.
            /** @type {?} */
            const components = tView.components;
            if (components !== null) {
                renderChildComponents(lView, components);
            }
        }
        catch (error) {
            // If we didn't manage to get past the first template pass due to
            // an error, mark the view as corrupted so we can try to recover.
            if (tView.firstCreatePass) {
                tView.incompleteFirstPass = true;
            }
            throw error;
        }
        finally {
            lView[FLAGS] &= ~4 /* CreationMode */;
            leaveView();
        }
    }
    /**
     * Processes a view in update mode. This includes a number of steps in a specific order:
     * - executing a template function in update mode;
     * - executing hooks;
     * - refreshing queries;
     * - setting host bindings;
     * - refreshing child (embedded and component) views.
     * @template T
     * @param {?} tView
     * @param {?} lView
     * @param {?} templateFn
     * @param {?} context
     * @return {?}
     */
    function refreshView(tView, lView, templateFn, context) {
        ngDevMode && assertEqual(isCreationMode(lView), false, 'Should be run in update mode');
        /** @type {?} */
        const flags = lView[FLAGS];
        if ((flags & 256 /* Destroyed */) === 256 /* Destroyed */)
            return;
        enterView(lView, lView[T_HOST]);
        /** @type {?} */
        const checkNoChangesMode = getCheckNoChangesMode();
        try {
            resetPreOrderHookFlags(lView);
            setBindingIndex(tView.bindingStartIndex);
            if (templateFn !== null) {
                executeTemplate(tView, lView, templateFn, 2 /* Update */, context);
            }
            /** @type {?} */
            const hooksInitPhaseCompleted = (flags & 3 /* InitPhaseStateMask */) === 3 /* InitPhaseCompleted */;
            // execute pre-order hooks (OnInit, OnChanges, DoCheck)
            // PERF WARNING: do NOT extract this to a separate function without running benchmarks
            if (!checkNoChangesMode) {
                if (hooksInitPhaseCompleted) {
                    /** @type {?} */
                    const preOrderCheckHooks = tView.preOrderCheckHooks;
                    if (preOrderCheckHooks !== null) {
                        executeCheckHooks(lView, preOrderCheckHooks, null);
                    }
                }
                else {
                    /** @type {?} */
                    const preOrderHooks = tView.preOrderHooks;
                    if (preOrderHooks !== null) {
                        executeInitAndCheckHooks(lView, preOrderHooks, 0 /* OnInitHooksToBeRun */, null);
                    }
                    incrementInitPhaseFlags(lView, 0 /* OnInitHooksToBeRun */);
                }
            }
            // First mark transplanted views that are declared in this lView as needing a refresh at their
            // insertion points. This is needed to avoid the situation where the template is defined in this
            // `LView` but its declaration appears after the insertion component.
            markTransplantedViewsForRefresh(lView);
            refreshEmbeddedViews(lView);
            // Content query results must be refreshed before content hooks are called.
            if (tView.contentQueries !== null) {
                refreshContentQueries(tView, lView);
            }
            // execute content hooks (AfterContentInit, AfterContentChecked)
            // PERF WARNING: do NOT extract this to a separate function without running benchmarks
            if (!checkNoChangesMode) {
                if (hooksInitPhaseCompleted) {
                    /** @type {?} */
                    const contentCheckHooks = tView.contentCheckHooks;
                    if (contentCheckHooks !== null) {
                        executeCheckHooks(lView, contentCheckHooks);
                    }
                }
                else {
                    /** @type {?} */
                    const contentHooks = tView.contentHooks;
                    if (contentHooks !== null) {
                        executeInitAndCheckHooks(lView, contentHooks, 1 /* AfterContentInitHooksToBeRun */);
                    }
                    incrementInitPhaseFlags(lView, 1 /* AfterContentInitHooksToBeRun */);
                }
            }
            setHostBindingsByExecutingExpandoInstructions(tView, lView);
            // Refresh child component views.
            /** @type {?} */
            const components = tView.components;
            if (components !== null) {
                refreshChildComponents(lView, components);
            }
            // View queries must execute after refreshing child components because a template in this view
            // could be inserted in a child component. If the view query executes before child component
            // refresh, the template might not yet be inserted.
            /** @type {?} */
            const viewQuery = tView.viewQuery;
            if (viewQuery !== null) {
                executeViewQueryFn(2 /* Update */, viewQuery, context);
            }
            // execute view hooks (AfterViewInit, AfterViewChecked)
            // PERF WARNING: do NOT extract this to a separate function without running benchmarks
            if (!checkNoChangesMode) {
                if (hooksInitPhaseCompleted) {
                    /** @type {?} */
                    const viewCheckHooks = tView.viewCheckHooks;
                    if (viewCheckHooks !== null) {
                        executeCheckHooks(lView, viewCheckHooks);
                    }
                }
                else {
                    /** @type {?} */
                    const viewHooks = tView.viewHooks;
                    if (viewHooks !== null) {
                        executeInitAndCheckHooks(lView, viewHooks, 2 /* AfterViewInitHooksToBeRun */);
                    }
                    incrementInitPhaseFlags(lView, 2 /* AfterViewInitHooksToBeRun */);
                }
            }
            if (tView.firstUpdatePass === true) {
                // We need to make sure that we only flip the flag on successful `refreshView` only
                // Don't do this in `finally` block.
                // If we did this in `finally` block then an exception could block the execution of styling
                // instructions which in turn would be unable to insert themselves into the styling linked
                // list. The result of this would be that if the exception would not be throw on subsequent CD
                // the styling would be unable to process it data and reflect to the DOM.
                tView.firstUpdatePass = false;
            }
            // Do not reset the dirty state when running in check no changes mode. We don't want components
            // to behave differently depending on whether check no changes is enabled or not. For example:
            // Marking an OnPush component as dirty from within the `ngAfterViewInit` hook in order to
            // refresh a `NgClass` binding should work. If we would reset the dirty state in the check
            // no changes cycle, the component would be not be dirty for the next update pass. This would
            // be different in production mode where the component dirty state is not reset.
            if (!checkNoChangesMode) {
                lView[FLAGS] &= ~(64 /* Dirty */ | 8 /* FirstLViewPass */);
            }
            if (lView[FLAGS] & 1024 /* RefreshTransplantedView */) {
                lView[FLAGS] &= ~1024 /* RefreshTransplantedView */;
                updateTransplantedViewCount((/** @type {?} */ (lView[PARENT])), -1);
            }
        }
        finally {
            leaveView();
        }
    }
    /**
     * @template T
     * @param {?} tView
     * @param {?} lView
     * @param {?} templateFn
     * @param {?} context
     * @return {?}
     */
    function renderComponentOrTemplate(tView, lView, templateFn, context) {
        /** @type {?} */
        const rendererFactory = lView[RENDERER_FACTORY];
        /** @type {?} */
        const normalExecutionPath = !getCheckNoChangesMode();
        /** @type {?} */
        const creationModeIsActive = isCreationMode(lView);
        try {
            if (normalExecutionPath && !creationModeIsActive && rendererFactory.begin) {
                rendererFactory.begin();
            }
            if (creationModeIsActive) {
                renderView(tView, lView, context);
            }
            refreshView(tView, lView, templateFn, context);
        }
        finally {
            if (normalExecutionPath && !creationModeIsActive && rendererFactory.end) {
                rendererFactory.end();
            }
        }
    }
    /**
     * @template T
     * @param {?} tView
     * @param {?} lView
     * @param {?} templateFn
     * @param {?} rf
     * @param {?} context
     * @return {?}
     */
    function executeTemplate(tView, lView, templateFn, rf, context) {
        /** @type {?} */
        const prevSelectedIndex = getSelectedIndex();
        try {
            setSelectedIndex(-1);
            if (rf & 2 /* Update */ && lView.length > HEADER_OFFSET) {
                // When we're updating, inherently select 0 so we don't
                // have to generate that instruction for most update blocks.
                selectIndexInternal(tView, lView, 0, getCheckNoChangesMode());
            }
            templateFn(rf, context);
        }
        finally {
            setSelectedIndex(prevSelectedIndex);
        }
    }
    //////////////////////////
    //// Element
    //////////////////////////
    /**
     * @param {?} tView
     * @param {?} tNode
     * @param {?} lView
     * @return {?}
     */
    function executeContentQueries(tView, tNode, lView) {
        if (isContentQueryHost(tNode)) {
            /** @type {?} */
            const start = tNode.directiveStart;
            /** @type {?} */
            const end = tNode.directiveEnd;
            for (let directiveIndex = start; directiveIndex < end; directiveIndex++) {
                /** @type {?} */
                const def = (/** @type {?} */ (tView.data[directiveIndex]));
                if (def.contentQueries) {
                    def.contentQueries(1 /* Create */, lView[directiveIndex], directiveIndex);
                }
            }
        }
    }
    /**
     * Creates directive instances.
     * @param {?} tView
     * @param {?} lView
     * @param {?} tNode
     * @return {?}
     */
    function createDirectivesInstances(tView, lView, tNode) {
        if (!getBindingsEnabled())
            return;
        instantiateAllDirectives(tView, lView, tNode, getNativeByTNode(tNode, lView));
        if ((tNode.flags & 128 /* hasHostBindings */) === 128 /* hasHostBindings */) {
            invokeDirectivesHostBindings(tView, lView, tNode);
        }
    }
    /**
     * Takes a list of local names and indices and pushes the resolved local variable values
     * to LView in the same order as they are loaded in the template with load().
     * @param {?} viewData
     * @param {?} tNode
     * @param {?=} localRefExtractor
     * @return {?}
     */
    function saveResolvedLocalsInData(viewData, tNode, localRefExtractor = getNativeByTNode) {
        /** @type {?} */
        const localNames = tNode.localNames;
        if (localNames !== null) {
            /** @type {?} */
            let localIndex = tNode.index + 1;
            for (let i = 0; i < localNames.length; i += 2) {
                /** @type {?} */
                const index = (/** @type {?} */ (localNames[i + 1]));
                /** @type {?} */
                const value = index === -1 ?
                    localRefExtractor((/** @type {?} */ (tNode)), viewData) :
                    viewData[index];
                viewData[localIndex++] = value;
            }
        }
    }
    /**
     * Gets TView from a template function or creates a new TView
     * if it doesn't already exist.
     *
     * @param {?} def ComponentDef
     * @return {?} TView
     */
    function getOrCreateTComponentView(def) {
        /** @type {?} */
        const tView = def.tView;
        // Create a TView if there isn't one, or recreate it if the first create pass didn't
        // complete successfuly since we can't know for sure whether it's in a usable shape.
        if (tView === null || tView.incompleteFirstPass) {
            return def.tView = createTView(1 /* Component */, -1, def.template, def.decls, def.vars, def.directiveDefs, def.pipeDefs, def.viewQuery, def.schemas, def.consts);
        }
        return tView;
    }
    /**
     * Creates a TView instance
     *
     * @param {?} type
     * @param {?} viewIndex The viewBlockId for inline views, or -1 if it's a component/dynamic
     * @param {?} templateFn Template function
     * @param {?} decls The number of nodes, local refs, and pipes in this template
     * @param {?} vars
     * @param {?} directives Registry of directives for this view
     * @param {?} pipes Registry of pipes for this view
     * @param {?} viewQuery View queries for this view
     * @param {?} schemas Schemas for this view
     * @param {?} consts Constants for this view
     * @return {?}
     */
    function createTView(type, viewIndex, templateFn, decls, vars, directives, pipes, viewQuery, schemas, consts) {
        ngDevMode && ngDevMode.tView++;
        /** @type {?} */
        const bindingStartIndex = HEADER_OFFSET + decls;
        // This length does not yet contain host bindings from child directives because at this point,
        // we don't know which directives are active on this template. As soon as a directive is matched
        // that has a host binding, we will update the blueprint with that def's hostVars count.
        /** @type {?} */
        const initialViewLength = bindingStartIndex + vars;
        /** @type {?} */
        const blueprint = createViewBlueprint(bindingStartIndex, initialViewLength);
        return blueprint[(/** @type {?} */ (TVIEW))] = ngDevMode ?
            new TViewConstructor(type, viewIndex, // id: number,
            blueprint, // blueprint: LView,
            templateFn, // template: ComponentTemplate<{}>|null,
            null, // queries: TQueries|null
            viewQuery, (/** @type {?} */ (null)), // node: TViewNode|TElementNode|null,
            cloneToTViewData(blueprint).fill(null, bindingStartIndex), // data: TData,
            bindingStartIndex, // bindingStartIndex: number,
            initialViewLength, // expandoStartIndex: number,
            null, // expandoInstructions: ExpandoInstructions|null,
            true, // firstCreatePass: boolean,
            true, // firstUpdatePass: boolean,
            false, // staticViewQueries: boolean,
            false, // staticContentQueries: boolean,
            null, // preOrderHooks: HookData|null,
            null, // preOrderCheckHooks: HookData|null,
            null, // contentHooks: HookData|null,
            null, // contentCheckHooks: HookData|null,
            null, // viewHooks: HookData|null,
            null, // viewCheckHooks: HookData|null,
            null, // destroyHooks: DestroyHookData|null,
            null, // cleanup: any[]|null,
            null, // contentQueries: number[]|null,
            null, // components: number[]|null,
            typeof directives === 'function' ?
                directives() :
                directives, // directiveRegistry: DirectiveDefList|null,
            typeof pipes === 'function' ? pipes() : pipes, // pipeRegistry: PipeDefList|null,
            null, // firstChild: TNode|null,
            schemas, // schemas: SchemaMetadata[]|null,
            consts, // consts: TConstants|null
            false // incompleteFirstPass: boolean
            ) :
            {
                type: type,
                id: viewIndex,
                blueprint: blueprint,
                template: templateFn,
                queries: null,
                viewQuery: viewQuery,
                node: (/** @type {?} */ (null)),
                data: blueprint.slice().fill(null, bindingStartIndex),
                bindingStartIndex: bindingStartIndex,
                expandoStartIndex: initialViewLength,
                expandoInstructions: null,
                firstCreatePass: true,
                firstUpdatePass: true,
                staticViewQueries: false,
                staticContentQueries: false,
                preOrderHooks: null,
                preOrderCheckHooks: null,
                contentHooks: null,
                contentCheckHooks: null,
                viewHooks: null,
                viewCheckHooks: null,
                destroyHooks: null,
                cleanup: null,
                contentQueries: null,
                components: null,
                directiveRegistry: typeof directives === 'function' ? directives() : directives,
                pipeRegistry: typeof pipes === 'function' ? pipes() : pipes,
                firstChild: null,
                schemas: schemas,
                consts: consts,
                incompleteFirstPass: false
            };
    }
    /**
     * @param {?} bindingStartIndex
     * @param {?} initialViewLength
     * @return {?}
     */
    function createViewBlueprint(bindingStartIndex, initialViewLength) {
        /** @type {?} */
        const blueprint = ngDevMode ? new LViewBlueprint() : [];
        for (let i = 0; i < initialViewLength; i++) {
            blueprint.push(i < bindingStartIndex ? null : NO_CHANGE);
        }
        return (/** @type {?} */ (blueprint));
    }
    /**
     * @param {?} text
     * @param {?} token
     * @return {?}
     */
    function createError(text, token) {
        return new Error(`Renderer: ${text} [${stringifyForError(token)}]`);
    }
    /**
     * @param {?} rElement
     * @param {?} elementOrSelector
     * @return {?}
     */
    function assertHostNodeExists(rElement, elementOrSelector) {
        if (!rElement) {
            if (typeof elementOrSelector === 'string') {
                throw createError('Host node with selector not found:', elementOrSelector);
            }
            else {
                throw createError('Host node is required:', elementOrSelector);
            }
        }
    }
    /**
     * Locates the host native element, used for bootstrapping existing nodes into rendering pipeline.
     *
     * @param {?} renderer
     * @param {?} elementOrSelector Render element or CSS selector to locate the element.
     * @param {?} encapsulation View Encapsulation defined for component that requests host element.
     * @return {?}
     */
    function locateHostElement(renderer, elementOrSelector, encapsulation) {
        if (isProceduralRenderer(renderer)) {
            // When using native Shadow DOM, do not clear host element to allow native slot projection
            /** @type {?} */
            const preserveContent = encapsulation === ViewEncapsulation$1.ShadowDom;
            return renderer.selectRootElement(elementOrSelector, preserveContent);
        }
        /** @type {?} */
        let rElement = typeof elementOrSelector === 'string' ?
            (/** @type {?} */ (renderer.querySelector(elementOrSelector))) :
            elementOrSelector;
        ngDevMode && assertHostNodeExists(rElement, elementOrSelector);
        // Always clear host element's content when Renderer3 is in use. For procedural renderer case we
        // make it depend on whether ShadowDom encapsulation is used (in which case the content should be
        // preserved to allow native slot projection). ShadowDom encapsulation requires procedural
        // renderer, and procedural renderer case is handled above.
        rElement.textContent = '';
        return rElement;
    }
    /**
     * Saves context for this cleanup function in LView.cleanupInstances.
     *
     * On the first template pass, saves in TView:
     * - Cleanup function
     * - Index of context we just saved in LView.cleanupInstances
     * @param {?} tView
     * @param {?} lView
     * @param {?} context
     * @param {?} cleanupFn
     * @return {?}
     */
    function storeCleanupWithContext(tView, lView, context, cleanupFn) {
        /** @type {?} */
        const lCleanup = getLCleanup(lView);
        lCleanup.push(context);
        if (tView.firstCreatePass) {
            getTViewCleanup(tView).push(cleanupFn, lCleanup.length - 1);
        }
    }
    /**
     * Saves the cleanup function itself in LView.cleanupInstances.
     *
     * This is necessary for functions that are wrapped with their contexts, like in renderer2
     * listeners.
     *
     * On the first template pass, the index of the cleanup function is saved in TView.
     * @param {?} tView
     * @param {?} lView
     * @param {?} cleanupFn
     * @return {?}
     */
    function storeCleanupFn(tView, lView, cleanupFn) {
        getLCleanup(lView).push(cleanupFn);
        if (tView.firstCreatePass) {
            getTViewCleanup(tView).push((/** @type {?} */ (lView[CLEANUP])).length - 1, null);
        }
    }
    /**
     * Constructs a TNode object from the arguments.
     *
     * @param {?} tView `TView` to which this `TNode` belongs (used only in `ngDevMode`)
     * @param {?} tParent
     * @param {?} type The type of the node
     * @param {?} adjustedIndex The index of the TNode in TView.data, adjusted for HEADER_OFFSET
     * @param {?} tagName The tag name of the node
     * @param {?} attrs The attributes defined on this node
     * @return {?} the TNode object
     */
    function createTNode(tView, tParent, type, adjustedIndex, tagName, attrs) {
        ngDevMode && ngDevMode.tNode++;
        /** @type {?} */
        let injectorIndex = tParent ? tParent.injectorIndex : -1;
        return ngDevMode ? new TNodeDebug(tView, // tView_: TView
        type, // type: TNodeType
        adjustedIndex, // index: number
        injectorIndex, // injectorIndex: number
        -1, // directiveStart: number
        -1, // directiveEnd: number
        -1, // directiveStylingLast: number
        null, // propertyBindings: number[]|null
        0, // flags: TNodeFlags
        0, // providerIndexes: TNodeProviderIndexes
        tagName, // tagName: string|null
        attrs, // attrs: (string|AttributeMarker|(string|SelectorFlags)[])[]|null
        null, // mergedAttrs
        null, // localNames: (string|number)[]|null
        undefined, // initialInputs: (string[]|null)[]|null|undefined
        null, // inputs: PropertyAliases|null
        null, // outputs: PropertyAliases|null
        null, // tViews: ITView|ITView[]|null
        null, // next: ITNode|null
        null, // projectionNext: ITNode|null
        null, // child: ITNode|null
        tParent, // parent: TElementNode|TContainerNode|null
        null, // projection: number|(ITNode|RNode[])[]|null
        null, // styles: string|null
        null, // stylesWithoutHost: string|null
        undefined, // residualStyles: string|null
        null, // classes: string|null
        null, // classesWithoutHost: string|null
        undefined, (/** @type {?} */ (0)), (/** @type {?} */ (0))) :
            {
                type: type,
                index: adjustedIndex,
                injectorIndex: injectorIndex,
                directiveStart: -1,
                directiveEnd: -1,
                directiveStylingLast: -1,
                propertyBindings: null,
                flags: 0,
                providerIndexes: 0,
                tagName: tagName,
                attrs: attrs,
                mergedAttrs: null,
                localNames: null,
                initialInputs: undefined,
                inputs: null,
                outputs: null,
                tViews: null,
                next: null,
                projectionNext: null,
                child: null,
                parent: tParent,
                projection: null,
                styles: null,
                stylesWithoutHost: null,
                residualStyles: undefined,
                classes: null,
                classesWithoutHost: null,
                residualClasses: undefined,
                classBindings: (/** @type {?} */ (0)),
                styleBindings: (/** @type {?} */ (0)),
            };
    }
    /**
     * @param {?} inputAliasMap
     * @param {?} directiveDefIdx
     * @param {?} propStore
     * @return {?}
     */
    function generatePropertyAliases(inputAliasMap, directiveDefIdx, propStore) {
        for (let publicName in inputAliasMap) {
            if (inputAliasMap.hasOwnProperty(publicName)) {
                propStore = propStore === null ? {} : propStore;
                /** @type {?} */
                const internalName = inputAliasMap[publicName];
                if (propStore.hasOwnProperty(publicName)) {
                    propStore[publicName].push(directiveDefIdx, internalName);
                }
                else {
                    (propStore[publicName] = [directiveDefIdx, internalName]);
                }
            }
        }
        return propStore;
    }
    /**
     * Initializes data structures required to work with directive outputs and outputs.
     * Initialization is done for all directives matched on a given TNode.
     * @param {?} tView
     * @param {?} tNode
     * @return {?}
     */
    function initializeInputAndOutputAliases(tView, tNode) {
        ngDevMode && assertFirstCreatePass(tView);
        /** @type {?} */
        const start = tNode.directiveStart;
        /** @type {?} */
        const end = tNode.directiveEnd;
        /** @type {?} */
        const defs = tView.data;
        /** @type {?} */
        const tNodeAttrs = tNode.attrs;
        /** @type {?} */
        const inputsFromAttrs = ngDevMode ? new TNodeInitialInputs() : [];
        /** @type {?} */
        let inputsStore = null;
        /** @type {?} */
        let outputsStore = null;
        for (let i = start; i < end; i++) {
            /** @type {?} */
            const directiveDef = (/** @type {?} */ (defs[i]));
            /** @type {?} */
            const directiveInputs = directiveDef.inputs;
            // Do not use unbound attributes as inputs to structural directives, since structural
            // directive inputs can only be set using microsyntax (e.g. `<div *dir="exp">`).
            // TODO(FW-1930): microsyntax expressions may also contain unbound/static attributes, which
            // should be set for inline templates.
            /** @type {?} */
            const initialInputs = (tNodeAttrs !== null && !isInlineTemplate(tNode)) ?
                generateInitialInputs(directiveInputs, tNodeAttrs) :
                null;
            inputsFromAttrs.push(initialInputs);
            inputsStore = generatePropertyAliases(directiveInputs, i, inputsStore);
            outputsStore = generatePropertyAliases(directiveDef.outputs, i, outputsStore);
        }
        if (inputsStore !== null) {
            if (inputsStore.hasOwnProperty('class')) {
                tNode.flags |= 16 /* hasClassInput */;
            }
            if (inputsStore.hasOwnProperty('style')) {
                tNode.flags |= 32 /* hasStyleInput */;
            }
        }
        tNode.initialInputs = inputsFromAttrs;
        tNode.inputs = inputsStore;
        tNode.outputs = outputsStore;
    }
    /**
     * Mapping between attributes names that don't correspond to their element property names.
     *
     * Performance note: this function is written as a series of if checks (instead of, say, a property
     * object lookup) for performance reasons - the series of `if` checks seems to be the fastest way of
     * mapping property names. Do NOT change without benchmarking.
     *
     * Note: this mapping has to be kept in sync with the equally named mapping in the template
     * type-checking machinery of ngtsc.
     * @param {?} name
     * @return {?}
     */
    function mapPropName(name) {
        if (name === 'class')
            return 'className';
        if (name === 'for')
            return 'htmlFor';
        if (name === 'formaction')
            return 'formAction';
        if (name === 'innerHtml')
            return 'innerHTML';
        if (name === 'readonly')
            return 'readOnly';
        if (name === 'tabindex')
            return 'tabIndex';
        return name;
    }
    /**
     * @template T
     * @param {?} tView
     * @param {?} tNode
     * @param {?} lView
     * @param {?} propName
     * @param {?} value
     * @param {?} renderer
     * @param {?} sanitizer
     * @param {?} nativeOnly
     * @return {?}
     */
    function elementPropertyInternal(tView, tNode, lView, propName, value, renderer, sanitizer, nativeOnly) {
        ngDevMode && assertNotSame(value, (/** @type {?} */ (NO_CHANGE)), 'Incoming value should never be NO_CHANGE.');
        /** @type {?} */
        const element = (/** @type {?} */ (getNativeByTNode(tNode, lView)));
        /** @type {?} */
        let inputData = tNode.inputs;
        /** @type {?} */
        let dataValue;
        if (!nativeOnly && inputData != null && (dataValue = inputData[propName])) {
            setInputsForProperty(tView, lView, dataValue, propName, value);
            if (isComponentHost(tNode))
                markDirtyIfOnPush(lView, tNode.index);
            if (ngDevMode) {
                setNgReflectProperties(lView, element, tNode.type, dataValue, value);
            }
        }
        else if (tNode.type === 3 /* Element */) {
            propName = mapPropName(propName);
            if (ngDevMode) {
                validateAgainstEventProperties(propName);
                if (!validateProperty(tView, lView, element, propName, tNode)) {
                    // Return here since we only log warnings for unknown properties.
                    warnAboutUnknownProperty(propName, tNode);
                    return;
                }
                ngDevMode.rendererSetProperty++;
            }
            // It is assumed that the sanitizer is only added when the compiler determines that the
            // property is risky, so sanitization can be done without further checks.
            value = sanitizer != null ? ((/** @type {?} */ (sanitizer(value, tNode.tagName || '', propName)))) : value;
            if (isProceduralRenderer(renderer)) {
                renderer.setProperty((/** @type {?} */ (element)), propName, value);
            }
            else if (!isAnimationProp(propName)) {
                ((/** @type {?} */ (element))).setProperty ? ((/** @type {?} */ (element))).setProperty(propName, value) :
                    ((/** @type {?} */ (element)))[propName] = value;
            }
        }
        else if (tNode.type === 0 /* Container */) {
            // If the node is a container and the property didn't
            // match any of the inputs or schemas we should throw.
            if (ngDevMode && !matchingSchemas(tView, lView, tNode.tagName)) {
                warnAboutUnknownProperty(propName, tNode);
            }
        }
    }
    /**
     * If node is an OnPush component, marks its LView dirty.
     * @param {?} lView
     * @param {?} viewIndex
     * @return {?}
     */
    function markDirtyIfOnPush(lView, viewIndex) {
        ngDevMode && assertLView(lView);
        /** @type {?} */
        const childComponentLView = getComponentLViewByIndex(viewIndex, lView);
        if (!(childComponentLView[FLAGS] & 16 /* CheckAlways */)) {
            childComponentLView[FLAGS] |= 64 /* Dirty */;
        }
    }
    /**
     * @param {?} lView
     * @param {?} element
     * @param {?} type
     * @param {?} attrName
     * @param {?} value
     * @return {?}
     */
    function setNgReflectProperty(lView, element, type, attrName, value) {
        /** @type {?} */
        const renderer = lView[RENDERER];
        attrName = normalizeDebugBindingName(attrName);
        /** @type {?} */
        const debugValue = normalizeDebugBindingValue(value);
        if (type === 3 /* Element */) {
            if (value == null) {
                isProceduralRenderer(renderer) ? renderer.removeAttribute(((/** @type {?} */ (element))), attrName) :
                    ((/** @type {?} */ (element))).removeAttribute(attrName);
            }
            else {
                isProceduralRenderer(renderer) ?
                    renderer.setAttribute(((/** @type {?} */ (element))), attrName, debugValue) :
                    ((/** @type {?} */ (element))).setAttribute(attrName, debugValue);
            }
        }
        else {
            /** @type {?} */
            const textContent = `bindings=${JSON.stringify({ [attrName]: debugValue }, null, 2)}`;
            if (isProceduralRenderer(renderer)) {
                renderer.setValue(((/** @type {?} */ (element))), textContent);
            }
            else {
                ((/** @type {?} */ (element))).textContent = textContent;
            }
        }
    }
    /**
     * @param {?} lView
     * @param {?} element
     * @param {?} type
     * @param {?} dataValue
     * @param {?} value
     * @return {?}
     */
    function setNgReflectProperties(lView, element, type, dataValue, value) {
        if (type === 3 /* Element */ || type === 0 /* Container */) {
            /**
             * dataValue is an array containing runtime input or output names for the directives:
             * i+0: directive instance index
             * i+1: privateName
             *
             * e.g. [0, 'change', 'change-minified']
             * we want to set the reflected property with the privateName: dataValue[i+1]
             */
            for (let i = 0; i < dataValue.length; i += 2) {
                setNgReflectProperty(lView, element, type, (/** @type {?} */ (dataValue[i + 1])), value);
            }
        }
    }
    /**
     * @param {?} tView
     * @param {?} lView
     * @param {?} element
     * @param {?} propName
     * @param {?} tNode
     * @return {?}
     */
    function validateProperty(tView, lView, element, propName, tNode) {
        // If `schemas` is set to `null`, that's an indication that this Component was compiled in AOT
        // mode where this check happens at compile time. In JIT mode, `schemas` is always present and
        // defined as an array (as an empty array in case `schemas` field is not defined) and we should
        // execute the check below.
        if (tView.schemas === null)
            return true;
        // The property is considered valid if the element matches the schema, it exists on the element
        // or it is synthetic, and we are in a browser context (web worker nodes should be skipped).
        if (matchingSchemas(tView, lView, tNode.tagName) || propName in element ||
            isAnimationProp(propName)) {
            return true;
        }
        // Note: `typeof Node` returns 'function' in most browsers, but on IE it is 'object' so we
        // need to account for both here, while being careful for `typeof null` also returning 'object'.
        return typeof Node === 'undefined' || Node === null || !(element instanceof Node);
    }
    /**
     * @param {?} tView
     * @param {?} lView
     * @param {?} tagName
     * @return {?}
     */
    function matchingSchemas(tView, lView, tagName) {
        /** @type {?} */
        const schemas = tView.schemas;
        if (schemas !== null) {
            for (let i = 0; i < schemas.length; i++) {
                /** @type {?} */
                const schema = schemas[i];
                if (schema === NO_ERRORS_SCHEMA ||
                    schema === CUSTOM_ELEMENTS_SCHEMA && tagName && tagName.indexOf('-') > -1) {
                    return true;
                }
            }
        }
        return false;
    }
    /**
     * Logs a warning that a property is not supported on an element.
     * @param {?} propName Name of the invalid property.
     * @param {?} tNode Node on which we encountered the property.
     * @return {?}
     */
    function warnAboutUnknownProperty(propName, tNode) {
        console.warn(`Can't bind to '${propName}' since it isn't a known property of '${tNode.tagName}'.`);
    }
    /**
     * Instantiate a root component.
     * @template T
     * @param {?} tView
     * @param {?} lView
     * @param {?} def
     * @return {?}
     */
    function instantiateRootComponent(tView, lView, def) {
        /** @type {?} */
        const rootTNode = getPreviousOrParentTNode();
        if (tView.firstCreatePass) {
            if (def.providersResolver)
                def.providersResolver(def);
            generateExpandoInstructionBlock(tView, rootTNode, 1);
            baseResolveDirective(tView, lView, def);
        }
        /** @type {?} */
        const directive = getNodeInjectable(lView, tView, lView.length - 1, (/** @type {?} */ (rootTNode)));
        attachPatchData(directive, lView);
        /** @type {?} */
        const native = getNativeByTNode(rootTNode, lView);
        if (native) {
            attachPatchData(native, lView);
        }
        return directive;
    }
    /**
     * Resolve the matched directives on a node.
     * @param {?} tView
     * @param {?} lView
     * @param {?} tNode
     * @param {?} localRefs
     * @return {?}
     */
    function resolveDirectives(tView, lView, tNode, localRefs) {
        // Please make sure to have explicit type for `exportsMap`. Inferred type triggers bug in
        // tsickle.
        ngDevMode && assertFirstCreatePass(tView);
        /** @type {?} */
        let hasDirectives = false;
        if (getBindingsEnabled()) {
            /** @type {?} */
            const directiveDefs = findDirectiveDefMatches(tView, lView, tNode);
            /** @type {?} */
            const exportsMap = localRefs === null ? null : { '': -1 };
            if (directiveDefs !== null) {
                /** @type {?} */
                let totalDirectiveHostVars = 0;
                hasDirectives = true;
                initTNodeFlags(tNode, tView.data.length, directiveDefs.length);
                // When the same token is provided by several directives on the same node, some rules apply in
                // the viewEngine:
                // - viewProviders have priority over providers
                // - the last directive in NgModule.declarations has priority over the previous one
                // So to match these rules, the order in which providers are added in the arrays is very
                // important.
                for (let i = 0; i < directiveDefs.length; i++) {
                    /** @type {?} */
                    const def = directiveDefs[i];
                    if (def.providersResolver)
                        def.providersResolver(def);
                }
                generateExpandoInstructionBlock(tView, tNode, directiveDefs.length);
                /** @type {?} */
                let preOrderHooksFound = false;
                /** @type {?} */
                let preOrderCheckHooksFound = false;
                for (let i = 0; i < directiveDefs.length; i++) {
                    /** @type {?} */
                    const def = directiveDefs[i];
                    // Merge the attrs in the order of matches. This assumes that the first directive is the
                    // component itself, so that the component has the least priority.
                    tNode.mergedAttrs = mergeHostAttrs(tNode.mergedAttrs, def.hostAttrs);
                    baseResolveDirective(tView, lView, def);
                    saveNameToExportMap((/** @type {?} */ (tView.data)).length - 1, def, exportsMap);
                    if (def.contentQueries !== null)
                        tNode.flags |= 8 /* hasContentQuery */;
                    if (def.hostBindings !== null || def.hostAttrs !== null || def.hostVars !== 0)
                        tNode.flags |= 128 /* hasHostBindings */;
                    // Only push a node index into the preOrderHooks array if this is the first
                    // pre-order hook found on this node.
                    if (!preOrderHooksFound && (def.onChanges || def.onInit || def.doCheck)) {
                        // We will push the actual hook function into this array later during dir instantiation.
                        // We cannot do it now because we must ensure hooks are registered in the same
                        // order that directives are created (i.e. injection order).
                        (tView.preOrderHooks || (tView.preOrderHooks = [])).push(tNode.index - HEADER_OFFSET);
                        preOrderHooksFound = true;
                    }
                    if (!preOrderCheckHooksFound && (def.onChanges || def.doCheck)) {
                        (tView.preOrderCheckHooks || (tView.preOrderCheckHooks = []))
                            .push(tNode.index - HEADER_OFFSET);
                        preOrderCheckHooksFound = true;
                    }
                    addHostBindingsToExpandoInstructions(tView, def);
                    totalDirectiveHostVars += def.hostVars;
                }
                initializeInputAndOutputAliases(tView, tNode);
                growHostVarsSpace(tView, lView, totalDirectiveHostVars);
            }
            if (exportsMap)
                cacheMatchingLocalNames(tNode, localRefs, exportsMap);
        }
        // Merge the template attrs last so that they have the highest priority.
        tNode.mergedAttrs = mergeHostAttrs(tNode.mergedAttrs, tNode.attrs);
        return hasDirectives;
    }
    /**
     * Add `hostBindings` to the `TView.expandoInstructions`.
     *
     * @param {?} tView `TView` to which the `hostBindings` should be added.
     * @param {?} def `ComponentDef`/`DirectiveDef`, which contains the `hostVars`/`hostBindings` to add.
     * @return {?}
     */
    function addHostBindingsToExpandoInstructions(tView, def) {
        ngDevMode && assertFirstCreatePass(tView);
        /** @type {?} */
        const expando = (/** @type {?} */ (tView.expandoInstructions));
        // TODO(misko): PERF we are adding `hostBindings` even if there is nothing to add! This is
        // suboptimal for performance. `def.hostBindings` may be null,
        // but we still need to push null to the array as a placeholder
        // to ensure the directive counter is incremented (so host
        // binding functions always line up with the corrective directive).
        // This is suboptimal for performance. See `currentDirectiveIndex`
        //  comment in `setHostBindingsByExecutingExpandoInstructions` for more
        // details.  expando.push(def.hostBindings);
        expando.push(def.hostBindings);
        /** @type {?} */
        const hostVars = def.hostVars;
        if (hostVars !== 0) {
            expando.push(def.hostVars);
        }
    }
    /**
     * Grow the `LView`, blueprint and `TView.data` to accommodate the `hostBindings`.
     *
     * To support locality we don't know ahead of time how many `hostVars` of the containing directives
     * we need to allocate. For this reason we allow growing these data structures as we discover more
     * directives to accommodate them.
     *
     * @param {?} tView `TView` which needs to be grown.
     * @param {?} lView `LView` which needs to be grown.
     * @param {?} count Size by which we need to grow the data structures.
     * @return {?}
     */
    function growHostVarsSpace(tView, lView, count) {
        ngDevMode && assertFirstCreatePass(tView);
        ngDevMode && assertSame(tView, lView[TVIEW], '`LView` must be associated with `TView`!');
        for (let i = 0; i < count; i++) {
            lView.push(NO_CHANGE);
            tView.blueprint.push(NO_CHANGE);
            tView.data.push(null);
        }
    }
    /**
     * Instantiate all the directives that were previously resolved on the current node.
     * @param {?} tView
     * @param {?} lView
     * @param {?} tNode
     * @param {?} native
     * @return {?}
     */
    function instantiateAllDirectives(tView, lView, tNode, native) {
        /** @type {?} */
        const start = tNode.directiveStart;
        /** @type {?} */
        const end = tNode.directiveEnd;
        if (!tView.firstCreatePass) {
            getOrCreateNodeInjectorForNode(tNode, lView);
        }
        attachPatchData(native, lView);
        /** @type {?} */
        const initialInputs = tNode.initialInputs;
        for (let i = start; i < end; i++) {
            /** @type {?} */
            const def = (/** @type {?} */ (tView.data[i]));
            /** @type {?} */
            const isComponent = isComponentDef(def);
            if (isComponent) {
                ngDevMode && assertNodeOfPossibleTypes(tNode, 3 /* Element */);
                addComponentLogic(lView, (/** @type {?} */ (tNode)), (/** @type {?} */ (def)));
            }
            /** @type {?} */
            const directive = getNodeInjectable(lView, tView, i, tNode);
            attachPatchData(directive, lView);
            if (initialInputs !== null) {
                setInputsFromAttrs(lView, i - start, directive, def, tNode, (/** @type {?} */ (initialInputs)));
            }
            if (isComponent) {
                /** @type {?} */
                const componentView = getComponentLViewByIndex(tNode.index, lView);
                componentView[CONTEXT] = directive;
            }
        }
    }
    /**
     * @param {?} tView
     * @param {?} lView
     * @param {?} tNode
     * @return {?}
     */
    function invokeDirectivesHostBindings(tView, lView, tNode) {
        /** @type {?} */
        const start = tNode.directiveStart;
        /** @type {?} */
        const end = tNode.directiveEnd;
        /** @type {?} */
        const expando = (/** @type {?} */ (tView.expandoInstructions));
        /** @type {?} */
        const firstCreatePass = tView.firstCreatePass;
        /** @type {?} */
        const elementIndex = tNode.index - HEADER_OFFSET;
        /** @type {?} */
        const currentDirectiveIndex = getCurrentDirectiveIndex();
        try {
            setSelectedIndex(elementIndex);
            for (let dirIndex = start; dirIndex < end; dirIndex++) {
                /** @type {?} */
                const def = (/** @type {?} */ (tView.data[dirIndex]));
                /** @type {?} */
                const directive = lView[dirIndex];
                setCurrentDirectiveIndex(dirIndex);
                if (def.hostBindings !== null || def.hostVars !== 0 || def.hostAttrs !== null) {
                    invokeHostBindingsInCreationMode(def, directive);
                }
                else if (firstCreatePass) {
                    expando.push(null);
                }
            }
        }
        finally {
            setSelectedIndex(-1);
            setCurrentDirectiveIndex(currentDirectiveIndex);
        }
    }
    /**
     * Invoke the host bindings in creation mode.
     *
     * @param {?} def `DirectiveDef` which may contain the `hostBindings` function.
     * @param {?} directive Instance of directive.
     * @return {?}
     */
    function invokeHostBindingsInCreationMode(def, directive) {
        if (def.hostBindings !== null) {
            (/** @type {?} */ (def.hostBindings))(1 /* Create */, directive);
        }
    }
    /**
     * Generates a new block in TView.expandoInstructions for this node.
     *
     * Each expando block starts with the element index (turned negative so we can distinguish
     * it from the hostVar count) and the directive count. See more in VIEW_DATA.md.
     * @param {?} tView
     * @param {?} tNode
     * @param {?} directiveCount
     * @return {?}
     */
    function generateExpandoInstructionBlock(tView, tNode, directiveCount) {
        ngDevMode &&
            assertEqual(tView.firstCreatePass, true, 'Expando block should only be generated on first create pass.');
        // Important: In JS `-x` and `0-x` is not the same! If `x===0` then `-x` will produce `-0` which
        // requires non standard math arithmetic and it can prevent VM optimizations.
        // `0-0` will always produce `0` and will not cause a potential deoptimization in VM.
        /** @type {?} */
        const elementIndex = HEADER_OFFSET - tNode.index;
        /** @type {?} */
        const providerStartIndex = tNode.providerIndexes & 65535 /* ProvidersStartIndexMask */;
        /** @type {?} */
        const providerCount = tView.data.length - providerStartIndex;
        (tView.expandoInstructions || (tView.expandoInstructions = []))
            .push(elementIndex, providerCount, directiveCount);
    }
    /**
     * Matches the current node against all available selectors.
     * If a component is matched (at most one), it is returned in first position in the array.
     * @param {?} tView
     * @param {?} viewData
     * @param {?} tNode
     * @return {?}
     */
    function findDirectiveDefMatches(tView, viewData, tNode) {
        ngDevMode && assertFirstCreatePass(tView);
        ngDevMode &&
            assertNodeOfPossibleTypes(tNode, 3 /* Element */, 4 /* ElementContainer */, 0 /* Container */);
        /** @type {?} */
        const registry = tView.directiveRegistry;
        /** @type {?} */
        let matches = null;
        if (registry) {
            for (let i = 0; i < registry.length; i++) {
                /** @type {?} */
                const def = (/** @type {?} */ (registry[i]));
                if (isNodeMatchingSelectorList(tNode, (/** @type {?} */ (def.selectors)), /* isProjectionMode */ false)) {
                    matches || (matches = ngDevMode ? new MatchesArray() : []);
                    diPublicInInjector(getOrCreateNodeInjectorForNode(tNode, viewData), tView, def.type);
                    if (isComponentDef(def)) {
                        if (tNode.flags & 2 /* isComponentHost */)
                            throwMultipleComponentError(tNode);
                        markAsComponentHost(tView, tNode);
                        // The component is always stored first with directives after.
                        matches.unshift(def);
                    }
                    else {
                        matches.push(def);
                    }
                }
            }
        }
        return matches;
    }
    /**
     * Marks a given TNode as a component's host. This consists of:
     * - setting appropriate TNode flags;
     * - storing index of component's host element so it will be queued for view refresh during CD.
     * @param {?} tView
     * @param {?} hostTNode
     * @return {?}
     */
    function markAsComponentHost(tView, hostTNode) {
        ngDevMode && assertFirstCreatePass(tView);
        hostTNode.flags |= 2 /* isComponentHost */;
        (tView.components || (tView.components = ngDevMode ? new TViewComponents() : []))
            .push(hostTNode.index);
    }
    /**
     * Caches local names and their matching directive indices for query and template lookups.
     * @param {?} tNode
     * @param {?} localRefs
     * @param {?} exportsMap
     * @return {?}
     */
    function cacheMatchingLocalNames(tNode, localRefs, exportsMap) {
        if (localRefs) {
            /** @type {?} */
            const localNames = tNode.localNames = ngDevMode ? new TNodeLocalNames() : [];
            // Local names must be stored in tNode in the same order that localRefs are defined
            // in the template to ensure the data is loaded in the same slots as their refs
            // in the template (for template queries).
            for (let i = 0; i < localRefs.length; i += 2) {
                /** @type {?} */
                const index = exportsMap[localRefs[i + 1]];
                if (index == null)
                    throw new Error(`Export of name '${localRefs[i + 1]}' not found!`);
                localNames.push(localRefs[i], index);
            }
        }
    }
    /**
     * Builds up an export map as directives are created, so local refs can be quickly mapped
     * to their directive instances.
     * @param {?} index
     * @param {?} def
     * @param {?} exportsMap
     * @return {?}
     */
    function saveNameToExportMap(index, def, exportsMap) {
        if (exportsMap) {
            if (def.exportAs) {
                for (let i = 0; i < def.exportAs.length; i++) {
                    exportsMap[def.exportAs[i]] = index;
                }
            }
            if (isComponentDef(def))
                exportsMap[''] = index;
        }
    }
    /**
     * Initializes the flags on the current node, setting all indices to the initial index,
     * the directive count to 0, and adding the isComponent flag.
     * @param {?} tNode
     * @param {?} index the initial index
     * @param {?} numberOfDirectives
     * @return {?}
     */
    function initTNodeFlags(tNode, index, numberOfDirectives) {
        ngDevMode &&
            assertNotEqual(numberOfDirectives, tNode.directiveEnd - tNode.directiveStart, 'Reached the max number of directives');
        tNode.flags |= 1 /* isDirectiveHost */;
        // When the first directive is created on a node, save the index
        tNode.directiveStart = index;
        tNode.directiveEnd = index + numberOfDirectives;
        tNode.providerIndexes = index;
    }
    /**
     * @template T
     * @param {?} tView
     * @param {?} viewData
     * @param {?} def
     * @return {?}
     */
    function baseResolveDirective(tView, viewData, def) {
        tView.data.push(def);
        /** @type {?} */
        const directiveFactory = def.factory || (((/** @type {?} */ (def))).factory = getFactoryDef(def.type, true));
        /** @type {?} */
        const nodeInjectorFactory = new NodeInjectorFactory(directiveFactory, isComponentDef(def), null);
        tView.blueprint.push(nodeInjectorFactory);
        viewData.push(nodeInjectorFactory);
    }
    /**
     * @template T
     * @param {?} lView
     * @param {?} hostTNode
     * @param {?} def
     * @return {?}
     */
    function addComponentLogic(lView, hostTNode, def) {
        /** @type {?} */
        const native = (/** @type {?} */ (getNativeByTNode(hostTNode, lView)));
        /** @type {?} */
        const tView = getOrCreateTComponentView(def);
        // Only component views should be added to the view tree directly. Embedded views are
        // accessed through their containers because they may be removed / re-added later.
        /** @type {?} */
        const rendererFactory = lView[RENDERER_FACTORY];
        /** @type {?} */
        const componentView = addToViewTree(lView, createLView(lView, tView, null, def.onPush ? 64 /* Dirty */ : 16 /* CheckAlways */, native, (/** @type {?} */ (hostTNode)), rendererFactory, rendererFactory.createRenderer(native, def)));
        // Component view will always be created before any injected LContainers,
        // so this is a regular element, wrap it with the component view
        lView[hostTNode.index] = componentView;
    }
    /**
     * @param {?} tNode
     * @param {?} lView
     * @param {?} name
     * @param {?} value
     * @param {?} sanitizer
     * @param {?} namespace
     * @return {?}
     */
    function elementAttributeInternal(tNode, lView, name, value, sanitizer, namespace) {
        ngDevMode && assertNotSame(value, (/** @type {?} */ (NO_CHANGE)), 'Incoming value should never be NO_CHANGE.');
        ngDevMode && validateAgainstEventAttributes(name);
        /** @type {?} */
        const element = (/** @type {?} */ (getNativeByTNode(tNode, lView)));
        /** @type {?} */
        const renderer = lView[RENDERER];
        if (value == null) {
            ngDevMode && ngDevMode.rendererRemoveAttribute++;
            isProceduralRenderer(renderer) ? renderer.removeAttribute(element, name, namespace) :
                element.removeAttribute(name);
        }
        else {
            ngDevMode && ngDevMode.rendererSetAttribute++;
            /** @type {?} */
            const strValue = sanitizer == null ? renderStringify(value) : sanitizer(value, tNode.tagName || '', name);
            if (isProceduralRenderer(renderer)) {
                renderer.setAttribute(element, name, strValue, namespace);
            }
            else {
                namespace ? element.setAttributeNS(namespace, name, strValue) :
                    element.setAttribute(name, strValue);
            }
        }
    }
    /**
     * Sets initial input properties on directive instances from attribute data
     *
     * @template T
     * @param {?} lView Current LView that is being processed.
     * @param {?} directiveIndex Index of the directive in directives array
     * @param {?} instance Instance of the directive on which to set the initial inputs
     * @param {?} def The directive def that contains the list of inputs
     * @param {?} tNode The static data for this node
     * @param {?} initialInputData
     * @return {?}
     */
    function setInputsFromAttrs(lView, directiveIndex, instance, def, tNode, initialInputData) {
        /** @type {?} */
        const initialInputs = (/** @type {?} */ (initialInputData))[directiveIndex];
        if (initialInputs !== null) {
            /** @type {?} */
            const setInput = def.setInput;
            for (let i = 0; i < initialInputs.length;) {
                /** @type {?} */
                const publicName = initialInputs[i++];
                /** @type {?} */
                const privateName = initialInputs[i++];
                /** @type {?} */
                const value = initialInputs[i++];
                if (setInput !== null) {
                    (/** @type {?} */ (def.setInput))(instance, value, publicName, privateName);
                }
                else {
                    ((/** @type {?} */ (instance)))[privateName] = value;
                }
                if (ngDevMode) {
                    /** @type {?} */
                    const nativeElement = (/** @type {?} */ (getNativeByTNode(tNode, lView)));
                    setNgReflectProperty(lView, nativeElement, tNode.type, privateName, value);
                }
            }
        }
    }
    /**
     * Generates initialInputData for a node and stores it in the template's static storage
     * so subsequent template invocations don't have to recalculate it.
     *
     * initialInputData is an array containing values that need to be set as input properties
     * for directives on this node, but only once on creation. We need this array to support
     * the case where you set an \@Input property of a directive using attribute-like syntax.
     * e.g. if you have a `name` \@Input, you can set it once like this:
     *
     * <my-component name="Bess"></my-component>
     *
     * @param {?} inputs The list of inputs from the directive def
     * @param {?} attrs The static attrs on this node
     * @return {?}
     */
    function generateInitialInputs(inputs, attrs) {
        /** @type {?} */
        let inputsToStore = null;
        /** @type {?} */
        let i = 0;
        while (i < attrs.length) {
            /** @type {?} */
            const attrName = attrs[i];
            if (attrName === 0 /* NamespaceURI */) {
                // We do not allow inputs on namespaced attributes.
                i += 4;
                continue;
            }
            else if (attrName === 5 /* ProjectAs */) {
                // Skip over the `ngProjectAs` value.
                i += 2;
                continue;
            }
            // If we hit any other attribute markers, we're done anyway. None of those are valid inputs.
            if (typeof attrName === 'number')
                break;
            if (inputs.hasOwnProperty((/** @type {?} */ (attrName)))) {
                if (inputsToStore === null)
                    inputsToStore = [];
                inputsToStore.push((/** @type {?} */ (attrName)), inputs[(/** @type {?} */ (attrName))], (/** @type {?} */ (attrs[i + 1])));
            }
            i += 2;
        }
        return inputsToStore;
    }
    //////////////////////////
    //// ViewContainer & View
    //////////////////////////
    // Not sure why I need to do `any` here but TS complains later.
    /** @type {?} */
    const LContainerArray = ((typeof ngDevMode === 'undefined' || ngDevMode) && initNgDevMode()) &&
        createNamedArrayType('LContainer');
    /**
     * Creates a LContainer, either from a container instruction, or for a ViewContainerRef.
     *
     * @param {?} hostNative The host element for the LContainer
     * @param {?} currentView The parent view of the LContainer
     * @param {?} native The native comment element
     * @param {?} tNode
     * @return {?} LContainer
     */
    function createLContainer(hostNative, currentView, native, tNode) {
        ngDevMode && assertLView(currentView);
        ngDevMode && !isProceduralRenderer(currentView[RENDERER]) && assertDomNode(native);
        // https://jsperf.com/array-literal-vs-new-array-really
        /** @type {?} */
        const lContainer = new (ngDevMode ? LContainerArray : Array)(hostNative, // host native
        true, // Boolean `true` in this position signifies that this is an `LContainer`
        false, // has transplanted views
        currentView, // parent
        null, // next
        0, // transplanted views to refresh count
        tNode, // t_host
        native, // native,
        null, // view refs
        null);
        ngDevMode &&
            assertEqual(lContainer.length, CONTAINER_HEADER_OFFSET, 'Should allocate correct number of slots for LContainer header.');
        ngDevMode && attachLContainerDebug(lContainer);
        return lContainer;
    }
    /**
     * Goes over embedded views (ones created through ViewContainerRef APIs) and refreshes
     * them by executing an associated template function.
     * @param {?} lView
     * @return {?}
     */
    function refreshEmbeddedViews(lView) {
        for (let lContainer = getFirstLContainer(lView); lContainer !== null; lContainer = getNextLContainer(lContainer)) {
            for (let i = CONTAINER_HEADER_OFFSET; i < lContainer.length; i++) {
                /** @type {?} */
                const embeddedLView = lContainer[i];
                /** @type {?} */
                const embeddedTView = embeddedLView[TVIEW];
                ngDevMode && assertDefined(embeddedTView, 'TView must be allocated');
                if (viewAttachedToChangeDetector(embeddedLView)) {
                    refreshView(embeddedTView, embeddedLView, embeddedTView.template, (/** @type {?} */ (embeddedLView[CONTEXT])));
                }
            }
        }
    }
    /**
     * Mark transplanted views as needing to be refreshed at their insertion points.
     *
     * @param {?} lView The `LView` that may have transplanted views.
     * @return {?}
     */
    function markTransplantedViewsForRefresh(lView) {
        for (let lContainer = getFirstLContainer(lView); lContainer !== null; lContainer = getNextLContainer(lContainer)) {
            if (!lContainer[HAS_TRANSPLANTED_VIEWS])
                continue;
            /** @type {?} */
            const movedViews = (/** @type {?} */ (lContainer[MOVED_VIEWS]));
            ngDevMode && assertDefined(movedViews, 'Transplanted View flags set but missing MOVED_VIEWS');
            for (let i = 0; i < movedViews.length; i++) {
                /** @type {?} */
                const movedLView = (/** @type {?} */ (movedViews[i]));
                /** @type {?} */
                const insertionLContainer = (/** @type {?} */ (movedLView[PARENT]));
                ngDevMode && assertLContainer(insertionLContainer);
                // We don't want to increment the counter if the moved LView was already marked for
                // refresh.
                if ((movedLView[FLAGS] & 1024 /* RefreshTransplantedView */) === 0) {
                    updateTransplantedViewCount(insertionLContainer, 1);
                }
                // Note, it is possible that the `movedViews` is tracking views that are transplanted *and*
                // those that aren't (declaration component === insertion component). In the latter case,
                // it's fine to add the flag, as we will clear it immediately in
                // `refreshEmbeddedViews` for the view currently being refreshed.
                movedLView[FLAGS] |= 1024 /* RefreshTransplantedView */;
            }
        }
    }
    /////////////
    /**
     * Refreshes components by entering the component view and processing its bindings, queries, etc.
     *
     * @param {?} hostLView
     * @param {?} componentHostIdx  Element index in LView[] (adjusted for HEADER_OFFSET)
     * @return {?}
     */
    function refreshComponent(hostLView, componentHostIdx) {
        ngDevMode && assertEqual(isCreationMode(hostLView), false, 'Should be run in update mode');
        /** @type {?} */
        const componentView = getComponentLViewByIndex(componentHostIdx, hostLView);
        // Only attached components that are CheckAlways or OnPush and dirty should be refreshed
        if (viewAttachedToChangeDetector(componentView)) {
            /** @type {?} */
            const tView = componentView[TVIEW];
            if (componentView[FLAGS] & (16 /* CheckAlways */ | 64 /* Dirty */)) {
                refreshView(tView, componentView, tView.template, componentView[CONTEXT]);
            }
            else if (componentView[TRANSPLANTED_VIEWS_TO_REFRESH] > 0) {
                // Only attached components that are CheckAlways or OnPush and dirty should be refreshed
                refreshContainsDirtyView(componentView);
            }
        }
    }
    /**
     * Refreshes all transplanted views marked with `LViewFlags.RefreshTransplantedView` that are
     * children or descendants of the given lView.
     *
     * @param {?} lView The lView which contains descendant transplanted views that need to be refreshed.
     * @return {?}
     */
    function refreshContainsDirtyView(lView) {
        for (let lContainer = getFirstLContainer(lView); lContainer !== null; lContainer = getNextLContainer(lContainer)) {
            for (let i = CONTAINER_HEADER_OFFSET; i < lContainer.length; i++) {
                /** @type {?} */
                const embeddedLView = lContainer[i];
                if (embeddedLView[FLAGS] & 1024 /* RefreshTransplantedView */) {
                    /** @type {?} */
                    const embeddedTView = embeddedLView[TVIEW];
                    ngDevMode && assertDefined(embeddedTView, 'TView must be allocated');
                    refreshView(embeddedTView, embeddedLView, embeddedTView.template, (/** @type {?} */ (embeddedLView[CONTEXT])));
                }
                else if (embeddedLView[TRANSPLANTED_VIEWS_TO_REFRESH] > 0) {
                    refreshContainsDirtyView(embeddedLView);
                }
            }
        }
        /** @type {?} */
        const tView = lView[TVIEW];
        // Refresh child component views.
        /** @type {?} */
        const components = tView.components;
        if (components !== null) {
            for (let i = 0; i < components.length; i++) {
                /** @type {?} */
                const componentView = getComponentLViewByIndex(components[i], lView);
                // Only attached components that are CheckAlways or OnPush and dirty should be refreshed
                if (viewAttachedToChangeDetector(componentView) &&
                    componentView[TRANSPLANTED_VIEWS_TO_REFRESH] > 0) {
                    refreshContainsDirtyView(componentView);
                }
            }
        }
    }
    /**
     * @param {?} hostLView
     * @param {?} componentHostIdx
     * @return {?}
     */
    function renderComponent(hostLView, componentHostIdx) {
        ngDevMode && assertEqual(isCreationMode(hostLView), true, 'Should be run in creation mode');
        /** @type {?} */
        const componentView = getComponentLViewByIndex(componentHostIdx, hostLView);
        /** @type {?} */
        const componentTView = componentView[TVIEW];
        syncViewWithBlueprint(componentTView, componentView);
        renderView(componentTView, componentView, componentView[CONTEXT]);
    }
    /**
     * Syncs an LView instance with its blueprint if they have gotten out of sync.
     *
     * Typically, blueprints and their view instances should always be in sync, so the loop here
     * will be skipped. However, consider this case of two components side-by-side:
     *
     * App template:
     * ```
     * <comp></comp>
     * <comp></comp>
     * ```
     *
     * The following will happen:
     * 1. App template begins processing.
     * 2. First <comp> is matched as a component and its LView is created.
     * 3. Second <comp> is matched as a component and its LView is created.
     * 4. App template completes processing, so it's time to check child templates.
     * 5. First <comp> template is checked. It has a directive, so its def is pushed to blueprint.
     * 6. Second <comp> template is checked. Its blueprint has been updated by the first
     * <comp> template, but its LView was created before this update, so it is out of sync.
     *
     * Note that embedded views inside ngFor loops will never be out of sync because these views
     * are processed as soon as they are created.
     *
     * @param {?} tView The `TView` that contains the blueprint for syncing
     * @param {?} lView The view to sync
     * @return {?}
     */
    function syncViewWithBlueprint(tView, lView) {
        for (let i = lView.length; i < tView.blueprint.length; i++) {
            lView.push(tView.blueprint[i]);
        }
    }
    /**
     * Adds LView or LContainer to the end of the current view tree.
     *
     * This structure will be used to traverse through nested views to remove listeners
     * and call onDestroy callbacks.
     *
     * @template T
     * @param {?} lView The view where LView or LContainer should be added
     * @param {?} lViewOrLContainer The LView or LContainer to add to the view tree
     * @return {?} The state passed in
     */
    function addToViewTree(lView, lViewOrLContainer) {
        // TODO(benlesh/misko): This implementation is incorrect, because it always adds the LContainer
        // to the end of the queue, which means if the developer retrieves the LContainers from RNodes out
        // of order, the change detection will run out of order, as the act of retrieving the the
        // LContainer from the RNode is what adds it to the queue.
        if (lView[CHILD_HEAD]) {
            (/** @type {?} */ (lView[CHILD_TAIL]))[NEXT] = lViewOrLContainer;
        }
        else {
            lView[CHILD_HEAD] = lViewOrLContainer;
        }
        lView[CHILD_TAIL] = lViewOrLContainer;
        return lViewOrLContainer;
    }
    ///////////////////////////////
    //// Change detection
    ///////////////////////////////
    /**
     * Marks current view and all ancestors dirty.
     *
     * Returns the root view because it is found as a byproduct of marking the view tree
     * dirty, and can be used by methods that consume markViewDirty() to easily schedule
     * change detection. Otherwise, such methods would need to traverse up the view tree
     * an additional time to get the root view and schedule a tick on it.
     *
     * @param {?} lView The starting LView to mark dirty
     * @return {?} the root LView
     */
    function markViewDirty(lView) {
        while (lView) {
            lView[FLAGS] |= 64 /* Dirty */;
            /** @type {?} */
            const parent = getLViewParent(lView);
            // Stop traversing up as soon as you find a root view that wasn't attached to any container
            if (isRootView(lView) && !parent) {
                return lView;
            }
            // continue otherwise
            lView = (/** @type {?} */ (parent));
        }
        return null;
    }
    /**
     * Used to schedule change detection on the whole application.
     *
     * Unlike `tick`, `scheduleTick` coalesces multiple calls into one change detection run.
     * It is usually called indirectly by calling `markDirty` when the view needs to be
     * re-rendered.
     *
     * Typically `scheduleTick` uses `requestAnimationFrame` to coalesce multiple
     * `scheduleTick` requests. The scheduling function can be overridden in
     * `renderComponent`'s `scheduler` option.
     * @param {?} rootContext
     * @param {?} flags
     * @return {?}
     */
    function scheduleTick(rootContext, flags) {
        /** @type {?} */
        const nothingScheduled = rootContext.flags === 0 /* Empty */;
        rootContext.flags |= flags;
        if (nothingScheduled && rootContext.clean == _CLEAN_PROMISE) {
            /** @type {?} */
            let res;
            rootContext.clean = new Promise((/**
             * @param {?} r
             * @return {?}
             */
            (r) => res = r));
            rootContext.scheduler((/**
             * @return {?}
             */
            () => {
                if (rootContext.flags & 1 /* DetectChanges */) {
                    rootContext.flags &= ~1 /* DetectChanges */;
                    tickRootContext(rootContext);
                }
                if (rootContext.flags & 2 /* FlushPlayers */) {
                    rootContext.flags &= ~2 /* FlushPlayers */;
                    /** @type {?} */
                    const playerHandler = rootContext.playerHandler;
                    if (playerHandler) {
                        playerHandler.flushPlayers();
                    }
                }
                rootContext.clean = _CLEAN_PROMISE;
                (/** @type {?} */ (res))(null);
            }));
        }
    }
    /**
     * @param {?} rootContext
     * @return {?}
     */
    function tickRootContext(rootContext) {
        for (let i = 0; i < rootContext.components.length; i++) {
            /** @type {?} */
            const rootComponent = rootContext.components[i];
            /** @type {?} */
            const lView = (/** @type {?} */ (readPatchedLView(rootComponent)));
            /** @type {?} */
            const tView = lView[TVIEW];
            renderComponentOrTemplate(tView, lView, tView.template, rootComponent);
        }
    }
    /**
     * @template T
     * @param {?} tView
     * @param {?} lView
     * @param {?} context
     * @return {?}
     */
    function detectChangesInternal(tView, lView, context) {
        /** @type {?} */
        const rendererFactory = lView[RENDERER_FACTORY];
        if (rendererFactory.begin)
            rendererFactory.begin();
        try {
            refreshView(tView, lView, tView.template, context);
        }
        catch (error) {
            handleError(lView, error);
            throw error;
        }
        finally {
            if (rendererFactory.end)
                rendererFactory.end();
        }
    }
    /**
     * Synchronously perform change detection on a root view and its components.
     *
     * @param {?} lView The view which the change detection should be performed on.
     * @return {?}
     */
    function detectChangesInRootView(lView) {
        tickRootContext((/** @type {?} */ (lView[CONTEXT])));
    }
    /**
     * @template T
     * @param {?} tView
     * @param {?} view
     * @param {?} context
     * @return {?}
     */
    function checkNoChangesInternal(tView, view, context) {
        setCheckNoChangesMode(true);
        try {
            detectChangesInternal(tView, view, context);
        }
        finally {
            setCheckNoChangesMode(false);
        }
    }
    /**
     * Checks the change detector on a root view and its components, and throws if any changes are
     * detected.
     *
     * This is used in development mode to verify that running change detection doesn't
     * introduce other changes.
     *
     * @param {?} lView The view which the change detection should be checked on.
     * @return {?}
     */
    function checkNoChangesInRootView(lView) {
        setCheckNoChangesMode(true);
        try {
            detectChangesInRootView(lView);
        }
        finally {
            setCheckNoChangesMode(false);
        }
    }
    /**
     * @template T
     * @param {?} flags
     * @param {?} viewQueryFn
     * @param {?} component
     * @return {?}
     */
    function executeViewQueryFn(flags, viewQueryFn, component) {
        ngDevMode && assertDefined(viewQueryFn, 'View queries function to execute must be defined.');
        setCurrentQueryIndex(0);
        viewQueryFn(flags, component);
    }
    ///////////////////////////////
    //// Bindings & interpolations
    ///////////////////////////////
    /**
     * Stores meta-data for a property binding to be used by TestBed's `DebugElement.properties`.
     *
     * In order to support TestBed's `DebugElement.properties` we need to save, for each binding:
     * - a bound property name;
     * - a static parts of interpolated strings;
     *
     * A given property metadata is saved at the binding's index in the `TView.data` (in other words, a
     * property binding metadata will be stored in `TView.data` at the same index as a bound value in
     * `LView`). Metadata are represented as `INTERPOLATION_DELIMITER`-delimited string with the
     * following format:
     * - `propertyName` for bound properties;
     * - `propertyName�prefix�interpolation_static_part1�..interpolation_static_partN�suffix` for
     * interpolated properties.
     *
     * @param {?} tData `TData` where meta-data will be saved;
     * @param {?} tNode `TNode` that is a target of the binding;
     * @param {?} propertyName bound property name;
     * @param {?} bindingIndex binding index in `LView`
     * @param {...?} interpolationParts static interpolation parts (for property interpolations)
     * @return {?}
     */
    function storePropertyBindingMetadata(tData, tNode, propertyName, bindingIndex, ...interpolationParts) {
        // Binding meta-data are stored only the first time a given property instruction is processed.
        // Since we don't have a concept of the "first update pass" we need to check for presence of the
        // binding meta-data to decide if one should be stored (or if was stored already).
        if (tData[bindingIndex] === null) {
            if (tNode.inputs == null || !tNode.inputs[propertyName]) {
                /** @type {?} */
                const propBindingIdxs = tNode.propertyBindings || (tNode.propertyBindings = []);
                propBindingIdxs.push(bindingIndex);
                /** @type {?} */
                let bindingMetadata = propertyName;
                if (interpolationParts.length > 0) {
                    bindingMetadata +=
                        INTERPOLATION_DELIMITER + interpolationParts.join(INTERPOLATION_DELIMITER);
                }
                tData[bindingIndex] = bindingMetadata;
            }
        }
    }
    /** @type {?} */
    const CLEAN_PROMISE = _CLEAN_PROMISE;
    /**
     * @param {?} view
     * @return {?}
     */
    function getLCleanup(view) {
        // top level variables should not be exported for performance reasons (PERF_NOTES.md)
        return view[CLEANUP] || (view[CLEANUP] = ngDevMode ? new LCleanup() : []);
    }
    /**
     * @param {?} tView
     * @return {?}
     */
    function getTViewCleanup(tView) {
        return tView.cleanup || (tView.cleanup = ngDevMode ? new TCleanup() : []);
    }
    /**
     * There are cases where the sub component's renderer needs to be included
     * instead of the current renderer (see the componentSyntheticHost* instructions).
     * @param {?} currentDef
     * @param {?} tNode
     * @param {?} lView
     * @return {?}
     */
    function loadComponentRenderer(currentDef, tNode, lView) {
        // TODO(FW-2043): the `currentDef` is null when host bindings are invoked while creating root
        // component (see packages/core/src/render3/component.ts). This is not consistent with the process
        // of creating inner components, when current directive index is available in the state. In order
        // to avoid relying on current def being `null` (thus special-casing root component creation), the
        // process of creating root component should be unified with the process of creating inner
        // components.
        if (currentDef === null || isComponentDef(currentDef)) {
            lView = (/** @type {?} */ (unwrapLView(lView[tNode.index])));
        }
        return lView[RENDERER];
    }
    /**
     * Handles an error thrown in an LView.
     * @param {?} lView
     * @param {?} error
     * @return {?}
     */
    function handleError(lView, error) {
        /** @type {?} */
        const injector = lView[INJECTOR$1];
        /** @type {?} */
        const errorHandler = injector ? injector.get(ErrorHandler, null) : null;
        errorHandler && errorHandler.handleError(error);
    }
    /**
     * Set the inputs of directives at the current node to corresponding value.
     *
     * @param {?} tView The current TView
     * @param {?} lView the `LView` which contains the directives.
     * @param {?} inputs mapping between the public "input" name and privately-known,
     *        possibly minified, property names to write to.
     * @param {?} publicName
     * @param {?} value Value to set.
     * @return {?}
     */
    function setInputsForProperty(tView, lView, inputs, publicName, value) {
        for (let i = 0; i < inputs.length;) {
            /** @type {?} */
            const index = (/** @type {?} */ (inputs[i++]));
            /** @type {?} */
            const privateName = (/** @type {?} */ (inputs[i++]));
            /** @type {?} */
            const instance = lView[index];
            ngDevMode && assertDataInRange(lView, index);
            /** @type {?} */
            const def = (/** @type {?} */ (tView.data[index]));
            if (def.setInput !== null) {
                (/** @type {?} */ (def.setInput))(instance, value, publicName, privateName);
            }
            else {
                instance[privateName] = value;
            }
        }
    }
    /**
     * Updates a text binding at a given index in a given LView.
     * @param {?} lView
     * @param {?} index
     * @param {?} value
     * @return {?}
     */
    function textBindingInternal(lView, index, value) {
        ngDevMode && assertNotSame(value, (/** @type {?} */ (NO_CHANGE)), 'value should not be NO_CHANGE');
        ngDevMode && assertDataInRange(lView, index + HEADER_OFFSET);
        /** @type {?} */
        const element = (/** @type {?} */ ((/** @type {?} */ (getNativeByIndex(index, lView)))));
        ngDevMode && assertDefined(element, 'native element should exist');
        ngDevMode && ngDevMode.rendererSetText++;
        /** @type {?} */
        const renderer = lView[RENDERER];
        isProceduralRenderer(renderer) ? renderer.setValue(element, value) : element.textContent = value;
    }
    /**
     * @param {?} tNode
     * @param {?} embeddedView
     * @return {?}
     */
    function getLContainer(tNode, embeddedView) {
        ngDevMode && assertLView(embeddedView);
        /** @type {?} */
        const container = (/** @type {?} */ (embeddedView[PARENT]));
        if (tNode.index === -1) {
            // This is a dynamically created view inside a dynamic container.
            // The parent isn't an LContainer if the embedded view hasn't been attached yet.
            return isLContainer(container) ? container : null;
        }
        else {
            ngDevMode && assertLContainer(container);
            // This is a inline view node (e.g. embeddedViewStart)
            return container;
        }
    }
    /**
     * Retrieves render parent for a given view.
     * Might be null if a view is not yet attached to any container.
     * @param {?} tViewNode
     * @param {?} view
     * @return {?}
     */
    function getContainerRenderParent(tViewNode, view) {
        /** @type {?} */
        const container = getLContainer(tViewNode, view);
        return container ? nativeParentNode(view[RENDERER], container[NATIVE]) : null;
    }
    /**
     * NOTE: for performance reasons, the possible actions are inlined within the function instead of
     * being passed as an argument.
     * @param {?} action
     * @param {?} renderer
     * @param {?} parent
     * @param {?} lNodeToHandle
     * @param {?=} beforeNode
     * @return {?}
     */
    function applyToElementOrContainer(action, renderer, parent, lNodeToHandle, beforeNode) {
        // If this slot was allocated for a text node dynamically created by i18n, the text node itself
        // won't be created until i18nApply() in the update block, so this node should be skipped.
        // For more info, see "ICU expressions should work inside an ngTemplateOutlet inside an ngFor"
        // in `i18n_spec.ts`.
        if (lNodeToHandle != null) {
            /** @type {?} */
            let lContainer;
            /** @type {?} */
            let isComponent = false;
            // We are expecting an RNode, but in the case of a component or LContainer the `RNode` is
            // wrapped in an array which needs to be unwrapped. We need to know if it is a component and if
            // it has LContainer so that we can process all of those cases appropriately.
            if (isLContainer(lNodeToHandle)) {
                lContainer = lNodeToHandle;
            }
            else if (isLView(lNodeToHandle)) {
                isComponent = true;
                ngDevMode && assertDefined(lNodeToHandle[HOST], 'HOST must be defined for a component LView');
                lNodeToHandle = (/** @type {?} */ (lNodeToHandle[HOST]));
            }
            /** @type {?} */
            const rNode = unwrapRNode(lNodeToHandle);
            ngDevMode && !isProceduralRenderer(renderer) && assertDomNode(rNode);
            if (action === 0 /* Create */ && parent !== null) {
                if (beforeNode == null) {
                    nativeAppendChild(renderer, parent, rNode);
                }
                else {
                    nativeInsertBefore(renderer, parent, rNode, beforeNode || null);
                }
            }
            else if (action === 1 /* Insert */ && parent !== null) {
                nativeInsertBefore(renderer, parent, rNode, beforeNode || null);
            }
            else if (action === 2 /* Detach */) {
                nativeRemoveNode(renderer, rNode, isComponent);
            }
            else if (action === 3 /* Destroy */) {
                ngDevMode && ngDevMode.rendererDestroyNode++;
                (/** @type {?} */ (((/** @type {?} */ (renderer))).destroyNode))(rNode);
            }
            if (lContainer != null) {
                applyContainer(renderer, action, lContainer, parent, beforeNode);
            }
        }
    }
    /**
     * @param {?} value
     * @param {?} renderer
     * @return {?}
     */
    function createTextNode(value, renderer) {
        ngDevMode && ngDevMode.rendererCreateTextNode++;
        ngDevMode && ngDevMode.rendererSetText++;
        return isProceduralRenderer(renderer) ? renderer.createText(value) :
            renderer.createTextNode(value);
    }
    /**
     * @param {?} tView
     * @param {?} lView
     * @param {?} insertMode
     * @param {?} beforeNode
     * @return {?}
     */
    function addRemoveViewFromContainer(tView, lView, insertMode, beforeNode) {
        /** @type {?} */
        const renderParent = getContainerRenderParent((/** @type {?} */ (tView.node)), lView);
        ngDevMode && assertNodeType((/** @type {?} */ (tView.node)), 2 /* View */);
        if (renderParent) {
            /** @type {?} */
            const renderer = lView[RENDERER];
            /** @type {?} */
            const action = insertMode ? 1 /* Insert */ : 2 /* Detach */;
            applyView(tView, lView, renderer, action, renderParent, beforeNode);
        }
    }
    /**
     * Detach a `LView` from the DOM by detaching its nodes.
     *
     * @param {?} tView The `TView' of the `LView` to be detached
     * @param {?} lView the `LView` to be detached.
     * @return {?}
     */
    function renderDetachView(tView, lView) {
        applyView(tView, lView, lView[RENDERER], 2 /* Detach */, null, null);
    }
    /**
     * Traverses down and up the tree of views and containers to remove listeners and
     * call onDestroy callbacks.
     *
     * Notes:
     *  - Because it's used for onDestroy calls, it needs to be bottom-up.
     *  - Must process containers instead of their views to avoid splicing
     *  when views are destroyed and re-added.
     *  - Using a while loop because it's faster than recursion
     *  - Destroy only called on movement to sibling or movement to parent (laterally or up)
     *
     * @param {?} rootView The view to destroy
     * @return {?}
     */
    function destroyViewTree(rootView) {
        // If the view has no children, we can clean it up and return early.
        /** @type {?} */
        let lViewOrLContainer = rootView[CHILD_HEAD];
        if (!lViewOrLContainer) {
            return cleanUpView(rootView[TVIEW], rootView);
        }
        while (lViewOrLContainer) {
            /** @type {?} */
            let next = null;
            if (isLView(lViewOrLContainer)) {
                // If LView, traverse down to child.
                next = lViewOrLContainer[CHILD_HEAD];
            }
            else {
                ngDevMode && assertLContainer(lViewOrLContainer);
                // If container, traverse down to its first LView.
                /** @type {?} */
                const firstView = lViewOrLContainer[CONTAINER_HEADER_OFFSET];
                if (firstView)
                    next = firstView;
            }
            if (!next) {
                // Only clean up view when moving to the side or up, as destroy hooks
                // should be called in order from the bottom up.
                while (lViewOrLContainer && !(/** @type {?} */ (lViewOrLContainer))[NEXT] && lViewOrLContainer !== rootView) {
                    isLView(lViewOrLContainer) && cleanUpView(lViewOrLContainer[TVIEW], lViewOrLContainer);
                    lViewOrLContainer = getParentState(lViewOrLContainer, rootView);
                }
                if (lViewOrLContainer === null)
                    lViewOrLContainer = rootView;
                isLView(lViewOrLContainer) && cleanUpView(lViewOrLContainer[TVIEW], lViewOrLContainer);
                next = lViewOrLContainer && (/** @type {?} */ (lViewOrLContainer))[NEXT];
            }
            lViewOrLContainer = next;
        }
    }
    /**
     * Inserts a view into a container.
     *
     * This adds the view to the container's array of active views in the correct
     * position. It also adds the view's elements to the DOM if the container isn't a
     * root node of another view (in that case, the view's elements will be added when
     * the container's parent view is added later).
     *
     * @param {?} tView The `TView' of the `LView` to insert
     * @param {?} lView The view to insert
     * @param {?} lContainer The container into which the view should be inserted
     * @param {?} index Which index in the container to insert the child view into
     * @return {?}
     */
    function insertView(tView, lView, lContainer, index) {
        ngDevMode && assertLView(lView);
        ngDevMode && assertLContainer(lContainer);
        /** @type {?} */
        const indexInContainer = CONTAINER_HEADER_OFFSET + index;
        /** @type {?} */
        const containerLength = lContainer.length;
        if (index > 0) {
            // This is a new view, we need to add it to the children.
            lContainer[indexInContainer - 1][NEXT] = lView;
        }
        if (index < containerLength - CONTAINER_HEADER_OFFSET) {
            lView[NEXT] = lContainer[indexInContainer];
            addToArray(lContainer, CONTAINER_HEADER_OFFSET + index, lView);
        }
        else {
            lContainer.push(lView);
            lView[NEXT] = null;
        }
        lView[PARENT] = lContainer;
        // track views where declaration and insertion points are different
        /** @type {?} */
        const declarationLContainer = lView[DECLARATION_LCONTAINER];
        if (declarationLContainer !== null && lContainer !== declarationLContainer) {
            trackMovedView(declarationLContainer, lView);
        }
        // notify query that a new view has been added
        /** @type {?} */
        const lQueries = lView[QUERIES];
        if (lQueries !== null) {
            lQueries.insertView(tView);
        }
        // Sets the attached flag
        lView[FLAGS] |= 128 /* Attached */;
    }
    /**
     * Track views created from the declaration container (TemplateRef) and inserted into a
     * different LContainer.
     * @param {?} declarationContainer
     * @param {?} lView
     * @return {?}
     */
    function trackMovedView(declarationContainer, lView) {
        ngDevMode && assertDefined(lView, 'LView required');
        ngDevMode && assertLContainer(declarationContainer);
        /** @type {?} */
        const movedViews = declarationContainer[MOVED_VIEWS];
        /** @type {?} */
        const insertedLContainer = (/** @type {?} */ (lView[PARENT]));
        ngDevMode && assertLContainer(insertedLContainer);
        /** @type {?} */
        const insertedComponentLView = (/** @type {?} */ (insertedLContainer[PARENT]))[DECLARATION_COMPONENT_VIEW];
        ngDevMode && assertDefined(insertedComponentLView, 'Missing insertedComponentLView');
        /** @type {?} */
        const declaredComponentLView = lView[DECLARATION_COMPONENT_VIEW];
        ngDevMode && assertDefined(declaredComponentLView, 'Missing declaredComponentLView');
        if (declaredComponentLView !== insertedComponentLView) {
            // At this point the declaration-component is not same as insertion-component; this means that
            // this is a transplanted view. Mark the declared lView as having transplanted views so that
            // those views can participate in CD.
            declarationContainer[HAS_TRANSPLANTED_VIEWS] = true;
        }
        if (movedViews === null) {
            declarationContainer[MOVED_VIEWS] = [lView];
        }
        else {
            movedViews.push(lView);
        }
    }
    /**
     * @param {?} declarationContainer
     * @param {?} lView
     * @return {?}
     */
    function detachMovedView(declarationContainer, lView) {
        ngDevMode && assertLContainer(declarationContainer);
        ngDevMode &&
            assertDefined(declarationContainer[MOVED_VIEWS], 'A projected view should belong to a non-empty projected views collection');
        /** @type {?} */
        const movedViews = (/** @type {?} */ (declarationContainer[MOVED_VIEWS]));
        /** @type {?} */
        const declarationViewIndex = movedViews.indexOf(lView);
        /** @type {?} */
        const insertionLContainer = (/** @type {?} */ (lView[PARENT]));
        ngDevMode && assertLContainer(insertionLContainer);
        // If the view was marked for refresh but then detached before it was checked (where the flag
        // would be cleared and the counter decremented), we need to decrement the view counter here
        // instead.
        if (lView[FLAGS] & 1024 /* RefreshTransplantedView */) {
            updateTransplantedViewCount(insertionLContainer, -1);
        }
        movedViews.splice(declarationViewIndex, 1);
    }
    /**
     * Detaches a view from a container.
     *
     * This method removes the view from the container's array of active views. It also
     * removes the view's elements from the DOM.
     *
     * @param {?} lContainer The container from which to detach a view
     * @param {?} removeIndex The index of the view to detach
     * @return {?} Detached LView instance.
     */
    function detachView(lContainer, removeIndex) {
        if (lContainer.length <= CONTAINER_HEADER_OFFSET)
            return;
        /** @type {?} */
        const indexInContainer = CONTAINER_HEADER_OFFSET + removeIndex;
        /** @type {?} */
        const viewToDetach = lContainer[indexInContainer];
        if (viewToDetach) {
            /** @type {?} */
            const declarationLContainer = viewToDetach[DECLARATION_LCONTAINER];
            if (declarationLContainer !== null && declarationLContainer !== lContainer) {
                detachMovedView(declarationLContainer, viewToDetach);
            }
            if (removeIndex > 0) {
                lContainer[indexInContainer - 1][NEXT] = (/** @type {?} */ (viewToDetach[NEXT]));
            }
            /** @type {?} */
            const removedLView = removeFromArray(lContainer, CONTAINER_HEADER_OFFSET + removeIndex);
            addRemoveViewFromContainer(viewToDetach[TVIEW], viewToDetach, false, null);
            // notify query that a view has been removed
            /** @type {?} */
            const lQueries = removedLView[QUERIES];
            if (lQueries !== null) {
                lQueries.detachView(removedLView[TVIEW]);
            }
            viewToDetach[PARENT] = null;
            viewToDetach[NEXT] = null;
            // Unsets the attached flag
            viewToDetach[FLAGS] &= ~128 /* Attached */;
        }
        return viewToDetach;
    }
    /**
     * Removes a view from a container, i.e. detaches it and then destroys the underlying LView.
     *
     * @param {?} lContainer The container from which to remove a view
     * @param {?} removeIndex The index of the view to remove
     * @return {?}
     */
    function removeView(lContainer, removeIndex) {
        /** @type {?} */
        const detachedView = detachView(lContainer, removeIndex);
        detachedView && destroyLView(detachedView[TVIEW], detachedView);
    }
    /**
     * A standalone function which destroys an LView,
     * conducting clean up (e.g. removing listeners, calling onDestroys).
     *
     * @param {?} tView The `TView' of the `LView` to be destroyed
     * @param {?} lView The view to be destroyed.
     * @return {?}
     */
    function destroyLView(tView, lView) {
        if (!(lView[FLAGS] & 256 /* Destroyed */)) {
            /** @type {?} */
            const renderer = lView[RENDERER];
            if (isProceduralRenderer(renderer) && renderer.destroyNode) {
                applyView(tView, lView, renderer, 3 /* Destroy */, null, null);
            }
            destroyViewTree(lView);
        }
    }
    /**
     * Determines which LViewOrLContainer to jump to when traversing back up the
     * tree in destroyViewTree.
     *
     * Normally, the view's parent LView should be checked, but in the case of
     * embedded views, the container (which is the view node's parent, but not the
     * LView's parent) needs to be checked for a possible next property.
     *
     * @param {?} lViewOrLContainer The LViewOrLContainer for which we need a parent state
     * @param {?} rootView The rootView, so we don't propagate too far up the view tree
     * @return {?} The correct parent LViewOrLContainer
     */
    function getParentState(lViewOrLContainer, rootView) {
        /** @type {?} */
        let tNode;
        if (isLView(lViewOrLContainer) && (tNode = lViewOrLContainer[T_HOST]) &&
            tNode.type === 2 /* View */) {
            // if it's an embedded view, the state needs to go up to the container, in case the
            // container has a next
            return getLContainer((/** @type {?} */ (tNode)), lViewOrLContainer);
        }
        else {
            // otherwise, use parent view for containers or component views
            return lViewOrLContainer[PARENT] === rootView ? null : lViewOrLContainer[PARENT];
        }
    }
    /**
     * Calls onDestroys hooks for all directives and pipes in a given view and then removes all
     * listeners. Listeners are removed as the last step so events delivered in the onDestroys hooks
     * can be propagated to \@Output listeners.
     *
     * @param {?} tView `TView` for the `LView` to clean up.
     * @param {?} lView The LView to clean up
     * @return {?}
     */
    function cleanUpView(tView, lView) {
        if (!(lView[FLAGS] & 256 /* Destroyed */)) {
            // Usually the Attached flag is removed when the view is detached from its parent, however
            // if it's a root view, the flag won't be unset hence why we're also removing on destroy.
            lView[FLAGS] &= ~128 /* Attached */;
            // Mark the LView as destroyed *before* executing the onDestroy hooks. An onDestroy hook
            // runs arbitrary user code, which could include its own `viewRef.destroy()` (or similar). If
            // We don't flag the view as destroyed before the hooks, this could lead to an infinite loop.
            // This also aligns with the ViewEngine behavior. It also means that the onDestroy hook is
            // really more of an "afterDestroy" hook if you think about it.
            lView[FLAGS] |= 256 /* Destroyed */;
            executeOnDestroys(tView, lView);
            removeListeners(tView, lView);
            /** @type {?} */
            const hostTNode = lView[T_HOST];
            // For component views only, the local renderer is destroyed as clean up time.
            if (hostTNode && hostTNode.type === 3 /* Element */ &&
                isProceduralRenderer(lView[RENDERER])) {
                ngDevMode && ngDevMode.rendererDestroy++;
                ((/** @type {?} */ (lView[RENDERER]))).destroy();
            }
            /** @type {?} */
            const declarationContainer = lView[DECLARATION_LCONTAINER];
            // we are dealing with an embedded view that is still inserted into a container
            if (declarationContainer !== null && isLContainer(lView[PARENT])) {
                // and this is a projected view
                if (declarationContainer !== lView[PARENT]) {
                    detachMovedView(declarationContainer, lView);
                }
                // For embedded views still attached to a container: remove query result from this view.
                /** @type {?} */
                const lQueries = lView[QUERIES];
                if (lQueries !== null) {
                    lQueries.detachView(tView);
                }
            }
        }
    }
    /**
     * Removes listeners and unsubscribes from output subscriptions
     * @param {?} tView
     * @param {?} lView
     * @return {?}
     */
    function removeListeners(tView, lView) {
        /** @type {?} */
        const tCleanup = tView.cleanup;
        if (tCleanup !== null) {
            /** @type {?} */
            const lCleanup = (/** @type {?} */ (lView[CLEANUP]));
            for (let i = 0; i < tCleanup.length - 1; i += 2) {
                if (typeof tCleanup[i] === 'string') {
                    // This is a native DOM listener
                    /** @type {?} */
                    const idxOrTargetGetter = tCleanup[i + 1];
                    /** @type {?} */
                    const target = typeof idxOrTargetGetter === 'function' ?
                        idxOrTargetGetter(lView) :
                        unwrapRNode(lView[idxOrTargetGetter]);
                    /** @type {?} */
                    const listener = lCleanup[tCleanup[i + 2]];
                    /** @type {?} */
                    const useCaptureOrSubIdx = tCleanup[i + 3];
                    if (typeof useCaptureOrSubIdx === 'boolean') {
                        // native DOM listener registered with Renderer3
                        target.removeEventListener(tCleanup[i], listener, useCaptureOrSubIdx);
                    }
                    else {
                        if (useCaptureOrSubIdx >= 0) {
                            // unregister
                            lCleanup[useCaptureOrSubIdx]();
                        }
                        else {
                            // Subscription
                            lCleanup[-useCaptureOrSubIdx].unsubscribe();
                        }
                    }
                    i += 2;
                }
                else {
                    // This is a cleanup function that is grouped with the index of its context
                    /** @type {?} */
                    const context = lCleanup[tCleanup[i + 1]];
                    tCleanup[i].call(context);
                }
            }
            lView[CLEANUP] = null;
        }
    }
    /**
     * Calls onDestroy hooks for this view
     * @param {?} tView
     * @param {?} lView
     * @return {?}
     */
    function executeOnDestroys(tView, lView) {
        /** @type {?} */
        let destroyHooks;
        if (tView != null && (destroyHooks = tView.destroyHooks) != null) {
            for (let i = 0; i < destroyHooks.length; i += 2) {
                /** @type {?} */
                const context = lView[(/** @type {?} */ (destroyHooks[i]))];
                // Only call the destroy hook if the context has been requested.
                if (!(context instanceof NodeInjectorFactory)) {
                    /** @type {?} */
                    const toCall = (/** @type {?} */ (destroyHooks[i + 1]));
                    if (Array.isArray(toCall)) {
                        for (let j = 0; j < toCall.length; j += 2) {
                            ((/** @type {?} */ (toCall[j + 1]))).call(context[(/** @type {?} */ (toCall[j]))]);
                        }
                    }
                    else {
                        toCall.call(context);
                    }
                }
            }
        }
    }
    /**
     * Returns a native element if a node can be inserted into the given parent.
     *
     * There are two reasons why we may not be able to insert a element immediately.
     * - Projection: When creating a child content element of a component, we have to skip the
     *   insertion because the content of a component will be projected.
     *   `<component><content>delayed due to projection</content></component>`
     * - Parent container is disconnected: This can happen when we are inserting a view into
     *   parent container, which itself is disconnected. For example the parent container is part
     *   of a View which has not be inserted or is made for projection but has not been inserted
     *   into destination.
     * @param {?} tView
     * @param {?} tNode
     * @param {?} currentView
     * @return {?}
     */
    function getRenderParent(tView, tNode, currentView) {
        // Skip over element and ICU containers as those are represented by a comment node and
        // can't be used as a render parent.
        /** @type {?} */
        let parentTNode = tNode.parent;
        while (parentTNode != null &&
            (parentTNode.type === 4 /* ElementContainer */ ||
                parentTNode.type === 5 /* IcuContainer */)) {
            tNode = parentTNode;
            parentTNode = tNode.parent;
        }
        // If the parent tNode is null, then we are inserting across views: either into an embedded view
        // or a component view.
        if (parentTNode == null) {
            /** @type {?} */
            const hostTNode = (/** @type {?} */ (currentView[T_HOST]));
            if (hostTNode.type === 2 /* View */) {
                // We are inserting a root element of an embedded view We might delay insertion of children
                // for a given view if it is disconnected. This might happen for 2 main reasons:
                // - view is not inserted into any container(view was created but not inserted yet)
                // - view is inserted into a container but the container itself is not inserted into the DOM
                // (container might be part of projection or child of a view that is not inserted yet).
                // In other words we can insert children of a given view if this view was inserted into a
                // container and the container itself has its render parent determined.
                return getContainerRenderParent((/** @type {?} */ (hostTNode)), currentView);
            }
            else {
                // We are inserting a root element of the component view into the component host element and
                // it should always be eager.
                ngDevMode && assertNodeOfPossibleTypes(hostTNode, 3 /* Element */);
                return currentView[HOST];
            }
        }
        else {
            /** @type {?} */
            const isIcuCase = tNode && tNode.type === 5 /* IcuContainer */;
            // If the parent of this node is an ICU container, then it is represented by comment node and we
            // need to use it as an anchor. If it is projected then it's direct parent node is the renderer.
            if (isIcuCase && tNode.flags & 4 /* isProjected */) {
                return (/** @type {?} */ (getNativeByTNode(tNode, currentView).parentNode));
            }
            ngDevMode && assertNodeType(parentTNode, 3 /* Element */);
            if (parentTNode.flags & 2 /* isComponentHost */) {
                /** @type {?} */
                const tData = tView.data;
                /** @type {?} */
                const tNode = (/** @type {?} */ (tData[parentTNode.index]));
                /** @type {?} */
                const encapsulation = ((/** @type {?} */ (tData[tNode.directiveStart]))).encapsulation;
                // We've got a parent which is an element in the current view. We just need to verify if the
                // parent element is not a component. Component's content nodes are not inserted immediately
                // because they will be projected, and so doing insert at this point would be wasteful.
                // Since the projection would then move it to its final destination. Note that we can't
                // make this assumption when using the Shadow DOM, because the native projection placeholders
                // (<content> or <slot>) have to be in place as elements are being inserted.
                if (encapsulation !== ViewEncapsulation$1.ShadowDom &&
                    encapsulation !== ViewEncapsulation$1.Native) {
                    return null;
                }
            }
            return (/** @type {?} */ (getNativeByTNode(parentTNode, currentView)));
        }
    }
    /**
     * Inserts a native node before another native node for a given parent using {\@link Renderer3}.
     * This is a utility function that can be used when native nodes were determined - it abstracts an
     * actual renderer being used.
     * @param {?} renderer
     * @param {?} parent
     * @param {?} child
     * @param {?} beforeNode
     * @return {?}
     */
    function nativeInsertBefore(renderer, parent, child, beforeNode) {
        ngDevMode && ngDevMode.rendererInsertBefore++;
        if (isProceduralRenderer(renderer)) {
            renderer.insertBefore(parent, child, beforeNode);
        }
        else {
            parent.insertBefore(child, beforeNode, true);
        }
    }
    /**
     * @param {?} renderer
     * @param {?} parent
     * @param {?} child
     * @return {?}
     */
    function nativeAppendChild(renderer, parent, child) {
        ngDevMode && ngDevMode.rendererAppendChild++;
        ngDevMode && assertDefined(parent, 'parent node must be defined');
        if (isProceduralRenderer(renderer)) {
            renderer.appendChild(parent, child);
        }
        else {
            parent.appendChild(child);
        }
    }
    /**
     * @param {?} renderer
     * @param {?} parent
     * @param {?} child
     * @param {?} beforeNode
     * @return {?}
     */
    function nativeAppendOrInsertBefore(renderer, parent, child, beforeNode) {
        if (beforeNode !== null) {
            nativeInsertBefore(renderer, parent, child, beforeNode);
        }
        else {
            nativeAppendChild(renderer, parent, child);
        }
    }
    /**
     * Removes a node from the DOM given its native parent.
     * @param {?} renderer
     * @param {?} parent
     * @param {?} child
     * @param {?=} isHostElement
     * @return {?}
     */
    function nativeRemoveChild(renderer, parent, child, isHostElement) {
        if (isProceduralRenderer(renderer)) {
            renderer.removeChild(parent, child, isHostElement);
        }
        else {
            parent.removeChild(child);
        }
    }
    /**
     * Returns a native parent of a given native node.
     * @param {?} renderer
     * @param {?} node
     * @return {?}
     */
    function nativeParentNode(renderer, node) {
        return (/** @type {?} */ ((isProceduralRenderer(renderer) ? renderer.parentNode(node) : node.parentNode)));
    }
    /**
     * Returns a native sibling of a given native node.
     * @param {?} renderer
     * @param {?} node
     * @return {?}
     */
    function nativeNextSibling(renderer, node) {
        return isProceduralRenderer(renderer) ? renderer.nextSibling(node) : node.nextSibling;
    }
    /**
     * Finds a native "anchor" node for cases where we can't append a native child directly
     * (`appendChild`) and need to use a reference (anchor) node for the `insertBefore` operation.
     * @param {?} parentTNode
     * @param {?} lView
     * @return {?}
     */
    function getNativeAnchorNode(parentTNode, lView) {
        if (parentTNode.type === 2 /* View */) {
            /** @type {?} */
            const lContainer = getLContainer((/** @type {?} */ (parentTNode)), lView);
            if (lContainer === null)
                return null;
            /** @type {?} */
            const index = lContainer.indexOf(lView, CONTAINER_HEADER_OFFSET) - CONTAINER_HEADER_OFFSET;
            return getBeforeNodeForView(index, lContainer);
        }
        else if (parentTNode.type === 4 /* ElementContainer */ ||
            parentTNode.type === 5 /* IcuContainer */) {
            return getNativeByTNode(parentTNode, lView);
        }
        return null;
    }
    /**
     * Appends the `child` native node (or a collection of nodes) to the `parent`.
     *
     * The element insertion might be delayed {\@link canInsertNativeNode}.
     *
     * @param {?} tView The `TView' to be appended
     * @param {?} lView The current LView
     * @param {?} childEl The native child (or children) that should be appended
     * @param {?} childTNode The TNode of the child element
     * @return {?} Whether or not the child was appended
     */
    function appendChild(tView, lView, childEl, childTNode) {
        /** @type {?} */
        const renderParent = getRenderParent(tView, childTNode, lView);
        if (renderParent != null) {
            /** @type {?} */
            const renderer = lView[RENDERER];
            /** @type {?} */
            const parentTNode = childTNode.parent || (/** @type {?} */ (lView[T_HOST]));
            /** @type {?} */
            const anchorNode = getNativeAnchorNode(parentTNode, lView);
            if (Array.isArray(childEl)) {
                for (let i = 0; i < childEl.length; i++) {
                    nativeAppendOrInsertBefore(renderer, renderParent, childEl[i], anchorNode);
                }
            }
            else {
                nativeAppendOrInsertBefore(renderer, renderParent, childEl, anchorNode);
            }
        }
    }
    /**
     * Returns the first native node for a given LView, starting from the provided TNode.
     *
     * Native nodes are returned in the order in which those appear in the native tree (DOM).
     * @param {?} lView
     * @param {?} tNode
     * @return {?}
     */
    function getFirstNativeNode(lView, tNode) {
        if (tNode !== null) {
            ngDevMode &&
                assertNodeOfPossibleTypes(tNode, 3 /* Element */, 0 /* Container */, 4 /* ElementContainer */, 5 /* IcuContainer */, 1 /* Projection */);
            /** @type {?} */
            const tNodeType = tNode.type;
            if (tNodeType === 3 /* Element */) {
                return getNativeByTNode(tNode, lView);
            }
            else if (tNodeType === 0 /* Container */) {
                return getBeforeNodeForView(-1, lView[tNode.index]);
            }
            else if (tNodeType === 4 /* ElementContainer */ || tNodeType === 5 /* IcuContainer */) {
                /** @type {?} */
                const elIcuContainerChild = tNode.child;
                if (elIcuContainerChild !== null) {
                    return getFirstNativeNode(lView, elIcuContainerChild);
                }
                else {
                    /** @type {?} */
                    const rNodeOrLContainer = lView[tNode.index];
                    if (isLContainer(rNodeOrLContainer)) {
                        return getBeforeNodeForView(-1, rNodeOrLContainer);
                    }
                    else {
                        return unwrapRNode(rNodeOrLContainer);
                    }
                }
            }
            else {
                /** @type {?} */
                const componentView = lView[DECLARATION_COMPONENT_VIEW];
                /** @type {?} */
                const componentHost = (/** @type {?} */ (componentView[T_HOST]));
                /** @type {?} */
                const parentView = getLViewParent(componentView);
                /** @type {?} */
                const firstProjectedTNode = ((/** @type {?} */ (componentHost.projection)))[(/** @type {?} */ (tNode.projection))];
                if (firstProjectedTNode != null) {
                    return getFirstNativeNode((/** @type {?} */ (parentView)), firstProjectedTNode);
                }
                else {
                    return getFirstNativeNode(lView, tNode.next);
                }
            }
        }
        return null;
    }
    /**
     * @param {?} viewIndexInContainer
     * @param {?} lContainer
     * @return {?}
     */
    function getBeforeNodeForView(viewIndexInContainer, lContainer) {
        /** @type {?} */
        const nextViewIndex = CONTAINER_HEADER_OFFSET + viewIndexInContainer + 1;
        if (nextViewIndex < lContainer.length) {
            /** @type {?} */
            const lView = (/** @type {?} */ (lContainer[nextViewIndex]));
            /** @type {?} */
            const firstTNodeOfView = lView[TVIEW].firstChild;
            if (firstTNodeOfView !== null) {
                return getFirstNativeNode(lView, firstTNodeOfView);
            }
        }
        return lContainer[NATIVE];
    }
    /**
     * Removes a native node itself using a given renderer. To remove the node we are looking up its
     * parent from the native tree as not all platforms / browsers support the equivalent of
     * node.remove().
     *
     * @param {?} renderer A renderer to be used
     * @param {?} rNode The native node that should be removed
     * @param {?=} isHostElement A flag indicating if a node to be removed is a host of a component.
     * @return {?}
     */
    function nativeRemoveNode(renderer, rNode, isHostElement) {
        /** @type {?} */
        const nativeParent = nativeParentNode(renderer, rNode);
        if (nativeParent) {
            nativeRemoveChild(renderer, nativeParent, rNode, isHostElement);
        }
    }
    /**
     * Performs the operation of `action` on the node. Typically this involves inserting or removing
     * nodes on the LView or projection boundary.
     * @param {?} renderer
     * @param {?} action
     * @param {?} tNode
     * @param {?} lView
     * @param {?} renderParent
     * @param {?} beforeNode
     * @param {?} isProjection
     * @return {?}
     */
    function applyNodes(renderer, action, tNode, lView, renderParent, beforeNode, isProjection) {
        while (tNode != null) {
            ngDevMode && assertTNodeForLView(tNode, lView);
            ngDevMode &&
                assertNodeOfPossibleTypes(tNode, 0 /* Container */, 3 /* Element */, 4 /* ElementContainer */, 1 /* Projection */, 1 /* Projection */, 5 /* IcuContainer */);
            /** @type {?} */
            const rawSlotValue = lView[tNode.index];
            /** @type {?} */
            const tNodeType = tNode.type;
            if (isProjection) {
                if (action === 0 /* Create */) {
                    rawSlotValue && attachPatchData(unwrapRNode(rawSlotValue), lView);
                    tNode.flags |= 4 /* isProjected */;
                }
            }
            if ((tNode.flags & 64 /* isDetached */) !== 64 /* isDetached */) {
                if (tNodeType === 4 /* ElementContainer */ || tNodeType === 5 /* IcuContainer */) {
                    applyNodes(renderer, action, tNode.child, lView, renderParent, beforeNode, false);
                    applyToElementOrContainer(action, renderer, renderParent, rawSlotValue, beforeNode);
                }
                else if (tNodeType === 1 /* Projection */) {
                    applyProjectionRecursive(renderer, action, lView, (/** @type {?} */ (tNode)), renderParent, beforeNode);
                }
                else {
                    ngDevMode && assertNodeOfPossibleTypes(tNode, 3 /* Element */, 0 /* Container */);
                    applyToElementOrContainer(action, renderer, renderParent, rawSlotValue, beforeNode);
                }
            }
            tNode = isProjection ? tNode.projectionNext : tNode.next;
        }
    }
    /**
     * `applyView` performs operation on the view as specified in `action` (insert, detach, destroy)
     *
     * Inserting a view without projection or containers at top level is simple. Just iterate over the
     * root nodes of the View, and for each node perform the `action`.
     *
     * Things get more complicated with containers and projections. That is because coming across:
     * - Container: implies that we have to insert/remove/destroy the views of that container as well
     *              which in turn can have their own Containers at the View roots.
     * - Projection: implies that we have to insert/remove/destroy the nodes of the projection. The
     *               complication is that the nodes we are projecting can themselves have Containers
     *               or other Projections.
     *
     * As you can see this is a very recursive problem. Yes recursion is not most efficient but the
     * code is complicated enough that trying to implemented with recursion becomes unmaintainable.
     *
     * @param {?} tView The `TView' which needs to be inserted, detached, destroyed
     * @param {?} lView The LView which needs to be inserted, detached, destroyed.
     * @param {?} renderer Renderer to use
     * @param {?} action action to perform (insert, detach, destroy)
     * @param {?} renderParent parent DOM element for insertion/removal.
     * @param {?} beforeNode Before which node the insertions should happen.
     * @return {?}
     */
    function applyView(tView, lView, renderer, action, renderParent, beforeNode) {
        ngDevMode && assertNodeType((/** @type {?} */ (tView.node)), 2 /* View */);
        /** @type {?} */
        const viewRootTNode = (/** @type {?} */ (tView.node)).child;
        applyNodes(renderer, action, viewRootTNode, lView, renderParent, beforeNode, false);
    }
    /**
     * `applyProjection` performs operation on the projection.
     *
     * Inserting a projection requires us to locate the projected nodes from the parent component. The
     * complication is that those nodes themselves could be re-projected from their parent component.
     *
     * @param {?} tView The `TView` of `LView` which needs to be inserted, detached, destroyed
     * @param {?} lView The `LView` which needs to be inserted, detached, destroyed.
     * @param {?} tProjectionNode node to project
     * @return {?}
     */
    function applyProjection(tView, lView, tProjectionNode) {
        /** @type {?} */
        const renderer = lView[RENDERER];
        /** @type {?} */
        const renderParent = getRenderParent(tView, tProjectionNode, lView);
        /** @type {?} */
        const parentTNode = tProjectionNode.parent || (/** @type {?} */ (lView[T_HOST]));
        /** @type {?} */
        let beforeNode = getNativeAnchorNode(parentTNode, lView);
        applyProjectionRecursive(renderer, 0 /* Create */, lView, tProjectionNode, renderParent, beforeNode);
    }
    /**
     * `applyProjectionRecursive` performs operation on the projection specified by `action` (insert,
     * detach, destroy)
     *
     * Inserting a projection requires us to locate the projected nodes from the parent component. The
     * complication is that those nodes themselves could be re-projected from their parent component.
     *
     * @param {?} renderer Render to use
     * @param {?} action action to perform (insert, detach, destroy)
     * @param {?} lView The LView which needs to be inserted, detached, destroyed.
     * @param {?} tProjectionNode node to project
     * @param {?} renderParent parent DOM element for insertion/removal.
     * @param {?} beforeNode Before which node the insertions should happen.
     * @return {?}
     */
    function applyProjectionRecursive(renderer, action, lView, tProjectionNode, renderParent, beforeNode) {
        /** @type {?} */
        const componentLView = lView[DECLARATION_COMPONENT_VIEW];
        /** @type {?} */
        const componentNode = (/** @type {?} */ (componentLView[T_HOST]));
        ngDevMode &&
            assertEqual(typeof tProjectionNode.projection, 'number', 'expecting projection index');
        /** @type {?} */
        const nodeToProjectOrRNodes = (/** @type {?} */ ((/** @type {?} */ (componentNode.projection))[tProjectionNode.projection]));
        if (Array.isArray(nodeToProjectOrRNodes)) {
            // This should not exist, it is a bit of a hack. When we bootstrap a top level node and we
            // need to support passing projectable nodes, so we cheat and put them in the TNode
            // of the Host TView. (Yes we put instance info at the T Level). We can get away with it
            // because we know that that TView is not shared and therefore it will not be a problem.
            // This should be refactored and cleaned up.
            for (let i = 0; i < nodeToProjectOrRNodes.length; i++) {
                /** @type {?} */
                const rNode = nodeToProjectOrRNodes[i];
                applyToElementOrContainer(action, renderer, renderParent, rNode, beforeNode);
            }
        }
        else {
            /** @type {?} */
            let nodeToProject = nodeToProjectOrRNodes;
            /** @type {?} */
            const projectedComponentLView = (/** @type {?} */ (componentLView[PARENT]));
            applyNodes(renderer, action, nodeToProject, projectedComponentLView, renderParent, beforeNode, true);
        }
    }
    /**
     * `applyContainer` performs an operation on the container and its views as specified by
     * `action` (insert, detach, destroy)
     *
     * Inserting a Container is complicated by the fact that the container may have Views which
     * themselves have containers or projections.
     *
     * @param {?} renderer Renderer to use
     * @param {?} action action to perform (insert, detach, destroy)
     * @param {?} lContainer The LContainer which needs to be inserted, detached, destroyed.
     * @param {?} renderParent parent DOM element for insertion/removal.
     * @param {?} beforeNode Before which node the insertions should happen.
     * @return {?}
     */
    function applyContainer(renderer, action, lContainer, renderParent, beforeNode) {
        ngDevMode && assertLContainer(lContainer);
        /** @type {?} */
        const anchor = lContainer[NATIVE];
        // LContainer has its own before node.
        /** @type {?} */
        const native = unwrapRNode(lContainer);
        // An LContainer can be created dynamically on any node by injecting ViewContainerRef.
        // Asking for a ViewContainerRef on an element will result in a creation of a separate anchor node
        // (comment in the DOM) that will be different from the LContainer's host node. In this particular
        // case we need to execute action on 2 nodes:
        // - container's host node (this is done in the executeActionOnElementOrContainer)
        // - container's host node (this is done here)
        if (anchor !== native) {
            // This is very strange to me (Misko). I would expect that the native is same as anchor. I don't
            // see a reason why they should be different, but they are.
            //
            // If they are we need to process the second anchor as well.
            applyToElementOrContainer(action, renderer, renderParent, anchor, beforeNode);
        }
        for (let i = CONTAINER_HEADER_OFFSET; i < lContainer.length; i++) {
            /** @type {?} */
            const lView = (/** @type {?} */ (lContainer[i]));
            applyView(lView[TVIEW], lView, renderer, action, renderParent, anchor);
        }
    }
    /**
     * Writes class/style to element.
     *
     * @param {?} renderer Renderer to use.
     * @param {?} isClassBased `true` if it should be written to `class` (`false` to write to `style`)
     * @param {?} rNode The Node to write to.
     * @param {?} prop Property to write to. This would be the class/style name.
     * @param {?} value Value to write. If `null`/`undefined`/`false` this is considered a remove (set/add
     *        otherwise).
     * @return {?}
     */
    function applyStyling(renderer, isClassBased, rNode, prop, value) {
        /** @type {?} */
        const isProcedural = isProceduralRenderer(renderer);
        if (isClassBased) {
            // We actually want JS true/false here because any truthy value should add the class
            if (!value) {
                ngDevMode && ngDevMode.rendererRemoveClass++;
                if (isProcedural) {
                    ((/** @type {?} */ (renderer))).removeClass(rNode, prop);
                }
                else {
                    ((/** @type {?} */ (rNode))).classList.remove(prop);
                }
            }
            else {
                ngDevMode && ngDevMode.rendererAddClass++;
                if (isProcedural) {
                    ((/** @type {?} */ (renderer))).addClass(rNode, prop);
                }
                else {
                    ngDevMode && assertDefined(((/** @type {?} */ (rNode))).classList, 'HTMLElement expected');
                    ((/** @type {?} */ (rNode))).classList.add(prop);
                }
            }
        }
        else {
            // TODO(misko): Can't import RendererStyleFlags2.DashCase as it causes imports to be resolved in
            // different order which causes failures. Using direct constant as workaround for now.
            /** @type {?} */
            const flags = prop.indexOf('-') == -1 ? undefined : 2 /* RendererStyleFlags2.DashCase */;
            if (value == null /** || value === undefined */) {
                ngDevMode && ngDevMode.rendererRemoveStyle++;
                if (isProcedural) {
                    ((/** @type {?} */ (renderer))).removeStyle(rNode, prop, flags);
                }
                else {
                    ((/** @type {?} */ (rNode))).style.removeProperty(prop);
                }
            }
            else {
                ngDevMode && ngDevMode.rendererSetStyle++;
                if (isProcedural) {
                    ((/** @type {?} */ (renderer))).setStyle(rNode, prop, value, flags);
                }
                else {
                    ngDevMode && assertDefined(((/** @type {?} */ (rNode))).style, 'HTMLElement expected');
                    ((/** @type {?} */ (rNode))).style.setProperty(prop, value);
                }
            }
        }
    }
    /**
     * Write `cssText` to `RElement`.
     *
     * This function does direct write without any reconciliation. Used for writing initial values, so
     * that static styling values do not pull in the style parser.
     *
     * @param {?} renderer Renderer to use
     * @param {?} element The element which needs to be updated.
     * @param {?} newValue The new class list to write.
     * @return {?}
     */
    function writeDirectStyle(renderer, element, newValue) {
        ngDevMode && assertString(newValue, '\'newValue\' should be a string');
        if (isProceduralRenderer(renderer)) {
            renderer.setAttribute(element, 'style', newValue);
        }
        else {
            ((/** @type {?} */ (element))).style.cssText = newValue;
        }
        ngDevMode && ngDevMode.rendererSetStyle++;
    }
    /**
     * Write `className` to `RElement`.
     *
     * This function does direct write without any reconciliation. Used for writing initial values, so
     * that static styling values do not pull in the style parser.
     *
     * @param {?} renderer Renderer to use
     * @param {?} element The element which needs to be updated.
     * @param {?} newValue The new class list to write.
     * @return {?}
     */
    function writeDirectClass(renderer, element, newValue) {
        ngDevMode && assertString(newValue, '\'newValue\' should be a string');
        if (isProceduralRenderer(renderer)) {
            if (newValue === '') {
                // There are tests in `google3` which expect `element.getAttribute('class')` to be `null`.
                renderer.removeAttribute(element, 'class');
            }
            else {
                renderer.setAttribute(element, 'class', newValue);
            }
        }
        else {
            element.className = newValue;
        }
        ngDevMode && ngDevMode.rendererSetClassName++;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/node_util.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * If `startTNode.parent` exists and has an injector, returns TNode for that injector.
     * Otherwise, unwraps a parent injector location number to find the view offset from the current
     * injector, then walks up the declaration view tree until the TNode of the parent injector is
     * found.
     *
     * @param {?} location The location of the parent injector, which contains the view offset
     * @param {?} startView The LView instance from which to start walking up the view tree
     * @param {?} startTNode The TNode instance of the starting element
     * @return {?} The TNode of the parent injector
     */
    function getParentInjectorTNode(location, startView, startTNode) {
        // If there is an injector on the parent TNode, retrieve the TNode for that injector.
        if (startTNode.parent && startTNode.parent.injectorIndex !== -1) {
            // view offset is 0
            /** @type {?} */
            const injectorIndex = startTNode.parent.injectorIndex;
            /** @type {?} */
            let tNode = startTNode.parent;
            // If tNode.injectorIndex === tNode.parent.injectorIndex, then the index belongs to a parent
            // injector.
            while (tNode.parent != null && injectorIndex == tNode.parent.injectorIndex) {
                tNode = tNode.parent;
            }
            return tNode;
        }
        /** @type {?} */
        let viewOffset = getParentInjectorViewOffset(location);
        // view offset is 1
        /** @type {?} */
        let parentView = startView;
        /** @type {?} */
        let parentTNode = (/** @type {?} */ (startView[T_HOST]));
        // view offset is superior to 1
        while (viewOffset > 1) {
            parentView = (/** @type {?} */ (parentView[DECLARATION_VIEW]));
            parentTNode = (/** @type {?} */ (parentView[T_HOST]));
            viewOffset--;
        }
        return parentTNode;
    }
    /**
     * @template T
     */
    class ViewRef {
        /**
         * @param {?} _lView
         * @param {?=} _cdRefInjectingView
         */
        constructor(_lView, _cdRefInjectingView) {
            this._lView = _lView;
            this._cdRefInjectingView = _cdRefInjectingView;
            this._appRef = null;
            this._viewContainerRef = null;
        }
        /**
         * @return {?}
         */
        get rootNodes() {
            /** @type {?} */
            const lView = this._lView;
            if (lView[HOST] == null) {
                /** @type {?} */
                const hostTView = (/** @type {?} */ (lView[T_HOST]));
                return collectNativeNodes(lView[TVIEW], lView, hostTView.child, []);
            }
            return [];
        }
        /**
         * @return {?}
         */
        get context() {
            return (/** @type {?} */ (this._lView[CONTEXT]));
        }
        /**
         * @return {?}
         */
        get destroyed() {
            return (this._lView[FLAGS] & 256 /* Destroyed */) === 256 /* Destroyed */;
        }
        /**
         * @return {?}
         */
        destroy() {
            if (this._appRef) {
                this._appRef.detachView(this);
            }
            else if (this._viewContainerRef) {
                /** @type {?} */
                const index = this._viewContainerRef.indexOf(this);
                if (index > -1) {
                    this._viewContainerRef.detach(index);
                }
                this._viewContainerRef = null;
            }
            destroyLView(this._lView[TVIEW], this._lView);
        }
        /**
         * @param {?} callback
         * @return {?}
         */
        onDestroy(callback) {
            storeCleanupFn(this._lView[TVIEW], this._lView, callback);
        }
        /**
         * Marks a view and all of its ancestors dirty.
         *
         * It also triggers change detection by calling `scheduleTick` internally, which coalesces
         * multiple `markForCheck` calls to into one change detection run.
         *
         * This can be used to ensure an {\@link ChangeDetectionStrategy#OnPush OnPush} component is
         * checked when it needs to be re-rendered but the two normal triggers haven't marked it
         * dirty (i.e. inputs haven't changed and events haven't fired in the view).
         *
         * <!-- TODO: Add a link to a chapter on OnPush components -->
         *
         * \@usageNotes
         * ### Example
         *
         * ```typescript
         * \@Component({
         *   selector: 'my-app',
         *   template: `Number of ticks: {{numberOfTicks}}`
         *   changeDetection: ChangeDetectionStrategy.OnPush,
         * })
         * class AppComponent {
         *   numberOfTicks = 0;
         *
         *   constructor(private ref: ChangeDetectorRef) {
         *     setInterval(() => {
         *       this.numberOfTicks++;
         *       // the following is required, otherwise the view will not be updated
         *       this.ref.markForCheck();
         *     }, 1000);
         *   }
         * }
         * ```
         * @return {?}
         */
        markForCheck() {
            markViewDirty(this._cdRefInjectingView || this._lView);
        }
        /**
         * Detaches the view from the change detection tree.
         *
         * Detached views will not be checked during change detection runs until they are
         * re-attached, even if they are dirty. `detach` can be used in combination with
         * {\@link ChangeDetectorRef#detectChanges detectChanges} to implement local change
         * detection checks.
         *
         * <!-- TODO: Add a link to a chapter on detach/reattach/local digest -->
         * <!-- TODO: Add a live demo once ref.detectChanges is merged into master -->
         *
         * \@usageNotes
         * ### Example
         *
         * The following example defines a component with a large list of readonly data.
         * Imagine the data changes constantly, many times per second. For performance reasons,
         * we want to check and update the list every five seconds. We can do that by detaching
         * the component's change detector and doing a local check every five seconds.
         *
         * ```typescript
         * class DataProvider {
         *   // in a real application the returned data will be different every time
         *   get data() {
         *     return [1,2,3,4,5];
         *   }
         * }
         *
         * \@Component({
         *   selector: 'giant-list',
         *   template: `
         *     <li *ngFor="let d of dataProvider.data">Data {{d}}</li>
         *   `,
         * })
         * class GiantList {
         *   constructor(private ref: ChangeDetectorRef, private dataProvider: DataProvider) {
         *     ref.detach();
         *     setInterval(() => {
         *       this.ref.detectChanges();
         *     }, 5000);
         *   }
         * }
         *  /
         *   selector: 'app',
         *   providers: [DataProvider],
         *   template: `
         *     <giant-list><giant-list>
         *   `,
         * })
         * class App {
         * }
         * ```
         * @return {?}
         */
        detach() {
            this._lView[FLAGS] &= ~128 /* Attached */;
        }
        /**
         * Re-attaches a view to the change detection tree.
         *
         * This can be used to re-attach views that were previously detached from the tree
         * using {\@link ChangeDetectorRef#detach detach}. Views are attached to the tree by default.
         *
         * <!-- TODO: Add a link to a chapter on detach/reattach/local digest -->
         *
         * \@usageNotes
         * ### Example
         *
         * The following example creates a component displaying `live` data. The component will detach
         * its change detector from the main change detector tree when the component's live property
         * is set to false.
         *
         * ```typescript
         * class DataProvider {
         *   data = 1;
         *
         *   constructor() {
         *     setInterval(() => {
         *       this.data = this.data * 2;
         *     }, 500);
         *   }
         * }
         *
         * \@Component({
         *   selector: 'live-data',
         *   inputs: ['live'],
         *   template: 'Data: {{dataProvider.data}}'
         * })
         * class LiveData {
         *   constructor(private ref: ChangeDetectorRef, private dataProvider: DataProvider) {}
         *
         *   set live(value) {
         *     if (value) {
         *       this.ref.reattach();
         *     } else {
         *       this.ref.detach();
         *     }
         *   }
         * }
         *  /
         *   selector: 'my-app',
         *   providers: [DataProvider],
         *   template: `
         *     Live Update: <input type="checkbox" [(ngModel)]="live">
         *     <live-data [live]="live"><live-data>
         *   `,
         * })
         * class AppComponent {
         *   live = true;
         * }
         * ```
         * @return {?}
         */
        reattach() {
            this._lView[FLAGS] |= 128 /* Attached */;
        }
        /**
         * Checks the view and its children.
         *
         * This can also be used in combination with {\@link ChangeDetectorRef#detach detach} to implement
         * local change detection checks.
         *
         * <!-- TODO: Add a link to a chapter on detach/reattach/local digest -->
         * <!-- TODO: Add a live demo once ref.detectChanges is merged into master -->
         *
         * \@usageNotes
         * ### Example
         *
         * The following example defines a component with a large list of readonly data.
         * Imagine, the data changes constantly, many times per second. For performance reasons,
         * we want to check and update the list every five seconds.
         *
         * We can do that by detaching the component's change detector and doing a local change detection
         * check every five seconds.
         *
         * See {\@link ChangeDetectorRef#detach detach} for more information.
         * @return {?}
         */
        detectChanges() {
            detectChangesInternal(this._lView[TVIEW], this._lView, this.context);
        }
        /**
         * Checks the change detector and its children, and throws if any changes are detected.
         *
         * This is used in development mode to verify that running change detection doesn't
         * introduce other changes.
         * @return {?}
         */
        checkNoChanges() {
            checkNoChangesInternal(this._lView[TVIEW], this._lView, this.context);
        }
        /**
         * @param {?} vcRef
         * @return {?}
         */
        attachToViewContainerRef(vcRef) {
            if (this._appRef) {
                throw new Error('This view is already attached directly to the ApplicationRef!');
            }
            this._viewContainerRef = vcRef;
        }
        /**
         * @return {?}
         */
        detachFromAppRef() {
            this._appRef = null;
            renderDetachView(this._lView[TVIEW], this._lView);
        }
        /**
         * @param {?} appRef
         * @return {?}
         */
        attachToAppRef(appRef) {
            if (this._viewContainerRef) {
                throw new Error('This view is already attached to a ViewContainer!');
            }
            this._appRef = appRef;
        }
    }
    /**
     * \@internal
     * @template T
     */
    class RootViewRef extends ViewRef {
        /**
         * @param {?} _view
         */
        constructor(_view) {
            super(_view);
            this._view = _view;
        }
        /**
         * @return {?}
         */
        detectChanges() {
            detectChangesInRootView(this._view);
        }
        /**
         * @return {?}
         */
        checkNoChanges() {
            checkNoChangesInRootView(this._view);
        }
        /**
         * @return {?}
         */
        get context() {
            return (/** @type {?} */ (null));
        }
    }
    /**
     * @param {?} tView
     * @param {?} lView
     * @param {?} tNode
     * @param {?} result
     * @param {?=} isProjection
     * @return {?}
     */
    function collectNativeNodes(tView, lView, tNode, result, isProjection = false) {
        while (tNode !== null) {
            ngDevMode &&
                assertNodeOfPossibleTypes(tNode, 3 /* Element */, 0 /* Container */, 1 /* Projection */, 4 /* ElementContainer */, 5 /* IcuContainer */);
            /** @type {?} */
            const lNode = lView[tNode.index];
            if (lNode !== null) {
                result.push(unwrapRNode(lNode));
            }
            // A given lNode can represent either a native node or a LContainer (when it is a host of a
            // ViewContainerRef). When we find a LContainer we need to descend into it to collect root nodes
            // from the views in this container.
            if (isLContainer(lNode)) {
                for (let i = CONTAINER_HEADER_OFFSET; i < lNode.length; i++) {
                    /** @type {?} */
                    const lViewInAContainer = lNode[i];
                    /** @type {?} */
                    const lViewFirstChildTNode = lViewInAContainer[TVIEW].firstChild;
                    if (lViewFirstChildTNode !== null) {
                        collectNativeNodes(lViewInAContainer[TVIEW], lViewInAContainer, lViewFirstChildTNode, result);
                    }
                }
            }
            /** @type {?} */
            const tNodeType = tNode.type;
            if (tNodeType === 4 /* ElementContainer */ || tNodeType === 5 /* IcuContainer */) {
                collectNativeNodes(tView, lView, tNode.child, result);
            }
            else if (tNodeType === 1 /* Projection */) {
                /** @type {?} */
                const componentView = lView[DECLARATION_COMPONENT_VIEW];
                /** @type {?} */
                const componentHost = (/** @type {?} */ (componentView[T_HOST]));
                /** @type {?} */
                const slotIdx = (/** @type {?} */ (tNode.projection));
                ngDevMode &&
                    assertDefined(componentHost.projection, 'Components with projection nodes (<ng-content>) must have projection slots defined.');
                /** @type {?} */
                const nodesInSlot = (/** @type {?} */ (componentHost.projection))[slotIdx];
                if (Array.isArray(nodesInSlot)) {
                    result.push(...nodesInSlot);
                }
                else {
                    /** @type {?} */
                    const parentView = (/** @type {?} */ (getLViewParent(componentView)));
                    ngDevMode &&
                        assertDefined(parentView, 'Component views should always have a parent view (component\'s host view)');
                    collectNativeNodes(parentView[TVIEW], parentView, nodesInSlot, result, true);
                }
            }
            tNode = isProjection ? tNode.projectionNext : tNode.next;
        }
        return result;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/view_engine_compatibility.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Creates an ElementRef from the most recent node.
     *
     * @param {?} ElementRefToken
     * @return {?} The ElementRef instance to use
     */
    function injectElementRef(ElementRefToken) {
        return createElementRef(ElementRefToken, getPreviousOrParentTNode(), getLView());
    }
    /** @type {?} */
    let R3ElementRef;
    /**
     * Creates an ElementRef given a node.
     *
     * @param {?} ElementRefToken The ElementRef type
     * @param {?} tNode The node for which you'd like an ElementRef
     * @param {?} view The view to which the node belongs
     * @return {?} The ElementRef instance to use
     */
    function createElementRef(ElementRefToken, tNode, view) {
        if (!R3ElementRef) {
            R3ElementRef = class ElementRef extends ElementRefToken {
            };
        }
        return new R3ElementRef((/** @type {?} */ (getNativeByTNode(tNode, view))));
    }
    /** @type {?} */
    let R3TemplateRef;
    /**
     * Creates a TemplateRef given a node.
     *
     * @template T
     * @param {?} TemplateRefToken
     * @param {?} ElementRefToken
     * @return {?} The TemplateRef instance to use
     */
    function injectTemplateRef(TemplateRefToken, ElementRefToken) {
        return createTemplateRef(TemplateRefToken, ElementRefToken, getPreviousOrParentTNode(), getLView());
    }
    /**
     * Creates a TemplateRef and stores it on the injector.
     *
     * @template T
     * @param {?} TemplateRefToken The TemplateRef type
     * @param {?} ElementRefToken The ElementRef type
     * @param {?} hostTNode The node on which a TemplateRef is requested
     * @param {?} hostView The view to which the node belongs
     * @return {?} The TemplateRef instance or null if we can't create a TemplateRef on a given node type
     */
    function createTemplateRef(TemplateRefToken, ElementRefToken, hostTNode, hostView) {
        if (!R3TemplateRef) {
            R3TemplateRef = class TemplateRef extends TemplateRefToken {
                /**
                 * @param {?} _declarationView
                 * @param {?} _declarationTContainer
                 * @param {?} elementRef
                 */
                constructor(_declarationView, _declarationTContainer, elementRef) {
                    super();
                    this._declarationView = _declarationView;
                    this._declarationTContainer = _declarationTContainer;
                    this.elementRef = elementRef;
                }
                /**
                 * @param {?} context
                 * @return {?}
                 */
                createEmbeddedView(context) {
                    /** @type {?} */
                    const embeddedTView = (/** @type {?} */ (this._declarationTContainer.tViews));
                    /** @type {?} */
                    const embeddedLView = createLView(this._declarationView, embeddedTView, context, 16 /* CheckAlways */, null, embeddedTView.node);
                    /** @type {?} */
                    const declarationLContainer = this._declarationView[this._declarationTContainer.index];
                    ngDevMode && assertLContainer(declarationLContainer);
                    embeddedLView[DECLARATION_LCONTAINER] = declarationLContainer;
                    /** @type {?} */
                    const declarationViewLQueries = this._declarationView[QUERIES];
                    if (declarationViewLQueries !== null) {
                        embeddedLView[QUERIES] = declarationViewLQueries.createEmbeddedView(embeddedTView);
                    }
                    renderView(embeddedTView, embeddedLView, context);
                    return new ViewRef(embeddedLView);
                }
            };
        }
        if (hostTNode.type === 0 /* Container */) {
            ngDevMode && assertDefined(hostTNode.tViews, 'TView must be allocated');
            return new R3TemplateRef(hostView, (/** @type {?} */ (hostTNode)), createElementRef(ElementRefToken, hostTNode, hostView));
        }
        else {
            return null;
        }
    }
    /** @type {?} */
    let R3ViewContainerRef;
    /**
     * Creates a ViewContainerRef and stores it on the injector. Or, if the ViewContainerRef
     * already exists, retrieves the existing ViewContainerRef.
     *
     * @param {?} ViewContainerRefToken
     * @param {?} ElementRefToken
     * @return {?} The ViewContainerRef instance to use
     */
    function injectViewContainerRef(ViewContainerRefToken, ElementRefToken) {
        /** @type {?} */
        const previousTNode = (/** @type {?} */ (getPreviousOrParentTNode()));
        return createContainerRef(ViewContainerRefToken, ElementRefToken, previousTNode, getLView());
    }
    /**
     * Creates a ViewContainerRef and stores it on the injector.
     *
     * @param {?} ViewContainerRefToken The ViewContainerRef type
     * @param {?} ElementRefToken The ElementRef type
     * @param {?} hostTNode The node that is requesting a ViewContainerRef
     * @param {?} hostView The view to which the node belongs
     * @return {?} The ViewContainerRef instance to use
     */
    function createContainerRef(ViewContainerRefToken, ElementRefToken, hostTNode, hostView) {
        if (!R3ViewContainerRef) {
            R3ViewContainerRef = class ViewContainerRef extends ViewContainerRefToken {
                /**
                 * @param {?} _lContainer
                 * @param {?} _hostTNode
                 * @param {?} _hostView
                 */
                constructor(_lContainer, _hostTNode, _hostView) {
                    super();
                    this._lContainer = _lContainer;
                    this._hostTNode = _hostTNode;
                    this._hostView = _hostView;
                }
                /**
                 * @return {?}
                 */
                get element() {
                    return createElementRef(ElementRefToken, this._hostTNode, this._hostView);
                }
                /**
                 * @return {?}
                 */
                get injector() {
                    return new NodeInjector(this._hostTNode, this._hostView);
                }
                /**
                 * @deprecated No replacement
                 * @return {?}
                 */
                get parentInjector() {
                    /** @type {?} */
                    const parentLocation = getParentInjectorLocation(this._hostTNode, this._hostView);
                    /** @type {?} */
                    const parentView = getParentInjectorView(parentLocation, this._hostView);
                    /** @type {?} */
                    const parentTNode = getParentInjectorTNode(parentLocation, this._hostView, this._hostTNode);
                    return !hasParentInjector(parentLocation) || parentTNode == null ?
                        new NodeInjector(null, this._hostView) :
                        new NodeInjector(parentTNode, parentView);
                }
                /**
                 * @return {?}
                 */
                clear() {
                    while (this.length > 0) {
                        this.remove(this.length - 1);
                    }
                }
                /**
                 * @param {?} index
                 * @return {?}
                 */
                get(index) {
                    return this._lContainer[VIEW_REFS] !== null && (/** @type {?} */ (this._lContainer[VIEW_REFS]))[index] || null;
                }
                /**
                 * @return {?}
                 */
                get length() {
                    return this._lContainer.length - CONTAINER_HEADER_OFFSET;
                }
                /**
                 * @template C
                 * @param {?} templateRef
                 * @param {?=} context
                 * @param {?=} index
                 * @return {?}
                 */
                createEmbeddedView(templateRef, context, index) {
                    /** @type {?} */
                    const viewRef = templateRef.createEmbeddedView(context || (/** @type {?} */ ({})));
                    this.insert(viewRef, index);
                    return viewRef;
                }
                /**
                 * @template C
                 * @param {?} componentFactory
                 * @param {?=} index
                 * @param {?=} injector
                 * @param {?=} projectableNodes
                 * @param {?=} ngModuleRef
                 * @return {?}
                 */
                createComponent(componentFactory, index, injector, projectableNodes, ngModuleRef) {
                    /** @type {?} */
                    const contextInjector = injector || this.parentInjector;
                    if (!ngModuleRef && ((/** @type {?} */ (componentFactory))).ngModule == null && contextInjector) {
                        // DO NOT REFACTOR. The code here used to have a `value || undefined` expression
                        // which seems to cause internal google apps to fail. This is documented in the
                        // following internal bug issue: go/b/142967802
                        /** @type {?} */
                        const result = contextInjector.get(NgModuleRef, null);
                        if (result) {
                            ngModuleRef = result;
                        }
                    }
                    /** @type {?} */
                    const componentRef = componentFactory.create(contextInjector, projectableNodes, undefined, ngModuleRef);
                    this.insert(componentRef.hostView, index);
                    return componentRef;
                }
                /**
                 * @param {?} viewRef
                 * @param {?=} index
                 * @return {?}
                 */
                insert(viewRef, index) {
                    /** @type {?} */
                    const lView = (/** @type {?} */ (((/** @type {?} */ (viewRef)))._lView));
                    /** @type {?} */
                    const tView = lView[TVIEW];
                    if (viewRef.destroyed) {
                        throw new Error('Cannot insert a destroyed View in a ViewContainer!');
                    }
                    this.allocateContainerIfNeeded();
                    if (viewAttachedToContainer(lView)) {
                        // If view is already attached, detach it first so we clean up references appropriately.
                        /** @type {?} */
                        const prevIdx = this.indexOf(viewRef);
                        // A view might be attached either to this or a different container. The `prevIdx` for
                        // those cases will be:
                        // equal to -1 for views attached to this ViewContainerRef
                        // >= 0 for views attached to a different ViewContainerRef
                        if (prevIdx !== -1) {
                            this.detach(prevIdx);
                        }
                        else {
                            /** @type {?} */
                            const prevLContainer = (/** @type {?} */ (lView[PARENT]));
                            ngDevMode &&
                                assertEqual(isLContainer(prevLContainer), true, 'An attached view should have its PARENT point to a container.');
                            // We need to re-create a R3ViewContainerRef instance since those are not stored on
                            // LView (nor anywhere else).
                            /** @type {?} */
                            const prevVCRef = new R3ViewContainerRef(prevLContainer, (/** @type {?} */ (prevLContainer[T_HOST])), prevLContainer[PARENT]);
                            prevVCRef.detach(prevVCRef.indexOf(viewRef));
                        }
                    }
                    /** @type {?} */
                    const adjustedIdx = this._adjustIndex(index);
                    insertView(tView, lView, this._lContainer, adjustedIdx);
                    /** @type {?} */
                    const beforeNode = getBeforeNodeForView(adjustedIdx, this._lContainer);
                    addRemoveViewFromContainer(tView, lView, true, beforeNode);
                    ((/** @type {?} */ (viewRef))).attachToViewContainerRef(this);
                    addToArray((/** @type {?} */ (this._lContainer[VIEW_REFS])), adjustedIdx, viewRef);
                    return viewRef;
                }
                /**
                 * @param {?} viewRef
                 * @param {?} newIndex
                 * @return {?}
                 */
                move(viewRef, newIndex) {
                    if (viewRef.destroyed) {
                        throw new Error('Cannot move a destroyed View in a ViewContainer!');
                    }
                    return this.insert(viewRef, newIndex);
                }
                /**
                 * @param {?} viewRef
                 * @return {?}
                 */
                indexOf(viewRef) {
                    /** @type {?} */
                    const viewRefsArr = this._lContainer[VIEW_REFS];
                    return viewRefsArr !== null ? viewRefsArr.indexOf(viewRef) : -1;
                }
                /**
                 * @param {?=} index
                 * @return {?}
                 */
                remove(index) {
                    this.allocateContainerIfNeeded();
                    /** @type {?} */
                    const adjustedIdx = this._adjustIndex(index, -1);
                    removeView(this._lContainer, adjustedIdx);
                    removeFromArray((/** @type {?} */ (this._lContainer[VIEW_REFS])), adjustedIdx);
                }
                /**
                 * @param {?=} index
                 * @return {?}
                 */
                detach(index) {
                    this.allocateContainerIfNeeded();
                    /** @type {?} */
                    const adjustedIdx = this._adjustIndex(index, -1);
                    /** @type {?} */
                    const view = detachView(this._lContainer, adjustedIdx);
                    /** @type {?} */
                    const wasDetached = view && removeFromArray((/** @type {?} */ (this._lContainer[VIEW_REFS])), adjustedIdx) != null;
                    return wasDetached ? new ViewRef((/** @type {?} */ (view))) : null;
                }
                /**
                 * @private
                 * @param {?=} index
                 * @param {?=} shift
                 * @return {?}
                 */
                _adjustIndex(index, shift = 0) {
                    if (index == null) {
                        return this.length + shift;
                    }
                    if (ngDevMode) {
                        assertGreaterThan(index, -1, `ViewRef index must be positive, got ${index}`);
                        // +1 because it's legal to insert at the end.
                        assertLessThan(index, this.length + 1 + shift, 'index');
                    }
                    return index;
                }
                /**
                 * @private
                 * @return {?}
                 */
                allocateContainerIfNeeded() {
                    if (this._lContainer[VIEW_REFS] === null) {
                        this._lContainer[VIEW_REFS] = [];
                    }
                }
            };
        }
        ngDevMode &&
            assertNodeOfPossibleTypes(hostTNode, 0 /* Container */, 3 /* Element */, 4 /* ElementContainer */);
        /** @type {?} */
        let lContainer;
        /** @type {?} */
        const slotValue = hostView[hostTNode.index];
        if (isLContainer(slotValue)) {
            // If the host is a container, we don't need to create a new LContainer
            lContainer = slotValue;
        }
        else {
            /** @type {?} */
            let commentNode;
            // If the host is an element container, the native host element is guaranteed to be a
            // comment and we can reuse that comment as anchor element for the new LContainer.
            // The comment node in question is already part of the DOM structure so we don't need to append
            // it again.
            if (hostTNode.type === 4 /* ElementContainer */) {
                commentNode = (/** @type {?} */ (unwrapRNode(slotValue)));
            }
            else {
                ngDevMode && ngDevMode.rendererCreateComment++;
                commentNode = hostView[RENDERER].createComment(ngDevMode ? 'container' : '');
                // A `ViewContainerRef` can be injected by the root (topmost / bootstrapped) component. In
                // this case we can't use TView / TNode data structures to insert container's marker node
                // (both a parent of a comment node and the comment node itself are not part of any view). In
                // this specific case we use low-level DOM manipulation to insert container's marker (comment)
                // node.
                if (isRootView(hostView)) {
                    /** @type {?} */
                    const renderer = hostView[RENDERER];
                    /** @type {?} */
                    const hostNative = (/** @type {?} */ (getNativeByTNode(hostTNode, hostView)));
                    /** @type {?} */
                    const parentOfHostNative = nativeParentNode(renderer, hostNative);
                    nativeInsertBefore(renderer, (/** @type {?} */ (parentOfHostNative)), commentNode, nativeNextSibling(renderer, hostNative));
                }
                else {
                    appendChild(hostView[TVIEW], hostView, commentNode, hostTNode);
                }
            }
            hostView[hostTNode.index] = lContainer =
                createLContainer(slotValue, hostView, commentNode, hostTNode);
            addToViewTree(hostView, lContainer);
        }
        return new R3ViewContainerRef(lContainer, hostTNode, hostView);
    }
    /**
     * Returns a ChangeDetectorRef (a.k.a. a ViewRef)
     * @param {?=} isPipe
     * @return {?}
     */
    function injectChangeDetectorRef(isPipe = false) {
        return createViewRef(getPreviousOrParentTNode(), getLView(), isPipe);
    }
    /**
     * Creates a ViewRef and stores it on the injector as ChangeDetectorRef (public alias).
     *
     * @param {?} tNode The node that is requesting a ChangeDetectorRef
     * @param {?} lView The view to which the node belongs
     * @param {?} isPipe Whether the view is being injected into a pipe.
     * @return {?} The ChangeDetectorRef to use
     */
    function createViewRef(tNode, lView, isPipe) {
        // `isComponentView` will be true for Component and Directives (but not for Pipes).
        // See https://github.com/angular/angular/pull/33072 for proper fix
        /** @type {?} */
        const isComponentView = !isPipe && isComponentHost(tNode);
        if (isComponentView) {
            // The LView represents the location where the component is declared.
            // Instead we want the LView for the component View and so we need to look it up.
            /** @type {?} */
            const componentView = getComponentLViewByIndex(tNode.index, lView);
            return new ViewRef(componentView, componentView);
        }
        else if (tNode.type === 3 /* Element */ || tNode.type === 0 /* Container */ ||
            tNode.type === 4 /* ElementContainer */ || tNode.type === 5 /* IcuContainer */) {
            // The LView represents the location where the injection is requested from.
            // We need to locate the containing LView (in case where the `lView` is an embedded view)
            /** @type {?} */
            const hostComponentView = lView[DECLARATION_COMPONENT_VIEW];
            return new ViewRef(hostComponentView, lView);
        }
        return (/** @type {?} */ (null));
    }
    /**
     * Returns a Renderer2 (or throws when application was bootstrapped with Renderer3)
     * @param {?} view
     * @return {?}
     */
    function getOrCreateRenderer2(view) {
        /** @type {?} */
        const renderer = view[RENDERER];
        if (isProceduralRenderer(renderer)) {
            return (/** @type {?} */ (renderer));
        }
        else {
            throw new Error('Cannot inject Renderer2 when the application uses Renderer3!');
        }
    }
    /**
     * Injects a Renderer2 for the current component.
     * @return {?}
     */
    function injectRenderer2() {
        // We need the Renderer to be based on the component that it's being injected into, however since
        // DI happens before we've entered its view, `getLView` will return the parent view instead.
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const tNode = getPreviousOrParentTNode();
        /** @type {?} */
        const nodeAtIndex = getComponentLViewByIndex(tNode.index, lView);
        return getOrCreateRenderer2(isLView(nodeAtIndex) ? nodeAtIndex : lView);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/change_detection/change_detector_ref.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Base class for Angular Views, provides change detection functionality.
     * A change-detection tree collects all views that are to be checked for changes.
     * Use the methods to add and remove views from the tree, initiate change-detection,
     * and explicitly mark views as _dirty_, meaning that they have changed and need to be rerendered.
     *
     * \@usageNotes
     *
     * The following examples demonstrate how to modify default change-detection behavior
     * to perform explicit detection when needed.
     *
     * ### Use `markForCheck()` with `CheckOnce` strategy
     *
     * The following example sets the `OnPush` change-detection strategy for a component
     * (`CheckOnce`, rather than the default `CheckAlways`), then forces a second check
     * after an interval. See [live demo](http://plnkr.co/edit/GC512b?p=preview).
     *
     * <code-example path="core/ts/change_detect/change-detection.ts"
     * region="mark-for-check"></code-example>
     *
     * ### Detach change detector to limit how often check occurs
     *
     * The following example defines a component with a large list of read-only data
     * that is expected to change constantly, many times per second.
     * To improve performance, we want to check and update the list
     * less often than the changes actually occur. To do that, we detach
     * the component's change detector and perform an explicit local check every five seconds.
     *
     * <code-example path="core/ts/change_detect/change-detection.ts" region="detach"></code-example>
     *
     *
     * ### Reattaching a detached component
     *
     * The following example creates a component displaying live data.
     * The component detaches its change detector from the main change detector tree
     * when the `live` property is set to false, and reattaches it when the property
     * becomes true.
     *
     * <code-example path="core/ts/change_detect/change-detection.ts" region="reattach"></code-example>
     *
     * \@publicApi
     * @abstract
     */
    class ChangeDetectorRef {
    }
    /**
     * \@internal
     * @nocollapse
     */
    ChangeDetectorRef.__NG_ELEMENT_ID__ = (/**
     * @return {?}
     */
    () => SWITCH_CHANGE_DETECTOR_REF_FACTORY());
    /** @type {?} */
    const SWITCH_CHANGE_DETECTOR_REF_FACTORY__POST_R3__ = injectChangeDetectorRef;
    /** @type {?} */
    const SWITCH_CHANGE_DETECTOR_REF_FACTORY = SWITCH_CHANGE_DETECTOR_REF_FACTORY__POST_R3__;

    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * @description
     *
     * Represents a type that a Component or other object is instances of.
     *
     * An example of a `Type` is `MyCustomComponent` class, which in JavaScript is be represented by
     * the `MyCustomComponent` constructor function.
     *
     * @publicApi
     */
    const Type = Function;
    function isType(v) {
        return typeof v === 'function';
    }

    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * Attention: These regex has to hold even if the code is minified!
     */
    const DELEGATE_CTOR = /^function\s+\S+\(\)\s*{[\s\S]+\.apply\(this,\s*arguments\)/;
    const INHERITED_CLASS = /^class\s+[A-Za-z\d$_]*\s*extends\s+[^{]+{/;
    const INHERITED_CLASS_WITH_CTOR = /^class\s+[A-Za-z\d$_]*\s*extends\s+[^{]+{[\s\S]*constructor\s*\(/;
    const INHERITED_CLASS_WITH_DELEGATE_CTOR = /^class\s+[A-Za-z\d$_]*\s*extends\s+[^{]+{[\s\S]*constructor\s*\(\)\s*{\s*super\(\.\.\.arguments\)/;
    /**
     * Determine whether a stringified type is a class which delegates its constructor
     * to its parent.
     *
     * This is not trivial since compiled code can actually contain a constructor function
     * even if the original source code did not. For instance, when the child class contains
     * an initialized instance property.
     */
    function isDelegateCtor(typeStr) {
        return DELEGATE_CTOR.test(typeStr) || INHERITED_CLASS_WITH_DELEGATE_CTOR.test(typeStr) ||
            (INHERITED_CLASS.test(typeStr) && !INHERITED_CLASS_WITH_CTOR.test(typeStr));
    }
    class ReflectionCapabilities {
        constructor(reflect) {
            this._reflect = reflect || _global['Reflect'];
        }
        isReflectionEnabled() {
            return true;
        }
        factory(t) {
            return (...args) => new t(...args);
        }
        /** @internal */
        _zipTypesAndAnnotations(paramTypes, paramAnnotations) {
            let result;
            if (typeof paramTypes === 'undefined') {
                result = newArray(paramAnnotations.length);
            }
            else {
                result = newArray(paramTypes.length);
            }
            for (let i = 0; i < result.length; i++) {
                // TS outputs Object for parameters without types, while Traceur omits
                // the annotations. For now we preserve the Traceur behavior to aid
                // migration, but this can be revisited.
                if (typeof paramTypes === 'undefined') {
                    result[i] = [];
                }
                else if (paramTypes[i] && paramTypes[i] != Object) {
                    result[i] = [paramTypes[i]];
                }
                else {
                    result[i] = [];
                }
                if (paramAnnotations && paramAnnotations[i] != null) {
                    result[i] = result[i].concat(paramAnnotations[i]);
                }
            }
            return result;
        }
        _ownParameters(type, parentCtor) {
            const typeStr = type.toString();
            // If we have no decorators, we only have function.length as metadata.
            // In that case, to detect whether a child class declared an own constructor or not,
            // we need to look inside of that constructor to check whether it is
            // just calling the parent.
            // This also helps to work around for https://github.com/Microsoft/TypeScript/issues/12439
            // that sets 'design:paramtypes' to []
            // if a class inherits from another class but has no ctor declared itself.
            if (isDelegateCtor(typeStr)) {
                return null;
            }
            // Prefer the direct API.
            if (type.parameters && type.parameters !== parentCtor.parameters) {
                return type.parameters;
            }
            // API of tsickle for lowering decorators to properties on the class.
            const tsickleCtorParams = type.ctorParameters;
            if (tsickleCtorParams && tsickleCtorParams !== parentCtor.ctorParameters) {
                // Newer tsickle uses a function closure
                // Retain the non-function case for compatibility with older tsickle
                const ctorParameters = typeof tsickleCtorParams === 'function' ? tsickleCtorParams() : tsickleCtorParams;
                const paramTypes = ctorParameters.map((ctorParam) => ctorParam && ctorParam.type);
                const paramAnnotations = ctorParameters.map((ctorParam) => ctorParam && convertTsickleDecoratorIntoMetadata(ctorParam.decorators));
                return this._zipTypesAndAnnotations(paramTypes, paramAnnotations);
            }
            // API for metadata created by invoking the decorators.
            const paramAnnotations = type.hasOwnProperty(PARAMETERS) && type[PARAMETERS];
            const paramTypes = this._reflect && this._reflect.getOwnMetadata &&
                this._reflect.getOwnMetadata('design:paramtypes', type);
            if (paramTypes || paramAnnotations) {
                return this._zipTypesAndAnnotations(paramTypes, paramAnnotations);
            }
            // If a class has no decorators, at least create metadata
            // based on function.length.
            // Note: We know that this is a real constructor as we checked
            // the content of the constructor above.
            return newArray(type.length);
        }
        parameters(type) {
            // Note: only report metadata if we have at least one class decorator
            // to stay in sync with the static reflector.
            if (!isType(type)) {
                return [];
            }
            const parentCtor = getParentCtor(type);
            let parameters = this._ownParameters(type, parentCtor);
            if (!parameters && parentCtor !== Object) {
                parameters = this.parameters(parentCtor);
            }
            return parameters || [];
        }
        _ownAnnotations(typeOrFunc, parentCtor) {
            // Prefer the direct API.
            if (typeOrFunc.annotations && typeOrFunc.annotations !== parentCtor.annotations) {
                let annotations = typeOrFunc.annotations;
                if (typeof annotations === 'function' && annotations.annotations) {
                    annotations = annotations.annotations;
                }
                return annotations;
            }
            // API of tsickle for lowering decorators to properties on the class.
            if (typeOrFunc.decorators && typeOrFunc.decorators !== parentCtor.decorators) {
                return convertTsickleDecoratorIntoMetadata(typeOrFunc.decorators);
            }
            // API for metadata created by invoking the decorators.
            if (typeOrFunc.hasOwnProperty(ANNOTATIONS)) {
                return typeOrFunc[ANNOTATIONS];
            }
            return null;
        }
        annotations(typeOrFunc) {
            if (!isType(typeOrFunc)) {
                return [];
            }
            const parentCtor = getParentCtor(typeOrFunc);
            const ownAnnotations = this._ownAnnotations(typeOrFunc, parentCtor) || [];
            const parentAnnotations = parentCtor !== Object ? this.annotations(parentCtor) : [];
            return parentAnnotations.concat(ownAnnotations);
        }
        _ownPropMetadata(typeOrFunc, parentCtor) {
            // Prefer the direct API.
            if (typeOrFunc.propMetadata &&
                typeOrFunc.propMetadata !== parentCtor.propMetadata) {
                let propMetadata = typeOrFunc.propMetadata;
                if (typeof propMetadata === 'function' && propMetadata.propMetadata) {
                    propMetadata = propMetadata.propMetadata;
                }
                return propMetadata;
            }
            // API of tsickle for lowering decorators to properties on the class.
            if (typeOrFunc.propDecorators &&
                typeOrFunc.propDecorators !== parentCtor.propDecorators) {
                const propDecorators = typeOrFunc.propDecorators;
                const propMetadata = {};
                Object.keys(propDecorators).forEach(prop => {
                    propMetadata[prop] = convertTsickleDecoratorIntoMetadata(propDecorators[prop]);
                });
                return propMetadata;
            }
            // API for metadata created by invoking the decorators.
            if (typeOrFunc.hasOwnProperty(PROP_METADATA)) {
                return typeOrFunc[PROP_METADATA];
            }
            return null;
        }
        propMetadata(typeOrFunc) {
            if (!isType(typeOrFunc)) {
                return {};
            }
            const parentCtor = getParentCtor(typeOrFunc);
            const propMetadata = {};
            if (parentCtor !== Object) {
                const parentPropMetadata = this.propMetadata(parentCtor);
                Object.keys(parentPropMetadata).forEach((propName) => {
                    propMetadata[propName] = parentPropMetadata[propName];
                });
            }
            const ownPropMetadata = this._ownPropMetadata(typeOrFunc, parentCtor);
            if (ownPropMetadata) {
                Object.keys(ownPropMetadata).forEach((propName) => {
                    const decorators = [];
                    if (propMetadata.hasOwnProperty(propName)) {
                        decorators.push(...propMetadata[propName]);
                    }
                    decorators.push(...ownPropMetadata[propName]);
                    propMetadata[propName] = decorators;
                });
            }
            return propMetadata;
        }
        ownPropMetadata(typeOrFunc) {
            if (!isType(typeOrFunc)) {
                return {};
            }
            return this._ownPropMetadata(typeOrFunc, getParentCtor(typeOrFunc)) || {};
        }
        hasLifecycleHook(type, lcProperty) {
            return type instanceof Type && lcProperty in type.prototype;
        }
        guards(type) {
            return {};
        }
        getter(name) {
            return new Function('o', 'return o.' + name + ';');
        }
        setter(name) {
            return new Function('o', 'v', 'return o.' + name + ' = v;');
        }
        method(name) {
            const functionBody = `if (!o.${name}) throw new Error('"${name}" is undefined');
        return o.${name}.apply(o, args);`;
            return new Function('o', 'args', functionBody);
        }
        // There is not a concept of import uri in Js, but this is useful in developing Dart applications.
        importUri(type) {
            // StaticSymbol
            if (typeof type === 'object' && type['filePath']) {
                return type['filePath'];
            }
            // Runtime type
            return `./${stringify(type)}`;
        }
        resourceUri(type) {
            return `./${stringify(type)}`;
        }
        resolveIdentifier(name, moduleUrl, members, runtime) {
            return runtime;
        }
        resolveEnum(enumIdentifier, name) {
            return enumIdentifier[name];
        }
    }
    function convertTsickleDecoratorIntoMetadata(decoratorInvocations) {
        if (!decoratorInvocations) {
            return [];
        }
        return decoratorInvocations.map(decoratorInvocation => {
            const decoratorType = decoratorInvocation.type;
            const annotationCls = decoratorType.annotationCls;
            const annotationArgs = decoratorInvocation.args ? decoratorInvocation.args : [];
            return new annotationCls(...annotationArgs);
        });
    }
    function getParentCtor(ctor) {
        const parentProto = ctor.prototype ? Object.getPrototypeOf(ctor.prototype) : null;
        const parentCtor = parentProto ? parentProto.constructor : null;
        // Note: We always use `Object` as the null value
        // to simplify checking later on.
        return parentCtor || Object;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/di/jit/util.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    let _reflect = null;
    /**
     * @return {?}
     */
    function getReflect() {
        return (_reflect = _reflect || new ReflectionCapabilities());
    }
    /**
     * @param {?} type
     * @return {?}
     */
    function reflectDependencies(type) {
        return convertDependencies(getReflect().parameters(type));
    }
    /**
     * @param {?} deps
     * @return {?}
     */
    function convertDependencies(deps) {
        /** @type {?} */
        const compiler = getCompilerFacade();
        return deps.map((/**
         * @param {?} dep
         * @return {?}
         */
        dep => reflectDependency(compiler, dep)));
    }
    /**
     * @param {?} compiler
     * @param {?} dep
     * @return {?}
     */
    function reflectDependency(compiler, dep) {
        /** @type {?} */
        const meta = {
            token: null,
            host: false,
            optional: false,
            resolved: compiler.R3ResolvedDependencyType.Token,
            self: false,
            skipSelf: false,
        };
        /**
         * @param {?} token
         * @return {?}
         */
        function setTokenAndResolvedType(token) {
            meta.resolved = compiler.R3ResolvedDependencyType.Token;
            meta.token = token;
        }
        if (Array.isArray(dep) && dep.length > 0) {
            for (let j = 0; j < dep.length; j++) {
                /** @type {?} */
                const param = dep[j];
                if (param === undefined) {
                    // param may be undefined if type of dep is not set by ngtsc
                    continue;
                }
                /** @type {?} */
                const proto = Object.getPrototypeOf(param);
                if (param instanceof Optional || proto.ngMetadataName === 'Optional') {
                    meta.optional = true;
                }
                else if (param instanceof SkipSelf || proto.ngMetadataName === 'SkipSelf') {
                    meta.skipSelf = true;
                }
                else if (param instanceof Self || proto.ngMetadataName === 'Self') {
                    meta.self = true;
                }
                else if (param instanceof Host || proto.ngMetadataName === 'Host') {
                    meta.host = true;
                }
                else if (param instanceof Inject) {
                    meta.token = param.token;
                }
                else if (param instanceof Attribute) {
                    if (param.attributeName === undefined) {
                        throw new Error(`Attribute name must be defined.`);
                    }
                    meta.token = param.attributeName;
                    meta.resolved = compiler.R3ResolvedDependencyType.Attribute;
                }
                else if (param === ChangeDetectorRef) {
                    meta.token = param;
                    meta.resolved = compiler.R3ResolvedDependencyType.ChangeDetectorRef;
                }
                else {
                    setTokenAndResolvedType(param);
                }
            }
        }
        else if (dep === undefined || (Array.isArray(dep) && dep.length === 0)) {
            meta.token = undefined;
            meta.resolved = R3ResolvedDependencyType.Invalid;
        }
        else {
            setTokenAndResolvedType(dep);
        }
        return meta;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/di/jit/injectable.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Compile an Angular injectable according to its `Injectable` metadata, and patch the resulting
     * injectable def (`ɵprov`) onto the injectable type.
     * @param {?} type
     * @param {?=} srcMeta
     * @return {?}
     */
    function compileInjectable(type, srcMeta) {
        /** @type {?} */
        let ngInjectableDef = null;
        /** @type {?} */
        let ngFactoryDef = null;
        // if NG_PROV_DEF is already defined on this class then don't overwrite it
        if (!type.hasOwnProperty(NG_PROV_DEF)) {
            Object.defineProperty(type, NG_PROV_DEF, {
                get: (/**
                 * @return {?}
                 */
                () => {
                    if (ngInjectableDef === null) {
                        ngInjectableDef = getCompilerFacade().compileInjectable(angularCoreDiEnv, `ng:///${type.name}/ɵprov.js`, getInjectableMetadata(type, srcMeta));
                    }
                    return ngInjectableDef;
                }),
            });
            // On IE10 properties defined via `defineProperty` won't be inherited by child classes,
            // which will break inheriting the injectable definition from a grandparent through an
            // undecorated parent class. We work around it by defining a method which should be used
            // as a fallback. This should only be a problem in JIT mode, because in AOT TypeScript
            // seems to have a workaround for static properties. When inheriting from an undecorated
            // parent is no longer supported in v10, this can safely be removed.
            if (!type.hasOwnProperty(NG_PROV_DEF_FALLBACK)) {
                ((/** @type {?} */ (type)))[NG_PROV_DEF_FALLBACK] = (/**
                 * @return {?}
                 */
                () => ((/** @type {?} */ (type)))[NG_PROV_DEF]);
            }
        }
        // if NG_FACTORY_DEF is already defined on this class then don't overwrite it
        if (!type.hasOwnProperty(NG_FACTORY_DEF)) {
            Object.defineProperty(type, NG_FACTORY_DEF, {
                get: (/**
                 * @return {?}
                 */
                () => {
                    if (ngFactoryDef === null) {
                        /** @type {?} */
                        const metadata = getInjectableMetadata(type, srcMeta);
                        /** @type {?} */
                        const compiler = getCompilerFacade();
                        ngFactoryDef = compiler.compileFactory(angularCoreDiEnv, `ng:///${type.name}/ɵfac.js`, {
                            name: metadata.name,
                            type: metadata.type,
                            typeArgumentCount: metadata.typeArgumentCount,
                            deps: reflectDependencies(type),
                            injectFn: 'inject',
                            target: compiler.R3FactoryTarget.Injectable
                        });
                    }
                    return ngFactoryDef;
                }),
                // Leave this configurable so that the factories from directives or pipes can take precedence.
                configurable: true
            });
        }
    }
    const ɵ0$6 = getClosureSafeProperty;
    /** @type {?} */
    const USE_VALUE$1 = getClosureSafeProperty({ provide: String, useValue: ɵ0$6 });
    /**
     * @param {?} meta
     * @return {?}
     */
    function isUseClassProvider(meta) {
        return ((/** @type {?} */ (meta))).useClass !== undefined;
    }
    /**
     * @param {?} meta
     * @return {?}
     */
    function isUseValueProvider(meta) {
        return USE_VALUE$1 in meta;
    }
    /**
     * @param {?} meta
     * @return {?}
     */
    function isUseFactoryProvider(meta) {
        return ((/** @type {?} */ (meta))).useFactory !== undefined;
    }
    /**
     * @param {?} meta
     * @return {?}
     */
    function isUseExistingProvider(meta) {
        return ((/** @type {?} */ (meta))).useExisting !== undefined;
    }
    /**
     * @param {?} type
     * @param {?=} srcMeta
     * @return {?}
     */
    function getInjectableMetadata(type, srcMeta) {
        // Allow the compilation of a class with a `@Injectable()` decorator without parameters
        /** @type {?} */
        const meta = srcMeta || { providedIn: null };
        /** @type {?} */
        const compilerMeta = {
            name: type.name,
            type: type,
            typeArgumentCount: 0,
            providedIn: meta.providedIn,
            userDeps: undefined,
        };
        if ((isUseClassProvider(meta) || isUseFactoryProvider(meta)) && meta.deps !== undefined) {
            compilerMeta.userDeps = convertDependencies(meta.deps);
        }
        if (isUseClassProvider(meta)) {
            // The user explicitly specified useClass, and may or may not have provided deps.
            compilerMeta.useClass = resolveForwardRef(meta.useClass);
        }
        else if (isUseValueProvider(meta)) {
            // The user explicitly specified useValue.
            compilerMeta.useValue = resolveForwardRef(meta.useValue);
        }
        else if (isUseFactoryProvider(meta)) {
            // The user explicitly specified useFactory.
            compilerMeta.useFactory = meta.useFactory;
        }
        else if (isUseExistingProvider(meta)) {
            // The user explicitly specified useExisting.
            compilerMeta.useExisting = resolveForwardRef(meta.useExisting);
        }
        return compilerMeta;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/di/util.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    const ɵ0$7 = getClosureSafeProperty;
    /** @type {?} */
    const USE_VALUE$2 = getClosureSafeProperty({ provide: String, useValue: ɵ0$7 });
    // WARNING: interface has both a type and a value, skipping emit
    const ɵ0$8 = /**
     * @param {?} type
     * @param {?} meta
     * @return {?}
     */
    (type, meta) => SWITCH_COMPILE_INJECTABLE((/** @type {?} */ (type)), meta);
    /**
     * Injectable decorator and metadata.
     *
     * \@Annotation
     * \@publicApi
     * @type {?}
     */
    const Injectable = makeDecorator('Injectable', undefined, undefined, undefined, (ɵ0$8));
    /** @type {?} */
    const SWITCH_COMPILE_INJECTABLE__POST_R3__ = compileInjectable;
    /** @type {?} */
    const SWITCH_COMPILE_INJECTABLE = SWITCH_COMPILE_INJECTABLE__POST_R3__;

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/di/scope.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * An internal token whose presence in an injector indicates that the injector should treat itself
     * as a root scoped injector when processing requests for unknown tokens which may indicate
     * they are provided in the root scope.
     * @type {?}
     */
    const INJECTOR_SCOPE = new InjectionToken('Set Injector scope.');

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/di/r3_injector.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Marker which indicates that a value has not yet been created from the factory function.
     * @type {?}
     */
    const NOT_YET = {};
    /**
     * Marker which indicates that the factory function for a token is in the process of being called.
     *
     * If the injector is asked to inject a token with its value set to CIRCULAR, that indicates
     * injection of a dependency has recursively attempted to inject the original token, and there is
     * a circular dependency among the providers.
     * @type {?}
     */
    const CIRCULAR = {};
    /** @type {?} */
    const EMPTY_ARRAY$2 = (/** @type {?} */ ([]));
    /**
     * A lazily initialized NullInjector.
     * @type {?}
     */
    let NULL_INJECTOR = undefined;
    /**
     * @return {?}
     */
    function getNullInjector() {
        if (NULL_INJECTOR === undefined) {
            NULL_INJECTOR = new NullInjector();
        }
        return NULL_INJECTOR;
    }
    /**
     * Create a new `Injector` which is configured using a `defType` of `InjectorType<any>`s.
     *
     * \@publicApi
     * @param {?} defType
     * @param {?=} parent
     * @param {?=} additionalProviders
     * @param {?=} name
     * @return {?}
     */
    function createInjector(defType, parent = null, additionalProviders = null, name) {
        /** @type {?} */
        const injector = createInjectorWithoutInjectorInstances(defType, parent, additionalProviders, name);
        injector._resolveInjectorDefTypes();
        return injector;
    }
    /**
     * Creates a new injector without eagerly resolving its injector types. Can be used in places
     * where resolving the injector types immediately can lead to an infinite loop. The injector types
     * should be resolved at a later point by calling `_resolveInjectorDefTypes`.
     * @param {?} defType
     * @param {?=} parent
     * @param {?=} additionalProviders
     * @param {?=} name
     * @return {?}
     */
    function createInjectorWithoutInjectorInstances(defType, parent = null, additionalProviders = null, name) {
        return new R3Injector(defType, additionalProviders, parent || getNullInjector(), name);
    }
    class R3Injector {
        /**
         * @param {?} def
         * @param {?} additionalProviders
         * @param {?} parent
         * @param {?=} source
         */
        constructor(def, additionalProviders, parent, source = null) {
            this.parent = parent;
            /**
             * Map of tokens to records which contain the instances of those tokens.
             * - `null` value implies that we don't have the record. Used by tree-shakable injectors
             * to prevent further searches.
             */
            this.records = new Map();
            /**
             * The transitive set of `InjectorType`s which define this injector.
             */
            this.injectorDefTypes = new Set();
            /**
             * Set of values instantiated by this injector which contain `ngOnDestroy` lifecycle hooks.
             */
            this.onDestroy = new Set();
            this._destroyed = false;
            /** @type {?} */
            const dedupStack = [];
            // Start off by creating Records for every provider declared in every InjectorType
            // included transitively in additional providers then do the same for `def`. This order is
            // important because `def` may include providers that override ones in additionalProviders.
            additionalProviders &&
                deepForEach(additionalProviders, (/**
                 * @param {?} provider
                 * @return {?}
                 */
                provider => this.processProvider(provider, def, additionalProviders)));
            deepForEach([def], (/**
             * @param {?} injectorDef
             * @return {?}
             */
            injectorDef => this.processInjectorType(injectorDef, [], dedupStack)));
            // Make sure the INJECTOR token provides this injector.
            this.records.set(INJECTOR, makeRecord(undefined, this));
            // Detect whether this injector has the APP_ROOT_SCOPE token and thus should provide
            // any injectable scoped to APP_ROOT_SCOPE.
            /** @type {?} */
            const record = this.records.get(INJECTOR_SCOPE);
            this.scope = record != null ? record.value : null;
            // Source name, used for debugging
            this.source = source || (typeof def === 'object' ? null : stringify(def));
        }
        /**
         * Flag indicating that this injector was previously destroyed.
         * @return {?}
         */
        get destroyed() {
            return this._destroyed;
        }
        /**
         * Destroy the injector and release references to every instance or provider associated with it.
         *
         * Also calls the `OnDestroy` lifecycle hooks of every instance that was created for which a
         * hook was found.
         * @return {?}
         */
        destroy() {
            this.assertNotDestroyed();
            // Set destroyed = true first, in case lifecycle hooks re-enter destroy().
            this._destroyed = true;
            try {
                // Call all the lifecycle hooks.
                this.onDestroy.forEach((/**
                 * @param {?} service
                 * @return {?}
                 */
                service => service.ngOnDestroy()));
            }
            finally {
                // Release all references.
                this.records.clear();
                this.onDestroy.clear();
                this.injectorDefTypes.clear();
            }
        }
        /**
         * @template T
         * @param {?} token
         * @param {?=} notFoundValue
         * @param {?=} flags
         * @return {?}
         */
        get(token, notFoundValue = THROW_IF_NOT_FOUND, flags = InjectFlags.Default) {
            this.assertNotDestroyed();
            // Set the injection context.
            /** @type {?} */
            const previousInjector = setCurrentInjector(this);
            try {
                // Check for the SkipSelf flag.
                if (!(flags & InjectFlags.SkipSelf)) {
                    // SkipSelf isn't set, check if the record belongs to this injector.
                    /** @type {?} */
                    let record = this.records.get(token);
                    if (record === undefined) {
                        // No record, but maybe the token is scoped to this injector. Look for an injectable
                        // def with a scope matching this injector.
                        /** @type {?} */
                        const def = couldBeInjectableType(token) && getInjectableDef(token);
                        if (def && this.injectableDefInScope(def)) {
                            // Found an injectable def and it's scoped to this injector. Pretend as if it was here
                            // all along.
                            record = makeRecord(injectableDefOrInjectorDefFactory(token), NOT_YET);
                        }
                        else {
                            record = null;
                        }
                        this.records.set(token, record);
                    }
                    // If a record was found, get the instance for it and return it.
                    if (record != null /* NOT null || undefined */) {
                        return this.hydrate(token, record);
                    }
                }
                // Select the next injector based on the Self flag - if self is set, the next injector is
                // the NullInjector, otherwise it's the parent.
                /** @type {?} */
                const nextInjector = !(flags & InjectFlags.Self) ? this.parent : getNullInjector();
                // Set the notFoundValue based on the Optional flag - if optional is set and notFoundValue
                // is undefined, the value is null, otherwise it's the notFoundValue.
                notFoundValue = (flags & InjectFlags.Optional) && notFoundValue === THROW_IF_NOT_FOUND ?
                    null :
                    notFoundValue;
                return nextInjector.get(token, notFoundValue);
            }
            catch (e) {
                if (e.name === 'NullInjectorError') {
                    /** @type {?} */
                    const path = e[NG_TEMP_TOKEN_PATH] = e[NG_TEMP_TOKEN_PATH] || [];
                    path.unshift(stringify(token));
                    if (previousInjector) {
                        // We still have a parent injector, keep throwing
                        throw e;
                    }
                    else {
                        // Format & throw the final error message when we don't have any previous injector
                        return catchInjectorError(e, token, 'R3InjectorError', this.source);
                    }
                }
                else {
                    throw e;
                }
            }
            finally {
                // Lastly, clean up the state by restoring the previous injector.
                setCurrentInjector(previousInjector);
            }
        }
        /**
         * \@internal
         * @return {?}
         */
        _resolveInjectorDefTypes() {
            this.injectorDefTypes.forEach((/**
             * @param {?} defType
             * @return {?}
             */
            defType => this.get(defType)));
        }
        /**
         * @return {?}
         */
        toString() {
            /** @type {?} */
            const tokens = (/** @type {?} */ ([]));
            /** @type {?} */
            const records = this.records;
            records.forEach((/**
             * @param {?} v
             * @param {?} token
             * @return {?}
             */
            (v, token) => tokens.push(stringify(token))));
            return `R3Injector[${tokens.join(', ')}]`;
        }
        /**
         * @private
         * @return {?}
         */
        assertNotDestroyed() {
            if (this._destroyed) {
                throw new Error('Injector has already been destroyed.');
            }
        }
        /**
         * Add an `InjectorType` or `InjectorTypeWithProviders` and all of its transitive providers
         * to this injector.
         *
         * If an `InjectorTypeWithProviders` that declares providers besides the type is specified,
         * the function will return "true" to indicate that the providers of the type definition need
         * to be processed. This allows us to process providers of injector types after all imports of
         * an injector definition are processed. (following View Engine semantics: see FW-1349)
         * @private
         * @param {?} defOrWrappedDef
         * @param {?} parents
         * @param {?} dedupStack
         * @return {?}
         */
        processInjectorType(defOrWrappedDef, parents, dedupStack) {
            defOrWrappedDef = resolveForwardRef(defOrWrappedDef);
            if (!defOrWrappedDef)
                return false;
            // Either the defOrWrappedDef is an InjectorType (with injector def) or an
            // InjectorDefTypeWithProviders (aka ModuleWithProviders). Detecting either is a megamorphic
            // read, so care is taken to only do the read once.
            // First attempt to read the injector def (`ɵinj`).
            /** @type {?} */
            let def = getInjectorDef(defOrWrappedDef);
            // If that's not present, then attempt to read ngModule from the InjectorDefTypeWithProviders.
            /** @type {?} */
            const ngModule = (def == null) && ((/** @type {?} */ (defOrWrappedDef))).ngModule || undefined;
            // Determine the InjectorType. In the case where `defOrWrappedDef` is an `InjectorType`,
            // then this is easy. In the case of an InjectorDefTypeWithProviders, then the definition type
            // is the `ngModule`.
            /** @type {?} */
            const defType = (ngModule === undefined) ? ((/** @type {?} */ (defOrWrappedDef))) : ngModule;
            // Check for circular dependencies.
            if (ngDevMode && parents.indexOf(defType) !== -1) {
                /** @type {?} */
                const defName = stringify(defType);
                throw new Error(`Circular dependency in DI detected for type ${defName}. Dependency path: ${parents.map((/**
             * @param {?} defType
             * @return {?}
             */
            defType => stringify(defType))).join(' > ')} > ${defName}.`);
            }
            // Check for multiple imports of the same module
            /** @type {?} */
            const isDuplicate = dedupStack.indexOf(defType) !== -1;
            // Finally, if defOrWrappedType was an `InjectorDefTypeWithProviders`, then the actual
            // `InjectorDef` is on its `ngModule`.
            if (ngModule !== undefined) {
                def = getInjectorDef(ngModule);
            }
            // If no definition was found, it might be from exports. Remove it.
            if (def == null) {
                return false;
            }
            // Add providers in the same way that @NgModule resolution did:
            // First, include providers from any imports.
            if (def.imports != null && !isDuplicate) {
                // Before processing defType's imports, add it to the set of parents. This way, if it ends
                // up deeply importing itself, this can be detected.
                ngDevMode && parents.push(defType);
                // Add it to the set of dedups. This way we can detect multiple imports of the same module
                dedupStack.push(defType);
                /** @type {?} */
                let importTypesWithProviders;
                try {
                    deepForEach(def.imports, (/**
                     * @param {?} imported
                     * @return {?}
                     */
                    imported => {
                        if (this.processInjectorType(imported, parents, dedupStack)) {
                            if (importTypesWithProviders === undefined)
                                importTypesWithProviders = [];
                            // If the processed import is an injector type with providers, we store it in the
                            // list of import types with providers, so that we can process those afterwards.
                            importTypesWithProviders.push(imported);
                        }
                    }));
                }
                finally {
                    // Remove it from the parents set when finished.
                    ngDevMode && parents.pop();
                }
                // Imports which are declared with providers (TypeWithProviders) need to be processed
                // after all imported modules are processed. This is similar to how View Engine
                // processes/merges module imports in the metadata resolver. See: FW-1349.
                if (importTypesWithProviders !== undefined) {
                    for (let i = 0; i < importTypesWithProviders.length; i++) {
                        const { ngModule, providers } = importTypesWithProviders[i];
                        deepForEach((/** @type {?} */ (providers)), (/**
                         * @param {?} provider
                         * @return {?}
                         */
                        provider => this.processProvider(provider, ngModule, providers || EMPTY_ARRAY$2)));
                    }
                }
            }
            // Track the InjectorType and add a provider for it. It's important that this is done after the
            // def's imports.
            this.injectorDefTypes.add(defType);
            this.records.set(defType, makeRecord(def.factory, NOT_YET));
            // Next, include providers listed on the definition itself.
            /** @type {?} */
            const defProviders = def.providers;
            if (defProviders != null && !isDuplicate) {
                /** @type {?} */
                const injectorType = (/** @type {?} */ (defOrWrappedDef));
                deepForEach(defProviders, (/**
                 * @param {?} provider
                 * @return {?}
                 */
                provider => this.processProvider(provider, injectorType, defProviders)));
            }
            return (ngModule !== undefined &&
                ((/** @type {?} */ (defOrWrappedDef))).providers !== undefined);
        }
        /**
         * Process a `SingleProvider` and add it.
         * @private
         * @param {?} provider
         * @param {?} ngModuleType
         * @param {?} providers
         * @return {?}
         */
        processProvider(provider, ngModuleType, providers) {
            // Determine the token from the provider. Either it's its own token, or has a {provide: ...}
            // property.
            provider = resolveForwardRef(provider);
            /** @type {?} */
            let token = isTypeProvider(provider) ? provider : resolveForwardRef(provider && provider.provide);
            // Construct a `Record` for the provider.
            /** @type {?} */
            const record = providerToRecord(provider, ngModuleType, providers);
            if (!isTypeProvider(provider) && provider.multi === true) {
                // If the provider indicates that it's a multi-provider, process it specially.
                // First check whether it's been defined already.
                /** @type {?} */
                let multiRecord = this.records.get(token);
                if (multiRecord) {
                    // It has. Throw a nice error if
                    if (multiRecord.multi === undefined) {
                        throwMixedMultiProviderError();
                    }
                }
                else {
                    multiRecord = makeRecord(undefined, NOT_YET, true);
                    multiRecord.factory = (/**
                     * @return {?}
                     */
                    () => injectArgs((/** @type {?} */ ((/** @type {?} */ (multiRecord)).multi))));
                    this.records.set(token, multiRecord);
                }
                token = provider;
                (/** @type {?} */ (multiRecord.multi)).push(provider);
            }
            else {
                /** @type {?} */
                const existing = this.records.get(token);
                if (existing && existing.multi !== undefined) {
                    throwMixedMultiProviderError();
                }
            }
            this.records.set(token, record);
        }
        /**
         * @private
         * @template T
         * @param {?} token
         * @param {?} record
         * @return {?}
         */
        hydrate(token, record) {
            if (record.value === CIRCULAR) {
                throwCyclicDependencyError(stringify(token));
            }
            else if (record.value === NOT_YET) {
                record.value = CIRCULAR;
                record.value = (/** @type {?} */ (record.factory))();
            }
            if (typeof record.value === 'object' && record.value && hasOnDestroy(record.value)) {
                this.onDestroy.add(record.value);
            }
            return (/** @type {?} */ (record.value));
        }
        /**
         * @private
         * @param {?} def
         * @return {?}
         */
        injectableDefInScope(def) {
            if (!def.providedIn) {
                return false;
            }
            else if (typeof def.providedIn === 'string') {
                return def.providedIn === 'any' || (def.providedIn === this.scope);
            }
            else {
                return this.injectorDefTypes.has(def.providedIn);
            }
        }
    }
    /**
     * @param {?} token
     * @return {?}
     */
    function injectableDefOrInjectorDefFactory(token) {
        // Most tokens will have an injectable def directly on them, which specifies a factory directly.
        /** @type {?} */
        const injectableDef = getInjectableDef(token);
        /** @type {?} */
        const factory = injectableDef !== null ? injectableDef.factory : getFactoryDef(token);
        if (factory !== null) {
            return factory;
        }
        // If the token is an NgModule, it's also injectable but the factory is on its injector def
        // (`ɵinj`)
        /** @type {?} */
        const injectorDef = getInjectorDef(token);
        if (injectorDef !== null) {
            return injectorDef.factory;
        }
        // InjectionTokens should have an injectable def (ɵprov) and thus should be handled above.
        // If it's missing that, it's an error.
        if (token instanceof InjectionToken) {
            throw new Error(`Token ${stringify(token)} is missing a ɵprov definition.`);
        }
        // Undecorated types can sometimes be created if they have no constructor arguments.
        if (token instanceof Function) {
            return getUndecoratedInjectableFactory(token);
        }
        // There was no way to resolve a factory for this token.
        throw new Error('unreachable');
    }
    /**
     * @param {?} token
     * @return {?}
     */
    function getUndecoratedInjectableFactory(token) {
        // If the token has parameters then it has dependencies that we cannot resolve implicitly.
        /** @type {?} */
        const paramLength = token.length;
        if (paramLength > 0) {
            /** @type {?} */
            const args = newArray(paramLength, '?');
            throw new Error(`Can't resolve all parameters for ${stringify(token)}: (${args.join(', ')}).`);
        }
        // The constructor function appears to have no parameters.
        // This might be because it inherits from a super-class. In which case, use an injectable
        // def from an ancestor if there is one.
        // Otherwise this really is a simple class with no dependencies, so return a factory that
        // just instantiates the zero-arg constructor.
        /** @type {?} */
        const inheritedInjectableDef = getInheritedInjectableDef(token);
        if (inheritedInjectableDef !== null) {
            return (/**
             * @return {?}
             */
            () => inheritedInjectableDef.factory((/** @type {?} */ (token))));
        }
        else {
            return (/**
             * @return {?}
             */
            () => new ((/** @type {?} */ (token)))());
        }
    }
    /**
     * @param {?} provider
     * @param {?} ngModuleType
     * @param {?} providers
     * @return {?}
     */
    function providerToRecord(provider, ngModuleType, providers) {
        if (isValueProvider(provider)) {
            return makeRecord(undefined, provider.useValue);
        }
        else {
            /** @type {?} */
            const factory = providerToFactory(provider, ngModuleType, providers);
            return makeRecord(factory, NOT_YET);
        }
    }
    /**
     * Converts a `SingleProvider` into a factory function.
     *
     * @param {?} provider provider to convert to factory
     * @param {?=} ngModuleType
     * @param {?=} providers
     * @return {?}
     */
    function providerToFactory(provider, ngModuleType, providers) {
        /** @type {?} */
        let factory = undefined;
        if (isTypeProvider(provider)) {
            /** @type {?} */
            const unwrappedProvider = resolveForwardRef(provider);
            return getFactoryDef(unwrappedProvider) || injectableDefOrInjectorDefFactory(unwrappedProvider);
        }
        else {
            if (isValueProvider(provider)) {
                factory = (/**
                 * @return {?}
                 */
                () => resolveForwardRef(provider.useValue));
            }
            else if (isFactoryProvider(provider)) {
                factory = (/**
                 * @return {?}
                 */
                () => provider.useFactory(...injectArgs(provider.deps || [])));
            }
            else if (isExistingProvider(provider)) {
                factory = (/**
                 * @return {?}
                 */
                () => ɵɵinject(resolveForwardRef(provider.useExisting)));
            }
            else {
                /** @type {?} */
                const classRef = resolveForwardRef(provider &&
                    (((/** @type {?} */ (provider))).useClass || provider.provide));
                if (!classRef) {
                    throwInvalidProviderError(ngModuleType, providers, provider);
                }
                if (hasDeps(provider)) {
                    factory = (/**
                     * @return {?}
                     */
                    () => new (classRef)(...injectArgs(provider.deps)));
                }
                else {
                    return getFactoryDef(classRef) || injectableDefOrInjectorDefFactory(classRef);
                }
            }
        }
        return factory;
    }
    /**
     * @template T
     * @param {?} factory
     * @param {?} value
     * @param {?=} multi
     * @return {?}
     */
    function makeRecord(factory, value, multi = false) {
        return {
            factory: factory,
            value: value,
            multi: multi ? [] : undefined,
        };
    }
    /**
     * @param {?} value
     * @return {?}
     */
    function isValueProvider(value) {
        return value !== null && typeof value == 'object' && USE_VALUE in value;
    }
    /**
     * @param {?} value
     * @return {?}
     */
    function isExistingProvider(value) {
        return !!(value && ((/** @type {?} */ (value))).useExisting);
    }
    /**
     * @param {?} value
     * @return {?}
     */
    function isFactoryProvider(value) {
        return !!(value && ((/** @type {?} */ (value))).useFactory);
    }
    /**
     * @param {?} value
     * @return {?}
     */
    function isTypeProvider(value) {
        return typeof value === 'function';
    }
    /**
     * @param {?} value
     * @return {?}
     */
    function isClassProvider(value) {
        return !!((/** @type {?} */ (value))).useClass;
    }
    /**
     * @param {?} value
     * @return {?}
     */
    function hasDeps(value) {
        return !!((/** @type {?} */ (value))).deps;
    }
    /**
     * @param {?} value
     * @return {?}
     */
    function hasOnDestroy(value) {
        return value !== null && typeof value === 'object' &&
            typeof ((/** @type {?} */ (value))).ngOnDestroy === 'function';
    }
    /**
     * @param {?} value
     * @return {?}
     */
    function couldBeInjectableType(value) {
        return (typeof value === 'function') ||
            (typeof value === 'object' && value instanceof InjectionToken);
    }
    /**
     * @param {?} providers
     * @param {?} parent
     * @param {?} name
     * @return {?}
     */
    function INJECTOR_IMPL__POST_R3__(providers, parent, name) {
        return createInjector({ name: name }, parent, providers, name);
    }
    /** @type {?} */
    const INJECTOR_IMPL = INJECTOR_IMPL__POST_R3__;
    /**
     * Concrete injectors implement this interface. Injectors are configured
     * with [providers](guide/glossary#provider) that associate
     * dependencies of various types with [injection tokens](guide/glossary#di-token).
     *
     * @see ["DI Providers"](guide/dependency-injection-providers).
     * @see `StaticProvider`
     *
     * \@usageNotes
     *
     *  The following example creates a service injector instance.
     *
     * {\@example core/di/ts/provider_spec.ts region='ConstructorProvider'}
     *
     * ### Usage example
     *
     * {\@example core/di/ts/injector_spec.ts region='Injector'}
     *
     * `Injector` returns itself when given `Injector` as a token:
     *
     * {\@example core/di/ts/injector_spec.ts region='injectInjector'}
     *
     * \@publicApi
     * @abstract
     */
    class Injector {
        /**
         * @param {?} options
         * @param {?=} parent
         * @return {?}
         */
        static create(options, parent) {
            if (Array.isArray(options)) {
                return INJECTOR_IMPL(options, parent, '');
            }
            else {
                return INJECTOR_IMPL(options.providers, options.parent, options.name || '');
            }
        }
    }
    Injector.THROW_IF_NOT_FOUND = THROW_IF_NOT_FOUND;
    Injector.NULL = new NullInjector();
    /** @nocollapse */
    Injector.ɵprov = ɵɵdefineInjectable({
        token: Injector,
        providedIn: (/** @type {?} */ ('any')),
        factory: (/**
         * @return {?}
         */
        () => ɵɵinject(INJECTOR)),
    });
    /**
     * \@internal
     * @nocollapse
     */
    Injector.__NG_ELEMENT_ID__ = -1;

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/di/reflective_errors.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @param {?} keys
     * @return {?}
     */
    function findFirstClosedCycle(keys) {
        /** @type {?} */
        const res = [];
        for (let i = 0; i < keys.length; ++i) {
            if (res.indexOf(keys[i]) > -1) {
                res.push(keys[i]);
                return res;
            }
            res.push(keys[i]);
        }
        return res;
    }
    /**
     * @param {?} keys
     * @return {?}
     */
    function constructResolvingPath(keys) {
        if (keys.length > 1) {
            /** @type {?} */
            const reversed = findFirstClosedCycle(keys.slice().reverse());
            /** @type {?} */
            const tokenStrs = reversed.map((/**
             * @param {?} k
             * @return {?}
             */
            k => stringify(k.token)));
            return ' (' + tokenStrs.join(' -> ') + ')';
        }
        return '';
    }
    /**
     * @param {?} injector
     * @param {?} key
     * @param {?} constructResolvingMessage
     * @param {?=} originalError
     * @return {?}
     */
    function injectionError(injector, key, constructResolvingMessage, originalError) {
        /** @type {?} */
        const keys = [key];
        /** @type {?} */
        const errMsg = constructResolvingMessage(keys);
        /** @type {?} */
        const error = (/** @type {?} */ ((originalError ? wrappedError(errMsg, originalError) : Error(errMsg))));
        error.addKey = addKey;
        error.keys = keys;
        error.injectors = [injector];
        error.constructResolvingMessage = constructResolvingMessage;
        ((/** @type {?} */ (error)))[ERROR_ORIGINAL_ERROR] = originalError;
        return error;
    }
    /**
     * @this {?}
     * @param {?} injector
     * @param {?} key
     * @return {?}
     */
    function addKey(injector, key) {
        this.injectors.push(injector);
        this.keys.push(key);
        // Note: This updated message won't be reflected in the `.stack` property
        this.message = this.constructResolvingMessage(this.keys);
    }
    /**
     * Thrown when trying to retrieve a dependency by key from {\@link Injector}, but the
     * {\@link Injector} does not have a {\@link Provider} for the given key.
     *
     * \@usageNotes
     * ### Example
     *
     * ```typescript
     * class A {
     *   constructor(b:B) {}
     * }
     *
     * expect(() => Injector.resolveAndCreate([A])).toThrowError();
     * ```
     * @param {?} injector
     * @param {?} key
     * @return {?}
     */
    function noProviderError(injector, key) {
        return injectionError(injector, key, (/**
         * @param {?} keys
         * @return {?}
         */
        function (keys) {
            /** @type {?} */
            const first = stringify(keys[0].token);
            return `No provider for ${first}!${constructResolvingPath(keys)}`;
        }));
    }
    /**
     * Thrown when dependencies form a cycle.
     *
     * \@usageNotes
     * ### Example
     *
     * ```typescript
     * var injector = Injector.resolveAndCreate([
     *   {provide: "one", useFactory: (two) => "two", deps: [[new Inject("two")]]},
     *   {provide: "two", useFactory: (one) => "one", deps: [[new Inject("one")]]}
     * ]);
     *
     * expect(() => injector.get("one")).toThrowError();
     * ```
     *
     * Retrieving `A` or `B` throws a `CyclicDependencyError` as the graph above cannot be constructed.
     * @param {?} injector
     * @param {?} key
     * @return {?}
     */
    function cyclicDependencyError(injector, key) {
        return injectionError(injector, key, (/**
         * @param {?} keys
         * @return {?}
         */
        function (keys) {
            return `Cannot instantiate cyclic dependency!${constructResolvingPath(keys)}`;
        }));
    }
    /**
     * Thrown when a constructing type returns with an Error.
     *
     * The `InstantiationError` class contains the original error plus the dependency graph which caused
     * this object to be instantiated.
     *
     * \@usageNotes
     * ### Example
     *
     * ```typescript
     * class A {
     *   constructor() {
     *     throw new Error('message');
     *   }
     * }
     *
     * var injector = Injector.resolveAndCreate([A]);
     * try {
     *   injector.get(A);
     * } catch (e) {
     *   expect(e instanceof InstantiationError).toBe(true);
     *   expect(e.originalException.message).toEqual("message");
     *   expect(e.originalStack).toBeDefined();
     * }
     * ```
     * @param {?} injector
     * @param {?} originalException
     * @param {?} originalStack
     * @param {?} key
     * @return {?}
     */
    function instantiationError(injector, originalException, originalStack, key) {
        return injectionError(injector, key, (/**
         * @param {?} keys
         * @return {?}
         */
        function (keys) {
            /** @type {?} */
            const first = stringify(keys[0].token);
            return `${originalException.message}: Error during instantiation of ${first}!${constructResolvingPath(keys)}.`;
        }), originalException);
    }
    /**
     * Thrown when an object other then {\@link Provider} (or `Type`) is passed to {\@link Injector}
     * creation.
     *
     * \@usageNotes
     * ### Example
     *
     * ```typescript
     * expect(() => Injector.resolveAndCreate(["not a type"])).toThrowError();
     * ```
     * @param {?} provider
     * @return {?}
     */
    function invalidProviderError(provider) {
        return Error(`Invalid provider - only instances of Provider and Type are allowed, got: ${provider}`);
    }
    /**
     * Thrown when the class has no annotation information.
     *
     * Lack of annotation information prevents the {\@link Injector} from determining which dependencies
     * need to be injected into the constructor.
     *
     * \@usageNotes
     * ### Example
     *
     * ```typescript
     * class A {
     *   constructor(b) {}
     * }
     *
     * expect(() => Injector.resolveAndCreate([A])).toThrowError();
     * ```
     *
     * This error is also thrown when the class not marked with {\@link Injectable} has parameter types.
     *
     * ```typescript
     * class B {}
     *
     * class A {
     *   constructor(b:B) {} // no information about the parameter types of A is available at runtime.
     * }
     *
     * expect(() => Injector.resolveAndCreate([A,B])).toThrowError();
     * ```
     *
     * @param {?} typeOrFunc
     * @param {?} params
     * @return {?}
     */
    function noAnnotationError(typeOrFunc, params) {
        /** @type {?} */
        const signature = [];
        for (let i = 0, ii = params.length; i < ii; i++) {
            /** @type {?} */
            const parameter = params[i];
            if (!parameter || parameter.length == 0) {
                signature.push('?');
            }
            else {
                signature.push(parameter.map(stringify).join(' '));
            }
        }
        return Error('Cannot resolve all parameters for \'' + stringify(typeOrFunc) + '\'(' +
            signature.join(', ') + '). ' +
            'Make sure that all the parameters are decorated with Inject or have valid type annotations and that \'' +
            stringify(typeOrFunc) + '\' is decorated with Injectable.');
    }
    /**
     * Thrown when getting an object by index.
     *
     * \@usageNotes
     * ### Example
     *
     * ```typescript
     * class A {}
     *
     * var injector = Injector.resolveAndCreate([A]);
     *
     * expect(() => injector.getAt(100)).toThrowError();
     * ```
     *
     * @param {?} index
     * @return {?}
     */
    function outOfBoundsError(index) {
        return Error(`Index ${index} is out-of-bounds.`);
    }
    // TODO: add a working example after alpha38 is released
    /**
     * Thrown when a multi provider and a regular provider are bound to the same token.
     *
     * \@usageNotes
     * ### Example
     *
     * ```typescript
     * expect(() => Injector.resolveAndCreate([
     *   { provide: "Strings", useValue: "string1", multi: true},
     *   { provide: "Strings", useValue: "string2", multi: false}
     * ])).toThrowError();
     * ```
     * @param {?} provider1
     * @param {?} provider2
     * @return {?}
     */
    function mixingMultiProvidersWithRegularProvidersError(provider1, provider2) {
        return Error(`Cannot mix multi providers and regular providers, got: ${provider1} ${provider2}`);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/di/reflective_key.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * A unique object used for retrieving items from the {\@link ReflectiveInjector}.
     *
     * Keys have:
     * - a system-wide unique `id`.
     * - a `token`.
     *
     * `Key` is used internally by {\@link ReflectiveInjector} because its system-wide unique `id` allows
     * the
     * injector to store created objects in a more efficient way.
     *
     * `Key` should not be created directly. {\@link ReflectiveInjector} creates keys automatically when
     * resolving
     * providers.
     *
     * @deprecated No replacement
     * \@publicApi
     */
    class ReflectiveKey {
        /**
         * Private
         * @param {?} token
         * @param {?} id
         */
        constructor(token, id) {
            this.token = token;
            this.id = id;
            if (!token) {
                throw new Error('Token must be defined!');
            }
            this.displayName = stringify(this.token);
        }
        /**
         * Retrieves a `Key` for a token.
         * @param {?} token
         * @return {?}
         */
        static get(token) {
            return _globalKeyRegistry.get(resolveForwardRef(token));
        }
        /**
         * @return {?} the number of keys registered in the system.
         */
        static get numberOfKeys() {
            return _globalKeyRegistry.numberOfKeys;
        }
    }
    class KeyRegistry {
        constructor() {
            this._allKeys = new Map();
        }
        /**
         * @param {?} token
         * @return {?}
         */
        get(token) {
            if (token instanceof ReflectiveKey)
                return token;
            if (this._allKeys.has(token)) {
                return (/** @type {?} */ (this._allKeys.get(token)));
            }
            /** @type {?} */
            const newKey = new ReflectiveKey(token, ReflectiveKey.numberOfKeys);
            this._allKeys.set(token, newKey);
            return newKey;
        }
        /**
         * @return {?}
         */
        get numberOfKeys() {
            return this._allKeys.size;
        }
    }
    /** @type {?} */
    const _globalKeyRegistry = new KeyRegistry();

    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * Provides access to reflection data about symbols. Used internally by Angular
     * to power dependency injection and compilation.
     */
    class Reflector {
        constructor(reflectionCapabilities) {
            this.reflectionCapabilities = reflectionCapabilities;
        }
        updateCapabilities(caps) {
            this.reflectionCapabilities = caps;
        }
        factory(type) {
            return this.reflectionCapabilities.factory(type);
        }
        parameters(typeOrFunc) {
            return this.reflectionCapabilities.parameters(typeOrFunc);
        }
        annotations(typeOrFunc) {
            return this.reflectionCapabilities.annotations(typeOrFunc);
        }
        propMetadata(typeOrFunc) {
            return this.reflectionCapabilities.propMetadata(typeOrFunc);
        }
        hasLifecycleHook(type, lcProperty) {
            return this.reflectionCapabilities.hasLifecycleHook(type, lcProperty);
        }
        getter(name) {
            return this.reflectionCapabilities.getter(name);
        }
        setter(name) {
            return this.reflectionCapabilities.setter(name);
        }
        method(name) {
            return this.reflectionCapabilities.method(name);
        }
        importUri(type) {
            return this.reflectionCapabilities.importUri(type);
        }
        resourceUri(type) {
            return this.reflectionCapabilities.resourceUri(type);
        }
        resolveIdentifier(name, moduleUrl, members, runtime) {
            return this.reflectionCapabilities.resolveIdentifier(name, moduleUrl, members, runtime);
        }
        resolveEnum(identifier, name) {
            return this.reflectionCapabilities.resolveEnum(identifier, name);
        }
    }

    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * The {@link Reflector} used internally in Angular to access metadata
     * about symbols.
     */
    const reflector = new Reflector(new ReflectionCapabilities());
    /**
     * `Dependency` is used by the framework to extend DI.
     * This is internal to Angular and should not be used directly.
     */
    class ReflectiveDependency {
        /**
         * @param {?} key
         * @param {?} optional
         * @param {?} visibility
         */
        constructor(key, optional, visibility) {
            this.key = key;
            this.optional = optional;
            this.visibility = visibility;
        }
        /**
         * @param {?} key
         * @return {?}
         */
        static fromKey(key) {
            return new ReflectiveDependency(key, false, null);
        }
    }
    /** @type {?} */
    const _EMPTY_LIST = [];
    class ResolvedReflectiveProvider_ {
        /**
         * @param {?} key
         * @param {?} resolvedFactories
         * @param {?} multiProvider
         */
        constructor(key, resolvedFactories, multiProvider) {
            this.key = key;
            this.resolvedFactories = resolvedFactories;
            this.multiProvider = multiProvider;
            this.resolvedFactory = this.resolvedFactories[0];
        }
    }
    /**
     * An internal resolved representation of a factory function created by resolving `Provider`.
     * \@publicApi
     */
    class ResolvedReflectiveFactory {
        /**
         * @param {?} factory
         * @param {?} dependencies
         */
        constructor(factory, dependencies) {
            this.factory = factory;
            this.dependencies = dependencies;
        }
    }
    /**
     * Resolve a single provider.
     * @param {?} provider
     * @return {?}
     */
    function resolveReflectiveFactory(provider) {
        /** @type {?} */
        let factoryFn;
        /** @type {?} */
        let resolvedDeps;
        if (provider.useClass) {
            /** @type {?} */
            const useClass = resolveForwardRef(provider.useClass);
            factoryFn = reflector.factory(useClass);
            resolvedDeps = _dependenciesFor(useClass);
        }
        else if (provider.useExisting) {
            factoryFn = (/**
             * @param {?} aliasInstance
             * @return {?}
             */
            (aliasInstance) => aliasInstance);
            resolvedDeps = [ReflectiveDependency.fromKey(ReflectiveKey.get(provider.useExisting))];
        }
        else if (provider.useFactory) {
            factoryFn = provider.useFactory;
            resolvedDeps = constructDependencies(provider.useFactory, provider.deps);
        }
        else {
            factoryFn = (/**
             * @return {?}
             */
            () => provider.useValue);
            resolvedDeps = _EMPTY_LIST;
        }
        return new ResolvedReflectiveFactory(factoryFn, resolvedDeps);
    }
    /**
     * Converts the `Provider` into `ResolvedProvider`.
     *
     * `Injector` internally only uses `ResolvedProvider`, `Provider` contains convenience provider
     * syntax.
     * @param {?} provider
     * @return {?}
     */
    function resolveReflectiveProvider(provider) {
        return new ResolvedReflectiveProvider_(ReflectiveKey.get(provider.provide), [resolveReflectiveFactory(provider)], provider.multi || false);
    }
    /**
     * Resolve a list of Providers.
     * @param {?} providers
     * @return {?}
     */
    function resolveReflectiveProviders(providers) {
        /** @type {?} */
        const normalized = _normalizeProviders(providers, []);
        /** @type {?} */
        const resolved = normalized.map(resolveReflectiveProvider);
        /** @type {?} */
        const resolvedProviderMap = mergeResolvedReflectiveProviders(resolved, new Map());
        return Array.from(resolvedProviderMap.values());
    }
    /**
     * Merges a list of ResolvedProviders into a list where each key is contained exactly once and
     * multi providers have been merged.
     * @param {?} providers
     * @param {?} normalizedProvidersMap
     * @return {?}
     */
    function mergeResolvedReflectiveProviders(providers, normalizedProvidersMap) {
        for (let i = 0; i < providers.length; i++) {
            /** @type {?} */
            const provider = providers[i];
            /** @type {?} */
            const existing = normalizedProvidersMap.get(provider.key.id);
            if (existing) {
                if (provider.multiProvider !== existing.multiProvider) {
                    throw mixingMultiProvidersWithRegularProvidersError(existing, provider);
                }
                if (provider.multiProvider) {
                    for (let j = 0; j < provider.resolvedFactories.length; j++) {
                        existing.resolvedFactories.push(provider.resolvedFactories[j]);
                    }
                }
                else {
                    normalizedProvidersMap.set(provider.key.id, provider);
                }
            }
            else {
                /** @type {?} */
                let resolvedProvider;
                if (provider.multiProvider) {
                    resolvedProvider = new ResolvedReflectiveProvider_(provider.key, provider.resolvedFactories.slice(), provider.multiProvider);
                }
                else {
                    resolvedProvider = provider;
                }
                normalizedProvidersMap.set(provider.key.id, resolvedProvider);
            }
        }
        return normalizedProvidersMap;
    }
    /**
     * @param {?} providers
     * @param {?} res
     * @return {?}
     */
    function _normalizeProviders(providers, res) {
        providers.forEach((/**
         * @param {?} b
         * @return {?}
         */
        b => {
            if (b instanceof Type) {
                res.push((/** @type {?} */ ({ provide: b, useClass: b })));
            }
            else if (b && typeof b == 'object' && ((/** @type {?} */ (b))).provide !== undefined) {
                res.push((/** @type {?} */ (b)));
            }
            else if (Array.isArray(b)) {
                _normalizeProviders(b, res);
            }
            else {
                throw invalidProviderError(b);
            }
        }));
        return res;
    }
    /**
     * @param {?} typeOrFunc
     * @param {?=} dependencies
     * @return {?}
     */
    function constructDependencies(typeOrFunc, dependencies) {
        if (!dependencies) {
            return _dependenciesFor(typeOrFunc);
        }
        else {
            /** @type {?} */
            const params = dependencies.map((/**
             * @param {?} t
             * @return {?}
             */
            t => [t]));
            return dependencies.map((/**
             * @param {?} t
             * @return {?}
             */
            t => _extractToken(typeOrFunc, t, params)));
        }
    }
    /**
     * @param {?} typeOrFunc
     * @return {?}
     */
    function _dependenciesFor(typeOrFunc) {
        /** @type {?} */
        const params = reflector.parameters(typeOrFunc);
        if (!params)
            return [];
        if (params.some((/**
         * @param {?} p
         * @return {?}
         */
        p => p == null))) {
            throw noAnnotationError(typeOrFunc, params);
        }
        return params.map((/**
         * @param {?} p
         * @return {?}
         */
        p => _extractToken(typeOrFunc, p, params)));
    }
    /**
     * @param {?} typeOrFunc
     * @param {?} metadata
     * @param {?} params
     * @return {?}
     */
    function _extractToken(typeOrFunc, metadata, params) {
        /** @type {?} */
        let token = null;
        /** @type {?} */
        let optional = false;
        if (!Array.isArray(metadata)) {
            if (metadata instanceof Inject) {
                return _createDependency(metadata.token, optional, null);
            }
            else {
                return _createDependency(metadata, optional, null);
            }
        }
        /** @type {?} */
        let visibility = null;
        for (let i = 0; i < metadata.length; ++i) {
            /** @type {?} */
            const paramMetadata = metadata[i];
            if (paramMetadata instanceof Type) {
                token = paramMetadata;
            }
            else if (paramMetadata instanceof Inject) {
                token = paramMetadata.token;
            }
            else if (paramMetadata instanceof Optional) {
                optional = true;
            }
            else if (paramMetadata instanceof Self || paramMetadata instanceof SkipSelf) {
                visibility = paramMetadata;
            }
            else if (paramMetadata instanceof InjectionToken) {
                token = paramMetadata;
            }
        }
        token = resolveForwardRef(token);
        if (token != null) {
            return _createDependency(token, optional, visibility);
        }
        else {
            throw noAnnotationError(typeOrFunc, params);
        }
    }
    /**
     * @param {?} token
     * @param {?} optional
     * @param {?} visibility
     * @return {?}
     */
    function _createDependency(token, optional, visibility) {
        return new ReflectiveDependency(ReflectiveKey.get(token), optional, visibility);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/di/reflective_injector.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    // Threshold for the dynamic version
    /** @type {?} */
    const UNDEFINED = {};
    /**
     * A ReflectiveDependency injection container used for instantiating objects and resolving
     * dependencies.
     *
     * An `Injector` is a replacement for a `new` operator, which can automatically resolve the
     * constructor dependencies.
     *
     * In typical use, application code asks for the dependencies in the constructor and they are
     * resolved by the `Injector`.
     *
     * \@usageNotes
     * ### Example
     *
     * The following example creates an `Injector` configured to create `Engine` and `Car`.
     *
     * ```typescript
     * \@Injectable()
     * class Engine {
     * }
     *
     * \@Injectable()
     * class Car {
     *   constructor(public engine:Engine) {}
     * }
     *
     * var injector = ReflectiveInjector.resolveAndCreate([Car, Engine]);
     * var car = injector.get(Car);
     * expect(car instanceof Car).toBe(true);
     * expect(car.engine instanceof Engine).toBe(true);
     * ```
     *
     * Notice, we don't use the `new` operator because we explicitly want to have the `Injector`
     * resolve all of the object's dependencies automatically.
     *
     * @deprecated from v5 - slow and brings in a lot of code, Use `Injector.create` instead.
     * \@publicApi
     * @abstract
     */
    class ReflectiveInjector {
        /**
         * Turns an array of provider definitions into an array of resolved providers.
         *
         * A resolution is a process of flattening multiple nested arrays and converting individual
         * providers into an array of `ResolvedReflectiveProvider`s.
         *
         * \@usageNotes
         * ### Example
         *
         * ```typescript
         * \@Injectable()
         * class Engine {
         * }
         *  /
         * class Car {
         *   constructor(public engine:Engine) {}
         * }
         *
         * var providers = ReflectiveInjector.resolve([Car, [[Engine]]]);
         *
         * expect(providers.length).toEqual(2);
         *
         * expect(providers[0] instanceof ResolvedReflectiveProvider).toBe(true);
         * expect(providers[0].key.displayName).toBe("Car");
         * expect(providers[0].dependencies.length).toEqual(1);
         * expect(providers[0].factory).toBeDefined();
         *
         * expect(providers[1].key.displayName).toBe("Engine");
         * });
         * ```
         *
         * @param {?} providers
         * @return {?}
         */
        static resolve(providers) {
            return resolveReflectiveProviders(providers);
        }
        /**
         * Resolves an array of providers and creates an injector from those providers.
         *
         * The passed-in providers can be an array of `Type`, `Provider`,
         * or a recursive array of more providers.
         *
         * \@usageNotes
         * ### Example
         *
         * ```typescript
         * \@Injectable()
         * class Engine {
         * }
         *  /
         * class Car {
         *   constructor(public engine:Engine) {}
         * }
         *
         * var injector = ReflectiveInjector.resolveAndCreate([Car, Engine]);
         * expect(injector.get(Car) instanceof Car).toBe(true);
         * ```
         * @param {?} providers
         * @param {?=} parent
         * @return {?}
         */
        static resolveAndCreate(providers, parent) {
            /** @type {?} */
            const ResolvedReflectiveProviders = ReflectiveInjector.resolve(providers);
            return ReflectiveInjector.fromResolvedProviders(ResolvedReflectiveProviders, parent);
        }
        /**
         * Creates an injector from previously resolved providers.
         *
         * This API is the recommended way to construct injectors in performance-sensitive parts.
         *
         * \@usageNotes
         * ### Example
         *
         * ```typescript
         * \@Injectable()
         * class Engine {
         * }
         *  /
         * class Car {
         *   constructor(public engine:Engine) {}
         * }
         *
         * var providers = ReflectiveInjector.resolve([Car, Engine]);
         * var injector = ReflectiveInjector.fromResolvedProviders(providers);
         * expect(injector.get(Car) instanceof Car).toBe(true);
         * ```
         * @param {?} providers
         * @param {?=} parent
         * @return {?}
         */
        static fromResolvedProviders(providers, parent) {
            return new ReflectiveInjector_(providers, parent);
        }
    }
    class ReflectiveInjector_ {
        /**
         * Private
         * @param {?} _providers
         * @param {?=} _parent
         */
        constructor(_providers, _parent) {
            /**
             * \@internal
             */
            this._constructionCounter = 0;
            this._providers = _providers;
            this.parent = _parent || null;
            /** @type {?} */
            const len = _providers.length;
            this.keyIds = [];
            this.objs = [];
            for (let i = 0; i < len; i++) {
                this.keyIds[i] = _providers[i].key.id;
                this.objs[i] = UNDEFINED;
            }
        }
        /**
         * @param {?} token
         * @param {?=} notFoundValue
         * @return {?}
         */
        get(token, notFoundValue = THROW_IF_NOT_FOUND) {
            return this._getByKey(ReflectiveKey.get(token), null, notFoundValue);
        }
        /**
         * @param {?} providers
         * @return {?}
         */
        resolveAndCreateChild(providers) {
            /** @type {?} */
            const ResolvedReflectiveProviders = ReflectiveInjector.resolve(providers);
            return this.createChildFromResolved(ResolvedReflectiveProviders);
        }
        /**
         * @param {?} providers
         * @return {?}
         */
        createChildFromResolved(providers) {
            /** @type {?} */
            const inj = new ReflectiveInjector_(providers);
            ((/** @type {?} */ (inj))).parent = this;
            return inj;
        }
        /**
         * @param {?} provider
         * @return {?}
         */
        resolveAndInstantiate(provider) {
            return this.instantiateResolved(ReflectiveInjector.resolve([provider])[0]);
        }
        /**
         * @param {?} provider
         * @return {?}
         */
        instantiateResolved(provider) {
            return this._instantiateProvider(provider);
        }
        /**
         * @param {?} index
         * @return {?}
         */
        getProviderAtIndex(index) {
            if (index < 0 || index >= this._providers.length) {
                throw outOfBoundsError(index);
            }
            return this._providers[index];
        }
        /**
         * \@internal
         * @param {?} provider
         * @return {?}
         */
        _new(provider) {
            if (this._constructionCounter++ > this._getMaxNumberOfObjects()) {
                throw cyclicDependencyError(this, provider.key);
            }
            return this._instantiateProvider(provider);
        }
        /**
         * @private
         * @return {?}
         */
        _getMaxNumberOfObjects() {
            return this.objs.length;
        }
        /**
         * @private
         * @param {?} provider
         * @return {?}
         */
        _instantiateProvider(provider) {
            if (provider.multiProvider) {
                /** @type {?} */
                const res = [];
                for (let i = 0; i < provider.resolvedFactories.length; ++i) {
                    res[i] = this._instantiate(provider, provider.resolvedFactories[i]);
                }
                return res;
            }
            else {
                return this._instantiate(provider, provider.resolvedFactories[0]);
            }
        }
        /**
         * @private
         * @param {?} provider
         * @param {?} ResolvedReflectiveFactory
         * @return {?}
         */
        _instantiate(provider, ResolvedReflectiveFactory) {
            /** @type {?} */
            const factory = ResolvedReflectiveFactory.factory;
            /** @type {?} */
            let deps;
            try {
                deps =
                    ResolvedReflectiveFactory.dependencies.map((/**
                     * @param {?} dep
                     * @return {?}
                     */
                    dep => this._getByReflectiveDependency(dep)));
            }
            catch (e) {
                if (e.addKey) {
                    e.addKey(this, provider.key);
                }
                throw e;
            }
            /** @type {?} */
            let obj;
            try {
                obj = factory(...deps);
            }
            catch (e) {
                throw instantiationError(this, e, e.stack, provider.key);
            }
            return obj;
        }
        /**
         * @private
         * @param {?} dep
         * @return {?}
         */
        _getByReflectiveDependency(dep) {
            return this._getByKey(dep.key, dep.visibility, dep.optional ? null : THROW_IF_NOT_FOUND);
        }
        /**
         * @private
         * @param {?} key
         * @param {?} visibility
         * @param {?} notFoundValue
         * @return {?}
         */
        _getByKey(key, visibility, notFoundValue) {
            if (key === ReflectiveInjector_.INJECTOR_KEY) {
                return this;
            }
            if (visibility instanceof Self) {
                return this._getByKeySelf(key, notFoundValue);
            }
            else {
                return this._getByKeyDefault(key, notFoundValue, visibility);
            }
        }
        /**
         * @private
         * @param {?} keyId
         * @return {?}
         */
        _getObjByKeyId(keyId) {
            for (let i = 0; i < this.keyIds.length; i++) {
                if (this.keyIds[i] === keyId) {
                    if (this.objs[i] === UNDEFINED) {
                        this.objs[i] = this._new(this._providers[i]);
                    }
                    return this.objs[i];
                }
            }
            return UNDEFINED;
        }
        /**
         * \@internal
         * @param {?} key
         * @param {?} notFoundValue
         * @return {?}
         */
        _throwOrNull(key, notFoundValue) {
            if (notFoundValue !== THROW_IF_NOT_FOUND) {
                return notFoundValue;
            }
            else {
                throw noProviderError(this, key);
            }
        }
        /**
         * \@internal
         * @param {?} key
         * @param {?} notFoundValue
         * @return {?}
         */
        _getByKeySelf(key, notFoundValue) {
            /** @type {?} */
            const obj = this._getObjByKeyId(key.id);
            return (obj !== UNDEFINED) ? obj : this._throwOrNull(key, notFoundValue);
        }
        /**
         * \@internal
         * @param {?} key
         * @param {?} notFoundValue
         * @param {?} visibility
         * @return {?}
         */
        _getByKeyDefault(key, notFoundValue, visibility) {
            /** @type {?} */
            let inj;
            if (visibility instanceof SkipSelf) {
                inj = this.parent;
            }
            else {
                inj = this;
            }
            while (inj instanceof ReflectiveInjector_) {
                /** @type {?} */
                const inj_ = (/** @type {?} */ (inj));
                /** @type {?} */
                const obj = inj_._getObjByKeyId(key.id);
                if (obj !== UNDEFINED)
                    return obj;
                inj = inj_.parent;
            }
            if (inj !== null) {
                return inj.get(key.token, notFoundValue);
            }
            else {
                return this._throwOrNull(key, notFoundValue);
            }
        }
        /**
         * @return {?}
         */
        get displayName() {
            /** @type {?} */
            const providers = _mapProviders(this, (/**
             * @param {?} b
             * @return {?}
             */
            (b) => ' "' + b.key.displayName + '" '))
                .join(', ');
            return `ReflectiveInjector(providers: [${providers}])`;
        }
        /**
         * @return {?}
         */
        toString() {
            return this.displayName;
        }
    }
    ReflectiveInjector_.INJECTOR_KEY = ReflectiveKey.get(Injector);
    /**
     * @param {?} injector
     * @param {?} fn
     * @return {?}
     */
    function _mapProviders(injector, fn) {
        /** @type {?} */
        const res = [];
        for (let i = 0; i < injector._providers.length; ++i) {
            res[i] = fn(injector.getProviderAtIndex(i));
        }
        return res;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/di/index.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/di.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/metadata/di.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * A DI token that you can use to create a virtual [provider](guide/glossary#provider)
     * that will populate the `entryComponents` field of components and NgModules
     * based on its `useValue` property value.
     * All components that are referenced in the `useValue` value (either directly
     * or in a nested array or map) are added to the `entryComponents` property.
     *
     * \@usageNotes
     *
     * The following example shows how the router can populate the `entryComponents`
     * field of an NgModule based on a router configuration that refers
     * to components.
     *
     * ```typescript
     * // helper function inside the router
     * function provideRoutes(routes) {
     *   return [
     *     {provide: ROUTES, useValue: routes},
     *     {provide: ANALYZE_FOR_ENTRY_COMPONENTS, useValue: routes, multi: true}
     *   ];
     * }
     *
     * // user code
     * let routes = [
     *   {path: '/root', component: RootComp},
     *   {path: '/teams', component: TeamsComp}
     * ];
     *
     * \@NgModule({
     *   providers: [provideRoutes(routes)]
     * })
     * class ModuleWithRoutes {}
     * ```
     *
     * \@publicApi
     * @deprecated Since 9.0.0. With Ivy, this property is no longer necessary.
     * @type {?}
     */
    const ANALYZE_FOR_ENTRY_COMPONENTS = new InjectionToken('AnalyzeForEntryComponents');
    // WARNING: interface has both a type and a value, skipping emit
    /**
     * Base class for query metadata.
     *
     * @see `ContentChildren`.
     * @see `ContentChild`.
     * @see `ViewChildren`.
     * @see `ViewChild`.
     *
     * \@publicApi
     * @abstract
     */
    class Query {
    }
    const ɵ0$a = /**
     * @param {?=} selector
     * @param {?=} data
     * @return {?}
     */
    (selector, data = {}) => (Object.assign({ selector, first: false, isViewQuery: false, descendants: false }, data));
    /**
     * ContentChildren decorator and metadata.
     *
     *
     * \@Annotation
     * \@publicApi
     * @type {?}
     */
    const ContentChildren = makePropDecorator('ContentChildren', (ɵ0$a), Query);
    const ɵ1$2 = /**
     * @param {?=} selector
     * @param {?=} data
     * @return {?}
     */
    (selector, data = {}) => (Object.assign({ selector, first: true, isViewQuery: false, descendants: true }, data));
    /**
     * ContentChild decorator and metadata.
     *
     *
     * \@Annotation
     *
     * \@publicApi
     * @type {?}
     */
    const ContentChild = makePropDecorator('ContentChild', (ɵ1$2), Query);
    const ɵ2 = /**
     * @param {?=} selector
     * @param {?=} data
     * @return {?}
     */
    (selector, data = {}) => (Object.assign({ selector, first: false, isViewQuery: true, descendants: true }, data));
    /**
     * ViewChildren decorator and metadata.
     *
     * \@Annotation
     * \@publicApi
     * @type {?}
     */
    const ViewChildren = makePropDecorator('ViewChildren', (ɵ2), Query);
    const ɵ3 = /**
     * @param {?} selector
     * @param {?} data
     * @return {?}
     */
    (selector, data) => (Object.assign({ selector, first: true, isViewQuery: true, descendants: true }, data));
    /**
     * ViewChild decorator and metadata.
     *
     * \@Annotation
     * \@publicApi
     * @type {?}
     */
    const ViewChild = makePropDecorator('ViewChild', (ɵ3), Query);

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/metadata/resource_loading.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * Used to resolve resource URLs on `\@Component` when used with JIT compilation.
     *
     * Example:
     * ```
     * \@Component({
     *   selector: 'my-comp',
     *   templateUrl: 'my-comp.html', // This requires asynchronous resolution
     * })
     * class MyComponent{
     * }
     *
     * // Calling `renderComponent` will fail because `renderComponent` is a synchronous process
     * // and `MyComponent`'s `\@Component.templateUrl` needs to be resolved asynchronously.
     *
     * // Calling `resolveComponentResources()` will resolve `\@Component.templateUrl` into
     * // `\@Component.template`, which allows `renderComponent` to proceed in a synchronous manner.
     *
     * // Use browser's `fetch()` function as the default resource resolution strategy.
     * resolveComponentResources(fetch).then(() => {
     *   // After resolution all URLs have been converted into `template` strings.
     *   renderComponent(MyComponent);
     * });
     *
     * ```
     *
     * NOTE: In AOT the resolution happens during compilation, and so there should be no need
     * to call this method outside JIT mode.
     *
     * @param {?} resourceResolver a function which is responsible for returning a `Promise` to the
     * contents of the resolved URL. Browser's `fetch()` method is a good default implementation.
     * @return {?}
     */
    function resolveComponentResources(resourceResolver) {
        // Store all promises which are fetching the resources.
        /** @type {?} */
        const componentResolved = [];
        // Cache so that we don't fetch the same resource more than once.
        /** @type {?} */
        const urlMap = new Map();
        /**
         * @param {?} url
         * @return {?}
         */
        function cachedResourceResolve(url) {
            /** @type {?} */
            let promise = urlMap.get(url);
            if (!promise) {
                /** @type {?} */
                const resp = resourceResolver(url);
                urlMap.set(url, promise = resp.then(unwrapResponse));
            }
            return promise;
        }
        componentResourceResolutionQueue.forEach((/**
         * @param {?} component
         * @param {?} type
         * @return {?}
         */
        (component, type) => {
            /** @type {?} */
            const promises = [];
            if (component.templateUrl) {
                promises.push(cachedResourceResolve(component.templateUrl).then((/**
                 * @param {?} template
                 * @return {?}
                 */
                (template) => {
                    component.template = template;
                })));
            }
            /** @type {?} */
            const styleUrls = component.styleUrls;
            /** @type {?} */
            const styles = component.styles || (component.styles = []);
            /** @type {?} */
            const styleOffset = component.styles.length;
            styleUrls && styleUrls.forEach((/**
             * @param {?} styleUrl
             * @param {?} index
             * @return {?}
             */
            (styleUrl, index) => {
                styles.push(''); // pre-allocate array.
                promises.push(cachedResourceResolve(styleUrl).then((/**
                 * @param {?} style
                 * @return {?}
                 */
                (style) => {
                    styles[styleOffset + index] = style;
                    styleUrls.splice(styleUrls.indexOf(styleUrl), 1);
                    if (styleUrls.length == 0) {
                        component.styleUrls = undefined;
                    }
                })));
            }));
            /** @type {?} */
            const fullyResolved = Promise.all(promises).then((/**
             * @return {?}
             */
            () => componentDefResolved(type)));
            componentResolved.push(fullyResolved);
        }));
        clearResolutionOfComponentResourcesQueue();
        return Promise.all(componentResolved).then((/**
         * @return {?}
         */
        () => undefined));
    }
    /** @type {?} */
    let componentResourceResolutionQueue = new Map();
    // Track when existing ɵcmp for a Type is waiting on resources.
    /** @type {?} */
    const componentDefPendingResolution = new Set();
    /**
     * @param {?} type
     * @param {?} metadata
     * @return {?}
     */
    function maybeQueueResolutionOfComponentResources(type, metadata) {
        if (componentNeedsResolution(metadata)) {
            componentResourceResolutionQueue.set(type, metadata);
            componentDefPendingResolution.add(type);
        }
    }
    /**
     * @param {?} component
     * @return {?}
     */
    function componentNeedsResolution(component) {
        return !!((component.templateUrl && !component.hasOwnProperty('template')) ||
            component.styleUrls && component.styleUrls.length);
    }
    /**
     * @return {?}
     */
    function clearResolutionOfComponentResourcesQueue() {
        /** @type {?} */
        const old = componentResourceResolutionQueue;
        componentResourceResolutionQueue = new Map();
        return old;
    }
    /**
     * @return {?}
     */
    function isComponentResourceResolutionQueueEmpty() {
        return componentResourceResolutionQueue.size === 0;
    }
    /**
     * @param {?} response
     * @return {?}
     */
    function unwrapResponse(response) {
        return typeof response == 'string' ? response : response.text();
    }
    /**
     * @param {?} type
     * @return {?}
     */
    function componentDefResolved(type) {
        componentDefPendingResolution.delete(type);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/styling/static_styling.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Compute the static styling (class/style) from `TAttributes`.
     *
     * This function should be called during `firstCreatePass` only.
     *
     * @param {?} tNode The `TNode` into which the styling information should be loaded.
     * @param {?} attrs `TAttributes` containing the styling information.
     * @param {?} writeToHost Where should the resulting static styles be written?
     *   - `false` Write to `TNode.stylesWithoutHost` / `TNode.classesWithoutHost`
     *   - `true` Write to `TNode.styles` / `TNode.classes`
     * @return {?}
     */
    function computeStaticStyling(tNode, attrs, writeToHost) {
        ngDevMode &&
            assertFirstCreatePass(getTView(), 'Expecting to be called in first template pass only');
        /** @type {?} */
        let styles = writeToHost ? tNode.styles : null;
        /** @type {?} */
        let classes = writeToHost ? tNode.classes : null;
        /** @type {?} */
        let mode = 0;
        if (attrs !== null) {
            for (let i = 0; i < attrs.length; i++) {
                /** @type {?} */
                const value = attrs[i];
                if (typeof value === 'number') {
                    mode = value;
                }
                else if (mode == 1 /* Classes */) {
                    classes = concatStringsWithSpace(classes, (/** @type {?} */ (value)));
                }
                else if (mode == 2 /* Styles */) {
                    /** @type {?} */
                    const style = (/** @type {?} */ (value));
                    /** @type {?} */
                    const styleValue = (/** @type {?} */ (attrs[++i]));
                    styles = concatStringsWithSpace(styles, style + ': ' + styleValue + ';');
                }
            }
        }
        writeToHost ? tNode.styles = styles : tNode.stylesWithoutHost = styles;
        writeToHost ? tNode.classes = classes : tNode.classesWithoutHost = classes;
    }

    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    let _symbolIterator = null;
    function getSymbolIterator$1() {
        if (!_symbolIterator) {
            const Symbol = _global['Symbol'];
            if (Symbol && Symbol.iterator) {
                _symbolIterator = Symbol.iterator;
            }
            else {
                // es6-shim specific logic
                const keys = Object.getOwnPropertyNames(Map.prototype);
                for (let i = 0; i < keys.length; ++i) {
                    const key = keys[i];
                    if (key !== 'entries' && key !== 'size' &&
                        Map.prototype[key] === Map.prototype['entries']) {
                        _symbolIterator = key;
                    }
                }
            }
        }
        return _symbolIterator;
    }

    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    // JS has NaN !== NaN
    function looseIdentical(a, b) {
        return a === b || typeof a === 'number' && typeof b === 'number' && isNaN(a) && isNaN(b);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/change_detection/change_detection_util.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @param {?} a
     * @param {?} b
     * @return {?}
     */
    function devModeEqual$1(a, b) {
        /** @type {?} */
        const isListLikeIterableA = isListLikeIterable$1(a);
        /** @type {?} */
        const isListLikeIterableB = isListLikeIterable$1(b);
        if (isListLikeIterableA && isListLikeIterableB) {
            return areIterablesEqual$1(a, b, devModeEqual$1);
        }
        else {
            /** @type {?} */
            const isAObject = a && (typeof a === 'object' || typeof a === 'function');
            /** @type {?} */
            const isBObject = b && (typeof b === 'object' || typeof b === 'function');
            if (!isListLikeIterableA && isAObject && !isListLikeIterableB && isBObject) {
                return true;
            }
            else {
                return looseIdentical(a, b);
            }
        }
    }
    /**
     * Indicates that the result of a {\@link Pipe} transformation has changed even though the
     * reference has not changed.
     *
     * Wrapped values are unwrapped automatically during the change detection, and the unwrapped value
     * is stored.
     *
     * Example:
     *
     * ```
     * if (this._latestValue === this._latestReturnedValue) {
     *    return this._latestReturnedValue;
     *  } else {
     *    this._latestReturnedValue = this._latestValue;
     *    return WrappedValue.wrap(this._latestValue); // this will force update
     *  }
     * ```
     *
     * \@publicApi
     */
    class WrappedValue {
        /**
         * @param {?} value
         */
        constructor(value) {
            this.wrapped = value;
        }
        /**
         * Creates a wrapped value.
         * @param {?} value
         * @return {?}
         */
        static wrap(value) {
            return new WrappedValue(value);
        }
        /**
         * Returns the underlying value of a wrapped value.
         * Returns the given `value` when it is not wrapped.
         *
         * @param {?} value
         * @return {?}
         */
        static unwrap(value) {
            return WrappedValue.isWrapped(value) ? value.wrapped : value;
        }
        /**
         * Returns true if `value` is a wrapped value.
         * @param {?} value
         * @return {?}
         */
        static isWrapped(value) {
            return value instanceof WrappedValue;
        }
    }
    /**
     * @param {?} obj
     * @return {?}
     */
    function isListLikeIterable$1(obj) {
        if (!isJsObject$1(obj))
            return false;
        return Array.isArray(obj) ||
            (!(obj instanceof Map) && // JS Map are iterables but return entries as [k, v]
                getSymbolIterator$1() in obj); // JS Iterable have a Symbol.iterator prop
    }
    /**
     * @param {?} a
     * @param {?} b
     * @param {?} comparator
     * @return {?}
     */
    function areIterablesEqual$1(a, b, comparator) {
        /** @type {?} */
        const iterator1 = a[getSymbolIterator$1()]();
        /** @type {?} */
        const iterator2 = b[getSymbolIterator$1()]();
        while (true) {
            /** @type {?} */
            const item1 = iterator1.next();
            /** @type {?} */
            const item2 = iterator2.next();
            if (item1.done && item2.done)
                return true;
            if (item1.done || item2.done)
                return false;
            if (!comparator(item1.value, item2.value))
                return false;
        }
    }
    /**
     * @param {?} obj
     * @param {?} fn
     * @return {?}
     */
    function iterateListLike$1(obj, fn) {
        if (Array.isArray(obj)) {
            for (let i = 0; i < obj.length; i++) {
                fn(obj[i]);
            }
        }
        else {
            /** @type {?} */
            const iterator = obj[getSymbolIterator$1()]();
            /** @type {?} */
            let item;
            while (!((item = iterator.next()).done)) {
                fn(item.value);
            }
        }
    }
    /**
     * @param {?} o
     * @return {?}
     */
    function isJsObject$1(o) {
        return o !== null && (typeof o === 'function' || typeof o === 'object');
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/bindings.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    // TODO(misko): consider inlining
    /**
     * Updates binding and returns the value.
     * @param {?} lView
     * @param {?} bindingIndex
     * @param {?} value
     * @return {?}
     */
    function updateBinding(lView, bindingIndex, value) {
        return lView[bindingIndex] = value;
    }
    /**
     * Gets the current binding value.
     * @param {?} lView
     * @param {?} bindingIndex
     * @return {?}
     */
    function getBinding(lView, bindingIndex) {
        ngDevMode && assertDataInRange(lView, bindingIndex);
        ngDevMode &&
            assertNotSame(lView[bindingIndex], NO_CHANGE, 'Stored value should never be NO_CHANGE.');
        return lView[bindingIndex];
    }
    /**
     * Updates binding if changed, then returns whether it was updated.
     *
     * This function also checks the `CheckNoChangesMode` and throws if changes are made.
     * Some changes (Objects/iterables) during `CheckNoChangesMode` are exempt to comply with VE
     * behavior.
     *
     * @param {?} lView current `LView`
     * @param {?} bindingIndex The binding in the `LView` to check
     * @param {?} value New value to check against `lView[bindingIndex]`
     * @return {?} `true` if the bindings has changed. (Throws if binding has changed during
     *          `CheckNoChangesMode`)
     */
    function bindingUpdated(lView, bindingIndex, value) {
        ngDevMode && assertNotSame(value, NO_CHANGE, 'Incoming value should never be NO_CHANGE.');
        ngDevMode &&
            assertLessThan(bindingIndex, lView.length, `Slot should have been initialized to NO_CHANGE`);
        /** @type {?} */
        const oldValue = lView[bindingIndex];
        if (Object.is(oldValue, value)) {
            return false;
        }
        else {
            if (ngDevMode && getCheckNoChangesMode()) {
                // View engine didn't report undefined values as changed on the first checkNoChanges pass
                // (before the change detection was run).
                /** @type {?} */
                const oldValueToCompare = oldValue !== NO_CHANGE ? oldValue : undefined;
                if (!devModeEqual$1(oldValueToCompare, value)) {
                    /** @type {?} */
                    const details = getExpressionChangedErrorDetails(lView, bindingIndex, oldValueToCompare, value);
                    throwErrorIfNoChangesMode(oldValue === NO_CHANGE, details.oldValue, details.newValue, details.propName);
                }
                // There was a change, but the `devModeEqual` decided that the change is exempt from an error.
                // For this reason we exit as if no change. The early exit is needed to prevent the changed
                // value to be written into `LView` (If we would write the new value that we would not see it
                // as change on next CD.)
                return false;
            }
            lView[bindingIndex] = value;
            return true;
        }
    }
    /**
     * Updates 2 bindings if changed, then returns whether either was updated.
     * @param {?} lView
     * @param {?} bindingIndex
     * @param {?} exp1
     * @param {?} exp2
     * @return {?}
     */
    function bindingUpdated2(lView, bindingIndex, exp1, exp2) {
        /** @type {?} */
        const different = bindingUpdated(lView, bindingIndex, exp1);
        return bindingUpdated(lView, bindingIndex + 1, exp2) || different;
    }
    /**
     * Updates 3 bindings if changed, then returns whether any was updated.
     * @param {?} lView
     * @param {?} bindingIndex
     * @param {?} exp1
     * @param {?} exp2
     * @param {?} exp3
     * @return {?}
     */
    function bindingUpdated3(lView, bindingIndex, exp1, exp2, exp3) {
        /** @type {?} */
        const different = bindingUpdated2(lView, bindingIndex, exp1, exp2);
        return bindingUpdated(lView, bindingIndex + 2, exp3) || different;
    }
    /**
     * Updates 4 bindings if changed, then returns whether any was updated.
     * @param {?} lView
     * @param {?} bindingIndex
     * @param {?} exp1
     * @param {?} exp2
     * @param {?} exp3
     * @param {?} exp4
     * @return {?}
     */
    function bindingUpdated4(lView, bindingIndex, exp1, exp2, exp3, exp4) {
        /** @type {?} */
        const different = bindingUpdated2(lView, bindingIndex, exp1, exp2);
        return bindingUpdated2(lView, bindingIndex + 2, exp3, exp4) || different;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/instructions/attribute.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Updates the value of or removes a bound attribute on an Element.
     *
     * Used in the case of `[attr.title]="value"`
     *
     * \@codeGenApi
     * @param {?} name name The name of the attribute.
     * @param {?} value value The attribute is removed when value is `null` or `undefined`.
     *                  Otherwise the attribute value is set to the stringified value.
     * @param {?=} sanitizer An optional function used to sanitize the value.
     * @param {?=} namespace Optional namespace to use when setting the attribute.
     *
     * @return {?}
     */
    function ɵɵattribute(name, value, sanitizer, namespace) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const bindingIndex = nextBindingIndex();
        if (bindingUpdated(lView, bindingIndex, value)) {
            /** @type {?} */
            const tView = getTView();
            /** @type {?} */
            const tNode = getSelectedTNode();
            elementAttributeInternal(tNode, lView, name, value, sanitizer, namespace);
            ngDevMode && storePropertyBindingMetadata(tView.data, tNode, 'attr.' + name, bindingIndex);
        }
        return ɵɵattribute;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/instructions/interpolation.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Create interpolation bindings with a variable number of expressions.
     *
     * If there are 1 to 8 expressions `interpolation1()` to `interpolation8()` should be used instead.
     * Those are faster because there is no need to create an array of expressions and iterate over it.
     *
     * `values`:
     * - has static text at even indexes,
     * - has evaluated expressions at odd indexes.
     *
     * Returns the concatenated string when any of the arguments changes, `NO_CHANGE` otherwise.
     * @param {?} lView
     * @param {?} values
     * @return {?}
     */
    function interpolationV(lView, values) {
        ngDevMode && assertLessThan(2, values.length, 'should have at least 3 values');
        ngDevMode && assertEqual(values.length % 2, 1, 'should have an odd number of values');
        /** @type {?} */
        let isBindingUpdated = false;
        /** @type {?} */
        let bindingIndex = getBindingIndex();
        for (let i = 1; i < values.length; i += 2) {
            // Check if bindings (odd indexes) have changed
            isBindingUpdated = bindingUpdated(lView, bindingIndex++, values[i]) || isBindingUpdated;
        }
        setBindingIndex(bindingIndex);
        if (!isBindingUpdated) {
            return NO_CHANGE;
        }
        // Build the updated content
        /** @type {?} */
        let content = values[0];
        for (let i = 1; i < values.length; i += 2) {
            content += renderStringify(values[i]) + values[i + 1];
        }
        return content;
    }
    /**
     * Creates an interpolation binding with 1 expression.
     *
     * @param {?} lView
     * @param {?} prefix static value used for concatenation only.
     * @param {?} v0 value checked for change.
     * @param {?} suffix static value used for concatenation only.
     * @return {?}
     */
    function interpolation1(lView, prefix, v0, suffix) {
        /** @type {?} */
        const different = bindingUpdated(lView, nextBindingIndex(), v0);
        return different ? prefix + renderStringify(v0) + suffix : NO_CHANGE;
    }
    /**
     * Creates an interpolation binding with 2 expressions.
     * @param {?} lView
     * @param {?} prefix
     * @param {?} v0
     * @param {?} i0
     * @param {?} v1
     * @param {?} suffix
     * @return {?}
     */
    function interpolation2(lView, prefix, v0, i0, v1, suffix) {
        /** @type {?} */
        const bindingIndex = getBindingIndex();
        /** @type {?} */
        const different = bindingUpdated2(lView, bindingIndex, v0, v1);
        incrementBindingIndex(2);
        return different ? prefix + renderStringify(v0) + i0 + renderStringify(v1) + suffix : NO_CHANGE;
    }
    /**
     * Creates an interpolation binding with 3 expressions.
     * @param {?} lView
     * @param {?} prefix
     * @param {?} v0
     * @param {?} i0
     * @param {?} v1
     * @param {?} i1
     * @param {?} v2
     * @param {?} suffix
     * @return {?}
     */
    function interpolation3(lView, prefix, v0, i0, v1, i1, v2, suffix) {
        /** @type {?} */
        const bindingIndex = getBindingIndex();
        /** @type {?} */
        const different = bindingUpdated3(lView, bindingIndex, v0, v1, v2);
        incrementBindingIndex(3);
        return different ?
            prefix + renderStringify(v0) + i0 + renderStringify(v1) + i1 + renderStringify(v2) + suffix :
            NO_CHANGE;
    }
    /**
     * Create an interpolation binding with 4 expressions.
     * @param {?} lView
     * @param {?} prefix
     * @param {?} v0
     * @param {?} i0
     * @param {?} v1
     * @param {?} i1
     * @param {?} v2
     * @param {?} i2
     * @param {?} v3
     * @param {?} suffix
     * @return {?}
     */
    function interpolation4(lView, prefix, v0, i0, v1, i1, v2, i2, v3, suffix) {
        /** @type {?} */
        const bindingIndex = getBindingIndex();
        /** @type {?} */
        const different = bindingUpdated4(lView, bindingIndex, v0, v1, v2, v3);
        incrementBindingIndex(4);
        return different ? prefix + renderStringify(v0) + i0 + renderStringify(v1) + i1 +
            renderStringify(v2) + i2 + renderStringify(v3) + suffix :
            NO_CHANGE;
    }
    /**
     * Creates an interpolation binding with 5 expressions.
     * @param {?} lView
     * @param {?} prefix
     * @param {?} v0
     * @param {?} i0
     * @param {?} v1
     * @param {?} i1
     * @param {?} v2
     * @param {?} i2
     * @param {?} v3
     * @param {?} i3
     * @param {?} v4
     * @param {?} suffix
     * @return {?}
     */
    function interpolation5(lView, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, suffix) {
        /** @type {?} */
        const bindingIndex = getBindingIndex();
        /** @type {?} */
        let different = bindingUpdated4(lView, bindingIndex, v0, v1, v2, v3);
        different = bindingUpdated(lView, bindingIndex + 4, v4) || different;
        incrementBindingIndex(5);
        return different ? prefix + renderStringify(v0) + i0 + renderStringify(v1) + i1 +
            renderStringify(v2) + i2 + renderStringify(v3) + i3 + renderStringify(v4) + suffix :
            NO_CHANGE;
    }
    /**
     * Creates an interpolation binding with 6 expressions.
     * @param {?} lView
     * @param {?} prefix
     * @param {?} v0
     * @param {?} i0
     * @param {?} v1
     * @param {?} i1
     * @param {?} v2
     * @param {?} i2
     * @param {?} v3
     * @param {?} i3
     * @param {?} v4
     * @param {?} i4
     * @param {?} v5
     * @param {?} suffix
     * @return {?}
     */
    function interpolation6(lView, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, suffix) {
        /** @type {?} */
        const bindingIndex = getBindingIndex();
        /** @type {?} */
        let different = bindingUpdated4(lView, bindingIndex, v0, v1, v2, v3);
        different = bindingUpdated2(lView, bindingIndex + 4, v4, v5) || different;
        incrementBindingIndex(6);
        return different ?
            prefix + renderStringify(v0) + i0 + renderStringify(v1) + i1 + renderStringify(v2) + i2 +
                renderStringify(v3) + i3 + renderStringify(v4) + i4 + renderStringify(v5) + suffix :
            NO_CHANGE;
    }
    /**
     * Creates an interpolation binding with 7 expressions.
     * @param {?} lView
     * @param {?} prefix
     * @param {?} v0
     * @param {?} i0
     * @param {?} v1
     * @param {?} i1
     * @param {?} v2
     * @param {?} i2
     * @param {?} v3
     * @param {?} i3
     * @param {?} v4
     * @param {?} i4
     * @param {?} v5
     * @param {?} i5
     * @param {?} v6
     * @param {?} suffix
     * @return {?}
     */
    function interpolation7(lView, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, i5, v6, suffix) {
        /** @type {?} */
        const bindingIndex = getBindingIndex();
        /** @type {?} */
        let different = bindingUpdated4(lView, bindingIndex, v0, v1, v2, v3);
        different = bindingUpdated3(lView, bindingIndex + 4, v4, v5, v6) || different;
        incrementBindingIndex(7);
        return different ? prefix + renderStringify(v0) + i0 + renderStringify(v1) + i1 +
            renderStringify(v2) + i2 + renderStringify(v3) + i3 + renderStringify(v4) + i4 +
            renderStringify(v5) + i5 + renderStringify(v6) + suffix :
            NO_CHANGE;
    }
    /**
     * Creates an interpolation binding with 8 expressions.
     * @param {?} lView
     * @param {?} prefix
     * @param {?} v0
     * @param {?} i0
     * @param {?} v1
     * @param {?} i1
     * @param {?} v2
     * @param {?} i2
     * @param {?} v3
     * @param {?} i3
     * @param {?} v4
     * @param {?} i4
     * @param {?} v5
     * @param {?} i5
     * @param {?} v6
     * @param {?} i6
     * @param {?} v7
     * @param {?} suffix
     * @return {?}
     */
    function interpolation8(lView, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, i5, v6, i6, v7, suffix) {
        /** @type {?} */
        const bindingIndex = getBindingIndex();
        /** @type {?} */
        let different = bindingUpdated4(lView, bindingIndex, v0, v1, v2, v3);
        different = bindingUpdated4(lView, bindingIndex + 4, v4, v5, v6, v7) || different;
        incrementBindingIndex(8);
        return different ? prefix + renderStringify(v0) + i0 + renderStringify(v1) + i1 +
            renderStringify(v2) + i2 + renderStringify(v3) + i3 + renderStringify(v4) + i4 +
            renderStringify(v5) + i5 + renderStringify(v6) + i6 + renderStringify(v7) + suffix :
            NO_CHANGE;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/instructions/attribute_interpolation.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     *
     * Update an interpolated attribute on an element with single bound value surrounded by text.
     *
     * Used when the value passed to a property has 1 interpolated value in it:
     *
     * ```html
     * <div attr.title="prefix{{v0}}suffix"></div>
     * ```
     *
     * Its compiled representation is::
     *
     * ```ts
     * ɵɵattributeInterpolate1('title', 'prefix', v0, 'suffix');
     * ```
     *
     * \@codeGenApi
     * @param {?} attrName The name of the attribute to update
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @param {?=} sanitizer An optional sanitizer function
     * @param {?=} namespace
     * @return {?} itself, so that it may be chained.
     */
    function ɵɵattributeInterpolate1(attrName, prefix, v0, suffix, sanitizer, namespace) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation1(lView, prefix, v0, suffix);
        if (interpolatedValue !== NO_CHANGE) {
            /** @type {?} */
            const tNode = getSelectedTNode();
            elementAttributeInternal(tNode, lView, attrName, interpolatedValue, sanitizer, namespace);
            ngDevMode &&
                storePropertyBindingMetadata(getTView().data, tNode, 'attr.' + attrName, getBindingIndex() - 1, prefix, suffix);
        }
        return ɵɵattributeInterpolate1;
    }
    /**
     *
     * Update an interpolated attribute on an element with 2 bound values surrounded by text.
     *
     * Used when the value passed to a property has 2 interpolated values in it:
     *
     * ```html
     * <div attr.title="prefix{{v0}}-{{v1}}suffix"></div>
     * ```
     *
     * Its compiled representation is::
     *
     * ```ts
     * ɵɵattributeInterpolate2('title', 'prefix', v0, '-', v1, 'suffix');
     * ```
     *
     * \@codeGenApi
     * @param {?} attrName The name of the attribute to update
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} i0 Static value used for concatenation only.
     * @param {?} v1 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @param {?=} sanitizer An optional sanitizer function
     * @param {?=} namespace
     * @return {?} itself, so that it may be chained.
     */
    function ɵɵattributeInterpolate2(attrName, prefix, v0, i0, v1, suffix, sanitizer, namespace) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation2(lView, prefix, v0, i0, v1, suffix);
        if (interpolatedValue !== NO_CHANGE) {
            /** @type {?} */
            const tNode = getSelectedTNode();
            elementAttributeInternal(tNode, lView, attrName, interpolatedValue, sanitizer, namespace);
            ngDevMode &&
                storePropertyBindingMetadata(getTView().data, tNode, 'attr.' + attrName, getBindingIndex() - 2, prefix, i0, suffix);
        }
        return ɵɵattributeInterpolate2;
    }
    /**
     *
     * Update an interpolated attribute on an element with 3 bound values surrounded by text.
     *
     * Used when the value passed to a property has 3 interpolated values in it:
     *
     * ```html
     * <div attr.title="prefix{{v0}}-{{v1}}-{{v2}}suffix"></div>
     * ```
     *
     * Its compiled representation is::
     *
     * ```ts
     * ɵɵattributeInterpolate3(
     * 'title', 'prefix', v0, '-', v1, '-', v2, 'suffix');
     * ```
     *
     * \@codeGenApi
     * @param {?} attrName The name of the attribute to update
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} i0 Static value used for concatenation only.
     * @param {?} v1 Value checked for change.
     * @param {?} i1 Static value used for concatenation only.
     * @param {?} v2 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @param {?=} sanitizer An optional sanitizer function
     * @param {?=} namespace
     * @return {?} itself, so that it may be chained.
     */
    function ɵɵattributeInterpolate3(attrName, prefix, v0, i0, v1, i1, v2, suffix, sanitizer, namespace) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation3(lView, prefix, v0, i0, v1, i1, v2, suffix);
        if (interpolatedValue !== NO_CHANGE) {
            /** @type {?} */
            const tNode = getSelectedTNode();
            elementAttributeInternal(tNode, lView, attrName, interpolatedValue, sanitizer, namespace);
            ngDevMode &&
                storePropertyBindingMetadata(getTView().data, tNode, 'attr.' + attrName, getBindingIndex() - 3, prefix, i0, i1, suffix);
        }
        return ɵɵattributeInterpolate3;
    }
    /**
     *
     * Update an interpolated attribute on an element with 4 bound values surrounded by text.
     *
     * Used when the value passed to a property has 4 interpolated values in it:
     *
     * ```html
     * <div attr.title="prefix{{v0}}-{{v1}}-{{v2}}-{{v3}}suffix"></div>
     * ```
     *
     * Its compiled representation is::
     *
     * ```ts
     * ɵɵattributeInterpolate4(
     * 'title', 'prefix', v0, '-', v1, '-', v2, '-', v3, 'suffix');
     * ```
     *
     * \@codeGenApi
     * @param {?} attrName The name of the attribute to update
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} i0 Static value used for concatenation only.
     * @param {?} v1 Value checked for change.
     * @param {?} i1 Static value used for concatenation only.
     * @param {?} v2 Value checked for change.
     * @param {?} i2 Static value used for concatenation only.
     * @param {?} v3 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @param {?=} sanitizer An optional sanitizer function
     * @param {?=} namespace
     * @return {?} itself, so that it may be chained.
     */
    function ɵɵattributeInterpolate4(attrName, prefix, v0, i0, v1, i1, v2, i2, v3, suffix, sanitizer, namespace) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation4(lView, prefix, v0, i0, v1, i1, v2, i2, v3, suffix);
        if (interpolatedValue !== NO_CHANGE) {
            /** @type {?} */
            const tNode = getSelectedTNode();
            elementAttributeInternal(tNode, lView, attrName, interpolatedValue, sanitizer, namespace);
            ngDevMode &&
                storePropertyBindingMetadata(getTView().data, tNode, 'attr.' + attrName, getBindingIndex() - 4, prefix, i0, i1, i2, suffix);
        }
        return ɵɵattributeInterpolate4;
    }
    /**
     *
     * Update an interpolated attribute on an element with 5 bound values surrounded by text.
     *
     * Used when the value passed to a property has 5 interpolated values in it:
     *
     * ```html
     * <div attr.title="prefix{{v0}}-{{v1}}-{{v2}}-{{v3}}-{{v4}}suffix"></div>
     * ```
     *
     * Its compiled representation is::
     *
     * ```ts
     * ɵɵattributeInterpolate5(
     * 'title', 'prefix', v0, '-', v1, '-', v2, '-', v3, '-', v4, 'suffix');
     * ```
     *
     * \@codeGenApi
     * @param {?} attrName The name of the attribute to update
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} i0 Static value used for concatenation only.
     * @param {?} v1 Value checked for change.
     * @param {?} i1 Static value used for concatenation only.
     * @param {?} v2 Value checked for change.
     * @param {?} i2 Static value used for concatenation only.
     * @param {?} v3 Value checked for change.
     * @param {?} i3 Static value used for concatenation only.
     * @param {?} v4 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @param {?=} sanitizer An optional sanitizer function
     * @param {?=} namespace
     * @return {?} itself, so that it may be chained.
     */
    function ɵɵattributeInterpolate5(attrName, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, suffix, sanitizer, namespace) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation5(lView, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, suffix);
        if (interpolatedValue !== NO_CHANGE) {
            /** @type {?} */
            const tNode = getSelectedTNode();
            elementAttributeInternal(tNode, lView, attrName, interpolatedValue, sanitizer, namespace);
            ngDevMode &&
                storePropertyBindingMetadata(getTView().data, tNode, 'attr.' + attrName, getBindingIndex() - 5, prefix, i0, i1, i2, i3, suffix);
        }
        return ɵɵattributeInterpolate5;
    }
    /**
     *
     * Update an interpolated attribute on an element with 6 bound values surrounded by text.
     *
     * Used when the value passed to a property has 6 interpolated values in it:
     *
     * ```html
     * <div attr.title="prefix{{v0}}-{{v1}}-{{v2}}-{{v3}}-{{v4}}-{{v5}}suffix"></div>
     * ```
     *
     * Its compiled representation is::
     *
     * ```ts
     * ɵɵattributeInterpolate6(
     *    'title', 'prefix', v0, '-', v1, '-', v2, '-', v3, '-', v4, '-', v5, 'suffix');
     * ```
     *
     * \@codeGenApi
     * @param {?} attrName The name of the attribute to update
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} i0 Static value used for concatenation only.
     * @param {?} v1 Value checked for change.
     * @param {?} i1 Static value used for concatenation only.
     * @param {?} v2 Value checked for change.
     * @param {?} i2 Static value used for concatenation only.
     * @param {?} v3 Value checked for change.
     * @param {?} i3 Static value used for concatenation only.
     * @param {?} v4 Value checked for change.
     * @param {?} i4 Static value used for concatenation only.
     * @param {?} v5 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @param {?=} sanitizer An optional sanitizer function
     * @param {?=} namespace
     * @return {?} itself, so that it may be chained.
     */
    function ɵɵattributeInterpolate6(attrName, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, suffix, sanitizer, namespace) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation6(lView, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, suffix);
        if (interpolatedValue !== NO_CHANGE) {
            /** @type {?} */
            const tNode = getSelectedTNode();
            elementAttributeInternal(tNode, lView, attrName, interpolatedValue, sanitizer, namespace);
            ngDevMode &&
                storePropertyBindingMetadata(getTView().data, tNode, 'attr.' + attrName, getBindingIndex() - 6, prefix, i0, i1, i2, i3, i4, suffix);
        }
        return ɵɵattributeInterpolate6;
    }
    /**
     *
     * Update an interpolated attribute on an element with 7 bound values surrounded by text.
     *
     * Used when the value passed to a property has 7 interpolated values in it:
     *
     * ```html
     * <div attr.title="prefix{{v0}}-{{v1}}-{{v2}}-{{v3}}-{{v4}}-{{v5}}-{{v6}}suffix"></div>
     * ```
     *
     * Its compiled representation is::
     *
     * ```ts
     * ɵɵattributeInterpolate7(
     *    'title', 'prefix', v0, '-', v1, '-', v2, '-', v3, '-', v4, '-', v5, '-', v6, 'suffix');
     * ```
     *
     * \@codeGenApi
     * @param {?} attrName The name of the attribute to update
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} i0 Static value used for concatenation only.
     * @param {?} v1 Value checked for change.
     * @param {?} i1 Static value used for concatenation only.
     * @param {?} v2 Value checked for change.
     * @param {?} i2 Static value used for concatenation only.
     * @param {?} v3 Value checked for change.
     * @param {?} i3 Static value used for concatenation only.
     * @param {?} v4 Value checked for change.
     * @param {?} i4 Static value used for concatenation only.
     * @param {?} v5 Value checked for change.
     * @param {?} i5 Static value used for concatenation only.
     * @param {?} v6 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @param {?=} sanitizer An optional sanitizer function
     * @param {?=} namespace
     * @return {?} itself, so that it may be chained.
     */
    function ɵɵattributeInterpolate7(attrName, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, i5, v6, suffix, sanitizer, namespace) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation7(lView, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, i5, v6, suffix);
        if (interpolatedValue !== NO_CHANGE) {
            /** @type {?} */
            const tNode = getSelectedTNode();
            elementAttributeInternal(tNode, lView, attrName, interpolatedValue, sanitizer, namespace);
            ngDevMode &&
                storePropertyBindingMetadata(getTView().data, tNode, 'attr.' + attrName, getBindingIndex() - 7, prefix, i0, i1, i2, i3, i4, i5, suffix);
        }
        return ɵɵattributeInterpolate7;
    }
    /**
     *
     * Update an interpolated attribute on an element with 8 bound values surrounded by text.
     *
     * Used when the value passed to a property has 8 interpolated values in it:
     *
     * ```html
     * <div attr.title="prefix{{v0}}-{{v1}}-{{v2}}-{{v3}}-{{v4}}-{{v5}}-{{v6}}-{{v7}}suffix"></div>
     * ```
     *
     * Its compiled representation is::
     *
     * ```ts
     * ɵɵattributeInterpolate8(
     *  'title', 'prefix', v0, '-', v1, '-', v2, '-', v3, '-', v4, '-', v5, '-', v6, '-', v7, 'suffix');
     * ```
     *
     * \@codeGenApi
     * @param {?} attrName The name of the attribute to update
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} i0 Static value used for concatenation only.
     * @param {?} v1 Value checked for change.
     * @param {?} i1 Static value used for concatenation only.
     * @param {?} v2 Value checked for change.
     * @param {?} i2 Static value used for concatenation only.
     * @param {?} v3 Value checked for change.
     * @param {?} i3 Static value used for concatenation only.
     * @param {?} v4 Value checked for change.
     * @param {?} i4 Static value used for concatenation only.
     * @param {?} v5 Value checked for change.
     * @param {?} i5 Static value used for concatenation only.
     * @param {?} v6 Value checked for change.
     * @param {?} i6 Static value used for concatenation only.
     * @param {?} v7 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @param {?=} sanitizer An optional sanitizer function
     * @param {?=} namespace
     * @return {?} itself, so that it may be chained.
     */
    function ɵɵattributeInterpolate8(attrName, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, i5, v6, i6, v7, suffix, sanitizer, namespace) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation8(lView, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, i5, v6, i6, v7, suffix);
        if (interpolatedValue !== NO_CHANGE) {
            /** @type {?} */
            const tNode = getSelectedTNode();
            elementAttributeInternal(tNode, lView, attrName, interpolatedValue, sanitizer, namespace);
            ngDevMode &&
                storePropertyBindingMetadata(getTView().data, tNode, 'attr.' + attrName, getBindingIndex() - 8, prefix, i0, i1, i2, i3, i4, i5, i6, suffix);
        }
        return ɵɵattributeInterpolate8;
    }
    /**
     * Update an interpolated attribute on an element with 9 or more bound values surrounded by text.
     *
     * Used when the number of interpolated values exceeds 8.
     *
     * ```html
     * <div
     *  title="prefix{{v0}}-{{v1}}-{{v2}}-{{v3}}-{{v4}}-{{v5}}-{{v6}}-{{v7}}-{{v8}}-{{v9}}suffix"></div>
     * ```
     *
     * Its compiled representation is::
     *
     * ```ts
     * ɵɵattributeInterpolateV(
     *  'title', ['prefix', v0, '-', v1, '-', v2, '-', v3, '-', v4, '-', v5, '-', v6, '-', v7, '-', v9,
     *  'suffix']);
     * ```
     *
     * \@codeGenApi
     * @param {?} attrName The name of the attribute to update.
     * @param {?} values The collection of values and the strings in-between those values, beginning with
     * a string prefix and ending with a string suffix.
     * (e.g. `['prefix', value0, '-', value1, '-', value2, ..., value99, 'suffix']`)
     * @param {?=} sanitizer An optional sanitizer function
     * @param {?=} namespace
     * @return {?} itself, so that it may be chained.
     */
    function ɵɵattributeInterpolateV(attrName, values, sanitizer, namespace) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolated = interpolationV(lView, values);
        if (interpolated !== NO_CHANGE) {
            /** @type {?} */
            const tNode = getSelectedTNode();
            elementAttributeInternal(tNode, lView, attrName, interpolated, sanitizer, namespace);
            if (ngDevMode) {
                /** @type {?} */
                const interpolationInBetween = [values[0]];
                for (let i = 2; i < values.length; i += 2) {
                    interpolationInBetween.push(values[i]);
                }
                storePropertyBindingMetadata(getTView().data, tNode, 'attr.' + attrName, getBindingIndex() - interpolationInBetween.length + 1, ...interpolationInBetween);
            }
        }
        return ɵɵattributeInterpolateV;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/instructions/change_detection.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Synchronously perform change detection on a component (and possibly its sub-components).
     *
     * This function triggers change detection in a synchronous way on a component.
     *
     * @param {?} component The component which the change detection should be performed on.
     * @return {?}
     */
    function detectChanges(component) {
        /** @type {?} */
        const view = getComponentViewByInstance(component);
        detectChangesInternal(view[TVIEW], view, component);
    }
    /**
     * Marks the component as dirty (needing change detection). Marking a component dirty will
     * schedule a change detection on it at some point in the future.
     *
     * Marking an already dirty component as dirty won't do anything. Only one outstanding change
     * detection can be scheduled per component tree.
     *
     * @param {?} component Component to mark as dirty.
     * @return {?}
     */
    function markDirty(component) {
        ngDevMode && assertDefined(component, 'component');
        /** @type {?} */
        const rootView = (/** @type {?} */ (markViewDirty(getComponentViewByInstance(component))));
        ngDevMode && assertDefined(rootView[CONTEXT], 'rootContext should be defined');
        scheduleTick((/** @type {?} */ (rootView[CONTEXT])), 1 /* DetectChanges */);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/instructions/template.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @param {?} index
     * @param {?} tView
     * @param {?} lView
     * @param {?} templateFn
     * @param {?} decls
     * @param {?} vars
     * @param {?=} tagName
     * @param {?=} attrsIndex
     * @param {?=} localRefsIndex
     * @return {?}
     */
    function templateFirstCreatePass(index, tView, lView, templateFn, decls, vars, tagName, attrsIndex, localRefsIndex) {
        ngDevMode && assertFirstCreatePass(tView);
        ngDevMode && ngDevMode.firstCreatePass++;
        /** @type {?} */
        const tViewConsts = tView.consts;
        // TODO(pk): refactor getOrCreateTNode to have the "create" only version
        /** @type {?} */
        const tNode = getOrCreateTNode(tView, lView[T_HOST], index, 0 /* Container */, tagName || null, getConstant(tViewConsts, attrsIndex));
        resolveDirectives(tView, lView, tNode, getConstant(tViewConsts, localRefsIndex));
        registerPostOrderHooks(tView, tNode);
        /** @type {?} */
        const embeddedTView = tNode.tViews = createTView(2 /* Embedded */, -1, templateFn, decls, vars, tView.directiveRegistry, tView.pipeRegistry, null, tView.schemas, tViewConsts);
        /** @type {?} */
        const embeddedTViewNode = (/** @type {?} */ (createTNode(tView, null, 2 /* View */, -1, null, null)));
        embeddedTViewNode.injectorIndex = tNode.injectorIndex;
        embeddedTView.node = embeddedTViewNode;
        if (tView.queries !== null) {
            tView.queries.template(tView, tNode);
            embeddedTView.queries = tView.queries.embeddedTView(tNode);
        }
        return tNode;
    }
    /**
     * Creates an LContainer for an ng-template (dynamically-inserted view), e.g.
     *
     * <ng-template #foo>
     *    <div></div>
     * </ng-template>
     *
     * \@codeGenApi
     * @param {?} index The index of the container in the data array
     * @param {?} templateFn Inline template
     * @param {?} decls The number of nodes, local refs, and pipes for this template
     * @param {?} vars The number of bindings for this template
     * @param {?=} tagName The name of the container element, if applicable
     * @param {?=} attrsIndex Index of template attributes in the `consts` array.
     * @param {?=} localRefsIndex
     * @param {?=} localRefExtractor A function which extracts local-refs values from the template.
     *        Defaults to the current element associated with the local-ref.
     *
     * @return {?}
     */
    function ɵɵtemplate(index, templateFn, decls, vars, tagName, attrsIndex, localRefsIndex, localRefExtractor) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const tView = getTView();
        /** @type {?} */
        const adjustedIndex = index + HEADER_OFFSET;
        /** @type {?} */
        const tNode = tView.firstCreatePass ?
            templateFirstCreatePass(index, tView, lView, templateFn, decls, vars, tagName, attrsIndex, localRefsIndex) :
            (/** @type {?} */ (tView.data[adjustedIndex]));
        setPreviousOrParentTNode(tNode, false);
        /** @type {?} */
        const comment = lView[RENDERER].createComment(ngDevMode ? 'container' : '');
        appendChild(tView, lView, comment, tNode);
        attachPatchData(comment, lView);
        addToViewTree(lView, lView[adjustedIndex] = createLContainer(comment, lView, comment, tNode));
        if (isDirectiveHost(tNode)) {
            createDirectivesInstances(tView, lView, tNode);
        }
        if (localRefsIndex != null) {
            saveResolvedLocalsInData(lView, tNode, localRefExtractor);
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/instructions/storage.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Store a value in the `data` at a given `index`.
     * @template T
     * @param {?} tView
     * @param {?} lView
     * @param {?} index
     * @param {?} value
     * @return {?}
     */
    function store(tView, lView, index, value) {
        // We don't store any static data for local variables, so the first time
        // we see the template, we should store as null to avoid a sparse array
        /** @type {?} */
        const adjustedIndex = index + HEADER_OFFSET;
        if (adjustedIndex >= tView.data.length) {
            tView.data[adjustedIndex] = null;
            tView.blueprint[adjustedIndex] = null;
        }
        lView[adjustedIndex] = value;
    }
    /**
     * Retrieves a local reference from the current contextViewData.
     *
     * If the reference to retrieve is in a parent view, this instruction is used in conjunction
     * with a nextContext() call, which walks up the tree and updates the contextViewData instance.
     *
     * \@codeGenApi
     * @template T
     * @param {?} index The index of the local ref in contextViewData.
     *
     * @return {?}
     */
    function ɵɵreference(index) {
        /** @type {?} */
        const contextLView = getContextLView();
        return load(contextLView, index);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/instructions/di.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @template T
     * @param {?} token
     * @param {?=} flags
     * @return {?}
     */
    function ɵɵdirectiveInject(token, flags = InjectFlags.Default) {
        /** @type {?} */
        const lView = getLView();
        // Fall back to inject() if view hasn't been created. This situation can happen in tests
        // if inject utilities are used before bootstrapping.
        if (lView == null)
            return ɵɵinject(token, flags);
        /** @type {?} */
        const tNode = getPreviousOrParentTNode();
        return getOrCreateInjectable((/** @type {?} */ (tNode)), lView, resolveForwardRef(token), flags);
    }
    /**
     * Facade for the attribute injection from DI.
     *
     * \@codeGenApi
     * @param {?} attrNameToInject
     * @return {?}
     */
    function ɵɵinjectAttribute(attrNameToInject) {
        return injectAttributeImpl(getPreviousOrParentTNode(), attrNameToInject);
    }
    /**
     * Throws an error indicating that a factory function could not be generated by the compiler for a
     * particular class.
     *
     * This instruction allows the actual error message to be optimized away when ngDevMode is turned
     * off, saving bytes of generated code while still providing a good experience in dev mode.
     *
     * The name of the class is not mentioned here, but will be in the generated factory function name
     * and thus in the stack trace.
     *
     * \@codeGenApi
     * @return {?}
     */
    function ɵɵinvalidFactory() {
        /** @type {?} */
        const msg = ngDevMode ? `This constructor was not compatible with Dependency Injection.` : 'invalid';
        throw new Error(msg);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/instructions/property.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Update a property on a selected element.
     *
     * Operates on the element selected by index via the {\@link select} instruction.
     *
     * If the property name also exists as an input property on one of the element's directives,
     * the component property will be set instead of the element property. This check must
     * be conducted at runtime so child components that add new `\@Inputs` don't have to be re-compiled
     *
     * \@codeGenApi
     * @template T
     * @param {?} propName Name of property. Because it is going to DOM, this is not subject to
     *        renaming as part of minification.
     * @param {?} value New value to write.
     * @param {?=} sanitizer An optional function used to sanitize the value.
     * @return {?} This function returns itself so that it may be chained
     * (e.g. `property('name', ctx.name)('title', ctx.title)`)
     *
     */
    function ɵɵproperty(propName, value, sanitizer) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const bindingIndex = nextBindingIndex();
        if (bindingUpdated(lView, bindingIndex, value)) {
            /** @type {?} */
            const tView = getTView();
            /** @type {?} */
            const tNode = getSelectedTNode();
            elementPropertyInternal(tView, tNode, lView, propName, value, lView[RENDERER], sanitizer, false);
            ngDevMode && storePropertyBindingMetadata(tView.data, tNode, propName, bindingIndex);
        }
        return ɵɵproperty;
    }
    /**
     * Given `<div style="..." my-dir>` and `MyDir` with `\@Input('style')` we need to write to
     * directive input.
     * @param {?} tView
     * @param {?} tNode
     * @param {?} lView
     * @param {?} value
     * @param {?} isClassBased
     * @return {?}
     */
    function setDirectiveInputsWhichShadowsStyling(tView, tNode, lView, value, isClassBased) {
        /** @type {?} */
        const inputs = (/** @type {?} */ (tNode.inputs));
        /** @type {?} */
        const property = isClassBased ? 'class' : 'style';
        // We support both 'class' and `className` hence the fallback.
        setInputsForProperty(tView, lView, inputs[property], property, value);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/instructions/element.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @param {?} index
     * @param {?} tView
     * @param {?} lView
     * @param {?} native
     * @param {?} name
     * @param {?=} attrsIndex
     * @param {?=} localRefsIndex
     * @return {?}
     */
    function elementStartFirstCreatePass(index, tView, lView, native, name, attrsIndex, localRefsIndex) {
        ngDevMode && assertFirstCreatePass(tView);
        ngDevMode && ngDevMode.firstCreatePass++;
        /** @type {?} */
        const tViewConsts = tView.consts;
        /** @type {?} */
        const attrs = getConstant(tViewConsts, attrsIndex);
        /** @type {?} */
        const tNode = getOrCreateTNode(tView, lView[T_HOST], index, 3 /* Element */, name, attrs);
        /** @type {?} */
        const hasDirectives = resolveDirectives(tView, lView, tNode, getConstant(tViewConsts, localRefsIndex));
        ngDevMode && warnAboutUnknownElement(tView, lView, native, tNode, hasDirectives);
        if (tNode.attrs !== null) {
            computeStaticStyling(tNode, tNode.attrs, false);
        }
        if (tNode.mergedAttrs !== null) {
            computeStaticStyling(tNode, tNode.mergedAttrs, true);
        }
        if (tView.queries !== null) {
            tView.queries.elementStart(tView, tNode);
        }
        return tNode;
    }
    /**
     * Create DOM element. The instruction must later be followed by `elementEnd()` call.
     *
     * \@codeGenApi
     * @param {?} index Index of the element in the LView array
     * @param {?} name Name of the DOM Node
     * @param {?=} attrsIndex Index of the element's attributes in the `consts` array.
     * @param {?=} localRefsIndex Index of the element's local references in the `consts` array.
     *
     * Attributes and localRefs are passed as an array of strings where elements with an even index
     * hold an attribute name and elements with an odd index hold an attribute value, ex.:
     * ['id', 'warning5', 'class', 'alert']
     *
     * @return {?}
     */
    function ɵɵelementStart(index, name, attrsIndex, localRefsIndex) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const tView = getTView();
        /** @type {?} */
        const adjustedIndex = HEADER_OFFSET + index;
        ngDevMode &&
            assertEqual(getBindingIndex(), tView.bindingStartIndex, 'elements should be created before any bindings');
        ngDevMode && ngDevMode.rendererCreateElement++;
        ngDevMode && assertDataInRange(lView, adjustedIndex);
        /** @type {?} */
        const renderer = lView[RENDERER];
        /** @type {?} */
        const native = lView[adjustedIndex] = elementCreate(name, renderer, getNamespace());
        /** @type {?} */
        const tNode = tView.firstCreatePass ?
            elementStartFirstCreatePass(index, tView, lView, native, name, attrsIndex, localRefsIndex) :
            (/** @type {?} */ (tView.data[adjustedIndex]));
        setPreviousOrParentTNode(tNode, true);
        /** @type {?} */
        const mergedAttrs = tNode.mergedAttrs;
        if (mergedAttrs !== null) {
            setUpAttributes(renderer, native, mergedAttrs);
        }
        /** @type {?} */
        const classes = tNode.classes;
        if (classes !== null) {
            writeDirectClass(renderer, native, classes);
        }
        /** @type {?} */
        const styles = tNode.styles;
        if (styles !== null) {
            writeDirectStyle(renderer, native, styles);
        }
        appendChild(tView, lView, native, tNode);
        // any immediate children of a component or template container must be pre-emptively
        // monkey-patched with the component view data so that the element can be inspected
        // later on using any element discovery utility methods (see `element_discovery.ts`)
        if (getElementDepthCount() === 0) {
            attachPatchData(native, lView);
        }
        increaseElementDepthCount();
        if (isDirectiveHost(tNode)) {
            createDirectivesInstances(tView, lView, tNode);
            executeContentQueries(tView, tNode, lView);
        }
        if (localRefsIndex !== null) {
            saveResolvedLocalsInData(lView, tNode);
        }
    }
    /**
     * Mark the end of the element.
     *
     * \@codeGenApi
     * @return {?}
     */
    function ɵɵelementEnd() {
        /** @type {?} */
        let previousOrParentTNode = getPreviousOrParentTNode();
        ngDevMode && assertDefined(previousOrParentTNode, 'No parent node to close.');
        if (getIsParent()) {
            setIsNotParent();
        }
        else {
            ngDevMode && assertHasParent(getPreviousOrParentTNode());
            previousOrParentTNode = (/** @type {?} */ (previousOrParentTNode.parent));
            setPreviousOrParentTNode(previousOrParentTNode, false);
        }
        /** @type {?} */
        const tNode = previousOrParentTNode;
        ngDevMode && assertNodeType(tNode, 3 /* Element */);
        decreaseElementDepthCount();
        /** @type {?} */
        const tView = getTView();
        if (tView.firstCreatePass) {
            registerPostOrderHooks(tView, previousOrParentTNode);
            if (isContentQueryHost(previousOrParentTNode)) {
                (/** @type {?} */ (tView.queries)).elementEnd(previousOrParentTNode);
            }
        }
        if (tNode.classesWithoutHost != null && hasClassInput(tNode)) {
            setDirectiveInputsWhichShadowsStyling(tView, tNode, getLView(), tNode.classesWithoutHost, true);
        }
        if (tNode.stylesWithoutHost != null && hasStyleInput(tNode)) {
            setDirectiveInputsWhichShadowsStyling(tView, tNode, getLView(), tNode.stylesWithoutHost, false);
        }
    }
    /**
     * Creates an empty element using {\@link elementStart} and {\@link elementEnd}
     *
     * \@codeGenApi
     * @param {?} index Index of the element in the data array
     * @param {?} name Name of the DOM Node
     * @param {?=} attrsIndex Index of the element's attributes in the `consts` array.
     * @param {?=} localRefsIndex Index of the element's local references in the `consts` array.
     *
     * @return {?}
     */
    function ɵɵelement(index, name, attrsIndex, localRefsIndex) {
        ɵɵelementStart(index, name, attrsIndex, localRefsIndex);
        ɵɵelementEnd();
    }
    /**
     * @param {?} tView
     * @param {?} lView
     * @param {?} element
     * @param {?} tNode
     * @param {?} hasDirectives
     * @return {?}
     */
    function warnAboutUnknownElement(tView, lView, element, tNode, hasDirectives) {
        /** @type {?} */
        const schemas = tView.schemas;
        // If `schemas` is set to `null`, that's an indication that this Component was compiled in AOT
        // mode where this check happens at compile time. In JIT mode, `schemas` is always present and
        // defined as an array (as an empty array in case `schemas` field is not defined) and we should
        // execute the check below.
        if (schemas === null)
            return;
        /** @type {?} */
        const tagName = tNode.tagName;
        // If the element matches any directive, it's considered as valid.
        if (!hasDirectives && tagName !== null) {
            // The element is unknown if it's an instance of HTMLUnknownElement or it isn't registered
            // as a custom element. Note that unknown elements with a dash in their name won't be instances
            // of HTMLUnknownElement in browsers that support web components.
            /** @type {?} */
            const isUnknown = 
            // Note that we can't check for `typeof HTMLUnknownElement === 'function'`,
            // because while most browsers return 'function', IE returns 'object'.
            (typeof HTMLUnknownElement !== 'undefined' && HTMLUnknownElement &&
                element instanceof HTMLUnknownElement) ||
                (typeof customElements !== 'undefined' && tagName.indexOf('-') > -1 &&
                    !customElements.get(tagName));
            if (isUnknown && !matchingSchemas(tView, lView, tagName)) {
                /** @type {?} */
                let warning = `'${tagName}' is not a known element:\n`;
                warning += `1. If '${tagName}' is an Angular component, then verify that it is part of this module.\n`;
                if (tagName && tagName.indexOf('-') > -1) {
                    warning += `2. If '${tagName}' is a Web Component then add 'CUSTOM_ELEMENTS_SCHEMA' to the '@NgModule.schemas' of this component to suppress this message.`;
                }
                else {
                    warning +=
                        `2. To allow any element add 'NO_ERRORS_SCHEMA' to the '@NgModule.schemas' of this component.`;
                }
                console.warn(warning);
            }
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/instructions/element_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @param {?} index
     * @param {?} tView
     * @param {?} lView
     * @param {?=} attrsIndex
     * @param {?=} localRefsIndex
     * @return {?}
     */
    function elementContainerStartFirstCreatePass(index, tView, lView, attrsIndex, localRefsIndex) {
        ngDevMode && ngDevMode.firstCreatePass++;
        /** @type {?} */
        const tViewConsts = tView.consts;
        /** @type {?} */
        const attrs = getConstant(tViewConsts, attrsIndex);
        /** @type {?} */
        const tNode = getOrCreateTNode(tView, lView[T_HOST], index, 4 /* ElementContainer */, 'ng-container', attrs);
        // While ng-container doesn't necessarily support styling, we use the style context to identify
        // and execute directives on the ng-container.
        if (attrs !== null) {
            computeStaticStyling(tNode, attrs, true);
        }
        /** @type {?} */
        const localRefs = getConstant(tViewConsts, localRefsIndex);
        resolveDirectives(tView, lView, tNode, localRefs);
        if (tView.queries !== null) {
            tView.queries.elementStart(tView, tNode);
        }
        return tNode;
    }
    /**
     * Creates a logical container for other nodes (<ng-container>) backed by a comment node in the DOM.
     * The instruction must later be followed by `elementContainerEnd()` call.
     *
     * \@codeGenApi
     * @param {?} index Index of the element in the LView array
     * @param {?=} attrsIndex Index of the container attributes in the `consts` array.
     * @param {?=} localRefsIndex Index of the container's local references in the `consts` array.
     *
     * Even if this instruction accepts a set of attributes no actual attribute values are propagated to
     * the DOM (as a comment node can't have attributes). Attributes are here only for directive
     * matching purposes and setting initial inputs of directives.
     *
     * @return {?}
     */
    function ɵɵelementContainerStart(index, attrsIndex, localRefsIndex) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const tView = getTView();
        /** @type {?} */
        const adjustedIndex = index + HEADER_OFFSET;
        ngDevMode && assertDataInRange(lView, adjustedIndex);
        ngDevMode &&
            assertEqual(getBindingIndex(), tView.bindingStartIndex, 'element containers should be created before any bindings');
        /** @type {?} */
        const tNode = tView.firstCreatePass ?
            elementContainerStartFirstCreatePass(index, tView, lView, attrsIndex, localRefsIndex) :
            (/** @type {?} */ (tView.data[adjustedIndex]));
        setPreviousOrParentTNode(tNode, true);
        ngDevMode && ngDevMode.rendererCreateComment++;
        /** @type {?} */
        const native = lView[adjustedIndex] =
            lView[RENDERER].createComment(ngDevMode ? 'ng-container' : '');
        appendChild(tView, lView, native, tNode);
        attachPatchData(native, lView);
        if (isDirectiveHost(tNode)) {
            createDirectivesInstances(tView, lView, tNode);
            executeContentQueries(tView, tNode, lView);
        }
        if (localRefsIndex != null) {
            saveResolvedLocalsInData(lView, tNode);
        }
    }
    /**
     * Mark the end of the <ng-container>.
     *
     * \@codeGenApi
     * @return {?}
     */
    function ɵɵelementContainerEnd() {
        /** @type {?} */
        let previousOrParentTNode = getPreviousOrParentTNode();
        /** @type {?} */
        const tView = getTView();
        if (getIsParent()) {
            setIsNotParent();
        }
        else {
            ngDevMode && assertHasParent(previousOrParentTNode);
            previousOrParentTNode = (/** @type {?} */ (previousOrParentTNode.parent));
            setPreviousOrParentTNode(previousOrParentTNode, false);
        }
        ngDevMode && assertNodeType(previousOrParentTNode, 4 /* ElementContainer */);
        if (tView.firstCreatePass) {
            registerPostOrderHooks(tView, previousOrParentTNode);
            if (isContentQueryHost(previousOrParentTNode)) {
                (/** @type {?} */ (tView.queries)).elementEnd(previousOrParentTNode);
            }
        }
    }
    /**
     * Creates an empty logical container using {\@link elementContainerStart}
     * and {\@link elementContainerEnd}
     *
     * \@codeGenApi
     * @param {?} index Index of the element in the LView array
     * @param {?=} attrsIndex Index of the container attributes in the `consts` array.
     * @param {?=} localRefsIndex Index of the container's local references in the `consts` array.
     *
     * @return {?}
     */
    function ɵɵelementContainer(index, attrsIndex, localRefsIndex) {
        ɵɵelementContainerStart(index, attrsIndex, localRefsIndex);
        ɵɵelementContainerEnd();
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/instructions/get_current_view.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Returns the current OpaqueViewState instance.
     *
     * Used in conjunction with the restoreView() instruction to save a snapshot
     * of the current view and restore it when listeners are invoked. This allows
     * walking the declaration view tree in listeners to get vars from parent views.
     *
     * \@codeGenApi
     * @return {?}
     */
    function ɵɵgetCurrentView() {
        return (/** @type {?} */ ((/** @type {?} */ (getLView()))));
    }

    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * Determine if the argument is shaped like a Promise
     */
    function isPromise$1(obj) {
        // allow any Promise/A+ compliant thenable.
        // It's up to the caller to ensure that obj.then conforms to the spec
        return !!obj && typeof obj.then === 'function';
    }
    /**
     * Determine if the argument is an Observable
     */
    function isObservable$1(obj) {
        // TODO: use isObservable once we update pass rxjs 6.1
        // https://github.com/ReactiveX/rxjs/blob/master/CHANGELOG.md#610-2018-05-03
        return !!obj && typeof obj.subscribe === 'function';
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/instructions/listener.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Adds an event listener to the current node.
     *
     * If an output exists on one of the node's directives, it also subscribes to the output
     * and saves the subscription for later cleanup.
     *
     * \@codeGenApi
     * @param {?} eventName Name of the event
     * @param {?} listenerFn The function to be called when event emits
     * @param {?=} useCapture Whether or not to use capture in event listener
     * @param {?=} eventTargetResolver Function that returns global target information in case this listener
     * should be attached to a global object like window, document or body
     *
     * @return {?}
     */
    function ɵɵlistener(eventName, listenerFn, useCapture = false, eventTargetResolver) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const tView = getTView();
        /** @type {?} */
        const tNode = getPreviousOrParentTNode();
        listenerInternal(tView, lView, lView[RENDERER], tNode, eventName, listenerFn, useCapture, eventTargetResolver);
        return ɵɵlistener;
    }
    /**
     * Registers a synthetic host listener (e.g. `(\@foo.start)`) on a component or directive.
     *
     * This instruction is for compatibility purposes and is designed to ensure that a
     * synthetic host listener (e.g. `\@HostListener('\@foo.start')`) properly gets rendered
     * in the component's renderer. Normally all host listeners are evaluated with the
     * parent component's renderer, but, in the case of animation \@triggers, they need
     * to be evaluated with the sub component's renderer (because that's where the
     * animation triggers are defined).
     *
     * Do not use this instruction as a replacement for `listener`. This instruction
     * only exists to ensure compatibility with the ViewEngine's host binding behavior.
     *
     * \@codeGenApi
     * @param {?} eventName Name of the event
     * @param {?} listenerFn The function to be called when event emits
     * @param {?=} useCapture Whether or not to use capture in event listener
     * @param {?=} eventTargetResolver Function that returns global target information in case this listener
     * should be attached to a global object like window, document or body
     *
     * @return {?}
     */
    function ɵɵcomponentHostSyntheticListener(eventName, listenerFn, useCapture = false, eventTargetResolver) {
        /** @type {?} */
        const tNode = getPreviousOrParentTNode();
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const tView = getTView();
        /** @type {?} */
        const currentDef = getCurrentDirectiveDef(tView.data);
        /** @type {?} */
        const renderer = loadComponentRenderer(currentDef, tNode, lView);
        listenerInternal(tView, lView, renderer, tNode, eventName, listenerFn, useCapture, eventTargetResolver);
        return ɵɵcomponentHostSyntheticListener;
    }
    /**
     * A utility function that checks if a given element has already an event handler registered for an
     * event with a specified name. The TView.cleanup data structure is used to find out which events
     * are registered for a given element.
     * @param {?} tView
     * @param {?} lView
     * @param {?} eventName
     * @param {?} tNodeIdx
     * @return {?}
     */
    function findExistingListener(tView, lView, eventName, tNodeIdx) {
        /** @type {?} */
        const tCleanup = tView.cleanup;
        if (tCleanup != null) {
            for (let i = 0; i < tCleanup.length - 1; i += 2) {
                /** @type {?} */
                const cleanupEventName = tCleanup[i];
                if (cleanupEventName === eventName && tCleanup[i + 1] === tNodeIdx) {
                    // We have found a matching event name on the same node but it might not have been
                    // registered yet, so we must explicitly verify entries in the LView cleanup data
                    // structures.
                    /** @type {?} */
                    const lCleanup = (/** @type {?} */ (lView[CLEANUP]));
                    /** @type {?} */
                    const listenerIdxInLCleanup = tCleanup[i + 2];
                    return lCleanup.length > listenerIdxInLCleanup ? lCleanup[listenerIdxInLCleanup] : null;
                }
                // TView.cleanup can have a mix of 4-elements entries (for event handler cleanups) or
                // 2-element entries (for directive and queries destroy hooks). As such we can encounter
                // blocks of 4 or 2 items in the tView.cleanup and this is why we iterate over 2 elements
                // first and jump another 2 elements if we detect listeners cleanup (4 elements). Also check
                // documentation of TView.cleanup for more details of this data structure layout.
                if (typeof cleanupEventName === 'string') {
                    i += 2;
                }
            }
        }
        return null;
    }
    /**
     * @param {?} tView
     * @param {?} lView
     * @param {?} renderer
     * @param {?} tNode
     * @param {?} eventName
     * @param {?} listenerFn
     * @param {?=} useCapture
     * @param {?=} eventTargetResolver
     * @return {?}
     */
    function listenerInternal(tView, lView, renderer, tNode, eventName, listenerFn, useCapture = false, eventTargetResolver) {
        /** @type {?} */
        const isTNodeDirectiveHost = isDirectiveHost(tNode);
        /** @type {?} */
        const firstCreatePass = tView.firstCreatePass;
        /** @type {?} */
        const tCleanup = firstCreatePass && (tView.cleanup || (tView.cleanup = []));
        // When the ɵɵlistener instruction was generated and is executed we know that there is either a
        // native listener or a directive output on this element. As such we we know that we will have to
        // register a listener and store its cleanup function on LView.
        /** @type {?} */
        const lCleanup = getLCleanup(lView);
        ngDevMode &&
            assertNodeOfPossibleTypes(tNode, 3 /* Element */, 0 /* Container */, 4 /* ElementContainer */);
        /** @type {?} */
        let processOutputs = true;
        // add native event listener - applicable to elements only
        if (tNode.type === 3 /* Element */) {
            /** @type {?} */
            const native = (/** @type {?} */ (getNativeByTNode(tNode, lView)));
            /** @type {?} */
            const resolved = eventTargetResolver ? eventTargetResolver(native) : (/** @type {?} */ (EMPTY_OBJ));
            /** @type {?} */
            const target = resolved.target || native;
            /** @type {?} */
            const lCleanupIndex = lCleanup.length;
            /** @type {?} */
            const idxOrTargetGetter = eventTargetResolver ?
                (/**
                 * @param {?} _lView
                 * @return {?}
                 */
                (_lView) => eventTargetResolver(unwrapRNode(_lView[tNode.index])).target) :
                tNode.index;
            // In order to match current behavior, native DOM event listeners must be added for all
            // events (including outputs).
            if (isProceduralRenderer(renderer)) {
                // There might be cases where multiple directives on the same element try to register an event
                // handler function for the same event. In this situation we want to avoid registration of
                // several native listeners as each registration would be intercepted by NgZone and
                // trigger change detection. This would mean that a single user action would result in several
                // change detections being invoked. To avoid this situation we want to have only one call to
                // native handler registration (for the same element and same type of event).
                //
                // In order to have just one native event handler in presence of multiple handler functions,
                // we just register a first handler function as a native event listener and then chain
                // (coalesce) other handler functions on top of the first native handler function.
                /** @type {?} */
                let existingListener = null;
                // Please note that the coalescing described here doesn't happen for events specifying an
                // alternative target (ex. (document:click)) - this is to keep backward compatibility with the
                // view engine.
                // Also, we don't have to search for existing listeners is there are no directives
                // matching on a given node as we can't register multiple event handlers for the same event in
                // a template (this would mean having duplicate attributes).
                if (!eventTargetResolver && isTNodeDirectiveHost) {
                    existingListener = findExistingListener(tView, lView, eventName, tNode.index);
                }
                if (existingListener !== null) {
                    // Attach a new listener to coalesced listeners list, maintaining the order in which
                    // listeners are registered. For performance reasons, we keep a reference to the last
                    // listener in that list (in `__ngLastListenerFn__` field), so we can avoid going through
                    // the entire set each time we need to add a new listener.
                    /** @type {?} */
                    const lastListenerFn = ((/** @type {?} */ (existingListener))).__ngLastListenerFn__ || existingListener;
                    lastListenerFn.__ngNextListenerFn__ = listenerFn;
                    ((/** @type {?} */ (existingListener))).__ngLastListenerFn__ = listenerFn;
                    processOutputs = false;
                }
                else {
                    // The first argument of `listen` function in Procedural Renderer is:
                    // - either a target name (as a string) in case of global target (window, document, body)
                    // - or element reference (in all other cases)
                    listenerFn = wrapListener(tNode, lView, listenerFn, false /** preventDefault */);
                    /** @type {?} */
                    const cleanupFn = renderer.listen(resolved.name || target, eventName, listenerFn);
                    ngDevMode && ngDevMode.rendererAddEventListener++;
                    lCleanup.push(listenerFn, cleanupFn);
                    tCleanup && tCleanup.push(eventName, idxOrTargetGetter, lCleanupIndex, lCleanupIndex + 1);
                }
            }
            else {
                listenerFn = wrapListener(tNode, lView, listenerFn, true /** preventDefault */);
                target.addEventListener(eventName, listenerFn, useCapture);
                ngDevMode && ngDevMode.rendererAddEventListener++;
                lCleanup.push(listenerFn);
                tCleanup && tCleanup.push(eventName, idxOrTargetGetter, lCleanupIndex, useCapture);
            }
        }
        // subscribe to directive outputs
        /** @type {?} */
        const outputs = tNode.outputs;
        /** @type {?} */
        let props;
        if (processOutputs && outputs !== null && (props = outputs[eventName])) {
            /** @type {?} */
            const propsLength = props.length;
            if (propsLength) {
                for (let i = 0; i < propsLength; i += 2) {
                    /** @type {?} */
                    const index = (/** @type {?} */ (props[i]));
                    ngDevMode && assertDataInRange(lView, index);
                    /** @type {?} */
                    const minifiedName = props[i + 1];
                    /** @type {?} */
                    const directiveInstance = lView[index];
                    /** @type {?} */
                    const output = directiveInstance[minifiedName];
                    if (ngDevMode && !isObservable$1(output)) {
                        throw new Error(`@Output ${minifiedName} not initialized in '${directiveInstance.constructor.name}'.`);
                    }
                    /** @type {?} */
                    const subscription = output.subscribe(listenerFn);
                    /** @type {?} */
                    const idx = lCleanup.length;
                    lCleanup.push(listenerFn, subscription);
                    tCleanup && tCleanup.push(eventName, tNode.index, idx, -(idx + 1));
                }
            }
        }
    }
    /**
     * @param {?} lView
     * @param {?} listenerFn
     * @param {?} e
     * @return {?}
     */
    function executeListenerWithErrorHandling(lView, listenerFn, e) {
        try {
            // Only explicitly returning false from a listener should preventDefault
            return listenerFn(e) !== false;
        }
        catch (error) {
            handleError(lView, error);
            return false;
        }
    }
    /**
     * Wraps an event listener with a function that marks ancestors dirty and prevents default behavior,
     * if applicable.
     *
     * @param {?} tNode The TNode associated with this listener
     * @param {?} lView The LView that contains this listener
     * @param {?} listenerFn The listener function to call
     * @param {?} wrapWithPreventDefault Whether or not to prevent default behavior
     * (the procedural renderer does this already, so in those cases, we should skip)
     * @return {?}
     */
    function wrapListener(tNode, lView, listenerFn, wrapWithPreventDefault) {
        // Note: we are performing most of the work in the listener function itself
        // to optimize listener registration.
        return (/**
         * @param {?} e
         * @return {?}
         */
        function wrapListenerIn_markDirtyAndPreventDefault(e) {
            // Ivy uses `Function` as a special token that allows us to unwrap the function
            // so that it can be invoked programmatically by `DebugNode.triggerEventHandler`.
            if (e === Function) {
                return listenerFn;
            }
            // In order to be backwards compatible with View Engine, events on component host nodes
            // must also mark the component view itself dirty (i.e. the view that it owns).
            /** @type {?} */
            const startView = tNode.flags & 2 /* isComponentHost */ ?
                getComponentLViewByIndex(tNode.index, lView) :
                lView;
            // See interfaces/view.ts for more on LViewFlags.ManualOnPush
            if ((lView[FLAGS] & 32 /* ManualOnPush */) === 0) {
                markViewDirty(startView);
            }
            /** @type {?} */
            let result = executeListenerWithErrorHandling(lView, listenerFn, e);
            // A just-invoked listener function might have coalesced listeners so we need to check for
            // their presence and invoke as needed.
            /** @type {?} */
            let nextListenerFn = ((/** @type {?} */ (wrapListenerIn_markDirtyAndPreventDefault))).__ngNextListenerFn__;
            while (nextListenerFn) {
                // We should prevent default if any of the listeners explicitly return false
                result = executeListenerWithErrorHandling(lView, nextListenerFn, e) && result;
                nextListenerFn = ((/** @type {?} */ (nextListenerFn))).__ngNextListenerFn__;
            }
            if (wrapWithPreventDefault && result === false) {
                e.preventDefault();
                // Necessary for legacy browsers that don't support preventDefault (e.g. IE)
                e.returnValue = false;
            }
            return result;
        });
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/instructions/namespace.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/instructions/next_context.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Retrieves a context at the level specified and saves it as the global, contextViewData.
     * Will get the next level up if level is not specified.
     *
     * This is used to save contexts of parent views so they can be bound in embedded views, or
     * in conjunction with reference() to bind a ref from a parent view.
     *
     * \@codeGenApi
     * @template T
     * @param {?=} level The relative level of the view from which to grab context compared to contextVewData
     * @return {?} context
     *
     */
    function ɵɵnextContext(level = 1) {
        return nextContextImpl(level);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/instructions/projection.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Checks a given node against matching projection slots and returns the
     * determined slot index. Returns "null" if no slot matched the given node.
     *
     * This function takes into account the parsed ngProjectAs selector from the
     * node's attributes. If present, it will check whether the ngProjectAs selector
     * matches any of the projection slot selectors.
     * @param {?} tNode
     * @param {?} projectionSlots
     * @return {?}
     */
    function matchingProjectionSlotIndex(tNode, projectionSlots) {
        /** @type {?} */
        let wildcardNgContentIndex = null;
        /** @type {?} */
        const ngProjectAsAttrVal = getProjectAsAttrValue(tNode);
        for (let i = 0; i < projectionSlots.length; i++) {
            /** @type {?} */
            const slotValue = projectionSlots[i];
            // The last wildcard projection slot should match all nodes which aren't matching
            // any selector. This is necessary to be backwards compatible with view engine.
            if (slotValue === '*') {
                wildcardNgContentIndex = i;
                continue;
            }
            // If we ran into an `ngProjectAs` attribute, we should match its parsed selector
            // to the list of selectors, otherwise we fall back to matching against the node.
            if (ngProjectAsAttrVal === null ?
                isNodeMatchingSelectorList(tNode, slotValue, /* isProjectionMode */ true) :
                isSelectorInSelectorList(ngProjectAsAttrVal, slotValue)) {
                return i; // first matching selector "captures" a given node
            }
        }
        return wildcardNgContentIndex;
    }
    /**
     * Instruction to distribute projectable nodes among <ng-content> occurrences in a given template.
     * It takes all the selectors from the entire component's template and decides where
     * each projected node belongs (it re-distributes nodes among "buckets" where each "bucket" is
     * backed by a selector).
     *
     * This function requires CSS selectors to be provided in 2 forms: parsed (by a compiler) and text,
     * un-parsed form.
     *
     * The parsed form is needed for efficient matching of a node against a given CSS selector.
     * The un-parsed, textual form is needed for support of the ngProjectAs attribute.
     *
     * Having a CSS selector in 2 different formats is not ideal, but alternatives have even more
     * drawbacks:
     * - having only a textual form would require runtime parsing of CSS selectors;
     * - we can't have only a parsed as we can't re-construct textual form from it (as entered by a
     * template author).
     *
     * \@codeGenApi
     * @param {?=} projectionSlots
     * @return {?}
     */
    function ɵɵprojectionDef(projectionSlots) {
        /** @type {?} */
        const componentNode = (/** @type {?} */ (getLView()[DECLARATION_COMPONENT_VIEW][T_HOST]));
        if (!componentNode.projection) {
            // If no explicit projection slots are defined, fall back to a single
            // projection slot with the wildcard selector.
            /** @type {?} */
            const numProjectionSlots = projectionSlots ? projectionSlots.length : 1;
            /** @type {?} */
            const projectionHeads = componentNode.projection =
                newArray(numProjectionSlots, (/** @type {?} */ ((/** @type {?} */ (null)))));
            /** @type {?} */
            const tails = projectionHeads.slice();
            /** @type {?} */
            let componentChild = componentNode.child;
            while (componentChild !== null) {
                /** @type {?} */
                const slotIndex = projectionSlots ? matchingProjectionSlotIndex(componentChild, projectionSlots) : 0;
                if (slotIndex !== null) {
                    if (tails[slotIndex]) {
                        (/** @type {?} */ (tails[slotIndex])).projectionNext = componentChild;
                    }
                    else {
                        projectionHeads[slotIndex] = componentChild;
                    }
                    tails[slotIndex] = componentChild;
                }
                componentChild = componentChild.next;
            }
        }
    }
    /** @type {?} */
    let delayProjection = false;
    /**
     * @param {?} value
     * @return {?}
     */
    function setDelayProjection(value) {
        delayProjection = value;
    }
    /**
     * Inserts previously re-distributed projected nodes. This instruction must be preceded by a call
     * to the projectionDef instruction.
     *
     * \@codeGenApi
     * @param {?} nodeIndex
     * @param {?=} selectorIndex
     * @param {?=} attrs
     * @return {?}
     */
    function ɵɵprojection(nodeIndex, selectorIndex = 0, attrs) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const tView = getTView();
        /** @type {?} */
        const tProjectionNode = getOrCreateTNode(tView, lView[T_HOST], nodeIndex, 1 /* Projection */, null, attrs || null);
        // We can't use viewData[HOST_NODE] because projection nodes can be nested in embedded views.
        if (tProjectionNode.projection === null)
            tProjectionNode.projection = selectorIndex;
        // `<ng-content>` has no content
        setIsNotParent();
        // We might need to delay the projection of nodes if they are in the middle of an i18n block
        if (!delayProjection) {
            // re-distribution of projectable nodes is stored on a component's view level
            applyProjection(tView, lView, tProjectionNode);
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/instructions/property_interpolation.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     *
     * Update an interpolated property on an element with a lone bound value
     *
     * Used when the value passed to a property has 1 interpolated value in it, an no additional text
     * surrounds that interpolated value:
     *
     * ```html
     * <div title="{{v0}}"></div>
     * ```
     *
     * Its compiled representation is::
     *
     * ```ts
     * ɵɵpropertyInterpolate('title', v0);
     * ```
     *
     * If the property name also exists as an input property on one of the element's directives,
     * the component property will be set instead of the element property. This check must
     * be conducted at runtime so child components that add new `\@Inputs` don't have to be re-compiled.
     *
     * \@codeGenApi
     * @param {?} propName The name of the property to update
     * @param {?} v0 Value checked for change.
     * @param {?=} sanitizer An optional sanitizer function
     * @return {?} itself, so that it may be chained.
     */
    function ɵɵpropertyInterpolate(propName, v0, sanitizer) {
        ɵɵpropertyInterpolate1(propName, '', v0, '', sanitizer);
        return ɵɵpropertyInterpolate;
    }
    /**
     *
     * Update an interpolated property on an element with single bound value surrounded by text.
     *
     * Used when the value passed to a property has 1 interpolated value in it:
     *
     * ```html
     * <div title="prefix{{v0}}suffix"></div>
     * ```
     *
     * Its compiled representation is::
     *
     * ```ts
     * ɵɵpropertyInterpolate1('title', 'prefix', v0, 'suffix');
     * ```
     *
     * If the property name also exists as an input property on one of the element's directives,
     * the component property will be set instead of the element property. This check must
     * be conducted at runtime so child components that add new `\@Inputs` don't have to be re-compiled.
     *
     * \@codeGenApi
     * @param {?} propName The name of the property to update
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @param {?=} sanitizer An optional sanitizer function
     * @return {?} itself, so that it may be chained.
     */
    function ɵɵpropertyInterpolate1(propName, prefix, v0, suffix, sanitizer) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation1(lView, prefix, v0, suffix);
        if (interpolatedValue !== NO_CHANGE) {
            /** @type {?} */
            const tView = getTView();
            /** @type {?} */
            const tNode = getSelectedTNode();
            elementPropertyInternal(tView, tNode, lView, propName, interpolatedValue, lView[RENDERER], sanitizer, false);
            ngDevMode &&
                storePropertyBindingMetadata(tView.data, tNode, propName, getBindingIndex() - 1, prefix, suffix);
        }
        return ɵɵpropertyInterpolate1;
    }
    /**
     *
     * Update an interpolated property on an element with 2 bound values surrounded by text.
     *
     * Used when the value passed to a property has 2 interpolated values in it:
     *
     * ```html
     * <div title="prefix{{v0}}-{{v1}}suffix"></div>
     * ```
     *
     * Its compiled representation is::
     *
     * ```ts
     * ɵɵpropertyInterpolate2('title', 'prefix', v0, '-', v1, 'suffix');
     * ```
     *
     * If the property name also exists as an input property on one of the element's directives,
     * the component property will be set instead of the element property. This check must
     * be conducted at runtime so child components that add new `\@Inputs` don't have to be re-compiled.
     *
     * \@codeGenApi
     * @param {?} propName The name of the property to update
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} i0 Static value used for concatenation only.
     * @param {?} v1 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @param {?=} sanitizer An optional sanitizer function
     * @return {?} itself, so that it may be chained.
     */
    function ɵɵpropertyInterpolate2(propName, prefix, v0, i0, v1, suffix, sanitizer) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation2(lView, prefix, v0, i0, v1, suffix);
        if (interpolatedValue !== NO_CHANGE) {
            /** @type {?} */
            const tView = getTView();
            /** @type {?} */
            const tNode = getSelectedTNode();
            elementPropertyInternal(tView, tNode, lView, propName, interpolatedValue, lView[RENDERER], sanitizer, false);
            ngDevMode &&
                storePropertyBindingMetadata(tView.data, tNode, propName, getBindingIndex() - 2, prefix, i0, suffix);
        }
        return ɵɵpropertyInterpolate2;
    }
    /**
     *
     * Update an interpolated property on an element with 3 bound values surrounded by text.
     *
     * Used when the value passed to a property has 3 interpolated values in it:
     *
     * ```html
     * <div title="prefix{{v0}}-{{v1}}-{{v2}}suffix"></div>
     * ```
     *
     * Its compiled representation is::
     *
     * ```ts
     * ɵɵpropertyInterpolate3(
     * 'title', 'prefix', v0, '-', v1, '-', v2, 'suffix');
     * ```
     *
     * If the property name also exists as an input property on one of the element's directives,
     * the component property will be set instead of the element property. This check must
     * be conducted at runtime so child components that add new `\@Inputs` don't have to be re-compiled.
     *
     * \@codeGenApi
     * @param {?} propName The name of the property to update
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} i0 Static value used for concatenation only.
     * @param {?} v1 Value checked for change.
     * @param {?} i1 Static value used for concatenation only.
     * @param {?} v2 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @param {?=} sanitizer An optional sanitizer function
     * @return {?} itself, so that it may be chained.
     */
    function ɵɵpropertyInterpolate3(propName, prefix, v0, i0, v1, i1, v2, suffix, sanitizer) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation3(lView, prefix, v0, i0, v1, i1, v2, suffix);
        if (interpolatedValue !== NO_CHANGE) {
            /** @type {?} */
            const tView = getTView();
            /** @type {?} */
            const tNode = getSelectedTNode();
            elementPropertyInternal(tView, tNode, lView, propName, interpolatedValue, lView[RENDERER], sanitizer, false);
            ngDevMode &&
                storePropertyBindingMetadata(tView.data, tNode, propName, getBindingIndex() - 3, prefix, i0, i1, suffix);
        }
        return ɵɵpropertyInterpolate3;
    }
    /**
     *
     * Update an interpolated property on an element with 4 bound values surrounded by text.
     *
     * Used when the value passed to a property has 4 interpolated values in it:
     *
     * ```html
     * <div title="prefix{{v0}}-{{v1}}-{{v2}}-{{v3}}suffix"></div>
     * ```
     *
     * Its compiled representation is::
     *
     * ```ts
     * ɵɵpropertyInterpolate4(
     * 'title', 'prefix', v0, '-', v1, '-', v2, '-', v3, 'suffix');
     * ```
     *
     * If the property name also exists as an input property on one of the element's directives,
     * the component property will be set instead of the element property. This check must
     * be conducted at runtime so child components that add new `\@Inputs` don't have to be re-compiled.
     *
     * \@codeGenApi
     * @param {?} propName The name of the property to update
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} i0 Static value used for concatenation only.
     * @param {?} v1 Value checked for change.
     * @param {?} i1 Static value used for concatenation only.
     * @param {?} v2 Value checked for change.
     * @param {?} i2 Static value used for concatenation only.
     * @param {?} v3 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @param {?=} sanitizer An optional sanitizer function
     * @return {?} itself, so that it may be chained.
     */
    function ɵɵpropertyInterpolate4(propName, prefix, v0, i0, v1, i1, v2, i2, v3, suffix, sanitizer) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation4(lView, prefix, v0, i0, v1, i1, v2, i2, v3, suffix);
        if (interpolatedValue !== NO_CHANGE) {
            /** @type {?} */
            const tView = getTView();
            /** @type {?} */
            const tNode = getSelectedTNode();
            elementPropertyInternal(tView, tNode, lView, propName, interpolatedValue, lView[RENDERER], sanitizer, false);
            ngDevMode &&
                storePropertyBindingMetadata(tView.data, tNode, propName, getBindingIndex() - 4, prefix, i0, i1, i2, suffix);
        }
        return ɵɵpropertyInterpolate4;
    }
    /**
     *
     * Update an interpolated property on an element with 5 bound values surrounded by text.
     *
     * Used when the value passed to a property has 5 interpolated values in it:
     *
     * ```html
     * <div title="prefix{{v0}}-{{v1}}-{{v2}}-{{v3}}-{{v4}}suffix"></div>
     * ```
     *
     * Its compiled representation is::
     *
     * ```ts
     * ɵɵpropertyInterpolate5(
     * 'title', 'prefix', v0, '-', v1, '-', v2, '-', v3, '-', v4, 'suffix');
     * ```
     *
     * If the property name also exists as an input property on one of the element's directives,
     * the component property will be set instead of the element property. This check must
     * be conducted at runtime so child components that add new `\@Inputs` don't have to be re-compiled.
     *
     * \@codeGenApi
     * @param {?} propName The name of the property to update
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} i0 Static value used for concatenation only.
     * @param {?} v1 Value checked for change.
     * @param {?} i1 Static value used for concatenation only.
     * @param {?} v2 Value checked for change.
     * @param {?} i2 Static value used for concatenation only.
     * @param {?} v3 Value checked for change.
     * @param {?} i3 Static value used for concatenation only.
     * @param {?} v4 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @param {?=} sanitizer An optional sanitizer function
     * @return {?} itself, so that it may be chained.
     */
    function ɵɵpropertyInterpolate5(propName, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, suffix, sanitizer) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation5(lView, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, suffix);
        if (interpolatedValue !== NO_CHANGE) {
            /** @type {?} */
            const tView = getTView();
            /** @type {?} */
            const tNode = getSelectedTNode();
            elementPropertyInternal(tView, tNode, lView, propName, interpolatedValue, lView[RENDERER], sanitizer, false);
            ngDevMode &&
                storePropertyBindingMetadata(tView.data, tNode, propName, getBindingIndex() - 5, prefix, i0, i1, i2, i3, suffix);
        }
        return ɵɵpropertyInterpolate5;
    }
    /**
     *
     * Update an interpolated property on an element with 6 bound values surrounded by text.
     *
     * Used when the value passed to a property has 6 interpolated values in it:
     *
     * ```html
     * <div title="prefix{{v0}}-{{v1}}-{{v2}}-{{v3}}-{{v4}}-{{v5}}suffix"></div>
     * ```
     *
     * Its compiled representation is::
     *
     * ```ts
     * ɵɵpropertyInterpolate6(
     *    'title', 'prefix', v0, '-', v1, '-', v2, '-', v3, '-', v4, '-', v5, 'suffix');
     * ```
     *
     * If the property name also exists as an input property on one of the element's directives,
     * the component property will be set instead of the element property. This check must
     * be conducted at runtime so child components that add new `\@Inputs` don't have to be re-compiled.
     *
     * \@codeGenApi
     * @param {?} propName The name of the property to update
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} i0 Static value used for concatenation only.
     * @param {?} v1 Value checked for change.
     * @param {?} i1 Static value used for concatenation only.
     * @param {?} v2 Value checked for change.
     * @param {?} i2 Static value used for concatenation only.
     * @param {?} v3 Value checked for change.
     * @param {?} i3 Static value used for concatenation only.
     * @param {?} v4 Value checked for change.
     * @param {?} i4 Static value used for concatenation only.
     * @param {?} v5 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @param {?=} sanitizer An optional sanitizer function
     * @return {?} itself, so that it may be chained.
     */
    function ɵɵpropertyInterpolate6(propName, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, suffix, sanitizer) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation6(lView, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, suffix);
        if (interpolatedValue !== NO_CHANGE) {
            /** @type {?} */
            const tView = getTView();
            /** @type {?} */
            const tNode = getSelectedTNode();
            elementPropertyInternal(tView, tNode, lView, propName, interpolatedValue, lView[RENDERER], sanitizer, false);
            ngDevMode &&
                storePropertyBindingMetadata(tView.data, tNode, propName, getBindingIndex() - 6, prefix, i0, i1, i2, i3, i4, suffix);
        }
        return ɵɵpropertyInterpolate6;
    }
    /**
     *
     * Update an interpolated property on an element with 7 bound values surrounded by text.
     *
     * Used when the value passed to a property has 7 interpolated values in it:
     *
     * ```html
     * <div title="prefix{{v0}}-{{v1}}-{{v2}}-{{v3}}-{{v4}}-{{v5}}-{{v6}}suffix"></div>
     * ```
     *
     * Its compiled representation is::
     *
     * ```ts
     * ɵɵpropertyInterpolate7(
     *    'title', 'prefix', v0, '-', v1, '-', v2, '-', v3, '-', v4, '-', v5, '-', v6, 'suffix');
     * ```
     *
     * If the property name also exists as an input property on one of the element's directives,
     * the component property will be set instead of the element property. This check must
     * be conducted at runtime so child components that add new `\@Inputs` don't have to be re-compiled.
     *
     * \@codeGenApi
     * @param {?} propName The name of the property to update
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} i0 Static value used for concatenation only.
     * @param {?} v1 Value checked for change.
     * @param {?} i1 Static value used for concatenation only.
     * @param {?} v2 Value checked for change.
     * @param {?} i2 Static value used for concatenation only.
     * @param {?} v3 Value checked for change.
     * @param {?} i3 Static value used for concatenation only.
     * @param {?} v4 Value checked for change.
     * @param {?} i4 Static value used for concatenation only.
     * @param {?} v5 Value checked for change.
     * @param {?} i5 Static value used for concatenation only.
     * @param {?} v6 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @param {?=} sanitizer An optional sanitizer function
     * @return {?} itself, so that it may be chained.
     */
    function ɵɵpropertyInterpolate7(propName, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, i5, v6, suffix, sanitizer) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation7(lView, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, i5, v6, suffix);
        if (interpolatedValue !== NO_CHANGE) {
            /** @type {?} */
            const tView = getTView();
            /** @type {?} */
            const tNode = getSelectedTNode();
            elementPropertyInternal(tView, tNode, lView, propName, interpolatedValue, lView[RENDERER], sanitizer, false);
            ngDevMode &&
                storePropertyBindingMetadata(tView.data, tNode, propName, getBindingIndex() - 7, prefix, i0, i1, i2, i3, i4, i5, suffix);
        }
        return ɵɵpropertyInterpolate7;
    }
    /**
     *
     * Update an interpolated property on an element with 8 bound values surrounded by text.
     *
     * Used when the value passed to a property has 8 interpolated values in it:
     *
     * ```html
     * <div title="prefix{{v0}}-{{v1}}-{{v2}}-{{v3}}-{{v4}}-{{v5}}-{{v6}}-{{v7}}suffix"></div>
     * ```
     *
     * Its compiled representation is::
     *
     * ```ts
     * ɵɵpropertyInterpolate8(
     *  'title', 'prefix', v0, '-', v1, '-', v2, '-', v3, '-', v4, '-', v5, '-', v6, '-', v7, 'suffix');
     * ```
     *
     * If the property name also exists as an input property on one of the element's directives,
     * the component property will be set instead of the element property. This check must
     * be conducted at runtime so child components that add new `\@Inputs` don't have to be re-compiled.
     *
     * \@codeGenApi
     * @param {?} propName The name of the property to update
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} i0 Static value used for concatenation only.
     * @param {?} v1 Value checked for change.
     * @param {?} i1 Static value used for concatenation only.
     * @param {?} v2 Value checked for change.
     * @param {?} i2 Static value used for concatenation only.
     * @param {?} v3 Value checked for change.
     * @param {?} i3 Static value used for concatenation only.
     * @param {?} v4 Value checked for change.
     * @param {?} i4 Static value used for concatenation only.
     * @param {?} v5 Value checked for change.
     * @param {?} i5 Static value used for concatenation only.
     * @param {?} v6 Value checked for change.
     * @param {?} i6 Static value used for concatenation only.
     * @param {?} v7 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @param {?=} sanitizer An optional sanitizer function
     * @return {?} itself, so that it may be chained.
     */
    function ɵɵpropertyInterpolate8(propName, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, i5, v6, i6, v7, suffix, sanitizer) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation8(lView, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, i5, v6, i6, v7, suffix);
        if (interpolatedValue !== NO_CHANGE) {
            /** @type {?} */
            const tView = getTView();
            /** @type {?} */
            const tNode = getSelectedTNode();
            elementPropertyInternal(tView, tNode, lView, propName, interpolatedValue, lView[RENDERER], sanitizer, false);
            ngDevMode &&
                storePropertyBindingMetadata(tView.data, tNode, propName, getBindingIndex() - 8, prefix, i0, i1, i2, i3, i4, i5, i6, suffix);
        }
        return ɵɵpropertyInterpolate8;
    }
    /**
     * Update an interpolated property on an element with 9 or more bound values surrounded by text.
     *
     * Used when the number of interpolated values exceeds 8.
     *
     * ```html
     * <div
     *  title="prefix{{v0}}-{{v1}}-{{v2}}-{{v3}}-{{v4}}-{{v5}}-{{v6}}-{{v7}}-{{v8}}-{{v9}}suffix"></div>
     * ```
     *
     * Its compiled representation is::
     *
     * ```ts
     * ɵɵpropertyInterpolateV(
     *  'title', ['prefix', v0, '-', v1, '-', v2, '-', v3, '-', v4, '-', v5, '-', v6, '-', v7, '-', v9,
     *  'suffix']);
     * ```
     *
     * If the property name also exists as an input property on one of the element's directives,
     * the component property will be set instead of the element property. This check must
     * be conducted at runtime so child components that add new `\@Inputs` don't have to be re-compiled.
     *
     * \@codeGenApi
     * @param {?} propName The name of the property to update.
     * @param {?} values The collection of values and the strings inbetween those values, beginning with a
     * string prefix and ending with a string suffix.
     * (e.g. `['prefix', value0, '-', value1, '-', value2, ..., value99, 'suffix']`)
     * @param {?=} sanitizer An optional sanitizer function
     * @return {?} itself, so that it may be chained.
     */
    function ɵɵpropertyInterpolateV(propName, values, sanitizer) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolationV(lView, values);
        if (interpolatedValue !== NO_CHANGE) {
            /** @type {?} */
            const tView = getTView();
            /** @type {?} */
            const tNode = getSelectedTNode();
            elementPropertyInternal(tView, tNode, lView, propName, interpolatedValue, lView[RENDERER], sanitizer, false);
            if (ngDevMode) {
                /** @type {?} */
                const interpolationInBetween = [values[0]];
                for (let i = 2; i < values.length; i += 2) {
                    interpolationInBetween.push(values[i]);
                }
                storePropertyBindingMetadata(tView.data, tNode, propName, getBindingIndex() - interpolationInBetween.length + 1, ...interpolationInBetween);
            }
        }
        return ɵɵpropertyInterpolateV;
    }

    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * This file contains reuseable "empty" symbols that can be used as default return values
     * in different parts of the rendering code. Because the same symbols are returned, this
     * allows for identity checks against these values to be consistently used by the framework
     * code.
     */
    const EMPTY_OBJ$1 = {};
    const EMPTY_ARRAY$3 = [];
    // freezing the values prevents any code from accidentally inserting new values in
    if ((typeof ngDevMode === 'undefined' || ngDevMode) && initNgDevMode()) {
        // These property accesses can be ignored because ngDevMode will be set to false
        // when optimizing code and the whole if statement will be dropped.
        // tslint:disable-next-line:no-toplevel-property-access
        Object.freeze(EMPTY_OBJ$1);
        // tslint:disable-next-line:no-toplevel-property-access
        Object.freeze(EMPTY_ARRAY$3);
    }
    /**
     * Insert new `tStyleValue` at `TData` and link existing style bindings such that we maintain linked
     * list of styles and compute the duplicate flag.
     *
     * Note: this function is executed during `firstUpdatePass` only to populate the `TView.data`.
     *
     * The function works by keeping track of `tStylingRange` which contains two pointers pointing to
     * the head/tail of the template portion of the styles.
     *  - if `isHost === false` (we are template) then insertion is at tail of `TStylingRange`
     *  - if `isHost === true` (we are host binding) then insertion is at head of `TStylingRange`
     *
     * @param {?} tData The `TData` to insert into.
     * @param {?} tNode `TNode` associated with the styling element.
     * @param {?} tStylingKeyWithStatic
     * @param {?} index location of where `tStyleValue` should be stored (and linked into list.)
     * @param {?} isHostBinding `true` if the insertion is for a `hostBinding`. (insertion is in front of
     *               template.)
     * @param {?} isClassBinding True if the associated `tStylingKey` as a `class` styling.
     *                       `tNode.classBindings` should be used (or `tNode.styleBindings` otherwise.)
     * @return {?}
     */
    function insertTStylingBinding(tData, tNode, tStylingKeyWithStatic, index, isHostBinding, isClassBinding) {
        ngDevMode && assertFirstUpdatePass(getTView());
        /** @type {?} */
        let tBindings = isClassBinding ? tNode.classBindings : tNode.styleBindings;
        /** @type {?} */
        let tmplHead = getTStylingRangePrev(tBindings);
        /** @type {?} */
        let tmplTail = getTStylingRangeNext(tBindings);
        tData[index] = tStylingKeyWithStatic;
        /** @type {?} */
        let isKeyDuplicateOfStatic = false;
        /** @type {?} */
        let tStylingKey;
        if (Array.isArray(tStylingKeyWithStatic)) {
            // We are case when the `TStylingKey` contains static fields as well.
            /** @type {?} */
            const staticKeyValueArray = (/** @type {?} */ (tStylingKeyWithStatic));
            tStylingKey = staticKeyValueArray[1]; // unwrap.
            // We need to check if our key is present in the static so that we can mark it as duplicate.
            if (tStylingKey === null ||
                keyValueArrayIndexOf(staticKeyValueArray, (/** @type {?} */ (tStylingKey))) > 0) {
                // tStylingKey is present in the statics, need to mark it as duplicate.
                isKeyDuplicateOfStatic = true;
            }
        }
        else {
            tStylingKey = tStylingKeyWithStatic;
        }
        if (isHostBinding) {
            // We are inserting host bindings
            // If we don't have template bindings then `tail` is 0.
            /** @type {?} */
            const hasTemplateBindings = tmplTail !== 0;
            // This is important to know because that means that the `head` can't point to the first
            // template bindings (there are none.) Instead the head points to the tail of the template.
            if (hasTemplateBindings) {
                // template head's "prev" will point to last host binding or to 0 if no host bindings yet
                /** @type {?} */
                const previousNode = getTStylingRangePrev((/** @type {?} */ (tData[tmplHead + 1])));
                tData[index + 1] = toTStylingRange(previousNode, tmplHead);
                // if a host binding has already been registered, we need to update the next of that host
                // binding to point to this one
                if (previousNode !== 0) {
                    // We need to update the template-tail value to point to us.
                    tData[previousNode + 1] =
                        setTStylingRangeNext((/** @type {?} */ (tData[previousNode + 1])), index);
                }
                // The "previous" of the template binding head should point to this host binding
                tData[tmplHead + 1] = setTStylingRangePrev((/** @type {?} */ (tData[tmplHead + 1])), index);
            }
            else {
                tData[index + 1] = toTStylingRange(tmplHead, 0);
                // if a host binding has already been registered, we need to update the next of that host
                // binding to point to this one
                if (tmplHead !== 0) {
                    // We need to update the template-tail value to point to us.
                    tData[tmplHead + 1] = setTStylingRangeNext((/** @type {?} */ (tData[tmplHead + 1])), index);
                }
                // if we don't have template, the head points to template-tail, and needs to be advanced.
                tmplHead = index;
            }
        }
        else {
            // We are inserting in template section.
            // We need to set this binding's "previous" to the current template tail
            tData[index + 1] = toTStylingRange(tmplTail, 0);
            ngDevMode &&
                assertEqual(tmplHead !== 0 && tmplTail === 0, false, 'Adding template bindings after hostBindings is not allowed.');
            if (tmplHead === 0) {
                tmplHead = index;
            }
            else {
                // We need to update the previous value "next" to point to this binding
                tData[tmplTail + 1] = setTStylingRangeNext((/** @type {?} */ (tData[tmplTail + 1])), index);
            }
            tmplTail = index;
        }
        // Now we need to update / compute the duplicates.
        // Starting with our location search towards head (least priority)
        if (isKeyDuplicateOfStatic) {
            tData[index + 1] = setTStylingRangePrevDuplicate((/** @type {?} */ (tData[index + 1])));
        }
        markDuplicates(tData, tStylingKey, index, true);
        markDuplicates(tData, tStylingKey, index, false);
        markDuplicateOfResidualStyling(tNode, tStylingKey, tData, index, isClassBinding);
        tBindings = toTStylingRange(tmplHead, tmplTail);
        if (isClassBinding) {
            tNode.classBindings = tBindings;
        }
        else {
            tNode.styleBindings = tBindings;
        }
    }
    /**
     * Look into the residual styling to see if the current `tStylingKey` is duplicate of residual.
     *
     * @param {?} tNode `TNode` where the residual is stored.
     * @param {?} tStylingKey `TStylingKey` to store.
     * @param {?} tData `TData` associated with the current `LView`.
     * @param {?} index location of where `tStyleValue` should be stored (and linked into list.)
     * @param {?} isClassBinding True if the associated `tStylingKey` as a `class` styling.
     *                       `tNode.classBindings` should be used (or `tNode.styleBindings` otherwise.)
     * @return {?}
     */
    function markDuplicateOfResidualStyling(tNode, tStylingKey, tData, index, isClassBinding) {
        /** @type {?} */
        const residual = isClassBinding ? tNode.residualClasses : tNode.residualStyles;
        if (residual != null /* or undefined */ && typeof tStylingKey == 'string' &&
            keyValueArrayIndexOf(residual, tStylingKey) >= 0) {
            // We have duplicate in the residual so mark ourselves as duplicate.
            tData[index + 1] = setTStylingRangeNextDuplicate((/** @type {?} */ (tData[index + 1])));
        }
    }
    /**
     * Marks `TStyleValue`s as duplicates if another style binding in the list has the same
     * `TStyleValue`.
     *
     * NOTE: this function is intended to be called twice once with `isPrevDir` set to `true` and once
     * with it set to `false` to search both the previous as well as next items in the list.
     *
     * No duplicate case
     * ```
     *   [style.color]
     *   [style.width.px] <<- index
     *   [style.height.px]
     * ```
     *
     * In the above case adding `[style.width.px]` to the existing `[style.color]` produces no
     * duplicates because `width` is not found in any other part of the linked list.
     *
     * Duplicate case
     * ```
     *   [style.color]
     *   [style.width.em]
     *   [style.width.px] <<- index
     * ```
     * In the above case adding `[style.width.px]` will produce a duplicate with `[style.width.em]`
     * because `width` is found in the chain.
     *
     * Map case 1
     * ```
     *   [style.width.px]
     *   [style.color]
     *   [style]  <<- index
     * ```
     * In the above case adding `[style]` will produce a duplicate with any other bindings because
     * `[style]` is a Map and as such is fully dynamic and could produce `color` or `width`.
     *
     * Map case 2
     * ```
     *   [style]
     *   [style.width.px]
     *   [style.color]  <<- index
     * ```
     * In the above case adding `[style.color]` will produce a duplicate because there is already a
     * `[style]` binding which is a Map and as such is fully dynamic and could produce `color` or
     * `width`.
     *
     * NOTE: Once `[style]` (Map) is added into the system all things are mapped as duplicates.
     * NOTE: We use `style` as example, but same logic is applied to `class`es as well.
     *
     * @param {?} tData `TData` where the linked list is stored.
     * @param {?} tStylingKey `TStylingKeyPrimitive` which contains the value to compare to other keys in
     *        the linked list.
     * @param {?} index Starting location in the linked list to search from
     * @param {?} isPrevDir Direction.
     *        - `true` for previous (lower priority);
     *        - `false` for next (higher priority).
     * @param {?} isClassBinding
     * @return {?}
     */
    function markDuplicates(tData, tStylingKey, index, isPrevDir, isClassBinding) {
        /** @type {?} */
        const tStylingAtIndex = (/** @type {?} */ (tData[index + 1]));
        /** @type {?} */
        const isMap = tStylingKey === null;
        /** @type {?} */
        let cursor = isPrevDir ? getTStylingRangePrev(tStylingAtIndex) : getTStylingRangeNext(tStylingAtIndex);
        /** @type {?} */
        let foundDuplicate = false;
        // We keep iterating as long as we have a cursor
        // AND either:
        // - we found what we are looking for, OR
        // - we are a map in which case we have to continue searching even after we find what we were
        //   looking for since we are a wild card and everything needs to be flipped to duplicate.
        while (cursor !== 0 && (foundDuplicate === false || isMap)) {
            ngDevMode && assertDataInRange(tData, cursor);
            /** @type {?} */
            const tStylingValueAtCursor = (/** @type {?} */ (tData[cursor]));
            /** @type {?} */
            const tStyleRangeAtCursor = (/** @type {?} */ (tData[cursor + 1]));
            if (isStylingMatch(tStylingValueAtCursor, tStylingKey)) {
                foundDuplicate = true;
                tData[cursor + 1] = isPrevDir ? setTStylingRangeNextDuplicate(tStyleRangeAtCursor) :
                    setTStylingRangePrevDuplicate(tStyleRangeAtCursor);
            }
            cursor = isPrevDir ? getTStylingRangePrev(tStyleRangeAtCursor) :
                getTStylingRangeNext(tStyleRangeAtCursor);
        }
        if (foundDuplicate) {
            // if we found a duplicate, than mark ourselves.
            tData[index + 1] = isPrevDir ? setTStylingRangePrevDuplicate(tStylingAtIndex) :
                setTStylingRangeNextDuplicate(tStylingAtIndex);
        }
    }
    /**
     * Determines if two `TStylingKey`s are a match.
     *
     * When computing weather a binding contains a duplicate, we need to compare if the instruction
     * `TStylingKey` has a match.
     *
     * Here are examples of `TStylingKey`s which match given `tStylingKeyCursor` is:
     * - `color`
     *    - `color`    // Match another color
     *    - `null`     // That means that `tStylingKey` is a `classMap`/`styleMap` instruction
     *    - `['', 'color', 'other', true]` // wrapped `color` so match
     *    - `['', null, 'other', true]`       // wrapped `null` so match
     *    - `['', 'width', 'color', 'value']` // wrapped static value contains a match on `'color'`
     * - `null`       // `tStylingKeyCursor` always match as it is `classMap`/`styleMap` instruction
     *
     * @param {?} tStylingKeyCursor
     * @param {?} tStylingKey
     * @return {?}
     */
    function isStylingMatch(tStylingKeyCursor, tStylingKey) {
        ngDevMode &&
            assertNotEqual(Array.isArray(tStylingKey), true, 'Expected that \'tStylingKey\' has been unwrapped');
        if (tStylingKeyCursor === null || // If the cursor is `null` it means that we have map at that
            // location so we must assume that we have a match.
            tStylingKey == null || // If `tStylingKey` is `null` then it is a map therefor assume that it
            // contains a match.
            (Array.isArray(tStylingKeyCursor) ? tStylingKeyCursor[1] : tStylingKeyCursor) ===
                tStylingKey // If the keys match explicitly than we are a match.
        ) {
            return true;
        }
        else if (Array.isArray(tStylingKeyCursor) && typeof tStylingKey === 'string') {
            // if we did not find a match, but `tStylingKeyCursor` is `KeyValueArray` that means cursor has
            // statics and we need to check those as well.
            return keyValueArrayIndexOf(tStylingKeyCursor, tStylingKey) >=
                0; // see if we are matching the key
        }
        return false;
    }
    // Global state of the parser. (This makes parser non-reentrant, but that is not an issue)
    /** @type {?} */
    const parserState = {
        textEnd: 0,
        key: 0,
        keyEnd: 0,
        value: 0,
        valueEnd: 0,
    };
    /**
     * Retrieves the last parsed `key` of style.
     * @param {?} text the text to substring the key from.
     * @return {?}
     */
    function getLastParsedKey(text) {
        return text.substring(parserState.key, parserState.keyEnd);
    }
    /**
     * Retrieves the last parsed `value` of style.
     * @param {?} text the text to substring the key from.
     * @return {?}
     */
    function getLastParsedValue(text) {
        return text.substring(parserState.value, parserState.valueEnd);
    }
    /**
     * Initializes `className` string for parsing and parses the first token.
     *
     * This function is intended to be used in this format:
     * ```
     * for (let i = parseClassName(text); i >= 0; i = parseClassNameNext(text, i)) {
     *   const key = getLastParsedKey();
     *   ...
     * }
     * ```
     * @param {?} text `className` to parse
     * @return {?} index where the next invocation of `parseClassNameNext` should resume.
     */
    function parseClassName(text) {
        resetParserState(text);
        return parseClassNameNext(text, consumeWhitespace(text, 0, parserState.textEnd));
    }
    /**
     * Parses next `className` token.
     *
     * This function is intended to be used in this format:
     * ```
     * for (let i = parseClassName(text); i >= 0; i = parseClassNameNext(text, i)) {
     *   const key = getLastParsedKey();
     *   ...
     * }
     * ```
     *
     * @param {?} text `className` to parse
     * @param {?} index where the parsing should resume.
     * @return {?} index where the next invocation of `parseClassNameNext` should resume.
     */
    function parseClassNameNext(text, index) {
        /** @type {?} */
        const end = parserState.textEnd;
        if (end === index) {
            return -1;
        }
        index = parserState.keyEnd = consumeClassToken(text, parserState.key = index, end);
        return consumeWhitespace(text, index, end);
    }
    /**
     * Initializes `cssText` string for parsing and parses the first key/values.
     *
     * This function is intended to be used in this format:
     * ```
     * for (let i = parseStyle(text); i >= 0; i = parseStyleNext(text, i))) {
     *   const key = getLastParsedKey();
     *   const value = getLastParsedValue();
     *   ...
     * }
     * ```
     * @param {?} text `cssText` to parse
     * @return {?} index where the next invocation of `parseStyleNext` should resume.
     */
    function parseStyle(text) {
        resetParserState(text);
        return parseStyleNext(text, consumeWhitespace(text, 0, parserState.textEnd));
    }
    /**
     * Parses the next `cssText` key/values.
     *
     * This function is intended to be used in this format:
     * ```
     * for (let i = parseStyle(text); i >= 0; i = parseStyleNext(text, i))) {
     *   const key = getLastParsedKey();
     *   const value = getLastParsedValue();
     *   ...
     * }
     *
     * @param {?} text `cssText` to parse
     * @param {?} startIndex
     * @return {?} index where the next invocation of `parseStyleNext` should resume.
     */
    function parseStyleNext(text, startIndex) {
        /** @type {?} */
        const end = parserState.textEnd;
        /** @type {?} */
        let index = parserState.key = consumeWhitespace(text, startIndex, end);
        if (end === index) {
            // we reached an end so just quit
            return -1;
        }
        index = parserState.keyEnd = consumeStyleKey(text, index, end);
        index = consumeSeparator(text, index, end, 58 /* COLON */);
        index = parserState.value = consumeWhitespace(text, index, end);
        index = parserState.valueEnd = consumeStyleValue(text, index, end);
        return consumeSeparator(text, index, end, 59 /* SEMI_COLON */);
    }
    /**
     * Reset the global state of the styling parser.
     * @param {?} text The styling text to parse.
     * @return {?}
     */
    function resetParserState(text) {
        parserState.key = 0;
        parserState.keyEnd = 0;
        parserState.value = 0;
        parserState.valueEnd = 0;
        parserState.textEnd = text.length;
    }
    /**
     * Returns index of next non-whitespace character.
     *
     * @param {?} text Text to scan
     * @param {?} startIndex Starting index of character where the scan should start.
     * @param {?} endIndex Ending index of character where the scan should end.
     * @return {?} Index of next non-whitespace character (May be the same as `start` if no whitespace at
     *          that location.)
     */
    function consumeWhitespace(text, startIndex, endIndex) {
        while (startIndex < endIndex && text.charCodeAt(startIndex) <= 32 /* SPACE */) {
            startIndex++;
        }
        return startIndex;
    }
    /**
     * Returns index of last char in class token.
     *
     * @param {?} text Text to scan
     * @param {?} startIndex Starting index of character where the scan should start.
     * @param {?} endIndex Ending index of character where the scan should end.
     * @return {?} Index after last char in class token.
     */
    function consumeClassToken(text, startIndex, endIndex) {
        while (startIndex < endIndex && text.charCodeAt(startIndex) > 32 /* SPACE */) {
            startIndex++;
        }
        return startIndex;
    }
    /**
     * Consumes all of the characters belonging to style key and token.
     *
     * @param {?} text Text to scan
     * @param {?} startIndex Starting index of character where the scan should start.
     * @param {?} endIndex Ending index of character where the scan should end.
     * @return {?} Index after last style key character.
     */
    function consumeStyleKey(text, startIndex, endIndex) {
        /** @type {?} */
        let ch;
        while (startIndex < endIndex &&
            ((ch = text.charCodeAt(startIndex)) === 45 /* DASH */ || ch === 95 /* UNDERSCORE */ ||
                ((ch & -33 /* UPPER_CASE */) >= 65 /* A */ && (ch & -33 /* UPPER_CASE */) <= 90 /* Z */))) {
            startIndex++;
        }
        return startIndex;
    }
    /**
     * Consumes all whitespace and the separator `:` after the style key.
     *
     * @param {?} text Text to scan
     * @param {?} startIndex Starting index of character where the scan should start.
     * @param {?} endIndex Ending index of character where the scan should end.
     * @param {?} separator
     * @return {?} Index after separator and surrounding whitespace.
     */
    function consumeSeparator(text, startIndex, endIndex, separator) {
        startIndex = consumeWhitespace(text, startIndex, endIndex);
        if (startIndex < endIndex) {
            if (ngDevMode && text.charCodeAt(startIndex) !== separator) {
                malformedStyleError(text, String.fromCharCode(separator), startIndex);
            }
            startIndex++;
        }
        return startIndex;
    }
    /**
     * Consumes style value honoring `url()` and `""` text.
     *
     * @param {?} text Text to scan
     * @param {?} startIndex Starting index of character where the scan should start.
     * @param {?} endIndex Ending index of character where the scan should end.
     * @return {?} Index after last style value character.
     */
    function consumeStyleValue(text, startIndex, endIndex) {
        /** @type {?} */
        let ch1 = -1;
        // 1st previous character
        /** @type {?} */
        let ch2 = -1;
        // 2nd previous character
        /** @type {?} */
        let ch3 = -1;
        // 3rd previous character
        /** @type {?} */
        let i = startIndex;
        /** @type {?} */
        let lastChIndex = i;
        while (i < endIndex) {
            /** @type {?} */
            const ch = text.charCodeAt(i++);
            if (ch === 59 /* SEMI_COLON */) {
                return lastChIndex;
            }
            else if (ch === 34 /* DOUBLE_QUOTE */ || ch === 39 /* SINGLE_QUOTE */) {
                lastChIndex = i = consumeQuotedText(text, ch, i, endIndex);
            }
            else if (startIndex ===
                i - 4 && // We have seen only 4 characters so far "URL(" (Ignore "foo_URL()")
                ch3 === 85 /* U */ &&
                ch2 === 82 /* R */ && ch1 === 76 /* L */ && ch === 40 /* OPEN_PAREN */) {
                lastChIndex = i = consumeQuotedText(text, 41 /* CLOSE_PAREN */, i, endIndex);
            }
            else if (ch > 32 /* SPACE */) {
                // if we have a non-whitespace character then capture its location
                lastChIndex = i;
            }
            ch3 = ch2;
            ch2 = ch1;
            ch1 = ch & -33 /* UPPER_CASE */;
        }
        return lastChIndex;
    }
    /**
     * Consumes all of the quoted characters.
     *
     * @param {?} text Text to scan
     * @param {?} quoteCharCode CharCode of either `"` or `'` quote or `)` for `url(...)`.
     * @param {?} startIndex Starting index of character where the scan should start.
     * @param {?} endIndex Ending index of character where the scan should end.
     * @return {?} Index after quoted characters.
     */
    function consumeQuotedText(text, quoteCharCode, startIndex, endIndex) {
        /** @type {?} */
        let ch1 = -1;
        // 1st previous character
        /** @type {?} */
        let index = startIndex;
        while (index < endIndex) {
            /** @type {?} */
            const ch = text.charCodeAt(index++);
            if (ch == quoteCharCode && ch1 !== 92 /* BACK_SLASH */) {
                return index;
            }
            if (ch == 92 /* BACK_SLASH */ && ch1 === 92 /* BACK_SLASH */) {
                // two back slashes cancel each other out. For example `"\\"` should properly end the
                // quotation. (It should not assume that the last `"` is escaped.)
                ch1 = 0;
            }
            else {
                ch1 = ch;
            }
        }
        throw ngDevMode ? malformedStyleError(text, String.fromCharCode(quoteCharCode), endIndex) :
            new Error();
    }
    /**
     * @param {?} text
     * @param {?} expecting
     * @param {?} index
     * @return {?}
     */
    function malformedStyleError(text, expecting, index) {
        ngDevMode && assertEqual(typeof text === 'string', true, 'String expected here');
        throw throwError$1(`Malformed style at location ${index} in string '` + text.substring(0, index) + '[>>' +
            text.substring(index, index + 1) + '<<]' + text.substr(index + 1) +
            `'. Expecting '${expecting}'.`);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/instructions/styling.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Sets the current style sanitizer function which will then be used
     * within all follow-up prop and map-based style binding instructions
     * for the given element.
     *
     * Note that once styling has been applied to the element (i.e. once
     * `advance(n)` is executed or the hostBindings/template function exits)
     * then the active `sanitizerFn` will be set to `null`. This means that
     * once styling is applied to another element then a another call to
     * `styleSanitizer` will need to be made.
     *
     * \@codeGenApi
     * @param {?} sanitizer
     * @return {?}
     */
    function ɵɵstyleSanitizer(sanitizer) {
        setCurrentStyleSanitizer(sanitizer);
    }
    /**
     * Update a style binding on an element with the provided value.
     *
     * If the style value is falsy then it will be removed from the element
     * (or assigned a different value depending if there are any styles placed
     * on the element with `styleMap` or any static styles that are
     * present from when the element was created with `styling`).
     *
     * Note that the styling element is updated as part of `stylingApply`.
     *
     * \@codeGenApi
     * @param {?} prop A valid CSS property.
     * @param {?} value New value to write (`null` or an empty string to remove).
     * @param {?=} suffix Optional suffix. Used with scalar values to add unit such as `px`.
     *        Note that when a suffix is provided then the underlying sanitizer will
     *        be ignored.
     *
     * Note that this will apply the provided style value to the host element if this function is called
     * within a host binding function.
     *
     * @return {?}
     */
    function ɵɵstyleProp(prop, value, suffix) {
        checkStylingProperty(prop, value, suffix, false);
        return ɵɵstyleProp;
    }
    /**
     * Update a class binding on an element with the provided value.
     *
     * This instruction is meant to handle the `[class.foo]="exp"` case and,
     * therefore, the class binding itself must already be allocated using
     * `styling` within the creation block.
     *
     * \@codeGenApi
     * @param {?} className
     * @param {?} value A true/false value which will turn the class on or off.
     *
     * Note that this will apply the provided class value to the host element if this function
     * is called within a host binding function.
     *
     * @return {?}
     */
    function ɵɵclassProp(className, value) {
        checkStylingProperty(className, value, null, true);
        return ɵɵclassProp;
    }
    /**
     * Update style bindings using an object literal on an element.
     *
     * This instruction is meant to apply styling via the `[style]="exp"` template bindings.
     * When styles are applied to the element they will then be updated with respect to
     * any styles/classes set via `styleProp`. If any styles are set to falsy
     * then they will be removed from the element.
     *
     * Note that the styling instruction will not be applied until `stylingApply` is called.
     *
     * \@codeGenApi
     * @param {?} styles A key/value style map of the styles that will be applied to the given element.
     *        Any missing styles (that have already been applied to the element beforehand) will be
     *        removed (unset) from the element's styling.
     *
     * Note that this will apply the provided styleMap value to the host element if this function
     * is called within a host binding.
     *
     * @return {?}
     */
    function ɵɵstyleMap(styles) {
        checkStylingMap(styleKeyValueArraySet, styleStringParser, styles, false);
    }
    /**
     * Parse text as style and add values to KeyValueArray.
     *
     * This code is pulled out to a separate function so that it can be tree shaken away if it is not
     * needed. It is only referenced from `ɵɵstyleMap`.
     *
     * @param {?} keyValueArray KeyValueArray to add parsed values to.
     * @param {?} text text to parse.
     * @return {?}
     */
    function styleStringParser(keyValueArray, text) {
        for (let i = parseStyle(text); i >= 0; i = parseStyleNext(text, i)) {
            styleKeyValueArraySet(keyValueArray, getLastParsedKey(text), getLastParsedValue(text));
        }
    }
    /**
     * Update class bindings using an object literal or class-string on an element.
     *
     * This instruction is meant to apply styling via the `[class]="exp"` template bindings.
     * When classes are applied to the element they will then be updated with
     * respect to any styles/classes set via `classProp`. If any
     * classes are set to falsy then they will be removed from the element.
     *
     * Note that the styling instruction will not be applied until `stylingApply` is called.
     * Note that this will the provided classMap value to the host element if this function is called
     * within a host binding.
     *
     * \@codeGenApi
     * @param {?} classes A key/value map or string of CSS classes that will be added to the
     *        given element. Any missing classes (that have already been applied to the element
     *        beforehand) will be removed (unset) from the element's list of CSS classes.
     *
     * @return {?}
     */
    function ɵɵclassMap(classes) {
        checkStylingMap(keyValueArraySet, classStringParser, classes, true);
    }
    /**
     * Parse text as class and add values to KeyValueArray.
     *
     * This code is pulled out to a separate function so that it can be tree shaken away if it is not
     * needed. It is only referenced from `ɵɵclassMap`.
     *
     * @param {?} keyValueArray KeyValueArray to add parsed values to.
     * @param {?} text text to parse.
     * @return {?}
     */
    function classStringParser(keyValueArray, text) {
        for (let i = parseClassName(text); i >= 0; i = parseClassNameNext(text, i)) {
            keyValueArraySet(keyValueArray, getLastParsedKey(text), true);
        }
    }
    /**
     * Common code between `ɵɵclassProp` and `ɵɵstyleProp`.
     *
     * @param {?} prop property name.
     * @param {?} value binding value.
     * @param {?} suffixOrSanitizer suffix or sanitization function
     * @param {?} isClassBased `true` if `class` change (`false` if `style`)
     * @return {?}
     */
    function checkStylingProperty(prop, value, suffixOrSanitizer, isClassBased) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const tView = getTView();
        // Styling instructions use 2 slots per binding.
        // 1. one for the value / TStylingKey
        // 2. one for the intermittent-value / TStylingRange
        /** @type {?} */
        const bindingIndex = incrementBindingIndex(2);
        if (tView.firstUpdatePass) {
            stylingFirstUpdatePass(tView, prop, bindingIndex, isClassBased);
        }
        if (value !== NO_CHANGE && bindingUpdated(lView, bindingIndex, value)) {
            // This is a work around. Once PR#34480 lands the sanitizer is passed explicitly and this line
            // can be removed.
            /** @type {?} */
            let styleSanitizer;
            if (suffixOrSanitizer == null) {
                if (styleSanitizer = getCurrentStyleSanitizer()) {
                    suffixOrSanitizer = (/** @type {?} */ (styleSanitizer));
                }
            }
            /** @type {?} */
            const tNode = (/** @type {?} */ (tView.data[getSelectedIndex() + HEADER_OFFSET]));
            updateStyling(tView, tNode, lView, lView[RENDERER], prop, lView[bindingIndex + 1] = normalizeAndApplySuffixOrSanitizer(value, suffixOrSanitizer), isClassBased, bindingIndex);
        }
    }
    /**
     * Common code between `ɵɵclassMap` and `ɵɵstyleMap`.
     *
     * @param {?} keyValueArraySet (See `keyValueArraySet` in "util/array_utils") Gets passed in as a
     * function so that
     *        `style` can pass in version which does sanitization. This is done for tree shaking
     *        purposes.
     * @param {?} stringParser Parser used to parse `value` if `string`. (Passed in as `style` and `class`
     *        have different parsers.)
     * @param {?} value bound value from application
     * @param {?} isClassBased `true` if `class` change (`false` if `style`)
     * @return {?}
     */
    function checkStylingMap(keyValueArraySet, stringParser, value, isClassBased) {
        /** @type {?} */
        const tView = getTView();
        /** @type {?} */
        const bindingIndex = incrementBindingIndex(2);
        if (tView.firstUpdatePass) {
            stylingFirstUpdatePass(tView, null, bindingIndex, isClassBased);
        }
        /** @type {?} */
        const lView = getLView();
        if (value !== NO_CHANGE && bindingUpdated(lView, bindingIndex, value)) {
            // `getSelectedIndex()` should be here (rather than in instruction) so that it is guarded by the
            // if so as not to read unnecessarily.
            /** @type {?} */
            const tNode = (/** @type {?} */ (tView.data[getSelectedIndex() + HEADER_OFFSET]));
            if (hasStylingInputShadow(tNode, isClassBased) && !isInHostBindings(tView, bindingIndex)) {
                if (ngDevMode) {
                    // verify that if we are shadowing then `TData` is appropriately marked so that we skip
                    // processing this binding in styling resolution.
                    /** @type {?} */
                    const tStylingKey = tView.data[bindingIndex];
                    assertEqual(Array.isArray(tStylingKey) ? tStylingKey[1] : tStylingKey, false, 'Styling linked list shadow input should be marked as \'false\'');
                }
                // VE does not concatenate the static portion like we are doing here.
                // Instead VE just ignores the static completely if dynamic binding is present.
                // Because of locality we have already set the static portion because we don't know if there
                // is a dynamic portion until later. If we would ignore the static portion it would look like
                // the binding has removed it. This would confuse `[ngStyle]`/`[ngClass]` to do the wrong
                // thing as it would think that the static portion was removed. For this reason we
                // concatenate it so that `[ngStyle]`/`[ngClass]`  can continue to work on changed.
                /** @type {?} */
                let staticPrefix = isClassBased ? tNode.classesWithoutHost : tNode.stylesWithoutHost;
                ngDevMode && isClassBased === false && staticPrefix !== null &&
                    assertEqual(staticPrefix.endsWith(';'), true, 'Expecting static portion to end with \';\'');
                if (staticPrefix !== null) {
                    // We want to make sure that falsy values of `value` become empty strings.
                    value = concatStringsWithSpace(staticPrefix, value ? value : '');
                }
                // Given `<div [style] my-dir>` such that `my-dir` has `@Input('style')`.
                // This takes over the `[style]` binding. (Same for `[class]`)
                setDirectiveInputsWhichShadowsStyling(tView, tNode, lView, value, isClassBased);
            }
            else {
                updateStylingMap(tView, tNode, lView, lView[RENDERER], lView[bindingIndex + 1], lView[bindingIndex + 1] = toStylingKeyValueArray(keyValueArraySet, stringParser, value), isClassBased, bindingIndex);
            }
        }
    }
    /**
     * Determines when the binding is in `hostBindings` section
     *
     * @param {?} tView Current `TView`
     * @param {?} bindingIndex index of binding which we would like if it is in `hostBindings`
     * @return {?}
     */
    function isInHostBindings(tView, bindingIndex) {
        // All host bindings are placed after the expando section.
        return bindingIndex >= tView.expandoStartIndex;
    }
    /**
     * Collects the necessary information to insert the binding into a linked list of style bindings
     * using `insertTStylingBinding`.
     *
     * @param {?} tView `TView` where the binding linked list will be stored.
     * @param {?} tStylingKey Property/key of the binding.
     * @param {?} bindingIndex Index of binding associated with the `prop`
     * @param {?} isClassBased `true` if `class` change (`false` if `style`)
     * @return {?}
     */
    function stylingFirstUpdatePass(tView, tStylingKey, bindingIndex, isClassBased) {
        ngDevMode && assertFirstUpdatePass(tView);
        /** @type {?} */
        const tData = tView.data;
        if (tData[bindingIndex + 1] === null) {
            // The above check is necessary because we don't clear first update pass until first successful
            // (no exception) template execution. This prevents the styling instruction from double adding
            // itself to the list.
            // `getSelectedIndex()` should be here (rather than in instruction) so that it is guarded by the
            // if so as not to read unnecessarily.
            /** @type {?} */
            const tNode = (/** @type {?} */ (tData[getSelectedIndex() + HEADER_OFFSET]));
            /** @type {?} */
            const isHostBindings = isInHostBindings(tView, bindingIndex);
            if (hasStylingInputShadow(tNode, isClassBased) && tStylingKey === null && !isHostBindings) {
                // `tStylingKey === null` implies that we are either `[style]` or `[class]` binding.
                // If there is a directive which uses `@Input('style')` or `@Input('class')` than
                // we need to neutralize this binding since that directive is shadowing it.
                // We turn this into a noop by setting the key to `false`
                tStylingKey = false;
            }
            tStylingKey = wrapInStaticStylingKey(tData, tNode, tStylingKey, isClassBased);
            insertTStylingBinding(tData, tNode, tStylingKey, bindingIndex, isHostBindings, isClassBased);
        }
    }
    /**
     * Adds static styling information to the binding if applicable.
     *
     * The linked list of styles not only stores the list and keys, but also stores static styling
     * information on some of the keys. This function determines if the key should contain the styling
     * information and computes it.
     *
     * See `TStylingStatic` for more details.
     *
     * @param {?} tData `TData` where the linked list is stored.
     * @param {?} tNode `TNode` for which the styling is being computed.
     * @param {?} stylingKey `TStylingKeyPrimitive` which may need to be wrapped into `TStylingKey`
     * @param {?} isClassBased `true` if `class` (`false` if `style`)
     * @return {?}
     */
    function wrapInStaticStylingKey(tData, tNode, stylingKey, isClassBased) {
        /** @type {?} */
        const hostDirectiveDef = getCurrentDirectiveDef(tData);
        /** @type {?} */
        let residual = isClassBased ? tNode.residualClasses : tNode.residualStyles;
        if (hostDirectiveDef === null) {
            // We are in template node.
            // If template node already had styling instruction then it has already collected the static
            // styling and there is no need to collect them again. We know that we are the first styling
            // instruction because the `TNode.*Bindings` points to 0 (nothing has been inserted yet).
            /** @type {?} */
            const isFirstStylingInstructionInTemplate = (/** @type {?} */ ((/** @type {?} */ ((isClassBased ? tNode.classBindings : tNode.styleBindings))))) === 0;
            if (isFirstStylingInstructionInTemplate) {
                // It would be nice to be able to get the statics from `mergeAttrs`, however, at this point
                // they are already merged and it would not be possible to figure which property belongs where
                // in the priority.
                stylingKey = collectStylingFromDirectives(null, tData, tNode, stylingKey, isClassBased);
                stylingKey = collectStylingFromTAttrs(stylingKey, tNode.attrs, isClassBased);
                // We know that if we have styling binding in template we can't have residual.
                residual = null;
            }
        }
        else {
            // We are in host binding node and there was no binding instruction in template node.
            // This means that we need to compute the residual.
            /** @type {?} */
            const directiveStylingLast = tNode.directiveStylingLast;
            /** @type {?} */
            const isFirstStylingInstructionInHostBinding = directiveStylingLast === -1 || tData[directiveStylingLast] !== hostDirectiveDef;
            if (isFirstStylingInstructionInHostBinding) {
                stylingKey =
                    collectStylingFromDirectives(hostDirectiveDef, tData, tNode, stylingKey, isClassBased);
                if (residual === null) {
                    // - If `null` than either:
                    //    - Template styling instruction already ran and it has consumed the static
                    //      styling into its `TStylingKey` and so there is no need to update residual. Instead
                    //      we need to update the `TStylingKey` associated with the first template node
                    //      instruction. OR
                    //    - Some other styling instruction ran and determined that there are no residuals
                    /** @type {?} */
                    let templateStylingKey = getTemplateHeadTStylingKey(tData, tNode, isClassBased);
                    if (templateStylingKey !== undefined && Array.isArray(templateStylingKey)) {
                        // Only recompute if `templateStylingKey` had static values. (If no static value found
                        // then there is nothing to do since this operation can only produce less static keys, not
                        // more.)
                        templateStylingKey = collectStylingFromDirectives(null, tData, tNode, templateStylingKey[1] /* unwrap previous statics */, isClassBased);
                        templateStylingKey =
                            collectStylingFromTAttrs(templateStylingKey, tNode.attrs, isClassBased);
                        setTemplateHeadTStylingKey(tData, tNode, isClassBased, templateStylingKey);
                    }
                }
                else {
                    // We only need to recompute residual if it is not `null`.
                    // - If existing residual (implies there was no template styling). This means that some of
                    //   the statics may have moved from the residual to the `stylingKey` and so we have to
                    //   recompute.
                    // - If `undefined` this is the first time we are running.
                    residual = collectResidual(tData, tNode, isClassBased);
                }
            }
        }
        if (residual !== undefined) {
            isClassBased ? (tNode.residualClasses = residual) : (tNode.residualStyles = residual);
        }
        return stylingKey;
    }
    /**
     * Retrieve the `TStylingKey` for the template styling instruction.
     *
     * This is needed since `hostBinding` styling instructions are inserted after the template
     * instruction. While the template instruction needs to update the residual in `TNode` the
     * `hostBinding` instructions need to update the `TStylingKey` of the template instruction because
     * the template instruction is downstream from the `hostBindings` instructions.
     *
     * @param {?} tData `TData` where the linked list is stored.
     * @param {?} tNode `TNode` for which the styling is being computed.
     * @param {?} isClassBased `true` if `class` (`false` if `style`)
     * @return {?} `TStylingKey` if found or `undefined` if not found.
     */
    function getTemplateHeadTStylingKey(tData, tNode, isClassBased) {
        /** @type {?} */
        const bindings = isClassBased ? tNode.classBindings : tNode.styleBindings;
        if (getTStylingRangeNext(bindings) === 0) {
            // There does not seem to be a styling instruction in the `template`.
            return undefined;
        }
        return (/** @type {?} */ (tData[getTStylingRangePrev(bindings)]));
    }
    /**
     * Update the `TStylingKey` of the first template instruction in `TNode`.
     *
     * Logically `hostBindings` styling instructions are of lower priority than that of the template.
     * However, they execute after the template styling instructions. This means that they get inserted
     * in front of the template styling instructions.
     *
     * If we have a template styling instruction and a new `hostBindings` styling instruction is
     * executed it means that it may need to steal static fields from the template instruction. This
     * method allows us to update the first template instruction `TStylingKey` with a new value.
     *
     * Assume:
     * ```
     * <div my-dir style="color: red" [style.color]="tmplExp"></div>
     *
     * \@Directive({
     *   host: {
     *     'style': 'width: 100px',
     *     '[style.color]': 'dirExp',
     *   }
     * })
     * class MyDir {}
     * ```
     *
     * when `[style.color]="tmplExp"` executes it creates this data structure.
     * ```
     *  ['', 'color', 'color', 'red', 'width', '100px'],
     * ```
     *
     * The reason for this is that the template instruction does not know if there are styling
     * instructions and must assume that there are none and must collect all of the static styling.
     * (both
     * `color' and 'width`)
     *
     * When `'[style.color]': 'dirExp',` executes we need to insert a new data into the linked list.
     * ```
     *  ['', 'color', 'width', '100px'],  // newly inserted
     *  ['', 'color', 'color', 'red', 'width', '100px'], // this is wrong
     * ```
     *
     * Notice that the template statics is now wrong as it incorrectly contains `width` so we need to
     * update it like so:
     * ```
     *  ['', 'color', 'width', '100px'],
     *  ['', 'color', 'color', 'red'],    // UPDATE
     * ```
     *
     * @param {?} tData `TData` where the linked list is stored.
     * @param {?} tNode `TNode` for which the styling is being computed.
     * @param {?} isClassBased `true` if `class` (`false` if `style`)
     * @param {?} tStylingKey New `TStylingKey` which is replacing the old one.
     * @return {?}
     */
    function setTemplateHeadTStylingKey(tData, tNode, isClassBased, tStylingKey) {
        /** @type {?} */
        const bindings = isClassBased ? tNode.classBindings : tNode.styleBindings;
        ngDevMode &&
            assertNotEqual(getTStylingRangeNext(bindings), 0, 'Expecting to have at least one template styling binding.');
        tData[getTStylingRangePrev(bindings)] = tStylingKey;
    }
    /**
     * Collect all static values after the current `TNode.directiveStylingLast` index.
     *
     * Collect the remaining styling information which has not yet been collected by an existing
     * styling instruction.
     *
     * @param {?} tData `TData` where the `DirectiveDefs` are stored.
     * @param {?} tNode `TNode` which contains the directive range.
     * @param {?} isClassBased `true` if `class` (`false` if `style`)
     * @return {?}
     */
    function collectResidual(tData, tNode, isClassBased) {
        /** @type {?} */
        let residual = undefined;
        /** @type {?} */
        const directiveEnd = tNode.directiveEnd;
        ngDevMode &&
            assertNotEqual(tNode.directiveStylingLast, -1, 'By the time this function gets called at least one hostBindings-node styling instruction must have executed.');
        // We add `1 + tNode.directiveStart` because we need to skip the current directive (as we are
        // collecting things after the last `hostBindings` directive which had a styling instruction.)
        for (let i = 1 + tNode.directiveStylingLast; i < directiveEnd; i++) {
            /** @type {?} */
            const attrs = ((/** @type {?} */ (tData[i]))).hostAttrs;
            residual = (/** @type {?} */ (collectStylingFromTAttrs(residual, attrs, isClassBased)));
        }
        return (/** @type {?} */ (collectStylingFromTAttrs(residual, tNode.attrs, isClassBased)));
    }
    /**
     * Collect the static styling information with lower priority than `hostDirectiveDef`.
     *
     * (This is opposite of residual styling.)
     *
     * @param {?} hostDirectiveDef `DirectiveDef` for which we want to collect lower priority static
     *        styling. (Or `null` if template styling)
     * @param {?} tData `TData` where the linked list is stored.
     * @param {?} tNode `TNode` for which the styling is being computed.
     * @param {?} stylingKey Existing `TStylingKey` to update or wrap.
     * @param {?} isClassBased `true` if `class` (`false` if `style`)
     * @return {?}
     */
    function collectStylingFromDirectives(hostDirectiveDef, tData, tNode, stylingKey, isClassBased) {
        // We need to loop because there can be directives which have `hostAttrs` but don't have
        // `hostBindings` so this loop catches up to the current directive..
        /** @type {?} */
        let currentDirective = null;
        /** @type {?} */
        const directiveEnd = tNode.directiveEnd;
        /** @type {?} */
        let directiveStylingLast = tNode.directiveStylingLast;
        if (directiveStylingLast === -1) {
            directiveStylingLast = tNode.directiveStart;
        }
        else {
            directiveStylingLast++;
        }
        while (directiveStylingLast < directiveEnd) {
            currentDirective = (/** @type {?} */ (tData[directiveStylingLast]));
            ngDevMode && assertDefined(currentDirective, 'expected to be defined');
            stylingKey = collectStylingFromTAttrs(stylingKey, currentDirective.hostAttrs, isClassBased);
            if (currentDirective === hostDirectiveDef)
                break;
            directiveStylingLast++;
        }
        if (hostDirectiveDef !== null) {
            // we only advance the styling cursor if we are collecting data from host bindings.
            // Template executes before host bindings and so if we would update the index,
            // host bindings would not get their statics.
            tNode.directiveStylingLast = directiveStylingLast;
        }
        return stylingKey;
    }
    /**
     * Convert `TAttrs` into `TStylingStatic`.
     *
     * @param {?} stylingKey existing `TStylingKey` to update or wrap.
     * @param {?} attrs `TAttributes` to process.
     * @param {?} isClassBased `true` if `class` (`false` if `style`)
     * @return {?}
     */
    function collectStylingFromTAttrs(stylingKey, attrs, isClassBased) {
        /** @type {?} */
        const desiredMarker = isClassBased ? 1 /* Classes */ : 2 /* Styles */;
        /** @type {?} */
        let currentMarker = -1 /* ImplicitAttributes */;
        if (attrs !== null) {
            for (let i = 0; i < attrs.length; i++) {
                /** @type {?} */
                const item = (/** @type {?} */ (attrs[i]));
                if (typeof item === 'number') {
                    currentMarker = item;
                }
                else {
                    if (currentMarker === desiredMarker) {
                        if (!Array.isArray(stylingKey)) {
                            stylingKey = stylingKey === undefined ? [] : (/** @type {?} */ (['', stylingKey]));
                        }
                        keyValueArraySet((/** @type {?} */ (stylingKey)), item, isClassBased ? true : attrs[++i]);
                    }
                }
            }
        }
        return stylingKey === undefined ? null : stylingKey;
    }
    /**
     * Convert user input to `KeyValueArray`.
     *
     * This function takes user input which could be `string`, Object literal, or iterable and converts
     * it into a consistent representation. The output of this is `KeyValueArray` (which is an array
     * where
     * even indexes contain keys and odd indexes contain values for those keys).
     *
     * The advantage of converting to `KeyValueArray` is that we can perform diff in an input
     * independent
     * way.
     * (ie we can compare `foo bar` to `['bar', 'baz'] and determine a set of changes which need to be
     * applied)
     *
     * The fact that `KeyValueArray` is sorted is very important because it allows us to compute the
     * difference in linear fashion without the need to allocate any additional data.
     *
     * For example if we kept this as a `Map` we would have to iterate over previous `Map` to determine
     * which values need to be deleted, over the new `Map` to determine additions, and we would have to
     * keep additional `Map` to keep track of duplicates or items which have not yet been visited.
     *
     * @param {?} keyValueArraySet (See `keyValueArraySet` in "util/array_utils") Gets passed in as a
     * function so that
     *        `style` can pass in version which does sanitization. This is done for tree shaking
     *        purposes.
     * @param {?} stringParser The parser is passed in so that it will be tree shakable. See
     *        `styleStringParser` and `classStringParser`
     * @param {?} value The value to parse/convert to `KeyValueArray`
     * @return {?}
     */
    function toStylingKeyValueArray(keyValueArraySet, stringParser, value) {
        if (value == null /*|| value === undefined */ || value === '')
            return (/** @type {?} */ (EMPTY_ARRAY$3));
        /** @type {?} */
        const styleKeyValueArray = (/** @type {?} */ ([]));
        /** @type {?} */
        const unwrappedValue = (/** @type {?} */ (unwrapSafeValue(value)));
        if (Array.isArray(unwrappedValue)) {
            for (let i = 0; i < unwrappedValue.length; i++) {
                keyValueArraySet(styleKeyValueArray, unwrappedValue[i], true);
            }
        }
        else if (typeof unwrappedValue === 'object') {
            for (const key in unwrappedValue) {
                if (unwrappedValue.hasOwnProperty(key)) {
                    keyValueArraySet(styleKeyValueArray, key, unwrappedValue[key]);
                }
            }
        }
        else if (typeof unwrappedValue === 'string') {
            stringParser(styleKeyValueArray, unwrappedValue);
        }
        else {
            ngDevMode &&
                throwError$1('Unsupported styling type ' + typeof unwrappedValue + ': ' + unwrappedValue);
        }
        return styleKeyValueArray;
    }
    /**
     * Set a `value` for a `key` taking style sanitization into account.
     *
     * See: `keyValueArraySet` for details
     *
     * @param {?} keyValueArray KeyValueArray to add to.
     * @param {?} key Style key to add. (This key will be checked if it needs sanitization)
     * @param {?} value The value to set (If key needs sanitization it will be sanitized)
     * @return {?}
     */
    function styleKeyValueArraySet(keyValueArray, key, value) {
        if (stylePropNeedsSanitization(key)) {
            value = ɵɵsanitizeStyle(value);
        }
        keyValueArraySet(keyValueArray, key, value);
    }
    /**
     * Update map based styling.
     *
     * Map based styling could be anything which contains more than one binding. For example `string`,
     * or object literal. Dealing with all of these types would complicate the logic so
     * instead this function expects that the complex input is first converted into normalized
     * `KeyValueArray`. The advantage of normalization is that we get the values sorted, which makes it
     * very cheap to compute deltas between the previous and current value.
     *
     * @param {?} tView Associated `TView.data` contains the linked list of binding priorities.
     * @param {?} tNode `TNode` where the binding is located.
     * @param {?} lView `LView` contains the values associated with other styling binding at this `TNode`.
     * @param {?} renderer Renderer to use if any updates.
     * @param {?} oldKeyValueArray Previous value represented as `KeyValueArray`
     * @param {?} newKeyValueArray Current value represented as `KeyValueArray`
     * @param {?} isClassBased `true` if `class` (`false` if `style`)
     * @param {?} bindingIndex Binding index of the binding.
     * @return {?}
     */
    function updateStylingMap(tView, tNode, lView, renderer, oldKeyValueArray, newKeyValueArray, isClassBased, bindingIndex) {
        if ((/** @type {?} */ (oldKeyValueArray)) === NO_CHANGE) {
            // On first execution the oldKeyValueArray is NO_CHANGE => treat it as empty KeyValueArray.
            oldKeyValueArray = (/** @type {?} */ (EMPTY_ARRAY$3));
        }
        /** @type {?} */
        let oldIndex = 0;
        /** @type {?} */
        let newIndex = 0;
        /** @type {?} */
        let oldKey = 0 < oldKeyValueArray.length ? oldKeyValueArray[0] : null;
        /** @type {?} */
        let newKey = 0 < newKeyValueArray.length ? newKeyValueArray[0] : null;
        while (oldKey !== null || newKey !== null) {
            ngDevMode && assertLessThan(oldIndex, 999, 'Are we stuck in infinite loop?');
            ngDevMode && assertLessThan(newIndex, 999, 'Are we stuck in infinite loop?');
            /** @type {?} */
            const oldValue = oldIndex < oldKeyValueArray.length ? oldKeyValueArray[oldIndex + 1] : undefined;
            /** @type {?} */
            const newValue = newIndex < newKeyValueArray.length ? newKeyValueArray[newIndex + 1] : undefined;
            /** @type {?} */
            let setKey = null;
            /** @type {?} */
            let setValue = undefined;
            if (oldKey === newKey) {
                // UPDATE: Keys are equal => new value is overwriting old value.
                oldIndex += 2;
                newIndex += 2;
                if (oldValue !== newValue) {
                    setKey = newKey;
                    setValue = newValue;
                }
            }
            else if (newKey === null || oldKey !== null && oldKey < (/** @type {?} */ (newKey))) {
                // DELETE: oldKey key is missing or we did not find the oldKey in the newValue
                // (because the keyValueArray is sorted and `newKey` is found later alphabetically).
                // `"background" < "color"` so we need to delete `"background"` because it is not found in the
                // new array.
                oldIndex += 2;
                setKey = oldKey;
            }
            else {
                // CREATE: newKey's is earlier alphabetically than oldKey's (or no oldKey) => we have new key.
                // `"color" > "background"` so we need to add `color` because it is in new array but not in
                // old array.
                ngDevMode && assertDefined(newKey, 'Expecting to have a valid key');
                newIndex += 2;
                setKey = newKey;
                setValue = newValue;
            }
            if (setKey !== null) {
                updateStyling(tView, tNode, lView, renderer, setKey, setValue, isClassBased, bindingIndex);
            }
            oldKey = oldIndex < oldKeyValueArray.length ? oldKeyValueArray[oldIndex] : null;
            newKey = newIndex < newKeyValueArray.length ? newKeyValueArray[newIndex] : null;
        }
    }
    /**
     * Update a simple (property name) styling.
     *
     * This function takes `prop` and updates the DOM to that value. The function takes the binding
     * value as well as binding priority into consideration to determine which value should be written
     * to DOM. (For example it may be determined that there is a higher priority overwrite which blocks
     * the DOM write, or if the value goes to `undefined` a lower priority overwrite may be consulted.)
     *
     * @param {?} tView Associated `TView.data` contains the linked list of binding priorities.
     * @param {?} tNode `TNode` where the binding is located.
     * @param {?} lView `LView` contains the values associated with other styling binding at this `TNode`.
     * @param {?} renderer Renderer to use if any updates.
     * @param {?} prop Either style property name or a class name.
     * @param {?} value Either style value for `prop` or `true`/`false` if `prop` is class.
     * @param {?} isClassBased `true` if `class` (`false` if `style`)
     * @param {?} bindingIndex Binding index of the binding.
     * @return {?}
     */
    function updateStyling(tView, tNode, lView, renderer, prop, value, isClassBased, bindingIndex) {
        if (tNode.type !== 3 /* Element */) {
            // It is possible to have styling on non-elements (such as ng-container).
            // This is rare, but it does happen. In such a case, just ignore the binding.
            return;
        }
        /** @type {?} */
        const tData = tView.data;
        /** @type {?} */
        const tRange = (/** @type {?} */ (tData[bindingIndex + 1]));
        /** @type {?} */
        const higherPriorityValue = getTStylingRangeNextDuplicate(tRange) ?
            findStylingValue(tData, tNode, lView, prop, getTStylingRangeNext(tRange), isClassBased) :
            undefined;
        if (!isStylingValuePresent(higherPriorityValue)) {
            // We don't have a next duplicate, or we did not find a duplicate value.
            if (!isStylingValuePresent(value)) {
                // We should delete current value or restore to lower priority value.
                if (getTStylingRangePrevDuplicate(tRange)) {
                    // We have a possible prev duplicate, let's retrieve it.
                    value = findStylingValue(tData, null, lView, prop, bindingIndex, isClassBased);
                }
            }
            /** @type {?} */
            const rNode = (/** @type {?} */ (getNativeByIndex(getSelectedIndex(), lView)));
            applyStyling(renderer, isClassBased, rNode, prop, value);
        }
    }
    /**
     * Search for styling value with higher priority which is overwriting current value, or a
     * value of lower priority to which we should fall back if the value is `undefined`.
     *
     * When value is being applied at a location, related values need to be consulted.
     * - If there is a higher priority binding, we should be using that one instead.
     *   For example `<div  [style]="{color:exp1}" [style.color]="exp2">` change to `exp1`
     *   requires that we check `exp2` to see if it is set to value other than `undefined`.
     * - If there is a lower priority binding and we are changing to `undefined`
     *   For example `<div  [style]="{color:exp1}" [style.color]="exp2">` change to `exp2` to
     *   `undefined` requires that we check `exp1` (and static values) and use that as new value.
     *
     * NOTE: The styling stores two values.
     * 1. The raw value which came from the application is stored at `index + 0` location. (This value
     *    is used for dirty checking).
     * 2. The normalized value (converted to `KeyValueArray` if map and sanitized) is stored at `index +
     * 1`.
     *    The advantage of storing the sanitized value is that once the value is written we don't need
     *    to worry about sanitizing it later or keeping track of the sanitizer.
     *
     * @param {?} tData `TData` used for traversing the priority.
     * @param {?} tNode `TNode` to use for resolving static styling. Also controls search direction.
     *   - `TNode` search next and quit as soon as `isStylingValuePresent(value)` is true.
     *      If no value found consult `tNode.residualStyle`/`tNode.residualClass` for default value.
     *   - `null` search prev and go all the way to end. Return last value where
     *     `isStylingValuePresent(value)` is true.
     * @param {?} lView `LView` used for retrieving the actual values.
     * @param {?} prop Property which we are interested in.
     * @param {?} index Starting index in the linked list of styling bindings where the search should start.
     * @param {?} isClassBased `true` if `class` (`false` if `style`)
     * @return {?}
     */
    function findStylingValue(tData, tNode, lView, prop, index, isClassBased) {
        // `TNode` to use for resolving static styling. Also controls search direction.
        //   - `TNode` search next and quit as soon as `isStylingValuePresent(value)` is true.
        //      If no value found consult `tNode.residualStyle`/`tNode.residualClass` for default value.
        //   - `null` search prev and go all the way to end. Return last value where
        //     `isStylingValuePresent(value)` is true.
        /** @type {?} */
        const isPrevDirection = tNode === null;
        /** @type {?} */
        let value = undefined;
        while (index > 0) {
            /** @type {?} */
            const rawKey = (/** @type {?} */ (tData[index]));
            /** @type {?} */
            const containsStatics = Array.isArray(rawKey);
            // Unwrap the key if we contain static values.
            /** @type {?} */
            const key = containsStatics ? ((/** @type {?} */ (rawKey)))[1] : rawKey;
            /** @type {?} */
            const isStylingMap = key === null;
            /** @type {?} */
            let valueAtLViewIndex = lView[index + 1];
            if (valueAtLViewIndex === NO_CHANGE) {
                // In firstUpdatePass the styling instructions create a linked list of styling.
                // On subsequent passes it is possible for a styling instruction to try to read a binding
                // which
                // has not yet executed. In that case we will find `NO_CHANGE` and we should assume that
                // we have `undefined` (or empty array in case of styling-map instruction) instead. This
                // allows the resolution to apply the value (which may later be overwritten when the
                // binding actually executes.)
                valueAtLViewIndex = isStylingMap ? EMPTY_ARRAY$3 : undefined;
            }
            /** @type {?} */
            let currentValue = isStylingMap ? keyValueArrayGet(valueAtLViewIndex, prop) :
                key === prop ? valueAtLViewIndex : undefined;
            if (containsStatics && !isStylingValuePresent(currentValue)) {
                currentValue = keyValueArrayGet((/** @type {?} */ (rawKey)), prop);
            }
            if (isStylingValuePresent(currentValue)) {
                value = currentValue;
                if (isPrevDirection) {
                    return value;
                }
            }
            /** @type {?} */
            const tRange = (/** @type {?} */ (tData[index + 1]));
            index = isPrevDirection ? getTStylingRangePrev(tRange) : getTStylingRangeNext(tRange);
        }
        if (tNode !== null) {
            // in case where we are going in next direction AND we did not find anything, we need to
            // consult residual styling
            /** @type {?} */
            let residual = isClassBased ? tNode.residualClasses : tNode.residualStyles;
            if (residual != null /** OR residual !=== undefined */) {
                value = keyValueArrayGet((/** @type {?} */ (residual)), prop);
            }
        }
        return value;
    }
    /**
     * Determines if the binding value should be used (or if the value is 'undefined' and hence priority
     * resolution should be used.)
     *
     * @param {?} value Binding style value.
     * @return {?}
     */
    function isStylingValuePresent(value) {
        // Currently only `undefined` value is considered non-binding. That is `undefined` says I don't
        // have an opinion as to what this binding should be and you should consult other bindings by
        // priority to determine the valid value.
        // This is extracted into a single function so that we have a single place to control this.
        return value !== undefined;
    }
    /**
     * Sanitizes or adds suffix to the value.
     *
     * If value is `null`/`undefined` no suffix is added
     * @param {?} value
     * @param {?} suffixOrSanitizer
     * @return {?}
     */
    function normalizeAndApplySuffixOrSanitizer(value, suffixOrSanitizer) {
        if (value == null /** || value === undefined */) ;
        else if (typeof suffixOrSanitizer === 'function') {
            // sanitize the value.
            value = suffixOrSanitizer(value);
        }
        else if (typeof suffixOrSanitizer === 'string') {
            value = value + suffixOrSanitizer;
        }
        else if (typeof value === 'object') {
            value = stringify(unwrapSafeValue(value));
        }
        return value;
    }
    /**
     * Tests if the `TNode` has input shadow.
     *
     * An input shadow is when a directive steals (shadows) the input by using `\@Input('style')` or
     * `\@Input('class')` as input.
     *
     * @param {?} tNode `TNode` which we would like to see if it has shadow.
     * @param {?} isClassBased `true` if `class` (`false` if `style`)
     * @return {?}
     */
    function hasStylingInputShadow(tNode, isClassBased) {
        return (tNode.flags & (isClassBased ? 16 /* hasClassInput */ : 32 /* hasStyleInput */)) !== 0;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/instructions/text.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Create static text node
     *
     * \@codeGenApi
     * @param {?} index Index of the node in the data array
     * @param {?=} value Static string value to write.
     *
     * @return {?}
     */
    function ɵɵtext(index, value = '') {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const tView = getTView();
        /** @type {?} */
        const adjustedIndex = index + HEADER_OFFSET;
        ngDevMode &&
            assertEqual(getBindingIndex(), tView.bindingStartIndex, 'text nodes should be created before any bindings');
        ngDevMode && assertDataInRange(lView, adjustedIndex);
        /** @type {?} */
        const tNode = tView.firstCreatePass ?
            getOrCreateTNode(tView, lView[T_HOST], index, 3 /* Element */, null, null) :
            (/** @type {?} */ (tView.data[adjustedIndex]));
        /** @type {?} */
        const textNative = lView[adjustedIndex] = createTextNode(value, lView[RENDERER]);
        appendChild(tView, lView, textNative, tNode);
        // Text nodes are self closing.
        setPreviousOrParentTNode(tNode, false);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/instructions/text_interpolation.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     *
     * Update text content with a lone bound value
     *
     * Used when a text node has 1 interpolated value in it, an no additional text
     * surrounds that interpolated value:
     *
     * ```html
     * <div>{{v0}}</div>
     * ```
     *
     * Its compiled representation is:
     *
     * ```ts
     * ɵɵtextInterpolate(v0);
     * ```
     * @see textInterpolateV
     * \@codeGenApi
     * @param {?} v0
     * @return {?} itself, so that it may be chained.
     */
    function ɵɵtextInterpolate(v0) {
        ɵɵtextInterpolate1('', v0, '');
        return ɵɵtextInterpolate;
    }
    /**
     *
     * Update text content with single bound value surrounded by other text.
     *
     * Used when a text node has 1 interpolated value in it:
     *
     * ```html
     * <div>prefix{{v0}}suffix</div>
     * ```
     *
     * Its compiled representation is:
     *
     * ```ts
     * ɵɵtextInterpolate1('prefix', v0, 'suffix');
     * ```
     * @see textInterpolateV
     * \@codeGenApi
     * @param {?} prefix
     * @param {?} v0
     * @param {?} suffix
     * @return {?} itself, so that it may be chained.
     */
    function ɵɵtextInterpolate1(prefix, v0, suffix) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolated = interpolation1(lView, prefix, v0, suffix);
        if (interpolated !== NO_CHANGE) {
            textBindingInternal(lView, getSelectedIndex(), (/** @type {?} */ (interpolated)));
        }
        return ɵɵtextInterpolate1;
    }
    /**
     *
     * Update text content with 2 bound values surrounded by other text.
     *
     * Used when a text node has 2 interpolated values in it:
     *
     * ```html
     * <div>prefix{{v0}}-{{v1}}suffix</div>
     * ```
     *
     * Its compiled representation is:
     *
     * ```ts
     * ɵɵtextInterpolate2('prefix', v0, '-', v1, 'suffix');
     * ```
     * @see textInterpolateV
     * \@codeGenApi
     * @param {?} prefix
     * @param {?} v0
     * @param {?} i0
     * @param {?} v1
     * @param {?} suffix
     * @return {?} itself, so that it may be chained.
     */
    function ɵɵtextInterpolate2(prefix, v0, i0, v1, suffix) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolated = interpolation2(lView, prefix, v0, i0, v1, suffix);
        if (interpolated !== NO_CHANGE) {
            textBindingInternal(lView, getSelectedIndex(), (/** @type {?} */ (interpolated)));
        }
        return ɵɵtextInterpolate2;
    }
    /**
     *
     * Update text content with 3 bound values surrounded by other text.
     *
     * Used when a text node has 3 interpolated values in it:
     *
     * ```html
     * <div>prefix{{v0}}-{{v1}}-{{v2}}suffix</div>
     * ```
     *
     * Its compiled representation is:
     *
     * ```ts
     * ɵɵtextInterpolate3(
     * 'prefix', v0, '-', v1, '-', v2, 'suffix');
     * ```
     * @see textInterpolateV
     * \@codeGenApi
     * @param {?} prefix
     * @param {?} v0
     * @param {?} i0
     * @param {?} v1
     * @param {?} i1
     * @param {?} v2
     * @param {?} suffix
     * @return {?} itself, so that it may be chained.
     */
    function ɵɵtextInterpolate3(prefix, v0, i0, v1, i1, v2, suffix) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolated = interpolation3(lView, prefix, v0, i0, v1, i1, v2, suffix);
        if (interpolated !== NO_CHANGE) {
            textBindingInternal(lView, getSelectedIndex(), (/** @type {?} */ (interpolated)));
        }
        return ɵɵtextInterpolate3;
    }
    /**
     *
     * Update text content with 4 bound values surrounded by other text.
     *
     * Used when a text node has 4 interpolated values in it:
     *
     * ```html
     * <div>prefix{{v0}}-{{v1}}-{{v2}}-{{v3}}suffix</div>
     * ```
     *
     * Its compiled representation is:
     *
     * ```ts
     * ɵɵtextInterpolate4(
     * 'prefix', v0, '-', v1, '-', v2, '-', v3, 'suffix');
     * ```
     * @see ɵɵtextInterpolateV
     * \@codeGenApi
     * @param {?} prefix
     * @param {?} v0
     * @param {?} i0
     * @param {?} v1
     * @param {?} i1
     * @param {?} v2
     * @param {?} i2
     * @param {?} v3
     * @param {?} suffix
     * @return {?} itself, so that it may be chained.
     */
    function ɵɵtextInterpolate4(prefix, v0, i0, v1, i1, v2, i2, v3, suffix) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolated = interpolation4(lView, prefix, v0, i0, v1, i1, v2, i2, v3, suffix);
        if (interpolated !== NO_CHANGE) {
            textBindingInternal(lView, getSelectedIndex(), (/** @type {?} */ (interpolated)));
        }
        return ɵɵtextInterpolate4;
    }
    /**
     *
     * Update text content with 5 bound values surrounded by other text.
     *
     * Used when a text node has 5 interpolated values in it:
     *
     * ```html
     * <div>prefix{{v0}}-{{v1}}-{{v2}}-{{v3}}-{{v4}}suffix</div>
     * ```
     *
     * Its compiled representation is:
     *
     * ```ts
     * ɵɵtextInterpolate5(
     * 'prefix', v0, '-', v1, '-', v2, '-', v3, '-', v4, 'suffix');
     * ```
     * @see textInterpolateV
     * \@codeGenApi
     * @param {?} prefix
     * @param {?} v0
     * @param {?} i0
     * @param {?} v1
     * @param {?} i1
     * @param {?} v2
     * @param {?} i2
     * @param {?} v3
     * @param {?} i3
     * @param {?} v4
     * @param {?} suffix
     * @return {?} itself, so that it may be chained.
     */
    function ɵɵtextInterpolate5(prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, suffix) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolated = interpolation5(lView, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, suffix);
        if (interpolated !== NO_CHANGE) {
            textBindingInternal(lView, getSelectedIndex(), (/** @type {?} */ (interpolated)));
        }
        return ɵɵtextInterpolate5;
    }
    /**
     *
     * Update text content with 6 bound values surrounded by other text.
     *
     * Used when a text node has 6 interpolated values in it:
     *
     * ```html
     * <div>prefix{{v0}}-{{v1}}-{{v2}}-{{v3}}-{{v4}}-{{v5}}suffix</div>
     * ```
     *
     * Its compiled representation is:
     *
     * ```ts
     * ɵɵtextInterpolate6(
     *    'prefix', v0, '-', v1, '-', v2, '-', v3, '-', v4, '-', v5, 'suffix');
     * ```
     *
     * @see textInterpolateV
     * \@codeGenApi
     * @param {?} prefix
     * @param {?} v0
     * @param {?} i0
     * @param {?} v1
     * @param {?} i1
     * @param {?} v2
     * @param {?} i2
     * @param {?} v3
     * @param {?} i3
     * @param {?} v4
     * @param {?} i4 Static value used for concatenation only.
     * @param {?} v5 Value checked for change. \@returns itself, so that it may be chained.
     * @param {?} suffix
     * @return {?}
     */
    function ɵɵtextInterpolate6(prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, suffix) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolated = interpolation6(lView, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, suffix);
        if (interpolated !== NO_CHANGE) {
            textBindingInternal(lView, getSelectedIndex(), (/** @type {?} */ (interpolated)));
        }
        return ɵɵtextInterpolate6;
    }
    /**
     *
     * Update text content with 7 bound values surrounded by other text.
     *
     * Used when a text node has 7 interpolated values in it:
     *
     * ```html
     * <div>prefix{{v0}}-{{v1}}-{{v2}}-{{v3}}-{{v4}}-{{v5}}-{{v6}}suffix</div>
     * ```
     *
     * Its compiled representation is:
     *
     * ```ts
     * ɵɵtextInterpolate7(
     *    'prefix', v0, '-', v1, '-', v2, '-', v3, '-', v4, '-', v5, '-', v6, 'suffix');
     * ```
     * @see textInterpolateV
     * \@codeGenApi
     * @param {?} prefix
     * @param {?} v0
     * @param {?} i0
     * @param {?} v1
     * @param {?} i1
     * @param {?} v2
     * @param {?} i2
     * @param {?} v3
     * @param {?} i3
     * @param {?} v4
     * @param {?} i4
     * @param {?} v5
     * @param {?} i5
     * @param {?} v6
     * @param {?} suffix
     * @return {?} itself, so that it may be chained.
     */
    function ɵɵtextInterpolate7(prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, i5, v6, suffix) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolated = interpolation7(lView, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, i5, v6, suffix);
        if (interpolated !== NO_CHANGE) {
            textBindingInternal(lView, getSelectedIndex(), (/** @type {?} */ (interpolated)));
        }
        return ɵɵtextInterpolate7;
    }
    /**
     *
     * Update text content with 8 bound values surrounded by other text.
     *
     * Used when a text node has 8 interpolated values in it:
     *
     * ```html
     * <div>prefix{{v0}}-{{v1}}-{{v2}}-{{v3}}-{{v4}}-{{v5}}-{{v6}}-{{v7}}suffix</div>
     * ```
     *
     * Its compiled representation is:
     *
     * ```ts
     * ɵɵtextInterpolate8(
     *  'prefix', v0, '-', v1, '-', v2, '-', v3, '-', v4, '-', v5, '-', v6, '-', v7, 'suffix');
     * ```
     * @see textInterpolateV
     * \@codeGenApi
     * @param {?} prefix
     * @param {?} v0
     * @param {?} i0
     * @param {?} v1
     * @param {?} i1
     * @param {?} v2
     * @param {?} i2
     * @param {?} v3
     * @param {?} i3
     * @param {?} v4
     * @param {?} i4
     * @param {?} v5
     * @param {?} i5
     * @param {?} v6
     * @param {?} i6
     * @param {?} v7
     * @param {?} suffix
     * @return {?} itself, so that it may be chained.
     */
    function ɵɵtextInterpolate8(prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, i5, v6, i6, v7, suffix) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolated = interpolation8(lView, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, i5, v6, i6, v7, suffix);
        if (interpolated !== NO_CHANGE) {
            textBindingInternal(lView, getSelectedIndex(), (/** @type {?} */ (interpolated)));
        }
        return ɵɵtextInterpolate8;
    }
    /**
     * Update text content with 9 or more bound values other surrounded by text.
     *
     * Used when the number of interpolated values exceeds 8.
     *
     * ```html
     * <div>prefix{{v0}}-{{v1}}-{{v2}}-{{v3}}-{{v4}}-{{v5}}-{{v6}}-{{v7}}-{{v8}}-{{v9}}suffix</div>
     * ```
     *
     * Its compiled representation is:
     *
     * ```ts
     * ɵɵtextInterpolateV(
     *  ['prefix', v0, '-', v1, '-', v2, '-', v3, '-', v4, '-', v5, '-', v6, '-', v7, '-', v9,
     *  'suffix']);
     * ```
     * .
     * \@codeGenApi
     * @param {?} values The collection of values and the strings in between those values, beginning with
     * a string prefix and ending with a string suffix.
     * (e.g. `['prefix', value0, '-', value1, '-', value2, ..., value99, 'suffix']`)
     *
     * @return {?} itself, so that it may be chained.
     */
    function ɵɵtextInterpolateV(values) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolated = interpolationV(lView, values);
        if (interpolated !== NO_CHANGE) {
            textBindingInternal(lView, getSelectedIndex(), (/** @type {?} */ (interpolated)));
        }
        return ɵɵtextInterpolateV;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/instructions/class_map_interpolation.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     *
     * Update an interpolated class on an element with single bound value surrounded by text.
     *
     * Used when the value passed to a property has 1 interpolated value in it:
     *
     * ```html
     * <div class="prefix{{v0}}suffix"></div>
     * ```
     *
     * Its compiled representation is:
     *
     * ```ts
     * ɵɵclassMapInterpolate1('prefix', v0, 'suffix');
     * ```
     *
     * \@codeGenApi
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @return {?}
     */
    function ɵɵclassMapInterpolate1(prefix, v0, suffix) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation1(lView, prefix, v0, suffix);
        checkStylingMap(keyValueArraySet, classStringParser, interpolatedValue, true);
    }
    /**
     *
     * Update an interpolated class on an element with 2 bound values surrounded by text.
     *
     * Used when the value passed to a property has 2 interpolated values in it:
     *
     * ```html
     * <div class="prefix{{v0}}-{{v1}}suffix"></div>
     * ```
     *
     * Its compiled representation is:
     *
     * ```ts
     * ɵɵclassMapInterpolate2('prefix', v0, '-', v1, 'suffix');
     * ```
     *
     * \@codeGenApi
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} i0 Static value used for concatenation only.
     * @param {?} v1 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @return {?}
     */
    function ɵɵclassMapInterpolate2(prefix, v0, i0, v1, suffix) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation2(lView, prefix, v0, i0, v1, suffix);
        checkStylingMap(keyValueArraySet, classStringParser, interpolatedValue, true);
    }
    /**
     *
     * Update an interpolated class on an element with 3 bound values surrounded by text.
     *
     * Used when the value passed to a property has 3 interpolated values in it:
     *
     * ```html
     * <div class="prefix{{v0}}-{{v1}}-{{v2}}suffix"></div>
     * ```
     *
     * Its compiled representation is:
     *
     * ```ts
     * ɵɵclassMapInterpolate3(
     * 'prefix', v0, '-', v1, '-', v2, 'suffix');
     * ```
     *
     * \@codeGenApi
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} i0 Static value used for concatenation only.
     * @param {?} v1 Value checked for change.
     * @param {?} i1 Static value used for concatenation only.
     * @param {?} v2 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @return {?}
     */
    function ɵɵclassMapInterpolate3(prefix, v0, i0, v1, i1, v2, suffix) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation3(lView, prefix, v0, i0, v1, i1, v2, suffix);
        checkStylingMap(keyValueArraySet, classStringParser, interpolatedValue, true);
    }
    /**
     *
     * Update an interpolated class on an element with 4 bound values surrounded by text.
     *
     * Used when the value passed to a property has 4 interpolated values in it:
     *
     * ```html
     * <div class="prefix{{v0}}-{{v1}}-{{v2}}-{{v3}}suffix"></div>
     * ```
     *
     * Its compiled representation is:
     *
     * ```ts
     * ɵɵclassMapInterpolate4(
     * 'prefix', v0, '-', v1, '-', v2, '-', v3, 'suffix');
     * ```
     *
     * \@codeGenApi
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} i0 Static value used for concatenation only.
     * @param {?} v1 Value checked for change.
     * @param {?} i1 Static value used for concatenation only.
     * @param {?} v2 Value checked for change.
     * @param {?} i2 Static value used for concatenation only.
     * @param {?} v3 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @return {?}
     */
    function ɵɵclassMapInterpolate4(prefix, v0, i0, v1, i1, v2, i2, v3, suffix) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation4(lView, prefix, v0, i0, v1, i1, v2, i2, v3, suffix);
        checkStylingMap(keyValueArraySet, classStringParser, interpolatedValue, true);
    }
    /**
     *
     * Update an interpolated class on an element with 5 bound values surrounded by text.
     *
     * Used when the value passed to a property has 5 interpolated values in it:
     *
     * ```html
     * <div class="prefix{{v0}}-{{v1}}-{{v2}}-{{v3}}-{{v4}}suffix"></div>
     * ```
     *
     * Its compiled representation is:
     *
     * ```ts
     * ɵɵclassMapInterpolate5(
     * 'prefix', v0, '-', v1, '-', v2, '-', v3, '-', v4, 'suffix');
     * ```
     *
     * \@codeGenApi
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} i0 Static value used for concatenation only.
     * @param {?} v1 Value checked for change.
     * @param {?} i1 Static value used for concatenation only.
     * @param {?} v2 Value checked for change.
     * @param {?} i2 Static value used for concatenation only.
     * @param {?} v3 Value checked for change.
     * @param {?} i3 Static value used for concatenation only.
     * @param {?} v4 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @return {?}
     */
    function ɵɵclassMapInterpolate5(prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, suffix) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation5(lView, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, suffix);
        checkStylingMap(keyValueArraySet, classStringParser, interpolatedValue, true);
    }
    /**
     *
     * Update an interpolated class on an element with 6 bound values surrounded by text.
     *
     * Used when the value passed to a property has 6 interpolated values in it:
     *
     * ```html
     * <div class="prefix{{v0}}-{{v1}}-{{v2}}-{{v3}}-{{v4}}-{{v5}}suffix"></div>
     * ```
     *
     * Its compiled representation is:
     *
     * ```ts
     * ɵɵclassMapInterpolate6(
     *    'prefix', v0, '-', v1, '-', v2, '-', v3, '-', v4, '-', v5, 'suffix');
     * ```
     *
     * \@codeGenApi
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} i0 Static value used for concatenation only.
     * @param {?} v1 Value checked for change.
     * @param {?} i1 Static value used for concatenation only.
     * @param {?} v2 Value checked for change.
     * @param {?} i2 Static value used for concatenation only.
     * @param {?} v3 Value checked for change.
     * @param {?} i3 Static value used for concatenation only.
     * @param {?} v4 Value checked for change.
     * @param {?} i4 Static value used for concatenation only.
     * @param {?} v5 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @return {?}
     */
    function ɵɵclassMapInterpolate6(prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, suffix) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation6(lView, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, suffix);
        checkStylingMap(keyValueArraySet, classStringParser, interpolatedValue, true);
    }
    /**
     *
     * Update an interpolated class on an element with 7 bound values surrounded by text.
     *
     * Used when the value passed to a property has 7 interpolated values in it:
     *
     * ```html
     * <div class="prefix{{v0}}-{{v1}}-{{v2}}-{{v3}}-{{v4}}-{{v5}}-{{v6}}suffix"></div>
     * ```
     *
     * Its compiled representation is:
     *
     * ```ts
     * ɵɵclassMapInterpolate7(
     *    'prefix', v0, '-', v1, '-', v2, '-', v3, '-', v4, '-', v5, '-', v6, 'suffix');
     * ```
     *
     * \@codeGenApi
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} i0 Static value used for concatenation only.
     * @param {?} v1 Value checked for change.
     * @param {?} i1 Static value used for concatenation only.
     * @param {?} v2 Value checked for change.
     * @param {?} i2 Static value used for concatenation only.
     * @param {?} v3 Value checked for change.
     * @param {?} i3 Static value used for concatenation only.
     * @param {?} v4 Value checked for change.
     * @param {?} i4 Static value used for concatenation only.
     * @param {?} v5 Value checked for change.
     * @param {?} i5 Static value used for concatenation only.
     * @param {?} v6 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @return {?}
     */
    function ɵɵclassMapInterpolate7(prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, i5, v6, suffix) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation7(lView, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, i5, v6, suffix);
        checkStylingMap(keyValueArraySet, classStringParser, interpolatedValue, true);
    }
    /**
     *
     * Update an interpolated class on an element with 8 bound values surrounded by text.
     *
     * Used when the value passed to a property has 8 interpolated values in it:
     *
     * ```html
     * <div class="prefix{{v0}}-{{v1}}-{{v2}}-{{v3}}-{{v4}}-{{v5}}-{{v6}}-{{v7}}suffix"></div>
     * ```
     *
     * Its compiled representation is:
     *
     * ```ts
     * ɵɵclassMapInterpolate8(
     *  'prefix', v0, '-', v1, '-', v2, '-', v3, '-', v4, '-', v5, '-', v6, '-', v7, 'suffix');
     * ```
     *
     * \@codeGenApi
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} i0 Static value used for concatenation only.
     * @param {?} v1 Value checked for change.
     * @param {?} i1 Static value used for concatenation only.
     * @param {?} v2 Value checked for change.
     * @param {?} i2 Static value used for concatenation only.
     * @param {?} v3 Value checked for change.
     * @param {?} i3 Static value used for concatenation only.
     * @param {?} v4 Value checked for change.
     * @param {?} i4 Static value used for concatenation only.
     * @param {?} v5 Value checked for change.
     * @param {?} i5 Static value used for concatenation only.
     * @param {?} v6 Value checked for change.
     * @param {?} i6 Static value used for concatenation only.
     * @param {?} v7 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @return {?}
     */
    function ɵɵclassMapInterpolate8(prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, i5, v6, i6, v7, suffix) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation8(lView, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, i5, v6, i6, v7, suffix);
        checkStylingMap(keyValueArraySet, classStringParser, interpolatedValue, true);
    }
    /**
     * Update an interpolated class on an element with 9 or more bound values surrounded by text.
     *
     * Used when the number of interpolated values exceeds 8.
     *
     * ```html
     * <div
     *  class="prefix{{v0}}-{{v1}}-{{v2}}-{{v3}}-{{v4}}-{{v5}}-{{v6}}-{{v7}}-{{v8}}-{{v9}}suffix"></div>
     * ```
     *
     * Its compiled representation is:
     *
     * ```ts
     * ɵɵclassMapInterpolateV(
     *  ['prefix', v0, '-', v1, '-', v2, '-', v3, '-', v4, '-', v5, '-', v6, '-', v7, '-', v9,
     *  'suffix']);
     * ```
     * .
     * \@codeGenApi
     * @param {?} values The collection of values and the strings in-between those values, beginning with
     * a string prefix and ending with a string suffix.
     * (e.g. `['prefix', value0, '-', value1, '-', value2, ..., value99, 'suffix']`)
     * @return {?}
     */
    function ɵɵclassMapInterpolateV(values) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolationV(lView, values);
        checkStylingMap(keyValueArraySet, classStringParser, interpolatedValue, true);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/instructions/style_map_interpolation.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     *
     * Update an interpolated style on an element with single bound value surrounded by text.
     *
     * Used when the value passed to a property has 1 interpolated value in it:
     *
     * ```html
     * <div style="key: {{v0}}suffix"></div>
     * ```
     *
     * Its compiled representation is:
     *
     * ```ts
     * ɵɵstyleMapInterpolate1('key: ', v0, 'suffix');
     * ```
     *
     * \@codeGenApi
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @return {?}
     */
    function ɵɵstyleMapInterpolate1(prefix, v0, suffix) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation1(lView, prefix, v0, suffix);
        ɵɵstyleMap(interpolatedValue);
    }
    /**
     *
     * Update an interpolated style on an element with 2 bound values surrounded by text.
     *
     * Used when the value passed to a property has 2 interpolated values in it:
     *
     * ```html
     * <div style="key: {{v0}}; key1: {{v1}}suffix"></div>
     * ```
     *
     * Its compiled representation is:
     *
     * ```ts
     * ɵɵstyleMapInterpolate2('key: ', v0, '; key1: ', v1, 'suffix');
     * ```
     *
     * \@codeGenApi
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} i0 Static value used for concatenation only.
     * @param {?} v1 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @return {?}
     */
    function ɵɵstyleMapInterpolate2(prefix, v0, i0, v1, suffix) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation2(lView, prefix, v0, i0, v1, suffix);
        ɵɵstyleMap(interpolatedValue);
    }
    /**
     *
     * Update an interpolated style on an element with 3 bound values surrounded by text.
     *
     * Used when the value passed to a property has 3 interpolated values in it:
     *
     * ```html
     * <div style="key: {{v0}}; key2: {{v1}}; key2: {{v2}}suffix"></div>
     * ```
     *
     * Its compiled representation is:
     *
     * ```ts
     * ɵɵstyleMapInterpolate3(
     *     'key: ', v0, '; key1: ', v1, '; key2: ', v2, 'suffix');
     * ```
     *
     * \@codeGenApi
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} i0 Static value used for concatenation only.
     * @param {?} v1 Value checked for change.
     * @param {?} i1 Static value used for concatenation only.
     * @param {?} v2 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @return {?}
     */
    function ɵɵstyleMapInterpolate3(prefix, v0, i0, v1, i1, v2, suffix) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation3(lView, prefix, v0, i0, v1, i1, v2, suffix);
        ɵɵstyleMap(interpolatedValue);
    }
    /**
     *
     * Update an interpolated style on an element with 4 bound values surrounded by text.
     *
     * Used when the value passed to a property has 4 interpolated values in it:
     *
     * ```html
     * <div style="key: {{v0}}; key1: {{v1}}; key2: {{v2}}; key3: {{v3}}suffix"></div>
     * ```
     *
     * Its compiled representation is:
     *
     * ```ts
     * ɵɵstyleMapInterpolate4(
     *     'key: ', v0, '; key1: ', v1, '; key2: ', v2, '; key3: ', v3, 'suffix');
     * ```
     *
     * \@codeGenApi
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} i0 Static value used for concatenation only.
     * @param {?} v1 Value checked for change.
     * @param {?} i1 Static value used for concatenation only.
     * @param {?} v2 Value checked for change.
     * @param {?} i2 Static value used for concatenation only.
     * @param {?} v3 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @return {?}
     */
    function ɵɵstyleMapInterpolate4(prefix, v0, i0, v1, i1, v2, i2, v3, suffix) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation4(lView, prefix, v0, i0, v1, i1, v2, i2, v3, suffix);
        ɵɵstyleMap(interpolatedValue);
    }
    /**
     *
     * Update an interpolated style on an element with 5 bound values surrounded by text.
     *
     * Used when the value passed to a property has 5 interpolated values in it:
     *
     * ```html
     * <div style="key: {{v0}}; key1: {{v1}}; key2: {{v2}}; key3: {{v3}}; key4: {{v4}}suffix"></div>
     * ```
     *
     * Its compiled representation is:
     *
     * ```ts
     * ɵɵstyleMapInterpolate5(
     *     'key: ', v0, '; key1: ', v1, '; key2: ', v2, '; key3: ', v3, '; key4: ', v4, 'suffix');
     * ```
     *
     * \@codeGenApi
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} i0 Static value used for concatenation only.
     * @param {?} v1 Value checked for change.
     * @param {?} i1 Static value used for concatenation only.
     * @param {?} v2 Value checked for change.
     * @param {?} i2 Static value used for concatenation only.
     * @param {?} v3 Value checked for change.
     * @param {?} i3 Static value used for concatenation only.
     * @param {?} v4 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @return {?}
     */
    function ɵɵstyleMapInterpolate5(prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, suffix) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation5(lView, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, suffix);
        ɵɵstyleMap(interpolatedValue);
    }
    /**
     *
     * Update an interpolated style on an element with 6 bound values surrounded by text.
     *
     * Used when the value passed to a property has 6 interpolated values in it:
     *
     * ```html
     * <div style="key: {{v0}}; key1: {{v1}}; key2: {{v2}}; key3: {{v3}}; key4: {{v4}};
     *             key5: {{v5}}suffix"></div>
     * ```
     *
     * Its compiled representation is:
     *
     * ```ts
     * ɵɵstyleMapInterpolate6(
     *    'key: ', v0, '; key1: ', v1, '; key2: ', v2, '; key3: ', v3, '; key4: ', v4, '; key5: ', v5,
     *    'suffix');
     * ```
     *
     * \@codeGenApi
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} i0 Static value used for concatenation only.
     * @param {?} v1 Value checked for change.
     * @param {?} i1 Static value used for concatenation only.
     * @param {?} v2 Value checked for change.
     * @param {?} i2 Static value used for concatenation only.
     * @param {?} v3 Value checked for change.
     * @param {?} i3 Static value used for concatenation only.
     * @param {?} v4 Value checked for change.
     * @param {?} i4 Static value used for concatenation only.
     * @param {?} v5 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @return {?}
     */
    function ɵɵstyleMapInterpolate6(prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, suffix) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation6(lView, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, suffix);
        ɵɵstyleMap(interpolatedValue);
    }
    /**
     *
     * Update an interpolated style on an element with 7 bound values surrounded by text.
     *
     * Used when the value passed to a property has 7 interpolated values in it:
     *
     * ```html
     * <div style="key: {{v0}}; key1: {{v1}}; key2: {{v2}}; key3: {{v3}}; key4: {{v4}}; key5: {{v5}};
     *             key6: {{v6}}suffix"></div>
     * ```
     *
     * Its compiled representation is:
     *
     * ```ts
     * ɵɵstyleMapInterpolate7(
     *    'key: ', v0, '; key1: ', v1, '; key2: ', v2, '; key3: ', v3, '; key4: ', v4, '; key5: ', v5,
     *    '; key6: ', v6, 'suffix');
     * ```
     *
     * \@codeGenApi
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} i0 Static value used for concatenation only.
     * @param {?} v1 Value checked for change.
     * @param {?} i1 Static value used for concatenation only.
     * @param {?} v2 Value checked for change.
     * @param {?} i2 Static value used for concatenation only.
     * @param {?} v3 Value checked for change.
     * @param {?} i3 Static value used for concatenation only.
     * @param {?} v4 Value checked for change.
     * @param {?} i4 Static value used for concatenation only.
     * @param {?} v5 Value checked for change.
     * @param {?} i5 Static value used for concatenation only.
     * @param {?} v6 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @return {?}
     */
    function ɵɵstyleMapInterpolate7(prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, i5, v6, suffix) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation7(lView, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, i5, v6, suffix);
        ɵɵstyleMap(interpolatedValue);
    }
    /**
     *
     * Update an interpolated style on an element with 8 bound values surrounded by text.
     *
     * Used when the value passed to a property has 8 interpolated values in it:
     *
     * ```html
     * <div style="key: {{v0}}; key1: {{v1}}; key2: {{v2}}; key3: {{v3}}; key4: {{v4}}; key5: {{v5}};
     *             key6: {{v6}}; key7: {{v7}}suffix"></div>
     * ```
     *
     * Its compiled representation is:
     *
     * ```ts
     * ɵɵstyleMapInterpolate8(
     *    'key: ', v0, '; key1: ', v1, '; key2: ', v2, '; key3: ', v3, '; key4: ', v4, '; key5: ', v5,
     *    '; key6: ', v6, '; key7: ', v7, 'suffix');
     * ```
     *
     * \@codeGenApi
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} i0 Static value used for concatenation only.
     * @param {?} v1 Value checked for change.
     * @param {?} i1 Static value used for concatenation only.
     * @param {?} v2 Value checked for change.
     * @param {?} i2 Static value used for concatenation only.
     * @param {?} v3 Value checked for change.
     * @param {?} i3 Static value used for concatenation only.
     * @param {?} v4 Value checked for change.
     * @param {?} i4 Static value used for concatenation only.
     * @param {?} v5 Value checked for change.
     * @param {?} i5 Static value used for concatenation only.
     * @param {?} v6 Value checked for change.
     * @param {?} i6 Static value used for concatenation only.
     * @param {?} v7 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @return {?}
     */
    function ɵɵstyleMapInterpolate8(prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, i5, v6, i6, v7, suffix) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation8(lView, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, i5, v6, i6, v7, suffix);
        ɵɵstyleMap(interpolatedValue);
    }
    /**
     * Update an interpolated style on an element with 9 or more bound values surrounded by text.
     *
     * Used when the number of interpolated values exceeds 8.
     *
     * ```html
     * <div
     *  class="key: {{v0}}; key1: {{v1}}; key2: {{v2}}; key3: {{v3}}; key4: {{v4}}; key5: {{v5}};
     *         key6: {{v6}}; key7: {{v7}}; key8: {{v8}}; key9: {{v9}}suffix"></div>
     * ```
     *
     * Its compiled representation is:
     *
     * ```ts
     * ɵɵstyleMapInterpolateV(
     *    ['key: ', v0, '; key1: ', v1, '; key2: ', v2, '; key3: ', v3, '; key4: ', v4, '; key5: ', v5,
     *     '; key6: ', v6, '; key7: ', v7, '; key8: ', v8, '; key9: ', v9, 'suffix']);
     * ```
     * .
     * \@codeGenApi
     * @param {?} values The collection of values and the strings in-between those values, beginning with
     * a string prefix and ending with a string suffix.
     * (e.g. `['prefix', value0, '; key2: ', value1, '; key2: ', value2, ..., value99, 'suffix']`)
     * @return {?}
     */
    function ɵɵstyleMapInterpolateV(values) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolationV(lView, values);
        ɵɵstyleMap(interpolatedValue);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/instructions/style_prop_interpolation.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     *
     * Update an interpolated style property on an element with single bound value surrounded by text.
     *
     * Used when the value passed to a property has 1 interpolated value in it:
     *
     * ```html
     * <div style.color="prefix{{v0}}suffix"></div>
     * ```
     *
     * Its compiled representation is:
     *
     * ```ts
     * ɵɵstylePropInterpolate1(0, 'prefix', v0, 'suffix');
     * ```
     *
     * \@codeGenApi
     * @param {?} prop
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @param {?=} valueSuffix Optional suffix. Used with scalar values to add unit such as `px`.
     * @return {?} itself, so that it may be chained.
     */
    function ɵɵstylePropInterpolate1(prop, prefix, v0, suffix, valueSuffix) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation1(lView, prefix, v0, suffix);
        checkStylingProperty(prop, interpolatedValue, valueSuffix, false);
        return ɵɵstylePropInterpolate1;
    }
    /**
     *
     * Update an interpolated style property on an element with 2 bound values surrounded by text.
     *
     * Used when the value passed to a property has 2 interpolated values in it:
     *
     * ```html
     * <div style.color="prefix{{v0}}-{{v1}}suffix"></div>
     * ```
     *
     * Its compiled representation is:
     *
     * ```ts
     * ɵɵstylePropInterpolate2(0, 'prefix', v0, '-', v1, 'suffix');
     * ```
     *
     * \@codeGenApi
     * @param {?} prop
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} i0 Static value used for concatenation only.
     * @param {?} v1 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @param {?=} valueSuffix Optional suffix. Used with scalar values to add unit such as `px`.
     * @return {?} itself, so that it may be chained.
     */
    function ɵɵstylePropInterpolate2(prop, prefix, v0, i0, v1, suffix, valueSuffix) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation2(lView, prefix, v0, i0, v1, suffix);
        checkStylingProperty(prop, interpolatedValue, valueSuffix, false);
        return ɵɵstylePropInterpolate2;
    }
    /**
     *
     * Update an interpolated style property on an element with 3 bound values surrounded by text.
     *
     * Used when the value passed to a property has 3 interpolated values in it:
     *
     * ```html
     * <div style.color="prefix{{v0}}-{{v1}}-{{v2}}suffix"></div>
     * ```
     *
     * Its compiled representation is:
     *
     * ```ts
     * ɵɵstylePropInterpolate3(0, 'prefix', v0, '-', v1, '-', v2, 'suffix');
     * ```
     *
     * \@codeGenApi
     * @param {?} prop
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} i0 Static value used for concatenation only.
     * @param {?} v1 Value checked for change.
     * @param {?} i1 Static value used for concatenation only.
     * @param {?} v2 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @param {?=} valueSuffix Optional suffix. Used with scalar values to add unit such as `px`.
     * @return {?} itself, so that it may be chained.
     */
    function ɵɵstylePropInterpolate3(prop, prefix, v0, i0, v1, i1, v2, suffix, valueSuffix) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation3(lView, prefix, v0, i0, v1, i1, v2, suffix);
        checkStylingProperty(prop, interpolatedValue, valueSuffix, false);
        return ɵɵstylePropInterpolate3;
    }
    /**
     *
     * Update an interpolated style property on an element with 4 bound values surrounded by text.
     *
     * Used when the value passed to a property has 4 interpolated values in it:
     *
     * ```html
     * <div style.color="prefix{{v0}}-{{v1}}-{{v2}}-{{v3}}suffix"></div>
     * ```
     *
     * Its compiled representation is:
     *
     * ```ts
     * ɵɵstylePropInterpolate4(0, 'prefix', v0, '-', v1, '-', v2, '-', v3, 'suffix');
     * ```
     *
     * \@codeGenApi
     * @param {?} prop
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} i0 Static value used for concatenation only.
     * @param {?} v1 Value checked for change.
     * @param {?} i1 Static value used for concatenation only.
     * @param {?} v2 Value checked for change.
     * @param {?} i2 Static value used for concatenation only.
     * @param {?} v3 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @param {?=} valueSuffix Optional suffix. Used with scalar values to add unit such as `px`.
     * @return {?} itself, so that it may be chained.
     */
    function ɵɵstylePropInterpolate4(prop, prefix, v0, i0, v1, i1, v2, i2, v3, suffix, valueSuffix) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation4(lView, prefix, v0, i0, v1, i1, v2, i2, v3, suffix);
        checkStylingProperty(prop, interpolatedValue, valueSuffix, false);
        return ɵɵstylePropInterpolate4;
    }
    /**
     *
     * Update an interpolated style property on an element with 5 bound values surrounded by text.
     *
     * Used when the value passed to a property has 5 interpolated values in it:
     *
     * ```html
     * <div style.color="prefix{{v0}}-{{v1}}-{{v2}}-{{v3}}-{{v4}}suffix"></div>
     * ```
     *
     * Its compiled representation is:
     *
     * ```ts
     * ɵɵstylePropInterpolate5(0, 'prefix', v0, '-', v1, '-', v2, '-', v3, '-', v4, 'suffix');
     * ```
     *
     * \@codeGenApi
     * @param {?} prop
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} i0 Static value used for concatenation only.
     * @param {?} v1 Value checked for change.
     * @param {?} i1 Static value used for concatenation only.
     * @param {?} v2 Value checked for change.
     * @param {?} i2 Static value used for concatenation only.
     * @param {?} v3 Value checked for change.
     * @param {?} i3 Static value used for concatenation only.
     * @param {?} v4 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @param {?=} valueSuffix Optional suffix. Used with scalar values to add unit such as `px`.
     * @return {?} itself, so that it may be chained.
     */
    function ɵɵstylePropInterpolate5(prop, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, suffix, valueSuffix) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation5(lView, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, suffix);
        checkStylingProperty(prop, interpolatedValue, valueSuffix, false);
        return ɵɵstylePropInterpolate5;
    }
    /**
     *
     * Update an interpolated style property on an element with 6 bound values surrounded by text.
     *
     * Used when the value passed to a property has 6 interpolated values in it:
     *
     * ```html
     * <div style.color="prefix{{v0}}-{{v1}}-{{v2}}-{{v3}}-{{v4}}-{{v5}}suffix"></div>
     * ```
     *
     * Its compiled representation is:
     *
     * ```ts
     * ɵɵstylePropInterpolate6(0, 'prefix', v0, '-', v1, '-', v2, '-', v3, '-', v4, '-', v5, 'suffix');
     * ```
     *
     * \@codeGenApi
     * @param {?} prop
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} i0 Static value used for concatenation only.
     * @param {?} v1 Value checked for change.
     * @param {?} i1 Static value used for concatenation only.
     * @param {?} v2 Value checked for change.
     * @param {?} i2 Static value used for concatenation only.
     * @param {?} v3 Value checked for change.
     * @param {?} i3 Static value used for concatenation only.
     * @param {?} v4 Value checked for change.
     * @param {?} i4 Static value used for concatenation only.
     * @param {?} v5 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @param {?=} valueSuffix Optional suffix. Used with scalar values to add unit such as `px`.
     * @return {?} itself, so that it may be chained.
     */
    function ɵɵstylePropInterpolate6(prop, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, suffix, valueSuffix) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation6(lView, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, suffix);
        checkStylingProperty(prop, interpolatedValue, valueSuffix, false);
        return ɵɵstylePropInterpolate6;
    }
    /**
     *
     * Update an interpolated style property on an element with 7 bound values surrounded by text.
     *
     * Used when the value passed to a property has 7 interpolated values in it:
     *
     * ```html
     * <div style.color="prefix{{v0}}-{{v1}}-{{v2}}-{{v3}}-{{v4}}-{{v5}}-{{v6}}suffix"></div>
     * ```
     *
     * Its compiled representation is:
     *
     * ```ts
     * ɵɵstylePropInterpolate7(
     *    0, 'prefix', v0, '-', v1, '-', v2, '-', v3, '-', v4, '-', v5, '-', v6, 'suffix');
     * ```
     *
     * \@codeGenApi
     * @param {?} prop
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} i0 Static value used for concatenation only.
     * @param {?} v1 Value checked for change.
     * @param {?} i1 Static value used for concatenation only.
     * @param {?} v2 Value checked for change.
     * @param {?} i2 Static value used for concatenation only.
     * @param {?} v3 Value checked for change.
     * @param {?} i3 Static value used for concatenation only.
     * @param {?} v4 Value checked for change.
     * @param {?} i4 Static value used for concatenation only.
     * @param {?} v5 Value checked for change.
     * @param {?} i5 Static value used for concatenation only.
     * @param {?} v6 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @param {?=} valueSuffix Optional suffix. Used with scalar values to add unit such as `px`.
     * @return {?} itself, so that it may be chained.
     */
    function ɵɵstylePropInterpolate7(prop, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, i5, v6, suffix, valueSuffix) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation7(lView, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, i5, v6, suffix);
        checkStylingProperty(prop, interpolatedValue, valueSuffix, false);
        return ɵɵstylePropInterpolate7;
    }
    /**
     *
     * Update an interpolated style property on an element with 8 bound values surrounded by text.
     *
     * Used when the value passed to a property has 8 interpolated values in it:
     *
     * ```html
     * <div style.color="prefix{{v0}}-{{v1}}-{{v2}}-{{v3}}-{{v4}}-{{v5}}-{{v6}}-{{v7}}suffix"></div>
     * ```
     *
     * Its compiled representation is:
     *
     * ```ts
     * ɵɵstylePropInterpolate8(0, 'prefix', v0, '-', v1, '-', v2, '-', v3, '-', v4, '-', v5, '-', v6,
     * '-', v7, 'suffix');
     * ```
     *
     * \@codeGenApi
     * @param {?} prop
     * @param {?} prefix Static value used for concatenation only.
     * @param {?} v0 Value checked for change.
     * @param {?} i0 Static value used for concatenation only.
     * @param {?} v1 Value checked for change.
     * @param {?} i1 Static value used for concatenation only.
     * @param {?} v2 Value checked for change.
     * @param {?} i2 Static value used for concatenation only.
     * @param {?} v3 Value checked for change.
     * @param {?} i3 Static value used for concatenation only.
     * @param {?} v4 Value checked for change.
     * @param {?} i4 Static value used for concatenation only.
     * @param {?} v5 Value checked for change.
     * @param {?} i5 Static value used for concatenation only.
     * @param {?} v6 Value checked for change.
     * @param {?} i6 Static value used for concatenation only.
     * @param {?} v7 Value checked for change.
     * @param {?} suffix Static value used for concatenation only.
     * @param {?=} valueSuffix Optional suffix. Used with scalar values to add unit such as `px`.
     * @return {?} itself, so that it may be chained.
     */
    function ɵɵstylePropInterpolate8(prop, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, i5, v6, i6, v7, suffix, valueSuffix) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolation8(lView, prefix, v0, i0, v1, i1, v2, i2, v3, i3, v4, i4, v5, i5, v6, i6, v7, suffix);
        checkStylingProperty(prop, interpolatedValue, valueSuffix, false);
        return ɵɵstylePropInterpolate8;
    }
    /**
     * Update an interpolated style property on an element with 9 or more bound values surrounded by
     * text.
     *
     * Used when the number of interpolated values exceeds 8.
     *
     * ```html
     * <div
     *  style.color="prefix{{v0}}-{{v1}}-{{v2}}-{{v3}}-{{v4}}-{{v5}}-{{v6}}-{{v7}}-{{v8}}-{{v9}}suffix">
     * </div>
     * ```
     *
     * Its compiled representation is:
     *
     * ```ts
     * ɵɵstylePropInterpolateV(
     *  0, ['prefix', v0, '-', v1, '-', v2, '-', v3, '-', v4, '-', v5, '-', v6, '-', v7, '-', v9,
     *  'suffix']);
     * ```
     *
     * \@codeGenApi
     * @param {?} prop
     * @param {?} values The collection of values and the strings in-between those values, beginning with
     * a string prefix and ending with a string suffix.
     * (e.g. `['prefix', value0, '-', value1, '-', value2, ..., value99, 'suffix']`)
     * @param {?=} valueSuffix Optional suffix. Used with scalar values to add unit such as `px`.
     * @return {?} itself, so that it may be chained.
     */
    function ɵɵstylePropInterpolateV(prop, values, valueSuffix) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const interpolatedValue = interpolationV(lView, values);
        checkStylingProperty(prop, interpolatedValue, valueSuffix, false);
        return ɵɵstylePropInterpolateV;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/instructions/host_property.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Update a property on a host element. Only applies to native node properties, not inputs.
     *
     * Operates on the element selected by index via the {\@link select} instruction.
     *
     * \@codeGenApi
     * @template T
     * @param {?} propName Name of property. Because it is going to DOM, this is not subject to
     *        renaming as part of minification.
     * @param {?} value New value to write.
     * @param {?=} sanitizer An optional function used to sanitize the value.
     * @return {?} This function returns itself so that it may be chained
     * (e.g. `property('name', ctx.name)('title', ctx.title)`)
     *
     */
    function ɵɵhostProperty(propName, value, sanitizer) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const bindingIndex = nextBindingIndex();
        if (bindingUpdated(lView, bindingIndex, value)) {
            /** @type {?} */
            const tView = getTView();
            /** @type {?} */
            const tNode = getSelectedTNode();
            elementPropertyInternal(tView, tNode, lView, propName, value, lView[RENDERER], sanitizer, true);
            ngDevMode && storePropertyBindingMetadata(tView.data, tNode, propName, bindingIndex);
        }
        return ɵɵhostProperty;
    }
    /**
     * Updates a synthetic host binding (e.g. `[\@foo]`) on a component or directive.
     *
     * This instruction is for compatibility purposes and is designed to ensure that a
     * synthetic host binding (e.g. `\@HostBinding('\@foo')`) properly gets rendered in
     * the component's renderer. Normally all host bindings are evaluated with the parent
     * component's renderer, but, in the case of animation \@triggers, they need to be
     * evaluated with the sub component's renderer (because that's where the animation
     * triggers are defined).
     *
     * Do not use this instruction as a replacement for `elementProperty`. This instruction
     * only exists to ensure compatibility with the ViewEngine's host binding behavior.
     *
     * \@codeGenApi
     * @template T
     * @param {?} propName Name of property. Because it is going to DOM, this is not subject to
     *        renaming as part of minification.
     * @param {?} value New value to write.
     * @param {?=} sanitizer An optional function used to sanitize the value.
     *
     * @return {?}
     */
    function ɵɵupdateSyntheticHostBinding(propName, value, sanitizer) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const bindingIndex = nextBindingIndex();
        if (bindingUpdated(lView, bindingIndex, value)) {
            /** @type {?} */
            const tView = getTView();
            /** @type {?} */
            const tNode = getSelectedTNode();
            /** @type {?} */
            const currentDef = getCurrentDirectiveDef(tView.data);
            /** @type {?} */
            const renderer = loadComponentRenderer(currentDef, tNode, lView);
            elementPropertyInternal(tView, tNode, lView, propName, value, renderer, sanitizer, true);
            ngDevMode && storePropertyBindingMetadata(tView.data, tNode, propName, bindingIndex);
        }
        return ɵɵupdateSyntheticHostBinding;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/instructions/all.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/util/discovery_utils.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Retrieves the component instance associated with a given DOM element.
     *
     * \@usageNotes
     * Given the following DOM structure:
     * ```html
     * <my-app>
     *   <div>
     *     <child-comp></child-comp>
     *   </div>
     * </my-app>
     * ```
     * Calling `getComponent` on `<child-comp>` will return the instance of `ChildComponent`
     * associated with this DOM element.
     *
     * Calling the function on `<my-app>` will return the `MyApp` instance.
     *
     *
     * \@publicApi
     * \@globalApi ng
     * @template T
     * @param {?} element DOM element from which the component should be retrieved.
     * @return {?} Component instance associated with the element or `null` if there
     *    is no component associated with it.
     *
     */
    function getComponent(element) {
        assertDomElement(element);
        /** @type {?} */
        const context = loadLContext(element, false);
        if (context === null)
            return null;
        if (context.component === undefined) {
            context.component = getComponentAtNodeIndex(context.nodeIndex, context.lView);
        }
        return (/** @type {?} */ (context.component));
    }
    /**
     * If inside an embedded view (e.g. `*ngIf` or `*ngFor`), retrieves the context of the embedded
     * view that the element is part of. Otherwise retrieves the instance of the component whose view
     * owns the element (in this case, the result is the same as calling `getOwningComponent`).
     *
     * \@publicApi
     * \@globalApi ng
     * @template T
     * @param {?} element Element for which to get the surrounding component instance.
     * @return {?} Instance of the component that is around the element or null if the element isn't
     *    inside any component.
     *
     */
    function getContext(element) {
        assertDomElement(element);
        /** @type {?} */
        const context = loadLContext(element, false);
        return context === null ? null : (/** @type {?} */ (context.lView[CONTEXT]));
    }
    /**
     * Retrieves the component instance whose view contains the DOM element.
     *
     * For example, if `<child-comp>` is used in the template of `<app-comp>`
     * (i.e. a `ViewChild` of `<app-comp>`), calling `getOwningComponent` on `<child-comp>`
     * would return `<app-comp>`.
     *
     * \@publicApi
     * \@globalApi ng
     * @template T
     * @param {?} elementOrDir DOM element, component or directive instance
     *    for which to retrieve the root components.
     * @return {?} Component instance whose view owns the DOM element or null if the element is not
     *    part of a component view.
     *
     */
    function getOwningComponent(elementOrDir) {
        /** @type {?} */
        const context = loadLContext(elementOrDir, false);
        if (context === null)
            return null;
        /** @type {?} */
        let lView = context.lView;
        /** @type {?} */
        let parent;
        ngDevMode && assertLView(lView);
        while (lView[HOST] === null && (parent = (/** @type {?} */ (getLViewParent(lView))))) {
            // As long as lView[HOST] is null we know we are part of sub-template such as `*ngIf`
            lView = parent;
        }
        return lView[FLAGS] & 512 /* IsRoot */ ? null : (/** @type {?} */ (lView[CONTEXT]));
    }
    /**
     * Retrieves all root components associated with a DOM element, directive or component instance.
     * Root components are those which have been bootstrapped by Angular.
     *
     * \@publicApi
     * \@globalApi ng
     * @param {?} elementOrDir DOM element, component or directive instance
     *    for which to retrieve the root components.
     * @return {?} Root components associated with the target object.
     *
     */
    function getRootComponents(elementOrDir) {
        return [...getRootContext(elementOrDir).components];
    }
    /**
     * Retrieves an `Injector` associated with an element, component or directive instance.
     *
     * \@publicApi
     * \@globalApi ng
     * @param {?} elementOrDir DOM element, component or directive instance for which to
     *    retrieve the injector.
     * @return {?} Injector associated with the element, component or directive instance.
     *
     */
    function getInjector(elementOrDir) {
        /** @type {?} */
        const context = loadLContext(elementOrDir, false);
        if (context === null)
            return Injector.NULL;
        /** @type {?} */
        const tNode = (/** @type {?} */ (context.lView[TVIEW].data[context.nodeIndex]));
        return new NodeInjector(tNode, context.lView);
    }
    /**
     * Retrieves directive instances associated with a given DOM element. Does not include
     * component instances.
     *
     * \@usageNotes
     * Given the following DOM structure:
     * ```
     * <my-app>
     *   <button my-button></button>
     *   <my-comp></my-comp>
     * </my-app>
     * ```
     * Calling `getDirectives` on `<button>` will return an array with an instance of the `MyButton`
     * directive that is associated with the DOM element.
     *
     * Calling `getDirectives` on `<my-comp>` will return an empty array.
     *
     * \@publicApi
     * \@globalApi ng
     * @param {?} element DOM element for which to get the directives.
     * @return {?} Array of directives associated with the element.
     *
     */
    function getDirectives(element) {
        /** @type {?} */
        const context = (/** @type {?} */ (loadLContext(element)));
        if (context.directives === undefined) {
            context.directives = getDirectivesAtNodeIndex(context.nodeIndex, context.lView, false);
        }
        // The `directives` in this case are a named array called `LComponentView`. Clone the
        // result so we don't expose an internal data structure in the user's console.
        return context.directives === null ? [] : [...context.directives];
    }
    /**
     * @param {?} target
     * @param {?=} throwOnNotFound
     * @return {?}
     */
    function loadLContext(target, throwOnNotFound = true) {
        /** @type {?} */
        const context = getLContext(target);
        if (!context && throwOnNotFound) {
            throw new Error(ngDevMode ? `Unable to find context associated with ${stringifyForError(target)}` :
                'Invalid ng target');
        }
        return context;
    }
    /**
     * Retrieves the host element of a component or directive instance.
     * The host element is the DOM element that matched the selector of the directive.
     *
     * \@publicApi
     * \@globalApi ng
     * @param {?} componentOrDirective Component or directive instance for which the host
     *     element should be retrieved.
     * @return {?} Host element of the target.
     *
     */
    function getHostElement(componentOrDirective) {
        return (/** @type {?} */ ((/** @type {?} */ ((/** @type {?} */ (getLContext(componentOrDirective))).native))));
    }
    /**
     * Retrieves a list of event listeners associated with a DOM element. The list does include host
     * listeners, but it does not include event listeners defined outside of the Angular context
     * (e.g. through `addEventListener`).
     *
     * \@usageNotes
     * Given the following DOM structure:
     * ```
     * <my-app>
     *   <div (click)="doSomething()"></div>
     * </my-app>
     *
     * ```
     * Calling `getListeners` on `<div>` will return an object that looks as follows:
     * ```
     * {
     *   name: 'click',
     *   element: <div>,
     *   callback: () => doSomething(),
     *   useCapture: false
     * }
     * ```
     *
     * \@publicApi
     * \@globalApi ng
     * @param {?} element Element for which the DOM listeners should be retrieved.
     * @return {?} Array of event listeners on the DOM element.
     *
     */
    function getListeners(element) {
        assertDomElement(element);
        /** @type {?} */
        const lContext = loadLContext(element, false);
        if (lContext === null)
            return [];
        /** @type {?} */
        const lView = lContext.lView;
        /** @type {?} */
        const tView = lView[TVIEW];
        /** @type {?} */
        const lCleanup = lView[CLEANUP];
        /** @type {?} */
        const tCleanup = tView.cleanup;
        /** @type {?} */
        const listeners = [];
        if (tCleanup && lCleanup) {
            for (let i = 0; i < tCleanup.length;) {
                /** @type {?} */
                const firstParam = tCleanup[i++];
                /** @type {?} */
                const secondParam = tCleanup[i++];
                if (typeof firstParam === 'string') {
                    /** @type {?} */
                    const name = firstParam;
                    /** @type {?} */
                    const listenerElement = (/** @type {?} */ ((/** @type {?} */ (unwrapRNode(lView[secondParam])))));
                    /** @type {?} */
                    const callback = lCleanup[tCleanup[i++]];
                    /** @type {?} */
                    const useCaptureOrIndx = tCleanup[i++];
                    // if useCaptureOrIndx is boolean then report it as is.
                    // if useCaptureOrIndx is positive number then it in unsubscribe method
                    // if useCaptureOrIndx is negative number then it is a Subscription
                    /** @type {?} */
                    const type = (typeof useCaptureOrIndx === 'boolean' || useCaptureOrIndx >= 0) ? 'dom' : 'output';
                    /** @type {?} */
                    const useCapture = typeof useCaptureOrIndx === 'boolean' ? useCaptureOrIndx : false;
                    if (element == listenerElement) {
                        listeners.push({ element, name, callback, useCapture, type });
                    }
                }
            }
        }
        listeners.sort(sortListeners);
        return listeners;
    }
    /**
     * @param {?} a
     * @param {?} b
     * @return {?}
     */
    function sortListeners(a, b) {
        if (a.name == b.name)
            return 0;
        return a.name < b.name ? -1 : 1;
    }
    /**
     * Asserts that a value is a DOM Element.
     * @param {?} value
     * @return {?}
     */
    function assertDomElement(value) {
        if (typeof Element !== 'undefined' && !(value instanceof Element)) {
            throw new Error('Expecting instance of DOM Element');
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/util/change_detection_utils.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Marks a component for check (in case of OnPush components) and synchronously
     * performs change detection on the application this component belongs to.
     *
     * \@publicApi
     * \@globalApi ng
     * @param {?} component Component to {\@link ChangeDetectorRef#markForCheck mark for check}.
     *
     * @return {?}
     */
    function applyChanges(component) {
        markDirty(component);
        getRootComponents(component).forEach((/**
         * @param {?} rootComponent
         * @return {?}
         */
        rootComponent => detectChanges(rootComponent)));
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/util/global_utils.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * This value reflects the property on the window where the dev
     * tools are patched (window.ng).
     *
     * @type {?}
     */
    const GLOBAL_PUBLISH_EXPANDO_KEY = 'ng';
    /** @type {?} */
    let _published = false;
    /**
     * Publishes a collection of default debug tools onto`window.ng`.
     *
     * These functions are available globally when Angular is in development
     * mode and are automatically stripped away from prod mode is on.
     * @return {?}
     */
    function publishDefaultGlobalUtils() {
        if (!_published) {
            _published = true;
            publishGlobalUtil('getComponent', getComponent);
            publishGlobalUtil('getContext', getContext);
            publishGlobalUtil('getListeners', getListeners);
            publishGlobalUtil('getOwningComponent', getOwningComponent);
            publishGlobalUtil('getHostElement', getHostElement);
            publishGlobalUtil('getInjector', getInjector);
            publishGlobalUtil('getRootComponents', getRootComponents);
            publishGlobalUtil('getDirectives', getDirectives);
            publishGlobalUtil('applyChanges', applyChanges);
        }
    }
    /**
     * Publishes the given function to `window.ng` so that it can be
     * used from the browser console when an application is not in production.
     * @param {?} name
     * @param {?} fn
     * @return {?}
     */
    function publishGlobalUtil(name, fn) {
        if (typeof COMPILED === 'undefined' || !COMPILED) {
            // Note: we can't export `ng` when using closure enhanced optimization as:
            // - closure declares globals itself for minified names, which sometimes clobber our `ng` global
            // - we can't declare a closure extern as the namespace `ng` is already used within Google
            //   for typings for AngularJS (via `goog.provide('ng....')`).
            /** @type {?} */
            const w = (/** @type {?} */ ((/** @type {?} */ (_global))));
            ngDevMode && assertDefined(fn, 'function not defined');
            if (w) {
                /** @type {?} */
                let container = w[GLOBAL_PUBLISH_EXPANDO_KEY];
                if (!container) {
                    container = w[GLOBAL_PUBLISH_EXPANDO_KEY] = {};
                }
                container[name] = fn;
            }
        }
    }
    /**
     * Creates the root component view and the root component node.
     *
     * @param {?} rNode Render host element.
     * @param {?} def ComponentDef
     * @param {?} rootView The parent view where the host node is stored
     * @param {?} rendererFactory
     * @param {?} hostRenderer The current renderer
     * @param {?=} sanitizer The sanitizer, if provided
     *
     * @return {?} Component view created
     */
    function createRootComponentView(rNode, def, rootView, rendererFactory, hostRenderer, sanitizer) {
        /** @type {?} */
        const tView = rootView[TVIEW];
        ngDevMode && assertDataInRange(rootView, 0 + HEADER_OFFSET);
        rootView[0 + HEADER_OFFSET] = rNode;
        /** @type {?} */
        const tNode = getOrCreateTNode(tView, null, 0, 3 /* Element */, null, null);
        /** @type {?} */
        const mergedAttrs = tNode.mergedAttrs = def.hostAttrs;
        if (mergedAttrs !== null) {
            computeStaticStyling(tNode, mergedAttrs, true);
            if (rNode !== null) {
                setUpAttributes(hostRenderer, rNode, mergedAttrs);
                if (tNode.classes !== null) {
                    writeDirectClass(hostRenderer, rNode, tNode.classes);
                }
                if (tNode.styles !== null) {
                    writeDirectStyle(hostRenderer, rNode, tNode.styles);
                }
            }
        }
        /** @type {?} */
        const viewRenderer = rendererFactory.createRenderer(rNode, def);
        /** @type {?} */
        const componentView = createLView(rootView, getOrCreateTComponentView(def), null, def.onPush ? 64 /* Dirty */ : 16 /* CheckAlways */, rootView[HEADER_OFFSET], tNode, rendererFactory, viewRenderer, sanitizer);
        if (tView.firstCreatePass) {
            diPublicInInjector(getOrCreateNodeInjectorForNode(tNode, rootView), tView, def.type);
            markAsComponentHost(tView, tNode);
            initTNodeFlags(tNode, rootView.length, 1);
        }
        addToViewTree(rootView, componentView);
        // Store component view at node index, with node as the HOST
        return rootView[HEADER_OFFSET] = componentView;
    }
    /**
     * Creates a root component and sets it up with features and host bindings. Shared by
     * renderComponent() and ViewContainerRef.createComponent().
     * @template T
     * @param {?} componentView
     * @param {?} componentDef
     * @param {?} rootLView
     * @param {?} rootContext
     * @param {?} hostFeatures
     * @return {?}
     */
    function createRootComponent(componentView, componentDef, rootLView, rootContext, hostFeatures) {
        /** @type {?} */
        const tView = rootLView[TVIEW];
        // Create directive instance with factory() and store at next index in viewData
        /** @type {?} */
        const component = instantiateRootComponent(tView, rootLView, componentDef);
        rootContext.components.push(component);
        componentView[CONTEXT] = component;
        hostFeatures && hostFeatures.forEach((/**
         * @param {?} feature
         * @return {?}
         */
        (feature) => feature(component, componentDef)));
        // We want to generate an empty QueryList for root content queries for backwards
        // compatibility with ViewEngine.
        if (componentDef.contentQueries) {
            componentDef.contentQueries(1 /* Create */, component, rootLView.length - 1);
        }
        /** @type {?} */
        const rootTNode = getPreviousOrParentTNode();
        if (tView.firstCreatePass &&
            (componentDef.hostBindings !== null || componentDef.hostAttrs !== null)) {
            /** @type {?} */
            const elementIndex = rootTNode.index - HEADER_OFFSET;
            setSelectedIndex(elementIndex);
            /** @type {?} */
            const rootTView = rootLView[TVIEW];
            addHostBindingsToExpandoInstructions(rootTView, componentDef);
            growHostVarsSpace(rootTView, rootLView, componentDef.hostVars);
            invokeHostBindingsInCreationMode(componentDef, component);
        }
        return component;
    }
    /**
     * @param {?=} scheduler
     * @param {?=} playerHandler
     * @return {?}
     */
    function createRootContext(scheduler, playerHandler) {
        return {
            components: [],
            scheduler: scheduler || defaultScheduler,
            clean: CLEAN_PROMISE,
            playerHandler: playerHandler || null,
            flags: 0 /* Empty */
        };
    }
    /**
     * Used to enable lifecycle hooks on the root component.
     *
     * Include this feature when calling `renderComponent` if the root component
     * you are rendering has lifecycle hooks defined. Otherwise, the hooks won't
     * be called properly.
     *
     * Example:
     *
     * ```
     * renderComponent(AppComponent, {hostFeatures: [LifecycleHooksFeature]});
     * ```
     * @param {?} component
     * @param {?} def
     * @return {?}
     */
    function LifecycleHooksFeature(component, def) {
        /** @type {?} */
        const rootTView = (/** @type {?} */ (readPatchedLView(component)))[TVIEW];
        /** @type {?} */
        const dirIndex = rootTView.data.length - 1;
        // TODO(misko): replace `as TNode` with createTNode call. (needs refactoring to lose dep on
        // LNode).
        registerPostOrderHooks(rootTView, (/** @type {?} */ ({ directiveStart: dirIndex, directiveEnd: dirIndex + 1 })));
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/features/inherit_definition_feature.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @param {?} type
     * @return {?}
     */
    function getSuperType(type) {
        return Object.getPrototypeOf(type.prototype).constructor;
    }
    /**
     * Merges the definition from a super class to a sub class.
     * \@codeGenApi
     * @param {?} definition The definition that is a SubClass of another directive of component
     *
     * @return {?}
     */
    function ɵɵInheritDefinitionFeature(definition) {
        /** @type {?} */
        let superType = getSuperType(definition.type);
        /** @type {?} */
        let shouldInheritFields = true;
        /** @type {?} */
        const inheritanceChain = [definition];
        while (superType) {
            /** @type {?} */
            let superDef = undefined;
            if (isComponentDef(definition)) {
                // Don't use getComponentDef/getDirectiveDef. This logic relies on inheritance.
                superDef = superType.ɵcmp || superType.ɵdir;
            }
            else {
                if (superType.ɵcmp) {
                    throw new Error('Directives cannot inherit Components');
                }
                // Don't use getComponentDef/getDirectiveDef. This logic relies on inheritance.
                superDef = superType.ɵdir;
            }
            if (superDef) {
                if (shouldInheritFields) {
                    inheritanceChain.push(superDef);
                    // Some fields in the definition may be empty, if there were no values to put in them that
                    // would've justified object creation. Unwrap them if necessary.
                    /** @type {?} */
                    const writeableDef = (/** @type {?} */ (definition));
                    writeableDef.inputs = maybeUnwrapEmpty(definition.inputs);
                    writeableDef.declaredInputs = maybeUnwrapEmpty(definition.declaredInputs);
                    writeableDef.outputs = maybeUnwrapEmpty(definition.outputs);
                    // Merge hostBindings
                    /** @type {?} */
                    const superHostBindings = superDef.hostBindings;
                    superHostBindings && inheritHostBindings(definition, superHostBindings);
                    // Merge queries
                    /** @type {?} */
                    const superViewQuery = superDef.viewQuery;
                    /** @type {?} */
                    const superContentQueries = superDef.contentQueries;
                    superViewQuery && inheritViewQuery(definition, superViewQuery);
                    superContentQueries && inheritContentQueries(definition, superContentQueries);
                    // Merge inputs and outputs
                    fillProperties(definition.inputs, superDef.inputs);
                    fillProperties(definition.declaredInputs, superDef.declaredInputs);
                    fillProperties(definition.outputs, superDef.outputs);
                    // Merge animations metadata.
                    // If `superDef` is a Component, the `data` field is present (defaults to an empty object).
                    if (isComponentDef(superDef) && superDef.data.animation) {
                        // If super def is a Component, the `definition` is also a Component, since Directives can
                        // not inherit Components (we throw an error above and cannot reach this code).
                        /** @type {?} */
                        const defData = ((/** @type {?} */ (definition))).data;
                        defData.animation = (defData.animation || []).concat(superDef.data.animation);
                    }
                    // Inherit hooks
                    // Assume super class inheritance feature has already run.
                    writeableDef.afterContentChecked =
                        writeableDef.afterContentChecked || superDef.afterContentChecked;
                    writeableDef.afterContentInit = definition.afterContentInit || superDef.afterContentInit;
                    writeableDef.afterViewChecked = definition.afterViewChecked || superDef.afterViewChecked;
                    writeableDef.afterViewInit = definition.afterViewInit || superDef.afterViewInit;
                    writeableDef.doCheck = definition.doCheck || superDef.doCheck;
                    writeableDef.onDestroy = definition.onDestroy || superDef.onDestroy;
                    writeableDef.onInit = definition.onInit || superDef.onInit;
                }
                // Run parent features
                /** @type {?} */
                const features = superDef.features;
                if (features) {
                    for (let i = 0; i < features.length; i++) {
                        /** @type {?} */
                        const feature = features[i];
                        if (feature && feature.ngInherit) {
                            ((/** @type {?} */ (feature)))(definition);
                        }
                        // If `InheritDefinitionFeature` is a part of the current `superDef`, it means that this
                        // def already has all the necessary information inherited from its super class(es), so we
                        // can stop merging fields from super classes. However we need to iterate through the
                        // prototype chain to look for classes that might contain other "features" (like
                        // NgOnChanges), which we should invoke for the original `definition`. We set the
                        // `shouldInheritFields` flag to indicate that, essentially skipping fields inheritance
                        // logic and only invoking functions from the "features" list.
                        if (feature === ɵɵInheritDefinitionFeature) {
                            shouldInheritFields = false;
                        }
                    }
                }
            }
            superType = Object.getPrototypeOf(superType);
        }
        mergeHostAttrsAcrossInheritance(inheritanceChain);
    }
    /**
     * Merge the `hostAttrs` and `hostVars` from the inherited parent to the base class.
     *
     * @param {?} inheritanceChain A list of `WritableDefs` starting at the top most type and listing
     * sub-types in order. For each type take the `hostAttrs` and `hostVars` and merge it with the child
     * type.
     * @return {?}
     */
    function mergeHostAttrsAcrossInheritance(inheritanceChain) {
        /** @type {?} */
        let hostVars = 0;
        /** @type {?} */
        let hostAttrs = null;
        // We process the inheritance order from the base to the leaves here.
        for (let i = inheritanceChain.length - 1; i >= 0; i--) {
            /** @type {?} */
            const def = inheritanceChain[i];
            // For each `hostVars`, we need to add the superclass amount.
            def.hostVars = (hostVars += def.hostVars);
            // for each `hostAttrs` we need to merge it with superclass.
            def.hostAttrs =
                mergeHostAttrs(def.hostAttrs, hostAttrs = mergeHostAttrs(hostAttrs, def.hostAttrs));
        }
    }
    /**
     * @param {?} value
     * @return {?}
     */
    function maybeUnwrapEmpty(value) {
        if (value === EMPTY_OBJ) {
            return {};
        }
        else if (value === EMPTY_ARRAY) {
            return [];
        }
        else {
            return value;
        }
    }
    /**
     * @param {?} definition
     * @param {?} superViewQuery
     * @return {?}
     */
    function inheritViewQuery(definition, superViewQuery) {
        /** @type {?} */
        const prevViewQuery = definition.viewQuery;
        if (prevViewQuery) {
            definition.viewQuery = (/**
             * @param {?} rf
             * @param {?} ctx
             * @return {?}
             */
            (rf, ctx) => {
                superViewQuery(rf, ctx);
                prevViewQuery(rf, ctx);
            });
        }
        else {
            definition.viewQuery = superViewQuery;
        }
    }
    /**
     * @param {?} definition
     * @param {?} superContentQueries
     * @return {?}
     */
    function inheritContentQueries(definition, superContentQueries) {
        /** @type {?} */
        const prevContentQueries = definition.contentQueries;
        if (prevContentQueries) {
            definition.contentQueries = (/**
             * @param {?} rf
             * @param {?} ctx
             * @param {?} directiveIndex
             * @return {?}
             */
            (rf, ctx, directiveIndex) => {
                superContentQueries(rf, ctx, directiveIndex);
                prevContentQueries(rf, ctx, directiveIndex);
            });
        }
        else {
            definition.contentQueries = superContentQueries;
        }
    }
    /**
     * @param {?} definition
     * @param {?} superHostBindings
     * @return {?}
     */
    function inheritHostBindings(definition, superHostBindings) {
        /** @type {?} */
        const prevHostBindings = definition.hostBindings;
        if (prevHostBindings) {
            definition.hostBindings = (/**
             * @param {?} rf
             * @param {?} ctx
             * @return {?}
             */
            (rf, ctx) => {
                superHostBindings(rf, ctx);
                prevHostBindings(rf, ctx);
            });
        }
        else {
            definition.hostBindings = superHostBindings;
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/features/copy_definition_feature.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Fields which exist on either directive or component definitions, and need to be copied from
     * parent to child classes by the `ɵɵCopyDefinitionFeature`.
     * @type {?}
     */
    const COPY_DIRECTIVE_FIELDS = [
        // The child class should use the providers of its parent.
        'providersResolver',
    ];
    /**
     * Fields which exist only on component definitions, and need to be copied from parent to child
     * classes by the `ɵɵCopyDefinitionFeature`.
     *
     * The type here allows any field of `ComponentDef` which is not also a property of `DirectiveDef`,
     * since those should go in `COPY_DIRECTIVE_FIELDS` above.
     * @type {?}
     */
    const COPY_COMPONENT_FIELDS = [
        // The child class should use the template function of its parent, including all template
        // semantics.
        'template',
        'decls',
        'consts',
        'vars',
        'onPush',
        'ngContentSelectors',
        // The child class should use the CSS styles of its parent, including all styling semantics.
        'styles',
        'encapsulation',
        // The child class should be checked by the runtime in the same way as its parent.
        'schemas',
    ];
    /**
     * Copies the fields not handled by the `ɵɵInheritDefinitionFeature` from the supertype of a
     * definition.
     *
     * This exists primarily to support ngcc migration of an existing View Engine pattern, where an
     * entire decorator is inherited from a parent to a child class. When ngcc detects this case, it
     * generates a skeleton definition on the child class, and applies this feature.
     *
     * The `ɵɵCopyDefinitionFeature` then copies any needed fields from the parent class' definition,
     * including things like the component template function.
     *
     * \@codeGenApi
     * @param {?} definition The definition of a child class which inherits from a parent class with its
     * own definition.
     *
     * @return {?}
     */
    function ɵɵCopyDefinitionFeature(definition) {
        /** @type {?} */
        let superType = (/** @type {?} */ (getSuperType(definition.type)));
        /** @type {?} */
        let superDef = undefined;
        if (isComponentDef(definition)) {
            // Don't use getComponentDef/getDirectiveDef. This logic relies on inheritance.
            superDef = (/** @type {?} */ (superType.ɵcmp));
        }
        else {
            // Don't use getComponentDef/getDirectiveDef. This logic relies on inheritance.
            superDef = (/** @type {?} */ (superType.ɵdir));
        }
        // Needed because `definition` fields are readonly.
        /** @type {?} */
        const defAny = ((/** @type {?} */ (definition)));
        // Copy over any fields that apply to either directives or components.
        for (const field of COPY_DIRECTIVE_FIELDS) {
            defAny[field] = superDef[field];
        }
        if (isComponentDef(superDef)) {
            // Copy over any component-specific fields.
            for (const field of COPY_COMPONENT_FIELDS) {
                defAny[field] = superDef[field];
            }
        }
    }

    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * Represents a basic change from a previous to a new value for a single
     * property on a directive instance. Passed as a value in a
     * {@link SimpleChanges} object to the `ngOnChanges` hook.
     *
     * @see `OnChanges`
     *
     * @publicApi
     */
    class SimpleChange {
        constructor(previousValue, currentValue, firstChange) {
            this.previousValue = previousValue;
            this.currentValue = currentValue;
            this.firstChange = firstChange;
        }
        /**
         * Check whether the new value is the first value assigned.
         */
        isFirstChange() {
            return this.firstChange;
        }
    }
    /**
     * The NgOnChangesFeature decorates a component with support for the ngOnChanges
     * lifecycle hook, so it should be included in any component that implements
     * that hook.
     *
     * If the component or directive uses inheritance, the NgOnChangesFeature MUST
     * be included as a feature AFTER {\@link InheritDefinitionFeature}, otherwise
     * inherited properties will not be propagated to the ngOnChanges lifecycle
     * hook.
     *
     * Example usage:
     *
     * ```
     * static ɵcmp = defineComponent({
     *   ...
     *   inputs: {name: 'publicName'},
     *   features: [NgOnChangesFeature]
     * });
     * ```
     *
     * \@codeGenApi
     * @template T
     * @param {?} definition
     * @return {?}
     */
    function ɵɵNgOnChangesFeature(definition) {
        if (definition.type.prototype.ngOnChanges) {
            definition.setInput = ngOnChangesSetInput;
            ((/** @type {?} */ (definition))).onChanges = wrapOnChanges();
        }
    }
    // This option ensures that the ngOnChanges lifecycle hook will be inherited
    // from superclasses (in InheritDefinitionFeature).
    /** @nocollapse */
    // tslint:disable-next-line:no-toplevel-property-access
    ((/** @type {?} */ (ɵɵNgOnChangesFeature))).ngInherit = true;
    /**
     * @return {?}
     */
    function wrapOnChanges() {
        return (/**
         * @this {?}
         * @return {?}
         */
        function wrapOnChangesHook_inPreviousChangesStorage() {
            /** @type {?} */
            const simpleChangesStore = getSimpleChangesStore(this);
            /** @type {?} */
            const current = simpleChangesStore && simpleChangesStore.current;
            if (current) {
                /** @type {?} */
                const previous = (/** @type {?} */ (simpleChangesStore)).previous;
                if (previous === EMPTY_OBJ) {
                    (/** @type {?} */ (simpleChangesStore)).previous = current;
                }
                else {
                    // New changes are copied to the previous store, so that we don't lose history for inputs
                    // which were not changed this time
                    for (let key in current) {
                        previous[key] = current[key];
                    }
                }
                (/** @type {?} */ (simpleChangesStore)).current = null;
                this.ngOnChanges(current);
            }
        });
    }
    /**
     * @template T
     * @this {?}
     * @param {?} instance
     * @param {?} value
     * @param {?} publicName
     * @param {?} privateName
     * @return {?}
     */
    function ngOnChangesSetInput(instance, value, publicName, privateName) {
        /** @type {?} */
        const simpleChangesStore = getSimpleChangesStore(instance) ||
            setSimpleChangesStore(instance, { previous: EMPTY_OBJ, current: null });
        /** @type {?} */
        const current = simpleChangesStore.current || (simpleChangesStore.current = {});
        /** @type {?} */
        const previous = simpleChangesStore.previous;
        /** @type {?} */
        const declaredName = ((/** @type {?} */ (this.declaredInputs)))[publicName];
        /** @type {?} */
        const previousChange = previous[declaredName];
        current[declaredName] = new SimpleChange(previousChange && previousChange.currentValue, value, previous === EMPTY_OBJ);
        ((/** @type {?} */ (instance)))[privateName] = value;
    }
    /** @type {?} */
    const SIMPLE_CHANGES_STORE = '__ngSimpleChanges__';
    /**
     * @param {?} instance
     * @return {?}
     */
    function getSimpleChangesStore(instance) {
        return instance[SIMPLE_CHANGES_STORE] || null;
    }
    /**
     * @param {?} instance
     * @param {?} store
     * @return {?}
     */
    function setSimpleChangesStore(instance, store) {
        return instance[SIMPLE_CHANGES_STORE] = store;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/di_setup.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Resolves the providers which are defined in the DirectiveDef.
     *
     * When inserting the tokens and the factories in their respective arrays, we can assume that
     * this method is called first for the component (if any), and then for other directives on the same
     * node.
     * As a consequence,the providers are always processed in that order:
     * 1) The view providers of the component
     * 2) The providers of the component
     * 3) The providers of the other directives
     * This matches the structure of the injectables arrays of a view (for each node).
     * So the tokens and the factories can be pushed at the end of the arrays, except
     * in one case for multi providers.
     *
     * @template T
     * @param {?} def the directive definition
     * @param {?} providers
     * @param {?} viewProviders
     * @return {?}
     */
    function providersResolver(def, providers, viewProviders) {
        /** @type {?} */
        const tView = getTView();
        if (tView.firstCreatePass) {
            /** @type {?} */
            const isComponent = isComponentDef(def);
            // The list of view providers is processed first, and the flags are updated
            resolveProvider$1(viewProviders, tView.data, tView.blueprint, isComponent, true);
            // Then, the list of providers is processed, and the flags are updated
            resolveProvider$1(providers, tView.data, tView.blueprint, isComponent, false);
        }
    }
    /**
     * Resolves a provider and publishes it to the DI system.
     * @param {?} provider
     * @param {?} tInjectables
     * @param {?} lInjectablesBlueprint
     * @param {?} isComponent
     * @param {?} isViewProvider
     * @return {?}
     */
    function resolveProvider$1(provider, tInjectables, lInjectablesBlueprint, isComponent, isViewProvider) {
        provider = resolveForwardRef(provider);
        if (Array.isArray(provider)) {
            // Recursively call `resolveProvider`
            // Recursion is OK in this case because this code will not be in hot-path once we implement
            // cloning of the initial state.
            for (let i = 0; i < provider.length; i++) {
                resolveProvider$1(provider[i], tInjectables, lInjectablesBlueprint, isComponent, isViewProvider);
            }
        }
        else {
            /** @type {?} */
            const tView = getTView();
            /** @type {?} */
            const lView = getLView();
            /** @type {?} */
            let token = isTypeProvider(provider) ? provider : resolveForwardRef(provider.provide);
            /** @type {?} */
            let providerFactory = providerToFactory(provider);
            /** @type {?} */
            const tNode = getPreviousOrParentTNode();
            /** @type {?} */
            const beginIndex = tNode.providerIndexes & 65535 /* ProvidersStartIndexMask */;
            /** @type {?} */
            const endIndex = tNode.directiveStart;
            /** @type {?} */
            const cptViewProvidersCount = tNode.providerIndexes >> 16 /* CptViewProvidersCountShift */;
            if (isTypeProvider(provider) || !provider.multi) {
                // Single provider case: the factory is created and pushed immediately
                /** @type {?} */
                const factory = new NodeInjectorFactory(providerFactory, isViewProvider, ɵɵdirectiveInject);
                /** @type {?} */
                const existingFactoryIndex = indexOf(token, tInjectables, isViewProvider ? beginIndex : beginIndex + cptViewProvidersCount, endIndex);
                if (existingFactoryIndex === -1) {
                    diPublicInInjector(getOrCreateNodeInjectorForNode((/** @type {?} */ (tNode)), lView), tView, token);
                    registerDestroyHooksIfSupported(tView, provider, tInjectables.length);
                    tInjectables.push(token);
                    tNode.directiveStart++;
                    tNode.directiveEnd++;
                    if (isViewProvider) {
                        tNode.providerIndexes += 65536 /* CptViewProvidersCountShifter */;
                    }
                    lInjectablesBlueprint.push(factory);
                    lView.push(factory);
                }
                else {
                    lInjectablesBlueprint[existingFactoryIndex] = factory;
                    lView[existingFactoryIndex] = factory;
                }
            }
            else {
                // Multi provider case:
                // We create a multi factory which is going to aggregate all the values.
                // Since the output of such a factory depends on content or view injection,
                // we create two of them, which are linked together.
                //
                // The first one (for view providers) is always in the first block of the injectables array,
                // and the second one (for providers) is always in the second block.
                // This is important because view providers have higher priority. When a multi token
                // is being looked up, the view providers should be found first.
                // Note that it is not possible to have a multi factory in the third block (directive block).
                //
                // The algorithm to process multi providers is as follows:
                // 1) If the multi provider comes from the `viewProviders` of the component:
                //   a) If the special view providers factory doesn't exist, it is created and pushed.
                //   b) Else, the multi provider is added to the existing multi factory.
                // 2) If the multi provider comes from the `providers` of the component or of another
                // directive:
                //   a) If the multi factory doesn't exist, it is created and provider pushed into it.
                //      It is also linked to the multi factory for view providers, if it exists.
                //   b) Else, the multi provider is added to the existing multi factory.
                /** @type {?} */
                const existingProvidersFactoryIndex = indexOf(token, tInjectables, beginIndex + cptViewProvidersCount, endIndex);
                /** @type {?} */
                const existingViewProvidersFactoryIndex = indexOf(token, tInjectables, beginIndex, beginIndex + cptViewProvidersCount);
                /** @type {?} */
                const doesProvidersFactoryExist = existingProvidersFactoryIndex >= 0 &&
                    lInjectablesBlueprint[existingProvidersFactoryIndex];
                /** @type {?} */
                const doesViewProvidersFactoryExist = existingViewProvidersFactoryIndex >= 0 &&
                    lInjectablesBlueprint[existingViewProvidersFactoryIndex];
                if (isViewProvider && !doesViewProvidersFactoryExist ||
                    !isViewProvider && !doesProvidersFactoryExist) {
                    // Cases 1.a and 2.a
                    diPublicInInjector(getOrCreateNodeInjectorForNode((/** @type {?} */ (tNode)), lView), tView, token);
                    /** @type {?} */
                    const factory = multiFactory(isViewProvider ? multiViewProvidersFactoryResolver : multiProvidersFactoryResolver, lInjectablesBlueprint.length, isViewProvider, isComponent, providerFactory);
                    if (!isViewProvider && doesViewProvidersFactoryExist) {
                        lInjectablesBlueprint[existingViewProvidersFactoryIndex].providerFactory = factory;
                    }
                    registerDestroyHooksIfSupported(tView, provider, tInjectables.length, 0);
                    tInjectables.push(token);
                    tNode.directiveStart++;
                    tNode.directiveEnd++;
                    if (isViewProvider) {
                        tNode.providerIndexes += 65536 /* CptViewProvidersCountShifter */;
                    }
                    lInjectablesBlueprint.push(factory);
                    lView.push(factory);
                }
                else {
                    // Cases 1.b and 2.b
                    /** @type {?} */
                    const indexInFactory = multiFactoryAdd((/** @type {?} */ (lInjectablesBlueprint))[isViewProvider ? existingViewProvidersFactoryIndex :
                        existingProvidersFactoryIndex], providerFactory, !isViewProvider && isComponent);
                    registerDestroyHooksIfSupported(tView, provider, existingProvidersFactoryIndex > -1 ? existingProvidersFactoryIndex :
                        existingViewProvidersFactoryIndex, indexInFactory);
                }
                if (!isViewProvider && isComponent && doesViewProvidersFactoryExist) {
                    (/** @type {?} */ (lInjectablesBlueprint[existingViewProvidersFactoryIndex].componentProviders))++;
                }
            }
        }
    }
    /**
     * Registers the `ngOnDestroy` hook of a provider, if the provider supports destroy hooks.
     * @param {?} tView `TView` in which to register the hook.
     * @param {?} provider Provider whose hook should be registered.
     * @param {?} contextIndex Index under which to find the context for the hook when it's being invoked.
     * @param {?=} indexInFactory Only required for `multi` providers. Index of the provider in the multi
     * provider factory.
     * @return {?}
     */
    function registerDestroyHooksIfSupported(tView, provider, contextIndex, indexInFactory) {
        /** @type {?} */
        const providerIsTypeProvider = isTypeProvider(provider);
        if (providerIsTypeProvider || isClassProvider(provider)) {
            /** @type {?} */
            const prototype = (((/** @type {?} */ (provider))).useClass || provider).prototype;
            /** @type {?} */
            const ngOnDestroy = prototype.ngOnDestroy;
            if (ngOnDestroy) {
                /** @type {?} */
                const hooks = tView.destroyHooks || (tView.destroyHooks = []);
                if (!providerIsTypeProvider && (((/** @type {?} */ (provider)))).multi) {
                    ngDevMode &&
                        assertDefined(indexInFactory, 'indexInFactory when registering multi factory destroy hook');
                    /** @type {?} */
                    const existingCallbacksIndex = hooks.indexOf(contextIndex);
                    if (existingCallbacksIndex === -1) {
                        hooks.push(contextIndex, [indexInFactory, ngOnDestroy]);
                    }
                    else {
                        ((/** @type {?} */ (hooks[existingCallbacksIndex + 1]))).push((/** @type {?} */ (indexInFactory)), ngOnDestroy);
                    }
                }
                else {
                    hooks.push(contextIndex, ngOnDestroy);
                }
            }
        }
    }
    /**
     * Add a factory in a multi factory.
     * @param {?} multiFactory
     * @param {?} factory
     * @param {?} isComponentProvider
     * @return {?} Index at which the factory was inserted.
     */
    function multiFactoryAdd(multiFactory, factory, isComponentProvider) {
        if (isComponentProvider) {
            (/** @type {?} */ (multiFactory.componentProviders))++;
        }
        return (/** @type {?} */ (multiFactory.multi)).push(factory) - 1;
    }
    /**
     * Returns the index of item in the array, but only in the begin to end range.
     * @param {?} item
     * @param {?} arr
     * @param {?} begin
     * @param {?} end
     * @return {?}
     */
    function indexOf(item, arr, begin, end) {
        for (let i = begin; i < end; i++) {
            if (arr[i] === item)
                return i;
        }
        return -1;
    }
    /**
     * Use this with `multi` `providers`.
     * @this {?}
     * @param {?} _
     * @param {?} tData
     * @param {?} lData
     * @param {?} tNode
     * @return {?}
     */
    function multiProvidersFactoryResolver(_, tData, lData, tNode) {
        return multiResolve((/** @type {?} */ (this.multi)), []);
    }
    /**
     * Use this with `multi` `viewProviders`.
     *
     * This factory knows how to concatenate itself with the existing `multi` `providers`.
     * @this {?}
     * @param {?} _
     * @param {?} tData
     * @param {?} lView
     * @param {?} tNode
     * @return {?}
     */
    function multiViewProvidersFactoryResolver(_, tData, lView, tNode) {
        /** @type {?} */
        const factories = (/** @type {?} */ (this.multi));
        /** @type {?} */
        let result;
        if (this.providerFactory) {
            /** @type {?} */
            const componentCount = (/** @type {?} */ (this.providerFactory.componentProviders));
            /** @type {?} */
            const multiProviders = getNodeInjectable(lView, lView[TVIEW], (/** @type {?} */ ((/** @type {?} */ (this.providerFactory)).index)), tNode);
            // Copy the section of the array which contains `multi` `providers` from the component
            result = multiProviders.slice(0, componentCount);
            // Insert the `viewProvider` instances.
            multiResolve(factories, result);
            // Copy the section of the array which contains `multi` `providers` from other directives
            for (let i = componentCount; i < multiProviders.length; i++) {
                result.push(multiProviders[i]);
            }
        }
        else {
            result = [];
            // Insert the `viewProvider` instances.
            multiResolve(factories, result);
        }
        return result;
    }
    /**
     * Maps an array of factories into an array of values.
     * @param {?} factories
     * @param {?} result
     * @return {?}
     */
    function multiResolve(factories, result) {
        for (let i = 0; i < factories.length; i++) {
            /** @type {?} */
            const factory = (/** @type {?} */ ((/** @type {?} */ (factories[i]))));
            result.push(factory());
        }
        return result;
    }
    /**
     * Creates a multi factory.
     * @param {?} factoryFn
     * @param {?} index
     * @param {?} isViewProvider
     * @param {?} isComponent
     * @param {?} f
     * @return {?}
     */
    function multiFactory(factoryFn, index, isViewProvider, isComponent, f) {
        /** @type {?} */
        const factory = new NodeInjectorFactory(factoryFn, isViewProvider, ɵɵdirectiveInject);
        factory.multi = [];
        factory.index = index;
        factory.componentProviders = 0;
        multiFactoryAdd(factory, f, isComponent && !isViewProvider);
        return factory;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/features/providers_feature.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * This feature resolves the providers of a directive (or component),
     * and publish them into the DI system, making it visible to others for injection.
     *
     * For example:
     * ```ts
     * class ComponentWithProviders {
     *   constructor(private greeter: GreeterDE) {}
     *
     *   static ɵcmp = defineComponent({
     *     type: ComponentWithProviders,
     *     selectors: [['component-with-providers']],
     *    factory: () => new ComponentWithProviders(directiveInject(GreeterDE as any)),
     *    decls: 1,
     *    vars: 1,
     *    template: function(fs: RenderFlags, ctx: ComponentWithProviders) {
     *      if (fs & RenderFlags.Create) {
     *        ɵɵtext(0);
     *      }
     *      if (fs & RenderFlags.Update) {
     *        ɵɵtextInterpolate(ctx.greeter.greet());
     *      }
     *    },
     *    features: [ProvidersFeature([GreeterDE])]
     *  });
     * }
     * ```
     *
     * \@codeGenApi
     * @template T
     * @param {?} providers
     * @param {?=} viewProviders
     * @return {?}
     */
    function ɵɵProvidersFeature(providers, viewProviders = []) {
        return (/**
         * @param {?} definition
         * @return {?}
         */
        (definition) => {
            definition.providersResolver =
                (/**
                 * @param {?} def
                 * @param {?=} processProvidersFn
                 * @return {?}
                 */
                (def, processProvidersFn) => {
                    return providersResolver(def, //
                    processProvidersFn ? processProvidersFn(providers) : providers, //
                    viewProviders);
                });
        });
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/linker/component_factory.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * Represents a component created by a `ComponentFactory`.
     * Provides access to the component instance and related objects,
     * and provides the means of destroying the instance.
     *
     * \@publicApi
     * @abstract
     * @template C
     */
    class ComponentRef {
    }
    /**
     * Base class for a factory that can create a component dynamically.
     * Instantiate a factory for a given type of component with `resolveComponentFactory()`.
     * Use the resulting `ComponentFactory.create()` method to create a component of that type.
     *
     * @see [Dynamic Components](guide/dynamic-component-loader)
     *
     * \@publicApi
     * @abstract
     * @template C
     */
    class ComponentFactory {
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/linker/component_factory_resolver.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @param {?} component
     * @return {?}
     */
    function noComponentFactoryError(component) {
        /** @type {?} */
        const error = Error(`No component factory found for ${stringify(component)}. Did you add it to @NgModule.entryComponents?`);
        ((/** @type {?} */ (error)))[ERROR_COMPONENT] = component;
        return error;
    }
    /** @type {?} */
    const ERROR_COMPONENT = 'ngComponent';
    class _NullComponentFactoryResolver {
        /**
         * @template T
         * @param {?} component
         * @return {?}
         */
        resolveComponentFactory(component) {
            throw noComponentFactoryError(component);
        }
    }
    /**
     * A simple registry that maps `Components` to generated `ComponentFactory` classes
     * that can be used to create instances of components.
     * Use to obtain the factory for a given component type,
     * then use the factory's `create()` method to create a component of that type.
     *
     * @see [Dynamic Components](guide/dynamic-component-loader)
     * \@publicApi
     * @abstract
     */
    class ComponentFactoryResolver {
    }
    ComponentFactoryResolver.NULL = new _NullComponentFactoryResolver();

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/linker/element_ref.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * A wrapper around a native element inside of a View.
     *
     * An `ElementRef` is backed by a render-specific element. In the browser, this is usually a DOM
     * element.
     *
     * \@security Permitting direct access to the DOM can make your application more vulnerable to
     * XSS attacks. Carefully review any use of `ElementRef` in your code. For more detail, see the
     * [Security Guide](http://g.co/ng/security).
     *
     * \@publicApi
     * @template T
     */
    // Note: We don't expose things like `Injector`, `ViewContainer`, ... here,
    // i.e. users have to ask for what they need. With that, we can build better analysis tools
    // and could do better codegen in the future.
    class ElementRef {
        /**
         * @param {?} nativeElement
         */
        constructor(nativeElement) {
            this.nativeElement = nativeElement;
        }
    }
    /**
     * \@internal
     * @nocollapse
     */
    ElementRef.__NG_ELEMENT_ID__ = (/**
     * @return {?}
     */
    () => SWITCH_ELEMENT_REF_FACTORY(ElementRef));
    /** @type {?} */
    const SWITCH_ELEMENT_REF_FACTORY__POST_R3__ = injectElementRef;
    /** @type {?} */
    const SWITCH_ELEMENT_REF_FACTORY = SWITCH_ELEMENT_REF_FACTORY__POST_R3__;

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render/api.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const Renderer2Interceptor = new InjectionToken('Renderer2Interceptor');
    /**
     * Creates and initializes a custom renderer that implements the `Renderer2` base class.
     *
     * \@publicApi
     * @abstract
     */
    class RendererFactory2 {
    }
    /** @enum {number} */
    const RendererStyleFlags2 = {
        // TODO(misko): This needs to be refactored into a separate file so that it can be imported from
        // `node_manipulation.ts` Currently doing the import cause resolution order to change and fails
        // the tests. The work around is to have hard coded value in `node_manipulation.ts` for now.
        /**
         * Marks a style as important.
         */
        Important: 1,
        /**
         * Marks a style as using dash case naming (this-is-dash-case).
         */
        DashCase: 2,
    };
    RendererStyleFlags2[RendererStyleFlags2.Important] = 'Important';
    RendererStyleFlags2[RendererStyleFlags2.DashCase] = 'DashCase';
    /**
     * Extend this base class to implement custom rendering. By default, Angular
     * renders a template into DOM. You can use custom rendering to intercept
     * rendering calls, or to render to something other than DOM.
     *
     * Create your custom renderer using `RendererFactory2`.
     *
     * Use a custom renderer to bypass Angular's templating and
     * make custom UI changes that can't be expressed declaratively.
     * For example if you need to set a property or an attribute whose name is
     * not statically known, use the `setProperty()` or
     * `setAttribute()` method.
     *
     * \@publicApi
     * @abstract
     */
    class Renderer2 {
    }
    /**
     * \@internal
     * @nocollapse
     */
    Renderer2.__NG_ELEMENT_ID__ = (/**
     * @return {?}
     */
    () => SWITCH_RENDERER2_FACTORY());
    /** @type {?} */
    const SWITCH_RENDERER2_FACTORY__POST_R3__ = injectRenderer2;
    /** @type {?} */
    const SWITCH_RENDERER2_FACTORY = SWITCH_RENDERER2_FACTORY__POST_R3__;

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/sanitization/sanitizer.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Sanitizer is used by the views to sanitize potentially dangerous values.
     *
     * \@publicApi
     * @abstract
     */
    class Sanitizer {
    }
    /** @nocollapse */
    Sanitizer.ɵprov = ɵɵdefineInjectable({
        token: Sanitizer,
        providedIn: 'root',
        factory: (/**
         * @return {?}
         */
        () => null),
    });

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/version.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * \@description Represents the version of Angular
     *
     * \@publicApi
     */
    class Version {
        /**
         * @param {?} full
         */
        constructor(full) {
            this.full = full;
            this.major = full.split('.')[0];
            this.minor = full.split('.')[1];
            this.patch = full.split('.').slice(2).join('.');
        }
    }
    /**
     * \@publicApi
     * @type {?}
     */
    const VERSION = new Version('9.1.12');

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/change_detection/differs/default_iterable_differ.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class DefaultIterableDifferFactory {
        constructor() { }
        /**
         * @param {?} obj
         * @return {?}
         */
        supports(obj) {
            return isListLikeIterable$1(obj);
        }
        /**
         * @template V
         * @param {?=} trackByFn
         * @return {?}
         */
        create(trackByFn) {
            return new DefaultIterableDiffer(trackByFn);
        }
    }
    /** @type {?} */
    const trackByIdentity = (/**
     * @param {?} index
     * @param {?} item
     * @return {?}
     */
    (index, item) => item);
    /**
     * @deprecated v4.0.0 - Should not be part of public API.
     * \@publicApi
     * @template V
     */
    class DefaultIterableDiffer {
        /**
         * @param {?=} trackByFn
         */
        constructor(trackByFn) {
            this.length = 0;
            // Keeps track of the used records at any point in time (during & across `_check()` calls)
            this._linkedRecords = null;
            // Keeps track of the removed records at any point in time during `_check()` calls.
            this._unlinkedRecords = null;
            this._previousItHead = null;
            this._itHead = null;
            this._itTail = null;
            this._additionsHead = null;
            this._additionsTail = null;
            this._movesHead = null;
            this._movesTail = null;
            this._removalsHead = null;
            this._removalsTail = null;
            // Keeps track of records where custom track by is the same, but item identity has changed
            this._identityChangesHead = null;
            this._identityChangesTail = null;
            this._trackByFn = trackByFn || trackByIdentity;
        }
        /**
         * @param {?} fn
         * @return {?}
         */
        forEachItem(fn) {
            /** @type {?} */
            let record;
            for (record = this._itHead; record !== null; record = record._next) {
                fn(record);
            }
        }
        /**
         * @param {?} fn
         * @return {?}
         */
        forEachOperation(fn) {
            /** @type {?} */
            let nextIt = this._itHead;
            /** @type {?} */
            let nextRemove = this._removalsHead;
            /** @type {?} */
            let addRemoveOffset = 0;
            /** @type {?} */
            let moveOffsets = null;
            while (nextIt || nextRemove) {
                // Figure out which is the next record to process
                // Order: remove, add, move
                /** @type {?} */
                const record = !nextRemove ||
                    nextIt &&
                        (/** @type {?} */ (nextIt.currentIndex)) <
                            getPreviousIndex(nextRemove, addRemoveOffset, moveOffsets) ?
                    (/** @type {?} */ (nextIt)) :
                    nextRemove;
                /** @type {?} */
                const adjPreviousIndex = getPreviousIndex(record, addRemoveOffset, moveOffsets);
                /** @type {?} */
                const currentIndex = record.currentIndex;
                // consume the item, and adjust the addRemoveOffset and update moveDistance if necessary
                if (record === nextRemove) {
                    addRemoveOffset--;
                    nextRemove = nextRemove._nextRemoved;
                }
                else {
                    nextIt = (/** @type {?} */ (nextIt))._next;
                    if (record.previousIndex == null) {
                        addRemoveOffset++;
                    }
                    else {
                        // INVARIANT:  currentIndex < previousIndex
                        if (!moveOffsets)
                            moveOffsets = [];
                        /** @type {?} */
                        const localMovePreviousIndex = adjPreviousIndex - addRemoveOffset;
                        /** @type {?} */
                        const localCurrentIndex = (/** @type {?} */ (currentIndex)) - addRemoveOffset;
                        if (localMovePreviousIndex != localCurrentIndex) {
                            for (let i = 0; i < localMovePreviousIndex; i++) {
                                /** @type {?} */
                                const offset = i < moveOffsets.length ? moveOffsets[i] : (moveOffsets[i] = 0);
                                /** @type {?} */
                                const index = offset + i;
                                if (localCurrentIndex <= index && index < localMovePreviousIndex) {
                                    moveOffsets[i] = offset + 1;
                                }
                            }
                            /** @type {?} */
                            const previousIndex = record.previousIndex;
                            moveOffsets[previousIndex] = localCurrentIndex - localMovePreviousIndex;
                        }
                    }
                }
                if (adjPreviousIndex !== currentIndex) {
                    fn(record, adjPreviousIndex, currentIndex);
                }
            }
        }
        /**
         * @param {?} fn
         * @return {?}
         */
        forEachPreviousItem(fn) {
            /** @type {?} */
            let record;
            for (record = this._previousItHead; record !== null; record = record._nextPrevious) {
                fn(record);
            }
        }
        /**
         * @param {?} fn
         * @return {?}
         */
        forEachAddedItem(fn) {
            /** @type {?} */
            let record;
            for (record = this._additionsHead; record !== null; record = record._nextAdded) {
                fn(record);
            }
        }
        /**
         * @param {?} fn
         * @return {?}
         */
        forEachMovedItem(fn) {
            /** @type {?} */
            let record;
            for (record = this._movesHead; record !== null; record = record._nextMoved) {
                fn(record);
            }
        }
        /**
         * @param {?} fn
         * @return {?}
         */
        forEachRemovedItem(fn) {
            /** @type {?} */
            let record;
            for (record = this._removalsHead; record !== null; record = record._nextRemoved) {
                fn(record);
            }
        }
        /**
         * @param {?} fn
         * @return {?}
         */
        forEachIdentityChange(fn) {
            /** @type {?} */
            let record;
            for (record = this._identityChangesHead; record !== null; record = record._nextIdentityChange) {
                fn(record);
            }
        }
        /**
         * @param {?} collection
         * @return {?}
         */
        diff(collection) {
            if (collection == null)
                collection = [];
            if (!isListLikeIterable$1(collection)) {
                throw new Error(`Error trying to diff '${stringify(collection)}'. Only arrays and iterables are allowed`);
            }
            if (this.check(collection)) {
                return this;
            }
            else {
                return null;
            }
        }
        /**
         * @return {?}
         */
        onDestroy() { }
        /**
         * @param {?} collection
         * @return {?}
         */
        check(collection) {
            this._reset();
            /** @type {?} */
            let record = this._itHead;
            /** @type {?} */
            let mayBeDirty = false;
            /** @type {?} */
            let index;
            /** @type {?} */
            let item;
            /** @type {?} */
            let itemTrackBy;
            if (Array.isArray(collection)) {
                ((/** @type {?} */ (this))).length = collection.length;
                for (let index = 0; index < this.length; index++) {
                    item = collection[index];
                    itemTrackBy = this._trackByFn(index, item);
                    if (record === null || !looseIdentical(record.trackById, itemTrackBy)) {
                        record = this._mismatch(record, item, itemTrackBy, index);
                        mayBeDirty = true;
                    }
                    else {
                        if (mayBeDirty) {
                            // TODO(misko): can we limit this to duplicates only?
                            record = this._verifyReinsertion(record, item, itemTrackBy, index);
                        }
                        if (!looseIdentical(record.item, item))
                            this._addIdentityChange(record, item);
                    }
                    record = record._next;
                }
            }
            else {
                index = 0;
                iterateListLike$1(collection, (/**
                 * @param {?} item
                 * @return {?}
                 */
                (item) => {
                    itemTrackBy = this._trackByFn(index, item);
                    if (record === null || !looseIdentical(record.trackById, itemTrackBy)) {
                        record = this._mismatch(record, item, itemTrackBy, index);
                        mayBeDirty = true;
                    }
                    else {
                        if (mayBeDirty) {
                            // TODO(misko): can we limit this to duplicates only?
                            record = this._verifyReinsertion(record, item, itemTrackBy, index);
                        }
                        if (!looseIdentical(record.item, item))
                            this._addIdentityChange(record, item);
                    }
                    record = record._next;
                    index++;
                }));
                ((/** @type {?} */ (this))).length = index;
            }
            this._truncate(record);
            ((/** @type {?} */ (this))).collection = collection;
            return this.isDirty;
        }
        /* CollectionChanges is considered dirty if it has any additions, moves, removals, or identity
           * changes.
           */
        /**
         * @return {?}
         */
        get isDirty() {
            return this._additionsHead !== null || this._movesHead !== null ||
                this._removalsHead !== null || this._identityChangesHead !== null;
        }
        /**
         * Reset the state of the change objects to show no changes. This means set previousKey to
         * currentKey, and clear all of the queues (additions, moves, removals).
         * Set the previousIndexes of moved and added items to their currentIndexes
         * Reset the list of additions, moves and removals
         *
         * \@internal
         * @return {?}
         */
        _reset() {
            if (this.isDirty) {
                /** @type {?} */
                let record;
                /** @type {?} */
                let nextRecord;
                for (record = this._previousItHead = this._itHead; record !== null; record = record._next) {
                    record._nextPrevious = record._next;
                }
                for (record = this._additionsHead; record !== null; record = record._nextAdded) {
                    record.previousIndex = record.currentIndex;
                }
                this._additionsHead = this._additionsTail = null;
                for (record = this._movesHead; record !== null; record = nextRecord) {
                    record.previousIndex = record.currentIndex;
                    nextRecord = record._nextMoved;
                }
                this._movesHead = this._movesTail = null;
                this._removalsHead = this._removalsTail = null;
                this._identityChangesHead = this._identityChangesTail = null;
                // TODO(vicb): when assert gets supported
                // assert(!this.isDirty);
            }
        }
        /**
         * This is the core function which handles differences between collections.
         *
         * - `record` is the record which we saw at this position last time. If null then it is a new
         *   item.
         * - `item` is the current item in the collection
         * - `index` is the position of the item in the collection
         *
         * \@internal
         * @param {?} record
         * @param {?} item
         * @param {?} itemTrackBy
         * @param {?} index
         * @return {?}
         */
        _mismatch(record, item, itemTrackBy, index) {
            // The previous record after which we will append the current one.
            /** @type {?} */
            let previousRecord;
            if (record === null) {
                previousRecord = this._itTail;
            }
            else {
                previousRecord = record._prev;
                // Remove the record from the collection since we know it does not match the item.
                this._remove(record);
            }
            // Attempt to see if we have seen the item before.
            record = this._linkedRecords === null ? null : this._linkedRecords.get(itemTrackBy, index);
            if (record !== null) {
                // We have seen this before, we need to move it forward in the collection.
                // But first we need to check if identity changed, so we can update in view if necessary
                if (!looseIdentical(record.item, item))
                    this._addIdentityChange(record, item);
                this._moveAfter(record, previousRecord, index);
            }
            else {
                // Never seen it, check evicted list.
                record = this._unlinkedRecords === null ? null : this._unlinkedRecords.get(itemTrackBy, null);
                if (record !== null) {
                    // It is an item which we have evicted earlier: reinsert it back into the list.
                    // But first we need to check if identity changed, so we can update in view if necessary
                    if (!looseIdentical(record.item, item))
                        this._addIdentityChange(record, item);
                    this._reinsertAfter(record, previousRecord, index);
                }
                else {
                    // It is a new item: add it.
                    record =
                        this._addAfter(new IterableChangeRecord_(item, itemTrackBy), previousRecord, index);
                }
            }
            return record;
        }
        /**
         * This check is only needed if an array contains duplicates. (Short circuit of nothing dirty)
         *
         * Use case: `[a, a]` => `[b, a, a]`
         *
         * If we did not have this check then the insertion of `b` would:
         *   1) evict first `a`
         *   2) insert `b` at `0` index.
         *   3) leave `a` at index `1` as is. <-- this is wrong!
         *   3) reinsert `a` at index 2. <-- this is wrong!
         *
         * The correct behavior is:
         *   1) evict first `a`
         *   2) insert `b` at `0` index.
         *   3) reinsert `a` at index 1.
         *   3) move `a` at from `1` to `2`.
         *
         *
         * Double check that we have not evicted a duplicate item. We need to check if the item type may
         * have already been removed:
         * The insertion of b will evict the first 'a'. If we don't reinsert it now it will be reinserted
         * at the end. Which will show up as the two 'a's switching position. This is incorrect, since a
         * better way to think of it is as insert of 'b' rather then switch 'a' with 'b' and then add 'a'
         * at the end.
         *
         * \@internal
         * @param {?} record
         * @param {?} item
         * @param {?} itemTrackBy
         * @param {?} index
         * @return {?}
         */
        _verifyReinsertion(record, item, itemTrackBy, index) {
            /** @type {?} */
            let reinsertRecord = this._unlinkedRecords === null ? null : this._unlinkedRecords.get(itemTrackBy, null);
            if (reinsertRecord !== null) {
                record = this._reinsertAfter(reinsertRecord, (/** @type {?} */ (record._prev)), index);
            }
            else if (record.currentIndex != index) {
                record.currentIndex = index;
                this._addToMoves(record, index);
            }
            return record;
        }
        /**
         * Get rid of any excess {\@link IterableChangeRecord_}s from the previous collection
         *
         * - `record` The first excess {\@link IterableChangeRecord_}.
         *
         * \@internal
         * @param {?} record
         * @return {?}
         */
        _truncate(record) {
            // Anything after that needs to be removed;
            while (record !== null) {
                /** @type {?} */
                const nextRecord = record._next;
                this._addToRemovals(this._unlink(record));
                record = nextRecord;
            }
            if (this._unlinkedRecords !== null) {
                this._unlinkedRecords.clear();
            }
            if (this._additionsTail !== null) {
                this._additionsTail._nextAdded = null;
            }
            if (this._movesTail !== null) {
                this._movesTail._nextMoved = null;
            }
            if (this._itTail !== null) {
                this._itTail._next = null;
            }
            if (this._removalsTail !== null) {
                this._removalsTail._nextRemoved = null;
            }
            if (this._identityChangesTail !== null) {
                this._identityChangesTail._nextIdentityChange = null;
            }
        }
        /**
         * \@internal
         * @param {?} record
         * @param {?} prevRecord
         * @param {?} index
         * @return {?}
         */
        _reinsertAfter(record, prevRecord, index) {
            if (this._unlinkedRecords !== null) {
                this._unlinkedRecords.remove(record);
            }
            /** @type {?} */
            const prev = record._prevRemoved;
            /** @type {?} */
            const next = record._nextRemoved;
            if (prev === null) {
                this._removalsHead = next;
            }
            else {
                prev._nextRemoved = next;
            }
            if (next === null) {
                this._removalsTail = prev;
            }
            else {
                next._prevRemoved = prev;
            }
            this._insertAfter(record, prevRecord, index);
            this._addToMoves(record, index);
            return record;
        }
        /**
         * \@internal
         * @param {?} record
         * @param {?} prevRecord
         * @param {?} index
         * @return {?}
         */
        _moveAfter(record, prevRecord, index) {
            this._unlink(record);
            this._insertAfter(record, prevRecord, index);
            this._addToMoves(record, index);
            return record;
        }
        /**
         * \@internal
         * @param {?} record
         * @param {?} prevRecord
         * @param {?} index
         * @return {?}
         */
        _addAfter(record, prevRecord, index) {
            this._insertAfter(record, prevRecord, index);
            if (this._additionsTail === null) {
                // TODO(vicb):
                // assert(this._additionsHead === null);
                this._additionsTail = this._additionsHead = record;
            }
            else {
                // TODO(vicb):
                // assert(_additionsTail._nextAdded === null);
                // assert(record._nextAdded === null);
                this._additionsTail = this._additionsTail._nextAdded = record;
            }
            return record;
        }
        /**
         * \@internal
         * @param {?} record
         * @param {?} prevRecord
         * @param {?} index
         * @return {?}
         */
        _insertAfter(record, prevRecord, index) {
            // TODO(vicb):
            // assert(record != prevRecord);
            // assert(record._next === null);
            // assert(record._prev === null);
            // TODO(vicb):
            // assert(record != prevRecord);
            // assert(record._next === null);
            // assert(record._prev === null);
            /** @type {?} */
            const next = prevRecord === null ? this._itHead : prevRecord._next;
            // TODO(vicb):
            // assert(next != record);
            // assert(prevRecord != record);
            record._next = next;
            record._prev = prevRecord;
            if (next === null) {
                this._itTail = record;
            }
            else {
                next._prev = record;
            }
            if (prevRecord === null) {
                this._itHead = record;
            }
            else {
                prevRecord._next = record;
            }
            if (this._linkedRecords === null) {
                this._linkedRecords = new _DuplicateMap();
            }
            this._linkedRecords.put(record);
            record.currentIndex = index;
            return record;
        }
        /**
         * \@internal
         * @param {?} record
         * @return {?}
         */
        _remove(record) {
            return this._addToRemovals(this._unlink(record));
        }
        /**
         * \@internal
         * @param {?} record
         * @return {?}
         */
        _unlink(record) {
            if (this._linkedRecords !== null) {
                this._linkedRecords.remove(record);
            }
            /** @type {?} */
            const prev = record._prev;
            /** @type {?} */
            const next = record._next;
            // TODO(vicb):
            // assert((record._prev = null) === null);
            // assert((record._next = null) === null);
            if (prev === null) {
                this._itHead = next;
            }
            else {
                prev._next = next;
            }
            if (next === null) {
                this._itTail = prev;
            }
            else {
                next._prev = prev;
            }
            return record;
        }
        /**
         * \@internal
         * @param {?} record
         * @param {?} toIndex
         * @return {?}
         */
        _addToMoves(record, toIndex) {
            // TODO(vicb):
            // assert(record._nextMoved === null);
            if (record.previousIndex === toIndex) {
                return record;
            }
            if (this._movesTail === null) {
                // TODO(vicb):
                // assert(_movesHead === null);
                this._movesTail = this._movesHead = record;
            }
            else {
                // TODO(vicb):
                // assert(_movesTail._nextMoved === null);
                this._movesTail = this._movesTail._nextMoved = record;
            }
            return record;
        }
        /**
         * @private
         * @param {?} record
         * @return {?}
         */
        _addToRemovals(record) {
            if (this._unlinkedRecords === null) {
                this._unlinkedRecords = new _DuplicateMap();
            }
            this._unlinkedRecords.put(record);
            record.currentIndex = null;
            record._nextRemoved = null;
            if (this._removalsTail === null) {
                // TODO(vicb):
                // assert(_removalsHead === null);
                this._removalsTail = this._removalsHead = record;
                record._prevRemoved = null;
            }
            else {
                // TODO(vicb):
                // assert(_removalsTail._nextRemoved === null);
                // assert(record._nextRemoved === null);
                record._prevRemoved = this._removalsTail;
                this._removalsTail = this._removalsTail._nextRemoved = record;
            }
            return record;
        }
        /**
         * \@internal
         * @param {?} record
         * @param {?} item
         * @return {?}
         */
        _addIdentityChange(record, item) {
            record.item = item;
            if (this._identityChangesTail === null) {
                this._identityChangesTail = this._identityChangesHead = record;
            }
            else {
                this._identityChangesTail = this._identityChangesTail._nextIdentityChange = record;
            }
            return record;
        }
    }
    /**
     * @template V
     */
    class IterableChangeRecord_ {
        /**
         * @param {?} item
         * @param {?} trackById
         */
        constructor(item, trackById) {
            this.item = item;
            this.trackById = trackById;
            this.currentIndex = null;
            this.previousIndex = null;
            /**
             * \@internal
             */
            this._nextPrevious = null;
            /**
             * \@internal
             */
            this._prev = null;
            /**
             * \@internal
             */
            this._next = null;
            /**
             * \@internal
             */
            this._prevDup = null;
            /**
             * \@internal
             */
            this._nextDup = null;
            /**
             * \@internal
             */
            this._prevRemoved = null;
            /**
             * \@internal
             */
            this._nextRemoved = null;
            /**
             * \@internal
             */
            this._nextAdded = null;
            /**
             * \@internal
             */
            this._nextMoved = null;
            /**
             * \@internal
             */
            this._nextIdentityChange = null;
        }
    }
    // A linked list of CollectionChangeRecords with the same IterableChangeRecord_.item
    /**
     * @template V
     */
    class _DuplicateItemRecordList {
        constructor() {
            /**
             * \@internal
             */
            this._head = null;
            /**
             * \@internal
             */
            this._tail = null;
        }
        /**
         * Append the record to the list of duplicates.
         *
         * Note: by design all records in the list of duplicates hold the same value in record.item.
         * @param {?} record
         * @return {?}
         */
        add(record) {
            if (this._head === null) {
                this._head = this._tail = record;
                record._nextDup = null;
                record._prevDup = null;
            }
            else {
                // TODO(vicb):
                // assert(record.item ==  _head.item ||
                //       record.item is num && record.item.isNaN && _head.item is num && _head.item.isNaN);
                (/** @type {?} */ (this._tail))._nextDup = record;
                record._prevDup = this._tail;
                record._nextDup = null;
                this._tail = record;
            }
        }
        // Returns a IterableChangeRecord_ having IterableChangeRecord_.trackById == trackById and
        // IterableChangeRecord_.currentIndex >= atOrAfterIndex
        /**
         * @param {?} trackById
         * @param {?} atOrAfterIndex
         * @return {?}
         */
        get(trackById, atOrAfterIndex) {
            /** @type {?} */
            let record;
            for (record = this._head; record !== null; record = record._nextDup) {
                if ((atOrAfterIndex === null || atOrAfterIndex <= (/** @type {?} */ (record.currentIndex))) &&
                    looseIdentical(record.trackById, trackById)) {
                    return record;
                }
            }
            return null;
        }
        /**
         * Remove one {\@link IterableChangeRecord_} from the list of duplicates.
         *
         * Returns whether the list of duplicates is empty.
         * @param {?} record
         * @return {?}
         */
        remove(record) {
            // TODO(vicb):
            // assert(() {
            //  // verify that the record being removed is in the list.
            //  for (IterableChangeRecord_ cursor = _head; cursor != null; cursor = cursor._nextDup) {
            //    if (identical(cursor, record)) return true;
            //  }
            //  return false;
            //});
            // TODO(vicb):
            // assert(() {
            //  // verify that the record being removed is in the list.
            //  for (IterableChangeRecord_ cursor = _head; cursor != null; cursor = cursor._nextDup) {
            //    if (identical(cursor, record)) return true;
            //  }
            //  return false;
            //});
            /** @type {?} */
            const prev = record._prevDup;
            /** @type {?} */
            const next = record._nextDup;
            if (prev === null) {
                this._head = next;
            }
            else {
                prev._nextDup = next;
            }
            if (next === null) {
                this._tail = prev;
            }
            else {
                next._prevDup = prev;
            }
            return this._head === null;
        }
    }
    /**
     * @template V
     */
    class _DuplicateMap {
        constructor() {
            this.map = new Map();
        }
        /**
         * @param {?} record
         * @return {?}
         */
        put(record) {
            /** @type {?} */
            const key = record.trackById;
            /** @type {?} */
            let duplicates = this.map.get(key);
            if (!duplicates) {
                duplicates = new _DuplicateItemRecordList();
                this.map.set(key, duplicates);
            }
            duplicates.add(record);
        }
        /**
         * Retrieve the `value` using key. Because the IterableChangeRecord_ value may be one which we
         * have already iterated over, we use the `atOrAfterIndex` to pretend it is not there.
         *
         * Use case: `[a, b, c, a, a]` if we are at index `3` which is the second `a` then asking if we
         * have any more `a`s needs to return the second `a`.
         * @param {?} trackById
         * @param {?} atOrAfterIndex
         * @return {?}
         */
        get(trackById, atOrAfterIndex) {
            /** @type {?} */
            const key = trackById;
            /** @type {?} */
            const recordList = this.map.get(key);
            return recordList ? recordList.get(trackById, atOrAfterIndex) : null;
        }
        /**
         * Removes a {\@link IterableChangeRecord_} from the list of duplicates.
         *
         * The list of duplicates also is removed from the map if it gets empty.
         * @param {?} record
         * @return {?}
         */
        remove(record) {
            /** @type {?} */
            const key = record.trackById;
            /** @type {?} */
            const recordList = (/** @type {?} */ (this.map.get(key)));
            // Remove the list of duplicates when it gets empty
            if (recordList.remove(record)) {
                this.map.delete(key);
            }
            return record;
        }
        /**
         * @return {?}
         */
        get isEmpty() {
            return this.map.size === 0;
        }
        /**
         * @return {?}
         */
        clear() {
            this.map.clear();
        }
    }
    /**
     * @param {?} item
     * @param {?} addRemoveOffset
     * @param {?} moveOffsets
     * @return {?}
     */
    function getPreviousIndex(item, addRemoveOffset, moveOffsets) {
        /** @type {?} */
        const previousIndex = item.previousIndex;
        if (previousIndex === null)
            return previousIndex;
        /** @type {?} */
        let moveOffset = 0;
        if (moveOffsets && previousIndex < moveOffsets.length) {
            moveOffset = moveOffsets[previousIndex];
        }
        return previousIndex + addRemoveOffset + moveOffset;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/change_detection/differs/default_keyvalue_differ.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @template K, V
     */
    class DefaultKeyValueDifferFactory {
        constructor() { }
        /**
         * @param {?} obj
         * @return {?}
         */
        supports(obj) {
            return obj instanceof Map || isJsObject$1(obj);
        }
        /**
         * @template K, V
         * @return {?}
         */
        create() {
            return new DefaultKeyValueDiffer();
        }
    }
    /**
     * @template K, V
     */
    class DefaultKeyValueDiffer {
        constructor() {
            this._records = new Map();
            this._mapHead = null;
            // _appendAfter is used in the check loop
            this._appendAfter = null;
            this._previousMapHead = null;
            this._changesHead = null;
            this._changesTail = null;
            this._additionsHead = null;
            this._additionsTail = null;
            this._removalsHead = null;
            this._removalsTail = null;
        }
        /**
         * @return {?}
         */
        get isDirty() {
            return this._additionsHead !== null || this._changesHead !== null ||
                this._removalsHead !== null;
        }
        /**
         * @param {?} fn
         * @return {?}
         */
        forEachItem(fn) {
            /** @type {?} */
            let record;
            for (record = this._mapHead; record !== null; record = record._next) {
                fn(record);
            }
        }
        /**
         * @param {?} fn
         * @return {?}
         */
        forEachPreviousItem(fn) {
            /** @type {?} */
            let record;
            for (record = this._previousMapHead; record !== null; record = record._nextPrevious) {
                fn(record);
            }
        }
        /**
         * @param {?} fn
         * @return {?}
         */
        forEachChangedItem(fn) {
            /** @type {?} */
            let record;
            for (record = this._changesHead; record !== null; record = record._nextChanged) {
                fn(record);
            }
        }
        /**
         * @param {?} fn
         * @return {?}
         */
        forEachAddedItem(fn) {
            /** @type {?} */
            let record;
            for (record = this._additionsHead; record !== null; record = record._nextAdded) {
                fn(record);
            }
        }
        /**
         * @param {?} fn
         * @return {?}
         */
        forEachRemovedItem(fn) {
            /** @type {?} */
            let record;
            for (record = this._removalsHead; record !== null; record = record._nextRemoved) {
                fn(record);
            }
        }
        /**
         * @param {?=} map
         * @return {?}
         */
        diff(map) {
            if (!map) {
                map = new Map();
            }
            else if (!(map instanceof Map || isJsObject$1(map))) {
                throw new Error(`Error trying to diff '${stringify(map)}'. Only maps and objects are allowed`);
            }
            return this.check(map) ? this : null;
        }
        /**
         * @return {?}
         */
        onDestroy() { }
        /**
         * Check the current state of the map vs the previous.
         * The algorithm is optimised for when the keys do no change.
         * @param {?} map
         * @return {?}
         */
        check(map) {
            this._reset();
            /** @type {?} */
            let insertBefore = this._mapHead;
            this._appendAfter = null;
            this._forEach(map, (/**
             * @param {?} value
             * @param {?} key
             * @return {?}
             */
            (value, key) => {
                if (insertBefore && insertBefore.key === key) {
                    this._maybeAddToChanges(insertBefore, value);
                    this._appendAfter = insertBefore;
                    insertBefore = insertBefore._next;
                }
                else {
                    /** @type {?} */
                    const record = this._getOrCreateRecordForKey(key, value);
                    insertBefore = this._insertBeforeOrAppend(insertBefore, record);
                }
            }));
            // Items remaining at the end of the list have been deleted
            if (insertBefore) {
                if (insertBefore._prev) {
                    insertBefore._prev._next = null;
                }
                this._removalsHead = insertBefore;
                for (let record = insertBefore; record !== null; record = record._nextRemoved) {
                    if (record === this._mapHead) {
                        this._mapHead = null;
                    }
                    this._records.delete(record.key);
                    record._nextRemoved = record._next;
                    record.previousValue = record.currentValue;
                    record.currentValue = null;
                    record._prev = null;
                    record._next = null;
                }
            }
            // Make sure tails have no next records from previous runs
            if (this._changesTail)
                this._changesTail._nextChanged = null;
            if (this._additionsTail)
                this._additionsTail._nextAdded = null;
            return this.isDirty;
        }
        /**
         * Inserts a record before `before` or append at the end of the list when `before` is null.
         *
         * Notes:
         * - This method appends at `this._appendAfter`,
         * - This method updates `this._appendAfter`,
         * - The return value is the new value for the insertion pointer.
         * @private
         * @param {?} before
         * @param {?} record
         * @return {?}
         */
        _insertBeforeOrAppend(before, record) {
            if (before) {
                /** @type {?} */
                const prev = before._prev;
                record._next = before;
                record._prev = prev;
                before._prev = record;
                if (prev) {
                    prev._next = record;
                }
                if (before === this._mapHead) {
                    this._mapHead = record;
                }
                this._appendAfter = before;
                return before;
            }
            if (this._appendAfter) {
                this._appendAfter._next = record;
                record._prev = this._appendAfter;
            }
            else {
                this._mapHead = record;
            }
            this._appendAfter = record;
            return null;
        }
        /**
         * @private
         * @param {?} key
         * @param {?} value
         * @return {?}
         */
        _getOrCreateRecordForKey(key, value) {
            if (this._records.has(key)) {
                /** @type {?} */
                const record = (/** @type {?} */ (this._records.get(key)));
                this._maybeAddToChanges(record, value);
                /** @type {?} */
                const prev = record._prev;
                /** @type {?} */
                const next = record._next;
                if (prev) {
                    prev._next = next;
                }
                if (next) {
                    next._prev = prev;
                }
                record._next = null;
                record._prev = null;
                return record;
            }
            /** @type {?} */
            const record = new KeyValueChangeRecord_(key);
            this._records.set(key, record);
            record.currentValue = value;
            this._addToAdditions(record);
            return record;
        }
        /**
         * \@internal
         * @return {?}
         */
        _reset() {
            if (this.isDirty) {
                /** @type {?} */
                let record;
                // let `_previousMapHead` contain the state of the map before the changes
                this._previousMapHead = this._mapHead;
                for (record = this._previousMapHead; record !== null; record = record._next) {
                    record._nextPrevious = record._next;
                }
                // Update `record.previousValue` with the value of the item before the changes
                // We need to update all changed items (that's those which have been added and changed)
                for (record = this._changesHead; record !== null; record = record._nextChanged) {
                    record.previousValue = record.currentValue;
                }
                for (record = this._additionsHead; record != null; record = record._nextAdded) {
                    record.previousValue = record.currentValue;
                }
                this._changesHead = this._changesTail = null;
                this._additionsHead = this._additionsTail = null;
                this._removalsHead = null;
            }
        }
        // Add the record or a given key to the list of changes only when the value has actually changed
        /**
         * @private
         * @param {?} record
         * @param {?} newValue
         * @return {?}
         */
        _maybeAddToChanges(record, newValue) {
            if (!looseIdentical(newValue, record.currentValue)) {
                record.previousValue = record.currentValue;
                record.currentValue = newValue;
                this._addToChanges(record);
            }
        }
        /**
         * @private
         * @param {?} record
         * @return {?}
         */
        _addToAdditions(record) {
            if (this._additionsHead === null) {
                this._additionsHead = this._additionsTail = record;
            }
            else {
                (/** @type {?} */ (this._additionsTail))._nextAdded = record;
                this._additionsTail = record;
            }
        }
        /**
         * @private
         * @param {?} record
         * @return {?}
         */
        _addToChanges(record) {
            if (this._changesHead === null) {
                this._changesHead = this._changesTail = record;
            }
            else {
                (/** @type {?} */ (this._changesTail))._nextChanged = record;
                this._changesTail = record;
            }
        }
        /**
         * \@internal
         * @private
         * @template K, V
         * @param {?} obj
         * @param {?} fn
         * @return {?}
         */
        _forEach(obj, fn) {
            if (obj instanceof Map) {
                obj.forEach(fn);
            }
            else {
                Object.keys(obj).forEach((/**
                 * @param {?} k
                 * @return {?}
                 */
                k => fn(obj[k], k)));
            }
        }
    }
    /**
     * @template K, V
     */
    class KeyValueChangeRecord_ {
        /**
         * @param {?} key
         */
        constructor(key) {
            this.key = key;
            this.previousValue = null;
            this.currentValue = null;
            /**
             * \@internal
             */
            this._nextPrevious = null;
            /**
             * \@internal
             */
            this._next = null;
            /**
             * \@internal
             */
            this._prev = null;
            /**
             * \@internal
             */
            this._nextAdded = null;
            /**
             * \@internal
             */
            this._nextRemoved = null;
            /**
             * \@internal
             */
            this._nextChanged = null;
        }
    }
    /**
     * A repository of different iterable diffing strategies used by NgFor, NgClass, and others.
     *
     * \@publicApi
     */
    class IterableDiffers {
        /**
         * @param {?} factories
         */
        constructor(factories) {
            this.factories = factories;
        }
        /**
         * @param {?} factories
         * @param {?=} parent
         * @return {?}
         */
        static create(factories, parent) {
            if (parent != null) {
                /** @type {?} */
                const copied = parent.factories.slice();
                factories = factories.concat(copied);
            }
            return new IterableDiffers(factories);
        }
        /**
         * Takes an array of {\@link IterableDifferFactory} and returns a provider used to extend the
         * inherited {\@link IterableDiffers} instance with the provided factories and return a new
         * {\@link IterableDiffers} instance.
         *
         * \@usageNotes
         * ### Example
         *
         * The following example shows how to extend an existing list of factories,
         * which will only be applied to the injector for this component and its children.
         * This step is all that's required to make a new {\@link IterableDiffer} available.
         *
         * ```
         * \@Component({
         *   viewProviders: [
         *     IterableDiffers.extend([new ImmutableListDiffer()])
         *   ]
         * })
         * ```
         * @param {?} factories
         * @return {?}
         */
        static extend(factories) {
            return {
                provide: IterableDiffers,
                useFactory: (/**
                 * @param {?} parent
                 * @return {?}
                 */
                (parent) => {
                    if (!parent) {
                        // Typically would occur when calling IterableDiffers.extend inside of dependencies passed
                        // to
                        // bootstrap(), which would override default pipes instead of extending them.
                        throw new Error('Cannot extend IterableDiffers without a parent injector');
                    }
                    return IterableDiffers.create(factories, parent);
                }),
                // Dependency technically isn't optional, but we can provide a better error message this way.
                deps: [[IterableDiffers, new SkipSelf(), new Optional()]]
            };
        }
        /**
         * @param {?} iterable
         * @return {?}
         */
        find(iterable) {
            /** @type {?} */
            const factory = this.factories.find((/**
             * @param {?} f
             * @return {?}
             */
            f => f.supports(iterable)));
            if (factory != null) {
                return factory;
            }
            else {
                throw new Error(`Cannot find a differ supporting object '${iterable}' of type '${getTypeNameForDebugging(iterable)}'`);
            }
        }
    }
    /** @nocollapse */
    IterableDiffers.ɵprov = ɵɵdefineInjectable({
        token: IterableDiffers,
        providedIn: 'root',
        factory: (/**
         * @return {?}
         */
        () => new IterableDiffers([new DefaultIterableDifferFactory()]))
    });
    /**
     * @param {?} type
     * @return {?}
     */
    function getTypeNameForDebugging(type) {
        return type['name'] || typeof type;
    }
    /**
     * A repository of different Map diffing strategies used by NgClass, NgStyle, and others.
     *
     * \@publicApi
     */
    class KeyValueDiffers {
        /**
         * @param {?} factories
         */
        constructor(factories) {
            this.factories = factories;
        }
        /**
         * @template S
         * @param {?} factories
         * @param {?=} parent
         * @return {?}
         */
        static create(factories, parent) {
            if (parent) {
                /** @type {?} */
                const copied = parent.factories.slice();
                factories = factories.concat(copied);
            }
            return new KeyValueDiffers(factories);
        }
        /**
         * Takes an array of {\@link KeyValueDifferFactory} and returns a provider used to extend the
         * inherited {\@link KeyValueDiffers} instance with the provided factories and return a new
         * {\@link KeyValueDiffers} instance.
         *
         * \@usageNotes
         * ### Example
         *
         * The following example shows how to extend an existing list of factories,
         * which will only be applied to the injector for this component and its children.
         * This step is all that's required to make a new {\@link KeyValueDiffer} available.
         *
         * ```
         * \@Component({
         *   viewProviders: [
         *     KeyValueDiffers.extend([new ImmutableMapDiffer()])
         *   ]
         * })
         * ```
         * @template S
         * @param {?} factories
         * @return {?}
         */
        static extend(factories) {
            return {
                provide: KeyValueDiffers,
                useFactory: (/**
                 * @param {?} parent
                 * @return {?}
                 */
                (parent) => {
                    if (!parent) {
                        // Typically would occur when calling KeyValueDiffers.extend inside of dependencies passed
                        // to bootstrap(), which would override default pipes instead of extending them.
                        throw new Error('Cannot extend KeyValueDiffers without a parent injector');
                    }
                    return KeyValueDiffers.create(factories, parent);
                }),
                // Dependency technically isn't optional, but we can provide a better error message this way.
                deps: [[KeyValueDiffers, new SkipSelf(), new Optional()]]
            };
        }
        /**
         * @param {?} kv
         * @return {?}
         */
        find(kv) {
            /** @type {?} */
            const factory = this.factories.find((/**
             * @param {?} f
             * @return {?}
             */
            f => f.supports(kv)));
            if (factory) {
                return factory;
            }
            throw new Error(`Cannot find a differ supporting object '${kv}'`);
        }
    }
    /** @nocollapse */
    KeyValueDiffers.ɵprov = ɵɵdefineInjectable({
        token: KeyValueDiffers,
        providedIn: 'root',
        factory: (/**
         * @return {?}
         */
        () => new KeyValueDiffers([new DefaultKeyValueDifferFactory()]))
    });

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/change_detection/change_detection.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Structural diffing for `Object`s and `Map`s.
     * @type {?}
     */
    const keyValDiff = [new DefaultKeyValueDifferFactory()];
    /**
     * Structural diffing for `Iterable` types such as `Array`s.
     * @type {?}
     */
    const iterableDiff = [new DefaultIterableDifferFactory()];
    /** @type {?} */
    const defaultIterableDiffers = new IterableDiffers(iterableDiff);
    /** @type {?} */
    const defaultKeyValueDiffers = new KeyValueDiffers(keyValDiff);

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/linker/template_ref.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Represents an embedded template that can be used to instantiate embedded views.
     * To instantiate embedded views based on a template, use the `ViewContainerRef`
     * method `createEmbeddedView()`.
     *
     * Access a `TemplateRef` instance by placing a directive on an `<ng-template>`
     * element (or directive prefixed with `*`). The `TemplateRef` for the embedded view
     * is injected into the constructor of the directive,
     * using the `TemplateRef` token.
     *
     * You can also use a `Query` to find a `TemplateRef` associated with
     * a component or a directive.
     *
     * @see `ViewContainerRef`
     * @see [Navigate the Component Tree with DI](guide/dependency-injection-navtree)
     *
     * \@publicApi
     * @abstract
     * @template C
     */
    class TemplateRef {
    }
    /**
     * \@internal
     * @nocollapse
     */
    TemplateRef.__NG_ELEMENT_ID__ = (/**
     * @return {?}
     */
    () => SWITCH_TEMPLATE_REF_FACTORY(TemplateRef, ElementRef));
    /** @type {?} */
    const SWITCH_TEMPLATE_REF_FACTORY__POST_R3__ = injectTemplateRef;
    /** @type {?} */
    const SWITCH_TEMPLATE_REF_FACTORY = SWITCH_TEMPLATE_REF_FACTORY__POST_R3__;

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/linker/view_container_ref.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Represents a container where one or more views can be attached to a component.
     *
     * Can contain *host views* (created by instantiating a
     * component with the `createComponent()` method), and *embedded views*
     * (created by instantiating a `TemplateRef` with the `createEmbeddedView()` method).
     *
     * A view container instance can contain other view containers,
     * creating a [view hierarchy](guide/glossary#view-tree).
     *
     * @see `ComponentRef`
     * @see `EmbeddedViewRef`
     *
     * \@publicApi
     * @abstract
     */
    class ViewContainerRef {
    }
    /**
     * \@internal
     * @nocollapse
     */
    ViewContainerRef.__NG_ELEMENT_ID__ = (/**
     * @return {?}
     */
    () => SWITCH_VIEW_CONTAINER_REF_FACTORY(ViewContainerRef, ElementRef));
    /** @type {?} */
    const SWITCH_VIEW_CONTAINER_REF_FACTORY__POST_R3__ = injectViewContainerRef;
    /** @type {?} */
    const SWITCH_VIEW_CONTAINER_REF_FACTORY = SWITCH_VIEW_CONTAINER_REF_FACTORY__POST_R3__;
    /** @type {?} */
    const _tokenKeyCache = new Map();
    /**
     * @param {?} token
     * @return {?}
     */
    function tokenKey(token) {
        /** @type {?} */
        let key = _tokenKeyCache.get(token);
        if (!key) {
            key = stringify(token) + '_' + _tokenKeyCache.size;
            _tokenKeyCache.set(token, key);
        }
        return key;
    }
    /** @type {?} */
    const InjectorRefTokenKey = tokenKey(Injector);
    /** @type {?} */
    const INJECTORRefTokenKey = tokenKey(INJECTOR);
    /** @type {?} */
    const NgModuleRefTokenKey = tokenKey(NgModuleRef);

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/view/provider.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const Renderer2TokenKey = tokenKey(Renderer2);
    /** @type {?} */
    const ElementRefTokenKey = tokenKey(ElementRef);
    /** @type {?} */
    const ViewContainerRefTokenKey = tokenKey(ViewContainerRef);
    /** @type {?} */
    const TemplateRefTokenKey = tokenKey(TemplateRef);
    /** @type {?} */
    const ChangeDetectorRefTokenKey = tokenKey(ChangeDetectorRef);
    /** @type {?} */
    const InjectorRefTokenKey$1 = tokenKey(Injector);
    /** @type {?} */
    const INJECTORRefTokenKey$1 = tokenKey(INJECTOR);
    // This default value is when checking the hierarchy for a token.
    //
    // It means both:
    // - the token is not provided by the current injector,
    // - only the element injectors should be checked (ie do not check module injectors
    //
    //          mod1
    //         /
    //       el1   mod2
    //         \  /
    //         el2
    //
    // When requesting el2.injector.get(token), we should check in the following order and return the
    // first found value:
    // - el2.injector.get(token, default)
    // - el1.injector.get(token, NOT_FOUND_CHECK_ONLY_ELEMENT_INJECTOR) -> do not check the module
    // - mod2.injector.get(token, default)
    /** @type {?} */
    const NOT_FOUND_CHECK_ONLY_ELEMENT_INJECTOR = {};

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/component_ref.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class ComponentFactoryResolver$1 extends ComponentFactoryResolver {
        /**
         * @param {?=} ngModule The NgModuleRef to which all resolved factories are bound.
         */
        constructor(ngModule) {
            super();
            this.ngModule = ngModule;
        }
        /**
         * @template T
         * @param {?} component
         * @return {?}
         */
        resolveComponentFactory(component) {
            ngDevMode && assertComponentType(component);
            /** @type {?} */
            const componentDef = (/** @type {?} */ (getComponentDef(component)));
            return new ComponentFactory$1(componentDef, this.ngModule);
        }
    }
    /**
     * @param {?} map
     * @return {?}
     */
    function toRefArray(map) {
        /** @type {?} */
        const array = [];
        for (let nonMinified in map) {
            if (map.hasOwnProperty(nonMinified)) {
                /** @type {?} */
                const minified = map[nonMinified];
                array.push({ propName: minified, templateName: nonMinified });
            }
        }
        return array;
    }
    /**
     * @param {?} elementName
     * @return {?}
     */
    function getNamespace$1(elementName) {
        /** @type {?} */
        const name = elementName.toLowerCase();
        return name === 'svg' ? SVG_NAMESPACE : (name === 'math' ? MATH_ML_NAMESPACE : null);
    }
    /**
     * A change detection scheduler token for {\@link RootContext}. This token is the default value used
     * for the default `RootContext` found in the {\@link ROOT_CONTEXT} token.
     * @type {?}
     */
    const SCHEDULER = new InjectionToken('SCHEDULER_TOKEN', {
        providedIn: 'root',
        factory: (/**
         * @return {?}
         */
        () => defaultScheduler),
    });
    /**
     * @param {?} rootViewInjector
     * @param {?} moduleInjector
     * @return {?}
     */
    function createChainedInjector(rootViewInjector, moduleInjector) {
        return {
            get: (/**
             * @template T
             * @param {?} token
             * @param {?=} notFoundValue
             * @param {?=} flags
             * @return {?}
             */
            (token, notFoundValue, flags) => {
                /** @type {?} */
                const value = rootViewInjector.get(token, (/** @type {?} */ (NOT_FOUND_CHECK_ONLY_ELEMENT_INJECTOR)), flags);
                if (value !== NOT_FOUND_CHECK_ONLY_ELEMENT_INJECTOR ||
                    notFoundValue === NOT_FOUND_CHECK_ONLY_ELEMENT_INJECTOR) {
                    // Return the value from the root element injector when
                    // - it provides it
                    //   (value !== NOT_FOUND_CHECK_ONLY_ELEMENT_INJECTOR)
                    // - the module injector should not be checked
                    //   (notFoundValue === NOT_FOUND_CHECK_ONLY_ELEMENT_INJECTOR)
                    return value;
                }
                return moduleInjector.get(token, notFoundValue, flags);
            })
        };
    }
    /**
     * Render3 implementation of {\@link viewEngine_ComponentFactory}.
     * @template T
     */
    class ComponentFactory$1 extends ComponentFactory {
        /**
         * @param {?} componentDef The component definition.
         * @param {?=} ngModule The NgModuleRef to which the factory is bound.
         */
        constructor(componentDef, ngModule) {
            super();
            this.componentDef = componentDef;
            this.ngModule = ngModule;
            this.componentType = componentDef.type;
            this.selector = stringifyCSSSelectorList(componentDef.selectors);
            this.ngContentSelectors =
                componentDef.ngContentSelectors ? componentDef.ngContentSelectors : [];
            this.isBoundToModule = !!ngModule;
        }
        /**
         * @return {?}
         */
        get inputs() {
            return toRefArray(this.componentDef.inputs);
        }
        /**
         * @return {?}
         */
        get outputs() {
            return toRefArray(this.componentDef.outputs);
        }
        /**
         * @param {?} injector
         * @param {?=} projectableNodes
         * @param {?=} rootSelectorOrNode
         * @param {?=} ngModule
         * @return {?}
         */
        create(injector, projectableNodes, rootSelectorOrNode, ngModule) {
            ngModule = ngModule || this.ngModule;
            /** @type {?} */
            const rootViewInjector = ngModule ? createChainedInjector(injector, ngModule.injector) : injector;
            /** @type {?} */
            const rendererFactory = (/** @type {?} */ (rootViewInjector.get(RendererFactory2, domRendererFactory3)));
            /** @type {?} */
            const sanitizer = rootViewInjector.get(Sanitizer, null);
            /** @type {?} */
            const hostRenderer = rendererFactory.createRenderer(null, this.componentDef);
            // Determine a tag name used for creating host elements when this component is created
            // dynamically. Default to 'div' if this component did not specify any tag name in its selector.
            /** @type {?} */
            const elementName = (/** @type {?} */ (this.componentDef.selectors[0][0])) || 'div';
            /** @type {?} */
            const hostRNode = rootSelectorOrNode ?
                locateHostElement(hostRenderer, rootSelectorOrNode, this.componentDef.encapsulation) :
                elementCreate(elementName, rendererFactory.createRenderer(null, this.componentDef), getNamespace$1(elementName));
            /** @type {?} */
            const rootFlags = this.componentDef.onPush ? 64 /* Dirty */ | 512 /* IsRoot */ :
                16 /* CheckAlways */ | 512 /* IsRoot */;
            // Check whether this Component needs to be isolated from other components, i.e. whether it
            // should be placed into its own (empty) root context or existing root context should be used.
            // Note: this is internal-only convention and might change in the future, so it should not be
            // relied upon externally.
            /** @type {?} */
            const isIsolated = typeof rootSelectorOrNode === 'string' &&
                /^#root-ng-internal-isolated-\d+/.test(rootSelectorOrNode);
            /** @type {?} */
            const rootContext = createRootContext();
            // Create the root view. Uses empty TView and ContentTemplate.
            /** @type {?} */
            const rootTView = createTView(0 /* Root */, -1, null, 1, 0, null, null, null, null, null);
            /** @type {?} */
            const rootLView = createLView(null, rootTView, rootContext, rootFlags, null, null, rendererFactory, hostRenderer, sanitizer, rootViewInjector);
            // rootView is the parent when bootstrapping
            // TODO(misko): it looks like we are entering view here but we don't really need to as
            // `renderView` does that. However as the code is written it is needed because
            // `createRootComponentView` and `createRootComponent` both read global state. Fixing those
            // issues would allow us to drop this.
            enterView(rootLView, null);
            /** @type {?} */
            let component;
            /** @type {?} */
            let tElementNode;
            try {
                /** @type {?} */
                const componentView = createRootComponentView(hostRNode, this.componentDef, rootLView, rendererFactory, hostRenderer);
                if (hostRNode) {
                    if (rootSelectorOrNode) {
                        setUpAttributes(hostRenderer, hostRNode, ['ng-version', VERSION.full]);
                    }
                    else {
                        // If host element is created as a part of this function call (i.e. `rootSelectorOrNode`
                        // is not defined), also apply attributes and classes extracted from component selector.
                        // Extract attributes and classes from the first selector only to match VE behavior.
                        const { attrs, classes } = extractAttrsAndClassesFromSelector(this.componentDef.selectors[0]);
                        if (attrs) {
                            setUpAttributes(hostRenderer, hostRNode, attrs);
                        }
                        if (classes && classes.length > 0) {
                            writeDirectClass(hostRenderer, hostRNode, classes.join(' '));
                        }
                    }
                }
                tElementNode = (/** @type {?} */ (getTNode(rootTView, 0)));
                if (projectableNodes !== undefined) {
                    /** @type {?} */
                    const projection = tElementNode.projection = [];
                    for (let i = 0; i < this.ngContentSelectors.length; i++) {
                        /** @type {?} */
                        const nodesforSlot = projectableNodes[i];
                        // Projectable nodes can be passed as array of arrays or an array of iterables (ngUpgrade
                        // case). Here we do normalize passed data structure to be an array of arrays to avoid
                        // complex checks down the line.
                        // We also normalize the length of the passed in projectable nodes (to match the number of
                        // <ng-container> slots defined by a component).
                        projection.push(nodesforSlot != null ? Array.from(nodesforSlot) : null);
                    }
                }
                // TODO: should LifecycleHooksFeature and other host features be generated by the compiler and
                // executed here?
                // Angular 5 reference: https://stackblitz.com/edit/lifecycle-hooks-vcref
                component = createRootComponent(componentView, this.componentDef, rootLView, rootContext, [LifecycleHooksFeature]);
                renderView(rootTView, rootLView, null);
            }
            finally {
                leaveView();
            }
            /** @type {?} */
            const componentRef = new ComponentRef$1(this.componentType, component, createElementRef(ElementRef, tElementNode, rootLView), rootLView, tElementNode);
            if (!rootSelectorOrNode || isIsolated) {
                // The host element of the internal or isolated root view is attached to the component's host
                // view node.
                ngDevMode && assertNodeOfPossibleTypes(rootTView.node, 2 /* View */);
                (/** @type {?} */ (rootTView.node)).child = tElementNode;
            }
            return componentRef;
        }
    }
    /** @type {?} */
    const componentFactoryResolver = new ComponentFactoryResolver$1();
    /**
     * Represents an instance of a Component created via a {\@link ComponentFactory}.
     *
     * `ComponentRef` provides access to the Component Instance as well other objects related to this
     * Component Instance and allows you to destroy the Component Instance via the {\@link #destroy}
     * method.
     *
     * @template T
     */
    class ComponentRef$1 extends ComponentRef {
        /**
         * @param {?} componentType
         * @param {?} instance
         * @param {?} location
         * @param {?} _rootLView
         * @param {?} _tNode
         */
        constructor(componentType, instance, location, _rootLView, _tNode) {
            super();
            this.location = location;
            this._rootLView = _rootLView;
            this._tNode = _tNode;
            this.destroyCbs = [];
            this.instance = instance;
            this.hostView = this.changeDetectorRef = new RootViewRef(_rootLView);
            assignTViewNodeToLView(_rootLView[TVIEW], null, -1, _rootLView);
            this.componentType = componentType;
        }
        /**
         * @return {?}
         */
        get injector() {
            return new NodeInjector(this._tNode, this._rootLView);
        }
        /**
         * @return {?}
         */
        destroy() {
            if (this.destroyCbs) {
                this.destroyCbs.forEach((/**
                 * @param {?} fn
                 * @return {?}
                 */
                fn => fn()));
                this.destroyCbs = null;
                !this.hostView.destroyed && this.hostView.destroy();
            }
        }
        /**
         * @param {?} callback
         * @return {?}
         */
        onDestroy(callback) {
            if (this.destroyCbs) {
                this.destroyCbs.push(callback);
            }
        }
    }

    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * NOTE: changes to the `ngI18nClosureMode` name must be synced with `compiler-cli/src/tooling.ts`.
     */
    if (typeof ngI18nClosureMode === 'undefined') {
        // These property accesses can be ignored because ngI18nClosureMode will be set to false
        // when optimizing code and the whole if statement will be dropped.
        // Make sure to refer to ngI18nClosureMode as ['ngI18nClosureMode'] for closure.
        // NOTE: we need to have it in IIFE so that the tree-shaker is happy.
        (function () {
            // tslint:disable-next-line:no-toplevel-property-access
            _global['ngI18nClosureMode'] =
                // TODO(FW-1250): validate that this actually, you know, works.
                // tslint:disable-next-line:no-toplevel-property-access
                typeof goog !== 'undefined' && typeof goog.getMsg === 'function';
        })();
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/i18n/locale_en.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    // THIS CODE IS GENERATED - DO NOT MODIFY
    // See angular/tools/gulp-tasks/cldr/extract.js
    /** @type {?} */
    const u = undefined;
    /**
     * @param {?} n
     * @return {?}
     */
    function plural(n) {
        /** @type {?} */
        let i = Math.floor(Math.abs(n));
        /** @type {?} */
        let v = n.toString().replace(/^[^.]*\.?/, '').length;
        if (i === 1 && v === 0)
            return 1;
        return 5;
    }
    var localeEn = [
        'en',
        [['a', 'p'], ['AM', 'PM'], u],
        [['AM', 'PM'], u, u],
        [
            ['S', 'M', 'T', 'W', 'T', 'F', 'S'], ['Sun', 'Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat'],
            ['Sunday', 'Monday', 'Tuesday', 'Wednesday', 'Thursday', 'Friday', 'Saturday'],
            ['Su', 'Mo', 'Tu', 'We', 'Th', 'Fr', 'Sa']
        ],
        u,
        [
            ['J', 'F', 'M', 'A', 'M', 'J', 'J', 'A', 'S', 'O', 'N', 'D'],
            ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec'],
            [
                'January', 'February', 'March', 'April', 'May', 'June', 'July', 'August', 'September',
                'October', 'November', 'December'
            ]
        ],
        u,
        [['B', 'A'], ['BC', 'AD'], ['Before Christ', 'Anno Domini']],
        0,
        [6, 0],
        ['M/d/yy', 'MMM d, y', 'MMMM d, y', 'EEEE, MMMM d, y'],
        ['h:mm a', 'h:mm:ss a', 'h:mm:ss a z', 'h:mm:ss a zzzz'],
        ['{1}, {0}', u, '{1} \'at\' {0}', u],
        ['.', ',', ';', '%', '+', '-', 'E', '×', '‰', '∞', 'NaN', ':'],
        ['#,##0.###', '#,##0%', '¤#,##0.00', '#E0'],
        'USD',
        '$',
        'US Dollar',
        {},
        'ltr',
        plural
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/i18n/locale_data_api.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * This const is used to store the locale data registered with `registerLocaleData`
     * @type {?}
     */
    let LOCALE_DATA = {};
    /**
     * Finds the locale data for a given locale.
     *
     * @see [Internationalization (i18n) Guide](https://angular.io/guide/i18n)
     * @param {?} locale The locale code.
     * @return {?} The locale data.
     */
    function findLocaleData(locale) {
        /** @type {?} */
        const normalizedLocale = normalizeLocale(locale);
        /** @type {?} */
        let match = getLocaleData(normalizedLocale);
        if (match) {
            return match;
        }
        // let's try to find a parent locale
        /** @type {?} */
        const parentLocale = normalizedLocale.split('-')[0];
        match = getLocaleData(parentLocale);
        if (match) {
            return match;
        }
        if (parentLocale === 'en') {
            return localeEn;
        }
        throw new Error(`Missing locale data for the locale "${locale}".`);
    }
    /**
     * Retrieves the plural function used by ICU expressions to determine the plural case to use
     * for a given locale.
     * @see `NgPlural` / [Internationalization (i18n) Guide](https://angular.io/guide/i18n)
     * @param {?} locale A locale code for the locale format rules to use.
     * @return {?} The plural function for the locale.
     */
    function getLocalePluralCase(locale) {
        /** @type {?} */
        const data = findLocaleData(locale);
        return data[LocaleDataIndex.PluralCase];
    }
    /**
     * Helper function to get the given `normalizedLocale` from `LOCALE_DATA`
     * or from the global `ng.common.locale`.
     * @param {?} normalizedLocale
     * @return {?}
     */
    function getLocaleData(normalizedLocale) {
        if (!(normalizedLocale in LOCALE_DATA)) {
            LOCALE_DATA[normalizedLocale] = _global.ng && _global.ng.common && _global.ng.common.locales &&
                _global.ng.common.locales[normalizedLocale];
        }
        return LOCALE_DATA[normalizedLocale];
    }
    /** @enum {number} */
    const LocaleDataIndex = {
        LocaleId: 0,
        DayPeriodsFormat: 1,
        DayPeriodsStandalone: 2,
        DaysFormat: 3,
        DaysStandalone: 4,
        MonthsFormat: 5,
        MonthsStandalone: 6,
        Eras: 7,
        FirstDayOfWeek: 8,
        WeekendRange: 9,
        DateFormat: 10,
        TimeFormat: 11,
        DateTimeFormat: 12,
        NumberSymbols: 13,
        NumberFormats: 14,
        CurrencyCode: 15,
        CurrencySymbol: 16,
        CurrencyName: 17,
        Currencies: 18,
        Directionality: 19,
        PluralCase: 20,
        ExtraData: 21,
    };
    LocaleDataIndex[LocaleDataIndex.LocaleId] = 'LocaleId';
    LocaleDataIndex[LocaleDataIndex.DayPeriodsFormat] = 'DayPeriodsFormat';
    LocaleDataIndex[LocaleDataIndex.DayPeriodsStandalone] = 'DayPeriodsStandalone';
    LocaleDataIndex[LocaleDataIndex.DaysFormat] = 'DaysFormat';
    LocaleDataIndex[LocaleDataIndex.DaysStandalone] = 'DaysStandalone';
    LocaleDataIndex[LocaleDataIndex.MonthsFormat] = 'MonthsFormat';
    LocaleDataIndex[LocaleDataIndex.MonthsStandalone] = 'MonthsStandalone';
    LocaleDataIndex[LocaleDataIndex.Eras] = 'Eras';
    LocaleDataIndex[LocaleDataIndex.FirstDayOfWeek] = 'FirstDayOfWeek';
    LocaleDataIndex[LocaleDataIndex.WeekendRange] = 'WeekendRange';
    LocaleDataIndex[LocaleDataIndex.DateFormat] = 'DateFormat';
    LocaleDataIndex[LocaleDataIndex.TimeFormat] = 'TimeFormat';
    LocaleDataIndex[LocaleDataIndex.DateTimeFormat] = 'DateTimeFormat';
    LocaleDataIndex[LocaleDataIndex.NumberSymbols] = 'NumberSymbols';
    LocaleDataIndex[LocaleDataIndex.NumberFormats] = 'NumberFormats';
    LocaleDataIndex[LocaleDataIndex.CurrencyCode] = 'CurrencyCode';
    LocaleDataIndex[LocaleDataIndex.CurrencySymbol] = 'CurrencySymbol';
    LocaleDataIndex[LocaleDataIndex.CurrencyName] = 'CurrencyName';
    LocaleDataIndex[LocaleDataIndex.Currencies] = 'Currencies';
    LocaleDataIndex[LocaleDataIndex.Directionality] = 'Directionality';
    LocaleDataIndex[LocaleDataIndex.PluralCase] = 'PluralCase';
    LocaleDataIndex[LocaleDataIndex.ExtraData] = 'ExtraData';
    /**
     * Returns the canonical form of a locale name - lowercase with `_` replaced with `-`.
     * @param {?} locale
     * @return {?}
     */
    function normalizeLocale(locale) {
        return locale.toLowerCase().replace(/_/g, '-');
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/i18n/localization.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const pluralMapping = ['zero', 'one', 'two', 'few', 'many'];
    /**
     * Returns the plural case based on the locale
     * @param {?} value
     * @param {?} locale
     * @return {?}
     */
    function getPluralCase(value, locale) {
        /** @type {?} */
        const plural = getLocalePluralCase(locale)(parseInt(value, 10));
        /** @type {?} */
        const result = pluralMapping[plural];
        return (result !== undefined) ? result : 'other';
    }
    /**
     * The locale id that the application is using by default (for translations and ICU expressions).
     * @type {?}
     */
    const DEFAULT_LOCALE_ID = 'en-US';
    /**
     * USD currency code that the application uses by default for CurrencyPipe when no
     * DEFAULT_CURRENCY_CODE is provided.
     * @type {?}
     */
    const USD_CURRENCY_CODE = 'USD';

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/i18n.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const MARKER = `�`;
    /** @type {?} */
    const ICU_BLOCK_REGEXP = /^\s*(�\d+:?\d*�)\s*,\s*(select|plural)\s*,/;
    /** @type {?} */
    const SUBTEMPLATE_REGEXP = /�\/?\*(\d+:\d+)�/gi;
    /** @type {?} */
    const PH_REGEXP = /�(\/?[#*!]\d+):?\d*�/gi;
    /** @type {?} */
    const BINDING_REGEXP = /�(\d+):?\d*�/gi;
    /** @type {?} */
    const ICU_REGEXP = /({\s*�\d+:?\d*�\s*,\s*\S{6}\s*,[\s\S]*})/gi;
    // i18nPostprocess consts
    /** @type {?} */
    const ROOT_TEMPLATE_ID = 0;
    /** @type {?} */
    const PP_MULTI_VALUE_PLACEHOLDERS_REGEXP = /\[(�.+?�?)\]/;
    /** @type {?} */
    const PP_PLACEHOLDERS_REGEXP = /\[(�.+?�?)\]|(�\/?\*\d+:\d+�)/g;
    /** @type {?} */
    const PP_ICU_VARS_REGEXP = /({\s*)(VAR_(PLURAL|SELECT)(_\d+)?)(\s*,)/g;
    /** @type {?} */
    const PP_ICU_PLACEHOLDERS_REGEXP = /{([A-Z0-9_]+)}/g;
    /** @type {?} */
    const PP_ICUS_REGEXP = /�I18N_EXP_(ICU(_\d+)?)�/g;
    /** @type {?} */
    const PP_CLOSE_TEMPLATE_REGEXP = /\/\*/;
    /** @type {?} */
    const PP_TEMPLATE_ID_REGEXP = /\d+\:(\d+)/;
    /**
     * Breaks pattern into strings and top level {...} blocks.
     * Can be used to break a message into text and ICU expressions, or to break an ICU expression into
     * keys and cases.
     * Original code from closure library, modified for Angular.
     *
     * @param {?} pattern (sub)Pattern to be broken.
     *
     * @return {?}
     */
    function extractParts(pattern) {
        if (!pattern) {
            return [];
        }
        /** @type {?} */
        let prevPos = 0;
        /** @type {?} */
        const braceStack = [];
        /** @type {?} */
        const results = [];
        /** @type {?} */
        const braces = /[{}]/g;
        // lastIndex doesn't get set to 0 so we have to.
        braces.lastIndex = 0;
        /** @type {?} */
        let match;
        while (match = braces.exec(pattern)) {
            /** @type {?} */
            const pos = match.index;
            if (match[0] == '}') {
                braceStack.pop();
                if (braceStack.length == 0) {
                    // End of the block.
                    /** @type {?} */
                    const block = pattern.substring(prevPos, pos);
                    if (ICU_BLOCK_REGEXP.test(block)) {
                        results.push(parseICUBlock(block));
                    }
                    else {
                        results.push(block);
                    }
                    prevPos = pos + 1;
                }
            }
            else {
                if (braceStack.length == 0) {
                    /** @type {?} */
                    const substring = pattern.substring(prevPos, pos);
                    results.push(substring);
                    prevPos = pos + 1;
                }
                braceStack.push('{');
            }
        }
        /** @type {?} */
        const substring = pattern.substring(prevPos);
        results.push(substring);
        return results;
    }
    /**
     * Parses text containing an ICU expression and produces a JSON object for it.
     * Original code from closure library, modified for Angular.
     *
     * @param {?} pattern Text containing an ICU expression that needs to be parsed.
     *
     * @return {?}
     */
    function parseICUBlock(pattern) {
        /** @type {?} */
        const cases = [];
        /** @type {?} */
        const values = [];
        /** @type {?} */
        let icuType = 1 /* plural */;
        /** @type {?} */
        let mainBinding = 0;
        pattern = pattern.replace(ICU_BLOCK_REGEXP, (/**
         * @param {?} str
         * @param {?} binding
         * @param {?} type
         * @return {?}
         */
        function (str, binding, type) {
            if (type === 'select') {
                icuType = 0 /* select */;
            }
            else {
                icuType = 1 /* plural */;
            }
            mainBinding = parseInt(binding.substr(1), 10);
            return '';
        }));
        /** @type {?} */
        const parts = (/** @type {?} */ (extractParts(pattern)));
        // Looking for (key block)+ sequence. One of the keys has to be "other".
        for (let pos = 0; pos < parts.length;) {
            /** @type {?} */
            let key = parts[pos++].trim();
            if (icuType === 1 /* plural */) {
                // Key can be "=x", we just want "x"
                key = key.replace(/\s*(?:=)?(\w+)\s*/, '$1');
            }
            if (key.length) {
                cases.push(key);
            }
            /** @type {?} */
            const blocks = (/** @type {?} */ (extractParts(parts[pos++])));
            if (cases.length > values.length) {
                values.push(blocks);
            }
        }
        // TODO(ocombe): support ICU expressions in attributes, see #21615
        return { type: icuType, mainBinding: mainBinding, cases, values };
    }
    /**
     * Removes everything inside the sub-templates of a message.
     * @param {?} message
     * @return {?}
     */
    function removeInnerTemplateTranslation(message) {
        /** @type {?} */
        let match;
        /** @type {?} */
        let res = '';
        /** @type {?} */
        let index = 0;
        /** @type {?} */
        let inTemplate = false;
        /** @type {?} */
        let tagMatched;
        while ((match = SUBTEMPLATE_REGEXP.exec(message)) !== null) {
            if (!inTemplate) {
                res += message.substring(index, match.index + match[0].length);
                tagMatched = match[1];
                inTemplate = true;
            }
            else {
                if (match[0] === `${MARKER}/*${tagMatched}${MARKER}`) {
                    index = match.index;
                    inTemplate = false;
                }
            }
        }
        ngDevMode &&
            assertEqual(inTemplate, false, `Tag mismatch: unable to find the end of the sub-template in the translation "${message}"`);
        res += message.substr(index);
        return res;
    }
    /**
     * Extracts a part of a message and removes the rest.
     *
     * This method is used for extracting a part of the message associated with a template. A translated
     * message can span multiple templates.
     *
     * Example:
     * ```
     * <div i18n>Translate <span *ngIf>me</span>!</div>
     * ```
     *
     * @param {?} message The message to crop
     * @param {?=} subTemplateIndex Index of the sub-template to extract. If undefined it returns the
     * external template and removes all sub-templates.
     * @return {?}
     */
    function getTranslationForTemplate(message, subTemplateIndex) {
        if (isRootTemplateMessage(subTemplateIndex)) {
            // We want the root template message, ignore all sub-templates
            return removeInnerTemplateTranslation(message);
        }
        else {
            // We want a specific sub-template
            /** @type {?} */
            const start = message.indexOf(`:${subTemplateIndex}${MARKER}`) + 2 + subTemplateIndex.toString().length;
            /** @type {?} */
            const end = message.search(new RegExp(`${MARKER}\\/\\*\\d+:${subTemplateIndex}${MARKER}`));
            return removeInnerTemplateTranslation(message.substring(start, end));
        }
    }
    /**
     * Generate the OpCodes to update the bindings of a string.
     *
     * @param {?} str The string containing the bindings.
     * @param {?} destinationNode Index of the destination node which will receive the binding.
     * @param {?=} attrName Name of the attribute, if the string belongs to an attribute.
     * @param {?=} sanitizeFn Sanitization function used to sanitize the string after update, if necessary.
     * @return {?}
     */
    function generateBindingUpdateOpCodes(str, destinationNode, attrName, sanitizeFn = null) {
        /** @type {?} */
        const updateOpCodes = [null, null];
        // Alloc space for mask and size
        /** @type {?} */
        const textParts = str.split(BINDING_REGEXP);
        /** @type {?} */
        let mask = 0;
        for (let j = 0; j < textParts.length; j++) {
            /** @type {?} */
            const textValue = textParts[j];
            if (j & 1) {
                // Odd indexes are bindings
                /** @type {?} */
                const bindingIndex = parseInt(textValue, 10);
                updateOpCodes.push(-1 - bindingIndex);
                mask = mask | toMaskBit(bindingIndex);
            }
            else if (textValue !== '') {
                // Even indexes are text
                updateOpCodes.push(textValue);
            }
        }
        updateOpCodes.push(destinationNode << 2 /* SHIFT_REF */ |
            (attrName ? 1 /* Attr */ : 0 /* Text */));
        if (attrName) {
            updateOpCodes.push(attrName, sanitizeFn);
        }
        updateOpCodes[0] = mask;
        updateOpCodes[1] = updateOpCodes.length - 2;
        return updateOpCodes;
    }
    /**
     * @param {?} icuExpression
     * @param {?=} mask
     * @return {?}
     */
    function getBindingMask(icuExpression, mask = 0) {
        mask = mask | toMaskBit(icuExpression.mainBinding);
        /** @type {?} */
        let match;
        for (let i = 0; i < icuExpression.values.length; i++) {
            /** @type {?} */
            const valueArr = icuExpression.values[i];
            for (let j = 0; j < valueArr.length; j++) {
                /** @type {?} */
                const value = valueArr[j];
                if (typeof value === 'string') {
                    while (match = BINDING_REGEXP.exec(value)) {
                        mask = mask | toMaskBit(parseInt(match[1], 10));
                    }
                }
                else {
                    mask = getBindingMask((/** @type {?} */ (value)), mask);
                }
            }
        }
        return mask;
    }
    /** @type {?} */
    const i18nIndexStack = [];
    /** @type {?} */
    let i18nIndexStackPointer = -1;
    /**
     * Convert binding index to mask bit.
     *
     * Each index represents a single bit on the bit-mask. Because bit-mask only has 32 bits, we make
     * the 32nd bit share all masks for all bindings higher than 32. Since it is extremely rare to have
     * more than 32 bindings this will be hit very rarely. The downside of hitting this corner case is
     * that we will execute binding code more often than necessary. (penalty of performance)
     * @param {?} bindingIndex
     * @return {?}
     */
    function toMaskBit(bindingIndex) {
        return 1 << Math.min(bindingIndex, 31);
    }
    /** @type {?} */
    const parentIndexStack = [];
    /**
     * Marks a block of text as translatable.
     *
     * The instructions `i18nStart` and `i18nEnd` mark the translation block in the template.
     * The translation `message` is the value which is locale specific. The translation string may
     * contain placeholders which associate inner elements and sub-templates within the translation.
     *
     * The translation `message` placeholders are:
     * - `�{index}(:{block})�`: *Binding Placeholder*: Marks a location where an expression will be
     *   interpolated into. The placeholder `index` points to the expression binding index. An optional
     *   `block` that matches the sub-template in which it was declared.
     * - `�#{index}(:{block})�`/`�/#{index}(:{block})�`: *Element Placeholder*:  Marks the beginning
     *   and end of DOM element that were embedded in the original translation block. The placeholder
     *   `index` points to the element index in the template instructions set. An optional `block` that
     *   matches the sub-template in which it was declared.
     * - `�!{index}(:{block})�`/`�/!{index}(:{block})�`: *Projection Placeholder*:  Marks the
     *   beginning and end of <ng-content> that was embedded in the original translation block.
     *   The placeholder `index` points to the element index in the template instructions set.
     *   An optional `block` that matches the sub-template in which it was declared.
     * - `�*{index}:{block}�`/`�/*{index}:{block}�`: *Sub-template Placeholder*: Sub-templates must be
     *   split up and translated separately in each angular template function. The `index` points to the
     *   `template` instruction index. A `block` that matches the sub-template in which it was declared.
     *
     * \@codeGenApi
     * @param {?} index A unique index of the translation in the static block.
     * @param {?} message The translation message.
     * @param {?=} subTemplateIndex Optional sub-template index in the `message`.
     *
     * @return {?}
     */
    function ɵɵi18nStart(index, message, subTemplateIndex) {
        /** @type {?} */
        const tView = getTView();
        ngDevMode && assertDefined(tView, `tView should be defined`);
        i18nIndexStack[++i18nIndexStackPointer] = index;
        // We need to delay projections until `i18nEnd`
        setDelayProjection(true);
        if (tView.firstCreatePass && tView.data[index + HEADER_OFFSET] === null) {
            i18nStartFirstPass(getLView(), tView, index, message, subTemplateIndex);
        }
    }
    // Count for the number of vars that will be allocated for each i18n block.
    // It is global because this is used in multiple functions that include loops and recursive calls.
    // This is reset to 0 when `i18nStartFirstPass` is called.
    /** @type {?} */
    let i18nVarsCount;
    /**
     * @param {?} startIndex
     * @return {?}
     */
    function allocNodeIndex(startIndex) {
        return startIndex + i18nVarsCount++;
    }
    /**
     * See `i18nStart` above.
     * @param {?} lView
     * @param {?} tView
     * @param {?} index
     * @param {?} message
     * @param {?=} subTemplateIndex
     * @return {?}
     */
    function i18nStartFirstPass(lView, tView, index, message, subTemplateIndex) {
        /** @type {?} */
        const startIndex = tView.blueprint.length - HEADER_OFFSET;
        i18nVarsCount = 0;
        /** @type {?} */
        const previousOrParentTNode = getPreviousOrParentTNode();
        /** @type {?} */
        const parentTNode = getIsParent() ? previousOrParentTNode : previousOrParentTNode && previousOrParentTNode.parent;
        /** @type {?} */
        let parentIndex = parentTNode && parentTNode !== lView[T_HOST] ? parentTNode.index - HEADER_OFFSET : index;
        /** @type {?} */
        let parentIndexPointer = 0;
        parentIndexStack[parentIndexPointer] = parentIndex;
        /** @type {?} */
        const createOpCodes = [];
        // If the previous node wasn't the direct parent then we have a translation without top level
        // element and we need to keep a reference of the previous element if there is one. We should also
        // keep track whether an element was a parent node or not, so that the logic that consumes
        // the generated `I18nMutateOpCode`s can leverage this information to properly set TNode state
        // (whether it's a parent or sibling).
        if (index > 0 && previousOrParentTNode !== parentTNode) {
            /** @type {?} */
            let previousTNodeIndex = previousOrParentTNode.index - HEADER_OFFSET;
            // If current TNode is a sibling node, encode it using a negative index. This information is
            // required when the `Select` action is processed (see the `readCreateOpCodes` function).
            if (!getIsParent()) {
                previousTNodeIndex = ~previousTNodeIndex;
            }
            // Create an OpCode to select the previous TNode
            createOpCodes.push(previousTNodeIndex << 3 /* SHIFT_REF */ | 0 /* Select */);
        }
        /** @type {?} */
        const updateOpCodes = [];
        /** @type {?} */
        const icuExpressions = [];
        if (message === '' && isRootTemplateMessage(subTemplateIndex)) {
            // If top level translation is an empty string, do not invoke additional processing
            // and just create op codes for empty text node instead.
            createOpCodes.push(message, allocNodeIndex(startIndex), parentIndex << 17 /* SHIFT_PARENT */ | 1 /* AppendChild */);
        }
        else {
            /** @type {?} */
            const templateTranslation = getTranslationForTemplate(message, subTemplateIndex);
            /** @type {?} */
            const msgParts = replaceNgsp(templateTranslation).split(PH_REGEXP);
            for (let i = 0; i < msgParts.length; i++) {
                /** @type {?} */
                let value = msgParts[i];
                if (i & 1) {
                    // Odd indexes are placeholders (elements and sub-templates)
                    if (value.charAt(0) === '/') {
                        // It is a closing tag
                        if (value.charAt(1) === "#" /* ELEMENT */) {
                            /** @type {?} */
                            const phIndex = parseInt(value.substr(2), 10);
                            parentIndex = parentIndexStack[--parentIndexPointer];
                            createOpCodes.push(phIndex << 3 /* SHIFT_REF */ | 5 /* ElementEnd */);
                        }
                    }
                    else {
                        /** @type {?} */
                        const phIndex = parseInt(value.substr(1), 10);
                        /** @type {?} */
                        const isElement = value.charAt(0) === "#" /* ELEMENT */;
                        // The value represents a placeholder that we move to the designated index.
                        // Note: positive indicies indicate that a TNode with a given index should also be marked
                        // as parent while executing `Select` instruction.
                        createOpCodes.push((isElement ? phIndex : ~phIndex) << 3 /* SHIFT_REF */ |
                            0 /* Select */, parentIndex << 17 /* SHIFT_PARENT */ | 1 /* AppendChild */);
                        if (isElement) {
                            parentIndexStack[++parentIndexPointer] = parentIndex = phIndex;
                        }
                    }
                }
                else {
                    // Even indexes are text (including bindings & ICU expressions)
                    /** @type {?} */
                    const parts = extractParts(value);
                    for (let j = 0; j < parts.length; j++) {
                        if (j & 1) {
                            // Odd indexes are ICU expressions
                            /** @type {?} */
                            const icuExpression = (/** @type {?} */ (parts[j]));
                            // Verify that ICU expression has the right shape. Translations might contain invalid
                            // constructions (while original messages were correct), so ICU parsing at runtime may
                            // not succeed (thus `icuExpression` remains a string).
                            if (typeof icuExpression !== 'object') {
                                throw new Error(`Unable to parse ICU expression in "${templateTranslation}" message.`);
                            }
                            // Create the comment node that will anchor the ICU expression
                            /** @type {?} */
                            const icuNodeIndex = allocNodeIndex(startIndex);
                            createOpCodes.push(COMMENT_MARKER, ngDevMode ? `ICU ${icuNodeIndex}` : '', icuNodeIndex, parentIndex << 17 /* SHIFT_PARENT */ | 1 /* AppendChild */);
                            // Update codes for the ICU expression
                            /** @type {?} */
                            const mask = getBindingMask(icuExpression);
                            icuStart(icuExpressions, icuExpression, icuNodeIndex, icuNodeIndex);
                            // Since this is recursive, the last TIcu that was pushed is the one we want
                            /** @type {?} */
                            const tIcuIndex = icuExpressions.length - 1;
                            updateOpCodes.push(toMaskBit(icuExpression.mainBinding), // mask of the main binding
                            3, // skip 3 opCodes if not changed
                            -1 - icuExpression.mainBinding, icuNodeIndex << 2 /* SHIFT_REF */ | 2 /* IcuSwitch */, tIcuIndex, mask, // mask of all the bindings of this ICU expression
                            2, // skip 2 opCodes if not changed
                            icuNodeIndex << 2 /* SHIFT_REF */ | 3 /* IcuUpdate */, tIcuIndex);
                        }
                        else if (parts[j] !== '') {
                            /** @type {?} */
                            const text = (/** @type {?} */ (parts[j]));
                            // Even indexes are text (including bindings)
                            /** @type {?} */
                            const hasBinding = text.match(BINDING_REGEXP);
                            // Create text nodes
                            /** @type {?} */
                            const textNodeIndex = allocNodeIndex(startIndex);
                            createOpCodes.push(
                            // If there is a binding, the value will be set during update
                            hasBinding ? '' : text, textNodeIndex, parentIndex << 17 /* SHIFT_PARENT */ | 1 /* AppendChild */);
                            if (hasBinding) {
                                addAllToArray(generateBindingUpdateOpCodes(text, textNodeIndex), updateOpCodes);
                            }
                        }
                    }
                }
            }
        }
        if (i18nVarsCount > 0) {
            allocExpando(tView, lView, i18nVarsCount);
        }
        ngDevMode &&
            attachI18nOpCodesDebug(createOpCodes, updateOpCodes, icuExpressions.length ? icuExpressions : null, lView);
        // NOTE: local var needed to properly assert the type of `TI18n`.
        /** @type {?} */
        const tI18n = {
            vars: i18nVarsCount,
            create: createOpCodes,
            update: updateOpCodes,
            icus: icuExpressions.length ? icuExpressions : null,
        };
        tView.data[index + HEADER_OFFSET] = tI18n;
    }
    /**
     * @param {?} tView
     * @param {?} tNode
     * @param {?} parentTNode
     * @param {?} previousTNode
     * @param {?} lView
     * @return {?}
     */
    function appendI18nNode(tView, tNode, parentTNode, previousTNode, lView) {
        ngDevMode && ngDevMode.rendererMoveNode++;
        /** @type {?} */
        const nextNode = tNode.next;
        if (!previousTNode) {
            previousTNode = parentTNode;
        }
        // Re-organize node tree to put this node in the correct position.
        if (previousTNode === parentTNode && tNode !== parentTNode.child) {
            tNode.next = parentTNode.child;
            parentTNode.child = tNode;
        }
        else if (previousTNode !== parentTNode && tNode !== previousTNode.next) {
            tNode.next = previousTNode.next;
            previousTNode.next = tNode;
        }
        else {
            tNode.next = null;
        }
        if (parentTNode !== lView[T_HOST]) {
            tNode.parent = (/** @type {?} */ (parentTNode));
        }
        // If tNode was moved around, we might need to fix a broken link.
        /** @type {?} */
        let cursor = tNode.next;
        while (cursor) {
            if (cursor.next === tNode) {
                cursor.next = nextNode;
            }
            cursor = cursor.next;
        }
        // If the placeholder to append is a projection, we need to move the projected nodes instead
        if (tNode.type === 1 /* Projection */) {
            applyProjection(tView, lView, (/** @type {?} */ (tNode)));
            return tNode;
        }
        appendChild(tView, lView, getNativeByTNode(tNode, lView), tNode);
        /** @type {?} */
        const slotValue = lView[tNode.index];
        if (tNode.type !== 0 /* Container */ && isLContainer(slotValue)) {
            // Nodes that inject ViewContainerRef also have a comment node that should be moved
            appendChild(tView, lView, slotValue[NATIVE], tNode);
        }
        return tNode;
    }
    /**
     * @param {?} subTemplateIndex
     * @return {?}
     */
    function isRootTemplateMessage(subTemplateIndex) {
        return subTemplateIndex === undefined;
    }
    /**
     * Handles message string post-processing for internationalization.
     *
     * Handles message string post-processing by transforming it from intermediate
     * format (that might contain some markers that we need to replace) to the final
     * form, consumable by i18nStart instruction. Post processing steps include:
     *
     * 1. Resolve all multi-value cases (like [�*1:1��#2:1�|�#4:1�|�5�])
     * 2. Replace all ICU vars (like "VAR_PLURAL")
     * 3. Replace all placeholders used inside ICUs in a form of {PLACEHOLDER}
     * 4. Replace all ICU references with corresponding values (like �ICU_EXP_ICU_1�)
     *    in case multiple ICUs have the same placeholder name
     *
     * \@codeGenApi
     * @param {?} message Raw translation string for post processing
     * @param {?=} replacements Set of replacements that should be applied
     *
     * @return {?} Transformed string that can be consumed by i18nStart instruction
     *
     */
    function ɵɵi18nPostprocess(message, replacements = {}) {
        /**
         * Step 1: resolve all multi-value placeholders like [�#5�|�*1:1��#2:1�|�#4:1�]
         *
         * Note: due to the way we process nested templates (BFS), multi-value placeholders are typically
         * grouped by templates, for example: [�#5�|�#6�|�#1:1�|�#3:2�] where �#5� and �#6� belong to root
         * template, �#1:1� belong to nested template with index 1 and �#1:2� - nested template with index
         * 3. However in real templates the order might be different: i.e. �#1:1� and/or �#3:2� may go in
         * front of �#6�. The post processing step restores the right order by keeping track of the
         * template id stack and looks for placeholders that belong to the currently active template.
         * @type {?}
         */
        let result = message;
        if (PP_MULTI_VALUE_PLACEHOLDERS_REGEXP.test(message)) {
            /** @type {?} */
            const matches = {};
            /** @type {?} */
            const templateIdsStack = [ROOT_TEMPLATE_ID];
            result = result.replace(PP_PLACEHOLDERS_REGEXP, (/**
             * @param {?} m
             * @param {?} phs
             * @param {?} tmpl
             * @return {?}
             */
            (m, phs, tmpl) => {
                /** @type {?} */
                const content = phs || tmpl;
                /** @type {?} */
                const placeholders = matches[content] || [];
                if (!placeholders.length) {
                    content.split('|').forEach((/**
                     * @param {?} placeholder
                     * @return {?}
                     */
                    (placeholder) => {
                        /** @type {?} */
                        const match = placeholder.match(PP_TEMPLATE_ID_REGEXP);
                        /** @type {?} */
                        const templateId = match ? parseInt(match[1], 10) : ROOT_TEMPLATE_ID;
                        /** @type {?} */
                        const isCloseTemplateTag = PP_CLOSE_TEMPLATE_REGEXP.test(placeholder);
                        placeholders.push([templateId, isCloseTemplateTag, placeholder]);
                    }));
                    matches[content] = placeholders;
                }
                if (!placeholders.length) {
                    throw new Error(`i18n postprocess: unmatched placeholder - ${content}`);
                }
                /** @type {?} */
                const currentTemplateId = templateIdsStack[templateIdsStack.length - 1];
                /** @type {?} */
                let idx = 0;
                // find placeholder index that matches current template id
                for (let i = 0; i < placeholders.length; i++) {
                    if (placeholders[i][0] === currentTemplateId) {
                        idx = i;
                        break;
                    }
                }
                // update template id stack based on the current tag extracted
                const [templateId, isCloseTemplateTag, placeholder] = placeholders[idx];
                if (isCloseTemplateTag) {
                    templateIdsStack.pop();
                }
                else if (currentTemplateId !== templateId) {
                    templateIdsStack.push(templateId);
                }
                // remove processed tag from the list
                placeholders.splice(idx, 1);
                return placeholder;
            }));
        }
        // return current result if no replacements specified
        if (!Object.keys(replacements).length) {
            return result;
        }
        /**
         * Step 2: replace all ICU vars (like "VAR_PLURAL")
         */
        result = result.replace(PP_ICU_VARS_REGEXP, (/**
         * @param {?} match
         * @param {?} start
         * @param {?} key
         * @param {?} _type
         * @param {?} _idx
         * @param {?} end
         * @return {?}
         */
        (match, start, key, _type, _idx, end) => {
            return replacements.hasOwnProperty(key) ? `${start}${replacements[key]}${end}` : match;
        }));
        /**
         * Step 3: replace all placeholders used inside ICUs in a form of {PLACEHOLDER}
         */
        result = result.replace(PP_ICU_PLACEHOLDERS_REGEXP, (/**
         * @param {?} match
         * @param {?} key
         * @return {?}
         */
        (match, key) => {
            return replacements.hasOwnProperty(key) ? (/** @type {?} */ (replacements[key])) : match;
        }));
        /**
         * Step 4: replace all ICU references with corresponding values (like �ICU_EXP_ICU_1�) in case
         * multiple ICUs have the same placeholder name
         */
        result = result.replace(PP_ICUS_REGEXP, (/**
         * @param {?} match
         * @param {?} key
         * @return {?}
         */
        (match, key) => {
            if (replacements.hasOwnProperty(key)) {
                /** @type {?} */
                const list = (/** @type {?} */ (replacements[key]));
                if (!list.length) {
                    throw new Error(`i18n postprocess: unmatched ICU - ${match} with key: ${key}`);
                }
                return (/** @type {?} */ (list.shift()));
            }
            return match;
        }));
        return result;
    }
    /**
     * Translates a translation block marked by `i18nStart` and `i18nEnd`. It inserts the text/ICU nodes
     * into the render tree, moves the placeholder nodes and removes the deleted nodes.
     *
     * \@codeGenApi
     * @return {?}
     */
    function ɵɵi18nEnd() {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const tView = getTView();
        ngDevMode && assertDefined(tView, `tView should be defined`);
        i18nEndFirstPass(tView, lView);
        // Stop delaying projections
        setDelayProjection(false);
    }
    /**
     * See `i18nEnd` above.
     * @param {?} tView
     * @param {?} lView
     * @return {?}
     */
    function i18nEndFirstPass(tView, lView) {
        ngDevMode &&
            assertEqual(getBindingIndex(), tView.bindingStartIndex, 'i18nEnd should be called before any binding');
        /** @type {?} */
        const rootIndex = i18nIndexStack[i18nIndexStackPointer--];
        /** @type {?} */
        const tI18n = (/** @type {?} */ (tView.data[rootIndex + HEADER_OFFSET]));
        ngDevMode && assertDefined(tI18n, `You should call i18nStart before i18nEnd`);
        // Find the last node that was added before `i18nEnd`
        /** @type {?} */
        const lastCreatedNode = getPreviousOrParentTNode();
        // Read the instructions to insert/move/remove DOM elements
        /** @type {?} */
        const visitedNodes = readCreateOpCodes(rootIndex, tI18n.create, tView, lView);
        // Remove deleted nodes
        /** @type {?} */
        let index = rootIndex + 1;
        while (index <= lastCreatedNode.index - HEADER_OFFSET) {
            if (visitedNodes.indexOf(index) === -1) {
                removeNode(tView, lView, index, /* markAsDetached */ true);
            }
            // Check if an element has any local refs and skip them
            /** @type {?} */
            const tNode = getTNode(tView, index);
            if (tNode &&
                (tNode.type === 0 /* Container */ || tNode.type === 3 /* Element */ ||
                    tNode.type === 4 /* ElementContainer */) &&
                tNode.localNames !== null) {
                // Divide by 2 to get the number of local refs,
                // since they are stored as an array that also includes directive indexes,
                // i.e. ["localRef", directiveIndex, ...]
                index += tNode.localNames.length >> 1;
            }
            index++;
        }
    }
    /**
     * Creates and stores the dynamic TNode, and unhooks it from the tree for now.
     * @param {?} tView
     * @param {?} lView
     * @param {?} index
     * @param {?} type
     * @param {?} native
     * @param {?} name
     * @return {?}
     */
    function createDynamicNodeAtIndex(tView, lView, index, type, native, name) {
        /** @type {?} */
        const previousOrParentTNode = getPreviousOrParentTNode();
        ngDevMode && assertDataInRange(lView, index + HEADER_OFFSET);
        lView[index + HEADER_OFFSET] = native;
        /** @type {?} */
        const tNode = getOrCreateTNode(tView, lView[T_HOST], index, (/** @type {?} */ (type)), name, null);
        // We are creating a dynamic node, the previous tNode might not be pointing at this node.
        // We will link ourselves into the tree later with `appendI18nNode`.
        if (previousOrParentTNode && previousOrParentTNode.next === tNode) {
            previousOrParentTNode.next = null;
        }
        return tNode;
    }
    /**
     * @param {?} index
     * @param {?} createOpCodes
     * @param {?} tView
     * @param {?} lView
     * @return {?}
     */
    function readCreateOpCodes(index, createOpCodes, tView, lView) {
        /** @type {?} */
        const renderer = lView[RENDERER];
        /** @type {?} */
        let currentTNode = null;
        /** @type {?} */
        let previousTNode = null;
        /** @type {?} */
        const visitedNodes = [];
        for (let i = 0; i < createOpCodes.length; i++) {
            /** @type {?} */
            const opCode = createOpCodes[i];
            if (typeof opCode == 'string') {
                /** @type {?} */
                const textRNode = createTextNode(opCode, renderer);
                /** @type {?} */
                const textNodeIndex = (/** @type {?} */ (createOpCodes[++i]));
                ngDevMode && ngDevMode.rendererCreateTextNode++;
                previousTNode = currentTNode;
                currentTNode =
                    createDynamicNodeAtIndex(tView, lView, textNodeIndex, 3 /* Element */, textRNode, null);
                visitedNodes.push(textNodeIndex);
                setIsNotParent();
            }
            else if (typeof opCode == 'number') {
                switch (opCode & 7 /* MASK_OPCODE */) {
                    case 1 /* AppendChild */:
                        /** @type {?} */
                        const destinationNodeIndex = opCode >>> 17 /* SHIFT_PARENT */;
                        /** @type {?} */
                        let destinationTNode;
                        if (destinationNodeIndex === index) {
                            // If the destination node is `i18nStart`, we don't have a
                            // top-level node and we should use the host node instead
                            destinationTNode = (/** @type {?} */ (lView[T_HOST]));
                        }
                        else {
                            destinationTNode = getTNode(tView, destinationNodeIndex);
                        }
                        ngDevMode &&
                            assertDefined((/** @type {?} */ (currentTNode)), `You need to create or select a node before you can insert it into the DOM`);
                        previousTNode =
                            appendI18nNode(tView, (/** @type {?} */ (currentTNode)), destinationTNode, previousTNode, lView);
                        break;
                    case 0 /* Select */:
                        // Negative indicies indicate that a given TNode is a sibling node, not a parent node
                        // (see `i18nStartFirstPass` for additional information).
                        /** @type {?} */
                        const isParent = opCode >= 0;
                        /** @type {?} */
                        const nodeIndex = (isParent ? opCode : ~opCode) >>> 3 /* SHIFT_REF */;
                        visitedNodes.push(nodeIndex);
                        previousTNode = currentTNode;
                        currentTNode = getTNode(tView, nodeIndex);
                        if (currentTNode) {
                            setPreviousOrParentTNode(currentTNode, isParent);
                        }
                        break;
                    case 5 /* ElementEnd */:
                        /** @type {?} */
                        const elementIndex = opCode >>> 3 /* SHIFT_REF */;
                        previousTNode = currentTNode = getTNode(tView, elementIndex);
                        setPreviousOrParentTNode(currentTNode, false);
                        break;
                    case 4 /* Attr */:
                        /** @type {?} */
                        const elementNodeIndex = opCode >>> 3 /* SHIFT_REF */;
                        /** @type {?} */
                        const attrName = (/** @type {?} */ (createOpCodes[++i]));
                        /** @type {?} */
                        const attrValue = (/** @type {?} */ (createOpCodes[++i]));
                        // This code is used for ICU expressions only, since we don't support
                        // directives/components in ICUs, we don't need to worry about inputs here
                        elementAttributeInternal(getTNode(tView, elementNodeIndex), lView, attrName, attrValue, null, null);
                        break;
                    default:
                        throw new Error(`Unable to determine the type of mutate operation for "${opCode}"`);
                }
            }
            else {
                switch (opCode) {
                    case COMMENT_MARKER:
                        /** @type {?} */
                        const commentValue = (/** @type {?} */ (createOpCodes[++i]));
                        /** @type {?} */
                        const commentNodeIndex = (/** @type {?} */ (createOpCodes[++i]));
                        ngDevMode &&
                            assertEqual(typeof commentValue, 'string', `Expected "${commentValue}" to be a comment node value`);
                        /** @type {?} */
                        const commentRNode = renderer.createComment(commentValue);
                        ngDevMode && ngDevMode.rendererCreateComment++;
                        previousTNode = currentTNode;
                        currentTNode = createDynamicNodeAtIndex(tView, lView, commentNodeIndex, 5 /* IcuContainer */, commentRNode, null);
                        visitedNodes.push(commentNodeIndex);
                        attachPatchData(commentRNode, lView);
                        ((/** @type {?} */ (currentTNode))).activeCaseIndex = null;
                        // We will add the case nodes later, during the update phase
                        setIsNotParent();
                        break;
                    case ELEMENT_MARKER:
                        /** @type {?} */
                        const tagNameValue = (/** @type {?} */ (createOpCodes[++i]));
                        /** @type {?} */
                        const elementNodeIndex = (/** @type {?} */ (createOpCodes[++i]));
                        ngDevMode &&
                            assertEqual(typeof tagNameValue, 'string', `Expected "${tagNameValue}" to be an element node tag name`);
                        /** @type {?} */
                        const elementRNode = renderer.createElement(tagNameValue);
                        ngDevMode && ngDevMode.rendererCreateElement++;
                        previousTNode = currentTNode;
                        currentTNode = createDynamicNodeAtIndex(tView, lView, elementNodeIndex, 3 /* Element */, elementRNode, tagNameValue);
                        visitedNodes.push(elementNodeIndex);
                        break;
                    default:
                        throw new Error(`Unable to determine the type of mutate operation for "${opCode}"`);
                }
            }
        }
        setIsNotParent();
        return visitedNodes;
    }
    /**
     * @param {?} updateOpCodes
     * @param {?} icus
     * @param {?} bindingsStartIndex
     * @param {?} changeMask
     * @param {?} tView
     * @param {?} lView
     * @param {?=} bypassCheckBit
     * @return {?}
     */
    function readUpdateOpCodes(updateOpCodes, icus, bindingsStartIndex, changeMask, tView, lView, bypassCheckBit = false) {
        /** @type {?} */
        let caseCreated = false;
        for (let i = 0; i < updateOpCodes.length; i++) {
            // bit code to check if we should apply the next update
            /** @type {?} */
            const checkBit = (/** @type {?} */ (updateOpCodes[i]));
            // Number of opCodes to skip until next set of update codes
            /** @type {?} */
            const skipCodes = (/** @type {?} */ (updateOpCodes[++i]));
            if (bypassCheckBit || (checkBit & changeMask)) {
                // The value has been updated since last checked
                /** @type {?} */
                let value = '';
                for (let j = i + 1; j <= (i + skipCodes); j++) {
                    /** @type {?} */
                    const opCode = updateOpCodes[j];
                    if (typeof opCode == 'string') {
                        value += opCode;
                    }
                    else if (typeof opCode == 'number') {
                        if (opCode < 0) {
                            // It's a binding index whose value is negative
                            value += renderStringify(lView[bindingsStartIndex - opCode]);
                        }
                        else {
                            /** @type {?} */
                            const nodeIndex = opCode >>> 2 /* SHIFT_REF */;
                            /** @type {?} */
                            let tIcuIndex;
                            /** @type {?} */
                            let tIcu;
                            /** @type {?} */
                            let icuTNode;
                            switch (opCode & 3 /* MASK_OPCODE */) {
                                case 1 /* Attr */:
                                    /** @type {?} */
                                    const propName = (/** @type {?} */ (updateOpCodes[++j]));
                                    /** @type {?} */
                                    const sanitizeFn = (/** @type {?} */ (updateOpCodes[++j]));
                                    elementPropertyInternal(tView, getTNode(tView, nodeIndex), lView, propName, value, lView[RENDERER], sanitizeFn, false);
                                    break;
                                case 0 /* Text */:
                                    textBindingInternal(lView, nodeIndex, value);
                                    break;
                                case 2 /* IcuSwitch */:
                                    tIcuIndex = (/** @type {?} */ (updateOpCodes[++j]));
                                    tIcu = (/** @type {?} */ (icus))[tIcuIndex];
                                    icuTNode = (/** @type {?} */ (getTNode(tView, nodeIndex)));
                                    // If there is an active case, delete the old nodes
                                    if (icuTNode.activeCaseIndex !== null) {
                                        /** @type {?} */
                                        const removeCodes = tIcu.remove[icuTNode.activeCaseIndex];
                                        for (let k = 0; k < removeCodes.length; k++) {
                                            /** @type {?} */
                                            const removeOpCode = (/** @type {?} */ (removeCodes[k]));
                                            switch (removeOpCode & 7 /* MASK_OPCODE */) {
                                                case 3 /* Remove */:
                                                    /** @type {?} */
                                                    const nodeIndex = removeOpCode >>> 3 /* SHIFT_REF */;
                                                    // Remove DOM element, but do *not* mark TNode as detached, since we are
                                                    // just switching ICU cases (while keeping the same TNode), so a DOM element
                                                    // representing a new ICU case will be re-created.
                                                    removeNode(tView, lView, nodeIndex, /* markAsDetached */ false);
                                                    break;
                                                case 6 /* RemoveNestedIcu */:
                                                    /** @type {?} */
                                                    const nestedIcuNodeIndex = (/** @type {?} */ (removeCodes[k + 1])) >>> 3 /* SHIFT_REF */;
                                                    /** @type {?} */
                                                    const nestedIcuTNode = (/** @type {?} */ (getTNode(tView, nestedIcuNodeIndex)));
                                                    /** @type {?} */
                                                    const activeIndex = nestedIcuTNode.activeCaseIndex;
                                                    if (activeIndex !== null) {
                                                        /** @type {?} */
                                                        const nestedIcuTIndex = removeOpCode >>> 3 /* SHIFT_REF */;
                                                        /** @type {?} */
                                                        const nestedTIcu = (/** @type {?} */ (icus))[nestedIcuTIndex];
                                                        addAllToArray(nestedTIcu.remove[activeIndex], removeCodes);
                                                    }
                                                    break;
                                            }
                                        }
                                    }
                                    // Update the active caseIndex
                                    /** @type {?} */
                                    const caseIndex = getCaseIndex(tIcu, value);
                                    icuTNode.activeCaseIndex = caseIndex !== -1 ? caseIndex : null;
                                    if (caseIndex > -1) {
                                        // Add the nodes for the new case
                                        readCreateOpCodes(-1, tIcu.create[caseIndex], tView, lView);
                                        caseCreated = true;
                                    }
                                    break;
                                case 3 /* IcuUpdate */:
                                    tIcuIndex = (/** @type {?} */ (updateOpCodes[++j]));
                                    tIcu = (/** @type {?} */ (icus))[tIcuIndex];
                                    icuTNode = (/** @type {?} */ (getTNode(tView, nodeIndex)));
                                    if (icuTNode.activeCaseIndex !== null) {
                                        readUpdateOpCodes(tIcu.update[icuTNode.activeCaseIndex], icus, bindingsStartIndex, changeMask, tView, lView, caseCreated);
                                    }
                                    break;
                            }
                        }
                    }
                }
            }
            i += skipCodes;
        }
    }
    /**
     * @param {?} tView
     * @param {?} lView
     * @param {?} index
     * @param {?} markAsDetached
     * @return {?}
     */
    function removeNode(tView, lView, index, markAsDetached) {
        /** @type {?} */
        const removedPhTNode = getTNode(tView, index);
        /** @type {?} */
        const removedPhRNode = getNativeByIndex(index, lView);
        if (removedPhRNode) {
            nativeRemoveNode(lView[RENDERER], removedPhRNode);
        }
        /** @type {?} */
        const slotValue = (/** @type {?} */ (load(lView, index)));
        if (isLContainer(slotValue)) {
            /** @type {?} */
            const lContainer = (/** @type {?} */ (slotValue));
            if (removedPhTNode.type !== 0 /* Container */) {
                nativeRemoveNode(lView[RENDERER], lContainer[NATIVE]);
            }
        }
        if (markAsDetached) {
            // Define this node as detached to avoid projecting it later
            removedPhTNode.flags |= 64 /* isDetached */;
        }
        ngDevMode && ngDevMode.rendererRemoveNode++;
    }
    /**
     *
     * Use this instruction to create a translation block that doesn't contain any placeholder.
     * It calls both {\@link i18nStart} and {\@link i18nEnd} in one instruction.
     *
     * The translation `message` is the value which is locale specific. The translation string may
     * contain placeholders which associate inner elements and sub-templates within the translation.
     *
     * The translation `message` placeholders are:
     * - `�{index}(:{block})�`: *Binding Placeholder*: Marks a location where an expression will be
     *   interpolated into. The placeholder `index` points to the expression binding index. An optional
     *   `block` that matches the sub-template in which it was declared.
     * - `�#{index}(:{block})�`/`�/#{index}(:{block})�`: *Element Placeholder*:  Marks the beginning
     *   and end of DOM element that were embedded in the original translation block. The placeholder
     *   `index` points to the element index in the template instructions set. An optional `block` that
     *   matches the sub-template in which it was declared.
     * - `�*{index}:{block}�`/`�/*{index}:{block}�`: *Sub-template Placeholder*: Sub-templates must be
     *   split up and translated separately in each angular template function. The `index` points to the
     *   `template` instruction index. A `block` that matches the sub-template in which it was declared.
     *
     * \@codeGenApi
     * @param {?} index A unique index of the translation in the static block.
     * @param {?} message The translation message.
     * @param {?=} subTemplateIndex Optional sub-template index in the `message`.
     *
     * @return {?}
     */
    function ɵɵi18n(index, message, subTemplateIndex) {
        ɵɵi18nStart(index, message, subTemplateIndex);
        ɵɵi18nEnd();
    }
    /**
     * Marks a list of attributes as translatable.
     *
     * \@codeGenApi
     * @param {?} index A unique index in the static block
     * @param {?} values
     *
     * @return {?}
     */
    function ɵɵi18nAttributes(index, values) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const tView = getTView();
        ngDevMode && assertDefined(tView, `tView should be defined`);
        i18nAttributesFirstPass(lView, tView, index, values);
    }
    /**
     * See `i18nAttributes` above.
     * @param {?} lView
     * @param {?} tView
     * @param {?} index
     * @param {?} values
     * @return {?}
     */
    function i18nAttributesFirstPass(lView, tView, index, values) {
        /** @type {?} */
        const previousElement = getPreviousOrParentTNode();
        /** @type {?} */
        const previousElementIndex = previousElement.index - HEADER_OFFSET;
        /** @type {?} */
        const updateOpCodes = [];
        for (let i = 0; i < values.length; i += 2) {
            /** @type {?} */
            const attrName = values[i];
            /** @type {?} */
            const message = values[i + 1];
            /** @type {?} */
            const parts = message.split(ICU_REGEXP);
            for (let j = 0; j < parts.length; j++) {
                /** @type {?} */
                const value = parts[j];
                if (j & 1) {
                    // Odd indexes are ICU expressions
                    // TODO(ocombe): support ICU expressions in attributes
                    throw new Error('ICU expressions are not yet supported in attributes');
                }
                else if (value !== '') {
                    // Even indexes are text (including bindings)
                    /** @type {?} */
                    const hasBinding = !!value.match(BINDING_REGEXP);
                    if (hasBinding) {
                        if (tView.firstCreatePass && tView.data[index + HEADER_OFFSET] === null) {
                            addAllToArray(generateBindingUpdateOpCodes(value, previousElementIndex, attrName), updateOpCodes);
                        }
                    }
                    else {
                        /** @type {?} */
                        const tNode = getTNode(tView, previousElementIndex);
                        // Set attributes for Elements only, for other types (like ElementContainer),
                        // only set inputs below
                        if (tNode.type === 3 /* Element */) {
                            elementAttributeInternal(tNode, lView, attrName, value, null, null);
                        }
                        // Check if that attribute is a directive input
                        /** @type {?} */
                        const dataValue = tNode.inputs !== null && tNode.inputs[attrName];
                        if (dataValue) {
                            setInputsForProperty(tView, lView, dataValue, attrName, value);
                            if (ngDevMode) {
                                /** @type {?} */
                                const element = (/** @type {?} */ (getNativeByIndex(previousElementIndex, lView)));
                                setNgReflectProperties(lView, element, tNode.type, dataValue, value);
                            }
                        }
                    }
                }
            }
        }
        if (tView.firstCreatePass && tView.data[index + HEADER_OFFSET] === null) {
            tView.data[index + HEADER_OFFSET] = updateOpCodes;
        }
    }
    /** @type {?} */
    let changeMask = 0b0;
    /** @type {?} */
    let shiftsCounter = 0;
    /**
     * Stores the values of the bindings during each update cycle in order to determine if we need to
     * update the translated nodes.
     *
     * \@codeGenApi
     * @template T
     * @param {?} value The binding's value
     * @return {?} This function returns itself so that it may be chained
     * (e.g. `i18nExp(ctx.name)(ctx.title)`)
     *
     */
    function ɵɵi18nExp(value) {
        /** @type {?} */
        const lView = getLView();
        if (bindingUpdated(lView, nextBindingIndex(), value)) {
            changeMask = changeMask | (1 << shiftsCounter);
        }
        shiftsCounter++;
        return ɵɵi18nExp;
    }
    /**
     * Updates a translation block or an i18n attribute when the bindings have changed.
     *
     * \@codeGenApi
     * @param {?} index Index of either {\@link i18nStart} (translation block) or {\@link i18nAttributes}
     * (i18n attribute) on which it should update the content.
     *
     * @return {?}
     */
    function ɵɵi18nApply(index) {
        if (shiftsCounter) {
            /** @type {?} */
            const tView = getTView();
            ngDevMode && assertDefined(tView, `tView should be defined`);
            /** @type {?} */
            const tI18n = tView.data[index + HEADER_OFFSET];
            /** @type {?} */
            let updateOpCodes;
            /** @type {?} */
            let icus = null;
            if (Array.isArray(tI18n)) {
                updateOpCodes = (/** @type {?} */ (tI18n));
            }
            else {
                updateOpCodes = ((/** @type {?} */ (tI18n))).update;
                icus = ((/** @type {?} */ (tI18n))).icus;
            }
            /** @type {?} */
            const bindingsStartIndex = getBindingIndex() - shiftsCounter - 1;
            /** @type {?} */
            const lView = getLView();
            readUpdateOpCodes(updateOpCodes, icus, bindingsStartIndex, changeMask, tView, lView);
            // Reset changeMask & maskBit to default for the next update cycle
            changeMask = 0b0;
            shiftsCounter = 0;
        }
    }
    /**
     * Returns the index of the current case of an ICU expression depending on the main binding value
     *
     * @param {?} icuExpression
     * @param {?} bindingValue The value of the main binding used by this ICU expression
     * @return {?}
     */
    function getCaseIndex(icuExpression, bindingValue) {
        /** @type {?} */
        let index = icuExpression.cases.indexOf(bindingValue);
        if (index === -1) {
            switch (icuExpression.type) {
                case 1 /* plural */: {
                    /** @type {?} */
                    const resolvedCase = getPluralCase(bindingValue, getLocaleId());
                    index = icuExpression.cases.indexOf(resolvedCase);
                    if (index === -1 && resolvedCase !== 'other') {
                        index = icuExpression.cases.indexOf('other');
                    }
                    break;
                }
                case 0 /* select */: {
                    index = icuExpression.cases.indexOf('other');
                    break;
                }
            }
        }
        return index;
    }
    /**
     * Generate the OpCodes for ICU expressions.
     *
     * @param {?} tIcus
     * @param {?} icuExpression
     * @param {?} startIndex
     * @param {?} expandoStartIndex
     * @return {?}
     */
    function icuStart(tIcus, icuExpression, startIndex, expandoStartIndex) {
        /** @type {?} */
        const createCodes = [];
        /** @type {?} */
        const removeCodes = [];
        /** @type {?} */
        const updateCodes = [];
        /** @type {?} */
        const vars = [];
        /** @type {?} */
        const childIcus = [];
        for (let i = 0; i < icuExpression.values.length; i++) {
            // Each value is an array of strings & other ICU expressions
            /** @type {?} */
            const valueArr = icuExpression.values[i];
            /** @type {?} */
            const nestedIcus = [];
            for (let j = 0; j < valueArr.length; j++) {
                /** @type {?} */
                const value = valueArr[j];
                if (typeof value !== 'string') {
                    // It is an nested ICU expression
                    /** @type {?} */
                    const icuIndex = nestedIcus.push((/** @type {?} */ (value))) - 1;
                    // Replace nested ICU expression by a comment node
                    valueArr[j] = `<!--�${icuIndex}�-->`;
                }
            }
            /** @type {?} */
            const icuCase = parseIcuCase(valueArr.join(''), startIndex, nestedIcus, tIcus, expandoStartIndex);
            createCodes.push(icuCase.create);
            removeCodes.push(icuCase.remove);
            updateCodes.push(icuCase.update);
            vars.push(icuCase.vars);
            childIcus.push(icuCase.childIcus);
        }
        /** @type {?} */
        const tIcu = {
            type: icuExpression.type,
            vars,
            childIcus,
            cases: icuExpression.cases,
            create: createCodes,
            remove: removeCodes,
            update: updateCodes
        };
        tIcus.push(tIcu);
        // Adding the maximum possible of vars needed (based on the cases with the most vars)
        i18nVarsCount += Math.max(...vars);
    }
    /**
     * Transforms a string template into an HTML template and a list of instructions used to update
     * attributes or nodes that contain bindings.
     *
     * @param {?} unsafeHtml The string to parse
     * @param {?} parentIndex
     * @param {?} nestedIcus
     * @param {?} tIcus
     * @param {?} expandoStartIndex
     * @return {?}
     */
    function parseIcuCase(unsafeHtml, parentIndex, nestedIcus, tIcus, expandoStartIndex) {
        /** @type {?} */
        const inertBodyHelper = new InertBodyHelper(getDocument());
        /** @type {?} */
        const inertBodyElement = inertBodyHelper.getInertBodyElement(unsafeHtml);
        if (!inertBodyElement) {
            throw new Error('Unable to generate inert body element');
        }
        /** @type {?} */
        const wrapper = (/** @type {?} */ (getTemplateContent((/** @type {?} */ (inertBodyElement))))) || inertBodyElement;
        /** @type {?} */
        const opCodes = { vars: 0, childIcus: [], create: [], remove: [], update: [] };
        parseNodes(wrapper.firstChild, opCodes, parentIndex, nestedIcus, tIcus, expandoStartIndex);
        return opCodes;
    }
    /** @type {?} */
    const NESTED_ICU = /�(\d+)�/;
    /**
     * Parses a node, its children and its siblings, and generates the mutate & update OpCodes.
     *
     * @param {?} currentNode The first node to parse
     * @param {?} icuCase The data for the ICU expression case that contains those nodes
     * @param {?} parentIndex Index of the current node's parent
     * @param {?} nestedIcus Data for the nested ICU expressions that this case contains
     * @param {?} tIcus Data for all ICU expressions of the current message
     * @param {?} expandoStartIndex Expando start index for the current ICU expression
     * @return {?}
     */
    function parseNodes(currentNode, icuCase, parentIndex, nestedIcus, tIcus, expandoStartIndex) {
        if (currentNode) {
            /** @type {?} */
            const nestedIcusToCreate = [];
            while (currentNode) {
                /** @type {?} */
                const nextNode = currentNode.nextSibling;
                /** @type {?} */
                const newIndex = expandoStartIndex + ++icuCase.vars;
                switch (currentNode.nodeType) {
                    case Node.ELEMENT_NODE:
                        /** @type {?} */
                        const element = (/** @type {?} */ (currentNode));
                        /** @type {?} */
                        const tagName = element.tagName.toLowerCase();
                        if (!VALID_ELEMENTS.hasOwnProperty(tagName)) {
                            // This isn't a valid element, we won't create an element for it
                            icuCase.vars--;
                        }
                        else {
                            icuCase.create.push(ELEMENT_MARKER, tagName, newIndex, parentIndex << 17 /* SHIFT_PARENT */ | 1 /* AppendChild */);
                            /** @type {?} */
                            const elAttrs = element.attributes;
                            for (let i = 0; i < elAttrs.length; i++) {
                                /** @type {?} */
                                const attr = (/** @type {?} */ (elAttrs.item(i)));
                                /** @type {?} */
                                const lowerAttrName = attr.name.toLowerCase();
                                /** @type {?} */
                                const hasBinding = !!attr.value.match(BINDING_REGEXP);
                                // we assume the input string is safe, unless it's using a binding
                                if (hasBinding) {
                                    if (VALID_ATTRS.hasOwnProperty(lowerAttrName)) {
                                        if (URI_ATTRS[lowerAttrName]) {
                                            addAllToArray(generateBindingUpdateOpCodes(attr.value, newIndex, attr.name, _sanitizeUrl), icuCase.update);
                                        }
                                        else if (SRCSET_ATTRS[lowerAttrName]) {
                                            addAllToArray(generateBindingUpdateOpCodes(attr.value, newIndex, attr.name, sanitizeSrcset), icuCase.update);
                                        }
                                        else {
                                            addAllToArray(generateBindingUpdateOpCodes(attr.value, newIndex, attr.name), icuCase.update);
                                        }
                                    }
                                    else {
                                        ngDevMode &&
                                            console.warn(`WARNING: ignoring unsafe attribute value ${lowerAttrName} on element ${tagName} (see http://g.co/ng/security#xss)`);
                                    }
                                }
                                else {
                                    icuCase.create.push(newIndex << 3 /* SHIFT_REF */ | 4 /* Attr */, attr.name, attr.value);
                                }
                            }
                            // Parse the children of this node (if any)
                            parseNodes(currentNode.firstChild, icuCase, newIndex, nestedIcus, tIcus, expandoStartIndex);
                            // Remove the parent node after the children
                            icuCase.remove.push(newIndex << 3 /* SHIFT_REF */ | 3 /* Remove */);
                        }
                        break;
                    case Node.TEXT_NODE:
                        /** @type {?} */
                        const value = currentNode.textContent || '';
                        /** @type {?} */
                        const hasBinding = value.match(BINDING_REGEXP);
                        icuCase.create.push(hasBinding ? '' : value, newIndex, parentIndex << 17 /* SHIFT_PARENT */ | 1 /* AppendChild */);
                        icuCase.remove.push(newIndex << 3 /* SHIFT_REF */ | 3 /* Remove */);
                        if (hasBinding) {
                            addAllToArray(generateBindingUpdateOpCodes(value, newIndex), icuCase.update);
                        }
                        break;
                    case Node.COMMENT_NODE:
                        // Check if the comment node is a placeholder for a nested ICU
                        /** @type {?} */
                        const match = NESTED_ICU.exec(currentNode.textContent || '');
                        if (match) {
                            /** @type {?} */
                            const nestedIcuIndex = parseInt(match[1], 10);
                            /** @type {?} */
                            const newLocal = ngDevMode ? `nested ICU ${nestedIcuIndex}` : '';
                            // Create the comment node that will anchor the ICU expression
                            icuCase.create.push(COMMENT_MARKER, newLocal, newIndex, parentIndex << 17 /* SHIFT_PARENT */ | 1 /* AppendChild */);
                            /** @type {?} */
                            const nestedIcu = nestedIcus[nestedIcuIndex];
                            nestedIcusToCreate.push([nestedIcu, newIndex]);
                        }
                        else {
                            // We do not handle any other type of comment
                            icuCase.vars--;
                        }
                        break;
                    default:
                        // We do not handle any other type of element
                        icuCase.vars--;
                }
                currentNode = (/** @type {?} */ (nextNode));
            }
            for (let i = 0; i < nestedIcusToCreate.length; i++) {
                /** @type {?} */
                const nestedIcu = nestedIcusToCreate[i][0];
                /** @type {?} */
                const nestedIcuNodeIndex = nestedIcusToCreate[i][1];
                icuStart(tIcus, nestedIcu, nestedIcuNodeIndex, expandoStartIndex + icuCase.vars);
                // Since this is recursive, the last TIcu that was pushed is the one we want
                /** @type {?} */
                const nestTIcuIndex = tIcus.length - 1;
                icuCase.vars += Math.max(...tIcus[nestTIcuIndex].vars);
                icuCase.childIcus.push(nestTIcuIndex);
                /** @type {?} */
                const mask = getBindingMask(nestedIcu);
                icuCase.update.push(toMaskBit(nestedIcu.mainBinding), // mask of the main binding
                3, // skip 3 opCodes if not changed
                -1 - nestedIcu.mainBinding, nestedIcuNodeIndex << 2 /* SHIFT_REF */ | 2 /* IcuSwitch */, nestTIcuIndex, mask, // mask of all the bindings of this ICU expression
                2, // skip 2 opCodes if not changed
                nestedIcuNodeIndex << 2 /* SHIFT_REF */ | 3 /* IcuUpdate */, nestTIcuIndex);
                icuCase.remove.push(nestTIcuIndex << 3 /* SHIFT_REF */ | 6 /* RemoveNestedIcu */, nestedIcuNodeIndex << 3 /* SHIFT_REF */ | 3 /* Remove */);
            }
        }
    }
    /**
     * Angular Dart introduced &ngsp; as a placeholder for non-removable space, see:
     * https://github.com/dart-lang/angular/blob/0bb611387d29d65b5af7f9d2515ab571fd3fbee4/_tests/test/compiler/preserve_whitespace_test.dart#L25-L32
     * In Angular Dart &ngsp; is converted to the 0xE500 PUA (Private Use Areas) unicode character
     * and later on replaced by a space. We are re-implementing the same idea here, since translations
     * might contain this special character.
     * @type {?}
     */
    const NGSP_UNICODE_REGEXP = /\uE500/g;
    /**
     * @param {?} value
     * @return {?}
     */
    function replaceNgsp(value) {
        return value.replace(NGSP_UNICODE_REGEXP, ' ');
    }
    /**
     * The locale id that the application is currently using (for translations and ICU expressions).
     * This is the ivy version of `LOCALE_ID` that was defined as an injection token for the view engine
     * but is now defined as a global value.
     * @type {?}
     */
    let LOCALE_ID = DEFAULT_LOCALE_ID;
    /**
     * Sets the locale id that will be used for translations and ICU expressions.
     * This is the ivy version of `LOCALE_ID` that was defined as an injection token for the view engine
     * but is now defined as a global value.
     *
     * @param {?} localeId
     * @return {?}
     */
    function setLocaleId(localeId) {
        assertDefined(localeId, `Expected localeId to be defined`);
        if (typeof localeId === 'string') {
            LOCALE_ID = localeId.toLowerCase().replace(/_/g, '-');
        }
    }
    /**
     * Gets the locale id that will be used for translations and ICU expressions.
     * This is the ivy version of `LOCALE_ID` that was defined as an injection token for the view engine
     * but is now defined as a global value.
     * @return {?}
     */
    function getLocaleId() {
        return LOCALE_ID;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/linker/ng_module_factory_registration.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Map of module-id to the corresponding NgModule.
     * - In pre Ivy we track NgModuleFactory,
     * - In post Ivy we track the NgModuleType
     * @type {?}
     */
    const modules = new Map();
    /**
     * @param {?} id
     * @param {?} type
     * @param {?} incoming
     * @return {?}
     */
    function assertSameOrNotExisting(id, type, incoming) {
        if (type && type !== incoming) {
            throw new Error(`Duplicate module registered for ${id} - ${stringify(type)} vs ${stringify(type.name)}`);
        }
    }
    /**
     * @param {?} ngModuleType
     * @return {?}
     */
    function registerNgModuleType(ngModuleType) {
        if (ngModuleType.ɵmod.id !== null) {
            /** @type {?} */
            const id = ngModuleType.ɵmod.id;
            /** @type {?} */
            const existing = (/** @type {?} */ (modules.get(id)));
            assertSameOrNotExisting(id, existing, ngModuleType);
            modules.set(id, ngModuleType);
        }
        /** @type {?} */
        let imports = ngModuleType.ɵmod.imports;
        if (imports instanceof Function) {
            imports = imports();
        }
        if (imports) {
            imports.forEach((/**
             * @param {?} i
             * @return {?}
             */
            i => registerNgModuleType((/** @type {?} */ (i)))));
        }
    }
    /**
     * @template T
     */
    class NgModuleRef$1 extends NgModuleRef {
        /**
         * @param {?} ngModuleType
         * @param {?} _parent
         */
        constructor(ngModuleType, _parent) {
            super();
            this._parent = _parent;
            // tslint:disable-next-line:require-internal-with-underscore
            this._bootstrapComponents = [];
            this.injector = this;
            this.destroyCbs = [];
            // When bootstrapping a module we have a dependency graph that looks like this:
            // ApplicationRef -> ComponentFactoryResolver -> NgModuleRef. The problem is that if the
            // module being resolved tries to inject the ComponentFactoryResolver, it'll create a
            // circular dependency which will result in a runtime error, because the injector doesn't
            // exist yet. We work around the issue by creating the ComponentFactoryResolver ourselves
            // and providing it, rather than letting the injector resolve it.
            this.componentFactoryResolver = new ComponentFactoryResolver$1(this);
            /** @type {?} */
            const ngModuleDef = getNgModuleDef(ngModuleType);
            ngDevMode &&
                assertDefined(ngModuleDef, `NgModule '${stringify(ngModuleType)}' is not a subtype of 'NgModuleType'.`);
            /** @type {?} */
            const ngLocaleIdDef = getNgLocaleIdDef(ngModuleType);
            ngLocaleIdDef && setLocaleId(ngLocaleIdDef);
            this._bootstrapComponents = maybeUnwrapFn((/** @type {?} */ (ngModuleDef)).bootstrap);
            this._r3Injector = (/** @type {?} */ (createInjectorWithoutInjectorInstances(ngModuleType, _parent, [
                { provide: NgModuleRef, useValue: this }, {
                    provide: ComponentFactoryResolver,
                    useValue: this.componentFactoryResolver
                }
            ], stringify(ngModuleType))));
            // We need to resolve the injector types separately from the injector creation, because
            // the module might be trying to use this ref in its contructor for DI which will cause a
            // circular error that will eventually error out, because the injector isn't created yet.
            this._r3Injector._resolveInjectorDefTypes();
            this.instance = this.get(ngModuleType);
        }
        /**
         * @param {?} token
         * @param {?=} notFoundValue
         * @param {?=} injectFlags
         * @return {?}
         */
        get(token, notFoundValue = Injector.THROW_IF_NOT_FOUND, injectFlags = InjectFlags.Default) {
            if (token === Injector || token === NgModuleRef || token === INJECTOR) {
                return this;
            }
            return this._r3Injector.get(token, notFoundValue, injectFlags);
        }
        /**
         * @return {?}
         */
        destroy() {
            ngDevMode && assertDefined(this.destroyCbs, 'NgModule already destroyed');
            /** @type {?} */
            const injector = this._r3Injector;
            !injector.destroyed && injector.destroy();
            (/** @type {?} */ (this.destroyCbs)).forEach((/**
             * @param {?} fn
             * @return {?}
             */
            fn => fn()));
            this.destroyCbs = null;
        }
        /**
         * @param {?} callback
         * @return {?}
         */
        onDestroy(callback) {
            ngDevMode && assertDefined(this.destroyCbs, 'NgModule already destroyed');
            (/** @type {?} */ (this.destroyCbs)).push(callback);
        }
    }
    /**
     * @template T
     */
    class NgModuleFactory$1 extends NgModuleFactory {
        /**
         * @param {?} moduleType
         */
        constructor(moduleType) {
            super();
            this.moduleType = moduleType;
            /** @type {?} */
            const ngModuleDef = getNgModuleDef(moduleType);
            if (ngModuleDef !== null) {
                // Register the NgModule with Angular's module registry. The location (and hence timing) of
                // this call is critical to ensure this works correctly (modules get registered when expected)
                // without bloating bundles (modules are registered when otherwise not referenced).
                //
                // In View Engine, registration occurs in the .ngfactory.js file as a side effect. This has
                // several practical consequences:
                //
                // - If an .ngfactory file is not imported from, the module won't be registered (and can be
                //   tree shaken).
                // - If an .ngfactory file is imported from, the module will be registered even if an instance
                //   is not actually created (via `create` below).
                // - Since an .ngfactory file in View Engine references the .ngfactory files of the NgModule's
                //   imports,
                //
                // In Ivy, things are a bit different. .ngfactory files still exist for compatibility, but are
                // not a required API to use - there are other ways to obtain an NgModuleFactory for a given
                // NgModule. Thus, relying on a side effect in the .ngfactory file is not sufficient. Instead,
                // the side effect of registration is added here, in the constructor of NgModuleFactory,
                // ensuring no matter how a factory is created, the module is registered correctly.
                //
                // An alternative would be to include the registration side effect inline following the actual
                // NgModule definition. This also has the correct timing, but breaks tree-shaking - modules
                // will be registered and retained even if they're otherwise never referenced.
                registerNgModuleType((/** @type {?} */ (moduleType)));
            }
        }
        /**
         * @param {?} parentInjector
         * @return {?}
         */
        create(parentInjector) {
            return new NgModuleRef$1(this.moduleType, parentInjector);
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/pure_function.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Bindings for pure functions are stored after regular bindings.
     *
     * |-------decls------|---------vars---------|                 |----- hostVars (dir1) ------|
     * ------------------------------------------------------------------------------------------
     * | nodes/refs/pipes | bindings | fn slots  | injector | dir1 | host bindings | host slots |
     * ------------------------------------------------------------------------------------------
     *                    ^                      ^
     *      TView.bindingStartIndex      TView.expandoStartIndex
     *
     * Pure function instructions are given an offset from the binding root. Adding the offset to the
     * binding root gives the first index where the bindings are stored. In component views, the binding
     * root is the bindingStartIndex. In host bindings, the binding root is the expandoStartIndex +
     * any directive instances + any hostVars in directives evaluated before it.
     *
     * See VIEW_DATA.md for more information about host binding resolution.
     */
    /**
     * If the value hasn't been saved, calls the pure function to store and return the
     * value. If it has been saved, returns the saved value.
     *
     * \@codeGenApi
     * @template T
     * @param {?} slotOffset the offset from binding root to the reserved slot
     * @param {?} pureFn Function that returns a value
     * @param {?=} thisArg Optional calling context of pureFn
     * @return {?} value
     *
     */
    function ɵɵpureFunction0(slotOffset, pureFn, thisArg) {
        /** @type {?} */
        const bindingIndex = getBindingRoot() + slotOffset;
        /** @type {?} */
        const lView = getLView();
        return lView[bindingIndex] === NO_CHANGE ?
            updateBinding(lView, bindingIndex, thisArg ? pureFn.call(thisArg) : pureFn()) :
            getBinding(lView, bindingIndex);
    }
    /**
     * If the value of the provided exp has changed, calls the pure function to return
     * an updated value. Or if the value has not changed, returns cached value.
     *
     * \@codeGenApi
     * @param {?} slotOffset the offset from binding root to the reserved slot
     * @param {?} pureFn Function that returns an updated value
     * @param {?} exp Updated expression value
     * @param {?=} thisArg Optional calling context of pureFn
     * @return {?} Updated or cached value
     *
     */
    function ɵɵpureFunction1(slotOffset, pureFn, exp, thisArg) {
        return pureFunction1Internal(getLView(), getBindingRoot(), slotOffset, pureFn, exp, thisArg);
    }
    /**
     * If the value of any provided exp has changed, calls the pure function to return
     * an updated value. Or if no values have changed, returns cached value.
     *
     * \@codeGenApi
     * @param {?} slotOffset the offset from binding root to the reserved slot
     * @param {?} pureFn
     * @param {?} exp1
     * @param {?} exp2
     * @param {?=} thisArg Optional calling context of pureFn
     * @return {?} Updated or cached value
     *
     */
    function ɵɵpureFunction2(slotOffset, pureFn, exp1, exp2, thisArg) {
        return pureFunction2Internal(getLView(), getBindingRoot(), slotOffset, pureFn, exp1, exp2, thisArg);
    }
    /**
     * If the value of any provided exp has changed, calls the pure function to return
     * an updated value. Or if no values have changed, returns cached value.
     *
     * \@codeGenApi
     * @param {?} slotOffset the offset from binding root to the reserved slot
     * @param {?} pureFn
     * @param {?} exp1
     * @param {?} exp2
     * @param {?} exp3
     * @param {?=} thisArg Optional calling context of pureFn
     * @return {?} Updated or cached value
     *
     */
    function ɵɵpureFunction3(slotOffset, pureFn, exp1, exp2, exp3, thisArg) {
        return pureFunction3Internal(getLView(), getBindingRoot(), slotOffset, pureFn, exp1, exp2, exp3, thisArg);
    }
    /**
     * If the value of any provided exp has changed, calls the pure function to return
     * an updated value. Or if no values have changed, returns cached value.
     *
     * \@codeGenApi
     * @param {?} slotOffset the offset from binding root to the reserved slot
     * @param {?} pureFn
     * @param {?} exp1
     * @param {?} exp2
     * @param {?} exp3
     * @param {?} exp4
     * @param {?=} thisArg Optional calling context of pureFn
     * @return {?} Updated or cached value
     *
     */
    function ɵɵpureFunction4(slotOffset, pureFn, exp1, exp2, exp3, exp4, thisArg) {
        return pureFunction4Internal(getLView(), getBindingRoot(), slotOffset, pureFn, exp1, exp2, exp3, exp4, thisArg);
    }
    /**
     * If the value of any provided exp has changed, calls the pure function to return
     * an updated value. Or if no values have changed, returns cached value.
     *
     * \@codeGenApi
     * @param {?} slotOffset the offset from binding root to the reserved slot
     * @param {?} pureFn
     * @param {?} exp1
     * @param {?} exp2
     * @param {?} exp3
     * @param {?} exp4
     * @param {?} exp5
     * @param {?=} thisArg Optional calling context of pureFn
     * @return {?} Updated or cached value
     *
     */
    function ɵɵpureFunction5(slotOffset, pureFn, exp1, exp2, exp3, exp4, exp5, thisArg) {
        /** @type {?} */
        const bindingIndex = getBindingRoot() + slotOffset;
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const different = bindingUpdated4(lView, bindingIndex, exp1, exp2, exp3, exp4);
        return bindingUpdated(lView, bindingIndex + 4, exp5) || different ?
            updateBinding(lView, bindingIndex + 5, thisArg ? pureFn.call(thisArg, exp1, exp2, exp3, exp4, exp5) :
                pureFn(exp1, exp2, exp3, exp4, exp5)) :
            getBinding(lView, bindingIndex + 5);
    }
    /**
     * If the value of any provided exp has changed, calls the pure function to return
     * an updated value. Or if no values have changed, returns cached value.
     *
     * \@codeGenApi
     * @param {?} slotOffset the offset from binding root to the reserved slot
     * @param {?} pureFn
     * @param {?} exp1
     * @param {?} exp2
     * @param {?} exp3
     * @param {?} exp4
     * @param {?} exp5
     * @param {?} exp6
     * @param {?=} thisArg Optional calling context of pureFn
     * @return {?} Updated or cached value
     *
     */
    function ɵɵpureFunction6(slotOffset, pureFn, exp1, exp2, exp3, exp4, exp5, exp6, thisArg) {
        /** @type {?} */
        const bindingIndex = getBindingRoot() + slotOffset;
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const different = bindingUpdated4(lView, bindingIndex, exp1, exp2, exp3, exp4);
        return bindingUpdated2(lView, bindingIndex + 4, exp5, exp6) || different ?
            updateBinding(lView, bindingIndex + 6, thisArg ? pureFn.call(thisArg, exp1, exp2, exp3, exp4, exp5, exp6) :
                pureFn(exp1, exp2, exp3, exp4, exp5, exp6)) :
            getBinding(lView, bindingIndex + 6);
    }
    /**
     * If the value of any provided exp has changed, calls the pure function to return
     * an updated value. Or if no values have changed, returns cached value.
     *
     * \@codeGenApi
     * @param {?} slotOffset the offset from binding root to the reserved slot
     * @param {?} pureFn
     * @param {?} exp1
     * @param {?} exp2
     * @param {?} exp3
     * @param {?} exp4
     * @param {?} exp5
     * @param {?} exp6
     * @param {?} exp7
     * @param {?=} thisArg Optional calling context of pureFn
     * @return {?} Updated or cached value
     *
     */
    function ɵɵpureFunction7(slotOffset, pureFn, exp1, exp2, exp3, exp4, exp5, exp6, exp7, thisArg) {
        /** @type {?} */
        const bindingIndex = getBindingRoot() + slotOffset;
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        let different = bindingUpdated4(lView, bindingIndex, exp1, exp2, exp3, exp4);
        return bindingUpdated3(lView, bindingIndex + 4, exp5, exp6, exp7) || different ?
            updateBinding(lView, bindingIndex + 7, thisArg ? pureFn.call(thisArg, exp1, exp2, exp3, exp4, exp5, exp6, exp7) :
                pureFn(exp1, exp2, exp3, exp4, exp5, exp6, exp7)) :
            getBinding(lView, bindingIndex + 7);
    }
    /**
     * If the value of any provided exp has changed, calls the pure function to return
     * an updated value. Or if no values have changed, returns cached value.
     *
     * \@codeGenApi
     * @param {?} slotOffset the offset from binding root to the reserved slot
     * @param {?} pureFn
     * @param {?} exp1
     * @param {?} exp2
     * @param {?} exp3
     * @param {?} exp4
     * @param {?} exp5
     * @param {?} exp6
     * @param {?} exp7
     * @param {?} exp8
     * @param {?=} thisArg Optional calling context of pureFn
     * @return {?} Updated or cached value
     *
     */
    function ɵɵpureFunction8(slotOffset, pureFn, exp1, exp2, exp3, exp4, exp5, exp6, exp7, exp8, thisArg) {
        /** @type {?} */
        const bindingIndex = getBindingRoot() + slotOffset;
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const different = bindingUpdated4(lView, bindingIndex, exp1, exp2, exp3, exp4);
        return bindingUpdated4(lView, bindingIndex + 4, exp5, exp6, exp7, exp8) || different ?
            updateBinding(lView, bindingIndex + 8, thisArg ? pureFn.call(thisArg, exp1, exp2, exp3, exp4, exp5, exp6, exp7, exp8) :
                pureFn(exp1, exp2, exp3, exp4, exp5, exp6, exp7, exp8)) :
            getBinding(lView, bindingIndex + 8);
    }
    /**
     * pureFunction instruction that can support any number of bindings.
     *
     * If the value of any provided exp has changed, calls the pure function to return
     * an updated value. Or if no values have changed, returns cached value.
     *
     * \@codeGenApi
     * @param {?} slotOffset the offset from binding root to the reserved slot
     * @param {?} pureFn A pure function that takes binding values and builds an object or array
     * containing those values.
     * @param {?} exps An array of binding values
     * @param {?=} thisArg Optional calling context of pureFn
     * @return {?} Updated or cached value
     *
     */
    function ɵɵpureFunctionV(slotOffset, pureFn, exps, thisArg) {
        return pureFunctionVInternal(getLView(), getBindingRoot(), slotOffset, pureFn, exps, thisArg);
    }
    /**
     * Results of a pure function invocation are stored in LView in a dedicated slot that is initialized
     * to NO_CHANGE. In rare situations a pure pipe might throw an exception on the very first
     * invocation and not produce any valid results. In this case LView would keep holding the NO_CHANGE
     * value. The NO_CHANGE is not something that we can use in expressions / bindings thus we convert
     * it to `undefined`.
     * @param {?} lView
     * @param {?} returnValueIndex
     * @return {?}
     */
    function getPureFunctionReturnValue(lView, returnValueIndex) {
        ngDevMode && assertDataInRange(lView, returnValueIndex);
        /** @type {?} */
        const lastReturnValue = lView[returnValueIndex];
        return lastReturnValue === NO_CHANGE ? undefined : lastReturnValue;
    }
    /**
     * If the value of the provided exp has changed, calls the pure function to return
     * an updated value. Or if the value has not changed, returns cached value.
     *
     * @param {?} lView LView in which the function is being executed.
     * @param {?} bindingRoot Binding root index.
     * @param {?} slotOffset the offset from binding root to the reserved slot
     * @param {?} pureFn Function that returns an updated value
     * @param {?} exp Updated expression value
     * @param {?=} thisArg Optional calling context of pureFn
     * @return {?} Updated or cached value
     */
    function pureFunction1Internal(lView, bindingRoot, slotOffset, pureFn, exp, thisArg) {
        /** @type {?} */
        const bindingIndex = bindingRoot + slotOffset;
        return bindingUpdated(lView, bindingIndex, exp) ?
            updateBinding(lView, bindingIndex + 1, thisArg ? pureFn.call(thisArg, exp) : pureFn(exp)) :
            getPureFunctionReturnValue(lView, bindingIndex + 1);
    }
    /**
     * If the value of any provided exp has changed, calls the pure function to return
     * an updated value. Or if no values have changed, returns cached value.
     *
     * @param {?} lView LView in which the function is being executed.
     * @param {?} bindingRoot Binding root index.
     * @param {?} slotOffset the offset from binding root to the reserved slot
     * @param {?} pureFn
     * @param {?} exp1
     * @param {?} exp2
     * @param {?=} thisArg Optional calling context of pureFn
     * @return {?} Updated or cached value
     */
    function pureFunction2Internal(lView, bindingRoot, slotOffset, pureFn, exp1, exp2, thisArg) {
        /** @type {?} */
        const bindingIndex = bindingRoot + slotOffset;
        return bindingUpdated2(lView, bindingIndex, exp1, exp2) ?
            updateBinding(lView, bindingIndex + 2, thisArg ? pureFn.call(thisArg, exp1, exp2) : pureFn(exp1, exp2)) :
            getPureFunctionReturnValue(lView, bindingIndex + 2);
    }
    /**
     * If the value of any provided exp has changed, calls the pure function to return
     * an updated value. Or if no values have changed, returns cached value.
     *
     * @param {?} lView LView in which the function is being executed.
     * @param {?} bindingRoot Binding root index.
     * @param {?} slotOffset the offset from binding root to the reserved slot
     * @param {?} pureFn
     * @param {?} exp1
     * @param {?} exp2
     * @param {?} exp3
     * @param {?=} thisArg Optional calling context of pureFn
     * @return {?} Updated or cached value
     */
    function pureFunction3Internal(lView, bindingRoot, slotOffset, pureFn, exp1, exp2, exp3, thisArg) {
        /** @type {?} */
        const bindingIndex = bindingRoot + slotOffset;
        return bindingUpdated3(lView, bindingIndex, exp1, exp2, exp3) ?
            updateBinding(lView, bindingIndex + 3, thisArg ? pureFn.call(thisArg, exp1, exp2, exp3) : pureFn(exp1, exp2, exp3)) :
            getPureFunctionReturnValue(lView, bindingIndex + 3);
    }
    /**
     * If the value of any provided exp has changed, calls the pure function to return
     * an updated value. Or if no values have changed, returns cached value.
     *
     * @param {?} lView LView in which the function is being executed.
     * @param {?} bindingRoot Binding root index.
     * @param {?} slotOffset the offset from binding root to the reserved slot
     * @param {?} pureFn
     * @param {?} exp1
     * @param {?} exp2
     * @param {?} exp3
     * @param {?} exp4
     * @param {?=} thisArg Optional calling context of pureFn
     * @return {?} Updated or cached value
     *
     */
    function pureFunction4Internal(lView, bindingRoot, slotOffset, pureFn, exp1, exp2, exp3, exp4, thisArg) {
        /** @type {?} */
        const bindingIndex = bindingRoot + slotOffset;
        return bindingUpdated4(lView, bindingIndex, exp1, exp2, exp3, exp4) ?
            updateBinding(lView, bindingIndex + 4, thisArg ? pureFn.call(thisArg, exp1, exp2, exp3, exp4) : pureFn(exp1, exp2, exp3, exp4)) :
            getPureFunctionReturnValue(lView, bindingIndex + 4);
    }
    /**
     * pureFunction instruction that can support any number of bindings.
     *
     * If the value of any provided exp has changed, calls the pure function to return
     * an updated value. Or if no values have changed, returns cached value.
     *
     * @param {?} lView LView in which the function is being executed.
     * @param {?} bindingRoot Binding root index.
     * @param {?} slotOffset the offset from binding root to the reserved slot
     * @param {?} pureFn A pure function that takes binding values and builds an object or array
     * containing those values.
     * @param {?} exps An array of binding values
     * @param {?=} thisArg Optional calling context of pureFn
     * @return {?} Updated or cached value
     */
    function pureFunctionVInternal(lView, bindingRoot, slotOffset, pureFn, exps, thisArg) {
        /** @type {?} */
        let bindingIndex = bindingRoot + slotOffset;
        /** @type {?} */
        let different = false;
        for (let i = 0; i < exps.length; i++) {
            bindingUpdated(lView, bindingIndex++, exps[i]) && (different = true);
        }
        return different ? updateBinding(lView, bindingIndex, pureFn.apply(thisArg, exps)) :
            getPureFunctionReturnValue(lView, bindingIndex);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/pipe.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Create a pipe.
     *
     * \@codeGenApi
     * @param {?} index Pipe index where the pipe will be stored.
     * @param {?} pipeName The name of the pipe
     * @return {?} T the instance of the pipe.
     *
     */
    function ɵɵpipe(index, pipeName) {
        /** @type {?} */
        const tView = getTView();
        /** @type {?} */
        let pipeDef;
        /** @type {?} */
        const adjustedIndex = index + HEADER_OFFSET;
        if (tView.firstCreatePass) {
            pipeDef = getPipeDef$1(pipeName, tView.pipeRegistry);
            tView.data[adjustedIndex] = pipeDef;
            if (pipeDef.onDestroy) {
                (tView.destroyHooks || (tView.destroyHooks = [])).push(adjustedIndex, pipeDef.onDestroy);
            }
        }
        else {
            pipeDef = (/** @type {?} */ (tView.data[adjustedIndex]));
        }
        /** @type {?} */
        const pipeFactory = pipeDef.factory || (pipeDef.factory = getFactoryDef(pipeDef.type, true));
        /** @type {?} */
        const previousInjectImplementation = setInjectImplementation(ɵɵdirectiveInject);
        // DI for pipes is supposed to behave like directives when placed on a component
        // host node, which means that we have to disable access to `viewProviders`.
        /** @type {?} */
        const previousIncludeViewProviders = setIncludeViewProviders(false);
        /** @type {?} */
        const pipeInstance = pipeFactory();
        setIncludeViewProviders(previousIncludeViewProviders);
        setInjectImplementation(previousInjectImplementation);
        store(tView, getLView(), index, pipeInstance);
        return pipeInstance;
    }
    /**
     * Searches the pipe registry for a pipe with the given name. If one is found,
     * returns the pipe. Otherwise, an error is thrown because the pipe cannot be resolved.
     *
     * @param {?} name Name of pipe to resolve
     * @param {?} registry Full list of available pipes
     * @return {?} Matching PipeDef
     */
    function getPipeDef$1(name, registry) {
        if (registry) {
            for (let i = registry.length - 1; i >= 0; i--) {
                /** @type {?} */
                const pipeDef = registry[i];
                if (name === pipeDef.name) {
                    return pipeDef;
                }
            }
        }
        throw new Error(`The pipe '${name}' could not be found!`);
    }
    /**
     * Invokes a pipe with 1 arguments.
     *
     * This instruction acts as a guard to {\@link PipeTransform#transform} invoking
     * the pipe only when an input to the pipe changes.
     *
     * \@codeGenApi
     * @param {?} index Pipe index where the pipe was stored on creation.
     * @param {?} slotOffset the offset in the reserved slot space
     * @param {?} v1 1st argument to {\@link PipeTransform#transform}.
     *
     * @return {?}
     */
    function ɵɵpipeBind1(index, slotOffset, v1) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const pipeInstance = load(lView, index);
        return unwrapValue$1(lView, isPure(lView, index) ?
            pureFunction1Internal(lView, getBindingRoot(), slotOffset, pipeInstance.transform, v1, pipeInstance) :
            pipeInstance.transform(v1));
    }
    /**
     * Invokes a pipe with 2 arguments.
     *
     * This instruction acts as a guard to {\@link PipeTransform#transform} invoking
     * the pipe only when an input to the pipe changes.
     *
     * \@codeGenApi
     * @param {?} index Pipe index where the pipe was stored on creation.
     * @param {?} slotOffset the offset in the reserved slot space
     * @param {?} v1 1st argument to {\@link PipeTransform#transform}.
     * @param {?} v2 2nd argument to {\@link PipeTransform#transform}.
     *
     * @return {?}
     */
    function ɵɵpipeBind2(index, slotOffset, v1, v2) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const pipeInstance = load(lView, index);
        return unwrapValue$1(lView, isPure(lView, index) ?
            pureFunction2Internal(lView, getBindingRoot(), slotOffset, pipeInstance.transform, v1, v2, pipeInstance) :
            pipeInstance.transform(v1, v2));
    }
    /**
     * Invokes a pipe with 3 arguments.
     *
     * This instruction acts as a guard to {\@link PipeTransform#transform} invoking
     * the pipe only when an input to the pipe changes.
     *
     * \@codeGenApi
     * @param {?} index Pipe index where the pipe was stored on creation.
     * @param {?} slotOffset the offset in the reserved slot space
     * @param {?} v1 1st argument to {\@link PipeTransform#transform}.
     * @param {?} v2 2nd argument to {\@link PipeTransform#transform}.
     * @param {?} v3 4rd argument to {\@link PipeTransform#transform}.
     *
     * @return {?}
     */
    function ɵɵpipeBind3(index, slotOffset, v1, v2, v3) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const pipeInstance = load(lView, index);
        return unwrapValue$1(lView, isPure(lView, index) ? pureFunction3Internal(lView, getBindingRoot(), slotOffset, pipeInstance.transform, v1, v2, v3, pipeInstance) :
            pipeInstance.transform(v1, v2, v3));
    }
    /**
     * Invokes a pipe with 4 arguments.
     *
     * This instruction acts as a guard to {\@link PipeTransform#transform} invoking
     * the pipe only when an input to the pipe changes.
     *
     * \@codeGenApi
     * @param {?} index Pipe index where the pipe was stored on creation.
     * @param {?} slotOffset the offset in the reserved slot space
     * @param {?} v1 1st argument to {\@link PipeTransform#transform}.
     * @param {?} v2 2nd argument to {\@link PipeTransform#transform}.
     * @param {?} v3 3rd argument to {\@link PipeTransform#transform}.
     * @param {?} v4 4th argument to {\@link PipeTransform#transform}.
     *
     * @return {?}
     */
    function ɵɵpipeBind4(index, slotOffset, v1, v2, v3, v4) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const pipeInstance = load(lView, index);
        return unwrapValue$1(lView, isPure(lView, index) ? pureFunction4Internal(lView, getBindingRoot(), slotOffset, pipeInstance.transform, v1, v2, v3, v4, pipeInstance) :
            pipeInstance.transform(v1, v2, v3, v4));
    }
    /**
     * Invokes a pipe with variable number of arguments.
     *
     * This instruction acts as a guard to {\@link PipeTransform#transform} invoking
     * the pipe only when an input to the pipe changes.
     *
     * \@codeGenApi
     * @param {?} index Pipe index where the pipe was stored on creation.
     * @param {?} slotOffset the offset in the reserved slot space
     * @param {?} values Array of arguments to pass to {\@link PipeTransform#transform} method.
     *
     * @return {?}
     */
    function ɵɵpipeBindV(index, slotOffset, values) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const pipeInstance = load(lView, index);
        return unwrapValue$1(lView, isPure(lView, index) ?
            pureFunctionVInternal(lView, getBindingRoot(), slotOffset, pipeInstance.transform, values, pipeInstance) :
            pipeInstance.transform.apply(pipeInstance, values));
    }
    /**
     * @param {?} lView
     * @param {?} index
     * @return {?}
     */
    function isPure(lView, index) {
        return ((/** @type {?} */ (lView[TVIEW].data[index + HEADER_OFFSET]))).pure;
    }
    /**
     * Unwrap the output of a pipe transformation.
     * In order to trick change detection into considering that the new value is always different from
     * the old one, the old value is overwritten by NO_CHANGE.
     *
     * @param {?} lView
     * @param {?} newValue the pipe transformation output.
     * @return {?}
     */
    function unwrapValue$1(lView, newValue) {
        if (WrappedValue.isWrapped(newValue)) {
            newValue = WrappedValue.unwrap(newValue);
            // The NO_CHANGE value needs to be written at the index where the impacted binding value is
            // stored
            /** @type {?} */
            const bindingToInvalidateIdx = getBindingIndex();
            lView[bindingToInvalidateIdx] = NO_CHANGE;
        }
        return newValue;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/event_emitter.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Use in components with the `\@Output` directive to emit custom events
     * synchronously or asynchronously, and register handlers for those events
     * by subscribing to an instance.
     *
     * \@usageNotes
     *
     * Extends
     * [RxJS `Subject`](https://rxjs.dev/api/index/class/Subject)
     * for Angular by adding the `emit()` method.
     *
     * In the following example, a component defines two output properties
     * that create event emitters. When the title is clicked, the emitter
     * emits an open or close event to toggle the current visibility state.
     *
     * ```html
     * \@Component({
     *   selector: 'zippy',
     *   template: `
     *   <div class="zippy">
     *     <div (click)="toggle()">Toggle</div>
     *     <div [hidden]="!visible">
     *       <ng-content></ng-content>
     *     </div>
     *  </div>`})
     * export class Zippy {
     *   visible: boolean = true;
     * \@Output() open: EventEmitter<any> = new EventEmitter();
     * \@Output() close: EventEmitter<any> = new EventEmitter();
     *
     *   toggle() {
     *     this.visible = !this.visible;
     *     if (this.visible) {
     *       this.open.emit(null);
     *     } else {
     *       this.close.emit(null);
     *     }
     *   }
     * }
     * ```
     *
     * Access the event object with the `$event` argument passed to the output event
     * handler:
     *
     * ```html
     * <zippy (open)="onOpen($event)" (close)="onClose($event)"></zippy>
     * ```
     *
     * @see [Observables in Angular](guide/observables-in-angular)
     * \@publicApi
     * @template T
     */
    class EventEmitter extends Subject {
        // tslint:disable-line
        /**
         * Creates an instance of this class that can
         * deliver events synchronously or asynchronously.
         *
         * @param {?=} isAsync When true, deliver events asynchronously.
         *
         */
        constructor(isAsync = false) {
            super();
            this.__isAsync = isAsync;
        }
        /**
         * Emits an event containing a given value.
         * @param {?=} value The value to emit.
         * @return {?}
         */
        emit(value) {
            super.next(value);
        }
        /**
         * Registers handlers for events emitted by this instance.
         * @param {?=} generatorOrNext When supplied, a custom handler for emitted events.
         * @param {?=} error When supplied, a custom handler for an error notification
         * from this emitter.
         * @param {?=} complete When supplied, a custom handler for a completion
         * notification from this emitter.
         * @return {?}
         */
        subscribe(generatorOrNext, error, complete) {
            /** @type {?} */
            let schedulerFn;
            /** @type {?} */
            let errorFn = (/**
             * @param {?} err
             * @return {?}
             */
            (err) => null);
            /** @type {?} */
            let completeFn = (/**
             * @return {?}
             */
            () => null);
            if (generatorOrNext && typeof generatorOrNext === 'object') {
                schedulerFn = this.__isAsync ? (/**
                 * @param {?} value
                 * @return {?}
                 */
                (value) => {
                    setTimeout((/**
                     * @return {?}
                     */
                    () => generatorOrNext.next(value)));
                }) : (/**
                 * @param {?} value
                 * @return {?}
                 */
                (value) => {
                    generatorOrNext.next(value);
                });
                if (generatorOrNext.error) {
                    errorFn = this.__isAsync ? (/**
                     * @param {?} err
                     * @return {?}
                     */
                    (err) => {
                        setTimeout((/**
                         * @return {?}
                         */
                        () => generatorOrNext.error(err)));
                    }) : (/**
                     * @param {?} err
                     * @return {?}
                     */
                    (err) => {
                        generatorOrNext.error(err);
                    });
                }
                if (generatorOrNext.complete) {
                    completeFn = this.__isAsync ? (/**
                     * @return {?}
                     */
                    () => {
                        setTimeout((/**
                         * @return {?}
                         */
                        () => generatorOrNext.complete()));
                    }) : (/**
                     * @return {?}
                     */
                    () => {
                        generatorOrNext.complete();
                    });
                }
            }
            else {
                schedulerFn = this.__isAsync ? (/**
                 * @param {?} value
                 * @return {?}
                 */
                (value) => {
                    setTimeout((/**
                     * @return {?}
                     */
                    () => generatorOrNext(value)));
                }) : (/**
                 * @param {?} value
                 * @return {?}
                 */
                (value) => {
                    generatorOrNext(value);
                });
                if (error) {
                    errorFn = this.__isAsync ? (/**
                     * @param {?} err
                     * @return {?}
                     */
                    (err) => {
                        setTimeout((/**
                         * @return {?}
                         */
                        () => error(err)));
                    }) : (/**
                     * @param {?} err
                     * @return {?}
                     */
                    (err) => {
                        error(err);
                    });
                }
                if (complete) {
                    completeFn = this.__isAsync ? (/**
                     * @return {?}
                     */
                    () => {
                        setTimeout((/**
                         * @return {?}
                         */
                        () => complete()));
                    }) : (/**
                     * @return {?}
                     */
                    () => {
                        complete();
                    });
                }
            }
            /** @type {?} */
            const sink = super.subscribe(schedulerFn, errorFn, completeFn);
            if (generatorOrNext instanceof Subscription) {
                generatorOrNext.add(sink);
            }
            return sink;
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/linker/query_list.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @template T
     * @this {?}
     * @return {?}
     */
    function symbolIterator() {
        return ((/** @type {?} */ (((/** @type {?} */ ((/** @type {?} */ (this)))))._results)))[getSymbolIterator$1()]();
    }
    /**
     * An unmodifiable list of items that Angular keeps up to date when the state
     * of the application changes.
     *
     * The type of object that {\@link ViewChildren}, {\@link ContentChildren}, and {\@link QueryList}
     * provide.
     *
     * Implements an iterable interface, therefore it can be used in both ES6
     * javascript `for (var i of items)` loops as well as in Angular templates with
     * `*ngFor="let i of myList"`.
     *
     * Changes can be observed by subscribing to the changes `Observable`.
     *
     * NOTE: In the future this class will implement an `Observable` interface.
     *
     * \@usageNotes
     * ### Example
     * ```typescript
     * \@Component({...})
     * class Container {
     * \@ViewChildren(Item) items:QueryList<Item>;
     * }
     * ```
     *
     * \@publicApi
     * @template T
     */
    class QueryList {
        constructor() {
            this.dirty = true;
            this._results = [];
            this.changes = new EventEmitter();
            this.length = 0;
            // This function should be declared on the prototype, but doing so there will cause the class
            // declaration to have side-effects and become not tree-shakable. For this reason we do it in
            // the constructor.
            // [getSymbolIterator()](): Iterator<T> { ... }
            /** @type {?} */
            const symbol = getSymbolIterator$1();
            /** @type {?} */
            const proto = (/** @type {?} */ (QueryList.prototype));
            if (!proto[symbol])
                proto[symbol] = symbolIterator;
        }
        /**
         * See
         * [Array.map](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/map)
         * @template U
         * @param {?} fn
         * @return {?}
         */
        map(fn) {
            return this._results.map(fn);
        }
        /**
         * See
         * [Array.filter](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/filter)
         * @param {?} fn
         * @return {?}
         */
        filter(fn) {
            return this._results.filter(fn);
        }
        /**
         * See
         * [Array.find](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/find)
         * @param {?} fn
         * @return {?}
         */
        find(fn) {
            return this._results.find(fn);
        }
        /**
         * See
         * [Array.reduce](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/reduce)
         * @template U
         * @param {?} fn
         * @param {?} init
         * @return {?}
         */
        reduce(fn, init) {
            return this._results.reduce(fn, init);
        }
        /**
         * See
         * [Array.forEach](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/forEach)
         * @param {?} fn
         * @return {?}
         */
        forEach(fn) {
            this._results.forEach(fn);
        }
        /**
         * See
         * [Array.some](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/some)
         * @param {?} fn
         * @return {?}
         */
        some(fn) {
            return this._results.some(fn);
        }
        /**
         * Returns a copy of the internal results list as an Array.
         * @return {?}
         */
        toArray() {
            return this._results.slice();
        }
        /**
         * @return {?}
         */
        toString() {
            return this._results.toString();
        }
        /**
         * Updates the stored data of the query list, and resets the `dirty` flag to `false`, so that
         * on change detection, it will not notify of changes to the queries, unless a new change
         * occurs.
         *
         * @param {?} resultsTree The query results to store
         * @return {?}
         */
        reset(resultsTree) {
            this._results = flatten(resultsTree);
            ((/** @type {?} */ (this))).dirty = false;
            ((/** @type {?} */ (this))).length = this._results.length;
            ((/** @type {?} */ (this))).last = this._results[this.length - 1];
            ((/** @type {?} */ (this))).first = this._results[0];
        }
        /**
         * Triggers a change event by emitting on the `changes` {\@link EventEmitter}.
         * @return {?}
         */
        notifyOnChanges() {
            ((/** @type {?} */ (this.changes))).emit(this);
        }
        /**
         * internal
         * @return {?}
         */
        setDirty() {
            ((/** @type {?} */ (this))).dirty = true;
        }
        /**
         * internal
         * @return {?}
         */
        destroy() {
            ((/** @type {?} */ (this.changes))).complete();
            ((/** @type {?} */ (this.changes))).unsubscribe();
        }
    }
    /**
     * @template T
     */
    class LQuery_ {
        /**
         * @param {?} queryList
         */
        constructor(queryList) {
            this.queryList = queryList;
            this.matches = null;
        }
        /**
         * @return {?}
         */
        clone() {
            return new LQuery_(this.queryList);
        }
        /**
         * @return {?}
         */
        setDirty() {
            this.queryList.setDirty();
        }
    }
    class LQueries_ {
        /**
         * @param {?=} queries
         */
        constructor(queries = []) {
            this.queries = queries;
        }
        /**
         * @param {?} tView
         * @return {?}
         */
        createEmbeddedView(tView) {
            /** @type {?} */
            const tQueries = tView.queries;
            if (tQueries !== null) {
                /** @type {?} */
                const noOfInheritedQueries = tView.contentQueries !== null ? tView.contentQueries[0] : tQueries.length;
                /** @type {?} */
                const viewLQueries = [];
                // An embedded view has queries propagated from a declaration view at the beginning of the
                // TQueries collection and up until a first content query declared in the embedded view. Only
                // propagated LQueries are created at this point (LQuery corresponding to declared content
                // queries will be instantiated from the content query instructions for each directive).
                for (let i = 0; i < noOfInheritedQueries; i++) {
                    /** @type {?} */
                    const tQuery = tQueries.getByIndex(i);
                    /** @type {?} */
                    const parentLQuery = this.queries[tQuery.indexInDeclarationView];
                    viewLQueries.push(parentLQuery.clone());
                }
                return new LQueries_(viewLQueries);
            }
            return null;
        }
        /**
         * @param {?} tView
         * @return {?}
         */
        insertView(tView) {
            this.dirtyQueriesWithMatches(tView);
        }
        /**
         * @param {?} tView
         * @return {?}
         */
        detachView(tView) {
            this.dirtyQueriesWithMatches(tView);
        }
        /**
         * @private
         * @param {?} tView
         * @return {?}
         */
        dirtyQueriesWithMatches(tView) {
            for (let i = 0; i < this.queries.length; i++) {
                if (getTQuery(tView, i).matches !== null) {
                    this.queries[i].setDirty();
                }
            }
        }
    }
    class TQueryMetadata_ {
        /**
         * @param {?} predicate
         * @param {?} descendants
         * @param {?} isStatic
         * @param {?=} read
         */
        constructor(predicate, descendants, isStatic, read = null) {
            this.predicate = predicate;
            this.descendants = descendants;
            this.isStatic = isStatic;
            this.read = read;
        }
    }
    class TQueries_ {
        /**
         * @param {?=} queries
         */
        constructor(queries = []) {
            this.queries = queries;
        }
        /**
         * @param {?} tView
         * @param {?} tNode
         * @return {?}
         */
        elementStart(tView, tNode) {
            ngDevMode &&
                assertFirstCreatePass(tView, 'Queries should collect results on the first template pass only');
            for (let i = 0; i < this.queries.length; i++) {
                this.queries[i].elementStart(tView, tNode);
            }
        }
        /**
         * @param {?} tNode
         * @return {?}
         */
        elementEnd(tNode) {
            for (let i = 0; i < this.queries.length; i++) {
                this.queries[i].elementEnd(tNode);
            }
        }
        /**
         * @param {?} tNode
         * @return {?}
         */
        embeddedTView(tNode) {
            /** @type {?} */
            let queriesForTemplateRef = null;
            for (let i = 0; i < this.length; i++) {
                /** @type {?} */
                const childQueryIndex = queriesForTemplateRef !== null ? queriesForTemplateRef.length : 0;
                /** @type {?} */
                const tqueryClone = this.getByIndex(i).embeddedTView(tNode, childQueryIndex);
                if (tqueryClone) {
                    tqueryClone.indexInDeclarationView = i;
                    if (queriesForTemplateRef !== null) {
                        queriesForTemplateRef.push(tqueryClone);
                    }
                    else {
                        queriesForTemplateRef = [tqueryClone];
                    }
                }
            }
            return queriesForTemplateRef !== null ? new TQueries_(queriesForTemplateRef) : null;
        }
        /**
         * @param {?} tView
         * @param {?} tNode
         * @return {?}
         */
        template(tView, tNode) {
            ngDevMode &&
                assertFirstCreatePass(tView, 'Queries should collect results on the first template pass only');
            for (let i = 0; i < this.queries.length; i++) {
                this.queries[i].template(tView, tNode);
            }
        }
        /**
         * @param {?} index
         * @return {?}
         */
        getByIndex(index) {
            ngDevMode && assertDataInRange(this.queries, index);
            return this.queries[index];
        }
        /**
         * @return {?}
         */
        get length() {
            return this.queries.length;
        }
        /**
         * @param {?} tquery
         * @return {?}
         */
        track(tquery) {
            this.queries.push(tquery);
        }
    }
    class TQuery_ {
        /**
         * @param {?} metadata
         * @param {?=} nodeIndex
         */
        constructor(metadata, nodeIndex = -1) {
            this.metadata = metadata;
            this.matches = null;
            this.indexInDeclarationView = -1;
            this.crossesNgTemplate = false;
            /**
             * A flag indicating if a given query still applies to nodes it is crossing. We use this flag
             * (alongside with _declarationNodeIndex) to know when to stop applying content queries to
             * elements in a template.
             */
            this._appliesToNextNode = true;
            this._declarationNodeIndex = nodeIndex;
        }
        /**
         * @param {?} tView
         * @param {?} tNode
         * @return {?}
         */
        elementStart(tView, tNode) {
            if (this.isApplyingToNode(tNode)) {
                this.matchTNode(tView, tNode);
            }
        }
        /**
         * @param {?} tNode
         * @return {?}
         */
        elementEnd(tNode) {
            if (this._declarationNodeIndex === tNode.index) {
                this._appliesToNextNode = false;
            }
        }
        /**
         * @param {?} tView
         * @param {?} tNode
         * @return {?}
         */
        template(tView, tNode) {
            this.elementStart(tView, tNode);
        }
        /**
         * @param {?} tNode
         * @param {?} childQueryIndex
         * @return {?}
         */
        embeddedTView(tNode, childQueryIndex) {
            if (this.isApplyingToNode(tNode)) {
                this.crossesNgTemplate = true;
                // A marker indicating a `<ng-template>` element (a placeholder for query results from
                // embedded views created based on this `<ng-template>`).
                this.addMatch(-tNode.index, childQueryIndex);
                return new TQuery_(this.metadata);
            }
            return null;
        }
        /**
         * @private
         * @param {?} tNode
         * @return {?}
         */
        isApplyingToNode(tNode) {
            if (this._appliesToNextNode && this.metadata.descendants === false) {
                /** @type {?} */
                const declarationNodeIdx = this._declarationNodeIndex;
                /** @type {?} */
                let parent = tNode.parent;
                // Determine if a given TNode is a "direct" child of a node on which a content query was
                // declared (only direct children of query's host node can match with the descendants: false
                // option). There are 3 main use-case / conditions to consider here:
                // - <needs-target><i #target></i></needs-target>: here <i #target> parent node is a query
                // host node;
                // - <needs-target><ng-template [ngIf]="true"><i #target></i></ng-template></needs-target>:
                // here <i #target> parent node is null;
                // - <needs-target><ng-container><i #target></i></ng-container></needs-target>: here we need
                // to go past `<ng-container>` to determine <i #target> parent node (but we shouldn't traverse
                // up past the query's host node!).
                while (parent !== null && parent.type === 4 /* ElementContainer */ &&
                    parent.index !== declarationNodeIdx) {
                    parent = parent.parent;
                }
                return declarationNodeIdx === (parent !== null ? parent.index : -1);
            }
            return this._appliesToNextNode;
        }
        /**
         * @private
         * @param {?} tView
         * @param {?} tNode
         * @return {?}
         */
        matchTNode(tView, tNode) {
            if (Array.isArray(this.metadata.predicate)) {
                /** @type {?} */
                const localNames = this.metadata.predicate;
                for (let i = 0; i < localNames.length; i++) {
                    this.matchTNodeWithReadOption(tView, tNode, getIdxOfMatchingSelector(tNode, localNames[i]));
                }
            }
            else {
                /** @type {?} */
                const typePredicate = (/** @type {?} */ (this.metadata.predicate));
                if (typePredicate === TemplateRef) {
                    if (tNode.type === 0 /* Container */) {
                        this.matchTNodeWithReadOption(tView, tNode, -1);
                    }
                }
                else {
                    this.matchTNodeWithReadOption(tView, tNode, locateDirectiveOrProvider(tNode, tView, typePredicate, false, false));
                }
            }
        }
        /**
         * @private
         * @param {?} tView
         * @param {?} tNode
         * @param {?} nodeMatchIdx
         * @return {?}
         */
        matchTNodeWithReadOption(tView, tNode, nodeMatchIdx) {
            if (nodeMatchIdx !== null) {
                /** @type {?} */
                const read = this.metadata.read;
                if (read !== null) {
                    if (read === ElementRef || read === ViewContainerRef ||
                        read === TemplateRef && tNode.type === 0 /* Container */) {
                        this.addMatch(tNode.index, -2);
                    }
                    else {
                        /** @type {?} */
                        const directiveOrProviderIdx = locateDirectiveOrProvider(tNode, tView, read, false, false);
                        if (directiveOrProviderIdx !== null) {
                            this.addMatch(tNode.index, directiveOrProviderIdx);
                        }
                    }
                }
                else {
                    this.addMatch(tNode.index, nodeMatchIdx);
                }
            }
        }
        /**
         * @private
         * @param {?} tNodeIdx
         * @param {?} matchIdx
         * @return {?}
         */
        addMatch(tNodeIdx, matchIdx) {
            if (this.matches === null) {
                this.matches = [tNodeIdx, matchIdx];
            }
            else {
                this.matches.push(tNodeIdx, matchIdx);
            }
        }
    }
    /**
     * Iterates over local names for a given node and returns directive index
     * (or -1 if a local name points to an element).
     *
     * @param {?} tNode static data of a node to check
     * @param {?} selector selector to match
     * @return {?} directive index, -1 or null if a selector didn't match any of the local names
     */
    function getIdxOfMatchingSelector(tNode, selector) {
        /** @type {?} */
        const localNames = tNode.localNames;
        if (localNames !== null) {
            for (let i = 0; i < localNames.length; i += 2) {
                if (localNames[i] === selector) {
                    return (/** @type {?} */ (localNames[i + 1]));
                }
            }
        }
        return null;
    }
    /**
     * @param {?} tNode
     * @param {?} currentView
     * @return {?}
     */
    function createResultByTNodeType(tNode, currentView) {
        if (tNode.type === 3 /* Element */ || tNode.type === 4 /* ElementContainer */) {
            return createElementRef(ElementRef, tNode, currentView);
        }
        else if (tNode.type === 0 /* Container */) {
            return createTemplateRef(TemplateRef, ElementRef, tNode, currentView);
        }
        return null;
    }
    /**
     * @param {?} lView
     * @param {?} tNode
     * @param {?} matchingIdx
     * @param {?} read
     * @return {?}
     */
    function createResultForNode(lView, tNode, matchingIdx, read) {
        if (matchingIdx === -1) {
            // if read token and / or strategy is not specified, detect it using appropriate tNode type
            return createResultByTNodeType(tNode, lView);
        }
        else if (matchingIdx === -2) {
            // read a special token from a node injector
            return createSpecialToken(lView, tNode, read);
        }
        else {
            // read a token
            return getNodeInjectable(lView, lView[TVIEW], matchingIdx, (/** @type {?} */ (tNode)));
        }
    }
    /**
     * @param {?} lView
     * @param {?} tNode
     * @param {?} read
     * @return {?}
     */
    function createSpecialToken(lView, tNode, read) {
        if (read === ElementRef) {
            return createElementRef(ElementRef, tNode, lView);
        }
        else if (read === TemplateRef) {
            return createTemplateRef(TemplateRef, ElementRef, tNode, lView);
        }
        else if (read === ViewContainerRef) {
            ngDevMode &&
                assertNodeOfPossibleTypes(tNode, 3 /* Element */, 0 /* Container */, 4 /* ElementContainer */);
            return createContainerRef(ViewContainerRef, ElementRef, (/** @type {?} */ (tNode)), lView);
        }
        else {
            ngDevMode &&
                throwError$1(`Special token to read should be one of ElementRef, TemplateRef or ViewContainerRef but got ${stringify(read)}.`);
        }
    }
    /**
     * A helper function that creates query results for a given view. This function is meant to do the
     * processing once and only once for a given view instance (a set of results for a given view
     * doesn't change).
     * @template T
     * @param {?} tView
     * @param {?} lView
     * @param {?} tQuery
     * @param {?} queryIndex
     * @return {?}
     */
    function materializeViewResults(tView, lView, tQuery, queryIndex) {
        /** @type {?} */
        const lQuery = (/** @type {?} */ ((/** @type {?} */ (lView[QUERIES])).queries))[queryIndex];
        if (lQuery.matches === null) {
            /** @type {?} */
            const tViewData = tView.data;
            /** @type {?} */
            const tQueryMatches = (/** @type {?} */ (tQuery.matches));
            /** @type {?} */
            const result = [];
            for (let i = 0; i < tQueryMatches.length; i += 2) {
                /** @type {?} */
                const matchedNodeIdx = tQueryMatches[i];
                if (matchedNodeIdx < 0) {
                    // we at the <ng-template> marker which might have results in views created based on this
                    // <ng-template> - those results will be in separate views though, so here we just leave
                    // null as a placeholder
                    result.push(null);
                }
                else {
                    ngDevMode && assertDataInRange(tViewData, matchedNodeIdx);
                    /** @type {?} */
                    const tNode = (/** @type {?} */ (tViewData[matchedNodeIdx]));
                    result.push(createResultForNode(lView, tNode, tQueryMatches[i + 1], tQuery.metadata.read));
                }
            }
            lQuery.matches = result;
        }
        return lQuery.matches;
    }
    /**
     * A helper function that collects (already materialized) query results from a tree of views,
     * starting with a provided LView.
     * @template T
     * @param {?} tView
     * @param {?} lView
     * @param {?} queryIndex
     * @param {?} result
     * @return {?}
     */
    function collectQueryResults(tView, lView, queryIndex, result) {
        /** @type {?} */
        const tQuery = (/** @type {?} */ (tView.queries)).getByIndex(queryIndex);
        /** @type {?} */
        const tQueryMatches = tQuery.matches;
        if (tQueryMatches !== null) {
            /** @type {?} */
            const lViewResults = materializeViewResults(tView, lView, tQuery, queryIndex);
            for (let i = 0; i < tQueryMatches.length; i += 2) {
                /** @type {?} */
                const tNodeIdx = tQueryMatches[i];
                if (tNodeIdx > 0) {
                    result.push((/** @type {?} */ (lViewResults[i / 2])));
                }
                else {
                    /** @type {?} */
                    const childQueryIndex = tQueryMatches[i + 1];
                    /** @type {?} */
                    const declarationLContainer = (/** @type {?} */ (lView[-tNodeIdx]));
                    ngDevMode && assertLContainer(declarationLContainer);
                    // collect matches for views inserted in this container
                    for (let i = CONTAINER_HEADER_OFFSET; i < declarationLContainer.length; i++) {
                        /** @type {?} */
                        const embeddedLView = declarationLContainer[i];
                        if (embeddedLView[DECLARATION_LCONTAINER] === embeddedLView[PARENT]) {
                            collectQueryResults(embeddedLView[TVIEW], embeddedLView, childQueryIndex, result);
                        }
                    }
                    // collect matches for views created from this declaration container and inserted into
                    // different containers
                    if (declarationLContainer[MOVED_VIEWS] !== null) {
                        /** @type {?} */
                        const embeddedLViews = (/** @type {?} */ (declarationLContainer[MOVED_VIEWS]));
                        for (let i = 0; i < embeddedLViews.length; i++) {
                            /** @type {?} */
                            const embeddedLView = embeddedLViews[i];
                            collectQueryResults(embeddedLView[TVIEW], embeddedLView, childQueryIndex, result);
                        }
                    }
                }
            }
        }
        return result;
    }
    /**
     * Refreshes a query by combining matches from all active views and removing matches from deleted
     * views.
     *
     * \@codeGenApi
     * @param {?} queryList
     * @return {?} `true` if a query got dirty during change detection or if this is a static query
     * resolving in creation mode, `false` otherwise.
     *
     */
    function ɵɵqueryRefresh(queryList) {
        /** @type {?} */
        const lView = getLView();
        /** @type {?} */
        const tView = getTView();
        /** @type {?} */
        const queryIndex = getCurrentQueryIndex();
        setCurrentQueryIndex(queryIndex + 1);
        /** @type {?} */
        const tQuery = getTQuery(tView, queryIndex);
        if (queryList.dirty && (isCreationMode(lView) === tQuery.metadata.isStatic)) {
            if (tQuery.matches === null) {
                queryList.reset([]);
            }
            else {
                /** @type {?} */
                const result = tQuery.crossesNgTemplate ?
                    collectQueryResults(tView, lView, queryIndex, []) :
                    materializeViewResults(tView, lView, tQuery, queryIndex);
                queryList.reset(result);
                queryList.notifyOnChanges();
            }
            return true;
        }
        return false;
    }
    /**
     * Creates new QueryList for a static view query.
     *
     * \@codeGenApi
     * @template T
     * @param {?} predicate The type for which the query will search
     * @param {?} descend Whether or not to descend into children
     * @param {?=} read What to save in the query
     *
     * @return {?}
     */
    function ɵɵstaticViewQuery(predicate, descend, read) {
        viewQueryInternal(getTView(), getLView(), predicate, descend, read, true);
    }
    /**
     * Creates new QueryList, stores the reference in LView and returns QueryList.
     *
     * \@codeGenApi
     * @template T
     * @param {?} predicate The type for which the query will search
     * @param {?} descend Whether or not to descend into children
     * @param {?=} read What to save in the query
     *
     * @return {?}
     */
    function ɵɵviewQuery(predicate, descend, read) {
        viewQueryInternal(getTView(), getLView(), predicate, descend, read, false);
    }
    /**
     * @template T
     * @param {?} tView
     * @param {?} lView
     * @param {?} predicate
     * @param {?} descend
     * @param {?} read
     * @param {?} isStatic
     * @return {?}
     */
    function viewQueryInternal(tView, lView, predicate, descend, read, isStatic) {
        if (tView.firstCreatePass) {
            createTQuery(tView, new TQueryMetadata_(predicate, descend, isStatic, read), -1);
            if (isStatic) {
                tView.staticViewQueries = true;
            }
        }
        createLQuery(tView, lView);
    }
    /**
     * Registers a QueryList, associated with a content query, for later refresh (part of a view
     * refresh).
     *
     * \@codeGenApi
     * @template T
     * @param {?} directiveIndex Current directive index
     * @param {?} predicate The type for which the query will search
     * @param {?} descend Whether or not to descend into children
     * @param {?=} read What to save in the query
     * @return {?} QueryList<T>
     *
     */
    function ɵɵcontentQuery(directiveIndex, predicate, descend, read) {
        contentQueryInternal(getTView(), getLView(), predicate, descend, read, false, getPreviousOrParentTNode(), directiveIndex);
    }
    /**
     * Registers a QueryList, associated with a static content query, for later refresh
     * (part of a view refresh).
     *
     * \@codeGenApi
     * @template T
     * @param {?} directiveIndex Current directive index
     * @param {?} predicate The type for which the query will search
     * @param {?} descend Whether or not to descend into children
     * @param {?=} read What to save in the query
     * @return {?} QueryList<T>
     *
     */
    function ɵɵstaticContentQuery(directiveIndex, predicate, descend, read) {
        contentQueryInternal(getTView(), getLView(), predicate, descend, read, true, getPreviousOrParentTNode(), directiveIndex);
    }
    /**
     * @template T
     * @param {?} tView
     * @param {?} lView
     * @param {?} predicate
     * @param {?} descend
     * @param {?} read
     * @param {?} isStatic
     * @param {?} tNode
     * @param {?} directiveIndex
     * @return {?}
     */
    function contentQueryInternal(tView, lView, predicate, descend, read, isStatic, tNode, directiveIndex) {
        if (tView.firstCreatePass) {
            createTQuery(tView, new TQueryMetadata_(predicate, descend, isStatic, read), tNode.index);
            saveContentQueryAndDirectiveIndex(tView, directiveIndex);
            if (isStatic) {
                tView.staticContentQueries = true;
            }
        }
        createLQuery(tView, lView);
    }
    /**
     * Loads a QueryList corresponding to the current view or content query.
     *
     * \@codeGenApi
     * @template T
     * @return {?}
     */
    function ɵɵloadQuery() {
        return loadQueryInternal(getLView(), getCurrentQueryIndex());
    }
    /**
     * @template T
     * @param {?} lView
     * @param {?} queryIndex
     * @return {?}
     */
    function loadQueryInternal(lView, queryIndex) {
        ngDevMode &&
            assertDefined(lView[QUERIES], 'LQueries should be defined when trying to load a query');
        ngDevMode && assertDataInRange((/** @type {?} */ (lView[QUERIES])).queries, queryIndex);
        return (/** @type {?} */ (lView[QUERIES])).queries[queryIndex].queryList;
    }
    /**
     * @template T
     * @param {?} tView
     * @param {?} lView
     * @return {?}
     */
    function createLQuery(tView, lView) {
        /** @type {?} */
        const queryList = new QueryList();
        storeCleanupWithContext(tView, lView, queryList, queryList.destroy);
        if (lView[QUERIES] === null)
            lView[QUERIES] = new LQueries_();
        (/** @type {?} */ (lView[QUERIES])).queries.push(new LQuery_(queryList));
    }
    /**
     * @param {?} tView
     * @param {?} metadata
     * @param {?} nodeIndex
     * @return {?}
     */
    function createTQuery(tView, metadata, nodeIndex) {
        if (tView.queries === null)
            tView.queries = new TQueries_();
        tView.queries.track(new TQuery_(metadata, nodeIndex));
    }
    /**
     * @param {?} tView
     * @param {?} directiveIndex
     * @return {?}
     */
    function saveContentQueryAndDirectiveIndex(tView, directiveIndex) {
        /** @type {?} */
        const tViewContentQueries = tView.contentQueries || (tView.contentQueries = []);
        /** @type {?} */
        const lastSavedDirectiveIndex = tView.contentQueries.length ? tViewContentQueries[tViewContentQueries.length - 1] : -1;
        if (directiveIndex !== lastSavedDirectiveIndex) {
            tViewContentQueries.push((/** @type {?} */ (tView.queries)).length - 1, directiveIndex);
        }
    }
    /**
     * @param {?} tView
     * @param {?} index
     * @return {?}
     */
    function getTQuery(tView, index) {
        ngDevMode && assertDefined(tView.queries, 'TQueries must be defined to retrieve a TQuery');
        return (/** @type {?} */ (tView.queries)).getByIndex(index);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/view_engine_compatibility_prebound.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Retrieves `TemplateRef` instance from `Injector` when a local reference is placed on the
     * `<ng-template>` element.
     *
     * \@codeGenApi
     * @param {?} tNode
     * @param {?} currentView
     * @return {?}
     */
    function ɵɵtemplateRefExtractor(tNode, currentView) {
        return createTemplateRef(TemplateRef, ElementRef, tNode, currentView);
    }
    /**
     * Returns the appropriate `ChangeDetectorRef` for a pipe.
     *
     * \@codeGenApi
     * @param {?=} flags
     * @return {?}
     */
    function ɵɵinjectPipeChangeDetectorRef(flags = InjectFlags.Default) {
        /** @type {?} */
        const value = injectChangeDetectorRef(true);
        if (value == null && !(flags & InjectFlags.Optional)) {
            throw new Error(`No provider for ChangeDetectorRef!`);
        }
        else {
            return value;
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/index.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/jit/environment.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    const ɵ0$d = /**
     * @return {?}
     */
    () => ({
        'ɵɵattribute': ɵɵattribute,
        'ɵɵattributeInterpolate1': ɵɵattributeInterpolate1,
        'ɵɵattributeInterpolate2': ɵɵattributeInterpolate2,
        'ɵɵattributeInterpolate3': ɵɵattributeInterpolate3,
        'ɵɵattributeInterpolate4': ɵɵattributeInterpolate4,
        'ɵɵattributeInterpolate5': ɵɵattributeInterpolate5,
        'ɵɵattributeInterpolate6': ɵɵattributeInterpolate6,
        'ɵɵattributeInterpolate7': ɵɵattributeInterpolate7,
        'ɵɵattributeInterpolate8': ɵɵattributeInterpolate8,
        'ɵɵattributeInterpolateV': ɵɵattributeInterpolateV,
        'ɵɵdefineComponent': ɵɵdefineComponent,
        'ɵɵdefineDirective': ɵɵdefineDirective,
        'ɵɵdefineInjectable': ɵɵdefineInjectable,
        'ɵɵdefineInjector': ɵɵdefineInjector,
        'ɵɵdefineNgModule': ɵɵdefineNgModule,
        'ɵɵdefinePipe': ɵɵdefinePipe,
        'ɵɵdirectiveInject': ɵɵdirectiveInject,
        'ɵɵgetFactoryOf': ɵɵgetFactoryOf,
        'ɵɵgetInheritedFactory': ɵɵgetInheritedFactory,
        'ɵɵinject': ɵɵinject,
        'ɵɵinjectAttribute': ɵɵinjectAttribute,
        'ɵɵinvalidFactory': ɵɵinvalidFactory,
        'ɵɵinvalidFactoryDep': ɵɵinvalidFactoryDep,
        'ɵɵinjectPipeChangeDetectorRef': ɵɵinjectPipeChangeDetectorRef,
        'ɵɵtemplateRefExtractor': ɵɵtemplateRefExtractor,
        'ɵɵNgOnChangesFeature': ɵɵNgOnChangesFeature,
        'ɵɵProvidersFeature': ɵɵProvidersFeature,
        'ɵɵCopyDefinitionFeature': ɵɵCopyDefinitionFeature,
        'ɵɵInheritDefinitionFeature': ɵɵInheritDefinitionFeature,
        'ɵɵnextContext': ɵɵnextContext,
        'ɵɵnamespaceHTML': ɵɵnamespaceHTML,
        'ɵɵnamespaceMathML': ɵɵnamespaceMathML,
        'ɵɵnamespaceSVG': ɵɵnamespaceSVG,
        'ɵɵenableBindings': ɵɵenableBindings,
        'ɵɵdisableBindings': ɵɵdisableBindings,
        'ɵɵelementStart': ɵɵelementStart,
        'ɵɵelementEnd': ɵɵelementEnd,
        'ɵɵelement': ɵɵelement,
        'ɵɵelementContainerStart': ɵɵelementContainerStart,
        'ɵɵelementContainerEnd': ɵɵelementContainerEnd,
        'ɵɵelementContainer': ɵɵelementContainer,
        'ɵɵpureFunction0': ɵɵpureFunction0,
        'ɵɵpureFunction1': ɵɵpureFunction1,
        'ɵɵpureFunction2': ɵɵpureFunction2,
        'ɵɵpureFunction3': ɵɵpureFunction3,
        'ɵɵpureFunction4': ɵɵpureFunction4,
        'ɵɵpureFunction5': ɵɵpureFunction5,
        'ɵɵpureFunction6': ɵɵpureFunction6,
        'ɵɵpureFunction7': ɵɵpureFunction7,
        'ɵɵpureFunction8': ɵɵpureFunction8,
        'ɵɵpureFunctionV': ɵɵpureFunctionV,
        'ɵɵgetCurrentView': ɵɵgetCurrentView,
        'ɵɵrestoreView': ɵɵrestoreView,
        'ɵɵlistener': ɵɵlistener,
        'ɵɵprojection': ɵɵprojection,
        'ɵɵupdateSyntheticHostBinding': ɵɵupdateSyntheticHostBinding,
        'ɵɵcomponentHostSyntheticListener': ɵɵcomponentHostSyntheticListener,
        'ɵɵpipeBind1': ɵɵpipeBind1,
        'ɵɵpipeBind2': ɵɵpipeBind2,
        'ɵɵpipeBind3': ɵɵpipeBind3,
        'ɵɵpipeBind4': ɵɵpipeBind4,
        'ɵɵpipeBindV': ɵɵpipeBindV,
        'ɵɵprojectionDef': ɵɵprojectionDef,
        'ɵɵhostProperty': ɵɵhostProperty,
        'ɵɵproperty': ɵɵproperty,
        'ɵɵpropertyInterpolate': ɵɵpropertyInterpolate,
        'ɵɵpropertyInterpolate1': ɵɵpropertyInterpolate1,
        'ɵɵpropertyInterpolate2': ɵɵpropertyInterpolate2,
        'ɵɵpropertyInterpolate3': ɵɵpropertyInterpolate3,
        'ɵɵpropertyInterpolate4': ɵɵpropertyInterpolate4,
        'ɵɵpropertyInterpolate5': ɵɵpropertyInterpolate5,
        'ɵɵpropertyInterpolate6': ɵɵpropertyInterpolate6,
        'ɵɵpropertyInterpolate7': ɵɵpropertyInterpolate7,
        'ɵɵpropertyInterpolate8': ɵɵpropertyInterpolate8,
        'ɵɵpropertyInterpolateV': ɵɵpropertyInterpolateV,
        'ɵɵpipe': ɵɵpipe,
        'ɵɵqueryRefresh': ɵɵqueryRefresh,
        'ɵɵviewQuery': ɵɵviewQuery,
        'ɵɵstaticViewQuery': ɵɵstaticViewQuery,
        'ɵɵstaticContentQuery': ɵɵstaticContentQuery,
        'ɵɵloadQuery': ɵɵloadQuery,
        'ɵɵcontentQuery': ɵɵcontentQuery,
        'ɵɵreference': ɵɵreference,
        'ɵɵclassMap': ɵɵclassMap,
        'ɵɵclassMapInterpolate1': ɵɵclassMapInterpolate1,
        'ɵɵclassMapInterpolate2': ɵɵclassMapInterpolate2,
        'ɵɵclassMapInterpolate3': ɵɵclassMapInterpolate3,
        'ɵɵclassMapInterpolate4': ɵɵclassMapInterpolate4,
        'ɵɵclassMapInterpolate5': ɵɵclassMapInterpolate5,
        'ɵɵclassMapInterpolate6': ɵɵclassMapInterpolate6,
        'ɵɵclassMapInterpolate7': ɵɵclassMapInterpolate7,
        'ɵɵclassMapInterpolate8': ɵɵclassMapInterpolate8,
        'ɵɵclassMapInterpolateV': ɵɵclassMapInterpolateV,
        'ɵɵstyleMap': ɵɵstyleMap,
        'ɵɵstyleMapInterpolate1': ɵɵstyleMapInterpolate1,
        'ɵɵstyleMapInterpolate2': ɵɵstyleMapInterpolate2,
        'ɵɵstyleMapInterpolate3': ɵɵstyleMapInterpolate3,
        'ɵɵstyleMapInterpolate4': ɵɵstyleMapInterpolate4,
        'ɵɵstyleMapInterpolate5': ɵɵstyleMapInterpolate5,
        'ɵɵstyleMapInterpolate6': ɵɵstyleMapInterpolate6,
        'ɵɵstyleMapInterpolate7': ɵɵstyleMapInterpolate7,
        'ɵɵstyleMapInterpolate8': ɵɵstyleMapInterpolate8,
        'ɵɵstyleMapInterpolateV': ɵɵstyleMapInterpolateV,
        'ɵɵstyleProp': ɵɵstyleProp,
        'ɵɵstylePropInterpolate1': ɵɵstylePropInterpolate1,
        'ɵɵstylePropInterpolate2': ɵɵstylePropInterpolate2,
        'ɵɵstylePropInterpolate3': ɵɵstylePropInterpolate3,
        'ɵɵstylePropInterpolate4': ɵɵstylePropInterpolate4,
        'ɵɵstylePropInterpolate5': ɵɵstylePropInterpolate5,
        'ɵɵstylePropInterpolate6': ɵɵstylePropInterpolate6,
        'ɵɵstylePropInterpolate7': ɵɵstylePropInterpolate7,
        'ɵɵstylePropInterpolate8': ɵɵstylePropInterpolate8,
        'ɵɵstylePropInterpolateV': ɵɵstylePropInterpolateV,
        'ɵɵstyleSanitizer': ɵɵstyleSanitizer,
        'ɵɵclassProp': ɵɵclassProp,
        'ɵɵselect': ɵɵselect,
        'ɵɵadvance': ɵɵadvance,
        'ɵɵtemplate': ɵɵtemplate,
        'ɵɵtext': ɵɵtext,
        'ɵɵtextInterpolate': ɵɵtextInterpolate,
        'ɵɵtextInterpolate1': ɵɵtextInterpolate1,
        'ɵɵtextInterpolate2': ɵɵtextInterpolate2,
        'ɵɵtextInterpolate3': ɵɵtextInterpolate3,
        'ɵɵtextInterpolate4': ɵɵtextInterpolate4,
        'ɵɵtextInterpolate5': ɵɵtextInterpolate5,
        'ɵɵtextInterpolate6': ɵɵtextInterpolate6,
        'ɵɵtextInterpolate7': ɵɵtextInterpolate7,
        'ɵɵtextInterpolate8': ɵɵtextInterpolate8,
        'ɵɵtextInterpolateV': ɵɵtextInterpolateV,
        'ɵɵi18n': ɵɵi18n,
        'ɵɵi18nAttributes': ɵɵi18nAttributes,
        'ɵɵi18nExp': ɵɵi18nExp,
        'ɵɵi18nStart': ɵɵi18nStart,
        'ɵɵi18nEnd': ɵɵi18nEnd,
        'ɵɵi18nApply': ɵɵi18nApply,
        'ɵɵi18nPostprocess': ɵɵi18nPostprocess,
        'ɵɵresolveWindow': ɵɵresolveWindow,
        'ɵɵresolveDocument': ɵɵresolveDocument,
        'ɵɵresolveBody': ɵɵresolveBody,
        'ɵɵsetComponentScope': ɵɵsetComponentScope,
        'ɵɵsetNgModuleScope': ɵɵsetNgModuleScope,
        'ɵɵsanitizeHtml': ɵɵsanitizeHtml,
        'ɵɵsanitizeStyle': ɵɵsanitizeStyle,
        'ɵɵdefaultStyleSanitizer': ɵɵdefaultStyleSanitizer,
        'ɵɵsanitizeResourceUrl': ɵɵsanitizeResourceUrl,
        'ɵɵsanitizeScript': ɵɵsanitizeScript,
        'ɵɵsanitizeUrl': ɵɵsanitizeUrl,
        'ɵɵsanitizeUrlOrResourceUrl': ɵɵsanitizeUrlOrResourceUrl,
    });
    /**
     * A mapping of the \@angular/core API surface used in generated expressions to the actual symbols.
     *
     * This should be kept up to date with the public exports of \@angular/core.
     * @type {?}
     */
    const angularCoreEnv = ((ɵ0$d))();
    /** @type {?} */
    let jitOptions = null;
    /**
     * @param {?} options
     * @return {?}
     */
    function setJitOptions(options) {
        if (jitOptions !== null) {
            if (options.defaultEncapsulation !== jitOptions.defaultEncapsulation) {
                ngDevMode &&
                    console.error('Provided value for `defaultEncapsulation` can not be changed once it has been set.');
                return;
            }
            if (options.preserveWhitespaces !== jitOptions.preserveWhitespaces) {
                ngDevMode &&
                    console.error('Provided value for `preserveWhitespaces` can not be changed once it has been set.');
                return;
            }
        }
        jitOptions = options;
    }
    /**
     * @return {?}
     */
    function getJitOptions() {
        return jitOptions;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/jit/module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const EMPTY_ARRAY$5 = [];
    /** @type {?} */
    const moduleQueue = [];
    /**
     * Enqueues moduleDef to be checked later to see if scope can be set on its
     * component declarations.
     * @param {?} moduleType
     * @param {?} ngModule
     * @return {?}
     */
    function enqueueModuleForDelayedScoping(moduleType, ngModule) {
        moduleQueue.push({ moduleType, ngModule });
    }
    /** @type {?} */
    let flushingModuleQueue = false;
    /**
     * Loops over queued module definitions, if a given module definition has all of its
     * declarations resolved, it dequeues that module definition and sets the scope on
     * its declarations.
     * @return {?}
     */
    function flushModuleScopingQueueAsMuchAsPossible() {
        if (!flushingModuleQueue) {
            flushingModuleQueue = true;
            try {
                for (let i = moduleQueue.length - 1; i >= 0; i--) {
                    const { moduleType, ngModule } = moduleQueue[i];
                    if (ngModule.declarations && ngModule.declarations.every(isResolvedDeclaration)) {
                        // dequeue
                        moduleQueue.splice(i, 1);
                        setScopeOnDeclaredComponents(moduleType, ngModule);
                    }
                }
            }
            finally {
                flushingModuleQueue = false;
            }
        }
    }
    /**
     * Returns truthy if a declaration has resolved. If the declaration happens to be
     * an array of declarations, it will recurse to check each declaration in that array
     * (which may also be arrays).
     * @param {?} declaration
     * @return {?}
     */
    function isResolvedDeclaration(declaration) {
        if (Array.isArray(declaration)) {
            return declaration.every(isResolvedDeclaration);
        }
        return !!resolveForwardRef(declaration);
    }
    /**
     * Compiles a module in JIT mode.
     *
     * This function automatically gets called when a class has a `\@NgModule` decorator.
     * @param {?} moduleType
     * @param {?=} ngModule
     * @return {?}
     */
    function compileNgModule(moduleType, ngModule = {}) {
        compileNgModuleDefs((/** @type {?} */ (moduleType)), ngModule);
        // Because we don't know if all declarations have resolved yet at the moment the
        // NgModule decorator is executing, we're enqueueing the setting of module scope
        // on its declarations to be run at a later time when all declarations for the module,
        // including forward refs, have resolved.
        enqueueModuleForDelayedScoping(moduleType, ngModule);
    }
    /**
     * Compiles and adds the `ɵmod` and `ɵinj` properties to the module class.
     *
     * It's possible to compile a module via this API which will allow duplicate declarations in its
     * root.
     * @param {?} moduleType
     * @param {?} ngModule
     * @param {?=} allowDuplicateDeclarationsInRoot
     * @return {?}
     */
    function compileNgModuleDefs(moduleType, ngModule, allowDuplicateDeclarationsInRoot = false) {
        ngDevMode && assertDefined(moduleType, 'Required value moduleType');
        ngDevMode && assertDefined(ngModule, 'Required value ngModule');
        /** @type {?} */
        const declarations = flatten(ngModule.declarations || EMPTY_ARRAY$5);
        /** @type {?} */
        let ngModuleDef = null;
        Object.defineProperty(moduleType, NG_MOD_DEF, {
            configurable: true,
            get: (/**
             * @return {?}
             */
            () => {
                if (ngModuleDef === null) {
                    if (ngDevMode && ngModule.imports && ngModule.imports.indexOf(moduleType) > -1) {
                        // We need to assert this immediately, because allowing it to continue will cause it to
                        // go into an infinite loop before we've reached the point where we throw all the errors.
                        throw new Error(`'${stringifyForError(moduleType)}' module can't import itself`);
                    }
                    ngModuleDef = getCompilerFacade().compileNgModule(angularCoreEnv, `ng:///${moduleType.name}/ɵmod.js`, {
                        type: moduleType,
                        bootstrap: flatten(ngModule.bootstrap || EMPTY_ARRAY$5).map(resolveForwardRef),
                        declarations: declarations.map(resolveForwardRef),
                        imports: flatten(ngModule.imports || EMPTY_ARRAY$5)
                            .map(resolveForwardRef)
                            .map(expandModuleWithProviders),
                        exports: flatten(ngModule.exports || EMPTY_ARRAY$5)
                            .map(resolveForwardRef)
                            .map(expandModuleWithProviders),
                        schemas: ngModule.schemas ? flatten(ngModule.schemas) : null,
                        id: ngModule.id || null,
                    });
                    // Set `schemas` on ngModuleDef to an empty array in JIT mode to indicate that runtime
                    // should verify that there are no unknown elements in a template. In AOT mode, that check
                    // happens at compile time and `schemas` information is not present on Component and Module
                    // defs after compilation (so the check doesn't happen the second time at runtime).
                    if (!ngModuleDef.schemas) {
                        ngModuleDef.schemas = [];
                    }
                }
                return ngModuleDef;
            })
        });
        /** @type {?} */
        let ngInjectorDef = null;
        Object.defineProperty(moduleType, NG_INJ_DEF, {
            get: (/**
             * @return {?}
             */
            () => {
                if (ngInjectorDef === null) {
                    ngDevMode &&
                        verifySemanticsOfNgModuleDef((/** @type {?} */ ((/** @type {?} */ (moduleType)))), allowDuplicateDeclarationsInRoot);
                    /** @type {?} */
                    const meta = {
                        name: moduleType.name,
                        type: moduleType,
                        deps: reflectDependencies(moduleType),
                        providers: ngModule.providers || EMPTY_ARRAY$5,
                        imports: [
                            (ngModule.imports || EMPTY_ARRAY$5).map(resolveForwardRef),
                            (ngModule.exports || EMPTY_ARRAY$5).map(resolveForwardRef),
                        ],
                    };
                    ngInjectorDef = getCompilerFacade().compileInjector(angularCoreEnv, `ng:///${moduleType.name}/ɵinj.js`, meta);
                }
                return ngInjectorDef;
            }),
            // Make the property configurable in dev mode to allow overriding in tests
            configurable: !!ngDevMode,
        });
    }
    /**
     * @param {?} moduleType
     * @param {?} allowDuplicateDeclarationsInRoot
     * @param {?=} importingModule
     * @return {?}
     */
    function verifySemanticsOfNgModuleDef(moduleType, allowDuplicateDeclarationsInRoot, importingModule) {
        if (verifiedNgModule.get(moduleType))
            return;
        verifiedNgModule.set(moduleType, true);
        moduleType = resolveForwardRef(moduleType);
        /** @type {?} */
        let ngModuleDef;
        if (importingModule) {
            ngModuleDef = (/** @type {?} */ (getNgModuleDef(moduleType)));
            if (!ngModuleDef) {
                throw new Error(`Unexpected value '${moduleType.name}' imported by the module '${importingModule.name}'. Please add an @NgModule annotation.`);
            }
        }
        else {
            ngModuleDef = getNgModuleDef(moduleType, true);
        }
        /** @type {?} */
        const errors = [];
        /** @type {?} */
        const declarations = maybeUnwrapFn(ngModuleDef.declarations);
        /** @type {?} */
        const imports = maybeUnwrapFn(ngModuleDef.imports);
        flatten(imports).map(unwrapModuleWithProvidersImports).forEach((/**
         * @param {?} mod
         * @return {?}
         */
        mod => {
            verifySemanticsOfNgModuleImport(mod, moduleType);
            verifySemanticsOfNgModuleDef(mod, false, moduleType);
        }));
        /** @type {?} */
        const exports = maybeUnwrapFn(ngModuleDef.exports);
        declarations.forEach(verifyDeclarationsHaveDefinitions);
        declarations.forEach(verifyDirectivesHaveSelector);
        /** @type {?} */
        const combinedDeclarations = [
            ...declarations.map(resolveForwardRef),
            ...flatten(imports.map(computeCombinedExports)).map(resolveForwardRef),
        ];
        exports.forEach(verifyExportsAreDeclaredOrReExported);
        declarations.forEach((/**
         * @param {?} decl
         * @return {?}
         */
        decl => verifyDeclarationIsUnique(decl, allowDuplicateDeclarationsInRoot)));
        declarations.forEach(verifyComponentEntryComponentsIsPartOfNgModule);
        /** @type {?} */
        const ngModule = getAnnotation(moduleType, 'NgModule');
        if (ngModule) {
            ngModule.imports &&
                flatten(ngModule.imports).map(unwrapModuleWithProvidersImports).forEach((/**
                 * @param {?} mod
                 * @return {?}
                 */
                mod => {
                    verifySemanticsOfNgModuleImport(mod, moduleType);
                    verifySemanticsOfNgModuleDef(mod, false, moduleType);
                }));
            ngModule.bootstrap && deepForEach(ngModule.bootstrap, verifyCorrectBootstrapType);
            ngModule.bootstrap && deepForEach(ngModule.bootstrap, verifyComponentIsPartOfNgModule);
            ngModule.entryComponents &&
                deepForEach(ngModule.entryComponents, verifyComponentIsPartOfNgModule);
        }
        // Throw Error if any errors were detected.
        if (errors.length) {
            throw new Error(errors.join('\n'));
        }
        ////////////////////////////////////////////////////////////////////////////////////////////////
        /**
         * @param {?} type
         * @return {?}
         */
        function verifyDeclarationsHaveDefinitions(type) {
            type = resolveForwardRef(type);
            /** @type {?} */
            const def = getComponentDef(type) || getDirectiveDef(type) || getPipeDef(type);
            if (!def) {
                errors.push(`Unexpected value '${stringifyForError(type)}' declared by the module '${stringifyForError(moduleType)}'. Please add a @Pipe/@Directive/@Component annotation.`);
            }
        }
        /**
         * @param {?} type
         * @return {?}
         */
        function verifyDirectivesHaveSelector(type) {
            type = resolveForwardRef(type);
            /** @type {?} */
            const def = getDirectiveDef(type);
            if (!getComponentDef(type) && def && def.selectors.length == 0) {
                errors.push(`Directive ${stringifyForError(type)} has no selector, please add it!`);
            }
        }
        /**
         * @param {?} type
         * @return {?}
         */
        function verifyExportsAreDeclaredOrReExported(type) {
            type = resolveForwardRef(type);
            /** @type {?} */
            const kind = getComponentDef(type) && 'component' || getDirectiveDef(type) && 'directive' ||
                getPipeDef(type) && 'pipe';
            if (kind) {
                // only checked if we are declared as Component, Directive, or Pipe
                // Modules don't need to be declared or imported.
                if (combinedDeclarations.lastIndexOf(type) === -1) {
                    // We are exporting something which we don't explicitly declare or import.
                    errors.push(`Can't export ${kind} ${stringifyForError(type)} from ${stringifyForError(moduleType)} as it was neither declared nor imported!`);
                }
            }
        }
        /**
         * @param {?} type
         * @param {?} suppressErrors
         * @return {?}
         */
        function verifyDeclarationIsUnique(type, suppressErrors) {
            type = resolveForwardRef(type);
            /** @type {?} */
            const existingModule = ownerNgModule.get(type);
            if (existingModule && existingModule !== moduleType) {
                if (!suppressErrors) {
                    /** @type {?} */
                    const modules = [existingModule, moduleType].map(stringifyForError).sort();
                    errors.push(`Type ${stringifyForError(type)} is part of the declarations of 2 modules: ${modules[0]} and ${modules[1]}! ` +
                        `Please consider moving ${stringifyForError(type)} to a higher module that imports ${modules[0]} and ${modules[1]}. ` +
                        `You can also create a new NgModule that exports and includes ${stringifyForError(type)} then import that NgModule in ${modules[0]} and ${modules[1]}.`);
                }
            }
            else {
                // Mark type as having owner.
                ownerNgModule.set(type, moduleType);
            }
        }
        /**
         * @param {?} type
         * @return {?}
         */
        function verifyComponentIsPartOfNgModule(type) {
            type = resolveForwardRef(type);
            /** @type {?} */
            const existingModule = ownerNgModule.get(type);
            if (!existingModule) {
                errors.push(`Component ${stringifyForError(type)} is not part of any NgModule or the module has not been imported into your module.`);
            }
        }
        /**
         * @param {?} type
         * @return {?}
         */
        function verifyCorrectBootstrapType(type) {
            type = resolveForwardRef(type);
            if (!getComponentDef(type)) {
                errors.push(`${stringifyForError(type)} cannot be used as an entry component.`);
            }
        }
        /**
         * @param {?} type
         * @return {?}
         */
        function verifyComponentEntryComponentsIsPartOfNgModule(type) {
            type = resolveForwardRef(type);
            if (getComponentDef(type)) {
                // We know we are component
                /** @type {?} */
                const component = getAnnotation(type, 'Component');
                if (component && component.entryComponents) {
                    deepForEach(component.entryComponents, verifyComponentIsPartOfNgModule);
                }
            }
        }
        /**
         * @param {?} type
         * @param {?} importingModule
         * @return {?}
         */
        function verifySemanticsOfNgModuleImport(type, importingModule) {
            type = resolveForwardRef(type);
            if (getComponentDef(type) || getDirectiveDef(type)) {
                throw new Error(`Unexpected directive '${type.name}' imported by the module '${importingModule.name}'. Please add an @NgModule annotation.`);
            }
            if (getPipeDef(type)) {
                throw new Error(`Unexpected pipe '${type.name}' imported by the module '${importingModule.name}'. Please add an @NgModule annotation.`);
            }
        }
    }
    /**
     * @param {?} typeOrWithProviders
     * @return {?}
     */
    function unwrapModuleWithProvidersImports(typeOrWithProviders) {
        typeOrWithProviders = resolveForwardRef(typeOrWithProviders);
        return ((/** @type {?} */ (typeOrWithProviders))).ngModule || typeOrWithProviders;
    }
    /**
     * @template T
     * @param {?} type
     * @param {?} name
     * @return {?}
     */
    function getAnnotation(type, name) {
        /** @type {?} */
        let annotation = null;
        collect(type.__annotations__);
        collect(type.decorators);
        return annotation;
        /**
         * @param {?} annotations
         * @return {?}
         */
        function collect(annotations) {
            if (annotations) {
                annotations.forEach(readAnnotation);
            }
        }
        /**
         * @param {?} decorator
         * @return {?}
         */
        function readAnnotation(decorator) {
            if (!annotation) {
                /** @type {?} */
                const proto = Object.getPrototypeOf(decorator);
                if (proto.ngMetadataName == name) {
                    annotation = (/** @type {?} */ (decorator));
                }
                else if (decorator.type) {
                    /** @type {?} */
                    const proto = Object.getPrototypeOf(decorator.type);
                    if (proto.ngMetadataName == name) {
                        annotation = decorator.args[0];
                    }
                }
            }
        }
    }
    /**
     * Keep track of compiled components. This is needed because in tests we often want to compile the
     * same component with more than one NgModule. This would cause an error unless we reset which
     * NgModule the component belongs to. We keep the list of compiled components here so that the
     * TestBed can reset it later.
     * @type {?}
     */
    let ownerNgModule = new Map();
    /** @type {?} */
    let verifiedNgModule = new Map();
    /**
     * Computes the combined declarations of explicit declarations, as well as declarations inherited by
     * traversing the exports of imported modules.
     * @param {?} type
     * @return {?}
     */
    function computeCombinedExports(type) {
        type = resolveForwardRef(type);
        /** @type {?} */
        const ngModuleDef = getNgModuleDef(type, true);
        return [...flatten(maybeUnwrapFn(ngModuleDef.exports).map((/**
             * @param {?} type
             * @return {?}
             */
            (type) => {
                /** @type {?} */
                const ngModuleDef = getNgModuleDef(type);
                if (ngModuleDef) {
                    verifySemanticsOfNgModuleDef((/** @type {?} */ ((/** @type {?} */ (type)))), false);
                    return computeCombinedExports(type);
                }
                else {
                    return type;
                }
            })))];
    }
    /**
     * Some declared components may be compiled asynchronously, and thus may not have their
     * ɵcmp set yet. If this is the case, then a reference to the module is written into
     * the `ngSelectorScope` property of the declared type.
     * @param {?} moduleType
     * @param {?} ngModule
     * @return {?}
     */
    function setScopeOnDeclaredComponents(moduleType, ngModule) {
        /** @type {?} */
        const declarations = flatten(ngModule.declarations || EMPTY_ARRAY$5);
        /** @type {?} */
        const transitiveScopes = transitiveScopesFor(moduleType);
        declarations.forEach((/**
         * @param {?} declaration
         * @return {?}
         */
        declaration => {
            if (declaration.hasOwnProperty(NG_COMP_DEF)) {
                // A `ɵcmp` field exists - go ahead and patch the component directly.
                /** @type {?} */
                const component = (/** @type {?} */ (declaration));
                /** @type {?} */
                const componentDef = (/** @type {?} */ (getComponentDef(component)));
                patchComponentDefWithScope(componentDef, transitiveScopes);
            }
            else if (!declaration.hasOwnProperty(NG_DIR_DEF) && !declaration.hasOwnProperty(NG_PIPE_DEF)) {
                // Set `ngSelectorScope` for future reference when the component compilation finishes.
                ((/** @type {?} */ (declaration))).ngSelectorScope = moduleType;
            }
        }));
    }
    /**
     * Patch the definition of a component with directives and pipes from the compilation scope of
     * a given module.
     * @template C
     * @param {?} componentDef
     * @param {?} transitiveScopes
     * @return {?}
     */
    function patchComponentDefWithScope(componentDef, transitiveScopes) {
        componentDef.directiveDefs = (/**
         * @return {?}
         */
        () => Array.from(transitiveScopes.compilation.directives)
            .map((/**
         * @param {?} dir
         * @return {?}
         */
        dir => dir.hasOwnProperty(NG_COMP_DEF) ? (/** @type {?} */ (getComponentDef(dir))) : (/** @type {?} */ (getDirectiveDef(dir)))))
            .filter((/**
         * @param {?} def
         * @return {?}
         */
        def => !!def)));
        componentDef.pipeDefs = (/**
         * @return {?}
         */
        () => Array.from(transitiveScopes.compilation.pipes).map((/**
         * @param {?} pipe
         * @return {?}
         */
        pipe => (/** @type {?} */ (getPipeDef(pipe))))));
        componentDef.schemas = transitiveScopes.schemas;
        // Since we avoid Components/Directives/Pipes recompiling in case there are no overrides, we
        // may face a problem where previously compiled defs available to a given Component/Directive
        // are cached in TView and may become stale (in case any of these defs gets recompiled). In
        // order to avoid this problem, we force fresh TView to be created.
        componentDef.tView = null;
    }
    /**
     * Compute the pair of transitive scopes (compilation scope and exported scope) for a given module.
     *
     * This operation is memoized and the result is cached on the module's definition. This function can
     * be called on modules with components that have not fully compiled yet, but the result should not
     * be used until they have.
     *
     * @template T
     * @param {?} moduleType module that transitive scope should be calculated for.
     * @return {?}
     */
    function transitiveScopesFor(moduleType) {
        if (!isNgModule(moduleType)) {
            throw new Error(`${moduleType.name} does not have a module def (ɵmod property)`);
        }
        /** @type {?} */
        const def = (/** @type {?} */ (getNgModuleDef(moduleType)));
        if (def.transitiveCompileScopes !== null) {
            return def.transitiveCompileScopes;
        }
        /** @type {?} */
        const scopes = {
            schemas: def.schemas || null,
            compilation: {
                directives: new Set(),
                pipes: new Set(),
            },
            exported: {
                directives: new Set(),
                pipes: new Set(),
            },
        };
        maybeUnwrapFn(def.imports).forEach((/**
         * @template I
         * @param {?} imported
         * @return {?}
         */
        (imported) => {
            /** @type {?} */
            const importedType = (/** @type {?} */ (imported));
            if (!isNgModule(importedType)) {
                throw new Error(`Importing ${importedType.name} which does not have a ɵmod property`);
            }
            // When this module imports another, the imported module's exported directives and pipes are
            // added to the compilation scope of this module.
            /** @type {?} */
            const importedScope = transitiveScopesFor(importedType);
            importedScope.exported.directives.forEach((/**
             * @param {?} entry
             * @return {?}
             */
            entry => scopes.compilation.directives.add(entry)));
            importedScope.exported.pipes.forEach((/**
             * @param {?} entry
             * @return {?}
             */
            entry => scopes.compilation.pipes.add(entry)));
        }));
        maybeUnwrapFn(def.declarations).forEach((/**
         * @param {?} declared
         * @return {?}
         */
        declared => {
            /** @type {?} */
            const declaredWithDefs = (/** @type {?} */ (declared));
            if (getPipeDef(declaredWithDefs)) {
                scopes.compilation.pipes.add(declared);
            }
            else {
                // Either declared has a ɵcmp or ɵdir, or it's a component which hasn't
                // had its template compiled yet. In either case, it gets added to the compilation's
                // directives.
                scopes.compilation.directives.add(declared);
            }
        }));
        maybeUnwrapFn(def.exports).forEach((/**
         * @template E
         * @param {?} exported
         * @return {?}
         */
        (exported) => {
            /** @type {?} */
            const exportedType = (/** @type {?} */ (exported));
            // Either the type is a module, a pipe, or a component/directive (which may not have a
            // ɵcmp as it might be compiled asynchronously).
            if (isNgModule(exportedType)) {
                // When this module exports another, the exported module's exported directives and pipes are
                // added to both the compilation and exported scopes of this module.
                /** @type {?} */
                const exportedScope = transitiveScopesFor(exportedType);
                exportedScope.exported.directives.forEach((/**
                 * @param {?} entry
                 * @return {?}
                 */
                entry => {
                    scopes.compilation.directives.add(entry);
                    scopes.exported.directives.add(entry);
                }));
                exportedScope.exported.pipes.forEach((/**
                 * @param {?} entry
                 * @return {?}
                 */
                entry => {
                    scopes.compilation.pipes.add(entry);
                    scopes.exported.pipes.add(entry);
                }));
            }
            else if (getPipeDef(exportedType)) {
                scopes.exported.pipes.add(exportedType);
            }
            else {
                scopes.exported.directives.add(exportedType);
            }
        }));
        def.transitiveCompileScopes = scopes;
        return scopes;
    }
    /**
     * @param {?} value
     * @return {?}
     */
    function expandModuleWithProviders(value) {
        if (isModuleWithProviders(value)) {
            return value.ngModule;
        }
        return value;
    }
    /**
     * @param {?} value
     * @return {?}
     */
    function isModuleWithProviders(value) {
        return ((/** @type {?} */ (value))).ngModule !== undefined;
    }
    /**
     * @template T
     * @param {?} value
     * @return {?}
     */
    function isNgModule(value) {
        return !!getNgModuleDef(value);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/jit/directive.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Compile an Angular component according to its decorator metadata, and patch the resulting
     * component def (ɵcmp) onto the component type.
     *
     * Compilation may be asynchronous (due to the need to resolve URLs for the component template or
     * other resources, for example). In the event that compilation is not immediate, `compileComponent`
     * will enqueue resource resolution into a global queue and will fail to return the `ɵcmp`
     * until the global queue has been resolved with a call to `resolveComponentResources`.
     * @param {?} type
     * @param {?} metadata
     * @return {?}
     */
    function compileComponent(type, metadata) {
        // Initialize ngDevMode. This must be the first statement in compileComponent.
        // See the `initNgDevMode` docstring for more information.
        (typeof ngDevMode === 'undefined' || ngDevMode) && initNgDevMode();
        /** @type {?} */
        let ngComponentDef = null;
        // Metadata may have resources which need to be resolved.
        maybeQueueResolutionOfComponentResources(type, metadata);
        // Note that we're using the same function as `Directive`, because that's only subset of metadata
        // that we need to create the ngFactoryDef. We're avoiding using the component metadata
        // because we'd have to resolve the asynchronous templates.
        addDirectiveFactoryDef(type, metadata);
        Object.defineProperty(type, NG_COMP_DEF, {
            get: (/**
             * @return {?}
             */
            () => {
                if (ngComponentDef === null) {
                    /** @type {?} */
                    const compiler = getCompilerFacade();
                    if (componentNeedsResolution(metadata)) {
                        /** @type {?} */
                        const error = [`Component '${type.name}' is not resolved:`];
                        if (metadata.templateUrl) {
                            error.push(` - templateUrl: ${metadata.templateUrl}`);
                        }
                        if (metadata.styleUrls && metadata.styleUrls.length) {
                            error.push(` - styleUrls: ${JSON.stringify(metadata.styleUrls)}`);
                        }
                        error.push(`Did you run and wait for 'resolveComponentResources()'?`);
                        throw new Error(error.join('\n'));
                    }
                    // This const was called `jitOptions` previously but had to be renamed to `options` because
                    // of a bug with Terser that caused optimized JIT builds to throw a `ReferenceError`.
                    // This bug was investigated in https://github.com/angular/angular-cli/issues/17264.
                    // We should not rename it back until https://github.com/terser/terser/issues/615 is fixed.
                    /** @type {?} */
                    const options = getJitOptions();
                    /** @type {?} */
                    let preserveWhitespaces = metadata.preserveWhitespaces;
                    if (preserveWhitespaces === undefined) {
                        if (options !== null && options.preserveWhitespaces !== undefined) {
                            preserveWhitespaces = options.preserveWhitespaces;
                        }
                        else {
                            preserveWhitespaces = false;
                        }
                    }
                    /** @type {?} */
                    let encapsulation = metadata.encapsulation;
                    if (encapsulation === undefined) {
                        if (options !== null && options.defaultEncapsulation !== undefined) {
                            encapsulation = options.defaultEncapsulation;
                        }
                        else {
                            encapsulation = ViewEncapsulation$1.Emulated;
                        }
                    }
                    /** @type {?} */
                    const templateUrl = metadata.templateUrl || `ng:///${type.name}/template.html`;
                    /** @type {?} */
                    const meta = Object.assign(Object.assign({}, directiveMetadata(type, metadata)), { typeSourceSpan: compiler.createParseSourceSpan('Component', type.name, templateUrl), template: metadata.template || '', preserveWhitespaces, styles: metadata.styles || EMPTY_ARRAY, animations: metadata.animations, directives: [], changeDetection: metadata.changeDetection, pipes: new Map(), encapsulation, interpolation: metadata.interpolation, viewProviders: metadata.viewProviders || null });
                    if (meta.usesInheritance) {
                        addDirectiveDefToUndecoratedParents(type);
                    }
                    ngComponentDef = compiler.compileComponent(angularCoreEnv, templateUrl, meta);
                    // When NgModule decorator executed, we enqueued the module definition such that
                    // it would only dequeue and add itself as module scope to all of its declarations,
                    // but only if  if all of its declarations had resolved. This call runs the check
                    // to see if any modules that are in the queue can be dequeued and add scope to
                    // their declarations.
                    flushModuleScopingQueueAsMuchAsPossible();
                    // If component compilation is async, then the @NgModule annotation which declares the
                    // component may execute and set an ngSelectorScope property on the component type. This
                    // allows the component to patch itself with directiveDefs from the module after it
                    // finishes compiling.
                    if (hasSelectorScope(type)) {
                        /** @type {?} */
                        const scopes = transitiveScopesFor(type.ngSelectorScope);
                        patchComponentDefWithScope(ngComponentDef, scopes);
                    }
                }
                return ngComponentDef;
            }),
            // Make the property configurable in dev mode to allow overriding in tests
            configurable: !!ngDevMode,
        });
    }
    /**
     * @template T
     * @param {?} component
     * @return {?}
     */
    function hasSelectorScope(component) {
        return ((/** @type {?} */ (component))).ngSelectorScope !== undefined;
    }
    /**
     * Compile an Angular directive according to its decorator metadata, and patch the resulting
     * directive def onto the component type.
     *
     * In the event that compilation is not immediate, `compileDirective` will return a `Promise` which
     * will resolve when compilation completes and the directive becomes usable.
     * @param {?} type
     * @param {?} directive
     * @return {?}
     */
    function compileDirective(type, directive) {
        /** @type {?} */
        let ngDirectiveDef = null;
        addDirectiveFactoryDef(type, directive || {});
        Object.defineProperty(type, NG_DIR_DEF, {
            get: (/**
             * @return {?}
             */
            () => {
                if (ngDirectiveDef === null) {
                    // `directive` can be null in the case of abstract directives as a base class
                    // that use `@Directive()` with no selector. In that case, pass empty object to the
                    // `directiveMetadata` function instead of null.
                    /** @type {?} */
                    const meta = getDirectiveMetadata(type, directive || {});
                    ngDirectiveDef =
                        getCompilerFacade().compileDirective(angularCoreEnv, meta.sourceMapUrl, meta.metadata);
                }
                return ngDirectiveDef;
            }),
            // Make the property configurable in dev mode to allow overriding in tests
            configurable: !!ngDevMode,
        });
    }
    /**
     * @param {?} type
     * @param {?} metadata
     * @return {?}
     */
    function getDirectiveMetadata(type, metadata) {
        /** @type {?} */
        const name = type && type.name;
        /** @type {?} */
        const sourceMapUrl = `ng:///${name}/ɵdir.js`;
        /** @type {?} */
        const compiler = getCompilerFacade();
        /** @type {?} */
        const facade = directiveMetadata((/** @type {?} */ (type)), metadata);
        facade.typeSourceSpan = compiler.createParseSourceSpan('Directive', name, sourceMapUrl);
        if (facade.usesInheritance) {
            addDirectiveDefToUndecoratedParents(type);
        }
        return { metadata: facade, sourceMapUrl };
    }
    /**
     * @param {?} type
     * @param {?} metadata
     * @return {?}
     */
    function addDirectiveFactoryDef(type, metadata) {
        /** @type {?} */
        let ngFactoryDef = null;
        Object.defineProperty(type, NG_FACTORY_DEF, {
            get: (/**
             * @return {?}
             */
            () => {
                if (ngFactoryDef === null) {
                    /** @type {?} */
                    const meta = getDirectiveMetadata(type, metadata);
                    /** @type {?} */
                    const compiler = getCompilerFacade();
                    ngFactoryDef = compiler.compileFactory(angularCoreEnv, `ng:///${type.name}/ɵfac.js`, Object.assign(Object.assign({}, meta.metadata), { injectFn: 'directiveInject', target: compiler.R3FactoryTarget.Directive }));
                }
                return ngFactoryDef;
            }),
            // Make the property configurable in dev mode to allow overriding in tests
            configurable: !!ngDevMode,
        });
    }
    /**
     * @param {?} type
     * @return {?}
     */
    function extendsDirectlyFromObject(type) {
        return Object.getPrototypeOf(type.prototype) === Object.prototype;
    }
    /**
     * Extract the `R3DirectiveMetadata` for a particular directive (either a `Directive` or a
     * `Component`).
     * @param {?} type
     * @param {?} metadata
     * @return {?}
     */
    function directiveMetadata(type, metadata) {
        // Reflect inputs and outputs.
        /** @type {?} */
        const reflect = getReflect();
        /** @type {?} */
        const propMetadata = reflect.ownPropMetadata(type);
        return {
            name: type.name,
            type: type,
            typeArgumentCount: 0,
            selector: metadata.selector !== undefined ? metadata.selector : null,
            deps: reflectDependencies(type),
            host: metadata.host || EMPTY_OBJ,
            propMetadata: propMetadata,
            inputs: metadata.inputs || EMPTY_ARRAY,
            outputs: metadata.outputs || EMPTY_ARRAY,
            queries: extractQueriesMetadata(type, propMetadata, isContentQuery),
            lifecycle: { usesOnChanges: reflect.hasLifecycleHook(type, 'ngOnChanges') },
            typeSourceSpan: (/** @type {?} */ (null)),
            usesInheritance: !extendsDirectlyFromObject(type),
            exportAs: extractExportAs(metadata.exportAs),
            providers: metadata.providers || null,
            viewQueries: extractQueriesMetadata(type, propMetadata, isViewQuery)
        };
    }
    /**
     * Adds a directive definition to all parent classes of a type that don't have an Angular decorator.
     * @param {?} type
     * @return {?}
     */
    function addDirectiveDefToUndecoratedParents(type) {
        /** @type {?} */
        const objPrototype = Object.prototype;
        /** @type {?} */
        let parent = Object.getPrototypeOf(type.prototype).constructor;
        // Go up the prototype until we hit `Object`.
        while (parent && parent !== objPrototype) {
            // Since inheritance works if the class was annotated already, we only need to add
            // the def if there are no annotations and the def hasn't been created already.
            if (!getDirectiveDef(parent) && !getComponentDef(parent) &&
                shouldAddAbstractDirective(parent)) {
                compileDirective(parent, null);
            }
            parent = Object.getPrototypeOf(parent);
        }
    }
    /**
     * @param {?} selector
     * @return {?}
     */
    function convertToR3QueryPredicate(selector) {
        return typeof selector === 'string' ? splitByComma(selector) : resolveForwardRef(selector);
    }
    /**
     * @param {?} propertyName
     * @param {?} ann
     * @return {?}
     */
    function convertToR3QueryMetadata(propertyName, ann) {
        return {
            propertyName: propertyName,
            predicate: convertToR3QueryPredicate(ann.selector),
            descendants: ann.descendants,
            first: ann.first,
            read: ann.read ? ann.read : null,
            static: !!ann.static
        };
    }
    /**
     * @param {?} type
     * @param {?} propMetadata
     * @param {?} isQueryAnn
     * @return {?}
     */
    function extractQueriesMetadata(type, propMetadata, isQueryAnn) {
        /** @type {?} */
        const queriesMeta = [];
        for (const field in propMetadata) {
            if (propMetadata.hasOwnProperty(field)) {
                /** @type {?} */
                const annotations = propMetadata[field];
                annotations.forEach((/**
                 * @param {?} ann
                 * @return {?}
                 */
                ann => {
                    if (isQueryAnn(ann)) {
                        if (!ann.selector) {
                            throw new Error(`Can't construct a query for the property "${field}" of ` +
                                `"${stringifyForError(type)}" since the query selector wasn't defined.`);
                        }
                        if (annotations.some(isInputAnnotation)) {
                            throw new Error(`Cannot combine @Input decorators with query decorators`);
                        }
                        queriesMeta.push(convertToR3QueryMetadata(field, ann));
                    }
                }));
            }
        }
        return queriesMeta;
    }
    /**
     * @param {?} exportAs
     * @return {?}
     */
    function extractExportAs(exportAs) {
        return exportAs === undefined ? null : splitByComma(exportAs);
    }
    /**
     * @param {?} value
     * @return {?}
     */
    function isContentQuery(value) {
        /** @type {?} */
        const name = value.ngMetadataName;
        return name === 'ContentChild' || name === 'ContentChildren';
    }
    /**
     * @param {?} value
     * @return {?}
     */
    function isViewQuery(value) {
        /** @type {?} */
        const name = value.ngMetadataName;
        return name === 'ViewChild' || name === 'ViewChildren';
    }
    /**
     * @param {?} value
     * @return {?}
     */
    function isInputAnnotation(value) {
        return value.ngMetadataName === 'Input';
    }
    /**
     * @param {?} value
     * @return {?}
     */
    function splitByComma(value) {
        return value.split(',').map((/**
         * @param {?} piece
         * @return {?}
         */
        piece => piece.trim()));
    }
    /** @type {?} */
    const LIFECYCLE_HOOKS = [
        'ngOnChanges', 'ngOnInit', 'ngOnDestroy', 'ngDoCheck', 'ngAfterViewInit', 'ngAfterViewChecked',
        'ngAfterContentInit', 'ngAfterContentChecked'
    ];
    /**
     * @param {?} type
     * @return {?}
     */
    function shouldAddAbstractDirective(type) {
        /** @type {?} */
        const reflect = getReflect();
        if (LIFECYCLE_HOOKS.some((/**
         * @param {?} hookName
         * @return {?}
         */
        hookName => reflect.hasLifecycleHook(type, hookName)))) {
            return true;
        }
        /** @type {?} */
        const propMetadata = reflect.propMetadata(type);
        for (const field in propMetadata) {
            /** @type {?} */
            const annotations = propMetadata[field];
            for (let i = 0; i < annotations.length; i++) {
                /** @type {?} */
                const current = annotations[i];
                /** @type {?} */
                const metadataName = current.ngMetadataName;
                if (isInputAnnotation(current) || isContentQuery(current) || isViewQuery(current) ||
                    metadataName === 'Output' || metadataName === 'HostBinding' ||
                    metadataName === 'HostListener') {
                    return true;
                }
            }
        }
        return false;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/render3/jit/pipe.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @param {?} type
     * @param {?} meta
     * @return {?}
     */
    function compilePipe(type, meta) {
        /** @type {?} */
        let ngPipeDef = null;
        /** @type {?} */
        let ngFactoryDef = null;
        Object.defineProperty(type, NG_FACTORY_DEF, {
            get: (/**
             * @return {?}
             */
            () => {
                if (ngFactoryDef === null) {
                    /** @type {?} */
                    const metadata = getPipeMetadata(type, meta);
                    /** @type {?} */
                    const compiler = getCompilerFacade();
                    ngFactoryDef = compiler.compileFactory(angularCoreEnv, `ng:///${metadata.name}/ɵfac.js`, Object.assign(Object.assign({}, metadata), { injectFn: 'directiveInject', target: compiler.R3FactoryTarget.Pipe }));
                }
                return ngFactoryDef;
            }),
            // Make the property configurable in dev mode to allow overriding in tests
            configurable: !!ngDevMode,
        });
        Object.defineProperty(type, NG_PIPE_DEF, {
            get: (/**
             * @return {?}
             */
            () => {
                if (ngPipeDef === null) {
                    /** @type {?} */
                    const metadata = getPipeMetadata(type, meta);
                    ngPipeDef = getCompilerFacade().compilePipe(angularCoreEnv, `ng:///${metadata.name}/ɵpipe.js`, metadata);
                }
                return ngPipeDef;
            }),
            // Make the property configurable in dev mode to allow overriding in tests
            configurable: !!ngDevMode,
        });
    }
    /**
     * @param {?} type
     * @param {?} meta
     * @return {?}
     */
    function getPipeMetadata(type, meta) {
        return {
            type: type,
            typeArgumentCount: 0,
            name: type.name,
            deps: reflectDependencies(type),
            pipeName: meta.name,
            pure: meta.pure !== undefined ? meta.pure : true
        };
    }
    // WARNING: interface has both a type and a value, skipping emit
    const ɵ0$e = /**
     * @param {?=} dir
     * @return {?}
     */
    (dir = {}) => dir, ɵ1$3 = /**
     * @param {?} type
     * @param {?} meta
     * @return {?}
     */
    (type, meta) => SWITCH_COMPILE_DIRECTIVE(type, meta);
    /**
     * Type of the Directive metadata.
     *
     * \@publicApi
     * @type {?}
     */
    const Directive = makeDecorator('Directive', (ɵ0$e), undefined, undefined, (ɵ1$3));
    // WARNING: interface has both a type and a value, skipping emit
    const ɵ2$1 = /**
     * @param {?=} c
     * @return {?}
     */
    (c = {}) => (Object.assign({ changeDetection: ChangeDetectionStrategy.Default }, c)), ɵ3$1 = /**
     * @param {?} type
     * @param {?} meta
     * @return {?}
     */
    (type, meta) => SWITCH_COMPILE_COMPONENT(type, meta);
    /**
     * Component decorator and metadata.
     *
     * \@Annotation
     * \@publicApi
     * @type {?}
     */
    const Component = makeDecorator('Component', (ɵ2$1), Directive, undefined, (ɵ3$1));
    // WARNING: interface has both a type and a value, skipping emit
    const ɵ4 = /**
     * @param {?} p
     * @return {?}
     */
    (p) => (Object.assign({ pure: true }, p)), ɵ5 = /**
     * @param {?} type
     * @param {?} meta
     * @return {?}
     */
    (type, meta) => SWITCH_COMPILE_PIPE(type, meta);
    /**
     * \@Annotation
     * \@publicApi
     * @type {?}
     */
    const Pipe = makeDecorator('Pipe', (ɵ4), undefined, undefined, (ɵ5));
    // WARNING: interface has both a type and a value, skipping emit
    const ɵ6 = /**
     * @param {?=} bindingPropertyName
     * @return {?}
     */
    (bindingPropertyName) => ({ bindingPropertyName });
    /**
     * \@Annotation
     * \@publicApi
     * @type {?}
     */
    const Input = makePropDecorator('Input', (ɵ6));
    // WARNING: interface has both a type and a value, skipping emit
    const ɵ7 = /**
     * @param {?=} bindingPropertyName
     * @return {?}
     */
    (bindingPropertyName) => ({ bindingPropertyName });
    /**
     * \@Annotation
     * \@publicApi
     * @type {?}
     */
    const Output = makePropDecorator('Output', (ɵ7));
    // WARNING: interface has both a type and a value, skipping emit
    const ɵ8 = /**
     * @param {?=} hostPropertyName
     * @return {?}
     */
    (hostPropertyName) => ({ hostPropertyName });
    /**
     * \@Annotation
     * \@publicApi
     * @type {?}
     */
    const HostBinding = makePropDecorator('HostBinding', (ɵ8));
    // WARNING: interface has both a type and a value, skipping emit
    const ɵ9 = /**
     * @param {?=} eventName
     * @param {?=} args
     * @return {?}
     */
    (eventName, args) => ({ eventName, args });
    /**
     * Decorator that binds a DOM event to a host listener and supplies configuration metadata.
     * Angular invokes the supplied handler method when the host element emits the specified event,
     * and updates the bound element with the result.
     *
     * If the handler method returns false, applies `preventDefault` on the bound element.
     *
     * \@usageNotes
     *
     * The following example declares a directive
     * that attaches a click listener to a button and counts clicks.
     *
     * ```ts
     * \@Directive({selector: 'button[counting]'})
     * class CountClicks {
     *   numberOfClicks = 0;
     *
     * \@HostListener('click', ['$event.target'])
     *   onClick(btn) {
     *     console.log('button', btn, 'number of clicks:', this.numberOfClicks++);
     *  }
     * }
     *
     * \@Component({
     *   selector: 'app',
     *   template: '<button counting>Increment</button>',
     * })
     * class App {}
     *
     * ```
     *
     * The following example registers another DOM event handler that listens for key-press events.
     * ``` ts
     * import { HostListener, Component } from "\@angular/core";
     *
     * \@Component({
     *   selector: 'app',
     *   template: `<h1>Hello, you have pressed keys {{counter}} number of times!</h1> Press any key to
     * increment the counter.
     *   <button (click)="resetCounter()">Reset Counter</button>`
     * })
     * class AppComponent {
     *   counter = 0;
     * \@HostListener('window:keydown', ['$event'])
     *   handleKeyDown(event: KeyboardEvent) {
     *     this.counter++;
     *   }
     *   resetCounter() {
     *     this.counter = 0;
     *   }
     * }
     * ```
     *
     * \@Annotation
     * \@publicApi
     * @type {?}
     */
    const HostListener = makePropDecorator('HostListener', (ɵ9));
    /** @type {?} */
    const SWITCH_COMPILE_COMPONENT__POST_R3__ = compileComponent;
    /** @type {?} */
    const SWITCH_COMPILE_DIRECTIVE__POST_R3__ = compileDirective;
    /** @type {?} */
    const SWITCH_COMPILE_PIPE__POST_R3__ = compilePipe;
    /** @type {?} */
    const SWITCH_COMPILE_COMPONENT = SWITCH_COMPILE_COMPONENT__POST_R3__;
    /** @type {?} */
    const SWITCH_COMPILE_DIRECTIVE = SWITCH_COMPILE_DIRECTIVE__POST_R3__;
    /** @type {?} */
    const SWITCH_COMPILE_PIPE = SWITCH_COMPILE_PIPE__POST_R3__;
    // WARNING: interface has both a type and a value, skipping emit
    const ɵ0$f = /**
     * @param {?} ngModule
     * @return {?}
     */
    (ngModule) => ngModule, ɵ1$4 = /**
     * Decorator that marks the following class as an NgModule, and supplies
     * configuration metadata for it.
     *
     * * The `declarations` and `entryComponents` options configure the compiler
     * with information about what belongs to the NgModule.
     * * The `providers` options configures the NgModule's injector to provide
     * dependencies the NgModule members.
     * * The `imports` and `exports` options bring in members from other modules, and make
     * this module's members available to others.
     * @param {?} type
     * @param {?} meta
     * @return {?}
     */
    (type, meta) => SWITCH_COMPILE_NGMODULE(type, meta);
    /**
     * \@Annotation
     * \@publicApi
     * @type {?}
     */
    const NgModule = makeDecorator('NgModule', (ɵ0$f), undefined, undefined, (ɵ1$4));
    /** @type {?} */
    const SWITCH_COMPILE_NGMODULE__POST_R3__ = compileNgModule;
    /** @type {?} */
    const SWITCH_COMPILE_NGMODULE = SWITCH_COMPILE_NGMODULE__POST_R3__;

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/metadata.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */

    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/application_init.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * An injection token that allows you to provide one or more initialization functions.
     * These function are injected at application startup and executed during
     * app initialization. If any of these functions returns a Promise, initialization
     * does not complete until the Promise is resolved.
     *
     * You can, for example, create a factory function that loads language data
     * or an external configuration, and provide that function to the `APP_INITIALIZER` token.
     * That way, the function is executed during the application bootstrap process,
     * and the needed data is available on startup.
     *
     * \@publicApi
     * @type {?}
     */
    const APP_INITIALIZER = new InjectionToken('Application Initializer');
    /**
     * A class that reflects the state of running {\@link APP_INITIALIZER}s.
     *
     * \@publicApi
     */
    class ApplicationInitStatus {
        /**
         * @param {?} appInits
         */
        constructor(appInits) {
            this.appInits = appInits;
            this.initialized = false;
            this.done = false;
            this.donePromise = new Promise((/**
             * @param {?} res
             * @param {?} rej
             * @return {?}
             */
            (res, rej) => {
                this.resolve = res;
                this.reject = rej;
            }));
        }
        /**
         * \@internal
         * @return {?}
         */
        runInitializers() {
            if (this.initialized) {
                return;
            }
            /** @type {?} */
            const asyncInitPromises = [];
            /** @type {?} */
            const complete = (/**
             * @return {?}
             */
            () => {
                ((/** @type {?} */ (this))).done = true;
                this.resolve();
            });
            if (this.appInits) {
                for (let i = 0; i < this.appInits.length; i++) {
                    /** @type {?} */
                    const initResult = this.appInits[i]();
                    if (isPromise$1(initResult)) {
                        asyncInitPromises.push(initResult);
                    }
                }
            }
            Promise.all(asyncInitPromises)
                .then((/**
             * @return {?}
             */
            () => {
                complete();
            }))
                .catch((/**
             * @param {?} e
             * @return {?}
             */
            e => {
                this.reject(e);
            }));
            if (asyncInitPromises.length === 0) {
                complete();
            }
            this.initialized = true;
        }
    }
    ApplicationInitStatus.ɵfac = function ApplicationInitStatus_Factory(t) { return new (t || ApplicationInitStatus)(ɵɵinject(APP_INITIALIZER, 8)); };
    ApplicationInitStatus.ɵprov = ɵɵdefineInjectable({ token: ApplicationInitStatus, factory: ApplicationInitStatus.ɵfac });
    /** @nocollapse */
    ApplicationInitStatus.ctorParameters = () => [
        { type: Array, decorators: [{ type: Inject, args: [APP_INITIALIZER,] }, { type: Optional }] }
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/application_tokens.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * A DI Token representing a unique string id assigned to the application by Angular and used
     * primarily for prefixing application attributes and CSS styles when
     * {\@link ViewEncapsulation#Emulated ViewEncapsulation.Emulated} is being used.
     *
     * If you need to avoid randomly generated value to be used as an application id, you can provide
     * a custom value via a DI provider <!-- TODO: provider --> configuring the root {\@link Injector}
     * using this token.
     * \@publicApi
     * @type {?}
     */
    const APP_ID = new InjectionToken('AppId');
    /**
     * @return {?}
     */
    function _appIdRandomProviderFactory() {
        return `${_randomChar()}${_randomChar()}${_randomChar()}`;
    }
    /**
     * Providers that will generate a random APP_ID_TOKEN.
     * \@publicApi
     * @type {?}
     */
    const APP_ID_RANDOM_PROVIDER = {
        provide: APP_ID,
        useFactory: _appIdRandomProviderFactory,
        deps: (/** @type {?} */ ([])),
    };
    /**
     * @return {?}
     */
    function _randomChar() {
        return String.fromCharCode(97 + Math.floor(Math.random() * 25));
    }
    /**
     * A function that will be executed when a platform is initialized.
     * \@publicApi
     * @type {?}
     */
    const PLATFORM_INITIALIZER = new InjectionToken('Platform Initializer');
    /**
     * A token that indicates an opaque platform id.
     * \@publicApi
     * @type {?}
     */
    const PLATFORM_ID = new InjectionToken('Platform ID');
    /**
     * All callbacks provided via this token will be called for every component that is bootstrapped.
     * Signature of the callback:
     *
     * `(componentRef: ComponentRef) => void`.
     *
     * \@publicApi
     * @type {?}
     */
    const APP_BOOTSTRAP_LISTENER = new InjectionToken('appBootstrapListener');
    /**
     * A token which indicates the root directory of the application
     * \@publicApi
     * @type {?}
     */
    const PACKAGE_ROOT_URL = new InjectionToken('Application Packages Root URL');

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/console.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class Console {
        /**
         * @param {?} message
         * @return {?}
         */
        log(message) {
            // tslint:disable-next-line:no-console
            console.log(message);
        }
        // Note: for reporting errors use `DOM.logError()` as it is platform specific
        /**
         * @param {?} message
         * @return {?}
         */
        warn(message) {
            // tslint:disable-next-line:no-console
            console.warn(message);
        }
    }
    Console.ɵfac = function Console_Factory(t) { return new (t || Console)(); };
    Console.ɵprov = ɵɵdefineInjectable({ token: Console, factory: Console.ɵfac });

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/i18n/tokens.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Provide this token to set the locale of your application.
     * It is used for i18n extraction, by i18n pipes (DatePipe, I18nPluralPipe, CurrencyPipe,
     * DecimalPipe and PercentPipe) and by ICU expressions.
     *
     * See the [i18n guide](guide/i18n#setting-up-locale) for more information.
     *
     * \@usageNotes
     * ### Example
     *
     * ```typescript
     * import { LOCALE_ID } from '\@angular/core';
     * import { platformBrowserDynamic } from '\@angular/platform-browser-dynamic';
     * import { AppModule } from './app/app.module';
     *
     * platformBrowserDynamic().bootstrapModule(AppModule, {
     *   providers: [{provide: LOCALE_ID, useValue: 'en-US' }]
     * });
     * ```
     *
     * \@publicApi
     * @type {?}
     */
    const LOCALE_ID$1 = new InjectionToken('LocaleId');
    /**
     * Provide this token to set the default currency code your application uses for
     * CurrencyPipe when there is no currency code passed into it. This is only used by
     * CurrencyPipe and has no relation to locale currency. Defaults to USD if not configured.
     *
     * See the [i18n guide](guide/i18n#setting-up-locale) for more information.
     *
     * <div class="alert is-helpful">
     *
     * **Deprecation notice:**
     *
     * The default currency code is currently always `USD` but this is deprecated from v9.
     *
     * **In v10 the default currency code will be taken from the current locale.**
     *
     * If you need the previous behavior then set it by creating a `DEFAULT_CURRENCY_CODE` provider in
     * your application `NgModule`:
     *
     * ```ts
     * {provide: DEFAULT_CURRENCY_CODE, useValue: 'USD'}
     * ```
     *
     * </div>
     *
     * \@usageNotes
     * ### Example
     *
     * ```typescript
     * import { platformBrowserDynamic } from '\@angular/platform-browser-dynamic';
     * import { AppModule } from './app/app.module';
     *
     * platformBrowserDynamic().bootstrapModule(AppModule, {
     *   providers: [{provide: DEFAULT_CURRENCY_CODE, useValue: 'EUR' }]
     * });
     * ```
     *
     * \@publicApi
     * @type {?}
     */
    const DEFAULT_CURRENCY_CODE = new InjectionToken('DefaultCurrencyCode');
    /**
     * Use this token at bootstrap to provide the content of your translation file (`xtb`,
     * `xlf` or `xlf2`) when you want to translate your application in another language.
     *
     * See the [i18n guide](guide/i18n#merge) for more information.
     *
     * \@usageNotes
     * ### Example
     *
     * ```typescript
     * import { TRANSLATIONS } from '\@angular/core';
     * import { platformBrowserDynamic } from '\@angular/platform-browser-dynamic';
     * import { AppModule } from './app/app.module';
     *
     * // content of your translation file
     * const translations = '....';
     *
     * platformBrowserDynamic().bootstrapModule(AppModule, {
     *   providers: [{provide: TRANSLATIONS, useValue: translations }]
     * });
     * ```
     *
     * \@publicApi
     * @type {?}
     */
    const TRANSLATIONS = new InjectionToken('Translations');
    /**
     * Provide this token at bootstrap to set the format of your {\@link TRANSLATIONS}: `xtb`,
     * `xlf` or `xlf2`.
     *
     * See the [i18n guide](guide/i18n#merge) for more information.
     *
     * \@usageNotes
     * ### Example
     *
     * ```typescript
     * import { TRANSLATIONS_FORMAT } from '\@angular/core';
     * import { platformBrowserDynamic } from '\@angular/platform-browser-dynamic';
     * import { AppModule } from './app/app.module';
     *
     * platformBrowserDynamic().bootstrapModule(AppModule, {
     *   providers: [{provide: TRANSLATIONS_FORMAT, useValue: 'xlf' }]
     * });
     * ```
     *
     * \@publicApi
     * @type {?}
     */
    const TRANSLATIONS_FORMAT = new InjectionToken('TranslationsFormat');

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/ivy_switch.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /** @type {?} */
    const SWITCH_IVY_ENABLED__POST_R3__ = true;
    /** @type {?} */
    const ivyEnabled = SWITCH_IVY_ENABLED__POST_R3__;

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/linker/compiler.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Combination of NgModuleFactory and ComponentFactorys.
     *
     * \@publicApi
     * @template T
     */
    class ModuleWithComponentFactories {
        /**
         * @param {?} ngModuleFactory
         * @param {?} componentFactories
         */
        constructor(ngModuleFactory, componentFactories) {
            this.ngModuleFactory = ngModuleFactory;
            this.componentFactories = componentFactories;
        }
    }
    /** @type {?} */
    const Compiler_compileModuleSync__POST_R3__ = (/**
     * @template T
     * @param {?} moduleType
     * @return {?}
     */
    function (moduleType) {
        return new NgModuleFactory$1(moduleType);
    });
    /** @type {?} */
    const Compiler_compileModuleSync = Compiler_compileModuleSync__POST_R3__;
    /** @type {?} */
    const Compiler_compileModuleAsync__POST_R3__ = (/**
     * @template T
     * @param {?} moduleType
     * @return {?}
     */
    function (moduleType) {
        return Promise.resolve(Compiler_compileModuleSync__POST_R3__(moduleType));
    });
    /** @type {?} */
    const Compiler_compileModuleAsync = Compiler_compileModuleAsync__POST_R3__;
    /** @type {?} */
    const Compiler_compileModuleAndAllComponentsSync__POST_R3__ = (/**
     * @template T
     * @param {?} moduleType
     * @return {?}
     */
    function (moduleType) {
        /** @type {?} */
        const ngModuleFactory = Compiler_compileModuleSync__POST_R3__(moduleType);
        /** @type {?} */
        const moduleDef = (/** @type {?} */ (getNgModuleDef(moduleType)));
        /** @type {?} */
        const componentFactories = maybeUnwrapFn(moduleDef.declarations)
            .reduce((/**
         * @param {?} factories
         * @param {?} declaration
         * @return {?}
         */
        (factories, declaration) => {
            /** @type {?} */
            const componentDef = getComponentDef(declaration);
            componentDef && factories.push(new ComponentFactory$1(componentDef));
            return factories;
        }), (/** @type {?} */ ([])));
        return new ModuleWithComponentFactories(ngModuleFactory, componentFactories);
    });
    /** @type {?} */
    const Compiler_compileModuleAndAllComponentsSync = Compiler_compileModuleAndAllComponentsSync__POST_R3__;
    /** @type {?} */
    const Compiler_compileModuleAndAllComponentsAsync__POST_R3__ = (/**
     * @template T
     * @param {?} moduleType
     * @return {?}
     */
    function (moduleType) {
        return Promise.resolve(Compiler_compileModuleAndAllComponentsSync__POST_R3__(moduleType));
    });
    /** @type {?} */
    const Compiler_compileModuleAndAllComponentsAsync = Compiler_compileModuleAndAllComponentsAsync__POST_R3__;
    /**
     * Low-level service for running the angular compiler during runtime
     * to create {\@link ComponentFactory}s, which
     * can later be used to create and render a Component instance.
     *
     * Each `\@NgModule` provides an own `Compiler` to its injector,
     * that will use the directives/pipes of the ng module for compilation
     * of components.
     *
     * \@publicApi
     */
    class Compiler {
        constructor() {
            /**
             * Compiles the given NgModule and all of its components. All templates of the components listed
             * in `entryComponents` have to be inlined.
             */
            this.compileModuleSync = Compiler_compileModuleSync;
            /**
             * Compiles the given NgModule and all of its components
             */
            this.compileModuleAsync = Compiler_compileModuleAsync;
            /**
             * Same as {\@link #compileModuleSync} but also creates ComponentFactories for all components.
             */
            this.compileModuleAndAllComponentsSync = Compiler_compileModuleAndAllComponentsSync;
            /**
             * Same as {\@link #compileModuleAsync} but also creates ComponentFactories for all components.
             */
            this.compileModuleAndAllComponentsAsync = Compiler_compileModuleAndAllComponentsAsync;
        }
        /**
         * Clears all caches.
         * @return {?}
         */
        clearCache() { }
        /**
         * Clears the cache for the given component/ngModule.
         * @param {?} type
         * @return {?}
         */
        clearCacheFor(type) { }
        /**
         * Returns the id for a given NgModule, if one is defined and known to the compiler.
         * @param {?} moduleType
         * @return {?}
         */
        getModuleId(moduleType) {
            return undefined;
        }
    }
    Compiler.ɵfac = function Compiler_Factory(t) { return new (t || Compiler)(); };
    Compiler.ɵprov = ɵɵdefineInjectable({ token: Compiler, factory: Compiler.ɵfac });
    /**
     * Token to provide CompilerOptions in the platform injector.
     *
     * \@publicApi
     * @type {?}
     */
    const COMPILER_OPTIONS = new InjectionToken('compilerOptions');

    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    const promise = (() => Promise.resolve(0))();
    function scheduleMicroTask(fn) {
        if (typeof Zone === 'undefined') {
            // use promise to schedule microTask instead of use Zone
            promise.then(() => {
                fn && fn.apply(null, null);
            });
        }
        else {
            Zone.current.scheduleMicroTask('scheduleMicrotask', fn);
        }
    }

    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    function getNativeRequestAnimationFrame() {
        let nativeRequestAnimationFrame = _global['requestAnimationFrame'];
        let nativeCancelAnimationFrame = _global['cancelAnimationFrame'];
        if (typeof Zone !== 'undefined' && nativeRequestAnimationFrame && nativeCancelAnimationFrame) {
            // use unpatched version of requestAnimationFrame(native delegate) if possible
            // to avoid another Change detection
            const unpatchedRequestAnimationFrame = nativeRequestAnimationFrame[Zone.__symbol__('OriginalDelegate')];
            if (unpatchedRequestAnimationFrame) {
                nativeRequestAnimationFrame = unpatchedRequestAnimationFrame;
            }
            const unpatchedCancelAnimationFrame = nativeCancelAnimationFrame[Zone.__symbol__('OriginalDelegate')];
            if (unpatchedCancelAnimationFrame) {
                nativeCancelAnimationFrame = unpatchedCancelAnimationFrame;
            }
        }
        return { nativeRequestAnimationFrame, nativeCancelAnimationFrame };
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/zone/ng_zone.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * An injectable service for executing work inside or outside of the Angular zone.
     *
     * The most common use of this service is to optimize performance when starting a work consisting of
     * one or more asynchronous tasks that don't require UI updates or error handling to be handled by
     * Angular. Such tasks can be kicked off via {\@link #runOutsideAngular} and if needed, these tasks
     * can reenter the Angular zone via {\@link #run}.
     *
     * <!-- TODO: add/fix links to:
     *   - docs explaining zones and the use of zones in Angular and change-detection
     *   - link to runOutsideAngular/run (throughout this file!)
     *   -->
     *
     * \@usageNotes
     * ### Example
     *
     * ```
     * import {Component, NgZone} from '\@angular/core';
     * import {NgIf} from '\@angular/common';
     *
     * \@Component({
     *   selector: 'ng-zone-demo',
     *   template: `
     *     <h2>Demo: NgZone</h2>
     *
     *     <p>Progress: {{progress}}%</p>
     *     <p *ngIf="progress >= 100">Done processing {{label}} of Angular zone!</p>
     *
     *     <button (click)="processWithinAngularZone()">Process within Angular zone</button>
     *     <button (click)="processOutsideOfAngularZone()">Process outside of Angular zone</button>
     *   `,
     * })
     * export class NgZoneDemo {
     *   progress: number = 0;
     *   label: string;
     *
     *   constructor(private _ngZone: NgZone) {}
     *
     *   // Loop inside the Angular zone
     *   // so the UI DOES refresh after each setTimeout cycle
     *   processWithinAngularZone() {
     *     this.label = 'inside';
     *     this.progress = 0;
     *     this._increaseProgress(() => console.log('Inside Done!'));
     *   }
     *
     *   // Loop outside of the Angular zone
     *   // so the UI DOES NOT refresh after each setTimeout cycle
     *   processOutsideOfAngularZone() {
     *     this.label = 'outside';
     *     this.progress = 0;
     *     this._ngZone.runOutsideAngular(() => {
     *       this._increaseProgress(() => {
     *         // reenter the Angular zone and display done
     *         this._ngZone.run(() => { console.log('Outside Done!'); });
     *       });
     *     });
     *   }
     *
     *   _increaseProgress(doneCallback: () => void) {
     *     this.progress += 1;
     *     console.log(`Current progress: ${this.progress}%`);
     *
     *     if (this.progress < 100) {
     *       window.setTimeout(() => this._increaseProgress(doneCallback), 10);
     *     } else {
     *       doneCallback();
     *     }
     *   }
     * }
     * ```
     *
     * \@publicApi
     */
    class NgZone {
        /**
         * @param {?} __0
         */
        constructor({ enableLongStackTrace = false, shouldCoalesceEventChangeDetection = false }) {
            this.hasPendingMacrotasks = false;
            this.hasPendingMicrotasks = false;
            /**
             * Whether there are no outstanding microtasks or macrotasks.
             */
            this.isStable = true;
            /**
             * Notifies when code enters Angular Zone. This gets fired first on VM Turn.
             */
            this.onUnstable = new EventEmitter(false);
            /**
             * Notifies when there is no more microtasks enqueued in the current VM Turn.
             * This is a hint for Angular to do change detection, which may enqueue more microtasks.
             * For this reason this event can fire multiple times per VM Turn.
             */
            this.onMicrotaskEmpty = new EventEmitter(false);
            /**
             * Notifies when the last `onMicrotaskEmpty` has run and there are no more microtasks, which
             * implies we are about to relinquish VM turn.
             * This event gets called just once.
             */
            this.onStable = new EventEmitter(false);
            /**
             * Notifies that an error has been delivered.
             */
            this.onError = new EventEmitter(false);
            if (typeof Zone == 'undefined') {
                throw new Error(`In this configuration Angular requires Zone.js`);
            }
            Zone.assertZonePatched();
            /** @type {?} */
            const self = (/** @type {?} */ ((/** @type {?} */ (this))));
            self._nesting = 0;
            self._outer = self._inner = Zone.current;
            if (((/** @type {?} */ (Zone)))['wtfZoneSpec']) {
                self._inner = self._inner.fork(((/** @type {?} */ (Zone)))['wtfZoneSpec']);
            }
            if (((/** @type {?} */ (Zone)))['TaskTrackingZoneSpec']) {
                self._inner = self._inner.fork(new ((/** @type {?} */ (((/** @type {?} */ (Zone)))['TaskTrackingZoneSpec']))));
            }
            if (enableLongStackTrace && ((/** @type {?} */ (Zone)))['longStackTraceZoneSpec']) {
                self._inner = self._inner.fork(((/** @type {?} */ (Zone)))['longStackTraceZoneSpec']);
            }
            self.shouldCoalesceEventChangeDetection = shouldCoalesceEventChangeDetection;
            self.lastRequestAnimationFrameId = -1;
            self.nativeRequestAnimationFrame = getNativeRequestAnimationFrame().nativeRequestAnimationFrame;
            forkInnerZoneWithAngularBehavior(self);
        }
        /**
         * @return {?}
         */
        static isInAngularZone() {
            return Zone.current.get('isAngularZone') === true;
        }
        /**
         * @return {?}
         */
        static assertInAngularZone() {
            if (!NgZone.isInAngularZone()) {
                throw new Error('Expected to be in Angular Zone, but it is not!');
            }
        }
        /**
         * @return {?}
         */
        static assertNotInAngularZone() {
            if (NgZone.isInAngularZone()) {
                throw new Error('Expected to not be in Angular Zone, but it is!');
            }
        }
        /**
         * Executes the `fn` function synchronously within the Angular zone and returns value returned by
         * the function.
         *
         * Running functions via `run` allows you to reenter Angular zone from a task that was executed
         * outside of the Angular zone (typically started via {\@link #runOutsideAngular}).
         *
         * Any future tasks or microtasks scheduled from within this function will continue executing from
         * within the Angular zone.
         *
         * If a synchronous error happens it will be rethrown and not reported via `onError`.
         * @template T
         * @param {?} fn
         * @param {?=} applyThis
         * @param {?=} applyArgs
         * @return {?}
         */
        run(fn, applyThis, applyArgs) {
            return (/** @type {?} */ (((/** @type {?} */ ((/** @type {?} */ (this)))))._inner.run(fn, applyThis, applyArgs)));
        }
        /**
         * Executes the `fn` function synchronously within the Angular zone as a task and returns value
         * returned by the function.
         *
         * Running functions via `run` allows you to reenter Angular zone from a task that was executed
         * outside of the Angular zone (typically started via {\@link #runOutsideAngular}).
         *
         * Any future tasks or microtasks scheduled from within this function will continue executing from
         * within the Angular zone.
         *
         * If a synchronous error happens it will be rethrown and not reported via `onError`.
         * @template T
         * @param {?} fn
         * @param {?=} applyThis
         * @param {?=} applyArgs
         * @param {?=} name
         * @return {?}
         */
        runTask(fn, applyThis, applyArgs, name) {
            /** @type {?} */
            const zone = ((/** @type {?} */ ((/** @type {?} */ (this)))))._inner;
            /** @type {?} */
            const task = zone.scheduleEventTask('NgZoneEvent: ' + name, fn, EMPTY_PAYLOAD, noop$1, noop$1);
            try {
                return (/** @type {?} */ (zone.runTask(task, applyThis, applyArgs)));
            }
            finally {
                zone.cancelTask(task);
            }
        }
        /**
         * Same as `run`, except that synchronous errors are caught and forwarded via `onError` and not
         * rethrown.
         * @template T
         * @param {?} fn
         * @param {?=} applyThis
         * @param {?=} applyArgs
         * @return {?}
         */
        runGuarded(fn, applyThis, applyArgs) {
            return (/** @type {?} */ (((/** @type {?} */ ((/** @type {?} */ (this)))))._inner.runGuarded(fn, applyThis, applyArgs)));
        }
        /**
         * Executes the `fn` function synchronously in Angular's parent zone and returns value returned by
         * the function.
         *
         * Running functions via {\@link #runOutsideAngular} allows you to escape Angular's zone and do
         * work that
         * doesn't trigger Angular change-detection or is subject to Angular's error handling.
         *
         * Any future tasks or microtasks scheduled from within this function will continue executing from
         * outside of the Angular zone.
         *
         * Use {\@link #run} to reenter the Angular zone and do work that updates the application model.
         * @template T
         * @param {?} fn
         * @return {?}
         */
        runOutsideAngular(fn) {
            return (/** @type {?} */ (((/** @type {?} */ ((/** @type {?} */ (this)))))._outer.run(fn)));
        }
    }
    /**
     * @return {?}
     */
    function noop$1() { }
    /** @type {?} */
    const EMPTY_PAYLOAD = {};
    /**
     * @param {?} zone
     * @return {?}
     */
    function checkStable(zone) {
        if (zone._nesting == 0 && !zone.hasPendingMicrotasks && !zone.isStable) {
            try {
                zone._nesting++;
                zone.onMicrotaskEmpty.emit(null);
            }
            finally {
                zone._nesting--;
                if (!zone.hasPendingMicrotasks) {
                    try {
                        zone.runOutsideAngular((/**
                         * @return {?}
                         */
                        () => zone.onStable.emit(null)));
                    }
                    finally {
                        zone.isStable = true;
                    }
                }
            }
        }
    }
    /**
     * @param {?} zone
     * @return {?}
     */
    function delayChangeDetectionForEvents(zone) {
        if (zone.lastRequestAnimationFrameId !== -1) {
            return;
        }
        zone.lastRequestAnimationFrameId = zone.nativeRequestAnimationFrame.call(_global, (/**
         * @return {?}
         */
        () => {
            zone.lastRequestAnimationFrameId = -1;
            updateMicroTaskStatus(zone);
            checkStable(zone);
        }));
        updateMicroTaskStatus(zone);
    }
    /**
     * @param {?} zone
     * @return {?}
     */
    function forkInnerZoneWithAngularBehavior(zone) {
        /** @type {?} */
        const delayChangeDetectionForEventsDelegate = (/**
         * @return {?}
         */
        () => {
            delayChangeDetectionForEvents(zone);
        });
        /** @type {?} */
        const maybeDelayChangeDetection = !!zone.shouldCoalesceEventChangeDetection &&
            zone.nativeRequestAnimationFrame && delayChangeDetectionForEventsDelegate;
        zone._inner = zone._inner.fork({
            name: 'angular',
            properties: (/** @type {?} */ ({ 'isAngularZone': true, 'maybeDelayChangeDetection': maybeDelayChangeDetection })),
            onInvokeTask: (/**
             * @param {?} delegate
             * @param {?} current
             * @param {?} target
             * @param {?} task
             * @param {?} applyThis
             * @param {?} applyArgs
             * @return {?}
             */
            (delegate, current, target, task, applyThis, applyArgs) => {
                try {
                    onEnter(zone);
                    return delegate.invokeTask(target, task, applyThis, applyArgs);
                }
                finally {
                    if (maybeDelayChangeDetection && task.type === 'eventTask') {
                        maybeDelayChangeDetection();
                    }
                    onLeave(zone);
                }
            }),
            onInvoke: (/**
             * @param {?} delegate
             * @param {?} current
             * @param {?} target
             * @param {?} callback
             * @param {?} applyThis
             * @param {?=} applyArgs
             * @param {?=} source
             * @return {?}
             */
            (delegate, current, target, callback, applyThis, applyArgs, source) => {
                try {
                    onEnter(zone);
                    return delegate.invoke(target, callback, applyThis, applyArgs, source);
                }
                finally {
                    onLeave(zone);
                }
            }),
            onHasTask: (/**
             * @param {?} delegate
             * @param {?} current
             * @param {?} target
             * @param {?} hasTaskState
             * @return {?}
             */
            (delegate, current, target, hasTaskState) => {
                delegate.hasTask(target, hasTaskState);
                if (current === target) {
                    // We are only interested in hasTask events which originate from our zone
                    // (A child hasTask event is not interesting to us)
                    if (hasTaskState.change == 'microTask') {
                        zone._hasPendingMicrotasks = hasTaskState.microTask;
                        updateMicroTaskStatus(zone);
                        checkStable(zone);
                    }
                    else if (hasTaskState.change == 'macroTask') {
                        zone.hasPendingMacrotasks = hasTaskState.macroTask;
                    }
                }
            }),
            onHandleError: (/**
             * @param {?} delegate
             * @param {?} current
             * @param {?} target
             * @param {?} error
             * @return {?}
             */
            (delegate, current, target, error) => {
                delegate.handleError(target, error);
                zone.runOutsideAngular((/**
                 * @return {?}
                 */
                () => zone.onError.emit(error)));
                return false;
            })
        });
    }
    /**
     * @param {?} zone
     * @return {?}
     */
    function updateMicroTaskStatus(zone) {
        if (zone._hasPendingMicrotasks ||
            (zone.shouldCoalesceEventChangeDetection && zone.lastRequestAnimationFrameId !== -1)) {
            zone.hasPendingMicrotasks = true;
        }
        else {
            zone.hasPendingMicrotasks = false;
        }
    }
    /**
     * @param {?} zone
     * @return {?}
     */
    function onEnter(zone) {
        zone._nesting++;
        if (zone.isStable) {
            zone.isStable = false;
            zone.onUnstable.emit(null);
        }
    }
    /**
     * @param {?} zone
     * @return {?}
     */
    function onLeave(zone) {
        zone._nesting--;
        checkStable(zone);
    }
    /**
     * Provides a noop implementation of `NgZone` which does nothing. This zone requires explicit calls
     * to framework to perform rendering.
     */
    class NoopNgZone {
        constructor() {
            this.hasPendingMicrotasks = false;
            this.hasPendingMacrotasks = false;
            this.isStable = true;
            this.onUnstable = new EventEmitter();
            this.onMicrotaskEmpty = new EventEmitter();
            this.onStable = new EventEmitter();
            this.onError = new EventEmitter();
        }
        /**
         * @param {?} fn
         * @param {?=} applyThis
         * @param {?=} applyArgs
         * @return {?}
         */
        run(fn, applyThis, applyArgs) {
            return fn.apply(applyThis, applyArgs);
        }
        /**
         * @param {?} fn
         * @param {?=} applyThis
         * @param {?=} applyArgs
         * @return {?}
         */
        runGuarded(fn, applyThis, applyArgs) {
            return fn.apply(applyThis, applyArgs);
        }
        /**
         * @param {?} fn
         * @return {?}
         */
        runOutsideAngular(fn) {
            return fn();
        }
        /**
         * @param {?} fn
         * @param {?=} applyThis
         * @param {?=} applyArgs
         * @param {?=} name
         * @return {?}
         */
        runTask(fn, applyThis, applyArgs, name) {
            return fn.apply(applyThis, applyArgs);
        }
    }
    /**
     * The Testability service provides testing hooks that can be accessed from
     * the browser and by services such as Protractor. Each bootstrapped Angular
     * application on the page will have an instance of Testability.
     * \@publicApi
     */
    class Testability {
        /**
         * @param {?} _ngZone
         */
        constructor(_ngZone) {
            this._ngZone = _ngZone;
            this._pendingCount = 0;
            this._isZoneStable = true;
            /**
             * Whether any work was done since the last 'whenStable' callback. This is
             * useful to detect if this could have potentially destabilized another
             * component while it is stabilizing.
             * \@internal
             */
            this._didWork = false;
            this._callbacks = [];
            this.taskTrackingZone = null;
            this._watchAngularEvents();
            _ngZone.run((/**
             * @return {?}
             */
            () => {
                this.taskTrackingZone =
                    typeof Zone == 'undefined' ? null : Zone.current.get('TaskTrackingZone');
            }));
        }
        /**
         * @private
         * @return {?}
         */
        _watchAngularEvents() {
            this._ngZone.onUnstable.subscribe({
                next: (/**
                 * @return {?}
                 */
                () => {
                    this._didWork = true;
                    this._isZoneStable = false;
                })
            });
            this._ngZone.runOutsideAngular((/**
             * @return {?}
             */
            () => {
                this._ngZone.onStable.subscribe({
                    next: (/**
                     * @return {?}
                     */
                    () => {
                        NgZone.assertNotInAngularZone();
                        scheduleMicroTask((/**
                         * @return {?}
                         */
                        () => {
                            this._isZoneStable = true;
                            this._runCallbacksIfReady();
                        }));
                    })
                });
            }));
        }
        /**
         * Increases the number of pending request
         * @deprecated pending requests are now tracked with zones.
         * @return {?}
         */
        increasePendingRequestCount() {
            this._pendingCount += 1;
            this._didWork = true;
            return this._pendingCount;
        }
        /**
         * Decreases the number of pending request
         * @deprecated pending requests are now tracked with zones
         * @return {?}
         */
        decreasePendingRequestCount() {
            this._pendingCount -= 1;
            if (this._pendingCount < 0) {
                throw new Error('pending async requests below zero');
            }
            this._runCallbacksIfReady();
            return this._pendingCount;
        }
        /**
         * Whether an associated application is stable
         * @return {?}
         */
        isStable() {
            return this._isZoneStable && this._pendingCount === 0 && !this._ngZone.hasPendingMacrotasks;
        }
        /**
         * @private
         * @return {?}
         */
        _runCallbacksIfReady() {
            if (this.isStable()) {
                // Schedules the call backs in a new frame so that it is always async.
                scheduleMicroTask((/**
                 * @return {?}
                 */
                () => {
                    while (this._callbacks.length !== 0) {
                        /** @type {?} */
                        let cb = (/** @type {?} */ (this._callbacks.pop()));
                        clearTimeout(cb.timeoutId);
                        cb.doneCb(this._didWork);
                    }
                    this._didWork = false;
                }));
            }
            else {
                // Still not stable, send updates.
                /** @type {?} */
                let pending = this.getPendingTasks();
                this._callbacks = this._callbacks.filter((/**
                 * @param {?} cb
                 * @return {?}
                 */
                (cb) => {
                    if (cb.updateCb && cb.updateCb(pending)) {
                        clearTimeout(cb.timeoutId);
                        return false;
                    }
                    return true;
                }));
                this._didWork = true;
            }
        }
        /**
         * @private
         * @return {?}
         */
        getPendingTasks() {
            if (!this.taskTrackingZone) {
                return [];
            }
            // Copy the tasks data so that we don't leak tasks.
            return this.taskTrackingZone.macroTasks.map((/**
             * @param {?} t
             * @return {?}
             */
            (t) => {
                return {
                    source: t.source,
                    // From TaskTrackingZone:
                    // https://github.com/angular/zone.js/blob/master/lib/zone-spec/task-tracking.ts#L40
                    creationLocation: (/** @type {?} */ (((/** @type {?} */ (t))).creationLocation)),
                    data: t.data
                };
            }));
        }
        /**
         * @private
         * @param {?} cb
         * @param {?=} timeout
         * @param {?=} updateCb
         * @return {?}
         */
        addCallback(cb, timeout, updateCb) {
            /** @type {?} */
            let timeoutId = -1;
            if (timeout && timeout > 0) {
                timeoutId = setTimeout((/**
                 * @return {?}
                 */
                () => {
                    this._callbacks = this._callbacks.filter((/**
                     * @param {?} cb
                     * @return {?}
                     */
                    (cb) => cb.timeoutId !== timeoutId));
                    cb(this._didWork, this.getPendingTasks());
                }), timeout);
            }
            this._callbacks.push((/** @type {?} */ ({ doneCb: cb, timeoutId: timeoutId, updateCb: updateCb })));
        }
        /**
         * Wait for the application to be stable with a timeout. If the timeout is reached before that
         * happens, the callback receives a list of the macro tasks that were pending, otherwise null.
         *
         * @param {?} doneCb The callback to invoke when Angular is stable or the timeout expires
         *    whichever comes first.
         * @param {?=} timeout Optional. The maximum time to wait for Angular to become stable. If not
         *    specified, whenStable() will wait forever.
         * @param {?=} updateCb Optional. If specified, this callback will be invoked whenever the set of
         *    pending macrotasks changes. If this callback returns true doneCb will not be invoked
         *    and no further updates will be issued.
         * @return {?}
         */
        whenStable(doneCb, timeout, updateCb) {
            if (updateCb && !this.taskTrackingZone) {
                throw new Error('Task tracking zone is required when passing an update callback to ' +
                    'whenStable(). Is "zone.js/dist/task-tracking.js" loaded?');
            }
            // These arguments are 'Function' above to keep the public API simple.
            this.addCallback((/** @type {?} */ (doneCb)), timeout, (/** @type {?} */ (updateCb)));
            this._runCallbacksIfReady();
        }
        /**
         * Get the number of pending requests
         * @deprecated pending requests are now tracked with zones
         * @return {?}
         */
        getPendingRequestCount() {
            return this._pendingCount;
        }
        /**
         * Find providers by name
         * @param {?} using The root element to search from
         * @param {?} provider The name of binding variable
         * @param {?} exactMatch Whether using exactMatch
         * @return {?}
         */
        findProviders(using, provider, exactMatch) {
            // TODO(juliemr): implement.
            return [];
        }
    }
    Testability.ɵfac = function Testability_Factory(t) { return new (t || Testability)(ɵɵinject(NgZone)); };
    Testability.ɵprov = ɵɵdefineInjectable({ token: Testability, factory: Testability.ɵfac });
    /** @nocollapse */
    Testability.ctorParameters = () => [
        { type: NgZone }
    ];
    /**
     * A global registry of {\@link Testability} instances for specific elements.
     * \@publicApi
     */
    class TestabilityRegistry {
        constructor() {
            /**
             * \@internal
             */
            this._applications = new Map();
            _testabilityGetter.addToWindow(this);
        }
        /**
         * Registers an application with a testability hook so that it can be tracked
         * @param {?} token token of application, root element
         * @param {?} testability Testability hook
         * @return {?}
         */
        registerApplication(token, testability) {
            this._applications.set(token, testability);
        }
        /**
         * Unregisters an application.
         * @param {?} token token of application, root element
         * @return {?}
         */
        unregisterApplication(token) {
            this._applications.delete(token);
        }
        /**
         * Unregisters all applications
         * @return {?}
         */
        unregisterAllApplications() {
            this._applications.clear();
        }
        /**
         * Get a testability hook associated with the application
         * @param {?} elem root element
         * @return {?}
         */
        getTestability(elem) {
            return this._applications.get(elem) || null;
        }
        /**
         * Get all registered testabilities
         * @return {?}
         */
        getAllTestabilities() {
            return Array.from(this._applications.values());
        }
        /**
         * Get all registered applications(root elements)
         * @return {?}
         */
        getAllRootElements() {
            return Array.from(this._applications.keys());
        }
        /**
         * Find testability of a node in the Tree
         * @param {?} elem node
         * @param {?=} findInAncestors whether finding testability in ancestors if testability was not found in
         * current node
         * @return {?}
         */
        findTestabilityInTree(elem, findInAncestors = true) {
            return _testabilityGetter.findTestabilityInTree(this, elem, findInAncestors);
        }
    }
    TestabilityRegistry.ɵfac = function TestabilityRegistry_Factory(t) { return new (t || TestabilityRegistry)(); };
    TestabilityRegistry.ɵprov = ɵɵdefineInjectable({ token: TestabilityRegistry, factory: TestabilityRegistry.ɵfac });
    /** @nocollapse */
    TestabilityRegistry.ctorParameters = () => [];
    class _NoopGetTestability {
        /**
         * @param {?} registry
         * @return {?}
         */
        addToWindow(registry) { }
        /**
         * @param {?} registry
         * @param {?} elem
         * @param {?} findInAncestors
         * @return {?}
         */
        findTestabilityInTree(registry, elem, findInAncestors) {
            return null;
        }
    }
    /**
     * Set the {\@link GetTestability} implementation used by the Angular testing framework.
     * \@publicApi
     * @param {?} getter
     * @return {?}
     */
    function setTestabilityGetter(getter) {
        _testabilityGetter = getter;
    }
    /** @type {?} */
    let _testabilityGetter = new _NoopGetTestability();

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/application_ref.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    let _platform;
    /** @type {?} */
    let compileNgModuleFactory = compileNgModuleFactory__POST_R3__;
    /**
     * @template M
     * @param {?} injector
     * @param {?} options
     * @param {?} moduleType
     * @return {?}
     */
    function compileNgModuleFactory__POST_R3__(injector, options, moduleType) {
        ngDevMode && assertNgModuleType(moduleType);
        /** @type {?} */
        const compilerOptions = injector.get(COMPILER_OPTIONS, []).concat(options);
        if (typeof ngJitMode === 'undefined' || ngJitMode) {
            // Configure the compiler to use the provided options. This call may fail when multiple modules
            // are bootstrapped with incompatible options, as a component can only be compiled according to
            // a single set of options.
            setJitOptions({
                defaultEncapsulation: _lastDefined(compilerOptions.map((/**
                 * @param {?} options
                 * @return {?}
                 */
                options => options.defaultEncapsulation))),
                preserveWhitespaces: _lastDefined(compilerOptions.map((/**
                 * @param {?} options
                 * @return {?}
                 */
                options => options.preserveWhitespaces))),
            });
        }
        /** @type {?} */
        const moduleFactory = new NgModuleFactory$1(moduleType);
        if (isComponentResourceResolutionQueueEmpty()) {
            return Promise.resolve(moduleFactory);
        }
        /** @type {?} */
        const compilerProviders = _mergeArrays(compilerOptions.map((/**
         * @param {?} o
         * @return {?}
         */
        o => (/** @type {?} */ (o.providers)))));
        // In case there are no compiler providers, we just return the module factory as
        // there won't be any resource loader. This can happen with Ivy, because AOT compiled
        // modules can be still passed through "bootstrapModule". In that case we shouldn't
        // unnecessarily require the JIT compiler.
        if (compilerProviders.length === 0) {
            return Promise.resolve(moduleFactory);
        }
        /** @type {?} */
        const compiler = getCompilerFacade();
        /** @type {?} */
        const compilerInjector = Injector.create({ providers: compilerProviders });
        /** @type {?} */
        const resourceLoader = compilerInjector.get(compiler.ResourceLoader);
        // The resource loader can also return a string while the "resolveComponentResources"
        // always expects a promise. Therefore we need to wrap the returned value in a promise.
        return resolveComponentResources((/**
         * @param {?} url
         * @return {?}
         */
        url => Promise.resolve(resourceLoader.get(url))))
            .then((/**
         * @return {?}
         */
        () => moduleFactory));
    }
    /**
     * @return {?}
     */
    function publishDefaultGlobalUtils__POST_R3__() {
        ngDevMode && publishDefaultGlobalUtils();
    }
    /** @type {?} */
    let publishDefaultGlobalUtils$1 = publishDefaultGlobalUtils__POST_R3__;
    /** @type {?} */
    let isBoundToModule = isBoundToModule__POST_R3__;
    /**
     * @template C
     * @param {?} cf
     * @return {?}
     */
    function isBoundToModule__POST_R3__(cf) {
        return ((/** @type {?} */ (cf))).isBoundToModule;
    }
    /** @type {?} */
    const ALLOW_MULTIPLE_PLATFORMS = new InjectionToken('AllowMultipleToken');
    /**
     * A token for third-party components that can register themselves with NgProbe.
     *
     * \@publicApi
     */
    class NgProbeToken {
        /**
         * @param {?} name
         * @param {?} token
         */
        constructor(name, token) {
            this.name = name;
            this.token = token;
        }
    }
    /**
     * Creates a platform.
     * Platforms have to be eagerly created via this function.
     *
     * \@publicApi
     * @param {?} injector
     * @return {?}
     */
    function createPlatform(injector) {
        if (_platform && !_platform.destroyed &&
            !_platform.injector.get(ALLOW_MULTIPLE_PLATFORMS, false)) {
            throw new Error('There can be only one platform. Destroy the previous one to create a new one.');
        }
        publishDefaultGlobalUtils$1();
        _platform = injector.get(PlatformRef);
        /** @type {?} */
        const inits = injector.get(PLATFORM_INITIALIZER, null);
        if (inits)
            inits.forEach((/**
             * @param {?} init
             * @return {?}
             */
            (init) => init()));
        return _platform;
    }
    /**
     * Creates a factory for a platform
     *
     * \@publicApi
     * @param {?} parentPlatformFactory
     * @param {?} name
     * @param {?=} providers
     * @return {?}
     */
    function createPlatformFactory(parentPlatformFactory, name, providers = []) {
        /** @type {?} */
        const desc = `Platform: ${name}`;
        /** @type {?} */
        const marker = new InjectionToken(desc);
        return (/**
         * @param {?=} extraProviders
         * @return {?}
         */
        (extraProviders = []) => {
            /** @type {?} */
            let platform = getPlatform();
            if (!platform || platform.injector.get(ALLOW_MULTIPLE_PLATFORMS, false)) {
                if (parentPlatformFactory) {
                    parentPlatformFactory(providers.concat(extraProviders).concat({ provide: marker, useValue: true }));
                }
                else {
                    /** @type {?} */
                    const injectedProviders = providers.concat(extraProviders).concat({ provide: marker, useValue: true }, {
                        provide: INJECTOR_SCOPE,
                        useValue: 'platform'
                    });
                    createPlatform(Injector.create({ providers: injectedProviders, name: desc }));
                }
            }
            return assertPlatform(marker);
        });
    }
    /**
     * Checks that there currently is a platform which contains the given token as a provider.
     *
     * \@publicApi
     * @param {?} requiredToken
     * @return {?}
     */
    function assertPlatform(requiredToken) {
        /** @type {?} */
        const platform = getPlatform();
        if (!platform) {
            throw new Error('No platform exists!');
        }
        if (!platform.injector.get(requiredToken, null)) {
            throw new Error('A platform with a different configuration has been created. Please destroy it first.');
        }
        return platform;
    }
    /**
     * Returns the current platform.
     *
     * \@publicApi
     * @return {?}
     */
    function getPlatform() {
        return _platform && !_platform.destroyed ? _platform : null;
    }
    /**
     * The Angular platform is the entry point for Angular on a web page. Each page
     * has exactly one platform, and services (such as reflection) which are common
     * to every Angular application running on the page are bound in its scope.
     *
     * A page's platform is initialized implicitly when a platform is created via a platform factory
     * (e.g. {\@link platformBrowser}), or explicitly by calling the {\@link createPlatform} function.
     *
     * \@publicApi
     */
    class PlatformRef {
        /**
         * \@internal
         * @param {?} _injector
         */
        constructor(_injector) {
            this._injector = _injector;
            this._modules = [];
            this._destroyListeners = [];
            this._destroyed = false;
        }
        /**
         * Creates an instance of an `\@NgModule` for the given platform
         * for offline compilation.
         *
         * \@usageNotes
         * ### Simple Example
         *
         * ```typescript
         * my_module.ts:
         *
         * \@NgModule({
         *   imports: [BrowserModule]
         * })
         * class MyModule {}
         *
         * main.ts:
         * import {MyModuleNgFactory} from './my_module.ngfactory';
         * import {platformBrowser} from '\@angular/platform-browser';
         *
         * let moduleRef = platformBrowser().bootstrapModuleFactory(MyModuleNgFactory);
         * ```
         * @template M
         * @param {?} moduleFactory
         * @param {?=} options
         * @return {?}
         */
        bootstrapModuleFactory(moduleFactory, options) {
            // Note: We need to create the NgZone _before_ we instantiate the module,
            // as instantiating the module creates some providers eagerly.
            // So we create a mini parent injector that just contains the new NgZone and
            // pass that as parent to the NgModuleFactory.
            /** @type {?} */
            const ngZoneOption = options ? options.ngZone : undefined;
            /** @type {?} */
            const ngZoneEventCoalescing = (options && options.ngZoneEventCoalescing) || false;
            /** @type {?} */
            const ngZone = getNgZone(ngZoneOption, ngZoneEventCoalescing);
            /** @type {?} */
            const providers = [{ provide: NgZone, useValue: ngZone }];
            // Attention: Don't use ApplicationRef.run here,
            // as we want to be sure that all possible constructor calls are inside `ngZone.run`!
            return ngZone.run((/**
             * @return {?}
             */
            () => {
                /** @type {?} */
                const ngZoneInjector = Injector.create({ providers: providers, parent: this.injector, name: moduleFactory.moduleType.name });
                /** @type {?} */
                const moduleRef = (/** @type {?} */ (moduleFactory.create(ngZoneInjector)));
                /** @type {?} */
                const exceptionHandler = moduleRef.injector.get(ErrorHandler, null);
                if (!exceptionHandler) {
                    throw new Error('No ErrorHandler. Is platform module (BrowserModule) included?');
                }
                moduleRef.onDestroy((/**
                 * @return {?}
                 */
                () => remove(this._modules, moduleRef)));
                (/** @type {?} */ (ngZone)).runOutsideAngular((/**
                 * @return {?}
                 */
                () => (/** @type {?} */ (ngZone)).onError.subscribe({
                    next: (/**
                     * @param {?} error
                     * @return {?}
                     */
                    (error) => {
                        exceptionHandler.handleError(error);
                    })
                })));
                return _callAndReportToErrorHandler(exceptionHandler, (/** @type {?} */ (ngZone)), (/**
                 * @return {?}
                 */
                () => {
                    /** @type {?} */
                    const initStatus = moduleRef.injector.get(ApplicationInitStatus);
                    initStatus.runInitializers();
                    return initStatus.donePromise.then((/**
                     * @return {?}
                     */
                    () => {
                        if (ivyEnabled) {
                            // If the `LOCALE_ID` provider is defined at bootstrap then we set the value for ivy
                            /** @type {?} */
                            const localeId = moduleRef.injector.get(LOCALE_ID$1, DEFAULT_LOCALE_ID);
                            setLocaleId(localeId || DEFAULT_LOCALE_ID);
                        }
                        this._moduleDoBootstrap(moduleRef);
                        return moduleRef;
                    }));
                }));
            }));
        }
        /**
         * Creates an instance of an `\@NgModule` for a given platform using the given runtime compiler.
         *
         * \@usageNotes
         * ### Simple Example
         *
         * ```typescript
         * \@NgModule({
         *   imports: [BrowserModule]
         * })
         * class MyModule {}
         *
         * let moduleRef = platformBrowser().bootstrapModule(MyModule);
         * ```
         *
         * @template M
         * @param {?} moduleType
         * @param {?=} compilerOptions
         * @return {?}
         */
        bootstrapModule(moduleType, compilerOptions = []) {
            /** @type {?} */
            const options = optionsReducer({}, compilerOptions);
            return compileNgModuleFactory(this.injector, options, moduleType)
                .then((/**
             * @param {?} moduleFactory
             * @return {?}
             */
            moduleFactory => this.bootstrapModuleFactory(moduleFactory, options)));
        }
        /**
         * @private
         * @param {?} moduleRef
         * @return {?}
         */
        _moduleDoBootstrap(moduleRef) {
            /** @type {?} */
            const appRef = (/** @type {?} */ (moduleRef.injector.get(ApplicationRef)));
            if (moduleRef._bootstrapComponents.length > 0) {
                moduleRef._bootstrapComponents.forEach((/**
                 * @param {?} f
                 * @return {?}
                 */
                f => appRef.bootstrap(f)));
            }
            else if (moduleRef.instance.ngDoBootstrap) {
                moduleRef.instance.ngDoBootstrap(appRef);
            }
            else {
                throw new Error(`The module ${stringify(moduleRef.instance
                .constructor)} was bootstrapped, but it does not declare "@NgModule.bootstrap" components nor a "ngDoBootstrap" method. ` +
                    `Please define one of these.`);
            }
            this._modules.push(moduleRef);
        }
        /**
         * Register a listener to be called when the platform is disposed.
         * @param {?} callback
         * @return {?}
         */
        onDestroy(callback) {
            this._destroyListeners.push(callback);
        }
        /**
         * Retrieve the platform {\@link Injector}, which is the parent injector for
         * every Angular application on the page and provides singleton providers.
         * @return {?}
         */
        get injector() {
            return this._injector;
        }
        /**
         * Destroy the Angular platform and all Angular applications on the page.
         * @return {?}
         */
        destroy() {
            if (this._destroyed) {
                throw new Error('The platform has already been destroyed!');
            }
            this._modules.slice().forEach((/**
             * @param {?} module
             * @return {?}
             */
            module => module.destroy()));
            this._destroyListeners.forEach((/**
             * @param {?} listener
             * @return {?}
             */
            listener => listener()));
            this._destroyed = true;
        }
        /**
         * @return {?}
         */
        get destroyed() {
            return this._destroyed;
        }
    }
    PlatformRef.ɵfac = function PlatformRef_Factory(t) { return new (t || PlatformRef)(ɵɵinject(Injector)); };
    PlatformRef.ɵprov = ɵɵdefineInjectable({ token: PlatformRef, factory: PlatformRef.ɵfac });
    /** @nocollapse */
    PlatformRef.ctorParameters = () => [
        { type: Injector }
    ];
    /**
     * @param {?} ngZoneOption
     * @param {?} ngZoneEventCoalescing
     * @return {?}
     */
    function getNgZone(ngZoneOption, ngZoneEventCoalescing) {
        /** @type {?} */
        let ngZone;
        if (ngZoneOption === 'noop') {
            ngZone = new NoopNgZone();
        }
        else {
            ngZone = (ngZoneOption === 'zone.js' ? undefined : ngZoneOption) || new NgZone({
                enableLongStackTrace: isDevMode(),
                shouldCoalesceEventChangeDetection: ngZoneEventCoalescing
            });
        }
        return ngZone;
    }
    /**
     * @param {?} errorHandler
     * @param {?} ngZone
     * @param {?} callback
     * @return {?}
     */
    function _callAndReportToErrorHandler(errorHandler, ngZone, callback) {
        try {
            /** @type {?} */
            const result = callback();
            if (isPromise$1(result)) {
                return result.catch((/**
                 * @param {?} e
                 * @return {?}
                 */
                (e) => {
                    ngZone.runOutsideAngular((/**
                     * @return {?}
                     */
                    () => errorHandler.handleError(e)));
                    // rethrow as the exception handler might not do it
                    throw e;
                }));
            }
            return result;
        }
        catch (e) {
            ngZone.runOutsideAngular((/**
             * @return {?}
             */
            () => errorHandler.handleError(e)));
            // rethrow as the exception handler might not do it
            throw e;
        }
    }
    /**
     * @template T
     * @param {?} dst
     * @param {?} objs
     * @return {?}
     */
    function optionsReducer(dst, objs) {
        if (Array.isArray(objs)) {
            dst = objs.reduce(optionsReducer, dst);
        }
        else {
            dst = Object.assign(Object.assign({}, dst), ((/** @type {?} */ (objs))));
        }
        return dst;
    }
    /**
     * A reference to an Angular application running on a page.
     *
     * \@usageNotes
     *
     * {\@a is-stable-examples}
     * ### isStable examples and caveats
     *
     * Note two important points about `isStable`, demonstrated in the examples below:
     * - the application will never be stable if you start any kind
     * of recurrent asynchronous task when the application starts
     * (for example for a polling process, started with a `setInterval`, a `setTimeout`
     * or using RxJS operators like `interval`);
     * - the `isStable` Observable runs outside of the Angular zone.
     *
     * Let's imagine that you start a recurrent task
     * (here incrementing a counter, using RxJS `interval`),
     * and at the same time subscribe to `isStable`.
     *
     * ```
     * constructor(appRef: ApplicationRef) {
     *   appRef.isStable.pipe(
     *      filter(stable => stable)
     *   ).subscribe(() => console.log('App is stable now');
     *   interval(1000).subscribe(counter => console.log(counter));
     * }
     * ```
     * In this example, `isStable` will never emit `true`,
     * and the trace "App is stable now" will never get logged.
     *
     * If you want to execute something when the app is stable,
     * you have to wait for the application to be stable
     * before starting your polling process.
     *
     * ```
     * constructor(appRef: ApplicationRef) {
     *   appRef.isStable.pipe(
     *     first(stable => stable),
     *     tap(stable => console.log('App is stable now')),
     *     switchMap(() => interval(1000))
     *   ).subscribe(counter => console.log(counter));
     * }
     * ```
     * In this example, the trace "App is stable now" will be logged
     * and then the counter starts incrementing every second.
     *
     * Note also that this Observable runs outside of the Angular zone,
     * which means that the code in the subscription
     * to this Observable will not trigger the change detection.
     *
     * Let's imagine that instead of logging the counter value,
     * you update a field of your component
     * and display it in its template.
     *
     * ```
     * constructor(appRef: ApplicationRef) {
     *   appRef.isStable.pipe(
     *     first(stable => stable),
     *     switchMap(() => interval(1000))
     *   ).subscribe(counter => this.value = counter);
     * }
     * ```
     * As the `isStable` Observable runs outside the zone,
     * the `value` field will be updated properly,
     * but the template will not be refreshed!
     *
     * You'll have to manually trigger the change detection to update the template.
     *
     * ```
     * constructor(appRef: ApplicationRef, cd: ChangeDetectorRef) {
     *   appRef.isStable.pipe(
     *     first(stable => stable),
     *     switchMap(() => interval(1000))
     *   ).subscribe(counter => {
     *     this.value = counter;
     *     cd.detectChanges();
     *   });
     * }
     * ```
     *
     * Or make the subscription callback run inside the zone.
     *
     * ```
     * constructor(appRef: ApplicationRef, zone: NgZone) {
     *   appRef.isStable.pipe(
     *     first(stable => stable),
     *     switchMap(() => interval(1000))
     *   ).subscribe(counter => zone.run(() => this.value = counter));
     * }
     * ```
     *
     * \@publicApi
     */
    class ApplicationRef {
        /**
         * \@internal
         * @param {?} _zone
         * @param {?} _console
         * @param {?} _injector
         * @param {?} _exceptionHandler
         * @param {?} _componentFactoryResolver
         * @param {?} _initStatus
         */
        constructor(_zone, _console, _injector, _exceptionHandler, _componentFactoryResolver, _initStatus) {
            this._zone = _zone;
            this._console = _console;
            this._injector = _injector;
            this._exceptionHandler = _exceptionHandler;
            this._componentFactoryResolver = _componentFactoryResolver;
            this._initStatus = _initStatus;
            /**
             * \@internal
             */
            this._bootstrapListeners = [];
            this._views = [];
            this._runningTick = false;
            this._enforceNoNewChanges = false;
            this._stable = true;
            /**
             * Get a list of component types registered to this application.
             * This list is populated even before the component is created.
             */
            this.componentTypes = [];
            /**
             * Get a list of components registered to this application.
             */
            this.components = [];
            this._enforceNoNewChanges = isDevMode();
            this._zone.onMicrotaskEmpty.subscribe({
                next: (/**
                 * @return {?}
                 */
                () => {
                    this._zone.run((/**
                     * @return {?}
                     */
                    () => {
                        this.tick();
                    }));
                })
            });
            /** @type {?} */
            const isCurrentlyStable = new Observable((/**
             * @param {?} observer
             * @return {?}
             */
            (observer) => {
                this._stable = this._zone.isStable && !this._zone.hasPendingMacrotasks &&
                    !this._zone.hasPendingMicrotasks;
                this._zone.runOutsideAngular((/**
                 * @return {?}
                 */
                () => {
                    observer.next(this._stable);
                    observer.complete();
                }));
            }));
            /** @type {?} */
            const isStable = new Observable((/**
             * @param {?} observer
             * @return {?}
             */
            (observer) => {
                // Create the subscription to onStable outside the Angular Zone so that
                // the callback is run outside the Angular Zone.
                /** @type {?} */
                let stableSub;
                this._zone.runOutsideAngular((/**
                 * @return {?}
                 */
                () => {
                    stableSub = this._zone.onStable.subscribe((/**
                     * @return {?}
                     */
                    () => {
                        NgZone.assertNotInAngularZone();
                        // Check whether there are no pending macro/micro tasks in the next tick
                        // to allow for NgZone to update the state.
                        scheduleMicroTask((/**
                         * @return {?}
                         */
                        () => {
                            if (!this._stable && !this._zone.hasPendingMacrotasks &&
                                !this._zone.hasPendingMicrotasks) {
                                this._stable = true;
                                observer.next(true);
                            }
                        }));
                    }));
                }));
                /** @type {?} */
                const unstableSub = this._zone.onUnstable.subscribe((/**
                 * @return {?}
                 */
                () => {
                    NgZone.assertInAngularZone();
                    if (this._stable) {
                        this._stable = false;
                        this._zone.runOutsideAngular((/**
                         * @return {?}
                         */
                        () => {
                            observer.next(false);
                        }));
                    }
                }));
                return (/**
                 * @return {?}
                 */
                () => {
                    stableSub.unsubscribe();
                    unstableSub.unsubscribe();
                });
            }));
            ((/** @type {?} */ (this))).isStable =
                merge(isCurrentlyStable, isStable.pipe(share()));
        }
        /**
         * Bootstrap a new component at the root level of the application.
         *
         * \@usageNotes
         * ### Bootstrap process
         *
         * When bootstrapping a new root component into an application, Angular mounts the
         * specified application component onto DOM elements identified by the componentType's
         * selector and kicks off automatic change detection to finish initializing the component.
         *
         * Optionally, a component can be mounted onto a DOM element that does not match the
         * componentType's selector.
         *
         * ### Example
         * {\@example core/ts/platform/platform.ts region='longform'}
         * @template C
         * @param {?} componentOrFactory
         * @param {?=} rootSelectorOrNode
         * @return {?}
         */
        bootstrap(componentOrFactory, rootSelectorOrNode) {
            if (!this._initStatus.done) {
                throw new Error('Cannot bootstrap as there are still asynchronous initializers running. Bootstrap components in the `ngDoBootstrap` method of the root module.');
            }
            /** @type {?} */
            let componentFactory;
            if (componentOrFactory instanceof ComponentFactory) {
                componentFactory = componentOrFactory;
            }
            else {
                componentFactory =
                    (/** @type {?} */ (this._componentFactoryResolver.resolveComponentFactory(componentOrFactory)));
            }
            this.componentTypes.push(componentFactory.componentType);
            // Create a factory associated with the current module if it's not bound to some other
            /** @type {?} */
            const ngModule = isBoundToModule(componentFactory) ? undefined : this._injector.get(NgModuleRef);
            /** @type {?} */
            const selectorOrNode = rootSelectorOrNode || componentFactory.selector;
            /** @type {?} */
            const compRef = componentFactory.create(Injector.NULL, [], selectorOrNode, ngModule);
            compRef.onDestroy((/**
             * @return {?}
             */
            () => {
                this._unloadComponent(compRef);
            }));
            /** @type {?} */
            const testability = compRef.injector.get(Testability, null);
            if (testability) {
                compRef.injector.get(TestabilityRegistry)
                    .registerApplication(compRef.location.nativeElement, testability);
            }
            this._loadComponent(compRef);
            if (isDevMode()) {
                this._console.log(`Angular is running in the development mode. Call enableProdMode() to enable the production mode.`);
            }
            return compRef;
        }
        /**
         * Invoke this method to explicitly process change detection and its side-effects.
         *
         * In development mode, `tick()` also performs a second change detection cycle to ensure that no
         * further changes are detected. If additional changes are picked up during this second cycle,
         * bindings in the app have side-effects that cannot be resolved in a single change detection
         * pass.
         * In this case, Angular throws an error, since an Angular application can only have one change
         * detection pass during which all change detection must complete.
         * @return {?}
         */
        tick() {
            if (this._runningTick) {
                throw new Error('ApplicationRef.tick is called recursively');
            }
            try {
                this._runningTick = true;
                for (let view of this._views) {
                    view.detectChanges();
                }
                if (this._enforceNoNewChanges) {
                    for (let view of this._views) {
                        view.checkNoChanges();
                    }
                }
            }
            catch (e) {
                // Attention: Don't rethrow as it could cancel subscriptions to Observables!
                this._zone.runOutsideAngular((/**
                 * @return {?}
                 */
                () => this._exceptionHandler.handleError(e)));
            }
            finally {
                this._runningTick = false;
            }
        }
        /**
         * Attaches a view so that it will be dirty checked.
         * The view will be automatically detached when it is destroyed.
         * This will throw if the view is already attached to a ViewContainer.
         * @param {?} viewRef
         * @return {?}
         */
        attachView(viewRef) {
            /** @type {?} */
            const view = ((/** @type {?} */ (viewRef)));
            this._views.push(view);
            view.attachToAppRef(this);
        }
        /**
         * Detaches a view from dirty checking again.
         * @param {?} viewRef
         * @return {?}
         */
        detachView(viewRef) {
            /** @type {?} */
            const view = ((/** @type {?} */ (viewRef)));
            remove(this._views, view);
            view.detachFromAppRef();
        }
        /**
         * @private
         * @param {?} componentRef
         * @return {?}
         */
        _loadComponent(componentRef) {
            this.attachView(componentRef.hostView);
            this.tick();
            this.components.push(componentRef);
            // Get the listeners lazily to prevent DI cycles.
            /** @type {?} */
            const listeners = this._injector.get(APP_BOOTSTRAP_LISTENER, []).concat(this._bootstrapListeners);
            listeners.forEach((/**
             * @param {?} listener
             * @return {?}
             */
            (listener) => listener(componentRef)));
        }
        /**
         * @private
         * @param {?} componentRef
         * @return {?}
         */
        _unloadComponent(componentRef) {
            this.detachView(componentRef.hostView);
            remove(this.components, componentRef);
        }
        /**
         * \@internal
         * @return {?}
         */
        ngOnDestroy() {
            // TODO(alxhub): Dispose of the NgZone.
            this._views.slice().forEach((/**
             * @param {?} view
             * @return {?}
             */
            (view) => view.destroy()));
        }
        /**
         * Returns the number of attached views.
         * @return {?}
         */
        get viewCount() {
            return this._views.length;
        }
    }
    ApplicationRef.ɵfac = function ApplicationRef_Factory(t) { return new (t || ApplicationRef)(ɵɵinject(NgZone), ɵɵinject(Console), ɵɵinject(Injector), ɵɵinject(ErrorHandler), ɵɵinject(ComponentFactoryResolver), ɵɵinject(ApplicationInitStatus)); };
    ApplicationRef.ɵprov = ɵɵdefineInjectable({ token: ApplicationRef, factory: ApplicationRef.ɵfac });
    /** @nocollapse */
    ApplicationRef.ctorParameters = () => [
        { type: NgZone },
        { type: Console },
        { type: Injector },
        { type: ErrorHandler },
        { type: ComponentFactoryResolver },
        { type: ApplicationInitStatus }
    ];
    /**
     * @template T
     * @param {?} list
     * @param {?} el
     * @return {?}
     */
    function remove(list, el) {
        /** @type {?} */
        const index = list.indexOf(el);
        if (index > -1) {
            list.splice(index, 1);
        }
    }
    /**
     * @template T
     * @param {?} args
     * @return {?}
     */
    function _lastDefined(args) {
        for (let i = args.length - 1; i >= 0; i--) {
            if (args[i] !== undefined) {
                return args[i];
            }
        }
        return undefined;
    }
    /**
     * @param {?} parts
     * @return {?}
     */
    function _mergeArrays(parts) {
        /** @type {?} */
        const result = [];
        parts.forEach((/**
         * @param {?} part
         * @return {?}
         */
        (part) => part && result.push(...part)));
        return result;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/linker/system_js_ng_module_factory_loader.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const _SEPARATOR = '#';
    /** @type {?} */
    const FACTORY_CLASS_SUFFIX = 'NgFactory';
    /**
     * Configuration for SystemJsNgModuleLoader.
     * token.
     *
     * \@publicApi
     * @deprecated the `string` form of `loadChildren` is deprecated, and `SystemJsNgModuleLoaderConfig`
     * is part of its implementation. See `LoadChildren` for more details.
     * @abstract
     */
    class SystemJsNgModuleLoaderConfig {
    }
    /** @type {?} */
    const DEFAULT_CONFIG = {
        factoryPathPrefix: '',
        factoryPathSuffix: '.ngfactory',
    };
    /**
     * NgModuleFactoryLoader that uses SystemJS to load NgModuleFactory
     * \@publicApi
     * @deprecated the `string` form of `loadChildren` is deprecated, and `SystemJsNgModuleLoader` is
     * part of its implementation. See `LoadChildren` for more details.
     */
    class SystemJsNgModuleLoader {
        /**
         * @param {?} _compiler
         * @param {?=} config
         */
        constructor(_compiler, config) {
            this._compiler = _compiler;
            this._config = config || DEFAULT_CONFIG;
        }
        /**
         * @param {?} path
         * @return {?}
         */
        load(path) {
            return  this.loadAndCompile(path);
        }
        /**
         * @private
         * @param {?} path
         * @return {?}
         */
        loadAndCompile(path) {
            let [module, exportName] = path.split(_SEPARATOR);
            if (exportName === undefined) {
                exportName = 'default';
            }
            return System.import(module)
                .then((/**
             * @param {?} module
             * @return {?}
             */
            (module) => module[exportName]))
                .then((/**
             * @param {?} type
             * @return {?}
             */
            (type) => checkNotEmpty(type, module, exportName)))
                .then((/**
             * @param {?} type
             * @return {?}
             */
            (type) => this._compiler.compileModuleAsync(type)));
        }
        /**
         * @private
         * @param {?} path
         * @return {?}
         */
        loadFactory(path) {
            let [module, exportName] = path.split(_SEPARATOR);
            /** @type {?} */
            let factoryClassSuffix = FACTORY_CLASS_SUFFIX;
            if (exportName === undefined) {
                exportName = 'default';
                factoryClassSuffix = '';
            }
            return System.import(this._config.factoryPathPrefix + module + this._config.factoryPathSuffix)
                .then((/**
             * @param {?} module
             * @return {?}
             */
            (module) => module[exportName + factoryClassSuffix]))
                .then((/**
             * @param {?} factory
             * @return {?}
             */
            (factory) => checkNotEmpty(factory, module, exportName)));
        }
    }
    SystemJsNgModuleLoader.ɵfac = function SystemJsNgModuleLoader_Factory(t) { return new (t || SystemJsNgModuleLoader)(ɵɵinject(Compiler), ɵɵinject(SystemJsNgModuleLoaderConfig, 8)); };
    SystemJsNgModuleLoader.ɵprov = ɵɵdefineInjectable({ token: SystemJsNgModuleLoader, factory: SystemJsNgModuleLoader.ɵfac });
    /** @nocollapse */
    SystemJsNgModuleLoader.ctorParameters = () => [
        { type: Compiler },
        { type: SystemJsNgModuleLoaderConfig, decorators: [{ type: Optional }] }
    ];
    /**
     * @param {?} value
     * @param {?} modulePath
     * @param {?} exportName
     * @return {?}
     */
    function checkNotEmpty(value, modulePath, exportName) {
        if (!value) {
            throw new Error(`Cannot find '${exportName}' in '${modulePath}'`);
        }
        return value;
    }
    /**
     * @param {?} _nativeNode
     * @return {?}
     */
    function getDebugNodeR2__POST_R3__(_nativeNode) {
        return null;
    }
    /** @type {?} */
    const getDebugNodeR2 = getDebugNodeR2__POST_R3__;

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/change_detection.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/platform_core_providers.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const _CORE_PLATFORM_PROVIDERS = [
        // Set a default platform name for platforms that don't set it explicitly.
        { provide: PLATFORM_ID, useValue: 'unknown' },
        { provide: PlatformRef, deps: [Injector] },
        { provide: TestabilityRegistry, deps: [] },
        { provide: Console, deps: [] },
    ];
    /**
     * This platform has to be included in any other platform
     *
     * \@publicApi
     * @type {?}
     */
    const platformCore = createPlatformFactory(null, 'core', _CORE_PLATFORM_PROVIDERS);

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/application_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @return {?}
     */
    function _iterableDiffersFactory() {
        return defaultIterableDiffers;
    }
    /**
     * @return {?}
     */
    function _keyValueDiffersFactory() {
        return defaultKeyValueDiffers;
    }
    /**
     * @param {?=} locale
     * @return {?}
     */
    function _localeFactory(locale) {
        locale = locale || getGlobalLocale();
        {
            setLocaleId(locale);
        }
        return locale;
    }
    /**
     * Work out the locale from the potential global properties.
     *
     * * Closure Compiler: use `goog.LOCALE`.
     * * Ivy enabled: use `$localize.locale`
     * @return {?}
     */
    function getGlobalLocale() {
        if (typeof ngI18nClosureMode !== 'undefined' && ngI18nClosureMode &&
            typeof goog !== 'undefined' && goog.LOCALE !== 'en') {
            // * The default `goog.LOCALE` value is `en`, while Angular used `en-US`.
            // * In order to preserve backwards compatibility, we use Angular default value over
            //   Closure Compiler's one.
            return goog.LOCALE;
        }
        else {
            // KEEP `typeof $localize !== 'undefined' && $localize.locale` IN SYNC WITH THE LOCALIZE
            // COMPILE-TIME INLINER.
            //
            // * During compile time inlining of translations the expression will be replaced
            //   with a string literal that is the current locale. Other forms of this expression are not
            //   guaranteed to be replaced.
            //
            // * During runtime translation evaluation, the developer is required to set `$localize.locale`
            //   if required, or just to provide their own `LOCALE_ID` provider.
            return ( typeof $localize !== 'undefined' && $localize.locale) ||
                DEFAULT_LOCALE_ID;
        }
    }
    const ɵ0$g = USD_CURRENCY_CODE;
    /**
     * A built-in [dependency injection token](guide/glossary#di-token)
     * that is used to configure the root injector for bootstrapping.
     * @type {?}
     */
    const APPLICATION_MODULE_PROVIDERS = [
        {
            provide: ApplicationRef,
            useClass: ApplicationRef,
            deps: [NgZone, Console, Injector, ErrorHandler, ComponentFactoryResolver, ApplicationInitStatus]
        },
        { provide: SCHEDULER, deps: [NgZone], useFactory: zoneSchedulerFactory },
        {
            provide: ApplicationInitStatus,
            useClass: ApplicationInitStatus,
            deps: [[new Optional(), APP_INITIALIZER]]
        },
        { provide: Compiler, useClass: Compiler, deps: [] },
        APP_ID_RANDOM_PROVIDER,
        { provide: IterableDiffers, useFactory: _iterableDiffersFactory, deps: [] },
        { provide: KeyValueDiffers, useFactory: _keyValueDiffersFactory, deps: [] },
        {
            provide: LOCALE_ID$1,
            useFactory: _localeFactory,
            deps: [[new Inject(LOCALE_ID$1), new Optional(), new SkipSelf()]]
        },
        { provide: DEFAULT_CURRENCY_CODE, useValue: ɵ0$g },
    ];
    /**
     * Schedule work at next available slot.
     *
     * In Ivy this is just `requestAnimationFrame`. For compatibility reasons when bootstrapped
     * using `platformRef.bootstrap` we need to use `NgZone.onStable` as the scheduling mechanism.
     * This overrides the scheduling mechanism in Ivy to `NgZone.onStable`.
     *
     * @param {?} ngZone NgZone to use for scheduling.
     * @return {?}
     */
    function zoneSchedulerFactory(ngZone) {
        /** @type {?} */
        let queue = [];
        ngZone.onStable.subscribe((/**
         * @return {?}
         */
        () => {
            while (queue.length) {
                (/** @type {?} */ (queue.pop()))();
            }
        }));
        return (/**
         * @param {?} fn
         * @return {?}
         */
        function (fn) {
            queue.push(fn);
        });
    }
    /**
     * Configures the root injector for an app with
     * providers of `\@angular/core` dependencies that `ApplicationRef` needs
     * to bootstrap components.
     *
     * Re-exported by `BrowserModule`, which is included automatically in the root
     * `AppModule` when you create a new app with the CLI `new` command.
     *
     * \@publicApi
     */
    class ApplicationModule {
        // Inject ApplicationRef to make it eager...
        /**
         * @param {?} appRef
         */
        constructor(appRef) { }
    }
    ApplicationModule.ɵmod = ɵɵdefineNgModule({ type: ApplicationModule });
    ApplicationModule.ɵinj = ɵɵdefineInjector({ factory: function ApplicationModule_Factory(t) { return new (t || ApplicationModule)(ɵɵinject(ApplicationRef)); }, providers: APPLICATION_MODULE_PROVIDERS });
    /** @nocollapse */
    ApplicationModule.ctorParameters = () => [
        { type: ApplicationRef }
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/view/index.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/core_private_export.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/core_render3_private_export.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/codegen_private_exports.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/core/src/core.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    if (ngDevMode) {
        // This helper is to give a reasonable error message to people upgrading to v9 that have not yet
        // installed `@angular/localize` in their app.
        // tslint:disable-next-line: no-toplevel-property-access
        _global.$localize = _global.$localize || (/**
         * @return {?}
         */
        function () {
            throw new Error('It looks like your application or one of its dependencies is using i18n.\n' +
                'Angular 9 introduced a global `$localize()` function that needs to be loaded.\n' +
                'Please run `ng add @angular/localize` from the Angular CLI.\n' +
                '(For non-CLI projects, add `import \'@angular/localize/init\';` to your `polyfills.ts` file.\n' +
                'For server-side rendering applications add the import to your `main.server.ts` file.)');
        });
    }

    /**
     * @license Angular v9.1.12
     * (c) 2010-2020 Google LLC. https://angular.io/
     * License: MIT
     */
    let _DOM = (/** @type {?} */ (null));
    /**
     * @return {?}
     */
    function getDOM() {
        return _DOM;
    }
    /**
     * @param {?} adapter
     * @return {?}
     */
    function setRootDomAdapter(adapter) {
        if (!_DOM) {
            _DOM = adapter;
        }
    }
    /* tslint:disable:requireParameterType */
    /**
     * Provides DOM operations in an environment-agnostic way.
     *
     * \@security Tread carefully! Interacting with the DOM directly is dangerous and
     * can introduce XSS risks.
     * @abstract
     */
    class DomAdapter {
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/src/dom_tokens.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * A DI Token representing the main rendering context. In a browser this is the DOM Document.
     *
     * Note: Document might not be available in the Application Context when Application and Rendering
     * Contexts are not the same (e.g. when running the application in a Web Worker).
     *
     * \@publicApi
     * @type {?}
     */
    const DOCUMENT$1 = new InjectionToken('DocumentToken');

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/src/location/platform_location.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * This class should not be used directly by an application developer. Instead, use
     * {\@link Location}.
     *
     * `PlatformLocation` encapsulates all calls to DOM APIs, which allows the Router to be
     * platform-agnostic.
     * This means that we can have different implementation of `PlatformLocation` for the different
     * platforms that Angular supports. For example, `\@angular/platform-browser` provides an
     * implementation specific to the browser environment, while `\@angular/platform-server` provides
     * one suitable for use with server-side rendering.
     *
     * The `PlatformLocation` class is used directly by all implementations of {\@link LocationStrategy}
     * when they need to interact with the DOM APIs like pushState, popState, etc.
     *
     * {\@link LocationStrategy} in turn is used by the {\@link Location} service which is used directly
     * by the {\@link Router} in order to navigate between routes. Since all interactions between {\@link
     * Router} /
     * {\@link Location} / {\@link LocationStrategy} and DOM APIs flow through the `PlatformLocation`
     * class, they are all platform-agnostic.
     *
     * \@publicApi
     * @abstract
     */
    class PlatformLocation {
    }
    PlatformLocation.ɵfac = function PlatformLocation_Factory(t) { return new (t || PlatformLocation)(); };
    /** @nocollapse */ PlatformLocation.ɵprov = ɵɵdefineInjectable({ factory: useBrowserPlatformLocation, token: PlatformLocation, providedIn: "platform" });
    /**
     * @return {?}
     */
    function useBrowserPlatformLocation() {
        return ɵɵinject(BrowserPlatformLocation);
    }
    /**
     * \@description
     * Indicates when a location is initialized.
     *
     * \@publicApi
     * @type {?}
     */
    const LOCATION_INITIALIZED = new InjectionToken('Location Initialized');
    /**
     * `PlatformLocation` encapsulates all of the direct calls to platform APIs.
     * This class should not be used directly by an application developer. Instead, use
     * {\@link Location}.
     */
    class BrowserPlatformLocation extends PlatformLocation {
        /**
         * @param {?} _doc
         */
        constructor(_doc) {
            super();
            this._doc = _doc;
            this._init();
        }
        // This is moved to its own method so that `MockPlatformLocationStrategy` can overwrite it
        /**
         * \@internal
         * @return {?}
         */
        _init() {
            ((/** @type {?} */ (this))).location = getDOM().getLocation();
            this._history = getDOM().getHistory();
        }
        /**
         * @return {?}
         */
        getBaseHrefFromDOM() {
            return (/** @type {?} */ (getDOM().getBaseHref(this._doc)));
        }
        /**
         * @param {?} fn
         * @return {?}
         */
        onPopState(fn) {
            getDOM().getGlobalEventTarget(this._doc, 'window').addEventListener('popstate', fn, false);
        }
        /**
         * @param {?} fn
         * @return {?}
         */
        onHashChange(fn) {
            getDOM().getGlobalEventTarget(this._doc, 'window').addEventListener('hashchange', fn, false);
        }
        /**
         * @return {?}
         */
        get href() {
            return this.location.href;
        }
        /**
         * @return {?}
         */
        get protocol() {
            return this.location.protocol;
        }
        /**
         * @return {?}
         */
        get hostname() {
            return this.location.hostname;
        }
        /**
         * @return {?}
         */
        get port() {
            return this.location.port;
        }
        /**
         * @return {?}
         */
        get pathname() {
            return this.location.pathname;
        }
        /**
         * @return {?}
         */
        get search() {
            return this.location.search;
        }
        /**
         * @return {?}
         */
        get hash() {
            return this.location.hash;
        }
        /**
         * @param {?} newPath
         * @return {?}
         */
        set pathname(newPath) {
            this.location.pathname = newPath;
        }
        /**
         * @param {?} state
         * @param {?} title
         * @param {?} url
         * @return {?}
         */
        pushState(state, title, url) {
            if (supportsState()) {
                this._history.pushState(state, title, url);
            }
            else {
                this.location.hash = url;
            }
        }
        /**
         * @param {?} state
         * @param {?} title
         * @param {?} url
         * @return {?}
         */
        replaceState(state, title, url) {
            if (supportsState()) {
                this._history.replaceState(state, title, url);
            }
            else {
                this.location.hash = url;
            }
        }
        /**
         * @return {?}
         */
        forward() {
            this._history.forward();
        }
        /**
         * @return {?}
         */
        back() {
            this._history.back();
        }
        /**
         * @return {?}
         */
        getState() {
            return this._history.state;
        }
    }
    BrowserPlatformLocation.ɵfac = function BrowserPlatformLocation_Factory(t) { return new (t || BrowserPlatformLocation)(ɵɵinject(DOCUMENT$1)); };
    /** @nocollapse */
    BrowserPlatformLocation.ctorParameters = () => [
        { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT$1,] }] }
    ];
    /** @nocollapse */ BrowserPlatformLocation.ɵprov = ɵɵdefineInjectable({ factory: createBrowserPlatformLocation, token: BrowserPlatformLocation, providedIn: "platform" });
    /**
     * @return {?}
     */
    function supportsState() {
        return !!window.history.pushState;
    }
    /**
     * @return {?}
     */
    function createBrowserPlatformLocation() {
        return new BrowserPlatformLocation(ɵɵinject(DOCUMENT$1));
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/src/private_export.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/src/location/util.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * Joins two parts of a URL with a slash if needed.
     *
     * @param {?} start  URL string
     * @param {?} end    URL string
     *
     *
     * @return {?} The joined URL string.
     */
    function joinWithSlash(start, end) {
        if (start.length == 0) {
            return end;
        }
        if (end.length == 0) {
            return start;
        }
        /** @type {?} */
        let slashes = 0;
        if (start.endsWith('/')) {
            slashes++;
        }
        if (end.startsWith('/')) {
            slashes++;
        }
        if (slashes == 2) {
            return start + end.substring(1);
        }
        if (slashes == 1) {
            return start + end;
        }
        return start + '/' + end;
    }
    /**
     * Removes a trailing slash from a URL string if needed.
     * Looks for the first occurrence of either `#`, `?`, or the end of the
     * line as `/` characters and removes the trailing slash if one exists.
     *
     * @param {?} url URL string.
     *
     * @return {?} The URL string, modified if needed.
     */
    function stripTrailingSlash(url) {
        /** @type {?} */
        const match = url.match(/#|\?|$/);
        /** @type {?} */
        const pathEndIdx = match && match.index || url.length;
        /** @type {?} */
        const droppedSlashIdx = pathEndIdx - (url[pathEndIdx - 1] === '/' ? 1 : 0);
        return url.slice(0, droppedSlashIdx) + url.slice(pathEndIdx);
    }
    /**
     * Normalizes URL parameters by prepending with `?` if needed.
     *
     * @param {?} params String of URL parameters.
     *
     * @return {?} The normalized URL parameters string.
     */
    function normalizeQueryParams(params) {
        return params && params[0] !== '?' ? '?' + params : params;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/src/location/location_strategy.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Enables the `Location` service to read route state from the browser's URL.
     * Angular provides two strategies:
     * `HashLocationStrategy` and `PathLocationStrategy`.
     *
     * Applications should use the `Router` or `Location` services to
     * interact with application route state.
     *
     * For instance, `HashLocationStrategy` produces URLs like
     * <code class="no-auto-link">http://example.com#/foo</code>,
     * and `PathLocationStrategy` produces
     * <code class="no-auto-link">http://example.com/foo</code> as an equivalent URL.
     *
     * See these two classes for more.
     *
     * \@publicApi
     * @abstract
     */
    class LocationStrategy {
    }
    LocationStrategy.ɵfac = function LocationStrategy_Factory(t) { return new (t || LocationStrategy)(); };
    /** @nocollapse */ LocationStrategy.ɵprov = ɵɵdefineInjectable({ factory: provideLocationStrategy, token: LocationStrategy, providedIn: "root" });
    /**
     * @param {?} platformLocation
     * @return {?}
     */
    function provideLocationStrategy(platformLocation) {
        // See #23917
        /** @type {?} */
        const location = ɵɵinject(DOCUMENT$1).location;
        return new PathLocationStrategy(ɵɵinject((/** @type {?} */ (PlatformLocation))), location && location.origin || '');
    }
    /**
     * A predefined [DI token](guide/glossary#di-token) for the base href
     * to be used with the `PathLocationStrategy`.
     * The base href is the URL prefix that should be preserved when generating
     * and recognizing URLs.
     *
     * \@usageNotes
     *
     * The following example shows how to use this token to configure the root app injector
     * with a base href value, so that the DI framework can supply the dependency anywhere in the app.
     *
     * ```typescript
     * import {Component, NgModule} from '\@angular/core';
     * import {APP_BASE_HREF} from '\@angular/common';
     *
     * \@NgModule({
     *   providers: [{provide: APP_BASE_HREF, useValue: '/my/app'}]
     * })
     * class AppModule {}
     * ```
     *
     * \@publicApi
     * @type {?}
     */
    const APP_BASE_HREF = new InjectionToken('appBaseHref');
    /**
     * \@description
     * A {\@link LocationStrategy} used to configure the {\@link Location} service to
     * represent its state in the
     * [path](https://en.wikipedia.org/wiki/Uniform_Resource_Locator#Syntax) of the
     * browser's URL.
     *
     * If you're using `PathLocationStrategy`, you must provide a {\@link APP_BASE_HREF}
     * or add a base element to the document. This URL prefix that will be preserved
     * when generating and recognizing URLs.
     *
     * For instance, if you provide an `APP_BASE_HREF` of `'/my/app'` and call
     * `location.go('/foo')`, the browser's URL will become
     * `example.com/my/app/foo`.
     *
     * Similarly, if you add `<base href='/my/app'/>` to the document and call
     * `location.go('/foo')`, the browser's URL will become
     * `example.com/my/app/foo`.
     *
     * \@usageNotes
     *
     * ### Example
     *
     * {\@example common/location/ts/path_location_component.ts region='LocationComponent'}
     *
     * \@publicApi
     */
    class PathLocationStrategy extends LocationStrategy {
        /**
         * @param {?} _platformLocation
         * @param {?=} href
         */
        constructor(_platformLocation, href) {
            super();
            this._platformLocation = _platformLocation;
            if (href == null) {
                href = this._platformLocation.getBaseHrefFromDOM();
            }
            if (href == null) {
                throw new Error(`No base href set. Please provide a value for the APP_BASE_HREF token or add a base element to the document.`);
            }
            this._baseHref = href;
        }
        /**
         * @param {?} fn
         * @return {?}
         */
        onPopState(fn) {
            this._platformLocation.onPopState(fn);
            this._platformLocation.onHashChange(fn);
        }
        /**
         * @return {?}
         */
        getBaseHref() {
            return this._baseHref;
        }
        /**
         * @param {?} internal
         * @return {?}
         */
        prepareExternalUrl(internal) {
            return joinWithSlash(this._baseHref, internal);
        }
        /**
         * @param {?=} includeHash
         * @return {?}
         */
        path(includeHash = false) {
            /** @type {?} */
            const pathname = this._platformLocation.pathname + normalizeQueryParams(this._platformLocation.search);
            /** @type {?} */
            const hash = this._platformLocation.hash;
            return hash && includeHash ? `${pathname}${hash}` : pathname;
        }
        /**
         * @param {?} state
         * @param {?} title
         * @param {?} url
         * @param {?} queryParams
         * @return {?}
         */
        pushState(state, title, url, queryParams) {
            /** @type {?} */
            const externalUrl = this.prepareExternalUrl(url + normalizeQueryParams(queryParams));
            this._platformLocation.pushState(state, title, externalUrl);
        }
        /**
         * @param {?} state
         * @param {?} title
         * @param {?} url
         * @param {?} queryParams
         * @return {?}
         */
        replaceState(state, title, url, queryParams) {
            /** @type {?} */
            const externalUrl = this.prepareExternalUrl(url + normalizeQueryParams(queryParams));
            this._platformLocation.replaceState(state, title, externalUrl);
        }
        /**
         * @return {?}
         */
        forward() {
            this._platformLocation.forward();
        }
        /**
         * @return {?}
         */
        back() {
            this._platformLocation.back();
        }
    }
    PathLocationStrategy.ɵfac = function PathLocationStrategy_Factory(t) { return new (t || PathLocationStrategy)(ɵɵinject(PlatformLocation), ɵɵinject(APP_BASE_HREF, 8)); };
    PathLocationStrategy.ɵprov = ɵɵdefineInjectable({ token: PathLocationStrategy, factory: PathLocationStrategy.ɵfac });
    /** @nocollapse */
    PathLocationStrategy.ctorParameters = () => [
        { type: PlatformLocation },
        { type: String, decorators: [{ type: Optional }, { type: Inject, args: [APP_BASE_HREF,] }] }
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/src/location/hash_location_strategy.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * \@description
     * A {\@link LocationStrategy} used to configure the {\@link Location} service to
     * represent its state in the
     * [hash fragment](https://en.wikipedia.org/wiki/Uniform_Resource_Locator#Syntax)
     * of the browser's URL.
     *
     * For instance, if you call `location.go('/foo')`, the browser's URL will become
     * `example.com#/foo`.
     *
     * \@usageNotes
     *
     * ### Example
     *
     * {\@example common/location/ts/hash_location_component.ts region='LocationComponent'}
     *
     * \@publicApi
     */
    class HashLocationStrategy extends LocationStrategy {
        /**
         * @param {?} _platformLocation
         * @param {?=} _baseHref
         */
        constructor(_platformLocation, _baseHref) {
            super();
            this._platformLocation = _platformLocation;
            this._baseHref = '';
            if (_baseHref != null) {
                this._baseHref = _baseHref;
            }
        }
        /**
         * @param {?} fn
         * @return {?}
         */
        onPopState(fn) {
            this._platformLocation.onPopState(fn);
            this._platformLocation.onHashChange(fn);
        }
        /**
         * @return {?}
         */
        getBaseHref() {
            return this._baseHref;
        }
        /**
         * @param {?=} includeHash
         * @return {?}
         */
        path(includeHash = false) {
            // the hash value is always prefixed with a `#`
            // and if it is empty then it will stay empty
            /** @type {?} */
            let path = this._platformLocation.hash;
            if (path == null)
                path = '#';
            return path.length > 0 ? path.substring(1) : path;
        }
        /**
         * @param {?} internal
         * @return {?}
         */
        prepareExternalUrl(internal) {
            /** @type {?} */
            const url = joinWithSlash(this._baseHref, internal);
            return url.length > 0 ? ('#' + url) : url;
        }
        /**
         * @param {?} state
         * @param {?} title
         * @param {?} path
         * @param {?} queryParams
         * @return {?}
         */
        pushState(state, title, path, queryParams) {
            /** @type {?} */
            let url = this.prepareExternalUrl(path + normalizeQueryParams(queryParams));
            if (url.length == 0) {
                url = this._platformLocation.pathname;
            }
            this._platformLocation.pushState(state, title, url);
        }
        /**
         * @param {?} state
         * @param {?} title
         * @param {?} path
         * @param {?} queryParams
         * @return {?}
         */
        replaceState(state, title, path, queryParams) {
            /** @type {?} */
            let url = this.prepareExternalUrl(path + normalizeQueryParams(queryParams));
            if (url.length == 0) {
                url = this._platformLocation.pathname;
            }
            this._platformLocation.replaceState(state, title, url);
        }
        /**
         * @return {?}
         */
        forward() {
            this._platformLocation.forward();
        }
        /**
         * @return {?}
         */
        back() {
            this._platformLocation.back();
        }
    }
    HashLocationStrategy.ɵfac = function HashLocationStrategy_Factory(t) { return new (t || HashLocationStrategy)(ɵɵinject(PlatformLocation), ɵɵinject(APP_BASE_HREF, 8)); };
    HashLocationStrategy.ɵprov = ɵɵdefineInjectable({ token: HashLocationStrategy, factory: HashLocationStrategy.ɵfac });
    /** @nocollapse */
    HashLocationStrategy.ctorParameters = () => [
        { type: PlatformLocation },
        { type: String, decorators: [{ type: Optional }, { type: Inject, args: [APP_BASE_HREF,] }] }
    ];
    /**
     * \@description
     *
     * A service that applications can use to interact with a browser's URL.
     *
     * Depending on the `LocationStrategy` used, `Location` persists
     * to the URL's path or the URL's hash segment.
     *
     * \@usageNotes
     *
     * It's better to use the `Router#navigate` service to trigger route changes. Use
     * `Location` only if you need to interact with or create normalized URLs outside of
     * routing.
     *
     * `Location` is responsible for normalizing the URL against the application's base href.
     * A normalized URL is absolute from the URL host, includes the application's base href, and has no
     * trailing slash:
     * - `/my/app/user/123` is normalized
     * - `my/app/user/123` **is not** normalized
     * - `/my/app/user/123/` **is not** normalized
     *
     * ### Example
     *
     * <code-example path='common/location/ts/path_location_component.ts'
     * region='LocationComponent'></code-example>
     *
     * \@publicApi
     */
    class Location {
        /**
         * @param {?} platformStrategy
         * @param {?} platformLocation
         */
        constructor(platformStrategy, platformLocation) {
            /**
             * \@internal
             */
            this._subject = new EventEmitter();
            /**
             * \@internal
             */
            this._urlChangeListeners = [];
            this._platformStrategy = platformStrategy;
            /** @type {?} */
            const browserBaseHref = this._platformStrategy.getBaseHref();
            this._platformLocation = platformLocation;
            this._baseHref = stripTrailingSlash(_stripIndexHtml(browserBaseHref));
            this._platformStrategy.onPopState((/**
             * @param {?} ev
             * @return {?}
             */
            (ev) => {
                this._subject.emit({
                    'url': this.path(true),
                    'pop': true,
                    'state': ev.state,
                    'type': ev.type,
                });
            }));
        }
        /**
         * Normalizes the URL path for this location.
         *
         * @param {?=} includeHash True to include an anchor fragment in the path.
         *
         * @return {?} The normalized URL path.
         */
        // TODO: vsavkin. Remove the boolean flag and always include hash once the deprecated router is
        // removed.
        path(includeHash = false) {
            return this.normalize(this._platformStrategy.path(includeHash));
        }
        /**
         * Reports the current state of the location history.
         * @return {?} The current value of the `history.state` object.
         */
        getState() {
            return this._platformLocation.getState();
        }
        /**
         * Normalizes the given path and compares to the current normalized path.
         *
         * @param {?} path The given URL path.
         * @param {?=} query Query parameters.
         *
         * @return {?} True if the given URL path is equal to the current normalized path, false
         * otherwise.
         */
        isCurrentPathEqualTo(path, query = '') {
            return this.path() == this.normalize(path + normalizeQueryParams(query));
        }
        /**
         * Normalizes a URL path by stripping any trailing slashes.
         *
         * @param {?} url String representing a URL.
         *
         * @return {?} The normalized URL string.
         */
        normalize(url) {
            return Location.stripTrailingSlash(_stripBaseHref(this._baseHref, _stripIndexHtml(url)));
        }
        /**
         * Normalizes an external URL path.
         * If the given URL doesn't begin with a leading slash (`'/'`), adds one
         * before normalizing. Adds a hash if `HashLocationStrategy` is
         * in use, or the `APP_BASE_HREF` if the `PathLocationStrategy` is in use.
         *
         * @param {?} url String representing a URL.
         *
         * @return {?} A normalized platform-specific URL.
         */
        prepareExternalUrl(url) {
            if (url && url[0] !== '/') {
                url = '/' + url;
            }
            return this._platformStrategy.prepareExternalUrl(url);
        }
        // TODO: rename this method to pushState
        /**
         * Changes the browser's URL to a normalized version of a given URL, and pushes a
         * new item onto the platform's history.
         *
         * @param {?} path  URL path to normalize.
         * @param {?=} query Query parameters.
         * @param {?=} state Location history state.
         *
         * @return {?}
         */
        go(path, query = '', state = null) {
            this._platformStrategy.pushState(state, '', path, query);
            this._notifyUrlChangeListeners(this.prepareExternalUrl(path + normalizeQueryParams(query)), state);
        }
        /**
         * Changes the browser's URL to a normalized version of the given URL, and replaces
         * the top item on the platform's history stack.
         *
         * @param {?} path  URL path to normalize.
         * @param {?=} query Query parameters.
         * @param {?=} state Location history state.
         * @return {?}
         */
        replaceState(path, query = '', state = null) {
            this._platformStrategy.replaceState(state, '', path, query);
            this._notifyUrlChangeListeners(this.prepareExternalUrl(path + normalizeQueryParams(query)), state);
        }
        /**
         * Navigates forward in the platform's history.
         * @return {?}
         */
        forward() {
            this._platformStrategy.forward();
        }
        /**
         * Navigates back in the platform's history.
         * @return {?}
         */
        back() {
            this._platformStrategy.back();
        }
        /**
         * Registers a URL change listener. Use to catch updates performed by the Angular
         * framework that are not detectible through "popstate" or "hashchange" events.
         *
         * @param {?} fn The change handler function, which take a URL and a location history state.
         * @return {?}
         */
        onUrlChange(fn) {
            this._urlChangeListeners.push(fn);
            this.subscribe((/**
             * @param {?} v
             * @return {?}
             */
            v => {
                this._notifyUrlChangeListeners(v.url, v.state);
            }));
        }
        /**
         * \@internal
         * @param {?=} url
         * @param {?=} state
         * @return {?}
         */
        _notifyUrlChangeListeners(url = '', state) {
            this._urlChangeListeners.forEach((/**
             * @param {?} fn
             * @return {?}
             */
            fn => fn(url, state)));
        }
        /**
         * Subscribes to the platform's `popState` events.
         *
         * @param {?} onNext
         * @param {?=} onThrow
         * @param {?=} onReturn
         * @return {?} Subscribed events.
         */
        subscribe(onNext, onThrow, onReturn) {
            return this._subject.subscribe({ next: onNext, error: onThrow, complete: onReturn });
        }
    }
    Location.ɵfac = function Location_Factory(t) { return new (t || Location)(ɵɵinject(LocationStrategy), ɵɵinject(PlatformLocation)); };
    /**
     * Normalizes URL parameters by prepending with `?` if needed.
     *
     * @param params String of URL parameters.
     *
     * @return The normalized URL parameters string.
     */
    Location.normalizeQueryParams = normalizeQueryParams;
    /**
     * Joins two parts of a URL with a slash if needed.
     *
     * @param start  URL string
     * @param end    URL string
     *
     *
     * @return The joined URL string.
     */
    Location.joinWithSlash = joinWithSlash;
    /**
     * Removes a trailing slash from a URL string if needed.
     * Looks for the first occurrence of either `#`, `?`, or the end of the
     * line as `/` characters and removes the trailing slash if one exists.
     *
     * @param url URL string.
     *
     * @return The URL string, modified if needed.
     */
    Location.stripTrailingSlash = stripTrailingSlash;
    /** @nocollapse */
    Location.ctorParameters = () => [
        { type: LocationStrategy },
        { type: PlatformLocation }
    ];
    /** @nocollapse */ Location.ɵprov = ɵɵdefineInjectable({ factory: createLocation, token: Location, providedIn: "root" });
    /**
     * @return {?}
     */
    function createLocation() {
        return new Location(ɵɵinject((/** @type {?} */ (LocationStrategy))), ɵɵinject((/** @type {?} */ (PlatformLocation))));
    }
    /**
     * @param {?} baseHref
     * @param {?} url
     * @return {?}
     */
    function _stripBaseHref(baseHref, url) {
        return baseHref && url.startsWith(baseHref) ? url.substring(baseHref.length) : url;
    }
    /**
     * @param {?} url
     * @return {?}
     */
    function _stripIndexHtml(url) {
        return url.replace(/\/index.html$/, '');
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/src/location/index.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/src/i18n/currencies.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    // THIS CODE IS GENERATED - DO NOT MODIFY
    // See angular/tools/gulp-tasks/cldr/extract.js
    /**
     * \@internal
     * @type {?}
     */
    const CURRENCIES_EN = {
        'ADP': [undefined, undefined, 0],
        'AFN': [undefined, undefined, 0],
        'ALL': [undefined, undefined, 0],
        'AMD': [undefined, undefined, 2],
        'AOA': [undefined, 'Kz'],
        'ARS': [undefined, '$'],
        'AUD': ['A$', '$'],
        'BAM': [undefined, 'KM'],
        'BBD': [undefined, '$'],
        'BDT': [undefined, '৳'],
        'BHD': [undefined, undefined, 3],
        'BIF': [undefined, undefined, 0],
        'BMD': [undefined, '$'],
        'BND': [undefined, '$'],
        'BOB': [undefined, 'Bs'],
        'BRL': ['R$'],
        'BSD': [undefined, '$'],
        'BWP': [undefined, 'P'],
        'BYN': [undefined, 'р.', 2],
        'BYR': [undefined, undefined, 0],
        'BZD': [undefined, '$'],
        'CAD': ['CA$', '$', 2],
        'CHF': [undefined, undefined, 2],
        'CLF': [undefined, undefined, 4],
        'CLP': [undefined, '$', 0],
        'CNY': ['CN¥', '¥'],
        'COP': [undefined, '$', 2],
        'CRC': [undefined, '₡', 2],
        'CUC': [undefined, '$'],
        'CUP': [undefined, '$'],
        'CZK': [undefined, 'Kč', 2],
        'DJF': [undefined, undefined, 0],
        'DKK': [undefined, 'kr', 2],
        'DOP': [undefined, '$'],
        'EGP': [undefined, 'E£'],
        'ESP': [undefined, '₧', 0],
        'EUR': ['€'],
        'FJD': [undefined, '$'],
        'FKP': [undefined, '£'],
        'GBP': ['£'],
        'GEL': [undefined, '₾'],
        'GIP': [undefined, '£'],
        'GNF': [undefined, 'FG', 0],
        'GTQ': [undefined, 'Q'],
        'GYD': [undefined, '$', 2],
        'HKD': ['HK$', '$'],
        'HNL': [undefined, 'L'],
        'HRK': [undefined, 'kn'],
        'HUF': [undefined, 'Ft', 2],
        'IDR': [undefined, 'Rp', 2],
        'ILS': ['₪'],
        'INR': ['₹'],
        'IQD': [undefined, undefined, 0],
        'IRR': [undefined, undefined, 0],
        'ISK': [undefined, 'kr', 0],
        'ITL': [undefined, undefined, 0],
        'JMD': [undefined, '$'],
        'JOD': [undefined, undefined, 3],
        'JPY': ['¥', undefined, 0],
        'KHR': [undefined, '៛'],
        'KMF': [undefined, 'CF', 0],
        'KPW': [undefined, '₩', 0],
        'KRW': ['₩', undefined, 0],
        'KWD': [undefined, undefined, 3],
        'KYD': [undefined, '$'],
        'KZT': [undefined, '₸'],
        'LAK': [undefined, '₭', 0],
        'LBP': [undefined, 'L£', 0],
        'LKR': [undefined, 'Rs'],
        'LRD': [undefined, '$'],
        'LTL': [undefined, 'Lt'],
        'LUF': [undefined, undefined, 0],
        'LVL': [undefined, 'Ls'],
        'LYD': [undefined, undefined, 3],
        'MGA': [undefined, 'Ar', 0],
        'MGF': [undefined, undefined, 0],
        'MMK': [undefined, 'K', 0],
        'MNT': [undefined, '₮', 2],
        'MRO': [undefined, undefined, 0],
        'MUR': [undefined, 'Rs', 2],
        'MXN': ['MX$', '$'],
        'MYR': [undefined, 'RM'],
        'NAD': [undefined, '$'],
        'NGN': [undefined, '₦'],
        'NIO': [undefined, 'C$'],
        'NOK': [undefined, 'kr', 2],
        'NPR': [undefined, 'Rs'],
        'NZD': ['NZ$', '$'],
        'OMR': [undefined, undefined, 3],
        'PHP': [undefined, '₱'],
        'PKR': [undefined, 'Rs', 2],
        'PLN': [undefined, 'zł'],
        'PYG': [undefined, '₲', 0],
        'RON': [undefined, 'lei'],
        'RSD': [undefined, undefined, 0],
        'RUB': [undefined, '₽'],
        'RUR': [undefined, 'р.'],
        'RWF': [undefined, 'RF', 0],
        'SBD': [undefined, '$'],
        'SEK': [undefined, 'kr', 2],
        'SGD': [undefined, '$'],
        'SHP': [undefined, '£'],
        'SLL': [undefined, undefined, 0],
        'SOS': [undefined, undefined, 0],
        'SRD': [undefined, '$'],
        'SSP': [undefined, '£'],
        'STD': [undefined, undefined, 0],
        'STN': [undefined, 'Db'],
        'SYP': [undefined, '£', 0],
        'THB': [undefined, '฿'],
        'TMM': [undefined, undefined, 0],
        'TND': [undefined, undefined, 3],
        'TOP': [undefined, 'T$'],
        'TRL': [undefined, undefined, 0],
        'TRY': [undefined, '₺'],
        'TTD': [undefined, '$'],
        'TWD': ['NT$', '$', 2],
        'TZS': [undefined, undefined, 2],
        'UAH': [undefined, '₴'],
        'UGX': [undefined, undefined, 0],
        'USD': ['$'],
        'UYI': [undefined, undefined, 0],
        'UYU': [undefined, '$'],
        'UYW': [undefined, undefined, 4],
        'UZS': [undefined, undefined, 2],
        'VEF': [undefined, 'Bs', 2],
        'VND': ['₫', undefined, 0],
        'VUV': [undefined, undefined, 0],
        'XAF': ['FCFA', undefined, 0],
        'XCD': ['EC$', '$'],
        'XOF': ['CFA', undefined, 0],
        'XPF': ['CFPF', undefined, 0],
        'XXX': ['¤'],
        'YER': [undefined, undefined, 0],
        'ZAR': [undefined, 'R'],
        'ZMK': [undefined, undefined, 0],
        'ZMW': [undefined, 'ZK'],
        'ZWD': [undefined, undefined, 0]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/src/i18n/locale_data_api.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @enum {number} */
    const NumberFormatStyle = {
        Decimal: 0,
        Percent: 1,
        Currency: 2,
        Scientific: 3,
    };
    NumberFormatStyle[NumberFormatStyle.Decimal] = 'Decimal';
    NumberFormatStyle[NumberFormatStyle.Percent] = 'Percent';
    NumberFormatStyle[NumberFormatStyle.Currency] = 'Currency';
    NumberFormatStyle[NumberFormatStyle.Scientific] = 'Scientific';
    /** @enum {number} */
    const Plural = {
        Zero: 0,
        One: 1,
        Two: 2,
        Few: 3,
        Many: 4,
        Other: 5,
    };
    Plural[Plural.Zero] = 'Zero';
    Plural[Plural.One] = 'One';
    Plural[Plural.Two] = 'Two';
    Plural[Plural.Few] = 'Few';
    Plural[Plural.Many] = 'Many';
    Plural[Plural.Other] = 'Other';
    /** @enum {number} */
    const FormStyle = {
        Format: 0,
        Standalone: 1,
    };
    FormStyle[FormStyle.Format] = 'Format';
    FormStyle[FormStyle.Standalone] = 'Standalone';
    /** @enum {number} */
    const TranslationWidth = {
        /** 1 character for `en-US`. For example: 'S' */
        Narrow: 0,
        /** 3 characters for `en-US`. For example: 'Sun' */
        Abbreviated: 1,
        /** Full length for `en-US`. For example: "Sunday" */
        Wide: 2,
        /** 2 characters for `en-US`, For example: "Su" */
        Short: 3,
    };
    TranslationWidth[TranslationWidth.Narrow] = 'Narrow';
    TranslationWidth[TranslationWidth.Abbreviated] = 'Abbreviated';
    TranslationWidth[TranslationWidth.Wide] = 'Wide';
    TranslationWidth[TranslationWidth.Short] = 'Short';
    /** @enum {number} */
    const FormatWidth = {
        /**
         * For `en-US`, 'M/d/yy, h:mm a'`
         * (Example: `6/15/15, 9:03 AM`)
         */
        Short: 0,
        /**
         * For `en-US`, `'MMM d, y, h:mm:ss a'`
         * (Example: `Jun 15, 2015, 9:03:01 AM`)
         */
        Medium: 1,
        /**
         * For `en-US`, `'MMMM d, y, h:mm:ss a z'`
         * (Example: `June 15, 2015 at 9:03:01 AM GMT+1`)
         */
        Long: 2,
        /**
         * For `en-US`, `'EEEE, MMMM d, y, h:mm:ss a zzzz'`
         * (Example: `Monday, June 15, 2015 at 9:03:01 AM GMT+01:00`)
         */
        Full: 3,
    };
    FormatWidth[FormatWidth.Short] = 'Short';
    FormatWidth[FormatWidth.Medium] = 'Medium';
    FormatWidth[FormatWidth.Long] = 'Long';
    FormatWidth[FormatWidth.Full] = 'Full';
    /** @enum {number} */
    const NumberSymbol = {
        /**
         * Decimal separator.
         * For `en-US`, the dot character.
         * Example : 2,345`.`67
         */
        Decimal: 0,
        /**
         * Grouping separator, typically for thousands.
         * For `en-US`, the comma character.
         * Example: 2`,`345.67
         */
        Group: 1,
        /**
         * List-item separator.
         * Example: "one, two, and three"
         */
        List: 2,
        /**
         * Sign for percentage (out of 100).
         * Example: 23.4%
         */
        PercentSign: 3,
        /**
         * Sign for positive numbers.
         * Example: +23
         */
        PlusSign: 4,
        /**
         * Sign for negative numbers.
         * Example: -23
         */
        MinusSign: 5,
        /**
         * Computer notation for exponential value (n times a power of 10).
         * Example: 1.2E3
         */
        Exponential: 6,
        /**
         * Human-readable format of exponential.
         * Example: 1.2x103
         */
        SuperscriptingExponent: 7,
        /**
         * Sign for permille (out of 1000).
         * Example: 23.4‰
         */
        PerMille: 8,
        /**
         * Infinity, can be used with plus and minus.
         * Example: ∞, +∞, -∞
         */
        Infinity: 9,
        /**
         * Not a number.
         * Example: NaN
         */
        NaN: 10,
        /**
         * Symbol used between time units.
         * Example: 10:52
         */
        TimeSeparator: 11,
        /**
         * Decimal separator for currency values (fallback to `Decimal`).
         * Example: $2,345.67
         */
        CurrencyDecimal: 12,
        /**
         * Group separator for currency values (fallback to `Group`).
         * Example: $2,345.67
         */
        CurrencyGroup: 13,
    };
    NumberSymbol[NumberSymbol.Decimal] = 'Decimal';
    NumberSymbol[NumberSymbol.Group] = 'Group';
    NumberSymbol[NumberSymbol.List] = 'List';
    NumberSymbol[NumberSymbol.PercentSign] = 'PercentSign';
    NumberSymbol[NumberSymbol.PlusSign] = 'PlusSign';
    NumberSymbol[NumberSymbol.MinusSign] = 'MinusSign';
    NumberSymbol[NumberSymbol.Exponential] = 'Exponential';
    NumberSymbol[NumberSymbol.SuperscriptingExponent] = 'SuperscriptingExponent';
    NumberSymbol[NumberSymbol.PerMille] = 'PerMille';
    NumberSymbol[NumberSymbol.Infinity] = 'Infinity';
    NumberSymbol[NumberSymbol.NaN] = 'NaN';
    NumberSymbol[NumberSymbol.TimeSeparator] = 'TimeSeparator';
    NumberSymbol[NumberSymbol.CurrencyDecimal] = 'CurrencyDecimal';
    NumberSymbol[NumberSymbol.CurrencyGroup] = 'CurrencyGroup';
    /**
     * Retrieves the locale ID from the currently loaded locale.
     * The loaded locale could be, for example, a global one rather than a regional one.
     * @see [Internationalization (i18n) Guide](https://angular.io/guide/i18n)
     *
     * \@publicApi
     * @param {?} locale A locale code, such as `fr-FR`.
     * @return {?} The locale code. For example, `fr`.
     */
    function getLocaleId$1(locale) {
        return findLocaleData(locale)[LocaleDataIndex.LocaleId];
    }
    /**
     * Retrieves day period strings for the given locale.
     *
     * @see [Internationalization (i18n) Guide](https://angular.io/guide/i18n)
     *
     * \@publicApi
     * @param {?} locale A locale code for the locale format rules to use.
     * @param {?} formStyle The required grammatical form.
     * @param {?} width The required character width.
     * @return {?} An array of localized period strings. For example, `[AM, PM]` for `en-US`.
     */
    function getLocaleDayPeriods(locale, formStyle, width) {
        /** @type {?} */
        const data = findLocaleData(locale);
        /** @type {?} */
        const amPmData = (/** @type {?} */ ([
            data[LocaleDataIndex.DayPeriodsFormat], data[LocaleDataIndex.DayPeriodsStandalone]
        ]));
        /** @type {?} */
        const amPm = getLastDefinedValue(amPmData, formStyle);
        return getLastDefinedValue(amPm, width);
    }
    /**
     * Retrieves days of the week for the given locale, using the Gregorian calendar.
     *
     * @see [Internationalization (i18n) Guide](https://angular.io/guide/i18n)
     *
     * \@publicApi
     * @param {?} locale A locale code for the locale format rules to use.
     * @param {?} formStyle The required grammatical form.
     * @param {?} width The required character width.
     * @return {?} An array of localized name strings.
     * For example,`[Sunday, Monday, ... Saturday]` for `en-US`.
     */
    function getLocaleDayNames(locale, formStyle, width) {
        /** @type {?} */
        const data = findLocaleData(locale);
        /** @type {?} */
        const daysData = (/** @type {?} */ ([data[LocaleDataIndex.DaysFormat], data[LocaleDataIndex.DaysStandalone]]));
        /** @type {?} */
        const days = getLastDefinedValue(daysData, formStyle);
        return getLastDefinedValue(days, width);
    }
    /**
     * Retrieves months of the year for the given locale, using the Gregorian calendar.
     *
     * @see [Internationalization (i18n) Guide](https://angular.io/guide/i18n)
     *
     * \@publicApi
     * @param {?} locale A locale code for the locale format rules to use.
     * @param {?} formStyle The required grammatical form.
     * @param {?} width The required character width.
     * @return {?} An array of localized name strings.
     * For example,  `[January, February, ...]` for `en-US`.
     */
    function getLocaleMonthNames(locale, formStyle, width) {
        /** @type {?} */
        const data = findLocaleData(locale);
        /** @type {?} */
        const monthsData = (/** @type {?} */ ([data[LocaleDataIndex.MonthsFormat], data[LocaleDataIndex.MonthsStandalone]]));
        /** @type {?} */
        const months = getLastDefinedValue(monthsData, formStyle);
        return getLastDefinedValue(months, width);
    }
    /**
     * Retrieves Gregorian-calendar eras for the given locale.
     * @see [Internationalization (i18n) Guide](https://angular.io/guide/i18n)
     *
     * \@publicApi
     * @param {?} locale A locale code for the locale format rules to use.
     * @param {?} width The required character width.
     * @return {?} An array of localized era strings.
     * For example, `[AD, BC]` for `en-US`.
     */
    function getLocaleEraNames(locale, width) {
        /** @type {?} */
        const data = findLocaleData(locale);
        /** @type {?} */
        const erasData = (/** @type {?} */ (data[LocaleDataIndex.Eras]));
        return getLastDefinedValue(erasData, width);
    }
    /**
     * Retrieves a localized date-value formating string.
     *
     * @see `FormatWidth` / [Internationalization (i18n) Guide](https://angular.io/guide/i18n)
     *
     * \@publicApi
     * @param {?} locale A locale code for the locale format rules to use.
     * @param {?} width The format type.
     * @return {?} The localized formating string.
     */
    function getLocaleDateFormat(locale, width) {
        /** @type {?} */
        const data = findLocaleData(locale);
        return getLastDefinedValue(data[LocaleDataIndex.DateFormat], width);
    }
    /**
     * Retrieves a localized time-value formatting string.
     *
     * @see `FormatWidth` / [Internationalization (i18n) Guide](https://angular.io/guide/i18n)
     * \@publicApi
     * @param {?} locale A locale code for the locale format rules to use.
     * @param {?} width The format type.
     * @return {?} The localized formatting string.
     */
    function getLocaleTimeFormat(locale, width) {
        /** @type {?} */
        const data = findLocaleData(locale);
        return getLastDefinedValue(data[LocaleDataIndex.TimeFormat], width);
    }
    /**
     * Retrieves a localized date-time formatting string.
     *
     * @see `FormatWidth` / [Internationalization (i18n) Guide](https://angular.io/guide/i18n)
     *
     * \@publicApi
     * @param {?} locale A locale code for the locale format rules to use.
     * @param {?} width The format type.
     * @return {?} The localized formatting string.
     */
    function getLocaleDateTimeFormat(locale, width) {
        /** @type {?} */
        const data = findLocaleData(locale);
        /** @type {?} */
        const dateTimeFormatData = (/** @type {?} */ (data[LocaleDataIndex.DateTimeFormat]));
        return getLastDefinedValue(dateTimeFormatData, width);
    }
    /**
     * Retrieves a localized number symbol that can be used to replace placeholders in number formats.
     * @see `NumberSymbol` / [Internationalization (i18n) Guide](https://angular.io/guide/i18n)
     *
     * \@publicApi
     * @param {?} locale The locale code.
     * @param {?} symbol The symbol to localize.
     * @return {?} The character for the localized symbol.
     */
    function getLocaleNumberSymbol(locale, symbol) {
        /** @type {?} */
        const data = findLocaleData(locale);
        /** @type {?} */
        const res = data[LocaleDataIndex.NumberSymbols][symbol];
        if (typeof res === 'undefined') {
            if (symbol === NumberSymbol.CurrencyDecimal) {
                return data[LocaleDataIndex.NumberSymbols][NumberSymbol.Decimal];
            }
            else if (symbol === NumberSymbol.CurrencyGroup) {
                return data[LocaleDataIndex.NumberSymbols][NumberSymbol.Group];
            }
        }
        return res;
    }
    /**
     * Retrieves a number format for a given locale.
     *
     * Numbers are formatted using patterns, like `#,###.00`. For example, the pattern `#,###.00`
     * when used to format the number 12345.678 could result in "12'345,678". That would happen if the
     * grouping separator for your language is an apostrophe, and the decimal separator is a comma.
     *
     * <b>Important:</b> The characters `.` `,` `0` `#` (and others below) are special placeholders
     * that stand for the decimal separator, and so on, and are NOT real characters.
     * You must NOT "translate" the placeholders. For example, don't change `.` to `,` even though in
     * your language the decimal point is written with a comma. The symbols should be replaced by the
     * local equivalents, using the appropriate `NumberSymbol` for your language.
     *
     * Here are the special characters used in number patterns:
     *
     * | Symbol | Meaning |
     * |--------|---------|
     * | . | Replaced automatically by the character used for the decimal point. |
     * | , | Replaced by the "grouping" (thousands) separator. |
     * | 0 | Replaced by a digit (or zero if there aren't enough digits). |
     * | # | Replaced by a digit (or nothing if there aren't enough). |
     * | ¤ | Replaced by a currency symbol, such as $ or USD. |
     * | % | Marks a percent format. The % symbol may change position, but must be retained. |
     * | E | Marks a scientific format. The E symbol may change position, but must be retained. |
     * | ' | Special characters used as literal characters are quoted with ASCII single quotes. |
     *
     * @see `NumberFormatStyle` / [CLDR website](http://cldr.unicode.org/translation/number-patterns) / [Internationalization (i18n) Guide](https://angular.io/guide/i18n)
     *
     * \@publicApi
     * @param {?} locale A locale code for the locale format rules to use.
     * @param {?} type The type of numeric value to be formatted (such as `Decimal` or `Currency`.)
     * @return {?} The localized format string.
     */
    function getLocaleNumberFormat(locale, type) {
        /** @type {?} */
        const data = findLocaleData(locale);
        return data[LocaleDataIndex.NumberFormats][type];
    }
    /**
     * Retrieves the currency values for a given locale.
     * @see [Internationalization (i18n) Guide](https://angular.io/guide/i18n)
     * @param {?} locale A locale code for the locale format rules to use.
     * @return {?} The currency values.
     */
    function getLocaleCurrencies(locale) {
        /** @type {?} */
        const data = findLocaleData(locale);
        return data[LocaleDataIndex.Currencies];
    }
    /**
     * \@alias core/ɵgetLocalePluralCase
     * \@publicApi
     * @type {?}
     */
    const getLocalePluralCase$1 = getLocalePluralCase;
    /**
     * @param {?} data
     * @return {?}
     */
    function checkFullData(data) {
        if (!data[LocaleDataIndex.ExtraData]) {
            throw new Error(`Missing extra locale data for the locale "${data[LocaleDataIndex
            .LocaleId]}". Use "registerLocaleData" to load new data. See the "I18n guide" on angular.io to know more.`);
        }
    }
    /**
     * Retrieves locale-specific rules used to determine which day period to use
     * when more than one period is defined for a locale.
     *
     * There is a rule for each defined day period. The
     * first rule is applied to the first day period and so on.
     * Fall back to AM/PM when no rules are available.
     *
     * A rule can specify a period as time range, or as a single time value.
     *
     * This functionality is only available when you have loaded the full locale data.
     * See the ["I18n guide"](guide/i18n#i18n-pipes).
     *
     * @see `getLocaleExtraDayPeriods()` / [Internationalization (i18n) Guide](https://angular.io/guide/i18n)
     *
     * \@publicApi
     * @param {?} locale A locale code for the locale format rules to use.
     * @return {?} The rules for the locale, a single time value or array of *from-time, to-time*,
     * or null if no periods are available.
     *
     */
    function getLocaleExtraDayPeriodRules(locale) {
        /** @type {?} */
        const data = findLocaleData(locale);
        checkFullData(data);
        /** @type {?} */
        const rules = data[LocaleDataIndex.ExtraData][2 /* ExtraDayPeriodsRules */] || [];
        return rules.map((/**
         * @param {?} rule
         * @return {?}
         */
        (rule) => {
            if (typeof rule === 'string') {
                return extractTime(rule);
            }
            return [extractTime(rule[0]), extractTime(rule[1])];
        }));
    }
    /**
     * Retrieves locale-specific day periods, which indicate roughly how a day is broken up
     * in different languages.
     * For example, for `en-US`, periods are morning, noon, afternoon, evening, and midnight.
     *
     * This functionality is only available when you have loaded the full locale data.
     * See the ["I18n guide"](guide/i18n#i18n-pipes).
     *
     * @see `getLocaleExtraDayPeriodRules()` / [Internationalization (i18n) Guide](https://angular.io/guide/i18n)
     *
     * \@publicApi
     * @param {?} locale A locale code for the locale format rules to use.
     * @param {?} formStyle The required grammatical form.
     * @param {?} width The required character width.
     * @return {?} The translated day-period strings.
     */
    function getLocaleExtraDayPeriods(locale, formStyle, width) {
        /** @type {?} */
        const data = findLocaleData(locale);
        checkFullData(data);
        /** @type {?} */
        const dayPeriodsData = (/** @type {?} */ ([
            data[LocaleDataIndex.ExtraData][0 /* ExtraDayPeriodFormats */],
            data[LocaleDataIndex.ExtraData][1 /* ExtraDayPeriodStandalone */]
        ]));
        /** @type {?} */
        const dayPeriods = getLastDefinedValue(dayPeriodsData, formStyle) || [];
        return getLastDefinedValue(dayPeriods, width) || [];
    }
    /**
     * Retrieves the first value that is defined in an array, going backwards from an index position.
     *
     * To avoid repeating the same data (as when the "format" and "standalone" forms are the same)
     * add the first value to the locale data arrays, and add other values only if they are different.
     *
     * @see [Internationalization (i18n) Guide](https://angular.io/guide/i18n)
     *
     * \@publicApi
     * @template T
     * @param {?} data The data array to retrieve from.
     * @param {?} index A 0-based index into the array to start from.
     * @return {?} The value immediately before the given index position.
     */
    function getLastDefinedValue(data, index) {
        for (let i = index; i > -1; i--) {
            if (typeof data[i] !== 'undefined') {
                return data[i];
            }
        }
        throw new Error('Locale data API: locale data undefined');
    }
    /**
     * Extracts the hours and minutes from a string like "15:45"
     * @param {?} time
     * @return {?}
     */
    function extractTime(time) {
        const [h, m] = time.split(':');
        return { hours: +h, minutes: +m };
    }
    /**
     * Retrieves the currency symbol for a given currency code.
     *
     * For example, for the default `en-US` locale, the code `USD` can
     * be represented by the narrow symbol `$` or the wide symbol `US$`.
     *
     * @see [Internationalization (i18n) Guide](https://angular.io/guide/i18n)
     *
     * \@publicApi
     * @param {?} code The currency code.
     * @param {?} format The format, `wide` or `narrow`.
     * @param {?=} locale A locale code for the locale format rules to use.
     *
     * @return {?} The symbol, or the currency code if no symbol is available.
     */
    function getCurrencySymbol(code, format, locale = 'en') {
        /** @type {?} */
        const currency = getLocaleCurrencies(locale)[code] || CURRENCIES_EN[code] || [];
        /** @type {?} */
        const symbolNarrow = currency[1 /* SymbolNarrow */];
        if (format === 'narrow' && typeof symbolNarrow === 'string') {
            return symbolNarrow;
        }
        return currency[0 /* Symbol */] || code;
    }
    // Most currencies have cents, that's why the default is 2
    /** @type {?} */
    const DEFAULT_NB_OF_CURRENCY_DIGITS = 2;
    /**
     * Reports the number of decimal digits for a given currency.
     * The value depends upon the presence of cents in that particular currency.
     *
     * @see [Internationalization (i18n) Guide](https://angular.io/guide/i18n)
     *
     * \@publicApi
     * @param {?} code The currency code.
     * @return {?} The number of decimal digits, typically 0 or 2.
     */
    function getNumberOfCurrencyDigits(code) {
        /** @type {?} */
        let digits;
        /** @type {?} */
        const currency = CURRENCIES_EN[code];
        if (currency) {
            digits = currency[2 /* NbOfDigits */];
        }
        return typeof digits === 'number' ? digits : DEFAULT_NB_OF_CURRENCY_DIGITS;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/src/i18n/format_date.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const ISO8601_DATE_REGEX = /^(\d{4})-?(\d\d)-?(\d\d)(?:T(\d\d)(?::?(\d\d)(?::?(\d\d)(?:\.(\d+))?)?)?(Z|([+-])(\d\d):?(\d\d))?)?$/;
    //    1        2       3         4          5          6          7          8  9     10      11
    /** @type {?} */
    const NAMED_FORMATS = {};
    /** @type {?} */
    const DATE_FORMATS_SPLIT = /((?:[^GyMLwWdEabBhHmsSzZO']+)|(?:'(?:[^']|'')*')|(?:G{1,5}|y{1,4}|M{1,5}|L{1,5}|w{1,2}|W{1}|d{1,2}|E{1,6}|a{1,5}|b{1,5}|B{1,5}|h{1,2}|H{1,2}|m{1,2}|s{1,2}|S{1,3}|z{1,4}|Z{1,5}|O{1,4}))([\s\S]*)/;
    /** @enum {number} */
    const ZoneWidth = {
        Short: 0,
        ShortGMT: 1,
        Long: 2,
        Extended: 3,
    };
    ZoneWidth[ZoneWidth.Short] = 'Short';
    ZoneWidth[ZoneWidth.ShortGMT] = 'ShortGMT';
    ZoneWidth[ZoneWidth.Long] = 'Long';
    ZoneWidth[ZoneWidth.Extended] = 'Extended';
    /** @enum {number} */
    const DateType = {
        FullYear: 0,
        Month: 1,
        Date: 2,
        Hours: 3,
        Minutes: 4,
        Seconds: 5,
        FractionalSeconds: 6,
        Day: 7,
    };
    DateType[DateType.FullYear] = 'FullYear';
    DateType[DateType.Month] = 'Month';
    DateType[DateType.Date] = 'Date';
    DateType[DateType.Hours] = 'Hours';
    DateType[DateType.Minutes] = 'Minutes';
    DateType[DateType.Seconds] = 'Seconds';
    DateType[DateType.FractionalSeconds] = 'FractionalSeconds';
    DateType[DateType.Day] = 'Day';
    /** @enum {number} */
    const TranslationType = {
        DayPeriods: 0,
        Days: 1,
        Months: 2,
        Eras: 3,
    };
    TranslationType[TranslationType.DayPeriods] = 'DayPeriods';
    TranslationType[TranslationType.Days] = 'Days';
    TranslationType[TranslationType.Months] = 'Months';
    TranslationType[TranslationType.Eras] = 'Eras';
    /**
     * \@ngModule CommonModule
     * \@description
     *
     * Formats a date according to locale rules.
     *
     * @see `DatePipe` / [Internationalization (i18n) Guide](https://angular.io/guide/i18n)
     *
     * \@publicApi
     * @param {?} value The date to format, as a Date, or a number (milliseconds since UTC epoch)
     * or an [ISO date-time string](https://www.w3.org/TR/NOTE-datetime).
     * @param {?} format The date-time components to include. See `DatePipe` for details.
     * @param {?} locale A locale code for the locale format rules to use.
     * @param {?=} timezone The time zone. A time zone offset from GMT (such as `'+0430'`),
     * or a standard UTC/GMT or continental US time zone abbreviation.
     * If not specified, uses host system settings.
     *
     * @return {?} The formatted date string.
     *
     */
    function formatDate(value, format, locale, timezone) {
        /** @type {?} */
        let date = toDate(value);
        /** @type {?} */
        const namedFormat = getNamedFormat(locale, format);
        format = namedFormat || format;
        /** @type {?} */
        let parts = [];
        /** @type {?} */
        let match;
        while (format) {
            match = DATE_FORMATS_SPLIT.exec(format);
            if (match) {
                parts = parts.concat(match.slice(1));
                /** @type {?} */
                const part = parts.pop();
                if (!part) {
                    break;
                }
                format = part;
            }
            else {
                parts.push(format);
                break;
            }
        }
        /** @type {?} */
        let dateTimezoneOffset = date.getTimezoneOffset();
        if (timezone) {
            dateTimezoneOffset = timezoneToOffset(timezone, dateTimezoneOffset);
            date = convertTimezoneToLocal(date, timezone, true);
        }
        /** @type {?} */
        let text = '';
        parts.forEach((/**
         * @param {?} value
         * @return {?}
         */
        value => {
            /** @type {?} */
            const dateFormatter = getDateFormatter(value);
            text += dateFormatter ?
                dateFormatter(date, locale, dateTimezoneOffset) :
                value === '\'\'' ? '\'' : value.replace(/(^'|'$)/g, '').replace(/''/g, '\'');
        }));
        return text;
    }
    /**
     * @param {?} locale
     * @param {?} format
     * @return {?}
     */
    function getNamedFormat(locale, format) {
        /** @type {?} */
        const localeId = getLocaleId$1(locale);
        NAMED_FORMATS[localeId] = NAMED_FORMATS[localeId] || {};
        if (NAMED_FORMATS[localeId][format]) {
            return NAMED_FORMATS[localeId][format];
        }
        /** @type {?} */
        let formatValue = '';
        switch (format) {
            case 'shortDate':
                formatValue = getLocaleDateFormat(locale, FormatWidth.Short);
                break;
            case 'mediumDate':
                formatValue = getLocaleDateFormat(locale, FormatWidth.Medium);
                break;
            case 'longDate':
                formatValue = getLocaleDateFormat(locale, FormatWidth.Long);
                break;
            case 'fullDate':
                formatValue = getLocaleDateFormat(locale, FormatWidth.Full);
                break;
            case 'shortTime':
                formatValue = getLocaleTimeFormat(locale, FormatWidth.Short);
                break;
            case 'mediumTime':
                formatValue = getLocaleTimeFormat(locale, FormatWidth.Medium);
                break;
            case 'longTime':
                formatValue = getLocaleTimeFormat(locale, FormatWidth.Long);
                break;
            case 'fullTime':
                formatValue = getLocaleTimeFormat(locale, FormatWidth.Full);
                break;
            case 'short':
                /** @type {?} */
                const shortTime = getNamedFormat(locale, 'shortTime');
                /** @type {?} */
                const shortDate = getNamedFormat(locale, 'shortDate');
                formatValue = formatDateTime(getLocaleDateTimeFormat(locale, FormatWidth.Short), [shortTime, shortDate]);
                break;
            case 'medium':
                /** @type {?} */
                const mediumTime = getNamedFormat(locale, 'mediumTime');
                /** @type {?} */
                const mediumDate = getNamedFormat(locale, 'mediumDate');
                formatValue = formatDateTime(getLocaleDateTimeFormat(locale, FormatWidth.Medium), [mediumTime, mediumDate]);
                break;
            case 'long':
                /** @type {?} */
                const longTime = getNamedFormat(locale, 'longTime');
                /** @type {?} */
                const longDate = getNamedFormat(locale, 'longDate');
                formatValue =
                    formatDateTime(getLocaleDateTimeFormat(locale, FormatWidth.Long), [longTime, longDate]);
                break;
            case 'full':
                /** @type {?} */
                const fullTime = getNamedFormat(locale, 'fullTime');
                /** @type {?} */
                const fullDate = getNamedFormat(locale, 'fullDate');
                formatValue =
                    formatDateTime(getLocaleDateTimeFormat(locale, FormatWidth.Full), [fullTime, fullDate]);
                break;
        }
        if (formatValue) {
            NAMED_FORMATS[localeId][format] = formatValue;
        }
        return formatValue;
    }
    /**
     * @param {?} str
     * @param {?} opt_values
     * @return {?}
     */
    function formatDateTime(str, opt_values) {
        if (opt_values) {
            str = str.replace(/\{([^}]+)}/g, (/**
             * @param {?} match
             * @param {?} key
             * @return {?}
             */
            function (match, key) {
                return (opt_values != null && key in opt_values) ? opt_values[key] : match;
            }));
        }
        return str;
    }
    /**
     * @param {?} num
     * @param {?} digits
     * @param {?=} minusSign
     * @param {?=} trim
     * @param {?=} negWrap
     * @return {?}
     */
    function padNumber(num, digits, minusSign = '-', trim, negWrap) {
        /** @type {?} */
        let neg = '';
        if (num < 0 || (negWrap && num <= 0)) {
            if (negWrap) {
                num = -num + 1;
            }
            else {
                num = -num;
                neg = minusSign;
            }
        }
        /** @type {?} */
        let strNum = String(num);
        while (strNum.length < digits) {
            strNum = '0' + strNum;
        }
        if (trim) {
            strNum = strNum.substr(strNum.length - digits);
        }
        return neg + strNum;
    }
    /**
     * @param {?} milliseconds
     * @param {?} digits
     * @return {?}
     */
    function formatFractionalSeconds(milliseconds, digits) {
        /** @type {?} */
        const strMs = padNumber(milliseconds, 3);
        return strMs.substr(0, digits);
    }
    /**
     * Returns a date formatter that transforms a date into its locale digit representation
     * @param {?} name
     * @param {?} size
     * @param {?=} offset
     * @param {?=} trim
     * @param {?=} negWrap
     * @return {?}
     */
    function dateGetter(name, size, offset = 0, trim = false, negWrap = false) {
        return (/**
         * @param {?} date
         * @param {?} locale
         * @return {?}
         */
        function (date, locale) {
            /** @type {?} */
            let part = getDatePart(name, date);
            if (offset > 0 || part > -offset) {
                part += offset;
            }
            if (name === DateType.Hours) {
                if (part === 0 && offset === -12) {
                    part = 12;
                }
            }
            else if (name === DateType.FractionalSeconds) {
                return formatFractionalSeconds(part, size);
            }
            /** @type {?} */
            const localeMinus = getLocaleNumberSymbol(locale, NumberSymbol.MinusSign);
            return padNumber(part, size, localeMinus, trim, negWrap);
        });
    }
    /**
     * @param {?} part
     * @param {?} date
     * @return {?}
     */
    function getDatePart(part, date) {
        switch (part) {
            case DateType.FullYear:
                return date.getFullYear();
            case DateType.Month:
                return date.getMonth();
            case DateType.Date:
                return date.getDate();
            case DateType.Hours:
                return date.getHours();
            case DateType.Minutes:
                return date.getMinutes();
            case DateType.Seconds:
                return date.getSeconds();
            case DateType.FractionalSeconds:
                return date.getMilliseconds();
            case DateType.Day:
                return date.getDay();
            default:
                throw new Error(`Unknown DateType value "${part}".`);
        }
    }
    /**
     * Returns a date formatter that transforms a date into its locale string representation
     * @param {?} name
     * @param {?} width
     * @param {?=} form
     * @param {?=} extended
     * @return {?}
     */
    function dateStrGetter(name, width, form = FormStyle.Format, extended = false) {
        return (/**
         * @param {?} date
         * @param {?} locale
         * @return {?}
         */
        function (date, locale) {
            return getDateTranslation(date, locale, name, width, form, extended);
        });
    }
    /**
     * Returns the locale translation of a date for a given form, type and width
     * @param {?} date
     * @param {?} locale
     * @param {?} name
     * @param {?} width
     * @param {?} form
     * @param {?} extended
     * @return {?}
     */
    function getDateTranslation(date, locale, name, width, form, extended) {
        switch (name) {
            case TranslationType.Months:
                return getLocaleMonthNames(locale, form, width)[date.getMonth()];
            case TranslationType.Days:
                return getLocaleDayNames(locale, form, width)[date.getDay()];
            case TranslationType.DayPeriods:
                /** @type {?} */
                const currentHours = date.getHours();
                /** @type {?} */
                const currentMinutes = date.getMinutes();
                if (extended) {
                    /** @type {?} */
                    const rules = getLocaleExtraDayPeriodRules(locale);
                    /** @type {?} */
                    const dayPeriods = getLocaleExtraDayPeriods(locale, form, width);
                    /** @type {?} */
                    let result;
                    rules.forEach((/**
                     * @param {?} rule
                     * @param {?} index
                     * @return {?}
                     */
                    (rule, index) => {
                        if (Array.isArray(rule)) {
                            // morning, afternoon, evening, night
                            const { hours: hoursFrom, minutes: minutesFrom } = rule[0];
                            const { hours: hoursTo, minutes: minutesTo } = rule[1];
                            if (currentHours >= hoursFrom && currentMinutes >= minutesFrom &&
                                (currentHours < hoursTo ||
                                    (currentHours === hoursTo && currentMinutes < minutesTo))) {
                                result = dayPeriods[index];
                            }
                        }
                        else { // noon or midnight
                            // noon or midnight
                            const { hours, minutes } = rule;
                            if (hours === currentHours && minutes === currentMinutes) {
                                result = dayPeriods[index];
                            }
                        }
                    }));
                    if (result) {
                        return result;
                    }
                }
                // if no rules for the day periods, we use am/pm by default
                return getLocaleDayPeriods(locale, form, (/** @type {?} */ (width)))[currentHours < 12 ? 0 : 1];
            case TranslationType.Eras:
                return getLocaleEraNames(locale, (/** @type {?} */ (width)))[date.getFullYear() <= 0 ? 0 : 1];
            default:
                // This default case is not needed by TypeScript compiler, as the switch is exhaustive.
                // However Closure Compiler does not understand that and reports an error in typed mode.
                // The `throw new Error` below works around the problem, and the unexpected: never variable
                // makes sure tsc still checks this code is unreachable.
                /** @type {?} */
                const unexpected = name;
                throw new Error(`unexpected translation type ${unexpected}`);
        }
    }
    /**
     * Returns a date formatter that transforms a date and an offset into a timezone with ISO8601 or
     * GMT format depending on the width (eg: short = +0430, short:GMT = GMT+4, long = GMT+04:30,
     * extended = +04:30)
     * @param {?} width
     * @return {?}
     */
    function timeZoneGetter(width) {
        return (/**
         * @param {?} date
         * @param {?} locale
         * @param {?} offset
         * @return {?}
         */
        function (date, locale, offset) {
            /** @type {?} */
            const zone = -1 * offset;
            /** @type {?} */
            const minusSign = getLocaleNumberSymbol(locale, NumberSymbol.MinusSign);
            /** @type {?} */
            const hours = zone > 0 ? Math.floor(zone / 60) : Math.ceil(zone / 60);
            switch (width) {
                case ZoneWidth.Short:
                    return ((zone >= 0) ? '+' : '') + padNumber(hours, 2, minusSign) +
                        padNumber(Math.abs(zone % 60), 2, minusSign);
                case ZoneWidth.ShortGMT:
                    return 'GMT' + ((zone >= 0) ? '+' : '') + padNumber(hours, 1, minusSign);
                case ZoneWidth.Long:
                    return 'GMT' + ((zone >= 0) ? '+' : '') + padNumber(hours, 2, minusSign) + ':' +
                        padNumber(Math.abs(zone % 60), 2, minusSign);
                case ZoneWidth.Extended:
                    if (offset === 0) {
                        return 'Z';
                    }
                    else {
                        return ((zone >= 0) ? '+' : '') + padNumber(hours, 2, minusSign) + ':' +
                            padNumber(Math.abs(zone % 60), 2, minusSign);
                    }
                default:
                    throw new Error(`Unknown zone width "${width}"`);
            }
        });
    }
    /** @type {?} */
    const JANUARY = 0;
    /** @type {?} */
    const THURSDAY = 4;
    /**
     * @param {?} year
     * @return {?}
     */
    function getFirstThursdayOfYear(year) {
        /** @type {?} */
        const firstDayOfYear = (new Date(year, JANUARY, 1)).getDay();
        return new Date(year, 0, 1 + ((firstDayOfYear <= THURSDAY) ? THURSDAY : THURSDAY + 7) - firstDayOfYear);
    }
    /**
     * @param {?} datetime
     * @return {?}
     */
    function getThursdayThisWeek(datetime) {
        return new Date(datetime.getFullYear(), datetime.getMonth(), datetime.getDate() + (THURSDAY - datetime.getDay()));
    }
    /**
     * @param {?} size
     * @param {?=} monthBased
     * @return {?}
     */
    function weekGetter(size, monthBased = false) {
        return (/**
         * @param {?} date
         * @param {?} locale
         * @return {?}
         */
        function (date, locale) {
            /** @type {?} */
            let result;
            if (monthBased) {
                /** @type {?} */
                const nbDaysBefore1stDayOfMonth = new Date(date.getFullYear(), date.getMonth(), 1).getDay() - 1;
                /** @type {?} */
                const today = date.getDate();
                result = 1 + Math.floor((today + nbDaysBefore1stDayOfMonth) / 7);
            }
            else {
                /** @type {?} */
                const firstThurs = getFirstThursdayOfYear(date.getFullYear());
                /** @type {?} */
                const thisThurs = getThursdayThisWeek(date);
                /** @type {?} */
                const diff = thisThurs.getTime() - firstThurs.getTime();
                result = 1 + Math.round(diff / 6.048e8); // 6.048e8 ms per week
            }
            return padNumber(result, size, getLocaleNumberSymbol(locale, NumberSymbol.MinusSign));
        });
    }
    /** @type {?} */
    const DATE_FORMATS = {};
    // Based on CLDR formats:
    // See complete list: http://www.unicode.org/reports/tr35/tr35-dates.html#Date_Field_Symbol_Table
    // See also explanations: http://cldr.unicode.org/translation/date-time
    // TODO(ocombe): support all missing cldr formats: Y, U, Q, D, F, e, c, j, J, C, A, v, V, X, x
    /**
     * @param {?} format
     * @return {?}
     */
    function getDateFormatter(format) {
        if (DATE_FORMATS[format]) {
            return DATE_FORMATS[format];
        }
        /** @type {?} */
        let formatter;
        switch (format) {
            // Era name (AD/BC)
            case 'G':
            case 'GG':
            case 'GGG':
                formatter = dateStrGetter(TranslationType.Eras, TranslationWidth.Abbreviated);
                break;
            case 'GGGG':
                formatter = dateStrGetter(TranslationType.Eras, TranslationWidth.Wide);
                break;
            case 'GGGGG':
                formatter = dateStrGetter(TranslationType.Eras, TranslationWidth.Narrow);
                break;
            // 1 digit representation of the year, e.g. (AD 1 => 1, AD 199 => 199)
            case 'y':
                formatter = dateGetter(DateType.FullYear, 1, 0, false, true);
                break;
            // 2 digit representation of the year, padded (00-99). (e.g. AD 2001 => 01, AD 2010 => 10)
            case 'yy':
                formatter = dateGetter(DateType.FullYear, 2, 0, true, true);
                break;
            // 3 digit representation of the year, padded (000-999). (e.g. AD 2001 => 01, AD 2010 => 10)
            case 'yyy':
                formatter = dateGetter(DateType.FullYear, 3, 0, false, true);
                break;
            // 4 digit representation of the year (e.g. AD 1 => 0001, AD 2010 => 2010)
            case 'yyyy':
                formatter = dateGetter(DateType.FullYear, 4, 0, false, true);
                break;
            // Month of the year (1-12), numeric
            case 'M':
            case 'L':
                formatter = dateGetter(DateType.Month, 1, 1);
                break;
            case 'MM':
            case 'LL':
                formatter = dateGetter(DateType.Month, 2, 1);
                break;
            // Month of the year (January, ...), string, format
            case 'MMM':
                formatter = dateStrGetter(TranslationType.Months, TranslationWidth.Abbreviated);
                break;
            case 'MMMM':
                formatter = dateStrGetter(TranslationType.Months, TranslationWidth.Wide);
                break;
            case 'MMMMM':
                formatter = dateStrGetter(TranslationType.Months, TranslationWidth.Narrow);
                break;
            // Month of the year (January, ...), string, standalone
            case 'LLL':
                formatter =
                    dateStrGetter(TranslationType.Months, TranslationWidth.Abbreviated, FormStyle.Standalone);
                break;
            case 'LLLL':
                formatter =
                    dateStrGetter(TranslationType.Months, TranslationWidth.Wide, FormStyle.Standalone);
                break;
            case 'LLLLL':
                formatter =
                    dateStrGetter(TranslationType.Months, TranslationWidth.Narrow, FormStyle.Standalone);
                break;
            // Week of the year (1, ... 52)
            case 'w':
                formatter = weekGetter(1);
                break;
            case 'ww':
                formatter = weekGetter(2);
                break;
            // Week of the month (1, ...)
            case 'W':
                formatter = weekGetter(1, true);
                break;
            // Day of the month (1-31)
            case 'd':
                formatter = dateGetter(DateType.Date, 1);
                break;
            case 'dd':
                formatter = dateGetter(DateType.Date, 2);
                break;
            // Day of the Week
            case 'E':
            case 'EE':
            case 'EEE':
                formatter = dateStrGetter(TranslationType.Days, TranslationWidth.Abbreviated);
                break;
            case 'EEEE':
                formatter = dateStrGetter(TranslationType.Days, TranslationWidth.Wide);
                break;
            case 'EEEEE':
                formatter = dateStrGetter(TranslationType.Days, TranslationWidth.Narrow);
                break;
            case 'EEEEEE':
                formatter = dateStrGetter(TranslationType.Days, TranslationWidth.Short);
                break;
            // Generic period of the day (am-pm)
            case 'a':
            case 'aa':
            case 'aaa':
                formatter = dateStrGetter(TranslationType.DayPeriods, TranslationWidth.Abbreviated);
                break;
            case 'aaaa':
                formatter = dateStrGetter(TranslationType.DayPeriods, TranslationWidth.Wide);
                break;
            case 'aaaaa':
                formatter = dateStrGetter(TranslationType.DayPeriods, TranslationWidth.Narrow);
                break;
            // Extended period of the day (midnight, at night, ...), standalone
            case 'b':
            case 'bb':
            case 'bbb':
                formatter = dateStrGetter(TranslationType.DayPeriods, TranslationWidth.Abbreviated, FormStyle.Standalone, true);
                break;
            case 'bbbb':
                formatter = dateStrGetter(TranslationType.DayPeriods, TranslationWidth.Wide, FormStyle.Standalone, true);
                break;
            case 'bbbbb':
                formatter = dateStrGetter(TranslationType.DayPeriods, TranslationWidth.Narrow, FormStyle.Standalone, true);
                break;
            // Extended period of the day (midnight, night, ...), standalone
            case 'B':
            case 'BB':
            case 'BBB':
                formatter = dateStrGetter(TranslationType.DayPeriods, TranslationWidth.Abbreviated, FormStyle.Format, true);
                break;
            case 'BBBB':
                formatter =
                    dateStrGetter(TranslationType.DayPeriods, TranslationWidth.Wide, FormStyle.Format, true);
                break;
            case 'BBBBB':
                formatter = dateStrGetter(TranslationType.DayPeriods, TranslationWidth.Narrow, FormStyle.Format, true);
                break;
            // Hour in AM/PM, (1-12)
            case 'h':
                formatter = dateGetter(DateType.Hours, 1, -12);
                break;
            case 'hh':
                formatter = dateGetter(DateType.Hours, 2, -12);
                break;
            // Hour of the day (0-23)
            case 'H':
                formatter = dateGetter(DateType.Hours, 1);
                break;
            // Hour in day, padded (00-23)
            case 'HH':
                formatter = dateGetter(DateType.Hours, 2);
                break;
            // Minute of the hour (0-59)
            case 'm':
                formatter = dateGetter(DateType.Minutes, 1);
                break;
            case 'mm':
                formatter = dateGetter(DateType.Minutes, 2);
                break;
            // Second of the minute (0-59)
            case 's':
                formatter = dateGetter(DateType.Seconds, 1);
                break;
            case 'ss':
                formatter = dateGetter(DateType.Seconds, 2);
                break;
            // Fractional second
            case 'S':
                formatter = dateGetter(DateType.FractionalSeconds, 1);
                break;
            case 'SS':
                formatter = dateGetter(DateType.FractionalSeconds, 2);
                break;
            case 'SSS':
                formatter = dateGetter(DateType.FractionalSeconds, 3);
                break;
            // Timezone ISO8601 short format (-0430)
            case 'Z':
            case 'ZZ':
            case 'ZZZ':
                formatter = timeZoneGetter(ZoneWidth.Short);
                break;
            // Timezone ISO8601 extended format (-04:30)
            case 'ZZZZZ':
                formatter = timeZoneGetter(ZoneWidth.Extended);
                break;
            // Timezone GMT short format (GMT+4)
            case 'O':
            case 'OO':
            case 'OOO':
            // Should be location, but fallback to format O instead because we don't have the data yet
            case 'z':
            case 'zz':
            case 'zzz':
                formatter = timeZoneGetter(ZoneWidth.ShortGMT);
                break;
            // Timezone GMT long format (GMT+0430)
            case 'OOOO':
            case 'ZZZZ':
            // Should be location, but fallback to format O instead because we don't have the data yet
            case 'zzzz':
                formatter = timeZoneGetter(ZoneWidth.Long);
                break;
            default:
                return null;
        }
        DATE_FORMATS[format] = formatter;
        return formatter;
    }
    /**
     * @param {?} timezone
     * @param {?} fallback
     * @return {?}
     */
    function timezoneToOffset(timezone, fallback) {
        // Support: IE 9-11 only, Edge 13-15+
        // IE/Edge do not "understand" colon (`:`) in timezone
        timezone = timezone.replace(/:/g, '');
        /** @type {?} */
        const requestedTimezoneOffset = Date.parse('Jan 01, 1970 00:00:00 ' + timezone) / 60000;
        return isNaN(requestedTimezoneOffset) ? fallback : requestedTimezoneOffset;
    }
    /**
     * @param {?} date
     * @param {?} minutes
     * @return {?}
     */
    function addDateMinutes(date, minutes) {
        date = new Date(date.getTime());
        date.setMinutes(date.getMinutes() + minutes);
        return date;
    }
    /**
     * @param {?} date
     * @param {?} timezone
     * @param {?} reverse
     * @return {?}
     */
    function convertTimezoneToLocal(date, timezone, reverse) {
        /** @type {?} */
        const reverseValue = reverse ? -1 : 1;
        /** @type {?} */
        const dateTimezoneOffset = date.getTimezoneOffset();
        /** @type {?} */
        const timezoneOffset = timezoneToOffset(timezone, dateTimezoneOffset);
        return addDateMinutes(date, reverseValue * (timezoneOffset - dateTimezoneOffset));
    }
    /**
     * Converts a value to date.
     *
     * Supported input formats:
     * - `Date`
     * - number: timestamp
     * - string: numeric (e.g. "1234"), ISO and date strings in a format supported by
     *   [Date.parse()](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Date/parse).
     *   Note: ISO strings without time return a date without timeoffset.
     *
     * Throws if unable to convert to a date.
     * @param {?} value
     * @return {?}
     */
    function toDate(value) {
        if (isDate$1(value)) {
            return value;
        }
        if (typeof value === 'number' && !isNaN(value)) {
            return new Date(value);
        }
        if (typeof value === 'string') {
            value = value.trim();
            /** @type {?} */
            const parsedNb = parseFloat(value);
            // any string that only contains numbers, like "1234" but not like "1234hello"
            if (!isNaN((/** @type {?} */ (value)) - parsedNb)) {
                return new Date(parsedNb);
            }
            if (/^(\d{4}-\d{1,2}-\d{1,2})$/.test(value)) {
                /* For ISO Strings without time the day, month and year must be extracted from the ISO String
                      before Date creation to avoid time offset and errors in the new Date.
                      If we only replace '-' with ',' in the ISO String ("2015,01,01"), and try to create a new
                      date, some browsers (e.g. IE 9) will throw an invalid Date error.
                      If we leave the '-' ("2015-01-01") and try to create a new Date("2015-01-01") the timeoffset
                      is applied.
                      Note: ISO months are 0 for January, 1 for February, ... */
                const [y, m, d] = value.split('-').map((/**
                 * @param {?} val
                 * @return {?}
                 */
                (val) => +val));
                return new Date(y, m - 1, d);
            }
            /** @type {?} */
            let match;
            if (match = value.match(ISO8601_DATE_REGEX)) {
                return isoStringToDate(match);
            }
        }
        /** @type {?} */
        const date = new Date((/** @type {?} */ (value)));
        if (!isDate$1(date)) {
            throw new Error(`Unable to convert "${value}" into a date`);
        }
        return date;
    }
    /**
     * Converts a date in ISO8601 to a Date.
     * Used instead of `Date.parse` because of browser discrepancies.
     * @param {?} match
     * @return {?}
     */
    function isoStringToDate(match) {
        /** @type {?} */
        const date = new Date(0);
        /** @type {?} */
        let tzHour = 0;
        /** @type {?} */
        let tzMin = 0;
        // match[8] means that the string contains "Z" (UTC) or a timezone like "+01:00" or "+0100"
        /** @type {?} */
        const dateSetter = match[8] ? date.setUTCFullYear : date.setFullYear;
        /** @type {?} */
        const timeSetter = match[8] ? date.setUTCHours : date.setHours;
        // if there is a timezone defined like "+01:00" or "+0100"
        if (match[9]) {
            tzHour = Number(match[9] + match[10]);
            tzMin = Number(match[9] + match[11]);
        }
        dateSetter.call(date, Number(match[1]), Number(match[2]) - 1, Number(match[3]));
        /** @type {?} */
        const h = Number(match[4] || 0) - tzHour;
        /** @type {?} */
        const m = Number(match[5] || 0) - tzMin;
        /** @type {?} */
        const s = Number(match[6] || 0);
        /** @type {?} */
        const ms = Math.round(parseFloat('0.' + (match[7] || 0)) * 1000);
        timeSetter.call(date, h, m, s, ms);
        return date;
    }
    /**
     * @param {?} value
     * @return {?}
     */
    function isDate$1(value) {
        return value instanceof Date && !isNaN(value.valueOf());
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/src/i18n/format_number.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const NUMBER_FORMAT_REGEXP = /^(\d+)?\.((\d+)(-(\d+))?)?$/;
    /** @type {?} */
    const MAX_DIGITS = 22;
    /** @type {?} */
    const DECIMAL_SEP = '.';
    /** @type {?} */
    const ZERO_CHAR = '0';
    /** @type {?} */
    const PATTERN_SEP = ';';
    /** @type {?} */
    const GROUP_SEP = ',';
    /** @type {?} */
    const DIGIT_CHAR = '#';
    /** @type {?} */
    const CURRENCY_CHAR = '¤';
    /** @type {?} */
    const PERCENT_CHAR = '%';
    /**
     * Transforms a number to a locale string based on a style and a format.
     * @param {?} value
     * @param {?} pattern
     * @param {?} locale
     * @param {?} groupSymbol
     * @param {?} decimalSymbol
     * @param {?=} digitsInfo
     * @param {?=} isPercent
     * @return {?}
     */
    function formatNumberToLocaleString(value, pattern, locale, groupSymbol, decimalSymbol, digitsInfo, isPercent = false) {
        /** @type {?} */
        let formattedText = '';
        /** @type {?} */
        let isZero = false;
        if (!isFinite(value)) {
            formattedText = getLocaleNumberSymbol(locale, NumberSymbol.Infinity);
        }
        else {
            /** @type {?} */
            let parsedNumber = parseNumber(value);
            if (isPercent) {
                parsedNumber = toPercent(parsedNumber);
            }
            /** @type {?} */
            let minInt = pattern.minInt;
            /** @type {?} */
            let minFraction = pattern.minFrac;
            /** @type {?} */
            let maxFraction = pattern.maxFrac;
            if (digitsInfo) {
                /** @type {?} */
                const parts = digitsInfo.match(NUMBER_FORMAT_REGEXP);
                if (parts === null) {
                    throw new Error(`${digitsInfo} is not a valid digit info`);
                }
                /** @type {?} */
                const minIntPart = parts[1];
                /** @type {?} */
                const minFractionPart = parts[3];
                /** @type {?} */
                const maxFractionPart = parts[5];
                if (minIntPart != null) {
                    minInt = parseIntAutoRadix(minIntPart);
                }
                if (minFractionPart != null) {
                    minFraction = parseIntAutoRadix(minFractionPart);
                }
                if (maxFractionPart != null) {
                    maxFraction = parseIntAutoRadix(maxFractionPart);
                }
                else if (minFractionPart != null && minFraction > maxFraction) {
                    maxFraction = minFraction;
                }
            }
            roundNumber(parsedNumber, minFraction, maxFraction);
            /** @type {?} */
            let digits = parsedNumber.digits;
            /** @type {?} */
            let integerLen = parsedNumber.integerLen;
            /** @type {?} */
            const exponent = parsedNumber.exponent;
            /** @type {?} */
            let decimals = [];
            isZero = digits.every((/**
             * @param {?} d
             * @return {?}
             */
            d => !d));
            // pad zeros for small numbers
            for (; integerLen < minInt; integerLen++) {
                digits.unshift(0);
            }
            // pad zeros for small numbers
            for (; integerLen < 0; integerLen++) {
                digits.unshift(0);
            }
            // extract decimals digits
            if (integerLen > 0) {
                decimals = digits.splice(integerLen, digits.length);
            }
            else {
                decimals = digits;
                digits = [0];
            }
            // format the integer digits with grouping separators
            /** @type {?} */
            const groups = [];
            if (digits.length >= pattern.lgSize) {
                groups.unshift(digits.splice(-pattern.lgSize, digits.length).join(''));
            }
            while (digits.length > pattern.gSize) {
                groups.unshift(digits.splice(-pattern.gSize, digits.length).join(''));
            }
            if (digits.length) {
                groups.unshift(digits.join(''));
            }
            formattedText = groups.join(getLocaleNumberSymbol(locale, groupSymbol));
            // append the decimal digits
            if (decimals.length) {
                formattedText += getLocaleNumberSymbol(locale, decimalSymbol) + decimals.join('');
            }
            if (exponent) {
                formattedText += getLocaleNumberSymbol(locale, NumberSymbol.Exponential) + '+' + exponent;
            }
        }
        if (value < 0 && !isZero) {
            formattedText = pattern.negPre + formattedText + pattern.negSuf;
        }
        else {
            formattedText = pattern.posPre + formattedText + pattern.posSuf;
        }
        return formattedText;
    }
    /**
     * \@ngModule CommonModule
     * \@description
     *
     * Formats a number as currency using locale rules.
     *
     * @see `formatNumber()` / `DecimalPipe` / [Internationalization (i18n) Guide](https://angular.io/guide/i18n)
     *
     * \@publicApi
     * @param {?} value The number to format.
     * @param {?} locale A locale code for the locale format rules to use.
     * @param {?} currency A string containing the currency symbol or its name,
     * such as "$" or "Canadian Dollar". Used in output string, but does not affect the operation
     * of the function.
     * @param {?=} currencyCode The [ISO 4217](https://en.wikipedia.org/wiki/ISO_4217)
     * currency code, such as `USD` for the US dollar and `EUR` for the euro.
     * Used to determine the number of digits in the decimal part.
     * @param {?=} digitsInfo
     * @return {?} The formatted currency value.
     *
     */
    function formatCurrency(value, locale, currency, currencyCode, digitsInfo) {
        /** @type {?} */
        const format = getLocaleNumberFormat(locale, NumberFormatStyle.Currency);
        /** @type {?} */
        const pattern = parseNumberFormat(format, getLocaleNumberSymbol(locale, NumberSymbol.MinusSign));
        pattern.minFrac = getNumberOfCurrencyDigits((/** @type {?} */ (currencyCode)));
        pattern.maxFrac = pattern.minFrac;
        /** @type {?} */
        const res = formatNumberToLocaleString(value, pattern, locale, NumberSymbol.CurrencyGroup, NumberSymbol.CurrencyDecimal, digitsInfo);
        return res
            .replace(CURRENCY_CHAR, currency)
            // if we have 2 time the currency character, the second one is ignored
            .replace(CURRENCY_CHAR, '')
            // If there is a spacing between currency character and the value and
            // the currency character is supressed by passing an empty string, the
            // spacing character would remain as part of the string. Then we
            // should remove it.
            .trim();
    }
    /**
     * \@ngModule CommonModule
     * \@description
     *
     * Formats a number as a percentage according to locale rules.
     *
     * @see `formatNumber()` / `DecimalPipe` / [Internationalization (i18n) Guide](https://angular.io/guide/i18n)
     * \@publicApi
     *
     * @param {?} value The number to format.
     * @param {?} locale A locale code for the locale format rules to use.
     * @param {?=} digitsInfo
     * @return {?} The formatted percentage value.
     *
     */
    function formatPercent(value, locale, digitsInfo) {
        /** @type {?} */
        const format = getLocaleNumberFormat(locale, NumberFormatStyle.Percent);
        /** @type {?} */
        const pattern = parseNumberFormat(format, getLocaleNumberSymbol(locale, NumberSymbol.MinusSign));
        /** @type {?} */
        const res = formatNumberToLocaleString(value, pattern, locale, NumberSymbol.Group, NumberSymbol.Decimal, digitsInfo, true);
        return res.replace(new RegExp(PERCENT_CHAR, 'g'), getLocaleNumberSymbol(locale, NumberSymbol.PercentSign));
    }
    /**
     * \@ngModule CommonModule
     * \@description
     *
     * Formats a number as text, with group sizing, separator, and other
     * parameters based on the locale.
     *
     * @see [Internationalization (i18n) Guide](https://angular.io/guide/i18n)
     *
     * \@publicApi
     * @param {?} value The number to format.
     * @param {?} locale A locale code for the locale format rules to use.
     * @param {?=} digitsInfo
     * @return {?} The formatted text string.
     */
    function formatNumber(value, locale, digitsInfo) {
        /** @type {?} */
        const format = getLocaleNumberFormat(locale, NumberFormatStyle.Decimal);
        /** @type {?} */
        const pattern = parseNumberFormat(format, getLocaleNumberSymbol(locale, NumberSymbol.MinusSign));
        return formatNumberToLocaleString(value, pattern, locale, NumberSymbol.Group, NumberSymbol.Decimal, digitsInfo);
    }
    /**
     * @param {?} format
     * @param {?=} minusSign
     * @return {?}
     */
    function parseNumberFormat(format, minusSign = '-') {
        /** @type {?} */
        const p = {
            minInt: 1,
            minFrac: 0,
            maxFrac: 0,
            posPre: '',
            posSuf: '',
            negPre: '',
            negSuf: '',
            gSize: 0,
            lgSize: 0
        };
        /** @type {?} */
        const patternParts = format.split(PATTERN_SEP);
        /** @type {?} */
        const positive = patternParts[0];
        /** @type {?} */
        const negative = patternParts[1];
        /** @type {?} */
        const positiveParts = positive.indexOf(DECIMAL_SEP) !== -1 ?
            positive.split(DECIMAL_SEP) :
            [
                positive.substring(0, positive.lastIndexOf(ZERO_CHAR) + 1),
                positive.substring(positive.lastIndexOf(ZERO_CHAR) + 1)
            ];
        /** @type {?} */
        const integer = positiveParts[0];
        /** @type {?} */
        const fraction = positiveParts[1] || '';
        p.posPre = integer.substr(0, integer.indexOf(DIGIT_CHAR));
        for (let i = 0; i < fraction.length; i++) {
            /** @type {?} */
            const ch = fraction.charAt(i);
            if (ch === ZERO_CHAR) {
                p.minFrac = p.maxFrac = i + 1;
            }
            else if (ch === DIGIT_CHAR) {
                p.maxFrac = i + 1;
            }
            else {
                p.posSuf += ch;
            }
        }
        /** @type {?} */
        const groups = integer.split(GROUP_SEP);
        p.gSize = groups[1] ? groups[1].length : 0;
        p.lgSize = (groups[2] || groups[1]) ? (groups[2] || groups[1]).length : 0;
        if (negative) {
            /** @type {?} */
            const trunkLen = positive.length - p.posPre.length - p.posSuf.length;
            /** @type {?} */
            const pos = negative.indexOf(DIGIT_CHAR);
            p.negPre = negative.substr(0, pos).replace(/'/g, '');
            p.negSuf = negative.substr(pos + trunkLen).replace(/'/g, '');
        }
        else {
            p.negPre = minusSign + p.posPre;
            p.negSuf = p.posSuf;
        }
        return p;
    }
    // Transforms a parsed number into a percentage by multiplying it by 100
    /**
     * @param {?} parsedNumber
     * @return {?}
     */
    function toPercent(parsedNumber) {
        // if the number is 0, don't do anything
        if (parsedNumber.digits[0] === 0) {
            return parsedNumber;
        }
        // Getting the current number of decimals
        /** @type {?} */
        const fractionLen = parsedNumber.digits.length - parsedNumber.integerLen;
        if (parsedNumber.exponent) {
            parsedNumber.exponent += 2;
        }
        else {
            if (fractionLen === 0) {
                parsedNumber.digits.push(0, 0);
            }
            else if (fractionLen === 1) {
                parsedNumber.digits.push(0);
            }
            parsedNumber.integerLen += 2;
        }
        return parsedNumber;
    }
    /**
     * Parses a number.
     * Significant bits of this parse algorithm came from https://github.com/MikeMcl/big.js/
     * @param {?} num
     * @return {?}
     */
    function parseNumber(num) {
        /** @type {?} */
        let numStr = Math.abs(num) + '';
        /** @type {?} */
        let exponent = 0;
        /** @type {?} */
        let digits;
        /** @type {?} */
        let integerLen;
        /** @type {?} */
        let i;
        /** @type {?} */
        let j;
        /** @type {?} */
        let zeros;
        // Decimal point?
        if ((integerLen = numStr.indexOf(DECIMAL_SEP)) > -1) {
            numStr = numStr.replace(DECIMAL_SEP, '');
        }
        // Exponential form?
        if ((i = numStr.search(/e/i)) > 0) {
            // Work out the exponent.
            if (integerLen < 0)
                integerLen = i;
            integerLen += +numStr.slice(i + 1);
            numStr = numStr.substring(0, i);
        }
        else if (integerLen < 0) {
            // There was no decimal point or exponent so it is an integer.
            integerLen = numStr.length;
        }
        // Count the number of leading zeros.
        for (i = 0; numStr.charAt(i) === ZERO_CHAR; i++) { /* empty */
        }
        if (i === (zeros = numStr.length)) {
            // The digits are all zero.
            digits = [0];
            integerLen = 1;
        }
        else {
            // Count the number of trailing zeros
            zeros--;
            while (numStr.charAt(zeros) === ZERO_CHAR)
                zeros--;
            // Trailing zeros are insignificant so ignore them
            integerLen -= i;
            digits = [];
            // Convert string to array of digits without leading/trailing zeros.
            for (j = 0; i <= zeros; i++, j++) {
                digits[j] = Number(numStr.charAt(i));
            }
        }
        // If the number overflows the maximum allowed digits then use an exponent.
        if (integerLen > MAX_DIGITS) {
            digits = digits.splice(0, MAX_DIGITS - 1);
            exponent = integerLen - 1;
            integerLen = 1;
        }
        return { digits, exponent, integerLen };
    }
    /**
     * Round the parsed number to the specified number of decimal places
     * This function changes the parsedNumber in-place
     * @param {?} parsedNumber
     * @param {?} minFrac
     * @param {?} maxFrac
     * @return {?}
     */
    function roundNumber(parsedNumber, minFrac, maxFrac) {
        if (minFrac > maxFrac) {
            throw new Error(`The minimum number of digits after fraction (${minFrac}) is higher than the maximum (${maxFrac}).`);
        }
        /** @type {?} */
        let digits = parsedNumber.digits;
        /** @type {?} */
        let fractionLen = digits.length - parsedNumber.integerLen;
        /** @type {?} */
        const fractionSize = Math.min(Math.max(minFrac, fractionLen), maxFrac);
        // The index of the digit to where rounding is to occur
        /** @type {?} */
        let roundAt = fractionSize + parsedNumber.integerLen;
        /** @type {?} */
        let digit = digits[roundAt];
        if (roundAt > 0) {
            // Drop fractional digits beyond `roundAt`
            digits.splice(Math.max(parsedNumber.integerLen, roundAt));
            // Set non-fractional digits beyond `roundAt` to 0
            for (let j = roundAt; j < digits.length; j++) {
                digits[j] = 0;
            }
        }
        else {
            // We rounded to zero so reset the parsedNumber
            fractionLen = Math.max(0, fractionLen);
            parsedNumber.integerLen = 1;
            digits.length = Math.max(1, roundAt = fractionSize + 1);
            digits[0] = 0;
            for (let i = 1; i < roundAt; i++)
                digits[i] = 0;
        }
        if (digit >= 5) {
            if (roundAt - 1 < 0) {
                for (let k = 0; k > roundAt; k--) {
                    digits.unshift(0);
                    parsedNumber.integerLen++;
                }
                digits.unshift(1);
                parsedNumber.integerLen++;
            }
            else {
                digits[roundAt - 1]++;
            }
        }
        // Pad out with zeros to get the required fraction length
        for (; fractionLen < Math.max(0, fractionSize); fractionLen++)
            digits.push(0);
        /** @type {?} */
        let dropTrailingZeros = fractionSize !== 0;
        // Minimal length = nb of decimals required + current nb of integers
        // Any number besides that is optional and can be removed if it's a trailing 0
        /** @type {?} */
        const minLen = minFrac + parsedNumber.integerLen;
        // Do any carrying, e.g. a digit was rounded up to 10
        /** @type {?} */
        const carry = digits.reduceRight((/**
         * @param {?} carry
         * @param {?} d
         * @param {?} i
         * @param {?} digits
         * @return {?}
         */
        function (carry, d, i, digits) {
            d = d + carry;
            digits[i] = d < 10 ? d : d - 10; // d % 10
            if (dropTrailingZeros) {
                // Do not keep meaningless fractional trailing zeros (e.g. 15.52000 --> 15.52)
                if (digits[i] === 0 && i >= minLen) {
                    digits.pop();
                }
                else {
                    dropTrailingZeros = false;
                }
            }
            return d >= 10 ? 1 : 0; // Math.floor(d / 10);
        }), 0);
        if (carry) {
            digits.unshift(carry);
            parsedNumber.integerLen++;
        }
    }
    /**
     * @param {?} text
     * @return {?}
     */
    function parseIntAutoRadix(text) {
        /** @type {?} */
        const result = parseInt(text);
        if (isNaN(result)) {
            throw new Error('Invalid integer literal when parsing ' + text);
        }
        return result;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/src/i18n/localization.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * \@publicApi
     * @abstract
     */
    class NgLocalization {
    }
    /**
     * Returns the plural category for a given value.
     * - "=value" when the case exists,
     * - the plural category otherwise
     * @param {?} value
     * @param {?} cases
     * @param {?} ngLocalization
     * @param {?=} locale
     * @return {?}
     */
    function getPluralCategory(value, cases, ngLocalization, locale) {
        /** @type {?} */
        let key = `=${value}`;
        if (cases.indexOf(key) > -1) {
            return key;
        }
        key = ngLocalization.getPluralCategory(value, locale);
        if (cases.indexOf(key) > -1) {
            return key;
        }
        if (cases.indexOf('other') > -1) {
            return 'other';
        }
        throw new Error(`No plural message found for value "${value}"`);
    }
    /**
     * Returns the plural case based on the locale
     *
     * \@publicApi
     */
    class NgLocaleLocalization extends NgLocalization {
        /**
         * @param {?} locale
         */
        constructor(locale) {
            super();
            this.locale = locale;
        }
        /**
         * @param {?} value
         * @param {?=} locale
         * @return {?}
         */
        getPluralCategory(value, locale) {
            /** @type {?} */
            const plural = getLocalePluralCase$1(locale || this.locale)(value);
            switch (plural) {
                case Plural.Zero:
                    return 'zero';
                case Plural.One:
                    return 'one';
                case Plural.Two:
                    return 'two';
                case Plural.Few:
                    return 'few';
                case Plural.Many:
                    return 'many';
                default:
                    return 'other';
            }
        }
    }
    NgLocaleLocalization.ɵfac = function NgLocaleLocalization_Factory(t) { return new (t || NgLocaleLocalization)(ɵɵinject(LOCALE_ID$1)); };
    NgLocaleLocalization.ɵprov = ɵɵdefineInjectable({ token: NgLocaleLocalization, factory: NgLocaleLocalization.ɵfac });
    /** @nocollapse */
    NgLocaleLocalization.ctorParameters = () => [
        { type: String, decorators: [{ type: Inject, args: [LOCALE_ID$1,] }] }
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/src/cookie.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * @param {?} cookieStr
     * @param {?} name
     * @return {?}
     */
    function parseCookieValue(cookieStr, name) {
        name = encodeURIComponent(name);
        for (const cookie of cookieStr.split(';')) {
            /** @type {?} */
            const eqIndex = cookie.indexOf('=');
            const [cookieName, cookieValue] = eqIndex == -1 ? [cookie, ''] : [cookie.slice(0, eqIndex), cookie.slice(eqIndex + 1)];
            if (cookieName.trim() === name) {
                return decodeURIComponent(cookieValue);
            }
        }
        return null;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/src/directives/ng_class.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * \@ngModule CommonModule
     *
     * \@usageNotes
     * ```
     *     <some-element [ngClass]="'first second'">...</some-element>
     *
     *     <some-element [ngClass]="['first', 'second']">...</some-element>
     *
     *     <some-element [ngClass]="{'first': true, 'second': true, 'third': false}">...</some-element>
     *
     *     <some-element [ngClass]="stringExp|arrayExp|objExp">...</some-element>
     *
     *     <some-element [ngClass]="{'class1 class2 class3' : true}">...</some-element>
     * ```
     *
     * \@description
     *
     * Adds and removes CSS classes on an HTML element.
     *
     * The CSS classes are updated as follows, depending on the type of the expression evaluation:
     * - `string` - the CSS classes listed in the string (space delimited) are added,
     * - `Array` - the CSS classes declared as Array elements are added,
     * - `Object` - keys are CSS classes that get added when the expression given in the value
     *              evaluates to a truthy value, otherwise they are removed.
     *
     * \@publicApi
     */
    class NgClass {
        /**
         * @param {?} _iterableDiffers
         * @param {?} _keyValueDiffers
         * @param {?} _ngEl
         * @param {?} _renderer
         */
        constructor(_iterableDiffers, _keyValueDiffers, _ngEl, _renderer) {
            this._iterableDiffers = _iterableDiffers;
            this._keyValueDiffers = _keyValueDiffers;
            this._ngEl = _ngEl;
            this._renderer = _renderer;
            this._iterableDiffer = null;
            this._keyValueDiffer = null;
            this._initialClasses = [];
            this._rawClass = null;
        }
        /**
         * @param {?} value
         * @return {?}
         */
        set klass(value) {
            this._removeClasses(this._initialClasses);
            this._initialClasses = typeof value === 'string' ? value.split(/\s+/) : [];
            this._applyClasses(this._initialClasses);
            this._applyClasses(this._rawClass);
        }
        /**
         * @param {?} value
         * @return {?}
         */
        set ngClass(value) {
            this._removeClasses(this._rawClass);
            this._applyClasses(this._initialClasses);
            this._iterableDiffer = null;
            this._keyValueDiffer = null;
            this._rawClass = typeof value === 'string' ? value.split(/\s+/) : value;
            if (this._rawClass) {
                if (isListLikeIterable$1(this._rawClass)) {
                    this._iterableDiffer = this._iterableDiffers.find(this._rawClass).create();
                }
                else {
                    this._keyValueDiffer = this._keyValueDiffers.find(this._rawClass).create();
                }
            }
        }
        /**
         * @return {?}
         */
        ngDoCheck() {
            if (this._iterableDiffer) {
                /** @type {?} */
                const iterableChanges = this._iterableDiffer.diff((/** @type {?} */ (this._rawClass)));
                if (iterableChanges) {
                    this._applyIterableChanges(iterableChanges);
                }
            }
            else if (this._keyValueDiffer) {
                /** @type {?} */
                const keyValueChanges = this._keyValueDiffer.diff((/** @type {?} */ (this._rawClass)));
                if (keyValueChanges) {
                    this._applyKeyValueChanges(keyValueChanges);
                }
            }
        }
        /**
         * @private
         * @param {?} changes
         * @return {?}
         */
        _applyKeyValueChanges(changes) {
            changes.forEachAddedItem((/**
             * @param {?} record
             * @return {?}
             */
            (record) => this._toggleClass(record.key, record.currentValue)));
            changes.forEachChangedItem((/**
             * @param {?} record
             * @return {?}
             */
            (record) => this._toggleClass(record.key, record.currentValue)));
            changes.forEachRemovedItem((/**
             * @param {?} record
             * @return {?}
             */
            (record) => {
                if (record.previousValue) {
                    this._toggleClass(record.key, false);
                }
            }));
        }
        /**
         * @private
         * @param {?} changes
         * @return {?}
         */
        _applyIterableChanges(changes) {
            changes.forEachAddedItem((/**
             * @param {?} record
             * @return {?}
             */
            (record) => {
                if (typeof record.item === 'string') {
                    this._toggleClass(record.item, true);
                }
                else {
                    throw new Error(`NgClass can only toggle CSS classes expressed as strings, got ${stringify(record.item)}`);
                }
            }));
            changes.forEachRemovedItem((/**
             * @param {?} record
             * @return {?}
             */
            (record) => this._toggleClass(record.item, false)));
        }
        /**
         * Applies a collection of CSS classes to the DOM element.
         *
         * For argument of type Set and Array CSS class names contained in those collections are always
         * added.
         * For argument of type Map CSS class name in the map's key is toggled based on the value (added
         * for truthy and removed for falsy).
         * @private
         * @param {?} rawClassVal
         * @return {?}
         */
        _applyClasses(rawClassVal) {
            if (rawClassVal) {
                if (Array.isArray(rawClassVal) || rawClassVal instanceof Set) {
                    ((/** @type {?} */ (rawClassVal))).forEach((/**
                     * @param {?} klass
                     * @return {?}
                     */
                    (klass) => this._toggleClass(klass, true)));
                }
                else {
                    Object.keys(rawClassVal).forEach((/**
                     * @param {?} klass
                     * @return {?}
                     */
                    klass => this._toggleClass(klass, !!rawClassVal[klass])));
                }
            }
        }
        /**
         * Removes a collection of CSS classes from the DOM element. This is mostly useful for cleanup
         * purposes.
         * @private
         * @param {?} rawClassVal
         * @return {?}
         */
        _removeClasses(rawClassVal) {
            if (rawClassVal) {
                if (Array.isArray(rawClassVal) || rawClassVal instanceof Set) {
                    ((/** @type {?} */ (rawClassVal))).forEach((/**
                     * @param {?} klass
                     * @return {?}
                     */
                    (klass) => this._toggleClass(klass, false)));
                }
                else {
                    Object.keys(rawClassVal).forEach((/**
                     * @param {?} klass
                     * @return {?}
                     */
                    klass => this._toggleClass(klass, false)));
                }
            }
        }
        /**
         * @private
         * @param {?} klass
         * @param {?} enabled
         * @return {?}
         */
        _toggleClass(klass, enabled) {
            klass = klass.trim();
            if (klass) {
                klass.split(/\s+/g).forEach((/**
                 * @param {?} klass
                 * @return {?}
                 */
                klass => {
                    if (enabled) {
                        this._renderer.addClass(this._ngEl.nativeElement, klass);
                    }
                    else {
                        this._renderer.removeClass(this._ngEl.nativeElement, klass);
                    }
                }));
            }
        }
    }
    NgClass.ɵfac = function NgClass_Factory(t) { return new (t || NgClass)(ɵɵdirectiveInject(IterableDiffers), ɵɵdirectiveInject(KeyValueDiffers), ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(Renderer2)); };
    NgClass.ɵdir = ɵɵdefineDirective({ type: NgClass, selectors: [["", "ngClass", ""]], inputs: { klass: ["class", "klass"], ngClass: "ngClass" } });
    /** @nocollapse */
    NgClass.ctorParameters = () => [
        { type: IterableDiffers },
        { type: KeyValueDiffers },
        { type: ElementRef },
        { type: Renderer2 }
    ];
    NgClass.propDecorators = {
        klass: [{ type: Input, args: ['class',] }],
        ngClass: [{ type: Input, args: ['ngClass',] }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/src/directives/ng_component_outlet.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Instantiates a single {\@link Component} type and inserts its Host View into current View.
     * `NgComponentOutlet` provides a declarative approach for dynamic component creation.
     *
     * `NgComponentOutlet` requires a component type, if a falsy value is set the view will clear and
     * any existing component will get destroyed.
     *
     * \@usageNotes
     *
     * ### Fine tune control
     *
     * You can control the component creation process by using the following optional attributes:
     *
     * * `ngComponentOutletInjector`: Optional custom {\@link Injector} that will be used as parent for
     * the Component. Defaults to the injector of the current view container.
     *
     * * `ngComponentOutletContent`: Optional list of projectable nodes to insert into the content
     * section of the component, if exists.
     *
     * * `ngComponentOutletNgModuleFactory`: Optional module factory to allow dynamically loading other
     * module, then load a component from that module.
     *
     * ### Syntax
     *
     * Simple
     * ```
     * <ng-container *ngComponentOutlet="componentTypeExpression"></ng-container>
     * ```
     *
     * Customized injector/content
     * ```
     * <ng-container *ngComponentOutlet="componentTypeExpression;
     *                                   injector: injectorExpression;
     *                                   content: contentNodesExpression;">
     * </ng-container>
     * ```
     *
     * Customized ngModuleFactory
     * ```
     * <ng-container *ngComponentOutlet="componentTypeExpression;
     *                                   ngModuleFactory: moduleFactory;">
     * </ng-container>
     * ```
     *
     * ### A simple example
     *
     * {\@example common/ngComponentOutlet/ts/module.ts region='SimpleExample'}
     *
     * A more complete example with additional options:
     *
     * {\@example common/ngComponentOutlet/ts/module.ts region='CompleteExample'}
     *
     * \@publicApi
     * \@ngModule CommonModule
     */
    class NgComponentOutlet {
        /**
         * @param {?} _viewContainerRef
         */
        constructor(_viewContainerRef) {
            this._viewContainerRef = _viewContainerRef;
            this._componentRef = null;
            this._moduleRef = null;
        }
        /**
         * @param {?} changes
         * @return {?}
         */
        ngOnChanges(changes) {
            this._viewContainerRef.clear();
            this._componentRef = null;
            if (this.ngComponentOutlet) {
                /** @type {?} */
                const elInjector = this.ngComponentOutletInjector || this._viewContainerRef.parentInjector;
                if (changes['ngComponentOutletNgModuleFactory']) {
                    if (this._moduleRef)
                        this._moduleRef.destroy();
                    if (this.ngComponentOutletNgModuleFactory) {
                        /** @type {?} */
                        const parentModule = elInjector.get(NgModuleRef);
                        this._moduleRef = this.ngComponentOutletNgModuleFactory.create(parentModule.injector);
                    }
                    else {
                        this._moduleRef = null;
                    }
                }
                /** @type {?} */
                const componentFactoryResolver = this._moduleRef ? this._moduleRef.componentFactoryResolver :
                    elInjector.get(ComponentFactoryResolver);
                /** @type {?} */
                const componentFactory = componentFactoryResolver.resolveComponentFactory(this.ngComponentOutlet);
                this._componentRef = this._viewContainerRef.createComponent(componentFactory, this._viewContainerRef.length, elInjector, this.ngComponentOutletContent);
            }
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            if (this._moduleRef)
                this._moduleRef.destroy();
        }
    }
    NgComponentOutlet.ɵfac = function NgComponentOutlet_Factory(t) { return new (t || NgComponentOutlet)(ɵɵdirectiveInject(ViewContainerRef)); };
    NgComponentOutlet.ɵdir = ɵɵdefineDirective({ type: NgComponentOutlet, selectors: [["", "ngComponentOutlet", ""]], inputs: { ngComponentOutlet: "ngComponentOutlet", ngComponentOutletInjector: "ngComponentOutletInjector", ngComponentOutletContent: "ngComponentOutletContent", ngComponentOutletNgModuleFactory: "ngComponentOutletNgModuleFactory" }, features: [ɵɵNgOnChangesFeature] });
    /** @nocollapse */
    NgComponentOutlet.ctorParameters = () => [
        { type: ViewContainerRef }
    ];
    NgComponentOutlet.propDecorators = {
        ngComponentOutlet: [{ type: Input }],
        ngComponentOutletInjector: [{ type: Input }],
        ngComponentOutletContent: [{ type: Input }],
        ngComponentOutletNgModuleFactory: [{ type: Input }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/src/directives/ng_for_of.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * \@publicApi
     * @template T, U
     */
    class NgForOfContext {
        /**
         * @param {?} $implicit
         * @param {?} ngForOf
         * @param {?} index
         * @param {?} count
         */
        constructor($implicit, ngForOf, index, count) {
            this.$implicit = $implicit;
            this.ngForOf = ngForOf;
            this.index = index;
            this.count = count;
        }
        /**
         * @return {?}
         */
        get first() {
            return this.index === 0;
        }
        /**
         * @return {?}
         */
        get last() {
            return this.index === this.count - 1;
        }
        /**
         * @return {?}
         */
        get even() {
            return this.index % 2 === 0;
        }
        /**
         * @return {?}
         */
        get odd() {
            return !this.even;
        }
    }
    /**
     * A [structural directive](guide/structural-directives) that renders
     * a template for each item in a collection.
     * The directive is placed on an element, which becomes the parent
     * of the cloned templates.
     *
     * The `ngForOf` directive is generally used in the
     * [shorthand form](guide/structural-directives#the-asterisk--prefix) `*ngFor`.
     * In this form, the template to be rendered for each iteration is the content
     * of an anchor element containing the directive.
     *
     * The following example shows the shorthand syntax with some options,
     * contained in an `<li>` element.
     *
     * ```
     * <li *ngFor="let item of items; index as i; trackBy: trackByFn">...</li>
     * ```
     *
     * The shorthand form expands into a long form that uses the `ngForOf` selector
     * on an `<ng-template>` element.
     * The content of the `<ng-template>` element is the `<li>` element that held the
     * short-form directive.
     *
     * Here is the expanded version of the short-form example.
     *
     * ```
     * <ng-template ngFor let-item [ngForOf]="items" let-i="index" [ngForTrackBy]="trackByFn">
     *   <li>...</li>
     * </ng-template>
     * ```
     *
     * Angular automatically expands the shorthand syntax as it compiles the template.
     * The context for each embedded view is logically merged to the current component
     * context according to its lexical position.
     *
     * When using the shorthand syntax, Angular allows only [one structural directive
     * on an element](guide/structural-directives#one-structural-directive-per-host-element).
     * If you want to iterate conditionally, for example,
     * put the `*ngIf` on a container element that wraps the `*ngFor` element.
     * For futher discussion, see
     * [Structural Directives](guide/structural-directives#one-per-element).
     *
     * \@usageNotes
     *
     * ### Local variables
     *
     * `NgForOf` provides exported values that can be aliased to local variables.
     * For example:
     *
     *  ```
     * <li *ngFor="let user of users; index as i; first as isFirst">
     *    {{i}}/{{users.length}}. {{user}} <span *ngIf="isFirst">default</span>
     * </li>
     * ```
     *
     * The following exported values can be aliased to local variables:
     *
     * - `$implicit: T`: The value of the individual items in the iterable (`ngForOf`).
     * - `ngForOf: NgIterable<T>`: The value of the iterable expression. Useful when the expression is
     * more complex then a property access, for example when using the async pipe (`userStreams |
     * async`).
     * - `index: number`: The index of the current item in the iterable.
     * - `count: number`: The length of the iterable.
     * - `first: boolean`: True when the item is the first item in the iterable.
     * - `last: boolean`: True when the item is the last item in the iterable.
     * - `even: boolean`: True when the item has an even index in the iterable.
     * - `odd: boolean`: True when the item has an odd index in the iterable.
     *
     * ### Change propagation
     *
     * When the contents of the iterator changes, `NgForOf` makes the corresponding changes to the DOM:
     *
     * * When an item is added, a new instance of the template is added to the DOM.
     * * When an item is removed, its template instance is removed from the DOM.
     * * When items are reordered, their respective templates are reordered in the DOM.
     *
     * Angular uses object identity to track insertions and deletions within the iterator and reproduce
     * those changes in the DOM. This has important implications for animations and any stateful
     * controls that are present, such as `<input>` elements that accept user input. Inserted rows can
     * be animated in, deleted rows can be animated out, and unchanged rows retain any unsaved state
     * such as user input.
     * For more on animations, see [Transitions and Triggers](guide/transition-and-triggers).
     *
     * The identities of elements in the iterator can change while the data does not.
     * This can happen, for example, if the iterator is produced from an RPC to the server, and that
     * RPC is re-run. Even if the data hasn't changed, the second response produces objects with
     * different identities, and Angular must tear down the entire DOM and rebuild it (as if all old
     * elements were deleted and all new elements inserted).
     *
     * To avoid this expensive operation, you can customize the default tracking algorithm.
     * by supplying the `trackBy` option to `NgForOf`.
     * `trackBy` takes a function that has two arguments: `index` and `item`.
     * If `trackBy` is given, Angular tracks changes by the return value of the function.
     *
     * @see [Structural Directives](guide/structural-directives)
     * \@ngModule CommonModule
     * \@publicApi
     * @template T, U
     */
    class NgForOf {
        /**
         * @param {?} _viewContainer
         * @param {?} _template
         * @param {?} _differs
         */
        constructor(_viewContainer, _template, _differs) {
            this._viewContainer = _viewContainer;
            this._template = _template;
            this._differs = _differs;
            this._ngForOf = null;
            this._ngForOfDirty = true;
            this._differ = null;
        }
        /**
         * The value of the iterable expression, which can be used as a
         * [template input variable](guide/structural-directives#template-input-variable).
         * @param {?} ngForOf
         * @return {?}
         */
        set ngForOf(ngForOf) {
            this._ngForOf = ngForOf;
            this._ngForOfDirty = true;
        }
        /**
         * A function that defines how to track changes for items in the iterable.
         *
         * When items are added, moved, or removed in the iterable,
         * the directive must re-render the appropriate DOM nodes.
         * To minimize churn in the DOM, only nodes that have changed
         * are re-rendered.
         *
         * By default, the change detector assumes that
         * the object instance identifies the node in the iterable.
         * When this function is supplied, the directive uses
         * the result of calling this function to identify the item node,
         * rather than the identity of the object itself.
         *
         * The function receives two inputs,
         * the iteration index and the node object ID.
         * @param {?} fn
         * @return {?}
         */
        set ngForTrackBy(fn) {
            if (isDevMode() && fn != null && typeof fn !== 'function') {
                // TODO(vicb): use a log service once there is a public one available
                if ((/** @type {?} */ (console)) && (/** @type {?} */ (console.warn))) {
                    console.warn(`trackBy must be a function, but received ${JSON.stringify(fn)}. ` +
                        `See https://angular.io/api/common/NgForOf#change-propagation for more information.`);
                }
            }
            this._trackByFn = fn;
        }
        /**
         * @return {?}
         */
        get ngForTrackBy() {
            return this._trackByFn;
        }
        /**
         * A reference to the template that is stamped out for each item in the iterable.
         * @see [template reference variable](guide/template-syntax#template-reference-variables--var-)
         * @param {?} value
         * @return {?}
         */
        set ngForTemplate(value) {
            // TODO(TS2.1): make TemplateRef<Partial<NgForRowOf<T>>> once we move to TS v2.1
            // The current type is too restrictive; a template that just uses index, for example,
            // should be acceptable.
            if (value) {
                this._template = value;
            }
        }
        /**
         * Applies the changes when needed.
         * @return {?}
         */
        ngDoCheck() {
            if (this._ngForOfDirty) {
                this._ngForOfDirty = false;
                // React on ngForOf changes only once all inputs have been initialized
                /** @type {?} */
                const value = this._ngForOf;
                if (!this._differ && value) {
                    try {
                        this._differ = this._differs.find(value).create(this.ngForTrackBy);
                    }
                    catch (_a) {
                        throw new Error(`Cannot find a differ supporting object '${value}' of type '${getTypeName$1(value)}'. NgFor only supports binding to Iterables such as Arrays.`);
                    }
                }
            }
            if (this._differ) {
                /** @type {?} */
                const changes = this._differ.diff(this._ngForOf);
                if (changes)
                    this._applyChanges(changes);
            }
        }
        /**
         * @private
         * @param {?} changes
         * @return {?}
         */
        _applyChanges(changes) {
            /** @type {?} */
            const insertTuples = [];
            changes.forEachOperation((/**
             * @param {?} item
             * @param {?} adjustedPreviousIndex
             * @param {?} currentIndex
             * @return {?}
             */
            (item, adjustedPreviousIndex, currentIndex) => {
                if (item.previousIndex == null) {
                    // NgForOf is never "null" or "undefined" here because the differ detected
                    // that a new item needs to be inserted from the iterable. This implies that
                    // there is an iterable value for "_ngForOf".
                    /** @type {?} */
                    const view = this._viewContainer.createEmbeddedView(this._template, new NgForOfContext((/** @type {?} */ (null)), (/** @type {?} */ (this._ngForOf)), -1, -1), currentIndex === null ? undefined : currentIndex);
                    /** @type {?} */
                    const tuple = new RecordViewTuple(item, view);
                    insertTuples.push(tuple);
                }
                else if (currentIndex == null) {
                    this._viewContainer.remove(adjustedPreviousIndex === null ? undefined : adjustedPreviousIndex);
                }
                else if (adjustedPreviousIndex !== null) {
                    /** @type {?} */
                    const view = (/** @type {?} */ (this._viewContainer.get(adjustedPreviousIndex)));
                    this._viewContainer.move(view, currentIndex);
                    /** @type {?} */
                    const tuple = new RecordViewTuple(item, (/** @type {?} */ (view)));
                    insertTuples.push(tuple);
                }
            }));
            for (let i = 0; i < insertTuples.length; i++) {
                this._perViewChange(insertTuples[i].view, insertTuples[i].record);
            }
            for (let i = 0, ilen = this._viewContainer.length; i < ilen; i++) {
                /** @type {?} */
                const viewRef = (/** @type {?} */ (this._viewContainer.get(i)));
                viewRef.context.index = i;
                viewRef.context.count = ilen;
                viewRef.context.ngForOf = (/** @type {?} */ (this._ngForOf));
            }
            changes.forEachIdentityChange((/**
             * @param {?} record
             * @return {?}
             */
            (record) => {
                /** @type {?} */
                const viewRef = (/** @type {?} */ (this._viewContainer.get(record.currentIndex)));
                viewRef.context.$implicit = record.item;
            }));
        }
        /**
         * @private
         * @param {?} view
         * @param {?} record
         * @return {?}
         */
        _perViewChange(view, record) {
            view.context.$implicit = record.item;
        }
        /**
         * Asserts the correct type of the context for the template that `NgForOf` will render.
         *
         * The presence of this method is a signal to the Ivy template type-check compiler that the
         * `NgForOf` structural directive renders its template with a specific context type.
         * @template T, U
         * @param {?} dir
         * @param {?} ctx
         * @return {?}
         */
        static ngTemplateContextGuard(dir, ctx) {
            return true;
        }
    }
    NgForOf.ɵfac = function NgForOf_Factory(t) { return new (t || NgForOf)(ɵɵdirectiveInject(ViewContainerRef), ɵɵdirectiveInject(TemplateRef), ɵɵdirectiveInject(IterableDiffers)); };
    NgForOf.ɵdir = ɵɵdefineDirective({ type: NgForOf, selectors: [["", "ngFor", "", "ngForOf", ""]], inputs: { ngForOf: "ngForOf", ngForTrackBy: "ngForTrackBy", ngForTemplate: "ngForTemplate" } });
    /** @nocollapse */
    NgForOf.ctorParameters = () => [
        { type: ViewContainerRef },
        { type: TemplateRef },
        { type: IterableDiffers }
    ];
    NgForOf.propDecorators = {
        ngForOf: [{ type: Input }],
        ngForTrackBy: [{ type: Input }],
        ngForTemplate: [{ type: Input }]
    };
    /**
     * @template T, U
     */
    class RecordViewTuple {
        /**
         * @param {?} record
         * @param {?} view
         */
        constructor(record, view) {
            this.record = record;
            this.view = view;
        }
    }
    /**
     * @param {?} type
     * @return {?}
     */
    function getTypeName$1(type) {
        return type['name'] || typeof type;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/src/directives/ng_if.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * A structural directive that conditionally includes a template based on the value of
     * an expression coerced to Boolean.
     * When the expression evaluates to true, Angular renders the template
     * provided in a `then` clause, and when  false or null,
     * Angular renders the template provided in an optional `else` clause. The default
     * template for the `else` clause is blank.
     *
     * A [shorthand form](guide/structural-directives#the-asterisk--prefix) of the directive,
     * `*ngIf="condition"`, is generally used, provided
     * as an attribute of the anchor element for the inserted template.
     * Angular expands this into a more explicit version, in which the anchor element
     * is contained in an `<ng-template>` element.
     *
     * Simple form with shorthand syntax:
     *
     * ```
     * <div *ngIf="condition">Content to render when condition is true.</div>
     * ```
     *
     * Simple form with expanded syntax:
     *
     * ```
     * <ng-template [ngIf]="condition"><div>Content to render when condition is
     * true.</div></ng-template>
     * ```
     *
     * Form with an "else" block:
     *
     * ```
     * <div *ngIf="condition; else elseBlock">Content to render when condition is true.</div>
     * <ng-template #elseBlock>Content to render when condition is false.</ng-template>
     * ```
     *
     * Shorthand form with "then" and "else" blocks:
     *
     * ```
     * <div *ngIf="condition; then thenBlock else elseBlock"></div>
     * <ng-template #thenBlock>Content to render when condition is true.</ng-template>
     * <ng-template #elseBlock>Content to render when condition is false.</ng-template>
     * ```
     *
     * Form with storing the value locally:
     *
     * ```
     * <div *ngIf="condition as value; else elseBlock">{{value}}</div>
     * <ng-template #elseBlock>Content to render when value is null.</ng-template>
     * ```
     *
     * \@usageNotes
     *
     * The `*ngIf` directive is most commonly used to conditionally show an inline template,
     * as seen in the following  example.
     * The default `else` template is blank.
     *
     * {\@example common/ngIf/ts/module.ts region='NgIfSimple'}
     *
     * ### Showing an alternative template using `else`
     *
     * To display a template when `expression` evaluates to false, use an `else` template
     * binding as shown in the following example.
     * The `else` binding points to an `<ng-template>`  element labeled `#elseBlock`.
     * The template can be defined anywhere in the component view, but is typically placed right after
     * `ngIf` for readability.
     *
     * {\@example common/ngIf/ts/module.ts region='NgIfElse'}
     *
     * ### Using an external `then` template
     *
     * In the previous example, the then-clause template is specified inline, as the content of the
     * tag that contains the `ngIf` directive. You can also specify a template that is defined
     * externally, by referencing a labeled `<ng-template>` element. When you do this, you can
     * change which template to use at runtime, as shown in the following example.
     *
     * {\@example common/ngIf/ts/module.ts region='NgIfThenElse'}
     *
     * ### Storing a conditional result in a variable
     *
     * You might want to show a set of properties from the same object. If you are waiting
     * for asynchronous data, the object can be undefined.
     * In this case, you can use `ngIf` and store the result of the condition in a local
     * variable as shown in the the following example.
     *
     * {\@example common/ngIf/ts/module.ts region='NgIfAs'}
     *
     * This code uses only one `AsyncPipe`, so only one subscription is created.
     * The conditional statement stores the result of `userStream|async` in the local variable `user`.
     * You can then bind the local `user` repeatedly.
     *
     * The conditional displays the data only if `userStream` returns a value,
     * so you don't need to use the
     * [safe-navigation-operator](guide/template-syntax#safe-navigation-operator) (`?.`)
     * to guard against null values when accessing properties.
     * You can display an alternative template while waiting for the data.
     *
     * ### Shorthand syntax
     *
     * The shorthand syntax `*ngIf` expands into two separate template specifications
     * for the "then" and "else" clauses. For example, consider the following shorthand statement,
     * that is meant to show a loading page while waiting for data to be loaded.
     *
     * ```
     * <div class="hero-list" *ngIf="heroes else loading">
     *  ...
     * </div>
     *
     * <ng-template #loading>
     *  <div>Loading...</div>
     * </ng-template>
     * ```
     *
     * You can see that the "else" clause references the `<ng-template>`
     * with the `#loading` label, and the template for the "then" clause
     * is provided as the content of the anchor element.
     *
     * However, when Angular expands the shorthand syntax, it creates
     * another `<ng-template>` tag, with `ngIf` and `ngIfElse` directives.
     * The anchor element containing the template for the "then" clause becomes
     * the content of this unlabeled `<ng-template>` tag.
     *
     * ```
     * <ng-template [ngIf]="heroes" [ngIfElse]="loading">
     *  <div class="hero-list">
     *   ...
     *  </div>
     * </ng-template>
     *
     * <ng-template #loading>
     *  <div>Loading...</div>
     * </ng-template>
     * ```
     *
     * The presence of the implicit template object has implications for the nesting of
     * structural directives. For more on this subject, see
     * [Structural Directives](https://angular.io/guide/structural-directives#one-per-element).
     *
     * \@ngModule CommonModule
     * \@publicApi
     * @template T
     */
    class NgIf {
        /**
         * @param {?} _viewContainer
         * @param {?} templateRef
         */
        constructor(_viewContainer, templateRef) {
            this._viewContainer = _viewContainer;
            this._context = new NgIfContext();
            this._thenTemplateRef = null;
            this._elseTemplateRef = null;
            this._thenViewRef = null;
            this._elseViewRef = null;
            this._thenTemplateRef = templateRef;
        }
        /**
         * The Boolean expression to evaluate as the condition for showing a template.
         * @param {?} condition
         * @return {?}
         */
        set ngIf(condition) {
            this._context.$implicit = this._context.ngIf = condition;
            this._updateView();
        }
        /**
         * A template to show if the condition expression evaluates to true.
         * @param {?} templateRef
         * @return {?}
         */
        set ngIfThen(templateRef) {
            assertTemplate('ngIfThen', templateRef);
            this._thenTemplateRef = templateRef;
            this._thenViewRef = null; // clear previous view if any.
            this._updateView();
        }
        /**
         * A template to show if the condition expression evaluates to false.
         * @param {?} templateRef
         * @return {?}
         */
        set ngIfElse(templateRef) {
            assertTemplate('ngIfElse', templateRef);
            this._elseTemplateRef = templateRef;
            this._elseViewRef = null; // clear previous view if any.
            this._updateView();
        }
        /**
         * @private
         * @return {?}
         */
        _updateView() {
            if (this._context.$implicit) {
                if (!this._thenViewRef) {
                    this._viewContainer.clear();
                    this._elseViewRef = null;
                    if (this._thenTemplateRef) {
                        this._thenViewRef =
                            this._viewContainer.createEmbeddedView(this._thenTemplateRef, this._context);
                    }
                }
            }
            else {
                if (!this._elseViewRef) {
                    this._viewContainer.clear();
                    this._thenViewRef = null;
                    if (this._elseTemplateRef) {
                        this._elseViewRef =
                            this._viewContainer.createEmbeddedView(this._elseTemplateRef, this._context);
                    }
                }
            }
        }
        /**
         * Asserts the correct type of the context for the template that `NgIf` will render.
         *
         * The presence of this method is a signal to the Ivy template type-check compiler that the
         * `NgIf` structural directive renders its template with a specific context type.
         * @template T
         * @param {?} dir
         * @param {?} ctx
         * @return {?}
         */
        static ngTemplateContextGuard(dir, ctx) {
            return true;
        }
    }
    NgIf.ɵfac = function NgIf_Factory(t) { return new (t || NgIf)(ɵɵdirectiveInject(ViewContainerRef), ɵɵdirectiveInject(TemplateRef)); };
    NgIf.ɵdir = ɵɵdefineDirective({ type: NgIf, selectors: [["", "ngIf", ""]], inputs: { ngIf: "ngIf", ngIfThen: "ngIfThen", ngIfElse: "ngIfElse" } });
    /** @nocollapse */
    NgIf.ctorParameters = () => [
        { type: ViewContainerRef },
        { type: TemplateRef }
    ];
    NgIf.propDecorators = {
        ngIf: [{ type: Input }],
        ngIfThen: [{ type: Input }],
        ngIfElse: [{ type: Input }]
    };
    /**
     * \@publicApi
     * @template T
     */
    class NgIfContext {
        constructor() {
            this.$implicit = (/** @type {?} */ (null));
            this.ngIf = (/** @type {?} */ (null));
        }
    }
    /**
     * @param {?} property
     * @param {?} templateRef
     * @return {?}
     */
    function assertTemplate(property, templateRef) {
        /** @type {?} */
        const isTemplateRefOrNull = !!(!templateRef || templateRef.createEmbeddedView);
        if (!isTemplateRefOrNull) {
            throw new Error(`${property} must be a TemplateRef, but received '${stringify(templateRef)}'.`);
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/src/directives/ng_switch.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class SwitchView {
        /**
         * @param {?} _viewContainerRef
         * @param {?} _templateRef
         */
        constructor(_viewContainerRef, _templateRef) {
            this._viewContainerRef = _viewContainerRef;
            this._templateRef = _templateRef;
            this._created = false;
        }
        /**
         * @return {?}
         */
        create() {
            this._created = true;
            this._viewContainerRef.createEmbeddedView(this._templateRef);
        }
        /**
         * @return {?}
         */
        destroy() {
            this._created = false;
            this._viewContainerRef.clear();
        }
        /**
         * @param {?} created
         * @return {?}
         */
        enforceState(created) {
            if (created && !this._created) {
                this.create();
            }
            else if (!created && this._created) {
                this.destroy();
            }
        }
    }
    /**
     * \@ngModule CommonModule
     *
     * \@description
     * The `[ngSwitch]` directive on a container specifies an expression to match against.
     * The expressions to match are provided by `ngSwitchCase` directives on views within the container.
     * - Every view that matches is rendered.
     * - If there are no matches, a view with the `ngSwitchDefault` directive is rendered.
     * - Elements within the `[NgSwitch]` statement but outside of any `NgSwitchCase`
     * or `ngSwitchDefault` directive are preserved at the location.
     *
     * \@usageNotes
     * Define a container element for the directive, and specify the switch expression
     * to match against as an attribute:
     *
     * ```
     * <container-element [ngSwitch]="switch_expression">
     * ```
     *
     * Within the container, `*ngSwitchCase` statements specify the match expressions
     * as attributes. Include `*ngSwitchDefault` as the final case.
     *
     * ```
     * <container-element [ngSwitch]="switch_expression">
     *    <some-element *ngSwitchCase="match_expression_1">...</some-element>
     * ...
     *    <some-element *ngSwitchDefault>...</some-element>
     * </container-element>
     * ```
     *
     * ### Usage Examples
     *
     * The following example shows how to use more than one case to display the same view:
     *
     * ```
     * <container-element [ngSwitch]="switch_expression">
     *   <!-- the same view can be shown in more than one case -->
     *   <some-element *ngSwitchCase="match_expression_1">...</some-element>
     *   <some-element *ngSwitchCase="match_expression_2">...</some-element>
     *   <some-other-element *ngSwitchCase="match_expression_3">...</some-other-element>
     *   <!--default case when there are no matches -->
     *   <some-element *ngSwitchDefault>...</some-element>
     * </container-element>
     * ```
     *
     * The following example shows how cases can be nested:
     * ```
     * <container-element [ngSwitch]="switch_expression">
     *       <some-element *ngSwitchCase="match_expression_1">...</some-element>
     *       <some-element *ngSwitchCase="match_expression_2">...</some-element>
     *       <some-other-element *ngSwitchCase="match_expression_3">...</some-other-element>
     *       <ng-container *ngSwitchCase="match_expression_3">
     *         <!-- use a ng-container to group multiple root nodes -->
     *         <inner-element></inner-element>
     *         <inner-other-element></inner-other-element>
     *       </ng-container>
     *       <some-element *ngSwitchDefault>...</some-element>
     *     </container-element>
     * ```
     *
     * \@publicApi
     * @see `NgSwitchCase`
     * @see `NgSwitchDefault`
     * @see [Structural Directives](guide/structural-directives)
     *
     */
    class NgSwitch {
        constructor() {
            this._defaultUsed = false;
            this._caseCount = 0;
            this._lastCaseCheckIndex = 0;
            this._lastCasesMatched = false;
        }
        /**
         * @param {?} newValue
         * @return {?}
         */
        set ngSwitch(newValue) {
            this._ngSwitch = newValue;
            if (this._caseCount === 0) {
                this._updateDefaultCases(true);
            }
        }
        /**
         * \@internal
         * @return {?}
         */
        _addCase() {
            return this._caseCount++;
        }
        /**
         * \@internal
         * @param {?} view
         * @return {?}
         */
        _addDefault(view) {
            if (!this._defaultViews) {
                this._defaultViews = [];
            }
            this._defaultViews.push(view);
        }
        /**
         * \@internal
         * @param {?} value
         * @return {?}
         */
        _matchCase(value) {
            /** @type {?} */
            const matched = value == this._ngSwitch;
            this._lastCasesMatched = this._lastCasesMatched || matched;
            this._lastCaseCheckIndex++;
            if (this._lastCaseCheckIndex === this._caseCount) {
                this._updateDefaultCases(!this._lastCasesMatched);
                this._lastCaseCheckIndex = 0;
                this._lastCasesMatched = false;
            }
            return matched;
        }
        /**
         * @private
         * @param {?} useDefault
         * @return {?}
         */
        _updateDefaultCases(useDefault) {
            if (this._defaultViews && useDefault !== this._defaultUsed) {
                this._defaultUsed = useDefault;
                for (let i = 0; i < this._defaultViews.length; i++) {
                    /** @type {?} */
                    const defaultView = this._defaultViews[i];
                    defaultView.enforceState(useDefault);
                }
            }
        }
    }
    NgSwitch.ɵfac = function NgSwitch_Factory(t) { return new (t || NgSwitch)(); };
    NgSwitch.ɵdir = ɵɵdefineDirective({ type: NgSwitch, selectors: [["", "ngSwitch", ""]], inputs: { ngSwitch: "ngSwitch" } });
    NgSwitch.propDecorators = {
        ngSwitch: [{ type: Input }]
    };
    /**
     * \@ngModule CommonModule
     *
     * \@description
     * Provides a switch case expression to match against an enclosing `ngSwitch` expression.
     * When the expressions match, the given `NgSwitchCase` template is rendered.
     * If multiple match expressions match the switch expression value, all of them are displayed.
     *
     * \@usageNotes
     *
     * Within a switch container, `*ngSwitchCase` statements specify the match expressions
     * as attributes. Include `*ngSwitchDefault` as the final case.
     *
     * ```
     * <container-element [ngSwitch]="switch_expression">
     *   <some-element *ngSwitchCase="match_expression_1">...</some-element>
     *   ...
     *   <some-element *ngSwitchDefault>...</some-element>
     * </container-element>
     * ```
     *
     * Each switch-case statement contains an in-line HTML template or template reference
     * that defines the subtree to be selected if the value of the match expression
     * matches the value of the switch expression.
     *
     * Unlike JavaScript, which uses strict equality, Angular uses loose equality.
     * This means that the empty string, `""` matches 0.
     *
     * \@publicApi
     * @see `NgSwitch`
     * @see `NgSwitchDefault`
     *
     */
    class NgSwitchCase {
        /**
         * @param {?} viewContainer
         * @param {?} templateRef
         * @param {?} ngSwitch
         */
        constructor(viewContainer, templateRef, ngSwitch) {
            this.ngSwitch = ngSwitch;
            ngSwitch._addCase();
            this._view = new SwitchView(viewContainer, templateRef);
        }
        /**
         * Performs case matching. For internal use only.
         * @return {?}
         */
        ngDoCheck() {
            this._view.enforceState(this.ngSwitch._matchCase(this.ngSwitchCase));
        }
    }
    NgSwitchCase.ɵfac = function NgSwitchCase_Factory(t) { return new (t || NgSwitchCase)(ɵɵdirectiveInject(ViewContainerRef), ɵɵdirectiveInject(TemplateRef), ɵɵdirectiveInject(NgSwitch, 1)); };
    NgSwitchCase.ɵdir = ɵɵdefineDirective({ type: NgSwitchCase, selectors: [["", "ngSwitchCase", ""]], inputs: { ngSwitchCase: "ngSwitchCase" } });
    /** @nocollapse */
    NgSwitchCase.ctorParameters = () => [
        { type: ViewContainerRef },
        { type: TemplateRef },
        { type: NgSwitch, decorators: [{ type: Host }] }
    ];
    NgSwitchCase.propDecorators = {
        ngSwitchCase: [{ type: Input }]
    };
    /**
     * \@ngModule CommonModule
     *
     * \@description
     *
     * Creates a view that is rendered when no `NgSwitchCase` expressions
     * match the `NgSwitch` expression.
     * This statement should be the final case in an `NgSwitch`.
     *
     * \@publicApi
     * @see `NgSwitch`
     * @see `NgSwitchCase`
     *
     */
    class NgSwitchDefault {
        /**
         * @param {?} viewContainer
         * @param {?} templateRef
         * @param {?} ngSwitch
         */
        constructor(viewContainer, templateRef, ngSwitch) {
            ngSwitch._addDefault(new SwitchView(viewContainer, templateRef));
        }
    }
    NgSwitchDefault.ɵfac = function NgSwitchDefault_Factory(t) { return new (t || NgSwitchDefault)(ɵɵdirectiveInject(ViewContainerRef), ɵɵdirectiveInject(TemplateRef), ɵɵdirectiveInject(NgSwitch, 1)); };
    NgSwitchDefault.ɵdir = ɵɵdefineDirective({ type: NgSwitchDefault, selectors: [["", "ngSwitchDefault", ""]] });
    /** @nocollapse */
    NgSwitchDefault.ctorParameters = () => [
        { type: ViewContainerRef },
        { type: TemplateRef },
        { type: NgSwitch, decorators: [{ type: Host }] }
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/src/directives/ng_plural.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * \@ngModule CommonModule
     *
     * \@usageNotes
     * ```
     * <some-element [ngPlural]="value">
     *   <ng-template ngPluralCase="=0">there is nothing</ng-template>
     *   <ng-template ngPluralCase="=1">there is one</ng-template>
     *   <ng-template ngPluralCase="few">there are a few</ng-template>
     * </some-element>
     * ```
     *
     * \@description
     *
     * Adds / removes DOM sub-trees based on a numeric value. Tailored for pluralization.
     *
     * Displays DOM sub-trees that match the switch expression value, or failing that, DOM sub-trees
     * that match the switch expression's pluralization category.
     *
     * To use this directive you must provide a container element that sets the `[ngPlural]` attribute
     * to a switch expression. Inner elements with a `[ngPluralCase]` will display based on their
     * expression:
     * - if `[ngPluralCase]` is set to a value starting with `=`, it will only display if the value
     *   matches the switch expression exactly,
     * - otherwise, the view will be treated as a "category match", and will only display if exact
     *   value matches aren't found and the value maps to its category for the defined locale.
     *
     * See http://cldr.unicode.org/index/cldr-spec/plural-rules
     *
     * \@publicApi
     */
    class NgPlural {
        /**
         * @param {?} _localization
         */
        constructor(_localization) {
            this._localization = _localization;
            this._caseViews = {};
        }
        /**
         * @param {?} value
         * @return {?}
         */
        set ngPlural(value) {
            this._switchValue = value;
            this._updateView();
        }
        /**
         * @param {?} value
         * @param {?} switchView
         * @return {?}
         */
        addCase(value, switchView) {
            this._caseViews[value] = switchView;
        }
        /**
         * @private
         * @return {?}
         */
        _updateView() {
            this._clearViews();
            /** @type {?} */
            const cases = Object.keys(this._caseViews);
            /** @type {?} */
            const key = getPluralCategory(this._switchValue, cases, this._localization);
            this._activateView(this._caseViews[key]);
        }
        /**
         * @private
         * @return {?}
         */
        _clearViews() {
            if (this._activeView)
                this._activeView.destroy();
        }
        /**
         * @private
         * @param {?} view
         * @return {?}
         */
        _activateView(view) {
            if (view) {
                this._activeView = view;
                this._activeView.create();
            }
        }
    }
    NgPlural.ɵfac = function NgPlural_Factory(t) { return new (t || NgPlural)(ɵɵdirectiveInject(NgLocalization)); };
    NgPlural.ɵdir = ɵɵdefineDirective({ type: NgPlural, selectors: [["", "ngPlural", ""]], inputs: { ngPlural: "ngPlural" } });
    /** @nocollapse */
    NgPlural.ctorParameters = () => [
        { type: NgLocalization }
    ];
    NgPlural.propDecorators = {
        ngPlural: [{ type: Input }]
    };
    /**
     * \@ngModule CommonModule
     *
     * \@description
     *
     * Creates a view that will be added/removed from the parent {\@link NgPlural} when the
     * given expression matches the plural expression according to CLDR rules.
     *
     * \@usageNotes
     * ```
     * <some-element [ngPlural]="value">
     *   <ng-template ngPluralCase="=0">...</ng-template>
     *   <ng-template ngPluralCase="other">...</ng-template>
     * </some-element>
     * ```
     *
     * See {\@link NgPlural} for more details and example.
     *
     * \@publicApi
     */
    class NgPluralCase {
        /**
         * @param {?} value
         * @param {?} template
         * @param {?} viewContainer
         * @param {?} ngPlural
         */
        constructor(value, template, viewContainer, ngPlural) {
            this.value = value;
            /** @type {?} */
            const isANumber = !isNaN(Number(value));
            ngPlural.addCase(isANumber ? `=${value}` : value, new SwitchView(viewContainer, template));
        }
    }
    NgPluralCase.ɵfac = function NgPluralCase_Factory(t) { return new (t || NgPluralCase)(ɵɵinjectAttribute('ngPluralCase'), ɵɵdirectiveInject(TemplateRef), ɵɵdirectiveInject(ViewContainerRef), ɵɵdirectiveInject(NgPlural, 1)); };
    NgPluralCase.ɵdir = ɵɵdefineDirective({ type: NgPluralCase, selectors: [["", "ngPluralCase", ""]] });
    /** @nocollapse */
    NgPluralCase.ctorParameters = () => [
        { type: String, decorators: [{ type: Attribute, args: ['ngPluralCase',] }] },
        { type: TemplateRef },
        { type: ViewContainerRef },
        { type: NgPlural, decorators: [{ type: Host }] }
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/src/directives/ng_style.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * \@ngModule CommonModule
     *
     * \@usageNotes
     *
     * Set the font of the containing element to the result of an expression.
     *
     * ```
     * <some-element [ngStyle]="{'font-style': styleExp}">...</some-element>
     * ```
     *
     * Set the width of the containing element to a pixel value returned by an expression.
     *
     * ```
     * <some-element [ngStyle]="{'max-width.px': widthExp}">...</some-element>
     * ```
     *
     * Set a collection of style values using an expression that returns key-value pairs.
     *
     * ```
     * <some-element [ngStyle]="objExp">...</some-element>
     * ```
     *
     * \@description
     *
     * An attribute directive that updates styles for the containing HTML element.
     * Sets one or more style properties, specified as colon-separated key-value pairs.
     * The key is a style name, with an optional `.<unit>` suffix
     * (such as 'top.px', 'font-style.em').
     * The value is an expression to be evaluated.
     * The resulting non-null value, expressed in the given unit,
     * is assigned to the given style property.
     * If the result of evaluation is null, the corresponding style is removed.
     *
     * \@publicApi
     */
    class NgStyle {
        /**
         * @param {?} _ngEl
         * @param {?} _differs
         * @param {?} _renderer
         */
        constructor(_ngEl, _differs, _renderer) {
            this._ngEl = _ngEl;
            this._differs = _differs;
            this._renderer = _renderer;
            this._ngStyle = null;
            this._differ = null;
        }
        /**
         * @param {?} values
         * @return {?}
         */
        set ngStyle(values) {
            this._ngStyle = values;
            if (!this._differ && values) {
                this._differ = this._differs.find(values).create();
            }
        }
        /**
         * @return {?}
         */
        ngDoCheck() {
            if (this._differ) {
                /** @type {?} */
                const changes = this._differ.diff((/** @type {?} */ (this._ngStyle)));
                if (changes) {
                    this._applyChanges(changes);
                }
            }
        }
        /**
         * @private
         * @param {?} nameAndUnit
         * @param {?} value
         * @return {?}
         */
        _setStyle(nameAndUnit, value) {
            const [name, unit] = nameAndUnit.split('.');
            value = value != null && unit ? `${value}${unit}` : value;
            if (value != null) {
                this._renderer.setStyle(this._ngEl.nativeElement, name, (/** @type {?} */ (value)));
            }
            else {
                this._renderer.removeStyle(this._ngEl.nativeElement, name);
            }
        }
        /**
         * @private
         * @param {?} changes
         * @return {?}
         */
        _applyChanges(changes) {
            changes.forEachRemovedItem((/**
             * @param {?} record
             * @return {?}
             */
            (record) => this._setStyle(record.key, null)));
            changes.forEachAddedItem((/**
             * @param {?} record
             * @return {?}
             */
            (record) => this._setStyle(record.key, record.currentValue)));
            changes.forEachChangedItem((/**
             * @param {?} record
             * @return {?}
             */
            (record) => this._setStyle(record.key, record.currentValue)));
        }
    }
    NgStyle.ɵfac = function NgStyle_Factory(t) { return new (t || NgStyle)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(KeyValueDiffers), ɵɵdirectiveInject(Renderer2)); };
    NgStyle.ɵdir = ɵɵdefineDirective({ type: NgStyle, selectors: [["", "ngStyle", ""]], inputs: { ngStyle: "ngStyle" } });
    /** @nocollapse */
    NgStyle.ctorParameters = () => [
        { type: ElementRef },
        { type: KeyValueDiffers },
        { type: Renderer2 }
    ];
    NgStyle.propDecorators = {
        ngStyle: [{ type: Input, args: ['ngStyle',] }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/src/directives/ng_template_outlet.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * \@ngModule CommonModule
     *
     * \@description
     *
     * Inserts an embedded view from a prepared `TemplateRef`.
     *
     * You can attach a context object to the `EmbeddedViewRef` by setting `[ngTemplateOutletContext]`.
     * `[ngTemplateOutletContext]` should be an object, the object's keys will be available for binding
     * by the local template `let` declarations.
     *
     * \@usageNotes
     * ```
     * <ng-container *ngTemplateOutlet="templateRefExp; context: contextExp"></ng-container>
     * ```
     *
     * Using the key `$implicit` in the context object will set its value as default.
     *
     * ### Example
     *
     * {\@example common/ngTemplateOutlet/ts/module.ts region='NgTemplateOutlet'}
     *
     * \@publicApi
     */
    class NgTemplateOutlet {
        /**
         * @param {?} _viewContainerRef
         */
        constructor(_viewContainerRef) {
            this._viewContainerRef = _viewContainerRef;
            this._viewRef = null;
            /**
             * A context object to attach to the {\@link EmbeddedViewRef}. This should be an
             * object, the object's keys will be available for binding by the local template `let`
             * declarations.
             * Using the key `$implicit` in the context object will set its value as default.
             */
            this.ngTemplateOutletContext = null;
            /**
             * A string defining the template reference and optionally the context object for the template.
             */
            this.ngTemplateOutlet = null;
        }
        /**
         * @param {?} changes
         * @return {?}
         */
        ngOnChanges(changes) {
            /** @type {?} */
            const recreateView = this._shouldRecreateView(changes);
            if (recreateView) {
                /** @type {?} */
                const viewContainerRef = this._viewContainerRef;
                if (this._viewRef) {
                    viewContainerRef.remove(viewContainerRef.indexOf(this._viewRef));
                }
                this._viewRef = this.ngTemplateOutlet ?
                    viewContainerRef.createEmbeddedView(this.ngTemplateOutlet, this.ngTemplateOutletContext) :
                    null;
            }
            else if (this._viewRef && this.ngTemplateOutletContext) {
                this._updateExistingContext(this.ngTemplateOutletContext);
            }
        }
        /**
         * We need to re-create existing embedded view if:
         * - templateRef has changed
         * - context has changes
         *
         * We mark context object as changed when the corresponding object
         * shape changes (new properties are added or existing properties are removed).
         * In other words we consider context with the same properties as "the same" even
         * if object reference changes (see https://github.com/angular/angular/issues/13407).
         * @private
         * @param {?} changes
         * @return {?}
         */
        _shouldRecreateView(changes) {
            /** @type {?} */
            const ctxChange = changes['ngTemplateOutletContext'];
            return !!changes['ngTemplateOutlet'] || (ctxChange && this._hasContextShapeChanged(ctxChange));
        }
        /**
         * @private
         * @param {?} ctxChange
         * @return {?}
         */
        _hasContextShapeChanged(ctxChange) {
            /** @type {?} */
            const prevCtxKeys = Object.keys(ctxChange.previousValue || {});
            /** @type {?} */
            const currCtxKeys = Object.keys(ctxChange.currentValue || {});
            if (prevCtxKeys.length === currCtxKeys.length) {
                for (let propName of currCtxKeys) {
                    if (prevCtxKeys.indexOf(propName) === -1) {
                        return true;
                    }
                }
                return false;
            }
            return true;
        }
        /**
         * @private
         * @param {?} ctx
         * @return {?}
         */
        _updateExistingContext(ctx) {
            for (let propName of Object.keys(ctx)) {
                ((/** @type {?} */ ((/** @type {?} */ (this._viewRef)).context)))[propName] = ((/** @type {?} */ (this.ngTemplateOutletContext)))[propName];
            }
        }
    }
    NgTemplateOutlet.ɵfac = function NgTemplateOutlet_Factory(t) { return new (t || NgTemplateOutlet)(ɵɵdirectiveInject(ViewContainerRef)); };
    NgTemplateOutlet.ɵdir = ɵɵdefineDirective({ type: NgTemplateOutlet, selectors: [["", "ngTemplateOutlet", ""]], inputs: { ngTemplateOutletContext: "ngTemplateOutletContext", ngTemplateOutlet: "ngTemplateOutlet" }, features: [ɵɵNgOnChangesFeature] });
    /** @nocollapse */
    NgTemplateOutlet.ctorParameters = () => [
        { type: ViewContainerRef }
    ];
    NgTemplateOutlet.propDecorators = {
        ngTemplateOutletContext: [{ type: Input }],
        ngTemplateOutlet: [{ type: Input }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/src/pipes/invalid_pipe_argument_error.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @param {?} type
     * @param {?} value
     * @return {?}
     */
    function invalidPipeArgumentError(type, value) {
        return Error(`InvalidPipeArgument: '${value}' for pipe '${stringify(type)}'`);
    }
    class ObservableStrategy {
        /**
         * @param {?} async
         * @param {?} updateLatestValue
         * @return {?}
         */
        createSubscription(async, updateLatestValue) {
            return async.subscribe({
                next: updateLatestValue,
                error: (/**
                 * @param {?} e
                 * @return {?}
                 */
                (e) => {
                    throw e;
                })
            });
        }
        /**
         * @param {?} subscription
         * @return {?}
         */
        dispose(subscription) {
            subscription.unsubscribe();
        }
        /**
         * @param {?} subscription
         * @return {?}
         */
        onDestroy(subscription) {
            subscription.unsubscribe();
        }
    }
    class PromiseStrategy {
        /**
         * @param {?} async
         * @param {?} updateLatestValue
         * @return {?}
         */
        createSubscription(async, updateLatestValue) {
            return async.then(updateLatestValue, (/**
             * @param {?} e
             * @return {?}
             */
            e => {
                throw e;
            }));
        }
        /**
         * @param {?} subscription
         * @return {?}
         */
        dispose(subscription) { }
        /**
         * @param {?} subscription
         * @return {?}
         */
        onDestroy(subscription) { }
    }
    /** @type {?} */
    const _promiseStrategy = new PromiseStrategy();
    /** @type {?} */
    const _observableStrategy = new ObservableStrategy();
    /**
     * \@ngModule CommonModule
     * \@description
     *
     * Unwraps a value from an asynchronous primitive.
     *
     * The `async` pipe subscribes to an `Observable` or `Promise` and returns the latest value it has
     * emitted. When a new value is emitted, the `async` pipe marks the component to be checked for
     * changes. When the component gets destroyed, the `async` pipe unsubscribes automatically to avoid
     * potential memory leaks.
     *
     * \@usageNotes
     *
     * ### Examples
     *
     * This example binds a `Promise` to the view. Clicking the `Resolve` button resolves the
     * promise.
     *
     * {\@example common/pipes/ts/async_pipe.ts region='AsyncPipePromise'}
     *
     * It's also possible to use `async` with Observables. The example below binds the `time` Observable
     * to the view. The Observable continuously updates the view with the current time.
     *
     * {\@example common/pipes/ts/async_pipe.ts region='AsyncPipeObservable'}
     *
     * \@publicApi
     */
    class AsyncPipe {
        /**
         * @param {?} _ref
         */
        constructor(_ref) {
            this._ref = _ref;
            this._latestValue = null;
            this._latestReturnedValue = null;
            this._subscription = null;
            this._obj = null;
            this._strategy = (/** @type {?} */ (null));
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            if (this._subscription) {
                this._dispose();
            }
        }
        /**
         * @param {?} obj
         * @return {?}
         */
        transform(obj) {
            if (!this._obj) {
                if (obj) {
                    this._subscribe(obj);
                }
                this._latestReturnedValue = this._latestValue;
                return this._latestValue;
            }
            if (obj !== this._obj) {
                this._dispose();
                return this.transform((/** @type {?} */ (obj)));
            }
            if (looseIdentical(this._latestValue, this._latestReturnedValue)) {
                return this._latestReturnedValue;
            }
            this._latestReturnedValue = this._latestValue;
            return WrappedValue.wrap(this._latestValue);
        }
        /**
         * @private
         * @param {?} obj
         * @return {?}
         */
        _subscribe(obj) {
            this._obj = obj;
            this._strategy = this._selectStrategy(obj);
            this._subscription = this._strategy.createSubscription(obj, (/**
             * @param {?} value
             * @return {?}
             */
            (value) => this._updateLatestValue(obj, value)));
        }
        /**
         * @private
         * @param {?} obj
         * @return {?}
         */
        _selectStrategy(obj) {
            if (isPromise$1(obj)) {
                return _promiseStrategy;
            }
            if (isObservable$1(obj)) {
                return _observableStrategy;
            }
            throw invalidPipeArgumentError(AsyncPipe, obj);
        }
        /**
         * @private
         * @return {?}
         */
        _dispose() {
            this._strategy.dispose((/** @type {?} */ (this._subscription)));
            this._latestValue = null;
            this._latestReturnedValue = null;
            this._subscription = null;
            this._obj = null;
        }
        /**
         * @private
         * @param {?} async
         * @param {?} value
         * @return {?}
         */
        _updateLatestValue(async, value) {
            if (async === this._obj) {
                this._latestValue = value;
                this._ref.markForCheck();
            }
        }
    }
    AsyncPipe.ɵfac = function AsyncPipe_Factory(t) { return new (t || AsyncPipe)(ɵɵinjectPipeChangeDetectorRef()); };
    AsyncPipe.ɵpipe = ɵɵdefinePipe({ name: "async", type: AsyncPipe, pure: false });
    /** @nocollapse */
    AsyncPipe.ctorParameters = () => [
        { type: ChangeDetectorRef }
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/src/pipes/case_conversion_pipes.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Transforms text to all lower case.
     *
     * @see `UpperCasePipe`
     * @see `TitleCasePipe`
     * \@usageNotes
     *
     * The following example defines a view that allows the user to enter
     * text, and then uses the pipe to convert the input text to all lower case.
     *
     * <code-example path="common/pipes/ts/lowerupper_pipe.ts" region='LowerUpperPipe'></code-example>
     *
     * \@ngModule CommonModule
     * \@publicApi
     */
    class LowerCasePipe {
        /**
         * @param {?} value The string to transform to lower case.
         * @return {?}
         */
        transform(value) {
            if (!value)
                return value;
            if (typeof value !== 'string') {
                throw invalidPipeArgumentError(LowerCasePipe, value);
            }
            return value.toLowerCase();
        }
    }
    LowerCasePipe.ɵfac = function LowerCasePipe_Factory(t) { return new (t || LowerCasePipe)(); };
    LowerCasePipe.ɵpipe = ɵɵdefinePipe({ name: "lowercase", type: LowerCasePipe, pure: true });
    //
    // Regex below matches any Unicode word and compatible with ES5. In ES2018 the same result
    // can be achieved by using /\p{L}\S*/gu and also known as Unicode Property Escapes
    // (http://2ality.com/2017/07/regexp-unicode-property-escapes.html). Since there is no
    // transpilation of this functionality down to ES5 without external tool, the only solution is
    // to use already transpiled form. Example can be found here -
    // https://mothereff.in/regexpu#input=var+regex+%3D+/%5Cp%7BL%7D/u%3B&unicodePropertyEscape=1
    //
    /** @type {?} */
    const unicodeWordMatch = /(?:[A-Za-z\xAA\xB5\xBA\xC0-\xD6\xD8-\xF6\xF8-\u02C1\u02C6-\u02D1\u02E0-\u02E4\u02EC\u02EE\u0370-\u0374\u0376\u0377\u037A-\u037D\u037F\u0386\u0388-\u038A\u038C\u038E-\u03A1\u03A3-\u03F5\u03F7-\u0481\u048A-\u052F\u0531-\u0556\u0559\u0561-\u0587\u05D0-\u05EA\u05F0-\u05F2\u0620-\u064A\u066E\u066F\u0671-\u06D3\u06D5\u06E5\u06E6\u06EE\u06EF\u06FA-\u06FC\u06FF\u0710\u0712-\u072F\u074D-\u07A5\u07B1\u07CA-\u07EA\u07F4\u07F5\u07FA\u0800-\u0815\u081A\u0824\u0828\u0840-\u0858\u0860-\u086A\u08A0-\u08B4\u08B6-\u08BD\u0904-\u0939\u093D\u0950\u0958-\u0961\u0971-\u0980\u0985-\u098C\u098F\u0990\u0993-\u09A8\u09AA-\u09B0\u09B2\u09B6-\u09B9\u09BD\u09CE\u09DC\u09DD\u09DF-\u09E1\u09F0\u09F1\u09FC\u0A05-\u0A0A\u0A0F\u0A10\u0A13-\u0A28\u0A2A-\u0A30\u0A32\u0A33\u0A35\u0A36\u0A38\u0A39\u0A59-\u0A5C\u0A5E\u0A72-\u0A74\u0A85-\u0A8D\u0A8F-\u0A91\u0A93-\u0AA8\u0AAA-\u0AB0\u0AB2\u0AB3\u0AB5-\u0AB9\u0ABD\u0AD0\u0AE0\u0AE1\u0AF9\u0B05-\u0B0C\u0B0F\u0B10\u0B13-\u0B28\u0B2A-\u0B30\u0B32\u0B33\u0B35-\u0B39\u0B3D\u0B5C\u0B5D\u0B5F-\u0B61\u0B71\u0B83\u0B85-\u0B8A\u0B8E-\u0B90\u0B92-\u0B95\u0B99\u0B9A\u0B9C\u0B9E\u0B9F\u0BA3\u0BA4\u0BA8-\u0BAA\u0BAE-\u0BB9\u0BD0\u0C05-\u0C0C\u0C0E-\u0C10\u0C12-\u0C28\u0C2A-\u0C39\u0C3D\u0C58-\u0C5A\u0C60\u0C61\u0C80\u0C85-\u0C8C\u0C8E-\u0C90\u0C92-\u0CA8\u0CAA-\u0CB3\u0CB5-\u0CB9\u0CBD\u0CDE\u0CE0\u0CE1\u0CF1\u0CF2\u0D05-\u0D0C\u0D0E-\u0D10\u0D12-\u0D3A\u0D3D\u0D4E\u0D54-\u0D56\u0D5F-\u0D61\u0D7A-\u0D7F\u0D85-\u0D96\u0D9A-\u0DB1\u0DB3-\u0DBB\u0DBD\u0DC0-\u0DC6\u0E01-\u0E30\u0E32\u0E33\u0E40-\u0E46\u0E81\u0E82\u0E84\u0E87\u0E88\u0E8A\u0E8D\u0E94-\u0E97\u0E99-\u0E9F\u0EA1-\u0EA3\u0EA5\u0EA7\u0EAA\u0EAB\u0EAD-\u0EB0\u0EB2\u0EB3\u0EBD\u0EC0-\u0EC4\u0EC6\u0EDC-\u0EDF\u0F00\u0F40-\u0F47\u0F49-\u0F6C\u0F88-\u0F8C\u1000-\u102A\u103F\u1050-\u1055\u105A-\u105D\u1061\u1065\u1066\u106E-\u1070\u1075-\u1081\u108E\u10A0-\u10C5\u10C7\u10CD\u10D0-\u10FA\u10FC-\u1248\u124A-\u124D\u1250-\u1256\u1258\u125A-\u125D\u1260-\u1288\u128A-\u128D\u1290-\u12B0\u12B2-\u12B5\u12B8-\u12BE\u12C0\u12C2-\u12C5\u12C8-\u12D6\u12D8-\u1310\u1312-\u1315\u1318-\u135A\u1380-\u138F\u13A0-\u13F5\u13F8-\u13FD\u1401-\u166C\u166F-\u167F\u1681-\u169A\u16A0-\u16EA\u16F1-\u16F8\u1700-\u170C\u170E-\u1711\u1720-\u1731\u1740-\u1751\u1760-\u176C\u176E-\u1770\u1780-\u17B3\u17D7\u17DC\u1820-\u1877\u1880-\u1884\u1887-\u18A8\u18AA\u18B0-\u18F5\u1900-\u191E\u1950-\u196D\u1970-\u1974\u1980-\u19AB\u19B0-\u19C9\u1A00-\u1A16\u1A20-\u1A54\u1AA7\u1B05-\u1B33\u1B45-\u1B4B\u1B83-\u1BA0\u1BAE\u1BAF\u1BBA-\u1BE5\u1C00-\u1C23\u1C4D-\u1C4F\u1C5A-\u1C7D\u1C80-\u1C88\u1CE9-\u1CEC\u1CEE-\u1CF1\u1CF5\u1CF6\u1D00-\u1DBF\u1E00-\u1F15\u1F18-\u1F1D\u1F20-\u1F45\u1F48-\u1F4D\u1F50-\u1F57\u1F59\u1F5B\u1F5D\u1F5F-\u1F7D\u1F80-\u1FB4\u1FB6-\u1FBC\u1FBE\u1FC2-\u1FC4\u1FC6-\u1FCC\u1FD0-\u1FD3\u1FD6-\u1FDB\u1FE0-\u1FEC\u1FF2-\u1FF4\u1FF6-\u1FFC\u2071\u207F\u2090-\u209C\u2102\u2107\u210A-\u2113\u2115\u2119-\u211D\u2124\u2126\u2128\u212A-\u212D\u212F-\u2139\u213C-\u213F\u2145-\u2149\u214E\u2183\u2184\u2C00-\u2C2E\u2C30-\u2C5E\u2C60-\u2CE4\u2CEB-\u2CEE\u2CF2\u2CF3\u2D00-\u2D25\u2D27\u2D2D\u2D30-\u2D67\u2D6F\u2D80-\u2D96\u2DA0-\u2DA6\u2DA8-\u2DAE\u2DB0-\u2DB6\u2DB8-\u2DBE\u2DC0-\u2DC6\u2DC8-\u2DCE\u2DD0-\u2DD6\u2DD8-\u2DDE\u2E2F\u3005\u3006\u3031-\u3035\u303B\u303C\u3041-\u3096\u309D-\u309F\u30A1-\u30FA\u30FC-\u30FF\u3105-\u312E\u3131-\u318E\u31A0-\u31BA\u31F0-\u31FF\u3400-\u4DB5\u4E00-\u9FEA\uA000-\uA48C\uA4D0-\uA4FD\uA500-\uA60C\uA610-\uA61F\uA62A\uA62B\uA640-\uA66E\uA67F-\uA69D\uA6A0-\uA6E5\uA717-\uA71F\uA722-\uA788\uA78B-\uA7AE\uA7B0-\uA7B7\uA7F7-\uA801\uA803-\uA805\uA807-\uA80A\uA80C-\uA822\uA840-\uA873\uA882-\uA8B3\uA8F2-\uA8F7\uA8FB\uA8FD\uA90A-\uA925\uA930-\uA946\uA960-\uA97C\uA984-\uA9B2\uA9CF\uA9E0-\uA9E4\uA9E6-\uA9EF\uA9FA-\uA9FE\uAA00-\uAA28\uAA40-\uAA42\uAA44-\uAA4B\uAA60-\uAA76\uAA7A\uAA7E-\uAAAF\uAAB1\uAAB5\uAAB6\uAAB9-\uAABD\uAAC0\uAAC2\uAADB-\uAADD\uAAE0-\uAAEA\uAAF2-\uAAF4\uAB01-\uAB06\uAB09-\uAB0E\uAB11-\uAB16\uAB20-\uAB26\uAB28-\uAB2E\uAB30-\uAB5A\uAB5C-\uAB65\uAB70-\uABE2\uAC00-\uD7A3\uD7B0-\uD7C6\uD7CB-\uD7FB\uF900-\uFA6D\uFA70-\uFAD9\uFB00-\uFB06\uFB13-\uFB17\uFB1D\uFB1F-\uFB28\uFB2A-\uFB36\uFB38-\uFB3C\uFB3E\uFB40\uFB41\uFB43\uFB44\uFB46-\uFBB1\uFBD3-\uFD3D\uFD50-\uFD8F\uFD92-\uFDC7\uFDF0-\uFDFB\uFE70-\uFE74\uFE76-\uFEFC\uFF21-\uFF3A\uFF41-\uFF5A\uFF66-\uFFBE\uFFC2-\uFFC7\uFFCA-\uFFCF\uFFD2-\uFFD7\uFFDA-\uFFDC]|\uD800[\uDC00-\uDC0B\uDC0D-\uDC26\uDC28-\uDC3A\uDC3C\uDC3D\uDC3F-\uDC4D\uDC50-\uDC5D\uDC80-\uDCFA\uDE80-\uDE9C\uDEA0-\uDED0\uDF00-\uDF1F\uDF2D-\uDF40\uDF42-\uDF49\uDF50-\uDF75\uDF80-\uDF9D\uDFA0-\uDFC3\uDFC8-\uDFCF]|\uD801[\uDC00-\uDC9D\uDCB0-\uDCD3\uDCD8-\uDCFB\uDD00-\uDD27\uDD30-\uDD63\uDE00-\uDF36\uDF40-\uDF55\uDF60-\uDF67]|\uD802[\uDC00-\uDC05\uDC08\uDC0A-\uDC35\uDC37\uDC38\uDC3C\uDC3F-\uDC55\uDC60-\uDC76\uDC80-\uDC9E\uDCE0-\uDCF2\uDCF4\uDCF5\uDD00-\uDD15\uDD20-\uDD39\uDD80-\uDDB7\uDDBE\uDDBF\uDE00\uDE10-\uDE13\uDE15-\uDE17\uDE19-\uDE33\uDE60-\uDE7C\uDE80-\uDE9C\uDEC0-\uDEC7\uDEC9-\uDEE4\uDF00-\uDF35\uDF40-\uDF55\uDF60-\uDF72\uDF80-\uDF91]|\uD803[\uDC00-\uDC48\uDC80-\uDCB2\uDCC0-\uDCF2]|\uD804[\uDC03-\uDC37\uDC83-\uDCAF\uDCD0-\uDCE8\uDD03-\uDD26\uDD50-\uDD72\uDD76\uDD83-\uDDB2\uDDC1-\uDDC4\uDDDA\uDDDC\uDE00-\uDE11\uDE13-\uDE2B\uDE80-\uDE86\uDE88\uDE8A-\uDE8D\uDE8F-\uDE9D\uDE9F-\uDEA8\uDEB0-\uDEDE\uDF05-\uDF0C\uDF0F\uDF10\uDF13-\uDF28\uDF2A-\uDF30\uDF32\uDF33\uDF35-\uDF39\uDF3D\uDF50\uDF5D-\uDF61]|\uD805[\uDC00-\uDC34\uDC47-\uDC4A\uDC80-\uDCAF\uDCC4\uDCC5\uDCC7\uDD80-\uDDAE\uDDD8-\uDDDB\uDE00-\uDE2F\uDE44\uDE80-\uDEAA\uDF00-\uDF19]|\uD806[\uDCA0-\uDCDF\uDCFF\uDE00\uDE0B-\uDE32\uDE3A\uDE50\uDE5C-\uDE83\uDE86-\uDE89\uDEC0-\uDEF8]|\uD807[\uDC00-\uDC08\uDC0A-\uDC2E\uDC40\uDC72-\uDC8F\uDD00-\uDD06\uDD08\uDD09\uDD0B-\uDD30\uDD46]|\uD808[\uDC00-\uDF99]|\uD809[\uDC80-\uDD43]|[\uD80C\uD81C-\uD820\uD840-\uD868\uD86A-\uD86C\uD86F-\uD872\uD874-\uD879][\uDC00-\uDFFF]|\uD80D[\uDC00-\uDC2E]|\uD811[\uDC00-\uDE46]|\uD81A[\uDC00-\uDE38\uDE40-\uDE5E\uDED0-\uDEED\uDF00-\uDF2F\uDF40-\uDF43\uDF63-\uDF77\uDF7D-\uDF8F]|\uD81B[\uDF00-\uDF44\uDF50\uDF93-\uDF9F\uDFE0\uDFE1]|\uD821[\uDC00-\uDFEC]|\uD822[\uDC00-\uDEF2]|\uD82C[\uDC00-\uDD1E\uDD70-\uDEFB]|\uD82F[\uDC00-\uDC6A\uDC70-\uDC7C\uDC80-\uDC88\uDC90-\uDC99]|\uD835[\uDC00-\uDC54\uDC56-\uDC9C\uDC9E\uDC9F\uDCA2\uDCA5\uDCA6\uDCA9-\uDCAC\uDCAE-\uDCB9\uDCBB\uDCBD-\uDCC3\uDCC5-\uDD05\uDD07-\uDD0A\uDD0D-\uDD14\uDD16-\uDD1C\uDD1E-\uDD39\uDD3B-\uDD3E\uDD40-\uDD44\uDD46\uDD4A-\uDD50\uDD52-\uDEA5\uDEA8-\uDEC0\uDEC2-\uDEDA\uDEDC-\uDEFA\uDEFC-\uDF14\uDF16-\uDF34\uDF36-\uDF4E\uDF50-\uDF6E\uDF70-\uDF88\uDF8A-\uDFA8\uDFAA-\uDFC2\uDFC4-\uDFCB]|\uD83A[\uDC00-\uDCC4\uDD00-\uDD43]|\uD83B[\uDE00-\uDE03\uDE05-\uDE1F\uDE21\uDE22\uDE24\uDE27\uDE29-\uDE32\uDE34-\uDE37\uDE39\uDE3B\uDE42\uDE47\uDE49\uDE4B\uDE4D-\uDE4F\uDE51\uDE52\uDE54\uDE57\uDE59\uDE5B\uDE5D\uDE5F\uDE61\uDE62\uDE64\uDE67-\uDE6A\uDE6C-\uDE72\uDE74-\uDE77\uDE79-\uDE7C\uDE7E\uDE80-\uDE89\uDE8B-\uDE9B\uDEA1-\uDEA3\uDEA5-\uDEA9\uDEAB-\uDEBB]|\uD869[\uDC00-\uDED6\uDF00-\uDFFF]|\uD86D[\uDC00-\uDF34\uDF40-\uDFFF]|\uD86E[\uDC00-\uDC1D\uDC20-\uDFFF]|\uD873[\uDC00-\uDEA1\uDEB0-\uDFFF]|\uD87A[\uDC00-\uDFE0]|\uD87E[\uDC00-\uDE1D])\S*/g;
    /**
     * Transforms text to title case.
     * Capitalizes the first letter of each word and transforms the
     * rest of the word to lower case.
     * Words are delimited by any whitespace character, such as a space, tab, or line-feed character.
     *
     * @see `LowerCasePipe`
     * @see `UpperCasePipe`
     *
     * \@usageNotes
     * The following example shows the result of transforming various strings into title case.
     *
     * <code-example path="common/pipes/ts/titlecase_pipe.ts" region='TitleCasePipe'></code-example>
     *
     * \@ngModule CommonModule
     * \@publicApi
     */
    class TitleCasePipe {
        /**
         * @param {?} value The string to transform to title case.
         * @return {?}
         */
        transform(value) {
            if (!value)
                return value;
            if (typeof value !== 'string') {
                throw invalidPipeArgumentError(TitleCasePipe, value);
            }
            return value.replace(unicodeWordMatch, ((/**
             * @param {?} txt
             * @return {?}
             */
            txt => txt[0].toUpperCase() + txt.substr(1).toLowerCase())));
        }
    }
    TitleCasePipe.ɵfac = function TitleCasePipe_Factory(t) { return new (t || TitleCasePipe)(); };
    TitleCasePipe.ɵpipe = ɵɵdefinePipe({ name: "titlecase", type: TitleCasePipe, pure: true });
    /**
     * Transforms text to all upper case.
     * @see `LowerCasePipe`
     * @see `TitleCasePipe`
     *
     * \@ngModule CommonModule
     * \@publicApi
     */
    class UpperCasePipe {
        /**
         * @param {?} value The string to transform to upper case.
         * @return {?}
         */
        transform(value) {
            if (!value)
                return value;
            if (typeof value !== 'string') {
                throw invalidPipeArgumentError(UpperCasePipe, value);
            }
            return value.toUpperCase();
        }
    }
    UpperCasePipe.ɵfac = function UpperCasePipe_Factory(t) { return new (t || UpperCasePipe)(); };
    UpperCasePipe.ɵpipe = ɵɵdefinePipe({ name: "uppercase", type: UpperCasePipe, pure: true });

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/src/pipes/date_pipe.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    // clang-format off
    /**
     * \@ngModule CommonModule
     * \@description
     *
     * Formats a date value according to locale rules.
     *
     * Only the `en-US` locale data comes with Angular. To localize dates
     * in another language, you must import the corresponding locale data.
     * See the [I18n guide](guide/i18n#i18n-pipes) for more information.
     *
     * @see `formatDate()`
     *
     *
     * \@usageNotes
     *
     * The result of this pipe is not reevaluated when the input is mutated. To avoid the need to
     * reformat the date on every change-detection cycle, treat the date as an immutable object
     * and change the reference when the pipe needs to run again.
     *
     * ### Pre-defined format options
     *
     * Examples are given in `en-US` locale.
     *
     * - `'short'`: equivalent to `'M/d/yy, h:mm a'` (`6/15/15, 9:03 AM`).
     * - `'medium'`: equivalent to `'MMM d, y, h:mm:ss a'` (`Jun 15, 2015, 9:03:01 AM`).
     * - `'long'`: equivalent to `'MMMM d, y, h:mm:ss a z'` (`June 15, 2015 at 9:03:01 AM
     * GMT+1`).
     * - `'full'`: equivalent to `'EEEE, MMMM d, y, h:mm:ss a zzzz'` (`Monday, June 15, 2015 at
     * 9:03:01 AM GMT+01:00`).
     * - `'shortDate'`: equivalent to `'M/d/yy'` (`6/15/15`).
     * - `'mediumDate'`: equivalent to `'MMM d, y'` (`Jun 15, 2015`).
     * - `'longDate'`: equivalent to `'MMMM d, y'` (`June 15, 2015`).
     * - `'fullDate'`: equivalent to `'EEEE, MMMM d, y'` (`Monday, June 15, 2015`).
     * - `'shortTime'`: equivalent to `'h:mm a'` (`9:03 AM`).
     * - `'mediumTime'`: equivalent to `'h:mm:ss a'` (`9:03:01 AM`).
     * - `'longTime'`: equivalent to `'h:mm:ss a z'` (`9:03:01 AM GMT+1`).
     * - `'fullTime'`: equivalent to `'h:mm:ss a zzzz'` (`9:03:01 AM GMT+01:00`).
     *
     * ### Custom format options
     *
     * You can construct a format string using symbols to specify the components
     * of a date-time value, as described in the following table.
     * Format details depend on the locale.
     * Fields marked with (*) are only available in the extra data set for the given locale.
     *
     *  | Field type         | Format      | Description                                                   | Example Value                                              |
     *  |--------------------|-------------|---------------------------------------------------------------|------------------------------------------------------------|
     *  | Era                | G, GG & GGG | Abbreviated                                                   | AD                                                         |
     *  |                    | GGGG        | Wide                                                          | Anno Domini                                                |
     *  |                    | GGGGG       | Narrow                                                        | A                                                          |
     *  | Year               | y           | Numeric: minimum digits                                       | 2, 20, 201, 2017, 20173                                    |
     *  |                    | yy          | Numeric: 2 digits + zero padded                               | 02, 20, 01, 17, 73                                         |
     *  |                    | yyy         | Numeric: 3 digits + zero padded                               | 002, 020, 201, 2017, 20173                                 |
     *  |                    | yyyy        | Numeric: 4 digits or more + zero padded                       | 0002, 0020, 0201, 2017, 20173                              |
     *  | Month              | M           | Numeric: 1 digit                                              | 9, 12                                                      |
     *  |                    | MM          | Numeric: 2 digits + zero padded                               | 09, 12                                                     |
     *  |                    | MMM         | Abbreviated                                                   | Sep                                                        |
     *  |                    | MMMM        | Wide                                                          | September                                                  |
     *  |                    | MMMMM       | Narrow                                                        | S                                                          |
     *  | Month standalone   | L           | Numeric: 1 digit                                              | 9, 12                                                      |
     *  |                    | LL          | Numeric: 2 digits + zero padded                               | 09, 12                                                     |
     *  |                    | LLL         | Abbreviated                                                   | Sep                                                        |
     *  |                    | LLLL        | Wide                                                          | September                                                  |
     *  |                    | LLLLL       | Narrow                                                        | S                                                          |
     *  | Week of year       | w           | Numeric: minimum digits                                       | 1... 53                                                    |
     *  |                    | ww          | Numeric: 2 digits + zero padded                               | 01... 53                                                   |
     *  | Week of month      | W           | Numeric: 1 digit                                              | 1... 5                                                     |
     *  | Day of month       | d           | Numeric: minimum digits                                       | 1                                                          |
     *  |                    | dd          | Numeric: 2 digits + zero padded                               | 01                                                          |
     *  | Week day           | E, EE & EEE | Abbreviated                                                   | Tue                                                        |
     *  |                    | EEEE        | Wide                                                          | Tuesday                                                    |
     *  |                    | EEEEE       | Narrow                                                        | T                                                          |
     *  |                    | EEEEEE      | Short                                                         | Tu                                                         |
     *  | Period             | a, aa & aaa | Abbreviated                                                   | am/pm or AM/PM                                             |
     *  |                    | aaaa        | Wide (fallback to `a` when missing)                           | ante meridiem/post meridiem                                |
     *  |                    | aaaaa       | Narrow                                                        | a/p                                                        |
     *  | Period*            | B, BB & BBB | Abbreviated                                                   | mid.                                                       |
     *  |                    | BBBB        | Wide                                                          | am, pm, midnight, noon, morning, afternoon, evening, night |
     *  |                    | BBBBB       | Narrow                                                        | md                                                         |
     *  | Period standalone* | b, bb & bbb | Abbreviated                                                   | mid.                                                       |
     *  |                    | bbbb        | Wide                                                          | am, pm, midnight, noon, morning, afternoon, evening, night |
     *  |                    | bbbbb       | Narrow                                                        | md                                                         |
     *  | Hour 1-12          | h           | Numeric: minimum digits                                       | 1, 12                                                      |
     *  |                    | hh          | Numeric: 2 digits + zero padded                               | 01, 12                                                     |
     *  | Hour 0-23          | H           | Numeric: minimum digits                                       | 0, 23                                                      |
     *  |                    | HH          | Numeric: 2 digits + zero padded                               | 00, 23                                                     |
     *  | Minute             | m           | Numeric: minimum digits                                       | 8, 59                                                      |
     *  |                    | mm          | Numeric: 2 digits + zero padded                               | 08, 59                                                     |
     *  | Second             | s           | Numeric: minimum digits                                       | 0... 59                                                    |
     *  |                    | ss          | Numeric: 2 digits + zero padded                               | 00... 59                                                   |
     *  | Fractional seconds | S           | Numeric: 1 digit                                              | 0... 9                                                     |
     *  |                    | SS          | Numeric: 2 digits + zero padded                               | 00... 99                                                   |
     *  |                    | SSS         | Numeric: 3 digits + zero padded (= milliseconds)              | 000... 999                                                 |
     *  | Zone               | z, zz & zzz | Short specific non location format (fallback to O)            | GMT-8                                                      |
     *  |                    | zzzz        | Long specific non location format (fallback to OOOO)          | GMT-08:00                                                  |
     *  |                    | Z, ZZ & ZZZ | ISO8601 basic format                                          | -0800                                                      |
     *  |                    | ZZZZ        | Long localized GMT format                                     | GMT-8:00                                                   |
     *  |                    | ZZZZZ       | ISO8601 extended format + Z indicator for offset 0 (= XXXXX)  | -08:00                                                     |
     *  |                    | O, OO & OOO | Short localized GMT format                                    | GMT-8                                                      |
     *  |                    | OOOO        | Long localized GMT format                                     | GMT-08:00                                                  |
     *
     * Note that timezone correction is not applied to an ISO string that has no time component, such as "2016-09-19"
     *
     * ### Format examples
     *
     * These examples transform a date into various formats,
     * assuming that `dateObj` is a JavaScript `Date` object for
     * year: 2015, month: 6, day: 15, hour: 21, minute: 43, second: 11,
     * given in the local time for the `en-US` locale.
     *
     * ```
     * {{ dateObj | date }}               // output is 'Jun 15, 2015'
     * {{ dateObj | date:'medium' }}      // output is 'Jun 15, 2015, 9:43:11 PM'
     * {{ dateObj | date:'shortTime' }}   // output is '9:43 PM'
     * {{ dateObj | date:'mm:ss' }}       // output is '43:11'
     * ```
     *
     * ### Usage example
     *
     * The following component uses a date pipe to display the current date in different formats.
     *
     * ```
     * \@Component({
     *  selector: 'date-pipe',
     *  template: `<div>
     *    <p>Today is {{today | date}}</p>
     *    <p>Or if you prefer, {{today | date:'fullDate'}}</p>
     *    <p>The time is {{today | date:'h:mm a z'}}</p>
     *  </div>`
     * })
     * // Get the current date and time as a date-time value.
     * export class DatePipeComponent {
     *   today: number = Date.now();
     * }
     * ```
     *
     * \@publicApi
     */
    // clang-format on
    class DatePipe {
        /**
         * @param {?} locale
         */
        constructor(locale) {
            this.locale = locale;
        }
        /**
         * @param {?} value The date expression: a `Date` object,  a number
         * (milliseconds since UTC epoch), or an ISO string (https://www.w3.org/TR/NOTE-datetime).
         * @param {?=} format The date/time components to include, using predefined options or a
         * custom format string.
         * @param {?=} timezone A timezone offset (such as `'+0430'`), or a standard
         * UTC/GMT or continental US timezone abbreviation.
         * When not supplied, uses the end-user's local system timezone.
         * @param {?=} locale A locale code for the locale format rules to use.
         * When not supplied, uses the value of `LOCALE_ID`, which is `en-US` by default.
         * See [Setting your app locale](guide/i18n#setting-up-the-locale-of-your-app).
         * @return {?} A date string in the desired format.
         */
        transform(value, format = 'mediumDate', timezone, locale) {
            if (value == null || value === '' || value !== value)
                return null;
            try {
                return formatDate(value, format, locale || this.locale, timezone);
            }
            catch (error) {
                throw invalidPipeArgumentError(DatePipe, error.message);
            }
        }
    }
    DatePipe.ɵfac = function DatePipe_Factory(t) { return new (t || DatePipe)(ɵɵdirectiveInject(LOCALE_ID$1)); };
    DatePipe.ɵpipe = ɵɵdefinePipe({ name: "date", type: DatePipe, pure: true });
    /** @nocollapse */
    DatePipe.ctorParameters = () => [
        { type: String, decorators: [{ type: Inject, args: [LOCALE_ID$1,] }] }
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/src/pipes/i18n_plural_pipe.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const _INTERPOLATION_REGEXP = /#/g;
    /**
     * \@ngModule CommonModule
     * \@description
     *
     * Maps a value to a string that pluralizes the value according to locale rules.
     *
     * \@usageNotes
     *
     * ### Example
     *
     * {\@example common/pipes/ts/i18n_pipe.ts region='I18nPluralPipeComponent'}
     *
     * \@publicApi
     */
    class I18nPluralPipe {
        /**
         * @param {?} _localization
         */
        constructor(_localization) {
            this._localization = _localization;
        }
        /**
         * @param {?} value the number to be formatted
         * @param {?} pluralMap an object that mimics the ICU format, see
         * http://userguide.icu-project.org/formatparse/messages.
         * @param {?=} locale a `string` defining the locale to use (uses the current {\@link LOCALE_ID} by
         * default).
         * @return {?}
         */
        transform(value, pluralMap, locale) {
            if (value == null)
                return '';
            if (typeof pluralMap !== 'object' || pluralMap === null) {
                throw invalidPipeArgumentError(I18nPluralPipe, pluralMap);
            }
            /** @type {?} */
            const key = getPluralCategory(value, Object.keys(pluralMap), this._localization, locale);
            return pluralMap[key].replace(_INTERPOLATION_REGEXP, value.toString());
        }
    }
    I18nPluralPipe.ɵfac = function I18nPluralPipe_Factory(t) { return new (t || I18nPluralPipe)(ɵɵdirectiveInject(NgLocalization)); };
    I18nPluralPipe.ɵpipe = ɵɵdefinePipe({ name: "i18nPlural", type: I18nPluralPipe, pure: true });
    /** @nocollapse */
    I18nPluralPipe.ctorParameters = () => [
        { type: NgLocalization }
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/src/pipes/i18n_select_pipe.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * \@ngModule CommonModule
     * \@description
     *
     * Generic selector that displays the string that matches the current value.
     *
     * If none of the keys of the `mapping` match the `value`, then the content
     * of the `other` key is returned when present, otherwise an empty string is returned.
     *
     * \@usageNotes
     *
     * ### Example
     *
     * {\@example common/pipes/ts/i18n_pipe.ts region='I18nSelectPipeComponent'}
     *
     * \@publicApi
     */
    class I18nSelectPipe {
        /**
         * @param {?} value a string to be internationalized.
         * @param {?} mapping an object that indicates the text that should be displayed
         * for different values of the provided `value`.
         * @return {?}
         */
        transform(value, mapping) {
            if (value == null)
                return '';
            if (typeof mapping !== 'object' || typeof value !== 'string') {
                throw invalidPipeArgumentError(I18nSelectPipe, mapping);
            }
            if (mapping.hasOwnProperty(value)) {
                return mapping[value];
            }
            if (mapping.hasOwnProperty('other')) {
                return mapping['other'];
            }
            return '';
        }
    }
    I18nSelectPipe.ɵfac = function I18nSelectPipe_Factory(t) { return new (t || I18nSelectPipe)(); };
    I18nSelectPipe.ɵpipe = ɵɵdefinePipe({ name: "i18nSelect", type: I18nSelectPipe, pure: true });

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/src/pipes/json_pipe.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * \@ngModule CommonModule
     * \@description
     *
     * Converts a value into its JSON-format representation.  Useful for debugging.
     *
     * \@usageNotes
     *
     * The following component uses a JSON pipe to convert an object
     * to JSON format, and displays the string in both formats for comparison.
     *
     * {\@example common/pipes/ts/json_pipe.ts region='JsonPipe'}
     *
     * \@publicApi
     */
    class JsonPipe {
        /**
         * @param {?} value A value of any type to convert into a JSON-format string.
         * @return {?}
         */
        transform(value) {
            return JSON.stringify(value, null, 2);
        }
    }
    JsonPipe.ɵfac = function JsonPipe_Factory(t) { return new (t || JsonPipe)(); };
    JsonPipe.ɵpipe = ɵɵdefinePipe({ name: "json", type: JsonPipe, pure: false });

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/src/pipes/keyvalue_pipe.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @template K, V
     * @param {?} key
     * @param {?} value
     * @return {?}
     */
    function makeKeyValuePair(key, value) {
        return { key: key, value: value };
    }
    /**
     * \@ngModule CommonModule
     * \@description
     *
     * Transforms Object or Map into an array of key value pairs.
     *
     * The output array will be ordered by keys.
     * By default the comparator will be by Unicode point value.
     * You can optionally pass a compareFn if your keys are complex types.
     *
     * \@usageNotes
     * ### Examples
     *
     * This examples show how an Object or a Map can be iterated by ngFor with the use of this keyvalue
     * pipe.
     *
     * {\@example common/pipes/ts/keyvalue_pipe.ts region='KeyValuePipe'}
     *
     * \@publicApi
     */
    class KeyValuePipe {
        /**
         * @param {?} differs
         */
        constructor(differs) {
            this.differs = differs;
            this.keyValues = [];
        }
        /**
         * @template K, V
         * @param {?} input
         * @param {?=} compareFn
         * @return {?}
         */
        transform(input, compareFn = defaultComparator) {
            if (!input || (!(input instanceof Map) && typeof input !== 'object')) {
                return null;
            }
            if (!this.differ) {
                // make a differ for whatever type we've been passed in
                this.differ = this.differs.find(input).create();
            }
            /** @type {?} */
            const differChanges = this.differ.diff((/** @type {?} */ (input)));
            if (differChanges) {
                this.keyValues = [];
                differChanges.forEachItem((/**
                 * @param {?} r
                 * @return {?}
                 */
                (r) => {
                    this.keyValues.push(makeKeyValuePair(r.key, (/** @type {?} */ (r.currentValue))));
                }));
                this.keyValues.sort(compareFn);
            }
            return this.keyValues;
        }
    }
    KeyValuePipe.ɵfac = function KeyValuePipe_Factory(t) { return new (t || KeyValuePipe)(ɵɵdirectiveInject(KeyValueDiffers)); };
    KeyValuePipe.ɵpipe = ɵɵdefinePipe({ name: "keyvalue", type: KeyValuePipe, pure: false });
    /** @nocollapse */
    KeyValuePipe.ctorParameters = () => [
        { type: KeyValueDiffers }
    ];
    /**
     * @template K, V
     * @param {?} keyValueA
     * @param {?} keyValueB
     * @return {?}
     */
    function defaultComparator(keyValueA, keyValueB) {
        /** @type {?} */
        const a = keyValueA.key;
        /** @type {?} */
        const b = keyValueB.key;
        // if same exit with 0;
        if (a === b)
            return 0;
        // make sure that undefined are at the end of the sort.
        if (a === undefined)
            return 1;
        if (b === undefined)
            return -1;
        // make sure that nulls are at the end of the sort.
        if (a === null)
            return 1;
        if (b === null)
            return -1;
        if (typeof a == 'string' && typeof b == 'string') {
            return a < b ? -1 : 1;
        }
        if (typeof a == 'number' && typeof b == 'number') {
            return a - b;
        }
        if (typeof a == 'boolean' && typeof b == 'boolean') {
            return a < b ? -1 : 1;
        }
        // `a` and `b` are of different types. Compare their string values.
        /** @type {?} */
        const aString = String(a);
        /** @type {?} */
        const bString = String(b);
        return aString == bString ? 0 : aString < bString ? -1 : 1;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/src/pipes/number_pipe.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * \@ngModule CommonModule
     * \@description
     *
     * Transforms a number into a string,
     * formatted according to locale rules that determine group sizing and
     * separator, decimal-point character, and other locale-specific
     * configurations.
     *
     * If no parameters are specified, the function rounds off to the nearest value using this
     * [rounding method](https://en.wikibooks.org/wiki/Arithmetic/Rounding).
     * The behavior differs from that of the JavaScript ```Math.round()``` function.
     * In the following case for example, the pipe rounds down where
     * ```Math.round()``` rounds up:
     *
     * ```html
     * -2.5 | number:'1.0-0'
     * > -3
     * Math.round(-2.5)
     * > -2
     * ```
     *
     * @see `formatNumber()`
     *
     * \@usageNotes
     * The following code shows how the pipe transforms numbers
     * into text strings, according to various format specifications,
     * where the caller's default locale is `en-US`.
     *
     * ### Example
     *
     * <code-example path="common/pipes/ts/number_pipe.ts" region='NumberPipe'></code-example>
     *
     * \@publicApi
     */
    class DecimalPipe {
        /**
         * @param {?} _locale
         */
        constructor(_locale) {
            this._locale = _locale;
        }
        /**
         * @param {?} value The number to be formatted.
         * @param {?=} digitsInfo Decimal representation options, specified by a string
         * in the following format:<br>
         * <code>{minIntegerDigits}.{minFractionDigits}-{maxFractionDigits}</code>.
         *   - `minIntegerDigits`: The minimum number of integer digits before the decimal point.
         * Default is `1`.
         *   - `minFractionDigits`: The minimum number of digits after the decimal point.
         * Default is `0`.
         *   - `maxFractionDigits`: The maximum number of digits after the decimal point.
         * Default is `3`.
         * @param {?=} locale A locale code for the locale format rules to use.
         * When not supplied, uses the value of `LOCALE_ID`, which is `en-US` by default.
         * See [Setting your app locale](guide/i18n#setting-up-the-locale-of-your-app).
         * @return {?}
         */
        transform(value, digitsInfo, locale) {
            if (isEmpty(value))
                return null;
            locale = locale || this._locale;
            try {
                /** @type {?} */
                const num = strToNumber(value);
                return formatNumber(num, locale, digitsInfo);
            }
            catch (error) {
                throw invalidPipeArgumentError(DecimalPipe, error.message);
            }
        }
    }
    DecimalPipe.ɵfac = function DecimalPipe_Factory(t) { return new (t || DecimalPipe)(ɵɵdirectiveInject(LOCALE_ID$1)); };
    DecimalPipe.ɵpipe = ɵɵdefinePipe({ name: "number", type: DecimalPipe, pure: true });
    /** @nocollapse */
    DecimalPipe.ctorParameters = () => [
        { type: String, decorators: [{ type: Inject, args: [LOCALE_ID$1,] }] }
    ];
    /**
     * \@ngModule CommonModule
     * \@description
     *
     * Transforms a number to a percentage
     * string, formatted according to locale rules that determine group sizing and
     * separator, decimal-point character, and other locale-specific
     * configurations.
     *
     * @see `formatPercent()`
     *
     * \@usageNotes
     * The following code shows how the pipe transforms numbers
     * into text strings, according to various format specifications,
     * where the caller's default locale is `en-US`.
     *
     * <code-example path="common/pipes/ts/percent_pipe.ts" region='PercentPipe'></code-example>
     *
     * \@publicApi
     */
    class PercentPipe {
        /**
         * @param {?} _locale
         */
        constructor(_locale) {
            this._locale = _locale;
        }
        /**
         *
         * @param {?} value The number to be formatted as a percentage.
         * @param {?=} digitsInfo Decimal representation options, specified by a string
         * in the following format:<br>
         * <code>{minIntegerDigits}.{minFractionDigits}-{maxFractionDigits}</code>.
         *   - `minIntegerDigits`: The minimum number of integer digits before the decimal point.
         * Default is `1`.
         *   - `minFractionDigits`: The minimum number of digits after the decimal point.
         * Default is `0`.
         *   - `maxFractionDigits`: The maximum number of digits after the decimal point.
         * Default is `0`.
         * @param {?=} locale A locale code for the locale format rules to use.
         * When not supplied, uses the value of `LOCALE_ID`, which is `en-US` by default.
         * See [Setting your app locale](guide/i18n#setting-up-the-locale-of-your-app).
         * @return {?}
         */
        transform(value, digitsInfo, locale) {
            if (isEmpty(value))
                return null;
            locale = locale || this._locale;
            try {
                /** @type {?} */
                const num = strToNumber(value);
                return formatPercent(num, locale, digitsInfo);
            }
            catch (error) {
                throw invalidPipeArgumentError(PercentPipe, error.message);
            }
        }
    }
    PercentPipe.ɵfac = function PercentPipe_Factory(t) { return new (t || PercentPipe)(ɵɵdirectiveInject(LOCALE_ID$1)); };
    PercentPipe.ɵpipe = ɵɵdefinePipe({ name: "percent", type: PercentPipe, pure: true });
    /** @nocollapse */
    PercentPipe.ctorParameters = () => [
        { type: String, decorators: [{ type: Inject, args: [LOCALE_ID$1,] }] }
    ];
    /**
     * \@ngModule CommonModule
     * \@description
     *
     * Transforms a number to a currency string, formatted according to locale rules
     * that determine group sizing and separator, decimal-point character,
     * and other locale-specific configurations.
     *
     * {\@a currency-code-deprecation}
     * <div class="alert is-helpful">
     *
     * **Deprecation notice:**
     *
     * The default currency code is currently always `USD` but this is deprecated from v9.
     *
     * **In v11 the default currency code will be taken from the current locale identified by
     * the `LOCAL_ID` token. See the [i18n guide](guide/i18n#setting-up-the-locale-of-your-app) for
     * more information.**
     *
     * If you need the previous behavior then set it by creating a `DEFAULT_CURRENCY_CODE` provider in
     * your application `NgModule`:
     *
     * ```ts
     * {provide: DEFAULT_CURRENCY_CODE, useValue: 'USD'}
     * ```
     *
     * </div>
     *
     * @see `getCurrencySymbol()`
     * @see `formatCurrency()`
     *
     * \@usageNotes
     * The following code shows how the pipe transforms numbers
     * into text strings, according to various format specifications,
     * where the caller's default locale is `en-US`.
     *
     * <code-example path="common/pipes/ts/currency_pipe.ts" region='CurrencyPipe'></code-example>
     *
     * \@publicApi
     */
    class CurrencyPipe {
        /**
         * @param {?} _locale
         * @param {?=} _defaultCurrencyCode
         */
        constructor(_locale, _defaultCurrencyCode = 'USD') {
            this._locale = _locale;
            this._defaultCurrencyCode = _defaultCurrencyCode;
        }
        /**
         *
         * @param {?} value The number to be formatted as currency.
         * @param {?=} currencyCode The [ISO 4217](https://en.wikipedia.org/wiki/ISO_4217) currency code,
         * such as `USD` for the US dollar and `EUR` for the euro. The default currency code can be
         * configured using the `DEFAULT_CURRENCY_CODE` injection token.
         * @param {?=} display The format for the currency indicator. One of the following:
         *   - `code`: Show the code (such as `USD`).
         *   - `symbol`(default): Show the symbol (such as `$`).
         *   - `symbol-narrow`: Use the narrow symbol for locales that have two symbols for their
         * currency.
         * For example, the Canadian dollar CAD has the symbol `CA$` and the symbol-narrow `$`. If the
         * locale has no narrow symbol, uses the standard symbol for the locale.
         *   - String: Use the given string value instead of a code or a symbol.
         * For example, an empty string will suppress the currency & symbol.
         *   - Boolean (marked deprecated in v5): `true` for symbol and false for `code`.
         *
         * @param {?=} digitsInfo Decimal representation options, specified by a string
         * in the following format:<br>
         * <code>{minIntegerDigits}.{minFractionDigits}-{maxFractionDigits}</code>.
         *   - `minIntegerDigits`: The minimum number of integer digits before the decimal point.
         * Default is `1`.
         *   - `minFractionDigits`: The minimum number of digits after the decimal point.
         * Default is `2`.
         *   - `maxFractionDigits`: The maximum number of digits after the decimal point.
         * Default is `2`.
         * If not provided, the number will be formatted with the proper amount of digits,
         * depending on what the [ISO 4217](https://en.wikipedia.org/wiki/ISO_4217) specifies.
         * For example, the Canadian dollar has 2 digits, whereas the Chilean peso has none.
         * @param {?=} locale A locale code for the locale format rules to use.
         * When not supplied, uses the value of `LOCALE_ID`, which is `en-US` by default.
         * See [Setting your app locale](guide/i18n#setting-up-the-locale-of-your-app).
         * @return {?}
         */
        transform(value, currencyCode, display = 'symbol', digitsInfo, locale) {
            if (isEmpty(value))
                return null;
            locale = locale || this._locale;
            if (typeof display === 'boolean') {
                if ((/** @type {?} */ (console)) && (/** @type {?} */ (console.warn))) {
                    console.warn(`Warning: the currency pipe has been changed in Angular v5. The symbolDisplay option (third parameter) is now a string instead of a boolean. The accepted values are "code", "symbol" or "symbol-narrow".`);
                }
                display = display ? 'symbol' : 'code';
            }
            /** @type {?} */
            let currency = currencyCode || this._defaultCurrencyCode;
            if (display !== 'code') {
                if (display === 'symbol' || display === 'symbol-narrow') {
                    currency = getCurrencySymbol(currency, display === 'symbol' ? 'wide' : 'narrow', locale);
                }
                else {
                    currency = display;
                }
            }
            try {
                /** @type {?} */
                const num = strToNumber(value);
                return formatCurrency(num, locale, currency, currencyCode, digitsInfo);
            }
            catch (error) {
                throw invalidPipeArgumentError(CurrencyPipe, error.message);
            }
        }
    }
    CurrencyPipe.ɵfac = function CurrencyPipe_Factory(t) { return new (t || CurrencyPipe)(ɵɵdirectiveInject(LOCALE_ID$1), ɵɵdirectiveInject(DEFAULT_CURRENCY_CODE)); };
    CurrencyPipe.ɵpipe = ɵɵdefinePipe({ name: "currency", type: CurrencyPipe, pure: true });
    /** @nocollapse */
    CurrencyPipe.ctorParameters = () => [
        { type: String, decorators: [{ type: Inject, args: [LOCALE_ID$1,] }] },
        { type: String, decorators: [{ type: Inject, args: [DEFAULT_CURRENCY_CODE,] }] }
    ];
    /**
     * @param {?} value
     * @return {?}
     */
    function isEmpty(value) {
        return value == null || value === '' || value !== value;
    }
    /**
     * Transforms a string into a number (if needed).
     * @param {?} value
     * @return {?}
     */
    function strToNumber(value) {
        // Convert strings to numbers
        if (typeof value === 'string' && !isNaN(Number(value) - parseFloat(value))) {
            return Number(value);
        }
        if (typeof value !== 'number') {
            throw new Error(`${value} is not a number`);
        }
        return value;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/src/pipes/slice_pipe.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * \@ngModule CommonModule
     * \@description
     *
     * Creates a new `Array` or `String` containing a subset (slice) of the elements.
     *
     * \@usageNotes
     *
     * All behavior is based on the expected behavior of the JavaScript API `Array.prototype.slice()`
     * and `String.prototype.slice()`.
     *
     * When operating on an `Array`, the returned `Array` is always a copy even when all
     * the elements are being returned.
     *
     * When operating on a blank value, the pipe returns the blank value.
     *
     * ### List Example
     *
     * This `ngFor` example:
     *
     * {\@example common/pipes/ts/slice_pipe.ts region='SlicePipe_list'}
     *
     * produces the following:
     *
     * ```html
     * <li>b</li>
     * <li>c</li>
     * ```
     *
     * ### String Examples
     *
     * {\@example common/pipes/ts/slice_pipe.ts region='SlicePipe_string'}
     *
     * \@publicApi
     */
    class SlicePipe {
        /**
         * @param {?} value
         * @param {?} start
         * @param {?=} end
         * @return {?}
         */
        transform(value, start, end) {
            if (value == null)
                return value;
            if (!this.supports(value)) {
                throw invalidPipeArgumentError(SlicePipe, value);
            }
            return value.slice(start, end);
        }
        /**
         * @private
         * @param {?} obj
         * @return {?}
         */
        supports(obj) {
            return typeof obj === 'string' || Array.isArray(obj);
        }
    }
    SlicePipe.ɵfac = function SlicePipe_Factory(t) { return new (t || SlicePipe)(); };
    SlicePipe.ɵpipe = ɵɵdefinePipe({ name: "slice", type: SlicePipe, pure: false });

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/src/common_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    // Note: This does not contain the location providers,
    // as they need some platform specific implementations to work.
    /**
     * Exports all the basic Angular directives and pipes,
     * such as `NgIf`, `NgForOf`, `DecimalPipe`, and so on.
     * Re-exported by `BrowserModule`, which is included automatically in the root
     * `AppModule` when you create a new app with the CLI `new` command.
     *
     * * The `providers` options configure the NgModule's injector to provide
     * localization dependencies to members.
     * * The `exports` options make the declared directives and pipes available for import
     * by other NgModules.
     *
     * \@publicApi
     */
    class CommonModule {
    }
    CommonModule.ɵmod = ɵɵdefineNgModule({ type: CommonModule });
    CommonModule.ɵinj = ɵɵdefineInjector({ factory: function CommonModule_Factory(t) { return new (t || CommonModule)(); }, providers: [
            { provide: NgLocalization, useClass: NgLocaleLocalization },
        ] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(CommonModule, { declarations: [NgClass,
            NgComponentOutlet,
            NgForOf,
            NgIf,
            NgTemplateOutlet,
            NgStyle,
            NgSwitch,
            NgSwitchCase,
            NgSwitchDefault,
            NgPlural,
            NgPluralCase,
            AsyncPipe,
            UpperCasePipe,
            LowerCasePipe,
            JsonPipe,
            SlicePipe,
            DecimalPipe,
            PercentPipe,
            TitleCasePipe,
            CurrencyPipe,
            DatePipe,
            I18nPluralPipe,
            I18nSelectPipe,
            KeyValuePipe], exports: [NgClass,
            NgComponentOutlet,
            NgForOf,
            NgIf,
            NgTemplateOutlet,
            NgStyle,
            NgSwitch,
            NgSwitchCase,
            NgSwitchDefault,
            NgPlural,
            NgPluralCase,
            AsyncPipe,
            UpperCasePipe,
            LowerCasePipe,
            JsonPipe,
            SlicePipe,
            DecimalPipe,
            PercentPipe,
            TitleCasePipe,
            CurrencyPipe,
            DatePipe,
            I18nPluralPipe,
            I18nSelectPipe,
            KeyValuePipe] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/src/platform_id.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /** @type {?} */
    const PLATFORM_BROWSER_ID = 'browser';
    /**
     * Returns whether a platform id represents a browser platform.
     * \@publicApi
     * @param {?} platformId
     * @return {?}
     */
    function isPlatformBrowser(platformId) {
        return platformId === PLATFORM_BROWSER_ID;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/src/version.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * \@publicApi
     * @type {?}
     */
    const VERSION$1 = new Version('9.1.12');

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/src/viewport_scroller.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Defines a scroll position manager. Implemented by `BrowserViewportScroller`.
     *
     * \@publicApi
     * @abstract
     */
    class ViewportScroller {
    }
    // De-sugared tree-shakable injection
    // See #23917
    /** @nocollapse */
    ViewportScroller.ɵprov = ɵɵdefineInjectable({
        token: ViewportScroller,
        providedIn: 'root',
        factory: (/**
         * @return {?}
         */
        () => new BrowserViewportScroller(ɵɵinject(DOCUMENT$1), window, ɵɵinject(ErrorHandler)))
    });
    /**
     * Manages the scroll position for a browser window.
     */
    class BrowserViewportScroller {
        /**
         * @param {?} document
         * @param {?} window
         * @param {?} errorHandler
         */
        constructor(document, window, errorHandler) {
            this.document = document;
            this.window = window;
            this.errorHandler = errorHandler;
            this.offset = (/**
             * @return {?}
             */
            () => [0, 0]);
        }
        /**
         * Configures the top offset used when scrolling to an anchor.
         * @param {?} offset A position in screen coordinates (a tuple with x and y values)
         * or a function that returns the top offset position.
         *
         * @return {?}
         */
        setOffset(offset) {
            if (Array.isArray(offset)) {
                this.offset = (/**
                 * @return {?}
                 */
                () => offset);
            }
            else {
                this.offset = offset;
            }
        }
        /**
         * Retrieves the current scroll position.
         * @return {?} The position in screen coordinates.
         */
        getScrollPosition() {
            if (this.supportScrollRestoration()) {
                return [this.window.scrollX, this.window.scrollY];
            }
            else {
                return [0, 0];
            }
        }
        /**
         * Sets the scroll position.
         * @param {?} position The new position in screen coordinates.
         * @return {?}
         */
        scrollToPosition(position) {
            if (this.supportScrollRestoration()) {
                this.window.scrollTo(position[0], position[1]);
            }
        }
        /**
         * Scrolls to an anchor element.
         * @param {?} anchor The ID of the anchor element.
         * @return {?}
         */
        scrollToAnchor(anchor) {
            if (this.supportScrollRestoration()) {
                // Escape anything passed to `querySelector` as it can throw errors and stop the application
                // from working if invalid values are passed.
                if (this.window.CSS && this.window.CSS.escape) {
                    anchor = this.window.CSS.escape(anchor);
                }
                else {
                    anchor = anchor.replace(/(\"|\'\ |:|\.|\[|\]|,|=)/g, '\\$1');
                }
                try {
                    /** @type {?} */
                    const elSelectedById = this.document.querySelector(`#${anchor}`);
                    if (elSelectedById) {
                        this.scrollToElement(elSelectedById);
                        return;
                    }
                    /** @type {?} */
                    const elSelectedByName = this.document.querySelector(`[name='${anchor}']`);
                    if (elSelectedByName) {
                        this.scrollToElement(elSelectedByName);
                        return;
                    }
                }
                catch (e) {
                    this.errorHandler.handleError(e);
                }
            }
        }
        /**
         * Disables automatic scroll restoration provided by the browser.
         * @param {?} scrollRestoration
         * @return {?}
         */
        setHistoryScrollRestoration(scrollRestoration) {
            if (this.supportScrollRestoration()) {
                /** @type {?} */
                const history = this.window.history;
                if (history && history.scrollRestoration) {
                    history.scrollRestoration = scrollRestoration;
                }
            }
        }
        /**
         * @private
         * @param {?} el
         * @return {?}
         */
        scrollToElement(el) {
            /** @type {?} */
            const rect = el.getBoundingClientRect();
            /** @type {?} */
            const left = rect.left + this.window.pageXOffset;
            /** @type {?} */
            const top = rect.top + this.window.pageYOffset;
            /** @type {?} */
            const offset = this.offset();
            this.window.scrollTo(left - offset[0], top - offset[1]);
        }
        /**
         * We only support scroll restoration when we can get a hold of window.
         * This means that we do not support this behavior when running in a web worker.
         *
         * Lifting this restriction right now would require more changes in the dom adapter.
         * Since webworkers aren't widely used, we will lift it once RouterScroller is
         * battle-tested.
         * @private
         * @return {?}
         */
        supportScrollRestoration() {
            try {
                return !!this.window && !!this.window.scrollTo;
            }
            catch (_a) {
                return false;
            }
        }
    }

    /**
     * @license Angular v9.1.12
     * (c) 2010-2020 Google LLC. https://angular.io/
     * License: MIT
     */

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/platform-browser/src/browser/generic_browser_adapter.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Provides DOM operations in any browser environment.
     *
     * \@security Tread carefully! Interacting with the DOM directly is dangerous and
     * can introduce XSS risks.
     * @abstract
     */
    class GenericBrowserDomAdapter extends DomAdapter {
        constructor() {
            super();
        }
        /**
         * @return {?}
         */
        supportsDOMEvents() {
            return true;
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/platform-browser/src/browser/browser_adapter.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    const ɵ0$5 = /**
     * @return {?}
     */
    () => {
        if (_global['Node']) {
            return _global['Node'].prototype.contains || (/**
             * @this {?}
             * @param {?} node
             * @return {?}
             */
            function (node) {
                return !!(this.compareDocumentPosition(node) & 16);
            });
        }
        return (/** @type {?} */ (undefined));
    };
    /** @type {?} */
    const nodeContains = ((ɵ0$5))();
    /**
     * A `DomAdapter` powered by full browser DOM APIs.
     *
     * \@security Tread carefully! Interacting with the DOM directly is dangerous and
     * can introduce XSS risks.
     */
    /* tslint:disable:requireParameterType no-console */
    class BrowserDomAdapter extends GenericBrowserDomAdapter {
        /**
         * @return {?}
         */
        static makeCurrent() {
            setRootDomAdapter(new BrowserDomAdapter());
        }
        /**
         * @param {?} el
         * @param {?} name
         * @return {?}
         */
        getProperty(el, name) {
            return ((/** @type {?} */ (el)))[name];
        }
        /**
         * @param {?} error
         * @return {?}
         */
        log(error) {
            if (window.console) {
                window.console.log && window.console.log(error);
            }
        }
        /**
         * @param {?} error
         * @return {?}
         */
        logGroup(error) {
            if (window.console) {
                window.console.group && window.console.group(error);
            }
        }
        /**
         * @return {?}
         */
        logGroupEnd() {
            if (window.console) {
                window.console.groupEnd && window.console.groupEnd();
            }
        }
        /**
         * @param {?} el
         * @param {?} evt
         * @param {?} listener
         * @return {?}
         */
        onAndCancel(el, evt, listener) {
            el.addEventListener(evt, listener, false);
            // Needed to follow Dart's subscription semantic, until fix of
            // https://code.google.com/p/dart/issues/detail?id=17406
            return (/**
             * @return {?}
             */
            () => {
                el.removeEventListener(evt, listener, false);
            });
        }
        /**
         * @param {?} el
         * @param {?} evt
         * @return {?}
         */
        dispatchEvent(el, evt) {
            el.dispatchEvent(evt);
        }
        /**
         * @param {?} node
         * @return {?}
         */
        remove(node) {
            if (node.parentNode) {
                node.parentNode.removeChild(node);
            }
            return node;
        }
        /**
         * @param {?} el
         * @return {?}
         */
        getValue(el) {
            return el.value;
        }
        /**
         * @param {?} tagName
         * @param {?=} doc
         * @return {?}
         */
        createElement(tagName, doc) {
            doc = doc || this.getDefaultDocument();
            return doc.createElement(tagName);
        }
        /**
         * @return {?}
         */
        createHtmlDocument() {
            return document.implementation.createHTMLDocument('fakeTitle');
        }
        /**
         * @return {?}
         */
        getDefaultDocument() {
            return document;
        }
        /**
         * @param {?} node
         * @return {?}
         */
        isElementNode(node) {
            return node.nodeType === Node.ELEMENT_NODE;
        }
        /**
         * @param {?} node
         * @return {?}
         */
        isShadowRoot(node) {
            return node instanceof DocumentFragment;
        }
        /**
         * @param {?} doc
         * @param {?} target
         * @return {?}
         */
        getGlobalEventTarget(doc, target) {
            if (target === 'window') {
                return window;
            }
            if (target === 'document') {
                return doc;
            }
            if (target === 'body') {
                return doc.body;
            }
            return null;
        }
        /**
         * @return {?}
         */
        getHistory() {
            return window.history;
        }
        /**
         * @return {?}
         */
        getLocation() {
            return window.location;
        }
        /**
         * @param {?} doc
         * @return {?}
         */
        getBaseHref(doc) {
            /** @type {?} */
            const href = getBaseElementHref();
            return href == null ? null : relativePath(href);
        }
        /**
         * @return {?}
         */
        resetBaseElement() {
            baseElement = null;
        }
        /**
         * @return {?}
         */
        getUserAgent() {
            return window.navigator.userAgent;
        }
        /**
         * @return {?}
         */
        performanceNow() {
            // performance.now() is not available in all browsers, see
            // http://caniuse.com/#search=performance.now
            return window.performance && window.performance.now ? window.performance.now() :
                new Date().getTime();
        }
        /**
         * @return {?}
         */
        supportsCookies() {
            return true;
        }
        /**
         * @param {?} name
         * @return {?}
         */
        getCookie(name) {
            return parseCookieValue(document.cookie, name);
        }
    }
    /** @type {?} */
    let baseElement = null;
    /**
     * @return {?}
     */
    function getBaseElementHref() {
        if (!baseElement) {
            baseElement = (/** @type {?} */ (document.querySelector('base')));
            if (!baseElement) {
                return null;
            }
        }
        return baseElement.getAttribute('href');
    }
    // based on urlUtils.js in AngularJS 1
    /** @type {?} */
    let urlParsingNode;
    /**
     * @param {?} url
     * @return {?}
     */
    function relativePath(url) {
        if (!urlParsingNode) {
            urlParsingNode = document.createElement('a');
        }
        urlParsingNode.setAttribute('href', url);
        return (urlParsingNode.pathname.charAt(0) === '/') ? urlParsingNode.pathname :
            '/' + urlParsingNode.pathname;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/platform-browser/src/browser/server-transition.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * An id that identifies a particular application being bootstrapped, that should
     * match across the client/server boundary.
     * @type {?}
     */
    const TRANSITION_ID = new InjectionToken('TRANSITION_ID');
    /**
     * @param {?} transitionId
     * @param {?} document
     * @param {?} injector
     * @return {?}
     */
    function appInitializerFactory(transitionId, document, injector) {
        return (/**
         * @return {?}
         */
        () => {
            // Wait for all application initializers to be completed before removing the styles set by
            // the server.
            injector.get(ApplicationInitStatus).donePromise.then((/**
             * @return {?}
             */
            () => {
                /** @type {?} */
                const dom = getDOM();
                /** @type {?} */
                const styles = Array.prototype.slice.apply(document.querySelectorAll(`style[ng-transition]`));
                styles.filter((/**
                 * @param {?} el
                 * @return {?}
                 */
                el => el.getAttribute('ng-transition') === transitionId))
                    .forEach((/**
                 * @param {?} el
                 * @return {?}
                 */
                el => dom.remove(el)));
            }));
        });
    }
    /** @type {?} */
    const SERVER_TRANSITION_PROVIDERS = [
        {
            provide: APP_INITIALIZER,
            useFactory: appInitializerFactory,
            deps: [TRANSITION_ID, DOCUMENT$1, Injector],
            multi: true
        },
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/platform-browser/src/browser/testability.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class BrowserGetTestability {
        /**
         * @return {?}
         */
        static init() {
            setTestabilityGetter(new BrowserGetTestability());
        }
        /**
         * @param {?} registry
         * @return {?}
         */
        addToWindow(registry) {
            _global['getAngularTestability'] = (/**
             * @param {?} elem
             * @param {?=} findInAncestors
             * @return {?}
             */
            (elem, findInAncestors = true) => {
                /** @type {?} */
                const testability = registry.findTestabilityInTree(elem, findInAncestors);
                if (testability == null) {
                    throw new Error('Could not find testability for element.');
                }
                return testability;
            });
            _global['getAllAngularTestabilities'] = (/**
             * @return {?}
             */
            () => registry.getAllTestabilities());
            _global['getAllAngularRootElements'] = (/**
             * @return {?}
             */
            () => registry.getAllRootElements());
            /** @type {?} */
            const whenAllStable = (/**
             * @param {?} callback
             * @return {?}
             */
            (callback /** TODO #9100 */) => {
                /** @type {?} */
                const testabilities = _global['getAllAngularTestabilities']();
                /** @type {?} */
                let count = testabilities.length;
                /** @type {?} */
                let didWork = false;
                /** @type {?} */
                const decrement = (/**
                 * @param {?} didWork_
                 * @return {?}
                 */
                function (didWork_ /** TODO #9100 */) {
                    didWork = didWork || didWork_;
                    count--;
                    if (count == 0) {
                        callback(didWork);
                    }
                });
                testabilities.forEach((/**
                 * @param {?} testability
                 * @return {?}
                 */
                function (testability /** TODO #9100 */) {
                    testability.whenStable(decrement);
                }));
            });
            if (!_global['frameworkStabilizers']) {
                _global['frameworkStabilizers'] = [];
            }
            _global['frameworkStabilizers'].push(whenAllStable);
        }
        /**
         * @param {?} registry
         * @param {?} elem
         * @param {?} findInAncestors
         * @return {?}
         */
        findTestabilityInTree(registry, elem, findInAncestors) {
            if (elem == null) {
                return null;
            }
            /** @type {?} */
            const t = registry.getTestability(elem);
            if (t != null) {
                return t;
            }
            else if (!findInAncestors) {
                return null;
            }
            if (getDOM().isShadowRoot(elem)) {
                return this.findTestabilityInTree(registry, ((/** @type {?} */ (elem))).host, true);
            }
            return this.findTestabilityInTree(registry, elem.parentElement, true);
        }
    }
    /**
     * Exports the value under a given `name` in the global property `ng`. For example `ng.probe` if
     * `name` is `'probe'`.
     * @param {?} name Name under which it will be exported. Keep in mind this will be a property of the
     * global `ng` object.
     * @param {?} value The value to export.
     * @return {?}
     */
    function exportNgVar(name, value) {
        if (typeof COMPILED === 'undefined' || !COMPILED) {
            // Note: we can't export `ng` when using closure enhanced optimization as:
            // - closure declares globals itself for minified names, which sometimes clobber our `ng` global
            // - we can't declare a closure extern as the namespace `ng` is already used within Google
            //   for typings for angularJS (via `goog.provide('ng....')`).
            /** @type {?} */
            const ng = _global['ng'] = ((/** @type {?} */ (_global['ng']))) || {};
            ng[name] = value;
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/platform-browser/src/dom/debug/ng_probe.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    const ɵ0$1$1 = /**
     * @return {?}
     */
    () => ({
        'ApplicationRef': ApplicationRef,
        'NgZone': NgZone,
    });
    /** @type {?} */
    const CORE_TOKENS = ((ɵ0$1$1))();
    /** @type {?} */
    const INSPECT_GLOBAL_NAME = 'probe';
    /** @type {?} */
    const CORE_TOKENS_GLOBAL_NAME = 'coreTokens';
    /**
     * Returns a {\@link DebugElement} for the given native DOM element, or
     * null if the given native element does not have an Angular view associated
     * with it.
     * @param {?} element
     * @return {?}
     */
    function inspectNativeElementR2(element) {
        return getDebugNodeR2();
    }
    /**
     * @param {?} coreTokens
     * @return {?}
     */
    function _createNgProbeR2(coreTokens) {
        exportNgVar(INSPECT_GLOBAL_NAME, inspectNativeElementR2);
        exportNgVar(CORE_TOKENS_GLOBAL_NAME, Object.assign(Object.assign({}, CORE_TOKENS), _ngProbeTokensToMap(coreTokens || [])));
        return (/**
         * @return {?}
         */
        () => inspectNativeElementR2);
    }
    /**
     * @param {?} tokens
     * @return {?}
     */
    function _ngProbeTokensToMap(tokens) {
        return tokens.reduce((/**
         * @param {?} prev
         * @param {?} t
         * @return {?}
         */
        (prev, t) => (prev[t.name] = t.token, prev)), {});
    }
    /**
     * In Ivy, we don't support NgProbe because we have our own set of testing utilities
     * with more robust functionality.
     *
     * We shouldn't bring in NgProbe because it prevents DebugNode and friends from
     * tree-shaking properly.
     * @type {?}
     */
    const ELEMENT_PROBE_PROVIDERS__POST_R3__ = [];
    /**
     * Providers which support debugging Angular applications (e.g. via `ng.probe`).
     * @type {?}
     */
    const ELEMENT_PROBE_PROVIDERS__PRE_R3__ = [
        {
            provide: APP_INITIALIZER,
            useFactory: _createNgProbeR2,
            deps: [
                [NgProbeToken, new Optional()],
            ],
            multi: true,
        },
    ];
    /** @type {?} */
    const ELEMENT_PROBE_PROVIDERS = ELEMENT_PROBE_PROVIDERS__POST_R3__;

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/platform-browser/src/dom/events/event_manager.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * The injection token for the event-manager plug-in service.
     *
     * \@publicApi
     * @type {?}
     */
    const EVENT_MANAGER_PLUGINS = new InjectionToken('EventManagerPlugins');
    /**
     * An injectable service that provides event management for Angular
     * through a browser plug-in.
     *
     * \@publicApi
     */
    class EventManager {
        /**
         * Initializes an instance of the event-manager service.
         * @param {?} plugins
         * @param {?} _zone
         */
        constructor(plugins, _zone) {
            this._zone = _zone;
            this._eventNameToPlugin = new Map();
            plugins.forEach((/**
             * @template THIS
             * @this {THIS}
             * @param {?} p
             * @return {THIS}
             */
            p => p.manager = this));
            this._plugins = plugins.slice().reverse();
        }
        /**
         * Registers a handler for a specific element and event.
         *
         * @param {?} element The HTML element to receive event notifications.
         * @param {?} eventName The name of the event to listen for.
         * @param {?} handler A function to call when the notification occurs. Receives the
         * event object as an argument.
         * @return {?} A callback function that can be used to remove the handler.
         */
        addEventListener(element, eventName, handler) {
            /** @type {?} */
            const plugin = this._findPluginFor(eventName);
            return plugin.addEventListener(element, eventName, handler);
        }
        /**
         * Registers a global handler for an event in a target view.
         *
         * @param {?} target A target for global event notifications. One of "window", "document", or "body".
         * @param {?} eventName The name of the event to listen for.
         * @param {?} handler A function to call when the notification occurs. Receives the
         * event object as an argument.
         * @return {?} A callback function that can be used to remove the handler.
         */
        addGlobalEventListener(target, eventName, handler) {
            /** @type {?} */
            const plugin = this._findPluginFor(eventName);
            return plugin.addGlobalEventListener(target, eventName, handler);
        }
        /**
         * Retrieves the compilation zone in which event listeners are registered.
         * @return {?}
         */
        getZone() {
            return this._zone;
        }
        /**
         * \@internal
         * @param {?} eventName
         * @return {?}
         */
        _findPluginFor(eventName) {
            /** @type {?} */
            const plugin = this._eventNameToPlugin.get(eventName);
            if (plugin) {
                return plugin;
            }
            /** @type {?} */
            const plugins = this._plugins;
            for (let i = 0; i < plugins.length; i++) {
                /** @type {?} */
                const plugin = plugins[i];
                if (plugin.supports(eventName)) {
                    this._eventNameToPlugin.set(eventName, plugin);
                    return plugin;
                }
            }
            throw new Error(`No event manager plugin found for event ${eventName}`);
        }
    }
    EventManager.ɵfac = function EventManager_Factory(t) { return new (t || EventManager)(ɵɵinject(EVENT_MANAGER_PLUGINS), ɵɵinject(NgZone)); };
    EventManager.ɵprov = ɵɵdefineInjectable({ token: EventManager, factory: EventManager.ɵfac });
    /** @nocollapse */
    EventManager.ctorParameters = () => [
        { type: Array, decorators: [{ type: Inject, args: [EVENT_MANAGER_PLUGINS,] }] },
        { type: NgZone }
    ];
    /**
     * @abstract
     */
    class EventManagerPlugin {
        /**
         * @param {?} _doc
         */
        constructor(_doc) {
            this._doc = _doc;
        }
        /**
         * @param {?} element
         * @param {?} eventName
         * @param {?} handler
         * @return {?}
         */
        addGlobalEventListener(element, eventName, handler) {
            /** @type {?} */
            const target = getDOM().getGlobalEventTarget(this._doc, element);
            if (!target) {
                throw new Error(`Unsupported event target ${target} for event ${eventName}`);
            }
            return this.addEventListener(target, eventName, handler);
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/platform-browser/src/dom/shared_styles_host.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class SharedStylesHost {
        constructor() {
            /**
             * \@internal
             */
            this._stylesSet = new Set();
        }
        /**
         * @param {?} styles
         * @return {?}
         */
        addStyles(styles) {
            /** @type {?} */
            const additions = new Set();
            styles.forEach((/**
             * @param {?} style
             * @return {?}
             */
            style => {
                if (!this._stylesSet.has(style)) {
                    this._stylesSet.add(style);
                    additions.add(style);
                }
            }));
            this.onStylesAdded(additions);
        }
        /**
         * @param {?} additions
         * @return {?}
         */
        onStylesAdded(additions) { }
        /**
         * @return {?}
         */
        getAllStyles() {
            return Array.from(this._stylesSet);
        }
    }
    SharedStylesHost.ɵfac = function SharedStylesHost_Factory(t) { return new (t || SharedStylesHost)(); };
    SharedStylesHost.ɵprov = ɵɵdefineInjectable({ token: SharedStylesHost, factory: SharedStylesHost.ɵfac });
    class DomSharedStylesHost extends SharedStylesHost {
        /**
         * @param {?} _doc
         */
        constructor(_doc) {
            super();
            this._doc = _doc;
            this._hostNodes = new Set();
            this._styleNodes = new Set();
            this._hostNodes.add(_doc.head);
        }
        /**
         * @private
         * @param {?} styles
         * @param {?} host
         * @return {?}
         */
        _addStylesToHost(styles, host) {
            styles.forEach((/**
             * @param {?} style
             * @return {?}
             */
            (style) => {
                /** @type {?} */
                const styleEl = this._doc.createElement('style');
                styleEl.textContent = style;
                this._styleNodes.add(host.appendChild(styleEl));
            }));
        }
        /**
         * @param {?} hostNode
         * @return {?}
         */
        addHost(hostNode) {
            this._addStylesToHost(this._stylesSet, hostNode);
            this._hostNodes.add(hostNode);
        }
        /**
         * @param {?} hostNode
         * @return {?}
         */
        removeHost(hostNode) {
            this._hostNodes.delete(hostNode);
        }
        /**
         * @param {?} additions
         * @return {?}
         */
        onStylesAdded(additions) {
            this._hostNodes.forEach((/**
             * @param {?} hostNode
             * @return {?}
             */
            hostNode => this._addStylesToHost(additions, hostNode)));
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this._styleNodes.forEach((/**
             * @param {?} styleNode
             * @return {?}
             */
            styleNode => getDOM().remove(styleNode)));
        }
    }
    DomSharedStylesHost.ɵfac = function DomSharedStylesHost_Factory(t) { return new (t || DomSharedStylesHost)(ɵɵinject(DOCUMENT$1)); };
    DomSharedStylesHost.ɵprov = ɵɵdefineInjectable({ token: DomSharedStylesHost, factory: DomSharedStylesHost.ɵfac });
    /** @nocollapse */
    DomSharedStylesHost.ctorParameters = () => [
        { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT$1,] }] }
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/platform-browser/src/dom/dom_renderer.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const NAMESPACE_URIS = {
        'svg': 'http://www.w3.org/2000/svg',
        'xhtml': 'http://www.w3.org/1999/xhtml',
        'xlink': 'http://www.w3.org/1999/xlink',
        'xml': 'http://www.w3.org/XML/1998/namespace',
        'xmlns': 'http://www.w3.org/2000/xmlns/',
    };
    /** @type {?} */
    const COMPONENT_REGEX = /%COMP%/g;
    /** @type {?} */
    const NG_DEV_MODE$1 = typeof ngDevMode === 'undefined' || !!ngDevMode;
    /** @type {?} */
    const COMPONENT_VARIABLE = '%COMP%';
    /** @type {?} */
    const HOST_ATTR = `_nghost-${COMPONENT_VARIABLE}`;
    /** @type {?} */
    const CONTENT_ATTR = `_ngcontent-${COMPONENT_VARIABLE}`;
    /**
     * @param {?} componentShortId
     * @return {?}
     */
    function shimContentAttribute(componentShortId) {
        return CONTENT_ATTR.replace(COMPONENT_REGEX, componentShortId);
    }
    /**
     * @param {?} componentShortId
     * @return {?}
     */
    function shimHostAttribute(componentShortId) {
        return HOST_ATTR.replace(COMPONENT_REGEX, componentShortId);
    }
    /**
     * @param {?} compId
     * @param {?} styles
     * @param {?} target
     * @return {?}
     */
    function flattenStyles(compId, styles, target) {
        for (let i = 0; i < styles.length; i++) {
            /** @type {?} */
            let style = styles[i];
            if (Array.isArray(style)) {
                flattenStyles(compId, style, target);
            }
            else {
                style = style.replace(COMPONENT_REGEX, compId);
                target.push(style);
            }
        }
        return target;
    }
    /**
     * @param {?} eventHandler
     * @return {?}
     */
    function decoratePreventDefault(eventHandler) {
        // `DebugNode.triggerEventHandler` needs to know if the listener was created with
        // decoratePreventDefault or is a listener added outside the Angular context so it can handle the
        // two differently. In the first case, the special '__ngUnwrap__' token is passed to the unwrap
        // the listener (see below).
        return (/**
         * @param {?} event
         * @return {?}
         */
        (event) => {
            // Ivy uses '__ngUnwrap__' as a special token that allows us to unwrap the function
            // so that it can be invoked programmatically by `DebugNode.triggerEventHandler`. The debug_node
            // can inspect the listener toString contents for the existence of this special token. Because
            // the token is a string literal, it is ensured to not be modified by compiled code.
            if (event === '__ngUnwrap__') {
                return eventHandler;
            }
            /** @type {?} */
            const allowDefaultBehavior = eventHandler(event);
            if (allowDefaultBehavior === false) {
                // TODO(tbosch): move preventDefault into event plugins...
                event.preventDefault();
                event.returnValue = false;
            }
            return undefined;
        });
    }
    class DomRendererFactory2 {
        /**
         * @param {?} eventManager
         * @param {?} sharedStylesHost
         * @param {?} appId
         */
        constructor(eventManager, sharedStylesHost, appId) {
            this.eventManager = eventManager;
            this.sharedStylesHost = sharedStylesHost;
            this.appId = appId;
            this.rendererByCompId = new Map();
            this.defaultRenderer = new DefaultDomRenderer2(eventManager);
        }
        /**
         * @param {?} element
         * @param {?} type
         * @return {?}
         */
        createRenderer(element, type) {
            if (!element || !type) {
                return this.defaultRenderer;
            }
            switch (type.encapsulation) {
                case ViewEncapsulation$1.Emulated: {
                    /** @type {?} */
                    let renderer = this.rendererByCompId.get(type.id);
                    if (!renderer) {
                        renderer = new EmulatedEncapsulationDomRenderer2(this.eventManager, this.sharedStylesHost, type, this.appId);
                        this.rendererByCompId.set(type.id, renderer);
                    }
                    ((/** @type {?} */ (renderer))).applyToHost(element);
                    return renderer;
                }
                case ViewEncapsulation$1.Native:
                case ViewEncapsulation$1.ShadowDom:
                    return new ShadowDomRenderer(this.eventManager, this.sharedStylesHost, element, type);
                default: {
                    if (!this.rendererByCompId.has(type.id)) {
                        /** @type {?} */
                        const styles = flattenStyles(type.id, type.styles, []);
                        this.sharedStylesHost.addStyles(styles);
                        this.rendererByCompId.set(type.id, this.defaultRenderer);
                    }
                    return this.defaultRenderer;
                }
            }
        }
        /**
         * @return {?}
         */
        begin() { }
        /**
         * @return {?}
         */
        end() { }
    }
    DomRendererFactory2.ɵfac = function DomRendererFactory2_Factory(t) { return new (t || DomRendererFactory2)(ɵɵinject(EventManager), ɵɵinject(DomSharedStylesHost), ɵɵinject(APP_ID)); };
    DomRendererFactory2.ɵprov = ɵɵdefineInjectable({ token: DomRendererFactory2, factory: DomRendererFactory2.ɵfac });
    /** @nocollapse */
    DomRendererFactory2.ctorParameters = () => [
        { type: EventManager },
        { type: DomSharedStylesHost },
        { type: String, decorators: [{ type: Inject, args: [APP_ID,] }] }
    ];
    class DefaultDomRenderer2 {
        /**
         * @param {?} eventManager
         */
        constructor(eventManager) {
            this.eventManager = eventManager;
            this.data = Object.create(null);
        }
        /**
         * @return {?}
         */
        destroy() { }
        /**
         * @param {?} name
         * @param {?=} namespace
         * @return {?}
         */
        createElement(name, namespace) {
            if (namespace) {
                // In cases where Ivy (not ViewEngine) is giving us the actual namespace, the look up by key
                // will result in undefined, so we just return the namespace here.
                return document.createElementNS(NAMESPACE_URIS[namespace] || namespace, name);
            }
            return document.createElement(name);
        }
        /**
         * @param {?} value
         * @return {?}
         */
        createComment(value) {
            return document.createComment(value);
        }
        /**
         * @param {?} value
         * @return {?}
         */
        createText(value) {
            return document.createTextNode(value);
        }
        /**
         * @param {?} parent
         * @param {?} newChild
         * @return {?}
         */
        appendChild(parent, newChild) {
            parent.appendChild(newChild);
        }
        /**
         * @param {?} parent
         * @param {?} newChild
         * @param {?} refChild
         * @return {?}
         */
        insertBefore(parent, newChild, refChild) {
            if (parent) {
                parent.insertBefore(newChild, refChild);
            }
        }
        /**
         * @param {?} parent
         * @param {?} oldChild
         * @return {?}
         */
        removeChild(parent, oldChild) {
            if (parent) {
                parent.removeChild(oldChild);
            }
        }
        /**
         * @param {?} selectorOrNode
         * @param {?=} preserveContent
         * @return {?}
         */
        selectRootElement(selectorOrNode, preserveContent) {
            /** @type {?} */
            let el = typeof selectorOrNode === 'string' ? document.querySelector(selectorOrNode) :
                selectorOrNode;
            if (!el) {
                throw new Error(`The selector "${selectorOrNode}" did not match any elements`);
            }
            if (!preserveContent) {
                el.textContent = '';
            }
            return el;
        }
        /**
         * @param {?} node
         * @return {?}
         */
        parentNode(node) {
            return node.parentNode;
        }
        /**
         * @param {?} node
         * @return {?}
         */
        nextSibling(node) {
            return node.nextSibling;
        }
        /**
         * @param {?} el
         * @param {?} name
         * @param {?} value
         * @param {?=} namespace
         * @return {?}
         */
        setAttribute(el, name, value, namespace) {
            if (namespace) {
                name = namespace + ':' + name;
                // TODO(FW-811): Ivy may cause issues here because it's passing around
                // full URIs for namespaces, therefore this lookup will fail.
                /** @type {?} */
                const namespaceUri = NAMESPACE_URIS[namespace];
                if (namespaceUri) {
                    el.setAttributeNS(namespaceUri, name, value);
                }
                else {
                    el.setAttribute(name, value);
                }
            }
            else {
                el.setAttribute(name, value);
            }
        }
        /**
         * @param {?} el
         * @param {?} name
         * @param {?=} namespace
         * @return {?}
         */
        removeAttribute(el, name, namespace) {
            if (namespace) {
                // TODO(FW-811): Ivy may cause issues here because it's passing around
                // full URIs for namespaces, therefore this lookup will fail.
                /** @type {?} */
                const namespaceUri = NAMESPACE_URIS[namespace];
                if (namespaceUri) {
                    el.removeAttributeNS(namespaceUri, name);
                }
                else {
                    // TODO(FW-811): Since ivy is passing around full URIs for namespaces
                    // this could result in properties like `http://www.w3.org/2000/svg:cx="123"`,
                    // which is wrong.
                    el.removeAttribute(`${namespace}:${name}`);
                }
            }
            else {
                el.removeAttribute(name);
            }
        }
        /**
         * @param {?} el
         * @param {?} name
         * @return {?}
         */
        addClass(el, name) {
            el.classList.add(name);
        }
        /**
         * @param {?} el
         * @param {?} name
         * @return {?}
         */
        removeClass(el, name) {
            el.classList.remove(name);
        }
        /**
         * @param {?} el
         * @param {?} style
         * @param {?} value
         * @param {?} flags
         * @return {?}
         */
        setStyle(el, style, value, flags) {
            if (flags & RendererStyleFlags2.DashCase) {
                el.style.setProperty(style, value, !!(flags & RendererStyleFlags2.Important) ? 'important' : '');
            }
            else {
                el.style[style] = value;
            }
        }
        /**
         * @param {?} el
         * @param {?} style
         * @param {?} flags
         * @return {?}
         */
        removeStyle(el, style, flags) {
            if (flags & RendererStyleFlags2.DashCase) {
                el.style.removeProperty(style);
            }
            else {
                // IE requires '' instead of null
                // see https://github.com/angular/angular/issues/7916
                el.style[style] = '';
            }
        }
        /**
         * @param {?} el
         * @param {?} name
         * @param {?} value
         * @return {?}
         */
        setProperty(el, name, value) {
            NG_DEV_MODE$1 && checkNoSyntheticProp(name, 'property');
            el[name] = value;
        }
        /**
         * @param {?} node
         * @param {?} value
         * @return {?}
         */
        setValue(node, value) {
            node.nodeValue = value;
        }
        /**
         * @param {?} target
         * @param {?} event
         * @param {?} callback
         * @return {?}
         */
        listen(target, event, callback) {
            NG_DEV_MODE$1 && checkNoSyntheticProp(event, 'listener');
            if (typeof target === 'string') {
                return (/** @type {?} */ (this.eventManager.addGlobalEventListener(target, event, decoratePreventDefault(callback))));
            }
            return (/** @type {?} */ ((/** @type {?} */ (this.eventManager.addEventListener(target, event, decoratePreventDefault(callback))))));
        }
    }
    const ɵ0$2$1 = /**
     * @return {?}
     */
    () => '@'.charCodeAt(0);
    /** @type {?} */
    const AT_CHARCODE = ((ɵ0$2$1))();
    /**
     * @param {?} name
     * @param {?} nameKind
     * @return {?}
     */
    function checkNoSyntheticProp(name, nameKind) {
        if (name.charCodeAt(0) === AT_CHARCODE) {
            throw new Error(`Found the synthetic ${nameKind} ${name}. Please include either "BrowserAnimationsModule" or "NoopAnimationsModule" in your application.`);
        }
    }
    class EmulatedEncapsulationDomRenderer2 extends DefaultDomRenderer2 {
        /**
         * @param {?} eventManager
         * @param {?} sharedStylesHost
         * @param {?} component
         * @param {?} appId
         */
        constructor(eventManager, sharedStylesHost, component, appId) {
            super(eventManager);
            this.component = component;
            /** @type {?} */
            const styles = flattenStyles(appId + '-' + component.id, component.styles, []);
            sharedStylesHost.addStyles(styles);
            this.contentAttr = shimContentAttribute(appId + '-' + component.id);
            this.hostAttr = shimHostAttribute(appId + '-' + component.id);
        }
        /**
         * @param {?} element
         * @return {?}
         */
        applyToHost(element) {
            super.setAttribute(element, this.hostAttr, '');
        }
        /**
         * @param {?} parent
         * @param {?} name
         * @return {?}
         */
        createElement(parent, name) {
            /** @type {?} */
            const el = super.createElement(parent, name);
            super.setAttribute(el, this.contentAttr, '');
            return el;
        }
    }
    class ShadowDomRenderer extends DefaultDomRenderer2 {
        /**
         * @param {?} eventManager
         * @param {?} sharedStylesHost
         * @param {?} hostEl
         * @param {?} component
         */
        constructor(eventManager, sharedStylesHost, hostEl, component) {
            super(eventManager);
            this.sharedStylesHost = sharedStylesHost;
            this.hostEl = hostEl;
            this.component = component;
            if (component.encapsulation === ViewEncapsulation$1.ShadowDom) {
                this.shadowRoot = ((/** @type {?} */ (hostEl))).attachShadow({ mode: 'open' });
            }
            else {
                this.shadowRoot = ((/** @type {?} */ (hostEl))).createShadowRoot();
            }
            this.sharedStylesHost.addHost(this.shadowRoot);
            /** @type {?} */
            const styles = flattenStyles(component.id, component.styles, []);
            for (let i = 0; i < styles.length; i++) {
                /** @type {?} */
                const styleEl = document.createElement('style');
                styleEl.textContent = styles[i];
                this.shadowRoot.appendChild(styleEl);
            }
        }
        /**
         * @private
         * @param {?} node
         * @return {?}
         */
        nodeOrShadowRoot(node) {
            return node === this.hostEl ? this.shadowRoot : node;
        }
        /**
         * @return {?}
         */
        destroy() {
            this.sharedStylesHost.removeHost(this.shadowRoot);
        }
        /**
         * @param {?} parent
         * @param {?} newChild
         * @return {?}
         */
        appendChild(parent, newChild) {
            return super.appendChild(this.nodeOrShadowRoot(parent), newChild);
        }
        /**
         * @param {?} parent
         * @param {?} newChild
         * @param {?} refChild
         * @return {?}
         */
        insertBefore(parent, newChild, refChild) {
            return super.insertBefore(this.nodeOrShadowRoot(parent), newChild, refChild);
        }
        /**
         * @param {?} parent
         * @param {?} oldChild
         * @return {?}
         */
        removeChild(parent, oldChild) {
            return super.removeChild(this.nodeOrShadowRoot(parent), oldChild);
        }
        /**
         * @param {?} node
         * @return {?}
         */
        parentNode(node) {
            return this.nodeOrShadowRoot(super.parentNode(this.nodeOrShadowRoot(node)));
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/platform-browser/src/dom/events/dom_events.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class DomEventsPlugin extends EventManagerPlugin {
        /**
         * @param {?} doc
         */
        constructor(doc) {
            super(doc);
        }
        // This plugin should come last in the list of plugins, because it accepts all
        // events.
        /**
         * @param {?} eventName
         * @return {?}
         */
        supports(eventName) {
            return true;
        }
        /**
         * @param {?} element
         * @param {?} eventName
         * @param {?} handler
         * @return {?}
         */
        addEventListener(element, eventName, handler) {
            element.addEventListener(eventName, (/** @type {?} */ (handler)), false);
            return (/**
             * @return {?}
             */
            () => this.removeEventListener(element, eventName, (/** @type {?} */ (handler))));
        }
        /**
         * @param {?} target
         * @param {?} eventName
         * @param {?} callback
         * @return {?}
         */
        removeEventListener(target, eventName, callback) {
            return target.removeEventListener(eventName, (/** @type {?} */ (callback)));
        }
    }
    DomEventsPlugin.ɵfac = function DomEventsPlugin_Factory(t) { return new (t || DomEventsPlugin)(ɵɵinject(DOCUMENT$1)); };
    DomEventsPlugin.ɵprov = ɵɵdefineInjectable({ token: DomEventsPlugin, factory: DomEventsPlugin.ɵfac });
    /** @nocollapse */
    DomEventsPlugin.ctorParameters = () => [
        { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT$1,] }] }
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/platform-browser/src/dom/events/hammer_gestures.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Supported HammerJS recognizer event names.
     * @type {?}
     */
    const EVENT_NAMES = {
        // pan
        'pan': true,
        'panstart': true,
        'panmove': true,
        'panend': true,
        'pancancel': true,
        'panleft': true,
        'panright': true,
        'panup': true,
        'pandown': true,
        // pinch
        'pinch': true,
        'pinchstart': true,
        'pinchmove': true,
        'pinchend': true,
        'pinchcancel': true,
        'pinchin': true,
        'pinchout': true,
        // press
        'press': true,
        'pressup': true,
        // rotate
        'rotate': true,
        'rotatestart': true,
        'rotatemove': true,
        'rotateend': true,
        'rotatecancel': true,
        // swipe
        'swipe': true,
        'swipeleft': true,
        'swiperight': true,
        'swipeup': true,
        'swipedown': true,
        // tap
        'tap': true,
    };
    /**
     * DI token for providing [HammerJS](http://hammerjs.github.io/) support to Angular.
     * @see `HammerGestureConfig`
     *
     * \@ngModule HammerModule
     * \@publicApi
     * @type {?}
     */
    const HAMMER_GESTURE_CONFIG = new InjectionToken('HammerGestureConfig');
    /**
     * Injection token used to provide a {\@link HammerLoader} to Angular.
     *
     * \@publicApi
     * @type {?}
     */
    const HAMMER_LOADER = new InjectionToken('HammerLoader');
    /**
     * An injectable [HammerJS Manager](http://hammerjs.github.io/api/#hammer.manager)
     * for gesture recognition. Configures specific event recognition.
     * \@publicApi
     */
    class HammerGestureConfig {
        constructor() {
            /**
             * A set of supported event names for gestures to be used in Angular.
             * Angular supports all built-in recognizers, as listed in
             * [HammerJS documentation](http://hammerjs.github.io/).
             */
            this.events = [];
            /**
             * Maps gesture event names to a set of configuration options
             * that specify overrides to the default values for specific properties.
             *
             * The key is a supported event name to be configured,
             * and the options object contains a set of properties, with override values
             * to be applied to the named recognizer event.
             * For example, to disable recognition of the rotate event, specify
             *  `{"rotate": {"enable": false}}`.
             *
             * Properties that are not present take the HammerJS default values.
             * For information about which properties are supported for which events,
             * and their allowed and default values, see
             * [HammerJS documentation](http://hammerjs.github.io/).
             *
             */
            this.overrides = {};
        }
        /**
         * Creates a [HammerJS Manager](http://hammerjs.github.io/api/#hammer.manager)
         * and attaches it to a given HTML element.
         * @param {?} element The element that will recognize gestures.
         * @return {?} A HammerJS event-manager object.
         */
        buildHammer(element) {
            /** @type {?} */
            const mc = new (/** @type {?} */ (Hammer))(element, this.options);
            mc.get('pinch').set({ enable: true });
            mc.get('rotate').set({ enable: true });
            for (const eventName in this.overrides) {
                mc.get(eventName).set(this.overrides[eventName]);
            }
            return mc;
        }
    }
    HammerGestureConfig.ɵfac = function HammerGestureConfig_Factory(t) { return new (t || HammerGestureConfig)(); };
    HammerGestureConfig.ɵprov = ɵɵdefineInjectable({ token: HammerGestureConfig, factory: HammerGestureConfig.ɵfac });
    /**
     * Event plugin that adds Hammer support to an application.
     *
     * \@ngModule HammerModule
     */
    class HammerGesturesPlugin extends EventManagerPlugin {
        /**
         * @param {?} doc
         * @param {?} _config
         * @param {?} console
         * @param {?=} loader
         */
        constructor(doc, _config, console, loader) {
            super(doc);
            this._config = _config;
            this.console = console;
            this.loader = loader;
        }
        /**
         * @param {?} eventName
         * @return {?}
         */
        supports(eventName) {
            if (!EVENT_NAMES.hasOwnProperty(eventName.toLowerCase()) && !this.isCustomEvent(eventName)) {
                return false;
            }
            if (!((/** @type {?} */ (window))).Hammer && !this.loader) {
                this.console.warn(`The "${eventName}" event cannot be bound because Hammer.JS is not ` +
                    `loaded and no custom loader has been specified.`);
                return false;
            }
            return true;
        }
        /**
         * @param {?} element
         * @param {?} eventName
         * @param {?} handler
         * @return {?}
         */
        addEventListener(element, eventName, handler) {
            /** @type {?} */
            const zone = this.manager.getZone();
            eventName = eventName.toLowerCase();
            // If Hammer is not present but a loader is specified, we defer adding the event listener
            // until Hammer is loaded.
            if (!((/** @type {?} */ (window))).Hammer && this.loader) {
                // This `addEventListener` method returns a function to remove the added listener.
                // Until Hammer is loaded, the returned function needs to *cancel* the registration rather
                // than remove anything.
                /** @type {?} */
                let cancelRegistration = false;
                /** @type {?} */
                let deregister = (/**
                 * @return {?}
                 */
                () => {
                    cancelRegistration = true;
                });
                this.loader()
                    .then((/**
                 * @return {?}
                 */
                () => {
                    // If Hammer isn't actually loaded when the custom loader resolves, give up.
                    if (!((/** @type {?} */ (window))).Hammer) {
                        this.console.warn(`The custom HAMMER_LOADER completed, but Hammer.JS is not present.`);
                        deregister = (/**
                         * @return {?}
                         */
                        () => { });
                        return;
                    }
                    if (!cancelRegistration) {
                        // Now that Hammer is loaded and the listener is being loaded for real,
                        // the deregistration function changes from canceling registration to removal.
                        deregister = this.addEventListener(element, eventName, handler);
                    }
                }))
                    .catch((/**
                 * @return {?}
                 */
                () => {
                    this.console.warn(`The "${eventName}" event cannot be bound because the custom ` +
                        `Hammer.JS loader failed.`);
                    deregister = (/**
                     * @return {?}
                     */
                    () => { });
                }));
                // Return a function that *executes* `deregister` (and not `deregister` itself) so that we
                // can change the behavior of `deregister` once the listener is added. Using a closure in
                // this way allows us to avoid any additional data structures to track listener removal.
                return (/**
                 * @return {?}
                 */
                () => {
                    deregister();
                });
            }
            return zone.runOutsideAngular((/**
             * @return {?}
             */
            () => {
                // Creating the manager bind events, must be done outside of angular
                /** @type {?} */
                const mc = this._config.buildHammer(element);
                /** @type {?} */
                const callback = (/**
                 * @param {?} eventObj
                 * @return {?}
                 */
                function (eventObj) {
                    zone.runGuarded((/**
                     * @return {?}
                     */
                    function () {
                        handler(eventObj);
                    }));
                });
                mc.on(eventName, callback);
                return (/**
                 * @return {?}
                 */
                () => {
                    mc.off(eventName, callback);
                    // destroy mc to prevent memory leak
                    if (typeof mc.destroy === 'function') {
                        mc.destroy();
                    }
                });
            }));
        }
        /**
         * @param {?} eventName
         * @return {?}
         */
        isCustomEvent(eventName) {
            return this._config.events.indexOf(eventName) > -1;
        }
    }
    HammerGesturesPlugin.ɵfac = function HammerGesturesPlugin_Factory(t) { return new (t || HammerGesturesPlugin)(ɵɵinject(DOCUMENT$1), ɵɵinject(HAMMER_GESTURE_CONFIG), ɵɵinject(Console), ɵɵinject(HAMMER_LOADER, 8)); };
    HammerGesturesPlugin.ɵprov = ɵɵdefineInjectable({ token: HammerGesturesPlugin, factory: HammerGesturesPlugin.ɵfac });
    /** @nocollapse */
    HammerGesturesPlugin.ctorParameters = () => [
        { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT$1,] }] },
        { type: HammerGestureConfig, decorators: [{ type: Inject, args: [HAMMER_GESTURE_CONFIG,] }] },
        { type: Console },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [HAMMER_LOADER,] }] }
    ];
    /**
     * In Ivy, support for Hammer gestures is optional, so applications must
     * import the `HammerModule` at root to turn on support. This means that
     * Hammer-specific code can be tree-shaken away if not needed.
     * @type {?}
     */
    const HAMMER_PROVIDERS__POST_R3__ = [];
    /**
     * In View Engine, support for Hammer gestures is built-in by default.
     * @type {?}
     */
    const HAMMER_PROVIDERS__PRE_R3__ = [
        {
            provide: EVENT_MANAGER_PLUGINS,
            useClass: HammerGesturesPlugin,
            multi: true,
            deps: [DOCUMENT$1, HAMMER_GESTURE_CONFIG, Console, [new Optional(), HAMMER_LOADER]]
        },
        { provide: HAMMER_GESTURE_CONFIG, useClass: HammerGestureConfig, deps: [] },
    ];
    /** @type {?} */
    const HAMMER_PROVIDERS = HAMMER_PROVIDERS__POST_R3__;
    /**
     * Adds support for HammerJS.
     *
     * Import this module at the root of your application so that Angular can work with
     * HammerJS to detect gesture events.
     *
     * Note that applications still need to include the HammerJS script itself. This module
     * simply sets up the coordination layer between HammerJS and Angular's EventManager.
     *
     * \@publicApi
     */
    class HammerModule {
    }
    HammerModule.ɵmod = ɵɵdefineNgModule({ type: HammerModule });
    HammerModule.ɵinj = ɵɵdefineInjector({ factory: function HammerModule_Factory(t) { return new (t || HammerModule)(); }, providers: HAMMER_PROVIDERS__PRE_R3__ });

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/platform-browser/src/dom/events/key_events.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Defines supported modifiers for key events.
     * @type {?}
     */
    const MODIFIER_KEYS = ['alt', 'control', 'meta', 'shift'];
    /** @type {?} */
    const DOM_KEY_LOCATION_NUMPAD = 3;
    // Map to convert some key or keyIdentifier values to what will be returned by getEventKey
    /** @type {?} */
    const _keyMap = {
        // The following values are here for cross-browser compatibility and to match the W3C standard
        // cf http://www.w3.org/TR/DOM-Level-3-Events-key/
        '\b': 'Backspace',
        '\t': 'Tab',
        '\x7F': 'Delete',
        '\x1B': 'Escape',
        'Del': 'Delete',
        'Esc': 'Escape',
        'Left': 'ArrowLeft',
        'Right': 'ArrowRight',
        'Up': 'ArrowUp',
        'Down': 'ArrowDown',
        'Menu': 'ContextMenu',
        'Scroll': 'ScrollLock',
        'Win': 'OS'
    };
    // There is a bug in Chrome for numeric keypad keys:
    // https://code.google.com/p/chromium/issues/detail?id=155654
    // 1, 2, 3 ... are reported as A, B, C ...
    /** @type {?} */
    const _chromeNumKeyPadMap = {
        'A': '1',
        'B': '2',
        'C': '3',
        'D': '4',
        'E': '5',
        'F': '6',
        'G': '7',
        'H': '8',
        'I': '9',
        'J': '*',
        'K': '+',
        'M': '-',
        'N': '.',
        'O': '/',
        '\x60': '0',
        '\x90': 'NumLock'
    };
    const ɵ0$3$1 = /**
     * @param {?} event
     * @return {?}
     */
    (event) => event.altKey, ɵ1$1 = /**
     * @param {?} event
     * @return {?}
     */
    (event) => event.ctrlKey, ɵ2$2 = /**
     * @param {?} event
     * @return {?}
     */
    (event) => event.metaKey, ɵ3$2 = /**
     * @param {?} event
     * @return {?}
     */
    (event) => event.shiftKey;
    /**
     * Retrieves modifiers from key-event objects.
     * @type {?}
     */
    const MODIFIER_KEY_GETTERS = {
        'alt': (ɵ0$3$1),
        'control': (ɵ1$1),
        'meta': (ɵ2$2),
        'shift': (ɵ3$2)
    };
    /**
     * \@publicApi
     * A browser plug-in that provides support for handling of key events in Angular.
     */
    class KeyEventsPlugin extends EventManagerPlugin {
        /**
         * Initializes an instance of the browser plug-in.
         * @param {?} doc The document in which key events will be detected.
         */
        constructor(doc) {
            super(doc);
        }
        /**
         * Reports whether a named key event is supported.
         * @param {?} eventName The event name to query.
         * @return {?} True if the named key event is supported.
         */
        supports(eventName) {
            return KeyEventsPlugin.parseEventName(eventName) != null;
        }
        /**
         * Registers a handler for a specific element and key event.
         * @param {?} element The HTML element to receive event notifications.
         * @param {?} eventName The name of the key event to listen for.
         * @param {?} handler A function to call when the notification occurs. Receives the
         * event object as an argument.
         * @return {?} The key event that was registered.
         */
        addEventListener(element, eventName, handler) {
            /** @type {?} */
            const parsedEvent = (/** @type {?} */ (KeyEventsPlugin.parseEventName(eventName)));
            /** @type {?} */
            const outsideHandler = KeyEventsPlugin.eventCallback(parsedEvent['fullKey'], handler, this.manager.getZone());
            return this.manager.getZone().runOutsideAngular((/**
             * @return {?}
             */
            () => {
                return getDOM().onAndCancel(element, parsedEvent['domEventName'], outsideHandler);
            }));
        }
        /**
         * @param {?} eventName
         * @return {?}
         */
        static parseEventName(eventName) {
            /** @type {?} */
            const parts = eventName.toLowerCase().split('.');
            /** @type {?} */
            const domEventName = parts.shift();
            if ((parts.length === 0) || !(domEventName === 'keydown' || domEventName === 'keyup')) {
                return null;
            }
            /** @type {?} */
            const key = KeyEventsPlugin._normalizeKey((/** @type {?} */ (parts.pop())));
            /** @type {?} */
            let fullKey = '';
            MODIFIER_KEYS.forEach((/**
             * @param {?} modifierName
             * @return {?}
             */
            modifierName => {
                /** @type {?} */
                const index = parts.indexOf(modifierName);
                if (index > -1) {
                    parts.splice(index, 1);
                    fullKey += modifierName + '.';
                }
            }));
            fullKey += key;
            if (parts.length != 0 || key.length === 0) {
                // returning null instead of throwing to let another plugin process the event
                return null;
            }
            /** @type {?} */
            const result = {};
            result['domEventName'] = domEventName;
            result['fullKey'] = fullKey;
            return result;
        }
        /**
         * @param {?} event
         * @return {?}
         */
        static getEventFullKey(event) {
            /** @type {?} */
            let fullKey = '';
            /** @type {?} */
            let key = getEventKey(event);
            key = key.toLowerCase();
            if (key === ' ') {
                key = 'space'; // for readability
            }
            else if (key === '.') {
                key = 'dot'; // because '.' is used as a separator in event names
            }
            MODIFIER_KEYS.forEach((/**
             * @param {?} modifierName
             * @return {?}
             */
            modifierName => {
                if (modifierName != key) {
                    /** @type {?} */
                    const modifierGetter = MODIFIER_KEY_GETTERS[modifierName];
                    if (modifierGetter(event)) {
                        fullKey += modifierName + '.';
                    }
                }
            }));
            fullKey += key;
            return fullKey;
        }
        /**
         * Configures a handler callback for a key event.
         * @param {?} fullKey The event name that combines all simultaneous keystrokes.
         * @param {?} handler The function that responds to the key event.
         * @param {?} zone The zone in which the event occurred.
         * @return {?} A callback function.
         */
        static eventCallback(fullKey, handler, zone) {
            return (/**
             * @param {?} event
             * @return {?}
             */
            (event /** TODO #9100 */) => {
                if (KeyEventsPlugin.getEventFullKey(event) === fullKey) {
                    zone.runGuarded((/**
                     * @return {?}
                     */
                    () => handler(event)));
                }
            });
        }
        /**
         * \@internal
         * @param {?} keyName
         * @return {?}
         */
        static _normalizeKey(keyName) {
            // TODO: switch to a Map if the mapping grows too much
            switch (keyName) {
                case 'esc':
                    return 'escape';
                default:
                    return keyName;
            }
        }
    }
    KeyEventsPlugin.ɵfac = function KeyEventsPlugin_Factory(t) { return new (t || KeyEventsPlugin)(ɵɵinject(DOCUMENT$1)); };
    KeyEventsPlugin.ɵprov = ɵɵdefineInjectable({ token: KeyEventsPlugin, factory: KeyEventsPlugin.ɵfac });
    /** @nocollapse */
    KeyEventsPlugin.ctorParameters = () => [
        { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT$1,] }] }
    ];
    /**
     * @param {?} event
     * @return {?}
     */
    function getEventKey(event) {
        /** @type {?} */
        let key = event.key;
        if (key == null) {
            key = event.keyIdentifier;
            // keyIdentifier is defined in the old draft of DOM Level 3 Events implemented by Chrome and
            // Safari cf
            // http://www.w3.org/TR/2007/WD-DOM-Level-3-Events-20071221/events.html#Events-KeyboardEvents-Interfaces
            if (key == null) {
                return 'Unidentified';
            }
            if (key.startsWith('U+')) {
                key = String.fromCharCode(parseInt(key.substring(2), 16));
                if (event.location === DOM_KEY_LOCATION_NUMPAD && _chromeNumKeyPadMap.hasOwnProperty(key)) {
                    // There is a bug in Chrome for numeric keypad keys:
                    // https://code.google.com/p/chromium/issues/detail?id=155654
                    // 1, 2, 3 ... are reported as A, B, C ...
                    key = ((/** @type {?} */ (_chromeNumKeyPadMap)))[key];
                }
            }
        }
        return _keyMap[key] || key;
    }
    /**
     * DomSanitizer helps preventing Cross Site Scripting Security bugs (XSS) by sanitizing
     * values to be safe to use in the different DOM contexts.
     *
     * For example, when binding a URL in an `<a [href]="someValue">` hyperlink, `someValue` will be
     * sanitized so that an attacker cannot inject e.g. a `javascript:` URL that would execute code on
     * the website.
     *
     * In specific situations, it might be necessary to disable sanitization, for example if the
     * application genuinely needs to produce a `javascript:` style link with a dynamic value in it.
     * Users can bypass security by constructing a value with one of the `bypassSecurityTrust...`
     * methods, and then binding to that value from the template.
     *
     * These situations should be very rare, and extraordinary care must be taken to avoid creating a
     * Cross Site Scripting (XSS) security bug!
     *
     * When using `bypassSecurityTrust...`, make sure to call the method as early as possible and as
     * close as possible to the source of the value, to make it easy to verify no security bug is
     * created by its use.
     *
     * It is not required (and not recommended) to bypass security if the value is safe, e.g. a URL that
     * does not start with a suspicious protocol, or an HTML snippet that does not contain dangerous
     * code. The sanitizer leaves safe values intact.
     *
     * \@security Calling any of the `bypassSecurityTrust...` APIs disables Angular's built-in
     * sanitization for the value passed in. Carefully check and audit all values and code paths going
     * into this call. Make sure any user data is appropriately escaped for this security context.
     * For more detail, see the [Security Guide](http://g.co/ng/security).
     *
     * \@publicApi
     * @abstract
     */
    class DomSanitizer {
    }
    DomSanitizer.ɵfac = function DomSanitizer_Factory(t) { return new (t || DomSanitizer)(); };
    /** @nocollapse */ DomSanitizer.ɵprov = ɵɵdefineInjectable({ factory: function DomSanitizer_Factory() { return ɵɵinject(DomSanitizerImpl); }, token: DomSanitizer, providedIn: "root" });
    /**
     * @param {?} injector
     * @return {?}
     */
    function domSanitizerImplFactory(injector) {
        return new DomSanitizerImpl(injector.get(DOCUMENT$1));
    }
    class DomSanitizerImpl extends DomSanitizer {
        /**
         * @param {?} _doc
         */
        constructor(_doc) {
            super();
            this._doc = _doc;
        }
        /**
         * @param {?} ctx
         * @param {?} value
         * @return {?}
         */
        sanitize(ctx, value) {
            if (value == null)
                return null;
            switch (ctx) {
                case SecurityContext.NONE:
                    return (/** @type {?} */ (value));
                case SecurityContext.HTML:
                    if (allowSanitizationBypassAndThrow(value, "HTML" /* Html */)) {
                        return unwrapSafeValue(value);
                    }
                    return _sanitizeHtml(this._doc, String(value));
                case SecurityContext.STYLE:
                    if (allowSanitizationBypassAndThrow(value, "Style" /* Style */)) {
                        return unwrapSafeValue(value);
                    }
                    return _sanitizeStyle((/** @type {?} */ (value)));
                case SecurityContext.SCRIPT:
                    if (allowSanitizationBypassAndThrow(value, "Script" /* Script */)) {
                        return unwrapSafeValue(value);
                    }
                    throw new Error('unsafe value used in a script context');
                case SecurityContext.URL:
                    /** @type {?} */
                    const type = getSanitizationBypassType(value);
                    if (allowSanitizationBypassAndThrow(value, "URL" /* Url */)) {
                        return unwrapSafeValue(value);
                    }
                    return _sanitizeUrl(String(value));
                case SecurityContext.RESOURCE_URL:
                    if (allowSanitizationBypassAndThrow(value, "ResourceURL" /* ResourceUrl */)) {
                        return unwrapSafeValue(value);
                    }
                    throw new Error('unsafe value used in a resource URL context (see http://g.co/ng/security#xss)');
                default:
                    throw new Error(`Unexpected SecurityContext ${ctx} (see http://g.co/ng/security#xss)`);
            }
        }
        /**
         * @param {?} value
         * @return {?}
         */
        bypassSecurityTrustHtml(value) {
            return bypassSanitizationTrustHtml(value);
        }
        /**
         * @param {?} value
         * @return {?}
         */
        bypassSecurityTrustStyle(value) {
            return bypassSanitizationTrustStyle(value);
        }
        /**
         * @param {?} value
         * @return {?}
         */
        bypassSecurityTrustScript(value) {
            return bypassSanitizationTrustScript(value);
        }
        /**
         * @param {?} value
         * @return {?}
         */
        bypassSecurityTrustUrl(value) {
            return bypassSanitizationTrustUrl(value);
        }
        /**
         * @param {?} value
         * @return {?}
         */
        bypassSecurityTrustResourceUrl(value) {
            return bypassSanitizationTrustResourceUrl(value);
        }
    }
    DomSanitizerImpl.ɵfac = function DomSanitizerImpl_Factory(t) { return new (t || DomSanitizerImpl)(ɵɵinject(DOCUMENT$1)); };
    /** @nocollapse */
    DomSanitizerImpl.ctorParameters = () => [
        { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT$1,] }] }
    ];
    /** @nocollapse */ DomSanitizerImpl.ɵprov = ɵɵdefineInjectable({ factory: function DomSanitizerImpl_Factory() { return domSanitizerImplFactory(ɵɵinject(INJECTOR)); }, token: DomSanitizerImpl, providedIn: "root" });

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/platform-browser/src/browser.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @return {?}
     */
    function initDomAdapter() {
        BrowserDomAdapter.makeCurrent();
        BrowserGetTestability.init();
    }
    /**
     * @return {?}
     */
    function errorHandler() {
        return new ErrorHandler();
    }
    /**
     * @return {?}
     */
    function _document() {
        // Tell ivy about the global document
        setDocument(document);
        return document;
    }
    const ɵ0$4$1 = PLATFORM_BROWSER_ID;
    /** @type {?} */
    const INTERNAL_BROWSER_PLATFORM_PROVIDERS = [
        { provide: PLATFORM_ID, useValue: ɵ0$4$1 },
        { provide: PLATFORM_INITIALIZER, useValue: initDomAdapter, multi: true },
        { provide: DOCUMENT$1, useFactory: _document, deps: [] },
    ];
    /** @type {?} */
    const BROWSER_SANITIZATION_PROVIDERS__POST_R3__ = [];
    /**
     * \@security Replacing built-in sanitization providers exposes the application to XSS risks.
     * Attacker-controlled data introduced by an unsanitized provider could expose your
     * application to XSS risks. For more detail, see the [Security Guide](http://g.co/ng/security).
     * \@publicApi
     * @type {?}
     */
    const BROWSER_SANITIZATION_PROVIDERS = BROWSER_SANITIZATION_PROVIDERS__POST_R3__;
    /**
     * \@publicApi
     * @type {?}
     */
    const platformBrowser = createPlatformFactory(platformCore, 'browser', INTERNAL_BROWSER_PLATFORM_PROVIDERS);
    /** @type {?} */
    const BROWSER_MODULE_PROVIDERS = [
        BROWSER_SANITIZATION_PROVIDERS,
        { provide: INJECTOR_SCOPE, useValue: 'root' },
        { provide: ErrorHandler, useFactory: errorHandler, deps: [] },
        {
            provide: EVENT_MANAGER_PLUGINS,
            useClass: DomEventsPlugin,
            multi: true,
            deps: [DOCUMENT$1, NgZone, PLATFORM_ID]
        },
        { provide: EVENT_MANAGER_PLUGINS, useClass: KeyEventsPlugin, multi: true, deps: [DOCUMENT$1] },
        HAMMER_PROVIDERS,
        {
            provide: DomRendererFactory2,
            useClass: DomRendererFactory2,
            deps: [EventManager, DomSharedStylesHost, APP_ID]
        },
        { provide: RendererFactory2, useExisting: DomRendererFactory2 },
        { provide: SharedStylesHost, useExisting: DomSharedStylesHost },
        { provide: DomSharedStylesHost, useClass: DomSharedStylesHost, deps: [DOCUMENT$1] },
        { provide: Testability, useClass: Testability, deps: [NgZone] },
        { provide: EventManager, useClass: EventManager, deps: [EVENT_MANAGER_PLUGINS, NgZone] },
        ELEMENT_PROBE_PROVIDERS,
    ];
    /**
     * Exports required infrastructure for all Angular apps.
     * Included by default in all Angular apps created with the CLI
     * `new` command.
     * Re-exports `CommonModule` and `ApplicationModule`, making their
     * exports and providers available to all apps.
     *
     * \@publicApi
     */
    class BrowserModule {
        /**
         * @param {?} parentModule
         */
        constructor(parentModule) {
            if (parentModule) {
                throw new Error(`BrowserModule has already been loaded. If you need access to common directives such as NgIf and NgFor from a lazy loaded module, import CommonModule instead.`);
            }
        }
        /**
         * Configures a browser-based app to transition from a server-rendered app, if
         * one is present on the page.
         *
         * @param {?} params An object containing an identifier for the app to transition.
         * The ID must match between the client and server versions of the app.
         * @return {?} The reconfigured `BrowserModule` to import into the app's root `AppModule`.
         */
        static withServerTransition(params) {
            return {
                ngModule: BrowserModule,
                providers: [
                    { provide: APP_ID, useValue: params.appId },
                    { provide: TRANSITION_ID, useExisting: APP_ID },
                    SERVER_TRANSITION_PROVIDERS,
                ],
            };
        }
    }
    BrowserModule.ɵmod = ɵɵdefineNgModule({ type: BrowserModule });
    BrowserModule.ɵinj = ɵɵdefineInjector({ factory: function BrowserModule_Factory(t) { return new (t || BrowserModule)(ɵɵinject(BrowserModule, 12)); }, providers: BROWSER_MODULE_PROVIDERS, imports: [CommonModule, ApplicationModule] });
    /** @nocollapse */
    BrowserModule.ctorParameters = () => [
        { type: undefined, decorators: [{ type: Optional }, { type: SkipSelf }, { type: Inject, args: [BrowserModule,] }] }
    ];
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(BrowserModule, { exports: function () { return [CommonModule, ApplicationModule]; } }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/platform-browser/src/browser/meta.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Factory to create Meta service.
     * @return {?}
     */
    function createMeta() {
        return new Meta(ɵɵinject(DOCUMENT$1));
    }
    /**
     * A service that can be used to get and add meta tags.
     *
     * \@publicApi
     */
    class Meta {
        /**
         * @param {?} _doc
         */
        constructor(_doc) {
            this._doc = _doc;
            this._dom = getDOM();
        }
        /**
         * @param {?} tag
         * @param {?=} forceCreation
         * @return {?}
         */
        addTag(tag, forceCreation = false) {
            if (!tag)
                return null;
            return this._getOrCreateElement(tag, forceCreation);
        }
        /**
         * @param {?} tags
         * @param {?=} forceCreation
         * @return {?}
         */
        addTags(tags, forceCreation = false) {
            if (!tags)
                return [];
            return tags.reduce((/**
             * @param {?} result
             * @param {?} tag
             * @return {?}
             */
            (result, tag) => {
                if (tag) {
                    result.push(this._getOrCreateElement(tag, forceCreation));
                }
                return result;
            }), []);
        }
        /**
         * @param {?} attrSelector
         * @return {?}
         */
        getTag(attrSelector) {
            if (!attrSelector)
                return null;
            return this._doc.querySelector(`meta[${attrSelector}]`) || null;
        }
        /**
         * @param {?} attrSelector
         * @return {?}
         */
        getTags(attrSelector) {
            if (!attrSelector)
                return [];
            /** @type {?} */
            const list /*NodeList*/ = this._doc.querySelectorAll(`meta[${attrSelector}]`);
            return list ? [].slice.call(list) : [];
        }
        /**
         * @param {?} tag
         * @param {?=} selector
         * @return {?}
         */
        updateTag(tag, selector) {
            if (!tag)
                return null;
            selector = selector || this._parseSelector(tag);
            /** @type {?} */
            const meta = (/** @type {?} */ (this.getTag(selector)));
            if (meta) {
                return this._setMetaElementAttributes(tag, meta);
            }
            return this._getOrCreateElement(tag, true);
        }
        /**
         * @param {?} attrSelector
         * @return {?}
         */
        removeTag(attrSelector) {
            this.removeTagElement((/** @type {?} */ (this.getTag(attrSelector))));
        }
        /**
         * @param {?} meta
         * @return {?}
         */
        removeTagElement(meta) {
            if (meta) {
                this._dom.remove(meta);
            }
        }
        /**
         * @private
         * @param {?} meta
         * @param {?=} forceCreation
         * @return {?}
         */
        _getOrCreateElement(meta, forceCreation = false) {
            if (!forceCreation) {
                /** @type {?} */
                const selector = this._parseSelector(meta);
                /** @type {?} */
                const elem = (/** @type {?} */ (this.getTag(selector)));
                // It's allowed to have multiple elements with the same name so it's not enough to
                // just check that element with the same name already present on the page. We also need to
                // check if element has tag attributes
                if (elem && this._containsAttributes(meta, elem))
                    return elem;
            }
            /** @type {?} */
            const element = (/** @type {?} */ (this._dom.createElement('meta')));
            this._setMetaElementAttributes(meta, element);
            /** @type {?} */
            const head = this._doc.getElementsByTagName('head')[0];
            head.appendChild(element);
            return element;
        }
        /**
         * @private
         * @param {?} tag
         * @param {?} el
         * @return {?}
         */
        _setMetaElementAttributes(tag, el) {
            Object.keys(tag).forEach((/**
             * @param {?} prop
             * @return {?}
             */
            (prop) => el.setAttribute(prop, tag[prop])));
            return el;
        }
        /**
         * @private
         * @param {?} tag
         * @return {?}
         */
        _parseSelector(tag) {
            /** @type {?} */
            const attr = tag.name ? 'name' : 'property';
            return `${attr}="${tag[attr]}"`;
        }
        /**
         * @private
         * @param {?} tag
         * @param {?} elem
         * @return {?}
         */
        _containsAttributes(tag, elem) {
            return Object.keys(tag).every((/**
             * @param {?} key
             * @return {?}
             */
            (key) => elem.getAttribute(key) === tag[key]));
        }
    }
    Meta.ɵfac = function Meta_Factory(t) { return new (t || Meta)(ɵɵinject(DOCUMENT$1)); };
    /** @nocollapse */
    Meta.ctorParameters = () => [
        { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT$1,] }] }
    ];
    /** @nocollapse */ Meta.ɵprov = ɵɵdefineInjectable({ factory: createMeta, token: Meta, providedIn: "root" });

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/platform-browser/src/browser/title.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Factory to create Title service.
     * @return {?}
     */
    function createTitle() {
        return new Title(ɵɵinject(DOCUMENT$1));
    }
    /**
     * A service that can be used to get and set the title of a current HTML document.
     *
     * Since an Angular application can't be bootstrapped on the entire HTML document (`<html>` tag)
     * it is not possible to bind to the `text` property of the `HTMLTitleElement` elements
     * (representing the `<title>` tag). Instead, this service can be used to set and get the current
     * title value.
     *
     * \@publicApi
     */
    class Title {
        /**
         * @param {?} _doc
         */
        constructor(_doc) {
            this._doc = _doc;
        }
        /**
         * Get the title of the current HTML document.
         * @return {?}
         */
        getTitle() {
            return this._doc.title;
        }
        /**
         * Set the title of the current HTML document.
         * @param {?} newTitle
         * @return {?}
         */
        setTitle(newTitle) {
            this._doc.title = newTitle || '';
        }
    }
    Title.ɵfac = function Title_Factory(t) { return new (t || Title)(ɵɵinject(DOCUMENT$1)); };
    /** @nocollapse */
    Title.ctorParameters = () => [
        { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT$1,] }] }
    ];
    /** @nocollapse */ Title.ɵprov = ɵɵdefineInjectable({ factory: createTitle, token: Title, providedIn: "root" });
    /**
     * @param {?} text
     * @return {?}
     */
    function unescapeHtml(text) {
        /** @type {?} */
        const unescapedText = {
            '&a;': '&',
            '&q;': '"',
            '&s;': '\'',
            '&l;': '<',
            '&g;': '>',
        };
        return text.replace(/&[^;]+;/g, (/**
         * @param {?} s
         * @return {?}
         */
        s => unescapedText[s]));
    }
    /**
     * A key value store that is transferred from the application on the server side to the application
     * on the client side.
     *
     * `TransferState` will be available as an injectable token. To use it import
     * `ServerTransferStateModule` on the server and `BrowserTransferStateModule` on the client.
     *
     * The values in the store are serialized/deserialized using JSON.stringify/JSON.parse. So only
     * boolean, number, string, null and non-class objects will be serialized and deserialzied in a
     * non-lossy manner.
     *
     * \@publicApi
     */
    class TransferState {
        constructor() {
            this.store = {};
            this.onSerializeCallbacks = {};
        }
        /**
         * \@internal
         * @param {?} initState
         * @return {?}
         */
        static init(initState) {
            /** @type {?} */
            const transferState = new TransferState();
            transferState.store = initState;
            return transferState;
        }
        /**
         * Get the value corresponding to a key. Return `defaultValue` if key is not found.
         * @template T
         * @param {?} key
         * @param {?} defaultValue
         * @return {?}
         */
        get(key, defaultValue) {
            return this.store[key] !== undefined ? (/** @type {?} */ (this.store[key])) : defaultValue;
        }
        /**
         * Set the value corresponding to a key.
         * @template T
         * @param {?} key
         * @param {?} value
         * @return {?}
         */
        set(key, value) {
            this.store[key] = value;
        }
        /**
         * Remove a key from the store.
         * @template T
         * @param {?} key
         * @return {?}
         */
        remove(key) {
            delete this.store[key];
        }
        /**
         * Test whether a key exists in the store.
         * @template T
         * @param {?} key
         * @return {?}
         */
        hasKey(key) {
            return this.store.hasOwnProperty(key);
        }
        /**
         * Register a callback to provide the value for a key when `toJson` is called.
         * @template T
         * @param {?} key
         * @param {?} callback
         * @return {?}
         */
        onSerialize(key, callback) {
            this.onSerializeCallbacks[key] = callback;
        }
        /**
         * Serialize the current state of the store to JSON.
         * @return {?}
         */
        toJson() {
            // Call the onSerialize callbacks and put those values into the store.
            for (const key in this.onSerializeCallbacks) {
                if (this.onSerializeCallbacks.hasOwnProperty(key)) {
                    try {
                        this.store[key] = this.onSerializeCallbacks[key]();
                    }
                    catch (e) {
                        console.warn('Exception in onSerialize callback: ', e);
                    }
                }
            }
            return JSON.stringify(this.store);
        }
    }
    TransferState.ɵfac = function TransferState_Factory(t) { return new (t || TransferState)(); };
    TransferState.ɵprov = ɵɵdefineInjectable({ token: TransferState, factory: TransferState.ɵfac });
    /**
     * @param {?} doc
     * @param {?} appId
     * @return {?}
     */
    function initTransferState(doc, appId) {
        // Locate the script tag with the JSON data transferred from the server.
        // The id of the script tag is set to the Angular appId + 'state'.
        /** @type {?} */
        const script = doc.getElementById(appId + '-state');
        /** @type {?} */
        let initialState = {};
        if (script && script.textContent) {
            try {
                initialState = JSON.parse(unescapeHtml(script.textContent));
            }
            catch (e) {
                console.warn('Exception while restoring TransferState for app ' + appId, e);
            }
        }
        return TransferState.init(initialState);
    }
    /**
     * NgModule to install on the client side while using the `TransferState` to transfer state from
     * server to client.
     *
     * \@publicApi
     */
    class BrowserTransferStateModule {
    }
    BrowserTransferStateModule.ɵmod = ɵɵdefineNgModule({ type: BrowserTransferStateModule });
    BrowserTransferStateModule.ɵinj = ɵɵdefineInjector({ factory: function BrowserTransferStateModule_Factory(t) { return new (t || BrowserTransferStateModule)(); }, providers: [{ provide: TransferState, useFactory: initTransferState, deps: [DOCUMENT$1, APP_ID] }] });

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/platform-browser/src/private_export.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/platform-browser/src/version.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * \@publicApi
     * @type {?}
     */
    const VERSION$2 = new Version('9.1.12');

    var commonjsGlobal = typeof globalThis !== 'undefined' ? globalThis : typeof window !== 'undefined' ? window : typeof global !== 'undefined' ? global : typeof self !== 'undefined' ? self : {};

    /**
    * @license Angular v9.1.0-next.4+61.sha-e552591.with-local-changes
    * (c) 2010-2020 Google LLC. https://angular.io/
    * License: MIT
    */
    (function (factory) {
        
            factory();
    }((function () {
        /**
         * @license
         * Copyright Google Inc. All Rights Reserved.
         *
         * Use of this source code is governed by an MIT-style license that can be
         * found in the LICENSE file at https://angular.io/license
         */
        var Zone$1 = (function (global) {
            var performance = global['performance'];
            function mark(name) { performance && performance['mark'] && performance['mark'](name); }
            function performanceMeasure(name, label) {
                performance && performance['measure'] && performance['measure'](name, label);
            }
            mark('Zone');
            // Initialize before it's accessed below.
            // __Zone_symbol_prefix global can be used to override the default zone
            // symbol prefix with a custom one if needed.
            var symbolPrefix = global['__Zone_symbol_prefix'] || '__zone_symbol__';
            function __symbol__(name) { return symbolPrefix + name; }
            var checkDuplicate = global[__symbol__('forceDuplicateZoneCheck')] === true;
            if (global['Zone']) {
                // if global['Zone'] already exists (maybe zone.js was already loaded or
                // some other lib also registered a global object named Zone), we may need
                // to throw an error, but sometimes user may not want this error.
                // For example,
                // we have two web pages, page1 includes zone.js, page2 doesn't.
                // and the 1st time user load page1 and page2, everything work fine,
                // but when user load page2 again, error occurs because global['Zone'] already exists.
                // so we add a flag to let user choose whether to throw this error or not.
                // By default, if existing Zone is from zone.js, we will not throw the error.
                if (checkDuplicate || typeof global['Zone'].__symbol__ !== 'function') {
                    throw new Error('Zone already loaded.');
                }
                else {
                    return global['Zone'];
                }
            }
            var Zone = /** @class */ (function () {
                function Zone(parent, zoneSpec) {
                    this._parent = parent;
                    this._name = zoneSpec ? zoneSpec.name || 'unnamed' : '<root>';
                    this._properties = zoneSpec && zoneSpec.properties || {};
                    this._zoneDelegate =
                        new ZoneDelegate(this, this._parent && this._parent._zoneDelegate, zoneSpec);
                }
                Zone.assertZonePatched = function () {
                    if (global['Promise'] !== patches['ZoneAwarePromise']) {
                        throw new Error('Zone.js has detected that ZoneAwarePromise `(window|global).Promise` ' +
                            'has been overwritten.\n' +
                            'Most likely cause is that a Promise polyfill has been loaded ' +
                            'after Zone.js (Polyfilling Promise api is not necessary when zone.js is loaded. ' +
                            'If you must load one, do so before loading zone.js.)');
                    }
                };
                Object.defineProperty(Zone, "root", {
                    get: function () {
                        var zone = Zone.current;
                        while (zone.parent) {
                            zone = zone.parent;
                        }
                        return zone;
                    },
                    enumerable: true,
                    configurable: true
                });
                Object.defineProperty(Zone, "current", {
                    get: function () { return _currentZoneFrame.zone; },
                    enumerable: true,
                    configurable: true
                });
                Object.defineProperty(Zone, "currentTask", {
                    get: function () { return _currentTask; },
                    enumerable: true,
                    configurable: true
                });
                // tslint:disable-next-line:require-internal-with-underscore
                Zone.__load_patch = function (name, fn) {
                    if (patches.hasOwnProperty(name)) {
                        if (checkDuplicate) {
                            throw Error('Already loaded patch: ' + name);
                        }
                    }
                    else if (!global['__Zone_disable_' + name]) {
                        var perfName = 'Zone:' + name;
                        mark(perfName);
                        patches[name] = fn(global, Zone, _api);
                        performanceMeasure(perfName, perfName);
                    }
                };
                Object.defineProperty(Zone.prototype, "parent", {
                    get: function () { return this._parent; },
                    enumerable: true,
                    configurable: true
                });
                Object.defineProperty(Zone.prototype, "name", {
                    get: function () { return this._name; },
                    enumerable: true,
                    configurable: true
                });
                Zone.prototype.get = function (key) {
                    var zone = this.getZoneWith(key);
                    if (zone)
                        return zone._properties[key];
                };
                Zone.prototype.getZoneWith = function (key) {
                    var current = this;
                    while (current) {
                        if (current._properties.hasOwnProperty(key)) {
                            return current;
                        }
                        current = current._parent;
                    }
                    return null;
                };
                Zone.prototype.fork = function (zoneSpec) {
                    if (!zoneSpec)
                        throw new Error('ZoneSpec required!');
                    return this._zoneDelegate.fork(this, zoneSpec);
                };
                Zone.prototype.wrap = function (callback, source) {
                    if (typeof callback !== 'function') {
                        throw new Error('Expecting function got: ' + callback);
                    }
                    var _callback = this._zoneDelegate.intercept(this, callback, source);
                    var zone = this;
                    return function () {
                        return zone.runGuarded(_callback, this, arguments, source);
                    };
                };
                Zone.prototype.run = function (callback, applyThis, applyArgs, source) {
                    _currentZoneFrame = { parent: _currentZoneFrame, zone: this };
                    try {
                        return this._zoneDelegate.invoke(this, callback, applyThis, applyArgs, source);
                    }
                    finally {
                        _currentZoneFrame = _currentZoneFrame.parent;
                    }
                };
                Zone.prototype.runGuarded = function (callback, applyThis, applyArgs, source) {
                    if (applyThis === void 0) { applyThis = null; }
                    _currentZoneFrame = { parent: _currentZoneFrame, zone: this };
                    try {
                        try {
                            return this._zoneDelegate.invoke(this, callback, applyThis, applyArgs, source);
                        }
                        catch (error) {
                            if (this._zoneDelegate.handleError(this, error)) {
                                throw error;
                            }
                        }
                    }
                    finally {
                        _currentZoneFrame = _currentZoneFrame.parent;
                    }
                };
                Zone.prototype.runTask = function (task, applyThis, applyArgs) {
                    if (task.zone != this) {
                        throw new Error('A task can only be run in the zone of creation! (Creation: ' +
                            (task.zone || NO_ZONE).name + '; Execution: ' + this.name + ')');
                    }
                    // https://github.com/angular/zone.js/issues/778, sometimes eventTask
                    // will run in notScheduled(canceled) state, we should not try to
                    // run such kind of task but just return
                    if (task.state === notScheduled && (task.type === eventTask || task.type === macroTask)) {
                        return;
                    }
                    var reEntryGuard = task.state != running;
                    reEntryGuard && task._transitionTo(running, scheduled);
                    task.runCount++;
                    var previousTask = _currentTask;
                    _currentTask = task;
                    _currentZoneFrame = { parent: _currentZoneFrame, zone: this };
                    try {
                        if (task.type == macroTask && task.data && !task.data.isPeriodic) {
                            task.cancelFn = undefined;
                        }
                        try {
                            return this._zoneDelegate.invokeTask(this, task, applyThis, applyArgs);
                        }
                        catch (error) {
                            if (this._zoneDelegate.handleError(this, error)) {
                                throw error;
                            }
                        }
                    }
                    finally {
                        // if the task's state is notScheduled or unknown, then it has already been cancelled
                        // we should not reset the state to scheduled
                        if (task.state !== notScheduled && task.state !== unknown) {
                            if (task.type == eventTask || (task.data && task.data.isPeriodic)) {
                                reEntryGuard && task._transitionTo(scheduled, running);
                            }
                            else {
                                task.runCount = 0;
                                this._updateTaskCount(task, -1);
                                reEntryGuard &&
                                    task._transitionTo(notScheduled, running, notScheduled);
                            }
                        }
                        _currentZoneFrame = _currentZoneFrame.parent;
                        _currentTask = previousTask;
                    }
                };
                Zone.prototype.scheduleTask = function (task) {
                    if (task.zone && task.zone !== this) {
                        // check if the task was rescheduled, the newZone
                        // should not be the children of the original zone
                        var newZone = this;
                        while (newZone) {
                            if (newZone === task.zone) {
                                throw Error("can not reschedule task to " + this.name + " which is descendants of the original zone " + task.zone.name);
                            }
                            newZone = newZone.parent;
                        }
                    }
                    task._transitionTo(scheduling, notScheduled);
                    var zoneDelegates = [];
                    task._zoneDelegates = zoneDelegates;
                    task._zone = this;
                    try {
                        task = this._zoneDelegate.scheduleTask(this, task);
                    }
                    catch (err) {
                        // should set task's state to unknown when scheduleTask throw error
                        // because the err may from reschedule, so the fromState maybe notScheduled
                        task._transitionTo(unknown, scheduling, notScheduled);
                        // TODO: @JiaLiPassion, should we check the result from handleError?
                        this._zoneDelegate.handleError(this, err);
                        throw err;
                    }
                    if (task._zoneDelegates === zoneDelegates) {
                        // we have to check because internally the delegate can reschedule the task.
                        this._updateTaskCount(task, 1);
                    }
                    if (task.state == scheduling) {
                        task._transitionTo(scheduled, scheduling);
                    }
                    return task;
                };
                Zone.prototype.scheduleMicroTask = function (source, callback, data, customSchedule) {
                    return this.scheduleTask(new ZoneTask(microTask, source, callback, data, customSchedule, undefined));
                };
                Zone.prototype.scheduleMacroTask = function (source, callback, data, customSchedule, customCancel) {
                    return this.scheduleTask(new ZoneTask(macroTask, source, callback, data, customSchedule, customCancel));
                };
                Zone.prototype.scheduleEventTask = function (source, callback, data, customSchedule, customCancel) {
                    return this.scheduleTask(new ZoneTask(eventTask, source, callback, data, customSchedule, customCancel));
                };
                Zone.prototype.cancelTask = function (task) {
                    if (task.zone != this)
                        throw new Error('A task can only be cancelled in the zone of creation! (Creation: ' +
                            (task.zone || NO_ZONE).name + '; Execution: ' + this.name + ')');
                    task._transitionTo(canceling, scheduled, running);
                    try {
                        this._zoneDelegate.cancelTask(this, task);
                    }
                    catch (err) {
                        // if error occurs when cancelTask, transit the state to unknown
                        task._transitionTo(unknown, canceling);
                        this._zoneDelegate.handleError(this, err);
                        throw err;
                    }
                    this._updateTaskCount(task, -1);
                    task._transitionTo(notScheduled, canceling);
                    task.runCount = 0;
                    return task;
                };
                Zone.prototype._updateTaskCount = function (task, count) {
                    var zoneDelegates = task._zoneDelegates;
                    if (count == -1) {
                        task._zoneDelegates = null;
                    }
                    for (var i = 0; i < zoneDelegates.length; i++) {
                        zoneDelegates[i]._updateTaskCount(task.type, count);
                    }
                };
                return Zone;
            }());
            // tslint:disable-next-line:require-internal-with-underscore
            Zone.__symbol__ = __symbol__;
            var DELEGATE_ZS = {
                name: '',
                onHasTask: function (delegate, _, target, hasTaskState) { return delegate.hasTask(target, hasTaskState); },
                onScheduleTask: function (delegate, _, target, task) { return delegate.scheduleTask(target, task); },
                onInvokeTask: function (delegate, _, target, task, applyThis, applyArgs) { return delegate.invokeTask(target, task, applyThis, applyArgs); },
                onCancelTask: function (delegate, _, target, task) { return delegate.cancelTask(target, task); }
            };
            var ZoneDelegate = /** @class */ (function () {
                function ZoneDelegate(zone, parentDelegate, zoneSpec) {
                    this._taskCounts = { 'microTask': 0, 'macroTask': 0, 'eventTask': 0 };
                    this.zone = zone;
                    this._parentDelegate = parentDelegate;
                    this._forkZS =
                        zoneSpec && (zoneSpec && zoneSpec.onFork ? zoneSpec : parentDelegate._forkZS);
                    this._forkDlgt = zoneSpec && (zoneSpec.onFork ? parentDelegate : parentDelegate._forkDlgt);
                    this._forkCurrZone =
                        zoneSpec && (zoneSpec.onFork ? this.zone : parentDelegate._forkCurrZone);
                    this._interceptZS =
                        zoneSpec && (zoneSpec.onIntercept ? zoneSpec : parentDelegate._interceptZS);
                    this._interceptDlgt =
                        zoneSpec && (zoneSpec.onIntercept ? parentDelegate : parentDelegate._interceptDlgt);
                    this._interceptCurrZone =
                        zoneSpec && (zoneSpec.onIntercept ? this.zone : parentDelegate._interceptCurrZone);
                    this._invokeZS = zoneSpec && (zoneSpec.onInvoke ? zoneSpec : parentDelegate._invokeZS);
                    this._invokeDlgt =
                        zoneSpec && (zoneSpec.onInvoke ? parentDelegate : parentDelegate._invokeDlgt);
                    this._invokeCurrZone =
                        zoneSpec && (zoneSpec.onInvoke ? this.zone : parentDelegate._invokeCurrZone);
                    this._handleErrorZS =
                        zoneSpec && (zoneSpec.onHandleError ? zoneSpec : parentDelegate._handleErrorZS);
                    this._handleErrorDlgt = zoneSpec &&
                        (zoneSpec.onHandleError ? parentDelegate : parentDelegate._handleErrorDlgt);
                    this._handleErrorCurrZone =
                        zoneSpec && (zoneSpec.onHandleError ? this.zone : parentDelegate._handleErrorCurrZone);
                    this._scheduleTaskZS =
                        zoneSpec && (zoneSpec.onScheduleTask ? zoneSpec : parentDelegate._scheduleTaskZS);
                    this._scheduleTaskDlgt = zoneSpec &&
                        (zoneSpec.onScheduleTask ? parentDelegate : parentDelegate._scheduleTaskDlgt);
                    this._scheduleTaskCurrZone = zoneSpec &&
                        (zoneSpec.onScheduleTask ? this.zone : parentDelegate._scheduleTaskCurrZone);
                    this._invokeTaskZS =
                        zoneSpec && (zoneSpec.onInvokeTask ? zoneSpec : parentDelegate._invokeTaskZS);
                    this._invokeTaskDlgt =
                        zoneSpec && (zoneSpec.onInvokeTask ? parentDelegate : parentDelegate._invokeTaskDlgt);
                    this._invokeTaskCurrZone =
                        zoneSpec && (zoneSpec.onInvokeTask ? this.zone : parentDelegate._invokeTaskCurrZone);
                    this._cancelTaskZS =
                        zoneSpec && (zoneSpec.onCancelTask ? zoneSpec : parentDelegate._cancelTaskZS);
                    this._cancelTaskDlgt =
                        zoneSpec && (zoneSpec.onCancelTask ? parentDelegate : parentDelegate._cancelTaskDlgt);
                    this._cancelTaskCurrZone =
                        zoneSpec && (zoneSpec.onCancelTask ? this.zone : parentDelegate._cancelTaskCurrZone);
                    this._hasTaskZS = null;
                    this._hasTaskDlgt = null;
                    this._hasTaskDlgtOwner = null;
                    this._hasTaskCurrZone = null;
                    var zoneSpecHasTask = zoneSpec && zoneSpec.onHasTask;
                    var parentHasTask = parentDelegate && parentDelegate._hasTaskZS;
                    if (zoneSpecHasTask || parentHasTask) {
                        // If we need to report hasTask, than this ZS needs to do ref counting on tasks. In such
                        // a case all task related interceptors must go through this ZD. We can't short circuit it.
                        this._hasTaskZS = zoneSpecHasTask ? zoneSpec : DELEGATE_ZS;
                        this._hasTaskDlgt = parentDelegate;
                        this._hasTaskDlgtOwner = this;
                        this._hasTaskCurrZone = zone;
                        if (!zoneSpec.onScheduleTask) {
                            this._scheduleTaskZS = DELEGATE_ZS;
                            this._scheduleTaskDlgt = parentDelegate;
                            this._scheduleTaskCurrZone = this.zone;
                        }
                        if (!zoneSpec.onInvokeTask) {
                            this._invokeTaskZS = DELEGATE_ZS;
                            this._invokeTaskDlgt = parentDelegate;
                            this._invokeTaskCurrZone = this.zone;
                        }
                        if (!zoneSpec.onCancelTask) {
                            this._cancelTaskZS = DELEGATE_ZS;
                            this._cancelTaskDlgt = parentDelegate;
                            this._cancelTaskCurrZone = this.zone;
                        }
                    }
                }
                ZoneDelegate.prototype.fork = function (targetZone, zoneSpec) {
                    return this._forkZS ?
                        this._forkZS.onFork(this._forkDlgt, this.zone, targetZone, zoneSpec) :
                        new Zone(targetZone, zoneSpec);
                };
                ZoneDelegate.prototype.intercept = function (targetZone, callback, source) {
                    return this._interceptZS ?
                        this._interceptZS.onIntercept(this._interceptDlgt, this._interceptCurrZone, targetZone, callback, source) :
                        callback;
                };
                ZoneDelegate.prototype.invoke = function (targetZone, callback, applyThis, applyArgs, source) {
                    return this._invokeZS ?
                        this._invokeZS.onInvoke(this._invokeDlgt, this._invokeCurrZone, targetZone, callback, applyThis, applyArgs, source) :
                        callback.apply(applyThis, applyArgs);
                };
                ZoneDelegate.prototype.handleError = function (targetZone, error) {
                    return this._handleErrorZS ?
                        this._handleErrorZS.onHandleError(this._handleErrorDlgt, this._handleErrorCurrZone, targetZone, error) :
                        true;
                };
                ZoneDelegate.prototype.scheduleTask = function (targetZone, task) {
                    var returnTask = task;
                    if (this._scheduleTaskZS) {
                        if (this._hasTaskZS) {
                            returnTask._zoneDelegates.push(this._hasTaskDlgtOwner);
                        }
                        // clang-format off
                        returnTask = this._scheduleTaskZS.onScheduleTask(this._scheduleTaskDlgt, this._scheduleTaskCurrZone, targetZone, task);
                        // clang-format on
                        if (!returnTask)
                            returnTask = task;
                    }
                    else {
                        if (task.scheduleFn) {
                            task.scheduleFn(task);
                        }
                        else if (task.type == microTask) {
                            scheduleMicroTask(task);
                        }
                        else {
                            throw new Error('Task is missing scheduleFn.');
                        }
                    }
                    return returnTask;
                };
                ZoneDelegate.prototype.invokeTask = function (targetZone, task, applyThis, applyArgs) {
                    return this._invokeTaskZS ?
                        this._invokeTaskZS.onInvokeTask(this._invokeTaskDlgt, this._invokeTaskCurrZone, targetZone, task, applyThis, applyArgs) :
                        task.callback.apply(applyThis, applyArgs);
                };
                ZoneDelegate.prototype.cancelTask = function (targetZone, task) {
                    var value;
                    if (this._cancelTaskZS) {
                        value = this._cancelTaskZS.onCancelTask(this._cancelTaskDlgt, this._cancelTaskCurrZone, targetZone, task);
                    }
                    else {
                        if (!task.cancelFn) {
                            throw Error('Task is not cancelable');
                        }
                        value = task.cancelFn(task);
                    }
                    return value;
                };
                ZoneDelegate.prototype.hasTask = function (targetZone, isEmpty) {
                    // hasTask should not throw error so other ZoneDelegate
                    // can still trigger hasTask callback
                    try {
                        this._hasTaskZS &&
                            this._hasTaskZS.onHasTask(this._hasTaskDlgt, this._hasTaskCurrZone, targetZone, isEmpty);
                    }
                    catch (err) {
                        this.handleError(targetZone, err);
                    }
                };
                // tslint:disable-next-line:require-internal-with-underscore
                ZoneDelegate.prototype._updateTaskCount = function (type, count) {
                    var counts = this._taskCounts;
                    var prev = counts[type];
                    var next = counts[type] = prev + count;
                    if (next < 0) {
                        throw new Error('More tasks executed then were scheduled.');
                    }
                    if (prev == 0 || next == 0) {
                        var isEmpty = {
                            microTask: counts['microTask'] > 0,
                            macroTask: counts['macroTask'] > 0,
                            eventTask: counts['eventTask'] > 0,
                            change: type
                        };
                        this.hasTask(this.zone, isEmpty);
                    }
                };
                return ZoneDelegate;
            }());
            var ZoneTask = /** @class */ (function () {
                function ZoneTask(type, source, callback, options, scheduleFn, cancelFn) {
                    // tslint:disable-next-line:require-internal-with-underscore
                    this._zone = null;
                    this.runCount = 0;
                    // tslint:disable-next-line:require-internal-with-underscore
                    this._zoneDelegates = null;
                    // tslint:disable-next-line:require-internal-with-underscore
                    this._state = 'notScheduled';
                    this.type = type;
                    this.source = source;
                    this.data = options;
                    this.scheduleFn = scheduleFn;
                    this.cancelFn = cancelFn;
                    if (!callback) {
                        throw new Error('callback is not defined');
                    }
                    this.callback = callback;
                    var self = this;
                    // TODO: @JiaLiPassion options should have interface
                    if (type === eventTask && options && options.useG) {
                        this.invoke = ZoneTask.invokeTask;
                    }
                    else {
                        this.invoke = function () {
                            return ZoneTask.invokeTask.call(global, self, this, arguments);
                        };
                    }
                }
                ZoneTask.invokeTask = function (task, target, args) {
                    if (!task) {
                        task = this;
                    }
                    _numberOfNestedTaskFrames++;
                    try {
                        task.runCount++;
                        return task.zone.runTask(task, target, args);
                    }
                    finally {
                        if (_numberOfNestedTaskFrames == 1) {
                            drainMicroTaskQueue();
                        }
                        _numberOfNestedTaskFrames--;
                    }
                };
                Object.defineProperty(ZoneTask.prototype, "zone", {
                    get: function () { return this._zone; },
                    enumerable: true,
                    configurable: true
                });
                Object.defineProperty(ZoneTask.prototype, "state", {
                    get: function () { return this._state; },
                    enumerable: true,
                    configurable: true
                });
                ZoneTask.prototype.cancelScheduleRequest = function () { this._transitionTo(notScheduled, scheduling); };
                // tslint:disable-next-line:require-internal-with-underscore
                ZoneTask.prototype._transitionTo = function (toState, fromState1, fromState2) {
                    if (this._state === fromState1 || this._state === fromState2) {
                        this._state = toState;
                        if (toState == notScheduled) {
                            this._zoneDelegates = null;
                        }
                    }
                    else {
                        throw new Error(this.type + " '" + this.source + "': can not transition to '" + toState + "', expecting state '" + fromState1 + "'" + (fromState2 ? ' or \'' + fromState2 + '\'' : '') + ", was '" + this._state + "'.");
                    }
                };
                ZoneTask.prototype.toString = function () {
                    if (this.data && typeof this.data.handleId !== 'undefined') {
                        return this.data.handleId.toString();
                    }
                    else {
                        return Object.prototype.toString.call(this);
                    }
                };
                // add toJSON method to prevent cyclic error when
                // call JSON.stringify(zoneTask)
                ZoneTask.prototype.toJSON = function () {
                    return {
                        type: this.type,
                        state: this.state,
                        source: this.source,
                        zone: this.zone.name,
                        runCount: this.runCount
                    };
                };
                return ZoneTask;
            }());
            //////////////////////////////////////////////////////
            //////////////////////////////////////////////////////
            ///  MICROTASK QUEUE
            //////////////////////////////////////////////////////
            //////////////////////////////////////////////////////
            var symbolSetTimeout = __symbol__('setTimeout');
            var symbolPromise = __symbol__('Promise');
            var symbolThen = __symbol__('then');
            var _microTaskQueue = [];
            var _isDrainingMicrotaskQueue = false;
            var nativeMicroTaskQueuePromise;
            function scheduleMicroTask(task) {
                // if we are not running in any task, and there has not been anything scheduled
                // we must bootstrap the initial task creation by manually scheduling the drain
                if (_numberOfNestedTaskFrames === 0 && _microTaskQueue.length === 0) {
                    // We are not running in Task, so we need to kickstart the microtask queue.
                    if (!nativeMicroTaskQueuePromise) {
                        if (global[symbolPromise]) {
                            nativeMicroTaskQueuePromise = global[symbolPromise].resolve(0);
                        }
                    }
                    if (nativeMicroTaskQueuePromise) {
                        var nativeThen = nativeMicroTaskQueuePromise[symbolThen];
                        if (!nativeThen) {
                            // native Promise is not patchable, we need to use `then` directly
                            // issue 1078
                            nativeThen = nativeMicroTaskQueuePromise['then'];
                        }
                        nativeThen.call(nativeMicroTaskQueuePromise, drainMicroTaskQueue);
                    }
                    else {
                        global[symbolSetTimeout](drainMicroTaskQueue, 0);
                    }
                }
                task && _microTaskQueue.push(task);
            }
            function drainMicroTaskQueue() {
                if (!_isDrainingMicrotaskQueue) {
                    _isDrainingMicrotaskQueue = true;
                    while (_microTaskQueue.length) {
                        var queue = _microTaskQueue;
                        _microTaskQueue = [];
                        for (var i = 0; i < queue.length; i++) {
                            var task = queue[i];
                            try {
                                task.zone.runTask(task, null, null);
                            }
                            catch (error) {
                                _api.onUnhandledError(error);
                            }
                        }
                    }
                    _api.microtaskDrainDone();
                    _isDrainingMicrotaskQueue = false;
                }
            }
            //////////////////////////////////////////////////////
            //////////////////////////////////////////////////////
            ///  BOOTSTRAP
            //////////////////////////////////////////////////////
            //////////////////////////////////////////////////////
            var NO_ZONE = { name: 'NO ZONE' };
            var notScheduled = 'notScheduled', scheduling = 'scheduling', scheduled = 'scheduled', running = 'running', canceling = 'canceling', unknown = 'unknown';
            var microTask = 'microTask', macroTask = 'macroTask', eventTask = 'eventTask';
            var patches = {};
            var _api = {
                symbol: __symbol__,
                currentZoneFrame: function () { return _currentZoneFrame; },
                onUnhandledError: noop,
                microtaskDrainDone: noop,
                scheduleMicroTask: scheduleMicroTask,
                showUncaughtError: function () { return !Zone[__symbol__('ignoreConsoleErrorUncaughtError')]; },
                patchEventTarget: function () { return []; },
                patchOnProperties: noop,
                patchMethod: function () { return noop; },
                bindArguments: function () { return []; },
                patchThen: function () { return noop; },
                patchMacroTask: function () { return noop; },
                setNativePromise: function (NativePromise) {
                    // sometimes NativePromise.resolve static function
                    // is not ready yet, (such as core-js/es6.promise)
                    // so we need to check here.
                    if (NativePromise && typeof NativePromise.resolve === 'function') {
                        nativeMicroTaskQueuePromise = NativePromise.resolve(0);
                    }
                },
                patchEventPrototype: function () { return noop; },
                isIEOrEdge: function () { return false; },
                getGlobalObjects: function () { return undefined; },
                ObjectDefineProperty: function () { return noop; },
                ObjectGetOwnPropertyDescriptor: function () { return undefined; },
                ObjectCreate: function () { return undefined; },
                ArraySlice: function () { return []; },
                patchClass: function () { return noop; },
                wrapWithCurrentZone: function () { return noop; },
                filterProperties: function () { return []; },
                attachOriginToPatched: function () { return noop; },
                _redefineProperty: function () { return noop; },
                patchCallbacks: function () { return noop; }
            };
            var _currentZoneFrame = { parent: null, zone: new Zone(null, null) };
            var _currentTask = null;
            var _numberOfNestedTaskFrames = 0;
            function noop() { }
            performanceMeasure('Zone', 'Zone');
            return global['Zone'] = Zone;
        })(typeof window !== 'undefined' && window || typeof self !== 'undefined' && self || commonjsGlobal);
        /**
         * @license
         * Copyright Google Inc. All Rights Reserved.
         *
         * Use of this source code is governed by an MIT-style license that can be
         * found in the LICENSE file at https://angular.io/license
         */
        Zone.__load_patch('ZoneAwarePromise', function (global, Zone, api) {
            var ObjectGetOwnPropertyDescriptor = Object.getOwnPropertyDescriptor;
            var ObjectDefineProperty = Object.defineProperty;
            function readableObjectToString(obj) {
                if (obj && obj.toString === Object.prototype.toString) {
                    var className = obj.constructor && obj.constructor.name;
                    return (className ? className : '') + ': ' + JSON.stringify(obj);
                }
                return obj ? obj.toString() : Object.prototype.toString.call(obj);
            }
            var __symbol__ = api.symbol;
            var _uncaughtPromiseErrors = [];
            var isDisableWrappingUncaughtPromiseRejection = global[__symbol__('DISABLE_WRAPPING_UNCAUGHT_PROMISE_REJECTION')] === true;
            var symbolPromise = __symbol__('Promise');
            var symbolThen = __symbol__('then');
            var creationTrace = '__creationTrace__';
            api.onUnhandledError = function (e) {
                if (api.showUncaughtError()) {
                    var rejection = e && e.rejection;
                    if (rejection) {
                        console.error('Unhandled Promise rejection:', rejection instanceof Error ? rejection.message : rejection, '; Zone:', e.zone.name, '; Task:', e.task && e.task.source, '; Value:', rejection, rejection instanceof Error ? rejection.stack : undefined);
                    }
                    else {
                        console.error(e);
                    }
                }
            };
            api.microtaskDrainDone = function () {
                var _loop_1 = function () {
                    var uncaughtPromiseError = _uncaughtPromiseErrors.shift();
                    try {
                        uncaughtPromiseError.zone.runGuarded(function () { throw uncaughtPromiseError; });
                    }
                    catch (error) {
                        handleUnhandledRejection(error);
                    }
                };
                while (_uncaughtPromiseErrors.length) {
                    _loop_1();
                }
            };
            var UNHANDLED_PROMISE_REJECTION_HANDLER_SYMBOL = __symbol__('unhandledPromiseRejectionHandler');
            function handleUnhandledRejection(e) {
                api.onUnhandledError(e);
                try {
                    var handler = Zone[UNHANDLED_PROMISE_REJECTION_HANDLER_SYMBOL];
                    if (typeof handler === 'function') {
                        handler.call(this, e);
                    }
                }
                catch (err) {
                }
            }
            function isThenable(value) { return value && value.then; }
            function forwardResolution(value) { return value; }
            function forwardRejection(rejection) { return ZoneAwarePromise.reject(rejection); }
            var symbolState = __symbol__('state');
            var symbolValue = __symbol__('value');
            var symbolFinally = __symbol__('finally');
            var symbolParentPromiseValue = __symbol__('parentPromiseValue');
            var symbolParentPromiseState = __symbol__('parentPromiseState');
            var source = 'Promise.then';
            var UNRESOLVED = null;
            var RESOLVED = true;
            var REJECTED = false;
            var REJECTED_NO_CATCH = 0;
            function makeResolver(promise, state) {
                return function (v) {
                    try {
                        resolvePromise(promise, state, v);
                    }
                    catch (err) {
                        resolvePromise(promise, false, err);
                    }
                    // Do not return value or you will break the Promise spec.
                };
            }
            var once = function () {
                var wasCalled = false;
                return function wrapper(wrappedFunction) {
                    return function () {
                        if (wasCalled) {
                            return;
                        }
                        wasCalled = true;
                        wrappedFunction.apply(null, arguments);
                    };
                };
            };
            var TYPE_ERROR = 'Promise resolved with itself';
            var CURRENT_TASK_TRACE_SYMBOL = __symbol__('currentTaskTrace');
            // Promise Resolution
            function resolvePromise(promise, state, value) {
                var onceWrapper = once();
                if (promise === value) {
                    throw new TypeError(TYPE_ERROR);
                }
                if (promise[symbolState] === UNRESOLVED) {
                    // should only get value.then once based on promise spec.
                    var then = null;
                    try {
                        if (typeof value === 'object' || typeof value === 'function') {
                            then = value && value.then;
                        }
                    }
                    catch (err) {
                        onceWrapper(function () { resolvePromise(promise, false, err); })();
                        return promise;
                    }
                    // if (value instanceof ZoneAwarePromise) {
                    if (state !== REJECTED && value instanceof ZoneAwarePromise &&
                        value.hasOwnProperty(symbolState) && value.hasOwnProperty(symbolValue) &&
                        value[symbolState] !== UNRESOLVED) {
                        clearRejectedNoCatch(value);
                        resolvePromise(promise, value[symbolState], value[symbolValue]);
                    }
                    else if (state !== REJECTED && typeof then === 'function') {
                        try {
                            then.call(value, onceWrapper(makeResolver(promise, state)), onceWrapper(makeResolver(promise, false)));
                        }
                        catch (err) {
                            onceWrapper(function () { resolvePromise(promise, false, err); })();
                        }
                    }
                    else {
                        promise[symbolState] = state;
                        var queue = promise[symbolValue];
                        promise[symbolValue] = value;
                        if (promise[symbolFinally] === symbolFinally) {
                            // the promise is generated by Promise.prototype.finally
                            if (state === RESOLVED) {
                                // the state is resolved, should ignore the value
                                // and use parent promise value
                                promise[symbolState] = promise[symbolParentPromiseState];
                                promise[symbolValue] = promise[symbolParentPromiseValue];
                            }
                        }
                        // record task information in value when error occurs, so we can
                        // do some additional work such as render longStackTrace
                        if (state === REJECTED && value instanceof Error) {
                            // check if longStackTraceZone is here
                            var trace = Zone.currentTask && Zone.currentTask.data &&
                                Zone.currentTask.data[creationTrace];
                            if (trace) {
                                // only keep the long stack trace into error when in longStackTraceZone
                                ObjectDefineProperty(value, CURRENT_TASK_TRACE_SYMBOL, { configurable: true, enumerable: false, writable: true, value: trace });
                            }
                        }
                        for (var i = 0; i < queue.length;) {
                            scheduleResolveOrReject(promise, queue[i++], queue[i++], queue[i++], queue[i++]);
                        }
                        if (queue.length == 0 && state == REJECTED) {
                            promise[symbolState] = REJECTED_NO_CATCH;
                            var uncaughtPromiseError = value;
                            if (!isDisableWrappingUncaughtPromiseRejection) {
                                // If disable wrapping uncaught promise reject
                                // and the rejected value is an Error object,
                                // use the value instead of wrapping it.
                                try {
                                    // Here we throws a new Error to print more readable error log
                                    // and if the value is not an error, zone.js builds an `Error`
                                    // Object here to attach the stack information.
                                    throw new Error('Uncaught (in promise): ' + readableObjectToString(value) +
                                        (value && value.stack ? '\n' + value.stack : ''));
                                }
                                catch (err) {
                                    uncaughtPromiseError = err;
                                }
                            }
                            uncaughtPromiseError.rejection = value;
                            uncaughtPromiseError.promise = promise;
                            uncaughtPromiseError.zone = Zone.current;
                            uncaughtPromiseError.task = Zone.currentTask;
                            _uncaughtPromiseErrors.push(uncaughtPromiseError);
                            api.scheduleMicroTask(); // to make sure that it is running
                        }
                    }
                }
                // Resolving an already resolved promise is a noop.
                return promise;
            }
            var REJECTION_HANDLED_HANDLER = __symbol__('rejectionHandledHandler');
            function clearRejectedNoCatch(promise) {
                if (promise[symbolState] === REJECTED_NO_CATCH) {
                    // if the promise is rejected no catch status
                    // and queue.length > 0, means there is a error handler
                    // here to handle the rejected promise, we should trigger
                    // windows.rejectionhandled eventHandler or nodejs rejectionHandled
                    // eventHandler
                    try {
                        var handler = Zone[REJECTION_HANDLED_HANDLER];
                        if (handler && typeof handler === 'function') {
                            handler.call(this, { rejection: promise[symbolValue], promise: promise });
                        }
                    }
                    catch (err) {
                    }
                    promise[symbolState] = REJECTED;
                    for (var i = 0; i < _uncaughtPromiseErrors.length; i++) {
                        if (promise === _uncaughtPromiseErrors[i].promise) {
                            _uncaughtPromiseErrors.splice(i, 1);
                        }
                    }
                }
            }
            function scheduleResolveOrReject(promise, zone, chainPromise, onFulfilled, onRejected) {
                clearRejectedNoCatch(promise);
                var promiseState = promise[symbolState];
                var delegate = promiseState ?
                    (typeof onFulfilled === 'function') ? onFulfilled : forwardResolution :
                    (typeof onRejected === 'function') ? onRejected : forwardRejection;
                zone.scheduleMicroTask(source, function () {
                    try {
                        var parentPromiseValue = promise[symbolValue];
                        var isFinallyPromise = !!chainPromise && symbolFinally === chainPromise[symbolFinally];
                        if (isFinallyPromise) {
                            // if the promise is generated from finally call, keep parent promise's state and value
                            chainPromise[symbolParentPromiseValue] = parentPromiseValue;
                            chainPromise[symbolParentPromiseState] = promiseState;
                        }
                        // should not pass value to finally callback
                        var value = zone.run(delegate, undefined, isFinallyPromise && delegate !== forwardRejection && delegate !== forwardResolution ?
                            [] :
                            [parentPromiseValue]);
                        resolvePromise(chainPromise, true, value);
                    }
                    catch (error) {
                        // if error occurs, should always return this error
                        resolvePromise(chainPromise, false, error);
                    }
                }, chainPromise);
            }
            var ZONE_AWARE_PROMISE_TO_STRING = 'function ZoneAwarePromise() { [native code] }';
            var noop = function () { };
            var ZoneAwarePromise = /** @class */ (function () {
                function ZoneAwarePromise(executor) {
                    var promise = this;
                    if (!(promise instanceof ZoneAwarePromise)) {
                        throw new Error('Must be an instanceof Promise.');
                    }
                    promise[symbolState] = UNRESOLVED;
                    promise[symbolValue] = []; // queue;
                    try {
                        executor && executor(makeResolver(promise, RESOLVED), makeResolver(promise, REJECTED));
                    }
                    catch (error) {
                        resolvePromise(promise, false, error);
                    }
                }
                ZoneAwarePromise.toString = function () { return ZONE_AWARE_PROMISE_TO_STRING; };
                ZoneAwarePromise.resolve = function (value) {
                    return resolvePromise(new this(null), RESOLVED, value);
                };
                ZoneAwarePromise.reject = function (error) {
                    return resolvePromise(new this(null), REJECTED, error);
                };
                ZoneAwarePromise.race = function (values) {
                    var resolve;
                    var reject;
                    var promise = new this(function (res, rej) {
                        resolve = res;
                        reject = rej;
                    });
                    function onResolve(value) { resolve(value); }
                    function onReject(error) { reject(error); }
                    for (var _i = 0, values_1 = values; _i < values_1.length; _i++) {
                        var value = values_1[_i];
                        if (!isThenable(value)) {
                            value = this.resolve(value);
                        }
                        value.then(onResolve, onReject);
                    }
                    return promise;
                };
                ZoneAwarePromise.all = function (values) { return ZoneAwarePromise.allWithCallback(values); };
                ZoneAwarePromise.allSettled = function (values) {
                    var P = this && this.prototype instanceof ZoneAwarePromise ? this : ZoneAwarePromise;
                    return P.allWithCallback(values, {
                        thenCallback: function (value) { return ({ status: 'fulfilled', value: value }); },
                        errorCallback: function (err) { return ({ status: 'rejected', reason: err }); }
                    });
                };
                ZoneAwarePromise.allWithCallback = function (values, callback) {
                    var resolve;
                    var reject;
                    var promise = new this(function (res, rej) {
                        resolve = res;
                        reject = rej;
                    });
                    // Start at 2 to prevent prematurely resolving if .then is called immediately.
                    var unresolvedCount = 2;
                    var valueIndex = 0;
                    var resolvedValues = [];
                    var _loop_2 = function (value) {
                        if (!isThenable(value)) {
                            value = this_1.resolve(value);
                        }
                        var curValueIndex = valueIndex;
                        try {
                            value.then(function (value) {
                                resolvedValues[curValueIndex] = callback ? callback.thenCallback(value) : value;
                                unresolvedCount--;
                                if (unresolvedCount === 0) {
                                    resolve(resolvedValues);
                                }
                            }, function (err) {
                                if (!callback) {
                                    reject(err);
                                }
                                else {
                                    resolvedValues[curValueIndex] = callback.errorCallback(err);
                                    unresolvedCount--;
                                    if (unresolvedCount === 0) {
                                        resolve(resolvedValues);
                                    }
                                }
                            });
                        }
                        catch (thenErr) {
                            reject(thenErr);
                        }
                        unresolvedCount++;
                        valueIndex++;
                    };
                    var this_1 = this;
                    for (var _i = 0, values_2 = values; _i < values_2.length; _i++) {
                        var value = values_2[_i];
                        _loop_2(value);
                    }
                    // Make the unresolvedCount zero-based again.
                    unresolvedCount -= 2;
                    if (unresolvedCount === 0) {
                        resolve(resolvedValues);
                    }
                    return promise;
                };
                Object.defineProperty(ZoneAwarePromise.prototype, Symbol.toStringTag, {
                    get: function () { return 'Promise'; },
                    enumerable: true,
                    configurable: true
                });
                Object.defineProperty(ZoneAwarePromise.prototype, Symbol.species, {
                    get: function () { return ZoneAwarePromise; },
                    enumerable: true,
                    configurable: true
                });
                ZoneAwarePromise.prototype.then = function (onFulfilled, onRejected) {
                    var C = this.constructor[Symbol.species];
                    if (!C || typeof C !== 'function') {
                        C = this.constructor || ZoneAwarePromise;
                    }
                    var chainPromise = new C(noop);
                    var zone = Zone.current;
                    if (this[symbolState] == UNRESOLVED) {
                        this[symbolValue].push(zone, chainPromise, onFulfilled, onRejected);
                    }
                    else {
                        scheduleResolveOrReject(this, zone, chainPromise, onFulfilled, onRejected);
                    }
                    return chainPromise;
                };
                ZoneAwarePromise.prototype.catch = function (onRejected) {
                    return this.then(null, onRejected);
                };
                ZoneAwarePromise.prototype.finally = function (onFinally) {
                    var C = this.constructor[Symbol.species];
                    if (!C || typeof C !== 'function') {
                        C = ZoneAwarePromise;
                    }
                    var chainPromise = new C(noop);
                    chainPromise[symbolFinally] = symbolFinally;
                    var zone = Zone.current;
                    if (this[symbolState] == UNRESOLVED) {
                        this[symbolValue].push(zone, chainPromise, onFinally, onFinally);
                    }
                    else {
                        scheduleResolveOrReject(this, zone, chainPromise, onFinally, onFinally);
                    }
                    return chainPromise;
                };
                return ZoneAwarePromise;
            }());
            // Protect against aggressive optimizers dropping seemingly unused properties.
            // E.g. Closure Compiler in advanced mode.
            ZoneAwarePromise['resolve'] = ZoneAwarePromise.resolve;
            ZoneAwarePromise['reject'] = ZoneAwarePromise.reject;
            ZoneAwarePromise['race'] = ZoneAwarePromise.race;
            ZoneAwarePromise['all'] = ZoneAwarePromise.all;
            var NativePromise = global[symbolPromise] = global['Promise'];
            var ZONE_AWARE_PROMISE = Zone.__symbol__('ZoneAwarePromise');
            var desc = ObjectGetOwnPropertyDescriptor(global, 'Promise');
            if (!desc || desc.configurable) {
                desc && delete desc.writable;
                desc && delete desc.value;
                if (!desc) {
                    desc = { configurable: true, enumerable: true };
                }
                desc.get = function () {
                    // if we already set ZoneAwarePromise, use patched one
                    // otherwise return native one.
                    return global[ZONE_AWARE_PROMISE] ? global[ZONE_AWARE_PROMISE] : global[symbolPromise];
                };
                desc.set = function (NewNativePromise) {
                    if (NewNativePromise === ZoneAwarePromise) {
                        // if the NewNativePromise is ZoneAwarePromise
                        // save to global
                        global[ZONE_AWARE_PROMISE] = NewNativePromise;
                    }
                    else {
                        // if the NewNativePromise is not ZoneAwarePromise
                        // for example: after load zone.js, some library just
                        // set es6-promise to global, if we set it to global
                        // directly, assertZonePatched will fail and angular
                        // will not loaded, so we just set the NewNativePromise
                        // to global[symbolPromise], so the result is just like
                        // we load ES6 Promise before zone.js
                        global[symbolPromise] = NewNativePromise;
                        if (!NewNativePromise.prototype[symbolThen]) {
                            patchThen(NewNativePromise);
                        }
                        api.setNativePromise(NewNativePromise);
                    }
                };
                ObjectDefineProperty(global, 'Promise', desc);
            }
            global['Promise'] = ZoneAwarePromise;
            var symbolThenPatched = __symbol__('thenPatched');
            function patchThen(Ctor) {
                var proto = Ctor.prototype;
                var prop = ObjectGetOwnPropertyDescriptor(proto, 'then');
                if (prop && (prop.writable === false || !prop.configurable)) {
                    // check Ctor.prototype.then propertyDescriptor is writable or not
                    // in meteor env, writable is false, we should ignore such case
                    return;
                }
                var originalThen = proto.then;
                // Keep a reference to the original method.
                proto[symbolThen] = originalThen;
                Ctor.prototype.then = function (onResolve, onReject) {
                    var _this = this;
                    var wrapped = new ZoneAwarePromise(function (resolve, reject) { originalThen.call(_this, resolve, reject); });
                    return wrapped.then(onResolve, onReject);
                };
                Ctor[symbolThenPatched] = true;
            }
            api.patchThen = patchThen;
            function zoneify(fn) {
                return function () {
                    var resultPromise = fn.apply(this, arguments);
                    if (resultPromise instanceof ZoneAwarePromise) {
                        return resultPromise;
                    }
                    var ctor = resultPromise.constructor;
                    if (!ctor[symbolThenPatched]) {
                        patchThen(ctor);
                    }
                    return resultPromise;
                };
            }
            if (NativePromise) {
                patchThen(NativePromise);
                var fetch_1 = global['fetch'];
                if (typeof fetch_1 == 'function') {
                    global[api.symbol('fetch')] = fetch_1;
                    global['fetch'] = zoneify(fetch_1);
                }
            }
            // This is not part of public API, but it is useful for tests, so we expose it.
            Promise[Zone.__symbol__('uncaughtPromiseErrors')] = _uncaughtPromiseErrors;
            return ZoneAwarePromise;
        });
        /**
         * @license
         * Copyright Google Inc. All Rights Reserved.
         *
         * Use of this source code is governed by an MIT-style license that can be
         * found in the LICENSE file at https://angular.io/license
         */
        /**
         * Suppress closure compiler errors about unknown 'Zone' variable
         * @fileoverview
         * @suppress {undefinedVars,globalThis,missingRequire}
         */
        /// <reference types="node"/>
        // issue #989, to reduce bundle size, use short name
        /** Object.getOwnPropertyDescriptor */
        var ObjectGetOwnPropertyDescriptor = Object.getOwnPropertyDescriptor;
        /** Object.defineProperty */
        var ObjectDefineProperty = Object.defineProperty;
        /** Object.getPrototypeOf */
        var ObjectGetPrototypeOf = Object.getPrototypeOf;
        /** Object.create */
        var ObjectCreate = Object.create;
        /** Array.prototype.slice */
        var ArraySlice = Array.prototype.slice;
        /** addEventListener string const */
        var ADD_EVENT_LISTENER_STR = 'addEventListener';
        /** removeEventListener string const */
        var REMOVE_EVENT_LISTENER_STR = 'removeEventListener';
        /** zoneSymbol addEventListener */
        var ZONE_SYMBOL_ADD_EVENT_LISTENER = Zone.__symbol__(ADD_EVENT_LISTENER_STR);
        /** zoneSymbol removeEventListener */
        var ZONE_SYMBOL_REMOVE_EVENT_LISTENER = Zone.__symbol__(REMOVE_EVENT_LISTENER_STR);
        /** true string const */
        var TRUE_STR = 'true';
        /** false string const */
        var FALSE_STR = 'false';
        /** Zone symbol prefix string const. */
        var ZONE_SYMBOL_PREFIX = Zone.__symbol__('');
        function wrapWithCurrentZone(callback, source) {
            return Zone.current.wrap(callback, source);
        }
        function scheduleMacroTaskWithCurrentZone(source, callback, data, customSchedule, customCancel) {
            return Zone.current.scheduleMacroTask(source, callback, data, customSchedule, customCancel);
        }
        var zoneSymbol = Zone.__symbol__;
        var isWindowExists = typeof window !== 'undefined';
        var internalWindow = isWindowExists ? window : undefined;
        var _global = isWindowExists && internalWindow || typeof self === 'object' && self || commonjsGlobal;
        var REMOVE_ATTRIBUTE = 'removeAttribute';
        var NULL_ON_PROP_VALUE = [null];
        function bindArguments(args, source) {
            for (var i = args.length - 1; i >= 0; i--) {
                if (typeof args[i] === 'function') {
                    args[i] = wrapWithCurrentZone(args[i], source + '_' + i);
                }
            }
            return args;
        }
        function patchPrototype(prototype, fnNames) {
            var source = prototype.constructor['name'];
            var _loop_3 = function (i) {
                var name_1 = fnNames[i];
                var delegate = prototype[name_1];
                if (delegate) {
                    var prototypeDesc = ObjectGetOwnPropertyDescriptor(prototype, name_1);
                    if (!isPropertyWritable(prototypeDesc)) {
                        return "continue";
                    }
                    prototype[name_1] = (function (delegate) {
                        var patched = function () {
                            return delegate.apply(this, bindArguments(arguments, source + '.' + name_1));
                        };
                        attachOriginToPatched(patched, delegate);
                        return patched;
                    })(delegate);
                }
            };
            for (var i = 0; i < fnNames.length; i++) {
                _loop_3(i);
            }
        }
        function isPropertyWritable(propertyDesc) {
            if (!propertyDesc) {
                return true;
            }
            if (propertyDesc.writable === false) {
                return false;
            }
            return !(typeof propertyDesc.get === 'function' && typeof propertyDesc.set === 'undefined');
        }
        var isWebWorker = (typeof WorkerGlobalScope !== 'undefined' && self instanceof WorkerGlobalScope);
        // Make sure to access `process` through `_global` so that WebPack does not accidentally browserify
        // this code.
        var isNode = (!('nw' in _global) && typeof _global.process !== 'undefined' &&
            {}.toString.call(_global.process) === '[object process]');
        var isBrowser = !isNode && !isWebWorker && !!(isWindowExists && internalWindow['HTMLElement']);
        // we are in electron of nw, so we are both browser and nodejs
        // Make sure to access `process` through `_global` so that WebPack does not accidentally browserify
        // this code.
        var isMix = typeof _global.process !== 'undefined' &&
            {}.toString.call(_global.process) === '[object process]' && !isWebWorker &&
            !!(isWindowExists && internalWindow['HTMLElement']);
        var zoneSymbolEventNames = {};
        var wrapFn = function (event) {
            // https://github.com/angular/zone.js/issues/911, in IE, sometimes
            // event will be undefined, so we need to use window.event
            event = event || _global.event;
            if (!event) {
                return;
            }
            var eventNameSymbol = zoneSymbolEventNames[event.type];
            if (!eventNameSymbol) {
                eventNameSymbol = zoneSymbolEventNames[event.type] = zoneSymbol('ON_PROPERTY' + event.type);
            }
            var target = this || event.target || _global;
            var listener = target[eventNameSymbol];
            var result;
            if (isBrowser && target === internalWindow && event.type === 'error') {
                // window.onerror have different signiture
                // https://developer.mozilla.org/en-US/docs/Web/API/GlobalEventHandlers/onerror#window.onerror
                // and onerror callback will prevent default when callback return true
                var errorEvent = event;
                result = listener &&
                    listener.call(this, errorEvent.message, errorEvent.filename, errorEvent.lineno, errorEvent.colno, errorEvent.error);
                if (result === true) {
                    event.preventDefault();
                }
            }
            else {
                result = listener && listener.apply(this, arguments);
                if (result != undefined && !result) {
                    event.preventDefault();
                }
            }
            return result;
        };
        function patchProperty(obj, prop, prototype) {
            var desc = ObjectGetOwnPropertyDescriptor(obj, prop);
            if (!desc && prototype) {
                // when patch window object, use prototype to check prop exist or not
                var prototypeDesc = ObjectGetOwnPropertyDescriptor(prototype, prop);
                if (prototypeDesc) {
                    desc = { enumerable: true, configurable: true };
                }
            }
            // if the descriptor not exists or is not configurable
            // just return
            if (!desc || !desc.configurable) {
                return;
            }
            var onPropPatchedSymbol = zoneSymbol('on' + prop + 'patched');
            if (obj.hasOwnProperty(onPropPatchedSymbol) && obj[onPropPatchedSymbol]) {
                return;
            }
            // A property descriptor cannot have getter/setter and be writable
            // deleting the writable and value properties avoids this error:
            //
            // TypeError: property descriptors must not specify a value or be writable when a
            // getter or setter has been specified
            delete desc.writable;
            delete desc.value;
            var originalDescGet = desc.get;
            var originalDescSet = desc.set;
            // substr(2) cuz 'onclick' -> 'click', etc
            var eventName = prop.substr(2);
            var eventNameSymbol = zoneSymbolEventNames[eventName];
            if (!eventNameSymbol) {
                eventNameSymbol = zoneSymbolEventNames[eventName] = zoneSymbol('ON_PROPERTY' + eventName);
            }
            desc.set = function (newValue) {
                // in some of windows's onproperty callback, this is undefined
                // so we need to check it
                var target = this;
                if (!target && obj === _global) {
                    target = _global;
                }
                if (!target) {
                    return;
                }
                var previousValue = target[eventNameSymbol];
                if (previousValue) {
                    target.removeEventListener(eventName, wrapFn);
                }
                // issue #978, when onload handler was added before loading zone.js
                // we should remove it with originalDescSet
                if (originalDescSet) {
                    originalDescSet.apply(target, NULL_ON_PROP_VALUE);
                }
                if (typeof newValue === 'function') {
                    target[eventNameSymbol] = newValue;
                    target.addEventListener(eventName, wrapFn, false);
                }
                else {
                    target[eventNameSymbol] = null;
                }
            };
            // The getter would return undefined for unassigned properties but the default value of an
            // unassigned property is null
            desc.get = function () {
                // in some of windows's onproperty callback, this is undefined
                // so we need to check it
                var target = this;
                if (!target && obj === _global) {
                    target = _global;
                }
                if (!target) {
                    return null;
                }
                var listener = target[eventNameSymbol];
                if (listener) {
                    return listener;
                }
                else if (originalDescGet) {
                    // result will be null when use inline event attribute,
                    // such as <button onclick="func();">OK</button>
                    // because the onclick function is internal raw uncompiled handler
                    // the onclick will be evaluated when first time event was triggered or
                    // the property is accessed, https://github.com/angular/zone.js/issues/525
                    // so we should use original native get to retrieve the handler
                    var value = originalDescGet && originalDescGet.call(this);
                    if (value) {
                        desc.set.call(this, value);
                        if (typeof target[REMOVE_ATTRIBUTE] === 'function') {
                            target.removeAttribute(prop);
                        }
                        return value;
                    }
                }
                return null;
            };
            ObjectDefineProperty(obj, prop, desc);
            obj[onPropPatchedSymbol] = true;
        }
        function patchOnProperties(obj, properties, prototype) {
            if (properties) {
                for (var i = 0; i < properties.length; i++) {
                    patchProperty(obj, 'on' + properties[i], prototype);
                }
            }
            else {
                var onProperties = [];
                for (var prop in obj) {
                    if (prop.substr(0, 2) == 'on') {
                        onProperties.push(prop);
                    }
                }
                for (var j = 0; j < onProperties.length; j++) {
                    patchProperty(obj, onProperties[j], prototype);
                }
            }
        }
        var originalInstanceKey = zoneSymbol('originalInstance');
        // wrap some native API on `window`
        function patchClass(className) {
            var OriginalClass = _global[className];
            if (!OriginalClass)
                return;
            // keep original class in global
            _global[zoneSymbol(className)] = OriginalClass;
            _global[className] = function () {
                var a = bindArguments(arguments, className);
                switch (a.length) {
                    case 0:
                        this[originalInstanceKey] = new OriginalClass();
                        break;
                    case 1:
                        this[originalInstanceKey] = new OriginalClass(a[0]);
                        break;
                    case 2:
                        this[originalInstanceKey] = new OriginalClass(a[0], a[1]);
                        break;
                    case 3:
                        this[originalInstanceKey] = new OriginalClass(a[0], a[1], a[2]);
                        break;
                    case 4:
                        this[originalInstanceKey] = new OriginalClass(a[0], a[1], a[2], a[3]);
                        break;
                    default:
                        throw new Error('Arg list too long.');
                }
            };
            // attach original delegate to patched function
            attachOriginToPatched(_global[className], OriginalClass);
            var instance = new OriginalClass(function () { });
            var prop;
            for (prop in instance) {
                // https://bugs.webkit.org/show_bug.cgi?id=44721
                if (className === 'XMLHttpRequest' && prop === 'responseBlob')
                    continue;
                (function (prop) {
                    if (typeof instance[prop] === 'function') {
                        _global[className].prototype[prop] = function () {
                            return this[originalInstanceKey][prop].apply(this[originalInstanceKey], arguments);
                        };
                    }
                    else {
                        ObjectDefineProperty(_global[className].prototype, prop, {
                            set: function (fn) {
                                if (typeof fn === 'function') {
                                    this[originalInstanceKey][prop] = wrapWithCurrentZone(fn, className + '.' + prop);
                                    // keep callback in wrapped function so we can
                                    // use it in Function.prototype.toString to return
                                    // the native one.
                                    attachOriginToPatched(this[originalInstanceKey][prop], fn);
                                }
                                else {
                                    this[originalInstanceKey][prop] = fn;
                                }
                            },
                            get: function () { return this[originalInstanceKey][prop]; }
                        });
                    }
                }(prop));
            }
            for (prop in OriginalClass) {
                if (prop !== 'prototype' && OriginalClass.hasOwnProperty(prop)) {
                    _global[className][prop] = OriginalClass[prop];
                }
            }
        }
        function patchMethod(target, name, patchFn) {
            var proto = target;
            while (proto && !proto.hasOwnProperty(name)) {
                proto = ObjectGetPrototypeOf(proto);
            }
            if (!proto && target[name]) {
                // somehow we did not find it, but we can see it. This happens on IE for Window properties.
                proto = target;
            }
            var delegateName = zoneSymbol(name);
            var delegate = null;
            if (proto && !(delegate = proto[delegateName])) {
                delegate = proto[delegateName] = proto[name];
                // check whether proto[name] is writable
                // some property is readonly in safari, such as HtmlCanvasElement.prototype.toBlob
                var desc = proto && ObjectGetOwnPropertyDescriptor(proto, name);
                if (isPropertyWritable(desc)) {
                    var patchDelegate_1 = patchFn(delegate, delegateName, name);
                    proto[name] = function () { return patchDelegate_1(this, arguments); };
                    attachOriginToPatched(proto[name], delegate);
                }
            }
            return delegate;
        }
        // TODO: @JiaLiPassion, support cancel task later if necessary
        function patchMacroTask(obj, funcName, metaCreator) {
            var setNative = null;
            function scheduleTask(task) {
                var data = task.data;
                data.args[data.cbIdx] = function () { task.invoke.apply(this, arguments); };
                setNative.apply(data.target, data.args);
                return task;
            }
            setNative = patchMethod(obj, funcName, function (delegate) { return function (self, args) {
                var meta = metaCreator(self, args);
                if (meta.cbIdx >= 0 && typeof args[meta.cbIdx] === 'function') {
                    return scheduleMacroTaskWithCurrentZone(meta.name, args[meta.cbIdx], meta, scheduleTask);
                }
                else {
                    // cause an error by calling it directly.
                    return delegate.apply(self, args);
                }
            }; });
        }
        function attachOriginToPatched(patched, original) {
            patched[zoneSymbol('OriginalDelegate')] = original;
        }
        var isDetectedIEOrEdge = false;
        var ieOrEdge = false;
        function isIE() {
            try {
                var ua = internalWindow.navigator.userAgent;
                if (ua.indexOf('MSIE ') !== -1 || ua.indexOf('Trident/') !== -1) {
                    return true;
                }
            }
            catch (error) {
            }
            return false;
        }
        function isIEOrEdge() {
            if (isDetectedIEOrEdge) {
                return ieOrEdge;
            }
            isDetectedIEOrEdge = true;
            try {
                var ua = internalWindow.navigator.userAgent;
                if (ua.indexOf('MSIE ') !== -1 || ua.indexOf('Trident/') !== -1 || ua.indexOf('Edge/') !== -1) {
                    ieOrEdge = true;
                }
            }
            catch (error) {
            }
            return ieOrEdge;
        }
        /**
         * @license
         * Copyright Google Inc. All Rights Reserved.
         *
         * Use of this source code is governed by an MIT-style license that can be
         * found in the LICENSE file at https://angular.io/license
         */
        // override Function.prototype.toString to make zone.js patched function
        // look like native function
        Zone.__load_patch('toString', function (global) {
            // patch Func.prototype.toString to let them look like native
            var originalFunctionToString = Function.prototype.toString;
            var ORIGINAL_DELEGATE_SYMBOL = zoneSymbol('OriginalDelegate');
            var PROMISE_SYMBOL = zoneSymbol('Promise');
            var ERROR_SYMBOL = zoneSymbol('Error');
            var newFunctionToString = function toString() {
                if (typeof this === 'function') {
                    var originalDelegate = this[ORIGINAL_DELEGATE_SYMBOL];
                    if (originalDelegate) {
                        if (typeof originalDelegate === 'function') {
                            return originalFunctionToString.call(originalDelegate);
                        }
                        else {
                            return Object.prototype.toString.call(originalDelegate);
                        }
                    }
                    if (this === Promise) {
                        var nativePromise = global[PROMISE_SYMBOL];
                        if (nativePromise) {
                            return originalFunctionToString.call(nativePromise);
                        }
                    }
                    if (this === Error) {
                        var nativeError = global[ERROR_SYMBOL];
                        if (nativeError) {
                            return originalFunctionToString.call(nativeError);
                        }
                    }
                }
                return originalFunctionToString.call(this);
            };
            newFunctionToString[ORIGINAL_DELEGATE_SYMBOL] = originalFunctionToString;
            Function.prototype.toString = newFunctionToString;
            // patch Object.prototype.toString to let them look like native
            var originalObjectToString = Object.prototype.toString;
            var PROMISE_OBJECT_TO_STRING = '[object Promise]';
            Object.prototype.toString = function () {
                if (this instanceof Promise) {
                    return PROMISE_OBJECT_TO_STRING;
                }
                return originalObjectToString.call(this);
            };
        });
        /**
         * @license
         * Copyright Google Inc. All Rights Reserved.
         *
         * Use of this source code is governed by an MIT-style license that can be
         * found in the LICENSE file at https://angular.io/license
         */
        var passiveSupported = false;
        if (typeof window !== 'undefined') {
            try {
                var options = Object.defineProperty({}, 'passive', { get: function () { passiveSupported = true; } });
                window.addEventListener('test', options, options);
                window.removeEventListener('test', options, options);
            }
            catch (err) {
                passiveSupported = false;
            }
        }
        // an identifier to tell ZoneTask do not create a new invoke closure
        var OPTIMIZED_ZONE_EVENT_TASK_DATA = {
            useG: true
        };
        var zoneSymbolEventNames$1 = {};
        var globalSources = {};
        var EVENT_NAME_SYMBOL_REGX = new RegExp('^' + ZONE_SYMBOL_PREFIX + '(\\w+)(true|false)$');
        var IMMEDIATE_PROPAGATION_SYMBOL = zoneSymbol('propagationStopped');
        function prepareEventNames(eventName, eventNameToString) {
            var falseEventName = (eventNameToString ? eventNameToString(eventName) : eventName) + FALSE_STR;
            var trueEventName = (eventNameToString ? eventNameToString(eventName) : eventName) + TRUE_STR;
            var symbol = ZONE_SYMBOL_PREFIX + falseEventName;
            var symbolCapture = ZONE_SYMBOL_PREFIX + trueEventName;
            zoneSymbolEventNames$1[eventName] = {};
            zoneSymbolEventNames$1[eventName][FALSE_STR] = symbol;
            zoneSymbolEventNames$1[eventName][TRUE_STR] = symbolCapture;
        }
        function patchEventTarget(_global, apis, patchOptions) {
            var ADD_EVENT_LISTENER = (patchOptions && patchOptions.add) || ADD_EVENT_LISTENER_STR;
            var REMOVE_EVENT_LISTENER = (patchOptions && patchOptions.rm) || REMOVE_EVENT_LISTENER_STR;
            var LISTENERS_EVENT_LISTENER = (patchOptions && patchOptions.listeners) || 'eventListeners';
            var REMOVE_ALL_LISTENERS_EVENT_LISTENER = (patchOptions && patchOptions.rmAll) || 'removeAllListeners';
            var zoneSymbolAddEventListener = zoneSymbol(ADD_EVENT_LISTENER);
            var ADD_EVENT_LISTENER_SOURCE = '.' + ADD_EVENT_LISTENER + ':';
            var PREPEND_EVENT_LISTENER = 'prependListener';
            var PREPEND_EVENT_LISTENER_SOURCE = '.' + PREPEND_EVENT_LISTENER + ':';
            var invokeTask = function (task, target, event) {
                // for better performance, check isRemoved which is set
                // by removeEventListener
                if (task.isRemoved) {
                    return;
                }
                var delegate = task.callback;
                if (typeof delegate === 'object' && delegate.handleEvent) {
                    // create the bind version of handleEvent when invoke
                    task.callback = function (event) { return delegate.handleEvent(event); };
                    task.originalDelegate = delegate;
                }
                // invoke static task.invoke
                task.invoke(task, target, [event]);
                var options = task.options;
                if (options && typeof options === 'object' && options.once) {
                    // if options.once is true, after invoke once remove listener here
                    // only browser need to do this, nodejs eventEmitter will cal removeListener
                    // inside EventEmitter.once
                    var delegate_1 = task.originalDelegate ? task.originalDelegate : task.callback;
                    target[REMOVE_EVENT_LISTENER].call(target, event.type, delegate_1, options);
                }
            };
            // global shared zoneAwareCallback to handle all event callback with capture = false
            var globalZoneAwareCallback = function (event) {
                // https://github.com/angular/zone.js/issues/911, in IE, sometimes
                // event will be undefined, so we need to use window.event
                event = event || _global.event;
                if (!event) {
                    return;
                }
                // event.target is needed for Samsung TV and SourceBuffer
                // || global is needed https://github.com/angular/zone.js/issues/190
                var target = this || event.target || _global;
                var tasks = target[zoneSymbolEventNames$1[event.type][FALSE_STR]];
                if (tasks) {
                    // invoke all tasks which attached to current target with given event.type and capture = false
                    // for performance concern, if task.length === 1, just invoke
                    if (tasks.length === 1) {
                        invokeTask(tasks[0], target, event);
                    }
                    else {
                        // https://github.com/angular/zone.js/issues/836
                        // copy the tasks array before invoke, to avoid
                        // the callback will remove itself or other listener
                        var copyTasks = tasks.slice();
                        for (var i = 0; i < copyTasks.length; i++) {
                            if (event && event[IMMEDIATE_PROPAGATION_SYMBOL] === true) {
                                break;
                            }
                            invokeTask(copyTasks[i], target, event);
                        }
                    }
                }
            };
            // global shared zoneAwareCallback to handle all event callback with capture = true
            var globalZoneAwareCaptureCallback = function (event) {
                // https://github.com/angular/zone.js/issues/911, in IE, sometimes
                // event will be undefined, so we need to use window.event
                event = event || _global.event;
                if (!event) {
                    return;
                }
                // event.target is needed for Samsung TV and SourceBuffer
                // || global is needed https://github.com/angular/zone.js/issues/190
                var target = this || event.target || _global;
                var tasks = target[zoneSymbolEventNames$1[event.type][TRUE_STR]];
                if (tasks) {
                    // invoke all tasks which attached to current target with given event.type and capture = false
                    // for performance concern, if task.length === 1, just invoke
                    if (tasks.length === 1) {
                        invokeTask(tasks[0], target, event);
                    }
                    else {
                        // https://github.com/angular/zone.js/issues/836
                        // copy the tasks array before invoke, to avoid
                        // the callback will remove itself or other listener
                        var copyTasks = tasks.slice();
                        for (var i = 0; i < copyTasks.length; i++) {
                            if (event && event[IMMEDIATE_PROPAGATION_SYMBOL] === true) {
                                break;
                            }
                            invokeTask(copyTasks[i], target, event);
                        }
                    }
                }
            };
            function patchEventTargetMethods(obj, patchOptions) {
                if (!obj) {
                    return false;
                }
                var useGlobalCallback = true;
                if (patchOptions && patchOptions.useG !== undefined) {
                    useGlobalCallback = patchOptions.useG;
                }
                var validateHandler = patchOptions && patchOptions.vh;
                var checkDuplicate = true;
                if (patchOptions && patchOptions.chkDup !== undefined) {
                    checkDuplicate = patchOptions.chkDup;
                }
                var returnTarget = false;
                if (patchOptions && patchOptions.rt !== undefined) {
                    returnTarget = patchOptions.rt;
                }
                var proto = obj;
                while (proto && !proto.hasOwnProperty(ADD_EVENT_LISTENER)) {
                    proto = ObjectGetPrototypeOf(proto);
                }
                if (!proto && obj[ADD_EVENT_LISTENER]) {
                    // somehow we did not find it, but we can see it. This happens on IE for Window properties.
                    proto = obj;
                }
                if (!proto) {
                    return false;
                }
                if (proto[zoneSymbolAddEventListener]) {
                    return false;
                }
                var eventNameToString = patchOptions && patchOptions.eventNameToString;
                // a shared global taskData to pass data for scheduleEventTask
                // so we do not need to create a new object just for pass some data
                var taskData = {};
                var nativeAddEventListener = proto[zoneSymbolAddEventListener] = proto[ADD_EVENT_LISTENER];
                var nativeRemoveEventListener = proto[zoneSymbol(REMOVE_EVENT_LISTENER)] =
                    proto[REMOVE_EVENT_LISTENER];
                var nativeListeners = proto[zoneSymbol(LISTENERS_EVENT_LISTENER)] =
                    proto[LISTENERS_EVENT_LISTENER];
                var nativeRemoveAllListeners = proto[zoneSymbol(REMOVE_ALL_LISTENERS_EVENT_LISTENER)] =
                    proto[REMOVE_ALL_LISTENERS_EVENT_LISTENER];
                var nativePrependEventListener;
                if (patchOptions && patchOptions.prepend) {
                    nativePrependEventListener = proto[zoneSymbol(patchOptions.prepend)] =
                        proto[patchOptions.prepend];
                }
                /**
                 * This util function will build an option object with passive option
                 * to handle all possible input from the user.
                 */
                function buildEventListenerOptions(options, passive) {
                    if (!passiveSupported && typeof options === 'object' && options) {
                        // doesn't support passive but user want to pass an object as options.
                        // this will not work on some old browser, so we just pass a boolean
                        // as useCapture parameter
                        return !!options.capture;
                    }
                    if (!passiveSupported || !passive) {
                        return options;
                    }
                    if (typeof options === 'boolean') {
                        return { capture: options, passive: true };
                    }
                    if (!options) {
                        return { passive: true };
                    }
                    if (typeof options === 'object' && options.passive !== false) {
                        return Object.assign(Object.assign({}, options), { passive: true });
                    }
                    return options;
                }
                var customScheduleGlobal = function (task) {
                    // if there is already a task for the eventName + capture,
                    // just return, because we use the shared globalZoneAwareCallback here.
                    if (taskData.isExisting) {
                        return;
                    }
                    return nativeAddEventListener.call(taskData.target, taskData.eventName, taskData.capture ? globalZoneAwareCaptureCallback : globalZoneAwareCallback, taskData.options);
                };
                var customCancelGlobal = function (task) {
                    // if task is not marked as isRemoved, this call is directly
                    // from Zone.prototype.cancelTask, we should remove the task
                    // from tasksList of target first
                    if (!task.isRemoved) {
                        var symbolEventNames = zoneSymbolEventNames$1[task.eventName];
                        var symbolEventName = void 0;
                        if (symbolEventNames) {
                            symbolEventName = symbolEventNames[task.capture ? TRUE_STR : FALSE_STR];
                        }
                        var existingTasks = symbolEventName && task.target[symbolEventName];
                        if (existingTasks) {
                            for (var i = 0; i < existingTasks.length; i++) {
                                var existingTask = existingTasks[i];
                                if (existingTask === task) {
                                    existingTasks.splice(i, 1);
                                    // set isRemoved to data for faster invokeTask check
                                    task.isRemoved = true;
                                    if (existingTasks.length === 0) {
                                        // all tasks for the eventName + capture have gone,
                                        // remove globalZoneAwareCallback and remove the task cache from target
                                        task.allRemoved = true;
                                        task.target[symbolEventName] = null;
                                    }
                                    break;
                                }
                            }
                        }
                    }
                    // if all tasks for the eventName + capture have gone,
                    // we will really remove the global event callback,
                    // if not, return
                    if (!task.allRemoved) {
                        return;
                    }
                    return nativeRemoveEventListener.call(task.target, task.eventName, task.capture ? globalZoneAwareCaptureCallback : globalZoneAwareCallback, task.options);
                };
                var customScheduleNonGlobal = function (task) {
                    return nativeAddEventListener.call(taskData.target, taskData.eventName, task.invoke, taskData.options);
                };
                var customSchedulePrepend = function (task) {
                    return nativePrependEventListener.call(taskData.target, taskData.eventName, task.invoke, taskData.options);
                };
                var customCancelNonGlobal = function (task) {
                    return nativeRemoveEventListener.call(task.target, task.eventName, task.invoke, task.options);
                };
                var customSchedule = useGlobalCallback ? customScheduleGlobal : customScheduleNonGlobal;
                var customCancel = useGlobalCallback ? customCancelGlobal : customCancelNonGlobal;
                var compareTaskCallbackVsDelegate = function (task, delegate) {
                    var typeOfDelegate = typeof delegate;
                    return (typeOfDelegate === 'function' && task.callback === delegate) ||
                        (typeOfDelegate === 'object' && task.originalDelegate === delegate);
                };
                var compare = (patchOptions && patchOptions.diff) ? patchOptions.diff : compareTaskCallbackVsDelegate;
                var blackListedEvents = Zone[zoneSymbol('BLACK_LISTED_EVENTS')];
                var passiveEvents = _global[zoneSymbol('PASSIVE_EVENTS')];
                var makeAddListener = function (nativeListener, addSource, customScheduleFn, customCancelFn, returnTarget, prepend) {
                    if (returnTarget === void 0) { returnTarget = false; }
                    if (prepend === void 0) { prepend = false; }
                    return function () {
                        var target = this || _global;
                        var eventName = arguments[0];
                        if (patchOptions && patchOptions.transferEventName) {
                            eventName = patchOptions.transferEventName(eventName);
                        }
                        var delegate = arguments[1];
                        if (!delegate) {
                            return nativeListener.apply(this, arguments);
                        }
                        if (isNode && eventName === 'uncaughtException') {
                            // don't patch uncaughtException of nodejs to prevent endless loop
                            return nativeListener.apply(this, arguments);
                        }
                        // don't create the bind delegate function for handleEvent
                        // case here to improve addEventListener performance
                        // we will create the bind delegate when invoke
                        var isHandleEvent = false;
                        if (typeof delegate !== 'function') {
                            if (!delegate.handleEvent) {
                                return nativeListener.apply(this, arguments);
                            }
                            isHandleEvent = true;
                        }
                        if (validateHandler && !validateHandler(nativeListener, delegate, target, arguments)) {
                            return;
                        }
                        var passive = passiveSupported && !!passiveEvents && passiveEvents.indexOf(eventName) !== -1;
                        var options = buildEventListenerOptions(arguments[2], passive);
                        if (blackListedEvents) {
                            // check black list
                            for (var i = 0; i < blackListedEvents.length; i++) {
                                if (eventName === blackListedEvents[i]) {
                                    if (passive) {
                                        return nativeListener.call(target, eventName, delegate, options);
                                    }
                                    else {
                                        return nativeListener.apply(this, arguments);
                                    }
                                }
                            }
                        }
                        var capture = !options ? false : typeof options === 'boolean' ? true : options.capture;
                        var once = options && typeof options === 'object' ? options.once : false;
                        var zone = Zone.current;
                        var symbolEventNames = zoneSymbolEventNames$1[eventName];
                        if (!symbolEventNames) {
                            prepareEventNames(eventName, eventNameToString);
                            symbolEventNames = zoneSymbolEventNames$1[eventName];
                        }
                        var symbolEventName = symbolEventNames[capture ? TRUE_STR : FALSE_STR];
                        var existingTasks = target[symbolEventName];
                        var isExisting = false;
                        if (existingTasks) {
                            // already have task registered
                            isExisting = true;
                            if (checkDuplicate) {
                                for (var i = 0; i < existingTasks.length; i++) {
                                    if (compare(existingTasks[i], delegate)) {
                                        // same callback, same capture, same event name, just return
                                        return;
                                    }
                                }
                            }
                        }
                        else {
                            existingTasks = target[symbolEventName] = [];
                        }
                        var source;
                        var constructorName = target.constructor['name'];
                        var targetSource = globalSources[constructorName];
                        if (targetSource) {
                            source = targetSource[eventName];
                        }
                        if (!source) {
                            source = constructorName + addSource +
                                (eventNameToString ? eventNameToString(eventName) : eventName);
                        }
                        // do not create a new object as task.data to pass those things
                        // just use the global shared one
                        taskData.options = options;
                        if (once) {
                            // if addEventListener with once options, we don't pass it to
                            // native addEventListener, instead we keep the once setting
                            // and handle ourselves.
                            taskData.options.once = false;
                        }
                        taskData.target = target;
                        taskData.capture = capture;
                        taskData.eventName = eventName;
                        taskData.isExisting = isExisting;
                        var data = useGlobalCallback ? OPTIMIZED_ZONE_EVENT_TASK_DATA : undefined;
                        // keep taskData into data to allow onScheduleEventTask to access the task information
                        if (data) {
                            data.taskData = taskData;
                        }
                        var task = zone.scheduleEventTask(source, delegate, data, customScheduleFn, customCancelFn);
                        // should clear taskData.target to avoid memory leak
                        // issue, https://github.com/angular/angular/issues/20442
                        taskData.target = null;
                        // need to clear up taskData because it is a global object
                        if (data) {
                            data.taskData = null;
                        }
                        // have to save those information to task in case
                        // application may call task.zone.cancelTask() directly
                        if (once) {
                            options.once = true;
                        }
                        if (!(!passiveSupported && typeof task.options === 'boolean')) {
                            // if not support passive, and we pass an option object
                            // to addEventListener, we should save the options to task
                            task.options = options;
                        }
                        task.target = target;
                        task.capture = capture;
                        task.eventName = eventName;
                        if (isHandleEvent) {
                            // save original delegate for compare to check duplicate
                            task.originalDelegate = delegate;
                        }
                        if (!prepend) {
                            existingTasks.push(task);
                        }
                        else {
                            existingTasks.unshift(task);
                        }
                        if (returnTarget) {
                            return target;
                        }
                    };
                };
                proto[ADD_EVENT_LISTENER] = makeAddListener(nativeAddEventListener, ADD_EVENT_LISTENER_SOURCE, customSchedule, customCancel, returnTarget);
                if (nativePrependEventListener) {
                    proto[PREPEND_EVENT_LISTENER] = makeAddListener(nativePrependEventListener, PREPEND_EVENT_LISTENER_SOURCE, customSchedulePrepend, customCancel, returnTarget, true);
                }
                proto[REMOVE_EVENT_LISTENER] = function () {
                    var target = this || _global;
                    var eventName = arguments[0];
                    if (patchOptions && patchOptions.transferEventName) {
                        eventName = patchOptions.transferEventName(eventName);
                    }
                    var options = arguments[2];
                    var capture = !options ? false : typeof options === 'boolean' ? true : options.capture;
                    var delegate = arguments[1];
                    if (!delegate) {
                        return nativeRemoveEventListener.apply(this, arguments);
                    }
                    if (validateHandler &&
                        !validateHandler(nativeRemoveEventListener, delegate, target, arguments)) {
                        return;
                    }
                    var symbolEventNames = zoneSymbolEventNames$1[eventName];
                    var symbolEventName;
                    if (symbolEventNames) {
                        symbolEventName = symbolEventNames[capture ? TRUE_STR : FALSE_STR];
                    }
                    var existingTasks = symbolEventName && target[symbolEventName];
                    if (existingTasks) {
                        for (var i = 0; i < existingTasks.length; i++) {
                            var existingTask = existingTasks[i];
                            if (compare(existingTask, delegate)) {
                                existingTasks.splice(i, 1);
                                // set isRemoved to data for faster invokeTask check
                                existingTask.isRemoved = true;
                                if (existingTasks.length === 0) {
                                    // all tasks for the eventName + capture have gone,
                                    // remove globalZoneAwareCallback and remove the task cache from target
                                    existingTask.allRemoved = true;
                                    target[symbolEventName] = null;
                                    // in the target, we have an event listener which is added by on_property
                                    // such as target.onclick = function() {}, so we need to clear this internal
                                    // property too if all delegates all removed
                                    if (typeof eventName === 'string') {
                                        var onPropertySymbol = ZONE_SYMBOL_PREFIX + 'ON_PROPERTY' + eventName;
                                        target[onPropertySymbol] = null;
                                    }
                                }
                                existingTask.zone.cancelTask(existingTask);
                                if (returnTarget) {
                                    return target;
                                }
                                return;
                            }
                        }
                    }
                    // issue 930, didn't find the event name or callback
                    // from zone kept existingTasks, the callback maybe
                    // added outside of zone, we need to call native removeEventListener
                    // to try to remove it.
                    return nativeRemoveEventListener.apply(this, arguments);
                };
                proto[LISTENERS_EVENT_LISTENER] = function () {
                    var target = this || _global;
                    var eventName = arguments[0];
                    if (patchOptions && patchOptions.transferEventName) {
                        eventName = patchOptions.transferEventName(eventName);
                    }
                    var listeners = [];
                    var tasks = findEventTasks(target, eventNameToString ? eventNameToString(eventName) : eventName);
                    for (var i = 0; i < tasks.length; i++) {
                        var task = tasks[i];
                        var delegate = task.originalDelegate ? task.originalDelegate : task.callback;
                        listeners.push(delegate);
                    }
                    return listeners;
                };
                proto[REMOVE_ALL_LISTENERS_EVENT_LISTENER] = function () {
                    var target = this || _global;
                    var eventName = arguments[0];
                    if (!eventName) {
                        var keys = Object.keys(target);
                        for (var i = 0; i < keys.length; i++) {
                            var prop = keys[i];
                            var match = EVENT_NAME_SYMBOL_REGX.exec(prop);
                            var evtName = match && match[1];
                            // in nodejs EventEmitter, removeListener event is
                            // used for monitoring the removeListener call,
                            // so just keep removeListener eventListener until
                            // all other eventListeners are removed
                            if (evtName && evtName !== 'removeListener') {
                                this[REMOVE_ALL_LISTENERS_EVENT_LISTENER].call(this, evtName);
                            }
                        }
                        // remove removeListener listener finally
                        this[REMOVE_ALL_LISTENERS_EVENT_LISTENER].call(this, 'removeListener');
                    }
                    else {
                        if (patchOptions && patchOptions.transferEventName) {
                            eventName = patchOptions.transferEventName(eventName);
                        }
                        var symbolEventNames = zoneSymbolEventNames$1[eventName];
                        if (symbolEventNames) {
                            var symbolEventName = symbolEventNames[FALSE_STR];
                            var symbolCaptureEventName = symbolEventNames[TRUE_STR];
                            var tasks = target[symbolEventName];
                            var captureTasks = target[symbolCaptureEventName];
                            if (tasks) {
                                var removeTasks = tasks.slice();
                                for (var i = 0; i < removeTasks.length; i++) {
                                    var task = removeTasks[i];
                                    var delegate = task.originalDelegate ? task.originalDelegate : task.callback;
                                    this[REMOVE_EVENT_LISTENER].call(this, eventName, delegate, task.options);
                                }
                            }
                            if (captureTasks) {
                                var removeTasks = captureTasks.slice();
                                for (var i = 0; i < removeTasks.length; i++) {
                                    var task = removeTasks[i];
                                    var delegate = task.originalDelegate ? task.originalDelegate : task.callback;
                                    this[REMOVE_EVENT_LISTENER].call(this, eventName, delegate, task.options);
                                }
                            }
                        }
                    }
                    if (returnTarget) {
                        return this;
                    }
                };
                // for native toString patch
                attachOriginToPatched(proto[ADD_EVENT_LISTENER], nativeAddEventListener);
                attachOriginToPatched(proto[REMOVE_EVENT_LISTENER], nativeRemoveEventListener);
                if (nativeRemoveAllListeners) {
                    attachOriginToPatched(proto[REMOVE_ALL_LISTENERS_EVENT_LISTENER], nativeRemoveAllListeners);
                }
                if (nativeListeners) {
                    attachOriginToPatched(proto[LISTENERS_EVENT_LISTENER], nativeListeners);
                }
                return true;
            }
            var results = [];
            for (var i = 0; i < apis.length; i++) {
                results[i] = patchEventTargetMethods(apis[i], patchOptions);
            }
            return results;
        }
        function findEventTasks(target, eventName) {
            if (!eventName) {
                var foundTasks = [];
                for (var prop in target) {
                    var match = EVENT_NAME_SYMBOL_REGX.exec(prop);
                    var evtName = match && match[1];
                    if (evtName && (!eventName || evtName === eventName)) {
                        var tasks = target[prop];
                        if (tasks) {
                            for (var i = 0; i < tasks.length; i++) {
                                foundTasks.push(tasks[i]);
                            }
                        }
                    }
                }
                return foundTasks;
            }
            var symbolEventName = zoneSymbolEventNames$1[eventName];
            if (!symbolEventName) {
                prepareEventNames(eventName);
                symbolEventName = zoneSymbolEventNames$1[eventName];
            }
            var captureFalseTasks = target[symbolEventName[FALSE_STR]];
            var captureTrueTasks = target[symbolEventName[TRUE_STR]];
            if (!captureFalseTasks) {
                return captureTrueTasks ? captureTrueTasks.slice() : [];
            }
            else {
                return captureTrueTasks ? captureFalseTasks.concat(captureTrueTasks) :
                    captureFalseTasks.slice();
            }
        }
        function patchEventPrototype(global, api) {
            var Event = global['Event'];
            if (Event && Event.prototype) {
                api.patchMethod(Event.prototype, 'stopImmediatePropagation', function (delegate) { return function (self, args) {
                    self[IMMEDIATE_PROPAGATION_SYMBOL] = true;
                    // we need to call the native stopImmediatePropagation
                    // in case in some hybrid application, some part of
                    // application will be controlled by zone, some are not
                    delegate && delegate.apply(self, args);
                }; });
            }
        }
        /**
         * @license
         * Copyright Google Inc. All Rights Reserved.
         *
         * Use of this source code is governed by an MIT-style license that can be
         * found in the LICENSE file at https://angular.io/license
         */
        function patchCallbacks(api, target, targetName, method, callbacks) {
            var symbol = Zone.__symbol__(method);
            if (target[symbol]) {
                return;
            }
            var nativeDelegate = target[symbol] = target[method];
            target[method] = function (name, opts, options) {
                if (opts && opts.prototype) {
                    callbacks.forEach(function (callback) {
                        var source = targetName + "." + method + "::" + callback;
                        var prototype = opts.prototype;
                        if (prototype.hasOwnProperty(callback)) {
                            var descriptor = api.ObjectGetOwnPropertyDescriptor(prototype, callback);
                            if (descriptor && descriptor.value) {
                                descriptor.value = api.wrapWithCurrentZone(descriptor.value, source);
                                api._redefineProperty(opts.prototype, callback, descriptor);
                            }
                            else if (prototype[callback]) {
                                prototype[callback] = api.wrapWithCurrentZone(prototype[callback], source);
                            }
                        }
                        else if (prototype[callback]) {
                            prototype[callback] = api.wrapWithCurrentZone(prototype[callback], source);
                        }
                    });
                }
                return nativeDelegate.call(target, name, opts, options);
            };
            api.attachOriginToPatched(target[method], nativeDelegate);
        }
        /**
         * @license
         * Copyright Google Inc. All Rights Reserved.
         *
         * Use of this source code is governed by an MIT-style license that can be
         * found in the LICENSE file at https://angular.io/license
         */
        var globalEventHandlersEventNames = [
            'abort',
            'animationcancel',
            'animationend',
            'animationiteration',
            'auxclick',
            'beforeinput',
            'blur',
            'cancel',
            'canplay',
            'canplaythrough',
            'change',
            'compositionstart',
            'compositionupdate',
            'compositionend',
            'cuechange',
            'click',
            'close',
            'contextmenu',
            'curechange',
            'dblclick',
            'drag',
            'dragend',
            'dragenter',
            'dragexit',
            'dragleave',
            'dragover',
            'drop',
            'durationchange',
            'emptied',
            'ended',
            'error',
            'focus',
            'focusin',
            'focusout',
            'gotpointercapture',
            'input',
            'invalid',
            'keydown',
            'keypress',
            'keyup',
            'load',
            'loadstart',
            'loadeddata',
            'loadedmetadata',
            'lostpointercapture',
            'mousedown',
            'mouseenter',
            'mouseleave',
            'mousemove',
            'mouseout',
            'mouseover',
            'mouseup',
            'mousewheel',
            'orientationchange',
            'pause',
            'play',
            'playing',
            'pointercancel',
            'pointerdown',
            'pointerenter',
            'pointerleave',
            'pointerlockchange',
            'mozpointerlockchange',
            'webkitpointerlockerchange',
            'pointerlockerror',
            'mozpointerlockerror',
            'webkitpointerlockerror',
            'pointermove',
            'pointout',
            'pointerover',
            'pointerup',
            'progress',
            'ratechange',
            'reset',
            'resize',
            'scroll',
            'seeked',
            'seeking',
            'select',
            'selectionchange',
            'selectstart',
            'show',
            'sort',
            'stalled',
            'submit',
            'suspend',
            'timeupdate',
            'volumechange',
            'touchcancel',
            'touchmove',
            'touchstart',
            'touchend',
            'transitioncancel',
            'transitionend',
            'waiting',
            'wheel'
        ];
        var documentEventNames = [
            'afterscriptexecute', 'beforescriptexecute', 'DOMContentLoaded', 'freeze', 'fullscreenchange',
            'mozfullscreenchange', 'webkitfullscreenchange', 'msfullscreenchange', 'fullscreenerror',
            'mozfullscreenerror', 'webkitfullscreenerror', 'msfullscreenerror', 'readystatechange',
            'visibilitychange', 'resume'
        ];
        var windowEventNames = [
            'absolutedeviceorientation',
            'afterinput',
            'afterprint',
            'appinstalled',
            'beforeinstallprompt',
            'beforeprint',
            'beforeunload',
            'devicelight',
            'devicemotion',
            'deviceorientation',
            'deviceorientationabsolute',
            'deviceproximity',
            'hashchange',
            'languagechange',
            'message',
            'mozbeforepaint',
            'offline',
            'online',
            'paint',
            'pageshow',
            'pagehide',
            'popstate',
            'rejectionhandled',
            'storage',
            'unhandledrejection',
            'unload',
            'userproximity',
            'vrdisplayconnected',
            'vrdisplaydisconnected',
            'vrdisplaypresentchange'
        ];
        var htmlElementEventNames = [
            'beforecopy', 'beforecut', 'beforepaste', 'copy', 'cut', 'paste', 'dragstart', 'loadend',
            'animationstart', 'search', 'transitionrun', 'transitionstart', 'webkitanimationend',
            'webkitanimationiteration', 'webkitanimationstart', 'webkittransitionend'
        ];
        var mediaElementEventNames = ['encrypted', 'waitingforkey', 'msneedkey', 'mozinterruptbegin', 'mozinterruptend'];
        var ieElementEventNames = [
            'activate',
            'afterupdate',
            'ariarequest',
            'beforeactivate',
            'beforedeactivate',
            'beforeeditfocus',
            'beforeupdate',
            'cellchange',
            'controlselect',
            'dataavailable',
            'datasetchanged',
            'datasetcomplete',
            'errorupdate',
            'filterchange',
            'layoutcomplete',
            'losecapture',
            'move',
            'moveend',
            'movestart',
            'propertychange',
            'resizeend',
            'resizestart',
            'rowenter',
            'rowexit',
            'rowsdelete',
            'rowsinserted',
            'command',
            'compassneedscalibration',
            'deactivate',
            'help',
            'mscontentzoom',
            'msmanipulationstatechanged',
            'msgesturechange',
            'msgesturedoubletap',
            'msgestureend',
            'msgesturehold',
            'msgesturestart',
            'msgesturetap',
            'msgotpointercapture',
            'msinertiastart',
            'mslostpointercapture',
            'mspointercancel',
            'mspointerdown',
            'mspointerenter',
            'mspointerhover',
            'mspointerleave',
            'mspointermove',
            'mspointerout',
            'mspointerover',
            'mspointerup',
            'pointerout',
            'mssitemodejumplistitemremoved',
            'msthumbnailclick',
            'stop',
            'storagecommit'
        ];
        var webglEventNames = ['webglcontextrestored', 'webglcontextlost', 'webglcontextcreationerror'];
        var formEventNames = ['autocomplete', 'autocompleteerror'];
        var detailEventNames = ['toggle'];
        var frameEventNames = ['load'];
        var frameSetEventNames = ['blur', 'error', 'focus', 'load', 'resize', 'scroll', 'messageerror'];
        var marqueeEventNames = ['bounce', 'finish', 'start'];
        var XMLHttpRequestEventNames = [
            'loadstart', 'progress', 'abort', 'error', 'load', 'progress', 'timeout', 'loadend',
            'readystatechange'
        ];
        var IDBIndexEventNames = ['upgradeneeded', 'complete', 'abort', 'success', 'error', 'blocked', 'versionchange', 'close'];
        var websocketEventNames = ['close', 'error', 'open', 'message'];
        var workerEventNames = ['error', 'message'];
        var eventNames = globalEventHandlersEventNames.concat(webglEventNames, formEventNames, detailEventNames, documentEventNames, windowEventNames, htmlElementEventNames, ieElementEventNames);
        function filterProperties(target, onProperties, ignoreProperties) {
            if (!ignoreProperties || ignoreProperties.length === 0) {
                return onProperties;
            }
            var tip = ignoreProperties.filter(function (ip) { return ip.target === target; });
            if (!tip || tip.length === 0) {
                return onProperties;
            }
            var targetIgnoreProperties = tip[0].ignoreProperties;
            return onProperties.filter(function (op) { return targetIgnoreProperties.indexOf(op) === -1; });
        }
        function patchFilteredProperties(target, onProperties, ignoreProperties, prototype) {
            // check whether target is available, sometimes target will be undefined
            // because different browser or some 3rd party plugin.
            if (!target) {
                return;
            }
            var filteredProperties = filterProperties(target, onProperties, ignoreProperties);
            patchOnProperties(target, filteredProperties, prototype);
        }
        function propertyDescriptorPatch(api, _global) {
            if (isNode && !isMix) {
                return;
            }
            if (Zone[api.symbol('patchEvents')]) {
                // events are already been patched by legacy patch.
                return;
            }
            var supportsWebSocket = typeof WebSocket !== 'undefined';
            var ignoreProperties = _global['__Zone_ignore_on_properties'];
            // for browsers that we can patch the descriptor:  Chrome & Firefox
            if (isBrowser) {
                var internalWindow_1 = window;
                var ignoreErrorProperties = isIE ? [{ target: internalWindow_1, ignoreProperties: ['error'] }] : [];
                // in IE/Edge, onProp not exist in window object, but in WindowPrototype
                // so we need to pass WindowPrototype to check onProp exist or not
                patchFilteredProperties(internalWindow_1, eventNames.concat(['messageerror']), ignoreProperties ? ignoreProperties.concat(ignoreErrorProperties) : ignoreProperties, ObjectGetPrototypeOf(internalWindow_1));
                patchFilteredProperties(Document.prototype, eventNames, ignoreProperties);
                if (typeof internalWindow_1['SVGElement'] !== 'undefined') {
                    patchFilteredProperties(internalWindow_1['SVGElement'].prototype, eventNames, ignoreProperties);
                }
                patchFilteredProperties(Element.prototype, eventNames, ignoreProperties);
                patchFilteredProperties(HTMLElement.prototype, eventNames, ignoreProperties);
                patchFilteredProperties(HTMLMediaElement.prototype, mediaElementEventNames, ignoreProperties);
                patchFilteredProperties(HTMLFrameSetElement.prototype, windowEventNames.concat(frameSetEventNames), ignoreProperties);
                patchFilteredProperties(HTMLBodyElement.prototype, windowEventNames.concat(frameSetEventNames), ignoreProperties);
                patchFilteredProperties(HTMLFrameElement.prototype, frameEventNames, ignoreProperties);
                patchFilteredProperties(HTMLIFrameElement.prototype, frameEventNames, ignoreProperties);
                var HTMLMarqueeElement_1 = internalWindow_1['HTMLMarqueeElement'];
                if (HTMLMarqueeElement_1) {
                    patchFilteredProperties(HTMLMarqueeElement_1.prototype, marqueeEventNames, ignoreProperties);
                }
                var Worker_1 = internalWindow_1['Worker'];
                if (Worker_1) {
                    patchFilteredProperties(Worker_1.prototype, workerEventNames, ignoreProperties);
                }
            }
            var XMLHttpRequest = _global['XMLHttpRequest'];
            if (XMLHttpRequest) {
                // XMLHttpRequest is not available in ServiceWorker, so we need to check here
                patchFilteredProperties(XMLHttpRequest.prototype, XMLHttpRequestEventNames, ignoreProperties);
            }
            var XMLHttpRequestEventTarget = _global['XMLHttpRequestEventTarget'];
            if (XMLHttpRequestEventTarget) {
                patchFilteredProperties(XMLHttpRequestEventTarget && XMLHttpRequestEventTarget.prototype, XMLHttpRequestEventNames, ignoreProperties);
            }
            if (typeof IDBIndex !== 'undefined') {
                patchFilteredProperties(IDBIndex.prototype, IDBIndexEventNames, ignoreProperties);
                patchFilteredProperties(IDBRequest.prototype, IDBIndexEventNames, ignoreProperties);
                patchFilteredProperties(IDBOpenDBRequest.prototype, IDBIndexEventNames, ignoreProperties);
                patchFilteredProperties(IDBDatabase.prototype, IDBIndexEventNames, ignoreProperties);
                patchFilteredProperties(IDBTransaction.prototype, IDBIndexEventNames, ignoreProperties);
                patchFilteredProperties(IDBCursor.prototype, IDBIndexEventNames, ignoreProperties);
            }
            if (supportsWebSocket) {
                patchFilteredProperties(WebSocket.prototype, websocketEventNames, ignoreProperties);
            }
        }
        /**
         * @license
         * Copyright Google Inc. All Rights Reserved.
         *
         * Use of this source code is governed by an MIT-style license that can be
         * found in the LICENSE file at https://angular.io/license
         */
        Zone.__load_patch('util', function (global, Zone, api) {
            api.patchOnProperties = patchOnProperties;
            api.patchMethod = patchMethod;
            api.bindArguments = bindArguments;
            api.patchMacroTask = patchMacroTask;
            // In earlier version of zone.js (<0.9.0), we use env name `__zone_symbol__BLACK_LISTED_EVENTS` to
            // define which events will not be patched by `Zone.js`.
            // In newer version (>=0.9.0), we change the env name to `__zone_symbol__UNPATCHED_EVENTS` to keep
            // the name consistent with angular repo.
            // The  `__zone_symbol__BLACK_LISTED_EVENTS` is deprecated, but it is still be supported for
            // backwards compatibility.
            var SYMBOL_BLACK_LISTED_EVENTS = Zone.__symbol__('BLACK_LISTED_EVENTS');
            var SYMBOL_UNPATCHED_EVENTS = Zone.__symbol__('UNPATCHED_EVENTS');
            if (global[SYMBOL_UNPATCHED_EVENTS]) {
                global[SYMBOL_BLACK_LISTED_EVENTS] = global[SYMBOL_UNPATCHED_EVENTS];
            }
            if (global[SYMBOL_BLACK_LISTED_EVENTS]) {
                Zone[SYMBOL_BLACK_LISTED_EVENTS] = Zone[SYMBOL_UNPATCHED_EVENTS] =
                    global[SYMBOL_BLACK_LISTED_EVENTS];
            }
            api.patchEventPrototype = patchEventPrototype;
            api.patchEventTarget = patchEventTarget;
            api.isIEOrEdge = isIEOrEdge;
            api.ObjectDefineProperty = ObjectDefineProperty;
            api.ObjectGetOwnPropertyDescriptor = ObjectGetOwnPropertyDescriptor;
            api.ObjectCreate = ObjectCreate;
            api.ArraySlice = ArraySlice;
            api.patchClass = patchClass;
            api.wrapWithCurrentZone = wrapWithCurrentZone;
            api.filterProperties = filterProperties;
            api.attachOriginToPatched = attachOriginToPatched;
            api._redefineProperty = Object.defineProperty;
            api.patchCallbacks = patchCallbacks;
            api.getGlobalObjects = function () { return ({ globalSources: globalSources, zoneSymbolEventNames: zoneSymbolEventNames$1, eventNames: eventNames, isBrowser: isBrowser, isMix: isMix, isNode: isNode, TRUE_STR: TRUE_STR,
                FALSE_STR: FALSE_STR, ZONE_SYMBOL_PREFIX: ZONE_SYMBOL_PREFIX, ADD_EVENT_LISTENER_STR: ADD_EVENT_LISTENER_STR, REMOVE_EVENT_LISTENER_STR: REMOVE_EVENT_LISTENER_STR }); };
        });
        /**
         * @license
         * Copyright Google Inc. All Rights Reserved.
         *
         * Use of this source code is governed by an MIT-style license that can be
         * found in the LICENSE file at https://angular.io/license
         */
        /*
         * This is necessary for Chrome and Chrome mobile, to enable
         * things like redefining `createdCallback` on an element.
         */
        var zoneSymbol$1;
        var _defineProperty;
        var _getOwnPropertyDescriptor;
        var _create;
        var unconfigurablesKey;
        function propertyPatch() {
            zoneSymbol$1 = Zone.__symbol__;
            _defineProperty = Object[zoneSymbol$1('defineProperty')] = Object.defineProperty;
            _getOwnPropertyDescriptor = Object[zoneSymbol$1('getOwnPropertyDescriptor')] =
                Object.getOwnPropertyDescriptor;
            _create = Object.create;
            unconfigurablesKey = zoneSymbol$1('unconfigurables');
            Object.defineProperty = function (obj, prop, desc) {
                if (isUnconfigurable(obj, prop)) {
                    throw new TypeError('Cannot assign to read only property \'' + prop + '\' of ' + obj);
                }
                var originalConfigurableFlag = desc.configurable;
                if (prop !== 'prototype') {
                    desc = rewriteDescriptor(obj, prop, desc);
                }
                return _tryDefineProperty(obj, prop, desc, originalConfigurableFlag);
            };
            Object.defineProperties = function (obj, props) {
                Object.keys(props).forEach(function (prop) { Object.defineProperty(obj, prop, props[prop]); });
                return obj;
            };
            Object.create = function (obj, proto) {
                if (typeof proto === 'object' && !Object.isFrozen(proto)) {
                    Object.keys(proto).forEach(function (prop) {
                        proto[prop] = rewriteDescriptor(obj, prop, proto[prop]);
                    });
                }
                return _create(obj, proto);
            };
            Object.getOwnPropertyDescriptor = function (obj, prop) {
                var desc = _getOwnPropertyDescriptor(obj, prop);
                if (desc && isUnconfigurable(obj, prop)) {
                    desc.configurable = false;
                }
                return desc;
            };
        }
        function _redefineProperty(obj, prop, desc) {
            var originalConfigurableFlag = desc.configurable;
            desc = rewriteDescriptor(obj, prop, desc);
            return _tryDefineProperty(obj, prop, desc, originalConfigurableFlag);
        }
        function isUnconfigurable(obj, prop) {
            return obj && obj[unconfigurablesKey] && obj[unconfigurablesKey][prop];
        }
        function rewriteDescriptor(obj, prop, desc) {
            // issue-927, if the desc is frozen, don't try to change the desc
            if (!Object.isFrozen(desc)) {
                desc.configurable = true;
            }
            if (!desc.configurable) {
                // issue-927, if the obj is frozen, don't try to set the desc to obj
                if (!obj[unconfigurablesKey] && !Object.isFrozen(obj)) {
                    _defineProperty(obj, unconfigurablesKey, { writable: true, value: {} });
                }
                if (obj[unconfigurablesKey]) {
                    obj[unconfigurablesKey][prop] = true;
                }
            }
            return desc;
        }
        function _tryDefineProperty(obj, prop, desc, originalConfigurableFlag) {
            try {
                return _defineProperty(obj, prop, desc);
            }
            catch (error) {
                if (desc.configurable) {
                    // In case of errors, when the configurable flag was likely set by rewriteDescriptor(), let's
                    // retry with the original flag value
                    if (typeof originalConfigurableFlag == 'undefined') {
                        delete desc.configurable;
                    }
                    else {
                        desc.configurable = originalConfigurableFlag;
                    }
                    try {
                        return _defineProperty(obj, prop, desc);
                    }
                    catch (error) {
                        var descJson = null;
                        try {
                            descJson = JSON.stringify(desc);
                        }
                        catch (error) {
                            descJson = desc.toString();
                        }
                        console.log("Attempting to configure '" + prop + "' with descriptor '" + descJson + "' on object '" + obj + "' and got error, giving up: " + error);
                    }
                }
                else {
                    throw error;
                }
            }
        }
        /**
         * @license
         * Copyright Google Inc. All Rights Reserved.
         *
         * Use of this source code is governed by an MIT-style license that can be
         * found in the LICENSE file at https://angular.io/license
         */
        function eventTargetLegacyPatch(_global, api) {
            var _a = api.getGlobalObjects(), eventNames = _a.eventNames, globalSources = _a.globalSources, zoneSymbolEventNames = _a.zoneSymbolEventNames, TRUE_STR = _a.TRUE_STR, FALSE_STR = _a.FALSE_STR, ZONE_SYMBOL_PREFIX = _a.ZONE_SYMBOL_PREFIX;
            var WTF_ISSUE_555 = 'Anchor,Area,Audio,BR,Base,BaseFont,Body,Button,Canvas,Content,DList,Directory,Div,Embed,FieldSet,Font,Form,Frame,FrameSet,HR,Head,Heading,Html,IFrame,Image,Input,Keygen,LI,Label,Legend,Link,Map,Marquee,Media,Menu,Meta,Meter,Mod,OList,Object,OptGroup,Option,Output,Paragraph,Pre,Progress,Quote,Script,Select,Source,Span,Style,TableCaption,TableCell,TableCol,Table,TableRow,TableSection,TextArea,Title,Track,UList,Unknown,Video';
            var NO_EVENT_TARGET = 'ApplicationCache,EventSource,FileReader,InputMethodContext,MediaController,MessagePort,Node,Performance,SVGElementInstance,SharedWorker,TextTrack,TextTrackCue,TextTrackList,WebKitNamedFlow,Window,Worker,WorkerGlobalScope,XMLHttpRequest,XMLHttpRequestEventTarget,XMLHttpRequestUpload,IDBRequest,IDBOpenDBRequest,IDBDatabase,IDBTransaction,IDBCursor,DBIndex,WebSocket'
                .split(',');
            var EVENT_TARGET = 'EventTarget';
            var apis = [];
            var isWtf = _global['wtf'];
            var WTF_ISSUE_555_ARRAY = WTF_ISSUE_555.split(',');
            if (isWtf) {
                // Workaround for: https://github.com/google/tracing-framework/issues/555
                apis = WTF_ISSUE_555_ARRAY.map(function (v) { return 'HTML' + v + 'Element'; }).concat(NO_EVENT_TARGET);
            }
            else if (_global[EVENT_TARGET]) {
                apis.push(EVENT_TARGET);
            }
            else {
                // Note: EventTarget is not available in all browsers,
                // if it's not available, we instead patch the APIs in the IDL that inherit from EventTarget
                apis = NO_EVENT_TARGET;
            }
            var isDisableIECheck = _global['__Zone_disable_IE_check'] || false;
            var isEnableCrossContextCheck = _global['__Zone_enable_cross_context_check'] || false;
            var ieOrEdge = api.isIEOrEdge();
            var ADD_EVENT_LISTENER_SOURCE = '.addEventListener:';
            var FUNCTION_WRAPPER = '[object FunctionWrapper]';
            var BROWSER_TOOLS = 'function __BROWSERTOOLS_CONSOLE_SAFEFUNC() { [native code] }';
            var pointerEventsMap = {
                'MSPointerCancel': 'pointercancel',
                'MSPointerDown': 'pointerdown',
                'MSPointerEnter': 'pointerenter',
                'MSPointerHover': 'pointerhover',
                'MSPointerLeave': 'pointerleave',
                'MSPointerMove': 'pointermove',
                'MSPointerOut': 'pointerout',
                'MSPointerOver': 'pointerover',
                'MSPointerUp': 'pointerup'
            };
            //  predefine all __zone_symbol__ + eventName + true/false string
            for (var i = 0; i < eventNames.length; i++) {
                var eventName = eventNames[i];
                var falseEventName = eventName + FALSE_STR;
                var trueEventName = eventName + TRUE_STR;
                var symbol = ZONE_SYMBOL_PREFIX + falseEventName;
                var symbolCapture = ZONE_SYMBOL_PREFIX + trueEventName;
                zoneSymbolEventNames[eventName] = {};
                zoneSymbolEventNames[eventName][FALSE_STR] = symbol;
                zoneSymbolEventNames[eventName][TRUE_STR] = symbolCapture;
            }
            //  predefine all task.source string
            for (var i = 0; i < WTF_ISSUE_555_ARRAY.length; i++) {
                var target = WTF_ISSUE_555_ARRAY[i];
                var targets = globalSources[target] = {};
                for (var j = 0; j < eventNames.length; j++) {
                    var eventName = eventNames[j];
                    targets[eventName] = target + ADD_EVENT_LISTENER_SOURCE + eventName;
                }
            }
            var checkIEAndCrossContext = function (nativeDelegate, delegate, target, args) {
                if (!isDisableIECheck && ieOrEdge) {
                    if (isEnableCrossContextCheck) {
                        try {
                            var testString = delegate.toString();
                            if ((testString === FUNCTION_WRAPPER || testString == BROWSER_TOOLS)) {
                                nativeDelegate.apply(target, args);
                                return false;
                            }
                        }
                        catch (error) {
                            nativeDelegate.apply(target, args);
                            return false;
                        }
                    }
                    else {
                        var testString = delegate.toString();
                        if ((testString === FUNCTION_WRAPPER || testString == BROWSER_TOOLS)) {
                            nativeDelegate.apply(target, args);
                            return false;
                        }
                    }
                }
                else if (isEnableCrossContextCheck) {
                    try {
                        delegate.toString();
                    }
                    catch (error) {
                        nativeDelegate.apply(target, args);
                        return false;
                    }
                }
                return true;
            };
            var apiTypes = [];
            for (var i = 0; i < apis.length; i++) {
                var type = _global[apis[i]];
                apiTypes.push(type && type.prototype);
            }
            // vh is validateHandler to check event handler
            // is valid or not(for security check)
            api.patchEventTarget(_global, apiTypes, {
                vh: checkIEAndCrossContext,
                transferEventName: function (eventName) {
                    var pointerEventName = pointerEventsMap[eventName];
                    return pointerEventName || eventName;
                }
            });
            Zone[api.symbol('patchEventTarget')] = !!_global[EVENT_TARGET];
            return true;
        }
        /**
         * @license
         * Copyright Google Inc. All Rights Reserved.
         *
         * Use of this source code is governed by an MIT-style license that can be
         * found in the LICENSE file at https://angular.io/license
         */
        // we have to patch the instance since the proto is non-configurable
        function apply(api, _global) {
            var _a = api.getGlobalObjects(), ADD_EVENT_LISTENER_STR = _a.ADD_EVENT_LISTENER_STR, REMOVE_EVENT_LISTENER_STR = _a.REMOVE_EVENT_LISTENER_STR;
            var WS = _global.WebSocket;
            // On Safari window.EventTarget doesn't exist so need to patch WS add/removeEventListener
            // On older Chrome, no need since EventTarget was already patched
            if (!_global.EventTarget) {
                api.patchEventTarget(_global, [WS.prototype]);
            }
            _global.WebSocket = function (x, y) {
                var socket = arguments.length > 1 ? new WS(x, y) : new WS(x);
                var proxySocket;
                var proxySocketProto;
                // Safari 7.0 has non-configurable own 'onmessage' and friends properties on the socket instance
                var onmessageDesc = api.ObjectGetOwnPropertyDescriptor(socket, 'onmessage');
                if (onmessageDesc && onmessageDesc.configurable === false) {
                    proxySocket = api.ObjectCreate(socket);
                    // socket have own property descriptor 'onopen', 'onmessage', 'onclose', 'onerror'
                    // but proxySocket not, so we will keep socket as prototype and pass it to
                    // patchOnProperties method
                    proxySocketProto = socket;
                    [ADD_EVENT_LISTENER_STR, REMOVE_EVENT_LISTENER_STR, 'send', 'close'].forEach(function (propName) {
                        proxySocket[propName] = function () {
                            var args = api.ArraySlice.call(arguments);
                            if (propName === ADD_EVENT_LISTENER_STR || propName === REMOVE_EVENT_LISTENER_STR) {
                                var eventName = args.length > 0 ? args[0] : undefined;
                                if (eventName) {
                                    var propertySymbol = Zone.__symbol__('ON_PROPERTY' + eventName);
                                    socket[propertySymbol] = proxySocket[propertySymbol];
                                }
                            }
                            return socket[propName].apply(socket, args);
                        };
                    });
                }
                else {
                    // we can patch the real socket
                    proxySocket = socket;
                }
                api.patchOnProperties(proxySocket, ['close', 'error', 'message', 'open'], proxySocketProto);
                return proxySocket;
            };
            var globalWebSocket = _global['WebSocket'];
            for (var prop in WS) {
                globalWebSocket[prop] = WS[prop];
            }
        }
        /**
         * @license
         * Copyright Google Inc. All Rights Reserved.
         *
         * Use of this source code is governed by an MIT-style license that can be
         * found in the LICENSE file at https://angular.io/license
         */
        function propertyDescriptorLegacyPatch(api, _global) {
            var _a = api.getGlobalObjects(), isNode = _a.isNode, isMix = _a.isMix;
            if (isNode && !isMix) {
                return;
            }
            if (!canPatchViaPropertyDescriptor(api, _global)) {
                var supportsWebSocket = typeof WebSocket !== 'undefined';
                // Safari, Android browsers (Jelly Bean)
                patchViaCapturingAllTheEvents(api);
                api.patchClass('XMLHttpRequest');
                if (supportsWebSocket) {
                    apply(api, _global);
                }
                Zone[api.symbol('patchEvents')] = true;
            }
        }
        function canPatchViaPropertyDescriptor(api, _global) {
            var _a = api.getGlobalObjects(), isBrowser = _a.isBrowser, isMix = _a.isMix;
            if ((isBrowser || isMix) &&
                !api.ObjectGetOwnPropertyDescriptor(HTMLElement.prototype, 'onclick') &&
                typeof Element !== 'undefined') {
                // WebKit https://bugs.webkit.org/show_bug.cgi?id=134364
                // IDL interface attributes are not configurable
                var desc = api.ObjectGetOwnPropertyDescriptor(Element.prototype, 'onclick');
                if (desc && !desc.configurable)
                    return false;
                // try to use onclick to detect whether we can patch via propertyDescriptor
                // because XMLHttpRequest is not available in service worker
                if (desc) {
                    api.ObjectDefineProperty(Element.prototype, 'onclick', { enumerable: true, configurable: true, get: function () { return true; } });
                    var div = document.createElement('div');
                    var result = !!div.onclick;
                    api.ObjectDefineProperty(Element.prototype, 'onclick', desc);
                    return result;
                }
            }
            var XMLHttpRequest = _global['XMLHttpRequest'];
            if (!XMLHttpRequest) {
                // XMLHttpRequest is not available in service worker
                return false;
            }
            var ON_READY_STATE_CHANGE = 'onreadystatechange';
            var XMLHttpRequestPrototype = XMLHttpRequest.prototype;
            var xhrDesc = api.ObjectGetOwnPropertyDescriptor(XMLHttpRequestPrototype, ON_READY_STATE_CHANGE);
            // add enumerable and configurable here because in opera
            // by default XMLHttpRequest.prototype.onreadystatechange is undefined
            // without adding enumerable and configurable will cause onreadystatechange
            // non-configurable
            // and if XMLHttpRequest.prototype.onreadystatechange is undefined,
            // we should set a real desc instead a fake one
            if (xhrDesc) {
                api.ObjectDefineProperty(XMLHttpRequestPrototype, ON_READY_STATE_CHANGE, { enumerable: true, configurable: true, get: function () { return true; } });
                var req = new XMLHttpRequest();
                var result = !!req.onreadystatechange;
                // restore original desc
                api.ObjectDefineProperty(XMLHttpRequestPrototype, ON_READY_STATE_CHANGE, xhrDesc || {});
                return result;
            }
            else {
                var SYMBOL_FAKE_ONREADYSTATECHANGE_1 = api.symbol('fake');
                api.ObjectDefineProperty(XMLHttpRequestPrototype, ON_READY_STATE_CHANGE, {
                    enumerable: true,
                    configurable: true,
                    get: function () { return this[SYMBOL_FAKE_ONREADYSTATECHANGE_1]; },
                    set: function (value) { this[SYMBOL_FAKE_ONREADYSTATECHANGE_1] = value; }
                });
                var req = new XMLHttpRequest();
                var detectFunc = function () { };
                req.onreadystatechange = detectFunc;
                var result = req[SYMBOL_FAKE_ONREADYSTATECHANGE_1] === detectFunc;
                req.onreadystatechange = null;
                return result;
            }
        }
        // Whenever any eventListener fires, we check the eventListener target and all parents
        // for `onwhatever` properties and replace them with zone-bound functions
        // - Chrome (for now)
        function patchViaCapturingAllTheEvents(api) {
            var eventNames = api.getGlobalObjects().eventNames;
            var unboundKey = api.symbol('unbound');
            var _loop_4 = function (i) {
                var property = eventNames[i];
                var onproperty = 'on' + property;
                self.addEventListener(property, function (event) {
                    var elt = event.target, bound, source;
                    if (elt) {
                        source = elt.constructor['name'] + '.' + onproperty;
                    }
                    else {
                        source = 'unknown.' + onproperty;
                    }
                    while (elt) {
                        if (elt[onproperty] && !elt[onproperty][unboundKey]) {
                            bound = api.wrapWithCurrentZone(elt[onproperty], source);
                            bound[unboundKey] = elt[onproperty];
                            elt[onproperty] = bound;
                        }
                        elt = elt.parentElement;
                    }
                }, true);
            };
            for (var i = 0; i < eventNames.length; i++) {
                _loop_4(i);
            }
        }
        /**
         * @license
         * Copyright Google Inc. All Rights Reserved.
         *
         * Use of this source code is governed by an MIT-style license that can be
         * found in the LICENSE file at https://angular.io/license
         */
        function registerElementPatch(_global, api) {
            var _a = api.getGlobalObjects(), isBrowser = _a.isBrowser, isMix = _a.isMix;
            if ((!isBrowser && !isMix) || !('registerElement' in _global.document)) {
                return;
            }
            var callbacks = ['createdCallback', 'attachedCallback', 'detachedCallback', 'attributeChangedCallback'];
            api.patchCallbacks(api, document, 'Document', 'registerElement', callbacks);
        }
        /**
         * @license
         * Copyright Google Inc. All Rights Reserved.
         *
         * Use of this source code is governed by an MIT-style license that can be
         * found in the LICENSE file at https://angular.io/license
         */
        (function (_global) {
            var symbolPrefix = _global['__Zone_symbol_prefix'] || '__zone_symbol__';
            function __symbol__(name) { return symbolPrefix + name; }
            _global[__symbol__('legacyPatch')] = function () {
                var Zone = _global['Zone'];
                Zone.__load_patch('defineProperty', function (global, Zone, api) {
                    api._redefineProperty = _redefineProperty;
                    propertyPatch();
                });
                Zone.__load_patch('registerElement', function (global, Zone, api) {
                    registerElementPatch(global, api);
                });
                Zone.__load_patch('EventTargetLegacy', function (global, Zone, api) {
                    eventTargetLegacyPatch(global, api);
                    propertyDescriptorLegacyPatch(api, global);
                });
            };
        })(typeof window !== 'undefined' ?
            window :
            typeof commonjsGlobal !== 'undefined' ? commonjsGlobal : typeof self !== 'undefined' ? self : {});
        /**
         * @license
         * Copyright Google Inc. All Rights Reserved.
         *
         * Use of this source code is governed by an MIT-style license that can be
         * found in the LICENSE file at https://angular.io/license
         */
        var taskSymbol = zoneSymbol('zoneTask');
        function patchTimer(window, setName, cancelName, nameSuffix) {
            var setNative = null;
            var clearNative = null;
            setName += nameSuffix;
            cancelName += nameSuffix;
            var tasksByHandleId = {};
            function scheduleTask(task) {
                var data = task.data;
                function timer() {
                    try {
                        task.invoke.apply(this, arguments);
                    }
                    finally {
                        // issue-934, task will be cancelled
                        // even it is a periodic task such as
                        // setInterval
                        if (!(task.data && task.data.isPeriodic)) {
                            if (typeof data.handleId === 'number') {
                                // in non-nodejs env, we remove timerId
                                // from local cache
                                delete tasksByHandleId[data.handleId];
                            }
                            else if (data.handleId) {
                                // Node returns complex objects as handleIds
                                // we remove task reference from timer object
                                data.handleId[taskSymbol] = null;
                            }
                        }
                    }
                }
                data.args[0] = timer;
                data.handleId = setNative.apply(window, data.args);
                return task;
            }
            function clearTask(task) { return clearNative(task.data.handleId); }
            setNative =
                patchMethod(window, setName, function (delegate) { return function (self, args) {
                    if (typeof args[0] === 'function') {
                        var options = {
                            isPeriodic: nameSuffix === 'Interval',
                            delay: (nameSuffix === 'Timeout' || nameSuffix === 'Interval') ? args[1] || 0 :
                                undefined,
                            args: args
                        };
                        var task = scheduleMacroTaskWithCurrentZone(setName, args[0], options, scheduleTask, clearTask);
                        if (!task) {
                            return task;
                        }
                        // Node.js must additionally support the ref and unref functions.
                        var handle = task.data.handleId;
                        if (typeof handle === 'number') {
                            // for non nodejs env, we save handleId: task
                            // mapping in local cache for clearTimeout
                            tasksByHandleId[handle] = task;
                        }
                        else if (handle) {
                            // for nodejs env, we save task
                            // reference in timerId Object for clearTimeout
                            handle[taskSymbol] = task;
                        }
                        // check whether handle is null, because some polyfill or browser
                        // may return undefined from setTimeout/setInterval/setImmediate/requestAnimationFrame
                        if (handle && handle.ref && handle.unref && typeof handle.ref === 'function' &&
                            typeof handle.unref === 'function') {
                            task.ref = handle.ref.bind(handle);
                            task.unref = handle.unref.bind(handle);
                        }
                        if (typeof handle === 'number' || handle) {
                            return handle;
                        }
                        return task;
                    }
                    else {
                        // cause an error by calling it directly.
                        return delegate.apply(window, args);
                    }
                }; });
            clearNative =
                patchMethod(window, cancelName, function (delegate) { return function (self, args) {
                    var id = args[0];
                    var task;
                    if (typeof id === 'number') {
                        // non nodejs env.
                        task = tasksByHandleId[id];
                    }
                    else {
                        // nodejs env.
                        task = id && id[taskSymbol];
                        // other environments.
                        if (!task) {
                            task = id;
                        }
                    }
                    if (task && typeof task.type === 'string') {
                        if (task.state !== 'notScheduled' &&
                            (task.cancelFn && task.data.isPeriodic || task.runCount === 0)) {
                            if (typeof id === 'number') {
                                delete tasksByHandleId[id];
                            }
                            else if (id) {
                                id[taskSymbol] = null;
                            }
                            // Do not cancel already canceled functions
                            task.zone.cancelTask(task);
                        }
                    }
                    else {
                        // cause an error by calling it directly.
                        delegate.apply(window, args);
                    }
                }; });
        }
        /**
         * @license
         * Copyright Google Inc. All Rights Reserved.
         *
         * Use of this source code is governed by an MIT-style license that can be
         * found in the LICENSE file at https://angular.io/license
         */
        function patchCustomElements(_global, api) {
            var _a = api.getGlobalObjects(), isBrowser = _a.isBrowser, isMix = _a.isMix;
            if ((!isBrowser && !isMix) || !_global['customElements'] || !('customElements' in _global)) {
                return;
            }
            var callbacks = ['connectedCallback', 'disconnectedCallback', 'adoptedCallback', 'attributeChangedCallback'];
            api.patchCallbacks(api, _global.customElements, 'customElements', 'define', callbacks);
        }
        /**
         * @license
         * Copyright Google Inc. All Rights Reserved.
         *
         * Use of this source code is governed by an MIT-style license that can be
         * found in the LICENSE file at https://angular.io/license
         */
        function eventTargetPatch(_global, api) {
            if (Zone[api.symbol('patchEventTarget')]) {
                // EventTarget is already patched.
                return;
            }
            var _a = api.getGlobalObjects(), eventNames = _a.eventNames, zoneSymbolEventNames = _a.zoneSymbolEventNames, TRUE_STR = _a.TRUE_STR, FALSE_STR = _a.FALSE_STR, ZONE_SYMBOL_PREFIX = _a.ZONE_SYMBOL_PREFIX;
            //  predefine all __zone_symbol__ + eventName + true/false string
            for (var i = 0; i < eventNames.length; i++) {
                var eventName = eventNames[i];
                var falseEventName = eventName + FALSE_STR;
                var trueEventName = eventName + TRUE_STR;
                var symbol = ZONE_SYMBOL_PREFIX + falseEventName;
                var symbolCapture = ZONE_SYMBOL_PREFIX + trueEventName;
                zoneSymbolEventNames[eventName] = {};
                zoneSymbolEventNames[eventName][FALSE_STR] = symbol;
                zoneSymbolEventNames[eventName][TRUE_STR] = symbolCapture;
            }
            var EVENT_TARGET = _global['EventTarget'];
            if (!EVENT_TARGET || !EVENT_TARGET.prototype) {
                return;
            }
            api.patchEventTarget(_global, [EVENT_TARGET && EVENT_TARGET.prototype]);
            return true;
        }
        function patchEvent(global, api) {
            api.patchEventPrototype(global, api);
        }
        /**
         * @license
         * Copyright Google Inc. All Rights Reserved.
         *
         * Use of this source code is governed by an MIT-style license that can be
         * found in the LICENSE file at https://angular.io/license
         */
        Zone.__load_patch('legacy', function (global) {
            var legacyPatch = global[Zone.__symbol__('legacyPatch')];
            if (legacyPatch) {
                legacyPatch();
            }
        });
        Zone.__load_patch('timers', function (global) {
            var set = 'set';
            var clear = 'clear';
            patchTimer(global, set, clear, 'Timeout');
            patchTimer(global, set, clear, 'Interval');
            patchTimer(global, set, clear, 'Immediate');
        });
        Zone.__load_patch('requestAnimationFrame', function (global) {
            patchTimer(global, 'request', 'cancel', 'AnimationFrame');
            patchTimer(global, 'mozRequest', 'mozCancel', 'AnimationFrame');
            patchTimer(global, 'webkitRequest', 'webkitCancel', 'AnimationFrame');
        });
        Zone.__load_patch('blocking', function (global, Zone) {
            var blockingMethods = ['alert', 'prompt', 'confirm'];
            for (var i = 0; i < blockingMethods.length; i++) {
                var name_2 = blockingMethods[i];
                patchMethod(global, name_2, function (delegate, symbol, name) {
                    return function (s, args) {
                        return Zone.current.run(delegate, global, args, name);
                    };
                });
            }
        });
        Zone.__load_patch('EventTarget', function (global, Zone, api) {
            patchEvent(global, api);
            eventTargetPatch(global, api);
            // patch XMLHttpRequestEventTarget's addEventListener/removeEventListener
            var XMLHttpRequestEventTarget = global['XMLHttpRequestEventTarget'];
            if (XMLHttpRequestEventTarget && XMLHttpRequestEventTarget.prototype) {
                api.patchEventTarget(global, [XMLHttpRequestEventTarget.prototype]);
            }
            patchClass('MutationObserver');
            patchClass('WebKitMutationObserver');
            patchClass('IntersectionObserver');
            patchClass('FileReader');
        });
        Zone.__load_patch('on_property', function (global, Zone, api) {
            propertyDescriptorPatch(api, global);
        });
        Zone.__load_patch('customElements', function (global, Zone, api) {
            patchCustomElements(global, api);
        });
        Zone.__load_patch('XHR', function (global, Zone) {
            // Treat XMLHttpRequest as a macrotask.
            patchXHR(global);
            var XHR_TASK = zoneSymbol('xhrTask');
            var XHR_SYNC = zoneSymbol('xhrSync');
            var XHR_LISTENER = zoneSymbol('xhrListener');
            var XHR_SCHEDULED = zoneSymbol('xhrScheduled');
            var XHR_URL = zoneSymbol('xhrURL');
            var XHR_ERROR_BEFORE_SCHEDULED = zoneSymbol('xhrErrorBeforeScheduled');
            function patchXHR(window) {
                var XMLHttpRequest = window['XMLHttpRequest'];
                if (!XMLHttpRequest) {
                    // XMLHttpRequest is not available in service worker
                    return;
                }
                var XMLHttpRequestPrototype = XMLHttpRequest.prototype;
                function findPendingTask(target) { return target[XHR_TASK]; }
                var oriAddListener = XMLHttpRequestPrototype[ZONE_SYMBOL_ADD_EVENT_LISTENER];
                var oriRemoveListener = XMLHttpRequestPrototype[ZONE_SYMBOL_REMOVE_EVENT_LISTENER];
                if (!oriAddListener) {
                    var XMLHttpRequestEventTarget_1 = window['XMLHttpRequestEventTarget'];
                    if (XMLHttpRequestEventTarget_1) {
                        var XMLHttpRequestEventTargetPrototype = XMLHttpRequestEventTarget_1.prototype;
                        oriAddListener = XMLHttpRequestEventTargetPrototype[ZONE_SYMBOL_ADD_EVENT_LISTENER];
                        oriRemoveListener = XMLHttpRequestEventTargetPrototype[ZONE_SYMBOL_REMOVE_EVENT_LISTENER];
                    }
                }
                var READY_STATE_CHANGE = 'readystatechange';
                var SCHEDULED = 'scheduled';
                function scheduleTask(task) {
                    var data = task.data;
                    var target = data.target;
                    target[XHR_SCHEDULED] = false;
                    target[XHR_ERROR_BEFORE_SCHEDULED] = false;
                    // remove existing event listener
                    var listener = target[XHR_LISTENER];
                    if (!oriAddListener) {
                        oriAddListener = target[ZONE_SYMBOL_ADD_EVENT_LISTENER];
                        oriRemoveListener = target[ZONE_SYMBOL_REMOVE_EVENT_LISTENER];
                    }
                    if (listener) {
                        oriRemoveListener.call(target, READY_STATE_CHANGE, listener);
                    }
                    var newListener = target[XHR_LISTENER] = function () {
                        if (target.readyState === target.DONE) {
                            // sometimes on some browsers XMLHttpRequest will fire onreadystatechange with
                            // readyState=4 multiple times, so we need to check task state here
                            if (!data.aborted && target[XHR_SCHEDULED] && task.state === SCHEDULED) {
                                // check whether the xhr has registered onload listener
                                // if that is the case, the task should invoke after all
                                // onload listeners finish.
                                var loadTasks = target[Zone.__symbol__('loadfalse')];
                                if (loadTasks && loadTasks.length > 0) {
                                    var oriInvoke_1 = task.invoke;
                                    task.invoke = function () {
                                        // need to load the tasks again, because in other
                                        // load listener, they may remove themselves
                                        var loadTasks = target[Zone.__symbol__('loadfalse')];
                                        for (var i = 0; i < loadTasks.length; i++) {
                                            if (loadTasks[i] === task) {
                                                loadTasks.splice(i, 1);
                                            }
                                        }
                                        if (!data.aborted && task.state === SCHEDULED) {
                                            oriInvoke_1.call(task);
                                        }
                                    };
                                    loadTasks.push(task);
                                }
                                else {
                                    task.invoke();
                                }
                            }
                            else if (!data.aborted && target[XHR_SCHEDULED] === false) {
                                // error occurs when xhr.send()
                                target[XHR_ERROR_BEFORE_SCHEDULED] = true;
                            }
                        }
                    };
                    oriAddListener.call(target, READY_STATE_CHANGE, newListener);
                    var storedTask = target[XHR_TASK];
                    if (!storedTask) {
                        target[XHR_TASK] = task;
                    }
                    sendNative.apply(target, data.args);
                    target[XHR_SCHEDULED] = true;
                    return task;
                }
                function placeholderCallback() { }
                function clearTask(task) {
                    var data = task.data;
                    // Note - ideally, we would call data.target.removeEventListener here, but it's too late
                    // to prevent it from firing. So instead, we store info for the event listener.
                    data.aborted = true;
                    return abortNative.apply(data.target, data.args);
                }
                var openNative = patchMethod(XMLHttpRequestPrototype, 'open', function () { return function (self, args) {
                    self[XHR_SYNC] = args[2] == false;
                    self[XHR_URL] = args[1];
                    return openNative.apply(self, args);
                }; });
                var XMLHTTPREQUEST_SOURCE = 'XMLHttpRequest.send';
                var fetchTaskAborting = zoneSymbol('fetchTaskAborting');
                var fetchTaskScheduling = zoneSymbol('fetchTaskScheduling');
                var sendNative = patchMethod(XMLHttpRequestPrototype, 'send', function () { return function (self, args) {
                    if (Zone.current[fetchTaskScheduling] === true) {
                        // a fetch is scheduling, so we are using xhr to polyfill fetch
                        // and because we already schedule macroTask for fetch, we should
                        // not schedule a macroTask for xhr again
                        return sendNative.apply(self, args);
                    }
                    if (self[XHR_SYNC]) {
                        // if the XHR is sync there is no task to schedule, just execute the code.
                        return sendNative.apply(self, args);
                    }
                    else {
                        var options = { target: self, url: self[XHR_URL], isPeriodic: false, args: args, aborted: false };
                        var task = scheduleMacroTaskWithCurrentZone(XMLHTTPREQUEST_SOURCE, placeholderCallback, options, scheduleTask, clearTask);
                        if (self && self[XHR_ERROR_BEFORE_SCHEDULED] === true && !options.aborted &&
                            task.state === SCHEDULED) {
                            // xhr request throw error when send
                            // we should invoke task instead of leaving a scheduled
                            // pending macroTask
                            task.invoke();
                        }
                    }
                }; });
                var abortNative = patchMethod(XMLHttpRequestPrototype, 'abort', function () { return function (self, args) {
                    var task = findPendingTask(self);
                    if (task && typeof task.type == 'string') {
                        // If the XHR has already completed, do nothing.
                        // If the XHR has already been aborted, do nothing.
                        // Fix #569, call abort multiple times before done will cause
                        // macroTask task count be negative number
                        if (task.cancelFn == null || (task.data && task.data.aborted)) {
                            return;
                        }
                        task.zone.cancelTask(task);
                    }
                    else if (Zone.current[fetchTaskAborting] === true) {
                        // the abort is called from fetch polyfill, we need to call native abort of XHR.
                        return abortNative.apply(self, args);
                    }
                    // Otherwise, we are trying to abort an XHR which has not yet been sent, so there is no
                    // task
                    // to cancel. Do nothing.
                }; });
            }
        });
        Zone.__load_patch('geolocation', function (global) {
            /// GEO_LOCATION
            if (global['navigator'] && global['navigator'].geolocation) {
                patchPrototype(global['navigator'].geolocation, ['getCurrentPosition', 'watchPosition']);
            }
        });
        Zone.__load_patch('PromiseRejectionEvent', function (global, Zone) {
            // handle unhandled promise rejection
            function findPromiseRejectionHandler(evtName) {
                return function (e) {
                    var eventTasks = findEventTasks(global, evtName);
                    eventTasks.forEach(function (eventTask) {
                        // windows has added unhandledrejection event listener
                        // trigger the event listener
                        var PromiseRejectionEvent = global['PromiseRejectionEvent'];
                        if (PromiseRejectionEvent) {
                            var evt = new PromiseRejectionEvent(evtName, { promise: e.promise, reason: e.rejection });
                            eventTask.invoke(evt);
                        }
                    });
                };
            }
            if (global['PromiseRejectionEvent']) {
                Zone[zoneSymbol('unhandledPromiseRejectionHandler')] =
                    findPromiseRejectionHandler('unhandledrejection');
                Zone[zoneSymbol('rejectionHandledHandler')] =
                    findPromiseRejectionHandler('rejectionhandled');
            }
        });
    })));

    /**
     * @license Angular v9.1.12
     * (c) 2010-2020 Google LLC. https://angular.io/
     * License: MIT
     */

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/animations/src/animation_builder.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * An injectable service that produces an animation sequence programmatically within an
     * Angular component or directive.
     * Provided by the `BrowserAnimationsModule` or `NoopAnimationsModule`.
     *
     * \@usageNotes
     *
     * To use this service, add it to your component or directive as a dependency.
     * The service is instantiated along with your component.
     *
     * Apps do not typically need to create their own animation players, but if you
     * do need to, follow these steps:
     *
     * 1. Use the `build()` method to create a programmatic animation using the
     * `animate()` function. The method returns an `AnimationFactory` instance.
     *
     * 2. Use the factory object to create an `AnimationPlayer` and attach it to a DOM element.
     *
     * 3. Use the player object to control the animation programmatically.
     *
     * For example:
     *
     * ```ts
     * // import the service from BrowserAnimationsModule
     * import {AnimationBuilder} from '\@angular/animations';
     * // require the service as a dependency
     * class MyCmp {
     *   constructor(private _builder: AnimationBuilder) {}
     *
     *   makeAnimation(element: any) {
     *     // first define a reusable animation
     *     const myAnimation = this._builder.build([
     *       style({ width: 0 }),
     *       animate(1000, style({ width: '100px' }))
     *     ]);
     *
     *     // use the returned factory object to create a player
     *     const player = myAnimation.create(element);
     *
     *     player.play();
     *   }
     * }
     * ```
     *
     * \@publicApi
     * @abstract
     */
    class AnimationBuilder {
    }
    /**
     * A factory object returned from the `AnimationBuilder`.`build()` method.
     *
     * \@publicApi
     * @abstract
     */
    class AnimationFactory {
    }
    /**
     * Specifies automatic styling.
     *
     * \@publicApi
     * @type {?}
     */
    const AUTO_STYLE = '*';
    /**
     * Creates a named animation trigger, containing a  list of `state()`
     * and `transition()` entries to be evaluated when the expression
     * bound to the trigger changes.
     *
     * \@usageNotes
     * Define an animation trigger in the `animations` section of `\@Component` metadata.
     * In the template, reference the trigger by name and bind it to a trigger expression that
     * evaluates to a defined animation state, using the following format:
     *
     * `[\@triggerName]="expression"`
     *
     * Animation trigger bindings convert all values to strings, and then match the
     * previous and current values against any linked transitions.
     * Booleans can be specified as `1` or `true` and `0` or `false`.
     *
     * ### Usage Example
     *
     * The following example creates an animation trigger reference based on the provided
     * name value.
     * The provided animation value is expected to be an array consisting of state and
     * transition declarations.
     *
     * ```typescript
     * \@Component({
     *   selector: "my-component",
     *   templateUrl: "my-component-tpl.html",
     *   animations: [
     *     trigger("myAnimationTrigger", [
     *       state(...),
     *       state(...),
     *       transition(...),
     *       transition(...)
     *     ])
     *   ]
     * })
     * class MyComponent {
     *   myStatusExp = "something";
     * }
     * ```
     *
     * The template associated with this component makes use of the defined trigger
     * by binding to an element within its template code.
     *
     * ```html
     * <!-- somewhere inside of my-component-tpl.html -->
     * <div [\@myAnimationTrigger]="myStatusExp">...</div>
     * ```
     *
     * ### Using an inline function
     * The `transition` animation method also supports reading an inline function which can decide
     * if its associated animation should be run.
     *
     * ```typescript
     * // this method is run each time the `myAnimationTrigger` trigger value changes.
     * function myInlineMatcherFn(fromState: string, toState: string, element: any, params: {[key:
     * string]: any}): boolean {
     *   // notice that `element` and `params` are also available here
     *   return toState == 'yes-please-animate';
     * }
     *  /
     *   selector: 'my-component',
     *   templateUrl: 'my-component-tpl.html',
     *   animations: [
     *     trigger('myAnimationTrigger', [
     *       transition(myInlineMatcherFn, [
     *         // the animation sequence code
     *       ]),
     *     ])
     *   ]
     * })
     * class MyComponent {
     *   myStatusExp = "yes-please-animate";
     * }
     * ```
     *
     * ### Disabling Animations
     * When true, the special animation control binding `\@.disabled` binding prevents
     * all animations from rendering.
     * Place the  `\@.disabled` binding on an element to disable
     * animations on the element itself, as well as any inner animation triggers
     * within the element.
     *
     * The following example shows how to use this feature:
     *
     * ```typescript /
     *   selector: 'my-component',
     *   template: `
     *     <div [\@.disabled]="isDisabled">
     *       <div [\@childAnimation]="exp"></div>
     *     </div>
     *   `,
     *   animations: [
     *     trigger("childAnimation", [
     *       // ...
     *     ])
     *   ]
     * })
     * class MyComponent {
     *   isDisabled = true;
     *   exp = '...';
     * }
     * ```
     *
     * When `\@.disabled` is true, it prevents the `\@childAnimation` trigger from animating,
     * along with any inner animations.
     *
     * ### Disable animations application-wide
     * When an area of the template is set to have animations disabled,
     * **all** inner components have their animations disabled as well.
     * This means that you can disable all animations for an app
     * by placing a host binding set on `\@.disabled` on the topmost Angular component.
     *
     * ```typescript
     * import {Component, HostBinding} from '\@angular/core';
     *  /
     *   selector: 'app-component',
     *   templateUrl: 'app.component.html',
     * })
     * class AppComponent {
     * \@HostBinding('@.disabled')
     *   public animationsDisabled = true;
     * }
     * ```
     *
     * ### Overriding disablement of inner animations
     * Despite inner animations being disabled, a parent animation can `query()`
     * for inner elements located in disabled areas of the template and still animate
     * them if needed. This is also the case for when a sub animation is
     * queried by a parent and then later animated using `animateChild()`.
     *
     * ### Detecting when an animation is disabled
     * If a region of the DOM (or the entire application) has its animations disabled, the animation
     * trigger callbacks still fire, but for zero seconds. When the callback fires, it provides
     * an instance of an `AnimationEvent`. If animations are disabled,
     * the `.disabled` flag on the event is true.
     *
     * \@publicApi
     * @param {?} name An identifying string.
     * @param {?} definitions  An animation definition object, containing an array of `state()`
     * and `transition()` declarations.
     *
     * @return {?} An object that encapsulates the trigger data.
     *
     */
    function trigger(name, definitions) {
        return { type: 7 /* Trigger */, name, definitions, options: {} };
    }
    /**
     * Defines an animation step that combines styling information with timing information.
     *
     * \@usageNotes
     * Call within an animation `sequence()`, `{\@link animations/group group()}`, or
     * `transition()` call to specify an animation step
     * that applies given style data to the parent animation for a given amount of time.
     *
     * ### Syntax Examples
     * **Timing examples**
     *
     * The following examples show various `timings` specifications.
     * - `animate(500)` : Duration is 500 milliseconds.
     * - `animate("1s")` : Duration is 1000 milliseconds.
     * - `animate("100ms 0.5s")` : Duration is 100 milliseconds, delay is 500 milliseconds.
     * - `animate("5s ease-in")` : Duration is 5000 milliseconds, easing in.
     * - `animate("5s 10ms cubic-bezier(.17,.67,.88,.1)")` : Duration is 5000 milliseconds, delay is 10
     * milliseconds, easing according to a bezier curve.
     *
     * **Style examples**
     *
     * The following example calls `style()` to set a single CSS style.
     * ```typescript
     * animate(500, style({ background: "red" }))
     * ```
     * The following example calls `keyframes()` to set a CSS style
     * to different values for successive keyframes.
     * ```typescript
     * animate(500, keyframes(
     *  [
     *   style({ background: "blue" })),
     *   style({ background: "red" }))
     *  ])
     * ```
     *
     * \@publicApi
     * @param {?} timings Sets `AnimateTimings` for the parent animation.
     * A string in the format "duration [delay] [easing]".
     *  - Duration and delay are expressed as a number and optional time unit,
     * such as "1s" or "10ms" for one second and 10 milliseconds, respectively.
     * The default unit is milliseconds.
     *  - The easing value controls how the animation accelerates and decelerates
     * during its runtime. Value is one of  `ease`, `ease-in`, `ease-out`,
     * `ease-in-out`, or a `cubic-bezier()` function call.
     * If not supplied, no easing is applied.
     *
     * For example, the string "1s 100ms ease-out" specifies a duration of
     * 1000 milliseconds, and delay of 100 ms, and the "ease-out" easing style,
     * which decelerates near the end of the duration.
     * @param {?=} styles Sets AnimationStyles for the parent animation.
     * A function call to either `style()` or `keyframes()`
     * that returns a collection of CSS style entries to be applied to the parent animation.
     * When null, uses the styles from the destination state.
     * This is useful when describing an animation step that will complete an animation;
     * see "Animating to the final state" in `transitions()`.
     * @return {?} An object that encapsulates the animation step.
     *
     */
    function animate(timings, styles = null) {
        return { type: 4 /* Animate */, styles, timings };
    }
    /**
     * \@description Defines a list of animation steps to be run in parallel.
     *
     * \@usageNotes
     * Grouped animations are useful when a series of styles must be
     * animated at different starting times and closed off at different ending times.
     *
     * When called within a `sequence()` or a
     * `transition()` call, does not continue to the next
     * instruction until all of the inner animation steps have completed.
     *
     * \@publicApi
     * @param {?} steps An array of animation step objects.
     * - When steps are defined by `style()` or `animate()`
     * function calls, each call within the group is executed instantly.
     * - To specify offset styles to be applied at a later time, define steps with
     * `keyframes()`, or use `animate()` calls with a delay value.
     * For example:
     *
     * ```typescript
     * group([
     *   animate("1s", style({ background: "black" })),
     *   animate("2s", style({ color: "white" }))
     * ])
     * ```
     *
     * @param {?=} options An options object containing a delay and
     * developer-defined parameters that provide styling defaults and
     * can be overridden on invocation.
     *
     * @return {?} An object that encapsulates the group data.
     *
     */
    function group(steps, options = null) {
        return { type: 3 /* Group */, steps, options };
    }
    /**
     * Defines a list of animation steps to be run sequentially, one by one.
     *
     * \@usageNotes
     * When you pass an array of steps to a
     * `transition()` call, the steps run sequentially by default.
     * Compare this to the `{\@link animations/group group()}` call, which runs animation steps in
     * parallel.
     *
     * When a sequence is used within a `{\@link animations/group group()}` or a `transition()` call,
     * execution continues to the next instruction only after each of the inner animation
     * steps have completed.
     *
     * \@publicApi
     *
     * @param {?} steps An array of animation step objects.
     * - Steps defined by `style()` calls apply the styling data immediately.
     * - Steps defined by `animate()` calls apply the styling data over time
     *   as specified by the timing data.
     *
     * ```typescript
     * sequence([
     *   style({ opacity: 0 }),
     *   animate("1s", style({ opacity: 1 }))
     * ])
     * ```
     *
     * @param {?=} options An options object containing a delay and
     * developer-defined parameters that provide styling defaults and
     * can be overridden on invocation.
     *
     * @return {?} An object that encapsulates the sequence data.
     *
     */
    function sequence(steps, options = null) {
        return { type: 2 /* Sequence */, steps, options };
    }
    /**
     * Declares a key/value object containing CSS properties/styles that
     * can then be used for an animation `state`, within an animation `sequence`,
     * or as styling data for calls to `animate()` and `keyframes()`.
     *
     * \@usageNotes
     * The following examples create animation styles that collect a set of
     * CSS property values:
     *
     * ```typescript
     * // string values for CSS properties
     * style({ background: "red", color: "blue" })
     *
     * // numerical pixel values
     * style({ width: 100, height: 0 })
     * ```
     *
     * The following example uses auto-styling to allow a component to animate from
     * a height of 0 up to the height of the parent element:
     *
     * ```
     * style({ height: 0 }),
     * animate("1s", style({ height: "*" }))
     * ```
     *
     * \@publicApi
     *
     * @param {?} tokens A set of CSS styles or HTML styles associated with an animation state.
     * The value can be any of the following:
     * - A key-value style pair associating a CSS property with a value.
     * - An array of key-value style pairs.
     * - An asterisk (*), to use auto-styling, where styles are derived from the element
     * being animated and applied to the animation when it starts.
     *
     * Auto-styling can be used to define a state that depends on layout or other
     * environmental factors.
     *
     * @return {?} An object that encapsulates the style data.
     *
     */
    function style(tokens) {
        return { type: 6 /* Style */, styles: tokens, offset: null };
    }
    /**
     * Declares an animation state within a trigger attached to an element.
     *
     * \@usageNotes
     * Use the `trigger()` function to register states to an animation trigger.
     * Use the `transition()` function to animate between states.
     * When a state is active within a component, its associated styles persist on the element,
     * even when the animation ends.
     *
     * \@publicApi
     *
     * @param {?} name One or more names for the defined state in a comma-separated string.
     * The following reserved state names can be supplied to define a style for specific use
     * cases:
     *
     * - `void` You can associate styles with this name to be used when
     * the element is detached from the application. For example, when an `ngIf` evaluates
     * to false, the state of the associated element is void.
     *  - `*` (asterisk) Indicates the default state. You can associate styles with this name
     * to be used as the fallback when the state that is being animated is not declared
     * within the trigger.
     *
     * @param {?} styles A set of CSS styles associated with this state, created using the
     * `style()` function.
     * This set of styles persists on the element once the state has been reached.
     * @param {?=} options Parameters that can be passed to the state when it is invoked.
     * 0 or more key-value pairs.
     * @return {?} An object that encapsulates the new state data.
     *
     */
    function state(name, styles, options) {
        return { type: 0 /* State */, name, styles, options };
    }
    /**
     * Defines a set of animation styles, associating each style with an optional `offset` value.
     *
     * \@usageNotes
     * Use with the `animate()` call. Instead of applying animations
     * from the current state
     * to the destination state, keyframes describe how each style entry is applied and at what point
     * within the animation arc.
     * Compare [CSS Keyframe Animations](https://www.w3schools.com/css/css3_animations.asp).
     *
     * ### Usage
     *
     * In the following example, the offset values describe
     * when each `backgroundColor` value is applied. The color is red at the start, and changes to
     * blue when 20% of the total time has elapsed.
     *
     * ```typescript
     * // the provided offset values
     * animate("5s", keyframes([
     *   style({ backgroundColor: "red", offset: 0 }),
     *   style({ backgroundColor: "blue", offset: 0.2 }),
     *   style({ backgroundColor: "orange", offset: 0.3 }),
     *   style({ backgroundColor: "black", offset: 1 })
     * ]))
     * ```
     *
     * If there are no `offset` values specified in the style entries, the offsets
     * are calculated automatically.
     *
     * ```typescript
     * animate("5s", keyframes([
     *   style({ backgroundColor: "red" }) // offset = 0
     *   style({ backgroundColor: "blue" }) // offset = 0.33
     *   style({ backgroundColor: "orange" }) // offset = 0.66
     *   style({ backgroundColor: "black" }) // offset = 1
     * ]))
     * ```
     * \@publicApi
     * @param {?} steps A set of animation styles with optional offset data.
     * The optional `offset` value for a style specifies a percentage of the total animation
     * time at which that style is applied.
     * @return {?} An object that encapsulates the keyframes data.
     *
     */
    function keyframes(steps) {
        return { type: 5 /* Keyframes */, steps };
    }
    /**
     * Declares an animation transition as a sequence of animation steps to run when a given
     * condition is satisfied. The condition is a Boolean expression or function that compares
     * the previous and current animation states, and returns true if this transition should occur.
     * When the state criteria of a defined transition are met, the associated animation is
     * triggered.
     *
     * \@usageNotes
     * The template associated with a component binds an animation trigger to an element.
     *
     * ```HTML
     * <!-- somewhere inside of my-component-tpl.html -->
     * <div [\@myAnimationTrigger]="myStatusExp">...</div>
     * ```
     *
     * All transitions are defined within an animation trigger,
     * along with named states that the transitions change to and from.
     *
     * ```typescript
     * trigger("myAnimationTrigger", [
     *  // define states
     *  state("on", style({ background: "green" })),
     *  state("off", style({ background: "grey" })),
     *  ...]
     * ```
     *
     * Note that when you call the `sequence()` function within a `{\@link animations/group group()}`
     * or a `transition()` call, execution does not continue to the next instruction
     * until each of the inner animation steps have completed.
     *
     * ### Syntax examples
     *
     * The following examples define transitions between the two defined states (and default states),
     * using various options:
     *
     * ```typescript
     * // Transition occurs when the state value
     * // bound to "myAnimationTrigger" changes from "on" to "off"
     * transition("on => off", animate(500))
     * // Run the same animation for both directions
     * transition("on <=> off", animate(500))
     * // Define multiple state-change pairs separated by commas
     * transition("on => off, off => void", animate(500))
     * ```
     *
     * ### Special values for state-change expressions
     *
     * - Catch-all state change for when an element is inserted into the page and the
     * destination state is unknown:
     *
     * ```typescript
     * transition("void => *", [
     *  style({ opacity: 0 }),
     *  animate(500)
     *  ])
     * ```
     *
     * - Capture a state change between any states:
     *
     *  `transition("* => *", animate("1s 0s"))`
     *
     * - Entry and exit transitions:
     *
     * ```typescript
     * transition(":enter", [
     *   style({ opacity: 0 }),
     *   animate(500, style({ opacity: 1 }))
     *   ]),
     * transition(":leave", [
     *   animate(500, style({ opacity: 0 }))
     *   ])
     * ```
     *
     * - Use `:increment` and `:decrement` to initiate transitions:
     *
     * ```typescript
     * transition(":increment", group([
     *  query(':enter', [
     *     style({ left: '100%' }),
     *     animate('0.5s ease-out', style('*'))
     *   ]),
     *  query(':leave', [
     *     animate('0.5s ease-out', style({ left: '-100%' }))
     *  ])
     * ]))
     *
     * transition(":decrement", group([
     *  query(':enter', [
     *     style({ left: '100%' }),
     *     animate('0.5s ease-out', style('*'))
     *   ]),
     *  query(':leave', [
     *     animate('0.5s ease-out', style({ left: '-100%' }))
     *  ])
     * ]))
     * ```
     *
     * ### State-change functions
     *
     * Here is an example of a `fromState` specified as a state-change function that invokes an
     * animation when true:
     *
     * ```typescript
     * transition((fromState, toState) =>
     *  {
     *   return fromState == "off" && toState == "on";
     *  },
     *  animate("1s 0s"))
     * ```
     *
     * ### Animating to the final state
     *
     * If the final step in a transition is a call to `animate()` that uses a timing value
     * with no style data, that step is automatically considered the final animation arc,
     * for the element to reach the final state. Angular automatically adds or removes
     * CSS styles to ensure that the element is in the correct final state.
     *
     * The following example defines a transition that starts by hiding the element,
     * then makes sure that it animates properly to whatever state is currently active for trigger:
     *
     * ```typescript
     * transition("void => *", [
     *   style({ opacity: 0 }),
     *   animate(500)
     *  ])
     * ```
     * ### Boolean value matching
     * If a trigger binding value is a Boolean, it can be matched using a transition expression
     * that compares true and false or 1 and 0. For example:
     *
     * ```
     * // in the template
     * <div [\@openClose]="open ? true : false">...</div>
     * // in the component metadata
     * trigger('openClose', [
     *   state('true', style({ height: '*' })),
     *   state('false', style({ height: '0px' })),
     *   transition('false <=> true', animate(500))
     * ])
     * ```
     *
     * \@publicApi
     *
     * @param {?} stateChangeExpr A Boolean expression or function that compares the previous and current
     * animation states, and returns true if this transition should occur. Note that  "true" and "false"
     * match 1 and 0, respectively. An expression is evaluated each time a state change occurs in the
     * animation trigger element.
     * The animation steps run when the expression evaluates to true.
     *
     * - A state-change string takes the form "state1 => state2", where each side is a defined animation
     * state, or an asterix (*) to refer to a dynamic start or end state.
     *   - The expression string can contain multiple comma-separated statements;
     * for example "state1 => state2, state3 => state4".
     *   - Special values `:enter` and `:leave` initiate a transition on the entry and exit states,
     * equivalent to  "void => *"  and "* => void".
     *   - Special values `:increment` and `:decrement` initiate a transition when a numeric value has
     * increased or decreased in value.
     * - A function is executed each time a state change occurs in the animation trigger element.
     * The animation steps run when the function returns true.
     *
     * @param {?} steps One or more animation objects, as returned by the `animate()` or
     * `sequence()` function, that form a transformation from one state to another.
     * A sequence is used by default when you pass an array.
     * @param {?=} options An options object that can contain a delay value for the start of the animation,
     * and additional developer-defined parameters. Provided values for additional parameters are used
     * as defaults, and override values can be passed to the caller on invocation.
     * @return {?} An object that encapsulates the transition data.
     *
     */
    function transition(stateChangeExpr, steps, options = null) {
        return { type: 1 /* Transition */, expr: stateChangeExpr, animation: steps, options };
    }
    /**
     * Executes a queried inner animation element within an animation sequence.
     *
     * \@usageNotes
     * Each time an animation is triggered in Angular, the parent animation
     * has priority and any child animations are blocked. In order
     * for a child animation to run, the parent animation must query each of the elements
     * containing child animations, and run them using this function.
     *
     * Note that this feature is designed to be used with `query()` and it will only work
     * with animations that are assigned using the Angular animation library. CSS keyframes
     * and transitions are not handled by this API.
     *
     * \@publicApi
     * @param {?=} options An options object that can contain a delay value for the start of the
     * animation, and additional override values for developer-defined parameters.
     * @return {?} An object that encapsulates the child animation data.
     *
     */
    function animateChild(options = null) {
        return { type: 9 /* AnimateChild */, options };
    }
    /**
     * Finds one or more inner elements within the current element that is
     * being animated within a sequence. Use with `animate()`.
     *
     * \@usageNotes
     * Tokens can be merged into a combined query selector string. For example:
     *
     * ```typescript
     *  query(':self, .record:enter, .record:leave, \@subTrigger', [...])
     * ```
     *
     * The `query()` function collects multiple elements and works internally by using
     * `element.querySelectorAll`. Use the `limit` field of an options object to limit
     * the total number of items to be collected. For example:
     *
     * ```js
     * query('div', [
     *   animate(...),
     *   animate(...)
     * ], { limit: 1 })
     * ```
     *
     * By default, throws an error when zero items are found. Set the
     * `optional` flag to ignore this error. For example:
     *
     * ```js
     * query('.some-element-that-may-not-be-there', [
     *   animate(...),
     *   animate(...)
     * ], { optional: true })
     * ```
     *
     * ### Usage Example
     *
     * The following example queries for inner elements and animates them
     * individually using `animate()`.
     *
     * ```typescript
     * \@Component({
     *   selector: 'inner',
     *   template: `
     *     <div [\@queryAnimation]="exp">
     *       <h1>Title</h1>
     *       <div class="content">
     *         Blah blah blah
     *       </div>
     *     </div>
     *   `,
     *   animations: [
     *    trigger('queryAnimation', [
     *      transition('* => goAnimate', [
     *        // hide the inner elements
     *        query('h1', style({ opacity: 0 })),
     *        query('.content', style({ opacity: 0 })),
     *
     *        // animate the inner elements in, one by one
     *        query('h1', animate(1000, style({ opacity: 1 }))),
     *        query('.content', animate(1000, style({ opacity: 1 }))),
     *      ])
     *    ])
     *  ]
     * })
     * class Cmp {
     *   exp = '';
     *
     *   goAnimate() {
     *     this.exp = 'goAnimate';
     *   }
     * }
     * ```
     *
     * \@publicApi
     * @param {?} selector The element to query, or a set of elements that contain Angular-specific
     * characteristics, specified with one or more of the following tokens.
     *  - `query(":enter")` or `query(":leave")` : Query for newly inserted/removed elements.
     *  - `query(":animating")` : Query all currently animating elements.
     *  - `query("\@triggerName")` : Query elements that contain an animation trigger.
     *  - `query("\@*")` : Query all elements that contain an animation triggers.
     *  - `query(":self")` : Include the current element into the animation sequence.
     *
     * @param {?} animation One or more animation steps to apply to the queried element or elements.
     * An array is treated as an animation sequence.
     * @param {?=} options An options object. Use the 'limit' field to limit the total number of
     * items to collect.
     * @return {?} An object that encapsulates the query data.
     *
     */
    function query(selector, animation, options = null) {
        return { type: 11 /* Query */, selector, animation, options };
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/animations/src/util.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     * @param {?} cb
     * @return {?}
     */
    function scheduleMicroTask$1(cb) {
        Promise.resolve(null).then(cb);
    }
    /**
     * An empty programmatic controller for reusable animations.
     * Used internally when animations are disabled, to avoid
     * checking for the null case when an animation player is expected.
     *
     * @see `animate()`
     * @see `AnimationPlayer`
     * @see `GroupPlayer`
     *
     * \@publicApi
     */
    class NoopAnimationPlayer {
        /**
         * @param {?=} duration
         * @param {?=} delay
         */
        constructor(duration = 0, delay = 0) {
            this._onDoneFns = [];
            this._onStartFns = [];
            this._onDestroyFns = [];
            this._started = false;
            this._destroyed = false;
            this._finished = false;
            this.parentPlayer = null;
            this.totalTime = duration + delay;
        }
        /**
         * @private
         * @return {?}
         */
        _onFinish() {
            if (!this._finished) {
                this._finished = true;
                this._onDoneFns.forEach((/**
                 * @param {?} fn
                 * @return {?}
                 */
                fn => fn()));
                this._onDoneFns = [];
            }
        }
        /**
         * @param {?} fn
         * @return {?}
         */
        onStart(fn) {
            this._onStartFns.push(fn);
        }
        /**
         * @param {?} fn
         * @return {?}
         */
        onDone(fn) {
            this._onDoneFns.push(fn);
        }
        /**
         * @param {?} fn
         * @return {?}
         */
        onDestroy(fn) {
            this._onDestroyFns.push(fn);
        }
        /**
         * @return {?}
         */
        hasStarted() {
            return this._started;
        }
        /**
         * @return {?}
         */
        init() { }
        /**
         * @return {?}
         */
        play() {
            if (!this.hasStarted()) {
                this._onStart();
                this.triggerMicrotask();
            }
            this._started = true;
        }
        /**
         * \@internal
         * @return {?}
         */
        triggerMicrotask() {
            scheduleMicroTask$1((/**
             * @return {?}
             */
            () => this._onFinish()));
        }
        /**
         * @private
         * @return {?}
         */
        _onStart() {
            this._onStartFns.forEach((/**
             * @param {?} fn
             * @return {?}
             */
            fn => fn()));
            this._onStartFns = [];
        }
        /**
         * @return {?}
         */
        pause() { }
        /**
         * @return {?}
         */
        restart() { }
        /**
         * @return {?}
         */
        finish() {
            this._onFinish();
        }
        /**
         * @return {?}
         */
        destroy() {
            if (!this._destroyed) {
                this._destroyed = true;
                if (!this.hasStarted()) {
                    this._onStart();
                }
                this.finish();
                this._onDestroyFns.forEach((/**
                 * @param {?} fn
                 * @return {?}
                 */
                fn => fn()));
                this._onDestroyFns = [];
            }
        }
        /**
         * @return {?}
         */
        reset() { }
        /**
         * @param {?} position
         * @return {?}
         */
        setPosition(position) { }
        /**
         * @return {?}
         */
        getPosition() {
            return 0;
        }
        /**
         * \@internal
         * @param {?} phaseName
         * @return {?}
         */
        triggerCallback(phaseName) {
            /** @type {?} */
            const methods = phaseName == 'start' ? this._onStartFns : this._onDoneFns;
            methods.forEach((/**
             * @param {?} fn
             * @return {?}
             */
            fn => fn()));
            methods.length = 0;
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/animations/src/players/animation_group_player.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * A programmatic controller for a group of reusable animations.
     * Used internally to control animations.
     *
     * @see `AnimationPlayer`
     * @see `{\@link animations/group group()}`
     *
     */
    class AnimationGroupPlayer {
        /**
         * @param {?} _players
         */
        constructor(_players) {
            this._onDoneFns = [];
            this._onStartFns = [];
            this._finished = false;
            this._started = false;
            this._destroyed = false;
            this._onDestroyFns = [];
            this.parentPlayer = null;
            this.totalTime = 0;
            this.players = _players;
            /** @type {?} */
            let doneCount = 0;
            /** @type {?} */
            let destroyCount = 0;
            /** @type {?} */
            let startCount = 0;
            /** @type {?} */
            const total = this.players.length;
            if (total == 0) {
                scheduleMicroTask$1((/**
                 * @return {?}
                 */
                () => this._onFinish()));
            }
            else {
                this.players.forEach((/**
                 * @param {?} player
                 * @return {?}
                 */
                player => {
                    player.onDone((/**
                     * @return {?}
                     */
                    () => {
                        if (++doneCount == total) {
                            this._onFinish();
                        }
                    }));
                    player.onDestroy((/**
                     * @return {?}
                     */
                    () => {
                        if (++destroyCount == total) {
                            this._onDestroy();
                        }
                    }));
                    player.onStart((/**
                     * @return {?}
                     */
                    () => {
                        if (++startCount == total) {
                            this._onStart();
                        }
                    }));
                }));
            }
            this.totalTime = this.players.reduce((/**
             * @param {?} time
             * @param {?} player
             * @return {?}
             */
            (time, player) => Math.max(time, player.totalTime)), 0);
        }
        /**
         * @private
         * @return {?}
         */
        _onFinish() {
            if (!this._finished) {
                this._finished = true;
                this._onDoneFns.forEach((/**
                 * @param {?} fn
                 * @return {?}
                 */
                fn => fn()));
                this._onDoneFns = [];
            }
        }
        /**
         * @return {?}
         */
        init() {
            this.players.forEach((/**
             * @param {?} player
             * @return {?}
             */
            player => player.init()));
        }
        /**
         * @param {?} fn
         * @return {?}
         */
        onStart(fn) {
            this._onStartFns.push(fn);
        }
        /**
         * @private
         * @return {?}
         */
        _onStart() {
            if (!this.hasStarted()) {
                this._started = true;
                this._onStartFns.forEach((/**
                 * @param {?} fn
                 * @return {?}
                 */
                fn => fn()));
                this._onStartFns = [];
            }
        }
        /**
         * @param {?} fn
         * @return {?}
         */
        onDone(fn) {
            this._onDoneFns.push(fn);
        }
        /**
         * @param {?} fn
         * @return {?}
         */
        onDestroy(fn) {
            this._onDestroyFns.push(fn);
        }
        /**
         * @return {?}
         */
        hasStarted() {
            return this._started;
        }
        /**
         * @return {?}
         */
        play() {
            if (!this.parentPlayer) {
                this.init();
            }
            this._onStart();
            this.players.forEach((/**
             * @param {?} player
             * @return {?}
             */
            player => player.play()));
        }
        /**
         * @return {?}
         */
        pause() {
            this.players.forEach((/**
             * @param {?} player
             * @return {?}
             */
            player => player.pause()));
        }
        /**
         * @return {?}
         */
        restart() {
            this.players.forEach((/**
             * @param {?} player
             * @return {?}
             */
            player => player.restart()));
        }
        /**
         * @return {?}
         */
        finish() {
            this._onFinish();
            this.players.forEach((/**
             * @param {?} player
             * @return {?}
             */
            player => player.finish()));
        }
        /**
         * @return {?}
         */
        destroy() {
            this._onDestroy();
        }
        /**
         * @private
         * @return {?}
         */
        _onDestroy() {
            if (!this._destroyed) {
                this._destroyed = true;
                this._onFinish();
                this.players.forEach((/**
                 * @param {?} player
                 * @return {?}
                 */
                player => player.destroy()));
                this._onDestroyFns.forEach((/**
                 * @param {?} fn
                 * @return {?}
                 */
                fn => fn()));
                this._onDestroyFns = [];
            }
        }
        /**
         * @return {?}
         */
        reset() {
            this.players.forEach((/**
             * @param {?} player
             * @return {?}
             */
            player => player.reset()));
            this._destroyed = false;
            this._finished = false;
            this._started = false;
        }
        /**
         * @param {?} p
         * @return {?}
         */
        setPosition(p) {
            /** @type {?} */
            const timeAtPosition = p * this.totalTime;
            this.players.forEach((/**
             * @param {?} player
             * @return {?}
             */
            player => {
                /** @type {?} */
                const position = player.totalTime ? Math.min(1, timeAtPosition / player.totalTime) : 1;
                player.setPosition(position);
            }));
        }
        /**
         * @return {?}
         */
        getPosition() {
            /** @type {?} */
            let min = 0;
            this.players.forEach((/**
             * @param {?} player
             * @return {?}
             */
            player => {
                /** @type {?} */
                const p = player.getPosition();
                min = Math.min(p, min);
            }));
            return min;
        }
        /**
         * @return {?}
         */
        beforeDestroy() {
            this.players.forEach((/**
             * @param {?} player
             * @return {?}
             */
            player => {
                if (player.beforeDestroy) {
                    player.beforeDestroy();
                }
            }));
        }
        /**
         * \@internal
         * @param {?} phaseName
         * @return {?}
         */
        triggerCallback(phaseName) {
            /** @type {?} */
            const methods = phaseName == 'start' ? this._onStartFns : this._onDoneFns;
            methods.forEach((/**
             * @param {?} fn
             * @return {?}
             */
            fn => fn()));
            methods.length = 0;
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/animations/src/private_export.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const ɵPRE_STYLE = '!';

    /**
     * @license Angular v9.1.12
     * (c) 2010-2020 Google LLC. https://angular.io/
     * License: MIT
     */
    function isBrowser() {
        return (typeof window !== 'undefined' && typeof window.document !== 'undefined');
    }
    /**
     * @return {?}
     */
    function isNode() {
        // Checking only for `process` isn't enough to identify whether or not we're in a Node
        // environment, because Webpack by default will polyfill the `process`. While we can discern
        // that Webpack polyfilled it by looking at `process.browser`, it's very Webpack-specific and
        // might not be future-proof. Instead we look at the stringified version of `process` which
        // is `[object process]` in Node and `[object Object]` when polyfilled.
        return typeof process !== 'undefined' && {}.toString.call(process) === '[object process]';
    }
    /**
     * @param {?} players
     * @return {?}
     */
    function optimizeGroupPlayer(players) {
        switch (players.length) {
            case 0:
                return new NoopAnimationPlayer();
            case 1:
                return players[0];
            default:
                return new AnimationGroupPlayer(players);
        }
    }
    /**
     * @param {?} driver
     * @param {?} normalizer
     * @param {?} element
     * @param {?} keyframes
     * @param {?=} preStyles
     * @param {?=} postStyles
     * @return {?}
     */
    function normalizeKeyframes(driver, normalizer, element, keyframes, preStyles = {}, postStyles = {}) {
        /** @type {?} */
        const errors = [];
        /** @type {?} */
        const normalizedKeyframes = [];
        /** @type {?} */
        let previousOffset = -1;
        /** @type {?} */
        let previousKeyframe = null;
        keyframes.forEach((/**
         * @param {?} kf
         * @return {?}
         */
        kf => {
            /** @type {?} */
            const offset = (/** @type {?} */ (kf['offset']));
            /** @type {?} */
            const isSameOffset = offset == previousOffset;
            /** @type {?} */
            const normalizedKeyframe = (isSameOffset && previousKeyframe) || {};
            Object.keys(kf).forEach((/**
             * @param {?} prop
             * @return {?}
             */
            prop => {
                /** @type {?} */
                let normalizedProp = prop;
                /** @type {?} */
                let normalizedValue = kf[prop];
                if (prop !== 'offset') {
                    normalizedProp = normalizer.normalizePropertyName(normalizedProp, errors);
                    switch (normalizedValue) {
                        case ɵPRE_STYLE:
                            normalizedValue = preStyles[prop];
                            break;
                        case AUTO_STYLE:
                            normalizedValue = postStyles[prop];
                            break;
                        default:
                            normalizedValue =
                                normalizer.normalizeStyleValue(prop, normalizedProp, normalizedValue, errors);
                            break;
                    }
                }
                normalizedKeyframe[normalizedProp] = normalizedValue;
            }));
            if (!isSameOffset) {
                normalizedKeyframes.push(normalizedKeyframe);
            }
            previousKeyframe = normalizedKeyframe;
            previousOffset = offset;
        }));
        if (errors.length) {
            /** @type {?} */
            const LINE_START = '\n - ';
            throw new Error(`Unable to animate due to the following errors:${LINE_START}${errors.join(LINE_START)}`);
        }
        return normalizedKeyframes;
    }
    /**
     * @param {?} player
     * @param {?} eventName
     * @param {?} event
     * @param {?} callback
     * @return {?}
     */
    function listenOnPlayer(player, eventName, event, callback) {
        switch (eventName) {
            case 'start':
                player.onStart((/**
                 * @return {?}
                 */
                () => callback(event && copyAnimationEvent(event, 'start', player))));
                break;
            case 'done':
                player.onDone((/**
                 * @return {?}
                 */
                () => callback(event && copyAnimationEvent(event, 'done', player))));
                break;
            case 'destroy':
                player.onDestroy((/**
                 * @return {?}
                 */
                () => callback(event && copyAnimationEvent(event, 'destroy', player))));
                break;
        }
    }
    /**
     * @param {?} e
     * @param {?} phaseName
     * @param {?} player
     * @return {?}
     */
    function copyAnimationEvent(e, phaseName, player) {
        /** @type {?} */
        const totalTime = player.totalTime;
        /** @type {?} */
        const disabled = ((/** @type {?} */ (player))).disabled ? true : false;
        /** @type {?} */
        const event = makeAnimationEvent(e.element, e.triggerName, e.fromState, e.toState, phaseName || e.phaseName, totalTime == undefined ? e.totalTime : totalTime, disabled);
        /** @type {?} */
        const data = ((/** @type {?} */ (e)))['_data'];
        if (data != null) {
            ((/** @type {?} */ (event)))['_data'] = data;
        }
        return event;
    }
    /**
     * @param {?} element
     * @param {?} triggerName
     * @param {?} fromState
     * @param {?} toState
     * @param {?=} phaseName
     * @param {?=} totalTime
     * @param {?=} disabled
     * @return {?}
     */
    function makeAnimationEvent(element, triggerName, fromState, toState, phaseName = '', totalTime = 0, disabled) {
        return { element, triggerName, fromState, toState, phaseName, totalTime, disabled: !!disabled };
    }
    /**
     * @param {?} map
     * @param {?} key
     * @param {?} defaultValue
     * @return {?}
     */
    function getOrSetAsInMap(map, key, defaultValue) {
        /** @type {?} */
        let value;
        if (map instanceof Map) {
            value = map.get(key);
            if (!value) {
                map.set(key, value = defaultValue);
            }
        }
        else {
            value = map[key];
            if (!value) {
                value = map[key] = defaultValue;
            }
        }
        return value;
    }
    /**
     * @param {?} command
     * @return {?}
     */
    function parseTimelineCommand(command) {
        /** @type {?} */
        const separatorPos = command.indexOf(':');
        /** @type {?} */
        const id = command.substring(1, separatorPos);
        /** @type {?} */
        const action = command.substr(separatorPos + 1);
        return [id, action];
    }
    /** @type {?} */
    let _contains = (/**
     * @param {?} elm1
     * @param {?} elm2
     * @return {?}
     */
    (elm1, elm2) => false);
    /** @type {?} */
    let _matches = (/**
     * @param {?} element
     * @param {?} selector
     * @return {?}
     */
    (element, selector) => false);
    /** @type {?} */
    let _query = (/**
     * @param {?} element
     * @param {?} selector
     * @param {?} multi
     * @return {?}
     */
    (element, selector, multi) => {
        return [];
    });
    // Define utility methods for browsers and platform-server(domino) where Element
    // and utility methods exist.
    /** @type {?} */
    const _isNode = isNode();
    if (_isNode || typeof Element !== 'undefined') {
        // this is well supported in all browsers
        _contains = (/**
         * @param {?} elm1
         * @param {?} elm2
         * @return {?}
         */
        (elm1, elm2) => {
            return (/** @type {?} */ (elm1.contains(elm2)));
        });
        _matches = ((/**
         * @return {?}
         */
        () => {
            if (_isNode || Element.prototype.matches) {
                return (/**
                 * @param {?} element
                 * @param {?} selector
                 * @return {?}
                 */
                (element, selector) => element.matches(selector));
            }
            else {
                /** @type {?} */
                const proto = (/** @type {?} */ (Element.prototype));
                /** @type {?} */
                const fn = proto.matchesSelector || proto.mozMatchesSelector || proto.msMatchesSelector ||
                    proto.oMatchesSelector || proto.webkitMatchesSelector;
                if (fn) {
                    return (/**
                     * @param {?} element
                     * @param {?} selector
                     * @return {?}
                     */
                    (element, selector) => fn.apply(element, [selector]));
                }
                else {
                    return _matches;
                }
            }
        }))();
        _query = (/**
         * @param {?} element
         * @param {?} selector
         * @param {?} multi
         * @return {?}
         */
        (element, selector, multi) => {
            /** @type {?} */
            let results = [];
            if (multi) {
                results.push(...element.querySelectorAll(selector));
            }
            else {
                /** @type {?} */
                const elm = element.querySelector(selector);
                if (elm) {
                    results.push(elm);
                }
            }
            return results;
        });
    }
    /**
     * @param {?} prop
     * @return {?}
     */
    function containsVendorPrefix(prop) {
        // Webkit is the only real popular vendor prefix nowadays
        // cc: http://shouldiprefix.com/
        return prop.substring(1, 6) == 'ebkit'; // webkit or Webkit
    }
    /** @type {?} */
    let _CACHED_BODY = null;
    /** @type {?} */
    let _IS_WEBKIT = false;
    /**
     * @param {?} prop
     * @return {?}
     */
    function validateStyleProperty(prop) {
        if (!_CACHED_BODY) {
            _CACHED_BODY = getBodyNode() || {};
            _IS_WEBKIT = (/** @type {?} */ (_CACHED_BODY)).style ? ('WebkitAppearance' in (/** @type {?} */ (_CACHED_BODY)).style) : false;
        }
        /** @type {?} */
        let result = true;
        if ((/** @type {?} */ (_CACHED_BODY)).style && !containsVendorPrefix(prop)) {
            result = prop in (/** @type {?} */ (_CACHED_BODY)).style;
            if (!result && _IS_WEBKIT) {
                /** @type {?} */
                const camelProp = 'Webkit' + prop.charAt(0).toUpperCase() + prop.substr(1);
                result = camelProp in (/** @type {?} */ (_CACHED_BODY)).style;
            }
        }
        return result;
    }
    /**
     * @return {?}
     */
    function getBodyNode() {
        if (typeof document != 'undefined') {
            return document.body;
        }
        return null;
    }
    /** @type {?} */
    const matchesElement = _matches;
    /** @type {?} */
    const containsElement = _contains;
    /** @type {?} */
    const invokeQuery = _query;
    /**
     * @param {?} object
     * @return {?}
     */
    function hypenatePropsObject(object) {
        /** @type {?} */
        const newObj = {};
        Object.keys(object).forEach((/**
         * @param {?} prop
         * @return {?}
         */
        prop => {
            /** @type {?} */
            const newProp = prop.replace(/([a-z])([A-Z])/g, '$1-$2');
            newObj[newProp] = object[prop];
        }));
        return newObj;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/animations/browser/src/render/animation_driver.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * \@publicApi
     */
    class NoopAnimationDriver {
        /**
         * @param {?} prop
         * @return {?}
         */
        validateStyleProperty(prop) {
            return validateStyleProperty(prop);
        }
        /**
         * @param {?} element
         * @param {?} selector
         * @return {?}
         */
        matchesElement(element, selector) {
            return matchesElement(element, selector);
        }
        /**
         * @param {?} elm1
         * @param {?} elm2
         * @return {?}
         */
        containsElement(elm1, elm2) {
            return containsElement(elm1, elm2);
        }
        /**
         * @param {?} element
         * @param {?} selector
         * @param {?} multi
         * @return {?}
         */
        query(element, selector, multi) {
            return invokeQuery(element, selector, multi);
        }
        /**
         * @param {?} element
         * @param {?} prop
         * @param {?=} defaultValue
         * @return {?}
         */
        computeStyle(element, prop, defaultValue) {
            return defaultValue || '';
        }
        /**
         * @param {?} element
         * @param {?} keyframes
         * @param {?} duration
         * @param {?} delay
         * @param {?} easing
         * @param {?=} previousPlayers
         * @param {?=} scrubberAccessRequested
         * @return {?}
         */
        animate(element, keyframes, duration, delay, easing, previousPlayers = [], scrubberAccessRequested) {
            return new NoopAnimationPlayer(duration, delay);
        }
    }
    NoopAnimationDriver.ɵfac = function NoopAnimationDriver_Factory(t) { return new (t || NoopAnimationDriver)(); };
    NoopAnimationDriver.ɵprov = ɵɵdefineInjectable({ token: NoopAnimationDriver, factory: NoopAnimationDriver.ɵfac });
    /**
     * \@publicApi
     * @abstract
     */
    class AnimationDriver {
    }
    AnimationDriver.NOOP = new NoopAnimationDriver();

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/animations/browser/src/util.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const ONE_SECOND = 1000;
    /** @type {?} */
    const SUBSTITUTION_EXPR_START = '{{';
    /** @type {?} */
    const SUBSTITUTION_EXPR_END = '}}';
    /** @type {?} */
    const ENTER_CLASSNAME = 'ng-enter';
    /** @type {?} */
    const LEAVE_CLASSNAME = 'ng-leave';
    /** @type {?} */
    const NG_TRIGGER_CLASSNAME = 'ng-trigger';
    /** @type {?} */
    const NG_TRIGGER_SELECTOR = '.ng-trigger';
    /** @type {?} */
    const NG_ANIMATING_CLASSNAME = 'ng-animating';
    /** @type {?} */
    const NG_ANIMATING_SELECTOR = '.ng-animating';
    /**
     * @param {?} value
     * @return {?}
     */
    function resolveTimingValue(value) {
        if (typeof value == 'number')
            return value;
        /** @type {?} */
        const matches = value.match(/^(-?[\.\d]+)(m?s)/);
        if (!matches || matches.length < 2)
            return 0;
        return _convertTimeValueToMS(parseFloat(matches[1]), matches[2]);
    }
    /**
     * @param {?} value
     * @param {?} unit
     * @return {?}
     */
    function _convertTimeValueToMS(value, unit) {
        switch (unit) {
            case 's':
                return value * ONE_SECOND;
            default: // ms or something else
                return value;
        }
    }
    /**
     * @param {?} timings
     * @param {?} errors
     * @param {?=} allowNegativeValues
     * @return {?}
     */
    function resolveTiming(timings, errors, allowNegativeValues) {
        return timings.hasOwnProperty('duration') ?
            (/** @type {?} */ (timings)) :
            parseTimeExpression((/** @type {?} */ (timings)), errors, allowNegativeValues);
    }
    /**
     * @param {?} exp
     * @param {?} errors
     * @param {?=} allowNegativeValues
     * @return {?}
     */
    function parseTimeExpression(exp, errors, allowNegativeValues) {
        /** @type {?} */
        const regex = /^(-?[\.\d]+)(m?s)(?:\s+(-?[\.\d]+)(m?s))?(?:\s+([-a-z]+(?:\(.+?\))?))?$/i;
        /** @type {?} */
        let duration;
        /** @type {?} */
        let delay = 0;
        /** @type {?} */
        let easing = '';
        if (typeof exp === 'string') {
            /** @type {?} */
            const matches = exp.match(regex);
            if (matches === null) {
                errors.push(`The provided timing value "${exp}" is invalid.`);
                return { duration: 0, delay: 0, easing: '' };
            }
            duration = _convertTimeValueToMS(parseFloat(matches[1]), matches[2]);
            /** @type {?} */
            const delayMatch = matches[3];
            if (delayMatch != null) {
                delay = _convertTimeValueToMS(parseFloat(delayMatch), matches[4]);
            }
            /** @type {?} */
            const easingVal = matches[5];
            if (easingVal) {
                easing = easingVal;
            }
        }
        else {
            duration = exp;
        }
        if (!allowNegativeValues) {
            /** @type {?} */
            let containsErrors = false;
            /** @type {?} */
            let startIndex = errors.length;
            if (duration < 0) {
                errors.push(`Duration values below 0 are not allowed for this animation step.`);
                containsErrors = true;
            }
            if (delay < 0) {
                errors.push(`Delay values below 0 are not allowed for this animation step.`);
                containsErrors = true;
            }
            if (containsErrors) {
                errors.splice(startIndex, 0, `The provided timing value "${exp}" is invalid.`);
            }
        }
        return { duration, delay, easing };
    }
    /**
     * @param {?} obj
     * @param {?=} destination
     * @return {?}
     */
    function copyObj(obj, destination = {}) {
        Object.keys(obj).forEach((/**
         * @param {?} prop
         * @return {?}
         */
        prop => {
            destination[prop] = obj[prop];
        }));
        return destination;
    }
    /**
     * @param {?} styles
     * @param {?} readPrototype
     * @param {?=} destination
     * @return {?}
     */
    function copyStyles(styles, readPrototype, destination = {}) {
        if (readPrototype) {
            // we make use of a for-in loop so that the
            // prototypically inherited properties are
            // revealed from the backFill map
            for (let prop in styles) {
                destination[prop] = styles[prop];
            }
        }
        else {
            copyObj(styles, destination);
        }
        return destination;
    }
    /**
     * @param {?} element
     * @param {?} key
     * @param {?} value
     * @return {?}
     */
    function getStyleAttributeString(element, key, value) {
        // Return the key-value pair string to be added to the style attribute for the
        // given CSS style key.
        if (value) {
            return key + ':' + value + ';';
        }
        else {
            return '';
        }
    }
    /**
     * @param {?} element
     * @return {?}
     */
    function writeStyleAttribute(element) {
        // Read the style property of the element and manually reflect it to the
        // style attribute. This is needed because Domino on platform-server doesn't
        // understand the full set of allowed CSS properties and doesn't reflect some
        // of them automatically.
        /** @type {?} */
        let styleAttrValue = '';
        for (let i = 0; i < element.style.length; i++) {
            /** @type {?} */
            const key = element.style.item(i);
            styleAttrValue += getStyleAttributeString(element, key, element.style.getPropertyValue(key));
        }
        for (const key in element.style) {
            // Skip internal Domino properties that don't need to be reflected.
            if (!element.style.hasOwnProperty(key) || key.startsWith('_')) {
                continue;
            }
            /** @type {?} */
            const dashKey = camelCaseToDashCase$1(key);
            styleAttrValue += getStyleAttributeString(element, dashKey, element.style[key]);
        }
        element.setAttribute('style', styleAttrValue);
    }
    /**
     * @param {?} element
     * @param {?} styles
     * @param {?=} formerStyles
     * @return {?}
     */
    function setStyles(element, styles, formerStyles) {
        if (element['style']) {
            Object.keys(styles).forEach((/**
             * @param {?} prop
             * @return {?}
             */
            prop => {
                /** @type {?} */
                const camelProp = dashCaseToCamelCase(prop);
                if (formerStyles && !formerStyles.hasOwnProperty(prop)) {
                    formerStyles[prop] = element.style[camelProp];
                }
                element.style[camelProp] = styles[prop];
            }));
            // On the server set the 'style' attribute since it's not automatically reflected.
            if (isNode()) {
                writeStyleAttribute(element);
            }
        }
    }
    /**
     * @param {?} element
     * @param {?} styles
     * @return {?}
     */
    function eraseStyles(element, styles) {
        if (element['style']) {
            Object.keys(styles).forEach((/**
             * @param {?} prop
             * @return {?}
             */
            prop => {
                /** @type {?} */
                const camelProp = dashCaseToCamelCase(prop);
                element.style[camelProp] = '';
            }));
            // On the server set the 'style' attribute since it's not automatically reflected.
            if (isNode()) {
                writeStyleAttribute(element);
            }
        }
    }
    /**
     * @param {?} steps
     * @return {?}
     */
    function normalizeAnimationEntry(steps) {
        if (Array.isArray(steps)) {
            if (steps.length == 1)
                return steps[0];
            return sequence(steps);
        }
        return (/** @type {?} */ (steps));
    }
    /**
     * @param {?} value
     * @param {?} options
     * @param {?} errors
     * @return {?}
     */
    function validateStyleParams(value, options, errors) {
        /** @type {?} */
        const params = options.params || {};
        /** @type {?} */
        const matches = extractStyleParams(value);
        if (matches.length) {
            matches.forEach((/**
             * @param {?} varName
             * @return {?}
             */
            varName => {
                if (!params.hasOwnProperty(varName)) {
                    errors.push(`Unable to resolve the local animation param ${varName} in the given list of values`);
                }
            }));
        }
    }
    /** @type {?} */
    const PARAM_REGEX = new RegExp(`${SUBSTITUTION_EXPR_START}\\s*(.+?)\\s*${SUBSTITUTION_EXPR_END}`, 'g');
    /**
     * @param {?} value
     * @return {?}
     */
    function extractStyleParams(value) {
        /** @type {?} */
        let params = [];
        if (typeof value === 'string') {
            /** @type {?} */
            let match;
            while (match = PARAM_REGEX.exec(value)) {
                params.push((/** @type {?} */ (match[1])));
            }
            PARAM_REGEX.lastIndex = 0;
        }
        return params;
    }
    /**
     * @param {?} value
     * @param {?} params
     * @param {?} errors
     * @return {?}
     */
    function interpolateParams(value, params, errors) {
        /** @type {?} */
        const original = value.toString();
        /** @type {?} */
        const str = original.replace(PARAM_REGEX, (/**
         * @param {?} _
         * @param {?} varName
         * @return {?}
         */
        (_, varName) => {
            /** @type {?} */
            let localVal = params[varName];
            // this means that the value was never overridden by the data passed in by the user
            if (!params.hasOwnProperty(varName)) {
                errors.push(`Please provide a value for the animation param ${varName}`);
                localVal = '';
            }
            return localVal.toString();
        }));
        // we do this to assert that numeric values stay as they are
        return str == original ? value : str;
    }
    /**
     * @param {?} iterator
     * @return {?}
     */
    function iteratorToArray(iterator) {
        /** @type {?} */
        const arr = [];
        /** @type {?} */
        let item = iterator.next();
        while (!item.done) {
            arr.push(item.value);
            item = iterator.next();
        }
        return arr;
    }
    /** @type {?} */
    const DASH_CASE_REGEXP = /-+([a-z0-9])/g;
    /**
     * @param {?} input
     * @return {?}
     */
    function dashCaseToCamelCase(input) {
        return input.replace(DASH_CASE_REGEXP, (/**
         * @param {...?} m
         * @return {?}
         */
        (...m) => m[1].toUpperCase()));
    }
    /**
     * @param {?} input
     * @return {?}
     */
    function camelCaseToDashCase$1(input) {
        return input.replace(/([a-z])([A-Z])/g, '$1-$2').toLowerCase();
    }
    /**
     * @param {?} duration
     * @param {?} delay
     * @return {?}
     */
    function allowPreviousPlayerStylesMerge(duration, delay) {
        return duration === 0 || delay === 0;
    }
    /**
     * @param {?} element
     * @param {?} keyframes
     * @param {?} previousStyles
     * @return {?}
     */
    function balancePreviousStylesIntoKeyframes(element, keyframes, previousStyles) {
        /** @type {?} */
        const previousStyleProps = Object.keys(previousStyles);
        if (previousStyleProps.length && keyframes.length) {
            /** @type {?} */
            let startingKeyframe = keyframes[0];
            /** @type {?} */
            let missingStyleProps = [];
            previousStyleProps.forEach((/**
             * @param {?} prop
             * @return {?}
             */
            prop => {
                if (!startingKeyframe.hasOwnProperty(prop)) {
                    missingStyleProps.push(prop);
                }
                startingKeyframe[prop] = previousStyles[prop];
            }));
            if (missingStyleProps.length) {
                // tslint:disable-next-line
                for (var i = 1; i < keyframes.length; i++) {
                    /** @type {?} */
                    let kf = keyframes[i];
                    missingStyleProps.forEach((/**
                     * @param {?} prop
                     * @return {?}
                     */
                    function (prop) {
                        kf[prop] = computeStyle(element, prop);
                    }));
                }
            }
        }
        return keyframes;
    }
    /**
     * @param {?} visitor
     * @param {?} node
     * @param {?} context
     * @return {?}
     */
    function visitDslNode(visitor, node, context) {
        switch (node.type) {
            case 7 /* Trigger */:
                return visitor.visitTrigger(node, context);
            case 0 /* State */:
                return visitor.visitState(node, context);
            case 1 /* Transition */:
                return visitor.visitTransition(node, context);
            case 2 /* Sequence */:
                return visitor.visitSequence(node, context);
            case 3 /* Group */:
                return visitor.visitGroup(node, context);
            case 4 /* Animate */:
                return visitor.visitAnimate(node, context);
            case 5 /* Keyframes */:
                return visitor.visitKeyframes(node, context);
            case 6 /* Style */:
                return visitor.visitStyle(node, context);
            case 8 /* Reference */:
                return visitor.visitReference(node, context);
            case 9 /* AnimateChild */:
                return visitor.visitAnimateChild(node, context);
            case 10 /* AnimateRef */:
                return visitor.visitAnimateRef(node, context);
            case 11 /* Query */:
                return visitor.visitQuery(node, context);
            case 12 /* Stagger */:
                return visitor.visitStagger(node, context);
            default:
                throw new Error(`Unable to resolve animation metadata node #${node.type}`);
        }
    }
    /**
     * @param {?} element
     * @param {?} prop
     * @return {?}
     */
    function computeStyle(element, prop) {
        return ((/** @type {?} */ (window.getComputedStyle(element))))[prop];
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/animations/browser/src/dsl/animation_transition_expr.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     * @type {?}
     */
    const ANY_STATE = '*';
    /**
     * @param {?} transitionValue
     * @param {?} errors
     * @return {?}
     */
    function parseTransitionExpr(transitionValue, errors) {
        /** @type {?} */
        const expressions = [];
        if (typeof transitionValue == 'string') {
            transitionValue.split(/\s*,\s*/).forEach((/**
             * @param {?} str
             * @return {?}
             */
            str => parseInnerTransitionStr(str, expressions, errors)));
        }
        else {
            expressions.push((/** @type {?} */ (transitionValue)));
        }
        return expressions;
    }
    /**
     * @param {?} eventStr
     * @param {?} expressions
     * @param {?} errors
     * @return {?}
     */
    function parseInnerTransitionStr(eventStr, expressions, errors) {
        if (eventStr[0] == ':') {
            /** @type {?} */
            const result = parseAnimationAlias(eventStr, errors);
            if (typeof result == 'function') {
                expressions.push(result);
                return;
            }
            eventStr = result;
        }
        /** @type {?} */
        const match = eventStr.match(/^(\*|[-\w]+)\s*(<?[=-]>)\s*(\*|[-\w]+)$/);
        if (match == null || match.length < 4) {
            errors.push(`The provided transition expression "${eventStr}" is not supported`);
            return expressions;
        }
        /** @type {?} */
        const fromState = match[1];
        /** @type {?} */
        const separator = match[2];
        /** @type {?} */
        const toState = match[3];
        expressions.push(makeLambdaFromStates(fromState, toState));
        /** @type {?} */
        const isFullAnyStateExpr = fromState == ANY_STATE && toState == ANY_STATE;
        if (separator[0] == '<' && !isFullAnyStateExpr) {
            expressions.push(makeLambdaFromStates(toState, fromState));
        }
    }
    /**
     * @param {?} alias
     * @param {?} errors
     * @return {?}
     */
    function parseAnimationAlias(alias, errors) {
        switch (alias) {
            case ':enter':
                return 'void => *';
            case ':leave':
                return '* => void';
            case ':increment':
                return (/**
                 * @param {?} fromState
                 * @param {?} toState
                 * @return {?}
                 */
                (fromState, toState) => parseFloat(toState) > parseFloat(fromState));
            case ':decrement':
                return (/**
                 * @param {?} fromState
                 * @param {?} toState
                 * @return {?}
                 */
                (fromState, toState) => parseFloat(toState) < parseFloat(fromState));
            default:
                errors.push(`The transition alias value "${alias}" is not supported`);
                return '* => *';
        }
    }
    // DO NOT REFACTOR ... keep the follow set instantiations
    // with the values intact (closure compiler for some reason
    // removes follow-up lines that add the values outside of
    // the constructor...
    /** @type {?} */
    const TRUE_BOOLEAN_VALUES = new Set(['true', '1']);
    /** @type {?} */
    const FALSE_BOOLEAN_VALUES = new Set(['false', '0']);
    /**
     * @param {?} lhs
     * @param {?} rhs
     * @return {?}
     */
    function makeLambdaFromStates(lhs, rhs) {
        /** @type {?} */
        const LHS_MATCH_BOOLEAN = TRUE_BOOLEAN_VALUES.has(lhs) || FALSE_BOOLEAN_VALUES.has(lhs);
        /** @type {?} */
        const RHS_MATCH_BOOLEAN = TRUE_BOOLEAN_VALUES.has(rhs) || FALSE_BOOLEAN_VALUES.has(rhs);
        return (/**
         * @param {?} fromState
         * @param {?} toState
         * @return {?}
         */
        (fromState, toState) => {
            /** @type {?} */
            let lhsMatch = lhs == ANY_STATE || lhs == fromState;
            /** @type {?} */
            let rhsMatch = rhs == ANY_STATE || rhs == toState;
            if (!lhsMatch && LHS_MATCH_BOOLEAN && typeof fromState === 'boolean') {
                lhsMatch = fromState ? TRUE_BOOLEAN_VALUES.has(lhs) : FALSE_BOOLEAN_VALUES.has(lhs);
            }
            if (!rhsMatch && RHS_MATCH_BOOLEAN && typeof toState === 'boolean') {
                rhsMatch = toState ? TRUE_BOOLEAN_VALUES.has(rhs) : FALSE_BOOLEAN_VALUES.has(rhs);
            }
            return lhsMatch && rhsMatch;
        });
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/animations/browser/src/dsl/animation_ast_builder.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const SELF_TOKEN = ':self';
    /** @type {?} */
    const SELF_TOKEN_REGEX = new RegExp(`\s*${SELF_TOKEN}\s*,?`, 'g');
    /*
     * [Validation]
     * The visitor code below will traverse the animation AST generated by the animation verb functions
     * (the output is a tree of objects) and attempt to perform a series of validations on the data. The
     * following corner-cases will be validated:
     *
     * 1. Overlap of animations
     * Given that a CSS property cannot be animated in more than one place at the same time, it's
     * important that this behavior is detected and validated. The way in which this occurs is that
     * each time a style property is examined, a string-map containing the property will be updated with
     * the start and end times for when the property is used within an animation step.
     *
     * If there are two or more parallel animations that are currently running (these are invoked by the
     * group()) on the same element then the validator will throw an error. Since the start/end timing
     * values are collected for each property then if the current animation step is animating the same
     * property and its timing values fall anywhere into the window of time that the property is
     * currently being animated within then this is what causes an error.
     *
     * 2. Timing values
     * The validator will validate to see if a timing value of `duration delay easing` or
     * `durationNumber` is valid or not.
     *
     * (note that upon validation the code below will replace the timing data with an object containing
     * {duration,delay,easing}.
     *
     * 3. Offset Validation
     * Each of the style() calls are allowed to have an offset value when placed inside of keyframes().
     * Offsets within keyframes() are considered valid when:
     *
     *   - No offsets are used at all
     *   - Each style() entry contains an offset value
     *   - Each offset is between 0 and 1
     *   - Each offset is greater to or equal than the previous one
     *
     * Otherwise an error will be thrown.
     */
    /**
     * @param {?} driver
     * @param {?} metadata
     * @param {?} errors
     * @return {?}
     */
    function buildAnimationAst(driver, metadata, errors) {
        return new AnimationAstBuilderVisitor(driver).build(metadata, errors);
    }
    /** @type {?} */
    const ROOT_SELECTOR = '';
    class AnimationAstBuilderVisitor {
        /**
         * @param {?} _driver
         */
        constructor(_driver) {
            this._driver = _driver;
        }
        /**
         * @param {?} metadata
         * @param {?} errors
         * @return {?}
         */
        build(metadata, errors) {
            /** @type {?} */
            const context = new AnimationAstBuilderContext(errors);
            this._resetContextStyleTimingState(context);
            return (/** @type {?} */ (visitDslNode(this, normalizeAnimationEntry(metadata), context)));
        }
        /**
         * @private
         * @param {?} context
         * @return {?}
         */
        _resetContextStyleTimingState(context) {
            context.currentQuerySelector = ROOT_SELECTOR;
            context.collectedStyles = {};
            context.collectedStyles[ROOT_SELECTOR] = {};
            context.currentTime = 0;
        }
        /**
         * @param {?} metadata
         * @param {?} context
         * @return {?}
         */
        visitTrigger(metadata, context) {
            /** @type {?} */
            let queryCount = context.queryCount = 0;
            /** @type {?} */
            let depCount = context.depCount = 0;
            /** @type {?} */
            const states = [];
            /** @type {?} */
            const transitions = [];
            if (metadata.name.charAt(0) == '@') {
                context.errors.push('animation triggers cannot be prefixed with an `@` sign (e.g. trigger(\'@foo\', [...]))');
            }
            metadata.definitions.forEach((/**
             * @param {?} def
             * @return {?}
             */
            def => {
                this._resetContextStyleTimingState(context);
                if (def.type == 0 /* State */) {
                    /** @type {?} */
                    const stateDef = (/** @type {?} */ (def));
                    /** @type {?} */
                    const name = stateDef.name;
                    name.toString().split(/\s*,\s*/).forEach((/**
                     * @param {?} n
                     * @return {?}
                     */
                    n => {
                        stateDef.name = n;
                        states.push(this.visitState(stateDef, context));
                    }));
                    stateDef.name = name;
                }
                else if (def.type == 1 /* Transition */) {
                    /** @type {?} */
                    const transition = this.visitTransition((/** @type {?} */ (def)), context);
                    queryCount += transition.queryCount;
                    depCount += transition.depCount;
                    transitions.push(transition);
                }
                else {
                    context.errors.push('only state() and transition() definitions can sit inside of a trigger()');
                }
            }));
            return {
                type: 7 /* Trigger */,
                name: metadata.name,
                states,
                transitions,
                queryCount,
                depCount,
                options: null
            };
        }
        /**
         * @param {?} metadata
         * @param {?} context
         * @return {?}
         */
        visitState(metadata, context) {
            /** @type {?} */
            const styleAst = this.visitStyle(metadata.styles, context);
            /** @type {?} */
            const astParams = (metadata.options && metadata.options.params) || null;
            if (styleAst.containsDynamicStyles) {
                /** @type {?} */
                const missingSubs = new Set();
                /** @type {?} */
                const params = astParams || {};
                styleAst.styles.forEach((/**
                 * @param {?} value
                 * @return {?}
                 */
                value => {
                    if (isObject$1(value)) {
                        /** @type {?} */
                        const stylesObj = (/** @type {?} */ (value));
                        Object.keys(stylesObj).forEach((/**
                         * @param {?} prop
                         * @return {?}
                         */
                        prop => {
                            extractStyleParams(stylesObj[prop]).forEach((/**
                             * @param {?} sub
                             * @return {?}
                             */
                            sub => {
                                if (!params.hasOwnProperty(sub)) {
                                    missingSubs.add(sub);
                                }
                            }));
                        }));
                    }
                }));
                if (missingSubs.size) {
                    /** @type {?} */
                    const missingSubsArr = iteratorToArray(missingSubs.values());
                    context.errors.push(`state("${metadata
                    .name}", ...) must define default values for all the following style substitutions: ${missingSubsArr.join(', ')}`);
                }
            }
            return {
                type: 0 /* State */,
                name: metadata.name,
                style: styleAst,
                options: astParams ? { params: astParams } : null
            };
        }
        /**
         * @param {?} metadata
         * @param {?} context
         * @return {?}
         */
        visitTransition(metadata, context) {
            context.queryCount = 0;
            context.depCount = 0;
            /** @type {?} */
            const animation = visitDslNode(this, normalizeAnimationEntry(metadata.animation), context);
            /** @type {?} */
            const matchers = parseTransitionExpr(metadata.expr, context.errors);
            return {
                type: 1 /* Transition */,
                matchers,
                animation,
                queryCount: context.queryCount,
                depCount: context.depCount,
                options: normalizeAnimationOptions(metadata.options)
            };
        }
        /**
         * @param {?} metadata
         * @param {?} context
         * @return {?}
         */
        visitSequence(metadata, context) {
            return {
                type: 2 /* Sequence */,
                steps: metadata.steps.map((/**
                 * @param {?} s
                 * @return {?}
                 */
                s => visitDslNode(this, s, context))),
                options: normalizeAnimationOptions(metadata.options)
            };
        }
        /**
         * @param {?} metadata
         * @param {?} context
         * @return {?}
         */
        visitGroup(metadata, context) {
            /** @type {?} */
            const currentTime = context.currentTime;
            /** @type {?} */
            let furthestTime = 0;
            /** @type {?} */
            const steps = metadata.steps.map((/**
             * @param {?} step
             * @return {?}
             */
            step => {
                context.currentTime = currentTime;
                /** @type {?} */
                const innerAst = visitDslNode(this, step, context);
                furthestTime = Math.max(furthestTime, context.currentTime);
                return innerAst;
            }));
            context.currentTime = furthestTime;
            return {
                type: 3 /* Group */,
                steps,
                options: normalizeAnimationOptions(metadata.options)
            };
        }
        /**
         * @param {?} metadata
         * @param {?} context
         * @return {?}
         */
        visitAnimate(metadata, context) {
            /** @type {?} */
            const timingAst = constructTimingAst(metadata.timings, context.errors);
            context.currentAnimateTimings = timingAst;
            /** @type {?} */
            let styleAst;
            /** @type {?} */
            let styleMetadata = metadata.styles ? metadata.styles : style({});
            if (styleMetadata.type == 5 /* Keyframes */) {
                styleAst = this.visitKeyframes((/** @type {?} */ (styleMetadata)), context);
            }
            else {
                /** @type {?} */
                let styleMetadata = (/** @type {?} */ (metadata.styles));
                /** @type {?} */
                let isEmpty = false;
                if (!styleMetadata) {
                    isEmpty = true;
                    /** @type {?} */
                    const newStyleData = {};
                    if (timingAst.easing) {
                        newStyleData['easing'] = timingAst.easing;
                    }
                    styleMetadata = style(newStyleData);
                }
                context.currentTime += timingAst.duration + timingAst.delay;
                /** @type {?} */
                const _styleAst = this.visitStyle(styleMetadata, context);
                _styleAst.isEmptyStep = isEmpty;
                styleAst = _styleAst;
            }
            context.currentAnimateTimings = null;
            return {
                type: 4 /* Animate */,
                timings: timingAst,
                style: styleAst,
                options: null
            };
        }
        /**
         * @param {?} metadata
         * @param {?} context
         * @return {?}
         */
        visitStyle(metadata, context) {
            /** @type {?} */
            const ast = this._makeStyleAst(metadata, context);
            this._validateStyleAst(ast, context);
            return ast;
        }
        /**
         * @private
         * @param {?} metadata
         * @param {?} context
         * @return {?}
         */
        _makeStyleAst(metadata, context) {
            /** @type {?} */
            const styles = [];
            if (Array.isArray(metadata.styles)) {
                ((/** @type {?} */ (metadata.styles))).forEach((/**
                 * @param {?} styleTuple
                 * @return {?}
                 */
                styleTuple => {
                    if (typeof styleTuple == 'string') {
                        if (styleTuple == AUTO_STYLE) {
                            styles.push(styleTuple);
                        }
                        else {
                            context.errors.push(`The provided style string value ${styleTuple} is not allowed.`);
                        }
                    }
                    else {
                        styles.push(styleTuple);
                    }
                }));
            }
            else {
                styles.push(metadata.styles);
            }
            /** @type {?} */
            let containsDynamicStyles = false;
            /** @type {?} */
            let collectedEasing = null;
            styles.forEach((/**
             * @param {?} styleData
             * @return {?}
             */
            styleData => {
                if (isObject$1(styleData)) {
                    /** @type {?} */
                    const styleMap = (/** @type {?} */ (styleData));
                    /** @type {?} */
                    const easing = styleMap['easing'];
                    if (easing) {
                        collectedEasing = (/** @type {?} */ (easing));
                        delete styleMap['easing'];
                    }
                    if (!containsDynamicStyles) {
                        for (let prop in styleMap) {
                            /** @type {?} */
                            const value = styleMap[prop];
                            if (value.toString().indexOf(SUBSTITUTION_EXPR_START) >= 0) {
                                containsDynamicStyles = true;
                                break;
                            }
                        }
                    }
                }
            }));
            return {
                type: 6 /* Style */,
                styles,
                easing: collectedEasing,
                offset: metadata.offset,
                containsDynamicStyles,
                options: null
            };
        }
        /**
         * @private
         * @param {?} ast
         * @param {?} context
         * @return {?}
         */
        _validateStyleAst(ast, context) {
            /** @type {?} */
            const timings = context.currentAnimateTimings;
            /** @type {?} */
            let endTime = context.currentTime;
            /** @type {?} */
            let startTime = context.currentTime;
            if (timings && startTime > 0) {
                startTime -= timings.duration + timings.delay;
            }
            ast.styles.forEach((/**
             * @param {?} tuple
             * @return {?}
             */
            tuple => {
                if (typeof tuple == 'string')
                    return;
                Object.keys(tuple).forEach((/**
                 * @param {?} prop
                 * @return {?}
                 */
                prop => {
                    if (!this._driver.validateStyleProperty(prop)) {
                        context.errors.push(`The provided animation property "${prop}" is not a supported CSS property for animations`);
                        return;
                    }
                    /** @type {?} */
                    const collectedStyles = context.collectedStyles[(/** @type {?} */ (context.currentQuerySelector))];
                    /** @type {?} */
                    const collectedEntry = collectedStyles[prop];
                    /** @type {?} */
                    let updateCollectedStyle = true;
                    if (collectedEntry) {
                        if (startTime != endTime && startTime >= collectedEntry.startTime &&
                            endTime <= collectedEntry.endTime) {
                            context.errors.push(`The CSS property "${prop}" that exists between the times of "${collectedEntry.startTime}ms" and "${collectedEntry
                            .endTime}ms" is also being animated in a parallel animation between the times of "${startTime}ms" and "${endTime}ms"`);
                            updateCollectedStyle = false;
                        }
                        // we always choose the smaller start time value since we
                        // want to have a record of the entire animation window where
                        // the style property is being animated in between
                        startTime = collectedEntry.startTime;
                    }
                    if (updateCollectedStyle) {
                        collectedStyles[prop] = { startTime, endTime };
                    }
                    if (context.options) {
                        validateStyleParams(tuple[prop], context.options, context.errors);
                    }
                }));
            }));
        }
        /**
         * @param {?} metadata
         * @param {?} context
         * @return {?}
         */
        visitKeyframes(metadata, context) {
            /** @type {?} */
            const ast = { type: 5 /* Keyframes */, styles: [], options: null };
            if (!context.currentAnimateTimings) {
                context.errors.push(`keyframes() must be placed inside of a call to animate()`);
                return ast;
            }
            /** @type {?} */
            const MAX_KEYFRAME_OFFSET = 1;
            /** @type {?} */
            let totalKeyframesWithOffsets = 0;
            /** @type {?} */
            const offsets = [];
            /** @type {?} */
            let offsetsOutOfOrder = false;
            /** @type {?} */
            let keyframesOutOfRange = false;
            /** @type {?} */
            let previousOffset = 0;
            /** @type {?} */
            const keyframes = metadata.steps.map((/**
             * @param {?} styles
             * @return {?}
             */
            styles => {
                /** @type {?} */
                const style = this._makeStyleAst(styles, context);
                /** @type {?} */
                let offsetVal = style.offset != null ? style.offset : consumeOffset(style.styles);
                /** @type {?} */
                let offset = 0;
                if (offsetVal != null) {
                    totalKeyframesWithOffsets++;
                    offset = style.offset = offsetVal;
                }
                keyframesOutOfRange = keyframesOutOfRange || offset < 0 || offset > 1;
                offsetsOutOfOrder = offsetsOutOfOrder || offset < previousOffset;
                previousOffset = offset;
                offsets.push(offset);
                return style;
            }));
            if (keyframesOutOfRange) {
                context.errors.push(`Please ensure that all keyframe offsets are between 0 and 1`);
            }
            if (offsetsOutOfOrder) {
                context.errors.push(`Please ensure that all keyframe offsets are in order`);
            }
            /** @type {?} */
            const length = metadata.steps.length;
            /** @type {?} */
            let generatedOffset = 0;
            if (totalKeyframesWithOffsets > 0 && totalKeyframesWithOffsets < length) {
                context.errors.push(`Not all style() steps within the declared keyframes() contain offsets`);
            }
            else if (totalKeyframesWithOffsets == 0) {
                generatedOffset = MAX_KEYFRAME_OFFSET / (length - 1);
            }
            /** @type {?} */
            const limit = length - 1;
            /** @type {?} */
            const currentTime = context.currentTime;
            /** @type {?} */
            const currentAnimateTimings = (/** @type {?} */ (context.currentAnimateTimings));
            /** @type {?} */
            const animateDuration = currentAnimateTimings.duration;
            keyframes.forEach((/**
             * @param {?} kf
             * @param {?} i
             * @return {?}
             */
            (kf, i) => {
                /** @type {?} */
                const offset = generatedOffset > 0 ? (i == limit ? 1 : (generatedOffset * i)) : offsets[i];
                /** @type {?} */
                const durationUpToThisFrame = offset * animateDuration;
                context.currentTime = currentTime + currentAnimateTimings.delay + durationUpToThisFrame;
                currentAnimateTimings.duration = durationUpToThisFrame;
                this._validateStyleAst(kf, context);
                kf.offset = offset;
                ast.styles.push(kf);
            }));
            return ast;
        }
        /**
         * @param {?} metadata
         * @param {?} context
         * @return {?}
         */
        visitReference(metadata, context) {
            return {
                type: 8 /* Reference */,
                animation: visitDslNode(this, normalizeAnimationEntry(metadata.animation), context),
                options: normalizeAnimationOptions(metadata.options)
            };
        }
        /**
         * @param {?} metadata
         * @param {?} context
         * @return {?}
         */
        visitAnimateChild(metadata, context) {
            context.depCount++;
            return {
                type: 9 /* AnimateChild */,
                options: normalizeAnimationOptions(metadata.options)
            };
        }
        /**
         * @param {?} metadata
         * @param {?} context
         * @return {?}
         */
        visitAnimateRef(metadata, context) {
            return {
                type: 10 /* AnimateRef */,
                animation: this.visitReference(metadata.animation, context),
                options: normalizeAnimationOptions(metadata.options)
            };
        }
        /**
         * @param {?} metadata
         * @param {?} context
         * @return {?}
         */
        visitQuery(metadata, context) {
            /** @type {?} */
            const parentSelector = (/** @type {?} */ (context.currentQuerySelector));
            /** @type {?} */
            const options = (/** @type {?} */ ((metadata.options || {})));
            context.queryCount++;
            context.currentQuery = metadata;
            const [selector, includeSelf] = normalizeSelector(metadata.selector);
            context.currentQuerySelector =
                parentSelector.length ? (parentSelector + ' ' + selector) : selector;
            getOrSetAsInMap(context.collectedStyles, context.currentQuerySelector, {});
            /** @type {?} */
            const animation = visitDslNode(this, normalizeAnimationEntry(metadata.animation), context);
            context.currentQuery = null;
            context.currentQuerySelector = parentSelector;
            return {
                type: 11 /* Query */,
                selector,
                limit: options.limit || 0,
                optional: !!options.optional,
                includeSelf,
                animation,
                originalSelector: metadata.selector,
                options: normalizeAnimationOptions(metadata.options)
            };
        }
        /**
         * @param {?} metadata
         * @param {?} context
         * @return {?}
         */
        visitStagger(metadata, context) {
            if (!context.currentQuery) {
                context.errors.push(`stagger() can only be used inside of query()`);
            }
            /** @type {?} */
            const timings = metadata.timings === 'full' ?
                { duration: 0, delay: 0, easing: 'full' } :
                resolveTiming(metadata.timings, context.errors, true);
            return {
                type: 12 /* Stagger */,
                animation: visitDslNode(this, normalizeAnimationEntry(metadata.animation), context),
                timings,
                options: null
            };
        }
    }
    /**
     * @param {?} selector
     * @return {?}
     */
    function normalizeSelector(selector) {
        /** @type {?} */
        const hasAmpersand = selector.split(/\s*,\s*/).find((/**
         * @param {?} token
         * @return {?}
         */
        token => token == SELF_TOKEN)) ? true : false;
        if (hasAmpersand) {
            selector = selector.replace(SELF_TOKEN_REGEX, '');
        }
        // the :enter and :leave selectors are filled in at runtime during timeline building
        selector = selector.replace(/@\*/g, NG_TRIGGER_SELECTOR)
            .replace(/@\w+/g, (/**
         * @param {?} match
         * @return {?}
         */
        match => NG_TRIGGER_SELECTOR + '-' + match.substr(1)))
            .replace(/:animating/g, NG_ANIMATING_SELECTOR);
        return [selector, hasAmpersand];
    }
    /**
     * @param {?} obj
     * @return {?}
     */
    function normalizeParams(obj) {
        return obj ? copyObj(obj) : null;
    }
    class AnimationAstBuilderContext {
        /**
         * @param {?} errors
         */
        constructor(errors) {
            this.errors = errors;
            this.queryCount = 0;
            this.depCount = 0;
            this.currentTransition = null;
            this.currentQuery = null;
            this.currentQuerySelector = null;
            this.currentAnimateTimings = null;
            this.currentTime = 0;
            this.collectedStyles = {};
            this.options = null;
        }
    }
    /**
     * @param {?} styles
     * @return {?}
     */
    function consumeOffset(styles) {
        if (typeof styles == 'string')
            return null;
        /** @type {?} */
        let offset = null;
        if (Array.isArray(styles)) {
            styles.forEach((/**
             * @param {?} styleTuple
             * @return {?}
             */
            styleTuple => {
                if (isObject$1(styleTuple) && styleTuple.hasOwnProperty('offset')) {
                    /** @type {?} */
                    const obj = (/** @type {?} */ (styleTuple));
                    offset = parseFloat((/** @type {?} */ (obj['offset'])));
                    delete obj['offset'];
                }
            }));
        }
        else if (isObject$1(styles) && styles.hasOwnProperty('offset')) {
            /** @type {?} */
            const obj = styles;
            offset = parseFloat((/** @type {?} */ (obj['offset'])));
            delete obj['offset'];
        }
        return offset;
    }
    /**
     * @param {?} value
     * @return {?}
     */
    function isObject$1(value) {
        return !Array.isArray(value) && typeof value == 'object';
    }
    /**
     * @param {?} value
     * @param {?} errors
     * @return {?}
     */
    function constructTimingAst(value, errors) {
        /** @type {?} */
        let timings = null;
        if (value.hasOwnProperty('duration')) {
            timings = (/** @type {?} */ (value));
        }
        else if (typeof value == 'number') {
            /** @type {?} */
            const duration = resolveTiming(value, errors).duration;
            return makeTimingAst(duration, 0, '');
        }
        /** @type {?} */
        const strValue = (/** @type {?} */ (value));
        /** @type {?} */
        const isDynamic = strValue.split(/\s+/).some((/**
         * @param {?} v
         * @return {?}
         */
        v => v.charAt(0) == '{' && v.charAt(1) == '{'));
        if (isDynamic) {
            /** @type {?} */
            const ast = (/** @type {?} */ (makeTimingAst(0, 0, '')));
            ast.dynamic = true;
            ast.strValue = strValue;
            return (/** @type {?} */ (ast));
        }
        timings = timings || resolveTiming(strValue, errors);
        return makeTimingAst(timings.duration, timings.delay, timings.easing);
    }
    /**
     * @param {?} options
     * @return {?}
     */
    function normalizeAnimationOptions(options) {
        if (options) {
            options = copyObj(options);
            if (options['params']) {
                options['params'] = (/** @type {?} */ (normalizeParams(options['params'])));
            }
        }
        else {
            options = {};
        }
        return options;
    }
    /**
     * @param {?} duration
     * @param {?} delay
     * @param {?} easing
     * @return {?}
     */
    function makeTimingAst(duration, delay, easing) {
        return { duration, delay, easing };
    }
    /**
     * @param {?} element
     * @param {?} keyframes
     * @param {?} preStyleProps
     * @param {?} postStyleProps
     * @param {?} duration
     * @param {?} delay
     * @param {?=} easing
     * @param {?=} subTimeline
     * @return {?}
     */
    function createTimelineInstruction(element, keyframes, preStyleProps, postStyleProps, duration, delay, easing = null, subTimeline = false) {
        return {
            type: 1 /* TimelineAnimation */,
            element,
            keyframes,
            preStyleProps,
            postStyleProps,
            duration,
            delay,
            totalTime: duration + delay,
            easing,
            subTimeline
        };
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/animations/browser/src/dsl/element_instruction_map.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class ElementInstructionMap {
        constructor() {
            this._map = new Map();
        }
        /**
         * @param {?} element
         * @return {?}
         */
        consume(element) {
            /** @type {?} */
            let instructions = this._map.get(element);
            if (instructions) {
                this._map.delete(element);
            }
            else {
                instructions = [];
            }
            return instructions;
        }
        /**
         * @param {?} element
         * @param {?} instructions
         * @return {?}
         */
        append(element, instructions) {
            /** @type {?} */
            let existingInstructions = this._map.get(element);
            if (!existingInstructions) {
                this._map.set(element, existingInstructions = []);
            }
            existingInstructions.push(...instructions);
        }
        /**
         * @param {?} element
         * @return {?}
         */
        has(element) {
            return this._map.has(element);
        }
        /**
         * @return {?}
         */
        clear() {
            this._map.clear();
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/animations/browser/src/dsl/animation_timeline_builder.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const ONE_FRAME_IN_MILLISECONDS = 1;
    /** @type {?} */
    const ENTER_TOKEN = ':enter';
    /** @type {?} */
    const ENTER_TOKEN_REGEX = new RegExp(ENTER_TOKEN, 'g');
    /** @type {?} */
    const LEAVE_TOKEN = ':leave';
    /** @type {?} */
    const LEAVE_TOKEN_REGEX = new RegExp(LEAVE_TOKEN, 'g');
    /*
     * The code within this file aims to generate web-animations-compatible keyframes from Angular's
     * animation DSL code.
     *
     * The code below will be converted from:
     *
     * ```
     * sequence([
     *   style({ opacity: 0 }),
     *   animate(1000, style({ opacity: 0 }))
     * ])
     * ```
     *
     * To:
     * ```
     * keyframes = [{ opacity: 0, offset: 0 }, { opacity: 1, offset: 1 }]
     * duration = 1000
     * delay = 0
     * easing = ''
     * ```
     *
     * For this operation to cover the combination of animation verbs (style, animate, group, etc...) a
     * combination of prototypical inheritance, AST traversal and merge-sort-like algorithms are used.
     *
     * [AST Traversal]
     * Each of the animation verbs, when executed, will return an string-map object representing what
     * type of action it is (style, animate, group, etc...) and the data associated with it. This means
     * that when functional composition mix of these functions is evaluated (like in the example above)
     * then it will end up producing a tree of objects representing the animation itself.
     *
     * When this animation object tree is processed by the visitor code below it will visit each of the
     * verb statements within the visitor. And during each visit it will build the context of the
     * animation keyframes by interacting with the `TimelineBuilder`.
     *
     * [TimelineBuilder]
     * This class is responsible for tracking the styles and building a series of keyframe objects for a
     * timeline between a start and end time. The builder starts off with an initial timeline and each
     * time the AST comes across a `group()`, `keyframes()` or a combination of the two wihtin a
     * `sequence()` then it will generate a sub timeline for each step as well as a new one after
     * they are complete.
     *
     * As the AST is traversed, the timing state on each of the timelines will be incremented. If a sub
     * timeline was created (based on one of the cases above) then the parent timeline will attempt to
     * merge the styles used within the sub timelines into itself (only with group() this will happen).
     * This happens with a merge operation (much like how the merge works in mergesort) and it will only
     * copy the most recently used styles from the sub timelines into the parent timeline. This ensures
     * that if the styles are used later on in another phase of the animation then they will be the most
     * up-to-date values.
     *
     * [How Missing Styles Are Updated]
     * Each timeline has a `backFill` property which is responsible for filling in new styles into
     * already processed keyframes if a new style shows up later within the animation sequence.
     *
     * ```
     * sequence([
     *   style({ width: 0 }),
     *   animate(1000, style({ width: 100 })),
     *   animate(1000, style({ width: 200 })),
     *   animate(1000, style({ width: 300 }))
     *   animate(1000, style({ width: 400, height: 400 })) // notice how `height` doesn't exist anywhere
     * else
     * ])
     * ```
     *
     * What is happening here is that the `height` value is added later in the sequence, but is missing
     * from all previous animation steps. Therefore when a keyframe is created it would also be missing
     * from all previous keyframes up until where it is first used. For the timeline keyframe generation
     * to properly fill in the style it will place the previous value (the value from the parent
     * timeline) or a default value of `*` into the backFill object. Given that each of the keyframe
     * styles are objects that prototypically inhert from the backFill object, this means that if a
     * value is added into the backFill then it will automatically propagate any missing values to all
     * keyframes. Therefore the missing `height` value will be properly filled into the already
     * processed keyframes.
     *
     * When a sub-timeline is created it will have its own backFill property. This is done so that
     * styles present within the sub-timeline do not accidentally seep into the previous/future timeline
     * keyframes
     *
     * (For prototypically-inherited contents to be detected a `for(i in obj)` loop must be used.)
     *
     * [Validation]
     * The code in this file is not responsible for validation. That functionality happens with within
     * the `AnimationValidatorVisitor` code.
     */
    /**
     * @param {?} driver
     * @param {?} rootElement
     * @param {?} ast
     * @param {?} enterClassName
     * @param {?} leaveClassName
     * @param {?=} startingStyles
     * @param {?=} finalStyles
     * @param {?=} options
     * @param {?=} subInstructions
     * @param {?=} errors
     * @return {?}
     */
    function buildAnimationTimelines(driver, rootElement, ast, enterClassName, leaveClassName, startingStyles = {}, finalStyles = {}, options, subInstructions, errors = []) {
        return new AnimationTimelineBuilderVisitor().buildKeyframes(driver, rootElement, ast, enterClassName, leaveClassName, startingStyles, finalStyles, options, subInstructions, errors);
    }
    class AnimationTimelineBuilderVisitor {
        /**
         * @param {?} driver
         * @param {?} rootElement
         * @param {?} ast
         * @param {?} enterClassName
         * @param {?} leaveClassName
         * @param {?} startingStyles
         * @param {?} finalStyles
         * @param {?} options
         * @param {?=} subInstructions
         * @param {?=} errors
         * @return {?}
         */
        buildKeyframes(driver, rootElement, ast, enterClassName, leaveClassName, startingStyles, finalStyles, options, subInstructions, errors = []) {
            subInstructions = subInstructions || new ElementInstructionMap();
            /** @type {?} */
            const context = new AnimationTimelineContext(driver, rootElement, subInstructions, enterClassName, leaveClassName, errors, []);
            context.options = options;
            context.currentTimeline.setStyles([startingStyles], null, context.errors, options);
            visitDslNode(this, ast, context);
            // this checks to see if an actual animation happened
            /** @type {?} */
            const timelines = context.timelines.filter((/**
             * @param {?} timeline
             * @return {?}
             */
            timeline => timeline.containsAnimation()));
            if (timelines.length && Object.keys(finalStyles).length) {
                /** @type {?} */
                const tl = timelines[timelines.length - 1];
                if (!tl.allowOnlyTimelineStyles()) {
                    tl.setStyles([finalStyles], null, context.errors, options);
                }
            }
            return timelines.length ? timelines.map((/**
             * @param {?} timeline
             * @return {?}
             */
            timeline => timeline.buildKeyframes())) :
                [createTimelineInstruction(rootElement, [], [], [], 0, 0, '', false)];
        }
        /**
         * @param {?} ast
         * @param {?} context
         * @return {?}
         */
        visitTrigger(ast, context) {
            // these values are not visited in this AST
        }
        /**
         * @param {?} ast
         * @param {?} context
         * @return {?}
         */
        visitState(ast, context) {
            // these values are not visited in this AST
        }
        /**
         * @param {?} ast
         * @param {?} context
         * @return {?}
         */
        visitTransition(ast, context) {
            // these values are not visited in this AST
        }
        /**
         * @param {?} ast
         * @param {?} context
         * @return {?}
         */
        visitAnimateChild(ast, context) {
            /** @type {?} */
            const elementInstructions = context.subInstructions.consume(context.element);
            if (elementInstructions) {
                /** @type {?} */
                const innerContext = context.createSubContext(ast.options);
                /** @type {?} */
                const startTime = context.currentTimeline.currentTime;
                /** @type {?} */
                const endTime = this._visitSubInstructions(elementInstructions, innerContext, (/** @type {?} */ (innerContext.options)));
                if (startTime != endTime) {
                    // we do this on the upper context because we created a sub context for
                    // the sub child animations
                    context.transformIntoNewTimeline(endTime);
                }
            }
            context.previousNode = ast;
        }
        /**
         * @param {?} ast
         * @param {?} context
         * @return {?}
         */
        visitAnimateRef(ast, context) {
            /** @type {?} */
            const innerContext = context.createSubContext(ast.options);
            innerContext.transformIntoNewTimeline();
            this.visitReference(ast.animation, innerContext);
            context.transformIntoNewTimeline(innerContext.currentTimeline.currentTime);
            context.previousNode = ast;
        }
        /**
         * @private
         * @param {?} instructions
         * @param {?} context
         * @param {?} options
         * @return {?}
         */
        _visitSubInstructions(instructions, context, options) {
            /** @type {?} */
            const startTime = context.currentTimeline.currentTime;
            /** @type {?} */
            let furthestTime = startTime;
            // this is a special-case for when a user wants to skip a sub
            // animation from being fired entirely.
            /** @type {?} */
            const duration = options.duration != null ? resolveTimingValue(options.duration) : null;
            /** @type {?} */
            const delay = options.delay != null ? resolveTimingValue(options.delay) : null;
            if (duration !== 0) {
                instructions.forEach((/**
                 * @param {?} instruction
                 * @return {?}
                 */
                instruction => {
                    /** @type {?} */
                    const instructionTimings = context.appendInstructionToTimeline(instruction, duration, delay);
                    furthestTime =
                        Math.max(furthestTime, instructionTimings.duration + instructionTimings.delay);
                }));
            }
            return furthestTime;
        }
        /**
         * @param {?} ast
         * @param {?} context
         * @return {?}
         */
        visitReference(ast, context) {
            context.updateOptions(ast.options, true);
            visitDslNode(this, ast.animation, context);
            context.previousNode = ast;
        }
        /**
         * @param {?} ast
         * @param {?} context
         * @return {?}
         */
        visitSequence(ast, context) {
            /** @type {?} */
            const subContextCount = context.subContextCount;
            /** @type {?} */
            let ctx = context;
            /** @type {?} */
            const options = ast.options;
            if (options && (options.params || options.delay)) {
                ctx = context.createSubContext(options);
                ctx.transformIntoNewTimeline();
                if (options.delay != null) {
                    if (ctx.previousNode.type == 6 /* Style */) {
                        ctx.currentTimeline.snapshotCurrentStyles();
                        ctx.previousNode = DEFAULT_NOOP_PREVIOUS_NODE;
                    }
                    /** @type {?} */
                    const delay = resolveTimingValue(options.delay);
                    ctx.delayNextStep(delay);
                }
            }
            if (ast.steps.length) {
                ast.steps.forEach((/**
                 * @param {?} s
                 * @return {?}
                 */
                s => visitDslNode(this, s, ctx)));
                // this is here just incase the inner steps only contain or end with a style() call
                ctx.currentTimeline.applyStylesToKeyframe();
                // this means that some animation function within the sequence
                // ended up creating a sub timeline (which means the current
                // timeline cannot overlap with the contents of the sequence)
                if (ctx.subContextCount > subContextCount) {
                    ctx.transformIntoNewTimeline();
                }
            }
            context.previousNode = ast;
        }
        /**
         * @param {?} ast
         * @param {?} context
         * @return {?}
         */
        visitGroup(ast, context) {
            /** @type {?} */
            const innerTimelines = [];
            /** @type {?} */
            let furthestTime = context.currentTimeline.currentTime;
            /** @type {?} */
            const delay = ast.options && ast.options.delay ? resolveTimingValue(ast.options.delay) : 0;
            ast.steps.forEach((/**
             * @param {?} s
             * @return {?}
             */
            s => {
                /** @type {?} */
                const innerContext = context.createSubContext(ast.options);
                if (delay) {
                    innerContext.delayNextStep(delay);
                }
                visitDslNode(this, s, innerContext);
                furthestTime = Math.max(furthestTime, innerContext.currentTimeline.currentTime);
                innerTimelines.push(innerContext.currentTimeline);
            }));
            // this operation is run after the AST loop because otherwise
            // if the parent timeline's collected styles were updated then
            // it would pass in invalid data into the new-to-be forked items
            innerTimelines.forEach((/**
             * @param {?} timeline
             * @return {?}
             */
            timeline => context.currentTimeline.mergeTimelineCollectedStyles(timeline)));
            context.transformIntoNewTimeline(furthestTime);
            context.previousNode = ast;
        }
        /**
         * @private
         * @param {?} ast
         * @param {?} context
         * @return {?}
         */
        _visitTiming(ast, context) {
            if (((/** @type {?} */ (ast))).dynamic) {
                /** @type {?} */
                const strValue = ((/** @type {?} */ (ast))).strValue;
                /** @type {?} */
                const timingValue = context.params ? interpolateParams(strValue, context.params, context.errors) : strValue;
                return resolveTiming(timingValue, context.errors);
            }
            else {
                return { duration: ast.duration, delay: ast.delay, easing: ast.easing };
            }
        }
        /**
         * @param {?} ast
         * @param {?} context
         * @return {?}
         */
        visitAnimate(ast, context) {
            /** @type {?} */
            const timings = context.currentAnimateTimings = this._visitTiming(ast.timings, context);
            /** @type {?} */
            const timeline = context.currentTimeline;
            if (timings.delay) {
                context.incrementTime(timings.delay);
                timeline.snapshotCurrentStyles();
            }
            /** @type {?} */
            const style = ast.style;
            if (style.type == 5 /* Keyframes */) {
                this.visitKeyframes(style, context);
            }
            else {
                context.incrementTime(timings.duration);
                this.visitStyle((/** @type {?} */ (style)), context);
                timeline.applyStylesToKeyframe();
            }
            context.currentAnimateTimings = null;
            context.previousNode = ast;
        }
        /**
         * @param {?} ast
         * @param {?} context
         * @return {?}
         */
        visitStyle(ast, context) {
            /** @type {?} */
            const timeline = context.currentTimeline;
            /** @type {?} */
            const timings = (/** @type {?} */ (context.currentAnimateTimings));
            // this is a special case for when a style() call
            // directly follows  an animate() call (but not inside of an animate() call)
            if (!timings && timeline.getCurrentStyleProperties().length) {
                timeline.forwardFrame();
            }
            /** @type {?} */
            const easing = (timings && timings.easing) || ast.easing;
            if (ast.isEmptyStep) {
                timeline.applyEmptyStep(easing);
            }
            else {
                timeline.setStyles(ast.styles, easing, context.errors, context.options);
            }
            context.previousNode = ast;
        }
        /**
         * @param {?} ast
         * @param {?} context
         * @return {?}
         */
        visitKeyframes(ast, context) {
            /** @type {?} */
            const currentAnimateTimings = (/** @type {?} */ (context.currentAnimateTimings));
            /** @type {?} */
            const startTime = ((/** @type {?} */ (context.currentTimeline))).duration;
            /** @type {?} */
            const duration = currentAnimateTimings.duration;
            /** @type {?} */
            const innerContext = context.createSubContext();
            /** @type {?} */
            const innerTimeline = innerContext.currentTimeline;
            innerTimeline.easing = currentAnimateTimings.easing;
            ast.styles.forEach((/**
             * @param {?} step
             * @return {?}
             */
            step => {
                /** @type {?} */
                const offset = step.offset || 0;
                innerTimeline.forwardTime(offset * duration);
                innerTimeline.setStyles(step.styles, step.easing, context.errors, context.options);
                innerTimeline.applyStylesToKeyframe();
            }));
            // this will ensure that the parent timeline gets all the styles from
            // the child even if the new timeline below is not used
            context.currentTimeline.mergeTimelineCollectedStyles(innerTimeline);
            // we do this because the window between this timeline and the sub timeline
            // should ensure that the styles within are exactly the same as they were before
            context.transformIntoNewTimeline(startTime + duration);
            context.previousNode = ast;
        }
        /**
         * @param {?} ast
         * @param {?} context
         * @return {?}
         */
        visitQuery(ast, context) {
            // in the event that the first step before this is a style step we need
            // to ensure the styles are applied before the children are animated
            /** @type {?} */
            const startTime = context.currentTimeline.currentTime;
            /** @type {?} */
            const options = (/** @type {?} */ ((ast.options || {})));
            /** @type {?} */
            const delay = options.delay ? resolveTimingValue(options.delay) : 0;
            if (delay &&
                (context.previousNode.type === 6 /* Style */ ||
                    (startTime == 0 && context.currentTimeline.getCurrentStyleProperties().length))) {
                context.currentTimeline.snapshotCurrentStyles();
                context.previousNode = DEFAULT_NOOP_PREVIOUS_NODE;
            }
            /** @type {?} */
            let furthestTime = startTime;
            /** @type {?} */
            const elms = context.invokeQuery(ast.selector, ast.originalSelector, ast.limit, ast.includeSelf, options.optional ? true : false, context.errors);
            context.currentQueryTotal = elms.length;
            /** @type {?} */
            let sameElementTimeline = null;
            elms.forEach((/**
             * @param {?} element
             * @param {?} i
             * @return {?}
             */
            (element, i) => {
                context.currentQueryIndex = i;
                /** @type {?} */
                const innerContext = context.createSubContext(ast.options, element);
                if (delay) {
                    innerContext.delayNextStep(delay);
                }
                if (element === context.element) {
                    sameElementTimeline = innerContext.currentTimeline;
                }
                visitDslNode(this, ast.animation, innerContext);
                // this is here just incase the inner steps only contain or end
                // with a style() call (which is here to signal that this is a preparatory
                // call to style an element before it is animated again)
                innerContext.currentTimeline.applyStylesToKeyframe();
                /** @type {?} */
                const endTime = innerContext.currentTimeline.currentTime;
                furthestTime = Math.max(furthestTime, endTime);
            }));
            context.currentQueryIndex = 0;
            context.currentQueryTotal = 0;
            context.transformIntoNewTimeline(furthestTime);
            if (sameElementTimeline) {
                context.currentTimeline.mergeTimelineCollectedStyles(sameElementTimeline);
                context.currentTimeline.snapshotCurrentStyles();
            }
            context.previousNode = ast;
        }
        /**
         * @param {?} ast
         * @param {?} context
         * @return {?}
         */
        visitStagger(ast, context) {
            /** @type {?} */
            const parentContext = (/** @type {?} */ (context.parentContext));
            /** @type {?} */
            const tl = context.currentTimeline;
            /** @type {?} */
            const timings = ast.timings;
            /** @type {?} */
            const duration = Math.abs(timings.duration);
            /** @type {?} */
            const maxTime = duration * (context.currentQueryTotal - 1);
            /** @type {?} */
            let delay = duration * context.currentQueryIndex;
            /** @type {?} */
            let staggerTransformer = timings.duration < 0 ? 'reverse' : timings.easing;
            switch (staggerTransformer) {
                case 'reverse':
                    delay = maxTime - delay;
                    break;
                case 'full':
                    delay = parentContext.currentStaggerTime;
                    break;
            }
            /** @type {?} */
            const timeline = context.currentTimeline;
            if (delay) {
                timeline.delayNextStep(delay);
            }
            /** @type {?} */
            const startingTime = timeline.currentTime;
            visitDslNode(this, ast.animation, context);
            context.previousNode = ast;
            // time = duration + delay
            // the reason why this computation is so complex is because
            // the inner timeline may either have a delay value or a stretched
            // keyframe depending on if a subtimeline is not used or is used.
            parentContext.currentStaggerTime =
                (tl.currentTime - startingTime) + (tl.startTime - parentContext.currentTimeline.startTime);
        }
    }
    /** @type {?} */
    const DEFAULT_NOOP_PREVIOUS_NODE = (/** @type {?} */ ({}));
    class AnimationTimelineContext {
        /**
         * @param {?} _driver
         * @param {?} element
         * @param {?} subInstructions
         * @param {?} _enterClassName
         * @param {?} _leaveClassName
         * @param {?} errors
         * @param {?} timelines
         * @param {?=} initialTimeline
         */
        constructor(_driver, element, subInstructions, _enterClassName, _leaveClassName, errors, timelines, initialTimeline) {
            this._driver = _driver;
            this.element = element;
            this.subInstructions = subInstructions;
            this._enterClassName = _enterClassName;
            this._leaveClassName = _leaveClassName;
            this.errors = errors;
            this.timelines = timelines;
            this.parentContext = null;
            this.currentAnimateTimings = null;
            this.previousNode = DEFAULT_NOOP_PREVIOUS_NODE;
            this.subContextCount = 0;
            this.options = {};
            this.currentQueryIndex = 0;
            this.currentQueryTotal = 0;
            this.currentStaggerTime = 0;
            this.currentTimeline = initialTimeline || new TimelineBuilder(this._driver, element, 0);
            timelines.push(this.currentTimeline);
        }
        /**
         * @return {?}
         */
        get params() {
            return this.options.params;
        }
        /**
         * @param {?} options
         * @param {?=} skipIfExists
         * @return {?}
         */
        updateOptions(options, skipIfExists) {
            if (!options)
                return;
            /** @type {?} */
            const newOptions = (/** @type {?} */ (options));
            /** @type {?} */
            let optionsToUpdate = this.options;
            // NOTE: this will get patched up when other animation methods support duration overrides
            if (newOptions.duration != null) {
                ((/** @type {?} */ (optionsToUpdate))).duration = resolveTimingValue(newOptions.duration);
            }
            if (newOptions.delay != null) {
                optionsToUpdate.delay = resolveTimingValue(newOptions.delay);
            }
            /** @type {?} */
            const newParams = newOptions.params;
            if (newParams) {
                /** @type {?} */
                let paramsToUpdate = (/** @type {?} */ (optionsToUpdate.params));
                if (!paramsToUpdate) {
                    paramsToUpdate = this.options.params = {};
                }
                Object.keys(newParams).forEach((/**
                 * @param {?} name
                 * @return {?}
                 */
                name => {
                    if (!skipIfExists || !paramsToUpdate.hasOwnProperty(name)) {
                        paramsToUpdate[name] = interpolateParams(newParams[name], paramsToUpdate, this.errors);
                    }
                }));
            }
        }
        /**
         * @private
         * @return {?}
         */
        _copyOptions() {
            /** @type {?} */
            const options = {};
            if (this.options) {
                /** @type {?} */
                const oldParams = this.options.params;
                if (oldParams) {
                    /** @type {?} */
                    const params = options['params'] = {};
                    Object.keys(oldParams).forEach((/**
                     * @param {?} name
                     * @return {?}
                     */
                    name => {
                        params[name] = oldParams[name];
                    }));
                }
            }
            return options;
        }
        /**
         * @param {?=} options
         * @param {?=} element
         * @param {?=} newTime
         * @return {?}
         */
        createSubContext(options = null, element, newTime) {
            /** @type {?} */
            const target = element || this.element;
            /** @type {?} */
            const context = new AnimationTimelineContext(this._driver, target, this.subInstructions, this._enterClassName, this._leaveClassName, this.errors, this.timelines, this.currentTimeline.fork(target, newTime || 0));
            context.previousNode = this.previousNode;
            context.currentAnimateTimings = this.currentAnimateTimings;
            context.options = this._copyOptions();
            context.updateOptions(options);
            context.currentQueryIndex = this.currentQueryIndex;
            context.currentQueryTotal = this.currentQueryTotal;
            context.parentContext = this;
            this.subContextCount++;
            return context;
        }
        /**
         * @param {?=} newTime
         * @return {?}
         */
        transformIntoNewTimeline(newTime) {
            this.previousNode = DEFAULT_NOOP_PREVIOUS_NODE;
            this.currentTimeline = this.currentTimeline.fork(this.element, newTime);
            this.timelines.push(this.currentTimeline);
            return this.currentTimeline;
        }
        /**
         * @param {?} instruction
         * @param {?} duration
         * @param {?} delay
         * @return {?}
         */
        appendInstructionToTimeline(instruction, duration, delay) {
            /** @type {?} */
            const updatedTimings = {
                duration: duration != null ? duration : instruction.duration,
                delay: this.currentTimeline.currentTime + (delay != null ? delay : 0) + instruction.delay,
                easing: ''
            };
            /** @type {?} */
            const builder = new SubTimelineBuilder(this._driver, instruction.element, instruction.keyframes, instruction.preStyleProps, instruction.postStyleProps, updatedTimings, instruction.stretchStartingKeyframe);
            this.timelines.push(builder);
            return updatedTimings;
        }
        /**
         * @param {?} time
         * @return {?}
         */
        incrementTime(time) {
            this.currentTimeline.forwardTime(this.currentTimeline.duration + time);
        }
        /**
         * @param {?} delay
         * @return {?}
         */
        delayNextStep(delay) {
            // negative delays are not yet supported
            if (delay > 0) {
                this.currentTimeline.delayNextStep(delay);
            }
        }
        /**
         * @param {?} selector
         * @param {?} originalSelector
         * @param {?} limit
         * @param {?} includeSelf
         * @param {?} optional
         * @param {?} errors
         * @return {?}
         */
        invokeQuery(selector, originalSelector, limit, includeSelf, optional, errors) {
            /** @type {?} */
            let results = [];
            if (includeSelf) {
                results.push(this.element);
            }
            if (selector.length > 0) { // if :self is only used then the selector is empty
                selector = selector.replace(ENTER_TOKEN_REGEX, '.' + this._enterClassName);
                selector = selector.replace(LEAVE_TOKEN_REGEX, '.' + this._leaveClassName);
                /** @type {?} */
                const multi = limit != 1;
                /** @type {?} */
                let elements = this._driver.query(this.element, selector, multi);
                if (limit !== 0) {
                    elements = limit < 0 ? elements.slice(elements.length + limit, elements.length) :
                        elements.slice(0, limit);
                }
                results.push(...elements);
            }
            if (!optional && results.length == 0) {
                errors.push(`\`query("${originalSelector}")\` returned zero elements. (Use \`query("${originalSelector}", { optional: true })\` if you wish to allow this.)`);
            }
            return results;
        }
    }
    class TimelineBuilder {
        /**
         * @param {?} _driver
         * @param {?} element
         * @param {?} startTime
         * @param {?=} _elementTimelineStylesLookup
         */
        constructor(_driver, element, startTime, _elementTimelineStylesLookup) {
            this._driver = _driver;
            this.element = element;
            this.startTime = startTime;
            this._elementTimelineStylesLookup = _elementTimelineStylesLookup;
            this.duration = 0;
            this._previousKeyframe = {};
            this._currentKeyframe = {};
            this._keyframes = new Map();
            this._styleSummary = {};
            this._pendingStyles = {};
            this._backFill = {};
            this._currentEmptyStepKeyframe = null;
            if (!this._elementTimelineStylesLookup) {
                this._elementTimelineStylesLookup = new Map();
            }
            this._localTimelineStyles = Object.create(this._backFill, {});
            this._globalTimelineStyles = (/** @type {?} */ (this._elementTimelineStylesLookup.get(element)));
            if (!this._globalTimelineStyles) {
                this._globalTimelineStyles = this._localTimelineStyles;
                this._elementTimelineStylesLookup.set(element, this._localTimelineStyles);
            }
            this._loadKeyframe();
        }
        /**
         * @return {?}
         */
        containsAnimation() {
            switch (this._keyframes.size) {
                case 0:
                    return false;
                case 1:
                    return this.getCurrentStyleProperties().length > 0;
                default:
                    return true;
            }
        }
        /**
         * @return {?}
         */
        getCurrentStyleProperties() {
            return Object.keys(this._currentKeyframe);
        }
        /**
         * @return {?}
         */
        get currentTime() {
            return this.startTime + this.duration;
        }
        /**
         * @param {?} delay
         * @return {?}
         */
        delayNextStep(delay) {
            // in the event that a style() step is placed right before a stagger()
            // and that style() step is the very first style() value in the animation
            // then we need to make a copy of the keyframe [0, copy, 1] so that the delay
            // properly applies the style() values to work with the stagger...
            /** @type {?} */
            const hasPreStyleStep = this._keyframes.size == 1 && Object.keys(this._pendingStyles).length;
            if (this.duration || hasPreStyleStep) {
                this.forwardTime(this.currentTime + delay);
                if (hasPreStyleStep) {
                    this.snapshotCurrentStyles();
                }
            }
            else {
                this.startTime += delay;
            }
        }
        /**
         * @param {?} element
         * @param {?=} currentTime
         * @return {?}
         */
        fork(element, currentTime) {
            this.applyStylesToKeyframe();
            return new TimelineBuilder(this._driver, element, currentTime || this.currentTime, this._elementTimelineStylesLookup);
        }
        /**
         * @private
         * @return {?}
         */
        _loadKeyframe() {
            if (this._currentKeyframe) {
                this._previousKeyframe = this._currentKeyframe;
            }
            this._currentKeyframe = (/** @type {?} */ (this._keyframes.get(this.duration)));
            if (!this._currentKeyframe) {
                this._currentKeyframe = Object.create(this._backFill, {});
                this._keyframes.set(this.duration, this._currentKeyframe);
            }
        }
        /**
         * @return {?}
         */
        forwardFrame() {
            this.duration += ONE_FRAME_IN_MILLISECONDS;
            this._loadKeyframe();
        }
        /**
         * @param {?} time
         * @return {?}
         */
        forwardTime(time) {
            this.applyStylesToKeyframe();
            this.duration = time;
            this._loadKeyframe();
        }
        /**
         * @private
         * @param {?} prop
         * @param {?} value
         * @return {?}
         */
        _updateStyle(prop, value) {
            this._localTimelineStyles[prop] = value;
            this._globalTimelineStyles[prop] = value;
            this._styleSummary[prop] = { time: this.currentTime, value };
        }
        /**
         * @return {?}
         */
        allowOnlyTimelineStyles() {
            return this._currentEmptyStepKeyframe !== this._currentKeyframe;
        }
        /**
         * @param {?} easing
         * @return {?}
         */
        applyEmptyStep(easing) {
            if (easing) {
                this._previousKeyframe['easing'] = easing;
            }
            // special case for animate(duration):
            // all missing styles are filled with a `*` value then
            // if any destination styles are filled in later on the same
            // keyframe then they will override the overridden styles
            // We use `_globalTimelineStyles` here because there may be
            // styles in previous keyframes that are not present in this timeline
            Object.keys(this._globalTimelineStyles).forEach((/**
             * @param {?} prop
             * @return {?}
             */
            prop => {
                this._backFill[prop] = this._globalTimelineStyles[prop] || AUTO_STYLE;
                this._currentKeyframe[prop] = AUTO_STYLE;
            }));
            this._currentEmptyStepKeyframe = this._currentKeyframe;
        }
        /**
         * @param {?} input
         * @param {?} easing
         * @param {?} errors
         * @param {?=} options
         * @return {?}
         */
        setStyles(input, easing, errors, options) {
            if (easing) {
                this._previousKeyframe['easing'] = easing;
            }
            /** @type {?} */
            const params = (options && options.params) || {};
            /** @type {?} */
            const styles = flattenStyles$1(input, this._globalTimelineStyles);
            Object.keys(styles).forEach((/**
             * @param {?} prop
             * @return {?}
             */
            prop => {
                /** @type {?} */
                const val = interpolateParams(styles[prop], params, errors);
                this._pendingStyles[prop] = val;
                if (!this._localTimelineStyles.hasOwnProperty(prop)) {
                    this._backFill[prop] = this._globalTimelineStyles.hasOwnProperty(prop) ?
                        this._globalTimelineStyles[prop] :
                        AUTO_STYLE;
                }
                this._updateStyle(prop, val);
            }));
        }
        /**
         * @return {?}
         */
        applyStylesToKeyframe() {
            /** @type {?} */
            const styles = this._pendingStyles;
            /** @type {?} */
            const props = Object.keys(styles);
            if (props.length == 0)
                return;
            this._pendingStyles = {};
            props.forEach((/**
             * @param {?} prop
             * @return {?}
             */
            prop => {
                /** @type {?} */
                const val = styles[prop];
                this._currentKeyframe[prop] = val;
            }));
            Object.keys(this._localTimelineStyles).forEach((/**
             * @param {?} prop
             * @return {?}
             */
            prop => {
                if (!this._currentKeyframe.hasOwnProperty(prop)) {
                    this._currentKeyframe[prop] = this._localTimelineStyles[prop];
                }
            }));
        }
        /**
         * @return {?}
         */
        snapshotCurrentStyles() {
            Object.keys(this._localTimelineStyles).forEach((/**
             * @param {?} prop
             * @return {?}
             */
            prop => {
                /** @type {?} */
                const val = this._localTimelineStyles[prop];
                this._pendingStyles[prop] = val;
                this._updateStyle(prop, val);
            }));
        }
        /**
         * @return {?}
         */
        getFinalKeyframe() {
            return this._keyframes.get(this.duration);
        }
        /**
         * @return {?}
         */
        get properties() {
            /** @type {?} */
            const properties = [];
            for (let prop in this._currentKeyframe) {
                properties.push(prop);
            }
            return properties;
        }
        /**
         * @param {?} timeline
         * @return {?}
         */
        mergeTimelineCollectedStyles(timeline) {
            Object.keys(timeline._styleSummary).forEach((/**
             * @param {?} prop
             * @return {?}
             */
            prop => {
                /** @type {?} */
                const details0 = this._styleSummary[prop];
                /** @type {?} */
                const details1 = timeline._styleSummary[prop];
                if (!details0 || details1.time > details0.time) {
                    this._updateStyle(prop, details1.value);
                }
            }));
        }
        /**
         * @return {?}
         */
        buildKeyframes() {
            this.applyStylesToKeyframe();
            /** @type {?} */
            const preStyleProps = new Set();
            /** @type {?} */
            const postStyleProps = new Set();
            /** @type {?} */
            const isEmpty = this._keyframes.size === 1 && this.duration === 0;
            /** @type {?} */
            let finalKeyframes = [];
            this._keyframes.forEach((/**
             * @param {?} keyframe
             * @param {?} time
             * @return {?}
             */
            (keyframe, time) => {
                /** @type {?} */
                const finalKeyframe = copyStyles(keyframe, true);
                Object.keys(finalKeyframe).forEach((/**
                 * @param {?} prop
                 * @return {?}
                 */
                prop => {
                    /** @type {?} */
                    const value = finalKeyframe[prop];
                    if (value == ɵPRE_STYLE) {
                        preStyleProps.add(prop);
                    }
                    else if (value == AUTO_STYLE) {
                        postStyleProps.add(prop);
                    }
                }));
                if (!isEmpty) {
                    finalKeyframe['offset'] = time / this.duration;
                }
                finalKeyframes.push(finalKeyframe);
            }));
            /** @type {?} */
            const preProps = preStyleProps.size ? iteratorToArray(preStyleProps.values()) : [];
            /** @type {?} */
            const postProps = postStyleProps.size ? iteratorToArray(postStyleProps.values()) : [];
            // special case for a 0-second animation (which is designed just to place styles onscreen)
            if (isEmpty) {
                /** @type {?} */
                const kf0 = finalKeyframes[0];
                /** @type {?} */
                const kf1 = copyObj(kf0);
                kf0['offset'] = 0;
                kf1['offset'] = 1;
                finalKeyframes = [kf0, kf1];
            }
            return createTimelineInstruction(this.element, finalKeyframes, preProps, postProps, this.duration, this.startTime, this.easing, false);
        }
    }
    class SubTimelineBuilder extends TimelineBuilder {
        /**
         * @param {?} driver
         * @param {?} element
         * @param {?} keyframes
         * @param {?} preStyleProps
         * @param {?} postStyleProps
         * @param {?} timings
         * @param {?=} _stretchStartingKeyframe
         */
        constructor(driver, element, keyframes, preStyleProps, postStyleProps, timings, _stretchStartingKeyframe = false) {
            super(driver, element, timings.delay);
            this.element = element;
            this.keyframes = keyframes;
            this.preStyleProps = preStyleProps;
            this.postStyleProps = postStyleProps;
            this._stretchStartingKeyframe = _stretchStartingKeyframe;
            this.timings = { duration: timings.duration, delay: timings.delay, easing: timings.easing };
        }
        /**
         * @return {?}
         */
        containsAnimation() {
            return this.keyframes.length > 1;
        }
        /**
         * @return {?}
         */
        buildKeyframes() {
            /** @type {?} */
            let keyframes = this.keyframes;
            let { delay, duration, easing } = this.timings;
            if (this._stretchStartingKeyframe && delay) {
                /** @type {?} */
                const newKeyframes = [];
                /** @type {?} */
                const totalTime = duration + delay;
                /** @type {?} */
                const startingGap = delay / totalTime;
                // the original starting keyframe now starts once the delay is done
                /** @type {?} */
                const newFirstKeyframe = copyStyles(keyframes[0], false);
                newFirstKeyframe['offset'] = 0;
                newKeyframes.push(newFirstKeyframe);
                /** @type {?} */
                const oldFirstKeyframe = copyStyles(keyframes[0], false);
                oldFirstKeyframe['offset'] = roundOffset(startingGap);
                newKeyframes.push(oldFirstKeyframe);
                /*
                        When the keyframe is stretched then it means that the delay before the animation
                        starts is gone. Instead the first keyframe is placed at the start of the animation
                        and it is then copied to where it starts when the original delay is over. This basically
                        means nothing animates during that delay, but the styles are still renderered. For this
                        to work the original offset values that exist in the original keyframes must be "warped"
                        so that they can take the new keyframe + delay into account.
                
                        delay=1000, duration=1000, keyframes = 0 .5 1
                
                        turns into
                
                        delay=0, duration=2000, keyframes = 0 .33 .66 1
                       */
                // offsets between 1 ... n -1 are all warped by the keyframe stretch
                /** @type {?} */
                const limit = keyframes.length - 1;
                for (let i = 1; i <= limit; i++) {
                    /** @type {?} */
                    let kf = copyStyles(keyframes[i], false);
                    /** @type {?} */
                    const oldOffset = (/** @type {?} */ (kf['offset']));
                    /** @type {?} */
                    const timeAtKeyframe = delay + oldOffset * duration;
                    kf['offset'] = roundOffset(timeAtKeyframe / totalTime);
                    newKeyframes.push(kf);
                }
                // the new starting keyframe should be added at the start
                duration = totalTime;
                delay = 0;
                easing = '';
                keyframes = newKeyframes;
            }
            return createTimelineInstruction(this.element, keyframes, this.preStyleProps, this.postStyleProps, duration, delay, easing, true);
        }
    }
    /**
     * @param {?} offset
     * @param {?=} decimalPoints
     * @return {?}
     */
    function roundOffset(offset, decimalPoints = 3) {
        /** @type {?} */
        const mult = Math.pow(10, decimalPoints - 1);
        return Math.round(offset * mult) / mult;
    }
    /**
     * @param {?} input
     * @param {?} allStyles
     * @return {?}
     */
    function flattenStyles$1(input, allStyles) {
        /** @type {?} */
        const styles = {};
        /** @type {?} */
        let allProperties;
        input.forEach((/**
         * @param {?} token
         * @return {?}
         */
        token => {
            if (token === '*') {
                allProperties = allProperties || Object.keys(allStyles);
                allProperties.forEach((/**
                 * @param {?} prop
                 * @return {?}
                 */
                prop => {
                    styles[prop] = AUTO_STYLE;
                }));
            }
            else {
                copyStyles((/** @type {?} */ (token)), false, styles);
            }
        }));
        return styles;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/animations/browser/src/dsl/style_normalization/animation_style_normalizer.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * \@publicApi
     * @abstract
     */
    class AnimationStyleNormalizer {
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/animations/browser/src/dsl/style_normalization/web_animations_style_normalizer.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class WebAnimationsStyleNormalizer extends AnimationStyleNormalizer {
        /**
         * @param {?} propertyName
         * @param {?} errors
         * @return {?}
         */
        normalizePropertyName(propertyName, errors) {
            return dashCaseToCamelCase(propertyName);
        }
        /**
         * @param {?} userProvidedProperty
         * @param {?} normalizedProperty
         * @param {?} value
         * @param {?} errors
         * @return {?}
         */
        normalizeStyleValue(userProvidedProperty, normalizedProperty, value, errors) {
            /** @type {?} */
            let unit = '';
            /** @type {?} */
            const strVal = value.toString().trim();
            if (DIMENSIONAL_PROP_MAP[normalizedProperty] && value !== 0 && value !== '0') {
                if (typeof value === 'number') {
                    unit = 'px';
                }
                else {
                    /** @type {?} */
                    const valAndSuffixMatch = value.match(/^[+-]?[\d\.]+([a-z]*)$/);
                    if (valAndSuffixMatch && valAndSuffixMatch[1].length == 0) {
                        errors.push(`Please provide a CSS unit value for ${userProvidedProperty}:${value}`);
                    }
                }
            }
            return strVal + unit;
        }
    }
    const ɵ0$1$2 = /**
     * @return {?}
     */
    () => makeBooleanMap('width,height,minWidth,minHeight,maxWidth,maxHeight,left,top,bottom,right,fontSize,outlineWidth,outlineOffset,paddingTop,paddingLeft,paddingBottom,paddingRight,marginTop,marginLeft,marginBottom,marginRight,borderRadius,borderWidth,borderTopWidth,borderLeftWidth,borderRightWidth,borderBottomWidth,textIndent,perspective'
        .split(','));
    /** @type {?} */
    const DIMENSIONAL_PROP_MAP = ((ɵ0$1$2))();
    /**
     * @param {?} keys
     * @return {?}
     */
    function makeBooleanMap(keys) {
        /** @type {?} */
        const map = {};
        keys.forEach((/**
         * @param {?} key
         * @return {?}
         */
        key => map[key] = true));
        return map;
    }
    /**
     * @param {?} element
     * @param {?} triggerName
     * @param {?} fromState
     * @param {?} toState
     * @param {?} isRemovalTransition
     * @param {?} fromStyles
     * @param {?} toStyles
     * @param {?} timelines
     * @param {?} queriedElements
     * @param {?} preStyleProps
     * @param {?} postStyleProps
     * @param {?} totalTime
     * @param {?=} errors
     * @return {?}
     */
    function createTransitionInstruction(element, triggerName, fromState, toState, isRemovalTransition, fromStyles, toStyles, timelines, queriedElements, preStyleProps, postStyleProps, totalTime, errors) {
        return {
            type: 0 /* TransitionAnimation */,
            element,
            triggerName,
            isRemovalTransition,
            fromState,
            fromStyles,
            toState,
            toStyles,
            timelines,
            queriedElements,
            preStyleProps,
            postStyleProps,
            totalTime,
            errors
        };
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/animations/browser/src/dsl/animation_transition_factory.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const EMPTY_OBJECT = {};
    class AnimationTransitionFactory {
        /**
         * @param {?} _triggerName
         * @param {?} ast
         * @param {?} _stateStyles
         */
        constructor(_triggerName, ast, _stateStyles) {
            this._triggerName = _triggerName;
            this.ast = ast;
            this._stateStyles = _stateStyles;
        }
        /**
         * @param {?} currentState
         * @param {?} nextState
         * @param {?} element
         * @param {?} params
         * @return {?}
         */
        match(currentState, nextState, element, params) {
            return oneOrMoreTransitionsMatch(this.ast.matchers, currentState, nextState, element, params);
        }
        /**
         * @param {?} stateName
         * @param {?} params
         * @param {?} errors
         * @return {?}
         */
        buildStyles(stateName, params, errors) {
            /** @type {?} */
            const backupStateStyler = this._stateStyles['*'];
            /** @type {?} */
            const stateStyler = this._stateStyles[stateName];
            /** @type {?} */
            const backupStyles = backupStateStyler ? backupStateStyler.buildStyles(params, errors) : {};
            return stateStyler ? stateStyler.buildStyles(params, errors) : backupStyles;
        }
        /**
         * @param {?} driver
         * @param {?} element
         * @param {?} currentState
         * @param {?} nextState
         * @param {?} enterClassName
         * @param {?} leaveClassName
         * @param {?=} currentOptions
         * @param {?=} nextOptions
         * @param {?=} subInstructions
         * @param {?=} skipAstBuild
         * @return {?}
         */
        build(driver, element, currentState, nextState, enterClassName, leaveClassName, currentOptions, nextOptions, subInstructions, skipAstBuild) {
            /** @type {?} */
            const errors = [];
            /** @type {?} */
            const transitionAnimationParams = this.ast.options && this.ast.options.params || EMPTY_OBJECT;
            /** @type {?} */
            const currentAnimationParams = currentOptions && currentOptions.params || EMPTY_OBJECT;
            /** @type {?} */
            const currentStateStyles = this.buildStyles(currentState, currentAnimationParams, errors);
            /** @type {?} */
            const nextAnimationParams = nextOptions && nextOptions.params || EMPTY_OBJECT;
            /** @type {?} */
            const nextStateStyles = this.buildStyles(nextState, nextAnimationParams, errors);
            /** @type {?} */
            const queriedElements = new Set();
            /** @type {?} */
            const preStyleMap = new Map();
            /** @type {?} */
            const postStyleMap = new Map();
            /** @type {?} */
            const isRemoval = nextState === 'void';
            /** @type {?} */
            const animationOptions = { params: Object.assign(Object.assign({}, transitionAnimationParams), nextAnimationParams) };
            /** @type {?} */
            const timelines = skipAstBuild ?
                [] :
                buildAnimationTimelines(driver, element, this.ast.animation, enterClassName, leaveClassName, currentStateStyles, nextStateStyles, animationOptions, subInstructions, errors);
            /** @type {?} */
            let totalTime = 0;
            timelines.forEach((/**
             * @param {?} tl
             * @return {?}
             */
            tl => {
                totalTime = Math.max(tl.duration + tl.delay, totalTime);
            }));
            if (errors.length) {
                return createTransitionInstruction(element, this._triggerName, currentState, nextState, isRemoval, currentStateStyles, nextStateStyles, [], [], preStyleMap, postStyleMap, totalTime, errors);
            }
            timelines.forEach((/**
             * @param {?} tl
             * @return {?}
             */
            tl => {
                /** @type {?} */
                const elm = tl.element;
                /** @type {?} */
                const preProps = getOrSetAsInMap(preStyleMap, elm, {});
                tl.preStyleProps.forEach((/**
                 * @param {?} prop
                 * @return {?}
                 */
                prop => preProps[prop] = true));
                /** @type {?} */
                const postProps = getOrSetAsInMap(postStyleMap, elm, {});
                tl.postStyleProps.forEach((/**
                 * @param {?} prop
                 * @return {?}
                 */
                prop => postProps[prop] = true));
                if (elm !== element) {
                    queriedElements.add(elm);
                }
            }));
            /** @type {?} */
            const queriedElementsList = iteratorToArray(queriedElements.values());
            return createTransitionInstruction(element, this._triggerName, currentState, nextState, isRemoval, currentStateStyles, nextStateStyles, timelines, queriedElementsList, preStyleMap, postStyleMap, totalTime);
        }
    }
    /**
     * @param {?} matchFns
     * @param {?} currentState
     * @param {?} nextState
     * @param {?} element
     * @param {?} params
     * @return {?}
     */
    function oneOrMoreTransitionsMatch(matchFns, currentState, nextState, element, params) {
        return matchFns.some((/**
         * @param {?} fn
         * @return {?}
         */
        fn => fn(currentState, nextState, element, params)));
    }
    class AnimationStateStyles {
        /**
         * @param {?} styles
         * @param {?} defaultParams
         */
        constructor(styles, defaultParams) {
            this.styles = styles;
            this.defaultParams = defaultParams;
        }
        /**
         * @param {?} params
         * @param {?} errors
         * @return {?}
         */
        buildStyles(params, errors) {
            /** @type {?} */
            const finalStyles = {};
            /** @type {?} */
            const combinedParams = copyObj(this.defaultParams);
            Object.keys(params).forEach((/**
             * @param {?} key
             * @return {?}
             */
            key => {
                /** @type {?} */
                const value = params[key];
                if (value != null) {
                    combinedParams[key] = value;
                }
            }));
            this.styles.styles.forEach((/**
             * @param {?} value
             * @return {?}
             */
            value => {
                if (typeof value !== 'string') {
                    /** @type {?} */
                    const styleObj = (/** @type {?} */ (value));
                    Object.keys(styleObj).forEach((/**
                     * @param {?} prop
                     * @return {?}
                     */
                    prop => {
                        /** @type {?} */
                        let val = styleObj[prop];
                        if (val.length > 1) {
                            val = interpolateParams(val, combinedParams, errors);
                        }
                        finalStyles[prop] = val;
                    }));
                }
            }));
            return finalStyles;
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/animations/browser/src/dsl/animation_trigger.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * \@publicApi
     * @param {?} name
     * @param {?} ast
     * @return {?}
     */
    function buildTrigger(name, ast) {
        return new AnimationTrigger(name, ast);
    }
    /**
     * \@publicApi
     */
    class AnimationTrigger {
        /**
         * @param {?} name
         * @param {?} ast
         */
        constructor(name, ast) {
            this.name = name;
            this.ast = ast;
            this.transitionFactories = [];
            this.states = {};
            ast.states.forEach((/**
             * @param {?} ast
             * @return {?}
             */
            ast => {
                /** @type {?} */
                const defaultParams = (ast.options && ast.options.params) || {};
                this.states[ast.name] = new AnimationStateStyles(ast.style, defaultParams);
            }));
            balanceProperties(this.states, 'true', '1');
            balanceProperties(this.states, 'false', '0');
            ast.transitions.forEach((/**
             * @param {?} ast
             * @return {?}
             */
            ast => {
                this.transitionFactories.push(new AnimationTransitionFactory(name, ast, this.states));
            }));
            this.fallbackTransition = createFallbackTransition(name, this.states);
        }
        /**
         * @return {?}
         */
        get containsQueries() {
            return this.ast.queryCount > 0;
        }
        /**
         * @param {?} currentState
         * @param {?} nextState
         * @param {?} element
         * @param {?} params
         * @return {?}
         */
        matchTransition(currentState, nextState, element, params) {
            /** @type {?} */
            const entry = this.transitionFactories.find((/**
             * @param {?} f
             * @return {?}
             */
            f => f.match(currentState, nextState, element, params)));
            return entry || null;
        }
        /**
         * @param {?} currentState
         * @param {?} params
         * @param {?} errors
         * @return {?}
         */
        matchStyles(currentState, params, errors) {
            return this.fallbackTransition.buildStyles(currentState, params, errors);
        }
    }
    /**
     * @param {?} triggerName
     * @param {?} states
     * @return {?}
     */
    function createFallbackTransition(triggerName, states) {
        /** @type {?} */
        const matchers = [(/**
             * @param {?} fromState
             * @param {?} toState
             * @return {?}
             */
            (fromState, toState) => true)];
        /** @type {?} */
        const animation = { type: 2 /* Sequence */, steps: [], options: null };
        /** @type {?} */
        const transition = {
            type: 1 /* Transition */,
            animation,
            matchers,
            options: null,
            queryCount: 0,
            depCount: 0
        };
        return new AnimationTransitionFactory(triggerName, transition, states);
    }
    /**
     * @param {?} obj
     * @param {?} key1
     * @param {?} key2
     * @return {?}
     */
    function balanceProperties(obj, key1, key2) {
        if (obj.hasOwnProperty(key1)) {
            if (!obj.hasOwnProperty(key2)) {
                obj[key2] = obj[key1];
            }
        }
        else if (obj.hasOwnProperty(key2)) {
            obj[key1] = obj[key2];
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/animations/browser/src/render/timeline_animation_engine.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const EMPTY_INSTRUCTION_MAP = new ElementInstructionMap();
    class TimelineAnimationEngine {
        /**
         * @param {?} bodyNode
         * @param {?} _driver
         * @param {?} _normalizer
         */
        constructor(bodyNode, _driver, _normalizer) {
            this.bodyNode = bodyNode;
            this._driver = _driver;
            this._normalizer = _normalizer;
            this._animations = {};
            this._playersById = {};
            this.players = [];
        }
        /**
         * @param {?} id
         * @param {?} metadata
         * @return {?}
         */
        register(id, metadata) {
            /** @type {?} */
            const errors = [];
            /** @type {?} */
            const ast = buildAnimationAst(this._driver, metadata, errors);
            if (errors.length) {
                throw new Error(`Unable to build the animation due to the following errors: ${errors.join('\n')}`);
            }
            else {
                this._animations[id] = ast;
            }
        }
        /**
         * @private
         * @param {?} i
         * @param {?} preStyles
         * @param {?=} postStyles
         * @return {?}
         */
        _buildPlayer(i, preStyles, postStyles) {
            /** @type {?} */
            const element = i.element;
            /** @type {?} */
            const keyframes = normalizeKeyframes(this._driver, this._normalizer, element, i.keyframes, preStyles, postStyles);
            return this._driver.animate(element, keyframes, i.duration, i.delay, i.easing, [], true);
        }
        /**
         * @param {?} id
         * @param {?} element
         * @param {?=} options
         * @return {?}
         */
        create(id, element, options = {}) {
            /** @type {?} */
            const errors = [];
            /** @type {?} */
            const ast = this._animations[id];
            /** @type {?} */
            let instructions;
            /** @type {?} */
            const autoStylesMap = new Map();
            if (ast) {
                instructions = buildAnimationTimelines(this._driver, element, ast, ENTER_CLASSNAME, LEAVE_CLASSNAME, {}, {}, options, EMPTY_INSTRUCTION_MAP, errors);
                instructions.forEach((/**
                 * @param {?} inst
                 * @return {?}
                 */
                inst => {
                    /** @type {?} */
                    const styles = getOrSetAsInMap(autoStylesMap, inst.element, {});
                    inst.postStyleProps.forEach((/**
                     * @param {?} prop
                     * @return {?}
                     */
                    prop => styles[prop] = null));
                }));
            }
            else {
                errors.push('The requested animation doesn\'t exist or has already been destroyed');
                instructions = [];
            }
            if (errors.length) {
                throw new Error(`Unable to create the animation due to the following errors: ${errors.join('\n')}`);
            }
            autoStylesMap.forEach((/**
             * @param {?} styles
             * @param {?} element
             * @return {?}
             */
            (styles, element) => {
                Object.keys(styles).forEach((/**
                 * @param {?} prop
                 * @return {?}
                 */
                prop => {
                    styles[prop] = this._driver.computeStyle(element, prop, AUTO_STYLE);
                }));
            }));
            /** @type {?} */
            const players = instructions.map((/**
             * @param {?} i
             * @return {?}
             */
            i => {
                /** @type {?} */
                const styles = autoStylesMap.get(i.element);
                return this._buildPlayer(i, {}, styles);
            }));
            /** @type {?} */
            const player = optimizeGroupPlayer(players);
            this._playersById[id] = player;
            player.onDestroy((/**
             * @return {?}
             */
            () => this.destroy(id)));
            this.players.push(player);
            return player;
        }
        /**
         * @param {?} id
         * @return {?}
         */
        destroy(id) {
            /** @type {?} */
            const player = this._getPlayer(id);
            player.destroy();
            delete this._playersById[id];
            /** @type {?} */
            const index = this.players.indexOf(player);
            if (index >= 0) {
                this.players.splice(index, 1);
            }
        }
        /**
         * @private
         * @param {?} id
         * @return {?}
         */
        _getPlayer(id) {
            /** @type {?} */
            const player = this._playersById[id];
            if (!player) {
                throw new Error(`Unable to find the timeline player referenced by ${id}`);
            }
            return player;
        }
        /**
         * @param {?} id
         * @param {?} element
         * @param {?} eventName
         * @param {?} callback
         * @return {?}
         */
        listen(id, element, eventName, callback) {
            // triggerName, fromState, toState are all ignored for timeline animations
            /** @type {?} */
            const baseEvent = makeAnimationEvent(element, '', '', '');
            listenOnPlayer(this._getPlayer(id), eventName, baseEvent, callback);
            return (/**
             * @return {?}
             */
            () => { });
        }
        /**
         * @param {?} id
         * @param {?} element
         * @param {?} command
         * @param {?} args
         * @return {?}
         */
        command(id, element, command, args) {
            if (command == 'register') {
                this.register(id, (/** @type {?} */ (args[0])));
                return;
            }
            if (command == 'create') {
                /** @type {?} */
                const options = (/** @type {?} */ ((args[0] || {})));
                this.create(id, element, options);
                return;
            }
            /** @type {?} */
            const player = this._getPlayer(id);
            switch (command) {
                case 'play':
                    player.play();
                    break;
                case 'pause':
                    player.pause();
                    break;
                case 'reset':
                    player.reset();
                    break;
                case 'restart':
                    player.restart();
                    break;
                case 'finish':
                    player.finish();
                    break;
                case 'init':
                    player.init();
                    break;
                case 'setPosition':
                    player.setPosition(parseFloat((/** @type {?} */ (args[0]))));
                    break;
                case 'destroy':
                    this.destroy(id);
                    break;
            }
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/animations/browser/src/render/transition_animation_engine.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const QUEUED_CLASSNAME = 'ng-animate-queued';
    /** @type {?} */
    const QUEUED_SELECTOR = '.ng-animate-queued';
    /** @type {?} */
    const DISABLED_CLASSNAME = 'ng-animate-disabled';
    /** @type {?} */
    const DISABLED_SELECTOR = '.ng-animate-disabled';
    /** @type {?} */
    const STAR_CLASSNAME = 'ng-star-inserted';
    /** @type {?} */
    const STAR_SELECTOR = '.ng-star-inserted';
    /** @type {?} */
    const EMPTY_PLAYER_ARRAY = [];
    /** @type {?} */
    const NULL_REMOVAL_STATE = {
        namespaceId: '',
        setForRemoval: false,
        setForMove: false,
        hasAnimation: false,
        removedBeforeQueried: false
    };
    /** @type {?} */
    const NULL_REMOVED_QUERIED_STATE = {
        namespaceId: '',
        setForMove: false,
        setForRemoval: false,
        hasAnimation: false,
        removedBeforeQueried: true
    };
    /** @type {?} */
    const REMOVAL_FLAG = '__ng_removed';
    class StateValue {
        /**
         * @param {?} input
         * @param {?=} namespaceId
         */
        constructor(input, namespaceId = '') {
            this.namespaceId = namespaceId;
            /** @type {?} */
            const isObj = input && input.hasOwnProperty('value');
            /** @type {?} */
            const value = isObj ? input['value'] : input;
            this.value = normalizeTriggerValue(value);
            if (isObj) {
                /** @type {?} */
                const options = copyObj((/** @type {?} */ (input)));
                delete options['value'];
                this.options = (/** @type {?} */ (options));
            }
            else {
                this.options = {};
            }
            if (!this.options.params) {
                this.options.params = {};
            }
        }
        /**
         * @return {?}
         */
        get params() {
            return (/** @type {?} */ (this.options.params));
        }
        /**
         * @param {?} options
         * @return {?}
         */
        absorbOptions(options) {
            /** @type {?} */
            const newParams = options.params;
            if (newParams) {
                /** @type {?} */
                const oldParams = (/** @type {?} */ (this.options.params));
                Object.keys(newParams).forEach((/**
                 * @param {?} prop
                 * @return {?}
                 */
                prop => {
                    if (oldParams[prop] == null) {
                        oldParams[prop] = newParams[prop];
                    }
                }));
            }
        }
    }
    /** @type {?} */
    const VOID_VALUE = 'void';
    /** @type {?} */
    const DEFAULT_STATE_VALUE = new StateValue(VOID_VALUE);
    class AnimationTransitionNamespace {
        /**
         * @param {?} id
         * @param {?} hostElement
         * @param {?} _engine
         */
        constructor(id, hostElement, _engine) {
            this.id = id;
            this.hostElement = hostElement;
            this._engine = _engine;
            this.players = [];
            this._triggers = {};
            this._queue = [];
            this._elementListeners = new Map();
            this._hostClassName = 'ng-tns-' + id;
            addClass(hostElement, this._hostClassName);
        }
        /**
         * @param {?} element
         * @param {?} name
         * @param {?} phase
         * @param {?} callback
         * @return {?}
         */
        listen(element, name, phase, callback) {
            if (!this._triggers.hasOwnProperty(name)) {
                throw new Error(`Unable to listen on the animation trigger event "${phase}" because the animation trigger "${name}" doesn\'t exist!`);
            }
            if (phase == null || phase.length == 0) {
                throw new Error(`Unable to listen on the animation trigger "${name}" because the provided event is undefined!`);
            }
            if (!isTriggerEventValid(phase)) {
                throw new Error(`The provided animation trigger event "${phase}" for the animation trigger "${name}" is not supported!`);
            }
            /** @type {?} */
            const listeners = getOrSetAsInMap(this._elementListeners, element, []);
            /** @type {?} */
            const data = { name, phase, callback };
            listeners.push(data);
            /** @type {?} */
            const triggersWithStates = getOrSetAsInMap(this._engine.statesByElement, element, {});
            if (!triggersWithStates.hasOwnProperty(name)) {
                addClass(element, NG_TRIGGER_CLASSNAME);
                addClass(element, NG_TRIGGER_CLASSNAME + '-' + name);
                triggersWithStates[name] = DEFAULT_STATE_VALUE;
            }
            return (/**
             * @return {?}
             */
            () => {
                // the event listener is removed AFTER the flush has occurred such
                // that leave animations callbacks can fire (otherwise if the node
                // is removed in between then the listeners would be deregistered)
                this._engine.afterFlush((/**
                 * @return {?}
                 */
                () => {
                    /** @type {?} */
                    const index = listeners.indexOf(data);
                    if (index >= 0) {
                        listeners.splice(index, 1);
                    }
                    if (!this._triggers[name]) {
                        delete triggersWithStates[name];
                    }
                }));
            });
        }
        /**
         * @param {?} name
         * @param {?} ast
         * @return {?}
         */
        register(name, ast) {
            if (this._triggers[name]) {
                // throw
                return false;
            }
            else {
                this._triggers[name] = ast;
                return true;
            }
        }
        /**
         * @private
         * @param {?} name
         * @return {?}
         */
        _getTrigger(name) {
            /** @type {?} */
            const trigger = this._triggers[name];
            if (!trigger) {
                throw new Error(`The provided animation trigger "${name}" has not been registered!`);
            }
            return trigger;
        }
        /**
         * @param {?} element
         * @param {?} triggerName
         * @param {?} value
         * @param {?=} defaultToFallback
         * @return {?}
         */
        trigger(element, triggerName, value, defaultToFallback = true) {
            /** @type {?} */
            const trigger = this._getTrigger(triggerName);
            /** @type {?} */
            const player = new TransitionAnimationPlayer(this.id, triggerName, element);
            /** @type {?} */
            let triggersWithStates = this._engine.statesByElement.get(element);
            if (!triggersWithStates) {
                addClass(element, NG_TRIGGER_CLASSNAME);
                addClass(element, NG_TRIGGER_CLASSNAME + '-' + triggerName);
                this._engine.statesByElement.set(element, triggersWithStates = {});
            }
            /** @type {?} */
            let fromState = triggersWithStates[triggerName];
            /** @type {?} */
            const toState = new StateValue(value, this.id);
            /** @type {?} */
            const isObj = value && value.hasOwnProperty('value');
            if (!isObj && fromState) {
                toState.absorbOptions(fromState.options);
            }
            triggersWithStates[triggerName] = toState;
            if (!fromState) {
                fromState = DEFAULT_STATE_VALUE;
            }
            /** @type {?} */
            const isRemoval = toState.value === VOID_VALUE;
            // normally this isn't reached by here, however, if an object expression
            // is passed in then it may be a new object each time. Comparing the value
            // is important since that will stay the same despite there being a new object.
            // The removal arc here is special cased because the same element is triggered
            // twice in the event that it contains animations on the outer/inner portions
            // of the host container
            if (!isRemoval && fromState.value === toState.value) {
                // this means that despite the value not changing, some inner params
                // have changed which means that the animation final styles need to be applied
                if (!objEquals(fromState.params, toState.params)) {
                    /** @type {?} */
                    const errors = [];
                    /** @type {?} */
                    const fromStyles = trigger.matchStyles(fromState.value, fromState.params, errors);
                    /** @type {?} */
                    const toStyles = trigger.matchStyles(toState.value, toState.params, errors);
                    if (errors.length) {
                        this._engine.reportError(errors);
                    }
                    else {
                        this._engine.afterFlush((/**
                         * @return {?}
                         */
                        () => {
                            eraseStyles(element, fromStyles);
                            setStyles(element, toStyles);
                        }));
                    }
                }
                return;
            }
            /** @type {?} */
            const playersOnElement = getOrSetAsInMap(this._engine.playersByElement, element, []);
            playersOnElement.forEach((/**
             * @param {?} player
             * @return {?}
             */
            player => {
                // only remove the player if it is queued on the EXACT same trigger/namespace
                // we only also deal with queued players here because if the animation has
                // started then we want to keep the player alive until the flush happens
                // (which is where the previousPlayers are passed into the new palyer)
                if (player.namespaceId == this.id && player.triggerName == triggerName && player.queued) {
                    player.destroy();
                }
            }));
            /** @type {?} */
            let transition = trigger.matchTransition(fromState.value, toState.value, element, toState.params);
            /** @type {?} */
            let isFallbackTransition = false;
            if (!transition) {
                if (!defaultToFallback)
                    return;
                transition = trigger.fallbackTransition;
                isFallbackTransition = true;
            }
            this._engine.totalQueuedPlayers++;
            this._queue.push({ element, triggerName, transition, fromState, toState, player, isFallbackTransition });
            if (!isFallbackTransition) {
                addClass(element, QUEUED_CLASSNAME);
                player.onStart((/**
                 * @return {?}
                 */
                () => {
                    removeClass(element, QUEUED_CLASSNAME);
                }));
            }
            player.onDone((/**
             * @return {?}
             */
            () => {
                /** @type {?} */
                let index = this.players.indexOf(player);
                if (index >= 0) {
                    this.players.splice(index, 1);
                }
                /** @type {?} */
                const players = this._engine.playersByElement.get(element);
                if (players) {
                    /** @type {?} */
                    let index = players.indexOf(player);
                    if (index >= 0) {
                        players.splice(index, 1);
                    }
                }
            }));
            this.players.push(player);
            playersOnElement.push(player);
            return player;
        }
        /**
         * @param {?} name
         * @return {?}
         */
        deregister(name) {
            delete this._triggers[name];
            this._engine.statesByElement.forEach((/**
             * @param {?} stateMap
             * @param {?} element
             * @return {?}
             */
            (stateMap, element) => {
                delete stateMap[name];
            }));
            this._elementListeners.forEach((/**
             * @param {?} listeners
             * @param {?} element
             * @return {?}
             */
            (listeners, element) => {
                this._elementListeners.set(element, listeners.filter((/**
                 * @param {?} entry
                 * @return {?}
                 */
                entry => {
                    return entry.name != name;
                })));
            }));
        }
        /**
         * @param {?} element
         * @return {?}
         */
        clearElementCache(element) {
            this._engine.statesByElement.delete(element);
            this._elementListeners.delete(element);
            /** @type {?} */
            const elementPlayers = this._engine.playersByElement.get(element);
            if (elementPlayers) {
                elementPlayers.forEach((/**
                 * @param {?} player
                 * @return {?}
                 */
                player => player.destroy()));
                this._engine.playersByElement.delete(element);
            }
        }
        /**
         * @private
         * @param {?} rootElement
         * @param {?} context
         * @return {?}
         */
        _signalRemovalForInnerTriggers(rootElement, context) {
            /** @type {?} */
            const elements = this._engine.driver.query(rootElement, NG_TRIGGER_SELECTOR, true);
            // emulate a leave animation for all inner nodes within this node.
            // If there are no animations found for any of the nodes then clear the cache
            // for the element.
            elements.forEach((/**
             * @param {?} elm
             * @return {?}
             */
            elm => {
                // this means that an inner remove() operation has already kicked off
                // the animation on this element...
                if (elm[REMOVAL_FLAG])
                    return;
                /** @type {?} */
                const namespaces = this._engine.fetchNamespacesByElement(elm);
                if (namespaces.size) {
                    namespaces.forEach((/**
                     * @param {?} ns
                     * @return {?}
                     */
                    ns => ns.triggerLeaveAnimation(elm, context, false, true)));
                }
                else {
                    this.clearElementCache(elm);
                }
            }));
            // If the child elements were removed along with the parent, their animations might not
            // have completed. Clear all the elements from the cache so we don't end up with a memory leak.
            this._engine.afterFlushAnimationsDone((/**
             * @return {?}
             */
            () => elements.forEach((/**
             * @param {?} elm
             * @return {?}
             */
            elm => this.clearElementCache(elm)))));
        }
        /**
         * @param {?} element
         * @param {?} context
         * @param {?=} destroyAfterComplete
         * @param {?=} defaultToFallback
         * @return {?}
         */
        triggerLeaveAnimation(element, context, destroyAfterComplete, defaultToFallback) {
            /** @type {?} */
            const triggerStates = this._engine.statesByElement.get(element);
            if (triggerStates) {
                /** @type {?} */
                const players = [];
                Object.keys(triggerStates).forEach((/**
                 * @param {?} triggerName
                 * @return {?}
                 */
                triggerName => {
                    // this check is here in the event that an element is removed
                    // twice (both on the host level and the component level)
                    if (this._triggers[triggerName]) {
                        /** @type {?} */
                        const player = this.trigger(element, triggerName, VOID_VALUE, defaultToFallback);
                        if (player) {
                            players.push(player);
                        }
                    }
                }));
                if (players.length) {
                    this._engine.markElementAsRemoved(this.id, element, true, context);
                    if (destroyAfterComplete) {
                        optimizeGroupPlayer(players).onDone((/**
                         * @return {?}
                         */
                        () => this._engine.processLeaveNode(element)));
                    }
                    return true;
                }
            }
            return false;
        }
        /**
         * @param {?} element
         * @return {?}
         */
        prepareLeaveAnimationListeners(element) {
            /** @type {?} */
            const listeners = this._elementListeners.get(element);
            if (listeners) {
                /** @type {?} */
                const visitedTriggers = new Set();
                listeners.forEach((/**
                 * @param {?} listener
                 * @return {?}
                 */
                listener => {
                    /** @type {?} */
                    const triggerName = listener.name;
                    if (visitedTriggers.has(triggerName))
                        return;
                    visitedTriggers.add(triggerName);
                    /** @type {?} */
                    const trigger = this._triggers[triggerName];
                    /** @type {?} */
                    const transition = trigger.fallbackTransition;
                    /** @type {?} */
                    const elementStates = (/** @type {?} */ (this._engine.statesByElement.get(element)));
                    /** @type {?} */
                    const fromState = elementStates[triggerName] || DEFAULT_STATE_VALUE;
                    /** @type {?} */
                    const toState = new StateValue(VOID_VALUE);
                    /** @type {?} */
                    const player = new TransitionAnimationPlayer(this.id, triggerName, element);
                    this._engine.totalQueuedPlayers++;
                    this._queue.push({
                        element,
                        triggerName,
                        transition,
                        fromState,
                        toState,
                        player,
                        isFallbackTransition: true
                    });
                }));
            }
        }
        /**
         * @param {?} element
         * @param {?} context
         * @return {?}
         */
        removeNode(element, context) {
            /** @type {?} */
            const engine = this._engine;
            if (element.childElementCount) {
                this._signalRemovalForInnerTriggers(element, context);
            }
            // this means that a * => VOID animation was detected and kicked off
            if (this.triggerLeaveAnimation(element, context, true))
                return;
            // find the player that is animating and make sure that the
            // removal is delayed until that player has completed
            /** @type {?} */
            let containsPotentialParentTransition = false;
            if (engine.totalAnimations) {
                /** @type {?} */
                const currentPlayers = engine.players.length ? engine.playersByQueriedElement.get(element) : [];
                // when this `if statement` does not continue forward it means that
                // a previous animation query has selected the current element and
                // is animating it. In this situation want to continue forwards and
                // allow the element to be queued up for animation later.
                if (currentPlayers && currentPlayers.length) {
                    containsPotentialParentTransition = true;
                }
                else {
                    /** @type {?} */
                    let parent = element;
                    while (parent = parent.parentNode) {
                        /** @type {?} */
                        const triggers = engine.statesByElement.get(parent);
                        if (triggers) {
                            containsPotentialParentTransition = true;
                            break;
                        }
                    }
                }
            }
            // at this stage we know that the element will either get removed
            // during flush or will be picked up by a parent query. Either way
            // we need to fire the listeners for this element when it DOES get
            // removed (once the query parent animation is done or after flush)
            this.prepareLeaveAnimationListeners(element);
            // whether or not a parent has an animation we need to delay the deferral of the leave
            // operation until we have more information (which we do after flush() has been called)
            if (containsPotentialParentTransition) {
                engine.markElementAsRemoved(this.id, element, false, context);
            }
            else {
                /** @type {?} */
                const removalFlag = element[REMOVAL_FLAG];
                if (!removalFlag || removalFlag === NULL_REMOVAL_STATE) {
                    // we do this after the flush has occurred such
                    // that the callbacks can be fired
                    engine.afterFlush((/**
                     * @return {?}
                     */
                    () => this.clearElementCache(element)));
                    engine.destroyInnerAnimations(element);
                    engine._onRemovalComplete(element, context);
                }
            }
        }
        /**
         * @param {?} element
         * @param {?} parent
         * @return {?}
         */
        insertNode(element, parent) {
            addClass(element, this._hostClassName);
        }
        /**
         * @param {?} microtaskId
         * @return {?}
         */
        drainQueuedTransitions(microtaskId) {
            /** @type {?} */
            const instructions = [];
            this._queue.forEach((/**
             * @param {?} entry
             * @return {?}
             */
            entry => {
                /** @type {?} */
                const player = entry.player;
                if (player.destroyed)
                    return;
                /** @type {?} */
                const element = entry.element;
                /** @type {?} */
                const listeners = this._elementListeners.get(element);
                if (listeners) {
                    listeners.forEach((/**
                     * @param {?} listener
                     * @return {?}
                     */
                    (listener) => {
                        if (listener.name == entry.triggerName) {
                            /** @type {?} */
                            const baseEvent = makeAnimationEvent(element, entry.triggerName, entry.fromState.value, entry.toState.value);
                            ((/** @type {?} */ (baseEvent)))['_data'] = microtaskId;
                            listenOnPlayer(entry.player, listener.phase, baseEvent, listener.callback);
                        }
                    }));
                }
                if (player.markedForDestroy) {
                    this._engine.afterFlush((/**
                     * @return {?}
                     */
                    () => {
                        // now we can destroy the element properly since the event listeners have
                        // been bound to the player
                        player.destroy();
                    }));
                }
                else {
                    instructions.push(entry);
                }
            }));
            this._queue = [];
            return instructions.sort((/**
             * @param {?} a
             * @param {?} b
             * @return {?}
             */
            (a, b) => {
                // if depCount == 0 them move to front
                // otherwise if a contains b then move back
                /** @type {?} */
                const d0 = a.transition.ast.depCount;
                /** @type {?} */
                const d1 = b.transition.ast.depCount;
                if (d0 == 0 || d1 == 0) {
                    return d0 - d1;
                }
                return this._engine.driver.containsElement(a.element, b.element) ? 1 : -1;
            }));
        }
        /**
         * @param {?} context
         * @return {?}
         */
        destroy(context) {
            this.players.forEach((/**
             * @param {?} p
             * @return {?}
             */
            p => p.destroy()));
            this._signalRemovalForInnerTriggers(this.hostElement, context);
        }
        /**
         * @param {?} element
         * @return {?}
         */
        elementContainsData(element) {
            /** @type {?} */
            let containsData = false;
            if (this._elementListeners.has(element))
                containsData = true;
            containsData =
                (this._queue.find((/**
                 * @param {?} entry
                 * @return {?}
                 */
                entry => entry.element === element)) ? true : false) || containsData;
            return containsData;
        }
    }
    class TransitionAnimationEngine {
        /**
         * @param {?} bodyNode
         * @param {?} driver
         * @param {?} _normalizer
         */
        constructor(bodyNode, driver, _normalizer) {
            this.bodyNode = bodyNode;
            this.driver = driver;
            this._normalizer = _normalizer;
            this.players = [];
            this.newHostElements = new Map();
            this.playersByElement = new Map();
            this.playersByQueriedElement = new Map();
            this.statesByElement = new Map();
            this.disabledNodes = new Set();
            this.totalAnimations = 0;
            this.totalQueuedPlayers = 0;
            this._namespaceLookup = {};
            this._namespaceList = [];
            this._flushFns = [];
            this._whenQuietFns = [];
            this.namespacesByHostElement = new Map();
            this.collectedEnterElements = [];
            this.collectedLeaveElements = [];
            // this method is designed to be overridden by the code that uses this engine
            this.onRemovalComplete = (/**
             * @param {?} element
             * @param {?} context
             * @return {?}
             */
            (element, context) => { });
        }
        /**
         * \@internal
         * @param {?} element
         * @param {?} context
         * @return {?}
         */
        _onRemovalComplete(element, context) {
            this.onRemovalComplete(element, context);
        }
        /**
         * @return {?}
         */
        get queuedPlayers() {
            /** @type {?} */
            const players = [];
            this._namespaceList.forEach((/**
             * @param {?} ns
             * @return {?}
             */
            ns => {
                ns.players.forEach((/**
                 * @param {?} player
                 * @return {?}
                 */
                player => {
                    if (player.queued) {
                        players.push(player);
                    }
                }));
            }));
            return players;
        }
        /**
         * @param {?} namespaceId
         * @param {?} hostElement
         * @return {?}
         */
        createNamespace(namespaceId, hostElement) {
            /** @type {?} */
            const ns = new AnimationTransitionNamespace(namespaceId, hostElement, this);
            if (hostElement.parentNode) {
                this._balanceNamespaceList(ns, hostElement);
            }
            else {
                // defer this later until flush during when the host element has
                // been inserted so that we know exactly where to place it in
                // the namespace list
                this.newHostElements.set(hostElement, ns);
                // given that this host element is apart of the animation code, it
                // may or may not be inserted by a parent node that is an of an
                // animation renderer type. If this happens then we can still have
                // access to this item when we query for :enter nodes. If the parent
                // is a renderer then the set data-structure will normalize the entry
                this.collectEnterElement(hostElement);
            }
            return this._namespaceLookup[namespaceId] = ns;
        }
        /**
         * @private
         * @param {?} ns
         * @param {?} hostElement
         * @return {?}
         */
        _balanceNamespaceList(ns, hostElement) {
            /** @type {?} */
            const limit = this._namespaceList.length - 1;
            if (limit >= 0) {
                /** @type {?} */
                let found = false;
                for (let i = limit; i >= 0; i--) {
                    /** @type {?} */
                    const nextNamespace = this._namespaceList[i];
                    if (this.driver.containsElement(nextNamespace.hostElement, hostElement)) {
                        this._namespaceList.splice(i + 1, 0, ns);
                        found = true;
                        break;
                    }
                }
                if (!found) {
                    this._namespaceList.splice(0, 0, ns);
                }
            }
            else {
                this._namespaceList.push(ns);
            }
            this.namespacesByHostElement.set(hostElement, ns);
            return ns;
        }
        /**
         * @param {?} namespaceId
         * @param {?} hostElement
         * @return {?}
         */
        register(namespaceId, hostElement) {
            /** @type {?} */
            let ns = this._namespaceLookup[namespaceId];
            if (!ns) {
                ns = this.createNamespace(namespaceId, hostElement);
            }
            return ns;
        }
        /**
         * @param {?} namespaceId
         * @param {?} name
         * @param {?} trigger
         * @return {?}
         */
        registerTrigger(namespaceId, name, trigger) {
            /** @type {?} */
            let ns = this._namespaceLookup[namespaceId];
            if (ns && ns.register(name, trigger)) {
                this.totalAnimations++;
            }
        }
        /**
         * @param {?} namespaceId
         * @param {?} context
         * @return {?}
         */
        destroy(namespaceId, context) {
            if (!namespaceId)
                return;
            /** @type {?} */
            const ns = this._fetchNamespace(namespaceId);
            this.afterFlush((/**
             * @return {?}
             */
            () => {
                this.namespacesByHostElement.delete(ns.hostElement);
                delete this._namespaceLookup[namespaceId];
                /** @type {?} */
                const index = this._namespaceList.indexOf(ns);
                if (index >= 0) {
                    this._namespaceList.splice(index, 1);
                }
            }));
            this.afterFlushAnimationsDone((/**
             * @return {?}
             */
            () => ns.destroy(context)));
        }
        /**
         * @private
         * @param {?} id
         * @return {?}
         */
        _fetchNamespace(id) {
            return this._namespaceLookup[id];
        }
        /**
         * @param {?} element
         * @return {?}
         */
        fetchNamespacesByElement(element) {
            // normally there should only be one namespace per element, however
            // if @triggers are placed on both the component element and then
            // its host element (within the component code) then there will be
            // two namespaces returned. We use a set here to simply the dedupe
            // of namespaces incase there are multiple triggers both the elm and host
            /** @type {?} */
            const namespaces = new Set();
            /** @type {?} */
            const elementStates = this.statesByElement.get(element);
            if (elementStates) {
                /** @type {?} */
                const keys = Object.keys(elementStates);
                for (let i = 0; i < keys.length; i++) {
                    /** @type {?} */
                    const nsId = elementStates[keys[i]].namespaceId;
                    if (nsId) {
                        /** @type {?} */
                        const ns = this._fetchNamespace(nsId);
                        if (ns) {
                            namespaces.add(ns);
                        }
                    }
                }
            }
            return namespaces;
        }
        /**
         * @param {?} namespaceId
         * @param {?} element
         * @param {?} name
         * @param {?} value
         * @return {?}
         */
        trigger(namespaceId, element, name, value) {
            if (isElementNode(element)) {
                /** @type {?} */
                const ns = this._fetchNamespace(namespaceId);
                if (ns) {
                    ns.trigger(element, name, value);
                    return true;
                }
            }
            return false;
        }
        /**
         * @param {?} namespaceId
         * @param {?} element
         * @param {?} parent
         * @param {?} insertBefore
         * @return {?}
         */
        insertNode(namespaceId, element, parent, insertBefore) {
            if (!isElementNode(element))
                return;
            // special case for when an element is removed and reinserted (move operation)
            // when this occurs we do not want to use the element for deletion later
            /** @type {?} */
            const details = (/** @type {?} */ (element[REMOVAL_FLAG]));
            if (details && details.setForRemoval) {
                details.setForRemoval = false;
                details.setForMove = true;
                /** @type {?} */
                const index = this.collectedLeaveElements.indexOf(element);
                if (index >= 0) {
                    this.collectedLeaveElements.splice(index, 1);
                }
            }
            // in the event that the namespaceId is blank then the caller
            // code does not contain any animation code in it, but it is
            // just being called so that the node is marked as being inserted
            if (namespaceId) {
                /** @type {?} */
                const ns = this._fetchNamespace(namespaceId);
                // This if-statement is a workaround for router issue #21947.
                // The router sometimes hits a race condition where while a route
                // is being instantiated a new navigation arrives, triggering leave
                // animation of DOM that has not been fully initialized, until this
                // is resolved, we need to handle the scenario when DOM is not in a
                // consistent state during the animation.
                if (ns) {
                    ns.insertNode(element, parent);
                }
            }
            // only *directives and host elements are inserted before
            if (insertBefore) {
                this.collectEnterElement(element);
            }
        }
        /**
         * @param {?} element
         * @return {?}
         */
        collectEnterElement(element) {
            this.collectedEnterElements.push(element);
        }
        /**
         * @param {?} element
         * @param {?} value
         * @return {?}
         */
        markElementAsDisabled(element, value) {
            if (value) {
                if (!this.disabledNodes.has(element)) {
                    this.disabledNodes.add(element);
                    addClass(element, DISABLED_CLASSNAME);
                }
            }
            else if (this.disabledNodes.has(element)) {
                this.disabledNodes.delete(element);
                removeClass(element, DISABLED_CLASSNAME);
            }
        }
        /**
         * @param {?} namespaceId
         * @param {?} element
         * @param {?} isHostElement
         * @param {?} context
         * @return {?}
         */
        removeNode(namespaceId, element, isHostElement, context) {
            if (isElementNode(element)) {
                /** @type {?} */
                const ns = namespaceId ? this._fetchNamespace(namespaceId) : null;
                if (ns) {
                    ns.removeNode(element, context);
                }
                else {
                    this.markElementAsRemoved(namespaceId, element, false, context);
                }
                if (isHostElement) {
                    /** @type {?} */
                    const hostNS = this.namespacesByHostElement.get(element);
                    if (hostNS && hostNS.id !== namespaceId) {
                        hostNS.removeNode(element, context);
                    }
                }
            }
            else {
                this._onRemovalComplete(element, context);
            }
        }
        /**
         * @param {?} namespaceId
         * @param {?} element
         * @param {?=} hasAnimation
         * @param {?=} context
         * @return {?}
         */
        markElementAsRemoved(namespaceId, element, hasAnimation, context) {
            this.collectedLeaveElements.push(element);
            element[REMOVAL_FLAG] =
                { namespaceId, setForRemoval: context, hasAnimation, removedBeforeQueried: false };
        }
        /**
         * @param {?} namespaceId
         * @param {?} element
         * @param {?} name
         * @param {?} phase
         * @param {?} callback
         * @return {?}
         */
        listen(namespaceId, element, name, phase, callback) {
            if (isElementNode(element)) {
                return this._fetchNamespace(namespaceId).listen(element, name, phase, callback);
            }
            return (/**
             * @return {?}
             */
            () => { });
        }
        /**
         * @private
         * @param {?} entry
         * @param {?} subTimelines
         * @param {?} enterClassName
         * @param {?} leaveClassName
         * @param {?=} skipBuildAst
         * @return {?}
         */
        _buildInstruction(entry, subTimelines, enterClassName, leaveClassName, skipBuildAst) {
            return entry.transition.build(this.driver, entry.element, entry.fromState.value, entry.toState.value, enterClassName, leaveClassName, entry.fromState.options, entry.toState.options, subTimelines, skipBuildAst);
        }
        /**
         * @param {?} containerElement
         * @return {?}
         */
        destroyInnerAnimations(containerElement) {
            /** @type {?} */
            let elements = this.driver.query(containerElement, NG_TRIGGER_SELECTOR, true);
            elements.forEach((/**
             * @param {?} element
             * @return {?}
             */
            element => this.destroyActiveAnimationsForElement(element)));
            if (this.playersByQueriedElement.size == 0)
                return;
            elements = this.driver.query(containerElement, NG_ANIMATING_SELECTOR, true);
            elements.forEach((/**
             * @param {?} element
             * @return {?}
             */
            element => this.finishActiveQueriedAnimationOnElement(element)));
        }
        /**
         * @param {?} element
         * @return {?}
         */
        destroyActiveAnimationsForElement(element) {
            /** @type {?} */
            const players = this.playersByElement.get(element);
            if (players) {
                players.forEach((/**
                 * @param {?} player
                 * @return {?}
                 */
                player => {
                    // special case for when an element is set for destruction, but hasn't started.
                    // in this situation we want to delay the destruction until the flush occurs
                    // so that any event listeners attached to the player are triggered.
                    if (player.queued) {
                        player.markedForDestroy = true;
                    }
                    else {
                        player.destroy();
                    }
                }));
            }
        }
        /**
         * @param {?} element
         * @return {?}
         */
        finishActiveQueriedAnimationOnElement(element) {
            /** @type {?} */
            const players = this.playersByQueriedElement.get(element);
            if (players) {
                players.forEach((/**
                 * @param {?} player
                 * @return {?}
                 */
                player => player.finish()));
            }
        }
        /**
         * @return {?}
         */
        whenRenderingDone() {
            return new Promise((/**
             * @param {?} resolve
             * @return {?}
             */
            resolve => {
                if (this.players.length) {
                    return optimizeGroupPlayer(this.players).onDone((/**
                     * @return {?}
                     */
                    () => resolve()));
                }
                else {
                    resolve();
                }
            }));
        }
        /**
         * @param {?} element
         * @return {?}
         */
        processLeaveNode(element) {
            /** @type {?} */
            const details = (/** @type {?} */ (element[REMOVAL_FLAG]));
            if (details && details.setForRemoval) {
                // this will prevent it from removing it twice
                element[REMOVAL_FLAG] = NULL_REMOVAL_STATE;
                if (details.namespaceId) {
                    this.destroyInnerAnimations(element);
                    /** @type {?} */
                    const ns = this._fetchNamespace(details.namespaceId);
                    if (ns) {
                        ns.clearElementCache(element);
                    }
                }
                this._onRemovalComplete(element, details.setForRemoval);
            }
            if (this.driver.matchesElement(element, DISABLED_SELECTOR)) {
                this.markElementAsDisabled(element, false);
            }
            this.driver.query(element, DISABLED_SELECTOR, true).forEach((/**
             * @param {?} node
             * @return {?}
             */
            node => {
                this.markElementAsDisabled(node, false);
            }));
        }
        /**
         * @param {?=} microtaskId
         * @return {?}
         */
        flush(microtaskId = -1) {
            /** @type {?} */
            let players = [];
            if (this.newHostElements.size) {
                this.newHostElements.forEach((/**
                 * @param {?} ns
                 * @param {?} element
                 * @return {?}
                 */
                (ns, element) => this._balanceNamespaceList(ns, element)));
                this.newHostElements.clear();
            }
            if (this.totalAnimations && this.collectedEnterElements.length) {
                for (let i = 0; i < this.collectedEnterElements.length; i++) {
                    /** @type {?} */
                    const elm = this.collectedEnterElements[i];
                    addClass(elm, STAR_CLASSNAME);
                }
            }
            if (this._namespaceList.length &&
                (this.totalQueuedPlayers || this.collectedLeaveElements.length)) {
                /** @type {?} */
                const cleanupFns = [];
                try {
                    players = this._flushAnimations(cleanupFns, microtaskId);
                }
                finally {
                    for (let i = 0; i < cleanupFns.length; i++) {
                        cleanupFns[i]();
                    }
                }
            }
            else {
                for (let i = 0; i < this.collectedLeaveElements.length; i++) {
                    /** @type {?} */
                    const element = this.collectedLeaveElements[i];
                    this.processLeaveNode(element);
                }
            }
            this.totalQueuedPlayers = 0;
            this.collectedEnterElements.length = 0;
            this.collectedLeaveElements.length = 0;
            this._flushFns.forEach((/**
             * @param {?} fn
             * @return {?}
             */
            fn => fn()));
            this._flushFns = [];
            if (this._whenQuietFns.length) {
                // we move these over to a variable so that
                // if any new callbacks are registered in another
                // flush they do not populate the existing set
                /** @type {?} */
                const quietFns = this._whenQuietFns;
                this._whenQuietFns = [];
                if (players.length) {
                    optimizeGroupPlayer(players).onDone((/**
                     * @return {?}
                     */
                    () => {
                        quietFns.forEach((/**
                         * @param {?} fn
                         * @return {?}
                         */
                        fn => fn()));
                    }));
                }
                else {
                    quietFns.forEach((/**
                     * @param {?} fn
                     * @return {?}
                     */
                    fn => fn()));
                }
            }
        }
        /**
         * @param {?} errors
         * @return {?}
         */
        reportError(errors) {
            throw new Error(`Unable to process animations due to the following failed trigger transitions\n ${errors.join('\n')}`);
        }
        /**
         * @private
         * @param {?} cleanupFns
         * @param {?} microtaskId
         * @return {?}
         */
        _flushAnimations(cleanupFns, microtaskId) {
            /** @type {?} */
            const subTimelines = new ElementInstructionMap();
            /** @type {?} */
            const skippedPlayers = [];
            /** @type {?} */
            const skippedPlayersMap = new Map();
            /** @type {?} */
            const queuedInstructions = [];
            /** @type {?} */
            const queriedElements = new Map();
            /** @type {?} */
            const allPreStyleElements = new Map();
            /** @type {?} */
            const allPostStyleElements = new Map();
            /** @type {?} */
            const disabledElementsSet = new Set();
            this.disabledNodes.forEach((/**
             * @param {?} node
             * @return {?}
             */
            node => {
                disabledElementsSet.add(node);
                /** @type {?} */
                const nodesThatAreDisabled = this.driver.query(node, QUEUED_SELECTOR, true);
                for (let i = 0; i < nodesThatAreDisabled.length; i++) {
                    disabledElementsSet.add(nodesThatAreDisabled[i]);
                }
            }));
            /** @type {?} */
            const bodyNode = this.bodyNode;
            /** @type {?} */
            const allTriggerElements = Array.from(this.statesByElement.keys());
            /** @type {?} */
            const enterNodeMap = buildRootMap(allTriggerElements, this.collectedEnterElements);
            // this must occur before the instructions are built below such that
            // the :enter queries match the elements (since the timeline queries
            // are fired during instruction building).
            /** @type {?} */
            const enterNodeMapIds = new Map();
            /** @type {?} */
            let i = 0;
            enterNodeMap.forEach((/**
             * @param {?} nodes
             * @param {?} root
             * @return {?}
             */
            (nodes, root) => {
                /** @type {?} */
                const className = ENTER_CLASSNAME + i++;
                enterNodeMapIds.set(root, className);
                nodes.forEach((/**
                 * @param {?} node
                 * @return {?}
                 */
                node => addClass(node, className)));
            }));
            /** @type {?} */
            const allLeaveNodes = [];
            /** @type {?} */
            const mergedLeaveNodes = new Set();
            /** @type {?} */
            const leaveNodesWithoutAnimations = new Set();
            for (let i = 0; i < this.collectedLeaveElements.length; i++) {
                /** @type {?} */
                const element = this.collectedLeaveElements[i];
                /** @type {?} */
                const details = (/** @type {?} */ (element[REMOVAL_FLAG]));
                if (details && details.setForRemoval) {
                    allLeaveNodes.push(element);
                    mergedLeaveNodes.add(element);
                    if (details.hasAnimation) {
                        this.driver.query(element, STAR_SELECTOR, true).forEach((/**
                         * @param {?} elm
                         * @return {?}
                         */
                        elm => mergedLeaveNodes.add(elm)));
                    }
                    else {
                        leaveNodesWithoutAnimations.add(element);
                    }
                }
            }
            /** @type {?} */
            const leaveNodeMapIds = new Map();
            /** @type {?} */
            const leaveNodeMap = buildRootMap(allTriggerElements, Array.from(mergedLeaveNodes));
            leaveNodeMap.forEach((/**
             * @param {?} nodes
             * @param {?} root
             * @return {?}
             */
            (nodes, root) => {
                /** @type {?} */
                const className = LEAVE_CLASSNAME + i++;
                leaveNodeMapIds.set(root, className);
                nodes.forEach((/**
                 * @param {?} node
                 * @return {?}
                 */
                node => addClass(node, className)));
            }));
            cleanupFns.push((/**
             * @return {?}
             */
            () => {
                enterNodeMap.forEach((/**
                 * @param {?} nodes
                 * @param {?} root
                 * @return {?}
                 */
                (nodes, root) => {
                    /** @type {?} */
                    const className = (/** @type {?} */ (enterNodeMapIds.get(root)));
                    nodes.forEach((/**
                     * @param {?} node
                     * @return {?}
                     */
                    node => removeClass(node, className)));
                }));
                leaveNodeMap.forEach((/**
                 * @param {?} nodes
                 * @param {?} root
                 * @return {?}
                 */
                (nodes, root) => {
                    /** @type {?} */
                    const className = (/** @type {?} */ (leaveNodeMapIds.get(root)));
                    nodes.forEach((/**
                     * @param {?} node
                     * @return {?}
                     */
                    node => removeClass(node, className)));
                }));
                allLeaveNodes.forEach((/**
                 * @param {?} element
                 * @return {?}
                 */
                element => {
                    this.processLeaveNode(element);
                }));
            }));
            /** @type {?} */
            const allPlayers = [];
            /** @type {?} */
            const erroneousTransitions = [];
            for (let i = this._namespaceList.length - 1; i >= 0; i--) {
                /** @type {?} */
                const ns = this._namespaceList[i];
                ns.drainQueuedTransitions(microtaskId).forEach((/**
                 * @param {?} entry
                 * @return {?}
                 */
                entry => {
                    /** @type {?} */
                    const player = entry.player;
                    /** @type {?} */
                    const element = entry.element;
                    allPlayers.push(player);
                    if (this.collectedEnterElements.length) {
                        /** @type {?} */
                        const details = (/** @type {?} */ (element[REMOVAL_FLAG]));
                        // move animations are currently not supported...
                        if (details && details.setForMove) {
                            player.destroy();
                            return;
                        }
                    }
                    /** @type {?} */
                    const nodeIsOrphaned = !bodyNode || !this.driver.containsElement(bodyNode, element);
                    /** @type {?} */
                    const leaveClassName = (/** @type {?} */ (leaveNodeMapIds.get(element)));
                    /** @type {?} */
                    const enterClassName = (/** @type {?} */ (enterNodeMapIds.get(element)));
                    /** @type {?} */
                    const instruction = (/** @type {?} */ (this._buildInstruction(entry, subTimelines, enterClassName, leaveClassName, nodeIsOrphaned)));
                    if (instruction.errors && instruction.errors.length) {
                        erroneousTransitions.push(instruction);
                        return;
                    }
                    // even though the element may not be apart of the DOM, it may
                    // still be added at a later point (due to the mechanics of content
                    // projection and/or dynamic component insertion) therefore it's
                    // important we still style the element.
                    if (nodeIsOrphaned) {
                        player.onStart((/**
                         * @return {?}
                         */
                        () => eraseStyles(element, instruction.fromStyles)));
                        player.onDestroy((/**
                         * @return {?}
                         */
                        () => setStyles(element, instruction.toStyles)));
                        skippedPlayers.push(player);
                        return;
                    }
                    // if a unmatched transition is queued to go then it SHOULD NOT render
                    // an animation and cancel the previously running animations.
                    if (entry.isFallbackTransition) {
                        player.onStart((/**
                         * @return {?}
                         */
                        () => eraseStyles(element, instruction.fromStyles)));
                        player.onDestroy((/**
                         * @return {?}
                         */
                        () => setStyles(element, instruction.toStyles)));
                        skippedPlayers.push(player);
                        return;
                    }
                    // this means that if a parent animation uses this animation as a sub trigger
                    // then it will instruct the timeline builder to not add a player delay, but
                    // instead stretch the first keyframe gap up until the animation starts. The
                    // reason this is important is to prevent extra initialization styles from being
                    // required by the user in the animation.
                    instruction.timelines.forEach((/**
                     * @param {?} tl
                     * @return {?}
                     */
                    tl => tl.stretchStartingKeyframe = true));
                    subTimelines.append(element, instruction.timelines);
                    /** @type {?} */
                    const tuple = { instruction, player, element };
                    queuedInstructions.push(tuple);
                    instruction.queriedElements.forEach((/**
                     * @param {?} element
                     * @return {?}
                     */
                    element => getOrSetAsInMap(queriedElements, element, []).push(player)));
                    instruction.preStyleProps.forEach((/**
                     * @param {?} stringMap
                     * @param {?} element
                     * @return {?}
                     */
                    (stringMap, element) => {
                        /** @type {?} */
                        const props = Object.keys(stringMap);
                        if (props.length) {
                            /** @type {?} */
                            let setVal = (/** @type {?} */ (allPreStyleElements.get(element)));
                            if (!setVal) {
                                allPreStyleElements.set(element, setVal = new Set());
                            }
                            props.forEach((/**
                             * @param {?} prop
                             * @return {?}
                             */
                            prop => setVal.add(prop)));
                        }
                    }));
                    instruction.postStyleProps.forEach((/**
                     * @param {?} stringMap
                     * @param {?} element
                     * @return {?}
                     */
                    (stringMap, element) => {
                        /** @type {?} */
                        const props = Object.keys(stringMap);
                        /** @type {?} */
                        let setVal = (/** @type {?} */ (allPostStyleElements.get(element)));
                        if (!setVal) {
                            allPostStyleElements.set(element, setVal = new Set());
                        }
                        props.forEach((/**
                         * @param {?} prop
                         * @return {?}
                         */
                        prop => setVal.add(prop)));
                    }));
                }));
            }
            if (erroneousTransitions.length) {
                /** @type {?} */
                const errors = [];
                erroneousTransitions.forEach((/**
                 * @param {?} instruction
                 * @return {?}
                 */
                instruction => {
                    errors.push(`@${instruction.triggerName} has failed due to:\n`);
                    (/** @type {?} */ (instruction.errors)).forEach((/**
                     * @param {?} error
                     * @return {?}
                     */
                    error => errors.push(`- ${error}\n`)));
                }));
                allPlayers.forEach((/**
                 * @param {?} player
                 * @return {?}
                 */
                player => player.destroy()));
                this.reportError(errors);
            }
            /** @type {?} */
            const allPreviousPlayersMap = new Map();
            // this map works to tell which element in the DOM tree is contained by
            // which animation. Further down below this map will get populated once
            // the players are built and in doing so it can efficiently figure out
            // if a sub player is skipped due to a parent player having priority.
            /** @type {?} */
            const animationElementMap = new Map();
            queuedInstructions.forEach((/**
             * @param {?} entry
             * @return {?}
             */
            entry => {
                /** @type {?} */
                const element = entry.element;
                if (subTimelines.has(element)) {
                    animationElementMap.set(element, element);
                    this._beforeAnimationBuild(entry.player.namespaceId, entry.instruction, allPreviousPlayersMap);
                }
            }));
            skippedPlayers.forEach((/**
             * @param {?} player
             * @return {?}
             */
            player => {
                /** @type {?} */
                const element = player.element;
                /** @type {?} */
                const previousPlayers = this._getPreviousPlayers(element, false, player.namespaceId, player.triggerName, null);
                previousPlayers.forEach((/**
                 * @param {?} prevPlayer
                 * @return {?}
                 */
                prevPlayer => {
                    getOrSetAsInMap(allPreviousPlayersMap, element, []).push(prevPlayer);
                    prevPlayer.destroy();
                }));
            }));
            // this is a special case for nodes that will be removed (either by)
            // having their own leave animations or by being queried in a container
            // that will be removed once a parent animation is complete. The idea
            // here is that * styles must be identical to ! styles because of
            // backwards compatibility (* is also filled in by default in many places).
            // Otherwise * styles will return an empty value or auto since the element
            // that is being getComputedStyle'd will not be visible (since * = destination)
            /** @type {?} */
            const replaceNodes = allLeaveNodes.filter((/**
             * @param {?} node
             * @return {?}
             */
            node => {
                return replacePostStylesAsPre(node, allPreStyleElements, allPostStyleElements);
            }));
            // POST STAGE: fill the * styles
            /** @type {?} */
            const postStylesMap = new Map();
            /** @type {?} */
            const allLeaveQueriedNodes = cloakAndComputeStyles(postStylesMap, this.driver, leaveNodesWithoutAnimations, allPostStyleElements, AUTO_STYLE);
            allLeaveQueriedNodes.forEach((/**
             * @param {?} node
             * @return {?}
             */
            node => {
                if (replacePostStylesAsPre(node, allPreStyleElements, allPostStyleElements)) {
                    replaceNodes.push(node);
                }
            }));
            // PRE STAGE: fill the ! styles
            /** @type {?} */
            const preStylesMap = new Map();
            enterNodeMap.forEach((/**
             * @param {?} nodes
             * @param {?} root
             * @return {?}
             */
            (nodes, root) => {
                cloakAndComputeStyles(preStylesMap, this.driver, new Set(nodes), allPreStyleElements, ɵPRE_STYLE);
            }));
            replaceNodes.forEach((/**
             * @param {?} node
             * @return {?}
             */
            node => {
                /** @type {?} */
                const post = postStylesMap.get(node);
                /** @type {?} */
                const pre = preStylesMap.get(node);
                postStylesMap.set(node, (/** @type {?} */ (Object.assign(Object.assign({}, post), pre))));
            }));
            /** @type {?} */
            const rootPlayers = [];
            /** @type {?} */
            const subPlayers = [];
            /** @type {?} */
            const NO_PARENT_ANIMATION_ELEMENT_DETECTED = {};
            queuedInstructions.forEach((/**
             * @param {?} entry
             * @return {?}
             */
            entry => {
                const { element, player, instruction } = entry;
                // this means that it was never consumed by a parent animation which
                // means that it is independent and therefore should be set for animation
                if (subTimelines.has(element)) {
                    if (disabledElementsSet.has(element)) {
                        player.onDestroy((/**
                         * @return {?}
                         */
                        () => setStyles(element, instruction.toStyles)));
                        player.disabled = true;
                        player.overrideTotalTime(instruction.totalTime);
                        skippedPlayers.push(player);
                        return;
                    }
                    // this will flow up the DOM and query the map to figure out
                    // if a parent animation has priority over it. In the situation
                    // that a parent is detected then it will cancel the loop. If
                    // nothing is detected, or it takes a few hops to find a parent,
                    // then it will fill in the missing nodes and signal them as having
                    // a detected parent (or a NO_PARENT value via a special constant).
                    /** @type {?} */
                    let parentWithAnimation = NO_PARENT_ANIMATION_ELEMENT_DETECTED;
                    if (animationElementMap.size > 1) {
                        /** @type {?} */
                        let elm = element;
                        /** @type {?} */
                        const parentsToAdd = [];
                        while (elm = elm.parentNode) {
                            /** @type {?} */
                            const detectedParent = animationElementMap.get(elm);
                            if (detectedParent) {
                                parentWithAnimation = detectedParent;
                                break;
                            }
                            parentsToAdd.push(elm);
                        }
                        parentsToAdd.forEach((/**
                         * @param {?} parent
                         * @return {?}
                         */
                        parent => animationElementMap.set(parent, parentWithAnimation)));
                    }
                    /** @type {?} */
                    const innerPlayer = this._buildAnimation(player.namespaceId, instruction, allPreviousPlayersMap, skippedPlayersMap, preStylesMap, postStylesMap);
                    player.setRealPlayer(innerPlayer);
                    if (parentWithAnimation === NO_PARENT_ANIMATION_ELEMENT_DETECTED) {
                        rootPlayers.push(player);
                    }
                    else {
                        /** @type {?} */
                        const parentPlayers = this.playersByElement.get(parentWithAnimation);
                        if (parentPlayers && parentPlayers.length) {
                            player.parentPlayer = optimizeGroupPlayer(parentPlayers);
                        }
                        skippedPlayers.push(player);
                    }
                }
                else {
                    eraseStyles(element, instruction.fromStyles);
                    player.onDestroy((/**
                     * @return {?}
                     */
                    () => setStyles(element, instruction.toStyles)));
                    // there still might be a ancestor player animating this
                    // element therefore we will still add it as a sub player
                    // even if its animation may be disabled
                    subPlayers.push(player);
                    if (disabledElementsSet.has(element)) {
                        skippedPlayers.push(player);
                    }
                }
            }));
            // find all of the sub players' corresponding inner animation player
            subPlayers.forEach((/**
             * @param {?} player
             * @return {?}
             */
            player => {
                // even if any players are not found for a sub animation then it
                // will still complete itself after the next tick since it's Noop
                /** @type {?} */
                const playersForElement = skippedPlayersMap.get(player.element);
                if (playersForElement && playersForElement.length) {
                    /** @type {?} */
                    const innerPlayer = optimizeGroupPlayer(playersForElement);
                    player.setRealPlayer(innerPlayer);
                }
            }));
            // the reason why we don't actually play the animation is
            // because all that a skipped player is designed to do is to
            // fire the start/done transition callback events
            skippedPlayers.forEach((/**
             * @param {?} player
             * @return {?}
             */
            player => {
                if (player.parentPlayer) {
                    player.syncPlayerEvents(player.parentPlayer);
                }
                else {
                    player.destroy();
                }
            }));
            // run through all of the queued removals and see if they
            // were picked up by a query. If not then perform the removal
            // operation right away unless a parent animation is ongoing.
            for (let i = 0; i < allLeaveNodes.length; i++) {
                /** @type {?} */
                const element = allLeaveNodes[i];
                /** @type {?} */
                const details = (/** @type {?} */ (element[REMOVAL_FLAG]));
                removeClass(element, LEAVE_CLASSNAME);
                // this means the element has a removal animation that is being
                // taken care of and therefore the inner elements will hang around
                // until that animation is over (or the parent queried animation)
                if (details && details.hasAnimation)
                    continue;
                /** @type {?} */
                let players = [];
                // if this element is queried or if it contains queried children
                // then we want for the element not to be removed from the page
                // until the queried animations have finished
                if (queriedElements.size) {
                    /** @type {?} */
                    let queriedPlayerResults = queriedElements.get(element);
                    if (queriedPlayerResults && queriedPlayerResults.length) {
                        players.push(...queriedPlayerResults);
                    }
                    /** @type {?} */
                    let queriedInnerElements = this.driver.query(element, NG_ANIMATING_SELECTOR, true);
                    for (let j = 0; j < queriedInnerElements.length; j++) {
                        /** @type {?} */
                        let queriedPlayers = queriedElements.get(queriedInnerElements[j]);
                        if (queriedPlayers && queriedPlayers.length) {
                            players.push(...queriedPlayers);
                        }
                    }
                }
                /** @type {?} */
                const activePlayers = players.filter((/**
                 * @param {?} p
                 * @return {?}
                 */
                p => !p.destroyed));
                if (activePlayers.length) {
                    removeNodesAfterAnimationDone(this, element, activePlayers);
                }
                else {
                    this.processLeaveNode(element);
                }
            }
            // this is required so the cleanup method doesn't remove them
            allLeaveNodes.length = 0;
            rootPlayers.forEach((/**
             * @param {?} player
             * @return {?}
             */
            player => {
                this.players.push(player);
                player.onDone((/**
                 * @return {?}
                 */
                () => {
                    player.destroy();
                    /** @type {?} */
                    const index = this.players.indexOf(player);
                    this.players.splice(index, 1);
                }));
                player.play();
            }));
            return rootPlayers;
        }
        /**
         * @param {?} namespaceId
         * @param {?} element
         * @return {?}
         */
        elementContainsData(namespaceId, element) {
            /** @type {?} */
            let containsData = false;
            /** @type {?} */
            const details = (/** @type {?} */ (element[REMOVAL_FLAG]));
            if (details && details.setForRemoval)
                containsData = true;
            if (this.playersByElement.has(element))
                containsData = true;
            if (this.playersByQueriedElement.has(element))
                containsData = true;
            if (this.statesByElement.has(element))
                containsData = true;
            return this._fetchNamespace(namespaceId).elementContainsData(element) || containsData;
        }
        /**
         * @param {?} callback
         * @return {?}
         */
        afterFlush(callback) {
            this._flushFns.push(callback);
        }
        /**
         * @param {?} callback
         * @return {?}
         */
        afterFlushAnimationsDone(callback) {
            this._whenQuietFns.push(callback);
        }
        /**
         * @private
         * @param {?} element
         * @param {?} isQueriedElement
         * @param {?=} namespaceId
         * @param {?=} triggerName
         * @param {?=} toStateValue
         * @return {?}
         */
        _getPreviousPlayers(element, isQueriedElement, namespaceId, triggerName, toStateValue) {
            /** @type {?} */
            let players = [];
            if (isQueriedElement) {
                /** @type {?} */
                const queriedElementPlayers = this.playersByQueriedElement.get(element);
                if (queriedElementPlayers) {
                    players = queriedElementPlayers;
                }
            }
            else {
                /** @type {?} */
                const elementPlayers = this.playersByElement.get(element);
                if (elementPlayers) {
                    /** @type {?} */
                    const isRemovalAnimation = !toStateValue || toStateValue == VOID_VALUE;
                    elementPlayers.forEach((/**
                     * @param {?} player
                     * @return {?}
                     */
                    player => {
                        if (player.queued)
                            return;
                        if (!isRemovalAnimation && player.triggerName != triggerName)
                            return;
                        players.push(player);
                    }));
                }
            }
            if (namespaceId || triggerName) {
                players = players.filter((/**
                 * @param {?} player
                 * @return {?}
                 */
                player => {
                    if (namespaceId && namespaceId != player.namespaceId)
                        return false;
                    if (triggerName && triggerName != player.triggerName)
                        return false;
                    return true;
                }));
            }
            return players;
        }
        /**
         * @private
         * @param {?} namespaceId
         * @param {?} instruction
         * @param {?} allPreviousPlayersMap
         * @return {?}
         */
        _beforeAnimationBuild(namespaceId, instruction, allPreviousPlayersMap) {
            /** @type {?} */
            const triggerName = instruction.triggerName;
            /** @type {?} */
            const rootElement = instruction.element;
            // when a removal animation occurs, ALL previous players are collected
            // and destroyed (even if they are outside of the current namespace)
            /** @type {?} */
            const targetNameSpaceId = instruction.isRemovalTransition ? undefined : namespaceId;
            /** @type {?} */
            const targetTriggerName = instruction.isRemovalTransition ? undefined : triggerName;
            for (const timelineInstruction of instruction.timelines) {
                /** @type {?} */
                const element = timelineInstruction.element;
                /** @type {?} */
                const isQueriedElement = element !== rootElement;
                /** @type {?} */
                const players = getOrSetAsInMap(allPreviousPlayersMap, element, []);
                /** @type {?} */
                const previousPlayers = this._getPreviousPlayers(element, isQueriedElement, targetNameSpaceId, targetTriggerName, instruction.toState);
                previousPlayers.forEach((/**
                 * @param {?} player
                 * @return {?}
                 */
                player => {
                    /** @type {?} */
                    const realPlayer = (/** @type {?} */ (((/** @type {?} */ (player))).getRealPlayer()));
                    if (realPlayer.beforeDestroy) {
                        realPlayer.beforeDestroy();
                    }
                    player.destroy();
                    players.push(player);
                }));
            }
            // this needs to be done so that the PRE/POST styles can be
            // computed properly without interfering with the previous animation
            eraseStyles(rootElement, instruction.fromStyles);
        }
        /**
         * @private
         * @param {?} namespaceId
         * @param {?} instruction
         * @param {?} allPreviousPlayersMap
         * @param {?} skippedPlayersMap
         * @param {?} preStylesMap
         * @param {?} postStylesMap
         * @return {?}
         */
        _buildAnimation(namespaceId, instruction, allPreviousPlayersMap, skippedPlayersMap, preStylesMap, postStylesMap) {
            /** @type {?} */
            const triggerName = instruction.triggerName;
            /** @type {?} */
            const rootElement = instruction.element;
            // we first run this so that the previous animation player
            // data can be passed into the successive animation players
            /** @type {?} */
            const allQueriedPlayers = [];
            /** @type {?} */
            const allConsumedElements = new Set();
            /** @type {?} */
            const allSubElements = new Set();
            /** @type {?} */
            const allNewPlayers = instruction.timelines.map((/**
             * @param {?} timelineInstruction
             * @return {?}
             */
            timelineInstruction => {
                /** @type {?} */
                const element = timelineInstruction.element;
                allConsumedElements.add(element);
                // FIXME (matsko): make sure to-be-removed animations are removed properly
                /** @type {?} */
                const details = element[REMOVAL_FLAG];
                if (details && details.removedBeforeQueried)
                    return new NoopAnimationPlayer(timelineInstruction.duration, timelineInstruction.delay);
                /** @type {?} */
                const isQueriedElement = element !== rootElement;
                /** @type {?} */
                const previousPlayers = flattenGroupPlayers((allPreviousPlayersMap.get(element) || EMPTY_PLAYER_ARRAY)
                    .map((/**
                 * @param {?} p
                 * @return {?}
                 */
                p => p.getRealPlayer())))
                    .filter((/**
                 * @param {?} p
                 * @return {?}
                 */
                p => {
                    // the `element` is not apart of the AnimationPlayer definition, but
                    // Mock/WebAnimations
                    // use the element within their implementation. This will be added in Angular5 to
                    // AnimationPlayer
                    /** @type {?} */
                    const pp = (/** @type {?} */ (p));
                    return pp.element ? pp.element === element : false;
                }));
                /** @type {?} */
                const preStyles = preStylesMap.get(element);
                /** @type {?} */
                const postStyles = postStylesMap.get(element);
                /** @type {?} */
                const keyframes = normalizeKeyframes(this.driver, this._normalizer, element, timelineInstruction.keyframes, preStyles, postStyles);
                /** @type {?} */
                const player = this._buildPlayer(timelineInstruction, keyframes, previousPlayers);
                // this means that this particular player belongs to a sub trigger. It is
                // important that we match this player up with the corresponding (@trigger.listener)
                if (timelineInstruction.subTimeline && skippedPlayersMap) {
                    allSubElements.add(element);
                }
                if (isQueriedElement) {
                    /** @type {?} */
                    const wrappedPlayer = new TransitionAnimationPlayer(namespaceId, triggerName, element);
                    wrappedPlayer.setRealPlayer(player);
                    allQueriedPlayers.push(wrappedPlayer);
                }
                return player;
            }));
            allQueriedPlayers.forEach((/**
             * @param {?} player
             * @return {?}
             */
            player => {
                getOrSetAsInMap(this.playersByQueriedElement, player.element, []).push(player);
                player.onDone((/**
                 * @return {?}
                 */
                () => deleteOrUnsetInMap(this.playersByQueriedElement, player.element, player)));
            }));
            allConsumedElements.forEach((/**
             * @param {?} element
             * @return {?}
             */
            element => addClass(element, NG_ANIMATING_CLASSNAME)));
            /** @type {?} */
            const player = optimizeGroupPlayer(allNewPlayers);
            player.onDestroy((/**
             * @return {?}
             */
            () => {
                allConsumedElements.forEach((/**
                 * @param {?} element
                 * @return {?}
                 */
                element => removeClass(element, NG_ANIMATING_CLASSNAME)));
                setStyles(rootElement, instruction.toStyles);
            }));
            // this basically makes all of the callbacks for sub element animations
            // be dependent on the upper players for when they finish
            allSubElements.forEach((/**
             * @param {?} element
             * @return {?}
             */
            element => {
                getOrSetAsInMap(skippedPlayersMap, element, []).push(player);
            }));
            return player;
        }
        /**
         * @private
         * @param {?} instruction
         * @param {?} keyframes
         * @param {?} previousPlayers
         * @return {?}
         */
        _buildPlayer(instruction, keyframes, previousPlayers) {
            if (keyframes.length > 0) {
                return this.driver.animate(instruction.element, keyframes, instruction.duration, instruction.delay, instruction.easing, previousPlayers);
            }
            // special case for when an empty transition|definition is provided
            // ... there is no point in rendering an empty animation
            return new NoopAnimationPlayer(instruction.duration, instruction.delay);
        }
    }
    class TransitionAnimationPlayer {
        /**
         * @param {?} namespaceId
         * @param {?} triggerName
         * @param {?} element
         */
        constructor(namespaceId, triggerName, element) {
            this.namespaceId = namespaceId;
            this.triggerName = triggerName;
            this.element = element;
            this._player = new NoopAnimationPlayer();
            this._containsRealPlayer = false;
            this._queuedCallbacks = {};
            this.destroyed = false;
            this.markedForDestroy = false;
            this.disabled = false;
            this.queued = true;
            this.totalTime = 0;
        }
        /**
         * @param {?} player
         * @return {?}
         */
        setRealPlayer(player) {
            if (this._containsRealPlayer)
                return;
            this._player = player;
            Object.keys(this._queuedCallbacks).forEach((/**
             * @param {?} phase
             * @return {?}
             */
            phase => {
                this._queuedCallbacks[phase].forEach((/**
                 * @param {?} callback
                 * @return {?}
                 */
                callback => listenOnPlayer(player, phase, undefined, callback)));
            }));
            this._queuedCallbacks = {};
            this._containsRealPlayer = true;
            this.overrideTotalTime(player.totalTime);
            ((/** @type {?} */ (this))).queued = false;
        }
        /**
         * @return {?}
         */
        getRealPlayer() {
            return this._player;
        }
        /**
         * @param {?} totalTime
         * @return {?}
         */
        overrideTotalTime(totalTime) {
            ((/** @type {?} */ (this))).totalTime = totalTime;
        }
        /**
         * @param {?} player
         * @return {?}
         */
        syncPlayerEvents(player) {
            /** @type {?} */
            const p = (/** @type {?} */ (this._player));
            if (p.triggerCallback) {
                player.onStart((/**
                 * @return {?}
                 */
                () => (/** @type {?} */ (p.triggerCallback))('start')));
            }
            player.onDone((/**
             * @return {?}
             */
            () => this.finish()));
            player.onDestroy((/**
             * @return {?}
             */
            () => this.destroy()));
        }
        /**
         * @private
         * @param {?} name
         * @param {?} callback
         * @return {?}
         */
        _queueEvent(name, callback) {
            getOrSetAsInMap(this._queuedCallbacks, name, []).push(callback);
        }
        /**
         * @param {?} fn
         * @return {?}
         */
        onDone(fn) {
            if (this.queued) {
                this._queueEvent('done', fn);
            }
            this._player.onDone(fn);
        }
        /**
         * @param {?} fn
         * @return {?}
         */
        onStart(fn) {
            if (this.queued) {
                this._queueEvent('start', fn);
            }
            this._player.onStart(fn);
        }
        /**
         * @param {?} fn
         * @return {?}
         */
        onDestroy(fn) {
            if (this.queued) {
                this._queueEvent('destroy', fn);
            }
            this._player.onDestroy(fn);
        }
        /**
         * @return {?}
         */
        init() {
            this._player.init();
        }
        /**
         * @return {?}
         */
        hasStarted() {
            return this.queued ? false : this._player.hasStarted();
        }
        /**
         * @return {?}
         */
        play() {
            !this.queued && this._player.play();
        }
        /**
         * @return {?}
         */
        pause() {
            !this.queued && this._player.pause();
        }
        /**
         * @return {?}
         */
        restart() {
            !this.queued && this._player.restart();
        }
        /**
         * @return {?}
         */
        finish() {
            this._player.finish();
        }
        /**
         * @return {?}
         */
        destroy() {
            ((/** @type {?} */ (this))).destroyed = true;
            this._player.destroy();
        }
        /**
         * @return {?}
         */
        reset() {
            !this.queued && this._player.reset();
        }
        /**
         * @param {?} p
         * @return {?}
         */
        setPosition(p) {
            if (!this.queued) {
                this._player.setPosition(p);
            }
        }
        /**
         * @return {?}
         */
        getPosition() {
            return this.queued ? 0 : this._player.getPosition();
        }
        /**
         * \@internal
         * @param {?} phaseName
         * @return {?}
         */
        triggerCallback(phaseName) {
            /** @type {?} */
            const p = (/** @type {?} */ (this._player));
            if (p.triggerCallback) {
                p.triggerCallback(phaseName);
            }
        }
    }
    /**
     * @param {?} map
     * @param {?} key
     * @param {?} value
     * @return {?}
     */
    function deleteOrUnsetInMap(map, key, value) {
        /** @type {?} */
        let currentValues;
        if (map instanceof Map) {
            currentValues = map.get(key);
            if (currentValues) {
                if (currentValues.length) {
                    /** @type {?} */
                    const index = currentValues.indexOf(value);
                    currentValues.splice(index, 1);
                }
                if (currentValues.length == 0) {
                    map.delete(key);
                }
            }
        }
        else {
            currentValues = map[key];
            if (currentValues) {
                if (currentValues.length) {
                    /** @type {?} */
                    const index = currentValues.indexOf(value);
                    currentValues.splice(index, 1);
                }
                if (currentValues.length == 0) {
                    delete map[key];
                }
            }
        }
        return currentValues;
    }
    /**
     * @param {?} value
     * @return {?}
     */
    function normalizeTriggerValue(value) {
        // we use `!= null` here because it's the most simple
        // way to test against a "falsy" value without mixing
        // in empty strings or a zero value. DO NOT OPTIMIZE.
        return value != null ? value : null;
    }
    /**
     * @param {?} node
     * @return {?}
     */
    function isElementNode(node) {
        return node && node['nodeType'] === 1;
    }
    /**
     * @param {?} eventName
     * @return {?}
     */
    function isTriggerEventValid(eventName) {
        return eventName == 'start' || eventName == 'done';
    }
    /**
     * @param {?} element
     * @param {?=} value
     * @return {?}
     */
    function cloakElement(element, value) {
        /** @type {?} */
        const oldValue = element.style.display;
        element.style.display = value != null ? value : 'none';
        return oldValue;
    }
    /**
     * @param {?} valuesMap
     * @param {?} driver
     * @param {?} elements
     * @param {?} elementPropsMap
     * @param {?} defaultStyle
     * @return {?}
     */
    function cloakAndComputeStyles(valuesMap, driver, elements, elementPropsMap, defaultStyle) {
        /** @type {?} */
        const cloakVals = [];
        elements.forEach((/**
         * @param {?} element
         * @return {?}
         */
        element => cloakVals.push(cloakElement(element))));
        /** @type {?} */
        const failedElements = [];
        elementPropsMap.forEach((/**
         * @param {?} props
         * @param {?} element
         * @return {?}
         */
        (props, element) => {
            /** @type {?} */
            const styles = {};
            props.forEach((/**
             * @param {?} prop
             * @return {?}
             */
            prop => {
                /** @type {?} */
                const value = styles[prop] = driver.computeStyle(element, prop, defaultStyle);
                // there is no easy way to detect this because a sub element could be removed
                // by a parent animation element being detached.
                if (!value || value.length == 0) {
                    element[REMOVAL_FLAG] = NULL_REMOVED_QUERIED_STATE;
                    failedElements.push(element);
                }
            }));
            valuesMap.set(element, styles);
        }));
        // we use a index variable here since Set.forEach(a, i) does not return
        // an index value for the closure (but instead just the value)
        /** @type {?} */
        let i = 0;
        elements.forEach((/**
         * @param {?} element
         * @return {?}
         */
        element => cloakElement(element, cloakVals[i++])));
        return failedElements;
    }
    /*
    Since the Angular renderer code will return a collection of inserted
    nodes in all areas of a DOM tree, it's up to this algorithm to figure
    out which nodes are roots for each animation @trigger.

    By placing each inserted node into a Set and traversing upwards, it
    is possible to find the @trigger elements and well any direct *star
    insertion nodes, if a @trigger root is found then the enter element
    is placed into the Map[@trigger] spot.
     */
    /**
     * @param {?} roots
     * @param {?} nodes
     * @return {?}
     */
    function buildRootMap(roots, nodes) {
        /** @type {?} */
        const rootMap = new Map();
        roots.forEach((/**
         * @param {?} root
         * @return {?}
         */
        root => rootMap.set(root, [])));
        if (nodes.length == 0)
            return rootMap;
        /** @type {?} */
        const NULL_NODE = 1;
        /** @type {?} */
        const nodeSet = new Set(nodes);
        /** @type {?} */
        const localRootMap = new Map();
        /**
         * @param {?} node
         * @return {?}
         */
        function getRoot(node) {
            if (!node)
                return NULL_NODE;
            /** @type {?} */
            let root = localRootMap.get(node);
            if (root)
                return root;
            /** @type {?} */
            const parent = node.parentNode;
            if (rootMap.has(parent)) { // ngIf inside @trigger
                root = parent;
            }
            else if (nodeSet.has(parent)) { // ngIf inside ngIf
                root = NULL_NODE;
            }
            else { // recurse upwards
                root = getRoot(parent);
            }
            localRootMap.set(node, root);
            return root;
        }
        nodes.forEach((/**
         * @param {?} node
         * @return {?}
         */
        node => {
            /** @type {?} */
            const root = getRoot(node);
            if (root !== NULL_NODE) {
                (/** @type {?} */ (rootMap.get(root))).push(node);
            }
        }));
        return rootMap;
    }
    /** @type {?} */
    const CLASSES_CACHE_KEY = '$$classes';
    /**
     * @param {?} element
     * @param {?} className
     * @return {?}
     */
    function addClass(element, className) {
        if (element.classList) {
            element.classList.add(className);
        }
        else {
            /** @type {?} */
            let classes = element[CLASSES_CACHE_KEY];
            if (!classes) {
                classes = element[CLASSES_CACHE_KEY] = {};
            }
            classes[className] = true;
        }
    }
    /**
     * @param {?} element
     * @param {?} className
     * @return {?}
     */
    function removeClass(element, className) {
        if (element.classList) {
            element.classList.remove(className);
        }
        else {
            /** @type {?} */
            let classes = element[CLASSES_CACHE_KEY];
            if (classes) {
                delete classes[className];
            }
        }
    }
    /**
     * @param {?} engine
     * @param {?} element
     * @param {?} players
     * @return {?}
     */
    function removeNodesAfterAnimationDone(engine, element, players) {
        optimizeGroupPlayer(players).onDone((/**
         * @return {?}
         */
        () => engine.processLeaveNode(element)));
    }
    /**
     * @param {?} players
     * @return {?}
     */
    function flattenGroupPlayers(players) {
        /** @type {?} */
        const finalPlayers = [];
        _flattenGroupPlayersRecur(players, finalPlayers);
        return finalPlayers;
    }
    /**
     * @param {?} players
     * @param {?} finalPlayers
     * @return {?}
     */
    function _flattenGroupPlayersRecur(players, finalPlayers) {
        for (let i = 0; i < players.length; i++) {
            /** @type {?} */
            const player = players[i];
            if (player instanceof AnimationGroupPlayer) {
                _flattenGroupPlayersRecur(player.players, finalPlayers);
            }
            else {
                finalPlayers.push(player);
            }
        }
    }
    /**
     * @param {?} a
     * @param {?} b
     * @return {?}
     */
    function objEquals(a, b) {
        /** @type {?} */
        const k1 = Object.keys(a);
        /** @type {?} */
        const k2 = Object.keys(b);
        if (k1.length != k2.length)
            return false;
        for (let i = 0; i < k1.length; i++) {
            /** @type {?} */
            const prop = k1[i];
            if (!b.hasOwnProperty(prop) || a[prop] !== b[prop])
                return false;
        }
        return true;
    }
    /**
     * @param {?} element
     * @param {?} allPreStyleElements
     * @param {?} allPostStyleElements
     * @return {?}
     */
    function replacePostStylesAsPre(element, allPreStyleElements, allPostStyleElements) {
        /** @type {?} */
        const postEntry = allPostStyleElements.get(element);
        if (!postEntry)
            return false;
        /** @type {?} */
        let preEntry = allPreStyleElements.get(element);
        if (preEntry) {
            postEntry.forEach((/**
             * @param {?} data
             * @return {?}
             */
            data => (/** @type {?} */ (preEntry)).add(data)));
        }
        else {
            allPreStyleElements.set(element, postEntry);
        }
        allPostStyleElements.delete(element);
        return true;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/animations/browser/src/render/animation_engine_next.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class AnimationEngine {
        /**
         * @param {?} bodyNode
         * @param {?} _driver
         * @param {?} normalizer
         */
        constructor(bodyNode, _driver, normalizer) {
            this.bodyNode = bodyNode;
            this._driver = _driver;
            this._triggerCache = {};
            // this method is designed to be overridden by the code that uses this engine
            this.onRemovalComplete = (/**
             * @param {?} element
             * @param {?} context
             * @return {?}
             */
            (element, context) => { });
            this._transitionEngine = new TransitionAnimationEngine(bodyNode, _driver, normalizer);
            this._timelineEngine = new TimelineAnimationEngine(bodyNode, _driver, normalizer);
            this._transitionEngine.onRemovalComplete = (/**
             * @param {?} element
             * @param {?} context
             * @return {?}
             */
            (element, context) => this.onRemovalComplete(element, context));
        }
        /**
         * @param {?} componentId
         * @param {?} namespaceId
         * @param {?} hostElement
         * @param {?} name
         * @param {?} metadata
         * @return {?}
         */
        registerTrigger(componentId, namespaceId, hostElement, name, metadata) {
            /** @type {?} */
            const cacheKey = componentId + '-' + name;
            /** @type {?} */
            let trigger = this._triggerCache[cacheKey];
            if (!trigger) {
                /** @type {?} */
                const errors = [];
                /** @type {?} */
                const ast = (/** @type {?} */ (buildAnimationAst(this._driver, (/** @type {?} */ (metadata)), errors)));
                if (errors.length) {
                    throw new Error(`The animation trigger "${name}" has failed to build due to the following errors:\n - ${errors.join('\n - ')}`);
                }
                trigger = buildTrigger(name, ast);
                this._triggerCache[cacheKey] = trigger;
            }
            this._transitionEngine.registerTrigger(namespaceId, name, trigger);
        }
        /**
         * @param {?} namespaceId
         * @param {?} hostElement
         * @return {?}
         */
        register(namespaceId, hostElement) {
            this._transitionEngine.register(namespaceId, hostElement);
        }
        /**
         * @param {?} namespaceId
         * @param {?} context
         * @return {?}
         */
        destroy(namespaceId, context) {
            this._transitionEngine.destroy(namespaceId, context);
        }
        /**
         * @param {?} namespaceId
         * @param {?} element
         * @param {?} parent
         * @param {?} insertBefore
         * @return {?}
         */
        onInsert(namespaceId, element, parent, insertBefore) {
            this._transitionEngine.insertNode(namespaceId, element, parent, insertBefore);
        }
        /**
         * @param {?} namespaceId
         * @param {?} element
         * @param {?} context
         * @param {?=} isHostElement
         * @return {?}
         */
        onRemove(namespaceId, element, context, isHostElement) {
            this._transitionEngine.removeNode(namespaceId, element, isHostElement || false, context);
        }
        /**
         * @param {?} element
         * @param {?} disable
         * @return {?}
         */
        disableAnimations(element, disable) {
            this._transitionEngine.markElementAsDisabled(element, disable);
        }
        /**
         * @param {?} namespaceId
         * @param {?} element
         * @param {?} property
         * @param {?} value
         * @return {?}
         */
        process(namespaceId, element, property, value) {
            if (property.charAt(0) == '@') {
                const [id, action] = parseTimelineCommand(property);
                /** @type {?} */
                const args = (/** @type {?} */ (value));
                this._timelineEngine.command(id, element, action, args);
            }
            else {
                this._transitionEngine.trigger(namespaceId, element, property, value);
            }
        }
        /**
         * @param {?} namespaceId
         * @param {?} element
         * @param {?} eventName
         * @param {?} eventPhase
         * @param {?} callback
         * @return {?}
         */
        listen(namespaceId, element, eventName, eventPhase, callback) {
            // @@listen
            if (eventName.charAt(0) == '@') {
                const [id, action] = parseTimelineCommand(eventName);
                return this._timelineEngine.listen(id, element, action, callback);
            }
            return this._transitionEngine.listen(namespaceId, element, eventName, eventPhase, callback);
        }
        /**
         * @param {?=} microtaskId
         * @return {?}
         */
        flush(microtaskId = -1) {
            this._transitionEngine.flush(microtaskId);
        }
        /**
         * @return {?}
         */
        get players() {
            return ((/** @type {?} */ (this._transitionEngine.players)))
                .concat((/** @type {?} */ (this._timelineEngine.players)));
        }
        /**
         * @return {?}
         */
        whenRenderingDone() {
            return this._transitionEngine.whenRenderingDone();
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/animations/browser/src/render/special_cased_styles.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Returns an instance of `SpecialCasedStyles` if and when any special (non animateable) styles are
     * detected.
     *
     * In CSS there exist properties that cannot be animated within a keyframe animation
     * (whether it be via CSS keyframes or web-animations) and the animation implementation
     * will ignore them. This function is designed to detect those special cased styles and
     * return a container that will be executed at the start and end of the animation.
     *
     * @param {?} element
     * @param {?} styles
     * @return {?} an instance of `SpecialCasedStyles` if any special styles are detected otherwise `null`
     */
    function packageNonAnimatableStyles(element, styles) {
        /** @type {?} */
        let startStyles = null;
        /** @type {?} */
        let endStyles = null;
        if (Array.isArray(styles) && styles.length) {
            startStyles = filterNonAnimatableStyles(styles[0]);
            if (styles.length > 1) {
                endStyles = filterNonAnimatableStyles(styles[styles.length - 1]);
            }
        }
        else if (styles) {
            startStyles = filterNonAnimatableStyles(styles);
        }
        return (startStyles || endStyles) ? new SpecialCasedStyles(element, startStyles, endStyles) :
            null;
    }
    /**
     * Designed to be executed during a keyframe-based animation to apply any special-cased styles.
     *
     * When started (when the `start()` method is run) then the provided `startStyles`
     * will be applied. When finished (when the `finish()` method is called) the
     * `endStyles` will be applied as well any any starting styles. Finally when
     * `destroy()` is called then all styles will be removed.
     */
    class SpecialCasedStyles {
        /**
         * @param {?} _element
         * @param {?} _startStyles
         * @param {?} _endStyles
         */
        constructor(_element, _startStyles, _endStyles) {
            this._element = _element;
            this._startStyles = _startStyles;
            this._endStyles = _endStyles;
            this._state = 0 /* Pending */;
            /** @type {?} */
            let initialStyles = SpecialCasedStyles.initialStylesByElement.get(_element);
            if (!initialStyles) {
                SpecialCasedStyles.initialStylesByElement.set(_element, initialStyles = {});
            }
            this._initialStyles = initialStyles;
        }
        /**
         * @return {?}
         */
        start() {
            if (this._state < 1 /* Started */) {
                if (this._startStyles) {
                    setStyles(this._element, this._startStyles, this._initialStyles);
                }
                this._state = 1 /* Started */;
            }
        }
        /**
         * @return {?}
         */
        finish() {
            this.start();
            if (this._state < 2 /* Finished */) {
                setStyles(this._element, this._initialStyles);
                if (this._endStyles) {
                    setStyles(this._element, this._endStyles);
                    this._endStyles = null;
                }
                this._state = 1 /* Started */;
            }
        }
        /**
         * @return {?}
         */
        destroy() {
            this.finish();
            if (this._state < 3 /* Destroyed */) {
                SpecialCasedStyles.initialStylesByElement.delete(this._element);
                if (this._startStyles) {
                    eraseStyles(this._element, this._startStyles);
                    this._endStyles = null;
                }
                if (this._endStyles) {
                    eraseStyles(this._element, this._endStyles);
                    this._endStyles = null;
                }
                setStyles(this._element, this._initialStyles);
                this._state = 3 /* Destroyed */;
            }
        }
    }
    SpecialCasedStyles.initialStylesByElement = new WeakMap();
    /**
     * @param {?} styles
     * @return {?}
     */
    function filterNonAnimatableStyles(styles) {
        /** @type {?} */
        let result = null;
        /** @type {?} */
        const props = Object.keys(styles);
        for (let i = 0; i < props.length; i++) {
            /** @type {?} */
            const prop = props[i];
            if (isNonAnimatableStyle(prop)) {
                result = result || {};
                result[prop] = styles[prop];
            }
        }
        return result;
    }
    /**
     * @param {?} prop
     * @return {?}
     */
    function isNonAnimatableStyle(prop) {
        return prop === 'display' || prop === 'position';
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/animations/browser/src/render/css_keyframes/element_animation_style_handler.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     * @type {?}
     */
    const ELAPSED_TIME_MAX_DECIMAL_PLACES = 3;
    /** @type {?} */
    const ANIMATION_PROP = 'animation';
    /** @type {?} */
    const ANIMATIONEND_EVENT = 'animationend';
    /** @type {?} */
    const ONE_SECOND$1 = 1000;
    class ElementAnimationStyleHandler {
        /**
         * @param {?} _element
         * @param {?} _name
         * @param {?} _duration
         * @param {?} _delay
         * @param {?} _easing
         * @param {?} _fillMode
         * @param {?} _onDoneFn
         */
        constructor(_element, _name, _duration, _delay, _easing, _fillMode, _onDoneFn) {
            this._element = _element;
            this._name = _name;
            this._duration = _duration;
            this._delay = _delay;
            this._easing = _easing;
            this._fillMode = _fillMode;
            this._onDoneFn = _onDoneFn;
            this._finished = false;
            this._destroyed = false;
            this._startTime = 0;
            this._position = 0;
            this._eventFn = (/**
             * @param {?} e
             * @return {?}
             */
            (e) => this._handleCallback(e));
        }
        /**
         * @return {?}
         */
        apply() {
            applyKeyframeAnimation(this._element, `${this._duration}ms ${this._easing} ${this._delay}ms 1 normal ${this._fillMode} ${this._name}`);
            addRemoveAnimationEvent(this._element, this._eventFn, false);
            this._startTime = Date.now();
        }
        /**
         * @return {?}
         */
        pause() {
            playPauseAnimation(this._element, this._name, 'paused');
        }
        /**
         * @return {?}
         */
        resume() {
            playPauseAnimation(this._element, this._name, 'running');
        }
        /**
         * @param {?} position
         * @return {?}
         */
        setPosition(position) {
            /** @type {?} */
            const index = findIndexForAnimation(this._element, this._name);
            this._position = position * this._duration;
            setAnimationStyle(this._element, 'Delay', `-${this._position}ms`, index);
        }
        /**
         * @return {?}
         */
        getPosition() {
            return this._position;
        }
        /**
         * @private
         * @param {?} event
         * @return {?}
         */
        _handleCallback(event) {
            /** @type {?} */
            const timestamp = event._ngTestManualTimestamp || Date.now();
            /** @type {?} */
            const elapsedTime = parseFloat(event.elapsedTime.toFixed(ELAPSED_TIME_MAX_DECIMAL_PLACES)) * ONE_SECOND$1;
            if (event.animationName == this._name &&
                Math.max(timestamp - this._startTime, 0) >= this._delay && elapsedTime >= this._duration) {
                this.finish();
            }
        }
        /**
         * @return {?}
         */
        finish() {
            if (this._finished)
                return;
            this._finished = true;
            this._onDoneFn();
            addRemoveAnimationEvent(this._element, this._eventFn, true);
        }
        /**
         * @return {?}
         */
        destroy() {
            if (this._destroyed)
                return;
            this._destroyed = true;
            this.finish();
            removeKeyframeAnimation(this._element, this._name);
        }
    }
    /**
     * @param {?} element
     * @param {?} name
     * @param {?} status
     * @return {?}
     */
    function playPauseAnimation(element, name, status) {
        /** @type {?} */
        const index = findIndexForAnimation(element, name);
        setAnimationStyle(element, 'PlayState', status, index);
    }
    /**
     * @param {?} element
     * @param {?} value
     * @return {?}
     */
    function applyKeyframeAnimation(element, value) {
        /** @type {?} */
        const anim = getAnimationStyle(element, '').trim();
        /** @type {?} */
        let index = 0;
        if (anim.length) {
            index = countChars(anim, ',') + 1;
            value = `${anim}, ${value}`;
        }
        setAnimationStyle(element, '', value);
        return index;
    }
    /**
     * @param {?} element
     * @param {?} name
     * @return {?}
     */
    function removeKeyframeAnimation(element, name) {
        /** @type {?} */
        const anim = getAnimationStyle(element, '');
        /** @type {?} */
        const tokens = anim.split(',');
        /** @type {?} */
        const index = findMatchingTokenIndex(tokens, name);
        if (index >= 0) {
            tokens.splice(index, 1);
            /** @type {?} */
            const newValue = tokens.join(',');
            setAnimationStyle(element, '', newValue);
        }
    }
    /**
     * @param {?} element
     * @param {?} value
     * @return {?}
     */
    function findIndexForAnimation(element, value) {
        /** @type {?} */
        const anim = getAnimationStyle(element, '');
        if (anim.indexOf(',') > 0) {
            /** @type {?} */
            const tokens = anim.split(',');
            return findMatchingTokenIndex(tokens, value);
        }
        return findMatchingTokenIndex([anim], value);
    }
    /**
     * @param {?} tokens
     * @param {?} searchToken
     * @return {?}
     */
    function findMatchingTokenIndex(tokens, searchToken) {
        for (let i = 0; i < tokens.length; i++) {
            if (tokens[i].indexOf(searchToken) >= 0) {
                return i;
            }
        }
        return -1;
    }
    /**
     * @param {?} element
     * @param {?} fn
     * @param {?} doRemove
     * @return {?}
     */
    function addRemoveAnimationEvent(element, fn, doRemove) {
        doRemove ? element.removeEventListener(ANIMATIONEND_EVENT, fn) :
            element.addEventListener(ANIMATIONEND_EVENT, fn);
    }
    /**
     * @param {?} element
     * @param {?} name
     * @param {?} value
     * @param {?=} index
     * @return {?}
     */
    function setAnimationStyle(element, name, value, index) {
        /** @type {?} */
        const prop = ANIMATION_PROP + name;
        if (index != null) {
            /** @type {?} */
            const oldValue = element.style[prop];
            if (oldValue.length) {
                /** @type {?} */
                const tokens = oldValue.split(',');
                tokens[index] = value;
                value = tokens.join(',');
            }
        }
        element.style[prop] = value;
    }
    /**
     * @param {?} element
     * @param {?} name
     * @return {?}
     */
    function getAnimationStyle(element, name) {
        return element.style[ANIMATION_PROP + name];
    }
    /**
     * @param {?} value
     * @param {?} char
     * @return {?}
     */
    function countChars(value, char) {
        /** @type {?} */
        let count = 0;
        for (let i = 0; i < value.length; i++) {
            /** @type {?} */
            const c = value.charAt(i);
            if (c === char)
                count++;
        }
        return count;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/animations/browser/src/render/css_keyframes/css_keyframes_player.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const DEFAULT_FILL_MODE = 'forwards';
    /** @type {?} */
    const DEFAULT_EASING = 'linear';
    class CssKeyframesPlayer {
        /**
         * @param {?} element
         * @param {?} keyframes
         * @param {?} animationName
         * @param {?} _duration
         * @param {?} _delay
         * @param {?} easing
         * @param {?} _finalStyles
         * @param {?=} _specialStyles
         */
        constructor(element, keyframes, animationName, _duration, _delay, easing, _finalStyles, _specialStyles) {
            this.element = element;
            this.keyframes = keyframes;
            this.animationName = animationName;
            this._duration = _duration;
            this._delay = _delay;
            this._finalStyles = _finalStyles;
            this._specialStyles = _specialStyles;
            this._onDoneFns = [];
            this._onStartFns = [];
            this._onDestroyFns = [];
            this._started = false;
            this.currentSnapshot = {};
            this._state = 0;
            this.easing = easing || DEFAULT_EASING;
            this.totalTime = _duration + _delay;
            this._buildStyler();
        }
        /**
         * @param {?} fn
         * @return {?}
         */
        onStart(fn) {
            this._onStartFns.push(fn);
        }
        /**
         * @param {?} fn
         * @return {?}
         */
        onDone(fn) {
            this._onDoneFns.push(fn);
        }
        /**
         * @param {?} fn
         * @return {?}
         */
        onDestroy(fn) {
            this._onDestroyFns.push(fn);
        }
        /**
         * @return {?}
         */
        destroy() {
            this.init();
            if (this._state >= 4 /* DESTROYED */)
                return;
            this._state = 4 /* DESTROYED */;
            this._styler.destroy();
            this._flushStartFns();
            this._flushDoneFns();
            if (this._specialStyles) {
                this._specialStyles.destroy();
            }
            this._onDestroyFns.forEach((/**
             * @param {?} fn
             * @return {?}
             */
            fn => fn()));
            this._onDestroyFns = [];
        }
        /**
         * @private
         * @return {?}
         */
        _flushDoneFns() {
            this._onDoneFns.forEach((/**
             * @param {?} fn
             * @return {?}
             */
            fn => fn()));
            this._onDoneFns = [];
        }
        /**
         * @private
         * @return {?}
         */
        _flushStartFns() {
            this._onStartFns.forEach((/**
             * @param {?} fn
             * @return {?}
             */
            fn => fn()));
            this._onStartFns = [];
        }
        /**
         * @return {?}
         */
        finish() {
            this.init();
            if (this._state >= 3 /* FINISHED */)
                return;
            this._state = 3 /* FINISHED */;
            this._styler.finish();
            this._flushStartFns();
            if (this._specialStyles) {
                this._specialStyles.finish();
            }
            this._flushDoneFns();
        }
        /**
         * @param {?} value
         * @return {?}
         */
        setPosition(value) {
            this._styler.setPosition(value);
        }
        /**
         * @return {?}
         */
        getPosition() {
            return this._styler.getPosition();
        }
        /**
         * @return {?}
         */
        hasStarted() {
            return this._state >= 2 /* STARTED */;
        }
        /**
         * @return {?}
         */
        init() {
            if (this._state >= 1 /* INITIALIZED */)
                return;
            this._state = 1 /* INITIALIZED */;
            /** @type {?} */
            const elm = this.element;
            this._styler.apply();
            if (this._delay) {
                this._styler.pause();
            }
        }
        /**
         * @return {?}
         */
        play() {
            this.init();
            if (!this.hasStarted()) {
                this._flushStartFns();
                this._state = 2 /* STARTED */;
                if (this._specialStyles) {
                    this._specialStyles.start();
                }
            }
            this._styler.resume();
        }
        /**
         * @return {?}
         */
        pause() {
            this.init();
            this._styler.pause();
        }
        /**
         * @return {?}
         */
        restart() {
            this.reset();
            this.play();
        }
        /**
         * @return {?}
         */
        reset() {
            this._styler.destroy();
            this._buildStyler();
            this._styler.apply();
        }
        /**
         * @private
         * @return {?}
         */
        _buildStyler() {
            this._styler = new ElementAnimationStyleHandler(this.element, this.animationName, this._duration, this._delay, this.easing, DEFAULT_FILL_MODE, (/**
             * @return {?}
             */
            () => this.finish()));
        }
        /**
         * \@internal
         * @param {?} phaseName
         * @return {?}
         */
        triggerCallback(phaseName) {
            /** @type {?} */
            const methods = phaseName == 'start' ? this._onStartFns : this._onDoneFns;
            methods.forEach((/**
             * @param {?} fn
             * @return {?}
             */
            fn => fn()));
            methods.length = 0;
        }
        /**
         * @return {?}
         */
        beforeDestroy() {
            this.init();
            /** @type {?} */
            const styles = {};
            if (this.hasStarted()) {
                /** @type {?} */
                const finished = this._state >= 3 /* FINISHED */;
                Object.keys(this._finalStyles).forEach((/**
                 * @param {?} prop
                 * @return {?}
                 */
                prop => {
                    if (prop != 'offset') {
                        styles[prop] = finished ? this._finalStyles[prop] : computeStyle(this.element, prop);
                    }
                }));
            }
            this.currentSnapshot = styles;
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/animations/browser/src/render/css_keyframes/direct_style_player.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class DirectStylePlayer extends NoopAnimationPlayer {
        /**
         * @param {?} element
         * @param {?} styles
         */
        constructor(element, styles) {
            super();
            this.element = element;
            this._startingStyles = {};
            this.__initialized = false;
            this._styles = hypenatePropsObject(styles);
        }
        /**
         * @return {?}
         */
        init() {
            if (this.__initialized || !this._startingStyles)
                return;
            this.__initialized = true;
            Object.keys(this._styles).forEach((/**
             * @param {?} prop
             * @return {?}
             */
            prop => {
                (/** @type {?} */ (this._startingStyles))[prop] = this.element.style[prop];
            }));
            super.init();
        }
        /**
         * @return {?}
         */
        play() {
            if (!this._startingStyles)
                return;
            this.init();
            Object.keys(this._styles)
                .forEach((/**
             * @param {?} prop
             * @return {?}
             */
            prop => this.element.style.setProperty(prop, this._styles[prop])));
            super.play();
        }
        /**
         * @return {?}
         */
        destroy() {
            if (!this._startingStyles)
                return;
            Object.keys(this._startingStyles).forEach((/**
             * @param {?} prop
             * @return {?}
             */
            prop => {
                /** @type {?} */
                const value = (/** @type {?} */ (this._startingStyles))[prop];
                if (value) {
                    this.element.style.setProperty(prop, value);
                }
                else {
                    this.element.style.removeProperty(prop);
                }
            }));
            this._startingStyles = null;
            super.destroy();
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/animations/browser/src/render/css_keyframes/css_keyframes_driver.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const KEYFRAMES_NAME_PREFIX = 'gen_css_kf_';
    /** @type {?} */
    const TAB_SPACE = ' ';
    class CssKeyframesDriver {
        constructor() {
            this._count = 0;
            this._head = document.querySelector('head');
            this._warningIssued = false;
        }
        /**
         * @param {?} prop
         * @return {?}
         */
        validateStyleProperty(prop) {
            return validateStyleProperty(prop);
        }
        /**
         * @param {?} element
         * @param {?} selector
         * @return {?}
         */
        matchesElement(element, selector) {
            return matchesElement(element, selector);
        }
        /**
         * @param {?} elm1
         * @param {?} elm2
         * @return {?}
         */
        containsElement(elm1, elm2) {
            return containsElement(elm1, elm2);
        }
        /**
         * @param {?} element
         * @param {?} selector
         * @param {?} multi
         * @return {?}
         */
        query(element, selector, multi) {
            return invokeQuery(element, selector, multi);
        }
        /**
         * @param {?} element
         * @param {?} prop
         * @param {?=} defaultValue
         * @return {?}
         */
        computeStyle(element, prop, defaultValue) {
            return (/** @type {?} */ (((/** @type {?} */ (window.getComputedStyle(element))))[prop]));
        }
        /**
         * @param {?} element
         * @param {?} name
         * @param {?} keyframes
         * @return {?}
         */
        buildKeyframeElement(element, name, keyframes) {
            keyframes = keyframes.map((/**
             * @param {?} kf
             * @return {?}
             */
            kf => hypenatePropsObject(kf)));
            /** @type {?} */
            let keyframeStr = `@keyframes ${name} {\n`;
            /** @type {?} */
            let tab = '';
            keyframes.forEach((/**
             * @param {?} kf
             * @return {?}
             */
            kf => {
                tab = TAB_SPACE;
                /** @type {?} */
                const offset = parseFloat(kf['offset']);
                keyframeStr += `${tab}${offset * 100}% {\n`;
                tab += TAB_SPACE;
                Object.keys(kf).forEach((/**
                 * @param {?} prop
                 * @return {?}
                 */
                prop => {
                    /** @type {?} */
                    const value = kf[prop];
                    switch (prop) {
                        case 'offset':
                            return;
                        case 'easing':
                            if (value) {
                                keyframeStr += `${tab}animation-timing-function: ${value};\n`;
                            }
                            return;
                        default:
                            keyframeStr += `${tab}${prop}: ${value};\n`;
                            return;
                    }
                }));
                keyframeStr += `${tab}}\n`;
            }));
            keyframeStr += `}\n`;
            /** @type {?} */
            const kfElm = document.createElement('style');
            kfElm.innerHTML = keyframeStr;
            return kfElm;
        }
        /**
         * @param {?} element
         * @param {?} keyframes
         * @param {?} duration
         * @param {?} delay
         * @param {?} easing
         * @param {?=} previousPlayers
         * @param {?=} scrubberAccessRequested
         * @return {?}
         */
        animate(element, keyframes, duration, delay, easing, previousPlayers = [], scrubberAccessRequested) {
            if (scrubberAccessRequested) {
                this._notifyFaultyScrubber();
            }
            /** @type {?} */
            const previousCssKeyframePlayers = (/** @type {?} */ (previousPlayers.filter((/**
             * @param {?} player
             * @return {?}
             */
            player => player instanceof CssKeyframesPlayer))));
            /** @type {?} */
            const previousStyles = {};
            if (allowPreviousPlayerStylesMerge(duration, delay)) {
                previousCssKeyframePlayers.forEach((/**
                 * @param {?} player
                 * @return {?}
                 */
                player => {
                    /** @type {?} */
                    let styles = player.currentSnapshot;
                    Object.keys(styles).forEach((/**
                     * @param {?} prop
                     * @return {?}
                     */
                    prop => previousStyles[prop] = styles[prop]));
                }));
            }
            keyframes = balancePreviousStylesIntoKeyframes(element, keyframes, previousStyles);
            /** @type {?} */
            const finalStyles = flattenKeyframesIntoStyles(keyframes);
            // if there is no animation then there is no point in applying
            // styles and waiting for an event to get fired. This causes lag.
            // It's better to just directly apply the styles to the element
            // via the direct styling animation player.
            if (duration == 0) {
                return new DirectStylePlayer(element, finalStyles);
            }
            /** @type {?} */
            const animationName = `${KEYFRAMES_NAME_PREFIX}${this._count++}`;
            /** @type {?} */
            const kfElm = this.buildKeyframeElement(element, animationName, keyframes);
            (/** @type {?} */ (document.querySelector('head'))).appendChild(kfElm);
            /** @type {?} */
            const specialStyles = packageNonAnimatableStyles(element, keyframes);
            /** @type {?} */
            const player = new CssKeyframesPlayer(element, keyframes, animationName, duration, delay, easing, finalStyles, specialStyles);
            player.onDestroy((/**
             * @return {?}
             */
            () => removeElement(kfElm)));
            return player;
        }
        /**
         * @private
         * @return {?}
         */
        _notifyFaultyScrubber() {
            if (!this._warningIssued) {
                console.warn('@angular/animations: please load the web-animations.js polyfill to allow programmatic access...\n', '  visit http://bit.ly/IWukam to learn more about using the web-animation-js polyfill.');
                this._warningIssued = true;
            }
        }
    }
    /**
     * @param {?} keyframes
     * @return {?}
     */
    function flattenKeyframesIntoStyles(keyframes) {
        /** @type {?} */
        let flatKeyframes = {};
        if (keyframes) {
            /** @type {?} */
            const kfs = Array.isArray(keyframes) ? keyframes : [keyframes];
            kfs.forEach((/**
             * @param {?} kf
             * @return {?}
             */
            kf => {
                Object.keys(kf).forEach((/**
                 * @param {?} prop
                 * @return {?}
                 */
                prop => {
                    if (prop == 'offset' || prop == 'easing')
                        return;
                    flatKeyframes[prop] = kf[prop];
                }));
            }));
        }
        return flatKeyframes;
    }
    /**
     * @param {?} node
     * @return {?}
     */
    function removeElement(node) {
        node.parentNode.removeChild(node);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/animations/browser/src/render/web_animations/web_animations_player.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class WebAnimationsPlayer {
        /**
         * @param {?} element
         * @param {?} keyframes
         * @param {?} options
         * @param {?=} _specialStyles
         */
        constructor(element, keyframes, options, _specialStyles) {
            this.element = element;
            this.keyframes = keyframes;
            this.options = options;
            this._specialStyles = _specialStyles;
            this._onDoneFns = [];
            this._onStartFns = [];
            this._onDestroyFns = [];
            this._initialized = false;
            this._finished = false;
            this._started = false;
            this._destroyed = false;
            this.time = 0;
            this.parentPlayer = null;
            this.currentSnapshot = {};
            this._duration = (/** @type {?} */ (options['duration']));
            this._delay = (/** @type {?} */ (options['delay'])) || 0;
            this.time = this._duration + this._delay;
        }
        /**
         * @private
         * @return {?}
         */
        _onFinish() {
            if (!this._finished) {
                this._finished = true;
                this._onDoneFns.forEach((/**
                 * @param {?} fn
                 * @return {?}
                 */
                fn => fn()));
                this._onDoneFns = [];
            }
        }
        /**
         * @return {?}
         */
        init() {
            this._buildPlayer();
            this._preparePlayerBeforeStart();
        }
        /**
         * @private
         * @return {?}
         */
        _buildPlayer() {
            if (this._initialized)
                return;
            this._initialized = true;
            /** @type {?} */
            const keyframes = this.keyframes;
            ((/** @type {?} */ (this))).domPlayer =
                this._triggerWebAnimation(this.element, keyframes, this.options);
            this._finalKeyframe = keyframes.length ? keyframes[keyframes.length - 1] : {};
            this.domPlayer.addEventListener('finish', (/**
             * @return {?}
             */
            () => this._onFinish()));
        }
        /**
         * @private
         * @return {?}
         */
        _preparePlayerBeforeStart() {
            // this is required so that the player doesn't start to animate right away
            if (this._delay) {
                this._resetDomPlayerState();
            }
            else {
                this.domPlayer.pause();
            }
        }
        /**
         * \@internal
         * @param {?} element
         * @param {?} keyframes
         * @param {?} options
         * @return {?}
         */
        _triggerWebAnimation(element, keyframes, options) {
            // jscompiler doesn't seem to know animate is a native property because it's not fully
            // supported yet across common browsers (we polyfill it for Edge/Safari) [CL #143630929]
            return (/** @type {?} */ (element['animate'](keyframes, options)));
        }
        /**
         * @param {?} fn
         * @return {?}
         */
        onStart(fn) {
            this._onStartFns.push(fn);
        }
        /**
         * @param {?} fn
         * @return {?}
         */
        onDone(fn) {
            this._onDoneFns.push(fn);
        }
        /**
         * @param {?} fn
         * @return {?}
         */
        onDestroy(fn) {
            this._onDestroyFns.push(fn);
        }
        /**
         * @return {?}
         */
        play() {
            this._buildPlayer();
            if (!this.hasStarted()) {
                this._onStartFns.forEach((/**
                 * @param {?} fn
                 * @return {?}
                 */
                fn => fn()));
                this._onStartFns = [];
                this._started = true;
                if (this._specialStyles) {
                    this._specialStyles.start();
                }
            }
            this.domPlayer.play();
        }
        /**
         * @return {?}
         */
        pause() {
            this.init();
            this.domPlayer.pause();
        }
        /**
         * @return {?}
         */
        finish() {
            this.init();
            if (this._specialStyles) {
                this._specialStyles.finish();
            }
            this._onFinish();
            this.domPlayer.finish();
        }
        /**
         * @return {?}
         */
        reset() {
            this._resetDomPlayerState();
            this._destroyed = false;
            this._finished = false;
            this._started = false;
        }
        /**
         * @private
         * @return {?}
         */
        _resetDomPlayerState() {
            if (this.domPlayer) {
                this.domPlayer.cancel();
            }
        }
        /**
         * @return {?}
         */
        restart() {
            this.reset();
            this.play();
        }
        /**
         * @return {?}
         */
        hasStarted() {
            return this._started;
        }
        /**
         * @return {?}
         */
        destroy() {
            if (!this._destroyed) {
                this._destroyed = true;
                this._resetDomPlayerState();
                this._onFinish();
                if (this._specialStyles) {
                    this._specialStyles.destroy();
                }
                this._onDestroyFns.forEach((/**
                 * @param {?} fn
                 * @return {?}
                 */
                fn => fn()));
                this._onDestroyFns = [];
            }
        }
        /**
         * @param {?} p
         * @return {?}
         */
        setPosition(p) {
            this.domPlayer.currentTime = p * this.time;
        }
        /**
         * @return {?}
         */
        getPosition() {
            return this.domPlayer.currentTime / this.time;
        }
        /**
         * @return {?}
         */
        get totalTime() {
            return this._delay + this._duration;
        }
        /**
         * @return {?}
         */
        beforeDestroy() {
            /** @type {?} */
            const styles = {};
            if (this.hasStarted()) {
                Object.keys(this._finalKeyframe).forEach((/**
                 * @param {?} prop
                 * @return {?}
                 */
                prop => {
                    if (prop != 'offset') {
                        styles[prop] =
                            this._finished ? this._finalKeyframe[prop] : computeStyle(this.element, prop);
                    }
                }));
            }
            this.currentSnapshot = styles;
        }
        /**
         * \@internal
         * @param {?} phaseName
         * @return {?}
         */
        triggerCallback(phaseName) {
            /** @type {?} */
            const methods = phaseName == 'start' ? this._onStartFns : this._onDoneFns;
            methods.forEach((/**
             * @param {?} fn
             * @return {?}
             */
            fn => fn()));
            methods.length = 0;
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/animations/browser/src/render/web_animations/web_animations_driver.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class WebAnimationsDriver {
        constructor() {
            this._isNativeImpl = /\{\s*\[native\s+code\]\s*\}/.test(getElementAnimateFn().toString());
            this._cssKeyframesDriver = new CssKeyframesDriver();
        }
        /**
         * @param {?} prop
         * @return {?}
         */
        validateStyleProperty(prop) {
            return validateStyleProperty(prop);
        }
        /**
         * @param {?} element
         * @param {?} selector
         * @return {?}
         */
        matchesElement(element, selector) {
            return matchesElement(element, selector);
        }
        /**
         * @param {?} elm1
         * @param {?} elm2
         * @return {?}
         */
        containsElement(elm1, elm2) {
            return containsElement(elm1, elm2);
        }
        /**
         * @param {?} element
         * @param {?} selector
         * @param {?} multi
         * @return {?}
         */
        query(element, selector, multi) {
            return invokeQuery(element, selector, multi);
        }
        /**
         * @param {?} element
         * @param {?} prop
         * @param {?=} defaultValue
         * @return {?}
         */
        computeStyle(element, prop, defaultValue) {
            return (/** @type {?} */ (((/** @type {?} */ (window.getComputedStyle(element))))[prop]));
        }
        /**
         * @param {?} supported
         * @return {?}
         */
        overrideWebAnimationsSupport(supported) {
            this._isNativeImpl = supported;
        }
        /**
         * @param {?} element
         * @param {?} keyframes
         * @param {?} duration
         * @param {?} delay
         * @param {?} easing
         * @param {?=} previousPlayers
         * @param {?=} scrubberAccessRequested
         * @return {?}
         */
        animate(element, keyframes, duration, delay, easing, previousPlayers = [], scrubberAccessRequested) {
            /** @type {?} */
            const useKeyframes = !scrubberAccessRequested && !this._isNativeImpl;
            if (useKeyframes) {
                return this._cssKeyframesDriver.animate(element, keyframes, duration, delay, easing, previousPlayers);
            }
            /** @type {?} */
            const fill = delay == 0 ? 'both' : 'forwards';
            /** @type {?} */
            const playerOptions = { duration, delay, fill };
            // we check for this to avoid having a null|undefined value be present
            // for the easing (which results in an error for certain browsers #9752)
            if (easing) {
                playerOptions['easing'] = easing;
            }
            /** @type {?} */
            const previousStyles = {};
            /** @type {?} */
            const previousWebAnimationPlayers = (/** @type {?} */ (previousPlayers.filter((/**
             * @param {?} player
             * @return {?}
             */
            player => player instanceof WebAnimationsPlayer))));
            if (allowPreviousPlayerStylesMerge(duration, delay)) {
                previousWebAnimationPlayers.forEach((/**
                 * @param {?} player
                 * @return {?}
                 */
                player => {
                    /** @type {?} */
                    let styles = player.currentSnapshot;
                    Object.keys(styles).forEach((/**
                     * @param {?} prop
                     * @return {?}
                     */
                    prop => previousStyles[prop] = styles[prop]));
                }));
            }
            keyframes = keyframes.map((/**
             * @param {?} styles
             * @return {?}
             */
            styles => copyStyles(styles, false)));
            keyframes = balancePreviousStylesIntoKeyframes(element, keyframes, previousStyles);
            /** @type {?} */
            const specialStyles = packageNonAnimatableStyles(element, keyframes);
            return new WebAnimationsPlayer(element, keyframes, playerOptions, specialStyles);
        }
    }
    /**
     * @return {?}
     */
    function supportsWebAnimations() {
        return typeof getElementAnimateFn() === 'function';
    }
    /**
     * @return {?}
     */
    function getElementAnimateFn() {
        return (isBrowser() && ((/** @type {?} */ (Element))).prototype['animate']) || {};
    }

    /**
     * @license Angular v9.1.12
     * (c) 2010-2020 Google LLC. https://angular.io/
     * License: MIT
     */
    class BrowserAnimationBuilder extends AnimationBuilder {
        /**
         * @param {?} rootRenderer
         * @param {?} doc
         */
        constructor(rootRenderer, doc) {
            super();
            this._nextAnimationId = 0;
            /** @type {?} */
            const typeData = (/** @type {?} */ ({ id: '0', encapsulation: ViewEncapsulation$1.None, styles: [], data: { animation: [] } }));
            this._renderer = (/** @type {?} */ (rootRenderer.createRenderer(doc.body, typeData)));
        }
        /**
         * @param {?} animation
         * @return {?}
         */
        build(animation) {
            /** @type {?} */
            const id = this._nextAnimationId.toString();
            this._nextAnimationId++;
            /** @type {?} */
            const entry = Array.isArray(animation) ? sequence(animation) : animation;
            issueAnimationCommand(this._renderer, null, id, 'register', [entry]);
            return new BrowserAnimationFactory(id, this._renderer);
        }
    }
    BrowserAnimationBuilder.ɵfac = function BrowserAnimationBuilder_Factory(t) { return new (t || BrowserAnimationBuilder)(ɵɵinject(RendererFactory2), ɵɵinject(DOCUMENT$1)); };
    BrowserAnimationBuilder.ɵprov = ɵɵdefineInjectable({ token: BrowserAnimationBuilder, factory: BrowserAnimationBuilder.ɵfac });
    /** @nocollapse */
    BrowserAnimationBuilder.ctorParameters = () => [
        { type: RendererFactory2 },
        { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT$1,] }] }
    ];
    class BrowserAnimationFactory extends AnimationFactory {
        /**
         * @param {?} _id
         * @param {?} _renderer
         */
        constructor(_id, _renderer) {
            super();
            this._id = _id;
            this._renderer = _renderer;
        }
        /**
         * @param {?} element
         * @param {?=} options
         * @return {?}
         */
        create(element, options) {
            return new RendererAnimationPlayer(this._id, element, options || {}, this._renderer);
        }
    }
    class RendererAnimationPlayer {
        /**
         * @param {?} id
         * @param {?} element
         * @param {?} options
         * @param {?} _renderer
         */
        constructor(id, element, options, _renderer) {
            this.id = id;
            this.element = element;
            this._renderer = _renderer;
            this.parentPlayer = null;
            this._started = false;
            this.totalTime = 0;
            this._command('create', options);
        }
        /**
         * @private
         * @param {?} eventName
         * @param {?} callback
         * @return {?}
         */
        _listen(eventName, callback) {
            return this._renderer.listen(this.element, `@@${this.id}:${eventName}`, callback);
        }
        /**
         * @private
         * @param {?} command
         * @param {...?} args
         * @return {?}
         */
        _command(command, ...args) {
            return issueAnimationCommand(this._renderer, this.element, this.id, command, args);
        }
        /**
         * @param {?} fn
         * @return {?}
         */
        onDone(fn) {
            this._listen('done', fn);
        }
        /**
         * @param {?} fn
         * @return {?}
         */
        onStart(fn) {
            this._listen('start', fn);
        }
        /**
         * @param {?} fn
         * @return {?}
         */
        onDestroy(fn) {
            this._listen('destroy', fn);
        }
        /**
         * @return {?}
         */
        init() {
            this._command('init');
        }
        /**
         * @return {?}
         */
        hasStarted() {
            return this._started;
        }
        /**
         * @return {?}
         */
        play() {
            this._command('play');
            this._started = true;
        }
        /**
         * @return {?}
         */
        pause() {
            this._command('pause');
        }
        /**
         * @return {?}
         */
        restart() {
            this._command('restart');
        }
        /**
         * @return {?}
         */
        finish() {
            this._command('finish');
        }
        /**
         * @return {?}
         */
        destroy() {
            this._command('destroy');
        }
        /**
         * @return {?}
         */
        reset() {
            this._command('reset');
        }
        /**
         * @param {?} p
         * @return {?}
         */
        setPosition(p) {
            this._command('setPosition', p);
        }
        /**
         * @return {?}
         */
        getPosition() {
            return 0;
        }
    }
    /**
     * @param {?} renderer
     * @param {?} element
     * @param {?} id
     * @param {?} command
     * @param {?} args
     * @return {?}
     */
    function issueAnimationCommand(renderer, element, id, command, args) {
        return renderer.setProperty(element, `@@${id}:${command}`, args);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/platform-browser/animations/src/animation_renderer.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const ANIMATION_PREFIX = '@';
    /** @type {?} */
    const DISABLE_ANIMATIONS_FLAG = '@.disabled';
    class AnimationRendererFactory {
        /**
         * @param {?} delegate
         * @param {?} engine
         * @param {?} _zone
         */
        constructor(delegate, engine, _zone) {
            this.delegate = delegate;
            this.engine = engine;
            this._zone = _zone;
            this._currentId = 0;
            this._microtaskId = 1;
            this._animationCallbacksBuffer = [];
            this._rendererCache = new Map();
            this._cdRecurDepth = 0;
            this.promise = Promise.resolve(0);
            engine.onRemovalComplete = (/**
             * @param {?} element
             * @param {?} delegate
             * @return {?}
             */
            (element, delegate) => {
                // Note: if an component element has a leave animation, and the component
                // a host leave animation, the view engine will call `removeChild` for the parent
                // component renderer as well as for the child component renderer.
                // Therefore, we need to check if we already removed the element.
                if (delegate && delegate.parentNode(element)) {
                    delegate.removeChild(element.parentNode, element);
                }
            });
        }
        /**
         * @param {?} hostElement
         * @param {?} type
         * @return {?}
         */
        createRenderer(hostElement, type) {
            /** @type {?} */
            const EMPTY_NAMESPACE_ID = '';
            // cache the delegates to find out which cached delegate can
            // be used by which cached renderer
            /** @type {?} */
            const delegate = this.delegate.createRenderer(hostElement, type);
            if (!hostElement || !type || !type.data || !type.data['animation']) {
                /** @type {?} */
                let renderer = this._rendererCache.get(delegate);
                if (!renderer) {
                    renderer = new BaseAnimationRenderer(EMPTY_NAMESPACE_ID, delegate, this.engine);
                    // only cache this result when the base renderer is used
                    this._rendererCache.set(delegate, renderer);
                }
                return renderer;
            }
            /** @type {?} */
            const componentId = type.id;
            /** @type {?} */
            const namespaceId = type.id + '-' + this._currentId;
            this._currentId++;
            this.engine.register(namespaceId, hostElement);
            /** @type {?} */
            const registerTrigger = (/**
             * @param {?} trigger
             * @return {?}
             */
            (trigger) => {
                if (Array.isArray(trigger)) {
                    trigger.forEach(registerTrigger);
                }
                else {
                    this.engine.registerTrigger(componentId, namespaceId, hostElement, trigger.name, trigger);
                }
            });
            /** @type {?} */
            const animationTriggers = (/** @type {?} */ (type.data['animation']));
            animationTriggers.forEach(registerTrigger);
            return new AnimationRenderer(this, namespaceId, delegate, this.engine);
        }
        /**
         * @return {?}
         */
        begin() {
            this._cdRecurDepth++;
            if (this.delegate.begin) {
                this.delegate.begin();
            }
        }
        /**
         * @private
         * @return {?}
         */
        _scheduleCountTask() {
            // always use promise to schedule microtask instead of use Zone
            this.promise.then((/**
             * @return {?}
             */
            () => {
                this._microtaskId++;
            }));
        }
        /**
         * \@internal
         * @param {?} count
         * @param {?} fn
         * @param {?} data
         * @return {?}
         */
        scheduleListenerCallback(count, fn, data) {
            if (count >= 0 && count < this._microtaskId) {
                this._zone.run((/**
                 * @return {?}
                 */
                () => fn(data)));
                return;
            }
            if (this._animationCallbacksBuffer.length == 0) {
                Promise.resolve(null).then((/**
                 * @return {?}
                 */
                () => {
                    this._zone.run((/**
                     * @return {?}
                     */
                    () => {
                        this._animationCallbacksBuffer.forEach((/**
                         * @param {?} tuple
                         * @return {?}
                         */
                        tuple => {
                            const [fn, data] = tuple;
                            fn(data);
                        }));
                        this._animationCallbacksBuffer = [];
                    }));
                }));
            }
            this._animationCallbacksBuffer.push([fn, data]);
        }
        /**
         * @return {?}
         */
        end() {
            this._cdRecurDepth--;
            // this is to prevent animations from running twice when an inner
            // component does CD when a parent component instead has inserted it
            if (this._cdRecurDepth == 0) {
                this._zone.runOutsideAngular((/**
                 * @return {?}
                 */
                () => {
                    this._scheduleCountTask();
                    this.engine.flush(this._microtaskId);
                }));
            }
            if (this.delegate.end) {
                this.delegate.end();
            }
        }
        /**
         * @return {?}
         */
        whenRenderingDone() {
            return this.engine.whenRenderingDone();
        }
    }
    AnimationRendererFactory.ɵfac = function AnimationRendererFactory_Factory(t) { return new (t || AnimationRendererFactory)(ɵɵinject(RendererFactory2), ɵɵinject(AnimationEngine), ɵɵinject(NgZone)); };
    AnimationRendererFactory.ɵprov = ɵɵdefineInjectable({ token: AnimationRendererFactory, factory: AnimationRendererFactory.ɵfac });
    /** @nocollapse */
    AnimationRendererFactory.ctorParameters = () => [
        { type: RendererFactory2 },
        { type: AnimationEngine },
        { type: NgZone }
    ];
    class BaseAnimationRenderer {
        /**
         * @param {?} namespaceId
         * @param {?} delegate
         * @param {?} engine
         */
        constructor(namespaceId, delegate, engine) {
            this.namespaceId = namespaceId;
            this.delegate = delegate;
            this.engine = engine;
            this.destroyNode = this.delegate.destroyNode ? (/**
             * @param {?} n
             * @return {?}
             */
            (n) => (/** @type {?} */ (delegate.destroyNode))(n)) : null;
        }
        /**
         * @return {?}
         */
        get data() {
            return this.delegate.data;
        }
        /**
         * @return {?}
         */
        destroy() {
            this.engine.destroy(this.namespaceId, this.delegate);
            this.delegate.destroy();
        }
        /**
         * @param {?} name
         * @param {?=} namespace
         * @return {?}
         */
        createElement(name, namespace) {
            return this.delegate.createElement(name, namespace);
        }
        /**
         * @param {?} value
         * @return {?}
         */
        createComment(value) {
            return this.delegate.createComment(value);
        }
        /**
         * @param {?} value
         * @return {?}
         */
        createText(value) {
            return this.delegate.createText(value);
        }
        /**
         * @param {?} parent
         * @param {?} newChild
         * @return {?}
         */
        appendChild(parent, newChild) {
            this.delegate.appendChild(parent, newChild);
            this.engine.onInsert(this.namespaceId, newChild, parent, false);
        }
        /**
         * @param {?} parent
         * @param {?} newChild
         * @param {?} refChild
         * @return {?}
         */
        insertBefore(parent, newChild, refChild) {
            this.delegate.insertBefore(parent, newChild, refChild);
            this.engine.onInsert(this.namespaceId, newChild, parent, true);
        }
        /**
         * @param {?} parent
         * @param {?} oldChild
         * @param {?} isHostElement
         * @return {?}
         */
        removeChild(parent, oldChild, isHostElement) {
            this.engine.onRemove(this.namespaceId, oldChild, this.delegate, isHostElement);
        }
        /**
         * @param {?} selectorOrNode
         * @param {?=} preserveContent
         * @return {?}
         */
        selectRootElement(selectorOrNode, preserveContent) {
            return this.delegate.selectRootElement(selectorOrNode, preserveContent);
        }
        /**
         * @param {?} node
         * @return {?}
         */
        parentNode(node) {
            return this.delegate.parentNode(node);
        }
        /**
         * @param {?} node
         * @return {?}
         */
        nextSibling(node) {
            return this.delegate.nextSibling(node);
        }
        /**
         * @param {?} el
         * @param {?} name
         * @param {?} value
         * @param {?=} namespace
         * @return {?}
         */
        setAttribute(el, name, value, namespace) {
            this.delegate.setAttribute(el, name, value, namespace);
        }
        /**
         * @param {?} el
         * @param {?} name
         * @param {?=} namespace
         * @return {?}
         */
        removeAttribute(el, name, namespace) {
            this.delegate.removeAttribute(el, name, namespace);
        }
        /**
         * @param {?} el
         * @param {?} name
         * @return {?}
         */
        addClass(el, name) {
            this.delegate.addClass(el, name);
        }
        /**
         * @param {?} el
         * @param {?} name
         * @return {?}
         */
        removeClass(el, name) {
            this.delegate.removeClass(el, name);
        }
        /**
         * @param {?} el
         * @param {?} style
         * @param {?} value
         * @param {?=} flags
         * @return {?}
         */
        setStyle(el, style, value, flags) {
            this.delegate.setStyle(el, style, value, flags);
        }
        /**
         * @param {?} el
         * @param {?} style
         * @param {?=} flags
         * @return {?}
         */
        removeStyle(el, style, flags) {
            this.delegate.removeStyle(el, style, flags);
        }
        /**
         * @param {?} el
         * @param {?} name
         * @param {?} value
         * @return {?}
         */
        setProperty(el, name, value) {
            if (name.charAt(0) == ANIMATION_PREFIX && name == DISABLE_ANIMATIONS_FLAG) {
                this.disableAnimations(el, !!value);
            }
            else {
                this.delegate.setProperty(el, name, value);
            }
        }
        /**
         * @param {?} node
         * @param {?} value
         * @return {?}
         */
        setValue(node, value) {
            this.delegate.setValue(node, value);
        }
        /**
         * @param {?} target
         * @param {?} eventName
         * @param {?} callback
         * @return {?}
         */
        listen(target, eventName, callback) {
            return this.delegate.listen(target, eventName, callback);
        }
        /**
         * @protected
         * @param {?} element
         * @param {?} value
         * @return {?}
         */
        disableAnimations(element, value) {
            this.engine.disableAnimations(element, value);
        }
    }
    class AnimationRenderer extends BaseAnimationRenderer {
        /**
         * @param {?} factory
         * @param {?} namespaceId
         * @param {?} delegate
         * @param {?} engine
         */
        constructor(factory, namespaceId, delegate, engine) {
            super(namespaceId, delegate, engine);
            this.factory = factory;
            this.namespaceId = namespaceId;
        }
        /**
         * @param {?} el
         * @param {?} name
         * @param {?} value
         * @return {?}
         */
        setProperty(el, name, value) {
            if (name.charAt(0) == ANIMATION_PREFIX) {
                if (name.charAt(1) == '.' && name == DISABLE_ANIMATIONS_FLAG) {
                    value = value === undefined ? true : !!value;
                    this.disableAnimations(el, (/** @type {?} */ (value)));
                }
                else {
                    this.engine.process(this.namespaceId, el, name.substr(1), value);
                }
            }
            else {
                this.delegate.setProperty(el, name, value);
            }
        }
        /**
         * @param {?} target
         * @param {?} eventName
         * @param {?} callback
         * @return {?}
         */
        listen(target, eventName, callback) {
            if (eventName.charAt(0) == ANIMATION_PREFIX) {
                /** @type {?} */
                const element = resolveElementFromTarget(target);
                /** @type {?} */
                let name = eventName.substr(1);
                /** @type {?} */
                let phase = '';
                // @listener.phase is for trigger animation callbacks
                // @@listener is for animation builder callbacks
                if (name.charAt(0) != ANIMATION_PREFIX) {
                    [name, phase] = parseTriggerCallbackName(name);
                }
                return this.engine.listen(this.namespaceId, element, name, phase, (/**
                 * @param {?} event
                 * @return {?}
                 */
                event => {
                    /** @type {?} */
                    const countId = ((/** @type {?} */ (event)))['_data'] || -1;
                    this.factory.scheduleListenerCallback(countId, callback, event);
                }));
            }
            return this.delegate.listen(target, eventName, callback);
        }
    }
    /**
     * @param {?} target
     * @return {?}
     */
    function resolveElementFromTarget(target) {
        switch (target) {
            case 'body':
                return document.body;
            case 'document':
                return document;
            case 'window':
                return window;
            default:
                return target;
        }
    }
    /**
     * @param {?} triggerName
     * @return {?}
     */
    function parseTriggerCallbackName(triggerName) {
        /** @type {?} */
        const dotIndex = triggerName.indexOf('.');
        /** @type {?} */
        const trigger = triggerName.substring(0, dotIndex);
        /** @type {?} */
        const phase = triggerName.substr(dotIndex + 1);
        return [trigger, phase];
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/platform-browser/animations/src/providers.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class InjectableAnimationEngine extends AnimationEngine {
        /**
         * @param {?} doc
         * @param {?} driver
         * @param {?} normalizer
         */
        constructor(doc, driver, normalizer) {
            super(doc.body, driver, normalizer);
        }
    }
    InjectableAnimationEngine.ɵfac = function InjectableAnimationEngine_Factory(t) { return new (t || InjectableAnimationEngine)(ɵɵinject(DOCUMENT$1), ɵɵinject(AnimationDriver), ɵɵinject(AnimationStyleNormalizer)); };
    InjectableAnimationEngine.ɵprov = ɵɵdefineInjectable({ token: InjectableAnimationEngine, factory: InjectableAnimationEngine.ɵfac });
    /** @nocollapse */
    InjectableAnimationEngine.ctorParameters = () => [
        { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT$1,] }] },
        { type: AnimationDriver },
        { type: AnimationStyleNormalizer }
    ];
    /**
     * @return {?}
     */
    function instantiateSupportedAnimationDriver() {
        return supportsWebAnimations() ? new WebAnimationsDriver() : new CssKeyframesDriver();
    }
    /**
     * @return {?}
     */
    function instantiateDefaultStyleNormalizer() {
        return new WebAnimationsStyleNormalizer();
    }
    /**
     * @param {?} renderer
     * @param {?} engine
     * @param {?} zone
     * @return {?}
     */
    function instantiateRendererFactory(renderer, engine, zone) {
        return new AnimationRendererFactory(renderer, engine, zone);
    }
    /**
     * \@publicApi
     * @type {?}
     */
    const ANIMATION_MODULE_TYPE = new InjectionToken('AnimationModuleType');
    /** @type {?} */
    const SHARED_ANIMATION_PROVIDERS = [
        { provide: AnimationBuilder, useClass: BrowserAnimationBuilder },
        { provide: AnimationStyleNormalizer, useFactory: instantiateDefaultStyleNormalizer },
        { provide: AnimationEngine, useClass: InjectableAnimationEngine }, {
            provide: RendererFactory2,
            useFactory: instantiateRendererFactory,
            deps: [DomRendererFactory2, AnimationEngine, NgZone]
        }
    ];
    /**
     * Separate providers from the actual module so that we can do a local modification in Google3 to
     * include them in the BrowserModule.
     * @type {?}
     */
    const BROWSER_ANIMATIONS_PROVIDERS = [
        { provide: AnimationDriver, useFactory: instantiateSupportedAnimationDriver },
        { provide: ANIMATION_MODULE_TYPE, useValue: 'BrowserAnimations' }, ...SHARED_ANIMATION_PROVIDERS
    ];
    /**
     * Separate providers from the actual module so that we can do a local modification in Google3 to
     * include them in the BrowserTestingModule.
     * @type {?}
     */
    const BROWSER_NOOP_ANIMATIONS_PROVIDERS = [
        { provide: AnimationDriver, useClass: NoopAnimationDriver },
        { provide: ANIMATION_MODULE_TYPE, useValue: 'NoopAnimations' }, ...SHARED_ANIMATION_PROVIDERS
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/platform-browser/animations/src/module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Exports `BrowserModule` with additional [dependency-injection providers](guide/glossary#provider)
     * for use with animations. See [Animations](guide/animations).
     * \@publicApi
     */
    class BrowserAnimationsModule {
    }
    BrowserAnimationsModule.ɵmod = ɵɵdefineNgModule({ type: BrowserAnimationsModule });
    BrowserAnimationsModule.ɵinj = ɵɵdefineInjector({ factory: function BrowserAnimationsModule_Factory(t) { return new (t || BrowserAnimationsModule)(); }, providers: BROWSER_ANIMATIONS_PROVIDERS, imports: [BrowserModule] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(BrowserAnimationsModule, { exports: function () { return [BrowserModule]; } }); })();
    /**
     * A null player that must be imported to allow disabling of animations.
     * \@publicApi
     */
    class NoopAnimationsModule {
    }
    NoopAnimationsModule.ɵmod = ɵɵdefineNgModule({ type: NoopAnimationsModule });
    NoopAnimationsModule.ɵinj = ɵɵdefineInjector({ factory: function NoopAnimationsModule_Factory(t) { return new (t || NoopAnimationsModule)(); }, providers: BROWSER_NOOP_ANIMATIONS_PROVIDERS, imports: [BrowserModule] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(NoopAnimationsModule, { exports: function () { return [BrowserModule]; } }); })();

    /**
     * @license NgRx 9.2.0
     * (c) 2015-2018 Brandon Roberts, Mike Ryan, Rob Wormald, Victor Savkin
     * License: MIT
     */
    let REGISTERED_ACTION_TYPES = {};

    /**
     * @fileoverview added by tsickle
     * Generated from: modules/store/src/action_creator.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * \@description
     * Creates a configured `Creator` function that, when called, returns an object in the shape of the `Action` interface.
     *
     * Action creators reduce the explicitness of class-based action creators.
     *
     * \@usageNotes
     *
     * **Declaring an action creator**
     *
     * Without additional metadata:
     * ```ts
     * export const increment = createAction('[Counter] Increment');
     * ```
     * With additional metadata:
     * ```ts
     * export const loginSuccess = createAction(
     *   '[Auth/API] Login Success',
     *   props<{ user: User }>()
     * );
     * ```
     * With a function:
     * ```ts
     * export const loginSuccess = createAction(
     *   '[Auth/API] Login Success',
     *   (response: Response) => response.user
     * );
     * ```
     *
     * **Dispatching an action**
     *
     * Without additional metadata:
     * ```ts
     * store.dispatch(increment());
     * ```
     * With additional metadata:
     * ```ts
     * store.dispatch(loginSuccess({ user: newUser }));
     * ```
     *
     * **Referencing an action in a reducer**
     *
     * Using a switch statement:
     * ```ts
     * switch (action.type) {
     *   // ...
     *   case AuthApiActions.loginSuccess.type: {
     *     return {
     *       ...state,
     *       user: action.user
     *     };
     *   }
     * }
     * ```
     * Using a reducer creator:
     * ```ts
     * on(AuthApiActions.loginSuccess, (state, { user }) => ({ ...state, user }))
     * ```
     *
     *  **Referencing an action in an effect**
     * ```ts
     * effectName$ = createEffect(
     *   () => this.actions$.pipe(
     *     ofType(AuthApiActions.loginSuccess),
     *     // ...
     *   )
     * );
     * ```
     * @template T, C
     * @param {?} type Describes the action that will be dispatched
     * @param {?=} config Additional metadata needed for the handling of the action.  See {\@link createAction#usage-notes Usage Notes}.
     *
     * @return {?}
     */
    function createAction(type, config) {
        REGISTERED_ACTION_TYPES[type] = (REGISTERED_ACTION_TYPES[type] || 0) + 1;
        if (typeof config === 'function') {
            return defineType(type, (/**
             * @param {...?} args
             * @return {?}
             */
            (...args) => (Object.assign(Object.assign({}, config(...args)), { type }))));
        }
        /** @type {?} */
        const as = config ? config._as : 'empty';
        switch (as) {
            case 'empty':
                return defineType(type, (/**
                 * @return {?}
                 */
                () => ({ type })));
            case 'props':
                return defineType(type, (/**
                 * @param {?} props
                 * @return {?}
                 */
                (props) => (Object.assign(Object.assign({}, props), { type }))));
            default:
                throw new Error('Unexpected config.');
        }
    }
    /**
     * @template P
     * @return {?}
     */
    function props() {
        return { _as: 'props', _p: (/** @type {?} */ (undefined)) };
    }
    /**
     * @template T
     * @param {?} type
     * @param {?} creator
     * @return {?}
     */
    function defineType(type, creator) {
        return Object.defineProperty(creator, 'type', {
            value: type,
            writable: false,
        });
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: modules/store/src/actions_subject.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const INIT = (/** @type {?} */ ('@ngrx/store/init'));
    class ActionsSubject extends BehaviorSubject {
        constructor() {
            super({ type: INIT });
        }
        /**
         * @param {?} action
         * @return {?}
         */
        next(action) {
            if (typeof action === 'function') {
                throw new TypeError(`
        Dispatch expected an object, instead it received a function.
        If you're using the createAction function, make sure to invoke the function
        before dispatching the action. For example, someAction should be someAction().`);
            }
            else if (typeof action === 'undefined') {
                throw new TypeError(`Actions must be objects`);
            }
            else if (typeof action.type === 'undefined') {
                throw new TypeError(`Actions must have a type property`);
            }
            super.next(action);
        }
        /**
         * @return {?}
         */
        complete() {
            /* noop */
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            super.complete();
        }
    }
    ActionsSubject.ɵfac = function ActionsSubject_Factory(t) { return new (t || ActionsSubject)(); };
    ActionsSubject.ɵprov = ɵɵdefineInjectable({ token: ActionsSubject, factory: ActionsSubject.ɵfac });
    /** @nocollapse */
    ActionsSubject.ctorParameters = () => [];
    /** @type {?} */
    const ACTIONS_SUBJECT_PROVIDERS = [ActionsSubject];

    /**
     * @fileoverview added by tsickle
     * Generated from: modules/store/src/tokens.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const _ROOT_STORE_GUARD = new InjectionToken('@ngrx/store Internal Root Guard');
    /** @type {?} */
    const _INITIAL_STATE = new InjectionToken('@ngrx/store Internal Initial State');
    /** @type {?} */
    const INITIAL_STATE = new InjectionToken('@ngrx/store Initial State');
    /** @type {?} */
    const REDUCER_FACTORY = new InjectionToken('@ngrx/store Reducer Factory');
    /** @type {?} */
    const _REDUCER_FACTORY = new InjectionToken('@ngrx/store Internal Reducer Factory Provider');
    /** @type {?} */
    const INITIAL_REDUCERS = new InjectionToken('@ngrx/store Initial Reducers');
    /** @type {?} */
    const _INITIAL_REDUCERS = new InjectionToken('@ngrx/store Internal Initial Reducers');
    /** @type {?} */
    const STORE_FEATURES = new InjectionToken('@ngrx/store Store Features');
    /** @type {?} */
    const _STORE_REDUCERS = new InjectionToken('@ngrx/store Internal Store Reducers');
    /** @type {?} */
    const _FEATURE_REDUCERS = new InjectionToken('@ngrx/store Internal Feature Reducers');
    /** @type {?} */
    const _FEATURE_CONFIGS = new InjectionToken('@ngrx/store Internal Feature Configs');
    /** @type {?} */
    const _STORE_FEATURES = new InjectionToken('@ngrx/store Internal Store Features');
    /** @type {?} */
    const _FEATURE_REDUCERS_TOKEN = new InjectionToken('@ngrx/store Internal Feature Reducers Token');
    /** @type {?} */
    const FEATURE_REDUCERS = new InjectionToken('@ngrx/store Feature Reducers');
    /**
     * User-defined meta reducers from StoreModule.forRoot()
     * @type {?}
     */
    const USER_PROVIDED_META_REDUCERS = new InjectionToken('@ngrx/store User Provided Meta Reducers');
    /**
     * Meta reducers defined either internally by \@ngrx/store or by library authors
     * @type {?}
     */
    const META_REDUCERS = new InjectionToken('@ngrx/store Meta Reducers');
    /**
     * Concats the user provided meta reducers and the meta reducers provided on the multi
     * injection token
     * @type {?}
     */
    const _RESOLVED_META_REDUCERS = new InjectionToken('@ngrx/store Internal Resolved Meta Reducers');
    /**
     * Runtime checks defined by the user via an InjectionToken
     * Defaults to `_USER_RUNTIME_CHECKS`
     * @type {?}
     */
    const USER_RUNTIME_CHECKS = new InjectionToken('@ngrx/store User Runtime Checks Config');
    /**
     * Runtime checks defined by the user via forRoot()
     * @type {?}
     */
    const _USER_RUNTIME_CHECKS = new InjectionToken('@ngrx/store Internal User Runtime Checks Config');
    /**
     * Runtime checks currently in use
     * @type {?}
     */
    const _ACTIVE_RUNTIME_CHECKS = new InjectionToken('@ngrx/store Internal Runtime Checks');
    /** @type {?} */
    const _ACTION_TYPE_UNIQUENESS_CHECK = new InjectionToken('@ngrx/store Check if Action types are unique');

    /**
     * @fileoverview added by tsickle
     * Generated from: modules/store/src/utils.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @param {?} reducers
     * @param {?=} initialState
     * @return {?}
     */
    function combineReducers(reducers, initialState = {}) {
        /** @type {?} */
        const reducerKeys = Object.keys(reducers);
        /** @type {?} */
        const finalReducers = {};
        for (let i = 0; i < reducerKeys.length; i++) {
            /** @type {?} */
            const key = reducerKeys[i];
            if (typeof reducers[key] === 'function') {
                finalReducers[key] = reducers[key];
            }
        }
        /** @type {?} */
        const finalReducerKeys = Object.keys(finalReducers);
        return (/**
         * @param {?} state
         * @param {?} action
         * @return {?}
         */
        function combination(state, action) {
            state = state === undefined ? initialState : state;
            /** @type {?} */
            let hasChanged = false;
            /** @type {?} */
            const nextState = {};
            for (let i = 0; i < finalReducerKeys.length; i++) {
                /** @type {?} */
                const key = finalReducerKeys[i];
                /** @type {?} */
                const reducer = finalReducers[key];
                /** @type {?} */
                const previousStateForKey = state[key];
                /** @type {?} */
                const nextStateForKey = reducer(previousStateForKey, action);
                nextState[key] = nextStateForKey;
                hasChanged = hasChanged || nextStateForKey !== previousStateForKey;
            }
            return hasChanged ? nextState : state;
        });
    }
    /**
     * @template T
     * @param {?} object
     * @param {?} keyToRemove
     * @return {?}
     */
    function omit(object, keyToRemove) {
        return Object.keys(object)
            .filter((/**
         * @param {?} key
         * @return {?}
         */
        key => key !== keyToRemove))
            .reduce((/**
         * @param {?} result
         * @param {?} key
         * @return {?}
         */
        (result, key) => Object.assign(result, { [key]: object[key] })), {});
    }
    /**
     * @param {...?} functions
     * @return {?}
     */
    function compose(...functions) {
        return (/**
         * @param {?} arg
         * @return {?}
         */
        function (arg) {
            if (functions.length === 0) {
                return arg;
            }
            /** @type {?} */
            const last = functions[functions.length - 1];
            /** @type {?} */
            const rest = functions.slice(0, -1);
            return rest.reduceRight((/**
             * @param {?} composed
             * @param {?} fn
             * @return {?}
             */
            (composed, fn) => fn(composed)), last(arg));
        });
    }
    /**
     * @template T, V
     * @param {?} reducerFactory
     * @param {?=} metaReducers
     * @return {?}
     */
    function createReducerFactory(reducerFactory, metaReducers) {
        if (Array.isArray(metaReducers) && metaReducers.length > 0) {
            ((/** @type {?} */ (reducerFactory))) = compose.apply(null, [
                ...metaReducers,
                reducerFactory,
            ]);
        }
        return (/**
         * @param {?} reducers
         * @param {?=} initialState
         * @return {?}
         */
        (reducers, initialState) => {
            /** @type {?} */
            const reducer = reducerFactory(reducers);
            return (/**
             * @param {?} state
             * @param {?} action
             * @return {?}
             */
            (state, action) => {
                state = state === undefined ? ((/** @type {?} */ (initialState))) : state;
                return reducer(state, action);
            });
        });
    }
    /**
     * @template T, V
     * @param {?=} metaReducers
     * @return {?}
     */
    function createFeatureReducerFactory(metaReducers) {
        /** @type {?} */
        const reducerFactory = Array.isArray(metaReducers) && metaReducers.length > 0
            ? compose(...metaReducers)
            : (/**
             * @param {?} r
             * @return {?}
             */
            (r) => r);
        return (/**
         * @param {?} reducer
         * @param {?=} initialState
         * @return {?}
         */
        (reducer, initialState) => {
            reducer = reducerFactory(reducer);
            return (/**
             * @param {?} state
             * @param {?} action
             * @return {?}
             */
            (state, action) => {
                state = state === undefined ? initialState : state;
                return reducer(state, action);
            });
        });
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: modules/store/src/reducer_manager.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @abstract
     */
    class ReducerObservable extends Observable {
    }
    /**
     * @abstract
     */
    class ReducerManagerDispatcher extends ActionsSubject {
    }
    /** @type {?} */
    const UPDATE = (/** @type {?} */ ('@ngrx/store/update-reducers'));
    class ReducerManager extends BehaviorSubject {
        /**
         * @param {?} dispatcher
         * @param {?} initialState
         * @param {?} reducers
         * @param {?} reducerFactory
         */
        constructor(dispatcher, initialState, reducers, reducerFactory) {
            super(reducerFactory(reducers, initialState));
            this.dispatcher = dispatcher;
            this.initialState = initialState;
            this.reducers = reducers;
            this.reducerFactory = reducerFactory;
        }
        /**
         * @param {?} feature
         * @return {?}
         */
        addFeature(feature) {
            this.addFeatures([feature]);
        }
        /**
         * @param {?} features
         * @return {?}
         */
        addFeatures(features) {
            /** @type {?} */
            const reducers = features.reduce((/**
             * @param {?} reducerDict
             * @param {?} __1
             * @return {?}
             */
            (reducerDict, { reducers, reducerFactory, metaReducers, initialState, key }) => {
                /** @type {?} */
                const reducer = typeof reducers === 'function'
                    ? createFeatureReducerFactory(metaReducers)(reducers, initialState)
                    : createReducerFactory(reducerFactory, metaReducers)(reducers, initialState);
                reducerDict[key] = reducer;
                return reducerDict;
            }), (/** @type {?} */ ({})));
            this.addReducers(reducers);
        }
        /**
         * @param {?} feature
         * @return {?}
         */
        removeFeature(feature) {
            this.removeFeatures([feature]);
        }
        /**
         * @param {?} features
         * @return {?}
         */
        removeFeatures(features) {
            this.removeReducers(features.map((/**
             * @param {?} p
             * @return {?}
             */
            p => p.key)));
        }
        /**
         * @param {?} key
         * @param {?} reducer
         * @return {?}
         */
        addReducer(key, reducer) {
            this.addReducers({ [key]: reducer });
        }
        /**
         * @param {?} reducers
         * @return {?}
         */
        addReducers(reducers) {
            this.reducers = Object.assign(Object.assign({}, this.reducers), reducers);
            this.updateReducers(Object.keys(reducers));
        }
        /**
         * @param {?} featureKey
         * @return {?}
         */
        removeReducer(featureKey) {
            this.removeReducers([featureKey]);
        }
        /**
         * @param {?} featureKeys
         * @return {?}
         */
        removeReducers(featureKeys) {
            featureKeys.forEach((/**
             * @param {?} key
             * @return {?}
             */
            key => {
                this.reducers = (/** @type {?} */ (omit(this.reducers, key) /*TODO(#823)*/));
            }));
            this.updateReducers(featureKeys);
        }
        /**
         * @private
         * @param {?} featureKeys
         * @return {?}
         */
        updateReducers(featureKeys) {
            this.next(this.reducerFactory(this.reducers, this.initialState));
            this.dispatcher.next((/** @type {?} */ ({
                type: UPDATE,
                features: featureKeys,
            })));
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this.complete();
        }
    }
    ReducerManager.ɵfac = function ReducerManager_Factory(t) { return new (t || ReducerManager)(ɵɵinject(ReducerManagerDispatcher), ɵɵinject(INITIAL_STATE), ɵɵinject(INITIAL_REDUCERS), ɵɵinject(REDUCER_FACTORY)); };
    ReducerManager.ɵprov = ɵɵdefineInjectable({ token: ReducerManager, factory: ReducerManager.ɵfac });
    /** @nocollapse */
    ReducerManager.ctorParameters = () => [
        { type: ReducerManagerDispatcher },
        { type: undefined, decorators: [{ type: Inject, args: [INITIAL_STATE,] }] },
        { type: undefined, decorators: [{ type: Inject, args: [INITIAL_REDUCERS,] }] },
        { type: undefined, decorators: [{ type: Inject, args: [REDUCER_FACTORY,] }] }
    ];
    /** @type {?} */
    const REDUCER_MANAGER_PROVIDERS = [
        ReducerManager,
        { provide: ReducerObservable, useExisting: ReducerManager },
        { provide: ReducerManagerDispatcher, useExisting: ActionsSubject },
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: modules/store/src/scanned_actions_subject.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class ScannedActionsSubject extends Subject {
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this.complete();
        }
    }
    ScannedActionsSubject.ɵfac = function ScannedActionsSubject_Factory(t) { return ɵScannedActionsSubject_BaseFactory(t || ScannedActionsSubject); };
    ScannedActionsSubject.ɵprov = ɵɵdefineInjectable({ token: ScannedActionsSubject, factory: ScannedActionsSubject.ɵfac });
    const ɵScannedActionsSubject_BaseFactory = ɵɵgetInheritedFactory(ScannedActionsSubject);
    /** @type {?} */
    const SCANNED_ACTIONS_SUBJECT_PROVIDERS = [
        ScannedActionsSubject,
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: modules/store/src/state.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @abstract
     */
    class StateObservable extends Observable {
    }
    /**
     * @template T
     */
    class State extends BehaviorSubject {
        /**
         * @param {?} actions$
         * @param {?} reducer$
         * @param {?} scannedActions
         * @param {?} initialState
         */
        constructor(actions$, reducer$, scannedActions, initialState) {
            super(initialState);
            /** @type {?} */
            const actionsOnQueue$ = actions$.pipe(observeOn(queue));
            /** @type {?} */
            const withLatestReducer$ = actionsOnQueue$.pipe(withLatestFrom(reducer$));
            /** @type {?} */
            const seed = { state: initialState };
            /** @type {?} */
            const stateAndAction$ = withLatestReducer$.pipe(scan(reduceState, seed));
            this.stateSubscription = stateAndAction$.subscribe((/**
             * @param {?} __0
             * @return {?}
             */
            ({ state, action }) => {
                this.next(state);
                scannedActions.next(action);
            }));
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this.stateSubscription.unsubscribe();
            this.complete();
        }
    }
    State.ɵfac = function State_Factory(t) { return new (t || State)(ɵɵinject(ActionsSubject), ɵɵinject(ReducerObservable), ɵɵinject(ScannedActionsSubject), ɵɵinject(INITIAL_STATE)); };
    State.ɵprov = ɵɵdefineInjectable({ token: State, factory: State.ɵfac });
    State.INIT = INIT;
    /** @nocollapse */
    State.ctorParameters = () => [
        { type: ActionsSubject },
        { type: ReducerObservable },
        { type: ScannedActionsSubject },
        { type: undefined, decorators: [{ type: Inject, args: [INITIAL_STATE,] }] }
    ];
    /**
     * @template T, V
     * @param {?=} stateActionPair
     * @param {?=} __1
     * @return {?}
     */
    function reduceState(stateActionPair = { state: undefined }, [action, reducer]) {
        const { state } = stateActionPair;
        return { state: reducer(state, action), action };
    }
    /** @type {?} */
    const STATE_PROVIDERS = [
        State,
        { provide: StateObservable, useExisting: State },
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: modules/store/src/store.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @template T
     */
    class Store extends Observable {
        /**
         * @param {?} state$
         * @param {?} actionsObserver
         * @param {?} reducerManager
         */
        constructor(state$, actionsObserver, reducerManager) {
            super();
            this.actionsObserver = actionsObserver;
            this.reducerManager = reducerManager;
            this.source = state$;
        }
        /**
         * @template Props, K
         * @param {?} pathOrMapFn
         * @param {...?} paths
         * @return {?}
         */
        select(pathOrMapFn, ...paths) {
            return ((/** @type {?} */ (select))).call(null, pathOrMapFn, ...paths)(this);
        }
        /**
         * @template R
         * @param {?} operator
         * @return {?}
         */
        lift(operator) {
            /** @type {?} */
            const store = new Store(this, this.actionsObserver, this.reducerManager);
            store.operator = operator;
            return store;
        }
        /**
         * @template V
         * @param {?} action
         * @return {?}
         */
        dispatch(action) {
            this.actionsObserver.next(action);
        }
        /**
         * @param {?} action
         * @return {?}
         */
        next(action) {
            this.actionsObserver.next(action);
        }
        /**
         * @param {?} err
         * @return {?}
         */
        error(err) {
            this.actionsObserver.error(err);
        }
        /**
         * @return {?}
         */
        complete() {
            this.actionsObserver.complete();
        }
        /**
         * @template State, Actions
         * @param {?} key
         * @param {?} reducer
         * @return {?}
         */
        addReducer(key, reducer) {
            this.reducerManager.addReducer(key, reducer);
        }
        /**
         * @template Key
         * @param {?} key
         * @return {?}
         */
        removeReducer(key) {
            this.reducerManager.removeReducer(key);
        }
    }
    Store.ɵfac = function Store_Factory(t) { return new (t || Store)(ɵɵinject(StateObservable), ɵɵinject(ActionsSubject), ɵɵinject(ReducerManager)); };
    Store.ɵprov = ɵɵdefineInjectable({ token: Store, factory: Store.ɵfac });
    /** @nocollapse */
    Store.ctorParameters = () => [
        { type: StateObservable },
        { type: ActionsSubject },
        { type: ReducerManager }
    ];
    /** @type {?} */
    const STORE_PROVIDERS = [Store];
    /**
     * @template T, Props, K
     * @param {?} pathOrMapFn
     * @param {?=} propsOrPath
     * @param {...?} paths
     * @return {?}
     */
    function select(pathOrMapFn, propsOrPath, ...paths) {
        return (/**
         * @param {?} source$
         * @return {?}
         */
        function selectOperator(source$) {
            /** @type {?} */
            let mapped$;
            if (typeof pathOrMapFn === 'string') {
                /** @type {?} */
                const pathSlices = [(/** @type {?} */ (propsOrPath)), ...paths].filter(Boolean);
                mapped$ = source$.pipe(pluck(pathOrMapFn, ...pathSlices));
            }
            else if (typeof pathOrMapFn === 'function') {
                mapped$ = source$.pipe(map((/**
                 * @param {?} source
                 * @return {?}
                 */
                source => pathOrMapFn(source, (/** @type {?} */ (propsOrPath))))));
            }
            else {
                throw new TypeError(`Unexpected type '${typeof pathOrMapFn}' in select operator,` +
                    ` expected 'string' or 'function'`);
            }
            return mapped$.pipe(distinctUntilChanged());
        });
    }
    /**
     * @param {?} a
     * @param {?} b
     * @return {?}
     */
    function isEqualCheck(a, b) {
        return a === b;
    }
    /**
     * @param {?} args
     * @param {?} lastArguments
     * @param {?} comparator
     * @return {?}
     */
    function isArgumentsChanged(args, lastArguments, comparator) {
        for (let i = 0; i < args.length; i++) {
            if (!comparator(args[i], lastArguments[i])) {
                return true;
            }
        }
        return false;
    }
    /**
     * @param {?} projectionFn
     * @param {?=} isArgumentsEqual
     * @param {?=} isResultEqual
     * @return {?}
     */
    function defaultMemoize(projectionFn, isArgumentsEqual = isEqualCheck, isResultEqual = isEqualCheck) {
        /** @type {?} */
        let lastArguments = null;
        // tslint:disable-next-line:no-any anything could be the result.
        /** @type {?} */
        let lastResult = null;
        /** @type {?} */
        let overrideResult;
        /**
         * @return {?}
         */
        function reset() {
            lastArguments = null;
            lastResult = null;
        }
        /**
         * @param {?=} result
         * @return {?}
         */
        function setResult(result = undefined) {
            overrideResult = { result };
        }
        /**
         * @return {?}
         */
        function clearResult() {
            overrideResult = undefined;
        }
        // tslint:disable-next-line:no-any anything could be the result.
        /**
         * @return {?}
         */
        function memoized() {
            if (overrideResult !== undefined) {
                return overrideResult.result;
            }
            if (!lastArguments) {
                lastResult = projectionFn.apply(null, (/** @type {?} */ (arguments)));
                lastArguments = arguments;
                return lastResult;
            }
            if (!isArgumentsChanged(arguments, lastArguments, isArgumentsEqual)) {
                return lastResult;
            }
            /** @type {?} */
            const newResult = projectionFn.apply(null, (/** @type {?} */ (arguments)));
            lastArguments = arguments;
            if (isResultEqual(lastResult, newResult)) {
                return lastResult;
            }
            lastResult = newResult;
            return newResult;
        }
        return { memoized, reset, setResult, clearResult };
    }
    /**
     * @param {...?} input
     * @return {?}
     */
    function createSelector(...input) {
        return createSelectorFactory(defaultMemoize)(...input);
    }
    /**
     * @param {?} state
     * @param {?} selectors
     * @param {?} props
     * @param {?} memoizedProjector
     * @return {?}
     */
    function defaultStateFn(state, selectors, props, memoizedProjector) {
        if (props === undefined) {
            /** @type {?} */
            const args = ((/** @type {?} */ (selectors))).map((/**
             * @param {?} fn
             * @return {?}
             */
            fn => fn(state)));
            return memoizedProjector.memoized.apply(null, args);
        }
        /** @type {?} */
        const args = ((/** @type {?} */ (selectors))).map((/**
         * @param {?} fn
         * @return {?}
         */
        fn => fn(state, props)));
        return memoizedProjector.memoized.apply(null, [...args, props]);
    }
    /**
     * @param {?} memoize
     * @param {?=} options
     * @return {?}
     */
    function createSelectorFactory(memoize, options = {
        stateFn: defaultStateFn,
    }) {
        return (/**
         * @param {...?} input
         * @return {?}
         */
        function (...input) {
            /** @type {?} */
            let args = input;
            if (Array.isArray(args[0])) {
                const [head, ...tail] = args;
                args = [...head, ...tail];
            }
            /** @type {?} */
            const selectors = args.slice(0, args.length - 1);
            /** @type {?} */
            const projector = args[args.length - 1];
            /** @type {?} */
            const memoizedSelectors = selectors.filter((/**
             * @param {?} selector
             * @return {?}
             */
            (selector) => selector.release && typeof selector.release === 'function'));
            /** @type {?} */
            const memoizedProjector = memoize((/**
             * @param {...?} selectors
             * @return {?}
             */
            function (...selectors) {
                return projector.apply(null, selectors);
            }));
            /** @type {?} */
            const memoizedState = defaultMemoize((/**
             * @param {?} state
             * @param {?} props
             * @return {?}
             */
            function (state, props) {
                return options.stateFn.apply(null, [
                    state,
                    selectors,
                    props,
                    memoizedProjector,
                ]);
            }));
            /**
             * @return {?}
             */
            function release() {
                memoizedState.reset();
                memoizedProjector.reset();
                memoizedSelectors.forEach((/**
                 * @param {?} selector
                 * @return {?}
                 */
                selector => selector.release()));
            }
            return Object.assign(memoizedState.memoized, {
                release,
                projector: memoizedProjector.memoized,
                setResult: memoizedState.setResult,
                clearResult: memoizedState.clearResult,
            });
        });
    }
    /**
     * @param {?} featureName
     * @return {?}
     */
    function createFeatureSelector(featureName) {
        return createSelector((/**
         * @param {?} state
         * @return {?}
         */
        (state) => {
            /** @type {?} */
            const featureState = state[featureName];
            if ( isDevMode() && !(featureName in state)) {
                console.warn(`@ngrx/store: The feature name \"${featureName}\" does ` +
                    'not exist in the state, therefore createFeatureSelector ' +
                    'cannot access it.  Be sure it is imported in a loaded module ' +
                    `using StoreModule.forRoot('${featureName}', ...) or ` +
                    `StoreModule.forFeature('${featureName}', ...).  If the default ` +
                    'state is intended to be undefined, as is the case with router ' +
                    'state, this development-only warning message can be ignored.');
            }
            return featureState;
        }), (/**
         * @param {?} featureState
         * @return {?}
         */
        (featureState) => featureState));
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: modules/store/src/meta-reducers/utils.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const RUNTIME_CHECK_URL = 'https://ngrx.io/guide/store/configuration/runtime-checks';
    /**
     * @param {?} target
     * @return {?}
     */
    function isUndefined(target) {
        return target === undefined;
    }
    /**
     * @param {?} target
     * @return {?}
     */
    function isNull(target) {
        return target === null;
    }
    /**
     * @param {?} target
     * @return {?}
     */
    function isArray$1(target) {
        return Array.isArray(target);
    }
    /**
     * @param {?} target
     * @return {?}
     */
    function isString(target) {
        return typeof target === 'string';
    }
    /**
     * @param {?} target
     * @return {?}
     */
    function isBoolean(target) {
        return typeof target === 'boolean';
    }
    /**
     * @param {?} target
     * @return {?}
     */
    function isNumber(target) {
        return typeof target === 'number';
    }
    /**
     * @param {?} target
     * @return {?}
     */
    function isObjectLike(target) {
        return typeof target === 'object' && target !== null;
    }
    /**
     * @param {?} target
     * @return {?}
     */
    function isObject$2(target) {
        return isObjectLike(target) && !isArray$1(target);
    }
    /**
     * @param {?} target
     * @return {?}
     */
    function isPlainObject(target) {
        if (!isObject$2(target)) {
            return false;
        }
        /** @type {?} */
        const targetPrototype = Object.getPrototypeOf(target);
        return targetPrototype === Object.prototype || targetPrototype === null;
    }
    /**
     * @param {?} target
     * @return {?}
     */
    function isFunction$1(target) {
        return typeof target === 'function';
    }
    /**
     * @param {?} target
     * @return {?}
     */
    function isComponent(target) {
        return isFunction$1(target) && target.hasOwnProperty('ɵcmp');
    }
    /**
     * @param {?} target
     * @param {?} propertyName
     * @return {?}
     */
    function hasOwnProperty(target, propertyName) {
        return Object.prototype.hasOwnProperty.call(target, propertyName);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: modules/store/src/meta-reducers/immutability_reducer.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @param {?} reducer
     * @param {?} checks
     * @return {?}
     */
    function immutabilityCheckMetaReducer(reducer, checks) {
        return (/**
         * @param {?} state
         * @param {?} action
         * @return {?}
         */
        function (state, action) {
            /** @type {?} */
            const act = checks.action(action) ? freeze(action) : action;
            /** @type {?} */
            const nextState = reducer(state, act);
            return checks.state() ? freeze(nextState) : nextState;
        });
    }
    /**
     * @param {?} target
     * @return {?}
     */
    function freeze(target) {
        Object.freeze(target);
        /** @type {?} */
        const targetIsFunction = isFunction$1(target);
        Object.getOwnPropertyNames(target).forEach((/**
         * @param {?} prop
         * @return {?}
         */
        prop => {
            // Ignore Ivy properties, ref: https://github.com/ngrx/platform/issues/2109#issuecomment-582689060
            if (prop.startsWith('ɵ')) {
                return;
            }
            if (hasOwnProperty(target, prop) &&
                (targetIsFunction
                    ? prop !== 'caller' && prop !== 'callee' && prop !== 'arguments'
                    : true)) {
                /** @type {?} */
                const propValue = target[prop];
                if ((isObjectLike(propValue) || isFunction$1(propValue)) &&
                    !Object.isFrozen(propValue)) {
                    freeze(propValue);
                }
            }
        }));
        return target;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: modules/store/src/meta-reducers/serialization_reducer.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @param {?} reducer
     * @param {?} checks
     * @return {?}
     */
    function serializationCheckMetaReducer(reducer, checks) {
        return (/**
         * @param {?} state
         * @param {?} action
         * @return {?}
         */
        function (state, action) {
            if (checks.action(action)) {
                /** @type {?} */
                const unserializableAction = getUnserializable(action);
                throwIfUnserializable(unserializableAction, 'action');
            }
            /** @type {?} */
            const nextState = reducer(state, action);
            if (checks.state()) {
                /** @type {?} */
                const unserializableState = getUnserializable(nextState);
                throwIfUnserializable(unserializableState, 'state');
            }
            return nextState;
        });
    }
    /**
     * @param {?=} target
     * @param {?=} path
     * @return {?}
     */
    function getUnserializable(target, path = []) {
        // Guard against undefined and null, e.g. a reducer that returns undefined
        if ((isUndefined(target) || isNull(target)) && path.length === 0) {
            return {
                path: ['root'],
                value: target,
            };
        }
        /** @type {?} */
        const keys = Object.keys(target);
        return keys.reduce((/**
         * @param {?} result
         * @param {?} key
         * @return {?}
         */
        (result, key) => {
            if (result) {
                return result;
            }
            /** @type {?} */
            const value = ((/** @type {?} */ (target)))[key];
            // Ignore Ivy components
            if (isComponent(value)) {
                return result;
            }
            if (isUndefined(value) ||
                isNull(value) ||
                isNumber(value) ||
                isBoolean(value) ||
                isString(value) ||
                isArray$1(value)) {
                return false;
            }
            if (isPlainObject(value)) {
                return getUnserializable(value, [...path, key]);
            }
            return {
                path: [...path, key],
                value,
            };
        }), false);
    }
    /**
     * @param {?} unserializable
     * @param {?} context
     * @return {?}
     */
    function throwIfUnserializable(unserializable, context) {
        if (unserializable === false) {
            return;
        }
        /** @type {?} */
        const unserializablePath = unserializable.path.join('.');
        /** @type {?} */
        const error = new Error(`Detected unserializable ${context} at "${unserializablePath}". ${RUNTIME_CHECK_URL}#strict${context}serializability`);
        error.value = unserializable.value;
        error.unserializablePath = unserializablePath;
        throw error;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: modules/store/src/meta-reducers/inNgZoneAssert_reducer.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @param {?} reducer
     * @param {?} checks
     * @return {?}
     */
    function inNgZoneAssertMetaReducer(reducer, checks) {
        return (/**
         * @param {?} state
         * @param {?} action
         * @return {?}
         */
        function (state, action) {
            if (checks.action(action) && !NgZone.isInAngularZone()) {
                throw new Error(`Action '${action.type}' running outside NgZone. ${RUNTIME_CHECK_URL}#strictactionwithinngzone`);
            }
            return reducer(state, action);
        });
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: modules/store/src/meta-reducers/index.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */

    /**
     * @fileoverview added by tsickle
     * Generated from: modules/store/src/runtime_checks.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @param {?=} runtimeChecks
     * @return {?}
     */
    function createActiveRuntimeChecks(runtimeChecks) {
        if (isDevMode()) {
            return Object.assign({ strictStateSerializability: false, strictActionSerializability: false, strictStateImmutability: true, strictActionImmutability: true, strictActionWithinNgZone: false, strictActionTypeUniqueness: false }, runtimeChecks);
        }
        return {
            strictStateSerializability: false,
            strictActionSerializability: false,
            strictStateImmutability: false,
            strictActionImmutability: false,
            strictActionWithinNgZone: false,
            strictActionTypeUniqueness: false,
        };
    }
    /**
     * @param {?} __0
     * @return {?}
     */
    function createSerializationCheckMetaReducer({ strictActionSerializability, strictStateSerializability, }) {
        return (/**
         * @param {?} reducer
         * @return {?}
         */
        (reducer) => strictActionSerializability || strictStateSerializability
            ? serializationCheckMetaReducer(reducer, {
                action: (/**
                 * @param {?} action
                 * @return {?}
                 */
                (action) => strictActionSerializability && !ignoreNgrxAction(action)),
                state: (/**
                 * @return {?}
                 */
                () => strictStateSerializability),
            })
            : reducer);
    }
    /**
     * @param {?} __0
     * @return {?}
     */
    function createImmutabilityCheckMetaReducer({ strictActionImmutability, strictStateImmutability, }) {
        return (/**
         * @param {?} reducer
         * @return {?}
         */
        (reducer) => strictActionImmutability || strictStateImmutability
            ? immutabilityCheckMetaReducer(reducer, {
                action: (/**
                 * @param {?} action
                 * @return {?}
                 */
                (action) => strictActionImmutability && !ignoreNgrxAction(action)),
                state: (/**
                 * @return {?}
                 */
                () => strictStateImmutability),
            })
            : reducer);
    }
    /**
     * @param {?} action
     * @return {?}
     */
    function ignoreNgrxAction(action) {
        return action.type.startsWith('@ngrx');
    }
    /**
     * @param {?} __0
     * @return {?}
     */
    function createInNgZoneCheckMetaReducer({ strictActionWithinNgZone, }) {
        return (/**
         * @param {?} reducer
         * @return {?}
         */
        (reducer) => strictActionWithinNgZone
            ? inNgZoneAssertMetaReducer(reducer, {
                action: (/**
                 * @param {?} action
                 * @return {?}
                 */
                (action) => strictActionWithinNgZone && !ignoreNgrxAction(action)),
            })
            : reducer);
    }
    /**
     * @param {?=} runtimeChecks
     * @return {?}
     */
    function provideRuntimeChecks(runtimeChecks) {
        return [
            {
                provide: _USER_RUNTIME_CHECKS,
                useValue: runtimeChecks,
            },
            {
                provide: USER_RUNTIME_CHECKS,
                useFactory: _runtimeChecksFactory,
                deps: [_USER_RUNTIME_CHECKS],
            },
            {
                provide: _ACTIVE_RUNTIME_CHECKS,
                deps: [USER_RUNTIME_CHECKS],
                useFactory: createActiveRuntimeChecks,
            },
            {
                provide: META_REDUCERS,
                multi: true,
                deps: [_ACTIVE_RUNTIME_CHECKS],
                useFactory: createImmutabilityCheckMetaReducer,
            },
            {
                provide: META_REDUCERS,
                multi: true,
                deps: [_ACTIVE_RUNTIME_CHECKS],
                useFactory: createSerializationCheckMetaReducer,
            },
            {
                provide: META_REDUCERS,
                multi: true,
                deps: [_ACTIVE_RUNTIME_CHECKS],
                useFactory: createInNgZoneCheckMetaReducer,
            },
        ];
    }
    /**
     * @return {?}
     */
    function checkForActionTypeUniqueness() {
        return [
            {
                provide: _ACTION_TYPE_UNIQUENESS_CHECK,
                multi: true,
                deps: [_ACTIVE_RUNTIME_CHECKS],
                useFactory: _actionTypeUniquenessCheck,
            },
        ];
    }
    /**
     * @param {?} runtimeChecks
     * @return {?}
     */
    function _runtimeChecksFactory(runtimeChecks) {
        return runtimeChecks;
    }
    /**
     * @param {?} config
     * @return {?}
     */
    function _actionTypeUniquenessCheck(config) {
        if (!config.strictActionTypeUniqueness) {
            return;
        }
        /** @type {?} */
        const duplicates = Object.entries(REGISTERED_ACTION_TYPES)
            .filter((/**
         * @param {?} __0
         * @return {?}
         */
        ([, registrations]) => registrations > 1))
            .map((/**
         * @param {?} __0
         * @return {?}
         */
        ([type]) => type));
        if (duplicates.length) {
            throw new Error(`Action types are registered more than once, ${duplicates
            .map((/**
         * @param {?} type
         * @return {?}
         */
        (type) => `"${type}"`))
            .join(', ')}. ${RUNTIME_CHECK_URL}#strictactiontypeuniqueness`);
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: modules/store/src/store_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class StoreRootModule {
        /**
         * @param {?} actions$
         * @param {?} reducer$
         * @param {?} scannedActions$
         * @param {?} store
         * @param {?} guard
         * @param {?} actionCheck
         */
        constructor(actions$, reducer$, scannedActions$, store, guard, actionCheck) {
        }
    }
    StoreRootModule.ɵmod = ɵɵdefineNgModule({ type: StoreRootModule });
    StoreRootModule.ɵinj = ɵɵdefineInjector({ factory: function StoreRootModule_Factory(t) { return new (t || StoreRootModule)(ɵɵinject(ActionsSubject), ɵɵinject(ReducerObservable), ɵɵinject(ScannedActionsSubject), ɵɵinject(Store), ɵɵinject(_ROOT_STORE_GUARD, 8), ɵɵinject(_ACTION_TYPE_UNIQUENESS_CHECK, 8)); } });
    /** @nocollapse */
    StoreRootModule.ctorParameters = () => [
        { type: ActionsSubject },
        { type: ReducerObservable },
        { type: ScannedActionsSubject },
        { type: Store },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [_ROOT_STORE_GUARD,] }] },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [_ACTION_TYPE_UNIQUENESS_CHECK,] }] }
    ];
    class StoreFeatureModule {
        /**
         * @param {?} features
         * @param {?} featureReducers
         * @param {?} reducerManager
         * @param {?} root
         * @param {?} actionCheck
         */
        constructor(features, featureReducers, reducerManager, root, actionCheck) {
            this.features = features;
            this.featureReducers = featureReducers;
            this.reducerManager = reducerManager;
            /** @type {?} */
            const feats = features.map((/**
             * @param {?} feature
             * @param {?} index
             * @return {?}
             */
            (feature, index) => {
                /** @type {?} */
                const featureReducerCollection = featureReducers.shift();
                /** @type {?} */
                const reducers = (/** @type {?} */ (featureReducerCollection /*TODO(#823)*/))[index];
                return Object.assign(Object.assign({}, feature), { reducers, initialState: _initialStateFactory(feature.initialState) });
            }));
            reducerManager.addFeatures(feats);
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this.reducerManager.removeFeatures(this.features);
        }
    }
    StoreFeatureModule.ɵmod = ɵɵdefineNgModule({ type: StoreFeatureModule });
    StoreFeatureModule.ɵinj = ɵɵdefineInjector({ factory: function StoreFeatureModule_Factory(t) { return new (t || StoreFeatureModule)(ɵɵinject(_STORE_FEATURES), ɵɵinject(FEATURE_REDUCERS), ɵɵinject(ReducerManager), ɵɵinject(StoreRootModule), ɵɵinject(_ACTION_TYPE_UNIQUENESS_CHECK, 8)); } });
    /** @nocollapse */
    StoreFeatureModule.ctorParameters = () => [
        { type: Array, decorators: [{ type: Inject, args: [_STORE_FEATURES,] }] },
        { type: Array, decorators: [{ type: Inject, args: [FEATURE_REDUCERS,] }] },
        { type: ReducerManager },
        { type: StoreRootModule },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [_ACTION_TYPE_UNIQUENESS_CHECK,] }] }
    ];
    class StoreModule {
        /**
         * @param {?} reducers
         * @param {?=} config
         * @return {?}
         */
        static forRoot(reducers, config = {}) {
            return {
                ngModule: StoreRootModule,
                providers: [
                    {
                        provide: _ROOT_STORE_GUARD,
                        useFactory: _provideForRootGuard,
                        deps: [[Store, new Optional(), new SkipSelf()]],
                    },
                    { provide: _INITIAL_STATE, useValue: config.initialState },
                    {
                        provide: INITIAL_STATE,
                        useFactory: _initialStateFactory,
                        deps: [_INITIAL_STATE],
                    },
                    { provide: _INITIAL_REDUCERS, useValue: reducers },
                    {
                        provide: _STORE_REDUCERS,
                        useExisting: reducers instanceof InjectionToken ? reducers : _INITIAL_REDUCERS,
                    },
                    {
                        provide: INITIAL_REDUCERS,
                        deps: [Injector, _INITIAL_REDUCERS, [new Inject(_STORE_REDUCERS)]],
                        useFactory: _createStoreReducers,
                    },
                    {
                        provide: USER_PROVIDED_META_REDUCERS,
                        useValue: config.metaReducers ? config.metaReducers : [],
                    },
                    {
                        provide: _RESOLVED_META_REDUCERS,
                        deps: [META_REDUCERS, USER_PROVIDED_META_REDUCERS],
                        useFactory: _concatMetaReducers,
                    },
                    {
                        provide: _REDUCER_FACTORY,
                        useValue: config.reducerFactory
                            ? config.reducerFactory
                            : combineReducers,
                    },
                    {
                        provide: REDUCER_FACTORY,
                        deps: [_REDUCER_FACTORY, _RESOLVED_META_REDUCERS],
                        useFactory: createReducerFactory,
                    },
                    ACTIONS_SUBJECT_PROVIDERS,
                    REDUCER_MANAGER_PROVIDERS,
                    SCANNED_ACTIONS_SUBJECT_PROVIDERS,
                    STATE_PROVIDERS,
                    STORE_PROVIDERS,
                    provideRuntimeChecks(config.runtimeChecks),
                    checkForActionTypeUniqueness(),
                ],
            };
        }
        /**
         * @param {?} featureName
         * @param {?} reducers
         * @param {?=} config
         * @return {?}
         */
        static forFeature(featureName, reducers, config = {}) {
            return {
                ngModule: StoreFeatureModule,
                providers: [
                    {
                        provide: _FEATURE_CONFIGS,
                        multi: true,
                        useValue: config,
                    },
                    {
                        provide: STORE_FEATURES,
                        multi: true,
                        useValue: {
                            key: featureName,
                            reducerFactory: !(config instanceof InjectionToken) && config.reducerFactory
                                ? config.reducerFactory
                                : combineReducers,
                            metaReducers: !(config instanceof InjectionToken) && config.metaReducers
                                ? config.metaReducers
                                : [],
                            initialState: !(config instanceof InjectionToken) && config.initialState
                                ? config.initialState
                                : undefined,
                        },
                    },
                    {
                        provide: _STORE_FEATURES,
                        deps: [Injector, _FEATURE_CONFIGS, STORE_FEATURES],
                        useFactory: _createFeatureStore,
                    },
                    { provide: _FEATURE_REDUCERS, multi: true, useValue: reducers },
                    {
                        provide: _FEATURE_REDUCERS_TOKEN,
                        multi: true,
                        useExisting: reducers instanceof InjectionToken ? reducers : _FEATURE_REDUCERS,
                    },
                    {
                        provide: FEATURE_REDUCERS,
                        multi: true,
                        deps: [
                            Injector,
                            _FEATURE_REDUCERS,
                            [new Inject(_FEATURE_REDUCERS_TOKEN)],
                        ],
                        useFactory: _createFeatureReducers,
                    },
                    checkForActionTypeUniqueness(),
                ],
            };
        }
    }
    StoreModule.ɵmod = ɵɵdefineNgModule({ type: StoreModule });
    StoreModule.ɵinj = ɵɵdefineInjector({ factory: function StoreModule_Factory(t) { return new (t || StoreModule)(); } });
    /**
     * @param {?} injector
     * @param {?} reducers
     * @return {?}
     */
    function _createStoreReducers(injector, reducers) {
        return reducers instanceof InjectionToken ? injector.get(reducers) : reducers;
    }
    /**
     * @param {?} injector
     * @param {?} configs
     * @param {?} featureStores
     * @return {?}
     */
    function _createFeatureStore(injector, configs, featureStores) {
        return featureStores.map((/**
         * @param {?} feat
         * @param {?} index
         * @return {?}
         */
        (feat, index) => {
            if (configs[index] instanceof InjectionToken) {
                /** @type {?} */
                const conf = injector.get(configs[index]);
                return {
                    key: feat.key,
                    reducerFactory: conf.reducerFactory
                        ? conf.reducerFactory
                        : combineReducers,
                    metaReducers: conf.metaReducers ? conf.metaReducers : [],
                    initialState: conf.initialState,
                };
            }
            return feat;
        }));
    }
    /**
     * @param {?} injector
     * @param {?} reducerCollection
     * @return {?}
     */
    function _createFeatureReducers(injector, reducerCollection) {
        /** @type {?} */
        const reducers = reducerCollection.map((/**
         * @param {?} reducer
         * @return {?}
         */
        reducer => {
            return reducer instanceof InjectionToken ? injector.get(reducer) : reducer;
        }));
        return reducers;
    }
    /**
     * @param {?} initialState
     * @return {?}
     */
    function _initialStateFactory(initialState) {
        if (typeof initialState === 'function') {
            return initialState();
        }
        return initialState;
    }
    /**
     * @param {?} metaReducers
     * @param {?} userProvidedMetaReducers
     * @return {?}
     */
    function _concatMetaReducers(metaReducers, userProvidedMetaReducers) {
        return metaReducers.concat(userProvidedMetaReducers);
    }
    /**
     * @param {?} store
     * @return {?}
     */
    function _provideForRootGuard(store) {
        if (store) {
            throw new TypeError(`StoreModule.forRoot() called twice. Feature modules should use StoreModule.forFeature() instead.`);
        }
        return 'guarded';
    }
    /**
     * \@description
     * Associates actions with a given state change function.
     * A state change function must be provided as the last parameter.
     *
     * @param {...?} args `ActionCreator`'s followed by a state change function.
     *
     * **To maintain type-safety**: pass 10 or less `ActionCreator`'s.
     * @return {?} an association of action types with a state change function.
     */
    function on(...args) {
        /** @type {?} */
        const reducer = (/** @type {?} */ (args.pop()));
        /** @type {?} */
        const types = args.reduce((/**
         * @param {?} result
         * @param {?} creator
         * @return {?}
         */
        (result, creator) => [...result, ((/** @type {?} */ (creator))).type]), (/** @type {?} */ ([])));
        return { reducer, types };
    }
    /**
     * \@description
     * Creates a reducer function to handle state transitions.
     *
     * Reducer creators reduce the explicitness of reducer functions with switch statements.
     *
     * \@usageNotes
     *
     * - Must be used with `ActionCreator`'s (returned by `createAction`).  Cannot be used with class-based action creators.
     * - The returned `ActionReducer` should additionally be returned from an exported `reducer` function, if you are using View Engine.
     * In case you are using Ivy the extra function `reducer` is not required.
     *
     * **Declaring a reducer creator with an exported reducer function**
     *
     * ```ts
     * const featureReducer = createReducer(
     *   initialState,
     *   on(
     *     featureActions.actionOne,
     *     featureActions.actionTwo,
     *     (state, { updatedValue }) => ({ ...state, prop: updatedValue })
     *   ),
     *   on(featureActions.actionThree, () => initialState);
     * );
     *
     * export function reducer(state: State | undefined, action: Action) {
     *   return featureReducer(state, action);
     * }
     * ```
     * @template S, A
     * @param {?} initialState Provides a state value if the current state is `undefined`, as it is initially.
     * @param {...?} ons Associations between actions and state changes.
     * @return {?} A reducer function.
     *
     */
    function createReducer(initialState, ...ons) {
        /** @type {?} */
        const map = new Map();
        for (let on of ons) {
            for (let type of on.types) {
                if (map.has(type)) {
                    /** @type {?} */
                    const existingReducer = (/** @type {?} */ (map.get(type)));
                    /** @type {?} */
                    const newReducer = (/**
                     * @param {?} state
                     * @param {?} action
                     * @return {?}
                     */
                    (state, action) => on.reducer(existingReducer(state, action), action));
                    map.set(type, newReducer);
                }
                else {
                    map.set(type, on.reducer);
                }
            }
        }
        return (/**
         * @param {?=} state
         * @param {?=} action
         * @return {?}
         */
        function (state = initialState, action) {
            /** @type {?} */
            const reducer = map.get(action.type);
            return reducer ? reducer(state, action) : state;
        });
    }

    /**
     * @license NgRx 9.2.0
     * (c) 2015-2018 Brandon Roberts, Mike Ryan, Rob Wormald, Victor Savkin
     * License: MIT
     */
    /** @type {?} */
    const DEFAULT_EFFECT_CONFIG = {
        dispatch: true,
        useEffectsErrorHandler: true,
    };
    /** @type {?} */
    const CREATE_EFFECT_METADATA_KEY = '__@ngrx/effects_create__';

    /**
     * @fileoverview added by tsickle
     * Generated from: modules/effects/src/effect_creator.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * \@description
     * Creates an effect from an `Observable` and an `EffectConfig`.
     *
     * \@usageNotes
     *
     * ** Mapping to a different action **
     * ```ts
     * effectName$ = createEffect(
     *   () => this.actions$.pipe(
     *     ofType(FeatureActions.actionOne),
     *     map(() => FeatureActions.actionTwo())
     *   )
     * );
     * ```
     *
     *  ** Non-dispatching effects **
     * ```ts
     * effectName$ = createEffect(
     *   () => this.actions$.pipe(
     *     ofType(FeatureActions.actionOne),
     *     tap(() => console.log('Action One Dispatched'))
     *   ),
     *   { dispatch: false }
     *   // FeatureActions.actionOne is not dispatched
     * );
     * ```
     * @template C, DT, OT, R
     * @param {?} source A function which returns an `Observable`.
     * @param {?=} config A `Partial<EffectConfig>` to configure the effect.  By default, `dispatch` is true and `useEffectsErrorHandler` is true.
     * @return {?} If `EffectConfig`#`dispatch` is true, returns `Observable<Action>`.  Else, returns `Observable<unknown>`.
     *
     */
    function createEffect(source, config) {
        /** @type {?} */
        const effect = source();
        /** @type {?} */
        const value = Object.assign(Object.assign({}, DEFAULT_EFFECT_CONFIG), config);
        Object.defineProperty(effect, CREATE_EFFECT_METADATA_KEY, {
            value,
        });
        return (/** @type {?} */ (effect));
    }
    /**
     * @template T
     * @param {?} instance
     * @return {?}
     */
    function getCreateEffectMetadata(instance) {
        /** @type {?} */
        const propertyNames = (/** @type {?} */ (Object.getOwnPropertyNames(instance)));
        /** @type {?} */
        const metadata = propertyNames
            .filter((/**
         * @param {?} propertyName
         * @return {?}
         */
        propertyName => instance[propertyName] &&
            instance[propertyName].hasOwnProperty(CREATE_EFFECT_METADATA_KEY)))
            .map((/**
         * @param {?} propertyName
         * @return {?}
         */
        propertyName => {
            /** @type {?} */
            const metaData = ((/** @type {?} */ (instance[propertyName])))[CREATE_EFFECT_METADATA_KEY];
            return Object.assign({ propertyName }, metaData);
        }));
        return metadata;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: modules/effects/src/utils.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @template T
     * @param {?} instance
     * @return {?}
     */
    function getSourceForInstance(instance) {
        return Object.getPrototypeOf(instance);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: modules/effects/src/effect_decorator.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const METADATA_KEY = '__@ngrx/effects__';
    /**
     * @template T
     * @param {?} instance
     * @return {?}
     */
    function getEffectDecoratorMetadata(instance) {
        /** @type {?} */
        const effectsDecorators = compose(getEffectMetadataEntries, getSourceForInstance)(instance);
        return effectsDecorators;
    }
    /**
     * Type guard to detemine whether METADATA_KEY is already present on the Class
     * constructor
     * @template T
     * @param {?} sourceProto
     * @return {?}
     */
    function hasMetadataEntries(sourceProto) {
        return sourceProto.constructor.hasOwnProperty(METADATA_KEY);
    }
    /**
     * @template T
     * @param {?} sourceProto
     * @return {?}
     */
    function getEffectMetadataEntries(sourceProto) {
        return hasMetadataEntries(sourceProto)
            ? sourceProto.constructor[METADATA_KEY]
            : [];
    }
    /**
     * @template T
     * @param {?} instance
     * @return {?}
     */
    function getSourceMetadata(instance) {
        /** @type {?} */
        const effects = [
            getEffectDecoratorMetadata,
            getCreateEffectMetadata,
        ];
        return effects.reduce((/**
         * @param {?} sources
         * @param {?} source
         * @return {?}
         */
        (sources, source) => sources.concat(source(instance))), []);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: modules/effects/src/effects_resolver.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @param {?} sourceInstance
     * @param {?} globalErrorHandler
     * @param {?} effectsErrorHandler
     * @return {?}
     */
    function mergeEffects(sourceInstance, globalErrorHandler, effectsErrorHandler) {
        /** @type {?} */
        const sourceName = getSourceForInstance(sourceInstance).constructor.name;
        /** @type {?} */
        const observables$ = getSourceMetadata(sourceInstance).map((/**
         * @param {?} __0
         * @return {?}
         */
        ({ propertyName, dispatch, useEffectsErrorHandler, }) => {
            /** @type {?} */
            const observable$ = typeof sourceInstance[propertyName] === 'function'
                ? sourceInstance[propertyName]()
                : sourceInstance[propertyName];
            /** @type {?} */
            const effectAction$ = useEffectsErrorHandler
                ? effectsErrorHandler(observable$, globalErrorHandler)
                : observable$;
            if (dispatch === false) {
                return effectAction$.pipe(ignoreElements());
            }
            /** @type {?} */
            const materialized$ = effectAction$.pipe(materialize());
            return materialized$.pipe(map((/**
             * @param {?} notification
             * @return {?}
             */
            (notification) => ({
                effect: sourceInstance[propertyName],
                notification,
                propertyName,
                sourceName,
                sourceInstance,
            }))));
        }));
        return merge(...observables$);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: modules/effects/src/effects_error_handler.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const MAX_NUMBER_OF_RETRY_ATTEMPTS = 10;
    /**
     * @template T
     * @param {?} observable$
     * @param {?} errorHandler
     * @param {?=} retryAttemptLeft
     * @return {?}
     */
    function defaultEffectsErrorHandler(observable$, errorHandler, retryAttemptLeft = MAX_NUMBER_OF_RETRY_ATTEMPTS) {
        return observable$.pipe(catchError((/**
         * @param {?} error
         * @return {?}
         */
        error => {
            if (errorHandler)
                errorHandler.handleError(error);
            if (retryAttemptLeft <= 1) {
                return observable$; // last attempt
            }
            // Return observable that produces this particular effect
            return defaultEffectsErrorHandler(observable$, errorHandler, retryAttemptLeft - 1);
        })));
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: modules/effects/src/actions.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @template V
     */
    class Actions extends Observable {
        /**
         * @param {?=} source
         */
        constructor(source) {
            super();
            if (source) {
                this.source = source;
            }
        }
        /**
         * @template R
         * @param {?} operator
         * @return {?}
         */
        lift(operator) {
            /** @type {?} */
            const observable = new Actions();
            observable.source = this;
            observable.operator = operator;
            return observable;
        }
    }
    Actions.ɵfac = function Actions_Factory(t) { return new (t || Actions)(ɵɵinject(ScannedActionsSubject)); };
    Actions.ɵprov = ɵɵdefineInjectable({ token: Actions, factory: Actions.ɵfac });
    /** @nocollapse */
    Actions.ctorParameters = () => [
        { type: Observable, decorators: [{ type: Inject, args: [ScannedActionsSubject,] }] }
    ];
    /**
     * @param {...?} allowedTypes
     * @return {?}
     */
    function ofType(...allowedTypes) {
        return filter((/**
         * @param {?} action
         * @return {?}
         */
        (action) => allowedTypes.some((/**
         * @param {?} typeOrActionCreator
         * @return {?}
         */
        typeOrActionCreator => {
            if (typeof typeOrActionCreator === 'string') {
                // Comparing the string to type
                return typeOrActionCreator === action.type;
            }
            // We are filtering by ActionCreator
            return typeOrActionCreator.type === action.type;
        }))));
    }
    /**
     * @param {?} output
     * @param {?} reporter
     * @return {?}
     */
    function reportInvalidActions(output, reporter) {
        if (output.notification.kind === 'N') {
            /** @type {?} */
            const action = output.notification.value;
            /** @type {?} */
            const isInvalidAction = !isAction(action);
            if (isInvalidAction) {
                reporter.handleError(new Error(`Effect ${getEffectName(output)} dispatched an invalid action: ${stringify$1(action)}`));
            }
        }
    }
    /**
     * @param {?} action
     * @return {?}
     */
    function isAction(action) {
        return (typeof action !== 'function' &&
            action &&
            action.type &&
            typeof action.type === 'string');
    }
    /**
     * @param {?} __0
     * @return {?}
     */
    function getEffectName({ propertyName, sourceInstance, sourceName, }) {
        /** @type {?} */
        const isMethod = typeof sourceInstance[propertyName] === 'function';
        return `"${sourceName}.${String(propertyName)}${isMethod ? '()' : ''}"`;
    }
    /**
     * @param {?} action
     * @return {?}
     */
    function stringify$1(action) {
        try {
            return JSON.stringify(action);
        }
        catch (_a) {
            return action;
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: modules/effects/src/lifecycle_hooks.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const onIdentifyEffectsKey = 'ngrxOnIdentifyEffects';
    /**
     * @param {?} instance
     * @return {?}
     */
    function isOnIdentifyEffects(instance) {
        return isFunction$2(instance, onIdentifyEffectsKey);
    }
    /** @type {?} */
    const onRunEffectsKey = 'ngrxOnRunEffects';
    /**
     * @param {?} instance
     * @return {?}
     */
    function isOnRunEffects(instance) {
        return isFunction$2(instance, onRunEffectsKey);
    }
    /** @type {?} */
    const onInitEffects = 'ngrxOnInitEffects';
    /**
     * @param {?} instance
     * @return {?}
     */
    function isOnInitEffects(instance) {
        return isFunction$2(instance, onInitEffects);
    }
    /**
     * @param {?} instance
     * @param {?} functionName
     * @return {?}
     */
    function isFunction$2(instance, functionName) {
        return (instance &&
            functionName in instance &&
            typeof instance[functionName] === 'function');
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: modules/effects/src/tokens.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const _ROOT_EFFECTS_GUARD = new InjectionToken('@ngrx/effects Internal Root Guard');
    /** @type {?} */
    const IMMEDIATE_EFFECTS = new InjectionToken('ngrx/effects: Immediate Effects');
    /** @type {?} */
    const USER_PROVIDED_EFFECTS = new InjectionToken('ngrx/effects: User Provided Effects');
    /** @type {?} */
    const _ROOT_EFFECTS = new InjectionToken('ngrx/effects: Internal Root Effects');
    /** @type {?} */
    const ROOT_EFFECTS = new InjectionToken('ngrx/effects: Root Effects');
    /** @type {?} */
    const _FEATURE_EFFECTS = new InjectionToken('ngrx/effects: Internal Feature Effects');
    /** @type {?} */
    const FEATURE_EFFECTS = new InjectionToken('ngrx/effects: Feature Effects');
    /** @type {?} */
    const EFFECTS_ERROR_HANDLER = new InjectionToken('ngrx/effects: Effects Error Handler');

    /**
     * @fileoverview added by tsickle
     * Generated from: modules/effects/src/effect_sources.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class EffectSources extends Subject {
        /**
         * @param {?} errorHandler
         * @param {?} effectsErrorHandler
         */
        constructor(errorHandler, effectsErrorHandler) {
            super();
            this.errorHandler = errorHandler;
            this.effectsErrorHandler = effectsErrorHandler;
        }
        /**
         * @param {?} effectSourceInstance
         * @return {?}
         */
        addEffects(effectSourceInstance) {
            this.next(effectSourceInstance);
        }
        /**
         * \@internal
         * @return {?}
         */
        toActions() {
            return this.pipe(groupBy(getSourceForInstance), mergeMap((/**
             * @param {?} source$
             * @return {?}
             */
            source$ => {
                return source$.pipe(groupBy(effectsInstance));
            })), mergeMap((/**
             * @param {?} source$
             * @return {?}
             */
            source$ => {
                /** @type {?} */
                const effect$ = source$.pipe(exhaustMap((/**
                 * @param {?} sourceInstance
                 * @return {?}
                 */
                sourceInstance => {
                    return resolveEffectSource(this.errorHandler, this.effectsErrorHandler)(sourceInstance);
                })), map((/**
                 * @param {?} output
                 * @return {?}
                 */
                output => {
                    reportInvalidActions(output, this.errorHandler);
                    return output.notification;
                })), filter((/**
                 * @param {?} notification
                 * @return {?}
                 */
                (notification) => notification.kind === 'N')), dematerialize());
                // start the stream with an INIT action
                // do this only for the first Effect instance
                /** @type {?} */
                const init$ = source$.pipe(take(1), filter(isOnInitEffects), map((/**
                 * @param {?} instance
                 * @return {?}
                 */
                instance => instance.ngrxOnInitEffects())));
                return merge(effect$, init$);
            })));
        }
    }
    EffectSources.ɵfac = function EffectSources_Factory(t) { return new (t || EffectSources)(ɵɵinject(ErrorHandler), ɵɵinject(EFFECTS_ERROR_HANDLER)); };
    EffectSources.ɵprov = ɵɵdefineInjectable({ token: EffectSources, factory: EffectSources.ɵfac });
    /** @nocollapse */
    EffectSources.ctorParameters = () => [
        { type: ErrorHandler },
        { type: undefined, decorators: [{ type: Inject, args: [EFFECTS_ERROR_HANDLER,] }] }
    ];
    /**
     * @param {?} sourceInstance
     * @return {?}
     */
    function effectsInstance(sourceInstance) {
        if (isOnIdentifyEffects(sourceInstance)) {
            return sourceInstance.ngrxOnIdentifyEffects();
        }
        return '';
    }
    /**
     * @param {?} errorHandler
     * @param {?} effectsErrorHandler
     * @return {?}
     */
    function resolveEffectSource(errorHandler, effectsErrorHandler) {
        return (/**
         * @param {?} sourceInstance
         * @return {?}
         */
        sourceInstance => {
            /** @type {?} */
            const mergedEffects$ = mergeEffects(sourceInstance, errorHandler, effectsErrorHandler);
            if (isOnRunEffects(sourceInstance)) {
                return sourceInstance.ngrxOnRunEffects(mergedEffects$);
            }
            return mergedEffects$;
        });
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: modules/effects/src/effects_runner.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class EffectsRunner {
        /**
         * @param {?} effectSources
         * @param {?} store
         */
        constructor(effectSources, store) {
            this.effectSources = effectSources;
            this.store = store;
            this.effectsSubscription = null;
        }
        /**
         * @return {?}
         */
        start() {
            if (!this.effectsSubscription) {
                this.effectsSubscription = this.effectSources
                    .toActions()
                    .subscribe(this.store);
            }
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            if (this.effectsSubscription) {
                this.effectsSubscription.unsubscribe();
                this.effectsSubscription = null;
            }
        }
    }
    EffectsRunner.ɵfac = function EffectsRunner_Factory(t) { return new (t || EffectsRunner)(ɵɵinject(EffectSources), ɵɵinject(Store)); };
    EffectsRunner.ɵprov = ɵɵdefineInjectable({ token: EffectsRunner, factory: EffectsRunner.ɵfac });
    /** @nocollapse */
    EffectsRunner.ctorParameters = () => [
        { type: EffectSources },
        { type: Store }
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: modules/effects/src/effects_root_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const ROOT_EFFECTS_INIT = '@ngrx/effects/init';
    /** @type {?} */
    const rootEffectsInit = createAction(ROOT_EFFECTS_INIT);
    class EffectsRootModule {
        /**
         * @param {?} sources
         * @param {?} runner
         * @param {?} store
         * @param {?} rootEffects
         * @param {?} storeRootModule
         * @param {?} storeFeatureModule
         * @param {?} guard
         */
        constructor(sources, runner, store, rootEffects, storeRootModule, storeFeatureModule, guard) {
            this.sources = sources;
            runner.start();
            rootEffects.forEach((/**
             * @param {?} effectSourceInstance
             * @return {?}
             */
            effectSourceInstance => sources.addEffects(effectSourceInstance)));
            store.dispatch({ type: ROOT_EFFECTS_INIT });
        }
        /**
         * @param {?} effectSourceInstance
         * @return {?}
         */
        addEffects(effectSourceInstance) {
            this.sources.addEffects(effectSourceInstance);
        }
    }
    EffectsRootModule.ɵmod = ɵɵdefineNgModule({ type: EffectsRootModule });
    EffectsRootModule.ɵinj = ɵɵdefineInjector({ factory: function EffectsRootModule_Factory(t) { return new (t || EffectsRootModule)(ɵɵinject(EffectSources), ɵɵinject(EffectsRunner), ɵɵinject(Store), ɵɵinject(ROOT_EFFECTS), ɵɵinject(StoreRootModule, 8), ɵɵinject(StoreFeatureModule, 8), ɵɵinject(_ROOT_EFFECTS_GUARD, 8)); } });
    /** @nocollapse */
    EffectsRootModule.ctorParameters = () => [
        { type: EffectSources },
        { type: EffectsRunner },
        { type: Store },
        { type: Array, decorators: [{ type: Inject, args: [ROOT_EFFECTS,] }] },
        { type: StoreRootModule, decorators: [{ type: Optional }] },
        { type: StoreFeatureModule, decorators: [{ type: Optional }] },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [_ROOT_EFFECTS_GUARD,] }] }
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: modules/effects/src/effects_feature_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class EffectsFeatureModule {
        /**
         * @param {?} root
         * @param {?} effectSourceGroups
         * @param {?} storeRootModule
         * @param {?} storeFeatureModule
         */
        constructor(root, effectSourceGroups, storeRootModule, storeFeatureModule) {
            effectSourceGroups.forEach((/**
             * @param {?} group
             * @return {?}
             */
            group => group.forEach((/**
             * @param {?} effectSourceInstance
             * @return {?}
             */
            effectSourceInstance => root.addEffects(effectSourceInstance)))));
        }
    }
    EffectsFeatureModule.ɵmod = ɵɵdefineNgModule({ type: EffectsFeatureModule });
    EffectsFeatureModule.ɵinj = ɵɵdefineInjector({ factory: function EffectsFeatureModule_Factory(t) { return new (t || EffectsFeatureModule)(ɵɵinject(EffectsRootModule), ɵɵinject(FEATURE_EFFECTS), ɵɵinject(StoreRootModule, 8), ɵɵinject(StoreFeatureModule, 8)); } });
    /** @nocollapse */
    EffectsFeatureModule.ctorParameters = () => [
        { type: EffectsRootModule },
        { type: Array, decorators: [{ type: Inject, args: [FEATURE_EFFECTS,] }] },
        { type: StoreRootModule, decorators: [{ type: Optional }] },
        { type: StoreFeatureModule, decorators: [{ type: Optional }] }
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: modules/effects/src/effects_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class EffectsModule {
        /**
         * @param {?=} featureEffects
         * @return {?}
         */
        static forFeature(featureEffects = []) {
            return {
                ngModule: EffectsFeatureModule,
                providers: [
                    featureEffects,
                    {
                        provide: _FEATURE_EFFECTS,
                        multi: true,
                        useValue: featureEffects,
                    },
                    {
                        provide: USER_PROVIDED_EFFECTS,
                        multi: true,
                        useValue: [],
                    },
                    {
                        provide: FEATURE_EFFECTS,
                        multi: true,
                        useFactory: createEffects,
                        deps: [Injector, _FEATURE_EFFECTS, USER_PROVIDED_EFFECTS],
                    },
                ],
            };
        }
        /**
         * @param {?=} rootEffects
         * @return {?}
         */
        static forRoot(rootEffects = []) {
            return {
                ngModule: EffectsRootModule,
                providers: [
                    {
                        provide: _ROOT_EFFECTS_GUARD,
                        useFactory: _provideForRootGuard$1,
                        deps: [[EffectsRunner, new Optional(), new SkipSelf()]],
                    },
                    {
                        provide: EFFECTS_ERROR_HANDLER,
                        useValue: defaultEffectsErrorHandler,
                    },
                    EffectsRunner,
                    EffectSources,
                    Actions,
                    rootEffects,
                    {
                        provide: _ROOT_EFFECTS,
                        useValue: [rootEffects],
                    },
                    {
                        provide: USER_PROVIDED_EFFECTS,
                        multi: true,
                        useValue: [],
                    },
                    {
                        provide: ROOT_EFFECTS,
                        useFactory: createEffects,
                        deps: [Injector, _ROOT_EFFECTS, USER_PROVIDED_EFFECTS],
                    },
                ],
            };
        }
    }
    EffectsModule.ɵmod = ɵɵdefineNgModule({ type: EffectsModule });
    EffectsModule.ɵinj = ɵɵdefineInjector({ factory: function EffectsModule_Factory(t) { return new (t || EffectsModule)(); } });
    /**
     * @param {?} injector
     * @param {?} effectGroups
     * @param {?} userProvidedEffectGroups
     * @return {?}
     */
    function createEffects(injector, effectGroups, userProvidedEffectGroups) {
        /** @type {?} */
        const mergedEffects = [];
        for (let effectGroup of effectGroups) {
            mergedEffects.push(...effectGroup);
        }
        for (let userProvidedEffectGroup of userProvidedEffectGroups) {
            mergedEffects.push(...userProvidedEffectGroup);
        }
        return createEffectInstances(injector, mergedEffects);
    }
    /**
     * @param {?} injector
     * @param {?} effects
     * @return {?}
     */
    function createEffectInstances(injector, effects) {
        return effects.map((/**
         * @param {?} effect
         * @return {?}
         */
        effect => injector.get(effect)));
    }
    /**
     * @param {?} runner
     * @return {?}
     */
    function _provideForRootGuard$1(runner) {
        if (runner) {
            throw new TypeError(`EffectsModule.forRoot() called twice. Feature modules should use EffectsModule.forFeature() instead.`);
        }
        return 'guarded';
    }

    /* Copyright 2019 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * User has clicked on a button in the header to change the plugin.
     */
    const changePlugin = createAction('[Core] Plugin Changed', props());
    /**
     * Plugin information in the hash is changed by user action.
     * e.g., user can use browser navigation button to change the hash.
     */
    const pluginUrlHashChanged = createAction('[Core] Plugin Url Hash Changed', props());
    const coreLoaded = createAction('[Core] Loaded');
    const manualReload = createAction('[Core] User Triggered Reload');
    const reload = createAction('[Core] Auto Reload');
    const pluginsListingRequested = createAction('[Core] PluginListing Fetch Requested');
    const pluginsListingLoaded = createAction('[Core] PluginListing Fetch Successful', props());
    const pluginsListingFailed = createAction('[Core] PluginListing Fetch Failed', props());
    /**
     * Action for when Environment data has been loaded from the WebApp server.
     */
    const environmentLoaded = createAction('[Core] Environment Fetch Successful', props());
    /**
     * Action for when user wants to enable/disable reload.
     */
    const toggleReloadEnabled = createAction('[Core] Reload Enable Toggled');
    /**
     * Action for when user wants to change the reload period.
     */
    const changeReloadPeriod = createAction('[Core] Reload Period Change', props());
    /**
     * Action for when user wants to an item count in a page of a paginated view.
     */
    const changePageSize = createAction('[Core] Page Size Change', props());
    /**
     * Action for when user wants to change the runs selection in the tf-runs-selector.
     *
     * Action is reserved for the wrapper for the Polymer based tf-runs-selector.
     */
    const polymerInteropRunSelectionChanged = createAction('[Core] Run Selection Changed', props());
    const fetchRunSucceeded = createAction('[Core] Run Fetch Successful', props());

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/keycodes/keycodes.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const BACKSPACE = 8;
    /** @type {?} */
    const TAB = 9;
    /** @type {?} */
    const ENTER = 13;
    /** @type {?} */
    const ESCAPE = 27;
    /** @type {?} */
    const SPACE = 32;
    /** @type {?} */
    const PAGE_UP = 33;
    /** @type {?} */
    const PAGE_DOWN = 34;
    /** @type {?} */
    const END = 35;
    /** @type {?} */
    const HOME = 36;
    /** @type {?} */
    const LEFT_ARROW = 37;
    /** @type {?} */
    const UP_ARROW = 38;
    /** @type {?} */
    const RIGHT_ARROW = 39;
    /** @type {?} */
    const DOWN_ARROW = 40;
    /** @type {?} */
    const DELETE = 46;
    /** @type {?} */
    const ZERO = 48;
    /** @type {?} */
    const NINE = 57;
    /** @type {?} */
    const A = 65;
    /** @type {?} */
    const Z = 90;

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/keycodes/modifiers.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google LLC All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * Checks whether a modifier key is pressed.
     * @param {?} event Event to be checked.
     * @param {...?} modifiers
     * @return {?}
     */
    function hasModifierKey(event, ...modifiers) {
        if (modifiers.length) {
            return modifiers.some((/**
             * @param {?} modifier
             * @return {?}
             */
            modifier => event[modifier]));
        }
        return event.altKey || event.shiftKey || event.ctrlKey || event.metaKey;
    }

    /**
     * @license
     * Copyright Google LLC All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /** Coerces a data-bound value (typically a string) to a boolean. */
    function coerceBooleanProperty(value) {
        return value != null && `${value}` !== 'false';
    }

    /**
     * @license
     * Copyright Google LLC All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    function coerceNumberProperty(value, fallbackValue = 0) {
        return _isNumberValue(value) ? Number(value) : fallbackValue;
    }
    /**
     * Whether the provided value is considered a number.
     * @docs-private
     */
    function _isNumberValue(value) {
        // parseFloat(value) handles most of the cases we're interested in (it treats null, empty string,
        // and other non-number values as NaN, where Number just uses 0) but it considers the string
        // '123hello' to be a valid number. Therefore we also check if Number(value) is NaN.
        return !isNaN(parseFloat(value)) && !isNaN(Number(value));
    }

    /**
     * @license
     * Copyright Google LLC All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    function coerceArray(value) {
        return Array.isArray(value) ? value : [value];
    }

    /**
     * @license
     * Copyright Google LLC All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /** Coerces a value to a CSS pixel value. */
    function coerceCssPixelValue(value) {
        if (value == null) {
            return '';
        }
        return typeof value === 'string' ? value : `${value}px`;
    }

    /**
     * @license
     * Copyright Google LLC All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * Coerces an ElementRef or an Element into an element.
     * Useful for APIs that can accept either a ref or the native element itself.
     */
    function coerceElement(elementOrRef) {
        return elementOrRef instanceof ElementRef ? elementOrRef.nativeElement : elementOrRef;
    }

    let hasV8BreakIterator;
    // We need a try/catch around the reference to `Intl`, because accessing it in some cases can
    // cause IE to throw. These cases are tied to particular versions of Windows and can happen if
    // the consumer is providing a polyfilled `Map`. See:
    // https://github.com/Microsoft/ChakraCore/issues/3189
    // https://github.com/angular/components/issues/15687
    try {
        hasV8BreakIterator = (typeof Intl !== 'undefined' && ((/** @type {?} */ (Intl))).v8BreakIterator);
    }
    catch (_a) {
        hasV8BreakIterator = false;
    }
    /**
     * Service to detect the current platform by comparing the userAgent strings and
     * checking browser-specific global properties.
     */
    class Platform {
        /**
         * \@breaking-change 8.0.0 remove optional decorator
         * @param {?=} _platformId
         */
        constructor(_platformId) {
            this._platformId = _platformId;
            // We want to use the Angular platform check because if the Document is shimmed
            // without the navigator, the following checks will fail. This is preferred because
            // sometimes the Document may be shimmed without the user's knowledge or intention
            /**
             * Whether the Angular application is being rendered in the browser.
             */
            this.isBrowser = this._platformId ?
                isPlatformBrowser(this._platformId) : typeof document === 'object' && !!document;
            /**
             * Whether the current browser is Microsoft Edge.
             */
            this.EDGE = this.isBrowser && /(edge)/i.test(navigator.userAgent);
            /**
             * Whether the current rendering engine is Microsoft Trident.
             */
            this.TRIDENT = this.isBrowser && /(msie|trident)/i.test(navigator.userAgent);
            // EdgeHTML and Trident mock Blink specific things and need to be excluded from this check.
            /**
             * Whether the current rendering engine is Blink.
             */
            this.BLINK = this.isBrowser && (!!(((/** @type {?} */ (window))).chrome || hasV8BreakIterator) &&
                typeof CSS !== 'undefined' && !this.EDGE && !this.TRIDENT);
            // Webkit is part of the userAgent in EdgeHTML, Blink and Trident. Therefore we need to
            // ensure that Webkit runs standalone and is not used as another engine's base.
            /**
             * Whether the current rendering engine is WebKit.
             */
            this.WEBKIT = this.isBrowser &&
                /AppleWebKit/i.test(navigator.userAgent) && !this.BLINK && !this.EDGE && !this.TRIDENT;
            /**
             * Whether the current platform is Apple iOS.
             */
            this.IOS = this.isBrowser && /iPad|iPhone|iPod/.test(navigator.userAgent) &&
                !('MSStream' in window);
            // It's difficult to detect the plain Gecko engine, because most of the browsers identify
            // them self as Gecko-like browsers and modify the userAgent's according to that.
            // Since we only cover one explicit Firefox case, we can simply check for Firefox
            // instead of having an unstable check for Gecko.
            /**
             * Whether the current browser is Firefox.
             */
            this.FIREFOX = this.isBrowser && /(firefox|minefield)/i.test(navigator.userAgent);
            /**
             * Whether the current platform is Android.
             */
            // Trident on mobile adds the android platform to the userAgent to trick detections.
            this.ANDROID = this.isBrowser && /android/i.test(navigator.userAgent) && !this.TRIDENT;
            // Safari browsers will include the Safari keyword in their userAgent. Some browsers may fake
            // this and just place the Safari keyword in the userAgent. To be more safe about Safari every
            // Safari browser should also use Webkit as its layout engine.
            /**
             * Whether the current browser is Safari.
             */
            this.SAFARI = this.isBrowser && /safari/i.test(navigator.userAgent) && this.WEBKIT;
        }
    }
    Platform.ɵfac = function Platform_Factory(t) { return new (t || Platform)(ɵɵinject(PLATFORM_ID, 8)); };
    /** @nocollapse */
    Platform.ctorParameters = () => [
        { type: Object, decorators: [{ type: Optional }, { type: Inject, args: [PLATFORM_ID,] }] }
    ];
    /** @nocollapse */ Platform.ɵprov = ɵɵdefineInjectable({ factory: function Platform_Factory() { return new Platform(ɵɵinject(PLATFORM_ID, 8)); }, token: Platform, providedIn: "root" });

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/platform/platform-module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class PlatformModule {
    }
    PlatformModule.ɵmod = ɵɵdefineNgModule({ type: PlatformModule });
    PlatformModule.ɵinj = ɵɵdefineInjector({ factory: function PlatformModule_Factory(t) { return new (t || PlatformModule)(); } });

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/platform/features/input-types.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google LLC All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * Cached result Set of input types support by the current browser.
     * @type {?}
     */
    let supportedInputTypes;
    /**
     * Types of `<input>` that *might* be supported.
     * @type {?}
     */
    const candidateInputTypes = [
        // `color` must come first. Chrome 56 shows a warning if we change the type to `color` after
        // first changing it to something else:
        // The specified value "" does not conform to the required format.
        // The format is "#rrggbb" where rr, gg, bb are two-digit hexadecimal numbers.
        'color',
        'button',
        'checkbox',
        'date',
        'datetime-local',
        'email',
        'file',
        'hidden',
        'image',
        'month',
        'number',
        'password',
        'radio',
        'range',
        'reset',
        'search',
        'submit',
        'tel',
        'text',
        'time',
        'url',
        'week',
    ];
    /**
     * @return {?} The input types supported by this browser.
     */
    function getSupportedInputTypes() {
        // Result is cached.
        if (supportedInputTypes) {
            return supportedInputTypes;
        }
        // We can't check if an input type is not supported until we're on the browser, so say that
        // everything is supported when not on the browser. We don't use `Platform` here since it's
        // just a helper function and can't inject it.
        if (typeof document !== 'object' || !document) {
            supportedInputTypes = new Set(candidateInputTypes);
            return supportedInputTypes;
        }
        /** @type {?} */
        let featureTestInput = document.createElement('input');
        supportedInputTypes = new Set(candidateInputTypes.filter((/**
         * @param {?} value
         * @return {?}
         */
        value => {
            featureTestInput.setAttribute('type', value);
            return featureTestInput.type === value;
        })));
        return supportedInputTypes;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/platform/features/passive-listeners.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google LLC All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * Cached result of whether the user's browser supports passive event listeners.
     * @type {?}
     */
    let supportsPassiveEvents;
    /**
     * Checks whether the user's browser supports passive event listeners.
     * See: https://github.com/WICG/EventListenerOptions/blob/gh-pages/explainer.md
     * @return {?}
     */
    function supportsPassiveEventListeners() {
        if (supportsPassiveEvents == null && typeof window !== 'undefined') {
            try {
                window.addEventListener('test', (/** @type {?} */ (null)), Object.defineProperty({}, 'passive', {
                    get: (/**
                     * @return {?}
                     */
                    () => supportsPassiveEvents = true)
                }));
            }
            finally {
                supportsPassiveEvents = supportsPassiveEvents || false;
            }
        }
        return supportsPassiveEvents;
    }
    /**
     * Normalizes an `AddEventListener` object to something that can be passed
     * to `addEventListener` on any browser, no matter whether it supports the
     * `options` parameter.
     * @param {?} options Object to be normalized.
     * @return {?}
     */
    function normalizePassiveListenerOptions(options) {
        return supportsPassiveEventListeners() ? options : !!options.capture;
    }
    /**
     * Cached result of the way the browser handles the horizontal scroll axis in RTL mode.
     * @type {?}
     */
    let rtlScrollAxisType;
    /**
     * Check whether the browser supports scroll behaviors.
     * @return {?}
     */
    function supportsScrollBehavior() {
        return !!(typeof document == 'object' && 'scrollBehavior' in (/** @type {?} */ (document.documentElement)).style);
    }
    /**
     * Checks the type of RTL scroll axis used by this browser. As of time of writing, Chrome is NORMAL,
     * Firefox & Safari are NEGATED, and IE & Edge are INVERTED.
     * @return {?}
     */
    function getRtlScrollAxisType() {
        // We can't check unless we're on the browser. Just assume 'normal' if we're not.
        if (typeof document !== 'object' || !document) {
            return 0 /* NORMAL */;
        }
        if (rtlScrollAxisType == null) {
            // Create a 1px wide scrolling container and a 2px wide content element.
            /** @type {?} */
            const scrollContainer = document.createElement('div');
            /** @type {?} */
            const containerStyle = scrollContainer.style;
            scrollContainer.dir = 'rtl';
            containerStyle.height = '1px';
            containerStyle.width = '1px';
            containerStyle.overflow = 'auto';
            containerStyle.visibility = 'hidden';
            containerStyle.pointerEvents = 'none';
            containerStyle.position = 'absolute';
            /** @type {?} */
            const content = document.createElement('div');
            /** @type {?} */
            const contentStyle = content.style;
            contentStyle.width = '2px';
            contentStyle.height = '1px';
            scrollContainer.appendChild(content);
            document.body.appendChild(scrollContainer);
            rtlScrollAxisType = 0 /* NORMAL */;
            // The viewport starts scrolled all the way to the right in RTL mode. If we are in a NORMAL
            // browser this would mean that the scrollLeft should be 1. If it's zero instead we know we're
            // dealing with one of the other two types of browsers.
            if (scrollContainer.scrollLeft === 0) {
                // In a NEGATED browser the scrollLeft is always somewhere in [-maxScrollAmount, 0]. For an
                // INVERTED browser it is always somewhere in [0, maxScrollAmount]. We can determine which by
                // setting to the scrollLeft to 1. This is past the max for a NEGATED browser, so it will
                // return 0 when we read it again.
                scrollContainer.scrollLeft = 1;
                rtlScrollAxisType =
                    scrollContainer.scrollLeft === 0 ? 1 /* NEGATED */ : 2 /* INVERTED */;
            }
            (/** @type {?} */ (scrollContainer.parentNode)).removeChild(scrollContainer);
        }
        return rtlScrollAxisType;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/platform/features/shadow-dom.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google LLC All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /** @type {?} */
    let shadowDomIsSupported;
    /**
     * Checks whether the user's browser support Shadow DOM.
     * @return {?}
     */
    function _supportsShadowDom() {
        if (shadowDomIsSupported == null) {
            /** @type {?} */
            const head = typeof document !== 'undefined' ? document.head : null;
            shadowDomIsSupported = !!(head && (((/** @type {?} */ (head))).createShadowRoot || head.attachShadow));
        }
        return shadowDomIsSupported;
    }
    /**
     * Gets the shadow root of an element, if supported and the element is inside the Shadow DOM.
     * @param {?} element
     * @return {?}
     */
    function _getShadowRoot(element) {
        if (_supportsShadowDom()) {
            /** @type {?} */
            const rootNode = element.getRootNode ? element.getRootNode() : null;
            // Note that this should be caught by `_supportsShadowDom`, but some
            // teams have been able to hit this code path on unsupported browsers.
            if (typeof ShadowRoot !== 'undefined' && ShadowRoot && rootNode instanceof ShadowRoot) {
                return rootNode;
            }
        }
        return null;
    }

    class MutationObserverFactory {
        /**
         * @param {?} callback
         * @return {?}
         */
        create(callback) {
            return typeof MutationObserver === 'undefined' ? null : new MutationObserver(callback);
        }
    }
    MutationObserverFactory.ɵfac = function MutationObserverFactory_Factory(t) { return new (t || MutationObserverFactory)(); };
    /** @nocollapse */ MutationObserverFactory.ɵprov = ɵɵdefineInjectable({ factory: function MutationObserverFactory_Factory() { return new MutationObserverFactory(); }, token: MutationObserverFactory, providedIn: "root" });
    /**
     * An injectable service that allows watching elements for changes to their content.
     */
    class ContentObserver {
        /**
         * @param {?} _mutationObserverFactory
         */
        constructor(_mutationObserverFactory) {
            this._mutationObserverFactory = _mutationObserverFactory;
            /**
             * Keeps track of the existing MutationObservers so they can be reused.
             */
            this._observedElements = new Map();
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this._observedElements.forEach((/**
             * @param {?} _
             * @param {?} element
             * @return {?}
             */
            (_, element) => this._cleanupObserver(element)));
        }
        /**
         * @param {?} elementOrRef
         * @return {?}
         */
        observe(elementOrRef) {
            /** @type {?} */
            const element = coerceElement(elementOrRef);
            return new Observable((/**
             * @param {?} observer
             * @return {?}
             */
            (observer) => {
                /** @type {?} */
                const stream = this._observeElement(element);
                /** @type {?} */
                const subscription = stream.subscribe(observer);
                return (/**
                 * @return {?}
                 */
                () => {
                    subscription.unsubscribe();
                    this._unobserveElement(element);
                });
            }));
        }
        /**
         * Observes the given element by using the existing MutationObserver if available, or creating a
         * new one if not.
         * @private
         * @param {?} element
         * @return {?}
         */
        _observeElement(element) {
            if (!this._observedElements.has(element)) {
                /** @type {?} */
                const stream = new Subject();
                /** @type {?} */
                const observer = this._mutationObserverFactory.create((/**
                 * @param {?} mutations
                 * @return {?}
                 */
                mutations => stream.next(mutations)));
                if (observer) {
                    observer.observe(element, {
                        characterData: true,
                        childList: true,
                        subtree: true
                    });
                }
                this._observedElements.set(element, { observer, stream, count: 1 });
            }
            else {
                (/** @type {?} */ (this._observedElements.get(element))).count++;
            }
            return (/** @type {?} */ (this._observedElements.get(element))).stream;
        }
        /**
         * Un-observes the given element and cleans up the underlying MutationObserver if nobody else is
         * observing this element.
         * @private
         * @param {?} element
         * @return {?}
         */
        _unobserveElement(element) {
            if (this._observedElements.has(element)) {
                (/** @type {?} */ (this._observedElements.get(element))).count--;
                if (!(/** @type {?} */ (this._observedElements.get(element))).count) {
                    this._cleanupObserver(element);
                }
            }
        }
        /**
         * Clean up the underlying MutationObserver for the specified element.
         * @private
         * @param {?} element
         * @return {?}
         */
        _cleanupObserver(element) {
            if (this._observedElements.has(element)) {
                const { observer, stream } = (/** @type {?} */ (this._observedElements.get(element)));
                if (observer) {
                    observer.disconnect();
                }
                stream.complete();
                this._observedElements.delete(element);
            }
        }
    }
    ContentObserver.ɵfac = function ContentObserver_Factory(t) { return new (t || ContentObserver)(ɵɵinject(MutationObserverFactory)); };
    /** @nocollapse */
    ContentObserver.ctorParameters = () => [
        { type: MutationObserverFactory }
    ];
    /** @nocollapse */ ContentObserver.ɵprov = ɵɵdefineInjectable({ factory: function ContentObserver_Factory() { return new ContentObserver(ɵɵinject(MutationObserverFactory)); }, token: ContentObserver, providedIn: "root" });
    /**
     * Directive that triggers a callback whenever the content of
     * its associated element has changed.
     */
    class CdkObserveContent {
        /**
         * @param {?} _contentObserver
         * @param {?} _elementRef
         * @param {?} _ngZone
         */
        constructor(_contentObserver, _elementRef, _ngZone) {
            this._contentObserver = _contentObserver;
            this._elementRef = _elementRef;
            this._ngZone = _ngZone;
            /**
             * Event emitted for each change in the element's content.
             */
            this.event = new EventEmitter();
            this._disabled = false;
            this._currentSubscription = null;
        }
        /**
         * Whether observing content is disabled. This option can be used
         * to disconnect the underlying MutationObserver until it is needed.
         * @return {?}
         */
        get disabled() { return this._disabled; }
        /**
         * @param {?} value
         * @return {?}
         */
        set disabled(value) {
            this._disabled = coerceBooleanProperty(value);
            this._disabled ? this._unsubscribe() : this._subscribe();
        }
        /**
         * Debounce interval for emitting the changes.
         * @return {?}
         */
        get debounce() { return this._debounce; }
        /**
         * @param {?} value
         * @return {?}
         */
        set debounce(value) {
            this._debounce = coerceNumberProperty(value);
            this._subscribe();
        }
        /**
         * @return {?}
         */
        ngAfterContentInit() {
            if (!this._currentSubscription && !this.disabled) {
                this._subscribe();
            }
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this._unsubscribe();
        }
        /**
         * @private
         * @return {?}
         */
        _subscribe() {
            this._unsubscribe();
            /** @type {?} */
            const stream = this._contentObserver.observe(this._elementRef);
            // TODO(mmalerba): We shouldn't be emitting on this @Output() outside the zone.
            // Consider brining it back inside the zone next time we're making breaking changes.
            // Bringing it back inside can cause things like infinite change detection loops and changed
            // after checked errors if people's code isn't handling it properly.
            this._ngZone.runOutsideAngular((/**
             * @return {?}
             */
            () => {
                this._currentSubscription =
                    (this.debounce ? stream.pipe(debounceTime(this.debounce)) : stream).subscribe(this.event);
            }));
        }
        /**
         * @private
         * @return {?}
         */
        _unsubscribe() {
            if (this._currentSubscription) {
                this._currentSubscription.unsubscribe();
            }
        }
    }
    CdkObserveContent.ɵfac = function CdkObserveContent_Factory(t) { return new (t || CdkObserveContent)(ɵɵdirectiveInject(ContentObserver), ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(NgZone)); };
    CdkObserveContent.ɵdir = ɵɵdefineDirective({ type: CdkObserveContent, selectors: [["", "cdkObserveContent", ""]], inputs: { disabled: ["cdkObserveContentDisabled", "disabled"], debounce: "debounce" }, outputs: { event: "cdkObserveContent" }, exportAs: ["cdkObserveContent"] });
    /** @nocollapse */
    CdkObserveContent.ctorParameters = () => [
        { type: ContentObserver },
        { type: ElementRef },
        { type: NgZone }
    ];
    CdkObserveContent.propDecorators = {
        event: [{ type: Output, args: ['cdkObserveContent',] }],
        disabled: [{ type: Input, args: ['cdkObserveContentDisabled',] }],
        debounce: [{ type: Input }]
    };
    class ObserversModule {
    }
    ObserversModule.ɵmod = ɵɵdefineNgModule({ type: ObserversModule });
    ObserversModule.ɵinj = ɵɵdefineInjector({ factory: function ObserversModule_Factory(t) { return new (t || ObserversModule)(); }, providers: [MutationObserverFactory] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(ObserversModule, { declarations: [CdkObserveContent], exports: [CdkObserveContent] }); })();

    const ID_DELIMITER = ' ';
    /**
     * Adds the given ID to the specified ARIA attribute on an element.
     * Used for attributes such as aria-labelledby, aria-owns, etc.
     * @param {?} el
     * @param {?} attr
     * @param {?} id
     * @return {?}
     */
    function addAriaReferencedId(el, attr, id) {
        /** @type {?} */
        const ids = getAriaReferenceIds(el, attr);
        if (ids.some((/**
         * @param {?} existingId
         * @return {?}
         */
        existingId => existingId.trim() == id.trim()))) {
            return;
        }
        ids.push(id.trim());
        el.setAttribute(attr, ids.join(ID_DELIMITER));
    }
    /**
     * Removes the given ID from the specified ARIA attribute on an element.
     * Used for attributes such as aria-labelledby, aria-owns, etc.
     * @param {?} el
     * @param {?} attr
     * @param {?} id
     * @return {?}
     */
    function removeAriaReferencedId(el, attr, id) {
        /** @type {?} */
        const ids = getAriaReferenceIds(el, attr);
        /** @type {?} */
        const filteredIds = ids.filter((/**
         * @param {?} val
         * @return {?}
         */
        val => val != id.trim()));
        if (filteredIds.length) {
            el.setAttribute(attr, filteredIds.join(ID_DELIMITER));
        }
        else {
            el.removeAttribute(attr);
        }
    }
    /**
     * Gets the list of IDs referenced by the given ARIA attribute on an element.
     * Used for attributes such as aria-labelledby, aria-owns, etc.
     * @param {?} el
     * @param {?} attr
     * @return {?}
     */
    function getAriaReferenceIds(el, attr) {
        // Get string array of all individual ids (whitespace delimited) in the attribute value
        return (el.getAttribute(attr) || '').match(/\S+/g) || [];
    }
    /**
     * ID used for the body container where all messages are appended.
     * @type {?}
     */
    const MESSAGES_CONTAINER_ID = 'cdk-describedby-message-container';
    /**
     * ID prefix used for each created message element.
     * @type {?}
     */
    const CDK_DESCRIBEDBY_ID_PREFIX = 'cdk-describedby-message';
    /**
     * Attribute given to each host element that is described by a message element.
     * @type {?}
     */
    const CDK_DESCRIBEDBY_HOST_ATTRIBUTE = 'cdk-describedby-host';
    /**
     * Global incremental identifier for each registered message element.
     * @type {?}
     */
    let nextId = 0;
    /**
     * Global map of all registered message elements that have been placed into the document.
     * @type {?}
     */
    const messageRegistry = new Map();
    /**
     * Container for all registered messages.
     * @type {?}
     */
    let messagesContainer = null;
    /**
     * Utility that creates visually hidden elements with a message content. Useful for elements that
     * want to use aria-describedby to further describe themselves without adding additional visual
     * content.
     */
    class AriaDescriber {
        /**
         * @param {?} _document
         */
        constructor(_document) {
            this._document = _document;
        }
        /**
         * Adds to the host element an aria-describedby reference to a hidden element that contains
         * the message. If the same message has already been registered, then it will reuse the created
         * message element.
         * @param {?} hostElement
         * @param {?} message
         * @return {?}
         */
        describe(hostElement, message) {
            if (!this._canBeDescribed(hostElement, message)) {
                return;
            }
            if (typeof message !== 'string') {
                // We need to ensure that the element has an ID.
                this._setMessageId(message);
                messageRegistry.set(message, { messageElement: message, referenceCount: 0 });
            }
            else if (!messageRegistry.has(message)) {
                this._createMessageElement(message);
            }
            if (!this._isElementDescribedByMessage(hostElement, message)) {
                this._addMessageReference(hostElement, message);
            }
        }
        /**
         * Removes the host element's aria-describedby reference to the message element.
         * @param {?} hostElement
         * @param {?} message
         * @return {?}
         */
        removeDescription(hostElement, message) {
            if (!this._isElementNode(hostElement)) {
                return;
            }
            if (this._isElementDescribedByMessage(hostElement, message)) {
                this._removeMessageReference(hostElement, message);
            }
            // If the message is a string, it means that it's one that we created for the
            // consumer so we can remove it safely, otherwise we should leave it in place.
            if (typeof message === 'string') {
                /** @type {?} */
                const registeredMessage = messageRegistry.get(message);
                if (registeredMessage && registeredMessage.referenceCount === 0) {
                    this._deleteMessageElement(message);
                }
            }
            if (messagesContainer && messagesContainer.childNodes.length === 0) {
                this._deleteMessagesContainer();
            }
        }
        /**
         * Unregisters all created message elements and removes the message container.
         * @return {?}
         */
        ngOnDestroy() {
            /** @type {?} */
            const describedElements = this._document.querySelectorAll(`[${CDK_DESCRIBEDBY_HOST_ATTRIBUTE}]`);
            for (let i = 0; i < describedElements.length; i++) {
                this._removeCdkDescribedByReferenceIds(describedElements[i]);
                describedElements[i].removeAttribute(CDK_DESCRIBEDBY_HOST_ATTRIBUTE);
            }
            if (messagesContainer) {
                this._deleteMessagesContainer();
            }
            messageRegistry.clear();
        }
        /**
         * Creates a new element in the visually hidden message container element with the message
         * as its content and adds it to the message registry.
         * @private
         * @param {?} message
         * @return {?}
         */
        _createMessageElement(message) {
            /** @type {?} */
            const messageElement = this._document.createElement('div');
            this._setMessageId(messageElement);
            messageElement.textContent = message;
            this._createMessagesContainer();
            (/** @type {?} */ (messagesContainer)).appendChild(messageElement);
            messageRegistry.set(message, { messageElement, referenceCount: 0 });
        }
        /**
         * Assigns a unique ID to an element, if it doesn't have one already.
         * @private
         * @param {?} element
         * @return {?}
         */
        _setMessageId(element) {
            if (!element.id) {
                element.id = `${CDK_DESCRIBEDBY_ID_PREFIX}-${nextId++}`;
            }
        }
        /**
         * Deletes the message element from the global messages container.
         * @private
         * @param {?} message
         * @return {?}
         */
        _deleteMessageElement(message) {
            /** @type {?} */
            const registeredMessage = messageRegistry.get(message);
            /** @type {?} */
            const messageElement = registeredMessage && registeredMessage.messageElement;
            if (messagesContainer && messageElement) {
                messagesContainer.removeChild(messageElement);
            }
            messageRegistry.delete(message);
        }
        /**
         * Creates the global container for all aria-describedby messages.
         * @private
         * @return {?}
         */
        _createMessagesContainer() {
            if (!messagesContainer) {
                /** @type {?} */
                const preExistingContainer = this._document.getElementById(MESSAGES_CONTAINER_ID);
                // When going from the server to the client, we may end up in a situation where there's
                // already a container on the page, but we don't have a reference to it. Clear the
                // old container so we don't get duplicates. Doing this, instead of emptying the previous
                // container, should be slightly faster.
                if (preExistingContainer) {
                    (/** @type {?} */ (preExistingContainer.parentNode)).removeChild(preExistingContainer);
                }
                messagesContainer = this._document.createElement('div');
                messagesContainer.id = MESSAGES_CONTAINER_ID;
                messagesContainer.setAttribute('aria-hidden', 'true');
                messagesContainer.style.display = 'none';
                this._document.body.appendChild(messagesContainer);
            }
        }
        /**
         * Deletes the global messages container.
         * @private
         * @return {?}
         */
        _deleteMessagesContainer() {
            if (messagesContainer && messagesContainer.parentNode) {
                messagesContainer.parentNode.removeChild(messagesContainer);
                messagesContainer = null;
            }
        }
        /**
         * Removes all cdk-describedby messages that are hosted through the element.
         * @private
         * @param {?} element
         * @return {?}
         */
        _removeCdkDescribedByReferenceIds(element) {
            // Remove all aria-describedby reference IDs that are prefixed by CDK_DESCRIBEDBY_ID_PREFIX
            /** @type {?} */
            const originalReferenceIds = getAriaReferenceIds(element, 'aria-describedby')
                .filter((/**
             * @param {?} id
             * @return {?}
             */
            id => id.indexOf(CDK_DESCRIBEDBY_ID_PREFIX) != 0));
            element.setAttribute('aria-describedby', originalReferenceIds.join(' '));
        }
        /**
         * Adds a message reference to the element using aria-describedby and increments the registered
         * message's reference count.
         * @private
         * @param {?} element
         * @param {?} message
         * @return {?}
         */
        _addMessageReference(element, message) {
            /** @type {?} */
            const registeredMessage = (/** @type {?} */ (messageRegistry.get(message)));
            // Add the aria-describedby reference and set the
            // describedby_host attribute to mark the element.
            addAriaReferencedId(element, 'aria-describedby', registeredMessage.messageElement.id);
            element.setAttribute(CDK_DESCRIBEDBY_HOST_ATTRIBUTE, '');
            registeredMessage.referenceCount++;
        }
        /**
         * Removes a message reference from the element using aria-describedby
         * and decrements the registered message's reference count.
         * @private
         * @param {?} element
         * @param {?} message
         * @return {?}
         */
        _removeMessageReference(element, message) {
            /** @type {?} */
            const registeredMessage = (/** @type {?} */ (messageRegistry.get(message)));
            registeredMessage.referenceCount--;
            removeAriaReferencedId(element, 'aria-describedby', registeredMessage.messageElement.id);
            element.removeAttribute(CDK_DESCRIBEDBY_HOST_ATTRIBUTE);
        }
        /**
         * Returns true if the element has been described by the provided message ID.
         * @private
         * @param {?} element
         * @param {?} message
         * @return {?}
         */
        _isElementDescribedByMessage(element, message) {
            /** @type {?} */
            const referenceIds = getAriaReferenceIds(element, 'aria-describedby');
            /** @type {?} */
            const registeredMessage = messageRegistry.get(message);
            /** @type {?} */
            const messageId = registeredMessage && registeredMessage.messageElement.id;
            return !!messageId && referenceIds.indexOf(messageId) != -1;
        }
        /**
         * Determines whether a message can be described on a particular element.
         * @private
         * @param {?} element
         * @param {?} message
         * @return {?}
         */
        _canBeDescribed(element, message) {
            if (!this._isElementNode(element)) {
                return false;
            }
            if (message && typeof message === 'object') {
                // We'd have to make some assumptions about the description element's text, if the consumer
                // passed in an element. Assume that if an element is passed in, the consumer has verified
                // that it can be used as a description.
                return true;
            }
            /** @type {?} */
            const trimmedMessage = message == null ? '' : `${message}`.trim();
            /** @type {?} */
            const ariaLabel = element.getAttribute('aria-label');
            // We shouldn't set descriptions if they're exactly the same as the `aria-label` of the
            // element, because screen readers will end up reading out the same text twice in a row.
            return trimmedMessage ? (!ariaLabel || ariaLabel.trim() !== trimmedMessage) : false;
        }
        /**
         * Checks whether a node is an Element node.
         * @private
         * @param {?} element
         * @return {?}
         */
        _isElementNode(element) {
            return element.nodeType === this._document.ELEMENT_NODE;
        }
    }
    AriaDescriber.ɵfac = function AriaDescriber_Factory(t) { return new (t || AriaDescriber)(ɵɵinject(DOCUMENT$1)); };
    /** @nocollapse */
    AriaDescriber.ctorParameters = () => [
        { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT$1,] }] }
    ];
    /** @nocollapse */ AriaDescriber.ɵprov = ɵɵdefineInjectable({ factory: function AriaDescriber_Factory() { return new AriaDescriber(ɵɵinject(DOCUMENT$1)); }, token: AriaDescriber, providedIn: "root" });
    /**
     * This class manages keyboard events for selectable lists. If you pass it a query list
     * of items, it will set the active item correctly when arrow events occur.
     * @template T
     */
    class ListKeyManager {
        /**
         * @param {?} _items
         */
        constructor(_items) {
            this._items = _items;
            this._activeItemIndex = -1;
            this._activeItem = null;
            this._wrap = false;
            this._letterKeyStream = new Subject();
            this._typeaheadSubscription = Subscription.EMPTY;
            this._vertical = true;
            this._allowedModifierKeys = [];
            /**
             * Predicate function that can be used to check whether an item should be skipped
             * by the key manager. By default, disabled items are skipped.
             */
            this._skipPredicateFn = (/**
             * @param {?} item
             * @return {?}
             */
            (item) => item.disabled);
            // Buffer for the letters that the user has pressed when the typeahead option is turned on.
            this._pressedLetters = [];
            /**
             * Stream that emits any time the TAB key is pressed, so components can react
             * when focus is shifted off of the list.
             */
            this.tabOut = new Subject();
            /**
             * Stream that emits whenever the active item of the list manager changes.
             */
            this.change = new Subject();
            // We allow for the items to be an array because, in some cases, the consumer may
            // not have access to a QueryList of the items they want to manage (e.g. when the
            // items aren't being collected via `ViewChildren` or `ContentChildren`).
            if (_items instanceof QueryList) {
                _items.changes.subscribe((/**
                 * @param {?} newItems
                 * @return {?}
                 */
                (newItems) => {
                    if (this._activeItem) {
                        /** @type {?} */
                        const itemArray = newItems.toArray();
                        /** @type {?} */
                        const newIndex = itemArray.indexOf(this._activeItem);
                        if (newIndex > -1 && newIndex !== this._activeItemIndex) {
                            this._activeItemIndex = newIndex;
                        }
                    }
                }));
            }
        }
        /**
         * Sets the predicate function that determines which items should be skipped by the
         * list key manager.
         * @template THIS
         * @this {THIS}
         * @param {?} predicate Function that determines whether the given item should be skipped.
         * @return {THIS}
         */
        skipPredicate(predicate) {
            (/** @type {?} */ (this))._skipPredicateFn = predicate;
            return (/** @type {?} */ (this));
        }
        /**
         * Configures wrapping mode, which determines whether the active item will wrap to
         * the other end of list when there are no more items in the given direction.
         * @template THIS
         * @this {THIS}
         * @param {?=} shouldWrap Whether the list should wrap when reaching the end.
         * @return {THIS}
         */
        withWrap(shouldWrap = true) {
            (/** @type {?} */ (this))._wrap = shouldWrap;
            return (/** @type {?} */ (this));
        }
        /**
         * Configures whether the key manager should be able to move the selection vertically.
         * @template THIS
         * @this {THIS}
         * @param {?=} enabled Whether vertical selection should be enabled.
         * @return {THIS}
         */
        withVerticalOrientation(enabled = true) {
            (/** @type {?} */ (this))._vertical = enabled;
            return (/** @type {?} */ (this));
        }
        /**
         * Configures the key manager to move the selection horizontally.
         * Passing in `null` will disable horizontal movement.
         * @template THIS
         * @this {THIS}
         * @param {?} direction Direction in which the selection can be moved.
         * @return {THIS}
         */
        withHorizontalOrientation(direction) {
            (/** @type {?} */ (this))._horizontal = direction;
            return (/** @type {?} */ (this));
        }
        /**
         * Modifier keys which are allowed to be held down and whose default actions will be prevented
         * as the user is pressing the arrow keys. Defaults to not allowing any modifier keys.
         * @template THIS
         * @this {THIS}
         * @param {?} keys
         * @return {THIS}
         */
        withAllowedModifierKeys(keys) {
            (/** @type {?} */ (this))._allowedModifierKeys = keys;
            return (/** @type {?} */ (this));
        }
        /**
         * Turns on typeahead mode which allows users to set the active item by typing.
         * @template THIS
         * @this {THIS}
         * @param {?=} debounceInterval Time to wait after the last keystroke before setting the active item.
         * @return {THIS}
         */
        withTypeAhead(debounceInterval = 200) {
            if ((/** @type {?} */ (this))._items.length && (/** @type {?} */ (this))._items.some((/**
             * @param {?} item
             * @return {?}
             */
            item => typeof item.getLabel !== 'function'))) {
                throw Error('ListKeyManager items in typeahead mode must implement the `getLabel` method.');
            }
            (/** @type {?} */ (this))._typeaheadSubscription.unsubscribe();
            // Debounce the presses of non-navigational keys, collect the ones that correspond to letters
            // and convert those letters back into a string. Afterwards find the first item that starts
            // with that string and select it.
            (/** @type {?} */ (this))._typeaheadSubscription = (/** @type {?} */ (this))._letterKeyStream.pipe(tap((/**
             * @param {?} letter
             * @return {?}
             */
            letter => (/** @type {?} */ (this))._pressedLetters.push(letter))), debounceTime(debounceInterval), filter((/**
             * @return {?}
             */
            () => (/** @type {?} */ (this))._pressedLetters.length > 0)), map((/**
             * @return {?}
             */
            () => (/** @type {?} */ (this))._pressedLetters.join('')))).subscribe((/**
             * @param {?} inputString
             * @return {?}
             */
            inputString => {
                /** @type {?} */
                const items = (/** @type {?} */ (this))._getItemsArray();
                // Start at 1 because we want to start searching at the item immediately
                // following the current active item.
                for (let i = 1; i < items.length + 1; i++) {
                    /** @type {?} */
                    const index = ((/** @type {?} */ (this))._activeItemIndex + i) % items.length;
                    /** @type {?} */
                    const item = items[index];
                    if (!(/** @type {?} */ (this))._skipPredicateFn(item) &&
                        (/** @type {?} */ (item.getLabel))().toUpperCase().trim().indexOf(inputString) === 0) {
                        (/** @type {?} */ (this)).setActiveItem(index);
                        break;
                    }
                }
                (/** @type {?} */ (this))._pressedLetters = [];
            }));
            return (/** @type {?} */ (this));
        }
        /**
         * @param {?} item
         * @return {?}
         */
        setActiveItem(item) {
            /** @type {?} */
            const previousIndex = this._activeItemIndex;
            this.updateActiveItem(item);
            if (this._activeItemIndex !== previousIndex) {
                this.change.next(this._activeItemIndex);
            }
        }
        /**
         * Sets the active item depending on the key event passed in.
         * @param {?} event Keyboard event to be used for determining which element should be active.
         * @return {?}
         */
        onKeydown(event) {
            /** @type {?} */
            const keyCode = event.keyCode;
            /** @type {?} */
            const modifiers = ['altKey', 'ctrlKey', 'metaKey', 'shiftKey'];
            /** @type {?} */
            const isModifierAllowed = modifiers.every((/**
             * @param {?} modifier
             * @return {?}
             */
            modifier => {
                return !event[modifier] || this._allowedModifierKeys.indexOf(modifier) > -1;
            }));
            switch (keyCode) {
                case TAB:
                    this.tabOut.next();
                    return;
                case DOWN_ARROW:
                    if (this._vertical && isModifierAllowed) {
                        this.setNextItemActive();
                        break;
                    }
                    else {
                        return;
                    }
                case UP_ARROW:
                    if (this._vertical && isModifierAllowed) {
                        this.setPreviousItemActive();
                        break;
                    }
                    else {
                        return;
                    }
                case RIGHT_ARROW:
                    if (this._horizontal && isModifierAllowed) {
                        this._horizontal === 'rtl' ? this.setPreviousItemActive() : this.setNextItemActive();
                        break;
                    }
                    else {
                        return;
                    }
                case LEFT_ARROW:
                    if (this._horizontal && isModifierAllowed) {
                        this._horizontal === 'rtl' ? this.setNextItemActive() : this.setPreviousItemActive();
                        break;
                    }
                    else {
                        return;
                    }
                default:
                    if (isModifierAllowed || hasModifierKey(event, 'shiftKey')) {
                        // Attempt to use the `event.key` which also maps it to the user's keyboard language,
                        // otherwise fall back to resolving alphanumeric characters via the keyCode.
                        if (event.key && event.key.length === 1) {
                            this._letterKeyStream.next(event.key.toLocaleUpperCase());
                        }
                        else if ((keyCode >= A && keyCode <= Z) || (keyCode >= ZERO && keyCode <= NINE)) {
                            this._letterKeyStream.next(String.fromCharCode(keyCode));
                        }
                    }
                    // Note that we return here, in order to avoid preventing
                    // the default action of non-navigational keys.
                    return;
            }
            this._pressedLetters = [];
            event.preventDefault();
        }
        /**
         * Index of the currently active item.
         * @return {?}
         */
        get activeItemIndex() {
            return this._activeItemIndex;
        }
        /**
         * The active item.
         * @return {?}
         */
        get activeItem() {
            return this._activeItem;
        }
        /**
         * Gets whether the user is currently typing into the manager using the typeahead feature.
         * @return {?}
         */
        isTyping() {
            return this._pressedLetters.length > 0;
        }
        /**
         * Sets the active item to the first enabled item in the list.
         * @return {?}
         */
        setFirstItemActive() {
            this._setActiveItemByIndex(0, 1);
        }
        /**
         * Sets the active item to the last enabled item in the list.
         * @return {?}
         */
        setLastItemActive() {
            this._setActiveItemByIndex(this._items.length - 1, -1);
        }
        /**
         * Sets the active item to the next enabled item in the list.
         * @return {?}
         */
        setNextItemActive() {
            this._activeItemIndex < 0 ? this.setFirstItemActive() : this._setActiveItemByDelta(1);
        }
        /**
         * Sets the active item to a previous enabled item in the list.
         * @return {?}
         */
        setPreviousItemActive() {
            this._activeItemIndex < 0 && this._wrap ? this.setLastItemActive()
                : this._setActiveItemByDelta(-1);
        }
        /**
         * @param {?} item
         * @return {?}
         */
        updateActiveItem(item) {
            /** @type {?} */
            const itemArray = this._getItemsArray();
            /** @type {?} */
            const index = typeof item === 'number' ? item : itemArray.indexOf(item);
            /** @type {?} */
            const activeItem = itemArray[index];
            // Explicitly check for `null` and `undefined` because other falsy values are valid.
            this._activeItem = activeItem == null ? null : activeItem;
            this._activeItemIndex = index;
        }
        /**
         * This method sets the active item, given a list of items and the delta between the
         * currently active item and the new active item. It will calculate differently
         * depending on whether wrap mode is turned on.
         * @private
         * @param {?} delta
         * @return {?}
         */
        _setActiveItemByDelta(delta) {
            this._wrap ? this._setActiveInWrapMode(delta) : this._setActiveInDefaultMode(delta);
        }
        /**
         * Sets the active item properly given "wrap" mode. In other words, it will continue to move
         * down the list until it finds an item that is not disabled, and it will wrap if it
         * encounters either end of the list.
         * @private
         * @param {?} delta
         * @return {?}
         */
        _setActiveInWrapMode(delta) {
            /** @type {?} */
            const items = this._getItemsArray();
            for (let i = 1; i <= items.length; i++) {
                /** @type {?} */
                const index = (this._activeItemIndex + (delta * i) + items.length) % items.length;
                /** @type {?} */
                const item = items[index];
                if (!this._skipPredicateFn(item)) {
                    this.setActiveItem(index);
                    return;
                }
            }
        }
        /**
         * Sets the active item properly given the default mode. In other words, it will
         * continue to move down the list until it finds an item that is not disabled. If
         * it encounters either end of the list, it will stop and not wrap.
         * @private
         * @param {?} delta
         * @return {?}
         */
        _setActiveInDefaultMode(delta) {
            this._setActiveItemByIndex(this._activeItemIndex + delta, delta);
        }
        /**
         * Sets the active item to the first enabled item starting at the index specified. If the
         * item is disabled, it will move in the fallbackDelta direction until it either
         * finds an enabled item or encounters the end of the list.
         * @private
         * @param {?} index
         * @param {?} fallbackDelta
         * @return {?}
         */
        _setActiveItemByIndex(index, fallbackDelta) {
            /** @type {?} */
            const items = this._getItemsArray();
            if (!items[index]) {
                return;
            }
            while (this._skipPredicateFn(items[index])) {
                index += fallbackDelta;
                if (!items[index]) {
                    return;
                }
            }
            this.setActiveItem(index);
        }
        /**
         * Returns the items as an array.
         * @private
         * @return {?}
         */
        _getItemsArray() {
            return this._items instanceof QueryList ? this._items.toArray() : this._items;
        }
    }
    /**
     * @template T
     */
    class ActiveDescendantKeyManager extends ListKeyManager {
        /**
         * @param {?} index
         * @return {?}
         */
        setActiveItem(index) {
            if (this.activeItem) {
                this.activeItem.setInactiveStyles();
            }
            super.setActiveItem(index);
            if (this.activeItem) {
                this.activeItem.setActiveStyles();
            }
        }
    }
    /**
     * @template T
     */
    class FocusKeyManager extends ListKeyManager {
        constructor() {
            super(...arguments);
            this._origin = 'program';
        }
        /**
         * Sets the focus origin that will be passed in to the items for any subsequent `focus` calls.
         * @template THIS
         * @this {THIS}
         * @param {?} origin Focus origin to be used when focusing items.
         * @return {THIS}
         */
        setFocusOrigin(origin) {
            (/** @type {?} */ (this))._origin = origin;
            return (/** @type {?} */ (this));
        }
        /**
         * @param {?} item
         * @return {?}
         */
        setActiveItem(item) {
            super.setActiveItem(item);
            if (this.activeItem) {
                this.activeItem.focus(this._origin);
            }
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/a11y/interactivity-checker/interactivity-checker.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    // The InteractivityChecker leans heavily on the ally.js accessibility utilities.
    // Methods like `isTabbable` are only covering specific edge-cases for the browsers which are
    // supported.
    /**
     * Utility for checking the interactivity of an element, such as whether is is focusable or
     * tabbable.
     */
    class InteractivityChecker {
        /**
         * @param {?} _platform
         */
        constructor(_platform) {
            this._platform = _platform;
        }
        /**
         * Gets whether an element is disabled.
         *
         * @param {?} element Element to be checked.
         * @return {?} Whether the element is disabled.
         */
        isDisabled(element) {
            // This does not capture some cases, such as a non-form control with a disabled attribute or
            // a form control inside of a disabled form, but should capture the most common cases.
            return element.hasAttribute('disabled');
        }
        /**
         * Gets whether an element is visible for the purposes of interactivity.
         *
         * This will capture states like `display: none` and `visibility: hidden`, but not things like
         * being clipped by an `overflow: hidden` parent or being outside the viewport.
         *
         * @param {?} element
         * @return {?} Whether the element is visible.
         */
        isVisible(element) {
            return hasGeometry(element) && getComputedStyle(element).visibility === 'visible';
        }
        /**
         * Gets whether an element can be reached via Tab key.
         * Assumes that the element has already been checked with isFocusable.
         *
         * @param {?} element Element to be checked.
         * @return {?} Whether the element is tabbable.
         */
        isTabbable(element) {
            // Nothing is tabbable on the server 😎
            if (!this._platform.isBrowser) {
                return false;
            }
            /** @type {?} */
            const frameElement = getFrameElement(getWindow(element));
            if (frameElement) {
                /** @type {?} */
                const frameType = frameElement && frameElement.nodeName.toLowerCase();
                // Frame elements inherit their tabindex onto all child elements.
                if (getTabIndexValue(frameElement) === -1) {
                    return false;
                }
                // Webkit and Blink consider anything inside of an <object> element as non-tabbable.
                if ((this._platform.BLINK || this._platform.WEBKIT) && frameType === 'object') {
                    return false;
                }
                // Webkit and Blink disable tabbing to an element inside of an invisible frame.
                if ((this._platform.BLINK || this._platform.WEBKIT) && !this.isVisible(frameElement)) {
                    return false;
                }
            }
            /** @type {?} */
            let nodeName = element.nodeName.toLowerCase();
            /** @type {?} */
            let tabIndexValue = getTabIndexValue(element);
            if (element.hasAttribute('contenteditable')) {
                return tabIndexValue !== -1;
            }
            if (nodeName === 'iframe') {
                // The frames may be tabbable depending on content, but it's not possibly to reliably
                // investigate the content of the frames.
                return false;
            }
            if (nodeName === 'audio') {
                if (!element.hasAttribute('controls')) {
                    // By default an <audio> element without the controls enabled is not tabbable.
                    return false;
                }
                else if (this._platform.BLINK) {
                    // In Blink <audio controls> elements are always tabbable.
                    return true;
                }
            }
            if (nodeName === 'video') {
                if (!element.hasAttribute('controls') && this._platform.TRIDENT) {
                    // In Trident a <video> element without the controls enabled is not tabbable.
                    return false;
                }
                else if (this._platform.BLINK || this._platform.FIREFOX) {
                    // In Chrome and Firefox <video controls> elements are always tabbable.
                    return true;
                }
            }
            if (nodeName === 'object' && (this._platform.BLINK || this._platform.WEBKIT)) {
                // In all Blink and WebKit based browsers <object> elements are never tabbable.
                return false;
            }
            // In iOS the browser only considers some specific elements as tabbable.
            if (this._platform.WEBKIT && this._platform.IOS && !isPotentiallyTabbableIOS(element)) {
                return false;
            }
            return element.tabIndex >= 0;
        }
        /**
         * Gets whether an element can be focused by the user.
         *
         * @param {?} element Element to be checked.
         * @return {?} Whether the element is focusable.
         */
        isFocusable(element) {
            // Perform checks in order of left to most expensive.
            // Again, naive approach that does not capture many edge cases and browser quirks.
            return isPotentiallyFocusable(element) && !this.isDisabled(element) && this.isVisible(element);
        }
    }
    InteractivityChecker.ɵfac = function InteractivityChecker_Factory(t) { return new (t || InteractivityChecker)(ɵɵinject(Platform)); };
    /** @nocollapse */
    InteractivityChecker.ctorParameters = () => [
        { type: Platform }
    ];
    /** @nocollapse */ InteractivityChecker.ɵprov = ɵɵdefineInjectable({ factory: function InteractivityChecker_Factory() { return new InteractivityChecker(ɵɵinject(Platform)); }, token: InteractivityChecker, providedIn: "root" });
    /**
     * Returns the frame element from a window object. Since browsers like MS Edge throw errors if
     * the frameElement property is being accessed from a different host address, this property
     * should be accessed carefully.
     * @param {?} window
     * @return {?}
     */
    function getFrameElement(window) {
        try {
            return (/** @type {?} */ (window.frameElement));
        }
        catch (_a) {
            return null;
        }
    }
    /**
     * Checks whether the specified element has any geometry / rectangles.
     * @param {?} element
     * @return {?}
     */
    function hasGeometry(element) {
        // Use logic from jQuery to check for an invisible element.
        // See https://github.com/jquery/jquery/blob/master/src/css/hiddenVisibleSelectors.js#L12
        return !!(element.offsetWidth || element.offsetHeight ||
            (typeof element.getClientRects === 'function' && element.getClientRects().length));
    }
    /**
     * Gets whether an element's
     * @param {?} element
     * @return {?}
     */
    function isNativeFormElement(element) {
        /** @type {?} */
        let nodeName = element.nodeName.toLowerCase();
        return nodeName === 'input' ||
            nodeName === 'select' ||
            nodeName === 'button' ||
            nodeName === 'textarea';
    }
    /**
     * Gets whether an element is an `<input type="hidden">`.
     * @param {?} element
     * @return {?}
     */
    function isHiddenInput(element) {
        return isInputElement(element) && element.type == 'hidden';
    }
    /**
     * Gets whether an element is an anchor that has an href attribute.
     * @param {?} element
     * @return {?}
     */
    function isAnchorWithHref(element) {
        return isAnchorElement(element) && element.hasAttribute('href');
    }
    /**
     * Gets whether an element is an input element.
     * @param {?} element
     * @return {?}
     */
    function isInputElement(element) {
        return element.nodeName.toLowerCase() == 'input';
    }
    /**
     * Gets whether an element is an anchor element.
     * @param {?} element
     * @return {?}
     */
    function isAnchorElement(element) {
        return element.nodeName.toLowerCase() == 'a';
    }
    /**
     * Gets whether an element has a valid tabindex.
     * @param {?} element
     * @return {?}
     */
    function hasValidTabIndex(element) {
        if (!element.hasAttribute('tabindex') || element.tabIndex === undefined) {
            return false;
        }
        /** @type {?} */
        let tabIndex = element.getAttribute('tabindex');
        // IE11 parses tabindex="" as the value "-32768"
        if (tabIndex == '-32768') {
            return false;
        }
        return !!(tabIndex && !isNaN(parseInt(tabIndex, 10)));
    }
    /**
     * Returns the parsed tabindex from the element attributes instead of returning the
     * evaluated tabindex from the browsers defaults.
     * @param {?} element
     * @return {?}
     */
    function getTabIndexValue(element) {
        if (!hasValidTabIndex(element)) {
            return null;
        }
        // See browser issue in Gecko https://bugzilla.mozilla.org/show_bug.cgi?id=1128054
        /** @type {?} */
        const tabIndex = parseInt(element.getAttribute('tabindex') || '', 10);
        return isNaN(tabIndex) ? -1 : tabIndex;
    }
    /**
     * Checks whether the specified element is potentially tabbable on iOS
     * @param {?} element
     * @return {?}
     */
    function isPotentiallyTabbableIOS(element) {
        /** @type {?} */
        let nodeName = element.nodeName.toLowerCase();
        /** @type {?} */
        let inputType = nodeName === 'input' && ((/** @type {?} */ (element))).type;
        return inputType === 'text'
            || inputType === 'password'
            || nodeName === 'select'
            || nodeName === 'textarea';
    }
    /**
     * Gets whether an element is potentially focusable without taking current visible/disabled state
     * into account.
     * @param {?} element
     * @return {?}
     */
    function isPotentiallyFocusable(element) {
        // Inputs are potentially focusable *unless* they're type="hidden".
        if (isHiddenInput(element)) {
            return false;
        }
        return isNativeFormElement(element) ||
            isAnchorWithHref(element) ||
            element.hasAttribute('contenteditable') ||
            hasValidTabIndex(element);
    }
    /**
     * Gets the parent window of a DOM node with regards of being inside of an iframe.
     * @param {?} node
     * @return {?}
     */
    function getWindow(node) {
        // ownerDocument is null if `node` itself *is* a document.
        return node.ownerDocument && node.ownerDocument.defaultView || window;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/a11y/focus-trap/focus-trap.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Class that allows for trapping focus within a DOM element.
     *
     * This class currently uses a relatively simple approach to focus trapping.
     * It assumes that the tab order is the same as DOM order, which is not necessarily true.
     * Things like `tabIndex > 0`, flex `order`, and shadow roots can cause to two to misalign.
     *
     * @deprecated Use `ConfigurableFocusTrap` instead.
     * \@breaking-change for 11.0.0 Remove this class.
     */
    class FocusTrap {
        /**
         * @param {?} _element
         * @param {?} _checker
         * @param {?} _ngZone
         * @param {?} _document
         * @param {?=} deferAnchors
         */
        constructor(_element, _checker, _ngZone, _document, deferAnchors = false) {
            this._element = _element;
            this._checker = _checker;
            this._ngZone = _ngZone;
            this._document = _document;
            this._hasAttached = false;
            // Event listeners for the anchors. Need to be regular functions so that we can unbind them later.
            this.startAnchorListener = (/**
             * @return {?}
             */
            () => this.focusLastTabbableElement());
            this.endAnchorListener = (/**
             * @return {?}
             */
            () => this.focusFirstTabbableElement());
            this._enabled = true;
            if (!deferAnchors) {
                this.attachAnchors();
            }
        }
        /**
         * Whether the focus trap is active.
         * @return {?}
         */
        get enabled() { return this._enabled; }
        /**
         * @param {?} value
         * @return {?}
         */
        set enabled(value) {
            this._enabled = value;
            if (this._startAnchor && this._endAnchor) {
                this._toggleAnchorTabIndex(value, this._startAnchor);
                this._toggleAnchorTabIndex(value, this._endAnchor);
            }
        }
        /**
         * Destroys the focus trap by cleaning up the anchors.
         * @return {?}
         */
        destroy() {
            /** @type {?} */
            const startAnchor = this._startAnchor;
            /** @type {?} */
            const endAnchor = this._endAnchor;
            if (startAnchor) {
                startAnchor.removeEventListener('focus', this.startAnchorListener);
                if (startAnchor.parentNode) {
                    startAnchor.parentNode.removeChild(startAnchor);
                }
            }
            if (endAnchor) {
                endAnchor.removeEventListener('focus', this.endAnchorListener);
                if (endAnchor.parentNode) {
                    endAnchor.parentNode.removeChild(endAnchor);
                }
            }
            this._startAnchor = this._endAnchor = null;
        }
        /**
         * Inserts the anchors into the DOM. This is usually done automatically
         * in the constructor, but can be deferred for cases like directives with `*ngIf`.
         * @return {?} Whether the focus trap managed to attach successfuly. This may not be the case
         * if the target element isn't currently in the DOM.
         */
        attachAnchors() {
            // If we're not on the browser, there can be no focus to trap.
            if (this._hasAttached) {
                return true;
            }
            this._ngZone.runOutsideAngular((/**
             * @return {?}
             */
            () => {
                if (!this._startAnchor) {
                    this._startAnchor = this._createAnchor();
                    (/** @type {?} */ (this._startAnchor)).addEventListener('focus', this.startAnchorListener);
                }
                if (!this._endAnchor) {
                    this._endAnchor = this._createAnchor();
                    (/** @type {?} */ (this._endAnchor)).addEventListener('focus', this.endAnchorListener);
                }
            }));
            if (this._element.parentNode) {
                this._element.parentNode.insertBefore((/** @type {?} */ (this._startAnchor)), this._element);
                this._element.parentNode.insertBefore((/** @type {?} */ (this._endAnchor)), this._element.nextSibling);
                this._hasAttached = true;
            }
            return this._hasAttached;
        }
        /**
         * Waits for the zone to stabilize, then either focuses the first element that the
         * user specified, or the first tabbable element.
         * @return {?} Returns a promise that resolves with a boolean, depending
         * on whether focus was moved successfuly.
         */
        focusInitialElementWhenReady() {
            return new Promise((/**
             * @param {?} resolve
             * @return {?}
             */
            resolve => {
                this._executeOnStable((/**
                 * @return {?}
                 */
                () => resolve(this.focusInitialElement())));
            }));
        }
        /**
         * Waits for the zone to stabilize, then focuses
         * the first tabbable element within the focus trap region.
         * @return {?} Returns a promise that resolves with a boolean, depending
         * on whether focus was moved successfuly.
         */
        focusFirstTabbableElementWhenReady() {
            return new Promise((/**
             * @param {?} resolve
             * @return {?}
             */
            resolve => {
                this._executeOnStable((/**
                 * @return {?}
                 */
                () => resolve(this.focusFirstTabbableElement())));
            }));
        }
        /**
         * Waits for the zone to stabilize, then focuses
         * the last tabbable element within the focus trap region.
         * @return {?} Returns a promise that resolves with a boolean, depending
         * on whether focus was moved successfuly.
         */
        focusLastTabbableElementWhenReady() {
            return new Promise((/**
             * @param {?} resolve
             * @return {?}
             */
            resolve => {
                this._executeOnStable((/**
                 * @return {?}
                 */
                () => resolve(this.focusLastTabbableElement())));
            }));
        }
        /**
         * Get the specified boundary element of the trapped region.
         * @private
         * @param {?} bound The boundary to get (start or end of trapped region).
         * @return {?} The boundary element.
         */
        _getRegionBoundary(bound) {
            // Contains the deprecated version of selector, for temporary backwards comparability.
            /** @type {?} */
            let markers = (/** @type {?} */ (this._element.querySelectorAll(`[cdk-focus-region-${bound}], ` +
                `[cdkFocusRegion${bound}], ` +
                `[cdk-focus-${bound}]`)));
            for (let i = 0; i < markers.length; i++) {
                // @breaking-change 8.0.0
                if (markers[i].hasAttribute(`cdk-focus-${bound}`)) {
                    console.warn(`Found use of deprecated attribute 'cdk-focus-${bound}', ` +
                        `use 'cdkFocusRegion${bound}' instead. The deprecated ` +
                        `attribute will be removed in 8.0.0.`, markers[i]);
                }
                else if (markers[i].hasAttribute(`cdk-focus-region-${bound}`)) {
                    console.warn(`Found use of deprecated attribute 'cdk-focus-region-${bound}', ` +
                        `use 'cdkFocusRegion${bound}' instead. The deprecated attribute ` +
                        `will be removed in 8.0.0.`, markers[i]);
                }
            }
            if (bound == 'start') {
                return markers.length ? markers[0] : this._getFirstTabbableElement(this._element);
            }
            return markers.length ?
                markers[markers.length - 1] : this._getLastTabbableElement(this._element);
        }
        /**
         * Focuses the element that should be focused when the focus trap is initialized.
         * @return {?} Whether focus was moved successfuly.
         */
        focusInitialElement() {
            // Contains the deprecated version of selector, for temporary backwards comparability.
            /** @type {?} */
            const redirectToElement = (/** @type {?} */ (this._element.querySelector(`[cdk-focus-initial], ` +
                `[cdkFocusInitial]`)));
            if (redirectToElement) {
                // @breaking-change 8.0.0
                if (redirectToElement.hasAttribute(`cdk-focus-initial`)) {
                    console.warn(`Found use of deprecated attribute 'cdk-focus-initial', ` +
                        `use 'cdkFocusInitial' instead. The deprecated attribute ` +
                        `will be removed in 8.0.0`, redirectToElement);
                }
                // Warn the consumer if the element they've pointed to
                // isn't focusable, when not in production mode.
                if (isDevMode() && !this._checker.isFocusable(redirectToElement)) {
                    console.warn(`Element matching '[cdkFocusInitial]' is not focusable.`, redirectToElement);
                }
                redirectToElement.focus();
                return true;
            }
            return this.focusFirstTabbableElement();
        }
        /**
         * Focuses the first tabbable element within the focus trap region.
         * @return {?} Whether focus was moved successfuly.
         */
        focusFirstTabbableElement() {
            /** @type {?} */
            const redirectToElement = this._getRegionBoundary('start');
            if (redirectToElement) {
                redirectToElement.focus();
            }
            return !!redirectToElement;
        }
        /**
         * Focuses the last tabbable element within the focus trap region.
         * @return {?} Whether focus was moved successfuly.
         */
        focusLastTabbableElement() {
            /** @type {?} */
            const redirectToElement = this._getRegionBoundary('end');
            if (redirectToElement) {
                redirectToElement.focus();
            }
            return !!redirectToElement;
        }
        /**
         * Checks whether the focus trap has successfuly been attached.
         * @return {?}
         */
        hasAttached() {
            return this._hasAttached;
        }
        /**
         * Get the first tabbable element from a DOM subtree (inclusive).
         * @private
         * @param {?} root
         * @return {?}
         */
        _getFirstTabbableElement(root) {
            if (this._checker.isFocusable(root) && this._checker.isTabbable(root)) {
                return root;
            }
            // Iterate in DOM order. Note that IE doesn't have `children` for SVG so we fall
            // back to `childNodes` which includes text nodes, comments etc.
            /** @type {?} */
            let children = root.children || root.childNodes;
            for (let i = 0; i < children.length; i++) {
                /** @type {?} */
                let tabbableChild = children[i].nodeType === this._document.ELEMENT_NODE ?
                    this._getFirstTabbableElement((/** @type {?} */ (children[i]))) :
                    null;
                if (tabbableChild) {
                    return tabbableChild;
                }
            }
            return null;
        }
        /**
         * Get the last tabbable element from a DOM subtree (inclusive).
         * @private
         * @param {?} root
         * @return {?}
         */
        _getLastTabbableElement(root) {
            if (this._checker.isFocusable(root) && this._checker.isTabbable(root)) {
                return root;
            }
            // Iterate in reverse DOM order.
            /** @type {?} */
            let children = root.children || root.childNodes;
            for (let i = children.length - 1; i >= 0; i--) {
                /** @type {?} */
                let tabbableChild = children[i].nodeType === this._document.ELEMENT_NODE ?
                    this._getLastTabbableElement((/** @type {?} */ (children[i]))) :
                    null;
                if (tabbableChild) {
                    return tabbableChild;
                }
            }
            return null;
        }
        /**
         * Creates an anchor element.
         * @private
         * @return {?}
         */
        _createAnchor() {
            /** @type {?} */
            const anchor = this._document.createElement('div');
            this._toggleAnchorTabIndex(this._enabled, anchor);
            anchor.classList.add('cdk-visually-hidden');
            anchor.classList.add('cdk-focus-trap-anchor');
            anchor.setAttribute('aria-hidden', 'true');
            return anchor;
        }
        /**
         * Toggles the `tabindex` of an anchor, based on the enabled state of the focus trap.
         * @private
         * @param {?} isEnabled Whether the focus trap is enabled.
         * @param {?} anchor Anchor on which to toggle the tabindex.
         * @return {?}
         */
        _toggleAnchorTabIndex(isEnabled, anchor) {
            // Remove the tabindex completely, rather than setting it to -1, because if the
            // element has a tabindex, the user might still hit it when navigating with the arrow keys.
            isEnabled ? anchor.setAttribute('tabindex', '0') : anchor.removeAttribute('tabindex');
        }
        /**
         * Toggles the`tabindex` of both anchors to either trap Tab focus or allow it to escape.
         * @protected
         * @param {?} enabled
         * @return {?}
         */
        toggleAnchors(enabled) {
            if (this._startAnchor && this._endAnchor) {
                this._toggleAnchorTabIndex(enabled, this._startAnchor);
                this._toggleAnchorTabIndex(enabled, this._endAnchor);
            }
        }
        /**
         * Executes a function when the zone is stable.
         * @private
         * @param {?} fn
         * @return {?}
         */
        _executeOnStable(fn) {
            if (this._ngZone.isStable) {
                fn();
            }
            else {
                this._ngZone.onStable.asObservable().pipe(take(1)).subscribe(fn);
            }
        }
    }
    /**
     * Factory that allows easy instantiation of focus traps.
     * @deprecated Use `ConfigurableFocusTrapFactory` instead.
     * \@breaking-change for 11.0.0 Remove this class.
     */
    class FocusTrapFactory {
        /**
         * @param {?} _checker
         * @param {?} _ngZone
         * @param {?} _document
         */
        constructor(_checker, _ngZone, _document) {
            this._checker = _checker;
            this._ngZone = _ngZone;
            this._document = _document;
        }
        /**
         * Creates a focus-trapped region around the given element.
         * @param {?} element The element around which focus will be trapped.
         * @param {?=} deferCaptureElements Defers the creation of focus-capturing elements to be done
         *     manually by the user.
         * @return {?} The created focus trap instance.
         */
        create(element, deferCaptureElements = false) {
            return new FocusTrap(element, this._checker, this._ngZone, this._document, deferCaptureElements);
        }
    }
    FocusTrapFactory.ɵfac = function FocusTrapFactory_Factory(t) { return new (t || FocusTrapFactory)(ɵɵinject(InteractivityChecker), ɵɵinject(NgZone), ɵɵinject(DOCUMENT$1)); };
    /** @nocollapse */
    FocusTrapFactory.ctorParameters = () => [
        { type: InteractivityChecker },
        { type: NgZone },
        { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT$1,] }] }
    ];
    /** @nocollapse */ FocusTrapFactory.ɵprov = ɵɵdefineInjectable({ factory: function FocusTrapFactory_Factory() { return new FocusTrapFactory(ɵɵinject(InteractivityChecker), ɵɵinject(NgZone), ɵɵinject(DOCUMENT$1)); }, token: FocusTrapFactory, providedIn: "root" });
    /**
     * Directive for trapping focus within a region.
     */
    class CdkTrapFocus {
        /**
         * @param {?} _elementRef
         * @param {?} _focusTrapFactory
         * @param {?} _document
         */
        constructor(_elementRef, _focusTrapFactory, _document) {
            this._elementRef = _elementRef;
            this._focusTrapFactory = _focusTrapFactory;
            /**
             * Previously focused element to restore focus to upon destroy when using autoCapture.
             */
            this._previouslyFocusedElement = null;
            this._document = _document;
            this.focusTrap = this._focusTrapFactory.create(this._elementRef.nativeElement, true);
        }
        /**
         * Whether the focus trap is active.
         * @return {?}
         */
        get enabled() { return this.focusTrap.enabled; }
        /**
         * @param {?} value
         * @return {?}
         */
        set enabled(value) { this.focusTrap.enabled = coerceBooleanProperty(value); }
        /**
         * Whether the directive should automatially move focus into the trapped region upon
         * initialization and return focus to the previous activeElement upon destruction.
         * @return {?}
         */
        get autoCapture() { return this._autoCapture; }
        /**
         * @param {?} value
         * @return {?}
         */
        set autoCapture(value) { this._autoCapture = coerceBooleanProperty(value); }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this.focusTrap.destroy();
            // If we stored a previously focused element when using autoCapture, return focus to that
            // element now that the trapped region is being destroyed.
            if (this._previouslyFocusedElement) {
                this._previouslyFocusedElement.focus();
                this._previouslyFocusedElement = null;
            }
        }
        /**
         * @return {?}
         */
        ngAfterContentInit() {
            this.focusTrap.attachAnchors();
            if (this.autoCapture) {
                this._previouslyFocusedElement = (/** @type {?} */ (this._document.activeElement));
                this.focusTrap.focusInitialElementWhenReady();
            }
        }
        /**
         * @return {?}
         */
        ngDoCheck() {
            if (!this.focusTrap.hasAttached()) {
                this.focusTrap.attachAnchors();
            }
        }
    }
    CdkTrapFocus.ɵfac = function CdkTrapFocus_Factory(t) { return new (t || CdkTrapFocus)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(FocusTrapFactory), ɵɵdirectiveInject(DOCUMENT$1)); };
    CdkTrapFocus.ɵdir = ɵɵdefineDirective({ type: CdkTrapFocus, selectors: [["", "cdkTrapFocus", ""]], inputs: { enabled: ["cdkTrapFocus", "enabled"], autoCapture: ["cdkTrapFocusAutoCapture", "autoCapture"] }, exportAs: ["cdkTrapFocus"] });
    /** @nocollapse */
    CdkTrapFocus.ctorParameters = () => [
        { type: ElementRef },
        { type: FocusTrapFactory },
        { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT$1,] }] }
    ];
    CdkTrapFocus.propDecorators = {
        enabled: [{ type: Input, args: ['cdkTrapFocus',] }],
        autoCapture: [{ type: Input, args: ['cdkTrapFocusAutoCapture',] }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/a11y/focus-trap/configurable-focus-trap.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Class that allows for trapping focus within a DOM element.
     *
     * This class uses a strategy pattern that determines how it traps focus.
     * See FocusTrapInertStrategy.
     */
    class ConfigurableFocusTrap extends FocusTrap {
        /**
         * @param {?} _element
         * @param {?} _checker
         * @param {?} _ngZone
         * @param {?} _document
         * @param {?} _focusTrapManager
         * @param {?} _inertStrategy
         * @param {?} config
         */
        constructor(_element, _checker, _ngZone, _document, _focusTrapManager, _inertStrategy, config) {
            super(_element, _checker, _ngZone, _document, config.defer);
            this._focusTrapManager = _focusTrapManager;
            this._inertStrategy = _inertStrategy;
            this._focusTrapManager.register(this);
        }
        /**
         * Whether the FocusTrap is enabled.
         * @return {?}
         */
        get enabled() { return this._enabled; }
        /**
         * @param {?} value
         * @return {?}
         */
        set enabled(value) {
            this._enabled = value;
            if (this._enabled) {
                this._focusTrapManager.register(this);
            }
            else {
                this._focusTrapManager.deregister(this);
            }
        }
        /**
         * Notifies the FocusTrapManager that this FocusTrap will be destroyed.
         * @return {?}
         */
        destroy() {
            this._focusTrapManager.deregister(this);
            super.destroy();
        }
        /**
         * \@docs-private Implemented as part of ManagedFocusTrap.
         * @return {?}
         */
        _enable() {
            this._inertStrategy.preventFocus(this);
            this.toggleAnchors(true);
        }
        /**
         * \@docs-private Implemented as part of ManagedFocusTrap.
         * @return {?}
         */
        _disable() {
            this._inertStrategy.allowFocus(this);
            this.toggleAnchors(false);
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/a11y/focus-trap/polyfill.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google LLC All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * IE 11 compatible closest implementation that is able to start from non-Element Nodes.
     * @param {?} element
     * @param {?} selector
     * @return {?}
     */
    function closest(element, selector) {
        if (!(element instanceof Node)) {
            return null;
        }
        /** @type {?} */
        let curr = element;
        while (curr != null && !(curr instanceof Element)) {
            curr = curr.parentNode;
        }
        return curr && (/** @type {?} */ ((hasNativeClosest ?
            curr.closest(selector) : polyfillClosest(curr, selector))));
    }
    /**
     * Polyfill for browsers without Element.closest.
     * @param {?} element
     * @param {?} selector
     * @return {?}
     */
    function polyfillClosest(element, selector) {
        /** @type {?} */
        let curr = element;
        while (curr != null && !(curr instanceof Element && matches(curr, selector))) {
            curr = curr.parentNode;
        }
        return (/** @type {?} */ ((curr || null)));
    }
    /** @type {?} */
    const hasNativeClosest = typeof Element != 'undefined' && !!Element.prototype.closest;
    /**
     * IE 11 compatible matches implementation.
     * @param {?} element
     * @param {?} selector
     * @return {?}
     */
    function matches(element, selector) {
        return element.matches ?
            element.matches(selector) :
            ((/** @type {?} */ (element)))['msMatchesSelector'](selector);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/a11y/focus-trap/event-listener-inert-strategy.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Lightweight FocusTrapInertStrategy that adds a document focus event
     * listener to redirect focus back inside the FocusTrap.
     */
    class EventListenerFocusTrapInertStrategy {
        constructor() {
            /**
             * Focus event handler.
             */
            this._listener = null;
        }
        /**
         * Adds a document event listener that keeps focus inside the FocusTrap.
         * @param {?} focusTrap
         * @return {?}
         */
        preventFocus(focusTrap) {
            // Ensure there's only one listener per document
            if (this._listener) {
                focusTrap._document.removeEventListener('focus', (/** @type {?} */ (this._listener)), true);
            }
            this._listener = (/**
             * @param {?} e
             * @return {?}
             */
            (e) => this._trapFocus(focusTrap, e));
            focusTrap._ngZone.runOutsideAngular((/**
             * @return {?}
             */
            () => {
                focusTrap._document.addEventListener('focus', (/** @type {?} */ (this._listener)), true);
            }));
        }
        /**
         * Removes the event listener added in preventFocus.
         * @param {?} focusTrap
         * @return {?}
         */
        allowFocus(focusTrap) {
            if (!this._listener) {
                return;
            }
            focusTrap._document.removeEventListener('focus', (/** @type {?} */ (this._listener)), true);
            this._listener = null;
        }
        /**
         * Refocuses the first element in the FocusTrap if the focus event target was outside
         * the FocusTrap.
         *
         * This is an event listener callback. The event listener is added in runOutsideAngular,
         * so all this code runs outside Angular as well.
         * @private
         * @param {?} focusTrap
         * @param {?} event
         * @return {?}
         */
        _trapFocus(focusTrap, event) {
            /** @type {?} */
            const target = (/** @type {?} */ (event.target));
            /** @type {?} */
            const focusTrapRoot = focusTrap._element;
            // Don't refocus if target was in an overlay, because the overlay might be associated
            // with an element inside the FocusTrap, ex. mat-select.
            if (!focusTrapRoot.contains(target) && closest(target, 'div.cdk-overlay-pane') === null) {
                // Some legacy FocusTrap usages have logic that focuses some element on the page
                // just before FocusTrap is destroyed. For backwards compatibility, wait
                // to be sure FocusTrap is still enabled before refocusing.
                setTimeout((/**
                 * @return {?}
                 */
                () => {
                    // Check whether focus wasn't put back into the focus trap while the timeout was pending.
                    if (focusTrap.enabled && !focusTrapRoot.contains(focusTrap._document.activeElement)) {
                        focusTrap.focusFirstTabbableElement();
                    }
                }));
            }
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/a11y/focus-trap/configurable-focus-trap-config.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google LLC All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * Configuration for creating a ConfigurableFocusTrap.
     */
    class ConfigurableFocusTrapConfig {
        constructor() {
            /**
             * Whether to defer the creation of FocusTrap elements to be
             * done manually by the user. Default is to create them
             * automatically.
             */
            this.defer = false;
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/a11y/focus-trap/focus-trap-inert-strategy.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * The injection token used to specify the inert strategy.
     * @type {?}
     */
    const FOCUS_TRAP_INERT_STRATEGY = new InjectionToken('FOCUS_TRAP_INERT_STRATEGY');
    /**
     * Injectable that ensures only the most recently enabled FocusTrap is active.
     */
    class FocusTrapManager {
        constructor() {
            // A stack of the FocusTraps on the page. Only the FocusTrap at the
            // top of the stack is active.
            this._focusTrapStack = [];
        }
        /**
         * Disables the FocusTrap at the top of the stack, and then pushes
         * the new FocusTrap onto the stack.
         * @param {?} focusTrap
         * @return {?}
         */
        register(focusTrap) {
            // Dedupe focusTraps that register multiple times.
            this._focusTrapStack = this._focusTrapStack.filter((/**
             * @param {?} ft
             * @return {?}
             */
            (ft) => ft !== focusTrap));
            /** @type {?} */
            let stack = this._focusTrapStack;
            if (stack.length) {
                stack[stack.length - 1]._disable();
            }
            stack.push(focusTrap);
            focusTrap._enable();
        }
        /**
         * Removes the FocusTrap from the stack, and activates the
         * FocusTrap that is the new top of the stack.
         * @param {?} focusTrap
         * @return {?}
         */
        deregister(focusTrap) {
            focusTrap._disable();
            /** @type {?} */
            const stack = this._focusTrapStack;
            /** @type {?} */
            const i = stack.indexOf(focusTrap);
            if (i !== -1) {
                stack.splice(i, 1);
                if (stack.length) {
                    stack[stack.length - 1]._enable();
                }
            }
        }
    }
    FocusTrapManager.ɵfac = function FocusTrapManager_Factory(t) { return new (t || FocusTrapManager)(); };
    /** @nocollapse */ FocusTrapManager.ɵprov = ɵɵdefineInjectable({ factory: function FocusTrapManager_Factory() { return new FocusTrapManager(); }, token: FocusTrapManager, providedIn: "root" });

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/a11y/focus-trap/configurable-focus-trap-factory.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Factory that allows easy instantiation of configurable focus traps.
     */
    class ConfigurableFocusTrapFactory {
        /**
         * @param {?} _checker
         * @param {?} _ngZone
         * @param {?} _focusTrapManager
         * @param {?} _document
         * @param {?=} _inertStrategy
         */
        constructor(_checker, _ngZone, _focusTrapManager, _document, _inertStrategy) {
            this._checker = _checker;
            this._ngZone = _ngZone;
            this._focusTrapManager = _focusTrapManager;
            this._document = _document;
            // TODO split up the strategies into different modules, similar to DateAdapter.
            this._inertStrategy = _inertStrategy || new EventListenerFocusTrapInertStrategy();
        }
        /**
         * @param {?} element
         * @param {?=} config
         * @return {?}
         */
        create(element, config = new ConfigurableFocusTrapConfig()) {
            /** @type {?} */
            let configObject;
            if (typeof config === 'boolean') {
                configObject = new ConfigurableFocusTrapConfig();
                configObject.defer = config;
            }
            else {
                configObject = config;
            }
            return new ConfigurableFocusTrap(element, this._checker, this._ngZone, this._document, this._focusTrapManager, this._inertStrategy, configObject);
        }
    }
    ConfigurableFocusTrapFactory.ɵfac = function ConfigurableFocusTrapFactory_Factory(t) { return new (t || ConfigurableFocusTrapFactory)(ɵɵinject(InteractivityChecker), ɵɵinject(NgZone), ɵɵinject(FocusTrapManager), ɵɵinject(DOCUMENT$1), ɵɵinject(FOCUS_TRAP_INERT_STRATEGY, 8)); };
    /** @nocollapse */
    ConfigurableFocusTrapFactory.ctorParameters = () => [
        { type: InteractivityChecker },
        { type: NgZone },
        { type: FocusTrapManager },
        { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT$1,] }] },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [FOCUS_TRAP_INERT_STRATEGY,] }] }
    ];
    /** @nocollapse */ ConfigurableFocusTrapFactory.ɵprov = ɵɵdefineInjectable({ factory: function ConfigurableFocusTrapFactory_Factory() { return new ConfigurableFocusTrapFactory(ɵɵinject(InteractivityChecker), ɵɵinject(NgZone), ɵɵinject(FocusTrapManager), ɵɵinject(DOCUMENT$1), ɵɵinject(FOCUS_TRAP_INERT_STRATEGY, 8)); }, token: ConfigurableFocusTrapFactory, providedIn: "root" });

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/a11y/live-announcer/live-announcer-tokens.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const LIVE_ANNOUNCER_ELEMENT_TOKEN = new InjectionToken('liveAnnouncerElement', {
        providedIn: 'root',
        factory: LIVE_ANNOUNCER_ELEMENT_TOKEN_FACTORY,
    });
    /**
     * \@docs-private
     * @return {?}
     */
    function LIVE_ANNOUNCER_ELEMENT_TOKEN_FACTORY() {
        return null;
    }
    /**
     * Injection token that can be used to configure the default options for the LiveAnnouncer.
     * @type {?}
     */
    const LIVE_ANNOUNCER_DEFAULT_OPTIONS = new InjectionToken('LIVE_ANNOUNCER_DEFAULT_OPTIONS');

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/a11y/live-announcer/live-announcer.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class LiveAnnouncer {
        /**
         * @param {?} elementToken
         * @param {?} _ngZone
         * @param {?} _document
         * @param {?=} _defaultOptions
         */
        constructor(elementToken, _ngZone, _document, _defaultOptions) {
            this._ngZone = _ngZone;
            this._defaultOptions = _defaultOptions;
            // We inject the live element and document as `any` because the constructor signature cannot
            // reference browser globals (HTMLElement, Document) on non-browser environments, since having
            // a class decorator causes TypeScript to preserve the constructor signature types.
            this._document = _document;
            this._liveElement = elementToken || this._createLiveElement();
        }
        /**
         * @param {?} message
         * @param {...?} args
         * @return {?}
         */
        announce(message, ...args) {
            /** @type {?} */
            const defaultOptions = this._defaultOptions;
            /** @type {?} */
            let politeness;
            /** @type {?} */
            let duration;
            if (args.length === 1 && typeof args[0] === 'number') {
                duration = args[0];
            }
            else {
                [politeness, duration] = args;
            }
            this.clear();
            clearTimeout(this._previousTimeout);
            if (!politeness) {
                politeness =
                    (defaultOptions && defaultOptions.politeness) ? defaultOptions.politeness : 'polite';
            }
            if (duration == null && defaultOptions) {
                duration = defaultOptions.duration;
            }
            // TODO: ensure changing the politeness works on all environments we support.
            this._liveElement.setAttribute('aria-live', politeness);
            // This 100ms timeout is necessary for some browser + screen-reader combinations:
            // - Both JAWS and NVDA over IE11 will not announce anything without a non-zero timeout.
            // - With Chrome and IE11 with NVDA or JAWS, a repeated (identical) message won't be read a
            //   second time without clearing and then using a non-zero delay.
            // (using JAWS 17 at time of this writing).
            return this._ngZone.runOutsideAngular((/**
             * @return {?}
             */
            () => {
                return new Promise((/**
                 * @param {?} resolve
                 * @return {?}
                 */
                resolve => {
                    clearTimeout(this._previousTimeout);
                    this._previousTimeout = setTimeout((/**
                     * @return {?}
                     */
                    () => {
                        this._liveElement.textContent = message;
                        resolve();
                        if (typeof duration === 'number') {
                            this._previousTimeout = setTimeout((/**
                             * @return {?}
                             */
                            () => this.clear()), duration);
                        }
                    }), 100);
                }));
            }));
        }
        /**
         * Clears the current text from the announcer element. Can be used to prevent
         * screen readers from reading the text out again while the user is going
         * through the page landmarks.
         * @return {?}
         */
        clear() {
            if (this._liveElement) {
                this._liveElement.textContent = '';
            }
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            clearTimeout(this._previousTimeout);
            if (this._liveElement && this._liveElement.parentNode) {
                this._liveElement.parentNode.removeChild(this._liveElement);
                this._liveElement = (/** @type {?} */ (null));
            }
        }
        /**
         * @private
         * @return {?}
         */
        _createLiveElement() {
            /** @type {?} */
            const elementClass = 'cdk-live-announcer-element';
            /** @type {?} */
            const previousElements = this._document.getElementsByClassName(elementClass);
            /** @type {?} */
            const liveEl = this._document.createElement('div');
            // Remove any old containers. This can happen when coming in from a server-side-rendered page.
            for (let i = 0; i < previousElements.length; i++) {
                (/** @type {?} */ (previousElements[i].parentNode)).removeChild(previousElements[i]);
            }
            liveEl.classList.add(elementClass);
            liveEl.classList.add('cdk-visually-hidden');
            liveEl.setAttribute('aria-atomic', 'true');
            liveEl.setAttribute('aria-live', 'polite');
            this._document.body.appendChild(liveEl);
            return liveEl;
        }
    }
    LiveAnnouncer.ɵfac = function LiveAnnouncer_Factory(t) { return new (t || LiveAnnouncer)(ɵɵinject(LIVE_ANNOUNCER_ELEMENT_TOKEN, 8), ɵɵinject(NgZone), ɵɵinject(DOCUMENT$1), ɵɵinject(LIVE_ANNOUNCER_DEFAULT_OPTIONS, 8)); };
    /** @nocollapse */
    LiveAnnouncer.ctorParameters = () => [
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [LIVE_ANNOUNCER_ELEMENT_TOKEN,] }] },
        { type: NgZone },
        { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT$1,] }] },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [LIVE_ANNOUNCER_DEFAULT_OPTIONS,] }] }
    ];
    /** @nocollapse */ LiveAnnouncer.ɵprov = ɵɵdefineInjectable({ factory: function LiveAnnouncer_Factory() { return new LiveAnnouncer(ɵɵinject(LIVE_ANNOUNCER_ELEMENT_TOKEN, 8), ɵɵinject(NgZone), ɵɵinject(DOCUMENT$1), ɵɵinject(LIVE_ANNOUNCER_DEFAULT_OPTIONS, 8)); }, token: LiveAnnouncer, providedIn: "root" });
    /**
     * A directive that works similarly to aria-live, but uses the LiveAnnouncer to ensure compatibility
     * with a wider range of browsers and screen readers.
     */
    class CdkAriaLive {
        /**
         * @param {?} _elementRef
         * @param {?} _liveAnnouncer
         * @param {?} _contentObserver
         * @param {?} _ngZone
         */
        constructor(_elementRef, _liveAnnouncer, _contentObserver, _ngZone) {
            this._elementRef = _elementRef;
            this._liveAnnouncer = _liveAnnouncer;
            this._contentObserver = _contentObserver;
            this._ngZone = _ngZone;
            this._politeness = 'off';
        }
        /**
         * The aria-live politeness level to use when announcing messages.
         * @return {?}
         */
        get politeness() { return this._politeness; }
        /**
         * @param {?} value
         * @return {?}
         */
        set politeness(value) {
            this._politeness = value === 'polite' || value === 'assertive' ? value : 'off';
            if (this._politeness === 'off') {
                if (this._subscription) {
                    this._subscription.unsubscribe();
                    this._subscription = null;
                }
            }
            else if (!this._subscription) {
                this._subscription = this._ngZone.runOutsideAngular((/**
                 * @return {?}
                 */
                () => {
                    return this._contentObserver
                        .observe(this._elementRef)
                        .subscribe((/**
                     * @return {?}
                     */
                    () => {
                        // Note that we use textContent here, rather than innerText, in order to avoid a reflow.
                        /** @type {?} */
                        const elementText = this._elementRef.nativeElement.textContent;
                        // The `MutationObserver` fires also for attribute
                        // changes which we don't want to announce.
                        if (elementText !== this._previousAnnouncedText) {
                            this._liveAnnouncer.announce(elementText, this._politeness);
                            this._previousAnnouncedText = elementText;
                        }
                    }));
                }));
            }
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            if (this._subscription) {
                this._subscription.unsubscribe();
            }
        }
    }
    CdkAriaLive.ɵfac = function CdkAriaLive_Factory(t) { return new (t || CdkAriaLive)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(LiveAnnouncer), ɵɵdirectiveInject(ContentObserver), ɵɵdirectiveInject(NgZone)); };
    CdkAriaLive.ɵdir = ɵɵdefineDirective({ type: CdkAriaLive, selectors: [["", "cdkAriaLive", ""]], inputs: { politeness: ["cdkAriaLive", "politeness"] }, exportAs: ["cdkAriaLive"] });
    /** @nocollapse */
    CdkAriaLive.ctorParameters = () => [
        { type: ElementRef },
        { type: LiveAnnouncer },
        { type: ContentObserver },
        { type: NgZone }
    ];
    CdkAriaLive.propDecorators = {
        politeness: [{ type: Input, args: ['cdkAriaLive',] }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/a11y/fake-mousedown.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google LLC All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * Screenreaders will often fire fake mousedown events when a focusable element
     * is activated using the keyboard. We can typically distinguish between these faked
     * mousedown events and real mousedown events using the "buttons" property. While
     * real mousedowns will indicate the mouse button that was pressed (e.g. "1" for
     * the left mouse button), faked mousedowns will usually set the property value to 0.
     * @param {?} event
     * @return {?}
     */
    function isFakeMousedownFromScreenReader(event) {
        return event.buttons === 0;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/a11y/focus-monitor/focus-monitor.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    // This is the value used by AngularJS Material. Through trial and error (on iPhone 6S) they found
    // that a value of around 650ms seems appropriate.
    /** @type {?} */
    const TOUCH_BUFFER_MS = 650;
    /**
     * InjectionToken for FocusMonitorOptions.
     * @type {?}
     */
    const FOCUS_MONITOR_DEFAULT_OPTIONS = new InjectionToken('cdk-focus-monitor-default-options');
    /**
     * Event listener options that enable capturing and also
     * mark the listener as passive if the browser supports it.
     * @type {?}
     */
    const captureEventListenerOptions = normalizePassiveListenerOptions({
        passive: true,
        capture: true
    });
    /**
     * Monitors mouse and keyboard events to determine the cause of focus events.
     */
    class FocusMonitor {
        /**
         * @param {?} _ngZone
         * @param {?} _platform
         * @param {?} document
         * @param {?} options
         */
        constructor(_ngZone, _platform, 
        /** @breaking-change 11.0.0 make document required */
        document, options) {
            this._ngZone = _ngZone;
            this._platform = _platform;
            /**
             * The focus origin that the next focus event is a result of.
             */
            this._origin = null;
            /**
             * Whether the window has just been focused.
             */
            this._windowFocused = false;
            /**
             * Map of elements being monitored to their info.
             */
            this._elementInfo = new Map();
            /**
             * The number of elements currently being monitored.
             */
            this._monitoredElementCount = 0;
            /**
             * Keeps track of the root nodes to which we've currently bound a focus/blur handler,
             * as well as the number of monitored elements that they contain. We have to treat focus/blur
             * handlers differently from the rest of the events, because the browser won't emit events
             * to the document when focus moves inside of a shadow root.
             */
            this._rootNodeFocusListenerCount = new Map();
            /**
             * Event listener for `keydown` events on the document.
             * Needs to be an arrow function in order to preserve the context when it gets bound.
             */
            this._documentKeydownListener = (/**
             * @return {?}
             */
            () => {
                // On keydown record the origin and clear any touch event that may be in progress.
                this._lastTouchTarget = null;
                this._setOriginForCurrentEventQueue('keyboard');
            });
            /**
             * Event listener for `mousedown` events on the document.
             * Needs to be an arrow function in order to preserve the context when it gets bound.
             */
            this._documentMousedownListener = (/**
             * @param {?} event
             * @return {?}
             */
            (event) => {
                // On mousedown record the origin only if there is not touch
                // target, since a mousedown can happen as a result of a touch event.
                if (!this._lastTouchTarget) {
                    // In some cases screen readers fire fake `mousedown` events instead of `keydown`.
                    // Resolve the focus source to `keyboard` if we detect one of them.
                    /** @type {?} */
                    const source = isFakeMousedownFromScreenReader(event) ? 'keyboard' : 'mouse';
                    this._setOriginForCurrentEventQueue(source);
                }
            });
            /**
             * Event listener for `touchstart` events on the document.
             * Needs to be an arrow function in order to preserve the context when it gets bound.
             */
            this._documentTouchstartListener = (/**
             * @param {?} event
             * @return {?}
             */
            (event) => {
                // When the touchstart event fires the focus event is not yet in the event queue. This means
                // we can't rely on the trick used above (setting timeout of 1ms). Instead we wait 650ms to
                // see if a focus happens.
                if (this._touchTimeoutId != null) {
                    clearTimeout(this._touchTimeoutId);
                }
                this._lastTouchTarget = getTarget(event);
                this._touchTimeoutId = setTimeout((/**
                 * @return {?}
                 */
                () => this._lastTouchTarget = null), TOUCH_BUFFER_MS);
            });
            /**
             * Event listener for `focus` events on the window.
             * Needs to be an arrow function in order to preserve the context when it gets bound.
             */
            this._windowFocusListener = (/**
             * @return {?}
             */
            () => {
                // Make a note of when the window regains focus, so we can
                // restore the origin info for the focused element.
                this._windowFocused = true;
                this._windowFocusTimeoutId = setTimeout((/**
                 * @return {?}
                 */
                () => this._windowFocused = false));
            });
            /**
             * Event listener for `focus` and 'blur' events on the document.
             * Needs to be an arrow function in order to preserve the context when it gets bound.
             */
            this._rootNodeFocusAndBlurListener = (/**
             * @param {?} event
             * @return {?}
             */
            (event) => {
                /** @type {?} */
                const target = getTarget(event);
                /** @type {?} */
                const handler = event.type === 'focus' ? this._onFocus : this._onBlur;
                // We need to walk up the ancestor chain in order to support `checkChildren`.
                for (let element = target; element; element = element.parentElement) {
                    handler.call(this, (/** @type {?} */ (event)), element);
                }
            });
            this._document = document;
            this._detectionMode = (options === null || options === void 0 ? void 0 : options.detectionMode) || 0 /* IMMEDIATE */;
        }
        /**
         * @param {?} element
         * @param {?=} checkChildren
         * @return {?}
         */
        monitor(element, checkChildren = false) {
            // Do nothing if we're not on the browser platform.
            if (!this._platform.isBrowser) {
                return of(null);
            }
            /** @type {?} */
            const nativeElement = coerceElement(element);
            // If the element is inside the shadow DOM, we need to bind our focus/blur listeners to
            // the shadow root, rather than the `document`, because the browser won't emit focus events
            // to the `document`, if focus is moving within the same shadow root.
            /** @type {?} */
            const rootNode = ((/** @type {?} */ (_getShadowRoot(nativeElement)))) || this._getDocument();
            /** @type {?} */
            const cachedInfo = this._elementInfo.get(nativeElement);
            // Check if we're already monitoring this element.
            if (cachedInfo) {
                if (checkChildren) {
                    // TODO(COMP-318): this can be problematic, because it'll turn all non-checkChildren
                    // observers into ones that behave as if `checkChildren` was turned on. We need a more
                    // robust solution.
                    cachedInfo.checkChildren = true;
                }
                return cachedInfo.subject.asObservable();
            }
            // Create monitored element info.
            /** @type {?} */
            const info = {
                checkChildren: checkChildren,
                subject: new Subject(),
                rootNode
            };
            this._elementInfo.set(nativeElement, info);
            this._registerGlobalListeners(info);
            return info.subject.asObservable();
        }
        /**
         * @param {?} element
         * @return {?}
         */
        stopMonitoring(element) {
            /** @type {?} */
            const nativeElement = coerceElement(element);
            /** @type {?} */
            const elementInfo = this._elementInfo.get(nativeElement);
            if (elementInfo) {
                elementInfo.subject.complete();
                this._setClasses(nativeElement);
                this._elementInfo.delete(nativeElement);
                this._removeGlobalListeners(elementInfo);
            }
        }
        /**
         * @param {?} element
         * @param {?} origin
         * @param {?=} options
         * @return {?}
         */
        focusVia(element, origin, options) {
            /** @type {?} */
            const nativeElement = coerceElement(element);
            this._setOriginForCurrentEventQueue(origin);
            // `focus` isn't available on the server
            if (typeof nativeElement.focus === 'function') {
                // Cast the element to `any`, because the TS typings don't have the `options` parameter yet.
                ((/** @type {?} */ (nativeElement))).focus(options);
            }
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this._elementInfo.forEach((/**
             * @param {?} _info
             * @param {?} element
             * @return {?}
             */
            (_info, element) => this.stopMonitoring(element)));
        }
        /**
         * Access injected document if available or fallback to global document reference
         * @private
         * @return {?}
         */
        _getDocument() {
            return this._document || document;
        }
        /**
         * Use defaultView of injected document if available or fallback to global window reference
         * @private
         * @return {?}
         */
        _getWindow() {
            /** @type {?} */
            const doc = this._getDocument();
            return doc.defaultView || window;
        }
        /**
         * @private
         * @param {?} element
         * @param {?} className
         * @param {?} shouldSet
         * @return {?}
         */
        _toggleClass(element, className, shouldSet) {
            if (shouldSet) {
                element.classList.add(className);
            }
            else {
                element.classList.remove(className);
            }
        }
        /**
         * @private
         * @param {?} event
         * @return {?}
         */
        _getFocusOrigin(event) {
            // If we couldn't detect a cause for the focus event, it's due to one of three reasons:
            // 1) The window has just regained focus, in which case we want to restore the focused state of
            //    the element from before the window blurred.
            // 2) It was caused by a touch event, in which case we mark the origin as 'touch'.
            // 3) The element was programmatically focused, in which case we should mark the origin as
            //    'program'.
            if (this._origin) {
                return this._origin;
            }
            if (this._windowFocused && this._lastFocusOrigin) {
                return this._lastFocusOrigin;
            }
            else if (this._wasCausedByTouch(event)) {
                return 'touch';
            }
            else {
                return 'program';
            }
        }
        /**
         * Sets the focus classes on the element based on the given focus origin.
         * @private
         * @param {?} element The element to update the classes on.
         * @param {?=} origin The focus origin.
         * @return {?}
         */
        _setClasses(element, origin) {
            this._toggleClass(element, 'cdk-focused', !!origin);
            this._toggleClass(element, 'cdk-touch-focused', origin === 'touch');
            this._toggleClass(element, 'cdk-keyboard-focused', origin === 'keyboard');
            this._toggleClass(element, 'cdk-mouse-focused', origin === 'mouse');
            this._toggleClass(element, 'cdk-program-focused', origin === 'program');
        }
        /**
         * Sets the origin and schedules an async function to clear it at the end of the event queue.
         * If the detection mode is 'eventual', the origin is never cleared.
         * @private
         * @param {?} origin The origin to set.
         * @return {?}
         */
        _setOriginForCurrentEventQueue(origin) {
            this._ngZone.runOutsideAngular((/**
             * @return {?}
             */
            () => {
                this._origin = origin;
                if (this._detectionMode === 0 /* IMMEDIATE */) {
                    // Sometimes the focus origin won't be valid in Firefox because Firefox seems to focus *one*
                    // tick after the interaction event fired. To ensure the focus origin is always correct,
                    // the focus origin will be determined at the beginning of the next tick.
                    this._originTimeoutId = setTimeout((/**
                     * @return {?}
                     */
                    () => this._origin = null), 1);
                }
            }));
        }
        /**
         * Checks whether the given focus event was caused by a touchstart event.
         * @private
         * @param {?} event The focus event to check.
         * @return {?} Whether the event was caused by a touch.
         */
        _wasCausedByTouch(event) {
            // Note(mmalerba): This implementation is not quite perfect, there is a small edge case.
            // Consider the following dom structure:
            //
            // <div #parent tabindex="0" cdkFocusClasses>
            //   <div #child (click)="#parent.focus()"></div>
            // </div>
            //
            // If the user touches the #child element and the #parent is programmatically focused as a
            // result, this code will still consider it to have been caused by the touch event and will
            // apply the cdk-touch-focused class rather than the cdk-program-focused class. This is a
            // relatively small edge-case that can be worked around by using
            // focusVia(parentEl, 'program') to focus the parent element.
            //
            // If we decide that we absolutely must handle this case correctly, we can do so by listening
            // for the first focus event after the touchstart, and then the first blur event after that
            // focus event. When that blur event fires we know that whatever follows is not a result of the
            // touchstart.
            /** @type {?} */
            const focusTarget = getTarget(event);
            return this._lastTouchTarget instanceof Node && focusTarget instanceof Node &&
                (focusTarget === this._lastTouchTarget || focusTarget.contains(this._lastTouchTarget));
        }
        /**
         * Handles focus events on a registered element.
         * @private
         * @param {?} event The focus event.
         * @param {?} element The monitored element.
         * @return {?}
         */
        _onFocus(event, element) {
            // NOTE(mmalerba): We currently set the classes based on the focus origin of the most recent
            // focus event affecting the monitored element. If we want to use the origin of the first event
            // instead we should check for the cdk-focused class here and return if the element already has
            // it. (This only matters for elements that have includesChildren = true).
            // NOTE(mmalerba): We currently set the classes based on the focus origin of the most recent
            // focus event affecting the monitored element. If we want to use the origin of the first event
            // instead we should check for the cdk-focused class here and return if the element already has
            // it. (This only matters for elements that have includesChildren = true).
            // If we are not counting child-element-focus as focused, make sure that the event target is the
            // monitored element itself.
            /** @type {?} */
            const elementInfo = this._elementInfo.get(element);
            if (!elementInfo || (!elementInfo.checkChildren && element !== getTarget(event))) {
                return;
            }
            /** @type {?} */
            const origin = this._getFocusOrigin(event);
            this._setClasses(element, origin);
            this._emitOrigin(elementInfo.subject, origin);
            this._lastFocusOrigin = origin;
        }
        /**
         * Handles blur events on a registered element.
         * @param {?} event The blur event.
         * @param {?} element The monitored element.
         * @return {?}
         */
        _onBlur(event, element) {
            // If we are counting child-element-focus as focused, make sure that we aren't just blurring in
            // order to focus another child of the monitored element.
            /** @type {?} */
            const elementInfo = this._elementInfo.get(element);
            if (!elementInfo || (elementInfo.checkChildren && event.relatedTarget instanceof Node &&
                element.contains(event.relatedTarget))) {
                return;
            }
            this._setClasses(element);
            this._emitOrigin(elementInfo.subject, null);
        }
        /**
         * @private
         * @param {?} subject
         * @param {?} origin
         * @return {?}
         */
        _emitOrigin(subject, origin) {
            this._ngZone.run((/**
             * @return {?}
             */
            () => subject.next(origin)));
        }
        /**
         * @private
         * @param {?} elementInfo
         * @return {?}
         */
        _registerGlobalListeners(elementInfo) {
            if (!this._platform.isBrowser) {
                return;
            }
            /** @type {?} */
            const rootNode = elementInfo.rootNode;
            /** @type {?} */
            const rootNodeFocusListeners = this._rootNodeFocusListenerCount.get(rootNode) || 0;
            if (!rootNodeFocusListeners) {
                this._ngZone.runOutsideAngular((/**
                 * @return {?}
                 */
                () => {
                    rootNode.addEventListener('focus', this._rootNodeFocusAndBlurListener, captureEventListenerOptions);
                    rootNode.addEventListener('blur', this._rootNodeFocusAndBlurListener, captureEventListenerOptions);
                }));
            }
            this._rootNodeFocusListenerCount.set(rootNode, rootNodeFocusListeners + 1);
            // Register global listeners when first element is monitored.
            if (++this._monitoredElementCount === 1) {
                // Note: we listen to events in the capture phase so we
                // can detect them even if the user stops propagation.
                this._ngZone.runOutsideAngular((/**
                 * @return {?}
                 */
                () => {
                    /** @type {?} */
                    const document = this._getDocument();
                    /** @type {?} */
                    const window = this._getWindow();
                    document.addEventListener('keydown', this._documentKeydownListener, captureEventListenerOptions);
                    document.addEventListener('mousedown', this._documentMousedownListener, captureEventListenerOptions);
                    document.addEventListener('touchstart', this._documentTouchstartListener, captureEventListenerOptions);
                    window.addEventListener('focus', this._windowFocusListener);
                }));
            }
        }
        /**
         * @private
         * @param {?} elementInfo
         * @return {?}
         */
        _removeGlobalListeners(elementInfo) {
            /** @type {?} */
            const rootNode = elementInfo.rootNode;
            if (this._rootNodeFocusListenerCount.has(rootNode)) {
                /** @type {?} */
                const rootNodeFocusListeners = (/** @type {?} */ (this._rootNodeFocusListenerCount.get(rootNode)));
                if (rootNodeFocusListeners > 1) {
                    this._rootNodeFocusListenerCount.set(rootNode, rootNodeFocusListeners - 1);
                }
                else {
                    rootNode.removeEventListener('focus', this._rootNodeFocusAndBlurListener, captureEventListenerOptions);
                    rootNode.removeEventListener('blur', this._rootNodeFocusAndBlurListener, captureEventListenerOptions);
                    this._rootNodeFocusListenerCount.delete(rootNode);
                }
            }
            // Unregister global listeners when last element is unmonitored.
            if (!--this._monitoredElementCount) {
                /** @type {?} */
                const document = this._getDocument();
                /** @type {?} */
                const window = this._getWindow();
                document.removeEventListener('keydown', this._documentKeydownListener, captureEventListenerOptions);
                document.removeEventListener('mousedown', this._documentMousedownListener, captureEventListenerOptions);
                document.removeEventListener('touchstart', this._documentTouchstartListener, captureEventListenerOptions);
                window.removeEventListener('focus', this._windowFocusListener);
                // Clear timeouts for all potentially pending timeouts to prevent the leaks.
                clearTimeout(this._windowFocusTimeoutId);
                clearTimeout(this._touchTimeoutId);
                clearTimeout(this._originTimeoutId);
            }
        }
    }
    FocusMonitor.ɵfac = function FocusMonitor_Factory(t) { return new (t || FocusMonitor)(ɵɵinject(NgZone), ɵɵinject(Platform), ɵɵinject(DOCUMENT$1, 8), ɵɵinject(FOCUS_MONITOR_DEFAULT_OPTIONS, 8)); };
    /** @nocollapse */
    FocusMonitor.ctorParameters = () => [
        { type: NgZone },
        { type: Platform },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [DOCUMENT$1,] }] },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [FOCUS_MONITOR_DEFAULT_OPTIONS,] }] }
    ];
    /** @nocollapse */ FocusMonitor.ɵprov = ɵɵdefineInjectable({ factory: function FocusMonitor_Factory() { return new FocusMonitor(ɵɵinject(NgZone), ɵɵinject(Platform), ɵɵinject(DOCUMENT$1, 8), ɵɵinject(FOCUS_MONITOR_DEFAULT_OPTIONS, 8)); }, token: FocusMonitor, providedIn: "root" });
    /**
     * Gets the target of an event, accounting for Shadow DOM.
     * @param {?} event
     * @return {?}
     */
    function getTarget(event) {
        // If an event is bound outside the Shadow DOM, the `event.target` will
        // point to the shadow root so we have to use `composedPath` instead.
        return (/** @type {?} */ ((event.composedPath ? event.composedPath()[0] : event.target)));
    }
    /**
     * Directive that determines how a particular element was focused (via keyboard, mouse, touch, or
     * programmatically) and adds corresponding classes to the element.
     *
     * There are two variants of this directive:
     * 1) cdkMonitorElementFocus: does not consider an element to be focused if one of its children is
     *    focused.
     * 2) cdkMonitorSubtreeFocus: considers an element focused if it or any of its children are focused.
     */
    class CdkMonitorFocus {
        /**
         * @param {?} _elementRef
         * @param {?} _focusMonitor
         */
        constructor(_elementRef, _focusMonitor) {
            this._elementRef = _elementRef;
            this._focusMonitor = _focusMonitor;
            this.cdkFocusChange = new EventEmitter();
            this._monitorSubscription = this._focusMonitor.monitor(this._elementRef, this._elementRef.nativeElement.hasAttribute('cdkMonitorSubtreeFocus'))
                .subscribe((/**
             * @param {?} origin
             * @return {?}
             */
            origin => this.cdkFocusChange.emit(origin)));
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this._focusMonitor.stopMonitoring(this._elementRef);
            this._monitorSubscription.unsubscribe();
        }
    }
    CdkMonitorFocus.ɵfac = function CdkMonitorFocus_Factory(t) { return new (t || CdkMonitorFocus)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(FocusMonitor)); };
    CdkMonitorFocus.ɵdir = ɵɵdefineDirective({ type: CdkMonitorFocus, selectors: [["", "cdkMonitorElementFocus", ""], ["", "cdkMonitorSubtreeFocus", ""]], outputs: { cdkFocusChange: "cdkFocusChange" } });
    /** @nocollapse */
    CdkMonitorFocus.ctorParameters = () => [
        { type: ElementRef },
        { type: FocusMonitor }
    ];
    CdkMonitorFocus.propDecorators = {
        cdkFocusChange: [{ type: Output }]
    };
    /**
     * CSS class applied to the document body when in black-on-white high-contrast mode.
     * @type {?}
     */
    const BLACK_ON_WHITE_CSS_CLASS = 'cdk-high-contrast-black-on-white';
    /**
     * CSS class applied to the document body when in white-on-black high-contrast mode.
     * @type {?}
     */
    const WHITE_ON_BLACK_CSS_CLASS = 'cdk-high-contrast-white-on-black';
    /**
     * CSS class applied to the document body when in high-contrast mode.
     * @type {?}
     */
    const HIGH_CONTRAST_MODE_ACTIVE_CSS_CLASS = 'cdk-high-contrast-active';
    /**
     * Service to determine whether the browser is currently in a high-contrast-mode environment.
     *
     * Microsoft Windows supports an accessibility feature called "High Contrast Mode". This mode
     * changes the appearance of all applications, including web applications, to dramatically increase
     * contrast.
     *
     * IE, Edge, and Firefox currently support this mode. Chrome does not support Windows High Contrast
     * Mode. This service does not detect high-contrast mode as added by the Chrome "High Contrast"
     * browser extension.
     */
    class HighContrastModeDetector {
        /**
         * @param {?} _platform
         * @param {?} document
         */
        constructor(_platform, document) {
            this._platform = _platform;
            this._document = document;
        }
        /**
         * Gets the current high-contrast-mode for the page.
         * @return {?}
         */
        getHighContrastMode() {
            if (!this._platform.isBrowser) {
                return 0 /* NONE */;
            }
            // Create a test element with an arbitrary background-color that is neither black nor
            // white; high-contrast mode will coerce the color to either black or white. Also ensure that
            // appending the test element to the DOM does not affect layout by absolutely positioning it
            /** @type {?} */
            const testElement = this._document.createElement('div');
            testElement.style.backgroundColor = 'rgb(1,2,3)';
            testElement.style.position = 'absolute';
            this._document.body.appendChild(testElement);
            // Get the computed style for the background color, collapsing spaces to normalize between
            // browsers. Once we get this color, we no longer need the test element. Access the `window`
            // via the document so we can fake it in tests. Note that we have extra null checks, because
            // this logic will likely run during app bootstrap and throwing can break the entire app.
            /** @type {?} */
            const documentWindow = this._document.defaultView || window;
            /** @type {?} */
            const computedStyle = (documentWindow && documentWindow.getComputedStyle) ?
                documentWindow.getComputedStyle(testElement) : null;
            /** @type {?} */
            const computedColor = (computedStyle && computedStyle.backgroundColor || '').replace(/ /g, '');
            this._document.body.removeChild(testElement);
            switch (computedColor) {
                case 'rgb(0,0,0)': return 2 /* WHITE_ON_BLACK */;
                case 'rgb(255,255,255)': return 1 /* BLACK_ON_WHITE */;
            }
            return 0 /* NONE */;
        }
        /**
         * Applies CSS classes indicating high-contrast mode to document body (browser-only).
         * @return {?}
         */
        _applyBodyHighContrastModeCssClasses() {
            if (this._platform.isBrowser && this._document.body) {
                /** @type {?} */
                const bodyClasses = this._document.body.classList;
                // IE11 doesn't support `classList` operations with multiple arguments
                bodyClasses.remove(HIGH_CONTRAST_MODE_ACTIVE_CSS_CLASS);
                bodyClasses.remove(BLACK_ON_WHITE_CSS_CLASS);
                bodyClasses.remove(WHITE_ON_BLACK_CSS_CLASS);
                /** @type {?} */
                const mode = this.getHighContrastMode();
                if (mode === 1 /* BLACK_ON_WHITE */) {
                    bodyClasses.add(HIGH_CONTRAST_MODE_ACTIVE_CSS_CLASS);
                    bodyClasses.add(BLACK_ON_WHITE_CSS_CLASS);
                }
                else if (mode === 2 /* WHITE_ON_BLACK */) {
                    bodyClasses.add(HIGH_CONTRAST_MODE_ACTIVE_CSS_CLASS);
                    bodyClasses.add(WHITE_ON_BLACK_CSS_CLASS);
                }
            }
        }
    }
    HighContrastModeDetector.ɵfac = function HighContrastModeDetector_Factory(t) { return new (t || HighContrastModeDetector)(ɵɵinject(Platform), ɵɵinject(DOCUMENT$1)); };
    /** @nocollapse */
    HighContrastModeDetector.ctorParameters = () => [
        { type: Platform },
        { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT$1,] }] }
    ];
    /** @nocollapse */ HighContrastModeDetector.ɵprov = ɵɵdefineInjectable({ factory: function HighContrastModeDetector_Factory() { return new HighContrastModeDetector(ɵɵinject(Platform), ɵɵinject(DOCUMENT$1)); }, token: HighContrastModeDetector, providedIn: "root" });

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/a11y/a11y-module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class A11yModule {
        /**
         * @param {?} highContrastModeDetector
         */
        constructor(highContrastModeDetector) {
            highContrastModeDetector._applyBodyHighContrastModeCssClasses();
        }
    }
    A11yModule.ɵmod = ɵɵdefineNgModule({ type: A11yModule });
    A11yModule.ɵinj = ɵɵdefineInjector({ factory: function A11yModule_Factory(t) { return new (t || A11yModule)(ɵɵinject(HighContrastModeDetector)); }, imports: [[PlatformModule, ObserversModule]] });
    /** @nocollapse */
    A11yModule.ctorParameters = () => [
        { type: HighContrastModeDetector }
    ];
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(A11yModule, { declarations: function () { return [CdkAriaLive,
            CdkTrapFocus,
            CdkMonitorFocus]; }, imports: function () { return [PlatformModule, ObserversModule]; }, exports: function () { return [CdkAriaLive,
            CdkTrapFocus,
            CdkMonitorFocus]; } }); })();

    const DIR_DOCUMENT = new InjectionToken('cdk-dir-doc', {
        providedIn: 'root',
        factory: DIR_DOCUMENT_FACTORY,
    });
    /**
     * \@docs-private
     * @return {?}
     */
    function DIR_DOCUMENT_FACTORY() {
        return inject(DOCUMENT$1);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/bidi/directionality.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * The directionality (LTR / RTL) context for the application (or a subtree of it).
     * Exposes the current direction and a stream of direction changes.
     */
    class Directionality {
        /**
         * @param {?=} _document
         */
        constructor(_document) {
            /**
             * The current 'ltr' or 'rtl' value.
             */
            this.value = 'ltr';
            /**
             * Stream that emits whenever the 'ltr' / 'rtl' state changes.
             */
            this.change = new EventEmitter();
            if (_document) {
                // TODO: handle 'auto' value -
                // We still need to account for dir="auto".
                // It looks like HTMLElemenet.dir is also "auto" when that's set to the attribute,
                // but getComputedStyle return either "ltr" or "rtl". avoiding getComputedStyle for now
                /** @type {?} */
                const bodyDir = _document.body ? _document.body.dir : null;
                /** @type {?} */
                const htmlDir = _document.documentElement ? _document.documentElement.dir : null;
                /** @type {?} */
                const value = bodyDir || htmlDir;
                this.value = (value === 'ltr' || value === 'rtl') ? value : 'ltr';
            }
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this.change.complete();
        }
    }
    Directionality.ɵfac = function Directionality_Factory(t) { return new (t || Directionality)(ɵɵinject(DIR_DOCUMENT, 8)); };
    /** @nocollapse */
    Directionality.ctorParameters = () => [
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [DIR_DOCUMENT,] }] }
    ];
    /** @nocollapse */ Directionality.ɵprov = ɵɵdefineInjectable({ factory: function Directionality_Factory() { return new Directionality(ɵɵinject(DIR_DOCUMENT, 8)); }, token: Directionality, providedIn: "root" });

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/bidi/dir.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Directive to listen for changes of direction of part of the DOM.
     *
     * Provides itself as Directionality such that descendant directives only need to ever inject
     * Directionality to get the closest direction.
     */
    class Dir {
        constructor() {
            /**
             * Normalized direction that accounts for invalid/unsupported values.
             */
            this._dir = 'ltr';
            /**
             * Whether the `value` has been set to its initial value.
             */
            this._isInitialized = false;
            /**
             * Event emitted when the direction changes.
             */
            this.change = new EventEmitter();
        }
        /**
         * \@docs-private
         * @return {?}
         */
        get dir() { return this._dir; }
        /**
         * @param {?} value
         * @return {?}
         */
        set dir(value) {
            /** @type {?} */
            const old = this._dir;
            /** @type {?} */
            const normalizedValue = value ? value.toLowerCase() : value;
            this._rawDir = value;
            this._dir = (normalizedValue === 'ltr' || normalizedValue === 'rtl') ? normalizedValue : 'ltr';
            if (old !== this._dir && this._isInitialized) {
                this.change.emit(this._dir);
            }
        }
        /**
         * Current layout direction of the element.
         * @return {?}
         */
        get value() { return this.dir; }
        /**
         * Initialize once default value has been set.
         * @return {?}
         */
        ngAfterContentInit() {
            this._isInitialized = true;
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this.change.complete();
        }
    }
    Dir.ɵfac = function Dir_Factory(t) { return new (t || Dir)(); };
    Dir.ɵdir = ɵɵdefineDirective({ type: Dir, selectors: [["", "dir", ""]], hostVars: 1, hostBindings: function Dir_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵattribute("dir", ctx._rawDir);
        } }, inputs: { dir: "dir" }, outputs: { change: "dirChange" }, exportAs: ["dir"], features: [ɵɵProvidersFeature([{ provide: Directionality, useExisting: Dir }])] });
    Dir.propDecorators = {
        change: [{ type: Output, args: ['dirChange',] }],
        dir: [{ type: Input }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/bidi/bidi-module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class BidiModule {
    }
    BidiModule.ɵmod = ɵɵdefineNgModule({ type: BidiModule });
    BidiModule.ɵinj = ɵɵdefineInjector({ factory: function BidiModule_Factory(t) { return new (t || BidiModule)(); } });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(BidiModule, { declarations: [Dir], exports: [Dir] }); })();

    /**
     * @license
     * Copyright Google LLC All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /** Current version of the Angular Component Development Kit. */
    const VERSION$3 = new Version('9.2.4');

    const _c0 = ["*", [["mat-option"], ["ng-container"]]];
    const _c1 = ["*", "mat-option, ng-container"];
    function MatOption_mat_pseudo_checkbox_0_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "mat-pseudo-checkbox", 3);
    } if (rf & 2) {
        const ctx_r0 = ɵɵnextContext();
        ɵɵproperty("state", ctx_r0.selected ? "checked" : "unchecked")("disabled", ctx_r0.disabled);
    } }
    const _c2 = ["*"];
    const VERSION$4 = new Version('9.2.4');

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/core/animation/animation.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google LLC All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * \@docs-private
     */
    class AnimationCurves {
    }
    AnimationCurves.STANDARD_CURVE = 'cubic-bezier(0.4,0.0,0.2,1)';
    AnimationCurves.DECELERATION_CURVE = 'cubic-bezier(0.0,0.0,0.2,1)';
    AnimationCurves.ACCELERATION_CURVE = 'cubic-bezier(0.4,0.0,1,1)';
    AnimationCurves.SHARP_CURVE = 'cubic-bezier(0.4,0.0,0.6,1)';
    /**
     * \@docs-private
     */
    class AnimationDurations {
    }
    AnimationDurations.COMPLEX = '375ms';
    AnimationDurations.ENTERING = '225ms';
    AnimationDurations.EXITING = '195ms';

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/core/common-behaviors/common-module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    // Private version constant to circumvent test/build issues,
    // i.e. avoid core to depend on the @angular/material primary entry-point
    // Can be removed once the Material primary entry-point no longer
    // re-exports all secondary entry-points
    /** @type {?} */
    const VERSION$1$1 = new Version('9.2.4');
    /**
     * \@docs-private
     * @return {?}
     */
    function MATERIAL_SANITY_CHECKS_FACTORY() {
        return true;
    }
    /**
     * Injection token that configures whether the Material sanity checks are enabled.
     * @type {?}
     */
    const MATERIAL_SANITY_CHECKS = new InjectionToken('mat-sanity-checks', {
        providedIn: 'root',
        factory: MATERIAL_SANITY_CHECKS_FACTORY,
    });
    /**
     * Module that captures anything that should be loaded and/or run for *all* Angular Material
     * components. This includes Bidi, etc.
     *
     * This module should be imported to each top-level component module (e.g., MatTabsModule).
     */
    class MatCommonModule {
        /**
         * @param {?} highContrastModeDetector
         * @param {?} sanityChecks
         * @param {?=} document
         */
        constructor(highContrastModeDetector, sanityChecks, 
        /** @breaking-change 11.0.0 make document required */
        document) {
            /**
             * Whether we've done the global sanity checks (e.g. a theme is loaded, there is a doctype).
             */
            this._hasDoneGlobalChecks = false;
            this._document = document;
            // While A11yModule also does this, we repeat it here to avoid importing A11yModule
            // in MatCommonModule.
            highContrastModeDetector._applyBodyHighContrastModeCssClasses();
            // Note that `_sanityChecks` is typed to `any`, because AoT
            // throws an error if we use the `SanityChecks` type directly.
            this._sanityChecks = sanityChecks;
            if (!this._hasDoneGlobalChecks) {
                this._checkDoctypeIsDefined();
                this._checkThemeIsPresent();
                this._checkCdkVersionMatch();
                this._hasDoneGlobalChecks = true;
            }
        }
        /**
         * Access injected document if available or fallback to global document reference
         * @private
         * @return {?}
         */
        _getDocument() {
            /** @type {?} */
            const doc = this._document || document;
            return typeof doc === 'object' && doc ? doc : null;
        }
        /**
         * Use defaultView of injected document if available or fallback to global window reference
         * @private
         * @return {?}
         */
        _getWindow() {
            /** @type {?} */
            const doc = this._getDocument();
            /** @type {?} */
            const win = (doc === null || doc === void 0 ? void 0 : doc.defaultView) || window;
            return typeof win === 'object' && win ? win : null;
        }
        /**
         * Whether any sanity checks are enabled.
         * @private
         * @return {?}
         */
        _checksAreEnabled() {
            return isDevMode() && !this._isTestEnv();
        }
        /**
         * Whether the code is running in tests.
         * @private
         * @return {?}
         */
        _isTestEnv() {
            /** @type {?} */
            const window = (/** @type {?} */ (this._getWindow()));
            return window && (window.__karma__ || window.jasmine);
        }
        /**
         * @private
         * @return {?}
         */
        _checkDoctypeIsDefined() {
            /** @type {?} */
            const isEnabled = this._checksAreEnabled() &&
                (this._sanityChecks === true || ((/** @type {?} */ (this._sanityChecks))).doctype);
            /** @type {?} */
            const document = this._getDocument();
            if (isEnabled && document && !document.doctype) {
                console.warn('Current document does not have a doctype. This may cause ' +
                    'some Angular Material components not to behave as expected.');
            }
        }
        /**
         * @private
         * @return {?}
         */
        _checkThemeIsPresent() {
            // We need to assert that the `body` is defined, because these checks run very early
            // and the `body` won't be defined if the consumer put their scripts in the `head`.
            /** @type {?} */
            const isDisabled = !this._checksAreEnabled() ||
                (this._sanityChecks === false || !((/** @type {?} */ (this._sanityChecks))).theme);
            /** @type {?} */
            const document = this._getDocument();
            if (isDisabled || !document || !document.body ||
                typeof getComputedStyle !== 'function') {
                return;
            }
            /** @type {?} */
            const testElement = document.createElement('div');
            testElement.classList.add('mat-theme-loaded-marker');
            document.body.appendChild(testElement);
            /** @type {?} */
            const computedStyle = getComputedStyle(testElement);
            // In some situations the computed style of the test element can be null. For example in
            // Firefox, the computed style is null if an application is running inside of a hidden iframe.
            // See: https://bugzilla.mozilla.org/show_bug.cgi?id=548397
            if (computedStyle && computedStyle.display !== 'none') {
                console.warn('Could not find Angular Material core theme. Most Material ' +
                    'components may not work as expected. For more info refer ' +
                    'to the theming guide: https://material.angular.io/guide/theming');
            }
            document.body.removeChild(testElement);
        }
        /**
         * Checks whether the material version matches the cdk version
         * @private
         * @return {?}
         */
        _checkCdkVersionMatch() {
            /** @type {?} */
            const isEnabled = this._checksAreEnabled() &&
                (this._sanityChecks === true || ((/** @type {?} */ (this._sanityChecks))).version);
            if (isEnabled && VERSION$1$1.full !== VERSION$3.full) {
                console.warn('The Angular Material version (' + VERSION$1$1.full + ') does not match ' +
                    'the Angular CDK version (' + VERSION$3.full + ').\n' +
                    'Please ensure the versions of these two packages exactly match.');
            }
        }
    }
    MatCommonModule.ɵmod = ɵɵdefineNgModule({ type: MatCommonModule });
    MatCommonModule.ɵinj = ɵɵdefineInjector({ factory: function MatCommonModule_Factory(t) { return new (t || MatCommonModule)(ɵɵinject(HighContrastModeDetector), ɵɵinject(MATERIAL_SANITY_CHECKS, 8), ɵɵinject(DOCUMENT$1, 8)); }, imports: [[BidiModule],
            BidiModule] });
    /** @nocollapse */
    MatCommonModule.ctorParameters = () => [
        { type: HighContrastModeDetector },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [MATERIAL_SANITY_CHECKS,] }] },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [DOCUMENT$1,] }] }
    ];
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(MatCommonModule, { imports: function () { return [BidiModule]; }, exports: function () { return [BidiModule]; } }); })();
    /**
     * Mixin to augment a directive with a `disabled` property.
     * @template T
     * @param {?} base
     * @return {?}
     */
    function mixinDisabled(base) {
        return class extends base {
            /**
             * @param {...?} args
             */
            constructor(...args) {
                super(...args);
                this._disabled = false;
            }
            /**
             * @return {?}
             */
            get disabled() { return this._disabled; }
            /**
             * @param {?} value
             * @return {?}
             */
            set disabled(value) { this._disabled = coerceBooleanProperty(value); }
        };
    }
    /**
     * Mixin to augment a directive with a `color` property.
     * @template T
     * @param {?} base
     * @param {?=} defaultColor
     * @return {?}
     */
    function mixinColor(base, defaultColor) {
        return class extends base {
            /**
             * @param {...?} args
             */
            constructor(...args) {
                super(...args);
                // Set the default color that can be specified from the mixin.
                this.color = defaultColor;
            }
            /**
             * @return {?}
             */
            get color() { return this._color; }
            /**
             * @param {?} value
             * @return {?}
             */
            set color(value) {
                /** @type {?} */
                const colorPalette = value || defaultColor;
                if (colorPalette !== this._color) {
                    if (this._color) {
                        this._elementRef.nativeElement.classList.remove(`mat-${this._color}`);
                    }
                    if (colorPalette) {
                        this._elementRef.nativeElement.classList.add(`mat-${colorPalette}`);
                    }
                    this._color = colorPalette;
                }
            }
        };
    }
    /**
     * Mixin to augment a directive with a `disableRipple` property.
     * @template T
     * @param {?} base
     * @return {?}
     */
    function mixinDisableRipple(base) {
        return class extends base {
            /**
             * @param {...?} args
             */
            constructor(...args) {
                super(...args);
                this._disableRipple = false;
            }
            /**
             * Whether the ripple effect is disabled or not.
             * @return {?}
             */
            get disableRipple() { return this._disableRipple; }
            /**
             * @param {?} value
             * @return {?}
             */
            set disableRipple(value) { this._disableRipple = coerceBooleanProperty(value); }
        };
    }
    /**
     * Mixin to augment a directive with a `tabIndex` property.
     * @template T
     * @param {?} base
     * @param {?=} defaultTabIndex
     * @return {?}
     */
    function mixinTabIndex(base, defaultTabIndex = 0) {
        return class extends base {
            /**
             * @param {...?} args
             */
            constructor(...args) {
                super(...args);
                this._tabIndex = defaultTabIndex;
            }
            /**
             * @return {?}
             */
            get tabIndex() { return this.disabled ? -1 : this._tabIndex; }
            /**
             * @param {?} value
             * @return {?}
             */
            set tabIndex(value) {
                // If the specified tabIndex value is null or undefined, fall back to the default value.
                this._tabIndex = value != null ? value : defaultTabIndex;
            }
        };
    }
    /**
     * Mixin to augment a directive with updateErrorState method.
     * For component with `errorState` and need to update `errorState`.
     * @template T
     * @param {?} base
     * @return {?}
     */
    function mixinErrorState(base) {
        return class extends base {
            /**
             * @param {...?} args
             */
            constructor(...args) {
                super(...args);
                /**
                 * Whether the component is in an error state.
                 */
                this.errorState = false;
                /**
                 * Stream that emits whenever the state of the input changes such that the wrapping
                 * `MatFormField` needs to run change detection.
                 */
                this.stateChanges = new Subject();
            }
            /**
             * @return {?}
             */
            updateErrorState() {
                /** @type {?} */
                const oldState = this.errorState;
                /** @type {?} */
                const parent = this._parentFormGroup || this._parentForm;
                /** @type {?} */
                const matcher = this.errorStateMatcher || this._defaultErrorStateMatcher;
                /** @type {?} */
                const control = this.ngControl ? (/** @type {?} */ (this.ngControl.control)) : null;
                /** @type {?} */
                const newState = matcher.isErrorState(control, parent);
                if (newState !== oldState) {
                    this.errorState = newState;
                    this.stateChanges.next();
                }
            }
        };
    }
    /**
     * Mixin to augment a directive with an initialized property that will emits when ngOnInit ends.
     * @template T
     * @param {?} base
     * @return {?}
     */
    function mixinInitialized(base) {
        return class extends base {
            /**
             * @param {...?} args
             */
            constructor(...args) {
                super(...args);
                /**
                 * Whether this directive has been marked as initialized.
                 */
                this._isInitialized = false;
                /**
                 * List of subscribers that subscribed before the directive was initialized. Should be notified
                 * during _markInitialized. Set to null after pending subscribers are notified, and should
                 * not expect to be populated after.
                 */
                this._pendingSubscribers = [];
                /**
                 * Observable stream that emits when the directive initializes. If already initialized, the
                 * subscriber is stored to be notified once _markInitialized is called.
                 */
                this.initialized = new Observable((/**
                 * @param {?} subscriber
                 * @return {?}
                 */
                subscriber => {
                    // If initialized, immediately notify the subscriber. Otherwise store the subscriber to notify
                    // when _markInitialized is called.
                    if (this._isInitialized) {
                        this._notifySubscriber(subscriber);
                    }
                    else {
                        (/** @type {?} */ (this._pendingSubscribers)).push(subscriber);
                    }
                }));
            }
            /**
             * Marks the state as initialized and notifies pending subscribers. Should be called at the end
             * of ngOnInit.
             * \@docs-private
             * @return {?}
             */
            _markInitialized() {
                if (this._isInitialized) {
                    throw Error('This directive has already been marked as initialized and ' +
                        'should not be called twice.');
                }
                this._isInitialized = true;
                (/** @type {?} */ (this._pendingSubscribers)).forEach(this._notifySubscriber);
                this._pendingSubscribers = null;
            }
            /**
             * Emits and completes the subscriber stream (should only emit once).
             * @param {?} subscriber
             * @return {?}
             */
            _notifySubscriber(subscriber) {
                subscriber.next();
                subscriber.complete();
            }
        };
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/core/common-behaviors/index.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/core/datetime/date-adapter.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * InjectionToken for datepicker that can be used to override default locale code.
     * @type {?}
     */
    const MAT_DATE_LOCALE = new InjectionToken('MAT_DATE_LOCALE', {
        providedIn: 'root',
        factory: MAT_DATE_LOCALE_FACTORY,
    });
    /**
     * \@docs-private
     * @return {?}
     */
    function MAT_DATE_LOCALE_FACTORY() {
        return inject(LOCALE_ID$1);
    }
    /**
     * Adapts type `D` to be usable as a date by cdk-based components that work with dates.
     * @abstract
     * @template D
     */
    class DateAdapter {
        constructor() {
            this._localeChanges = new Subject();
        }
        /**
         * A stream that emits when the locale changes.
         * @return {?}
         */
        get localeChanges() { return this._localeChanges; }
        /**
         * Attempts to deserialize a value to a valid date object. This is different from parsing in that
         * deserialize should only accept non-ambiguous, locale-independent formats (e.g. a ISO 8601
         * string). The default implementation does not allow any deserialization, it simply checks that
         * the given value is already a valid date object or null. The `<mat-datepicker>` will call this
         * method on all of its `\@Input()` properties that accept dates. It is therefore possible to
         * support passing values from your backend directly to these properties by overriding this method
         * to also deserialize the format used by your backend.
         * @param {?} value The value to be deserialized into a date object.
         * @return {?} The deserialized date object, either a valid date, null if the value can be
         *     deserialized into a null date (e.g. the empty string), or an invalid date.
         */
        deserialize(value) {
            if (value == null || this.isDateInstance(value) && this.isValid(value)) {
                return value;
            }
            return this.invalid();
        }
        /**
         * Sets the locale used for all dates.
         * @param {?} locale The new locale.
         * @return {?}
         */
        setLocale(locale) {
            this.locale = locale;
            this._localeChanges.next();
        }
        /**
         * Compares two dates.
         * @param {?} first The first date to compare.
         * @param {?} second The second date to compare.
         * @return {?} 0 if the dates are equal, a number less than 0 if the first date is earlier,
         *     a number greater than 0 if the first date is later.
         */
        compareDate(first, second) {
            return this.getYear(first) - this.getYear(second) ||
                this.getMonth(first) - this.getMonth(second) ||
                this.getDate(first) - this.getDate(second);
        }
        /**
         * Checks if two dates are equal.
         * @param {?} first The first date to check.
         * @param {?} second The second date to check.
         * @return {?} Whether the two dates are equal.
         *     Null dates are considered equal to other null dates.
         */
        sameDate(first, second) {
            if (first && second) {
                /** @type {?} */
                let firstValid = this.isValid(first);
                /** @type {?} */
                let secondValid = this.isValid(second);
                if (firstValid && secondValid) {
                    return !this.compareDate(first, second);
                }
                return firstValid == secondValid;
            }
            return first == second;
        }
        /**
         * Clamp the given date between min and max dates.
         * @param {?} date The date to clamp.
         * @param {?=} min The minimum value to allow. If null or omitted no min is enforced.
         * @param {?=} max The maximum value to allow. If null or omitted no max is enforced.
         * @return {?} `min` if `date` is less than `min`, `max` if date is greater than `max`,
         *     otherwise `date`.
         */
        clampDate(date, min, max) {
            if (min && this.compareDate(date, min) < 0) {
                return min;
            }
            if (max && this.compareDate(date, max) > 0) {
                return max;
            }
            return date;
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/core/datetime/date-formats.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const MAT_DATE_FORMATS = new InjectionToken('mat-date-formats');

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/core/datetime/native-date-adapter.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    // TODO(mmalerba): Remove when we no longer support safari 9.
    /**
     * Whether the browser supports the Intl API.
     * @type {?}
     */
    let SUPPORTS_INTL_API;
    // We need a try/catch around the reference to `Intl`, because accessing it in some cases can
    // cause IE to throw. These cases are tied to particular versions of Windows and can happen if
    // the consumer is providing a polyfilled `Map`. See:
    // https://github.com/Microsoft/ChakraCore/issues/3189
    // https://github.com/angular/components/issues/15687
    try {
        SUPPORTS_INTL_API = typeof Intl != 'undefined';
    }
    catch (_a) {
        SUPPORTS_INTL_API = false;
    }
    /**
     * The default month names to use if Intl API is not available.
     * @type {?}
     */
    const DEFAULT_MONTH_NAMES = {
        'long': [
            'January', 'February', 'March', 'April', 'May', 'June', 'July', 'August', 'September',
            'October', 'November', 'December'
        ],
        'short': ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec'],
        'narrow': ['J', 'F', 'M', 'A', 'M', 'J', 'J', 'A', 'S', 'O', 'N', 'D']
    };
    const ɵ0$9 = /**
     * @param {?} i
     * @return {?}
     */
    i => String(i + 1);
    /**
     * The default date names to use if Intl API is not available.
     * @type {?}
     */
    const DEFAULT_DATE_NAMES = range(31, (ɵ0$9));
    /**
     * The default day of the week names to use if Intl API is not available.
     * @type {?}
     */
    const DEFAULT_DAY_OF_WEEK_NAMES = {
        'long': ['Sunday', 'Monday', 'Tuesday', 'Wednesday', 'Thursday', 'Friday', 'Saturday'],
        'short': ['Sun', 'Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat'],
        'narrow': ['S', 'M', 'T', 'W', 'T', 'F', 'S']
    };
    /**
     * Matches strings that have the form of a valid RFC 3339 string
     * (https://tools.ietf.org/html/rfc3339). Note that the string may not actually be a valid date
     * because the regex will match strings an with out of bounds month, date, etc.
     * @type {?}
     */
    const ISO_8601_REGEX = /^\d{4}-\d{2}-\d{2}(?:T\d{2}:\d{2}:\d{2}(?:\.\d+)?(?:Z|(?:(?:\+|-)\d{2}:\d{2}))?)?$/;
    /**
     * Creates an array and fills it with values.
     * @template T
     * @param {?} length
     * @param {?} valueFunction
     * @return {?}
     */
    function range(length, valueFunction) {
        /** @type {?} */
        const valuesArray = Array(length);
        for (let i = 0; i < length; i++) {
            valuesArray[i] = valueFunction(i);
        }
        return valuesArray;
    }
    /**
     * Adapts the native JS Date for use with cdk-based components that work with dates.
     */
    class NativeDateAdapter extends DateAdapter {
        /**
         * @param {?} matDateLocale
         * @param {?} platform
         */
        constructor(matDateLocale, platform) {
            super();
            /**
             * Whether to use `timeZone: 'utc'` with `Intl.DateTimeFormat` when formatting dates.
             * Without this `Intl.DateTimeFormat` sometimes chooses the wrong timeZone, which can throw off
             * the result. (e.g. in the en-US locale `new Date(1800, 7, 14).toLocaleDateString()`
             * will produce `'8/13/1800'`.
             *
             * TODO(mmalerba): drop this variable. It's not being used in the code right now. We're now
             * getting the string representation of a Date object from its utc representation. We're keeping
             * it here for sometime, just for precaution, in case we decide to revert some of these changes
             * though.
             */
            this.useUtcForDisplay = true;
            super.setLocale(matDateLocale);
            // IE does its own time zone correction, so we disable this on IE.
            this.useUtcForDisplay = !platform.TRIDENT;
            this._clampDate = platform.TRIDENT || platform.EDGE;
        }
        /**
         * @param {?} date
         * @return {?}
         */
        getYear(date) {
            return date.getFullYear();
        }
        /**
         * @param {?} date
         * @return {?}
         */
        getMonth(date) {
            return date.getMonth();
        }
        /**
         * @param {?} date
         * @return {?}
         */
        getDate(date) {
            return date.getDate();
        }
        /**
         * @param {?} date
         * @return {?}
         */
        getDayOfWeek(date) {
            return date.getDay();
        }
        /**
         * @param {?} style
         * @return {?}
         */
        getMonthNames(style) {
            if (SUPPORTS_INTL_API) {
                /** @type {?} */
                const dtf = new Intl.DateTimeFormat(this.locale, { month: style, timeZone: 'utc' });
                return range(12, (/**
                 * @param {?} i
                 * @return {?}
                 */
                i => this._stripDirectionalityCharacters(this._format(dtf, new Date(2017, i, 1)))));
            }
            return DEFAULT_MONTH_NAMES[style];
        }
        /**
         * @return {?}
         */
        getDateNames() {
            if (SUPPORTS_INTL_API) {
                /** @type {?} */
                const dtf = new Intl.DateTimeFormat(this.locale, { day: 'numeric', timeZone: 'utc' });
                return range(31, (/**
                 * @param {?} i
                 * @return {?}
                 */
                i => this._stripDirectionalityCharacters(this._format(dtf, new Date(2017, 0, i + 1)))));
            }
            return DEFAULT_DATE_NAMES;
        }
        /**
         * @param {?} style
         * @return {?}
         */
        getDayOfWeekNames(style) {
            if (SUPPORTS_INTL_API) {
                /** @type {?} */
                const dtf = new Intl.DateTimeFormat(this.locale, { weekday: style, timeZone: 'utc' });
                return range(7, (/**
                 * @param {?} i
                 * @return {?}
                 */
                i => this._stripDirectionalityCharacters(this._format(dtf, new Date(2017, 0, i + 1)))));
            }
            return DEFAULT_DAY_OF_WEEK_NAMES[style];
        }
        /**
         * @param {?} date
         * @return {?}
         */
        getYearName(date) {
            if (SUPPORTS_INTL_API) {
                /** @type {?} */
                const dtf = new Intl.DateTimeFormat(this.locale, { year: 'numeric', timeZone: 'utc' });
                return this._stripDirectionalityCharacters(this._format(dtf, date));
            }
            return String(this.getYear(date));
        }
        /**
         * @return {?}
         */
        getFirstDayOfWeek() {
            // We can't tell using native JS Date what the first day of the week is, we default to Sunday.
            return 0;
        }
        /**
         * @param {?} date
         * @return {?}
         */
        getNumDaysInMonth(date) {
            return this.getDate(this._createDateWithOverflow(this.getYear(date), this.getMonth(date) + 1, 0));
        }
        /**
         * @param {?} date
         * @return {?}
         */
        clone(date) {
            return new Date(date.getTime());
        }
        /**
         * @param {?} year
         * @param {?} month
         * @param {?} date
         * @return {?}
         */
        createDate(year, month, date) {
            // Check for invalid month and date (except upper bound on date which we have to check after
            // creating the Date).
            if (month < 0 || month > 11) {
                throw Error(`Invalid month index "${month}". Month index has to be between 0 and 11.`);
            }
            if (date < 1) {
                throw Error(`Invalid date "${date}". Date has to be greater than 0.`);
            }
            /** @type {?} */
            let result = this._createDateWithOverflow(year, month, date);
            // Check that the date wasn't above the upper bound for the month, causing the month to overflow
            if (result.getMonth() != month) {
                throw Error(`Invalid date "${date}" for month with index "${month}".`);
            }
            return result;
        }
        /**
         * @return {?}
         */
        today() {
            return new Date();
        }
        /**
         * @param {?} value
         * @return {?}
         */
        parse(value) {
            // We have no way using the native JS Date to set the parse format or locale, so we ignore these
            // parameters.
            if (typeof value == 'number') {
                return new Date(value);
            }
            return value ? new Date(Date.parse(value)) : null;
        }
        /**
         * @param {?} date
         * @param {?} displayFormat
         * @return {?}
         */
        format(date, displayFormat) {
            if (!this.isValid(date)) {
                throw Error('NativeDateAdapter: Cannot format invalid date.');
            }
            if (SUPPORTS_INTL_API) {
                // On IE and Edge the i18n API will throw a hard error that can crash the entire app
                // if we attempt to format a date whose year is less than 1 or greater than 9999.
                if (this._clampDate && (date.getFullYear() < 1 || date.getFullYear() > 9999)) {
                    date = this.clone(date);
                    date.setFullYear(Math.max(1, Math.min(9999, date.getFullYear())));
                }
                displayFormat = Object.assign(Object.assign({}, displayFormat), { timeZone: 'utc' });
                /** @type {?} */
                const dtf = new Intl.DateTimeFormat(this.locale, displayFormat);
                return this._stripDirectionalityCharacters(this._format(dtf, date));
            }
            return this._stripDirectionalityCharacters(date.toDateString());
        }
        /**
         * @param {?} date
         * @param {?} years
         * @return {?}
         */
        addCalendarYears(date, years) {
            return this.addCalendarMonths(date, years * 12);
        }
        /**
         * @param {?} date
         * @param {?} months
         * @return {?}
         */
        addCalendarMonths(date, months) {
            /** @type {?} */
            let newDate = this._createDateWithOverflow(this.getYear(date), this.getMonth(date) + months, this.getDate(date));
            // It's possible to wind up in the wrong month if the original month has more days than the new
            // month. In this case we want to go to the last day of the desired month.
            // Note: the additional + 12 % 12 ensures we end up with a positive number, since JS % doesn't
            // guarantee this.
            if (this.getMonth(newDate) != ((this.getMonth(date) + months) % 12 + 12) % 12) {
                newDate = this._createDateWithOverflow(this.getYear(newDate), this.getMonth(newDate), 0);
            }
            return newDate;
        }
        /**
         * @param {?} date
         * @param {?} days
         * @return {?}
         */
        addCalendarDays(date, days) {
            return this._createDateWithOverflow(this.getYear(date), this.getMonth(date), this.getDate(date) + days);
        }
        /**
         * @param {?} date
         * @return {?}
         */
        toIso8601(date) {
            return [
                date.getUTCFullYear(),
                this._2digit(date.getUTCMonth() + 1),
                this._2digit(date.getUTCDate())
            ].join('-');
        }
        /**
         * Returns the given value if given a valid Date or null. Deserializes valid ISO 8601 strings
         * (https://www.ietf.org/rfc/rfc3339.txt) into valid Dates and empty string into null. Returns an
         * invalid date for all other values.
         * @param {?} value
         * @return {?}
         */
        deserialize(value) {
            if (typeof value === 'string') {
                if (!value) {
                    return null;
                }
                // The `Date` constructor accepts formats other than ISO 8601, so we need to make sure the
                // string is the right format first.
                if (ISO_8601_REGEX.test(value)) {
                    /** @type {?} */
                    let date = new Date(value);
                    if (this.isValid(date)) {
                        return date;
                    }
                }
            }
            return super.deserialize(value);
        }
        /**
         * @param {?} obj
         * @return {?}
         */
        isDateInstance(obj) {
            return obj instanceof Date;
        }
        /**
         * @param {?} date
         * @return {?}
         */
        isValid(date) {
            return !isNaN(date.getTime());
        }
        /**
         * @return {?}
         */
        invalid() {
            return new Date(NaN);
        }
        /**
         * Creates a date but allows the month and date to overflow.
         * @private
         * @param {?} year
         * @param {?} month
         * @param {?} date
         * @return {?}
         */
        _createDateWithOverflow(year, month, date) {
            /** @type {?} */
            const result = new Date(year, month, date);
            // We need to correct for the fact that JS native Date treats years in range [0, 99] as
            // abbreviations for 19xx.
            if (year >= 0 && year < 100) {
                result.setFullYear(this.getYear(result) - 1900);
            }
            return result;
        }
        /**
         * Pads a number to make it two digits.
         * @private
         * @param {?} n The number to pad.
         * @return {?} The padded number.
         */
        _2digit(n) {
            return ('00' + n).slice(-2);
        }
        /**
         * Strip out unicode LTR and RTL characters. Edge and IE insert these into formatted dates while
         * other browsers do not. We remove them to make output consistent and because they interfere with
         * date parsing.
         * @private
         * @param {?} str The string to strip direction characters from.
         * @return {?} The stripped string.
         */
        _stripDirectionalityCharacters(str) {
            return str.replace(/[\u200e\u200f]/g, '');
        }
        /**
         * When converting Date object to string, javascript built-in functions may return wrong
         * results because it applies its internal DST rules. The DST rules around the world change
         * very frequently, and the current valid rule is not always valid in previous years though.
         * We work around this problem building a new Date object which has its internal UTC
         * representation with the local date and time.
         * @private
         * @param {?} dtf Intl.DateTimeFormat object, containg the desired string format. It must have
         *    timeZone set to 'utc' to work fine.
         * @param {?} date Date from which we want to get the string representation according to dtf
         * @return {?} A Date object with its UTC representation based on the passed in date info
         */
        _format(dtf, date) {
            /** @type {?} */
            const d = new Date(Date.UTC(date.getFullYear(), date.getMonth(), date.getDate(), date.getHours(), date.getMinutes(), date.getSeconds(), date.getMilliseconds()));
            return dtf.format(d);
        }
    }
    NativeDateAdapter.ɵfac = function NativeDateAdapter_Factory(t) { return new (t || NativeDateAdapter)(ɵɵinject(MAT_DATE_LOCALE, 8), ɵɵinject(Platform)); };
    NativeDateAdapter.ɵprov = ɵɵdefineInjectable({ token: NativeDateAdapter, factory: NativeDateAdapter.ɵfac });
    /** @nocollapse */
    NativeDateAdapter.ctorParameters = () => [
        { type: String, decorators: [{ type: Optional }, { type: Inject, args: [MAT_DATE_LOCALE,] }] },
        { type: Platform }
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/core/datetime/native-date-formats.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google LLC All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /** @type {?} */
    const MAT_NATIVE_DATE_FORMATS = {
        parse: {
            dateInput: null,
        },
        display: {
            dateInput: { year: 'numeric', month: 'numeric', day: 'numeric' },
            monthYearLabel: { year: 'numeric', month: 'short' },
            dateA11yLabel: { year: 'numeric', month: 'long', day: 'numeric' },
            monthYearA11yLabel: { year: 'numeric', month: 'long' },
        }
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/core/datetime/index.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class NativeDateModule {
    }
    NativeDateModule.ɵmod = ɵɵdefineNgModule({ type: NativeDateModule });
    NativeDateModule.ɵinj = ɵɵdefineInjector({ factory: function NativeDateModule_Factory(t) { return new (t || NativeDateModule)(); }, providers: [
            { provide: DateAdapter, useClass: NativeDateAdapter },
        ], imports: [[PlatformModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(NativeDateModule, { imports: function () { return [PlatformModule]; } }); })();
    const ɵ0$1$3 = MAT_NATIVE_DATE_FORMATS;
    class MatNativeDateModule {
    }
    MatNativeDateModule.ɵmod = ɵɵdefineNgModule({ type: MatNativeDateModule });
    MatNativeDateModule.ɵinj = ɵɵdefineInjector({ factory: function MatNativeDateModule_Factory(t) { return new (t || MatNativeDateModule)(); }, providers: [{ provide: MAT_DATE_FORMATS, useValue: ɵ0$1$3 }], imports: [[NativeDateModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(MatNativeDateModule, { imports: [NativeDateModule] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/core/error/error-options.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Error state matcher that matches when a control is invalid and dirty.
     */
    class ShowOnDirtyErrorStateMatcher {
        /**
         * @param {?} control
         * @param {?} form
         * @return {?}
         */
        isErrorState(control, form) {
            return !!(control && control.invalid && (control.dirty || (form && form.submitted)));
        }
    }
    ShowOnDirtyErrorStateMatcher.ɵfac = function ShowOnDirtyErrorStateMatcher_Factory(t) { return new (t || ShowOnDirtyErrorStateMatcher)(); };
    ShowOnDirtyErrorStateMatcher.ɵprov = ɵɵdefineInjectable({ token: ShowOnDirtyErrorStateMatcher, factory: ShowOnDirtyErrorStateMatcher.ɵfac });
    /**
     * Provider that defines how form controls behave with regards to displaying error messages.
     */
    class ErrorStateMatcher {
        /**
         * @param {?} control
         * @param {?} form
         * @return {?}
         */
        isErrorState(control, form) {
            return !!(control && control.invalid && (control.touched || (form && form.submitted)));
        }
    }
    ErrorStateMatcher.ɵfac = function ErrorStateMatcher_Factory(t) { return new (t || ErrorStateMatcher)(); };
    /** @nocollapse */ ErrorStateMatcher.ɵprov = ɵɵdefineInjectable({ factory: function ErrorStateMatcher_Factory() { return new ErrorStateMatcher(); }, token: ErrorStateMatcher, providedIn: "root" });

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/core/gestures/gesture-config.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Injection token that can be used to provide options to the Hammerjs instance.
     * More info at http://hammerjs.github.io/api/.
     * @deprecated No longer being used. To be removed.
     * \@breaking-change 10.0.0
     * @type {?}
     */
    const MAT_HAMMER_OPTIONS = new InjectionToken('MAT_HAMMER_OPTIONS');
    /** @type {?} */
    const ANGULAR_MATERIAL_SUPPORTED_HAMMER_GESTURES = [
        'longpress',
        'slide',
        'slidestart',
        'slideend',
        'slideright',
        'slideleft'
    ];
    const ɵ0$2$2 = /**
     * @return {?}
     */
    () => { }, ɵ1$5 = /**
     * @return {?}
     */
    () => { };
    /**
     * Fake HammerInstance that is used when a Hammer instance is requested when HammerJS has not
     * been loaded on the page.
     * @type {?}
     */
    const noopHammerInstance = {
        on: (ɵ0$2$2),
        off: (ɵ1$5),
    };
    /**
     * Adjusts configuration of our gesture library, Hammer.
     * @deprecated No longer being used. To be removed.
     * \@breaking-change 10.0.0
     */
    class GestureConfig extends HammerGestureConfig {
        /**
         * @param {?=} _hammerOptions
         * @param {?=} _commonModule
         */
        constructor(_hammerOptions, _commonModule) {
            super();
            this._hammerOptions = _hammerOptions;
            /**
             * List of new event names to add to the gesture support list
             */
            this.events = ANGULAR_MATERIAL_SUPPORTED_HAMMER_GESTURES;
        }
        /**
         * Builds Hammer instance manually to add custom recognizers that match the Material Design spec.
         *
         * Our gesture names come from the Material Design gestures spec:
         * https://material.io/design/#gestures-touch-mechanics
         *
         * More information on default recognizers can be found in Hammer docs:
         * http://hammerjs.github.io/recognizer-pan/
         * http://hammerjs.github.io/recognizer-press/
         *
         * @param {?} element Element to which to assign the new HammerJS gestures.
         * @return {?} Newly-created HammerJS instance.
         */
        buildHammer(element) {
            /** @type {?} */
            const hammer = typeof window !== 'undefined' ? ((/** @type {?} */ (window))).Hammer : null;
            if (!hammer) {
                // If HammerJS is not loaded here, return the noop HammerInstance. This is necessary to
                // ensure that omitting HammerJS completely will not cause any errors while *also* supporting
                // the lazy-loading of HammerJS via the HAMMER_LOADER token introduced in Angular 6.1.
                // Because we can't depend on HAMMER_LOADER's existance until 7.0, we have to always set
                // `this.events` to the set we support, instead of conditionally setting it to `[]` if
                // `HAMMER_LOADER` is present (and then throwing an Error here if `window.Hammer` is
                // undefined).
                // @breaking-change 8.0.0
                return noopHammerInstance;
            }
            /** @type {?} */
            const mc = new hammer(element, this._hammerOptions || undefined);
            // Default Hammer Recognizers.
            /** @type {?} */
            const pan = new hammer.Pan();
            /** @type {?} */
            const swipe = new hammer.Swipe();
            /** @type {?} */
            const press = new hammer.Press();
            // Notice that a HammerJS recognizer can only depend on one other recognizer once.
            // Otherwise the previous `recognizeWith` will be dropped.
            // TODO: Confirm threshold numbers with Material Design UX Team
            /** @type {?} */
            const slide = this._createRecognizer(pan, { event: 'slide', threshold: 0 }, swipe);
            /** @type {?} */
            const longpress = this._createRecognizer(press, { event: 'longpress', time: 500 });
            // Overwrite the default `pan` event to use the swipe event.
            pan.recognizeWith(swipe);
            // Since the slide event threshold is set to zero, the slide recognizer can fire and
            // accidentally reset the longpress recognizer. In order to make sure that the two
            // recognizers can run simultaneously but don't affect each other, we allow the slide
            // recognizer to recognize while a longpress is being processed.
            // See: https://github.com/hammerjs/hammer.js/blob/master/src/manager.js#L123-L124
            longpress.recognizeWith(slide);
            // Add customized gestures to Hammer manager
            mc.add([swipe, press, pan, slide, longpress]);
            return (/** @type {?} */ (mc));
        }
        /**
         * Creates a new recognizer, without affecting the default recognizers of HammerJS
         * @private
         * @param {?} base
         * @param {?} options
         * @param {...?} inheritances
         * @return {?}
         */
        _createRecognizer(base, options, ...inheritances) {
            /** @type {?} */
            let recognizer = new ((/** @type {?} */ (base.constructor)))(options);
            inheritances.push(base);
            inheritances.forEach((/**
             * @param {?} item
             * @return {?}
             */
            item => recognizer.recognizeWith(item)));
            return recognizer;
        }
    }
    GestureConfig.ɵfac = function GestureConfig_Factory(t) { return new (t || GestureConfig)(ɵɵinject(MAT_HAMMER_OPTIONS, 8), ɵɵinject(MatCommonModule, 8)); };
    GestureConfig.ɵprov = ɵɵdefineInjectable({ token: GestureConfig, factory: GestureConfig.ɵfac });
    /** @nocollapse */
    GestureConfig.ctorParameters = () => [
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [MAT_HAMMER_OPTIONS,] }] },
        { type: MatCommonModule, decorators: [{ type: Optional }] }
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/core/line/line.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Shared directive to count lines inside a text area, such as a list item.
     * Line elements can be extracted with a \@ContentChildren(MatLine) query, then
     * counted by checking the query list's length.
     */
    class MatLine {
    }
    MatLine.ɵfac = function MatLine_Factory(t) { return new (t || MatLine)(); };
    MatLine.ɵdir = ɵɵdefineDirective({ type: MatLine, selectors: [["", "mat-line", ""], ["", "matLine", ""]], hostAttrs: [1, "mat-line"] });
    class MatLineModule {
    }
    MatLineModule.ɵmod = ɵɵdefineNgModule({ type: MatLineModule });
    MatLineModule.ɵinj = ɵɵdefineInjector({ factory: function MatLineModule_Factory(t) { return new (t || MatLineModule)(); }, imports: [[MatCommonModule],
            MatCommonModule] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(MatLineModule, { declarations: [MatLine], imports: [MatCommonModule], exports: [MatLine,
            MatCommonModule] }); })();
    /**
     * Reference to a previously launched ripple element.
     */
    class RippleRef {
        /**
         * @param {?} _renderer
         * @param {?} element
         * @param {?} config
         */
        constructor(_renderer, element, config) {
            this._renderer = _renderer;
            this.element = element;
            this.config = config;
            /**
             * Current state of the ripple.
             */
            this.state = 3 /* HIDDEN */;
        }
        /**
         * Fades out the ripple element.
         * @return {?}
         */
        fadeOut() {
            this._renderer.fadeOutRipple(this);
        }
    }
    /**
     * Default ripple animation configuration for ripples without an explicit
     * animation config specified.
     * @type {?}
     */
    const defaultRippleAnimationConfig = {
        enterDuration: 450,
        exitDuration: 400
    };
    /**
     * Timeout for ignoring mouse events. Mouse events will be temporary ignored after touch
     * events to avoid synthetic mouse events.
     * @type {?}
     */
    const ignoreMouseEventsTimeout = 800;
    /**
     * Options that apply to all the event listeners that are bound by the ripple renderer.
     * @type {?}
     */
    const passiveEventOptions = normalizePassiveListenerOptions({ passive: true });
    /**
     * Events that signal that the pointer is down.
     * @type {?}
     */
    const pointerDownEvents = ['mousedown', 'touchstart'];
    /**
     * Events that signal that the pointer is up.
     * @type {?}
     */
    const pointerUpEvents = ['mouseup', 'mouseleave', 'touchend', 'touchcancel'];
    /**
     * Helper service that performs DOM manipulations. Not intended to be used outside this module.
     * The constructor takes a reference to the ripple directive's host element and a map of DOM
     * event handlers to be installed on the element that triggers ripple animations.
     * This will eventually become a custom renderer once Angular support exists.
     * \@docs-private
     */
    class RippleRenderer {
        /**
         * @param {?} _target
         * @param {?} _ngZone
         * @param {?} elementOrElementRef
         * @param {?} platform
         */
        constructor(_target, _ngZone, elementOrElementRef, platform) {
            this._target = _target;
            this._ngZone = _ngZone;
            /**
             * Whether the pointer is currently down or not.
             */
            this._isPointerDown = false;
            /**
             * Set of currently active ripple references.
             */
            this._activeRipples = new Set();
            /**
             * Whether pointer-up event listeners have been registered.
             */
            this._pointerUpEventsRegistered = false;
            // Only do anything if we're on the browser.
            if (platform.isBrowser) {
                this._containerElement = coerceElement(elementOrElementRef);
            }
        }
        /**
         * Fades in a ripple at the given coordinates.
         * @param {?} x Coordinate within the element, along the X axis at which to start the ripple.
         * @param {?} y Coordinate within the element, along the Y axis at which to start the ripple.
         * @param {?=} config Extra ripple options.
         * @return {?}
         */
        fadeInRipple(x, y, config = {}) {
            /** @type {?} */
            const containerRect = this._containerRect =
                this._containerRect || this._containerElement.getBoundingClientRect();
            /** @type {?} */
            const animationConfig = Object.assign(Object.assign({}, defaultRippleAnimationConfig), config.animation);
            if (config.centered) {
                x = containerRect.left + containerRect.width / 2;
                y = containerRect.top + containerRect.height / 2;
            }
            /** @type {?} */
            const radius = config.radius || distanceToFurthestCorner(x, y, containerRect);
            /** @type {?} */
            const offsetX = x - containerRect.left;
            /** @type {?} */
            const offsetY = y - containerRect.top;
            /** @type {?} */
            const duration = animationConfig.enterDuration;
            /** @type {?} */
            const ripple = document.createElement('div');
            ripple.classList.add('mat-ripple-element');
            ripple.style.left = `${offsetX - radius}px`;
            ripple.style.top = `${offsetY - radius}px`;
            ripple.style.height = `${radius * 2}px`;
            ripple.style.width = `${radius * 2}px`;
            // If a custom color has been specified, set it as inline style. If no color is
            // set, the default color will be applied through the ripple theme styles.
            if (config.color != null) {
                ripple.style.backgroundColor = config.color;
            }
            ripple.style.transitionDuration = `${duration}ms`;
            this._containerElement.appendChild(ripple);
            // By default the browser does not recalculate the styles of dynamically created
            // ripple elements. This is critical because then the `scale` would not animate properly.
            enforceStyleRecalculation(ripple);
            ripple.style.transform = 'scale(1)';
            // Exposed reference to the ripple that will be returned.
            /** @type {?} */
            const rippleRef = new RippleRef(this, ripple, config);
            rippleRef.state = 0 /* FADING_IN */;
            // Add the ripple reference to the list of all active ripples.
            this._activeRipples.add(rippleRef);
            if (!config.persistent) {
                this._mostRecentTransientRipple = rippleRef;
            }
            // Wait for the ripple element to be completely faded in.
            // Once it's faded in, the ripple can be hidden immediately if the mouse is released.
            this._runTimeoutOutsideZone((/**
             * @return {?}
             */
            () => {
                /** @type {?} */
                const isMostRecentTransientRipple = rippleRef === this._mostRecentTransientRipple;
                rippleRef.state = 1 /* VISIBLE */;
                // When the timer runs out while the user has kept their pointer down, we want to
                // keep only the persistent ripples and the latest transient ripple. We do this,
                // because we don't want stacked transient ripples to appear after their enter
                // animation has finished.
                if (!config.persistent && (!isMostRecentTransientRipple || !this._isPointerDown)) {
                    rippleRef.fadeOut();
                }
            }), duration);
            return rippleRef;
        }
        /**
         * Fades out a ripple reference.
         * @param {?} rippleRef
         * @return {?}
         */
        fadeOutRipple(rippleRef) {
            /** @type {?} */
            const wasActive = this._activeRipples.delete(rippleRef);
            if (rippleRef === this._mostRecentTransientRipple) {
                this._mostRecentTransientRipple = null;
            }
            // Clear out the cached bounding rect if we have no more ripples.
            if (!this._activeRipples.size) {
                this._containerRect = null;
            }
            // For ripples that are not active anymore, don't re-run the fade-out animation.
            if (!wasActive) {
                return;
            }
            /** @type {?} */
            const rippleEl = rippleRef.element;
            /** @type {?} */
            const animationConfig = Object.assign(Object.assign({}, defaultRippleAnimationConfig), rippleRef.config.animation);
            rippleEl.style.transitionDuration = `${animationConfig.exitDuration}ms`;
            rippleEl.style.opacity = '0';
            rippleRef.state = 2 /* FADING_OUT */;
            // Once the ripple faded out, the ripple can be safely removed from the DOM.
            this._runTimeoutOutsideZone((/**
             * @return {?}
             */
            () => {
                rippleRef.state = 3 /* HIDDEN */;
                (/** @type {?} */ (rippleEl.parentNode)).removeChild(rippleEl);
            }), animationConfig.exitDuration);
        }
        /**
         * Fades out all currently active ripples.
         * @return {?}
         */
        fadeOutAll() {
            this._activeRipples.forEach((/**
             * @param {?} ripple
             * @return {?}
             */
            ripple => ripple.fadeOut()));
        }
        /**
         * Sets up the trigger event listeners
         * @param {?} elementOrElementRef
         * @return {?}
         */
        setupTriggerEvents(elementOrElementRef) {
            /** @type {?} */
            const element = coerceElement(elementOrElementRef);
            if (!element || element === this._triggerElement) {
                return;
            }
            // Remove all previously registered event listeners from the trigger element.
            this._removeTriggerEvents();
            this._triggerElement = element;
            this._registerEvents(pointerDownEvents);
        }
        /**
         * Handles all registered events.
         * \@docs-private
         * @param {?} event
         * @return {?}
         */
        handleEvent(event) {
            if (event.type === 'mousedown') {
                this._onMousedown((/** @type {?} */ (event)));
            }
            else if (event.type === 'touchstart') {
                this._onTouchStart((/** @type {?} */ (event)));
            }
            else {
                this._onPointerUp();
            }
            // If pointer-up events haven't been registered yet, do so now.
            // We do this on-demand in order to reduce the total number of event listeners
            // registered by the ripples, which speeds up the rendering time for large UIs.
            if (!this._pointerUpEventsRegistered) {
                this._registerEvents(pointerUpEvents);
                this._pointerUpEventsRegistered = true;
            }
        }
        /**
         * Function being called whenever the trigger is being pressed using mouse.
         * @private
         * @param {?} event
         * @return {?}
         */
        _onMousedown(event) {
            // Screen readers will fire fake mouse events for space/enter. Skip launching a
            // ripple in this case for consistency with the non-screen-reader experience.
            /** @type {?} */
            const isFakeMousedown = isFakeMousedownFromScreenReader(event);
            /** @type {?} */
            const isSyntheticEvent = this._lastTouchStartEvent &&
                Date.now() < this._lastTouchStartEvent + ignoreMouseEventsTimeout;
            if (!this._target.rippleDisabled && !isFakeMousedown && !isSyntheticEvent) {
                this._isPointerDown = true;
                this.fadeInRipple(event.clientX, event.clientY, this._target.rippleConfig);
            }
        }
        /**
         * Function being called whenever the trigger is being pressed using touch.
         * @private
         * @param {?} event
         * @return {?}
         */
        _onTouchStart(event) {
            if (!this._target.rippleDisabled) {
                // Some browsers fire mouse events after a `touchstart` event. Those synthetic mouse
                // events will launch a second ripple if we don't ignore mouse events for a specific
                // time after a touchstart event.
                this._lastTouchStartEvent = Date.now();
                this._isPointerDown = true;
                // Use `changedTouches` so we skip any touches where the user put
                // their finger down, but used another finger to tap the element again.
                /** @type {?} */
                const touches = event.changedTouches;
                for (let i = 0; i < touches.length; i++) {
                    this.fadeInRipple(touches[i].clientX, touches[i].clientY, this._target.rippleConfig);
                }
            }
        }
        /**
         * Function being called whenever the trigger is being released.
         * @private
         * @return {?}
         */
        _onPointerUp() {
            if (!this._isPointerDown) {
                return;
            }
            this._isPointerDown = false;
            // Fade-out all ripples that are visible and not persistent.
            this._activeRipples.forEach((/**
             * @param {?} ripple
             * @return {?}
             */
            ripple => {
                // By default, only ripples that are completely visible will fade out on pointer release.
                // If the `terminateOnPointerUp` option is set, ripples that still fade in will also fade out.
                /** @type {?} */
                const isVisible = ripple.state === 1 /* VISIBLE */ ||
                    ripple.config.terminateOnPointerUp && ripple.state === 0 /* FADING_IN */;
                if (!ripple.config.persistent && isVisible) {
                    ripple.fadeOut();
                }
            }));
        }
        /**
         * Runs a timeout outside of the Angular zone to avoid triggering the change detection.
         * @private
         * @param {?} fn
         * @param {?=} delay
         * @return {?}
         */
        _runTimeoutOutsideZone(fn, delay = 0) {
            this._ngZone.runOutsideAngular((/**
             * @return {?}
             */
            () => setTimeout(fn, delay)));
        }
        /**
         * Registers event listeners for a given list of events.
         * @private
         * @param {?} eventTypes
         * @return {?}
         */
        _registerEvents(eventTypes) {
            this._ngZone.runOutsideAngular((/**
             * @return {?}
             */
            () => {
                eventTypes.forEach((/**
                 * @param {?} type
                 * @return {?}
                 */
                (type) => {
                    (/** @type {?} */ (this._triggerElement)).addEventListener(type, this, passiveEventOptions);
                }));
            }));
        }
        /**
         * Removes previously registered event listeners from the trigger element.
         * @return {?}
         */
        _removeTriggerEvents() {
            if (this._triggerElement) {
                pointerDownEvents.forEach((/**
                 * @param {?} type
                 * @return {?}
                 */
                (type) => {
                    (/** @type {?} */ (this._triggerElement)).removeEventListener(type, this, passiveEventOptions);
                }));
                if (this._pointerUpEventsRegistered) {
                    pointerUpEvents.forEach((/**
                     * @param {?} type
                     * @return {?}
                     */
                    (type) => {
                        (/** @type {?} */ (this._triggerElement)).removeEventListener(type, this, passiveEventOptions);
                    }));
                }
            }
        }
    }
    /**
     * Enforces a style recalculation of a DOM element by computing its styles.
     * @param {?} element
     * @return {?}
     */
    function enforceStyleRecalculation(element) {
        // Enforce a style recalculation by calling `getComputedStyle` and accessing any property.
        // Calling `getPropertyValue` is important to let optimizers know that this is not a noop.
        // See: https://gist.github.com/paulirish/5d52fb081b3570c81e3a
        window.getComputedStyle(element).getPropertyValue('opacity');
    }
    /**
     * Returns the distance from the point (x, y) to the furthest corner of a rectangle.
     * @param {?} x
     * @param {?} y
     * @param {?} rect
     * @return {?}
     */
    function distanceToFurthestCorner(x, y, rect) {
        /** @type {?} */
        const distX = Math.max(Math.abs(x - rect.left), Math.abs(x - rect.right));
        /** @type {?} */
        const distY = Math.max(Math.abs(y - rect.top), Math.abs(y - rect.bottom));
        return Math.sqrt(distX * distX + distY * distY);
    }
    /**
     * Injection token that can be used to specify the global ripple options.
     * @type {?}
     */
    const MAT_RIPPLE_GLOBAL_OPTIONS = new InjectionToken('mat-ripple-global-options');
    class MatRipple {
        /**
         * @param {?} _elementRef
         * @param {?} ngZone
         * @param {?} platform
         * @param {?=} globalOptions
         * @param {?=} _animationMode
         */
        constructor(_elementRef, ngZone, platform, globalOptions, _animationMode) {
            this._elementRef = _elementRef;
            this._animationMode = _animationMode;
            /**
             * If set, the radius in pixels of foreground ripples when fully expanded. If unset, the radius
             * will be the distance from the center of the ripple to the furthest corner of the host element's
             * bounding rectangle.
             */
            this.radius = 0;
            this._disabled = false;
            /**
             * Whether ripple directive is initialized and the input bindings are set.
             */
            this._isInitialized = false;
            this._globalOptions = globalOptions || {};
            this._rippleRenderer = new RippleRenderer(this, ngZone, _elementRef, platform);
        }
        /**
         * Whether click events will not trigger the ripple. Ripples can be still launched manually
         * by using the `launch()` method.
         * @return {?}
         */
        get disabled() { return this._disabled; }
        /**
         * @param {?} value
         * @return {?}
         */
        set disabled(value) {
            this._disabled = value;
            this._setupTriggerEventsIfEnabled();
        }
        /**
         * The element that triggers the ripple when click events are received.
         * Defaults to the directive's host element.
         * @return {?}
         */
        get trigger() { return this._trigger || this._elementRef.nativeElement; }
        /**
         * @param {?} trigger
         * @return {?}
         */
        set trigger(trigger) {
            this._trigger = trigger;
            this._setupTriggerEventsIfEnabled();
        }
        /**
         * @return {?}
         */
        ngOnInit() {
            this._isInitialized = true;
            this._setupTriggerEventsIfEnabled();
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this._rippleRenderer._removeTriggerEvents();
        }
        /**
         * Fades out all currently showing ripple elements.
         * @return {?}
         */
        fadeOutAll() {
            this._rippleRenderer.fadeOutAll();
        }
        /**
         * Ripple configuration from the directive's input values.
         * \@docs-private Implemented as part of RippleTarget
         * @return {?}
         */
        get rippleConfig() {
            return {
                centered: this.centered,
                radius: this.radius,
                color: this.color,
                animation: Object.assign(Object.assign(Object.assign({}, this._globalOptions.animation), (this._animationMode === 'NoopAnimations' ? { enterDuration: 0, exitDuration: 0 } : {})), this.animation),
                terminateOnPointerUp: this._globalOptions.terminateOnPointerUp,
            };
        }
        /**
         * Whether ripples on pointer-down are disabled or not.
         * \@docs-private Implemented as part of RippleTarget
         * @return {?}
         */
        get rippleDisabled() {
            return this.disabled || !!this._globalOptions.disabled;
        }
        /**
         * Sets up the trigger event listeners if ripples are enabled.
         * @private
         * @return {?}
         */
        _setupTriggerEventsIfEnabled() {
            if (!this.disabled && this._isInitialized) {
                this._rippleRenderer.setupTriggerEvents(this.trigger);
            }
        }
        /**
         * Launches a manual ripple at the specified coordinated or just by the ripple config.
         * @param {?} configOrX
         * @param {?=} y
         * @param {?=} config
         * @return {?}
         */
        launch(configOrX, y = 0, config) {
            if (typeof configOrX === 'number') {
                return this._rippleRenderer.fadeInRipple(configOrX, y, Object.assign(Object.assign({}, this.rippleConfig), config));
            }
            else {
                return this._rippleRenderer.fadeInRipple(0, 0, Object.assign(Object.assign({}, this.rippleConfig), configOrX));
            }
        }
    }
    MatRipple.ɵfac = function MatRipple_Factory(t) { return new (t || MatRipple)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(NgZone), ɵɵdirectiveInject(Platform), ɵɵdirectiveInject(MAT_RIPPLE_GLOBAL_OPTIONS, 8), ɵɵdirectiveInject(ANIMATION_MODULE_TYPE, 8)); };
    MatRipple.ɵdir = ɵɵdefineDirective({ type: MatRipple, selectors: [["", "mat-ripple", ""], ["", "matRipple", ""]], hostAttrs: [1, "mat-ripple"], hostVars: 2, hostBindings: function MatRipple_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵclassProp("mat-ripple-unbounded", ctx.unbounded);
        } }, inputs: { radius: ["matRippleRadius", "radius"], disabled: ["matRippleDisabled", "disabled"], trigger: ["matRippleTrigger", "trigger"], color: ["matRippleColor", "color"], unbounded: ["matRippleUnbounded", "unbounded"], centered: ["matRippleCentered", "centered"], animation: ["matRippleAnimation", "animation"] }, exportAs: ["matRipple"] });
    /** @nocollapse */
    MatRipple.ctorParameters = () => [
        { type: ElementRef },
        { type: NgZone },
        { type: Platform },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [MAT_RIPPLE_GLOBAL_OPTIONS,] }] },
        { type: String, decorators: [{ type: Optional }, { type: Inject, args: [ANIMATION_MODULE_TYPE,] }] }
    ];
    MatRipple.propDecorators = {
        color: [{ type: Input, args: ['matRippleColor',] }],
        unbounded: [{ type: Input, args: ['matRippleUnbounded',] }],
        centered: [{ type: Input, args: ['matRippleCentered',] }],
        radius: [{ type: Input, args: ['matRippleRadius',] }],
        animation: [{ type: Input, args: ['matRippleAnimation',] }],
        disabled: [{ type: Input, args: ['matRippleDisabled',] }],
        trigger: [{ type: Input, args: ['matRippleTrigger',] }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/core/ripple/index.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class MatRippleModule {
    }
    MatRippleModule.ɵmod = ɵɵdefineNgModule({ type: MatRippleModule });
    MatRippleModule.ɵinj = ɵɵdefineInjector({ factory: function MatRippleModule_Factory(t) { return new (t || MatRippleModule)(); }, imports: [[MatCommonModule, PlatformModule],
            MatCommonModule] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(MatRippleModule, { declarations: function () { return [MatRipple]; }, imports: function () { return [MatCommonModule,
            PlatformModule]; }, exports: function () { return [MatRipple,
            MatCommonModule]; } }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/core/selection/pseudo-checkbox/pseudo-checkbox.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Component that shows a simplified checkbox without including any kind of "real" checkbox.
     * Meant to be used when the checkbox is purely decorative and a large number of them will be
     * included, such as for the options in a multi-select. Uses no SVGs or complex animations.
     * Note that theming is meant to be handled by the parent element, e.g.
     * `mat-primary .mat-pseudo-checkbox`.
     *
     * Note that this component will be completely invisible to screen-reader users. This is *not*
     * interchangeable with `<mat-checkbox>` and should *not* be used if the user would directly
     * interact with the checkbox. The pseudo-checkbox should only be used as an implementation detail
     * of more complex components that appropriately handle selected / checked state.
     * \@docs-private
     */
    class MatPseudoCheckbox {
        /**
         * @param {?=} _animationMode
         */
        constructor(_animationMode) {
            this._animationMode = _animationMode;
            /**
             * Display state of the checkbox.
             */
            this.state = 'unchecked';
            /**
             * Whether the checkbox is disabled.
             */
            this.disabled = false;
        }
    }
    MatPseudoCheckbox.ɵfac = function MatPseudoCheckbox_Factory(t) { return new (t || MatPseudoCheckbox)(ɵɵdirectiveInject(ANIMATION_MODULE_TYPE, 8)); };
    MatPseudoCheckbox.ɵcmp = ɵɵdefineComponent({ type: MatPseudoCheckbox, selectors: [["mat-pseudo-checkbox"]], hostAttrs: [1, "mat-pseudo-checkbox"], hostVars: 8, hostBindings: function MatPseudoCheckbox_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵclassProp("mat-pseudo-checkbox-indeterminate", ctx.state === "indeterminate")("mat-pseudo-checkbox-checked", ctx.state === "checked")("mat-pseudo-checkbox-disabled", ctx.disabled)("_mat-animation-noopable", ctx._animationMode === "NoopAnimations");
        } }, inputs: { state: "state", disabled: "disabled" }, decls: 0, vars: 0, template: function MatPseudoCheckbox_Template(rf, ctx) { }, styles: [".mat-pseudo-checkbox{width:16px;height:16px;border:2px solid;border-radius:2px;cursor:pointer;display:inline-block;vertical-align:middle;box-sizing:border-box;position:relative;flex-shrink:0;transition:border-color 90ms cubic-bezier(0, 0, 0.2, 0.1),background-color 90ms cubic-bezier(0, 0, 0.2, 0.1)}.mat-pseudo-checkbox::after{position:absolute;opacity:0;content:\"\";border-bottom:2px solid currentColor;transition:opacity 90ms cubic-bezier(0, 0, 0.2, 0.1)}.mat-pseudo-checkbox.mat-pseudo-checkbox-checked,.mat-pseudo-checkbox.mat-pseudo-checkbox-indeterminate{border-color:transparent}._mat-animation-noopable.mat-pseudo-checkbox{transition:none;animation:none}._mat-animation-noopable.mat-pseudo-checkbox::after{transition:none}.mat-pseudo-checkbox-disabled{cursor:default}.mat-pseudo-checkbox-indeterminate::after{top:5px;left:1px;width:10px;opacity:1;border-radius:2px}.mat-pseudo-checkbox-checked::after{top:2.4px;left:1px;width:8px;height:3px;border-left:2px solid currentColor;transform:rotate(-45deg);opacity:1;box-sizing:content-box}\n"], encapsulation: 2, changeDetection: 0 });
    /** @nocollapse */
    MatPseudoCheckbox.ctorParameters = () => [
        { type: String, decorators: [{ type: Optional }, { type: Inject, args: [ANIMATION_MODULE_TYPE,] }] }
    ];
    MatPseudoCheckbox.propDecorators = {
        state: [{ type: Input }],
        disabled: [{ type: Input }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/core/selection/index.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class MatPseudoCheckboxModule {
    }
    MatPseudoCheckboxModule.ɵmod = ɵɵdefineNgModule({ type: MatPseudoCheckboxModule });
    MatPseudoCheckboxModule.ɵinj = ɵɵdefineInjector({ factory: function MatPseudoCheckboxModule_Factory(t) { return new (t || MatPseudoCheckboxModule)(); } });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(MatPseudoCheckboxModule, { declarations: [MatPseudoCheckbox], exports: [MatPseudoCheckbox] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/core/option/optgroup.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    // Boilerplate for applying mixins to MatOptgroup.
    /**
     * \@docs-private
     */
    class MatOptgroupBase {
    }
    /** @type {?} */
    const _MatOptgroupMixinBase = mixinDisabled(MatOptgroupBase);
    // Counter for unique group ids.
    /** @type {?} */
    let _uniqueOptgroupIdCounter = 0;
    /**
     * Component that is used to group instances of `mat-option`.
     */
    class MatOptgroup extends _MatOptgroupMixinBase {
        constructor() {
            super(...arguments);
            /**
             * Unique id for the underlying label.
             */
            this._labelId = `mat-optgroup-label-${_uniqueOptgroupIdCounter++}`;
        }
    }
    MatOptgroup.ɵfac = function MatOptgroup_Factory(t) { return ɵMatOptgroup_BaseFactory(t || MatOptgroup); };
    MatOptgroup.ɵcmp = ɵɵdefineComponent({ type: MatOptgroup, selectors: [["mat-optgroup"]], hostAttrs: ["role", "group", 1, "mat-optgroup"], hostVars: 4, hostBindings: function MatOptgroup_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵattribute("aria-disabled", ctx.disabled.toString())("aria-labelledby", ctx._labelId);
            ɵɵclassProp("mat-optgroup-disabled", ctx.disabled);
        } }, inputs: { disabled: "disabled", label: "label" }, exportAs: ["matOptgroup"], features: [ɵɵInheritDefinitionFeature], ngContentSelectors: _c1, decls: 4, vars: 2, consts: [[1, "mat-optgroup-label", 3, "id"]], template: function MatOptgroup_Template(rf, ctx) { if (rf & 1) {
            ɵɵprojectionDef(_c0);
            ɵɵelementStart(0, "label", 0);
            ɵɵtext(1);
            ɵɵprojection(2);
            ɵɵelementEnd();
            ɵɵprojection(3, 1);
        } if (rf & 2) {
            ɵɵproperty("id", ctx._labelId);
            ɵɵadvance(1);
            ɵɵtextInterpolate1("", ctx.label, " ");
        } }, styles: [".mat-optgroup-label{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;line-height:48px;height:48px;padding:0 16px;text-align:left;text-decoration:none;max-width:100%;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;cursor:default}.mat-optgroup-label[disabled]{cursor:default}[dir=rtl] .mat-optgroup-label{text-align:right}.mat-optgroup-label .mat-icon{margin-right:16px;vertical-align:middle}.mat-optgroup-label .mat-icon svg{vertical-align:top}[dir=rtl] .mat-optgroup-label .mat-icon{margin-left:16px;margin-right:0}\n"], encapsulation: 2, changeDetection: 0 });
    MatOptgroup.propDecorators = {
        label: [{ type: Input }]
    };
    const ɵMatOptgroup_BaseFactory = ɵɵgetInheritedFactory(MatOptgroup);

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/core/option/option.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Option IDs need to be unique across components, so this counter exists outside of
     * the component definition.
     * @type {?}
     */
    let _uniqueIdCounter = 0;
    /**
     * Event object emitted by MatOption when selected or deselected.
     */
    class MatOptionSelectionChange {
        /**
         * @param {?} source
         * @param {?=} isUserInput
         */
        constructor(source, isUserInput = false) {
            this.source = source;
            this.isUserInput = isUserInput;
        }
    }
    /**
     * Injection token used to provide the parent component to options.
     * @type {?}
     */
    const MAT_OPTION_PARENT_COMPONENT = new InjectionToken('MAT_OPTION_PARENT_COMPONENT');
    /**
     * Single option inside of a `<mat-select>` element.
     */
    class MatOption {
        /**
         * @param {?} _element
         * @param {?} _changeDetectorRef
         * @param {?} _parent
         * @param {?} group
         */
        constructor(_element, _changeDetectorRef, _parent, group) {
            this._element = _element;
            this._changeDetectorRef = _changeDetectorRef;
            this._parent = _parent;
            this.group = group;
            this._selected = false;
            this._active = false;
            this._disabled = false;
            this._mostRecentViewValue = '';
            /**
             * The unique ID of the option.
             */
            this.id = `mat-option-${_uniqueIdCounter++}`;
            /**
             * Event emitted when the option is selected or deselected.
             */
            // tslint:disable-next-line:no-output-on-prefix
            this.onSelectionChange = new EventEmitter();
            /**
             * Emits when the state of the option changes and any parents have to be notified.
             */
            this._stateChanges = new Subject();
        }
        /**
         * Whether the wrapping component is in multiple selection mode.
         * @return {?}
         */
        get multiple() { return this._parent && this._parent.multiple; }
        /**
         * Whether or not the option is currently selected.
         * @return {?}
         */
        get selected() { return this._selected; }
        /**
         * Whether the option is disabled.
         * @return {?}
         */
        get disabled() { return (this.group && this.group.disabled) || this._disabled; }
        /**
         * @param {?} value
         * @return {?}
         */
        set disabled(value) { this._disabled = coerceBooleanProperty(value); }
        /**
         * Whether ripples for the option are disabled.
         * @return {?}
         */
        get disableRipple() { return this._parent && this._parent.disableRipple; }
        /**
         * Whether or not the option is currently active and ready to be selected.
         * An active option displays styles as if it is focused, but the
         * focus is actually retained somewhere else. This comes in handy
         * for components like autocomplete where focus must remain on the input.
         * @return {?}
         */
        get active() {
            return this._active;
        }
        /**
         * The displayed value of the option. It is necessary to show the selected option in the
         * select's trigger.
         * @return {?}
         */
        get viewValue() {
            // TODO(kara): Add input property alternative for node envs.
            return (this._getHostElement().textContent || '').trim();
        }
        /**
         * Selects the option.
         * @return {?}
         */
        select() {
            if (!this._selected) {
                this._selected = true;
                this._changeDetectorRef.markForCheck();
                this._emitSelectionChangeEvent();
            }
        }
        /**
         * Deselects the option.
         * @return {?}
         */
        deselect() {
            if (this._selected) {
                this._selected = false;
                this._changeDetectorRef.markForCheck();
                this._emitSelectionChangeEvent();
            }
        }
        /**
         * Sets focus onto this option.
         * @param {?=} _origin
         * @param {?=} options
         * @return {?}
         */
        focus(_origin, options) {
            // Note that we aren't using `_origin`, but we need to keep it because some internal consumers
            // use `MatOption` in a `FocusKeyManager` and we need it to match `FocusableOption`.
            /** @type {?} */
            const element = this._getHostElement();
            if (typeof element.focus === 'function') {
                element.focus(options);
            }
        }
        /**
         * This method sets display styles on the option to make it appear
         * active. This is used by the ActiveDescendantKeyManager so key
         * events will display the proper options as active on arrow key events.
         * @return {?}
         */
        setActiveStyles() {
            if (!this._active) {
                this._active = true;
                this._changeDetectorRef.markForCheck();
            }
        }
        /**
         * This method removes display styles on the option that made it appear
         * active. This is used by the ActiveDescendantKeyManager so key
         * events will display the proper options as active on arrow key events.
         * @return {?}
         */
        setInactiveStyles() {
            if (this._active) {
                this._active = false;
                this._changeDetectorRef.markForCheck();
            }
        }
        /**
         * Gets the label to be used when determining whether the option should be focused.
         * @return {?}
         */
        getLabel() {
            return this.viewValue;
        }
        /**
         * Ensures the option is selected when activated from the keyboard.
         * @param {?} event
         * @return {?}
         */
        _handleKeydown(event) {
            if ((event.keyCode === ENTER || event.keyCode === SPACE) && !hasModifierKey(event)) {
                this._selectViaInteraction();
                // Prevent the page from scrolling down and form submits.
                event.preventDefault();
            }
        }
        /**
         * `Selects the option while indicating the selection came from the user. Used to
         * determine if the select's view -> model callback should be invoked.`
         * @return {?}
         */
        _selectViaInteraction() {
            if (!this.disabled) {
                this._selected = this.multiple ? !this._selected : true;
                this._changeDetectorRef.markForCheck();
                this._emitSelectionChangeEvent(true);
            }
        }
        /**
         * Gets the `aria-selected` value for the option. We explicitly omit the `aria-selected`
         * attribute from single-selection, unselected options. Including the `aria-selected="false"`
         * attributes adds a significant amount of noise to screen-reader users without providing useful
         * information.
         * @return {?}
         */
        _getAriaSelected() {
            return this.selected || (this.multiple ? false : null);
        }
        /**
         * Returns the correct tabindex for the option depending on disabled state.
         * @return {?}
         */
        _getTabIndex() {
            return this.disabled ? '-1' : '0';
        }
        /**
         * Gets the host DOM element.
         * @return {?}
         */
        _getHostElement() {
            return this._element.nativeElement;
        }
        /**
         * @return {?}
         */
        ngAfterViewChecked() {
            // Since parent components could be using the option's label to display the selected values
            // (e.g. `mat-select`) and they don't have a way of knowing if the option's label has changed
            // we have to check for changes in the DOM ourselves and dispatch an event. These checks are
            // relatively cheap, however we still limit them only to selected options in order to avoid
            // hitting the DOM too often.
            if (this._selected) {
                /** @type {?} */
                const viewValue = this.viewValue;
                if (viewValue !== this._mostRecentViewValue) {
                    this._mostRecentViewValue = viewValue;
                    this._stateChanges.next();
                }
            }
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this._stateChanges.complete();
        }
        /**
         * Emits the selection change event.
         * @private
         * @param {?=} isUserInput
         * @return {?}
         */
        _emitSelectionChangeEvent(isUserInput = false) {
            this.onSelectionChange.emit(new MatOptionSelectionChange(this, isUserInput));
        }
    }
    MatOption.ɵfac = function MatOption_Factory(t) { return new (t || MatOption)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(ChangeDetectorRef), ɵɵdirectiveInject(MAT_OPTION_PARENT_COMPONENT, 8), ɵɵdirectiveInject(MatOptgroup, 8)); };
    MatOption.ɵcmp = ɵɵdefineComponent({ type: MatOption, selectors: [["mat-option"]], hostAttrs: ["role", "option", 1, "mat-option", "mat-focus-indicator"], hostVars: 12, hostBindings: function MatOption_HostBindings(rf, ctx) { if (rf & 1) {
            ɵɵlistener("click", function MatOption_click_HostBindingHandler() { return ctx._selectViaInteraction(); })("keydown", function MatOption_keydown_HostBindingHandler($event) { return ctx._handleKeydown($event); });
        } if (rf & 2) {
            ɵɵhostProperty("id", ctx.id);
            ɵɵattribute("tabindex", ctx._getTabIndex())("aria-selected", ctx._getAriaSelected())("aria-disabled", ctx.disabled.toString());
            ɵɵclassProp("mat-selected", ctx.selected)("mat-option-multiple", ctx.multiple)("mat-active", ctx.active)("mat-option-disabled", ctx.disabled);
        } }, inputs: { id: "id", disabled: "disabled", value: "value" }, outputs: { onSelectionChange: "onSelectionChange" }, exportAs: ["matOption"], ngContentSelectors: _c2, decls: 4, vars: 3, consts: [["class", "mat-option-pseudo-checkbox", 3, "state", "disabled", 4, "ngIf"], [1, "mat-option-text"], ["mat-ripple", "", 1, "mat-option-ripple", 3, "matRippleTrigger", "matRippleDisabled"], [1, "mat-option-pseudo-checkbox", 3, "state", "disabled"]], template: function MatOption_Template(rf, ctx) { if (rf & 1) {
            ɵɵprojectionDef();
            ɵɵtemplate(0, MatOption_mat_pseudo_checkbox_0_Template, 1, 2, "mat-pseudo-checkbox", 0);
            ɵɵelementStart(1, "span", 1);
            ɵɵprojection(2);
            ɵɵelementEnd();
            ɵɵelement(3, "div", 2);
        } if (rf & 2) {
            ɵɵproperty("ngIf", ctx.multiple);
            ɵɵadvance(3);
            ɵɵproperty("matRippleTrigger", ctx._getHostElement())("matRippleDisabled", ctx.disabled || ctx.disableRipple);
        } }, directives: [NgIf, MatRipple,
            MatPseudoCheckbox], styles: [".mat-option{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;line-height:48px;height:48px;padding:0 16px;text-align:left;text-decoration:none;max-width:100%;position:relative;cursor:pointer;outline:none;display:flex;flex-direction:row;max-width:100%;box-sizing:border-box;align-items:center;-webkit-tap-highlight-color:transparent}.mat-option[disabled]{cursor:default}[dir=rtl] .mat-option{text-align:right}.mat-option .mat-icon{margin-right:16px;vertical-align:middle}.mat-option .mat-icon svg{vertical-align:top}[dir=rtl] .mat-option .mat-icon{margin-left:16px;margin-right:0}.mat-option[aria-disabled=true]{-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;cursor:default}.mat-optgroup .mat-option:not(.mat-option-multiple){padding-left:32px}[dir=rtl] .mat-optgroup .mat-option:not(.mat-option-multiple){padding-left:16px;padding-right:32px}.cdk-high-contrast-active .mat-option{margin:0 1px}.cdk-high-contrast-active .mat-option.mat-active{border:solid 1px currentColor;margin:0}.mat-option-text{display:inline-block;flex-grow:1;overflow:hidden;text-overflow:ellipsis}.mat-option .mat-option-ripple{top:0;left:0;right:0;bottom:0;position:absolute;pointer-events:none}.cdk-high-contrast-active .mat-option .mat-option-ripple{opacity:.5}.mat-option-pseudo-checkbox{margin-right:8px}[dir=rtl] .mat-option-pseudo-checkbox{margin-left:8px;margin-right:0}\n"], encapsulation: 2, changeDetection: 0 });
    /** @nocollapse */
    MatOption.ctorParameters = () => [
        { type: ElementRef },
        { type: ChangeDetectorRef },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [MAT_OPTION_PARENT_COMPONENT,] }] },
        { type: MatOptgroup, decorators: [{ type: Optional }] }
    ];
    MatOption.propDecorators = {
        value: [{ type: Input }],
        id: [{ type: Input }],
        disabled: [{ type: Input }],
        onSelectionChange: [{ type: Output }]
    };
    /**
     * Counts the amount of option group labels that precede the specified option.
     * \@docs-private
     * @param {?} optionIndex Index of the option at which to start counting.
     * @param {?} options Flat list of all of the options.
     * @param {?} optionGroups Flat list of all of the option groups.
     * @return {?}
     */
    function _countGroupLabelsBeforeOption(optionIndex, options, optionGroups) {
        if (optionGroups.length) {
            /** @type {?} */
            let optionsArray = options.toArray();
            /** @type {?} */
            let groups = optionGroups.toArray();
            /** @type {?} */
            let groupCounter = 0;
            for (let i = 0; i < optionIndex + 1; i++) {
                if (optionsArray[i].group && optionsArray[i].group === groups[groupCounter]) {
                    groupCounter++;
                }
            }
            return groupCounter;
        }
        return 0;
    }
    /**
     * Determines the position to which to scroll a panel in order for an option to be into view.
     * \@docs-private
     * @param {?} optionIndex Index of the option to be scrolled into the view.
     * @param {?} optionHeight Height of the options.
     * @param {?} currentScrollPosition Current scroll position of the panel.
     * @param {?} panelHeight Height of the panel.
     * @return {?}
     */
    function _getOptionScrollPosition(optionIndex, optionHeight, currentScrollPosition, panelHeight) {
        /** @type {?} */
        const optionOffset = optionIndex * optionHeight;
        if (optionOffset < currentScrollPosition) {
            return optionOffset;
        }
        if (optionOffset + optionHeight > currentScrollPosition + panelHeight) {
            return Math.max(0, optionOffset - panelHeight + optionHeight);
        }
        return currentScrollPosition;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/core/option/index.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class MatOptionModule {
    }
    MatOptionModule.ɵmod = ɵɵdefineNgModule({ type: MatOptionModule });
    MatOptionModule.ɵinj = ɵɵdefineInjector({ factory: function MatOptionModule_Factory(t) { return new (t || MatOptionModule)(); }, imports: [[MatRippleModule, CommonModule, MatPseudoCheckboxModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(MatOptionModule, { declarations: function () { return [MatOption,
            MatOptgroup]; }, imports: function () { return [MatRippleModule,
            CommonModule,
            MatPseudoCheckboxModule]; }, exports: function () { return [MatOption,
            MatOptgroup]; } }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/core/label/label-options.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * InjectionToken that can be used to specify the global label options.
     * @deprecated Use `MAT_FORM_FIELD_DEFAULT_OPTIONS` injection token from
     *     `\@angular/material/form-field` instead.
     * \@breaking-change 11.0.0
     * @type {?}
     */
    const MAT_LABEL_GLOBAL_OPTIONS = new InjectionToken('mat-label-global-options');

    const _c0$1 = ["*", [["mat-toolbar-row"]]];
    const _c1$1 = ["*", "mat-toolbar-row"];
    class MatToolbarBase {
        /**
         * @param {?} _elementRef
         */
        constructor(_elementRef) {
            this._elementRef = _elementRef;
        }
    }
    /** @type {?} */
    const _MatToolbarMixinBase = mixinColor(MatToolbarBase);
    class MatToolbarRow {
    }
    MatToolbarRow.ɵfac = function MatToolbarRow_Factory(t) { return new (t || MatToolbarRow)(); };
    MatToolbarRow.ɵdir = ɵɵdefineDirective({ type: MatToolbarRow, selectors: [["mat-toolbar-row"]], hostAttrs: [1, "mat-toolbar-row"], exportAs: ["matToolbarRow"] });
    class MatToolbar extends _MatToolbarMixinBase {
        /**
         * @param {?} elementRef
         * @param {?} _platform
         * @param {?=} document
         */
        constructor(elementRef, _platform, document) {
            super(elementRef);
            this._platform = _platform;
            // TODO: make the document a required param when doing breaking changes.
            this._document = document;
        }
        /**
         * @return {?}
         */
        ngAfterViewInit() {
            if (!isDevMode() || !this._platform.isBrowser) {
                return;
            }
            this._checkToolbarMixedModes();
            this._toolbarRows.changes.subscribe((/**
             * @return {?}
             */
            () => this._checkToolbarMixedModes()));
        }
        /**
         * Throws an exception when developers are attempting to combine the different toolbar row modes.
         * @private
         * @return {?}
         */
        _checkToolbarMixedModes() {
            if (!this._toolbarRows.length) {
                return;
            }
            // Check if there are any other DOM nodes that can display content but aren't inside of
            // a <mat-toolbar-row> element.
            /** @type {?} */
            const isCombinedUsage = Array.from(this._elementRef.nativeElement.childNodes)
                .filter((/**
             * @param {?} node
             * @return {?}
             */
            node => !(node.classList && node.classList.contains('mat-toolbar-row'))))
                .filter((/**
             * @param {?} node
             * @return {?}
             */
            node => node.nodeType !== (this._document ? this._document.COMMENT_NODE : 8)))
                .some((/**
             * @param {?} node
             * @return {?}
             */
            node => !!(node.textContent && node.textContent.trim())));
            if (isCombinedUsage) {
                throwToolbarMixedModesError();
            }
        }
    }
    MatToolbar.ɵfac = function MatToolbar_Factory(t) { return new (t || MatToolbar)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(Platform), ɵɵdirectiveInject(DOCUMENT$1)); };
    MatToolbar.ɵcmp = ɵɵdefineComponent({ type: MatToolbar, selectors: [["mat-toolbar"]], contentQueries: function MatToolbar_ContentQueries(rf, ctx, dirIndex) { if (rf & 1) {
            ɵɵcontentQuery(dirIndex, MatToolbarRow, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._toolbarRows = _t);
        } }, hostAttrs: [1, "mat-toolbar"], hostVars: 4, hostBindings: function MatToolbar_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵclassProp("mat-toolbar-multiple-rows", ctx._toolbarRows.length > 0)("mat-toolbar-single-row", ctx._toolbarRows.length === 0);
        } }, inputs: { color: "color" }, exportAs: ["matToolbar"], features: [ɵɵInheritDefinitionFeature], ngContentSelectors: _c1$1, decls: 2, vars: 0, template: function MatToolbar_Template(rf, ctx) { if (rf & 1) {
            ɵɵprojectionDef(_c0$1);
            ɵɵprojection(0);
            ɵɵprojection(1, 1);
        } }, styles: [".cdk-high-contrast-active .mat-toolbar{outline:solid 1px}.mat-toolbar-row,.mat-toolbar-single-row{display:flex;box-sizing:border-box;padding:0 16px;width:100%;flex-direction:row;align-items:center;white-space:nowrap}.mat-toolbar-multiple-rows{display:flex;box-sizing:border-box;flex-direction:column;width:100%}.mat-toolbar-multiple-rows{min-height:64px}.mat-toolbar-row,.mat-toolbar-single-row{height:64px}@media(max-width: 599px){.mat-toolbar-multiple-rows{min-height:56px}.mat-toolbar-row,.mat-toolbar-single-row{height:56px}}\n"], encapsulation: 2, changeDetection: 0 });
    /** @nocollapse */
    MatToolbar.ctorParameters = () => [
        { type: ElementRef },
        { type: Platform },
        { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT$1,] }] }
    ];
    MatToolbar.propDecorators = {
        _toolbarRows: [{ type: ContentChildren, args: [MatToolbarRow, { descendants: true },] }]
    };
    /**
     * Throws an exception when attempting to combine the different toolbar row modes.
     * \@docs-private
     * @return {?}
     */
    function throwToolbarMixedModesError() {
        throw Error('MatToolbar: Attempting to combine different toolbar modes. ' +
            'Either specify multiple `<mat-toolbar-row>` elements explicitly or just place content ' +
            'inside of a `<mat-toolbar>` for a single row.');
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/toolbar/toolbar-module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class MatToolbarModule {
    }
    MatToolbarModule.ɵmod = ɵɵdefineNgModule({ type: MatToolbarModule });
    MatToolbarModule.ɵinj = ɵɵdefineInjector({ factory: function MatToolbarModule_Factory(t) { return new (t || MatToolbarModule)(); }, imports: [[MatCommonModule],
            MatCommonModule] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(MatToolbarModule, { declarations: function () { return [MatToolbar,
            MatToolbarRow]; }, imports: function () { return [MatCommonModule]; }, exports: function () { return [MatToolbar,
            MatToolbarRow,
            MatCommonModule]; } }); })();

    /* Copyright 2019 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /** Data-related types */
    var DataLoadState;
    (function (DataLoadState) {
        DataLoadState[DataLoadState["NOT_LOADED"] = 0] = "NOT_LOADED";
        DataLoadState[DataLoadState["LOADED"] = 1] = "LOADED";
        DataLoadState[DataLoadState["LOADING"] = 2] = "LOADING";
        DataLoadState[DataLoadState["FAILED"] = 3] = "FAILED";
    })(DataLoadState || (DataLoadState = {}));

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/core/store/core_types.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const CORE_FEATURE_KEY = 'core';
    /** @type {?} */
    const initialState = {
        activePlugin: null,
        plugins: {},
        pluginsListLoaded: {
            state: DataLoadState.NOT_LOADED,
            lastLoadedTimeInMs: null,
            failureCode: null,
        },
        reloadPeriodInMs: 30000,
        reloadEnabled: false,
        pageSize: 12,
        environment: {
            data_location: '',
            window_title: '',
        },
        polymerInteropRuns: [],
        polymerInteropRunSelection: new Set(),
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/core/store/core_reducers.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const reducer = createReducer(initialState, on(changePlugin, pluginUrlHashChanged, (/**
     * @param {?} state
     * @param {?} __1
     * @return {?}
     */
    (state, { plugin }) => {
        return Object.assign(Object.assign({}, state), { activePlugin: plugin });
    })), on(pluginsListingRequested, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return Object.assign(Object.assign({}, state), { pluginsListLoaded: Object.assign(Object.assign({}, state.pluginsListLoaded), { state: DataLoadState.LOADING }) });
    })), on(pluginsListingFailed, (/**
     * @param {?} state
     * @param {?} __1
     * @return {?}
     */
    (state, { failureCode }) => {
        return Object.assign(Object.assign({}, state), { pluginsListLoaded: Object.assign(Object.assign({}, state.pluginsListLoaded), { state: DataLoadState.FAILED, failureCode }) });
    })), on(pluginsListingLoaded, (/**
     * @param {?} state
     * @param {?} __1
     * @return {?}
     */
    (state, { plugins }) => {
        /** @type {?} */
        const firstEnabledPluginId = Object.keys(plugins).find((/**
         * @param {?} pluginId
         * @return {?}
         */
        (pluginId) => {
            return plugins[pluginId].enabled;
        })) || null;
        /** @type {?} */
        const activePlugin = state.activePlugin || firstEnabledPluginId;
        return Object.assign(Object.assign({}, state), { activePlugin,
            plugins, pluginsListLoaded: {
                state: DataLoadState.LOADED,
                lastLoadedTimeInMs: Date.now(),
                failureCode: null,
            } });
    })), on(environmentLoaded, (/**
     * @param {?} state
     * @param {?} __1
     * @return {?}
     */
    (state, { environment }) => {
        return Object.assign(Object.assign({}, state), { environment: environment });
    })), on(toggleReloadEnabled, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return Object.assign(Object.assign({}, state), { reloadEnabled: !state.reloadEnabled });
    })), on(changeReloadPeriod, (/**
     * @param {?} state
     * @param {?} __1
     * @return {?}
     */
    (state, { periodInMs }) => {
        /** @type {?} */
        const nextReloadPeriod = periodInMs > 0 ? periodInMs : state.reloadPeriodInMs;
        return Object.assign(Object.assign({}, state), { reloadPeriodInMs: nextReloadPeriod });
    })), on(changePageSize, (/**
     * @param {?} state
     * @param {?} __1
     * @return {?}
     */
    (state, { size }) => {
        /** @type {?} */
        const nextPageSize = size > 0 ? size : state.pageSize;
        return Object.assign(Object.assign({}, state), { pageSize: nextPageSize });
    })), on(fetchRunSucceeded, (/**
     * @param {?} state
     * @param {?} __1
     * @return {?}
     */
    (state, { runs }) => {
        // Do not modify the runSelection since the Polymer component is the
        // source of truth for the Polymer Interop.
        return Object.assign(Object.assign({}, state), { polymerInteropRuns: runs });
    })), on(polymerInteropRunSelectionChanged, (/**
     * @param {?} state
     * @param {?} __1
     * @return {?}
     */
    (state, { nextSelection }) => {
        return Object.assign(Object.assign({}, state), { polymerInteropRunSelection: new Set(nextSelection) });
    })));
    /**
     * @param {?} state
     * @param {?} action
     * @return {?}
     */
    function reducers(state, action) {
        return reducer(state, action);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/core/store/core_selectors.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const selectCoreState = createFeatureSelector(CORE_FEATURE_KEY);
    /** @type {?} */
    const getPluginsListLoaded = createSelector(selectCoreState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => state.pluginsListLoaded));
    /** @type {?} */
    const getActivePlugin = createSelector(selectCoreState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return state.activePlugin;
    }));
    /** @type {?} */
    const getPlugins = createSelector(selectCoreState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return state.plugins;
    }));
    /** @type {?} */
    const getEnvironment = createSelector(selectCoreState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return state.environment;
    }));
    /** @type {?} */
    const getReloadEnabled = createSelector(selectCoreState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return state.reloadEnabled;
    }));
    /** @type {?} */
    const getReloadPeriodInMs = createSelector(selectCoreState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return state.reloadPeriodInMs;
    }));
    /** @type {?} */
    const getPageSize = createSelector(selectCoreState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return state.pageSize;
    }));

    function throwNullPortalError() {
        throw Error('Must provide a portal to attach');
    }
    /**
     * Throws an exception when attempting to attach a portal to a host that is already attached.
     * \@docs-private
     * @return {?}
     */
    function throwPortalAlreadyAttachedError() {
        throw Error('Host already has a portal attached');
    }
    /**
     * Throws an exception when attempting to attach a portal to an already-disposed host.
     * \@docs-private
     * @return {?}
     */
    function throwPortalOutletAlreadyDisposedError() {
        throw Error('This PortalOutlet has already been disposed');
    }
    /**
     * Throws an exception when attempting to attach an unknown portal type.
     * \@docs-private
     * @return {?}
     */
    function throwUnknownPortalTypeError() {
        throw Error('Attempting to attach an unknown Portal type. BasePortalOutlet accepts either ' +
            'a ComponentPortal or a TemplatePortal.');
    }
    /**
     * Throws an exception when attempting to attach a portal to a null host.
     * \@docs-private
     * @return {?}
     */
    function throwNullPortalOutletError() {
        throw Error('Attempting to attach a portal to a null PortalOutlet');
    }
    /**
     * Throws an exception when attempting to detach a portal that is not attached.
     * \@docs-private
     * @return {?}
     */
    function throwNoPortalAttachedError() {
        throw Error('Attempting to detach a portal that is not attached to a host');
    }
    /**
     * A `Portal` is something that you want to render somewhere else.
     * It can be attach to / detached from a `PortalOutlet`.
     * @abstract
     * @template T
     */
    class Portal {
        /**
         * Attach this portal to a host.
         * @param {?} host
         * @return {?}
         */
        attach(host) {
            if (host == null) {
                throwNullPortalOutletError();
            }
            if (host.hasAttached()) {
                throwPortalAlreadyAttachedError();
            }
            this._attachedHost = host;
            return (/** @type {?} */ (host.attach(this)));
        }
        /**
         * Detach this portal from its host
         * @return {?}
         */
        detach() {
            /** @type {?} */
            let host = this._attachedHost;
            if (host == null) {
                throwNoPortalAttachedError();
            }
            else {
                this._attachedHost = null;
                host.detach();
            }
        }
        /**
         * Whether this portal is attached to a host.
         * @return {?}
         */
        get isAttached() {
            return this._attachedHost != null;
        }
        /**
         * Sets the PortalOutlet reference without performing `attach()`. This is used directly by
         * the PortalOutlet when it is performing an `attach()` or `detach()`.
         * @param {?} host
         * @return {?}
         */
        setAttachedHost(host) {
            this._attachedHost = host;
        }
    }
    /**
     * A `ComponentPortal` is a portal that instantiates some Component upon attachment.
     * @template T
     */
    class ComponentPortal extends Portal {
        /**
         * @param {?} component
         * @param {?=} viewContainerRef
         * @param {?=} injector
         * @param {?=} componentFactoryResolver
         */
        constructor(component, viewContainerRef, injector, componentFactoryResolver) {
            super();
            this.component = component;
            this.viewContainerRef = viewContainerRef;
            this.injector = injector;
            this.componentFactoryResolver = componentFactoryResolver;
        }
    }
    /**
     * A `TemplatePortal` is a portal that represents some embedded template (TemplateRef).
     * @template C
     */
    class TemplatePortal extends Portal {
        /**
         * @param {?} template
         * @param {?} viewContainerRef
         * @param {?=} context
         */
        constructor(template, viewContainerRef, context) {
            super();
            this.templateRef = template;
            this.viewContainerRef = viewContainerRef;
            this.context = context;
        }
        /**
         * @return {?}
         */
        get origin() {
            return this.templateRef.elementRef;
        }
        /**
         * Attach the portal to the provided `PortalOutlet`.
         * When a context is provided it will override the `context` property of the `TemplatePortal`
         * instance.
         * @param {?} host
         * @param {?=} context
         * @return {?}
         */
        attach(host, context = this.context) {
            this.context = context;
            return super.attach(host);
        }
        /**
         * @return {?}
         */
        detach() {
            this.context = undefined;
            return super.detach();
        }
    }
    /**
     * A `DomPortal` is a portal whose DOM element will be taken from its current position
     * in the DOM and moved into a portal outlet, when it is attached. On detach, the content
     * will be restored to its original position.
     * @template T
     */
    class DomPortal extends Portal {
        /**
         * @param {?} element
         */
        constructor(element) {
            super();
            this.element = element instanceof ElementRef ? element.nativeElement : element;
        }
    }
    /**
     * Partial implementation of PortalOutlet that handles attaching
     * ComponentPortal and TemplatePortal.
     * @abstract
     */
    class BasePortalOutlet {
        constructor() {
            /**
             * Whether this host has already been permanently disposed.
             */
            this._isDisposed = false;
            // @breaking-change 10.0.0 `attachDomPortal` to become a required abstract method.
            this.attachDomPortal = null;
        }
        /**
         * Whether this host has an attached portal.
         * @return {?}
         */
        hasAttached() {
            return !!this._attachedPortal;
        }
        /**
         * Attaches a portal.
         * @param {?} portal
         * @return {?}
         */
        attach(portal) {
            if (!portal) {
                throwNullPortalError();
            }
            if (this.hasAttached()) {
                throwPortalAlreadyAttachedError();
            }
            if (this._isDisposed) {
                throwPortalOutletAlreadyDisposedError();
            }
            if (portal instanceof ComponentPortal) {
                this._attachedPortal = portal;
                return this.attachComponentPortal(portal);
            }
            else if (portal instanceof TemplatePortal) {
                this._attachedPortal = portal;
                return this.attachTemplatePortal(portal);
                // @breaking-change 10.0.0 remove null check for `this.attachDomPortal`.
            }
            else if (this.attachDomPortal && portal instanceof DomPortal) {
                this._attachedPortal = portal;
                return this.attachDomPortal(portal);
            }
            throwUnknownPortalTypeError();
        }
        /**
         * Detaches a previously attached portal.
         * @return {?}
         */
        detach() {
            if (this._attachedPortal) {
                this._attachedPortal.setAttachedHost(null);
                this._attachedPortal = null;
            }
            this._invokeDisposeFn();
        }
        /**
         * Permanently dispose of this portal host.
         * @return {?}
         */
        dispose() {
            if (this.hasAttached()) {
                this.detach();
            }
            this._invokeDisposeFn();
            this._isDisposed = true;
        }
        /**
         * \@docs-private
         * @param {?} fn
         * @return {?}
         */
        setDisposeFn(fn) {
            this._disposeFn = fn;
        }
        /**
         * @private
         * @return {?}
         */
        _invokeDisposeFn() {
            if (this._disposeFn) {
                this._disposeFn();
                this._disposeFn = null;
            }
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/portal/dom-portal-outlet.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * A PortalOutlet for attaching portals to an arbitrary DOM element outside of the Angular
     * application context.
     */
    class DomPortalOutlet extends BasePortalOutlet {
        /**
         * @param {?} outletElement
         * @param {?} _componentFactoryResolver
         * @param {?} _appRef
         * @param {?} _defaultInjector
         * @param {?=} _document
         */
        constructor(outletElement, _componentFactoryResolver, _appRef, _defaultInjector, 
        /**
         * @deprecated `_document` Parameter to be made required.
         * @breaking-change 10.0.0
         */
        _document) {
            super();
            this.outletElement = outletElement;
            this._componentFactoryResolver = _componentFactoryResolver;
            this._appRef = _appRef;
            this._defaultInjector = _defaultInjector;
            /**
             * Attaches a DOM portal by transferring its content into the outlet.
             * @param portal Portal to be attached.
             * @deprecated To be turned into a method.
             * \@breaking-change 10.0.0
             */
            this.attachDomPortal = (/**
             * @param {?} portal
             * @return {?}
             */
            (portal) => {
                // @breaking-change 10.0.0 Remove check and error once the
                // `_document` constructor parameter is required.
                if (!this._document) {
                    throw Error('Cannot attach DOM portal without _document constructor parameter');
                }
                /** @type {?} */
                const element = portal.element;
                if (!element.parentNode) {
                    throw Error('DOM portal content must be attached to a parent node.');
                }
                // Anchor used to save the element's previous position so
                // that we can restore it when the portal is detached.
                /** @type {?} */
                const anchorNode = this._document.createComment('dom-portal');
                element.parentNode.insertBefore(anchorNode, element);
                this.outletElement.appendChild(element);
                super.setDisposeFn((/**
                 * @return {?}
                 */
                () => {
                    // We can't use `replaceWith` here because IE doesn't support it.
                    if (anchorNode.parentNode) {
                        anchorNode.parentNode.replaceChild(element, anchorNode);
                    }
                }));
            });
            this._document = _document;
        }
        /**
         * Attach the given ComponentPortal to DOM element using the ComponentFactoryResolver.
         * @template T
         * @param {?} portal Portal to be attached
         * @return {?} Reference to the created component.
         */
        attachComponentPortal(portal) {
            /** @type {?} */
            const resolver = portal.componentFactoryResolver || this._componentFactoryResolver;
            /** @type {?} */
            const componentFactory = resolver.resolveComponentFactory(portal.component);
            /** @type {?} */
            let componentRef;
            // If the portal specifies a ViewContainerRef, we will use that as the attachment point
            // for the component (in terms of Angular's component tree, not rendering).
            // When the ViewContainerRef is missing, we use the factory to create the component directly
            // and then manually attach the view to the application.
            if (portal.viewContainerRef) {
                componentRef = portal.viewContainerRef.createComponent(componentFactory, portal.viewContainerRef.length, portal.injector || portal.viewContainerRef.injector);
                this.setDisposeFn((/**
                 * @return {?}
                 */
                () => componentRef.destroy()));
            }
            else {
                componentRef = componentFactory.create(portal.injector || this._defaultInjector);
                this._appRef.attachView(componentRef.hostView);
                this.setDisposeFn((/**
                 * @return {?}
                 */
                () => {
                    this._appRef.detachView(componentRef.hostView);
                    componentRef.destroy();
                }));
            }
            // At this point the component has been instantiated, so we move it to the location in the DOM
            // where we want it to be rendered.
            this.outletElement.appendChild(this._getComponentRootNode(componentRef));
            return componentRef;
        }
        /**
         * Attaches a template portal to the DOM as an embedded view.
         * @template C
         * @param {?} portal Portal to be attached.
         * @return {?} Reference to the created embedded view.
         */
        attachTemplatePortal(portal) {
            /** @type {?} */
            let viewContainer = portal.viewContainerRef;
            /** @type {?} */
            let viewRef = viewContainer.createEmbeddedView(portal.templateRef, portal.context);
            viewRef.detectChanges();
            // The method `createEmbeddedView` will add the view as a child of the viewContainer.
            // But for the DomPortalOutlet the view can be added everywhere in the DOM
            // (e.g Overlay Container) To move the view to the specified host element. We just
            // re-append the existing root nodes.
            viewRef.rootNodes.forEach((/**
             * @param {?} rootNode
             * @return {?}
             */
            rootNode => this.outletElement.appendChild(rootNode)));
            this.setDisposeFn(((/**
             * @return {?}
             */
            () => {
                /** @type {?} */
                let index = viewContainer.indexOf(viewRef);
                if (index !== -1) {
                    viewContainer.remove(index);
                }
            })));
            // TODO(jelbourn): Return locals from view.
            return viewRef;
        }
        /**
         * Clears out a portal from the DOM.
         * @return {?}
         */
        dispose() {
            super.dispose();
            if (this.outletElement.parentNode != null) {
                this.outletElement.parentNode.removeChild(this.outletElement);
            }
        }
        /**
         * Gets the root HTMLElement for an instantiated component.
         * @private
         * @param {?} componentRef
         * @return {?}
         */
        _getComponentRootNode(componentRef) {
            return (/** @type {?} */ (((/** @type {?} */ (componentRef.hostView))).rootNodes[0]));
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/portal/portal-directives.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Directive version of a `TemplatePortal`. Because the directive *is* a TemplatePortal,
     * the directive instance itself can be attached to a host, enabling declarative use of portals.
     */
    class CdkPortal extends TemplatePortal {
        /**
         * @param {?} templateRef
         * @param {?} viewContainerRef
         */
        constructor(templateRef, viewContainerRef) {
            super(templateRef, viewContainerRef);
        }
    }
    CdkPortal.ɵfac = function CdkPortal_Factory(t) { return new (t || CdkPortal)(ɵɵdirectiveInject(TemplateRef), ɵɵdirectiveInject(ViewContainerRef)); };
    CdkPortal.ɵdir = ɵɵdefineDirective({ type: CdkPortal, selectors: [["", "cdkPortal", ""]], exportAs: ["cdkPortal"], features: [ɵɵInheritDefinitionFeature] });
    /** @nocollapse */
    CdkPortal.ctorParameters = () => [
        { type: TemplateRef },
        { type: ViewContainerRef }
    ];
    /**
     * @deprecated Use `CdkPortal` instead.
     * \@breaking-change 9.0.0
     */
    class TemplatePortalDirective extends CdkPortal {
    }
    TemplatePortalDirective.ɵfac = function TemplatePortalDirective_Factory(t) { return ɵTemplatePortalDirective_BaseFactory(t || TemplatePortalDirective); };
    TemplatePortalDirective.ɵdir = ɵɵdefineDirective({ type: TemplatePortalDirective, selectors: [["", "cdk-portal", ""], ["", "portal", ""]], exportAs: ["cdkPortal"], features: [ɵɵProvidersFeature([{
                    provide: CdkPortal,
                    useExisting: TemplatePortalDirective
                }]), ɵɵInheritDefinitionFeature] });
    const ɵTemplatePortalDirective_BaseFactory = ɵɵgetInheritedFactory(TemplatePortalDirective);
    /**
     * Directive version of a PortalOutlet. Because the directive *is* a PortalOutlet, portals can be
     * directly attached to it, enabling declarative use.
     *
     * Usage:
     * `<ng-template [cdkPortalOutlet]="greeting"></ng-template>`
     */
    class CdkPortalOutlet extends BasePortalOutlet {
        /**
         * @param {?} _componentFactoryResolver
         * @param {?} _viewContainerRef
         * @param {?=} _document
         */
        constructor(_componentFactoryResolver, _viewContainerRef, 
        /**
         * @deprecated `_document` parameter to be made required.
         * @breaking-change 9.0.0
         */
        _document) {
            super();
            this._componentFactoryResolver = _componentFactoryResolver;
            this._viewContainerRef = _viewContainerRef;
            /**
             * Whether the portal component is initialized.
             */
            this._isInitialized = false;
            /**
             * Emits when a portal is attached to the outlet.
             */
            this.attached = new EventEmitter();
            /**
             * Attaches the given DomPortal to this PortalHost by moving all of the portal content into it.
             * @param portal Portal to be attached.
             * @deprecated To be turned into a method.
             * \@breaking-change 10.0.0
             */
            this.attachDomPortal = (/**
             * @param {?} portal
             * @return {?}
             */
            (portal) => {
                // @breaking-change 9.0.0 Remove check and error once the
                // `_document` constructor parameter is required.
                if (!this._document) {
                    throw Error('Cannot attach DOM portal without _document constructor parameter');
                }
                /** @type {?} */
                const element = portal.element;
                if (!element.parentNode) {
                    throw Error('DOM portal content must be attached to a parent node.');
                }
                // Anchor used to save the element's previous position so
                // that we can restore it when the portal is detached.
                /** @type {?} */
                const anchorNode = this._document.createComment('dom-portal');
                portal.setAttachedHost(this);
                element.parentNode.insertBefore(anchorNode, element);
                this._getRootNode().appendChild(element);
                super.setDisposeFn((/**
                 * @return {?}
                 */
                () => {
                    if (anchorNode.parentNode) {
                        (/** @type {?} */ (anchorNode.parentNode)).replaceChild(element, anchorNode);
                    }
                }));
            });
            this._document = _document;
        }
        /**
         * Portal associated with the Portal outlet.
         * @return {?}
         */
        get portal() {
            return this._attachedPortal;
        }
        /**
         * @param {?} portal
         * @return {?}
         */
        set portal(portal) {
            // Ignore the cases where the `portal` is set to a falsy value before the lifecycle hooks have
            // run. This handles the cases where the user might do something like `<div cdkPortalOutlet>`
            // and attach a portal programmatically in the parent component. When Angular does the first CD
            // round, it will fire the setter with empty string, causing the user's content to be cleared.
            if (this.hasAttached() && !portal && !this._isInitialized) {
                return;
            }
            if (this.hasAttached()) {
                super.detach();
            }
            if (portal) {
                super.attach(portal);
            }
            this._attachedPortal = portal;
        }
        /**
         * Component or view reference that is attached to the portal.
         * @return {?}
         */
        get attachedRef() {
            return this._attachedRef;
        }
        /**
         * @return {?}
         */
        ngOnInit() {
            this._isInitialized = true;
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            super.dispose();
            this._attachedPortal = null;
            this._attachedRef = null;
        }
        /**
         * Attach the given ComponentPortal to this PortalOutlet using the ComponentFactoryResolver.
         *
         * @template T
         * @param {?} portal Portal to be attached to the portal outlet.
         * @return {?} Reference to the created component.
         */
        attachComponentPortal(portal) {
            portal.setAttachedHost(this);
            // If the portal specifies an origin, use that as the logical location of the component
            // in the application tree. Otherwise use the location of this PortalOutlet.
            /** @type {?} */
            const viewContainerRef = portal.viewContainerRef != null ?
                portal.viewContainerRef :
                this._viewContainerRef;
            /** @type {?} */
            const resolver = portal.componentFactoryResolver || this._componentFactoryResolver;
            /** @type {?} */
            const componentFactory = resolver.resolveComponentFactory(portal.component);
            /** @type {?} */
            const ref = viewContainerRef.createComponent(componentFactory, viewContainerRef.length, portal.injector || viewContainerRef.injector);
            // If we're using a view container that's different from the injected one (e.g. when the portal
            // specifies its own) we need to move the component into the outlet, otherwise it'll be rendered
            // inside of the alternate view container.
            if (viewContainerRef !== this._viewContainerRef) {
                this._getRootNode().appendChild(((/** @type {?} */ (ref.hostView))).rootNodes[0]);
            }
            super.setDisposeFn((/**
             * @return {?}
             */
            () => ref.destroy()));
            this._attachedPortal = portal;
            this._attachedRef = ref;
            this.attached.emit(ref);
            return ref;
        }
        /**
         * Attach the given TemplatePortal to this PortalHost as an embedded View.
         * @template C
         * @param {?} portal Portal to be attached.
         * @return {?} Reference to the created embedded view.
         */
        attachTemplatePortal(portal) {
            portal.setAttachedHost(this);
            /** @type {?} */
            const viewRef = this._viewContainerRef.createEmbeddedView(portal.templateRef, portal.context);
            super.setDisposeFn((/**
             * @return {?}
             */
            () => this._viewContainerRef.clear()));
            this._attachedPortal = portal;
            this._attachedRef = viewRef;
            this.attached.emit(viewRef);
            return viewRef;
        }
        /**
         * Gets the root node of the portal outlet.
         * @private
         * @return {?}
         */
        _getRootNode() {
            /** @type {?} */
            const nativeElement = this._viewContainerRef.element.nativeElement;
            // The directive could be set on a template which will result in a comment
            // node being the root. Use the comment's parent node if that is the case.
            return (/** @type {?} */ ((nativeElement.nodeType === nativeElement.ELEMENT_NODE ?
                nativeElement : (/** @type {?} */ (nativeElement.parentNode)))));
        }
    }
    CdkPortalOutlet.ɵfac = function CdkPortalOutlet_Factory(t) { return new (t || CdkPortalOutlet)(ɵɵdirectiveInject(ComponentFactoryResolver), ɵɵdirectiveInject(ViewContainerRef), ɵɵdirectiveInject(DOCUMENT$1)); };
    CdkPortalOutlet.ɵdir = ɵɵdefineDirective({ type: CdkPortalOutlet, selectors: [["", "cdkPortalOutlet", ""]], inputs: { portal: ["cdkPortalOutlet", "portal"] }, outputs: { attached: "attached" }, exportAs: ["cdkPortalOutlet"], features: [ɵɵInheritDefinitionFeature] });
    /** @nocollapse */
    CdkPortalOutlet.ctorParameters = () => [
        { type: ComponentFactoryResolver },
        { type: ViewContainerRef },
        { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT$1,] }] }
    ];
    CdkPortalOutlet.propDecorators = {
        attached: [{ type: Output }]
    };
    /**
     * @deprecated Use `CdkPortalOutlet` instead.
     * \@breaking-change 9.0.0
     */
    class PortalHostDirective extends CdkPortalOutlet {
    }
    PortalHostDirective.ɵfac = function PortalHostDirective_Factory(t) { return ɵPortalHostDirective_BaseFactory(t || PortalHostDirective); };
    PortalHostDirective.ɵdir = ɵɵdefineDirective({ type: PortalHostDirective, selectors: [["", "cdkPortalHost", ""], ["", "portalHost", ""]], inputs: { portal: ["cdkPortalHost", "portal"] }, exportAs: ["cdkPortalHost"], features: [ɵɵProvidersFeature([{
                    provide: CdkPortalOutlet,
                    useExisting: PortalHostDirective
                }]), ɵɵInheritDefinitionFeature] });
    const ɵPortalHostDirective_BaseFactory = ɵɵgetInheritedFactory(PortalHostDirective);
    class PortalModule {
    }
    PortalModule.ɵmod = ɵɵdefineNgModule({ type: PortalModule });
    PortalModule.ɵinj = ɵɵdefineInjector({ factory: function PortalModule_Factory(t) { return new (t || PortalModule)(); } });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(PortalModule, { declarations: [CdkPortal,
            CdkPortalOutlet,
            TemplatePortalDirective,
            PortalHostDirective], exports: [CdkPortal,
            CdkPortalOutlet,
            TemplatePortalDirective,
            PortalHostDirective] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/portal/portal-injector.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google LLC All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * Custom injector to be used when providing custom
     * injection tokens to components inside a portal.
     * \@docs-private
     */
    class PortalInjector {
        /**
         * @param {?} _parentInjector
         * @param {?} _customTokens
         */
        constructor(_parentInjector, _customTokens) {
            this._parentInjector = _parentInjector;
            this._customTokens = _customTokens;
        }
        /**
         * @param {?} token
         * @param {?=} notFoundValue
         * @return {?}
         */
        get(token, notFoundValue) {
            /** @type {?} */
            const value = this._customTokens.get(token);
            if (typeof value !== 'undefined') {
                return value;
            }
            return this._parentInjector.get(token, notFoundValue);
        }
    }

    class DataSource {
    }
    /**
     * Checks whether an object is a data source.
     * @param {?} value
     * @return {?}
     */
    function isDataSource(value) {
        // Check if the value is a DataSource by observing if it has a connect function. Cannot
        // be checked as an `instanceof DataSource` since people could create their own sources
        // that match the interface, but don't extend DataSource.
        return value && typeof value.connect === 'function';
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/collections/array-data-source.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * DataSource wrapper for a native array.
     * @template T
     */
    class ArrayDataSource extends DataSource {
        /**
         * @param {?} _data
         */
        constructor(_data) {
            super();
            this._data = _data;
        }
        /**
         * @return {?}
         */
        connect() {
            return isObservable(this._data) ? this._data : of(this._data);
        }
        /**
         * @return {?}
         */
        disconnect() { }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/collections/selection-model.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Class to be used to power selecting one or more options from a list.
     * @template T
     */
    class SelectionModel {
        /**
         * @param {?=} _multiple
         * @param {?=} initiallySelectedValues
         * @param {?=} _emitChanges
         */
        constructor(_multiple = false, initiallySelectedValues, _emitChanges = true) {
            this._multiple = _multiple;
            this._emitChanges = _emitChanges;
            /**
             * Currently-selected values.
             */
            this._selection = new Set();
            /**
             * Keeps track of the deselected options that haven't been emitted by the change event.
             */
            this._deselectedToEmit = [];
            /**
             * Keeps track of the selected options that haven't been emitted by the change event.
             */
            this._selectedToEmit = [];
            /**
             * Event emitted when the value has changed.
             */
            this.changed = new Subject();
            if (initiallySelectedValues && initiallySelectedValues.length) {
                if (_multiple) {
                    initiallySelectedValues.forEach((/**
                     * @param {?} value
                     * @return {?}
                     */
                    value => this._markSelected(value)));
                }
                else {
                    this._markSelected(initiallySelectedValues[0]);
                }
                // Clear the array in order to avoid firing the change event for preselected values.
                this._selectedToEmit.length = 0;
            }
        }
        /**
         * Selected values.
         * @return {?}
         */
        get selected() {
            if (!this._selected) {
                this._selected = Array.from(this._selection.values());
            }
            return this._selected;
        }
        /**
         * Selects a value or an array of values.
         * @param {...?} values
         * @return {?}
         */
        select(...values) {
            this._verifyValueAssignment(values);
            values.forEach((/**
             * @param {?} value
             * @return {?}
             */
            value => this._markSelected(value)));
            this._emitChangeEvent();
        }
        /**
         * Deselects a value or an array of values.
         * @param {...?} values
         * @return {?}
         */
        deselect(...values) {
            this._verifyValueAssignment(values);
            values.forEach((/**
             * @param {?} value
             * @return {?}
             */
            value => this._unmarkSelected(value)));
            this._emitChangeEvent();
        }
        /**
         * Toggles a value between selected and deselected.
         * @param {?} value
         * @return {?}
         */
        toggle(value) {
            this.isSelected(value) ? this.deselect(value) : this.select(value);
        }
        /**
         * Clears all of the selected values.
         * @return {?}
         */
        clear() {
            this._unmarkAll();
            this._emitChangeEvent();
        }
        /**
         * Determines whether a value is selected.
         * @param {?} value
         * @return {?}
         */
        isSelected(value) {
            return this._selection.has(value);
        }
        /**
         * Determines whether the model does not have a value.
         * @return {?}
         */
        isEmpty() {
            return this._selection.size === 0;
        }
        /**
         * Determines whether the model has a value.
         * @return {?}
         */
        hasValue() {
            return !this.isEmpty();
        }
        /**
         * Sorts the selected values based on a predicate function.
         * @param {?=} predicate
         * @return {?}
         */
        sort(predicate) {
            if (this._multiple && this.selected) {
                (/** @type {?} */ (this._selected)).sort(predicate);
            }
        }
        /**
         * Gets whether multiple values can be selected.
         * @return {?}
         */
        isMultipleSelection() {
            return this._multiple;
        }
        /**
         * Emits a change event and clears the records of selected and deselected values.
         * @private
         * @return {?}
         */
        _emitChangeEvent() {
            // Clear the selected values so they can be re-cached.
            this._selected = null;
            if (this._selectedToEmit.length || this._deselectedToEmit.length) {
                this.changed.next({
                    source: this,
                    added: this._selectedToEmit,
                    removed: this._deselectedToEmit
                });
                this._deselectedToEmit = [];
                this._selectedToEmit = [];
            }
        }
        /**
         * Selects a value.
         * @private
         * @param {?} value
         * @return {?}
         */
        _markSelected(value) {
            if (!this.isSelected(value)) {
                if (!this._multiple) {
                    this._unmarkAll();
                }
                this._selection.add(value);
                if (this._emitChanges) {
                    this._selectedToEmit.push(value);
                }
            }
        }
        /**
         * Deselects a value.
         * @private
         * @param {?} value
         * @return {?}
         */
        _unmarkSelected(value) {
            if (this.isSelected(value)) {
                this._selection.delete(value);
                if (this._emitChanges) {
                    this._deselectedToEmit.push(value);
                }
            }
        }
        /**
         * Clears out the selected values.
         * @private
         * @return {?}
         */
        _unmarkAll() {
            if (!this.isEmpty()) {
                this._selection.forEach((/**
                 * @param {?} value
                 * @return {?}
                 */
                value => this._unmarkSelected(value)));
            }
        }
        /**
         * Verifies the value assignment and throws an error if the specified value array is
         * including multiple values while the selection model is not supporting multiple values.
         * @private
         * @param {?} values
         * @return {?}
         */
        _verifyValueAssignment(values) {
            if (values.length > 1 && !this._multiple) {
                throw getMultipleValuesInSingleSelectionError();
            }
        }
    }
    /**
     * Returns an error that reports that multiple values are passed into a selection model
     * with a single value.
     * \@docs-private
     * @return {?}
     */
    function getMultipleValuesInSingleSelectionError() {
        return Error('Cannot pass multiple values into SelectionModel with single-value mode.');
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/collections/unique-selection-dispatcher.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Class to coordinate unique selection based on name.
     * Intended to be consumed as an Angular service.
     * This service is needed because native radio change events are only fired on the item currently
     * being selected, and we still need to uncheck the previous selection.
     *
     * This service does not *store* any IDs and names because they may change at any time, so it is
     * less error-prone if they are simply passed through when the events occur.
     */
    class UniqueSelectionDispatcher {
        constructor() {
            this._listeners = [];
        }
        /**
         * Notify other items that selection for the given name has been set.
         * @param {?} id ID of the item.
         * @param {?} name Name of the item.
         * @return {?}
         */
        notify(id, name) {
            for (let listener of this._listeners) {
                listener(id, name);
            }
        }
        /**
         * Listen for future changes to item selection.
         * @param {?} listener
         * @return {?} Function used to deregister listener
         */
        listen(listener) {
            this._listeners.push(listener);
            return (/**
             * @return {?}
             */
            () => {
                this._listeners = this._listeners.filter((/**
                 * @param {?} registered
                 * @return {?}
                 */
                (registered) => {
                    return listener !== registered;
                }));
            });
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this._listeners = [];
        }
    }
    UniqueSelectionDispatcher.ɵfac = function UniqueSelectionDispatcher_Factory(t) { return new (t || UniqueSelectionDispatcher)(); };
    /** @nocollapse */ UniqueSelectionDispatcher.ɵprov = ɵɵdefineInjectable({ factory: function UniqueSelectionDispatcher_Factory() { return new UniqueSelectionDispatcher(); }, token: UniqueSelectionDispatcher, providedIn: "root" });

    const _c0$2 = ["contentWrapper"];
    const _c1$2 = ["*"];
    const VIRTUAL_SCROLL_STRATEGY = new InjectionToken('VIRTUAL_SCROLL_STRATEGY');

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/scrolling/fixed-size-virtual-scroll.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Virtual scrolling strategy for lists with items of known fixed size.
     */
    class FixedSizeVirtualScrollStrategy {
        /**
         * @param {?} itemSize The size of the items in the virtually scrolling list.
         * @param {?} minBufferPx The minimum amount of buffer (in pixels) before needing to render more
         * @param {?} maxBufferPx The amount of buffer (in pixels) to render when rendering more.
         */
        constructor(itemSize, minBufferPx, maxBufferPx) {
            this._scrolledIndexChange = new Subject();
            /**
             * \@docs-private Implemented as part of VirtualScrollStrategy.
             */
            this.scrolledIndexChange = this._scrolledIndexChange.pipe(distinctUntilChanged());
            /**
             * The attached viewport.
             */
            this._viewport = null;
            this._itemSize = itemSize;
            this._minBufferPx = minBufferPx;
            this._maxBufferPx = maxBufferPx;
        }
        /**
         * Attaches this scroll strategy to a viewport.
         * @param {?} viewport The viewport to attach this strategy to.
         * @return {?}
         */
        attach(viewport) {
            this._viewport = viewport;
            this._updateTotalContentSize();
            this._updateRenderedRange();
        }
        /**
         * Detaches this scroll strategy from the currently attached viewport.
         * @return {?}
         */
        detach() {
            this._scrolledIndexChange.complete();
            this._viewport = null;
        }
        /**
         * Update the item size and buffer size.
         * @param {?} itemSize The size of the items in the virtually scrolling list.
         * @param {?} minBufferPx The minimum amount of buffer (in pixels) before needing to render more
         * @param {?} maxBufferPx The amount of buffer (in pixels) to render when rendering more.
         * @return {?}
         */
        updateItemAndBufferSize(itemSize, minBufferPx, maxBufferPx) {
            if (maxBufferPx < minBufferPx) {
                throw Error('CDK virtual scroll: maxBufferPx must be greater than or equal to minBufferPx');
            }
            this._itemSize = itemSize;
            this._minBufferPx = minBufferPx;
            this._maxBufferPx = maxBufferPx;
            this._updateTotalContentSize();
            this._updateRenderedRange();
        }
        /**
         * \@docs-private Implemented as part of VirtualScrollStrategy.
         * @return {?}
         */
        onContentScrolled() {
            this._updateRenderedRange();
        }
        /**
         * \@docs-private Implemented as part of VirtualScrollStrategy.
         * @return {?}
         */
        onDataLengthChanged() {
            this._updateTotalContentSize();
            this._updateRenderedRange();
        }
        /**
         * \@docs-private Implemented as part of VirtualScrollStrategy.
         * @return {?}
         */
        onContentRendered() { }
        /**
         * \@docs-private Implemented as part of VirtualScrollStrategy.
         * @return {?}
         */
        onRenderedOffsetChanged() { }
        /**
         * Scroll to the offset for the given index.
         * @param {?} index The index of the element to scroll to.
         * @param {?} behavior The ScrollBehavior to use when scrolling.
         * @return {?}
         */
        scrollToIndex(index, behavior) {
            if (this._viewport) {
                this._viewport.scrollToOffset(index * this._itemSize, behavior);
            }
        }
        /**
         * Update the viewport's total content size.
         * @private
         * @return {?}
         */
        _updateTotalContentSize() {
            if (!this._viewport) {
                return;
            }
            this._viewport.setTotalContentSize(this._viewport.getDataLength() * this._itemSize);
        }
        /**
         * Update the viewport's rendered range.
         * @private
         * @return {?}
         */
        _updateRenderedRange() {
            if (!this._viewport) {
                return;
            }
            /** @type {?} */
            const scrollOffset = this._viewport.measureScrollOffset();
            /** @type {?} */
            const firstVisibleIndex = scrollOffset / this._itemSize;
            /** @type {?} */
            const renderedRange = this._viewport.getRenderedRange();
            /** @type {?} */
            const newRange = { start: renderedRange.start, end: renderedRange.end };
            /** @type {?} */
            const viewportSize = this._viewport.getViewportSize();
            /** @type {?} */
            const dataLength = this._viewport.getDataLength();
            /** @type {?} */
            const startBuffer = scrollOffset - newRange.start * this._itemSize;
            if (startBuffer < this._minBufferPx && newRange.start != 0) {
                /** @type {?} */
                const expandStart = Math.ceil((this._maxBufferPx - startBuffer) / this._itemSize);
                newRange.start = Math.max(0, newRange.start - expandStart);
                newRange.end = Math.min(dataLength, Math.ceil(firstVisibleIndex + (viewportSize + this._minBufferPx) / this._itemSize));
            }
            else {
                /** @type {?} */
                const endBuffer = newRange.end * this._itemSize - (scrollOffset + viewportSize);
                if (endBuffer < this._minBufferPx && newRange.end != dataLength) {
                    /** @type {?} */
                    const expandEnd = Math.ceil((this._maxBufferPx - endBuffer) / this._itemSize);
                    if (expandEnd > 0) {
                        newRange.end = Math.min(dataLength, newRange.end + expandEnd);
                        newRange.start = Math.max(0, Math.floor(firstVisibleIndex - this._minBufferPx / this._itemSize));
                    }
                }
            }
            this._viewport.setRenderedRange(newRange);
            this._viewport.setRenderedContentOffset(this._itemSize * newRange.start);
            this._scrolledIndexChange.next(Math.floor(firstVisibleIndex));
        }
    }
    /**
     * Provider factory for `FixedSizeVirtualScrollStrategy` that simply extracts the already created
     * `FixedSizeVirtualScrollStrategy` from the given directive.
     * @param {?} fixedSizeDir The instance of `CdkFixedSizeVirtualScroll` to extract the
     *     `FixedSizeVirtualScrollStrategy` from.
     * @return {?}
     */
    function _fixedSizeVirtualScrollStrategyFactory(fixedSizeDir) {
        return fixedSizeDir._scrollStrategy;
    }
    /**
     * A virtual scroll strategy that supports fixed-size items.
     */
    class CdkFixedSizeVirtualScroll {
        constructor() {
            this._itemSize = 20;
            this._minBufferPx = 100;
            this._maxBufferPx = 200;
            /**
             * The scroll strategy used by this directive.
             */
            this._scrollStrategy = new FixedSizeVirtualScrollStrategy(this.itemSize, this.minBufferPx, this.maxBufferPx);
        }
        /**
         * The size of the items in the list (in pixels).
         * @return {?}
         */
        get itemSize() { return this._itemSize; }
        /**
         * @param {?} value
         * @return {?}
         */
        set itemSize(value) { this._itemSize = coerceNumberProperty(value); }
        /**
         * The minimum amount of buffer rendered beyond the viewport (in pixels).
         * If the amount of buffer dips below this number, more items will be rendered. Defaults to 100px.
         * @return {?}
         */
        get minBufferPx() { return this._minBufferPx; }
        /**
         * @param {?} value
         * @return {?}
         */
        set minBufferPx(value) { this._minBufferPx = coerceNumberProperty(value); }
        /**
         * The number of pixels worth of buffer to render for when rendering new items. Defaults to 200px.
         * @return {?}
         */
        get maxBufferPx() { return this._maxBufferPx; }
        /**
         * @param {?} value
         * @return {?}
         */
        set maxBufferPx(value) { this._maxBufferPx = coerceNumberProperty(value); }
        /**
         * @return {?}
         */
        ngOnChanges() {
            this._scrollStrategy.updateItemAndBufferSize(this.itemSize, this.minBufferPx, this.maxBufferPx);
        }
    }
    CdkFixedSizeVirtualScroll.ɵfac = function CdkFixedSizeVirtualScroll_Factory(t) { return new (t || CdkFixedSizeVirtualScroll)(); };
    CdkFixedSizeVirtualScroll.ɵdir = ɵɵdefineDirective({ type: CdkFixedSizeVirtualScroll, selectors: [["cdk-virtual-scroll-viewport", "itemSize", ""]], inputs: { itemSize: "itemSize", minBufferPx: "minBufferPx", maxBufferPx: "maxBufferPx" }, features: [ɵɵProvidersFeature([{
                    provide: VIRTUAL_SCROLL_STRATEGY,
                    useFactory: _fixedSizeVirtualScrollStrategyFactory,
                    deps: [forwardRef(( /**
                             * @return {?}
                             */() => CdkFixedSizeVirtualScroll))]
                }]), ɵɵNgOnChangesFeature] });
    CdkFixedSizeVirtualScroll.propDecorators = {
        itemSize: [{ type: Input }],
        minBufferPx: [{ type: Input }],
        maxBufferPx: [{ type: Input }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/scrolling/scroll-dispatcher.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Time in ms to throttle the scrolling events by default.
     * @type {?}
     */
    const DEFAULT_SCROLL_TIME = 20;
    /**
     * Service contained all registered Scrollable references and emits an event when any one of the
     * Scrollable references emit a scrolled event.
     */
    class ScrollDispatcher {
        /**
         * @param {?} _ngZone
         * @param {?} _platform
         * @param {?=} document
         */
        constructor(_ngZone, _platform, 
        /** @breaking-change 11.0.0 make document required */
        document) {
            this._ngZone = _ngZone;
            this._platform = _platform;
            /**
             * Subject for notifying that a registered scrollable reference element has been scrolled.
             */
            this._scrolled = new Subject();
            /**
             * Keeps track of the global `scroll` and `resize` subscriptions.
             */
            this._globalSubscription = null;
            /**
             * Keeps track of the amount of subscriptions to `scrolled`. Used for cleaning up afterwards.
             */
            this._scrolledCount = 0;
            /**
             * Map of all the scrollable references that are registered with the service and their
             * scroll event subscriptions.
             */
            this.scrollContainers = new Map();
            this._document = document;
        }
        /**
         * Registers a scrollable instance with the service and listens for its scrolled events. When the
         * scrollable is scrolled, the service emits the event to its scrolled observable.
         * @param {?} scrollable Scrollable instance to be registered.
         * @return {?}
         */
        register(scrollable) {
            if (!this.scrollContainers.has(scrollable)) {
                this.scrollContainers.set(scrollable, scrollable.elementScrolled()
                    .subscribe((/**
                 * @return {?}
                 */
                () => this._scrolled.next(scrollable))));
            }
        }
        /**
         * Deregisters a Scrollable reference and unsubscribes from its scroll event observable.
         * @param {?} scrollable Scrollable instance to be deregistered.
         * @return {?}
         */
        deregister(scrollable) {
            /** @type {?} */
            const scrollableReference = this.scrollContainers.get(scrollable);
            if (scrollableReference) {
                scrollableReference.unsubscribe();
                this.scrollContainers.delete(scrollable);
            }
        }
        /**
         * Returns an observable that emits an event whenever any of the registered Scrollable
         * references (or window, document, or body) fire a scrolled event. Can provide a time in ms
         * to override the default "throttle" time.
         *
         * **Note:** in order to avoid hitting change detection for every scroll event,
         * all of the events emitted from this stream will be run outside the Angular zone.
         * If you need to update any data bindings as a result of a scroll event, you have
         * to run the callback using `NgZone.run`.
         * @param {?=} auditTimeInMs
         * @return {?}
         */
        scrolled(auditTimeInMs = DEFAULT_SCROLL_TIME) {
            if (!this._platform.isBrowser) {
                return of();
            }
            return new Observable((/**
             * @param {?} observer
             * @return {?}
             */
            (observer) => {
                if (!this._globalSubscription) {
                    this._addGlobalListener();
                }
                // In the case of a 0ms delay, use an observable without auditTime
                // since it does add a perceptible delay in processing overhead.
                /** @type {?} */
                const subscription = auditTimeInMs > 0 ?
                    this._scrolled.pipe(auditTime(auditTimeInMs)).subscribe(observer) :
                    this._scrolled.subscribe(observer);
                this._scrolledCount++;
                return (/**
                 * @return {?}
                 */
                () => {
                    subscription.unsubscribe();
                    this._scrolledCount--;
                    if (!this._scrolledCount) {
                        this._removeGlobalListener();
                    }
                });
            }));
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this._removeGlobalListener();
            this.scrollContainers.forEach((/**
             * @param {?} _
             * @param {?} container
             * @return {?}
             */
            (_, container) => this.deregister(container)));
            this._scrolled.complete();
        }
        /**
         * Returns an observable that emits whenever any of the
         * scrollable ancestors of an element are scrolled.
         * @param {?} elementRef Element whose ancestors to listen for.
         * @param {?=} auditTimeInMs Time to throttle the scroll events.
         * @return {?}
         */
        ancestorScrolled(elementRef, auditTimeInMs) {
            /** @type {?} */
            const ancestors = this.getAncestorScrollContainers(elementRef);
            return this.scrolled(auditTimeInMs).pipe(filter((/**
             * @param {?} target
             * @return {?}
             */
            target => {
                return !target || ancestors.indexOf(target) > -1;
            })));
        }
        /**
         * Returns all registered Scrollables that contain the provided element.
         * @param {?} elementRef
         * @return {?}
         */
        getAncestorScrollContainers(elementRef) {
            /** @type {?} */
            const scrollingContainers = [];
            this.scrollContainers.forEach((/**
             * @param {?} _subscription
             * @param {?} scrollable
             * @return {?}
             */
            (_subscription, scrollable) => {
                if (this._scrollableContainsElement(scrollable, elementRef)) {
                    scrollingContainers.push(scrollable);
                }
            }));
            return scrollingContainers;
        }
        /**
         * Access injected document if available or fallback to global document reference
         * @private
         * @return {?}
         */
        _getDocument() {
            return this._document || document;
        }
        /**
         * Use defaultView of injected document if available or fallback to global window reference
         * @private
         * @return {?}
         */
        _getWindow() {
            /** @type {?} */
            const doc = this._getDocument();
            return doc.defaultView || window;
        }
        /**
         * Returns true if the element is contained within the provided Scrollable.
         * @private
         * @param {?} scrollable
         * @param {?} elementRef
         * @return {?}
         */
        _scrollableContainsElement(scrollable, elementRef) {
            /** @type {?} */
            let element = elementRef.nativeElement;
            /** @type {?} */
            let scrollableElement = scrollable.getElementRef().nativeElement;
            // Traverse through the element parents until we reach null, checking if any of the elements
            // are the scrollable's element.
            do {
                if (element == scrollableElement) {
                    return true;
                }
            } while (element = (/** @type {?} */ (element)).parentElement);
            return false;
        }
        /**
         * Sets up the global scroll listeners.
         * @private
         * @return {?}
         */
        _addGlobalListener() {
            this._globalSubscription = this._ngZone.runOutsideAngular((/**
             * @return {?}
             */
            () => {
                /** @type {?} */
                const window = this._getWindow();
                return fromEvent(window.document, 'scroll').subscribe((/**
                 * @return {?}
                 */
                () => this._scrolled.next()));
            }));
        }
        /**
         * Cleans up the global scroll listener.
         * @private
         * @return {?}
         */
        _removeGlobalListener() {
            if (this._globalSubscription) {
                this._globalSubscription.unsubscribe();
                this._globalSubscription = null;
            }
        }
    }
    ScrollDispatcher.ɵfac = function ScrollDispatcher_Factory(t) { return new (t || ScrollDispatcher)(ɵɵinject(NgZone), ɵɵinject(Platform), ɵɵinject(DOCUMENT$1, 8)); };
    /** @nocollapse */
    ScrollDispatcher.ctorParameters = () => [
        { type: NgZone },
        { type: Platform },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [DOCUMENT$1,] }] }
    ];
    /** @nocollapse */ ScrollDispatcher.ɵprov = ɵɵdefineInjectable({ factory: function ScrollDispatcher_Factory() { return new ScrollDispatcher(ɵɵinject(NgZone), ɵɵinject(Platform), ɵɵinject(DOCUMENT$1, 8)); }, token: ScrollDispatcher, providedIn: "root" });

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/scrolling/scrollable.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Sends an event when the directive's element is scrolled. Registers itself with the
     * ScrollDispatcher service to include itself as part of its collection of scrolling events that it
     * can be listened to through the service.
     */
    class CdkScrollable {
        /**
         * @param {?} elementRef
         * @param {?} scrollDispatcher
         * @param {?} ngZone
         * @param {?=} dir
         */
        constructor(elementRef, scrollDispatcher, ngZone, dir) {
            this.elementRef = elementRef;
            this.scrollDispatcher = scrollDispatcher;
            this.ngZone = ngZone;
            this.dir = dir;
            this._destroyed = new Subject();
            this._elementScrolled = new Observable((/**
             * @param {?} observer
             * @return {?}
             */
            (observer) => this.ngZone.runOutsideAngular((/**
             * @return {?}
             */
            () => fromEvent(this.elementRef.nativeElement, 'scroll').pipe(takeUntil(this._destroyed))
                .subscribe(observer)))));
        }
        /**
         * @return {?}
         */
        ngOnInit() {
            this.scrollDispatcher.register(this);
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this.scrollDispatcher.deregister(this);
            this._destroyed.next();
            this._destroyed.complete();
        }
        /**
         * Returns observable that emits when a scroll event is fired on the host element.
         * @return {?}
         */
        elementScrolled() {
            return this._elementScrolled;
        }
        /**
         * Gets the ElementRef for the viewport.
         * @return {?}
         */
        getElementRef() {
            return this.elementRef;
        }
        /**
         * Scrolls to the specified offsets. This is a normalized version of the browser's native scrollTo
         * method, since browsers are not consistent about what scrollLeft means in RTL. For this method
         * left and right always refer to the left and right side of the scrolling container irrespective
         * of the layout direction. start and end refer to left and right in an LTR context and vice-versa
         * in an RTL context.
         * @param {?} options specified the offsets to scroll to.
         * @return {?}
         */
        scrollTo(options) {
            /** @type {?} */
            const el = this.elementRef.nativeElement;
            /** @type {?} */
            const isRtl = this.dir && this.dir.value == 'rtl';
            // Rewrite start & end offsets as right or left offsets.
            if (options.left == null) {
                options.left = isRtl ? options.end : options.start;
            }
            if (options.right == null) {
                options.right = isRtl ? options.start : options.end;
            }
            // Rewrite the bottom offset as a top offset.
            if (options.bottom != null) {
                ((/** @type {?} */ (options))).top =
                    el.scrollHeight - el.clientHeight - options.bottom;
            }
            // Rewrite the right offset as a left offset.
            if (isRtl && getRtlScrollAxisType() != 0 /* NORMAL */) {
                if (options.left != null) {
                    ((/** @type {?} */ (options))).right =
                        el.scrollWidth - el.clientWidth - options.left;
                }
                if (getRtlScrollAxisType() == 2 /* INVERTED */) {
                    options.left = options.right;
                }
                else if (getRtlScrollAxisType() == 1 /* NEGATED */) {
                    options.left = options.right ? -options.right : options.right;
                }
            }
            else {
                if (options.right != null) {
                    ((/** @type {?} */ (options))).left =
                        el.scrollWidth - el.clientWidth - options.right;
                }
            }
            this._applyScrollToOptions(options);
        }
        /**
         * @private
         * @param {?} options
         * @return {?}
         */
        _applyScrollToOptions(options) {
            /** @type {?} */
            const el = this.elementRef.nativeElement;
            if (supportsScrollBehavior()) {
                el.scrollTo(options);
            }
            else {
                if (options.top != null) {
                    el.scrollTop = options.top;
                }
                if (options.left != null) {
                    el.scrollLeft = options.left;
                }
            }
        }
        /**
         * Measures the scroll offset relative to the specified edge of the viewport. This method can be
         * used instead of directly checking scrollLeft or scrollTop, since browsers are not consistent
         * about what scrollLeft means in RTL. The values returned by this method are normalized such that
         * left and right always refer to the left and right side of the scrolling container irrespective
         * of the layout direction. start and end refer to left and right in an LTR context and vice-versa
         * in an RTL context.
         * @param {?} from The edge to measure from.
         * @return {?}
         */
        measureScrollOffset(from) {
            /** @type {?} */
            const LEFT = 'left';
            /** @type {?} */
            const RIGHT = 'right';
            /** @type {?} */
            const el = this.elementRef.nativeElement;
            if (from == 'top') {
                return el.scrollTop;
            }
            if (from == 'bottom') {
                return el.scrollHeight - el.clientHeight - el.scrollTop;
            }
            // Rewrite start & end as left or right offsets.
            /** @type {?} */
            const isRtl = this.dir && this.dir.value == 'rtl';
            if (from == 'start') {
                from = isRtl ? RIGHT : LEFT;
            }
            else if (from == 'end') {
                from = isRtl ? LEFT : RIGHT;
            }
            if (isRtl && getRtlScrollAxisType() == 2 /* INVERTED */) {
                // For INVERTED, scrollLeft is (scrollWidth - clientWidth) when scrolled all the way left and
                // 0 when scrolled all the way right.
                if (from == LEFT) {
                    return el.scrollWidth - el.clientWidth - el.scrollLeft;
                }
                else {
                    return el.scrollLeft;
                }
            }
            else if (isRtl && getRtlScrollAxisType() == 1 /* NEGATED */) {
                // For NEGATED, scrollLeft is -(scrollWidth - clientWidth) when scrolled all the way left and
                // 0 when scrolled all the way right.
                if (from == LEFT) {
                    return el.scrollLeft + el.scrollWidth - el.clientWidth;
                }
                else {
                    return -el.scrollLeft;
                }
            }
            else {
                // For NORMAL, as well as non-RTL contexts, scrollLeft is 0 when scrolled all the way left and
                // (scrollWidth - clientWidth) when scrolled all the way right.
                if (from == LEFT) {
                    return el.scrollLeft;
                }
                else {
                    return el.scrollWidth - el.clientWidth - el.scrollLeft;
                }
            }
        }
    }
    CdkScrollable.ɵfac = function CdkScrollable_Factory(t) { return new (t || CdkScrollable)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(ScrollDispatcher), ɵɵdirectiveInject(NgZone), ɵɵdirectiveInject(Directionality, 8)); };
    CdkScrollable.ɵdir = ɵɵdefineDirective({ type: CdkScrollable, selectors: [["", "cdk-scrollable", ""], ["", "cdkScrollable", ""]] });
    /** @nocollapse */
    CdkScrollable.ctorParameters = () => [
        { type: ElementRef },
        { type: ScrollDispatcher },
        { type: NgZone },
        { type: Directionality, decorators: [{ type: Optional }] }
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/scrolling/viewport-ruler.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Time in ms to throttle the resize events by default.
     * @type {?}
     */
    const DEFAULT_RESIZE_TIME = 20;
    /**
     * Simple utility for getting the bounds of the browser viewport.
     * \@docs-private
     */
    class ViewportRuler {
        /**
         * @param {?} _platform
         * @param {?} ngZone
         * @param {?=} document
         */
        constructor(_platform, ngZone, 
        /** @breaking-change 11.0.0 make document required */
        document) {
            this._platform = _platform;
            this._document = document;
            ngZone.runOutsideAngular((/**
             * @return {?}
             */
            () => {
                /** @type {?} */
                const window = this._getWindow();
                this._change = _platform.isBrowser ?
                    merge(fromEvent(window, 'resize'), fromEvent(window, 'orientationchange')) :
                    of();
                // Note that we need to do the subscription inside `runOutsideAngular`
                // since subscribing is what causes the event listener to be added.
                this._invalidateCache = this.change().subscribe((/**
                 * @return {?}
                 */
                () => this._updateViewportSize()));
            }));
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this._invalidateCache.unsubscribe();
        }
        /**
         * Returns the viewport's width and height.
         * @return {?}
         */
        getViewportSize() {
            if (!this._viewportSize) {
                this._updateViewportSize();
            }
            /** @type {?} */
            const output = { width: this._viewportSize.width, height: this._viewportSize.height };
            // If we're not on a browser, don't cache the size since it'll be mocked out anyway.
            if (!this._platform.isBrowser) {
                this._viewportSize = (/** @type {?} */ (null));
            }
            return output;
        }
        /**
         * Gets a ClientRect for the viewport's bounds.
         * @return {?}
         */
        getViewportRect() {
            // Use the document element's bounding rect rather than the window scroll properties
            // (e.g. pageYOffset, scrollY) due to in issue in Chrome and IE where window scroll
            // properties and client coordinates (boundingClientRect, clientX/Y, etc.) are in different
            // conceptual viewports. Under most circumstances these viewports are equivalent, but they
            // can disagree when the page is pinch-zoomed (on devices that support touch).
            // See https://bugs.chromium.org/p/chromium/issues/detail?id=489206#c4
            // We use the documentElement instead of the body because, by default (without a css reset)
            // browsers typically give the document body an 8px margin, which is not included in
            // getBoundingClientRect().
            /** @type {?} */
            const scrollPosition = this.getViewportScrollPosition();
            const { width, height } = this.getViewportSize();
            return {
                top: scrollPosition.top,
                left: scrollPosition.left,
                bottom: scrollPosition.top + height,
                right: scrollPosition.left + width,
                height,
                width,
            };
        }
        /**
         * Gets the (top, left) scroll position of the viewport.
         * @return {?}
         */
        getViewportScrollPosition() {
            // While we can get a reference to the fake document
            // during SSR, it doesn't have getBoundingClientRect.
            if (!this._platform.isBrowser) {
                return { top: 0, left: 0 };
            }
            // The top-left-corner of the viewport is determined by the scroll position of the document
            // body, normally just (scrollLeft, scrollTop). However, Chrome and Firefox disagree about
            // whether `document.body` or `document.documentElement` is the scrolled element, so reading
            // `scrollTop` and `scrollLeft` is inconsistent. However, using the bounding rect of
            // `document.documentElement` works consistently, where the `top` and `left` values will
            // equal negative the scroll position.
            /** @type {?} */
            const document = this._getDocument();
            /** @type {?} */
            const window = this._getWindow();
            /** @type {?} */
            const documentElement = (/** @type {?} */ (document.documentElement));
            /** @type {?} */
            const documentRect = documentElement.getBoundingClientRect();
            /** @type {?} */
            const top = -documentRect.top || document.body.scrollTop || window.scrollY ||
                documentElement.scrollTop || 0;
            /** @type {?} */
            const left = -documentRect.left || document.body.scrollLeft || window.scrollX ||
                documentElement.scrollLeft || 0;
            return { top, left };
        }
        /**
         * Returns a stream that emits whenever the size of the viewport changes.
         * @param {?=} throttleTime Time in milliseconds to throttle the stream.
         * @return {?}
         */
        change(throttleTime = DEFAULT_RESIZE_TIME) {
            return throttleTime > 0 ? this._change.pipe(auditTime(throttleTime)) : this._change;
        }
        /**
         * Access injected document if available or fallback to global document reference
         * @private
         * @return {?}
         */
        _getDocument() {
            return this._document || document;
        }
        /**
         * Use defaultView of injected document if available or fallback to global window reference
         * @private
         * @return {?}
         */
        _getWindow() {
            /** @type {?} */
            const doc = this._getDocument();
            return doc.defaultView || window;
        }
        /**
         * Updates the cached viewport size.
         * @private
         * @return {?}
         */
        _updateViewportSize() {
            /** @type {?} */
            const window = this._getWindow();
            this._viewportSize = this._platform.isBrowser ?
                { width: window.innerWidth, height: window.innerHeight } :
                { width: 0, height: 0 };
        }
    }
    ViewportRuler.ɵfac = function ViewportRuler_Factory(t) { return new (t || ViewportRuler)(ɵɵinject(Platform), ɵɵinject(NgZone), ɵɵinject(DOCUMENT$1, 8)); };
    /** @nocollapse */
    ViewportRuler.ctorParameters = () => [
        { type: Platform },
        { type: NgZone },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [DOCUMENT$1,] }] }
    ];
    /** @nocollapse */ ViewportRuler.ɵprov = ɵɵdefineInjectable({ factory: function ViewportRuler_Factory() { return new ViewportRuler(ɵɵinject(Platform), ɵɵinject(NgZone), ɵɵinject(DOCUMENT$1, 8)); }, token: ViewportRuler, providedIn: "root" });

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/scrolling/virtual-scroll-viewport.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Checks if the given ranges are equal.
     * @param {?} r1
     * @param {?} r2
     * @return {?}
     */
    function rangesEqual(r1, r2) {
        return r1.start == r2.start && r1.end == r2.end;
    }
    /**
     * Scheduler to be used for scroll events. Needs to fall back to
     * something that doesn't rely on requestAnimationFrame on environments
     * that don't support it (e.g. server-side rendering).
     * @type {?}
     */
    const SCROLL_SCHEDULER = typeof requestAnimationFrame !== 'undefined' ? animationFrame : asap;
    /**
     * A viewport that virtualizes its scrolling with the help of `CdkVirtualForOf`.
     */
    class CdkVirtualScrollViewport extends CdkScrollable {
        /**
         * @param {?} elementRef
         * @param {?} _changeDetectorRef
         * @param {?} ngZone
         * @param {?} _scrollStrategy
         * @param {?} dir
         * @param {?} scrollDispatcher
         * @param {?=} viewportRuler
         */
        constructor(elementRef, _changeDetectorRef, ngZone, _scrollStrategy, dir, scrollDispatcher, 
        /**
         * @deprecated `viewportRuler` parameter to become required.
         * @breaking-change 11.0.0
         */
        viewportRuler) {
            super(elementRef, scrollDispatcher, ngZone, dir);
            this.elementRef = elementRef;
            this._changeDetectorRef = _changeDetectorRef;
            this._scrollStrategy = _scrollStrategy;
            /**
             * Emits when the viewport is detached from a CdkVirtualForOf.
             */
            this._detachedSubject = new Subject();
            /**
             * Emits when the rendered range changes.
             */
            this._renderedRangeSubject = new Subject();
            this._orientation = 'vertical';
            // Note: we don't use the typical EventEmitter here because we need to subscribe to the scroll
            // strategy lazily (i.e. only if the user is actually listening to the events). We do this because
            // depending on how the strategy calculates the scrolled index, it may come at a cost to
            // performance.
            /**
             * Emits when the index of the first element visible in the viewport changes.
             */
            this.scrolledIndexChange = new Observable((/**
             * @param {?} observer
             * @return {?}
             */
            (observer) => this._scrollStrategy.scrolledIndexChange.subscribe((/**
             * @param {?} index
             * @return {?}
             */
            index => Promise.resolve().then((/**
             * @return {?}
             */
            () => this.ngZone.run((/**
             * @return {?}
             */
            () => observer.next(index)))))))));
            /**
             * A stream that emits whenever the rendered range changes.
             */
            this.renderedRangeStream = this._renderedRangeSubject.asObservable();
            /**
             * The total size of all content (in pixels), including content that is not currently rendered.
             */
            this._totalContentSize = 0;
            /**
             * A string representing the `style.width` property value to be used for the spacer element.
             */
            this._totalContentWidth = '';
            /**
             * A string representing the `style.height` property value to be used for the spacer element.
             */
            this._totalContentHeight = '';
            /**
             * The currently rendered range of indices.
             */
            this._renderedRange = { start: 0, end: 0 };
            /**
             * The length of the data bound to this viewport (in number of items).
             */
            this._dataLength = 0;
            /**
             * The size of the viewport (in pixels).
             */
            this._viewportSize = 0;
            /**
             * The last rendered content offset that was set.
             */
            this._renderedContentOffset = 0;
            /**
             * Whether the last rendered content offset was to the end of the content (and therefore needs to
             * be rewritten as an offset to the start of the content).
             */
            this._renderedContentOffsetNeedsRewrite = false;
            /**
             * Whether there is a pending change detection cycle.
             */
            this._isChangeDetectionPending = false;
            /**
             * A list of functions to run after the next change detection cycle.
             */
            this._runAfterChangeDetection = [];
            /**
             * Subscription to changes in the viewport size.
             */
            this._viewportChanges = Subscription.EMPTY;
            if (!_scrollStrategy) {
                throw Error('Error: cdk-virtual-scroll-viewport requires the "itemSize" property to be set.');
            }
            // @breaking-change 11.0.0 Remove null check for `viewportRuler`.
            if (viewportRuler) {
                this._viewportChanges = viewportRuler.change().subscribe((/**
                 * @return {?}
                 */
                () => {
                    this.checkViewportSize();
                }));
            }
        }
        /**
         * The direction the viewport scrolls.
         * @return {?}
         */
        get orientation() {
            return this._orientation;
        }
        /**
         * @param {?} orientation
         * @return {?}
         */
        set orientation(orientation) {
            if (this._orientation !== orientation) {
                this._orientation = orientation;
                this._calculateSpacerSize();
            }
        }
        /**
         * @return {?}
         */
        ngOnInit() {
            super.ngOnInit();
            // It's still too early to measure the viewport at this point. Deferring with a promise allows
            // the Viewport to be rendered with the correct size before we measure. We run this outside the
            // zone to avoid causing more change detection cycles. We handle the change detection loop
            // ourselves instead.
            this.ngZone.runOutsideAngular((/**
             * @return {?}
             */
            () => Promise.resolve().then((/**
             * @return {?}
             */
            () => {
                this._measureViewportSize();
                this._scrollStrategy.attach(this);
                this.elementScrolled()
                    .pipe(
                // Start off with a fake scroll event so we properly detect our initial position.
                startWith((/** @type {?} */ (null))), 
                // Collect multiple events into one until the next animation frame. This way if
                // there are multiple scroll events in the same frame we only need to recheck
                // our layout once.
                auditTime(0, SCROLL_SCHEDULER))
                    .subscribe((/**
                 * @return {?}
                 */
                () => this._scrollStrategy.onContentScrolled()));
                this._markChangeDetectionNeeded();
            }))));
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this.detach();
            this._scrollStrategy.detach();
            // Complete all subjects
            this._renderedRangeSubject.complete();
            this._detachedSubject.complete();
            this._viewportChanges.unsubscribe();
            super.ngOnDestroy();
        }
        /**
         * Attaches a `CdkVirtualForOf` to this viewport.
         * @param {?} forOf
         * @return {?}
         */
        attach(forOf) {
            if (this._forOf) {
                throw Error('CdkVirtualScrollViewport is already attached.');
            }
            // Subscribe to the data stream of the CdkVirtualForOf to keep track of when the data length
            // changes. Run outside the zone to avoid triggering change detection, since we're managing the
            // change detection loop ourselves.
            this.ngZone.runOutsideAngular((/**
             * @return {?}
             */
            () => {
                this._forOf = forOf;
                this._forOf.dataStream.pipe(takeUntil(this._detachedSubject)).subscribe((/**
                 * @param {?} data
                 * @return {?}
                 */
                data => {
                    /** @type {?} */
                    const newLength = data.length;
                    if (newLength !== this._dataLength) {
                        this._dataLength = newLength;
                        this._scrollStrategy.onDataLengthChanged();
                    }
                    this._doChangeDetection();
                }));
            }));
        }
        /**
         * Detaches the current `CdkVirtualForOf`.
         * @return {?}
         */
        detach() {
            this._forOf = null;
            this._detachedSubject.next();
        }
        /**
         * Gets the length of the data bound to this viewport (in number of items).
         * @return {?}
         */
        getDataLength() {
            return this._dataLength;
        }
        /**
         * Gets the size of the viewport (in pixels).
         * @return {?}
         */
        getViewportSize() {
            return this._viewportSize;
        }
        // TODO(mmalerba): This is technically out of sync with what's really rendered until a render
        // cycle happens. I'm being careful to only call it after the render cycle is complete and before
        // setting it to something else, but its error prone and should probably be split into
        // `pendingRange` and `renderedRange`, the latter reflecting whats actually in the DOM.
        /**
         * Get the current rendered range of items.
         * @return {?}
         */
        getRenderedRange() {
            return this._renderedRange;
        }
        /**
         * Sets the total size of all content (in pixels), including content that is not currently
         * rendered.
         * @param {?} size
         * @return {?}
         */
        setTotalContentSize(size) {
            if (this._totalContentSize !== size) {
                this._totalContentSize = size;
                this._calculateSpacerSize();
                this._markChangeDetectionNeeded();
            }
        }
        /**
         * Sets the currently rendered range of indices.
         * @param {?} range
         * @return {?}
         */
        setRenderedRange(range) {
            if (!rangesEqual(this._renderedRange, range)) {
                this._renderedRangeSubject.next(this._renderedRange = range);
                this._markChangeDetectionNeeded((/**
                 * @return {?}
                 */
                () => this._scrollStrategy.onContentRendered()));
            }
        }
        /**
         * Gets the offset from the start of the viewport to the start of the rendered data (in pixels).
         * @return {?}
         */
        getOffsetToRenderedContentStart() {
            return this._renderedContentOffsetNeedsRewrite ? null : this._renderedContentOffset;
        }
        /**
         * Sets the offset from the start of the viewport to either the start or end of the rendered data
         * (in pixels).
         * @param {?} offset
         * @param {?=} to
         * @return {?}
         */
        setRenderedContentOffset(offset, to = 'to-start') {
            // For a horizontal viewport in a right-to-left language we need to translate along the x-axis
            // in the negative direction.
            /** @type {?} */
            const isRtl = this.dir && this.dir.value == 'rtl';
            /** @type {?} */
            const isHorizontal = this.orientation == 'horizontal';
            /** @type {?} */
            const axis = isHorizontal ? 'X' : 'Y';
            /** @type {?} */
            const axisDirection = isHorizontal && isRtl ? -1 : 1;
            /** @type {?} */
            let transform = `translate${axis}(${Number(axisDirection * offset)}px)`;
            this._renderedContentOffset = offset;
            if (to === 'to-end') {
                transform += ` translate${axis}(-100%)`;
                // The viewport should rewrite this as a `to-start` offset on the next render cycle. Otherwise
                // elements will appear to expand in the wrong direction (e.g. `mat-expansion-panel` would
                // expand upward).
                this._renderedContentOffsetNeedsRewrite = true;
            }
            if (this._renderedContentTransform != transform) {
                // We know this value is safe because we parse `offset` with `Number()` before passing it
                // into the string.
                this._renderedContentTransform = transform;
                this._markChangeDetectionNeeded((/**
                 * @return {?}
                 */
                () => {
                    if (this._renderedContentOffsetNeedsRewrite) {
                        this._renderedContentOffset -= this.measureRenderedContentSize();
                        this._renderedContentOffsetNeedsRewrite = false;
                        this.setRenderedContentOffset(this._renderedContentOffset);
                    }
                    else {
                        this._scrollStrategy.onRenderedOffsetChanged();
                    }
                }));
            }
        }
        /**
         * Scrolls to the given offset from the start of the viewport. Please note that this is not always
         * the same as setting `scrollTop` or `scrollLeft`. In a horizontal viewport with right-to-left
         * direction, this would be the equivalent of setting a fictional `scrollRight` property.
         * @param {?} offset The offset to scroll to.
         * @param {?=} behavior The ScrollBehavior to use when scrolling. Default is behavior is `auto`.
         * @return {?}
         */
        scrollToOffset(offset, behavior = 'auto') {
            /** @type {?} */
            const options = { behavior };
            if (this.orientation === 'horizontal') {
                options.start = offset;
            }
            else {
                options.top = offset;
            }
            this.scrollTo(options);
        }
        /**
         * Scrolls to the offset for the given index.
         * @param {?} index The index of the element to scroll to.
         * @param {?=} behavior The ScrollBehavior to use when scrolling. Default is behavior is `auto`.
         * @return {?}
         */
        scrollToIndex(index, behavior = 'auto') {
            this._scrollStrategy.scrollToIndex(index, behavior);
        }
        /**
         * Gets the current scroll offset from the start of the viewport (in pixels).
         * @param {?=} from The edge to measure the offset from. Defaults to 'top' in vertical mode and 'start'
         *     in horizontal mode.
         * @return {?}
         */
        measureScrollOffset(from) {
            return from ?
                super.measureScrollOffset(from) :
                super.measureScrollOffset(this.orientation === 'horizontal' ? 'start' : 'top');
        }
        /**
         * Measure the combined size of all of the rendered items.
         * @return {?}
         */
        measureRenderedContentSize() {
            /** @type {?} */
            const contentEl = this._contentWrapper.nativeElement;
            return this.orientation === 'horizontal' ? contentEl.offsetWidth : contentEl.offsetHeight;
        }
        /**
         * Measure the total combined size of the given range. Throws if the range includes items that are
         * not rendered.
         * @param {?} range
         * @return {?}
         */
        measureRangeSize(range) {
            if (!this._forOf) {
                return 0;
            }
            return this._forOf.measureRangeSize(range, this.orientation);
        }
        /**
         * Update the viewport dimensions and re-render.
         * @return {?}
         */
        checkViewportSize() {
            // TODO: Cleanup later when add logic for handling content resize
            this._measureViewportSize();
            this._scrollStrategy.onDataLengthChanged();
        }
        /**
         * Measure the viewport size.
         * @private
         * @return {?}
         */
        _measureViewportSize() {
            /** @type {?} */
            const viewportEl = this.elementRef.nativeElement;
            this._viewportSize = this.orientation === 'horizontal' ?
                viewportEl.clientWidth : viewportEl.clientHeight;
        }
        /**
         * Queue up change detection to run.
         * @private
         * @param {?=} runAfter
         * @return {?}
         */
        _markChangeDetectionNeeded(runAfter) {
            if (runAfter) {
                this._runAfterChangeDetection.push(runAfter);
            }
            // Use a Promise to batch together calls to `_doChangeDetection`. This way if we set a bunch of
            // properties sequentially we only have to run `_doChangeDetection` once at the end.
            if (!this._isChangeDetectionPending) {
                this._isChangeDetectionPending = true;
                this.ngZone.runOutsideAngular((/**
                 * @return {?}
                 */
                () => Promise.resolve().then((/**
                 * @return {?}
                 */
                () => {
                    this._doChangeDetection();
                }))));
            }
        }
        /**
         * Run change detection.
         * @private
         * @return {?}
         */
        _doChangeDetection() {
            this._isChangeDetectionPending = false;
            // Apply the content transform. The transform can't be set via an Angular binding because
            // bypassSecurityTrustStyle is banned in Google. However the value is safe, it's composed of
            // string literals, a variable that can only be 'X' or 'Y', and user input that is run through
            // the `Number` function first to coerce it to a numeric value.
            this._contentWrapper.nativeElement.style.transform = this._renderedContentTransform;
            // Apply changes to Angular bindings. Note: We must call `markForCheck` to run change detection
            // from the root, since the repeated items are content projected in. Calling `detectChanges`
            // instead does not properly check the projected content.
            this.ngZone.run((/**
             * @return {?}
             */
            () => this._changeDetectorRef.markForCheck()));
            /** @type {?} */
            const runAfterChangeDetection = this._runAfterChangeDetection;
            this._runAfterChangeDetection = [];
            for (const fn of runAfterChangeDetection) {
                fn();
            }
        }
        /**
         * Calculates the `style.width` and `style.height` for the spacer element.
         * @private
         * @return {?}
         */
        _calculateSpacerSize() {
            this._totalContentHeight =
                this.orientation === 'horizontal' ? '' : `${this._totalContentSize}px`;
            this._totalContentWidth =
                this.orientation === 'horizontal' ? `${this._totalContentSize}px` : '';
        }
    }
    CdkVirtualScrollViewport.ɵfac = function CdkVirtualScrollViewport_Factory(t) { return new (t || CdkVirtualScrollViewport)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(ChangeDetectorRef), ɵɵdirectiveInject(NgZone), ɵɵdirectiveInject(VIRTUAL_SCROLL_STRATEGY, 8), ɵɵdirectiveInject(Directionality, 8), ɵɵdirectiveInject(ScrollDispatcher), ɵɵdirectiveInject(ViewportRuler, 8)); };
    CdkVirtualScrollViewport.ɵcmp = ɵɵdefineComponent({ type: CdkVirtualScrollViewport, selectors: [["cdk-virtual-scroll-viewport"]], viewQuery: function CdkVirtualScrollViewport_Query(rf, ctx) { if (rf & 1) {
            ɵɵstaticViewQuery(_c0$2, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._contentWrapper = _t.first);
        } }, hostAttrs: [1, "cdk-virtual-scroll-viewport"], hostVars: 4, hostBindings: function CdkVirtualScrollViewport_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵclassProp("cdk-virtual-scroll-orientation-horizontal", ctx.orientation === "horizontal")("cdk-virtual-scroll-orientation-vertical", ctx.orientation !== "horizontal");
        } }, inputs: { orientation: "orientation" }, outputs: { scrolledIndexChange: "scrolledIndexChange" }, features: [ɵɵProvidersFeature([{
                    provide: CdkScrollable,
                    useExisting: CdkVirtualScrollViewport
                }]), ɵɵInheritDefinitionFeature], ngContentSelectors: _c1$2, decls: 4, vars: 4, consts: [[1, "cdk-virtual-scroll-content-wrapper"], ["contentWrapper", ""], [1, "cdk-virtual-scroll-spacer"]], template: function CdkVirtualScrollViewport_Template(rf, ctx) { if (rf & 1) {
            ɵɵprojectionDef();
            ɵɵelementStart(0, "div", 0, 1);
            ɵɵprojection(2);
            ɵɵelementEnd();
            ɵɵelement(3, "div", 2);
        } if (rf & 2) {
            ɵɵadvance(3);
            ɵɵstyleProp("width", ctx._totalContentWidth)("height", ctx._totalContentHeight);
        } }, styles: ["cdk-virtual-scroll-viewport{display:block;position:relative;overflow:auto;contain:strict;transform:translateZ(0);will-change:scroll-position;-webkit-overflow-scrolling:touch}.cdk-virtual-scroll-content-wrapper{position:absolute;top:0;left:0;contain:content}[dir=rtl] .cdk-virtual-scroll-content-wrapper{right:0;left:auto}.cdk-virtual-scroll-orientation-horizontal .cdk-virtual-scroll-content-wrapper{min-height:100%}.cdk-virtual-scroll-orientation-horizontal .cdk-virtual-scroll-content-wrapper>dl:not([cdkVirtualFor]),.cdk-virtual-scroll-orientation-horizontal .cdk-virtual-scroll-content-wrapper>ol:not([cdkVirtualFor]),.cdk-virtual-scroll-orientation-horizontal .cdk-virtual-scroll-content-wrapper>table:not([cdkVirtualFor]),.cdk-virtual-scroll-orientation-horizontal .cdk-virtual-scroll-content-wrapper>ul:not([cdkVirtualFor]){padding-left:0;padding-right:0;margin-left:0;margin-right:0;border-left-width:0;border-right-width:0;outline:none}.cdk-virtual-scroll-orientation-vertical .cdk-virtual-scroll-content-wrapper{min-width:100%}.cdk-virtual-scroll-orientation-vertical .cdk-virtual-scroll-content-wrapper>dl:not([cdkVirtualFor]),.cdk-virtual-scroll-orientation-vertical .cdk-virtual-scroll-content-wrapper>ol:not([cdkVirtualFor]),.cdk-virtual-scroll-orientation-vertical .cdk-virtual-scroll-content-wrapper>table:not([cdkVirtualFor]),.cdk-virtual-scroll-orientation-vertical .cdk-virtual-scroll-content-wrapper>ul:not([cdkVirtualFor]){padding-top:0;padding-bottom:0;margin-top:0;margin-bottom:0;border-top-width:0;border-bottom-width:0;outline:none}.cdk-virtual-scroll-spacer{position:absolute;top:0;left:0;height:1px;width:1px;transform-origin:0 0}[dir=rtl] .cdk-virtual-scroll-spacer{right:0;left:auto;transform-origin:100% 0}\n"], encapsulation: 2, changeDetection: 0 });
    /** @nocollapse */
    CdkVirtualScrollViewport.ctorParameters = () => [
        { type: ElementRef },
        { type: ChangeDetectorRef },
        { type: NgZone },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [VIRTUAL_SCROLL_STRATEGY,] }] },
        { type: Directionality, decorators: [{ type: Optional }] },
        { type: ScrollDispatcher },
        { type: ViewportRuler, decorators: [{ type: Optional }] }
    ];
    CdkVirtualScrollViewport.propDecorators = {
        orientation: [{ type: Input }],
        scrolledIndexChange: [{ type: Output }],
        _contentWrapper: [{ type: ViewChild, args: ['contentWrapper', { static: true },] }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/scrolling/virtual-for-of.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Helper to extract size from a DOM Node.
     * @param {?} orientation
     * @param {?} node
     * @return {?}
     */
    function getSize(orientation, node) {
        /** @type {?} */
        const el = (/** @type {?} */ (node));
        if (!el.getBoundingClientRect) {
            return 0;
        }
        /** @type {?} */
        const rect = el.getBoundingClientRect();
        return orientation == 'horizontal' ? rect.width : rect.height;
    }
    /**
     * A directive similar to `ngForOf` to be used for rendering data inside a virtual scrolling
     * container.
     * @template T
     */
    class CdkVirtualForOf {
        /**
         * @param {?} _viewContainerRef
         * @param {?} _template
         * @param {?} _differs
         * @param {?} _viewport
         * @param {?} ngZone
         */
        constructor(_viewContainerRef, _template, _differs, _viewport, ngZone) {
            this._viewContainerRef = _viewContainerRef;
            this._template = _template;
            this._differs = _differs;
            this._viewport = _viewport;
            /**
             * Emits when the rendered view of the data changes.
             */
            this.viewChange = new Subject();
            /**
             * Subject that emits when a new DataSource instance is given.
             */
            this._dataSourceChanges = new Subject();
            /**
             * The size of the cache used to store templates that are not being used for re-use later.
             * Setting the cache size to `0` will disable caching. Defaults to 20 templates.
             */
            this.cdkVirtualForTemplateCacheSize = 20;
            /**
             * Emits whenever the data in the current DataSource changes.
             */
            this.dataStream = this._dataSourceChanges
                .pipe(
            // Start off with null `DataSource`.
            startWith((/** @type {?} */ (null))), 
            // Bundle up the previous and current data sources so we can work with both.
            pairwise(), 
            // Use `_changeDataSource` to disconnect from the previous data source and connect to the
            // new one, passing back a stream of data changes which we run through `switchMap` to give
            // us a data stream that emits the latest data from whatever the current `DataSource` is.
            switchMap((/**
             * @param {?} __0
             * @return {?}
             */
            ([prev, cur]) => this._changeDataSource(prev, cur))), 
            // Replay the last emitted data when someone subscribes.
            shareReplay(1));
            /**
             * The differ used to calculate changes to the data.
             */
            this._differ = null;
            /**
             * The template cache used to hold on ot template instancess that have been stamped out, but don't
             * currently need to be rendered. These instances will be reused in the future rather than
             * stamping out brand new ones.
             */
            this._templateCache = [];
            /**
             * Whether the rendered data should be updated during the next ngDoCheck cycle.
             */
            this._needsUpdate = false;
            this._destroyed = new Subject();
            this.dataStream.subscribe((/**
             * @param {?} data
             * @return {?}
             */
            data => {
                this._data = data;
                this._onRenderedDataChange();
            }));
            this._viewport.renderedRangeStream.pipe(takeUntil(this._destroyed)).subscribe((/**
             * @param {?} range
             * @return {?}
             */
            range => {
                this._renderedRange = range;
                ngZone.run((/**
                 * @return {?}
                 */
                () => this.viewChange.next(this._renderedRange)));
                this._onRenderedDataChange();
            }));
            this._viewport.attach(this);
        }
        /**
         * The DataSource to display.
         * @return {?}
         */
        get cdkVirtualForOf() {
            return this._cdkVirtualForOf;
        }
        /**
         * @param {?} value
         * @return {?}
         */
        set cdkVirtualForOf(value) {
            this._cdkVirtualForOf = value;
            if (isDataSource(value)) {
                this._dataSourceChanges.next(value);
            }
            else {
                // Slice the value if its an NgIterable to ensure we're working with an array.
                this._dataSourceChanges.next(new ArrayDataSource(isObservable(value) ? value : Array.prototype.slice.call(value || [])));
            }
        }
        /**
         * The `TrackByFunction` to use for tracking changes. The `TrackByFunction` takes the index and
         * the item and produces a value to be used as the item's identity when tracking changes.
         * @return {?}
         */
        get cdkVirtualForTrackBy() {
            return this._cdkVirtualForTrackBy;
        }
        /**
         * @param {?} fn
         * @return {?}
         */
        set cdkVirtualForTrackBy(fn) {
            this._needsUpdate = true;
            this._cdkVirtualForTrackBy = fn ?
                (/**
                 * @param {?} index
                 * @param {?} item
                 * @return {?}
                 */
                (index, item) => fn(index + (this._renderedRange ? this._renderedRange.start : 0), item)) :
                undefined;
        }
        /**
         * The template used to stamp out new elements.
         * @param {?} value
         * @return {?}
         */
        set cdkVirtualForTemplate(value) {
            if (value) {
                this._needsUpdate = true;
                this._template = value;
            }
        }
        /**
         * Measures the combined size (width for horizontal orientation, height for vertical) of all items
         * in the specified range. Throws an error if the range includes items that are not currently
         * rendered.
         * @param {?} range
         * @param {?} orientation
         * @return {?}
         */
        measureRangeSize(range, orientation) {
            if (range.start >= range.end) {
                return 0;
            }
            if (range.start < this._renderedRange.start || range.end > this._renderedRange.end) {
                throw Error(`Error: attempted to measure an item that isn't rendered.`);
            }
            // The index into the list of rendered views for the first item in the range.
            /** @type {?} */
            const renderedStartIndex = range.start - this._renderedRange.start;
            // The length of the range we're measuring.
            /** @type {?} */
            const rangeLen = range.end - range.start;
            // Loop over all root nodes for all items in the range and sum up their size.
            /** @type {?} */
            let totalSize = 0;
            /** @type {?} */
            let i = rangeLen;
            while (i--) {
                /** @type {?} */
                const view = (/** @type {?} */ (this._viewContainerRef.get(i + renderedStartIndex)));
                /** @type {?} */
                let j = view ? view.rootNodes.length : 0;
                while (j--) {
                    totalSize += getSize(orientation, (/** @type {?} */ (view)).rootNodes[j]);
                }
            }
            return totalSize;
        }
        /**
         * @return {?}
         */
        ngDoCheck() {
            if (this._differ && this._needsUpdate) {
                // TODO(mmalerba): We should differentiate needs update due to scrolling and a new portion of
                // this list being rendered (can use simpler algorithm) vs needs update due to data actually
                // changing (need to do this diff).
                /** @type {?} */
                const changes = this._differ.diff(this._renderedItems);
                if (!changes) {
                    this._updateContext();
                }
                else {
                    this._applyChanges(changes);
                }
                this._needsUpdate = false;
            }
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this._viewport.detach();
            this._dataSourceChanges.next();
            this._dataSourceChanges.complete();
            this.viewChange.complete();
            this._destroyed.next();
            this._destroyed.complete();
            for (let view of this._templateCache) {
                view.destroy();
            }
        }
        /**
         * React to scroll state changes in the viewport.
         * @private
         * @return {?}
         */
        _onRenderedDataChange() {
            if (!this._renderedRange) {
                return;
            }
            this._renderedItems = this._data.slice(this._renderedRange.start, this._renderedRange.end);
            if (!this._differ) {
                this._differ = this._differs.find(this._renderedItems).create(this.cdkVirtualForTrackBy);
            }
            this._needsUpdate = true;
        }
        /**
         * Swap out one `DataSource` for another.
         * @private
         * @param {?} oldDs
         * @param {?} newDs
         * @return {?}
         */
        _changeDataSource(oldDs, newDs) {
            if (oldDs) {
                oldDs.disconnect(this);
            }
            this._needsUpdate = true;
            return newDs ? newDs.connect(this) : of();
        }
        /**
         * Update the `CdkVirtualForOfContext` for all views.
         * @private
         * @return {?}
         */
        _updateContext() {
            /** @type {?} */
            const count = this._data.length;
            /** @type {?} */
            let i = this._viewContainerRef.length;
            while (i--) {
                /** @type {?} */
                let view = (/** @type {?} */ (this._viewContainerRef.get(i)));
                view.context.index = this._renderedRange.start + i;
                view.context.count = count;
                this._updateComputedContextProperties(view.context);
                view.detectChanges();
            }
        }
        /**
         * Apply changes to the DOM.
         * @private
         * @param {?} changes
         * @return {?}
         */
        _applyChanges(changes) {
            // Rearrange the views to put them in the right location.
            changes.forEachOperation((/**
             * @param {?} record
             * @param {?} adjustedPreviousIndex
             * @param {?} currentIndex
             * @return {?}
             */
            (record, adjustedPreviousIndex, currentIndex) => {
                if (record.previousIndex == null) { // Item added.
                    // Item added.
                    /** @type {?} */
                    const view = this._insertViewForNewItem((/** @type {?} */ (currentIndex)));
                    view.context.$implicit = record.item;
                }
                else if (currentIndex == null) { // Item removed.
                    this._cacheView(this._detachView((/** @type {?} */ (adjustedPreviousIndex))));
                }
                else { // Item moved.
                    // Item moved.
                    /** @type {?} */
                    const view = (/** @type {?} */ (this._viewContainerRef.get((/** @type {?} */ (adjustedPreviousIndex)))));
                    this._viewContainerRef.move(view, currentIndex);
                    view.context.$implicit = record.item;
                }
            }));
            // Update $implicit for any items that had an identity change.
            changes.forEachIdentityChange((/**
             * @param {?} record
             * @return {?}
             */
            (record) => {
                /** @type {?} */
                const view = (/** @type {?} */ (this._viewContainerRef.get((/** @type {?} */ (record.currentIndex)))));
                view.context.$implicit = record.item;
            }));
            // Update the context variables on all items.
            /** @type {?} */
            const count = this._data.length;
            /** @type {?} */
            let i = this._viewContainerRef.length;
            while (i--) {
                /** @type {?} */
                const view = (/** @type {?} */ (this._viewContainerRef.get(i)));
                view.context.index = this._renderedRange.start + i;
                view.context.count = count;
                this._updateComputedContextProperties(view.context);
            }
        }
        /**
         * Cache the given detached view.
         * @private
         * @param {?} view
         * @return {?}
         */
        _cacheView(view) {
            if (this._templateCache.length < this.cdkVirtualForTemplateCacheSize) {
                this._templateCache.push(view);
            }
            else {
                /** @type {?} */
                const index = this._viewContainerRef.indexOf(view);
                // It's very unlikely that the index will ever be -1, but just in case,
                // destroy the view on its own, otherwise destroy it through the
                // container to ensure that all the references are removed.
                if (index === -1) {
                    view.destroy();
                }
                else {
                    this._viewContainerRef.remove(index);
                }
            }
        }
        /**
         * Inserts a view for a new item, either from the cache or by creating a new one.
         * @private
         * @param {?} index
         * @return {?}
         */
        _insertViewForNewItem(index) {
            return this._insertViewFromCache(index) || this._createEmbeddedViewAt(index);
        }
        /**
         * Update the computed properties on the `CdkVirtualForOfContext`.
         * @private
         * @param {?} context
         * @return {?}
         */
        _updateComputedContextProperties(context) {
            context.first = context.index === 0;
            context.last = context.index === context.count - 1;
            context.even = context.index % 2 === 0;
            context.odd = !context.even;
        }
        /**
         * Creates a new embedded view and moves it to the given index
         * @private
         * @param {?} index
         * @return {?}
         */
        _createEmbeddedViewAt(index) {
            // Note that it's important that we insert the item directly at the proper index,
            // rather than inserting it and the moving it in place, because if there's a directive
            // on the same node that injects the `ViewContainerRef`, Angular will insert another
            // comment node which can throw off the move when it's being repeated for all items.
            return this._viewContainerRef.createEmbeddedView(this._template, {
                $implicit: (/** @type {?} */ (null)),
                // It's guaranteed that the iterable is not "undefined" or "null" because we only
                // generate views for elements if the "cdkVirtualForOf" iterable has elements.
                cdkVirtualForOf: (/** @type {?} */ (this._cdkVirtualForOf)),
                index: -1,
                count: -1,
                first: false,
                last: false,
                odd: false,
                even: false
            }, index);
        }
        /**
         * Inserts a recycled view from the cache at the given index.
         * @private
         * @param {?} index
         * @return {?}
         */
        _insertViewFromCache(index) {
            /** @type {?} */
            const cachedView = this._templateCache.pop();
            if (cachedView) {
                this._viewContainerRef.insert(cachedView, index);
            }
            return cachedView || null;
        }
        /**
         * Detaches the embedded view at the given index.
         * @private
         * @param {?} index
         * @return {?}
         */
        _detachView(index) {
            return (/** @type {?} */ (this._viewContainerRef.detach(index)));
        }
    }
    CdkVirtualForOf.ɵfac = function CdkVirtualForOf_Factory(t) { return new (t || CdkVirtualForOf)(ɵɵdirectiveInject(ViewContainerRef), ɵɵdirectiveInject(TemplateRef), ɵɵdirectiveInject(IterableDiffers), ɵɵdirectiveInject(CdkVirtualScrollViewport, 4), ɵɵdirectiveInject(NgZone)); };
    CdkVirtualForOf.ɵdir = ɵɵdefineDirective({ type: CdkVirtualForOf, selectors: [["", "cdkVirtualFor", "", "cdkVirtualForOf", ""]], inputs: { cdkVirtualForTemplateCacheSize: "cdkVirtualForTemplateCacheSize", cdkVirtualForOf: "cdkVirtualForOf", cdkVirtualForTrackBy: "cdkVirtualForTrackBy", cdkVirtualForTemplate: "cdkVirtualForTemplate" } });
    /** @nocollapse */
    CdkVirtualForOf.ctorParameters = () => [
        { type: ViewContainerRef },
        { type: TemplateRef },
        { type: IterableDiffers },
        { type: CdkVirtualScrollViewport, decorators: [{ type: SkipSelf }] },
        { type: NgZone }
    ];
    CdkVirtualForOf.propDecorators = {
        cdkVirtualForOf: [{ type: Input }],
        cdkVirtualForTrackBy: [{ type: Input }],
        cdkVirtualForTemplate: [{ type: Input }],
        cdkVirtualForTemplateCacheSize: [{ type: Input }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/scrolling/scrolling-module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class CdkScrollableModule {
    }
    CdkScrollableModule.ɵmod = ɵɵdefineNgModule({ type: CdkScrollableModule });
    CdkScrollableModule.ɵinj = ɵɵdefineInjector({ factory: function CdkScrollableModule_Factory(t) { return new (t || CdkScrollableModule)(); } });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(CdkScrollableModule, { declarations: [CdkScrollable], exports: [CdkScrollable] }); })();
    class ScrollingModule {
    }
    ScrollingModule.ɵmod = ɵɵdefineNgModule({ type: ScrollingModule });
    ScrollingModule.ɵinj = ɵɵdefineInjector({ factory: function ScrollingModule_Factory(t) { return new (t || ScrollingModule)(); }, imports: [[
                BidiModule,
                PlatformModule,
                CdkScrollableModule
            ],
            BidiModule,
            CdkScrollableModule] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(ScrollingModule, { declarations: function () { return [CdkFixedSizeVirtualScroll,
            CdkVirtualForOf,
            CdkVirtualScrollViewport]; }, imports: function () { return [BidiModule,
            PlatformModule,
            CdkScrollableModule]; }, exports: function () { return [BidiModule,
            CdkScrollableModule,
            CdkFixedSizeVirtualScroll,
            CdkVirtualForOf,
            CdkVirtualScrollViewport]; } }); })();

    function MatTab_ng_template_0_Template(rf, ctx) { if (rf & 1) {
        ɵɵprojection(0);
    } }
    const _c0$3 = ["*"];
    function MatTabBody_ng_template_2_Template(rf, ctx) { }
    const _c1$3 = function (a0) { return { animationDuration: a0 }; };
    const _c2$1 = function (a0, a1) { return { value: a0, params: a1 }; };
    const _c3 = ["tabBodyWrapper"];
    const _c4 = ["tabHeader"];
    function MatTabGroup_div_2_ng_template_2_ng_template_0_Template(rf, ctx) { }
    function MatTabGroup_div_2_ng_template_2_Template(rf, ctx) { if (rf & 1) {
        ɵɵtemplate(0, MatTabGroup_div_2_ng_template_2_ng_template_0_Template, 0, 0, "ng-template", 9);
    } if (rf & 2) {
        const tab_r4 = ɵɵnextContext().$implicit;
        ɵɵproperty("cdkPortalOutlet", tab_r4.templateLabel);
    } }
    function MatTabGroup_div_2_ng_template_3_Template(rf, ctx) { if (rf & 1) {
        ɵɵtext(0);
    } if (rf & 2) {
        const tab_r4 = ɵɵnextContext().$implicit;
        ɵɵtextInterpolate(tab_r4.textLabel);
    } }
    function MatTabGroup_div_2_Template(rf, ctx) { if (rf & 1) {
        const _r12 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "div", 6);
        ɵɵlistener("click", function MatTabGroup_div_2_Template_div_click_0_listener() { ɵɵrestoreView(_r12); const tab_r4 = ctx.$implicit; const i_r5 = ctx.index; const ctx_r11 = ɵɵnextContext(); const _r0 = ɵɵreference(1); return ctx_r11._handleClick(tab_r4, _r0, i_r5); });
        ɵɵelementStart(1, "div", 7);
        ɵɵtemplate(2, MatTabGroup_div_2_ng_template_2_Template, 1, 1, "ng-template", 8);
        ɵɵtemplate(3, MatTabGroup_div_2_ng_template_3_Template, 1, 1, "ng-template", 8);
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const tab_r4 = ctx.$implicit;
        const i_r5 = ctx.index;
        const ctx_r1 = ɵɵnextContext();
        ɵɵclassProp("mat-tab-label-active", ctx_r1.selectedIndex == i_r5);
        ɵɵproperty("id", ctx_r1._getTabLabelId(i_r5))("disabled", tab_r4.disabled)("matRippleDisabled", tab_r4.disabled || ctx_r1.disableRipple);
        ɵɵattribute("tabIndex", ctx_r1._getTabIndex(tab_r4, i_r5))("aria-posinset", i_r5 + 1)("aria-setsize", ctx_r1._tabs.length)("aria-controls", ctx_r1._getTabContentId(i_r5))("aria-selected", ctx_r1.selectedIndex == i_r5)("aria-label", tab_r4.ariaLabel || null)("aria-labelledby", !tab_r4.ariaLabel && tab_r4.ariaLabelledby ? tab_r4.ariaLabelledby : null);
        ɵɵadvance(2);
        ɵɵproperty("ngIf", tab_r4.templateLabel);
        ɵɵadvance(1);
        ɵɵproperty("ngIf", !tab_r4.templateLabel);
    } }
    function MatTabGroup_mat_tab_body_5_Template(rf, ctx) { if (rf & 1) {
        const _r16 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "mat-tab-body", 10);
        ɵɵlistener("_onCentered", function MatTabGroup_mat_tab_body_5_Template_mat_tab_body__onCentered_0_listener() { ɵɵrestoreView(_r16); const ctx_r15 = ɵɵnextContext(); return ctx_r15._removeTabBodyWrapperHeight(); })("_onCentering", function MatTabGroup_mat_tab_body_5_Template_mat_tab_body__onCentering_0_listener($event) { ɵɵrestoreView(_r16); const ctx_r17 = ɵɵnextContext(); return ctx_r17._setTabBodyWrapperHeight($event); });
        ɵɵelementEnd();
    } if (rf & 2) {
        const tab_r13 = ctx.$implicit;
        const i_r14 = ctx.index;
        const ctx_r3 = ɵɵnextContext();
        ɵɵclassProp("mat-tab-body-active", ctx_r3.selectedIndex == i_r14);
        ɵɵproperty("id", ctx_r3._getTabContentId(i_r14))("content", tab_r13.content)("position", tab_r13.position)("origin", tab_r13.origin)("animationDuration", ctx_r3.animationDuration);
        ɵɵattribute("aria-labelledby", ctx_r3._getTabLabelId(i_r14));
    } }
    const _c5 = ["tabListContainer"];
    const _c6 = ["tabList"];
    const _c7 = ["nextPaginator"];
    const _c8 = ["previousPaginator"];
    const _c9 = ["mat-tab-nav-bar", ""];
    /**
     * Injection token for the MatInkBar's Positioner.
     * @type {?}
     */
    const _MAT_INK_BAR_POSITIONER = new InjectionToken('MatInkBarPositioner', {
        providedIn: 'root',
        factory: _MAT_INK_BAR_POSITIONER_FACTORY
    });
    /**
     * The default positioner function for the MatInkBar.
     * \@docs-private
     * @return {?}
     */
    function _MAT_INK_BAR_POSITIONER_FACTORY() {
        /** @type {?} */
        const method = (/**
         * @param {?} element
         * @return {?}
         */
        (element) => ({
            left: element ? (element.offsetLeft || 0) + 'px' : '0',
            width: element ? (element.offsetWidth || 0) + 'px' : '0',
        }));
        return method;
    }
    /**
     * The ink-bar is used to display and animate the line underneath the current active tab label.
     * \@docs-private
     */
    class MatInkBar {
        /**
         * @param {?} _elementRef
         * @param {?} _ngZone
         * @param {?} _inkBarPositioner
         * @param {?=} _animationMode
         */
        constructor(_elementRef, _ngZone, _inkBarPositioner, _animationMode) {
            this._elementRef = _elementRef;
            this._ngZone = _ngZone;
            this._inkBarPositioner = _inkBarPositioner;
            this._animationMode = _animationMode;
        }
        /**
         * Calculates the styles from the provided element in order to align the ink-bar to that element.
         * Shows the ink bar if previously set as hidden.
         * @param {?} element
         * @return {?}
         */
        alignToElement(element) {
            this.show();
            if (typeof requestAnimationFrame !== 'undefined') {
                this._ngZone.runOutsideAngular((/**
                 * @return {?}
                 */
                () => {
                    requestAnimationFrame((/**
                     * @return {?}
                     */
                    () => this._setStyles(element)));
                }));
            }
            else {
                this._setStyles(element);
            }
        }
        /**
         * Shows the ink bar.
         * @return {?}
         */
        show() {
            this._elementRef.nativeElement.style.visibility = 'visible';
        }
        /**
         * Hides the ink bar.
         * @return {?}
         */
        hide() {
            this._elementRef.nativeElement.style.visibility = 'hidden';
        }
        /**
         * Sets the proper styles to the ink bar element.
         * @private
         * @param {?} element
         * @return {?}
         */
        _setStyles(element) {
            /** @type {?} */
            const positions = this._inkBarPositioner(element);
            /** @type {?} */
            const inkBar = this._elementRef.nativeElement;
            inkBar.style.left = positions.left;
            inkBar.style.width = positions.width;
        }
    }
    MatInkBar.ɵfac = function MatInkBar_Factory(t) { return new (t || MatInkBar)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(NgZone), ɵɵdirectiveInject(_MAT_INK_BAR_POSITIONER), ɵɵdirectiveInject(ANIMATION_MODULE_TYPE, 8)); };
    MatInkBar.ɵdir = ɵɵdefineDirective({ type: MatInkBar, selectors: [["mat-ink-bar"]], hostAttrs: [1, "mat-ink-bar"], hostVars: 2, hostBindings: function MatInkBar_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵclassProp("_mat-animation-noopable", ctx._animationMode === "NoopAnimations");
        } } });
    /** @nocollapse */
    MatInkBar.ctorParameters = () => [
        { type: ElementRef },
        { type: NgZone },
        { type: undefined, decorators: [{ type: Inject, args: [_MAT_INK_BAR_POSITIONER,] }] },
        { type: String, decorators: [{ type: Optional }, { type: Inject, args: [ANIMATION_MODULE_TYPE,] }] }
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/tabs/tab-content.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Decorates the `ng-template` tags and reads out the template from it.
     */
    class MatTabContent {
        /**
         * @param {?} template
         */
        constructor(template) {
            this.template = template;
        }
    }
    MatTabContent.ɵfac = function MatTabContent_Factory(t) { return new (t || MatTabContent)(ɵɵdirectiveInject(TemplateRef)); };
    MatTabContent.ɵdir = ɵɵdefineDirective({ type: MatTabContent, selectors: [["", "matTabContent", ""]] });
    /** @nocollapse */
    MatTabContent.ctorParameters = () => [
        { type: TemplateRef }
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/tabs/tab-label.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Used to flag tab labels for use with the portal directive
     */
    class MatTabLabel extends CdkPortal {
    }
    MatTabLabel.ɵfac = function MatTabLabel_Factory(t) { return ɵMatTabLabel_BaseFactory(t || MatTabLabel); };
    MatTabLabel.ɵdir = ɵɵdefineDirective({ type: MatTabLabel, selectors: [["", "mat-tab-label", ""], ["", "matTabLabel", ""]], features: [ɵɵInheritDefinitionFeature] });
    const ɵMatTabLabel_BaseFactory = ɵɵgetInheritedFactory(MatTabLabel);

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/tabs/tab.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    // Boilerplate for applying mixins to MatTab.
    /**
     * \@docs-private
     */
    class MatTabBase {
    }
    /** @type {?} */
    const _MatTabMixinBase = mixinDisabled(MatTabBase);
    /**
     * Used to provide a tab group to a tab without causing a circular dependency.
     * \@docs-private
     * @type {?}
     */
    const MAT_TAB_GROUP = new InjectionToken('MAT_TAB_GROUP');
    class MatTab extends _MatTabMixinBase {
        /**
         * @param {?} _viewContainerRef
         * @param {?=} _closestTabGroup
         */
        constructor(_viewContainerRef, _closestTabGroup) {
            super();
            this._viewContainerRef = _viewContainerRef;
            this._closestTabGroup = _closestTabGroup;
            /**
             * Plain text label for the tab, used when there is no template label.
             */
            this.textLabel = '';
            /**
             * Portal that will be the hosted content of the tab
             */
            this._contentPortal = null;
            /**
             * Emits whenever the internal state of the tab changes.
             */
            this._stateChanges = new Subject();
            /**
             * The relatively indexed position where 0 represents the center, negative is left, and positive
             * represents the right.
             */
            this.position = null;
            /**
             * The initial relatively index origin of the tab if it was created and selected after there
             * was already a selected tab. Provides context of what position the tab should originate from.
             */
            this.origin = null;
            /**
             * Whether the tab is currently active.
             */
            this.isActive = false;
        }
        /**
         * Content for the tab label given by `<ng-template mat-tab-label>`.
         * @return {?}
         */
        get templateLabel() { return this._templateLabel; }
        /**
         * @param {?} value
         * @return {?}
         */
        set templateLabel(value) {
            // Only update the templateLabel via query if there is actually
            // a MatTabLabel found. This works around an issue where a user may have
            // manually set `templateLabel` during creation mode, which would then get clobbered
            // by `undefined` when this query resolves.
            if (value) {
                this._templateLabel = value;
            }
        }
        /**
         * \@docs-private
         * @return {?}
         */
        get content() {
            return this._contentPortal;
        }
        /**
         * @param {?} changes
         * @return {?}
         */
        ngOnChanges(changes) {
            if (changes.hasOwnProperty('textLabel') || changes.hasOwnProperty('disabled')) {
                this._stateChanges.next();
            }
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this._stateChanges.complete();
        }
        /**
         * @return {?}
         */
        ngOnInit() {
            this._contentPortal = new TemplatePortal(this._explicitContent || this._implicitContent, this._viewContainerRef);
        }
    }
    MatTab.ɵfac = function MatTab_Factory(t) { return new (t || MatTab)(ɵɵdirectiveInject(ViewContainerRef), ɵɵdirectiveInject(MAT_TAB_GROUP, 8)); };
    MatTab.ɵcmp = ɵɵdefineComponent({ type: MatTab, selectors: [["mat-tab"]], contentQueries: function MatTab_ContentQueries(rf, ctx, dirIndex) { if (rf & 1) {
            ɵɵcontentQuery(dirIndex, MatTabLabel, true);
            ɵɵstaticContentQuery(dirIndex, MatTabContent, true, TemplateRef);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.templateLabel = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._explicitContent = _t.first);
        } }, viewQuery: function MatTab_Query(rf, ctx) { if (rf & 1) {
            ɵɵstaticViewQuery(TemplateRef, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._implicitContent = _t.first);
        } }, inputs: { disabled: "disabled", textLabel: ["label", "textLabel"], ariaLabel: ["aria-label", "ariaLabel"], ariaLabelledby: ["aria-labelledby", "ariaLabelledby"] }, exportAs: ["matTab"], features: [ɵɵInheritDefinitionFeature, ɵɵNgOnChangesFeature], ngContentSelectors: _c0$3, decls: 1, vars: 0, template: function MatTab_Template(rf, ctx) { if (rf & 1) {
            ɵɵprojectionDef();
            ɵɵtemplate(0, MatTab_ng_template_0_Template, 1, 0, "ng-template");
        } }, encapsulation: 2 });
    /** @nocollapse */
    MatTab.ctorParameters = () => [
        { type: ViewContainerRef },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [MAT_TAB_GROUP,] }] }
    ];
    MatTab.propDecorators = {
        templateLabel: [{ type: ContentChild, args: [MatTabLabel,] }],
        _explicitContent: [{ type: ContentChild, args: [MatTabContent, { read: TemplateRef, static: true },] }],
        _implicitContent: [{ type: ViewChild, args: [TemplateRef, { static: true },] }],
        textLabel: [{ type: Input, args: ['label',] }],
        ariaLabel: [{ type: Input, args: ['aria-label',] }],
        ariaLabelledby: [{ type: Input, args: ['aria-labelledby',] }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/tabs/tabs-animations.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Animations used by the Material tabs.
     * \@docs-private
     * @type {?}
     */
    const matTabsAnimations = {
        /**
         * Animation translates a tab along the X axis.
         */
        translateTab: trigger('translateTab', [
            // Note: transitions to `none` instead of 0, because some browsers might blur the content.
            state('center, void, left-origin-center, right-origin-center', style({ transform: 'none' })),
            // If the tab is either on the left or right, we additionally add a `min-height` of 1px
            // in order to ensure that the element has a height before its state changes. This is
            // necessary because Chrome does seem to skip the transition in RTL mode if the element does
            // not have a static height and is not rendered. See related issue: #9465
            state('left', style({ transform: 'translate3d(-100%, 0, 0)', minHeight: '1px' })),
            state('right', style({ transform: 'translate3d(100%, 0, 0)', minHeight: '1px' })),
            transition('* => left, * => right, left => center, right => center', animate('{{animationDuration}} cubic-bezier(0.35, 0, 0.25, 1)')),
            transition('void => left-origin-center', [
                style({ transform: 'translate3d(-100%, 0, 0)' }),
                animate('{{animationDuration}} cubic-bezier(0.35, 0, 0.25, 1)')
            ]),
            transition('void => right-origin-center', [
                style({ transform: 'translate3d(100%, 0, 0)' }),
                animate('{{animationDuration}} cubic-bezier(0.35, 0, 0.25, 1)')
            ])
        ])
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/tabs/tab-body.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * The portal host directive for the contents of the tab.
     * \@docs-private
     */
    class MatTabBodyPortal extends CdkPortalOutlet {
        /**
         * @param {?} componentFactoryResolver
         * @param {?} viewContainerRef
         * @param {?} _host
         * @param {?=} _document
         */
        constructor(componentFactoryResolver, viewContainerRef, _host, 
        /**
         * @deprecated `_document` parameter to be made required.
         * @breaking-change 9.0.0
         */
        _document) {
            super(componentFactoryResolver, viewContainerRef, _document);
            this._host = _host;
            /**
             * Subscription to events for when the tab body begins centering.
             */
            this._centeringSub = Subscription.EMPTY;
            /**
             * Subscription to events for when the tab body finishes leaving from center position.
             */
            this._leavingSub = Subscription.EMPTY;
        }
        /**
         * Set initial visibility or set up subscription for changing visibility.
         * @return {?}
         */
        ngOnInit() {
            super.ngOnInit();
            this._centeringSub = this._host._beforeCentering
                .pipe(startWith(this._host._isCenterPosition(this._host._position)))
                .subscribe((/**
             * @param {?} isCentering
             * @return {?}
             */
            (isCentering) => {
                if (isCentering && !this.hasAttached()) {
                    this.attach(this._host._content);
                }
            }));
            this._leavingSub = this._host._afterLeavingCenter.subscribe((/**
             * @return {?}
             */
            () => {
                this.detach();
            }));
        }
        /**
         * Clean up centering subscription.
         * @return {?}
         */
        ngOnDestroy() {
            super.ngOnDestroy();
            this._centeringSub.unsubscribe();
            this._leavingSub.unsubscribe();
        }
    }
    MatTabBodyPortal.ɵfac = function MatTabBodyPortal_Factory(t) { return new (t || MatTabBodyPortal)(ɵɵdirectiveInject(ComponentFactoryResolver), ɵɵdirectiveInject(ViewContainerRef), ɵɵdirectiveInject(forwardRef(( /**
                     * @return {?}
                     */() => MatTabBody))), ɵɵdirectiveInject(DOCUMENT$1)); };
    MatTabBodyPortal.ɵdir = ɵɵdefineDirective({ type: MatTabBodyPortal, selectors: [["", "matTabBodyHost", ""]], features: [ɵɵInheritDefinitionFeature] });
    /** @nocollapse */
    MatTabBodyPortal.ctorParameters = () => [
        { type: ComponentFactoryResolver },
        { type: ViewContainerRef },
        { type: MatTabBody, decorators: [{ type: Inject, args: [forwardRef((/**
                         * @return {?}
                         */
                        () => MatTabBody)),] }] },
        { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT$1,] }] }
    ];
    /**
     * Base class with all of the `MatTabBody` functionality.
     * \@docs-private
     * @abstract
     */
    // tslint:disable-next-line:class-name
    class _MatTabBodyBase {
        /**
         * @param {?} _elementRef
         * @param {?} _dir
         * @param {?} changeDetectorRef
         */
        constructor(_elementRef, _dir, changeDetectorRef) {
            this._elementRef = _elementRef;
            this._dir = _dir;
            /**
             * Subscription to the directionality change observable.
             */
            this._dirChangeSubscription = Subscription.EMPTY;
            /**
             * Emits when an animation on the tab is complete.
             */
            this._translateTabComplete = new Subject();
            /**
             * Event emitted when the tab begins to animate towards the center as the active tab.
             */
            this._onCentering = new EventEmitter();
            /**
             * Event emitted before the centering of the tab begins.
             */
            this._beforeCentering = new EventEmitter();
            /**
             * Event emitted before the centering of the tab begins.
             */
            this._afterLeavingCenter = new EventEmitter();
            /**
             * Event emitted when the tab completes its animation towards the center.
             */
            this._onCentered = new EventEmitter(true);
            // Note that the default value will always be overwritten by `MatTabBody`, but we need one
            // anyway to prevent the animations module from throwing an error if the body is used on its own.
            /**
             * Duration for the tab's animation.
             */
            this.animationDuration = '500ms';
            if (_dir) {
                this._dirChangeSubscription = _dir.change.subscribe((/**
                 * @param {?} dir
                 * @return {?}
                 */
                (dir) => {
                    this._computePositionAnimationState(dir);
                    changeDetectorRef.markForCheck();
                }));
            }
            // Ensure that we get unique animation events, because the `.done` callback can get
            // invoked twice in some browsers. See https://github.com/angular/angular/issues/24084.
            this._translateTabComplete.pipe(distinctUntilChanged((/**
             * @param {?} x
             * @param {?} y
             * @return {?}
             */
            (x, y) => {
                return x.fromState === y.fromState && x.toState === y.toState;
            }))).subscribe((/**
             * @param {?} event
             * @return {?}
             */
            event => {
                // If the transition to the center is complete, emit an event.
                if (this._isCenterPosition(event.toState) && this._isCenterPosition(this._position)) {
                    this._onCentered.emit();
                }
                if (this._isCenterPosition(event.fromState) && !this._isCenterPosition(this._position)) {
                    this._afterLeavingCenter.emit();
                }
            }));
        }
        /**
         * The shifted index position of the tab body, where zero represents the active center tab.
         * @param {?} position
         * @return {?}
         */
        set position(position) {
            this._positionIndex = position;
            this._computePositionAnimationState();
        }
        /**
         * After initialized, check if the content is centered and has an origin. If so, set the
         * special position states that transition the tab from the left or right before centering.
         * @return {?}
         */
        ngOnInit() {
            if (this._position == 'center' && this.origin != null) {
                this._position = this._computePositionFromOrigin(this.origin);
            }
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this._dirChangeSubscription.unsubscribe();
            this._translateTabComplete.complete();
        }
        /**
         * @param {?} event
         * @return {?}
         */
        _onTranslateTabStarted(event) {
            /** @type {?} */
            const isCentering = this._isCenterPosition(event.toState);
            this._beforeCentering.emit(isCentering);
            if (isCentering) {
                this._onCentering.emit(this._elementRef.nativeElement.clientHeight);
            }
        }
        /**
         * The text direction of the containing app.
         * @return {?}
         */
        _getLayoutDirection() {
            return this._dir && this._dir.value === 'rtl' ? 'rtl' : 'ltr';
        }
        /**
         * Whether the provided position state is considered center, regardless of origin.
         * @param {?} position
         * @return {?}
         */
        _isCenterPosition(position) {
            return position == 'center' ||
                position == 'left-origin-center' ||
                position == 'right-origin-center';
        }
        /**
         * Computes the position state that will be used for the tab-body animation trigger.
         * @private
         * @param {?=} dir
         * @return {?}
         */
        _computePositionAnimationState(dir = this._getLayoutDirection()) {
            if (this._positionIndex < 0) {
                this._position = dir == 'ltr' ? 'left' : 'right';
            }
            else if (this._positionIndex > 0) {
                this._position = dir == 'ltr' ? 'right' : 'left';
            }
            else {
                this._position = 'center';
            }
        }
        /**
         * Computes the position state based on the specified origin position. This is used if the
         * tab is becoming visible immediately after creation.
         * @private
         * @param {?} origin
         * @return {?}
         */
        _computePositionFromOrigin(origin) {
            /** @type {?} */
            const dir = this._getLayoutDirection();
            if ((dir == 'ltr' && origin <= 0) || (dir == 'rtl' && origin > 0)) {
                return 'left-origin-center';
            }
            return 'right-origin-center';
        }
    }
    _MatTabBodyBase.ɵfac = function _MatTabBodyBase_Factory(t) { return new (t || _MatTabBodyBase)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(Directionality, 8), ɵɵdirectiveInject(ChangeDetectorRef)); };
    _MatTabBodyBase.ɵdir = ɵɵdefineDirective({ type: _MatTabBodyBase, inputs: { animationDuration: "animationDuration", position: "position", _content: ["content", "_content"], origin: "origin" }, outputs: { _onCentering: "_onCentering", _beforeCentering: "_beforeCentering", _afterLeavingCenter: "_afterLeavingCenter", _onCentered: "_onCentered" } });
    /** @nocollapse */
    _MatTabBodyBase.ctorParameters = () => [
        { type: ElementRef },
        { type: Directionality, decorators: [{ type: Optional }] },
        { type: ChangeDetectorRef }
    ];
    _MatTabBodyBase.propDecorators = {
        _onCentering: [{ type: Output }],
        _beforeCentering: [{ type: Output }],
        _afterLeavingCenter: [{ type: Output }],
        _onCentered: [{ type: Output }],
        _content: [{ type: Input, args: ['content',] }],
        origin: [{ type: Input }],
        animationDuration: [{ type: Input }],
        position: [{ type: Input }]
    };
    /**
     * Wrapper for the contents of a tab.
     * \@docs-private
     */
    class MatTabBody extends _MatTabBodyBase {
        /**
         * @param {?} elementRef
         * @param {?} dir
         * @param {?} changeDetectorRef
         */
        constructor(elementRef, dir, changeDetectorRef) {
            super(elementRef, dir, changeDetectorRef);
        }
    }
    MatTabBody.ɵfac = function MatTabBody_Factory(t) { return new (t || MatTabBody)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(Directionality, 8), ɵɵdirectiveInject(ChangeDetectorRef)); };
    MatTabBody.ɵcmp = ɵɵdefineComponent({ type: MatTabBody, selectors: [["mat-tab-body"]], viewQuery: function MatTabBody_Query(rf, ctx) { if (rf & 1) {
            ɵɵviewQuery(PortalHostDirective, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._portalHost = _t.first);
        } }, hostAttrs: [1, "mat-tab-body"], features: [ɵɵInheritDefinitionFeature], decls: 3, vars: 6, consts: [[1, "mat-tab-body-content"], ["content", ""], ["matTabBodyHost", ""]], template: function MatTabBody_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "div", 0, 1);
            ɵɵlistener("@translateTab.start", function MatTabBody_Template_div_animation_translateTab_start_0_listener($event) { return ctx._onTranslateTabStarted($event); })("@translateTab.done", function MatTabBody_Template_div_animation_translateTab_done_0_listener($event) { return ctx._translateTabComplete.next($event); });
            ɵɵtemplate(2, MatTabBody_ng_template_2_Template, 0, 0, "ng-template", 2);
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵproperty("@translateTab", ɵɵpureFunction2(3, _c2$1, ctx._position, ɵɵpureFunction1(1, _c1$3, ctx.animationDuration)));
        } }, directives: [MatTabBodyPortal], styles: [".mat-tab-body-content{height:100%;overflow:auto}.mat-tab-group-dynamic-height .mat-tab-body-content{overflow:hidden}\n"], encapsulation: 2, data: { animation: [matTabsAnimations.translateTab] } });
    /** @nocollapse */
    MatTabBody.ctorParameters = () => [
        { type: ElementRef },
        { type: Directionality, decorators: [{ type: Optional }] },
        { type: ChangeDetectorRef }
    ];
    MatTabBody.propDecorators = {
        _portalHost: [{ type: ViewChild, args: [PortalHostDirective,] }]
    };
    /**
     * Injection token that can be used to provide the default options the tabs module.
     * @type {?}
     */
    const MAT_TABS_CONFIG = new InjectionToken('MAT_TABS_CONFIG');

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/tabs/tab-group.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Used to generate unique ID's for each tab component
     * @type {?}
     */
    let nextId$1 = 0;
    /**
     * A simple change event emitted on focus or selection changes.
     */
    class MatTabChangeEvent {
    }
    // Boilerplate for applying mixins to MatTabGroup.
    /**
     * \@docs-private
     */
    class MatTabGroupMixinBase {
        /**
         * @param {?} _elementRef
         */
        constructor(_elementRef) {
            this._elementRef = _elementRef;
        }
    }
    /** @type {?} */
    const _MatTabGroupMixinBase = mixinColor(mixinDisableRipple(MatTabGroupMixinBase), 'primary');
    /**
     * Base class with all of the `MatTabGroupBase` functionality.
     * \@docs-private
     * @abstract
     */
    // tslint:disable-next-line:class-name
    class _MatTabGroupBase extends _MatTabGroupMixinBase {
        /**
         * @param {?} elementRef
         * @param {?} _changeDetectorRef
         * @param {?=} defaultConfig
         * @param {?=} _animationMode
         */
        constructor(elementRef, _changeDetectorRef, defaultConfig, _animationMode) {
            super(elementRef);
            this._changeDetectorRef = _changeDetectorRef;
            this._animationMode = _animationMode;
            /**
             * All of the tabs that belong to the group.
             */
            this._tabs = new QueryList();
            /**
             * The tab index that should be selected after the content has been checked.
             */
            this._indexToSelect = 0;
            /**
             * Snapshot of the height of the tab body wrapper before another tab is activated.
             */
            this._tabBodyWrapperHeight = 0;
            /**
             * Subscription to tabs being added/removed.
             */
            this._tabsSubscription = Subscription.EMPTY;
            /**
             * Subscription to changes in the tab labels.
             */
            this._tabLabelSubscription = Subscription.EMPTY;
            this._dynamicHeight = false;
            this._selectedIndex = null;
            /**
             * Position of the tab header.
             */
            this.headerPosition = 'above';
            /**
             * Output to enable support for two-way binding on `[(selectedIndex)]`
             */
            this.selectedIndexChange = new EventEmitter();
            /**
             * Event emitted when focus has changed within a tab group.
             */
            this.focusChange = new EventEmitter();
            /**
             * Event emitted when the body animation has completed
             */
            this.animationDone = new EventEmitter();
            /**
             * Event emitted when the tab selection has changed.
             */
            this.selectedTabChange = new EventEmitter(true);
            this._groupId = nextId$1++;
            this.animationDuration = defaultConfig && defaultConfig.animationDuration ?
                defaultConfig.animationDuration : '500ms';
            this.disablePagination = defaultConfig && defaultConfig.disablePagination != null ?
                defaultConfig.disablePagination : false;
        }
        /**
         * Whether the tab group should grow to the size of the active tab.
         * @return {?}
         */
        get dynamicHeight() { return this._dynamicHeight; }
        /**
         * @param {?} value
         * @return {?}
         */
        set dynamicHeight(value) { this._dynamicHeight = coerceBooleanProperty(value); }
        /**
         * The index of the active tab.
         * @return {?}
         */
        get selectedIndex() { return this._selectedIndex; }
        /**
         * @param {?} value
         * @return {?}
         */
        set selectedIndex(value) {
            this._indexToSelect = coerceNumberProperty(value, null);
        }
        /**
         * Duration for the tab animation. Will be normalized to milliseconds if no units are set.
         * @return {?}
         */
        get animationDuration() { return this._animationDuration; }
        /**
         * @param {?} value
         * @return {?}
         */
        set animationDuration(value) {
            this._animationDuration = /^\d+$/.test(value) ? value + 'ms' : value;
        }
        /**
         * Background color of the tab group.
         * @return {?}
         */
        get backgroundColor() { return this._backgroundColor; }
        /**
         * @param {?} value
         * @return {?}
         */
        set backgroundColor(value) {
            /** @type {?} */
            const nativeElement = this._elementRef.nativeElement;
            nativeElement.classList.remove(`mat-background-${this.backgroundColor}`);
            if (value) {
                nativeElement.classList.add(`mat-background-${value}`);
            }
            this._backgroundColor = value;
        }
        /**
         * After the content is checked, this component knows what tabs have been defined
         * and what the selected index should be. This is where we can know exactly what position
         * each tab should be in according to the new selected index, and additionally we know how
         * a new selected tab should transition in (from the left or right).
         * @return {?}
         */
        ngAfterContentChecked() {
            // Don't clamp the `indexToSelect` immediately in the setter because it can happen that
            // the amount of tabs changes before the actual change detection runs.
            /** @type {?} */
            const indexToSelect = this._indexToSelect = this._clampTabIndex(this._indexToSelect);
            // If there is a change in selected index, emit a change event. Should not trigger if
            // the selected index has not yet been initialized.
            if (this._selectedIndex != indexToSelect) {
                /** @type {?} */
                const isFirstRun = this._selectedIndex == null;
                if (!isFirstRun) {
                    this.selectedTabChange.emit(this._createChangeEvent(indexToSelect));
                }
                // Changing these values after change detection has run
                // since the checked content may contain references to them.
                Promise.resolve().then((/**
                 * @return {?}
                 */
                () => {
                    this._tabs.forEach((/**
                     * @param {?} tab
                     * @param {?} index
                     * @return {?}
                     */
                    (tab, index) => tab.isActive = index === indexToSelect));
                    if (!isFirstRun) {
                        this.selectedIndexChange.emit(indexToSelect);
                    }
                }));
            }
            // Setup the position for each tab and optionally setup an origin on the next selected tab.
            this._tabs.forEach((/**
             * @param {?} tab
             * @param {?} index
             * @return {?}
             */
            (tab, index) => {
                tab.position = index - indexToSelect;
                // If there is already a selected tab, then set up an origin for the next selected tab
                // if it doesn't have one already.
                if (this._selectedIndex != null && tab.position == 0 && !tab.origin) {
                    tab.origin = indexToSelect - this._selectedIndex;
                }
            }));
            if (this._selectedIndex !== indexToSelect) {
                this._selectedIndex = indexToSelect;
                this._changeDetectorRef.markForCheck();
            }
        }
        /**
         * @return {?}
         */
        ngAfterContentInit() {
            this._subscribeToAllTabChanges();
            this._subscribeToTabLabels();
            // Subscribe to changes in the amount of tabs, in order to be
            // able to re-render the content as new tabs are added or removed.
            this._tabsSubscription = this._tabs.changes.subscribe((/**
             * @return {?}
             */
            () => {
                /** @type {?} */
                const indexToSelect = this._clampTabIndex(this._indexToSelect);
                // Maintain the previously-selected tab if a new tab is added or removed and there is no
                // explicit change that selects a different tab.
                if (indexToSelect === this._selectedIndex) {
                    /** @type {?} */
                    const tabs = this._tabs.toArray();
                    for (let i = 0; i < tabs.length; i++) {
                        if (tabs[i].isActive) {
                            // Assign both to the `_indexToSelect` and `_selectedIndex` so we don't fire a changed
                            // event, otherwise the consumer may end up in an infinite loop in some edge cases like
                            // adding a tab within the `selectedIndexChange` event.
                            this._indexToSelect = this._selectedIndex = i;
                            break;
                        }
                    }
                }
                this._changeDetectorRef.markForCheck();
            }));
        }
        /**
         * Listens to changes in all of the tabs.
         * @private
         * @return {?}
         */
        _subscribeToAllTabChanges() {
            // Since we use a query with `descendants: true` to pick up the tabs, we may end up catching
            // some that are inside of nested tab groups. We filter them out manually by checking that
            // the closest group to the tab is the current one.
            this._allTabs.changes
                .pipe(startWith(this._allTabs))
                .subscribe((/**
             * @param {?} tabs
             * @return {?}
             */
            (tabs) => {
                this._tabs.reset(tabs.filter((/**
                 * @param {?} tab
                 * @return {?}
                 */
                tab => {
                    // @breaking-change 10.0.0 Remove null check for `_closestTabGroup`
                    // once it becomes a required parameter in MatTab.
                    return !tab._closestTabGroup || tab._closestTabGroup === this;
                })));
                this._tabs.notifyOnChanges();
            }));
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this._tabs.destroy();
            this._tabsSubscription.unsubscribe();
            this._tabLabelSubscription.unsubscribe();
        }
        /**
         * Re-aligns the ink bar to the selected tab element.
         * @return {?}
         */
        realignInkBar() {
            if (this._tabHeader) {
                this._tabHeader._alignInkBarToSelectedTab();
            }
        }
        /**
         * @param {?} index
         * @return {?}
         */
        _focusChanged(index) {
            this.focusChange.emit(this._createChangeEvent(index));
        }
        /**
         * @private
         * @param {?} index
         * @return {?}
         */
        _createChangeEvent(index) {
            /** @type {?} */
            const event = new MatTabChangeEvent;
            event.index = index;
            if (this._tabs && this._tabs.length) {
                event.tab = this._tabs.toArray()[index];
            }
            return event;
        }
        /**
         * Subscribes to changes in the tab labels. This is needed, because the \@Input for the label is
         * on the MatTab component, whereas the data binding is inside the MatTabGroup. In order for the
         * binding to be updated, we need to subscribe to changes in it and trigger change detection
         * manually.
         * @private
         * @return {?}
         */
        _subscribeToTabLabels() {
            if (this._tabLabelSubscription) {
                this._tabLabelSubscription.unsubscribe();
            }
            this._tabLabelSubscription = merge(...this._tabs.map((/**
             * @param {?} tab
             * @return {?}
             */
            tab => tab._stateChanges)))
                .subscribe((/**
             * @return {?}
             */
            () => this._changeDetectorRef.markForCheck()));
        }
        /**
         * Clamps the given index to the bounds of 0 and the tabs length.
         * @private
         * @param {?} index
         * @return {?}
         */
        _clampTabIndex(index) {
            // Note the `|| 0`, which ensures that values like NaN can't get through
            // and which would otherwise throw the component into an infinite loop
            // (since Math.max(NaN, 0) === NaN).
            return Math.min(this._tabs.length - 1, Math.max(index || 0, 0));
        }
        /**
         * Returns a unique id for each tab label element
         * @param {?} i
         * @return {?}
         */
        _getTabLabelId(i) {
            return `mat-tab-label-${this._groupId}-${i}`;
        }
        /**
         * Returns a unique id for each tab content element
         * @param {?} i
         * @return {?}
         */
        _getTabContentId(i) {
            return `mat-tab-content-${this._groupId}-${i}`;
        }
        /**
         * Sets the height of the body wrapper to the height of the activating tab if dynamic
         * height property is true.
         * @param {?} tabHeight
         * @return {?}
         */
        _setTabBodyWrapperHeight(tabHeight) {
            if (!this._dynamicHeight || !this._tabBodyWrapperHeight) {
                return;
            }
            /** @type {?} */
            const wrapper = this._tabBodyWrapper.nativeElement;
            wrapper.style.height = this._tabBodyWrapperHeight + 'px';
            // This conditional forces the browser to paint the height so that
            // the animation to the new height can have an origin.
            if (this._tabBodyWrapper.nativeElement.offsetHeight) {
                wrapper.style.height = tabHeight + 'px';
            }
        }
        /**
         * Removes the height of the tab body wrapper.
         * @return {?}
         */
        _removeTabBodyWrapperHeight() {
            /** @type {?} */
            const wrapper = this._tabBodyWrapper.nativeElement;
            this._tabBodyWrapperHeight = wrapper.clientHeight;
            wrapper.style.height = '';
            this.animationDone.emit();
        }
        /**
         * Handle click events, setting new selected index if appropriate.
         * @param {?} tab
         * @param {?} tabHeader
         * @param {?} index
         * @return {?}
         */
        _handleClick(tab, tabHeader, index) {
            if (!tab.disabled) {
                this.selectedIndex = tabHeader.focusIndex = index;
            }
        }
        /**
         * Retrieves the tabindex for the tab.
         * @param {?} tab
         * @param {?} idx
         * @return {?}
         */
        _getTabIndex(tab, idx) {
            if (tab.disabled) {
                return null;
            }
            return this.selectedIndex === idx ? 0 : -1;
        }
    }
    _MatTabGroupBase.ɵfac = function _MatTabGroupBase_Factory(t) { return new (t || _MatTabGroupBase)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(ChangeDetectorRef), ɵɵdirectiveInject(MAT_TABS_CONFIG, 8), ɵɵdirectiveInject(ANIMATION_MODULE_TYPE, 8)); };
    _MatTabGroupBase.ɵdir = ɵɵdefineDirective({ type: _MatTabGroupBase, inputs: { headerPosition: "headerPosition", animationDuration: "animationDuration", disablePagination: "disablePagination", dynamicHeight: "dynamicHeight", selectedIndex: "selectedIndex", backgroundColor: "backgroundColor" }, outputs: { selectedIndexChange: "selectedIndexChange", focusChange: "focusChange", animationDone: "animationDone", selectedTabChange: "selectedTabChange" }, features: [ɵɵInheritDefinitionFeature] });
    /** @nocollapse */
    _MatTabGroupBase.ctorParameters = () => [
        { type: ElementRef },
        { type: ChangeDetectorRef },
        { type: undefined, decorators: [{ type: Inject, args: [MAT_TABS_CONFIG,] }, { type: Optional }] },
        { type: String, decorators: [{ type: Optional }, { type: Inject, args: [ANIMATION_MODULE_TYPE,] }] }
    ];
    _MatTabGroupBase.propDecorators = {
        dynamicHeight: [{ type: Input }],
        selectedIndex: [{ type: Input }],
        headerPosition: [{ type: Input }],
        animationDuration: [{ type: Input }],
        disablePagination: [{ type: Input }],
        backgroundColor: [{ type: Input }],
        selectedIndexChange: [{ type: Output }],
        focusChange: [{ type: Output }],
        animationDone: [{ type: Output }],
        selectedTabChange: [{ type: Output }]
    };
    /**
     * Material design tab-group component. Supports basic tab pairs (label + content) and includes
     * animated ink-bar, keyboard navigation, and screen reader.
     * See: https://material.io/design/components/tabs.html
     */
    class MatTabGroup extends _MatTabGroupBase {
        /**
         * @param {?} elementRef
         * @param {?} changeDetectorRef
         * @param {?=} defaultConfig
         * @param {?=} animationMode
         */
        constructor(elementRef, changeDetectorRef, defaultConfig, animationMode) {
            super(elementRef, changeDetectorRef, defaultConfig, animationMode);
        }
    }
    MatTabGroup.ɵfac = function MatTabGroup_Factory(t) { return new (t || MatTabGroup)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(ChangeDetectorRef), ɵɵdirectiveInject(MAT_TABS_CONFIG, 8), ɵɵdirectiveInject(ANIMATION_MODULE_TYPE, 8)); };
    MatTabGroup.ɵcmp = ɵɵdefineComponent({ type: MatTabGroup, selectors: [["mat-tab-group"]], contentQueries: function MatTabGroup_ContentQueries(rf, ctx, dirIndex) { if (rf & 1) {
            ɵɵcontentQuery(dirIndex, MatTab, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._allTabs = _t);
        } }, viewQuery: function MatTabGroup_Query(rf, ctx) { if (rf & 1) {
            ɵɵviewQuery(_c3, true);
            ɵɵviewQuery(_c4, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._tabBodyWrapper = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._tabHeader = _t.first);
        } }, hostAttrs: [1, "mat-tab-group"], hostVars: 4, hostBindings: function MatTabGroup_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵclassProp("mat-tab-group-dynamic-height", ctx.dynamicHeight)("mat-tab-group-inverted-header", ctx.headerPosition === "below");
        } }, inputs: { color: "color", disableRipple: "disableRipple" }, exportAs: ["matTabGroup"], features: [ɵɵProvidersFeature([{
                    provide: MAT_TAB_GROUP,
                    useExisting: MatTabGroup
                }]), ɵɵInheritDefinitionFeature], decls: 6, vars: 7, consts: [[3, "selectedIndex", "disableRipple", "disablePagination", "indexFocused", "selectFocusedIndex"], ["tabHeader", ""], ["class", "mat-tab-label mat-focus-indicator", "role", "tab", "matTabLabelWrapper", "", "mat-ripple", "", "cdkMonitorElementFocus", "", 3, "id", "mat-tab-label-active", "disabled", "matRippleDisabled", "click", 4, "ngFor", "ngForOf"], [1, "mat-tab-body-wrapper"], ["tabBodyWrapper", ""], ["role", "tabpanel", 3, "id", "mat-tab-body-active", "content", "position", "origin", "animationDuration", "_onCentered", "_onCentering", 4, "ngFor", "ngForOf"], ["role", "tab", "matTabLabelWrapper", "", "mat-ripple", "", "cdkMonitorElementFocus", "", 1, "mat-tab-label", "mat-focus-indicator", 3, "id", "disabled", "matRippleDisabled", "click"], [1, "mat-tab-label-content"], [3, "ngIf"], [3, "cdkPortalOutlet"], ["role", "tabpanel", 3, "id", "content", "position", "origin", "animationDuration", "_onCentered", "_onCentering"]], template: function MatTabGroup_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "mat-tab-header", 0, 1);
            ɵɵlistener("indexFocused", function MatTabGroup_Template_mat_tab_header_indexFocused_0_listener($event) { return ctx._focusChanged($event); })("selectFocusedIndex", function MatTabGroup_Template_mat_tab_header_selectFocusedIndex_0_listener($event) { return ctx.selectedIndex = $event; });
            ɵɵtemplate(2, MatTabGroup_div_2_Template, 4, 14, "div", 2);
            ɵɵelementEnd();
            ɵɵelementStart(3, "div", 3, 4);
            ɵɵtemplate(5, MatTabGroup_mat_tab_body_5_Template, 1, 8, "mat-tab-body", 5);
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵproperty("selectedIndex", ctx.selectedIndex || 0)("disableRipple", ctx.disableRipple)("disablePagination", ctx.disablePagination);
            ɵɵadvance(2);
            ɵɵproperty("ngForOf", ctx._tabs);
            ɵɵadvance(1);
            ɵɵclassProp("_mat-animation-noopable", ctx._animationMode === "NoopAnimations");
            ɵɵadvance(2);
            ɵɵproperty("ngForOf", ctx._tabs);
        } }, directives: function () { return [MatTabHeader, NgForOf, MatTabLabelWrapper, MatRipple, CdkMonitorFocus, NgIf, CdkPortalOutlet, MatTabBody]; }, styles: [".mat-tab-group{display:flex;flex-direction:column}.mat-tab-group.mat-tab-group-inverted-header{flex-direction:column-reverse}.mat-tab-label{height:48px;padding:0 24px;cursor:pointer;box-sizing:border-box;opacity:.6;min-width:160px;text-align:center;display:inline-flex;justify-content:center;align-items:center;white-space:nowrap;position:relative}.mat-tab-label:focus{outline:none}.mat-tab-label:focus:not(.mat-tab-disabled){opacity:1}.cdk-high-contrast-active .mat-tab-label:focus{outline:dotted 2px;outline-offset:-2px}.mat-tab-label.mat-tab-disabled{cursor:default}.cdk-high-contrast-active .mat-tab-label.mat-tab-disabled{opacity:.5}.mat-tab-label .mat-tab-label-content{display:inline-flex;justify-content:center;align-items:center;white-space:nowrap}.cdk-high-contrast-active .mat-tab-label{opacity:1}@media(max-width: 599px){.mat-tab-label{padding:0 12px}}@media(max-width: 959px){.mat-tab-label{padding:0 12px}}.mat-tab-group[mat-stretch-tabs]>.mat-tab-header .mat-tab-label{flex-basis:0;flex-grow:1}.mat-tab-body-wrapper{position:relative;overflow:hidden;display:flex;transition:height 500ms cubic-bezier(0.35, 0, 0.25, 1)}._mat-animation-noopable.mat-tab-body-wrapper{transition:none;animation:none}.mat-tab-body{top:0;left:0;right:0;bottom:0;position:absolute;display:block;overflow:hidden;flex-basis:100%}.mat-tab-body.mat-tab-body-active{position:relative;overflow-x:hidden;overflow-y:auto;z-index:1;flex-grow:1}.mat-tab-group.mat-tab-group-dynamic-height .mat-tab-body.mat-tab-body-active{overflow-y:hidden}\n"], encapsulation: 2 });
    /** @nocollapse */
    MatTabGroup.ctorParameters = () => [
        { type: ElementRef },
        { type: ChangeDetectorRef },
        { type: undefined, decorators: [{ type: Inject, args: [MAT_TABS_CONFIG,] }, { type: Optional }] },
        { type: String, decorators: [{ type: Optional }, { type: Inject, args: [ANIMATION_MODULE_TYPE,] }] }
    ];
    MatTabGroup.propDecorators = {
        _allTabs: [{ type: ContentChildren, args: [MatTab, { descendants: true },] }],
        _tabBodyWrapper: [{ type: ViewChild, args: ['tabBodyWrapper',] }],
        _tabHeader: [{ type: ViewChild, args: ['tabHeader',] }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/tabs/tab-label-wrapper.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    // Boilerplate for applying mixins to MatTabLabelWrapper.
    /**
     * \@docs-private
     */
    class MatTabLabelWrapperBase {
    }
    /** @type {?} */
    const _MatTabLabelWrapperMixinBase = mixinDisabled(MatTabLabelWrapperBase);
    /**
     * Used in the `mat-tab-group` view to display tab labels.
     * \@docs-private
     */
    class MatTabLabelWrapper extends _MatTabLabelWrapperMixinBase {
        /**
         * @param {?} elementRef
         */
        constructor(elementRef) {
            super();
            this.elementRef = elementRef;
        }
        /**
         * Sets focus on the wrapper element
         * @return {?}
         */
        focus() {
            this.elementRef.nativeElement.focus();
        }
        /**
         * @return {?}
         */
        getOffsetLeft() {
            return this.elementRef.nativeElement.offsetLeft;
        }
        /**
         * @return {?}
         */
        getOffsetWidth() {
            return this.elementRef.nativeElement.offsetWidth;
        }
    }
    MatTabLabelWrapper.ɵfac = function MatTabLabelWrapper_Factory(t) { return new (t || MatTabLabelWrapper)(ɵɵdirectiveInject(ElementRef)); };
    MatTabLabelWrapper.ɵdir = ɵɵdefineDirective({ type: MatTabLabelWrapper, selectors: [["", "matTabLabelWrapper", ""]], hostVars: 3, hostBindings: function MatTabLabelWrapper_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵattribute("aria-disabled", !!ctx.disabled);
            ɵɵclassProp("mat-tab-disabled", ctx.disabled);
        } }, inputs: { disabled: "disabled" }, features: [ɵɵInheritDefinitionFeature] });
    /** @nocollapse */
    MatTabLabelWrapper.ctorParameters = () => [
        { type: ElementRef }
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/tabs/paginated-tab-header.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Config used to bind passive event listeners
     * @type {?}
     */
    const passiveEventListenerOptions = (/** @type {?} */ (normalizePassiveListenerOptions({ passive: true })));
    /**
     * The distance in pixels that will be overshot when scrolling a tab label into view. This helps
     * provide a small affordance to the label next to it.
     * @type {?}
     */
    const EXAGGERATED_OVERSCROLL = 60;
    /**
     * Amount of milliseconds to wait before starting to scroll the header automatically.
     * Set a little conservatively in order to handle fake events dispatched on touch devices.
     * @type {?}
     */
    const HEADER_SCROLL_DELAY = 650;
    /**
     * Interval in milliseconds at which to scroll the header
     * while the user is holding their pointer.
     * @type {?}
     */
    const HEADER_SCROLL_INTERVAL = 100;
    /**
     * Base class for a tab header that supported pagination.
     * \@docs-private
     * @abstract
     */
    class MatPaginatedTabHeader {
        /**
         * @param {?} _elementRef
         * @param {?} _changeDetectorRef
         * @param {?} _viewportRuler
         * @param {?} _dir
         * @param {?} _ngZone
         * @param {?=} _platform
         * @param {?=} _animationMode
         */
        constructor(_elementRef, _changeDetectorRef, _viewportRuler, _dir, _ngZone, _platform, _animationMode) {
            this._elementRef = _elementRef;
            this._changeDetectorRef = _changeDetectorRef;
            this._viewportRuler = _viewportRuler;
            this._dir = _dir;
            this._ngZone = _ngZone;
            this._platform = _platform;
            this._animationMode = _animationMode;
            /**
             * The distance in pixels that the tab labels should be translated to the left.
             */
            this._scrollDistance = 0;
            /**
             * Whether the header should scroll to the selected index after the view has been checked.
             */
            this._selectedIndexChanged = false;
            /**
             * Emits when the component is destroyed.
             */
            this._destroyed = new Subject();
            /**
             * Whether the controls for pagination should be displayed
             */
            this._showPaginationControls = false;
            /**
             * Whether the tab list can be scrolled more towards the end of the tab label list.
             */
            this._disableScrollAfter = true;
            /**
             * Whether the tab list can be scrolled more towards the beginning of the tab label list.
             */
            this._disableScrollBefore = true;
            /**
             * Stream that will stop the automated scrolling.
             */
            this._stopScrolling = new Subject();
            /**
             * Whether pagination should be disabled. This can be used to avoid unnecessary
             * layout recalculations if it's known that pagination won't be required.
             */
            this.disablePagination = false;
            this._selectedIndex = 0;
            /**
             * Event emitted when the option is selected.
             */
            this.selectFocusedIndex = new EventEmitter();
            /**
             * Event emitted when a label is focused.
             */
            this.indexFocused = new EventEmitter();
            // Bind the `mouseleave` event on the outside since it doesn't change anything in the view.
            _ngZone.runOutsideAngular((/**
             * @return {?}
             */
            () => {
                fromEvent(_elementRef.nativeElement, 'mouseleave')
                    .pipe(takeUntil(this._destroyed))
                    .subscribe((/**
                 * @return {?}
                 */
                () => {
                    this._stopInterval();
                }));
            }));
        }
        /**
         * The index of the active tab.
         * @return {?}
         */
        get selectedIndex() { return this._selectedIndex; }
        /**
         * @param {?} value
         * @return {?}
         */
        set selectedIndex(value) {
            value = coerceNumberProperty(value);
            if (this._selectedIndex != value) {
                this._selectedIndexChanged = true;
                this._selectedIndex = value;
                if (this._keyManager) {
                    this._keyManager.updateActiveItem(value);
                }
            }
        }
        /**
         * @return {?}
         */
        ngAfterViewInit() {
            // We need to handle these events manually, because we want to bind passive event listeners.
            fromEvent(this._previousPaginator.nativeElement, 'touchstart', passiveEventListenerOptions)
                .pipe(takeUntil(this._destroyed))
                .subscribe((/**
             * @return {?}
             */
            () => {
                this._handlePaginatorPress('before');
            }));
            fromEvent(this._nextPaginator.nativeElement, 'touchstart', passiveEventListenerOptions)
                .pipe(takeUntil(this._destroyed))
                .subscribe((/**
             * @return {?}
             */
            () => {
                this._handlePaginatorPress('after');
            }));
        }
        /**
         * @return {?}
         */
        ngAfterContentInit() {
            /** @type {?} */
            const dirChange = this._dir ? this._dir.change : of(null);
            /** @type {?} */
            const resize = this._viewportRuler.change(150);
            /** @type {?} */
            const realign = (/**
             * @return {?}
             */
            () => {
                this.updatePagination();
                this._alignInkBarToSelectedTab();
            });
            this._keyManager = new FocusKeyManager(this._items)
                .withHorizontalOrientation(this._getLayoutDirection())
                .withWrap();
            this._keyManager.updateActiveItem(0);
            // Defer the first call in order to allow for slower browsers to lay out the elements.
            // This helps in cases where the user lands directly on a page with paginated tabs.
            typeof requestAnimationFrame !== 'undefined' ? requestAnimationFrame(realign) : realign();
            // On dir change or window resize, realign the ink bar and update the orientation of
            // the key manager if the direction has changed.
            merge(dirChange, resize, this._items.changes).pipe(takeUntil(this._destroyed)).subscribe((/**
             * @return {?}
             */
            () => {
                // We need to defer this to give the browser some time to recalculate the element dimensions.
                Promise.resolve().then(realign);
                this._keyManager.withHorizontalOrientation(this._getLayoutDirection());
            }));
            // If there is a change in the focus key manager we need to emit the `indexFocused`
            // event in order to provide a public event that notifies about focus changes. Also we realign
            // the tabs container by scrolling the new focused tab into the visible section.
            this._keyManager.change.pipe(takeUntil(this._destroyed)).subscribe((/**
             * @param {?} newFocusIndex
             * @return {?}
             */
            newFocusIndex => {
                this.indexFocused.emit(newFocusIndex);
                this._setTabFocus(newFocusIndex);
            }));
        }
        /**
         * @return {?}
         */
        ngAfterContentChecked() {
            // If the number of tab labels have changed, check if scrolling should be enabled
            if (this._tabLabelCount != this._items.length) {
                this.updatePagination();
                this._tabLabelCount = this._items.length;
                this._changeDetectorRef.markForCheck();
            }
            // If the selected index has changed, scroll to the label and check if the scrolling controls
            // should be disabled.
            if (this._selectedIndexChanged) {
                this._scrollToLabel(this._selectedIndex);
                this._checkScrollingControls();
                this._alignInkBarToSelectedTab();
                this._selectedIndexChanged = false;
                this._changeDetectorRef.markForCheck();
            }
            // If the scroll distance has been changed (tab selected, focused, scroll controls activated),
            // then translate the header to reflect this.
            if (this._scrollDistanceChanged) {
                this._updateTabScrollPosition();
                this._scrollDistanceChanged = false;
                this._changeDetectorRef.markForCheck();
            }
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this._destroyed.next();
            this._destroyed.complete();
            this._stopScrolling.complete();
        }
        /**
         * Handles keyboard events on the header.
         * @param {?} event
         * @return {?}
         */
        _handleKeydown(event) {
            // We don't handle any key bindings with a modifier key.
            if (hasModifierKey(event)) {
                return;
            }
            switch (event.keyCode) {
                case HOME:
                    this._keyManager.setFirstItemActive();
                    event.preventDefault();
                    break;
                case END:
                    this._keyManager.setLastItemActive();
                    event.preventDefault();
                    break;
                case ENTER:
                case SPACE:
                    if (this.focusIndex !== this.selectedIndex) {
                        this.selectFocusedIndex.emit(this.focusIndex);
                        this._itemSelected(event);
                    }
                    break;
                default:
                    this._keyManager.onKeydown(event);
            }
        }
        /**
         * Callback for when the MutationObserver detects that the content has changed.
         * @return {?}
         */
        _onContentChanges() {
            /** @type {?} */
            const textContent = this._elementRef.nativeElement.textContent;
            // We need to diff the text content of the header, because the MutationObserver callback
            // will fire even if the text content didn't change which is inefficient and is prone
            // to infinite loops if a poorly constructed expression is passed in (see #14249).
            if (textContent !== this._currentTextContent) {
                this._currentTextContent = textContent || '';
                // The content observer runs outside the `NgZone` by default, which
                // means that we need to bring the callback back in ourselves.
                this._ngZone.run((/**
                 * @return {?}
                 */
                () => {
                    this.updatePagination();
                    this._alignInkBarToSelectedTab();
                    this._changeDetectorRef.markForCheck();
                }));
            }
        }
        /**
         * Updates the view whether pagination should be enabled or not.
         *
         * WARNING: Calling this method can be very costly in terms of performance. It should be called
         * as infrequently as possible from outside of the Tabs component as it causes a reflow of the
         * page.
         * @return {?}
         */
        updatePagination() {
            this._checkPaginationEnabled();
            this._checkScrollingControls();
            this._updateTabScrollPosition();
        }
        /**
         * Tracks which element has focus; used for keyboard navigation
         * @return {?}
         */
        get focusIndex() {
            return this._keyManager ? (/** @type {?} */ (this._keyManager.activeItemIndex)) : 0;
        }
        /**
         * When the focus index is set, we must manually send focus to the correct label
         * @param {?} value
         * @return {?}
         */
        set focusIndex(value) {
            if (!this._isValidIndex(value) || this.focusIndex === value || !this._keyManager) {
                return;
            }
            this._keyManager.setActiveItem(value);
        }
        /**
         * Determines if an index is valid.  If the tabs are not ready yet, we assume that the user is
         * providing a valid index and return true.
         * @param {?} index
         * @return {?}
         */
        _isValidIndex(index) {
            if (!this._items) {
                return true;
            }
            /** @type {?} */
            const tab = this._items ? this._items.toArray()[index] : null;
            return !!tab && !tab.disabled;
        }
        /**
         * Sets focus on the HTML element for the label wrapper and scrolls it into the view if
         * scrolling is enabled.
         * @param {?} tabIndex
         * @return {?}
         */
        _setTabFocus(tabIndex) {
            if (this._showPaginationControls) {
                this._scrollToLabel(tabIndex);
            }
            if (this._items && this._items.length) {
                this._items.toArray()[tabIndex].focus();
                // Do not let the browser manage scrolling to focus the element, this will be handled
                // by using translation. In LTR, the scroll left should be 0. In RTL, the scroll width
                // should be the full width minus the offset width.
                /** @type {?} */
                const containerEl = this._tabListContainer.nativeElement;
                /** @type {?} */
                const dir = this._getLayoutDirection();
                if (dir == 'ltr') {
                    containerEl.scrollLeft = 0;
                }
                else {
                    containerEl.scrollLeft = containerEl.scrollWidth - containerEl.offsetWidth;
                }
            }
        }
        /**
         * The layout direction of the containing app.
         * @return {?}
         */
        _getLayoutDirection() {
            return this._dir && this._dir.value === 'rtl' ? 'rtl' : 'ltr';
        }
        /**
         * Performs the CSS transformation on the tab list that will cause the list to scroll.
         * @return {?}
         */
        _updateTabScrollPosition() {
            if (this.disablePagination) {
                return;
            }
            /** @type {?} */
            const scrollDistance = this.scrollDistance;
            /** @type {?} */
            const platform = this._platform;
            /** @type {?} */
            const translateX = this._getLayoutDirection() === 'ltr' ? -scrollDistance : scrollDistance;
            // Don't use `translate3d` here because we don't want to create a new layer. A new layer
            // seems to cause flickering and overflow in Internet Explorer. For example, the ink bar
            // and ripples will exceed the boundaries of the visible tab bar.
            // See: https://github.com/angular/components/issues/10276
            // We round the `transform` here, because transforms with sub-pixel precision cause some
            // browsers to blur the content of the element.
            this._tabList.nativeElement.style.transform = `translateX(${Math.round(translateX)}px)`;
            // Setting the `transform` on IE will change the scroll offset of the parent, causing the
            // position to be thrown off in some cases. We have to reset it ourselves to ensure that
            // it doesn't get thrown off. Note that we scope it only to IE and Edge, because messing
            // with the scroll position throws off Chrome 71+ in RTL mode (see #14689).
            // @breaking-change 9.0.0 Remove null check for `platform` after it can no longer be undefined.
            if (platform && (platform.TRIDENT || platform.EDGE)) {
                this._tabListContainer.nativeElement.scrollLeft = 0;
            }
        }
        /**
         * Sets the distance in pixels that the tab header should be transformed in the X-axis.
         * @return {?}
         */
        get scrollDistance() { return this._scrollDistance; }
        /**
         * @param {?} value
         * @return {?}
         */
        set scrollDistance(value) {
            this._scrollTo(value);
        }
        /**
         * Moves the tab list in the 'before' or 'after' direction (towards the beginning of the list or
         * the end of the list, respectively). The distance to scroll is computed to be a third of the
         * length of the tab list view window.
         *
         * This is an expensive call that forces a layout reflow to compute box and scroll metrics and
         * should be called sparingly.
         * @param {?} direction
         * @return {?}
         */
        _scrollHeader(direction) {
            /** @type {?} */
            const viewLength = this._tabListContainer.nativeElement.offsetWidth;
            // Move the scroll distance one-third the length of the tab list's viewport.
            /** @type {?} */
            const scrollAmount = (direction == 'before' ? -1 : 1) * viewLength / 3;
            return this._scrollTo(this._scrollDistance + scrollAmount);
        }
        /**
         * Handles click events on the pagination arrows.
         * @param {?} direction
         * @return {?}
         */
        _handlePaginatorClick(direction) {
            this._stopInterval();
            this._scrollHeader(direction);
        }
        /**
         * Moves the tab list such that the desired tab label (marked by index) is moved into view.
         *
         * This is an expensive call that forces a layout reflow to compute box and scroll metrics and
         * should be called sparingly.
         * @param {?} labelIndex
         * @return {?}
         */
        _scrollToLabel(labelIndex) {
            if (this.disablePagination) {
                return;
            }
            /** @type {?} */
            const selectedLabel = this._items ? this._items.toArray()[labelIndex] : null;
            if (!selectedLabel) {
                return;
            }
            // The view length is the visible width of the tab labels.
            /** @type {?} */
            const viewLength = this._tabListContainer.nativeElement.offsetWidth;
            const { offsetLeft, offsetWidth } = selectedLabel.elementRef.nativeElement;
            /** @type {?} */
            let labelBeforePos;
            /** @type {?} */
            let labelAfterPos;
            if (this._getLayoutDirection() == 'ltr') {
                labelBeforePos = offsetLeft;
                labelAfterPos = labelBeforePos + offsetWidth;
            }
            else {
                labelAfterPos = this._tabList.nativeElement.offsetWidth - offsetLeft;
                labelBeforePos = labelAfterPos - offsetWidth;
            }
            /** @type {?} */
            const beforeVisiblePos = this.scrollDistance;
            /** @type {?} */
            const afterVisiblePos = this.scrollDistance + viewLength;
            if (labelBeforePos < beforeVisiblePos) {
                // Scroll header to move label to the before direction
                this.scrollDistance -= beforeVisiblePos - labelBeforePos + EXAGGERATED_OVERSCROLL;
            }
            else if (labelAfterPos > afterVisiblePos) {
                // Scroll header to move label to the after direction
                this.scrollDistance += labelAfterPos - afterVisiblePos + EXAGGERATED_OVERSCROLL;
            }
        }
        /**
         * Evaluate whether the pagination controls should be displayed. If the scroll width of the
         * tab list is wider than the size of the header container, then the pagination controls should
         * be shown.
         *
         * This is an expensive call that forces a layout reflow to compute box and scroll metrics and
         * should be called sparingly.
         * @return {?}
         */
        _checkPaginationEnabled() {
            if (this.disablePagination) {
                this._showPaginationControls = false;
            }
            else {
                /** @type {?} */
                const isEnabled = this._tabList.nativeElement.scrollWidth > this._elementRef.nativeElement.offsetWidth;
                if (!isEnabled) {
                    this.scrollDistance = 0;
                }
                if (isEnabled !== this._showPaginationControls) {
                    this._changeDetectorRef.markForCheck();
                }
                this._showPaginationControls = isEnabled;
            }
        }
        /**
         * Evaluate whether the before and after controls should be enabled or disabled.
         * If the header is at the beginning of the list (scroll distance is equal to 0) then disable the
         * before button. If the header is at the end of the list (scroll distance is equal to the
         * maximum distance we can scroll), then disable the after button.
         *
         * This is an expensive call that forces a layout reflow to compute box and scroll metrics and
         * should be called sparingly.
         * @return {?}
         */
        _checkScrollingControls() {
            if (this.disablePagination) {
                this._disableScrollAfter = this._disableScrollBefore = true;
            }
            else {
                // Check if the pagination arrows should be activated.
                this._disableScrollBefore = this.scrollDistance == 0;
                this._disableScrollAfter = this.scrollDistance == this._getMaxScrollDistance();
                this._changeDetectorRef.markForCheck();
            }
        }
        /**
         * Determines what is the maximum length in pixels that can be set for the scroll distance. This
         * is equal to the difference in width between the tab list container and tab header container.
         *
         * This is an expensive call that forces a layout reflow to compute box and scroll metrics and
         * should be called sparingly.
         * @return {?}
         */
        _getMaxScrollDistance() {
            /** @type {?} */
            const lengthOfTabList = this._tabList.nativeElement.scrollWidth;
            /** @type {?} */
            const viewLength = this._tabListContainer.nativeElement.offsetWidth;
            return (lengthOfTabList - viewLength) || 0;
        }
        /**
         * Tells the ink-bar to align itself to the current label wrapper
         * @return {?}
         */
        _alignInkBarToSelectedTab() {
            /** @type {?} */
            const selectedItem = this._items && this._items.length ?
                this._items.toArray()[this.selectedIndex] : null;
            /** @type {?} */
            const selectedLabelWrapper = selectedItem ? selectedItem.elementRef.nativeElement : null;
            if (selectedLabelWrapper) {
                this._inkBar.alignToElement(selectedLabelWrapper);
            }
            else {
                this._inkBar.hide();
            }
        }
        /**
         * Stops the currently-running paginator interval.
         * @return {?}
         */
        _stopInterval() {
            this._stopScrolling.next();
        }
        /**
         * Handles the user pressing down on one of the paginators.
         * Starts scrolling the header after a certain amount of time.
         * @param {?} direction In which direction the paginator should be scrolled.
         * @param {?=} mouseEvent
         * @return {?}
         */
        _handlePaginatorPress(direction, mouseEvent) {
            // Don't start auto scrolling for right mouse button clicks. Note that we shouldn't have to
            // null check the `button`, but we do it so we don't break tests that use fake events.
            if (mouseEvent && mouseEvent.button != null && mouseEvent.button !== 0) {
                return;
            }
            // Avoid overlapping timers.
            this._stopInterval();
            // Start a timer after the delay and keep firing based on the interval.
            timer(HEADER_SCROLL_DELAY, HEADER_SCROLL_INTERVAL)
                // Keep the timer going until something tells it to stop or the component is destroyed.
                .pipe(takeUntil(merge(this._stopScrolling, this._destroyed)))
                .subscribe((/**
             * @return {?}
             */
            () => {
                const { maxScrollDistance, distance } = this._scrollHeader(direction);
                // Stop the timer if we've reached the start or the end.
                if (distance === 0 || distance >= maxScrollDistance) {
                    this._stopInterval();
                }
            }));
        }
        /**
         * Scrolls the header to a given position.
         * @private
         * @param {?} position Position to which to scroll.
         * @return {?} Information on the current scroll distance and the maximum.
         */
        _scrollTo(position) {
            if (this.disablePagination) {
                return { maxScrollDistance: 0, distance: 0 };
            }
            /** @type {?} */
            const maxScrollDistance = this._getMaxScrollDistance();
            this._scrollDistance = Math.max(0, Math.min(maxScrollDistance, position));
            // Mark that the scroll distance has changed so that after the view is checked, the CSS
            // transformation can move the header.
            this._scrollDistanceChanged = true;
            this._checkScrollingControls();
            return { maxScrollDistance, distance: this._scrollDistance };
        }
    }
    MatPaginatedTabHeader.ɵfac = function MatPaginatedTabHeader_Factory(t) { return new (t || MatPaginatedTabHeader)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(ChangeDetectorRef), ɵɵdirectiveInject(ViewportRuler), ɵɵdirectiveInject(Directionality, 8), ɵɵdirectiveInject(NgZone), ɵɵdirectiveInject(Platform), ɵɵdirectiveInject(ANIMATION_MODULE_TYPE, 8)); };
    MatPaginatedTabHeader.ɵdir = ɵɵdefineDirective({ type: MatPaginatedTabHeader, inputs: { disablePagination: "disablePagination" } });
    /** @nocollapse */
    MatPaginatedTabHeader.ctorParameters = () => [
        { type: ElementRef },
        { type: ChangeDetectorRef },
        { type: ViewportRuler },
        { type: Directionality, decorators: [{ type: Optional }] },
        { type: NgZone },
        { type: Platform },
        { type: String, decorators: [{ type: Optional }, { type: Inject, args: [ANIMATION_MODULE_TYPE,] }] }
    ];
    MatPaginatedTabHeader.propDecorators = {
        disablePagination: [{ type: Input }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/tabs/tab-header.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Base class with all of the `MatTabHeader` functionality.
     * \@docs-private
     * @abstract
     */
    // tslint:disable-next-line:class-name
    class _MatTabHeaderBase extends MatPaginatedTabHeader {
        /**
         * @param {?} elementRef
         * @param {?} changeDetectorRef
         * @param {?} viewportRuler
         * @param {?} dir
         * @param {?} ngZone
         * @param {?} platform
         * @param {?=} animationMode
         */
        constructor(elementRef, changeDetectorRef, viewportRuler, dir, ngZone, platform, 
        // @breaking-change 9.0.0 `_animationMode` parameter to be made required.
        animationMode) {
            super(elementRef, changeDetectorRef, viewportRuler, dir, ngZone, platform, animationMode);
            this._disableRipple = false;
        }
        /**
         * Whether the ripple effect is disabled or not.
         * @return {?}
         */
        get disableRipple() { return this._disableRipple; }
        /**
         * @param {?} value
         * @return {?}
         */
        set disableRipple(value) { this._disableRipple = coerceBooleanProperty(value); }
        /**
         * @protected
         * @param {?} event
         * @return {?}
         */
        _itemSelected(event) {
            event.preventDefault();
        }
    }
    _MatTabHeaderBase.ɵfac = function _MatTabHeaderBase_Factory(t) { return new (t || _MatTabHeaderBase)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(ChangeDetectorRef), ɵɵdirectiveInject(ViewportRuler), ɵɵdirectiveInject(Directionality, 8), ɵɵdirectiveInject(NgZone), ɵɵdirectiveInject(Platform), ɵɵdirectiveInject(ANIMATION_MODULE_TYPE, 8)); };
    _MatTabHeaderBase.ɵdir = ɵɵdefineDirective({ type: _MatTabHeaderBase, inputs: { disableRipple: "disableRipple" }, features: [ɵɵInheritDefinitionFeature] });
    /** @nocollapse */
    _MatTabHeaderBase.ctorParameters = () => [
        { type: ElementRef },
        { type: ChangeDetectorRef },
        { type: ViewportRuler },
        { type: Directionality, decorators: [{ type: Optional }] },
        { type: NgZone },
        { type: Platform },
        { type: String, decorators: [{ type: Optional }, { type: Inject, args: [ANIMATION_MODULE_TYPE,] }] }
    ];
    _MatTabHeaderBase.propDecorators = {
        disableRipple: [{ type: Input }]
    };
    /**
     * The header of the tab group which displays a list of all the tabs in the tab group. Includes
     * an ink bar that follows the currently selected tab. When the tabs list's width exceeds the
     * width of the header container, then arrows will be displayed to allow the user to scroll
     * left and right across the header.
     * \@docs-private
     */
    class MatTabHeader extends _MatTabHeaderBase {
        /**
         * @param {?} elementRef
         * @param {?} changeDetectorRef
         * @param {?} viewportRuler
         * @param {?} dir
         * @param {?} ngZone
         * @param {?} platform
         * @param {?=} animationMode
         */
        constructor(elementRef, changeDetectorRef, viewportRuler, dir, ngZone, platform, 
        // @breaking-change 9.0.0 `_animationMode` parameter to be made required.
        animationMode) {
            super(elementRef, changeDetectorRef, viewportRuler, dir, ngZone, platform, animationMode);
        }
    }
    MatTabHeader.ɵfac = function MatTabHeader_Factory(t) { return new (t || MatTabHeader)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(ChangeDetectorRef), ɵɵdirectiveInject(ViewportRuler), ɵɵdirectiveInject(Directionality, 8), ɵɵdirectiveInject(NgZone), ɵɵdirectiveInject(Platform), ɵɵdirectiveInject(ANIMATION_MODULE_TYPE, 8)); };
    MatTabHeader.ɵcmp = ɵɵdefineComponent({ type: MatTabHeader, selectors: [["mat-tab-header"]], contentQueries: function MatTabHeader_ContentQueries(rf, ctx, dirIndex) { if (rf & 1) {
            ɵɵcontentQuery(dirIndex, MatTabLabelWrapper, false);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._items = _t);
        } }, viewQuery: function MatTabHeader_Query(rf, ctx) { if (rf & 1) {
            ɵɵstaticViewQuery(MatInkBar, true);
            ɵɵstaticViewQuery(_c5, true);
            ɵɵstaticViewQuery(_c6, true);
            ɵɵviewQuery(_c7, true);
            ɵɵviewQuery(_c8, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._inkBar = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._tabListContainer = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._tabList = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._nextPaginator = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._previousPaginator = _t.first);
        } }, hostAttrs: [1, "mat-tab-header"], hostVars: 4, hostBindings: function MatTabHeader_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵclassProp("mat-tab-header-pagination-controls-enabled", ctx._showPaginationControls)("mat-tab-header-rtl", ctx._getLayoutDirection() == "rtl");
        } }, inputs: { selectedIndex: "selectedIndex" }, outputs: { selectFocusedIndex: "selectFocusedIndex", indexFocused: "indexFocused" }, features: [ɵɵInheritDefinitionFeature], ngContentSelectors: _c0$3, decls: 13, vars: 8, consts: [["aria-hidden", "true", "mat-ripple", "", 1, "mat-tab-header-pagination", "mat-tab-header-pagination-before", "mat-elevation-z4", 3, "matRippleDisabled", "click", "mousedown", "touchend"], ["previousPaginator", ""], [1, "mat-tab-header-pagination-chevron"], [1, "mat-tab-label-container", 3, "keydown"], ["tabListContainer", ""], ["role", "tablist", 1, "mat-tab-list", 3, "cdkObserveContent"], ["tabList", ""], [1, "mat-tab-labels"], ["aria-hidden", "true", "mat-ripple", "", 1, "mat-tab-header-pagination", "mat-tab-header-pagination-after", "mat-elevation-z4", 3, "matRippleDisabled", "mousedown", "click", "touchend"], ["nextPaginator", ""]], template: function MatTabHeader_Template(rf, ctx) { if (rf & 1) {
            ɵɵprojectionDef();
            ɵɵelementStart(0, "div", 0, 1);
            ɵɵlistener("click", function MatTabHeader_Template_div_click_0_listener() { return ctx._handlePaginatorClick("before"); })("mousedown", function MatTabHeader_Template_div_mousedown_0_listener($event) { return ctx._handlePaginatorPress("before", $event); })("touchend", function MatTabHeader_Template_div_touchend_0_listener() { return ctx._stopInterval(); });
            ɵɵelement(2, "div", 2);
            ɵɵelementEnd();
            ɵɵelementStart(3, "div", 3, 4);
            ɵɵlistener("keydown", function MatTabHeader_Template_div_keydown_3_listener($event) { return ctx._handleKeydown($event); });
            ɵɵelementStart(5, "div", 5, 6);
            ɵɵlistener("cdkObserveContent", function MatTabHeader_Template_div_cdkObserveContent_5_listener() { return ctx._onContentChanges(); });
            ɵɵelementStart(7, "div", 7);
            ɵɵprojection(8);
            ɵɵelementEnd();
            ɵɵelement(9, "mat-ink-bar");
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementStart(10, "div", 8, 9);
            ɵɵlistener("mousedown", function MatTabHeader_Template_div_mousedown_10_listener($event) { return ctx._handlePaginatorPress("after", $event); })("click", function MatTabHeader_Template_div_click_10_listener() { return ctx._handlePaginatorClick("after"); })("touchend", function MatTabHeader_Template_div_touchend_10_listener() { return ctx._stopInterval(); });
            ɵɵelement(12, "div", 2);
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵclassProp("mat-tab-header-pagination-disabled", ctx._disableScrollBefore);
            ɵɵproperty("matRippleDisabled", ctx._disableScrollBefore || ctx.disableRipple);
            ɵɵadvance(5);
            ɵɵclassProp("_mat-animation-noopable", ctx._animationMode === "NoopAnimations");
            ɵɵadvance(5);
            ɵɵclassProp("mat-tab-header-pagination-disabled", ctx._disableScrollAfter);
            ɵɵproperty("matRippleDisabled", ctx._disableScrollAfter || ctx.disableRipple);
        } }, directives: [MatRipple, CdkObserveContent, MatInkBar], styles: [".mat-tab-header{display:flex;overflow:hidden;position:relative;flex-shrink:0}.mat-tab-header-pagination{-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;position:relative;display:none;justify-content:center;align-items:center;min-width:32px;cursor:pointer;z-index:2;-webkit-tap-highlight-color:transparent;touch-action:none}.mat-tab-header-pagination-controls-enabled .mat-tab-header-pagination{display:flex}.mat-tab-header-pagination-before,.mat-tab-header-rtl .mat-tab-header-pagination-after{padding-left:4px}.mat-tab-header-pagination-before .mat-tab-header-pagination-chevron,.mat-tab-header-rtl .mat-tab-header-pagination-after .mat-tab-header-pagination-chevron{transform:rotate(-135deg)}.mat-tab-header-rtl .mat-tab-header-pagination-before,.mat-tab-header-pagination-after{padding-right:4px}.mat-tab-header-rtl .mat-tab-header-pagination-before .mat-tab-header-pagination-chevron,.mat-tab-header-pagination-after .mat-tab-header-pagination-chevron{transform:rotate(45deg)}.mat-tab-header-pagination-chevron{border-style:solid;border-width:2px 2px 0 0;content:\"\";height:8px;width:8px}.mat-tab-header-pagination-disabled{box-shadow:none;cursor:default}.mat-tab-list{flex-grow:1;position:relative;transition:transform 500ms cubic-bezier(0.35, 0, 0.25, 1)}.mat-ink-bar{position:absolute;bottom:0;height:2px;transition:500ms cubic-bezier(0.35, 0, 0.25, 1)}._mat-animation-noopable.mat-ink-bar{transition:none;animation:none}.mat-tab-group-inverted-header .mat-ink-bar{bottom:auto;top:0}.cdk-high-contrast-active .mat-ink-bar{outline:solid 2px;height:0}.mat-tab-labels{display:flex}[mat-align-tabs=center]>.mat-tab-header .mat-tab-labels{justify-content:center}[mat-align-tabs=end]>.mat-tab-header .mat-tab-labels{justify-content:flex-end}.mat-tab-label-container{display:flex;flex-grow:1;overflow:hidden;z-index:1}._mat-animation-noopable.mat-tab-list{transition:none;animation:none}.mat-tab-label{height:48px;padding:0 24px;cursor:pointer;box-sizing:border-box;opacity:.6;min-width:160px;text-align:center;display:inline-flex;justify-content:center;align-items:center;white-space:nowrap;position:relative}.mat-tab-label:focus{outline:none}.mat-tab-label:focus:not(.mat-tab-disabled){opacity:1}.cdk-high-contrast-active .mat-tab-label:focus{outline:dotted 2px;outline-offset:-2px}.mat-tab-label.mat-tab-disabled{cursor:default}.cdk-high-contrast-active .mat-tab-label.mat-tab-disabled{opacity:.5}.mat-tab-label .mat-tab-label-content{display:inline-flex;justify-content:center;align-items:center;white-space:nowrap}.cdk-high-contrast-active .mat-tab-label{opacity:1}@media(max-width: 599px){.mat-tab-label{min-width:72px}}\n"], encapsulation: 2 });
    /** @nocollapse */
    MatTabHeader.ctorParameters = () => [
        { type: ElementRef },
        { type: ChangeDetectorRef },
        { type: ViewportRuler },
        { type: Directionality, decorators: [{ type: Optional }] },
        { type: NgZone },
        { type: Platform },
        { type: String, decorators: [{ type: Optional }, { type: Inject, args: [ANIMATION_MODULE_TYPE,] }] }
    ];
    MatTabHeader.propDecorators = {
        _items: [{ type: ContentChildren, args: [MatTabLabelWrapper, { descendants: false },] }],
        _inkBar: [{ type: ViewChild, args: [MatInkBar, { static: true },] }],
        _tabListContainer: [{ type: ViewChild, args: ['tabListContainer', { static: true },] }],
        _tabList: [{ type: ViewChild, args: ['tabList', { static: true },] }],
        _nextPaginator: [{ type: ViewChild, args: ['nextPaginator',] }],
        _previousPaginator: [{ type: ViewChild, args: ['previousPaginator',] }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/tabs/tab-nav-bar/tab-nav-bar.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Base class with all of the `MatTabNav` functionality.
     * \@docs-private
     * @abstract
     */
    // tslint:disable-next-line:class-name
    class _MatTabNavBase extends MatPaginatedTabHeader {
        /**
         * @param {?} elementRef
         * @param {?} dir
         * @param {?} ngZone
         * @param {?} changeDetectorRef
         * @param {?} viewportRuler
         * @param {?=} platform
         * @param {?=} animationMode
         */
        constructor(elementRef, dir, ngZone, changeDetectorRef, viewportRuler, 
        /**
         * @deprecated @breaking-change 9.0.0 `platform` parameter to become required.
         */
        platform, animationMode) {
            super(elementRef, changeDetectorRef, viewportRuler, dir, ngZone, platform, animationMode);
            this._disableRipple = false;
            /**
             * Theme color of the nav bar.
             */
            this.color = 'primary';
        }
        /**
         * Background color of the tab nav.
         * @return {?}
         */
        get backgroundColor() { return this._backgroundColor; }
        /**
         * @param {?} value
         * @return {?}
         */
        set backgroundColor(value) {
            /** @type {?} */
            const classList = this._elementRef.nativeElement.classList;
            classList.remove(`mat-background-${this.backgroundColor}`);
            if (value) {
                classList.add(`mat-background-${value}`);
            }
            this._backgroundColor = value;
        }
        /**
         * Whether the ripple effect is disabled or not.
         * @return {?}
         */
        get disableRipple() { return this._disableRipple; }
        /**
         * @param {?} value
         * @return {?}
         */
        set disableRipple(value) { this._disableRipple = coerceBooleanProperty(value); }
        /**
         * @protected
         * @return {?}
         */
        _itemSelected() {
            // noop
        }
        /**
         * @return {?}
         */
        ngAfterContentInit() {
            // We need this to run before the `changes` subscription in parent to ensure that the
            // selectedIndex is up-to-date by the time the super class starts looking for it.
            this._items.changes.pipe(startWith(null), takeUntil(this._destroyed)).subscribe((/**
             * @return {?}
             */
            () => {
                this.updateActiveLink();
            }));
            super.ngAfterContentInit();
        }
        /**
         * Notifies the component that the active link has been changed.
         * \@breaking-change 8.0.0 `element` parameter to be removed.
         * @param {?=} _element
         * @return {?}
         */
        updateActiveLink(_element) {
            if (!this._items) {
                return;
            }
            /** @type {?} */
            const items = this._items.toArray();
            for (let i = 0; i < items.length; i++) {
                if (items[i].active) {
                    this.selectedIndex = i;
                    this._changeDetectorRef.markForCheck();
                    return;
                }
            }
            // The ink bar should hide itself if no items are active.
            this.selectedIndex = -1;
            this._inkBar.hide();
        }
    }
    _MatTabNavBase.ɵfac = function _MatTabNavBase_Factory(t) { return new (t || _MatTabNavBase)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(Directionality, 8), ɵɵdirectiveInject(NgZone), ɵɵdirectiveInject(ChangeDetectorRef), ɵɵdirectiveInject(ViewportRuler), ɵɵdirectiveInject(Platform, 8), ɵɵdirectiveInject(ANIMATION_MODULE_TYPE, 8)); };
    _MatTabNavBase.ɵdir = ɵɵdefineDirective({ type: _MatTabNavBase, inputs: { color: "color", backgroundColor: "backgroundColor", disableRipple: "disableRipple" }, features: [ɵɵInheritDefinitionFeature] });
    /** @nocollapse */
    _MatTabNavBase.ctorParameters = () => [
        { type: ElementRef },
        { type: Directionality, decorators: [{ type: Optional }] },
        { type: NgZone },
        { type: ChangeDetectorRef },
        { type: ViewportRuler },
        { type: Platform, decorators: [{ type: Optional }] },
        { type: String, decorators: [{ type: Optional }, { type: Inject, args: [ANIMATION_MODULE_TYPE,] }] }
    ];
    _MatTabNavBase.propDecorators = {
        backgroundColor: [{ type: Input }],
        disableRipple: [{ type: Input }],
        color: [{ type: Input }]
    };
    /**
     * Navigation component matching the styles of the tab group header.
     * Provides anchored navigation with animated ink bar.
     */
    class MatTabNav extends _MatTabNavBase {
        /**
         * @param {?} elementRef
         * @param {?} dir
         * @param {?} ngZone
         * @param {?} changeDetectorRef
         * @param {?} viewportRuler
         * @param {?=} platform
         * @param {?=} animationMode
         */
        constructor(elementRef, dir, ngZone, changeDetectorRef, viewportRuler, 
        /**
         * @deprecated @breaking-change 9.0.0 `platform` parameter to become required.
         */
        platform, animationMode) {
            super(elementRef, dir, ngZone, changeDetectorRef, viewportRuler, platform, animationMode);
        }
    }
    MatTabNav.ɵfac = function MatTabNav_Factory(t) { return new (t || MatTabNav)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(Directionality, 8), ɵɵdirectiveInject(NgZone), ɵɵdirectiveInject(ChangeDetectorRef), ɵɵdirectiveInject(ViewportRuler), ɵɵdirectiveInject(Platform, 8), ɵɵdirectiveInject(ANIMATION_MODULE_TYPE, 8)); };
    MatTabNav.ɵcmp = ɵɵdefineComponent({ type: MatTabNav, selectors: [["", "mat-tab-nav-bar", ""]], contentQueries: function MatTabNav_ContentQueries(rf, ctx, dirIndex) { if (rf & 1) {
            ɵɵcontentQuery(dirIndex, MatTabLink, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._items = _t);
        } }, viewQuery: function MatTabNav_Query(rf, ctx) { if (rf & 1) {
            ɵɵstaticViewQuery(MatInkBar, true);
            ɵɵstaticViewQuery(_c5, true);
            ɵɵstaticViewQuery(_c6, true);
            ɵɵviewQuery(_c7, true);
            ɵɵviewQuery(_c8, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._inkBar = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._tabListContainer = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._tabList = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._nextPaginator = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._previousPaginator = _t.first);
        } }, hostAttrs: [1, "mat-tab-nav-bar", "mat-tab-header"], hostVars: 10, hostBindings: function MatTabNav_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵclassProp("mat-tab-header-pagination-controls-enabled", ctx._showPaginationControls)("mat-tab-header-rtl", ctx._getLayoutDirection() == "rtl")("mat-primary", ctx.color !== "warn" && ctx.color !== "accent")("mat-accent", ctx.color === "accent")("mat-warn", ctx.color === "warn");
        } }, inputs: { color: "color" }, exportAs: ["matTabNavBar", "matTabNav"], features: [ɵɵInheritDefinitionFeature], attrs: _c9, ngContentSelectors: _c0$3, decls: 13, vars: 8, consts: [["aria-hidden", "true", "mat-ripple", "", 1, "mat-tab-header-pagination", "mat-tab-header-pagination-before", "mat-elevation-z4", 3, "matRippleDisabled", "click", "mousedown", "touchend"], ["previousPaginator", ""], [1, "mat-tab-header-pagination-chevron"], [1, "mat-tab-link-container", 3, "keydown"], ["tabListContainer", ""], [1, "mat-tab-list", 3, "cdkObserveContent"], ["tabList", ""], [1, "mat-tab-links"], ["aria-hidden", "true", "mat-ripple", "", 1, "mat-tab-header-pagination", "mat-tab-header-pagination-after", "mat-elevation-z4", 3, "matRippleDisabled", "mousedown", "click", "touchend"], ["nextPaginator", ""]], template: function MatTabNav_Template(rf, ctx) { if (rf & 1) {
            ɵɵprojectionDef();
            ɵɵelementStart(0, "div", 0, 1);
            ɵɵlistener("click", function MatTabNav_Template_div_click_0_listener() { return ctx._handlePaginatorClick("before"); })("mousedown", function MatTabNav_Template_div_mousedown_0_listener($event) { return ctx._handlePaginatorPress("before", $event); })("touchend", function MatTabNav_Template_div_touchend_0_listener() { return ctx._stopInterval(); });
            ɵɵelement(2, "div", 2);
            ɵɵelementEnd();
            ɵɵelementStart(3, "div", 3, 4);
            ɵɵlistener("keydown", function MatTabNav_Template_div_keydown_3_listener($event) { return ctx._handleKeydown($event); });
            ɵɵelementStart(5, "div", 5, 6);
            ɵɵlistener("cdkObserveContent", function MatTabNav_Template_div_cdkObserveContent_5_listener() { return ctx._onContentChanges(); });
            ɵɵelementStart(7, "div", 7);
            ɵɵprojection(8);
            ɵɵelementEnd();
            ɵɵelement(9, "mat-ink-bar");
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementStart(10, "div", 8, 9);
            ɵɵlistener("mousedown", function MatTabNav_Template_div_mousedown_10_listener($event) { return ctx._handlePaginatorPress("after", $event); })("click", function MatTabNav_Template_div_click_10_listener() { return ctx._handlePaginatorClick("after"); })("touchend", function MatTabNav_Template_div_touchend_10_listener() { return ctx._stopInterval(); });
            ɵɵelement(12, "div", 2);
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵclassProp("mat-tab-header-pagination-disabled", ctx._disableScrollBefore);
            ɵɵproperty("matRippleDisabled", ctx._disableScrollBefore || ctx.disableRipple);
            ɵɵadvance(5);
            ɵɵclassProp("_mat-animation-noopable", ctx._animationMode === "NoopAnimations");
            ɵɵadvance(5);
            ɵɵclassProp("mat-tab-header-pagination-disabled", ctx._disableScrollAfter);
            ɵɵproperty("matRippleDisabled", ctx._disableScrollAfter || ctx.disableRipple);
        } }, directives: [MatRipple, CdkObserveContent, MatInkBar], styles: [".mat-tab-header{display:flex;overflow:hidden;position:relative;flex-shrink:0}.mat-tab-header-pagination{-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;position:relative;display:none;justify-content:center;align-items:center;min-width:32px;cursor:pointer;z-index:2;-webkit-tap-highlight-color:transparent;touch-action:none}.mat-tab-header-pagination-controls-enabled .mat-tab-header-pagination{display:flex}.mat-tab-header-pagination-before,.mat-tab-header-rtl .mat-tab-header-pagination-after{padding-left:4px}.mat-tab-header-pagination-before .mat-tab-header-pagination-chevron,.mat-tab-header-rtl .mat-tab-header-pagination-after .mat-tab-header-pagination-chevron{transform:rotate(-135deg)}.mat-tab-header-rtl .mat-tab-header-pagination-before,.mat-tab-header-pagination-after{padding-right:4px}.mat-tab-header-rtl .mat-tab-header-pagination-before .mat-tab-header-pagination-chevron,.mat-tab-header-pagination-after .mat-tab-header-pagination-chevron{transform:rotate(45deg)}.mat-tab-header-pagination-chevron{border-style:solid;border-width:2px 2px 0 0;content:\"\";height:8px;width:8px}.mat-tab-header-pagination-disabled{box-shadow:none;cursor:default}.mat-tab-list{flex-grow:1;position:relative;transition:transform 500ms cubic-bezier(0.35, 0, 0.25, 1)}.mat-tab-links{display:flex}[mat-align-tabs=center]>.mat-tab-link-container .mat-tab-links{justify-content:center}[mat-align-tabs=end]>.mat-tab-link-container .mat-tab-links{justify-content:flex-end}.mat-ink-bar{position:absolute;bottom:0;height:2px;transition:500ms cubic-bezier(0.35, 0, 0.25, 1)}._mat-animation-noopable.mat-ink-bar{transition:none;animation:none}.mat-tab-group-inverted-header .mat-ink-bar{bottom:auto;top:0}.cdk-high-contrast-active .mat-ink-bar{outline:solid 2px;height:0}.mat-tab-link-container{display:flex;flex-grow:1;overflow:hidden;z-index:1}.mat-tab-link{height:48px;padding:0 24px;cursor:pointer;box-sizing:border-box;opacity:.6;min-width:160px;text-align:center;display:inline-flex;justify-content:center;align-items:center;white-space:nowrap;vertical-align:top;text-decoration:none;position:relative;overflow:hidden;-webkit-tap-highlight-color:transparent}.mat-tab-link:focus{outline:none}.mat-tab-link:focus:not(.mat-tab-disabled){opacity:1}.cdk-high-contrast-active .mat-tab-link:focus{outline:dotted 2px;outline-offset:-2px}.mat-tab-link.mat-tab-disabled{cursor:default}.cdk-high-contrast-active .mat-tab-link.mat-tab-disabled{opacity:.5}.mat-tab-link .mat-tab-label-content{display:inline-flex;justify-content:center;align-items:center;white-space:nowrap}.cdk-high-contrast-active .mat-tab-link{opacity:1}[mat-stretch-tabs] .mat-tab-link{flex-basis:0;flex-grow:1}.mat-tab-link.mat-tab-disabled{pointer-events:none}@media(max-width: 599px){.mat-tab-link{min-width:72px}}\n"], encapsulation: 2 });
    /** @nocollapse */
    MatTabNav.ctorParameters = () => [
        { type: ElementRef },
        { type: Directionality, decorators: [{ type: Optional }] },
        { type: NgZone },
        { type: ChangeDetectorRef },
        { type: ViewportRuler },
        { type: Platform, decorators: [{ type: Optional }] },
        { type: String, decorators: [{ type: Optional }, { type: Inject, args: [ANIMATION_MODULE_TYPE,] }] }
    ];
    MatTabNav.propDecorators = {
        _items: [{ type: ContentChildren, args: [forwardRef((/**
                     * @return {?}
                     */
                    () => MatTabLink)), { descendants: true },] }],
        _inkBar: [{ type: ViewChild, args: [MatInkBar, { static: true },] }],
        _tabListContainer: [{ type: ViewChild, args: ['tabListContainer', { static: true },] }],
        _tabList: [{ type: ViewChild, args: ['tabList', { static: true },] }],
        _nextPaginator: [{ type: ViewChild, args: ['nextPaginator',] }],
        _previousPaginator: [{ type: ViewChild, args: ['previousPaginator',] }]
    };
    // Boilerplate for applying mixins to MatTabLink.
    class MatTabLinkMixinBase {
    }
    /** @type {?} */
    const _MatTabLinkMixinBase = mixinTabIndex(mixinDisableRipple(mixinDisabled(MatTabLinkMixinBase)));
    /**
     * Base class with all of the `MatTabLink` functionality.
     */
    // tslint:disable-next-line:class-name
    class _MatTabLinkBase extends _MatTabLinkMixinBase {
        /**
         * @param {?} _tabNavBar
         * @param {?} elementRef
         * @param {?} globalRippleOptions
         * @param {?} tabIndex
         * @param {?} _focusMonitor
         * @param {?=} animationMode
         */
        constructor(_tabNavBar, elementRef, globalRippleOptions, tabIndex, _focusMonitor, animationMode) {
            super();
            this._tabNavBar = _tabNavBar;
            this.elementRef = elementRef;
            this._focusMonitor = _focusMonitor;
            /**
             * Whether the tab link is active or not.
             */
            this._isActive = false;
            this.rippleConfig = globalRippleOptions || {};
            this.tabIndex = parseInt(tabIndex) || 0;
            if (animationMode === 'NoopAnimations') {
                this.rippleConfig.animation = { enterDuration: 0, exitDuration: 0 };
            }
            _focusMonitor.monitor(elementRef);
        }
        /**
         * Whether the link is active.
         * @return {?}
         */
        get active() { return this._isActive; }
        /**
         * @param {?} value
         * @return {?}
         */
        set active(value) {
            if (value !== this._isActive) {
                this._isActive = value;
                this._tabNavBar.updateActiveLink(this.elementRef);
            }
        }
        /**
         * Whether ripples are disabled on interaction.
         * \@docs-private
         * @return {?}
         */
        get rippleDisabled() {
            return this.disabled || this.disableRipple || this._tabNavBar.disableRipple ||
                !!this.rippleConfig.disabled;
        }
        /**
         * @return {?}
         */
        focus() {
            this.elementRef.nativeElement.focus();
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this._focusMonitor.stopMonitoring(this.elementRef);
        }
    }
    _MatTabLinkBase.ɵfac = function _MatTabLinkBase_Factory(t) { return new (t || _MatTabLinkBase)(ɵɵdirectiveInject(_MatTabNavBase), ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(MAT_RIPPLE_GLOBAL_OPTIONS, 8), ɵɵinjectAttribute('tabindex'), ɵɵdirectiveInject(FocusMonitor), ɵɵdirectiveInject(ANIMATION_MODULE_TYPE, 8)); };
    _MatTabLinkBase.ɵdir = ɵɵdefineDirective({ type: _MatTabLinkBase, inputs: { active: "active" }, features: [ɵɵInheritDefinitionFeature] });
    /** @nocollapse */
    _MatTabLinkBase.ctorParameters = () => [
        { type: _MatTabNavBase },
        { type: ElementRef },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [MAT_RIPPLE_GLOBAL_OPTIONS,] }] },
        { type: String, decorators: [{ type: Attribute, args: ['tabindex',] }] },
        { type: FocusMonitor },
        { type: String, decorators: [{ type: Optional }, { type: Inject, args: [ANIMATION_MODULE_TYPE,] }] }
    ];
    _MatTabLinkBase.propDecorators = {
        active: [{ type: Input }]
    };
    /**
     * Link inside of a `mat-tab-nav-bar`.
     */
    class MatTabLink extends _MatTabLinkBase {
        /**
         * @param {?} tabNavBar
         * @param {?} elementRef
         * @param {?} ngZone
         * @param {?} platform
         * @param {?} globalRippleOptions
         * @param {?} tabIndex
         * @param {?} focusMonitor
         * @param {?=} animationMode
         */
        constructor(tabNavBar, elementRef, ngZone, platform, globalRippleOptions, tabIndex, focusMonitor, animationMode) {
            super(tabNavBar, elementRef, globalRippleOptions, tabIndex, focusMonitor, animationMode);
            this._tabLinkRipple = new RippleRenderer(this, ngZone, elementRef, platform);
            this._tabLinkRipple.setupTriggerEvents(elementRef.nativeElement);
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            super.ngOnDestroy();
            this._tabLinkRipple._removeTriggerEvents();
        }
    }
    MatTabLink.ɵfac = function MatTabLink_Factory(t) { return new (t || MatTabLink)(ɵɵdirectiveInject(MatTabNav), ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(NgZone), ɵɵdirectiveInject(Platform), ɵɵdirectiveInject(MAT_RIPPLE_GLOBAL_OPTIONS, 8), ɵɵinjectAttribute('tabindex'), ɵɵdirectiveInject(FocusMonitor), ɵɵdirectiveInject(ANIMATION_MODULE_TYPE, 8)); };
    MatTabLink.ɵdir = ɵɵdefineDirective({ type: MatTabLink, selectors: [["", "mat-tab-link", ""], ["", "matTabLink", ""]], hostAttrs: [1, "mat-tab-link", "mat-focus-indicator"], hostVars: 7, hostBindings: function MatTabLink_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵattribute("aria-current", ctx.active ? "page" : null)("aria-disabled", ctx.disabled)("tabIndex", ctx.tabIndex);
            ɵɵclassProp("mat-tab-disabled", ctx.disabled)("mat-tab-label-active", ctx.active);
        } }, inputs: { disabled: "disabled", disableRipple: "disableRipple", tabIndex: "tabIndex" }, exportAs: ["matTabLink"], features: [ɵɵInheritDefinitionFeature] });
    /** @nocollapse */
    MatTabLink.ctorParameters = () => [
        { type: MatTabNav },
        { type: ElementRef },
        { type: NgZone },
        { type: Platform },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [MAT_RIPPLE_GLOBAL_OPTIONS,] }] },
        { type: String, decorators: [{ type: Attribute, args: ['tabindex',] }] },
        { type: FocusMonitor },
        { type: String, decorators: [{ type: Optional }, { type: Inject, args: [ANIMATION_MODULE_TYPE,] }] }
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/tabs/tabs-module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class MatTabsModule {
    }
    MatTabsModule.ɵmod = ɵɵdefineNgModule({ type: MatTabsModule });
    MatTabsModule.ɵinj = ɵɵdefineInjector({ factory: function MatTabsModule_Factory(t) { return new (t || MatTabsModule)(); }, imports: [[
                CommonModule,
                MatCommonModule,
                PortalModule,
                MatRippleModule,
                ObserversModule,
                A11yModule,
            ],
            MatCommonModule] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(MatTabsModule, { declarations: function () { return [MatTabGroup,
            MatTabLabel,
            MatTab,
            MatInkBar,
            MatTabLabelWrapper,
            MatTabNav,
            MatTabLink,
            MatTabBody,
            MatTabBodyPortal,
            MatTabHeader,
            MatTabContent]; }, imports: function () { return [CommonModule,
            MatCommonModule,
            PortalModule,
            MatRippleModule,
            ObserversModule,
            A11yModule]; }, exports: function () { return [MatCommonModule,
            MatTabGroup,
            MatTabLabel,
            MatTab,
            MatTabNav,
            MatTabLink,
            MatTabContent]; } }); })();

    const _c0$4 = ["underline"];
    const _c1$4 = ["connectionContainer"];
    const _c2$2 = ["inputContainer"];
    const _c3$1 = ["label"];
    function MatFormField_ng_container_3_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementContainerStart(0);
        ɵɵelementStart(1, "div", 14);
        ɵɵelement(2, "div", 15);
        ɵɵelement(3, "div", 16);
        ɵɵelement(4, "div", 17);
        ɵɵelementEnd();
        ɵɵelementStart(5, "div", 18);
        ɵɵelement(6, "div", 15);
        ɵɵelement(7, "div", 16);
        ɵɵelement(8, "div", 17);
        ɵɵelementEnd();
        ɵɵelementContainerEnd();
    } }
    function MatFormField_div_4_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 19);
        ɵɵprojection(1, 1);
        ɵɵelementEnd();
    } }
    function MatFormField_label_9_ng_container_2_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementContainerStart(0);
        ɵɵprojection(1, 2);
        ɵɵelementStart(2, "span");
        ɵɵtext(3);
        ɵɵelementEnd();
        ɵɵelementContainerEnd();
    } if (rf & 2) {
        const ctx_r10 = ɵɵnextContext(2);
        ɵɵadvance(3);
        ɵɵtextInterpolate(ctx_r10._control.placeholder);
    } }
    function MatFormField_label_9_3_Template(rf, ctx) { if (rf & 1) {
        ɵɵprojection(0, 3, ["*ngSwitchCase", "true"]);
    } }
    function MatFormField_label_9_span_4_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "span", 23);
        ɵɵtext(1, " *");
        ɵɵelementEnd();
    } }
    function MatFormField_label_9_Template(rf, ctx) { if (rf & 1) {
        const _r14 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "label", 20, 21);
        ɵɵlistener("cdkObserveContent", function MatFormField_label_9_Template_label_cdkObserveContent_0_listener() { ɵɵrestoreView(_r14); const ctx_r13 = ɵɵnextContext(); return ctx_r13.updateOutlineGap(); });
        ɵɵtemplate(2, MatFormField_label_9_ng_container_2_Template, 4, 1, "ng-container", 12);
        ɵɵtemplate(3, MatFormField_label_9_3_Template, 1, 0, undefined, 12);
        ɵɵtemplate(4, MatFormField_label_9_span_4_Template, 2, 0, "span", 22);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r4 = ɵɵnextContext();
        ɵɵclassProp("mat-empty", ctx_r4._control.empty && !ctx_r4._shouldAlwaysFloat)("mat-form-field-empty", ctx_r4._control.empty && !ctx_r4._shouldAlwaysFloat)("mat-accent", ctx_r4.color == "accent")("mat-warn", ctx_r4.color == "warn");
        ɵɵproperty("cdkObserveContentDisabled", ctx_r4.appearance != "outline")("id", ctx_r4._labelId)("ngSwitch", ctx_r4._hasLabel());
        ɵɵattribute("for", ctx_r4._control.id)("aria-owns", ctx_r4._control.id);
        ɵɵadvance(2);
        ɵɵproperty("ngSwitchCase", false);
        ɵɵadvance(1);
        ɵɵproperty("ngSwitchCase", true);
        ɵɵadvance(1);
        ɵɵproperty("ngIf", !ctx_r4.hideRequiredMarker && ctx_r4._control.required && !ctx_r4._control.disabled);
    } }
    function MatFormField_div_10_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 24);
        ɵɵprojection(1, 4);
        ɵɵelementEnd();
    } }
    function MatFormField_div_11_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 25, 26);
        ɵɵelement(2, "span", 27);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r6 = ɵɵnextContext();
        ɵɵadvance(2);
        ɵɵclassProp("mat-accent", ctx_r6.color == "accent")("mat-warn", ctx_r6.color == "warn");
    } }
    function MatFormField_div_13_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div");
        ɵɵprojection(1, 5);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r7 = ɵɵnextContext();
        ɵɵproperty("@transitionMessages", ctx_r7._subscriptAnimationState);
    } }
    function MatFormField_div_14_div_1_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 31);
        ɵɵtext(1);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r16 = ɵɵnextContext(2);
        ɵɵproperty("id", ctx_r16._hintLabelId);
        ɵɵadvance(1);
        ɵɵtextInterpolate(ctx_r16.hintLabel);
    } }
    function MatFormField_div_14_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 28);
        ɵɵtemplate(1, MatFormField_div_14_div_1_Template, 2, 2, "div", 29);
        ɵɵprojection(2, 6);
        ɵɵelement(3, "div", 30);
        ɵɵprojection(4, 7);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r8 = ɵɵnextContext();
        ɵɵproperty("@transitionMessages", ctx_r8._subscriptAnimationState);
        ɵɵadvance(1);
        ɵɵproperty("ngIf", ctx_r8.hintLabel);
    } }
    const _c4$1 = ["*", [["", "matPrefix", ""]], [["mat-placeholder"]], [["mat-label"]], [["", "matSuffix", ""]], [["mat-error"]], [["mat-hint", 3, "align", "end"]], [["mat-hint", "align", "end"]]];
    const _c5$1 = ["*", "[matPrefix]", "mat-placeholder", "mat-label", "[matSuffix]", "mat-error", "mat-hint:not([align='end'])", "mat-hint[align='end']"];
    let nextUniqueId = 0;
    /**
     * Single error message to be shown underneath the form field.
     */
    class MatError {
        constructor() {
            this.id = `mat-error-${nextUniqueId++}`;
        }
    }
    MatError.ɵfac = function MatError_Factory(t) { return new (t || MatError)(); };
    MatError.ɵdir = ɵɵdefineDirective({ type: MatError, selectors: [["mat-error"]], hostAttrs: ["role", "alert", 1, "mat-error"], hostVars: 1, hostBindings: function MatError_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵattribute("id", ctx.id);
        } }, inputs: { id: "id" } });
    MatError.propDecorators = {
        id: [{ type: Input }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/form-field/form-field-animations.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Animations used by the MatFormField.
     * \@docs-private
     * @type {?}
     */
    const matFormFieldAnimations = {
        /**
         * Animation that transitions the form field's error and hint messages.
         */
        transitionMessages: trigger('transitionMessages', [
            // TODO(mmalerba): Use angular animations for label animation as well.
            state('enter', style({ opacity: 1, transform: 'translateY(0%)' })),
            transition('void => enter', [
                style({ opacity: 0, transform: 'translateY(-100%)' }),
                animate('300ms cubic-bezier(0.55, 0, 0.55, 0.2)'),
            ]),
        ])
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/form-field/form-field-control.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * An interface which allows a control to work inside of a `MatFormField`.
     * @abstract
     * @template T
     */
    class MatFormFieldControl {
    }
    MatFormFieldControl.ɵfac = function MatFormFieldControl_Factory(t) { return new (t || MatFormFieldControl)(); };
    MatFormFieldControl.ɵdir = ɵɵdefineDirective({ type: MatFormFieldControl });

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/form-field/form-field-errors.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google LLC All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * \@docs-private
     * @return {?}
     */
    function getMatFormFieldPlaceholderConflictError() {
        return Error('Placeholder attribute and child element were both specified.');
    }
    /**
     * \@docs-private
     * @param {?} align
     * @return {?}
     */
    function getMatFormFieldDuplicatedHintError(align) {
        return Error(`A hint was already declared for 'align="${align}"'.`);
    }
    /**
     * \@docs-private
     * @return {?}
     */
    function getMatFormFieldMissingControlError() {
        return Error('mat-form-field must contain a MatFormFieldControl.');
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/form-field/hint.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    let nextUniqueId$1 = 0;
    /**
     * Hint text to be shown underneath the form field control.
     */
    class MatHint {
        constructor() {
            /**
             * Whether to align the hint label at the start or end of the line.
             */
            this.align = 'start';
            /**
             * Unique ID for the hint. Used for the aria-describedby on the form field control.
             */
            this.id = `mat-hint-${nextUniqueId$1++}`;
        }
    }
    MatHint.ɵfac = function MatHint_Factory(t) { return new (t || MatHint)(); };
    MatHint.ɵdir = ɵɵdefineDirective({ type: MatHint, selectors: [["mat-hint"]], hostAttrs: [1, "mat-hint"], hostVars: 4, hostBindings: function MatHint_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵattribute("id", ctx.id)("align", null);
            ɵɵclassProp("mat-right", ctx.align == "end");
        } }, inputs: { align: "align", id: "id" } });
    MatHint.propDecorators = {
        align: [{ type: Input }],
        id: [{ type: Input }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/form-field/label.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * The floating label for a `mat-form-field`.
     */
    class MatLabel {
    }
    MatLabel.ɵfac = function MatLabel_Factory(t) { return new (t || MatLabel)(); };
    MatLabel.ɵdir = ɵɵdefineDirective({ type: MatLabel, selectors: [["mat-label"]] });

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/form-field/placeholder.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * The placeholder text for an `MatFormField`.
     * @deprecated Use `<mat-label>` to specify the label and the `placeholder` attribute to specify the
     *     placeholder.
     * \@breaking-change 8.0.0
     */
    class MatPlaceholder {
    }
    MatPlaceholder.ɵfac = function MatPlaceholder_Factory(t) { return new (t || MatPlaceholder)(); };
    MatPlaceholder.ɵdir = ɵɵdefineDirective({ type: MatPlaceholder, selectors: [["mat-placeholder"]] });

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/form-field/prefix.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Prefix to be placed in front of the form field.
     */
    class MatPrefix {
    }
    MatPrefix.ɵfac = function MatPrefix_Factory(t) { return new (t || MatPrefix)(); };
    MatPrefix.ɵdir = ɵɵdefineDirective({ type: MatPrefix, selectors: [["", "matPrefix", ""]] });

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/form-field/suffix.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Suffix to be placed at the end of the form field.
     */
    class MatSuffix {
    }
    MatSuffix.ɵfac = function MatSuffix_Factory(t) { return new (t || MatSuffix)(); };
    MatSuffix.ɵdir = ɵɵdefineDirective({ type: MatSuffix, selectors: [["", "matSuffix", ""]] });

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/form-field/form-field.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    let nextUniqueId$2 = 0;
    /** @type {?} */
    const floatingLabelScale = 0.75;
    /** @type {?} */
    const outlineGapPadding = 5;
    /**
     * Boilerplate for applying mixins to MatFormField.
     * \@docs-private
     */
    class MatFormFieldBase {
        /**
         * @param {?} _elementRef
         */
        constructor(_elementRef) {
            this._elementRef = _elementRef;
        }
    }
    /**
     * Base class to which we're applying the form field mixins.
     * \@docs-private
     * @type {?}
     */
    const _MatFormFieldMixinBase = mixinColor(MatFormFieldBase, 'primary');
    /**
     * Injection token that can be used to configure the
     * default options for all form field within an app.
     * @type {?}
     */
    const MAT_FORM_FIELD_DEFAULT_OPTIONS = new InjectionToken('MAT_FORM_FIELD_DEFAULT_OPTIONS');
    /**
     * Injection token that can be used to inject an instances of `MatFormField`. It serves
     * as alternative token to the actual `MatFormField` class which would cause unnecessary
     * retention of the `MatFormField` class and its component metadata.
     * @type {?}
     */
    const MAT_FORM_FIELD = new InjectionToken('MatFormField');
    /**
     * Container for form controls that applies Material Design styling and behavior.
     */
    class MatFormField extends _MatFormFieldMixinBase {
        /**
         * @param {?} _elementRef
         * @param {?} _changeDetectorRef
         * @param {?} labelOptions
         * @param {?} _dir
         * @param {?} _defaults
         * @param {?} _platform
         * @param {?} _ngZone
         * @param {?} _animationMode
         */
        constructor(_elementRef, _changeDetectorRef, labelOptions, _dir, _defaults, _platform, _ngZone, _animationMode) {
            super(_elementRef);
            this._elementRef = _elementRef;
            this._changeDetectorRef = _changeDetectorRef;
            this._dir = _dir;
            this._defaults = _defaults;
            this._platform = _platform;
            this._ngZone = _ngZone;
            /**
             * Whether the outline gap needs to be calculated
             * immediately on the next change detection run.
             */
            this._outlineGapCalculationNeededImmediately = false;
            /**
             * Whether the outline gap needs to be calculated next time the zone has stabilized.
             */
            this._outlineGapCalculationNeededOnStable = false;
            this._destroyed = new Subject();
            /**
             * Override for the logic that disables the label animation in certain cases.
             */
            this._showAlwaysAnimate = false;
            /**
             * State of the mat-hint and mat-error animations.
             */
            this._subscriptAnimationState = '';
            this._hintLabel = '';
            // Unique id for the hint label.
            this._hintLabelId = `mat-hint-${nextUniqueId$2++}`;
            // Unique id for the internal form field label.
            this._labelId = `mat-form-field-label-${nextUniqueId$2++}`;
            this._labelOptions = labelOptions ? labelOptions : {};
            this.floatLabel = this._getDefaultFloatLabelState();
            this._animationsEnabled = _animationMode !== 'NoopAnimations';
            // Set the default through here so we invoke the setter on the first run.
            this.appearance = (_defaults && _defaults.appearance) ? _defaults.appearance : 'legacy';
            this._hideRequiredMarker = (_defaults && _defaults.hideRequiredMarker != null) ?
                _defaults.hideRequiredMarker : false;
        }
        /**
         * The form-field appearance style.
         * @return {?}
         */
        get appearance() { return this._appearance; }
        /**
         * @param {?} value
         * @return {?}
         */
        set appearance(value) {
            /** @type {?} */
            const oldValue = this._appearance;
            this._appearance = value || (this._defaults && this._defaults.appearance) || 'legacy';
            if (this._appearance === 'outline' && oldValue !== value) {
                this._outlineGapCalculationNeededOnStable = true;
            }
        }
        /**
         * Whether the required marker should be hidden.
         * @return {?}
         */
        get hideRequiredMarker() { return this._hideRequiredMarker; }
        /**
         * @param {?} value
         * @return {?}
         */
        set hideRequiredMarker(value) {
            this._hideRequiredMarker = coerceBooleanProperty(value);
        }
        /**
         * Whether the floating label should always float or not.
         * @return {?}
         */
        get _shouldAlwaysFloat() {
            return this.floatLabel === 'always' && !this._showAlwaysAnimate;
        }
        /**
         * Whether the label can float or not.
         * @return {?}
         */
        get _canLabelFloat() { return this.floatLabel !== 'never'; }
        /**
         * Text for the form field hint.
         * @return {?}
         */
        get hintLabel() { return this._hintLabel; }
        /**
         * @param {?} value
         * @return {?}
         */
        set hintLabel(value) {
            this._hintLabel = value;
            this._processHints();
        }
        /**
         * Whether the label should always float, never float or float as the user types.
         *
         * Note: only the legacy appearance supports the `never` option. `never` was originally added as a
         * way to make the floating label emulate the behavior of a standard input placeholder. However
         * the form field now supports both floating labels and placeholders. Therefore in the non-legacy
         * appearances the `never` option has been disabled in favor of just using the placeholder.
         * @return {?}
         */
        get floatLabel() {
            return this.appearance !== 'legacy' && this._floatLabel === 'never' ? 'auto' : this._floatLabel;
        }
        /**
         * @param {?} value
         * @return {?}
         */
        set floatLabel(value) {
            if (value !== this._floatLabel) {
                this._floatLabel = value || this._getDefaultFloatLabelState();
                this._changeDetectorRef.markForCheck();
            }
        }
        /**
         * @return {?}
         */
        get _control() {
            // TODO(crisbeto): we need this workaround in order to support both Ivy and ViewEngine.
            //  We should clean this up once Ivy is the default renderer.
            return this._explicitFormFieldControl || this._controlNonStatic || this._controlStatic;
        }
        /**
         * @param {?} value
         * @return {?}
         */
        set _control(value) {
            this._explicitFormFieldControl = value;
        }
        /**
         * @return {?}
         */
        get _labelChild() {
            return this._labelChildNonStatic || this._labelChildStatic;
        }
        /**
         * Gets an ElementRef for the element that a overlay attached to the form-field should be
         * positioned relative to.
         * @return {?}
         */
        getConnectedOverlayOrigin() {
            return this._connectionContainerRef || this._elementRef;
        }
        /**
         * @return {?}
         */
        ngAfterContentInit() {
            this._validateControlChild();
            /** @type {?} */
            const control = this._control;
            if (control.controlType) {
                this._elementRef.nativeElement.classList.add(`mat-form-field-type-${control.controlType}`);
            }
            // Subscribe to changes in the child control state in order to update the form field UI.
            control.stateChanges.pipe(startWith((/** @type {?} */ (null)))).subscribe((/**
             * @return {?}
             */
            () => {
                this._validatePlaceholders();
                this._syncDescribedByIds();
                this._changeDetectorRef.markForCheck();
            }));
            // Run change detection if the value changes.
            if (control.ngControl && control.ngControl.valueChanges) {
                control.ngControl.valueChanges
                    .pipe(takeUntil(this._destroyed))
                    .subscribe((/**
                 * @return {?}
                 */
                () => this._changeDetectorRef.markForCheck()));
            }
            // Note that we have to run outside of the `NgZone` explicitly,
            // in order to avoid throwing users into an infinite loop
            // if `zone-patch-rxjs` is included.
            this._ngZone.runOutsideAngular((/**
             * @return {?}
             */
            () => {
                this._ngZone.onStable.asObservable().pipe(takeUntil(this._destroyed)).subscribe((/**
                 * @return {?}
                 */
                () => {
                    if (this._outlineGapCalculationNeededOnStable) {
                        this.updateOutlineGap();
                    }
                }));
            }));
            // Run change detection and update the outline if the suffix or prefix changes.
            merge(this._prefixChildren.changes, this._suffixChildren.changes).subscribe((/**
             * @return {?}
             */
            () => {
                this._outlineGapCalculationNeededOnStable = true;
                this._changeDetectorRef.markForCheck();
            }));
            // Re-validate when the number of hints changes.
            this._hintChildren.changes.pipe(startWith(null)).subscribe((/**
             * @return {?}
             */
            () => {
                this._processHints();
                this._changeDetectorRef.markForCheck();
            }));
            // Update the aria-described by when the number of errors changes.
            this._errorChildren.changes.pipe(startWith(null)).subscribe((/**
             * @return {?}
             */
            () => {
                this._syncDescribedByIds();
                this._changeDetectorRef.markForCheck();
            }));
            if (this._dir) {
                this._dir.change.pipe(takeUntil(this._destroyed)).subscribe((/**
                 * @return {?}
                 */
                () => {
                    if (typeof requestAnimationFrame === 'function') {
                        this._ngZone.runOutsideAngular((/**
                         * @return {?}
                         */
                        () => {
                            requestAnimationFrame((/**
                             * @return {?}
                             */
                            () => this.updateOutlineGap()));
                        }));
                    }
                    else {
                        this.updateOutlineGap();
                    }
                }));
            }
        }
        /**
         * @return {?}
         */
        ngAfterContentChecked() {
            this._validateControlChild();
            if (this._outlineGapCalculationNeededImmediately) {
                this.updateOutlineGap();
            }
        }
        /**
         * @return {?}
         */
        ngAfterViewInit() {
            // Avoid animations on load.
            this._subscriptAnimationState = 'enter';
            this._changeDetectorRef.detectChanges();
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this._destroyed.next();
            this._destroyed.complete();
        }
        /**
         * Determines whether a class from the NgControl should be forwarded to the host element.
         * @param {?} prop
         * @return {?}
         */
        _shouldForward(prop) {
            /** @type {?} */
            const ngControl = this._control ? this._control.ngControl : null;
            return ngControl && ngControl[prop];
        }
        /**
         * @return {?}
         */
        _hasPlaceholder() {
            return !!(this._control && this._control.placeholder || this._placeholderChild);
        }
        /**
         * @return {?}
         */
        _hasLabel() {
            return !!this._labelChild;
        }
        /**
         * @return {?}
         */
        _shouldLabelFloat() {
            return this._canLabelFloat && (this._control.shouldLabelFloat || this._shouldAlwaysFloat);
        }
        /**
         * @return {?}
         */
        _hideControlPlaceholder() {
            // In the legacy appearance the placeholder is promoted to a label if no label is given.
            return this.appearance === 'legacy' && !this._hasLabel() ||
                this._hasLabel() && !this._shouldLabelFloat();
        }
        /**
         * @return {?}
         */
        _hasFloatingLabel() {
            // In the legacy appearance the placeholder is promoted to a label if no label is given.
            return this._hasLabel() || this.appearance === 'legacy' && this._hasPlaceholder();
        }
        /**
         * Determines whether to display hints or errors.
         * @return {?}
         */
        _getDisplayedMessages() {
            return (this._errorChildren && this._errorChildren.length > 0 &&
                this._control.errorState) ? 'error' : 'hint';
        }
        /**
         * Animates the placeholder up and locks it in position.
         * @return {?}
         */
        _animateAndLockLabel() {
            if (this._hasFloatingLabel() && this._canLabelFloat) {
                // If animations are disabled, we shouldn't go in here,
                // because the `transitionend` will never fire.
                if (this._animationsEnabled && this._label) {
                    this._showAlwaysAnimate = true;
                    fromEvent(this._label.nativeElement, 'transitionend').pipe(take(1)).subscribe((/**
                     * @return {?}
                     */
                    () => {
                        this._showAlwaysAnimate = false;
                    }));
                }
                this.floatLabel = 'always';
                this._changeDetectorRef.markForCheck();
            }
        }
        /**
         * Ensure that there is only one placeholder (either `placeholder` attribute on the child control
         * or child element with the `mat-placeholder` directive).
         * @private
         * @return {?}
         */
        _validatePlaceholders() {
            if (this._control.placeholder && this._placeholderChild) {
                throw getMatFormFieldPlaceholderConflictError();
            }
        }
        /**
         * Does any extra processing that is required when handling the hints.
         * @private
         * @return {?}
         */
        _processHints() {
            this._validateHints();
            this._syncDescribedByIds();
        }
        /**
         * Ensure that there is a maximum of one of each `<mat-hint>` alignment specified, with the
         * attribute being considered as `align="start"`.
         * @private
         * @return {?}
         */
        _validateHints() {
            if (this._hintChildren) {
                /** @type {?} */
                let startHint;
                /** @type {?} */
                let endHint;
                this._hintChildren.forEach((/**
                 * @param {?} hint
                 * @return {?}
                 */
                (hint) => {
                    if (hint.align === 'start') {
                        if (startHint || this.hintLabel) {
                            throw getMatFormFieldDuplicatedHintError('start');
                        }
                        startHint = hint;
                    }
                    else if (hint.align === 'end') {
                        if (endHint) {
                            throw getMatFormFieldDuplicatedHintError('end');
                        }
                        endHint = hint;
                    }
                }));
            }
        }
        /**
         * Gets the default float label state.
         * @private
         * @return {?}
         */
        _getDefaultFloatLabelState() {
            return (this._defaults && this._defaults.floatLabel) || this._labelOptions.float || 'auto';
        }
        /**
         * Sets the list of element IDs that describe the child control. This allows the control to update
         * its `aria-describedby` attribute accordingly.
         * @private
         * @return {?}
         */
        _syncDescribedByIds() {
            if (this._control) {
                /** @type {?} */
                let ids = [];
                if (this._getDisplayedMessages() === 'hint') {
                    /** @type {?} */
                    const startHint = this._hintChildren ?
                        this._hintChildren.find((/**
                         * @param {?} hint
                         * @return {?}
                         */
                        hint => hint.align === 'start')) : null;
                    /** @type {?} */
                    const endHint = this._hintChildren ?
                        this._hintChildren.find((/**
                         * @param {?} hint
                         * @return {?}
                         */
                        hint => hint.align === 'end')) : null;
                    if (startHint) {
                        ids.push(startHint.id);
                    }
                    else if (this._hintLabel) {
                        ids.push(this._hintLabelId);
                    }
                    if (endHint) {
                        ids.push(endHint.id);
                    }
                }
                else if (this._errorChildren) {
                    ids = this._errorChildren.map((/**
                     * @param {?} error
                     * @return {?}
                     */
                    error => error.id));
                }
                this._control.setDescribedByIds(ids);
            }
        }
        /**
         * Throws an error if the form field's control is missing.
         * @protected
         * @return {?}
         */
        _validateControlChild() {
            if (!this._control) {
                throw getMatFormFieldMissingControlError();
            }
        }
        /**
         * Updates the width and position of the gap in the outline. Only relevant for the outline
         * appearance.
         * @return {?}
         */
        updateOutlineGap() {
            /** @type {?} */
            const labelEl = this._label ? this._label.nativeElement : null;
            if (this.appearance !== 'outline' || !labelEl || !labelEl.children.length ||
                !labelEl.textContent.trim()) {
                return;
            }
            if (!this._platform.isBrowser) {
                // getBoundingClientRect isn't available on the server.
                return;
            }
            // If the element is not present in the DOM, the outline gap will need to be calculated
            // the next time it is checked and in the DOM.
            if (!this._isAttachedToDOM()) {
                this._outlineGapCalculationNeededImmediately = true;
                return;
            }
            /** @type {?} */
            let startWidth = 0;
            /** @type {?} */
            let gapWidth = 0;
            /** @type {?} */
            const container = this._connectionContainerRef.nativeElement;
            /** @type {?} */
            const startEls = container.querySelectorAll('.mat-form-field-outline-start');
            /** @type {?} */
            const gapEls = container.querySelectorAll('.mat-form-field-outline-gap');
            if (this._label && this._label.nativeElement.children.length) {
                /** @type {?} */
                const containerRect = container.getBoundingClientRect();
                // If the container's width and height are zero, it means that the element is
                // invisible and we can't calculate the outline gap. Mark the element as needing
                // to be checked the next time the zone stabilizes. We can't do this immediately
                // on the next change detection, because even if the element becomes visible,
                // the `ClientRect` won't be reclaculated immediately. We reset the
                // `_outlineGapCalculationNeededImmediately` flag some we don't run the checks twice.
                if (containerRect.width === 0 && containerRect.height === 0) {
                    this._outlineGapCalculationNeededOnStable = true;
                    this._outlineGapCalculationNeededImmediately = false;
                    return;
                }
                /** @type {?} */
                const containerStart = this._getStartEnd(containerRect);
                /** @type {?} */
                const labelStart = this._getStartEnd(labelEl.children[0].getBoundingClientRect());
                /** @type {?} */
                let labelWidth = 0;
                for (const child of labelEl.children) {
                    labelWidth += child.offsetWidth;
                }
                startWidth = Math.abs(labelStart - containerStart) - outlineGapPadding;
                gapWidth = labelWidth > 0 ? labelWidth * floatingLabelScale + outlineGapPadding * 2 : 0;
            }
            for (let i = 0; i < startEls.length; i++) {
                startEls[i].style.width = `${startWidth}px`;
            }
            for (let i = 0; i < gapEls.length; i++) {
                gapEls[i].style.width = `${gapWidth}px`;
            }
            this._outlineGapCalculationNeededOnStable =
                this._outlineGapCalculationNeededImmediately = false;
        }
        /**
         * Gets the start end of the rect considering the current directionality.
         * @private
         * @param {?} rect
         * @return {?}
         */
        _getStartEnd(rect) {
            return (this._dir && this._dir.value === 'rtl') ? rect.right : rect.left;
        }
        /**
         * Checks whether the form field is attached to the DOM.
         * @private
         * @return {?}
         */
        _isAttachedToDOM() {
            /** @type {?} */
            const element = this._elementRef.nativeElement;
            if (element.getRootNode) {
                /** @type {?} */
                const rootNode = element.getRootNode();
                // If the element is inside the DOM the root node will be either the document
                // or the closest shadow root, otherwise it'll be the element itself.
                return rootNode && rootNode !== element;
            }
            // Otherwise fall back to checking if it's in the document. This doesn't account for
            // shadow DOM, however browser that support shadow DOM should support `getRootNode` as well.
            return (/** @type {?} */ (document.documentElement)).contains(element);
        }
    }
    MatFormField.ɵfac = function MatFormField_Factory(t) { return new (t || MatFormField)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(ChangeDetectorRef), ɵɵdirectiveInject(MAT_LABEL_GLOBAL_OPTIONS, 8), ɵɵdirectiveInject(Directionality, 8), ɵɵdirectiveInject(MAT_FORM_FIELD_DEFAULT_OPTIONS, 8), ɵɵdirectiveInject(Platform), ɵɵdirectiveInject(NgZone), ɵɵdirectiveInject(ANIMATION_MODULE_TYPE, 8)); };
    MatFormField.ɵcmp = ɵɵdefineComponent({ type: MatFormField, selectors: [["mat-form-field"]], contentQueries: function MatFormField_ContentQueries(rf, ctx, dirIndex) { if (rf & 1) {
            ɵɵcontentQuery(dirIndex, MatFormFieldControl, true);
            ɵɵstaticContentQuery(dirIndex, MatFormFieldControl, true);
            ɵɵcontentQuery(dirIndex, MatLabel, true);
            ɵɵstaticContentQuery(dirIndex, MatLabel, true);
            ɵɵcontentQuery(dirIndex, MatPlaceholder, true);
            ɵɵcontentQuery(dirIndex, MatError, true);
            ɵɵcontentQuery(dirIndex, MatHint, true);
            ɵɵcontentQuery(dirIndex, MatPrefix, true);
            ɵɵcontentQuery(dirIndex, MatSuffix, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._controlNonStatic = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._controlStatic = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._labelChildNonStatic = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._labelChildStatic = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._placeholderChild = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._errorChildren = _t);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._hintChildren = _t);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._prefixChildren = _t);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._suffixChildren = _t);
        } }, viewQuery: function MatFormField_Query(rf, ctx) { if (rf & 1) {
            ɵɵviewQuery(_c0$4, true);
            ɵɵstaticViewQuery(_c1$4, true);
            ɵɵviewQuery(_c2$2, true);
            ɵɵviewQuery(_c3$1, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.underlineRef = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._connectionContainerRef = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._inputContainerRef = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._label = _t.first);
        } }, hostAttrs: [1, "mat-form-field"], hostVars: 44, hostBindings: function MatFormField_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵclassProp("mat-form-field-appearance-standard", ctx.appearance == "standard")("mat-form-field-appearance-fill", ctx.appearance == "fill")("mat-form-field-appearance-outline", ctx.appearance == "outline")("mat-form-field-appearance-legacy", ctx.appearance == "legacy")("mat-form-field-invalid", ctx._control.errorState)("mat-form-field-can-float", ctx._canLabelFloat)("mat-form-field-should-float", ctx._shouldLabelFloat())("mat-form-field-has-label", ctx._hasFloatingLabel())("mat-form-field-hide-placeholder", ctx._hideControlPlaceholder())("mat-form-field-disabled", ctx._control.disabled)("mat-form-field-autofilled", ctx._control.autofilled)("mat-focused", ctx._control.focused)("mat-accent", ctx.color == "accent")("mat-warn", ctx.color == "warn")("ng-untouched", ctx._shouldForward("untouched"))("ng-touched", ctx._shouldForward("touched"))("ng-pristine", ctx._shouldForward("pristine"))("ng-dirty", ctx._shouldForward("dirty"))("ng-valid", ctx._shouldForward("valid"))("ng-invalid", ctx._shouldForward("invalid"))("ng-pending", ctx._shouldForward("pending"))("_mat-animation-noopable", !ctx._animationsEnabled);
        } }, inputs: { color: "color", floatLabel: "floatLabel", appearance: "appearance", hideRequiredMarker: "hideRequiredMarker", hintLabel: "hintLabel" }, exportAs: ["matFormField"], features: [ɵɵProvidersFeature([
                { provide: MAT_FORM_FIELD, useExisting: MatFormField },
            ]), ɵɵInheritDefinitionFeature], ngContentSelectors: _c5$1, decls: 15, vars: 8, consts: [[1, "mat-form-field-wrapper"], [1, "mat-form-field-flex", 3, "click"], ["connectionContainer", ""], [4, "ngIf"], ["class", "mat-form-field-prefix", 4, "ngIf"], [1, "mat-form-field-infix"], ["inputContainer", ""], [1, "mat-form-field-label-wrapper"], ["class", "mat-form-field-label", 3, "cdkObserveContentDisabled", "id", "mat-empty", "mat-form-field-empty", "mat-accent", "mat-warn", "ngSwitch", "cdkObserveContent", 4, "ngIf"], ["class", "mat-form-field-suffix", 4, "ngIf"], ["class", "mat-form-field-underline", 4, "ngIf"], [1, "mat-form-field-subscript-wrapper", 3, "ngSwitch"], [4, "ngSwitchCase"], ["class", "mat-form-field-hint-wrapper", 4, "ngSwitchCase"], [1, "mat-form-field-outline"], [1, "mat-form-field-outline-start"], [1, "mat-form-field-outline-gap"], [1, "mat-form-field-outline-end"], [1, "mat-form-field-outline", "mat-form-field-outline-thick"], [1, "mat-form-field-prefix"], [1, "mat-form-field-label", 3, "cdkObserveContentDisabled", "id", "ngSwitch", "cdkObserveContent"], ["label", ""], ["class", "mat-placeholder-required mat-form-field-required-marker", "aria-hidden", "true", 4, "ngIf"], ["aria-hidden", "true", 1, "mat-placeholder-required", "mat-form-field-required-marker"], [1, "mat-form-field-suffix"], [1, "mat-form-field-underline"], ["underline", ""], [1, "mat-form-field-ripple"], [1, "mat-form-field-hint-wrapper"], ["class", "mat-hint", 3, "id", 4, "ngIf"], [1, "mat-form-field-hint-spacer"], [1, "mat-hint", 3, "id"]], template: function MatFormField_Template(rf, ctx) { if (rf & 1) {
            ɵɵprojectionDef(_c4$1);
            ɵɵelementStart(0, "div", 0);
            ɵɵelementStart(1, "div", 1, 2);
            ɵɵlistener("click", function MatFormField_Template_div_click_1_listener($event) { return ctx._control.onContainerClick && ctx._control.onContainerClick($event); });
            ɵɵtemplate(3, MatFormField_ng_container_3_Template, 9, 0, "ng-container", 3);
            ɵɵtemplate(4, MatFormField_div_4_Template, 2, 0, "div", 4);
            ɵɵelementStart(5, "div", 5, 6);
            ɵɵprojection(7);
            ɵɵelementStart(8, "span", 7);
            ɵɵtemplate(9, MatFormField_label_9_Template, 5, 16, "label", 8);
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵtemplate(10, MatFormField_div_10_Template, 2, 0, "div", 9);
            ɵɵelementEnd();
            ɵɵtemplate(11, MatFormField_div_11_Template, 3, 4, "div", 10);
            ɵɵelementStart(12, "div", 11);
            ɵɵtemplate(13, MatFormField_div_13_Template, 2, 1, "div", 12);
            ɵɵtemplate(14, MatFormField_div_14_Template, 5, 2, "div", 13);
            ɵɵelementEnd();
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵadvance(3);
            ɵɵproperty("ngIf", ctx.appearance == "outline");
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx._prefixChildren.length);
            ɵɵadvance(5);
            ɵɵproperty("ngIf", ctx._hasFloatingLabel());
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx._suffixChildren.length);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx.appearance != "outline");
            ɵɵadvance(1);
            ɵɵproperty("ngSwitch", ctx._getDisplayedMessages());
            ɵɵadvance(1);
            ɵɵproperty("ngSwitchCase", "error");
            ɵɵadvance(1);
            ɵɵproperty("ngSwitchCase", "hint");
        } }, directives: [NgIf, NgSwitch, NgSwitchCase, CdkObserveContent], styles: [".mat-form-field{display:inline-block;position:relative;text-align:left}[dir=rtl] .mat-form-field{text-align:right}.mat-form-field-wrapper{position:relative}.mat-form-field-flex{display:inline-flex;align-items:baseline;box-sizing:border-box;width:100%}.mat-form-field-prefix,.mat-form-field-suffix{white-space:nowrap;flex:none;position:relative}.mat-form-field-infix{display:block;position:relative;flex:auto;min-width:0;width:180px}.cdk-high-contrast-active .mat-form-field-infix{border-image:linear-gradient(transparent, transparent)}.mat-form-field-label-wrapper{position:absolute;left:0;box-sizing:content-box;width:100%;height:100%;overflow:hidden;pointer-events:none}[dir=rtl] .mat-form-field-label-wrapper{left:auto;right:0}.mat-form-field-label{position:absolute;left:0;font:inherit;pointer-events:none;width:100%;white-space:nowrap;text-overflow:ellipsis;overflow:hidden;transform-origin:0 0;transition:transform 400ms cubic-bezier(0.25, 0.8, 0.25, 1),color 400ms cubic-bezier(0.25, 0.8, 0.25, 1),width 400ms cubic-bezier(0.25, 0.8, 0.25, 1);display:none}[dir=rtl] .mat-form-field-label{transform-origin:100% 0;left:auto;right:0}.mat-form-field-empty.mat-form-field-label,.mat-form-field-can-float.mat-form-field-should-float .mat-form-field-label{display:block}.mat-form-field-autofill-control:-webkit-autofill+.mat-form-field-label-wrapper .mat-form-field-label{display:none}.mat-form-field-can-float .mat-form-field-autofill-control:-webkit-autofill+.mat-form-field-label-wrapper .mat-form-field-label{display:block;transition:none}.mat-input-server:focus+.mat-form-field-label-wrapper .mat-form-field-label,.mat-input-server[placeholder]:not(:placeholder-shown)+.mat-form-field-label-wrapper .mat-form-field-label{display:none}.mat-form-field-can-float .mat-input-server:focus+.mat-form-field-label-wrapper .mat-form-field-label,.mat-form-field-can-float .mat-input-server[placeholder]:not(:placeholder-shown)+.mat-form-field-label-wrapper .mat-form-field-label{display:block}.mat-form-field-label:not(.mat-form-field-empty){transition:none}.mat-form-field-underline{position:absolute;width:100%;pointer-events:none;transform:scale3d(1, 1.0001, 1)}.mat-form-field-ripple{position:absolute;left:0;width:100%;transform-origin:50%;transform:scaleX(0.5);opacity:0;transition:background-color 300ms cubic-bezier(0.55, 0, 0.55, 0.2)}.mat-form-field.mat-focused .mat-form-field-ripple,.mat-form-field.mat-form-field-invalid .mat-form-field-ripple{opacity:1;transform:scaleX(1);transition:transform 300ms cubic-bezier(0.25, 0.8, 0.25, 1),opacity 100ms cubic-bezier(0.25, 0.8, 0.25, 1),background-color 300ms cubic-bezier(0.25, 0.8, 0.25, 1)}.mat-form-field-subscript-wrapper{position:absolute;box-sizing:border-box;width:100%;overflow:hidden}.mat-form-field-subscript-wrapper .mat-icon,.mat-form-field-label-wrapper .mat-icon{width:1em;height:1em;font-size:inherit;vertical-align:baseline}.mat-form-field-hint-wrapper{display:flex}.mat-form-field-hint-spacer{flex:1 0 1em}.mat-error{display:block}.mat-form-field-control-wrapper{position:relative}.mat-form-field._mat-animation-noopable .mat-form-field-label,.mat-form-field._mat-animation-noopable .mat-form-field-ripple{transition:none}\n", ".mat-form-field-appearance-fill .mat-form-field-flex{border-radius:4px 4px 0 0;padding:.75em .75em 0 .75em}.cdk-high-contrast-active .mat-form-field-appearance-fill .mat-form-field-flex{outline:solid 1px}.mat-form-field-appearance-fill .mat-form-field-underline::before{content:\"\";display:block;position:absolute;bottom:0;height:1px;width:100%}.mat-form-field-appearance-fill .mat-form-field-ripple{bottom:0;height:2px}.cdk-high-contrast-active .mat-form-field-appearance-fill .mat-form-field-ripple{height:0;border-top:solid 2px}.mat-form-field-appearance-fill:not(.mat-form-field-disabled) .mat-form-field-flex:hover~.mat-form-field-underline .mat-form-field-ripple{opacity:1;transform:none;transition:opacity 600ms cubic-bezier(0.25, 0.8, 0.25, 1)}.mat-form-field-appearance-fill._mat-animation-noopable:not(.mat-form-field-disabled) .mat-form-field-flex:hover~.mat-form-field-underline .mat-form-field-ripple{transition:none}.mat-form-field-appearance-fill .mat-form-field-subscript-wrapper{padding:0 1em}\n", ".mat-input-element{font:inherit;background:transparent;color:currentColor;border:none;outline:none;padding:0;margin:0;width:100%;max-width:100%;vertical-align:bottom;text-align:inherit}.mat-input-element:-moz-ui-invalid{box-shadow:none}.mat-input-element::-ms-clear,.mat-input-element::-ms-reveal{display:none}.mat-input-element,.mat-input-element::-webkit-search-cancel-button,.mat-input-element::-webkit-search-decoration,.mat-input-element::-webkit-search-results-button,.mat-input-element::-webkit-search-results-decoration{-webkit-appearance:none}.mat-input-element::-webkit-contacts-auto-fill-button,.mat-input-element::-webkit-caps-lock-indicator,.mat-input-element::-webkit-credentials-auto-fill-button{visibility:hidden}.mat-input-element[type=date],.mat-input-element[type=datetime],.mat-input-element[type=datetime-local],.mat-input-element[type=month],.mat-input-element[type=week],.mat-input-element[type=time]{line-height:1}.mat-input-element[type=date]::after,.mat-input-element[type=datetime]::after,.mat-input-element[type=datetime-local]::after,.mat-input-element[type=month]::after,.mat-input-element[type=week]::after,.mat-input-element[type=time]::after{content:\" \";white-space:pre;width:1px}.mat-input-element::-webkit-inner-spin-button,.mat-input-element::-webkit-calendar-picker-indicator,.mat-input-element::-webkit-clear-button{font-size:.75em}.mat-input-element::placeholder{-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;transition:color 400ms 133.3333333333ms cubic-bezier(0.25, 0.8, 0.25, 1)}.mat-input-element::placeholder:-ms-input-placeholder{-ms-user-select:text}.mat-input-element::-moz-placeholder{-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;transition:color 400ms 133.3333333333ms cubic-bezier(0.25, 0.8, 0.25, 1)}.mat-input-element::-moz-placeholder:-ms-input-placeholder{-ms-user-select:text}.mat-input-element::-webkit-input-placeholder{-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;transition:color 400ms 133.3333333333ms cubic-bezier(0.25, 0.8, 0.25, 1)}.mat-input-element::-webkit-input-placeholder:-ms-input-placeholder{-ms-user-select:text}.mat-input-element:-ms-input-placeholder{-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;transition:color 400ms 133.3333333333ms cubic-bezier(0.25, 0.8, 0.25, 1)}.mat-input-element:-ms-input-placeholder:-ms-input-placeholder{-ms-user-select:text}.mat-form-field-hide-placeholder .mat-input-element::placeholder{color:transparent !important;-webkit-text-fill-color:transparent;transition:none}.mat-form-field-hide-placeholder .mat-input-element::-moz-placeholder{color:transparent !important;-webkit-text-fill-color:transparent;transition:none}.mat-form-field-hide-placeholder .mat-input-element::-webkit-input-placeholder{color:transparent !important;-webkit-text-fill-color:transparent;transition:none}.mat-form-field-hide-placeholder .mat-input-element:-ms-input-placeholder{color:transparent !important;-webkit-text-fill-color:transparent;transition:none}textarea.mat-input-element{resize:vertical;overflow:auto}textarea.mat-input-element.cdk-textarea-autosize{resize:none}textarea.mat-input-element{padding:2px 0;margin:-2px 0}select.mat-input-element{-moz-appearance:none;-webkit-appearance:none;position:relative;background-color:transparent;display:inline-flex;box-sizing:border-box;padding-top:1em;top:-1em;margin-bottom:-1em}select.mat-input-element::-ms-expand{display:none}select.mat-input-element::-moz-focus-inner{border:0}select.mat-input-element:not(:disabled){cursor:pointer}select.mat-input-element::-ms-value{color:inherit;background:none}.mat-focused .cdk-high-contrast-active select.mat-input-element::-ms-value{color:inherit}.mat-form-field-type-mat-native-select .mat-form-field-infix::after{content:\"\";width:0;height:0;border-left:5px solid transparent;border-right:5px solid transparent;border-top:5px solid;position:absolute;top:50%;right:0;margin-top:-2.5px;pointer-events:none}[dir=rtl] .mat-form-field-type-mat-native-select .mat-form-field-infix::after{right:auto;left:0}.mat-form-field-type-mat-native-select .mat-input-element{padding-right:15px}[dir=rtl] .mat-form-field-type-mat-native-select .mat-input-element{padding-right:0;padding-left:15px}.mat-form-field-type-mat-native-select .mat-form-field-label-wrapper{max-width:calc(100% - 10px)}.mat-form-field-type-mat-native-select.mat-form-field-appearance-outline .mat-form-field-infix::after{margin-top:-5px}.mat-form-field-type-mat-native-select.mat-form-field-appearance-fill .mat-form-field-infix::after{margin-top:-10px}\n", ".mat-form-field-appearance-legacy .mat-form-field-label{transform:perspective(100px);-ms-transform:none}.mat-form-field-appearance-legacy .mat-form-field-prefix .mat-icon,.mat-form-field-appearance-legacy .mat-form-field-suffix .mat-icon{width:1em}.mat-form-field-appearance-legacy .mat-form-field-prefix .mat-icon-button,.mat-form-field-appearance-legacy .mat-form-field-suffix .mat-icon-button{font:inherit;vertical-align:baseline}.mat-form-field-appearance-legacy .mat-form-field-prefix .mat-icon-button .mat-icon,.mat-form-field-appearance-legacy .mat-form-field-suffix .mat-icon-button .mat-icon{font-size:inherit}.mat-form-field-appearance-legacy .mat-form-field-underline{height:1px}.cdk-high-contrast-active .mat-form-field-appearance-legacy .mat-form-field-underline{height:0;border-top:solid 1px}.mat-form-field-appearance-legacy .mat-form-field-ripple{top:0;height:2px;overflow:hidden}.cdk-high-contrast-active .mat-form-field-appearance-legacy .mat-form-field-ripple{height:0;border-top:solid 2px}.mat-form-field-appearance-legacy.mat-form-field-disabled .mat-form-field-underline{background-position:0;background-color:transparent}.cdk-high-contrast-active .mat-form-field-appearance-legacy.mat-form-field-disabled .mat-form-field-underline{border-top-style:dotted;border-top-width:2px}.mat-form-field-appearance-legacy.mat-form-field-invalid:not(.mat-focused) .mat-form-field-ripple{height:1px}\n", ".mat-form-field-appearance-outline .mat-form-field-wrapper{margin:.25em 0}.mat-form-field-appearance-outline .mat-form-field-flex{padding:0 .75em 0 .75em;margin-top:-0.25em;position:relative}.mat-form-field-appearance-outline .mat-form-field-prefix,.mat-form-field-appearance-outline .mat-form-field-suffix{top:.25em}.mat-form-field-appearance-outline .mat-form-field-outline{display:flex;position:absolute;top:.25em;left:0;right:0;bottom:0;pointer-events:none}.mat-form-field-appearance-outline .mat-form-field-outline-start,.mat-form-field-appearance-outline .mat-form-field-outline-end{border:1px solid currentColor;min-width:5px}.mat-form-field-appearance-outline .mat-form-field-outline-start{border-radius:5px 0 0 5px;border-right-style:none}[dir=rtl] .mat-form-field-appearance-outline .mat-form-field-outline-start{border-right-style:solid;border-left-style:none;border-radius:0 5px 5px 0}.mat-form-field-appearance-outline .mat-form-field-outline-end{border-radius:0 5px 5px 0;border-left-style:none;flex-grow:1}[dir=rtl] .mat-form-field-appearance-outline .mat-form-field-outline-end{border-left-style:solid;border-right-style:none;border-radius:5px 0 0 5px}.mat-form-field-appearance-outline .mat-form-field-outline-gap{border-radius:.000001px;border:1px solid currentColor;border-left-style:none;border-right-style:none}.mat-form-field-appearance-outline.mat-form-field-can-float.mat-form-field-should-float .mat-form-field-outline-gap{border-top-color:transparent}.mat-form-field-appearance-outline .mat-form-field-outline-thick{opacity:0}.mat-form-field-appearance-outline .mat-form-field-outline-thick .mat-form-field-outline-start,.mat-form-field-appearance-outline .mat-form-field-outline-thick .mat-form-field-outline-end,.mat-form-field-appearance-outline .mat-form-field-outline-thick .mat-form-field-outline-gap{border-width:2px}.mat-form-field-appearance-outline.mat-focused .mat-form-field-outline,.mat-form-field-appearance-outline.mat-form-field-invalid .mat-form-field-outline{opacity:0;transition:opacity 100ms cubic-bezier(0.25, 0.8, 0.25, 1)}.mat-form-field-appearance-outline.mat-focused .mat-form-field-outline-thick,.mat-form-field-appearance-outline.mat-form-field-invalid .mat-form-field-outline-thick{opacity:1}.mat-form-field-appearance-outline:not(.mat-form-field-disabled) .mat-form-field-flex:hover .mat-form-field-outline{opacity:0;transition:opacity 600ms cubic-bezier(0.25, 0.8, 0.25, 1)}.mat-form-field-appearance-outline:not(.mat-form-field-disabled) .mat-form-field-flex:hover .mat-form-field-outline-thick{opacity:1}.mat-form-field-appearance-outline .mat-form-field-subscript-wrapper{padding:0 1em}.mat-form-field-appearance-outline._mat-animation-noopable:not(.mat-form-field-disabled) .mat-form-field-flex:hover~.mat-form-field-outline,.mat-form-field-appearance-outline._mat-animation-noopable .mat-form-field-outline,.mat-form-field-appearance-outline._mat-animation-noopable .mat-form-field-outline-start,.mat-form-field-appearance-outline._mat-animation-noopable .mat-form-field-outline-end,.mat-form-field-appearance-outline._mat-animation-noopable .mat-form-field-outline-gap{transition:none}\n", ".mat-form-field-appearance-standard .mat-form-field-flex{padding-top:.75em}.mat-form-field-appearance-standard .mat-form-field-underline{height:1px}.cdk-high-contrast-active .mat-form-field-appearance-standard .mat-form-field-underline{height:0;border-top:solid 1px}.mat-form-field-appearance-standard .mat-form-field-ripple{bottom:0;height:2px}.cdk-high-contrast-active .mat-form-field-appearance-standard .mat-form-field-ripple{height:0;border-top:2px}.mat-form-field-appearance-standard.mat-form-field-disabled .mat-form-field-underline{background-position:0;background-color:transparent}.cdk-high-contrast-active .mat-form-field-appearance-standard.mat-form-field-disabled .mat-form-field-underline{border-top-style:dotted;border-top-width:2px}.mat-form-field-appearance-standard:not(.mat-form-field-disabled) .mat-form-field-flex:hover~.mat-form-field-underline .mat-form-field-ripple{opacity:1;transform:none;transition:opacity 600ms cubic-bezier(0.25, 0.8, 0.25, 1)}.mat-form-field-appearance-standard._mat-animation-noopable:not(.mat-form-field-disabled) .mat-form-field-flex:hover~.mat-form-field-underline .mat-form-field-ripple{transition:none}\n"], encapsulation: 2, data: { animation: [matFormFieldAnimations.transitionMessages] }, changeDetection: 0 });
    /** @nocollapse */
    MatFormField.ctorParameters = () => [
        { type: ElementRef },
        { type: ChangeDetectorRef },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [MAT_LABEL_GLOBAL_OPTIONS,] }] },
        { type: Directionality, decorators: [{ type: Optional }] },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [MAT_FORM_FIELD_DEFAULT_OPTIONS,] }] },
        { type: Platform },
        { type: NgZone },
        { type: String, decorators: [{ type: Optional }, { type: Inject, args: [ANIMATION_MODULE_TYPE,] }] }
    ];
    MatFormField.propDecorators = {
        appearance: [{ type: Input }],
        hideRequiredMarker: [{ type: Input }],
        hintLabel: [{ type: Input }],
        floatLabel: [{ type: Input }],
        underlineRef: [{ type: ViewChild, args: ['underline',] }],
        _connectionContainerRef: [{ type: ViewChild, args: ['connectionContainer', { static: true },] }],
        _inputContainerRef: [{ type: ViewChild, args: ['inputContainer',] }],
        _label: [{ type: ViewChild, args: ['label',] }],
        _controlNonStatic: [{ type: ContentChild, args: [MatFormFieldControl,] }],
        _controlStatic: [{ type: ContentChild, args: [MatFormFieldControl, { static: true },] }],
        _labelChildNonStatic: [{ type: ContentChild, args: [MatLabel,] }],
        _labelChildStatic: [{ type: ContentChild, args: [MatLabel, { static: true },] }],
        _placeholderChild: [{ type: ContentChild, args: [MatPlaceholder,] }],
        _errorChildren: [{ type: ContentChildren, args: [MatError, { descendants: true },] }],
        _hintChildren: [{ type: ContentChildren, args: [MatHint, { descendants: true },] }],
        _prefixChildren: [{ type: ContentChildren, args: [MatPrefix, { descendants: true },] }],
        _suffixChildren: [{ type: ContentChildren, args: [MatSuffix, { descendants: true },] }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/form-field/form-field-module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class MatFormFieldModule {
    }
    MatFormFieldModule.ɵmod = ɵɵdefineNgModule({ type: MatFormFieldModule });
    MatFormFieldModule.ɵinj = ɵɵdefineInjector({ factory: function MatFormFieldModule_Factory(t) { return new (t || MatFormFieldModule)(); }, imports: [[
                CommonModule,
                MatCommonModule,
                ObserversModule,
            ],
            MatCommonModule] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(MatFormFieldModule, { declarations: function () { return [MatError,
            MatFormField,
            MatHint,
            MatLabel,
            MatPlaceholder,
            MatPrefix,
            MatSuffix]; }, imports: function () { return [CommonModule,
            MatCommonModule,
            ObserversModule]; }, exports: function () { return [MatCommonModule,
            MatError,
            MatFormField,
            MatHint,
            MatLabel,
            MatPlaceholder,
            MatPrefix,
            MatSuffix]; } }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/overlay/scroll/block-scroll-strategy.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Strategy that will prevent the user from scrolling while the overlay is visible.
     */
    class BlockScrollStrategy {
        /**
         * @param {?} _viewportRuler
         * @param {?} document
         */
        constructor(_viewportRuler, document) {
            this._viewportRuler = _viewportRuler;
            this._previousHTMLStyles = { top: '', left: '' };
            this._isEnabled = false;
            this._document = document;
        }
        /**
         * Attaches this scroll strategy to an overlay.
         * @return {?}
         */
        attach() { }
        /**
         * Blocks page-level scroll while the attached overlay is open.
         * @return {?}
         */
        enable() {
            if (this._canBeEnabled()) {
                /** @type {?} */
                const root = (/** @type {?} */ (this._document.documentElement));
                this._previousScrollPosition = this._viewportRuler.getViewportScrollPosition();
                // Cache the previous inline styles in case the user had set them.
                this._previousHTMLStyles.left = root.style.left || '';
                this._previousHTMLStyles.top = root.style.top || '';
                // Note: we're using the `html` node, instead of the `body`, because the `body` may
                // have the user agent margin, whereas the `html` is guaranteed not to have one.
                root.style.left = coerceCssPixelValue(-this._previousScrollPosition.left);
                root.style.top = coerceCssPixelValue(-this._previousScrollPosition.top);
                root.classList.add('cdk-global-scrollblock');
                this._isEnabled = true;
            }
        }
        /**
         * Unblocks page-level scroll while the attached overlay is open.
         * @return {?}
         */
        disable() {
            if (this._isEnabled) {
                /** @type {?} */
                const html = (/** @type {?} */ (this._document.documentElement));
                /** @type {?} */
                const body = (/** @type {?} */ (this._document.body));
                /** @type {?} */
                const htmlStyle = (/** @type {?} */ (html.style));
                /** @type {?} */
                const bodyStyle = (/** @type {?} */ (body.style));
                /** @type {?} */
                const previousHtmlScrollBehavior = htmlStyle.scrollBehavior || '';
                /** @type {?} */
                const previousBodyScrollBehavior = bodyStyle.scrollBehavior || '';
                this._isEnabled = false;
                htmlStyle.left = this._previousHTMLStyles.left;
                htmlStyle.top = this._previousHTMLStyles.top;
                html.classList.remove('cdk-global-scrollblock');
                // Disable user-defined smooth scrolling temporarily while we restore the scroll position.
                // See https://developer.mozilla.org/en-US/docs/Web/CSS/scroll-behavior
                htmlStyle.scrollBehavior = bodyStyle.scrollBehavior = 'auto';
                window.scroll(this._previousScrollPosition.left, this._previousScrollPosition.top);
                htmlStyle.scrollBehavior = previousHtmlScrollBehavior;
                bodyStyle.scrollBehavior = previousBodyScrollBehavior;
            }
        }
        /**
         * @private
         * @return {?}
         */
        _canBeEnabled() {
            // Since the scroll strategies can't be singletons, we have to use a global CSS class
            // (`cdk-global-scrollblock`) to make sure that we don't try to disable global
            // scrolling multiple times.
            /** @type {?} */
            const html = (/** @type {?} */ (this._document.documentElement));
            if (html.classList.contains('cdk-global-scrollblock') || this._isEnabled) {
                return false;
            }
            /** @type {?} */
            const body = this._document.body;
            /** @type {?} */
            const viewport = this._viewportRuler.getViewportSize();
            return body.scrollHeight > viewport.height || body.scrollWidth > viewport.width;
        }
    }
    /**
     * Returns an error to be thrown when attempting to attach an already-attached scroll strategy.
     * @return {?}
     */
    function getMatScrollStrategyAlreadyAttachedError() {
        return Error(`Scroll strategy has already been attached.`);
    }
    /**
     * Strategy that will close the overlay as soon as the user starts scrolling.
     */
    class CloseScrollStrategy {
        /**
         * @param {?} _scrollDispatcher
         * @param {?} _ngZone
         * @param {?} _viewportRuler
         * @param {?=} _config
         */
        constructor(_scrollDispatcher, _ngZone, _viewportRuler, _config) {
            this._scrollDispatcher = _scrollDispatcher;
            this._ngZone = _ngZone;
            this._viewportRuler = _viewportRuler;
            this._config = _config;
            this._scrollSubscription = null;
            /**
             * Detaches the overlay ref and disables the scroll strategy.
             */
            this._detach = (/**
             * @return {?}
             */
            () => {
                this.disable();
                if (this._overlayRef.hasAttached()) {
                    this._ngZone.run((/**
                     * @return {?}
                     */
                    () => this._overlayRef.detach()));
                }
            });
        }
        /**
         * Attaches this scroll strategy to an overlay.
         * @param {?} overlayRef
         * @return {?}
         */
        attach(overlayRef) {
            if (this._overlayRef) {
                throw getMatScrollStrategyAlreadyAttachedError();
            }
            this._overlayRef = overlayRef;
        }
        /**
         * Enables the closing of the attached overlay on scroll.
         * @return {?}
         */
        enable() {
            if (this._scrollSubscription) {
                return;
            }
            /** @type {?} */
            const stream = this._scrollDispatcher.scrolled(0);
            if (this._config && this._config.threshold && this._config.threshold > 1) {
                this._initialScrollPosition = this._viewportRuler.getViewportScrollPosition().top;
                this._scrollSubscription = stream.subscribe((/**
                 * @return {?}
                 */
                () => {
                    /** @type {?} */
                    const scrollPosition = this._viewportRuler.getViewportScrollPosition().top;
                    if (Math.abs(scrollPosition - this._initialScrollPosition) > (/** @type {?} */ ((/** @type {?} */ (this._config)).threshold))) {
                        this._detach();
                    }
                    else {
                        this._overlayRef.updatePosition();
                    }
                }));
            }
            else {
                this._scrollSubscription = stream.subscribe(this._detach);
            }
        }
        /**
         * Disables the closing the attached overlay on scroll.
         * @return {?}
         */
        disable() {
            if (this._scrollSubscription) {
                this._scrollSubscription.unsubscribe();
                this._scrollSubscription = null;
            }
        }
        /**
         * @return {?}
         */
        detach() {
            this.disable();
            this._overlayRef = (/** @type {?} */ (null));
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/overlay/scroll/noop-scroll-strategy.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google LLC All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * Scroll strategy that doesn't do anything.
     */
    class NoopScrollStrategy {
        /**
         * Does nothing, as this scroll strategy is a no-op.
         * @return {?}
         */
        enable() { }
        /**
         * Does nothing, as this scroll strategy is a no-op.
         * @return {?}
         */
        disable() { }
        /**
         * Does nothing, as this scroll strategy is a no-op.
         * @return {?}
         */
        attach() { }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/overlay/position/scroll-clip.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google LLC All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    // TODO(jelbourn): move this to live with the rest of the scrolling code
    // TODO(jelbourn): someday replace this with IntersectionObservers
    /**
     * Gets whether an element is scrolled outside of view by any of its parent scrolling containers.
     * \@docs-private
     * @param {?} element Dimensions of the element (from getBoundingClientRect)
     * @param {?} scrollContainers Dimensions of element's scrolling containers (from getBoundingClientRect)
     * @return {?} Whether the element is scrolled out of view
     */
    function isElementScrolledOutsideView(element, scrollContainers) {
        return scrollContainers.some((/**
         * @param {?} containerBounds
         * @return {?}
         */
        containerBounds => {
            /** @type {?} */
            const outsideAbove = element.bottom < containerBounds.top;
            /** @type {?} */
            const outsideBelow = element.top > containerBounds.bottom;
            /** @type {?} */
            const outsideLeft = element.right < containerBounds.left;
            /** @type {?} */
            const outsideRight = element.left > containerBounds.right;
            return outsideAbove || outsideBelow || outsideLeft || outsideRight;
        }));
    }
    /**
     * Gets whether an element is clipped by any of its scrolling containers.
     * \@docs-private
     * @param {?} element Dimensions of the element (from getBoundingClientRect)
     * @param {?} scrollContainers Dimensions of element's scrolling containers (from getBoundingClientRect)
     * @return {?} Whether the element is clipped
     */
    function isElementClippedByScrolling(element, scrollContainers) {
        return scrollContainers.some((/**
         * @param {?} scrollContainerRect
         * @return {?}
         */
        scrollContainerRect => {
            /** @type {?} */
            const clippedAbove = element.top < scrollContainerRect.top;
            /** @type {?} */
            const clippedBelow = element.bottom > scrollContainerRect.bottom;
            /** @type {?} */
            const clippedLeft = element.left < scrollContainerRect.left;
            /** @type {?} */
            const clippedRight = element.right > scrollContainerRect.right;
            return clippedAbove || clippedBelow || clippedLeft || clippedRight;
        }));
    }
    /**
     * Strategy that will update the element position as the user is scrolling.
     */
    class RepositionScrollStrategy {
        /**
         * @param {?} _scrollDispatcher
         * @param {?} _viewportRuler
         * @param {?} _ngZone
         * @param {?=} _config
         */
        constructor(_scrollDispatcher, _viewportRuler, _ngZone, _config) {
            this._scrollDispatcher = _scrollDispatcher;
            this._viewportRuler = _viewportRuler;
            this._ngZone = _ngZone;
            this._config = _config;
            this._scrollSubscription = null;
        }
        /**
         * Attaches this scroll strategy to an overlay.
         * @param {?} overlayRef
         * @return {?}
         */
        attach(overlayRef) {
            if (this._overlayRef) {
                throw getMatScrollStrategyAlreadyAttachedError();
            }
            this._overlayRef = overlayRef;
        }
        /**
         * Enables repositioning of the attached overlay on scroll.
         * @return {?}
         */
        enable() {
            if (!this._scrollSubscription) {
                /** @type {?} */
                const throttle = this._config ? this._config.scrollThrottle : 0;
                this._scrollSubscription = this._scrollDispatcher.scrolled(throttle).subscribe((/**
                 * @return {?}
                 */
                () => {
                    this._overlayRef.updatePosition();
                    // TODO(crisbeto): make `close` on by default once all components can handle it.
                    if (this._config && this._config.autoClose) {
                        /** @type {?} */
                        const overlayRect = this._overlayRef.overlayElement.getBoundingClientRect();
                        const { width, height } = this._viewportRuler.getViewportSize();
                        // TODO(crisbeto): include all ancestor scroll containers here once
                        // we have a way of exposing the trigger element to the scroll strategy.
                        /** @type {?} */
                        const parentRects = [{ width, height, bottom: height, right: width, top: 0, left: 0 }];
                        if (isElementScrolledOutsideView(overlayRect, parentRects)) {
                            this.disable();
                            this._ngZone.run((/**
                             * @return {?}
                             */
                            () => this._overlayRef.detach()));
                        }
                    }
                }));
            }
        }
        /**
         * Disables repositioning of the attached overlay on scroll.
         * @return {?}
         */
        disable() {
            if (this._scrollSubscription) {
                this._scrollSubscription.unsubscribe();
                this._scrollSubscription = null;
            }
        }
        /**
         * @return {?}
         */
        detach() {
            this.disable();
            this._overlayRef = (/** @type {?} */ (null));
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/overlay/scroll/scroll-strategy-options.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Options for how an overlay will handle scrolling.
     *
     * Users can provide a custom value for `ScrollStrategyOptions` to replace the default
     * behaviors. This class primarily acts as a factory for ScrollStrategy instances.
     */
    class ScrollStrategyOptions {
        /**
         * @param {?} _scrollDispatcher
         * @param {?} _viewportRuler
         * @param {?} _ngZone
         * @param {?} document
         */
        constructor(_scrollDispatcher, _viewportRuler, _ngZone, document) {
            this._scrollDispatcher = _scrollDispatcher;
            this._viewportRuler = _viewportRuler;
            this._ngZone = _ngZone;
            /**
             * Do nothing on scroll.
             */
            this.noop = (/**
             * @return {?}
             */
            () => new NoopScrollStrategy());
            /**
             * Close the overlay as soon as the user scrolls.
             * @param config Configuration to be used inside the scroll strategy.
             */
            this.close = (/**
             * @param {?=} config
             * @return {?}
             */
            (config) => new CloseScrollStrategy(this._scrollDispatcher, this._ngZone, this._viewportRuler, config));
            /**
             * Block scrolling.
             */
            this.block = (/**
             * @return {?}
             */
            () => new BlockScrollStrategy(this._viewportRuler, this._document));
            /**
             * Update the overlay's position on scroll.
             * @param config Configuration to be used inside the scroll strategy.
             * Allows debouncing the reposition calls.
             */
            this.reposition = (/**
             * @param {?=} config
             * @return {?}
             */
            (config) => new RepositionScrollStrategy(this._scrollDispatcher, this._viewportRuler, this._ngZone, config));
            this._document = document;
        }
    }
    ScrollStrategyOptions.ɵfac = function ScrollStrategyOptions_Factory(t) { return new (t || ScrollStrategyOptions)(ɵɵinject(ScrollDispatcher), ɵɵinject(ViewportRuler), ɵɵinject(NgZone), ɵɵinject(DOCUMENT$1)); };
    /** @nocollapse */
    ScrollStrategyOptions.ctorParameters = () => [
        { type: ScrollDispatcher },
        { type: ViewportRuler },
        { type: NgZone },
        { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT$1,] }] }
    ];
    /** @nocollapse */ ScrollStrategyOptions.ɵprov = ɵɵdefineInjectable({ factory: function ScrollStrategyOptions_Factory() { return new ScrollStrategyOptions(ɵɵinject(ScrollDispatcher), ɵɵinject(ViewportRuler), ɵɵinject(NgZone), ɵɵinject(DOCUMENT$1)); }, token: ScrollStrategyOptions, providedIn: "root" });

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/overlay/scroll/index.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/overlay/overlay-config.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Initial configuration used when creating an overlay.
     */
    class OverlayConfig {
        /**
         * @param {?=} config
         */
        constructor(config) {
            /**
             * Strategy to be used when handling scroll events while the overlay is open.
             */
            this.scrollStrategy = new NoopScrollStrategy();
            /**
             * Custom class to add to the overlay pane.
             */
            this.panelClass = '';
            /**
             * Whether the overlay has a backdrop.
             */
            this.hasBackdrop = false;
            /**
             * Custom class to add to the backdrop
             */
            this.backdropClass = 'cdk-overlay-dark-backdrop';
            /**
             * Whether the overlay should be disposed of when the user goes backwards/forwards in history.
             * Note that this usually doesn't include clicking on links (unless the user is using
             * the `HashLocationStrategy`).
             */
            this.disposeOnNavigation = false;
            if (config) {
                // Use `Iterable` instead of `Array` because TypeScript, as of 3.6.3,
                // loses the array generic type in the `for of`. But we *also* have to use `Array` because
                // typescript won't iterate over an `Iterable` unless you compile with `--downlevelIteration`
                /** @type {?} */
                const configKeys = (/** @type {?} */ (Object.keys(config)));
                for (const key of configKeys) {
                    if (config[key] !== undefined) {
                        // TypeScript, as of version 3.5, sees the left-hand-side of this expression
                        // as "I don't know *which* key this is, so the only valid value is the intersection
                        // of all the posible values." In this case, that happens to be `undefined`. TypeScript
                        // is not smart enough to see that the right-hand-side is actually an access of the same
                        // exact type with the same exact key, meaning that the value type must be identical.
                        // So we use `any` to work around this.
                        this[key] = (/** @type {?} */ (config[key]));
                    }
                }
            }
        }
    }
    /**
     * The points of the origin element and the overlay element to connect.
     */
    class ConnectionPositionPair {
        /**
         * @param {?} origin
         * @param {?} overlay
         * @param {?=} offsetX
         * @param {?=} offsetY
         * @param {?=} panelClass
         */
        constructor(origin, overlay, offsetX, offsetY, panelClass) {
            this.offsetX = offsetX;
            this.offsetY = offsetY;
            this.panelClass = panelClass;
            this.originX = origin.originX;
            this.originY = origin.originY;
            this.overlayX = overlay.overlayX;
            this.overlayY = overlay.overlayY;
        }
    }
    /**
     * Set of properties regarding the position of the origin and overlay relative to the viewport
     * with respect to the containing Scrollable elements.
     *
     * The overlay and origin are clipped if any part of their bounding client rectangle exceeds the
     * bounds of any one of the strategy's Scrollable's bounding client rectangle.
     *
     * The overlay and origin are outside view if there is no overlap between their bounding client
     * rectangle and any one of the strategy's Scrollable's bounding client rectangle.
     *
     *       -----------                    -----------
     *       | outside |                    | clipped |
     *       |  view   |              --------------------------
     *       |         |              |     |         |        |
     *       ----------               |     -----------        |
     *  --------------------------    |                        |
     *  |                        |    |      Scrollable        |
     *  |                        |    |                        |
     *  |                        |     --------------------------
     *  |      Scrollable        |
     *  |                        |
     *  --------------------------
     *
     * \@docs-private
     */
    class ScrollingVisibility {
    }
    /**
     * The change event emitted by the strategy when a fallback position is used.
     */
    class ConnectedOverlayPositionChange {
        /**
         * @param {?} connectionPair
         * @param {?} scrollableViewProperties
         */
        constructor(connectionPair, scrollableViewProperties) {
            this.connectionPair = connectionPair;
            this.scrollableViewProperties = scrollableViewProperties;
        }
    }
    /** @nocollapse */
    ConnectedOverlayPositionChange.ctorParameters = () => [
        { type: ConnectionPositionPair },
        { type: ScrollingVisibility, decorators: [{ type: Optional }] }
    ];
    /**
     * Validates whether a vertical position property matches the expected values.
     * \@docs-private
     * @param {?} property Name of the property being validated.
     * @param {?} value Value of the property being validated.
     * @return {?}
     */
    function validateVerticalPosition(property, value) {
        if (value !== 'top' && value !== 'bottom' && value !== 'center') {
            throw Error(`ConnectedPosition: Invalid ${property} "${value}". ` +
                `Expected "top", "bottom" or "center".`);
        }
    }
    /**
     * Validates whether a horizontal position property matches the expected values.
     * \@docs-private
     * @param {?} property Name of the property being validated.
     * @param {?} value Value of the property being validated.
     * @return {?}
     */
    function validateHorizontalPosition(property, value) {
        if (value !== 'start' && value !== 'end' && value !== 'center') {
            throw Error(`ConnectedPosition: Invalid ${property} "${value}". ` +
                `Expected "start", "end" or "center".`);
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/overlay/keyboard/overlay-keyboard-dispatcher.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Service for dispatching keyboard events that land on the body to appropriate overlay ref,
     * if any. It maintains a list of attached overlays to determine best suited overlay based
     * on event target and order of overlay opens.
     */
    class OverlayKeyboardDispatcher {
        /**
         * @param {?} document
         */
        constructor(document) {
            /**
             * Currently attached overlays in the order they were attached.
             */
            this._attachedOverlays = [];
            /**
             * Keyboard event listener that will be attached to the body.
             */
            this._keydownListener = (/**
             * @param {?} event
             * @return {?}
             */
            (event) => {
                /** @type {?} */
                const overlays = this._attachedOverlays;
                for (let i = overlays.length - 1; i > -1; i--) {
                    // Dispatch the keydown event to the top overlay which has subscribers to its keydown events.
                    // We want to target the most recent overlay, rather than trying to match where the event came
                    // from, because some components might open an overlay, but keep focus on a trigger element
                    // (e.g. for select and autocomplete). We skip overlays without keydown event subscriptions,
                    // because we don't want overlays that don't handle keyboard events to block the ones below
                    // them that do.
                    if (overlays[i]._keydownEvents.observers.length > 0) {
                        overlays[i]._keydownEvents.next(event);
                        break;
                    }
                }
            });
            this._document = document;
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this._detach();
        }
        /**
         * Add a new overlay to the list of attached overlay refs.
         * @param {?} overlayRef
         * @return {?}
         */
        add(overlayRef) {
            // Ensure that we don't get the same overlay multiple times.
            this.remove(overlayRef);
            // Lazily start dispatcher once first overlay is added
            if (!this._isAttached) {
                this._document.body.addEventListener('keydown', this._keydownListener);
                this._isAttached = true;
            }
            this._attachedOverlays.push(overlayRef);
        }
        /**
         * Remove an overlay from the list of attached overlay refs.
         * @param {?} overlayRef
         * @return {?}
         */
        remove(overlayRef) {
            /** @type {?} */
            const index = this._attachedOverlays.indexOf(overlayRef);
            if (index > -1) {
                this._attachedOverlays.splice(index, 1);
            }
            // Remove the global listener once there are no more overlays.
            if (this._attachedOverlays.length === 0) {
                this._detach();
            }
        }
        /**
         * Detaches the global keyboard event listener.
         * @private
         * @return {?}
         */
        _detach() {
            if (this._isAttached) {
                this._document.body.removeEventListener('keydown', this._keydownListener);
                this._isAttached = false;
            }
        }
    }
    OverlayKeyboardDispatcher.ɵfac = function OverlayKeyboardDispatcher_Factory(t) { return new (t || OverlayKeyboardDispatcher)(ɵɵinject(DOCUMENT$1)); };
    /** @nocollapse */
    OverlayKeyboardDispatcher.ctorParameters = () => [
        { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT$1,] }] }
    ];
    /** @nocollapse */ OverlayKeyboardDispatcher.ɵprov = ɵɵdefineInjectable({ factory: function OverlayKeyboardDispatcher_Factory() { return new OverlayKeyboardDispatcher(ɵɵinject(DOCUMENT$1)); }, token: OverlayKeyboardDispatcher, providedIn: "root" });
    /**
     * \@docs-private \@deprecated \@breaking-change 8.0.0
     * @param {?} dispatcher
     * @param {?} _document
     * @return {?}
     */
    function OVERLAY_KEYBOARD_DISPATCHER_PROVIDER_FACTORY(dispatcher, _document) {
        return dispatcher || new OverlayKeyboardDispatcher(_document);
    }
    /**
     * \@docs-private \@deprecated \@breaking-change 8.0.0
     * @type {?}
     */
    const OVERLAY_KEYBOARD_DISPATCHER_PROVIDER = {
        // If there is already an OverlayKeyboardDispatcher available, use that.
        // Otherwise, provide a new one.
        provide: OverlayKeyboardDispatcher,
        deps: [
            [new Optional(), new SkipSelf(), OverlayKeyboardDispatcher],
            (/** @type {?} */ (
            // Coerce to `InjectionToken` so that the `deps` match the "shape"
            // of the type expected by Angular
            DOCUMENT$1))
        ],
        useFactory: OVERLAY_KEYBOARD_DISPATCHER_PROVIDER_FACTORY
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/overlay/overlay-container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Whether we're in a testing environment.
     * TODO(crisbeto): remove this once we have an overlay testing module.
     * @type {?}
     */
    const isTestEnvironment = typeof window !== 'undefined' && !!window &&
        !!(((/** @type {?} */ (window))).__karma__ || ((/** @type {?} */ (window))).jasmine);
    /**
     * Container inside which all overlays will render.
     */
    class OverlayContainer {
        /**
         * @param {?} document
         * @param {?=} _platform
         */
        constructor(document, _platform) {
            this._platform = _platform;
            this._document = document;
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            /** @type {?} */
            const container = this._containerElement;
            if (container && container.parentNode) {
                container.parentNode.removeChild(container);
            }
        }
        /**
         * This method returns the overlay container element. It will lazily
         * create the element the first time  it is called to facilitate using
         * the container in non-browser environments.
         * @return {?} the container element
         */
        getContainerElement() {
            if (!this._containerElement) {
                this._createContainer();
            }
            return this._containerElement;
        }
        /**
         * Create the overlay container element, which is simply a div
         * with the 'cdk-overlay-container' class on the document body.
         * @protected
         * @return {?}
         */
        _createContainer() {
            // @breaking-change 10.0.0 Remove null check for `_platform`.
            /** @type {?} */
            const isBrowser = this._platform ? this._platform.isBrowser : typeof window !== 'undefined';
            /** @type {?} */
            const containerClass = 'cdk-overlay-container';
            if (isBrowser || isTestEnvironment) {
                /** @type {?} */
                const oppositePlatformContainers = this._document.querySelectorAll(`.${containerClass}[platform="server"], ` +
                    `.${containerClass}[platform="test"]`);
                // Remove any old containers from the opposite platform.
                // This can happen when transitioning from the server to the client.
                for (let i = 0; i < oppositePlatformContainers.length; i++) {
                    (/** @type {?} */ (oppositePlatformContainers[i].parentNode)).removeChild(oppositePlatformContainers[i]);
                }
            }
            /** @type {?} */
            const container = this._document.createElement('div');
            container.classList.add(containerClass);
            // A long time ago we kept adding new overlay containers whenever a new app was instantiated,
            // but at some point we added logic which clears the duplicate ones in order to avoid leaks.
            // The new logic was a little too aggressive since it was breaking some legitimate use cases.
            // To mitigate the problem we made it so that only containers from a different platform are
            // cleared, but the side-effect was that people started depending on the overly-aggressive
            // logic to clean up their tests for them. Until we can introduce an overlay-specific testing
            // module which does the cleanup, we try to detect that we're in a test environment and we
            // always clear the container. See #17006.
            // TODO(crisbeto): remove the test environment check once we have an overlay testing module.
            if (isTestEnvironment) {
                container.setAttribute('platform', 'test');
            }
            else if (!isBrowser) {
                container.setAttribute('platform', 'server');
            }
            this._document.body.appendChild(container);
            this._containerElement = container;
        }
    }
    OverlayContainer.ɵfac = function OverlayContainer_Factory(t) { return new (t || OverlayContainer)(ɵɵinject(DOCUMENT$1), ɵɵinject(Platform)); };
    /** @nocollapse */
    OverlayContainer.ctorParameters = () => [
        { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT$1,] }] },
        { type: Platform }
    ];
    /** @nocollapse */ OverlayContainer.ɵprov = ɵɵdefineInjectable({ factory: function OverlayContainer_Factory() { return new OverlayContainer(ɵɵinject(DOCUMENT$1), ɵɵinject(Platform)); }, token: OverlayContainer, providedIn: "root" });
    /**
     * \@docs-private \@deprecated \@breaking-change 8.0.0
     * @param {?} parentContainer
     * @param {?} _document
     * @return {?}
     */
    function OVERLAY_CONTAINER_PROVIDER_FACTORY(parentContainer, _document) {
        return parentContainer || new OverlayContainer(_document);
    }
    /**
     * \@docs-private \@deprecated \@breaking-change 8.0.0
     * @type {?}
     */
    const OVERLAY_CONTAINER_PROVIDER = {
        // If there is already an OverlayContainer available, use that. Otherwise, provide a new one.
        provide: OverlayContainer,
        deps: [
            [new Optional(), new SkipSelf(), OverlayContainer],
            (/** @type {?} */ (DOCUMENT$1))
        ],
        useFactory: OVERLAY_CONTAINER_PROVIDER_FACTORY
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/overlay/overlay-ref.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Reference to an overlay that has been created with the Overlay service.
     * Used to manipulate or dispose of said overlay.
     */
    class OverlayRef {
        /**
         * @param {?} _portalOutlet
         * @param {?} _host
         * @param {?} _pane
         * @param {?} _config
         * @param {?} _ngZone
         * @param {?} _keyboardDispatcher
         * @param {?} _document
         * @param {?=} _location
         */
        constructor(_portalOutlet, _host, _pane, _config, _ngZone, _keyboardDispatcher, _document, _location) {
            this._portalOutlet = _portalOutlet;
            this._host = _host;
            this._pane = _pane;
            this._config = _config;
            this._ngZone = _ngZone;
            this._keyboardDispatcher = _keyboardDispatcher;
            this._document = _document;
            this._location = _location;
            this._backdropElement = null;
            this._backdropClick = new Subject();
            this._attachments = new Subject();
            this._detachments = new Subject();
            this._locationChanges = Subscription.EMPTY;
            this._backdropClickHandler = (/**
             * @param {?} event
             * @return {?}
             */
            (event) => this._backdropClick.next(event));
            /**
             * Stream of keydown events dispatched to this overlay.
             */
            this._keydownEvents = new Subject();
            if (_config.scrollStrategy) {
                this._scrollStrategy = _config.scrollStrategy;
                this._scrollStrategy.attach(this);
            }
            this._positionStrategy = _config.positionStrategy;
        }
        /**
         * The overlay's HTML element
         * @return {?}
         */
        get overlayElement() {
            return this._pane;
        }
        /**
         * The overlay's backdrop HTML element.
         * @return {?}
         */
        get backdropElement() {
            return this._backdropElement;
        }
        /**
         * Wrapper around the panel element. Can be used for advanced
         * positioning where a wrapper with specific styling is
         * required around the overlay pane.
         * @return {?}
         */
        get hostElement() {
            return this._host;
        }
        /**
         * Attaches content, given via a Portal, to the overlay.
         * If the overlay is configured to have a backdrop, it will be created.
         *
         * @param {?} portal Portal instance to which to attach the overlay.
         * @return {?} The portal attachment result.
         */
        attach(portal) {
            /** @type {?} */
            let attachResult = this._portalOutlet.attach(portal);
            // Update the pane element with the given configuration.
            if (!this._host.parentElement && this._previousHostParent) {
                this._previousHostParent.appendChild(this._host);
            }
            if (this._positionStrategy) {
                this._positionStrategy.attach(this);
            }
            this._updateStackingOrder();
            this._updateElementSize();
            this._updateElementDirection();
            if (this._scrollStrategy) {
                this._scrollStrategy.enable();
            }
            // Update the position once the zone is stable so that the overlay will be fully rendered
            // before attempting to position it, as the position may depend on the size of the rendered
            // content.
            this._ngZone.onStable
                .asObservable()
                .pipe(take(1))
                .subscribe((/**
             * @return {?}
             */
            () => {
                // The overlay could've been detached before the zone has stabilized.
                if (this.hasAttached()) {
                    this.updatePosition();
                }
            }));
            // Enable pointer events for the overlay pane element.
            this._togglePointerEvents(true);
            if (this._config.hasBackdrop) {
                this._attachBackdrop();
            }
            if (this._config.panelClass) {
                this._toggleClasses(this._pane, this._config.panelClass, true);
            }
            // Only emit the `attachments` event once all other setup is done.
            this._attachments.next();
            // Track this overlay by the keyboard dispatcher
            this._keyboardDispatcher.add(this);
            // @breaking-change 8.0.0 remove the null check for `_location`
            // once the constructor parameter is made required.
            if (this._config.disposeOnNavigation && this._location) {
                this._locationChanges = this._location.subscribe((/**
                 * @return {?}
                 */
                () => this.dispose()));
            }
            return attachResult;
        }
        /**
         * Detaches an overlay from a portal.
         * @return {?} The portal detachment result.
         */
        detach() {
            if (!this.hasAttached()) {
                return;
            }
            this.detachBackdrop();
            // When the overlay is detached, the pane element should disable pointer events.
            // This is necessary because otherwise the pane element will cover the page and disable
            // pointer events therefore. Depends on the position strategy and the applied pane boundaries.
            this._togglePointerEvents(false);
            if (this._positionStrategy && this._positionStrategy.detach) {
                this._positionStrategy.detach();
            }
            if (this._scrollStrategy) {
                this._scrollStrategy.disable();
            }
            /** @type {?} */
            const detachmentResult = this._portalOutlet.detach();
            // Only emit after everything is detached.
            this._detachments.next();
            // Remove this overlay from keyboard dispatcher tracking.
            this._keyboardDispatcher.remove(this);
            // Keeping the host element in the DOM can cause scroll jank, because it still gets
            // rendered, even though it's transparent and unclickable which is why we remove it.
            this._detachContentWhenStable();
            // Stop listening for location changes.
            this._locationChanges.unsubscribe();
            return detachmentResult;
        }
        /**
         * Cleans up the overlay from the DOM.
         * @return {?}
         */
        dispose() {
            /** @type {?} */
            const isAttached = this.hasAttached();
            if (this._positionStrategy) {
                this._positionStrategy.dispose();
            }
            this._disposeScrollStrategy();
            this.detachBackdrop();
            this._locationChanges.unsubscribe();
            this._keyboardDispatcher.remove(this);
            this._portalOutlet.dispose();
            this._attachments.complete();
            this._backdropClick.complete();
            this._keydownEvents.complete();
            if (this._host && this._host.parentNode) {
                this._host.parentNode.removeChild(this._host);
                this._host = (/** @type {?} */ (null));
            }
            this._previousHostParent = this._pane = (/** @type {?} */ (null));
            if (isAttached) {
                this._detachments.next();
            }
            this._detachments.complete();
        }
        /**
         * Whether the overlay has attached content.
         * @return {?}
         */
        hasAttached() {
            return this._portalOutlet.hasAttached();
        }
        /**
         * Gets an observable that emits when the backdrop has been clicked.
         * @return {?}
         */
        backdropClick() {
            return this._backdropClick.asObservable();
        }
        /**
         * Gets an observable that emits when the overlay has been attached.
         * @return {?}
         */
        attachments() {
            return this._attachments.asObservable();
        }
        /**
         * Gets an observable that emits when the overlay has been detached.
         * @return {?}
         */
        detachments() {
            return this._detachments.asObservable();
        }
        /**
         * Gets an observable of keydown events targeted to this overlay.
         * @return {?}
         */
        keydownEvents() {
            return this._keydownEvents.asObservable();
        }
        /**
         * Gets the current overlay configuration, which is immutable.
         * @return {?}
         */
        getConfig() {
            return this._config;
        }
        /**
         * Updates the position of the overlay based on the position strategy.
         * @return {?}
         */
        updatePosition() {
            if (this._positionStrategy) {
                this._positionStrategy.apply();
            }
        }
        /**
         * Switches to a new position strategy and updates the overlay position.
         * @param {?} strategy
         * @return {?}
         */
        updatePositionStrategy(strategy) {
            if (strategy === this._positionStrategy) {
                return;
            }
            if (this._positionStrategy) {
                this._positionStrategy.dispose();
            }
            this._positionStrategy = strategy;
            if (this.hasAttached()) {
                strategy.attach(this);
                this.updatePosition();
            }
        }
        /**
         * Update the size properties of the overlay.
         * @param {?} sizeConfig
         * @return {?}
         */
        updateSize(sizeConfig) {
            this._config = Object.assign(Object.assign({}, this._config), sizeConfig);
            this._updateElementSize();
        }
        /**
         * Sets the LTR/RTL direction for the overlay.
         * @param {?} dir
         * @return {?}
         */
        setDirection(dir) {
            this._config = Object.assign(Object.assign({}, this._config), { direction: dir });
            this._updateElementDirection();
        }
        /**
         * Add a CSS class or an array of classes to the overlay pane.
         * @param {?} classes
         * @return {?}
         */
        addPanelClass(classes) {
            if (this._pane) {
                this._toggleClasses(this._pane, classes, true);
            }
        }
        /**
         * Remove a CSS class or an array of classes from the overlay pane.
         * @param {?} classes
         * @return {?}
         */
        removePanelClass(classes) {
            if (this._pane) {
                this._toggleClasses(this._pane, classes, false);
            }
        }
        /**
         * Returns the layout direction of the overlay panel.
         * @return {?}
         */
        getDirection() {
            /** @type {?} */
            const direction = this._config.direction;
            if (!direction) {
                return 'ltr';
            }
            return typeof direction === 'string' ? direction : direction.value;
        }
        /**
         * Switches to a new scroll strategy.
         * @param {?} strategy
         * @return {?}
         */
        updateScrollStrategy(strategy) {
            if (strategy === this._scrollStrategy) {
                return;
            }
            this._disposeScrollStrategy();
            this._scrollStrategy = strategy;
            if (this.hasAttached()) {
                strategy.attach(this);
                strategy.enable();
            }
        }
        /**
         * Updates the text direction of the overlay panel.
         * @private
         * @return {?}
         */
        _updateElementDirection() {
            this._host.setAttribute('dir', this.getDirection());
        }
        /**
         * Updates the size of the overlay element based on the overlay config.
         * @private
         * @return {?}
         */
        _updateElementSize() {
            if (!this._pane) {
                return;
            }
            /** @type {?} */
            const style = this._pane.style;
            style.width = coerceCssPixelValue(this._config.width);
            style.height = coerceCssPixelValue(this._config.height);
            style.minWidth = coerceCssPixelValue(this._config.minWidth);
            style.minHeight = coerceCssPixelValue(this._config.minHeight);
            style.maxWidth = coerceCssPixelValue(this._config.maxWidth);
            style.maxHeight = coerceCssPixelValue(this._config.maxHeight);
        }
        /**
         * Toggles the pointer events for the overlay pane element.
         * @private
         * @param {?} enablePointer
         * @return {?}
         */
        _togglePointerEvents(enablePointer) {
            this._pane.style.pointerEvents = enablePointer ? 'auto' : 'none';
        }
        /**
         * Attaches a backdrop for this overlay.
         * @private
         * @return {?}
         */
        _attachBackdrop() {
            /** @type {?} */
            const showingClass = 'cdk-overlay-backdrop-showing';
            this._backdropElement = this._document.createElement('div');
            this._backdropElement.classList.add('cdk-overlay-backdrop');
            if (this._config.backdropClass) {
                this._toggleClasses(this._backdropElement, this._config.backdropClass, true);
            }
            // Insert the backdrop before the pane in the DOM order,
            // in order to handle stacked overlays properly.
            (/** @type {?} */ (this._host.parentElement)).insertBefore(this._backdropElement, this._host);
            // Forward backdrop clicks such that the consumer of the overlay can perform whatever
            // action desired when such a click occurs (usually closing the overlay).
            this._backdropElement.addEventListener('click', this._backdropClickHandler);
            // Add class to fade-in the backdrop after one frame.
            if (typeof requestAnimationFrame !== 'undefined') {
                this._ngZone.runOutsideAngular((/**
                 * @return {?}
                 */
                () => {
                    requestAnimationFrame((/**
                     * @return {?}
                     */
                    () => {
                        if (this._backdropElement) {
                            this._backdropElement.classList.add(showingClass);
                        }
                    }));
                }));
            }
            else {
                this._backdropElement.classList.add(showingClass);
            }
        }
        /**
         * Updates the stacking order of the element, moving it to the top if necessary.
         * This is required in cases where one overlay was detached, while another one,
         * that should be behind it, was destroyed. The next time both of them are opened,
         * the stacking will be wrong, because the detached element's pane will still be
         * in its original DOM position.
         * @private
         * @return {?}
         */
        _updateStackingOrder() {
            if (this._host.nextSibling) {
                (/** @type {?} */ (this._host.parentNode)).appendChild(this._host);
            }
        }
        /**
         * Detaches the backdrop (if any) associated with the overlay.
         * @return {?}
         */
        detachBackdrop() {
            /** @type {?} */
            let backdropToDetach = this._backdropElement;
            if (!backdropToDetach) {
                return;
            }
            /** @type {?} */
            let timeoutId;
            /** @type {?} */
            let finishDetach = (/**
             * @return {?}
             */
            () => {
                // It may not be attached to anything in certain cases (e.g. unit tests).
                if (backdropToDetach) {
                    backdropToDetach.removeEventListener('click', this._backdropClickHandler);
                    backdropToDetach.removeEventListener('transitionend', finishDetach);
                    if (backdropToDetach.parentNode) {
                        backdropToDetach.parentNode.removeChild(backdropToDetach);
                    }
                }
                // It is possible that a new portal has been attached to this overlay since we started
                // removing the backdrop. If that is the case, only clear the backdrop reference if it
                // is still the same instance that we started to remove.
                if (this._backdropElement == backdropToDetach) {
                    this._backdropElement = null;
                }
                if (this._config.backdropClass) {
                    this._toggleClasses((/** @type {?} */ (backdropToDetach)), this._config.backdropClass, false);
                }
                clearTimeout(timeoutId);
            });
            backdropToDetach.classList.remove('cdk-overlay-backdrop-showing');
            this._ngZone.runOutsideAngular((/**
             * @return {?}
             */
            () => {
                (/** @type {?} */ (backdropToDetach)).addEventListener('transitionend', finishDetach);
            }));
            // If the backdrop doesn't have a transition, the `transitionend` event won't fire.
            // In this case we make it unclickable and we try to remove it after a delay.
            backdropToDetach.style.pointerEvents = 'none';
            // Run this outside the Angular zone because there's nothing that Angular cares about.
            // If it were to run inside the Angular zone, every test that used Overlay would have to be
            // either async or fakeAsync.
            timeoutId = this._ngZone.runOutsideAngular((/**
             * @return {?}
             */
            () => setTimeout(finishDetach, 500)));
        }
        /**
         * Toggles a single CSS class or an array of classes on an element.
         * @private
         * @param {?} element
         * @param {?} cssClasses
         * @param {?} isAdd
         * @return {?}
         */
        _toggleClasses(element, cssClasses, isAdd) {
            /** @type {?} */
            const classList = element.classList;
            coerceArray(cssClasses).forEach((/**
             * @param {?} cssClass
             * @return {?}
             */
            cssClass => {
                // We can't do a spread here, because IE doesn't support setting multiple classes.
                // Also trying to add an empty string to a DOMTokenList will throw.
                if (cssClass) {
                    isAdd ? classList.add(cssClass) : classList.remove(cssClass);
                }
            }));
        }
        /**
         * Detaches the overlay content next time the zone stabilizes.
         * @private
         * @return {?}
         */
        _detachContentWhenStable() {
            // Normally we wouldn't have to explicitly run this outside the `NgZone`, however
            // if the consumer is using `zone-patch-rxjs`, the `Subscription.unsubscribe` call will
            // be patched to run inside the zone, which will throw us into an infinite loop.
            this._ngZone.runOutsideAngular((/**
             * @return {?}
             */
            () => {
                // We can't remove the host here immediately, because the overlay pane's content
                // might still be animating. This stream helps us avoid interrupting the animation
                // by waiting for the pane to become empty.
                /** @type {?} */
                const subscription = this._ngZone.onStable
                    .asObservable()
                    .pipe(takeUntil(merge(this._attachments, this._detachments)))
                    .subscribe((/**
                 * @return {?}
                 */
                () => {
                    // Needs a couple of checks for the pane and host, because
                    // they may have been removed by the time the zone stabilizes.
                    if (!this._pane || !this._host || this._pane.children.length === 0) {
                        if (this._pane && this._config.panelClass) {
                            this._toggleClasses(this._pane, this._config.panelClass, false);
                        }
                        if (this._host && this._host.parentElement) {
                            this._previousHostParent = this._host.parentElement;
                            this._previousHostParent.removeChild(this._host);
                        }
                        subscription.unsubscribe();
                    }
                }));
            }));
        }
        /**
         * Disposes of a scroll strategy.
         * @private
         * @return {?}
         */
        _disposeScrollStrategy() {
            /** @type {?} */
            const scrollStrategy = this._scrollStrategy;
            if (scrollStrategy) {
                scrollStrategy.disable();
                if (scrollStrategy.detach) {
                    scrollStrategy.detach();
                }
            }
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/overlay/position/flexible-connected-position-strategy.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    // TODO: refactor clipping detection into a separate thing (part of scrolling module)
    // TODO: doesn't handle both flexible width and height when it has to scroll along both axis.
    /**
     * Class to be added to the overlay bounding box.
     * @type {?}
     */
    const boundingBoxClass = 'cdk-overlay-connected-position-bounding-box';
    /**
     * Regex used to split a string on its CSS units.
     * @type {?}
     */
    const cssUnitPattern = /([A-Za-z%]+)$/;
    /**
     * A strategy for positioning overlays. Using this strategy, an overlay is given an
     * implicit position relative some origin element. The relative position is defined in terms of
     * a point on the origin element that is connected to a point on the overlay element. For example,
     * a basic dropdown is connecting the bottom-left corner of the origin to the top-left corner
     * of the overlay.
     */
    class FlexibleConnectedPositionStrategy {
        /**
         * @param {?} connectedTo
         * @param {?} _viewportRuler
         * @param {?} _document
         * @param {?} _platform
         * @param {?} _overlayContainer
         */
        constructor(connectedTo, _viewportRuler, _document, _platform, _overlayContainer) {
            this._viewportRuler = _viewportRuler;
            this._document = _document;
            this._platform = _platform;
            this._overlayContainer = _overlayContainer;
            /**
             * Last size used for the bounding box. Used to avoid resizing the overlay after open.
             */
            this._lastBoundingBoxSize = { width: 0, height: 0 };
            /**
             * Whether the overlay was pushed in a previous positioning.
             */
            this._isPushed = false;
            /**
             * Whether the overlay can be pushed on-screen on the initial open.
             */
            this._canPush = true;
            /**
             * Whether the overlay can grow via flexible width/height after the initial open.
             */
            this._growAfterOpen = false;
            /**
             * Whether the overlay's width and height can be constrained to fit within the viewport.
             */
            this._hasFlexibleDimensions = true;
            /**
             * Whether the overlay position is locked.
             */
            this._positionLocked = false;
            /**
             * Amount of space that must be maintained between the overlay and the edge of the viewport.
             */
            this._viewportMargin = 0;
            /**
             * The Scrollable containers used to check scrollable view properties on position change.
             */
            this._scrollables = [];
            /**
             * Ordered list of preferred positions, from most to least desirable.
             */
            this._preferredPositions = [];
            /**
             * Subject that emits whenever the position changes.
             */
            this._positionChanges = new Subject();
            /**
             * Subscription to viewport size changes.
             */
            this._resizeSubscription = Subscription.EMPTY;
            /**
             * Default offset for the overlay along the x axis.
             */
            this._offsetX = 0;
            /**
             * Default offset for the overlay along the y axis.
             */
            this._offsetY = 0;
            /**
             * Keeps track of the CSS classes that the position strategy has applied on the overlay panel.
             */
            this._appliedPanelClasses = [];
            /**
             * Observable sequence of position changes.
             */
            this.positionChanges = this._positionChanges.asObservable();
            this.setOrigin(connectedTo);
        }
        /**
         * Ordered list of preferred positions, from most to least desirable.
         * @return {?}
         */
        get positions() {
            return this._preferredPositions;
        }
        /**
         * Attaches this position strategy to an overlay.
         * @param {?} overlayRef
         * @return {?}
         */
        attach(overlayRef) {
            if (this._overlayRef && overlayRef !== this._overlayRef) {
                throw Error('This position strategy is already attached to an overlay');
            }
            this._validatePositions();
            overlayRef.hostElement.classList.add(boundingBoxClass);
            this._overlayRef = overlayRef;
            this._boundingBox = overlayRef.hostElement;
            this._pane = overlayRef.overlayElement;
            this._isDisposed = false;
            this._isInitialRender = true;
            this._lastPosition = null;
            this._resizeSubscription.unsubscribe();
            this._resizeSubscription = this._viewportRuler.change().subscribe((/**
             * @return {?}
             */
            () => {
                // When the window is resized, we want to trigger the next reposition as if it
                // was an initial render, in order for the strategy to pick a new optimal position,
                // otherwise position locking will cause it to stay at the old one.
                this._isInitialRender = true;
                this.apply();
            }));
        }
        /**
         * Updates the position of the overlay element, using whichever preferred position relative
         * to the origin best fits on-screen.
         *
         * The selection of a position goes as follows:
         *  - If any positions fit completely within the viewport as-is,
         *      choose the first position that does so.
         *  - If flexible dimensions are enabled and at least one satifies the given minimum width/height,
         *      choose the position with the greatest available size modified by the positions' weight.
         *  - If pushing is enabled, take the position that went off-screen the least and push it
         *      on-screen.
         *  - If none of the previous criteria were met, use the position that goes off-screen the least.
         * \@docs-private
         * @return {?}
         */
        apply() {
            // We shouldn't do anything if the strategy was disposed or we're on the server.
            if (this._isDisposed || !this._platform.isBrowser) {
                return;
            }
            // If the position has been applied already (e.g. when the overlay was opened) and the
            // consumer opted into locking in the position, re-use the old position, in order to
            // prevent the overlay from jumping around.
            if (!this._isInitialRender && this._positionLocked && this._lastPosition) {
                this.reapplyLastPosition();
                return;
            }
            this._clearPanelClasses();
            this._resetOverlayElementStyles();
            this._resetBoundingBoxStyles();
            // We need the bounding rects for the origin and the overlay to determine how to position
            // the overlay relative to the origin.
            // We use the viewport rect to determine whether a position would go off-screen.
            this._viewportRect = this._getNarrowedViewportRect();
            this._originRect = this._getOriginRect();
            this._overlayRect = this._pane.getBoundingClientRect();
            /** @type {?} */
            const originRect = this._originRect;
            /** @type {?} */
            const overlayRect = this._overlayRect;
            /** @type {?} */
            const viewportRect = this._viewportRect;
            // Positions where the overlay will fit with flexible dimensions.
            /** @type {?} */
            const flexibleFits = [];
            // Fallback if none of the preferred positions fit within the viewport.
            /** @type {?} */
            let fallback;
            // Go through each of the preferred positions looking for a good fit.
            // If a good fit is found, it will be applied immediately.
            for (let pos of this._preferredPositions) {
                // Get the exact (x, y) coordinate for the point-of-origin on the origin element.
                /** @type {?} */
                let originPoint = this._getOriginPoint(originRect, pos);
                // From that point-of-origin, get the exact (x, y) coordinate for the top-left corner of the
                // overlay in this position. We use the top-left corner for calculations and later translate
                // this into an appropriate (top, left, bottom, right) style.
                /** @type {?} */
                let overlayPoint = this._getOverlayPoint(originPoint, overlayRect, pos);
                // Calculate how well the overlay would fit into the viewport with this point.
                /** @type {?} */
                let overlayFit = this._getOverlayFit(overlayPoint, overlayRect, viewportRect, pos);
                // If the overlay, without any further work, fits into the viewport, use this position.
                if (overlayFit.isCompletelyWithinViewport) {
                    this._isPushed = false;
                    this._applyPosition(pos, originPoint);
                    return;
                }
                // If the overlay has flexible dimensions, we can use this position
                // so long as there's enough space for the minimum dimensions.
                if (this._canFitWithFlexibleDimensions(overlayFit, overlayPoint, viewportRect)) {
                    // Save positions where the overlay will fit with flexible dimensions. We will use these
                    // if none of the positions fit *without* flexible dimensions.
                    flexibleFits.push({
                        position: pos,
                        origin: originPoint,
                        overlayRect,
                        boundingBoxRect: this._calculateBoundingBoxRect(originPoint, pos)
                    });
                    continue;
                }
                // If the current preferred position does not fit on the screen, remember the position
                // if it has more visible area on-screen than we've seen and move onto the next preferred
                // position.
                if (!fallback || fallback.overlayFit.visibleArea < overlayFit.visibleArea) {
                    fallback = { overlayFit, overlayPoint, originPoint, position: pos, overlayRect };
                }
            }
            // If there are any positions where the overlay would fit with flexible dimensions, choose the
            // one that has the greatest area available modified by the position's weight
            if (flexibleFits.length) {
                /** @type {?} */
                let bestFit = null;
                /** @type {?} */
                let bestScore = -1;
                for (const fit of flexibleFits) {
                    /** @type {?} */
                    const score = fit.boundingBoxRect.width * fit.boundingBoxRect.height * (fit.position.weight || 1);
                    if (score > bestScore) {
                        bestScore = score;
                        bestFit = fit;
                    }
                }
                this._isPushed = false;
                this._applyPosition((/** @type {?} */ (bestFit)).position, (/** @type {?} */ (bestFit)).origin);
                return;
            }
            // When none of the preferred positions fit within the viewport, take the position
            // that went off-screen the least and attempt to push it on-screen.
            if (this._canPush) {
                // TODO(jelbourn): after pushing, the opening "direction" of the overlay might not make sense.
                this._isPushed = true;
                this._applyPosition((/** @type {?} */ (fallback)).position, (/** @type {?} */ (fallback)).originPoint);
                return;
            }
            // All options for getting the overlay within the viewport have been exhausted, so go with the
            // position that went off-screen the least.
            this._applyPosition((/** @type {?} */ (fallback)).position, (/** @type {?} */ (fallback)).originPoint);
        }
        /**
         * @return {?}
         */
        detach() {
            this._clearPanelClasses();
            this._lastPosition = null;
            this._previousPushAmount = null;
            this._resizeSubscription.unsubscribe();
        }
        /**
         * Cleanup after the element gets destroyed.
         * @return {?}
         */
        dispose() {
            if (this._isDisposed) {
                return;
            }
            // We can't use `_resetBoundingBoxStyles` here, because it resets
            // some properties to zero, rather than removing them.
            if (this._boundingBox) {
                extendStyles(this._boundingBox.style, (/** @type {?} */ ({
                    top: '',
                    left: '',
                    right: '',
                    bottom: '',
                    height: '',
                    width: '',
                    alignItems: '',
                    justifyContent: '',
                })));
            }
            if (this._pane) {
                this._resetOverlayElementStyles();
            }
            if (this._overlayRef) {
                this._overlayRef.hostElement.classList.remove(boundingBoxClass);
            }
            this.detach();
            this._positionChanges.complete();
            this._overlayRef = this._boundingBox = (/** @type {?} */ (null));
            this._isDisposed = true;
        }
        /**
         * This re-aligns the overlay element with the trigger in its last calculated position,
         * even if a position higher in the "preferred positions" list would now fit. This
         * allows one to re-align the panel without changing the orientation of the panel.
         * @return {?}
         */
        reapplyLastPosition() {
            if (!this._isDisposed && (!this._platform || this._platform.isBrowser)) {
                this._originRect = this._getOriginRect();
                this._overlayRect = this._pane.getBoundingClientRect();
                this._viewportRect = this._getNarrowedViewportRect();
                /** @type {?} */
                const lastPosition = this._lastPosition || this._preferredPositions[0];
                /** @type {?} */
                const originPoint = this._getOriginPoint(this._originRect, lastPosition);
                this._applyPosition(lastPosition, originPoint);
            }
        }
        /**
         * Sets the list of Scrollable containers that host the origin element so that
         * on reposition we can evaluate if it or the overlay has been clipped or outside view. Every
         * Scrollable must be an ancestor element of the strategy's origin element.
         * @template THIS
         * @this {THIS}
         * @param {?} scrollables
         * @return {THIS}
         */
        withScrollableContainers(scrollables) {
            (/** @type {?} */ (this))._scrollables = scrollables;
            return (/** @type {?} */ (this));
        }
        /**
         * Adds new preferred positions.
         * @template THIS
         * @this {THIS}
         * @param {?} positions List of positions options for this overlay.
         * @return {THIS}
         */
        withPositions(positions) {
            (/** @type {?} */ (this))._preferredPositions = positions;
            // If the last calculated position object isn't part of the positions anymore, clear
            // it in order to avoid it being picked up if the consumer tries to re-apply.
            if (positions.indexOf((/** @type {?} */ ((/** @type {?} */ (this))._lastPosition))) === -1) {
                (/** @type {?} */ (this))._lastPosition = null;
            }
            (/** @type {?} */ (this))._validatePositions();
            return (/** @type {?} */ (this));
        }
        /**
         * Sets a minimum distance the overlay may be positioned to the edge of the viewport.
         * @template THIS
         * @this {THIS}
         * @param {?} margin Required margin between the overlay and the viewport edge in pixels.
         * @return {THIS}
         */
        withViewportMargin(margin) {
            (/** @type {?} */ (this))._viewportMargin = margin;
            return (/** @type {?} */ (this));
        }
        /**
         * Sets whether the overlay's width and height can be constrained to fit within the viewport.
         * @template THIS
         * @this {THIS}
         * @param {?=} flexibleDimensions
         * @return {THIS}
         */
        withFlexibleDimensions(flexibleDimensions = true) {
            (/** @type {?} */ (this))._hasFlexibleDimensions = flexibleDimensions;
            return (/** @type {?} */ (this));
        }
        /**
         * Sets whether the overlay can grow after the initial open via flexible width/height.
         * @template THIS
         * @this {THIS}
         * @param {?=} growAfterOpen
         * @return {THIS}
         */
        withGrowAfterOpen(growAfterOpen = true) {
            (/** @type {?} */ (this))._growAfterOpen = growAfterOpen;
            return (/** @type {?} */ (this));
        }
        /**
         * Sets whether the overlay can be pushed on-screen if none of the provided positions fit.
         * @template THIS
         * @this {THIS}
         * @param {?=} canPush
         * @return {THIS}
         */
        withPush(canPush = true) {
            (/** @type {?} */ (this))._canPush = canPush;
            return (/** @type {?} */ (this));
        }
        /**
         * Sets whether the overlay's position should be locked in after it is positioned
         * initially. When an overlay is locked in, it won't attempt to reposition itself
         * when the position is re-applied (e.g. when the user scrolls away).
         * @template THIS
         * @this {THIS}
         * @param {?=} isLocked Whether the overlay should locked in.
         * @return {THIS}
         */
        withLockedPosition(isLocked = true) {
            (/** @type {?} */ (this))._positionLocked = isLocked;
            return (/** @type {?} */ (this));
        }
        /**
         * Sets the origin, relative to which to position the overlay.
         * Using an element origin is useful for building components that need to be positioned
         * relatively to a trigger (e.g. dropdown menus or tooltips), whereas using a point can be
         * used for cases like contextual menus which open relative to the user's pointer.
         * @template THIS
         * @this {THIS}
         * @param {?} origin Reference to the new origin.
         * @return {THIS}
         */
        setOrigin(origin) {
            (/** @type {?} */ (this))._origin = origin;
            return (/** @type {?} */ (this));
        }
        /**
         * Sets the default offset for the overlay's connection point on the x-axis.
         * @template THIS
         * @this {THIS}
         * @param {?} offset New offset in the X axis.
         * @return {THIS}
         */
        withDefaultOffsetX(offset) {
            (/** @type {?} */ (this))._offsetX = offset;
            return (/** @type {?} */ (this));
        }
        /**
         * Sets the default offset for the overlay's connection point on the y-axis.
         * @template THIS
         * @this {THIS}
         * @param {?} offset New offset in the Y axis.
         * @return {THIS}
         */
        withDefaultOffsetY(offset) {
            (/** @type {?} */ (this))._offsetY = offset;
            return (/** @type {?} */ (this));
        }
        /**
         * Configures that the position strategy should set a `transform-origin` on some elements
         * inside the overlay, depending on the current position that is being applied. This is
         * useful for the cases where the origin of an animation can change depending on the
         * alignment of the overlay.
         * @template THIS
         * @this {THIS}
         * @param {?} selector CSS selector that will be used to find the target
         *    elements onto which to set the transform origin.
         * @return {THIS}
         */
        withTransformOriginOn(selector) {
            (/** @type {?} */ (this))._transformOriginSelector = selector;
            return (/** @type {?} */ (this));
        }
        /**
         * Gets the (x, y) coordinate of a connection point on the origin based on a relative position.
         * @private
         * @param {?} originRect
         * @param {?} pos
         * @return {?}
         */
        _getOriginPoint(originRect, pos) {
            /** @type {?} */
            let x;
            if (pos.originX == 'center') {
                // Note: when centering we should always use the `left`
                // offset, otherwise the position will be wrong in RTL.
                x = originRect.left + (originRect.width / 2);
            }
            else {
                /** @type {?} */
                const startX = this._isRtl() ? originRect.right : originRect.left;
                /** @type {?} */
                const endX = this._isRtl() ? originRect.left : originRect.right;
                x = pos.originX == 'start' ? startX : endX;
            }
            /** @type {?} */
            let y;
            if (pos.originY == 'center') {
                y = originRect.top + (originRect.height / 2);
            }
            else {
                y = pos.originY == 'top' ? originRect.top : originRect.bottom;
            }
            return { x, y };
        }
        /**
         * Gets the (x, y) coordinate of the top-left corner of the overlay given a given position and
         * origin point to which the overlay should be connected.
         * @private
         * @param {?} originPoint
         * @param {?} overlayRect
         * @param {?} pos
         * @return {?}
         */
        _getOverlayPoint(originPoint, overlayRect, pos) {
            // Calculate the (overlayStartX, overlayStartY), the start of the
            // potential overlay position relative to the origin point.
            /** @type {?} */
            let overlayStartX;
            if (pos.overlayX == 'center') {
                overlayStartX = -overlayRect.width / 2;
            }
            else if (pos.overlayX === 'start') {
                overlayStartX = this._isRtl() ? -overlayRect.width : 0;
            }
            else {
                overlayStartX = this._isRtl() ? 0 : -overlayRect.width;
            }
            /** @type {?} */
            let overlayStartY;
            if (pos.overlayY == 'center') {
                overlayStartY = -overlayRect.height / 2;
            }
            else {
                overlayStartY = pos.overlayY == 'top' ? 0 : -overlayRect.height;
            }
            // The (x, y) coordinates of the overlay.
            return {
                x: originPoint.x + overlayStartX,
                y: originPoint.y + overlayStartY,
            };
        }
        /**
         * Gets how well an overlay at the given point will fit within the viewport.
         * @private
         * @param {?} point
         * @param {?} overlay
         * @param {?} viewport
         * @param {?} position
         * @return {?}
         */
        _getOverlayFit(point, overlay, viewport, position) {
            let { x, y } = point;
            /** @type {?} */
            let offsetX = this._getOffset(position, 'x');
            /** @type {?} */
            let offsetY = this._getOffset(position, 'y');
            // Account for the offsets since they could push the overlay out of the viewport.
            if (offsetX) {
                x += offsetX;
            }
            if (offsetY) {
                y += offsetY;
            }
            // How much the overlay would overflow at this position, on each side.
            /** @type {?} */
            let leftOverflow = 0 - x;
            /** @type {?} */
            let rightOverflow = (x + overlay.width) - viewport.width;
            /** @type {?} */
            let topOverflow = 0 - y;
            /** @type {?} */
            let bottomOverflow = (y + overlay.height) - viewport.height;
            // Visible parts of the element on each axis.
            /** @type {?} */
            let visibleWidth = this._subtractOverflows(overlay.width, leftOverflow, rightOverflow);
            /** @type {?} */
            let visibleHeight = this._subtractOverflows(overlay.height, topOverflow, bottomOverflow);
            /** @type {?} */
            let visibleArea = visibleWidth * visibleHeight;
            return {
                visibleArea,
                isCompletelyWithinViewport: (overlay.width * overlay.height) === visibleArea,
                fitsInViewportVertically: visibleHeight === overlay.height,
                fitsInViewportHorizontally: visibleWidth == overlay.width,
            };
        }
        /**
         * Whether the overlay can fit within the viewport when it may resize either its width or height.
         * @private
         * @param {?} fit How well the overlay fits in the viewport at some position.
         * @param {?} point The (x, y) coordinates of the overlat at some position.
         * @param {?} viewport The geometry of the viewport.
         * @return {?}
         */
        _canFitWithFlexibleDimensions(fit, point, viewport) {
            if (this._hasFlexibleDimensions) {
                /** @type {?} */
                const availableHeight = viewport.bottom - point.y;
                /** @type {?} */
                const availableWidth = viewport.right - point.x;
                /** @type {?} */
                const minHeight = getPixelValue(this._overlayRef.getConfig().minHeight);
                /** @type {?} */
                const minWidth = getPixelValue(this._overlayRef.getConfig().minWidth);
                /** @type {?} */
                const verticalFit = fit.fitsInViewportVertically ||
                    (minHeight != null && minHeight <= availableHeight);
                /** @type {?} */
                const horizontalFit = fit.fitsInViewportHorizontally ||
                    (minWidth != null && minWidth <= availableWidth);
                return verticalFit && horizontalFit;
            }
            return false;
        }
        /**
         * Gets the point at which the overlay can be "pushed" on-screen. If the overlay is larger than
         * the viewport, the top-left corner will be pushed on-screen (with overflow occuring on the
         * right and bottom).
         *
         * @private
         * @param {?} start Starting point from which the overlay is pushed.
         * @param {?} overlay Dimensions of the overlay.
         * @param {?} scrollPosition Current viewport scroll position.
         * @return {?} The point at which to position the overlay after pushing. This is effectively a new
         *     originPoint.
         */
        _pushOverlayOnScreen(start, overlay, scrollPosition) {
            // If the position is locked and we've pushed the overlay already, reuse the previous push
            // amount, rather than pushing it again. If we were to continue pushing, the element would
            // remain in the viewport, which goes against the expectations when position locking is enabled.
            if (this._previousPushAmount && this._positionLocked) {
                return {
                    x: start.x + this._previousPushAmount.x,
                    y: start.y + this._previousPushAmount.y
                };
            }
            /** @type {?} */
            const viewport = this._viewportRect;
            // Determine how much the overlay goes outside the viewport on each
            // side, which we'll use to decide which direction to push it.
            /** @type {?} */
            const overflowRight = Math.max(start.x + overlay.width - viewport.right, 0);
            /** @type {?} */
            const overflowBottom = Math.max(start.y + overlay.height - viewport.bottom, 0);
            /** @type {?} */
            const overflowTop = Math.max(viewport.top - scrollPosition.top - start.y, 0);
            /** @type {?} */
            const overflowLeft = Math.max(viewport.left - scrollPosition.left - start.x, 0);
            // Amount by which to push the overlay in each axis such that it remains on-screen.
            /** @type {?} */
            let pushX = 0;
            /** @type {?} */
            let pushY = 0;
            // If the overlay fits completely within the bounds of the viewport, push it from whichever
            // direction is goes off-screen. Otherwise, push the top-left corner such that its in the
            // viewport and allow for the trailing end of the overlay to go out of bounds.
            if (overlay.width <= viewport.width) {
                pushX = overflowLeft || -overflowRight;
            }
            else {
                pushX = start.x < this._viewportMargin ? (viewport.left - scrollPosition.left) - start.x : 0;
            }
            if (overlay.height <= viewport.height) {
                pushY = overflowTop || -overflowBottom;
            }
            else {
                pushY = start.y < this._viewportMargin ? (viewport.top - scrollPosition.top) - start.y : 0;
            }
            this._previousPushAmount = { x: pushX, y: pushY };
            return {
                x: start.x + pushX,
                y: start.y + pushY,
            };
        }
        /**
         * Applies a computed position to the overlay and emits a position change.
         * @private
         * @param {?} position The position preference
         * @param {?} originPoint The point on the origin element where the overlay is connected.
         * @return {?}
         */
        _applyPosition(position, originPoint) {
            this._setTransformOrigin(position);
            this._setOverlayElementStyles(originPoint, position);
            this._setBoundingBoxStyles(originPoint, position);
            if (position.panelClass) {
                this._addPanelClasses(position.panelClass);
            }
            // Save the last connected position in case the position needs to be re-calculated.
            this._lastPosition = position;
            // Notify that the position has been changed along with its change properties.
            // We only emit if we've got any subscriptions, because the scroll visibility
            // calculcations can be somewhat expensive.
            if (this._positionChanges.observers.length) {
                /** @type {?} */
                const scrollableViewProperties = this._getScrollVisibility();
                /** @type {?} */
                const changeEvent = new ConnectedOverlayPositionChange(position, scrollableViewProperties);
                this._positionChanges.next(changeEvent);
            }
            this._isInitialRender = false;
        }
        /**
         * Sets the transform origin based on the configured selector and the passed-in position.
         * @private
         * @param {?} position
         * @return {?}
         */
        _setTransformOrigin(position) {
            if (!this._transformOriginSelector) {
                return;
            }
            /** @type {?} */
            const elements = (/** @type {?} */ (this._boundingBox)).querySelectorAll(this._transformOriginSelector);
            /** @type {?} */
            let xOrigin;
            /** @type {?} */
            let yOrigin = position.overlayY;
            if (position.overlayX === 'center') {
                xOrigin = 'center';
            }
            else if (this._isRtl()) {
                xOrigin = position.overlayX === 'start' ? 'right' : 'left';
            }
            else {
                xOrigin = position.overlayX === 'start' ? 'left' : 'right';
            }
            for (let i = 0; i < elements.length; i++) {
                elements[i].style.transformOrigin = `${xOrigin} ${yOrigin}`;
            }
        }
        /**
         * Gets the position and size of the overlay's sizing container.
         *
         * This method does no measuring and applies no styles so that we can cheaply compute the
         * bounds for all positions and choose the best fit based on these results.
         * @private
         * @param {?} origin
         * @param {?} position
         * @return {?}
         */
        _calculateBoundingBoxRect(origin, position) {
            /** @type {?} */
            const viewport = this._viewportRect;
            /** @type {?} */
            const isRtl = this._isRtl();
            /** @type {?} */
            let height;
            /** @type {?} */
            let top;
            /** @type {?} */
            let bottom;
            if (position.overlayY === 'top') {
                // Overlay is opening "downward" and thus is bound by the bottom viewport edge.
                top = origin.y;
                height = viewport.height - top + this._viewportMargin;
            }
            else if (position.overlayY === 'bottom') {
                // Overlay is opening "upward" and thus is bound by the top viewport edge. We need to add
                // the viewport margin back in, because the viewport rect is narrowed down to remove the
                // margin, whereas the `origin` position is calculated based on its `ClientRect`.
                bottom = viewport.height - origin.y + this._viewportMargin * 2;
                height = viewport.height - bottom + this._viewportMargin;
            }
            else {
                // If neither top nor bottom, it means that the overlay is vertically centered on the
                // origin point. Note that we want the position relative to the viewport, rather than
                // the page, which is why we don't use something like `viewport.bottom - origin.y` and
                // `origin.y - viewport.top`.
                /** @type {?} */
                const smallestDistanceToViewportEdge = Math.min(viewport.bottom - origin.y + viewport.top, origin.y);
                /** @type {?} */
                const previousHeight = this._lastBoundingBoxSize.height;
                height = smallestDistanceToViewportEdge * 2;
                top = origin.y - smallestDistanceToViewportEdge;
                if (height > previousHeight && !this._isInitialRender && !this._growAfterOpen) {
                    top = origin.y - (previousHeight / 2);
                }
            }
            // The overlay is opening 'right-ward' (the content flows to the right).
            /** @type {?} */
            const isBoundedByRightViewportEdge = (position.overlayX === 'start' && !isRtl) ||
                (position.overlayX === 'end' && isRtl);
            // The overlay is opening 'left-ward' (the content flows to the left).
            /** @type {?} */
            const isBoundedByLeftViewportEdge = (position.overlayX === 'end' && !isRtl) ||
                (position.overlayX === 'start' && isRtl);
            /** @type {?} */
            let width;
            /** @type {?} */
            let left;
            /** @type {?} */
            let right;
            if (isBoundedByLeftViewportEdge) {
                right = viewport.width - origin.x + this._viewportMargin;
                width = origin.x - this._viewportMargin;
            }
            else if (isBoundedByRightViewportEdge) {
                left = origin.x;
                width = viewport.right - origin.x;
            }
            else {
                // If neither start nor end, it means that the overlay is horizontally centered on the
                // origin point. Note that we want the position relative to the viewport, rather than
                // the page, which is why we don't use something like `viewport.right - origin.x` and
                // `origin.x - viewport.left`.
                /** @type {?} */
                const smallestDistanceToViewportEdge = Math.min(viewport.right - origin.x + viewport.left, origin.x);
                /** @type {?} */
                const previousWidth = this._lastBoundingBoxSize.width;
                width = smallestDistanceToViewportEdge * 2;
                left = origin.x - smallestDistanceToViewportEdge;
                if (width > previousWidth && !this._isInitialRender && !this._growAfterOpen) {
                    left = origin.x - (previousWidth / 2);
                }
            }
            return { top: (/** @type {?} */ (top)), left: (/** @type {?} */ (left)), bottom: (/** @type {?} */ (bottom)), right: (/** @type {?} */ (right)), width, height };
        }
        /**
         * Sets the position and size of the overlay's sizing wrapper. The wrapper is positioned on the
         * origin's connection point and stetches to the bounds of the viewport.
         *
         * @private
         * @param {?} origin The point on the origin element where the overlay is connected.
         * @param {?} position The position preference
         * @return {?}
         */
        _setBoundingBoxStyles(origin, position) {
            /** @type {?} */
            const boundingBoxRect = this._calculateBoundingBoxRect(origin, position);
            // It's weird if the overlay *grows* while scrolling, so we take the last size into account
            // when applying a new size.
            if (!this._isInitialRender && !this._growAfterOpen) {
                boundingBoxRect.height = Math.min(boundingBoxRect.height, this._lastBoundingBoxSize.height);
                boundingBoxRect.width = Math.min(boundingBoxRect.width, this._lastBoundingBoxSize.width);
            }
            /** @type {?} */
            const styles = (/** @type {?} */ ({}));
            if (this._hasExactPosition()) {
                styles.top = styles.left = '0';
                styles.bottom = styles.right = styles.maxHeight = styles.maxWidth = '';
                styles.width = styles.height = '100%';
            }
            else {
                /** @type {?} */
                const maxHeight = this._overlayRef.getConfig().maxHeight;
                /** @type {?} */
                const maxWidth = this._overlayRef.getConfig().maxWidth;
                styles.height = coerceCssPixelValue(boundingBoxRect.height);
                styles.top = coerceCssPixelValue(boundingBoxRect.top);
                styles.bottom = coerceCssPixelValue(boundingBoxRect.bottom);
                styles.width = coerceCssPixelValue(boundingBoxRect.width);
                styles.left = coerceCssPixelValue(boundingBoxRect.left);
                styles.right = coerceCssPixelValue(boundingBoxRect.right);
                // Push the pane content towards the proper direction.
                if (position.overlayX === 'center') {
                    styles.alignItems = 'center';
                }
                else {
                    styles.alignItems = position.overlayX === 'end' ? 'flex-end' : 'flex-start';
                }
                if (position.overlayY === 'center') {
                    styles.justifyContent = 'center';
                }
                else {
                    styles.justifyContent = position.overlayY === 'bottom' ? 'flex-end' : 'flex-start';
                }
                if (maxHeight) {
                    styles.maxHeight = coerceCssPixelValue(maxHeight);
                }
                if (maxWidth) {
                    styles.maxWidth = coerceCssPixelValue(maxWidth);
                }
            }
            this._lastBoundingBoxSize = boundingBoxRect;
            extendStyles((/** @type {?} */ (this._boundingBox)).style, styles);
        }
        /**
         * Resets the styles for the bounding box so that a new positioning can be computed.
         * @private
         * @return {?}
         */
        _resetBoundingBoxStyles() {
            extendStyles((/** @type {?} */ (this._boundingBox)).style, (/** @type {?} */ ({
                top: '0',
                left: '0',
                right: '0',
                bottom: '0',
                height: '',
                width: '',
                alignItems: '',
                justifyContent: '',
            })));
        }
        /**
         * Resets the styles for the overlay pane so that a new positioning can be computed.
         * @private
         * @return {?}
         */
        _resetOverlayElementStyles() {
            extendStyles(this._pane.style, (/** @type {?} */ ({
                top: '',
                left: '',
                bottom: '',
                right: '',
                position: '',
                transform: '',
            })));
        }
        /**
         * Sets positioning styles to the overlay element.
         * @private
         * @param {?} originPoint
         * @param {?} position
         * @return {?}
         */
        _setOverlayElementStyles(originPoint, position) {
            /** @type {?} */
            const styles = (/** @type {?} */ ({}));
            /** @type {?} */
            const hasExactPosition = this._hasExactPosition();
            /** @type {?} */
            const hasFlexibleDimensions = this._hasFlexibleDimensions;
            /** @type {?} */
            const config = this._overlayRef.getConfig();
            if (hasExactPosition) {
                /** @type {?} */
                const scrollPosition = this._viewportRuler.getViewportScrollPosition();
                extendStyles(styles, this._getExactOverlayY(position, originPoint, scrollPosition));
                extendStyles(styles, this._getExactOverlayX(position, originPoint, scrollPosition));
            }
            else {
                styles.position = 'static';
            }
            // Use a transform to apply the offsets. We do this because the `center` positions rely on
            // being in the normal flex flow and setting a `top` / `left` at all will completely throw
            // off the position. We also can't use margins, because they won't have an effect in some
            // cases where the element doesn't have anything to "push off of". Finally, this works
            // better both with flexible and non-flexible positioning.
            /** @type {?} */
            let transformString = '';
            /** @type {?} */
            let offsetX = this._getOffset(position, 'x');
            /** @type {?} */
            let offsetY = this._getOffset(position, 'y');
            if (offsetX) {
                transformString += `translateX(${offsetX}px) `;
            }
            if (offsetY) {
                transformString += `translateY(${offsetY}px)`;
            }
            styles.transform = transformString.trim();
            // If a maxWidth or maxHeight is specified on the overlay, we remove them. We do this because
            // we need these values to both be set to "100%" for the automatic flexible sizing to work.
            // The maxHeight and maxWidth are set on the boundingBox in order to enforce the constraint.
            // Note that this doesn't apply when we have an exact position, in which case we do want to
            // apply them because they'll be cleared from the bounding box.
            if (config.maxHeight) {
                if (hasExactPosition) {
                    styles.maxHeight = coerceCssPixelValue(config.maxHeight);
                }
                else if (hasFlexibleDimensions) {
                    styles.maxHeight = '';
                }
            }
            if (config.maxWidth) {
                if (hasExactPosition) {
                    styles.maxWidth = coerceCssPixelValue(config.maxWidth);
                }
                else if (hasFlexibleDimensions) {
                    styles.maxWidth = '';
                }
            }
            extendStyles(this._pane.style, styles);
        }
        /**
         * Gets the exact top/bottom for the overlay when not using flexible sizing or when pushing.
         * @private
         * @param {?} position
         * @param {?} originPoint
         * @param {?} scrollPosition
         * @return {?}
         */
        _getExactOverlayY(position, originPoint, scrollPosition) {
            // Reset any existing styles. This is necessary in case the
            // preferred position has changed since the last `apply`.
            /** @type {?} */
            let styles = (/** @type {?} */ ({ top: '', bottom: '' }));
            /** @type {?} */
            let overlayPoint = this._getOverlayPoint(originPoint, this._overlayRect, position);
            if (this._isPushed) {
                overlayPoint = this._pushOverlayOnScreen(overlayPoint, this._overlayRect, scrollPosition);
            }
            /** @type {?} */
            let virtualKeyboardOffset = this._overlayContainer.getContainerElement().getBoundingClientRect().top;
            // Normally this would be zero, however when the overlay is attached to an input (e.g. in an
            // autocomplete), mobile browsers will shift everything in order to put the input in the middle
            // of the screen and to make space for the virtual keyboard. We need to account for this offset,
            // otherwise our positioning will be thrown off.
            overlayPoint.y -= virtualKeyboardOffset;
            // We want to set either `top` or `bottom` based on whether the overlay wants to appear
            // above or below the origin and the direction in which the element will expand.
            if (position.overlayY === 'bottom') {
                // When using `bottom`, we adjust the y position such that it is the distance
                // from the bottom of the viewport rather than the top.
                /** @type {?} */
                const documentHeight = (/** @type {?} */ (this._document.documentElement)).clientHeight;
                styles.bottom = `${documentHeight - (overlayPoint.y + this._overlayRect.height)}px`;
            }
            else {
                styles.top = coerceCssPixelValue(overlayPoint.y);
            }
            return styles;
        }
        /**
         * Gets the exact left/right for the overlay when not using flexible sizing or when pushing.
         * @private
         * @param {?} position
         * @param {?} originPoint
         * @param {?} scrollPosition
         * @return {?}
         */
        _getExactOverlayX(position, originPoint, scrollPosition) {
            // Reset any existing styles. This is necessary in case the preferred position has
            // changed since the last `apply`.
            /** @type {?} */
            let styles = (/** @type {?} */ ({ left: '', right: '' }));
            /** @type {?} */
            let overlayPoint = this._getOverlayPoint(originPoint, this._overlayRect, position);
            if (this._isPushed) {
                overlayPoint = this._pushOverlayOnScreen(overlayPoint, this._overlayRect, scrollPosition);
            }
            // We want to set either `left` or `right` based on whether the overlay wants to appear "before"
            // or "after" the origin, which determines the direction in which the element will expand.
            // For the horizontal axis, the meaning of "before" and "after" change based on whether the
            // page is in RTL or LTR.
            /** @type {?} */
            let horizontalStyleProperty;
            if (this._isRtl()) {
                horizontalStyleProperty = position.overlayX === 'end' ? 'left' : 'right';
            }
            else {
                horizontalStyleProperty = position.overlayX === 'end' ? 'right' : 'left';
            }
            // When we're setting `right`, we adjust the x position such that it is the distance
            // from the right edge of the viewport rather than the left edge.
            if (horizontalStyleProperty === 'right') {
                /** @type {?} */
                const documentWidth = (/** @type {?} */ (this._document.documentElement)).clientWidth;
                styles.right = `${documentWidth - (overlayPoint.x + this._overlayRect.width)}px`;
            }
            else {
                styles.left = coerceCssPixelValue(overlayPoint.x);
            }
            return styles;
        }
        /**
         * Gets the view properties of the trigger and overlay, including whether they are clipped
         * or completely outside the view of any of the strategy's scrollables.
         * @private
         * @return {?}
         */
        _getScrollVisibility() {
            // Note: needs fresh rects since the position could've changed.
            /** @type {?} */
            const originBounds = this._getOriginRect();
            /** @type {?} */
            const overlayBounds = this._pane.getBoundingClientRect();
            // TODO(jelbourn): instead of needing all of the client rects for these scrolling containers
            // every time, we should be able to use the scrollTop of the containers if the size of those
            // containers hasn't changed.
            /** @type {?} */
            const scrollContainerBounds = this._scrollables.map((/**
             * @param {?} scrollable
             * @return {?}
             */
            scrollable => {
                return scrollable.getElementRef().nativeElement.getBoundingClientRect();
            }));
            return {
                isOriginClipped: isElementClippedByScrolling(originBounds, scrollContainerBounds),
                isOriginOutsideView: isElementScrolledOutsideView(originBounds, scrollContainerBounds),
                isOverlayClipped: isElementClippedByScrolling(overlayBounds, scrollContainerBounds),
                isOverlayOutsideView: isElementScrolledOutsideView(overlayBounds, scrollContainerBounds),
            };
        }
        /**
         * Subtracts the amount that an element is overflowing on an axis from its length.
         * @private
         * @param {?} length
         * @param {...?} overflows
         * @return {?}
         */
        _subtractOverflows(length, ...overflows) {
            return overflows.reduce((/**
             * @param {?} currentValue
             * @param {?} currentOverflow
             * @return {?}
             */
            (currentValue, currentOverflow) => {
                return currentValue - Math.max(currentOverflow, 0);
            }), length);
        }
        /**
         * Narrows the given viewport rect by the current _viewportMargin.
         * @private
         * @return {?}
         */
        _getNarrowedViewportRect() {
            // We recalculate the viewport rect here ourselves, rather than using the ViewportRuler,
            // because we want to use the `clientWidth` and `clientHeight` as the base. The difference
            // being that the client properties don't include the scrollbar, as opposed to `innerWidth`
            // and `innerHeight` that do. This is necessary, because the overlay container uses
            // 100% `width` and `height` which don't include the scrollbar either.
            /** @type {?} */
            const width = (/** @type {?} */ (this._document.documentElement)).clientWidth;
            /** @type {?} */
            const height = (/** @type {?} */ (this._document.documentElement)).clientHeight;
            /** @type {?} */
            const scrollPosition = this._viewportRuler.getViewportScrollPosition();
            return {
                top: scrollPosition.top + this._viewportMargin,
                left: scrollPosition.left + this._viewportMargin,
                right: scrollPosition.left + width - this._viewportMargin,
                bottom: scrollPosition.top + height - this._viewportMargin,
                width: width - (2 * this._viewportMargin),
                height: height - (2 * this._viewportMargin),
            };
        }
        /**
         * Whether the we're dealing with an RTL context
         * @private
         * @return {?}
         */
        _isRtl() {
            return this._overlayRef.getDirection() === 'rtl';
        }
        /**
         * Determines whether the overlay uses exact or flexible positioning.
         * @private
         * @return {?}
         */
        _hasExactPosition() {
            return !this._hasFlexibleDimensions || this._isPushed;
        }
        /**
         * Retrieves the offset of a position along the x or y axis.
         * @private
         * @param {?} position
         * @param {?} axis
         * @return {?}
         */
        _getOffset(position, axis) {
            if (axis === 'x') {
                // We don't do something like `position['offset' + axis]` in
                // order to avoid breking minifiers that rename properties.
                return position.offsetX == null ? this._offsetX : position.offsetX;
            }
            return position.offsetY == null ? this._offsetY : position.offsetY;
        }
        /**
         * Validates that the current position match the expected values.
         * @private
         * @return {?}
         */
        _validatePositions() {
            if (!this._preferredPositions.length) {
                throw Error('FlexibleConnectedPositionStrategy: At least one position is required.');
            }
            // TODO(crisbeto): remove these once Angular's template type
            // checking is advanced enough to catch these cases.
            this._preferredPositions.forEach((/**
             * @param {?} pair
             * @return {?}
             */
            pair => {
                validateHorizontalPosition('originX', pair.originX);
                validateVerticalPosition('originY', pair.originY);
                validateHorizontalPosition('overlayX', pair.overlayX);
                validateVerticalPosition('overlayY', pair.overlayY);
            }));
        }
        /**
         * Adds a single CSS class or an array of classes on the overlay panel.
         * @private
         * @param {?} cssClasses
         * @return {?}
         */
        _addPanelClasses(cssClasses) {
            if (this._pane) {
                coerceArray(cssClasses).forEach((/**
                 * @param {?} cssClass
                 * @return {?}
                 */
                cssClass => {
                    if (cssClass !== '' && this._appliedPanelClasses.indexOf(cssClass) === -1) {
                        this._appliedPanelClasses.push(cssClass);
                        this._pane.classList.add(cssClass);
                    }
                }));
            }
        }
        /**
         * Clears the classes that the position strategy has applied from the overlay panel.
         * @private
         * @return {?}
         */
        _clearPanelClasses() {
            if (this._pane) {
                this._appliedPanelClasses.forEach((/**
                 * @param {?} cssClass
                 * @return {?}
                 */
                cssClass => {
                    this._pane.classList.remove(cssClass);
                }));
                this._appliedPanelClasses = [];
            }
        }
        /**
         * Returns the ClientRect of the current origin.
         * @private
         * @return {?}
         */
        _getOriginRect() {
            /** @type {?} */
            const origin = this._origin;
            if (origin instanceof ElementRef) {
                return origin.nativeElement.getBoundingClientRect();
            }
            // Check for Element so SVG elements are also supported.
            if (origin instanceof Element) {
                return origin.getBoundingClientRect();
            }
            /** @type {?} */
            const width = origin.width || 0;
            /** @type {?} */
            const height = origin.height || 0;
            // If the origin is a point, return a client rect as if it was a 0x0 element at the point.
            return {
                top: origin.y,
                bottom: origin.y + height,
                left: origin.x,
                right: origin.x + width,
                height,
                width
            };
        }
    }
    /**
     * Shallow-extends a stylesheet object with another stylesheet object.
     * @param {?} destination
     * @param {?} source
     * @return {?}
     */
    function extendStyles(destination, source) {
        for (let key in source) {
            if (source.hasOwnProperty(key)) {
                destination[key] = source[key];
            }
        }
        return destination;
    }
    /**
     * Extracts the pixel value as a number from a value, if it's a number
     * or a CSS pixel string (e.g. `1337px`). Otherwise returns null.
     * @param {?} input
     * @return {?}
     */
    function getPixelValue(input) {
        if (typeof input !== 'number' && input != null) {
            const [value, units] = input.split(cssUnitPattern);
            return (!units || units === 'px') ? parseFloat(value) : null;
        }
        return input || null;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/overlay/position/connected-position-strategy.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * A strategy for positioning overlays. Using this strategy, an overlay is given an
     * implicit position relative to some origin element. The relative position is defined in terms of
     * a point on the origin element that is connected to a point on the overlay element. For example,
     * a basic dropdown is connecting the bottom-left corner of the origin to the top-left corner
     * of the overlay.
     * @deprecated Use `FlexibleConnectedPositionStrategy` instead.
     * \@breaking-change 8.0.0
     */
    class ConnectedPositionStrategy {
        /**
         * @param {?} originPos
         * @param {?} overlayPos
         * @param {?} connectedTo
         * @param {?} viewportRuler
         * @param {?} document
         * @param {?} platform
         * @param {?} overlayContainer
         */
        constructor(originPos, overlayPos, connectedTo, viewportRuler, document, platform, overlayContainer) {
            /**
             * Ordered list of preferred positions, from most to least desirable.
             */
            this._preferredPositions = [];
            // Since the `ConnectedPositionStrategy` is deprecated and we don't want to maintain
            // the extra logic, we create an instance of the positioning strategy that has some
            // defaults that make it behave as the old position strategy and to which we'll
            // proxy all of the API calls.
            this._positionStrategy = new FlexibleConnectedPositionStrategy(connectedTo, viewportRuler, document, platform, overlayContainer)
                .withFlexibleDimensions(false)
                .withPush(false)
                .withViewportMargin(0);
            this.withFallbackPosition(originPos, overlayPos);
        }
        /**
         * Whether the we're dealing with an RTL context
         * @return {?}
         */
        get _isRtl() {
            return this._overlayRef.getDirection() === 'rtl';
        }
        /**
         * Emits an event when the connection point changes.
         * @return {?}
         */
        get onPositionChange() {
            return this._positionStrategy.positionChanges;
        }
        /**
         * Ordered list of preferred positions, from most to least desirable.
         * @return {?}
         */
        get positions() {
            return this._preferredPositions;
        }
        /**
         * Attach this position strategy to an overlay.
         * @param {?} overlayRef
         * @return {?}
         */
        attach(overlayRef) {
            this._overlayRef = overlayRef;
            this._positionStrategy.attach(overlayRef);
            if (this._direction) {
                overlayRef.setDirection(this._direction);
                this._direction = null;
            }
        }
        /**
         * Disposes all resources used by the position strategy.
         * @return {?}
         */
        dispose() {
            this._positionStrategy.dispose();
        }
        /**
         * \@docs-private
         * @return {?}
         */
        detach() {
            this._positionStrategy.detach();
        }
        /**
         * Updates the position of the overlay element, using whichever preferred position relative
         * to the origin fits on-screen.
         * \@docs-private
         * @return {?}
         */
        apply() {
            this._positionStrategy.apply();
        }
        /**
         * Re-positions the overlay element with the trigger in its last calculated position,
         * even if a position higher in the "preferred positions" list would now fit. This
         * allows one to re-align the panel without changing the orientation of the panel.
         * @return {?}
         */
        recalculateLastPosition() {
            this._positionStrategy.reapplyLastPosition();
        }
        /**
         * Sets the list of Scrollable containers that host the origin element so that
         * on reposition we can evaluate if it or the overlay has been clipped or outside view. Every
         * Scrollable must be an ancestor element of the strategy's origin element.
         * @param {?} scrollables
         * @return {?}
         */
        withScrollableContainers(scrollables) {
            this._positionStrategy.withScrollableContainers(scrollables);
        }
        /**
         * Adds a new preferred fallback position.
         * @template THIS
         * @this {THIS}
         * @param {?} originPos
         * @param {?} overlayPos
         * @param {?=} offsetX
         * @param {?=} offsetY
         * @return {THIS}
         */
        withFallbackPosition(originPos, overlayPos, offsetX, offsetY) {
            /** @type {?} */
            const position = new ConnectionPositionPair(originPos, overlayPos, offsetX, offsetY);
            (/** @type {?} */ (this))._preferredPositions.push(position);
            (/** @type {?} */ (this))._positionStrategy.withPositions((/** @type {?} */ (this))._preferredPositions);
            return (/** @type {?} */ (this));
        }
        /**
         * Sets the layout direction so the overlay's position can be adjusted to match.
         * @template THIS
         * @this {THIS}
         * @param {?} dir New layout direction.
         * @return {THIS}
         */
        withDirection(dir) {
            // Since the direction might be declared before the strategy is attached,
            // we save the value in a temporary property and we'll transfer it to the
            // overlay ref on attachment.
            if ((/** @type {?} */ (this))._overlayRef) {
                (/** @type {?} */ (this))._overlayRef.setDirection(dir);
            }
            else {
                (/** @type {?} */ (this))._direction = dir;
            }
            return (/** @type {?} */ (this));
        }
        /**
         * Sets an offset for the overlay's connection point on the x-axis
         * @template THIS
         * @this {THIS}
         * @param {?} offset New offset in the X axis.
         * @return {THIS}
         */
        withOffsetX(offset) {
            (/** @type {?} */ (this))._positionStrategy.withDefaultOffsetX(offset);
            return (/** @type {?} */ (this));
        }
        /**
         * Sets an offset for the overlay's connection point on the y-axis
         * @template THIS
         * @this {THIS}
         * @param {?} offset New offset in the Y axis.
         * @return {THIS}
         */
        withOffsetY(offset) {
            (/** @type {?} */ (this))._positionStrategy.withDefaultOffsetY(offset);
            return (/** @type {?} */ (this));
        }
        /**
         * Sets whether the overlay's position should be locked in after it is positioned
         * initially. When an overlay is locked in, it won't attempt to reposition itself
         * when the position is re-applied (e.g. when the user scrolls away).
         * @template THIS
         * @this {THIS}
         * @param {?} isLocked Whether the overlay should locked in.
         * @return {THIS}
         */
        withLockedPosition(isLocked) {
            (/** @type {?} */ (this))._positionStrategy.withLockedPosition(isLocked);
            return (/** @type {?} */ (this));
        }
        /**
         * Overwrites the current set of positions with an array of new ones.
         * @template THIS
         * @this {THIS}
         * @param {?} positions Position pairs to be set on the strategy.
         * @return {THIS}
         */
        withPositions(positions) {
            (/** @type {?} */ (this))._preferredPositions = positions.slice();
            (/** @type {?} */ (this))._positionStrategy.withPositions((/** @type {?} */ (this))._preferredPositions);
            return (/** @type {?} */ (this));
        }
        /**
         * Sets the origin element, relative to which to position the overlay.
         * @template THIS
         * @this {THIS}
         * @param {?} origin Reference to the new origin element.
         * @return {THIS}
         */
        setOrigin(origin) {
            (/** @type {?} */ (this))._positionStrategy.setOrigin(origin);
            return (/** @type {?} */ (this));
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/overlay/position/global-position-strategy.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google LLC All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * Class to be added to the overlay pane wrapper.
     * @type {?}
     */
    const wrapperClass = 'cdk-global-overlay-wrapper';
    /**
     * A strategy for positioning overlays. Using this strategy, an overlay is given an
     * explicit position relative to the browser's viewport. We use flexbox, instead of
     * transforms, in order to avoid issues with subpixel rendering which can cause the
     * element to become blurry.
     */
    class GlobalPositionStrategy {
        constructor() {
            this._cssPosition = 'static';
            this._topOffset = '';
            this._bottomOffset = '';
            this._leftOffset = '';
            this._rightOffset = '';
            this._alignItems = '';
            this._justifyContent = '';
            this._width = '';
            this._height = '';
        }
        /**
         * @param {?} overlayRef
         * @return {?}
         */
        attach(overlayRef) {
            /** @type {?} */
            const config = overlayRef.getConfig();
            this._overlayRef = overlayRef;
            if (this._width && !config.width) {
                overlayRef.updateSize({ width: this._width });
            }
            if (this._height && !config.height) {
                overlayRef.updateSize({ height: this._height });
            }
            overlayRef.hostElement.classList.add(wrapperClass);
            this._isDisposed = false;
        }
        /**
         * Sets the top position of the overlay. Clears any previously set vertical position.
         * @template THIS
         * @this {THIS}
         * @param {?=} value New top offset.
         * @return {THIS}
         */
        top(value = '') {
            (/** @type {?} */ (this))._bottomOffset = '';
            (/** @type {?} */ (this))._topOffset = value;
            (/** @type {?} */ (this))._alignItems = 'flex-start';
            return (/** @type {?} */ (this));
        }
        /**
         * Sets the left position of the overlay. Clears any previously set horizontal position.
         * @template THIS
         * @this {THIS}
         * @param {?=} value New left offset.
         * @return {THIS}
         */
        left(value = '') {
            (/** @type {?} */ (this))._rightOffset = '';
            (/** @type {?} */ (this))._leftOffset = value;
            (/** @type {?} */ (this))._justifyContent = 'flex-start';
            return (/** @type {?} */ (this));
        }
        /**
         * Sets the bottom position of the overlay. Clears any previously set vertical position.
         * @template THIS
         * @this {THIS}
         * @param {?=} value New bottom offset.
         * @return {THIS}
         */
        bottom(value = '') {
            (/** @type {?} */ (this))._topOffset = '';
            (/** @type {?} */ (this))._bottomOffset = value;
            (/** @type {?} */ (this))._alignItems = 'flex-end';
            return (/** @type {?} */ (this));
        }
        /**
         * Sets the right position of the overlay. Clears any previously set horizontal position.
         * @template THIS
         * @this {THIS}
         * @param {?=} value New right offset.
         * @return {THIS}
         */
        right(value = '') {
            (/** @type {?} */ (this))._leftOffset = '';
            (/** @type {?} */ (this))._rightOffset = value;
            (/** @type {?} */ (this))._justifyContent = 'flex-end';
            return (/** @type {?} */ (this));
        }
        /**
         * Sets the overlay width and clears any previously set width.
         * @deprecated Pass the `width` through the `OverlayConfig`.
         * \@breaking-change 8.0.0
         * @template THIS
         * @this {THIS}
         * @param {?=} value New width for the overlay
         * @return {THIS}
         */
        width(value = '') {
            if ((/** @type {?} */ (this))._overlayRef) {
                (/** @type {?} */ (this))._overlayRef.updateSize({ width: value });
            }
            else {
                (/** @type {?} */ (this))._width = value;
            }
            return (/** @type {?} */ (this));
        }
        /**
         * Sets the overlay height and clears any previously set height.
         * @deprecated Pass the `height` through the `OverlayConfig`.
         * \@breaking-change 8.0.0
         * @template THIS
         * @this {THIS}
         * @param {?=} value New height for the overlay
         * @return {THIS}
         */
        height(value = '') {
            if ((/** @type {?} */ (this))._overlayRef) {
                (/** @type {?} */ (this))._overlayRef.updateSize({ height: value });
            }
            else {
                (/** @type {?} */ (this))._height = value;
            }
            return (/** @type {?} */ (this));
        }
        /**
         * Centers the overlay horizontally with an optional offset.
         * Clears any previously set horizontal position.
         *
         * @template THIS
         * @this {THIS}
         * @param {?=} offset Overlay offset from the horizontal center.
         * @return {THIS}
         */
        centerHorizontally(offset = '') {
            (/** @type {?} */ (this)).left(offset);
            (/** @type {?} */ (this))._justifyContent = 'center';
            return (/** @type {?} */ (this));
        }
        /**
         * Centers the overlay vertically with an optional offset.
         * Clears any previously set vertical position.
         *
         * @template THIS
         * @this {THIS}
         * @param {?=} offset Overlay offset from the vertical center.
         * @return {THIS}
         */
        centerVertically(offset = '') {
            (/** @type {?} */ (this)).top(offset);
            (/** @type {?} */ (this))._alignItems = 'center';
            return (/** @type {?} */ (this));
        }
        /**
         * Apply the position to the element.
         * \@docs-private
         * @return {?}
         */
        apply() {
            // Since the overlay ref applies the strategy asynchronously, it could
            // have been disposed before it ends up being applied. If that is the
            // case, we shouldn't do anything.
            if (!this._overlayRef || !this._overlayRef.hasAttached()) {
                return;
            }
            /** @type {?} */
            const styles = this._overlayRef.overlayElement.style;
            /** @type {?} */
            const parentStyles = this._overlayRef.hostElement.style;
            /** @type {?} */
            const config = this._overlayRef.getConfig();
            const { width, height, maxWidth, maxHeight } = config;
            /** @type {?} */
            const shouldBeFlushHorizontally = (width === '100%' || width === '100vw') &&
                (!maxWidth || maxWidth === '100%' || maxWidth === '100vw');
            /** @type {?} */
            const shouldBeFlushVertically = (height === '100%' || height === '100vh') &&
                (!maxHeight || maxHeight === '100%' || maxHeight === '100vh');
            styles.position = this._cssPosition;
            styles.marginLeft = shouldBeFlushHorizontally ? '0' : this._leftOffset;
            styles.marginTop = shouldBeFlushVertically ? '0' : this._topOffset;
            styles.marginBottom = this._bottomOffset;
            styles.marginRight = this._rightOffset;
            if (shouldBeFlushHorizontally) {
                parentStyles.justifyContent = 'flex-start';
            }
            else if (this._justifyContent === 'center') {
                parentStyles.justifyContent = 'center';
            }
            else if (this._overlayRef.getConfig().direction === 'rtl') {
                // In RTL the browser will invert `flex-start` and `flex-end` automatically, but we
                // don't want that because our positioning is explicitly `left` and `right`, hence
                // why we do another inversion to ensure that the overlay stays in the same position.
                // TODO: reconsider this if we add `start` and `end` methods.
                if (this._justifyContent === 'flex-start') {
                    parentStyles.justifyContent = 'flex-end';
                }
                else if (this._justifyContent === 'flex-end') {
                    parentStyles.justifyContent = 'flex-start';
                }
            }
            else {
                parentStyles.justifyContent = this._justifyContent;
            }
            parentStyles.alignItems = shouldBeFlushVertically ? 'flex-start' : this._alignItems;
        }
        /**
         * Cleans up the DOM changes from the position strategy.
         * \@docs-private
         * @return {?}
         */
        dispose() {
            if (this._isDisposed || !this._overlayRef) {
                return;
            }
            /** @type {?} */
            const styles = this._overlayRef.overlayElement.style;
            /** @type {?} */
            const parent = this._overlayRef.hostElement;
            /** @type {?} */
            const parentStyles = parent.style;
            parent.classList.remove(wrapperClass);
            parentStyles.justifyContent = parentStyles.alignItems = styles.marginTop =
                styles.marginBottom = styles.marginLeft = styles.marginRight = styles.position = '';
            this._overlayRef = (/** @type {?} */ (null));
            this._isDisposed = true;
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/overlay/position/overlay-position-builder.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Builder for overlay position strategy.
     */
    class OverlayPositionBuilder {
        /**
         * @param {?} _viewportRuler
         * @param {?} _document
         * @param {?} _platform
         * @param {?} _overlayContainer
         */
        constructor(_viewportRuler, _document, _platform, _overlayContainer) {
            this._viewportRuler = _viewportRuler;
            this._document = _document;
            this._platform = _platform;
            this._overlayContainer = _overlayContainer;
        }
        /**
         * Creates a global position strategy.
         * @return {?}
         */
        global() {
            return new GlobalPositionStrategy();
        }
        /**
         * Creates a relative position strategy.
         * @deprecated Use `flexibleConnectedTo` instead.
         * \@breaking-change 8.0.0
         * @param {?} elementRef
         * @param {?} originPos
         * @param {?} overlayPos
         * @return {?}
         */
        connectedTo(elementRef, originPos, overlayPos) {
            return new ConnectedPositionStrategy(originPos, overlayPos, elementRef, this._viewportRuler, this._document, this._platform, this._overlayContainer);
        }
        /**
         * Creates a flexible position strategy.
         * @param {?} origin Origin relative to which to position the overlay.
         * @return {?}
         */
        flexibleConnectedTo(origin) {
            return new FlexibleConnectedPositionStrategy(origin, this._viewportRuler, this._document, this._platform, this._overlayContainer);
        }
    }
    OverlayPositionBuilder.ɵfac = function OverlayPositionBuilder_Factory(t) { return new (t || OverlayPositionBuilder)(ɵɵinject(ViewportRuler), ɵɵinject(DOCUMENT$1), ɵɵinject(Platform), ɵɵinject(OverlayContainer)); };
    /** @nocollapse */
    OverlayPositionBuilder.ctorParameters = () => [
        { type: ViewportRuler },
        { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT$1,] }] },
        { type: Platform },
        { type: OverlayContainer }
    ];
    /** @nocollapse */ OverlayPositionBuilder.ɵprov = ɵɵdefineInjectable({ factory: function OverlayPositionBuilder_Factory() { return new OverlayPositionBuilder(ɵɵinject(ViewportRuler), ɵɵinject(DOCUMENT$1), ɵɵinject(Platform), ɵɵinject(OverlayContainer)); }, token: OverlayPositionBuilder, providedIn: "root" });

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/overlay/overlay.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Next overlay unique ID.
     * @type {?}
     */
    let nextUniqueId$3 = 0;
    // Note that Overlay is *not* scoped to the app root because of the ComponentFactoryResolver
    // which needs to be different depending on where OverlayModule is imported.
    /**
     * Service to create Overlays. Overlays are dynamically added pieces of floating UI, meant to be
     * used as a low-level building block for other components. Dialogs, tooltips, menus,
     * selects, etc. can all be built using overlays. The service should primarily be used by authors
     * of re-usable components rather than developers building end-user applications.
     *
     * An overlay *is* a PortalOutlet, so any kind of Portal can be loaded into one.
     */
    class Overlay {
        /**
         * @param {?} scrollStrategies
         * @param {?} _overlayContainer
         * @param {?} _componentFactoryResolver
         * @param {?} _positionBuilder
         * @param {?} _keyboardDispatcher
         * @param {?} _injector
         * @param {?} _ngZone
         * @param {?} _document
         * @param {?} _directionality
         * @param {?=} _location
         */
        constructor(scrollStrategies, _overlayContainer, _componentFactoryResolver, _positionBuilder, _keyboardDispatcher, _injector, _ngZone, _document, _directionality, _location) {
            this.scrollStrategies = scrollStrategies;
            this._overlayContainer = _overlayContainer;
            this._componentFactoryResolver = _componentFactoryResolver;
            this._positionBuilder = _positionBuilder;
            this._keyboardDispatcher = _keyboardDispatcher;
            this._injector = _injector;
            this._ngZone = _ngZone;
            this._document = _document;
            this._directionality = _directionality;
            this._location = _location;
        }
        /**
         * Creates an overlay.
         * @param {?=} config Configuration applied to the overlay.
         * @return {?} Reference to the created overlay.
         */
        create(config) {
            /** @type {?} */
            const host = this._createHostElement();
            /** @type {?} */
            const pane = this._createPaneElement(host);
            /** @type {?} */
            const portalOutlet = this._createPortalOutlet(pane);
            /** @type {?} */
            const overlayConfig = new OverlayConfig(config);
            overlayConfig.direction = overlayConfig.direction || this._directionality.value;
            return new OverlayRef(portalOutlet, host, pane, overlayConfig, this._ngZone, this._keyboardDispatcher, this._document, this._location);
        }
        /**
         * Gets a position builder that can be used, via fluent API,
         * to construct and configure a position strategy.
         * @return {?} An overlay position builder.
         */
        position() {
            return this._positionBuilder;
        }
        /**
         * Creates the DOM element for an overlay and appends it to the overlay container.
         * @private
         * @param {?} host
         * @return {?} Newly-created pane element
         */
        _createPaneElement(host) {
            /** @type {?} */
            const pane = this._document.createElement('div');
            pane.id = `cdk-overlay-${nextUniqueId$3++}`;
            pane.classList.add('cdk-overlay-pane');
            host.appendChild(pane);
            return pane;
        }
        /**
         * Creates the host element that wraps around an overlay
         * and can be used for advanced positioning.
         * @private
         * @return {?} Newly-create host element.
         */
        _createHostElement() {
            /** @type {?} */
            const host = this._document.createElement('div');
            this._overlayContainer.getContainerElement().appendChild(host);
            return host;
        }
        /**
         * Create a DomPortalOutlet into which the overlay content can be loaded.
         * @private
         * @param {?} pane The DOM element to turn into a portal outlet.
         * @return {?} A portal outlet for the given DOM element.
         */
        _createPortalOutlet(pane) {
            // We have to resolve the ApplicationRef later in order to allow people
            // to use overlay-based providers during app initialization.
            if (!this._appRef) {
                this._appRef = this._injector.get(ApplicationRef);
            }
            return new DomPortalOutlet(pane, this._componentFactoryResolver, this._appRef, this._injector, this._document);
        }
    }
    Overlay.ɵfac = function Overlay_Factory(t) { return new (t || Overlay)(ɵɵinject(ScrollStrategyOptions), ɵɵinject(OverlayContainer), ɵɵinject(ComponentFactoryResolver), ɵɵinject(OverlayPositionBuilder), ɵɵinject(OverlayKeyboardDispatcher), ɵɵinject(Injector), ɵɵinject(NgZone), ɵɵinject(DOCUMENT$1), ɵɵinject(Directionality), ɵɵinject(Location, 8)); };
    Overlay.ɵprov = ɵɵdefineInjectable({ token: Overlay, factory: Overlay.ɵfac });
    /** @nocollapse */
    Overlay.ctorParameters = () => [
        { type: ScrollStrategyOptions },
        { type: OverlayContainer },
        { type: ComponentFactoryResolver },
        { type: OverlayPositionBuilder },
        { type: OverlayKeyboardDispatcher },
        { type: Injector },
        { type: NgZone },
        { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT$1,] }] },
        { type: Directionality },
        { type: Location, decorators: [{ type: Optional }] }
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/overlay/overlay-directives.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Default set of positions for the overlay. Follows the behavior of a dropdown.
     * @type {?}
     */
    const defaultPositionList = [
        {
            originX: 'start',
            originY: 'bottom',
            overlayX: 'start',
            overlayY: 'top'
        },
        {
            originX: 'start',
            originY: 'top',
            overlayX: 'start',
            overlayY: 'bottom'
        },
        {
            originX: 'end',
            originY: 'top',
            overlayX: 'end',
            overlayY: 'bottom'
        },
        {
            originX: 'end',
            originY: 'bottom',
            overlayX: 'end',
            overlayY: 'top'
        }
    ];
    /**
     * Injection token that determines the scroll handling while the connected overlay is open.
     * @type {?}
     */
    const CDK_CONNECTED_OVERLAY_SCROLL_STRATEGY = new InjectionToken('cdk-connected-overlay-scroll-strategy');
    /**
     * Directive applied to an element to make it usable as an origin for an Overlay using a
     * ConnectedPositionStrategy.
     */
    class CdkOverlayOrigin {
        /**
         * @param {?} elementRef
         */
        constructor(elementRef) {
            this.elementRef = elementRef;
        }
    }
    CdkOverlayOrigin.ɵfac = function CdkOverlayOrigin_Factory(t) { return new (t || CdkOverlayOrigin)(ɵɵdirectiveInject(ElementRef)); };
    CdkOverlayOrigin.ɵdir = ɵɵdefineDirective({ type: CdkOverlayOrigin, selectors: [["", "cdk-overlay-origin", ""], ["", "overlay-origin", ""], ["", "cdkOverlayOrigin", ""]], exportAs: ["cdkOverlayOrigin"] });
    /** @nocollapse */
    CdkOverlayOrigin.ctorParameters = () => [
        { type: ElementRef }
    ];
    /**
     * Directive to facilitate declarative creation of an
     * Overlay using a FlexibleConnectedPositionStrategy.
     */
    class CdkConnectedOverlay {
        // TODO(jelbourn): inputs for size, scroll behavior, animation, etc.
        /**
         * @param {?} _overlay
         * @param {?} templateRef
         * @param {?} viewContainerRef
         * @param {?} scrollStrategyFactory
         * @param {?} _dir
         */
        constructor(_overlay, templateRef, viewContainerRef, scrollStrategyFactory, _dir) {
            this._overlay = _overlay;
            this._dir = _dir;
            this._hasBackdrop = false;
            this._lockPosition = false;
            this._growAfterOpen = false;
            this._flexibleDimensions = false;
            this._push = false;
            this._backdropSubscription = Subscription.EMPTY;
            /**
             * Margin between the overlay and the viewport edges.
             */
            this.viewportMargin = 0;
            /**
             * Whether the overlay is open.
             */
            this.open = false;
            /**
             * Event emitted when the backdrop is clicked.
             */
            this.backdropClick = new EventEmitter();
            /**
             * Event emitted when the position has changed.
             */
            this.positionChange = new EventEmitter();
            /**
             * Event emitted when the overlay has been attached.
             */
            this.attach = new EventEmitter();
            /**
             * Event emitted when the overlay has been detached.
             */
            this.detach = new EventEmitter();
            /**
             * Emits when there are keyboard events that are targeted at the overlay.
             */
            this.overlayKeydown = new EventEmitter();
            this._templatePortal = new TemplatePortal(templateRef, viewContainerRef);
            this._scrollStrategyFactory = scrollStrategyFactory;
            this.scrollStrategy = this._scrollStrategyFactory();
        }
        /**
         * The offset in pixels for the overlay connection point on the x-axis
         * @return {?}
         */
        get offsetX() { return this._offsetX; }
        /**
         * @param {?} offsetX
         * @return {?}
         */
        set offsetX(offsetX) {
            this._offsetX = offsetX;
            if (this._position) {
                this._updatePositionStrategy(this._position);
            }
        }
        /**
         * The offset in pixels for the overlay connection point on the y-axis
         * @return {?}
         */
        get offsetY() { return this._offsetY; }
        /**
         * @param {?} offsetY
         * @return {?}
         */
        set offsetY(offsetY) {
            this._offsetY = offsetY;
            if (this._position) {
                this._updatePositionStrategy(this._position);
            }
        }
        /**
         * Whether or not the overlay should attach a backdrop.
         * @return {?}
         */
        get hasBackdrop() { return this._hasBackdrop; }
        /**
         * @param {?} value
         * @return {?}
         */
        set hasBackdrop(value) { this._hasBackdrop = coerceBooleanProperty(value); }
        /**
         * Whether or not the overlay should be locked when scrolling.
         * @return {?}
         */
        get lockPosition() { return this._lockPosition; }
        /**
         * @param {?} value
         * @return {?}
         */
        set lockPosition(value) { this._lockPosition = coerceBooleanProperty(value); }
        /**
         * Whether the overlay's width and height can be constrained to fit within the viewport.
         * @return {?}
         */
        get flexibleDimensions() { return this._flexibleDimensions; }
        /**
         * @param {?} value
         * @return {?}
         */
        set flexibleDimensions(value) {
            this._flexibleDimensions = coerceBooleanProperty(value);
        }
        /**
         * Whether the overlay can grow after the initial open when flexible positioning is turned on.
         * @return {?}
         */
        get growAfterOpen() { return this._growAfterOpen; }
        /**
         * @param {?} value
         * @return {?}
         */
        set growAfterOpen(value) { this._growAfterOpen = coerceBooleanProperty(value); }
        /**
         * Whether the overlay can be pushed on-screen if none of the provided positions fit.
         * @return {?}
         */
        get push() { return this._push; }
        /**
         * @param {?} value
         * @return {?}
         */
        set push(value) { this._push = coerceBooleanProperty(value); }
        /**
         * The associated overlay reference.
         * @return {?}
         */
        get overlayRef() {
            return this._overlayRef;
        }
        /**
         * The element's layout direction.
         * @return {?}
         */
        get dir() {
            return this._dir ? this._dir.value : 'ltr';
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            if (this._overlayRef) {
                this._overlayRef.dispose();
            }
            this._backdropSubscription.unsubscribe();
        }
        /**
         * @param {?} changes
         * @return {?}
         */
        ngOnChanges(changes) {
            if (this._position) {
                this._updatePositionStrategy(this._position);
                this._overlayRef.updateSize({
                    width: this.width,
                    minWidth: this.minWidth,
                    height: this.height,
                    minHeight: this.minHeight,
                });
                if (changes['origin'] && this.open) {
                    this._position.apply();
                }
            }
            if (changes['open']) {
                this.open ? this._attachOverlay() : this._detachOverlay();
            }
        }
        /**
         * Creates an overlay
         * @private
         * @return {?}
         */
        _createOverlay() {
            if (!this.positions || !this.positions.length) {
                this.positions = defaultPositionList;
            }
            this._overlayRef = this._overlay.create(this._buildConfig());
            this._overlayRef.keydownEvents().subscribe((/**
             * @param {?} event
             * @return {?}
             */
            (event) => {
                this.overlayKeydown.next(event);
                if (event.keyCode === ESCAPE && !hasModifierKey(event)) {
                    event.preventDefault();
                    this._detachOverlay();
                }
            }));
        }
        /**
         * Builds the overlay config based on the directive's inputs
         * @private
         * @return {?}
         */
        _buildConfig() {
            /** @type {?} */
            const positionStrategy = this._position =
                this.positionStrategy || this._createPositionStrategy();
            /** @type {?} */
            const overlayConfig = new OverlayConfig({
                direction: this._dir,
                positionStrategy,
                scrollStrategy: this.scrollStrategy,
                hasBackdrop: this.hasBackdrop
            });
            if (this.width || this.width === 0) {
                overlayConfig.width = this.width;
            }
            if (this.height || this.height === 0) {
                overlayConfig.height = this.height;
            }
            if (this.minWidth || this.minWidth === 0) {
                overlayConfig.minWidth = this.minWidth;
            }
            if (this.minHeight || this.minHeight === 0) {
                overlayConfig.minHeight = this.minHeight;
            }
            if (this.backdropClass) {
                overlayConfig.backdropClass = this.backdropClass;
            }
            if (this.panelClass) {
                overlayConfig.panelClass = this.panelClass;
            }
            return overlayConfig;
        }
        /**
         * Updates the state of a position strategy, based on the values of the directive inputs.
         * @private
         * @param {?} positionStrategy
         * @return {?}
         */
        _updatePositionStrategy(positionStrategy) {
            /** @type {?} */
            const positions = this.positions.map((/**
             * @param {?} currentPosition
             * @return {?}
             */
            currentPosition => ({
                originX: currentPosition.originX,
                originY: currentPosition.originY,
                overlayX: currentPosition.overlayX,
                overlayY: currentPosition.overlayY,
                offsetX: currentPosition.offsetX || this.offsetX,
                offsetY: currentPosition.offsetY || this.offsetY,
                panelClass: currentPosition.panelClass || undefined,
            })));
            return positionStrategy
                .setOrigin(this.origin.elementRef)
                .withPositions(positions)
                .withFlexibleDimensions(this.flexibleDimensions)
                .withPush(this.push)
                .withGrowAfterOpen(this.growAfterOpen)
                .withViewportMargin(this.viewportMargin)
                .withLockedPosition(this.lockPosition)
                .withTransformOriginOn(this.transformOriginSelector);
        }
        /**
         * Returns the position strategy of the overlay to be set on the overlay config
         * @private
         * @return {?}
         */
        _createPositionStrategy() {
            /** @type {?} */
            const strategy = this._overlay.position().flexibleConnectedTo(this.origin.elementRef);
            this._updatePositionStrategy(strategy);
            strategy.positionChanges.subscribe((/**
             * @param {?} p
             * @return {?}
             */
            p => this.positionChange.emit(p)));
            return strategy;
        }
        /**
         * Attaches the overlay and subscribes to backdrop clicks if backdrop exists
         * @private
         * @return {?}
         */
        _attachOverlay() {
            if (!this._overlayRef) {
                this._createOverlay();
            }
            else {
                // Update the overlay size, in case the directive's inputs have changed
                this._overlayRef.getConfig().hasBackdrop = this.hasBackdrop;
            }
            if (!this._overlayRef.hasAttached()) {
                this._overlayRef.attach(this._templatePortal);
                this.attach.emit();
            }
            if (this.hasBackdrop) {
                this._backdropSubscription = this._overlayRef.backdropClick().subscribe((/**
                 * @param {?} event
                 * @return {?}
                 */
                event => {
                    this.backdropClick.emit(event);
                }));
            }
            else {
                this._backdropSubscription.unsubscribe();
            }
        }
        /**
         * Detaches the overlay and unsubscribes to backdrop clicks if backdrop exists
         * @private
         * @return {?}
         */
        _detachOverlay() {
            if (this._overlayRef) {
                this._overlayRef.detach();
                this.detach.emit();
            }
            this._backdropSubscription.unsubscribe();
        }
    }
    CdkConnectedOverlay.ɵfac = function CdkConnectedOverlay_Factory(t) { return new (t || CdkConnectedOverlay)(ɵɵdirectiveInject(Overlay), ɵɵdirectiveInject(TemplateRef), ɵɵdirectiveInject(ViewContainerRef), ɵɵdirectiveInject(CDK_CONNECTED_OVERLAY_SCROLL_STRATEGY), ɵɵdirectiveInject(Directionality, 8)); };
    CdkConnectedOverlay.ɵdir = ɵɵdefineDirective({ type: CdkConnectedOverlay, selectors: [["", "cdk-connected-overlay", ""], ["", "connected-overlay", ""], ["", "cdkConnectedOverlay", ""]], inputs: { viewportMargin: ["cdkConnectedOverlayViewportMargin", "viewportMargin"], open: ["cdkConnectedOverlayOpen", "open"], scrollStrategy: ["cdkConnectedOverlayScrollStrategy", "scrollStrategy"], offsetX: ["cdkConnectedOverlayOffsetX", "offsetX"], offsetY: ["cdkConnectedOverlayOffsetY", "offsetY"], hasBackdrop: ["cdkConnectedOverlayHasBackdrop", "hasBackdrop"], lockPosition: ["cdkConnectedOverlayLockPosition", "lockPosition"], flexibleDimensions: ["cdkConnectedOverlayFlexibleDimensions", "flexibleDimensions"], growAfterOpen: ["cdkConnectedOverlayGrowAfterOpen", "growAfterOpen"], push: ["cdkConnectedOverlayPush", "push"], positions: ["cdkConnectedOverlayPositions", "positions"], origin: ["cdkConnectedOverlayOrigin", "origin"], positionStrategy: ["cdkConnectedOverlayPositionStrategy", "positionStrategy"], width: ["cdkConnectedOverlayWidth", "width"], height: ["cdkConnectedOverlayHeight", "height"], minWidth: ["cdkConnectedOverlayMinWidth", "minWidth"], minHeight: ["cdkConnectedOverlayMinHeight", "minHeight"], backdropClass: ["cdkConnectedOverlayBackdropClass", "backdropClass"], panelClass: ["cdkConnectedOverlayPanelClass", "panelClass"], transformOriginSelector: ["cdkConnectedOverlayTransformOriginOn", "transformOriginSelector"] }, outputs: { backdropClick: "backdropClick", positionChange: "positionChange", attach: "attach", detach: "detach", overlayKeydown: "overlayKeydown" }, exportAs: ["cdkConnectedOverlay"], features: [ɵɵNgOnChangesFeature] });
    /** @nocollapse */
    CdkConnectedOverlay.ctorParameters = () => [
        { type: Overlay },
        { type: TemplateRef },
        { type: ViewContainerRef },
        { type: undefined, decorators: [{ type: Inject, args: [CDK_CONNECTED_OVERLAY_SCROLL_STRATEGY,] }] },
        { type: Directionality, decorators: [{ type: Optional }] }
    ];
    CdkConnectedOverlay.propDecorators = {
        origin: [{ type: Input, args: ['cdkConnectedOverlayOrigin',] }],
        positions: [{ type: Input, args: ['cdkConnectedOverlayPositions',] }],
        positionStrategy: [{ type: Input, args: ['cdkConnectedOverlayPositionStrategy',] }],
        offsetX: [{ type: Input, args: ['cdkConnectedOverlayOffsetX',] }],
        offsetY: [{ type: Input, args: ['cdkConnectedOverlayOffsetY',] }],
        width: [{ type: Input, args: ['cdkConnectedOverlayWidth',] }],
        height: [{ type: Input, args: ['cdkConnectedOverlayHeight',] }],
        minWidth: [{ type: Input, args: ['cdkConnectedOverlayMinWidth',] }],
        minHeight: [{ type: Input, args: ['cdkConnectedOverlayMinHeight',] }],
        backdropClass: [{ type: Input, args: ['cdkConnectedOverlayBackdropClass',] }],
        panelClass: [{ type: Input, args: ['cdkConnectedOverlayPanelClass',] }],
        viewportMargin: [{ type: Input, args: ['cdkConnectedOverlayViewportMargin',] }],
        scrollStrategy: [{ type: Input, args: ['cdkConnectedOverlayScrollStrategy',] }],
        open: [{ type: Input, args: ['cdkConnectedOverlayOpen',] }],
        transformOriginSelector: [{ type: Input, args: ['cdkConnectedOverlayTransformOriginOn',] }],
        hasBackdrop: [{ type: Input, args: ['cdkConnectedOverlayHasBackdrop',] }],
        lockPosition: [{ type: Input, args: ['cdkConnectedOverlayLockPosition',] }],
        flexibleDimensions: [{ type: Input, args: ['cdkConnectedOverlayFlexibleDimensions',] }],
        growAfterOpen: [{ type: Input, args: ['cdkConnectedOverlayGrowAfterOpen',] }],
        push: [{ type: Input, args: ['cdkConnectedOverlayPush',] }],
        backdropClick: [{ type: Output }],
        positionChange: [{ type: Output }],
        attach: [{ type: Output }],
        detach: [{ type: Output }],
        overlayKeydown: [{ type: Output }]
    };
    /**
     * \@docs-private
     * @param {?} overlay
     * @return {?}
     */
    function CDK_CONNECTED_OVERLAY_SCROLL_STRATEGY_PROVIDER_FACTORY(overlay) {
        return (/**
         * @return {?}
         */
        () => overlay.scrollStrategies.reposition());
    }
    /**
     * \@docs-private
     * @type {?}
     */
    const CDK_CONNECTED_OVERLAY_SCROLL_STRATEGY_PROVIDER = {
        provide: CDK_CONNECTED_OVERLAY_SCROLL_STRATEGY,
        deps: [Overlay],
        useFactory: CDK_CONNECTED_OVERLAY_SCROLL_STRATEGY_PROVIDER_FACTORY,
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/overlay/overlay-module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class OverlayModule {
    }
    OverlayModule.ɵmod = ɵɵdefineNgModule({ type: OverlayModule });
    OverlayModule.ɵinj = ɵɵdefineInjector({ factory: function OverlayModule_Factory(t) { return new (t || OverlayModule)(); }, providers: [
            Overlay,
            CDK_CONNECTED_OVERLAY_SCROLL_STRATEGY_PROVIDER,
        ], imports: [[BidiModule, PortalModule, ScrollingModule],
            ScrollingModule] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(OverlayModule, { declarations: function () { return [CdkConnectedOverlay,
            CdkOverlayOrigin]; }, imports: function () { return [BidiModule, PortalModule, ScrollingModule]; }, exports: function () { return [CdkConnectedOverlay,
            CdkOverlayOrigin,
            ScrollingModule]; } }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/overlay/fullscreen-overlay-container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Alternative to OverlayContainer that supports correct displaying of overlay elements in
     * Fullscreen mode
     * https://developer.mozilla.org/en-US/docs/Web/API/Element/requestFullScreen
     *
     * Should be provided in the root component.
     */
    class FullscreenOverlayContainer extends OverlayContainer {
        /**
         * @param {?} _document
         * @param {?=} platform
         */
        constructor(_document, 
        /**
         * @deprecated `platform` parameter to become required.
         * @breaking-change 10.0.0
         */
        platform) {
            super(_document, platform);
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            super.ngOnDestroy();
            if (this._fullScreenEventName && this._fullScreenListener) {
                this._document.removeEventListener(this._fullScreenEventName, this._fullScreenListener);
            }
        }
        /**
         * @protected
         * @return {?}
         */
        _createContainer() {
            super._createContainer();
            this._adjustParentForFullscreenChange();
            this._addFullscreenChangeListener((/**
             * @return {?}
             */
            () => this._adjustParentForFullscreenChange()));
        }
        /**
         * @private
         * @return {?}
         */
        _adjustParentForFullscreenChange() {
            if (!this._containerElement) {
                return;
            }
            /** @type {?} */
            const fullscreenElement = this.getFullscreenElement();
            /** @type {?} */
            const parent = fullscreenElement || this._document.body;
            parent.appendChild(this._containerElement);
        }
        /**
         * @private
         * @param {?} fn
         * @return {?}
         */
        _addFullscreenChangeListener(fn) {
            /** @type {?} */
            const eventName = this._getEventName();
            if (eventName) {
                if (this._fullScreenListener) {
                    this._document.removeEventListener(eventName, this._fullScreenListener);
                }
                this._document.addEventListener(eventName, fn);
                this._fullScreenListener = fn;
            }
        }
        /**
         * @private
         * @return {?}
         */
        _getEventName() {
            if (!this._fullScreenEventName) {
                /** @type {?} */
                const _document = (/** @type {?} */ (this._document));
                if (_document.fullscreenEnabled) {
                    this._fullScreenEventName = 'fullscreenchange';
                }
                else if (_document.webkitFullscreenEnabled) {
                    this._fullScreenEventName = 'webkitfullscreenchange';
                }
                else if (_document.mozFullScreenEnabled) {
                    this._fullScreenEventName = 'mozfullscreenchange';
                }
                else if (_document.msFullscreenEnabled) {
                    this._fullScreenEventName = 'MSFullscreenChange';
                }
            }
            return this._fullScreenEventName;
        }
        /**
         * When the page is put into fullscreen mode, a specific element is specified.
         * Only that element and its children are visible when in fullscreen mode.
         * @return {?}
         */
        getFullscreenElement() {
            /** @type {?} */
            const _document = (/** @type {?} */ (this._document));
            return _document.fullscreenElement ||
                _document.webkitFullscreenElement ||
                _document.mozFullScreenElement ||
                _document.msFullscreenElement ||
                null;
        }
    }
    FullscreenOverlayContainer.ɵfac = function FullscreenOverlayContainer_Factory(t) { return new (t || FullscreenOverlayContainer)(ɵɵinject(DOCUMENT$1), ɵɵinject(Platform)); };
    /** @nocollapse */
    FullscreenOverlayContainer.ctorParameters = () => [
        { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT$1,] }] },
        { type: Platform }
    ];
    /** @nocollapse */ FullscreenOverlayContainer.ɵprov = ɵɵdefineInjectable({ factory: function FullscreenOverlayContainer_Factory() { return new FullscreenOverlayContainer(ɵɵinject(DOCUMENT$1), ɵɵinject(Platform)); }, token: FullscreenOverlayContainer, providedIn: "root" });

    /**
     * @license Angular v9.1.12
     * (c) 2010-2020 Google LLC. https://angular.io/
     * License: MIT
     */
    /**
     * Used to provide a `ControlValueAccessor` for form controls.
     *
     * See `DefaultValueAccessor` for how to implement one.
     *
     * \@publicApi
     * @type {?}
     */
    const NG_VALUE_ACCESSOR = new InjectionToken('NgValueAccessor');

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/forms/src/directives/checkbox_value_accessor.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const CHECKBOX_VALUE_ACCESSOR = {
        provide: NG_VALUE_ACCESSOR,
        useExisting: forwardRef((/**
         * @return {?}
         */
        () => CheckboxControlValueAccessor)),
        multi: true,
    };
    /**
     * \@description
     * A `ControlValueAccessor` for writing a value and listening to changes on a checkbox input
     * element.
     *
     * \@usageNotes
     *
     * ### Using a checkbox with a reactive form.
     *
     * The following example shows how to use a checkbox with a reactive form.
     *
     * ```ts
     * const rememberLoginControl = new FormControl();
     * ```
     *
     * ```
     * <input type="checkbox" [formControl]="rememberLoginControl">
     * ```
     *
     * \@ngModule ReactiveFormsModule
     * \@ngModule FormsModule
     * \@publicApi
     */
    class CheckboxControlValueAccessor {
        /**
         * @param {?} _renderer
         * @param {?} _elementRef
         */
        constructor(_renderer, _elementRef) {
            this._renderer = _renderer;
            this._elementRef = _elementRef;
            /**
             * \@description
             * The registered callback function called when a change event occurs on the input element.
             */
            this.onChange = (/**
             * @param {?} _
             * @return {?}
             */
            (_) => { });
            /**
             * \@description
             * The registered callback function called when a blur event occurs on the input element.
             */
            this.onTouched = (/**
             * @return {?}
             */
            () => { });
        }
        /**
         * Sets the "checked" property on the input element.
         *
         * @param {?} value The checked value
         * @return {?}
         */
        writeValue(value) {
            this._renderer.setProperty(this._elementRef.nativeElement, 'checked', value);
        }
        /**
         * \@description
         * Registers a function called when the control value changes.
         *
         * @param {?} fn The callback function
         * @return {?}
         */
        registerOnChange(fn) {
            this.onChange = fn;
        }
        /**
         * \@description
         * Registers a function called when the control is touched.
         *
         * @param {?} fn The callback function
         * @return {?}
         */
        registerOnTouched(fn) {
            this.onTouched = fn;
        }
        /**
         * Sets the "disabled" property on the input element.
         *
         * @param {?} isDisabled The disabled value
         * @return {?}
         */
        setDisabledState(isDisabled) {
            this._renderer.setProperty(this._elementRef.nativeElement, 'disabled', isDisabled);
        }
    }
    CheckboxControlValueAccessor.ɵfac = function CheckboxControlValueAccessor_Factory(t) { return new (t || CheckboxControlValueAccessor)(ɵɵdirectiveInject(Renderer2), ɵɵdirectiveInject(ElementRef)); };
    CheckboxControlValueAccessor.ɵdir = ɵɵdefineDirective({ type: CheckboxControlValueAccessor, selectors: [["input", "type", "checkbox", "formControlName", ""], ["input", "type", "checkbox", "formControl", ""], ["input", "type", "checkbox", "ngModel", ""]], hostBindings: function CheckboxControlValueAccessor_HostBindings(rf, ctx) { if (rf & 1) {
            ɵɵlistener("change", function CheckboxControlValueAccessor_change_HostBindingHandler($event) { return ctx.onChange($event.target.checked); })("blur", function CheckboxControlValueAccessor_blur_HostBindingHandler() { return ctx.onTouched(); });
        } }, features: [ɵɵProvidersFeature([CHECKBOX_VALUE_ACCESSOR])] });
    /** @nocollapse */
    CheckboxControlValueAccessor.ctorParameters = () => [
        { type: Renderer2 },
        { type: ElementRef }
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/forms/src/directives/default_value_accessor.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const DEFAULT_VALUE_ACCESSOR = {
        provide: NG_VALUE_ACCESSOR,
        useExisting: forwardRef((/**
         * @return {?}
         */
        () => DefaultValueAccessor)),
        multi: true
    };
    /**
     * We must check whether the agent is Android because composition events
     * behave differently between iOS and Android.
     * @return {?}
     */
    function _isAndroid() {
        /** @type {?} */
        const userAgent = getDOM() ? getDOM().getUserAgent() : '';
        return /android (\d+)/.test(userAgent.toLowerCase());
    }
    /**
     * \@description
     * Provide this token to control if form directives buffer IME input until
     * the "compositionend" event occurs.
     * \@publicApi
     * @type {?}
     */
    const COMPOSITION_BUFFER_MODE = new InjectionToken('CompositionEventMode');
    /**
     * \@description
     * The default `ControlValueAccessor` for writing a value and listening to changes on input
     * elements. The accessor is used by the `FormControlDirective`, `FormControlName`, and
     * `NgModel` directives.
     *
     * \@usageNotes
     *
     * ### Using the default value accessor
     *
     * The following example shows how to use an input element that activates the default value accessor
     * (in this case, a text field).
     *
     * ```ts
     * const firstNameControl = new FormControl();
     * ```
     *
     * ```
     * <input type="text" [formControl]="firstNameControl">
     * ```
     *
     * \@ngModule ReactiveFormsModule
     * \@ngModule FormsModule
     * \@publicApi
     */
    class DefaultValueAccessor {
        /**
         * @param {?} _renderer
         * @param {?} _elementRef
         * @param {?} _compositionMode
         */
        constructor(_renderer, _elementRef, _compositionMode) {
            this._renderer = _renderer;
            this._elementRef = _elementRef;
            this._compositionMode = _compositionMode;
            /**
             * \@description
             * The registered callback function called when an input event occurs on the input element.
             */
            this.onChange = (/**
             * @param {?} _
             * @return {?}
             */
            (_) => { });
            /**
             * \@description
             * The registered callback function called when a blur event occurs on the input element.
             */
            this.onTouched = (/**
             * @return {?}
             */
            () => { });
            /**
             * Whether the user is creating a composition string (IME events).
             */
            this._composing = false;
            if (this._compositionMode == null) {
                this._compositionMode = !_isAndroid();
            }
        }
        /**
         * Sets the "value" property on the input element.
         *
         * @param {?} value The checked value
         * @return {?}
         */
        writeValue(value) {
            /** @type {?} */
            const normalizedValue = value == null ? '' : value;
            this._renderer.setProperty(this._elementRef.nativeElement, 'value', normalizedValue);
        }
        /**
         * \@description
         * Registers a function called when the control value changes.
         *
         * @param {?} fn The callback function
         * @return {?}
         */
        registerOnChange(fn) {
            this.onChange = fn;
        }
        /**
         * \@description
         * Registers a function called when the control is touched.
         *
         * @param {?} fn The callback function
         * @return {?}
         */
        registerOnTouched(fn) {
            this.onTouched = fn;
        }
        /**
         * Sets the "disabled" property on the input element.
         *
         * @param {?} isDisabled The disabled value
         * @return {?}
         */
        setDisabledState(isDisabled) {
            this._renderer.setProperty(this._elementRef.nativeElement, 'disabled', isDisabled);
        }
        /**
         * \@internal
         * @param {?} value
         * @return {?}
         */
        _handleInput(value) {
            if (!this._compositionMode || (this._compositionMode && !this._composing)) {
                this.onChange(value);
            }
        }
        /**
         * \@internal
         * @return {?}
         */
        _compositionStart() {
            this._composing = true;
        }
        /**
         * \@internal
         * @param {?} value
         * @return {?}
         */
        _compositionEnd(value) {
            this._composing = false;
            this._compositionMode && this.onChange(value);
        }
    }
    DefaultValueAccessor.ɵfac = function DefaultValueAccessor_Factory(t) { return new (t || DefaultValueAccessor)(ɵɵdirectiveInject(Renderer2), ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(COMPOSITION_BUFFER_MODE, 8)); };
    DefaultValueAccessor.ɵdir = ɵɵdefineDirective({ type: DefaultValueAccessor, selectors: [["input", "formControlName", "", 3, "type", "checkbox"], ["textarea", "formControlName", ""], ["input", "formControl", "", 3, "type", "checkbox"], ["textarea", "formControl", ""], ["input", "ngModel", "", 3, "type", "checkbox"], ["textarea", "ngModel", ""], ["", "ngDefaultControl", ""]], hostBindings: function DefaultValueAccessor_HostBindings(rf, ctx) { if (rf & 1) {
            ɵɵlistener("input", function DefaultValueAccessor_input_HostBindingHandler($event) { return ctx._handleInput($event.target.value); })("blur", function DefaultValueAccessor_blur_HostBindingHandler() { return ctx.onTouched(); })("compositionstart", function DefaultValueAccessor_compositionstart_HostBindingHandler() { return ctx._compositionStart(); })("compositionend", function DefaultValueAccessor_compositionend_HostBindingHandler($event) { return ctx._compositionEnd($event.target.value); });
        } }, features: [ɵɵProvidersFeature([DEFAULT_VALUE_ACCESSOR])] });
    /** @nocollapse */
    DefaultValueAccessor.ctorParameters = () => [
        { type: Renderer2 },
        { type: ElementRef },
        { type: Boolean, decorators: [{ type: Optional }, { type: Inject, args: [COMPOSITION_BUFFER_MODE,] }] }
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/forms/src/directives/abstract_control_directive.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * \@description
     * Base class for control directives.
     *
     * This class is only used internally in the `ReactiveFormsModule` and the `FormsModule`.
     *
     * \@publicApi
     * @abstract
     */
    class AbstractControlDirective {
        /**
         * \@description
         * Reports the value of the control if it is present, otherwise null.
         * @return {?}
         */
        get value() {
            return this.control ? this.control.value : null;
        }
        /**
         * \@description
         * Reports whether the control is valid. A control is considered valid if no
         * validation errors exist with the current value.
         * If the control is not present, null is returned.
         * @return {?}
         */
        get valid() {
            return this.control ? this.control.valid : null;
        }
        /**
         * \@description
         * Reports whether the control is invalid, meaning that an error exists in the input value.
         * If the control is not present, null is returned.
         * @return {?}
         */
        get invalid() {
            return this.control ? this.control.invalid : null;
        }
        /**
         * \@description
         * Reports whether a control is pending, meaning that that async validation is occurring and
         * errors are not yet available for the input value. If the control is not present, null is
         * returned.
         * @return {?}
         */
        get pending() {
            return this.control ? this.control.pending : null;
        }
        /**
         * \@description
         * Reports whether the control is disabled, meaning that the control is disabled
         * in the UI and is exempt from validation checks and excluded from aggregate
         * values of ancestor controls. If the control is not present, null is returned.
         * @return {?}
         */
        get disabled() {
            return this.control ? this.control.disabled : null;
        }
        /**
         * \@description
         * Reports whether the control is enabled, meaning that the control is included in ancestor
         * calculations of validity or value. If the control is not present, null is returned.
         * @return {?}
         */
        get enabled() {
            return this.control ? this.control.enabled : null;
        }
        /**
         * \@description
         * Reports the control's validation errors. If the control is not present, null is returned.
         * @return {?}
         */
        get errors() {
            return this.control ? this.control.errors : null;
        }
        /**
         * \@description
         * Reports whether the control is pristine, meaning that the user has not yet changed
         * the value in the UI. If the control is not present, null is returned.
         * @return {?}
         */
        get pristine() {
            return this.control ? this.control.pristine : null;
        }
        /**
         * \@description
         * Reports whether the control is dirty, meaning that the user has changed
         * the value in the UI. If the control is not present, null is returned.
         * @return {?}
         */
        get dirty() {
            return this.control ? this.control.dirty : null;
        }
        /**
         * \@description
         * Reports whether the control is touched, meaning that the user has triggered
         * a `blur` event on it. If the control is not present, null is returned.
         * @return {?}
         */
        get touched() {
            return this.control ? this.control.touched : null;
        }
        /**
         * \@description
         * Reports the validation status of the control. Possible values include:
         * 'VALID', 'INVALID', 'DISABLED', and 'PENDING'.
         * If the control is not present, null is returned.
         * @return {?}
         */
        get status() {
            return this.control ? this.control.status : null;
        }
        /**
         * \@description
         * Reports whether the control is untouched, meaning that the user has not yet triggered
         * a `blur` event on it. If the control is not present, null is returned.
         * @return {?}
         */
        get untouched() {
            return this.control ? this.control.untouched : null;
        }
        /**
         * \@description
         * Returns a multicasting observable that emits a validation status whenever it is
         * calculated for the control. If the control is not present, null is returned.
         * @return {?}
         */
        get statusChanges() {
            return this.control ? this.control.statusChanges : null;
        }
        /**
         * \@description
         * Returns a multicasting observable of value changes for the control that emits every time the
         * value of the control changes in the UI or programmatically.
         * If the control is not present, null is returned.
         * @return {?}
         */
        get valueChanges() {
            return this.control ? this.control.valueChanges : null;
        }
        /**
         * \@description
         * Returns an array that represents the path from the top-level form to this control.
         * Each index is the string name of the control on that level.
         * @return {?}
         */
        get path() {
            return null;
        }
        /**
         * \@description
         * Resets the control with the provided value if the control is present.
         * @param {?=} value
         * @return {?}
         */
        reset(value = undefined) {
            if (this.control)
                this.control.reset(value);
        }
        /**
         * \@description
         * Reports whether the control with the given path has the error specified.
         *
         * \@usageNotes
         * For example, for the following `FormGroup`:
         *
         * ```
         * form = new FormGroup({
         *   address: new FormGroup({ street: new FormControl() })
         * });
         * ```
         *
         * The path to the 'street' control from the root form would be 'address' -> 'street'.
         *
         * It can be provided to this method in one of two formats:
         *
         * 1. An array of string control names, e.g. `['address', 'street']`
         * 1. A period-delimited list of control names in one string, e.g. `'address.street'`
         *
         * If no path is given, this method checks for the error on the current control.
         *
         * @param {?} errorCode The code of the error to check
         * @param {?=} path A list of control names that designates how to move from the current control
         * to the control that should be queried for errors.
         *
         * @return {?} whether the given error is present in the control at the given path.
         *
         * If the control is not present, false is returned.
         */
        hasError(errorCode, path) {
            return this.control ? this.control.hasError(errorCode, path) : false;
        }
        /**
         * \@description
         * Reports error data for the control with the given path.
         *
         * \@usageNotes
         * For example, for the following `FormGroup`:
         *
         * ```
         * form = new FormGroup({
         *   address: new FormGroup({ street: new FormControl() })
         * });
         * ```
         *
         * The path to the 'street' control from the root form would be 'address' -> 'street'.
         *
         * It can be provided to this method in one of two formats:
         *
         * 1. An array of string control names, e.g. `['address', 'street']`
         * 1. A period-delimited list of control names in one string, e.g. `'address.street'`
         *
         * @param {?} errorCode The code of the error to check
         * @param {?=} path A list of control names that designates how to move from the current control
         * to the control that should be queried for errors.
         *
         * @return {?} error data for that particular error. If the control or error is not present,
         * null is returned.
         */
        getError(errorCode, path) {
            return this.control ? this.control.getError(errorCode, path) : null;
        }
    }
    AbstractControlDirective.ɵfac = function AbstractControlDirective_Factory(t) { return new (t || AbstractControlDirective)(); };
    AbstractControlDirective.ɵdir = ɵɵdefineDirective({ type: AbstractControlDirective });

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/forms/src/directives/control_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * \@description
     * A base class for directives that contain multiple registered instances of `NgControl`.
     * Only used by the forms module.
     *
     * \@publicApi
     * @abstract
     */
    class ControlContainer extends AbstractControlDirective {
        /**
         * \@description
         * The top-level form directive for the control.
         * @return {?}
         */
        get formDirective() {
            return null;
        }
        /**
         * \@description
         * The path to this group.
         * @return {?}
         */
        get path() {
            return null;
        }
    }
    ControlContainer.ɵfac = function ControlContainer_Factory(t) { return ɵControlContainer_BaseFactory(t || ControlContainer); };
    ControlContainer.ɵdir = ɵɵdefineDirective({ type: ControlContainer, features: [ɵɵInheritDefinitionFeature] });
    const ɵControlContainer_BaseFactory = ɵɵgetInheritedFactory(ControlContainer);

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/forms/src/directives/ng_control.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @return {?}
     */
    function unimplemented() {
        throw new Error('unimplemented');
    }
    /**
     * \@description
     * A base class that all control `FormControl`-based directives extend. It binds a `FormControl`
     * object to a DOM element.
     *
     * \@publicApi
     * @abstract
     */
    class NgControl extends AbstractControlDirective {
        constructor() {
            super(...arguments);
            /**
             * \@description
             * The parent form for the control.
             *
             * \@internal
             */
            this._parent = null;
            /**
             * \@description
             * The name for the control
             */
            this.name = null;
            /**
             * \@description
             * The value accessor for the control
             */
            this.valueAccessor = null;
            /**
             * \@description
             * The uncomposed array of synchronous validators for the control
             *
             * \@internal
             */
            this._rawValidators = [];
            /**
             * \@description
             * The uncomposed array of async validators for the control
             *
             * \@internal
             */
            this._rawAsyncValidators = [];
        }
        /**
         * \@description
         * The registered synchronous validator function for the control
         *
         * @throws An exception that this method is not implemented
         * @return {?}
         */
        get validator() {
            return (/** @type {?} */ (unimplemented()));
        }
        /**
         * \@description
         * The registered async validator function for the control
         *
         * @throws An exception that this method is not implemented
         * @return {?}
         */
        get asyncValidator() {
            return (/** @type {?} */ (unimplemented()));
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/forms/src/directives/ng_control_status.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class AbstractControlStatus {
        /**
         * @param {?} cd
         */
        constructor(cd) {
            this._cd = cd;
        }
        /**
         * @return {?}
         */
        get ngClassUntouched() {
            return this._cd.control ? this._cd.control.untouched : false;
        }
        /**
         * @return {?}
         */
        get ngClassTouched() {
            return this._cd.control ? this._cd.control.touched : false;
        }
        /**
         * @return {?}
         */
        get ngClassPristine() {
            return this._cd.control ? this._cd.control.pristine : false;
        }
        /**
         * @return {?}
         */
        get ngClassDirty() {
            return this._cd.control ? this._cd.control.dirty : false;
        }
        /**
         * @return {?}
         */
        get ngClassValid() {
            return this._cd.control ? this._cd.control.valid : false;
        }
        /**
         * @return {?}
         */
        get ngClassInvalid() {
            return this._cd.control ? this._cd.control.invalid : false;
        }
        /**
         * @return {?}
         */
        get ngClassPending() {
            return this._cd.control ? this._cd.control.pending : false;
        }
    }
    /**
     * \@description
     * Directive automatically applied to Angular form controls that sets CSS classes
     * based on control status.
     *
     * \@usageNotes
     *
     * ### CSS classes applied
     *
     * The following classes are applied as the properties become true:
     *
     * * ng-valid
     * * ng-invalid
     * * ng-pending
     * * ng-pristine
     * * ng-dirty
     * * ng-untouched
     * * ng-touched
     *
     * \@ngModule ReactiveFormsModule
     * \@ngModule FormsModule
     * \@publicApi
     */
    class NgControlStatus extends AbstractControlStatus {
        /**
         * @param {?} cd
         */
        constructor(cd) {
            super(cd);
        }
    }
    NgControlStatus.ɵfac = function NgControlStatus_Factory(t) { return new (t || NgControlStatus)(ɵɵdirectiveInject(NgControl, 2)); };
    NgControlStatus.ɵdir = ɵɵdefineDirective({ type: NgControlStatus, selectors: [["", "formControlName", ""], ["", "ngModel", ""], ["", "formControl", ""]], hostVars: 14, hostBindings: function NgControlStatus_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵclassProp("ng-untouched", ctx.ngClassUntouched)("ng-touched", ctx.ngClassTouched)("ng-pristine", ctx.ngClassPristine)("ng-dirty", ctx.ngClassDirty)("ng-valid", ctx.ngClassValid)("ng-invalid", ctx.ngClassInvalid)("ng-pending", ctx.ngClassPending);
        } }, features: [ɵɵInheritDefinitionFeature] });
    /** @nocollapse */
    NgControlStatus.ctorParameters = () => [
        { type: NgControl, decorators: [{ type: Self }] }
    ];
    /**
     * \@description
     * Directive automatically applied to Angular form groups that sets CSS classes
     * based on control status (valid/invalid/dirty/etc).
     *
     * @see `NgControlStatus`
     *
     * \@ngModule ReactiveFormsModule
     * \@ngModule FormsModule
     * \@publicApi
     */
    class NgControlStatusGroup extends AbstractControlStatus {
        /**
         * @param {?} cd
         */
        constructor(cd) {
            super(cd);
        }
    }
    NgControlStatusGroup.ɵfac = function NgControlStatusGroup_Factory(t) { return new (t || NgControlStatusGroup)(ɵɵdirectiveInject(ControlContainer, 2)); };
    NgControlStatusGroup.ɵdir = ɵɵdefineDirective({ type: NgControlStatusGroup, selectors: [["", "formGroupName", ""], ["", "formArrayName", ""], ["", "ngModelGroup", ""], ["", "formGroup", ""], ["form", 3, "ngNoForm", ""], ["", "ngForm", ""]], hostVars: 14, hostBindings: function NgControlStatusGroup_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵclassProp("ng-untouched", ctx.ngClassUntouched)("ng-touched", ctx.ngClassTouched)("ng-pristine", ctx.ngClassPristine)("ng-dirty", ctx.ngClassDirty)("ng-valid", ctx.ngClassValid)("ng-invalid", ctx.ngClassInvalid)("ng-pending", ctx.ngClassPending);
        } }, features: [ɵɵInheritDefinitionFeature] });
    /** @nocollapse */
    NgControlStatusGroup.ctorParameters = () => [
        { type: ControlContainer, decorators: [{ type: Self }] }
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/forms/src/validators.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @param {?} value
     * @return {?}
     */
    function isEmptyInputValue(value) {
        // we don't check for string here so it also works with arrays
        return value == null || value.length === 0;
    }
    /**
     * \@description
     * An `InjectionToken` for registering additional synchronous validators used with
     * `AbstractControl`s.
     *
     * @see `NG_ASYNC_VALIDATORS`
     *
     * \@usageNotes
     *
     * ### Providing a custom validator
     *
     * The following example registers a custom validator directive. Adding the validator to the
     * existing collection of validators requires the `multi: true` option.
     *
     * ```typescript
     * \@Directive({
     *   selector: '[customValidator]',
     *   providers: [{provide: NG_VALIDATORS, useExisting: CustomValidatorDirective, multi: true}]
     * })
     * class CustomValidatorDirective implements Validator {
     *   validate(control: AbstractControl): ValidationErrors | null {
     *     return { 'custom': true };
     *   }
     * }
     * ```
     *
     * \@publicApi
     * @type {?}
     */
    const NG_VALIDATORS = new InjectionToken('NgValidators');
    /**
     * \@description
     * An `InjectionToken` for registering additional asynchronous validators used with
     * `AbstractControl`s.
     *
     * @see `NG_VALIDATORS`
     *
     * \@publicApi
     * @type {?}
     */
    const NG_ASYNC_VALIDATORS = new InjectionToken('NgAsyncValidators');
    /**
     * A regular expression that matches valid e-mail addresses.
     *
     * At a high level, this regexp matches e-mail addresses of the format `local-part\@tld`, where:
     * - `local-part` consists of one or more of the allowed characters (alphanumeric and some
     *   punctuation symbols).
     * - `local-part` cannot begin or end with a period (`.`).
     * - `local-part` cannot be longer than 64 characters.
     * - `tld` consists of one or more `labels` separated by periods (`.`). For example `localhost` or
     *   `foo.com`.
     * - A `label` consists of one or more of the allowed characters (alphanumeric, dashes (`-`) and
     *   periods (`.`)).
     * - A `label` cannot begin or end with a dash (`-`) or a period (`.`).
     * - A `label` cannot be longer than 63 characters.
     * - The whole address cannot be longer than 254 characters.
     *
     * ## Implementation background
     *
     * This regexp was ported over from AngularJS (see there for git history):
     * https://github.com/angular/angular.js/blob/c133ef836/src/ng/directive/input.js#L27
     * It is based on the
     * [WHATWG version](https://html.spec.whatwg.org/multipage/input.html#valid-e-mail-address) with
     * some enhancements to incorporate more RFC rules (such as rules related to domain names and the
     * lengths of different parts of the address). The main differences from the WHATWG version are:
     *   - Disallow `local-part` to begin or end with a period (`.`).
     *   - Disallow `local-part` length to exceed 64 characters.
     *   - Disallow total address length to exceed 254 characters.
     *
     * See [this commit](https://github.com/angular/angular.js/commit/f3f5cf72e) for more details.
     * @type {?}
     */
    const EMAIL_REGEXP = /^(?=.{1,254}$)(?=.{1,64}@)[a-zA-Z0-9!#$%&'*+/=?^_`{|}~-]+(?:\.[a-zA-Z0-9!#$%&'*+/=?^_`{|}~-]+)*@[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?(?:\.[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?)*$/;
    /**
     * \@description
     * Provides a set of built-in validators that can be used by form controls.
     *
     * A validator is a function that processes a `FormControl` or collection of
     * controls and returns an error map or null. A null map means that validation has passed.
     *
     * @see [Form Validation](/guide/form-validation)
     *
     * \@publicApi
     */
    class Validators {
        /**
         * \@description
         * Validator that requires the control's value to be greater than or equal to the provided number.
         * The validator exists only as a function and not as a directive.
         *
         * \@usageNotes
         *
         * ### Validate against a minimum of 3
         *
         * ```typescript
         * const control = new FormControl(2, Validators.min(3));
         *
         * console.log(control.errors); // {min: {min: 3, actual: 2}}
         * ```
         *
         * @see `updateValueAndValidity()`
         *
         * @param {?} min
         * @return {?} A validator function that returns an error map with the
         * `min` property if the validation check fails, otherwise `null`.
         *
         */
        static min(min) {
            return (/**
             * @param {?} control
             * @return {?}
             */
            (control) => {
                if (isEmptyInputValue(control.value) || isEmptyInputValue(min)) {
                    return null; // don't validate empty values to allow optional controls
                }
                /** @type {?} */
                const value = parseFloat(control.value);
                // Controls with NaN values after parsing should be treated as not having a
                // minimum, per the HTML forms spec: https://www.w3.org/TR/html5/forms.html#attr-input-min
                return !isNaN(value) && value < min ? { 'min': { 'min': min, 'actual': control.value } } : null;
            });
        }
        /**
         * \@description
         * Validator that requires the control's value to be less than or equal to the provided number.
         * The validator exists only as a function and not as a directive.
         *
         * \@usageNotes
         *
         * ### Validate against a maximum of 15
         *
         * ```typescript
         * const control = new FormControl(16, Validators.max(15));
         *
         * console.log(control.errors); // {max: {max: 15, actual: 16}}
         * ```
         *
         * @see `updateValueAndValidity()`
         *
         * @param {?} max
         * @return {?} A validator function that returns an error map with the
         * `max` property if the validation check fails, otherwise `null`.
         *
         */
        static max(max) {
            return (/**
             * @param {?} control
             * @return {?}
             */
            (control) => {
                if (isEmptyInputValue(control.value) || isEmptyInputValue(max)) {
                    return null; // don't validate empty values to allow optional controls
                }
                /** @type {?} */
                const value = parseFloat(control.value);
                // Controls with NaN values after parsing should be treated as not having a
                // maximum, per the HTML forms spec: https://www.w3.org/TR/html5/forms.html#attr-input-max
                return !isNaN(value) && value > max ? { 'max': { 'max': max, 'actual': control.value } } : null;
            });
        }
        /**
         * \@description
         * Validator that requires the control have a non-empty value.
         *
         * \@usageNotes
         *
         * ### Validate that the field is non-empty
         *
         * ```typescript
         * const control = new FormControl('', Validators.required);
         *
         * console.log(control.errors); // {required: true}
         * ```
         *
         * @see `updateValueAndValidity()`
         *
         * @param {?} control
         * @return {?} An error map with the `required` property
         * if the validation check fails, otherwise `null`.
         *
         */
        static required(control) {
            return isEmptyInputValue(control.value) ? { 'required': true } : null;
        }
        /**
         * \@description
         * Validator that requires the control's value be true. This validator is commonly
         * used for required checkboxes.
         *
         * \@usageNotes
         *
         * ### Validate that the field value is true
         *
         * ```typescript
         * const control = new FormControl('', Validators.requiredTrue);
         *
         * console.log(control.errors); // {required: true}
         * ```
         *
         * @see `updateValueAndValidity()`
         *
         * @param {?} control
         * @return {?} An error map that contains the `required` property
         * set to `true` if the validation check fails, otherwise `null`.
         *
         */
        static requiredTrue(control) {
            return control.value === true ? null : { 'required': true };
        }
        /**
         * \@description
         * Validator that requires the control's value pass an email validation test.
         *
         * Tests the value using a [regular
         * expression](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Guide/Regular_Expressions)
         * pattern suitable for common usecases. The pattern is based on the definition of a valid email
         * address in the [WHATWG HTML
         * specification](https://html.spec.whatwg.org/multipage/input.html#valid-e-mail-address) with
         * some enhancements to incorporate more RFC rules (such as rules related to domain names and the
         * lengths of different parts of the address).
         *
         * The differences from the WHATWG version include:
         * - Disallow `local-part` (the part before the `\@` symbol) to begin or end with a period (`.`).
         * - Disallow `local-part` to be longer than 64 characters.
         * - Disallow the whole address to be longer than 254 characters.
         *
         * If this pattern does not satisfy your business needs, you can use `Validators.pattern()` to
         * validate the value against a different pattern.
         *
         * \@usageNotes
         *
         * ### Validate that the field matches a valid email pattern
         *
         * ```typescript
         * const control = new FormControl('bad\@', Validators.email);
         *
         * console.log(control.errors); // {email: true}
         * ```
         *
         * @see `updateValueAndValidity()`
         *
         * @param {?} control
         * @return {?} An error map with the `email` property
         * if the validation check fails, otherwise `null`.
         *
         */
        static email(control) {
            if (isEmptyInputValue(control.value)) {
                return null; // don't validate empty values to allow optional controls
            }
            return EMAIL_REGEXP.test(control.value) ? null : { 'email': true };
        }
        /**
         * \@description
         * Validator that requires the length of the control's value to be greater than or equal
         * to the provided minimum length. This validator is also provided by default if you use the
         * the HTML5 `minlength` attribute. Note that the `minLength` validator is intended to be used
         * only for types that have a numeric `length` property, such as strings or arrays. The
         * `minLength` validator logic is also not invoked for values when their `length` property is 0
         * (for example in case of an empty string or an empty array), to support optional controls. You
         * can use the standard `required` validator if empty values should not be considered valid.
         *
         * \@usageNotes
         *
         * ### Validate that the field has a minimum of 3 characters
         *
         * ```typescript
         * const control = new FormControl('ng', Validators.minLength(3));
         *
         * console.log(control.errors); // {minlength: {requiredLength: 3, actualLength: 2}}
         * ```
         *
         * ```html
         * <input minlength="5">
         * ```
         *
         * @see `updateValueAndValidity()`
         *
         * @param {?} minLength
         * @return {?} A validator function that returns an error map with the
         * `minlength` if the validation check fails, otherwise `null`.
         *
         */
        static minLength(minLength) {
            return (/**
             * @param {?} control
             * @return {?}
             */
            (control) => {
                if (isEmptyInputValue(control.value)) {
                    return null; // don't validate empty values to allow optional controls
                }
                /** @type {?} */
                const length = control.value ? control.value.length : 0;
                return length < minLength ?
                    { 'minlength': { 'requiredLength': minLength, 'actualLength': length } } :
                    null;
            });
        }
        /**
         * \@description
         * Validator that requires the length of the control's value to be less than or equal
         * to the provided maximum length. This validator is also provided by default if you use the
         * the HTML5 `maxlength` attribute. Note that the `maxLength` validator is intended to be used
         * only for types that have a numeric `length` property, such as strings or arrays.
         *
         * \@usageNotes
         *
         * ### Validate that the field has maximum of 5 characters
         *
         * ```typescript
         * const control = new FormControl('Angular', Validators.maxLength(5));
         *
         * console.log(control.errors); // {maxlength: {requiredLength: 5, actualLength: 7}}
         * ```
         *
         * ```html
         * <input maxlength="5">
         * ```
         *
         * @see `updateValueAndValidity()`
         *
         * @param {?} maxLength
         * @return {?} A validator function that returns an error map with the
         * `maxlength` property if the validation check fails, otherwise `null`.
         *
         */
        static maxLength(maxLength) {
            return (/**
             * @param {?} control
             * @return {?}
             */
            (control) => {
                /** @type {?} */
                const length = control.value ? control.value.length : 0;
                return length > maxLength ?
                    { 'maxlength': { 'requiredLength': maxLength, 'actualLength': length } } :
                    null;
            });
        }
        /**
         * \@description
         * Validator that requires the control's value to match a regex pattern. This validator is also
         * provided by default if you use the HTML5 `pattern` attribute.
         *
         * \@usageNotes
         *
         * ### Validate that the field only contains letters or spaces
         *
         * ```typescript
         * const control = new FormControl('1', Validators.pattern('[a-zA-Z ]*'));
         *
         * console.log(control.errors); // {pattern: {requiredPattern: '^[a-zA-Z ]*$', actualValue: '1'}}
         * ```
         *
         * ```html
         * <input pattern="[a-zA-Z ]*">
         * ```
         *
         * @see `updateValueAndValidity()`
         *
         * @param {?} pattern A regular expression to be used as is to test the values, or a string.
         * If a string is passed, the `^` character is prepended and the `$` character is
         * appended to the provided string (if not already present), and the resulting regular
         * expression is used to test the values.
         *
         * @return {?} A validator function that returns an error map with the
         * `pattern` property if the validation check fails, otherwise `null`.
         *
         */
        static pattern(pattern) {
            if (!pattern)
                return Validators.nullValidator;
            /** @type {?} */
            let regex;
            /** @type {?} */
            let regexStr;
            if (typeof pattern === 'string') {
                regexStr = '';
                if (pattern.charAt(0) !== '^')
                    regexStr += '^';
                regexStr += pattern;
                if (pattern.charAt(pattern.length - 1) !== '$')
                    regexStr += '$';
                regex = new RegExp(regexStr);
            }
            else {
                regexStr = pattern.toString();
                regex = pattern;
            }
            return (/**
             * @param {?} control
             * @return {?}
             */
            (control) => {
                if (isEmptyInputValue(control.value)) {
                    return null; // don't validate empty values to allow optional controls
                }
                /** @type {?} */
                const value = control.value;
                return regex.test(value) ? null :
                    { 'pattern': { 'requiredPattern': regexStr, 'actualValue': value } };
            });
        }
        /**
         * \@description
         * Validator that performs no operation.
         *
         * @see `updateValueAndValidity()`
         *
         * @param {?} control
         * @return {?}
         */
        static nullValidator(control) {
            return null;
        }
        /**
         * @param {?} validators
         * @return {?}
         */
        static compose(validators) {
            if (!validators)
                return null;
            /** @type {?} */
            const presentValidators = (/** @type {?} */ (validators.filter(isPresent)));
            if (presentValidators.length == 0)
                return null;
            return (/**
             * @param {?} control
             * @return {?}
             */
            function (control) {
                return _mergeErrors(_executeValidators(control, presentValidators));
            });
        }
        /**
         * \@description
         * Compose multiple async validators into a single function that returns the union
         * of the individual error objects for the provided control.
         *
         * @see `updateValueAndValidity()`
         *
         * @param {?} validators
         * @return {?} A validator function that returns an error map with the
         * merged error objects of the async validators if the validation check fails, otherwise `null`.
         *
         */
        static composeAsync(validators) {
            if (!validators)
                return null;
            /** @type {?} */
            const presentValidators = (/** @type {?} */ (validators.filter(isPresent)));
            if (presentValidators.length == 0)
                return null;
            return (/**
             * @param {?} control
             * @return {?}
             */
            function (control) {
                /** @type {?} */
                const observables = _executeAsyncValidators(control, presentValidators).map(toObservable);
                return forkJoin(observables).pipe(map(_mergeErrors));
            });
        }
    }
    /**
     * @param {?} o
     * @return {?}
     */
    function isPresent(o) {
        return o != null;
    }
    /**
     * @param {?} r
     * @return {?}
     */
    function toObservable(r) {
        /** @type {?} */
        const obs = isPromise$1(r) ? from(r) : r;
        if (!(isObservable$1(obs))) {
            throw new Error(`Expected validator to return Promise or Observable.`);
        }
        return obs;
    }
    /**
     * @param {?} control
     * @param {?} validators
     * @return {?}
     */
    function _executeValidators(control, validators) {
        return validators.map((/**
         * @param {?} v
         * @return {?}
         */
        v => v(control)));
    }
    /**
     * @param {?} control
     * @param {?} validators
     * @return {?}
     */
    function _executeAsyncValidators(control, validators) {
        return validators.map((/**
         * @param {?} v
         * @return {?}
         */
        v => v(control)));
    }
    /**
     * @param {?} arrayOfErrors
     * @return {?}
     */
    function _mergeErrors(arrayOfErrors) {
        /** @type {?} */
        let res = {};
        // Not using Array.reduce here due to a Chrome 80 bug
        // https://bugs.chromium.org/p/chromium/issues/detail?id=1049982
        arrayOfErrors.forEach((/**
         * @param {?} errors
         * @return {?}
         */
        (errors) => {
            res = errors != null ? Object.assign(Object.assign({}, (/** @type {?} */ (res))), errors) : (/** @type {?} */ (res));
        }));
        return Object.keys(res).length === 0 ? null : res;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/forms/src/directives/normalize_validator.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * @param {?} validator
     * @return {?}
     */
    function normalizeValidator(validator) {
        if (((/** @type {?} */ (validator))).validate) {
            return (/**
             * @param {?} c
             * @return {?}
             */
            (c) => ((/** @type {?} */ (validator))).validate(c));
        }
        else {
            return (/** @type {?} */ (validator));
        }
    }
    /**
     * @param {?} validator
     * @return {?}
     */
    function normalizeAsyncValidator(validator) {
        if (((/** @type {?} */ (validator))).validate) {
            return (/**
             * @param {?} c
             * @return {?}
             */
            (c) => ((/** @type {?} */ (validator))).validate(c));
        }
        else {
            return (/** @type {?} */ (validator));
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/forms/src/directives/number_value_accessor.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const NUMBER_VALUE_ACCESSOR = {
        provide: NG_VALUE_ACCESSOR,
        useExisting: forwardRef((/**
         * @return {?}
         */
        () => NumberValueAccessor)),
        multi: true
    };
    /**
     * \@description
     * The `ControlValueAccessor` for writing a number value and listening to number input changes.
     * The value accessor is used by the `FormControlDirective`, `FormControlName`, and  `NgModel`
     * directives.
     *
     * \@usageNotes
     *
     * ### Using a number input with a reactive form.
     *
     * The following example shows how to use a number input with a reactive form.
     *
     * ```ts
     * const totalCountControl = new FormControl();
     * ```
     *
     * ```
     * <input type="number" [formControl]="totalCountControl">
     * ```
     *
     * \@ngModule ReactiveFormsModule
     * \@ngModule FormsModule
     * \@publicApi
     */
    class NumberValueAccessor {
        /**
         * @param {?} _renderer
         * @param {?} _elementRef
         */
        constructor(_renderer, _elementRef) {
            this._renderer = _renderer;
            this._elementRef = _elementRef;
            /**
             * \@description
             * The registered callback function called when a change or input event occurs on the input
             * element.
             */
            this.onChange = (/**
             * @param {?} _
             * @return {?}
             */
            (_) => { });
            /**
             * \@description
             * The registered callback function called when a blur event occurs on the input element.
             */
            this.onTouched = (/**
             * @return {?}
             */
            () => { });
        }
        /**
         * Sets the "value" property on the input element.
         *
         * @param {?} value The checked value
         * @return {?}
         */
        writeValue(value) {
            // The value needs to be normalized for IE9, otherwise it is set to 'null' when null
            /** @type {?} */
            const normalizedValue = value == null ? '' : value;
            this._renderer.setProperty(this._elementRef.nativeElement, 'value', normalizedValue);
        }
        /**
         * \@description
         * Registers a function called when the control value changes.
         *
         * @param {?} fn The callback function
         * @return {?}
         */
        registerOnChange(fn) {
            this.onChange = (/**
             * @param {?} value
             * @return {?}
             */
            (value) => {
                fn(value == '' ? null : parseFloat(value));
            });
        }
        /**
         * \@description
         * Registers a function called when the control is touched.
         *
         * @param {?} fn The callback function
         * @return {?}
         */
        registerOnTouched(fn) {
            this.onTouched = fn;
        }
        /**
         * Sets the "disabled" property on the input element.
         *
         * @param {?} isDisabled The disabled value
         * @return {?}
         */
        setDisabledState(isDisabled) {
            this._renderer.setProperty(this._elementRef.nativeElement, 'disabled', isDisabled);
        }
    }
    NumberValueAccessor.ɵfac = function NumberValueAccessor_Factory(t) { return new (t || NumberValueAccessor)(ɵɵdirectiveInject(Renderer2), ɵɵdirectiveInject(ElementRef)); };
    NumberValueAccessor.ɵdir = ɵɵdefineDirective({ type: NumberValueAccessor, selectors: [["input", "type", "number", "formControlName", ""], ["input", "type", "number", "formControl", ""], ["input", "type", "number", "ngModel", ""]], hostBindings: function NumberValueAccessor_HostBindings(rf, ctx) { if (rf & 1) {
            ɵɵlistener("change", function NumberValueAccessor_change_HostBindingHandler($event) { return ctx.onChange($event.target.value); })("input", function NumberValueAccessor_input_HostBindingHandler($event) { return ctx.onChange($event.target.value); })("blur", function NumberValueAccessor_blur_HostBindingHandler() { return ctx.onTouched(); });
        } }, features: [ɵɵProvidersFeature([NUMBER_VALUE_ACCESSOR])] });
    /** @nocollapse */
    NumberValueAccessor.ctorParameters = () => [
        { type: Renderer2 },
        { type: ElementRef }
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/forms/src/directives/radio_control_value_accessor.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const RADIO_VALUE_ACCESSOR = {
        provide: NG_VALUE_ACCESSOR,
        useExisting: forwardRef((/**
         * @return {?}
         */
        () => RadioControlValueAccessor)),
        multi: true
    };
    /**
     * \@description
     * Class used by Angular to track radio buttons. For internal use only.
     */
    class RadioControlRegistry {
        constructor() {
            this._accessors = [];
        }
        /**
         * \@description
         * Adds a control to the internal registry. For internal use only.
         * @param {?} control
         * @param {?} accessor
         * @return {?}
         */
        add(control, accessor) {
            this._accessors.push([control, accessor]);
        }
        /**
         * \@description
         * Removes a control from the internal registry. For internal use only.
         * @param {?} accessor
         * @return {?}
         */
        remove(accessor) {
            for (let i = this._accessors.length - 1; i >= 0; --i) {
                if (this._accessors[i][1] === accessor) {
                    this._accessors.splice(i, 1);
                    return;
                }
            }
        }
        /**
         * \@description
         * Selects a radio button. For internal use only.
         * @param {?} accessor
         * @return {?}
         */
        select(accessor) {
            this._accessors.forEach((/**
             * @param {?} c
             * @return {?}
             */
            (c) => {
                if (this._isSameGroup(c, accessor) && c[1] !== accessor) {
                    c[1].fireUncheck(accessor.value);
                }
            }));
        }
        /**
         * @private
         * @param {?} controlPair
         * @param {?} accessor
         * @return {?}
         */
        _isSameGroup(controlPair, accessor) {
            if (!controlPair[0].control)
                return false;
            return controlPair[0]._parent === accessor._control._parent &&
                controlPair[1].name === accessor.name;
        }
    }
    RadioControlRegistry.ɵfac = function RadioControlRegistry_Factory(t) { return new (t || RadioControlRegistry)(); };
    RadioControlRegistry.ɵprov = ɵɵdefineInjectable({ token: RadioControlRegistry, factory: RadioControlRegistry.ɵfac });
    /**
     * \@description
     * The `ControlValueAccessor` for writing radio control values and listening to radio control
     * changes. The value accessor is used by the `FormControlDirective`, `FormControlName`, and
     * `NgModel` directives.
     *
     * \@usageNotes
     *
     * ### Using radio buttons with reactive form directives
     *
     * The follow example shows how to use radio buttons in a reactive form. When using radio buttons in
     * a reactive form, radio buttons in the same group should have the same `formControlName`.
     * Providing a `name` attribute is optional.
     *
     * {\@example forms/ts/reactiveRadioButtons/reactive_radio_button_example.ts region='Reactive'}
     *
     * \@ngModule ReactiveFormsModule
     * \@ngModule FormsModule
     * \@publicApi
     */
    class RadioControlValueAccessor {
        /**
         * @param {?} _renderer
         * @param {?} _elementRef
         * @param {?} _registry
         * @param {?} _injector
         */
        constructor(_renderer, _elementRef, _registry, _injector) {
            this._renderer = _renderer;
            this._elementRef = _elementRef;
            this._registry = _registry;
            this._injector = _injector;
            /**
             * \@description
             * The registered callback function called when a change event occurs on the input element.
             */
            this.onChange = (/**
             * @return {?}
             */
            () => { });
            /**
             * \@description
             * The registered callback function called when a blur event occurs on the input element.
             */
            this.onTouched = (/**
             * @return {?}
             */
            () => { });
        }
        /**
         * \@description
         * A lifecycle method called when the directive is initialized. For internal use only.
         * @return {?}
         */
        ngOnInit() {
            this._control = this._injector.get(NgControl);
            this._checkName();
            this._registry.add(this._control, this);
        }
        /**
         * \@description
         * Lifecycle method called before the directive's instance is destroyed. For internal use only.
         * @return {?}
         */
        ngOnDestroy() {
            this._registry.remove(this);
        }
        /**
         * \@description
         * Sets the "checked" property value on the radio input element.
         *
         * @param {?} value The checked value
         * @return {?}
         */
        writeValue(value) {
            this._state = value === this.value;
            this._renderer.setProperty(this._elementRef.nativeElement, 'checked', this._state);
        }
        /**
         * \@description
         * Registers a function called when the control value changes.
         *
         * @param {?} fn The callback function
         * @return {?}
         */
        registerOnChange(fn) {
            this._fn = fn;
            this.onChange = (/**
             * @return {?}
             */
            () => {
                fn(this.value);
                this._registry.select(this);
            });
        }
        /**
         * Sets the "value" on the radio input element and unchecks it.
         *
         * @param {?} value
         * @return {?}
         */
        fireUncheck(value) {
            this.writeValue(value);
        }
        /**
         * \@description
         * Registers a function called when the control is touched.
         *
         * @param {?} fn The callback function
         * @return {?}
         */
        registerOnTouched(fn) {
            this.onTouched = fn;
        }
        /**
         * Sets the "disabled" property on the input element.
         *
         * @param {?} isDisabled The disabled value
         * @return {?}
         */
        setDisabledState(isDisabled) {
            this._renderer.setProperty(this._elementRef.nativeElement, 'disabled', isDisabled);
        }
        /**
         * @private
         * @return {?}
         */
        _checkName() {
            if (this.name && this.formControlName && this.name !== this.formControlName) {
                this._throwNameError();
            }
            if (!this.name && this.formControlName)
                this.name = this.formControlName;
        }
        /**
         * @private
         * @return {?}
         */
        _throwNameError() {
            throw new Error(`
      If you define both a name and a formControlName attribute on your radio button, their values
      must match. Ex: <input type="radio" formControlName="food" name="food">
    `);
        }
    }
    RadioControlValueAccessor.ɵfac = function RadioControlValueAccessor_Factory(t) { return new (t || RadioControlValueAccessor)(ɵɵdirectiveInject(Renderer2), ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(RadioControlRegistry), ɵɵdirectiveInject(Injector)); };
    RadioControlValueAccessor.ɵdir = ɵɵdefineDirective({ type: RadioControlValueAccessor, selectors: [["input", "type", "radio", "formControlName", ""], ["input", "type", "radio", "formControl", ""], ["input", "type", "radio", "ngModel", ""]], hostBindings: function RadioControlValueAccessor_HostBindings(rf, ctx) { if (rf & 1) {
            ɵɵlistener("change", function RadioControlValueAccessor_change_HostBindingHandler() { return ctx.onChange(); })("blur", function RadioControlValueAccessor_blur_HostBindingHandler() { return ctx.onTouched(); });
        } }, inputs: { name: "name", formControlName: "formControlName", value: "value" }, features: [ɵɵProvidersFeature([RADIO_VALUE_ACCESSOR])] });
    /** @nocollapse */
    RadioControlValueAccessor.ctorParameters = () => [
        { type: Renderer2 },
        { type: ElementRef },
        { type: RadioControlRegistry },
        { type: Injector }
    ];
    RadioControlValueAccessor.propDecorators = {
        name: [{ type: Input }],
        formControlName: [{ type: Input }],
        value: [{ type: Input }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/forms/src/directives/range_value_accessor.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const RANGE_VALUE_ACCESSOR = {
        provide: NG_VALUE_ACCESSOR,
        useExisting: forwardRef((/**
         * @return {?}
         */
        () => RangeValueAccessor)),
        multi: true
    };
    /**
     * \@description
     * The `ControlValueAccessor` for writing a range value and listening to range input changes.
     * The value accessor is used by the `FormControlDirective`, `FormControlName`, and  `NgModel`
     * directives.
     *
     * \@usageNotes
     *
     * ### Using a range input with a reactive form
     *
     * The following example shows how to use a range input with a reactive form.
     *
     * ```ts
     * const ageControl = new FormControl();
     * ```
     *
     * ```
     * <input type="range" [formControl]="ageControl">
     * ```
     *
     * \@ngModule ReactiveFormsModule
     * \@ngModule FormsModule
     * \@publicApi
     */
    class RangeValueAccessor {
        /**
         * @param {?} _renderer
         * @param {?} _elementRef
         */
        constructor(_renderer, _elementRef) {
            this._renderer = _renderer;
            this._elementRef = _elementRef;
            /**
             * \@description
             * The registered callback function called when a change or input event occurs on the input
             * element.
             */
            this.onChange = (/**
             * @param {?} _
             * @return {?}
             */
            (_) => { });
            /**
             * \@description
             * The registered callback function called when a blur event occurs on the input element.
             */
            this.onTouched = (/**
             * @return {?}
             */
            () => { });
        }
        /**
         * Sets the "value" property on the input element.
         *
         * @param {?} value The checked value
         * @return {?}
         */
        writeValue(value) {
            this._renderer.setProperty(this._elementRef.nativeElement, 'value', parseFloat(value));
        }
        /**
         * \@description
         * Registers a function called when the control value changes.
         *
         * @param {?} fn The callback function
         * @return {?}
         */
        registerOnChange(fn) {
            this.onChange = (/**
             * @param {?} value
             * @return {?}
             */
            (value) => {
                fn(value == '' ? null : parseFloat(value));
            });
        }
        /**
         * \@description
         * Registers a function called when the control is touched.
         *
         * @param {?} fn The callback function
         * @return {?}
         */
        registerOnTouched(fn) {
            this.onTouched = fn;
        }
        /**
         * Sets the "disabled" property on the range input element.
         *
         * @param {?} isDisabled The disabled value
         * @return {?}
         */
        setDisabledState(isDisabled) {
            this._renderer.setProperty(this._elementRef.nativeElement, 'disabled', isDisabled);
        }
    }
    RangeValueAccessor.ɵfac = function RangeValueAccessor_Factory(t) { return new (t || RangeValueAccessor)(ɵɵdirectiveInject(Renderer2), ɵɵdirectiveInject(ElementRef)); };
    RangeValueAccessor.ɵdir = ɵɵdefineDirective({ type: RangeValueAccessor, selectors: [["input", "type", "range", "formControlName", ""], ["input", "type", "range", "formControl", ""], ["input", "type", "range", "ngModel", ""]], hostBindings: function RangeValueAccessor_HostBindings(rf, ctx) { if (rf & 1) {
            ɵɵlistener("change", function RangeValueAccessor_change_HostBindingHandler($event) { return ctx.onChange($event.target.value); })("input", function RangeValueAccessor_input_HostBindingHandler($event) { return ctx.onChange($event.target.value); })("blur", function RangeValueAccessor_blur_HostBindingHandler() { return ctx.onTouched(); });
        } }, features: [ɵɵProvidersFeature([RANGE_VALUE_ACCESSOR])] });
    /** @nocollapse */
    RangeValueAccessor.ctorParameters = () => [
        { type: Renderer2 },
        { type: ElementRef }
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/forms/src/directives/error_examples.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google Inc. All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /** @type {?} */
    const FormErrorExamples = {
        formControlName: `
    <div [formGroup]="myGroup">
      <input formControlName="firstName">
    </div>

    In your class:

    this.myGroup = new FormGroup({
       firstName: new FormControl()
    });`,
        formGroupName: `
    <div [formGroup]="myGroup">
       <div formGroupName="person">
          <input formControlName="firstName">
       </div>
    </div>

    In your class:

    this.myGroup = new FormGroup({
       person: new FormGroup({ firstName: new FormControl() })
    });`,
        formArrayName: `
    <div [formGroup]="myGroup">
      <div formArrayName="cities">
        <div *ngFor="let city of cityArray.controls; index as i">
          <input [formControlName]="i">
        </div>
      </div>
    </div>

    In your class:

    this.cityArray = new FormArray([new FormControl('SF')]);
    this.myGroup = new FormGroup({
      cities: this.cityArray
    });`,
        ngModelGroup: `
    <form>
       <div ngModelGroup="person">
          <input [(ngModel)]="person.name" name="firstName">
       </div>
    </form>`,
        ngModelWithFormGroup: `
    <div [formGroup]="myGroup">
       <input formControlName="firstName">
       <input [(ngModel)]="showMoreControls" [ngModelOptions]="{standalone: true}">
    </div>
  `
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/forms/src/directives/reactive_errors.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class ReactiveErrors {
        /**
         * @return {?}
         */
        static controlParentException() {
            throw new Error(`formControlName must be used with a parent formGroup directive.  You'll want to add a formGroup
       directive and pass it an existing FormGroup instance (you can create one in your class).

      Example:

      ${FormErrorExamples.formControlName}`);
        }
        /**
         * @return {?}
         */
        static ngModelGroupException() {
            throw new Error(`formControlName cannot be used with an ngModelGroup parent. It is only compatible with parents
       that also have a "form" prefix: formGroupName, formArrayName, or formGroup.

       Option 1:  Update the parent to be formGroupName (reactive form strategy)

        ${FormErrorExamples.formGroupName}

        Option 2: Use ngModel instead of formControlName (template-driven strategy)

        ${FormErrorExamples.ngModelGroup}`);
        }
        /**
         * @return {?}
         */
        static missingFormException() {
            throw new Error(`formGroup expects a FormGroup instance. Please pass one in.

       Example:

       ${FormErrorExamples.formControlName}`);
        }
        /**
         * @return {?}
         */
        static groupParentException() {
            throw new Error(`formGroupName must be used with a parent formGroup directive.  You'll want to add a formGroup
      directive and pass it an existing FormGroup instance (you can create one in your class).

      Example:

      ${FormErrorExamples.formGroupName}`);
        }
        /**
         * @return {?}
         */
        static arrayParentException() {
            throw new Error(`formArrayName must be used with a parent formGroup directive.  You'll want to add a formGroup
       directive and pass it an existing FormGroup instance (you can create one in your class).

        Example:

        ${FormErrorExamples.formArrayName}`);
        }
        /**
         * @return {?}
         */
        static disabledAttrWarning() {
            console.warn(`
      It looks like you're using the disabled attribute with a reactive form directive. If you set disabled to true
      when you set up this control in your component class, the disabled attribute will actually be set in the DOM for
      you. We recommend using this approach to avoid 'changed after checked' errors.
       
      Example: 
      form = new FormGroup({
        first: new FormControl({value: 'Nancy', disabled: true}, Validators.required),
        last: new FormControl('Drew', Validators.required)
      });
    `);
        }
        /**
         * @param {?} directiveName
         * @return {?}
         */
        static ngModelWarning(directiveName) {
            console.warn(`
    It looks like you're using ngModel on the same form field as ${directiveName}. 
    Support for using the ngModel input property and ngModelChange event with 
    reactive form directives has been deprecated in Angular v6 and will be removed 
    in Angular v7.
    
    For more information on this, see our API docs here:
    https://angular.io/api/forms/${directiveName === 'formControl' ? 'FormControlDirective' :
            'FormControlName'}#use-with-ngmodel
    `);
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/forms/src/directives/select_control_value_accessor.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const SELECT_VALUE_ACCESSOR = {
        provide: NG_VALUE_ACCESSOR,
        useExisting: forwardRef((/**
         * @return {?}
         */
        () => SelectControlValueAccessor)),
        multi: true
    };
    /**
     * @param {?} id
     * @param {?} value
     * @return {?}
     */
    function _buildValueString(id, value) {
        if (id == null)
            return `${value}`;
        if (value && typeof value === 'object')
            value = 'Object';
        return `${id}: ${value}`.slice(0, 50);
    }
    /**
     * @param {?} valueString
     * @return {?}
     */
    function _extractId(valueString) {
        return valueString.split(':')[0];
    }
    /**
     * \@description
     * The `ControlValueAccessor` for writing select control values and listening to select control
     * changes. The value accessor is used by the `FormControlDirective`, `FormControlName`, and
     * `NgModel` directives.
     *
     * \@usageNotes
     *
     * ### Using select controls in a reactive form
     *
     * The following examples show how to use a select control in a reactive form.
     *
     * {\@example forms/ts/reactiveSelectControl/reactive_select_control_example.ts region='Component'}
     *
     * ### Using select controls in a template-driven form
     *
     * To use a select in a template-driven form, simply add an `ngModel` and a `name`
     * attribute to the main `<select>` tag.
     *
     * {\@example forms/ts/selectControl/select_control_example.ts region='Component'}
     *
     * ### Customizing option selection
     *
     * Angular uses object identity to select option. It's possible for the identities of items
     * to change while the data does not. This can happen, for example, if the items are produced
     * from an RPC to the server, and that RPC is re-run. Even if the data hasn't changed, the
     * second response will produce objects with different identities.
     *
     * To customize the default option comparison algorithm, `<select>` supports `compareWith` input.
     * `compareWith` takes a **function** which has two arguments: `option1` and `option2`.
     * If `compareWith` is given, Angular selects option by the return value of the function.
     *
     * ```ts
     * const selectedCountriesControl = new FormControl();
     * ```
     *
     * ```
     * <select [compareWith]="compareFn"  [formControl]="selectedCountriesControl">
     *     <option *ngFor="let country of countries" [ngValue]="country">
     *         {{country.name}}
     *     </option>
     * </select>
     *
     * compareFn(c1: Country, c2: Country): boolean {
     *     return c1 && c2 ? c1.id === c2.id : c1 === c2;
     * }
     * ```
     *
     * **Note:** We listen to the 'change' event because 'input' events aren't fired
     * for selects in Firefox and IE:
     * https://bugzilla.mozilla.org/show_bug.cgi?id=1024350
     * https://developer.microsoft.com/en-us/microsoft-edge/platform/issues/4660045/
     *
     * \@ngModule ReactiveFormsModule
     * \@ngModule FormsModule
     * \@publicApi
     */
    class SelectControlValueAccessor {
        /**
         * @param {?} _renderer
         * @param {?} _elementRef
         */
        constructor(_renderer, _elementRef) {
            this._renderer = _renderer;
            this._elementRef = _elementRef;
            /**
             * \@internal
             */
            this._optionMap = new Map();
            /**
             * \@internal
             */
            this._idCounter = 0;
            /**
             * \@description
             * The registered callback function called when a change event occurs on the input element.
             */
            this.onChange = (/**
             * @param {?} _
             * @return {?}
             */
            (_) => { });
            /**
             * \@description
             * The registered callback function called when a blur event occurs on the input element.
             */
            this.onTouched = (/**
             * @return {?}
             */
            () => { });
            this._compareWith = looseIdentical;
        }
        /**
         * \@description
         * Tracks the option comparison algorithm for tracking identities when
         * checking for changes.
         * @param {?} fn
         * @return {?}
         */
        set compareWith(fn) {
            if (typeof fn !== 'function') {
                throw new Error(`compareWith must be a function, but received ${JSON.stringify(fn)}`);
            }
            this._compareWith = fn;
        }
        /**
         * Sets the "value" property on the input element. The "selectedIndex"
         * property is also set if an ID is provided on the option element.
         *
         * @param {?} value The checked value
         * @return {?}
         */
        writeValue(value) {
            this.value = value;
            /** @type {?} */
            const id = this._getOptionId(value);
            if (id == null) {
                this._renderer.setProperty(this._elementRef.nativeElement, 'selectedIndex', -1);
            }
            /** @type {?} */
            const valueString = _buildValueString(id, value);
            this._renderer.setProperty(this._elementRef.nativeElement, 'value', valueString);
        }
        /**
         * \@description
         * Registers a function called when the control value changes.
         *
         * @param {?} fn The callback function
         * @return {?}
         */
        registerOnChange(fn) {
            this.onChange = (/**
             * @param {?} valueString
             * @return {?}
             */
            (valueString) => {
                this.value = this._getOptionValue(valueString);
                fn(this.value);
            });
        }
        /**
         * \@description
         * Registers a function called when the control is touched.
         *
         * @param {?} fn The callback function
         * @return {?}
         */
        registerOnTouched(fn) {
            this.onTouched = fn;
        }
        /**
         * Sets the "disabled" property on the select input element.
         *
         * @param {?} isDisabled The disabled value
         * @return {?}
         */
        setDisabledState(isDisabled) {
            this._renderer.setProperty(this._elementRef.nativeElement, 'disabled', isDisabled);
        }
        /**
         * \@internal
         * @return {?}
         */
        _registerOption() {
            return (this._idCounter++).toString();
        }
        /**
         * \@internal
         * @param {?} value
         * @return {?}
         */
        _getOptionId(value) {
            for (const id of Array.from(this._optionMap.keys())) {
                if (this._compareWith(this._optionMap.get(id), value))
                    return id;
            }
            return null;
        }
        /**
         * \@internal
         * @param {?} valueString
         * @return {?}
         */
        _getOptionValue(valueString) {
            /** @type {?} */
            const id = _extractId(valueString);
            return this._optionMap.has(id) ? this._optionMap.get(id) : valueString;
        }
    }
    SelectControlValueAccessor.ɵfac = function SelectControlValueAccessor_Factory(t) { return new (t || SelectControlValueAccessor)(ɵɵdirectiveInject(Renderer2), ɵɵdirectiveInject(ElementRef)); };
    SelectControlValueAccessor.ɵdir = ɵɵdefineDirective({ type: SelectControlValueAccessor, selectors: [["select", "formControlName", "", 3, "multiple", ""], ["select", "formControl", "", 3, "multiple", ""], ["select", "ngModel", "", 3, "multiple", ""]], hostBindings: function SelectControlValueAccessor_HostBindings(rf, ctx) { if (rf & 1) {
            ɵɵlistener("change", function SelectControlValueAccessor_change_HostBindingHandler($event) { return ctx.onChange($event.target.value); })("blur", function SelectControlValueAccessor_blur_HostBindingHandler() { return ctx.onTouched(); });
        } }, inputs: { compareWith: "compareWith" }, features: [ɵɵProvidersFeature([SELECT_VALUE_ACCESSOR])] });
    /** @nocollapse */
    SelectControlValueAccessor.ctorParameters = () => [
        { type: Renderer2 },
        { type: ElementRef }
    ];
    SelectControlValueAccessor.propDecorators = {
        compareWith: [{ type: Input }]
    };
    /**
     * \@description
     * Marks `<option>` as dynamic, so Angular can be notified when options change.
     *
     * @see `SelectControlValueAccessor`
     *
     * \@ngModule ReactiveFormsModule
     * \@ngModule FormsModule
     * \@publicApi
     */
    class NgSelectOption {
        /**
         * @param {?} _element
         * @param {?} _renderer
         * @param {?} _select
         */
        constructor(_element, _renderer, _select) {
            this._element = _element;
            this._renderer = _renderer;
            this._select = _select;
            if (this._select)
                this.id = this._select._registerOption();
        }
        /**
         * \@description
         * Tracks the value bound to the option element. Unlike the value binding,
         * ngValue supports binding to objects.
         * @param {?} value
         * @return {?}
         */
        set ngValue(value) {
            if (this._select == null)
                return;
            this._select._optionMap.set(this.id, value);
            this._setElementValue(_buildValueString(this.id, value));
            this._select.writeValue(this._select.value);
        }
        /**
         * \@description
         * Tracks simple string values bound to the option element.
         * For objects, use the `ngValue` input binding.
         * @param {?} value
         * @return {?}
         */
        set value(value) {
            this._setElementValue(value);
            if (this._select)
                this._select.writeValue(this._select.value);
        }
        /**
         * \@internal
         * @param {?} value
         * @return {?}
         */
        _setElementValue(value) {
            this._renderer.setProperty(this._element.nativeElement, 'value', value);
        }
        /**
         * \@description
         * Lifecycle method called before the directive's instance is destroyed. For internal use only.
         * @return {?}
         */
        ngOnDestroy() {
            if (this._select) {
                this._select._optionMap.delete(this.id);
                this._select.writeValue(this._select.value);
            }
        }
    }
    NgSelectOption.ɵfac = function NgSelectOption_Factory(t) { return new (t || NgSelectOption)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(Renderer2), ɵɵdirectiveInject(SelectControlValueAccessor, 9)); };
    NgSelectOption.ɵdir = ɵɵdefineDirective({ type: NgSelectOption, selectors: [["option"]], inputs: { ngValue: "ngValue", value: "value" } });
    /** @nocollapse */
    NgSelectOption.ctorParameters = () => [
        { type: ElementRef },
        { type: Renderer2 },
        { type: SelectControlValueAccessor, decorators: [{ type: Optional }, { type: Host }] }
    ];
    NgSelectOption.propDecorators = {
        ngValue: [{ type: Input, args: ['ngValue',] }],
        value: [{ type: Input, args: ['value',] }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/forms/src/directives/select_multiple_control_value_accessor.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const SELECT_MULTIPLE_VALUE_ACCESSOR = {
        provide: NG_VALUE_ACCESSOR,
        useExisting: forwardRef((/**
         * @return {?}
         */
        () => SelectMultipleControlValueAccessor)),
        multi: true
    };
    /**
     * @param {?} id
     * @param {?} value
     * @return {?}
     */
    function _buildValueString$1(id, value) {
        if (id == null)
            return `${value}`;
        if (typeof value === 'string')
            value = `'${value}'`;
        if (value && typeof value === 'object')
            value = 'Object';
        return `${id}: ${value}`.slice(0, 50);
    }
    /**
     * @param {?} valueString
     * @return {?}
     */
    function _extractId$1(valueString) {
        return valueString.split(':')[0];
    }
    /**
     * \@description
     * The `ControlValueAccessor` for writing multi-select control values and listening to multi-select
     * control changes. The value accessor is used by the `FormControlDirective`, `FormControlName`, and
     * `NgModel` directives.
     *
     * @see `SelectControlValueAccessor`
     *
     * \@usageNotes
     *
     * ### Using a multi-select control
     *
     * The follow example shows you how to use a multi-select control with a reactive form.
     *
     * ```ts
     * const countryControl = new FormControl();
     * ```
     *
     * ```
     * <select multiple name="countries" [formControl]="countryControl">
     *   <option *ngFor="let country of countries" [ngValue]="country">
     *     {{ country.name }}
     *   </option>
     * </select>
     * ```
     *
     * ### Customizing option selection
     *
     * To customize the default option comparison algorithm, `<select>` supports `compareWith` input.
     * See the `SelectControlValueAccessor` for usage.
     *
     * \@ngModule ReactiveFormsModule
     * \@ngModule FormsModule
     * \@publicApi
     */
    class SelectMultipleControlValueAccessor {
        /**
         * @param {?} _renderer
         * @param {?} _elementRef
         */
        constructor(_renderer, _elementRef) {
            this._renderer = _renderer;
            this._elementRef = _elementRef;
            /**
             * \@internal
             */
            this._optionMap = new Map();
            /**
             * \@internal
             */
            this._idCounter = 0;
            /**
             * \@description
             * The registered callback function called when a change event occurs on the input element.
             */
            this.onChange = (/**
             * @param {?} _
             * @return {?}
             */
            (_) => { });
            /**
             * \@description
             * The registered callback function called when a blur event occurs on the input element.
             */
            this.onTouched = (/**
             * @return {?}
             */
            () => { });
            this._compareWith = looseIdentical;
        }
        /**
         * \@description
         * Tracks the option comparison algorithm for tracking identities when
         * checking for changes.
         * @param {?} fn
         * @return {?}
         */
        set compareWith(fn) {
            if (typeof fn !== 'function') {
                throw new Error(`compareWith must be a function, but received ${JSON.stringify(fn)}`);
            }
            this._compareWith = fn;
        }
        /**
         * \@description
         * Sets the "value" property on one or of more
         * of the select's options.
         *
         * @param {?} value The value
         * @return {?}
         */
        writeValue(value) {
            this.value = value;
            /** @type {?} */
            let optionSelectedStateSetter;
            if (Array.isArray(value)) {
                // convert values to ids
                /** @type {?} */
                const ids = value.map((/**
                 * @param {?} v
                 * @return {?}
                 */
                (v) => this._getOptionId(v)));
                optionSelectedStateSetter = (/**
                 * @param {?} opt
                 * @param {?} o
                 * @return {?}
                 */
                (opt, o) => {
                    opt._setSelected(ids.indexOf(o.toString()) > -1);
                });
            }
            else {
                optionSelectedStateSetter = (/**
                 * @param {?} opt
                 * @param {?} o
                 * @return {?}
                 */
                (opt, o) => {
                    opt._setSelected(false);
                });
            }
            this._optionMap.forEach(optionSelectedStateSetter);
        }
        /**
         * \@description
         * Registers a function called when the control value changes
         * and writes an array of the selected options.
         *
         * @param {?} fn The callback function
         * @return {?}
         */
        registerOnChange(fn) {
            this.onChange = (/**
             * @param {?} _
             * @return {?}
             */
            (_) => {
                /** @type {?} */
                const selected = [];
                if (_.hasOwnProperty('selectedOptions')) {
                    /** @type {?} */
                    const options = _.selectedOptions;
                    for (let i = 0; i < options.length; i++) {
                        /** @type {?} */
                        const opt = options.item(i);
                        /** @type {?} */
                        const val = this._getOptionValue(opt.value);
                        selected.push(val);
                    }
                }
                // Degrade on IE
                else {
                    /** @type {?} */
                    const options = (/** @type {?} */ (_.options));
                    for (let i = 0; i < options.length; i++) {
                        /** @type {?} */
                        const opt = options.item(i);
                        if (opt.selected) {
                            /** @type {?} */
                            const val = this._getOptionValue(opt.value);
                            selected.push(val);
                        }
                    }
                }
                this.value = selected;
                fn(selected);
            });
        }
        /**
         * \@description
         * Registers a function called when the control is touched.
         *
         * @param {?} fn The callback function
         * @return {?}
         */
        registerOnTouched(fn) {
            this.onTouched = fn;
        }
        /**
         * Sets the "disabled" property on the select input element.
         *
         * @param {?} isDisabled The disabled value
         * @return {?}
         */
        setDisabledState(isDisabled) {
            this._renderer.setProperty(this._elementRef.nativeElement, 'disabled', isDisabled);
        }
        /**
         * \@internal
         * @param {?} value
         * @return {?}
         */
        _registerOption(value) {
            /** @type {?} */
            const id = (this._idCounter++).toString();
            this._optionMap.set(id, value);
            return id;
        }
        /**
         * \@internal
         * @param {?} value
         * @return {?}
         */
        _getOptionId(value) {
            for (const id of Array.from(this._optionMap.keys())) {
                if (this._compareWith((/** @type {?} */ (this._optionMap.get(id)))._value, value))
                    return id;
            }
            return null;
        }
        /**
         * \@internal
         * @param {?} valueString
         * @return {?}
         */
        _getOptionValue(valueString) {
            /** @type {?} */
            const id = _extractId$1(valueString);
            return this._optionMap.has(id) ? (/** @type {?} */ (this._optionMap.get(id)))._value : valueString;
        }
    }
    SelectMultipleControlValueAccessor.ɵfac = function SelectMultipleControlValueAccessor_Factory(t) { return new (t || SelectMultipleControlValueAccessor)(ɵɵdirectiveInject(Renderer2), ɵɵdirectiveInject(ElementRef)); };
    SelectMultipleControlValueAccessor.ɵdir = ɵɵdefineDirective({ type: SelectMultipleControlValueAccessor, selectors: [["select", "multiple", "", "formControlName", ""], ["select", "multiple", "", "formControl", ""], ["select", "multiple", "", "ngModel", ""]], hostBindings: function SelectMultipleControlValueAccessor_HostBindings(rf, ctx) { if (rf & 1) {
            ɵɵlistener("change", function SelectMultipleControlValueAccessor_change_HostBindingHandler($event) { return ctx.onChange($event.target); })("blur", function SelectMultipleControlValueAccessor_blur_HostBindingHandler() { return ctx.onTouched(); });
        } }, inputs: { compareWith: "compareWith" }, features: [ɵɵProvidersFeature([SELECT_MULTIPLE_VALUE_ACCESSOR])] });
    /** @nocollapse */
    SelectMultipleControlValueAccessor.ctorParameters = () => [
        { type: Renderer2 },
        { type: ElementRef }
    ];
    SelectMultipleControlValueAccessor.propDecorators = {
        compareWith: [{ type: Input }]
    };
    /**
     * \@description
     * Marks `<option>` as dynamic, so Angular can be notified when options change.
     *
     * @see `SelectMultipleControlValueAccessor`
     *
     * \@ngModule ReactiveFormsModule
     * \@ngModule FormsModule
     * \@publicApi
     */
    class ɵNgSelectMultipleOption {
        /**
         * @param {?} _element
         * @param {?} _renderer
         * @param {?} _select
         */
        constructor(_element, _renderer, _select) {
            this._element = _element;
            this._renderer = _renderer;
            this._select = _select;
            if (this._select) {
                this.id = this._select._registerOption(this);
            }
        }
        /**
         * \@description
         * Tracks the value bound to the option element. Unlike the value binding,
         * ngValue supports binding to objects.
         * @param {?} value
         * @return {?}
         */
        set ngValue(value) {
            if (this._select == null)
                return;
            this._value = value;
            this._setElementValue(_buildValueString$1(this.id, value));
            this._select.writeValue(this._select.value);
        }
        /**
         * \@description
         * Tracks simple string values bound to the option element.
         * For objects, use the `ngValue` input binding.
         * @param {?} value
         * @return {?}
         */
        set value(value) {
            if (this._select) {
                this._value = value;
                this._setElementValue(_buildValueString$1(this.id, value));
                this._select.writeValue(this._select.value);
            }
            else {
                this._setElementValue(value);
            }
        }
        /**
         * \@internal
         * @param {?} value
         * @return {?}
         */
        _setElementValue(value) {
            this._renderer.setProperty(this._element.nativeElement, 'value', value);
        }
        /**
         * \@internal
         * @param {?} selected
         * @return {?}
         */
        _setSelected(selected) {
            this._renderer.setProperty(this._element.nativeElement, 'selected', selected);
        }
        /**
         * \@description
         * Lifecycle method called before the directive's instance is destroyed. For internal use only.
         * @return {?}
         */
        ngOnDestroy() {
            if (this._select) {
                this._select._optionMap.delete(this.id);
                this._select.writeValue(this._select.value);
            }
        }
    }
    ɵNgSelectMultipleOption.ɵfac = function ɵNgSelectMultipleOption_Factory(t) { return new (t || ɵNgSelectMultipleOption)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(Renderer2), ɵɵdirectiveInject(SelectMultipleControlValueAccessor, 9)); };
    ɵNgSelectMultipleOption.ɵdir = ɵɵdefineDirective({ type: ɵNgSelectMultipleOption, selectors: [["option"]], inputs: { ngValue: "ngValue", value: "value" } });
    /** @nocollapse */
    ɵNgSelectMultipleOption.ctorParameters = () => [
        { type: ElementRef },
        { type: Renderer2 },
        { type: SelectMultipleControlValueAccessor, decorators: [{ type: Optional }, { type: Host }] }
    ];
    ɵNgSelectMultipleOption.propDecorators = {
        ngValue: [{ type: Input, args: ['ngValue',] }],
        value: [{ type: Input, args: ['value',] }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/forms/src/directives/shared.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @param {?} name
     * @param {?} parent
     * @return {?}
     */
    function controlPath(name, parent) {
        return [...(/** @type {?} */ (parent.path)), (/** @type {?} */ (name))];
    }
    /**
     * @param {?} control
     * @param {?} dir
     * @return {?}
     */
    function setUpControl(control, dir) {
        if (!control)
            _throwError(dir, 'Cannot find control with');
        if (!dir.valueAccessor)
            _throwError(dir, 'No value accessor for form control with');
        control.validator = Validators.compose([(/** @type {?} */ (control.validator)), dir.validator]);
        control.asyncValidator = Validators.composeAsync([(/** @type {?} */ (control.asyncValidator)), dir.asyncValidator]);
        (/** @type {?} */ (dir.valueAccessor)).writeValue(control.value);
        setUpViewChangePipeline(control, dir);
        setUpModelChangePipeline(control, dir);
        setUpBlurPipeline(control, dir);
        if ((/** @type {?} */ (dir.valueAccessor)).setDisabledState) {
            control.registerOnDisabledChange((/**
             * @param {?} isDisabled
             * @return {?}
             */
            (isDisabled) => {
                (/** @type {?} */ ((/** @type {?} */ (dir.valueAccessor)).setDisabledState))(isDisabled);
            }));
        }
        // re-run validation when validator binding changes, e.g. minlength=3 -> minlength=4
        dir._rawValidators.forEach((/**
         * @param {?} validator
         * @return {?}
         */
        (validator) => {
            if (((/** @type {?} */ (validator))).registerOnValidatorChange)
                (/** @type {?} */ (((/** @type {?} */ (validator))).registerOnValidatorChange))((/**
                 * @return {?}
                 */
                () => control.updateValueAndValidity()));
        }));
        dir._rawAsyncValidators.forEach((/**
         * @param {?} validator
         * @return {?}
         */
        (validator) => {
            if (((/** @type {?} */ (validator))).registerOnValidatorChange)
                (/** @type {?} */ (((/** @type {?} */ (validator))).registerOnValidatorChange))((/**
                 * @return {?}
                 */
                () => control.updateValueAndValidity()));
        }));
    }
    /**
     * @param {?} control
     * @param {?} dir
     * @return {?}
     */
    function cleanUpControl(control, dir) {
        (/** @type {?} */ (dir.valueAccessor)).registerOnChange((/**
         * @return {?}
         */
        () => _noControlError(dir)));
        (/** @type {?} */ (dir.valueAccessor)).registerOnTouched((/**
         * @return {?}
         */
        () => _noControlError(dir)));
        dir._rawValidators.forEach((/**
         * @param {?} validator
         * @return {?}
         */
        (validator) => {
            if (validator.registerOnValidatorChange) {
                validator.registerOnValidatorChange(null);
            }
        }));
        dir._rawAsyncValidators.forEach((/**
         * @param {?} validator
         * @return {?}
         */
        (validator) => {
            if (validator.registerOnValidatorChange) {
                validator.registerOnValidatorChange(null);
            }
        }));
        if (control)
            control._clearChangeFns();
    }
    /**
     * @param {?} control
     * @param {?} dir
     * @return {?}
     */
    function setUpViewChangePipeline(control, dir) {
        (/** @type {?} */ (dir.valueAccessor)).registerOnChange((/**
         * @param {?} newValue
         * @return {?}
         */
        (newValue) => {
            control._pendingValue = newValue;
            control._pendingChange = true;
            control._pendingDirty = true;
            if (control.updateOn === 'change')
                updateControl(control, dir);
        }));
    }
    /**
     * @param {?} control
     * @param {?} dir
     * @return {?}
     */
    function setUpBlurPipeline(control, dir) {
        (/** @type {?} */ (dir.valueAccessor)).registerOnTouched((/**
         * @return {?}
         */
        () => {
            control._pendingTouched = true;
            if (control.updateOn === 'blur' && control._pendingChange)
                updateControl(control, dir);
            if (control.updateOn !== 'submit')
                control.markAsTouched();
        }));
    }
    /**
     * @param {?} control
     * @param {?} dir
     * @return {?}
     */
    function updateControl(control, dir) {
        if (control._pendingDirty)
            control.markAsDirty();
        control.setValue(control._pendingValue, { emitModelToViewChange: false });
        dir.viewToModelUpdate(control._pendingValue);
        control._pendingChange = false;
    }
    /**
     * @param {?} control
     * @param {?} dir
     * @return {?}
     */
    function setUpModelChangePipeline(control, dir) {
        control.registerOnChange((/**
         * @param {?} newValue
         * @param {?} emitModelEvent
         * @return {?}
         */
        (newValue, emitModelEvent) => {
            // control -> view
            (/** @type {?} */ (dir.valueAccessor)).writeValue(newValue);
            // control -> ngModel
            if (emitModelEvent)
                dir.viewToModelUpdate(newValue);
        }));
    }
    /**
     * @param {?} control
     * @param {?} dir
     * @return {?}
     */
    function setUpFormContainer(control, dir) {
        if (control == null)
            _throwError(dir, 'Cannot find control with');
        control.validator = Validators.compose([control.validator, dir.validator]);
        control.asyncValidator = Validators.composeAsync([control.asyncValidator, dir.asyncValidator]);
    }
    /**
     * @param {?} dir
     * @return {?}
     */
    function _noControlError(dir) {
        return _throwError(dir, 'There is no FormControl instance attached to form control element with');
    }
    /**
     * @param {?} dir
     * @param {?} message
     * @return {?}
     */
    function _throwError(dir, message) {
        /** @type {?} */
        let messageEnd;
        if ((/** @type {?} */ (dir.path)).length > 1) {
            messageEnd = `path: '${(/** @type {?} */ (dir.path)).join(' -> ')}'`;
        }
        else if ((/** @type {?} */ (dir.path))[0]) {
            messageEnd = `name: '${dir.path}'`;
        }
        else {
            messageEnd = 'unspecified name attribute';
        }
        throw new Error(`${message} ${messageEnd}`);
    }
    /**
     * @param {?} validators
     * @return {?}
     */
    function composeValidators(validators) {
        return validators != null ? Validators.compose(validators.map(normalizeValidator)) : null;
    }
    /**
     * @param {?} validators
     * @return {?}
     */
    function composeAsyncValidators(validators) {
        return validators != null ? Validators.composeAsync(validators.map(normalizeAsyncValidator)) :
            null;
    }
    /**
     * @param {?} changes
     * @param {?} viewModel
     * @return {?}
     */
    function isPropertyUpdated(changes, viewModel) {
        if (!changes.hasOwnProperty('model'))
            return false;
        /** @type {?} */
        const change = changes['model'];
        if (change.isFirstChange())
            return true;
        return !looseIdentical(viewModel, change.currentValue);
    }
    /** @type {?} */
    const BUILTIN_ACCESSORS = [
        CheckboxControlValueAccessor,
        RangeValueAccessor,
        NumberValueAccessor,
        SelectControlValueAccessor,
        SelectMultipleControlValueAccessor,
        RadioControlValueAccessor,
    ];
    /**
     * @param {?} valueAccessor
     * @return {?}
     */
    function isBuiltInAccessor(valueAccessor) {
        return BUILTIN_ACCESSORS.some((/**
         * @param {?} a
         * @return {?}
         */
        a => valueAccessor.constructor === a));
    }
    /**
     * @param {?} form
     * @param {?} directives
     * @return {?}
     */
    function syncPendingControls(form, directives) {
        form._syncPendingControls();
        directives.forEach((/**
         * @param {?} dir
         * @return {?}
         */
        dir => {
            /** @type {?} */
            const control = (/** @type {?} */ (dir.control));
            if (control.updateOn === 'submit' && control._pendingChange) {
                dir.viewToModelUpdate(control._pendingValue);
                control._pendingChange = false;
            }
        }));
    }
    // TODO: vsavkin remove it once https://github.com/angular/angular/issues/3011 is implemented
    /**
     * @param {?} dir
     * @param {?} valueAccessors
     * @return {?}
     */
    function selectValueAccessor(dir, valueAccessors) {
        if (!valueAccessors)
            return null;
        if (!Array.isArray(valueAccessors))
            _throwError(dir, 'Value accessor was not provided as an array for form control with');
        /** @type {?} */
        let defaultAccessor = undefined;
        /** @type {?} */
        let builtinAccessor = undefined;
        /** @type {?} */
        let customAccessor = undefined;
        valueAccessors.forEach((/**
         * @param {?} v
         * @return {?}
         */
        (v) => {
            if (v.constructor === DefaultValueAccessor) {
                defaultAccessor = v;
            }
            else if (isBuiltInAccessor(v)) {
                if (builtinAccessor)
                    _throwError(dir, 'More than one built-in value accessor matches form control with');
                builtinAccessor = v;
            }
            else {
                if (customAccessor)
                    _throwError(dir, 'More than one custom value accessor matches form control with');
                customAccessor = v;
            }
        }));
        if (customAccessor)
            return customAccessor;
        if (builtinAccessor)
            return builtinAccessor;
        if (defaultAccessor)
            return defaultAccessor;
        _throwError(dir, 'No valid value accessor for form control with');
        return null;
    }
    /**
     * @template T
     * @param {?} list
     * @param {?} el
     * @return {?}
     */
    function removeDir(list, el) {
        /** @type {?} */
        const index = list.indexOf(el);
        if (index > -1)
            list.splice(index, 1);
    }
    // TODO(kara): remove after deprecation period
    /**
     * @param {?} name
     * @param {?} type
     * @param {?} instance
     * @param {?} warningConfig
     * @return {?}
     */
    function _ngModelWarning(name, type, instance, warningConfig) {
        if (!isDevMode() || warningConfig === 'never')
            return;
        if (((warningConfig === null || warningConfig === 'once') && !type._ngModelWarningSentOnce) ||
            (warningConfig === 'always' && !instance._ngModelWarningSent)) {
            ReactiveErrors.ngModelWarning(name);
            type._ngModelWarningSentOnce = true;
            instance._ngModelWarningSent = true;
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/forms/src/model.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Reports that a FormControl is valid, meaning that no errors exist in the input value.
     *
     * @see `status`
     * @type {?}
     */
    const VALID = 'VALID';
    /**
     * Reports that a FormControl is invalid, meaning that an error exists in the input value.
     *
     * @see `status`
     * @type {?}
     */
    const INVALID = 'INVALID';
    /**
     * Reports that a FormControl is pending, meaning that that async validation is occurring and
     * errors are not yet available for the input value.
     *
     * @see `markAsPending`
     * @see `status`
     * @type {?}
     */
    const PENDING = 'PENDING';
    /**
     * Reports that a FormControl is disabled, meaning that the control is exempt from ancestor
     * calculations of validity or value.
     *
     * @see `markAsDisabled`
     * @see `status`
     * @type {?}
     */
    const DISABLED = 'DISABLED';
    /**
     * @param {?} control
     * @param {?} path
     * @param {?} delimiter
     * @return {?}
     */
    function _find(control, path, delimiter) {
        if (path == null)
            return null;
        if (!Array.isArray(path)) {
            path = path.split(delimiter);
        }
        if (Array.isArray(path) && path.length === 0)
            return null;
        // Not using Array.reduce here due to a Chrome 80 bug
        // https://bugs.chromium.org/p/chromium/issues/detail?id=1049982
        /** @type {?} */
        let controlToFind = control;
        path.forEach((/**
         * @param {?} name
         * @return {?}
         */
        (name) => {
            if (controlToFind instanceof FormGroup) {
                controlToFind = controlToFind.controls.hasOwnProperty((/** @type {?} */ (name))) ?
                    controlToFind.controls[name] :
                    null;
            }
            else if (controlToFind instanceof FormArray) {
                controlToFind = controlToFind.at((/** @type {?} */ (name))) || null;
            }
            else {
                controlToFind = null;
            }
        }));
        return controlToFind;
    }
    /**
     * @param {?=} validatorOrOpts
     * @return {?}
     */
    function coerceToValidator(validatorOrOpts) {
        /** @type {?} */
        const validator = (/** @type {?} */ ((isOptionsObj(validatorOrOpts) ? ((/** @type {?} */ (validatorOrOpts))).validators :
            validatorOrOpts)));
        return Array.isArray(validator) ? composeValidators(validator) : validator || null;
    }
    /**
     * @param {?=} asyncValidator
     * @param {?=} validatorOrOpts
     * @return {?}
     */
    function coerceToAsyncValidator(asyncValidator, validatorOrOpts) {
        /** @type {?} */
        const origAsyncValidator = (/** @type {?} */ ((isOptionsObj(validatorOrOpts) ? ((/** @type {?} */ (validatorOrOpts))).asyncValidators :
            asyncValidator)));
        return Array.isArray(origAsyncValidator) ? composeAsyncValidators(origAsyncValidator) :
            origAsyncValidator || null;
    }
    /**
     * @param {?=} validatorOrOpts
     * @return {?}
     */
    function isOptionsObj(validatorOrOpts) {
        return validatorOrOpts != null && !Array.isArray(validatorOrOpts) &&
            typeof validatorOrOpts === 'object';
    }
    /**
     * This is the base class for `FormControl`, `FormGroup`, and `FormArray`.
     *
     * It provides some of the shared behavior that all controls and groups of controls have, like
     * running validators, calculating status, and resetting state. It also defines the properties
     * that are shared between all sub-classes, like `value`, `valid`, and `dirty`. It shouldn't be
     * instantiated directly.
     *
     * @see [Forms Guide](/guide/forms)
     * @see [Reactive Forms Guide](/guide/reactive-forms)
     * @see [Dynamic Forms Guide](/guide/dynamic-form)
     *
     * \@publicApi
     * @abstract
     */
    class AbstractControl {
        /**
         * Initialize the AbstractControl instance.
         *
         * @param {?} validator The function that determines the synchronous validity of this control.
         * @param {?} asyncValidator The function that determines the asynchronous validity of this
         * control.
         */
        constructor(validator, asyncValidator) {
            this.validator = validator;
            this.asyncValidator = asyncValidator;
            /**
             * \@internal
             */
            this._onCollectionChange = (/**
             * @return {?}
             */
            () => { });
            /**
             * A control is `pristine` if the user has not yet changed
             * the value in the UI.
             *
             * @return True if the user has not yet changed the value in the UI; compare `dirty`.
             * Programmatic changes to a control's value do not mark it dirty.
             */
            this.pristine = true;
            /**
             * True if the control is marked as `touched`.
             *
             * A control is marked `touched` once the user has triggered
             * a `blur` event on it.
             */
            this.touched = false;
            /**
             * \@internal
             */
            this._onDisabledChange = [];
        }
        /**
         * The parent control.
         * @return {?}
         */
        get parent() {
            return this._parent;
        }
        /**
         * A control is `valid` when its `status` is `VALID`.
         *
         * @see {\@link AbstractControl.status}
         *
         * @return {?} True if the control has passed all of its validation tests,
         * false otherwise.
         */
        get valid() {
            return this.status === VALID;
        }
        /**
         * A control is `invalid` when its `status` is `INVALID`.
         *
         * @see {\@link AbstractControl.status}
         *
         * @return {?} True if this control has failed one or more of its validation checks,
         * false otherwise.
         */
        get invalid() {
            return this.status === INVALID;
        }
        /**
         * A control is `pending` when its `status` is `PENDING`.
         *
         * @see {\@link AbstractControl.status}
         *
         * @return {?} True if this control is in the process of conducting a validation check,
         * false otherwise.
         */
        get pending() {
            return this.status == PENDING;
        }
        /**
         * A control is `disabled` when its `status` is `DISABLED`.
         *
         * Disabled controls are exempt from validation checks and
         * are not included in the aggregate value of their ancestor
         * controls.
         *
         * @see {\@link AbstractControl.status}
         *
         * @return {?} True if the control is disabled, false otherwise.
         */
        get disabled() {
            return this.status === DISABLED;
        }
        /**
         * A control is `enabled` as long as its `status` is not `DISABLED`.
         *
         * @see {\@link AbstractControl.status}
         *
         * @return {?} True if the control has any status other than 'DISABLED',
         * false if the status is 'DISABLED'.
         *
         */
        get enabled() {
            return this.status !== DISABLED;
        }
        /**
         * A control is `dirty` if the user has changed the value
         * in the UI.
         *
         * @return {?} True if the user has changed the value of this control in the UI; compare `pristine`.
         * Programmatic changes to a control's value do not mark it dirty.
         */
        get dirty() {
            return !this.pristine;
        }
        /**
         * True if the control has not been marked as touched
         *
         * A control is `untouched` if the user has not yet triggered
         * a `blur` event on it.
         * @return {?}
         */
        get untouched() {
            return !this.touched;
        }
        /**
         * Reports the update strategy of the `AbstractControl` (meaning
         * the event on which the control updates itself).
         * Possible values: `'change'` | `'blur'` | `'submit'`
         * Default value: `'change'`
         * @return {?}
         */
        get updateOn() {
            return this._updateOn ? this._updateOn : (this.parent ? this.parent.updateOn : 'change');
        }
        /**
         * Sets the synchronous validators that are active on this control.  Calling
         * this overwrites any existing sync validators.
         *
         * When you add or remove a validator at run time, you must call
         * `updateValueAndValidity()` for the new validation to take effect.
         *
         * @param {?} newValidator
         * @return {?}
         */
        setValidators(newValidator) {
            this.validator = coerceToValidator(newValidator);
        }
        /**
         * Sets the async validators that are active on this control. Calling this
         * overwrites any existing async validators.
         *
         * When you add or remove a validator at run time, you must call
         * `updateValueAndValidity()` for the new validation to take effect.
         *
         * @param {?} newValidator
         * @return {?}
         */
        setAsyncValidators(newValidator) {
            this.asyncValidator = coerceToAsyncValidator(newValidator);
        }
        /**
         * Empties out the sync validator list.
         *
         * When you add or remove a validator at run time, you must call
         * `updateValueAndValidity()` for the new validation to take effect.
         *
         * @return {?}
         */
        clearValidators() {
            this.validator = null;
        }
        /**
         * Empties out the async validator list.
         *
         * When you add or remove a validator at run time, you must call
         * `updateValueAndValidity()` for the new validation to take effect.
         *
         * @return {?}
         */
        clearAsyncValidators() {
            this.asyncValidator = null;
        }
        /**
         * Marks the control as `touched`. A control is touched by focus and
         * blur events that do not change the value.
         *
         * @see `markAsUntouched()` / `markAsDirty()` / `markAsPristine()`
         *
         * @param {?=} opts Configuration options that determine how the control propagates changes
         * and emits events after marking is applied.
         * * `onlySelf`: When true, mark only this control. When false or not supplied,
         * marks all direct ancestors. Default is false.
         * @return {?}
         */
        markAsTouched(opts = {}) {
            ((/** @type {?} */ (this))).touched = true;
            if (this._parent && !opts.onlySelf) {
                this._parent.markAsTouched(opts);
            }
        }
        /**
         * Marks the control and all its descendant controls as `touched`.
         * @see `markAsTouched()`
         * @return {?}
         */
        markAllAsTouched() {
            this.markAsTouched({ onlySelf: true });
            this._forEachChild((/**
             * @param {?} control
             * @return {?}
             */
            (control) => control.markAllAsTouched()));
        }
        /**
         * Marks the control as `untouched`.
         *
         * If the control has any children, also marks all children as `untouched`
         * and recalculates the `touched` status of all parent controls.
         *
         * @see `markAsTouched()` / `markAsDirty()` / `markAsPristine()`
         *
         * @param {?=} opts Configuration options that determine how the control propagates changes
         * and emits events after the marking is applied.
         * * `onlySelf`: When true, mark only this control. When false or not supplied,
         * marks all direct ancestors. Default is false.
         * @return {?}
         */
        markAsUntouched(opts = {}) {
            ((/** @type {?} */ (this))).touched = false;
            this._pendingTouched = false;
            this._forEachChild((/**
             * @param {?} control
             * @return {?}
             */
            (control) => {
                control.markAsUntouched({ onlySelf: true });
            }));
            if (this._parent && !opts.onlySelf) {
                this._parent._updateTouched(opts);
            }
        }
        /**
         * Marks the control as `dirty`. A control becomes dirty when
         * the control's value is changed through the UI; compare `markAsTouched`.
         *
         * @see `markAsTouched()` / `markAsUntouched()` / `markAsPristine()`
         *
         * @param {?=} opts Configuration options that determine how the control propagates changes
         * and emits events after marking is applied.
         * * `onlySelf`: When true, mark only this control. When false or not supplied,
         * marks all direct ancestors. Default is false.
         * @return {?}
         */
        markAsDirty(opts = {}) {
            ((/** @type {?} */ (this))).pristine = false;
            if (this._parent && !opts.onlySelf) {
                this._parent.markAsDirty(opts);
            }
        }
        /**
         * Marks the control as `pristine`.
         *
         * If the control has any children, marks all children as `pristine`,
         * and recalculates the `pristine` status of all parent
         * controls.
         *
         * @see `markAsTouched()` / `markAsUntouched()` / `markAsDirty()`
         *
         * @param {?=} opts Configuration options that determine how the control emits events after
         * marking is applied.
         * * `onlySelf`: When true, mark only this control. When false or not supplied,
         * marks all direct ancestors. Default is false.
         * @return {?}
         */
        markAsPristine(opts = {}) {
            ((/** @type {?} */ (this))).pristine = true;
            this._pendingDirty = false;
            this._forEachChild((/**
             * @param {?} control
             * @return {?}
             */
            (control) => {
                control.markAsPristine({ onlySelf: true });
            }));
            if (this._parent && !opts.onlySelf) {
                this._parent._updatePristine(opts);
            }
        }
        /**
         * Marks the control as `pending`.
         *
         * A control is pending while the control performs async validation.
         *
         * @see {\@link AbstractControl.status}
         *
         * @param {?=} opts Configuration options that determine how the control propagates changes and
         * emits events after marking is applied.
         * * `onlySelf`: When true, mark only this control. When false or not supplied,
         * marks all direct ancestors. Default is false.
         * * `emitEvent`: When true or not supplied (the default), the `statusChanges`
         * observable emits an event with the latest status the control is marked pending.
         * When false, no events are emitted.
         *
         * @return {?}
         */
        markAsPending(opts = {}) {
            ((/** @type {?} */ (this))).status = PENDING;
            if (opts.emitEvent !== false) {
                ((/** @type {?} */ (this.statusChanges))).emit(this.status);
            }
            if (this._parent && !opts.onlySelf) {
                this._parent.markAsPending(opts);
            }
        }
        /**
         * Disables the control. This means the control is exempt from validation checks and
         * excluded from the aggregate value of any parent. Its status is `DISABLED`.
         *
         * If the control has children, all children are also disabled.
         *
         * @see {\@link AbstractControl.status}
         *
         * @param {?=} opts Configuration options that determine how the control propagates
         * changes and emits events after the control is disabled.
         * * `onlySelf`: When true, mark only this control. When false or not supplied,
         * marks all direct ancestors. Default is false.
         * * `emitEvent`: When true or not supplied (the default), both the `statusChanges` and
         * `valueChanges`
         * observables emit events with the latest status and value when the control is disabled.
         * When false, no events are emitted.
         * @return {?}
         */
        disable(opts = {}) {
            // If parent has been marked artificially dirty we don't want to re-calculate the
            // parent's dirtiness based on the children.
            /** @type {?} */
            const skipPristineCheck = this._parentMarkedDirty(opts.onlySelf);
            ((/** @type {?} */ (this))).status = DISABLED;
            ((/** @type {?} */ (this))).errors = null;
            this._forEachChild((/**
             * @param {?} control
             * @return {?}
             */
            (control) => {
                control.disable(Object.assign(Object.assign({}, opts), { onlySelf: true }));
            }));
            this._updateValue();
            if (opts.emitEvent !== false) {
                ((/** @type {?} */ (this.valueChanges))).emit(this.value);
                ((/** @type {?} */ (this.statusChanges))).emit(this.status);
            }
            this._updateAncestors(Object.assign(Object.assign({}, opts), { skipPristineCheck }));
            this._onDisabledChange.forEach((/**
             * @param {?} changeFn
             * @return {?}
             */
            (changeFn) => changeFn(true)));
        }
        /**
         * Enables the control. This means the control is included in validation checks and
         * the aggregate value of its parent. Its status recalculates based on its value and
         * its validators.
         *
         * By default, if the control has children, all children are enabled.
         *
         * @see {\@link AbstractControl.status}
         *
         * @param {?=} opts Configure options that control how the control propagates changes and
         * emits events when marked as untouched
         * * `onlySelf`: When true, mark only this control. When false or not supplied,
         * marks all direct ancestors. Default is false.
         * * `emitEvent`: When true or not supplied (the default), both the `statusChanges` and
         * `valueChanges`
         * observables emit events with the latest status and value when the control is enabled.
         * When false, no events are emitted.
         * @return {?}
         */
        enable(opts = {}) {
            // If parent has been marked artificially dirty we don't want to re-calculate the
            // parent's dirtiness based on the children.
            /** @type {?} */
            const skipPristineCheck = this._parentMarkedDirty(opts.onlySelf);
            ((/** @type {?} */ (this))).status = VALID;
            this._forEachChild((/**
             * @param {?} control
             * @return {?}
             */
            (control) => {
                control.enable(Object.assign(Object.assign({}, opts), { onlySelf: true }));
            }));
            this.updateValueAndValidity({ onlySelf: true, emitEvent: opts.emitEvent });
            this._updateAncestors(Object.assign(Object.assign({}, opts), { skipPristineCheck }));
            this._onDisabledChange.forEach((/**
             * @param {?} changeFn
             * @return {?}
             */
            (changeFn) => changeFn(false)));
        }
        /**
         * @private
         * @param {?} opts
         * @return {?}
         */
        _updateAncestors(opts) {
            if (this._parent && !opts.onlySelf) {
                this._parent.updateValueAndValidity(opts);
                if (!opts.skipPristineCheck) {
                    this._parent._updatePristine();
                }
                this._parent._updateTouched();
            }
        }
        /**
         * @param {?} parent Sets the parent of the control
         * @return {?}
         */
        setParent(parent) {
            this._parent = parent;
        }
        /**
         * Recalculates the value and validation status of the control.
         *
         * By default, it also updates the value and validity of its ancestors.
         *
         * @param {?=} opts Configuration options determine how the control propagates changes and emits events
         * after updates and validity checks are applied.
         * * `onlySelf`: When true, only update this control. When false or not supplied,
         * update all direct ancestors. Default is false.
         * * `emitEvent`: When true or not supplied (the default), both the `statusChanges` and
         * `valueChanges`
         * observables emit events with the latest status and value when the control is updated.
         * When false, no events are emitted.
         * @return {?}
         */
        updateValueAndValidity(opts = {}) {
            this._setInitialStatus();
            this._updateValue();
            if (this.enabled) {
                this._cancelExistingSubscription();
                ((/** @type {?} */ (this))).errors = this._runValidator();
                ((/** @type {?} */ (this))).status = this._calculateStatus();
                if (this.status === VALID || this.status === PENDING) {
                    this._runAsyncValidator(opts.emitEvent);
                }
            }
            if (opts.emitEvent !== false) {
                ((/** @type {?} */ (this.valueChanges))).emit(this.value);
                ((/** @type {?} */ (this.statusChanges))).emit(this.status);
            }
            if (this._parent && !opts.onlySelf) {
                this._parent.updateValueAndValidity(opts);
            }
        }
        /**
         * \@internal
         * @param {?=} opts
         * @return {?}
         */
        _updateTreeValidity(opts = { emitEvent: true }) {
            this._forEachChild((/**
             * @param {?} ctrl
             * @return {?}
             */
            (ctrl) => ctrl._updateTreeValidity(opts)));
            this.updateValueAndValidity({ onlySelf: true, emitEvent: opts.emitEvent });
        }
        /**
         * @private
         * @return {?}
         */
        _setInitialStatus() {
            ((/** @type {?} */ (this))).status = this._allControlsDisabled() ? DISABLED : VALID;
        }
        /**
         * @private
         * @return {?}
         */
        _runValidator() {
            return this.validator ? this.validator(this) : null;
        }
        /**
         * @private
         * @param {?=} emitEvent
         * @return {?}
         */
        _runAsyncValidator(emitEvent) {
            if (this.asyncValidator) {
                ((/** @type {?} */ (this))).status = PENDING;
                /** @type {?} */
                const obs = toObservable(this.asyncValidator(this));
                this._asyncValidationSubscription =
                    obs.subscribe((/**
                     * @param {?} errors
                     * @return {?}
                     */
                    (errors) => this.setErrors(errors, { emitEvent })));
            }
        }
        /**
         * @private
         * @return {?}
         */
        _cancelExistingSubscription() {
            if (this._asyncValidationSubscription) {
                this._asyncValidationSubscription.unsubscribe();
            }
        }
        /**
         * Sets errors on a form control when running validations manually, rather than automatically.
         *
         * Calling `setErrors` also updates the validity of the parent control.
         *
         * \@usageNotes
         *
         * ### Manually set the errors for a control
         *
         * ```
         * const login = new FormControl('someLogin');
         * login.setErrors({
         *   notUnique: true
         * });
         *
         * expect(login.valid).toEqual(false);
         * expect(login.errors).toEqual({ notUnique: true });
         *
         * login.setValue('someOtherLogin');
         *
         * expect(login.valid).toEqual(true);
         * ```
         * @param {?} errors
         * @param {?=} opts
         * @return {?}
         */
        setErrors(errors, opts = {}) {
            ((/** @type {?} */ (this))).errors = errors;
            this._updateControlsErrors(opts.emitEvent !== false);
        }
        /**
         * Retrieves a child control given the control's name or path.
         *
         * \@usageNotes
         * ### Retrieve a nested control
         *
         * For example, to get a `name` control nested within a `person` sub-group:
         *
         * * `this.form.get('person.name');`
         *
         * -OR-
         *
         * * `this.form.get(['person', 'name']);`
         * @param {?} path A dot-delimited string or array of string/number values that define the path to the
         * control.
         *
         * @return {?}
         */
        get(path) {
            return _find(this, path, '.');
        }
        /**
         * \@description
         * Reports error data for the control with the given path.
         *
         * \@usageNotes
         * For example, for the following `FormGroup`:
         *
         * ```
         * form = new FormGroup({
         *   address: new FormGroup({ street: new FormControl() })
         * });
         * ```
         *
         * The path to the 'street' control from the root form would be 'address' -> 'street'.
         *
         * It can be provided to this method in one of two formats:
         *
         * 1. An array of string control names, e.g. `['address', 'street']`
         * 1. A period-delimited list of control names in one string, e.g. `'address.street'`
         *
         * @param {?} errorCode The code of the error to check
         * @param {?=} path A list of control names that designates how to move from the current control
         * to the control that should be queried for errors.
         *
         * @return {?} error data for that particular error. If the control or error is not present,
         * null is returned.
         */
        getError(errorCode, path) {
            /** @type {?} */
            const control = path ? this.get(path) : this;
            return control && control.errors ? control.errors[errorCode] : null;
        }
        /**
         * \@description
         * Reports whether the control with the given path has the error specified.
         *
         * \@usageNotes
         * For example, for the following `FormGroup`:
         *
         * ```
         * form = new FormGroup({
         *   address: new FormGroup({ street: new FormControl() })
         * });
         * ```
         *
         * The path to the 'street' control from the root form would be 'address' -> 'street'.
         *
         * It can be provided to this method in one of two formats:
         *
         * 1. An array of string control names, e.g. `['address', 'street']`
         * 1. A period-delimited list of control names in one string, e.g. `'address.street'`
         *
         * If no path is given, this method checks for the error on the current control.
         *
         * @param {?} errorCode The code of the error to check
         * @param {?=} path A list of control names that designates how to move from the current control
         * to the control that should be queried for errors.
         *
         * @return {?} whether the given error is present in the control at the given path.
         *
         * If the control is not present, false is returned.
         */
        hasError(errorCode, path) {
            return !!this.getError(errorCode, path);
        }
        /**
         * Retrieves the top-level ancestor of this control.
         * @return {?}
         */
        get root() {
            /** @type {?} */
            let x = this;
            while (x._parent) {
                x = x._parent;
            }
            return x;
        }
        /**
         * \@internal
         * @param {?} emitEvent
         * @return {?}
         */
        _updateControlsErrors(emitEvent) {
            ((/** @type {?} */ (this))).status = this._calculateStatus();
            if (emitEvent) {
                ((/** @type {?} */ (this.statusChanges))).emit(this.status);
            }
            if (this._parent) {
                this._parent._updateControlsErrors(emitEvent);
            }
        }
        /**
         * \@internal
         * @return {?}
         */
        _initObservables() {
            ((/** @type {?} */ (this))).valueChanges = new EventEmitter();
            ((/** @type {?} */ (this))).statusChanges = new EventEmitter();
        }
        /**
         * @private
         * @return {?}
         */
        _calculateStatus() {
            if (this._allControlsDisabled())
                return DISABLED;
            if (this.errors)
                return INVALID;
            if (this._anyControlsHaveStatus(PENDING))
                return PENDING;
            if (this._anyControlsHaveStatus(INVALID))
                return INVALID;
            return VALID;
        }
        /**
         * \@internal
         * @param {?} status
         * @return {?}
         */
        _anyControlsHaveStatus(status) {
            return this._anyControls((/**
             * @param {?} control
             * @return {?}
             */
            (control) => control.status === status));
        }
        /**
         * \@internal
         * @return {?}
         */
        _anyControlsDirty() {
            return this._anyControls((/**
             * @param {?} control
             * @return {?}
             */
            (control) => control.dirty));
        }
        /**
         * \@internal
         * @return {?}
         */
        _anyControlsTouched() {
            return this._anyControls((/**
             * @param {?} control
             * @return {?}
             */
            (control) => control.touched));
        }
        /**
         * \@internal
         * @param {?=} opts
         * @return {?}
         */
        _updatePristine(opts = {}) {
            ((/** @type {?} */ (this))).pristine = !this._anyControlsDirty();
            if (this._parent && !opts.onlySelf) {
                this._parent._updatePristine(opts);
            }
        }
        /**
         * \@internal
         * @param {?=} opts
         * @return {?}
         */
        _updateTouched(opts = {}) {
            ((/** @type {?} */ (this))).touched = this._anyControlsTouched();
            if (this._parent && !opts.onlySelf) {
                this._parent._updateTouched(opts);
            }
        }
        /**
         * \@internal
         * @param {?} formState
         * @return {?}
         */
        _isBoxedValue(formState) {
            return typeof formState === 'object' && formState !== null &&
                Object.keys(formState).length === 2 && 'value' in formState && 'disabled' in formState;
        }
        /**
         * \@internal
         * @param {?} fn
         * @return {?}
         */
        _registerOnCollectionChange(fn) {
            this._onCollectionChange = fn;
        }
        /**
         * \@internal
         * @param {?=} opts
         * @return {?}
         */
        _setUpdateStrategy(opts) {
            if (isOptionsObj(opts) && ((/** @type {?} */ (opts))).updateOn != null) {
                this._updateOn = (/** @type {?} */ (((/** @type {?} */ (opts))).updateOn));
            }
        }
        /**
         * Check to see if parent has been marked artificially dirty.
         *
         * \@internal
         * @private
         * @param {?=} onlySelf
         * @return {?}
         */
        _parentMarkedDirty(onlySelf) {
            /** @type {?} */
            const parentDirty = this._parent && this._parent.dirty;
            return !onlySelf && parentDirty && !this._parent._anyControlsDirty();
        }
    }
    /**
     * Tracks the value and validation status of an individual form control.
     *
     * This is one of the three fundamental building blocks of Angular forms, along with
     * `FormGroup` and `FormArray`. It extends the `AbstractControl` class that
     * implements most of the base functionality for accessing the value, validation status,
     * user interactions and events. See [usage examples below](#usage-notes).
     *
     * @see `AbstractControl`
     * @see [Reactive Forms Guide](guide/reactive-forms)
     * @see [Usage Notes](#usage-notes)
     *
     * \@usageNotes
     *
     * ### Initializing Form Controls
     *
     * Instantiate a `FormControl`, with an initial value.
     *
     * ```ts
     * const control = new FormControl('some value');
     * console.log(control.value);     // 'some value'
     * ```
     *
     * The following example initializes the control with a form state object. The `value`
     * and `disabled` keys are required in this case.
     *
     * ```ts
     * const control = new FormControl({ value: 'n/a', disabled: true });
     * console.log(control.value);     // 'n/a'
     * console.log(control.status);    // 'DISABLED'
     * ```
     *
     * The following example initializes the control with a sync validator.
     *
     * ```ts
     * const control = new FormControl('', Validators.required);
     * console.log(control.value);      // ''
     * console.log(control.status);     // 'INVALID'
     * ```
     *
     * The following example initializes the control using an options object.
     *
     * ```ts
     * const control = new FormControl('', {
     *    validators: Validators.required,
     *    asyncValidators: myAsyncValidator
     * });
     * ```
     *
     * ### Configure the control to update on a blur event
     *
     * Set the `updateOn` option to `'blur'` to update on the blur `event`.
     *
     * ```ts
     * const control = new FormControl('', { updateOn: 'blur' });
     * ```
     *
     * ### Configure the control to update on a submit event
     *
     * Set the `updateOn` option to `'submit'` to update on a submit `event`.
     *
     * ```ts
     * const control = new FormControl('', { updateOn: 'submit' });
     * ```
     *
     * ### Reset the control back to an initial value
     *
     * You reset to a specific form state by passing through a standalone
     * value or a form state object that contains both a value and a disabled state
     * (these are the only two properties that cannot be calculated).
     *
     * ```ts
     * const control = new FormControl('Nancy');
     *
     * console.log(control.value); // 'Nancy'
     *
     * control.reset('Drew');
     *
     * console.log(control.value); // 'Drew'
     * ```
     *
     * ### Reset the control back to an initial value and disabled
     *
     * ```
     * const control = new FormControl('Nancy');
     *
     * console.log(control.value); // 'Nancy'
     * console.log(control.status); // 'VALID'
     *
     * control.reset({ value: 'Drew', disabled: true });
     *
     * console.log(control.value); // 'Drew'
     * console.log(control.status); // 'DISABLED'
     * ```
     *
     * \@publicApi
     */
    class FormControl extends AbstractControl {
        /**
         * Creates a new `FormControl` instance.
         *
         * @param {?=} formState Initializes the control with an initial value,
         * or an object that defines the initial value and disabled state.
         *
         * @param {?=} validatorOrOpts A synchronous validator function, or an array of
         * such functions, or an `AbstractControlOptions` object that contains validation functions
         * and a validation trigger.
         *
         * @param {?=} asyncValidator A single async validator or array of async validator functions
         *
         */
        constructor(formState = null, validatorOrOpts, asyncValidator) {
            super(coerceToValidator(validatorOrOpts), coerceToAsyncValidator(asyncValidator, validatorOrOpts));
            /**
             * \@internal
             */
            this._onChange = [];
            this._applyFormState(formState);
            this._setUpdateStrategy(validatorOrOpts);
            this.updateValueAndValidity({ onlySelf: true, emitEvent: false });
            this._initObservables();
        }
        /**
         * Sets a new value for the form control.
         *
         * @param {?} value The new value for the control.
         * @param {?=} options Configuration options that determine how the control propagates changes
         * and emits events when the value changes.
         * The configuration options are passed to the {\@link AbstractControl#updateValueAndValidity
         * updateValueAndValidity} method.
         *
         * * `onlySelf`: When true, each change only affects this control, and not its parent. Default is
         * false.
         * * `emitEvent`: When true or not supplied (the default), both the `statusChanges` and
         * `valueChanges`
         * observables emit events with the latest status and value when the control value is updated.
         * When false, no events are emitted.
         * * `emitModelToViewChange`: When true or not supplied  (the default), each change triggers an
         * `onChange` event to
         * update the view.
         * * `emitViewToModelChange`: When true or not supplied (the default), each change triggers an
         * `ngModelChange`
         * event to update the model.
         *
         * @return {?}
         */
        setValue(value, options = {}) {
            ((/** @type {?} */ (this))).value = this._pendingValue = value;
            if (this._onChange.length && options.emitModelToViewChange !== false) {
                this._onChange.forEach((/**
                 * @param {?} changeFn
                 * @return {?}
                 */
                (changeFn) => changeFn(this.value, options.emitViewToModelChange !== false)));
            }
            this.updateValueAndValidity(options);
        }
        /**
         * Patches the value of a control.
         *
         * This function is functionally the same as {\@link FormControl#setValue setValue} at this level.
         * It exists for symmetry with {\@link FormGroup#patchValue patchValue} on `FormGroups` and
         * `FormArrays`, where it does behave differently.
         *
         * @see `setValue` for options
         * @param {?} value
         * @param {?=} options
         * @return {?}
         */
        patchValue(value, options = {}) {
            this.setValue(value, options);
        }
        /**
         * Resets the form control, marking it `pristine` and `untouched`, and setting
         * the value to null.
         *
         * @param {?=} formState Resets the control with an initial value,
         * or an object that defines the initial value and disabled state.
         *
         * @param {?=} options Configuration options that determine how the control propagates changes
         * and emits events after the value changes.
         *
         * * `onlySelf`: When true, each change only affects this control, and not its parent. Default is
         * false.
         * * `emitEvent`: When true or not supplied (the default), both the `statusChanges` and
         * `valueChanges`
         * observables emit events with the latest status and value when the control is reset.
         * When false, no events are emitted.
         *
         * @return {?}
         */
        reset(formState = null, options = {}) {
            this._applyFormState(formState);
            this.markAsPristine(options);
            this.markAsUntouched(options);
            this.setValue(this.value, options);
            this._pendingChange = false;
        }
        /**
         * \@internal
         * @return {?}
         */
        _updateValue() { }
        /**
         * \@internal
         * @param {?} condition
         * @return {?}
         */
        _anyControls(condition) {
            return false;
        }
        /**
         * \@internal
         * @return {?}
         */
        _allControlsDisabled() {
            return this.disabled;
        }
        /**
         * Register a listener for change events.
         *
         * @param {?} fn The method that is called when the value changes
         * @return {?}
         */
        registerOnChange(fn) {
            this._onChange.push(fn);
        }
        /**
         * \@internal
         * @return {?}
         */
        _clearChangeFns() {
            this._onChange = [];
            this._onDisabledChange = [];
            this._onCollectionChange = (/**
             * @return {?}
             */
            () => { });
        }
        /**
         * Register a listener for disabled events.
         *
         * @param {?} fn The method that is called when the disabled status changes.
         * @return {?}
         */
        registerOnDisabledChange(fn) {
            this._onDisabledChange.push(fn);
        }
        /**
         * \@internal
         * @param {?} cb
         * @return {?}
         */
        _forEachChild(cb) { }
        /**
         * \@internal
         * @return {?}
         */
        _syncPendingControls() {
            if (this.updateOn === 'submit') {
                if (this._pendingDirty)
                    this.markAsDirty();
                if (this._pendingTouched)
                    this.markAsTouched();
                if (this._pendingChange) {
                    this.setValue(this._pendingValue, { onlySelf: true, emitModelToViewChange: false });
                    return true;
                }
            }
            return false;
        }
        /**
         * @private
         * @param {?} formState
         * @return {?}
         */
        _applyFormState(formState) {
            if (this._isBoxedValue(formState)) {
                ((/** @type {?} */ (this))).value = this._pendingValue = formState.value;
                formState.disabled ? this.disable({ onlySelf: true, emitEvent: false }) :
                    this.enable({ onlySelf: true, emitEvent: false });
            }
            else {
                ((/** @type {?} */ (this))).value = this._pendingValue = formState;
            }
        }
    }
    /**
     * Tracks the value and validity state of a group of `FormControl` instances.
     *
     * A `FormGroup` aggregates the values of each child `FormControl` into one object,
     * with each control name as the key.  It calculates its status by reducing the status values
     * of its children. For example, if one of the controls in a group is invalid, the entire
     * group becomes invalid.
     *
     * `FormGroup` is one of the three fundamental building blocks used to define forms in Angular,
     * along with `FormControl` and `FormArray`.
     *
     * When instantiating a `FormGroup`, pass in a collection of child controls as the first
     * argument. The key for each child registers the name for the control.
     *
     * \@usageNotes
     *
     * ### Create a form group with 2 controls
     *
     * ```
     * const form = new FormGroup({
     *   first: new FormControl('Nancy', Validators.minLength(2)),
     *   last: new FormControl('Drew'),
     * });
     *
     * console.log(form.value);   // {first: 'Nancy', last; 'Drew'}
     * console.log(form.status);  // 'VALID'
     * ```
     *
     * ### Create a form group with a group-level validator
     *
     * You include group-level validators as the second arg, or group-level async
     * validators as the third arg. These come in handy when you want to perform validation
     * that considers the value of more than one child control.
     *
     * ```
     * const form = new FormGroup({
     *   password: new FormControl('', Validators.minLength(2)),
     *   passwordConfirm: new FormControl('', Validators.minLength(2)),
     * }, passwordMatchValidator);
     *
     *
     * function passwordMatchValidator(g: FormGroup) {
     *    return g.get('password').value === g.get('passwordConfirm').value
     *       ? null : {'mismatch': true};
     * }
     * ```
     *
     * Like `FormControl` instances, you choose to pass in
     * validators and async validators as part of an options object.
     *
     * ```
     * const form = new FormGroup({
     *   password: new FormControl('')
     *   passwordConfirm: new FormControl('')
     * }, { validators: passwordMatchValidator, asyncValidators: otherValidator });
     * ```
     *
     * ### Set the updateOn property for all controls in a form group
     *
     * The options object is used to set a default value for each child
     * control's `updateOn` property. If you set `updateOn` to `'blur'` at the
     * group level, all child controls default to 'blur', unless the child
     * has explicitly specified a different `updateOn` value.
     *
     * ```ts
     * const c = new FormGroup({
     *   one: new FormControl()
     * }, { updateOn: 'blur' });
     * ```
     *
     * \@publicApi
     */
    class FormGroup extends AbstractControl {
        /**
         * Creates a new `FormGroup` instance.
         *
         * @param {?} controls A collection of child controls. The key for each child is the name
         * under which it is registered.
         *
         * @param {?=} validatorOrOpts A synchronous validator function, or an array of
         * such functions, or an `AbstractControlOptions` object that contains validation functions
         * and a validation trigger.
         *
         * @param {?=} asyncValidator A single async validator or array of async validator functions
         *
         */
        constructor(controls, validatorOrOpts, asyncValidator) {
            super(coerceToValidator(validatorOrOpts), coerceToAsyncValidator(asyncValidator, validatorOrOpts));
            this.controls = controls;
            this._initObservables();
            this._setUpdateStrategy(validatorOrOpts);
            this._setUpControls();
            this.updateValueAndValidity({ onlySelf: true, emitEvent: false });
        }
        /**
         * Registers a control with the group's list of controls.
         *
         * This method does not update the value or validity of the control.
         * Use {\@link FormGroup#addControl addControl} instead.
         *
         * @param {?} name The control name to register in the collection
         * @param {?} control Provides the control for the given name
         * @return {?}
         */
        registerControl(name, control) {
            if (this.controls[name])
                return this.controls[name];
            this.controls[name] = control;
            control.setParent(this);
            control._registerOnCollectionChange(this._onCollectionChange);
            return control;
        }
        /**
         * Add a control to this group.
         *
         * This method also updates the value and validity of the control.
         *
         * @param {?} name The control name to add to the collection
         * @param {?} control Provides the control for the given name
         * @return {?}
         */
        addControl(name, control) {
            this.registerControl(name, control);
            this.updateValueAndValidity();
            this._onCollectionChange();
        }
        /**
         * Remove a control from this group.
         *
         * @param {?} name The control name to remove from the collection
         * @return {?}
         */
        removeControl(name) {
            if (this.controls[name])
                this.controls[name]._registerOnCollectionChange((/**
                 * @return {?}
                 */
                () => { }));
            delete (this.controls[name]);
            this.updateValueAndValidity();
            this._onCollectionChange();
        }
        /**
         * Replace an existing control.
         *
         * @param {?} name The control name to replace in the collection
         * @param {?} control Provides the control for the given name
         * @return {?}
         */
        setControl(name, control) {
            if (this.controls[name])
                this.controls[name]._registerOnCollectionChange((/**
                 * @return {?}
                 */
                () => { }));
            delete (this.controls[name]);
            if (control)
                this.registerControl(name, control);
            this.updateValueAndValidity();
            this._onCollectionChange();
        }
        /**
         * Check whether there is an enabled control with the given name in the group.
         *
         * Reports false for disabled controls. If you'd like to check for existence in the group
         * only, use {\@link AbstractControl#get get} instead.
         *
         * @param {?} controlName The control name to check for existence in the collection
         *
         * @return {?} false for disabled controls, true otherwise.
         */
        contains(controlName) {
            return this.controls.hasOwnProperty(controlName) && this.controls[controlName].enabled;
        }
        /**
         * Sets the value of the `FormGroup`. It accepts an object that matches
         * the structure of the group, with control names as keys.
         *
         * \@usageNotes
         * ### Set the complete value for the form group
         *
         * ```
         * const form = new FormGroup({
         *   first: new FormControl(),
         *   last: new FormControl()
         * });
         *
         * console.log(form.value);   // {first: null, last: null}
         *
         * form.setValue({first: 'Nancy', last: 'Drew'});
         * console.log(form.value);   // {first: 'Nancy', last: 'Drew'}
         * ```
         *
         * @throws When strict checks fail, such as setting the value of a control
         * that doesn't exist or if you exclude a value of a control that does exist.
         *
         * @param {?} value The new value for the control that matches the structure of the group.
         * @param {?=} options Configuration options that determine how the control propagates changes
         * and emits events after the value changes.
         * The configuration options are passed to the {\@link AbstractControl#updateValueAndValidity
         * updateValueAndValidity} method.
         *
         * * `onlySelf`: When true, each change only affects this control, and not its parent. Default is
         * false.
         * * `emitEvent`: When true or not supplied (the default), both the `statusChanges` and
         * `valueChanges`
         * observables emit events with the latest status and value when the control value is updated.
         * When false, no events are emitted.
         * @return {?}
         */
        setValue(value, options = {}) {
            this._checkAllValuesPresent(value);
            Object.keys(value).forEach((/**
             * @param {?} name
             * @return {?}
             */
            name => {
                this._throwIfControlMissing(name);
                this.controls[name].setValue(value[name], { onlySelf: true, emitEvent: options.emitEvent });
            }));
            this.updateValueAndValidity(options);
        }
        /**
         * Patches the value of the `FormGroup`. It accepts an object with control
         * names as keys, and does its best to match the values to the correct controls
         * in the group.
         *
         * It accepts both super-sets and sub-sets of the group without throwing an error.
         *
         * \@usageNotes
         * ### Patch the value for a form group
         *
         * ```
         * const form = new FormGroup({
         *    first: new FormControl(),
         *    last: new FormControl()
         * });
         * console.log(form.value);   // {first: null, last: null}
         *
         * form.patchValue({first: 'Nancy'});
         * console.log(form.value);   // {first: 'Nancy', last: null}
         * ```
         *
         * @param {?} value The object that matches the structure of the group.
         * @param {?=} options Configuration options that determine how the control propagates changes and
         * emits events after the value is patched.
         * * `onlySelf`: When true, each change only affects this control and not its parent. Default is
         * true.
         * * `emitEvent`: When true or not supplied (the default), both the `statusChanges` and
         * `valueChanges`
         * observables emit events with the latest status and value when the control value is updated.
         * When false, no events are emitted.
         * The configuration options are passed to the {\@link AbstractControl#updateValueAndValidity
         * updateValueAndValidity} method.
         * @return {?}
         */
        patchValue(value, options = {}) {
            Object.keys(value).forEach((/**
             * @param {?} name
             * @return {?}
             */
            name => {
                if (this.controls[name]) {
                    this.controls[name].patchValue(value[name], { onlySelf: true, emitEvent: options.emitEvent });
                }
            }));
            this.updateValueAndValidity(options);
        }
        /**
         * Resets the `FormGroup`, marks all descendants are marked `pristine` and `untouched`, and
         * the value of all descendants to null.
         *
         * You reset to a specific form state by passing in a map of states
         * that matches the structure of your form, with control names as keys. The state
         * is a standalone value or a form state object with both a value and a disabled
         * status.
         *
         * \@usageNotes
         *
         * ### Reset the form group values
         *
         * ```ts
         * const form = new FormGroup({
         *   first: new FormControl('first name'),
         *   last: new FormControl('last name')
         * });
         *
         * console.log(form.value);  // {first: 'first name', last: 'last name'}
         *
         * form.reset({ first: 'name', last: 'last name' });
         *
         * console.log(form.value);  // {first: 'name', last: 'last name'}
         * ```
         *
         * ### Reset the form group values and disabled status
         *
         * ```
         * const form = new FormGroup({
         *   first: new FormControl('first name'),
         *   last: new FormControl('last name')
         * });
         *
         * form.reset({
         *   first: {value: 'name', disabled: true},
         *   last: 'last'
         * });
         *
         * console.log(this.form.value);  // {first: 'name', last: 'last name'}
         * console.log(this.form.get('first').status);  // 'DISABLED'
         * ```
         * @param {?=} value Resets the control with an initial value,
         * or an object that defines the initial value and disabled state.
         *
         * @param {?=} options Configuration options that determine how the control propagates changes
         * and emits events when the group is reset.
         * * `onlySelf`: When true, each change only affects this control, and not its parent. Default is
         * false.
         * * `emitEvent`: When true or not supplied (the default), both the `statusChanges` and
         * `valueChanges`
         * observables emit events with the latest status and value when the control is reset.
         * When false, no events are emitted.
         * The configuration options are passed to the {\@link AbstractControl#updateValueAndValidity
         * updateValueAndValidity} method.
         *
         * @return {?}
         */
        reset(value = {}, options = {}) {
            this._forEachChild((/**
             * @param {?} control
             * @param {?} name
             * @return {?}
             */
            (control, name) => {
                control.reset(value[name], { onlySelf: true, emitEvent: options.emitEvent });
            }));
            this._updatePristine(options);
            this._updateTouched(options);
            this.updateValueAndValidity(options);
        }
        /**
         * The aggregate value of the `FormGroup`, including any disabled controls.
         *
         * Retrieves all values regardless of disabled status.
         * The `value` property is the best way to get the value of the group, because
         * it excludes disabled controls in the `FormGroup`.
         * @return {?}
         */
        getRawValue() {
            return this._reduceChildren({}, (/**
             * @param {?} acc
             * @param {?} control
             * @param {?} name
             * @return {?}
             */
            (acc, control, name) => {
                acc[name] = control instanceof FormControl ? control.value : ((/** @type {?} */ (control))).getRawValue();
                return acc;
            }));
        }
        /**
         * \@internal
         * @return {?}
         */
        _syncPendingControls() {
            /** @type {?} */
            let subtreeUpdated = this._reduceChildren(false, (/**
             * @param {?} updated
             * @param {?} child
             * @return {?}
             */
            (updated, child) => {
                return child._syncPendingControls() ? true : updated;
            }));
            if (subtreeUpdated)
                this.updateValueAndValidity({ onlySelf: true });
            return subtreeUpdated;
        }
        /**
         * \@internal
         * @param {?} name
         * @return {?}
         */
        _throwIfControlMissing(name) {
            if (!Object.keys(this.controls).length) {
                throw new Error(`
        There are no form controls registered with this group yet.  If you're using ngModel,
        you may want to check next tick (e.g. use setTimeout).
      `);
            }
            if (!this.controls[name]) {
                throw new Error(`Cannot find form control with name: ${name}.`);
            }
        }
        /**
         * \@internal
         * @param {?} cb
         * @return {?}
         */
        _forEachChild(cb) {
            Object.keys(this.controls).forEach((/**
             * @param {?} k
             * @return {?}
             */
            k => cb(this.controls[k], k)));
        }
        /**
         * \@internal
         * @return {?}
         */
        _setUpControls() {
            this._forEachChild((/**
             * @param {?} control
             * @return {?}
             */
            (control) => {
                control.setParent(this);
                control._registerOnCollectionChange(this._onCollectionChange);
            }));
        }
        /**
         * \@internal
         * @return {?}
         */
        _updateValue() {
            ((/** @type {?} */ (this))).value = this._reduceValue();
        }
        /**
         * \@internal
         * @param {?} condition
         * @return {?}
         */
        _anyControls(condition) {
            /** @type {?} */
            let res = false;
            this._forEachChild((/**
             * @param {?} control
             * @param {?} name
             * @return {?}
             */
            (control, name) => {
                res = res || (this.contains(name) && condition(control));
            }));
            return res;
        }
        /**
         * \@internal
         * @return {?}
         */
        _reduceValue() {
            return this._reduceChildren({}, (/**
             * @param {?} acc
             * @param {?} control
             * @param {?} name
             * @return {?}
             */
            (acc, control, name) => {
                if (control.enabled || this.disabled) {
                    acc[name] = control.value;
                }
                return acc;
            }));
        }
        /**
         * \@internal
         * @param {?} initValue
         * @param {?} fn
         * @return {?}
         */
        _reduceChildren(initValue, fn) {
            /** @type {?} */
            let res = initValue;
            this._forEachChild((/**
             * @param {?} control
             * @param {?} name
             * @return {?}
             */
            (control, name) => {
                res = fn(res, control, name);
            }));
            return res;
        }
        /**
         * \@internal
         * @return {?}
         */
        _allControlsDisabled() {
            for (const controlName of Object.keys(this.controls)) {
                if (this.controls[controlName].enabled) {
                    return false;
                }
            }
            return Object.keys(this.controls).length > 0 || this.disabled;
        }
        /**
         * \@internal
         * @param {?} value
         * @return {?}
         */
        _checkAllValuesPresent(value) {
            this._forEachChild((/**
             * @param {?} control
             * @param {?} name
             * @return {?}
             */
            (control, name) => {
                if (value[name] === undefined) {
                    throw new Error(`Must supply a value for form control with name: '${name}'.`);
                }
            }));
        }
    }
    /**
     * Tracks the value and validity state of an array of `FormControl`,
     * `FormGroup` or `FormArray` instances.
     *
     * A `FormArray` aggregates the values of each child `FormControl` into an array.
     * It calculates its status by reducing the status values of its children. For example, if one of
     * the controls in a `FormArray` is invalid, the entire array becomes invalid.
     *
     * `FormArray` is one of the three fundamental building blocks used to define forms in Angular,
     * along with `FormControl` and `FormGroup`.
     *
     * \@usageNotes
     *
     * ### Create an array of form controls
     *
     * ```
     * const arr = new FormArray([
     *   new FormControl('Nancy', Validators.minLength(2)),
     *   new FormControl('Drew'),
     * ]);
     *
     * console.log(arr.value);   // ['Nancy', 'Drew']
     * console.log(arr.status);  // 'VALID'
     * ```
     *
     * ### Create a form array with array-level validators
     *
     * You include array-level validators and async validators. These come in handy
     * when you want to perform validation that considers the value of more than one child
     * control.
     *
     * The two types of validators are passed in separately as the second and third arg
     * respectively, or together as part of an options object.
     *
     * ```
     * const arr = new FormArray([
     *   new FormControl('Nancy'),
     *   new FormControl('Drew')
     * ], {validators: myValidator, asyncValidators: myAsyncValidator});
     * ```
     *
     * ### Set the updateOn property for all controls in a form array
     *
     * The options object is used to set a default value for each child
     * control's `updateOn` property. If you set `updateOn` to `'blur'` at the
     * array level, all child controls default to 'blur', unless the child
     * has explicitly specified a different `updateOn` value.
     *
     * ```ts
     * const arr = new FormArray([
     *    new FormControl()
     * ], {updateOn: 'blur'});
     * ```
     *
     * ### Adding or removing controls from a form array
     *
     * To change the controls in the array, use the `push`, `insert`, `removeAt` or `clear` methods
     * in `FormArray` itself. These methods ensure the controls are properly tracked in the
     * form's hierarchy. Do not modify the array of `AbstractControl`s used to instantiate
     * the `FormArray` directly, as that result in strange and unexpected behavior such
     * as broken change detection.
     *
     * \@publicApi
     */
    class FormArray extends AbstractControl {
        /**
         * Creates a new `FormArray` instance.
         *
         * @param {?} controls An array of child controls. Each child control is given an index
         * where it is registered.
         *
         * @param {?=} validatorOrOpts A synchronous validator function, or an array of
         * such functions, or an `AbstractControlOptions` object that contains validation functions
         * and a validation trigger.
         *
         * @param {?=} asyncValidator A single async validator or array of async validator functions
         *
         */
        constructor(controls, validatorOrOpts, asyncValidator) {
            super(coerceToValidator(validatorOrOpts), coerceToAsyncValidator(asyncValidator, validatorOrOpts));
            this.controls = controls;
            this._initObservables();
            this._setUpdateStrategy(validatorOrOpts);
            this._setUpControls();
            this.updateValueAndValidity({ onlySelf: true, emitEvent: false });
        }
        /**
         * Get the `AbstractControl` at the given `index` in the array.
         *
         * @param {?} index Index in the array to retrieve the control
         * @return {?}
         */
        at(index) {
            return this.controls[index];
        }
        /**
         * Insert a new `AbstractControl` at the end of the array.
         *
         * @param {?} control Form control to be inserted
         * @return {?}
         */
        push(control) {
            this.controls.push(control);
            this._registerControl(control);
            this.updateValueAndValidity();
            this._onCollectionChange();
        }
        /**
         * Insert a new `AbstractControl` at the given `index` in the array.
         *
         * @param {?} index Index in the array to insert the control
         * @param {?} control Form control to be inserted
         * @return {?}
         */
        insert(index, control) {
            this.controls.splice(index, 0, control);
            this._registerControl(control);
            this.updateValueAndValidity();
        }
        /**
         * Remove the control at the given `index` in the array.
         *
         * @param {?} index Index in the array to remove the control
         * @return {?}
         */
        removeAt(index) {
            if (this.controls[index])
                this.controls[index]._registerOnCollectionChange((/**
                 * @return {?}
                 */
                () => { }));
            this.controls.splice(index, 1);
            this.updateValueAndValidity();
        }
        /**
         * Replace an existing control.
         *
         * @param {?} index Index in the array to replace the control
         * @param {?} control The `AbstractControl` control to replace the existing control
         * @return {?}
         */
        setControl(index, control) {
            if (this.controls[index])
                this.controls[index]._registerOnCollectionChange((/**
                 * @return {?}
                 */
                () => { }));
            this.controls.splice(index, 1);
            if (control) {
                this.controls.splice(index, 0, control);
                this._registerControl(control);
            }
            this.updateValueAndValidity();
            this._onCollectionChange();
        }
        /**
         * Length of the control array.
         * @return {?}
         */
        get length() {
            return this.controls.length;
        }
        /**
         * Sets the value of the `FormArray`. It accepts an array that matches
         * the structure of the control.
         *
         * This method performs strict checks, and throws an error if you try
         * to set the value of a control that doesn't exist or if you exclude the
         * value of a control.
         *
         * \@usageNotes
         * ### Set the values for the controls in the form array
         *
         * ```
         * const arr = new FormArray([
         *   new FormControl(),
         *   new FormControl()
         * ]);
         * console.log(arr.value);   // [null, null]
         *
         * arr.setValue(['Nancy', 'Drew']);
         * console.log(arr.value);   // ['Nancy', 'Drew']
         * ```
         *
         * @param {?} value Array of values for the controls
         * @param {?=} options Configure options that determine how the control propagates changes and
         * emits events after the value changes
         *
         * * `onlySelf`: When true, each change only affects this control, and not its parent. Default
         * is false.
         * * `emitEvent`: When true or not supplied (the default), both the `statusChanges` and
         * `valueChanges`
         * observables emit events with the latest status and value when the control value is updated.
         * When false, no events are emitted.
         * The configuration options are passed to the {\@link AbstractControl#updateValueAndValidity
         * updateValueAndValidity} method.
         * @return {?}
         */
        setValue(value, options = {}) {
            this._checkAllValuesPresent(value);
            value.forEach((/**
             * @param {?} newValue
             * @param {?} index
             * @return {?}
             */
            (newValue, index) => {
                this._throwIfControlMissing(index);
                this.at(index).setValue(newValue, { onlySelf: true, emitEvent: options.emitEvent });
            }));
            this.updateValueAndValidity(options);
        }
        /**
         * Patches the value of the `FormArray`. It accepts an array that matches the
         * structure of the control, and does its best to match the values to the correct
         * controls in the group.
         *
         * It accepts both super-sets and sub-sets of the array without throwing an error.
         *
         * \@usageNotes
         * ### Patch the values for controls in a form array
         *
         * ```
         * const arr = new FormArray([
         *    new FormControl(),
         *    new FormControl()
         * ]);
         * console.log(arr.value);   // [null, null]
         *
         * arr.patchValue(['Nancy']);
         * console.log(arr.value);   // ['Nancy', null]
         * ```
         *
         * @param {?} value Array of latest values for the controls
         * @param {?=} options Configure options that determine how the control propagates changes and
         * emits events after the value changes
         *
         * * `onlySelf`: When true, each change only affects this control, and not its parent. Default
         * is false.
         * * `emitEvent`: When true or not supplied (the default), both the `statusChanges` and
         * `valueChanges`
         * observables emit events with the latest status and value when the control value is updated.
         * When false, no events are emitted.
         * The configuration options are passed to the {\@link AbstractControl#updateValueAndValidity
         * updateValueAndValidity} method.
         * @return {?}
         */
        patchValue(value, options = {}) {
            value.forEach((/**
             * @param {?} newValue
             * @param {?} index
             * @return {?}
             */
            (newValue, index) => {
                if (this.at(index)) {
                    this.at(index).patchValue(newValue, { onlySelf: true, emitEvent: options.emitEvent });
                }
            }));
            this.updateValueAndValidity(options);
        }
        /**
         * Resets the `FormArray` and all descendants are marked `pristine` and `untouched`, and the
         * value of all descendants to null or null maps.
         *
         * You reset to a specific form state by passing in an array of states
         * that matches the structure of the control. The state is a standalone value
         * or a form state object with both a value and a disabled status.
         *
         * \@usageNotes
         * ### Reset the values in a form array
         *
         * ```ts
         * const arr = new FormArray([
         *    new FormControl(),
         *    new FormControl()
         * ]);
         * arr.reset(['name', 'last name']);
         *
         * console.log(this.arr.value);  // ['name', 'last name']
         * ```
         *
         * ### Reset the values in a form array and the disabled status for the first control
         *
         * ```
         * this.arr.reset([
         *   {value: 'name', disabled: true},
         *   'last'
         * ]);
         *
         * console.log(this.arr.value);  // ['name', 'last name']
         * console.log(this.arr.get(0).status);  // 'DISABLED'
         * ```
         *
         * @param {?=} value Array of values for the controls
         * @param {?=} options Configure options that determine how the control propagates changes and
         * emits events after the value changes
         *
         * * `onlySelf`: When true, each change only affects this control, and not its parent. Default
         * is false.
         * * `emitEvent`: When true or not supplied (the default), both the `statusChanges` and
         * `valueChanges`
         * observables emit events with the latest status and value when the control is reset.
         * When false, no events are emitted.
         * The configuration options are passed to the {\@link AbstractControl#updateValueAndValidity
         * updateValueAndValidity} method.
         * @return {?}
         */
        reset(value = [], options = {}) {
            this._forEachChild((/**
             * @param {?} control
             * @param {?} index
             * @return {?}
             */
            (control, index) => {
                control.reset(value[index], { onlySelf: true, emitEvent: options.emitEvent });
            }));
            this._updatePristine(options);
            this._updateTouched(options);
            this.updateValueAndValidity(options);
        }
        /**
         * The aggregate value of the array, including any disabled controls.
         *
         * Reports all values regardless of disabled status.
         * For enabled controls only, the `value` property is the best way to get the value of the array.
         * @return {?}
         */
        getRawValue() {
            return this.controls.map((/**
             * @param {?} control
             * @return {?}
             */
            (control) => {
                return control instanceof FormControl ? control.value : ((/** @type {?} */ (control))).getRawValue();
            }));
        }
        /**
         * Remove all controls in the `FormArray`.
         *
         * \@usageNotes
         * ### Remove all elements from a FormArray
         *
         * ```ts
         * const arr = new FormArray([
         *    new FormControl(),
         *    new FormControl()
         * ]);
         * console.log(arr.length);  // 2
         *
         * arr.clear();
         * console.log(arr.length);  // 0
         * ```
         *
         * It's a simpler and more efficient alternative to removing all elements one by one:
         *
         * ```ts
         * const arr = new FormArray([
         *    new FormControl(),
         *    new FormControl()
         * ]);
         *
         * while (arr.length) {
         *    arr.removeAt(0);
         * }
         * ```
         * @return {?}
         */
        clear() {
            if (this.controls.length < 1)
                return;
            this._forEachChild((/**
             * @param {?} control
             * @return {?}
             */
            (control) => control._registerOnCollectionChange((/**
             * @return {?}
             */
            () => { }))));
            this.controls.splice(0);
            this.updateValueAndValidity();
        }
        /**
         * \@internal
         * @return {?}
         */
        _syncPendingControls() {
            /** @type {?} */
            let subtreeUpdated = this.controls.reduce((/**
             * @param {?} updated
             * @param {?} child
             * @return {?}
             */
            (updated, child) => {
                return child._syncPendingControls() ? true : updated;
            }), false);
            if (subtreeUpdated)
                this.updateValueAndValidity({ onlySelf: true });
            return subtreeUpdated;
        }
        /**
         * \@internal
         * @param {?} index
         * @return {?}
         */
        _throwIfControlMissing(index) {
            if (!this.controls.length) {
                throw new Error(`
        There are no form controls registered with this array yet.  If you're using ngModel,
        you may want to check next tick (e.g. use setTimeout).
      `);
            }
            if (!this.at(index)) {
                throw new Error(`Cannot find form control at index ${index}`);
            }
        }
        /**
         * \@internal
         * @param {?} cb
         * @return {?}
         */
        _forEachChild(cb) {
            this.controls.forEach((/**
             * @param {?} control
             * @param {?} index
             * @return {?}
             */
            (control, index) => {
                cb(control, index);
            }));
        }
        /**
         * \@internal
         * @return {?}
         */
        _updateValue() {
            ((/** @type {?} */ (this))).value =
                this.controls.filter((/**
                 * @param {?} control
                 * @return {?}
                 */
                (control) => control.enabled || this.disabled))
                    .map((/**
                 * @param {?} control
                 * @return {?}
                 */
                (control) => control.value));
        }
        /**
         * \@internal
         * @param {?} condition
         * @return {?}
         */
        _anyControls(condition) {
            return this.controls.some((/**
             * @param {?} control
             * @return {?}
             */
            (control) => control.enabled && condition(control)));
        }
        /**
         * \@internal
         * @return {?}
         */
        _setUpControls() {
            this._forEachChild((/**
             * @param {?} control
             * @return {?}
             */
            (control) => this._registerControl(control)));
        }
        /**
         * \@internal
         * @param {?} value
         * @return {?}
         */
        _checkAllValuesPresent(value) {
            this._forEachChild((/**
             * @param {?} control
             * @param {?} i
             * @return {?}
             */
            (control, i) => {
                if (value[i] === undefined) {
                    throw new Error(`Must supply a value for form control at index: ${i}.`);
                }
            }));
        }
        /**
         * \@internal
         * @return {?}
         */
        _allControlsDisabled() {
            for (const control of this.controls) {
                if (control.enabled)
                    return false;
            }
            return this.controls.length > 0 || this.disabled;
        }
        /**
         * @private
         * @param {?} control
         * @return {?}
         */
        _registerControl(control) {
            control.setParent(this);
            control._registerOnCollectionChange(this._onCollectionChange);
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/forms/src/directives/ng_form.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const formDirectiveProvider = {
        provide: ControlContainer,
        useExisting: forwardRef((/**
         * @return {?}
         */
        () => NgForm))
    };
    const ɵ0$b = /**
     * @return {?}
     */
    () => Promise.resolve(null);
    /** @type {?} */
    const resolvedPromise = ((ɵ0$b))();
    /**
     * \@description
     * Creates a top-level `FormGroup` instance and binds it to a form
     * to track aggregate form value and validation status.
     *
     * As soon as you import the `FormsModule`, this directive becomes active by default on
     * all `<form>` tags.  You don't need to add a special selector.
     *
     * You optionally export the directive into a local template variable using `ngForm` as the key
     * (ex: `#myForm="ngForm"`). This is optional, but useful.  Many properties from the underlying
     * `FormGroup` instance are duplicated on the directive itself, so a reference to it
     * gives you access to the aggregate value and validity status of the form, as well as
     * user interaction properties like `dirty` and `touched`.
     *
     * To register child controls with the form, use `NgModel` with a `name`
     * attribute. You may use `NgModelGroup` to create sub-groups within the form.
     *
     * If necessary, listen to the directive's `ngSubmit` event to be notified when the user has
     * triggered a form submission. The `ngSubmit` event emits the original form
     * submission event.
     *
     * In template driven forms, all `<form>` tags are automatically tagged as `NgForm`.
     * To import the `FormsModule` but skip its usage in some forms,
     * for example, to use native HTML5 validation, add the `ngNoForm` and the `<form>`
     * tags won't create an `NgForm` directive. In reactive forms, using `ngNoForm` is
     * unnecessary because the `<form>` tags are inert. In that case, you would
     * refrain from using the `formGroup` directive.
     *
     * \@usageNotes
     *
     * ### Listening for form submission
     *
     * The following example shows how to capture the form values from the "ngSubmit" event.
     *
     * {\@example forms/ts/simpleForm/simple_form_example.ts region='Component'}
     *
     * ### Setting the update options
     *
     * The following example shows you how to change the "updateOn" option from its default using
     * ngFormOptions.
     *
     * ```html
     * <form [ngFormOptions]="{updateOn: 'blur'}">
     *    <input name="one" ngModel>  <!-- this ngModel will update on blur -->
     * </form>
     * ```
     *
     * ### Native DOM validation UI
     *
     * In order to prevent the native DOM form validation UI from interfering with Angular's form
     * validation, Angular automatically adds the `novalidate` attribute on any `<form>` whenever
     * `FormModule` or `ReactiveFormModule` are imported into the application.
     * If you want to explicitly enable native DOM validation UI with Angular forms, you can add the
     * `ngNativeValidate` attribute to the `<form>` element:
     *
     * ```html
     * <form ngNativeValidate>
     *   ...
     * </form>
     * ```
     *
     * \@ngModule FormsModule
     * \@publicApi
     */
    class NgForm extends ControlContainer {
        /**
         * @param {?} validators
         * @param {?} asyncValidators
         */
        constructor(validators, asyncValidators) {
            super();
            /**
             * \@description
             * Returns whether the form submission has been triggered.
             */
            this.submitted = false;
            this._directives = [];
            /**
             * \@description
             * Event emitter for the "ngSubmit" event
             */
            this.ngSubmit = new EventEmitter();
            this.form =
                new FormGroup({}, composeValidators(validators), composeAsyncValidators(asyncValidators));
        }
        /**
         * \@description
         * Lifecycle method called after the view is initialized. For internal use only.
         * @return {?}
         */
        ngAfterViewInit() {
            this._setUpdateStrategy();
        }
        /**
         * \@description
         * The directive instance.
         * @return {?}
         */
        get formDirective() {
            return this;
        }
        /**
         * \@description
         * The internal `FormGroup` instance.
         * @return {?}
         */
        get control() {
            return this.form;
        }
        /**
         * \@description
         * Returns an array representing the path to this group. Because this directive
         * always lives at the top level of a form, it is always an empty array.
         * @return {?}
         */
        get path() {
            return [];
        }
        /**
         * \@description
         * Returns a map of the controls in this group.
         * @return {?}
         */
        get controls() {
            return this.form.controls;
        }
        /**
         * \@description
         * Method that sets up the control directive in this group, re-calculates its value
         * and validity, and adds the instance to the internal list of directives.
         *
         * @param {?} dir The `NgModel` directive instance.
         * @return {?}
         */
        addControl(dir) {
            resolvedPromise.then((/**
             * @return {?}
             */
            () => {
                /** @type {?} */
                const container = this._findContainer(dir.path);
                ((/** @type {?} */ (dir))).control =
                    (/** @type {?} */ (container.registerControl(dir.name, dir.control)));
                setUpControl(dir.control, dir);
                dir.control.updateValueAndValidity({ emitEvent: false });
                this._directives.push(dir);
            }));
        }
        /**
         * \@description
         * Retrieves the `FormControl` instance from the provided `NgModel` directive.
         *
         * @param {?} dir The `NgModel` directive instance.
         * @return {?}
         */
        getControl(dir) {
            return (/** @type {?} */ (this.form.get(dir.path)));
        }
        /**
         * \@description
         * Removes the `NgModel` instance from the internal list of directives
         *
         * @param {?} dir The `NgModel` directive instance.
         * @return {?}
         */
        removeControl(dir) {
            resolvedPromise.then((/**
             * @return {?}
             */
            () => {
                /** @type {?} */
                const container = this._findContainer(dir.path);
                if (container) {
                    container.removeControl(dir.name);
                }
                removeDir(this._directives, dir);
            }));
        }
        /**
         * \@description
         * Adds a new `NgModelGroup` directive instance to the form.
         *
         * @param {?} dir The `NgModelGroup` directive instance.
         * @return {?}
         */
        addFormGroup(dir) {
            resolvedPromise.then((/**
             * @return {?}
             */
            () => {
                /** @type {?} */
                const container = this._findContainer(dir.path);
                /** @type {?} */
                const group = new FormGroup({});
                setUpFormContainer(group, dir);
                container.registerControl(dir.name, group);
                group.updateValueAndValidity({ emitEvent: false });
            }));
        }
        /**
         * \@description
         * Removes the `NgModelGroup` directive instance from the form.
         *
         * @param {?} dir The `NgModelGroup` directive instance.
         * @return {?}
         */
        removeFormGroup(dir) {
            resolvedPromise.then((/**
             * @return {?}
             */
            () => {
                /** @type {?} */
                const container = this._findContainer(dir.path);
                if (container) {
                    container.removeControl(dir.name);
                }
            }));
        }
        /**
         * \@description
         * Retrieves the `FormGroup` for a provided `NgModelGroup` directive instance
         *
         * @param {?} dir The `NgModelGroup` directive instance.
         * @return {?}
         */
        getFormGroup(dir) {
            return (/** @type {?} */ (this.form.get(dir.path)));
        }
        /**
         * Sets the new value for the provided `NgControl` directive.
         *
         * @param {?} dir The `NgControl` directive instance.
         * @param {?} value The new value for the directive's control.
         * @return {?}
         */
        updateModel(dir, value) {
            resolvedPromise.then((/**
             * @return {?}
             */
            () => {
                /** @type {?} */
                const ctrl = (/** @type {?} */ (this.form.get((/** @type {?} */ (dir.path)))));
                ctrl.setValue(value);
            }));
        }
        /**
         * \@description
         * Sets the value for this `FormGroup`.
         *
         * @param {?} value The new value
         * @return {?}
         */
        setValue(value) {
            this.control.setValue(value);
        }
        /**
         * \@description
         * Method called when the "submit" event is triggered on the form.
         * Triggers the `ngSubmit` emitter to emit the "submit" event as its payload.
         *
         * @param {?} $event The "submit" event object
         * @return {?}
         */
        onSubmit($event) {
            ((/** @type {?} */ (this))).submitted = true;
            syncPendingControls(this.form, this._directives);
            this.ngSubmit.emit($event);
            return false;
        }
        /**
         * \@description
         * Method called when the "reset" event is triggered on the form.
         * @return {?}
         */
        onReset() {
            this.resetForm();
        }
        /**
         * \@description
         * Resets the form to an initial value and resets its submitted status.
         *
         * @param {?=} value The new value for the form.
         * @return {?}
         */
        resetForm(value = undefined) {
            this.form.reset(value);
            ((/** @type {?} */ (this))).submitted = false;
        }
        /**
         * @private
         * @return {?}
         */
        _setUpdateStrategy() {
            if (this.options && this.options.updateOn != null) {
                this.form._updateOn = this.options.updateOn;
            }
        }
        /**
         * \@internal
         * @param {?} path
         * @return {?}
         */
        _findContainer(path) {
            path.pop();
            return path.length ? (/** @type {?} */ (this.form.get(path))) : this.form;
        }
    }
    NgForm.ɵfac = function NgForm_Factory(t) { return new (t || NgForm)(ɵɵdirectiveInject(NG_VALIDATORS, 10), ɵɵdirectiveInject(NG_ASYNC_VALIDATORS, 10)); };
    NgForm.ɵdir = ɵɵdefineDirective({ type: NgForm, selectors: [["form", 3, "ngNoForm", "", 3, "formGroup", ""], ["ng-form"], ["", "ngForm", ""]], hostBindings: function NgForm_HostBindings(rf, ctx) { if (rf & 1) {
            ɵɵlistener("submit", function NgForm_submit_HostBindingHandler($event) { return ctx.onSubmit($event); })("reset", function NgForm_reset_HostBindingHandler() { return ctx.onReset(); });
        } }, inputs: { options: ["ngFormOptions", "options"] }, outputs: { ngSubmit: "ngSubmit" }, exportAs: ["ngForm"], features: [ɵɵProvidersFeature([formDirectiveProvider]), ɵɵInheritDefinitionFeature] });
    /** @nocollapse */
    NgForm.ctorParameters = () => [
        { type: Array, decorators: [{ type: Optional }, { type: Self }, { type: Inject, args: [NG_VALIDATORS,] }] },
        { type: Array, decorators: [{ type: Optional }, { type: Self }, { type: Inject, args: [NG_ASYNC_VALIDATORS,] }] }
    ];
    NgForm.propDecorators = {
        options: [{ type: Input, args: ['ngFormOptions',] }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/forms/src/directives/abstract_form_group_directive.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * \@description
     * A base class for code shared between the `NgModelGroup` and `FormGroupName` directives.
     *
     * \@publicApi
     */
    class AbstractFormGroupDirective extends ControlContainer {
        /**
         * \@description
         * An internal callback method triggered on the instance after the inputs are set.
         * Registers the group with its parent group.
         * @return {?}
         */
        ngOnInit() {
            this._checkParentType();
            (/** @type {?} */ (this.formDirective)).addFormGroup(this);
        }
        /**
         * \@description
         * An internal callback method triggered before the instance is destroyed.
         * Removes the group from its parent group.
         * @return {?}
         */
        ngOnDestroy() {
            if (this.formDirective) {
                this.formDirective.removeFormGroup(this);
            }
        }
        /**
         * \@description
         * The `FormGroup` bound to this directive.
         * @return {?}
         */
        get control() {
            return (/** @type {?} */ (this.formDirective)).getFormGroup(this);
        }
        /**
         * \@description
         * The path to this group from the top-level directive.
         * @return {?}
         */
        get path() {
            return controlPath(this.name == null ? this.name : this.name.toString(), this._parent);
        }
        /**
         * \@description
         * The top-level directive for this group if present, otherwise null.
         * @return {?}
         */
        get formDirective() {
            return this._parent ? this._parent.formDirective : null;
        }
        /**
         * \@description
         * The synchronous validators registered with this group.
         * @return {?}
         */
        get validator() {
            return composeValidators(this._validators);
        }
        /**
         * \@description
         * The async validators registered with this group.
         * @return {?}
         */
        get asyncValidator() {
            return composeAsyncValidators(this._asyncValidators);
        }
        /**
         * \@internal
         * @return {?}
         */
        _checkParentType() { }
    }
    AbstractFormGroupDirective.ɵfac = function AbstractFormGroupDirective_Factory(t) { return ɵAbstractFormGroupDirective_BaseFactory(t || AbstractFormGroupDirective); };
    AbstractFormGroupDirective.ɵdir = ɵɵdefineDirective({ type: AbstractFormGroupDirective, features: [ɵɵInheritDefinitionFeature] });
    const ɵAbstractFormGroupDirective_BaseFactory = ɵɵgetInheritedFactory(AbstractFormGroupDirective);

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/forms/src/directives/template_driven_errors.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class TemplateDrivenErrors {
        /**
         * @return {?}
         */
        static modelParentException() {
            throw new Error(`
      ngModel cannot be used to register form controls with a parent formGroup directive.  Try using
      formGroup's partner directive "formControlName" instead.  Example:

      ${FormErrorExamples.formControlName}

      Or, if you'd like to avoid registering this form control, indicate that it's standalone in ngModelOptions:

      Example:

      ${FormErrorExamples.ngModelWithFormGroup}`);
        }
        /**
         * @return {?}
         */
        static formGroupNameException() {
            throw new Error(`
      ngModel cannot be used to register form controls with a parent formGroupName or formArrayName directive.

      Option 1: Use formControlName instead of ngModel (reactive strategy):

      ${FormErrorExamples.formGroupName}

      Option 2:  Update ngModel's parent be ngModelGroup (template-driven strategy):

      ${FormErrorExamples.ngModelGroup}`);
        }
        /**
         * @return {?}
         */
        static missingNameException() {
            throw new Error(`If ngModel is used within a form tag, either the name attribute must be set or the form
      control must be defined as 'standalone' in ngModelOptions.

      Example 1: <input [(ngModel)]="person.firstName" name="first">
      Example 2: <input [(ngModel)]="person.firstName" [ngModelOptions]="{standalone: true}">`);
        }
        /**
         * @return {?}
         */
        static modelGroupParentException() {
            throw new Error(`
      ngModelGroup cannot be used with a parent formGroup directive.

      Option 1: Use formGroupName instead of ngModelGroup (reactive strategy):

      ${FormErrorExamples.formGroupName}

      Option 2:  Use a regular form tag instead of the formGroup directive (template-driven strategy):

      ${FormErrorExamples.ngModelGroup}`);
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/forms/src/directives/ng_model_group.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const modelGroupProvider = {
        provide: ControlContainer,
        useExisting: forwardRef((/**
         * @return {?}
         */
        () => NgModelGroup))
    };
    /**
     * \@description
     * Creates and binds a `FormGroup` instance to a DOM element.
     *
     * This directive can only be used as a child of `NgForm` (within `<form>` tags).
     *
     * Use this directive to validate a sub-group of your form separately from the
     * rest of your form, or if some values in your domain model make more sense
     * to consume together in a nested object.
     *
     * Provide a name for the sub-group and it will become the key
     * for the sub-group in the form's full value. If you need direct access, export the directive into
     * a local template variable using `ngModelGroup` (ex: `#myGroup="ngModelGroup"`).
     *
     * \@usageNotes
     *
     * ### Consuming controls in a grouping
     *
     * The following example shows you how to combine controls together in a sub-group
     * of the form.
     *
     * {\@example forms/ts/ngModelGroup/ng_model_group_example.ts region='Component'}
     *
     * \@ngModule FormsModule
     * \@publicApi
     */
    class NgModelGroup extends AbstractFormGroupDirective {
        /**
         * @param {?} parent
         * @param {?} validators
         * @param {?} asyncValidators
         */
        constructor(parent, validators, asyncValidators) {
            super();
            this._parent = parent;
            this._validators = validators;
            this._asyncValidators = asyncValidators;
        }
        /**
         * \@internal
         * @return {?}
         */
        _checkParentType() {
            if (!(this._parent instanceof NgModelGroup) && !(this._parent instanceof NgForm)) {
                TemplateDrivenErrors.modelGroupParentException();
            }
        }
    }
    NgModelGroup.ɵfac = function NgModelGroup_Factory(t) { return new (t || NgModelGroup)(ɵɵdirectiveInject(ControlContainer, 5), ɵɵdirectiveInject(NG_VALIDATORS, 10), ɵɵdirectiveInject(NG_ASYNC_VALIDATORS, 10)); };
    NgModelGroup.ɵdir = ɵɵdefineDirective({ type: NgModelGroup, selectors: [["", "ngModelGroup", ""]], inputs: { name: ["ngModelGroup", "name"] }, exportAs: ["ngModelGroup"], features: [ɵɵProvidersFeature([modelGroupProvider]), ɵɵInheritDefinitionFeature] });
    /** @nocollapse */
    NgModelGroup.ctorParameters = () => [
        { type: ControlContainer, decorators: [{ type: Host }, { type: SkipSelf }] },
        { type: Array, decorators: [{ type: Optional }, { type: Self }, { type: Inject, args: [NG_VALIDATORS,] }] },
        { type: Array, decorators: [{ type: Optional }, { type: Self }, { type: Inject, args: [NG_ASYNC_VALIDATORS,] }] }
    ];
    NgModelGroup.propDecorators = {
        name: [{ type: Input, args: ['ngModelGroup',] }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/forms/src/directives/ng_model.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const formControlBinding = {
        provide: NgControl,
        useExisting: forwardRef((/**
         * @return {?}
         */
        () => NgModel))
    };
    const ɵ0$1$4 = /**
     * @return {?}
     */
    () => Promise.resolve(null);
    /**
     * `ngModel` forces an additional change detection run when its inputs change:
     * E.g.:
     * ```
     * <div>{{myModel.valid}}</div>
     * <input [(ngModel)]="myValue" #myModel="ngModel">
     * ```
     * I.e. `ngModel` can export itself on the element and then be used in the template.
     * Normally, this would result in expressions before the `input` that use the exported directive
     * to have and old value as they have been
     * dirty checked before. As this is a very common case for `ngModel`, we added this second change
     * detection run.
     *
     * Notes:
     * - this is just one extra run no matter how many `ngModel` have been changed.
     * - this is a general problem when using `exportAs` for directives!
     * @type {?}
     */
    const resolvedPromise$1 = ((ɵ0$1$4))();
    /**
     * \@description
     * Creates a `FormControl` instance from a domain model and binds it
     * to a form control element.
     *
     * The `FormControl` instance tracks the value, user interaction, and
     * validation status of the control and keeps the view synced with the model. If used
     * within a parent form, the directive also registers itself with the form as a child
     * control.
     *
     * This directive is used by itself or as part of a larger form. Use the
     * `ngModel` selector to activate it.
     *
     * It accepts a domain model as an optional `Input`. If you have a one-way binding
     * to `ngModel` with `[]` syntax, changing the value of the domain model in the component
     * class sets the value in the view. If you have a two-way binding with `[()]` syntax
     * (also known as 'banana-box syntax'), the value in the UI always syncs back to
     * the domain model in your class.
     *
     * To inspect the properties of the associated `FormControl` (like validity state),
     * export the directive into a local template variable using `ngModel` as the key (ex:
     * `#myVar="ngModel"`). You then access the control using the directive's `control` property, but
     * most properties used (like `valid` and `dirty`) fall through to the control anyway for direct
     * access. See a full list of properties directly available in `AbstractControlDirective`.
     *
     * @see `RadioControlValueAccessor`
     * @see `SelectControlValueAccessor`
     *
     * \@usageNotes
     *
     * ### Using ngModel on a standalone control
     *
     * The following examples show a simple standalone control using `ngModel`:
     *
     * {\@example forms/ts/simpleNgModel/simple_ng_model_example.ts region='Component'}
     *
     * When using the `ngModel` within `<form>` tags, you'll also need to supply a `name` attribute
     * so that the control can be registered with the parent form under that name.
     *
     * In the context of a parent form, it's often unnecessary to include one-way or two-way binding,
     * as the parent form syncs the value for you. You access its properties by exporting it into a
     * local template variable using `ngForm` such as (`#f="ngForm"`). Use the variable where
     * needed on form submission.
     *
     * If you do need to populate initial values into your form, using a one-way binding for
     * `ngModel` tends to be sufficient as long as you use the exported form's value rather
     * than the domain model's value on submit.
     *
     * ### Using ngModel within a form
     *
     * The following example shows controls using `ngModel` within a form:
     *
     * {\@example forms/ts/simpleForm/simple_form_example.ts region='Component'}
     *
     * ### Using a standalone ngModel within a group
     *
     * The following example shows you how to use a standalone ngModel control
     * within a form. This controls the display of the form, but doesn't contain form data.
     *
     * ```html
     * <form>
     *   <input name="login" ngModel placeholder="Login">
     *   <input type="checkbox" ngModel [ngModelOptions]="{standalone: true}"> Show more options?
     * </form>
     * <!-- form value: {login: ''} -->
     * ```
     *
     * ### Setting the ngModel name attribute through options
     *
     * The following example shows you an alternate way to set the name attribute. The name attribute is
     * used within a custom form component, and the name `\@Input` property serves a different purpose.
     *
     * ```html
     * <form>
     *   <my-person-control name="Nancy" ngModel [ngModelOptions]="{name: 'user'}">
     *   </my-person-control>
     * </form>
     * <!-- form value: {user: ''} -->
     * ```
     *
     * \@ngModule FormsModule
     * \@publicApi
     */
    class NgModel extends NgControl {
        /**
         * @param {?} parent
         * @param {?} validators
         * @param {?} asyncValidators
         * @param {?} valueAccessors
         */
        constructor(parent, validators, asyncValidators, valueAccessors) {
            super();
            this.control = new FormControl();
            /**
             * \@internal
             */
            this._registered = false;
            /**
             * \@description
             * Event emitter for producing the `ngModelChange` event after
             * the view model updates.
             */
            this.update = new EventEmitter();
            this._parent = parent;
            this._rawValidators = validators || [];
            this._rawAsyncValidators = asyncValidators || [];
            this.valueAccessor = selectValueAccessor(this, valueAccessors);
        }
        /**
         * \@description
         * A lifecycle method called when the directive's inputs change. For internal use
         * only.
         *
         * @param {?} changes A object of key/value pairs for the set of changed inputs.
         * @return {?}
         */
        ngOnChanges(changes) {
            this._checkForErrors();
            if (!this._registered)
                this._setUpControl();
            if ('isDisabled' in changes) {
                this._updateDisabled(changes);
            }
            if (isPropertyUpdated(changes, this.viewModel)) {
                this._updateValue(this.model);
                this.viewModel = this.model;
            }
        }
        /**
         * \@description
         * Lifecycle method called before the directive's instance is destroyed. For internal
         * use only.
         * @return {?}
         */
        ngOnDestroy() {
            this.formDirective && this.formDirective.removeControl(this);
        }
        /**
         * \@description
         * Returns an array that represents the path from the top-level form to this control.
         * Each index is the string name of the control on that level.
         * @return {?}
         */
        get path() {
            return this._parent ? controlPath(this.name, this._parent) : [this.name];
        }
        /**
         * \@description
         * The top-level directive for this control if present, otherwise null.
         * @return {?}
         */
        get formDirective() {
            return this._parent ? this._parent.formDirective : null;
        }
        /**
         * \@description
         * Synchronous validator function composed of all the synchronous validators
         * registered with this directive.
         * @return {?}
         */
        get validator() {
            return composeValidators(this._rawValidators);
        }
        /**
         * \@description
         * Async validator function composed of all the async validators registered with this
         * directive.
         * @return {?}
         */
        get asyncValidator() {
            return composeAsyncValidators(this._rawAsyncValidators);
        }
        /**
         * \@description
         * Sets the new value for the view model and emits an `ngModelChange` event.
         *
         * @param {?} newValue The new value emitted by `ngModelChange`.
         * @return {?}
         */
        viewToModelUpdate(newValue) {
            this.viewModel = newValue;
            this.update.emit(newValue);
        }
        /**
         * @private
         * @return {?}
         */
        _setUpControl() {
            this._setUpdateStrategy();
            this._isStandalone() ? this._setUpStandalone() : this.formDirective.addControl(this);
            this._registered = true;
        }
        /**
         * @private
         * @return {?}
         */
        _setUpdateStrategy() {
            if (this.options && this.options.updateOn != null) {
                this.control._updateOn = this.options.updateOn;
            }
        }
        /**
         * @private
         * @return {?}
         */
        _isStandalone() {
            return !this._parent || !!(this.options && this.options.standalone);
        }
        /**
         * @private
         * @return {?}
         */
        _setUpStandalone() {
            setUpControl(this.control, this);
            this.control.updateValueAndValidity({ emitEvent: false });
        }
        /**
         * @private
         * @return {?}
         */
        _checkForErrors() {
            if (!this._isStandalone()) {
                this._checkParentType();
            }
            this._checkName();
        }
        /**
         * @private
         * @return {?}
         */
        _checkParentType() {
            if (!(this._parent instanceof NgModelGroup) &&
                this._parent instanceof AbstractFormGroupDirective) {
                TemplateDrivenErrors.formGroupNameException();
            }
            else if (!(this._parent instanceof NgModelGroup) && !(this._parent instanceof NgForm)) {
                TemplateDrivenErrors.modelParentException();
            }
        }
        /**
         * @private
         * @return {?}
         */
        _checkName() {
            if (this.options && this.options.name)
                this.name = this.options.name;
            if (!this._isStandalone() && !this.name) {
                TemplateDrivenErrors.missingNameException();
            }
        }
        /**
         * @private
         * @param {?} value
         * @return {?}
         */
        _updateValue(value) {
            resolvedPromise$1.then((/**
             * @return {?}
             */
            () => {
                this.control.setValue(value, { emitViewToModelChange: false });
            }));
        }
        /**
         * @private
         * @param {?} changes
         * @return {?}
         */
        _updateDisabled(changes) {
            /** @type {?} */
            const disabledValue = changes['isDisabled'].currentValue;
            /** @type {?} */
            const isDisabled = disabledValue === '' || (disabledValue && disabledValue !== 'false');
            resolvedPromise$1.then((/**
             * @return {?}
             */
            () => {
                if (isDisabled && !this.control.disabled) {
                    this.control.disable();
                }
                else if (!isDisabled && this.control.disabled) {
                    this.control.enable();
                }
            }));
        }
    }
    NgModel.ɵfac = function NgModel_Factory(t) { return new (t || NgModel)(ɵɵdirectiveInject(ControlContainer, 9), ɵɵdirectiveInject(NG_VALIDATORS, 10), ɵɵdirectiveInject(NG_ASYNC_VALIDATORS, 10), ɵɵdirectiveInject(NG_VALUE_ACCESSOR, 10)); };
    NgModel.ɵdir = ɵɵdefineDirective({ type: NgModel, selectors: [["", "ngModel", "", 3, "formControlName", "", 3, "formControl", ""]], inputs: { name: "name", isDisabled: ["disabled", "isDisabled"], model: ["ngModel", "model"], options: ["ngModelOptions", "options"] }, outputs: { update: "ngModelChange" }, exportAs: ["ngModel"], features: [ɵɵProvidersFeature([formControlBinding]), ɵɵInheritDefinitionFeature, ɵɵNgOnChangesFeature] });
    /** @nocollapse */
    NgModel.ctorParameters = () => [
        { type: ControlContainer, decorators: [{ type: Optional }, { type: Host }] },
        { type: Array, decorators: [{ type: Optional }, { type: Self }, { type: Inject, args: [NG_VALIDATORS,] }] },
        { type: Array, decorators: [{ type: Optional }, { type: Self }, { type: Inject, args: [NG_ASYNC_VALIDATORS,] }] },
        { type: Array, decorators: [{ type: Optional }, { type: Self }, { type: Inject, args: [NG_VALUE_ACCESSOR,] }] }
    ];
    NgModel.propDecorators = {
        name: [{ type: Input }],
        isDisabled: [{ type: Input, args: ['disabled',] }],
        model: [{ type: Input, args: ['ngModel',] }],
        options: [{ type: Input, args: ['ngModelOptions',] }],
        update: [{ type: Output, args: ['ngModelChange',] }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/forms/src/directives/ng_no_validate_directive.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * \@description
     *
     * Adds `novalidate` attribute to all forms by default.
     *
     * `novalidate` is used to disable browser's native form validation.
     *
     * If you want to use native validation with Angular forms, just add `ngNativeValidate` attribute:
     *
     * ```
     * <form ngNativeValidate></form>
     * ```
     *
     * \@publicApi
     * \@ngModule ReactiveFormsModule
     * \@ngModule FormsModule
     */
    class ɵNgNoValidate {
    }
    ɵNgNoValidate.ɵfac = function ɵNgNoValidate_Factory(t) { return new (t || ɵNgNoValidate)(); };
    ɵNgNoValidate.ɵdir = ɵɵdefineDirective({ type: ɵNgNoValidate, selectors: [["form", 3, "ngNoForm", "", 3, "ngNativeValidate", ""]], hostAttrs: ["novalidate", ""] });

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/forms/src/directives/reactive_directives/form_control_directive.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Token to provide to turn off the ngModel warning on formControl and formControlName.
     * @type {?}
     */
    const NG_MODEL_WITH_FORM_CONTROL_WARNING = new InjectionToken('NgModelWithFormControlWarning');
    /** @type {?} */
    const formControlBinding$1 = {
        provide: NgControl,
        useExisting: forwardRef((/**
         * @return {?}
         */
        () => FormControlDirective))
    };
    /**
     * \@description
     * Synchronizes a standalone `FormControl` instance to a form control element.
     *
     * Note that support for using the `ngModel` input property and `ngModelChange` event with reactive
     * form directives was deprecated in Angular v6 and is scheduled for removal in
     * a future version of Angular.
     * For details, see [Deprecated features](guide/deprecations#ngmodel-with-reactive-forms).
     *
     * @see [Reactive Forms Guide](guide/reactive-forms)
     * @see `FormControl`
     * @see `AbstractControl`
     *
     * \@usageNotes
     *
     * The following example shows how to register a standalone control and set its value.
     *
     * {\@example forms/ts/simpleFormControl/simple_form_control_example.ts region='Component'}
     *
     * \@ngModule ReactiveFormsModule
     * \@publicApi
     */
    class FormControlDirective extends NgControl {
        /**
         * @param {?} validators
         * @param {?} asyncValidators
         * @param {?} valueAccessors
         * @param {?} _ngModelWarningConfig
         */
        constructor(validators, asyncValidators, valueAccessors, _ngModelWarningConfig) {
            super();
            this._ngModelWarningConfig = _ngModelWarningConfig;
            /**
             * @deprecated as of v6
             */
            this.update = new EventEmitter();
            /**
             * \@description
             * Instance property used to track whether an ngModel warning has been sent out for this
             * particular `FormControlDirective` instance. Used to support warning config of "always".
             *
             * \@internal
             */
            this._ngModelWarningSent = false;
            this._rawValidators = validators || [];
            this._rawAsyncValidators = asyncValidators || [];
            this.valueAccessor = selectValueAccessor(this, valueAccessors);
        }
        /**
         * \@description
         * Triggers a warning that this input should not be used with reactive forms.
         * @param {?} isDisabled
         * @return {?}
         */
        set isDisabled(isDisabled) {
            ReactiveErrors.disabledAttrWarning();
        }
        /**
         * \@description
         * A lifecycle method called when the directive's inputs change. For internal use
         * only.
         *
         * @param {?} changes A object of key/value pairs for the set of changed inputs.
         * @return {?}
         */
        ngOnChanges(changes) {
            if (this._isControlChanged(changes)) {
                setUpControl(this.form, this);
                if (this.control.disabled && (/** @type {?} */ (this.valueAccessor)).setDisabledState) {
                    (/** @type {?} */ ((/** @type {?} */ (this.valueAccessor)).setDisabledState))(true);
                }
                this.form.updateValueAndValidity({ emitEvent: false });
            }
            if (isPropertyUpdated(changes, this.viewModel)) {
                _ngModelWarning('formControl', FormControlDirective, this, this._ngModelWarningConfig);
                this.form.setValue(this.model);
                this.viewModel = this.model;
            }
        }
        /**
         * \@description
         * Returns an array that represents the path from the top-level form to this control.
         * Each index is the string name of the control on that level.
         * @return {?}
         */
        get path() {
            return [];
        }
        /**
         * \@description
         * Synchronous validator function composed of all the synchronous validators
         * registered with this directive.
         * @return {?}
         */
        get validator() {
            return composeValidators(this._rawValidators);
        }
        /**
         * \@description
         * Async validator function composed of all the async validators registered with this
         * directive.
         * @return {?}
         */
        get asyncValidator() {
            return composeAsyncValidators(this._rawAsyncValidators);
        }
        /**
         * \@description
         * The `FormControl` bound to this directive.
         * @return {?}
         */
        get control() {
            return this.form;
        }
        /**
         * \@description
         * Sets the new value for the view model and emits an `ngModelChange` event.
         *
         * @param {?} newValue The new value for the view model.
         * @return {?}
         */
        viewToModelUpdate(newValue) {
            this.viewModel = newValue;
            this.update.emit(newValue);
        }
        /**
         * @private
         * @param {?} changes
         * @return {?}
         */
        _isControlChanged(changes) {
            return changes.hasOwnProperty('form');
        }
    }
    FormControlDirective.ɵfac = function FormControlDirective_Factory(t) { return new (t || FormControlDirective)(ɵɵdirectiveInject(NG_VALIDATORS, 10), ɵɵdirectiveInject(NG_ASYNC_VALIDATORS, 10), ɵɵdirectiveInject(NG_VALUE_ACCESSOR, 10), ɵɵdirectiveInject(NG_MODEL_WITH_FORM_CONTROL_WARNING, 8)); };
    FormControlDirective.ɵdir = ɵɵdefineDirective({ type: FormControlDirective, selectors: [["", "formControl", ""]], inputs: { isDisabled: ["disabled", "isDisabled"], form: ["formControl", "form"], model: ["ngModel", "model"] }, outputs: { update: "ngModelChange" }, exportAs: ["ngForm"], features: [ɵɵProvidersFeature([formControlBinding$1]), ɵɵInheritDefinitionFeature, ɵɵNgOnChangesFeature] });
    /**
     * \@description
     * Static property used to track whether any ngModel warnings have been sent across
     * all instances of FormControlDirective. Used to support warning config of "once".
     *
     * \@internal
     */
    FormControlDirective._ngModelWarningSentOnce = false;
    /** @nocollapse */
    FormControlDirective.ctorParameters = () => [
        { type: Array, decorators: [{ type: Optional }, { type: Self }, { type: Inject, args: [NG_VALIDATORS,] }] },
        { type: Array, decorators: [{ type: Optional }, { type: Self }, { type: Inject, args: [NG_ASYNC_VALIDATORS,] }] },
        { type: Array, decorators: [{ type: Optional }, { type: Self }, { type: Inject, args: [NG_VALUE_ACCESSOR,] }] },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [NG_MODEL_WITH_FORM_CONTROL_WARNING,] }] }
    ];
    FormControlDirective.propDecorators = {
        form: [{ type: Input, args: ['formControl',] }],
        isDisabled: [{ type: Input, args: ['disabled',] }],
        model: [{ type: Input, args: ['ngModel',] }],
        update: [{ type: Output, args: ['ngModelChange',] }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/forms/src/directives/reactive_directives/form_group_directive.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const formDirectiveProvider$1 = {
        provide: ControlContainer,
        useExisting: forwardRef((/**
         * @return {?}
         */
        () => FormGroupDirective))
    };
    /**
     * \@description
     *
     * Binds an existing `FormGroup` to a DOM element.
     *
     * This directive accepts an existing `FormGroup` instance. It will then use this
     * `FormGroup` instance to match any child `FormControl`, `FormGroup`,
     * and `FormArray` instances to child `FormControlName`, `FormGroupName`,
     * and `FormArrayName` directives.
     *
     * @see [Reactive Forms Guide](guide/reactive-forms)
     * @see `AbstractControl`
     *
     * ### Register Form Group
     *
     * The following example registers a `FormGroup` with first name and last name controls,
     * and listens for the *ngSubmit* event when the button is clicked.
     *
     * {\@example forms/ts/simpleFormGroup/simple_form_group_example.ts region='Component'}
     *
     * \@ngModule ReactiveFormsModule
     * \@publicApi
     */
    class FormGroupDirective extends ControlContainer {
        /**
         * @param {?} _validators
         * @param {?} _asyncValidators
         */
        constructor(_validators, _asyncValidators) {
            super();
            this._validators = _validators;
            this._asyncValidators = _asyncValidators;
            /**
             * \@description
             * Reports whether the form submission has been triggered.
             */
            this.submitted = false;
            /**
             * \@description
             * Tracks the list of added `FormControlName` instances
             */
            this.directives = [];
            /**
             * \@description
             * Tracks the `FormGroup` bound to this directive.
             */
            this.form = (/** @type {?} */ (null));
            /**
             * \@description
             * Emits an event when the form submission has been triggered.
             */
            this.ngSubmit = new EventEmitter();
        }
        /**
         * \@description
         * A lifecycle method called when the directive's inputs change. For internal use only.
         *
         * @param {?} changes A object of key/value pairs for the set of changed inputs.
         * @return {?}
         */
        ngOnChanges(changes) {
            this._checkFormPresent();
            if (changes.hasOwnProperty('form')) {
                this._updateValidators();
                this._updateDomValue();
                this._updateRegistrations();
            }
        }
        /**
         * \@description
         * Returns this directive's instance.
         * @return {?}
         */
        get formDirective() {
            return this;
        }
        /**
         * \@description
         * Returns the `FormGroup` bound to this directive.
         * @return {?}
         */
        get control() {
            return this.form;
        }
        /**
         * \@description
         * Returns an array representing the path to this group. Because this directive
         * always lives at the top level of a form, it always an empty array.
         * @return {?}
         */
        get path() {
            return [];
        }
        /**
         * \@description
         * Method that sets up the control directive in this group, re-calculates its value
         * and validity, and adds the instance to the internal list of directives.
         *
         * @param {?} dir The `FormControlName` directive instance.
         * @return {?}
         */
        addControl(dir) {
            /** @type {?} */
            const ctrl = this.form.get(dir.path);
            setUpControl(ctrl, dir);
            ctrl.updateValueAndValidity({ emitEvent: false });
            this.directives.push(dir);
            return ctrl;
        }
        /**
         * \@description
         * Retrieves the `FormControl` instance from the provided `FormControlName` directive
         *
         * @param {?} dir The `FormControlName` directive instance.
         * @return {?}
         */
        getControl(dir) {
            return (/** @type {?} */ (this.form.get(dir.path)));
        }
        /**
         * \@description
         * Removes the `FormControlName` instance from the internal list of directives
         *
         * @param {?} dir The `FormControlName` directive instance.
         * @return {?}
         */
        removeControl(dir) {
            removeDir(this.directives, dir);
        }
        /**
         * Adds a new `FormGroupName` directive instance to the form.
         *
         * @param {?} dir The `FormGroupName` directive instance.
         * @return {?}
         */
        addFormGroup(dir) {
            /** @type {?} */
            const ctrl = this.form.get(dir.path);
            setUpFormContainer(ctrl, dir);
            ctrl.updateValueAndValidity({ emitEvent: false });
        }
        /**
         * No-op method to remove the form group.
         *
         * @param {?} dir The `FormGroupName` directive instance.
         * @return {?}
         */
        removeFormGroup(dir) { }
        /**
         * \@description
         * Retrieves the `FormGroup` for a provided `FormGroupName` directive instance
         *
         * @param {?} dir The `FormGroupName` directive instance.
         * @return {?}
         */
        getFormGroup(dir) {
            return (/** @type {?} */ (this.form.get(dir.path)));
        }
        /**
         * Adds a new `FormArrayName` directive instance to the form.
         *
         * @param {?} dir The `FormArrayName` directive instance.
         * @return {?}
         */
        addFormArray(dir) {
            /** @type {?} */
            const ctrl = this.form.get(dir.path);
            setUpFormContainer(ctrl, dir);
            ctrl.updateValueAndValidity({ emitEvent: false });
        }
        /**
         * No-op method to remove the form array.
         *
         * @param {?} dir The `FormArrayName` directive instance.
         * @return {?}
         */
        removeFormArray(dir) { }
        /**
         * \@description
         * Retrieves the `FormArray` for a provided `FormArrayName` directive instance.
         *
         * @param {?} dir The `FormArrayName` directive instance.
         * @return {?}
         */
        getFormArray(dir) {
            return (/** @type {?} */ (this.form.get(dir.path)));
        }
        /**
         * Sets the new value for the provided `FormControlName` directive.
         *
         * @param {?} dir The `FormControlName` directive instance.
         * @param {?} value The new value for the directive's control.
         * @return {?}
         */
        updateModel(dir, value) {
            /** @type {?} */
            const ctrl = (/** @type {?} */ (this.form.get(dir.path)));
            ctrl.setValue(value);
        }
        /**
         * \@description
         * Method called with the "submit" event is triggered on the form.
         * Triggers the `ngSubmit` emitter to emit the "submit" event as its payload.
         *
         * @param {?} $event The "submit" event object
         * @return {?}
         */
        onSubmit($event) {
            ((/** @type {?} */ (this))).submitted = true;
            syncPendingControls(this.form, this.directives);
            this.ngSubmit.emit($event);
            return false;
        }
        /**
         * \@description
         * Method called when the "reset" event is triggered on the form.
         * @return {?}
         */
        onReset() {
            this.resetForm();
        }
        /**
         * \@description
         * Resets the form to an initial value and resets its submitted status.
         *
         * @param {?=} value The new value for the form.
         * @return {?}
         */
        resetForm(value = undefined) {
            this.form.reset(value);
            ((/** @type {?} */ (this))).submitted = false;
        }
        /**
         * \@internal
         * @return {?}
         */
        _updateDomValue() {
            this.directives.forEach((/**
             * @param {?} dir
             * @return {?}
             */
            dir => {
                /** @type {?} */
                const newCtrl = this.form.get(dir.path);
                if (dir.control !== newCtrl) {
                    cleanUpControl(dir.control, dir);
                    if (newCtrl)
                        setUpControl(newCtrl, dir);
                    ((/** @type {?} */ (dir))).control = newCtrl;
                }
            }));
            this.form._updateTreeValidity({ emitEvent: false });
        }
        /**
         * @private
         * @return {?}
         */
        _updateRegistrations() {
            this.form._registerOnCollectionChange((/**
             * @return {?}
             */
            () => this._updateDomValue()));
            if (this._oldForm)
                this._oldForm._registerOnCollectionChange((/**
                 * @return {?}
                 */
                () => { }));
            this._oldForm = this.form;
        }
        /**
         * @private
         * @return {?}
         */
        _updateValidators() {
            /** @type {?} */
            const sync = composeValidators(this._validators);
            this.form.validator = Validators.compose([(/** @type {?} */ (this.form.validator)), (/** @type {?} */ (sync))]);
            /** @type {?} */
            const async = composeAsyncValidators(this._asyncValidators);
            this.form.asyncValidator = Validators.composeAsync([(/** @type {?} */ (this.form.asyncValidator)), (/** @type {?} */ (async))]);
        }
        /**
         * @private
         * @return {?}
         */
        _checkFormPresent() {
            if (!this.form) {
                ReactiveErrors.missingFormException();
            }
        }
    }
    FormGroupDirective.ɵfac = function FormGroupDirective_Factory(t) { return new (t || FormGroupDirective)(ɵɵdirectiveInject(NG_VALIDATORS, 10), ɵɵdirectiveInject(NG_ASYNC_VALIDATORS, 10)); };
    FormGroupDirective.ɵdir = ɵɵdefineDirective({ type: FormGroupDirective, selectors: [["", "formGroup", ""]], hostBindings: function FormGroupDirective_HostBindings(rf, ctx) { if (rf & 1) {
            ɵɵlistener("submit", function FormGroupDirective_submit_HostBindingHandler($event) { return ctx.onSubmit($event); })("reset", function FormGroupDirective_reset_HostBindingHandler() { return ctx.onReset(); });
        } }, inputs: { form: ["formGroup", "form"] }, outputs: { ngSubmit: "ngSubmit" }, exportAs: ["ngForm"], features: [ɵɵProvidersFeature([formDirectiveProvider$1]), ɵɵInheritDefinitionFeature, ɵɵNgOnChangesFeature] });
    /** @nocollapse */
    FormGroupDirective.ctorParameters = () => [
        { type: Array, decorators: [{ type: Optional }, { type: Self }, { type: Inject, args: [NG_VALIDATORS,] }] },
        { type: Array, decorators: [{ type: Optional }, { type: Self }, { type: Inject, args: [NG_ASYNC_VALIDATORS,] }] }
    ];
    FormGroupDirective.propDecorators = {
        form: [{ type: Input, args: ['formGroup',] }],
        ngSubmit: [{ type: Output }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/forms/src/directives/reactive_directives/form_group_name.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const formGroupNameProvider = {
        provide: ControlContainer,
        useExisting: forwardRef((/**
         * @return {?}
         */
        () => FormGroupName))
    };
    /**
     * \@description
     *
     * Syncs a nested `FormGroup` to a DOM element.
     *
     * This directive can only be used with a parent `FormGroupDirective`.
     *
     * It accepts the string name of the nested `FormGroup` to link, and
     * looks for a `FormGroup` registered with that name in the parent
     * `FormGroup` instance you passed into `FormGroupDirective`.
     *
     * Use nested form groups to validate a sub-group of a
     * form separately from the rest or to group the values of certain
     * controls into their own nested object.
     *
     * @see [Reactive Forms Guide](guide/reactive-forms)
     *
     * \@usageNotes
     *
     * ### Access the group by name
     *
     * The following example uses the {\@link AbstractControl#get get} method to access the
     * associated `FormGroup`
     *
     * ```ts
     *   this.form.get('name');
     * ```
     *
     * ### Access individual controls in the group
     *
     * The following example uses the {\@link AbstractControl#get get} method to access
     * individual controls within the group using dot syntax.
     *
     * ```ts
     *   this.form.get('name.first');
     * ```
     *
     * ### Register a nested `FormGroup`.
     *
     * The following example registers a nested *name* `FormGroup` within an existing `FormGroup`,
     * and provides methods to retrieve the nested `FormGroup` and individual controls.
     *
     * {\@example forms/ts/nestedFormGroup/nested_form_group_example.ts region='Component'}
     *
     * \@ngModule ReactiveFormsModule
     * \@publicApi
     */
    class FormGroupName extends AbstractFormGroupDirective {
        /**
         * @param {?} parent
         * @param {?} validators
         * @param {?} asyncValidators
         */
        constructor(parent, validators, asyncValidators) {
            super();
            this._parent = parent;
            this._validators = validators;
            this._asyncValidators = asyncValidators;
        }
        /**
         * \@internal
         * @return {?}
         */
        _checkParentType() {
            if (_hasInvalidParent(this._parent)) {
                ReactiveErrors.groupParentException();
            }
        }
    }
    FormGroupName.ɵfac = function FormGroupName_Factory(t) { return new (t || FormGroupName)(ɵɵdirectiveInject(ControlContainer, 13), ɵɵdirectiveInject(NG_VALIDATORS, 10), ɵɵdirectiveInject(NG_ASYNC_VALIDATORS, 10)); };
    FormGroupName.ɵdir = ɵɵdefineDirective({ type: FormGroupName, selectors: [["", "formGroupName", ""]], inputs: { name: ["formGroupName", "name"] }, features: [ɵɵProvidersFeature([formGroupNameProvider]), ɵɵInheritDefinitionFeature] });
    /** @nocollapse */
    FormGroupName.ctorParameters = () => [
        { type: ControlContainer, decorators: [{ type: Optional }, { type: Host }, { type: SkipSelf }] },
        { type: Array, decorators: [{ type: Optional }, { type: Self }, { type: Inject, args: [NG_VALIDATORS,] }] },
        { type: Array, decorators: [{ type: Optional }, { type: Self }, { type: Inject, args: [NG_ASYNC_VALIDATORS,] }] }
    ];
    FormGroupName.propDecorators = {
        name: [{ type: Input, args: ['formGroupName',] }]
    };
    /** @type {?} */
    const formArrayNameProvider = {
        provide: ControlContainer,
        useExisting: forwardRef((/**
         * @return {?}
         */
        () => FormArrayName))
    };
    /**
     * \@description
     *
     * Syncs a nested `FormArray` to a DOM element.
     *
     * This directive is designed to be used with a parent `FormGroupDirective` (selector:
     * `[formGroup]`).
     *
     * It accepts the string name of the nested `FormArray` you want to link, and
     * will look for a `FormArray` registered with that name in the parent
     * `FormGroup` instance you passed into `FormGroupDirective`.
     *
     * @see [Reactive Forms Guide](guide/reactive-forms)
     * @see `AbstractControl`
     *
     * \@usageNotes
     *
     * ### Example
     *
     * {\@example forms/ts/nestedFormArray/nested_form_array_example.ts region='Component'}
     *
     * \@ngModule ReactiveFormsModule
     * \@publicApi
     */
    class FormArrayName extends ControlContainer {
        /**
         * @param {?} parent
         * @param {?} validators
         * @param {?} asyncValidators
         */
        constructor(parent, validators, asyncValidators) {
            super();
            this._parent = parent;
            this._validators = validators;
            this._asyncValidators = asyncValidators;
        }
        /**
         * \@description
         * A lifecycle method called when the directive's inputs are initialized. For internal use only.
         *
         * @throws If the directive does not have a valid parent.
         * @return {?}
         */
        ngOnInit() {
            this._checkParentType();
            (/** @type {?} */ (this.formDirective)).addFormArray(this);
        }
        /**
         * \@description
         * A lifecycle method called before the directive's instance is destroyed. For internal use only.
         * @return {?}
         */
        ngOnDestroy() {
            if (this.formDirective) {
                this.formDirective.removeFormArray(this);
            }
        }
        /**
         * \@description
         * The `FormArray` bound to this directive.
         * @return {?}
         */
        get control() {
            return (/** @type {?} */ (this.formDirective)).getFormArray(this);
        }
        /**
         * \@description
         * The top-level directive for this group if present, otherwise null.
         * @return {?}
         */
        get formDirective() {
            return this._parent ? (/** @type {?} */ (this._parent.formDirective)) : null;
        }
        /**
         * \@description
         * Returns an array that represents the path from the top-level form to this control.
         * Each index is the string name of the control on that level.
         * @return {?}
         */
        get path() {
            return controlPath(this.name == null ? this.name : this.name.toString(), this._parent);
        }
        /**
         * \@description
         * Synchronous validator function composed of all the synchronous validators registered with this
         * directive.
         * @return {?}
         */
        get validator() {
            return composeValidators(this._validators);
        }
        /**
         * \@description
         * Async validator function composed of all the async validators registered with this directive.
         * @return {?}
         */
        get asyncValidator() {
            return composeAsyncValidators(this._asyncValidators);
        }
        /**
         * @private
         * @return {?}
         */
        _checkParentType() {
            if (_hasInvalidParent(this._parent)) {
                ReactiveErrors.arrayParentException();
            }
        }
    }
    FormArrayName.ɵfac = function FormArrayName_Factory(t) { return new (t || FormArrayName)(ɵɵdirectiveInject(ControlContainer, 13), ɵɵdirectiveInject(NG_VALIDATORS, 10), ɵɵdirectiveInject(NG_ASYNC_VALIDATORS, 10)); };
    FormArrayName.ɵdir = ɵɵdefineDirective({ type: FormArrayName, selectors: [["", "formArrayName", ""]], inputs: { name: ["formArrayName", "name"] }, features: [ɵɵProvidersFeature([formArrayNameProvider]), ɵɵInheritDefinitionFeature] });
    /** @nocollapse */
    FormArrayName.ctorParameters = () => [
        { type: ControlContainer, decorators: [{ type: Optional }, { type: Host }, { type: SkipSelf }] },
        { type: Array, decorators: [{ type: Optional }, { type: Self }, { type: Inject, args: [NG_VALIDATORS,] }] },
        { type: Array, decorators: [{ type: Optional }, { type: Self }, { type: Inject, args: [NG_ASYNC_VALIDATORS,] }] }
    ];
    FormArrayName.propDecorators = {
        name: [{ type: Input, args: ['formArrayName',] }]
    };
    /**
     * @param {?} parent
     * @return {?}
     */
    function _hasInvalidParent(parent) {
        return !(parent instanceof FormGroupName) && !(parent instanceof FormGroupDirective) &&
            !(parent instanceof FormArrayName);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/forms/src/directives/reactive_directives/form_control_name.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const controlNameBinding = {
        provide: NgControl,
        useExisting: forwardRef((/**
         * @return {?}
         */
        () => FormControlName))
    };
    /**
     * \@description
     * Syncs a `FormControl` in an existing `FormGroup` to a form control
     * element by name.
     *
     * @see [Reactive Forms Guide](guide/reactive-forms)
     * @see `FormControl`
     * @see `AbstractControl`
     *
     * \@usageNotes
     *
     * ### Register `FormControl` within a group
     *
     * The following example shows how to register multiple form controls within a form group
     * and set their value.
     *
     * {\@example forms/ts/simpleFormGroup/simple_form_group_example.ts region='Component'}
     *
     * To see `formControlName` examples with different form control types, see:
     *
     * * Radio buttons: `RadioControlValueAccessor`
     * * Selects: `SelectControlValueAccessor`
     *
     * ### Use with ngModel is deprecated
     *
     * Support for using the `ngModel` input property and `ngModelChange` event with reactive
     * form directives has been deprecated in Angular v6 and is scheduled for removal in
     * a future version of Angular.
     *
     * For details, see [Deprecated features](guide/deprecations#ngmodel-with-reactive-forms).
     *
     * \@ngModule ReactiveFormsModule
     * \@publicApi
     */
    class FormControlName extends NgControl {
        /**
         * @param {?} parent
         * @param {?} validators
         * @param {?} asyncValidators
         * @param {?} valueAccessors
         * @param {?} _ngModelWarningConfig
         */
        constructor(parent, validators, asyncValidators, valueAccessors, _ngModelWarningConfig) {
            super();
            this._ngModelWarningConfig = _ngModelWarningConfig;
            this._added = false;
            /**
             * @deprecated as of v6
             */
            this.update = new EventEmitter();
            /**
             * \@description
             * Instance property used to track whether an ngModel warning has been sent out for this
             * particular FormControlName instance. Used to support warning config of "always".
             *
             * \@internal
             */
            this._ngModelWarningSent = false;
            this._parent = parent;
            this._rawValidators = validators || [];
            this._rawAsyncValidators = asyncValidators || [];
            this.valueAccessor = selectValueAccessor(this, valueAccessors);
        }
        /**
         * \@description
         * Triggers a warning that this input should not be used with reactive forms.
         * @param {?} isDisabled
         * @return {?}
         */
        set isDisabled(isDisabled) {
            ReactiveErrors.disabledAttrWarning();
        }
        /**
         * \@description
         * A lifecycle method called when the directive's inputs change. For internal use only.
         *
         * @param {?} changes A object of key/value pairs for the set of changed inputs.
         * @return {?}
         */
        ngOnChanges(changes) {
            if (!this._added)
                this._setUpControl();
            if (isPropertyUpdated(changes, this.viewModel)) {
                _ngModelWarning('formControlName', FormControlName, this, this._ngModelWarningConfig);
                this.viewModel = this.model;
                this.formDirective.updateModel(this, this.model);
            }
        }
        /**
         * \@description
         * Lifecycle method called before the directive's instance is destroyed. For internal use only.
         * @return {?}
         */
        ngOnDestroy() {
            if (this.formDirective) {
                this.formDirective.removeControl(this);
            }
        }
        /**
         * \@description
         * Sets the new value for the view model and emits an `ngModelChange` event.
         *
         * @param {?} newValue The new value for the view model.
         * @return {?}
         */
        viewToModelUpdate(newValue) {
            this.viewModel = newValue;
            this.update.emit(newValue);
        }
        /**
         * \@description
         * Returns an array that represents the path from the top-level form to this control.
         * Each index is the string name of the control on that level.
         * @return {?}
         */
        get path() {
            return controlPath(this.name == null ? this.name : this.name.toString(), (/** @type {?} */ (this._parent)));
        }
        /**
         * \@description
         * The top-level directive for this group if present, otherwise null.
         * @return {?}
         */
        get formDirective() {
            return this._parent ? this._parent.formDirective : null;
        }
        /**
         * \@description
         * Synchronous validator function composed of all the synchronous validators
         * registered with this directive.
         * @return {?}
         */
        get validator() {
            return composeValidators(this._rawValidators);
        }
        /**
         * \@description
         * Async validator function composed of all the async validators registered with this
         * directive.
         * @return {?}
         */
        get asyncValidator() {
            return (/** @type {?} */ (composeAsyncValidators(this._rawAsyncValidators)));
        }
        /**
         * @private
         * @return {?}
         */
        _checkParentType() {
            if (!(this._parent instanceof FormGroupName) &&
                this._parent instanceof AbstractFormGroupDirective) {
                ReactiveErrors.ngModelGroupException();
            }
            else if (!(this._parent instanceof FormGroupName) && !(this._parent instanceof FormGroupDirective) &&
                !(this._parent instanceof FormArrayName)) {
                ReactiveErrors.controlParentException();
            }
        }
        /**
         * @private
         * @return {?}
         */
        _setUpControl() {
            this._checkParentType();
            ((/** @type {?} */ (this))).control = this.formDirective.addControl(this);
            if (this.control.disabled && (/** @type {?} */ (this.valueAccessor)).setDisabledState) {
                (/** @type {?} */ ((/** @type {?} */ (this.valueAccessor)).setDisabledState))(true);
            }
            this._added = true;
        }
    }
    FormControlName.ɵfac = function FormControlName_Factory(t) { return new (t || FormControlName)(ɵɵdirectiveInject(ControlContainer, 13), ɵɵdirectiveInject(NG_VALIDATORS, 10), ɵɵdirectiveInject(NG_ASYNC_VALIDATORS, 10), ɵɵdirectiveInject(NG_VALUE_ACCESSOR, 10), ɵɵdirectiveInject(NG_MODEL_WITH_FORM_CONTROL_WARNING, 8)); };
    FormControlName.ɵdir = ɵɵdefineDirective({ type: FormControlName, selectors: [["", "formControlName", ""]], inputs: { isDisabled: ["disabled", "isDisabled"], name: ["formControlName", "name"], model: ["ngModel", "model"] }, outputs: { update: "ngModelChange" }, features: [ɵɵProvidersFeature([controlNameBinding]), ɵɵInheritDefinitionFeature, ɵɵNgOnChangesFeature] });
    /**
     * \@description
     * Static property used to track whether any ngModel warnings have been sent across
     * all instances of FormControlName. Used to support warning config of "once".
     *
     * \@internal
     */
    FormControlName._ngModelWarningSentOnce = false;
    /** @nocollapse */
    FormControlName.ctorParameters = () => [
        { type: ControlContainer, decorators: [{ type: Optional }, { type: Host }, { type: SkipSelf }] },
        { type: Array, decorators: [{ type: Optional }, { type: Self }, { type: Inject, args: [NG_VALIDATORS,] }] },
        { type: Array, decorators: [{ type: Optional }, { type: Self }, { type: Inject, args: [NG_ASYNC_VALIDATORS,] }] },
        { type: Array, decorators: [{ type: Optional }, { type: Self }, { type: Inject, args: [NG_VALUE_ACCESSOR,] }] },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [NG_MODEL_WITH_FORM_CONTROL_WARNING,] }] }
    ];
    FormControlName.propDecorators = {
        name: [{ type: Input, args: ['formControlName',] }],
        isDisabled: [{ type: Input, args: ['disabled',] }],
        model: [{ type: Input, args: ['ngModel',] }],
        update: [{ type: Output, args: ['ngModelChange',] }]
    };
    /**
     * \@description
     * Provider which adds `RequiredValidator` to the `NG_VALIDATORS` multi-provider list.
     * @type {?}
     */
    const REQUIRED_VALIDATOR = {
        provide: NG_VALIDATORS,
        useExisting: forwardRef((/**
         * @return {?}
         */
        () => RequiredValidator)),
        multi: true
    };
    /**
     * \@description
     * Provider which adds `CheckboxRequiredValidator` to the `NG_VALIDATORS` multi-provider list.
     * @type {?}
     */
    const CHECKBOX_REQUIRED_VALIDATOR = {
        provide: NG_VALIDATORS,
        useExisting: forwardRef((/**
         * @return {?}
         */
        () => CheckboxRequiredValidator)),
        multi: true
    };
    /**
     * \@description
     * A directive that adds the `required` validator to any controls marked with the
     * `required` attribute. The directive is provided with the `NG_VALIDATORS` multi-provider list.
     *
     * @see [Form Validation](guide/form-validation)
     *
     * \@usageNotes
     *
     * ### Adding a required validator using template-driven forms
     *
     * ```
     * <input name="fullName" ngModel required>
     * ```
     *
     * \@ngModule FormsModule
     * \@ngModule ReactiveFormsModule
     * \@publicApi
     */
    class RequiredValidator {
        /**
         * \@description
         * Tracks changes to the required attribute bound to this directive.
         * @return {?}
         */
        get required() {
            return this._required;
        }
        /**
         * @param {?} value
         * @return {?}
         */
        set required(value) {
            this._required = value != null && value !== false && `${value}` !== 'false';
            if (this._onChange)
                this._onChange();
        }
        /**
         * \@description
         * Method that validates whether the control is empty.
         * Returns the validation result if enabled, otherwise null.
         * @param {?} control
         * @return {?}
         */
        validate(control) {
            return this.required ? Validators.required(control) : null;
        }
        /**
         * \@description
         * Registers a callback function to call when the validator inputs change.
         *
         * @param {?} fn The callback function
         * @return {?}
         */
        registerOnValidatorChange(fn) {
            this._onChange = fn;
        }
    }
    RequiredValidator.ɵfac = function RequiredValidator_Factory(t) { return new (t || RequiredValidator)(); };
    RequiredValidator.ɵdir = ɵɵdefineDirective({ type: RequiredValidator, selectors: [["", "required", "", "formControlName", "", 3, "type", "checkbox"], ["", "required", "", "formControl", "", 3, "type", "checkbox"], ["", "required", "", "ngModel", "", 3, "type", "checkbox"]], hostVars: 1, hostBindings: function RequiredValidator_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵattribute("required", ctx.required ? "" : null);
        } }, inputs: { required: "required" }, features: [ɵɵProvidersFeature([REQUIRED_VALIDATOR])] });
    RequiredValidator.propDecorators = {
        required: [{ type: Input }]
    };
    /**
     * A Directive that adds the `required` validator to checkbox controls marked with the
     * `required` attribute. The directive is provided with the `NG_VALIDATORS` multi-provider list.
     *
     * @see [Form Validation](guide/form-validation)
     *
     * \@usageNotes
     *
     * ### Adding a required checkbox validator using template-driven forms
     *
     * The following example shows how to add a checkbox required validator to an input attached to an
     * ngModel binding.
     *
     * ```
     * <input type="checkbox" name="active" ngModel required>
     * ```
     *
     * \@publicApi
     * \@ngModule FormsModule
     * \@ngModule ReactiveFormsModule
     */
    class CheckboxRequiredValidator extends RequiredValidator {
        /**
         * \@description
         * Method that validates whether or not the checkbox has been checked.
         * Returns the validation result if enabled, otherwise null.
         * @param {?} control
         * @return {?}
         */
        validate(control) {
            return this.required ? Validators.requiredTrue(control) : null;
        }
    }
    CheckboxRequiredValidator.ɵfac = function CheckboxRequiredValidator_Factory(t) { return ɵCheckboxRequiredValidator_BaseFactory(t || CheckboxRequiredValidator); };
    CheckboxRequiredValidator.ɵdir = ɵɵdefineDirective({ type: CheckboxRequiredValidator, selectors: [["input", "type", "checkbox", "required", "", "formControlName", ""], ["input", "type", "checkbox", "required", "", "formControl", ""], ["input", "type", "checkbox", "required", "", "ngModel", ""]], hostVars: 1, hostBindings: function CheckboxRequiredValidator_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵattribute("required", ctx.required ? "" : null);
        } }, features: [ɵɵProvidersFeature([CHECKBOX_REQUIRED_VALIDATOR]), ɵɵInheritDefinitionFeature] });
    const ɵCheckboxRequiredValidator_BaseFactory = ɵɵgetInheritedFactory(CheckboxRequiredValidator);
    /**
     * \@description
     * Provider which adds `EmailValidator` to the `NG_VALIDATORS` multi-provider list.
     * @type {?}
     */
    const EMAIL_VALIDATOR = {
        provide: NG_VALIDATORS,
        useExisting: forwardRef((/**
         * @return {?}
         */
        () => EmailValidator)),
        multi: true
    };
    /**
     * A directive that adds the `email` validator to controls marked with the
     * `email` attribute. The directive is provided with the `NG_VALIDATORS` multi-provider list.
     *
     * @see [Form Validation](guide/form-validation)
     *
     * \@usageNotes
     *
     * ### Adding an email validator
     *
     * The following example shows how to add an email validator to an input attached to an ngModel
     * binding.
     *
     * ```
     * <input type="email" name="email" ngModel email>
     * <input type="email" name="email" ngModel email="true">
     * <input type="email" name="email" ngModel [email]="true">
     * ```
     *
     * \@publicApi
     * \@ngModule FormsModule
     * \@ngModule ReactiveFormsModule
     */
    class EmailValidator {
        /**
         * \@description
         * Tracks changes to the email attribute bound to this directive.
         * @param {?} value
         * @return {?}
         */
        set email(value) {
            this._enabled = value === '' || value === true || value === 'true';
            if (this._onChange)
                this._onChange();
        }
        /**
         * \@description
         * Method that validates whether an email address is valid.
         * Returns the validation result if enabled, otherwise null.
         * @param {?} control
         * @return {?}
         */
        validate(control) {
            return this._enabled ? Validators.email(control) : null;
        }
        /**
         * \@description
         * Registers a callback function to call when the validator inputs change.
         *
         * @param {?} fn The callback function
         * @return {?}
         */
        registerOnValidatorChange(fn) {
            this._onChange = fn;
        }
    }
    EmailValidator.ɵfac = function EmailValidator_Factory(t) { return new (t || EmailValidator)(); };
    EmailValidator.ɵdir = ɵɵdefineDirective({ type: EmailValidator, selectors: [["", "email", "", "formControlName", ""], ["", "email", "", "formControl", ""], ["", "email", "", "ngModel", ""]], inputs: { email: "email" }, features: [ɵɵProvidersFeature([EMAIL_VALIDATOR])] });
    EmailValidator.propDecorators = {
        email: [{ type: Input }]
    };
    /**
     * \@description
     * Provider which adds `MinLengthValidator` to the `NG_VALIDATORS` multi-provider list.
     * @type {?}
     */
    const MIN_LENGTH_VALIDATOR = {
        provide: NG_VALIDATORS,
        useExisting: forwardRef((/**
         * @return {?}
         */
        () => MinLengthValidator)),
        multi: true
    };
    /**
     * A directive that adds minimum length validation to controls marked with the
     * `minlength` attribute. The directive is provided with the `NG_VALIDATORS` multi-provider list.
     *
     * @see [Form Validation](guide/form-validation)
     *
     * \@usageNotes
     *
     * ### Adding a minimum length validator
     *
     * The following example shows how to add a minimum length validator to an input attached to an
     * ngModel binding.
     *
     * ```html
     * <input name="firstName" ngModel minlength="4">
     * ```
     *
     * \@ngModule ReactiveFormsModule
     * \@ngModule FormsModule
     * \@publicApi
     */
    class MinLengthValidator {
        /**
         * \@description
         * A lifecycle method called when the directive's inputs change. For internal use
         * only.
         *
         * @param {?} changes A object of key/value pairs for the set of changed inputs.
         * @return {?}
         */
        ngOnChanges(changes) {
            if ('minlength' in changes) {
                this._createValidator();
                if (this._onChange)
                    this._onChange();
            }
        }
        /**
         * \@description
         * Method that validates whether the value meets a minimum length
         * requirement. Returns the validation result if enabled, otherwise null.
         * @param {?} control
         * @return {?}
         */
        validate(control) {
            return this.minlength == null ? null : this._validator(control);
        }
        /**
         * \@description
         * Registers a callback function to call when the validator inputs change.
         *
         * @param {?} fn The callback function
         * @return {?}
         */
        registerOnValidatorChange(fn) {
            this._onChange = fn;
        }
        /**
         * @private
         * @return {?}
         */
        _createValidator() {
            this._validator = Validators.minLength(typeof this.minlength === 'number' ? this.minlength : parseInt(this.minlength, 10));
        }
    }
    MinLengthValidator.ɵfac = function MinLengthValidator_Factory(t) { return new (t || MinLengthValidator)(); };
    MinLengthValidator.ɵdir = ɵɵdefineDirective({ type: MinLengthValidator, selectors: [["", "minlength", "", "formControlName", ""], ["", "minlength", "", "formControl", ""], ["", "minlength", "", "ngModel", ""]], hostVars: 1, hostBindings: function MinLengthValidator_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵattribute("minlength", ctx.minlength ? ctx.minlength : null);
        } }, inputs: { minlength: "minlength" }, features: [ɵɵProvidersFeature([MIN_LENGTH_VALIDATOR]), ɵɵNgOnChangesFeature] });
    MinLengthValidator.propDecorators = {
        minlength: [{ type: Input }]
    };
    /**
     * \@description
     * Provider which adds `MaxLengthValidator` to the `NG_VALIDATORS` multi-provider list.
     * @type {?}
     */
    const MAX_LENGTH_VALIDATOR = {
        provide: NG_VALIDATORS,
        useExisting: forwardRef((/**
         * @return {?}
         */
        () => MaxLengthValidator)),
        multi: true
    };
    /**
     * A directive that adds max length validation to controls marked with the
     * `maxlength` attribute. The directive is provided with the `NG_VALIDATORS` multi-provider list.
     *
     * @see [Form Validation](guide/form-validation)
     *
     * \@usageNotes
     *
     * ### Adding a maximum length validator
     *
     * The following example shows how to add a maximum length validator to an input attached to an
     * ngModel binding.
     *
     * ```html
     * <input name="firstName" ngModel maxlength="25">
     * ```
     *
     * \@ngModule ReactiveFormsModule
     * \@ngModule FormsModule
     * \@publicApi
     */
    class MaxLengthValidator {
        /**
         * \@description
         * A lifecycle method called when the directive's inputs change. For internal use
         * only.
         *
         * @param {?} changes A object of key/value pairs for the set of changed inputs.
         * @return {?}
         */
        ngOnChanges(changes) {
            if ('maxlength' in changes) {
                this._createValidator();
                if (this._onChange)
                    this._onChange();
            }
        }
        /**
         * \@description
         * Method that validates whether the value exceeds
         * the maximum length requirement.
         * @param {?} control
         * @return {?}
         */
        validate(control) {
            return this.maxlength != null ? this._validator(control) : null;
        }
        /**
         * \@description
         * Registers a callback function to call when the validator inputs change.
         *
         * @param {?} fn The callback function
         * @return {?}
         */
        registerOnValidatorChange(fn) {
            this._onChange = fn;
        }
        /**
         * @private
         * @return {?}
         */
        _createValidator() {
            this._validator = Validators.maxLength(typeof this.maxlength === 'number' ? this.maxlength : parseInt(this.maxlength, 10));
        }
    }
    MaxLengthValidator.ɵfac = function MaxLengthValidator_Factory(t) { return new (t || MaxLengthValidator)(); };
    MaxLengthValidator.ɵdir = ɵɵdefineDirective({ type: MaxLengthValidator, selectors: [["", "maxlength", "", "formControlName", ""], ["", "maxlength", "", "formControl", ""], ["", "maxlength", "", "ngModel", ""]], hostVars: 1, hostBindings: function MaxLengthValidator_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵattribute("maxlength", ctx.maxlength ? ctx.maxlength : null);
        } }, inputs: { maxlength: "maxlength" }, features: [ɵɵProvidersFeature([MAX_LENGTH_VALIDATOR]), ɵɵNgOnChangesFeature] });
    MaxLengthValidator.propDecorators = {
        maxlength: [{ type: Input }]
    };
    /**
     * \@description
     * Provider which adds `PatternValidator` to the `NG_VALIDATORS` multi-provider list.
     * @type {?}
     */
    const PATTERN_VALIDATOR = {
        provide: NG_VALIDATORS,
        useExisting: forwardRef((/**
         * @return {?}
         */
        () => PatternValidator)),
        multi: true
    };
    /**
     * \@description
     * A directive that adds regex pattern validation to controls marked with the
     * `pattern` attribute. The regex must match the entire control value.
     * The directive is provided with the `NG_VALIDATORS` multi-provider list.
     *
     * @see [Form Validation](guide/form-validation)
     *
     * \@usageNotes
     *
     * ### Adding a pattern validator
     *
     * The following example shows how to add a pattern validator to an input attached to an
     * ngModel binding.
     *
     * ```html
     * <input name="firstName" ngModel pattern="[a-zA-Z ]*">
     * ```
     *
     * \@ngModule ReactiveFormsModule
     * \@ngModule FormsModule
     * \@publicApi
     */
    class PatternValidator {
        /**
         * \@description
         * A lifecycle method called when the directive's inputs change. For internal use
         * only.
         *
         * @param {?} changes A object of key/value pairs for the set of changed inputs.
         * @return {?}
         */
        ngOnChanges(changes) {
            if ('pattern' in changes) {
                this._createValidator();
                if (this._onChange)
                    this._onChange();
            }
        }
        /**
         * \@description
         * Method that validates whether the value matches the
         * the pattern requirement.
         * @param {?} control
         * @return {?}
         */
        validate(control) {
            return this._validator(control);
        }
        /**
         * \@description
         * Registers a callback function to call when the validator inputs change.
         *
         * @param {?} fn The callback function
         * @return {?}
         */
        registerOnValidatorChange(fn) {
            this._onChange = fn;
        }
        /**
         * @private
         * @return {?}
         */
        _createValidator() {
            this._validator = Validators.pattern(this.pattern);
        }
    }
    PatternValidator.ɵfac = function PatternValidator_Factory(t) { return new (t || PatternValidator)(); };
    PatternValidator.ɵdir = ɵɵdefineDirective({ type: PatternValidator, selectors: [["", "pattern", "", "formControlName", ""], ["", "pattern", "", "formControl", ""], ["", "pattern", "", "ngModel", ""]], hostVars: 1, hostBindings: function PatternValidator_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵattribute("pattern", ctx.pattern ? ctx.pattern : null);
        } }, inputs: { pattern: "pattern" }, features: [ɵɵProvidersFeature([PATTERN_VALIDATOR]), ɵɵNgOnChangesFeature] });
    PatternValidator.propDecorators = {
        pattern: [{ type: Input }]
    };
    /**
     * Internal module used for sharing directives between FormsModule and ReactiveFormsModule
     */
    class ɵInternalFormsSharedModule {
    }
    ɵInternalFormsSharedModule.ɵmod = ɵɵdefineNgModule({ type: ɵInternalFormsSharedModule });
    ɵInternalFormsSharedModule.ɵinj = ɵɵdefineInjector({ factory: function ɵInternalFormsSharedModule_Factory(t) { return new (t || ɵInternalFormsSharedModule)(); } });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(ɵInternalFormsSharedModule, { declarations: [ɵNgNoValidate,
            NgSelectOption,
            ɵNgSelectMultipleOption,
            DefaultValueAccessor,
            NumberValueAccessor,
            RangeValueAccessor,
            CheckboxControlValueAccessor,
            SelectControlValueAccessor,
            SelectMultipleControlValueAccessor,
            RadioControlValueAccessor,
            NgControlStatus,
            NgControlStatusGroup,
            RequiredValidator,
            MinLengthValidator,
            MaxLengthValidator,
            PatternValidator,
            CheckboxRequiredValidator,
            EmailValidator], exports: [ɵNgNoValidate,
            NgSelectOption,
            ɵNgSelectMultipleOption,
            DefaultValueAccessor,
            NumberValueAccessor,
            RangeValueAccessor,
            CheckboxControlValueAccessor,
            SelectControlValueAccessor,
            SelectMultipleControlValueAccessor,
            RadioControlValueAccessor,
            NgControlStatus,
            NgControlStatusGroup,
            RequiredValidator,
            MinLengthValidator,
            MaxLengthValidator,
            PatternValidator,
            CheckboxRequiredValidator,
            EmailValidator] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/forms/src/form_builder.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @param {?} options
     * @return {?}
     */
    function isAbstractControlOptions(options) {
        return ((/** @type {?} */ (options))).asyncValidators !== undefined ||
            ((/** @type {?} */ (options))).validators !== undefined ||
            ((/** @type {?} */ (options))).updateOn !== undefined;
    }
    /**
     * \@description
     * Creates an `AbstractControl` from a user-specified configuration.
     *
     * The `FormBuilder` provides syntactic sugar that shortens creating instances of a `FormControl`,
     * `FormGroup`, or `FormArray`. It reduces the amount of boilerplate needed to build complex
     * forms.
     *
     * @see [Reactive Forms Guide](/guide/reactive-forms)
     *
     * \@publicApi
     */
    class FormBuilder {
        /**
         * \@description
         * Construct a new `FormGroup` instance.
         *
         * @param {?} controlsConfig A collection of child controls. The key for each child is the name
         * under which it is registered.
         *
         * @param {?=} options Configuration options object for the `FormGroup`. The object can
         * have two shapes:
         *
         * 1) `AbstractControlOptions` object (preferred), which consists of:
         * * `validators`: A synchronous validator function, or an array of validator functions
         * * `asyncValidators`: A single async validator or array of async validator functions
         * * `updateOn`: The event upon which the control should be updated (options: 'change' | 'blur' |
         * submit')
         *
         * 2) Legacy configuration object, which consists of:
         * * `validator`: A synchronous validator function, or an array of validator functions
         * * `asyncValidator`: A single async validator or array of async validator functions
         *
         * @return {?}
         */
        group(controlsConfig, options = null) {
            /** @type {?} */
            const controls = this._reduceControls(controlsConfig);
            /** @type {?} */
            let validators = null;
            /** @type {?} */
            let asyncValidators = null;
            /** @type {?} */
            let updateOn = undefined;
            if (options != null) {
                if (isAbstractControlOptions(options)) {
                    // `options` are `AbstractControlOptions`
                    validators = options.validators != null ? options.validators : null;
                    asyncValidators = options.asyncValidators != null ? options.asyncValidators : null;
                    updateOn = options.updateOn != null ? options.updateOn : undefined;
                }
                else {
                    // `options` are legacy form group options
                    validators = options['validator'] != null ? options['validator'] : null;
                    asyncValidators = options['asyncValidator'] != null ? options['asyncValidator'] : null;
                }
            }
            return new FormGroup(controls, { asyncValidators, updateOn, validators });
        }
        /**
         * \@description
         * Construct a new `FormControl` with the given state, validators and options.
         *
         * \@usageNotes
         *
         * ### Initialize a control as disabled
         *
         * The following example returns a control with an initial value in a disabled state.
         *
         * <code-example path="forms/ts/formBuilder/form_builder_example.ts" region="disabled-control">
         * </code-example>
         * @param {?} formState Initializes the control with an initial state value, or
         * with an object that contains both a value and a disabled status.
         *
         * @param {?=} validatorOrOpts A synchronous validator function, or an array of
         * such functions, or an `AbstractControlOptions` object that contains
         * validation functions and a validation trigger.
         *
         * @param {?=} asyncValidator A single async validator or array of async validator
         * functions.
         *
         * @return {?}
         */
        control(formState, validatorOrOpts, asyncValidator) {
            return new FormControl(formState, validatorOrOpts, asyncValidator);
        }
        /**
         * Constructs a new `FormArray` from the given array of configurations,
         * validators and options.
         *
         * @param {?} controlsConfig An array of child controls or control configs. Each
         * child control is given an index when it is registered.
         *
         * @param {?=} validatorOrOpts A synchronous validator function, or an array of
         * such functions, or an `AbstractControlOptions` object that contains
         * validation functions and a validation trigger.
         *
         * @param {?=} asyncValidator A single async validator or array of async validator
         * functions.
         * @return {?}
         */
        array(controlsConfig, validatorOrOpts, asyncValidator) {
            /** @type {?} */
            const controls = controlsConfig.map((/**
             * @param {?} c
             * @return {?}
             */
            c => this._createControl(c)));
            return new FormArray(controls, validatorOrOpts, asyncValidator);
        }
        /**
         * \@internal
         * @param {?} controlsConfig
         * @return {?}
         */
        _reduceControls(controlsConfig) {
            /** @type {?} */
            const controls = {};
            Object.keys(controlsConfig).forEach((/**
             * @param {?} controlName
             * @return {?}
             */
            controlName => {
                controls[controlName] = this._createControl(controlsConfig[controlName]);
            }));
            return controls;
        }
        /**
         * \@internal
         * @param {?} controlConfig
         * @return {?}
         */
        _createControl(controlConfig) {
            if (controlConfig instanceof FormControl || controlConfig instanceof FormGroup ||
                controlConfig instanceof FormArray) {
                return controlConfig;
            }
            else if (Array.isArray(controlConfig)) {
                /** @type {?} */
                const value = controlConfig[0];
                /** @type {?} */
                const validator = controlConfig.length > 1 ? controlConfig[1] : null;
                /** @type {?} */
                const asyncValidator = controlConfig.length > 2 ? controlConfig[2] : null;
                return this.control(value, validator, asyncValidator);
            }
            else {
                return this.control(controlConfig);
            }
        }
    }
    FormBuilder.ɵfac = function FormBuilder_Factory(t) { return new (t || FormBuilder)(); };
    FormBuilder.ɵprov = ɵɵdefineInjectable({ token: FormBuilder, factory: FormBuilder.ɵfac });

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/forms/src/version.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * \@publicApi
     * @type {?}
     */
    const VERSION$5 = new Version('9.1.12');

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/forms/src/form_providers.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Exports the required providers and directives for template-driven forms,
     * making them available for import by NgModules that import this module.
     *
     * @see [Forms Overview](/guide/forms-overview)
     * @see [Template-driven Forms Guide](/guide/forms)
     *
     * \@publicApi
     */
    class FormsModule {
    }
    FormsModule.ɵmod = ɵɵdefineNgModule({ type: FormsModule });
    FormsModule.ɵinj = ɵɵdefineInjector({ factory: function FormsModule_Factory(t) { return new (t || FormsModule)(); }, providers: [RadioControlRegistry], imports: [ɵInternalFormsSharedModule] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(FormsModule, { declarations: [NgModel,
            NgModelGroup,
            NgForm], exports: [ɵInternalFormsSharedModule,
            NgModel,
            NgModelGroup,
            NgForm] }); })();
    /**
     * Exports the required infrastructure and directives for reactive forms,
     * making them available for import by NgModules that import this module.
     *
     * @see [Forms Overview](guide/forms-overview)
     * @see [Reactive Forms Guide](guide/reactive-forms)
     *
     * \@publicApi
     */
    class ReactiveFormsModule {
        /**
         * \@description
         * Provides options for configuring the reactive forms module.
         *
         * @param {?} opts An object of configuration options
         * * `warnOnNgModelWithFormControl` Configures when to emit a warning when an `ngModel`
         * binding is used with reactive form directives.
         * @return {?}
         */
        static withConfig(opts) {
            return {
                ngModule: ReactiveFormsModule,
                providers: [
                    { provide: NG_MODEL_WITH_FORM_CONTROL_WARNING, useValue: opts.warnOnNgModelWithFormControl }
                ]
            };
        }
    }
    ReactiveFormsModule.ɵmod = ɵɵdefineNgModule({ type: ReactiveFormsModule });
    ReactiveFormsModule.ɵinj = ɵɵdefineInjector({ factory: function ReactiveFormsModule_Factory(t) { return new (t || ReactiveFormsModule)(); }, providers: [FormBuilder, RadioControlRegistry], imports: [ɵInternalFormsSharedModule] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(ReactiveFormsModule, { declarations: [FormControlDirective,
            FormGroupDirective,
            FormControlName,
            FormGroupName,
            FormArrayName], exports: [ɵInternalFormsSharedModule,
            FormControlDirective,
            FormGroupDirective,
            FormControlName,
            FormGroupName,
            FormArrayName] }); })();

    const _c0$5 = ["trigger"];
    const _c1$5 = ["panel"];
    function MatSelect_span_4_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "span", 8);
        ɵɵtext(1);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r2 = ɵɵnextContext();
        ɵɵadvance(1);
        ɵɵtextInterpolate(ctx_r2.placeholder || "\u00A0");
    } }
    function MatSelect_span_5_span_1_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "span");
        ɵɵtext(1);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r5 = ɵɵnextContext(2);
        ɵɵadvance(1);
        ɵɵtextInterpolate(ctx_r5.triggerValue || "\u00A0");
    } }
    function MatSelect_span_5_2_Template(rf, ctx) { if (rf & 1) {
        ɵɵprojection(0, 0, ["*ngSwitchCase", "true"]);
    } }
    function MatSelect_span_5_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "span", 9);
        ɵɵtemplate(1, MatSelect_span_5_span_1_Template, 2, 1, "span", 10);
        ɵɵtemplate(2, MatSelect_span_5_2_Template, 1, 0, undefined, 11);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r3 = ɵɵnextContext();
        ɵɵproperty("ngSwitch", !!ctx_r3.customTrigger);
        ɵɵadvance(2);
        ɵɵproperty("ngSwitchCase", true);
    } }
    function MatSelect_ng_template_8_Template(rf, ctx) { if (rf & 1) {
        const _r9 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "div", 12);
        ɵɵelementStart(1, "div", 13, 14);
        ɵɵlistener("@transformPanel.done", function MatSelect_ng_template_8_Template_div_animation_transformPanel_done_1_listener($event) { ɵɵrestoreView(_r9); const ctx_r8 = ɵɵnextContext(); return ctx_r8._panelDoneAnimatingStream.next($event.toState); })("keydown", function MatSelect_ng_template_8_Template_div_keydown_1_listener($event) { ɵɵrestoreView(_r9); const ctx_r10 = ɵɵnextContext(); return ctx_r10._handleKeydown($event); });
        ɵɵprojection(3, 1);
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r4 = ɵɵnextContext();
        ɵɵproperty("@transformPanelWrap", undefined);
        ɵɵadvance(1);
        ɵɵclassMapInterpolate1("mat-select-panel ", ctx_r4._getPanelTheme(), "");
        ɵɵstyleProp("transform-origin", ctx_r4._transformOrigin)("font-size", ctx_r4._triggerFontSize, "px");
        ɵɵproperty("ngClass", ctx_r4.panelClass)("@transformPanel", ctx_r4.multiple ? "showing-multiple" : "showing");
        ɵɵattribute("id", ctx_r4.id + "-panel");
    } }
    const _c2$3 = [[["mat-select-trigger"]], "*"];
    const _c3$2 = ["mat-select-trigger", "*"];
    const matSelectAnimations = {
        /**
         * This animation ensures the select's overlay panel animation (transformPanel) is called when
         * closing the select.
         * This is needed due to https://github.com/angular/angular/issues/23302
         */
        transformPanelWrap: trigger('transformPanelWrap', [
            transition('* => void', query('@transformPanel', [animateChild()], { optional: true }))
        ]),
        /**
         * This animation transforms the select's overlay panel on and off the page.
         *
         * When the panel is attached to the DOM, it expands its width by the amount of padding, scales it
         * up to 100% on the Y axis, fades in its border, and translates slightly up and to the
         * side to ensure the option text correctly overlaps the trigger text.
         *
         * When the panel is removed from the DOM, it simply fades out linearly.
         */
        transformPanel: trigger('transformPanel', [
            state('void', style({
                transform: 'scaleY(0.8)',
                minWidth: '100%',
                opacity: 0
            })),
            state('showing', style({
                opacity: 1,
                minWidth: 'calc(100% + 32px)',
                // 32px = 2 * 16px padding
                transform: 'scaleY(1)'
            })),
            state('showing-multiple', style({
                opacity: 1,
                minWidth: 'calc(100% + 64px)',
                // 64px = 48px padding on the left + 16px padding on the right
                transform: 'scaleY(1)'
            })),
            transition('void => *', animate('120ms cubic-bezier(0, 0, 0.2, 1)')),
            transition('* => void', animate('100ms 25ms linear', style({ opacity: 0 })))
        ])
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/select/select-errors.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google LLC All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * Returns an exception to be thrown when attempting to change a select's `multiple` option
     * after initialization.
     * \@docs-private
     * @return {?}
     */
    function getMatSelectDynamicMultipleError() {
        return Error('Cannot change `multiple` mode of select after initialization.');
    }
    /**
     * Returns an exception to be thrown when attempting to assign a non-array value to a select
     * in `multiple` mode. Note that `undefined` and `null` are still valid values to allow for
     * resetting the value.
     * \@docs-private
     * @return {?}
     */
    function getMatSelectNonArrayValueError() {
        return Error('Value must be an array in multiple-selection mode.');
    }
    /**
     * Returns an exception to be thrown when assigning a non-function value to the comparator
     * used to determine if a value corresponds to an option. Note that whether the function
     * actually takes two values and returns a boolean is not checked.
     * @return {?}
     */
    function getMatSelectNonFunctionValueError() {
        return Error('`compareWith` must be a function.');
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/select/select.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    let nextUniqueId$4 = 0;
    /**
     * The max height of the select's overlay panel
     * @type {?}
     */
    const SELECT_PANEL_MAX_HEIGHT = 256;
    /**
     * The panel's padding on the x-axis
     * @type {?}
     */
    const SELECT_PANEL_PADDING_X = 16;
    /**
     * The panel's x axis padding if it is indented (e.g. there is an option group).
     * @type {?}
     */
    const SELECT_PANEL_INDENT_PADDING_X = SELECT_PANEL_PADDING_X * 2;
    /**
     * The height of the select items in `em` units.
     * @type {?}
     */
    const SELECT_ITEM_HEIGHT_EM = 3;
    // TODO(josephperrott): Revert to a constant after 2018 spec updates are fully merged.
    /**
     * Distance between the panel edge and the option text in
     * multi-selection mode.
     *
     * Calculated as:
     * (SELECT_PANEL_PADDING_X * 1.5) + 16 = 40
     * The padding is multiplied by 1.5 because the checkbox's margin is half the padding.
     * The checkbox width is 16px.
     * @type {?}
     */
    const SELECT_MULTIPLE_PANEL_PADDING_X = SELECT_PANEL_PADDING_X * 1.5 + 16;
    /**
     * The select panel will only "fit" inside the viewport if it is positioned at
     * this value or more away from the viewport boundary.
     * @type {?}
     */
    const SELECT_PANEL_VIEWPORT_PADDING = 8;
    /**
     * Injection token that determines the scroll handling while a select is open.
     * @type {?}
     */
    const MAT_SELECT_SCROLL_STRATEGY = new InjectionToken('mat-select-scroll-strategy');
    /**
     * \@docs-private
     * @param {?} overlay
     * @return {?}
     */
    function MAT_SELECT_SCROLL_STRATEGY_PROVIDER_FACTORY(overlay) {
        return (/**
         * @return {?}
         */
        () => overlay.scrollStrategies.reposition());
    }
    /**
     * Injection token that can be used to provide the default options the select module.
     * @type {?}
     */
    const MAT_SELECT_CONFIG = new InjectionToken('MAT_SELECT_CONFIG');
    /**
     * \@docs-private
     * @type {?}
     */
    const MAT_SELECT_SCROLL_STRATEGY_PROVIDER = {
        provide: MAT_SELECT_SCROLL_STRATEGY,
        deps: [Overlay],
        useFactory: MAT_SELECT_SCROLL_STRATEGY_PROVIDER_FACTORY,
    };
    /**
     * Change event object that is emitted when the select value has changed.
     */
    class MatSelectChange {
        /**
         * @param {?} source
         * @param {?} value
         */
        constructor(source, value) {
            this.source = source;
            this.value = value;
        }
    }
    // Boilerplate for applying mixins to MatSelect.
    /**
     * \@docs-private
     */
    class MatSelectBase {
        /**
         * @param {?} _elementRef
         * @param {?} _defaultErrorStateMatcher
         * @param {?} _parentForm
         * @param {?} _parentFormGroup
         * @param {?} ngControl
         */
        constructor(_elementRef, _defaultErrorStateMatcher, _parentForm, _parentFormGroup, ngControl) {
            this._elementRef = _elementRef;
            this._defaultErrorStateMatcher = _defaultErrorStateMatcher;
            this._parentForm = _parentForm;
            this._parentFormGroup = _parentFormGroup;
            this.ngControl = ngControl;
        }
    }
    /** @type {?} */
    const _MatSelectMixinBase = mixinDisableRipple(mixinTabIndex(mixinDisabled(mixinErrorState(MatSelectBase))));
    /**
     * Allows the user to customize the trigger that is displayed when the select has a value.
     */
    class MatSelectTrigger {
    }
    MatSelectTrigger.ɵfac = function MatSelectTrigger_Factory(t) { return new (t || MatSelectTrigger)(); };
    MatSelectTrigger.ɵdir = ɵɵdefineDirective({ type: MatSelectTrigger, selectors: [["mat-select-trigger"]] });
    class MatSelect extends _MatSelectMixinBase {
        /**
         * @param {?} _viewportRuler
         * @param {?} _changeDetectorRef
         * @param {?} _ngZone
         * @param {?} _defaultErrorStateMatcher
         * @param {?} elementRef
         * @param {?} _dir
         * @param {?} _parentForm
         * @param {?} _parentFormGroup
         * @param {?} _parentFormField
         * @param {?} ngControl
         * @param {?} tabIndex
         * @param {?} scrollStrategyFactory
         * @param {?} _liveAnnouncer
         * @param {?=} defaults
         */
        constructor(_viewportRuler, _changeDetectorRef, _ngZone, _defaultErrorStateMatcher, elementRef, _dir, _parentForm, _parentFormGroup, _parentFormField, ngControl, tabIndex, scrollStrategyFactory, _liveAnnouncer, defaults) {
            super(elementRef, _defaultErrorStateMatcher, _parentForm, _parentFormGroup, ngControl);
            this._viewportRuler = _viewportRuler;
            this._changeDetectorRef = _changeDetectorRef;
            this._ngZone = _ngZone;
            this._dir = _dir;
            this._parentFormField = _parentFormField;
            this.ngControl = ngControl;
            this._liveAnnouncer = _liveAnnouncer;
            /**
             * Whether or not the overlay panel is open.
             */
            this._panelOpen = false;
            /**
             * Whether filling out the select is required in the form.
             */
            this._required = false;
            /**
             * The scroll position of the overlay panel, calculated to center the selected option.
             */
            this._scrollTop = 0;
            /**
             * Whether the component is in multiple selection mode.
             */
            this._multiple = false;
            /**
             * Comparison function to specify which option is displayed. Defaults to object equality.
             */
            this._compareWith = (/**
             * @param {?} o1
             * @param {?} o2
             * @return {?}
             */
            (o1, o2) => o1 === o2);
            /**
             * Unique id for this input.
             */
            this._uid = `mat-select-${nextUniqueId$4++}`;
            /**
             * Emits whenever the component is destroyed.
             */
            this._destroy = new Subject();
            /**
             * The cached font-size of the trigger element.
             */
            this._triggerFontSize = 0;
            /**
             * `View -> model callback called when value changes`
             */
            this._onChange = (/**
             * @return {?}
             */
            () => { });
            /**
             * `View -> model callback called when select has been touched`
             */
            this._onTouched = (/**
             * @return {?}
             */
            () => { });
            /**
             * The IDs of child options to be passed to the aria-owns attribute.
             */
            this._optionIds = '';
            /**
             * The value of the select panel's transform-origin property.
             */
            this._transformOrigin = 'top';
            /**
             * Emits when the panel element is finished transforming in.
             */
            this._panelDoneAnimatingStream = new Subject();
            /**
             * The y-offset of the overlay panel in relation to the trigger's top start corner.
             * This must be adjusted to align the selected option text over the trigger text.
             * when the panel opens. Will change based on the y-position of the selected option.
             */
            this._offsetY = 0;
            /**
             * This position config ensures that the top "start" corner of the overlay
             * is aligned with with the top "start" of the origin by default (overlapping
             * the trigger completely). If the panel cannot fit below the trigger, it
             * will fall back to a position above the trigger.
             */
            this._positions = [
                {
                    originX: 'start',
                    originY: 'top',
                    overlayX: 'start',
                    overlayY: 'top',
                },
                {
                    originX: 'start',
                    originY: 'bottom',
                    overlayX: 'start',
                    overlayY: 'bottom',
                },
            ];
            /**
             * Whether the component is disabling centering of the active option over the trigger.
             */
            this._disableOptionCentering = false;
            this._focused = false;
            /**
             * A name for this control that can be used by `mat-form-field`.
             */
            this.controlType = 'mat-select';
            /**
             * Aria label of the select. If not specified, the placeholder will be used as label.
             */
            this.ariaLabel = '';
            /**
             * Combined stream of all of the child options' change events.
             */
            this.optionSelectionChanges = (/** @type {?} */ (defer((/**
             * @return {?}
             */
            () => {
                /** @type {?} */
                const options = this.options;
                if (options) {
                    return options.changes.pipe(startWith(options), switchMap((/**
                     * @return {?}
                     */
                    () => merge(...options.map((/**
                     * @param {?} option
                     * @return {?}
                     */
                    option => option.onSelectionChange))))));
                }
                return this._ngZone.onStable
                    .asObservable()
                    .pipe(take(1), switchMap((/**
                 * @return {?}
                 */
                () => this.optionSelectionChanges)));
            }))));
            /**
             * Event emitted when the select panel has been toggled.
             */
            this.openedChange = new EventEmitter();
            /**
             * Event emitted when the select has been opened.
             */
            this._openedStream = this.openedChange.pipe(filter((/**
             * @param {?} o
             * @return {?}
             */
            o => o)), map((/**
             * @return {?}
             */
            () => { })));
            /**
             * Event emitted when the select has been closed.
             */
            this._closedStream = this.openedChange.pipe(filter((/**
             * @param {?} o
             * @return {?}
             */
            o => !o)), map((/**
             * @return {?}
             */
            () => { })));
            /**
             * Event emitted when the selected value has been changed by the user.
             */
            this.selectionChange = new EventEmitter();
            /**
             * Event that emits whenever the raw value of the select changes. This is here primarily
             * to facilitate the two-way binding for the `value` input.
             * \@docs-private
             */
            this.valueChange = new EventEmitter();
            if (this.ngControl) {
                // Note: we provide the value accessor through here, instead of
                // the `providers` to avoid running into a circular import.
                this.ngControl.valueAccessor = this;
            }
            this._scrollStrategyFactory = scrollStrategyFactory;
            this._scrollStrategy = this._scrollStrategyFactory();
            this.tabIndex = parseInt(tabIndex) || 0;
            // Force setter to be called in case id was not specified.
            this.id = this.id;
            if (defaults) {
                if (defaults.disableOptionCentering != null) {
                    this.disableOptionCentering = defaults.disableOptionCentering;
                }
                if (defaults.typeaheadDebounceInterval != null) {
                    this.typeaheadDebounceInterval = defaults.typeaheadDebounceInterval;
                }
            }
        }
        /**
         * Whether the select is focused.
         * @return {?}
         */
        get focused() {
            return this._focused || this._panelOpen;
        }
        /**
         * Placeholder to be shown if no value has been selected.
         * @return {?}
         */
        get placeholder() { return this._placeholder; }
        /**
         * @param {?} value
         * @return {?}
         */
        set placeholder(value) {
            this._placeholder = value;
            this.stateChanges.next();
        }
        /**
         * Whether the component is required.
         * @return {?}
         */
        get required() { return this._required; }
        /**
         * @param {?} value
         * @return {?}
         */
        set required(value) {
            this._required = coerceBooleanProperty(value);
            this.stateChanges.next();
        }
        /**
         * Whether the user should be allowed to select multiple options.
         * @return {?}
         */
        get multiple() { return this._multiple; }
        /**
         * @param {?} value
         * @return {?}
         */
        set multiple(value) {
            if (this._selectionModel) {
                throw getMatSelectDynamicMultipleError();
            }
            this._multiple = coerceBooleanProperty(value);
        }
        /**
         * Whether to center the active option over the trigger.
         * @return {?}
         */
        get disableOptionCentering() { return this._disableOptionCentering; }
        /**
         * @param {?} value
         * @return {?}
         */
        set disableOptionCentering(value) {
            this._disableOptionCentering = coerceBooleanProperty(value);
        }
        /**
         * Function to compare the option values with the selected values. The first argument
         * is a value from an option. The second is a value from the selection. A boolean
         * should be returned.
         * @return {?}
         */
        get compareWith() { return this._compareWith; }
        /**
         * @param {?} fn
         * @return {?}
         */
        set compareWith(fn) {
            if (typeof fn !== 'function') {
                throw getMatSelectNonFunctionValueError();
            }
            this._compareWith = fn;
            if (this._selectionModel) {
                // A different comparator means the selection could change.
                this._initializeSelection();
            }
        }
        /**
         * Value of the select control.
         * @return {?}
         */
        get value() { return this._value; }
        /**
         * @param {?} newValue
         * @return {?}
         */
        set value(newValue) {
            if (newValue !== this._value) {
                this.writeValue(newValue);
                this._value = newValue;
            }
        }
        /**
         * Time to wait in milliseconds after the last keystroke before moving focus to an item.
         * @return {?}
         */
        get typeaheadDebounceInterval() { return this._typeaheadDebounceInterval; }
        /**
         * @param {?} value
         * @return {?}
         */
        set typeaheadDebounceInterval(value) {
            this._typeaheadDebounceInterval = coerceNumberProperty(value);
        }
        /**
         * Unique id of the element.
         * @return {?}
         */
        get id() { return this._id; }
        /**
         * @param {?} value
         * @return {?}
         */
        set id(value) {
            this._id = value || this._uid;
            this.stateChanges.next();
        }
        /**
         * @return {?}
         */
        ngOnInit() {
            this._selectionModel = new SelectionModel(this.multiple);
            this.stateChanges.next();
            // We need `distinctUntilChanged` here, because some browsers will
            // fire the animation end event twice for the same animation. See:
            // https://github.com/angular/angular/issues/24084
            this._panelDoneAnimatingStream
                .pipe(distinctUntilChanged(), takeUntil(this._destroy))
                .subscribe((/**
             * @return {?}
             */
            () => {
                if (this.panelOpen) {
                    this._scrollTop = 0;
                    this.openedChange.emit(true);
                }
                else {
                    this.openedChange.emit(false);
                    this.overlayDir.offsetX = 0;
                    this._changeDetectorRef.markForCheck();
                }
            }));
            this._viewportRuler.change()
                .pipe(takeUntil(this._destroy))
                .subscribe((/**
             * @return {?}
             */
            () => {
                if (this._panelOpen) {
                    this._triggerRect = this.trigger.nativeElement.getBoundingClientRect();
                    this._changeDetectorRef.markForCheck();
                }
            }));
        }
        /**
         * @return {?}
         */
        ngAfterContentInit() {
            this._initKeyManager();
            this._selectionModel.changed.pipe(takeUntil(this._destroy)).subscribe((/**
             * @param {?} event
             * @return {?}
             */
            event => {
                event.added.forEach((/**
                 * @param {?} option
                 * @return {?}
                 */
                option => option.select()));
                event.removed.forEach((/**
                 * @param {?} option
                 * @return {?}
                 */
                option => option.deselect()));
            }));
            this.options.changes.pipe(startWith(null), takeUntil(this._destroy)).subscribe((/**
             * @return {?}
             */
            () => {
                this._resetOptions();
                this._initializeSelection();
            }));
        }
        /**
         * @return {?}
         */
        ngDoCheck() {
            if (this.ngControl) {
                this.updateErrorState();
            }
        }
        /**
         * @param {?} changes
         * @return {?}
         */
        ngOnChanges(changes) {
            // Updating the disabled state is handled by `mixinDisabled`, but we need to additionally let
            // the parent form field know to run change detection when the disabled state changes.
            if (changes['disabled']) {
                this.stateChanges.next();
            }
            if (changes['typeaheadDebounceInterval'] && this._keyManager) {
                this._keyManager.withTypeAhead(this._typeaheadDebounceInterval);
            }
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this._destroy.next();
            this._destroy.complete();
            this.stateChanges.complete();
        }
        /**
         * Toggles the overlay panel open or closed.
         * @return {?}
         */
        toggle() {
            this.panelOpen ? this.close() : this.open();
        }
        /**
         * Opens the overlay panel.
         * @return {?}
         */
        open() {
            if (this.disabled || !this.options || !this.options.length || this._panelOpen) {
                return;
            }
            this._triggerRect = this.trigger.nativeElement.getBoundingClientRect();
            // Note: The computed font-size will be a string pixel value (e.g. "16px").
            // `parseInt` ignores the trailing 'px' and converts this to a number.
            this._triggerFontSize = parseInt(getComputedStyle(this.trigger.nativeElement).fontSize || '0');
            this._panelOpen = true;
            this._keyManager.withHorizontalOrientation(null);
            this._calculateOverlayPosition();
            this._highlightCorrectOption();
            this._changeDetectorRef.markForCheck();
            // Set the font size on the panel element once it exists.
            this._ngZone.onStable.asObservable().pipe(take(1)).subscribe((/**
             * @return {?}
             */
            () => {
                if (this._triggerFontSize && this.overlayDir.overlayRef &&
                    this.overlayDir.overlayRef.overlayElement) {
                    this.overlayDir.overlayRef.overlayElement.style.fontSize = `${this._triggerFontSize}px`;
                }
            }));
        }
        /**
         * Closes the overlay panel and focuses the host element.
         * @return {?}
         */
        close() {
            if (this._panelOpen) {
                this._panelOpen = false;
                this._keyManager.withHorizontalOrientation(this._isRtl() ? 'rtl' : 'ltr');
                this._changeDetectorRef.markForCheck();
                this._onTouched();
            }
        }
        /**
         * Sets the select's value. Part of the ControlValueAccessor interface
         * required to integrate with Angular's core forms API.
         *
         * @param {?} value New value to be written to the model.
         * @return {?}
         */
        writeValue(value) {
            if (this.options) {
                this._setSelectionByValue(value);
            }
        }
        /**
         * Saves a callback function to be invoked when the select's value
         * changes from user input. Part of the ControlValueAccessor interface
         * required to integrate with Angular's core forms API.
         *
         * @param {?} fn Callback to be triggered when the value changes.
         * @return {?}
         */
        registerOnChange(fn) {
            this._onChange = fn;
        }
        /**
         * Saves a callback function to be invoked when the select is blurred
         * by the user. Part of the ControlValueAccessor interface required
         * to integrate with Angular's core forms API.
         *
         * @param {?} fn Callback to be triggered when the component has been touched.
         * @return {?}
         */
        registerOnTouched(fn) {
            this._onTouched = fn;
        }
        /**
         * Disables the select. Part of the ControlValueAccessor interface required
         * to integrate with Angular's core forms API.
         *
         * @param {?} isDisabled Sets whether the component is disabled.
         * @return {?}
         */
        setDisabledState(isDisabled) {
            this.disabled = isDisabled;
            this._changeDetectorRef.markForCheck();
            this.stateChanges.next();
        }
        /**
         * Whether or not the overlay panel is open.
         * @return {?}
         */
        get panelOpen() {
            return this._panelOpen;
        }
        /**
         * The currently selected option.
         * @return {?}
         */
        get selected() {
            return this.multiple ? this._selectionModel.selected : this._selectionModel.selected[0];
        }
        /**
         * The value displayed in the trigger.
         * @return {?}
         */
        get triggerValue() {
            if (this.empty) {
                return '';
            }
            if (this._multiple) {
                /** @type {?} */
                const selectedOptions = this._selectionModel.selected.map((/**
                 * @param {?} option
                 * @return {?}
                 */
                option => option.viewValue));
                if (this._isRtl()) {
                    selectedOptions.reverse();
                }
                // TODO(crisbeto): delimiter should be configurable for proper localization.
                return selectedOptions.join(', ');
            }
            return this._selectionModel.selected[0].viewValue;
        }
        /**
         * Whether the element is in RTL mode.
         * @return {?}
         */
        _isRtl() {
            return this._dir ? this._dir.value === 'rtl' : false;
        }
        /**
         * Handles all keydown events on the select.
         * @param {?} event
         * @return {?}
         */
        _handleKeydown(event) {
            if (!this.disabled) {
                this.panelOpen ? this._handleOpenKeydown(event) : this._handleClosedKeydown(event);
            }
        }
        /**
         * Handles keyboard events while the select is closed.
         * @private
         * @param {?} event
         * @return {?}
         */
        _handleClosedKeydown(event) {
            /** @type {?} */
            const keyCode = event.keyCode;
            /** @type {?} */
            const isArrowKey = keyCode === DOWN_ARROW || keyCode === UP_ARROW ||
                keyCode === LEFT_ARROW || keyCode === RIGHT_ARROW;
            /** @type {?} */
            const isOpenKey = keyCode === ENTER || keyCode === SPACE;
            /** @type {?} */
            const manager = this._keyManager;
            // Open the select on ALT + arrow key to match the native <select>
            if (!manager.isTyping() && (isOpenKey && !hasModifierKey(event)) ||
                ((this.multiple || event.altKey) && isArrowKey)) {
                event.preventDefault(); // prevents the page from scrolling down when pressing space
                this.open();
            }
            else if (!this.multiple) {
                /** @type {?} */
                const previouslySelectedOption = this.selected;
                if (keyCode === HOME || keyCode === END) {
                    keyCode === HOME ? manager.setFirstItemActive() : manager.setLastItemActive();
                    event.preventDefault();
                }
                else {
                    manager.onKeydown(event);
                }
                /** @type {?} */
                const selectedOption = this.selected;
                // Since the value has changed, we need to announce it ourselves.
                if (selectedOption && previouslySelectedOption !== selectedOption) {
                    // We set a duration on the live announcement, because we want the live element to be
                    // cleared after a while so that users can't navigate to it using the arrow keys.
                    this._liveAnnouncer.announce(((/** @type {?} */ (selectedOption))).viewValue, 10000);
                }
            }
        }
        /**
         * Handles keyboard events when the selected is open.
         * @private
         * @param {?} event
         * @return {?}
         */
        _handleOpenKeydown(event) {
            /** @type {?} */
            const manager = this._keyManager;
            /** @type {?} */
            const keyCode = event.keyCode;
            /** @type {?} */
            const isArrowKey = keyCode === DOWN_ARROW || keyCode === UP_ARROW;
            /** @type {?} */
            const isTyping = manager.isTyping();
            if (keyCode === HOME || keyCode === END) {
                event.preventDefault();
                keyCode === HOME ? manager.setFirstItemActive() : manager.setLastItemActive();
            }
            else if (isArrowKey && event.altKey) {
                // Close the select on ALT + arrow key to match the native <select>
                event.preventDefault();
                this.close();
                // Don't do anything in this case if the user is typing,
                // because the typing sequence can include the space key.
            }
            else if (!isTyping && (keyCode === ENTER || keyCode === SPACE) && manager.activeItem &&
                !hasModifierKey(event)) {
                event.preventDefault();
                manager.activeItem._selectViaInteraction();
            }
            else if (!isTyping && this._multiple && keyCode === A && event.ctrlKey) {
                event.preventDefault();
                /** @type {?} */
                const hasDeselectedOptions = this.options.some((/**
                 * @param {?} opt
                 * @return {?}
                 */
                opt => !opt.disabled && !opt.selected));
                this.options.forEach((/**
                 * @param {?} option
                 * @return {?}
                 */
                option => {
                    if (!option.disabled) {
                        hasDeselectedOptions ? option.select() : option.deselect();
                    }
                }));
            }
            else {
                /** @type {?} */
                const previouslyFocusedIndex = manager.activeItemIndex;
                manager.onKeydown(event);
                if (this._multiple && isArrowKey && event.shiftKey && manager.activeItem &&
                    manager.activeItemIndex !== previouslyFocusedIndex) {
                    manager.activeItem._selectViaInteraction();
                }
            }
        }
        /**
         * @return {?}
         */
        _onFocus() {
            if (!this.disabled) {
                this._focused = true;
                this.stateChanges.next();
            }
        }
        /**
         * Calls the touched callback only if the panel is closed. Otherwise, the trigger will
         * "blur" to the panel when it opens, causing a false positive.
         * @return {?}
         */
        _onBlur() {
            this._focused = false;
            if (!this.disabled && !this.panelOpen) {
                this._onTouched();
                this._changeDetectorRef.markForCheck();
                this.stateChanges.next();
            }
        }
        /**
         * Callback that is invoked when the overlay panel has been attached.
         * @return {?}
         */
        _onAttached() {
            this.overlayDir.positionChange.pipe(take(1)).subscribe((/**
             * @return {?}
             */
            () => {
                this._changeDetectorRef.detectChanges();
                this._calculateOverlayOffsetX();
                this.panel.nativeElement.scrollTop = this._scrollTop;
            }));
        }
        /**
         * Returns the theme to be used on the panel.
         * @return {?}
         */
        _getPanelTheme() {
            return this._parentFormField ? `mat-${this._parentFormField.color}` : '';
        }
        /**
         * Whether the select has a value.
         * @return {?}
         */
        get empty() {
            return !this._selectionModel || this._selectionModel.isEmpty();
        }
        /**
         * @private
         * @return {?}
         */
        _initializeSelection() {
            // Defer setting the value in order to avoid the "Expression
            // has changed after it was checked" errors from Angular.
            Promise.resolve().then((/**
             * @return {?}
             */
            () => {
                this._setSelectionByValue(this.ngControl ? this.ngControl.value : this._value);
                this.stateChanges.next();
            }));
        }
        /**
         * Sets the selected option based on a value. If no option can be
         * found with the designated value, the select trigger is cleared.
         * @private
         * @param {?} value
         * @return {?}
         */
        _setSelectionByValue(value) {
            if (this.multiple && value) {
                if (!Array.isArray(value)) {
                    throw getMatSelectNonArrayValueError();
                }
                this._selectionModel.clear();
                value.forEach((/**
                 * @param {?} currentValue
                 * @return {?}
                 */
                (currentValue) => this._selectValue(currentValue)));
                this._sortValues();
            }
            else {
                this._selectionModel.clear();
                /** @type {?} */
                const correspondingOption = this._selectValue(value);
                // Shift focus to the active item. Note that we shouldn't do this in multiple
                // mode, because we don't know what option the user interacted with last.
                if (correspondingOption) {
                    this._keyManager.setActiveItem(correspondingOption);
                }
                else if (!this.panelOpen) {
                    // Otherwise reset the highlighted option. Note that we only want to do this while
                    // closed, because doing it while open can shift the user's focus unnecessarily.
                    this._keyManager.setActiveItem(-1);
                }
            }
            this._changeDetectorRef.markForCheck();
        }
        /**
         * Finds and selects and option based on its value.
         * @private
         * @param {?} value
         * @return {?} Option that has the corresponding value.
         */
        _selectValue(value) {
            /** @type {?} */
            const correspondingOption = this.options.find((/**
             * @param {?} option
             * @return {?}
             */
            (option) => {
                try {
                    // Treat null as a special reset value.
                    return option.value != null && this._compareWith(option.value, value);
                }
                catch (error) {
                    if (isDevMode()) {
                        // Notify developers of errors in their comparator.
                        console.warn(error);
                    }
                    return false;
                }
            }));
            if (correspondingOption) {
                this._selectionModel.select(correspondingOption);
            }
            return correspondingOption;
        }
        /**
         * Sets up a key manager to listen to keyboard events on the overlay panel.
         * @private
         * @return {?}
         */
        _initKeyManager() {
            this._keyManager = new ActiveDescendantKeyManager(this.options)
                .withTypeAhead(this._typeaheadDebounceInterval)
                .withVerticalOrientation()
                .withHorizontalOrientation(this._isRtl() ? 'rtl' : 'ltr')
                .withAllowedModifierKeys(['shiftKey']);
            this._keyManager.tabOut.pipe(takeUntil(this._destroy)).subscribe((/**
             * @return {?}
             */
            () => {
                if (this.panelOpen) {
                    // Select the active item when tabbing away. This is consistent with how the native
                    // select behaves. Note that we only want to do this in single selection mode.
                    if (!this.multiple && this._keyManager.activeItem) {
                        this._keyManager.activeItem._selectViaInteraction();
                    }
                    // Restore focus to the trigger before closing. Ensures that the focus
                    // position won't be lost if the user got focus into the overlay.
                    this.focus();
                    this.close();
                }
            }));
            this._keyManager.change.pipe(takeUntil(this._destroy)).subscribe((/**
             * @return {?}
             */
            () => {
                if (this._panelOpen && this.panel) {
                    this._scrollActiveOptionIntoView();
                }
                else if (!this._panelOpen && !this.multiple && this._keyManager.activeItem) {
                    this._keyManager.activeItem._selectViaInteraction();
                }
            }));
        }
        /**
         * Drops current option subscriptions and IDs and resets from scratch.
         * @private
         * @return {?}
         */
        _resetOptions() {
            /** @type {?} */
            const changedOrDestroyed = merge(this.options.changes, this._destroy);
            this.optionSelectionChanges.pipe(takeUntil(changedOrDestroyed)).subscribe((/**
             * @param {?} event
             * @return {?}
             */
            event => {
                this._onSelect(event.source, event.isUserInput);
                if (event.isUserInput && !this.multiple && this._panelOpen) {
                    this.close();
                    this.focus();
                }
            }));
            // Listen to changes in the internal state of the options and react accordingly.
            // Handles cases like the labels of the selected options changing.
            merge(...this.options.map((/**
             * @param {?} option
             * @return {?}
             */
            option => option._stateChanges)))
                .pipe(takeUntil(changedOrDestroyed))
                .subscribe((/**
             * @return {?}
             */
            () => {
                this._changeDetectorRef.markForCheck();
                this.stateChanges.next();
            }));
            this._setOptionIds();
        }
        /**
         * Invoked when an option is clicked.
         * @private
         * @param {?} option
         * @param {?} isUserInput
         * @return {?}
         */
        _onSelect(option, isUserInput) {
            /** @type {?} */
            const wasSelected = this._selectionModel.isSelected(option);
            if (option.value == null && !this._multiple) {
                option.deselect();
                this._selectionModel.clear();
                this._propagateChanges(option.value);
            }
            else {
                if (wasSelected !== option.selected) {
                    option.selected ? this._selectionModel.select(option) :
                        this._selectionModel.deselect(option);
                }
                if (isUserInput) {
                    this._keyManager.setActiveItem(option);
                }
                if (this.multiple) {
                    this._sortValues();
                    if (isUserInput) {
                        // In case the user selected the option with their mouse, we
                        // want to restore focus back to the trigger, in order to
                        // prevent the select keyboard controls from clashing with
                        // the ones from `mat-option`.
                        this.focus();
                    }
                }
            }
            if (wasSelected !== this._selectionModel.isSelected(option)) {
                this._propagateChanges();
            }
            this.stateChanges.next();
        }
        /**
         * Sorts the selected values in the selected based on their order in the panel.
         * @private
         * @return {?}
         */
        _sortValues() {
            if (this.multiple) {
                /** @type {?} */
                const options = this.options.toArray();
                this._selectionModel.sort((/**
                 * @param {?} a
                 * @param {?} b
                 * @return {?}
                 */
                (a, b) => {
                    return this.sortComparator ? this.sortComparator(a, b, options) :
                        options.indexOf(a) - options.indexOf(b);
                }));
                this.stateChanges.next();
            }
        }
        /**
         * Emits change event to set the model value.
         * @private
         * @param {?=} fallbackValue
         * @return {?}
         */
        _propagateChanges(fallbackValue) {
            /** @type {?} */
            let valueToEmit = null;
            if (this.multiple) {
                valueToEmit = ((/** @type {?} */ (this.selected))).map((/**
                 * @param {?} option
                 * @return {?}
                 */
                option => option.value));
            }
            else {
                valueToEmit = this.selected ? ((/** @type {?} */ (this.selected))).value : fallbackValue;
            }
            this._value = valueToEmit;
            this.valueChange.emit(valueToEmit);
            this._onChange(valueToEmit);
            this.selectionChange.emit(new MatSelectChange(this, valueToEmit));
            this._changeDetectorRef.markForCheck();
        }
        /**
         * Records option IDs to pass to the aria-owns property.
         * @private
         * @return {?}
         */
        _setOptionIds() {
            this._optionIds = this.options.map((/**
             * @param {?} option
             * @return {?}
             */
            option => option.id)).join(' ');
        }
        /**
         * Highlights the selected item. If no option is selected, it will highlight
         * the first item instead.
         * @private
         * @return {?}
         */
        _highlightCorrectOption() {
            if (this._keyManager) {
                if (this.empty) {
                    this._keyManager.setFirstItemActive();
                }
                else {
                    this._keyManager.setActiveItem(this._selectionModel.selected[0]);
                }
            }
        }
        /**
         * Scrolls the active option into view.
         * @private
         * @return {?}
         */
        _scrollActiveOptionIntoView() {
            /** @type {?} */
            const activeOptionIndex = this._keyManager.activeItemIndex || 0;
            /** @type {?} */
            const labelCount = _countGroupLabelsBeforeOption(activeOptionIndex, this.options, this.optionGroups);
            this.panel.nativeElement.scrollTop = _getOptionScrollPosition(activeOptionIndex + labelCount, this._getItemHeight(), this.panel.nativeElement.scrollTop, SELECT_PANEL_MAX_HEIGHT);
        }
        /**
         * Focuses the select element.
         * @param {?=} options
         * @return {?}
         */
        focus(options) {
            this._elementRef.nativeElement.focus(options);
        }
        /**
         * Gets the index of the provided option in the option list.
         * @private
         * @param {?} option
         * @return {?}
         */
        _getOptionIndex(option) {
            return this.options.reduce((/**
             * @param {?} result
             * @param {?} current
             * @param {?} index
             * @return {?}
             */
            (result, current, index) => {
                if (result !== undefined) {
                    return result;
                }
                return option === current ? index : undefined;
            }), undefined);
        }
        /**
         * Calculates the scroll position and x- and y-offsets of the overlay panel.
         * @private
         * @return {?}
         */
        _calculateOverlayPosition() {
            /** @type {?} */
            const itemHeight = this._getItemHeight();
            /** @type {?} */
            const items = this._getItemCount();
            /** @type {?} */
            const panelHeight = Math.min(items * itemHeight, SELECT_PANEL_MAX_HEIGHT);
            /** @type {?} */
            const scrollContainerHeight = items * itemHeight;
            // The farthest the panel can be scrolled before it hits the bottom
            /** @type {?} */
            const maxScroll = scrollContainerHeight - panelHeight;
            // If no value is selected we open the popup to the first item.
            /** @type {?} */
            let selectedOptionOffset = this.empty ? 0 : (/** @type {?} */ (this._getOptionIndex(this._selectionModel.selected[0])));
            selectedOptionOffset += _countGroupLabelsBeforeOption(selectedOptionOffset, this.options, this.optionGroups);
            // We must maintain a scroll buffer so the selected option will be scrolled to the
            // center of the overlay panel rather than the top.
            /** @type {?} */
            const scrollBuffer = panelHeight / 2;
            this._scrollTop = this._calculateOverlayScroll(selectedOptionOffset, scrollBuffer, maxScroll);
            this._offsetY = this._calculateOverlayOffsetY(selectedOptionOffset, scrollBuffer, maxScroll);
            this._checkOverlayWithinViewport(maxScroll);
        }
        /**
         * Calculates the scroll position of the select's overlay panel.
         *
         * Attempts to center the selected option in the panel. If the option is
         * too high or too low in the panel to be scrolled to the center, it clamps the
         * scroll position to the min or max scroll positions respectively.
         * @param {?} selectedIndex
         * @param {?} scrollBuffer
         * @param {?} maxScroll
         * @return {?}
         */
        _calculateOverlayScroll(selectedIndex, scrollBuffer, maxScroll) {
            /** @type {?} */
            const itemHeight = this._getItemHeight();
            /** @type {?} */
            const optionOffsetFromScrollTop = itemHeight * selectedIndex;
            /** @type {?} */
            const halfOptionHeight = itemHeight / 2;
            // Starts at the optionOffsetFromScrollTop, which scrolls the option to the top of the
            // scroll container, then subtracts the scroll buffer to scroll the option down to
            // the center of the overlay panel. Half the option height must be re-added to the
            // scrollTop so the option is centered based on its middle, not its top edge.
            /** @type {?} */
            const optimalScrollPosition = optionOffsetFromScrollTop - scrollBuffer + halfOptionHeight;
            return Math.min(Math.max(0, optimalScrollPosition), maxScroll);
        }
        /**
         * Returns the aria-label of the select component.
         * @return {?}
         */
        _getAriaLabel() {
            // If an ariaLabelledby value has been set by the consumer, the select should not overwrite the
            // `aria-labelledby` value by setting the ariaLabel to the placeholder.
            return this.ariaLabelledby ? null : this.ariaLabel || this.placeholder;
        }
        /**
         * Returns the aria-labelledby of the select component.
         * @return {?}
         */
        _getAriaLabelledby() {
            if (this.ariaLabelledby) {
                return this.ariaLabelledby;
            }
            // Note: we use `_getAriaLabel` here, because we want to check whether there's a
            // computed label. `this.ariaLabel` is only the user-specified label.
            if (!this._parentFormField || !this._parentFormField._hasFloatingLabel() ||
                this._getAriaLabel()) {
                return null;
            }
            return this._parentFormField._labelId || null;
        }
        /**
         * Determines the `aria-activedescendant` to be set on the host.
         * @return {?}
         */
        _getAriaActiveDescendant() {
            if (this.panelOpen && this._keyManager && this._keyManager.activeItem) {
                return this._keyManager.activeItem.id;
            }
            return null;
        }
        /**
         * Sets the x-offset of the overlay panel in relation to the trigger's top start corner.
         * This must be adjusted to align the selected option text over the trigger text when
         * the panel opens. Will change based on LTR or RTL text direction. Note that the offset
         * can't be calculated until the panel has been attached, because we need to know the
         * content width in order to constrain the panel within the viewport.
         * @private
         * @return {?}
         */
        _calculateOverlayOffsetX() {
            /** @type {?} */
            const overlayRect = this.overlayDir.overlayRef.overlayElement.getBoundingClientRect();
            /** @type {?} */
            const viewportSize = this._viewportRuler.getViewportSize();
            /** @type {?} */
            const isRtl = this._isRtl();
            /** @type {?} */
            const paddingWidth = this.multiple ? SELECT_MULTIPLE_PANEL_PADDING_X + SELECT_PANEL_PADDING_X :
                SELECT_PANEL_PADDING_X * 2;
            /** @type {?} */
            let offsetX;
            // Adjust the offset, depending on the option padding.
            if (this.multiple) {
                offsetX = SELECT_MULTIPLE_PANEL_PADDING_X;
            }
            else {
                /** @type {?} */
                let selected = this._selectionModel.selected[0] || this.options.first;
                offsetX = selected && selected.group ? SELECT_PANEL_INDENT_PADDING_X : SELECT_PANEL_PADDING_X;
            }
            // Invert the offset in LTR.
            if (!isRtl) {
                offsetX *= -1;
            }
            // Determine how much the select overflows on each side.
            /** @type {?} */
            const leftOverflow = 0 - (overlayRect.left + offsetX - (isRtl ? paddingWidth : 0));
            /** @type {?} */
            const rightOverflow = overlayRect.right + offsetX - viewportSize.width
                + (isRtl ? 0 : paddingWidth);
            // If the element overflows on either side, reduce the offset to allow it to fit.
            if (leftOverflow > 0) {
                offsetX += leftOverflow + SELECT_PANEL_VIEWPORT_PADDING;
            }
            else if (rightOverflow > 0) {
                offsetX -= rightOverflow + SELECT_PANEL_VIEWPORT_PADDING;
            }
            // Set the offset directly in order to avoid having to go through change detection and
            // potentially triggering "changed after it was checked" errors. Round the value to avoid
            // blurry content in some browsers.
            this.overlayDir.offsetX = Math.round(offsetX);
            this.overlayDir.overlayRef.updatePosition();
        }
        /**
         * Calculates the y-offset of the select's overlay panel in relation to the
         * top start corner of the trigger. It has to be adjusted in order for the
         * selected option to be aligned over the trigger when the panel opens.
         * @private
         * @param {?} selectedIndex
         * @param {?} scrollBuffer
         * @param {?} maxScroll
         * @return {?}
         */
        _calculateOverlayOffsetY(selectedIndex, scrollBuffer, maxScroll) {
            /** @type {?} */
            const itemHeight = this._getItemHeight();
            /** @type {?} */
            const optionHeightAdjustment = (itemHeight - this._triggerRect.height) / 2;
            /** @type {?} */
            const maxOptionsDisplayed = Math.floor(SELECT_PANEL_MAX_HEIGHT / itemHeight);
            /** @type {?} */
            let optionOffsetFromPanelTop;
            // Disable offset if requested by user by returning 0 as value to offset
            if (this._disableOptionCentering) {
                return 0;
            }
            if (this._scrollTop === 0) {
                optionOffsetFromPanelTop = selectedIndex * itemHeight;
            }
            else if (this._scrollTop === maxScroll) {
                /** @type {?} */
                const firstDisplayedIndex = this._getItemCount() - maxOptionsDisplayed;
                /** @type {?} */
                const selectedDisplayIndex = selectedIndex - firstDisplayedIndex;
                // The first item is partially out of the viewport. Therefore we need to calculate what
                // portion of it is shown in the viewport and account for it in our offset.
                /** @type {?} */
                let partialItemHeight = itemHeight - (this._getItemCount() * itemHeight - SELECT_PANEL_MAX_HEIGHT) % itemHeight;
                // Because the panel height is longer than the height of the options alone,
                // there is always extra padding at the top or bottom of the panel. When
                // scrolled to the very bottom, this padding is at the top of the panel and
                // must be added to the offset.
                optionOffsetFromPanelTop = selectedDisplayIndex * itemHeight + partialItemHeight;
            }
            else {
                // If the option was scrolled to the middle of the panel using a scroll buffer,
                // its offset will be the scroll buffer minus the half height that was added to
                // center it.
                optionOffsetFromPanelTop = scrollBuffer - itemHeight / 2;
            }
            // The final offset is the option's offset from the top, adjusted for the height difference,
            // multiplied by -1 to ensure that the overlay moves in the correct direction up the page.
            // The value is rounded to prevent some browsers from blurring the content.
            return Math.round(optionOffsetFromPanelTop * -1 - optionHeightAdjustment);
        }
        /**
         * Checks that the attempted overlay position will fit within the viewport.
         * If it will not fit, tries to adjust the scroll position and the associated
         * y-offset so the panel can open fully on-screen. If it still won't fit,
         * sets the offset back to 0 to allow the fallback position to take over.
         * @private
         * @param {?} maxScroll
         * @return {?}
         */
        _checkOverlayWithinViewport(maxScroll) {
            /** @type {?} */
            const itemHeight = this._getItemHeight();
            /** @type {?} */
            const viewportSize = this._viewportRuler.getViewportSize();
            /** @type {?} */
            const topSpaceAvailable = this._triggerRect.top - SELECT_PANEL_VIEWPORT_PADDING;
            /** @type {?} */
            const bottomSpaceAvailable = viewportSize.height - this._triggerRect.bottom - SELECT_PANEL_VIEWPORT_PADDING;
            /** @type {?} */
            const panelHeightTop = Math.abs(this._offsetY);
            /** @type {?} */
            const totalPanelHeight = Math.min(this._getItemCount() * itemHeight, SELECT_PANEL_MAX_HEIGHT);
            /** @type {?} */
            const panelHeightBottom = totalPanelHeight - panelHeightTop - this._triggerRect.height;
            if (panelHeightBottom > bottomSpaceAvailable) {
                this._adjustPanelUp(panelHeightBottom, bottomSpaceAvailable);
            }
            else if (panelHeightTop > topSpaceAvailable) {
                this._adjustPanelDown(panelHeightTop, topSpaceAvailable, maxScroll);
            }
            else {
                this._transformOrigin = this._getOriginBasedOnOption();
            }
        }
        /**
         * Adjusts the overlay panel up to fit in the viewport.
         * @private
         * @param {?} panelHeightBottom
         * @param {?} bottomSpaceAvailable
         * @return {?}
         */
        _adjustPanelUp(panelHeightBottom, bottomSpaceAvailable) {
            // Browsers ignore fractional scroll offsets, so we need to round.
            /** @type {?} */
            const distanceBelowViewport = Math.round(panelHeightBottom - bottomSpaceAvailable);
            // Scrolls the panel up by the distance it was extending past the boundary, then
            // adjusts the offset by that amount to move the panel up into the viewport.
            this._scrollTop -= distanceBelowViewport;
            this._offsetY -= distanceBelowViewport;
            this._transformOrigin = this._getOriginBasedOnOption();
            // If the panel is scrolled to the very top, it won't be able to fit the panel
            // by scrolling, so set the offset to 0 to allow the fallback position to take
            // effect.
            if (this._scrollTop <= 0) {
                this._scrollTop = 0;
                this._offsetY = 0;
                this._transformOrigin = `50% bottom 0px`;
            }
        }
        /**
         * Adjusts the overlay panel down to fit in the viewport.
         * @private
         * @param {?} panelHeightTop
         * @param {?} topSpaceAvailable
         * @param {?} maxScroll
         * @return {?}
         */
        _adjustPanelDown(panelHeightTop, topSpaceAvailable, maxScroll) {
            // Browsers ignore fractional scroll offsets, so we need to round.
            /** @type {?} */
            const distanceAboveViewport = Math.round(panelHeightTop - topSpaceAvailable);
            // Scrolls the panel down by the distance it was extending past the boundary, then
            // adjusts the offset by that amount to move the panel down into the viewport.
            this._scrollTop += distanceAboveViewport;
            this._offsetY += distanceAboveViewport;
            this._transformOrigin = this._getOriginBasedOnOption();
            // If the panel is scrolled to the very bottom, it won't be able to fit the
            // panel by scrolling, so set the offset to 0 to allow the fallback position
            // to take effect.
            if (this._scrollTop >= maxScroll) {
                this._scrollTop = maxScroll;
                this._offsetY = 0;
                this._transformOrigin = `50% top 0px`;
                return;
            }
        }
        /**
         * Sets the transform origin point based on the selected option.
         * @private
         * @return {?}
         */
        _getOriginBasedOnOption() {
            /** @type {?} */
            const itemHeight = this._getItemHeight();
            /** @type {?} */
            const optionHeightAdjustment = (itemHeight - this._triggerRect.height) / 2;
            /** @type {?} */
            const originY = Math.abs(this._offsetY) - optionHeightAdjustment + itemHeight / 2;
            return `50% ${originY}px 0px`;
        }
        /**
         * Calculates the amount of items in the select. This includes options and group labels.
         * @private
         * @return {?}
         */
        _getItemCount() {
            return this.options.length + this.optionGroups.length;
        }
        /**
         * Calculates the height of the select's options.
         * @private
         * @return {?}
         */
        _getItemHeight() {
            return this._triggerFontSize * SELECT_ITEM_HEIGHT_EM;
        }
        /**
         * Implemented as part of MatFormFieldControl.
         * \@docs-private
         * @param {?} ids
         * @return {?}
         */
        setDescribedByIds(ids) {
            this._ariaDescribedby = ids.join(' ');
        }
        /**
         * Implemented as part of MatFormFieldControl.
         * \@docs-private
         * @return {?}
         */
        onContainerClick() {
            this.focus();
            this.open();
        }
        /**
         * Implemented as part of MatFormFieldControl.
         * \@docs-private
         * @return {?}
         */
        get shouldLabelFloat() {
            return this._panelOpen || !this.empty;
        }
    }
    MatSelect.ɵfac = function MatSelect_Factory(t) { return new (t || MatSelect)(ɵɵdirectiveInject(ViewportRuler), ɵɵdirectiveInject(ChangeDetectorRef), ɵɵdirectiveInject(NgZone), ɵɵdirectiveInject(ErrorStateMatcher), ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(Directionality, 8), ɵɵdirectiveInject(NgForm, 8), ɵɵdirectiveInject(FormGroupDirective, 8), ɵɵdirectiveInject(MAT_FORM_FIELD, 8), ɵɵdirectiveInject(NgControl, 10), ɵɵinjectAttribute('tabindex'), ɵɵdirectiveInject(MAT_SELECT_SCROLL_STRATEGY), ɵɵdirectiveInject(LiveAnnouncer), ɵɵdirectiveInject(MAT_SELECT_CONFIG, 8)); };
    MatSelect.ɵcmp = ɵɵdefineComponent({ type: MatSelect, selectors: [["mat-select"]], contentQueries: function MatSelect_ContentQueries(rf, ctx, dirIndex) { if (rf & 1) {
            ɵɵcontentQuery(dirIndex, MatSelectTrigger, true);
            ɵɵcontentQuery(dirIndex, MatOption, true);
            ɵɵcontentQuery(dirIndex, MatOptgroup, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.customTrigger = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.options = _t);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.optionGroups = _t);
        } }, viewQuery: function MatSelect_Query(rf, ctx) { if (rf & 1) {
            ɵɵviewQuery(_c0$5, true);
            ɵɵviewQuery(_c1$5, true);
            ɵɵviewQuery(CdkConnectedOverlay, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.trigger = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.panel = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.overlayDir = _t.first);
        } }, hostAttrs: ["role", "listbox", 1, "mat-select"], hostVars: 19, hostBindings: function MatSelect_HostBindings(rf, ctx) { if (rf & 1) {
            ɵɵlistener("keydown", function MatSelect_keydown_HostBindingHandler($event) { return ctx._handleKeydown($event); })("focus", function MatSelect_focus_HostBindingHandler() { return ctx._onFocus(); })("blur", function MatSelect_blur_HostBindingHandler() { return ctx._onBlur(); });
        } if (rf & 2) {
            ɵɵattribute("id", ctx.id)("tabindex", ctx.tabIndex)("aria-label", ctx._getAriaLabel())("aria-labelledby", ctx._getAriaLabelledby())("aria-required", ctx.required.toString())("aria-disabled", ctx.disabled.toString())("aria-invalid", ctx.errorState)("aria-owns", ctx.panelOpen ? ctx._optionIds : null)("aria-multiselectable", ctx.multiple)("aria-describedby", ctx._ariaDescribedby || null)("aria-activedescendant", ctx._getAriaActiveDescendant());
            ɵɵclassProp("mat-select-disabled", ctx.disabled)("mat-select-invalid", ctx.errorState)("mat-select-required", ctx.required)("mat-select-empty", ctx.empty);
        } }, inputs: { disabled: "disabled", disableRipple: "disableRipple", tabIndex: "tabIndex", ariaLabel: ["aria-label", "ariaLabel"], id: "id", disableOptionCentering: "disableOptionCentering", typeaheadDebounceInterval: "typeaheadDebounceInterval", placeholder: "placeholder", required: "required", multiple: "multiple", compareWith: "compareWith", value: "value", panelClass: "panelClass", ariaLabelledby: ["aria-labelledby", "ariaLabelledby"], errorStateMatcher: "errorStateMatcher", sortComparator: "sortComparator" }, outputs: { openedChange: "openedChange", _openedStream: "opened", _closedStream: "closed", selectionChange: "selectionChange", valueChange: "valueChange" }, exportAs: ["matSelect"], features: [ɵɵProvidersFeature([
                { provide: MatFormFieldControl, useExisting: MatSelect },
                { provide: MAT_OPTION_PARENT_COMPONENT, useExisting: MatSelect }
            ]), ɵɵInheritDefinitionFeature, ɵɵNgOnChangesFeature], ngContentSelectors: _c3$2, decls: 9, vars: 9, consts: [["cdk-overlay-origin", "", "aria-hidden", "true", 1, "mat-select-trigger", 3, "click"], ["origin", "cdkOverlayOrigin", "trigger", ""], [1, "mat-select-value", 3, "ngSwitch"], ["class", "mat-select-placeholder", 4, "ngSwitchCase"], ["class", "mat-select-value-text", 3, "ngSwitch", 4, "ngSwitchCase"], [1, "mat-select-arrow-wrapper"], [1, "mat-select-arrow"], ["cdk-connected-overlay", "", "cdkConnectedOverlayLockPosition", "", "cdkConnectedOverlayHasBackdrop", "", "cdkConnectedOverlayBackdropClass", "cdk-overlay-transparent-backdrop", 3, "cdkConnectedOverlayScrollStrategy", "cdkConnectedOverlayOrigin", "cdkConnectedOverlayOpen", "cdkConnectedOverlayPositions", "cdkConnectedOverlayMinWidth", "cdkConnectedOverlayOffsetY", "backdropClick", "attach", "detach"], [1, "mat-select-placeholder"], [1, "mat-select-value-text", 3, "ngSwitch"], [4, "ngSwitchDefault"], [4, "ngSwitchCase"], [1, "mat-select-panel-wrap"], [3, "ngClass", "keydown"], ["panel", ""]], template: function MatSelect_Template(rf, ctx) { if (rf & 1) {
            ɵɵprojectionDef(_c2$3);
            ɵɵelementStart(0, "div", 0, 1);
            ɵɵlistener("click", function MatSelect_Template_div_click_0_listener() { return ctx.toggle(); });
            ɵɵelementStart(3, "div", 2);
            ɵɵtemplate(4, MatSelect_span_4_Template, 2, 1, "span", 3);
            ɵɵtemplate(5, MatSelect_span_5_Template, 3, 2, "span", 4);
            ɵɵelementEnd();
            ɵɵelementStart(6, "div", 5);
            ɵɵelement(7, "div", 6);
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵtemplate(8, MatSelect_ng_template_8_Template, 4, 11, "ng-template", 7);
            ɵɵlistener("backdropClick", function MatSelect_Template_ng_template_backdropClick_8_listener() { return ctx.close(); })("attach", function MatSelect_Template_ng_template_attach_8_listener() { return ctx._onAttached(); })("detach", function MatSelect_Template_ng_template_detach_8_listener() { return ctx.close(); });
        } if (rf & 2) {
            const _r0 = ɵɵreference(1);
            ɵɵadvance(3);
            ɵɵproperty("ngSwitch", ctx.empty);
            ɵɵadvance(1);
            ɵɵproperty("ngSwitchCase", true);
            ɵɵadvance(1);
            ɵɵproperty("ngSwitchCase", false);
            ɵɵadvance(3);
            ɵɵproperty("cdkConnectedOverlayScrollStrategy", ctx._scrollStrategy)("cdkConnectedOverlayOrigin", _r0)("cdkConnectedOverlayOpen", ctx.panelOpen)("cdkConnectedOverlayPositions", ctx._positions)("cdkConnectedOverlayMinWidth", ctx._triggerRect == null ? null : ctx._triggerRect.width)("cdkConnectedOverlayOffsetY", ctx._offsetY);
        } }, directives: [CdkOverlayOrigin, NgSwitch, NgSwitchCase, CdkConnectedOverlay, NgSwitchDefault, NgClass], styles: [".mat-select{display:inline-block;width:100%;outline:none}.mat-select-trigger{display:inline-table;cursor:pointer;position:relative;box-sizing:border-box}.mat-select-disabled .mat-select-trigger{-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;cursor:default}.mat-select-value{display:table-cell;max-width:0;width:100%;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.mat-select-value-text{white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.mat-select-arrow-wrapper{display:table-cell;vertical-align:middle}.mat-form-field-appearance-fill .mat-select-arrow-wrapper{transform:translateY(-50%)}.mat-form-field-appearance-outline .mat-select-arrow-wrapper{transform:translateY(-25%)}.mat-form-field-appearance-standard.mat-form-field-has-label .mat-select:not(.mat-select-empty) .mat-select-arrow-wrapper{transform:translateY(-50%)}.mat-form-field-appearance-standard .mat-select.mat-select-empty .mat-select-arrow-wrapper{transition:transform 400ms cubic-bezier(0.25, 0.8, 0.25, 1)}._mat-animation-noopable.mat-form-field-appearance-standard .mat-select.mat-select-empty .mat-select-arrow-wrapper{transition:none}.mat-select-arrow{width:0;height:0;border-left:5px solid transparent;border-right:5px solid transparent;border-top:5px solid;margin:0 4px}.mat-select-panel-wrap{flex-basis:100%}.mat-select-panel{min-width:112px;max-width:280px;overflow:auto;-webkit-overflow-scrolling:touch;padding-top:0;padding-bottom:0;max-height:256px;min-width:100%;border-radius:4px}.cdk-high-contrast-active .mat-select-panel{outline:solid 1px}.mat-select-panel .mat-optgroup-label,.mat-select-panel .mat-option{font-size:inherit;line-height:3em;height:3em}.mat-form-field-type-mat-select:not(.mat-form-field-disabled) .mat-form-field-flex{cursor:pointer}.mat-form-field-type-mat-select .mat-form-field-label{width:calc(100% - 18px)}.mat-select-placeholder{transition:color 400ms 133.3333333333ms cubic-bezier(0.25, 0.8, 0.25, 1)}._mat-animation-noopable .mat-select-placeholder{transition:none}.mat-form-field-hide-placeholder .mat-select-placeholder{color:transparent;-webkit-text-fill-color:transparent;transition:none;display:block}\n"], encapsulation: 2, data: { animation: [
                matSelectAnimations.transformPanelWrap,
                matSelectAnimations.transformPanel
            ] }, changeDetection: 0 });
    /** @nocollapse */
    MatSelect.ctorParameters = () => [
        { type: ViewportRuler },
        { type: ChangeDetectorRef },
        { type: NgZone },
        { type: ErrorStateMatcher },
        { type: ElementRef },
        { type: Directionality, decorators: [{ type: Optional }] },
        { type: NgForm, decorators: [{ type: Optional }] },
        { type: FormGroupDirective, decorators: [{ type: Optional }] },
        { type: MatFormField, decorators: [{ type: Optional }, { type: Inject, args: [MAT_FORM_FIELD,] }] },
        { type: NgControl, decorators: [{ type: Self }, { type: Optional }] },
        { type: String, decorators: [{ type: Attribute, args: ['tabindex',] }] },
        { type: undefined, decorators: [{ type: Inject, args: [MAT_SELECT_SCROLL_STRATEGY,] }] },
        { type: LiveAnnouncer },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [MAT_SELECT_CONFIG,] }] }
    ];
    MatSelect.propDecorators = {
        trigger: [{ type: ViewChild, args: ['trigger',] }],
        panel: [{ type: ViewChild, args: ['panel',] }],
        overlayDir: [{ type: ViewChild, args: [CdkConnectedOverlay,] }],
        options: [{ type: ContentChildren, args: [MatOption, { descendants: true },] }],
        optionGroups: [{ type: ContentChildren, args: [MatOptgroup, { descendants: true },] }],
        panelClass: [{ type: Input }],
        customTrigger: [{ type: ContentChild, args: [MatSelectTrigger,] }],
        placeholder: [{ type: Input }],
        required: [{ type: Input }],
        multiple: [{ type: Input }],
        disableOptionCentering: [{ type: Input }],
        compareWith: [{ type: Input }],
        value: [{ type: Input }],
        ariaLabel: [{ type: Input, args: ['aria-label',] }],
        ariaLabelledby: [{ type: Input, args: ['aria-labelledby',] }],
        errorStateMatcher: [{ type: Input }],
        typeaheadDebounceInterval: [{ type: Input }],
        sortComparator: [{ type: Input }],
        id: [{ type: Input }],
        openedChange: [{ type: Output }],
        _openedStream: [{ type: Output, args: ['opened',] }],
        _closedStream: [{ type: Output, args: ['closed',] }],
        selectionChange: [{ type: Output }],
        valueChange: [{ type: Output }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/select/select-module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class MatSelectModule {
    }
    MatSelectModule.ɵmod = ɵɵdefineNgModule({ type: MatSelectModule });
    MatSelectModule.ɵinj = ɵɵdefineInjector({ factory: function MatSelectModule_Factory(t) { return new (t || MatSelectModule)(); }, providers: [MAT_SELECT_SCROLL_STRATEGY_PROVIDER], imports: [[
                CommonModule,
                OverlayModule,
                MatOptionModule,
                MatCommonModule,
            ],
            CdkScrollableModule,
            MatFormFieldModule,
            MatOptionModule,
            MatCommonModule] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(MatSelectModule, { declarations: function () { return [MatSelect,
            MatSelectTrigger]; }, imports: function () { return [CommonModule,
            OverlayModule,
            MatOptionModule,
            MatCommonModule]; }, exports: function () { return [CdkScrollableModule,
            MatFormFieldModule,
            MatSelect,
            MatSelectTrigger,
            MatOptionModule,
            MatCommonModule]; } }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/header/plugin_selector_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    function PluginSelectorComponent_mat_tab_1_ng_template_1_Template(rf, ctx) { if (rf & 1) {
        const _r5 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "span", 5);
        ɵɵlistener("click", function PluginSelectorComponent_mat_tab_1_ng_template_1_Template_span_click_0_listener($event) { ɵɵrestoreView(_r5); const plugin_r2 = ɵɵnextContext().$implicit; const ctx_r4 = ɵɵnextContext(); return ctx_r4.onActivePluginSelection($event, plugin_r2.id); });
        ɵɵtext(1);
        ɵɵelementEnd();
    } if (rf & 2) {
        const plugin_r2 = ɵɵnextContext().$implicit;
        ɵɵattribute("data-plugin-id", plugin_r2.id);
        ɵɵadvance(1);
        ɵɵtextInterpolate1(" ", plugin_r2.tab_name, " ");
    } }
    function PluginSelectorComponent_mat_tab_1_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "mat-tab", 3);
        ɵɵtemplate(1, PluginSelectorComponent_mat_tab_1_ng_template_1_Template, 2, 2, "ng-template", 4);
        ɵɵelementEnd();
    } if (rf & 2) {
        const plugin_r2 = ctx.$implicit;
        ɵɵproperty("disabled", !plugin_r2.enabled);
    } }
    function PluginSelectorComponent_mat_form_field_2_mat_option_4_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "mat-option", 9);
        ɵɵtext(1);
        ɵɵelementEnd();
    } if (rf & 2) {
        const plugin_r9 = ctx.$implicit;
        ɵɵproperty("value", plugin_r9.id);
        ɵɵattribute("data-plugin-id", plugin_r9.id);
        ɵɵadvance(1);
        ɵɵtextInterpolate1(" ", plugin_r9.tab_name, " ");
    } }
    function PluginSelectorComponent_mat_form_field_2_Template(rf, ctx) { if (rf & 1) {
        const _r11 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "mat-form-field", 6);
        ɵɵelementStart(1, "mat-label");
        ɵɵtext(2, "Inactive");
        ɵɵelementEnd();
        ɵɵelementStart(3, "mat-select", 7);
        ɵɵlistener("selectionChange", function PluginSelectorComponent_mat_form_field_2_Template_mat_select_selectionChange_3_listener($event) { ɵɵrestoreView(_r11); const ctx_r10 = ɵɵnextContext(); return ctx_r10.onDisabledPluginSelectionChanged($event); });
        ɵɵtemplate(4, PluginSelectorComponent_mat_form_field_2_mat_option_4_Template, 2, 3, "mat-option", 8);
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r1 = ɵɵnextContext();
        ɵɵadvance(3);
        ɵɵproperty("value", ctx_r1.selectedPlugin);
        ɵɵadvance(1);
        ɵɵproperty("ngForOf", ctx_r1.disabledPlugins);
    } }
    class PluginSelectorComponent {
        constructor() {
            this.onPluginSelectionChanged = new EventEmitter();
        }
        /**
         * @return {?}
         */
        getActivePluginIndex() {
            return this.activePlugins.findIndex((/**
             * @param {?} __0
             * @return {?}
             */
            ({ id }) => id === this.selectedPlugin));
        }
        /**
         * @param {?} event
         * @param {?} pluginId
         * @return {?}
         */
        onActivePluginSelection(event, pluginId) {
            event.stopPropagation();
            this.onPluginSelectionChanged.emit(pluginId);
        }
        /**
         * @param {?} selectChangeEvent
         * @return {?}
         */
        onDisabledPluginSelectionChanged(selectChangeEvent) {
            this.onPluginSelectionChanged.emit(selectChangeEvent.value);
        }
    }
    PluginSelectorComponent.decorators = [
        { type: Component, args: [{
                    selector: 'plugin-selector-component',
                    templateUrl: './plugin_selector_component.ng.html',
                    styleUrls: ['./plugin_selector_component.css'],
                },] },
    ];
    PluginSelectorComponent.propDecorators = {
        activePlugins: [{ type: Input }],
        disabledPlugins: [{ type: Input }],
        selectedPlugin: [{ type: Input }],
        onPluginSelectionChanged: [{ type: Output }]
    };
    /** @nocollapse */ PluginSelectorComponent.ɵfac = function PluginSelectorComponent_Factory(t) { return new (t || PluginSelectorComponent)(); };
    /** @nocollapse */ PluginSelectorComponent.ɵcmp = ɵɵdefineComponent({ type: PluginSelectorComponent, selectors: [["plugin-selector-component"]], inputs: { activePlugins: "activePlugins", disabledPlugins: "disabledPlugins", selectedPlugin: "selectedPlugin" }, outputs: { onPluginSelectionChanged: "onPluginSelectionChanged" }, decls: 3, vars: 3, consts: [["animationDuration", "100ms", 1, "active-plugin-list", 3, "selectedIndex"], [3, "disabled", 4, "ngFor", "ngForOf"], ["floatLabel", "never", 4, "ngIf"], [3, "disabled"], ["mat-tab-label", ""], [1, "plugin-name", 3, "click"], ["floatLabel", "never"], [3, "value", "selectionChange"], [3, "value", 4, "ngFor", "ngForOf"], [3, "value"]], template: function PluginSelectorComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "mat-tab-group", 0);
            ɵɵtemplate(1, PluginSelectorComponent_mat_tab_1_Template, 2, 1, "mat-tab", 1);
            ɵɵelementEnd();
            ɵɵtemplate(2, PluginSelectorComponent_mat_form_field_2_Template, 5, 2, "mat-form-field", 2);
        } if (rf & 2) {
            ɵɵproperty("selectedIndex", ctx.getActivePluginIndex());
            ɵɵadvance(1);
            ɵɵproperty("ngForOf", ctx.activePlugins);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx.disabledPlugins.length > 0);
        } }, directives: [MatTabGroup, NgForOf, NgIf, MatTab, MatTabLabel, MatFormField, MatLabel, MatSelect, MatOption], styles: ["[_nghost-%COMP%]{align-items:center;display:flex;flex:1 1 auto;font-size:14px;height:100%;overflow:hidden}mat-form-field[_ngcontent-%COMP%]{flex:0 0;margin-top:5px;width:130px}mat-label[_ngcontent-%COMP%], mat-select[_ngcontent-%COMP%], mat-option[_ngcontent-%COMP%]{font-weight:500;text-transform:uppercase}.active-plugin-list[_ngcontent-%COMP%]{align-self:stretch;flex:1 1 auto;overflow:hidden}.plugin-name[_ngcontent-%COMP%]{align-items:center;display:inline-flex;height:100%;justify-content:center;padding:0 12px;width:100%}[_nghost-%COMP%]     .active-plugin-list.mat-primary .mat-ink-bar{background-color:currentColor}[_nghost-%COMP%]     .active-plugin-list .mat-tab-label, [_nghost-%COMP%]     .active-plugin-list .mat-tab-link{color:inherit;opacity:.7}[_nghost-%COMP%]     .active-plugin-list .mat-tab-label.mat-tab-label-active, [_nghost-%COMP%]     .active-plugin-list .mat-tab-link.mat-tab-label-active{opacity:1}[_nghost-%COMP%]     .active-plugin-list .mat-tab-header-pagination-chevron{border-color:currentColor}[_nghost-%COMP%]     .active-plugin-list .mat-tab-header-pagination-disabled{visibility:hidden}[_nghost-%COMP%]     .active-plugin-list .mat-tab-disabled{display:none}[_nghost-%COMP%]     .active-plugin-list .mat-tab-header, [_nghost-%COMP%]     .active-plugin-list .mat-tab-labels, [_nghost-%COMP%]     .active-plugin-list .mat-tab-label{height:100%}[_nghost-%COMP%]     .active-plugin-list .mat-tab-label{min-width:48px;padding:0;text-transform:uppercase}[_nghost-%COMP%]     .active-plugin-list .mat-tab-label-content{height:100%}[_nghost-%COMP%]     .active-plugin-list mat-tab-header:not(.mat-tab-header-pagination-controls-enabled){padding:0 36px}"] });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/header/plugin_selector_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const getUiPlugins = createSelector(getPlugins, (/**
     * @param {?} listing
     * @return {?}
     */
    (listing) => Object.keys(listing).map((/**
     * @param {?} key
     * @return {?}
     */
    (key) => Object.assign({}, { id: key }, listing[key])))));
    /** @type {?} */
    const getDisabledPlugins = createSelector(getUiPlugins, (/**
     * @param {?} plugins
     * @return {?}
     */
    (plugins) => plugins.filter((/**
     * @param {?} plugin
     * @return {?}
     */
    (plugin) => !plugin.enabled))));
    class PluginSelectorContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.activePlugin$ = this.store.pipe(select(getActivePlugin));
            this.plugins$ = this.store.pipe(select(getUiPlugins));
            this.disabledPlugins$ = this.store.pipe(select(getDisabledPlugins));
        }
        /**
         * @param {?} pluginId
         * @return {?}
         */
        onPluginSelectionChange(pluginId) {
            this.store.dispatch(changePlugin({ plugin: pluginId }));
        }
    }
    PluginSelectorContainer.decorators = [
        { type: Component, args: [{
                    selector: 'plugin-selector',
                    template: `
    <plugin-selector-component
      [activePlugins]="plugins$ | async"
      [disabledPlugins]="disabledPlugins$ | async"
      [selectedPlugin]="activePlugin$ | async"
      (onPluginSelectionChanged)="onPluginSelectionChange($event)"
    ></plugin-selector-component>
  `,
                },] },
    ];
    /** @nocollapse */
    PluginSelectorContainer.ctorParameters = () => [
        { type: Store }
    ];
    /** @nocollapse */ PluginSelectorContainer.ɵfac = function PluginSelectorContainer_Factory(t) { return new (t || PluginSelectorContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ PluginSelectorContainer.ɵcmp = ɵɵdefineComponent({ type: PluginSelectorContainer, selectors: [["plugin-selector"]], decls: 4, vars: 9, consts: [[3, "activePlugins", "disabledPlugins", "selectedPlugin", "onPluginSelectionChanged"]], template: function PluginSelectorContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "plugin-selector-component", 0);
            ɵɵlistener("onPluginSelectionChanged", function PluginSelectorContainer_Template_plugin_selector_component_onPluginSelectionChanged_0_listener($event) { return ctx.onPluginSelectionChange($event); });
            ɵɵpipe(1, "async");
            ɵɵpipe(2, "async");
            ɵɵpipe(3, "async");
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵproperty("activePlugins", ɵɵpipeBind1(1, 3, ctx.plugins$))("disabledPlugins", ɵɵpipeBind1(2, 5, ctx.disabledPlugins$))("selectedPlugin", ɵɵpipeBind1(3, 7, ctx.activePlugin$));
        } }, directives: [PluginSelectorComponent], pipes: [AsyncPipe], encapsulation: 2 });

    function MatDialogContainer_ng_template_0_Template(rf, ctx) { }
    /**
     * Configuration for opening a modal dialog with the MatDialog service.
     * @template D
     */
    class MatDialogConfig {
        constructor() {
            /**
             * The ARIA role of the dialog element.
             */
            this.role = 'dialog';
            /**
             * Custom class for the overlay pane.
             */
            this.panelClass = '';
            /**
             * Whether the dialog has a backdrop.
             */
            this.hasBackdrop = true;
            /**
             * Custom class for the backdrop.
             */
            this.backdropClass = '';
            /**
             * Whether the user can use escape or clicking on the backdrop to close the modal.
             */
            this.disableClose = false;
            /**
             * Width of the dialog.
             */
            this.width = '';
            /**
             * Height of the dialog.
             */
            this.height = '';
            /**
             * Max-width of the dialog. If a number is provided, assumes pixel units. Defaults to 80vw.
             */
            this.maxWidth = '80vw';
            /**
             * Data being injected into the child component.
             */
            this.data = null;
            /**
             * ID of the element that describes the dialog.
             */
            this.ariaDescribedBy = null;
            /**
             * ID of the element that labels the dialog.
             */
            this.ariaLabelledBy = null;
            /**
             * Aria label to assign to the dialog element.
             */
            this.ariaLabel = null;
            /**
             * Whether the dialog should focus the first focusable element on open.
             */
            this.autoFocus = true;
            /**
             * Whether the dialog should restore focus to the
             * previously-focused element, after it's closed.
             */
            this.restoreFocus = true;
            /**
             * Whether the dialog should close when the user goes backwards/forwards in history.
             * Note that this usually doesn't include clicking on links (unless the user is using
             * the `HashLocationStrategy`).
             */
            this.closeOnNavigation = true;
            // TODO(jelbourn): add configuration for lifecycle hooks, ARIA labelling.
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/dialog/dialog-animations.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Animations used by MatDialog.
     * \@docs-private
     * @type {?}
     */
    const matDialogAnimations = {
        /**
         * Animation that is applied on the dialog container by defalt.
         */
        dialogContainer: trigger('dialogContainer', [
            // Note: The `enter` animation transitions to `transform: none`, because for some reason
            // specifying the transform explicitly, causes IE both to blur the dialog content and
            // decimate the animation performance. Leaving it as `none` solves both issues.
            state('void, exit', style({ opacity: 0, transform: 'scale(0.7)' })),
            state('enter', style({ transform: 'none' })),
            transition('* => enter', animate('150ms cubic-bezier(0, 0, 0.2, 1)', style({ transform: 'none', opacity: 1 }))),
            transition('* => void, * => exit', animate('75ms cubic-bezier(0.4, 0.0, 0.2, 1)', style({ opacity: 0 }))),
        ])
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/dialog/dialog-container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Throws an exception for the case when a ComponentPortal is
     * attached to a DomPortalOutlet without an origin.
     * \@docs-private
     * @return {?}
     */
    function throwMatDialogContentAlreadyAttachedError() {
        throw Error('Attempting to attach dialog content after content is already attached');
    }
    /**
     * Internal component that wraps user-provided dialog content.
     * Animation is based on https://material.io/guidelines/motion/choreography.html.
     * \@docs-private
     */
    class MatDialogContainer extends BasePortalOutlet {
        /**
         * @param {?} _elementRef
         * @param {?} _focusTrapFactory
         * @param {?} _changeDetectorRef
         * @param {?} _document
         * @param {?} _config
         */
        constructor(_elementRef, _focusTrapFactory, _changeDetectorRef, _document, _config) {
            super();
            this._elementRef = _elementRef;
            this._focusTrapFactory = _focusTrapFactory;
            this._changeDetectorRef = _changeDetectorRef;
            this._config = _config;
            /**
             * Element that was focused before the dialog was opened. Save this to restore upon close.
             */
            this._elementFocusedBeforeDialogWasOpened = null;
            /**
             * State of the dialog animation.
             */
            this._state = 'enter';
            /**
             * Emits when an animation state changes.
             */
            this._animationStateChanged = new EventEmitter();
            /**
             * Attaches a DOM portal to the dialog container.
             * @param portal Portal to be attached.
             * @deprecated To be turned into a method.
             * \@breaking-change 10.0.0
             */
            this.attachDomPortal = (/**
             * @param {?} portal
             * @return {?}
             */
            (portal) => {
                if (this._portalOutlet.hasAttached()) {
                    throwMatDialogContentAlreadyAttachedError();
                }
                this._setupFocusTrap();
                return this._portalOutlet.attachDomPortal(portal);
            });
            this._ariaLabelledBy = _config.ariaLabelledBy || null;
            this._document = _document;
        }
        /**
         * Attach a ComponentPortal as content to this dialog container.
         * @template T
         * @param {?} portal Portal to be attached as the dialog content.
         * @return {?}
         */
        attachComponentPortal(portal) {
            if (this._portalOutlet.hasAttached()) {
                throwMatDialogContentAlreadyAttachedError();
            }
            this._setupFocusTrap();
            return this._portalOutlet.attachComponentPortal(portal);
        }
        /**
         * Attach a TemplatePortal as content to this dialog container.
         * @template C
         * @param {?} portal Portal to be attached as the dialog content.
         * @return {?}
         */
        attachTemplatePortal(portal) {
            if (this._portalOutlet.hasAttached()) {
                throwMatDialogContentAlreadyAttachedError();
            }
            this._setupFocusTrap();
            return this._portalOutlet.attachTemplatePortal(portal);
        }
        /**
         * Moves focus back into the dialog if it was moved out.
         * @return {?}
         */
        _recaptureFocus() {
            if (!this._containsFocus()) {
                /** @type {?} */
                const focusWasTrapped = this._focusTrap.focusInitialElement();
                if (!focusWasTrapped) {
                    this._elementRef.nativeElement.focus();
                }
            }
        }
        /**
         * Moves the focus inside the focus trap.
         * @private
         * @return {?}
         */
        _trapFocus() {
            // If we were to attempt to focus immediately, then the content of the dialog would not yet be
            // ready in instances where change detection has to run first. To deal with this, we simply
            // wait for the microtask queue to be empty.
            if (this._config.autoFocus) {
                this._focusTrap.focusInitialElementWhenReady();
            }
            else if (!this._containsFocus()) {
                // Otherwise ensure that focus is on the dialog container. It's possible that a different
                // component tried to move focus while the open animation was running. See:
                // https://github.com/angular/components/issues/16215. Note that we only want to do this
                // if the focus isn't inside the dialog already, because it's possible that the consumer
                // turned off `autoFocus` in order to move focus themselves.
                this._elementRef.nativeElement.focus();
            }
        }
        /**
         * Restores focus to the element that was focused before the dialog opened.
         * @private
         * @return {?}
         */
        _restoreFocus() {
            /** @type {?} */
            const toFocus = this._elementFocusedBeforeDialogWasOpened;
            // We need the extra check, because IE can set the `activeElement` to null in some cases.
            if (this._config.restoreFocus && toFocus && typeof toFocus.focus === 'function') {
                /** @type {?} */
                const activeElement = this._document.activeElement;
                /** @type {?} */
                const element = this._elementRef.nativeElement;
                // Make sure that focus is still inside the dialog or is on the body (usually because a
                // non-focusable element like the backdrop was clicked) before moving it. It's possible that
                // the consumer moved it themselves before the animation was done, in which case we shouldn't
                // do anything.
                if (!activeElement || activeElement === this._document.body || activeElement === element ||
                    element.contains(activeElement)) {
                    toFocus.focus();
                }
            }
            if (this._focusTrap) {
                this._focusTrap.destroy();
            }
        }
        /**
         * Sets up the focus trand and saves a reference to the
         * element that was focused before the dialog was opened.
         * @private
         * @return {?}
         */
        _setupFocusTrap() {
            if (!this._focusTrap) {
                this._focusTrap = this._focusTrapFactory.create(this._elementRef.nativeElement);
            }
            if (this._document) {
                this._elementFocusedBeforeDialogWasOpened = (/** @type {?} */ (this._document.activeElement));
                // Note that there is no focus method when rendering on the server.
                if (this._elementRef.nativeElement.focus) {
                    // Move focus onto the dialog immediately in order to prevent the user from accidentally
                    // opening multiple dialogs at the same time. Needs to be async, because the element
                    // may not be focusable immediately.
                    Promise.resolve().then((/**
                     * @return {?}
                     */
                    () => this._elementRef.nativeElement.focus()));
                }
            }
        }
        /**
         * Returns whether focus is inside the dialog.
         * @private
         * @return {?}
         */
        _containsFocus() {
            /** @type {?} */
            const element = this._elementRef.nativeElement;
            /** @type {?} */
            const activeElement = this._document.activeElement;
            return element === activeElement || element.contains(activeElement);
        }
        /**
         * Callback, invoked whenever an animation on the host completes.
         * @param {?} event
         * @return {?}
         */
        _onAnimationDone(event) {
            if (event.toState === 'enter') {
                this._trapFocus();
            }
            else if (event.toState === 'exit') {
                this._restoreFocus();
            }
            this._animationStateChanged.emit(event);
        }
        /**
         * Callback, invoked when an animation on the host starts.
         * @param {?} event
         * @return {?}
         */
        _onAnimationStart(event) {
            this._animationStateChanged.emit(event);
        }
        /**
         * Starts the dialog exit animation.
         * @return {?}
         */
        _startExitAnimation() {
            this._state = 'exit';
            // Mark the container for check so it can react if the
            // view container is using OnPush change detection.
            this._changeDetectorRef.markForCheck();
        }
    }
    MatDialogContainer.ɵfac = function MatDialogContainer_Factory(t) { return new (t || MatDialogContainer)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(FocusTrapFactory), ɵɵdirectiveInject(ChangeDetectorRef), ɵɵdirectiveInject(DOCUMENT$1, 8), ɵɵdirectiveInject(MatDialogConfig)); };
    MatDialogContainer.ɵcmp = ɵɵdefineComponent({ type: MatDialogContainer, selectors: [["mat-dialog-container"]], viewQuery: function MatDialogContainer_Query(rf, ctx) { if (rf & 1) {
            ɵɵstaticViewQuery(CdkPortalOutlet, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._portalOutlet = _t.first);
        } }, hostAttrs: ["tabindex", "-1", "aria-modal", "true", 1, "mat-dialog-container"], hostVars: 6, hostBindings: function MatDialogContainer_HostBindings(rf, ctx) { if (rf & 1) {
            ɵɵcomponentHostSyntheticListener("@dialogContainer.start", function MatDialogContainer_animation_dialogContainer_start_HostBindingHandler($event) { return ctx._onAnimationStart($event); })("@dialogContainer.done", function MatDialogContainer_animation_dialogContainer_done_HostBindingHandler($event) { return ctx._onAnimationDone($event); });
        } if (rf & 2) {
            ɵɵattribute("id", ctx._id)("role", ctx._config.role)("aria-labelledby", ctx._config.ariaLabel ? null : ctx._ariaLabelledBy)("aria-label", ctx._config.ariaLabel)("aria-describedby", ctx._config.ariaDescribedBy || null);
            ɵɵupdateSyntheticHostBinding("@dialogContainer", ctx._state);
        } }, features: [ɵɵInheritDefinitionFeature], decls: 1, vars: 0, consts: [["cdkPortalOutlet", ""]], template: function MatDialogContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵtemplate(0, MatDialogContainer_ng_template_0_Template, 0, 0, "ng-template", 0);
        } }, directives: [CdkPortalOutlet], styles: [".mat-dialog-container{display:block;padding:24px;border-radius:4px;box-sizing:border-box;overflow:auto;outline:0;width:100%;height:100%;min-height:inherit;max-height:inherit}.cdk-high-contrast-active .mat-dialog-container{outline:solid 1px}.mat-dialog-content{display:block;margin:0 -24px;padding:0 24px;max-height:65vh;overflow:auto;-webkit-overflow-scrolling:touch}.mat-dialog-title{margin:0 0 20px;display:block}.mat-dialog-actions{padding:8px 0;display:flex;flex-wrap:wrap;min-height:52px;align-items:center;margin-bottom:-24px}.mat-dialog-actions[align=end]{justify-content:flex-end}.mat-dialog-actions[align=center]{justify-content:center}.mat-dialog-actions .mat-button-base+.mat-button-base,.mat-dialog-actions .mat-mdc-button-base+.mat-mdc-button-base{margin-left:8px}[dir=rtl] .mat-dialog-actions .mat-button-base+.mat-button-base,[dir=rtl] .mat-dialog-actions .mat-mdc-button-base+.mat-mdc-button-base{margin-left:0;margin-right:8px}\n"], encapsulation: 2, data: { animation: [matDialogAnimations.dialogContainer] } });
    /** @nocollapse */
    MatDialogContainer.ctorParameters = () => [
        { type: ElementRef },
        { type: FocusTrapFactory },
        { type: ChangeDetectorRef },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [DOCUMENT$1,] }] },
        { type: MatDialogConfig }
    ];
    MatDialogContainer.propDecorators = {
        _portalOutlet: [{ type: ViewChild, args: [CdkPortalOutlet, { static: true },] }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/dialog/dialog-ref.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    // TODO(jelbourn): resizing
    // Counter for unique dialog ids.
    /** @type {?} */
    let uniqueId = 0;
    /**
     * Reference to a dialog opened via the MatDialog service.
     * @template T, R
     */
    class MatDialogRef {
        /**
         * @param {?} _overlayRef
         * @param {?} _containerInstance
         * @param {?=} id
         */
        constructor(_overlayRef, _containerInstance, id = `mat-dialog-${uniqueId++}`) {
            this._overlayRef = _overlayRef;
            this._containerInstance = _containerInstance;
            this.id = id;
            /**
             * Whether the user is allowed to close the dialog.
             */
            this.disableClose = this._containerInstance._config.disableClose;
            /**
             * Subject for notifying the user that the dialog has finished opening.
             */
            this._afterOpened = new Subject();
            /**
             * Subject for notifying the user that the dialog has finished closing.
             */
            this._afterClosed = new Subject();
            /**
             * Subject for notifying the user that the dialog has started closing.
             */
            this._beforeClosed = new Subject();
            /**
             * Current state of the dialog.
             */
            this._state = 0 /* OPEN */;
            // Pass the id along to the container.
            _containerInstance._id = id;
            // Emit when opening animation completes
            _containerInstance._animationStateChanged.pipe(filter((/**
             * @param {?} event
             * @return {?}
             */
            event => event.phaseName === 'done' && event.toState === 'enter')), take(1))
                .subscribe((/**
             * @return {?}
             */
            () => {
                this._afterOpened.next();
                this._afterOpened.complete();
            }));
            // Dispose overlay when closing animation is complete
            _containerInstance._animationStateChanged.pipe(filter((/**
             * @param {?} event
             * @return {?}
             */
            event => event.phaseName === 'done' && event.toState === 'exit')), take(1)).subscribe((/**
             * @return {?}
             */
            () => {
                clearTimeout(this._closeFallbackTimeout);
                this._finishDialogClose();
            }));
            _overlayRef.detachments().subscribe((/**
             * @return {?}
             */
            () => {
                this._beforeClosed.next(this._result);
                this._beforeClosed.complete();
                this._afterClosed.next(this._result);
                this._afterClosed.complete();
                this.componentInstance = (/** @type {?} */ (null));
                this._overlayRef.dispose();
            }));
            _overlayRef.keydownEvents()
                .pipe(filter((/**
             * @param {?} event
             * @return {?}
             */
            event => {
                return event.keyCode === ESCAPE && !this.disableClose && !hasModifierKey(event);
            })))
                .subscribe((/**
             * @param {?} event
             * @return {?}
             */
            event => {
                event.preventDefault();
                this.close();
            }));
            _overlayRef.backdropClick().subscribe((/**
             * @return {?}
             */
            () => {
                if (this.disableClose) {
                    this._containerInstance._recaptureFocus();
                }
                else {
                    this.close();
                }
            }));
        }
        /**
         * Close the dialog.
         * @param {?=} dialogResult Optional result to return to the dialog opener.
         * @return {?}
         */
        close(dialogResult) {
            this._result = dialogResult;
            // Transition the backdrop in parallel to the dialog.
            this._containerInstance._animationStateChanged.pipe(filter((/**
             * @param {?} event
             * @return {?}
             */
            event => event.phaseName === 'start')), take(1))
                .subscribe((/**
             * @param {?} event
             * @return {?}
             */
            event => {
                this._beforeClosed.next(dialogResult);
                this._beforeClosed.complete();
                this._overlayRef.detachBackdrop();
                // The logic that disposes of the overlay depends on the exit animation completing, however
                // it isn't guaranteed if the parent view is destroyed while it's running. Add a fallback
                // timeout which will clean everything up if the animation hasn't fired within the specified
                // amount of time plus 100ms. We don't need to run this outside the NgZone, because for the
                // vast majority of cases the timeout will have been cleared before it has the chance to fire.
                this._closeFallbackTimeout = setTimeout((/**
                 * @return {?}
                 */
                () => this._finishDialogClose()), event.totalTime + 100);
            }));
            this._containerInstance._startExitAnimation();
            this._state = 1 /* CLOSING */;
        }
        /**
         * Gets an observable that is notified when the dialog is finished opening.
         * @return {?}
         */
        afterOpened() {
            return this._afterOpened.asObservable();
        }
        /**
         * Gets an observable that is notified when the dialog is finished closing.
         * @return {?}
         */
        afterClosed() {
            return this._afterClosed.asObservable();
        }
        /**
         * Gets an observable that is notified when the dialog has started closing.
         * @return {?}
         */
        beforeClosed() {
            return this._beforeClosed.asObservable();
        }
        /**
         * Gets an observable that emits when the overlay's backdrop has been clicked.
         * @return {?}
         */
        backdropClick() {
            return this._overlayRef.backdropClick();
        }
        /**
         * Gets an observable that emits when keydown events are targeted on the overlay.
         * @return {?}
         */
        keydownEvents() {
            return this._overlayRef.keydownEvents();
        }
        /**
         * Updates the dialog's position.
         * @template THIS
         * @this {THIS}
         * @param {?=} position New dialog position.
         * @return {THIS}
         */
        updatePosition(position) {
            /** @type {?} */
            let strategy = (/** @type {?} */ (this))._getPositionStrategy();
            if (position && (position.left || position.right)) {
                position.left ? strategy.left(position.left) : strategy.right(position.right);
            }
            else {
                strategy.centerHorizontally();
            }
            if (position && (position.top || position.bottom)) {
                position.top ? strategy.top(position.top) : strategy.bottom(position.bottom);
            }
            else {
                strategy.centerVertically();
            }
            (/** @type {?} */ (this))._overlayRef.updatePosition();
            return (/** @type {?} */ (this));
        }
        /**
         * Updates the dialog's width and height.
         * @template THIS
         * @this {THIS}
         * @param {?=} width New width of the dialog.
         * @param {?=} height New height of the dialog.
         * @return {THIS}
         */
        updateSize(width = '', height = '') {
            (/** @type {?} */ (this))._getPositionStrategy().width(width).height(height);
            (/** @type {?} */ (this))._overlayRef.updatePosition();
            return (/** @type {?} */ (this));
        }
        /**
         * Add a CSS class or an array of classes to the overlay pane.
         * @template THIS
         * @this {THIS}
         * @param {?} classes
         * @return {THIS}
         */
        addPanelClass(classes) {
            (/** @type {?} */ (this))._overlayRef.addPanelClass(classes);
            return (/** @type {?} */ (this));
        }
        /**
         * Remove a CSS class or an array of classes from the overlay pane.
         * @template THIS
         * @this {THIS}
         * @param {?} classes
         * @return {THIS}
         */
        removePanelClass(classes) {
            (/** @type {?} */ (this))._overlayRef.removePanelClass(classes);
            return (/** @type {?} */ (this));
        }
        /**
         * Gets the current state of the dialog's lifecycle.
         * @return {?}
         */
        getState() {
            return this._state;
        }
        /**
         * Finishes the dialog close by updating the state of the dialog
         * and disposing the overlay.
         * @private
         * @return {?}
         */
        _finishDialogClose() {
            this._state = 2 /* CLOSED */;
            this._overlayRef.dispose();
        }
        /**
         * Fetches the position strategy object from the overlay ref.
         * @private
         * @return {?}
         */
        _getPositionStrategy() {
            return (/** @type {?} */ (this._overlayRef.getConfig().positionStrategy));
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/dialog/dialog.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Injection token that can be used to access the data that was passed in to a dialog.
     * @type {?}
     */
    const MAT_DIALOG_DATA = new InjectionToken('MatDialogData');
    /**
     * Injection token that can be used to specify default dialog options.
     * @type {?}
     */
    const MAT_DIALOG_DEFAULT_OPTIONS = new InjectionToken('mat-dialog-default-options');
    /**
     * Injection token that determines the scroll handling while the dialog is open.
     * @type {?}
     */
    const MAT_DIALOG_SCROLL_STRATEGY = new InjectionToken('mat-dialog-scroll-strategy');
    /**
     * \@docs-private
     * @param {?} overlay
     * @return {?}
     */
    function MAT_DIALOG_SCROLL_STRATEGY_PROVIDER_FACTORY(overlay) {
        return (/**
         * @return {?}
         */
        () => overlay.scrollStrategies.block());
    }
    /**
     * \@docs-private
     * @type {?}
     */
    const MAT_DIALOG_SCROLL_STRATEGY_PROVIDER = {
        provide: MAT_DIALOG_SCROLL_STRATEGY,
        deps: [Overlay],
        useFactory: MAT_DIALOG_SCROLL_STRATEGY_PROVIDER_FACTORY,
    };
    /**
     * Service to open Material Design modal dialogs.
     */
    class MatDialog {
        /**
         * @param {?} _overlay
         * @param {?} _injector
         * @param {?} _location
         * @param {?} _defaultOptions
         * @param {?} scrollStrategy
         * @param {?} _parentDialog
         * @param {?} _overlayContainer
         */
        constructor(_overlay, _injector, 
        /**
         * @deprecated `_location` parameter to be removed.
         * @breaking-change 10.0.0
         */
        _location, _defaultOptions, scrollStrategy, _parentDialog, _overlayContainer) {
            this._overlay = _overlay;
            this._injector = _injector;
            this._defaultOptions = _defaultOptions;
            this._parentDialog = _parentDialog;
            this._overlayContainer = _overlayContainer;
            this._openDialogsAtThisLevel = [];
            this._afterAllClosedAtThisLevel = new Subject();
            this._afterOpenedAtThisLevel = new Subject();
            this._ariaHiddenElements = new Map();
            // TODO (jelbourn): tighten the typing right-hand side of this expression.
            /**
             * Stream that emits when all open dialog have finished closing.
             * Will emit on subscribe if there are no open dialogs to begin with.
             */
            this.afterAllClosed = (/** @type {?} */ (defer((/**
             * @return {?}
             */
            () => this.openDialogs.length ?
                this._afterAllClosed :
                this._afterAllClosed.pipe(startWith(undefined))))));
            this._scrollStrategy = scrollStrategy;
        }
        /**
         * Keeps track of the currently-open dialogs.
         * @return {?}
         */
        get openDialogs() {
            return this._parentDialog ? this._parentDialog.openDialogs : this._openDialogsAtThisLevel;
        }
        /**
         * Stream that emits when a dialog has been opened.
         * @return {?}
         */
        get afterOpened() {
            return this._parentDialog ? this._parentDialog.afterOpened : this._afterOpenedAtThisLevel;
        }
        /**
         * @return {?}
         */
        get _afterAllClosed() {
            /** @type {?} */
            const parent = this._parentDialog;
            return parent ? parent._afterAllClosed : this._afterAllClosedAtThisLevel;
        }
        /**
         * Opens a modal dialog containing the given component.
         * @template T, D, R
         * @param {?} componentOrTemplateRef Type of the component to load into the dialog,
         *     or a TemplateRef to instantiate as the dialog content.
         * @param {?=} config Extra configuration options.
         * @return {?} Reference to the newly-opened dialog.
         */
        open(componentOrTemplateRef, config) {
            config = _applyConfigDefaults(config, this._defaultOptions || new MatDialogConfig());
            if (config.id && this.getDialogById(config.id)) {
                throw Error(`Dialog with id "${config.id}" exists already. The dialog id must be unique.`);
            }
            /** @type {?} */
            const overlayRef = this._createOverlay(config);
            /** @type {?} */
            const dialogContainer = this._attachDialogContainer(overlayRef, config);
            /** @type {?} */
            const dialogRef = this._attachDialogContent(componentOrTemplateRef, dialogContainer, overlayRef, config);
            // If this is the first dialog that we're opening, hide all the non-overlay content.
            if (!this.openDialogs.length) {
                this._hideNonDialogContentFromAssistiveTechnology();
            }
            this.openDialogs.push(dialogRef);
            dialogRef.afterClosed().subscribe((/**
             * @return {?}
             */
            () => this._removeOpenDialog(dialogRef)));
            this.afterOpened.next(dialogRef);
            return dialogRef;
        }
        /**
         * Closes all of the currently-open dialogs.
         * @return {?}
         */
        closeAll() {
            this._closeDialogs(this.openDialogs);
        }
        /**
         * Finds an open dialog by its id.
         * @param {?} id ID to use when looking up the dialog.
         * @return {?}
         */
        getDialogById(id) {
            return this.openDialogs.find((/**
             * @param {?} dialog
             * @return {?}
             */
            dialog => dialog.id === id));
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            // Only close the dialogs at this level on destroy
            // since the parent service may still be active.
            this._closeDialogs(this._openDialogsAtThisLevel);
            this._afterAllClosedAtThisLevel.complete();
            this._afterOpenedAtThisLevel.complete();
        }
        /**
         * Creates the overlay into which the dialog will be loaded.
         * @private
         * @param {?} config The dialog configuration.
         * @return {?} A promise resolving to the OverlayRef for the created overlay.
         */
        _createOverlay(config) {
            /** @type {?} */
            const overlayConfig = this._getOverlayConfig(config);
            return this._overlay.create(overlayConfig);
        }
        /**
         * Creates an overlay config from a dialog config.
         * @private
         * @param {?} dialogConfig The dialog configuration.
         * @return {?} The overlay configuration.
         */
        _getOverlayConfig(dialogConfig) {
            /** @type {?} */
            const state = new OverlayConfig({
                positionStrategy: this._overlay.position().global(),
                scrollStrategy: dialogConfig.scrollStrategy || this._scrollStrategy(),
                panelClass: dialogConfig.panelClass,
                hasBackdrop: dialogConfig.hasBackdrop,
                direction: dialogConfig.direction,
                minWidth: dialogConfig.minWidth,
                minHeight: dialogConfig.minHeight,
                maxWidth: dialogConfig.maxWidth,
                maxHeight: dialogConfig.maxHeight,
                disposeOnNavigation: dialogConfig.closeOnNavigation
            });
            if (dialogConfig.backdropClass) {
                state.backdropClass = dialogConfig.backdropClass;
            }
            return state;
        }
        /**
         * Attaches an MatDialogContainer to a dialog's already-created overlay.
         * @private
         * @param {?} overlay Reference to the dialog's underlying overlay.
         * @param {?} config The dialog configuration.
         * @return {?} A promise resolving to a ComponentRef for the attached container.
         */
        _attachDialogContainer(overlay, config) {
            /** @type {?} */
            const userInjector = config && config.viewContainerRef && config.viewContainerRef.injector;
            /** @type {?} */
            const injector = Injector.create({
                parent: userInjector || this._injector,
                providers: [{ provide: MatDialogConfig, useValue: config }]
            });
            /** @type {?} */
            const containerPortal = new ComponentPortal(MatDialogContainer, config.viewContainerRef, injector, config.componentFactoryResolver);
            /** @type {?} */
            const containerRef = overlay.attach(containerPortal);
            return containerRef.instance;
        }
        /**
         * Attaches the user-provided component to the already-created MatDialogContainer.
         * @private
         * @template T, R
         * @param {?} componentOrTemplateRef The type of component being loaded into the dialog,
         *     or a TemplateRef to instantiate as the content.
         * @param {?} dialogContainer Reference to the wrapping MatDialogContainer.
         * @param {?} overlayRef Reference to the overlay in which the dialog resides.
         * @param {?} config The dialog configuration.
         * @return {?} A promise resolving to the MatDialogRef that should be returned to the user.
         */
        _attachDialogContent(componentOrTemplateRef, dialogContainer, overlayRef, config) {
            // Create a reference to the dialog we're creating in order to give the user a handle
            // to modify and close it.
            /** @type {?} */
            const dialogRef = new MatDialogRef(overlayRef, dialogContainer, config.id);
            if (componentOrTemplateRef instanceof TemplateRef) {
                dialogContainer.attachTemplatePortal(new TemplatePortal(componentOrTemplateRef, (/** @type {?} */ (null)), (/** @type {?} */ ({ $implicit: config.data, dialogRef }))));
            }
            else {
                /** @type {?} */
                const injector = this._createInjector(config, dialogRef, dialogContainer);
                /** @type {?} */
                const contentRef = dialogContainer.attachComponentPortal(new ComponentPortal(componentOrTemplateRef, config.viewContainerRef, injector));
                dialogRef.componentInstance = contentRef.instance;
            }
            dialogRef
                .updateSize(config.width, config.height)
                .updatePosition(config.position);
            return dialogRef;
        }
        /**
         * Creates a custom injector to be used inside the dialog. This allows a component loaded inside
         * of a dialog to close itself and, optionally, to return a value.
         * @private
         * @template T
         * @param {?} config Config object that is used to construct the dialog.
         * @param {?} dialogRef Reference to the dialog.
         * @param {?} dialogContainer
         * @return {?} The custom injector that can be used inside the dialog.
         */
        _createInjector(config, dialogRef, dialogContainer) {
            /** @type {?} */
            const userInjector = config && config.viewContainerRef && config.viewContainerRef.injector;
            // The MatDialogContainer is injected in the portal as the MatDialogContainer and the dialog's
            // content are created out of the same ViewContainerRef and as such, are siblings for injector
            // purposes. To allow the hierarchy that is expected, the MatDialogContainer is explicitly
            // added to the injection tokens.
            /** @type {?} */
            const providers = [
                { provide: MatDialogContainer, useValue: dialogContainer },
                { provide: MAT_DIALOG_DATA, useValue: config.data },
                { provide: MatDialogRef, useValue: dialogRef }
            ];
            if (config.direction &&
                (!userInjector || !userInjector.get(Directionality, null))) {
                providers.push({
                    provide: Directionality,
                    useValue: { value: config.direction, change: of() }
                });
            }
            return Injector.create({ parent: userInjector || this._injector, providers });
        }
        /**
         * Removes a dialog from the array of open dialogs.
         * @private
         * @param {?} dialogRef Dialog to be removed.
         * @return {?}
         */
        _removeOpenDialog(dialogRef) {
            /** @type {?} */
            const index = this.openDialogs.indexOf(dialogRef);
            if (index > -1) {
                this.openDialogs.splice(index, 1);
                // If all the dialogs were closed, remove/restore the `aria-hidden`
                // to a the siblings and emit to the `afterAllClosed` stream.
                if (!this.openDialogs.length) {
                    this._ariaHiddenElements.forEach((/**
                     * @param {?} previousValue
                     * @param {?} element
                     * @return {?}
                     */
                    (previousValue, element) => {
                        if (previousValue) {
                            element.setAttribute('aria-hidden', previousValue);
                        }
                        else {
                            element.removeAttribute('aria-hidden');
                        }
                    }));
                    this._ariaHiddenElements.clear();
                    this._afterAllClosed.next();
                }
            }
        }
        /**
         * Hides all of the content that isn't an overlay from assistive technology.
         * @private
         * @return {?}
         */
        _hideNonDialogContentFromAssistiveTechnology() {
            /** @type {?} */
            const overlayContainer = this._overlayContainer.getContainerElement();
            // Ensure that the overlay container is attached to the DOM.
            if (overlayContainer.parentElement) {
                /** @type {?} */
                const siblings = overlayContainer.parentElement.children;
                for (let i = siblings.length - 1; i > -1; i--) {
                    /** @type {?} */
                    let sibling = siblings[i];
                    if (sibling !== overlayContainer &&
                        sibling.nodeName !== 'SCRIPT' &&
                        sibling.nodeName !== 'STYLE' &&
                        !sibling.hasAttribute('aria-live')) {
                        this._ariaHiddenElements.set(sibling, sibling.getAttribute('aria-hidden'));
                        sibling.setAttribute('aria-hidden', 'true');
                    }
                }
            }
        }
        /**
         * Closes all of the dialogs in an array.
         * @private
         * @param {?} dialogs
         * @return {?}
         */
        _closeDialogs(dialogs) {
            /** @type {?} */
            let i = dialogs.length;
            while (i--) {
                // The `_openDialogs` property isn't updated after close until the rxjs subscription
                // runs on the next microtask, in addition to modifying the array as we're going
                // through it. We loop through all of them and call close without assuming that
                // they'll be removed from the list instantaneously.
                dialogs[i].close();
            }
        }
    }
    MatDialog.ɵfac = function MatDialog_Factory(t) { return new (t || MatDialog)(ɵɵinject(Overlay), ɵɵinject(Injector), ɵɵinject(Location, 8), ɵɵinject(MAT_DIALOG_DEFAULT_OPTIONS, 8), ɵɵinject(MAT_DIALOG_SCROLL_STRATEGY), ɵɵinject(MatDialog, 12), ɵɵinject(OverlayContainer)); };
    MatDialog.ɵprov = ɵɵdefineInjectable({ token: MatDialog, factory: MatDialog.ɵfac });
    /** @nocollapse */
    MatDialog.ctorParameters = () => [
        { type: Overlay },
        { type: Injector },
        { type: Location, decorators: [{ type: Optional }] },
        { type: MatDialogConfig, decorators: [{ type: Optional }, { type: Inject, args: [MAT_DIALOG_DEFAULT_OPTIONS,] }] },
        { type: undefined, decorators: [{ type: Inject, args: [MAT_DIALOG_SCROLL_STRATEGY,] }] },
        { type: MatDialog, decorators: [{ type: Optional }, { type: SkipSelf }] },
        { type: OverlayContainer }
    ];
    /**
     * Applies default options to the dialog config.
     * @param {?=} config Config to be modified.
     * @param {?=} defaultOptions Default options provided.
     * @return {?} The new configuration object.
     */
    function _applyConfigDefaults(config, defaultOptions) {
        return Object.assign(Object.assign({}, defaultOptions), config);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/dialog/dialog-content-directives.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Counter used to generate unique IDs for dialog elements.
     * @type {?}
     */
    let dialogElementUid = 0;
    /**
     * Button that will close the current dialog.
     */
    class MatDialogClose {
        /**
         * @param {?} dialogRef
         * @param {?} _elementRef
         * @param {?} _dialog
         */
        constructor(dialogRef, _elementRef, _dialog) {
            this.dialogRef = dialogRef;
            this._elementRef = _elementRef;
            this._dialog = _dialog;
            /**
             * Default to "button" to prevents accidental form submits.
             */
            this.type = 'button';
        }
        /**
         * @return {?}
         */
        ngOnInit() {
            if (!this.dialogRef) {
                // When this directive is included in a dialog via TemplateRef (rather than being
                // in a Component), the DialogRef isn't available via injection because embedded
                // views cannot be given a custom injector. Instead, we look up the DialogRef by
                // ID. This must occur in `onInit`, as the ID binding for the dialog container won't
                // be resolved at constructor time.
                this.dialogRef = (/** @type {?} */ (getClosestDialog(this._elementRef, this._dialog.openDialogs)));
            }
        }
        /**
         * @param {?} changes
         * @return {?}
         */
        ngOnChanges(changes) {
            /** @type {?} */
            const proxiedChange = changes['_matDialogClose'] || changes['_matDialogCloseResult'];
            if (proxiedChange) {
                this.dialogResult = proxiedChange.currentValue;
            }
        }
    }
    MatDialogClose.ɵfac = function MatDialogClose_Factory(t) { return new (t || MatDialogClose)(ɵɵdirectiveInject(MatDialogRef, 8), ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(MatDialog)); };
    MatDialogClose.ɵdir = ɵɵdefineDirective({ type: MatDialogClose, selectors: [["", "mat-dialog-close", ""], ["", "matDialogClose", ""]], hostVars: 2, hostBindings: function MatDialogClose_HostBindings(rf, ctx) { if (rf & 1) {
            ɵɵlistener("click", function MatDialogClose_click_HostBindingHandler() { return ctx.dialogRef.close(ctx.dialogResult); });
        } if (rf & 2) {
            ɵɵattribute("aria-label", ctx.ariaLabel || null)("type", ctx.type);
        } }, inputs: { type: "type", dialogResult: ["mat-dialog-close", "dialogResult"], ariaLabel: ["aria-label", "ariaLabel"], _matDialogClose: ["matDialogClose", "_matDialogClose"] }, exportAs: ["matDialogClose"], features: [ɵɵNgOnChangesFeature] });
    /** @nocollapse */
    MatDialogClose.ctorParameters = () => [
        { type: MatDialogRef, decorators: [{ type: Optional }] },
        { type: ElementRef },
        { type: MatDialog }
    ];
    MatDialogClose.propDecorators = {
        ariaLabel: [{ type: Input, args: ['aria-label',] }],
        type: [{ type: Input }],
        dialogResult: [{ type: Input, args: ['mat-dialog-close',] }],
        _matDialogClose: [{ type: Input, args: ['matDialogClose',] }]
    };
    /**
     * Title of a dialog element. Stays fixed to the top of the dialog when scrolling.
     */
    class MatDialogTitle {
        /**
         * @param {?} _dialogRef
         * @param {?} _elementRef
         * @param {?} _dialog
         */
        constructor(_dialogRef, _elementRef, _dialog) {
            this._dialogRef = _dialogRef;
            this._elementRef = _elementRef;
            this._dialog = _dialog;
            this.id = `mat-dialog-title-${dialogElementUid++}`;
        }
        /**
         * @return {?}
         */
        ngOnInit() {
            if (!this._dialogRef) {
                this._dialogRef = (/** @type {?} */ (getClosestDialog(this._elementRef, this._dialog.openDialogs)));
            }
            if (this._dialogRef) {
                Promise.resolve().then((/**
                 * @return {?}
                 */
                () => {
                    /** @type {?} */
                    const container = this._dialogRef._containerInstance;
                    if (container && !container._ariaLabelledBy) {
                        container._ariaLabelledBy = this.id;
                    }
                }));
            }
        }
    }
    MatDialogTitle.ɵfac = function MatDialogTitle_Factory(t) { return new (t || MatDialogTitle)(ɵɵdirectiveInject(MatDialogRef, 8), ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(MatDialog)); };
    MatDialogTitle.ɵdir = ɵɵdefineDirective({ type: MatDialogTitle, selectors: [["", "mat-dialog-title", ""], ["", "matDialogTitle", ""]], hostAttrs: [1, "mat-dialog-title"], hostVars: 1, hostBindings: function MatDialogTitle_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵhostProperty("id", ctx.id);
        } }, inputs: { id: "id" }, exportAs: ["matDialogTitle"] });
    /** @nocollapse */
    MatDialogTitle.ctorParameters = () => [
        { type: MatDialogRef, decorators: [{ type: Optional }] },
        { type: ElementRef },
        { type: MatDialog }
    ];
    MatDialogTitle.propDecorators = {
        id: [{ type: Input }]
    };
    /**
     * Scrollable content container of a dialog.
     */
    class MatDialogContent {
    }
    MatDialogContent.ɵfac = function MatDialogContent_Factory(t) { return new (t || MatDialogContent)(); };
    MatDialogContent.ɵdir = ɵɵdefineDirective({ type: MatDialogContent, selectors: [["", "mat-dialog-content", ""], ["mat-dialog-content"], ["", "matDialogContent", ""]], hostAttrs: [1, "mat-dialog-content"] });
    /**
     * Container for the bottom action buttons in a dialog.
     * Stays fixed to the bottom when scrolling.
     */
    class MatDialogActions {
    }
    MatDialogActions.ɵfac = function MatDialogActions_Factory(t) { return new (t || MatDialogActions)(); };
    MatDialogActions.ɵdir = ɵɵdefineDirective({ type: MatDialogActions, selectors: [["", "mat-dialog-actions", ""], ["mat-dialog-actions"], ["", "matDialogActions", ""]], hostAttrs: [1, "mat-dialog-actions"] });
    /**
     * Finds the closest MatDialogRef to an element by looking at the DOM.
     * @param {?} element Element relative to which to look for a dialog.
     * @param {?} openDialogs References to the currently-open dialogs.
     * @return {?}
     */
    function getClosestDialog(element, openDialogs) {
        /** @type {?} */
        let parent = element.nativeElement.parentElement;
        while (parent && !parent.classList.contains('mat-dialog-container')) {
            parent = parent.parentElement;
        }
        return parent ? openDialogs.find((/**
         * @param {?} dialog
         * @return {?}
         */
        dialog => dialog.id === (/** @type {?} */ (parent)).id)) : null;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/dialog/dialog-module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class MatDialogModule {
    }
    MatDialogModule.ɵmod = ɵɵdefineNgModule({ type: MatDialogModule });
    MatDialogModule.ɵinj = ɵɵdefineInjector({ factory: function MatDialogModule_Factory(t) { return new (t || MatDialogModule)(); }, providers: [
            MatDialog,
            MAT_DIALOG_SCROLL_STRATEGY_PROVIDER,
        ], imports: [[
                OverlayModule,
                PortalModule,
                MatCommonModule,
            ],
            MatCommonModule] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(MatDialogModule, { declarations: function () { return [MatDialogContainer,
            MatDialogClose,
            MatDialogTitle,
            MatDialogActions,
            MatDialogContent]; }, imports: function () { return [OverlayModule,
            PortalModule,
            MatCommonModule]; }, exports: function () { return [MatDialogContainer,
            MatDialogClose,
            MatDialogTitle,
            MatDialogContent,
            MatDialogActions,
            MatCommonModule]; } }); })();

    const _c0$6 = ["mat-button", ""];
    const _c1$6 = ["*"];
    const DEFAULT_ROUND_BUTTON_COLOR = 'accent';
    /**
     * List of classes to add to MatButton instances based on host attributes to
     * style as different variants.
     * @type {?}
     */
    const BUTTON_HOST_ATTRIBUTES = [
        'mat-button',
        'mat-flat-button',
        'mat-icon-button',
        'mat-raised-button',
        'mat-stroked-button',
        'mat-mini-fab',
        'mat-fab',
    ];
    // Boilerplate for applying mixins to MatButton.
    /**
     * \@docs-private
     */
    class MatButtonBase {
        /**
         * @param {?} _elementRef
         */
        constructor(_elementRef) {
            this._elementRef = _elementRef;
        }
    }
    /** @type {?} */
    const _MatButtonMixinBase = mixinColor(mixinDisabled(mixinDisableRipple(MatButtonBase)));
    /**
     * Material design button.
     */
    class MatButton extends _MatButtonMixinBase {
        /**
         * @param {?} elementRef
         * @param {?} _focusMonitor
         * @param {?} _animationMode
         */
        constructor(elementRef, _focusMonitor, _animationMode) {
            super(elementRef);
            this._focusMonitor = _focusMonitor;
            this._animationMode = _animationMode;
            /**
             * Whether the button is round.
             */
            this.isRoundButton = this._hasHostAttributes('mat-fab', 'mat-mini-fab');
            /**
             * Whether the button is icon button.
             */
            this.isIconButton = this._hasHostAttributes('mat-icon-button');
            // For each of the variant selectors that is present in the button's host
            // attributes, add the correct corresponding class.
            for (const attr of BUTTON_HOST_ATTRIBUTES) {
                if (this._hasHostAttributes(attr)) {
                    ((/** @type {?} */ (this._getHostElement()))).classList.add(attr);
                }
            }
            // Add a class that applies to all buttons. This makes it easier to target if somebody
            // wants to target all Material buttons. We do it here rather than `host` to ensure that
            // the class is applied to derived classes.
            elementRef.nativeElement.classList.add('mat-button-base');
            this._focusMonitor.monitor(this._elementRef, true);
            if (this.isRoundButton) {
                this.color = DEFAULT_ROUND_BUTTON_COLOR;
            }
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this._focusMonitor.stopMonitoring(this._elementRef);
        }
        /**
         * Focuses the button.
         * @param {?=} origin
         * @param {?=} options
         * @return {?}
         */
        focus(origin = 'program', options) {
            this._focusMonitor.focusVia(this._getHostElement(), origin, options);
        }
        /**
         * @return {?}
         */
        _getHostElement() {
            return this._elementRef.nativeElement;
        }
        /**
         * @return {?}
         */
        _isRippleDisabled() {
            return this.disableRipple || this.disabled;
        }
        /**
         * Gets whether the button has one of the given attributes.
         * @param {...?} attributes
         * @return {?}
         */
        _hasHostAttributes(...attributes) {
            return attributes.some((/**
             * @param {?} attribute
             * @return {?}
             */
            attribute => this._getHostElement().hasAttribute(attribute)));
        }
    }
    MatButton.ɵfac = function MatButton_Factory(t) { return new (t || MatButton)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(FocusMonitor), ɵɵdirectiveInject(ANIMATION_MODULE_TYPE, 8)); };
    MatButton.ɵcmp = ɵɵdefineComponent({ type: MatButton, selectors: [["button", "mat-button", ""], ["button", "mat-raised-button", ""], ["button", "mat-icon-button", ""], ["button", "mat-fab", ""], ["button", "mat-mini-fab", ""], ["button", "mat-stroked-button", ""], ["button", "mat-flat-button", ""]], viewQuery: function MatButton_Query(rf, ctx) { if (rf & 1) {
            ɵɵviewQuery(MatRipple, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.ripple = _t.first);
        } }, hostAttrs: [1, "mat-focus-indicator"], hostVars: 3, hostBindings: function MatButton_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵattribute("disabled", ctx.disabled || null);
            ɵɵclassProp("_mat-animation-noopable", ctx._animationMode === "NoopAnimations");
        } }, inputs: { disabled: "disabled", disableRipple: "disableRipple", color: "color" }, exportAs: ["matButton"], features: [ɵɵInheritDefinitionFeature], attrs: _c0$6, ngContentSelectors: _c1$6, decls: 4, vars: 5, consts: [[1, "mat-button-wrapper"], ["matRipple", "", 1, "mat-button-ripple", 3, "matRippleDisabled", "matRippleCentered", "matRippleTrigger"], [1, "mat-button-focus-overlay"]], template: function MatButton_Template(rf, ctx) { if (rf & 1) {
            ɵɵprojectionDef();
            ɵɵelementStart(0, "span", 0);
            ɵɵprojection(1);
            ɵɵelementEnd();
            ɵɵelement(2, "div", 1);
            ɵɵelement(3, "div", 2);
        } if (rf & 2) {
            ɵɵadvance(2);
            ɵɵclassProp("mat-button-ripple-round", ctx.isRoundButton || ctx.isIconButton);
            ɵɵproperty("matRippleDisabled", ctx._isRippleDisabled())("matRippleCentered", ctx.isIconButton)("matRippleTrigger", ctx._getHostElement());
        } }, directives: [MatRipple], styles: [".mat-button .mat-button-focus-overlay,.mat-icon-button .mat-button-focus-overlay{opacity:0}.mat-button:hover .mat-button-focus-overlay,.mat-stroked-button:hover .mat-button-focus-overlay{opacity:.04}@media(hover: none){.mat-button:hover .mat-button-focus-overlay,.mat-stroked-button:hover .mat-button-focus-overlay{opacity:0}}.mat-button,.mat-icon-button,.mat-stroked-button,.mat-flat-button{box-sizing:border-box;position:relative;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;cursor:pointer;outline:none;border:none;-webkit-tap-highlight-color:transparent;display:inline-block;white-space:nowrap;text-decoration:none;vertical-align:baseline;text-align:center;margin:0;min-width:64px;line-height:36px;padding:0 16px;border-radius:4px;overflow:visible}.mat-button::-moz-focus-inner,.mat-icon-button::-moz-focus-inner,.mat-stroked-button::-moz-focus-inner,.mat-flat-button::-moz-focus-inner{border:0}.mat-button[disabled],.mat-icon-button[disabled],.mat-stroked-button[disabled],.mat-flat-button[disabled]{cursor:default}.mat-button.cdk-keyboard-focused .mat-button-focus-overlay,.mat-button.cdk-program-focused .mat-button-focus-overlay,.mat-icon-button.cdk-keyboard-focused .mat-button-focus-overlay,.mat-icon-button.cdk-program-focused .mat-button-focus-overlay,.mat-stroked-button.cdk-keyboard-focused .mat-button-focus-overlay,.mat-stroked-button.cdk-program-focused .mat-button-focus-overlay,.mat-flat-button.cdk-keyboard-focused .mat-button-focus-overlay,.mat-flat-button.cdk-program-focused .mat-button-focus-overlay{opacity:.12}.mat-button::-moz-focus-inner,.mat-icon-button::-moz-focus-inner,.mat-stroked-button::-moz-focus-inner,.mat-flat-button::-moz-focus-inner{border:0}.mat-raised-button{box-sizing:border-box;position:relative;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;cursor:pointer;outline:none;border:none;-webkit-tap-highlight-color:transparent;display:inline-block;white-space:nowrap;text-decoration:none;vertical-align:baseline;text-align:center;margin:0;min-width:64px;line-height:36px;padding:0 16px;border-radius:4px;overflow:visible;transform:translate3d(0, 0, 0);transition:background 400ms cubic-bezier(0.25, 0.8, 0.25, 1),box-shadow 280ms cubic-bezier(0.4, 0, 0.2, 1)}.mat-raised-button::-moz-focus-inner{border:0}.mat-raised-button[disabled]{cursor:default}.mat-raised-button.cdk-keyboard-focused .mat-button-focus-overlay,.mat-raised-button.cdk-program-focused .mat-button-focus-overlay{opacity:.12}.mat-raised-button::-moz-focus-inner{border:0}._mat-animation-noopable.mat-raised-button{transition:none;animation:none}.mat-stroked-button{border:1px solid currentColor;padding:0 15px;line-height:34px}.mat-stroked-button .mat-button-ripple.mat-ripple,.mat-stroked-button .mat-button-focus-overlay{top:-1px;left:-1px;right:-1px;bottom:-1px}.mat-fab{box-sizing:border-box;position:relative;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;cursor:pointer;outline:none;border:none;-webkit-tap-highlight-color:transparent;display:inline-block;white-space:nowrap;text-decoration:none;vertical-align:baseline;text-align:center;margin:0;min-width:64px;line-height:36px;padding:0 16px;border-radius:4px;overflow:visible;transform:translate3d(0, 0, 0);transition:background 400ms cubic-bezier(0.25, 0.8, 0.25, 1),box-shadow 280ms cubic-bezier(0.4, 0, 0.2, 1);min-width:0;border-radius:50%;width:56px;height:56px;padding:0;flex-shrink:0}.mat-fab::-moz-focus-inner{border:0}.mat-fab[disabled]{cursor:default}.mat-fab.cdk-keyboard-focused .mat-button-focus-overlay,.mat-fab.cdk-program-focused .mat-button-focus-overlay{opacity:.12}.mat-fab::-moz-focus-inner{border:0}._mat-animation-noopable.mat-fab{transition:none;animation:none}.mat-fab .mat-button-wrapper{padding:16px 0;display:inline-block;line-height:24px}.mat-mini-fab{box-sizing:border-box;position:relative;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;cursor:pointer;outline:none;border:none;-webkit-tap-highlight-color:transparent;display:inline-block;white-space:nowrap;text-decoration:none;vertical-align:baseline;text-align:center;margin:0;min-width:64px;line-height:36px;padding:0 16px;border-radius:4px;overflow:visible;transform:translate3d(0, 0, 0);transition:background 400ms cubic-bezier(0.25, 0.8, 0.25, 1),box-shadow 280ms cubic-bezier(0.4, 0, 0.2, 1);min-width:0;border-radius:50%;width:40px;height:40px;padding:0;flex-shrink:0}.mat-mini-fab::-moz-focus-inner{border:0}.mat-mini-fab[disabled]{cursor:default}.mat-mini-fab.cdk-keyboard-focused .mat-button-focus-overlay,.mat-mini-fab.cdk-program-focused .mat-button-focus-overlay{opacity:.12}.mat-mini-fab::-moz-focus-inner{border:0}._mat-animation-noopable.mat-mini-fab{transition:none;animation:none}.mat-mini-fab .mat-button-wrapper{padding:8px 0;display:inline-block;line-height:24px}.mat-icon-button{padding:0;min-width:0;width:40px;height:40px;flex-shrink:0;line-height:40px;border-radius:50%}.mat-icon-button i,.mat-icon-button .mat-icon{line-height:24px}.mat-button-ripple.mat-ripple,.mat-button-focus-overlay{top:0;left:0;right:0;bottom:0;position:absolute;pointer-events:none;border-radius:inherit}.mat-button-ripple.mat-ripple:not(:empty){transform:translateZ(0)}.mat-button-focus-overlay{opacity:0;transition:opacity 200ms cubic-bezier(0.35, 0, 0.25, 1),background-color 200ms cubic-bezier(0.35, 0, 0.25, 1)}._mat-animation-noopable .mat-button-focus-overlay{transition:none}.cdk-high-contrast-active .mat-button-focus-overlay{background-color:#fff}.cdk-high-contrast-black-on-white .mat-button-focus-overlay{background-color:#000}.mat-button-ripple-round{border-radius:50%;z-index:1}.mat-button .mat-button-wrapper>*,.mat-flat-button .mat-button-wrapper>*,.mat-stroked-button .mat-button-wrapper>*,.mat-raised-button .mat-button-wrapper>*,.mat-icon-button .mat-button-wrapper>*,.mat-fab .mat-button-wrapper>*,.mat-mini-fab .mat-button-wrapper>*{vertical-align:middle}.mat-form-field:not(.mat-form-field-appearance-legacy) .mat-form-field-prefix .mat-icon-button,.mat-form-field:not(.mat-form-field-appearance-legacy) .mat-form-field-suffix .mat-icon-button{display:block;font-size:inherit;width:2.5em;height:2.5em}.cdk-high-contrast-active .mat-button,.cdk-high-contrast-active .mat-flat-button,.cdk-high-contrast-active .mat-raised-button,.cdk-high-contrast-active .mat-icon-button,.cdk-high-contrast-active .mat-fab,.cdk-high-contrast-active .mat-mini-fab{outline:solid 1px}\n"], encapsulation: 2, changeDetection: 0 });
    /** @nocollapse */
    MatButton.ctorParameters = () => [
        { type: ElementRef },
        { type: FocusMonitor },
        { type: String, decorators: [{ type: Optional }, { type: Inject, args: [ANIMATION_MODULE_TYPE,] }] }
    ];
    MatButton.propDecorators = {
        ripple: [{ type: ViewChild, args: [MatRipple,] }]
    };
    /**
     * Material design anchor button.
     */
    class MatAnchor extends MatButton {
        /**
         * @param {?} focusMonitor
         * @param {?} elementRef
         * @param {?} animationMode
         */
        constructor(focusMonitor, elementRef, animationMode) {
            super(elementRef, focusMonitor, animationMode);
        }
        /**
         * @param {?} event
         * @return {?}
         */
        _haltDisabledEvents(event) {
            // A disabled button shouldn't apply any actions
            if (this.disabled) {
                event.preventDefault();
                event.stopImmediatePropagation();
            }
        }
    }
    MatAnchor.ɵfac = function MatAnchor_Factory(t) { return new (t || MatAnchor)(ɵɵdirectiveInject(FocusMonitor), ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(ANIMATION_MODULE_TYPE, 8)); };
    MatAnchor.ɵcmp = ɵɵdefineComponent({ type: MatAnchor, selectors: [["a", "mat-button", ""], ["a", "mat-raised-button", ""], ["a", "mat-icon-button", ""], ["a", "mat-fab", ""], ["a", "mat-mini-fab", ""], ["a", "mat-stroked-button", ""], ["a", "mat-flat-button", ""]], hostAttrs: [1, "mat-focus-indicator"], hostVars: 5, hostBindings: function MatAnchor_HostBindings(rf, ctx) { if (rf & 1) {
            ɵɵlistener("click", function MatAnchor_click_HostBindingHandler($event) { return ctx._haltDisabledEvents($event); });
        } if (rf & 2) {
            ɵɵattribute("tabindex", ctx.disabled ? 0 - 1 : ctx.tabIndex || 0)("disabled", ctx.disabled || null)("aria-disabled", ctx.disabled.toString());
            ɵɵclassProp("_mat-animation-noopable", ctx._animationMode === "NoopAnimations");
        } }, inputs: { disabled: "disabled", disableRipple: "disableRipple", color: "color", tabIndex: "tabIndex" }, exportAs: ["matButton", "matAnchor"], features: [ɵɵInheritDefinitionFeature], attrs: _c0$6, ngContentSelectors: _c1$6, decls: 4, vars: 5, consts: [[1, "mat-button-wrapper"], ["matRipple", "", 1, "mat-button-ripple", 3, "matRippleDisabled", "matRippleCentered", "matRippleTrigger"], [1, "mat-button-focus-overlay"]], template: function MatAnchor_Template(rf, ctx) { if (rf & 1) {
            ɵɵprojectionDef();
            ɵɵelementStart(0, "span", 0);
            ɵɵprojection(1);
            ɵɵelementEnd();
            ɵɵelement(2, "div", 1);
            ɵɵelement(3, "div", 2);
        } if (rf & 2) {
            ɵɵadvance(2);
            ɵɵclassProp("mat-button-ripple-round", ctx.isRoundButton || ctx.isIconButton);
            ɵɵproperty("matRippleDisabled", ctx._isRippleDisabled())("matRippleCentered", ctx.isIconButton)("matRippleTrigger", ctx._getHostElement());
        } }, directives: [MatRipple], styles: [".mat-button .mat-button-focus-overlay,.mat-icon-button .mat-button-focus-overlay{opacity:0}.mat-button:hover .mat-button-focus-overlay,.mat-stroked-button:hover .mat-button-focus-overlay{opacity:.04}@media(hover: none){.mat-button:hover .mat-button-focus-overlay,.mat-stroked-button:hover .mat-button-focus-overlay{opacity:0}}.mat-button,.mat-icon-button,.mat-stroked-button,.mat-flat-button{box-sizing:border-box;position:relative;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;cursor:pointer;outline:none;border:none;-webkit-tap-highlight-color:transparent;display:inline-block;white-space:nowrap;text-decoration:none;vertical-align:baseline;text-align:center;margin:0;min-width:64px;line-height:36px;padding:0 16px;border-radius:4px;overflow:visible}.mat-button::-moz-focus-inner,.mat-icon-button::-moz-focus-inner,.mat-stroked-button::-moz-focus-inner,.mat-flat-button::-moz-focus-inner{border:0}.mat-button[disabled],.mat-icon-button[disabled],.mat-stroked-button[disabled],.mat-flat-button[disabled]{cursor:default}.mat-button.cdk-keyboard-focused .mat-button-focus-overlay,.mat-button.cdk-program-focused .mat-button-focus-overlay,.mat-icon-button.cdk-keyboard-focused .mat-button-focus-overlay,.mat-icon-button.cdk-program-focused .mat-button-focus-overlay,.mat-stroked-button.cdk-keyboard-focused .mat-button-focus-overlay,.mat-stroked-button.cdk-program-focused .mat-button-focus-overlay,.mat-flat-button.cdk-keyboard-focused .mat-button-focus-overlay,.mat-flat-button.cdk-program-focused .mat-button-focus-overlay{opacity:.12}.mat-button::-moz-focus-inner,.mat-icon-button::-moz-focus-inner,.mat-stroked-button::-moz-focus-inner,.mat-flat-button::-moz-focus-inner{border:0}.mat-raised-button{box-sizing:border-box;position:relative;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;cursor:pointer;outline:none;border:none;-webkit-tap-highlight-color:transparent;display:inline-block;white-space:nowrap;text-decoration:none;vertical-align:baseline;text-align:center;margin:0;min-width:64px;line-height:36px;padding:0 16px;border-radius:4px;overflow:visible;transform:translate3d(0, 0, 0);transition:background 400ms cubic-bezier(0.25, 0.8, 0.25, 1),box-shadow 280ms cubic-bezier(0.4, 0, 0.2, 1)}.mat-raised-button::-moz-focus-inner{border:0}.mat-raised-button[disabled]{cursor:default}.mat-raised-button.cdk-keyboard-focused .mat-button-focus-overlay,.mat-raised-button.cdk-program-focused .mat-button-focus-overlay{opacity:.12}.mat-raised-button::-moz-focus-inner{border:0}._mat-animation-noopable.mat-raised-button{transition:none;animation:none}.mat-stroked-button{border:1px solid currentColor;padding:0 15px;line-height:34px}.mat-stroked-button .mat-button-ripple.mat-ripple,.mat-stroked-button .mat-button-focus-overlay{top:-1px;left:-1px;right:-1px;bottom:-1px}.mat-fab{box-sizing:border-box;position:relative;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;cursor:pointer;outline:none;border:none;-webkit-tap-highlight-color:transparent;display:inline-block;white-space:nowrap;text-decoration:none;vertical-align:baseline;text-align:center;margin:0;min-width:64px;line-height:36px;padding:0 16px;border-radius:4px;overflow:visible;transform:translate3d(0, 0, 0);transition:background 400ms cubic-bezier(0.25, 0.8, 0.25, 1),box-shadow 280ms cubic-bezier(0.4, 0, 0.2, 1);min-width:0;border-radius:50%;width:56px;height:56px;padding:0;flex-shrink:0}.mat-fab::-moz-focus-inner{border:0}.mat-fab[disabled]{cursor:default}.mat-fab.cdk-keyboard-focused .mat-button-focus-overlay,.mat-fab.cdk-program-focused .mat-button-focus-overlay{opacity:.12}.mat-fab::-moz-focus-inner{border:0}._mat-animation-noopable.mat-fab{transition:none;animation:none}.mat-fab .mat-button-wrapper{padding:16px 0;display:inline-block;line-height:24px}.mat-mini-fab{box-sizing:border-box;position:relative;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;cursor:pointer;outline:none;border:none;-webkit-tap-highlight-color:transparent;display:inline-block;white-space:nowrap;text-decoration:none;vertical-align:baseline;text-align:center;margin:0;min-width:64px;line-height:36px;padding:0 16px;border-radius:4px;overflow:visible;transform:translate3d(0, 0, 0);transition:background 400ms cubic-bezier(0.25, 0.8, 0.25, 1),box-shadow 280ms cubic-bezier(0.4, 0, 0.2, 1);min-width:0;border-radius:50%;width:40px;height:40px;padding:0;flex-shrink:0}.mat-mini-fab::-moz-focus-inner{border:0}.mat-mini-fab[disabled]{cursor:default}.mat-mini-fab.cdk-keyboard-focused .mat-button-focus-overlay,.mat-mini-fab.cdk-program-focused .mat-button-focus-overlay{opacity:.12}.mat-mini-fab::-moz-focus-inner{border:0}._mat-animation-noopable.mat-mini-fab{transition:none;animation:none}.mat-mini-fab .mat-button-wrapper{padding:8px 0;display:inline-block;line-height:24px}.mat-icon-button{padding:0;min-width:0;width:40px;height:40px;flex-shrink:0;line-height:40px;border-radius:50%}.mat-icon-button i,.mat-icon-button .mat-icon{line-height:24px}.mat-button-ripple.mat-ripple,.mat-button-focus-overlay{top:0;left:0;right:0;bottom:0;position:absolute;pointer-events:none;border-radius:inherit}.mat-button-ripple.mat-ripple:not(:empty){transform:translateZ(0)}.mat-button-focus-overlay{opacity:0;transition:opacity 200ms cubic-bezier(0.35, 0, 0.25, 1),background-color 200ms cubic-bezier(0.35, 0, 0.25, 1)}._mat-animation-noopable .mat-button-focus-overlay{transition:none}.cdk-high-contrast-active .mat-button-focus-overlay{background-color:#fff}.cdk-high-contrast-black-on-white .mat-button-focus-overlay{background-color:#000}.mat-button-ripple-round{border-radius:50%;z-index:1}.mat-button .mat-button-wrapper>*,.mat-flat-button .mat-button-wrapper>*,.mat-stroked-button .mat-button-wrapper>*,.mat-raised-button .mat-button-wrapper>*,.mat-icon-button .mat-button-wrapper>*,.mat-fab .mat-button-wrapper>*,.mat-mini-fab .mat-button-wrapper>*{vertical-align:middle}.mat-form-field:not(.mat-form-field-appearance-legacy) .mat-form-field-prefix .mat-icon-button,.mat-form-field:not(.mat-form-field-appearance-legacy) .mat-form-field-suffix .mat-icon-button{display:block;font-size:inherit;width:2.5em;height:2.5em}.cdk-high-contrast-active .mat-button,.cdk-high-contrast-active .mat-flat-button,.cdk-high-contrast-active .mat-raised-button,.cdk-high-contrast-active .mat-icon-button,.cdk-high-contrast-active .mat-fab,.cdk-high-contrast-active .mat-mini-fab{outline:solid 1px}\n"], encapsulation: 2, changeDetection: 0 });
    /** @nocollapse */
    MatAnchor.ctorParameters = () => [
        { type: FocusMonitor },
        { type: ElementRef },
        { type: String, decorators: [{ type: Optional }, { type: Inject, args: [ANIMATION_MODULE_TYPE,] }] }
    ];
    MatAnchor.propDecorators = {
        tabIndex: [{ type: Input }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/button/button-module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class MatButtonModule {
    }
    MatButtonModule.ɵmod = ɵɵdefineNgModule({ type: MatButtonModule });
    MatButtonModule.ɵinj = ɵɵdefineInjector({ factory: function MatButtonModule_Factory(t) { return new (t || MatButtonModule)(); }, imports: [[
                MatRippleModule,
                MatCommonModule,
            ],
            MatCommonModule] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(MatButtonModule, { declarations: function () { return [MatButton,
            MatAnchor]; }, imports: function () { return [MatRippleModule,
            MatCommonModule]; }, exports: function () { return [MatButton,
            MatAnchor,
            MatCommonModule]; } }); })();

    class PendingCopy {
        /**
         * @param {?} text
         * @param {?} _document
         */
        constructor(text, _document) {
            this._document = _document;
            /** @type {?} */
            const textarea = this._textarea = this._document.createElement('textarea');
            /** @type {?} */
            const styles = textarea.style;
            // Hide the element for display and accessibility. Set an
            // absolute position so the page layout isn't affected.
            styles.opacity = '0';
            styles.position = 'absolute';
            styles.left = styles.top = '-999em';
            textarea.setAttribute('aria-hidden', 'true');
            textarea.value = text;
            this._document.body.appendChild(textarea);
        }
        /**
         * Finishes copying the text.
         * @return {?}
         */
        copy() {
            /** @type {?} */
            const textarea = this._textarea;
            /** @type {?} */
            let successful = false;
            try { // Older browsers could throw if copy is not supported.
                if (textarea) {
                    /** @type {?} */
                    const currentFocus = (/** @type {?} */ (this._document.activeElement));
                    textarea.select();
                    textarea.setSelectionRange(0, textarea.value.length);
                    successful = this._document.execCommand('copy');
                    if (currentFocus) {
                        currentFocus.focus();
                    }
                }
            }
            catch (_a) {
                // Discard error.
                // Initial setting of {@code successful} will represent failure here.
            }
            return successful;
        }
        /**
         * Cleans up DOM changes used to perform the copy operation.
         * @return {?}
         */
        destroy() {
            /** @type {?} */
            const textarea = this._textarea;
            if (textarea) {
                if (textarea.parentNode) {
                    textarea.parentNode.removeChild(textarea);
                }
                this._textarea = undefined;
            }
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/clipboard/clipboard.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * A service for copying text to the clipboard.
     */
    class Clipboard {
        /**
         * @param {?} document
         */
        constructor(document) {
            this._document = document;
        }
        /**
         * Copies the provided text into the user's clipboard.
         *
         * @param {?} text The string to copy.
         * @return {?} Whether the operation was successful.
         */
        copy(text) {
            /** @type {?} */
            const pendingCopy = this.beginCopy(text);
            /** @type {?} */
            const successful = pendingCopy.copy();
            pendingCopy.destroy();
            return successful;
        }
        /**
         * Prepares a string to be copied later. This is useful for large strings
         * which take too long to successfully render and be copied in the same tick.
         *
         * The caller must call `destroy` on the returned `PendingCopy`.
         *
         * @param {?} text The string to copy.
         * @return {?} the pending copy operation.
         */
        beginCopy(text) {
            return new PendingCopy(text, this._document);
        }
    }
    Clipboard.ɵfac = function Clipboard_Factory(t) { return new (t || Clipboard)(ɵɵinject(DOCUMENT$1)); };
    /** @nocollapse */
    Clipboard.ctorParameters = () => [
        { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT$1,] }] }
    ];
    /** @nocollapse */ Clipboard.ɵprov = ɵɵdefineInjectable({ factory: function Clipboard_Factory() { return new Clipboard(ɵɵinject(DOCUMENT$1)); }, token: Clipboard, providedIn: "root" });
    /**
     * Injection token that can be used to provide the default options to `CdkCopyToClipboard`.
     * @type {?}
     */
    const CKD_COPY_TO_CLIPBOARD_CONFIG = new InjectionToken('CKD_COPY_TO_CLIPBOARD_CONFIG');
    /**
     * Provides behavior for a button that when clicked copies content into user's
     * clipboard.
     */
    class CdkCopyToClipboard {
        /**
         * @param {?} _clipboard
         * @param {?=} _ngZone
         * @param {?=} config
         */
        constructor(_clipboard, _ngZone, config) {
            this._clipboard = _clipboard;
            this._ngZone = _ngZone;
            /**
             * Content to be copied.
             */
            this.text = '';
            /**
             * How many times to attempt to copy the text. This may be necessary for longer text, because
             * the browser needs time to fill an intermediate textarea element and copy the content.
             */
            this.attempts = 1;
            /**
             * Emits when some text is copied to the clipboard. The
             * emitted value indicates whether copying was successful.
             */
            this.copied = new EventEmitter();
            /**
             * Emits when some text is copied to the clipboard. The
             * emitted value indicates whether copying was successful.
             * @deprecated Use `cdkCopyToClipboardCopied` instead.
             * \@breaking-change 10.0.0
             */
            this._deprecatedCopied = this.copied;
            /**
             * Copies that are currently being attempted.
             */
            this._pending = new Set();
            if (config && config.attempts != null) {
                this.attempts = config.attempts;
            }
        }
        /**
         * Copies the current text to the clipboard.
         * @param {?=} attempts
         * @return {?}
         */
        copy(attempts = this.attempts) {
            if (attempts > 1) {
                /** @type {?} */
                let remainingAttempts = attempts;
                /** @type {?} */
                const pending = this._clipboard.beginCopy(this.text);
                this._pending.add(pending);
                /** @type {?} */
                const attempt = (/**
                 * @return {?}
                 */
                () => {
                    /** @type {?} */
                    const successful = pending.copy();
                    if (!successful && --remainingAttempts && !this._destroyed) {
                        // @breaking-change 10.0.0 Remove null check for `_ngZone`.
                        if (this._ngZone) {
                            this._currentTimeout = this._ngZone.runOutsideAngular((/**
                             * @return {?}
                             */
                            () => setTimeout(attempt, 1)));
                        }
                        else {
                            // We use 1 for the timeout since it's more predictable when flushing in unit tests.
                            this._currentTimeout = setTimeout(attempt, 1);
                        }
                    }
                    else {
                        this._currentTimeout = null;
                        this._pending.delete(pending);
                        pending.destroy();
                        this.copied.emit(successful);
                    }
                });
                attempt();
            }
            else {
                this.copied.emit(this._clipboard.copy(this.text));
            }
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            if (this._currentTimeout) {
                clearTimeout(this._currentTimeout);
            }
            this._pending.forEach((/**
             * @param {?} copy
             * @return {?}
             */
            copy => copy.destroy()));
            this._pending.clear();
            this._destroyed = true;
        }
    }
    CdkCopyToClipboard.ɵfac = function CdkCopyToClipboard_Factory(t) { return new (t || CdkCopyToClipboard)(ɵɵdirectiveInject(Clipboard), ɵɵdirectiveInject(NgZone), ɵɵdirectiveInject(CKD_COPY_TO_CLIPBOARD_CONFIG, 8)); };
    CdkCopyToClipboard.ɵdir = ɵɵdefineDirective({ type: CdkCopyToClipboard, selectors: [["", "cdkCopyToClipboard", ""]], hostBindings: function CdkCopyToClipboard_HostBindings(rf, ctx) { if (rf & 1) {
            ɵɵlistener("click", function CdkCopyToClipboard_click_HostBindingHandler() { return ctx.copy(); });
        } }, inputs: { text: ["cdkCopyToClipboard", "text"], attempts: ["cdkCopyToClipboardAttempts", "attempts"] }, outputs: { copied: "cdkCopyToClipboardCopied", _deprecatedCopied: "copied" } });
    /** @nocollapse */
    CdkCopyToClipboard.ctorParameters = () => [
        { type: Clipboard },
        { type: NgZone },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [CKD_COPY_TO_CLIPBOARD_CONFIG,] }] }
    ];
    CdkCopyToClipboard.propDecorators = {
        text: [{ type: Input, args: ['cdkCopyToClipboard',] }],
        attempts: [{ type: Input, args: ['cdkCopyToClipboardAttempts',] }],
        copied: [{ type: Output, args: ['cdkCopyToClipboardCopied',] }],
        _deprecatedCopied: [{ type: Output, args: ['copied',] }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/clipboard/clipboard-module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class ClipboardModule {
    }
    ClipboardModule.ɵmod = ɵɵdefineNgModule({ type: ClipboardModule });
    ClipboardModule.ɵinj = ɵɵdefineInjector({ factory: function ClipboardModule_Factory(t) { return new (t || ClipboardModule)(); } });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(ClipboardModule, { declarations: [CdkCopyToClipboard], exports: [CdkCopyToClipboard] }); })();

    /**
     * @license Angular v9.1.12
     * (c) 2010-2020 Google LLC. https://angular.io/
     * License: MIT
     */
    class HttpHandler {
    }
    /**
     * A final `HttpHandler` which will dispatch the request via browser HTTP APIs to a backend.
     *
     * Interceptors sit between the `HttpClient` interface and the `HttpBackend`.
     *
     * When injected, `HttpBackend` dispatches requests directly to the backend, without going
     * through the interceptor chain.
     *
     * \@publicApi
     * @abstract
     */
    class HttpBackend {
    }
    /**
     * Represents the header configuration options for an HTTP request.
     * Instances are immutable. Modifying methods return a cloned
     * instance with the change. The original object is never changed.
     *
     * \@publicApi
     */
    class HttpHeaders {
        /**
         * Constructs a new HTTP header object with the given values.
         * @param {?=} headers
         */
        constructor(headers) {
            /**
             * Internal map of lowercased header names to the normalized
             * form of the name (the form seen first).
             */
            this.normalizedNames = new Map();
            /**
             * Queued updates to be materialized the next initialization.
             */
            this.lazyUpdate = null;
            if (!headers) {
                this.headers = new Map();
            }
            else if (typeof headers === 'string') {
                this.lazyInit = (/**
                 * @return {?}
                 */
                () => {
                    this.headers = new Map();
                    headers.split('\n').forEach((/**
                     * @param {?} line
                     * @return {?}
                     */
                    line => {
                        /** @type {?} */
                        const index = line.indexOf(':');
                        if (index > 0) {
                            /** @type {?} */
                            const name = line.slice(0, index);
                            /** @type {?} */
                            const key = name.toLowerCase();
                            /** @type {?} */
                            const value = line.slice(index + 1).trim();
                            this.maybeSetNormalizedName(name, key);
                            if (this.headers.has(key)) {
                                (/** @type {?} */ (this.headers.get(key))).push(value);
                            }
                            else {
                                this.headers.set(key, [value]);
                            }
                        }
                    }));
                });
            }
            else {
                this.lazyInit = (/**
                 * @return {?}
                 */
                () => {
                    this.headers = new Map();
                    Object.keys(headers).forEach((/**
                     * @param {?} name
                     * @return {?}
                     */
                    name => {
                        /** @type {?} */
                        let values = headers[name];
                        /** @type {?} */
                        const key = name.toLowerCase();
                        if (typeof values === 'string') {
                            values = [values];
                        }
                        if (values.length > 0) {
                            this.headers.set(key, values);
                            this.maybeSetNormalizedName(name, key);
                        }
                    }));
                });
            }
        }
        /**
         * Checks for existence of a given header.
         *
         * @param {?} name The header name to check for existence.
         *
         * @return {?} True if the header exists, false otherwise.
         */
        has(name) {
            this.init();
            return this.headers.has(name.toLowerCase());
        }
        /**
         * Retrieves the first value of a given header.
         *
         * @param {?} name The header name.
         *
         * @return {?} The value string if the header exists, null otherwise
         */
        get(name) {
            this.init();
            /** @type {?} */
            const values = this.headers.get(name.toLowerCase());
            return values && values.length > 0 ? values[0] : null;
        }
        /**
         * Retrieves the names of the headers.
         *
         * @return {?} A list of header names.
         */
        keys() {
            this.init();
            return Array.from(this.normalizedNames.values());
        }
        /**
         * Retrieves a list of values for a given header.
         *
         * @param {?} name The header name from which to retrieve values.
         *
         * @return {?} A string of values if the header exists, null otherwise.
         */
        getAll(name) {
            this.init();
            return this.headers.get(name.toLowerCase()) || null;
        }
        /**
         * Appends a new value to the existing set of values for a header
         * and returns them in a clone of the original instance.
         *
         * @param {?} name The header name for which to append the values.
         * @param {?} value The value to append.
         *
         * @return {?} A clone of the HTTP headers object with the value appended to the given header.
         */
        append(name, value) {
            return this.clone({ name, value, op: 'a' });
        }
        /**
         * Sets or modifies a value for a given header in a clone of the original instance.
         * If the header already exists, its value is replaced with the given value
         * in the returned object.
         *
         * @param {?} name The header name.
         * @param {?} value The value or values to set or overide for the given header.
         *
         * @return {?} A clone of the HTTP headers object with the newly set header value.
         */
        set(name, value) {
            return this.clone({ name, value, op: 's' });
        }
        /**
         * Deletes values for a given header in a clone of the original instance.
         *
         * @param {?} name The header name.
         * @param {?=} value The value or values to delete for the given header.
         *
         * @return {?} A clone of the HTTP headers object with the given value deleted.
         */
        delete(name, value) {
            return this.clone({ name, value, op: 'd' });
        }
        /**
         * @private
         * @param {?} name
         * @param {?} lcName
         * @return {?}
         */
        maybeSetNormalizedName(name, lcName) {
            if (!this.normalizedNames.has(lcName)) {
                this.normalizedNames.set(lcName, name);
            }
        }
        /**
         * @private
         * @return {?}
         */
        init() {
            if (!!this.lazyInit) {
                if (this.lazyInit instanceof HttpHeaders) {
                    this.copyFrom(this.lazyInit);
                }
                else {
                    this.lazyInit();
                }
                this.lazyInit = null;
                if (!!this.lazyUpdate) {
                    this.lazyUpdate.forEach((/**
                     * @param {?} update
                     * @return {?}
                     */
                    update => this.applyUpdate(update)));
                    this.lazyUpdate = null;
                }
            }
        }
        /**
         * @private
         * @param {?} other
         * @return {?}
         */
        copyFrom(other) {
            other.init();
            Array.from(other.headers.keys()).forEach((/**
             * @param {?} key
             * @return {?}
             */
            key => {
                this.headers.set(key, (/** @type {?} */ (other.headers.get(key))));
                this.normalizedNames.set(key, (/** @type {?} */ (other.normalizedNames.get(key))));
            }));
        }
        /**
         * @private
         * @param {?} update
         * @return {?}
         */
        clone(update) {
            /** @type {?} */
            const clone = new HttpHeaders();
            clone.lazyInit =
                (!!this.lazyInit && this.lazyInit instanceof HttpHeaders) ? this.lazyInit : this;
            clone.lazyUpdate = (this.lazyUpdate || []).concat([update]);
            return clone;
        }
        /**
         * @private
         * @param {?} update
         * @return {?}
         */
        applyUpdate(update) {
            /** @type {?} */
            const key = update.name.toLowerCase();
            switch (update.op) {
                case 'a':
                case 's':
                    /** @type {?} */
                    let value = (/** @type {?} */ (update.value));
                    if (typeof value === 'string') {
                        value = [value];
                    }
                    if (value.length === 0) {
                        return;
                    }
                    this.maybeSetNormalizedName(update.name, key);
                    /** @type {?} */
                    const base = (update.op === 'a' ? this.headers.get(key) : undefined) || [];
                    base.push(...value);
                    this.headers.set(key, base);
                    break;
                case 'd':
                    /** @type {?} */
                    const toDelete = (/** @type {?} */ (update.value));
                    if (!toDelete) {
                        this.headers.delete(key);
                        this.normalizedNames.delete(key);
                    }
                    else {
                        /** @type {?} */
                        let existing = this.headers.get(key);
                        if (!existing) {
                            return;
                        }
                        existing = existing.filter((/**
                         * @param {?} value
                         * @return {?}
                         */
                        value => toDelete.indexOf(value) === -1));
                        if (existing.length === 0) {
                            this.headers.delete(key);
                            this.normalizedNames.delete(key);
                        }
                        else {
                            this.headers.set(key, existing);
                        }
                    }
                    break;
            }
        }
        /**
         * \@internal
         * @param {?} fn
         * @return {?}
         */
        forEach(fn) {
            this.init();
            Array.from(this.normalizedNames.keys())
                .forEach((/**
             * @param {?} key
             * @return {?}
             */
            key => fn((/** @type {?} */ (this.normalizedNames.get(key))), (/** @type {?} */ (this.headers.get(key))))));
        }
    }
    /**
     * Provides encoding and decoding of URL parameter and query-string values.
     *
     * Serializes and parses URL parameter keys and values to encode and decode them.
     * If you pass URL query parameters without encoding,
     * the query parameters can be misinterpreted at the receiving end.
     *
     *
     * \@publicApi
     */
    class HttpUrlEncodingCodec {
        /**
         * Encodes a key name for a URL parameter or query-string.
         * @param {?} key The key name.
         * @return {?} The encoded key name.
         */
        encodeKey(key) {
            return standardEncoding(key);
        }
        /**
         * Encodes the value of a URL parameter or query-string.
         * @param {?} value The value.
         * @return {?} The encoded value.
         */
        encodeValue(value) {
            return standardEncoding(value);
        }
        /**
         * Decodes an encoded URL parameter or query-string key.
         * @param {?} key The encoded key name.
         * @return {?} The decoded key name.
         */
        decodeKey(key) {
            return decodeURIComponent(key);
        }
        /**
         * Decodes an encoded URL parameter or query-string value.
         * @param {?} value The encoded value.
         * @return {?} The decoded value.
         */
        decodeValue(value) {
            return decodeURIComponent(value);
        }
    }
    /**
     * @param {?} rawParams
     * @param {?} codec
     * @return {?}
     */
    function paramParser(rawParams, codec) {
        /** @type {?} */
        const map = new Map();
        if (rawParams.length > 0) {
            /** @type {?} */
            const params = rawParams.split('&');
            params.forEach((/**
             * @param {?} param
             * @return {?}
             */
            (param) => {
                /** @type {?} */
                const eqIdx = param.indexOf('=');
                const [key, val] = eqIdx == -1 ?
                    [codec.decodeKey(param), ''] :
                    [codec.decodeKey(param.slice(0, eqIdx)), codec.decodeValue(param.slice(eqIdx + 1))];
                /** @type {?} */
                const list = map.get(key) || [];
                list.push(val);
                map.set(key, list);
            }));
        }
        return map;
    }
    /**
     * @param {?} v
     * @return {?}
     */
    function standardEncoding(v) {
        return encodeURIComponent(v)
            .replace(/%40/gi, '@')
            .replace(/%3A/gi, ':')
            .replace(/%24/gi, '$')
            .replace(/%2C/gi, ',')
            .replace(/%3B/gi, ';')
            .replace(/%2B/gi, '+')
            .replace(/%3D/gi, '=')
            .replace(/%3F/gi, '?')
            .replace(/%2F/gi, '/');
    }
    /**
     * An HTTP request/response body that represents serialized parameters,
     * per the MIME type `application/x-www-form-urlencoded`.
     *
     * This class is immutable; all mutation operations return a new instance.
     *
     * \@publicApi
     */
    class HttpParams {
        /**
         * @param {?=} options
         */
        constructor(options = (/** @type {?} */ ({}))) {
            this.updates = null;
            this.cloneFrom = null;
            this.encoder = options.encoder || new HttpUrlEncodingCodec();
            if (!!options.fromString) {
                if (!!options.fromObject) {
                    throw new Error(`Cannot specify both fromString and fromObject.`);
                }
                this.map = paramParser(options.fromString, this.encoder);
            }
            else if (!!options.fromObject) {
                this.map = new Map();
                Object.keys(options.fromObject).forEach((/**
                 * @param {?} key
                 * @return {?}
                 */
                key => {
                    /** @type {?} */
                    const value = ((/** @type {?} */ (options.fromObject)))[key];
                    (/** @type {?} */ (this.map)).set(key, Array.isArray(value) ? value : [value]);
                }));
            }
            else {
                this.map = null;
            }
        }
        /**
         * Reports whether the body includes one or more values for a given parameter.
         * @param {?} param The parameter name.
         * @return {?} True if the parameter has one or more values,
         * false if it has no value or is not present.
         */
        has(param) {
            this.init();
            return (/** @type {?} */ (this.map)).has(param);
        }
        /**
         * Retrieves the first value for a parameter.
         * @param {?} param The parameter name.
         * @return {?} The first value of the given parameter,
         * or `null` if the parameter is not present.
         */
        get(param) {
            this.init();
            /** @type {?} */
            const res = (/** @type {?} */ (this.map)).get(param);
            return !!res ? res[0] : null;
        }
        /**
         * Retrieves all values for a  parameter.
         * @param {?} param The parameter name.
         * @return {?} All values in a string array,
         * or `null` if the parameter not present.
         */
        getAll(param) {
            this.init();
            return (/** @type {?} */ (this.map)).get(param) || null;
        }
        /**
         * Retrieves all the parameters for this body.
         * @return {?} The parameter names in a string array.
         */
        keys() {
            this.init();
            return Array.from((/** @type {?} */ (this.map)).keys());
        }
        /**
         * Appends a new value to existing values for a parameter.
         * @param {?} param The parameter name.
         * @param {?} value The new value to add.
         * @return {?} A new body with the appended value.
         */
        append(param, value) {
            return this.clone({ param, value, op: 'a' });
        }
        /**
         * Replaces the value for a parameter.
         * @param {?} param The parameter name.
         * @param {?} value The new value.
         * @return {?} A new body with the new value.
         */
        set(param, value) {
            return this.clone({ param, value, op: 's' });
        }
        /**
         * Removes a given value or all values from a parameter.
         * @param {?} param The parameter name.
         * @param {?=} value The value to remove, if provided.
         * @return {?} A new body with the given value removed, or with all values
         * removed if no value is specified.
         */
        delete(param, value) {
            return this.clone({ param, value, op: 'd' });
        }
        /**
         * Serializes the body to an encoded string, where key-value pairs (separated by `=`) are
         * separated by `&`s.
         * @return {?}
         */
        toString() {
            this.init();
            return this.keys()
                .map((/**
             * @param {?} key
             * @return {?}
             */
            key => {
                /** @type {?} */
                const eKey = this.encoder.encodeKey(key);
                // `a: ['1']` produces `'a=1'`
                // `b: []` produces `''`
                // `c: ['1', '2']` produces `'c=1&c=2'`
                return (/** @type {?} */ ((/** @type {?} */ (this.map)).get(key))).map((/**
                 * @param {?} value
                 * @return {?}
                 */
                value => eKey + '=' + this.encoder.encodeValue(value)))
                    .join('&');
            }))
                // filter out empty values because `b: []` produces `''`
                // which results in `a=1&&c=1&c=2` instead of `a=1&c=1&c=2` if we don't
                .filter((/**
             * @param {?} param
             * @return {?}
             */
            param => param !== ''))
                .join('&');
        }
        /**
         * @private
         * @param {?} update
         * @return {?}
         */
        clone(update) {
            /** @type {?} */
            const clone = new HttpParams((/** @type {?} */ ({ encoder: this.encoder })));
            clone.cloneFrom = this.cloneFrom || this;
            clone.updates = (this.updates || []).concat([update]);
            return clone;
        }
        /**
         * @private
         * @return {?}
         */
        init() {
            if (this.map === null) {
                this.map = new Map();
            }
            if (this.cloneFrom !== null) {
                this.cloneFrom.init();
                this.cloneFrom.keys().forEach((/**
                 * @param {?} key
                 * @return {?}
                 */
                key => (/** @type {?} */ (this.map)).set(key, (/** @type {?} */ ((/** @type {?} */ ((/** @type {?} */ (this.cloneFrom)).map)).get(key))))));
                (/** @type {?} */ (this.updates)).forEach((/**
                 * @param {?} update
                 * @return {?}
                 */
                update => {
                    switch (update.op) {
                        case 'a':
                        case 's':
                            /** @type {?} */
                            const base = (update.op === 'a' ? (/** @type {?} */ (this.map)).get(update.param) : undefined) || [];
                            base.push((/** @type {?} */ (update.value)));
                            (/** @type {?} */ (this.map)).set(update.param, base);
                            break;
                        case 'd':
                            if (update.value !== undefined) {
                                /** @type {?} */
                                let base = (/** @type {?} */ (this.map)).get(update.param) || [];
                                /** @type {?} */
                                const idx = base.indexOf(update.value);
                                if (idx !== -1) {
                                    base.splice(idx, 1);
                                }
                                if (base.length > 0) {
                                    (/** @type {?} */ (this.map)).set(update.param, base);
                                }
                                else {
                                    (/** @type {?} */ (this.map)).delete(update.param);
                                }
                            }
                            else {
                                (/** @type {?} */ (this.map)).delete(update.param);
                                break;
                            }
                    }
                }));
                this.cloneFrom = this.updates = null;
            }
        }
    }
    /**
     * Determine whether the given HTTP method may include a body.
     * @param {?} method
     * @return {?}
     */
    function mightHaveBody(method) {
        switch (method) {
            case 'DELETE':
            case 'GET':
            case 'HEAD':
            case 'OPTIONS':
            case 'JSONP':
                return false;
            default:
                return true;
        }
    }
    /**
     * Safely assert whether the given value is an ArrayBuffer.
     *
     * In some execution environments ArrayBuffer is not defined.
     * @param {?} value
     * @return {?}
     */
    function isArrayBuffer(value) {
        return typeof ArrayBuffer !== 'undefined' && value instanceof ArrayBuffer;
    }
    /**
     * Safely assert whether the given value is a Blob.
     *
     * In some execution environments Blob is not defined.
     * @param {?} value
     * @return {?}
     */
    function isBlob(value) {
        return typeof Blob !== 'undefined' && value instanceof Blob;
    }
    /**
     * Safely assert whether the given value is a FormData instance.
     *
     * In some execution environments FormData is not defined.
     * @param {?} value
     * @return {?}
     */
    function isFormData(value) {
        return typeof FormData !== 'undefined' && value instanceof FormData;
    }
    /**
     * An outgoing HTTP request with an optional typed body.
     *
     * `HttpRequest` represents an outgoing request, including URL, method,
     * headers, body, and other request configuration options. Instances should be
     * assumed to be immutable. To modify a `HttpRequest`, the `clone`
     * method should be used.
     *
     * \@publicApi
     * @template T
     */
    class HttpRequest {
        /**
         * @param {?} method
         * @param {?} url
         * @param {?=} third
         * @param {?=} fourth
         */
        constructor(method, url, third, fourth) {
            this.url = url;
            /**
             * The request body, or `null` if one isn't set.
             *
             * Bodies are not enforced to be immutable, as they can include a reference to any
             * user-defined data type. However, interceptors should take care to preserve
             * idempotence by treating them as such.
             */
            this.body = null;
            /**
             * Whether this request should be made in a way that exposes progress events.
             *
             * Progress events are expensive (change detection runs on each event) and so
             * they should only be requested if the consumer intends to monitor them.
             */
            this.reportProgress = false;
            /**
             * Whether this request should be sent with outgoing credentials (cookies).
             */
            this.withCredentials = false;
            /**
             * The expected response type of the server.
             *
             * This is used to parse the response appropriately before returning it to
             * the requestee.
             */
            this.responseType = 'json';
            this.method = method.toUpperCase();
            // Next, need to figure out which argument holds the HttpRequestInit
            // options, if any.
            /** @type {?} */
            let options;
            // Check whether a body argument is expected. The only valid way to omit
            // the body argument is to use a known no-body method like GET.
            if (mightHaveBody(this.method) || !!fourth) {
                // Body is the third argument, options are the fourth.
                this.body = (third !== undefined) ? (/** @type {?} */ (third)) : null;
                options = fourth;
            }
            else {
                // No body required, options are the third argument. The body stays null.
                options = (/** @type {?} */ (third));
            }
            // If options have been passed, interpret them.
            if (options) {
                // Normalize reportProgress and withCredentials.
                this.reportProgress = !!options.reportProgress;
                this.withCredentials = !!options.withCredentials;
                // Override default response type of 'json' if one is provided.
                if (!!options.responseType) {
                    this.responseType = options.responseType;
                }
                // Override headers if they're provided.
                if (!!options.headers) {
                    this.headers = options.headers;
                }
                if (!!options.params) {
                    this.params = options.params;
                }
            }
            // If no headers have been passed in, construct a new HttpHeaders instance.
            if (!this.headers) {
                this.headers = new HttpHeaders();
            }
            // If no parameters have been passed in, construct a new HttpUrlEncodedParams instance.
            if (!this.params) {
                this.params = new HttpParams();
                this.urlWithParams = url;
            }
            else {
                // Encode the parameters to a string in preparation for inclusion in the URL.
                /** @type {?} */
                const params = this.params.toString();
                if (params.length === 0) {
                    // No parameters, the visible URL is just the URL given at creation time.
                    this.urlWithParams = url;
                }
                else {
                    // Does the URL already have query parameters? Look for '?'.
                    /** @type {?} */
                    const qIdx = url.indexOf('?');
                    // There are 3 cases to handle:
                    // 1) No existing parameters -> append '?' followed by params.
                    // 2) '?' exists and is followed by existing query string ->
                    //    append '&' followed by params.
                    // 3) '?' exists at the end of the url -> append params directly.
                    // This basically amounts to determining the character, if any, with
                    // which to join the URL and parameters.
                    /** @type {?} */
                    const sep = qIdx === -1 ? '?' : (qIdx < url.length - 1 ? '&' : '');
                    this.urlWithParams = url + sep + params;
                }
            }
        }
        /**
         * Transform the free-form body into a serialized format suitable for
         * transmission to the server.
         * @return {?}
         */
        serializeBody() {
            // If no body is present, no need to serialize it.
            if (this.body === null) {
                return null;
            }
            // Check whether the body is already in a serialized form. If so,
            // it can just be returned directly.
            if (isArrayBuffer(this.body) || isBlob(this.body) || isFormData(this.body) ||
                typeof this.body === 'string') {
                return this.body;
            }
            // Check whether the body is an instance of HttpUrlEncodedParams.
            if (this.body instanceof HttpParams) {
                return this.body.toString();
            }
            // Check whether the body is an object or array, and serialize with JSON if so.
            if (typeof this.body === 'object' || typeof this.body === 'boolean' ||
                Array.isArray(this.body)) {
                return JSON.stringify(this.body);
            }
            // Fall back on toString() for everything else.
            return ((/** @type {?} */ (this.body))).toString();
        }
        /**
         * Examine the body and attempt to infer an appropriate MIME type
         * for it.
         *
         * If no such type can be inferred, this method will return `null`.
         * @return {?}
         */
        detectContentTypeHeader() {
            // An empty body has no content type.
            if (this.body === null) {
                return null;
            }
            // FormData bodies rely on the browser's content type assignment.
            if (isFormData(this.body)) {
                return null;
            }
            // Blobs usually have their own content type. If it doesn't, then
            // no type can be inferred.
            if (isBlob(this.body)) {
                return this.body.type || null;
            }
            // Array buffers have unknown contents and thus no type can be inferred.
            if (isArrayBuffer(this.body)) {
                return null;
            }
            // Technically, strings could be a form of JSON data, but it's safe enough
            // to assume they're plain strings.
            if (typeof this.body === 'string') {
                return 'text/plain';
            }
            // `HttpUrlEncodedParams` has its own content-type.
            if (this.body instanceof HttpParams) {
                return 'application/x-www-form-urlencoded;charset=UTF-8';
            }
            // Arrays, objects, and numbers will be encoded as JSON.
            if (typeof this.body === 'object' || typeof this.body === 'number' ||
                Array.isArray(this.body)) {
                return 'application/json';
            }
            // No type could be inferred.
            return null;
        }
        /**
         * @param {?=} update
         * @return {?}
         */
        clone(update = {}) {
            // For method, url, and responseType, take the current value unless
            // it is overridden in the update hash.
            /** @type {?} */
            const method = update.method || this.method;
            /** @type {?} */
            const url = update.url || this.url;
            /** @type {?} */
            const responseType = update.responseType || this.responseType;
            // The body is somewhat special - a `null` value in update.body means
            // whatever current body is present is being overridden with an empty
            // body, whereas an `undefined` value in update.body implies no
            // override.
            /** @type {?} */
            const body = (update.body !== undefined) ? update.body : this.body;
            // Carefully handle the boolean options to differentiate between
            // `false` and `undefined` in the update args.
            /** @type {?} */
            const withCredentials = (update.withCredentials !== undefined) ? update.withCredentials : this.withCredentials;
            /** @type {?} */
            const reportProgress = (update.reportProgress !== undefined) ? update.reportProgress : this.reportProgress;
            // Headers and params may be appended to if `setHeaders` or
            // `setParams` are used.
            /** @type {?} */
            let headers = update.headers || this.headers;
            /** @type {?} */
            let params = update.params || this.params;
            // Check whether the caller has asked to add headers.
            if (update.setHeaders !== undefined) {
                // Set every requested header.
                headers =
                    Object.keys(update.setHeaders)
                        .reduce((/**
                     * @param {?} headers
                     * @param {?} name
                     * @return {?}
                     */
                    (headers, name) => headers.set(name, (/** @type {?} */ (update.setHeaders))[name])), headers);
            }
            // Check whether the caller has asked to set params.
            if (update.setParams) {
                // Set every requested param.
                params = Object.keys(update.setParams)
                    .reduce((/**
                 * @param {?} params
                 * @param {?} param
                 * @return {?}
                 */
                (params, param) => params.set(param, (/** @type {?} */ (update.setParams))[param])), params);
            }
            // Finally, construct the new HttpRequest using the pieces from above.
            return new HttpRequest(method, url, body, {
                params,
                headers,
                reportProgress,
                responseType,
                withCredentials,
            });
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/http/src/response.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @enum {number} */
    const HttpEventType = {
        /**
         * The request was sent out over the wire.
         */
        Sent: 0,
        /**
         * An upload progress event was received.
         */
        UploadProgress: 1,
        /**
         * The response status code and headers were received.
         */
        ResponseHeader: 2,
        /**
         * A download progress event was received.
         */
        DownloadProgress: 3,
        /**
         * The full response including the body was received.
         */
        Response: 4,
        /**
         * A custom event from an interceptor or a backend.
         */
        User: 5,
    };
    HttpEventType[HttpEventType.Sent] = 'Sent';
    HttpEventType[HttpEventType.UploadProgress] = 'UploadProgress';
    HttpEventType[HttpEventType.ResponseHeader] = 'ResponseHeader';
    HttpEventType[HttpEventType.DownloadProgress] = 'DownloadProgress';
    HttpEventType[HttpEventType.Response] = 'Response';
    HttpEventType[HttpEventType.User] = 'User';
    /**
     * Base class for both `HttpResponse` and `HttpHeaderResponse`.
     *
     * \@publicApi
     * @abstract
     */
    class HttpResponseBase {
        /**
         * Super-constructor for all responses.
         *
         * The single parameter accepted is an initialization hash. Any properties
         * of the response passed there will override the default values.
         * @param {?} init
         * @param {?=} defaultStatus
         * @param {?=} defaultStatusText
         */
        constructor(init, defaultStatus = 200, defaultStatusText = 'OK') {
            // If the hash has values passed, use them to initialize the response.
            // Otherwise use the default values.
            this.headers = init.headers || new HttpHeaders();
            this.status = init.status !== undefined ? init.status : defaultStatus;
            this.statusText = init.statusText || defaultStatusText;
            this.url = init.url || null;
            // Cache the ok value to avoid defining a getter.
            this.ok = this.status >= 200 && this.status < 300;
        }
    }
    /**
     * A partial HTTP response which only includes the status and header data,
     * but no response body.
     *
     * `HttpHeaderResponse` is a `HttpEvent` available on the response
     * event stream, only when progress events are requested.
     *
     * \@publicApi
     */
    class HttpHeaderResponse extends HttpResponseBase {
        /**
         * Create a new `HttpHeaderResponse` with the given parameters.
         * @param {?=} init
         */
        constructor(init = {}) {
            super(init);
            this.type = HttpEventType.ResponseHeader;
        }
        /**
         * Copy this `HttpHeaderResponse`, overriding its contents with the
         * given parameter hash.
         * @param {?=} update
         * @return {?}
         */
        clone(update = {}) {
            // Perform a straightforward initialization of the new HttpHeaderResponse,
            // overriding the current parameters with new ones if given.
            return new HttpHeaderResponse({
                headers: update.headers || this.headers,
                status: update.status !== undefined ? update.status : this.status,
                statusText: update.statusText || this.statusText,
                url: update.url || this.url || undefined,
            });
        }
    }
    /**
     * A full HTTP response, including a typed response body (which may be `null`
     * if one was not returned).
     *
     * `HttpResponse` is a `HttpEvent` available on the response event
     * stream.
     *
     * \@publicApi
     * @template T
     */
    class HttpResponse extends HttpResponseBase {
        /**
         * Construct a new `HttpResponse`.
         * @param {?=} init
         */
        constructor(init = {}) {
            super(init);
            this.type = HttpEventType.Response;
            this.body = init.body !== undefined ? init.body : null;
        }
        /**
         * @param {?=} update
         * @return {?}
         */
        clone(update = {}) {
            return new HttpResponse({
                body: (update.body !== undefined) ? update.body : this.body,
                headers: update.headers || this.headers,
                status: (update.status !== undefined) ? update.status : this.status,
                statusText: update.statusText || this.statusText,
                url: update.url || this.url || undefined,
            });
        }
    }
    /**
     * A response that represents an error or failure, either from a
     * non-successful HTTP status, an error while executing the request,
     * or some other failure which occurred during the parsing of the response.
     *
     * Any error returned on the `Observable` response stream will be
     * wrapped in an `HttpErrorResponse` to provide additional context about
     * the state of the HTTP layer when the error occurred. The error property
     * will contain either a wrapped Error object or the error response returned
     * from the server.
     *
     * \@publicApi
     */
    class HttpErrorResponse extends HttpResponseBase {
        /**
         * @param {?} init
         */
        constructor(init) {
            // Initialize with a default status of 0 / Unknown Error.
            super(init, 0, 'Unknown Error');
            this.name = 'HttpErrorResponse';
            /**
             * Errors are never okay, even when the status code is in the 2xx success range.
             */
            this.ok = false;
            // If the response was successful, then this was a parse error. Otherwise, it was
            // a protocol-level failure of some sort. Either the request failed in transit
            // or the server returned an unsuccessful status code.
            if (this.status >= 200 && this.status < 300) {
                this.message = `Http failure during parsing for ${init.url || '(unknown url)'}`;
            }
            else {
                this.message = `Http failure response for ${init.url || '(unknown url)'}: ${init.status} ${init.statusText}`;
            }
            this.error = init.error || null;
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/http/src/client.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Constructs an instance of `HttpRequestOptions<T>` from a source `HttpMethodOptions` and
     * the given `body`. This function clones the object and adds the body.
     *
     * Note that the `responseType` *options* value is a String that identifies the
     * single data type of the response.
     * A single overload version of the method handles each response type.
     * The value of `responseType` cannot be a union, as the combined signature could imply.
     *
     * @template T
     * @param {?} options
     * @param {?} body
     * @return {?}
     */
    function addBody(options, body) {
        return {
            body,
            headers: options.headers,
            observe: options.observe,
            params: options.params,
            reportProgress: options.reportProgress,
            responseType: options.responseType,
            withCredentials: options.withCredentials,
        };
    }
    /**
     * Performs HTTP requests.
     * This service is available as an injectable class, with methods to perform HTTP requests.
     * Each request method has multiple signatures, and the return type varies based on
     * the signature that is called (mainly the values of `observe` and `responseType`).
     *
     * Note that the `responseType` *options* value is a String that identifies the
     * single data type of the response.
     * A single overload version of the method handles each response type.
     * The value of `responseType` cannot be a union, as the combined signature could imply.
     *
     * \@usageNotes
     * Sample HTTP requests for the [Tour of Heroes](/tutorial/toh-pt0) application.
     *
     * ### HTTP Request Example
     *
     * ```
     *  // GET heroes whose name contains search term
     * searchHeroes(term: string): observable<Hero[]>{
     *
     *  const params = new HttpParams({fromString: 'name=term'});
     *    return this.httpClient.request('GET', this.heroesUrl, {responseType:'json', params});
     * }
     * ```
     * ### JSONP Example
     * ```
     * requestJsonp(url, callback = 'callback') {
     *  return this.httpClient.jsonp(this.heroesURL, callback);
     * }
     * ```
     *
     * ### PATCH Example
     * ```
     * // PATCH one of the heroes' name
     * patchHero (id: number, heroName: string): Observable<{}> {
     * const url = `${this.heroesUrl}/${id}`;   // PATCH api/heroes/42
     *  return this.httpClient.patch(url, {name: heroName}, httpOptions)
     *    .pipe(catchError(this.handleError('patchHero')));
     * }
     * ```
     *
     * @see [HTTP Guide](guide/http)
     *
     * \@publicApi
     */
    class HttpClient {
        /**
         * @param {?} handler
         */
        constructor(handler) {
            this.handler = handler;
        }
        /**
         * Constructs an observable for a generic HTTP request that, when subscribed,
         * fires the request through the chain of registered interceptors and on to the
         * server.
         *
         * You can pass an `HttpRequest` directly as the only parameter. In this case,
         * the call returns an observable of the raw `HttpEvent` stream.
         *
         * Alternatively you can pass an HTTP method as the first parameter,
         * a URL string as the second, and an options hash containing the request body as the third.
         * See `addBody()`. In this case, the specified `responseType` and `observe` options determine the
         * type of returned observable.
         *   * The `responseType` value determines how a successful response body is parsed.
         *   * If `responseType` is the default `json`, you can pass a type interface for the resulting
         * object as a type parameter to the call.
         *
         * The `observe` value determines the return type, according to what you are interested in
         * observing.
         *   * An `observe` value of events returns an observable of the raw `HttpEvent` stream, including
         * progress events by default.
         *   * An `observe` value of response returns an observable of `HttpResponse<T>`,
         * where the `T` parameter depends on the `responseType` and any optionally provided type
         * parameter.
         *   * An `observe` value of body returns an observable of `<T>` with the same `T` body type.
         *
         * @param {?} first
         * @param {?=} url
         * @param {?=} options
         * @return {?}
         */
        request(first, url, options = {}) {
            /** @type {?} */
            let req;
            // First, check whether the primary argument is an instance of `HttpRequest`.
            if (first instanceof HttpRequest) {
                // It is. The other arguments must be undefined (per the signatures) and can be
                // ignored.
                req = first;
            }
            else {
                // It's a string, so it represents a URL. Construct a request based on it,
                // and incorporate the remaining arguments (assuming `GET` unless a method is
                // provided.
                // Figure out the headers.
                /** @type {?} */
                let headers = undefined;
                if (options.headers instanceof HttpHeaders) {
                    headers = options.headers;
                }
                else {
                    headers = new HttpHeaders(options.headers);
                }
                // Sort out parameters.
                /** @type {?} */
                let params = undefined;
                if (!!options.params) {
                    if (options.params instanceof HttpParams) {
                        params = options.params;
                    }
                    else {
                        params = new HttpParams((/** @type {?} */ ({ fromObject: options.params })));
                    }
                }
                // Construct the request.
                req = new HttpRequest(first, (/** @type {?} */ (url)), (options.body !== undefined ? options.body : null), {
                    headers,
                    params,
                    reportProgress: options.reportProgress,
                    // By default, JSON is assumed to be returned for all calls.
                    responseType: options.responseType || 'json',
                    withCredentials: options.withCredentials,
                });
            }
            // Start with an Observable.of() the initial request, and run the handler (which
            // includes all interceptors) inside a concatMap(). This way, the handler runs
            // inside an Observable chain, which causes interceptors to be re-run on every
            // subscription (this also makes retries re-run the handler, including interceptors).
            /** @type {?} */
            const events$ = of(req).pipe(concatMap((/**
             * @param {?} req
             * @return {?}
             */
            (req) => this.handler.handle(req))));
            // If coming via the API signature which accepts a previously constructed HttpRequest,
            // the only option is to get the event stream. Otherwise, return the event stream if
            // that is what was requested.
            if (first instanceof HttpRequest || options.observe === 'events') {
                return events$;
            }
            // The requested stream contains either the full response or the body. In either
            // case, the first step is to filter the event stream to extract a stream of
            // responses(s).
            /** @type {?} */
            const res$ = (/** @type {?} */ (events$.pipe(filter((/**
             * @param {?} event
             * @return {?}
             */
            (event) => event instanceof HttpResponse)))));
            // Decide which stream to return.
            switch (options.observe || 'body') {
                case 'body':
                    // The requested stream is the body. Map the response stream to the response
                    // body. This could be done more simply, but a misbehaving interceptor might
                    // transform the response body into a different format and ignore the requested
                    // responseType. Guard against this by validating that the response is of the
                    // requested type.
                    switch (req.responseType) {
                        case 'arraybuffer':
                            return res$.pipe(map((/**
                             * @param {?} res
                             * @return {?}
                             */
                            (res) => {
                                // Validate that the body is an ArrayBuffer.
                                if (res.body !== null && !(res.body instanceof ArrayBuffer)) {
                                    throw new Error('Response is not an ArrayBuffer.');
                                }
                                return res.body;
                            })));
                        case 'blob':
                            return res$.pipe(map((/**
                             * @param {?} res
                             * @return {?}
                             */
                            (res) => {
                                // Validate that the body is a Blob.
                                if (res.body !== null && !(res.body instanceof Blob)) {
                                    throw new Error('Response is not a Blob.');
                                }
                                return res.body;
                            })));
                        case 'text':
                            return res$.pipe(map((/**
                             * @param {?} res
                             * @return {?}
                             */
                            (res) => {
                                // Validate that the body is a string.
                                if (res.body !== null && typeof res.body !== 'string') {
                                    throw new Error('Response is not a string.');
                                }
                                return res.body;
                            })));
                        case 'json':
                        default:
                            // No validation needed for JSON responses, as they can be of any type.
                            return res$.pipe(map((/**
                             * @param {?} res
                             * @return {?}
                             */
                            (res) => res.body)));
                    }
                case 'response':
                    // The response stream was requested directly, so return it.
                    return res$;
                default:
                    // Guard against new future observe types being added.
                    throw new Error(`Unreachable: unhandled observe type ${options.observe}}`);
            }
        }
        /**
         * Constructs an observable that, when subscribed, causes the configured
         * `DELETE` request to execute on the server. See the individual overloads for
         * details on the return type.
         *
         * @param {?} url     The endpoint URL.
         * @param {?=} options The HTTP options to send with the request.
         *
         * @return {?}
         */
        delete(url, options = {}) {
            return this.request('DELETE', url, (/** @type {?} */ (options)));
        }
        /**
         * Constructs an observable that, when subscribed, causes the configured
         * `GET` request to execute on the server. See the individual overloads for
         * details on the return type.
         * @param {?} url
         * @param {?=} options
         * @return {?}
         */
        get(url, options = {}) {
            return this.request('GET', url, (/** @type {?} */ (options)));
        }
        /**
         * Constructs an observable that, when subscribed, causes the configured
         * `HEAD` request to execute on the server. The `HEAD` method returns
         * meta information about the resource without transferring the
         * resource itself. See the individual overloads for
         * details on the return type.
         * @param {?} url
         * @param {?=} options
         * @return {?}
         */
        head(url, options = {}) {
            return this.request('HEAD', url, (/** @type {?} */ (options)));
        }
        /**
         * Constructs an `Observable` that, when subscribed, causes a request with the special method
         * `JSONP` to be dispatched via the interceptor pipeline.
         * The [JSONP pattern](https://en.wikipedia.org/wiki/JSONP) works around limitations of certain
         * API endpoints that don't support newer,
         * and preferable [CORS](https://developer.mozilla.org/en-US/docs/Web/HTTP/CORS) protocol.
         * JSONP treats the endpoint API as a JavaScript file and tricks the browser to process the
         * requests even if the API endpoint is not located on the same domain (origin) as the client-side
         * application making the request.
         * The endpoint API must support JSONP callback for JSONP requests to work.
         * The resource API returns the JSON response wrapped in a callback function.
         * You can pass the callback function name as one of the query parameters.
         * Note that JSONP requests can only be used with `GET` requests.
         *
         * @template T
         * @param {?} url The resource URL.
         * @param {?} callbackParam The callback function name.
         *
         * @return {?}
         */
        jsonp(url, callbackParam) {
            return this.request('JSONP', url, {
                params: new HttpParams().append(callbackParam, 'JSONP_CALLBACK'),
                observe: 'body',
                responseType: 'json',
            });
        }
        /**
         * Constructs an `Observable` that, when subscribed, causes the configured
         * `OPTIONS` request to execute on the server. This method allows the client
         * to determine the supported HTTP methods and other capabilites of an endpoint,
         * without implying a resource action. See the individual overloads for
         * details on the return type.
         * @param {?} url
         * @param {?=} options
         * @return {?}
         */
        options(url, options = {}) {
            return this.request('OPTIONS', url, (/** @type {?} */ (options)));
        }
        /**
         * Constructs an observable that, when subscribed, causes the configured
         * `PATCH` request to execute on the server. See the individual overloads for
         * details on the return type.
         * @param {?} url
         * @param {?} body
         * @param {?=} options
         * @return {?}
         */
        patch(url, body, options = {}) {
            return this.request('PATCH', url, addBody(options, body));
        }
        /**
         * Constructs an observable that, when subscribed, causes the configured
         * `POST` request to execute on the server. The server responds with the location of
         * the replaced resource. See the individual overloads for
         * details on the return type.
         * @param {?} url
         * @param {?} body
         * @param {?=} options
         * @return {?}
         */
        post(url, body, options = {}) {
            return this.request('POST', url, addBody(options, body));
        }
        /**
         * Constructs an observable that, when subscribed, causes the configured
         * `PUT` request to execute on the server. The `PUT` method replaces an existing resource
         * with a new set of values.
         * See the individual overloads for details on the return type.
         * @param {?} url
         * @param {?} body
         * @param {?=} options
         * @return {?}
         */
        put(url, body, options = {}) {
            return this.request('PUT', url, addBody(options, body));
        }
    }
    HttpClient.ɵfac = function HttpClient_Factory(t) { return new (t || HttpClient)(ɵɵinject(HttpHandler)); };
    HttpClient.ɵprov = ɵɵdefineInjectable({ token: HttpClient, factory: HttpClient.ɵfac });
    /** @nocollapse */
    HttpClient.ctorParameters = () => [
        { type: HttpHandler }
    ];
    /**
     * `HttpHandler` which applies an `HttpInterceptor` to an `HttpRequest`.
     *
     *
     */
    class HttpInterceptorHandler {
        /**
         * @param {?} next
         * @param {?} interceptor
         */
        constructor(next, interceptor) {
            this.next = next;
            this.interceptor = interceptor;
        }
        /**
         * @param {?} req
         * @return {?}
         */
        handle(req) {
            return this.interceptor.intercept(req, this.next);
        }
    }
    /**
     * A multi-provider token that represents the array of registered
     * `HttpInterceptor` objects.
     *
     * \@publicApi
     * @type {?}
     */
    const HTTP_INTERCEPTORS = new InjectionToken('HTTP_INTERCEPTORS');
    class NoopInterceptor {
        /**
         * @param {?} req
         * @param {?} next
         * @return {?}
         */
        intercept(req, next) {
            return next.handle(req);
        }
    }
    NoopInterceptor.ɵfac = function NoopInterceptor_Factory(t) { return new (t || NoopInterceptor)(); };
    NoopInterceptor.ɵprov = ɵɵdefineInjectable({ token: NoopInterceptor, factory: NoopInterceptor.ɵfac });

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/http/src/jsonp.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    // Every request made through JSONP needs a callback name that's unique across the
    // whole page. Each request is assigned an id and the callback name is constructed
    // from that. The next id to be assigned is tracked in a global variable here that
    // is shared among all applications on the page.
    /** @type {?} */
    let nextRequestId = 0;
    // Error text given when a JSONP script is injected, but doesn't invoke the callback
    // passed in its URL.
    /** @type {?} */
    const JSONP_ERR_NO_CALLBACK = 'JSONP injected script did not invoke callback.';
    // Error text given when a request is passed to the JsonpClientBackend that doesn't
    // have a request method JSONP.
    /** @type {?} */
    const JSONP_ERR_WRONG_METHOD = 'JSONP requests must use JSONP request method.';
    /** @type {?} */
    const JSONP_ERR_WRONG_RESPONSE_TYPE = 'JSONP requests must use Json response type.';
    /**
     * DI token/abstract type representing a map of JSONP callbacks.
     *
     * In the browser, this should always be the `window` object.
     *
     *
     * @abstract
     */
    class JsonpCallbackContext {
    }
    /**
     * Processes an `HttpRequest` with the JSONP method,
     * by performing JSONP style requests.
     * @see `HttpHandler`
     * @see `HttpXhrBackend`
     *
     * \@publicApi
     */
    class JsonpClientBackend {
        /**
         * @param {?} callbackMap
         * @param {?} document
         */
        constructor(callbackMap, document) {
            this.callbackMap = callbackMap;
            this.document = document;
        }
        /**
         * Get the name of the next callback method, by incrementing the global `nextRequestId`.
         * @private
         * @return {?}
         */
        nextCallback() {
            return `ng_jsonp_callback_${nextRequestId++}`;
        }
        /**
         * Processes a JSONP request and returns an event stream of the results.
         * @param {?} req The request object.
         * @return {?} An observable of the response events.
         *
         */
        handle(req) {
            // Firstly, check both the method and response type. If either doesn't match
            // then the request was improperly routed here and cannot be handled.
            if (req.method !== 'JSONP') {
                throw new Error(JSONP_ERR_WRONG_METHOD);
            }
            else if (req.responseType !== 'json') {
                throw new Error(JSONP_ERR_WRONG_RESPONSE_TYPE);
            }
            // Everything else happens inside the Observable boundary.
            return new Observable((/**
             * @param {?} observer
             * @return {?}
             */
            (observer) => {
                // The first step to make a request is to generate the callback name, and replace the
                // callback placeholder in the URL with the name. Care has to be taken here to ensure
                // a trailing &, if matched, gets inserted back into the URL in the correct place.
                /** @type {?} */
                const callback = this.nextCallback();
                /** @type {?} */
                const url = req.urlWithParams.replace(/=JSONP_CALLBACK(&|$)/, `=${callback}$1`);
                // Construct the <script> tag and point it at the URL.
                /** @type {?} */
                const node = this.document.createElement('script');
                node.src = url;
                // A JSONP request requires waiting for multiple callbacks. These variables
                // are closed over and track state across those callbacks.
                // The response object, if one has been received, or null otherwise.
                /** @type {?} */
                let body = null;
                // Whether the response callback has been called.
                /** @type {?} */
                let finished = false;
                // Whether the request has been cancelled (and thus any other callbacks)
                // should be ignored.
                /** @type {?} */
                let cancelled = false;
                // Set the response callback in this.callbackMap (which will be the window
                // object in the browser. The script being loaded via the <script> tag will
                // eventually call this callback.
                this.callbackMap[callback] = (/**
                 * @param {?=} data
                 * @return {?}
                 */
                (data) => {
                    // Data has been received from the JSONP script. Firstly, delete this callback.
                    delete this.callbackMap[callback];
                    // Next, make sure the request wasn't cancelled in the meantime.
                    if (cancelled) {
                        return;
                    }
                    // Set state to indicate data was received.
                    body = data;
                    finished = true;
                });
                // cleanup() is a utility closure that removes the <script> from the page and
                // the response callback from the window. This logic is used in both the
                // success, error, and cancellation paths, so it's extracted out for convenience.
                /** @type {?} */
                const cleanup = (/**
                 * @return {?}
                 */
                () => {
                    // Remove the <script> tag if it's still on the page.
                    if (node.parentNode) {
                        node.parentNode.removeChild(node);
                    }
                    // Remove the response callback from the callbackMap (window object in the
                    // browser).
                    delete this.callbackMap[callback];
                });
                // onLoad() is the success callback which runs after the response callback
                // if the JSONP script loads successfully. The event itself is unimportant.
                // If something went wrong, onLoad() may run without the response callback
                // having been invoked.
                /** @type {?} */
                const onLoad = (/**
                 * @param {?} event
                 * @return {?}
                 */
                (event) => {
                    // Do nothing if the request has been cancelled.
                    if (cancelled) {
                        return;
                    }
                    // Cleanup the page.
                    cleanup();
                    // Check whether the response callback has run.
                    if (!finished) {
                        // It hasn't, something went wrong with the request. Return an error via
                        // the Observable error path. All JSONP errors have status 0.
                        observer.error(new HttpErrorResponse({
                            url,
                            status: 0,
                            statusText: 'JSONP Error',
                            error: new Error(JSONP_ERR_NO_CALLBACK),
                        }));
                        return;
                    }
                    // Success. body either contains the response body or null if none was
                    // returned.
                    observer.next(new HttpResponse({
                        body,
                        status: 200,
                        statusText: 'OK',
                        url,
                    }));
                    // Complete the stream, the response is over.
                    observer.complete();
                });
                // onError() is the error callback, which runs if the script returned generates
                // a Javascript error. It emits the error via the Observable error channel as
                // a HttpErrorResponse.
                /** @type {?} */
                const onError = (/**
                 * @param {?} error
                 * @return {?}
                 */
                (error) => {
                    // If the request was already cancelled, no need to emit anything.
                    if (cancelled) {
                        return;
                    }
                    cleanup();
                    // Wrap the error in a HttpErrorResponse.
                    observer.error(new HttpErrorResponse({
                        error,
                        status: 0,
                        statusText: 'JSONP Error',
                        url,
                    }));
                });
                // Subscribe to both the success (load) and error events on the <script> tag,
                // and add it to the page.
                node.addEventListener('load', onLoad);
                node.addEventListener('error', onError);
                this.document.body.appendChild(node);
                // The request has now been successfully sent.
                observer.next({ type: HttpEventType.Sent });
                // Cancellation handler.
                return (/**
                 * @return {?}
                 */
                () => {
                    // Track the cancellation so event listeners won't do anything even if already scheduled.
                    cancelled = true;
                    // Remove the event listeners so they won't run if the events later fire.
                    node.removeEventListener('load', onLoad);
                    node.removeEventListener('error', onError);
                    // And finally, clean up the page.
                    cleanup();
                });
            }));
        }
    }
    JsonpClientBackend.ɵfac = function JsonpClientBackend_Factory(t) { return new (t || JsonpClientBackend)(ɵɵinject(JsonpCallbackContext), ɵɵinject(DOCUMENT$1)); };
    JsonpClientBackend.ɵprov = ɵɵdefineInjectable({ token: JsonpClientBackend, factory: JsonpClientBackend.ɵfac });
    /** @nocollapse */
    JsonpClientBackend.ctorParameters = () => [
        { type: JsonpCallbackContext },
        { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT$1,] }] }
    ];
    /**
     * Identifies requests with the method JSONP and
     * shifts them to the `JsonpClientBackend`.
     *
     * @see `HttpInterceptor`
     *
     * \@publicApi
     */
    class JsonpInterceptor {
        /**
         * @param {?} jsonp
         */
        constructor(jsonp) {
            this.jsonp = jsonp;
        }
        /**
         * Identifies and handles a given JSONP request.
         * @param {?} req The outgoing request object to handle.
         * @param {?} next The next interceptor in the chain, or the backend
         * if no interceptors remain in the chain.
         * @return {?} An observable of the event stream.
         */
        intercept(req, next) {
            if (req.method === 'JSONP') {
                return this.jsonp.handle((/** @type {?} */ (req)));
            }
            // Fall through for normal HTTP requests.
            return next.handle(req);
        }
    }
    JsonpInterceptor.ɵfac = function JsonpInterceptor_Factory(t) { return new (t || JsonpInterceptor)(ɵɵinject(JsonpClientBackend)); };
    JsonpInterceptor.ɵprov = ɵɵdefineInjectable({ token: JsonpInterceptor, factory: JsonpInterceptor.ɵfac });
    /** @nocollapse */
    JsonpInterceptor.ctorParameters = () => [
        { type: JsonpClientBackend }
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/http/src/xhr.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const XSSI_PREFIX = /^\)\]\}',?\n/;
    /**
     * Determine an appropriate URL for the response, by checking either
     * XMLHttpRequest.responseURL or the X-Request-URL header.
     * @param {?} xhr
     * @return {?}
     */
    function getResponseUrl(xhr) {
        if ('responseURL' in xhr && xhr.responseURL) {
            return xhr.responseURL;
        }
        if (/^X-Request-URL:/m.test(xhr.getAllResponseHeaders())) {
            return xhr.getResponseHeader('X-Request-URL');
        }
        return null;
    }
    /**
     * A wrapper around the `XMLHttpRequest` constructor.
     *
     * \@publicApi
     * @abstract
     */
    class XhrFactory {
    }
    /**
     * A factory for `HttpXhrBackend` that uses the `XMLHttpRequest` browser API.
     *
     */
    class BrowserXhr {
        constructor() { }
        /**
         * @return {?}
         */
        build() {
            return (/** @type {?} */ ((new XMLHttpRequest())));
        }
    }
    BrowserXhr.ɵfac = function BrowserXhr_Factory(t) { return new (t || BrowserXhr)(); };
    BrowserXhr.ɵprov = ɵɵdefineInjectable({ token: BrowserXhr, factory: BrowserXhr.ɵfac });
    /** @nocollapse */
    BrowserXhr.ctorParameters = () => [];
    /**
     * Uses `XMLHttpRequest` to send requests to a backend server.
     * @see `HttpHandler`
     * @see `JsonpClientBackend`
     *
     * \@publicApi
     */
    class HttpXhrBackend {
        /**
         * @param {?} xhrFactory
         */
        constructor(xhrFactory) {
            this.xhrFactory = xhrFactory;
        }
        /**
         * Processes a request and returns a stream of response events.
         * @param {?} req The request object.
         * @return {?} An observable of the response events.
         */
        handle(req) {
            // Quick check to give a better error message when a user attempts to use
            // HttpClient.jsonp() without installing the JsonpClientModule
            if (req.method === 'JSONP') {
                throw new Error(`Attempted to construct Jsonp request without JsonpClientModule installed.`);
            }
            // Everything happens on Observable subscription.
            return new Observable((/**
             * @param {?} observer
             * @return {?}
             */
            (observer) => {
                // Start by setting up the XHR object with request method, URL, and withCredentials flag.
                /** @type {?} */
                const xhr = this.xhrFactory.build();
                xhr.open(req.method, req.urlWithParams);
                if (!!req.withCredentials) {
                    xhr.withCredentials = true;
                }
                // Add all the requested headers.
                req.headers.forEach((/**
                 * @param {?} name
                 * @param {?} values
                 * @return {?}
                 */
                (name, values) => xhr.setRequestHeader(name, values.join(','))));
                // Add an Accept header if one isn't present already.
                if (!req.headers.has('Accept')) {
                    xhr.setRequestHeader('Accept', 'application/json, text/plain, */*');
                }
                // Auto-detect the Content-Type header if one isn't present already.
                if (!req.headers.has('Content-Type')) {
                    /** @type {?} */
                    const detectedType = req.detectContentTypeHeader();
                    // Sometimes Content-Type detection fails.
                    if (detectedType !== null) {
                        xhr.setRequestHeader('Content-Type', detectedType);
                    }
                }
                // Set the responseType if one was requested.
                if (req.responseType) {
                    /** @type {?} */
                    const responseType = req.responseType.toLowerCase();
                    // JSON responses need to be processed as text. This is because if the server
                    // returns an XSSI-prefixed JSON response, the browser will fail to parse it,
                    // xhr.response will be null, and xhr.responseText cannot be accessed to
                    // retrieve the prefixed JSON data in order to strip the prefix. Thus, all JSON
                    // is parsed by first requesting text and then applying JSON.parse.
                    xhr.responseType = (/** @type {?} */ (((responseType !== 'json') ? responseType : 'text')));
                }
                // Serialize the request body if one is present. If not, this will be set to null.
                /** @type {?} */
                const reqBody = req.serializeBody();
                // If progress events are enabled, response headers will be delivered
                // in two events - the HttpHeaderResponse event and the full HttpResponse
                // event. However, since response headers don't change in between these
                // two events, it doesn't make sense to parse them twice. So headerResponse
                // caches the data extracted from the response whenever it's first parsed,
                // to ensure parsing isn't duplicated.
                /** @type {?} */
                let headerResponse = null;
                // partialFromXhr extracts the HttpHeaderResponse from the current XMLHttpRequest
                // state, and memoizes it into headerResponse.
                /** @type {?} */
                const partialFromXhr = (/**
                 * @return {?}
                 */
                () => {
                    if (headerResponse !== null) {
                        return headerResponse;
                    }
                    // Read status and normalize an IE9 bug (http://bugs.jquery.com/ticket/1450).
                    /** @type {?} */
                    const status = xhr.status === 1223 ? 204 : xhr.status;
                    /** @type {?} */
                    const statusText = xhr.statusText || 'OK';
                    // Parse headers from XMLHttpRequest - this step is lazy.
                    /** @type {?} */
                    const headers = new HttpHeaders(xhr.getAllResponseHeaders());
                    // Read the response URL from the XMLHttpResponse instance and fall back on the
                    // request URL.
                    /** @type {?} */
                    const url = getResponseUrl(xhr) || req.url;
                    // Construct the HttpHeaderResponse and memoize it.
                    headerResponse = new HttpHeaderResponse({ headers, status, statusText, url });
                    return headerResponse;
                });
                // Next, a few closures are defined for the various events which XMLHttpRequest can
                // emit. This allows them to be unregistered as event listeners later.
                // First up is the load event, which represents a response being fully available.
                /** @type {?} */
                const onLoad = (/**
                 * @return {?}
                 */
                () => {
                    // Read response state from the memoized partial data.
                    let { headers, status, statusText, url } = partialFromXhr();
                    // The body will be read out if present.
                    /** @type {?} */
                    let body = null;
                    if (status !== 204) {
                        // Use XMLHttpRequest.response if set, responseText otherwise.
                        body = (typeof xhr.response === 'undefined') ? xhr.responseText : xhr.response;
                    }
                    // Normalize another potential bug (this one comes from CORS).
                    if (status === 0) {
                        status = !!body ? 200 : 0;
                    }
                    // ok determines whether the response will be transmitted on the event or
                    // error channel. Unsuccessful status codes (not 2xx) will always be errors,
                    // but a successful status code can still result in an error if the user
                    // asked for JSON data and the body cannot be parsed as such.
                    /** @type {?} */
                    let ok = status >= 200 && status < 300;
                    // Check whether the body needs to be parsed as JSON (in many cases the browser
                    // will have done that already).
                    if (req.responseType === 'json' && typeof body === 'string') {
                        // Save the original body, before attempting XSSI prefix stripping.
                        /** @type {?} */
                        const originalBody = body;
                        body = body.replace(XSSI_PREFIX, '');
                        try {
                            // Attempt the parse. If it fails, a parse error should be delivered to the user.
                            body = body !== '' ? JSON.parse(body) : null;
                        }
                        catch (error) {
                            // Since the JSON.parse failed, it's reasonable to assume this might not have been a
                            // JSON response. Restore the original body (including any XSSI prefix) to deliver
                            // a better error response.
                            body = originalBody;
                            // If this was an error request to begin with, leave it as a string, it probably
                            // just isn't JSON. Otherwise, deliver the parsing error to the user.
                            if (ok) {
                                // Even though the response status was 2xx, this is still an error.
                                ok = false;
                                // The parse error contains the text of the body that failed to parse.
                                body = (/** @type {?} */ ({ error, text: body }));
                            }
                        }
                    }
                    if (ok) {
                        // A successful response is delivered on the event stream.
                        observer.next(new HttpResponse({
                            body,
                            headers,
                            status,
                            statusText,
                            url: url || undefined,
                        }));
                        // The full body has been received and delivered, no further events
                        // are possible. This request is complete.
                        observer.complete();
                    }
                    else {
                        // An unsuccessful request is delivered on the error channel.
                        observer.error(new HttpErrorResponse({
                            // The error in this case is the response body (error from the server).
                            error: body,
                            headers,
                            status,
                            statusText,
                            url: url || undefined,
                        }));
                    }
                });
                // The onError callback is called when something goes wrong at the network level.
                // Connection timeout, DNS error, offline, etc. These are actual errors, and are
                // transmitted on the error channel.
                /** @type {?} */
                const onError = (/**
                 * @param {?} error
                 * @return {?}
                 */
                (error) => {
                    const { url } = partialFromXhr();
                    /** @type {?} */
                    const res = new HttpErrorResponse({
                        error,
                        status: xhr.status || 0,
                        statusText: xhr.statusText || 'Unknown Error',
                        url: url || undefined,
                    });
                    observer.error(res);
                });
                // The sentHeaders flag tracks whether the HttpResponseHeaders event
                // has been sent on the stream. This is necessary to track if progress
                // is enabled since the event will be sent on only the first download
                // progerss event.
                /** @type {?} */
                let sentHeaders = false;
                // The download progress event handler, which is only registered if
                // progress events are enabled.
                /** @type {?} */
                const onDownProgress = (/**
                 * @param {?} event
                 * @return {?}
                 */
                (event) => {
                    // Send the HttpResponseHeaders event if it hasn't been sent already.
                    if (!sentHeaders) {
                        observer.next(partialFromXhr());
                        sentHeaders = true;
                    }
                    // Start building the download progress event to deliver on the response
                    // event stream.
                    /** @type {?} */
                    let progressEvent = {
                        type: HttpEventType.DownloadProgress,
                        loaded: event.loaded,
                    };
                    // Set the total number of bytes in the event if it's available.
                    if (event.lengthComputable) {
                        progressEvent.total = event.total;
                    }
                    // If the request was for text content and a partial response is
                    // available on XMLHttpRequest, include it in the progress event
                    // to allow for streaming reads.
                    if (req.responseType === 'text' && !!xhr.responseText) {
                        progressEvent.partialText = xhr.responseText;
                    }
                    // Finally, fire the event.
                    observer.next(progressEvent);
                });
                // The upload progress event handler, which is only registered if
                // progress events are enabled.
                /** @type {?} */
                const onUpProgress = (/**
                 * @param {?} event
                 * @return {?}
                 */
                (event) => {
                    // Upload progress events are simpler. Begin building the progress
                    // event.
                    /** @type {?} */
                    let progress = {
                        type: HttpEventType.UploadProgress,
                        loaded: event.loaded,
                    };
                    // If the total number of bytes being uploaded is available, include
                    // it.
                    if (event.lengthComputable) {
                        progress.total = event.total;
                    }
                    // Send the event.
                    observer.next(progress);
                });
                // By default, register for load and error events.
                xhr.addEventListener('load', onLoad);
                xhr.addEventListener('error', onError);
                // Progress events are only enabled if requested.
                if (req.reportProgress) {
                    // Download progress is always enabled if requested.
                    xhr.addEventListener('progress', onDownProgress);
                    // Upload progress depends on whether there is a body to upload.
                    if (reqBody !== null && xhr.upload) {
                        xhr.upload.addEventListener('progress', onUpProgress);
                    }
                }
                // Fire the request, and notify the event stream that it was fired.
                xhr.send((/** @type {?} */ (reqBody)));
                observer.next({ type: HttpEventType.Sent });
                // This is the return from the Observable function, which is the
                // request cancellation handler.
                return (/**
                 * @return {?}
                 */
                () => {
                    // On a cancellation, remove all registered event listeners.
                    xhr.removeEventListener('error', onError);
                    xhr.removeEventListener('load', onLoad);
                    if (req.reportProgress) {
                        xhr.removeEventListener('progress', onDownProgress);
                        if (reqBody !== null && xhr.upload) {
                            xhr.upload.removeEventListener('progress', onUpProgress);
                        }
                    }
                    // Finally, abort the in-flight request.
                    xhr.abort();
                });
            }));
        }
    }
    HttpXhrBackend.ɵfac = function HttpXhrBackend_Factory(t) { return new (t || HttpXhrBackend)(ɵɵinject(XhrFactory)); };
    HttpXhrBackend.ɵprov = ɵɵdefineInjectable({ token: HttpXhrBackend, factory: HttpXhrBackend.ɵfac });
    /** @nocollapse */
    HttpXhrBackend.ctorParameters = () => [
        { type: XhrFactory }
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/http/src/xsrf.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const XSRF_COOKIE_NAME = new InjectionToken('XSRF_COOKIE_NAME');
    /** @type {?} */
    const XSRF_HEADER_NAME = new InjectionToken('XSRF_HEADER_NAME');
    /**
     * Retrieves the current XSRF token to use with the next outgoing request.
     *
     * \@publicApi
     * @abstract
     */
    class HttpXsrfTokenExtractor {
    }
    /**
     * `HttpXsrfTokenExtractor` which retrieves the token from a cookie.
     */
    class HttpXsrfCookieExtractor {
        /**
         * @param {?} doc
         * @param {?} platform
         * @param {?} cookieName
         */
        constructor(doc, platform, cookieName) {
            this.doc = doc;
            this.platform = platform;
            this.cookieName = cookieName;
            this.lastCookieString = '';
            this.lastToken = null;
            /**
             * \@internal for testing
             */
            this.parseCount = 0;
        }
        /**
         * @return {?}
         */
        getToken() {
            if (this.platform === 'server') {
                return null;
            }
            /** @type {?} */
            const cookieString = this.doc.cookie || '';
            if (cookieString !== this.lastCookieString) {
                this.parseCount++;
                this.lastToken = parseCookieValue(cookieString, this.cookieName);
                this.lastCookieString = cookieString;
            }
            return this.lastToken;
        }
    }
    HttpXsrfCookieExtractor.ɵfac = function HttpXsrfCookieExtractor_Factory(t) { return new (t || HttpXsrfCookieExtractor)(ɵɵinject(DOCUMENT$1), ɵɵinject(PLATFORM_ID), ɵɵinject(XSRF_COOKIE_NAME)); };
    HttpXsrfCookieExtractor.ɵprov = ɵɵdefineInjectable({ token: HttpXsrfCookieExtractor, factory: HttpXsrfCookieExtractor.ɵfac });
    /** @nocollapse */
    HttpXsrfCookieExtractor.ctorParameters = () => [
        { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT$1,] }] },
        { type: String, decorators: [{ type: Inject, args: [PLATFORM_ID,] }] },
        { type: String, decorators: [{ type: Inject, args: [XSRF_COOKIE_NAME,] }] }
    ];
    /**
     * `HttpInterceptor` which adds an XSRF token to eligible outgoing requests.
     */
    class HttpXsrfInterceptor {
        /**
         * @param {?} tokenService
         * @param {?} headerName
         */
        constructor(tokenService, headerName) {
            this.tokenService = tokenService;
            this.headerName = headerName;
        }
        /**
         * @param {?} req
         * @param {?} next
         * @return {?}
         */
        intercept(req, next) {
            /** @type {?} */
            const lcUrl = req.url.toLowerCase();
            // Skip both non-mutating requests and absolute URLs.
            // Non-mutating requests don't require a token, and absolute URLs require special handling
            // anyway as the cookie set
            // on our origin is not the same as the token expected by another origin.
            if (req.method === 'GET' || req.method === 'HEAD' || lcUrl.startsWith('http://') ||
                lcUrl.startsWith('https://')) {
                return next.handle(req);
            }
            /** @type {?} */
            const token = this.tokenService.getToken();
            // Be careful not to overwrite an existing header of the same name.
            if (token !== null && !req.headers.has(this.headerName)) {
                req = req.clone({ headers: req.headers.set(this.headerName, token) });
            }
            return next.handle(req);
        }
    }
    HttpXsrfInterceptor.ɵfac = function HttpXsrfInterceptor_Factory(t) { return new (t || HttpXsrfInterceptor)(ɵɵinject(HttpXsrfTokenExtractor), ɵɵinject(XSRF_HEADER_NAME)); };
    HttpXsrfInterceptor.ɵprov = ɵɵdefineInjectable({ token: HttpXsrfInterceptor, factory: HttpXsrfInterceptor.ɵfac });
    /** @nocollapse */
    HttpXsrfInterceptor.ctorParameters = () => [
        { type: HttpXsrfTokenExtractor },
        { type: String, decorators: [{ type: Inject, args: [XSRF_HEADER_NAME,] }] }
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: packages/common/http/src/module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * An injectable `HttpHandler` that applies multiple interceptors
     * to a request before passing it to the given `HttpBackend`.
     *
     * The interceptors are loaded lazily from the injector, to allow
     * interceptors to themselves inject classes depending indirectly
     * on `HttpInterceptingHandler` itself.
     * @see `HttpInterceptor`
     */
    class HttpInterceptingHandler {
        /**
         * @param {?} backend
         * @param {?} injector
         */
        constructor(backend, injector) {
            this.backend = backend;
            this.injector = injector;
            this.chain = null;
        }
        /**
         * @param {?} req
         * @return {?}
         */
        handle(req) {
            if (this.chain === null) {
                /** @type {?} */
                const interceptors = this.injector.get(HTTP_INTERCEPTORS, []);
                this.chain = interceptors.reduceRight((/**
                 * @param {?} next
                 * @param {?} interceptor
                 * @return {?}
                 */
                (next, interceptor) => new HttpInterceptorHandler(next, interceptor)), this.backend);
            }
            return this.chain.handle(req);
        }
    }
    HttpInterceptingHandler.ɵfac = function HttpInterceptingHandler_Factory(t) { return new (t || HttpInterceptingHandler)(ɵɵinject(HttpBackend), ɵɵinject(Injector)); };
    HttpInterceptingHandler.ɵprov = ɵɵdefineInjectable({ token: HttpInterceptingHandler, factory: HttpInterceptingHandler.ɵfac });
    /** @nocollapse */
    HttpInterceptingHandler.ctorParameters = () => [
        { type: HttpBackend },
        { type: Injector }
    ];
    /**
     * Factory function that determines where to store JSONP callbacks.
     *
     * Ordinarily JSONP callbacks are stored on the `window` object, but this may not exist
     * in test environments. In that case, callbacks are stored on an anonymous object instead.
     *
     *
     * @return {?}
     */
    function jsonpCallbackContext() {
        if (typeof window === 'object') {
            return window;
        }
        return {};
    }
    /**
     * Configures XSRF protection support for outgoing requests.
     *
     * For a server that supports a cookie-based XSRF protection system,
     * use directly to configure XSRF protection with the correct
     * cookie and header names.
     *
     * If no names are supplied, the default cookie name is `XSRF-TOKEN`
     * and the default header name is `X-XSRF-TOKEN`.
     *
     * \@publicApi
     */
    class HttpClientXsrfModule {
        /**
         * Disable the default XSRF protection.
         * @return {?}
         */
        static disable() {
            return {
                ngModule: HttpClientXsrfModule,
                providers: [
                    { provide: HttpXsrfInterceptor, useClass: NoopInterceptor },
                ],
            };
        }
        /**
         * Configure XSRF protection.
         * @param {?=} options An object that can specify either or both
         * cookie name or header name.
         * - Cookie name default is `XSRF-TOKEN`.
         * - Header name default is `X-XSRF-TOKEN`.
         *
         * @return {?}
         */
        static withOptions(options = {}) {
            return {
                ngModule: HttpClientXsrfModule,
                providers: [
                    options.cookieName ? { provide: XSRF_COOKIE_NAME, useValue: options.cookieName } : [],
                    options.headerName ? { provide: XSRF_HEADER_NAME, useValue: options.headerName } : [],
                ],
            };
        }
    }
    HttpClientXsrfModule.ɵmod = ɵɵdefineNgModule({ type: HttpClientXsrfModule });
    HttpClientXsrfModule.ɵinj = ɵɵdefineInjector({ factory: function HttpClientXsrfModule_Factory(t) { return new (t || HttpClientXsrfModule)(); }, providers: [
            HttpXsrfInterceptor,
            { provide: HTTP_INTERCEPTORS, useExisting: HttpXsrfInterceptor, multi: true },
            { provide: HttpXsrfTokenExtractor, useClass: HttpXsrfCookieExtractor },
            { provide: XSRF_COOKIE_NAME, useValue: 'XSRF-TOKEN' },
            { provide: XSRF_HEADER_NAME, useValue: 'X-XSRF-TOKEN' },
        ] });
    /**
     * Configures the [dependency injector](guide/glossary#injector) for `HttpClient`
     * with supporting services for XSRF. Automatically imported by `HttpClientModule`.
     *
     * You can add interceptors to the chain behind `HttpClient` by binding them to the
     * multiprovider for built-in [DI token](guide/glossary#di-token) `HTTP_INTERCEPTORS`.
     *
     * \@publicApi
     */
    class HttpClientModule {
    }
    HttpClientModule.ɵmod = ɵɵdefineNgModule({ type: HttpClientModule });
    HttpClientModule.ɵinj = ɵɵdefineInjector({ factory: function HttpClientModule_Factory(t) { return new (t || HttpClientModule)(); }, providers: [
            HttpClient,
            { provide: HttpHandler, useClass: HttpInterceptingHandler },
            HttpXhrBackend,
            { provide: HttpBackend, useExisting: HttpXhrBackend },
            BrowserXhr,
            { provide: XhrFactory, useExisting: BrowserXhr },
        ], imports: [[
                HttpClientXsrfModule.withOptions({
                    cookieName: 'XSRF-TOKEN',
                    headerName: 'X-XSRF-TOKEN'
                }),
            ]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(HttpClientModule, { imports: [HttpClientXsrfModule] }); })();
    /**
     * Configures the [dependency injector](guide/glossary#injector) for `HttpClient`
     * with supporting services for JSONP.
     * Without this module, Jsonp requests reach the backend
     * with method JSONP, where they are rejected.
     *
     * You can add interceptors to the chain behind `HttpClient` by binding them to the
     * multiprovider for built-in [DI token](guide/glossary#di-token) `HTTP_INTERCEPTORS`.
     *
     * \@publicApi
     */
    class HttpClientJsonpModule {
    }
    HttpClientJsonpModule.ɵmod = ɵɵdefineNgModule({ type: HttpClientJsonpModule });
    HttpClientJsonpModule.ɵinj = ɵɵdefineInjector({ factory: function HttpClientJsonpModule_Factory(t) { return new (t || HttpClientJsonpModule)(); }, providers: [
            JsonpClientBackend,
            { provide: JsonpCallbackContext, useFactory: jsonpCallbackContext },
            { provide: HTTP_INTERCEPTORS, useClass: JsonpInterceptor, multi: true },
        ] });

    const _c0$7 = ["*"];
    function getMatIconNameNotFoundError(iconName) {
        return Error(`Unable to find icon with the name "${iconName}"`);
    }
    /**
     * Returns an exception to be thrown when the consumer attempts to use
     * `<mat-icon>` without including \@angular/common/http.
     * \@docs-private
     * @return {?}
     */
    function getMatIconNoHttpProviderError() {
        return Error('Could not find HttpClient provider for use with Angular Material icons. ' +
            'Please include the HttpClientModule from @angular/common/http in your ' +
            'app imports.');
    }
    /**
     * Returns an exception to be thrown when a URL couldn't be sanitized.
     * \@docs-private
     * @param {?} url URL that was attempted to be sanitized.
     * @return {?}
     */
    function getMatIconFailedToSanitizeUrlError(url) {
        return Error(`The URL provided to MatIconRegistry was not trusted as a resource URL ` +
            `via Angular's DomSanitizer. Attempted URL was "${url}".`);
    }
    /**
     * Returns an exception to be thrown when a HTML string couldn't be sanitized.
     * \@docs-private
     * @param {?} literal HTML that was attempted to be sanitized.
     * @return {?}
     */
    function getMatIconFailedToSanitizeLiteralError(literal) {
        return Error(`The literal provided to MatIconRegistry was not trusted as safe HTML by ` +
            `Angular's DomSanitizer. Attempted literal was "${literal}".`);
    }
    /**
     * Configuration for an icon, including the URL and possibly the cached SVG element.
     * \@docs-private
     */
    class SvgIconConfig {
        /**
         * @param {?} data
         * @param {?=} options
         */
        constructor(data, options) {
            this.options = options;
            // Note that we can't use `instanceof SVGElement` here,
            // because it'll break during server-side rendering.
            if (!!((/** @type {?} */ (data))).nodeName) {
                this.svgElement = (/** @type {?} */ (data));
            }
            else {
                this.url = (/** @type {?} */ (data));
            }
        }
    }
    /**
     * Service to register and display icons used by the `<mat-icon>` component.
     * - Registers icon URLs by namespace and name.
     * - Registers icon set URLs by namespace.
     * - Registers aliases for CSS classes, for use with icon fonts.
     * - Loads icons from URLs and extracts individual icons from icon sets.
     */
    class MatIconRegistry {
        /**
         * @param {?} _httpClient
         * @param {?} _sanitizer
         * @param {?} document
         * @param {?=} _errorHandler
         */
        constructor(_httpClient, _sanitizer, document, _errorHandler) {
            this._httpClient = _httpClient;
            this._sanitizer = _sanitizer;
            this._errorHandler = _errorHandler;
            /**
             * URLs and cached SVG elements for individual icons. Keys are of the format "[namespace]:[icon]".
             */
            this._svgIconConfigs = new Map();
            /**
             * SvgIconConfig objects and cached SVG elements for icon sets, keyed by namespace.
             * Multiple icon sets can be registered under the same namespace.
             */
            this._iconSetConfigs = new Map();
            /**
             * Cache for icons loaded by direct URLs.
             */
            this._cachedIconsByUrl = new Map();
            /**
             * In-progress icon fetches. Used to coalesce multiple requests to the same URL.
             */
            this._inProgressUrlFetches = new Map();
            /**
             * Map from font identifiers to their CSS class names. Used for icon fonts.
             */
            this._fontCssClassesByAlias = new Map();
            /**
             * The CSS class to apply when an `<mat-icon>` component has no icon name, url, or font specified.
             * The default 'material-icons' value assumes that the material icon font has been loaded as
             * described at http://google.github.io/material-design-icons/#icon-font-for-the-web
             */
            this._defaultFontSetClass = 'material-icons';
            this._document = document;
        }
        /**
         * Registers an icon by URL in the default namespace.
         * @template THIS
         * @this {THIS}
         * @param {?} iconName Name under which the icon should be registered.
         * @param {?} url
         * @param {?=} options
         * @return {THIS}
         */
        addSvgIcon(iconName, url, options) {
            return (/** @type {?} */ (this)).addSvgIconInNamespace('', iconName, url, options);
        }
        /**
         * Registers an icon using an HTML string in the default namespace.
         * @template THIS
         * @this {THIS}
         * @param {?} iconName Name under which the icon should be registered.
         * @param {?} literal SVG source of the icon.
         * @param {?=} options
         * @return {THIS}
         */
        addSvgIconLiteral(iconName, literal, options) {
            return (/** @type {?} */ (this)).addSvgIconLiteralInNamespace('', iconName, literal, options);
        }
        /**
         * Registers an icon by URL in the specified namespace.
         * @template THIS
         * @this {THIS}
         * @param {?} namespace Namespace in which the icon should be registered.
         * @param {?} iconName Name under which the icon should be registered.
         * @param {?} url
         * @param {?=} options
         * @return {THIS}
         */
        addSvgIconInNamespace(namespace, iconName, url, options) {
            return (/** @type {?} */ (this))._addSvgIconConfig(namespace, iconName, new SvgIconConfig(url, options));
        }
        /**
         * Registers an icon using an HTML string in the specified namespace.
         * @template THIS
         * @this {THIS}
         * @param {?} namespace Namespace in which the icon should be registered.
         * @param {?} iconName Name under which the icon should be registered.
         * @param {?} literal SVG source of the icon.
         * @param {?=} options
         * @return {THIS}
         */
        addSvgIconLiteralInNamespace(namespace, iconName, literal, options) {
            /** @type {?} */
            const sanitizedLiteral = (/** @type {?} */ (this))._sanitizer.sanitize(SecurityContext.HTML, literal);
            if (!sanitizedLiteral) {
                throw getMatIconFailedToSanitizeLiteralError(literal);
            }
            /** @type {?} */
            const svgElement = (/** @type {?} */ (this))._createSvgElementForSingleIcon(sanitizedLiteral, options);
            return (/** @type {?} */ (this))._addSvgIconConfig(namespace, iconName, new SvgIconConfig(svgElement, options));
        }
        /**
         * Registers an icon set by URL in the default namespace.
         * @template THIS
         * @this {THIS}
         * @param {?} url
         * @param {?=} options
         * @return {THIS}
         */
        addSvgIconSet(url, options) {
            return (/** @type {?} */ (this)).addSvgIconSetInNamespace('', url, options);
        }
        /**
         * Registers an icon set using an HTML string in the default namespace.
         * @template THIS
         * @this {THIS}
         * @param {?} literal SVG source of the icon set.
         * @param {?=} options
         * @return {THIS}
         */
        addSvgIconSetLiteral(literal, options) {
            return (/** @type {?} */ (this)).addSvgIconSetLiteralInNamespace('', literal, options);
        }
        /**
         * Registers an icon set by URL in the specified namespace.
         * @template THIS
         * @this {THIS}
         * @param {?} namespace Namespace in which to register the icon set.
         * @param {?} url
         * @param {?=} options
         * @return {THIS}
         */
        addSvgIconSetInNamespace(namespace, url, options) {
            return (/** @type {?} */ (this))._addSvgIconSetConfig(namespace, new SvgIconConfig(url, options));
        }
        /**
         * Registers an icon set using an HTML string in the specified namespace.
         * @template THIS
         * @this {THIS}
         * @param {?} namespace Namespace in which to register the icon set.
         * @param {?} literal SVG source of the icon set.
         * @param {?=} options
         * @return {THIS}
         */
        addSvgIconSetLiteralInNamespace(namespace, literal, options) {
            /** @type {?} */
            const sanitizedLiteral = (/** @type {?} */ (this))._sanitizer.sanitize(SecurityContext.HTML, literal);
            if (!sanitizedLiteral) {
                throw getMatIconFailedToSanitizeLiteralError(literal);
            }
            /** @type {?} */
            const svgElement = (/** @type {?} */ (this))._svgElementFromString(sanitizedLiteral);
            return (/** @type {?} */ (this))._addSvgIconSetConfig(namespace, new SvgIconConfig(svgElement, options));
        }
        /**
         * Defines an alias for a CSS class name to be used for icon fonts. Creating an matIcon
         * component with the alias as the fontSet input will cause the class name to be applied
         * to the `<mat-icon>` element.
         *
         * @template THIS
         * @this {THIS}
         * @param {?} alias Alias for the font.
         * @param {?=} className Class name override to be used instead of the alias.
         * @return {THIS}
         */
        registerFontClassAlias(alias, className = alias) {
            (/** @type {?} */ (this))._fontCssClassesByAlias.set(alias, className);
            return (/** @type {?} */ (this));
        }
        /**
         * Returns the CSS class name associated with the alias by a previous call to
         * registerFontClassAlias. If no CSS class has been associated, returns the alias unmodified.
         * @param {?} alias
         * @return {?}
         */
        classNameForFontAlias(alias) {
            return this._fontCssClassesByAlias.get(alias) || alias;
        }
        /**
         * Sets the CSS class name to be used for icon fonts when an `<mat-icon>` component does not
         * have a fontSet input value, and is not loading an icon by name or URL.
         *
         * @template THIS
         * @this {THIS}
         * @param {?} className
         * @return {THIS}
         */
        setDefaultFontSetClass(className) {
            (/** @type {?} */ (this))._defaultFontSetClass = className;
            return (/** @type {?} */ (this));
        }
        /**
         * Returns the CSS class name to be used for icon fonts when an `<mat-icon>` component does not
         * have a fontSet input value, and is not loading an icon by name or URL.
         * @return {?}
         */
        getDefaultFontSetClass() {
            return this._defaultFontSetClass;
        }
        /**
         * Returns an Observable that produces the icon (as an `<svg>` DOM element) from the given URL.
         * The response from the URL may be cached so this will not always cause an HTTP request, but
         * the produced element will always be a new copy of the originally fetched icon. (That is,
         * it will not contain any modifications made to elements previously returned).
         *
         * @param {?} safeUrl URL from which to fetch the SVG icon.
         * @return {?}
         */
        getSvgIconFromUrl(safeUrl) {
            /** @type {?} */
            const url = this._sanitizer.sanitize(SecurityContext.RESOURCE_URL, safeUrl);
            if (!url) {
                throw getMatIconFailedToSanitizeUrlError(safeUrl);
            }
            /** @type {?} */
            const cachedIcon = this._cachedIconsByUrl.get(url);
            if (cachedIcon) {
                return of(cloneSvg(cachedIcon));
            }
            return this._loadSvgIconFromConfig(new SvgIconConfig(safeUrl)).pipe(tap((/**
             * @param {?} svg
             * @return {?}
             */
            svg => this._cachedIconsByUrl.set((/** @type {?} */ (url)), svg))), map((/**
             * @param {?} svg
             * @return {?}
             */
            svg => cloneSvg(svg))));
        }
        /**
         * Returns an Observable that produces the icon (as an `<svg>` DOM element) with the given name
         * and namespace. The icon must have been previously registered with addIcon or addIconSet;
         * if not, the Observable will throw an error.
         *
         * @param {?} name Name of the icon to be retrieved.
         * @param {?=} namespace Namespace in which to look for the icon.
         * @return {?}
         */
        getNamedSvgIcon(name, namespace = '') {
            // Return (copy of) cached icon if possible.
            /** @type {?} */
            const key = iconKey(namespace, name);
            /** @type {?} */
            const config = this._svgIconConfigs.get(key);
            if (config) {
                return this._getSvgFromConfig(config);
            }
            // See if we have any icon sets registered for the namespace.
            /** @type {?} */
            const iconSetConfigs = this._iconSetConfigs.get(namespace);
            if (iconSetConfigs) {
                return this._getSvgFromIconSetConfigs(name, iconSetConfigs);
            }
            return throwError(getMatIconNameNotFoundError(key));
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this._svgIconConfigs.clear();
            this._iconSetConfigs.clear();
            this._cachedIconsByUrl.clear();
        }
        /**
         * Returns the cached icon for a SvgIconConfig if available, or fetches it from its URL if not.
         * @private
         * @param {?} config
         * @return {?}
         */
        _getSvgFromConfig(config) {
            if (config.svgElement) {
                // We already have the SVG element for this icon, return a copy.
                return of(cloneSvg(config.svgElement));
            }
            else {
                // Fetch the icon from the config's URL, cache it, and return a copy.
                return this._loadSvgIconFromConfig(config).pipe(tap((/**
                 * @param {?} svg
                 * @return {?}
                 */
                svg => config.svgElement = svg)), map((/**
                 * @param {?} svg
                 * @return {?}
                 */
                svg => cloneSvg(svg))));
            }
        }
        /**
         * Attempts to find an icon with the specified name in any of the SVG icon sets.
         * First searches the available cached icons for a nested element with a matching name, and
         * if found copies the element to a new `<svg>` element. If not found, fetches all icon sets
         * that have not been cached, and searches again after all fetches are completed.
         * The returned Observable produces the SVG element if possible, and throws
         * an error if no icon with the specified name can be found.
         * @private
         * @param {?} name
         * @param {?} iconSetConfigs
         * @return {?}
         */
        _getSvgFromIconSetConfigs(name, iconSetConfigs) {
            // For all the icon set SVG elements we've fetched, see if any contain an icon with the
            // requested name.
            /** @type {?} */
            const namedIcon = this._extractIconWithNameFromAnySet(name, iconSetConfigs);
            if (namedIcon) {
                // We could cache namedIcon in _svgIconConfigs, but since we have to make a copy every
                // time anyway, there's probably not much advantage compared to just always extracting
                // it from the icon set.
                return of(namedIcon);
            }
            // Not found in any cached icon sets. If there are icon sets with URLs that we haven't
            // fetched, fetch them now and look for iconName in the results.
            /** @type {?} */
            const iconSetFetchRequests = iconSetConfigs
                .filter((/**
             * @param {?} iconSetConfig
             * @return {?}
             */
            iconSetConfig => !iconSetConfig.svgElement))
                .map((/**
             * @param {?} iconSetConfig
             * @return {?}
             */
            iconSetConfig => {
                return this._loadSvgIconSetFromConfig(iconSetConfig).pipe(catchError((/**
                 * @param {?} err
                 * @return {?}
                 */
                (err) => {
                    /** @type {?} */
                    const url = this._sanitizer.sanitize(SecurityContext.RESOURCE_URL, iconSetConfig.url);
                    // Swallow errors fetching individual URLs so the
                    // combined Observable won't necessarily fail.
                    /** @type {?} */
                    const errorMessage = `Loading icon set URL: ${url} failed: ${err.message}`;
                    // @breaking-change 9.0.0 _errorHandler parameter to be made required
                    if (this._errorHandler) {
                        this._errorHandler.handleError(new Error(errorMessage));
                    }
                    else {
                        console.error(errorMessage);
                    }
                    return of(null);
                })));
            }));
            // Fetch all the icon set URLs. When the requests complete, every IconSet should have a
            // cached SVG element (unless the request failed), and we can check again for the icon.
            return forkJoin(iconSetFetchRequests).pipe(map((/**
             * @return {?}
             */
            () => {
                /** @type {?} */
                const foundIcon = this._extractIconWithNameFromAnySet(name, iconSetConfigs);
                if (!foundIcon) {
                    throw getMatIconNameNotFoundError(name);
                }
                return foundIcon;
            })));
        }
        /**
         * Searches the cached SVG elements for the given icon sets for a nested icon element whose "id"
         * tag matches the specified name. If found, copies the nested element to a new SVG element and
         * returns it. Returns null if no matching element is found.
         * @private
         * @param {?} iconName
         * @param {?} iconSetConfigs
         * @return {?}
         */
        _extractIconWithNameFromAnySet(iconName, iconSetConfigs) {
            // Iterate backwards, so icon sets added later have precedence.
            for (let i = iconSetConfigs.length - 1; i >= 0; i--) {
                /** @type {?} */
                const config = iconSetConfigs[i];
                if (config.svgElement) {
                    /** @type {?} */
                    const foundIcon = this._extractSvgIconFromSet(config.svgElement, iconName, config.options);
                    if (foundIcon) {
                        return foundIcon;
                    }
                }
            }
            return null;
        }
        /**
         * Loads the content of the icon URL specified in the SvgIconConfig and creates an SVG element
         * from it.
         * @private
         * @param {?} config
         * @return {?}
         */
        _loadSvgIconFromConfig(config) {
            return this._fetchUrl(config.url)
                .pipe(map((/**
             * @param {?} svgText
             * @return {?}
             */
            svgText => this._createSvgElementForSingleIcon(svgText, config.options))));
        }
        /**
         * Loads the content of the icon set URL specified in the SvgIconConfig and creates an SVG element
         * from it.
         * @private
         * @param {?} config
         * @return {?}
         */
        _loadSvgIconSetFromConfig(config) {
            // If the SVG for this icon set has already been parsed, do nothing.
            if (config.svgElement) {
                return of(config.svgElement);
            }
            return this._fetchUrl(config.url).pipe(map((/**
             * @param {?} svgText
             * @return {?}
             */
            svgText => {
                // It is possible that the icon set was parsed and cached by an earlier request, so parsing
                // only needs to occur if the cache is yet unset.
                if (!config.svgElement) {
                    config.svgElement = this._svgElementFromString(svgText);
                }
                return config.svgElement;
            })));
        }
        /**
         * Creates a DOM element from the given SVG string, and adds default attributes.
         * @private
         * @param {?} responseText
         * @param {?=} options
         * @return {?}
         */
        _createSvgElementForSingleIcon(responseText, options) {
            /** @type {?} */
            const svg = this._svgElementFromString(responseText);
            this._setSvgAttributes(svg, options);
            return svg;
        }
        /**
         * Searches the cached element of the given SvgIconConfig for a nested icon element whose "id"
         * tag matches the specified name. If found, copies the nested element to a new SVG element and
         * returns it. Returns null if no matching element is found.
         * @private
         * @param {?} iconSet
         * @param {?} iconName
         * @param {?=} options
         * @return {?}
         */
        _extractSvgIconFromSet(iconSet, iconName, options) {
            // Use the `id="iconName"` syntax in order to escape special
            // characters in the ID (versus using the #iconName syntax).
            /** @type {?} */
            const iconSource = iconSet.querySelector(`[id="${iconName}"]`);
            if (!iconSource) {
                return null;
            }
            // Clone the element and remove the ID to prevent multiple elements from being added
            // to the page with the same ID.
            /** @type {?} */
            const iconElement = (/** @type {?} */ (iconSource.cloneNode(true)));
            iconElement.removeAttribute('id');
            // If the icon node is itself an <svg> node, clone and return it directly. If not, set it as
            // the content of a new <svg> node.
            if (iconElement.nodeName.toLowerCase() === 'svg') {
                return this._setSvgAttributes((/** @type {?} */ (iconElement)), options);
            }
            // If the node is a <symbol>, it won't be rendered so we have to convert it into <svg>. Note
            // that the same could be achieved by referring to it via <use href="#id">, however the <use>
            // tag is problematic on Firefox, because it needs to include the current page path.
            if (iconElement.nodeName.toLowerCase() === 'symbol') {
                return this._setSvgAttributes(this._toSvgElement(iconElement), options);
            }
            // createElement('SVG') doesn't work as expected; the DOM ends up with
            // the correct nodes, but the SVG content doesn't render. Instead we
            // have to create an empty SVG node using innerHTML and append its content.
            // Elements created using DOMParser.parseFromString have the same problem.
            // http://stackoverflow.com/questions/23003278/svg-innerhtml-in-firefox-can-not-display
            /** @type {?} */
            const svg = this._svgElementFromString('<svg></svg>');
            // Clone the node so we don't remove it from the parent icon set element.
            svg.appendChild(iconElement);
            return this._setSvgAttributes(svg, options);
        }
        /**
         * Creates a DOM element from the given SVG string.
         * @private
         * @param {?} str
         * @return {?}
         */
        _svgElementFromString(str) {
            /** @type {?} */
            const div = this._document.createElement('DIV');
            div.innerHTML = str;
            /** @type {?} */
            const svg = (/** @type {?} */ (div.querySelector('svg')));
            if (!svg) {
                throw Error('<svg> tag not found');
            }
            return svg;
        }
        /**
         * Converts an element into an SVG node by cloning all of its children.
         * @private
         * @param {?} element
         * @return {?}
         */
        _toSvgElement(element) {
            /** @type {?} */
            const svg = this._svgElementFromString('<svg></svg>');
            /** @type {?} */
            const attributes = element.attributes;
            // Copy over all the attributes from the `symbol` to the new SVG, except the id.
            for (let i = 0; i < attributes.length; i++) {
                const { name, value } = attributes[i];
                if (name !== 'id') {
                    svg.setAttribute(name, value);
                }
            }
            for (let i = 0; i < element.childNodes.length; i++) {
                if (element.childNodes[i].nodeType === this._document.ELEMENT_NODE) {
                    svg.appendChild(element.childNodes[i].cloneNode(true));
                }
            }
            return svg;
        }
        /**
         * Sets the default attributes for an SVG element to be used as an icon.
         * @private
         * @param {?} svg
         * @param {?=} options
         * @return {?}
         */
        _setSvgAttributes(svg, options) {
            svg.setAttribute('fit', '');
            svg.setAttribute('height', '100%');
            svg.setAttribute('width', '100%');
            svg.setAttribute('preserveAspectRatio', 'xMidYMid meet');
            svg.setAttribute('focusable', 'false'); // Disable IE11 default behavior to make SVGs focusable.
            if (options && options.viewBox) {
                svg.setAttribute('viewBox', options.viewBox);
            }
            return svg;
        }
        /**
         * Returns an Observable which produces the string contents of the given URL. Results may be
         * cached, so future calls with the same URL may not cause another HTTP request.
         * @private
         * @param {?} safeUrl
         * @return {?}
         */
        _fetchUrl(safeUrl) {
            if (!this._httpClient) {
                throw getMatIconNoHttpProviderError();
            }
            if (safeUrl == null) {
                throw Error(`Cannot fetch icon from URL "${safeUrl}".`);
            }
            /** @type {?} */
            const url = this._sanitizer.sanitize(SecurityContext.RESOURCE_URL, safeUrl);
            if (!url) {
                throw getMatIconFailedToSanitizeUrlError(safeUrl);
            }
            // Store in-progress fetches to avoid sending a duplicate request for a URL when there is
            // already a request in progress for that URL. It's necessary to call share() on the
            // Observable returned by http.get() so that multiple subscribers don't cause multiple XHRs.
            /** @type {?} */
            const inProgressFetch = this._inProgressUrlFetches.get(url);
            if (inProgressFetch) {
                return inProgressFetch;
            }
            // TODO(jelbourn): for some reason, the `finalize` operator "loses" the generic type on the
            // Observable. Figure out why and fix it.
            /** @type {?} */
            const req = this._httpClient.get(url, { responseType: 'text' }).pipe(finalize((/**
             * @return {?}
             */
            () => this._inProgressUrlFetches.delete(url))), share());
            this._inProgressUrlFetches.set(url, req);
            return req;
        }
        /**
         * Registers an icon config by name in the specified namespace.
         * @private
         * @template THIS
         * @this {THIS}
         * @param {?} namespace Namespace in which to register the icon config.
         * @param {?} iconName Name under which to register the config.
         * @param {?} config Config to be registered.
         * @return {THIS}
         */
        _addSvgIconConfig(namespace, iconName, config) {
            (/** @type {?} */ (this))._svgIconConfigs.set(iconKey(namespace, iconName), config);
            return (/** @type {?} */ (this));
        }
        /**
         * Registers an icon set config in the specified namespace.
         * @private
         * @template THIS
         * @this {THIS}
         * @param {?} namespace Namespace in which to register the icon config.
         * @param {?} config Config to be registered.
         * @return {THIS}
         */
        _addSvgIconSetConfig(namespace, config) {
            /** @type {?} */
            const configNamespace = (/** @type {?} */ (this))._iconSetConfigs.get(namespace);
            if (configNamespace) {
                configNamespace.push(config);
            }
            else {
                (/** @type {?} */ (this))._iconSetConfigs.set(namespace, [config]);
            }
            return (/** @type {?} */ (this));
        }
    }
    MatIconRegistry.ɵfac = function MatIconRegistry_Factory(t) { return new (t || MatIconRegistry)(ɵɵinject(HttpClient, 8), ɵɵinject(DomSanitizer), ɵɵinject(DOCUMENT$1, 8), ɵɵinject(ErrorHandler, 8)); };
    /** @nocollapse */
    MatIconRegistry.ctorParameters = () => [
        { type: HttpClient, decorators: [{ type: Optional }] },
        { type: DomSanitizer },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [DOCUMENT$1,] }] },
        { type: ErrorHandler, decorators: [{ type: Optional }] }
    ];
    /** @nocollapse */ MatIconRegistry.ɵprov = ɵɵdefineInjectable({ factory: function MatIconRegistry_Factory() { return new MatIconRegistry(ɵɵinject(HttpClient, 8), ɵɵinject(DomSanitizer), ɵɵinject(DOCUMENT$1, 8), ɵɵinject(ErrorHandler, 8)); }, token: MatIconRegistry, providedIn: "root" });
    /**
     * \@docs-private
     * @param {?} parentRegistry
     * @param {?} httpClient
     * @param {?} sanitizer
     * @param {?=} document
     * @param {?=} errorHandler
     * @return {?}
     */
    function ICON_REGISTRY_PROVIDER_FACTORY(parentRegistry, httpClient, sanitizer, document, errorHandler) {
        return parentRegistry || new MatIconRegistry(httpClient, sanitizer, document, errorHandler);
    }
    /**
     * \@docs-private
     * @type {?}
     */
    const ICON_REGISTRY_PROVIDER = {
        // If there is already an MatIconRegistry available, use that. Otherwise, provide a new one.
        provide: MatIconRegistry,
        deps: [
            [new Optional(), new SkipSelf(), MatIconRegistry],
            [new Optional(), HttpClient],
            DomSanitizer,
            [new Optional(), ErrorHandler],
            [new Optional(), (/** @type {?} */ (DOCUMENT$1))],
        ],
        useFactory: ICON_REGISTRY_PROVIDER_FACTORY,
    };
    /**
     * Clones an SVGElement while preserving type information.
     * @param {?} svg
     * @return {?}
     */
    function cloneSvg(svg) {
        return (/** @type {?} */ (svg.cloneNode(true)));
    }
    /**
     * Returns the cache key to use for an icon namespace and name.
     * @param {?} namespace
     * @param {?} name
     * @return {?}
     */
    function iconKey(namespace, name) {
        return namespace + ':' + name;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/icon/icon.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    // Boilerplate for applying mixins to MatIcon.
    /**
     * \@docs-private
     */
    class MatIconBase {
        /**
         * @param {?} _elementRef
         */
        constructor(_elementRef) {
            this._elementRef = _elementRef;
        }
    }
    /** @type {?} */
    const _MatIconMixinBase = mixinColor(MatIconBase);
    /**
     * Injection token used to provide the current location to `MatIcon`.
     * Used to handle server-side rendering and to stub out during unit tests.
     * \@docs-private
     * @type {?}
     */
    const MAT_ICON_LOCATION = new InjectionToken('mat-icon-location', {
        providedIn: 'root',
        factory: MAT_ICON_LOCATION_FACTORY
    });
    /**
     * \@docs-private
     * @return {?}
     */
    function MAT_ICON_LOCATION_FACTORY() {
        /** @type {?} */
        const _document = inject(DOCUMENT$1);
        /** @type {?} */
        const _location = _document ? _document.location : null;
        return {
            // Note that this needs to be a function, rather than a property, because Angular
            // will only resolve it once, but we want the current path on each call.
            getPathname: (/**
             * @return {?}
             */
            () => _location ? (_location.pathname + _location.search) : '')
        };
    }
    /**
     * SVG attributes that accept a FuncIRI (e.g. `url(<something>)`).
     * @type {?}
     */
    const funcIriAttributes = [
        'clip-path',
        'color-profile',
        'src',
        'cursor',
        'fill',
        'filter',
        'marker',
        'marker-start',
        'marker-mid',
        'marker-end',
        'mask',
        'stroke'
    ];
    const ɵ0$c = /**
     * @param {?} attr
     * @return {?}
     */
    attr => `[${attr}]`;
    /**
     * Selector that can be used to find all elements that are using a `FuncIRI`.
     * @type {?}
     */
    const funcIriAttributeSelector = funcIriAttributes.map((ɵ0$c)).join(', ');
    /**
     * Regex that can be used to extract the id out of a FuncIRI.
     * @type {?}
     */
    const funcIriPattern = /^url\(['"]?#(.*?)['"]?\)$/;
    /**
     * Component to display an icon. It can be used in the following ways:
     *
     * - Specify the svgIcon input to load an SVG icon from a URL previously registered with the
     *   addSvgIcon, addSvgIconInNamespace, addSvgIconSet, or addSvgIconSetInNamespace methods of
     *   MatIconRegistry. If the svgIcon value contains a colon it is assumed to be in the format
     *   "[namespace]:[name]", if not the value will be the name of an icon in the default namespace.
     *   Examples:
     *     `<mat-icon svgIcon="left-arrow"></mat-icon>
     *     <mat-icon svgIcon="animals:cat"></mat-icon>`
     *
     * - Use a font ligature as an icon by putting the ligature text in the content of the `<mat-icon>`
     *   component. By default the Material icons font is used as described at
     *   http://google.github.io/material-design-icons/#icon-font-for-the-web. You can specify an
     *   alternate font by setting the fontSet input to either the CSS class to apply to use the
     *   desired font, or to an alias previously registered with MatIconRegistry.registerFontClassAlias.
     *   Examples:
     *     `<mat-icon>home</mat-icon>
     *     <mat-icon fontSet="myfont">sun</mat-icon>`
     *
     * - Specify a font glyph to be included via CSS rules by setting the fontSet input to specify the
     *   font, and the fontIcon input to specify the icon. Typically the fontIcon will specify a
     *   CSS class which causes the glyph to be displayed via a :before selector, as in
     *   https://fortawesome.github.io/Font-Awesome/examples/
     *   Example:
     *     `<mat-icon fontSet="fa" fontIcon="alarm"></mat-icon>`
     */
    class MatIcon extends _MatIconMixinBase {
        /**
         * @param {?} elementRef
         * @param {?} _iconRegistry
         * @param {?} ariaHidden
         * @param {?=} _location
         * @param {?=} _errorHandler
         */
        constructor(elementRef, _iconRegistry, ariaHidden, _location, _errorHandler) {
            super(elementRef);
            this._iconRegistry = _iconRegistry;
            this._location = _location;
            this._errorHandler = _errorHandler;
            this._inline = false;
            /**
             * Subscription to the current in-progress SVG icon request.
             */
            this._currentIconFetch = Subscription.EMPTY;
            // If the user has not explicitly set aria-hidden, mark the icon as hidden, as this is
            // the right thing to do for the majority of icon use-cases.
            if (!ariaHidden) {
                elementRef.nativeElement.setAttribute('aria-hidden', 'true');
            }
        }
        /**
         * Whether the icon should be inlined, automatically sizing the icon to match the font size of
         * the element the icon is contained in.
         * @return {?}
         */
        get inline() {
            return this._inline;
        }
        /**
         * @param {?} inline
         * @return {?}
         */
        set inline(inline) {
            this._inline = coerceBooleanProperty(inline);
        }
        /**
         * Font set that the icon is a part of.
         * @return {?}
         */
        get fontSet() { return this._fontSet; }
        /**
         * @param {?} value
         * @return {?}
         */
        set fontSet(value) {
            this._fontSet = this._cleanupFontValue(value);
        }
        /**
         * Name of an icon within a font set.
         * @return {?}
         */
        get fontIcon() { return this._fontIcon; }
        /**
         * @param {?} value
         * @return {?}
         */
        set fontIcon(value) {
            this._fontIcon = this._cleanupFontValue(value);
        }
        /**
         * Splits an svgIcon binding value into its icon set and icon name components.
         * Returns a 2-element array of [(icon set), (icon name)].
         * The separator for the two fields is ':'. If there is no separator, an empty
         * string is returned for the icon set and the entire value is returned for
         * the icon name. If the argument is falsy, returns an array of two empty strings.
         * Throws an error if the name contains two or more ':' separators.
         * Examples:
         *   `'social:cake' -> ['social', 'cake']
         *   'penguin' -> ['', 'penguin']
         *   null -> ['', '']
         *   'a:b:c' -> (throws Error)`
         * @private
         * @param {?} iconName
         * @return {?}
         */
        _splitIconName(iconName) {
            if (!iconName) {
                return ['', ''];
            }
            /** @type {?} */
            const parts = iconName.split(':');
            switch (parts.length) {
                case 1: return ['', parts[0]]; // Use default namespace.
                case 2: return (/** @type {?} */ (parts));
                default: throw Error(`Invalid icon name: "${iconName}"`);
            }
        }
        /**
         * @param {?} changes
         * @return {?}
         */
        ngOnChanges(changes) {
            // Only update the inline SVG icon if the inputs changed, to avoid unnecessary DOM operations.
            /** @type {?} */
            const svgIconChanges = changes['svgIcon'];
            if (svgIconChanges) {
                this._currentIconFetch.unsubscribe();
                if (this.svgIcon) {
                    const [namespace, iconName] = this._splitIconName(this.svgIcon);
                    this._currentIconFetch = this._iconRegistry.getNamedSvgIcon(iconName, namespace)
                        .pipe(take(1))
                        .subscribe((/**
                     * @param {?} svg
                     * @return {?}
                     */
                    svg => this._setSvgElement(svg)), (/**
                     * @param {?} err
                     * @return {?}
                     */
                    (err) => {
                        /** @type {?} */
                        const errorMessage = `Error retrieving icon ${namespace}:${iconName}! ${err.message}`;
                        // @breaking-change 9.0.0 _errorHandler parameter to be made required.
                        if (this._errorHandler) {
                            this._errorHandler.handleError(new Error(errorMessage));
                        }
                        else {
                            console.error(errorMessage);
                        }
                    }));
                }
                else if (svgIconChanges.previousValue) {
                    this._clearSvgElement();
                }
            }
            if (this._usingFontIcon()) {
                this._updateFontIconClasses();
            }
        }
        /**
         * @return {?}
         */
        ngOnInit() {
            // Update font classes because ngOnChanges won't be called if none of the inputs are present,
            // e.g. <mat-icon>arrow</mat-icon> In this case we need to add a CSS class for the default font.
            if (this._usingFontIcon()) {
                this._updateFontIconClasses();
            }
        }
        /**
         * @return {?}
         */
        ngAfterViewChecked() {
            /** @type {?} */
            const cachedElements = this._elementsWithExternalReferences;
            if (cachedElements && this._location && cachedElements.size) {
                /** @type {?} */
                const newPath = this._location.getPathname();
                // We need to check whether the URL has changed on each change detection since
                // the browser doesn't have an API that will let us react on link clicks and
                // we can't depend on the Angular router. The references need to be updated,
                // because while most browsers don't care whether the URL is correct after
                // the first render, Safari will break if the user navigates to a different
                // page and the SVG isn't re-rendered.
                if (newPath !== this._previousPath) {
                    this._previousPath = newPath;
                    this._prependPathToReferences(newPath);
                }
            }
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this._currentIconFetch.unsubscribe();
            if (this._elementsWithExternalReferences) {
                this._elementsWithExternalReferences.clear();
            }
        }
        /**
         * @private
         * @return {?}
         */
        _usingFontIcon() {
            return !this.svgIcon;
        }
        /**
         * @private
         * @param {?} svg
         * @return {?}
         */
        _setSvgElement(svg) {
            this._clearSvgElement();
            // Workaround for IE11 and Edge ignoring `style` tags inside dynamically-created SVGs.
            // See: https://developer.microsoft.com/en-us/microsoft-edge/platform/issues/10898469/
            // Do this before inserting the element into the DOM, in order to avoid a style recalculation.
            /** @type {?} */
            const styleTags = (/** @type {?} */ (svg.querySelectorAll('style')));
            for (let i = 0; i < styleTags.length; i++) {
                styleTags[i].textContent += ' ';
            }
            // Note: we do this fix here, rather than the icon registry, because the
            // references have to point to the URL at the time that the icon was created.
            if (this._location) {
                /** @type {?} */
                const path = this._location.getPathname();
                this._previousPath = path;
                this._cacheChildrenWithExternalReferences(svg);
                this._prependPathToReferences(path);
            }
            this._elementRef.nativeElement.appendChild(svg);
        }
        /**
         * @private
         * @return {?}
         */
        _clearSvgElement() {
            /** @type {?} */
            const layoutElement = this._elementRef.nativeElement;
            /** @type {?} */
            let childCount = layoutElement.childNodes.length;
            if (this._elementsWithExternalReferences) {
                this._elementsWithExternalReferences.clear();
            }
            // Remove existing non-element child nodes and SVGs, and add the new SVG element. Note that
            // we can't use innerHTML, because IE will throw if the element has a data binding.
            while (childCount--) {
                /** @type {?} */
                const child = layoutElement.childNodes[childCount];
                // 1 corresponds to Node.ELEMENT_NODE. We remove all non-element nodes in order to get rid
                // of any loose text nodes, as well as any SVG elements in order to remove any old icons.
                if (child.nodeType !== 1 || child.nodeName.toLowerCase() === 'svg') {
                    layoutElement.removeChild(child);
                }
            }
        }
        /**
         * @private
         * @return {?}
         */
        _updateFontIconClasses() {
            if (!this._usingFontIcon()) {
                return;
            }
            /** @type {?} */
            const elem = this._elementRef.nativeElement;
            /** @type {?} */
            const fontSetClass = this.fontSet ?
                this._iconRegistry.classNameForFontAlias(this.fontSet) :
                this._iconRegistry.getDefaultFontSetClass();
            if (fontSetClass != this._previousFontSetClass) {
                if (this._previousFontSetClass) {
                    elem.classList.remove(this._previousFontSetClass);
                }
                if (fontSetClass) {
                    elem.classList.add(fontSetClass);
                }
                this._previousFontSetClass = fontSetClass;
            }
            if (this.fontIcon != this._previousFontIconClass) {
                if (this._previousFontIconClass) {
                    elem.classList.remove(this._previousFontIconClass);
                }
                if (this.fontIcon) {
                    elem.classList.add(this.fontIcon);
                }
                this._previousFontIconClass = this.fontIcon;
            }
        }
        /**
         * Cleans up a value to be used as a fontIcon or fontSet.
         * Since the value ends up being assigned as a CSS class, we
         * have to trim the value and omit space-separated values.
         * @private
         * @param {?} value
         * @return {?}
         */
        _cleanupFontValue(value) {
            return typeof value === 'string' ? value.trim().split(' ')[0] : value;
        }
        /**
         * Prepends the current path to all elements that have an attribute pointing to a `FuncIRI`
         * reference. This is required because WebKit browsers require references to be prefixed with
         * the current path, if the page has a `base` tag.
         * @private
         * @param {?} path
         * @return {?}
         */
        _prependPathToReferences(path) {
            /** @type {?} */
            const elements = this._elementsWithExternalReferences;
            if (elements) {
                elements.forEach((/**
                 * @param {?} attrs
                 * @param {?} element
                 * @return {?}
                 */
                (attrs, element) => {
                    attrs.forEach((/**
                     * @param {?} attr
                     * @return {?}
                     */
                    attr => {
                        element.setAttribute(attr.name, `url('${path}#${attr.value}')`);
                    }));
                }));
            }
        }
        /**
         * Caches the children of an SVG element that have `url()`
         * references that we need to prefix with the current path.
         * @private
         * @param {?} element
         * @return {?}
         */
        _cacheChildrenWithExternalReferences(element) {
            /** @type {?} */
            const elementsWithFuncIri = element.querySelectorAll(funcIriAttributeSelector);
            /** @type {?} */
            const elements = this._elementsWithExternalReferences =
                this._elementsWithExternalReferences || new Map();
            for (let i = 0; i < elementsWithFuncIri.length; i++) {
                funcIriAttributes.forEach((/**
                 * @param {?} attr
                 * @return {?}
                 */
                attr => {
                    /** @type {?} */
                    const elementWithReference = elementsWithFuncIri[i];
                    /** @type {?} */
                    const value = elementWithReference.getAttribute(attr);
                    /** @type {?} */
                    const match = value ? value.match(funcIriPattern) : null;
                    if (match) {
                        /** @type {?} */
                        let attributes = elements.get(elementWithReference);
                        if (!attributes) {
                            attributes = [];
                            elements.set(elementWithReference, attributes);
                        }
                        (/** @type {?} */ (attributes)).push({ name: attr, value: match[1] });
                    }
                }));
            }
        }
    }
    MatIcon.ɵfac = function MatIcon_Factory(t) { return new (t || MatIcon)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(MatIconRegistry), ɵɵinjectAttribute('aria-hidden'), ɵɵdirectiveInject(MAT_ICON_LOCATION, 8), ɵɵdirectiveInject(ErrorHandler, 8)); };
    MatIcon.ɵcmp = ɵɵdefineComponent({ type: MatIcon, selectors: [["mat-icon"]], hostAttrs: ["role", "img", 1, "mat-icon", "notranslate"], hostVars: 4, hostBindings: function MatIcon_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵclassProp("mat-icon-inline", ctx.inline)("mat-icon-no-color", ctx.color !== "primary" && ctx.color !== "accent" && ctx.color !== "warn");
        } }, inputs: { color: "color", inline: "inline", fontSet: "fontSet", fontIcon: "fontIcon", svgIcon: "svgIcon" }, exportAs: ["matIcon"], features: [ɵɵInheritDefinitionFeature, ɵɵNgOnChangesFeature], ngContentSelectors: _c0$7, decls: 1, vars: 0, template: function MatIcon_Template(rf, ctx) { if (rf & 1) {
            ɵɵprojectionDef();
            ɵɵprojection(0);
        } }, styles: [".mat-icon{background-repeat:no-repeat;display:inline-block;fill:currentColor;height:24px;width:24px}.mat-icon.mat-icon-inline{font-size:inherit;height:inherit;line-height:inherit;width:inherit}[dir=rtl] .mat-icon-rtl-mirror{transform:scale(-1, 1)}.mat-form-field:not(.mat-form-field-appearance-legacy) .mat-form-field-prefix .mat-icon,.mat-form-field:not(.mat-form-field-appearance-legacy) .mat-form-field-suffix .mat-icon{display:block}.mat-form-field:not(.mat-form-field-appearance-legacy) .mat-form-field-prefix .mat-icon-button .mat-icon,.mat-form-field:not(.mat-form-field-appearance-legacy) .mat-form-field-suffix .mat-icon-button .mat-icon{margin:auto}\n"], encapsulation: 2, changeDetection: 0 });
    /** @nocollapse */
    MatIcon.ctorParameters = () => [
        { type: ElementRef },
        { type: MatIconRegistry },
        { type: String, decorators: [{ type: Attribute, args: ['aria-hidden',] }] },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [MAT_ICON_LOCATION,] }] },
        { type: ErrorHandler, decorators: [{ type: Optional }] }
    ];
    MatIcon.propDecorators = {
        inline: [{ type: Input }],
        svgIcon: [{ type: Input }],
        fontSet: [{ type: Input }],
        fontIcon: [{ type: Input }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/icon/icon-module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class MatIconModule {
    }
    MatIconModule.ɵmod = ɵɵdefineNgModule({ type: MatIconModule });
    MatIconModule.ɵinj = ɵɵdefineInjector({ factory: function MatIconModule_Factory(t) { return new (t || MatIconModule)(); }, imports: [[MatCommonModule],
            MatCommonModule] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(MatIconModule, { declarations: function () { return [MatIcon]; }, imports: function () { return [MatCommonModule]; }, exports: function () { return [MatIcon,
            MatCommonModule]; } }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/tbdev_upload/tbdev_upload_dialog_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class TbdevUploadDialogComponent {
        /**
         * @param {?} dialogRef
         */
        constructor(dialogRef) {
            this.dialogRef = dialogRef;
            this.tensorboardDotDevUrl = 'https://tensorboard.dev/?utm_source=tensorboard';
        }
        /**
         * @return {?}
         */
        onClose() {
            this.dialogRef.close();
        }
        /**
         * @return {?}
         */
        getCommandText() {
            if (!this.logdir) {
                // Without logdir we print a literal '{logdir}' string that user will
                // have to manually substitute.
                return 'tensorboard dev upload --logdir {logdir}';
            }
            else {
                // With logdir we substitute the value into the command for the user.
                // We assume that logdir is sufficiently long that we want to print it on
                // the next line. If the logdir value is still too long then the CSS will
                // render a scrollbar underneath.
                /** @type {?} */
                const escapedLogdir = this.logdir.replace(/'/g, "'\\''");
                return "tensorboard dev upload --logdir \\\n    '" + escapedLogdir + "'";
            }
        }
    }
    TbdevUploadDialogComponent.decorators = [
        { type: Component, args: [{
                    selector: 'tbdev-upload-dialog-component',
                    templateUrl: './tbdev_upload_dialog_component.ng.html',
                    styleUrls: ['./tbdev_upload_dialog_component.css'],
                },] },
    ];
    /** @nocollapse */
    TbdevUploadDialogComponent.ctorParameters = () => [
        { type: MatDialogRef }
    ];
    TbdevUploadDialogComponent.propDecorators = {
        logdir: [{ type: Input }]
    };
    /** @nocollapse */ TbdevUploadDialogComponent.ɵfac = function TbdevUploadDialogComponent_Factory(t) { return new (t || TbdevUploadDialogComponent)(ɵɵdirectiveInject(MatDialogRef)); };
    /** @nocollapse */ TbdevUploadDialogComponent.ɵcmp = ɵɵdefineComponent({ type: TbdevUploadDialogComponent, selectors: [["tbdev-upload-dialog-component"]], inputs: { logdir: "logdir" }, decls: 21, vars: 4, consts: [["target", "_blank", "rel", "noreferrer noopener", 1, "anchor-text", 3, "href"], [1, "command"], ["mat-icon-button", "", "title", "Click to copy the command", 1, "command-copy", 3, "cdkCopyToClipboard"], ["svgIcon", "content_copy_24px"], [1, "bottom-buttons"], ["mat-flat-button", "", 1, "close-button", 3, "click"], ["mat-flat-button", "", "target", "_blank", "rel", "noreferrer noopener", 1, "learn-more-button", 3, "href"]], template: function TbdevUploadDialogComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "h3");
            ɵɵtext(1, "Upload to TensorBoard.dev");
            ɵɵelementEnd();
            ɵɵelementStart(2, "p");
            ɵɵelementStart(3, "a", 0);
            ɵɵtext(4, " TensorBoard.dev");
            ɵɵelementEnd();
            ɵɵtext(5, " enables you to easily host, track, and share your ML experiments with everyone. You can share a link to the uploaded TensorBoard in papers, blog posts, and social media. This can showcase the results more effectively and helps reproducibility.\n");
            ɵɵelementEnd();
            ɵɵelementStart(6, "p");
            ɵɵtext(7, "To upload a logdir to TensorBoard.dev, run the command:");
            ɵɵelementEnd();
            ɵɵelementStart(8, "div", 1);
            ɵɵelementStart(9, "pre");
            ɵɵelementStart(10, "code");
            ɵɵtext(11);
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementStart(12, "button", 2);
            ɵɵelement(13, "mat-icon", 3);
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementStart(14, "p");
            ɵɵtext(15, " Only certain plugins are currently supported. Uploaded TensorBoards are public and visible to everyone; do not upload sensitive data.\n");
            ɵɵelementEnd();
            ɵɵelementStart(16, "div", 4);
            ɵɵelementStart(17, "button", 5);
            ɵɵlistener("click", function TbdevUploadDialogComponent_Template_button_click_17_listener() { return ctx.onClose(); });
            ɵɵtext(18, " Close ");
            ɵɵelementEnd();
            ɵɵelementStart(19, "a", 6);
            ɵɵtext(20, " Learn more ");
            ɵɵelementEnd();
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵadvance(3);
            ɵɵpropertyInterpolate("href", ctx.tensorboardDotDevUrl, ɵɵsanitizeUrl);
            ɵɵadvance(8);
            ɵɵtextInterpolate(ctx.getCommandText());
            ɵɵadvance(1);
            ɵɵproperty("cdkCopyToClipboard", ctx.getCommandText());
            ɵɵadvance(7);
            ɵɵpropertyInterpolate("href", ctx.tensorboardDotDevUrl, ɵɵsanitizeUrl);
        } }, directives: [MatButton, CdkCopyToClipboard, MatIcon, MatAnchor], styles: [".mat-badge-content[_ngcontent-%COMP%]{font-weight:600;font-size:12px;font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:9px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:24px}.mat-h1[_ngcontent-%COMP%], .mat-headline[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%]{font:400 24px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h2[_ngcontent-%COMP%], .mat-title[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h3[_ngcontent-%COMP%], .mat-subheading-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%]{font:400 16px/28px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h4[_ngcontent-%COMP%], .mat-subheading-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%]{font:400 15px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h5[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%]{font:400 calc(14px * 0.83)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-h6[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:400 calc(14px * 0.67)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-body-strong[_ngcontent-%COMP%], .mat-body-2[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   p[_ngcontent-%COMP%]{margin:0 0 12px}.mat-small[_ngcontent-%COMP%], .mat-caption[_ngcontent-%COMP%]{font:400 12px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-display-4[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-4[_ngcontent-%COMP%]{font:300 112px/112px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.05em;margin:0 0 56px}.mat-display-3[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-3[_ngcontent-%COMP%]{font:400 56px/56px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.02em;margin:0 0 64px}.mat-display-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-2[_ngcontent-%COMP%]{font:400 45px/48px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.005em;margin:0 0 64px}.mat-display-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-1[_ngcontent-%COMP%]{font:400 34px/40px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 64px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%], .mat-flat-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-button-toggle[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card-title[_ngcontent-%COMP%]{font-size:24px;font-weight:500}.mat-card-header[_ngcontent-%COMP%]   .mat-card-title[_ngcontent-%COMP%]{font-size:20px}.mat-card-subtitle[_ngcontent-%COMP%], .mat-card-content[_ngcontent-%COMP%]{font-size:14px}.mat-checkbox[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-checkbox-layout[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{line-height:24px}.mat-chip[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-chip[_ngcontent-%COMP%]   .mat-chip-trailing-icon.mat-icon[_ngcontent-%COMP%], .mat-chip[_ngcontent-%COMP%]   .mat-chip-remove.mat-icon[_ngcontent-%COMP%]{font-size:18px}.mat-table[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-header-cell[_ngcontent-%COMP%]{font-size:12px;font-weight:500}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{font-size:14px}.mat-calendar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-calendar-body[_ngcontent-%COMP%]{font-size:13px}.mat-calendar-body-label[_ngcontent-%COMP%], .mat-calendar-period-button[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-calendar-table-header[_ngcontent-%COMP%]   th[_ngcontent-%COMP%]{font-size:11px;font-weight:400}.mat-dialog-title[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-expansion-panel-header[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:15px;font-weight:400}.mat-expansion-panel-content[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field[_ngcontent-%COMP%]{font-size:inherit;font-weight:400;line-height:1.125;font-family:Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.34375em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{font-size:150%;line-height:1.125}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]{height:1.5em;width:1.5em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{height:1.125em;line-height:1.125}.mat-form-field-infix[_ngcontent-%COMP%]{padding:.5em 0;border-top:.84375em solid transparent}.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34375em) scale(0.75);width:133.3333333333%}.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34374em) scale(0.75);width:133.3333433333%}.mat-form-field-label-wrapper[_ngcontent-%COMP%]{top:-0.84375em;padding-top:.84375em}.mat-form-field-label[_ngcontent-%COMP%]{top:1.34375em}.mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.34375em}.mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{font-size:75%;margin-top:.6666666667em;top:calc(100% - 1.7916666667em)}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.4375em 0}.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.001px);-ms-transform:translateY(-1.28125em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00101px);-ms-transform:translateY(-1.28124em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00102px);-ms-transform:translateY(-1.28123em) scale(0.75);width:133.3333533333%}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.28125em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{margin-top:.5416666667em;top:calc(100% - 1.6666666667em)}@media print{.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28122em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28121em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.2812em) scale(0.75)}}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.25em 0 .75em 0}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.09375em;margin-top:-0.5em}.mat-form-field-appearance-fill.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59374em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:1em 0 1em 0}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.84375em;margin-top:-0.25em}.mat-form-field-appearance-outline.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59374em) scale(0.75);width:133.3333433333%}.mat-grid-tile-header[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]{font-size:14px}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2), .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}input.mat-input-element[_ngcontent-%COMP%]{margin-top:-0.0625em}.mat-menu-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:400}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px}.mat-radio-button[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select-trigger[_ngcontent-%COMP%]{height:1.125em}.mat-slide-toggle-content[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-slider-thumb-label-text[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-stepper-vertical[_ngcontent-%COMP%], .mat-stepper-horizontal[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-step-label[_ngcontent-%COMP%]{font-size:14px;font-weight:400}.mat-step-sub-label-error[_ngcontent-%COMP%]{font-weight:normal}.mat-step-label-error[_ngcontent-%COMP%]{font-size:14px}.mat-step-label-selected[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-tab-group[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-toolbar[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0}.mat-tooltip[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:10px;padding-top:6px;padding-bottom:6px}.mat-tooltip-handset[_ngcontent-%COMP%]{font-size:14px;padding-top:8px;padding-bottom:8px}.mat-list-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:16px}.mat-optgroup-label[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-simple-snackbar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px}.mat-simple-snackbar-action[_ngcontent-%COMP%]{line-height:1;font-family:inherit;font-size:inherit;font-weight:500}.mat-tree[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{font-weight:400;font-size:14px}.mat-ripple[_ngcontent-%COMP%]{overflow:hidden;position:relative}.mat-ripple[_ngcontent-%COMP%]:not(:empty){transform:translateZ(0)}.mat-ripple.mat-ripple-unbounded[_ngcontent-%COMP%]{overflow:visible}.mat-ripple-element[_ngcontent-%COMP%]{position:absolute;border-radius:50%;pointer-events:none;transition:opacity,transform 0ms cubic-bezier(0, 0, 0.2, 1);transform:scale(0)}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{display:none}.cdk-visually-hidden[_ngcontent-%COMP%]{border:0;clip:rect(0 0 0 0);height:1px;margin:-1px;overflow:hidden;padding:0;position:absolute;width:1px;outline:0;-webkit-appearance:none;-moz-appearance:none}.cdk-overlay-container[_ngcontent-%COMP%], .cdk-global-overlay-wrapper[_ngcontent-%COMP%]{pointer-events:none;top:0;left:0;height:100%;width:100%}.cdk-overlay-container[_ngcontent-%COMP%]{position:fixed;z-index:1000}.cdk-overlay-container[_ngcontent-%COMP%]:empty{display:none}.cdk-global-overlay-wrapper[_ngcontent-%COMP%]{display:flex;position:absolute;z-index:1000}.cdk-overlay-pane[_ngcontent-%COMP%]{position:absolute;pointer-events:auto;box-sizing:border-box;z-index:1000;display:flex;max-width:100%;max-height:100%}.cdk-overlay-backdrop[_ngcontent-%COMP%]{position:absolute;top:0;bottom:0;left:0;right:0;z-index:1000;pointer-events:auto;-webkit-tap-highlight-color:transparent;transition:opacity 400ms cubic-bezier(0.25, 0.8, 0.25, 1);opacity:0}.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:1}@media screen and (-ms-high-contrast: active){.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:.6}}.cdk-overlay-dark-backdrop[_ngcontent-%COMP%]{background:rgba(0,0,0,.32)}.cdk-overlay-transparent-backdrop[_ngcontent-%COMP%], .cdk-overlay-transparent-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:0}.cdk-overlay-connected-position-bounding-box[_ngcontent-%COMP%]{position:absolute;z-index:1000;display:flex;flex-direction:column;min-width:1px;min-height:1px}.cdk-global-scrollblock[_ngcontent-%COMP%]{position:fixed;width:100%;overflow-y:scroll}@keyframes cdk-text-field-autofill-start{}@keyframes cdk-text-field-autofill-end{}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:-webkit-autofill{animation:cdk-text-field-autofill-start 0s 1ms}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:not(:-webkit-autofill){animation:cdk-text-field-autofill-end 0s 1ms}textarea.cdk-textarea-autosize[_ngcontent-%COMP%]{resize:none}textarea.cdk-textarea-autosize-measuring[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:auto !important;overflow:hidden !important}textarea.cdk-textarea-autosize-measuring-firefox[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:0 !important}.mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.1)}.mat-option[_ngcontent-%COMP%]{color:#212121}.mat-option[_ngcontent-%COMP%]:hover:not(.mat-option-disabled), .mat-option[_ngcontent-%COMP%]:focus:not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple):not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-active[_ngcontent-%COMP%]{background:rgba(0,0,0,.04);color:#212121}.mat-option.mat-option-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-primary[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f57c00}.mat-accent[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f44336}.mat-optgroup-label[_ngcontent-%COMP%]{color:#616161}.mat-optgroup-disabled[_ngcontent-%COMP%]   .mat-optgroup-label[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-pseudo-checkbox[_ngcontent-%COMP%]{color:#616161}.mat-pseudo-checkbox[_ngcontent-%COMP%]::after{color:#fafafa}.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{color:#b0b0b0}.mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f57c00}.mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f44336}.mat-pseudo-checkbox-checked.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{background:#b0b0b0}.mat-elevation-z0[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z1[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z2[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z3[_ngcontent-%COMP%]{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z4[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z5[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 5px 8px 0px rgba(0, 0, 0, 0.14),0px 1px 14px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z6[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z7[_ngcontent-%COMP%]{box-shadow:0px 4px 5px -2px rgba(0, 0, 0, 0.2),0px 7px 10px 1px rgba(0, 0, 0, 0.14),0px 2px 16px 1px rgba(0, 0, 0, 0.12)}.mat-elevation-z8[_ngcontent-%COMP%]{box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z9[_ngcontent-%COMP%]{box-shadow:0px 5px 6px -3px rgba(0, 0, 0, 0.2),0px 9px 12px 1px rgba(0, 0, 0, 0.14),0px 3px 16px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z10[_ngcontent-%COMP%]{box-shadow:0px 6px 6px -3px rgba(0, 0, 0, 0.2),0px 10px 14px 1px rgba(0, 0, 0, 0.14),0px 4px 18px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z11[_ngcontent-%COMP%]{box-shadow:0px 6px 7px -4px rgba(0, 0, 0, 0.2),0px 11px 15px 1px rgba(0, 0, 0, 0.14),0px 4px 20px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z12[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z13[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 13px 19px 2px rgba(0, 0, 0, 0.14),0px 5px 24px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z14[_ngcontent-%COMP%]{box-shadow:0px 7px 9px -4px rgba(0, 0, 0, 0.2),0px 14px 21px 2px rgba(0, 0, 0, 0.14),0px 5px 26px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z15[_ngcontent-%COMP%]{box-shadow:0px 8px 9px -5px rgba(0, 0, 0, 0.2),0px 15px 22px 2px rgba(0, 0, 0, 0.14),0px 6px 28px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z16[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z17[_ngcontent-%COMP%]{box-shadow:0px 8px 11px -5px rgba(0, 0, 0, 0.2),0px 17px 26px 2px rgba(0, 0, 0, 0.14),0px 6px 32px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z18[_ngcontent-%COMP%]{box-shadow:0px 9px 11px -5px rgba(0, 0, 0, 0.2),0px 18px 28px 2px rgba(0, 0, 0, 0.14),0px 7px 34px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z19[_ngcontent-%COMP%]{box-shadow:0px 9px 12px -6px rgba(0, 0, 0, 0.2),0px 19px 29px 2px rgba(0, 0, 0, 0.14),0px 7px 36px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z20[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 20px 31px 3px rgba(0, 0, 0, 0.14),0px 8px 38px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z21[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 21px 33px 3px rgba(0, 0, 0, 0.14),0px 8px 40px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z22[_ngcontent-%COMP%]{box-shadow:0px 10px 14px -6px rgba(0, 0, 0, 0.2),0px 22px 35px 3px rgba(0, 0, 0, 0.14),0px 8px 42px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z23[_ngcontent-%COMP%]{box-shadow:0px 11px 14px -7px rgba(0, 0, 0, 0.2),0px 23px 36px 3px rgba(0, 0, 0, 0.14),0px 9px 44px 8px rgba(0, 0, 0, 0.12)}.mat-elevation-z24[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12)}.mat-app-background[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-theme-loaded-marker[_ngcontent-%COMP%]{display:none}.mat-autocomplete-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-autocomplete-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover){background:#fff}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover):not(.mat-option-disabled){color:#212121}.mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f57c00}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{outline:solid 1px;border-radius:0}.mat-badge-accent[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-badge-warn[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f44336}.mat-badge[_ngcontent-%COMP%]{position:relative}.mat-badge-hidden[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{display:none}.mat-badge-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#b9b9b9;color:#757575}.mat-badge-content[_ngcontent-%COMP%]{position:absolute;text-align:center;display:inline-block;border-radius:50%;transition:transform 200ms ease-in-out;transform:scale(0.6);overflow:hidden;white-space:nowrap;text-overflow:ellipsis;pointer-events:none}.ng-animate-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%], .mat-badge-content._mat-animation-noopable[_ngcontent-%COMP%]{transition:none}.mat-badge-content.mat-badge-active[_ngcontent-%COMP%]{transform:none}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:16px;height:16px;line-height:16px}.mat-badge-small.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-8px}.mat-badge-small.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-8px}.mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-16px}.mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-16px}.mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-8px}.mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-8px}.mat-badge-medium[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:22px;height:22px;line-height:22px}.mat-badge-medium.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-11px}.mat-badge-medium.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-11px}.mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-22px}.mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-22px}.mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-11px}.mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-11px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:28px;height:28px;line-height:28px}.mat-badge-large.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-14px}.mat-badge-large.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-14px}.mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-28px}.mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-28px}.mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-14px}.mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-14px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]{color:inherit;background:transparent}.mat-button.mat-primary[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-button[disabled][disabled][_ngcontent-%COMP%], .mat-icon-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-icon-button[disabled][disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-stroked-button[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f44336}.mat-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:transparent}.mat-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{opacity:.1;background-color:currentColor}.mat-button-focus-overlay[_ngcontent-%COMP%]{background:#000}.mat-stroked-button[_ngcontent-%COMP%]:not([disabled]){border-color:rgba(0,0,0,.12)}.mat-flat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{color:#212121;background-color:#fff}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-flat-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-stroked-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-flat-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-raised-button[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone[_ngcontent-%COMP%], .mat-button-toggle-group[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{box-shadow:none}.mat-button-toggle[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-button-toggle[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121;background:#fff}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:#000}.mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12)}[dir=rtl][_ngcontent-%COMP%]   .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-group-appearance-standard.mat-button-toggle-vertical[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:none;border-top:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#616161}.mat-button-toggle-checked.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121}.mat-button-toggle-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.26);background-color:#eee}.mat-button-toggle-disabled.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{background:#fff}.mat-button-toggle-disabled.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#bdbdbd}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{border:solid 1px rgba(0,0,0,.12)}.mat-card[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-card[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-card.mat-card-flat[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-card-subtitle[_ngcontent-%COMP%]{color:#616161}.mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#616161}.mat-checkbox-checkmark[_ngcontent-%COMP%]{fill:#fafafa}.mat-checkbox-checkmark-path[_ngcontent-%COMP%]{stroke:#fafafa !important}.mat-checkbox-mixedmark[_ngcontent-%COMP%]{background-color:#fafafa}.mat-checkbox-indeterminate.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f57c00}.mat-checkbox-indeterminate.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#ff9800}.mat-checkbox-indeterminate.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f44336}.mat-checkbox-disabled.mat-checkbox-checked[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-disabled.mat-checkbox-indeterminate[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]:not(.mat-checkbox-checked)   .mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{color:#616161}.mat-checkbox[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%]{background:#f57c00}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%]{background:#ff9800}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%]{background:#f44336}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#212121}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#212121;opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled):active{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled)   .mat-chip-remove[_ngcontent-%COMP%]:hover{opacity:.54}.mat-chip.mat-standard-chip.mat-chip-disabled[_ngcontent-%COMP%]{opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]::after{background:#000}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-table[_ngcontent-%COMP%]{background:#fff}.mat-table[_ngcontent-%COMP%]   thead[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tbody[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tfoot[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], [mat-header-row][_ngcontent-%COMP%], [mat-row][_ngcontent-%COMP%], [mat-footer-row][_ngcontent-%COMP%], .mat-table-sticky[_ngcontent-%COMP%]{background:inherit}mat-row[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], th.mat-header-cell[_ngcontent-%COMP%], td.mat-cell[_ngcontent-%COMP%], td.mat-footer-cell[_ngcontent-%COMP%]{border-bottom-color:rgba(0,0,0,.12)}.mat-header-cell[_ngcontent-%COMP%]{color:#616161}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{color:#212121}.mat-calendar-arrow[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.54)}.mat-datepicker-toggle[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-next-button[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-previous-button[_ngcontent-%COMP%]{color:rgba(0,0,0,.54)}.mat-calendar-table-header[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-calendar-table-header-divider[_ngcontent-%COMP%]::after{background:rgba(0,0,0,.12)}.mat-calendar-body-label[_ngcontent-%COMP%]{color:#616161}.mat-calendar-body-cell-content[_ngcontent-%COMP%]{color:#212121;border-color:transparent}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){color:#757575}.mat-calendar-body-cell[_ngcontent-%COMP%]:not(.mat-calendar-body-disabled):hover > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-keyboard-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-program-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){background-color:rgba(0,0,0,.04)}.mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.38)}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.18)}.mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.4)}.mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12);background-color:#fff;color:#212121}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.4)}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.4)}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content-touch[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-datepicker-toggle-active[_ngcontent-%COMP%]{color:#f57c00}.mat-datepicker-toggle-active.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-datepicker-toggle-active.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-dialog-container[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-divider[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-divider-vertical[_ngcontent-%COMP%]{border-right-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-expansion-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-action-row[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-keyboard-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-program-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded)   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover:not([aria-disabled=true]){background:rgba(0,0,0,.04)}@media(hover: none){.mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded):not([aria-disabled=true])   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover{background:#fff}}.mat-expansion-panel-header-title[_ngcontent-%COMP%]{color:#212121}.mat-expansion-panel-header-description[_ngcontent-%COMP%], .mat-expansion-indicator[_ngcontent-%COMP%]::after{color:#616161}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-title[_ngcontent-%COMP%], .mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-description[_ngcontent-%COMP%]{color:inherit}.mat-form-field-label[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-hint[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-focused[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-ripple[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid)   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f57c00}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-accent   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#ff9800}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-warn   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#f44336}.mat-error[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-hint[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-legacy.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-standard[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-standard.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.04)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.02)}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:transparent}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.12)}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:rgba(0,0,0,.87)}.mat-form-field-appearance-outline.mat-focused[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field-appearance-outline.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-appearance-outline.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-invalid.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.06)}.mat-icon.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-icon.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-icon.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-type-mat-native-select[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#616161}.mat-input-element[_ngcontent-%COMP%]:disabled, .mat-form-field-type-mat-native-select.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#757575}.mat-input-element[_ngcontent-%COMP%]{caret-color:#f57c00}.mat-input-element[_ngcontent-%COMP%]::placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-moz-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-webkit-input-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]:-ms-input-placeholder{color:rgba(97,97,97,.42)}.mat-accent[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%], .mat-form-field-invalid[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#f44336}.mat-form-field-type-mat-native-select.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{color:#616161}.mat-list-item-disabled[_ngcontent-%COMP%]{background-color:#eee}.mat-list-option[_ngcontent-%COMP%]:hover, .mat-list-option[_ngcontent-%COMP%]:focus, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.04)}.mat-list-single-selected-option[_ngcontent-%COMP%], .mat-list-single-selected-option[_ngcontent-%COMP%]:hover, .mat-list-single-selected-option[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.12)}.mat-menu-panel[_ngcontent-%COMP%]{background:#fff}.mat-menu-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-menu-item[_ngcontent-%COMP%]{background:transparent;color:#212121}.mat-menu-item[disabled][_ngcontent-%COMP%], .mat-menu-item[disabled][_ngcontent-%COMP%]::after{color:rgba(0,0,0,.38)}.mat-menu-item[_ngcontent-%COMP%]   .mat-icon-no-color[_ngcontent-%COMP%], .mat-menu-item-submenu-trigger[_ngcontent-%COMP%]::after{color:rgba(0,0,0,.54)}.mat-menu-item[_ngcontent-%COMP%]:hover:not([disabled]), .mat-menu-item.cdk-program-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item.cdk-keyboard-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item-highlighted[_ngcontent-%COMP%]:not([disabled]){background:rgba(0,0,0,.04)}.mat-paginator[_ngcontent-%COMP%]{background:#fff}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{color:#616161}.mat-paginator-decrement[_ngcontent-%COMP%], .mat-paginator-increment[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54);border-right:2px solid rgba(0,0,0,.54)}.mat-paginator-first[_ngcontent-%COMP%], .mat-paginator-last[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54)}.mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-decrement[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-increment[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-first[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-last[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f57c00}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#ff9800}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f44336}.mat-progress-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f57c00}.mat-progress-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#ff9800}.mat-progress-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f44336}.mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#616161}.mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f57c00}.mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#ff9800}.mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#ff9800}.mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f44336}.mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f44336}.mat-radio-button.mat-radio-disabled.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-label-content[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-radio-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-select-value[_ngcontent-%COMP%]{color:#212121}.mat-select-placeholder[_ngcontent-%COMP%]{color:rgba(97,97,97,.42)}.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%]{color:#757575}.mat-select-arrow[_ngcontent-%COMP%]{color:#616161}.mat-select-panel[_ngcontent-%COMP%]{background:#fff}.mat-select-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-select-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple){background:rgba(0,0,0,.12)}.mat-form-field.mat-focused.mat-primary[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-invalid[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#757575}.mat-drawer-container[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-drawer[_ngcontent-%COMP%]{background-color:#fff;color:#212121}.mat-drawer.mat-drawer-push[_ngcontent-%COMP%]{background-color:#fff}.mat-drawer[_ngcontent-%COMP%]:not(.mat-drawer-side){box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-drawer-side[_ngcontent-%COMP%]{border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-backdrop.mat-drawer-shown[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.6)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.54)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.54)}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.54)}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle[_ngcontent-%COMP%]:not(.mat-checked)   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-slide-toggle-thumb[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12);background-color:#fafafa}.mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-primary[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f57c00}.mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-accent[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#ff9800}.mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-warn[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f44336}.mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.2)}.mat-slider[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%], .cdk-focused[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-disabled[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]   .mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing)   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26);background-color:transparent}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26)}.mat-slider-has-ticks[_ngcontent-%COMP%]   .mat-slider-wrapper[_ngcontent-%COMP%]::after{border-color:rgba(0,0,0,.7)}.mat-slider-horizontal[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to right, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent);background-image:-moz-repeating-linear-gradient(0.0001deg, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-slider-vertical[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to bottom, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-step-header.cdk-keyboard-focused[_ngcontent-%COMP%], .mat-step-header.cdk-program-focused[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]:hover{background-color:rgba(0,0,0,.04)}@media(hover: none){.mat-step-header[_ngcontent-%COMP%]:hover{background:none}}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-optional[_ngcontent-%COMP%]{color:#616161}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon[_ngcontent-%COMP%]{background-color:#616161;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-selected[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-done[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-edit[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-error[_ngcontent-%COMP%]{background-color:transparent;color:#f44336}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-active[_ngcontent-%COMP%]{color:#212121}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-error[_ngcontent-%COMP%]{color:#f44336}.mat-stepper-horizontal[_ngcontent-%COMP%], .mat-stepper-vertical[_ngcontent-%COMP%]{background-color:#fff}.mat-stepper-vertical-line[_ngcontent-%COMP%]::before{border-left-color:rgba(0,0,0,.12)}.mat-horizontal-stepper-header[_ngcontent-%COMP%]::before, .mat-horizontal-stepper-header[_ngcontent-%COMP%]::after, .mat-stepper-horizontal-line[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-sort-header-arrow[_ngcontent-%COMP%]{color:#616161}.mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-header[_ngcontent-%COMP%]{border-bottom:1px solid rgba(0,0,0,.12)}.mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%]{border-top:1px solid rgba(0,0,0,.12);border-bottom:none}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{color:#212121}.mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:#757575}.mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#212121}.mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#757575}.mat-tab-group[class*=mat-background-][_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar[class*=mat-background-][_ngcontent-%COMP%]{border-bottom:none;border-top:none}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-toolbar[_ngcontent-%COMP%]{background:#f5f5f5;color:#212121}.mat-toolbar.mat-primary[_ngcontent-%COMP%]{background:#f57c00;color:#fff}.mat-toolbar.mat-accent[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-toolbar.mat-warn[_ngcontent-%COMP%]{background:#f44336;color:#fff}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:currentColor}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:inherit}.mat-toolbar[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:currentColor}.mat-tooltip[_ngcontent-%COMP%]{background:rgba(97,97,97,.9)}.mat-tree[_ngcontent-%COMP%]{background:#fff}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{color:#212121}.mat-snack-bar-container[_ngcontent-%COMP%]{color:rgba(255,255,255,.7);background:#323232;box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-simple-snackbar-action[_ngcontent-%COMP%]{color:#ff9800}div[_ngcontent-%COMP%], p[_ngcontent-%COMP%]{margin:16px 0}[_nghost-%COMP%] > [_ngcontent-%COMP%]:first-child{margin-top:0}[_nghost-%COMP%] > [_ngcontent-%COMP%]:last-child{margin-bottom:0}h3[_ngcontent-%COMP%]{font-size:14px;font-weight:500;line-height:1.5}p[_ngcontent-%COMP%]{color:#212121;font-size:12px;line-height:1.5}.anchor-text[_ngcontent-%COMP%]{color:#1976d2;text-decoration:none}.command[_ngcontent-%COMP%]{align-items:center;background:#f5f6f7;border-radius:4px;display:flex;justify-content:space-between;padding:2px 12px}pre[_ngcontent-%COMP%]{overflow-x:auto}code[_ngcontent-%COMP%]{font-size:14px;line-height:1.5}.bottom-buttons[_ngcontent-%COMP%]{align-items:center;display:flex;justify-content:flex-end}.close-button[_ngcontent-%COMP%]{color:#616161;text-transform:uppercase;margin-right:8px}.learn-more-button[_ngcontent-%COMP%]{color:#1976d2;text-transform:uppercase}"] });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/tbdev_upload/tbdev_upload_dialog_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const getLogdir = createSelector(getEnvironment, (/**
     * @param {?} environment
     * @return {?}
     */
    (environment) => environment.data_location));
    class TbdevUploadDialogContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.logdir$ = this.store.pipe(select(getLogdir));
        }
    }
    TbdevUploadDialogContainer.decorators = [
        { type: Component, args: [{
                    selector: 'tbdev-upload-dialog',
                    template: `
    <tbdev-upload-dialog-component
      [logdir]="logdir$ | async"
    ></tbdev-upload-dialog-component>
  `,
                },] },
    ];
    /** @nocollapse */
    TbdevUploadDialogContainer.ctorParameters = () => [
        { type: Store }
    ];
    /** @nocollapse */ TbdevUploadDialogContainer.ɵfac = function TbdevUploadDialogContainer_Factory(t) { return new (t || TbdevUploadDialogContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ TbdevUploadDialogContainer.ɵcmp = ɵɵdefineComponent({ type: TbdevUploadDialogContainer, selectors: [["tbdev-upload-dialog"]], decls: 2, vars: 3, consts: [[3, "logdir"]], template: function TbdevUploadDialogContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelement(0, "tbdev-upload-dialog-component", 0);
            ɵɵpipe(1, "async");
        } if (rf & 2) {
            ɵɵproperty("logdir", ɵɵpipeBind1(1, 1, ctx.logdir$));
        } }, directives: [TbdevUploadDialogComponent], pipes: [AsyncPipe], encapsulation: 2 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/tbdev_upload/tbdev_upload_button_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    function TbdevUploadButtonComponent_button_0_Template(rf, ctx) { if (rf & 1) {
        const _r2 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "button", 1);
        ɵɵlistener("click", function TbdevUploadButtonComponent_button_0_Template_button_click_0_listener() { ɵɵrestoreView(_r2); const ctx_r1 = ɵɵnextContext(); return ctx_r1.openDialog(); });
        ɵɵelementStart(1, "span", 2);
        ɵɵelement(2, "mat-icon", 3);
        ɵɵtext(3, " Upload ");
        ɵɵelementEnd();
        ɵɵelementEnd();
    } }
    // A list of hostname values that will trigger the button to appear.
    /** @type {?} */
    const LOCAL_HOSTNAMES = ['localhost', '127.0.0.1'];
    class TbdevUploadButtonComponent {
        /**
         * @param {?} window
         * @param {?} dialog
         */
        constructor(window, dialog) {
            this.window = window;
            this.dialog = dialog;
            this.shown = LOCAL_HOSTNAMES.includes(window.location.hostname);
        }
        /**
         * @return {?}
         */
        openDialog() {
            this.dialog.open(TbdevUploadDialogContainer, {
                width: '560px',
            });
        }
    }
    TbdevUploadButtonComponent.decorators = [
        { type: Component, args: [{
                    selector: 'tbdev-upload-button',
                    templateUrl: './tbdev_upload_button_component.ng.html',
                    styleUrls: ['./tbdev_upload_button_component.css'],
                },] },
    ];
    /** @nocollapse */
    TbdevUploadButtonComponent.ctorParameters = () => [
        { type: Window, decorators: [{ type: Inject, args: ['window',] }] },
        { type: MatDialog }
    ];
    TbdevUploadButtonComponent.propDecorators = {
        shown: [{ type: HostBinding, args: ['class.shown',] }]
    };
    /** @nocollapse */ TbdevUploadButtonComponent.ɵfac = function TbdevUploadButtonComponent_Factory(t) { return new (t || TbdevUploadButtonComponent)(ɵɵdirectiveInject('window'), ɵɵdirectiveInject(MatDialog)); };
    /** @nocollapse */ TbdevUploadButtonComponent.ɵcmp = ɵɵdefineComponent({ type: TbdevUploadButtonComponent, selectors: [["tbdev-upload-button"]], hostVars: 2, hostBindings: function TbdevUploadButtonComponent_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵclassProp("shown", ctx.shown);
        } }, decls: 1, vars: 1, consts: [["mat-stroked-button", "", 3, "click", 4, "ngIf"], ["mat-stroked-button", "", 3, "click"], [1, "button-contents"], ["svgIcon", "info_outline_24px"]], template: function TbdevUploadButtonComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵtemplate(0, TbdevUploadButtonComponent_button_0_Template, 4, 0, "button", 0);
        } if (rf & 2) {
            ɵɵproperty("ngIf", ctx.shown);
        } }, directives: [NgIf, MatButton, MatIcon], styles: [".mat-badge-content[_ngcontent-%COMP%]{font-weight:600;font-size:12px;font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:9px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:24px}.mat-h1[_ngcontent-%COMP%], .mat-headline[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%]{font:400 24px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h2[_ngcontent-%COMP%], .mat-title[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h3[_ngcontent-%COMP%], .mat-subheading-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%]{font:400 16px/28px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h4[_ngcontent-%COMP%], .mat-subheading-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%]{font:400 15px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h5[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%]{font:400 calc(14px * 0.83)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-h6[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:400 calc(14px * 0.67)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-body-strong[_ngcontent-%COMP%], .mat-body-2[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   p[_ngcontent-%COMP%]{margin:0 0 12px}.mat-small[_ngcontent-%COMP%], .mat-caption[_ngcontent-%COMP%]{font:400 12px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-display-4[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-4[_ngcontent-%COMP%]{font:300 112px/112px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.05em;margin:0 0 56px}.mat-display-3[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-3[_ngcontent-%COMP%]{font:400 56px/56px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.02em;margin:0 0 64px}.mat-display-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-2[_ngcontent-%COMP%]{font:400 45px/48px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.005em;margin:0 0 64px}.mat-display-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-1[_ngcontent-%COMP%]{font:400 34px/40px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 64px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%], .mat-flat-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-button-toggle[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card-title[_ngcontent-%COMP%]{font-size:24px;font-weight:500}.mat-card-header[_ngcontent-%COMP%]   .mat-card-title[_ngcontent-%COMP%]{font-size:20px}.mat-card-subtitle[_ngcontent-%COMP%], .mat-card-content[_ngcontent-%COMP%]{font-size:14px}.mat-checkbox[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-checkbox-layout[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{line-height:24px}.mat-chip[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-chip[_ngcontent-%COMP%]   .mat-chip-trailing-icon.mat-icon[_ngcontent-%COMP%], .mat-chip[_ngcontent-%COMP%]   .mat-chip-remove.mat-icon[_ngcontent-%COMP%]{font-size:18px}.mat-table[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-header-cell[_ngcontent-%COMP%]{font-size:12px;font-weight:500}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{font-size:14px}.mat-calendar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-calendar-body[_ngcontent-%COMP%]{font-size:13px}.mat-calendar-body-label[_ngcontent-%COMP%], .mat-calendar-period-button[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-calendar-table-header[_ngcontent-%COMP%]   th[_ngcontent-%COMP%]{font-size:11px;font-weight:400}.mat-dialog-title[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-expansion-panel-header[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:15px;font-weight:400}.mat-expansion-panel-content[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field[_ngcontent-%COMP%]{font-size:inherit;font-weight:400;line-height:1.125;font-family:Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.34375em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{font-size:150%;line-height:1.125}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]{height:1.5em;width:1.5em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{height:1.125em;line-height:1.125}.mat-form-field-infix[_ngcontent-%COMP%]{padding:.5em 0;border-top:.84375em solid transparent}.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34375em) scale(0.75);width:133.3333333333%}.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34374em) scale(0.75);width:133.3333433333%}.mat-form-field-label-wrapper[_ngcontent-%COMP%]{top:-0.84375em;padding-top:.84375em}.mat-form-field-label[_ngcontent-%COMP%]{top:1.34375em}.mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.34375em}.mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{font-size:75%;margin-top:.6666666667em;top:calc(100% - 1.7916666667em)}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.4375em 0}.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.001px);-ms-transform:translateY(-1.28125em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00101px);-ms-transform:translateY(-1.28124em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00102px);-ms-transform:translateY(-1.28123em) scale(0.75);width:133.3333533333%}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.28125em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{margin-top:.5416666667em;top:calc(100% - 1.6666666667em)}@media print{.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28122em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28121em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.2812em) scale(0.75)}}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.25em 0 .75em 0}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.09375em;margin-top:-0.5em}.mat-form-field-appearance-fill.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59374em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:1em 0 1em 0}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.84375em;margin-top:-0.25em}.mat-form-field-appearance-outline.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59374em) scale(0.75);width:133.3333433333%}.mat-grid-tile-header[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]{font-size:14px}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2), .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}input.mat-input-element[_ngcontent-%COMP%]{margin-top:-0.0625em}.mat-menu-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:400}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px}.mat-radio-button[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select-trigger[_ngcontent-%COMP%]{height:1.125em}.mat-slide-toggle-content[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-slider-thumb-label-text[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-stepper-vertical[_ngcontent-%COMP%], .mat-stepper-horizontal[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-step-label[_ngcontent-%COMP%]{font-size:14px;font-weight:400}.mat-step-sub-label-error[_ngcontent-%COMP%]{font-weight:normal}.mat-step-label-error[_ngcontent-%COMP%]{font-size:14px}.mat-step-label-selected[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-tab-group[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-toolbar[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0}.mat-tooltip[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:10px;padding-top:6px;padding-bottom:6px}.mat-tooltip-handset[_ngcontent-%COMP%]{font-size:14px;padding-top:8px;padding-bottom:8px}.mat-list-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:16px}.mat-optgroup-label[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-simple-snackbar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px}.mat-simple-snackbar-action[_ngcontent-%COMP%]{line-height:1;font-family:inherit;font-size:inherit;font-weight:500}.mat-tree[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{font-weight:400;font-size:14px}.mat-ripple[_ngcontent-%COMP%]{overflow:hidden;position:relative}.mat-ripple[_ngcontent-%COMP%]:not(:empty){transform:translateZ(0)}.mat-ripple.mat-ripple-unbounded[_ngcontent-%COMP%]{overflow:visible}.mat-ripple-element[_ngcontent-%COMP%]{position:absolute;border-radius:50%;pointer-events:none;transition:opacity,transform 0ms cubic-bezier(0, 0, 0.2, 1);transform:scale(0)}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{display:none}.cdk-visually-hidden[_ngcontent-%COMP%]{border:0;clip:rect(0 0 0 0);height:1px;margin:-1px;overflow:hidden;padding:0;position:absolute;width:1px;outline:0;-webkit-appearance:none;-moz-appearance:none}.cdk-overlay-container[_ngcontent-%COMP%], .cdk-global-overlay-wrapper[_ngcontent-%COMP%]{pointer-events:none;top:0;left:0;height:100%;width:100%}.cdk-overlay-container[_ngcontent-%COMP%]{position:fixed;z-index:1000}.cdk-overlay-container[_ngcontent-%COMP%]:empty{display:none}.cdk-global-overlay-wrapper[_ngcontent-%COMP%]{display:flex;position:absolute;z-index:1000}.cdk-overlay-pane[_ngcontent-%COMP%]{position:absolute;pointer-events:auto;box-sizing:border-box;z-index:1000;display:flex;max-width:100%;max-height:100%}.cdk-overlay-backdrop[_ngcontent-%COMP%]{position:absolute;top:0;bottom:0;left:0;right:0;z-index:1000;pointer-events:auto;-webkit-tap-highlight-color:transparent;transition:opacity 400ms cubic-bezier(0.25, 0.8, 0.25, 1);opacity:0}.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:1}@media screen and (-ms-high-contrast: active){.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:.6}}.cdk-overlay-dark-backdrop[_ngcontent-%COMP%]{background:rgba(0,0,0,.32)}.cdk-overlay-transparent-backdrop[_ngcontent-%COMP%], .cdk-overlay-transparent-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:0}.cdk-overlay-connected-position-bounding-box[_ngcontent-%COMP%]{position:absolute;z-index:1000;display:flex;flex-direction:column;min-width:1px;min-height:1px}.cdk-global-scrollblock[_ngcontent-%COMP%]{position:fixed;width:100%;overflow-y:scroll}@keyframes cdk-text-field-autofill-start{}@keyframes cdk-text-field-autofill-end{}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:-webkit-autofill{animation:cdk-text-field-autofill-start 0s 1ms}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:not(:-webkit-autofill){animation:cdk-text-field-autofill-end 0s 1ms}textarea.cdk-textarea-autosize[_ngcontent-%COMP%]{resize:none}textarea.cdk-textarea-autosize-measuring[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:auto !important;overflow:hidden !important}textarea.cdk-textarea-autosize-measuring-firefox[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:0 !important}.mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.1)}.mat-option[_ngcontent-%COMP%]{color:#212121}.mat-option[_ngcontent-%COMP%]:hover:not(.mat-option-disabled), .mat-option[_ngcontent-%COMP%]:focus:not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple):not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-active[_ngcontent-%COMP%]{background:rgba(0,0,0,.04);color:#212121}.mat-option.mat-option-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-primary[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f57c00}.mat-accent[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f44336}.mat-optgroup-label[_ngcontent-%COMP%]{color:#616161}.mat-optgroup-disabled[_ngcontent-%COMP%]   .mat-optgroup-label[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-pseudo-checkbox[_ngcontent-%COMP%]{color:#616161}.mat-pseudo-checkbox[_ngcontent-%COMP%]::after{color:#fafafa}.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{color:#b0b0b0}.mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f57c00}.mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f44336}.mat-pseudo-checkbox-checked.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{background:#b0b0b0}.mat-elevation-z0[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z1[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z2[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z3[_ngcontent-%COMP%]{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z4[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z5[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 5px 8px 0px rgba(0, 0, 0, 0.14),0px 1px 14px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z6[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z7[_ngcontent-%COMP%]{box-shadow:0px 4px 5px -2px rgba(0, 0, 0, 0.2),0px 7px 10px 1px rgba(0, 0, 0, 0.14),0px 2px 16px 1px rgba(0, 0, 0, 0.12)}.mat-elevation-z8[_ngcontent-%COMP%]{box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z9[_ngcontent-%COMP%]{box-shadow:0px 5px 6px -3px rgba(0, 0, 0, 0.2),0px 9px 12px 1px rgba(0, 0, 0, 0.14),0px 3px 16px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z10[_ngcontent-%COMP%]{box-shadow:0px 6px 6px -3px rgba(0, 0, 0, 0.2),0px 10px 14px 1px rgba(0, 0, 0, 0.14),0px 4px 18px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z11[_ngcontent-%COMP%]{box-shadow:0px 6px 7px -4px rgba(0, 0, 0, 0.2),0px 11px 15px 1px rgba(0, 0, 0, 0.14),0px 4px 20px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z12[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z13[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 13px 19px 2px rgba(0, 0, 0, 0.14),0px 5px 24px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z14[_ngcontent-%COMP%]{box-shadow:0px 7px 9px -4px rgba(0, 0, 0, 0.2),0px 14px 21px 2px rgba(0, 0, 0, 0.14),0px 5px 26px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z15[_ngcontent-%COMP%]{box-shadow:0px 8px 9px -5px rgba(0, 0, 0, 0.2),0px 15px 22px 2px rgba(0, 0, 0, 0.14),0px 6px 28px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z16[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z17[_ngcontent-%COMP%]{box-shadow:0px 8px 11px -5px rgba(0, 0, 0, 0.2),0px 17px 26px 2px rgba(0, 0, 0, 0.14),0px 6px 32px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z18[_ngcontent-%COMP%]{box-shadow:0px 9px 11px -5px rgba(0, 0, 0, 0.2),0px 18px 28px 2px rgba(0, 0, 0, 0.14),0px 7px 34px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z19[_ngcontent-%COMP%]{box-shadow:0px 9px 12px -6px rgba(0, 0, 0, 0.2),0px 19px 29px 2px rgba(0, 0, 0, 0.14),0px 7px 36px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z20[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 20px 31px 3px rgba(0, 0, 0, 0.14),0px 8px 38px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z21[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 21px 33px 3px rgba(0, 0, 0, 0.14),0px 8px 40px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z22[_ngcontent-%COMP%]{box-shadow:0px 10px 14px -6px rgba(0, 0, 0, 0.2),0px 22px 35px 3px rgba(0, 0, 0, 0.14),0px 8px 42px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z23[_ngcontent-%COMP%]{box-shadow:0px 11px 14px -7px rgba(0, 0, 0, 0.2),0px 23px 36px 3px rgba(0, 0, 0, 0.14),0px 9px 44px 8px rgba(0, 0, 0, 0.12)}.mat-elevation-z24[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12)}.mat-app-background[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-theme-loaded-marker[_ngcontent-%COMP%]{display:none}.mat-autocomplete-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-autocomplete-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover){background:#fff}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover):not(.mat-option-disabled){color:#212121}.mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f57c00}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{outline:solid 1px;border-radius:0}.mat-badge-accent[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-badge-warn[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f44336}.mat-badge[_ngcontent-%COMP%]{position:relative}.mat-badge-hidden[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{display:none}.mat-badge-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#b9b9b9;color:#757575}.mat-badge-content[_ngcontent-%COMP%]{position:absolute;text-align:center;display:inline-block;border-radius:50%;transition:transform 200ms ease-in-out;transform:scale(0.6);overflow:hidden;white-space:nowrap;text-overflow:ellipsis;pointer-events:none}.ng-animate-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%], .mat-badge-content._mat-animation-noopable[_ngcontent-%COMP%]{transition:none}.mat-badge-content.mat-badge-active[_ngcontent-%COMP%]{transform:none}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:16px;height:16px;line-height:16px}.mat-badge-small.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-8px}.mat-badge-small.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-8px}.mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-16px}.mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-16px}.mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-8px}.mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-8px}.mat-badge-medium[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:22px;height:22px;line-height:22px}.mat-badge-medium.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-11px}.mat-badge-medium.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-11px}.mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-22px}.mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-22px}.mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-11px}.mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-11px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:28px;height:28px;line-height:28px}.mat-badge-large.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-14px}.mat-badge-large.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-14px}.mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-28px}.mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-28px}.mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-14px}.mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-14px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]{color:inherit;background:transparent}.mat-button.mat-primary[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-button[disabled][disabled][_ngcontent-%COMP%], .mat-icon-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-icon-button[disabled][disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-stroked-button[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f44336}.mat-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:transparent}.mat-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{opacity:.1;background-color:currentColor}.mat-button-focus-overlay[_ngcontent-%COMP%]{background:#000}.mat-stroked-button[_ngcontent-%COMP%]:not([disabled]){border-color:rgba(0,0,0,.12)}.mat-flat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{color:#212121;background-color:#fff}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-flat-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-stroked-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-flat-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-raised-button[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone[_ngcontent-%COMP%], .mat-button-toggle-group[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{box-shadow:none}.mat-button-toggle[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-button-toggle[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121;background:#fff}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:#000}.mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12)}[dir=rtl][_ngcontent-%COMP%]   .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-group-appearance-standard.mat-button-toggle-vertical[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:none;border-top:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#616161}.mat-button-toggle-checked.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121}.mat-button-toggle-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.26);background-color:#eee}.mat-button-toggle-disabled.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{background:#fff}.mat-button-toggle-disabled.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#bdbdbd}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{border:solid 1px rgba(0,0,0,.12)}.mat-card[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-card[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-card.mat-card-flat[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-card-subtitle[_ngcontent-%COMP%]{color:#616161}.mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#616161}.mat-checkbox-checkmark[_ngcontent-%COMP%]{fill:#fafafa}.mat-checkbox-checkmark-path[_ngcontent-%COMP%]{stroke:#fafafa !important}.mat-checkbox-mixedmark[_ngcontent-%COMP%]{background-color:#fafafa}.mat-checkbox-indeterminate.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f57c00}.mat-checkbox-indeterminate.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#ff9800}.mat-checkbox-indeterminate.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f44336}.mat-checkbox-disabled.mat-checkbox-checked[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-disabled.mat-checkbox-indeterminate[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]:not(.mat-checkbox-checked)   .mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{color:#616161}.mat-checkbox[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%]{background:#f57c00}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%]{background:#ff9800}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%]{background:#f44336}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#212121}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#212121;opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled):active{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled)   .mat-chip-remove[_ngcontent-%COMP%]:hover{opacity:.54}.mat-chip.mat-standard-chip.mat-chip-disabled[_ngcontent-%COMP%]{opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]::after{background:#000}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-table[_ngcontent-%COMP%]{background:#fff}.mat-table[_ngcontent-%COMP%]   thead[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tbody[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tfoot[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], [mat-header-row][_ngcontent-%COMP%], [mat-row][_ngcontent-%COMP%], [mat-footer-row][_ngcontent-%COMP%], .mat-table-sticky[_ngcontent-%COMP%]{background:inherit}mat-row[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], th.mat-header-cell[_ngcontent-%COMP%], td.mat-cell[_ngcontent-%COMP%], td.mat-footer-cell[_ngcontent-%COMP%]{border-bottom-color:rgba(0,0,0,.12)}.mat-header-cell[_ngcontent-%COMP%]{color:#616161}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{color:#212121}.mat-calendar-arrow[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.54)}.mat-datepicker-toggle[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-next-button[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-previous-button[_ngcontent-%COMP%]{color:rgba(0,0,0,.54)}.mat-calendar-table-header[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-calendar-table-header-divider[_ngcontent-%COMP%]::after{background:rgba(0,0,0,.12)}.mat-calendar-body-label[_ngcontent-%COMP%]{color:#616161}.mat-calendar-body-cell-content[_ngcontent-%COMP%]{color:#212121;border-color:transparent}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){color:#757575}.mat-calendar-body-cell[_ngcontent-%COMP%]:not(.mat-calendar-body-disabled):hover > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-keyboard-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-program-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){background-color:rgba(0,0,0,.04)}.mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.38)}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.18)}.mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.4)}.mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12);background-color:#fff;color:#212121}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.4)}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.4)}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content-touch[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-datepicker-toggle-active[_ngcontent-%COMP%]{color:#f57c00}.mat-datepicker-toggle-active.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-datepicker-toggle-active.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-dialog-container[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-divider[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-divider-vertical[_ngcontent-%COMP%]{border-right-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-expansion-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-action-row[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-keyboard-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-program-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded)   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover:not([aria-disabled=true]){background:rgba(0,0,0,.04)}@media(hover: none){.mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded):not([aria-disabled=true])   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover{background:#fff}}.mat-expansion-panel-header-title[_ngcontent-%COMP%]{color:#212121}.mat-expansion-panel-header-description[_ngcontent-%COMP%], .mat-expansion-indicator[_ngcontent-%COMP%]::after{color:#616161}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-title[_ngcontent-%COMP%], .mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-description[_ngcontent-%COMP%]{color:inherit}.mat-form-field-label[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-hint[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-focused[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-ripple[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid)   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f57c00}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-accent   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#ff9800}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-warn   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#f44336}.mat-error[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-hint[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-legacy.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-standard[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-standard.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.04)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.02)}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:transparent}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.12)}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:rgba(0,0,0,.87)}.mat-form-field-appearance-outline.mat-focused[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field-appearance-outline.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-appearance-outline.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-invalid.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.06)}.mat-icon.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-icon.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-icon.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-type-mat-native-select[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#616161}.mat-input-element[_ngcontent-%COMP%]:disabled, .mat-form-field-type-mat-native-select.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#757575}.mat-input-element[_ngcontent-%COMP%]{caret-color:#f57c00}.mat-input-element[_ngcontent-%COMP%]::placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-moz-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-webkit-input-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]:-ms-input-placeholder{color:rgba(97,97,97,.42)}.mat-accent[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%], .mat-form-field-invalid[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#f44336}.mat-form-field-type-mat-native-select.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{color:#616161}.mat-list-item-disabled[_ngcontent-%COMP%]{background-color:#eee}.mat-list-option[_ngcontent-%COMP%]:hover, .mat-list-option[_ngcontent-%COMP%]:focus, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.04)}.mat-list-single-selected-option[_ngcontent-%COMP%], .mat-list-single-selected-option[_ngcontent-%COMP%]:hover, .mat-list-single-selected-option[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.12)}.mat-menu-panel[_ngcontent-%COMP%]{background:#fff}.mat-menu-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-menu-item[_ngcontent-%COMP%]{background:transparent;color:#212121}.mat-menu-item[disabled][_ngcontent-%COMP%], .mat-menu-item[disabled][_ngcontent-%COMP%]::after{color:rgba(0,0,0,.38)}.mat-menu-item[_ngcontent-%COMP%]   .mat-icon-no-color[_ngcontent-%COMP%], .mat-menu-item-submenu-trigger[_ngcontent-%COMP%]::after{color:rgba(0,0,0,.54)}.mat-menu-item[_ngcontent-%COMP%]:hover:not([disabled]), .mat-menu-item.cdk-program-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item.cdk-keyboard-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item-highlighted[_ngcontent-%COMP%]:not([disabled]){background:rgba(0,0,0,.04)}.mat-paginator[_ngcontent-%COMP%]{background:#fff}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{color:#616161}.mat-paginator-decrement[_ngcontent-%COMP%], .mat-paginator-increment[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54);border-right:2px solid rgba(0,0,0,.54)}.mat-paginator-first[_ngcontent-%COMP%], .mat-paginator-last[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54)}.mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-decrement[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-increment[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-first[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-last[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f57c00}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#ff9800}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f44336}.mat-progress-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f57c00}.mat-progress-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#ff9800}.mat-progress-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f44336}.mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#616161}.mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f57c00}.mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#ff9800}.mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#ff9800}.mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f44336}.mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f44336}.mat-radio-button.mat-radio-disabled.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-label-content[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-radio-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-select-value[_ngcontent-%COMP%]{color:#212121}.mat-select-placeholder[_ngcontent-%COMP%]{color:rgba(97,97,97,.42)}.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%]{color:#757575}.mat-select-arrow[_ngcontent-%COMP%]{color:#616161}.mat-select-panel[_ngcontent-%COMP%]{background:#fff}.mat-select-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-select-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple){background:rgba(0,0,0,.12)}.mat-form-field.mat-focused.mat-primary[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-invalid[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#757575}.mat-drawer-container[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-drawer[_ngcontent-%COMP%]{background-color:#fff;color:#212121}.mat-drawer.mat-drawer-push[_ngcontent-%COMP%]{background-color:#fff}.mat-drawer[_ngcontent-%COMP%]:not(.mat-drawer-side){box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-drawer-side[_ngcontent-%COMP%]{border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-backdrop.mat-drawer-shown[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.6)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.54)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.54)}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.54)}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle[_ngcontent-%COMP%]:not(.mat-checked)   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-slide-toggle-thumb[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12);background-color:#fafafa}.mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-primary[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f57c00}.mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-accent[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#ff9800}.mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-warn[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f44336}.mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.2)}.mat-slider[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%], .cdk-focused[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-disabled[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]   .mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing)   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26);background-color:transparent}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26)}.mat-slider-has-ticks[_ngcontent-%COMP%]   .mat-slider-wrapper[_ngcontent-%COMP%]::after{border-color:rgba(0,0,0,.7)}.mat-slider-horizontal[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to right, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent);background-image:-moz-repeating-linear-gradient(0.0001deg, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-slider-vertical[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to bottom, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-step-header.cdk-keyboard-focused[_ngcontent-%COMP%], .mat-step-header.cdk-program-focused[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]:hover{background-color:rgba(0,0,0,.04)}@media(hover: none){.mat-step-header[_ngcontent-%COMP%]:hover{background:none}}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-optional[_ngcontent-%COMP%]{color:#616161}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon[_ngcontent-%COMP%]{background-color:#616161;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-selected[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-done[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-edit[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-error[_ngcontent-%COMP%]{background-color:transparent;color:#f44336}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-active[_ngcontent-%COMP%]{color:#212121}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-error[_ngcontent-%COMP%]{color:#f44336}.mat-stepper-horizontal[_ngcontent-%COMP%], .mat-stepper-vertical[_ngcontent-%COMP%]{background-color:#fff}.mat-stepper-vertical-line[_ngcontent-%COMP%]::before{border-left-color:rgba(0,0,0,.12)}.mat-horizontal-stepper-header[_ngcontent-%COMP%]::before, .mat-horizontal-stepper-header[_ngcontent-%COMP%]::after, .mat-stepper-horizontal-line[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-sort-header-arrow[_ngcontent-%COMP%]{color:#616161}.mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-header[_ngcontent-%COMP%]{border-bottom:1px solid rgba(0,0,0,.12)}.mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%]{border-top:1px solid rgba(0,0,0,.12);border-bottom:none}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{color:#212121}.mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:#757575}.mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#212121}.mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#757575}.mat-tab-group[class*=mat-background-][_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar[class*=mat-background-][_ngcontent-%COMP%]{border-bottom:none;border-top:none}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-toolbar[_ngcontent-%COMP%]{background:#f5f5f5;color:#212121}.mat-toolbar.mat-primary[_ngcontent-%COMP%]{background:#f57c00;color:#fff}.mat-toolbar.mat-accent[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-toolbar.mat-warn[_ngcontent-%COMP%]{background:#f44336;color:#fff}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:currentColor}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:inherit}.mat-toolbar[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:currentColor}.mat-tooltip[_ngcontent-%COMP%]{background:rgba(97,97,97,.9)}.mat-tree[_ngcontent-%COMP%]{background:#fff}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{color:#212121}.mat-snack-bar-container[_ngcontent-%COMP%]{color:rgba(255,255,255,.7);background:#323232;box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-simple-snackbar-action[_ngcontent-%COMP%]{color:#ff9800}[_nghost-%COMP%]   button.mat-stroked-button[_ngcontent-%COMP%]{background-color:#ff9800;border:1px solid #ebebeb}.button-contents[_ngcontent-%COMP%]{align-items:center;display:flex;text-transform:uppercase}mat-icon[_ngcontent-%COMP%]{margin-right:6px}"] });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/header/reload_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const isReloadDisabledByPlugin = createSelector(getPlugins, getActivePlugin, (/**
     * @param {?} plugins
     * @param {?} id
     * @return {?}
     */
    (plugins, id) => {
        if (!id || !plugins[id])
            return false;
        return plugins[id].disable_reload;
    }));
    class ReloadContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.reloadDisabled$ = this.store.select(isReloadDisabledByPlugin);
            this.isReloading$ = this.store
                .select(getPluginsListLoaded)
                .pipe(withLatestFrom(this.reloadDisabled$), map((/**
             * @param {?} __0
             * @return {?}
             */
            ([loaded, reloadDisabled]) => {
                return !reloadDisabled && loaded.state === DataLoadState.LOADING;
            })));
            this.lastLoadedTimeInMs$ = this.store
                .select(getPluginsListLoaded)
                .pipe(map((/**
             * @param {?} loaded
             * @return {?}
             */
            (loaded) => {
                return loaded.lastLoadedTimeInMs;
            })));
        }
        /**
         * @return {?}
         */
        triggerReload() {
            this.store.dispatch(manualReload());
        }
        /**
         * @param {?} dateString
         * @return {?}
         */
        getReloadTitle(dateString) {
            if (!dateString) {
                return 'Loading...';
            }
            return `Last Updated: ${dateString}`;
        }
    }
    ReloadContainer.decorators = [
        { type: Component, args: [{
                    selector: 'app-header-reload',
                    template: `
    <button
      class="reload-button"
      [class.loading]="isReloading$ | async"
      mat-icon-button
      (click)="triggerReload()"
      [title]="getReloadTitle(lastLoadedTimeInMs$ | async | date: 'medium')"
      [disabled]="reloadDisabled$ | async"
    >
      <mat-icon class="refresh-icon" svgIcon="refresh_24px"></mat-icon>
    </button>
  `,
                    styles: [
                        `
      .reload-button,
      .refresh-icon {
        align-items: center;
        display: flex;
        justify-content: center;
      }

      .reload-button.loading {
        animation: rotate 2s linear infinite;
      }

      @keyframes rotate {
        0% {
          transform: rotate(0deg);
        }
        50% {
          transform: rotate(180deg);
        }
        100% {
          transform: rotate(360deg);
        }
      }
    `,
                    ],
                },] },
    ];
    /** @nocollapse */
    ReloadContainer.ctorParameters = () => [
        { type: Store }
    ];
    /** @nocollapse */ ReloadContainer.ɵfac = function ReloadContainer_Factory(t) { return new (t || ReloadContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ ReloadContainer.ɵcmp = ɵɵdefineComponent({ type: ReloadContainer, selectors: [["app-header-reload"]], decls: 6, vars: 13, consts: [["mat-icon-button", "", 1, "reload-button", 3, "title", "disabled", "click"], ["svgIcon", "refresh_24px", 1, "refresh-icon"]], template: function ReloadContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "button", 0);
            ɵɵlistener("click", function ReloadContainer_Template_button_click_0_listener() { return ctx.triggerReload(); });
            ɵɵpipe(1, "async");
            ɵɵpipe(2, "date");
            ɵɵpipe(3, "async");
            ɵɵpipe(4, "async");
            ɵɵelement(5, "mat-icon", 1);
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵclassProp("loading", ɵɵpipeBind1(1, 4, ctx.isReloading$));
            ɵɵproperty("title", ctx.getReloadTitle(ɵɵpipeBind2(2, 6, ɵɵpipeBind1(3, 9, ctx.lastLoadedTimeInMs$), "medium")))("disabled", ɵɵpipeBind1(4, 11, ctx.reloadDisabled$));
        } }, directives: [MatButton, MatIcon], pipes: [AsyncPipe, DatePipe], styles: [".reload-button[_ngcontent-%COMP%], .refresh-icon[_ngcontent-%COMP%] {\n        align-items: center;\n        display: flex;\n        justify-content: center;\n      }\n\n      .reload-button.loading[_ngcontent-%COMP%] {\n        animation: rotate 2s linear infinite;\n      }\n\n      @keyframes rotate {\n        0% {\n          transform: rotate(0deg);\n        }\n        50% {\n          transform: rotate(180deg);\n        }\n        100% {\n          transform: rotate(360deg);\n        }\n      }"] });

    const _c0$8 = ["input"];
    const _c1$7 = function () { return { enterDuration: 150 }; };
    const _c2$4 = ["*"];
    /**
     * Injection token to be used to override the default options for `mat-checkbox`.
     * @type {?}
     */
    const MAT_CHECKBOX_DEFAULT_OPTIONS = new InjectionToken('mat-checkbox-default-options', {
        providedIn: 'root',
        factory: MAT_CHECKBOX_DEFAULT_OPTIONS_FACTORY
    });
    /**
     * \@docs-private
     * @return {?}
     */
    function MAT_CHECKBOX_DEFAULT_OPTIONS_FACTORY() {
        return {
            color: 'accent',
            clickAction: 'check-indeterminate',
        };
    }
    /**
     * Injection token that can be used to specify the checkbox click behavior.
     * @deprecated Injection token will be removed, use `MAT_CHECKBOX_DEFAULT_OPTIONS` instead.
     * \@breaking-change 10.0.0
     * @type {?}
     */
    const MAT_CHECKBOX_CLICK_ACTION = new InjectionToken('mat-checkbox-click-action');

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/checkbox/checkbox.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    // Increasing integer for generating unique ids for checkbox components.
    /** @type {?} */
    let nextUniqueId$5 = 0;
    /**
     * Provider Expression that allows mat-checkbox to register as a ControlValueAccessor.
     * This allows it to support [(ngModel)].
     * \@docs-private
     * @type {?}
     */
    const MAT_CHECKBOX_CONTROL_VALUE_ACCESSOR = {
        provide: NG_VALUE_ACCESSOR,
        useExisting: forwardRef((/**
         * @return {?}
         */
        () => MatCheckbox)),
        multi: true
    };
    /**
     * Change event object emitted by MatCheckbox.
     */
    class MatCheckboxChange {
    }
    // Boilerplate for applying mixins to MatCheckbox.
    /**
     * \@docs-private
     */
    class MatCheckboxBase {
        /**
         * @param {?} _elementRef
         */
        constructor(_elementRef) {
            this._elementRef = _elementRef;
        }
    }
    /** @type {?} */
    const _MatCheckboxMixinBase = mixinTabIndex(mixinColor(mixinDisableRipple(mixinDisabled(MatCheckboxBase))));
    /**
     * A material design checkbox component. Supports all of the functionality of an HTML5 checkbox,
     * and exposes a similar API. A MatCheckbox can be either checked, unchecked, indeterminate, or
     * disabled. Note that all additional accessibility attributes are taken care of by the component,
     * so there is no need to provide them yourself. However, if you want to omit a label and still
     * have the checkbox be accessible, you may supply an [aria-label] input.
     * See: https://material.io/design/components/selection-controls.html
     */
    class MatCheckbox extends _MatCheckboxMixinBase {
        /**
         * @param {?} elementRef
         * @param {?} _changeDetectorRef
         * @param {?} _focusMonitor
         * @param {?} _ngZone
         * @param {?} tabIndex
         * @param {?} _clickAction
         * @param {?=} _animationMode
         * @param {?=} _options
         */
        constructor(elementRef, _changeDetectorRef, _focusMonitor, _ngZone, tabIndex, _clickAction, _animationMode, _options) {
            super(elementRef);
            this._changeDetectorRef = _changeDetectorRef;
            this._focusMonitor = _focusMonitor;
            this._ngZone = _ngZone;
            this._clickAction = _clickAction;
            this._animationMode = _animationMode;
            this._options = _options;
            /**
             * Attached to the aria-label attribute of the host element. In most cases, aria-labelledby will
             * take precedence so this may be omitted.
             */
            this.ariaLabel = '';
            /**
             * Users can specify the `aria-labelledby` attribute which will be forwarded to the input element
             */
            this.ariaLabelledby = null;
            this._uniqueId = `mat-checkbox-${++nextUniqueId$5}`;
            /**
             * A unique id for the checkbox input. If none is supplied, it will be auto-generated.
             */
            this.id = this._uniqueId;
            /**
             * Whether the label should appear after or before the checkbox. Defaults to 'after'
             */
            this.labelPosition = 'after';
            /**
             * Name value will be applied to the input element if present
             */
            this.name = null;
            /**
             * Event emitted when the checkbox's `checked` value changes.
             */
            this.change = new EventEmitter();
            /**
             * Event emitted when the checkbox's `indeterminate` value changes.
             */
            this.indeterminateChange = new EventEmitter();
            /**
             * Called when the checkbox is blurred. Needed to properly implement ControlValueAccessor.
             * \@docs-private
             */
            this._onTouched = (/**
             * @return {?}
             */
            () => { });
            this._currentAnimationClass = '';
            this._currentCheckState = 0 /* Init */;
            this._controlValueAccessorChangeFn = (/**
             * @return {?}
             */
            () => { });
            this._checked = false;
            this._disabled = false;
            this._indeterminate = false;
            this._options = this._options || {};
            if (this._options.color) {
                this.color = this._options.color;
            }
            this.tabIndex = parseInt(tabIndex) || 0;
            this._focusMonitor.monitor(elementRef, true).subscribe((/**
             * @param {?} focusOrigin
             * @return {?}
             */
            focusOrigin => {
                if (!focusOrigin) {
                    // When a focused element becomes disabled, the browser *immediately* fires a blur event.
                    // Angular does not expect events to be raised during change detection, so any state change
                    // (such as a form control's 'ng-touched') will cause a changed-after-checked error.
                    // See https://github.com/angular/angular/issues/17793. To work around this, we defer
                    // telling the form control it has been touched until the next tick.
                    Promise.resolve().then((/**
                     * @return {?}
                     */
                    () => {
                        this._onTouched();
                        _changeDetectorRef.markForCheck();
                    }));
                }
            }));
            // TODO: Remove this after the `_clickAction` parameter is removed as an injection parameter.
            this._clickAction = this._clickAction || this._options.clickAction;
        }
        /**
         * Returns the unique id for the visual hidden input.
         * @return {?}
         */
        get inputId() { return `${this.id || this._uniqueId}-input`; }
        /**
         * Whether the checkbox is required.
         * @return {?}
         */
        get required() { return this._required; }
        /**
         * @param {?} value
         * @return {?}
         */
        set required(value) { this._required = coerceBooleanProperty(value); }
        /**
         * @return {?}
         */
        ngAfterViewInit() {
            this._syncIndeterminate(this._indeterminate);
        }
        // TODO: Delete next major revision.
        /**
         * @return {?}
         */
        ngAfterViewChecked() { }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this._focusMonitor.stopMonitoring(this._elementRef);
        }
        /**
         * Whether the checkbox is checked.
         * @return {?}
         */
        get checked() { return this._checked; }
        /**
         * @param {?} value
         * @return {?}
         */
        set checked(value) {
            if (value != this.checked) {
                this._checked = value;
                this._changeDetectorRef.markForCheck();
            }
        }
        /**
         * Whether the checkbox is disabled. This fully overrides the implementation provided by
         * mixinDisabled, but the mixin is still required because mixinTabIndex requires it.
         * @return {?}
         */
        get disabled() { return this._disabled; }
        /**
         * @param {?} value
         * @return {?}
         */
        set disabled(value) {
            /** @type {?} */
            const newValue = coerceBooleanProperty(value);
            if (newValue !== this.disabled) {
                this._disabled = newValue;
                this._changeDetectorRef.markForCheck();
            }
        }
        /**
         * Whether the checkbox is indeterminate. This is also known as "mixed" mode and can be used to
         * represent a checkbox with three states, e.g. a checkbox that represents a nested list of
         * checkable items. Note that whenever checkbox is manually clicked, indeterminate is immediately
         * set to false.
         * @return {?}
         */
        get indeterminate() { return this._indeterminate; }
        /**
         * @param {?} value
         * @return {?}
         */
        set indeterminate(value) {
            /** @type {?} */
            const changed = value != this._indeterminate;
            this._indeterminate = coerceBooleanProperty(value);
            if (changed) {
                if (this._indeterminate) {
                    this._transitionCheckState(3 /* Indeterminate */);
                }
                else {
                    this._transitionCheckState(this.checked ? 1 /* Checked */ : 2 /* Unchecked */);
                }
                this.indeterminateChange.emit(this._indeterminate);
            }
            this._syncIndeterminate(this._indeterminate);
        }
        /**
         * @return {?}
         */
        _isRippleDisabled() {
            return this.disableRipple || this.disabled;
        }
        /**
         * Method being called whenever the label text changes.
         * @return {?}
         */
        _onLabelTextChange() {
            // Since the event of the `cdkObserveContent` directive runs outside of the zone, the checkbox
            // component will be only marked for check, but no actual change detection runs automatically.
            // Instead of going back into the zone in order to trigger a change detection which causes
            // *all* components to be checked (if explicitly marked or not using OnPush), we only trigger
            // an explicit change detection for the checkbox view and its children.
            this._changeDetectorRef.detectChanges();
        }
        // Implemented as part of ControlValueAccessor.
        /**
         * @param {?} value
         * @return {?}
         */
        writeValue(value) {
            this.checked = !!value;
        }
        // Implemented as part of ControlValueAccessor.
        /**
         * @param {?} fn
         * @return {?}
         */
        registerOnChange(fn) {
            this._controlValueAccessorChangeFn = fn;
        }
        // Implemented as part of ControlValueAccessor.
        /**
         * @param {?} fn
         * @return {?}
         */
        registerOnTouched(fn) {
            this._onTouched = fn;
        }
        // Implemented as part of ControlValueAccessor.
        /**
         * @param {?} isDisabled
         * @return {?}
         */
        setDisabledState(isDisabled) {
            this.disabled = isDisabled;
        }
        /**
         * @return {?}
         */
        _getAriaChecked() {
            if (this.checked) {
                return 'true';
            }
            return this.indeterminate ? 'mixed' : 'false';
        }
        /**
         * @private
         * @param {?} newState
         * @return {?}
         */
        _transitionCheckState(newState) {
            /** @type {?} */
            let oldState = this._currentCheckState;
            /** @type {?} */
            let element = this._elementRef.nativeElement;
            if (oldState === newState) {
                return;
            }
            if (this._currentAnimationClass.length > 0) {
                element.classList.remove(this._currentAnimationClass);
            }
            this._currentAnimationClass = this._getAnimationClassForCheckStateTransition(oldState, newState);
            this._currentCheckState = newState;
            if (this._currentAnimationClass.length > 0) {
                element.classList.add(this._currentAnimationClass);
                // Remove the animation class to avoid animation when the checkbox is moved between containers
                /** @type {?} */
                const animationClass = this._currentAnimationClass;
                this._ngZone.runOutsideAngular((/**
                 * @return {?}
                 */
                () => {
                    setTimeout((/**
                     * @return {?}
                     */
                    () => {
                        element.classList.remove(animationClass);
                    }), 1000);
                }));
            }
        }
        /**
         * @private
         * @return {?}
         */
        _emitChangeEvent() {
            /** @type {?} */
            const event = new MatCheckboxChange();
            event.source = this;
            event.checked = this.checked;
            this._controlValueAccessorChangeFn(this.checked);
            this.change.emit(event);
        }
        /**
         * Toggles the `checked` state of the checkbox.
         * @return {?}
         */
        toggle() {
            this.checked = !this.checked;
        }
        /**
         * Event handler for checkbox input element.
         * Toggles checked state if element is not disabled.
         * Do not toggle on (change) event since IE doesn't fire change event when
         *   indeterminate checkbox is clicked.
         * @param {?} event
         * @return {?}
         */
        _onInputClick(event) {
            // We have to stop propagation for click events on the visual hidden input element.
            // By default, when a user clicks on a label element, a generated click event will be
            // dispatched on the associated input element. Since we are using a label element as our
            // root container, the click event on the `checkbox` will be executed twice.
            // The real click event will bubble up, and the generated click event also tries to bubble up.
            // This will lead to multiple click events.
            // Preventing bubbling for the second event will solve that issue.
            event.stopPropagation();
            // If resetIndeterminate is false, and the current state is indeterminate, do nothing on click
            if (!this.disabled && this._clickAction !== 'noop') {
                // When user manually click on the checkbox, `indeterminate` is set to false.
                if (this.indeterminate && this._clickAction !== 'check') {
                    Promise.resolve().then((/**
                     * @return {?}
                     */
                    () => {
                        this._indeterminate = false;
                        this.indeterminateChange.emit(this._indeterminate);
                    }));
                }
                this.toggle();
                this._transitionCheckState(this._checked ? 1 /* Checked */ : 2 /* Unchecked */);
                // Emit our custom change event if the native input emitted one.
                // It is important to only emit it, if the native input triggered one, because
                // we don't want to trigger a change event, when the `checked` variable changes for example.
                this._emitChangeEvent();
            }
            else if (!this.disabled && this._clickAction === 'noop') {
                // Reset native input when clicked with noop. The native checkbox becomes checked after
                // click, reset it to be align with `checked` value of `mat-checkbox`.
                this._inputElement.nativeElement.checked = this.checked;
                this._inputElement.nativeElement.indeterminate = this.indeterminate;
            }
        }
        /**
         * Focuses the checkbox.
         * @param {?=} origin
         * @param {?=} options
         * @return {?}
         */
        focus(origin = 'keyboard', options) {
            this._focusMonitor.focusVia(this._inputElement, origin, options);
        }
        /**
         * @param {?} event
         * @return {?}
         */
        _onInteractionEvent(event) {
            // We always have to stop propagation on the change event.
            // Otherwise the change event, from the input element, will bubble up and
            // emit its event object to the `change` output.
            event.stopPropagation();
        }
        /**
         * @private
         * @param {?} oldState
         * @param {?} newState
         * @return {?}
         */
        _getAnimationClassForCheckStateTransition(oldState, newState) {
            // Don't transition if animations are disabled.
            if (this._animationMode === 'NoopAnimations') {
                return '';
            }
            /** @type {?} */
            let animSuffix = '';
            switch (oldState) {
                case 0 /* Init */:
                    // Handle edge case where user interacts with checkbox that does not have [(ngModel)] or
                    // [checked] bound to it.
                    if (newState === 1 /* Checked */) {
                        animSuffix = 'unchecked-checked';
                    }
                    else if (newState == 3 /* Indeterminate */) {
                        animSuffix = 'unchecked-indeterminate';
                    }
                    else {
                        return '';
                    }
                    break;
                case 2 /* Unchecked */:
                    animSuffix = newState === 1 /* Checked */ ?
                        'unchecked-checked' : 'unchecked-indeterminate';
                    break;
                case 1 /* Checked */:
                    animSuffix = newState === 2 /* Unchecked */ ?
                        'checked-unchecked' : 'checked-indeterminate';
                    break;
                case 3 /* Indeterminate */:
                    animSuffix = newState === 1 /* Checked */ ?
                        'indeterminate-checked' : 'indeterminate-unchecked';
                    break;
            }
            return `mat-checkbox-anim-${animSuffix}`;
        }
        /**
         * Syncs the indeterminate value with the checkbox DOM node.
         *
         * We sync `indeterminate` directly on the DOM node, because in Ivy the check for whether a
         * property is supported on an element boils down to `if (propName in element)`. Domino's
         * HTMLInputElement doesn't have an `indeterminate` property so Ivy will warn during
         * server-side rendering.
         * @private
         * @param {?} value
         * @return {?}
         */
        _syncIndeterminate(value) {
            /** @type {?} */
            const nativeCheckbox = this._inputElement;
            if (nativeCheckbox) {
                nativeCheckbox.nativeElement.indeterminate = value;
            }
        }
    }
    MatCheckbox.ɵfac = function MatCheckbox_Factory(t) { return new (t || MatCheckbox)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(ChangeDetectorRef), ɵɵdirectiveInject(FocusMonitor), ɵɵdirectiveInject(NgZone), ɵɵinjectAttribute('tabindex'), ɵɵdirectiveInject(MAT_CHECKBOX_CLICK_ACTION, 8), ɵɵdirectiveInject(ANIMATION_MODULE_TYPE, 8), ɵɵdirectiveInject(MAT_CHECKBOX_DEFAULT_OPTIONS, 8)); };
    MatCheckbox.ɵcmp = ɵɵdefineComponent({ type: MatCheckbox, selectors: [["mat-checkbox"]], viewQuery: function MatCheckbox_Query(rf, ctx) { if (rf & 1) {
            ɵɵviewQuery(_c0$8, true);
            ɵɵviewQuery(MatRipple, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._inputElement = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.ripple = _t.first);
        } }, hostAttrs: [1, "mat-checkbox"], hostVars: 12, hostBindings: function MatCheckbox_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵhostProperty("id", ctx.id);
            ɵɵattribute("tabindex", null);
            ɵɵclassProp("mat-checkbox-indeterminate", ctx.indeterminate)("mat-checkbox-checked", ctx.checked)("mat-checkbox-disabled", ctx.disabled)("mat-checkbox-label-before", ctx.labelPosition == "before")("_mat-animation-noopable", ctx._animationMode === "NoopAnimations");
        } }, inputs: { disableRipple: "disableRipple", color: "color", tabIndex: "tabIndex", ariaLabel: ["aria-label", "ariaLabel"], ariaLabelledby: ["aria-labelledby", "ariaLabelledby"], id: "id", labelPosition: "labelPosition", name: "name", required: "required", checked: "checked", disabled: "disabled", indeterminate: "indeterminate", value: "value" }, outputs: { change: "change", indeterminateChange: "indeterminateChange" }, exportAs: ["matCheckbox"], features: [ɵɵProvidersFeature([MAT_CHECKBOX_CONTROL_VALUE_ACCESSOR]), ɵɵInheritDefinitionFeature], ngContentSelectors: _c2$4, decls: 17, vars: 19, consts: [[1, "mat-checkbox-layout"], ["label", ""], [1, "mat-checkbox-inner-container"], ["type", "checkbox", 1, "mat-checkbox-input", "cdk-visually-hidden", 3, "id", "required", "checked", "disabled", "tabIndex", "change", "click"], ["input", ""], ["matRipple", "", 1, "mat-checkbox-ripple", "mat-focus-indicator", 3, "matRippleTrigger", "matRippleDisabled", "matRippleRadius", "matRippleCentered", "matRippleAnimation"], [1, "mat-ripple-element", "mat-checkbox-persistent-ripple"], [1, "mat-checkbox-frame"], [1, "mat-checkbox-background"], ["version", "1.1", "focusable", "false", "viewBox", "0 0 24 24", 0, "xml", "space", "preserve", 1, "mat-checkbox-checkmark"], ["fill", "none", "stroke", "white", "d", "M4.1,12.7 9,17.6 20.3,6.3", 1, "mat-checkbox-checkmark-path"], [1, "mat-checkbox-mixedmark"], [1, "mat-checkbox-label", 3, "cdkObserveContent"], ["checkboxLabel", ""], [2, "display", "none"]], template: function MatCheckbox_Template(rf, ctx) { if (rf & 1) {
            ɵɵprojectionDef();
            ɵɵelementStart(0, "label", 0, 1);
            ɵɵelementStart(2, "div", 2);
            ɵɵelementStart(3, "input", 3, 4);
            ɵɵlistener("change", function MatCheckbox_Template_input_change_3_listener($event) { return ctx._onInteractionEvent($event); })("click", function MatCheckbox_Template_input_click_3_listener($event) { return ctx._onInputClick($event); });
            ɵɵelementEnd();
            ɵɵelementStart(5, "div", 5);
            ɵɵelement(6, "div", 6);
            ɵɵelementEnd();
            ɵɵelement(7, "div", 7);
            ɵɵelementStart(8, "div", 8);
            ɵɵnamespaceSVG();
            ɵɵelementStart(9, "svg", 9);
            ɵɵelement(10, "path", 10);
            ɵɵelementEnd();
            ɵɵnamespaceHTML();
            ɵɵelement(11, "div", 11);
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementStart(12, "span", 12, 13);
            ɵɵlistener("cdkObserveContent", function MatCheckbox_Template_span_cdkObserveContent_12_listener() { return ctx._onLabelTextChange(); });
            ɵɵelementStart(14, "span", 14);
            ɵɵtext(15, "\u00A0");
            ɵɵelementEnd();
            ɵɵprojection(16);
            ɵɵelementEnd();
            ɵɵelementEnd();
        } if (rf & 2) {
            const _r0 = ɵɵreference(1);
            const _r2 = ɵɵreference(13);
            ɵɵattribute("for", ctx.inputId);
            ɵɵadvance(2);
            ɵɵclassProp("mat-checkbox-inner-container-no-side-margin", !_r2.textContent || !_r2.textContent.trim());
            ɵɵadvance(1);
            ɵɵproperty("id", ctx.inputId)("required", ctx.required)("checked", ctx.checked)("disabled", ctx.disabled)("tabIndex", ctx.tabIndex);
            ɵɵattribute("value", ctx.value)("name", ctx.name)("aria-label", ctx.ariaLabel || null)("aria-labelledby", ctx.ariaLabelledby)("aria-checked", ctx._getAriaChecked());
            ɵɵadvance(2);
            ɵɵproperty("matRippleTrigger", _r0)("matRippleDisabled", ctx._isRippleDisabled())("matRippleRadius", 20)("matRippleCentered", true)("matRippleAnimation", ɵɵpureFunction0(18, _c1$7));
        } }, directives: [MatRipple, CdkObserveContent], styles: ["@keyframes mat-checkbox-fade-in-background{0%{opacity:0}50%{opacity:1}}@keyframes mat-checkbox-fade-out-background{0%,50%{opacity:1}100%{opacity:0}}@keyframes mat-checkbox-unchecked-checked-checkmark-path{0%,50%{stroke-dashoffset:22.910259}50%{animation-timing-function:cubic-bezier(0, 0, 0.2, 0.1)}100%{stroke-dashoffset:0}}@keyframes mat-checkbox-unchecked-indeterminate-mixedmark{0%,68.2%{transform:scaleX(0)}68.2%{animation-timing-function:cubic-bezier(0, 0, 0, 1)}100%{transform:scaleX(1)}}@keyframes mat-checkbox-checked-unchecked-checkmark-path{from{animation-timing-function:cubic-bezier(0.4, 0, 1, 1);stroke-dashoffset:0}to{stroke-dashoffset:-22.910259}}@keyframes mat-checkbox-checked-indeterminate-checkmark{from{animation-timing-function:cubic-bezier(0, 0, 0.2, 0.1);opacity:1;transform:rotate(0deg)}to{opacity:0;transform:rotate(45deg)}}@keyframes mat-checkbox-indeterminate-checked-checkmark{from{animation-timing-function:cubic-bezier(0.14, 0, 0, 1);opacity:0;transform:rotate(45deg)}to{opacity:1;transform:rotate(360deg)}}@keyframes mat-checkbox-checked-indeterminate-mixedmark{from{animation-timing-function:cubic-bezier(0, 0, 0.2, 0.1);opacity:0;transform:rotate(-45deg)}to{opacity:1;transform:rotate(0deg)}}@keyframes mat-checkbox-indeterminate-checked-mixedmark{from{animation-timing-function:cubic-bezier(0.14, 0, 0, 1);opacity:1;transform:rotate(0deg)}to{opacity:0;transform:rotate(315deg)}}@keyframes mat-checkbox-indeterminate-unchecked-mixedmark{0%{animation-timing-function:linear;opacity:1;transform:scaleX(1)}32.8%,100%{opacity:0;transform:scaleX(0)}}.mat-checkbox-background,.mat-checkbox-frame{top:0;left:0;right:0;bottom:0;position:absolute;border-radius:2px;box-sizing:border-box;pointer-events:none}.mat-checkbox{transition:background 400ms cubic-bezier(0.25, 0.8, 0.25, 1),box-shadow 280ms cubic-bezier(0.4, 0, 0.2, 1);cursor:pointer;-webkit-tap-highlight-color:transparent}._mat-animation-noopable.mat-checkbox{transition:none;animation:none}.mat-checkbox .mat-ripple-element:not(.mat-checkbox-persistent-ripple){opacity:.16}.mat-checkbox-layout{-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;cursor:inherit;align-items:baseline;vertical-align:middle;display:inline-flex;white-space:nowrap}.mat-checkbox-label{-webkit-user-select:auto;-moz-user-select:auto;-ms-user-select:auto;user-select:auto}.mat-checkbox-inner-container{display:inline-block;height:16px;line-height:0;margin:auto;margin-right:8px;order:0;position:relative;vertical-align:middle;white-space:nowrap;width:16px;flex-shrink:0}[dir=rtl] .mat-checkbox-inner-container{margin-left:8px;margin-right:auto}.mat-checkbox-inner-container-no-side-margin{margin-left:0;margin-right:0}.mat-checkbox-frame{background-color:transparent;transition:border-color 90ms cubic-bezier(0, 0, 0.2, 0.1);border-width:2px;border-style:solid}._mat-animation-noopable .mat-checkbox-frame{transition:none}.mat-checkbox.cdk-keyboard-focused .cdk-high-contrast-active .mat-checkbox-frame{border-style:dotted}.mat-checkbox-background{align-items:center;display:inline-flex;justify-content:center;transition:background-color 90ms cubic-bezier(0, 0, 0.2, 0.1),opacity 90ms cubic-bezier(0, 0, 0.2, 0.1)}._mat-animation-noopable .mat-checkbox-background{transition:none}.cdk-high-contrast-active .mat-checkbox .mat-checkbox-background{background:none}.mat-checkbox-persistent-ripple{width:100%;height:100%;transform:none}.mat-checkbox-inner-container:hover .mat-checkbox-persistent-ripple{opacity:.04}.mat-checkbox.cdk-keyboard-focused .mat-checkbox-persistent-ripple{opacity:.12}.mat-checkbox-persistent-ripple,.mat-checkbox.mat-checkbox-disabled .mat-checkbox-inner-container:hover .mat-checkbox-persistent-ripple{opacity:0}@media(hover: none){.mat-checkbox-inner-container:hover .mat-checkbox-persistent-ripple{display:none}}.mat-checkbox-checkmark{top:0;left:0;right:0;bottom:0;position:absolute;width:100%}.mat-checkbox-checkmark-path{stroke-dashoffset:22.910259;stroke-dasharray:22.910259;stroke-width:2.1333333333px}.cdk-high-contrast-black-on-white .mat-checkbox-checkmark-path{stroke:#000 !important}.mat-checkbox-mixedmark{width:calc(100% - 6px);height:2px;opacity:0;transform:scaleX(0) rotate(0deg);border-radius:2px}.cdk-high-contrast-active .mat-checkbox-mixedmark{height:0;border-top:solid 2px;margin-top:2px}.mat-checkbox-label-before .mat-checkbox-inner-container{order:1;margin-left:8px;margin-right:auto}[dir=rtl] .mat-checkbox-label-before .mat-checkbox-inner-container{margin-left:auto;margin-right:8px}.mat-checkbox-checked .mat-checkbox-checkmark{opacity:1}.mat-checkbox-checked .mat-checkbox-checkmark-path{stroke-dashoffset:0}.mat-checkbox-checked .mat-checkbox-mixedmark{transform:scaleX(1) rotate(-45deg)}.mat-checkbox-indeterminate .mat-checkbox-checkmark{opacity:0;transform:rotate(45deg)}.mat-checkbox-indeterminate .mat-checkbox-checkmark-path{stroke-dashoffset:0}.mat-checkbox-indeterminate .mat-checkbox-mixedmark{opacity:1;transform:scaleX(1) rotate(0deg)}.mat-checkbox-unchecked .mat-checkbox-background{background-color:transparent}.mat-checkbox-disabled{cursor:default}.cdk-high-contrast-active .mat-checkbox-disabled{opacity:.5}.mat-checkbox-anim-unchecked-checked .mat-checkbox-background{animation:180ms linear 0ms mat-checkbox-fade-in-background}.mat-checkbox-anim-unchecked-checked .mat-checkbox-checkmark-path{animation:180ms linear 0ms mat-checkbox-unchecked-checked-checkmark-path}.mat-checkbox-anim-unchecked-indeterminate .mat-checkbox-background{animation:180ms linear 0ms mat-checkbox-fade-in-background}.mat-checkbox-anim-unchecked-indeterminate .mat-checkbox-mixedmark{animation:90ms linear 0ms mat-checkbox-unchecked-indeterminate-mixedmark}.mat-checkbox-anim-checked-unchecked .mat-checkbox-background{animation:180ms linear 0ms mat-checkbox-fade-out-background}.mat-checkbox-anim-checked-unchecked .mat-checkbox-checkmark-path{animation:90ms linear 0ms mat-checkbox-checked-unchecked-checkmark-path}.mat-checkbox-anim-checked-indeterminate .mat-checkbox-checkmark{animation:90ms linear 0ms mat-checkbox-checked-indeterminate-checkmark}.mat-checkbox-anim-checked-indeterminate .mat-checkbox-mixedmark{animation:90ms linear 0ms mat-checkbox-checked-indeterminate-mixedmark}.mat-checkbox-anim-indeterminate-checked .mat-checkbox-checkmark{animation:500ms linear 0ms mat-checkbox-indeterminate-checked-checkmark}.mat-checkbox-anim-indeterminate-checked .mat-checkbox-mixedmark{animation:500ms linear 0ms mat-checkbox-indeterminate-checked-mixedmark}.mat-checkbox-anim-indeterminate-unchecked .mat-checkbox-background{animation:180ms linear 0ms mat-checkbox-fade-out-background}.mat-checkbox-anim-indeterminate-unchecked .mat-checkbox-mixedmark{animation:300ms linear 0ms mat-checkbox-indeterminate-unchecked-mixedmark}.mat-checkbox-input{bottom:0;left:50%}.mat-checkbox .mat-checkbox-ripple{position:absolute;left:calc(50% - 20px);top:calc(50% - 20px);height:40px;width:40px;z-index:1;pointer-events:none}\n"], encapsulation: 2, changeDetection: 0 });
    /** @nocollapse */
    MatCheckbox.ctorParameters = () => [
        { type: ElementRef },
        { type: ChangeDetectorRef },
        { type: FocusMonitor },
        { type: NgZone },
        { type: String, decorators: [{ type: Attribute, args: ['tabindex',] }] },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [MAT_CHECKBOX_CLICK_ACTION,] }] },
        { type: String, decorators: [{ type: Optional }, { type: Inject, args: [ANIMATION_MODULE_TYPE,] }] },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [MAT_CHECKBOX_DEFAULT_OPTIONS,] }] }
    ];
    MatCheckbox.propDecorators = {
        ariaLabel: [{ type: Input, args: ['aria-label',] }],
        ariaLabelledby: [{ type: Input, args: ['aria-labelledby',] }],
        id: [{ type: Input }],
        required: [{ type: Input }],
        labelPosition: [{ type: Input }],
        name: [{ type: Input }],
        change: [{ type: Output }],
        indeterminateChange: [{ type: Output }],
        value: [{ type: Input }],
        _inputElement: [{ type: ViewChild, args: ['input',] }],
        ripple: [{ type: ViewChild, args: [MatRipple,] }],
        checked: [{ type: Input }],
        disabled: [{ type: Input }],
        indeterminate: [{ type: Input }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/checkbox/checkbox-required-validator.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const MAT_CHECKBOX_REQUIRED_VALIDATOR = {
        provide: NG_VALIDATORS,
        useExisting: forwardRef((/**
         * @return {?}
         */
        () => MatCheckboxRequiredValidator)),
        multi: true
    };
    /**
     * Validator for Material checkbox's required attribute in template-driven checkbox.
     * Current CheckboxRequiredValidator only work with `input type=checkbox` and does not
     * work with `mat-checkbox`.
     */
    class MatCheckboxRequiredValidator extends CheckboxRequiredValidator {
    }
    MatCheckboxRequiredValidator.ɵfac = function MatCheckboxRequiredValidator_Factory(t) { return ɵMatCheckboxRequiredValidator_BaseFactory(t || MatCheckboxRequiredValidator); };
    MatCheckboxRequiredValidator.ɵdir = ɵɵdefineDirective({ type: MatCheckboxRequiredValidator, selectors: [["mat-checkbox", "required", "", "formControlName", ""], ["mat-checkbox", "required", "", "formControl", ""], ["mat-checkbox", "required", "", "ngModel", ""]], features: [ɵɵProvidersFeature([MAT_CHECKBOX_REQUIRED_VALIDATOR]), ɵɵInheritDefinitionFeature] });
    const ɵMatCheckboxRequiredValidator_BaseFactory = ɵɵgetInheritedFactory(MatCheckboxRequiredValidator);

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/checkbox/checkbox-module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * This module is used by both original and MDC-based checkbox implementations.
     */
    // tslint:disable-next-line:class-name
    class _MatCheckboxRequiredValidatorModule {
    }
    _MatCheckboxRequiredValidatorModule.ɵmod = ɵɵdefineNgModule({ type: _MatCheckboxRequiredValidatorModule });
    _MatCheckboxRequiredValidatorModule.ɵinj = ɵɵdefineInjector({ factory: function _MatCheckboxRequiredValidatorModule_Factory(t) { return new (t || _MatCheckboxRequiredValidatorModule)(); } });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(_MatCheckboxRequiredValidatorModule, { declarations: [MatCheckboxRequiredValidator], exports: [MatCheckboxRequiredValidator] }); })();
    class MatCheckboxModule {
    }
    MatCheckboxModule.ɵmod = ɵɵdefineNgModule({ type: MatCheckboxModule });
    MatCheckboxModule.ɵinj = ɵɵdefineInjector({ factory: function MatCheckboxModule_Factory(t) { return new (t || MatCheckboxModule)(); }, imports: [[
                MatRippleModule, MatCommonModule, ObserversModule,
                _MatCheckboxRequiredValidatorModule
            ],
            MatCommonModule,
            _MatCheckboxRequiredValidatorModule] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(MatCheckboxModule, { declarations: function () { return [MatCheckbox]; }, imports: function () { return [MatRippleModule, MatCommonModule, ObserversModule,
            _MatCheckboxRequiredValidatorModule]; }, exports: function () { return [MatCheckbox,
            MatCommonModule,
            _MatCheckboxRequiredValidatorModule]; } }); })();

    const listenerOptions = normalizePassiveListenerOptions({ passive: true });
    /**
     * An injectable service that can be used to monitor the autofill state of an input.
     * Based on the following blog post:
     * https://medium.com/\@brunn/detecting-autofilled-fields-in-javascript-aed598d25da7
     */
    class AutofillMonitor {
        /**
         * @param {?} _platform
         * @param {?} _ngZone
         */
        constructor(_platform, _ngZone) {
            this._platform = _platform;
            this._ngZone = _ngZone;
            this._monitoredElements = new Map();
        }
        /**
         * @param {?} elementOrRef
         * @return {?}
         */
        monitor(elementOrRef) {
            if (!this._platform.isBrowser) {
                return EMPTY;
            }
            /** @type {?} */
            const element = coerceElement(elementOrRef);
            /** @type {?} */
            const info = this._monitoredElements.get(element);
            if (info) {
                return info.subject.asObservable();
            }
            /** @type {?} */
            const result = new Subject();
            /** @type {?} */
            const cssClass = 'cdk-text-field-autofilled';
            /** @type {?} */
            const listener = (/** @type {?} */ (((/**
             * @param {?} event
             * @return {?}
             */
            (event) => {
                // Animation events fire on initial element render, we check for the presence of the autofill
                // CSS class to make sure this is a real change in state, not just the initial render before
                // we fire off events.
                if (event.animationName === 'cdk-text-field-autofill-start' &&
                    !element.classList.contains(cssClass)) {
                    element.classList.add(cssClass);
                    this._ngZone.run((/**
                     * @return {?}
                     */
                    () => result.next({ target: (/** @type {?} */ (event.target)), isAutofilled: true })));
                }
                else if (event.animationName === 'cdk-text-field-autofill-end' &&
                    element.classList.contains(cssClass)) {
                    element.classList.remove(cssClass);
                    this._ngZone.run((/**
                     * @return {?}
                     */
                    () => result.next({ target: (/** @type {?} */ (event.target)), isAutofilled: false })));
                }
            }))));
            this._ngZone.runOutsideAngular((/**
             * @return {?}
             */
            () => {
                element.addEventListener('animationstart', listener, listenerOptions);
                element.classList.add('cdk-text-field-autofill-monitored');
            }));
            this._monitoredElements.set(element, {
                subject: result,
                unlisten: (/**
                 * @return {?}
                 */
                () => {
                    element.removeEventListener('animationstart', listener, listenerOptions);
                })
            });
            return result.asObservable();
        }
        /**
         * @param {?} elementOrRef
         * @return {?}
         */
        stopMonitoring(elementOrRef) {
            /** @type {?} */
            const element = coerceElement(elementOrRef);
            /** @type {?} */
            const info = this._monitoredElements.get(element);
            if (info) {
                info.unlisten();
                info.subject.complete();
                element.classList.remove('cdk-text-field-autofill-monitored');
                element.classList.remove('cdk-text-field-autofilled');
                this._monitoredElements.delete(element);
            }
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this._monitoredElements.forEach((/**
             * @param {?} _info
             * @param {?} element
             * @return {?}
             */
            (_info, element) => this.stopMonitoring(element)));
        }
    }
    AutofillMonitor.ɵfac = function AutofillMonitor_Factory(t) { return new (t || AutofillMonitor)(ɵɵinject(Platform), ɵɵinject(NgZone)); };
    /** @nocollapse */
    AutofillMonitor.ctorParameters = () => [
        { type: Platform },
        { type: NgZone }
    ];
    /** @nocollapse */ AutofillMonitor.ɵprov = ɵɵdefineInjectable({ factory: function AutofillMonitor_Factory() { return new AutofillMonitor(ɵɵinject(Platform), ɵɵinject(NgZone)); }, token: AutofillMonitor, providedIn: "root" });
    /**
     * A directive that can be used to monitor the autofill state of an input.
     */
    class CdkAutofill {
        /**
         * @param {?} _elementRef
         * @param {?} _autofillMonitor
         */
        constructor(_elementRef, _autofillMonitor) {
            this._elementRef = _elementRef;
            this._autofillMonitor = _autofillMonitor;
            /**
             * Emits when the autofill state of the element changes.
             */
            this.cdkAutofill = new EventEmitter();
        }
        /**
         * @return {?}
         */
        ngOnInit() {
            this._autofillMonitor
                .monitor(this._elementRef)
                .subscribe((/**
             * @param {?} event
             * @return {?}
             */
            event => this.cdkAutofill.emit(event)));
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this._autofillMonitor.stopMonitoring(this._elementRef);
        }
    }
    CdkAutofill.ɵfac = function CdkAutofill_Factory(t) { return new (t || CdkAutofill)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(AutofillMonitor)); };
    CdkAutofill.ɵdir = ɵɵdefineDirective({ type: CdkAutofill, selectors: [["", "cdkAutofill", ""]], outputs: { cdkAutofill: "cdkAutofill" } });
    /** @nocollapse */
    CdkAutofill.ctorParameters = () => [
        { type: ElementRef },
        { type: AutofillMonitor }
    ];
    CdkAutofill.propDecorators = {
        cdkAutofill: [{ type: Output }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/text-field/autosize.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Directive to automatically resize a textarea to fit its content.
     */
    class CdkTextareaAutosize {
        /**
         * @param {?} _elementRef
         * @param {?} _platform
         * @param {?} _ngZone
         * @param {?=} document
         */
        constructor(_elementRef, _platform, _ngZone, 
        /** @breaking-change 11.0.0 make document required */
        document) {
            this._elementRef = _elementRef;
            this._platform = _platform;
            this._ngZone = _ngZone;
            this._destroyed = new Subject();
            this._enabled = true;
            /**
             * Value of minRows as of last resize. If the minRows has decreased, the
             * height of the textarea needs to be recomputed to reflect the new minimum. The maxHeight
             * does not have the same problem because it does not affect the textarea's scrollHeight.
             */
            this._previousMinRows = -1;
            this._document = document;
            this._textareaElement = (/** @type {?} */ (this._elementRef.nativeElement));
            this._measuringClass = _platform.FIREFOX ?
                'cdk-textarea-autosize-measuring-firefox' :
                'cdk-textarea-autosize-measuring';
        }
        /**
         * Minimum amount of rows in the textarea.
         * @return {?}
         */
        get minRows() { return this._minRows; }
        /**
         * @param {?} value
         * @return {?}
         */
        set minRows(value) {
            this._minRows = coerceNumberProperty(value);
            this._setMinHeight();
        }
        /**
         * Maximum amount of rows in the textarea.
         * @return {?}
         */
        get maxRows() { return this._maxRows; }
        /**
         * @param {?} value
         * @return {?}
         */
        set maxRows(value) {
            this._maxRows = coerceNumberProperty(value);
            this._setMaxHeight();
        }
        /**
         * Whether autosizing is enabled or not
         * @return {?}
         */
        get enabled() { return this._enabled; }
        /**
         * @param {?} value
         * @return {?}
         */
        set enabled(value) {
            value = coerceBooleanProperty(value);
            // Only act if the actual value changed. This specifically helps to not run
            // resizeToFitContent too early (i.e. before ngAfterViewInit)
            if (this._enabled !== value) {
                (this._enabled = value) ? this.resizeToFitContent(true) : this.reset();
            }
        }
        /**
         * Sets the minimum height of the textarea as determined by minRows.
         * @return {?}
         */
        _setMinHeight() {
            /** @type {?} */
            const minHeight = this.minRows && this._cachedLineHeight ?
                `${this.minRows * this._cachedLineHeight}px` : null;
            if (minHeight) {
                this._textareaElement.style.minHeight = minHeight;
            }
        }
        /**
         * Sets the maximum height of the textarea as determined by maxRows.
         * @return {?}
         */
        _setMaxHeight() {
            /** @type {?} */
            const maxHeight = this.maxRows && this._cachedLineHeight ?
                `${this.maxRows * this._cachedLineHeight}px` : null;
            if (maxHeight) {
                this._textareaElement.style.maxHeight = maxHeight;
            }
        }
        /**
         * @return {?}
         */
        ngAfterViewInit() {
            if (this._platform.isBrowser) {
                // Remember the height which we started with in case autosizing is disabled
                this._initialHeight = this._textareaElement.style.height;
                this.resizeToFitContent();
                this._ngZone.runOutsideAngular((/**
                 * @return {?}
                 */
                () => {
                    /** @type {?} */
                    const window = this._getWindow();
                    fromEvent(window, 'resize')
                        .pipe(auditTime(16), takeUntil(this._destroyed))
                        .subscribe((/**
                     * @return {?}
                     */
                    () => this.resizeToFitContent(true)));
                }));
            }
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this._destroyed.next();
            this._destroyed.complete();
        }
        /**
         * Cache the height of a single-row textarea if it has not already been cached.
         *
         * We need to know how large a single "row" of a textarea is in order to apply minRows and
         * maxRows. For the initial version, we will assume that the height of a single line in the
         * textarea does not ever change.
         * @private
         * @return {?}
         */
        _cacheTextareaLineHeight() {
            if (this._cachedLineHeight) {
                return;
            }
            // Use a clone element because we have to override some styles.
            /** @type {?} */
            let textareaClone = (/** @type {?} */ (this._textareaElement.cloneNode(false)));
            textareaClone.rows = 1;
            // Use `position: absolute` so that this doesn't cause a browser layout and use
            // `visibility: hidden` so that nothing is rendered. Clear any other styles that
            // would affect the height.
            textareaClone.style.position = 'absolute';
            textareaClone.style.visibility = 'hidden';
            textareaClone.style.border = 'none';
            textareaClone.style.padding = '0';
            textareaClone.style.height = '';
            textareaClone.style.minHeight = '';
            textareaClone.style.maxHeight = '';
            // In Firefox it happens that textarea elements are always bigger than the specified amount
            // of rows. This is because Firefox tries to add extra space for the horizontal scrollbar.
            // As a workaround that removes the extra space for the scrollbar, we can just set overflow
            // to hidden. This ensures that there is no invalid calculation of the line height.
            // See Firefox bug report: https://bugzilla.mozilla.org/show_bug.cgi?id=33654
            textareaClone.style.overflow = 'hidden';
            (/** @type {?} */ (this._textareaElement.parentNode)).appendChild(textareaClone);
            this._cachedLineHeight = textareaClone.clientHeight;
            (/** @type {?} */ (this._textareaElement.parentNode)).removeChild(textareaClone);
            // Min and max heights have to be re-calculated if the cached line height changes
            this._setMinHeight();
            this._setMaxHeight();
        }
        /**
         * @return {?}
         */
        ngDoCheck() {
            if (this._platform.isBrowser) {
                this.resizeToFitContent();
            }
        }
        /**
         * Resize the textarea to fit its content.
         * @param {?=} force Whether to force a height recalculation. By default the height will be
         *    recalculated only if the value changed since the last call.
         * @return {?}
         */
        resizeToFitContent(force = false) {
            // If autosizing is disabled, just skip everything else
            if (!this._enabled) {
                return;
            }
            this._cacheTextareaLineHeight();
            // If we haven't determined the line-height yet, we know we're still hidden and there's no point
            // in checking the height of the textarea.
            if (!this._cachedLineHeight) {
                return;
            }
            /** @type {?} */
            const textarea = (/** @type {?} */ (this._elementRef.nativeElement));
            /** @type {?} */
            const value = textarea.value;
            // Only resize if the value or minRows have changed since these calculations can be expensive.
            if (!force && this._minRows === this._previousMinRows && value === this._previousValue) {
                return;
            }
            /** @type {?} */
            const placeholderText = textarea.placeholder;
            // Reset the textarea height to auto in order to shrink back to its default size.
            // Also temporarily force overflow:hidden, so scroll bars do not interfere with calculations.
            // Long placeholders that are wider than the textarea width may lead to a bigger scrollHeight
            // value. To ensure that the scrollHeight is not bigger than the content, the placeholders
            // need to be removed temporarily.
            textarea.classList.add(this._measuringClass);
            textarea.placeholder = '';
            // The measuring class includes a 2px padding to workaround an issue with Chrome,
            // so we account for that extra space here by subtracting 4 (2px top + 2px bottom).
            /** @type {?} */
            const height = textarea.scrollHeight - 4;
            // Use the scrollHeight to know how large the textarea *would* be if fit its entire value.
            textarea.style.height = `${height}px`;
            textarea.classList.remove(this._measuringClass);
            textarea.placeholder = placeholderText;
            this._ngZone.runOutsideAngular((/**
             * @return {?}
             */
            () => {
                if (typeof requestAnimationFrame !== 'undefined') {
                    requestAnimationFrame((/**
                     * @return {?}
                     */
                    () => this._scrollToCaretPosition(textarea)));
                }
                else {
                    setTimeout((/**
                     * @return {?}
                     */
                    () => this._scrollToCaretPosition(textarea)));
                }
            }));
            this._previousValue = value;
            this._previousMinRows = this._minRows;
        }
        /**
         * Resets the textarea to its original size
         * @return {?}
         */
        reset() {
            // Do not try to change the textarea, if the initialHeight has not been determined yet
            // This might potentially remove styles when reset() is called before ngAfterViewInit
            if (this._initialHeight !== undefined) {
                this._textareaElement.style.height = this._initialHeight;
            }
        }
        // In Ivy the `host` metadata will be merged, whereas in ViewEngine it is overridden. In order
        // to avoid double event listeners, we need to use `HostListener`. Once Ivy is the default, we
        // can move this back into `host`.
        // tslint:disable:no-host-decorator-in-concrete
        /**
         * @return {?}
         */
        _noopInputHandler() {
            // no-op handler that ensures we're running change detection on input events.
        }
        /**
         * Access injected document if available or fallback to global document reference
         * @private
         * @return {?}
         */
        _getDocument() {
            return this._document || document;
        }
        /**
         * Use defaultView of injected document if available or fallback to global window reference
         * @private
         * @return {?}
         */
        _getWindow() {
            /** @type {?} */
            const doc = this._getDocument();
            return doc.defaultView || window;
        }
        /**
         * Scrolls a textarea to the caret position. On Firefox resizing the textarea will
         * prevent it from scrolling to the caret position. We need to re-set the selection
         * in order for it to scroll to the proper position.
         * @private
         * @param {?} textarea
         * @return {?}
         */
        _scrollToCaretPosition(textarea) {
            const { selectionStart, selectionEnd } = textarea;
            /** @type {?} */
            const document = this._getDocument();
            // IE will throw an "Unspecified error" if we try to set the selection range after the
            // element has been removed from the DOM. Assert that the directive hasn't been destroyed
            // between the time we requested the animation frame and when it was executed.
            // Also note that we have to assert that the textarea is focused before we set the
            // selection range. Setting the selection range on a non-focused textarea will cause
            // it to receive focus on IE and Edge.
            if (!this._destroyed.isStopped && document.activeElement === textarea) {
                textarea.setSelectionRange(selectionStart, selectionEnd);
            }
        }
    }
    CdkTextareaAutosize.ɵfac = function CdkTextareaAutosize_Factory(t) { return new (t || CdkTextareaAutosize)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(Platform), ɵɵdirectiveInject(NgZone), ɵɵdirectiveInject(DOCUMENT$1, 8)); };
    CdkTextareaAutosize.ɵdir = ɵɵdefineDirective({ type: CdkTextareaAutosize, selectors: [["textarea", "cdkTextareaAutosize", ""]], hostAttrs: ["rows", "1", 1, "cdk-textarea-autosize"], hostBindings: function CdkTextareaAutosize_HostBindings(rf, ctx) { if (rf & 1) {
            ɵɵlistener("input", function CdkTextareaAutosize_input_HostBindingHandler() { return ctx._noopInputHandler(); });
        } }, inputs: { minRows: ["cdkAutosizeMinRows", "minRows"], maxRows: ["cdkAutosizeMaxRows", "maxRows"], enabled: ["cdkTextareaAutosize", "enabled"] }, exportAs: ["cdkTextareaAutosize"] });
    /** @nocollapse */
    CdkTextareaAutosize.ctorParameters = () => [
        { type: ElementRef },
        { type: Platform },
        { type: NgZone },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [DOCUMENT$1,] }] }
    ];
    CdkTextareaAutosize.propDecorators = {
        minRows: [{ type: Input, args: ['cdkAutosizeMinRows',] }],
        maxRows: [{ type: Input, args: ['cdkAutosizeMaxRows',] }],
        enabled: [{ type: Input, args: ['cdkTextareaAutosize',] }],
        _noopInputHandler: [{ type: HostListener, args: ['input',] }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/text-field/text-field-module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class TextFieldModule {
    }
    TextFieldModule.ɵmod = ɵɵdefineNgModule({ type: TextFieldModule });
    TextFieldModule.ɵinj = ɵɵdefineInjector({ factory: function TextFieldModule_Factory(t) { return new (t || TextFieldModule)(); }, imports: [[PlatformModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(TextFieldModule, { declarations: function () { return [CdkAutofill,
            CdkTextareaAutosize]; }, imports: function () { return [PlatformModule]; }, exports: function () { return [CdkAutofill,
            CdkTextareaAutosize]; } }); })();

    class MatTextareaAutosize extends CdkTextareaAutosize {
        /**
         * @return {?}
         */
        get matAutosizeMinRows() { return this.minRows; }
        /**
         * @param {?} value
         * @return {?}
         */
        set matAutosizeMinRows(value) { this.minRows = value; }
        /**
         * @return {?}
         */
        get matAutosizeMaxRows() { return this.maxRows; }
        /**
         * @param {?} value
         * @return {?}
         */
        set matAutosizeMaxRows(value) { this.maxRows = value; }
        /**
         * @return {?}
         */
        get matAutosize() { return this.enabled; }
        /**
         * @param {?} value
         * @return {?}
         */
        set matAutosize(value) { this.enabled = value; }
        /**
         * @return {?}
         */
        get matTextareaAutosize() { return this.enabled; }
        /**
         * @param {?} value
         * @return {?}
         */
        set matTextareaAutosize(value) { this.enabled = value; }
    }
    MatTextareaAutosize.ɵfac = function MatTextareaAutosize_Factory(t) { return ɵMatTextareaAutosize_BaseFactory(t || MatTextareaAutosize); };
    MatTextareaAutosize.ɵdir = ɵɵdefineDirective({ type: MatTextareaAutosize, selectors: [["textarea", "mat-autosize", ""], ["textarea", "matTextareaAutosize", ""]], hostAttrs: ["rows", "1", 1, "cdk-textarea-autosize", "mat-autosize"], inputs: { cdkAutosizeMinRows: "cdkAutosizeMinRows", cdkAutosizeMaxRows: "cdkAutosizeMaxRows", matAutosizeMinRows: "matAutosizeMinRows", matAutosizeMaxRows: "matAutosizeMaxRows", matAutosize: ["mat-autosize", "matAutosize"], matTextareaAutosize: "matTextareaAutosize" }, exportAs: ["matTextareaAutosize"], features: [ɵɵInheritDefinitionFeature] });
    MatTextareaAutosize.propDecorators = {
        matAutosizeMinRows: [{ type: Input }],
        matAutosizeMaxRows: [{ type: Input }],
        matAutosize: [{ type: Input, args: ['mat-autosize',] }],
        matTextareaAutosize: [{ type: Input }]
    };
    const ɵMatTextareaAutosize_BaseFactory = ɵɵgetInheritedFactory(MatTextareaAutosize);

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/input/input-errors.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google LLC All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * \@docs-private
     * @param {?} type
     * @return {?}
     */
    function getMatInputUnsupportedTypeError(type) {
        return Error(`Input type "${type}" isn't supported by matInput.`);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/input/input-value-accessor.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * This token is used to inject the object whose value should be set into `MatInput`. If none is
     * provided, the native `HTMLInputElement` is used. Directives like `MatDatepickerInput` can provide
     * themselves for this token, in order to make `MatInput` delegate the getting and setting of the
     * value to them.
     * @type {?}
     */
    const MAT_INPUT_VALUE_ACCESSOR = new InjectionToken('MAT_INPUT_VALUE_ACCESSOR');

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/input/input.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    // Invalid input type. Using one of these will throw an MatInputUnsupportedTypeError.
    /** @type {?} */
    const MAT_INPUT_INVALID_TYPES = [
        'button',
        'checkbox',
        'file',
        'hidden',
        'image',
        'radio',
        'range',
        'reset',
        'submit'
    ];
    /** @type {?} */
    let nextUniqueId$6 = 0;
    // Boilerplate for applying mixins to MatInput.
    /**
     * \@docs-private
     */
    class MatInputBase {
        /**
         * @param {?} _defaultErrorStateMatcher
         * @param {?} _parentForm
         * @param {?} _parentFormGroup
         * @param {?} ngControl
         */
        constructor(_defaultErrorStateMatcher, _parentForm, _parentFormGroup, ngControl) {
            this._defaultErrorStateMatcher = _defaultErrorStateMatcher;
            this._parentForm = _parentForm;
            this._parentFormGroup = _parentFormGroup;
            this.ngControl = ngControl;
        }
    }
    /** @type {?} */
    const _MatInputMixinBase = mixinErrorState(MatInputBase);
    /**
     * Directive that allows a native input to work inside a `MatFormField`.
     */
    class MatInput extends _MatInputMixinBase {
        /**
         * @param {?} _elementRef
         * @param {?} _platform
         * @param {?} ngControl
         * @param {?} _parentForm
         * @param {?} _parentFormGroup
         * @param {?} _defaultErrorStateMatcher
         * @param {?} inputValueAccessor
         * @param {?} _autofillMonitor
         * @param {?} ngZone
         */
        constructor(_elementRef, _platform, ngControl, _parentForm, _parentFormGroup, _defaultErrorStateMatcher, inputValueAccessor, _autofillMonitor, ngZone) {
            super(_defaultErrorStateMatcher, _parentForm, _parentFormGroup, ngControl);
            this._elementRef = _elementRef;
            this._platform = _platform;
            this.ngControl = ngControl;
            this._autofillMonitor = _autofillMonitor;
            this._uid = `mat-input-${nextUniqueId$6++}`;
            /**
             * Implemented as part of MatFormFieldControl.
             * \@docs-private
             */
            this.focused = false;
            /**
             * Implemented as part of MatFormFieldControl.
             * \@docs-private
             */
            this.stateChanges = new Subject();
            /**
             * Implemented as part of MatFormFieldControl.
             * \@docs-private
             */
            this.controlType = 'mat-input';
            /**
             * Implemented as part of MatFormFieldControl.
             * \@docs-private
             */
            this.autofilled = false;
            this._disabled = false;
            this._required = false;
            this._type = 'text';
            this._readonly = false;
            this._neverEmptyInputTypes = [
                'date',
                'datetime',
                'datetime-local',
                'month',
                'time',
                'week'
            ].filter((/**
             * @param {?} t
             * @return {?}
             */
            t => getSupportedInputTypes().has(t)));
            /** @type {?} */
            const element = this._elementRef.nativeElement;
            /** @type {?} */
            const nodeName = element.nodeName.toLowerCase();
            // If no input value accessor was explicitly specified, use the element as the input value
            // accessor.
            this._inputValueAccessor = inputValueAccessor || element;
            this._previousNativeValue = this.value;
            // Force setter to be called in case id was not specified.
            this.id = this.id;
            // On some versions of iOS the caret gets stuck in the wrong place when holding down the delete
            // key. In order to get around this we need to "jiggle" the caret loose. Since this bug only
            // exists on iOS, we only bother to install the listener on iOS.
            if (_platform.IOS) {
                ngZone.runOutsideAngular((/**
                 * @return {?}
                 */
                () => {
                    _elementRef.nativeElement.addEventListener('keyup', (/**
                     * @param {?} event
                     * @return {?}
                     */
                    (event) => {
                        /** @type {?} */
                        let el = (/** @type {?} */ (event.target));
                        if (!el.value && !el.selectionStart && !el.selectionEnd) {
                            // Note: Just setting `0, 0` doesn't fix the issue. Setting
                            // `1, 1` fixes it for the first time that you type text and
                            // then hold delete. Toggling to `1, 1` and then back to
                            // `0, 0` seems to completely fix it.
                            el.setSelectionRange(1, 1);
                            el.setSelectionRange(0, 0);
                        }
                    }));
                }));
            }
            this._isServer = !this._platform.isBrowser;
            this._isNativeSelect = nodeName === 'select';
            this._isTextarea = nodeName === 'textarea';
            if (this._isNativeSelect) {
                this.controlType = ((/** @type {?} */ (element))).multiple ? 'mat-native-select-multiple' :
                    'mat-native-select';
            }
        }
        /**
         * Implemented as part of MatFormFieldControl.
         * \@docs-private
         * @return {?}
         */
        get disabled() {
            if (this.ngControl && this.ngControl.disabled !== null) {
                return this.ngControl.disabled;
            }
            return this._disabled;
        }
        /**
         * @param {?} value
         * @return {?}
         */
        set disabled(value) {
            this._disabled = coerceBooleanProperty(value);
            // Browsers may not fire the blur event if the input is disabled too quickly.
            // Reset from here to ensure that the element doesn't become stuck.
            if (this.focused) {
                this.focused = false;
                this.stateChanges.next();
            }
        }
        /**
         * Implemented as part of MatFormFieldControl.
         * \@docs-private
         * @return {?}
         */
        get id() { return this._id; }
        /**
         * @param {?} value
         * @return {?}
         */
        set id(value) { this._id = value || this._uid; }
        /**
         * Implemented as part of MatFormFieldControl.
         * \@docs-private
         * @return {?}
         */
        get required() { return this._required; }
        /**
         * @param {?} value
         * @return {?}
         */
        set required(value) { this._required = coerceBooleanProperty(value); }
        /**
         * Input type of the element.
         * @return {?}
         */
        get type() { return this._type; }
        /**
         * @param {?} value
         * @return {?}
         */
        set type(value) {
            this._type = value || 'text';
            this._validateType();
            // When using Angular inputs, developers are no longer able to set the properties on the native
            // input element. To ensure that bindings for `type` work, we need to sync the setter
            // with the native property. Textarea elements don't support the type property or attribute.
            if (!this._isTextarea && getSupportedInputTypes().has(this._type)) {
                ((/** @type {?} */ (this._elementRef.nativeElement))).type = this._type;
            }
        }
        /**
         * Implemented as part of MatFormFieldControl.
         * \@docs-private
         * @return {?}
         */
        get value() { return this._inputValueAccessor.value; }
        /**
         * @param {?} value
         * @return {?}
         */
        set value(value) {
            if (value !== this.value) {
                this._inputValueAccessor.value = value;
                this.stateChanges.next();
            }
        }
        /**
         * Whether the element is readonly.
         * @return {?}
         */
        get readonly() { return this._readonly; }
        /**
         * @param {?} value
         * @return {?}
         */
        set readonly(value) { this._readonly = coerceBooleanProperty(value); }
        /**
         * @return {?}
         */
        ngOnInit() {
            if (this._platform.isBrowser) {
                this._autofillMonitor.monitor(this._elementRef.nativeElement).subscribe((/**
                 * @param {?} event
                 * @return {?}
                 */
                event => {
                    this.autofilled = event.isAutofilled;
                    this.stateChanges.next();
                }));
            }
        }
        /**
         * @return {?}
         */
        ngOnChanges() {
            this.stateChanges.next();
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this.stateChanges.complete();
            if (this._platform.isBrowser) {
                this._autofillMonitor.stopMonitoring(this._elementRef.nativeElement);
            }
        }
        /**
         * @return {?}
         */
        ngDoCheck() {
            if (this.ngControl) {
                // We need to re-evaluate this on every change detection cycle, because there are some
                // error triggers that we can't subscribe to (e.g. parent form submissions). This means
                // that whatever logic is in here has to be super lean or we risk destroying the performance.
                this.updateErrorState();
            }
            // We need to dirty-check the native element's value, because there are some cases where
            // we won't be notified when it changes (e.g. the consumer isn't using forms or they're
            // updating the value using `emitEvent: false`).
            this._dirtyCheckNativeValue();
        }
        /**
         * Focuses the input.
         * @param {?=} options
         * @return {?}
         */
        focus(options) {
            this._elementRef.nativeElement.focus(options);
        }
        // We have to use a `HostListener` here in order to support both Ivy and ViewEngine.
        // In Ivy the `host` bindings will be merged when this class is extended, whereas in
        // ViewEngine they're overwritten.
        // TODO(crisbeto): we move this back into `host` once Ivy is turned on by default.
        /**
         * Callback for the cases where the focused state of the input changes.
         * @param {?} isFocused
         * @return {?}
         */
        // tslint:disable:no-host-decorator-in-concrete
        // tslint:enable:no-host-decorator-in-concrete
        _focusChanged(isFocused) {
            if (isFocused !== this.focused && (!this.readonly || !isFocused)) {
                this.focused = isFocused;
                this.stateChanges.next();
            }
        }
        // We have to use a `HostListener` here in order to support both Ivy and ViewEngine.
        // In Ivy the `host` bindings will be merged when this class is extended, whereas in
        // ViewEngine they're overwritten.
        // TODO(crisbeto): we move this back into `host` once Ivy is turned on by default.
        // tslint:disable-next-line:no-host-decorator-in-concrete
        /**
         * @return {?}
         */
        _onInput() {
            // This is a noop function and is used to let Angular know whenever the value changes.
            // Angular will run a new change detection each time the `input` event has been dispatched.
            // It's necessary that Angular recognizes the value change, because when floatingLabel
            // is set to false and Angular forms aren't used, the placeholder won't recognize the
            // value changes and will not disappear.
            // Listening to the input event wouldn't be necessary when the input is using the
            // FormsModule or ReactiveFormsModule, because Angular forms also listens to input events.
        }
        /**
         * Does some manual dirty checking on the native input `value` property.
         * @protected
         * @return {?}
         */
        _dirtyCheckNativeValue() {
            /** @type {?} */
            const newValue = this._elementRef.nativeElement.value;
            if (this._previousNativeValue !== newValue) {
                this._previousNativeValue = newValue;
                this.stateChanges.next();
            }
        }
        /**
         * Make sure the input is a supported type.
         * @protected
         * @return {?}
         */
        _validateType() {
            if (MAT_INPUT_INVALID_TYPES.indexOf(this._type) > -1) {
                throw getMatInputUnsupportedTypeError(this._type);
            }
        }
        /**
         * Checks whether the input type is one of the types that are never empty.
         * @protected
         * @return {?}
         */
        _isNeverEmpty() {
            return this._neverEmptyInputTypes.indexOf(this._type) > -1;
        }
        /**
         * Checks whether the input is invalid based on the native validation.
         * @protected
         * @return {?}
         */
        _isBadInput() {
            // The `validity` property won't be present on platform-server.
            /** @type {?} */
            let validity = ((/** @type {?} */ (this._elementRef.nativeElement))).validity;
            return validity && validity.badInput;
        }
        /**
         * Implemented as part of MatFormFieldControl.
         * \@docs-private
         * @return {?}
         */
        get empty() {
            return !this._isNeverEmpty() && !this._elementRef.nativeElement.value && !this._isBadInput() &&
                !this.autofilled;
        }
        /**
         * Implemented as part of MatFormFieldControl.
         * \@docs-private
         * @return {?}
         */
        get shouldLabelFloat() {
            if (this._isNativeSelect) {
                // For a single-selection `<select>`, the label should float when the selected option has
                // a non-empty display value. For a `<select multiple>`, the label *always* floats to avoid
                // overlapping the label with the options.
                /** @type {?} */
                const selectElement = (/** @type {?} */ (this._elementRef.nativeElement));
                /** @type {?} */
                const firstOption = selectElement.options[0];
                // On most browsers the `selectedIndex` will always be 0, however on IE and Edge it'll be
                // -1 if the `value` is set to something, that isn't in the list of options, at a later point.
                return this.focused || selectElement.multiple || !this.empty ||
                    !!(selectElement.selectedIndex > -1 && firstOption && firstOption.label);
            }
            else {
                return this.focused || !this.empty;
            }
        }
        /**
         * Implemented as part of MatFormFieldControl.
         * \@docs-private
         * @param {?} ids
         * @return {?}
         */
        setDescribedByIds(ids) {
            this._ariaDescribedby = ids.join(' ');
        }
        /**
         * Implemented as part of MatFormFieldControl.
         * \@docs-private
         * @return {?}
         */
        onContainerClick() {
            // Do not re-focus the input element if the element is already focused. Otherwise it can happen
            // that someone clicks on a time input and the cursor resets to the "hours" field while the
            // "minutes" field was actually clicked. See: https://github.com/angular/components/issues/12849
            if (!this.focused) {
                this.focus();
            }
        }
    }
    MatInput.ɵfac = function MatInput_Factory(t) { return new (t || MatInput)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(Platform), ɵɵdirectiveInject(NgControl, 10), ɵɵdirectiveInject(NgForm, 8), ɵɵdirectiveInject(FormGroupDirective, 8), ɵɵdirectiveInject(ErrorStateMatcher), ɵɵdirectiveInject(MAT_INPUT_VALUE_ACCESSOR, 10), ɵɵdirectiveInject(AutofillMonitor), ɵɵdirectiveInject(NgZone)); };
    MatInput.ɵdir = ɵɵdefineDirective({ type: MatInput, selectors: [["input", "matInput", ""], ["textarea", "matInput", ""], ["select", "matNativeControl", ""], ["input", "matNativeControl", ""], ["textarea", "matNativeControl", ""]], hostAttrs: [1, "mat-input-element", "mat-form-field-autofill-control"], hostVars: 10, hostBindings: function MatInput_HostBindings(rf, ctx) { if (rf & 1) {
            ɵɵlistener("focus", function MatInput_focus_HostBindingHandler() { return ctx._focusChanged(true); })("blur", function MatInput_blur_HostBindingHandler() { return ctx._focusChanged(false); })("input", function MatInput_input_HostBindingHandler() { return ctx._onInput(); });
        } if (rf & 2) {
            ɵɵhostProperty("disabled", ctx.disabled)("required", ctx.required);
            ɵɵattribute("id", ctx.id)("placeholder", ctx.placeholder)("readonly", ctx.readonly && !ctx._isNativeSelect || null)("aria-describedby", ctx._ariaDescribedby || null)("aria-invalid", ctx.errorState)("aria-required", ctx.required.toString());
            ɵɵclassProp("mat-input-server", ctx._isServer);
        } }, inputs: { id: "id", disabled: "disabled", required: "required", type: "type", value: "value", readonly: "readonly", placeholder: "placeholder", errorStateMatcher: "errorStateMatcher" }, exportAs: ["matInput"], features: [ɵɵProvidersFeature([{ provide: MatFormFieldControl, useExisting: MatInput }]), ɵɵInheritDefinitionFeature, ɵɵNgOnChangesFeature] });
    /** @nocollapse */
    MatInput.ctorParameters = () => [
        { type: ElementRef },
        { type: Platform },
        { type: NgControl, decorators: [{ type: Optional }, { type: Self }] },
        { type: NgForm, decorators: [{ type: Optional }] },
        { type: FormGroupDirective, decorators: [{ type: Optional }] },
        { type: ErrorStateMatcher },
        { type: undefined, decorators: [{ type: Optional }, { type: Self }, { type: Inject, args: [MAT_INPUT_VALUE_ACCESSOR,] }] },
        { type: AutofillMonitor },
        { type: NgZone }
    ];
    MatInput.propDecorators = {
        disabled: [{ type: Input }],
        id: [{ type: Input }],
        placeholder: [{ type: Input }],
        required: [{ type: Input }],
        type: [{ type: Input }],
        errorStateMatcher: [{ type: Input }],
        value: [{ type: Input }],
        readonly: [{ type: Input }],
        _focusChanged: [{ type: HostListener, args: ['focus', ['true'],] }, { type: HostListener, args: ['blur', ['false'],] }],
        _onInput: [{ type: HostListener, args: ['input',] }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/input/input-module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class MatInputModule {
    }
    MatInputModule.ɵmod = ɵɵdefineNgModule({ type: MatInputModule });
    MatInputModule.ɵinj = ɵɵdefineInjector({ factory: function MatInputModule_Factory(t) { return new (t || MatInputModule)(); }, providers: [ErrorStateMatcher], imports: [[
                TextFieldModule,
                MatFormFieldModule,
            ],
            TextFieldModule,
            // We re-export the `MatFormFieldModule` since `MatInput` will almost always
            // be used together with `MatFormField`.
            MatFormFieldModule] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(MatInputModule, { declarations: function () { return [MatInput,
            MatTextareaAutosize]; }, imports: function () { return [TextFieldModule,
            MatFormFieldModule]; }, exports: function () { return [TextFieldModule,
            // We re-export the `MatFormFieldModule` since `MatInput` will almost always
            // be used together with `MatFormField`.
            MatFormFieldModule,
            MatInput,
            MatTextareaAutosize]; } }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/settings/dialog_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    function SettingsDialogComponent_mat_error_10_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "mat-error");
        ɵɵtext(1, " Reload period has to be minimum of 15 seconds. ");
        ɵɵelementEnd();
    } }
    function SettingsDialogComponent_mat_error_14_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "mat-error");
        ɵɵtext(1, " Page size has to be a positive integer. ");
        ɵɵelementEnd();
    } }
    /** @type {?} */
    const getReloadPeriodInSec = createSelector(getReloadPeriodInMs, (/**
     * @param {?} periodInMs
     * @return {?}
     */
    (periodInMs) => Math.round(periodInMs / 1000)));
    /**
     * @return {?}
     */
    function createIntegerValidator() {
        return (/**
         * @param {?} control
         * @return {?}
         */
        (control) => {
            /** @type {?} */
            const numValue = Number(control.value);
            /** @type {?} */
            const valid = Math.round(numValue) === control.value;
            return valid ? null : { integer: { value: control.value } };
        });
    }
    class SettingsDialogComponent {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.reloadEnabled$ = this.store.pipe(select(getReloadEnabled));
            this.pageSize$ = this.store.pipe(select(getPageSize));
            this.reloadPeriodInSec$ = this.store.pipe(select(getReloadPeriodInSec));
            this.reloadPeriodControl = new FormControl(15, [
                Validators.required,
                Validators.min(15),
            ]);
            this.paginationControl = new FormControl(1, [
                Validators.required,
                Validators.min(1),
                createIntegerValidator(),
            ]);
            this.ngUnsubscribe = new Subject();
        }
        /**
         * @return {?}
         */
        ngOnInit() {
            this.reloadPeriodInSec$
                .pipe(takeUntil(this.ngUnsubscribe), filter((/**
             * @param {?} value
             * @return {?}
             */
            (value) => value !== this.reloadPeriodControl.value)))
                .subscribe((/**
             * @param {?} value
             * @return {?}
             */
            (value) => {
                this.reloadPeriodControl.setValue(value);
            }));
            this.reloadEnabled$
                .pipe(takeUntil(this.ngUnsubscribe))
                .subscribe((/**
             * @param {?} value
             * @return {?}
             */
            (value) => {
                if (value)
                    this.reloadPeriodControl.enable();
                else
                    this.reloadPeriodControl.disable();
            }));
            this.reloadPeriodControl.valueChanges
                .pipe(takeUntil(this.ngUnsubscribe), debounceTime(500), filter((/**
             * @return {?}
             */
            () => this.reloadPeriodControl.valid)))
                .subscribe((/**
             * @return {?}
             */
            () => {
                if (!this.reloadPeriodControl.valid) {
                    return;
                }
                /** @type {?} */
                const periodInMs = this.reloadPeriodControl.value * 1000;
                this.store.dispatch(changeReloadPeriod({ periodInMs }));
            }));
            this.pageSize$
                .pipe(takeUntil(this.ngUnsubscribe), filter((/**
             * @param {?} value
             * @return {?}
             */
            (value) => value !== this.paginationControl.value)))
                .subscribe((/**
             * @param {?} value
             * @return {?}
             */
            (value) => {
                this.paginationControl.setValue(value);
            }));
            this.paginationControl.valueChanges
                .pipe(takeUntil(this.ngUnsubscribe), debounceTime(500), filter((/**
             * @return {?}
             */
            () => this.paginationControl.valid)))
                .subscribe((/**
             * @return {?}
             */
            () => {
                this.store.dispatch(changePageSize({ size: this.paginationControl.value }));
            }));
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this.ngUnsubscribe.next();
            this.ngUnsubscribe.complete();
        }
        /**
         * @return {?}
         */
        onReloadToggle() {
            this.store.dispatch(toggleReloadEnabled());
        }
    }
    SettingsDialogComponent.decorators = [
        { type: Component, args: [{
                    selector: 'settings-dialog',
                    template: `
    <h3>Settings</h3>
    <div>
      <div class="reload-toggle">
        <mat-checkbox
          [checked]="reloadEnabled$ | async"
          (change)="onReloadToggle()"
          >Reload data</mat-checkbox
        >
      </div>
      <div>
        <mat-form-field>
          <input
            class="reload-period"
            matInput
            type="number"
            placeholder="Reload Period"
            [formControl]="reloadPeriodControl"
          />
        </mat-form-field>
        <mat-error
          *ngIf="
            reloadPeriodControl.hasError('min') ||
            reloadPeriodControl.hasError('required')
          "
        >
          Reload period has to be minimum of 15 seconds.
        </mat-error>
      </div>
    </div>
    <div>
      <mat-form-field>
        <input
          class="page-size"
          matInput
          type="number"
          placeholder="Pagination Limit"
          [formControl]="paginationControl"
        />
      </mat-form-field>
      <mat-error *ngIf="paginationControl.invalid">
        Page size has to be a positive integer.
      </mat-error>
    </div>
  `,
                    styleUrls: ['./dialog_component.css'],
                },] },
    ];
    /** @nocollapse */
    SettingsDialogComponent.ctorParameters = () => [
        { type: Store }
    ];
    /** @nocollapse */ SettingsDialogComponent.ɵfac = function SettingsDialogComponent_Factory(t) { return new (t || SettingsDialogComponent)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ SettingsDialogComponent.ɵcmp = ɵɵdefineComponent({ type: SettingsDialogComponent, selectors: [["settings-dialog"]], decls: 15, vars: 7, consts: [[1, "reload-toggle"], [3, "checked", "change"], ["matInput", "", "type", "number", "placeholder", "Reload Period", 1, "reload-period", 3, "formControl"], [4, "ngIf"], ["matInput", "", "type", "number", "placeholder", "Pagination Limit", 1, "page-size", 3, "formControl"]], template: function SettingsDialogComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "h3");
            ɵɵtext(1, "Settings");
            ɵɵelementEnd();
            ɵɵelementStart(2, "div");
            ɵɵelementStart(3, "div", 0);
            ɵɵelementStart(4, "mat-checkbox", 1);
            ɵɵlistener("change", function SettingsDialogComponent_Template_mat_checkbox_change_4_listener() { return ctx.onReloadToggle(); });
            ɵɵpipe(5, "async");
            ɵɵtext(6, "Reload data");
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementStart(7, "div");
            ɵɵelementStart(8, "mat-form-field");
            ɵɵelement(9, "input", 2);
            ɵɵelementEnd();
            ɵɵtemplate(10, SettingsDialogComponent_mat_error_10_Template, 2, 0, "mat-error", 3);
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementStart(11, "div");
            ɵɵelementStart(12, "mat-form-field");
            ɵɵelement(13, "input", 4);
            ɵɵelementEnd();
            ɵɵtemplate(14, SettingsDialogComponent_mat_error_14_Template, 2, 0, "mat-error", 3);
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵadvance(4);
            ɵɵproperty("checked", ɵɵpipeBind1(5, 5, ctx.reloadEnabled$));
            ɵɵadvance(5);
            ɵɵproperty("formControl", ctx.reloadPeriodControl);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx.reloadPeriodControl.hasError("min") || ctx.reloadPeriodControl.hasError("required"));
            ɵɵadvance(3);
            ɵɵproperty("formControl", ctx.paginationControl);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx.paginationControl.invalid);
        } }, directives: [MatCheckbox, MatFormField, MatInput, NumberValueAccessor, DefaultValueAccessor, NgControlStatus, FormControlDirective, NgIf, MatError], pipes: [AsyncPipe], styles: ["[_nghost-%COMP%] {\n  font-size: 15px;\n}\n\n[_nghost-%COMP%]    > div[_ngcontent-%COMP%] {\n  margin: 10px 0;\n}\n\n[_nghost-%COMP%]    > [_ngcontent-%COMP%]:first-child {\n  margin-top: 0;\n}\n\n[_nghost-%COMP%]    > [_ngcontent-%COMP%]:last-child {\n  margin-bottom: 0;\n}\n\nh3[_ngcontent-%COMP%] {\n  font-size: 20px;\n}\n\n.reload-toggle[_ngcontent-%COMP%] {\n  margin-bottom: 10px;\n}"] });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/settings/settings_button_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class SettingsButtonComponent {
        /**
         * @param {?} dialog
         */
        constructor(dialog) {
            this.dialog = dialog;
        }
        /**
         * @return {?}
         */
        openDialog() {
            this.dialog.open(SettingsDialogComponent, {
                width: '400px',
            });
        }
    }
    SettingsButtonComponent.decorators = [
        { type: Component, args: [{
                    selector: 'settings-button',
                    template: `
    <button mat-icon-button (click)="openDialog()">
      <mat-icon svgIcon="settings_24px"></mat-icon>
    </button>
  `,
                },] },
    ];
    /** @nocollapse */
    SettingsButtonComponent.ctorParameters = () => [
        { type: MatDialog }
    ];
    /** @nocollapse */ SettingsButtonComponent.ɵfac = function SettingsButtonComponent_Factory(t) { return new (t || SettingsButtonComponent)(ɵɵdirectiveInject(MatDialog)); };
    /** @nocollapse */ SettingsButtonComponent.ɵcmp = ɵɵdefineComponent({ type: SettingsButtonComponent, selectors: [["settings-button"]], decls: 2, vars: 0, consts: [["mat-icon-button", "", 3, "click"], ["svgIcon", "settings_24px"]], template: function SettingsButtonComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "button", 0);
            ɵɵlistener("click", function SettingsButtonComponent_Template_button_click_0_listener() { return ctx.openDialog(); });
            ɵɵelement(1, "mat-icon", 1);
            ɵɵelementEnd();
        } }, directives: [MatButton, MatIcon], encapsulation: 2 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/header/header_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class HeaderComponent {
    }
    HeaderComponent.decorators = [
        { type: Component, args: [{
                    selector: 'app-header',
                    template: `
    <mat-toolbar color="primary">
      <span class="brand">TensorBoard</span>
      <plugin-selector class="plugins"></plugin-selector>
      <tbdev-upload-button></tbdev-upload-button>
      <app-header-reload></app-header-reload>
      <settings-button></settings-button>
      <a
        class="readme"
        mat-icon-button
        href="https://github.com/tensorflow/tensorboard/blob/master/README.md"
        rel="noopener noreferrer"
        target="_blank"
        aria-label="Help"
      >
        <mat-icon svgIcon="help_outline_24px"></mat-icon>
      </a>
    </mat-toolbar>
  `,
                    styles: [
                        `
      mat-toolbar {
        align-items: center;
        display: flex;
        height: 64px;
        overflow: hidden;
        width: 100%;
      }

      tbdev-upload-button.shown {
        margin: 0 8px 0 16px;
      }

      .brand,
      .readme,
      app-header-reload,
      settings-button {
        flex: 0 0 auto;
      }

      .brand {
        letter-spacing: -0.025em;
        margin-left: 10px;
        text-rendering: optimizeLegibility;
      }

      .plugins {
        align-items: center;
        display: flex;
        flex: 1 1 auto;
        font-size: 14px;
        height: 100%;
        overflow: hidden;
      }
    `,
                    ],
                },] },
    ];
    /** @nocollapse */ HeaderComponent.ɵfac = function HeaderComponent_Factory(t) { return new (t || HeaderComponent)(); };
    /** @nocollapse */ HeaderComponent.ɵcmp = ɵɵdefineComponent({ type: HeaderComponent, selectors: [["app-header"]], decls: 9, vars: 0, consts: [["color", "primary"], [1, "brand"], [1, "plugins"], ["mat-icon-button", "", "href", "https://github.com/tensorflow/tensorboard/blob/master/README.md", "rel", "noopener noreferrer", "target", "_blank", "aria-label", "Help", 1, "readme"], ["svgIcon", "help_outline_24px"]], template: function HeaderComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "mat-toolbar", 0);
            ɵɵelementStart(1, "span", 1);
            ɵɵtext(2, "TensorBoard");
            ɵɵelementEnd();
            ɵɵelement(3, "plugin-selector", 2);
            ɵɵelement(4, "tbdev-upload-button");
            ɵɵelement(5, "app-header-reload");
            ɵɵelement(6, "settings-button");
            ɵɵelementStart(7, "a", 3);
            ɵɵelement(8, "mat-icon", 4);
            ɵɵelementEnd();
            ɵɵelementEnd();
        } }, directives: [MatToolbar, PluginSelectorContainer, TbdevUploadButtonComponent, ReloadContainer, SettingsButtonComponent, MatAnchor, MatIcon], styles: ["mat-toolbar[_ngcontent-%COMP%] {\n        align-items: center;\n        display: flex;\n        height: 64px;\n        overflow: hidden;\n        width: 100%;\n      }\n\n      tbdev-upload-button.shown[_ngcontent-%COMP%] {\n        margin: 0 8px 0 16px;\n      }\n\n      .brand[_ngcontent-%COMP%], .readme[_ngcontent-%COMP%], app-header-reload[_ngcontent-%COMP%], settings-button[_ngcontent-%COMP%] {\n        flex: 0 0 auto;\n      }\n\n      .brand[_ngcontent-%COMP%] {\n        letter-spacing: -0.025em;\n        margin-left: 10px;\n        text-rendering: optimizeLegibility;\n      }\n\n      .plugins[_ngcontent-%COMP%] {\n        align-items: center;\n        display: flex;\n        flex: 1 1 auto;\n        font-size: 14px;\n        height: 100%;\n        overflow: hidden;\n      }"] });

    var RouteKind;
    (function (RouteKind) {
        RouteKind[RouteKind["UNKNOWN"] = 0] = "UNKNOWN";
        RouteKind[RouteKind["EXPERIMENTS"] = 1] = "EXPERIMENTS";
        RouteKind[RouteKind["EXPERIMENT"] = 2] = "EXPERIMENT";
        RouteKind[RouteKind["COMPARE_EXPERIMENT"] = 3] = "COMPARE_EXPERIMENT";
    })(RouteKind || (RouteKind = {}));
    const DEFAULT_EXPERIMENT_ID = 'defaultExperimentId';

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * Parses short experiment name to id map encoded in string for compare route.
     *
     * e.g., /compare/exp1:12345,exp2:99999 results,
     * [
     *   {name: 'exp1', id: '12345'},
     *   {name: 'exp2', id: '99999'}
     * ]
     */
    function parseCompareExperimentStr(eidStr) {
        return eidStr.split(',').map((nameToIdStr) => {
            const colonIndex = nameToIdStr.indexOf(':');
            if (colonIndex < 0) {
                throw new Error(`Expect colon delimiting name and ID: ${nameToIdStr}`);
            }
            const name = nameToIdStr.slice(0, colonIndex);
            const id = nameToIdStr.slice(colonIndex + 1);
            if (!name) {
                throw new Error(`Expect name to be non-falsy: ${nameToIdStr}`);
            }
            if (!id) {
                throw new Error(`Expect id to be non-falsy: ${nameToIdStr}`);
            }
            return { name, id };
        });
    }
    /**
     * Returns experimentIds from route parameter. For a route that does not contain
     * any experiment ids, it returns null.
     */
    function getExperimentIdsFromRouteParams(routeKind, params) {
        switch (routeKind) {
            case RouteKind.EXPERIMENT: {
                // The route may rely on the implicit, default experiment id, if the URL
                // does not contain the experiment param.
                if (Object.prototype.hasOwnProperty.call(params, 'experimentId')) {
                    return [params.experimentId];
                }
                return [DEFAULT_EXPERIMENT_ID];
            }
            case RouteKind.COMPARE_EXPERIMENT: {
                const typedParams = params;
                return parseCompareExperimentStr(typedParams.experimentIds).map(({ id }) => id);
            }
            case RouteKind.EXPERIMENTS:
            default:
                return null;
        }
    }
    /**
     * Returns an identifier that identifies known top-level routes. It is similar
     * to `url` but ignores hash and returns "unknown" for unknown routes.
     */
    function getRouteId(routeKind, params) {
        switch (routeKind) {
            case RouteKind.COMPARE_EXPERIMENT: {
                const typedParams = params;
                return `${routeKind}/${typedParams.experimentIds}`;
            }
            case RouteKind.EXPERIMENT: {
                const typedParams = params;
                return `${routeKind}/${typedParams.experimentId}`;
            }
            case RouteKind.EXPERIMENTS:
                return String(routeKind);
            default:
                return '';
        }
    }
    function createURLSearchParamsFromSerializableQueryParams(params) {
        const searchParams = new URLSearchParams();
        for (const { key, value } of params) {
            searchParams.append(key, value);
        }
        return searchParams;
    }
    /**
     * Checks whether two RouteOrNavs are equal. RouteOrNav is defined to be Route
     * or Navigation.
     *
     * Limitations: currently, they only match pathname and query parameter (order
     * of the params have to be equal too; e.g., a=1&b=2 will not equal b=2&a=1).
     */
    function areRoutesEqual(aRoute, bRoute) {
        // TODO(stephanwlee): support hashes.
        if (aRoute.pathname !== bRoute.pathname ||
            aRoute.queryParams.length !== bRoute.queryParams.length) {
            return false;
        }
        return aRoute.queryParams.every((paramA, index) => {
            const paramB = bRoute.queryParams[index];
            return paramA.key === paramB.key && paramA.value === paramB.value;
        });
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/app_routing/route_config.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @param {?} path
     * @return {?}
     */
    function getPathFragments(path) {
        /** @type {?} */
        const parts = getPathParts(path);
        return parts.map((/**
         * @param {?} part
         * @return {?}
         */
        (part) => {
            /** @type {?} */
            const isParam = part.startsWith(':');
            if (!isParam) {
                return { pathPart: part, isParam };
            }
            return {
                pathPart: part,
                isParam: true,
                paramName: part.slice(1),
            };
        }));
    }
    class RouteConfigMatcher {
        /**
         * @param {?} config
         */
        constructor(config) {
            this.config = config;
            this.validateConfig(config);
            this.pathFragments = getPathFragments(config.path);
            this.pathMatchers = this.getPathMatchers(this.pathFragments);
            this.redirectionFragments =
                config.routeKind === null
                    ? getPathFragments(config.redirectionPath)
                    : null;
        }
        /**
         * @private
         * @param {?} __0
         * @return {?}
         */
        validateConfig({ path }) {
            if (!path.startsWith('/')) {
                throw new RangeError(`config.path should start with '/'. ${path}`);
            }
            /** @type {?} */
            let colonIndex = 0;
            while ((colonIndex = path.indexOf(':', colonIndex + 1)) >= 0) {
                if (path[colonIndex - 1] !== '/') {
                    throw new RangeError(`config.path parameter should come after '/'. ${path}`);
                }
                if (path[colonIndex + 1] === undefined || path[colonIndex + 1] === '/') {
                    throw new RangeError(`config.path parameter should have non-empty name. ${path}`);
                }
            }
        }
        /**
         * @private
         * @param {?} pathFragments
         * @return {?}
         */
        getPathMatchers(pathFragments) {
            return pathFragments.map((/**
             * @param {?} fragment
             * @return {?}
             */
            (fragment) => {
                const { pathPart } = fragment;
                if (fragment.isParam) {
                    return (/**
                     * @param {?} pathPartQuery
                     * @return {?}
                     */
                    (pathPartQuery) => {
                        return {
                            isParamPathPart: true,
                            partMatched: true,
                            paramName: fragment.paramName,
                            paramValue: pathPartQuery,
                        };
                    });
                }
                return (/**
                 * @param {?} pathPartQuery
                 * @return {?}
                 */
                (pathPartQuery) => {
                    return {
                        isParamPathPart: false,
                        partMatched: pathPartQuery === pathPart,
                    };
                });
            }));
        }
        /**
         * In case of parameter reprojection failure (cannot create redirection
         * pathname from parameter present in current pathname), it throws a
         * RangeError.
         * @param {?} pathParts
         * @return {?}
         */
        match(pathParts) {
            /** @type {?} */
            let combinedParams = {};
            if (this.pathMatchers.length !== pathParts.length) {
                return { result: false };
            }
            /** @type {?} */
            let pathIndex = 0;
            for (const matcher of this.pathMatchers) {
                /** @type {?} */
                const pathToMatch = pathParts[pathIndex++];
                /** @type {?} */
                const match = matcher(pathToMatch);
                if (!match.partMatched) {
                    return { result: false };
                }
                if (match.isParamPathPart) {
                    combinedParams = Object.assign(Object.assign({}, combinedParams), { [match.paramName]: match.paramValue });
                }
            }
            if (this.redirectionFragments) {
                /** @type {?} */
                const newPathParts = this.reprojectPathByParams(this.redirectionFragments, combinedParams);
                return {
                    result: true,
                    params: combinedParams,
                    pathParts: newPathParts,
                    isRedirection: true,
                };
            }
            return {
                result: true,
                params: combinedParams,
                pathParts,
                isRedirection: false,
            };
        }
        /**
         * Reprojects route parameter values to path fragments for path parts.
         * @private
         * @param {?} pathFragments
         * @param {?} params
         * @return {?}
         */
        reprojectPathByParams(pathFragments, params) {
            /** @type {?} */
            const pathParts = [];
            for (const fragment of pathFragments) {
                if (fragment.isParam) {
                    const { paramName } = fragment;
                    if (!params.hasOwnProperty(paramName)) {
                        throw new RangeError(`Failed to reproject parameter. "${paramName}" parameter ` +
                            'should be present.');
                    }
                    pathParts.push(params[paramName]);
                }
                else {
                    pathParts.push(fragment.pathPart);
                }
            }
            return pathParts;
        }
        /**
         * In case of parameter projection failure (cannot create pathname), it throws
         * a RangeError.
         * @param {?} params
         * @return {?}
         */
        matchByParams(params) {
            /** @type {?} */
            const pathParts = this.reprojectPathByParams(this.pathFragments, params);
            return {
                result: true,
                params,
                pathParts,
                isRedirection: false,
            };
        }
    }
    class RouteConfigs {
        /**
         * @param {?} configs
         * @param {?=} maxRedirection
         */
        constructor(configs, maxRedirection = 3) {
            this.maxRedirection = maxRedirection;
            if (maxRedirection < 0) {
                throw new RangeError('maxRedirection has to be non-negative number');
            }
            this.validateRouteConfigs(configs);
            this.defaultRouteConfig = null;
            this.routeKindToConfigMatchers = new Map();
            this.configMatchers = [];
            for (const config of configs) {
                /** @type {?} */
                const matcher = new RouteConfigMatcher(config);
                this.configMatchers.push(matcher);
                if (matcher.config.routeKind !== null) {
                    this.routeKindToConfigMatchers.set(matcher.config.routeKind, matcher);
                    if (matcher.config.defaultRoute) {
                        this.defaultRouteConfig = matcher;
                    }
                }
            }
        }
        /**
         * @private
         * @param {?} configs
         * @return {?}
         */
        validateRouteConfigs(configs) {
            /** @type {?} */
            const defaultRoutes = configs.filter((/**
             * @param {?} routeDef
             * @return {?}
             */
            (routeDef) => {
                return Boolean(routeDef.routeKind !== null && routeDef.defaultRoute);
            }));
            if (defaultRoutes.length > 1) {
                /** @type {?} */
                const paths = defaultRoutes.map((/**
                 * @param {?} __0
                 * @return {?}
                 */
                ({ path }) => path)).join(', ');
                throw new RangeError(`There are more than one defaultRoutes. ${paths}`);
            }
            else if (defaultRoutes.length === 1) {
                const { path } = defaultRoutes[0];
                /** @type {?} */
                const hasParam = Boolean(getPathFragments(path).find((/**
                 * @param {?} __0
                 * @return {?}
                 */
                ({ isParam }) => isParam)));
                if (hasParam) {
                    throw new RangeError(`A defaultRoute cannot have any params. ${path}`);
                }
            }
            /** @type {?} */
            const routeKindConfig = new Set();
            for (const { routeKind } of configs) {
                if (routeKind === null) {
                    continue;
                }
                if (routeKindConfig.has(routeKind)) {
                    throw new RangeError(`Multiple route configuration for kind: ${routeKind}. ` +
                        'Configurations should have unique routeKinds');
                }
                routeKindConfig.add(routeKind);
            }
        }
        /**
         * @param {?} navigation
         * @return {?}
         */
        match(navigation) {
            if (!navigation.pathname.startsWith('/')) {
                throw new RangeError('Navigation has to made with pathname that starts with "/"');
            }
            /** @type {?} */
            let pathParts = getPathParts(navigation.pathname);
            /** @type {?} */
            let redirectionCount = 0;
            while (true) {
                /** @type {?} */
                let wasRedirected = false;
                for (const matcher of this.configMatchers) {
                    /** @type {?} */
                    const match = matcher.match(pathParts);
                    if (match.result) {
                        const { params, pathParts: newPathParts, isRedirection } = match;
                        if (isRedirection) {
                            pathParts = newPathParts;
                            wasRedirected = true;
                            break;
                        }
                        /** @type {?} */
                        const config = (/** @type {?} */ (matcher.config));
                        return {
                            routeKind: config.routeKind,
                            params,
                            pathname: getPathFromParts(newPathParts),
                            deepLinkProvider: config.deepLinkProvider || null,
                        };
                    }
                }
                if (wasRedirected) {
                    redirectionCount++;
                }
                if (redirectionCount > this.maxRedirection || !wasRedirected) {
                    // If not redirected, no need to rematch the routes. Abort.
                    break;
                }
            }
            if (redirectionCount > this.maxRedirection) {
                throw new Error(`Potential redirection loop (redirecting more than ` +
                    `${this.maxRedirection} times. Please do not have cycles in the ` +
                    `routes.`);
            }
            if (this.defaultRouteConfig) {
                /** @type {?} */
                const config = (/** @type {?} */ (this.defaultRouteConfig.config));
                return {
                    routeKind: config.routeKind,
                    params: {},
                    pathname: config.path,
                    deepLinkProvider: config.deepLinkProvider || null,
                };
            }
            return null;
        }
        /**
         * @param {?} routeKind
         * @param {?} params
         * @return {?}
         */
        matchByRouteKind(routeKind, params) {
            /** @type {?} */
            const matcher = this.routeKindToConfigMatchers.get(routeKind);
            if (!matcher || !matcher.config.routeKind) {
                throw new RangeError(`Requires configuration for routeKind: ${routeKind}`);
            }
            /** @type {?} */
            const match = matcher.matchByParams(params);
            /** @type {?} */
            const config = matcher.config;
            return {
                routeKind,
                params,
                pathname: getPathFromParts(match.pathParts),
                deepLinkProvider: config.deepLinkProvider || null,
            };
        }
    }
    /**
     * @param {?} path
     * @return {?}
     */
    function getPathParts(path) {
        return path.split('/').slice(1);
    }
    /**
     * @param {?} pathParts
     * @return {?}
     */
    function getPathFromParts(pathParts) {
        return '/' + pathParts.join('/');
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/app_routing/route_registry_types.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const ROUTE_CONFIGS_TOKEN = new InjectionToken('[App Routing] Route Config');

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/app_routing/route_registry_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class RouteRegistryModule {
        /**
         * @param {?} configsList
         */
        constructor(configsList) {
            this.routeConfigs = null;
            this.routeKindToNgComponent = new Map();
            if (!configsList) {
                return;
            }
            /** @type {?} */
            const configs = [];
            for (const routeDefList of configsList) {
                for (const routeDef of routeDefList) {
                    configs.push(routeDef);
                }
            }
            this.routeConfigs = new RouteConfigs(configs);
            configs.forEach((/**
             * @param {?} config
             * @return {?}
             */
            (config) => {
                if (config.routeKind) {
                    this.routeKindToNgComponent.set(config.routeKind, config.ngComponent);
                }
            }));
        }
        /**
         * Returns RouteConfigs of current route configuration. Returnsn null if no
         * routes are registered.
         * @return {?}
         */
        getRouteConfigs() {
            return this.routeConfigs;
        }
        /**
         * @param {?} routeKind
         * @return {?}
         */
        getNgComponentByRouteKind(routeKind) {
            return this.routeKindToNgComponent.get(routeKind) || null;
        }
        /**
         * An NgModule that registers routes.
         *
         * Note: especially because Polymer based TensorBoard requires relative paths
         * for making requests, although not required, prefer to have path that ends
         * with "/".
         *
         * Example:
         *
         * function routeProvider() {
         *   return [{
         *     path: '/experiments/',
         *     ngComponent: ScalarsDashboard,
         *     routeKind: RouteKind.EXPERIMENTS,
         *   }];
         * }
         *
         * \@NgModule({
         *   imports: [
         *     RouteRegistryModule.registerRoutes(routesProvider),
         *   ],
         *   entryComponents: [ScalarsDashboard]
         * })
         * @param {?} routeConfigProvider
         * @return {?}
         */
        static registerRoutes(routeConfigProvider) {
            return {
                ngModule: RouteRegistryModule,
                providers: [
                    {
                        provide: ROUTE_CONFIGS_TOKEN,
                        multi: true,
                        useFactory: routeConfigProvider,
                    },
                ],
            };
        }
    }
    RouteRegistryModule.decorators = [
        { type: NgModule, args: [{},] },
    ];
    /** @nocollapse */
    RouteRegistryModule.ctorParameters = () => [
        { type: Array, decorators: [{ type: Optional }, { type: Inject, args: [ROUTE_CONFIGS_TOKEN,] }] }
    ];
    /** @nocollapse */ RouteRegistryModule.ɵmod = ɵɵdefineNgModule({ type: RouteRegistryModule });
    /** @nocollapse */ RouteRegistryModule.ɵinj = ɵɵdefineInjector({ factory: function RouteRegistryModule_Factory(t) { return new (t || RouteRegistryModule)(ɵɵinject(ROUTE_CONFIGS_TOKEN, 8)); } });

    const APP_ROUTING_FEATURE_KEY = 'app_routing';

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    const getAppRoutingState = createFeatureSelector(APP_ROUTING_FEATURE_KEY);
    const getActiveRoute = createSelector(getAppRoutingState, (state) => {
        return state.activeRoute;
    });
    const getNextRouteForRouterOutletOnly = createSelector(getAppRoutingState, (state) => {
        return state.nextRoute;
    });
    const getRouteKind = createSelector(getActiveRoute, (activeRoute) => {
        return activeRoute ? activeRoute.routeKind : RouteKind.UNKNOWN;
    });
    const getRouteParams = createSelector(getActiveRoute, (activeRoute) => {
        return activeRoute ? activeRoute.params : {};
    });
    /**
     * Returns experiment ids activated by route. The value can be null if current
     * route does not have eids.
     */
    const getExperimentIdsFromRoute = createSelector(getRouteKind, getRouteParams, (routeKind, routeParams) => {
        return getExperimentIdsFromRouteParams(routeKind, routeParams);
    });
    const getRouteId$1 = createSelector(getRouteKind, getRouteParams, (routeKind, routeParams) => {
        return getRouteId(routeKind, routeParams);
    });
    const getExperimentIdToAliasMap = createSelector(getRouteKind, getRouteParams, (routeKind, routeParams) => {
        const idToDisplayName = {};
        if (routeKind !== RouteKind.COMPARE_EXPERIMENT) {
            return idToDisplayName;
        }
        const compareParams = routeParams;
        const nameAndIds = parseCompareExperimentStr(compareParams.experimentIds);
        for (const { id, name } of nameAndIds) {
            idToDisplayName[id] = name;
        }
        return idToDisplayName;
    });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/app_routing/views/router_outlet_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    const _c0$9 = ["routeContainer"];
    class RouterOutletComponent {
        /**
         * @param {?} componentFactoryResolver
         */
        constructor(componentFactoryResolver) {
            this.componentFactoryResolver = componentFactoryResolver;
        }
        /**
         * @param {?} changes
         * @return {?}
         */
        ngOnChanges(changes) {
            /** @type {?} */
            const activeComponentChange = changes['activeNgComponent'];
            if (activeComponentChange) {
                this.routeContainer.clear();
                if (activeComponentChange.currentValue) {
                    /** @type {?} */
                    const componentFactory = this.componentFactoryResolver.resolveComponentFactory(activeComponentChange.currentValue);
                    this.routeContainer.createComponent(componentFactory);
                }
            }
        }
    }
    RouterOutletComponent.decorators = [
        { type: Component, args: [{
                    selector: 'router-outlet-component',
                    template: ` <ng-container #routeContainer></ng-container> `,
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    RouterOutletComponent.ctorParameters = () => [
        { type: ComponentFactoryResolver }
    ];
    RouterOutletComponent.propDecorators = {
        routeContainer: [{ type: ViewChild, args: ['routeContainer', { static: true, read: ViewContainerRef },] }],
        activeNgComponent: [{ type: Input }]
    };
    /** @nocollapse */ RouterOutletComponent.ɵfac = function RouterOutletComponent_Factory(t) { return new (t || RouterOutletComponent)(ɵɵdirectiveInject(ComponentFactoryResolver)); };
    /** @nocollapse */ RouterOutletComponent.ɵcmp = ɵɵdefineComponent({ type: RouterOutletComponent, selectors: [["router-outlet-component"]], viewQuery: function RouterOutletComponent_Query(rf, ctx) { if (rf & 1) {
            ɵɵstaticViewQuery(_c0$9, true, ViewContainerRef);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.routeContainer = _t.first);
        } }, inputs: { activeNgComponent: "activeNgComponent" }, features: [ɵɵNgOnChangesFeature], decls: 2, vars: 0, consts: [["routeContainer", ""]], template: function RouterOutletComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementContainer(0, null, 0);
        } }, encapsulation: 2, changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/app_routing/views/router_outlet_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class RouterOutletContainer {
        /**
         * @param {?} store
         * @param {?} registry
         */
        constructor(store, registry) {
            this.store = store;
            this.registry = registry;
            this.activeNgComponent$ = combineLatest([
                this.store.select(getActiveRoute),
                this.store.select(getNextRouteForRouterOutletOnly),
            ]).pipe(map((/**
             * @param {?} __0
             * @return {?}
             */
            ([activeRoute, nextRoute]) => {
                if (!activeRoute) {
                    return null;
                }
                /** @type {?} */
                const isRouteTransitioning = nextRoute !== null &&
                    getRouteId(nextRoute.routeKind, nextRoute.params) !==
                        getRouteId(activeRoute.routeKind, activeRoute.params);
                return isRouteTransitioning
                    ? null
                    : this.registry.getNgComponentByRouteKind(activeRoute.routeKind);
            })));
        }
    }
    RouterOutletContainer.decorators = [
        { type: Component, args: [{
                    selector: 'router-outlet',
                    template: `
    <router-outlet-component
      [activeNgComponent]="activeNgComponent$ | async"
    ></router-outlet-component>
  `,
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    RouterOutletContainer.ctorParameters = () => [
        { type: Store },
        { type: RouteRegistryModule }
    ];
    /** @nocollapse */ RouterOutletContainer.ɵfac = function RouterOutletContainer_Factory(t) { return new (t || RouterOutletContainer)(ɵɵdirectiveInject(Store), ɵɵdirectiveInject(RouteRegistryModule)); };
    /** @nocollapse */ RouterOutletContainer.ɵcmp = ɵɵdefineComponent({ type: RouterOutletContainer, selectors: [["router-outlet"]], decls: 2, vars: 3, consts: [[3, "activeNgComponent"]], template: function RouterOutletContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelement(0, "router-outlet-component", 0);
            ɵɵpipe(1, "async");
        } if (rf & 2) {
            ɵɵproperty("activeNgComponent", ɵɵpipeBind1(1, 1, ctx.activeNgComponent$));
        } }, directives: [RouterOutletComponent], pipes: [AsyncPipe], encapsulation: 2, changeDetection: 0 });

    class LayoutModule {
    }
    LayoutModule.ɵmod = ɵɵdefineNgModule({ type: LayoutModule });
    LayoutModule.ɵinj = ɵɵdefineInjector({ factory: function LayoutModule_Factory(t) { return new (t || LayoutModule)(); } });

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/layout/media-matcher.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Global registry for all dynamically-created, injected media queries.
     * @type {?}
     */
    const mediaQueriesForWebkitCompatibility = new Set();
    /**
     * Style tag that holds all of the dynamically-created media queries.
     * @type {?}
     */
    let mediaQueryStyleNode;
    /**
     * A utility for calling matchMedia queries.
     */
    class MediaMatcher {
        /**
         * @param {?} _platform
         */
        constructor(_platform) {
            this._platform = _platform;
            this._matchMedia = this._platform.isBrowser && window.matchMedia ?
                // matchMedia is bound to the window scope intentionally as it is an illegal invocation to
                // call it from a different scope.
                window.matchMedia.bind(window) :
                noopMatchMedia;
        }
        /**
         * Evaluates the given media query and returns the native MediaQueryList from which results
         * can be retrieved.
         * Confirms the layout engine will trigger for the selector query provided and returns the
         * MediaQueryList for the query provided.
         * @param {?} query
         * @return {?}
         */
        matchMedia(query) {
            if (this._platform.WEBKIT) {
                createEmptyStyleRule(query);
            }
            return this._matchMedia(query);
        }
    }
    MediaMatcher.ɵfac = function MediaMatcher_Factory(t) { return new (t || MediaMatcher)(ɵɵinject(Platform)); };
    /** @nocollapse */
    MediaMatcher.ctorParameters = () => [
        { type: Platform }
    ];
    /** @nocollapse */ MediaMatcher.ɵprov = ɵɵdefineInjectable({ factory: function MediaMatcher_Factory() { return new MediaMatcher(ɵɵinject(Platform)); }, token: MediaMatcher, providedIn: "root" });
    /**
     * For Webkit engines that only trigger the MediaQueryListListener when
     * there is at least one CSS selector for the respective media query.
     * @param {?} query
     * @return {?}
     */
    function createEmptyStyleRule(query) {
        if (mediaQueriesForWebkitCompatibility.has(query)) {
            return;
        }
        try {
            if (!mediaQueryStyleNode) {
                mediaQueryStyleNode = document.createElement('style');
                mediaQueryStyleNode.setAttribute('type', 'text/css');
                (/** @type {?} */ (document.head)).appendChild(mediaQueryStyleNode);
            }
            if (mediaQueryStyleNode.sheet) {
                ((/** @type {?} */ (mediaQueryStyleNode.sheet)))
                    .insertRule(`@media ${query} {.fx-query-test{ }}`, 0);
                mediaQueriesForWebkitCompatibility.add(query);
            }
        }
        catch (e) {
            console.error(e);
        }
    }
    /**
     * No-op matchMedia replacement for non-browser platforms.
     * @param {?} query
     * @return {?}
     */
    function noopMatchMedia(query) {
        // Use `as any` here to avoid adding additional necessary properties for
        // the noop matcher.
        return (/** @type {?} */ ({
            matches: query === 'all' || query === '',
            media: query,
            addListener: (/**
             * @return {?}
             */
            () => { }),
            removeListener: (/**
             * @return {?}
             */
            () => { })
        }));
    }
    /**
     * Utility for checking the matching state of \@media queries.
     */
    class BreakpointObserver {
        /**
         * @param {?} _mediaMatcher
         * @param {?} _zone
         */
        constructor(_mediaMatcher, _zone) {
            this._mediaMatcher = _mediaMatcher;
            this._zone = _zone;
            /**
             * A map of all media queries currently being listened for.
             */
            this._queries = new Map();
            /**
             * A subject for all other observables to takeUntil based on.
             */
            this._destroySubject = new Subject();
        }
        /**
         * Completes the active subject, signalling to all other observables to complete.
         * @return {?}
         */
        ngOnDestroy() {
            this._destroySubject.next();
            this._destroySubject.complete();
        }
        /**
         * Whether one or more media queries match the current viewport size.
         * @param {?} value One or more media queries to check.
         * @return {?} Whether any of the media queries match.
         */
        isMatched(value) {
            /** @type {?} */
            const queries = splitQueries(coerceArray(value));
            return queries.some((/**
             * @param {?} mediaQuery
             * @return {?}
             */
            mediaQuery => this._registerQuery(mediaQuery).mql.matches));
        }
        /**
         * Gets an observable of results for the given queries that will emit new results for any changes
         * in matching of the given queries.
         * @param {?} value One or more media queries to check.
         * @return {?} A stream of matches for the given queries.
         */
        observe(value) {
            /** @type {?} */
            const queries = splitQueries(coerceArray(value));
            /** @type {?} */
            const observables = queries.map((/**
             * @param {?} query
             * @return {?}
             */
            query => this._registerQuery(query).observable));
            /** @type {?} */
            let stateObservable = combineLatest(observables);
            // Emit the first state immediately, and then debounce the subsequent emissions.
            stateObservable = concat(stateObservable.pipe(take(1)), stateObservable.pipe(skip(1), debounceTime(0)));
            return stateObservable.pipe(map((/**
             * @param {?} breakpointStates
             * @return {?}
             */
            (breakpointStates) => {
                /** @type {?} */
                const response = {
                    matches: false,
                    breakpoints: {},
                };
                breakpointStates.forEach((/**
                 * @param {?} state
                 * @return {?}
                 */
                (state) => {
                    response.matches = response.matches || state.matches;
                    response.breakpoints[state.query] = state.matches;
                }));
                return response;
            })));
        }
        /**
         * Registers a specific query to be listened for.
         * @private
         * @param {?} query
         * @return {?}
         */
        _registerQuery(query) {
            // Only set up a new MediaQueryList if it is not already being listened for.
            if (this._queries.has(query)) {
                return (/** @type {?} */ (this._queries.get(query)));
            }
            /** @type {?} */
            const mql = this._mediaMatcher.matchMedia(query);
            // Create callback for match changes and add it is as a listener.
            /** @type {?} */
            const queryObservable = new Observable((/**
             * @param {?} observer
             * @return {?}
             */
            (observer) => {
                // Listener callback methods are wrapped to be placed back in ngZone. Callbacks must be placed
                // back into the zone because matchMedia is only included in Zone.js by loading the
                // webapis-media-query.js file alongside the zone.js file.  Additionally, some browsers do not
                // have MediaQueryList inherit from EventTarget, which causes inconsistencies in how Zone.js
                // patches it.
                /** @type {?} */
                const handler = (/**
                 * @param {?} e
                 * @return {?}
                 */
                (e) => this._zone.run((/**
                 * @return {?}
                 */
                () => observer.next(e))));
                mql.addListener(handler);
                return (/**
                 * @return {?}
                 */
                () => {
                    mql.removeListener(handler);
                });
            })).pipe(startWith(mql), map((/**
             * @param {?} nextMql
             * @return {?}
             */
            (nextMql) => ({ query, matches: nextMql.matches }))), takeUntil(this._destroySubject));
            // Add the MediaQueryList to the set of queries.
            /** @type {?} */
            const output = { observable: queryObservable, mql };
            this._queries.set(query, output);
            return output;
        }
    }
    BreakpointObserver.ɵfac = function BreakpointObserver_Factory(t) { return new (t || BreakpointObserver)(ɵɵinject(MediaMatcher), ɵɵinject(NgZone)); };
    /** @nocollapse */
    BreakpointObserver.ctorParameters = () => [
        { type: MediaMatcher },
        { type: NgZone }
    ];
    /** @nocollapse */ BreakpointObserver.ɵprov = ɵɵdefineInjectable({ factory: function BreakpointObserver_Factory() { return new BreakpointObserver(ɵɵinject(MediaMatcher), ɵɵinject(NgZone)); }, token: BreakpointObserver, providedIn: "root" });
    /**
     * Split each query string into separate query strings if two queries are provided as comma
     * separated.
     * @param {?} queries
     * @return {?}
     */
    function splitQueries(queries) {
        return queries.map((/**
         * @param {?} query
         * @return {?}
         */
        (query) => query.split(',')))
            .reduce((/**
         * @param {?} a1
         * @param {?} a2
         * @return {?}
         */
        (a1, a2) => a1.concat(a2)))
            .map((/**
         * @param {?} query
         * @return {?}
         */
        query => query.trim()));
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/layout/breakpoints.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    // PascalCase is being used as Breakpoints is used like an enum.
    // tslint:disable-next-line:variable-name
    /**
     * @license
     * Copyright Google LLC All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     * @type {?}
     */
    const Breakpoints = {
        XSmall: '(max-width: 599.99px)',
        Small: '(min-width: 600px) and (max-width: 959.99px)',
        Medium: '(min-width: 960px) and (max-width: 1279.99px)',
        Large: '(min-width: 1280px) and (max-width: 1919.99px)',
        XLarge: '(min-width: 1920px)',
        Handset: '(max-width: 599.99px) and (orientation: portrait), ' +
            '(max-width: 959.99px) and (orientation: landscape)',
        Tablet: '(min-width: 600px) and (max-width: 839.99px) and (orientation: portrait), ' +
            '(min-width: 960px) and (max-width: 1279.99px) and (orientation: landscape)',
        Web: '(min-width: 840px) and (orientation: portrait), ' +
            '(min-width: 1280px) and (orientation: landscape)',
        HandsetPortrait: '(max-width: 599.99px) and (orientation: portrait)',
        TabletPortrait: '(min-width: 600px) and (max-width: 839.99px) and (orientation: portrait)',
        WebPortrait: '(min-width: 840px) and (orientation: portrait)',
        HandsetLandscape: '(max-width: 959.99px) and (orientation: landscape)',
        TabletLandscape: '(min-width: 960px) and (max-width: 1279.99px) and (orientation: landscape)',
        WebLandscape: '(min-width: 1280px) and (orientation: landscape)',
    };

    function SimpleSnackBar_div_2_Template(rf, ctx) { if (rf & 1) {
        const _r2 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "div", 1);
        ɵɵelementStart(1, "button", 2);
        ɵɵlistener("click", function SimpleSnackBar_div_2_Template_button_click_1_listener() { ɵɵrestoreView(_r2); const ctx_r1 = ɵɵnextContext(); return ctx_r1.action(); });
        ɵɵtext(2);
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r0 = ɵɵnextContext();
        ɵɵadvance(2);
        ɵɵtextInterpolate(ctx_r0.data.action);
    } }
    function MatSnackBarContainer_ng_template_0_Template(rf, ctx) { }
    /**
     * Maximum amount of milliseconds that can be passed into setTimeout.
     * @type {?}
     */
    const MAX_TIMEOUT = Math.pow(2, 31) - 1;
    /**
     * Reference to a snack bar dispatched from the snack bar service.
     * @template T
     */
    class MatSnackBarRef {
        /**
         * @param {?} containerInstance
         * @param {?} _overlayRef
         */
        constructor(containerInstance, _overlayRef) {
            this._overlayRef = _overlayRef;
            /**
             * Subject for notifying the user that the snack bar has been dismissed.
             */
            this._afterDismissed = new Subject();
            /**
             * Subject for notifying the user that the snack bar has opened and appeared.
             */
            this._afterOpened = new Subject();
            /**
             * Subject for notifying the user that the snack bar action was called.
             */
            this._onAction = new Subject();
            /**
             * Whether the snack bar was dismissed using the action button.
             */
            this._dismissedByAction = false;
            this.containerInstance = containerInstance;
            // Dismiss snackbar on action.
            this.onAction().subscribe((/**
             * @return {?}
             */
            () => this.dismiss()));
            containerInstance._onExit.subscribe((/**
             * @return {?}
             */
            () => this._finishDismiss()));
        }
        /**
         * Dismisses the snack bar.
         * @return {?}
         */
        dismiss() {
            if (!this._afterDismissed.closed) {
                this.containerInstance.exit();
            }
            clearTimeout(this._durationTimeoutId);
        }
        /**
         * Marks the snackbar action clicked.
         * @return {?}
         */
        dismissWithAction() {
            if (!this._onAction.closed) {
                this._dismissedByAction = true;
                this._onAction.next();
                this._onAction.complete();
            }
        }
        /**
         * Marks the snackbar action clicked.
         * @deprecated Use `dismissWithAction` instead.
         * \@breaking-change 8.0.0
         * @return {?}
         */
        closeWithAction() {
            this.dismissWithAction();
        }
        /**
         * Dismisses the snack bar after some duration
         * @param {?} duration
         * @return {?}
         */
        _dismissAfter(duration) {
            // Note that we need to cap the duration to the maximum value for setTimeout, because
            // it'll revert to 1 if somebody passes in something greater (e.g. `Infinity`). See #17234.
            this._durationTimeoutId = setTimeout((/**
             * @return {?}
             */
            () => this.dismiss()), Math.min(duration, MAX_TIMEOUT));
        }
        /**
         * Marks the snackbar as opened
         * @return {?}
         */
        _open() {
            if (!this._afterOpened.closed) {
                this._afterOpened.next();
                this._afterOpened.complete();
            }
        }
        /**
         * Cleans up the DOM after closing.
         * @private
         * @return {?}
         */
        _finishDismiss() {
            this._overlayRef.dispose();
            if (!this._onAction.closed) {
                this._onAction.complete();
            }
            this._afterDismissed.next({ dismissedByAction: this._dismissedByAction });
            this._afterDismissed.complete();
            this._dismissedByAction = false;
        }
        /**
         * Gets an observable that is notified when the snack bar is finished closing.
         * @return {?}
         */
        afterDismissed() {
            return this._afterDismissed.asObservable();
        }
        /**
         * Gets an observable that is notified when the snack bar has opened and appeared.
         * @return {?}
         */
        afterOpened() {
            return this.containerInstance._onEnter;
        }
        /**
         * Gets an observable that is notified when the snack bar action is called.
         * @return {?}
         */
        onAction() {
            return this._onAction.asObservable();
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/snack-bar/snack-bar-config.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Injection token that can be used to access the data that was passed in to a snack bar.
     * @type {?}
     */
    const MAT_SNACK_BAR_DATA = new InjectionToken('MatSnackBarData');
    /**
     * Configuration used when opening a snack-bar.
     * @template D
     */
    class MatSnackBarConfig {
        constructor() {
            /**
             * The politeness level for the MatAriaLiveAnnouncer announcement.
             */
            this.politeness = 'assertive';
            /**
             * Message to be announced by the LiveAnnouncer. When opening a snackbar without a custom
             * component or template, the announcement message will default to the specified message.
             */
            this.announcementMessage = '';
            /**
             * The length of time in milliseconds to wait before automatically dismissing the snack bar.
             */
            this.duration = 0;
            /**
             * Data being injected into the child component.
             */
            this.data = null;
            /**
             * The horizontal position to place the snack bar.
             */
            this.horizontalPosition = 'center';
            /**
             * The vertical position to place the snack bar.
             */
            this.verticalPosition = 'bottom';
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/snack-bar/simple-snack-bar.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * A component used to open as the default snack bar, matching material spec.
     * This should only be used internally by the snack bar service.
     */
    class SimpleSnackBar {
        /**
         * @param {?} snackBarRef
         * @param {?} data
         */
        constructor(snackBarRef, data) {
            this.snackBarRef = snackBarRef;
            this.data = data;
        }
        /**
         * Performs the action on the snack bar.
         * @return {?}
         */
        action() {
            this.snackBarRef.dismissWithAction();
        }
        /**
         * If the action button should be shown.
         * @return {?}
         */
        get hasAction() {
            return !!this.data.action;
        }
    }
    SimpleSnackBar.ɵfac = function SimpleSnackBar_Factory(t) { return new (t || SimpleSnackBar)(ɵɵdirectiveInject(MatSnackBarRef), ɵɵdirectiveInject(MAT_SNACK_BAR_DATA)); };
    SimpleSnackBar.ɵcmp = ɵɵdefineComponent({ type: SimpleSnackBar, selectors: [["simple-snack-bar"]], hostAttrs: [1, "mat-simple-snackbar"], decls: 3, vars: 2, consts: [["class", "mat-simple-snackbar-action", 4, "ngIf"], [1, "mat-simple-snackbar-action"], ["mat-button", "", 3, "click"]], template: function SimpleSnackBar_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "span");
            ɵɵtext(1);
            ɵɵelementEnd();
            ɵɵtemplate(2, SimpleSnackBar_div_2_Template, 3, 1, "div", 0);
        } if (rf & 2) {
            ɵɵadvance(1);
            ɵɵtextInterpolate(ctx.data.message);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx.hasAction);
        } }, directives: [NgIf, MatButton], styles: [".mat-simple-snackbar{display:flex;justify-content:space-between;align-items:center;line-height:20px;opacity:1}.mat-simple-snackbar-action{flex-shrink:0;margin:-8px -8px -8px 8px}.mat-simple-snackbar-action button{max-height:36px;min-width:0}[dir=rtl] .mat-simple-snackbar-action{margin-left:-8px;margin-right:8px}\n"], encapsulation: 2, changeDetection: 0 });
    /** @nocollapse */
    SimpleSnackBar.ctorParameters = () => [
        { type: MatSnackBarRef },
        { type: undefined, decorators: [{ type: Inject, args: [MAT_SNACK_BAR_DATA,] }] }
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/snack-bar/snack-bar-animations.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Animations used by the Material snack bar.
     * \@docs-private
     * @type {?}
     */
    const matSnackBarAnimations = {
        /**
         * Animation that shows and hides a snack bar.
         */
        snackBarState: trigger('state', [
            state('void, hidden', style({
                transform: 'scale(0.8)',
                opacity: 0,
            })),
            state('visible', style({
                transform: 'scale(1)',
                opacity: 1,
            })),
            transition('* => visible', animate('150ms cubic-bezier(0, 0, 0.2, 1)')),
            transition('* => void, * => hidden', animate('75ms cubic-bezier(0.4, 0.0, 1, 1)', style({
                opacity: 0
            }))),
        ])
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/snack-bar/snack-bar-container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Internal component that wraps user-provided snack bar content.
     * \@docs-private
     */
    class MatSnackBarContainer extends BasePortalOutlet {
        /**
         * @param {?} _ngZone
         * @param {?} _elementRef
         * @param {?} _changeDetectorRef
         * @param {?} snackBarConfig
         */
        constructor(_ngZone, _elementRef, _changeDetectorRef, snackBarConfig) {
            super();
            this._ngZone = _ngZone;
            this._elementRef = _elementRef;
            this._changeDetectorRef = _changeDetectorRef;
            this.snackBarConfig = snackBarConfig;
            /**
             * Whether the component has been destroyed.
             */
            this._destroyed = false;
            /**
             * Subject for notifying that the snack bar has exited from view.
             */
            this._onExit = new Subject();
            /**
             * Subject for notifying that the snack bar has finished entering the view.
             */
            this._onEnter = new Subject();
            /**
             * The state of the snack bar animations.
             */
            this._animationState = 'void';
            /**
             * Attaches a DOM portal to the snack bar container.
             * @deprecated To be turned into a method.
             * \@breaking-change 10.0.0
             */
            this.attachDomPortal = (/**
             * @param {?} portal
             * @return {?}
             */
            (portal) => {
                this._assertNotAttached();
                this._applySnackBarClasses();
                return this._portalOutlet.attachDomPortal(portal);
            });
            // Based on the ARIA spec, `alert` and `status` roles have an
            // implicit `assertive` and `polite` politeness respectively.
            if (snackBarConfig.politeness === 'assertive' && !snackBarConfig.announcementMessage) {
                this._role = 'alert';
            }
            else if (snackBarConfig.politeness === 'off') {
                this._role = null;
            }
            else {
                this._role = 'status';
            }
        }
        /**
         * Attach a component portal as content to this snack bar container.
         * @template T
         * @param {?} portal
         * @return {?}
         */
        attachComponentPortal(portal) {
            this._assertNotAttached();
            this._applySnackBarClasses();
            return this._portalOutlet.attachComponentPortal(portal);
        }
        /**
         * Attach a template portal as content to this snack bar container.
         * @template C
         * @param {?} portal
         * @return {?}
         */
        attachTemplatePortal(portal) {
            this._assertNotAttached();
            this._applySnackBarClasses();
            return this._portalOutlet.attachTemplatePortal(portal);
        }
        /**
         * Handle end of animations, updating the state of the snackbar.
         * @param {?} event
         * @return {?}
         */
        onAnimationEnd(event) {
            const { fromState, toState } = event;
            if ((toState === 'void' && fromState !== 'void') || toState === 'hidden') {
                this._completeExit();
            }
            if (toState === 'visible') {
                // Note: we shouldn't use `this` inside the zone callback,
                // because it can cause a memory leak.
                /** @type {?} */
                const onEnter = this._onEnter;
                this._ngZone.run((/**
                 * @return {?}
                 */
                () => {
                    onEnter.next();
                    onEnter.complete();
                }));
            }
        }
        /**
         * Begin animation of snack bar entrance into view.
         * @return {?}
         */
        enter() {
            if (!this._destroyed) {
                this._animationState = 'visible';
                this._changeDetectorRef.detectChanges();
            }
        }
        /**
         * Begin animation of the snack bar exiting from view.
         * @return {?}
         */
        exit() {
            // Note: this one transitions to `hidden`, rather than `void`, in order to handle the case
            // where multiple snack bars are opened in quick succession (e.g. two consecutive calls to
            // `MatSnackBar.open`).
            this._animationState = 'hidden';
            // Mark this element with an 'exit' attribute to indicate that the snackbar has
            // been dismissed and will soon be removed from the DOM. This is used by the snackbar
            // test harness.
            this._elementRef.nativeElement.setAttribute('mat-exit', '');
            return this._onExit;
        }
        /**
         * Makes sure the exit callbacks have been invoked when the element is destroyed.
         * @return {?}
         */
        ngOnDestroy() {
            this._destroyed = true;
            this._completeExit();
        }
        /**
         * Waits for the zone to settle before removing the element. Helps prevent
         * errors where we end up removing an element which is in the middle of an animation.
         * @private
         * @return {?}
         */
        _completeExit() {
            this._ngZone.onMicrotaskEmpty.asObservable().pipe(take(1)).subscribe((/**
             * @return {?}
             */
            () => {
                this._onExit.next();
                this._onExit.complete();
            }));
        }
        /**
         * Applies the various positioning and user-configured CSS classes to the snack bar.
         * @private
         * @return {?}
         */
        _applySnackBarClasses() {
            /** @type {?} */
            const element = this._elementRef.nativeElement;
            /** @type {?} */
            const panelClasses = this.snackBarConfig.panelClass;
            if (panelClasses) {
                if (Array.isArray(panelClasses)) {
                    // Note that we can't use a spread here, because IE doesn't support multiple arguments.
                    panelClasses.forEach((/**
                     * @param {?} cssClass
                     * @return {?}
                     */
                    cssClass => element.classList.add(cssClass)));
                }
                else {
                    element.classList.add(panelClasses);
                }
            }
            if (this.snackBarConfig.horizontalPosition === 'center') {
                element.classList.add('mat-snack-bar-center');
            }
            if (this.snackBarConfig.verticalPosition === 'top') {
                element.classList.add('mat-snack-bar-top');
            }
        }
        /**
         * Asserts that no content is already attached to the container.
         * @private
         * @return {?}
         */
        _assertNotAttached() {
            if (this._portalOutlet.hasAttached()) {
                throw Error('Attempting to attach snack bar content after content is already attached');
            }
        }
    }
    MatSnackBarContainer.ɵfac = function MatSnackBarContainer_Factory(t) { return new (t || MatSnackBarContainer)(ɵɵdirectiveInject(NgZone), ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(ChangeDetectorRef), ɵɵdirectiveInject(MatSnackBarConfig)); };
    MatSnackBarContainer.ɵcmp = ɵɵdefineComponent({ type: MatSnackBarContainer, selectors: [["snack-bar-container"]], viewQuery: function MatSnackBarContainer_Query(rf, ctx) { if (rf & 1) {
            ɵɵstaticViewQuery(CdkPortalOutlet, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._portalOutlet = _t.first);
        } }, hostAttrs: [1, "mat-snack-bar-container"], hostVars: 2, hostBindings: function MatSnackBarContainer_HostBindings(rf, ctx) { if (rf & 1) {
            ɵɵcomponentHostSyntheticListener("@state.done", function MatSnackBarContainer_animation_state_done_HostBindingHandler($event) { return ctx.onAnimationEnd($event); });
        } if (rf & 2) {
            ɵɵattribute("role", ctx._role);
            ɵɵupdateSyntheticHostBinding("@state", ctx._animationState);
        } }, features: [ɵɵInheritDefinitionFeature], decls: 1, vars: 0, consts: [["cdkPortalOutlet", ""]], template: function MatSnackBarContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵtemplate(0, MatSnackBarContainer_ng_template_0_Template, 0, 0, "ng-template", 0);
        } }, directives: [CdkPortalOutlet], styles: [".mat-snack-bar-container{border-radius:4px;box-sizing:border-box;display:block;margin:24px;max-width:33vw;min-width:344px;padding:14px 16px;min-height:48px;transform-origin:center}.cdk-high-contrast-active .mat-snack-bar-container{border:solid 1px}.mat-snack-bar-handset{width:100%}.mat-snack-bar-handset .mat-snack-bar-container{margin:8px;max-width:100%;min-width:0;width:100%}\n"], encapsulation: 2, data: { animation: [matSnackBarAnimations.snackBarState] } });
    /** @nocollapse */
    MatSnackBarContainer.ctorParameters = () => [
        { type: NgZone },
        { type: ElementRef },
        { type: ChangeDetectorRef },
        { type: MatSnackBarConfig }
    ];
    MatSnackBarContainer.propDecorators = {
        _portalOutlet: [{ type: ViewChild, args: [CdkPortalOutlet, { static: true },] }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/snack-bar/snack-bar-module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class MatSnackBarModule {
    }
    MatSnackBarModule.ɵmod = ɵɵdefineNgModule({ type: MatSnackBarModule });
    MatSnackBarModule.ɵinj = ɵɵdefineInjector({ factory: function MatSnackBarModule_Factory(t) { return new (t || MatSnackBarModule)(); }, imports: [[
                OverlayModule,
                PortalModule,
                CommonModule,
                MatButtonModule,
                MatCommonModule,
            ],
            MatCommonModule] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(MatSnackBarModule, { declarations: function () { return [MatSnackBarContainer,
            SimpleSnackBar]; }, imports: function () { return [OverlayModule,
            PortalModule,
            CommonModule,
            MatButtonModule,
            MatCommonModule]; }, exports: function () { return [MatSnackBarContainer,
            MatCommonModule]; } }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/snack-bar/snack-bar.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Injection token that can be used to specify default snack bar.
     * @type {?}
     */
    const MAT_SNACK_BAR_DEFAULT_OPTIONS = new InjectionToken('mat-snack-bar-default-options', {
        providedIn: 'root',
        factory: MAT_SNACK_BAR_DEFAULT_OPTIONS_FACTORY,
    });
    /**
     * \@docs-private
     * @return {?}
     */
    function MAT_SNACK_BAR_DEFAULT_OPTIONS_FACTORY() {
        return new MatSnackBarConfig();
    }
    /**
     * Service to dispatch Material Design snack bar messages.
     */
    class MatSnackBar {
        /**
         * @param {?} _overlay
         * @param {?} _live
         * @param {?} _injector
         * @param {?} _breakpointObserver
         * @param {?} _parentSnackBar
         * @param {?} _defaultConfig
         */
        constructor(_overlay, _live, _injector, _breakpointObserver, _parentSnackBar, _defaultConfig) {
            this._overlay = _overlay;
            this._live = _live;
            this._injector = _injector;
            this._breakpointObserver = _breakpointObserver;
            this._parentSnackBar = _parentSnackBar;
            this._defaultConfig = _defaultConfig;
            /**
             * Reference to the current snack bar in the view *at this level* (in the Angular injector tree).
             * If there is a parent snack-bar service, all operations should delegate to that parent
             * via `_openedSnackBarRef`.
             */
            this._snackBarRefAtThisLevel = null;
        }
        /**
         * Reference to the currently opened snackbar at *any* level.
         * @return {?}
         */
        get _openedSnackBarRef() {
            /** @type {?} */
            const parent = this._parentSnackBar;
            return parent ? parent._openedSnackBarRef : this._snackBarRefAtThisLevel;
        }
        /**
         * @param {?} value
         * @return {?}
         */
        set _openedSnackBarRef(value) {
            if (this._parentSnackBar) {
                this._parentSnackBar._openedSnackBarRef = value;
            }
            else {
                this._snackBarRefAtThisLevel = value;
            }
        }
        /**
         * Creates and dispatches a snack bar with a custom component for the content, removing any
         * currently opened snack bars.
         *
         * @template T
         * @param {?} component Component to be instantiated.
         * @param {?=} config Extra configuration for the snack bar.
         * @return {?}
         */
        openFromComponent(component, config) {
            return (/** @type {?} */ (this._attach(component, config)));
        }
        /**
         * Creates and dispatches a snack bar with a custom template for the content, removing any
         * currently opened snack bars.
         *
         * @param {?} template Template to be instantiated.
         * @param {?=} config Extra configuration for the snack bar.
         * @return {?}
         */
        openFromTemplate(template, config) {
            return this._attach(template, config);
        }
        /**
         * Opens a snackbar with a message and an optional action.
         * @param {?} message The message to show in the snackbar.
         * @param {?=} action The label for the snackbar action.
         * @param {?=} config Additional configuration options for the snackbar.
         * @return {?}
         */
        open(message, action = '', config) {
            /** @type {?} */
            const _config = Object.assign(Object.assign({}, this._defaultConfig), config);
            // Since the user doesn't have access to the component, we can
            // override the data to pass in our own message and action.
            _config.data = { message, action };
            if (!_config.announcementMessage) {
                _config.announcementMessage = message;
            }
            return this.openFromComponent(SimpleSnackBar, _config);
        }
        /**
         * Dismisses the currently-visible snack bar.
         * @return {?}
         */
        dismiss() {
            if (this._openedSnackBarRef) {
                this._openedSnackBarRef.dismiss();
            }
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            // Only dismiss the snack bar at the current level on destroy.
            if (this._snackBarRefAtThisLevel) {
                this._snackBarRefAtThisLevel.dismiss();
            }
        }
        /**
         * Attaches the snack bar container component to the overlay.
         * @private
         * @param {?} overlayRef
         * @param {?} config
         * @return {?}
         */
        _attachSnackBarContainer(overlayRef, config) {
            /** @type {?} */
            const userInjector = config && config.viewContainerRef && config.viewContainerRef.injector;
            /** @type {?} */
            const injector = new PortalInjector(userInjector || this._injector, new WeakMap([
                [MatSnackBarConfig, config]
            ]));
            /** @type {?} */
            const containerPortal = new ComponentPortal(MatSnackBarContainer, config.viewContainerRef, injector);
            /** @type {?} */
            const containerRef = overlayRef.attach(containerPortal);
            containerRef.instance.snackBarConfig = config;
            return containerRef.instance;
        }
        /**
         * Places a new component or a template as the content of the snack bar container.
         * @private
         * @template T
         * @param {?} content
         * @param {?=} userConfig
         * @return {?}
         */
        _attach(content, userConfig) {
            /** @type {?} */
            const config = Object.assign(Object.assign(Object.assign({}, new MatSnackBarConfig()), this._defaultConfig), userConfig);
            /** @type {?} */
            const overlayRef = this._createOverlay(config);
            /** @type {?} */
            const container = this._attachSnackBarContainer(overlayRef, config);
            /** @type {?} */
            const snackBarRef = new MatSnackBarRef(container, overlayRef);
            if (content instanceof TemplateRef) {
                /** @type {?} */
                const portal = new TemplatePortal(content, (/** @type {?} */ (null)), (/** @type {?} */ ({
                    $implicit: config.data,
                    snackBarRef
                })));
                snackBarRef.instance = container.attachTemplatePortal(portal);
            }
            else {
                /** @type {?} */
                const injector = this._createInjector(config, snackBarRef);
                /** @type {?} */
                const portal = new ComponentPortal(content, undefined, injector);
                /** @type {?} */
                const contentRef = container.attachComponentPortal(portal);
                // We can't pass this via the injector, because the injector is created earlier.
                snackBarRef.instance = contentRef.instance;
            }
            // Subscribe to the breakpoint observer and attach the mat-snack-bar-handset class as
            // appropriate. This class is applied to the overlay element because the overlay must expand to
            // fill the width of the screen for full width snackbars.
            this._breakpointObserver.observe(Breakpoints.HandsetPortrait).pipe(takeUntil(overlayRef.detachments())).subscribe((/**
             * @param {?} state
             * @return {?}
             */
            state => {
                /** @type {?} */
                const classList = overlayRef.overlayElement.classList;
                /** @type {?} */
                const className = 'mat-snack-bar-handset';
                state.matches ? classList.add(className) : classList.remove(className);
            }));
            this._animateSnackBar(snackBarRef, config);
            this._openedSnackBarRef = snackBarRef;
            return this._openedSnackBarRef;
        }
        /**
         * Animates the old snack bar out and the new one in.
         * @private
         * @param {?} snackBarRef
         * @param {?} config
         * @return {?}
         */
        _animateSnackBar(snackBarRef, config) {
            // When the snackbar is dismissed, clear the reference to it.
            snackBarRef.afterDismissed().subscribe((/**
             * @return {?}
             */
            () => {
                // Clear the snackbar ref if it hasn't already been replaced by a newer snackbar.
                if (this._openedSnackBarRef == snackBarRef) {
                    this._openedSnackBarRef = null;
                }
                if (config.announcementMessage) {
                    this._live.clear();
                }
            }));
            if (this._openedSnackBarRef) {
                // If a snack bar is already in view, dismiss it and enter the
                // new snack bar after exit animation is complete.
                this._openedSnackBarRef.afterDismissed().subscribe((/**
                 * @return {?}
                 */
                () => {
                    snackBarRef.containerInstance.enter();
                }));
                this._openedSnackBarRef.dismiss();
            }
            else {
                // If no snack bar is in view, enter the new snack bar.
                snackBarRef.containerInstance.enter();
            }
            // If a dismiss timeout is provided, set up dismiss based on after the snackbar is opened.
            if (config.duration && config.duration > 0) {
                snackBarRef.afterOpened().subscribe((/**
                 * @return {?}
                 */
                () => snackBarRef._dismissAfter((/** @type {?} */ (config.duration)))));
            }
            if (config.announcementMessage) {
                this._live.announce(config.announcementMessage, config.politeness);
            }
        }
        /**
         * Creates a new overlay and places it in the correct location.
         * @private
         * @param {?} config The user-specified snack bar config.
         * @return {?}
         */
        _createOverlay(config) {
            /** @type {?} */
            const overlayConfig = new OverlayConfig();
            overlayConfig.direction = config.direction;
            /** @type {?} */
            let positionStrategy = this._overlay.position().global();
            // Set horizontal position.
            /** @type {?} */
            const isRtl = config.direction === 'rtl';
            /** @type {?} */
            const isLeft = (config.horizontalPosition === 'left' ||
                (config.horizontalPosition === 'start' && !isRtl) ||
                (config.horizontalPosition === 'end' && isRtl));
            /** @type {?} */
            const isRight = !isLeft && config.horizontalPosition !== 'center';
            if (isLeft) {
                positionStrategy.left('0');
            }
            else if (isRight) {
                positionStrategy.right('0');
            }
            else {
                positionStrategy.centerHorizontally();
            }
            // Set horizontal position.
            if (config.verticalPosition === 'top') {
                positionStrategy.top('0');
            }
            else {
                positionStrategy.bottom('0');
            }
            overlayConfig.positionStrategy = positionStrategy;
            return this._overlay.create(overlayConfig);
        }
        /**
         * Creates an injector to be used inside of a snack bar component.
         * @private
         * @template T
         * @param {?} config Config that was used to create the snack bar.
         * @param {?} snackBarRef Reference to the snack bar.
         * @return {?}
         */
        _createInjector(config, snackBarRef) {
            /** @type {?} */
            const userInjector = config && config.viewContainerRef && config.viewContainerRef.injector;
            return new PortalInjector(userInjector || this._injector, new WeakMap([
                [MatSnackBarRef, snackBarRef],
                [MAT_SNACK_BAR_DATA, config.data]
            ]));
        }
    }
    MatSnackBar.ɵfac = function MatSnackBar_Factory(t) { return new (t || MatSnackBar)(ɵɵinject(Overlay), ɵɵinject(LiveAnnouncer), ɵɵinject(Injector), ɵɵinject(BreakpointObserver), ɵɵinject(MatSnackBar, 12), ɵɵinject(MAT_SNACK_BAR_DEFAULT_OPTIONS)); };
    /** @nocollapse */
    MatSnackBar.ctorParameters = () => [
        { type: Overlay },
        { type: LiveAnnouncer },
        { type: Injector },
        { type: BreakpointObserver },
        { type: MatSnackBar, decorators: [{ type: Optional }, { type: SkipSelf }] },
        { type: MatSnackBarConfig, decorators: [{ type: Inject, args: [MAT_SNACK_BAR_DEFAULT_OPTIONS,] }] }
    ];
    /** @nocollapse */ MatSnackBar.ɵprov = ɵɵdefineInjectable({ factory: function MatSnackBar_Factory() { return new MatSnackBar(ɵɵinject(Overlay), ɵɵinject(LiveAnnouncer), ɵɵinject(INJECTOR), ɵɵinject(BreakpointObserver), ɵɵinject(MatSnackBar, 12), ɵɵinject(MAT_SNACK_BAR_DEFAULT_OPTIONS)); }, token: MatSnackBar, providedIn: MatSnackBarModule });

    const EXPERIMENTS_FEATURE_KEY = 'experiments';

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    const getExperimentsState = createFeatureSelector(EXPERIMENTS_FEATURE_KEY);
    const getDataState = createSelector(getExperimentsState, (state) => {
        return state.data;
    });
    /**
     * Returns Observable that emits an experiment.
     */
    const getExperiment = createSelector(getDataState, (state, props) => {
        const { experimentId } = props;
        return state.experimentMap[experimentId] || null;
    });

    const ALERT_FEATURE_KEY = 'alerts';

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    const selectAlertState = createFeatureSelector(ALERT_FEATURE_KEY);
    const getLatestAlert = createSelector(selectAlertState, (state) => {
        return state.latestAlert;
    });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/feature_flag/store/feature_flag_types.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /** @type {?} */
    const FEAUTURE_FLAG_FEATURE_KEY = 'feature';

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/feature_flag/store/feature_flag_selectors.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const selectFeatureFlagState = createFeatureSelector(FEAUTURE_FLAG_FEATURE_KEY);
    /** @type {?} */
    const getIsFeatureFlagsLoaded = createSelector(selectFeatureFlagState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return state.isFeatureFlagsLoaded;
    }));
    /** @type {?} */
    const getFeature = createSelector(selectFeatureFlagState, (/**
     * @param {?} state
     * @param {?} featureId
     * @return {?}
     */
    (state, featureId) => {
        return state.features[featureId] || null;
    }));
    /** @type {?} */
    const getEnabledExperimentalPlugins = createSelector(selectFeatureFlagState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return state.features.enabledExperimentalPlugins || [];
    }));
    /** @type {?} */
    const getIsInColab = createSelector(selectFeatureFlagState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return !!state.features.inColab;
    }));

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/webapp_data_source/tb_http_client.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @param {?} formData
     * @return {?}
     */
    function convertFormDataToObject(formData) {
        /** @type {?} */
        const result = (/** @type {?} */ ({}));
        for (const [key, value] of formData.entries()) {
            result[key] = (/** @type {?} */ (value));
        }
        return result;
    }
    class TBHttpClient {
        /**
         * @param {?} http
         * @param {?} store
         */
        constructor(http, store) {
            this.http = http;
            this.store = store;
        }
        /**
         * @template ResponseType
         * @param {?} path
         * @param {?=} options
         * @return {?}
         */
        get(path, options = {}) {
            return this.http.get(path, options);
        }
        /**
         * @template ResponseType
         * @param {?} path
         * @param {?} body
         * @param {?=} options
         * @return {?}
         */
        post(path, body, options = {}) {
            return this.store.select(getIsFeatureFlagsLoaded).pipe(filter((/**
             * @param {?} isLoaded
             * @return {?}
             */
            (isLoaded) => Boolean(isLoaded))), take(1), withLatestFrom(this.store.select(getIsInColab)), mergeMap((/**
             * @param {?} __0
             * @return {?}
             */
            ([, isInColab]) => {
                // Google-internal Colab does not support HTTP POST requests, so we fall
                // back to HTTP GET (even though public Colab supports POST)
                // See b/72932164.
                if (isInColab) {
                    return this.http.get(path, {
                        headers: options.headers,
                        params: convertFormDataToObject(body),
                    });
                }
                else {
                    return this.http.post(path, body, options);
                }
            })));
        }
        /**
         * @template ResponseType
         * @param {?} path
         * @param {?} body
         * @param {?=} options
         * @return {?}
         */
        put(path, body, options = {}) {
            return this.http.put(path, body, options);
        }
        /**
         * @template ResponseType
         * @param {?} path
         * @param {?=} options
         * @return {?}
         */
        delete(path, options = {}) {
            return this.http.delete(path, options);
        }
    }
    TBHttpClient.decorators = [
        { type: Injectable },
    ];
    /** @nocollapse */
    TBHttpClient.ctorParameters = () => [
        { type: HttpClient },
        { type: Store }
    ];
    /** @nocollapse */ TBHttpClient.ɵfac = function TBHttpClient_Factory(t) { return new (t || TBHttpClient)(ɵɵinject(HttpClient), ɵɵinject(Store)); };
    /** @nocollapse */ TBHttpClient.ɵprov = ɵɵdefineInjectable({ token: TBHttpClient, factory: TBHttpClient.ɵfac });

    const METRICS_PLUGIN_ID = 'timeseries';
    var PluginType;
    (function (PluginType) {
        PluginType["SCALARS"] = "scalars";
        PluginType["HISTOGRAMS"] = "histograms";
        PluginType["IMAGES"] = "images";
    })(PluginType || (PluginType = {}));
    function isPluginType(text) {
        return (text === PluginType.SCALARS ||
            text === PluginType.HISTOGRAMS ||
            text === PluginType.IMAGES);
    }
    const sampledPluginTypes = [PluginType.IMAGES];
    function isSampledPlugin(plugin) {
        return sampledPluginTypes.includes(plugin);
    }
    const singleRunPluginTypes = [PluginType.HISTOGRAMS, PluginType.IMAGES];
    function isSingleRunPlugin(plugin) {
        return singleRunPluginTypes.includes(plugin);
    }
    function isSingleRunTimeSeriesRequest(request) {
        return isSingleRunPlugin(request.plugin);
    }
    class MetricsDataSource {
    }
    function isFailedTimeSeriesResponse(response) {
        return response.hasOwnProperty('error');
    }

    /** @type {?} */
    const HTTP_PATH_PREFIX = 'data/plugin/timeseries';
    /**
     * @param {?} runId
     * @return {?}
     */
    function parseRunId(runId) {
        /** @type {?} */
        const slashIndex = runId.indexOf('/');
        return {
            run: runId.substring(slashIndex + 1),
            experimentId: runId.substring(0, slashIndex),
        };
    }
    /**
     * @param {?} run
     * @param {?} experimentId
     * @return {?}
     */
    function runToRunId(run, experimentId) {
        return `${experimentId}/${run}`;
    }
    /**
     * @param {?} backendResponse
     * @param {?} experimentId
     * @return {?}
     */
    function buildFrontendTimeSeriesResponse(backendResponse, experimentId) {
        const { runToSeries, run } = backendResponse, responseRest = __rest(backendResponse, ["runToSeries", "run"]);
        /** @type {?} */
        const response = (/** @type {?} */ (Object.assign({}, responseRest)));
        if (runToSeries) {
            response.runToSeries = buildRunIdKeyedObject(runToSeries, experimentId);
        }
        if (run) {
            response.runId = runToRunId(run, experimentId);
        }
        return response;
    }
    /**
     * @template T
     * @param {?} backendObject
     * @param {?} experimentId
     * @return {?}
     */
    function buildRunIdKeyedObject(backendObject, experimentId) {
        /** @type {?} */
        const frontendObject = (/** @type {?} */ ({}));
        for (const run in backendObject) {
            if (backendObject.hasOwnProperty(run)) {
                /** @type {?} */
                const runId = runToRunId(run, experimentId);
                frontendObject[runId] = backendObject[run];
            }
        }
        return (/** @type {?} */ (frontendObject));
    }
    /**
     * @param {?} backendTagMetadata
     * @param {?} experimentId
     * @return {?}
     */
    function buildFrontendTagMetadata(backendTagMetadata, experimentId) {
        /** @type {?} */
        const tagMetadata = (/** @type {?} */ ({}));
        for (const pluginType of Object.keys(backendTagMetadata)) {
            /** @type {?} */
            const plugin = (/** @type {?} */ (pluginType));
            if (isSampledPlugin(plugin)) {
                const _a = backendTagMetadata[plugin], { tagRunSampledInfo } = _a, rest = __rest(_a, ["tagRunSampledInfo"]);
                /** @type {?} */
                const frontendTagRunSampledInfo = (/** @type {?} */ ({}));
                for (const tag in tagRunSampledInfo) {
                    if (tagRunSampledInfo.hasOwnProperty(tag)) {
                        frontendTagRunSampledInfo[tag] = buildRunIdKeyedObject(tagRunSampledInfo[tag], experimentId);
                    }
                }
                tagMetadata[plugin] = Object.assign(Object.assign({}, rest), { tagRunSampledInfo: frontendTagRunSampledInfo });
            }
            else {
                const _b = backendTagMetadata[plugin], { runTagInfo } = _b, rest = __rest(_b, ["runTagInfo"]);
                tagMetadata[plugin] = Object.assign(Object.assign({}, rest), { runTagInfo: buildRunIdKeyedObject(runTagInfo, experimentId) });
            }
        }
        return tagMetadata;
    }
    /**
     * @param {?} results
     * @return {?}
     */
    function buildCombinedTagMetadata(results) {
        // Collate results from different experiments.
        /** @type {?} */
        const tagMetadata = (/** @type {?} */ ({}));
        for (const experimentTagMetadata of results) {
            for (const plugin of Object.values(PluginType)) {
                if (isSampledPlugin(plugin)) {
                    tagMetadata[plugin] = tagMetadata[plugin] || {
                        tagDescriptions: {},
                        tagRunSampledInfo: {},
                    };
                    const { tagDescriptions, tagRunSampledInfo } = experimentTagMetadata[plugin];
                    tagMetadata[plugin].tagDescriptions = Object.assign(Object.assign({}, tagMetadata[plugin].tagDescriptions), tagDescriptions);
                    /** @type {?} */
                    const combinedTagRunSampledInfo = tagMetadata[plugin].tagRunSampledInfo;
                    for (const tag of Object.keys(tagRunSampledInfo)) {
                        combinedTagRunSampledInfo[tag] = combinedTagRunSampledInfo[tag] || {};
                        for (const runId of Object.keys(tagRunSampledInfo[tag])) {
                            combinedTagRunSampledInfo[tag][runId] =
                                tagRunSampledInfo[tag][runId];
                        }
                    }
                }
                else {
                    tagMetadata[plugin] = tagMetadata[plugin] || {
                        tagDescriptions: {},
                        runTagInfo: {},
                    };
                    const { tagDescriptions, runTagInfo } = experimentTagMetadata[plugin];
                    tagMetadata[plugin].tagDescriptions = Object.assign(Object.assign({}, tagMetadata[plugin].tagDescriptions), tagDescriptions);
                    tagMetadata[plugin].runTagInfo = Object.assign(Object.assign({}, tagMetadata[plugin].runTagInfo), runTagInfo);
                }
            }
        }
        return tagMetadata;
    }
    /**
     * An implementation of MetricsDataSource that treats RunIds as identifiers
     * containing run name and experimentId.
     */
    class TBMetricsDataSource {
        /**
         * @param {?} http
         */
        constructor(http) {
            this.http = http;
        }
        /**
         * @param {?} experimentIds
         * @return {?}
         */
        fetchTagMetadata(experimentIds) {
            /** @type {?} */
            const fetches = experimentIds.map((/**
             * @param {?} experimentId
             * @return {?}
             */
            (experimentId) => {
                /** @type {?} */
                const url = `/experiment/${experimentId}/${HTTP_PATH_PREFIX}/tags`;
                return this.http.get(url).pipe(map((/**
                 * @param {?} tagMetadata
                 * @return {?}
                 */
                (tagMetadata) => {
                    return buildFrontendTagMetadata(tagMetadata, experimentId);
                })));
            }));
            return forkJoin(fetches).pipe(map((/**
             * @param {?} results
             * @return {?}
             */
            (results) => buildCombinedTagMetadata(results))));
        }
        /**
         * TODO(psybuzz): we only request 1 at a time, consider updating the backend to
         * take a BackendTimeSeriesRequest instead of an array.
         * @param {?} requests
         * @return {?}
         */
        fetchTimeSeries(requests) {
            /** @type {?} */
            const fetches = requests.map((/**
             * @param {?} request
             * @return {?}
             */
            (request) => {
                // One single-run request.
                if (isSingleRunPlugin(request.plugin)) {
                    const _a = (/** @type {?} */ (request)), { runId } = _a, requestRest = __rest(_a, ["runId"]);
                    const { run, experimentId } = parseRunId(runId);
                    /** @type {?} */
                    const backendRequest = Object.assign(Object.assign({}, requestRest), { run });
                    return this.fetchTimeSeriesBackendRequest(backendRequest, experimentId).pipe(map((/**
                     * @param {?} __0
                     * @return {?}
                     */
                    ({ response, experimentId }) => {
                        return buildFrontendTimeSeriesResponse(response, experimentId);
                    })));
                }
                // One multi-run request generates many responses with different
                // 'runToSeries', 'error' fields. Combine them into one.
                const _b = (/** @type {?} */ (request)), { experimentIds } = _b, requestRest = __rest(_b, ["experimentIds"]);
                /** @type {?} */
                const perExperimentRequests = experimentIds.map((/**
                 * @param {?} experimentId
                 * @return {?}
                 */
                (experimentId) => {
                    return this.fetchTimeSeriesBackendRequest(requestRest, experimentId);
                }));
                return forkJoin(perExperimentRequests).pipe(map((/**
                 * @param {?} perExperimentResults
                 * @return {?}
                 */
                (perExperimentResults) => {
                    const _a = perExperimentResults[0].response, responseRest = __rest(_a, ["runToSeries", "error"]);
                    /** @type {?} */
                    const combinedResponse = (/** @type {?} */ (responseRest));
                    for (const { response, experimentId } of perExperimentResults) {
                        /** @type {?} */
                        const frontendResponse = buildFrontendTimeSeriesResponse(response, experimentId);
                        if (combinedResponse.error) {
                            continue;
                        }
                        const { runToSeries, error } = frontendResponse;
                        if (error) {
                            combinedResponse.error = error;
                            combinedResponse.runToSeries = undefined;
                        }
                        else {
                            combinedResponse.runToSeries = combinedResponse.runToSeries || {};
                            for (const run of Object.keys((/** @type {?} */ (runToSeries)))) {
                                combinedResponse.runToSeries[run] = (/** @type {?} */ (runToSeries))[run];
                            }
                        }
                    }
                    return combinedResponse;
                })));
            }));
            return forkJoin(fetches);
        }
        /**
         * @private
         * @param {?} backendRequest
         * @param {?} experimentId
         * @return {?}
         */
        fetchTimeSeriesBackendRequest(backendRequest, experimentId) {
            /** @type {?} */
            const body = new FormData();
            body.append('requests', JSON.stringify([backendRequest]));
            return this.http
                .post(`/experiment/${experimentId}/${HTTP_PATH_PREFIX}/timeSeries`, body)
                .pipe(map((/**
             * @param {?} responses
             * @return {?}
             */
            (responses) => {
                return { response: responses[0], experimentId };
            })));
        }
        /**
         * @param {?} imageId
         * @return {?}
         */
        imageUrl(imageId) {
            return `${HTTP_PATH_PREFIX}/imageData?imageId=${imageId}`;
        }
    }
    TBMetricsDataSource.decorators = [
        { type: Injectable },
    ];
    /** @nocollapse */
    TBMetricsDataSource.ctorParameters = () => [
        { type: TBHttpClient }
    ];
    /** @nocollapse */ TBMetricsDataSource.ɵfac = function TBMetricsDataSource_Factory(t) { return new (t || TBMetricsDataSource)(ɵɵinject(TBHttpClient)); };
    /** @nocollapse */ TBMetricsDataSource.ɵprov = ɵɵdefineInjectable({ token: TBMetricsDataSource, factory: TBMetricsDataSource.ɵfac });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/webapp_data_source/tb_http_client_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class TBHttpClientModule {
    }
    TBHttpClientModule.decorators = [
        { type: NgModule, args: [{
                    imports: [HttpClientModule],
                    providers: [TBHttpClient],
                },] },
    ];
    /** @nocollapse */ TBHttpClientModule.ɵmod = ɵɵdefineNgModule({ type: TBHttpClientModule });
    /** @nocollapse */ TBHttpClientModule.ɵinj = ɵɵdefineInjector({ factory: function TBHttpClientModule_Factory(t) { return new (t || TBHttpClientModule)(); }, providers: [TBHttpClient], imports: [[HttpClientModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(TBHttpClientModule, { imports: [HttpClientModule] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/data_source/metrics_data_source_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class MetricsDataSourceModule {
    }
    MetricsDataSourceModule.decorators = [
        { type: NgModule, args: [{
                    imports: [TBHttpClientModule],
                    providers: [{ provide: MetricsDataSource, useClass: TBMetricsDataSource }],
                },] },
    ];
    /** @nocollapse */ MetricsDataSourceModule.ɵmod = ɵɵdefineNgModule({ type: MetricsDataSourceModule });
    /** @nocollapse */ MetricsDataSourceModule.ɵinj = ɵɵdefineInjector({ factory: function MetricsDataSourceModule_Factory(t) { return new (t || MetricsDataSourceModule)(); }, providers: [{ provide: MetricsDataSource, useClass: TBMetricsDataSource }], imports: [[TBHttpClientModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(MetricsDataSourceModule, { imports: [TBHttpClientModule] }); })();

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * Returns the loadable information for a specific tag, containing its series
     * data and load state. Returns `null` when the requested tag has no initial
     * loadable in `timeSeriesData`.
     */
    function getTimeSeriesLoadable(timeSeriesData, plugin, tag, sample) {
        const pluginData = timeSeriesData[plugin];
        if (!pluginData.hasOwnProperty(tag)) {
            return null;
        }
        if (isSampledPlugin(plugin)) {
            if (!timeSeriesData[plugin][tag].hasOwnProperty(sample)) {
                return null;
            }
            return timeSeriesData[plugin][tag][sample];
        }
        return timeSeriesData[plugin][tag];
    }
    /**
     * Create a new plugin data with new references to a new time series loadable.
     * The return object is a shallow clone, so consumers must clone fields as
     * needed.
     */
    function createPluginDataWithLoadable(timeSeriesData, plugin, tag, sample) {
        if (isSampledPlugin(plugin)) {
            const pluginData = Object.assign({}, timeSeriesData[plugin]);
            const tagData = createSampledTagDataWithLoadable(pluginData, tag, sample);
            pluginData[tag] = tagData;
            return pluginData;
        }
        const pluginData = Object.assign({}, timeSeriesData[plugin]);
        const hasTag = pluginData.hasOwnProperty(tag);
        pluginData[tag] = hasTag
            ? Object.assign({}, pluginData[tag]) : buildTimeSeriesLoadable();
        return pluginData;
    }
    function createSampledTagDataWithLoadable(pluginData, tag, sample) {
        const hasTag = pluginData.hasOwnProperty(tag);
        const tagData = hasTag ? Object.assign({}, pluginData[tag]) : {};
        const hasSample = tagData.hasOwnProperty(sample);
        tagData[sample] = hasSample
            ? Object.assign({}, tagData[sample]) : buildTimeSeriesLoadable();
        return tagData;
    }
    function buildTimeSeriesLoadable() {
        return {
            runToSeries: {},
            runToLoadState: {},
        };
    }
    /**
     * Note: do not rely on the implementation details of these ID generators below.
     * Clients should operate on `CardId`s, whose type may be open to change.
     */
    function getCardId(cardMetadata) {
        return JSON.stringify(cardMetadata);
    }
    function getPinnedCardId(baseCardId) {
        return JSON.stringify({ baseCardId });
    }
    /**
     * Creates a RunToLoadState with a specific load state for all specified runs.
     */
    function createRunToLoadState(loadState, runs, prevRunToLoadState) {
        const runToLoadState = Object.assign({}, prevRunToLoadState);
        for (const run of runs) {
            runToLoadState[run] = loadState;
        }
        return runToLoadState;
    }
    function getRunIds(tagMetadata, plugin, tag, sample) {
        if (isSampledPlugin(plugin)) {
            const tagRunSampledInfo = tagMetadata[plugin].tagRunSampledInfo;
            if (!tagRunSampledInfo.hasOwnProperty(tag)) {
                return [];
            }
            const runIds = Object.keys(tagRunSampledInfo[tag]);
            return runIds.filter((runId) => {
                return sample < tagRunSampledInfo[tag][runId].maxSamplesPerStep;
            });
        }
        const tagToRunIds = tagMetadata[plugin].tagToRuns;
        return tagToRunIds.hasOwnProperty(tag) ? tagToRunIds[tag] : [];
    }
    /**
     * Returns whether the CardMetadata exactly matches the pinned card from
     * storage.
     */
    function cardMatchesCardUniqueInfo(cardMetadata, cardUniqueInfo) {
        const noRunId = !cardMetadata.runId && !cardUniqueInfo.runId;
        return (cardMetadata.plugin === cardUniqueInfo.plugin &&
            cardMetadata.tag === cardUniqueInfo.tag &&
            cardMetadata.sample === cardUniqueInfo.sample &&
            (cardMetadata.runId === cardUniqueInfo.runId || noRunId));
    }
    /**
     * Attempts to resolve the imported pins against the list of non-pinned cards
     * provided. Returns the resulting state.
     *
     * Note: this assumes input has already been sanitized and validated. Untrusted
     * data from URLs must be cleaned before being passed to the store.
     */
    function buildOrReturnStateWithUnresolvedImportedPins(unresolvedImportedPinnedCards, nonPinnedCards, cardMetadataMap, cardToPinnedCopy, pinnedCardToOriginal, cardStepIndexMap) {
        const unresolvedPinSet = new Set(unresolvedImportedPinnedCards);
        const nonPinnedCardsWithMatch = [];
        for (const unresolvedPin of unresolvedImportedPinnedCards) {
            for (const nonPinnedCardId of nonPinnedCards) {
                const cardMetadata = cardMetadataMap[nonPinnedCardId];
                if (cardMatchesCardUniqueInfo(cardMetadata, unresolvedPin)) {
                    nonPinnedCardsWithMatch.push(nonPinnedCardId);
                    unresolvedPinSet.delete(unresolvedPin);
                    break;
                }
            }
        }
        if (!nonPinnedCardsWithMatch.length) {
            return {
                unresolvedImportedPinnedCards,
                cardMetadataMap,
                cardToPinnedCopy,
                pinnedCardToOriginal,
                cardStepIndex: cardStepIndexMap,
            };
        }
        let stateWithResolvedPins = {
            cardToPinnedCopy,
            pinnedCardToOriginal,
            cardStepIndex: cardStepIndexMap,
            cardMetadataMap,
        };
        for (const cardToPin of nonPinnedCardsWithMatch) {
            stateWithResolvedPins = buildOrReturnStateWithPinnedCopy(cardToPin, stateWithResolvedPins.cardToPinnedCopy, stateWithResolvedPins.pinnedCardToOriginal, stateWithResolvedPins.cardStepIndex, stateWithResolvedPins.cardMetadataMap);
        }
        return Object.assign(Object.assign({}, stateWithResolvedPins), { unresolvedImportedPinnedCards: [...unresolvedPinSet] });
    }
    /**
     * Return the state produced by creating a new pinned copy of the provided card.
     * May throw if the card provided has no metadata.
     */
    function buildOrReturnStateWithPinnedCopy(cardId, cardToPinnedCopy, pinnedCardToOriginal, cardStepIndexMap, cardMetadataMap) {
        // No-op if the card already has a pinned copy.
        if (cardToPinnedCopy.has(cardId)) {
            return {
                cardToPinnedCopy,
                pinnedCardToOriginal,
                cardStepIndex: cardStepIndexMap,
                cardMetadataMap,
            };
        }
        const nextCardToPinnedCopy = new Map(cardToPinnedCopy);
        const nextPinnedCardToOriginal = new Map(pinnedCardToOriginal);
        const nextCardStepIndexMap = Object.assign({}, cardStepIndexMap);
        const nextCardMetadataMap = Object.assign({}, cardMetadataMap);
        // Create a pinned copy. Copies step index from the original card.
        const pinnedCardId = getPinnedCardId(cardId);
        nextCardToPinnedCopy.set(cardId, pinnedCardId);
        nextPinnedCardToOriginal.set(pinnedCardId, cardId);
        if (cardStepIndexMap.hasOwnProperty(cardId)) {
            nextCardStepIndexMap[pinnedCardId] = cardStepIndexMap[cardId];
        }
        const metadata = cardMetadataMap[cardId];
        if (!metadata) {
            throw new Error('Cannot pin a card without metadata');
        }
        nextCardMetadataMap[pinnedCardId] = metadata;
        return {
            cardToPinnedCopy: nextCardToPinnedCopy,
            pinnedCardToOriginal: nextPinnedCardToOriginal,
            cardStepIndex: nextCardStepIndexMap,
            cardMetadataMap: nextCardMetadataMap,
        };
    }
    /**
     * The maximum number of pins we allow the user to create. This is intentionally
     * finite at the moment to mitigate super long URL lengths, until there is more
     * durable value storage for pins.
     * https://github.com/tensorflow/tensorboard/issues/4242
     */
    const util = {
        MAX_PIN_COUNT: 10,
    };
    function canCreateNewPins(state) {
        const pinCountInURL = state.pinnedCardToOriginal.size +
            state.unresolvedImportedPinnedCards.length;
        return pinCountInURL < util.MAX_PIN_COUNT;
    }

    const METRICS_FEATURE_KEY = 'metrics';

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    const selectMetricsState = createFeatureSelector(METRICS_FEATURE_KEY);
    const getMetricsTagMetadataLoaded = createSelector(selectMetricsState, (state) => state.tagMetadataLoaded);
    const getMetricsTagMetadata = createSelector(selectMetricsState, (state) => {
        return state.tagMetadata;
    });
    /**
     * Cards
     */
    const getCardIds = createSelector(selectMetricsState, (state) => {
        return state.cardList;
    });
    const getCardLoadState = createSelector(selectMetricsState, (state, cardId) => {
        if (!state.cardMetadataMap.hasOwnProperty(cardId)) {
            return DataLoadState.NOT_LOADED;
        }
        const { plugin, tag, runId, sample } = state.cardMetadataMap[cardId];
        const loadable = getTimeSeriesLoadable(state.timeSeriesData, plugin, tag, sample);
        if (!loadable) {
            return DataLoadState.NOT_LOADED;
        }
        const runToLoadState = loadable.runToLoadState;
        if (runId) {
            return runToLoadState.hasOwnProperty(runId)
                ? runToLoadState[runId]
                : DataLoadState.NOT_LOADED;
        }
        const runIds = getRunIds(state.tagMetadata, plugin, tag, sample);
        if (!runIds.length) {
            throw new Error('Cannot load a card whose tag has no runs');
        }
        if (runIds.every((runId) => runToLoadState[runId] === DataLoadState.LOADED)) {
            return DataLoadState.LOADED;
        }
        return runIds.some((runId) => runToLoadState[runId] === DataLoadState.LOADING)
            ? DataLoadState.LOADING
            : DataLoadState.NOT_LOADED;
    });
    const getCardTimeSeries = createSelector(selectMetricsState, (state, cardId) => {
        if (!state.cardMetadataMap.hasOwnProperty(cardId)) {
            return null;
        }
        const { plugin, tag, sample } = state.cardMetadataMap[cardId];
        const loadable = getTimeSeriesLoadable(state.timeSeriesData, plugin, tag, sample);
        return loadable ? loadable.runToSeries : null;
    });
    const getCardMetadataMap = createSelector(selectMetricsState, (state) => {
        return state.cardMetadataMap;
    });
    const getCardMetadata = createSelector(getCardMetadataMap, (metadataMap, cardId) => {
        if (!metadataMap.hasOwnProperty(cardId)) {
            return null;
        }
        return metadataMap[cardId];
    });
    // A cheap identity selector to skip recomputing selectors.
    const getVisibleCardIdSet = createSelector(selectMetricsState, (state) => {
        return state.visibleCards;
    });
    const getVisibleCardIds = createSelector(getVisibleCardIdSet, (cardIdSet) => {
        return [...cardIdSet];
    });
    /**
     * Returns current list of card data whose metadata is loaded.
     */
    const getNonEmptyCardIdsWithMetadata = createSelector(getCardIds, getCardMetadataMap, (cardIds, metadataMap) => {
        return cardIds
            .filter((cardId) => {
            return metadataMap.hasOwnProperty(cardId);
        })
            .map((cardId) => {
            return Object.assign({ cardId }, metadataMap[cardId]);
        });
    });
    /**
     * The index into the step values array for a card's UI. This may be greater
     * than the number of step values available, if time series data is not loaded.
     */
    const getCardStepIndex = createSelector(selectMetricsState, (state, cardId) => {
        if (!state.cardStepIndex.hasOwnProperty(cardId)) {
            return null;
        }
        return state.cardStepIndex[cardId];
    });
    const getCardToPinnedCopy = createSelector(selectMetricsState, (state) => {
        return state.cardToPinnedCopy;
    });
    const getPinnedCardToOriginal = createSelector(selectMetricsState, (state) => {
        return state.pinnedCardToOriginal;
    });
    /**
     * Returns an ordered list of the cards in the pinned location.
     */
    const getPinnedCardsWithMetadata = createSelector(getCardToPinnedCopy, getCardMetadataMap, (cardToPinnedCopy, metadataMap) => {
        return [...cardToPinnedCopy.values()]
            .filter((cardId) => {
            return metadataMap.hasOwnProperty(cardId);
        })
            .map((cardId) => {
            return Object.assign({ cardId }, metadataMap[cardId]);
        });
    });
    /**
     * Returns true if a card is pinned or a separate card exists that is a pinned
     * copy of this card. Defaults to false if the card is unknown.
     */
    const getCardPinnedState = createSelector(getCardToPinnedCopy, getPinnedCardToOriginal, (cardToPinnedCopy, pinnedCardToOriginal, cardId) => {
        return cardToPinnedCopy.has(cardId) || pinnedCardToOriginal.has(cardId);
    });
    const getUnresolvedImportedPinnedCards = createSelector(selectMetricsState, (state) => {
        return state.unresolvedImportedPinnedCards;
    });
    /**
     * Whether the UI is allowed to pin more cards. This may be limited if the URL
     * contains too many pins already.
     */
    const getCanCreateNewPins = createSelector(selectMetricsState, (state) => {
        return canCreateNewPins(state);
    });
    /**
     * Settings.
     */
    const getMetricsTooltipSort = createSelector(selectMetricsState, (state) => state.settings.tooltipSort);
    const getMetricsIgnoreOutliers = createSelector(selectMetricsState, (state) => state.settings.ignoreOutliers);
    const getMetricsXAxisType = createSelector(selectMetricsState, (state) => state.settings.xAxisType);
    const getMetricsHistogramMode = createSelector(selectMetricsState, (state) => state.settings.histogramMode);
    const getMetricsScalarSmoothing = createSelector(selectMetricsState, (state) => state.settings.scalarSmoothing);
    const getMetricsImageBrightnessInMilli = createSelector(selectMetricsState, (state) => state.settings.imageBrightnessInMilli);
    const getMetricsImageContrastInMilli = createSelector(selectMetricsState, (state) => state.settings.imageContrastInMilli);
    const getMetricsImageShowActualSize = createSelector(selectMetricsState, (state) => state.settings.imageShowActualSize);
    const getMetricsTagFilter = createSelector(selectMetricsState, (state) => state.tagFilter);
    const getMetricsTagGroupExpansionState = createSelector(selectMetricsState, (state, tagGroup) => {
        return Boolean(state.tagGroupExpanded.get(tagGroup));
    });

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * @fileoverview Types defined in tensorboard/plugins/hparams/api.proto and
     * tensorboard/plugins/hparams/http_api.md.
     */
    var DatasetType;
    (function (DatasetType) {
        DatasetType["DATASET_UNKNOWN"] = "DATASET_UNKNOWN";
        DatasetType["DATASET_TRAINING"] = "DATASET_TRAINING";
        DatasetType["DATASET_VALIDATION"] = "DATASET_VALIDATION";
    })(DatasetType || (DatasetType = {}));
    var RunStatus;
    (function (RunStatus) {
        RunStatus["STATUS_UNKNOWN"] = "STATUS_UNKNOWN";
        RunStatus["STATUS_SUCCESS"] = "STATUS_SUCCESS";
        RunStatus["STATUS_FAILURE"] = "STATUS_FAILURE";
        RunStatus["STATUS_RUNNING"] = "STATUS_RUNNING";
    })(RunStatus || (RunStatus = {}));
    var BackendHparamsValueType;
    (function (BackendHparamsValueType) {
        BackendHparamsValueType["DATA_TYPE_UNSET"] = "DATA_TYPE_UNSET";
        BackendHparamsValueType["DATA_TYPE_STRING"] = "DATA_TYPE_STRING";
        BackendHparamsValueType["DATA_TYPE_BOOL"] = "DATA_TYPE_BOOL";
        BackendHparamsValueType["DATA_TYPE_FLOAT64"] = "DATA_TYPE_FLOAT64";
    })(BackendHparamsValueType || (BackendHparamsValueType = {}));

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/runs/data_source/runs_data_source_types.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @enum {number} */
    const DomainType = {
        DISCRETE: 0,
        INTERVAL: 1,
    };
    DomainType[DomainType.DISCRETE] = 'DISCRETE';
    DomainType[DomainType.INTERVAL] = 'INTERVAL';
    /**
     * @abstract
     */
    class RunsDataSource {
    }
    RunsDataSource.decorators = [
        { type: Injectable, args: [{ providedIn: 'root' },] },
    ];
    /** @nocollapse */ RunsDataSource.ɵfac = function RunsDataSource_Factory(t) { return new (t || RunsDataSource)(); };
    /** @nocollapse */ RunsDataSource.ɵprov = ɵɵdefineInjectable({ token: RunsDataSource, factory: RunsDataSource.ɵfac, providedIn: 'root' });

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * Combines hparams and metrics spec from different experiments.
     *
     * Different experiments can be testing hypothesis on different sets of hparams
     * and metrics. This utility combines the specs when possible and throws
     * validation errors when key (`name` for hparams and `tag` for metrics) collide
     * with different specs.
     */
    function combineHparamAndMetricSpecs(...specs) {
        const hparams = new Map();
        const hparamDisplayNames = new Map();
        const metrics = new Map();
        const metricDisplayNames = new Map();
        const validationError = [];
        for (const spec of specs) {
            for (const hparamSpec of spec.hparams) {
                if (!hparamDisplayNames.has(hparamSpec.name)) {
                    hparamDisplayNames.set(hparamSpec.name, new Set());
                }
                hparamDisplayNames.get(hparamSpec.name).add(hparamSpec.displayName);
                if (!hparams.has(hparamSpec.name)) {
                    hparams.set(hparamSpec.name, Object.assign({}, hparamSpec));
                }
                else {
                    const specA = hparams.get(hparamSpec.name);
                    const specB = hparamSpec;
                    if (specA.type !== specB.type) {
                        validationError.push(`Hparam, ${specB.name}, types have to match. Got: ` +
                            `${specA.type} vs. ${specB.type}`);
                    }
                    if (specA.domain.type === DomainType.INTERVAL &&
                        specB.domain.type === DomainType.INTERVAL) {
                        if (specA.domain.minValue !== specB.domain.minValue ||
                            specA.domain.maxValue !== specB.domain.maxValue) {
                            validationError.push(`Hparam, ${specB.name}, domains have to match. ` +
                                `Got: ${specA.domain} vs. ${specB.domain}`);
                        }
                    }
                    else if (specA.domain.type === DomainType.DISCRETE &&
                        specB.domain.type === DomainType.DISCRETE) {
                        const valueSet = new Set([
                            ...specA.domain.values,
                            ...specB.domain.values,
                        ]);
                        if (specA.domain.values.length !== specB.domain.values.length ||
                            specA.domain.values.length !== valueSet.size) {
                            validationError.push(`Hparam, ${specB.name}, domains have to match. ` +
                                `Got: ${specA.domain} vs. ${specB.domain}`);
                        }
                    }
                    else {
                        validationError.push(`Hparam, ${specB.name}, domains have to match. ` +
                            `Got: ${specA.domain} vs. ${specB.domain}`);
                    }
                }
            }
            for (const metricSpec of spec.metrics) {
                if (!metricDisplayNames.has(metricSpec.tag)) {
                    metricDisplayNames.set(metricSpec.tag, new Set());
                }
                metricDisplayNames.get(metricSpec.tag).add(metricSpec.displayName);
                if (!metrics.has(metricSpec.tag)) {
                    metrics.set(metricSpec.tag, Object.assign({}, metricSpec));
                }
                else {
                    const specA = metrics.get(metricSpec.tag);
                    const specB = metricSpec;
                    if (specA.datasetType !== specB.datasetType) {
                        validationError.push(`Metric, ${specB.tag}, datasetTypes have to match. Got: ` +
                            `${specA.datasetType} vs. ${specB.datasetType}`);
                    }
                }
            }
        }
        if (validationError.length) {
            // TODO(b/157733179): handle the error gracefully on the view side.
            throw new Error(`Validation error:
${validationError.join('\n')}`);
        }
        return {
            hparams: [...hparams].map(([hparamName, spec]) => {
                return Object.assign(Object.assign({}, spec), { displayName: [...hparamDisplayNames.get(hparamName)].join(' or ') });
            }),
            metrics: [...metrics].map(([metricTag, spec]) => {
                return Object.assign(Object.assign({}, spec), { displayName: [...metricDisplayNames.get(metricTag)].join(' or ') });
            }),
        };
    }

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * Key used to namespace the experiments reducer.
     */
    const RUNS_FEATURE_KEY = 'runs';
    /**
     * When runs selector was never modified by user and there are less than or
     * equal to MAX_NUM_RUNS_TO_ENABLE_BY_DEFAULT in an experiment, we default
     * select all runs.
     */
    const MAX_NUM_RUNS_TO_ENABLE_BY_DEFAULT = 40;

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    function serializeExperimentIds(experimentIds) {
        return JSON.stringify(experimentIds.slice().sort());
    }

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    const getRunsState = createFeatureSelector(RUNS_FEATURE_KEY);
    const getDataState$1 = createSelector(getRunsState, (state) => {
        return state.data;
    });
    /**
     * Returns Observable that emits run object.
     */
    const getExperimentIdForRunId = createSelector(getDataState$1, (state, props) => {
        var _a;
        return (_a = state.runIdToExpId[props.runId]) !== null && _a !== void 0 ? _a : null;
    });
    /**
     * Returns Observable that emits run object.
     */
    const getRun = createSelector(getDataState$1, (state, props) => {
        var _a;
        return (_a = state.runMetadata[props.runId]) !== null && _a !== void 0 ? _a : null;
    });
    /**
     * Returns Observable that emits runs list for an experiment.
     */
    const getRuns = createSelector(getDataState$1, (state, props) => {
        const runIds = state.runIds[props.experimentId] || [];
        return runIds
            .filter((id) => Boolean(state.runMetadata[id]))
            .map((id) => state.runMetadata[id]);
    });
    /**
     * Returns Observable that emits load state of the runs list.
     */
    const getRunsLoadState = createSelector(getDataState$1, (state, props) => {
        return (state.runsLoadState[props.experimentId] || {
            lastLoadedTimeInMs: null,
            state: DataLoadState.NOT_LOADED,
        });
    });
    /**
     * `selectHparamAndMetricSpec` is meant for performance optimization that
     * leverages the memoization.
     */
    const selectHparamAndMetricSpec = createSelector(getDataState$1, (state) => {
        return state.hparamAndMetricSpec;
    });
    /**
     * Returns Observable that emits hparams and metrics specs of experiments.
     */
    const getExperimentsHparamsAndMetrics = createSelector(selectHparamAndMetricSpec, (record, props) => {
        return combineHparamAndMetricSpecs(...props.experimentIds.map((eid) => record[eid]).filter(Boolean));
    });
    /**
     * Returns Observable that emits selection state of runs. If the runs for the
     * current route are desired, please see ui_selectors.ts's
     * getCurrentRouteRunSelection instead.
     */
    const getRunSelectionMap = createSelector(getDataState$1, (dataState, props) => {
        const stateKey = serializeExperimentIds(props.experimentIds);
        return dataState.selectionState.get(stateKey) || new Map();
    });
    const getUiState = createSelector(getRunsState, (state) => {
        return state.ui;
    });
    /**
     * Returns Observable that emits pagination option on the run selector.
     */
    const getRunSelectorPaginationOption = createSelector(getUiState, (state) => {
        return state.paginationOption;
    });
    /**
     * Returns Observable that emits regex filter on the run selector.
     */
    const getRunSelectorRegexFilter = createSelector(getUiState, (state) => {
        return state.regexFilter;
    });
    /**
     * Returns Observable that emits sort options on the run selector.
     */
    const getRunSelectorSort = createSelector(getUiState, (state) => {
        return state.sort;
    });
    /**
     * Returns Observable that emits map of run id to run color (hex).
     */
    const getRunColorMap = createSelector(getUiState, (state) => {
        const map = new Map([...state.defaultRunColor, ...state.runColorOverride]);
        const colorObject = {};
        map.forEach((value, key) => {
            colorObject[key] = value;
        });
        return colorObject;
    });
    // Cheap identity selectors to skip recomputing selectors.
    const getHparamDefaultFilter = createSelector(getUiState, (state) => {
        return state.hparamDefaultFilters;
    });
    const getHparamFilter = createSelector(getUiState, (state) => {
        return state.hparamFilters;
    });
    const getMetricDefaultFilter = createSelector(getUiState, (state) => {
        return state.metricDefaultFilters;
    });
    const getMetricFilter = createSelector(getUiState, (state) => {
        return state.metricFilters;
    });
    /**
     * Returns Observable that emits map of hparam name to filter values.
     */
    const getRunHparamFilterMap = createSelector(getHparamDefaultFilter, getHparamFilter, (defaultFilterMap, filterMap) => {
        return new Map([...defaultFilterMap, ...filterMap]);
    });
    /**
     * Returns Observable that emits map of metric tag to filter values.
     */
    const getRunMetricFilterMap = createSelector(getMetricDefaultFilter, getMetricFilter, (defaultFilterMap, filterMap) => {
        return new Map([...defaultFilterMap, ...filterMap]);
    });

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * Selects the run selection (runId to boolean) of current routeId.
     *
     * Note that emits null when current route is not about an experiment.
     */
    const getCurrentRouteRunSelection = createSelector((state) => {
        const experimentIds = getExperimentIdsFromRoute(state);
        if (experimentIds === null) {
            return null;
        }
        return getRunSelectionMap(state, { experimentIds });
    }, (runSelection) => runSelection);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * This regex escape character set is also used in lodash's `escapeRegExp`.
     */
    const REGEXP_ESCAPE_CHARS = /[\\^$.*+?()[\]{}|]/g;
    /**
     * Converts a string into a form that has been escaped for use as a literal
     * argument to a regular expression constructor.
     *
     * Takes a string V and escapes characters to produce a new string E, such that
     * new RegExp(E).test(V) === true.
     */
    function escapeForRegex(value) {
        // '$&' in a regex replacement indicates the last match.
        return value.replace(REGEXP_ESCAPE_CHARS, '\\$&');
    }
    /**
     * Processes text from left-to-right, splitting it into pieces based on the
     * regex. Each piece is either an unmatched or a matched substring.
     *
     * For example,
     * splitByRegex("a input1 b input2 c input3", /input\d/)
     * returns
     * [
     *    {index: 0,  matchesRegex: false, text: "a "},
     *    {index: 2,  matchesRegex: true,  text: "input1"},
     *    {index: 8,  matchesRegex: false, text: " b "},
     *    {index: 11, matchesRegex: true,  text: "input2"},
     *    {index: 17, matchesRegex: false, text: " c "},
     *    {index: 20, matchesRegex: true,  text: "input3"},
     * ]
     */
    function splitByRegex(text, regex) {
        // 'matchAll' requires a regex with the 'global' flag.
        if (!regex.flags.includes('g')) {
            regex = new RegExp(regex, regex.flags + 'g');
        }
        const result = [];
        // Index of the earliest unvisited character.
        let lastIndex = 0;
        for (const match of text.matchAll(regex)) {
            const index = match.index;
            const matchingText = match[0];
            // Add any text between the last match and this current one.
            if (index > lastIndex) {
                result.push({
                    index: lastIndex,
                    text: text.substring(lastIndex, index),
                    matchesRegex: false,
                });
            }
            result.push({
                index,
                text: matchingText,
                matchesRegex: true,
            });
            lastIndex = index + matchingText.length;
        }
        // Add the remaining text piece, if any.
        if (text.length > lastIndex) {
            result.push({
                index: lastIndex,
                text: text.substring(lastIndex, text.length),
                matchesRegex: false,
            });
        }
        return result;
    }
    // Based on
    // https://cs.chromium.org/chromium/src/third_party/devtools-frontend/src/front_end/console/ConsoleViewMessage.js?q=linkstringregex
    const URL_CONTROL_CODES = '\\u0000-\\u0020\\u007f-\\u009f';
    const LINKIFY_URL_REGEX = new RegExp('(?:[a-zA-Z][a-zA-Z0-9+.-]{2,}:\\/\\/|data:|www\\.)[^\\s' +
        URL_CONTROL_CODES +
        '"]{2,}[^\\s' +
        URL_CONTROL_CODES +
        '"\')}\\],:;.!?]', 'gu');
    /**
     * Splits the string into pieces that are URLs and non-URLs for linkification.
     * Invalid links (e.g. 'javascript:') are not linkified.
     *
     * For example,
     * splitByURL("visit http://example.com today")
     * Returns
     * [
     *   {isUrl: false, text: "visit "},
     *   {isUrl: true,  text: "http://example.com"},
     *   {isUrl: false, text: " today"},
     * ]
     */
    function splitByURL(text) {
        return splitByRegex(text, LINKIFY_URL_REGEX).map(({ matchesRegex, text }) => {
            return { isURL: matchesRegex, text };
        });
    }

    var I18N_0;
    if (typeof ngI18nClosureMode !== "undefined" && ngI18nClosureMode) {
        /**
         * @desc A button to close the snackbar message
         */ 
        const MSG_EXTERNAL_1536087519743707362$$TENSORBOARD_WEBAPP_ALERT_VIEWS_ALERT_DISPLAY_SNACKBAR_CONTAINER_TS_1 = goog.getMsg("Dismiss");
        I18N_0 = MSG_EXTERNAL_1536087519743707362$$TENSORBOARD_WEBAPP_ALERT_VIEWS_ALERT_DISPLAY_SNACKBAR_CONTAINER_TS_1;
    }
    else {
        I18N_0 = $localize `:A button to close the snackbar message␟ea4d9fe61420a3fce81cf54c4c615e3c19c646a6␟1536087519743707362:Dismiss`;
    }
    const _c2$5 = ["aria-label", I18N_0];
    function AlertDisplaySnackbarContainer_ng_container_1_ng_container_1_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementContainerStart(0);
        ɵɵtext(1);
        ɵɵelementContainerEnd();
    } if (rf & 2) {
        const piece_r2 = ɵɵnextContext().$implicit;
        ɵɵadvance(1);
        ɵɵtextInterpolate1(" ", piece_r2.text, " ");
    } }
    function AlertDisplaySnackbarContainer_ng_container_1_ng_template_2_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "a", 7);
        ɵɵtext(1);
        ɵɵelementEnd();
    } if (rf & 2) {
        const piece_r2 = ɵɵnextContext().$implicit;
        ɵɵpropertyInterpolate("href", piece_r2.text, ɵɵsanitizeUrl);
        ɵɵadvance(1);
        ɵɵtextInterpolate(piece_r2.text);
    } }
    function AlertDisplaySnackbarContainer_ng_container_1_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementContainerStart(0);
        ɵɵtemplate(1, AlertDisplaySnackbarContainer_ng_container_1_ng_container_1_Template, 2, 1, "ng-container", 5);
        ɵɵtemplate(2, AlertDisplaySnackbarContainer_ng_container_1_ng_template_2_Template, 2, 2, "ng-template", null, 6, ɵɵtemplateRefExtractor);
        ɵɵelementContainerEnd();
    } if (rf & 2) {
        const piece_r2 = ctx.$implicit;
        const _r4 = ɵɵreference(3);
        ɵɵadvance(1);
        ɵɵproperty("ngIf", !piece_r2.isURL)("ngIfElse", _r4);
    } }
    function AlertDisplaySnackbarContainer_button_3_Template(rf, ctx) { if (rf & 1) {
        const _r9 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "button", 8);
        ɵɵlistener("click", function AlertDisplaySnackbarContainer_button_3_Template_button_click_0_listener() { ɵɵrestoreView(_r9); const ctx_r8 = ɵɵnextContext(); return ctx_r8.onActionButtonClicked(); });
        ɵɵtext(1);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r1 = ɵɵnextContext();
        ɵɵadvance(1);
        ɵɵtextInterpolate1(" ", ctx_r1.alert.followupAction.localizedLabel, " ");
    } }
    class AlertDisplaySnackbarContainer {
        /**
         * @param {?} snackBarRef
         * @param {?} unknownData
         * @param {?} store
         */
        constructor(snackBarRef, unknownData, store) {
            this.snackBarRef = snackBarRef;
            this.unknownData = unknownData;
            this.store = store;
            this.splitByURL = splitByURL;
            this.alert = (/** @type {?} */ (unknownData));
        }
        /**
         * @return {?}
         */
        onActionButtonClicked() {
            return __awaiter(this, void 0, void 0, function* () {
                this.snackBarRef.dismiss();
                /** @type {?} */
                const followupAction = yield (/** @type {?} */ (this.alert.followupAction)).getFollowupAction(this.store);
                this.store.dispatch(followupAction);
            });
        }
        /**
         * @return {?}
         */
        onCloseButtonClicked() {
            this.snackBarRef.dismiss();
        }
    }
    AlertDisplaySnackbarContainer.decorators = [
        { type: Component, args: [{
                    selector: 'alert-display-snackbar',
                    templateUrl: './alert_display_snackbar_container.ng.html',
                    styleUrls: ['./alert_display_snackbar_container.css'],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    AlertDisplaySnackbarContainer.ctorParameters = () => [
        { type: MatSnackBarRef },
        { type: undefined, decorators: [{ type: Inject, args: [MAT_SNACK_BAR_DATA,] }] },
        { type: Store }
    ];
    /** @nocollapse */ AlertDisplaySnackbarContainer.ɵfac = function AlertDisplaySnackbarContainer_Factory(t) { return new (t || AlertDisplaySnackbarContainer)(ɵɵdirectiveInject(MatSnackBarRef), ɵɵdirectiveInject(MAT_SNACK_BAR_DATA), ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ AlertDisplaySnackbarContainer.ɵcmp = ɵɵdefineComponent({ type: AlertDisplaySnackbarContainer, selectors: [["alert-display-snackbar"]], decls: 7, vars: 2, consts: [[1, "message"], [4, "ngFor", "ngForOf"], [1, "controls"], ["mat-button", "", "class", "followup-button", 3, "click", 4, "ngIf"], ["mat-button", "", 1, "dismiss-button", 3, "click", 6, "aria-label"], [4, "ngIf", "ngIfElse"], ["linkPiece", ""], ["rel", "noreferrer noopener", "target", "_blank", 3, "href"], ["mat-button", "", 1, "followup-button", 3, "click"]], template: function AlertDisplaySnackbarContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "div", 0);
            ɵɵtemplate(1, AlertDisplaySnackbarContainer_ng_container_1_Template, 4, 2, "ng-container", 1);
            ɵɵelementEnd();
            ɵɵelementStart(2, "div", 2);
            ɵɵtemplate(3, AlertDisplaySnackbarContainer_button_3_Template, 2, 1, "button", 3);
            ɵɵelementStart(4, "button", 4);
            ɵɵi18nAttributes(5, _c2$5);
            ɵɵlistener("click", function AlertDisplaySnackbarContainer_Template_button_click_4_listener() { return ctx.onCloseButtonClicked(); });
            ɵɵtext(6, " Dismiss ");
            ɵɵelementEnd();
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵadvance(1);
            ɵɵproperty("ngForOf", ctx.splitByURL(ctx.alert.localizedMessage));
            ɵɵadvance(2);
            ɵɵproperty("ngIf", ctx.alert.followupAction);
        } }, directives: [NgForOf, NgIf, MatButton], styles: ["[_nghost-%COMP%]{display:flex;flex-wrap:wrap}.message[_ngcontent-%COMP%]{font-size:14px;align-self:center;margin:5px 0;word-break:break-word}.message[_ngcontent-%COMP%]   a[_ngcontent-%COMP%]{color:inherit}.controls[_ngcontent-%COMP%]{white-space:nowrap;margin-left:auto}button[_ngcontent-%COMP%]{text-transform:uppercase}"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/alert/views/alert_snackbar_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Renders alerts in a 'snackbar' to indicate them to the user.
     */
    class AlertSnackbarContainer {
        /**
         * @param {?} store
         * @param {?} snackBar
         */
        constructor(store, snackBar) {
            this.store = store;
            this.snackBar = snackBar;
            this.ngUnsubscribe = new Subject();
        }
        /**
         * @return {?}
         */
        ngOnInit() {
            this.store
                .select(getLatestAlert)
                .pipe(takeUntil(this.ngUnsubscribe), filter((/**
             * @param {?} alert
             * @return {?}
             */
            (alert) => Boolean(alert))))
                .subscribe((/**
             * @param {?} alert
             * @return {?}
             */
            (alert) => {
                this.showAlert((/** @type {?} */ (alert)));
            }));
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this.ngUnsubscribe.next();
            this.ngUnsubscribe.complete();
        }
        /**
         * @private
         * @param {?} alertInfo
         * @return {?}
         */
        showAlert(alertInfo) {
            this.snackBar.openFromComponent(AlertDisplaySnackbarContainer, {
                duration: 5000,
                horizontalPosition: 'start',
                verticalPosition: 'bottom',
                data: alertInfo,
            });
        }
    }
    AlertSnackbarContainer.decorators = [
        { type: Component, args: [{
                    selector: 'alert-snackbar',
                    template: '',
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    AlertSnackbarContainer.ctorParameters = () => [
        { type: Store },
        { type: MatSnackBar }
    ];
    /** @nocollapse */ AlertSnackbarContainer.ɵfac = function AlertSnackbarContainer_Factory(t) { return new (t || AlertSnackbarContainer)(ɵɵdirectiveInject(Store), ɵɵdirectiveInject(MatSnackBar)); };
    /** @nocollapse */ AlertSnackbarContainer.ɵcmp = ɵɵdefineComponent({ type: AlertSnackbarContainer, selectors: [["alert-snackbar"]], decls: 0, vars: 0, template: function AlertSnackbarContainer_Template(rf, ctx) { }, encapsulation: 2, changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/deeplink/hash.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    // TODO(tensorboard-team): merge this module with tf_storage/storage.ts when
    // tf_ts_library can be referenced by tf_web_library.
    /** @type {?} */
    const TAB$1 = '__tab__';
    class HashDeepLinker {
        constructor() {
            this.tfStorage = (/** @type {?} */ (document.createElement('tf-storage')));
            /** @type {?} */
            const tfGlobals = (/** @type {?} */ (document.createElement('tf-globals')));
            // Note: `migrateLegacyURLScheme()` must be called before `setUseHash`, so
            // that tfStorage reads from the actual URL, not the fake hash for tests
            // only.
            tfGlobals.tf_globals.setUseHash(true);
            this.tfStorage.tf_storage.migrateLegacyURLScheme();
        }
        /**
         * @param {?} key
         * @return {?}
         */
        getString(key) {
            return this.tfStorage.tf_storage.getString(key);
        }
        /**
         * @param {?} key
         * @param {?} value
         * @param {?=} options
         * @return {?}
         */
        setString(key, value, options) {
            this.tfStorage.tf_storage.setString(key, value, options);
        }
        /**
         * @return {?}
         */
        getPluginId() {
            return this.getString(TAB$1);
        }
        /**
         * @param {?} pluginId
         * @param {?=} options
         * @return {?}
         */
        setPluginId(pluginId, options) {
            this.setString(TAB$1, pluginId, options);
        }
    }
    HashDeepLinker.decorators = [
        { type: Injectable },
    ];
    /** @nocollapse */
    HashDeepLinker.ctorParameters = () => [];
    /** @nocollapse */ HashDeepLinker.ɵfac = function HashDeepLinker_Factory(t) { return new (t || HashDeepLinker)(); };
    /** @nocollapse */ HashDeepLinker.ɵprov = ɵɵdefineInjectable({ token: HashDeepLinker, factory: HashDeepLinker.ɵfac });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/core/views/hash_storage_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @enum {number} */
    const ChangedProp = {
        ACTIVE_PLUGIN: 0,
    };
    ChangedProp[ChangedProp.ACTIVE_PLUGIN] = 'ACTIVE_PLUGIN';
    class HashStorageComponent {
        /**
         * @param {?} deepLinker
         */
        constructor(deepLinker) {
            this.deepLinker = deepLinker;
            this.onHashChange = this.onHashChangedImpl.bind(this);
            this.onValueChange = new EventEmitter();
        }
        /**
         * @private
         * @return {?}
         */
        onHashChangedImpl() {
            /** @type {?} */
            const activePluginId = this.deepLinker.getPluginId();
            if (activePluginId !== this.activePluginId) {
                this.onValueChange.emit({
                    prop: ChangedProp.ACTIVE_PLUGIN,
                    value: activePluginId,
                });
            }
        }
        /**
         * @return {?}
         */
        ngOnInit() {
            // Cannot use the tf_storage hash listener because it binds to event before the
            // zone.js patch. According to [1], zone.js patches various asynchronos calls and
            // event listeners to detect "changes" and mark components as dirty for re-render.
            // When using tf_storage hash listener, it causes bad renders in Angular due to
            // missing dirtiness detection.
            // [1]: https://blog.angular-university.io/how-does-angular-2-change-detection-really-work/
            window.addEventListener('hashchange', this.onHashChange);
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            window.removeEventListener('hashchange', this.onHashChange);
        }
        /**
         * @param {?} changes
         * @return {?}
         */
        ngOnChanges(changes) {
            if (changes['activePluginId']) {
                /** @type {?} */
                const activePluginIdChange = changes['activePluginId'];
                /** @type {?} */
                const option = {
                    defaultValue: '',
                    useLocationReplace: activePluginIdChange.previousValue === null ||
                        activePluginIdChange.firstChange,
                };
                /** @type {?} */
                const value = activePluginIdChange.currentValue === null
                    ? ''
                    : activePluginIdChange.currentValue;
                this.deepLinker.setPluginId(value, option);
            }
        }
    }
    HashStorageComponent.decorators = [
        { type: Component, args: [{
                    selector: 'hash-storage-component',
                    template: '',
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    HashStorageComponent.ctorParameters = () => [
        { type: HashDeepLinker }
    ];
    HashStorageComponent.propDecorators = {
        activePluginId: [{ type: Input }],
        onValueChange: [{ type: Output }]
    };
    /** @nocollapse */ HashStorageComponent.ɵfac = function HashStorageComponent_Factory(t) { return new (t || HashStorageComponent)(ɵɵdirectiveInject(HashDeepLinker)); };
    /** @nocollapse */ HashStorageComponent.ɵcmp = ɵɵdefineComponent({ type: HashStorageComponent, selectors: [["hash-storage-component"]], inputs: { activePluginId: "activePluginId" }, outputs: { onValueChange: "onValueChange" }, features: [ɵɵNgOnChangesFeature], decls: 0, vars: 0, template: function HashStorageComponent_Template(rf, ctx) { }, encapsulation: 2, changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/core/views/hash_storage_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class HashStorageContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.activePluginId$ = this.store.pipe(select(getActivePlugin));
        }
        /**
         * @param {?} change
         * @return {?}
         */
        onValueChanged(change) {
            switch (change.prop) {
                case ChangedProp.ACTIVE_PLUGIN:
                    this.store.dispatch(pluginUrlHashChanged({ plugin: change.value }));
                    break;
            }
        }
    }
    HashStorageContainer.decorators = [
        { type: Component, args: [{
                    selector: 'hash-storage',
                    template: `
    <hash-storage-component
      [activePluginId]="activePluginId$ | async"
      (onValueChange)="onValueChanged($event)"
    >
    </hash-storage-component>
  `,
                    styles: [
                        `
      :host {
        display: none;
      }
    `,
                    ],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    HashStorageContainer.ctorParameters = () => [
        { type: Store }
    ];
    /** @nocollapse */ HashStorageContainer.ɵfac = function HashStorageContainer_Factory(t) { return new (t || HashStorageContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ HashStorageContainer.ɵcmp = ɵɵdefineComponent({ type: HashStorageContainer, selectors: [["hash-storage"]], decls: 2, vars: 3, consts: [[3, "activePluginId", "onValueChange"]], template: function HashStorageContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "hash-storage-component", 0);
            ɵɵlistener("onValueChange", function HashStorageContainer_Template_hash_storage_component_onValueChange_0_listener($event) { return ctx.onValueChanged($event); });
            ɵɵpipe(1, "async");
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵproperty("activePluginId", ɵɵpipeBind1(1, 1, ctx.activePluginId$));
        } }, directives: [HashStorageComponent], pipes: [AsyncPipe], styles: ["[_nghost-%COMP%] {\n        display: none;\n      }"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/core/views/page_title_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class PageTitleComponent {
        /**
         * @param {?} changes
         * @return {?}
         */
        ngOnChanges(changes) {
            if (changes['title']) {
                document.title = changes['title'].currentValue;
            }
        }
    }
    PageTitleComponent.decorators = [
        { type: Component, args: [{
                    selector: 'page-title-component',
                    template: '',
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    PageTitleComponent.propDecorators = {
        title: [{ type: Input }]
    };
    /** @nocollapse */ PageTitleComponent.ɵfac = function PageTitleComponent_Factory(t) { return new (t || PageTitleComponent)(); };
    /** @nocollapse */ PageTitleComponent.ɵcmp = ɵɵdefineComponent({ type: PageTitleComponent, selectors: [["page-title-component"]], inputs: { title: "title" }, features: [ɵɵNgOnChangesFeature], decls: 0, vars: 0, template: function PageTitleComponent_Template(rf, ctx) { }, encapsulation: 2, changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/core/views/page_title_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class PageTitleContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.title$ = this.store.pipe(select(getEnvironment), map((/**
             * @param {?} env
             * @return {?}
             */
            (env) => env.window_title || 'TensorBoard')), 
            // (it's an empty string when the `--window_title` flag is not set)
            distinctUntilChanged());
        }
    }
    PageTitleContainer.decorators = [
        { type: Component, args: [{
                    selector: 'page-title',
                    template: `
    <page-title-component [title]="title$ | async"></page-title-component>
  `,
                    styles: [
                        `
      :host {
        display: none;
      }
    `,
                    ],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    PageTitleContainer.ctorParameters = () => [
        { type: Store }
    ];
    /** @nocollapse */ PageTitleContainer.ɵfac = function PageTitleContainer_Factory(t) { return new (t || PageTitleContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ PageTitleContainer.ɵcmp = ɵɵdefineComponent({ type: PageTitleContainer, selectors: [["page-title"]], decls: 2, vars: 3, consts: [[3, "title"]], template: function PageTitleContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelement(0, "page-title-component", 0);
            ɵɵpipe(1, "async");
        } if (rf & 2) {
            ɵɵproperty("title", ɵɵpipeBind1(1, 1, ctx.title$));
        } }, directives: [PageTitleComponent], pipes: [AsyncPipe], styles: ["[_nghost-%COMP%] {\n        display: none;\n      }"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/settings/polymer_interop_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * SettingsPolymerInterop is a temporary interop module that writes settings in
     * Ngrx store to Polymer-based TensorBoard components. As long as TensorBoard
     * renders Polymer components, this module should be used.
     *
     * NOTE: there are two classes of settings in the Polymer land: (1) ones
     * persisted in URL and (2) ones stored in JavaScript. This module interops with
     * only (2) kinds. For (1), please refer to the hash_storage.
     */
    class SettingsPolymerInteropContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.ngUnsubscribe = new Subject();
            this.getPageSize$ = this.store.pipe(select(getPageSize));
            this.paginatedViewStore = ((/** @type {?} */ (document.createElement('tf-paginated-view-store')))).tf_paginated_view;
        }
        /**
         * @return {?}
         */
        ngOnInit() {
            this.getPageSize$
                .pipe(takeUntil(this.ngUnsubscribe), distinctUntilChanged())
                .subscribe((/**
             * @param {?} pageSize
             * @return {?}
             */
            (pageSize) => {
                this.paginatedViewStore.setLimit(pageSize);
            }));
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this.ngUnsubscribe.next();
            this.ngUnsubscribe.complete();
        }
    }
    SettingsPolymerInteropContainer.decorators = [
        { type: Component, args: [{
                    selector: 'settings-polymer-interop',
                    template: '',
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    SettingsPolymerInteropContainer.ctorParameters = () => [
        { type: Store }
    ];
    /** @nocollapse */ SettingsPolymerInteropContainer.ɵfac = function SettingsPolymerInteropContainer_Factory(t) { return new (t || SettingsPolymerInteropContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ SettingsPolymerInteropContainer.ɵcmp = ɵɵdefineComponent({ type: SettingsPolymerInteropContainer, selectors: [["settings-polymer-interop"]], decls: 0, vars: 0, template: function SettingsPolymerInteropContainer_Template(rf, ctx) { }, encapsulation: 2, changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/app_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class AppContainer {
        // vcRef is required by ngx-color-picker in order for it to place the popup
        // in the root node in a modal mode.
        // https://github.com/zefoy/ngx-color-picker/blob/94a7c862bb61d7207f21281526fcd94453219b54/projects/lib/src/lib/color-picker.directive.ts#L168-L175
        /**
         * @param {?} store
         * @param {?} vcRef
         */
        constructor(store, vcRef) {
            this.store = store;
            this.vcRef = vcRef;
        }
        /**
         * @return {?}
         */
        ngOnInit() {
            this.store.dispatch(coreLoaded());
        }
    }
    AppContainer.decorators = [
        { type: Component, args: [{
                    selector: 'tb-webapp',
                    templateUrl: './app_container.ng.html',
                    styleUrls: ['./app_container.css'],
                },] },
    ];
    /** @nocollapse */
    AppContainer.ctorParameters = () => [
        { type: Store },
        { type: ViewContainerRef }
    ];
    /** @nocollapse */ AppContainer.ɵfac = function AppContainer_Factory(t) { return new (t || AppContainer)(ɵɵdirectiveInject(Store), ɵɵdirectiveInject(ViewContainerRef)); };
    /** @nocollapse */ AppContainer.ɵcmp = ɵɵdefineComponent({ type: AppContainer, selectors: [["tb-webapp"]], decls: 8, vars: 0, consts: [["main", ""]], template: function AppContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelement(0, "app-header");
            ɵɵelementStart(1, "main", null, 0);
            ɵɵelement(3, "router-outlet");
            ɵɵelementEnd();
            ɵɵelement(4, "alert-snackbar");
            ɵɵelement(5, "hash-storage");
            ɵɵelement(6, "page-title");
            ɵɵelement(7, "settings-polymer-interop");
        } }, directives: [HeaderComponent, RouterOutletContainer, AlertSnackbarContainer, HashStorageContainer, PageTitleContainer, SettingsPolymerInteropContainer], styles: ["html[_ngcontent-%COMP%], body[_ngcontent-%COMP%] {\n  font-family: Roboto, sans-serif;\n  height: 100%;\n  margin: 0;\n  padding: 0;\n}\n\n[_nghost-%COMP%] {\n  background: #f5f5f5;\n  display: flex;\n  flex-direction: column;\n  height: 100%;\n}\n\napp-header[_ngcontent-%COMP%] {\n  box-shadow: 0 1px 3px 3px rgba(0, 0, 0, 0.25);\n  flex: 0 0;\n  z-index: 1; \n}\n\nmain[_ngcontent-%COMP%] {\n  flex-grow: 1;\n  overflow: auto;\n}"] });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/app_routing/location.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const utils = {
        /**
         * @return {?}
         */
        getHref() {
            return window.location.href;
        },
    };
    /**
     * @param {?} navOrRoute
     * @return {?}
     */
    function isNavigation(navOrRoute) {
        return (navOrRoute.hasOwnProperty('pathname') &&
            !navOrRoute.hasOwnProperty('queryParams'));
    }
    class Location$1 {
        /**
         * @return {?}
         */
        getHref() {
            return utils.getHref();
        }
        /**
         * @return {?}
         */
        getSearch() {
            /** @type {?} */
            const searchParams = new URLSearchParams(window.location.search);
            /** @type {?} */
            const serializableSearchParams = [];
            // URLSearchParams is a Iterable but TypeScript does not know about that.
            searchParams.forEach((/**
             * @param {?} value
             * @param {?} key
             * @return {?}
             */
            (value, key) => {
                serializableSearchParams.push({ key, value });
            }));
            return serializableSearchParams;
        }
        /**
         * @return {?}
         */
        getHash() {
            return window.location.hash;
        }
        /**
         * @return {?}
         */
        getPath() {
            return window.location.pathname;
        }
        /**
         * @param {?} path
         * @return {?}
         */
        replaceState(path) {
            window.history.replaceState(null, '', path);
        }
        /**
         * @param {?} path
         * @return {?}
         */
        pushState(path) {
            window.history.pushState(null, '', path);
        }
        /**
         * @return {?}
         */
        onPopState() {
            return fromEvent(window, 'popstate').pipe(map((/**
             * @return {?}
             */
            () => {
                return {
                    pathname: this.getPath(),
                    queryParams: this.getSearch(),
                };
            })));
        }
        /**
         * @param {?} relativePath
         * @return {?}
         */
        getResolvedPath(relativePath) {
            /** @type {?} */
            const url = new URL(relativePath, utils.getHref());
            return url.pathname;
        }
        /**
         * @param {?} routeLike
         * @param {?=} shouldPreserveHash
         * @return {?}
         */
        getFullPathFromRouteOrNav(routeLike, shouldPreserveHash) {
            // TODO(stephanwlee): support hashes in the routeLike.
            /** @type {?} */
            const pathname = this.getResolvedPath(routeLike.pathname);
            /** @type {?} */
            let search = '';
            if (!isNavigation(routeLike) && routeLike.queryParams.length) {
                search =
                    '?' +
                        createURLSearchParamsFromSerializableQueryParams(routeLike.queryParams).toString();
            }
            /** @type {?} */
            const hash = shouldPreserveHash ? this.getHash() : '';
            return `${pathname}${search}${hash}`;
        }
    }
    Location$1.decorators = [
        { type: Injectable },
    ];
    /** @nocollapse */ Location$1.ɵfac = function Location_Factory(t) { return new (t || Location$1)(); };
    /** @nocollapse */ Location$1.ɵprov = ɵɵdefineInjectable({ token: Location$1, factory: Location$1.ɵfac });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/app_routing/app_root.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class AppRootProvider {
        /**
         * @param {?} location
         */
        constructor(location) {
            this.appRoot = this.getAppRootFromMetaElement(location);
        }
        /**
         * appRoot path starts with `/` and always end with `/`.
         * @private
         * @param {?} location
         * @return {?}
         */
        getAppRootFromMetaElement(location) {
            /** @type {?} */
            const metaEl = document.querySelector('head meta[name="tb-relative-root"]');
            if (!metaEl)
                return '/';
            const { pathname } = new URL(((/** @type {?} */ (metaEl))).content, location.getHref());
            return pathname.replace(/\/*$/, '/');
        }
        /**
         * Returns actual full pathname that includes path prefix of the application.
         * @param {?} absPathname
         * @return {?}
         */
        getAbsPathnameWithAppRoot(absPathname) {
            // appRoot has trailing '/'. Remove one so we don't have "//".
            return this.appRoot.slice(0, -1) + absPathname;
        }
        /**
         * @param {?} absPathname
         * @return {?}
         */
        getAppRootlessPathname(absPathname) {
            if (absPathname.startsWith(this.appRoot)) {
                // appRoot ends with "/" and we need the trimmed pathname to start with "/" since
                // routes are defined with starting "/".
                return '/' + absPathname.slice(this.appRoot.length);
            }
            return absPathname;
        }
    }
    AppRootProvider.decorators = [
        { type: Injectable },
    ];
    /** @nocollapse */
    AppRootProvider.ctorParameters = () => [
        { type: Location$1 }
    ];
    /** @nocollapse */ AppRootProvider.ɵfac = function AppRootProvider_Factory(t) { return new (t || AppRootProvider)(ɵɵinject(Location$1)); };
    /** @nocollapse */ AppRootProvider.ɵprov = ɵɵdefineInjectable({ token: AppRootProvider, factory: AppRootProvider.ɵfac });
    class TestableAppRootProvider extends AppRootProvider {
        /**
         * @return {?}
         */
        getAppRoot() {
            return this.appRoot;
        }
        /**
         * @param {?} appRoot
         * @return {?}
         */
        setAppRoot(appRoot) {
            this.appRoot = appRoot;
        }
    }
    TestableAppRootProvider.decorators = [
        { type: Injectable },
    ];
    /** @nocollapse */ TestableAppRootProvider.ɵfac = function TestableAppRootProvider_Factory(t) { return ɵTestableAppRootProvider_BaseFactory(t || TestableAppRootProvider); };
    /** @nocollapse */ TestableAppRootProvider.ɵprov = ɵɵdefineInjectable({ token: TestableAppRootProvider, factory: TestableAppRootProvider.ɵfac });
    const ɵTestableAppRootProvider_BaseFactory = ɵɵgetInheritedFactory(TestableAppRootProvider);

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * Created when router rehydrates state from the URL after a browser initiated
     * event. Please do note that the action is fired before `navigated` so make
     * sure the view can handle inconsistent state right before a navigation.
     */
    const stateRehydratedFromUrl = createAction('[App Routing] State Rehydrated From Url', props());
    /**
     * Created when user intends to navigate in the application
     */
    const navigationRequested = createAction('[App Routing] In App Navigation Requested', props());
    /**
     * Created after a route is matched but before navigated.
     */
    const navigating = createAction('[App Routing] In App Navigating', props());
    /**
     * Created after navigation is successful.
     */
    const navigated = createAction('[App Routing] In App Navigated', props());

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/app_routing/programmatical_navigation_types.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const NAVIGATION_PROVIDER = new InjectionToken('[App Routing] Programmatical Navigation Provider');

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/app_routing/programmatical_navigation_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class ProgrammaticalNavigationModule {
        /**
         * @param {?} providers
         */
        constructor(providers) {
            this.providers = new Map();
            for (const provider of providers || []) {
                if (this.providers.has(provider.actionCreator.type)) {
                    throw new RangeError(`"${provider.actionCreator.type}" is already registered for nav.` +
                        ' Multiple navigations on same kick is not allowed.');
                }
                this.providers.set(provider.actionCreator.type, provider.lambda);
            }
        }
        /**
         * @param {?} action
         * @return {?}
         */
        getNavigation(action) {
            /** @type {?} */
            const lambda = this.providers.get(action.type);
            if (!lambda) {
                return null;
            }
            return lambda(action);
        }
        /**
         * An NgModule that provides programmmatic navigation routines.
         * On an action subscribed fire, it invokes a navigation.
         *
         * WARN: internally, it is implemented with Ngrx composite actions so there
         * may be a timing issue. Mitigate the issue by account for potential data
         * inconsistency issue in the container.
         *
         * function provider() {
         *   return {
         *     actionCreator: somethingHappened,
         *     lambda: (action: typeof somethingHappened) => {
         *       return {routeKind: RouteKind.EXPERIMENT, routeParams: {}};
         *     },
         *   };
         * }
         *
         * \@NgModule({
         *   imports: [
         *     RouteRegistryModule.registerProgrammaticalNavigation(provider),
         *     RouteRegistryModule.registerProgrammaticalNavigation(anotherProvider),
         *   ],
         * })
         * @template AC, U
         * @param {?} providerFactory
         * @return {?}
         */
        static registerProgrammaticalNavigation(providerFactory) {
            return {
                ngModule: ProgrammaticalNavigationModule,
                providers: [
                    {
                        provide: NAVIGATION_PROVIDER,
                        multi: true,
                        useFactory: providerFactory,
                    },
                ],
            };
        }
    }
    ProgrammaticalNavigationModule.decorators = [
        { type: NgModule, args: [{},] },
    ];
    /** @nocollapse */
    ProgrammaticalNavigationModule.ctorParameters = () => [
        { type: Array, decorators: [{ type: Optional }, { type: Inject, args: [NAVIGATION_PROVIDER,] }] }
    ];
    /** @nocollapse */ ProgrammaticalNavigationModule.ɵmod = ɵɵdefineNgModule({ type: ProgrammaticalNavigationModule });
    /** @nocollapse */ ProgrammaticalNavigationModule.ɵinj = ɵɵdefineInjector({ factory: function ProgrammaticalNavigationModule_Factory(t) { return new (t || ProgrammaticalNavigationModule)(ɵɵinject(NAVIGATION_PROVIDER, 8)); } });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/app_routing/effects/app_routing_effects.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const initAction = createAction('[App Routing] Effects Init');
    class AppRoutingEffects {
        /**
         * @param {?} actions$
         * @param {?} store
         * @param {?} location
         * @param {?} registry
         * @param {?} programmaticalNavModule
         * @param {?} appRootProvider
         */
        constructor(actions$, store, location, registry, programmaticalNavModule, appRootProvider) {
            this.actions$ = actions$;
            this.store = store;
            this.location = location;
            this.programmaticalNavModule = programmaticalNavModule;
            this.appRootProvider = appRootProvider;
            this.onNavigationRequested$ = this.actions$.pipe(ofType(navigationRequested), map((/**
             * @param {?} navigation
             * @return {?}
             */
            (navigation) => {
                /** @type {?} */
                const resolvedPathname = navigation.pathname.startsWith('/')
                    ? this.appRootProvider.getAbsPathnameWithAppRoot(navigation.pathname)
                    : this.location.getResolvedPath(navigation.pathname);
                return Object.assign(Object.assign({}, navigation), { pathname: resolvedPathname });
            })));
            this.onInit$ = this.actions$
                .pipe(ofType(initAction))
                .pipe(delay(0), map((/**
             * @return {?}
             */
            () => {
                return {
                    pathname: this.location.getPath(),
                    queryParams: this.location.getSearch(),
                    replaceState: true,
                    browserInitiated: true,
                };
            })));
            /**
             * Input observable must have absolute pathname with, when appRoot is present,
             * appRoot prefixed (e.g., window.location.pathname).
             */
            this.userInitNavRoute$ = merge(this.onNavigationRequested$, this.onInit$, this.location.onPopState().pipe(map((/**
             * @param {?} navigation
             * @return {?}
             */
            (navigation) => {
                return {
                    pathname: navigation.pathname,
                    replaceState: navigation.replaceState,
                    browserInitiated: true,
                };
            })))).pipe(map((/**
             * @param {?} navigation
             * @return {?}
             */
            (navigation) => {
                // Expect to have absolute navigation here.
                if (!navigation.pathname.startsWith('/')) {
                    throw new Error(`[App routing] pathname must start with '/'. Got: ${navigation.pathname}`);
                }
                return Object.assign(Object.assign({}, navigation), { pathname: this.appRootProvider.getAppRootlessPathname(navigation.pathname) });
            })), map((/**
             * @param {?} navigationWithAbsolutePath
             * @return {?}
             */
            (navigationWithAbsolutePath) => {
                /** @type {?} */
                const routeMatch = this.routeConfigs
                    ? this.routeConfigs.match(navigationWithAbsolutePath)
                    : null;
                return {
                    routeMatch,
                    options: {
                        replaceState: navigationWithAbsolutePath.replaceState,
                        browserInitiated: navigationWithAbsolutePath.browserInitiated,
                    },
                };
            })));
            this.programmticalNavRoute$ = this.actions$.pipe(map((/**
             * @param {?} action
             * @return {?}
             */
            (action) => {
                return this.programmaticalNavModule.getNavigation(action);
            })), filter((/**
             * @param {?} nav
             * @return {?}
             */
            (nav) => {
                return nav !== null;
            })), map((/**
             * @param {?} programmaticalNavigation
             * @return {?}
             */
            (programmaticalNavigation) => {
                const { routeKind, routeParams } = (/** @type {?} */ (programmaticalNavigation));
                /** @type {?} */
                const routeMatch = this.routeConfigs
                    ? this.routeConfigs.matchByRouteKind(routeKind, routeParams)
                    : null;
                return {
                    routeMatch,
                    options: {
                        replaceState: false,
                        browserInitiated: false,
                    },
                };
            })));
            this.validatedRoute$ = merge(this.userInitNavRoute$, this.programmticalNavRoute$).pipe(filter((/**
             * @param {?} __0
             * @return {?}
             */
            ({ routeMatch }) => Boolean(routeMatch))), map((/**
             * @param {?} routeMatchAndOptions
             * @return {?}
             */
            (routeMatchAndOptions) => {
                return {
                    routeMatch: (/** @type {?} */ (routeMatchAndOptions.routeMatch)),
                    options: routeMatchAndOptions.options,
                };
            })));
            /**
             * @export
             */
            this.fireNavigatedIfValidRoute$ = createEffect((/**
             * @return {?}
             */
            () => {
                return this.validatedRoute$.pipe(tap((/**
                 * @param {?} __0
                 * @return {?}
                 */
                ({ routeMatch, options }) => {
                    if (options.browserInitiated && routeMatch.deepLinkProvider) {
                        /** @type {?} */
                        const rehydratingState = routeMatch.deepLinkProvider.deserializeQueryParams(this.location.getSearch());
                        this.store.dispatch(stateRehydratedFromUrl({
                            routeKind: routeMatch.routeKind,
                            partialState: rehydratingState,
                        }));
                    }
                })), switchMap((/**
                 * @param {?} __0
                 * @return {?}
                 */
                ({ routeMatch, options }) => {
                    var _a;
                    /** @type {?} */
                    const navigationOptions = {
                        replaceState: (_a = options.replaceState) !== null && _a !== void 0 ? _a : false,
                    };
                    /** @type {?} */
                    const routeObservableWithoutQuery = of({
                        routeKind: routeMatch.routeKind,
                        params: routeMatch.params,
                        pathname: routeMatch.pathname,
                        queryParams: [],
                        navigationOptions,
                    });
                    if (routeMatch.deepLinkProvider === null) {
                        return routeObservableWithoutQuery;
                    }
                    return (/** @type {?} */ (routeMatch
                        .deepLinkProvider)).serializeStateToQueryParams(this.store)
                        .pipe(map((/**
                     * @param {?} queryParams
                     * @param {?} index
                     * @return {?}
                     */
                    (queryParams, index) => {
                        return {
                            routeKind: routeMatch.routeKind,
                            params: routeMatch.params,
                            pathname: routeMatch.pathname,
                            queryParams,
                            navigationOptions: index === 0
                                ? navigationOptions
                                : Object.assign(Object.assign({}, navigationOptions), { replaceState: true }),
                        };
                    })));
                })), tap((/**
                 * @param {?} route
                 * @return {?}
                 */
                (route) => {
                    // b/160185039: Allows the route store + router outlet to change
                    // before the route change so all components do not have to
                    // safeguard against the case when `routeId` (routeKind and
                    // routeParams) do not have unexpected values. Because we
                    // debounceTime, technically, it does not fire two actions
                    // sequentially.
                    this.store.dispatch(navigating({ after: route }));
                })), 
                // Let the router-outlet flush the change in a microtask.
                debounceTime(0), withLatestFrom(this.store.select(getActiveRoute)), map((/**
                 * @param {?} __0
                 * @return {?}
                 */
                ([route, oldRoute]) => {
                    return navigated({ before: oldRoute, after: route });
                })));
            }));
            // TODO(stephanwlee): move this to a "view".
            /**
             * @export
             */
            this.changeBrowserUrl$ = createEffect((/**
             * @return {?}
             */
            () => {
                return this.actions$.pipe(ofType(navigated), withLatestFrom(this.store.select(getActiveRoute)), filter((/**
                 * @param {?} __0
                 * @return {?}
                 */
                ([, route]) => Boolean(route))), map((/**
                 * @param {?} __0
                 * @return {?}
                 */
                ([navigatedAction, route]) => {
                    // The URL hash can be set via HashStorageComponent (which uses
                    // Polymer's tf-storage). DeepLinkProviders also modify the URL when
                    // a provider's serializeStateToQueryParams() emits. These result in
                    // the URL updated without the previous hash. HashStorageComponent
                    // makes no attempt to restore the hash, so it is dropped.
                    // The URL hash can be set via HashStorageComponent (which uses
                    // Polymer's tf-storage). DeepLinkProviders also modify the URL when
                    // a provider's serializeStateToQueryParams() emits. These result in
                    // the URL updated without the previous hash. HashStorageComponent
                    // makes no attempt to restore the hash, so it is dropped.
                    // This results in bad behavior when refreshing (e.g. lost active
                    // plugin) and when changing dashboards (e.g. lost tagFilter).
                    // TODO(b/169799696): either AppRouting should manage the URL entirely
                    // (including hash), or we make the app wait for AppRouting to
                    // initialize before setting the active plugin hash.
                    // See https://github.com/tensorflow/tensorboard/issues/4207.
                    /** @type {?} */
                    const oldRoute = navigatedAction.before;
                    /** @type {?} */
                    const preserveHash = oldRoute === null ||
                        getRouteId(oldRoute.routeKind, oldRoute.params) ===
                            getRouteId((/** @type {?} */ (route)).routeKind, (/** @type {?} */ (route)).params);
                    return {
                        preserveHash,
                        route: (/** @type {?} */ (route)),
                    };
                })), filter((/**
                 * @param {?} __0
                 * @return {?}
                 */
                ({ route }) => {
                    return !areRoutesEqual(route, {
                        pathname: this.appRootProvider.getAppRootlessPathname(this.location.getPath()),
                        queryParams: this.location.getSearch(),
                    });
                })), tap((/**
                 * @param {?} __0
                 * @return {?}
                 */
                ({ preserveHash, route }) => {
                    if (route.navigationOptions.replaceState) {
                        this.location.replaceState(this.appRootProvider.getAbsPathnameWithAppRoot(this.location.getFullPathFromRouteOrNav(route, preserveHash)));
                    }
                    else {
                        this.location.pushState(this.appRootProvider.getAbsPathnameWithAppRoot(this.location.getFullPathFromRouteOrNav(route, preserveHash)));
                    }
                })));
            }), { dispatch: false });
            this.routeConfigs = registry.getRouteConfigs();
        }
        /**
         * @export
         * @return {?}
         */
        ngrxOnInitEffects() {
            return initAction();
        }
    }
    AppRoutingEffects.decorators = [
        { type: Injectable },
    ];
    /** @nocollapse */
    AppRoutingEffects.ctorParameters = () => [
        { type: Actions },
        { type: Store },
        { type: Location$1 },
        { type: RouteRegistryModule },
        { type: ProgrammaticalNavigationModule },
        { type: AppRootProvider }
    ];
    /** @nocollapse */ AppRoutingEffects.ɵfac = function AppRoutingEffects_Factory(t) { return new (t || AppRoutingEffects)(ɵɵinject(Actions), ɵɵinject(Store), ɵɵinject(Location$1), ɵɵinject(RouteRegistryModule), ɵɵinject(ProgrammaticalNavigationModule), ɵɵinject(AppRootProvider)); };
    /** @nocollapse */ AppRoutingEffects.ɵprov = ɵɵdefineInjectable({ token: AppRoutingEffects, factory: AppRoutingEffects.ɵfac });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/app_routing/location_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class LocationModule {
    }
    LocationModule.decorators = [
        { type: NgModule, args: [{
                    providers: [Location$1],
                },] },
    ];
    /** @nocollapse */ LocationModule.ɵmod = ɵɵdefineNgModule({ type: LocationModule });
    /** @nocollapse */ LocationModule.ɵinj = ɵɵdefineInjector({ factory: function LocationModule_Factory(t) { return new (t || LocationModule)(); }, providers: [Location$1] });

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    const initialState$1 = {
        activeRoute: null,
        nextRoute: null,
    };
    const reducer$1 = createReducer(initialState$1, on(navigating, (state, { after }) => {
        return Object.assign(Object.assign({}, state), { nextRoute: after });
    }), on(navigated, (state, { after }) => {
        return Object.assign(Object.assign({}, state), { activeRoute: after, nextRoute: null });
    }));
    function reducers$1(state, action) {
        return reducer$1(state, action);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/app_routing/app_routing_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class AppRoutingModule {
    }
    AppRoutingModule.decorators = [
        { type: NgModule, args: [{
                    imports: [
                        StoreModule.forFeature(APP_ROUTING_FEATURE_KEY, reducers$1),
                        EffectsModule.forFeature([AppRoutingEffects]),
                        LocationModule,
                    ],
                    providers: [ProgrammaticalNavigationModule, AppRootProvider],
                },] },
    ];
    /** @nocollapse */ AppRoutingModule.ɵmod = ɵɵdefineNgModule({ type: AppRoutingModule });
    /** @nocollapse */ AppRoutingModule.ɵinj = ɵɵdefineInjector({ factory: function AppRoutingModule_Factory(t) { return new (t || AppRoutingModule)(); }, providers: [ProgrammaticalNavigationModule, AppRootProvider], imports: [[
                StoreModule.forFeature(APP_ROUTING_FEATURE_KEY, reducers$1),
                EffectsModule.forFeature([AppRoutingEffects]),
                LocationModule,
            ]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(AppRoutingModule, { imports: [StoreFeatureModule, EffectsFeatureModule, LocationModule] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/app_routing/views/router_link_directive_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class RouterLinkDirectiveContainer {
        /**
         * @param {?} store
         * @param {?} location
         * @param {?} appRootProvider
         */
        constructor(store, location, appRootProvider) {
            this.store = store;
            this.location = location;
            this.appRootProvider = appRootProvider;
            this.pathname = null;
        }
        /**
         * @param {?} event
         * @return {?}
         */
        handleClick(event) {
            if (!this.pathname || event.ctrlKey || event.metaKey) {
                return;
            }
            event.preventDefault();
            event.stopPropagation();
            this.store.dispatch(navigationRequested({
                pathname: this.pathname,
            }));
        }
        /**
         * @return {?}
         */
        get href() {
            if (!this.pathname)
                return null;
            return this.appRootProvider.getAbsPathnameWithAppRoot(this.location.getFullPathFromRouteOrNav({
                pathname: this.pathname,
            }));
        }
        /**
         * Add pathname to a route.
         *
         * Unlike \@angular/router's `routerLink`, the path is made sure to end  with
         * "/".
         * e.g., <a [routerLink]="['foo', someParamWithValueForBar]"> -> 'foo/bar/'.
         * @param {?} pathParts
         * @return {?}
         */
        set routerLink(pathParts) {
            if (typeof pathParts === 'string') {
                pathParts = [pathParts];
            }
            if (pathParts.length === 0) {
                throw new RangeError('routeLink should have proper path. Got nothing.');
            }
            // Append "/" suffix.
            /** @type {?} */
            const pathname = [...pathParts].join('/');
            this.pathname = pathname.endsWith('/') ? pathname : pathname + '/';
        }
    }
    RouterLinkDirectiveContainer.decorators = [
        { type: Directive, args: [{ selector: 'a[routerLink]' },] },
    ];
    /** @nocollapse */
    RouterLinkDirectiveContainer.ctorParameters = () => [
        { type: Store },
        { type: Location$1 },
        { type: AppRootProvider }
    ];
    RouterLinkDirectiveContainer.propDecorators = {
        handleClick: [{ type: HostListener, args: ['click', ['$event'],] }],
        href: [{ type: HostBinding, args: ['attr.href',] }],
        routerLink: [{ type: Input }]
    };
    /** @nocollapse */ RouterLinkDirectiveContainer.ɵfac = function RouterLinkDirectiveContainer_Factory(t) { return new (t || RouterLinkDirectiveContainer)(ɵɵdirectiveInject(Store), ɵɵdirectiveInject(Location$1), ɵɵdirectiveInject(AppRootProvider)); };
    /** @nocollapse */ RouterLinkDirectiveContainer.ɵdir = ɵɵdefineDirective({ type: RouterLinkDirectiveContainer, selectors: [["a", "routerLink", ""]], hostVars: 1, hostBindings: function RouterLinkDirectiveContainer_HostBindings(rf, ctx) { if (rf & 1) {
            ɵɵlistener("click", function RouterLinkDirectiveContainer_click_HostBindingHandler($event) { return ctx.handleClick($event); });
        } if (rf & 2) {
            ɵɵattribute("href", ctx.href, ɵɵsanitizeUrl);
        } }, inputs: { routerLink: "routerLink" } });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/app_routing/views/app_routing_view_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class AppRoutingViewModule {
    }
    AppRoutingViewModule.decorators = [
        { type: NgModule, args: [{
                    imports: [CommonModule, LocationModule, RouteRegistryModule],
                    exports: [RouterOutletContainer, RouterLinkDirectiveContainer],
                    declarations: [
                        RouterOutletContainer,
                        RouterOutletComponent,
                        RouterLinkDirectiveContainer,
                    ],
                    providers: [AppRootProvider],
                },] },
    ];
    /** @nocollapse */ AppRoutingViewModule.ɵmod = ɵɵdefineNgModule({ type: AppRoutingViewModule });
    /** @nocollapse */ AppRoutingViewModule.ɵinj = ɵɵdefineInjector({ factory: function AppRoutingViewModule_Factory(t) { return new (t || AppRoutingViewModule)(); }, providers: [AppRootProvider], imports: [[CommonModule, LocationModule, RouteRegistryModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(AppRoutingViewModule, { declarations: [RouterOutletContainer,
            RouterOutletComponent,
            RouterLinkDirectiveContainer], imports: [CommonModule, LocationModule, RouteRegistryModule], exports: [RouterOutletContainer, RouterLinkDirectiveContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/core/types.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @enum {string} */
    const PluginsListFailureCode = {
        UNKNOWN: "UNKNOWN",
        NOT_FOUND: "NOT_FOUND",
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/webapp_data_source/tb_server_data_source.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @param {?} enabledExperimentPluginIds
     * @return {?}
     */
    function getPluginsListingQueryParams(enabledExperimentPluginIds) {
        if (!enabledExperimentPluginIds.length) {
            return null;
        }
        /** @type {?} */
        const params = new URLSearchParams();
        for (const pluginId of enabledExperimentPluginIds) {
            params.append('experimentalPlugin', pluginId);
        }
        return params;
    }
    /**
     * @param {?} e
     * @return {?}
     */
    function handleError$1(e) {
        /** @type {?} */
        let status = PluginsListFailureCode.UNKNOWN;
        if (e instanceof HttpErrorResponse) {
            if (e.status === 404) {
                status = PluginsListFailureCode.NOT_FOUND;
            }
        }
        return throwError(new TBServerError(status));
    }
    class TBServerError {
        /**
         * @param {?} failureCode
         */
        constructor(failureCode) {
            this.failureCode = failureCode;
        }
    }
    class TBServerDataSource {
        /**
         * @param {?} http
         */
        constructor(http) {
            this.http = http;
            // TODO(soergel): implements WebappDataSource
            this.tfBackend = ((/** @type {?} */ (document.createElement('tf-backend')))).tf_backend;
        }
        /**
         * @param {?} enabledExperimentPluginIds
         * @return {?}
         */
        fetchPluginsListing(enabledExperimentPluginIds) {
            /** @type {?} */
            const params = getPluginsListingQueryParams(enabledExperimentPluginIds);
            /** @type {?} */
            const pathWithParams = params
                ? `data/plugins_listing?${params.toString()}`
                : 'data/plugins_listing';
            return this.http
                .get(pathWithParams)
                .pipe(catchError(handleError$1));
        }
        /**
         * @return {?}
         */
        fetchEnvironment() {
            // Make a request for data for the angular-specific portion of the app.
            /** @type {?} */
            const dataFetch = this.http.get('data/environment');
            // Force a data load for the polymer-specific portion of the app.
            // This leads to duplicate requests but hopefully the state is temporary until
            // we migrate everything from polymer to angular.
            /** @type {?} */
            const polymerEnvironmentRefresh = from(this.tfBackend.environmentStore.refresh());
            // Wait for both operations to complete and return the response from the
            // explicit http get call.
            return forkJoin([dataFetch, polymerEnvironmentRefresh]).pipe(map((/**
             * @param {?} __0
             * @return {?}
             */
            ([data]) => data)), catchError(handleError$1));
        }
    }
    TBServerDataSource.decorators = [
        { type: Injectable },
    ];
    /** @nocollapse */
    TBServerDataSource.ctorParameters = () => [
        { type: TBHttpClient }
    ];
    /** @nocollapse */ TBServerDataSource.ɵfac = function TBServerDataSource_Factory(t) { return new (t || TBServerDataSource)(ɵɵinject(TBHttpClient)); };
    /** @nocollapse */ TBServerDataSource.ɵprov = ɵɵdefineInjectable({ token: TBServerDataSource, factory: TBServerDataSource.ɵfac });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/webapp_data_source/tb_server_data_source_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class TBServerDataSourceModule {
    }
    TBServerDataSourceModule.decorators = [
        { type: NgModule, args: [{
                    imports: [TBHttpClientModule],
                    providers: [TBServerDataSource],
                },] },
    ];
    /** @nocollapse */ TBServerDataSourceModule.ɵmod = ɵɵdefineNgModule({ type: TBServerDataSourceModule });
    /** @nocollapse */ TBServerDataSourceModule.ɵinj = ɵɵdefineInjector({ factory: function TBServerDataSourceModule_Factory(t) { return new (t || TBServerDataSourceModule)(); }, providers: [TBServerDataSource], imports: [[TBHttpClientModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(TBServerDataSourceModule, { imports: [TBHttpClientModule] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/core/effects/core_effects.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class CoreEffects {
        /**
         * @param {?} actions$
         * @param {?} store
         * @param {?} webappDataSource
         */
        constructor(actions$, store, webappDataSource) {
            this.actions$ = actions$;
            this.store = store;
            this.webappDataSource = webappDataSource;
            // Ngrx assumes all Effect classes have properties that inherit from the base
            // JS Object. `tf_backend` does not, so we wrap it.
            this.tfBackend = {
                ref: ((/** @type {?} */ (document.createElement('tf-backend')))).tf_backend,
            };
            /**
               * Requires to be exported for JSCompiler. JSCompiler, otherwise,
               * think it is unused property and deadcode eliminate away.
               */
            /**
             * @export
             */
            this.fetchWebAppData$ = createEffect((/**
             * @return {?}
             */
            () => this.actions$.pipe(ofType(coreLoaded, reload, manualReload), withLatestFrom(this.store.select(getPluginsListLoaded), this.store.select(getEnabledExperimentalPlugins)), filter((/**
             * @param {?} __0
             * @return {?}
             */
            ([, { state }]) => state !== DataLoadState.LOADING)), tap((/**
             * @return {?}
             */
            () => this.store.dispatch(pluginsListingRequested()))), mergeMap((/**
             * @param {?} __0
             * @return {?}
             */
            ([, , enabledExperimentalPlugins]) => {
                return zip(this.webappDataSource.fetchPluginsListing(enabledExperimentalPlugins), this.fetchEnvironment(), this.refreshPolymerRuns()).pipe(map((/**
                 * @param {?} __0
                 * @return {?}
                 */
                ([plugins]) => {
                    this.store.dispatch(pluginsListingLoaded({ plugins }));
                })), catchError((/**
                 * @param {?} e
                 * @return {?}
                 */
                (e) => {
                    if (e instanceof TBServerError) {
                        this.store.dispatch(pluginsListingFailed({ failureCode: e.failureCode }));
                    }
                    else {
                        this.store.dispatch(pluginsListingFailed({
                            failureCode: PluginsListFailureCode.UNKNOWN,
                        }));
                    }
                    return EMPTY;
                })));
            })))), { dispatch: false });
            /**
             * HACK: COMPOSITE ACTION -- Fire `changePlugin` on first truthy value of
             * activePlugin on coreLoaded or pluginsListingLoaded.
             *
             * Rationale: most plugins want to be able to tell when it becomes active in
             * order to, for example, fetch necessary data. By firing changePlugin on
             * activePlugin first value set, we can prevent (1) other feature developer
             * from responding to values changes from the store (creates composite
             * actions) and (2) re-implement complex and brittle observable pattern.
             *
             * @export
             */
            this.dispatchChangePlugin$ = createEffect((/**
             * @return {?}
             */
            () => {
                return this.actions$.pipe(ofType(coreLoaded, pluginsListingLoaded), withLatestFrom(this.store.select(getActivePlugin)), map((/**
                 * @param {?} __0
                 * @return {?}
                 */
                ([, activePlugin]) => activePlugin)), distinctUntilChanged(), filter((/**
                 * @param {?} activePlugin
                 * @return {?}
                 */
                (activePlugin) => activePlugin !== null)), take(1), tap((/**
                 * @param {?} plugin
                 * @return {?}
                 */
                (plugin) => {
                    this.store.dispatch(changePlugin({ plugin: (/** @type {?} */ (plugin)) }));
                })));
            }), { dispatch: false });
        }
        /**
         * Force a data load for the Polymer-specific portion of the app. This leads
         * to duplicate requests but hopefully the state is temporary until we migrate
         * everything from Polymer to Angular.
         *
         * This is intentionally called in core/ rather than runs/ so that TB
         * embedders may use runs outside of the core dashboard page, without relying
         * on the Polymer runsStore.
         * @private
         * @return {?}
         */
        refreshPolymerRuns() {
            return from(this.tfBackend.ref.runsStore.refresh());
        }
        /**
         * @private
         * @return {?}
         */
        fetchEnvironment() {
            return this.webappDataSource.fetchEnvironment().pipe(tap((/**
             * @param {?} environment
             * @return {?}
             */
            (environment) => {
                this.store.dispatch(environmentLoaded({ environment }));
            })));
        }
    }
    CoreEffects.decorators = [
        { type: Injectable },
    ];
    /** @nocollapse */
    CoreEffects.ctorParameters = () => [
        { type: Actions },
        { type: Store },
        { type: TBServerDataSource }
    ];
    /** @nocollapse */ CoreEffects.ɵfac = function CoreEffects_Factory(t) { return new (t || CoreEffects)(ɵɵinject(Actions), ɵɵinject(Store), ɵɵinject(TBServerDataSource)); };
    /** @nocollapse */ CoreEffects.ɵprov = ɵɵdefineInjectable({ token: CoreEffects, factory: CoreEffects.ɵfac });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/core/store/core_initial_state_provider.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const CORE_STORE_CONFIG_TOKEN = new InjectionToken('Core Feature Config');
    /**
     * @param {?} deepLinker
     * @return {?}
     */
    function getConfig(deepLinker) {
        return {
            initialState: (/** @type {?} */ (Object.assign(Object.assign({}, initialState), { activePlugin: deepLinker.getPluginId() || null }))),
        };
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/core/core_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class CoreModule {
    }
    CoreModule.decorators = [
        { type: NgModule, args: [{
                    imports: [
                        EffectsModule.forFeature([CoreEffects]),
                        StoreModule.forFeature(CORE_FEATURE_KEY, reducers, CORE_STORE_CONFIG_TOKEN),
                        TBServerDataSourceModule,
                    ],
                    providers: [
                        {
                            provide: CORE_STORE_CONFIG_TOKEN,
                            deps: [HashDeepLinker],
                            useFactory: getConfig,
                        },
                    ],
                },] },
    ];
    /** @nocollapse */ CoreModule.ɵmod = ɵɵdefineNgModule({ type: CoreModule });
    /** @nocollapse */ CoreModule.ɵinj = ɵɵdefineInjector({ factory: function CoreModule_Factory(t) { return new (t || CoreModule)(); }, providers: [
            {
                provide: CORE_STORE_CONFIG_TOKEN,
                deps: [HashDeepLinker],
                useFactory: getConfig,
            },
        ], imports: [[
                EffectsModule.forFeature([CoreEffects]),
                StoreModule.forFeature(CORE_FEATURE_KEY, reducers, CORE_STORE_CONFIG_TOKEN),
                TBServerDataSourceModule,
            ]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(CoreModule, { imports: [EffectsFeatureModule, StoreFeatureModule, TBServerDataSourceModule] }); })();

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    const defaultExperiment = {
        id: DEFAULT_EXPERIMENT_ID,
        name: 'Default experiment',
        start_time: 0,
    };
    const initialDataState = {
        experimentMap: { [defaultExperiment.id]: defaultExperiment },
    };
    const dataReducer = createReducer(initialDataState);
    function reducers$2(state, action) {
        return combineReducers({ data: dataReducer })(state, action);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/experiments/experiments_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class ExperimentsModule {
    }
    ExperimentsModule.decorators = [
        { type: NgModule, args: [{
                    imports: [StoreModule.forFeature(EXPERIMENTS_FEATURE_KEY, reducers$2)],
                },] },
    ];
    /** @nocollapse */ ExperimentsModule.ɵmod = ɵɵdefineNgModule({ type: ExperimentsModule });
    /** @nocollapse */ ExperimentsModule.ɵinj = ɵɵdefineInjector({ factory: function ExperimentsModule_Factory(t) { return new (t || ExperimentsModule)(); }, imports: [[StoreModule.forFeature(EXPERIMENTS_FEATURE_KEY, reducers$2)]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(ExperimentsModule, { imports: [StoreFeatureModule] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/alert/alert_action_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    // While this token is not used outside, it must be exported so that stricter
    // build tools may discover it during compilation.
    /** @type {?} */
    const ACTION_TO_ALERT_PROVIDER = new InjectionToken('[Alert] Action-To-Alert Provider');
    /**
     * An NgModule that provides alert-producing actions. These action configs are
     * collected by AlertModule, which tracks application alerts.
     *
     * When the configured action fires, the AlertModule may respond.
     *
     * \@NgModule({
     *   imports: [
     *     AlertActionModule.registerAlertActions([
     *       {
     *         action: fetchKeysFailed,
     *         alertFromAction: () => {localizedMessage: "Keys failed to fetch."},
     *       },
     *       {
     *         action: greenButtonClicked,
     *         alertFromAction: (actionPayload) => {
     *           if (!actionPayload.wasButtonEnabled) {
     *             return {localizedMessage: "Green button failed."};
     *           }
     *           return null;
     *         }
     *       }
     *     ]),
     *   ],
     * })
     */
    class AlertActionModule {
        /**
         * @param {?} providers
         */
        constructor(providers) {
            /**
             * Map from action creator type to transformer function.
             */
            this.providers = new Map();
            for (const configs of providers || []) {
                for (const config of configs) {
                    if (this.providers.has(config.actionCreator.type)) {
                        throw new RangeError(`"${config.actionCreator.type}" is already registered for alerts.` +
                            ' Multiple alerts for the same action is not allowed.');
                    }
                    this.providers.set(config.actionCreator.type, config.alertFromAction);
                }
            }
        }
        /**
         * @param {?} action
         * @return {?}
         */
        getAlertFromAction(action) {
            /** @type {?} */
            const lambda = this.providers.get(action.type);
            if (!lambda) {
                return null;
            }
            return lambda(action);
        }
        /**
         * @param {?} providerFactory
         * @return {?}
         */
        static registerAlertActions(providerFactory) {
            return {
                ngModule: AlertActionModule,
                providers: [
                    {
                        provide: ACTION_TO_ALERT_PROVIDER,
                        multi: true,
                        useFactory: providerFactory,
                    },
                ],
            };
        }
    }
    AlertActionModule.decorators = [
        { type: NgModule, args: [{},] },
    ];
    /** @nocollapse */
    AlertActionModule.ctorParameters = () => [
        { type: Array, decorators: [{ type: Optional }, { type: Inject, args: [ACTION_TO_ALERT_PROVIDER,] }] }
    ];
    /** @nocollapse */ AlertActionModule.ɵmod = ɵɵdefineNgModule({ type: AlertActionModule });
    /** @nocollapse */ AlertActionModule.ɵinj = ɵɵdefineInjector({ factory: function AlertActionModule_Factory(t) { return new (t || AlertActionModule)(ɵɵinject(ACTION_TO_ALERT_PROVIDER, 8)); } });

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * Fires when an alert is to be reported.
     */
    const alertReported = createAction('[Alert] Alert Reported', props());

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/alert/effects/index.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class AlertEffects {
        /**
         * @param {?} actions$
         * @param {?} store
         * @param {?} alertActionModule
         */
        constructor(actions$, store, alertActionModule) {
            this.actions$ = actions$;
            this.store = store;
            this.alertActionModule = alertActionModule;
            /**
             * @export
             */
            this.reportRegisteredActionAlerts$ = createEffect((/**
             * @return {?}
             */
            () => {
                return this.actions$.pipe(tap((/**
                 * @param {?} action
                 * @return {?}
                 */
                (action) => {
                    /** @type {?} */
                    const alertInfo = this.alertActionModule.getAlertFromAction(action);
                    if (alertInfo) {
                        this.store.dispatch(alertReported(alertInfo));
                    }
                })));
            }), { dispatch: false });
        }
    }
    AlertEffects.decorators = [
        { type: Injectable },
    ];
    /** @nocollapse */
    AlertEffects.ctorParameters = () => [
        { type: Actions },
        { type: Store },
        { type: AlertActionModule }
    ];
    /** @nocollapse */ AlertEffects.ɵfac = function AlertEffects_Factory(t) { return new (t || AlertEffects)(ɵɵinject(Actions), ɵɵinject(Store), ɵɵinject(AlertActionModule)); };
    /** @nocollapse */ AlertEffects.ɵprov = ɵɵdefineInjectable({ token: AlertEffects, factory: AlertEffects.ɵfac });

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    const initialState$2 = {
        latestAlert: null,
    };
    const reducer$2 = createReducer(initialState$2, on(alertReported, (state, { localizedMessage }) => {
        return Object.assign(Object.assign({}, state), { latestAlert: { localizedMessage, created: Date.now() } });
    }));
    function reducers$3(state, action) {
        return reducer$2(state, action);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/alert/views/alert_snackbar_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Provides the 'alert snackbar' view.
     */
    class AlertSnackbarModule {
    }
    AlertSnackbarModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [AlertSnackbarContainer, AlertDisplaySnackbarContainer],
                    exports: [AlertSnackbarContainer],
                    imports: [CommonModule, MatButtonModule, MatSnackBarModule],
                    entryComponents: [
                        // Required for non-Ivy Angular apps.
                        AlertDisplaySnackbarContainer,
                    ],
                },] },
    ];
    /** @nocollapse */ AlertSnackbarModule.ɵmod = ɵɵdefineNgModule({ type: AlertSnackbarModule });
    /** @nocollapse */ AlertSnackbarModule.ɵinj = ɵɵdefineInjector({ factory: function AlertSnackbarModule_Factory(t) { return new (t || AlertSnackbarModule)(); }, imports: [[CommonModule, MatButtonModule, MatSnackBarModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(AlertSnackbarModule, { declarations: [AlertSnackbarContainer, AlertDisplaySnackbarContainer], imports: [CommonModule, MatButtonModule, MatSnackBarModule], exports: [AlertSnackbarContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/alert/alert_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class AlertModule {
    }
    AlertModule.decorators = [
        { type: NgModule, args: [{
                    imports: [
                        AlertActionModule,
                        AlertSnackbarModule,
                        StoreModule.forFeature(ALERT_FEATURE_KEY, reducers$3),
                        EffectsModule.forFeature([AlertEffects]),
                    ],
                },] },
    ];
    /** @nocollapse */ AlertModule.ɵmod = ɵɵdefineNgModule({ type: AlertModule });
    /** @nocollapse */ AlertModule.ɵinj = ɵɵdefineInjector({ factory: function AlertModule_Factory(t) { return new (t || AlertModule)(); }, imports: [[
                AlertActionModule,
                AlertSnackbarModule,
                StoreModule.forFeature(ALERT_FEATURE_KEY, reducers$3),
                EffectsModule.forFeature([AlertEffects]),
            ]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(AlertModule, { imports: [AlertActionModule,
            AlertSnackbarModule, StoreFeatureModule, EffectsFeatureModule] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/deeplink/deeplink_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class DeepLinkModule {
    }
    DeepLinkModule.decorators = [
        { type: NgModule, args: [{
                    providers: [HashDeepLinker],
                },] },
    ];
    /** @nocollapse */ DeepLinkModule.ɵmod = ɵɵdefineNgModule({ type: DeepLinkModule });
    /** @nocollapse */ DeepLinkModule.ɵinj = ɵɵdefineInjector({ factory: function DeepLinkModule_Factory(t) { return new (t || DeepLinkModule)(); }, providers: [HashDeepLinker] });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/core/views/hash_storage_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class HashStorageModule {
    }
    HashStorageModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [HashStorageContainer, HashStorageComponent],
                    exports: [HashStorageContainer],
                    imports: [CommonModule, DeepLinkModule],
                },] },
    ];
    /** @nocollapse */ HashStorageModule.ɵmod = ɵɵdefineNgModule({ type: HashStorageModule });
    /** @nocollapse */ HashStorageModule.ɵinj = ɵɵdefineInjector({ factory: function HashStorageModule_Factory(t) { return new (t || HashStorageModule)(); }, imports: [[CommonModule, DeepLinkModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(HashStorageModule, { declarations: [HashStorageContainer, HashStorageComponent], imports: [CommonModule, DeepLinkModule], exports: [HashStorageContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/core/views/page_title_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class PageTitleModule {
    }
    PageTitleModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [PageTitleContainer, PageTitleComponent],
                    exports: [PageTitleContainer],
                    imports: [CommonModule],
                },] },
    ];
    /** @nocollapse */ PageTitleModule.ɵmod = ɵɵdefineNgModule({ type: PageTitleModule });
    /** @nocollapse */ PageTitleModule.ɵinj = ɵɵdefineInjector({ factory: function PageTitleModule_Factory(t) { return new (t || PageTitleModule)(); }, imports: [[CommonModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(PageTitleModule, { declarations: [PageTitleContainer, PageTitleComponent], imports: [CommonModule], exports: [PageTitleContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/webapp_data_source/tb_feature_flag_data_source_types.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @abstract
     */
    class TBFeatureFlagDataSource {
    }
    TBFeatureFlagDataSource.decorators = [
        { type: Injectable },
    ];
    /** @nocollapse */ TBFeatureFlagDataSource.ɵfac = function TBFeatureFlagDataSource_Factory(t) { return new (t || TBFeatureFlagDataSource)(); };
    /** @nocollapse */ TBFeatureFlagDataSource.ɵprov = ɵɵdefineInjectable({ token: TBFeatureFlagDataSource, factory: TBFeatureFlagDataSource.ɵfac });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/webapp_data_source/tb_feature_flag_data_source.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Save the initial URL query params, before the AppRoutingEffects initialize.
     * @type {?}
     */
    const initialURLSearchParams = new URLSearchParams(window.location.search);
    /** @type {?} */
    const util$1 = {
        /**
         * @return {?}
         */
        getParams() {
            return initialURLSearchParams;
        },
    };
    class QueryParamsFeatureFlagDataSource extends TBFeatureFlagDataSource {
        /**
         * @return {?}
         */
        getFeatures() {
            /** @type {?} */
            const params = util$1.getParams();
            return {
                enabledExperimentalPlugins: params.getAll('experimentalPlugin'),
                inColab: params.get('tensorboardColab') === 'true',
            };
        }
    }
    QueryParamsFeatureFlagDataSource.decorators = [
        { type: Injectable },
    ];
    /** @nocollapse */ QueryParamsFeatureFlagDataSource.ɵfac = function QueryParamsFeatureFlagDataSource_Factory(t) { return ɵQueryParamsFeatureFlagDataSource_BaseFactory(t || QueryParamsFeatureFlagDataSource); };
    /** @nocollapse */ QueryParamsFeatureFlagDataSource.ɵprov = ɵɵdefineInjectable({ token: QueryParamsFeatureFlagDataSource, factory: QueryParamsFeatureFlagDataSource.ɵfac });
    const ɵQueryParamsFeatureFlagDataSource_BaseFactory = ɵɵgetInheritedFactory(QueryParamsFeatureFlagDataSource);

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/webapp_data_source/tb_feature_flag_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    class TBFeatureFlagModule {
    }
    TBFeatureFlagModule.decorators = [
        { type: NgModule, args: [{
                    providers: [
                        {
                            provide: TBFeatureFlagDataSource,
                            useClass: QueryParamsFeatureFlagDataSource,
                        },
                    ],
                },] },
    ];
    /** @nocollapse */ TBFeatureFlagModule.ɵmod = ɵɵdefineNgModule({ type: TBFeatureFlagModule });
    /** @nocollapse */ TBFeatureFlagModule.ɵinj = ɵɵdefineInjector({ factory: function TBFeatureFlagModule_Factory(t) { return new (t || TBFeatureFlagModule)(); }, providers: [
            {
                provide: TBFeatureFlagDataSource,
                useClass: QueryParamsFeatureFlagDataSource,
            },
        ] });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/feature_flag/actions/feature_flag_actions.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const featuresLoaded = createAction('[FEATURE FLAG] Features Loaded', props());

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/feature_flag/store/feature_flag_reducers.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const initialState$3 = {
        isFeatureFlagsLoaded: false,
        features: {
            enabledExperimentalPlugins: [],
            inColab: false,
        },
    };
    /** @type {?} */
    const reducer$3 = createReducer(initialState$3, on(featuresLoaded, (/**
     * @param {?} state
     * @param {?} __1
     * @return {?}
     */
    (state, { features }) => {
        return Object.assign(Object.assign({}, state), { isFeatureFlagsLoaded: true, features: Object.assign(Object.assign({}, state.features), features) });
    })));
    /**
     * @param {?} state
     * @param {?} action
     * @return {?}
     */
    function reducers$4(state, action) {
        return reducer$3(state, action);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/feature_flag/effects/feature_flag_effects.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /** @type {?} */
    const effectsInitialized = createAction('[FEATURE FLAG] Effects Init');
    class FeatureFlagEffects {
        /**
         * @param {?} actions$
         * @param {?} dataSource
         */
        constructor(actions$, dataSource) {
            this.actions$ = actions$;
            this.dataSource = dataSource;
            /**
             * @export
             */
            this.getFeatureFlags$ = createEffect((/**
             * @return {?}
             */
            () => this.actions$.pipe(ofType(effectsInitialized), map((/**
             * @return {?}
             */
            () => {
                /** @type {?} */
                const features = this.dataSource.getFeatures();
                return featuresLoaded({ features });
            })))));
        }
        /**
         * @export
         * @return {?}
         */
        ngrxOnInitEffects() {
            return effectsInitialized();
        }
    }
    FeatureFlagEffects.decorators = [
        { type: Injectable },
    ];
    /** @nocollapse */
    FeatureFlagEffects.ctorParameters = () => [
        { type: Actions },
        { type: TBFeatureFlagDataSource }
    ];
    /** @nocollapse */ FeatureFlagEffects.ɵfac = function FeatureFlagEffects_Factory(t) { return new (t || FeatureFlagEffects)(ɵɵinject(Actions), ɵɵinject(TBFeatureFlagDataSource)); };
    /** @nocollapse */ FeatureFlagEffects.ɵprov = ɵɵdefineInjectable({ token: FeatureFlagEffects, factory: FeatureFlagEffects.ɵfac });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/feature_flag/feature_flag_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    class FeatureFlagModule {
    }
    FeatureFlagModule.decorators = [
        { type: NgModule, args: [{
                    imports: [
                        TBFeatureFlagModule,
                        StoreModule.forFeature(FEAUTURE_FLAG_FEATURE_KEY, reducers$4),
                        EffectsModule.forFeature([FeatureFlagEffects]),
                    ],
                },] },
    ];
    /** @nocollapse */ FeatureFlagModule.ɵmod = ɵɵdefineNgModule({ type: FeatureFlagModule });
    /** @nocollapse */ FeatureFlagModule.ɵinj = ɵɵdefineInjector({ factory: function FeatureFlagModule_Factory(t) { return new (t || FeatureFlagModule)(); }, imports: [[
                TBFeatureFlagModule,
                StoreModule.forFeature(FEAUTURE_FLAG_FEATURE_KEY, reducers$4),
                EffectsModule.forFeature([FeatureFlagEffects]),
            ]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(FeatureFlagModule, { imports: [TBFeatureFlagModule, StoreFeatureModule, EffectsFeatureModule] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/settings/settings_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class SettingsModule {
    }
    SettingsModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [
                        SettingsButtonComponent,
                        SettingsDialogComponent,
                        SettingsPolymerInteropContainer,
                    ],
                    exports: [SettingsButtonComponent, SettingsPolymerInteropContainer],
                    entryComponents: [SettingsDialogComponent],
                    imports: [
                        CommonModule,
                        FormsModule,
                        ReactiveFormsModule,
                        MatButtonModule,
                        MatCheckboxModule,
                        MatDialogModule,
                        MatIconModule,
                        MatInputModule,
                    ],
                },] },
    ];
    /** @nocollapse */ SettingsModule.ɵmod = ɵɵdefineNgModule({ type: SettingsModule });
    /** @nocollapse */ SettingsModule.ɵinj = ɵɵdefineInjector({ factory: function SettingsModule_Factory(t) { return new (t || SettingsModule)(); }, imports: [[
                CommonModule,
                FormsModule,
                ReactiveFormsModule,
                MatButtonModule,
                MatCheckboxModule,
                MatDialogModule,
                MatIconModule,
                MatInputModule,
            ]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(SettingsModule, { declarations: [SettingsButtonComponent,
            SettingsDialogComponent,
            SettingsPolymerInteropContainer], imports: [CommonModule,
            FormsModule,
            ReactiveFormsModule,
            MatButtonModule,
            MatCheckboxModule,
            MatDialogModule,
            MatIconModule,
            MatInputModule], exports: [SettingsButtonComponent, SettingsPolymerInteropContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/tbdev_upload/tbdev_upload_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class TbdevUploadModule {
    }
    TbdevUploadModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [
                        TbdevUploadButtonComponent,
                        TbdevUploadDialogComponent,
                        TbdevUploadDialogContainer,
                    ],
                    exports: [TbdevUploadButtonComponent],
                    entryComponents: [TbdevUploadDialogContainer],
                    imports: [
                        ClipboardModule,
                        CommonModule,
                        MatButtonModule,
                        MatDialogModule,
                        MatIconModule,
                    ],
                    providers: [{ provide: 'window', useValue: window }],
                },] },
    ];
    /** @nocollapse */ TbdevUploadModule.ɵmod = ɵɵdefineNgModule({ type: TbdevUploadModule });
    /** @nocollapse */ TbdevUploadModule.ɵinj = ɵɵdefineInjector({ factory: function TbdevUploadModule_Factory(t) { return new (t || TbdevUploadModule)(); }, providers: [{ provide: 'window', useValue: window }], imports: [[
                ClipboardModule,
                CommonModule,
                MatButtonModule,
                MatDialogModule,
                MatIconModule,
            ]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(TbdevUploadModule, { declarations: [TbdevUploadButtonComponent,
            TbdevUploadDialogComponent,
            TbdevUploadDialogContainer], imports: [ClipboardModule,
            CommonModule,
            MatButtonModule,
            MatDialogModule,
            MatIconModule], exports: [TbdevUploadButtonComponent] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/header/header_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class HeaderModule {
    }
    HeaderModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [
                        HeaderComponent,
                        PluginSelectorComponent,
                        PluginSelectorContainer,
                        ReloadContainer,
                    ],
                    exports: [HeaderComponent, PluginSelectorContainer, ReloadContainer],
                    providers: [],
                    imports: [
                        MatButtonModule,
                        MatIconModule,
                        MatTabsModule,
                        MatToolbarModule,
                        MatSelectModule,
                        CommonModule,
                        CoreModule,
                        SettingsModule,
                        TbdevUploadModule,
                    ],
                },] },
    ];
    /** @nocollapse */ HeaderModule.ɵmod = ɵɵdefineNgModule({ type: HeaderModule });
    /** @nocollapse */ HeaderModule.ɵinj = ɵɵdefineInjector({ factory: function HeaderModule_Factory(t) { return new (t || HeaderModule)(); }, providers: [], imports: [[
                MatButtonModule,
                MatIconModule,
                MatTabsModule,
                MatToolbarModule,
                MatSelectModule,
                CommonModule,
                CoreModule,
                SettingsModule,
                TbdevUploadModule,
            ]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(HeaderModule, { declarations: [HeaderComponent,
            PluginSelectorComponent,
            PluginSelectorContainer,
            ReloadContainer], imports: [MatButtonModule,
            MatIconModule,
            MatTabsModule,
            MatToolbarModule,
            MatSelectModule,
            CommonModule,
            CoreModule,
            SettingsModule,
            TbdevUploadModule], exports: [HeaderComponent, PluginSelectorContainer, ReloadContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/mat_icon_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const SVG_PATH = './icon_bundle.svg';
    class MatIconModule$1 {
        /**
         * @param {?} domSanitizer
         * @param {?} iconRegistry
         */
        constructor(domSanitizer, iconRegistry) {
            /** @type {?} */
            const url = domSanitizer.bypassSecurityTrustResourceUrl(SVG_PATH);
            iconRegistry.addSvgIconSet(url);
        }
    }
    MatIconModule$1.decorators = [
        { type: NgModule, args: [{
                    imports: [MatIconModule],
                },] },
    ];
    /** @nocollapse */
    MatIconModule$1.ctorParameters = () => [
        { type: DomSanitizer },
        { type: MatIconRegistry }
    ];
    /** @nocollapse */ MatIconModule$1.ɵmod = ɵɵdefineNgModule({ type: MatIconModule$1 });
    /** @nocollapse */ MatIconModule$1.ɵinj = ɵɵdefineInjector({ factory: function MatIconModule_Factory(t) { return new (t || MatIconModule$1)(ɵɵinject(DomSanitizer), ɵɵinject(MatIconRegistry)); }, imports: [[MatIconModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(MatIconModule$1, { imports: [MatIconModule] }); })();

    /* Copyright 2019 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    var LoadingMechanismType;
    (function (LoadingMechanismType) {
        LoadingMechanismType["CUSTOM_ELEMENT"] = "CUSTOM_ELEMENT";
        LoadingMechanismType["IFRAME"] = "IFRAME";
        LoadingMechanismType["NG_COMPONENT"] = "NG_COMPONENT";
        LoadingMechanismType["NONE"] = "NONE";
    })(LoadingMechanismType || (LoadingMechanismType = {}));

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/plugin_registry_types.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    // This DI token must not be defined in the same file as PluginRegistryModule
    // or else its constructor injection causes Angular's AOT compiler to think
    // there is a circular dependency.
    /** @type {?} */
    const PLUGIN_CONFIG_TOKEN = new InjectionToken('[plugins] Plugin registry config');

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/plugin_registry_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const pluginNameToComponent = new Map();
    class PluginRegistryModule {
        /**
         * @param {?} configs
         */
        constructor(configs) {
            if (!configs) {
                return;
            }
            /** @type {?} */
            const pluginNames = new Set(configs.map((/**
             * @param {?} config
             * @return {?}
             */
            (config) => config.pluginName)));
            console.assert(pluginNames.size === configs.length, 'Cannot register the same plugin multiple times.');
            for (const config of configs) {
                const { pluginName, componentClass } = config;
                pluginNameToComponent.set(pluginName, (/** @type {?} */ (componentClass)));
            }
        }
        /**
         * An NgModule that declares a Component can call this method to register the
         * Component to be rendered when a plugin is requested by name. Modules
         * register by adding the following configuration:
         *
         * \@NgModule({
         *   imports: [
         *     PluginRegistryModule.forPlugin('scalars', ScalarsDashboard)
         *   ],
         *   entryComponents: [ScalarsDashboard]
         * })
         * @param {?} pluginName
         * @param {?} componentClass
         * @return {?}
         */
        static forPlugin(pluginName, componentClass) {
            return {
                ngModule: PluginRegistryModule,
                providers: [
                    {
                        provide: PLUGIN_CONFIG_TOKEN,
                        multi: true,
                        useValue: { pluginName, componentClass },
                    },
                ],
            };
        }
        /**
         * @param {?} pluginName
         * @return {?}
         */
        getComponent(pluginName) {
            return pluginNameToComponent.get(pluginName) || null;
        }
    }
    PluginRegistryModule.decorators = [
        { type: NgModule, args: [{},] },
    ];
    /** @nocollapse */
    PluginRegistryModule.ctorParameters = () => [
        { type: Array, decorators: [{ type: Optional }, { type: Inject, args: [PLUGIN_CONFIG_TOKEN,] }] }
    ];
    /** @nocollapse */ PluginRegistryModule.ɵmod = ɵɵdefineNgModule({ type: PluginRegistryModule });
    /** @nocollapse */ PluginRegistryModule.ɵinj = ɵɵdefineInjector({ factory: function PluginRegistryModule_Factory(t) { return new (t || PluginRegistryModule)(ɵɵinject(PLUGIN_CONFIG_TOKEN, 8)); } });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/plugins_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    const _c0$a = ["pluginContainer"];
    const _c1$8 = ["ngPluginContainer"];
    function PluginsComponent_div_4_ng_container_2_ng_container_1_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementContainer(0);
    } }
    function PluginsComponent_div_4_ng_container_2_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementContainerStart(0);
        ɵɵtemplate(1, PluginsComponent_div_4_ng_container_2_ng_container_1_Template, 1, 0, "ng-container", 9);
        ɵɵelementContainerEnd();
    } if (rf & 2) {
        const ctx_r7 = ɵɵnextContext(2);
        const _r3 = ɵɵreference(6);
        ɵɵadvance(1);
        ɵɵproperty("ngTemplateOutlet", ctx_r7.environmentFailureNotFoundTemplate ? ctx_r7.environmentFailureNotFoundTemplate : _r3);
    } }
    function PluginsComponent_div_4_ng_container_3_ng_container_1_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementContainer(0);
    } }
    function PluginsComponent_div_4_ng_container_3_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementContainerStart(0);
        ɵɵtemplate(1, PluginsComponent_div_4_ng_container_3_ng_container_1_Template, 1, 0, "ng-container", 9);
        ɵɵelementContainerEnd();
    } if (rf & 2) {
        const ctx_r8 = ɵɵnextContext(2);
        const _r3 = ɵɵreference(6);
        ɵɵadvance(1);
        ɵɵproperty("ngTemplateOutlet", ctx_r8.environmentFailureUnknownTemplate ? ctx_r8.environmentFailureUnknownTemplate : _r3);
    } }
    function PluginsComponent_div_4_ng_container_4_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementContainerStart(0);
        ɵɵelementStart(1, "h3", 10);
        ɵɵtext(2, " There\u2019s no dashboard by the name of \u201C");
        ɵɵelementStart(3, "code");
        ɵɵtext(4);
        ɵɵelementEnd();
        ɵɵtext(5, "\u201D. ");
        ɵɵelementEnd();
        ɵɵelementStart(6, "p");
        ɵɵtext(7, "You can select a dashboard from the list above.");
        ɵɵelementEnd();
        ɵɵelementStart(8, "p");
        ɵɵelementContainer(9, 11);
        ɵɵelementEnd();
        ɵɵelementContainerEnd();
    } if (rf & 2) {
        const ctx_r9 = ɵɵnextContext(2);
        const _r5 = ɵɵreference(8);
        ɵɵadvance(4);
        ɵɵtextInterpolate(ctx_r9.activePluginId);
        ɵɵadvance(5);
        ɵɵproperty("ngTemplateOutlet", _r5);
    } }
    function PluginsComponent_div_4_ng_container_5_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementContainerStart(0);
        ɵɵelementStart(1, "h3", 12);
        ɵɵtext(2, " No dashboards are active for the current data set. ");
        ɵɵelementEnd();
        ɵɵelementStart(3, "p");
        ɵɵtext(4, "Probable causes:");
        ɵɵelementEnd();
        ɵɵelementStart(5, "ul");
        ɵɵelementStart(6, "li");
        ɵɵtext(7, "You haven\u2019t written any data to your event files.");
        ɵɵelementEnd();
        ɵɵelementStart(8, "li");
        ɵɵtext(9, "TensorBoard can\u2019t find your event files.");
        ɵɵelementEnd();
        ɵɵelementEnd();
        ɵɵtext(10, " If you\u2019re new to using TensorBoard, and want to find out how to add data and set up your event files, check out the ");
        ɵɵelementStart(11, "a", 13);
        ɵɵtext(12, "README");
        ɵɵelementEnd();
        ɵɵtext(13, " and perhaps the ");
        ɵɵelementStart(14, "a", 14);
        ɵɵtext(15, "TensorBoard tutorial");
        ɵɵelementEnd();
        ɵɵtext(16, ". ");
        ɵɵelementStart(17, "p");
        ɵɵtext(18, " If you think TensorBoard is configured properly, please see ");
        ɵɵelementStart(19, "a", 15);
        ɵɵtext(20, "the section of the README devoted to missing data problems");
        ɵɵelementEnd();
        ɵɵtext(21, " and consider filing an issue on GitHub. ");
        ɵɵelementEnd();
        ɵɵelementStart(22, "p");
        ɵɵelementContainer(23, 11);
        ɵɵelementEnd();
        ɵɵelementContainerEnd();
    } if (rf & 2) {
        ɵɵnextContext(2);
        const _r5 = ɵɵreference(8);
        ɵɵadvance(23);
        ɵɵproperty("ngTemplateOutlet", _r5);
    } }
    function PluginsComponent_div_4_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 6);
        ɵɵelementStart(1, "div", 7);
        ɵɵtemplate(2, PluginsComponent_div_4_ng_container_2_Template, 2, 1, "ng-container", 8);
        ɵɵtemplate(3, PluginsComponent_div_4_ng_container_3_Template, 2, 1, "ng-container", 8);
        ɵɵtemplate(4, PluginsComponent_div_4_ng_container_4_Template, 10, 2, "ng-container", 8);
        ɵɵtemplate(5, PluginsComponent_div_4_ng_container_5_Template, 24, 1, "ng-container", 8);
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r2 = ɵɵnextContext();
        ɵɵproperty("ngSwitch", ctx_r2.pluginLoadState);
        ɵɵadvance(2);
        ɵɵproperty("ngSwitchCase", ctx_r2.PluginLoadState.ENVIRONMENT_FAILURE_NOT_FOUND);
        ɵɵadvance(1);
        ɵɵproperty("ngSwitchCase", ctx_r2.PluginLoadState.ENVIRONMENT_FAILURE_UNKNOWN);
        ɵɵadvance(1);
        ɵɵproperty("ngSwitchCase", ctx_r2.PluginLoadState.UNKNOWN_PLUGIN_ID);
        ɵɵadvance(1);
        ɵɵproperty("ngSwitchCase", ctx_r2.PluginLoadState.NO_ENABLED_PLUGINS);
    } }
    function PluginsComponent_ng_template_5_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "h3", 16);
        ɵɵtext(1, "Data could not be loaded.");
        ɵɵelementEnd();
        ɵɵelementStart(2, "p");
        ɵɵtext(3, "The TensorBoard server may be down or inaccessible.");
        ɵɵelementEnd();
        ɵɵelementStart(4, "p");
        ɵɵelementContainer(5, 11);
        ɵɵelementEnd();
    } if (rf & 2) {
        ɵɵnextContext();
        const _r5 = ɵɵreference(8);
        ɵɵadvance(5);
        ɵɵproperty("ngTemplateOutlet", _r5);
    } }
    function PluginsComponent_ng_template_7_p_3_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "p", 19);
        ɵɵelementStart(1, "i");
        ɵɵtext(2, "Log directory: ");
        ɵɵelementStart(3, "span");
        ɵɵtext(4);
        ɵɵelementEnd();
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r13 = ɵɵnextContext(2);
        ɵɵadvance(4);
        ɵɵtextInterpolate(ctx_r13.dataLocation);
    } }
    function PluginsComponent_ng_template_7_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "span", 17);
        ɵɵtext(1);
        ɵɵpipe(2, "date");
        ɵɵelementEnd();
        ɵɵtemplate(3, PluginsComponent_ng_template_7_p_3_Template, 5, 1, "p", 18);
    } if (rf & 2) {
        const ctx_r6 = ɵɵnextContext();
        ɵɵadvance(1);
        ɵɵtextInterpolate1("Last reload: ", ɵɵpipeBind2(2, 2, ctx_r6.lastUpdated, "medium"), "");
        ɵɵadvance(2);
        ɵɵproperty("ngIf", ctx_r6.dataLocation);
    } }
    /** @enum {number} */
    const PluginLoadState = {
        ENVIRONMENT_FAILURE_NOT_FOUND: 0,
        ENVIRONMENT_FAILURE_UNKNOWN: 1,
        NO_ENABLED_PLUGINS: 2,
        UNKNOWN_PLUGIN_ID: 3,
        LOADED: 4,
        LOADING: 5,
    };
    PluginLoadState[PluginLoadState.ENVIRONMENT_FAILURE_NOT_FOUND] = 'ENVIRONMENT_FAILURE_NOT_FOUND';
    PluginLoadState[PluginLoadState.ENVIRONMENT_FAILURE_UNKNOWN] = 'ENVIRONMENT_FAILURE_UNKNOWN';
    PluginLoadState[PluginLoadState.NO_ENABLED_PLUGINS] = 'NO_ENABLED_PLUGINS';
    PluginLoadState[PluginLoadState.UNKNOWN_PLUGIN_ID] = 'UNKNOWN_PLUGIN_ID';
    PluginLoadState[PluginLoadState.LOADED] = 'LOADED';
    PluginLoadState[PluginLoadState.LOADING] = 'LOADING';
    class PluginsComponent {
        /**
         * @param {?} componentFactoryResolver
         * @param {?} pluginRegistry
         */
        constructor(componentFactoryResolver, pluginRegistry) {
            this.componentFactoryResolver = componentFactoryResolver;
            this.pluginRegistry = pluginRegistry;
            this.experimentPluginHostLib = (/** @type {?} */ (document.createElement('tf-experimental-plugin-host-lib')));
            this.PluginLoadState = PluginLoadState;
            this.LoadingMechanismType = LoadingMechanismType;
            this.pluginInstances = new Map();
        }
        /**
         * @param {?} change
         * @return {?}
         */
        ngOnChanges(change) {
            /** @type {?} */
            const shouldCreatePlugin = Boolean(this.activeKnownPlugin &&
                !this.pluginInstances.has(this.activeKnownPlugin.id));
            if (change['activeKnownPlugin'] && this.activeKnownPlugin) {
                /** @type {?} */
                const prevActiveKnownPlugin = change['activeKnownPlugin'].previousValue;
                if (prevActiveKnownPlugin &&
                    prevActiveKnownPlugin.id !== this.activeKnownPlugin.id) {
                    this.hidePlugin(prevActiveKnownPlugin);
                }
                if (shouldCreatePlugin) {
                    /** @type {?} */
                    const pluginElement = this.createPlugin(this.activeKnownPlugin);
                    if (pluginElement) {
                        this.pluginInstances.set(this.activeKnownPlugin.id, pluginElement);
                    }
                }
                else {
                    this.showPlugin(this.activeKnownPlugin);
                }
            }
            if (this.activeKnownPlugin &&
                (shouldCreatePlugin || change['lastUpdated'])) {
                this.reload(this.activeKnownPlugin, shouldCreatePlugin);
            }
        }
        /**
         * @private
         * @param {?} plugin
         * @return {?}
         */
        hidePlugin(plugin) {
            // In case the active plugin does not have a DOM, for example, core plugin, the
            // instance can be falsy.
            if (!this.pluginInstances.has(plugin.id))
                return;
            /** @type {?} */
            const instance = (/** @type {?} */ (this.pluginInstances.get(plugin.id)));
            Object.assign(instance.style, {
                maxHeight: 0,
                overflow: 'hidden',
                /**
                 * We further make containers invisible. Some elements may anchor to
                 * the viewport instead of the container, in which case setting the max
                 * height here to 0 will not hide them.
                 *
                 */
                visibility: 'hidden',
                position: 'absolute',
            });
        }
        /**
         * @private
         * @param {?} plugin
         * @return {?}
         */
        showPlugin(plugin) {
            // In case the active plugin does not have a DOM, for example, core plugin, the
            // instance can be falsy.
            if (!this.pluginInstances.has(plugin.id))
                return;
            /** @type {?} */
            const instance = (/** @type {?} */ (this.pluginInstances.get(plugin.id)));
            Object.assign(instance.style, {
                maxHeight: null,
                overflow: null,
                visibility: null,
                position: null,
            });
        }
        /**
         * @private
         * @param {?} plugin
         * @return {?}
         */
        createPlugin(plugin) {
            /** @type {?} */
            let pluginElement = null;
            switch (plugin.loading_mechanism.type) {
                case LoadingMechanismType.CUSTOM_ELEMENT: {
                    /** @type {?} */
                    const customElementPlugin = (/** @type {?} */ (plugin.loading_mechanism));
                    pluginElement = document.createElement(customElementPlugin.element_name);
                    ((/** @type {?} */ (pluginElement))).reloadOnReady = false;
                    this.pluginsContainer.nativeElement.appendChild(pluginElement);
                    break;
                }
                case LoadingMechanismType.IFRAME: {
                    pluginElement = document.createElement('iframe');
                    // Ideally should use the DOMSanitizer but it is not usable in TypeScript.
                    pluginElement.setAttribute('src', `data/plugin_entry.html?name=${plugin.id}`);
                    this.experimentPluginHostLib.registerPluginIframe(pluginElement, plugin.id);
                    this.pluginsContainer.nativeElement.appendChild(pluginElement);
                    break;
                }
                case LoadingMechanismType.NG_COMPONENT:
                    /** @type {?} */
                    const ngComponentClass = this.pluginRegistry.getComponent(plugin.id);
                    if (ngComponentClass) {
                        /** @type {?} */
                        const componentFactory = this.componentFactoryResolver.resolveComponentFactory(ngComponentClass);
                        /** @type {?} */
                        const pluginComponent = this.ngPluginContainer.createComponent(componentFactory);
                        pluginElement = pluginComponent.location.nativeElement;
                    }
                    else {
                        console.error(`No registered Angular component for plugin: ${plugin.id}`);
                    }
                    break;
                case LoadingMechanismType.NONE:
                    break;
                default:
                    console.error('Unexpected plugin');
            }
            return pluginElement;
        }
        /**
         * @private
         * @param {?} plugin
         * @param {?} initialStamp
         * @return {?}
         */
        reload(plugin, initialStamp) {
            if (!initialStamp && plugin.disable_reload) {
                return;
            }
            /** @type {?} */
            const maybeDashboard = (/** @type {?} */ (this.pluginInstances.get(plugin.id)));
            if (maybeDashboard && maybeDashboard.reload) {
                maybeDashboard.reload();
            }
        }
    }
    PluginsComponent.decorators = [
        { type: Component, args: [{
                    selector: 'plugins-component',
                    templateUrl: './plugins_component.ng.html',
                    styles: [
                        `
      :host {
        display: block;
        position: relative;
      }
      .plugins {
        height: 100%;
        position: relative;
      }
      .warning {
        background-color: #fff;
        bottom: 0;
        left: 0;
        position: absolute;
        right: 0;
        top: 0;
      }
      .warning-message {
        margin: 80px auto 0;
        max-width: 540px;
      }
      .last-reload-time {
        font-style: italic;
      }
      .plugins ::ng-deep iframe {
        border: 0;
        height: 100%;
        width: 100%;
      }
    `,
                    ],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    PluginsComponent.ctorParameters = () => [
        { type: ComponentFactoryResolver },
        { type: PluginRegistryModule }
    ];
    PluginsComponent.propDecorators = {
        pluginsContainer: [{ type: ViewChild, args: ['pluginContainer', { static: true, read: ElementRef },] }],
        ngPluginContainer: [{ type: ViewChild, args: ['ngPluginContainer', { static: true, read: ViewContainerRef },] }],
        activePluginId: [{ type: Input }],
        activeKnownPlugin: [{ type: Input }],
        pluginLoadState: [{ type: Input }],
        dataLocation: [{ type: Input }],
        lastUpdated: [{ type: Input }],
        environmentFailureNotFoundTemplate: [{ type: Input }],
        environmentFailureUnknownTemplate: [{ type: Input }]
    };
    /** @nocollapse */ PluginsComponent.ɵfac = function PluginsComponent_Factory(t) { return new (t || PluginsComponent)(ɵɵdirectiveInject(ComponentFactoryResolver), ɵɵdirectiveInject(PluginRegistryModule)); };
    /** @nocollapse */ PluginsComponent.ɵcmp = ɵɵdefineComponent({ type: PluginsComponent, selectors: [["plugins-component"]], viewQuery: function PluginsComponent_Query(rf, ctx) { if (rf & 1) {
            ɵɵstaticViewQuery(_c0$a, true, ElementRef);
            ɵɵstaticViewQuery(_c1$8, true, ViewContainerRef);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.pluginsContainer = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.ngPluginContainer = _t.first);
        } }, inputs: { activePluginId: "activePluginId", activeKnownPlugin: "activeKnownPlugin", pluginLoadState: "pluginLoadState", dataLocation: "dataLocation", lastUpdated: "lastUpdated", environmentFailureNotFoundTemplate: "environmentFailureNotFoundTemplate", environmentFailureUnknownTemplate: "environmentFailureUnknownTemplate" }, features: [ɵɵNgOnChangesFeature], decls: 9, vars: 1, consts: [[1, "plugins"], ["pluginContainer", ""], ["ngPluginContainer", ""], ["class", "warning", 3, "ngSwitch", 4, "ngIf"], ["environmentFailureDefaultTemplate", ""], ["dateAndDataLocation", ""], [1, "warning", 3, "ngSwitch"], [1, "warning-message"], [4, "ngSwitchCase"], [4, "ngTemplateOutlet"], [1, "unknown-plugin"], [3, "ngTemplateOutlet"], [1, "no-active-plugin"], ["href", "https://github.com/tensorflow/tensorboard/blob/master/README.md"], ["href", "https://www.tensorflow.org/get_started/summaries_and_tensorboard"], ["href", "https://github.com/tensorflow/tensorboard/blob/master/README.md#my-tensorboard-isnt-showing-any-data-whats-wrong"], [1, "environment-not-loaded"], [1, "last-reload-time"], ["class", "data-location", 4, "ngIf"], [1, "data-location"]], template: function PluginsComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "div", 0, 1);
            ɵɵelementContainer(2, null, 2);
            ɵɵelementEnd();
            ɵɵtemplate(4, PluginsComponent_div_4_Template, 6, 5, "div", 3);
            ɵɵtemplate(5, PluginsComponent_ng_template_5_Template, 6, 1, "ng-template", null, 4, ɵɵtemplateRefExtractor);
            ɵɵtemplate(7, PluginsComponent_ng_template_7_Template, 4, 5, "ng-template", null, 5, ɵɵtemplateRefExtractor);
        } if (rf & 2) {
            ɵɵadvance(4);
            ɵɵproperty("ngIf", ctx.pluginLoadState !== ctx.PluginLoadState.LOADED && ctx.pluginLoadState !== ctx.PluginLoadState.LOADING);
        } }, directives: [NgIf, NgSwitch, NgSwitchCase, NgTemplateOutlet], pipes: [DatePipe], styles: ["[_nghost-%COMP%] {\n        display: block;\n        position: relative;\n      }\n      .plugins[_ngcontent-%COMP%] {\n        height: 100%;\n        position: relative;\n      }\n      .warning[_ngcontent-%COMP%] {\n        background-color: #fff;\n        bottom: 0;\n        left: 0;\n        position: absolute;\n        right: 0;\n        top: 0;\n      }\n      .warning-message[_ngcontent-%COMP%] {\n        margin: 80px auto 0;\n        max-width: 540px;\n      }\n      .last-reload-time[_ngcontent-%COMP%] {\n        font-style: italic;\n      }\n      .plugins[_ngcontent-%COMP%]     iframe {\n        border: 0;\n        height: 100%;\n        width: 100%;\n      }"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/plugins_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const activePlugin = createSelector(getPlugins, getActivePlugin, (/**
     * @param {?} plugins
     * @param {?} id
     * @return {?}
     */
    (plugins, id) => {
        if (!id || !plugins[id])
            return null;
        return Object.assign({ id }, plugins[id]);
    }));
    /** @type {?} */
    const lastLoadedTimeInMs = createSelector(getPluginsListLoaded, (/**
     * @param {?} loadState
     * @return {?}
     */
    (loadState) => {
        return loadState.lastLoadedTimeInMs;
    }));
    class PluginsContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.activeKnownPlugin$ = this.store.select(activePlugin);
            this.activePluginId$ = this.store.select(getActivePlugin);
            this.pluginLoadState$ = combineLatest(this.activeKnownPlugin$, this.activePluginId$, this.store.select(getPluginsListLoaded)).pipe(map((/**
             * @param {?} __0
             * @return {?}
             */
            ([activePlugin, activePluginId, loadState]) => {
                if (loadState.failureCode !== null) {
                    // Despite its 'Plugins'-specific name, getPluginsListLoaded actually
                    // encapsulates multiple requests to load different parts of the
                    // environment.
                    if (loadState.failureCode === PluginsListFailureCode.NOT_FOUND) {
                        return PluginLoadState.ENVIRONMENT_FAILURE_NOT_FOUND;
                    }
                    else {
                        return PluginLoadState.ENVIRONMENT_FAILURE_UNKNOWN;
                    }
                }
                if (activePlugin !== null) {
                    return PluginLoadState.LOADED;
                }
                if (loadState.lastLoadedTimeInMs === null &&
                    loadState.state === DataLoadState.LOADING) {
                    return PluginLoadState.LOADING;
                }
                if (activePluginId) {
                    return PluginLoadState.UNKNOWN_PLUGIN_ID;
                }
                return PluginLoadState.NO_ENABLED_PLUGINS;
            })));
            this.lastLoadedTimeInMs$ = this.store.select(lastLoadedTimeInMs);
            this.dataLocation$ = this.store.select(getEnvironment).pipe(map((/**
             * @param {?} env
             * @return {?}
             */
            (env) => {
                return env.data_location;
            })));
        }
    }
    PluginsContainer.decorators = [
        { type: Component, args: [{
                    selector: 'plugins',
                    template: `
    <plugins-component
      [activeKnownPlugin]="activeKnownPlugin$ | async"
      [activePluginId]="activePluginId$ | async"
      [dataLocation]="dataLocation$ | async"
      [lastUpdated]="lastLoadedTimeInMs$ | async"
      [pluginLoadState]="pluginLoadState$ | async"
      [environmentFailureNotFoundTemplate]="environmentFailureNotFoundTemplate"
      [environmentFailureUnknownTemplate]="environmentFailureUnknownTemplate"
    ></plugins-component>
  `,
                    styles: ['plugins-component { height: 100%; }'],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    PluginsContainer.ctorParameters = () => [
        { type: Store }
    ];
    PluginsContainer.propDecorators = {
        environmentFailureNotFoundTemplate: [{ type: Input }],
        environmentFailureUnknownTemplate: [{ type: Input }]
    };
    /** @nocollapse */ PluginsContainer.ɵfac = function PluginsContainer_Factory(t) { return new (t || PluginsContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ PluginsContainer.ɵcmp = ɵɵdefineComponent({ type: PluginsContainer, selectors: [["plugins"]], inputs: { environmentFailureNotFoundTemplate: "environmentFailureNotFoundTemplate", environmentFailureUnknownTemplate: "environmentFailureUnknownTemplate" }, decls: 6, vars: 17, consts: [[3, "activeKnownPlugin", "activePluginId", "dataLocation", "lastUpdated", "pluginLoadState", "environmentFailureNotFoundTemplate", "environmentFailureUnknownTemplate"]], template: function PluginsContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelement(0, "plugins-component", 0);
            ɵɵpipe(1, "async");
            ɵɵpipe(2, "async");
            ɵɵpipe(3, "async");
            ɵɵpipe(4, "async");
            ɵɵpipe(5, "async");
        } if (rf & 2) {
            ɵɵproperty("activeKnownPlugin", ɵɵpipeBind1(1, 7, ctx.activeKnownPlugin$))("activePluginId", ɵɵpipeBind1(2, 9, ctx.activePluginId$))("dataLocation", ɵɵpipeBind1(3, 11, ctx.dataLocation$))("lastUpdated", ɵɵpipeBind1(4, 13, ctx.lastLoadedTimeInMs$))("pluginLoadState", ɵɵpipeBind1(5, 15, ctx.pluginLoadState$))("environmentFailureNotFoundTemplate", ctx.environmentFailureNotFoundTemplate)("environmentFailureUnknownTemplate", ctx.environmentFailureUnknownTemplate);
        } }, styles: ["plugins-component[_ngcontent-%COMP%] { height: 100%; }"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/plugins_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class PluginsModule {
    }
    PluginsModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [PluginsContainer, PluginsComponent],
                    exports: [PluginsContainer],
                    imports: [CoreModule, CommonModule, PluginRegistryModule],
                },] },
    ];
    /** @nocollapse */ PluginsModule.ɵmod = ɵɵdefineNgModule({ type: PluginsModule });
    /** @nocollapse */ PluginsModule.ɵinj = ɵɵdefineInjector({ factory: function PluginsModule_Factory(t) { return new (t || PluginsModule)(); }, imports: [[CoreModule, CommonModule, PluginRegistryModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(PluginsModule, { declarations: [PluginsContainer, PluginsComponent], imports: [CoreModule, CommonModule, PluginRegistryModule], exports: [PluginsContainer] }); })();
    ɵɵsetComponentScope(PluginsContainer, [NgClass, NgComponentOutlet, NgForOf, NgIf, NgTemplateOutlet, NgStyle, NgSwitch, NgSwitchCase, NgSwitchDefault, NgPlural, NgPluralCase, PluginsContainer, PluginsComponent], [AsyncPipe, UpperCasePipe, LowerCasePipe, JsonPipe, SlicePipe, DecimalPipe, PercentPipe, TitleCasePipe, CurrencyPipe, DatePipe, I18nPluralPipe, I18nSelectPipe, KeyValuePipe]);

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/reducer_config.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    // console.log all actions
    /**
     * @param {?} reducer
     * @return {?}
     */
    function logger(reducer) {
        return (/**
         * @param {?} state
         * @param {?} action
         * @return {?}
         */
        (state, action) => {
            /** @type {?} */
            const result = reducer(state, action);
            console.groupCollapsed(action.type);
            console.log('prev state', state);
            console.log('action', action);
            console.log('next state', result);
            console.groupEnd();
            return result;
        });
    }
    /**
     * @return {?}
     */
    function loggerMetaReducerFactory() {
        return !isDevMode()
            ? (/**
             * @param {?} reducer
             * @return {?}
             */
            (reducer) => (/**
             * @param {?} state
             * @param {?} action
             * @return {?}
             */
            (state, action) => {
                return reducer(state, action);
            }))
            : logger;
    }
    /** @type {?} */
    const ROOT_REDUCERS = new InjectionToken('Root reducers token', {
        factory: (/**
         * @return {?}
         */
        () => ({})),
    });

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * The action can fire when no requests are actually made (i.e., an empty
     * requestedExperimentIds).
     */
    const fetchRunsRequested = createAction('[Runs] Fetch Runs Requested', props());
    /**
     * The action can fire when no requests are actually made (i.e., an empty
     * requestedExperimentIds).
     */
    const fetchRunsSucceeded = createAction('[Runs] Fetch Runs Succeeded', props());
    const fetchRunsFailed = createAction('[Runs] Fetch Runs Failed', props());
    const runSelectionToggled = createAction('[Runs] Run Selection Toggled', props());
    const runPageSelectionToggled = createAction('[Runs] Run Page Selection Toggled', props());
    const runsSelectAll = createAction('[Runs] Runs Select All', props());
    const runSelectorPaginationOptionChanged = createAction('[Runs] Run Selector Pagination Option Changed', props());
    const runSelectorSortChanged = createAction('[Runs] Run Selector Sort Changed', props());
    const runSelectorRegexFilterChanged = createAction('[Runs] Run Selector Regex Filter Changed', props());
    const runColorChanged = createAction('[Runs] Run Color Changed', props());
    const runTableShown = createAction('[Runs] Run Table Shown', props());
    const runDiscreteHparamFilterChanged = createAction('[Runs] Run Discrete Hparam Filter Changed', props());
    const runIntervalHparamFilterChanged = createAction('[Runs] Run Interval Hparam Filter Changed', props());
    const runMetricFilterChanged = createAction('[Runs] Run Metric Filter Changed', props());

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/runs/data_source/runs_data_source.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @param {?} run
     * @param {?} experimentId
     * @return {?}
     */
    function runToRunId$1(run, experimentId) {
        return `${experimentId}/${run}`;
    }
    class TBRunsDataSource {
        /**
         * @param {?} http
         */
        constructor(http) {
            this.http = http;
        }
        /**
         * @param {?} experimentId
         * @return {?}
         */
        fetchRuns(experimentId) {
            return this.http.get('data/runs').pipe(map((/**
             * @param {?} runs
             * @return {?}
             */
            (runs) => {
                return runs.map((/**
                 * @param {?} run
                 * @return {?}
                 */
                (run) => {
                    return {
                        id: runToRunId$1(run, experimentId),
                        name: run,
                        // Use a dummy startTime for now, until there is backend support.
                        startTime: 0,
                    };
                }));
            })));
        }
        /**
         * @param {?} experimentId
         * @return {?}
         */
        fetchHparamsMetadata(experimentId) {
            // Return a stub implementation.
            return of({
                hparamSpecs: [],
                metricSpecs: [],
                runToHparamsAndMetrics: {},
            });
        }
    }
    TBRunsDataSource.decorators = [
        { type: Injectable },
    ];
    /** @nocollapse */
    TBRunsDataSource.ctorParameters = () => [
        { type: TBHttpClient }
    ];
    /** @nocollapse */ TBRunsDataSource.ɵfac = function TBRunsDataSource_Factory(t) { return new (t || TBRunsDataSource)(ɵɵinject(TBHttpClient)); };
    /** @nocollapse */ TBRunsDataSource.ɵprov = ɵɵdefineInjectable({ token: TBRunsDataSource, factory: TBRunsDataSource.ɵfac });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/runs/data_source/runs_data_source_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class RunsDataSourceModule {
    }
    RunsDataSourceModule.decorators = [
        { type: NgModule, args: [{
                    providers: [{ provide: RunsDataSource, useClass: TBRunsDataSource }],
                },] },
    ];
    /** @nocollapse */ RunsDataSourceModule.ɵmod = ɵɵdefineNgModule({ type: RunsDataSourceModule });
    /** @nocollapse */ RunsDataSourceModule.ɵinj = ɵɵdefineInjector({ factory: function RunsDataSourceModule_Factory(t) { return new (t || RunsDataSourceModule)(); }, providers: [{ provide: RunsDataSource, useClass: TBRunsDataSource }] });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/runs/effects/runs_effects.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Runs effect for fetching data from the backend.
     */
    class RunsEffects {
        /**
         * @param {?} actions$
         * @param {?} store
         * @param {?} runsDataSource
         */
        constructor(actions$, store, runsDataSource) {
            this.actions$ = actions$;
            this.store = store;
            this.runsDataSource = runsDataSource;
            /**
             * Ensures runs are loaded when a run table is shown.
             *
             * @export
             */
            this.loadRunsOnRunTableShown$ = createEffect((/**
             * @return {?}
             */
            () => this.actions$.pipe(ofType(runTableShown), mergeMap((/**
             * @param {?} __0
             * @return {?}
             */
            ({ experimentIds }) => {
                /** @type {?} */
                const experimentsToFetch$ = this.getExperimentsWithLoadState(experimentIds, (/**
                 * @param {?} state
                 * @return {?}
                 */
                (state) => {
                    return (state === DataLoadState.FAILED ||
                        state === DataLoadState.NOT_LOADED);
                }));
                return experimentsToFetch$.pipe(filter((/**
                 * @param {?} experimentIds
                 * @return {?}
                 */
                (experimentIds) => !!experimentIds.length)), mergeMap((/**
                 * @param {?} experimentIdsToBeFetched
                 * @return {?}
                 */
                (experimentIdsToBeFetched) => {
                    return this.fetchAllRunsList(experimentIds, experimentIdsToBeFetched);
                })));
            })))), { dispatch: false });
            this.experimentsWithStaleRunsOnRouteChange$ = this.actions$.pipe(ofType(navigated), withLatestFrom(this.store.select(getRouteId$1)), distinctUntilChanged((/**
             * @param {?} __0
             * @param {?} __1
             * @return {?}
             */
            ([, prevRouteId], [, currRouteId]) => {
                return prevRouteId === currRouteId;
            })), withLatestFrom(this.store.select(getExperimentIdsFromRoute)), filter((/**
             * @param {?} __0
             * @return {?}
             */
            ([, experimentIds]) => !!experimentIds)), map((/**
             * @param {?} __0
             * @return {?}
             */
            ([, experimentIds]) => (/** @type {?} */ (experimentIds)))), mergeMap((/**
             * @param {?} experimentIds
             * @return {?}
             */
            (experimentIds) => {
                return this.getExperimentsWithLoadState(experimentIds, (/**
                 * @param {?} state
                 * @return {?}
                 */
                (state) => {
                    return (state === DataLoadState.FAILED || state === DataLoadState.NOT_LOADED);
                })).pipe(map((/**
                 * @param {?} experimentIdsToBeFetched
                 * @return {?}
                 */
                (experimentIdsToBeFetched) => {
                    return { experimentIds, experimentIdsToBeFetched };
                })));
            })));
            this.experimentsWithStaleRunsOnReload$ = this.actions$.pipe(ofType(reload, manualReload), withLatestFrom(this.store.select(getExperimentIdsFromRoute)), filter((/**
             * @param {?} __0
             * @return {?}
             */
            ([, experimentIds]) => !!experimentIds)), map((/**
             * @param {?} __0
             * @return {?}
             */
            ([, experimentIds]) => (/** @type {?} */ (experimentIds)))), mergeMap((/**
             * @param {?} experimentIds
             * @return {?}
             */
            (experimentIds) => {
                return this.getExperimentsWithLoadState(experimentIds, (/**
                 * @param {?} state
                 * @return {?}
                 */
                (state) => {
                    return state !== DataLoadState.LOADING;
                })).pipe(map((/**
                 * @param {?} experimentIdsToBeFetched
                 * @return {?}
                 */
                (experimentIdsToBeFetched) => {
                    return { experimentIds, experimentIdsToBeFetched };
                })));
            })));
            /**
             * Fetches runs on navigation or in-app reload.
             *
             * @export
             */
            this.loadRunsOnNavigationOrReload$ = createEffect((/**
             * @return {?}
             */
            () => {
                return merge(this.experimentsWithStaleRunsOnRouteChange$, this.experimentsWithStaleRunsOnReload$).pipe(mergeMap((/**
                 * @param {?} __0
                 * @return {?}
                 */
                ({ experimentIds, experimentIdsToBeFetched }) => {
                    return this.fetchAllRunsList(experimentIds, experimentIdsToBeFetched);
                })));
            }), { dispatch: false });
        }
        /**
         * @private
         * @param {?} experimentId
         * @return {?}
         */
        getRunsListLoadState(experimentId) {
            return this.store.select(getRunsLoadState, { experimentId }).pipe(take(1));
        }
        /**
         * @private
         * @param {?} experimentIds
         * @param {?} loadStateMatcher
         * @return {?}
         */
        getExperimentsWithLoadState(experimentIds, loadStateMatcher) {
            return forkJoin(experimentIds.map((/**
             * @param {?} eid
             * @return {?}
             */
            (eid) => {
                return this.getRunsListLoadState(eid);
            }))).pipe(map((/**
             * @param {?} loadStates
             * @return {?}
             */
            (loadStates) => {
                return experimentIds.filter((/**
                 * @param {?} unused
                 * @param {?} index
                 * @return {?}
                 */
                (unused, index) => {
                    return loadStateMatcher(loadStates[index].state);
                }));
            })));
        }
        /**
         * IMPORTANT: actions are dispatched even when there are no experiments to
         * fetch.
         *
         * Observable organization:
         * 1. dispatch requested action
         * 2. make requests for experiments that require fetching while waiting for
         *    runs if already loading and return runs
         * 3. combine the result from local + server where server data takaes
         *    precedence.
         * 4. dispatch succeeded if successful. else, dispatch failed.
         * @private
         * @param {?} experimentIds
         * @param {?} experimentIdsToBeFetched
         * @return {?}
         */
        fetchAllRunsList(experimentIds, experimentIdsToBeFetched) {
            return of({ experimentIds, experimentIdsToBeFetched }).pipe(tap((/**
             * @return {?}
             */
            () => {
                this.store.dispatch(fetchRunsRequested({
                    experimentIds,
                    requestedExperimentIds: experimentIdsToBeFetched,
                }));
            })), mergeMap((/**
             * @return {?}
             */
            () => {
                /** @type {?} */
                const eidsToBeFetched = new Set(experimentIdsToBeFetched);
                /** @type {?} */
                const fetchOrGetRuns = experimentIds.map((/**
                 * @param {?} experimentId
                 * @return {?}
                 */
                (experimentId) => {
                    if (eidsToBeFetched.has(experimentId)) {
                        return this.fetchRunsForExperiment(experimentId);
                    }
                    return this.maybeWaitForRunsAndGetRuns(experimentId);
                }));
                return forkJoin(fetchOrGetRuns);
            })), map((/**
             * @param {?} runsAndMedataList
             * @return {?}
             */
            (runsAndMedataList) => {
                /** @type {?} */
                const newRunsAndMetadata = (/** @type {?} */ ({}));
                /** @type {?} */
                const runsForAllExperiments = [];
                for (const runsAndMedata of runsAndMedataList) {
                    runsForAllExperiments.push(...runsAndMedata.runs);
                    if (runsAndMedata.fromRemote) {
                        newRunsAndMetadata[runsAndMedata.experimentId] = {
                            runs: runsAndMedata.runs,
                            metadata: runsAndMedata.metadata,
                        };
                    }
                }
                return { newRunsAndMetadata, runsForAllExperiments };
            })), tap((/**
             * @param {?} __0
             * @return {?}
             */
            ({ newRunsAndMetadata, runsForAllExperiments }) => {
                this.store.dispatch(fetchRunsSucceeded({
                    experimentIds,
                    newRunsAndMetadata,
                    runsForAllExperiments,
                }));
            })), catchError((/**
             * @param {?} error
             * @return {?}
             */
            (error) => {
                this.store.dispatch(fetchRunsFailed({
                    experimentIds,
                    requestedExperimentIds: experimentIdsToBeFetched,
                }));
                return of(null);
            })), map((/**
             * @return {?}
             */
            () => null)));
        }
        /**
         * @private
         * @param {?} experimentId
         * @return {?}
         */
        maybeWaitForRunsAndGetRuns(experimentId) {
            return this.store.select(getRunsLoadState, { experimentId }).pipe(filter((/**
             * @param {?} loadState
             * @return {?}
             */
            (loadState) => loadState.state !== DataLoadState.LOADING)), take(1), mergeMap((/**
             * @param {?} loadState
             * @return {?}
             */
            (loadState) => {
                if (loadState.state === DataLoadState.FAILED) {
                    return throwError(new Error('Pending request failed'));
                }
                return of(loadState);
            })), withLatestFrom(this.store.select(getRuns, { experimentId })), map((/**
             * @param {?} __0
             * @return {?}
             */
            ([, runs]) => ({ fromRemote: false, experimentId, runs }))));
        }
        /**
         * @private
         * @param {?} experimentId
         * @return {?}
         */
        fetchRunsForExperiment(experimentId) {
            return forkJoin([
                this.runsDataSource.fetchRuns(experimentId),
                this.runsDataSource.fetchHparamsMetadata(experimentId),
            ]).pipe(map((/**
             * @param {?} __0
             * @return {?}
             */
            ([runs, metadata]) => {
                return { fromRemote: true, experimentId, runs, metadata };
            })));
        }
    }
    RunsEffects.decorators = [
        { type: Injectable },
    ];
    /** @nocollapse */
    RunsEffects.ctorParameters = () => [
        { type: Actions },
        { type: Store },
        { type: RunsDataSource }
    ];
    /** @nocollapse */ RunsEffects.ɵfac = function RunsEffects_Factory(t) { return new (t || RunsEffects)(ɵɵinject(Actions), ɵɵinject(Store), ɵɵinject(RunsDataSource)); };
    /** @nocollapse */ RunsEffects.ɵprov = ɵɵdefineInjectable({ token: RunsEffects, factory: RunsEffects.ɵfac });

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * Utility for managing routeful states. It returns route contexted
     * `initialState` and `reducers` that help manage the routeful state.
     *
     * An optional `onRouteIdChanged` function will modify the state after it
     * is loaded from the cache.
     *
     * Example usage:
     *
     * const {initialState, reducers: routeReducers} =
     *    createRouteContextedState(
     *        {myRoutefulState: 0},
     *        {nonRoutefulState: 'one'},
     *        (state) => {
     *          console.log('Reset state upon mounting a new route');
     *          return {myRoutefulState: 0, nonRoutefulState: 'one'};
     *        }
     *    );
     *
     * export const reducers = composeReducers(routeReducers, reducer);
     */
    function createRouteContextedState(routefulInitialState, nonRoutefulInitialState, onRouteIdChanged) {
        const keys = Object.keys(routefulInitialState);
        const initialState = Object.assign(Object.assign(Object.assign({}, routefulInitialState), nonRoutefulInitialState), { privateRouteContextedState: {} });
        const reducers = createReducer({}, on(navigated, (state, { before, after }) => {
            const afterRouteId = getRouteId(after.routeKind, after.params);
            const beforeRouteId = before
                ? getRouteId(before.routeKind, before.params)
                : null;
            // When the routeIds are the same, do not modify the state.
            if (beforeRouteId === afterRouteId) {
                return state;
            }
            let nextContextedStateCache = Object.assign({}, state.privateRouteContextedState);
            if (beforeRouteId) {
                const currRoutefulState = {};
                for (const key of keys) {
                    currRoutefulState[key] = state[key];
                }
                nextContextedStateCache = Object.assign(Object.assign({}, nextContextedStateCache), { [beforeRouteId]: currRoutefulState });
            }
            let nextRoutefulState = state.privateRouteContextedState &&
                state.privateRouteContextedState[afterRouteId]
                ? state.privateRouteContextedState[afterRouteId]
                : null;
            // Set `nextRoutefulState` to the initialState when `before`
            // is non-empty. On the initial load when `before` is null, the
            // `state` can already have values from bootstraping deeplinks and it
            // should not overwrite the values.
            if (beforeRouteId && nextRoutefulState === null) {
                nextRoutefulState = routefulInitialState;
            }
            let nextFullState = Object.assign(Object.assign(Object.assign({}, state), nextRoutefulState), { privateRouteContextedState: nextContextedStateCache });
            if (onRouteIdChanged) {
                return onRouteIdChanged(nextFullState);
            }
            return nextFullState;
        }));
        return {
            initialState,
            reducers,
        };
    }

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * @fileoverview Type definition of the UI data that spans action, store, view,
     * and data sources.
     */
    var SortDirection;
    (function (SortDirection) {
        SortDirection["ASC"] = "asc";
        SortDirection["DESC"] = "desc";
        SortDirection["UNSET"] = "";
    })(SortDirection || (SortDirection = {}));
    var SearchTokenKey;
    (function (SearchTokenKey) {
        SearchTokenKey["USER"] = "user";
        SearchTokenKey["BEFORE"] = "before";
        SearchTokenKey["AFTER"] = "after";
        SearchTokenKey["REGEX"] = "regex";
    })(SearchTokenKey || (SearchTokenKey = {}));

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    const CHART_COLOR_PALLETE = [
        '#425066',
        '#12b5cb',
        '#e52592',
        '#f9ab00',
        '#9334e6',
        '#7cb342',
        '#e8710a',
    ];
    let colorIndex = 0;
    /**
     * Returns hex color for charts.
     */
    function getNextChartColor() {
        const color = CHART_COLOR_PALLETE[colorIndex];
        colorIndex = (colorIndex + 1) % CHART_COLOR_PALLETE.length;
        return color;
    }

    function composeReducers(...actionReducers) {
        return (state, action) => {
            let intermediateState = state;
            for (const reducer of actionReducers) {
                intermediateState = reducer(intermediateState, action);
            }
            return intermediateState;
        };
    }

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    const dataInitialState = {
        runIds: {},
        runIdToExpId: {},
        runMetadata: {},
        runsLoadState: {},
        hparamAndMetricSpec: {},
        selectionState: new Map(),
    };
    const dataReducer$1 = createReducer(dataInitialState, on(fetchRunsRequested, (state, action) => {
        const nextRunsLoadState = Object.assign({}, state.runsLoadState);
        for (const eid of action.requestedExperimentIds) {
            nextRunsLoadState[eid] = Object.assign(Object.assign({ lastLoadedTimeInMs: null }, nextRunsLoadState[eid]), { state: DataLoadState.LOADING });
        }
        return Object.assign(Object.assign({}, state), { runsLoadState: nextRunsLoadState });
    }), on(fetchRunsSucceeded, (state, action) => {
        const nextRunIds = Object.assign({}, state.runIds);
        const nextRunMetadata = Object.assign({}, state.runMetadata);
        const nextRunIdToExpId = Object.assign({}, state.runIdToExpId);
        const nextRunsLoadState = Object.assign({}, state.runsLoadState);
        const nextHparamAndMetricSpec = Object.assign({}, state.hparamAndMetricSpec);
        const nextSelectionState = new Map(state.selectionState);
        for (const eid of Object.keys(action.newRunsAndMetadata)) {
            const { runs, metadata } = action.newRunsAndMetadata[eid];
            nextRunIds[eid] = runs.map(({ id }) => id);
            nextRunsLoadState[eid] = Object.assign(Object.assign({}, nextRunsLoadState[eid]), { lastLoadedTimeInMs: Date.now(), state: DataLoadState.LOADED });
            for (const run of runs) {
                const hparamAndMetrics = metadata.runToHparamsAndMetrics[run.id];
                nextRunMetadata[run.id] = Object.assign(Object.assign({}, run), { hparams: hparamAndMetrics ? hparamAndMetrics.hparams : null, metrics: hparamAndMetrics ? hparamAndMetrics.metrics : null });
                nextRunIdToExpId[run.id] = eid;
            }
            nextHparamAndMetricSpec[eid] = {
                hparams: metadata.hparamSpecs,
                metrics: metadata.metricSpecs,
            };
        }
        const eidsBasedKey = serializeExperimentIds(action.experimentIds);
        if (!nextSelectionState.has(eidsBasedKey)) {
            const selectionMap = new Map();
            const runSelected = action.runsForAllExperiments.length <=
                MAX_NUM_RUNS_TO_ENABLE_BY_DEFAULT;
            for (const run of action.runsForAllExperiments) {
                selectionMap.set(run.id, runSelected);
            }
            nextSelectionState.set(eidsBasedKey, selectionMap);
        }
        else {
            // There could be new runs that were previously unseen.
            // Populate their selection states.
            const selectionMap = new Map(nextSelectionState.get(eidsBasedKey));
            for (const run of action.runsForAllExperiments) {
                if (!selectionMap.has(run.id)) {
                    selectionMap.set(run.id, false);
                }
            }
            nextSelectionState.set(eidsBasedKey, selectionMap);
        }
        return Object.assign(Object.assign({}, state), { runIds: nextRunIds, runIdToExpId: nextRunIdToExpId, runMetadata: nextRunMetadata, runsLoadState: nextRunsLoadState, hparamAndMetricSpec: nextHparamAndMetricSpec, selectionState: nextSelectionState });
    }), on(fetchRunsFailed, (state, action) => {
        const nextRunsLoadState = Object.assign({}, state.runsLoadState);
        for (const eid of action.requestedExperimentIds) {
            nextRunsLoadState[eid] = Object.assign(Object.assign({ lastLoadedTimeInMs: null }, nextRunsLoadState[eid]), { state: DataLoadState.FAILED });
        }
        return Object.assign(Object.assign({}, state), { runsLoadState: nextRunsLoadState });
    }), on(runSelectionToggled, (state, { experimentIds, runId }) => {
        var _a;
        const stateKey = serializeExperimentIds(experimentIds);
        const nextSelectionState = new Map(state.selectionState);
        const subSelectionState = new Map((_a = nextSelectionState.get(stateKey)) !== null && _a !== void 0 ? _a : []);
        subSelectionState.set(runId, !Boolean(subSelectionState.get(runId)));
        nextSelectionState.set(stateKey, subSelectionState);
        return Object.assign(Object.assign({}, state), { selectionState: nextSelectionState });
    }), on(runPageSelectionToggled, (state, { experimentIds, runIds }) => {
        var _a;
        const stateKey = serializeExperimentIds(experimentIds);
        const nextSelectionState = new Map(state.selectionState);
        const subSelectionState = new Map((_a = nextSelectionState.get(stateKey)) !== null && _a !== void 0 ? _a : []);
        const nextValue = !runIds.every((runId) => {
            return Boolean(subSelectionState.get(runId));
        });
        for (const runId of runIds) {
            subSelectionState.set(runId, nextValue);
        }
        nextSelectionState.set(stateKey, subSelectionState);
        return Object.assign(Object.assign({}, state), { selectionState: nextSelectionState });
    }), on(runsSelectAll, (state, { experimentIds }) => {
        var _a;
        const stateKey = serializeExperimentIds(experimentIds);
        const nextSelectionState = new Map(state.selectionState);
        const subSelectionState = new Map((_a = nextSelectionState.get(stateKey)) !== null && _a !== void 0 ? _a : []);
        for (const experimentId of experimentIds) {
            for (const runId of state.runIds[experimentId]) {
                subSelectionState.set(runId, true);
            }
        }
        nextSelectionState.set(stateKey, subSelectionState);
        return Object.assign(Object.assign({}, state), { selectionState: nextSelectionState });
    }));
    const { initialState: uiInitialState, reducers: uiRouteContextReducers, } = createRouteContextedState({
        paginationOption: {
            pageIndex: 0,
            pageSize: 10,
        },
        regexFilter: '',
        sort: {
            column: null,
            direction: SortDirection.UNSET,
        },
        hparamFilters: new Map(),
        metricFilters: new Map(),
        runColorOverride: new Map(),
    }, {
        hparamDefaultFilters: new Map(),
        metricDefaultFilters: new Map(),
        defaultRunColor: new Map(),
    });
    const uiReducer = createReducer(uiInitialState, on(runSelectorPaginationOptionChanged, (state, { pageSize, pageIndex }) => {
        return Object.assign(Object.assign({}, state), { paginationOption: {
                pageSize,
                pageIndex,
            } });
    }), on(runSelectorRegexFilterChanged, (state, action) => {
        return Object.assign(Object.assign({}, state), { regexFilter: action.regexString, paginationOption: Object.assign(Object.assign({}, state.paginationOption), { 
                // Reset the page index to 0 to emulate mat-table behavior.
                pageIndex: 0 }) });
    }), on(runSelectorSortChanged, (state, action) => {
        return Object.assign(Object.assign({}, state), { sort: {
                column: action.column,
                direction: action.direction,
            } });
    }), on(fetchRunsSucceeded, (state, action) => {
        const nextDefaultRunColor = new Map(state.defaultRunColor);
        action.runsForAllExperiments
            .filter((run) => !Boolean(nextDefaultRunColor.get(run.id)))
            .forEach((run) => {
            nextDefaultRunColor.set(run.id, getNextChartColor());
        });
        return Object.assign(Object.assign({}, state), { defaultRunColor: nextDefaultRunColor });
    }), on(runColorChanged, (state, { runId, newColor }) => {
        const nextRunColorOverride = new Map(state.runColorOverride);
        nextRunColorOverride.set(runId, newColor);
        return Object.assign(Object.assign({}, state), { runColorOverride: nextRunColorOverride });
    }), on(runDiscreteHparamFilterChanged, (state, { hparamName, filterValues, includeUndefined }) => {
        const defaultFilter = state.hparamDefaultFilters.get(hparamName);
        if (!defaultFilter) {
            throw new Error(`Unknown hparams: ${hparamName}`);
        }
        if (defaultFilter.type === DomainType.INTERVAL) {
            throw new Error(`Invariant error: Hparams filter is INTERVAL but got a ' +
               'DISCRETE change`);
        }
        const existingFilter = Object.assign(Object.assign({}, defaultFilter), state.hparamFilters.get(hparamName));
        const newHparamFilters = new Map(state.hparamFilters);
        newHparamFilters.set(hparamName, Object.assign(Object.assign({}, existingFilter), { includeUndefined,
            filterValues }));
        return Object.assign(Object.assign({}, state), { hparamFilters: newHparamFilters });
    }), on(runIntervalHparamFilterChanged, (state, action) => {
        const { hparamName, filterLowerValue, filterUpperValue, includeUndefined, } = action;
        const defaultFilter = state.hparamDefaultFilters.get(hparamName);
        if (!defaultFilter) {
            throw new Error(`Unknown hparams: ${hparamName}`);
        }
        if (defaultFilter.type === DomainType.DISCRETE) {
            throw new Error(`Invariant error: Hparams filter is DISCRETE but got a ' +
               'INTERVAL change`);
        }
        const existingFilter = Object.assign(Object.assign({}, defaultFilter), state.hparamFilters.get(hparamName));
        const newHparamFilters = new Map(state.hparamFilters);
        newHparamFilters.set(hparamName, Object.assign(Object.assign({}, existingFilter), { includeUndefined,
            filterLowerValue,
            filterUpperValue }));
        return Object.assign(Object.assign({}, state), { hparamFilters: newHparamFilters });
    }), on(runMetricFilterChanged, (state, change) => {
        const { metricTag, filterLowerValue, filterUpperValue, includeUndefined, } = change;
        const defaultFilter = state.metricDefaultFilters.get(metricTag);
        if (!defaultFilter) {
            throw new Error(`Unknown metric: ${metricTag}`);
        }
        const existingFilter = Object.assign(Object.assign({}, defaultFilter), state.metricFilters.get(metricTag));
        const newMetricFilters = new Map(state.metricFilters);
        newMetricFilters.set(metricTag, Object.assign(Object.assign({}, existingFilter), { filterLowerValue,
            filterUpperValue,
            includeUndefined }));
        return Object.assign(Object.assign({}, state), { metricFilters: newMetricFilters });
    }), 
    /**
     * Sets default filter values.
     *
     * Implementation note: hparam values are defined as part of the spec but
     * metrics are defined with the `runToHparamsAndMetrics`. We need to collect
     * all the values for metrics, then compute the bound to create the default
     * filter value. When the metric values are missing, we set the bound to (0,
     * 1).
     */
    on(fetchRunsSucceeded, (state, action) => {
        var _a, _b, _c, _d;
        if (Object.keys(action.newRunsAndMetadata).length === 0) {
            return state;
        }
        const newHparamFilters = new Map();
        const newMetricFilters = new Map();
        const discreteHparams = new Map();
        const intervalHparams = new Map();
        // Arbitrary ordered collection of metric values collected across
        // experiments and runs to compute extents of metrics.
        const metricValueMinAndMax = new Map();
        const metricTags = new Set();
        for (const eid of Object.keys(action.newRunsAndMetadata)) {
            const { runs, metadata } = action.newRunsAndMetadata[eid];
            // Tabulate all the metric values from runs.
            for (const run of runs) {
                const hparamAndMetrics = metadata.runToHparamsAndMetrics[run.id];
                if (!hparamAndMetrics) {
                    continue;
                }
                for (const metric of hparamAndMetrics.metrics) {
                    const minAndMax = metricValueMinAndMax.get(metric.tag);
                    metricValueMinAndMax.set(metric.tag, {
                        min: minAndMax
                            ? Math.min(minAndMax.min, metric.value)
                            : metric.value,
                        max: minAndMax
                            ? Math.max(minAndMax.max, metric.value)
                            : metric.value,
                    });
                }
            }
            // Record and combine all hparam specs (multiple experiments can
            // have hparams of same name but disjoint set of domain).
            for (const { name, domain } of metadata.hparamSpecs) {
                if (domain.type === DomainType.DISCRETE) {
                    const values = discreteHparams.get(name) || new Set();
                    for (const value of domain.values) {
                        values.add(value);
                    }
                    discreteHparams.set(name, values);
                }
                else {
                    const existing = intervalHparams.get(name);
                    intervalHparams.set(name, {
                        minValue: existing
                            ? Math.min(domain.minValue, existing.minValue)
                            : domain.minValue,
                        maxValue: existing
                            ? Math.max(domain.maxValue, existing.maxValue)
                            : domain.maxValue,
                    });
                }
            }
            for (const metricSpec of metadata.metricSpecs) {
                metricTags.add(metricSpec.tag);
            }
        }
        for (const [name, values] of discreteHparams) {
            newHparamFilters.set(name, {
                type: DomainType.DISCRETE,
                includeUndefined: true,
                possibleValues: [...values],
                filterValues: [...values],
            });
        }
        for (const [name, { minValue, maxValue }] of intervalHparams) {
            newHparamFilters.set(name, {
                type: DomainType.INTERVAL,
                includeUndefined: true,
                minValue,
                maxValue,
                filterLowerValue: minValue,
                filterUpperValue: maxValue,
            });
        }
        for (const metricTag of metricTags) {
            const minAndMax = metricValueMinAndMax.get(metricTag);
            newMetricFilters.set(metricTag, {
                type: DomainType.INTERVAL,
                includeUndefined: true,
                minValue: (_a = minAndMax === null || minAndMax === void 0 ? void 0 : minAndMax.min) !== null && _a !== void 0 ? _a : 0,
                maxValue: (_b = minAndMax === null || minAndMax === void 0 ? void 0 : minAndMax.max) !== null && _b !== void 0 ? _b : 0,
                filterLowerValue: (_c = minAndMax === null || minAndMax === void 0 ? void 0 : minAndMax.min) !== null && _c !== void 0 ? _c : 0,
                filterUpperValue: (_d = minAndMax === null || minAndMax === void 0 ? void 0 : minAndMax.max) !== null && _d !== void 0 ? _d : 0,
            });
        }
        return Object.assign(Object.assign({}, state), { hparamDefaultFilters: newHparamFilters, metricDefaultFilters: newMetricFilters });
    }));
    const routeStatefulUiReducers = composeReducers(uiReducer, uiRouteContextReducers);
    /**
     * Reducers for the experiments.
     */
    function reducers$5(state, action) {
        return combineReducers({
            data: dataReducer$1,
            ui: routeStatefulUiReducers,
        })(state, action);
    }

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     *
     * @fileoverview Module for runs.
     *
     * Generated from: tensorboard/webapp/runs/runs_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @return {?}
     */
    function alertActionProvider() {
        return [
            {
                actionCreator: fetchRunsFailed,
                alertFromAction: (/**
                 * @return {?}
                 */
                () => {
                    return { localizedMessage: 'Failed to fetch runs' };
                }),
            },
        ];
    }
    class RunsModule {
    }
    RunsModule.decorators = [
        { type: NgModule, args: [{
                    imports: [
                        StoreModule.forFeature(RUNS_FEATURE_KEY, reducers$5),
                        EffectsModule.forFeature([RunsEffects]),
                        RunsDataSourceModule,
                        AlertActionModule.registerAlertActions(alertActionProvider),
                    ],
                },] },
    ];
    /** @nocollapse */ RunsModule.ɵmod = ɵɵdefineNgModule({ type: RunsModule });
    /** @nocollapse */ RunsModule.ɵinj = ɵɵdefineInjector({ factory: function RunsModule_Factory(t) { return new (t || RunsModule)(); }, imports: [[
                StoreModule.forFeature(RUNS_FEATURE_KEY, reducers$5),
                EffectsModule.forFeature([RunsEffects]),
                RunsDataSourceModule,
                AlertActionModule.registerAlertActions(alertActionProvider),
            ]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(RunsModule, { imports: [StoreFeatureModule, EffectsFeatureModule, RunsDataSourceModule, AlertActionModule] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/reloader/reloader_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class ReloaderComponent {
        /**
         * @param {?} store
         * @param {?} document
         */
        constructor(store, document) {
            this.store = store;
            this.document = document;
            this.onVisibilityChange = this.onVisibilityChangeImpl.bind(this);
            this.reloadEnabled$ = this.store.pipe(select(getReloadEnabled));
            this.reloadPeriodInMs$ = this.store.pipe(select(getReloadPeriodInMs));
            this.reloadTimerId = null;
            this.missedAutoReload = false;
        }
        /**
         * @return {?}
         */
        ngOnInit() {
            this.document.addEventListener('visibilitychange', this.onVisibilityChange);
            combineLatest(this.reloadEnabled$.pipe(distinctUntilChanged()), this.reloadPeriodInMs$.pipe(distinctUntilChanged())).subscribe((/**
             * @param {?} __0
             * @return {?}
             */
            ([enabled, reloadPeriodInMs]) => {
                this.cancelLoad();
                if (enabled) {
                    this.load((/** @type {?} */ (reloadPeriodInMs)));
                }
            }));
        }
        /**
         * @private
         * @return {?}
         */
        onVisibilityChangeImpl() {
            if (this.document.visibilityState === 'visible' && this.missedAutoReload) {
                this.missedAutoReload = false;
                this.store.dispatch(reload());
            }
        }
        /**
         * @private
         * @param {?} reloadPeriodInMs
         * @return {?}
         */
        load(reloadPeriodInMs) {
            this.reloadTimerId = setTimeout((/**
             * @return {?}
             */
            () => {
                if (this.document.visibilityState === 'visible') {
                    this.store.dispatch(reload());
                }
                else {
                    this.missedAutoReload = true;
                }
                this.load(reloadPeriodInMs);
            }), reloadPeriodInMs);
        }
        /**
         * @private
         * @return {?}
         */
        cancelLoad() {
            if (this.reloadTimerId !== null) {
                clearTimeout(this.reloadTimerId);
            }
            this.reloadTimerId = null;
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this.cancelLoad();
            this.document.removeEventListener('visibilitychange', this.onVisibilityChange);
        }
    }
    ReloaderComponent.decorators = [
        { type: Component, args: [{
                    selector: 'reloader',
                    template: '',
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    ReloaderComponent.ctorParameters = () => [
        { type: Store },
        { type: Document, decorators: [{ type: Inject, args: [DOCUMENT$1,] }] }
    ];
    /** @nocollapse */ ReloaderComponent.ɵfac = function ReloaderComponent_Factory(t) { return new (t || ReloaderComponent)(ɵɵdirectiveInject(Store), ɵɵdirectiveInject(DOCUMENT$1)); };
    /** @nocollapse */ ReloaderComponent.ɵcmp = ɵɵdefineComponent({ type: ReloaderComponent, selectors: [["reloader"]], decls: 0, vars: 0, template: function ReloaderComponent_Template(rf, ctx) { }, encapsulation: 2, changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/reloader/reloader_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class ReloaderModule {
    }
    ReloaderModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [ReloaderComponent],
                    exports: [ReloaderComponent],
                },] },
    ];
    /** @nocollapse */ ReloaderModule.ɵmod = ɵɵdefineNgModule({ type: ReloaderModule });
    /** @nocollapse */ ReloaderModule.ɵinj = ɵɵdefineInjector({ factory: function ReloaderModule_Factory(t) { return new (t || ReloaderModule)(); } });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(ReloaderModule, { declarations: [ReloaderComponent], exports: [ReloaderComponent] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/tb_wrapper/tb_wrapper_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class TensorBoardWrapperComponent {
    }
    TensorBoardWrapperComponent.decorators = [
        { type: Component, args: [{
                    selector: 'tensorboard-wrapper-component',
                    template: `
    <plugins class="plugins"></plugins>
    <reloader></reloader>
  `,
                    styles: [
                        `
      :host {
        display: flex;
        flex-direction: column;
        height: 100%;
      }

      .plugins {
        flex: 1 1;
        overflow: auto;
        position: relative;
      }
    `,
                    ],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */ TensorBoardWrapperComponent.ɵfac = function TensorBoardWrapperComponent_Factory(t) { return new (t || TensorBoardWrapperComponent)(); };
    /** @nocollapse */ TensorBoardWrapperComponent.ɵcmp = ɵɵdefineComponent({ type: TensorBoardWrapperComponent, selectors: [["tensorboard-wrapper-component"]], decls: 2, vars: 0, consts: [[1, "plugins"]], template: function TensorBoardWrapperComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelement(0, "plugins", 0);
            ɵɵelement(1, "reloader");
        } }, directives: [PluginsContainer, ReloaderComponent], styles: ["[_nghost-%COMP%] {\n        display: flex;\n        flex-direction: column;\n        height: 100%;\n      }\n\n      .plugins[_ngcontent-%COMP%] {\n        flex: 1 1;\n        overflow: auto;\n        position: relative;\n      }"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/tb_wrapper/tb_wrapper_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Provides the wrapper component that renders the main dashboard page.
     */
    class TensorBoardWrapperModule {
    }
    TensorBoardWrapperModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [TensorBoardWrapperComponent],
                    imports: [CommonModule, PluginsModule, ReloaderModule],
                    exports: [TensorBoardWrapperComponent],
                    entryComponents: [TensorBoardWrapperComponent],
                },] },
    ];
    /** @nocollapse */ TensorBoardWrapperModule.ɵmod = ɵɵdefineNgModule({ type: TensorBoardWrapperModule });
    /** @nocollapse */ TensorBoardWrapperModule.ɵinj = ɵɵdefineInjector({ factory: function TensorBoardWrapperModule_Factory(t) { return new (t || TensorBoardWrapperModule)(); }, imports: [[CommonModule, PluginsModule, ReloaderModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(TensorBoardWrapperModule, { declarations: [TensorBoardWrapperComponent], imports: [CommonModule, PluginsModule, ReloaderModule], exports: [TensorBoardWrapperComponent] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/data_source/tfdbg2_data_source.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Converts the array representation of a stack frame into the object
     * representation.
     * @param {?} array
     * @return {?}
     */
    function stackFrameAsArrayToStackFrame(array) {
        return {
            host_name: array[0],
            file_path: array[1],
            lineno: array[2],
            function_name: array[3],
        };
    }
    class Tfdbg2HttpServerDataSource {
        /**
         * @param {?} http
         */
        constructor(http) {
            this.http = http;
            this.httpPathPrefix = 'data/plugin/debugger-v2';
        }
        /**
         * @return {?}
         */
        fetchRuns() {
            // TODO(cais): Once the backend uses an DataProvider that unifies tfdbg and
            // non-tfdbg plugins, switch to using `tf_backend.runStore.refresh()`.
            return this.http.get(this.httpPathPrefix + '/runs');
        }
        /**
         * @param {?} run
         * @param {?} begin
         * @param {?} end
         * @return {?}
         */
        fetchExecutionDigests(run, begin, end) {
            return this.http.get(this.httpPathPrefix + '/execution/digests', {
                params: {
                    run,
                    begin: String(begin),
                    end: String(end),
                },
            });
        }
        /**
         * @param {?} run
         * @param {?} begin
         * @param {?} end
         * @return {?}
         */
        fetchExecutionData(run, begin, end) {
            return this.http.get(this.httpPathPrefix + '/execution/data', {
                params: {
                    run,
                    begin: String(begin),
                    end: String(end),
                },
            });
        }
        /**
         * @param {?} run
         * @param {?} begin
         * @param {?} end
         * @param {?=} trace_id
         * @return {?}
         */
        fetchGraphExecutionDigests(run, begin, end, trace_id) {
            if (trace_id !== undefined) {
                throw new Error('trace_id is not implemented for fetchGraphExecutionDigests() yet');
            }
            return this.http.get(this.httpPathPrefix + '/graph_execution/digests', {
                params: {
                    run,
                    begin: String(begin),
                    end: String(end),
                },
            });
        }
        /**
         * @param {?} run
         * @param {?} begin
         * @param {?} end
         * @param {?=} trace_id
         * @return {?}
         */
        fetchGraphExecutionData(run, begin, end, trace_id) {
            if (trace_id !== undefined) {
                throw new Error('trace_id is not implemented for fetchGraphExecutionData() yet');
            }
            return this.http.get(this.httpPathPrefix + '/graph_execution/data', {
                params: {
                    run,
                    begin: String(begin),
                    end: String(end),
                },
            });
        }
        /**
         * @param {?} run
         * @param {?} graph_id
         * @param {?} op_name
         * @return {?}
         */
        fetchGraphOpInfo(run, graph_id, op_name) {
            return this.http.get(this.httpPathPrefix + '/graphs/op_info', {
                params: {
                    run,
                    graph_id,
                    op_name,
                },
            });
        }
        /**
         * @param {?} run
         * @return {?}
         */
        fetchSourceFileList(run) {
            return this.http.get(this.httpPathPrefix + '/source_files/list', {
                params: {
                    run,
                },
            });
        }
        /**
         * @param {?} run
         * @param {?} fileIndex
         * @return {?}
         */
        fetchSourceFile(run, fileIndex) {
            return this.http.get(this.httpPathPrefix + '/source_files/file', {
                params: {
                    run,
                    index: String(fileIndex),
                },
            });
        }
        /**
         * @param {?} run
         * @param {?} stackFrameIds
         * @return {?}
         */
        fetchStackFrames(run, stackFrameIds) {
            return this.http
                .get(this.httpPathPrefix + '/stack_frames/stack_frames', {
                params: {
                    run,
                    stack_frame_ids: stackFrameIds.join(','),
                },
            })
                .pipe(map((/**
             * @param {?} rawResponse
             * @return {?}
             */
            (rawResponse) => {
                return {
                    stack_frames: rawResponse.stack_frames.map((/**
                     * @param {?} stackFrameAsArray
                     * @return {?}
                     */
                    (stackFrameAsArray) => stackFrameAsArrayToStackFrame(stackFrameAsArray))),
                };
            })));
        }
        /**
         * @param {?} run
         * @param {?} begin
         * @param {?} end
         * @param {?=} alert_type
         * @return {?}
         */
        fetchAlerts(run, begin, end, alert_type) {
            /** @type {?} */
            const params = {
                run,
                begin: String(begin),
                end: String(end),
            };
            if (alert_type !== undefined) {
                params['alert_type'] = alert_type;
            }
            return this.http.get(this.httpPathPrefix + '/alerts', {
                params,
            });
        }
    }
    Tfdbg2HttpServerDataSource.decorators = [
        { type: Injectable },
    ];
    /** @nocollapse */
    Tfdbg2HttpServerDataSource.ctorParameters = () => [
        { type: TBHttpClient }
    ];
    /** @nocollapse */ Tfdbg2HttpServerDataSource.ɵfac = function Tfdbg2HttpServerDataSource_Factory(t) { return new (t || Tfdbg2HttpServerDataSource)(ɵɵinject(TBHttpClient)); };
    /** @nocollapse */ Tfdbg2HttpServerDataSource.ɵprov = ɵɵdefineInjectable({ token: Tfdbg2HttpServerDataSource, factory: Tfdbg2HttpServerDataSource.ɵfac });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/data_source/tfdbg2_data_source_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class Tfdbg2ServerDataSourceModule {
    }
    Tfdbg2ServerDataSourceModule.decorators = [
        { type: NgModule, args: [{
                    imports: [TBHttpClientModule],
                    providers: [Tfdbg2HttpServerDataSource],
                },] },
    ];
    /** @nocollapse */ Tfdbg2ServerDataSourceModule.ɵmod = ɵɵdefineNgModule({ type: Tfdbg2ServerDataSourceModule });
    /** @nocollapse */ Tfdbg2ServerDataSourceModule.ɵinj = ɵɵdefineInjector({ factory: function Tfdbg2ServerDataSourceModule_Factory(t) { return new (t || Tfdbg2ServerDataSourceModule)(); }, providers: [Tfdbg2HttpServerDataSource], imports: [[TBHttpClientModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(Tfdbg2ServerDataSourceModule, { imports: [TBHttpClientModule] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/views/inactive/inactive_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class InactiveComponent {
    }
    InactiveComponent.decorators = [
        { type: Component, args: [{
                    selector: 'inactive-component',
                    templateUrl: './inactive_component.ng.html',
                    styleUrls: ['./inactive_component.css'],
                },] },
    ];
    /** @nocollapse */ InactiveComponent.ɵfac = function InactiveComponent_Factory(t) { return new (t || InactiveComponent)(); };
    /** @nocollapse */ InactiveComponent.ɵcmp = ɵɵdefineComponent({ type: InactiveComponent, selectors: [["inactive-component"]], decls: 54, vars: 0, consts: [[1, "container"], [1, "title"], [1, "code"], [1, "arg"], [1, "exhibits-container"], [1, "exhibit"], [1, "screenshot"], ["src", "data:image/png;base64,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"], [1, "description"], ["src", "data:image/png;base64,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"], ["src", "data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAATYAAAE2CAYAAADrvL6pAABKonpUWHRSYXcgcHJvZmlsZSB0eXBlIGV4aWYAAHjarZ1tluW4rWX/axRvCBK/ORzxQ2v1DHr4vTcjs1xVtl/7rW6XqyIz4sa9EgkcnAMC0LX/9//6rv/6r/96SmvhSrm20ku5+V/qqYeXP7T753/9/Pe50/nv+V8sv/70/PX71x8/CHyNvvLnB/X9+fq8fD//4xd+f8Yz/vr9q/36SWi/3ujXD3jjnyvwk/3z+vNF8v3w8/0n/Xqjvn/+UHqrf77U8euN5q8Xnkv59W/647J+vvj36y/fqKzSynxQDGHHJ97nv+nnCuLPvy//1vPfwOueGPnzE9t1vvy+JRbkL7f3++t9/3mB/rLIv/90/X31//jT3xY/vL++H/+2luXXGt3lX//gyX/7fvzjY8JfzOGPKwp//cEY6Z9v59e/37fa9+2fu3tTYUXLL4s6i/38fhteOFjyeH6t8E/l38yf6/mn80+733uy5eue9+Cf+fQnsCvf9aRnPe/zPft8nc/kElPYofI1hBni+V6LNfQwo/uU/Of5Qo09rtjYyxn2xfalGP64lud8bj+fN5/GJ6+Hl4aHN3v4lX/7z/Xf/fB/8s/1fdMleu72x1pxXUG75jLcOf/Lq9iD5/u1b/ks8O9/fm3//Sf7wVTZwXyWuXGD7z1+3mLk5x+2Fc8+R16X+fqzyc9V1683YIn47MzFPJEduMsT81Oeu4ZQn4d1bGzQy5WHmMJgB56cw+IiQ4qxhKuGFvxsfqc+57UhhxL8NtjERuRY8K3GDr1sVkoZ+6mpYUNvjjnlnEuuuV2557fEkkoupdQiyL011lRzLbXWVnt9W2yp5VZaba319vbQIxiYe+m1t977+4br5YNe3uvl9S/fGWHEkUYeZdTRRh/vxHxmmnmWWWebfb4rrLiAiVVWXW319e7n2iDFTjvvsutuu+/3w9a++KUvf+WrX/v69/6xa7929Z/++R/s2vNr18LZKV9X/9g1vnvV+vstHuEku2fsWEgPO17dAQw6uGd3e1IK7px7dveAU+TARWb35lqPO8YWpv2E/D1/7N0/du4/2rcrt/9o38L/becut+7/x85dbN0/79u/2LVlnJtnx3680DW9I97Ha97AFX1jffy1xC+/8avj27NuYdIf/+c/vV7vZ7/14V7Lu+pTc0ilvWxuHJ3LGHPiV2lEL2W/iZeMHdLXWny5g9nWM+89rjBnHLu2J+86dsxfe4G6b+y1V54Je9qTz1q132vt8XAfM+93jTVCGnzuE/oMqV5P460m64X13XvXmQY2sL+Y7/58Ne368LOY573mHp3tHPUeT8pjs2OA/8suYlTXTKLVs+aqL8Qkx014fdfLOnQuMeTBnlbWJ4zSAfY3z/qNVN/wLSxrfiXOXEe6IDV13ixGbQmD7kSZ3sv3sMFf3SzGHPxxxa/NL9xcnv4zoCF18q6xzoBzzq9dbeWS0mKdZr5HuVsP7PscQH7fq4U1S1xhrPtL+VsY0Fz4QY95rxpq3yWwESmXq7SHPflq5uaMN+u72YzO69zYf7P5z8yrcTHAQSz9u0cq1zDgxbuUtQawwq2FuPEjDBOzjWsUtjlyKXPE/PIJBcd/B5+846hcyoMh8GvXGMeq/juL+xc/m6WMWPaY8cOp92zfFdYqHfeOibDOPWMbKUxXpsc2V/bW1j63+XwvYP/HW2MtIbAhRPr2fBeh/QF9FmCCKfW5ucf+xlAx4meXwXKX8RU+vRR88yPCYyftTfebWnt5T8x9rnW18uUJiH3HnMqLDX7/7gq+fvPir6Vvx/pxQTGNsHtboOP1soP8tXwRAMK7uUpQDN8HFUbnfnMAScSs2OV+77NebCAaBRf/5o9X769eUh4uvtfvid/Z/f01XeFli2rBWp5Zguay84QmLjxk8/6sIeyGT320lTBhtYSTxIckPQx3/Paj5w5saoRnvPVjK7j+Y5cdMK6wBRcu1dbfL/LhkXe4NlG6l8aODugSK1RT+bBjXjlHL9t7XvyhFZakh6fnG2sLKQfub7/S77rG816pjFWed/IpgGtvBAKwGON/8lvzR+BvNU7wlL/OAz4zpxb3yCzMfjTHcWNU18M9VZz0TbM27h5He9dREJm9lI7JLlLA10JgJf+dyV7/uT0/XvHKGw4XJjfNBRciT+cepDU3xjeBVTxaOlpDwTKJrvUtGyNs7y7lG4DAgpDlcvfRimvERY9KGJ1VTHoHV9QzVDEBppOQprkWPrTzyg1uwnn4//eyirENYJKV4p750zD2ATvsyVpPv8rTdLWONQJQfOJu8dCuzXs9hC8CYkxExMdPGRu8igRVImkpsd0E8/RiGvta/U45go+dncenS8ZQHlyrPx18YJ9m5BcqccZPWqW+Xwt7EjIn/sRvbaIAcY37wFrZ5kBEXunexM/+eNsviAOyDdB114JoWkDDA0ZhMc/gd0dq/T7sG++5QOFkVDWCDYJc21Xkxei+AtBkQhPocE+C/iD+s7J1Pjvi9x3GUx9Dzq45XWPi7vEREMDYxkJy+bzlnAT+GrmiMcqzuM3b0AQRgOh8HSwB/iN+GSArfO9K8SU8chv8zSUhqmC65dkpG8bAJVZ/84tn8TGLlgFrIxW/jzklsAEyEa+RFosOQI6axShoFPzq/naAOsC29odYLQNCRdQjXozAHcAiqmja0iIcf5N1uzZOCIZFSEeqwD6UY0lCptYDjESQsS8iTSG4VjCsNL6uvZ8JfMSVhbEvvVdJa98gwd73Owi3kWCeXuw8Ay8f7/HtYpC5U2ElZ9vIcbAK/0OLsVvYPpztJkC+hOQbupd0pzuvtUIm3L7bCwUESsz5WblBHOuHS3PRmMKQJRLSYDEfsBHYNRy+lwx29lstBbziQRvCxwdjfdAUGCAxzUWFxLVaCOFjA9DgxQfdZHMJJRe6zHCCzRPeIUh4N5fP4sP1QHZ43s0NsU4pDaImdnALBjKn/v5cFjeW4xUyRGi+HXLYC/6CxAKVcCP8jhshqOZnvwTmsDb2nEDViTMQZe/aiX7YKJav8IvQxg5m94F5YEz3B9oAG4mL78Aw9Kjg3UR0dh4EZ3XenNhZ3pdbxWT73ntdS35BNAoYd4sDz3g7LvpJA/AaKBZvgBMX1n29WMmcHdbRcyCYJSVW5Xd6vfSeRPgI8BtC9audwCgmuA6lI/7cB/WQBkAMZBK+WN5QFwqh5AFY1c6PBm/UePls3nXiVrEMpCB0no1DX1eMIxvneoNJggELrc6aakubVXlYl4VlBQIkbMrNTc0dRcekGghoqIc2c+t54okTMwAuQAbee2E1i1D6QMYC0gCqSeRYF8yumSKAjqUqHiBIcA55GVEwsZRQpDk/sG+xIYSrCU0BR3ZuWgpsYcL52DXAvGvKPUaiAwIKlwWpWHPzPfwWVBNkjtAwTCOWb75gnot+QhWvwKnrvnrCm9YL2zTyPTUAzIQL4DJkdAWXDgqhPkCYZ23eHRwBaNxq2GrqOAg7eoPZgAh4DuJqhvdTjTNqsBcJXTDCRHR92MDpZwgON2E7w8pBawIzDloD4HoB5HCLUDJ49QbWCxTiAon+yADs5e2wi/KCnEQXjHb35x4vCxzBUoIIW8rWoUWgr0Qb/HONpBWj/1ixTuBhY++pz550GysERtxw5p4jITdOIhiUAEJ8A/IQrSeVhYYJQjs77H2+cCQuGX8kHBC8OziFHG0hJcIsYVMgLtgvUqUSP9UMF0Qpju8uHx6sWgnzYzHnU8Hkr+cKxzdSATyTDYIkTrbugTACPBWTfrGrhG9eL9HKP6JM+RNGUJP+6y91biXBs7iqgJ4C1AHLhQRCVgSkTJQ9Yke4LNQPvIkdJoCOilI5CdGAk9UXhI4EKZx8+8GGOt8XqpbCy9oDU5CKypK1WceFfUwYwrcCohX93RMsEH7oTsAE0bm8AQgwCMPpUbRB0I3FL6QEnIa0sMt3vdAk30NchKGypHDaJjXet1IbU+qTi9k9ZJgf9Pg1jHkFEzKGG6KF77Ih1wnhR6j1piDjjSUOoNINFBprByGO/+BZesz75c51FRTlBolbIOJl/BpJx1ZeEcv+jI13l4+CwoEY9LK2REVeyJ0vTN0PYVMJFx2TawgxgjOYixHtPICkC7cYH+8P4n1LTf69N06vjgTmUmhgC4uHw4Eo2CTOAgED6iBVnwuGRY/8tgs0jHshBFAFDfcNmEr2cxC4i+BuKqyz6gMn7qAQwZNojnc11SvmGHC8uHERXgkMbsxG5MiGYQJQfkt6ANe34VHNnWcNDiD1k1ZE435vOzFp4jAD4cfHshMJwhJg5TAFFuf81gA8MCeDUdsGIly5JxOaIxLp2DswCskMnsB8LvgAfJx9NwoMwPreIN3NvRC7MeMXO4bETpYfPv6Cg1wFi8V11wTregsb2Ua4ZHMgBaoeiWsuFaNJXLM8pBNp9tEomMeICeD7yrGzl6VlKTE/SDZLP5EQgAvk+NXjl2ZG4AdloBO4SCVSwfiIIZV/v/5EMzRIiUeXJmpOBd4IY8zrTRg8kQW+zW9XWGE0RTJCx4eJEgAgdwHyfFN5XzES1P49sFzo33oEQZYlXy0Z7DGL73O7IjIcTcl6vQCIv04UbkZofiWPD1I+EySBeAPHbfzpXQiJHS50YM69w2vL/gy6BPPKVvxaDhcjEYhQib2bgIACszhPTTdRqbG1yAQWtJlhf/g73xXHS5ABbKQs1o6jCOyAUiozebAwETVfRg36SYg3KG3JN+vNG7XwBNjd4HKI+MAPcaki5HBtDDej+9gzjGKYMMTJiAAg2534XFkN/CGGhcVdE0veCk6k5bciK/1zQdyElyTqHa725UGYQDHzuZG/onpB5YrSfRCTfUNGYdY3nHPIZAJyD/6cXzwKBonPcSPAFZrPe4ZVwjjGqMXwSiR8NPDOxSCzqvD5r5aVgIiYRJMURD5e8b4qPvi7F5MC70cobxgpkpArYi+hKYqLpd8k4iN3+mHpkKqgZsm5hHPklXlv+IqZtI478T2E229bv/7pQhAeKPBzJgFSYY6sCOxtC3h3AKID2ijeRllW+8GLHiRRuvhl6TucK+JVhIYMOUIOQDUKZP8LLMhmlXBsyDAhyjwml7/+tjvX93tzcNEcDjvmDbihnROOj/aRfkc2CyZ1CGDbkDFwFxSqK5vUK19ZF/rmIRwT7LxTj2ZMFWf2Fib/nZM5KBwoQHjd3BZs9g9z/cdFsUbbK68ZK/lu1E8wzO+h60K8wGz8DVnReK8lr0VaxNdML6Ru93Gb6UWWhnjBfFOrpsrUZVkUhXGEof39q08mZki1TX+iLYAa+XJ9uKJ9m9tmHWCFHbqyZiDWoLiAfCzHCAdYYqcmvBFQD3+Dt2bsGV4HqQan0KkQ9g4TO4mntyJCS/w3V8IK6LGxwjhYrFANf8p+IBUyeRGuCtETMYVA8JR2m0MC89+B/X51w+agKOwOkfdO5uKWymigBAvqLGgK4M51n8v4/uyof76Qjz0G9pCxjShC7MYPwR+2FOWNr/D+cPTSG8CGVmgNJXt/BVYG+0PBhyGhBNw7ITPBAGWqRP/xqo0gaZgY4P7eADikATVy7eeFtu6t9oJOQc4FFRg6vPrF1MGeXT/E+gPqpg9WhtSFgAO+CIeZ3IsKxF7QQeL1czwjm3ssCMiCeqvTo5S4hzkn2Q1vnDo6HVokhqsLwTTC2AMD2RehnsWAM3So2wfthyZAGD535Xn5VaQlwmumwm0QNMCFCD3CNcvu7DMeDUyOZj77hx7h5NgHUtIs2DsViIbKbCoaVp0CMah5jAP3fTFLeBUEligHP+SSL/UulkzQ3rfHHJAOUCx6qty0cykAersXkyUTGBz/GgavhGQ+ZwPlHFinQ//QO/DBV/kHcGWcAdnU4G4gazDB4Q7yljBbrjdr/3g/P2L1kW9msBASY6FYRM134ZKYDfsiAQEvCSd8CjsPi3if20TIRGs8xOzrg/3ihqBPWDCLYWoMANvb7BbLSpyG1SCQQF5AGCdpGyf/zLmiTJ/2Fc+uIFq/TvmFrMdfOBnxyWVLW7DviVEikeDxA1WFm0XshHUB5zBrNMGE1m6PMwhRYE0zw5nQHezaDVjeIBNvjA757p8U74RjLPOk3Pd4bu+fUBRZ+vTGcvklv8o9InBP6A+pJJvhkVhBIxdThBDegxBQNqJkIAiGj//4nTtj8N++VLrfCbXw54VBBROiGwGA5bNlvC1vTOAJCizTcYZ0ogIB7mNlZQ4vtg8egYVQQS4+eKbNdiDcCasDBzRN5BF9N+WJVumHPRV+BMi8T/mMfql51C0befdEWw+wrnkwAy0E4wlhDxwTi0DTWH4w3EvoYz/ZHRV2B7wzeI2wN+nrYkTPr5Q9ipXF/zKMinj/IgMBXWxptiZv73jjo/GCaWyfKQK2c0ETL6wNIbonuxXDAJmO3GkRhtFudriaakVJ85Y38SSZRMdAuFzIU3xuXm+eZ14maj0BwyaA/cwOssqmGEBSLn1gGbeHmhNGUQMBCUbKy7iU0RVjNcCSAngEarksmdXAIVj3CnSjDJI/xcyBTGQzO6I8B52R1dy4aGguK3dAGkBo/WLVokeqss9sDu4xVyAPUGW+z8R+sfR1ksEpn3OuAFA0z4V0jqmSe+KBEai/0oV1YFfLB+eDPYMWw6QFwQd2jIwbgNs34FqLHYMIRYTvs8dkj+HR14KNNc2dT9CbH8wo9C946NWM/+IsscnrJKBDLwLXNF5onmeBhIZBnOrJzOiRzgM6RhDk39iJc0hoEx13rHC8WabWvW+cBTh7MiTZ/DrxIRHKsDdCdoNY8RVHKADz9qzQQ8e5IANQ/O+clj4BGYB8RI48ARTngrEn3ByFjRDbrPiFq+UYAWhCTZldowwv8cjM2F43l2C++b6Jt81cXs9nfYgoA0YIPgLsQNp3efztSTBG0HCb6aHWLJ5wuki7Rs+8AbSHT4WDZUTpjScXeYfpXT4cWCxgNpu1c4QVEsC5a6ydpRv6gjowPmq8knEJNADg4nnBB1OQqd0InIDYAsm/q58jMNg0PgjHwO88qnjjS+wJUB8wCg7XIJVQHUDQOLl2nh0HnazfnHxa7dZE+KXiZMEcjwoc3v6kKkX6CBALFh1MsWWuAlT99ovp4AbfIpRg21gX6Hvl0nNJrYsBH/x+mUgz5nxoHogVLGsS3yswhgxhdSqLFE50ckssBYBA1+fiM/OTQCTIDPjJdXsYiD29tb1cUoQbwuXPgcOQy6AFcytmg7cnuR9Wh6yLvBG8CRabMpH9PWwHO5dQB5T9F5I6KmMdkxg/CZuYHavJHsN70LYYo6xmsmvfnc08YKFpHAT4ILEsFR/VNvEPs/9kh6wKOrOhPzPW4WFDfWDxeDnsk8VO7gX6EwaSYWa+cUlR8XtzSWAqXA4XGcrewd3k7nEQ77mQ0O93kt+s/cVGJpOkkf801haVQcSIriEOzJ5ixCw8zGMcdc7mBvVSvNELj9iNC/Mp2NEDS9kfYQLW0IEmNIeqHl4UjKTbsybQtN14WItYK38DqTIbWGckJmQzOJc6enq4bV1TzPLiDjRX7NwETUoJO0LachmIfFnFkqFGkwOtsxz3SoUPvgjdcAsUyklDwTAwNy7VE0+WMJ6DG+Jsgw1zf6sQusfN9d+fmsLjzGzKflzKQcB/Qy9gJ7D3iea2kqYAcuFQAhz3y6YoP09cf30l3DeLT27Yygt3voJOnY39hI5Ugp8FyuQP6s77wiLxB4H+eVjogo/ib3w1iJuPBLOJV1h28Iv5WyI7saiuTrA+BRiyvmaxgWyWWIbQRgghzl5p5VgQmmXUAsshiVfPppotZyjz3mb5ZGKygQQ/AhMAX0Se/s5tTEIHEWcrNWGPOW00H0HrC9fHvWYiDBbvMXD7zVluS49+cRYs8JAWv/6mLYh5/Izl/34Y0OV5EJdrjUzGVq2AyIqQLnFqP5UR6Bze3+NpfAU4+/WDX99mO/jBZdVqOnUGeEo3kfewycSjYKkSzrJY0YDdAOlw5KVT+xPjnZw6QQuQ3OkCQaLRk9jmAcNM2XIlwjYUtv2cSSSDAzH8sW4l+doJnnvuFst4T6VJwtegTSVsXq2cYy8NRdwfF2/QQgdmeAgumM38RejJhKVHVwl7+9QpiAbzkIsYIOhEM/taEuoGSME9VOsep7LNg40IrAXhXm4FkrHqnqxtDBz9BB6h9EHC4AnIUj19oSVjGwoCvGietd0VrujB7Cs30E/efFsTgT5CT8YxPYLuUGbc4gVyI+ov/Nb6ZiwJZ4MYWcw4xWaVIjwhn+NwZCfil0jbQXYo97pYfggk5AnUJuKBDoVYAHsYagresHl+DiyW5scBeFhwSajTG3OFE8wP0vaua91cEiIdcK4pvqaMMYLCBsCX2P3jJADL10XmAMgRCDcRgsuHfSWJ2gIuLgt7IKe3CJCe13wjvjbUCp4aqhLZyjpZJjYeQNMeu0kEzwq4dzZ1LewI4ZVbgoRYVQbP3dCW5f2gQRcXkzyIAwdxV/QM1MlUTAgnSnbPkvBsDDxcxIluLugG/kCNeJLjryI/IBpZvVcjmM3ggkyTI4VzNvmybcCFrKSyINfRd6g1BFEqT0qYx4Cdm3j1vByuZCGJlHatDwGRcFHrbKZVXNGjQeyJv18eFM7igfXLCyuYg5582FwQFvkLuTWBg0IpaXJ/cIt8vx60bj0zG9O7SdfruSXjaE9M7+3qcGT9nntCLKzVYg8CbxPuUaflY00rGpiAFRgEBTggai7Aj5TCViX252Y9DSXE7IZy5BfhJ54xEm2I1NE1h7EHawLwjAIp+F7vBv5aLjaPxSdAh6cvrMLjuV86z2RpRet8eg8MDPa8CHCsARjSIA4xntShZSPzMtewPlQCWDgtlUisMk6VwZNQLC+BuRPA8ps8IwCNR3ri1ug9o0Jye2yd+2XVmjU70CPIQmFhkXgxwm1aLO7IPOVRrDu8FDK4grC05b78g6S2/qL171IZnmz6tKpkekqHCsEisPbaMiCD7xKT8WoY0IhGGSL+YzURTIRVg4vw29fDj6As0NEGu1+o9KP9PzyYy3yQc317wKWdNeQTsodIIiMR13F/lCoWvK8ydrvNaYEGXPCG0zZPX96Ap2Yr1mBmWBp0ylMmgiZuisQGKwZ3+p1jPmwcqIWAFCBn4DSSK1HZPIWnaQ+h4UPyWU7SMP/k0WgldOHuOxdUB3t/d0jAvE42B/fS4MDZLxD4Jlpye7KN8WYgk4iAmv3wKIgr8dwCCfYiW511EzoeNNpF8J4PSJHQ0F5kxKPjB85jDAnRAuN59cJDY15PgXB9hMs85YHEGeBC2LhqhfmD+Oz7SqwvhJT1jE+c3708Wue9EadVs4JhGdC+U1h1kty9KOx19CtCMVXEpZtiydY2A4gm1vsA7Voat/k24ezTxhvG3ixXwi8JZMNyO24nXEQf1L5HL+YfeVt8InDNCOfQBuuZVSjd0+iZwk8d2wxAyMO2z4gfHZHbMEhCCtzHYgFYTFvN0mNC1SL2ViUOuLIHjNrikw1fSwZiYt0H0CowuYHArRGwYBEBsnvyleaMuZzHcl9WelqISxytw0zHnrn/8JZkaYFHIOM7osjDTHD5ZERY0fMVDv2NMHIA2YiMh218YD5s62RnWMnztcJ34K2Q5GDRHero5w2aqvpwEWIFHCT8/ODP34ZXEwweZDi0LCNFE2w7Aeg3WEaABPII++iXB9CKyYNKXNyTSciJuRTEZxjwNYvZiZZEbiBZo0k/nCmz3595SFPXVuATYqEDEMgW8Q8zw5D/uAiY4ZX5VigTdmoRT8Aipb2gKy44ZuaKCHMA6iufH0b7YvEOYe9Fz2GcgMHOlinwTlCA5d4Qc9NX3l09+QVDCQDxarCohk5A7vKGj3mKlw/HeODlnk9+fgpy5z0V3HC9YgUMuBWhLVkjBBgGeIQNV9ADUWPJiQ5H+MxQ0UWc/zwnhPdgHcXcRRECiTFrYXgEqRuHRvPixlcNnjKhBIgzsmIQGJwwc5gsvHzhiMP1gLv4ZZ38BTc39/ZI5yRHalwW6GZMHOWwMnaDGOJtlfyYJh6I54lrnm9ks1c1Hck7X15iEaHJ3L4TF36xmyxky3B7AJWwV4CQDM+zBo8QoFKKCCIP8r9dz2k6oihZMotVgFGvRRnzQp6475mwljaMoignPdv4uvKdLQHKSucjUrf20vqSGy3AlQHjA5pkzuZNOG30HV+PejeuT3TyyJ7w00MFo2AcjU1m//kKApv3BdS9KStVLJf0iCBf+SlWVuLVZhCx53mDqAU/gYdbQW7efptOWNXaM6v2Tf4s+SrUCmqQiIDzgpWWbXEZsSyKDp2VG3wGYT6ZjK9f5dVdhWNtwcZM0SKobnhvDB4AwpFDvN5H/4GeI+zFSU90XWggTSpHeAkh5Bt05h1domqlFtwUiEP3YaHqjVQv0yLFnpUveoZrwhhv+1DB8OwMsC57ixDq32uldit4bJFoKTFmy8SIOU2yeOpcLRxJ1oyqWZDVqkZW633MoirdcOBs0cBjwkOrT5/tDnyG9SoQjXhJaV+2We8CmFs3tXBbV2dxhpWPeLF5SnMY1VDHdqNMsaF9J6QJEheXbNjRSbhmONtU7rJ5sDri4/QkZXlYhfhA5QH9w3wVovSzFSF8CvlqQodXjMs6lTZOq48x1Co/UM6KGqTwub/NlXODy2agRBhv5i3R/jaNQFbOr5d2HUjr1sT09/EM/xRXnaQjMRD+1QjgXHALnu/BsORtuBDrV6ymzycfcpdL/yDG4ZVJbc/VECp9O242ZQFg5IddsR6MOIeEYYkw6BiyroQvhJMrujywxgC2x6ZFaZUs9kV+cuGbWBHb9ybPuoq8W8e77ZRBEcJEVOoCAJd03d9J9AKEHlwPP1VFCKS+ZgJhaUHLt3j4sW1v2wBRoJHYP4uBuXm+/qzLE1l3ik0BvDpo10zNQDRv6ycaPNaTwGLhHgSYSMa/CDNo2XrVcYX3gnddDZ5PIMSIXuvhez5H0uhnqHCsyzwvJvwN3rPewSKf21REVkfekH6C/3oAuwsz2gVCYS8EiwrqYALZJPO4t1k2cBM1iHwHYS1X9zT121BY1mY/HkStFye+bmzhtlp2veVWb4b2eULwJQtQ9osUjB9eCGUKcACbH9gQc4anIMncaJSlXAt2+LzQGmEyWvEHkPYOVyESY49JNfxpBzAW5TB6hLvCUV8EnGkspGSDRJy4D79FBJU5UeIstvmbEuAejb2vpupNMlqTXC13xzqgU4QFlvQJ1dOSuS/LJInFmEUlDIAUcHvTW3Vwz7n3WlgskNgiD6IfEey9TxUzqqtY3yYvhAZciN+fbkwreTEn8QEweHswo3MbgLcFmhmiTgRAsYBnA15k9AFfXGnwZVlWbUkZ8ftU8yK4CWkKqSetU3eAIIHQsjAT1Q2J8KgjDfRMh52752wyCAjPPtX0liTeNxodadV3ttihDo3NIum0jDK5SneFJVu82t8yTdd/nmr6t5kmIDKnC+PIp4p+3h7YWc5a2o0/P2huv/Haj/x//3r9Jy/EQL+mtA67bos3Qdx49hescuNmey/uD6WPClD0gRI5qq4JZcGMwekHGqvUo1ZPqRpEQ+h3j1Hvtqfs50NCwJyzZ9+QzWl5/pQ02+FQoRkV/PMwbs3XukELOOp9ihyN17CQ5Fnua8fWZSmeaFttASlTkYYrRQ81KsIhvxWqASEAg1BdHgiikpDWuGDDXwlhfruni/hb9ylQtGTWg7nyU1aR0KwZNg0rTs02mvshUBDFTO5a8E+4Gqj5tO27aVff2gost7HpS4K1QZ6A5xxbh8pCPsx/VGLPT0Z5ekhyjrb5UJQmSic/VwalrLt6rFIDAP5hPNjUL/OBmmlAftWANJ/8VwHxpOvfCAh8b2Hf4AuIZGFEBlDuczxukyA+rEJ8gkSmTUjRxY5i5x5927kQZCWteyjXIEUGzmrYGzZ/efB+E2qIw/3e7Gu0H+WzyaLe14ZzxeGh2puh5zgs4d3DD5s7oh1gFja1IiGC0sL4tgbIUsIUIViJuwdw7cy0zhDmmKxlJwxoRejVIxIMTy/kiIV8vmL8Ak6gVwaFdH+IZhDr5/SXXcvaDsDLioI79oFHGZtZHytxNhwEjIJC8tbbVuO1LH38ioMKCOLpNaV+3ed0+g74QrTO9AUouWUTGPaCTItNi8eGq4GH2KbcFPHoYehLkLDJG1VaL8u8xvshnTz5Nmca33daS5TNisrgp81LrMOw2C16V4QLnDd5/tenDRlpXj8V56lmy7rY7h1gCcNOEH4X/4QUPfOwJa8M9vjaHORZSNVO7ZPO/Hhd/IdtvKsnnuFWAVb7v4LntDJcXBmw9uhVCRc8rbfXgMAy8eUMsFuqkPMFtVXzArCsd7E0IwR71BIXxMojQSe/xtvi4zDL560QiI5rNu0hxg99wEaNq2cCucK5n46B/mpAo2WPKm7dKU575lFBH4T/FVc2Fu0pebwJoqeWFZpD7EetQBrSY43x6f+ImCCKFp5vTY0o5XkXvLopqgDzz2pKYjMqS1UH2MuPhLpvnPwot9HwopowDuLaOZ14BSC2FKKBayxgDS4MH8M+JWFQoKdB5uBHZZ5ySmJ8ZmM8XmLFYLxsOpojtPHefMcQDkLcLg9AcXumImSo2c5XSz2naYFlUSMBBaOTQWer0EGK0WXnqX+b+95owzsE9bklJDMah8xc4ybXSgRl7g3ebDkaZlTa5w5Pu07ZOKT6d5vwXaaNMD3COYrfPjU+PdSfvFHxXMQSGsnptlLHXM2rXn4tnER6oGme6nExGtOaYDSNxVbvuonXYNwgSAIy12N5VLRaPxN1YGxQLSwrvKb4PQc1tI14Tqfk72AGLm9DH26CqViY5pnClc/Fwspv00eBNUfcN/jzY+DxXJLbQmTY9xTPedfckGnrTtGZYDx0z2kCl6mdnQ6PMNtdT4KnmOF5f51M/SXD853WQfH5B50PNjsL5PpTfucHtX/yOz/ZnZPbsTTnj+zO7xwOQeeP1I7iD6eVeuIUixV5bZn/ULjI7Z0lqF8LlmEZidNtZ+YtswkSSMW7ZwLEPJHhQuh7BIA2twSAnQsmw3MFym0kiWzTGsBGPN3FMl37kirxLRMI8bTaPW0cF5QLXqXYRzB+tp/cMtD53tY6vTavIpJZPxQoQoYrgCl7OpWG4iKAra0C8dd0ZU99iF04t5oUrwhWLthsADYQEoqdLQ2Zc0/zRVtTWqza8OApYLoQdhQuJNagggyGo4SIiJ6jHWmBUPPUsyTBRCQhZEG1V8RD8frcT8+/fTL1gsmiYvC3k+MiimWPdj0IFVCt9a0b338LgsPurWHRHCrVE+/bOHATir8XcdxNTrKiqIaAZxe49ldqBpG5eJj1Mn9IrGuv9SPdcoA2rfCyRWUoXb6RG6LGPlLP80aB6VtDliwv+ObtMRq+vORCaOFTIzm9ZX6KrMdaoKE21BGhQr7QrURiODhLEk9uGCQYd7YRFCTBYfkQC/Q9s7MSiAV98Va4GLaEUWzbqEu92GHoU7cYHCH+JqjEAqqEZpyYz5As+2nwejPab3DKQiJYRz0xPuVUbrFrX7adDSnWLUfFNYj2pdslAJswhrMRHms0ecyK/LD5SS392pDnlXRFt39hk8mSHmSMKQM7ZK1obTbp3gdJ8/3HV6z6Bp0hL/jeHZ5oDcG8L3Yci9z394y1IT5rQ7FwtiOCWUyTJJWL+CyNeH4T4mCbSDt0eKav8WaoI/j3p0DKJsjMZSQ4hb2+mCpAsCwz6m9VZ25DVJvWXZsVreEzbMUPLnDZ1736GCYIhqzNXmYYL3zZytLkoYKtFXqSLc3ECADHmlxrkjC3uLlJFrusZU9LLBkVuosFB7jSyW6CjfZzdohVwe/tBOr7HtawoxIeK7ZCLmYPiO4Xq2Mu1KRK9jiGTcPCh0mfaobZTnXL16ENxfQ9/AoDYXfhNXg0G/VY8diu0/qS2Tv7p9EqWBC3ZXdVuAfExWKtGN7q+ZLua9ebdS+IbMikWcqkpoWN7OUpaDShS4z7qfgb0NOD2KPGO3Qp42N3poY92dFpCjQCsMmZS8D+wPt/uiTZu060YC9YeVuH4PO+2tBDlAPWQ7enoEO3hiX7G55rdFH1E1b7ZQUc3gWu4ogf8NXttQn2COGnYIcja9gRZPuw9RV7wFAklieKz9cGeRih3avIaHS04ahCMhbaN30eS8cBVTp5LHvOiBgILWLGwBpqq/CciY9wkZ8Wf9mXh7MDb8qLhBoHZjxVtD0ZAnjbBo0CZxsAxmE5ClSHBbuf+/ToRGM4lp1huBqN80FKtuEADdR+8o+LqIqoQFnclkx3xV52RsrCPtijU6ZsUe5T9mUhZTRxiynXE+y38rz20+KEW0Nj2VjCYvY8sv35L8NhWZ+Ca6XLGkkLDgPB1JzebSV0hRtvDzGsEbTvPiePFk/N4unRt+3JUssxwy42+0BGocovsY8oBLCds/IbzmEnyNL2uEeW08khhVU3EftuAAT/tkevyuEQwd0yxuHBIvYR7twqBMCT+fgkG3rs5g/8PxFYHP0wjVeYSzLp9UXAAujj93DyC5rOj9wASFP7wC9iRMI4t+VXHXO38OJ97jQVivKwc+CDLTlOAWM0y9HWhSLHJgnpnpt9Nt22Eu7XI2g/2KJp2xUgJXaqCqSPR7p2OCdoKVKjgaXhueyGx5kxnuerg7j3vd+pow0TUbB4jySBJOSjZZ+bALvMPUHogKJ5ivPx74bwe/NyHlh7fiZ5gKrA6YAwnTadapU2v4VfPxjuRJTA3+0pOt237Nxp4Brj8iIdJWEdQn6IYg4mwR75tIDWA67i+ozcINxI7CJXIAm0svBUoTm8AKO6TlUQH/C6St30pbNiGiyDhZX/elZhJTGLx4IE5x6wU55W5Rv5Z+maJyKXfTABLOtWcMGS0Z3oPkSgxRiwTlQ3DDE61aMN96h6hAPgj2dpQDY1QdjCFZuNY8l45kgnGaCDJrTcRwVaT64R48ckrSz781/k3GAzpvLWCwI9ZRtQZDzgBkAgr7YXY6NWwUvmMzpG1p+T1Q/NA7HHRoDaRJLGngWH3/FDa3uJHsRKLTBCn5Ndss8P3qzbngyL4IIVAYXwLsOxpwmeOg0c/bv+jI5QSGTJzbJMA2qyNADu9J0gl85Qg5utugW2fWaACFWvRwH9UhR8AWFzGr/6GgRdC9miZ32YdSK8YUDIU3PFJpWwQbuBTnExftQ2TmBf9sOqOBrkCMM/raOJRrPGOJpF1g+03eJeGFSxzhIz8Kwev6xHkTlKJUlkUfVnJ4d53gkzt17HAzxEsH2QxV5cawq3KNIeR8l4oAJFA+gfMLu+Z4oI4GLHbLRp1riYPVG6k45sif+fcbHx42Xzr4HNQmTs8bsiUJmsl+H6oYjddPSRcpYd3wLnPoTcZlJLm93/ZYVrqnzOy1XaYbbjhbDG+sYngOpJnn+ritf3TBMrsI3HDnmEYBqnZUiCN5Ehj7KnwKB7hmperKAzZj7LWuAcx5Bem0msj0AWlQpW22GTmkIyTZtlTpE0DNuKXmtPU3NkIWygyVN1o3qa07ICgKtsZ54OQIEkKFbrmKt0lk0wjJ4pQhFotT2+XizG+OxbrQtUYDXsrVmwhmDbO9u1gaVqjSnfXBGGUx3Y0rFR8G85EMVVTRCtaSryjOqxhONMAbqjfcof9L3ZRbOkyY1Lg/TBqW7Lfx9l1ynYu60OOrtm9SqanHu1drVVm2KRXonvITtmb6cfFJbSNrGIq3jXT66wnuEhppDudYHwd7YB9kW9tpoJO9u5QsPXeL6uBII9IIoejNyhAMsUEOwbFsHW3vdJZeEiP6MVuGtUiAXMdzInAmdp44dax7/XP/yr8ofrf1z/AH5aOGQZO3bNxRIdH5uXnRATPUi9ZTXbimrzDcsTd/TdAhJqslEXM/FkDnchdjvKxsk5YZ5E0/1c5/TaOszuSJLQGsQ7Gy/53G3lanDwTbWHAlcfiu9RFpbZi50t3SiBWA5XsElaiZQ/HLyaEicQI/BOSMbVb49RktVQNsecmT5IZM+Jam+YJyLWZNiFgWBiu34d47JcnTASTd3OBVXDVzxChpJVhw0sPqNnC8nPIFD9+H5snrN7FST2YoZVOtgzr3JYxy0ldBpB98DZQThHvYP1EV3LdRMZ+KkdydmESLmKszXgnugxjDoHlF+HeEGQuhMucLsVz/TD10WqT7JPMHuAwGus5jwdiS+WjUqr4yHaFPSD1RB24Xy1IgkscJq2Xzu8IJyY+k0/1cl/0p6RceQ62AVLzwF+y7yJcnwiW9BtWPyQDOq+ACEytW391jQLjZmAvixikl9NW7ZsH0CuV8d/neFUL1j+TRu92NwP7o09mR07dbb3Z4t2RBSjM4vVZc+ZRQDlRfzNcdmRBbWxlfO0QrWfwXNIWd4gvuxcEjG+3EU9JJ1C2uzptHJKQhSVCM3hd6DNuKXXZufAAVsuABTe3x5OIgnsdlqxlxzgYF1B1bWJxOaVIbU2AF1l8Olcm2UaJXJ9dbP/7H7C0LupC9MNT7cshNWp9s89Fg4oKr/KfgCohPirEXmbLd4murViGyedaTo9RbJPffwa7mH7DPAMS7FEXrE0zsgR1NM4M1lgyrejRCp4R6gwGWHVxvT8uhLWqi2H6BtH3jiJYlrniZscopgywYKYM6/37rzagQhOauGusPeq8s9mw+61SiUo1QQdt3venGMhyGKU4ZEmhAfk+3K4bkvdLIjZ0abK95yLz27wNYsL+/PYDkFDOPO08lSgckfVRYd5RuvdUKSXc4lQlIuAg3iAN5xCcSs+X+gAShjL6l4JKApff5TXwYKjKm3y3F9hNePFO8O77Gx0yoVDNMvHhUAA8PXFx8Enp2OAyntS8Cwia2dFryPvHNXG31AT4NFW4kDzl7yOEDXUk+akAu5z6hE9MNsqyY26syi1v90jV/zb7hdE2k3sP+2uNdmz224z5OKMKQGP6i0QKkQ4O0KSs8KSUyzDZ+HEMzdE1apS/eVaezmvACOHwryZC24QEHAA112W6eJdkHA7WeP4GXlkSXyw6rbaAEuYQzsL/k4JGStsxzQgvmxPsFt62WBu3S50VciqGyrwgWbN2X6wgd4tcWBX4Bb7JmQDsDdx8Sxss14dU/CuMFuYnK3ejvcszZYvQLNzg47UO3Uh+hwqE7eDZ79ndqmd/bAvq+mxHJQX/o/Lopo7n2FnfJlIs1jOXLPIWw52Nm+h0plel1PgLIh+oYJwAxTx+IgcLwp1OavFgGFyxhPmYpR4i3XJaaNlx5k0Z4FXypc+mexaaSdXx35Us3Zap6MyW9WhlmnKdUYIWtD7hM+IYSP48tAc5QHPtn8mBieDfB66no6MWw6JqF3OnH3sxc24DB5p3QyMZzYHXkA9UwIR7Z/tF9q2m54man+Clnq2OmvmlM50BwQkSwNNRW5HXjoExghJRLJRHRo/laQ4LSQxerpR7JDEe+6GfHqJK8EaWOk3N3R7KZ99RqfkvpjfxDbQ5Ny8Qz+uFu/tdJdKGFzO7JCkTYfueYRp28c0LXyfYtyK2OGNq/OMnK4QPT64tzHgsgESIZswYtTga6/5s7mZynI0S/UdHQsYsBHTuTSdKBRMvzwqDXCUiOI0pGtGu0YtOB23g8LEHYdRsMUOQam8u/lKLOzGgZFKEKBoCgkKmKEheqHocFk8bL+UfgNLm9tGMkdLwR6xk96zhcNeR7fu0x45ZxjlJDsBu0R+s2jX/1NLzoZiz1fzchJz9ADjzOODHSED0iMvzWM5GcB2QPgu24cm6hkbCedIedllk08+tSckrXO0VhNBbX970U7JZv9zWo8avJXNBaACX97XYiSrdo1VtsVAdXrNZ+oALkJIcm7HerIt7cXZQexNi+uzwRAPFmWylA5cl9eZsLASewDfgJk7bTfEtVn1Ot5o+lIytvdjUf3KYIB15RCFbncLhKfp2UoEi5nSizVlt6Jx/wFgK9Bni9j4O7HKo+/HUvV6cq3Bocr2cxS0PIji4Bau6JwqoAGiudO7m8qzhp0ddw4eDMxBY3g8kY+Y1q3OMjtn/qfHc2FeRDN97/DIMTMh0Pl4tfbrcfII8qzeVgDcQPHJ85g5KI4Daf9hMcv1+w98NG72RaUOXBDT7y9ah8vbrPeYaHPHB5UzVTOY33QjUC/wxM9BAade/u4OhSVsOhHr6U67O+dW2AoB1SPHIySywwCyrjf/ii0J5m/bCQRinvGoe6xpetKqeZTWZ8uv9AXFDlNAHHY1iD3s9puACflM5QXx6uXs8edMMYCPDYJRse+1nlE3ds1uaF6AkiTnNcBibidxva7/GbNilsfx1+1CN4TTAWDnUbQ9c/vBXgxbo70NJ1+zSO8w7WLxbHewXieunuoaXBf/1fu3vYgOob2nk4sc6fXZxIw9gOdFeEMxDKCsORjqM1Lb/P0R9wh6jr7zeLWj8sqpKt72oiCWnLlcFYNgVmgeTFhatM6Arhbwp+bsQ2uKAIDPCvz+1Mvo8jqHpXoV1qIUD/1wNccdjXbKy2EOCYWxPfMmJiArpsWNnvTfZ7z3Gy6o2D+hURpK0L9qyhczQ76J0KzhGW7WXYoJO33x5suFcwTV4/gjO2mKQyT3GVnt/DnIvPUzUuZhvaPdOZK/gbdOq/8QRQ5yQWUTl92kZe8Ea2nCDS98AXLH2EXVpS104TQvTRObMCMH+U5Fp3l4q2autwHJAWtIDrRGFhabRAh6zlotHltuhwE4woaVczj27zqVw8XBDlPrBWBDdA1FaPSQyfqQGRyMLQ5XuN1YDjEujoSdiddt1pPdW6ZxP5DW53aY5r6cE1ych+c0RMzZ05pwL2ASos/lpDMvdYGf6RyR8gc4+TINzmdmS7kJ7c9jybCeaHcrK2qgsrkPicj2eNv8BJoBBwnm9jwucJZo6dGRHiwglMUsuI8FWZaneI560ruZbzg1BDq0CrRtWpjwOpjN/C4XHdwSIOWxNGQ648GpV+uKEDIfrYG2QpBOOx2mJTvNUYLWu6O3y+ePcMCiuL2d+CAkL+iCrUMEDJh/+dgkDxCczSBb3PlRvFsDAX0O9ZSiOsn67bcnqtnxVB2uPZJjm0GbZY3xhds/s2Qkq7MWIcT3GeNbYch5b/mPO2W1lmePVg7dDhMJ02Ebc8p6IRbluTw2i07A4z4rfNRBjkGVH50zb9YgrQ7vMVUMt0JEJ8hF6iCb9SKITWhbSemaLp1HdpAKzziqh+jBI8xzybxNUg47TM2TdWxEPhyMvg7ZcdZnXJC9iwXMGX70dsftoT6RMuhah7o5E+AI4tcyWBaDn5mJPKV1Dnq5XysEir0FnSuKZ74HRLPYrU0g9shXd+O3+4yCw7IkJwcf42Eq0tL0PTE/bOHB2eZ+y3VsbuO0hzHx4a3ZruxgOEzbTm3r5m+VdbKs3+Il5zTCVI7saM7NnnnZwfJ5EjRO9iKAQwTZDWKxWl93wuFg/4ad5cFaxDO47MOOihVwzsg8R+/zqi3rjzbNgdXRmmREaY2n6PuFSjgnSfYQzomOfeQYBKLMrvkzwc7O8Zwvx+6f4fYPBEnceCVKz6lWTaYrm22pcTRLEdA92qv1Mc5UuCHa2/72BRsBLB12AK6IHpPLxrPf5BxcB0bZeBftFLGUzEcB5FdetU32EBfL3Rxfibq6APIUrb602T6fKdKQvAAvY43t5q02ycV5YOqgXPusDDWr9VqN8JX96/ki2ITVUD7UYVoXoPTnsj3/btyCjcYj+RqcyzqNxy4beeXjmMnbU2Wg4xqKr/sdUX2KQK02rseyp7x3EJ+RCC9fQKvl5B+RfG3HndsRlAa3QwAImV1LJvee4uhlQkRG77wJKZfsCK3zOQfUZyqmwzBtV0qENLYWuFymdqqjVsDswj3sEuwtNlnotCKz/U6fHUsmDOdeGofzs3ed5s6ndar7DGQMW0leiLQ+gcNR2FxoHR0Ae+CIQL6HScV4Dx+YE2ZekakYFrIKsicLbAZgG7TtRL7OI4GsBHpQ0Q6i9jERnUh4SmlN9cVfqfx7QueQDi94VQnEDooMTtXalSDDGtnPCoIQ0VlsA7WH/289Y5ws4flr6HYic2tSAKDwTHzGA9As1ynutCop+cwIT/BRbaue2RZOInJ8qVxPmntOWJxJjiJbBVVxjuCQZqiCy7ZrZ0+YPYlORAyEYPSiFLt1/gVcxzl3sTXkOZO4nM1ol/QZLsRrPM+/wnQuHIDCIjpubluJvz5BzJ76hB63NNKZXO8r6xvdY4Jkd5BJq+K0brt8tmM/SnWosMODMDE04HJi3F2Dx4XP8rEv5u8dwuNQLtMP3T6+nKxbCcN+hivDb/OZieuAGQdOZP2IG95o1fc8pMqSptPbrfJTgAYfKGNvkvORt08bmRf20W1tsLTyS+Ep43ac4nO6/Z1Ivu0wR7XE6uxi/NmSaf6+7QJ87GiS5TiuGJbkdUrHipO2TjshPtuztU1WPEVkxmsjkGcsHta4zK9jpGAxXDkhZV9WhLPQY0hCQfZYnC9vnslGD0enTo8NLSs2eLCgtiELFkQPMBq/Qk3EdKESB9x3OGG/Oqzbc+U7O/QMEmp9ozM5YXVO7Qke3TksIU4fIwM9ZINZmzHqZRdet34ZWUU0HQSax0IAEB1OjHkR8fFC3OXn+VUzNbOwTnM2geTI7GXdnNPPVtWj9+1J8nmETsse1TTsOYmiqZzJzt1CUjbEkllIghJrnmdbFQdLXeM8mCG+CCs73Vs8PU59ntH0HiwFj+3nuT0rN62EhqciqOepsbFXQWRCiwyu3cdVCIPgzHSKl9UVhA/7jXyaFVx3nyFxzucbEuFDageCEfRpxrKLGMcG1ec8xoVFC5ZMrs/2UgLHZ8NdPsOassY/HOASpJy9rQcdhc/scxh5WRT9AlkejBIC2FrHrfLSRcCP8uUWwmknTE6OV2DY4FOrJg6dradH8CvXqxc/ngwAzM2nXZSNpGvyhGaCgqBKWJ4wlBMUXrMaPlfDM905LbnkXUu5HBy1bt4hoImcQ/QYmjxrH5AdbsWZyCU5Bs4HjHmUO8/MhmzruMhkw+ALhwTmpJrrg1TpH181TLNyMSAfIYTrNn+EJYR0Coo9wfKwa5xnRtgwNCqRFnri0HmHPUrhcv2iU1ocg2/eHvNn+3BcZzKkdUoxXgLkZhcMlOq4M0z74k1ZM0cqP7aGmvZXgHm4pENx0305TeFk+e4zsvR9bSgbU/VOjEdsFwwy3za2n2eZfOYZ4Juw5mBvG07xRMf7Q2xYRCdeWpwRg8fjzuD0ESLTm8MjrwotdEhlQ+ZDf15PU6EhzYnh32mEdcKMFd3bigUnp8X71NF/tjlLM1gOpCjUaJuHRHR/Z043GGC3yLf9n8nQnREMHi6fKv79yUBAtMUF31a4qbqLbIS1U7tvHMXiV+dOOtLD6hNYKiE/zuDJOD+NMjZ7Ps9P18/IH2wEsL+cYgSXxtijE8EsNTwtzq8A/d22xz/2CHT2xOYWJCOOC2EGX+4/qsaJ/Q7bQQBZoYcGulWu1eovrNE5rtspR58YbMUL7KA6o/h965lruTBM1NYGN88AnDtgA8FhmQjN6SDoBeZDBLOPBbDjHELwVd5cfYmOtpUCTQI32mdI0hfSZW1vP4/VIapDxl6k9nvLABOoKfKuM7NCPLCjvyK1ROqebc947AhSt9/XuO2UKI+jMnwCDjQu2+4EPKlWNNtn2vIyvEkvdqTEpQ4Njt07z/0g4NjCMB2+DnTDmBD0Tsf3IVBzgNvQ6/YzhlNarrpSlUGfu+fFzQHF88Ug48etoWA8gB/wMzvMaznV/A7wtqrq7s6Uen12hF1QvNdtIVpYBMkJ06sFTMRRL7O4lhU2s8CB2HYGaEqwT/XfaYHycMix+X34kMJMJB0OnLc40rZrKfgHrXFWg6jN+/u4HMT64c/bKGeWDQJjgYxj7506higkDFnFA4KmeVt5DFe8QBYY9Ld6IM5j8HbTzPEN060d8wUoPXEkfi0PPQikQP7y2Qjs2mzB/FVKlnvYHx58OA9s3PySTUfuR8F1LA83CVi4MwdL329wYpWx3j6+n/KoauMprNbusn6e8TMOhXXWPQr9cQd9LBDs0UknZpnt04A9qKzY4ona8azTqdd5Xs4w7aazQIwkFCDpABAgwrmcr+c0spXZzItmuBsRoDoUGg2f5DanYdE5EdFILmqdgvBmdtDxdMNX1WoNvQ9uEQibdaDJPDXrLULb/DDHeSbaew1jwuNkfvjiyatbtGdBuPXDyNH7V5nc58D+4PS1BwL5Fojh6ZE5U7QxyN5fH5PnU8hYTZ8QUcyrByKoTdYBH+4PwYnI5qMeszOybSa1NspsMSCTnMd+fUKyU358IpUPEwiOkrWb/NThY1CFNVbk2FPsOB3ErJVqb/I5KpaPyKzLNZ1Ac0odo49uyEFQxq8QCfeZxDzETZ9T1qWzDnTcHqC9fFr0jM0n6OwFrQlODSv2fSVrVsrphMVJUNdnyCCY0IFMI96Q2Jw83j7Pl4yOWK8RefLBIZ/wtXxSHjaUm+oZ77QrJWPxXE5xUCwK1AesbKOSTfGOfPm6RPnMgcV6zNUSwy2sHiXaO0v4SIMXOeQSNDxJ0tP4fabaqfFh/staJAICv/Y4bzt9VxYpJFUO0XYuTzq5/nkou/HA1OxJLHv2vJ20B7Np5rPGMvBxXyxFgvotDw6XB+jFbs+5+liO3LfQGedIp+resrG57ujc3Y3dOsYfY+/evoO178vBEneyBAMi5EAR9bQdLuO0cym9zzRVrMsDTsNgRm56G/bQmCY7QYPFvsEW3pwLJFZ9TkaWIjhzxEfvQWKcAu/4cXjVNgfpWSFrXWdBqvuAh4odXwg6foXP5PqwG+xoBnz6tIRbtM8iOaSpOffUsbmOknjOuIzpufAcyrfw5ssRryyJFPPXs9MwyGkTuM86eTE+YNb5z44r8fzEdoTPyAv4PU5XmV124OgCq3duMLnIbCto+8gFAFmk1Dqzo3zonOyLBeRmpaMeRcFJzT6ZPvuei41nS57XGu3guY/TWQBTKHeFV69tTZkH+yomi2zOc9ueHcbnQx/0Fx+O1a/3TCi11MhOYGIHrLH7bCerstlDJ8dVJ+h6zOfDPRwlkp05WnwKTPIuTd1cI7pXzzPyT5leNlw63gJG9qAClHCPLftQis8ChjPPzw4561EcVplfR/Ze2IlPnYOaxAwn83gZcIgoC/T6yVqDbPY8nKddzWov+sNKB4PR8/j4Fy5s+rA7a3F8opZ1mymcZFx03Nupt/YxdP0tUH689R1nWpWHNuwKMuWR0QMHM67ru0/5KfvzhVVOVt/SN5DUJ8Q5UMaHPVhqNMyUmOqLxB+WmY/3sYcIe7a2XY6XdiiibW44689MZGwaxvu5JyOcxzjsbf/TY17fMjubCyCbzmm4mweb4zqRO/lYYehw8rGPy9Qw6H6ysdsH1+K257A22hZjh2UxkxN9hA8MEZeE+DprbNkskaMP8+0+oc7RTEfwLivGiIReCnbh8YHsvpsycJw6EIsf5VPSc5lWULf6YLvnHGDYeK0cRJ7ATx7iBcyL3fRJIw4/xPM9KX0j7lrsuzlnlldDyviIhec2LY25fPFMBw/LTr5gfYRThwg2xAe22tkE6OzhU+HWcv64o+S0o21foMmje5xHe5bwFJ+rMQFlx1h9BHMf+uRjHGBR+deoqfv0z5iYPX4MHjkSx/NqWGUj2G2HDDrgKxNpYXHdZpXiQ5kLuOCcsmbwRQIaXyyz8DPfeom86Vc/I4rqtawfNeq0Uj2A3bI4x0c32bnsQ17sQD1PubKpGNsw7zz2RbSYRhWHnpqt4rctmyZ8EoAmxBb5ZrqbkB3nMx5baXwixOcjHh3DiTpsFugCls2uMqtwy7bSBDn3sDh8EgBPwPhTTyPB5PndnV7FEvv/eZ8xh7N97CKMSCGfAFsJRh6bnTGrYzq5PXusWZ2CGIfPd4IF/r3hkdW4tkd4Pn7rkYGbqHLeEO8C6zBYE2nETZOgTjqRQTsICXeK2eecdsc6xLauU1WrBZitOgPba21OF/SQzvG3VkxguXtGB5WdpyCdVv7uWCYTUyLseC4HWFrtYMtKdAoQisQJ6+NzNCvY8ZyRKw48xQCha8gcOKNZVCuKgk8pfXzCQFXKrvA5E98O2ezjB6r9vijf5TN/G+K1OPy6O8oSUnI7Gr9De7F8I+DsnmWj9T2fxyiLHIoFgI+68J574SxO3a7L58P4bJ0A/r3qpSRB/EzxeRqRfQITxhiAVpjFmD7cbJu29Rjfkv0zXu07Bx+O5xC+fW5WPumoecoYEV8Acrv2PLNMkGfwRVZKUMSJsyUiHjHG19mHqTu+3x6THs6zSaBewIfD+ooPdA72HH+VuxzWJ9mpBut3jHy2rQzF3CroamIfmLudMMCNQdRsr/A5j5s3w2BZvevMASk/lcSOLDzPJ4WOL+NGsRlyqDzfXj21Dz42OPtoicfkcbSmfJ2z24ulHo7E5iqQihakTx+TGpDeMz1C2Uf4zB315fNULPW52+NjqZ97+1ynYBHU4yNTnTRzpjeic30Gefl5xtgpH/LJAsQY47wdaWCVA4JWcTTpMN3twTT/smu7V8+++C7rADkE9x7HCVhw8npg7IOpHXXo8xp9qOOp/vFhyL4Yk37U09FBkzbH3agDBxuwWt24Oka1NWAlj0IRFD6zaZYGeXriOcC3WJ/lrcuWu+887ayfrv7hgaXj8py+VyJmd5qHIc2WrjjMwGMgKBfU2No8y+Ur0vi+cVuj8+V4g20Bk0+sw4Y9huXCiUGOTXfqGpTZJyF5+j5fn7smf9v24zqx/wT4uupVPHIgNHgW5IFuP7M9voRLfdVHD8GIAffsAcR9GveTD0AbBdmHA3yPeeEXLdKGm4r7oZ7PXEf8BCJvnRqv9xmNVt1GHw59hp1jH8RtK1Usi2FlfYbtDD6pwnz7zhHEc5zzrqdQDBamhSzp55nzZasZvoE5Lgnk4ZTJs1vlfEnTcXzz1JdEH5xmkiDyBq9D8rFjbIcFl/LOxfWbg7QmyhHWDwSBz74/bqf7nGOs6vU0wQfNARnOrLtt5ahq6/c8xWPeOTncs0WTfFYr3R4ppzOgTzzBRy6rnACE88y0Jw2WMqmni+N9baaxe8CGfomCwwLs/fg8ezYVPBxkritagWCTggczPTvE34duJp+imB2wdSr1X2vZAEFj6DeAzWKaxIdwPzA3u3kiouM6D1JKX5qefMtKudJlhpto2H1MwZudtwXEquec8wuasWOArGO+67Qji+W+Ph9/+4qTFiW0DiB6SNKG0/htz0W4wpi2Rl83u5ES8gPdyfZ4PoiRGbviZbmlNYaWfFoliLBYpjSfd3YF4OuEFLuX1vDJu2vY8u2D6t/2ZWQIdu5Um3wVC9W6c+Rt8bPZGmdjyZ3YxBrbMXtb7vn6CHEcw/P56pAVbKlZOkC45IrS9VkMdduUZ8NEsN3ktkN3wbbl7GBpQwC2ng0+noo48jiar8dj7LUAZv8PPPfvEa6O0U4AAAGDaUNDUElDQyBwcm9maWxlAAB4nH2RPUjDQBzFX1OlohUFO4g4ZKhOFkRFHLUKRagQaoVWHUwu/YImDUmKi6PgWnDwY7Hq4OKsq4OrIAh+gDg5Oim6SIn/SwotYjw47se7e4+7d4BQLzPN6hgHNN02U4m4mMmuiqFXhCCgHz0IyMwy5iQpCd/xdY8AX+9iPMv/3J+jV81ZDAiIxLPMMG3iDeLpTdvgvE8cYUVZJT4nHjPpgsSPXFc8fuNccFngmREznZonjhCLhTZW2pgVTY14ijiqajrlCxmPVc5bnLVylTXvyV8Yzukry1ynOYwEFrEECSIUVFFCGTZitOqkWEjRftzHP+T6JXIp5CqBkWMBFWiQXT/4H/zu1spPTnhJ4TjQ+eI4HyNAaBdo1Bzn+9hxGidA8Bm40lv+Sh2Y+SS91tKiR0DfNnBx3dKUPeByBxh8MmRTdqUgTSGfB97P6JuywMAt0L3m9dbcx+kDkKaukjfAwSEwWqDsdZ93d7X39u+ZZn8/xrtyYvDRfG8AAAAGYktHRAD/AP8A/6C9p5MAAAAJcEhZcwAACxMAAAsTAQCanBgAAAAHdElNRQfkBwIREg+sh0OqAAAgAElEQVR42uxdd3gU1d5+z2zLpveEkIRiCC0QaghdOqGINBFEUcEGIoIFpPhhuV7Acm3X3pArKgoiICAivYl0EkpCSO9le5vZmfP9sTvD7GZDUUTU/T3PPJvdzcyeOXPmnffXCaUUfvntYrPZUF5ejurqatTW1kKn00Gv18NkMsFsNsNiscBms8Fms8HhcMDhcIBlWbAsC47jwHEcnE6ntPE8L73yPA9BEOR/E0EQIN8opeIroZRC3AAQ71cAoJQS99ClV6/PSF5eXlxUVFQTjUYTp1Ao4hiGiWEYJooQEgkgAkAoISQUQBCAQABaAGr3pgDAuI8nAOABsO7NBsAKwEIpNQIwUkr1lNJ6Smkdz/O1HMdV2e326oqKior09PRq93GobGvwnhDi/Z34GQBQQkiDV9lGGYYBIQQMw0AQBKjVajAMA4ZhqEKhgEKhAMMwEP9WKpXSq7ipVCqoVCqo1Wqo1WpoNBpoNBpotVpotVoEBQUhODgYISEhCA8PR0REBKKjoxEbG4uEhARotVr/zXQdhfiB7bcJy7IoLy9HZWUlamtrUV9fD71eD6PRKAGa1WqF3W6H3W6/IqDJwUy+CYJARIDzAjMPIBPBSQ5S3oAlf19dXZ0aHBycolQqb2EYpiUhpDkhpBmAZgACbpJptgMoppQWCYJQyPN8AcuyF3U6XX6zZs0ueAEcvQzgNQBAEfgIIR5ARymlSqVSBDYJ0OQgJ9/k4NYYwAUEBCAgIACBgYESwIWGhiI8PByRkZGIjo5GfHw8EhISoFar/TeXH9j+HCkpKUFlZSVqampQV1cHnU4Ho9EIk8kkAZrNZrssoHEcJ7Ezb1DzBWYiKxOBTbx2PoDM431eXl5sQkJCB5VK1UGhULQnhLQH0BaA5q98DXied5SWlp5NSko643Q6z9hstpzc3NycHj161LqZohzUBDdT82B23kAnApsIaCKbuxLIiaCmUCigUqkuC3BarVYCuJCQEISGhiIiIgJRUVGIiYlBfHw8kpKS/DeZH9hunFRVVTVQOw0Gg6R2ygHNbrc3ALQrsDQiqp1egEYaYWUNGNnEiRMVH330UfeAgIBuCoWiKyGkC4DUv+O14HkepaWlaNasmfdXeYIgHOM47rjRaDx2xx13HNu9e7cgAzsJ9AghgjejcwNbAzbHMAz1AXBQKBT0SuxNDnAiexMBTlRPw8LCGqincXFx/pvOD2x/nBgMBpSVlUkszZfa6cuOxnEcWJaVGJoPOxqR29F8MDPiw14msjLyxBNPKJYuXdpHo9H0ZBimFyGkJ4Dgf8I1uQyweYuZUnrI6XQeMplMh2fOnHlozZo1Th9AJ4jA5gY06sMuR72ZnJf9jXrb30QGp1arJYBrzP4mV09F9ta0aVOEhYX5b0I/sF0/4TgOJSUlHixNr9dLLO1q7Gi+nAMioHmxM0nNlDEyyIGMUsrU19e3Cw4OvlWhUPQjhPQH8I9c9dcAbA2eU4Ig7OU4bl9VVdW+5s2bn3ODWwNWJ2d0InuT/y06GOQgJwJcY86FK9nfRPYWHh7uwd6SkpKgUqn8N6Uf2H6fVFZWorS0tIFzQK52iqAmsjSRoV0toPE8L6mZgiA0CmZWq3WAWq0ezDDMYAAd/FfndwGbt2Q7nc4dRqNxV3R09D64vLiCN9g1BnIMw0jqqkKhoFcLcCKDE9mbCG5y9dTbuZCYmIj4+Hj/xfcD27WLxWJBcXExysvLUVVV5dM5cKXwDS+1k3g5BuTsjHgxNAnMbDZbllqtziKEZAFo6r8yN0TKeJ7/0WQybY+MjNyOS+EqPkFOpq56qKoii/PyoNIrsTdv9dTbuRAXF4eEhAQkJycjKCjIf7X8wHZ1UlJSIrE0uS3Nl3PA25bmg6VdCdCIt83MYrEM1Gg0txFCRvnB7KYAuS16vX5LTEyMyOQ8QM6bxclArlGAUyqVjQKcnL15Oxe8bW+JiYl+76kf2C4vJpMJRUVFKCsr82BpBoPBIybtKoNsiah6iurmZdgZo9Pp0oKDg8cyDHM7gHb+JXlTylmWZTeWl5dvbtmy5RkZwPEykKOy1wYsTq6muoGNXk1wrzz2TfSciuytadOmaNasGUJCQvxXyA9sDVma6CDw5fFsLIRDBDVfgCbPEpADmghmmzZtChg8ePBEpVI5gRAyzL8M/zoiCMI2q9W64a233tq4aNEiGwCnHORkLE4Q2Zu3muoVHuIBcCK4NRYa4u05FR0LfvbmBzYAgNVqRUFBQQMHgcjSzGazZEu7QubA5QCNyNVOvV7fISgoaBLDMJMA+FfiX/yZyHHc2pKSkvUpKSlnZaoq762qygCOXgngLhfYq9VqERwcLLE3b8dCixYtEBgY6L8y/1Rgq6iokFTP6upqDwfBleLSZKDmE9DcAbUe6qbVas1Sq9WTCSET/Uvub8ni1un1+rXR0dE73AxODnC8LzVVFibSAOAas735insTVdPY2FhJNW3SpMk//poo/0kny/M88vPzUVxcjIqKCiklymAwwGg0wmKxSKDmcDh8Zg+IgCazo/kEtLvvvlvx0Ucf3a1UKu8G0Md/+/99hWGYcZGRkeN4nt9vsVi+Hjx48NrDhw87RZCjlDbwrPI8TwVBoGLiPc/zVB4S5HQ6qTxcSNQQ5KYQ8aErPoStVivMZjNMJhNuueUWKBSKfy5jKyws/EecqNFo9AloosdTrnZezo7mS+WUsTOyadOmkNatW9+rUCimwR9z9o8USmm2w+H4cv369WsWLFhgJITwMibXQE2VsThvBndZ+5tcPRUDe8Wc0yZNmiA5ORmhoaH/TGDjOO5vf5JlZWUoLCyUvJ5iBoG36tlYGIeX2klkaU8SqGVnZ0e0bNnyfoVCcT+AFP/t7RcA+Xa7fdWmTZu+njJlis5LTeXd4CapqO5gXyrPQfVWT32FhXinZEVHR0te0+bNm6Np039e5NDf2sZGKUVubi6KioqkgNvLeT29MwgaUTtFLydDKSUnT54Mbdu27QyFQjEDwC3+e9kvPuSizWb7/Kuvvvpy+vTphssBHCFEkCXde2cuUF8ZC415TcWA3mbNmiE1NRXubAk/sP2VxWQy4cKFC5L6WV1dLXk95QG3jeR5Eu9UKC9PJ7NkyRLFkiVLHlQoFA/CH3/ml6t70J61Wq0r77jjjtVbtmxh3QAnqagygBPkHlTv1CyVSkUbyzeVq6WRkZGIjY2V1NKUlJR/TMzb3xLYqqqqkJ+fj5KSElRUVDQI5fBOXpfb1LxYmodjQPRy2u32O1Uq1SMAevhvV7/8BoA7bDQaPwsPD99ICOG8AM6bwVFZBgP1Zm/eMW/eAb1iSEiTJk2QlJSEW2655R9RDulv5xUtLCzExYsXPeLT5Lme3snrcpYmOgrkaqfMjsZYrdaBarV6pjvlyS9++W1sgpCMsLCwDJ7nR1VXV3/WpEmTA3KA8/ai8jxP3QJBEKhX/T7xPXx5UUWziuhFtdvtsNlsaN68uR/Y/ir2tPPnz+PixYuNOgkuU9m2AUuTq5319fWpoaGhswghD/tvS79cL2EYZkR8fPwIlmU/ysnJ+bxz5875MoATw0R4AIIgCILY38K9UTFlTw5u3iAnOsC8wc1ut6N169Z/W7vb3wLYbDYbzp8/7+H5lMenNWZP82ZpblCT1M57771X8cEHH8xWKBSPAUj034p++SNEpVLN6NSpU5bFYnm/b9++nx8/fpwFwLnZGyOqqJRSqRim24FF5cUVvAHOu1qzHOBYlpXA7e/YSOYvD2w6nQ65ubkoLCyUCkLKq3L4KgbpforJQU3O0hhKKbFarUM1Gs3jAAb5bz2/3ABpGhgY+Pyvv/46oLq6+qMmTZrsJYSwMvWU4FJOKvVicHKAo76aAvlST8UogNTUVERERPiB7WaRqqoqj3AOMYndV4VbWSgHEem5D5bGlJaWxsfFxT3OMMxc/73mlz9BPR0QHx8/wOFw/HfHjh2fZmVlVbrVU0YOcK4K5kRojL3JVFVcjZqampr6t3Iq/GWBraSkBHl5eSgqKpLCOeTNVXw5CbxZGs/zcpbG2O32CSqVam7pueNdyx3ByEhvdVOdc319PWpra5GamupHgL+A6PV6bNq0CQ6HA4MGDbomg71arZ41bNiwXnV1de9HRUVtIYQo3OopJ7O9Ebn9TQ5wLMsSlmUbtb15MziO41BTUwOz2QwAiIiIQOvWrX838SgoKAAAxMfHX9X5V1RUwOFw/G7nBtPoN6wenyx7EoP7ZyIjsz8eWfgqykzOm2LB5Ofn48yZMx5xapfYmg7FF87g2JFfkZ2dg8LiMlhsdjgcDiICnDs1igiCwAiCoKisrEx2Op2vqVSq/wHouvo/83DPwnduinM9fPgw3nzzTQDAkSNH8Nprr/2pY/g9wnEcBg0aBIPB8KeN4XrL2bNnYbfbG3zudDoxYcIEFBUVITAwEFar9ZqPTQjpHBkZ+R7Lsot37NjRjFIaQCnVwNWYWkUpVVBKFYIgKARBYHieJ06nk7AsC5PJBKPRSEQvqDyP1GAwSFWiKyoqUFxcjAsXLuDQoUP4+OOP8fbbb//mdfbSSy/hzJkzAICCggKsW7cOK1aswKpVq65qPWzZssXn/17remgU2LZ98Cxe/HQ3Rk5/Bv+34D7k/fgeHnn24z99IZ07dw5nz55Ffn6+R7Vbka1VFechv6QaysBQhAQHgHNYSH29noiMzc3UiCAIIksbFx0d/RUhZObN+NT/8ssvkZGR8bcYg0qlwvvvv/+bui7dDPPgS+bMmYOSkpIGn+fl5YEQgkWLFmHy5Mlo167d75m3B/r37//fEydOZAHQwtUTVgQ3pRvgGBHc3L00CM/zcDgcxG63ExHcLBYLTCYTTp48iYKCAtTU1Ej9PViWxZgxYzBw4MDfNE6TyYTdu3ejTZs2AIDMzEysWLECt9122w1fD40CW1FREUhEc0yZNAYjb78XLy9/EcO6JMEJQF94FHPvn4CMjEzcNvkh7D5XBQA4uv5NDM+6F3k2AHDixYdG4uFlX7p5+UlMyhqOd7/YiDlTb0dmRiY+213oYmCHvsc944YiI6MXpj76HPJrHSJtxNZPlmP0gL7I7DsYDy54BafPnUNBQYHUDs+7J0FFeTXAqBAaFACVWkvUSgWo4HSBmtNJOI5lnE4n02PAiPBv9p15QaVSfQmg24Fv38Tw/r0w8LZ7cLjQJKeuHmNY8uYaOG7QTWM0GnH27Fn06HEpDthqtWLevHno378/HnvsMTgcrtGYzWY89dRTGDp0KCZNmoRTp04BAL744gs8+uijGDBgABYvXoysrCzMmTNHbLqMtWvXYuTIkcjKyvL5pPQ1hl9++QXjx4/H4MGDsWTJEthsNgBAbW0tZs+ejaysLIwfPx6//PKLtM9dd92FkSNHYs6cOZK6I8qnn36K4cOHY8iQIXjttdfgHTTuPYbvv/8e7733nvT96dOnsWDBAokpLVu2DEOHDsXYsWOlMZjNZkyfPh2vv/46hgwZgokTJ6KoqAgA8MILL2D//v3S8VavXo3Vq1df9trs2bMHEydORGlpKebOnYuJEydi586dAICnn34aTzzxBCoqKjBx4kRMnDgRLMte9njr16/H66+/jhkzZqBfv35YunSpNA/Lly/HqFGjuixcuPCDXbt2zf3www+jKaUBADSUUhWlVCkIgpLneUbc3MBGDAYDamtrUVtbS2w2G3Q6HU6fPg2j0Yjs7Gxs27YNR48eRWVlJcrKylBQUIDi4mLU1dVJv5+Xl4d77rkHgiBI4509ezaOHj3qcQ4bN27EqFGjwDDMFdf25dbDuXPnMGHCBAwYMACffvrpFdfkhQsXMHPmTDzwwAPo378/Fi5cCMh0c4+t8sQG9GjfEp16jcDSV97B6RLXiVLegFkjOqH7sGn4btN6zJpwK1r3mIASO8Wuj+ejRYs+OGmmoJTDI4NSMGz2W679qvehS4sWaNm6E2bMm49F85/E3nwz+LoTGNC+JUY/+Cw2rF+N0T3bYtTjrn1OrFuGli3aYv7rK/HK0jlIiolC1rQnMH/+fEyfPh3jxo3D4MGDkZmZiY4dOyI1NRUaBQFRahAZGUlCQkKIVqslarWaUSqVCuKyKapXrfl28OIHR//Ysm0feqyao47SPbRrSgs69qGldP3alXRYt1TaesTjlFJKT6xbRlu2aEufeet/dO2nK2inlJZ02boT9EbIqlWr6Ntvvy29//HHH2n//v1pWVkZdTgc9M4776Rr166llFK6YMEC+q9//YsKgkAPHz5Me/ToQa1WK3399dfpf/7zH3rkyBE6dOhQynEc7dWrF62traUHDx6kI0eOpHq9nur1ejpkyBB68uTJy45Bp9PR7t2707Nnz1Ke5+nChQvpG2+8QSml9P3336eff/45pZTSQ4cO0X79+kn7mUwmajQaaXp6OjUYDNLn58+fp/369aMWi4VaLBY6ZcoU+ssvv1x2DDU1NbR3796U4zhKKaVLliyha9asoZRS+vrrr9Onn36aOp1OevbsWZqZmUntdjs1GAw0JSWFHjhwgFJK6YoVK+j8+fOleZ05c6Z0/GHDhtHCwsKrukZDhgyhubm5DT4/ceIEHTNmzFVf648//pjedtttVKfTUbPZTAcOHEh//fVXSimlNpuNGo1GOmPGDPrjjz9Snud/KiwsnACgGYAmAKIIIWEAggAEMAyjJoQoCSEKtVrNBAQEEAAkLCyMxMfHIzk5GQEBAWjevDkyMzMxePBgjBs3DtOnT8f8+fPx0EMPYeDAgTh+/DhYlgWlFGPGjMHOnTtBKcW5c+fQt29fOJ1OD8wYN24cysrKGmDJp59+iueff97jM7eqjPT0dBgMBunzjz/+GGPHjoXZbEZVVRUyMzORnZ3tse+qVavw9ttvS+9Pnz6Nrl27oqysDFarFRMnTmycscWlj8bWrd/jvqwO2P/NexgzeAg++fkchOID2HHWgIkPPo3bR47BosfuBlt9FLtP1l8VCxn16HJ8+OoyvLjsZfRpGYSCg7tRaGUwfd4ijB4zGW+8+wFmjuvtegJ8vwE0LAF8xRnkFBkQoHDi9PGjUpyad0iHzWYDTykg8B6qp8vrKTAUUMx+Y9PDUyeOX7lk/syhGnsZtu3ORf6vh1DPB+DBp5/BmHH3YGCXhEtPoe83ABFJSFAYUWULRGIEg1279t0QxrZ27VqMGzfO47M2bdogISEBarUa3bp1Q0VFBQDgp59+wowZM0AIQffu3ZGcnIwTJ04AgFR1NSQkBEqlEkFBQWBZFps3b0aTJk2wdu1arF27FjExMTh06NBlx3D48GF06NABbdq0AcMwWLBgAcaPHw8AmDFjBhISEvD2229j586dqKmpkfYTf987IDQuLg4qlQrvvPMOTp8+jZUrVzZQMbzHEB0djS5dumD37t1gWRa7d+/GqFGuZJAffvgBISEhWLlyJQ4cOACGYXD+/HlpDD179gQAZGRkoKrKpWkMHDgQOTk50Ol0OHnyJOLi4q5HW79rlu7duyM8PBxBQUFo3749KisrAQABAQEICQmR6qsxDDO4WbNmb3zzzTdT3KppgKiaAhBVUwUARsyeIYQQtyON2O128DwPu90uqaZ6vR51dXVSYLvRaERubi5OnToFu92OGTNm4PPPPwcAfPXVV7jnnns86r3l5uYiJCQECQkJV3Wuja0HAOjcuTOCgoIQGxuLYcOGeTD/xu6LhIQEJCQkQKvV4q677moM2FhsW/MZ9hcyeGzxMmzbsxe3d1Dhldfed3lQAGg0ruatKvcrx/LufQXIGGsDCfeKl2E5BwAFNBqXg7ZFeh9k9e3sopwmCzibHhfOn0dhYRHCk1IRG6K8XPcoQkEIpYJUO83p5Bie55nivO0JvVq2eC1Aq30DQIJSo4YCAMez4Jw8AAWUmoZOYrvNDiXDo6ayEpWV1eiSNQVZvdreEFtieHj4ZauhMgwjqQscx0Gj0UjfabXaK6o/NpsNoaGh0jZ27Fj06dPnsmPgOA5qtVp6HxISIpXFee2117B+/Xr06NEDI0aMuKrzDAsLw8aNG5GamopvvvkGw4YNQ1lZ2RXnYerUqfj222+xfft2DBo0SAoytdlsiIyMlM5p7ty5PudQPndKpRLjx4/H999/j3Xr1uGuu+760213hBBcIY87Qa1WvzRz5sxnNm3alCiqpjKAE4GNcduTIQgCcQfnEkEQiMPh8HAq6PV66d6yWCwoLCyUwK1v377Iz8/HuXPnsG3bNkyaNMljMGvWrMGECROu+zyo1WrIS6tdzX2h0WgaAzYljv+4Ek8teBZ7T+ajtLQQerPrxlE064ZO0Sr8sO5/KCwtxaovNwCBLdGtYwziY+IA1OGXg9nIP7kLF6r5Kw68ZZcMRChYbPpmHcpK8/DYuEz0vudFmEwmNEm+BZbaKgih8bilZTMoBRYBakWDODV3xVvCsiwUhABUICzLEZ7nGUGgzNCHX+3XpHnmh5ntw2bu2rgG54pL8e2qNbCQEGR0aY1bOnREICxYv2odCs8fxcm8Oml8fXr3BKfXISEjCw8/OA1t4sOQkdHlD1/Ya9aswR133HFNT/stW7ZIbvbs7Gx07NjxivtUVVVhzJgxmDBhAlq3bu2xYHyNoXPnzjh69Cj0er30Px988IHktR03bhy6d+8uqQlXkl27duHll1/GbbfdhldeeQVt27bFwYMHrzgP3bt3R2lpKVauXIkpU6ZIn2dkZIBlWUyYMAETJkxAQkLCVQWfTpo0CevXr8fhw4evyXiu1WpRV1f3p4Bfeno6jh49OqNXr14vFRQU9PDlWACgEASBEdmb6DWllMKXU0HM1rHZbFIJ/by8PJw9exaTJk3Co48+iqFDh3pUCeE4Drt27cLQoUOvy3kVFhaCUgqHw4Hdu3ejQ4cOV1wPOp1Osjf/9NNPjcWxMXjspf+iev5CzBg3BBwFolt2wcsvz4FS3QTLXv8XZs57HgP7rYI6rCmeeeVtpIUCGHwnRqV9g+UP34ZPW/dDi6Zh0F3hJDTJ/fHK0gfx1L8Xoe+7HKKad8VzC6fg9OnTiO44BJmtL+Dnbz4D6+Sh0IagZYoGAXB6x6hJ8WlQKAgRBCIIPAHADJ74yNTX/+/hZwjRpix4+XUUPTIPI27tB0VABO5b+hqGttICGIolDw7Bsx8+jR9Xx6J9wqWqo8Mffh6Plz6BNx6fguU8EN2qK17qPxlA+B+2YFmWxZ49eySD+NXI0qVLMXPmTGzatAkVFRVYunTpFW/o8ePH4/Tp08jKykJMTAwUCoXk5m9sDAkJCZg7dy4mTJiAmJgYAJDc7jNmzMDChQsRGxuL2NhYCIIAg8GAsLAwjBs3DvX19TCZTBg1ahSCgoKwZcsWZGZmYvXq1RgzZozrwalQSDfI5eaBEIKJEydiy5YtSEm5VNdz4cKFePTRRzF27FhQStGuXbur8qaKPQNatWoFpfLqwzunTp2KOXPmoHnz5hg9ejSmTp163dfD888/jx07dqC2thbHjh3DSy+9hPfffx+tW7fG1KlTMWrUqIFxcXHNBg0a9NbPP/+80Z2GJa/szAMApVRwx25CoVBQkRCo1WoaHx8Ps9mMU6dOSWzxs88+Q/v27TFy5EgIgoA2bdqguLgY9957r8f4fv75Z/Tt29eDyQPAhg0b8Nprr8FkMsHpdOLnn3/GI488gkmTJjW6HkSQGjt2LPR6PXr06CGZDy63HhwOB+69914p9OaKZYtYhwlWB4PwUK/O04IT9Xo9gkOjofZYBwL09XoER0ZeU/SvwDqgN9sABjibk+MRzlFdXYXaeh0cNluDbAI5qMmyCJhpd09VLP/PfxdEhwctgkcgsmt82uBwaNSehNVm0QPKYGh9qKSswwKjjUd0+B9favns2bM4cuQI7r777mvet66uDqGhoVCpVFe9j81mA8dxHmWkrzQGlmVhNpsRGRnZYIHZbDaEh18b8JtMJhBCEBwcfF3mwWAwQK1W/+48yClTpjSIu9u4caOH50/skXGlcy4uLsYjjzzi8Vn79u2xYsWK3zy+mpoaaLVaMAyDf/3rX87ExMTX3nvvvY9OnTpld6dkce7N6a7zJhBCBHkhS41GA7VaTcXabmLpI7FoZVRUFOLj42E0GpGTk4NVq1Z5nOunn36KPn36oFWr6xfQLibry0NBGlsP2dnZWLhwIb777jvo9XpERkbeXPXYdDodcrxATXQSiM1W5N2jZKAmr5vGVFZWJkVFRT1DCJkBv/jlbyqCIODuu+9G06ZNwfM8Tp06ha+//hohISErd+zY8eHw4cNLCSEOL3DjvcBNKj0ugpu8G5YIbqWlpSgoKMCsWbPQp08ftG/f/qbJLxWBbcOGDZcY/c0CbHJQKykp8QA1ecMVH0xNTI1iBEEgJpOpm1arXeivmeaXfwq4ZWdnw263Iz09XXIg8Ty/JTc397127dqdBuCQsTenu7kM7+6xIIiNY+TgJpYcF72XYg23lJQUqWDlzQJuVqsVhYWFHkHQNwWw6fV6ZGdn+wQ1kan5Uj9lWQSEUsrYbLZharV6EfyVbf3iF1BKD5eXl7+TmJi41wvcODe4SczNXX68Abg1ppaK4JaWlnbNZocbIX96ErzBYPDJ1HQ6nc8KHV7qpwRqDodjslKpXASglX9J+8Uvrkq9CQkJkbW1tW9FR0dvppQyxBU4RiilHNxOBcGdUuAmORQuxwP1AZS+fgNpaWm/KS3qbwtsJpPJp01NBLXLqZ9yUGNZ9mGFQrEYQIx/OfvFLx7AkxIVFbVYr9cHh4eHf+Mud0S8NTYR3JxOp09wawzUxK1Dhw43VaMY5s/6YYvF0qijwEcTY9IYqHEc95RCoVjuB7XG7Q/33XffTTWmwsJC/NUadb/88svo3bs3hg4dKqUaTZ48GZMnT8b06dOv++/t378f17Hnb0xYWNgSs9l8LwCtLJhXnqkgZSk4nU7irlpJR7QAACAASURBVNMmJdDbbDaPQN66ujopeT4/Px85OTmwWCw4e/aslNHhLfPmzZMyZRqTzZs3SxkOfzlgs9vt1+L9JPIOUqJN7Z133lFxHPcswzAvwpVS4hcfwvM8jhw5clON6cUXX4TFYpHeG41GzJ49+6adw5MnT2LLli346aefsG3bNnTq1AkqlQqvvPIKnnvuues+vzU1NVi+fPk1hez4kosXL+K5554T3wYEBQUtrKysfEij0QQ1Am4KQRCIvPyRu9KuBG4WiwVGo7FRcHviiSc8Uq3kcurUKalgQmNSWVkpFSf4S6miHMdd1vvpZVMTcz4hL8fy66+/Bqenpy9iGObJvyMYHThwAM2bN8f+/fsRFxeHvn37Sjl1ZrMZ27dvBwD06dMH0dHR0n7l5eXYt28fIiMjMXDgQJ9VFk6dOgWNRiMVEaytrcXOnTsRGBiIfv36eagTR48eRV5eHvr164fc3Fzceuutklqyb98+lJeXo3fv3khMvNQOoqysDPv27UNoaCgGDhzokeYFuLIiampq0L59ewBATk4Ozp49i927d2Pbtm0AgL59+0rxZ8XFxThw4ACSk5PRq1cvCQgvXLiAmJgY7N+/H6mpqejSpcsVx0Apxd69e1FRUYGePXsiOTlZ2icnJwdRUVEoLS1Fbm4u+vXrh8TEROzcuROHDh1CREQE9u3bB7VaLc1D06ZNYTKZGmWlhw4dQlJSEnr16gVCCI4dO4bU1FRotVqcPn0anTp1woULFxAYGOiRY7lu3TqMHTv2iuvh+PHjiI6ORlJSEgBIebODBg3CsWPHcPToUezatUsKcB06dCi0Wu3c1q1bq4cNG/bRyy+/rHerlKLdzelWS3l3qXxRLYXD4SCCIFCj0Qi1Wo3ExEQPVZQQgpKSEgQEBHjY28Tr56ts09GjR5Gbm4tu3bp5xMAJgoAdO3agvr4eQ4YMkY73yy+/oE2bNggLCwPLsjhw4IB0Lex2O7Zt2walUol27drdWMZGKfUANXkyu3fTFVlvAg9QO3/+fER6evrSvyuoAa4yOk8++ST0ej2WLVsmZQPodDrcdtttKC0tRWFhIUaPHo3a2loAwLFjxzBp0iTo9Xps3rwZDzzwQIPjbtmyBfPnz5fA8OLFi5g4caKU/D1mzBgpLWX16tV4+umnYTQasWjRIjz99NMeKsXXX38No9GIKVOmIDs7GwBw/Phx3H333TAajdi5cycmT57cwDazbt06jwTmuro6lJaWQhAEFBUVoaioSLTzYOfOnXjggQdgsVjw7rvvSoGspaWlmDVrFt544w3odDrMmjULu3btuuIY5s2bhy+++AImkwlTp071KFW0atUqTJ8+He+++y5ycnJQX+8q6lBSUoKamhrYbDYUFRWhtLT0itdv586duO+++2CxWPDRRx9JkfIbN27EkSNHcPbsWTzxxBMAgPfff99DPaOU4vvvv8ftt99+xfVw/vx5PP/889L/bd26FZ9//jkYhkF1dTUqKyvBsqw0r6KwLDvL4XA8PHHixGgAYvFKFQClIAgKAApKKWFZVlJLTSYTqqqqYLFYUF1djZycHA/mVlVVha1bt6Jdu3bIyckBpRRnzpzB+PHjUV1djc8//9xj7l599VW8/vrrsFqteOSRR7Bjxw7puw0bNuDo0aM4ffo0xowZI7H75cuXSyYMo9GIJ598UgLCqVOnYvv27cjPz3dlfzRWtuiP2E6fPo1vv/0Wr7zyCubNm4epU6di5MiR6NOnDzp16oTU1FQkJSUhNjYW4eHhJCgoiGg0GkapVCoUCoWysLAwThCEN+nfXIYPH06zs7MppZSWlpbSDh06UKfTSWtra+mxY8copZSyLEtnzJhBt2zZQiml9P7776ebNm2i7tLQdMGCBVSn01Gj0UjT0tLo1q1b6ZAhQ2hVVZX0O6WlpfTMmTPS8bKysujp06cppZT2799fGkN5eTnt3r07pZTS48eP0xEjRlBXNWpKv/nmG/rUU09RSin99NNP6ezZs6nT6ZS+s9ls0u8JgkCzsrKoTqfzON+amhratWvXBvMwePBgaQwWi4V27tyZOhwOmpOTQwcNGiSN4YMPPqDPPffcZcdw5swZeuutt1KxR+euXbvo+PHjpd+aP38+ff75531ej02bNtFZs2b5/M5oNNKOHTt6fDZq1Ch6+PBhSimlHMfRnj170vz8fPrdd9/Rd955h3744Yd04MCBtLKyko4dO9Zjjo4cOdLgtxpbD3a7nWZkZNDi4mJKKaV33nkn3bFjx2VLJxkMBpqWlkY5jqN2u/2D2bNnZwBoASABQBSAUEJIIMMwGgBKhmEUGo2GIYSQwMBAEhsbi4SEBDAMg06dOqFPnz4YOXIkJk+ejCZNmmDFihX49ttvpRp5H3/8sXT/9+vXD/n5+aisrERGRgbsdjsopTh48CCmTJkilS1atGiRtM/06dOxfv16UEoxduxYnDhxApRS1NTUoGvXrqCU4pdffsGIESMgssy33nrrxqmi586dw4ULFyT1U97zU7SpNRJ8KzG15OTkRYSQ2fgHiJiv2LRpUzAMA6PRiJCQEKxevRorVqxAaGgocnJypCd7YWGhpF4SQvDvf/8bgMvzbLfbMX/+fPTs2VPK7wRcpWOef/55WCwWBAcHo6KiAjzPS6xIPJ7c1pObm4vy8nKMHDlSsuG1beuqdjJ58mQUFRVh2LBhSE1NxbRp0xAQcMn8eeTIEaSkpFxV3BPP88jPz8dTTz3lMV4x+V6r1UrqeXBwsFTJpLExFBQUoG3btpJ6npaWJtXjF+V6pQRdvHhRUr2USiXatGmDwsJCpKenY+fOnXA4HJg6dSoOHjwIpVLpMUeNJXn7Wg8RERGYOnUqVq1ahcmTJ6Ompgb9+/e/qrXl3h4YN24c9u7d+8GJEyfqvbUrd84oL+aX8jxPWZYV+//BZrNJamh5eTliY2NRXV0NlUoFhUKB06dPS+Wk5OsoPz8fFotFUrcFQfBYE3LzRWpqqke1F19SVlbm0SNVpVLdGGDLz8/3CWo+KnQ0COkQbWotWrT4x4CaXIxGI1iWRXBwMLZs2QKj0Ygvv3RVJX700Uel/4uOjkZFRYWUEL5nzx50797d5SFiGGzevBnz5s3De++9J+Urrl69GomJiViyZAkAeKg/UVFRqKys9LCfiZ+3bdvWZ4XZvLw8PPzww3j22Wdx7NgxPPjgg9i2bRuioqIue9MSQjyqswKAQqFAeHg4PvzwQ6kskiii+u1LGhtDTEyMxw1SVlbmAfLXU8TfEpvuiL/VvHlzVFZWIjAwEIMGDcLixYs9GqZYLBYcP35ceihdaT0AriT8ESNGgGVZTJs2zcOu6mtevUWtVj/Qpk0bYdSoUe+/+OKLaAzcRADiOE4KBREdAYQQXLx4Ed26dUNtbS0UCoXkQDh+/Dh69+4tHU9cQ9HR0di0aZNPO7B8zOXl5cjMzBTHKj3A5CaOqKioBur8Hw5sJSUlUtMVb6bmbVPzFdLxzjvvqNyOgjn/JEB77733MGXKFKxatQojRoyASqVCbGysVA+ruLgYu3btkkrsTJs2TfKk5ebm4quvvsKGDRvgdDqhVquRkJCAt956C7fffjvatWuH/v37IyYmBlu3bsWOHTtw4sQJnDlzRrKxTZo0CQsXLsSDDz7okYPXt29fvPLKK3j11VfRv39/7NixA0lJSZg8eTKOHDmCbdu2Yc6cOaipqYFarZaevmLliOXLlzc41/DwcCgUCnz11Vdo0qQJ2rVrh5iYGDzwwAOYM2cO5s2bh7q6OmzZsgX//e9/LztvjY2ha9euIITg7bffRkZGBpYtW9agSsW1eps3bNgAu90OjuPw3XffITQ0FIMGDcL999+PJUuW4Mknn8TevXsRFhaGtLQ0EEIQGBiItm3bIjk5GUVFRR79ADZv3ozhw4f7vNl9rQfA1U1q8ODB2LBhA/bt8yyAmpSUhOLiYvzwww8ICAiQAMJbFArFQwsXLnTu3Lnz/f379zewi7rf84IgSHFulFLicDgoIQTypuN6vV4CttTUVLz77ruIjIyEzWZDcXExAKB169Zo3rw5Fi1ahPHjx+PYsWOwWCyYO3euNA9du3aFyWTC/v37sXjxYgBA165d8fnnn8PhcOCHH36QxpeZmYn/+7//w3vvvYdWrVphzZo1UCxduvQPuzmrqqpwzt2jQN7M2GAweBSI9E6TkseprV+/fgnDMAv+SaD2xRdfYNSoUTh48CBatGiBuXPnQqVSITExEYmJidi3bx/i4+MxYcIEREVFITExEa1atUJSUpKrFpVSiRdeeEHyLCoUCnTv3h1BQUHo0aMHzpw5g/T0dLRr1w4qlQqHDh1CWloahg4dioSEBMTExCAjIwNWqxVHjhxBt27dcOjQIUyfPh0KhQK33XYbcnNzcfToUbRp0wYTJkwAwzBIT09HUFAQdu7cCYvFgsWLF0u9KtetW4cmTZpIHjq5MAyDXr16Yd++faioqEBaWhpCQ0OlirKi+jZr1iyJqQQGBnrU6YqLi0OLFi0aHQMhBCNHjkR2djaOHz+OiRMnNmgy0rJlS8TGxvq8JlFRUQ08d9u3b4fFYkG3bt1gMBjgdDqRnp6O9PR0hIWFYfv27YiKisKzzz4rlfSJiIhAt27dEB0djejoaGRmZkqe6Oeeew6PP/54gyj+xtaDnDw0b94cAwYM8NhPq9WiY8eO2LNnD3Q6HTp16oSAgAAoFAp069ZN+r+wsDB07Ngx484777S/9NJLp2WMT0Q4KqqccianVCqJ6NQKCQnxqKRCKUVkZCSaNWuGEydOICkpCVlZWUhLS4NWq0VWVhaqq6tx8OBBxMbG4t5775XU7d69eyMvLw9lZWV49tlnJe91t27dUFpaipycHEyaNAmxsbHo2rWrVOrq8OHDqKqqQnx8/B+XK6rT6ZCdne3B1ryzCrzDOuRxamLwrTtOzS9+8csNELPZ/O+QkJD/AbC5K4M4ALCQVQYRc0uVSiV1M2Iq5pWKSfMRERFSXmlycjJSUlKQlpZ2w5Lm/xBV1GazSR3aKyoqUFtbK7XHu5z6yfM85GlSDMMs9i81v/jlxklwcPBcnU5njYiIWOdmZhTuWDav9CvqInXElaNFCGUYBgzDgBAC8W+lUgmVSgV3cj06duz4u2vk/SnARinF+fPnUVhY6KF+yqt0iB5QeZFIt6uWERPa3bmf/owCv/jlxkpAeHj4Y1VVVZa4uLit1JVc6m1vg7uHguB0OkXPqAhyIIRIdjZxEwFOo9EgPT3dZxOXmxrY5KAmNjIWHQUWi+WylW8FQSA2m22Yu0qHP/fTL375cyQmJiZmZkFBgbFFixYHvJgbBUAFQRC9o5ABGhGLV4qbHNREYAsICJCaKv8lgK2wsBAXL15EWVmZT0eB3W5vkFUgC+tgTCZTN3c9NX/pIb/45U8UQkhKcnLyQ8ePH9d37tw5x83cRJWUin0RRNYGmYNBVEW9GZtSqYRarYZarZb6mv5Rct28olVVVZJdrby83KddrTEPqFjOOzQ09CVCyED/svKLX24KcGsaExMT2qlTp9Nff/21zQ1ekqcUPmq2uVVS8W/555B/Loa+yHtcXFfGdj1ah5nNZpw5c0Zia2IOqHebPIfD4eEBFdXPu+66SwHgmfr6en85b7/45eaS4ZmZmTUpKSmvXbhwQZAxN1f6gfsP9yvcWQnEnUIH+eZ0OqXNXTUE7dq1+0PATfl7az5RSpGXl4fi4mJUVFQ0sKs1VtLb7SwgAJhly5YtcDqdN6TxithT0bvqxJXEYrEgMFALQhj/UvfLP4253b1169bqlJSUz904JsjBjRARxwTK8zzhOI66aRsVvaRyFdXbY9q+ffvr7kxQBgUF/q4D5OXloba2Bnq9DmazCVarBXa7pHLKKnQ4iRidLPeA1tbWTHU7C26I8LwTTiePaz1vnU6HyMiIa+o5aTAYERIS7DOS3C9++StJUFDgnPz8C5W33JKyFYA3c5P+TxAEged5OJ1OKBQM4TiOKpVKiaHZ7TZYrRaYzSbo9TrU1tagoqL8urbuAwDltdyo3lJWVoaysnJUV9egvl4Hg8EIs7lBjwI3uEmgRtxUldTW1vQNCAh4BjewLpzraSHgt5y3aAC9WjGZTAgLC8XvmWO/+OUmEWV8fPzDx48fq+zcucsxX8zN/UoEQQDP85TjnGAYzivkQwmVSg21WgONJgBabSCCgoIRGhraICf4dw22sWqXV5LS8jIcP3sBOndGgcFwKandbLbAbLWBvQRsIlsjvCAQKlCmqKgwISgoaD6AlBtLqxkpzuZaVW55cu8ftc8fIfpje6BoloGQqBsTFkg5O6w19VBGREOjVQNwomLrJthsgKppGyRltPlb3Om/ZV5thTmwOKMRnRLXyH/YUbXjMCJv7QeVm+g7ik6j7Fg+AILYQaMRHMpcx7FEIDol4Wp3uaV169b3r1y5snzatGnlAMRAXfGVip5SnufhDtqVQkAueUct7tAPtbsTlhbBwUEICwu7pqYwjtJcGIxaxLZLanBOymtWk6gViz7YgP1F5SB1ZSivqUfXuFAoLSZYLGaUlJWhsN4MwtkgCECIEsTpdMLJccTBOQngsqu9k2t++rn4+CHkT1iQlOKa1UMxP+5a9vst+/wRUvLZcgRN/gxhvQP/8N+q2/YZDr/yJbTNWsBRVoBm899Fm/5JUAUEwnJuN84dqEKzzHZ/C2D7LfNqOLAGhfpMxKaObIznI/+9zxDY91Zo3ESfUWqg1gbg4tsvQJ1+G0LDG66nyu8/RsDtaQiL8RxLwftLoez1GJI6RDbYx3ziR5QYOiM2NfFaNJ4B48aNK5w2bdpbMtYmyFgb4Ipzg1slpQzDEIWCoQ6HAkqlaxMzEdTqS23+goOD0bFjx6smApbjm3H2VBLi05o1OKdrBzao0T+tDXo2DUFhoRq79pZi7zkdukcoYNbX4WKdFYnhgeDtQEV1HTErlVAITnCuCGVCAMZSm//QyDf2zMzJTEXHG5xbIDdi/jY1lvnD9wEEUDBw6qvA0SAERnh6jSgEEDCwV5VD0IQjMPzSYrZXlYJXhiIoKvTSDqwdoATOumoImjAEhHqmtDiNdbAZHQiMi4dC9dtBmC/dj/2LV6LH2o1o0iwUEGyw1nNgGA3iB49AMFOO/AKLj/kQ4KipBscpEJQQA8+HHQ9beQUEVTACY8I9vqMOK8xVtdDExEGt1dxk8+r7nAjPgToFUKsBNgsQHBfhOT5nDPp++xmIbI60zdqhebN2qPxoCYCGa5c6nWj/3CoQZcPPzacOQNlqOogQDiiV0jio4ET87fMQL/vs0jA4WMsrwETGISBQ02Ct8EbHfVUVVYVxTeLWA+B9qaSCIAiEEOp0OsEwBA4bJYQw1KXBKKFUutOsVGoEaNRQOOzQKLUIDQ1129sE2CrLQQMiPa4DAHC6KnAkBETgQJ28NB/yc1JeqzeioqIeak6PkqpK1NXVgzhZ2MwmWNQq1OoMIAoCcC67moJQcCwLEEqoq6Y6sVhM/ZTOuvkco0SwkoCQGw1slwDutzC2a9nvt+wDAOefmYiLuSowSiWEyosIvn0h+s4d7frSeArr+z+DpGExqDylA8w6dPjkJzRLMuLQfVNRZ42EylIGkjYRt746GyoGoE4OhW/NR7FKAXvBGcTPXoHud7kKEhb+51Gc3FSCiOYRMF6oQ8Z3GxErPdwdOPnkTKgGzUK7rC5XXhs/rIeq+wQkNHerE4pABMnyRwgVQHnBaz6qsXPweLAxLaGCASZdKPqu+hCRMRpQUzH2TLkH9tCWUAs62BMGIutVV0k+/e7PsWfRJwhtmwJ72UUkP/kJ2g1sfpPMa+PnBIGH8eeV2H7gKzCWMqD1nRj4xiwoAVSu+y9OfbkD+mN1GHxmLyK9sZrjAHiuJ/bsz9j9/EewnP4Vrd/ei7a3NnGpZMc2Yd/LX8CSUwJS8DjKP9Ei+rZH0GVyPwBlOHD3k9AVnkfI7YvR/+lLpdrtuXuwe9YiME1agS3MQ9xDL6HbXX19rZWH9xzILuzXK+2YSxWlAkAEUS11u0kJz7Jw2u1USQjsOh346Hg3a1OC6MtQ9QuLWpUdQepgXFT+iIB3XkCgvQx5zzwHNrQZUH8RQUPnovcT40Ag4OJrs3FyYyHCYjXgWAdIm2T3fHie0zUBm9VqlcI6amvroK+vxvHCSsQFh7rac9nsILwAo9EGo8NJFFQAFSihDAhDwIxbsiL8kTU7n6qv0ydMGz8YKeo/QxH9bcD2e/a7ZhDlnXBG9sKoj54CMeXhh8zxqJgyAgnxShevqDkFReZmjF7eEQALJ6tE1eo3UaXqh9HfL4bCacCuIZnI2zMO7QckArwTyrQsDH5uCpzle7F+wBK0GrcLEYHVOPfhXmTsOoWEOABOFjwjf9gIMJw6Ak173VWdg+XCBQQn92z8f6kAwekNbBHos3471CEu6p73whSc+WoX+j42HOYDP6A+bDDGfeEqhMmzrLRv4ecfoNnTn6HTGNfTnWeFK47xxs1r4+cEgYcVyRj77ZtQCwbsHNQD+YfuQJuesWgyfg6ajB+LH9pMdAOY1/idHEA9gU3TbjCGfjUYR+7qBtBLay2w62gM/Wo0jt/fG6o730La0GTZkRLR58uvUPL+Y7hQT2XHs+LIY3OQsOBrdBzcGoLpPDb3m4H6MXsRFdJgrTRhqWLasmX/Lliw4BmBUgiEQIDMY0opBQUEyiig0AZQFZzEXFtDtaGBUCjsICwLc1014oePRGyLJoiLjkZtRQV2/udVJI18GcNnDQGcOuwYOACl40aiqfoQjn5yAUMPb0V4kIDj9w9GlfSg9Dyna3LXFRQUoKzMnQNaX4uDJ87AoQpCssIJs9kOjmUhcAKc1EnAS4ntEAQQQaDEkdzpybn92w2tLzmPV7cfwajUEYhTwS8+JLZPL1ennZBWiGkD1J8qR0K8e3FqUnHL6I6SaUCpBnQnTyKu75NQAIAyDE16t0f16dPAAJf9JK6nq4qpMqE7oqIuoj7fjogOkYjvGYkjD0xD8pB+iB84DPHt5faWQPT/6dQffKYMqje+g9zvdsNmtoOtLkKIu5y0Nq0rmFMPYNccB+J6dEXyyJEIUrvnp2cvHPjXLDjPjkBUj35IHtDlJppXZaPnBABRmZlQMwCYMMR3b4X67Gyg502QcGPKRXW2DYq93+DYQZdqTp2l0BWYEdXR51oZOHv27LMLFjzzsVsl5b3sbdSN0JTneTAqBgLlic3OUqVCAcJycAZGQCAsjEYjNBoNAmsKUH+sCHz0ehyr/wUUAOt0oD6nBBGqM2DSMhAexABgEN+3J6rONLaqrtZQWlKC0tJSVFdXoa6uFkdOnEKJQ4HWoUrY7a5qHVQQIHAcGMJAzQC8QAlACaWUmfb8f8ZNHdrn6Z7N4zGybx+05wqxrdDiR7BG6cVlvtMG4fdFkBD3lVei80fb0HvxdKi5MuwfPxR5x2p+81GDWjaHsaD4mvax7F+Jgx+fRJd3/oeRW7ag6/1D3doMoEzsgdH7fkLKkE4w7VqFH25/DKx7v6YPvIwRX65AeHwAzi+5B4fe33HTzOvlzsnnXoziJll0AqCKRPzILCRmZSEpayR6fLoWTW8JbHStBAYGziopKe4LVyUeDaVUDVf4loJSqqAAQyl19yrlwQPgOBYs64CD48BBgNVqhclkhsFgQF2dDjqOAduqLUiXLkjKykKXN/+HlF7XFgpyVcBmMpk8eoAeP3UKZ4wUHaI0YO022O0OsKwDCgCUtROHK2aNUEEgBGCqqkqTQgPVTxTVGlzTZzegxEIQHOCP72pMqvcecBkrTHmoOQdEdry8Sz4ivQOq9u4FDwBOAyr25yBSVmG26uBel/G3/FfU6VohqkUAABasRUBURj+0m/csUvrFwpgv7+LNIuf/5uD89qtjbQmjx4HdtxplBQb3fWKFpdpwye0UGQm2rAROuXmjrAwBrdIRHhMECBaU/bT3kqHaZALC4pA4agK6v7YUypJzsLp3Zk0mBN3SCSn3PoxOD94G/bm8m2ZeL3dOAFC3bx9YwXW8yl/zENkh7ercdpHhsFyhsYm3KDUasHX1V/fPIW0Q29oGqykcsd26IrZbV4S3aAZVEHPZtdK0adN7v/lmTRMR3ABI4AaAoQLPuOxtTsITQHByhGU5ODgnWN4Jm80VtOsCNw60aSDKCqtgiwiHtnUqolq3hDokEEGdOoBmH4beIgBwonLvwcbP+2rO15XY7i4YWVOKA3mVsLF27Cm3QWDtgAAkBCnhpCAqxtW9BtRVh44BmKiouHmPjuzd7YFvN+LokRjoa2uR0LEnRiRp/AjWiChqD+LHcXeAr76I0AdecNuBGpf4Ox9H3A934ocRJ6CylYGkPYRW/S+pldzJLdg6cSvsBWfQfMlbcDmaqrFr2FjwCalQox5G8y0YOFQeisGhctM3UMcOR+vBHa98EyX3R79/T8H+2/sjoEULOEqLkfLSanQY6nImaLqMRKuWX2F9t97QdBmB0R8sQtSgSVC/dRe2TDwKxlaPgOhLDgDzwf/h58VfIqJNC9gKziB65gKEuafh7II7UHRRi5AYNXS51ej6weM3zbwKlzknANAqyvHT7ePAWMrAdHoMt2REAdDh4D33QFfvgNFQgQPjRkOpbYE+a9+E2MK69dz52PnYOOQvDUb7l79Eq54JyFvxKPL2FMGaWw1Sci8KX9Mibfn/kNzetVfyPQ9j1xOPoOyTEMTd+TQypg9G7dZ3cPjtLWArL8LB/YrN+1eiyX0L0Xl8T3R783XsfnAqSt5LgZKrhdmSgCE/fYzgy6wVQkjnUaNG3QHgvzKVlHfTVIFSSgWnkwoAVSrVcDqd4DiOEJ6nnEBht9vdwbuuyh9RPQegavtKbDz+E3IiAxBlDUPfDd8jJr4vuj7QEj8PG47QWC14qnE1DAQanBOhGmnQgQAAIABJREFUVLiiCiomuJeWiuWI6qDXG3x1mXInuHOE5wVCKVXY7baJKpXqfy7rL4dygwUabRCitH+Oca26uhosyyEx8dqobXZ2DlJTW0m16/+ofQDg7FOjYem6FB2HJYITghEUFXTV+7rCEkIQFNUw0NFRUwkhIBTakEAP9cNWWQknVAiOj8H1cOdQ1gpzeTVUMfEICLqKeB6Bham4EurYBGgClV5fWWGprIUqIhYBIZ7HYg01cBgcCExIhEJ5k83rZc4JAHhLHWxmxivc42YRAdbyUlBNBIKiQq56rZSVlc1NTEzaBcACV2lxO6WUJYRw7rLivHdJca1Wi8DAQAQFBSEkJATh4WGIjIxCbGws4kID0axZK3TunYnkpEtBuGxdFZxMMAIjgn4bY7NYLCgtLUVVlby2mgkWi9WdD3opbcrplHJBiSC4gnBLS0viVSrV3EuPSxUSIsPhl6sTdUQc1Ne4T0Bc48GWmph4n9YIbXzCdR03UQci5FpqbTFqhDRPbuSrQIQk+/5OHRYDddhNOq+XOScAUARFITjoZl15DAITkq95rSQkJEz79NNPcu67734eAA9QibXhUoFKMZeUKhQKwrIsdWUkKKBSKWGxqKTOYoGBWhhYO8pKSxEVGYmgINeEqaOufP0ua2O71ITF1TJPBDVXaW8HHA4WHOeRMiV6QgmllMTFxT0OoKsfoq5NEmf8H1JuTfFPhH9e/1JCCOl0xx13THDb2TSUQu5IYERckGUlgOM4wnEsHA4WdrsDNpsNFovVnSSvR11dLSorK6XWfVdtW2zsi8rKSq/y3iaYzRbYbFYPpsaynFRjSRB4KcHdarUMZRhmrv9yX7uEtO7mnwT/vP4lJTAw8MHc3PNHU1NbHwbgdDG3S7Y2SInyUroVWJaDQsFKSfI2mwpms9qdJK9DUFAQQkNDERERgfj4+KvknD6E4zgvFdRVW81ms8Jms7u9oKxU2luuglJKyb33TlNoNJrH/ZfZL37550nLli2npKSkyD2kKjeJYuAO/xAEMQTECafTSTiOA8uKrM0Om80Ks9kMg8GI+vp6VFVVo7S0FFdbP9InsLlCO6rcKqhBqrNms8mZGuvRAdpdJZNQSpkPPvhgNoBB/kt87VJdXY3S0jL/RPjlLysKhaL/4cO/jBaBjVKqEsGNUqqglDLu6tlylVTCFYfDBW6X6rYZ3CppFUpKSn4bsBkMBnfPghrodHqYTK62eWLLPHmNNaeTI/IuUwBIfX1dqkKheMx/ef3il3+uRERETNuyZXOSF7hJsW0AiCAIMtbGSazNVZDS7ra3WWAyGaHT6VFbW4Py8nIYDIZrBzZ5hylXeW8LrFZPUDOazKg1mKA3WWFjObnDgBE466xXNuxMXLBuL/aWmS4d2GnHN7sP4cmvtuHf20+gykH9V/96i9OC2tPnXcGkv1MqVr+Ag+9vu+5D5Ct/xY7Zyz3GaDi2HceWzMe+2XNx5ru9EH7XLwjQn9iP8x++gxNvroTD67uqrStxcPZjOPLKh7BaZb/krMO++2fC5E+GuV7S9NZbb73dDWweQbsia/N0JHgyNxHcrFYbzGYLjEaXSlpdXY2yqwhS9gC2qqoqVFZWuhLc3XFqLhXUJrOr2WGw2MEQVyEPwckRpytmjVgNFQNnf7T5YS4qESNb/j975x0fRZ3//+fMluxm03s2gSQkhBYgGCDSIQKCoKACIvYGnOW8s5evDU85z3Lqz3YqlkNREbFgpBcFRKQHCM2QhPReN9tnfn/sJtkUUhCU03k9zAOzm5n5zOfzmfe86+tt4KH3vuFAvWsLf5T+HUtzbMwdMYjg2jxuXvHzr9zACtrAdJQt1z7aVHb0q2RkXRmWytqzPsTjLz1L8MUz8SwiyvliJX7DJxE3YxTZj99M5obcX3GFKjJfeIXqzD1kvvpxi7mo3vw2P/zja6LnXoe2aCOb7nu1ucJKHUzMEC0H3t+g7KOzBJ1Od0Nm5uHBHlrb6aKkLbS2lv62xqoEl0laXl5BcXExJSUlXRdshYWuZizV1VUtTFCr1YLN1ijYJPy9tYJWFBAFBNFVxi8AYuHJ7NuP+MVy/6jejEm+gLkJAiv2FwBWNhwu4dZJIxkaE8H8S0dgP3aUg2Zl8c8Wir5ewv6XluKoO8nhxYvZv/h5qiqkRilF1pvPs/2OO9n78hIaTF17pch1hRxa/Ajb732Egox8T6lH9jsvsf2OO9n90pKWmk+HgvcIx9aaiJvcp8XHg595g4RLJxI1cRaRKSGYCot/xUwEM/Kj5aTefw2tKzCzP1xK7MKH6TFqGIMWPYL9u/9S5mHVRM2+nOL/vodFeeOeNfTu3ftKWgYRNB7maKPW1lRL6hlIsNmsWK0tTdLq6irKysooLCzqmmDLy3MVuLeOgrpMUFdah0eT4+aAgSuCK1os5rnHy6unJ0SE4Gyo51SdlcSIEI6XVgFOHJKMSuW+nKhCJddzstKurPxZgk//YYSNGYKoCSI4LY2wtLHo9a75PvH0Ao7uqSH+tlvR5G9kw90vdklbLlrxGeoh04idEMdP18yjvMIBOMi4ew6/ZDpJWDAfH/MeNv713106X93PP2BPHI7fabIri5ctIju3F/0uH3YOZshC9dE8/PsnUn/yGDavRAIiyqnJanaXiBEpBIl7Kc6yKRvqLEGtVl+cn583mqbctmZfW2NDJzedeItAgku42d35bW2jpKWlJeTl5Xcs2Gw2G8XFrlrQ6upqtwna4BEFPU16h9MpyLIs3vHQYyqNRvMXs82Gt1bFp2vWcs2Kn9Fr1NTaHYA3I+KDWPFjBmX1Dazduo9cWcBsdSgrf5bg23sQxmFJiLoAIlJTMaamovMGKCN75S6SHn2ciORkBi56FOe65VTUd35O/6lX03fKKKKm30b8MCc5GzKQS3dybKOToYvuIigxkfi770XcsYKqhs7PV3vyJIbomHa/a9i/gm2v7SNt+RL8fM8FlboJmxlUmmK2TBxLZnouaj046jzNBh98YnTUZRUrG+osIjIycnZKSorOrbW1CSQ0mqXtpX+4tLbGKGmD2yStpry8nOLiImy29l9CapcJWtiiH6ir/rOxZKpZW2tMxHU6na42dpIkCIIgvPL0Y/OBVL1WS4PJydXTL+ESp5q9Ozfjp3HVhN42YwrWtTu5fWkWA/v0ZXzQCbwVdo9zD0c1DdX+eIe51STvUPS+1VgqgU761Bo8kiH1xkjKK8txlNZitRSy69Ybm9R9r6QksAKdUP/LOOA0NOkVO7YRcPFsAkPPFVe8Aa0enPYIJv60D01IED8sBrWvvu2fSk5l35xFiKI4bM2a1ZNDQ8O+BezuHwduWnFcybuCuy9pk5zx1Nq0WlcgQadzBRKqqhpN0kJi2ynfU5vN5iY6IlctqGeT45b5ag6HQ3ALNcHucAm1jAP7/VQq1XyAPqGB/HK0HNFrMKHA8eJyEnu4OhJpvQO45/KL3S6aQiZs8+X+UIVl8uzuIAFBcrakHFMH4B1QQ0OpDeK0YCrDXBeALqgLOo6Hr8tcWIR+sBF1mAEvQzyjPv4UQzffS36RcZg3tJ+HFDTxWpJsoe0bkSX54B2IzvfXFFfqCOjfg5rM4+guGAamDKpLwxgQ79vCXG0oMBESF9nuGJxafwyBvso+OxPPZ3Dw7Eceefj7Z59dbAPssiw73AwgTrfW1qJ1n9sylJujpDZ3lLTRJK2hosIVSAgPD0evb/mCEpvoiKoa2+c1eDQ8tjbWguJwNPrWHNhsDgABWRb21DhuXbb/VH+A2L596Vebw/PbT7Bl/14+zZKZlewqHi46lctHe06w6eAx/r50E4MuHEqcItfOslISg6/6BJnvfUHeug00mABCib1iGIee+QfFGfs5+MRiVJPnEOzT+elq1n7M0TXbKfj2HbL2aOg5sT9CWCp9xtr48f7nKd6fQdHmNex+4Z0upZj4jRiJkLkHUzseiMpNKzm+ene7x+25dgzbX17V5Wko+vYjjny+Aae9nKwPPiR7+yEAel13PTlvPEPu9zs58PizaKZcS6hHEb1ccYBy0yAi+rTVGg/cMYXvn12u7LEzhCAIfe+7775JtK1G8Ez9aJG063A0+vWbTVKLxVVLWldX585tK283kKCaO/cqj9y1Kmpra9ppemzFZnOFYp0ORxN7xyPvfRak9vZ53Spqg4ZHB4HKi7R+URw4lkVGpZM7Lx3HMHefQ6fNwpbMbPaX1DIwaTAPjInn9+ANNZlMOJ0Sfn5+3TqutLSM4ODgbvUIPZNjznR8rt3jjXHShZiOHqQhvwjDwAvRewsEj5mMquwweau3IMaOZPijf0HbhX4TweMmYzuwlpLMWgY+/U/CY/wAkfApM1CVHiQvfQO1hbUEjxxLUFxk55RH+kjsGUup8R5DWK+WLC+StQF1WCxBca1qAaViDv/zbYx3PEZYbNeYYap+2kRDrZaw4QMRGkyIfj0J7m1EF3cBIVF2TqWvRYgezfBHFqDVNI+66JPXaOg1ncQx7fc99RswjOD4SEVKnanOrNMZNRrN+k2bNltx1482UokLgtDYWd6zk5wgigKiqHL/iO4OVyp3lys1Wq0Wnc6LoKCgFvRgwjfffO3Ry6BRuNW6mTwak3PNWCyNXGsOwel0irIsizab9V6VSvXs/9Lknu98bGc6vv8V2LI2sPHpH5n8weNderHZMr5g1SM/M+Pb5zinHllHBVuu+AuDPvyUoEBRkULnCNXV1f8KDAxaiYuzzSQIQgMuD61NEASHIAiSSqWSGjnbdDovWafT4+2tx2Aw4OPji5+fH0FBgYSGhhEVZSQuLo7+/fsTHx/fbIp6BgzcZJGNGhp2u8OtEjoFd3pHU+nUnj27fVUq1c3KUinoDrTxE5naRaEGoE2aweVfLeach5nUwYz/Zrki1M4x/P39r7z//vt8G81Rd4RURatSK3duGw6HK1Jqtzuw2eweUVJXbltjIKG0tJSGhubQvOgqcq9vZXq2zFlrj2tt4MCBNwIKuZWCcwtRjahWhM0fyNcW/+CDD05wC7Z2fW2tOdva5rZZWwQSXEXyrkBC07ZpWeTemK/WNhLqZu4QAOG6665VqVSqG5RlUqBAQXcRFBQ0o1evXtp2tDaxUWvzqEhoyp1tDiR4klI2Fsm70j8sFotLsLm0NVeReztCzZOSqElbe/fdd68DBipLdPbhItVTUgoU/KG1tgHbtm0d0xWtrX3NrVG4NRbJt0z/ABBbpnfYWvjWnE6nO2+tmRkXENVq9XXK8pxLweanTISCPzTCw8OnuQVao9amphUZZWM1QqMMaulrs502/cPhcKBurAdt1NYa89WaKwyafGu4Kb+nAqOra2upsYtEBfrQrgvEKYOqdQKAk6KqOgzevvh5qZTVPQ2K13xKVbEVMaQXfaaP6dIxtvwjlBSK9Bjep8vXsRcfIWvNz8hA+MQrCIo+V5qihORoW00qqNVnpStWiytV5ZKzt4xeF3WDBrwun2NfbEQCAlMvJqJf1+inK7avRtV7HAFh3l2+VMXWdEqzyhF04cTPncLZSuWUGioo3LINq0nEeOml6HW/3X41ndhNrTWUyKSYLh8jiuKI48ePJbspxBuL4+1urc3pNkebNDan04nDZsFUX49Tq4XQSLy8vFrVkTZqbSWIrnrQ5goDD22tTcAAELRq29V3/fsDpr31LX//6EtGv7SCHaWN9XZWnvvoC0Y+9R8SFi3Hs0TVYapg/isfseCzzVz64lI+PFKmSLDTQOVlQFNzhD2vfN7lYxr2fcu+D9Z2zyQQtWgMBoo/eZHCzKpzdj+n3nmYlSlDWZmSzCfGaD4fMpSVKUPZ+MRHv+q8+R8t5siGIy1fnQW7+PmF/3bvRKIajcFA5doPyf7xeJcPy37jMYoOVXfvUl56tGI5+596nbNXau9g902X8MumQ5hLy5HOITvJwYduoriw5QUqNy0j88td3T5XbGzsRE9ztFWEVGgdSHA6nThlGWt1OfWVdU0VCWazpUlrc9WRliE2UhJ5CjV3hYFn0ABZloXq6qqBgko7e8GsmWx/YB4r776OB/p7sWjDwcZHkskjL2TVgoto/e5f9f2P1PUczDcLL2flNUN5+avtVCj0MO0idMKl9JqcenrtrKKE2lOFONuZP2dDHXX5pbSh8ZRsmE7lYDFZmj5Sh8UTP3sWgVG+p31gGvJPUV9Sxa+hBe1523PMOrCfWQdW4ecdyKi1+5l1YD8Tn7q2eXiSS7Mz5edgqbN0rqE4HJiOHaDmVDmSw0Gb9riSjbqcfOy2LmwyQwS9Zs8iNPE0reUcNkz5pzBVtM/cai0txFzT0M5a1FCbk99inQKHpxE/4yJOF+iVLCZqc3KwNli6OLsSkiOfsn0iSYsepP9fbsDgoUA6TRXUncpvd6/IAA4LdTk5XZonyeGgeucWTDU215y3u/eK2zK9uK9hbWhZcqLRaC7buHFDDw+NrckcbfK1uVsOuIQagujtDRoVDmdjLakHtVFdDaU52RRk56H2TO9oLpty4nRKSFKzxgaIBoPhKhBJ8ujg3jPIH0tJI/2QmiG9oqAmp80U7sgqZMJFYwEIjulFP8dWdpY5uSRcMUm7Y9LtvGoMZeZIvH0lqrMbSH3/E6L6uJru2g6vZs2szWiowqwfxuRPnkevBfOxLWye/yAqYyLW7ONE3P4vhl8/rsMryXW5bJk9D4tvL7RyJZaoSUx7pbk/j+3YJrYuWk7yG28R7P8r7+rIl3yy4HMSetVRViRhr3Iwev1aLJ/ew/HSkVz02CzAwpaR/Yj9OJsYzQ7W/e1FLCcOYjc8RfV3ARhGzmHMPXPc5mgWW66bi62unupSPyavX0HAGVrZ9hPrWHXN0/j27oVUfhIpbjoXvfEgWrdgyn71XnLVKizZmUTe/RKp148HJE698wi7l/yEf3wQtfl2Rn64jPDYjgdRvfl9Nj/4Lv79E7AUZBHz4FIGTIzr8Ji89/7B4W92UltTxE/z5qJWBzDs/XcJ9oWC9x7hpze2ERAXSHWujZGffUpknGuxCpfcz4GdEmL2Lpwab6z+ycz85J+ndQ2Ur/sPu99eQ122mYp7ruaEQUXP+U/Rf/IAAMz7vmb9tcuRKn5B6nMNU9++FxVQvWM5P9z9PPo+fTEd+4XEf7zbdAzA8OGpY4FTbq3NBrINBBWy7JRlWZRAlpxOWRBE1GqX5iZLMoLkbJH+0VBygr0rviQy2kiJ1IC6tW+t0VHn0taaTdBvv/1WJ4riVS1X3cR7O7OYM+myTraHlbI6ByP8RF74bA3Jo0cR7gNltWYI91HkVdcNGYa8uwatr+sBKfrgEfa/s5KoF25xzXK1lqlrv8Sgs7H7qlFkfr2flNmJ7P7r34h+7DMGTeyDVHeM78beSsXlWwnu4Dmr355OVfAUrvj4MdfbuBU9jLOulIr9h7CdJXtKPrGHwBc2kzrcCA4LTqDwdCZ09AimrFjB8SfmUhX3d1JvbKndSqUmBq79mmB/kd1XDSf3+ywCpsefmVsgagSX/rgVjRrAwrYpQ8ndPZ/ew10vE+3gS5n41DwchVv5asJjJM7agiF7NTv/c4yLN2/Cz1ek/Itn+On595n+esetQLKXvkvcox+QPKM3IOHsghbV4+bH6XHzKVYlXM3IFSsIasxkrsvg53+uZ/S2rYSH6ch//W52Pfc+l73V/HKq2nmUSzatIyBYh9Ni6dDfGTJ5AVMmL2DrhDiiXv2cXr1bpkybHUYu/+Y51JYc0gfNpKziXiJ8C9l5+z9IWrqRXkmh2PO38PXMp4ibuBw3VSA+PoZp8+bNW7Fs2TKryxxFIyDbZRAFwfWf4CKzxel0yk6nU5AlWRacTnctqbsBjCGA6NmziIwMp2fPCNSNUVCXGer0SPFo6V+bOHHibKC5z7xk5/UvVuNISGVB/+CuenUw6DRoREGRUWeImu0ryfw4HVNFHY7qEqS+ze8aQ8poDDoALSGjh3H8wCGYAqWHzKi2fs7eHSAjITvyqcquJ3jQ6V8q+qQUxIzb2HK3lfDUFHpOm4bBozJMP3Qucw7PPXs3FnMhvYa7TUG17lfVEavikwj2dz14huhI6qqrz/hcgmjhxPOLyd95ALtVwpxTS0BlDeASbOEjRrmGbBxGcPBJKrMsmHdvA2/IevlZVwv0yhPUHPTC6XYenQ5hI0by4zN34DhyCcGpY+k54YIzHrflWAbmqGGEhbmiCBFjx1C7NL3FGAImziTAXcut0v26aENgyhBXdYguAkNIDZZacJbvorzGi6Av/0PVl4BkRqzMpKYK9M0iI/r55/+VumzZss2N5qjsMkcdIDtlGcnlb5NkSXK20tg8oqRaM8U/baGipp4CjRfqRhPU6Wyd4tGckAuIarVqlsf7mk9XrWarGMf7U/vTeV64F6G+aspqnfxlxkVAA/+ph1A/vSKpOrY8W7oqTq5jy6PLGffFfwmLDabq6xfZ8rWtIwXPdRJNEBHTpuLjXqgeU6fjG+/d4bXU0alcum09RVu3UvjNUtLfWcvMzW+hPUe3KuoNp9lHUtO/chd5SQVPB1Y3ihbae91mv/oQ+aUpjF22CJ23mt3XDUOW5NOfwX09r5jBRE2d6v58Kj1vDmgxlPauFXXb81yStp+irT9x7LHrKbzuVUYtSDtn20vrc/aeP0E8TdGbfzRRU6c2lcT1mDa7jVsgLCwsDdjWytfmmbAru1I/XJRGsowgOJ2yS2a53GhVmYdBNjJgShrRPYyIzcm4zvbSO5BlWaiqqkwSBOHiRn9Z+vr1fFYXwjtXpjSplJ1tmZGJRjYfywagLPckx9SRpIYq/rXTzlhwCOryXEwecsteVIQc2Zfg2GDAQd66lo1HTHu2UW8BsFG+bRehAweBb1/C+phpqAsgbGgKYUNTCIiLQWNoXjhtSAh1BS0bqDjq6sA/nOjpsxj20pOo847i6fu1ZW1l+10PU1l37ubAKyCABjf9s7P4AJVFLcmR1HoDlsrys/eghwRhOtWSbrqhoICAIcPQeatxVhyiaGdLPrmSHVtd81W4i4qq3gTH6QgZPhpr1iH0/YY0zbmfMahZmBmC0DqKMVW1lNS2ujoM8ckk3LiQ5PmXUX30xBnfi67PIHQFuygtdQUhin/Yil/y4F/NqKP21WMr71oEXRU3jBBysRpim+chLgpNq7ejWq1O27BhfY92BJuq0WJslEeSJCEL4LDZWnS2spgaEPz9cTit1Jdno24Uam2rDJqCBoKPj8/lzcZ0CYs3ZVGr8yLtWVeoXRvai+0LJwDw5NsfkF5so87mYMaiJSQNHsaHMwYxfcwovntnFZe9eZKaqlr+evk0gpUSwNMLtrARDJ4ZwuoLR6A2pnDxt6+hHzYVI2+zeuZctGIduqCQloIgSOKHWZejpgqL70QmzxgEwNBXX+b7+deS91YCans59SYjk9YvaSLQ7XXLPWy6+T6+fF9H3N9fJPnKodTv+IiN//cJgX3jMGdnEnL7Q/h7vJSd5SfJ/uwrYh5bTNA5Sn8LmXgFqn/OIX3mT6j9ItGFtNRzjJdfz9FbH+Kb1a/gN/56xj967a+6XvTcBZyYN5+VI98k4pqHGHnHNGKuuZX1CxZQt3EglhrQJ7RkXbEfWM2a2WuwZGcS+9j/I8AbGDCV1Ju2s37COPz7xWI+lY3fpQ8y9p5L3U9yTwbfdxFb00ag8othTPoKgn3gyENzyD2pxzdUS9XxUlLe/tuZ34zvIFIfupjt0yfjHx9MdbaNkZ8s+tVrknDTQrbfPZNf/A3E3P0SA6cndfCmMJL62kNsvWkKJ+L7IleVYglKZfpHi9oI2GHDhqUCuZ7CTZZlu+CKkEqyILjNUQmVlw7JVE9VtglbYDjaWC904UZqM38iq/wkJYIOYfDgwe7+BmYsFotgtVqx2+2Cw+EQ3CFXldPp2AP0Pxu2VUlVPXqDAT/t76Otne+0RZ3DgSk/H1VgBDpDW7+Is6GOhloLPhGhrcwdiYbCfGSvQAzBXZNEkq0BU3E5msAwdL6632W9ZJuJ+tI6vKMj+L30e2dDHQ2lNXj3jEbVzsvYWlaMpPND79vSvJcsJkylFXiFRaDVdW0P2GrKsNZY8TZGozoLlCZOUw0NFSa8o43tjv23canYMOUXI/qHoPdvP5lZluWjoqi6A6gH6nBRGpkBi4BgF0TBKQiCpFarZY1GI3t5eaHT6WS9Xo+3t7eL0shbj49WT1jPHi6NrdkEbdbYcCfImUz1aWdHqLkcHuGBSrnQrzQGMETHnl799/bF19u33bn3Nvbs3mppvfHt2fN3vVtBa8A32vC7jkHl7YtvB6kaXqHtVyqIOgO+Pbs3dq1/KFr/szh2gz++Bv/fd8uKWgyd7CNBEPoePJjRd+DAQfvbmKPITmQkQRAESXJpbY2BBM86UockI2nV2GwWxMbGLM25a83+NUDw8vK6TBEmChQoONeIj48f0coUbc34QXOZlYSn7PIskLdabYgevjXB7VsTPBx2oiAI05UpV6BAwbmGXq9Po7n6oN0Wfa4fqZH5Q2ihsTV1trI1amySR+DApakBgtncMBWIUqZcgQIFvwGMR48eGUjL8qpGxo/G1DNBkjyL45s1t2b2jybB1oZzzeXe0GqnKnP926K0tJT8/AJlIhT8KRETEzPMU2Nzm6OqVuZoC642N4V4C61NdAcMmlS7xgPdZqgi2BQoUPCbQafTjW+lrbVnjraQWa0DCXa7HdHTDG3U1AChocE0oX0z1MnOjMMsWrmBe5dv4v19uR70KzJZeXl8um0vr248SK3HUQ5rA9sPHeetjTtZnlmirOBviEP3XcGJHcXnwUgkfr56PAWn/jhzayvNpbqw4gyPrmPbZZdS2qXDbVRlHMJ+FhhxajctYcMTS8/qPMgNFZQeyW3z+c/XjCUvp3vmaEbGgT4eWlsLja3xpyVXm9QmQiq2MkPxMEMntr836/hqbyF9YmPEzRKUAAAgAElEQVS4tH8Um7ds4PEfGkdu5r31ezhSXMJbP2S2EGyleTm8szeb/b+c5ItjimD7LWEtK8Butp0HIxGJnrcQv8A/kOtg1SvsWbL+jAW9pSAPR1e6TVPOjqtvpNp8Fl4vpkoays4u/56U+wMb/vpim88teTnY7d07V69evZI9NbbTmKN4mqPNWptLsKmbo6GNTB4uM1QUxfYFmxjA4hsnNf0a2FDGX/bnwthYwJtnbp4JNTl8c3BnSzGc0J8PEvqzesNaPqhThM25QPXOdI4vX48dDQEjp9H/yvFNSbq24qPse+x1zFZfEv92NyFGX8BC3rIPKdp7FLtdTfCkWfSZnooAVP+cTmWlnoY9G6ipFIm76XaMSe4idUcd2e+/Q+H+k3jFD6H/wpvw9u44+9N06AdOrNoOohbv0a0Eb85+Mpd8gqnaiiF+CH0X3tA5A+xpxlCzdw1FBf70vXQEAJWbV1IlJRJ/URJl6z6iQW2kcnM6FmcwCfPvIDTW3/2wl3L8nXeoyCrBkDSO/rdciVbdqI2UceK99yg7UoBXz0QSbr6VAP9qMl9+n4o9GdTVlbF/cTZCZDKDb3R5bxwVWRx990OqT9USMPYy+l2V5k4wtpH30Vuc2nmSgLTJdEUBq/x+Bad+3I/ZVMOJFxZToBWJnLOQ8Hh/1/mWvUvejqOojQkkzp/fVNjesQJYzbGXn6U8u4rIWTfTa0y/Js0w//P3yPvhEKqIRBIXdn6+3A9eovLgAeTiXPYtXoygCqT33xY2kSaYT+xk95JPsXtFM+Ceu/Fzl7BY8jM49s7H1FU7CJ16NYlThiIA3t7eI4DPWpuijeYosiQ5JUnA6UQlqmVJIwlOu0021dRgkyQsXt6IjVLPnZSLLCNUVlb0p4vNWo6UlpNoDFGkyu8MuWQnG+Y/T+jMa+lz3RUI5poWRIAnl39F6CWzCdBls/XB15pMoZpCO1Gzr6P3vGkUvXI7B9ccA6BmVzo7/vY4mpRpxE7owY6rr6WiRgIcZNw9h18ynSQsmI+PeQ8b//rvTh9QTXgsYWmjKfn8VVqSbTSw86brcfa5mH7zb8Q/QMLa6Rv+9GPw6zuIwlfv48TuQmxZW/nh4f8QMCjRpfOs/4wdD7xI4OS5GBMdbLn2TswOlxWy85oZlJmMJN52Ixz9jB+eW95kofw07zIKCn1JuG0+4b28qM2tBLUvwePS8E8MQxvdl7C0NMKHJrpvKZ8tM+fQEJBCn9uuov7bZ9n9wXYASj59ll2fHiLuphsR9q2krLDzyn5d3EDC0lJRa70IHJNGWNp4fIJcnIiFSx5j16cZxN50MwGak6yfd1+XmHnr1n1KQ+SFxM8cyaE7ryL3mItEM+u5v7B/VTaxN80nNKyUzdc91On5/IeOIWRYEvhGEJ6WRti4C900Ty6tNGdFOsZZ16At3cJPzy9zObRK97Nu5i2IgybRZ95lFL58B4c3ZQGuZi+rV38X2Z6fDVkWZARBEAQQaHKjOe1WnKjRBIeiDdChbl3wjqs2dHxXHqZT2Yd587CTd+7sp0iW39t71VCJw6FHb4wmJD6CkOSW38dddyfRI/ojR11NxuUfYAO0hDLgrzdRsX8/5mob/oPiKN91EKa4+iYEXHwNfaaMAkYRt+I9cjcdI2hUJcc2Opm46y58NBB0971kpV5LVcO9BHdA/a8N7Ykx1MgvbTQ7K5ZqK4FhYQQMSCJ4YHLnQrx0ZwdjMDLytadZc8tt5Bhq6fviRwSHNpczRc5bSOyYYTBmMLkfDiB/bxUx6vXklvdn5r1XowL8772Dr2e8juXROYgZq8kpTuDKr253MZskN9dGhqemYj8UQZk2HmNqMydc1frl1BinMf6Gi5GBQXfdyKpFXzH8xlHkrlhJ779+jfGCOIz97+fEh1d2er/ePfvg3dMXjUZHSGoqoYZmoZG94iv6PvQ9xgsiIPkJcpZeQMGJl4jr3XEJl+aCS0m+Kg0B6DfzM3LTfyAmfjBHluwiedOPhIaokfvfT+7SERRlOYiJP319V0DSMHxV+QiGvBbz0IjeC/+OcWgggfZZrHrOVV9e8Pn7aMbfSv9po5GAgQuuYMfydJLSXJx1KSkp/YE8T3NUEISmZF0EQUAUZUHEpZhpvND5anA6rNgFL5dga+zu3vijUqnGdjbZFUU5LPj8AItumEmiQalm/72hipvEyHsyOHDnHH4otGK85u+MeGBuU32lV6Cr5F3QaREcDiTAWZHJ+hk34T1yKgER/jQU1yP5NFNSe0eENv2/PjSM6soSHKXlWC2F7Lr1xiYaHq+kJLAC3mcy8kCGvfoM+199mC/+nkfAiMsY8cqT+HawpxylRR2OQddnPDGJizheMJjxo1o2GNGHNdJ/a/E2+tNQXk2DsxipMoNtN9zYPKoB8UiAtbAYTUx8t+ia6ovzcRzdw5YbmvsnBPce7jpfaR1BYWFuVSwCQ8CvKQg1Yyk1o2u0mMRADGFgq6wDOuZI1IVFNbkpdMYwGrIqoaYIs9nMoXtv5WjTm+ACVNjc8uXM/KrqQJdTVaVRIVutrpEXFWD68Qibb9jcrPkNaS5yCggISAY2eGhrrh8BAVkWJKdTkJ2SIIoqJEmSHeZaoabUKht8fBF0trYa27333qsSBKFD3ui6igJu+Wgbd86dwbgIb0WqnBdQ0/OWB+l5y4PYSjNYN242RdfOJdrYge9m4wocQ25g7L9uB+D4E7vwDFg2FDc33DGXleJ1YTjqMA1ehnhGffwpBvXZGXngmFlMGDML2VLFjnmTOf7dHFJmn541Qh0W2eEYyr95kdzqIcT3+oWfX9/AqDua3cXm4kIgBbDRUFhDZEgA3uoIVJGpTPjstTaF9jZjBPbcndgkmujAPSEIAnKr7ik+EdF4pYQy8b2H2wqUcH/MpaVAHFiKMVU7uiwgBFFq1dtBjy5Mj6WwHHpHgFSFqRSMXaBbsZQWILs1GUthKd4RQeAfjt7gz5C3lhEeeAbKSje6yOgjjfhfNJOJT7fPyKLRaEYMHjxIc+BARqtEXUF06WuCgCzhtNuRNGrsZgsan2B0Yf7ovHWoW5mhPPnkE6OB01fN2uu4c0k6uug+VBZks7QAVPoA5iW7ily3HzjEyYpyrE4zX27PICE8kqkJoUgN1Sw7cIrDp6optUks3QGpA/qR6KdRZNJZgC1rO0c2FRCZOgBH/k6s2hh8AjoxcaJ60LAnnYKfR0PxLo59uw/99EuagxFrP+bomkH4Og6Rvc/A+Ff6IPhL9Blr48f7n2fgDRcjVxVSsCePIffd1iH7RtX+7ZhqrJhNTip3bkF9youQ1BHodKUc/NcKgsaNREsl9aV2gmPCOxy3EJZ62jE4s7az7R9rGfXNN4T4lrF+yhxOpPan91CXhC/65C1yUiMQsldTYr+QYRcEohWn0tP3RbY/s4Q+l6biKDxGSbbEBX+ZjXbQVGIj/s32/3uD/nPHYsvdgxw5iZ7u8/nH96Xqg684OTQIbWgM0cP7EThpDj4vXM7Pb/Ymbkwi5qxDVJvCGDRvIjFzrmTbq4sJCbmdqq/eoL6rco0QAmIcnHhzKdbkSAJSRuMbrCNu1kx+evFJAnwXUr/pA2oiphHVu3P90r43nX2fXYwxooYj3xxlyKfjQe1Lv5sGsvuuh0n+6zWo7eUUbdpPwmP30BmBvyo6Hn3BMxz6fBX+gUGEjx/VFHxpD1Gzb+TAJX8ho38IxgFG6o7uxOqTTN9LhjUpcO+9936vlJSUQzQ3U1YJICIIIrLsREZAFGRZlpFFAbu5FluDBrm6FJVarRbcibkCID711JNXi6J4eh+bZKfKpiLcoMFid2CxO7CKWoZHBwFwLPcUeVY1Q2NDweFArfclKcwX2VrPD9ll+PgH0i/IG4vdQVR4BBH635aMxmQy4XRK3W5KXFpaRnBwMCqV6pwec6bjQ3BStWs7RZs2U1sqMHDR04Qam7ejb78UDIHu6JYmgLBh/dH1HExgUA05X67BjJGk22Zj6JFAYFw4NbvSccRchL5sD6XHTQx8+p+E9/AFRMKnzEBVepC89A3UFtYSPHIsQXGRHXLml61bSdXxfHyTUhFrCzEXlOBzQSo6LzBnHaBo0ybKDucQfcujJI5N6FR7Od0YyrdvJnT2XzD2jUDQBhA5oi8VhwsJGRhPxfpP0I2di+PgJqrKfEh57gn8A3QgeBF12UxsR7eTv+4HTFV2wsaOwz8qBAQvoi+bgT17J/lrf8DqCMA4YQQ6b9dTq40dTEiUQPXh41jMWsIH9waNHz1nTKLu580UbNyK1aolYux4fMN8MQwYga+YT+43m9GNmU1MSi+Chg7Fq1NZJBI+MQ1b1iHq8/LRxSVjCNLhe8E4fOQCTq1aj80Qz/BnH8bg3bkqbRiehlfBz+T/nEXC/c8Sk+wS1EGjL8FHKCIv/Tuqsorxu2A0Yf1iOici9gonamw/ajIOYS6sxn94Co3MZAHJI9C7DTvBN5ywQQmIhkhiLxlBxZY1FG77CYcYTOTY0XgHNFuAer0+67nnnjuBq9+oHXAI4ACcgCQIoiyqRFkQBNReOlSSE6nBgtoQiKDT6URP7jWHw/6NIAiT/qiazfnOx3am4zvbyH39drJNkxj/wOV/mLU/cv+lmFKeZOjcFEXF/x+Aw+HYotFon6WZo61eEAQTYBUEwSYIzRxtWq2Lo83Ly8XRJroTcpFlWZg9e5ZKEIQRypQq8AqLwyci6A91T7qYRLxDFJ/w/4zXWK0eGhMT0xw4aNlMWWx0n7mSdRt/XKlras80j3fffXcYoPTDU0DE7PuJ+IPdU9ydLyoL+78Fn2XLPu45atToYzT3QBDdaR+NpVWCLMuyLDfn40qS1KyxAeh0uqHKXCpQoOB8Qe/evXu5hZpn2kejYKOlxubq1SJJUsvCUpVKpTgfziLCwkK7FTgAMBgM+Pn5KpOnQAHg7+/fp5UZKnr8CK00NyTJRWmk9qApEgRBuECZyrMp2MK6fYzBYFAmToECNzQaTRJtfWyNxfBNTB+eRfGeGhsnThwPAxK7esHq2lpyK+pxnC4nz9m2qazVbCa3rBqzQ1ZWTIECBZ1CEISEF1543qelYJNbm6NCcz6uR/AAEIxGY5eK3pHqueuVFey1awn3clJo1/Pv66cxIkwPWHnuo2/5OqucUimALU9dRbT7sPdXruSNzDp6BHiRV2VhwRWXcOuAMGXlFChQ0CEmTZpkBCpoIpxEFAQED80NaPa1ybKEutFG1Wi0XRNsooYFs2aS1MOV1r7yu1Us2nCQ1fOGAyomj7yQWyeamfjmnhaHjUgdxTUzw9GKkH90D5NXbOPyAVd0UtGmQIGCPzuioqJ6ApluQdaicxVtoqMu4Sa6JZ2gUokDunYZryahBtAzyB9LE8+MmiG9ogjWt8187hsV3lRrFxbkh+iwY5WURVOgQEHH8PHxifMQaq1b8jXlsrk7WCFJktCU5CYIwoBuX9Fu4r2dWcxJ7duNg2Q+2XqQYcOTMCqkIAoUKOgEGo0moZVQa+Fja52oK8tyUwmYAHSPVE2y8/oXq3EkpLKgf9cNyh0/b+Xjcj9emDRAWTEFChR0ClEU+3hoaW6hJnuao9AigOASbEJpaUki4NX1Szn5dNVqtopxvDK1P11VvA5m7OKxn2p557o0ghVSDwUKFHQNXitXfhHiIdxEWaYFh6RnAEGWXeke+Pj4JHT9GjLp69fzWV0I71yZgr6LUu3E8QP8fVMBb9w0hRhvxQZVoEBB19G3b99QTp+k6yHUXOaoGhDUanV8l69gLmHxpixqdV6kPeui+dWG9mL7wgkAPPn2B6QX26izOZixaAlJg4fx4YxB/Gf1DgorRa55+UP3ifS8d988BuqVRVOgQEHHCAkJDW9rjrbU2jzMUVkQBEHvcNhfFkVx/p9hgn5L2iIFChScHZjN5mXe3oaluOiL6oA6N4WRBbALguAQRVFSq9WyRqORRQBBEGKVqVOgQMH5Co1GE+nWysTTmKKeaR+uLwRBiFGmToECBecrVCpV9GmEmuAh3AAZkJu+VASbAgUKzlsIgtCjlcYmyHKLlA939QHIMognThwPB3TK1ClQoOA8hu7f/37Jp3ONzWWOisHBwZHKnClQoOB8x8CBA/08BFmbdI9GoSbLMqKXl1f4mVzkTGiLzOYGcsuqMdn/PLRFsuRAkpSiWAUKfi0iIiL8TqOttf5BrVKpuifYzpC26KOvVvLmURPhvloKK03cOGMqCwf9wZXFwk0sT74eVY8IZJsFr7gUhr/4EhG9Oy9BM+3+mgPbHYy8+0plRytQAPj7+7fW2E5nigqiKIqh3Tq7m7Zo+wPzWHn3dTzQ34tFGw66v3TRFq1acBGtya2vnDqd7Q9dx8o7riL92qG8+s1PVPwZVsMrkUl7djP74H76DnGw/f/+X4uvHTVl1OYXI7XQ8iTsxdmUHvwFydE9jU8GcFioy8nBbpPc53N9Z6sqo760ps0x9poyak/l47ApmqWC8xfe3t5+pxFqTdTgjaVValEUu0mJ5kVSj+ay0p5B/lhKWtIWUZPT5ii9R0dYCVCrxA4b7P7xoCZseAqWzUdxAiosHHv6Lg5vyMM/VENtbRATPltCQGAdO+ffRsWJHBrKJNbN3QUBSUx490m8KCS97+UM37OTUANUfrmYbesNXPbGX10K4pL7ObBTQszehVPjjdU/mZmf/JO9N6VSIQxBNlXTcDSDiL+/xYibxwKQ/cJC9n2TS2BsELW/lHPhqrWEN+0IC/v/vgD1RXeRNF3p86Pg94VWq+3Ux+bW2FALgnDmzSMbaYsmXdalP/9pzy5e3ZXNyQor/5h3KUF/htWQ6ijetIlqSw0nX/uCuGueQgVUb3iPjB0aLtu4Bi815Lw8n31vrWLCw5dz4bsrqPnuVTZ/ZWXK2/d363JVO49yyaZ1BATrcFosTauu6TeVCQ/OwLJ7GV/eu4zhN49FRSlHl/zIhVsyMIYDDhvOFmW8EjWH9uE1sFp5qhT8/qqBWu1Hx741oVmNgMAze2C7T1s0sG8/njLGsG3vHl7duJ+Lbh3PH751iVRL8apvEYqPUaMfyuQbJgJQtn07GkQyF/8DCbDnVlBlPQT8us7rARNnEhDsyt5R6ZqzeMKGDAZAFx2NWF2NHVARRMSIIHbfdgM9J40lIu1iIgZEeyr/jFufoTxRCs4LqFSqdkxRGRCE1g4ZEfDr/iXOjLbIYPChd2QYN00djV/+MX6s+BNER9VRJP/7JcZ+sopekcfY9Vp60+QbklKJmjqVHlOn0mvhI4x9/JauSMrm5WsnJK31aZ9VQFC7V0lsaR4PeXcdo/7vFrT2ArZfOZkTe8uUJ0jBeQlRFH1o41trWVLl/gxREIRuCrYzoS2SKKoxNf1WWVpMnkNLoO7P5GUT6Xf/A5QseZ7KGonQUSOpzTyBf3IKYUNTCBs6GENQs/6q9jVgL6+kpejyxcunlobCBpfWt2/PrxyTDZtJInj4WPrf8zgJY8OozSpp8f3hJ+7m2AZFa1Pw+0MQBJ9OTNAmXjY1dNMaPCPaor68svQztpt1hOpFcitMzJl+EUP/ZC00NbFp9B3/LAf/k864B24lacfdpI+bTECvcOpzfyHqjpdIuWqES7tNmUokN/LNuIloIlK46LPn0OFL7+snsXPWxRxLCEdrEMHn14yolC0XX47TmIiWSmrr40mb3N/jezvF336ONmwKfSYOUp4sBb+3YDPQsui9TaoHzRJOKgCMv8XATA0mKiwOgn19MWh+H7LJ8422yGmqoaGqDn2YEbW2a3NiLStE0gWg9/U+CyOQMBcX40CDT0TonyxSreB/DEWCIC4ETLioi2qBekEQzIBVEAS7IAhOlUolqYHfjOrR4G3A4K2sjidUBn98Df7dOsYr9Gy+h0T0EUZlIRT8L0DXgRnqAVfwQGFOVKBAwf8CWsuqdg0MWUYRbAoUKPifEmxtAgbtCbhG/nAFChQoON+h6kBjayHgGssSFJwDHDp0GJvNpkyEAgVnB2JHWlp7f6hAgQIF/wsQOvkdQBCBblI6ONmZcZhFKzdw7/JNvL8vl2adRCYrL49Pt+3l1Y0HqW3vcIeJ9zf+xJb8GmWJOoKtmMx/PUfe/tzf5fK/vHQv6ZdMYvnoG2g4h9cxZ6STPv1SVg4fwtHv83/XKa/POoSpRtGwz2N0WVaJgLN7p67jq72F9ImN4dL+UWzesoHHf8hp3Ka8t34PR4pLeOuHzHYF2+ot23jjxwy2FNQqy9TRQ7bjaw59sJT9//nyd7l+wj0vMu2T5xGrKjmXhW/6QdOY9u0qeiV547D+vrRJJ55ZQPbOYmXznb9wnkZba6O1qQEboOm6lRvA4hsnNf0a2FDGX/bnwthYwJtnbp4JNTl8c3Bnm0Nri7NZWujD7Djvc6oF/BGQt3oNve59iPyXP6bW9jf8uhC7rt6ZzvHl67GjIWDkNPpfOb5pxU1HtnNs2TeYayX8L7yYfldPRIWFvGUfUrT3KHa7muBJs+gzPbXTJN2K7avIXf0jDVUmfAeNoe9Ns2lkparbu47SmiC8Sn4id+tRfMZcweC5aWf8gi75binZa3eBXxTxtywkNNbF2SA3lHLs7XcoP1GElzGGXjcsJDjaxQJoOryVIx+uxGoV8Rk0lv63zOhwg1uzdnFk+QZKjpYjrngTxx4//EZeSq9xSU3nO7b0K8xWLVFzbyQ2tQ8AhV+8g2wcROl3KzFbfUn8292EGF1jsObsJ3PJJ5iqrRjih9B34Q3odR3fE5g48fpSgqdcSM6SpVgcvvT52/0EGw3KA+G2Y7qjsf0q3ftIaTmJxpAu7FEbz337MwunDsNLSW/vZK6qyF+XTeTUOUQmlpDfBS1CLtnJhvnPEzrzWvpcdwWCuaZJ07IcWcOaeQ+gG3oxfW6ag1DW6D6oo6bQTtTs6+g9bxpFr9zOwTXHOr1WXVYugWnT6bfgeqTMz/jhmY+av9u3nr33/Y1T2V7EzJqJ2lF3xhpf9YbX+OHZdKLm3kxknJMtV95ErfuNePzZ28kvDKTv7QuJHNIDS6W7FtmWz/fz7sI7bQ59b5mLHhOOTq6jDooiLC0N7xADPoMuJCwtDf+48Oa5u/5RfNJmknDFOI7eez3ZB6sAKPrmXXY+9yGhl8wmQJfN1gdfc5+xgZ03XY+zz8X0m38j/gESVnvn9wQmsl57kR//7zX8R15M9KgELOV1yvNwesEmt/q3hcZm5gypi05lH+bNw07eubNfp3/7884dVPdIZmyoln3KAnUIa8ZGKnxSCTdqkdLGcGTtOvqPub5jWdhQicOhR2+MJiQ+gpDk5u9ylr5N6HWP0n+GS3MKSR7m/iaUAX+9iYr9+zFX2/AfFEf5roMwpU+H14q9/nZqj+ynpqAc/6T+HFuxG7i2+W2ZkMaIh29DAKJ/xTzkLl9B3O0v0WPEBTBiEIUrB5H3Uz4D0qKxVlaiiQrBL74vIQOSPLZ+NVaTCl1YBIGD4ggeNKzT66gCjRhTjZSE6NEkDsGY2rN57t7/D2E3PESvscOQgb6zh5Gdvom4gS7K9rjr7iR6RH/kqKvJuPwDbIAWK5ZqK4FhYQQMSCJ4YHKX7sn1iJrpc+9zxA8NVh6EtrB0R2M7I6uwoiiHBZ8fYNENU0k0dBJctVXwj43ZDA3XsP7wSU7W2MkrLOZQhUlZqnZQuG4tGr2Fo6+/QUVePWVr12Lt7OGMm8TIe8Zz4M45LB+cyrZ/fdrkkDAXFOAbF9fWYVGRydq0NDJXrKfq0GEaiuuRHJ3tHRO7r7+YH5/9L5X7D1OXXYRks7V4ZfomJp6VmlNLZSXeIWFN72B9WCCW0nIA+jzwAl55X/PtiKF8fcm1FJ5wE837JDHypds4+eQCvhiQzKYHXsbqOPMxmEoKKfvyTbbccCPf33Ajv/xYjk9ws2noFehiIRB0WgSHw+3dDmTYq89Q/s7DfDEwmXW3Pk6dSer0nlwLGYJvvCLU2rVKZNncjnbWnkEgq3EVlHYLdRUF3PLRNu6cO4NxEV0o/pRVjB0YS27uKXKB7Ho79tJSTlTEkhSs+A9av5Ty128lfMK9qAxaVH1HEpD+TwoOmug1sKO5UtPzlgfpecuD2EozWDduNkXXziXaCPoeUZRkZwMtNevKjStwDLmBsf+63WXePbGLUy2kpRrR4Wixc+T8HZw4HMGVe15GC9RueI3DOzJddH+Nb8szSCIS1CqkRnvNDV1QEA3lpUBPwIG5tAq/MJfbwyv2Ai5862PAQdbiBex/awXGFxcAEH7ZQsIvW4izJp8t0yaQs/cG+gzvolHSqr+EIdxI5Oh7GHHbmG7dT+CYWUwYMwvZUsWOeZM5/t0cUmYndXhPdDJ/topibA4vfMID/6yCzeQhzDr0cKhlWa4VhG68X+113LkkHV10HyoLsllaACp9APOSXer79gOHOFlRjtVp5svtGSSERzI1IZT7ZoxvOsUrH+dTlTCIyxNDFTnWWovK/5HC3J5Mve+2poCBfPAr8jd+T6+Bl5xeKc7azpFNBUSmDsCRvxOrNgafALfpeM18Ds97isx4HWFxvpR+n0HcXbfgHdWDhj3pFPw8Gop3cezbfeine1zDpxehsUVkvPAhEX0iMU6djC4wCm/zQbLW7SDYYOLwqx8DQ371fYemjmDXf1/BIE3Cf9CFhMSF0nPOLDY8+S/C4x5CylpNflEiF1/oMtmyP3gVoecwfEM1VJzIwyf5KrcpcYgDy/YQPiYFoSaLBpMvvl10vvv1TuDY8nfxFcfiHTeI0N4RxN6wgMM3PMbRyCcJ6elH9d4fEHvPoNeouA7OVMrBf60gaNxItFRSX2onOMbls+vonjrDL09eT2bpSGZ99uSfVbDVd6CttRB2qieeeOJKQRASu+7YtlNlUxFu0GCxO7DYHVhFLcOjXR0MjuWeIs+qZmhsKDgcqPW+JIX5tjlNdHg4sX6/fbcOsiwAACAASURBVAN6k8mE0ynh59c9fs3S0jKCg4NRqVTn9BhrQSHqvsPokRzf9Jl3lBGHXUNo/9gOVB4nVbu2U7RpM7WlAgMXPU2o0WUmqUMTiBk3gJK16RT/fAh971TCB8bi1XMwgUE15Hy5BjNGkm6bjaFHAoFxjR0ZNRgvvgjzsQwsJSYChqag9Q7FmBpLwcovqcqtp99dt+MdHkV4crP56RUZ73GOrsE3eQw+YiW1WbmoIxPxjwxA32sYoUYbp75Kx1QfxJB/LiIo1EVG46jMpWjTZkp+3IchdS7Jt16CSgREgdqMHRRt2UTlL1UkPvg00f3CuqZlDRuP2lxM3S854BdFYGwY6rAEYsb0pnhNOsU/HUDwjcU4bjg6gyvO6tsvBUOgex9rAggb5mKUNmcdoGjTJsoO5xB9y6Mkjk0A6PCeXNASNGwoWnX7YzT0HkzogNg/50vf6Tz69NP/+BlXEMHa+K8gCHbAKQiCJAiCLAiCLDidjo9EUbzmzzI5vyUf27nicFOg4E/ppLFYVur13v+lLR9bAy342ERJlGW5UpkyBQoUnO9wOBy1HianfHpfm4Aoy3KFMmUKFCg432G1WhvLlVoLtDYCTnQ6neXKlClQoOB8R0NDQx0tk3JPGx0V7XZ7iTJl5wZhYaHdChwoUKDg9Kiurj6dKdok7BozPESLxVKqTNm5EmxhimBToOAsoaioqK61QBOENmVVsiCAWFRUXHRG0rO2ltyKehynI2RwttIQJRmHU2r6kaQ/z4LI7nx0uR3WldMVuzV9LikbWoECgIyMg/XuR0NqFm5CO4+PgHrw4MGlkuS04OoA0zmkeu56ZQV77VrCvZwU2vX8+/ppjAjTA1ae++hbvs4qp1QKYMtTVzXVCn61bjWP7CjAz52gM/rCUbxwUeIffzVq9vDFpH8z4+cnSB9wD5MOr2pq5Jq/5B5O1kygn/EAB44kMPGpuU2HbZuQROyHhyi8J5nQp3fTq59a2dkK/syw3H///Q0dmKIt/G2NT8spoGtSRtSwYNZMknq40tpXfreKRRsOsnrecEDF5JEXcutEMxPfbNulfOqECbw4PkFZIgUKFHTP6pHlvFYCTPLQ3Frqa4KA6DpG7gZNq1eTUAPoGeSPpanGT82QXlEE69vXLqqrq9hyJIfsGsufaEkEVF5egIhKr29RHC6qvBC1IoJKhdgq1VzU60EE0Uvv4jlWoOBPDKfTWdBKqLXytQmyIAg0Bg/UgCxJUs4ZObntJt7bmcWcSZd1+qcaLx32knJW7S9h6/J1TJ8yhcc96GH+sPC/gMu3LgHg0t3LW3xlvHExrlbF02hNxTjqu10A9Ph4h7KrFfzpYbPZitrR2NpN+RAEQVYDstPpzO62YJPsvP7FahwJqSzo3znNyrQJaUxz/39Z7mEmvLed6y/oSaxGWTQFChR0jPr6+qJ2TNDT+tlEtzQ82U3FkE9XrWarGMcrU/t3u9VVaHQkUY56Tin84AoUKOgCSktLy+iCj63RHBUBuaqqKqvrl5BJX7+ez+pCeOfKFPRi146prDc3/ZZ17CQF+mD6+CgLpkCBgs6RmZlZ4SHMpE5MUQRBELwBrSQ5iwGvTq9gLmb0oi+o1XnhJbocddrQXmxfOAGAJ9/+gPRiG1VmB/56L5IGD+PDGQO475X32OPwwV8jkVcrce/sqczrE/KbT9Bvye6hQIGCswKrIIjX4mpjUI+L2aNOEIR692c2QRAcoig6VSpR1mg0siAIgt4t2LYAyedubBJl1fWYnQLhAb54/U4J+YpgU6DgfwuSJGWoVOpFuNoYeAo2k4dgc7oEm0rWaDRyY46BLElSpiiK51CwiYQG+CmrpECBgm7BZrNl0da31trHJnuYoYiCIMiA7HA4MpUpVKBAwfmGurq63HYEWpOPrXUOW1PwAJDNZvNhZQoVKFBwvuHUqVMFrQSak9OkfDQKucaYpnz8+AlFsClQoOC8w9q160rcgsvpIdTapHs0maGi4Ko8AOTU1NRySXKeAHr/pqN+59K2n11wNaTMU1ZUgYI/OWRZ/uXRRx81tzZD3XRFnpqah3ATm4rgZUCSJGmvKIq9/8gT5VWTi6owE6q715TW79QpRMcJUCssGwoU/Faw2eyH2zVDZSQEWQJBRpZBlmRBlhGcIDjkpuABgGy32/d1fiknOzMOs2jlBu5dvon39+Via5avZOXl8em2vby68SC1rW3l/Bye+XIj963YzEeZhX+6RSrft5mxT37R4rOT275l+r9Wt/v3P65ZwejZd5E861H21nh84ahl0ZP/YNAVd3PJ4q9/25twFHPtTQ8y9IqFXPvGD10+7IUnH+G/e6u6dall/3mN0df+negZz3A22VC/W7n6/7N37nFR1fn/f54zF4ZhGEYYYERERFJDREMyJDUzc83MzFxzy8y1sjIz13Vba11z/fU1t227mGtmdjMz18zMTM1YNDPXyszIC5kRIRIiIgzDMMzlfH5/zIUZwGvqVs778TgPZc7l8znn8zmv876+PpzR7HM3sKuolBMtKL/mjZfJGTWZnmP/ScV5e/Au9u0rxtaMn2/5i08z5/1vz+hKBatfp++46aRd+wd2285hFxUbU6bP44D93JEIHrfaDrZihgb72PCTTUpIyFJT8AB/ZNRqte46defrWLOrnC6pHbghox2bt+Qza2uJb2cDL3/4BfsrjrBo674QYDtWdoDfvbqNuLbJDL00GdnlvuiAze2op/RobchvsR0yuOvqLq0cXcfsf77HzPlPsnvV/5Ed07Sn/LOPeLk8gZ2rn2X9wzde2JtQW1j2yt+ZNyqLGrvr9EH9aCVWu+eMmrr1nslsWzgJda2Vc/Gq5G94j4IDxziw5xuq7ceZ//Jaqk5nGtpLGX7/i7S6OIi9mBkvFrJs6bPsXvZHLOftwdcycdJcipoR41irq6mud53RlQaOvJ1tS/9CkqvuxESxZyGFm9ayr00WnfXnjo6mqPjQD820tWAfm5D84OYDOF+qh1AH26mjR9+ya/PmAhtw4mIn2cTj468N/NnGfpT7dv8A/VMBPf83YQTUlrD2609DTluxbRfXDRrIvblJFxWYOY4f5smX11Jcr2Voui7kC7zstbcosipYLs0OOWfntv+w5stvKax2s3rlW2xTRTLxzptJibSz+MV32LX3axxVGuY8t4zYjplMG9bzZ3GvblsVi1fms/v7ChSdkdE33cDgjPgmcCv9hofmvIlNn8CU34+ga5y30MV+/DAL3tjInnIbmbl9mTq8F6dKg1YajrNo2Xvs+P4YsYmJjBs9guy2+pO/0AOuYtnKtbz8SSFblDU8NP4mzKfwLGz5YB35XxdjrT/ME88tQ4/MrWPHkNFGJv/9tWzZ8y1ljgaWvPQmel0c0+6+DtMJr9bI0tc3UeOq5qA9gaGdnKz8/BjT77+djDgNuz//hBUFeyivbSDt0h5MHXM1Jg3s/u9mVu08SIm9nkWLlmHRSAy94UbyUg0+Bb6KJxcsZs9RGD/2twy4pM1PGEQ7y99Yw6ZvjmDpeAlTbxuKxQdUZUW7eWLFRyhtkhmeFY8jthPDe7QLWHKL/l3ArVOfDrHuVix/B2Nqe9Z/8Cm6xI5M//31WHx1mDUl+1m230mO9keWFBSR0OkyZk64Cj0u1q9ex6ovfrC9sWHrkWZamgdQhNcc9VODC0lCKIqCrcGDrdGF7HO6CUD56KOPFCHEjjN5Dvsrq+icdKrSKBd7S2uJEce5c/Fb3PryBj74oeZi8BAwfdpsDuq7MGlYNsveyQ/apyI7pyddtUdZlB+aQpjcIZ0hud0wSRHk5WYzJK87Rg2Aht69LyOvcyIGcyJDrsym/6VtT9oD66EDzFn4JrObbZuKzv3iZM7qCqq1CYy/ZThj+7Vn6oN/YXtVk5a25J0tDBo2hNyoHxnyhxexAzQcZdSER6mM6cyUW6+hZNPrTP/316dsa8n8p1h/JJppd9zI0O4JVB1vOOU5xQf2s+OHOpLj40nQOdj0yd4Wpl1zSet8KUN6X4JOY6T/ldkMubInCTrvi5ne9VKG5KSj0xkYeGU2gy/vcgoa6kaWvvImcqfelH+8nMV7teRGVfDEu18BcLCkgpwr85g2dggcKGDMs5sASErpxJArM4jVaOjdO5shV15GSpsm6F+97j+k9erLsI6C0X95iZ/CLfHqM/9gfmE9E28dRuyPXzD4ryu86lFtMUMnzScl9ypG9zIybfZ8Vn8ZZNDbfmDTgSj6dosOAbZVb77BjNc/Z/So64it3MnIuU2uk5ofipj95HPM/28NI6+/ilSdCwdQsHwJU979jtt+e90XTpdbacUE9ZuhiiKEkPDqbY1uBY1GTYxeG4iKBsDN7Xbv0Gg0g07nIZR+v5fn93p4cfKlp/oMYHO5Wb/vCM+PGYat/BsmvLaRjg+N4ddMDq5U7GXF93HseWkgFhlm3t6fUR/4H7dMRrcM5MO74fvQ8yztO2Bpr8OgiSCnVwaZAa1CQ88eGajLd2Moj6RvdsapLUeNlgRzSx3CoDn3NW36lEymD6/hs2++wyZMZCQ4+GxfNXn9vVrbmNt+y+DsS6DnHbz81n1sOTyJ1D0F7Evsw6rRVyADc+4cSsbTW5l/S/eTtlV93Iq+bQxpqR3o2SXttPonGyzMe+Q+Xp77DENm3AdF3yOfwmpK6diJlDg3Wm00udkZIaZmaqdLSI2pRxcRTV52xkk0teDvWRyD+l2K80MzSq/LGOCpZ9UX3o/MqN/eRPG3Byn6sZbMrqnMX1cEDCahXQoJ7fQYNVqyszPIaaaYDhx2HSP7ZMDl0cxeOI8SJ2ScVeVfDUvX72fmq4+Ql6Ihr9M4lvSfy277rcgff4IzaxDTh3itgwlXf0hh8EftxzLKYxJIaUUDvvfO2+nfPYb+KWNY+JtFFD96E2m+4+yGNBbPHOM1EfMAFJa//zFTJj/HZR21e1oxQz2AR5K84CYEICH8a7qo1SoiIyJQ+3I/hPcgIerq6j6LjY095SM49mMJ97z1FXPuGEHnqFPZ1Gr0ahjW+zJSjZFgzKJP1HY+K7f/qoHNWVOLrU0cZt/jscTGAxd2UTA5QofF3NI0MepPHd1dtuBJZq7dD8g88dx8RneJPPmHblc+g2a+y5BBuSREqamye3C4m/w/yWbfvJKNWGJcVNY0IldWYj34DSMfPBQ4rnfHS1HgpHRY995/P48sWEH2sNfRJXZk/v+bysCOJy/ZS+vUAYAhN19PEmDs2vHCTwpZjVoGWaVC1khoZRX2RgE0MHPKI2wRqQzqZoHjVThcpxeBNxt9961WoVa5UcRZ+xKorDVgMftIEiNNJETZKD8O6toaLOYmWv+E2JjQjzigyFIrF9WQlOA7NiaWBMlKpR3SfF1O7pjczO/VSEVVI0MsJg6UfFvcGqgF+9eEd0OSZbQo1Dc4sNkdhPjYAGXSpPt3rFjxZi0Qc6L7rzt2mDuXbWPymBu5yqI/jSemoWOiAZfbHXgMLo+ERv51L02njTVhPH6MKgUsMlRUHb3wPr7qo6z/6PMWzvchplQy20af9Nyxk6czdvIJ7k2jxuEKDQasemcDIydOZd6IToCL4i3vhOwvq6oCzKBYqajVkRQbQXJCAglZbdj41NgTaDgqZLenRUTS1L4zC/8+C/CwbP4/mPnaFrbPHn5az6RrtzPMaJIlZMVzxgEMm7WWKrsg1XIautyPe1n4TRylHzyIASjZ+jZPfFESdIAE0pmv7ua02yg/3khyuziakc+jVSk4godQbSAhxkZFlQtSNGCvobLeQFIb0JnjKd3WFPMtqzgKQd8RXWI8CccrKVcgTQ51Q5UfqYEUE9RWUymMmIMgQ24BhhFYzBGUV9TUPvr4M2UnALWQkirZV3EQqdPSRqsRGhVejS1IxMqVK93Ll7/xsSzLw1p3l9Ux+aX30SV3ofrw97x+GFSRJm7t6aX5/uSrPRQfq6LR08A7nxSSntiW69LjubHXpUwo+C89jTnUHz7I54qFv7aL+FUDm5yQwa3px5jxwmYmX2HisTc+hoS+XsCpPcq2fYcpPfAj9qNq8v+7G1NCMjmdzi2Vk6lTdxb/rfs5v7es7t0oeukdFmfpSU1uz+AeHUhpl8CKrVvY2VVi38cfsP4HJUQjX/nG2wxOGU3lp+9T2qEffdvK6E0DSVv0CNNea8etV6RQWVLMHnsbHhqZ4z0pqh0D2h9jzvMbGNzJzMCBlxOrhlX/XgXtutLZrGbn95WkZ15z/gZSn0hXdRkLVnxEXtsoci7P4XS+5xvfXMyYdR5s78049RJwMfEkOb5j2Ud7yNE7eOKlDwnNlTeR0d7DoqUbqexmJiMri7Q2p7Y3Cze9zeVzd/HtZ88SuoxSDP17RPHUorcp72UhL68PyQYT44ZeymPPLiX2zv5sW/Umuv5X01MPcr9+JD89k+mvpZATWcWrnx4hN3hwjekM7ljL9qIG0jJCNftFLy8jM/I3bF+1kuRrB5B+UkVU5tbr+3HP8y99evD78mAz1B0CbsK/MqWvRlQo1DsEkiSjuD2o1Gq1JIRACCH5tH/54YcfTlCr1a372RQXx50qEqM0OFxuHC43jbKW3sleM+ObH0o51KgmJzUe3G7UkdFkJkRjTkyim76RDV//wI9KJA+P6Ecngxp2vdmyjbbdIan7eZmjjdXlKHVH0ev1Z3RebW0t0dHRyPKZhLJVDOiXw4GdO/jP/qPcccswOqUkkdu5LbaKUlZ+vIdaKZoeSVGUVlRh15rI7hgMbFp6ZXcmqhUNPyougZyO8f8z0NaZUxjc1cgXRT9Qr4mhd3oiGd2zEOUHePujQqIu7cu913SkU3pnUtp4P2DXX3U5/8nfzA9SIgv+fCuJkSrQRHHzb3L49ssvWffJV1Q6tQzscxkd4vQBN8Y1A3L48btiSqoa6NGjC9FqsNdUsOmTXeR/foCU7EHMvjWXiPO16I2k4zf9u1F8sJji8mOkXZpBfGTQoGgM9OmRSmvva3xSCld1Twkyq1X06JFOJJDcsRPJMVp0JjOXXZLG4Oy2rFu/lc/LGnhgwk20izfTr1v7wAt/Vb9eVJV8z7eHj5KYkk57k9YXXOhIekJUYM7kZHch2DsUbbYwoNclNHck9L3yCuorSvjux+Okde1Ggl6iZ+/e6Ku/Z+XmL9EkZ7LgwZuI1kigMXLToCwOfF1Ejb49/RIaOGrqyg3dLYH7ShBVLNhVz219OgaCByvffI8bxo5ie/4WGi0ZLHjgBqI0Tc9OZzLTOz00SaZj957EWg+98872vYcAJ9AIOHz/OgGXH+BUsqyoJAm1SkZRBE63B6ciIel0OtntdktCCFkIoRJCaEtKvu+ekpKy7YK8IRe4pKr2289xl+8jLu7MKg9KS0tJSkpCHa48CMtFKkXffEdqp06once59fd/ZOjMpxnfPchj5aph1L1PMefZ2WQYZMDJqBvGMXbRCkacGf0h7+Rvv2/kH+b9SCgHm02CeiQaJSSnJOGRJElRy7LQqmURodUQEaER/uCBkCQJ4Y0vCEBJTe1YpCiePUBmeDjDEpawAGz9cA3jZpehSJEMHTWJcd2bueE1Jla9NCfEL5jetROxZ7hgkyLEvpF/mHesmRnqpmW6h5B8OWz+IKhKklHJEmpZlgnyswXAze12F6jV6l8dsLkjYnDHpkFC4pmZsLUSIj4VNOFltcJyccrEOS8w8QzPmffaW2fcTr2t/tMT+NY8SJIX1CQEkoQky94Is1qLSqNF1kYgR+gC6R5IkuRT2ryIaLVat8TGxk75tQ2OR2fCJUdBwpnpxo2VLkT8JRCmBg9LWM6rfFO6cx+haR5uwCNJUpDGJiFJspBkFZJKg6zRImt1yFodqgh9k8bWLJ9NMZvjtymK5zDQ7rzeRfbvWv7Wtnt4dMMSlotQhBDll1/eu7SZCerffJRFUghrrizLgU2lUqFSyV5gk2VZeDyeYFNUAB6Px/OBSqWacF7vJMy7FpawhMUnDQ0NW1vT1oK2EB42WZaFH8O8wOYFN7k1jc2PjHV1tvwTdaDGauWHY7YTswN4QtOfFUXB7QndfvWiVLNi8SIWLVrE+u1Fv7rbqykvYvWK5SxfdyEC6E4KVq+m0nERvN2OKtbnf3bC3fu2b6SwNLTWeveWNRRVtF4lWlm8k5XLl7Nmy+5m89PJZwVrWb5sGXtKm7h4PitYT5Xzf3PrBw9+91UzUHMD7lAzlBYaWxOgqVCp1F5g828Q4DYSgBIbG5sPHA59GDYeePpVrl+0jj8se4e+T63iv5X+AuRG/r7sbfL+9gLpc1ZSFjjJxUP/eo2+83zb4y9z6cwlfFQjfuUzVMZgMFC64y0Wrt7xi+u9u2QLo6c81fpO6x4G5V3P9qJSKiprOP+fKTtPTHuQwouAO2HdM39i/b4TE6WtX/wwq7eXhfy28dW/s+2AtSVGlmwid+Cd7Ckto7I69OGtnfd77n3ibcrKK7Dam2o7bHvWMH3+xv+JGdqjR4/vfeDV3AxtKqXyFYYGY5cX0Jo2tU+Fk5rSPiQhSShCCAVQPB7PBpVKdVfTu6rhnlEjyGzvLRNZvf495uR/zYZbewMqBuflctegBgY9/0VQlzU8+cDvA3+VF33JyM3H6WOSfuW4ZmLYrWPRVX9CYckJtJ7KMmrcOlKbM6QoTspKy5ANZpLMoTWQDls15ZVWzEnJGHXnKa9OUXDUlLFp+1e43W6QZdT+5GRFwVq0k1LLQObNnO6dWKd1TUAGe00VlTZITW66Z6e9hrIKK0mpKeiCL+Z2UFJWiSXlDJduVJyUlZajj7UQa9SFdKKyrBS33kxSrOH0AN5hpcoGllgtZeU1JCVbmu7XN046cxJmQ2g71RXlWJ1qUlIsIc9HARS7lbIqG8kpSaGJvc5S5i3ZycLdr7R4eBWlpejMLWm/FLeb6Us+Qa2WW1hJxTt2YOh7C7OmT0OW1SHnbN++g3FTNjN5cGh+5sDxk5ia/QClU4aQcgFjZXa7fcsJfGueIP+aAgEfW8AM9WtsarUKtVqN2mePEmyS+sKjCuCpqanZEBcX1wRsRJDZvqkUKiU2BscRf6GzmsvS2kFtyUlv4K2dexmWfTUXdXxRsTF3/A28WugmRV9DlbEvG9c8j0UH9rIdDB50G7r0nsjWMmL738+Kx8YBsOXVv3DX3PVkZaZQeuAAM5Z/zqgswznv3hOTh7N+5w/YDlQxaMh1qOVEFq5dRmcdPDZxCJt2H6bmYA1DhlyHnHoVG5fMPCW4je/dBn3/CWzN34JedjBgygs8MSGPtfMfZOqCrWR0NlNU2sira9bTN82IrXQ7gwb9Dl16Fg6Hi5ogjqExmZGMWtXAqK5Qs2MJXad+SsWOFwE4sG0po8c/SmznLJxVJQya9iKzx/TGXr6L0SNuo8qYBlUHSB3xKMtnjz1pv8fntqXEnEvprs/omtsHa/FX9Jz4AgsmDaRy3yZGjrkfbXIGNQf3MXD68zw5cRBQwcisflQmdCaWGg5Um1i5/m2yLDo2PnM3j6w9QiyN2KuKcGbcyfYVswLvQknBKio6/4aQIXVWMWnENWyrScCEA7etmsFDfIpz4TpGTnuWol3bmL78W6YOSfadVMWUkb9j58EDFFfKDBmyGUvuLSx77C5Kty5jwpzXKNpVjqH8NtY9pWPcrJcY199bFomxJ4M7l7OqoIRpQ1Iv2Cuxb9/+L09ihgZpbKFBgyYT1GuGqtVqSE5uh9lslqKjo6WIiAhZrVarZFnWSJIUKUmSUZKkBCGUfUIoosXmrBP3PfOSWLj3aOjvNcWi56w3xaFWznHVHRJ5s5eK/Q2tXO8CbEeOVIhDhw6d8Xlff/21aGx0nFWbHz53j7j+jy+H/PbDB0+KxF63iFqPIoRwiD9ff6l48IWtQghF/Oe5e8XV9y0MHBvc7u2XxYhXPrf6/naKxkb3Sdv+8aPFom1iokhstj36xmen7HfD18tFTK87Wt13/POXReKV95zRc7ijV5S45r5nhMt//Qa7qP1qlUjseJX4vtZ7H1+8MUP0un2uEEIRz97dR/zu8TXe9r5aIWKkRPHhj95zb+kWId7a7+vLfxeLxCvu9LbjKhNXt48Xr3xS5mvXLY4drxFCKOLRm7uL+57N9x1XJW7s1k68+6395H2+IlG8td8pXvvz9eL2f6wRxz5ZIrr9do4QwiZ+f3k78ff39wghFOGp3SN6tb9UfFHrHc/a2qbrvvbn68TtvvvY8PSdol2/e0W9UIRoOCh6tWknPj7a1N5bf7tF3Pjw0pA+fPHGI6LjtdNEo1CEaCgVV7XTiEffLAw55s/XdxBPbyhtOW/fnCV63PFEq/f2x+s7iOc+LG913+sP3yhu+dvKC/ZeKoqnCBgNDAcGAblANyBVkqQEHxZFyrKsUavVqoiICDk6Oloym81ScnI70tPTycrKok+fPlx77bWoZVnljyhIsiwLRVEkXxDBb456nE7ne1qtNpR0TXHxr7c34E6/gnsyTr88advO/cRf2pWuOi5q2bXrS3r2vQajDKBlwOA8Htu5Gyb2JbN3fw7M/TPj5RL65vVl5Mih+ImkBgzsy5y7buTAyOvoO2goQ/NOzslm6X8X5RV3/Wx8jmPGjg2YXjqdjh3bClD0sGjuX7zTqqqIPbt0KMCunXsYOMFLGmDKGkBP86kNXveBbXwm92ZtXlKgzViTEahky9YDxJo/YMaMD706jdtB4e4Shqd3PckV9RgMajAaMBliMRjtOOw2sO5jy247csHrzNjqNRWd7lL2HLSRna1jx6onWbjiQyqtDmoqikkddXPgij1ze6MH0CWRmlBDVQ3gs8qrq6swJYeygRTu/pK8AXd6tTpdMgPz0s+7T9MYa6Kq7MI5NGtqarbQstIgxBSVJEkJrBsaEg0N8q2p1Wg0Gq8p2swcFVJTtq4CeMrLy9enpqY+FBTyZMV7G/hY7sgr12Vw2rXHSiMrsSOgpAAAIABJREFUdxYzcsRFluJxhsXZCb3HsG93LgX5Baxe+jeeXLGdfWvnIgMTnnyX/ru2UbBlCzNG9WP3wo94ZMSJC0Qqd6xg8L1/b/H7uNmvMW1E1oUPphhbkg9Y0noxYoR/7YYbGTU59pSPzOvV8zrtFEVpetFPeqKeAUNvoneC968RI24kJT3l9AZP9jFEeOETUFC0ZgaPvJEU2X+9m0nrrKd8y0ImLPiC/I3v0TXBwPon72R+RRMUqeUT+0WNJhNWq7WVPsgXdKRsVismk+GCtbd168dfNQM110nMUBGatxbkW/MBm+xPaGsCtwBVeCA6mpbWaZ+iKJt8sQve//BD/l1n5sWbexF5Bs/76A/f8ImnLcPTDBcVrlli21JcXBLylc3Ovozd2/6DVQFwsmXjdnJyegYmlSEhlRG3TmDxMw9TsW8/Dp8D2Wq1k57dn4nTZvHQ+L7s2XfgpG2bs4ezft17rGu23TW46yn7rdXpUKqrfH08fVn2xBTmLs0/rWOz+w6k6sBXJGX2Jjc3l9zcXNJ8QYXs3J4UbPGmktQUbmFXVVNHYs1GSku8kcEd23YEnq06vS+9lc9YGYgaKtTUWIEEBvRP5mCFM9BOdkbnVoH29FSaTAZm2Cm1xgaul5mehtEgU1FaSkLXXnRNMIBiY+26gtO+bE52L/btCaWKz86+jO1btnlXg3OUkb/94Pn3d+3ZQ052zgV5P9xu9+YRI0YcCwI2Fy0Tc4Ny16SQZNxg35pGo0ar1aJWq9V+UPNFR2UhSQrNzVG7vWGtwRA1mIYjPF7wHVZdBAPn7ve+APFpfHLv1QDMXvwq71c4qXO6uXHOS2T2uJzXbvRqBm9/vp9+vXphki8qXCNz+ATyXr6JtK5vkTX0PtY+NZmUQROZsvx9cnKuIkVnpcrYn43jvGbXZytmM/GZj8nsmkJx4W4mPfKS13TBzUPDe7BLSSfJ6KSwxM3y9YNPrm9o9SQln93LK6cPZGrvZ8nJugyjIZVlBe/Q9TQutWPTW+xO78wj407NMG/MGsHiSR8xsGd3umamU1VykK6j/sarM0dx1yN/Z+Xg3zFg64s43YLkIFN03MS7GTq+H5uyOmMxRyDjW/tBncSi5f9gzLirWNY1E6WqjEFTn2fmmN48NP8Vxo68nZzlyVi0Dg6WCZZt3UrOWVHg6Zn36kuMHnMDa5/qis5ZSamtHfk7V5E1dDzmJ64nb/Cn6OxVWBLSTvuq6YNGYph2Gwcdj+Ff+ydr1BSGLb2GnLxriZVdKOYmmp+XZ9/By/nFFO+rQFdyE6se0zNj4XsMyzKe/YS1H2DjLiPLl6VfkPfj0KGyT1oxQV0nMEOFJMlB0dAmTc2vrWm1WqRu3brR0NBAQ0MDjY0OqbHRicvlkhRFkRVFkYUQakD7f//3mPHhhx/eBrT/JYNMZWUlTqeL5OQzqxTbs2cvnTtfgvYc14rWVJVjdWtJsYS+XU67lfLKakwJSZj02hbnVNtlUpItqH8lHwm3w0Z5RRWxliQMOm3wDsrKqkhISUbb7F5tVeXYMGIxt2IBKE7KS8vRmS3EGkIdutUVpTjQY7GYz4GBp1BRVgq6WCzBaTmKk9KScoyWZEz6M0vJWT7rJnanPcwT43uHtFNZXobOlHRatO4/RbYvmcaTpTmsnnP+XUZCiLKbbhr553fffddOE0WRDbBJkmTHy8PmlCTJLcuyIsuyotFoRESElogInYiMjCQyMpKoqCgMBgNGo5HY2DZIPXr0wG6309DQgMPhkBobG3G5XFIzjjYNoGtsdDyi0WimhoEtLGE5f6JYi1m6qYTxowb+T9ovWPUq6UPGkWI4/19Nq9X6UkyM6QOggVDutXpJkhoAhyRJLkmSvNxrarXwAlsEOp0X2PR6PVFRUURHRxMTE0NcXJzXFPWboyqVSqhUKsnj8QhZlvB4RHCJlfvQoUNr0tLSpoanXlh+PVLD8oXLqG7Fj2jJ7MuoARd+zVbZmMb4UWn/sycycNT4C9bW5s1bdjXzrZ3QvybLkvCbnz6sCjJB1Wi1GiIiIoiM1PmBTUWQr80XHZXxxkYD0VElPf2S/W63a7UsyyPDL0RYfh1i4tZJk8OP4X8gjY2N60aMGFFFqF/NBbgkSQqweRCoDQ1l8QjGLr9vTaeLIDJSj6zRaAIHBKd+BBfG+wIJHryVCG+HhyQsYQnLT5WioqJPWtHWgoMGntYoipqneGg0GjQaLVptBDqdDr0+AGyaQLjUr+KpVLJozvYBeMzm+AIhxCfhYQlLWMJytuLxeHb07HnZ9yfR1lqweahUTWaoF6vUqNWaFtqawRCFWqvVoNF4bVSXy4uAbrcbb0WCgqIoAXATQngAd319/b/dinJlrUumXRtD65E5jwBVsyJ3j4vy4zYiogzERYYptn/OsrtgNTsOVIIuiXHjh6MPP5KwnEM5dKjsPyfR1vyMuUposbsqxAT145bft6bTeQMJ0dHRqLVaLRqNFrXa2UoQwd1ioRfAk9Il8+1LH5w7MTHCk1nuiuTpcdfTJyESL23ROt79ropKxcSWv92CvyT3yKFvGb9sG3GWeI5XHqVzrzz+OagLcniMW5XyHTBnCyya8b9pX6vTY5ArmfrQG4wIA1tYzqEoirKvX7++e1rX1pqW1sO/YIskCVmWWgka+E1Qv7YWicEQhdEYgxwREYFXa/OapE3gJvu0NjmQrOsHtuPlh9z5D9z05uoHb+ehjAjm5H/t67KXtui9e66h+Rrjb2z9lO55V7Ps98N4d9K1fLp5O181XCQD6fvXWgVlFaH77DVQXErISueKArZy2LoL3G7OaPVvBe9UKSkGuzO0A7bqlu2fSDLyhjB29FBOFPF3O6wUF5dgd7bsXE1lGSWl5bTYpTgpKymmvMoafrsvYqmsrNxYVnY4GNScBNYLDSmhUkK1teCggd8EbdLWoqK82lqbNibkiAgdWm2EzwHXlMGrUqmFvyIhKIjgawz38uXLVwLfpcTG4GgMpS2Ki2yZQGjQalH5l7OXALUWg+riGMj+Fpj2EOQNguGDYIGvOG3xDOjZH6ZMgKzeUFTt/f2hMTBuNpRsgSFDYPAY8O0i1wy7fRyEe1ZAz7FN7eQvgAFjYHAujBkL2SO8v88YCUNGwsgxMCgbJi/8iWbq2mfo3LkXU6bcQ9fOPVi5o9S3x8mTE6+h79A7mDr59/TMHo5/j71sB30zLmX85D8xfuS1jJm5NPyGX4QihCieNevRL09hhrYoePeVTYng9A4vsEUQEaHzJekaiImJITY2FnVkpI6GBq8653RqfL42dfMIabBJ6gE8d9890Xrb6BGvv/zpd7NHXzv8lDd0x3XX8MDrmxhfvJfqYzXcOfpaLrlYcl3dEtWxgj0+ZmaHA4rXw2NboXAXmNSwdi489BSsfQyeXAkH18CIFZC/4sya2rkNdu6GDLO3ncCHJRNWzYGaHZA6EZ6ZBGeVv65UMm3S/+Oxtd9wa7aZfatmM2DaY4zYvhhtzW4WrLKypepzUmVQnM6AtrpjzVK0A/9E/sJ7vRDodIbf8otQjh49+t6LL77Y2Jq2JkmS3wz10ET/LUIjocGgpiUiQktkpC6grZlMJszmeOTISD06nY4mX5uGZkm7wekfIakfk2b+37/d6VcUnw5t0ZbPvqBSZ+Geq7K55/J2vJG/kwA/5a9dZMGtQZqVTgfbtoAemDcTZsyAjbthT+FPb6rnUC+o+dvxS56vOseUArpqb3r32Tn/Cim0ZzAg29tIxsBBqAs/p1QBjOn0T69kzLDbmfvMInaX1QTAM7N3fw6seZzxk2ewZPk6bIo6/JZffNra90888cQXzUAtWGNrkeLRHNRa+tZ0vkioIVB1YLEkIkdF6dHpIvH62rT4o6ShqR8qEQRqCt4IqfuV+c8c//uAjktPHQBoZOWn33LbtVfSJ9XC9f360s1VwqaS+osE2MBoaOkLS+kJI0Z4t3HTYNUTZ+azU9wt9xlPUPscYMo5n9EaOZal2/cyf/otyBW7GNz7craWeTUzLxXTR4zI68zWpX8jb/RMlPC7frFpa2v/+c+nHM1MUL+21uoSe8EYFFxh4NXWWvrWzGYzarUa2WAwoNdHhmhtTUm76ta42nzgJnkkSXKvWbniTSHE/pPfkgpDhIofqmq9L6SjlkP1EgbdxfvV7j8ADhRC1xzIzfVuKUF18HojWCtDgwoACUYo9THybPvsPHfSYMbsLqekOqgXSZlk6fexZVcVAPsK8nFnXe7lJHPasLr19B44jBnzFjIi3UFRiVc3PDEVU1guEm3tm9mzZ+9sBdScQf61VlI8ZEJLp0K1Nb0+spm25mU+UUdHG6mrq6O+3o7D0UBjYyNOpxaNxoXL5UKtVgu32y15PB6heMNzQggUEIoQeCZMuLP2hTrLazumXDcPTkxbNPn6K7l71Xt8sTOemqoqkrL6MDRo7YSLTdKGwsytkJMFmeneKObQh2DuOB9+5Hr5kbOywJAEGzdCLDBhItw1CBZ2BbMBOJ/UdupUZs8ayq09u6A3pbFi+4dkGiw8teCvjBrZh+WZaezZV86C5e972V1rCumfezvmjCzkmhJqEsYwL9fL/XtiKqawXAzy448/rn3++UWn8q2FJOQG0RKJlqAWEWD1iI420qZNG+Lj49H5/C/Sa6+9SmlpKRUVRzh2rIqamlrq6qzYbPWny/qhve66Ifp169atkiSp90nvzuOivLaeiMio/1mC7s+N3cNhg4pqsFhAd5qXrq4ERQdm4/9uorodNsoqqkhITkEfnKGtOKmoKMetNpGcEEpxfTIqprD8ekVRlJ2Zmd2f2r9/vwMvi0e9b7MBdh+Lh/N0WDwMBi+QmUwxxMWZsVgSSUlJ4ZJLLqFHjx7o9d7PpbpNmzbU1NRQV2fDbq+nocFBY2MEWq0Tl8sbIVWrVcLt9rN+yHg8nkCyLuDZsGGj02q1vhoTE3NyYFNpSIo1hUc6SHQGSD1DrSs24X/fb7XOQGprHZe1WJJSWz1HqzeSmmoMD/pFJsXFxe/v37/fzakjoQpNq7v7tLXmvrUIH4OHV1szGr3aWkJCQgDUAGSz2YzJZCI62oBeH0VkpI6IiAifr611f1tQ1CKw+ILJ1OY9RVHWh4cxLGEJi19cLteHl1zSeX8zQDulb601v5pGow3QEun1UURHG3zpHeaAby0AbAkJicTGxmI0xmAwROFN/4igqSJB7UsBUYkgCvGQEiu8iy64KisrXw0PZVjCEha/fPnll5sITcYNAJtPW3OfXFvTNKsHbSp0Nxq9ybgJCYlERUWFAltSUlvMZjNt2piIjo4mNP0jIkhr0zSPkrbQ2tq2TdrucrmWhIczLGEJS319/fIrrsg91AzUGk+krfmX1GteNuWnJDpRekdSUtuWHpHIyEgsFgtxcXHExMTgTf/wJu1GRGh9KSABczS4zEo0qyF1S5Lk2rt371Lg8K9tkDIzu4VpwcMSltMUIcThNWvWbG1NU2umrYXkrbUWCfVXGPi51pqnd0RGRrYENoCkpCTi4+Np06YNRqORqKgo9PpIIiL84KY5bV/bZZdlf2e3218ID21YwnLxSmVl5cqxY28/3kxTO6G2diLfmtcE1RIR4c1ZCw4YxMfHk5SU1Gr73nXItVosFq9J6g0kRKPX6wOBBK02IlhzE4oQON0e4fYoQhFNrB9+dO7aI3vps/mFm0M4HNwO3vpoB9NXbOLx/N0caRTh0Q9LWH6F4na7P87Jyfn8NLS1kKX1/NUFvlQPXyVURFDAILge1IzF0vaEVlQgAal9+2T8gYSYGCMGgwF/HWmw1qbRyDjdoNZoUKlUIATC1zk/uB06eMC5YMN/lwQD27L31/N6iZMxfbKIsx5iwqrPwiU1YQnLr1AKCwvfLys73Ly6oBFoDAY1H6ApwTWhTVHQJm0ttB7UGAgYtG+ffMI+hFQOJiW1JT4+HpOpDdHRXpM0MtJrkjZpbTrMbaKFITKCCK1GqFSy8GJbwFb2SBKug0/d97HL6fyX98qN5O89wl3X5pHTwcLEG/rg+qaIrxvCkyAsYfk1idVqfbVXr5wSH3gFg1pwwXtzvjWftqZqRVvThVQYmEx+E7TtSfsRAmyJiYlYLBbM5jhMphifSeoFN52uyRz1RUmFWq1GBNGLNGltkhtwbisoeEUI8SV4cCsClcrXnKxCJWwUV7vCMyEsYfmViKIoXz3//KLmAYOAb02SJCet+9Z8JqhGNGlrTWVT3py1aEymGMxmb8AgMTHx9IENoF27diQkJPhy24wYDFGBInk/A4g/kAAKipBQa9QBIkpJCjBgusddN7Ti+PHjL4CePp1iWbW9kKM2Ox98/CU/CImGRnd4NoQlLL8S+fbbb1fPmDHD1gqo+TU2P6gFp3i0WHGqibnDX+Qe5VvhPZaEhATatTt1OWQLYIuJiSEpKQmzOd6X29ZkkgaDmyxBncMjdHo9GpWq1aRdJMkVF2fe4HK5Xrz7xiF05wiTXt/Ap0oiA2JV6HVhTq6whOXXIDab7Y2uXS/9JsgEbWymrbWajNsUMNCI5qAWWuTuJZBMSkoiJibmlP1pFVnat29PdXU1VquV+vp6XyF8I06nE6fTiaOhniO2RowmE2qPk8ZGBUVREEIIRVEUEB4hkCSQkSTVJ5988vJVV13Va9pNv8kGcNeVc/W2aP4UH16pKixh+RWYoLtfffW1za341YJBzYU319UjSZLSPBm3ya8WWjZlMESHFLy3b9/+tPrUKrBpNBqSk5Opra3FZqvDbrfjcDi8lEaOBr4+VocqMgpJuGhwK8ItJEmtUgkhFDwe8NIaSYoANwLX1VcPLNvz5a5FX3qiFidpFd75+HOycnvTMYxrYQnLL16++eab1Q888ID1JCZoi2Rc36pTLdI7vH61llHQxMQEkpOT0WhODzROyKdqsViaJe5G+2pJI4mPi8OkjwCVCkmlQpK9qCvLKiFJCCRJSJKvYl/CJUmSM/Oy7E2lJSXPri86TM/efXl6cOfwjAhLWH7hUltbuyQjo9uBk2hrfmDz+9Z8AYMAqIUw4nqjoP5a0OiQRNzmhe5nrLH5JSUlJYjSyO6jNGokPVnxcbbZqK+309DQIBobGyWPx4Narcbj8QifBBZZliTJmVhT+tKM3/++myzLg8JTIixh+WWL2+3e8uijsz9uBmoOmnLWThQFDTZBhZe1I5g8Uu8zQU0+E9RCSkrKGfXtpAz4UVFRJCcnk5iY4EvcjcFgCC6Ubwom+Gu7VL5Agn+pPoIqEu666+7jhw4degEoD0+LsITlFy0/fvrpp+89++yzDa2YoI5mJmigwiA0YKAWwZqav8DdYIgOLKPnN0Gbs3f8JI0NvIGE2tragNYW8LU5vdThbrcbt9uNx+NBUZqCCEIIfOAmCSHcgCRJkpya2vFzq7V2QXR09Nzw3AhLWH6Zcvjw4WV9+/Yr84FXMKA1N0HdQRUGomlRltBggb/APSrK4EvtaEN8vDe143QDBmcEbAAdOnTwRUhtNDQ00NjowOl04nI5g4DNjcfjER6PRwoGOJ/WJvlMUpckSSqjMebfjY2ONK1We1d4ioQlLL8ssdlsbyQnt99FaGpHcxPURWjZVAsTtIljLbjAPZSOqEOHDmfVx9NajC06Opr27dsH0RsFM+76qxL8xJSaU5mkTqDxww/zlyiKUhCeJmEJyy9H3G73R4sWLSqg9QhoQGM7mQnqBzWtNqJZdYGBmBhTgI6offv2REdHn1U/TztDtn379lit3kVeQrU2v0nqwePx4PEoJzJJEUJIPtXUOWzYsPLvvy9+vkOH1A6SRKfwlAlLWH7eIoQo3rFjx9o//emhulZMUActAwaeE5ugXk1Np4v0sXb4Oda8Be7JyclnZYKekcbml44dO9KunTcFJFKvB00k+qioIGLKCLQaDZIsC1nlX0k+hG03EEiQJMnZsWPazpqa48/RcvnMi0aspXtYv23PubnO9n0XpM/b1q+gzPpzf66wftuFas1OwfotOH8hc6W5fLZpNQerT/kKuou//u/yfv36lzXT1lozQf2JuL7qAq8JqtH4tbVQv5q/ZKpNm1ji4+Np1y6Jjh07/qR7OiNg0+v1dGhv5q1PCnn96zI+KSln03eVNKi9NrIkPJRU13HM7sRqdwi7S0FWqYUsS0KSUBThXYtUCOEWApckSY2xsXHvNTQ0PPVLAaInJo9kW9m5I1yqLMznmZVbfvJ1yj9bxSML116Qe1rw0O/Z8TOPa1cWwjMrW3lOO+Deeee2rd0r5/FkwUG0v5C50lxennM/24ptJz3m2LHqF9N79N0TpKk5gjdJkhqDQa3JBJWESqUWXlDTihPVgXrz1cy0bZtEhw4dQlacOu/ABtC2XXsm/m4M838/ggduGES/9kb21ziJiorCEB3DpR3akZ6UQLuENkhCwYWMLKuEIoQASciyv1Be+BkAGnNzc5e4XK7XfgH+BQq3fUhZjTdoEireCaw4bBQXl+IMms/2miqKi0uwu0MnuaIopA6ZzMb5k1u9lsNWRUlZ1Zn10W6lpLSiBdedw1ZDSXExNof7DO7Ju35ocUkpdmfLF7SmsoyK6pYvhK2qvNV+K047JcXFVFnt53WcFAVSh8DG+S1/t5XD1l3gdnv/DtnvgJISsLeieil47YqS4ub77Tz5xBtMmjT2lzVXFCdlJSXYWhnX5uNkt9v/3aNHj+B8tRYmKC1TO3xrGKhEMMda82CBnziyKV+tPW3btv3Jc+AsqtAjuK5vT776SsZubyDFksCntdUYDBG4XG4kWcaGgsfjQa1RC6FSS2oZHI1OZFkSEiKQAiK8KSBSYeHXckFBwYuDBg1KUKlU1/0cQa1w3TNMe+ZdCg/a2X3XdSwxqBk29Z9MHZYFwKvTb2R1TUcqP9sMeh3qzJvZtmQGT9w7iGW7PaQl6Nizr5zZr77N2L7pgINZ44aTX/gNtqz72LNsRqCtRVN/w4oDcajdx6k6WEjK6MdZO2/8KftYXbiGAYM+QEc1Ffo8CtY/T4IWVs4Zy6zV39M51czBwiLGP/EGD43KOeU9rVvwIFOe2kRGVmfKDxQzd+0XDEn36iXLZo9nsfUI+3buYurL/2H6sExwVzNr/M2sLFJI0dupNvRm3ep/YdFBxe41DB71J1Iys7BVHiR7/NM8NXHgeRmrWeMgvxBsWbBnWdPvD42BbUVQUgFDhgCxsHIlxAI7VsL4GZCeCQeKYO5yGJXjPS9/AczZBrqDYNVCjQmKfAtNOorWU2DPYUma7hczV5Sag4wc/BsqDF2R7Q04Kp3k+vY1H6eet/8jP831zYeHDx9uLVetNRNUCa4F9a8JqtFoRdMqU5GBOtCYGBOxsXEkJiaSnNyeTp3OkbtdCOWstuPHq/l48wf0v2U8193/Z+677z5Gj/4t1157LVdccQWdL0lDNsQRb2lLbGwbSVKpJJVWK0uyrJJkWQNSBJIUJUmSSZKkRKDDvn17hyuK51NvwcL52Y4cqRCHDh066/Nv6xEl3trvbPH7K3+8XsT3GiV+rHcLIRTR0GgXQiiitrYmcMz3G54QHa6dGnLe/rdmiW63zQ357fkHrxGX3zZXeIQiXIcLRLu4K8QPnpP3a/9bs4SmXT9xqEERQjjEH3/TUfzp9c9a9KFu/yoR3+VmUX+Ke6r9apWIa3eF2H/M4f2tsUYcr/Pe2y3dVOKPL24XQihi779nik43zBBCKGLzc5NEj5sfEY1CEUK4xdN3Xyn+8OI23/O5Ttzxj7WB6zc2Ok7+rBv3iT5tE0ViYuh29f3zT2uc9r+liG63tfz923cU0e2W0N88RxXRKVERnxz2/v3jx4po308RLt/+D59TRGQ7Rew96v27oaHp3C9emyG6/+6xX9RceffxsaLffd7n2Hhkh+gUqRKvfF7dYpzcbtfO995b+wDwW2A4MBjoC2QDXYEOkiQl+t7hKFmWI1QqlUatVqsiIiJkg8EgxcbGSm3btiUtrSOZmd244ooruPbaaxk9+rfcd999/O1vf+OVV17m44+3cvx49VnjUfPtrHmDTMYotuwtJrJbH4bFeygvK/NFSN00NtRzyOrCkhiHxt2IQ3iEJMmSJKuE1/yWFCF5kLxqqwTIkiRJGRndvi4rO7QwKSkpVpKk9F9aIGDwyFux6L3WvU7r/YJXFG5i8jOvcKCsGsVRRaWcd1rX6p2XiwzIScmYndVYnYDu5Oek5V5Nsg5AS/+BeSzcuRvG5mAr3cmMef9i98EKFLeNmmojVuBkXoxdOwpIHXQzXWN9niOtEZO2SdHvnedVZ5JSUrBVfQ7A1q2b0dpzmDXjYQBKS+xU6AqBPLL79mPG5D9xb8Vn5PXtz6hhgzjpml/armwv//GCjFvJdihXw5r5sNZrpWHd4y2P8Rfy9BwKGWbv/3VB41BZfRSjMfkXNVd27fyc/iOnex9zQg59M4yBfU3j9Ol3SW2T3np0+v1HmgUKgiOgJ/CryaJpBXd/sKApAmowGHxrgrYhIcFbB5qamorJZDpnY3qWwOZhxXsb2BOfwxPXGPj+myKcPlojh72OT47W0S45mWjFQX291z8gq1RIspoILcLjceNUEEiSgi9x1w9wycntP66qOvpcXFzcTCD+lwRsOqOhmbOpkFGj/spj6z9gWHYK9l1LsUz48PQGRg4aGvnMHdCK34OqVDFh+G0MW/Qh8wdnoq7cgiXzrz85DC0H98/nrFKArNxrGTHY900acSPGhFQAskbMYF/OcPI3FbB03n0s3fJH8p+598QNOA8yIu+3lDTraNdhD7DisQnnfOyMyTByRNPfI8dAQvB+Y+vnmY3R2OzWX9ZckeXW3HTB41S17PV/v/Hg9Pu/J7Sw3dGKXy1A9R202LHwrwnaMgJq8AULvPxqbdt6k3CTk5PP6XieBbAJ3v/wQ/5dZ2bp73phkAWehgYcDgcOWw3rdhwjOS2dRBxYrbV4zFNiAAAgAElEQVSBUquoyAhR5xSSTq1G8bgBSZElCQF4CxS8/jYhhGQ2x6+vqTluiImJ+eup9ZQLKwajnqrKauiacOqDq8soVafQt2cKMrB+zbu40Z+3vhXv2EyZYzbJOifbCraTM/YP4K6ktNpI/7xM1MCO9eupVE59T9m5AymZ8w+Kqqd4tTa3lRqHAZPhxPGmAf2v5qFtxWTPGuvVxhQ7VTXexqxWK7HJGYyekEFukpu8J4tOfjPaVBavfY/mvm2t4ad91fVGsFZ6vdz+yZ+aB4ZSkFOht49Aoqrq9CZe1545lD+zASe00EB/rnOld04v5m3bCrdm4azcybYiKwN8+6xWq6NBinnlwUf+3z6a0jocQEMIqLVM7fBra8H8aqK1ygL/KlNt21pISelA587nnunnzIGt4QiPF3yHVRfBwLn7AVBFW3iqdwqFhV9ySNGiPnKMb+x1uBvstNFHYtbpiDXFUH/kmKhv9EjCI4RKrZZk/MSUIfWkkhBCNpnavGWz1UVFRUU98nMCtgmT/sD4CVex2GRgzCMv8tDInic+OGUAU/IepW/uNaQZFMwp8YEHXrFjBaOm/wt7VQklVTJ9+75P5rAHWTRj1Fn3zWL2MHbQ1aipptI4lILR2aBVmD6hM8Nz+9AzWY+c1B6LfOp7MmaN4NUZHzE0pwcZWZ2pLC3lsZWfMjj9xAbkgIlzGPrZb8nseSUZKSYOHihm8oL3uXeQgbVPjOex9UfJSI1lX2ERjyx8+5RTMyHpzL/iFTtg1HSwV0FJFfTtC5nDYJHP356UC4OArCwwJMHGjRBrhuXzYVyeN3hgLQdtDuQvOo0PXc+h5LkfZWuZk0HJ2l/EXBly7yxeHnQ9uQM/QO10YbQ0qaPLHv3di/c/s2FXM1BzBIGbtxa0lWCBf6FjXwRUtBYBjYkxBdYtaN8+hfT0TkiSdM7fU0mIc5Nnc+TIEfbu3ct3333HoUNlHDlSwbFjxzh+vIa6ujrq671F9PX1DThdTsntLaCXPB6PJISQfZsa0ABaIYQOiLTb6++JjIz8w7m64crKSpxOF8nJ7S4YGFaVleDWm7HEGs57W267lfIaB8lJCSG5PDWVZdgwkJxwZhqP22GjvLIasyUZvfb0soPsNZVUWd1YkpLQBn06HbYqyqtsWJJSTvtaF1IUN5SVgdEMpjMYqm1LpjK/NI+Vc0b/guaKm/LSckxB43r8+PEXYmPjtjQzPRsAu+9fr9YmSU7Jm2QfADZfZYE/T000AZqB6Ghv/WdcXByJiRbat0+mU6dOdOvW7ZSLspytqGbPfvTcmGgGA5Ik0djoLY53ufyF8QqK4vGXWPlDsT5QFU2/Bdu6Xs0NgK+++mrPyJEjUavVvc9FP+vr6/F4FIwncpqcB9EbTRgitRekLVkTQUx0FM2/gbooI8aoM7fqZbWWmJgYNKrT/6pqdFHExESjaoZdaq2eNibTGV3rQookQ4wJdGc4VClZvWk8XEbXHl1R/WLmikx00LjW1ta+PGjQtZt//PFHVyvA5k3C9fnVJElyNTM/A6DmNz/9Cxz7AwVt2rTxrVnQNmB+nmu/2jkIHrQuqampOBwOnE4vrZHb7QpQGnk8ig/kvGDWBGpCeJMlFcVXV4oQwuV3uq1b9770+OPzljzyyMPqiIiIe7hopYblC5dR3YqCbcnsy6gBPQnL/0jUsYwdN+IXO1fq6upee+CBBzZ/8cUXzfPUmkDNGwH1a2oefMwd/gho8wRcbwTUn6sWusJUWlpHUlNTz++QnOsLdunSxcfZ5gykf/gL5P2am28TQgif1kZAlVMU4WcCkfw+tzlz5tTEx5uX3H333XJERMTdF+fbY+LWSZPDIBKWczpX6urqXv/rX2cVvP76soYgTa0hePODGkGVBbIstRYBDfjVvISRhqAa0IRAWkeXLl3O/7fmnKvzkkSXLl1obGz08bW5/FxtAc2tKZFOBIGbQAih+IhAQkxUSZJ44IEpx7Ra7ZI77riDixfcwhKWcyd1dXWvP/bYY/nPPvtsPaGBgoA/zQdqIflqXhNUbhEB9S/E0pww0pur1pbU1A506dLlvAQLzjuwAURGRtK5c+cgWiOv1tbka/OapD6z1A9uwqel+SOlweAmSRLSPffce7SxsXHx3Xffreh0unvCUzMsYTlrUHvtr3+dVXASUGuQpNBVpoKXzguOgDZpak0JuDExxoBfzZ+r1rlzZyIjIy/I/Z2z4EFr4KbT6QLg5vH4wS3A1ebX2vybD8bFiXRBAbBhw0aXJEl7+vTJdWs0mjMOKPwvggdhCcvPSWpra1+ePHlyweLFL9pPpKnRSmVBE6jJwr/AcWgENKrFegV+n9qll16K2Wy+YPd4FjF3D58W7mXO6nz+uLKAV778IcBD1Vhfw/KPPmXGvzfxl3e3U6xEcckll9ChQwoJ5jgaFInD9Y3UCK1vdXl98OryQqPRoFKphT+D2R91oYnDrRFwPPbYY7U33zzqBbvd/vSFnhQH9+yk2nGuruZk367d2E4746ackTm/4cBZEH9ZK4opOgP2h22vPsK9T6752b+kI3OgqJXnsW8XZ/BcT0fczLtrNFtK7D/puZ5KKkuLKKk8S7I79wGGZN9M2UkOOX78+AvXXDMoP8in1kJToyUNUXBhu4+GKNSnFhmp9yXgeqsK4uPNWCxt6dAhhUsuueS8pXWcO2BT6lizq5wuqR24IaMdm7fkM2trCQA/FJfwdb3M1d07kx0reOCFVRw2tCU9PZ1onYb9DRL1HsERRYPRaMRgaEZS6QM3tdoPbl5yymYElY2AY8OGDfVRUYaXbDbb476BuSAye+zVFJScK17MSiYOvYki++kDYWlJMc6zeGE/W/U40xesO+3jbTVHqaiy/eyBbfxUSGhlFk8cyhk819MYqR1LWVGRRv9U/U96rqeStfP/wLzVhWf9oSwpLj1RuZzj6NGjz8XGxm0Jin6eDqh5JEny+IMFLc3PULJIf1WBn4IoPT39JzHhXjgfm2zi8fHXBv5sYz/Kfbt/gP6pdO7ek8e7+/eksmvvfrZ+X81D2Z1wuQYSk9CO9zeup6Dcgckk+czSQNoHPr9awNfmduMrJcUT7HPzpYQAEB1tXHb8eLXdZDJN4TzWlpbuXMfiNZ9SWOHE/syj7DbL5I64m2E53jLpfVtXs2T5+1jlNoy5dyqDsrw5Ooq9gkXPzGdH0WFikzsy7t5pZKcY2Z2/jFVbvqDEVsOiOX/FopUZOu4P5HU+dQLt7vylzF+1mYSMq5g+bTwmNbitZSxe9DK7i75H0ccxesIUBmengLOcp+Y+z84duzhgrWTmzO/QJfdi5r0jAi9D/srFrNr4BRjaMmrCJAb19PXdWc2SuY+w7eAxhk34I6P6+kpf3FZWLJzPpp3fktD5cqZOmxQo6N63bTWLlr6PDS0ZA25i2q2DOR+puPu2wvJN3hncO+TZwKotUGKDRXPAooWh4yDP1/Xt62DZGnAbYOI0yPFXuTvgmQUwZDAsXgB2A0yfBem+4Xh18QsMH/tC072c4rnaqw6wYP4i9pTUkjnot0wdNyRQcrVj3WJeXf1fnFojecNv466hvVGq9jHn/7d37uFRlHf7/8zsIZvNZrMsIQkhiTFgiCHEiBgRIyIiIlJERKSIiJQiWqrWorVKKS8vPzzUIqUUrVI8IZ4QEREpIqUUERARMUaMMcQQYgghhGWz2ezOzvP7Y/YwuzmAeKK+Ptc1155nZmeeuef+nu7vwhfYtbWCurK/M6vmn6QVXsaMcYNPwNJcLF/4CBtLv2LI2IujkayhgoULF1NWfawhPjn9qScefWA30bWfLTHmZ7ugpmtwHAVqkUBBAomJ0aCmVRX0+vZkiL5zYIsZn9Y3kJt+VtsP/MepOKIyKTUJgLy8PBRFoVtKN+JbWkhORudrCy1h8AqDG0Ag0Cm4iS5dnKsOHapr7tat223flSqIPSOP4cNtbH/xEYpKLmNItpm0DM1PV75hESPufI0Fi/6XZF8lM8ZcxdJt71OcYmbpfTeyzncF82aOo75yDw0Nbsiyk96rH8MtFtYseZHiIVeQb4Os5JNIoFWqWPZyOQ/e/ktWzL+DyXMsrJ43Hl99JY2WbCbfNgpf/W5uG30ZS3d+ysAUO4OGXoHs2k1lbR+GD78C2RapXXxx9g3M22ZnwZxbsXpqKN1XHga2LS8/x8glDzEhexsTxt5McfW7ZJkVHpxyBZvNVzD7zjsoXfkoIyY3suvl2VC3lRHj/4cFz/6dbLuPbXs0wcvOgG3bqsfZsPdQzM2zK7fdN4OUTmZnSjYMHwpTR8PYeyEt+N30XjDcAmuWQPEQgsc1aF4vg6lPwMJHgFoYPww27YEsSxDY5sG67TB1MtAIjS7AAdDA+vWV3DevUHfldHJcPdWMHXwF+VMf4vZRaSyb+xtmeiwsmj6Yxt0rGD3zFZYvewiH2sDuas2Mla0pDB1+Bd7yDchZFzB8eBHW5BM3CV457xcs3JvBovt+ybpF91Kho2tzpv6M+nNnfXH99bbnR468JlT7GZunFlpaOwK1UFVBZ+anBmpdSUtLJTMzk169epKXl/eDMflvBGzV+z/hsU8CPDnj7Fh7lWfWbCTu7AFc2SMu/G6fPn1IT08nobmZ1FQ6CiIEvy0EhKOlHYIbaICYmpq2fv/+/a6srMxbZFk+/9s+UI60XpSk9SLNZiS//yBK8iKHbunCvzBx1quMKMkD+jN16FOsWLuL4ikDaWxowJqVQk5uAUWFRboLM5+UbDt2s4V+JSX0P9kKGtnKzLlzKM42Ujj3t6SMeg7XvPHYew1i5qQ8du7ZjVtNIT/dzc7dtQwckUX/khKa9qSTbM6lpKREB5LVLFy8lQV7vmRYlhkoRvcpBSNvZNqoIUAxA2f9mb21kGXZysJ1Cpsr7yXbDP3u+wNP9PoZpZ7Z5LsbaFKspGRk0S83nX79T/x3LA4naWltbiMYT0DzkrOgJAtsxraAl5INdjP0KyHquC56BO56AgYH92v007B6K9w+NPgFD9y7CIakx2ysqZJKVwbZ+iJb2dbhcS1b+yxlGdewcvooZGDuvbeSf8+LLJo+GG9DI6rFRlpGNgVZ/QgfImsyJSUl7FvtpKlXUfR56thhwPIVG5m56iADC6wUz72HJav/EPGnHWr8oLXV/crIkTfup23tp65EilaJjkHNYDAIk8mI2WzSgVpsUXuoXCozXC71Q45TBrYjX1VxyysfMfem0eQm6GehYMOmjbx6PJnlNxZG3a0lSSI7O5vE4y4yM9CZomqUORpaj8bgoDPmBhIaeRPizDPP3Pbhh7ub+vbtO9VgMAz/vg5idXUtZYt/x65nQ//WwtCh2vPpc//OfbP+l36592PJKGLRsqcYkvcNokNyCulBemJJy8DaVE8j0LT1aYZOfpThY64mxW6mwa3g9Z0gyuCupUZNIyer/RKeZIczPE3MZgVFAW9dDY3eGu4cd03ke0XnoHrB2Gskz87azaxJV1Be42X41Pt5fM7kTnXXbI4U0tJiviHbv5M8pOpaWDoLVumIcT/9hhyQm96uYxlF/jrb+RJX6XbGjP40/F5x3kWoQPrQyTw4poKpIy+k2iUz+d5HmT/9VKdqE3X1FjKSNb+fMS0NZ3A//X7/xosn3r7mxhm3HtKZn3pQCxe1S7RR6ogBNRMmU3sNWEKy3m1B7fvIVfvWge34kYP8YvlWZoy/mkvSop2p7773H/7yuczTN1+MvZ3COYPRgCOrJz3PFKhqIIathYAs5G8Lm6QdgVsIDUXIND333H6frFr16p+uuuqqw2az+cZvPdoiy6gxYvlZWRkUTf8b945qawU7copZsuINQGH5rJ8za8Fytj1xZyR2I6tttPc7D97UU1OrUJRjxFtXg8eZQTLwxLLHGDPrKR6c3A/wUbnmz7E73ma/saWTIddRWe2jV9bJ1Sea0zJw2nJ5eu1bYfNPP51GzZjLqBlzcdXtZnDh5WyeOplhnZQEVu7ZwtptB2JWk0rxiCHYv4lzTm7b0yArHcYshvEFXzOU5sgmx1JHTQP0SuGExzUr/QxSBqSyfuW8drZhY8rshUyZvZD60lUUDfw9U6YOp5dRP79ONjjlID3FS02DB9KsKHV1NKrQ4vG89NjSf7x9xx1/cHUCal6dWaqPfrYBtUhhe7xOqcOu61WQGixsz6FPnz6YTKYfPKj09YHNf5wZ/3gTS0ZvGg/u57mDYIh3MKEoi/3le7j1rc+54dL+rP9Ai+ycdcaZDEhPRPU0seKjaj6pbuKIT2VvNyf2tDPIUKN9bDpQCz0XIavT79dy2bREXwIRbAszQiGEEGPGXHswLy9vwfvv76y32Wx38C0mIufn92L5ogVYRhbQq3Ag+VlOpt75K4bf9UtyzP9DrlNm56b15Iy6h6H5dlY+/iBkDyQ3zcyufVX06n+T3nAiP0fh8QVPUN8/g/wBQ8g5kZ9N9bJgzixSbh/Fivl/ZtjE32EDsrKzeXHt8+wqkilb9wTryhX0Kle5uX3YveQlXhyQjCMth+EDC8CYxZ0zSrhryq0smPMrrN4a9jXamDqu414EcloJdw71MXn6HGZNHwWNNazdXsXc2bfjK9/MovXVDCkpwlO9lQZzDtnOzv/O8CmzGX4KupG7t0KjF1w+2L4JGqwwcHBEFTg/Bx5fAPX9IX8A5CTD7TNh6jSwzdeCClvWw+Dbod8JCXQaw4amsG17OYNHRWuHtXdc80dOImfupdy1II8JQ/KpL99DqTuNe6aMoGrralaV+xjcL5fa3buQs7JJ1s3OvF5n8+CKf7Aqw01yRh6D+nXmMrYxceJw5s++n6z7fs6aRY8oxxXlyUtze//nUNseBXpA80qS1CqEaCf5Vh8oMIaYmoiPD5mfNh1T00At1KugT58+WCynh3zi10/QVf0c9RlITTDh9St4/QqtspniDCfHm1uw2myYRCD8WVIXJzlJFkSrmy37D2NL6sLZTit+IZGXm4vTKFAUBVWNlFupqgibojqyRoi9CdHh3glJkgQgGhoa1AceeKDsttturbJarT0lSXLCN0/QLR5yFUrdPj6tqCK+Rx9y05Po2nMAVxY5WfPqq2x+vwxrZl+GXnweiXEynoZKNqz/Jxs37ySrZDJzfj2aODlCES4ZcSUN5Xv4fH81qb36k9n1BBNDTmXUsDN4/pmVOM/9OQ/dM544GfIvuBRRtYNX3/gXCf3GMX10ET37FJOVrGV6J+Wcx4Asid0ffUqdx8zF52mO3YJLR5NrqeO1V1/n42oXFw+9guxU7dg4M3pTkJMa3tc+xReTHC9TMmoclroPWbVqHZ8eOEa/S4ZyTq8eGCSFD7f9iw3r/0nZVxJ/WPAo5/VI/E4m7tsvw0cV0PdCcB+E6ho4d1AE2C4ZAQ3l8Pl+SO0FmV0h61wY0B1WvQo7PoK0PjD4fLAYIrf5AQMhrp3tZXfxM+fpXUwbe0nU++0eV1MS1467is/fXc/atzZR3xrHkKHDOCPVjoyX7Zvf5q317/BlSzce/ssDnJkUOeeZ55ZwVtwxPiwrx210cn5+dqfH4eyLhmE/9AEvrXu3Mmfw1Y+tX/nY+83Hj3ek0KFP5/AFAS0q+TY2+hlqvhIfr5mfiYn2oASRJhSpBQp6UVBQgM1m43QZ35oe26mOY8eOUVpaSkXFFxw4cIC6ujqOHGmI0XFrCRbWt+Lz+SR/UMtNVVVJVdWQnpsBMKDpuZmEEHFoc9RSVvZJUW5u7hSDwXDpD6HH9tP4MQwPd46+klEL32JItvW02jNFUf69ffv2NR00M+6gT4HwC0FAluWAPvm2/Ty1WF21UEqHFv0sKCggKSnptDomxh96B5KSkujTp4/O9NSbllFETU/fJNB6J2imadjJIUJ+t6BZqgJqfn6f3c8//3ztmDHXVAE3n86XT/muTeytamznEzvDxg7jp0KwH2pYWbj639/7Vn0NFazZvKfdz3KLh5LjND7/+OOPb7r77nuOE53OEWt+6tvk+SVJDkgSqr5Mqr2KglCeWqz5qReLPN1A7bQANgCHw0GfPvk6UJN0j1IY4CRJCi1CkiRJ87kpQUBTUVUh9PltIZ8bIG644YZa4K+ffbavympNmA50Px0vn4y8IuwZ7UQzZSO2n9Dl/9ww2jMYVGKjHbvqK7fHtTwxMTsk461P59CboO31/gwAAa2yR1bbL2jXfGoh8zMU/dSYWgjU8r/VzlI/OmAD6NKlSzD3JQJmGoiFmJsUfh3L3ABJVSUVVCkEanrmBqhB9iZ6985bvWnTpqru3dNuMhgMQ063E2K1ObH+hGA/jdD9zGwhJSbRT1GUzTt27HijpOTiGjrvzq6vJGiv8UqoqXGbioJI8m1HgYJ8unTpctoet+9M3eNURnx8PHa7PazfFgokhLI6OggadJYwE7ZtQwAHiGeeeeaw1Wrd0a9fP8+pKIT8NH4aP9Q4duzY0vvuu//1W2655YjO9DxRzWdUMbtOeigmncOi69IeKz2k96n1Oa1BDU6D4EEHJ49PPinjiy++oKbmAHV1hzhy5AhNTU24XC6am5vxeDztBBT8KEogFFCIDSoY0YIKZoJBBSDuwIHqi3v06DFZkqRzf7psfhqn61BVdc9nn322Kj+/T3mQpcX60/QmaJTpiSbjHWJqItLQ2ETHPjW9+RnN1E5Hn9q3xtiaXC4Ouf3YLGZkPWcKKNQ2HsMjDNjM0Rm6rS0tHGxqJi4uDpPcMdGyWCw4HI5wT9Lo6gTCvriY5GapI/IWSgHRm6ah148++ujBgoKCnTk5OX6j0fi9g5urupR3Pq7nrKyUk/+Rt55Va3bQ8+wzv3HzkNNpVO3dyvp3tlDXYiGnR9fvfwdO8bhuXfsicloB9rjvZrfcbvfzTz659OWRI0cepP16T30qh1fquOenru7THGZqQvFhtjmw2xJITIw0XwlJD4VUOgoKCk6YJrV3yzoCXXuSaJZAbeTFJ59h6/u7qPfZOCvz+9Nj+/o+NtXNr/+ykt1+M6lxAWr98Tw66SouTImn5vOPue7FHTgddjyu43Tv1Ze/X1dMogxPrVrFkrLjZDriOHDUyy1jRjC1T8cXc2JiIn379kWWZWTZEHyMLJKkf4wEFCRJwu/3i2CFghBCoKpqVCAhqPGmCk06JHDddeNqgL/V1BzYm56efpMkSd9qZ5T1S2dRkT6eGSPaprzX793Iwg1GRpQUnPwKPVXMf3Apg8cMwflfBFx71ixkVWMBcycPbfNZ3bYnGDTlOaZP/hlyiuc724fa7S8zd7PC4/dO+JaOq8qKBX9kVMF4Mr7lkLWqqh99/vnnq/Lyzv5Mx9JiO7N3xNIUSZICQqiqEAjZaFINBgOREqlInlrdoa8wdc0iMTExqqA9UialVRQkJCR0fl+o3szkWS+wZcuIkIcQm83G3g1PsbbCwoiBeacxsMkmbhk7moJMLRqyat0bzN34MW9NKMbqTOfV395EutWE6nMzacFzvFSex9Q8OxdecBE3jE7FLEPNvg8YtnIr1/QZQ2f35YSEBAoLC9FOiIzB0B7ISUFgkwmKU6JFTP3BoEK4ITNhpNMYmypJUiiooAKBjIzMzU89teyTcePGjbVardPavXs21VPvUsnKSos6eCrgdTXQ4JbJSo9cFqqiUF32AaW+ISiKgiwbkeXwxCV7+AzWjzS2uVjcjfX4zE7ssoc6t0xGij38mWLvx87ty9tUAClKdCmObDRGvqP6qKmuwZKcTrLNErs5kFXqqqswOtJJtn/72eOqqlBX9Qm7a+ztHAeFndveo9/YW7hn5nhk48lNS8Xrorq2kbSMmD6lKqiKm+raJtKzM8K1qqqq4q6tYMtub3AftDl0wuPqddHghjSnmZraJtIz0sLfURSVRRs/bVO0rypKVCRT/38Bmuqqccv2Nn1eQ2ooB/eXP/23f6zY8sD/m+smEiDQl0fpzdD2TE9VkiRVVTXrRKgiupmx2Ywsy8RZrJhkiE+w4XA4tMVuw2KOIy09g9zcs4IVBWZtDtU2kp6RRkNNNba0LKy6U/XswgcYMvl/ItF72cHICROxNL7L3qoO7tFN9dS5FLKy0qOuJ8Xrpqa2HkdKOo7Y+Ro8So211fgszqgerKrPQ3Vtw6lEReMoyIxw7ixnEt5DfgCcXSMwJZsTyLAbafFpF1tej4iCZorTjqz4aT2Rpk3QLI2AmyEG5CKsLcLgpBDASZIkidDFHgQ4gS5i2g57U26+eUrg5punPFle/tkHOTk5EwwGwyWhg/n0fT9nzupK8tNkKtzprF73EvnJZlY/ciOzVh7AbjQge6oxF93CumUzMVdvYfiU/6Vm326abHdTscpJ9uAbWTprEuBl9qRRbNz7Ge7CWyldHmxV7t5DbtbPKCjJY/veRgb0S2Pf7r3MW/0xY4ocrHx4Bo+v2cG2KifVNW8TJvc1Wxg+/v6g4qaP0p27WbT9KJP62agv28CY8b/CnJFPU0UZQ2Y+xiPTNNaklL6IY/xzTMk9xtYagadRYcWuHfT7NqP43nImjfoVZdWlVLm3MXz3C1hyL2ftknvAVcr4sb+hvLyUGnUXw7c/Q9HoO3hkxsgTsr8xM/5GfmEOe0treeTFNxk3IIum7UvJmfAYRSlWLFaFfe50Nmx8hV72Ju4Zfz1b91VQVacyfPh74DyHl19+BCd0eFwnD+hOVfIAqnfvJG/AhbgqP6Jo2t9ZfNsQyjcs5baHn2fv9u0s2dPC2HD1k4u7xv2MXfUatFXv283QWa+w7M4RqK5Kpo29jt0eJw5fPRT8nLXL7g1WTNRxcWr//5wxrOjNF5avq+qApenNUF8Q0Hx6lha6aUtauE3EmQyiVVEJ9f00G+Gr2oOoBhPyV4eQVLAmJtG1q5NDn/yH1ys9ZGek8Pwzy7nl0Zc57zwLy++9hvmbmrG7P8WYcR425Svqk69k98tzgpdHPS++XMbse082FufliacxvDgAACAASURBVHtv4pF1VeSmmKl0JbNq/SvkOY1sffo+Jj/8NgW56VSXlTLwtr+w+M7gfGjaRXberYwdncqGXUdQXI3MW/MR+aWPMPzhLRQ5BE1Nteg6Rp3C4jvOrQv/wZJPDrf5rHZ/KRfOf4UD3tjfBXh65Uomv/nx19qWqgb4+OO9vPbaayxa9Bfuu+/3/PKXv+S668YybNgwLrpoIEVFReTl9SY7O5vu3buTnJwsJSUlSQkJVikuLk42mUwGg8FglGXZJMtynCRJVkmSEiVJ6iJJUgqQAeQAZwNFvXr1GtDYeOT/CaHWNH+6SnTtfqHY36wIIVTxt19dKn52/7NCCFW89qcbRLcLJovjAVUIf4O4+uyu4qkdDUIIVQihiid/e4W45bEt4df65dNXZos+N8yPvHd8tzirx6XisFDEzecliX+83yhee+AG8avHNkW+c3iL6NHjMnG4nfUJoYp3Hvu1OPfqe7T9EW5x8/k9xENvlgohVBE4VirOyzxbfHBM+67/4xXCYOginnq3Wvu93yNa/O2vVwhVvP/8bJGamhqzdBdP/ru2w9+Elrf/+gtx1e+WtfvZG3+6SVz7xxUnXIcQqhCBOnFpjy7i+Q/qhRCq+OSV2aLbhVNFq1DF0feeEIb4s8SOw14hhCr+/MuLxE0PrQ3/9vPX5os+189uf73tHNebLkgVr3zqE8/87ipx459WiyPvLhV9rpur+51PXN8nQbzyefv7+tWHK0Xv3iXi4+D+PHX3VeLK3y4TAaEKITzijsvOEn95p1qoaqCmru6jBTL8HBgLjAauAi4HBgEXAEXBuZkDZEiSlBKcu4mSJFllWY6TZdlkMBiMJpPJYJQlGckgJdkTJAkkZ2p3srOzSXPEYbKnctFFA7l88PmYJCi5ehL33fd7HnroIV577TU+/ngvrrKVdOt9Lc1C5bnfXc2vH9vMl288TOZVdyIC5Zx75iUcCl6fgf3r6NL9Mg63c+2+/ddbuOq3y6Le++LNh8m88AaO+rXXr/+/6/nZ/SsQQuX4sSYCofUe3U7PrufyRfB74uhOzjCZ+MMLu4Lr8tLSqvDpK3OJP/MKDrWqCOH6Bnlsqp+/vfoWSq8LuCU/2qBsbvyKW1/+gPtvuIaMGIfqezv/w/MNdl6Y8vX0miRJoqCgAKPRiNFowGAwhhmcLOsfo3xxYb+bJCkh9iZUVQ0rhujYW6i/QiDE3ioqKhSns+uqt95a937XQ9tHOwdcelN2UCl26JDLmP/EbmAiAMWDBmGTAdnJ4IE57NpdyuTiQad0aGWbHRsydocdu82OzWbD7Tk5me7qrc8yfUk567a8oe2Pq4zNezzIm57j3i0a8/Qp1ZRWuOnXT6Pw5pxBjB8YlOAwWujMEO0/YQ51E+b8sM662r3s9eQzOFi9nj9kKMbJv6ZahWTAnj+Q/smaATpk0MUsW7sLGHGKG7NisxnBbsNhc2Kze/Ce5LnwNZQybvz9zH/xXxQE92fTxi34crO5797PACh3qxx5f/uKFV+1bpk48cajJ2BpoRpPnyRJoTSOgC7qKfT5aT7FL4xx8Zjj4jEbPKLVD9au8bgP+enaozt2exJ2eyKZ9t0kdU2hR48MEkQjLy5dTHWDB1Vx09RoJ9SBwW6zY3XYcdo8INuw4ibkDfU2NeCxOzhZor9187+xIvPgrN9rHLeynlLvHmA8alMF9971J3aWVeFVfdS7G2hwQ05o5ZZ8Jo3tF3xhxhL0NWT3v4AUM4DtVIEtwItvvMV/5DN56sr8KGuy1d3Arc9u5Nqrr+KqrOhM04/3vs8ftrv4x9QRdD1FZZO8vDxCzk+j0YjBEAK6iJmq+eHCvjgRfJSCEVGdbzYK4FQdwIXALQAoV145Yj+wJPOyX76rKMq1RqPxipNxKn/zIUds9ZNYnad2N+On/JnFq/9FL4cxvB+qOZlhY64mK7iq0aOvJSc3Uu8oW22crFdt7+oFTJrzXJv37338n4wfkMKPb8jhh8jZOAlZIaWJ28ffQMnspxhTlBa1vgFDr2ZkgQ1FUd4+66xeG6ZOHXdgeQTQOgI1vY9N0ZmeYX9xqE+ILBuE0SDwekH1ukVDazMIgb/FS0KCjTizjMlqo0uXLjidDqxGia6pPeh5Zlfm/fIPXL9sM9OHFWCs30xawR9i/m37c9Jit2P2uHBB28CL3N7VoZJVdDGjR4du/lcz1ZEBKNw36Wosk55nwxODMFNNcfJF0dPfasPSDnLJsvEbBA8QvPn227x0PJlnf34e8bqdVjxN3PHUegYOGcaNvaP/3uflH/GbTQdZ8ouRnGH9Zir4PXv2xGw2BwHOiMkUDXDtMLcgwElSRHo8UqEQYm26wEJAx97Ck+jAO09+YDI9WVpTc+Cdt9/ZeF2//lPCSr07t2zBrU7GpjayeVslo6ZGZKRtVhuNDfXf7fXnq2f6uBsY+eALDMvXHXt7AUPyPVS7nIwbrknuuBsaMNpO7RzkDZvK2v7j2sxcZ8qJQ/lWayINDV+3o5OPhTOnYx0yg2kjgnfp9AIKrWVs3t3AhH7JlG3aiFJ4PlkyeABX2VZ2NfgoTjazact/6N/v95F9sNtw1deg8F2W3agsvedGagp/zZIJA6I+GTJ0IK9+XrNr0uCSN3uflftp6OYZA2jtgJrklyQN+IQQAQEBOZxwixoIqALJgNloEkajEeFrBtkkzshM1wQiZS8ff/YVRlsi6ald+dztoVu3bjitXg42BRiecSb5uQkccXdl0MACjMD2deuoP8n7s5zdj0LfPsoaoSQG2dKc3ancWBUlEz9o8CXMmf8Zef3vQbsHK5psPh6qqz1MLSnGLEPVpjXsPYWmXV//3LYc4oFNX+CyxDFkvqYQau6Ww7vTL2XHhx+yse44H7y+ln+8HmQHlw/j/gGZ/P2t96htlLlh4TPBFcWzbOYE+p5i/9TMzEwduJnQkg2NYVM1msWFgU4IIaRgPamQpEAoVy4cWNCxtxDARbE3wJ+Rkfkv2Rj/3r8/eX6YEOI6IC+FSoYNugyjpwZzye+ZMCBydoeNv4VF435F0eoHyBt2Cy/On0rd9hcZO/NveBqqqGqQKSl5k4KRd/D4jE70t7z7GDfsl9R6XdTXVzOy5GLMGRez8cX51G96keW76iicezMvz9VO7bwV/2ZkvpUHn/4H48b/jDUL8rD46ql292DjrpVknMJxN1vtZFhPLa+h/8hJOB6/gYKiF3AWXcOWp2edFLBtXPkcjrRREWCT01iw+A+MHXMhKwpyKC2rZfGKNzGjAZs9zc69Iy/FalUpd2ew7pGIQm36gNEM5RoKC8/Dln4B69cvwdnJce1szJ92BevK3Oyr9LB33EUstDpYsOpNiu17eXDResiroV/R37TrYOZf+eMNF3029Bez1vyq76APez8cUHRRTL0ihy+WoUngF5oahxKch2oE1CQhywhFEUJIMmazdj0cb3YJc0JyuAuc1erEYTyAqzWO4uKLaFj3T15+zUW80YSzazwZPfPI7NmLmVNyGTXgQooyrMjpmaSd7P1PzmDc6CzWb9hDyfjobKmCUVMYuOwacvJeoXDEraxZMIOcEbcza8vN9C88n4Je6VRV7mPEPU8yf9IgbrtzAlOHX8CygjS81u70OoWcptOy8uDrjKNHj1JeXk5V1ZfU1tZSX19PY+MRmpqOBWWPQlUKLXi9rXg8Hnw+H0KoktahXiEQCEjB/guhagUpWLEgB8HfCBiDVQsmtMoFMxB3332/TzzbUnH56sA11y2/a2hOg1cmI+V0zC5TqaupBouTtOQfh0aI4nVTU9dASkYW1mC+RdP2peTN/IDaTX+ips5FekwawQ8xhBD7Dx8+vGbOnDm7Hnvs8VadKelrx/QMPfqDEU9Ft6h6X1qwioBg0m3QejGHGhljsUS6s2uqt4k4HEk4nV1JSelKvNnC2eecR2Gf3lElUk31NbixtUlFOdFwla2hZMYGdm5afNKuDa+7ibpGF2lpGVh0KTvuxnoaPDJZGcmn1OXM+N8+ubt06UJhYSFxcXHExZnDdyw9g9MzNyHUUIcrIct+SZZlFEURgUAUewsV18sx5mkAUILmqR/wz5//gA94rfcg8Y73zuFDe3RLuhboefodKZm0jOwflffLaLGRnd2+YoBstpGVZfuhAa3y8OHDbzz88MMf/PnPC7w6gPJ3AGixgQFFHxwIApsIlUUF5bvRQE3LTwtWE0SBms2mqd4mJYX6E2gd2rOzs+nduzfx8dFmkyMlg1PJ9rHnj2LhjEZqXSo5J6nrbrE5yLa13ZrNmYLtG/CD/3rGpptEfPbZZ1RW7ufgwRoOHTpEQ0OkvtTtduPxeHC5jtHS4kUIgc/nw+fz4ff7JUVpl71JQgg5CHChmlNDkL2ZguzNrF/uvntm4u9+97tLnU7n1ZIk9eGn8f2yuKYatuxzMWRA/g/HjVW1rL6+fv3s2X/88Mknn2ztBNBiF38Q1NoDND2ohVlaENREMEcNveRQBNT01QSp9OiRQU7OmfTu3fsHb7ryXY0fDbCFRlVVFZWVldTU1FBXV0dDg6bG63Id4/hxN0ePHsXj8SCECBbQt+Lz+fH7/SFl3iBzC0iqKqRg5DRkooZCQgadedouwOXk5Ji3bv3PxampqVfJsnzhT5Dz4x+BQGD7gQM171x8cUlpTc1BJSYw0Bmo6QEt7EcLVQ8AQh/xbMvSTJjNcegVb61Wfd1npJFxRkYGOTk5ZGdn/6jPhfHH9oeys7OJj4/HYoknLi4OszkuHGQwGk0EAgFkWUYIoTNTWzEYDKE7YYi9iUAggCRJoby3kEkqxwYXYsxTH2CurKw0p6f3eAfYUl7+WVF2dvZQk8k06qfL/8c3Wltb1+7bt+/doqJz9+vATM/SfO2Amr8DQAvoUjjCZmcsSzMajUGWpoGaxWLRyXgnYLMlkpSUhNOplx3KomfPHFJTU3/058T4Y/xTqampWK3WIMBZsFgiABcIKBgMBlRVDUdQvV5jCNzw+XxCluX2fG8iaJrGRk8DBJtioIXiTXqAA8y5ub13AnveeWfjS8XFFwyy2RKuglMKSv40Th/XR83x48f/+a9/bd49evTohg4ArT1Q83fG0DoyO0MsrX3TU+9P04IESUkOnE4nKSnd6N69O1lZZ9CrV08SExP/T5yfUzZFm1wujvllenSxRRcABxRqjx7HEJ9AakJ0r8qWFg/1bh/JjiQSTNL3MfkoLy/nyy+1iOmhQ/UcOFDN0aNNqGoAt1uLmLa0RJrFtLa24vf78Pm0hjE63xuh5jFqQKAKIQfbmYZM1I58cHpT1QSYJ0yYYPnTnx6+ICUlZYjRaOxQxXfP5tVY8oaRl3byzUNqy6BCgUGFP66JWrFrM2pGMbnhY6GiKFFl5hiDE/FEJciqquqK36M+ob6mGp/Z0SYi2NRQQ6Mb0tLT/nXoYM27v/nNbz59/fXX/ScJaP52AC3QAUNTQ/XOwQAB0Swt0ry4rT8toqPmdGpBgvT0dM444wxyc3O/U39a9d6tuJ2F5GecHhH37022aPnqVTy2r5nURDO1jc1MvvpKphd+t20HJEmid+/eJCQkBPN54vH7/RgMRlQ1EJXkG5k8BlpbjRgMPvx+v9C+L0uKEkBRFKH4AvhVFUkSKtpNNawWEsx3MwQna4jBGYOA5gsB24oVK1pXrFjxL2Drxo1vP3X++edfkJiYOFSSpChdl/VPP0TylAFfC9j2bYMVjT8uYFNd5UyZ+v9Ysf3t8Hsr593AxAfXk+ywhafyI2s+Yny/Ropt5zJv31GGt8eJPWUUpBVgH/cw25fOjABXxVbGjbuJWmMGdqUeufBmtj59D3jruWPC1Z+t/sS3OcnQ/NHHn5Yf0YFSe4Dm7wDQlBMAmmg/OGDAaDTppYaIizMTF2dpx5+mKd527ZpMamoqGRk92P36YtzXLaL3dxwksCk1jJm+lm1rH8R8GsyZU2BsrZQeaImSLXqyKZW3JhTTeOQI3nh7lGzR4NE3MDXPTkurj/g47S8f2b+Xi5//gv/MuobvS06wqamJysr9lJaWUltbSyAQ4OjRRo4d04IKsfluQWXeUNQURVHw+/1S8/EAfjVAnDkSPdUFGCQhkKFNkMEQZHDGGBYXtXz88d68nj17XhgfHz9EVZR0VcdCTgoAVG2RjTGMJUhh3I3Q5IOMtK8JLF6oqoWMbAilGnmaoK4R0rKIkq4J1b5UV0FyBlh1s1zxQnUdZGW3vaOqwc9S0qN/A7Bx0TSWNA5h1ZzxOmD7OQtrLmHr49Nj1lRF/06ArXLtg4xdsgt32WE2V/ybdCOAm0n9e2Kf9hKLpw0GoLS0rPbMM8/Y/NJD0z78xf++UB3yqZ4EoOkXJcjQAu2ZnASFT4M1zTFmpwGDwRRsWmyiI9MzISGBRJsVCYkuKen0zM4IBgkyyc7K4pFpA3FMfpO7RmRHy1gB7oZaGrxmsjOS28yVxrpqFGsyKfbom6qnqZ66Ji9p6RnRUlH4mD6oD4MXvs/4fj98g5fvTbYoBGqhY2c0yHyfgWaHw8E55xTS2urFZDKhKH7i4y3ExVmCQQbNEdvcbMJobMFkMuL1aizO5/Phb/VzpMEgVFWRBAqtrYqQ5ABmk4qEJHx+DdxCQQZZDgcZAkJgAClYxSNCzK0NyPXtW7gH+AR4+oILLuh74LMPz7/7hYrBdw7PSD+Z/zhtOOyqgOwJsHpe5P17x8AeGVQ3VJfC0Fmw+LYTr+/pmbCqCep3AlYwFsDWpfDwdFi+B3JSoLQM5jwNE0vAvQcyxsHIXGhwwZ462LILcu1QtxsGj4aMQu2f79sMW5sgG9j+Mky+F3oVQPk+mL8CxvYPwyErVrzO6EUPfyvzYM3qNxg64VG8S69l7fYGppUk4923ntUVWeyZfEmtx+PZUlHxxUfnnHPOfj276gDQ2gO20HuhoFIgDGiyrEpCqGqoIzgIJEkYTSYMsiS8LS3I5gS6WOPA3ywajrrIzM7GatFAzWKJIy4u1JHdSkKCjXjJzesvv4E17Qzi1GZSisfx5JzB2Bp2M278PZTvqcFYeRMbFlkZMP5u5k0dBkojsydfy8v7VLKsHhptxaxd9TfSLDBzRCbbPH0xmg3UV+5l9OzneHCyVs/54twbmPVyJYW9kqnYV88TW3cwIIyJZkaNuoglL69lfL+JUcfcVbaeCfc8y7zlKyhynLbAphv+Zpbt+IJxl7cN9n1V9Slbjjl5WVczuv2D91n0/n4qj7Qyb8LPvnf1V4PBQM+ePYmLs+DztWK1JgS7XMcCnJmWFi2K2traqoGb0ccZ2UaO1vmFK+CnS6IWPQ0EAqLFFwBJxWxShaqqkqIgCRCSFI6iBgCDJKGAZAhOfGPQTDW2A3TGHTt27AL2/ObKrKcu2/tR75ycnCKr1XphZ7lxSzfA2oWwtJ1yTFsBrJwLTdshexosvO3kTv72PbB3O6RZwRvsCjj9Ybgn6EqpWg+D58LEDcE7eh3csxMKHXDPcFi5Ee4bAw/fC2MfhHkToGoD9NoYvMk1wMTb4dndMDAd6rZC8V0wektw/3wV7Cy1c3tB2yuidM1fGFrxqjaR0y5g9fJ5nWe8q3WsXlfF7Hn98dReyuNr1zF14MSy8g/37fB0cZT1jDPomZmqY1mdAZqiY2eKHswkCTXYlFhoiguSkGRJGMORTolWb6tQhYF4s4n4OBONTcdFgHiOHWshLasnSbb4sD8t0j3Kis2mqd0e+bSUhL4jefiucfTokU56ejpZWVmQm8vGjeOZNfZsHJOfZ+bI7PBh2Pz4bNZ4L2LvrnmYUVk47RIeXr6NBVMHaiRgwDjWPTwZT/VGevW/m2njd5BjqWPJ4s0s23uQQWmA4sMXY0wUFhax68GdhBRvwrcmVx27dn2E2/ffEBU9Bdmivnln8z/pZ7B19wcsemcPl00dTMIPQFNTUlJwOruQmJiIzZYQFUHVJtFx3G4zZrOHlhaN4YVM02azD8lvxGLxi2BgQfK4FQymgDAatR4NAUUVICRZDkdRJQ3khBy8UOTghWAImqnhsq12gM5YWHjOR0Em99Jbb63rft555+U7HI4ik8l0IXBSnTUGBvX/HFlgaQQ3nFR2+bAxGqgBYXmYur0wYyGU12gmZL1ugttyNVADyMqCUO/nXbth9tPa8+yBkBGceVXboNYIqxfBGkD1gasUaoEsAFcDjUYHjnbcjDmDrmPxnKDEt9F2Qt9O4/a1lKZdduz8roEd9UXFe9+Z9NcvDA9NPhLykXbCztoDNb3CRrTvDFSBpGqPQoR7bsiyMMgI7YaoCJC03pEGIxaLRRiNRtKNcPBwA/bUbLp3tesCBBqo2WzRqRzZCefz+qKVPLsygWHDRpBbdGKhxy1b/oXZ05/Z9/4+6DbwUGfZC2jANmSQZo5bswZSbC9ld5VCTl4yQwYlc9u4axg38jIGDx/FoMKsqPXanQ5c9UfabM85YDJ1dZNPd1MUTlW2KCHBxlkJNs66soQ35q5g25FLuLzrD5P5bLVa6dOnD3a7ncTERBISbFitVqzWeJqaNAbndsdhNjdjNpvwek14va0cNxsx4Cc+Pux7E7LkB2NAMps1Budr1RicLGspIoGAQJKFhAgH7KTgXV3WmJzk1/niDDEgpwc6Y1BC6QCw8ZxzCk3Llj2Vk5NzZm+bzVZoNBr7Q/t9lcOKLl+z8M4SG+Ryw9ixMG8djOwHnt2QNqWd7cRsSzaCT4n4IvSeXXsGjBkdeT1mPIQFkOwO7IoLlxdi6ZjVkU5e3gl19N2KonzgcrlKFz7+VOWRD7fX2M3L1RhmFojxg3UEaHow0/9G1fvPgo0iVUlCIBCyLKMGAkKWDahKqxDIJNoTMZuMHGtsEAaTBl4mkwmP5wiyLOHzK9gSbcTHtMQL9fl0Op1065ZCevqFbLvmJj779FNWPfsgf33tA8rWzO88KgwUDric0cOCggujr8aekn1CqJjz8vuM3r6FLZs2MHHI+Ty47mMmFEekqtxNLmzO06OD1fckW6Ty1bEWuidp/Kyxvo4Dipkulh++nCMzMxOHwxEEOFsY4OLjj2KxaOCmBRZaMJlaOBZnwoCPhARjqGKBhDgjLsUvLHYDii8guYWCyRwQJlklEAjg96tIkiDYTybU6FlvpsrBC8QgSVI46KADufbAzvjRR3uN5513XimwD1h7xhlnGG4b9XxWq+GsHJ8vqbfJZCqQJKnXt3rAGqHaCCVF2k6vW31S6mQMGgjr18CIGbB3PdQoEfZmqwY5G4qDQY2GBh2GmXMpzm2ktNxDYeGJo8NCiApViE+OHq6r2Fyx78sbb5x0qKamRo1hZe2xs45YWvizoNKLErMOVQihT9UQWgtcScgyBFSEwWBA8fuFwWRCVVoxmOKxxscJgxRAUSE+Tmt7J7xNfHVM0K+4mOq9H1Djyub8NIcu4VYf9dTqPbt26UJqjx6cfe6FDO/nIG3kc3g1l6jGoC0W6hsa0LyZ2hg86FLu2VpJv9kTNZaremhoitxqNm3ZxMyRU/BUb2Onq4AF2UbAh8utUjRwKEUDh9JU+k/KymtBB2xlZaUU9W/LGN3lm5gxfxX3LFpMvv10BbZTki1K5S/PvcS7LRa6xct8eaSZcSMvo3/CaQHuJCYmUlBQQFJSkg7gErBaG2lqisdicREX10xcnJnGODMm4cVmM9HaqkVN0zJ9uCtM1B/2I1RFyGYFe4IiBZN7hdcbQDaqQhaqFEr01QUaQiAXZnHBvDg/0blxhhiAa/P8yy+/NPzuryWfARVxC3kHMDzyyJ9sh5ouT0909chqbbWdaRKmXiD3RlMo+fojC24fCCUDIMcGyVknN4nuehhGj4H8pZDbH7KCdqOcDCsWwaSBWvDAVQvm/rDx8YhTeuy4K1i5fiMTCkdFh+eF+Mzr9X5x/PjxL6urqw/+858bDt1///0tgDqhX7o4AZh1BGqxz8NL8PxEJdIG2wMJiKRrIElCQsJokIRf0dKKfK1eTOY4LCZVHGlyc7jBC0LCaJAwx8VjNcPefXXk9RtIalc7GYMv4u2NO/Hk9ya7W3JQlcNJt27dwqVRmZmZbHriLi5e+B8K8rKo3LuH2+77B3r4HzvtN4yddgMbF9sZPPl/WDBjBIOnzWXEzusoKLqI/CwHFeWVzFj8JtOHamS/afvLDBr6Gk2VpUx6+BlyLAB1jC++FHdGPnYaqHTnsmZkYRR5Wbvm34ydO6/NuffWl/PMMy8x8cHvD9i+11rRZk8zR7wKXRMTSTDJPwiI1dfX4/P5ycjo0f4+NjdTXV0dTug9ciRUa+oKyyC1tGi+t9ZWbxDctHpTj9uHIhSEqqWHhIrqQ9ULwRpUgjWooUJ7QkCne5SDgCfrlpCMkiGmCfSJnhtifw/Iq1a9mpyXl9ctOblbqs2WkG4ymbobDIYMSZIy2xp9bUdDDShWSPuaESBFAWohayiUl0fsZlWBmhqwJ4MjYkx7hRAHWg/vOjhw+B+/WrH26a+UxvrDZWVlR66/fvxRIhp64fSJDsCsI4bW4fNQ8bnu9/oUjXCqRugx2B0tqNYsiZBcvW6JlEMZZAQS1gQblrhQnac5GPEMBQgSwu3wunRxhFlaKDigb4Xn87iorW/UOjpZTz6LzNNUT4NLIS09HXPw7jRzeCbZM99nYqGCYnGSHJXuodJQW4sbM1npKdGpI/vWMnD6OrZvXoL1NCArP7oi+G8KbKFRV1cXLKQ/REPDYRobG4MabxGlEI+n/YoFv98fzn3TVy4EAooUCKhhgAsE1CCwqahqGNTQMbqOQC4MUDGqI50tcsxzOeZ5aBvSo48usPXt29eelpZmT0pKslutVrvZbLYbjUa7wWCwy7JskyTJJklSgiRJ8UEgDAkAGHTetRAw+Gq34pv5LN5ze4uWba+J5vQJwr3wl6o7EAgcVxTF1dra6vJ4PMebmppcX3311fG9ez9233333R4deIkYIBOdb+/04gAAC8xJREFUgJkeyDpialFLJ0AWBWZ6IAOELEtC65ImYTDIOkCTMRiMsZUDRBJt21YQaMm2miJHYqI9zNKSk7sFu7FrOWrf5QgB24yhX287+7atpcFRTEn+6SEP/xOwdZbN4vdz4MCBsIClJoN0lGPHXO2KWEarhfiCAKe0A3ARFqd1uldD5VrEMLjOQK5dsNOVd8mdgJqhHWCLXfTb6WyhncfY59FO2rbP9Y+dLbFAFrt0xNLaPA9Jv7cHYicCs9DzmJ4aUeysLaCFQC1SuN6RGGRSkh2HowvJyV1xyG7+82ElDmcXDIbo/vT9h41nQO632119387NWHsNJMtp/q++zn8CtpMYx44d4+DBg9TV1XH4cIi9RXTemps9OvM0UrUQqTmNBrhAIBBlpqqqGgVyQqiSqoqwqUq44bOQhEAPdFI7IBQDUMKgVUO0y9I6AzepA4CT2wG1zgCuM3CLBTTaAbHOGJraAVuLeq2l2kidgljQZ6brNxv2m4XALGhqSkhSNJjpQE2EFGP0gGY0RhhaqHqgbV6aTVfnGfGl9ejRg6SkJH4a33lU9P/eSEpKIikpia5du1Jb+xX19YeCOm9HgyVZx4MF9c3hgnqvV1+SpWdvIRM1EKUeoglcBggEVFRVFUGQE6qqSqF+M1q/BqJYXPACjmV0OgCSZEmKBTwhBwGyM0DrDNy+DnP7OsD2dczOdgEuCEidAhjRzbLDoKr3mYVeBwEtiplpQCaH2j6KEFOL1HYawwrOEZYWqR6wRKVwhFhaEl26dCE5OZmUlFTS07v/n5AX+gnYfuBRWvoJublnkZqayoEDNdTVfcXhw4c5cuRIlJBlyDzVK4boa04135sGcoFAGOAIBAJ6iSRJx+BEhMmJ0KOIMVXRR1cjbFxI7YNSFNhFfacdc1c6AbB1xNpiTVLRCbh1BnBqe691oNQZs4v9TdS29K9jTE1kWdYFA/SAZiCkuhELaAZDhKHpi9bbU7bVzM6IEGTXrl2DLK07mZk/KVr9BGw/wMjMzCA1NYXa2tqwSm9b87Q5GFwIAZwvxv/Whr2FAgxBkFM7BDkdwBHN5qRYoIt63s5nbZZgHl0nzExL0voufWxB1nWyTK6jBb1pGfOa9lhZiJnp/GftgFnI7DR2wtL0fjRzUBMwXicCGW12htRt09PTMZvNP11gPwywBdixdx//rPiKY4pMwVk9ueHcMzADrc1NvLrrM/bWHcNgsTGyuJALu8ckwivNPPXvjzmz99kMzvjv9R2YzWays7NJTU1tY57qo6fR/jc9g+s4eqooCoq/lZZWCYtZigI5XbChPZAj0iYVKfZRVVUkWQ4J2MQyOzphYrrXUkjW62TBrCPGFvWeCNnZJ2ZydAJgkd8LIVRVYDDIUUxM5y9Dn6IRC2ahwIC+CbeeoXUW7Yz40Sxt/GihaGes2RnbUKW9UbYbsoqg3Zaw7iomTHmYxSuW4NRd1bXle5BT8klzfDuA6SldxeC5u9j58vxvvJ4hc3ex/RTWczL/6RT02I6zenctRYXZpJpVlr29kc+OX8aDg7L5srKKj5tlLu2bi/vIQX7995UsvmMiA7pENvPW5q0see9LrrL1+K8GttCIj4+nZ88c0tJSqauro76+Pmie6v1vIYBraSOL1FH0tLnhCIcbA+RkpejTQ4TOD6cHOYRQhRZwUAkGHsJAF7zOJSEEkiwjRcCOk2FdMcB3IjA72XKSNqapiPw4CqRC7KojtqcHxPDz4P8zGAzh92MCAEQeY8FM1jXd1gOaIdiYu6NoZ7SsUEiFQwO0iB+ta9eupKSkkJaWFpWPdqIxbQQsrID+7RTNrVsyB8uA8VGgBrDkvp/jmPxWVBH8N7qhpxVw+6RvntKhel1UVR8+pd+ezH/6+sAmO3hg8uXhl108h7l1z5cwKJvcvkU80Df0STa7P/mULfsbGdBFOxCuuv08V2vjujOteH5k1DchIYGePXvSvXv3cPT0SO1nPP/Ce/QtKcHechzXoUo2fVJLn765JLS2BAMMsQzOy4EvvqDpWDNCERw5cgRVMpGS4oRAgIDi5fDhY1itRnHM5UJFxp7kQBZ+yeVyowS0WIIpziJMRgkhBAG/D18gIABJlg1IhMBORVFULSChqqCxlpMFrTYApkXYpTClC6bkI7cvcijC3wmBrPb9YF4YYYAK7m2Q1WmAJIQqhNB+L0kyRqMhBHAEFD8CCUnWopQhUAv4W1ExorS2EhBgtdmxxpnCQNZO2gZtfWntMzSt70AQ0Izx7PnESu4ZNnZsS6RbThI3T3SQ3k3zox2tSONvz1hxyTB+OgwtBDywcBGMvxPSLOCrh0eehRl3QeUmWLkZqtzw+FxIM8OISTAwN4QSjSx+4l/ctWVp+OA2lW9jwbNvsbm0HsvyP9O03UHe4GuZOFRrZly5cy1bm5JJqd3Ci5s+IXfIz7lv8nAqd23g6ZfforqukfTc85lx+3TS7RpMrHv6YbZVHMeadX7UiazYvobtjXa8ezawteIII6f8lrEluSdx1fhYvWQea3Z+Sb+RN3Hb2BJkoL58O0ufXUV59SGcGb2Zdvud5KVZT/ifVE89yxYvZmvpAb5x+v+n9Q3kpreTS+M/TsURlT6pQVam+nho7U6mX3k+cT/Ojl+AVlyfk5PDOeecQ/65gzjH0cCLG8vIykpjx/o1mDPyyUxLpVu3FJKTu+J0OnE4uoTLuex2O6k9Mkh1JiKZE0hLSyM1pSsJVo0BxMcZcLuPc+RoM3aHE7vNgtEcR5xRFrIpTtiTkoQtwSx8Lc1gDGa0WzQ5Js02MwmTySSMRqMwGiQRBAhhMBqE1sRBUg0GgyrLcntLILiEGvdGL5r2nCJJkqZ4odVVhl/HLAFJkgKyJAVkWQ4myWrrlmUpvB2BUIWGumpwu6rBYFAltI5NJpNRSKhCCajCaDQKk8kkTCazMEiIgCp0irNxiIBCa2ursCQkCptZFm7XcREXr/m/Io794HGOj8dqtXC8sZ5Dhw5RV1dHbW0tBw8e5Ksjbux2O0lJSTgcXYJ+sq5065ZCamoq6cndeW9LBmUNZ3D99Tk07sxlW10+55xzDkpFDtfeaaVkLEwaCXeNgZ31gBWyfDDpHk0sdNYUcKWBXYb0XjB8ODjNUDxEe56lu+R85ZvYKfdnQHqEp1iSsxg2/AqyU2z06ncxw4dfQb9eEVm/yp1vct+0X7CqwsL4iddjVVwAVFWU02vAldx+1x1kud9j2JR5YcGCvH6DKckK8MiyN2KA7Q1mTLsbtWAoE4ZmMn3szVSfhERR0+5VbHD1YvqUa1k9+zoWb6gEoKayAmd+CXfedQclKYcYNuIW3Cf6T6qLGSMuZl2NnWm33/HNggfV+z/hsU8CPDnj7FiiyTNrNhJ39gCu7KGVJO7c8R5NmUUM6mbmw/8DzkuLxUJ2djZ3LX6BTy+/mMcf/4DGzJHMvu4S3C59ioinTYmW1ZqITbio9PjJyMiI9sEZtRmTcVZvEg1CZ6ZaMMf7aPY0o0jxyHKLUDEQF2dGmM3EWVQaWr2S0RxHnCFosiqCFq9fxNsSkIWgtaUZRYDRaCRiwYqQORomWkEiFkW+hNDipqo/ALKMjCoFAmAwGJGljk1QjWRJ+H0+JIMRoxx5D0AEAgQkCYslLsqkNJtNBBSFgCowGmR8AUFcnBlJkpEsFpQWF65WSVgscWFzs/W4hCkuiWSnA1lK4HhlLXJcPPFGfbWAPjAg02pLJI5oc9Sc2AWns0uUD03fSMVOEiajg9/+0Um/1G50P57KqzVGLBZYuhAmzoIRJdr/nzoUVqyF4ikwZhZsHgFjxkBTCmwMypqlZGuL3Qz9StqaorXlVVhyMqNkXSzODEpKMlifYsWRX0xJSVuzzZh3BY/Pux0ZGB58b8j4GdRW7GVveQ0ZhedS9+xbNALJQE5hMSm+MuRlbU3IgpE3Mm3UEKCYgbP+zN5aTSm505E+iPn3jscBzJpxDfe8uIrbh82k3/CJZFfvY09ZJeZe52CpfIAKNxR18p9cu1ezvDaPms0zsQP/H3ixgJmQO5wuAAAAAElFTkSuQmCC"], [1, "details-container"], [1, "details"], ["href", "https://www.tensorflow.org/api_docs/python/tf/debugging/experimental/enable_dump_debug_info", "target", "blank", "rel", "noreferrer noopener"], ["href", "https://www.tensorflow.org/api_docs/python/tf/debugging", "target", "blank", "rel", "noreferrer noopener"]], template: function InactiveComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "div", 0);
            ɵɵelementStart(1, "div", 1);
            ɵɵtext(2, "Debugger V2 is inactive because no data is available.");
            ɵɵelementEnd();
            ɵɵelementStart(3, "div");
            ɵɵtext(4, "To use the debugger,");
            ɵɵelementEnd();
            ɵɵelementStart(5, "div");
            ɵɵelementStart(6, "ol");
            ɵɵelementStart(7, "li");
            ɵɵtext(8, " Add the following line to the beginning of your program: ");
            ɵɵelementStart(9, "div", 2);
            ɵɵelementStart(10, "span");
            ɵɵtext(11, "tf.debugging.experimental.enable_dump_debug_info(");
            ɵɵelementEnd();
            ɵɵelementStart(12, "span", 3);
            ɵɵtext(13, "logdir");
            ɵɵelementEnd();
            ɵɵtext(14, ", ");
            ɵɵelementStart(15, "span", 3);
            ɵɵtext(16, "tensor_debug_mode=\"FULL_HEALTH\"");
            ɵɵelementEnd();
            ɵɵtext(17, ", ");
            ɵɵelementStart(18, "span", 3);
            ɵɵtext(19, "circular_buffer_size=-1");
            ɵɵelementEnd();
            ɵɵelementStart(20, "span");
            ɵɵtext(21, ")");
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementStart(22, "li");
            ɵɵtext(23, "Re-run the program.");
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementStart(24, "div", 4);
            ɵɵelementStart(25, "div", 5);
            ɵɵelementStart(26, "div", 6);
            ɵɵelement(27, "img", 7);
            ɵɵelementEnd();
            ɵɵelementStart(28, "div", 8);
            ɵɵtext(29, "Auto-alerts for problems found");
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementStart(30, "div", 5);
            ɵɵelementStart(31, "div", 6);
            ɵɵelement(32, "img", 9);
            ɵɵelementEnd();
            ɵɵelementStart(33, "div", 8);
            ɵɵtext(34, " Integrated debugging to trace problems to their causes ");
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementStart(35, "div", 5);
            ɵɵelementStart(36, "div", 6);
            ɵɵelement(37, "img", 10);
            ɵɵelementEnd();
            ɵɵelementStart(38, "div", 8);
            ɵɵtext(39, "Link log to code");
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementStart(40, "div", 11);
            ɵɵelementStart(41, "div", 12);
            ɵɵtext(42, " The log directory must contain TensorFlow Debugger (V2) data. tf.debugging.experimental.enable_dump_debug_info() will collect tensor data, graph structures, the associated stack traces, and source code to the specificed directory logdir as the instrumented TensorFlow program executes. ");
            ɵɵelementEnd();
            ɵɵelementStart(43, "div", 12);
            ɵɵelementStart(44, "div");
            ɵɵtext(45, " See ");
            ɵɵelementStart(46, "a", 13);
            ɵɵtext(47, " documentation ");
            ɵɵelementEnd();
            ɵɵtext(48, " of the Python API of Debugger V2. ");
            ɵɵelementEnd();
            ɵɵelementStart(49, "div");
            ɵɵtext(50, " See ");
            ɵɵelementStart(51, "a", 14);
            ɵɵtext(52, " here ");
            ɵɵelementEnd();
            ɵɵtext(53, " for other TensorFlow debugging APIs. ");
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementEnd();
        } }, styles: [".arg[_ngcontent-%COMP%] {\n  color: lightblue;\n  font-style: italic;\n  margin: 2px;\n}\n\n.code[_ngcontent-%COMP%] {\n  font-family: 'Roboto Mono', monospace;\n  margin: 10px;\n}\n\n.container[_ngcontent-%COMP%] {\n  height: 100%;\n  font-family: Roboto;\n  font-size: 15px;\n  overflow-y: auto;\n  padding: 50px;\n}\n\n.details-container[_ngcontent-%COMP%] {\n  display: inline-flex;\n  vertical-align: middle;\n  width: 100%;\n}\n\n.details[_ngcontent-%COMP%] {\n  display: inline-block;\n  margin: 10px 60px;\n  width: 50%;\n}\n\n.exhibit-container[_ngcontent-%COMP%] {\n  white-space: nowrap;\n  width: 100%;\n}\n\n.exhibit[_ngcontent-%COMP%] {\n  align-content: center;\n  display: inline-block;\n  margin: 10px 60px;\n  vertical-align: top;\n  width: 310px;\n}\n\n.exhibit[_ngcontent-%COMP%]   .description[_ngcontent-%COMP%] {\n  font-weight: bold;\n  text-align: center;\n  width: 310px;\n}\n\n.exhibit[_ngcontent-%COMP%]   .screenshot[_ngcontent-%COMP%]   canvas[_ngcontent-%COMP%] {\n  height: 200px;\n  width: 100%;\n}\n\n.title[_ngcontent-%COMP%] {\n  font-size: 135%;\n  font-weight: bold;\n  margin-bottom: 25px;\n}"] });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/views/inactive/inactive_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class InactiveContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
        }
    }
    InactiveContainer.decorators = [
        { type: Component, args: [{
                    selector: 'tf-debugger-v2-inactive',
                    template: ` <inactive-component></inactive-component> `,
                },] },
    ];
    /** @nocollapse */
    InactiveContainer.ctorParameters = () => [
        { type: Store }
    ];
    /** @nocollapse */ InactiveContainer.ɵfac = function InactiveContainer_Factory(t) { return new (t || InactiveContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ InactiveContainer.ɵcmp = ɵɵdefineComponent({ type: InactiveContainer, selectors: [["tf-debugger-v2-inactive"]], decls: 1, vars: 0, template: function InactiveContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelement(0, "inactive-component");
        } }, directives: [InactiveComponent], encapsulation: 2 });

    /* Copyright 2019 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * Actions for Debugger V2.
     */
    /**
     * Actions for the Debugger Component.
     */
    const debuggerLoaded = createAction('[Debugger] Debugger Loaded');
    const debuggerUnloaded = createAction('[Debugger] Debugger Unloaded');
    const debuggerDataPollOnset = createAction('[Debugger] A New Debugger Data Polling Event Begins');
    const debuggerRunsRequested = createAction('[Debugger] Debugger Runs Requested');
    const debuggerRunsLoaded = createAction('[Debugger] Debugger Runs Loaded', props());
    const debuggerRunsRequestFailed = createAction('[Debugger] Debugger Runs Request Failed');
    /**
     * Number of alerts and their type breakdown or detailed alerts are requested.
     */
    const numAlertsAndBreakdownRequested = createAction('[Debugger] Number and Breakdown of Alerts Requested');
    /**
     * Number of alerts and their type breakdown are loaded.
     */
    const numAlertsAndBreakdownLoaded = createAction('[Debugger] Number and Breakdown of Alerts Loaded', props());
    const alertsOfTypeLoaded = createAction('[Debugger] Alerts Data of an AlertType Is Loaded', props());
    const alertTypeFocusToggled = createAction('[Debugger] Alert Type Focus Toggled', props());
    /**
     * Actions related to top-level (eager) execution
     */
    const numExecutionsRequested = createAction('[Debugger] Number of Top-Level Executions Requested');
    const numExecutionsLoaded = createAction('[Debugger] Number of Top-Level Executions Loaded', props());
    const executionDigestsRequested = createAction('[Debugger] ExecutionDigests Requested', props());
    const executionDigestsLoaded = createAction('[Debugger] ExecutionDigests Loaded', props());
    const executionScrollLeft = createAction('[Debugger] Scroll Leftward on the Execution Timeline');
    const executionScrollRight = createAction('[Debugger] Scroll Rightward on the Execution Timeline');
    const executionScrollToIndex = createAction('[Debugger] Scroll the Execution Timeline to Given Index', props());
    const executionDigestFocused = createAction('[Debugger] Execution Data Objects Being Focused On', props());
    const executionDataLoaded = createAction('[Debugger] Execution Data Objects Loaded', props());
    /**
     * Actions related to intra-graph execution
     */
    const numGraphExecutionsRequested = createAction('[Debugger] Number of Intra-Graph Executions Requested');
    const numGraphExecutionsLoaded = createAction('[Debugger] Number of Intra-Graph Executions Loaded', props());
    const graphExecutionDataRequested = createAction('[Debugger] Intra-Graph Execution Data Requested', props());
    const graphExecutionDataLoaded = createAction('[Debugger] Intra-Graph Execution Data Loaded', props());
    const graphExecutionScrollToIndex = createAction('[Debugger] Scroll Intra-Graph Execution List to Given Index', props());
    const graphExecutionFocused = createAction('[Debugger] Graph Execution is Focused On', props());
    /**
     * Actions related to graph ops.
     */
    const graphOpFocused = createAction('[Debugger] Graph Op Is Focused On', props());
    const graphOpInfoRequested = createAction('[Debugger] Graph Op Info Requested', props());
    const graphOpInfoLoaded = createAction('[Debugger] Graph Op Info Loaded', props());
    /**
     * Actions related to source files and stack traces.
     */
    const sourceFileListRequested = createAction('[Debugger] Source File List Requested.');
    const sourceFileListLoaded = createAction('[Debugger] Source File List Loaded', props());
    const sourceLineFocused = createAction('[Debugger] Source File Line Is Focused on', props());
    const sourceFileRequested = createAction('[Debugger] Source File Requested', props());
    const sourceFileLoaded = createAction('[Debugger] Source File Loaded', props());
    const stackFramesLoaded = createAction('[Debugger] A Set of Stack Frames Have Been Loaded', props());

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/store/debugger_types.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const DEBUGGER_FEATURE_KEY = 'debugger';
    /** @enum {number} */
    const TensorDebugMode = {
        // NOTE(cais): The string name and number values of these enums
        // need to match TensorDebugMode in tensorflow. See
        // https://github.com/tensorflow/tensorflow/blob/master/tensorflow/core/protobuf/debug_event.proto
        UNSPECIFIED: 0,
        NO_TENSOR: 1,
        CURT_HEALTH: 2,
        CONCISE_HEALTH: 3,
        FULL_HEALTH: 4,
        SHAPE: 5,
        FULL_NUMERICS: 6,
        FULL_TENSOR: 7,
        REDUCE_INF_NAN_THREE_SLOTS: 8,
    };
    TensorDebugMode[TensorDebugMode.UNSPECIFIED] = 'UNSPECIFIED';
    TensorDebugMode[TensorDebugMode.NO_TENSOR] = 'NO_TENSOR';
    TensorDebugMode[TensorDebugMode.CURT_HEALTH] = 'CURT_HEALTH';
    TensorDebugMode[TensorDebugMode.CONCISE_HEALTH] = 'CONCISE_HEALTH';
    TensorDebugMode[TensorDebugMode.FULL_HEALTH] = 'FULL_HEALTH';
    TensorDebugMode[TensorDebugMode.SHAPE] = 'SHAPE';
    TensorDebugMode[TensorDebugMode.FULL_NUMERICS] = 'FULL_NUMERICS';
    TensorDebugMode[TensorDebugMode.FULL_TENSOR] = 'FULL_TENSOR';
    TensorDebugMode[TensorDebugMode.REDUCE_INF_NAN_THREE_SLOTS] = 'REDUCE_INF_NAN_THREE_SLOTS';
    /** @enum {string} */
    const AlertType = {
        FUNCTION_RECOMPILE_ALERT: "FunctionRecompilesAlert",
        INF_NAN_ALERT: "InfNanAlert",
        TENSOR_SHAPE_ALERT: "TensorShapeAlert",
    };
    /** @enum {number} */
    const CodeLocationType = {
        // The code location for an eager (top-level) execution.
        EXECUTION: 0,
        // The code location for the creation of of an op (node) in a graph.
        GRAPH_OP_CREATION: 1,
    };
    CodeLocationType[CodeLocationType.EXECUTION] = 'EXECUTION';
    CodeLocationType[CodeLocationType.GRAPH_OP_CREATION] = 'GRAPH_OP_CREATION';

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/store/debugger_store_helpers.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Helper function that extracts the stack trace being focused on.
     *
     * This examines whether the current focused code location is for an
     * eager (top-level) execution or a graph-op creation, and then queries
     * the corresponding substates accordingly.
     *
     * @param {?} state
     * @return {?}
     */
    function getFocusedStackFramesHelper(state) {
        if (state.codeLocationFocusType === null) {
            return null;
        }
        /** @type {?} */
        let stackFrameIds = [];
        if (state.codeLocationFocusType === CodeLocationType.EXECUTION) {
            const { focusIndex, executionData } = state.executions;
            if (focusIndex === null || executionData[focusIndex] === undefined) {
                return null;
            }
            stackFrameIds = executionData[focusIndex].stack_frame_ids;
        }
        else {
            // This is CodeLocationType.GRAPH_OP_CREATION.
            if (state.graphs.focusedOp === null) {
                return null;
            }
            const { graphId, opName } = state.graphs.focusedOp;
            if (state.graphs.ops[graphId] === undefined ||
                !state.graphs.ops[graphId].has(opName)) {
                return null;
            }
            stackFrameIds = (/** @type {?} */ (state.graphs.ops[graphId].get(opName))).stack_frame_ids;
        }
        /** @type {?} */
        const stackFrames = [];
        for (const stackFrameId of stackFrameIds) {
            if (state.stackFrames[stackFrameId] != null) {
                stackFrames.push(state.stackFrames[stackFrameId]);
            }
            else {
                return null;
            }
        }
        return stackFrames;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/store/debugger_store_utils.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Find the index of a file spec among an array of file specs.
     * @param {?} fileList
     * @param {?} fileSpec
     * @return {?} The index of `fileSpec` in `fileList`. If not found, `-1`.
     */
    function findFileIndex(fileList, fileSpec) {
        return fileList.findIndex((/**
         * @param {?} item
         * @return {?}
         */
        (item) => item.host_name === fileSpec.host_name &&
            item.file_path === fileSpec.file_path));
    }
    /**
     * Determines if a source-line spec points at the bottommost in its file
     * in a given stack trace.
     *
     * @throws Error if `stackFrame` is not a frame in `stackFrames`.
     * @param {?} stackFrames The stack trace to examine.
     * @param {?} stackFrame A spec that describes a frame in the stack trace.
     * @return {?} Whether `stackFrame` points to the bottommost stack frame
     *   (in the Python sense) of the file it belongs to among the frames in
     *   `stackFrames`.
     */
    function isFrameBottommostInStackTrace(stackFrames, stackFrame) {
        /** @type {?} */
        let matchingIndex = -1;
        /** @type {?} */
        let bottommostIndex = -1;
        stackFrames.forEach((/**
         * @param {?} __0
         * @param {?} i
         * @return {?}
         */
        ({ file_path, lineno }, i) => {
            if (file_path === stackFrame.file_path) {
                bottommostIndex = i;
                if (lineno === stackFrame.lineno) {
                    matchingIndex = i;
                }
            }
        }));
        if (matchingIndex === -1) {
            throw new Error(`Stack frame ${JSON.stringify(stackFrame)} is not found.`);
        }
        return matchingIndex === bottommostIndex;
    }
    /**
     * Finds the bottommost stack frame in a stack trace.
     *
     * @param {?} stackFrames Stack frames of the stack trace to look in.
     * @param {?} focusedSourceLineSpec The currently focused stack frame.
     * @return {?} The stack frame that is in the same file as `focusedSourceLineSpec`,
     *   but at the bottommost location.
     */
    function getBottommostStackFrameInFocusedFile(stackFrames, focusedSourceLineSpec) {
        if (focusedSourceLineSpec === null) {
            return null;
        }
        for (let i = stackFrames.length - 1; i >= 0; --i) {
            /** @type {?} */
            const stackFrame = stackFrames[i];
            const { host_name, file_path } = stackFrame;
            if (host_name === focusedSourceLineSpec.host_name &&
                file_path === focusedSourceLineSpec.file_path) {
                return stackFrame;
            }
        }
        return null;
    }
    /**
     * Find the first range (with begin and end properties) in an array of ranges
     * that equals `[begin, end)`.
     *
     * @param {?} ranges The ranges to search in.
     * @param {?} begin The begin of the range to search for (inclusive).
     * @param {?} end The end of the range to search for (exclusive).
     * @return {?} Index (>=0) if found. -1 if not found.
     */
    function findBeginEndRangeIndex(ranges, begin, end) {
        if (begin >= end) {
            throw new Error(`Expected begin to be less than end, ` +
                `but got begin=${begin}, end=${end}`);
        }
        return ranges.findIndex((/**
         * @param {?} range
         * @return {?}
         */
        (range) => range.begin === begin && range.end === end));
    }
    /**
     * Determines if an array of ranges contains a range that strictly includes
     * `[begin, end)`.
     *
     * @param {?} ranges The ranges to search in.
     * @param {?} begin The begin of the range to search for (inclusive).
     * @param {?} end The end of the range to search for (exclusive).
     * @return {?} `True` if and only if one of the ranges of `ranges` strictly
     *   includes `[begin, end)`.
     */
    function beginEndRangesInclude(ranges, begin, end) {
        if (begin >= end) {
            throw new Error(`Expected begin to be less than end, ` +
                `but got begin=${begin}, end=${end}`);
        }
        return (ranges.findIndex((/**
         * @param {?} range
         * @return {?}
         */
        (range) => range.begin >= begin && range.end <= end)) !== -1);
    }
    /**
     * Computes bottommost stack frame in the currently-focused stack trace.
     *
     * @param {?} state Input DebuggerState (will not be mutated).
     * @return {?} If `stickToBottommostFrameInFocusedFile` and there is a stack trace
     *   in focus and if a bottommost frame can be found in the file currently in
     *   focus, return the bottommost frame. Else, return the current value of
     *   `focusLineSpec` in `state.sourceCode`.
     */
    function computeBottommostLineSpec(state) {
        /** @type {?} */
        const currentFocusLineSpec = state.sourceCode.focusLineSpec;
        if (!state.stickToBottommostFrameInFocusedFile) {
            return currentFocusLineSpec;
        }
        /** @type {?} */
        const focusedStackFrame = getFocusedStackFramesHelper(state);
        if (focusedStackFrame === null) {
            return currentFocusLineSpec;
        }
        /** @type {?} */
        const bottommost = getBottommostStackFrameInFocusedFile(focusedStackFrame, currentFocusLineSpec);
        if (bottommost === null) {
            return currentFocusLineSpec;
        }
        else {
            return bottommost;
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/store/debugger_reducers.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const DEFAULT_EXECUTION_PAGE_SIZE = 100;
    /** @type {?} */
    const DEFAULT_GRAPH_EXECUTION_PAGE_SIZE = 200;
    /**
     * @return {?}
     */
    function createInitialExecutionsState() {
        return {
            numExecutionsLoaded: {
                state: DataLoadState.NOT_LOADED,
                lastLoadedTimeInMs: null,
            },
            executionDigestsLoaded: {
                loadingRanges: [],
                numExecutions: 0,
                pageLoadedSizes: {},
            },
            // TODO(cais) Remove the hardcoding of this, which is coupled with css width
            // properties.
            displayCount: 50,
            pageSize: DEFAULT_EXECUTION_PAGE_SIZE,
            scrollBeginIndex: 0,
            focusIndex: null,
            executionDigests: {},
            executionData: {},
        };
    }
    /**
     * @return {?}
     */
    function createInitialGraphExecutionsState() {
        return {
            numExecutionsLoaded: {
                state: DataLoadState.NOT_LOADED,
                lastLoadedTimeInMs: null,
            },
            executionDigestsLoaded: {
                loadingRanges: [],
                numExecutions: 0,
                pageLoadedSizes: {},
            },
            // TODO(cais) Remove the hardcoding of this, which is coupled with css width
            // properties.
            displayCount: 100,
            pageSize: DEFAULT_GRAPH_EXECUTION_PAGE_SIZE,
            scrollBeginIndex: 0,
            focusIndex: null,
            graphExecutionDigests: {},
            graphExecutionDataLoadingPages: [],
            graphExecutionDataPageLoadedSizes: {},
            graphExecutionData: {},
        };
    }
    /**
     * @return {?}
     */
    function createInitialGraphsState() {
        return {
            ops: {},
            loadingOps: {},
            focusedOp: null,
        };
    }
    /** @type {?} */
    const initialState$4 = {
        runs: {},
        runsLoaded: {
            state: DataLoadState.NOT_LOADED,
            lastLoadedTimeInMs: null,
        },
        activeRunId: null,
        // The initial values of lastDataPollOnsetTimeMs and
        // lastNonEmptyPollDataTimeMs ensures that initially, before
        // the onset of any data polling, the difference of the two
        // is 0, which causes a backing-off polling algorithm to use
        // the lower-bound polling interval.
        lastDataPollOnsetTimeMs: -1,
        lastNonEmptyPollDataTimeMs: 1,
        alerts: {
            alertsLoaded: {
                state: DataLoadState.NOT_LOADED,
                lastLoadedTimeInMs: null,
            },
            numAlerts: 0,
            alertsBreakdown: {},
            alerts: {},
            executionIndices: {},
            graphExecutionIndices: {},
            focusType: null,
        },
        executions: createInitialExecutionsState(),
        graphExecutions: createInitialGraphExecutionsState(),
        graphs: createInitialGraphsState(),
        stackFrames: {},
        codeLocationFocusType: null,
        stickToBottommostFrameInFocusedFile: false,
        sourceCode: {
            sourceFileListLoaded: {
                state: DataLoadState.NOT_LOADED,
                lastLoadedTimeInMs: null,
            },
            sourceFileList: [],
            fileContents: [],
            focusLineSpec: null,
        },
    };
    // TODO(cais): As `executions` is getting large, create a subreducer for it.
    /** @type {?} */
    const reducer$4 = createReducer(initialState$4, on(debuggerRunsRequested, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return Object.assign(Object.assign({}, state), { runsLoaded: Object.assign(Object.assign({}, state.runsLoaded), { state: DataLoadState.LOADING }) });
    })), on(debuggerRunsRequestFailed, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return Object.assign(Object.assign({}, state), { runsLoaded: Object.assign(Object.assign({}, state.runsLoaded), { state: DataLoadState.FAILED }) });
    })), on(debuggerRunsLoaded, (/**
     * @param {?} state
     * @param {?} __1
     * @return {?}
     */
    (state, { runs }) => {
        /** @type {?} */
        const runIds = Object.keys(runs);
        /** @type {?} */
        const activeRunChanged = runIds.length > 0 && state.activeRunId === null;
        return Object.assign(Object.assign({}, state), { lastNonEmptyPollDataTimeMs: activeRunChanged
                ? Date.now()
                : state.lastNonEmptyPollDataTimeMs, runs, runsLoaded: {
                state: DataLoadState.LOADED,
                lastLoadedTimeInMs: Date.now(),
            }, activeRunId: runIds.length > 0 ? runIds[0] : null });
    })), on(debuggerDataPollOnset, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return Object.assign(Object.assign({}, state), { lastDataPollOnsetTimeMs: Date.now() });
    })), on(numAlertsAndBreakdownRequested, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        /** @type {?} */
        const runId = state.activeRunId;
        if (runId === null) {
            return state;
        }
        return Object.assign(Object.assign({}, state), { alerts: Object.assign(Object.assign({}, state.alerts), { alertsLoaded: Object.assign(Object.assign({}, state.alerts.alertsLoaded), { state: DataLoadState.LOADING }) }) });
    })), on(numAlertsAndBreakdownLoaded, (/**
     * @param {?} state
     * @param {?} __1
     * @return {?}
     */
    (state, { numAlerts, alertsBreakdown }) => {
        /** @type {?} */
        const runId = state.activeRunId;
        if (runId === null) {
            return state;
        }
        /** @type {?} */
        const numAlertsIncreased = numAlerts > state.alerts.numAlerts;
        return Object.assign(Object.assign({}, state), { lastNonEmptyPollDataTimeMs: numAlertsIncreased
                ? Date.now()
                : state.lastNonEmptyPollDataTimeMs, alerts: Object.assign(Object.assign({}, state.alerts), { alertsLoaded: Object.assign(Object.assign({}, state.alerts.alertsLoaded), { state: DataLoadState.LOADED, lastLoadedTimeInMs: Date.now() }), numAlerts,
                alertsBreakdown }) });
    })), on(alertsOfTypeLoaded, (/**
     * @param {?} state
     * @param {?} __1
     * @return {?}
     */
    (state, { numAlerts, alertsBreakdown, alertType, begin, alerts }) => {
        /** @type {?} */
        const runId = state.activeRunId;
        if (runId === null) {
            return state;
        }
        /** @type {?} */
        const updatedAlerts = {};
        /** @type {?} */
        const executionIndices = state.alerts.executionIndices[alertType]
            ? state.alerts.executionIndices[alertType].slice()
            : [];
        /** @type {?} */
        const graphExecutionIndices = state.alerts
            .graphExecutionIndices[alertType]
            ? state.alerts.graphExecutionIndices[alertType].slice()
            : [];
        for (let i = 0; i < alerts.length; ++i) {
            /** @type {?} */
            const alertIndex = begin + i;
            /** @type {?} */
            const alert = alerts[i];
            updatedAlerts[alertIndex] = alert;
            if (alert.alert_type === AlertType.INF_NAN_ALERT) {
                // TOOD(cais): Deal with other alert types with execution index.
                /** @type {?} */
                const infNanAlert = (/** @type {?} */ (alert));
                executionIndices[alertIndex] = infNanAlert.execution_index;
                if (infNanAlert.graph_execution_trace_index !== null) {
                    graphExecutionIndices[alertIndex] =
                        infNanAlert.graph_execution_trace_index;
                }
            }
        }
        if (state.alerts.alerts[alertType] !== undefined) {
            Object.assign(updatedAlerts, state.alerts.alerts[alertType]);
        }
        /** @type {?} */
        let scrollBeginIndex = state.executions.scrollBeginIndex;
        /** @type {?} */
        let graphExecutionFocusIndex = state.graphExecutions.focusIndex;
        if (alertType === AlertType.INF_NAN_ALERT && begin === 0) {
            // TOOD(cais): Deal with other alert types with execution index.
            /** @type {?} */
            const alert = (/** @type {?} */ (alerts[0]));
            /** @type {?} */
            const executionIndex = alert.execution_index;
            // Try to scroll the first alert to the center of the view.
            scrollBeginIndex = Math.max(0, executionIndex - Math.floor(state.executions.displayCount / 2));
            if (alert.graph_execution_trace_index !== null) {
                graphExecutionFocusIndex = alert.graph_execution_trace_index;
            }
        }
        return Object.assign(Object.assign({}, state), { executions: Object.assign(Object.assign({}, state.executions), { scrollBeginIndex }), graphExecutions: Object.assign(Object.assign({}, state.graphExecutions), { focusIndex: graphExecutionFocusIndex }), alerts: Object.assign(Object.assign({}, state.alerts), { alertsLoaded: Object.assign(Object.assign({}, state.alerts.alertsLoaded), { state: DataLoadState.LOADED, lastLoadedTimeInMs: Date.now() }), numAlerts,
                alertsBreakdown, alerts: Object.assign(Object.assign({}, state.alerts.alerts), { [alertType]: updatedAlerts }), executionIndices: Object.assign(Object.assign({}, state.alerts.executionIndices), { [alertType]: executionIndices }), graphExecutionIndices: Object.assign(Object.assign({}, state.alerts.graphExecutionIndices), { [alertType]: graphExecutionIndices }) }) });
    })), on(alertTypeFocusToggled, (/**
     * @param {?} state
     * @param {?} __1
     * @return {?}
     */
    (state, { alertType }) => {
        /** @type {?} */
        const newState = Object.assign(Object.assign({}, state), { alerts: Object.assign(Object.assign({}, state.alerts), { focusType: state.alerts.focusType === alertType ? null : alertType }) });
        // If alert data is available, focus onto the execution digest that
        // corresponds to the first alert.
        /** @type {?} */
        const currentFocusType = newState.alerts.focusType;
        if (currentFocusType !== null) {
            /** @type {?} */
            const executionIndices = newState.alerts.executionIndices[currentFocusType] || [];
            // Try to put the execution digest that corresponds to the first
            // alert at the center of the view.
            if (executionIndices[0] !== undefined) {
                newState.executions.scrollBeginIndex = Math.max(0, Number(executionIndices[0]) -
                    Math.floor(newState.executions.displayCount / 2));
            }
        }
        return newState;
    })), 
    //////////////////////////////////////////////
    // Reducers related to top-level execution. //
    //////////////////////////////////////////////
    on(numExecutionsRequested, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        /** @type {?} */
        const runId = state.activeRunId;
        if (runId === null) {
            return state;
        }
        return Object.assign(Object.assign({}, state), { executions: Object.assign(Object.assign({}, state.executions), { numExecutionsLoaded: Object.assign(Object.assign({}, state.executions.numExecutionsLoaded), { state: DataLoadState.LOADING }) }) });
    })), on(numExecutionsLoaded, (/**
     * @param {?} state
     * @param {?} __1
     * @return {?}
     */
    (state, { numExecutions }) => {
        /** @type {?} */
        const runId = state.activeRunId;
        if (runId === null) {
            return state;
        }
        /** @type {?} */
        const numExecutionsIncreased = numExecutions > state.executions.executionDigestsLoaded.numExecutions;
        /** @type {?} */
        const newState = Object.assign(Object.assign({}, state), { lastNonEmptyPollDataTimeMs: numExecutionsIncreased
                ? Date.now()
                : state.lastNonEmptyPollDataTimeMs, executions: Object.assign(Object.assign({}, state.executions), { numExecutionsLoaded: Object.assign(Object.assign({}, state.executions.numExecutionsLoaded), { state: DataLoadState.LOADED, lastLoadedTimeInMs: Date.now() }), executionDigestsLoaded: Object.assign(Object.assign({}, state.executions.executionDigestsLoaded), { numExecutions }) }) });
        if (numExecutions > 0 && state.executions.focusIndex === null) {
            newState.executions.focusIndex = 0;
        }
        return newState;
    })), on(executionDigestsRequested, (/**
     * @param {?} state
     * @param {?} range
     * @return {?}
     */
    (state, range) => {
        /** @type {?} */
        const runId = state.activeRunId;
        if (runId === null) {
            return state;
        }
        /** @type {?} */
        const loadingRanges = [
            ...state.executions.executionDigestsLoaded.loadingRanges,
        ];
        /** @type {?} */
        const match = findBeginEndRangeIndex(loadingRanges, range.begin, range.end);
        if (match === -1) {
            loadingRanges.push({ begin: range.begin, end: range.end });
        }
        /** @type {?} */
        const newState = Object.assign(Object.assign({}, state), { executions: Object.assign(Object.assign({}, state.executions), { executionDigestsLoaded: Object.assign(Object.assign({}, state.executions.executionDigestsLoaded), { loadingRanges }) }) });
        return newState;
    })), on(executionDigestsLoaded, (/**
     * @param {?} state
     * @param {?} digests
     * @return {?}
     */
    (state, digests) => {
        /** @type {?} */
        const runId = state.activeRunId;
        if (runId === null) {
            return state;
        }
        /** @type {?} */
        const loadingRanges = [
            ...state.executions.executionDigestsLoaded.loadingRanges,
        ];
        /** @type {?} */
        const matchIndex = findBeginEndRangeIndex(loadingRanges, digests.begin, digests.end);
        if (matchIndex !== -1) {
            loadingRanges.splice(matchIndex, 1);
        }
        /** @type {?} */
        const newState = Object.assign(Object.assign({}, state), { executions: Object.assign(Object.assign({}, state.executions), { executionDigestsLoaded: Object.assign(Object.assign({}, state.executions.executionDigestsLoaded), { numExecutions: digests.num_digests, loadingRanges }), executionDigests: Object.assign({}, state.executions.executionDigests) }) });
        for (let i = digests.begin; i < digests.end; ++i) {
            newState.executions.executionDigests[i] =
                digests.execution_digests[i - digests.begin];
        }
        // Update pagesLoadedInFull.
        if (digests.end > digests.begin) {
            /** @type {?} */
            const pageIndex = digests.begin / state.executions.pageSize;
            newState.executions.executionDigestsLoaded.pageLoadedSizes = Object.assign(Object.assign({}, newState.executions.executionDigestsLoaded.pageLoadedSizes), { [pageIndex]: digests.end - digests.begin });
        }
        return newState;
    })), on(executionScrollLeft, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        // TODO(cais): Left-right navigation should have more context-depedent
        // behavior, e.g., when alerts are present.
        /** @type {?} */
        const runId = state.activeRunId;
        if (runId === null) {
            return state;
        }
        /** @type {?} */
        let scrollBeginIndex = state.executions.scrollBeginIndex;
        if (scrollBeginIndex > 0) {
            scrollBeginIndex--;
        }
        return Object.assign(Object.assign({}, state), { executions: Object.assign(Object.assign({}, state.executions), { scrollBeginIndex }) });
    })), on(executionScrollRight, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        // TODO(cais): Left-right navigation should have more context-depedent
        // behavior, e.g., when alerts are present.
        /** @type {?} */
        const runId = state.activeRunId;
        if (runId === null) {
            return state;
        }
        /** @type {?} */
        let scrollBeginIndex = state.executions.scrollBeginIndex;
        if (scrollBeginIndex + state.executions.displayCount + 1 <=
            state.executions.executionDigestsLoaded.numExecutions) {
            scrollBeginIndex++;
        }
        return Object.assign(Object.assign({}, state), { executions: Object.assign(Object.assign({}, state.executions), { scrollBeginIndex }) });
    })), on(executionScrollToIndex, (/**
     * @param {?} state
     * @param {?} action
     * @return {?}
     */
    (state, action) => {
        if (action.index < 0 || !Number.isInteger(action.index)) {
            throw new Error(`Attempt to scroll to negative or non-integer execution index ` +
                `(${action.index})`);
        }
        const { displayCount } = state.executions;
        const { numExecutions } = state.executions.executionDigestsLoaded;
        if (action.index > Math.max(0, numExecutions - displayCount)) {
            throw new Error(`Attempt to scroll to execution index (${action.index}), ` +
                `which exceeds maximum allowed index ` +
                `(numExecutions=${numExecutions}; displayCount=${displayCount})`);
        }
        return Object.assign(Object.assign({}, state), { executions: Object.assign(Object.assign({}, state.executions), { scrollBeginIndex: action.index }) });
    })), on(executionDigestFocused, (/**
     * @param {?} state
     * @param {?} action
     * @return {?}
     */
    (state, action) => {
        /** @type {?} */
        const newState = Object.assign(Object.assign({}, state), { executions: Object.assign(Object.assign({}, state.executions), { focusIndex: state.executions.scrollBeginIndex + action.displayIndex }), codeLocationFocusType: CodeLocationType.EXECUTION, sourceCode: Object.assign({}, state.sourceCode) });
        newState.sourceCode.focusLineSpec = computeBottommostLineSpec(newState);
        return newState;
    })), on(executionDataLoaded, (/**
     * @param {?} state
     * @param {?} data
     * @return {?}
     */
    (state, data) => {
        /** @type {?} */
        const runId = state.activeRunId;
        if (runId === null) {
            return state;
        }
        /** @type {?} */
        const newState = Object.assign(Object.assign({}, state), { executions: Object.assign(Object.assign({}, state.executions), { executionData: Object.assign({}, state.executions.executionData) }) });
        for (let i = data.begin; i < data.end; ++i) {
            newState.executions.executionData[i] = data.executions[i - data.begin];
        }
        return newState;
    })), 
    ////////////////////////////////////////////////
    // Reducers related to intra-graph execution. //
    ////////////////////////////////////////////////
    on(numGraphExecutionsRequested, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        if (state.activeRunId === null) {
            return state;
        }
        return Object.assign(Object.assign({}, state), { graphExecutions: Object.assign(Object.assign({}, state.graphExecutions), { numExecutionsLoaded: Object.assign(Object.assign({}, state.graphExecutions.numExecutionsLoaded), { state: DataLoadState.LOADING }) }) });
    })), on(numGraphExecutionsLoaded, (/**
     * @param {?} state
     * @param {?} __1
     * @return {?}
     */
    (state, { numGraphExecutions }) => {
        if (state.activeRunId === null) {
            return state;
        }
        /** @type {?} */
        const numGraphExecutionsIncreased = numGraphExecutions >
            state.graphExecutions.executionDigestsLoaded.numExecutions;
        /** @type {?} */
        const newState = Object.assign(Object.assign({}, state), { lastNonEmptyPollDataTimeMs: numGraphExecutionsIncreased
                ? Date.now()
                : state.lastNonEmptyPollDataTimeMs, graphExecutions: Object.assign(Object.assign({}, state.graphExecutions), { numExecutionsLoaded: Object.assign(Object.assign({}, state.graphExecutions.numExecutionsLoaded), { state: DataLoadState.LOADED, lastLoadedTimeInMs: Date.now() }), executionDigestsLoaded: Object.assign(Object.assign({}, state.graphExecutions.executionDigestsLoaded), { numExecutions: numGraphExecutions }) }) });
        if (numGraphExecutions > 0 && state.graphExecutions.focusIndex === null) {
            newState.graphExecutions.focusIndex = 0;
        }
        return newState;
    })), on(graphExecutionDataRequested, (/**
     * @param {?} state
     * @param {?} __1
     * @return {?}
     */
    (state, { pageIndex }) => {
        if (state.activeRunId === null) {
            return state;
        }
        /** @type {?} */
        const graphExecutionDataLoadingPages = state.graphExecutions.graphExecutionDataLoadingPages.slice();
        if (graphExecutionDataLoadingPages.indexOf(pageIndex) === -1) {
            graphExecutionDataLoadingPages.push(pageIndex);
        }
        return Object.assign(Object.assign({}, state), { graphExecutions: Object.assign(Object.assign({}, state.graphExecutions), { graphExecutionDataLoadingPages }) });
    })), on(graphExecutionDataLoaded, (/**
     * @param {?} state
     * @param {?} data
     * @return {?}
     */
    (state, data) => {
        if (state.activeRunId === null) {
            return state;
        }
        const { pageSize } = state.graphExecutions;
        /** @type {?} */
        const graphExecutionDataLoadingPages = state.graphExecutions.graphExecutionDataLoadingPages.slice();
        /** @type {?} */
        const graphExecutionDataPageLoadedSizes = Object.assign({}, state.graphExecutions.graphExecutionDataPageLoadedSizes);
        /** @type {?} */
        const graphExecutionData = Object.assign({}, state.graphExecutions.graphExecutionData);
        for (let i = data.begin; i < data.end; ++i) {
            /** @type {?} */
            const pageIndex = Math.floor(i / pageSize);
            if (graphExecutionDataLoadingPages.indexOf(pageIndex) !== -1) {
                graphExecutionDataLoadingPages.splice(graphExecutionDataLoadingPages.indexOf(pageIndex), 1);
            }
            if (graphExecutionDataPageLoadedSizes[pageIndex] === undefined) {
                graphExecutionDataPageLoadedSizes[pageIndex] = 0;
            }
            if (graphExecutionData[i] === undefined) {
                graphExecutionDataPageLoadedSizes[pageIndex]++;
            }
            graphExecutionData[i] = data.graph_executions[i - data.begin];
        }
        return Object.assign(Object.assign({}, state), { graphExecutions: Object.assign(Object.assign({}, state.graphExecutions), { graphExecutionDataLoadingPages,
                graphExecutionDataPageLoadedSizes,
                graphExecutionData }) });
    })), on(graphExecutionScrollToIndex, (/**
     * @param {?} state
     * @param {?} action
     * @return {?}
     */
    (state, action) => {
        if (action.index < 0 || !Number.isInteger(action.index)) {
            throw new Error(`Attempt to scroll to negative or non-integer graph-execution ` +
                `index (${action.index})`);
        }
        return Object.assign(Object.assign({}, state), { graphExecutions: Object.assign(Object.assign({}, state.graphExecutions), { scrollBeginIndex: action.index }) });
    })), on(graphExecutionFocused, (/**
     * @param {?} state
     * @param {?} action
     * @return {?}
     */
    (state, action) => {
        return graphOpFocusReducerHelper(state, action.graph_id, action.op_name, action.index);
    })), 
    ////////////////////////////////////////////////
    // Reducers related to graph structures.      //
    ////////////////////////////////////////////////
    on(graphOpFocused, (/**
     * @param {?} state
     * @param {?} action
     * @return {?}
     */
    (state, action) => {
        return graphOpFocusReducerHelper(state, action.graph_id, action.op_name);
    })), on(graphOpInfoRequested, (/**
     * @param {?} state
     * @param {?} data
     * @return {?}
     */
    (state, data) => {
        const { graph_id, op_name } = data;
        /** @type {?} */
        const newState = Object.assign(Object.assign({}, state), { graphs: Object.assign(Object.assign({}, state.graphs), { loadingOps: Object.assign({}, state.graphs.loadingOps) }) });
        if (newState.graphs.loadingOps[graph_id] === undefined) {
            newState.graphs.loadingOps[graph_id] = new Map();
        }
        if (!newState.graphs.loadingOps[graph_id].has(op_name)) {
            newState.graphs.loadingOps[graph_id].set(op_name, DataLoadState.LOADING);
        }
        return newState;
    })), on(graphOpInfoLoaded, (/**
     * @param {?} state
     * @param {?} data
     * @return {?}
     */
    (state, data) => {
        const { graphOpInfoResponse } = data;
        const { graph_ids } = graphOpInfoResponse;
        /** @type {?} */
        const graphId = graph_ids[graph_ids.length - 1];
        /** @type {?} */
        const newState = Object.assign(Object.assign({}, state), { graphs: Object.assign(Object.assign({}, state.graphs), { ops: Object.assign(Object.assign({}, state.graphs.ops), { [graphId]: new Map(state.graphs.ops[graphId]) }), loadingOps: Object.assign(Object.assign({}, state.graphs.loadingOps), { [graphId]: new Map(state.graphs.loadingOps[graphId]) }) }) });
        for (const input of graphOpInfoResponse.inputs) {
            if (!input.data) {
                // `input.data` can be undefined when the backend fails to look up
                // detailed information regarding the input op (e.g., for certain
                // TF-internal ops such as StatefulPartitionedCall).
                // Same for `consumer.data` below.
                continue;
            }
            newState.graphs.ops[graphId].set(input.op_name, input.data);
        }
        for (let i = 0; i < graphOpInfoResponse.consumers.length; ++i) {
            for (const consumer of graphOpInfoResponse.consumers[i]) {
                if (!consumer.data) {
                    continue;
                }
                newState.graphs.ops[graphId].set(consumer.op_name, consumer.data);
            }
        }
        newState.graphs.ops[graphId].set(graphOpInfoResponse.op_name, Object.assign(Object.assign({}, graphOpInfoResponse), { inputs: graphOpInfoResponse.inputs.map((/**
             * @param {?} input
             * @return {?}
             */
            (input) => ({
                op_name: input.op_name,
                output_slot: input.output_slot,
            }))), consumers: graphOpInfoResponse.consumers.map((/**
             * @param {?} slotConsumers
             * @return {?}
             */
            (slotConsumers) => {
                return slotConsumers.map((/**
                 * @param {?} consumer
                 * @return {?}
                 */
                (consumer) => ({
                    op_name: consumer.op_name,
                    input_slot: consumer.input_slot,
                })));
            })) }));
        // Remove the loading marker for the op.
        newState.graphs.loadingOps[graphId].set(graphOpInfoResponse.op_name, DataLoadState.LOADED);
        return newState;
    })), 
    ////////////////////////////////////////////////////////
    // Reducers related to source files and stack traces. //
    ////////////////////////////////////////////////////////
    on(sourceFileListRequested, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return Object.assign(Object.assign({}, state), { sourceCode: Object.assign(Object.assign({}, state.sourceCode), { sourceFileListLoaded: Object.assign(Object.assign({}, state.sourceCode.sourceFileListLoaded), { state: DataLoadState.LOADING }) }) });
    })), on(sourceFileListLoaded, (/**
     * @param {?} state
     * @param {?} sourceFileList
     * @return {?}
     */
    (state, sourceFileList) => {
        /** @type {?} */
        const newState = Object.assign(Object.assign({}, state), { sourceCode: Object.assign(Object.assign({}, state.sourceCode), { sourceFileListLoaded: Object.assign(Object.assign({}, state.sourceCode.sourceFileListLoaded), { state: DataLoadState.LOADED, lastLoadedTimeInMs: Date.now() }), sourceFileList: sourceFileList.sourceFiles, fileContents: state.sourceCode.fileContents.slice() }) });
        /** @type {?} */
        const newNumFiles = sourceFileList.sourceFiles.length;
        const { fileContents } = newState.sourceCode;
        for (let i = 0; i < newNumFiles; ++i) {
            fileContents[i] = state.sourceCode.fileContents[i] || {
                loadState: DataLoadState.NOT_LOADED,
                lines: null,
            };
        }
        return newState;
    })), on(sourceLineFocused, (/**
     * @param {?} state
     * @param {?} focus
     * @return {?}
     */
    (state, focus) => {
        /** @type {?} */
        const focusedStackTrace = getFocusedStackFramesHelper(state);
        /** @type {?} */
        const newState = Object.assign(Object.assign({}, state), { sourceCode: Object.assign(Object.assign({}, state.sourceCode), { focusLineSpec: focus.stackFrame }) });
        if (focusedStackTrace !== null) {
            newState.stickToBottommostFrameInFocusedFile = isFrameBottommostInStackTrace(focusedStackTrace, focus.stackFrame);
        }
        return newState;
    })), on(sourceFileRequested, (/**
     * @param {?} state
     * @param {?} sourceFileSpec
     * @return {?}
     */
    (state, sourceFileSpec) => {
        /** @type {?} */
        const newState = Object.assign(Object.assign({}, state), { sourceCode: Object.assign(Object.assign({}, state.sourceCode), { fileContents: state.sourceCode.fileContents.slice() }) });
        /** @type {?} */
        const fileIndex = findFileIndex(newState.sourceCode.sourceFileList, sourceFileSpec);
        if (fileIndex >= 0) {
            newState.sourceCode.fileContents[fileIndex].loadState =
                DataLoadState.LOADING;
        }
        else {
            throw new Error(`Cannot find the following file in file list: ` +
                `host_name="${sourceFileSpec.host_name}", ` +
                `file_path="${sourceFileSpec.file_path}"`);
        }
        return newState;
    })), on(sourceFileLoaded, (/**
     * @param {?} state
     * @param {?} sourceFileResponse
     * @return {?}
     */
    (state, sourceFileResponse) => {
        /** @type {?} */
        const newState = Object.assign(Object.assign({}, state), { sourceCode: Object.assign(Object.assign({}, state.sourceCode), { fileContents: state.sourceCode.fileContents.slice() }) });
        /** @type {?} */
        const fileIndex = findFileIndex(newState.sourceCode.sourceFileList, sourceFileResponse);
        if (fileIndex >= 0) {
            newState.sourceCode.fileContents[fileIndex] = {
                loadState: DataLoadState.LOADED,
                lines: sourceFileResponse.lines,
            };
        }
        else {
            throw new Error(`Cannot find the following file in file list: ` +
                `host_name="${sourceFileResponse.host_name}", ` +
                `file_path="${sourceFileResponse.file_path}"`);
        }
        return newState;
    })), on(stackFramesLoaded, (/**
     * @param {?} state
     * @param {?} stackFrames
     * @return {?}
     */
    (state, stackFrames) => {
        /** @type {?} */
        const runId = state.activeRunId;
        if (runId === null) {
            return state;
        }
        /** @type {?} */
        const newState = Object.assign(Object.assign({}, state), { stackFrames: Object.assign(Object.assign({}, state.stackFrames), stackFrames.stackFrames), sourceCode: Object.assign({}, state.sourceCode) });
        newState.sourceCode.focusLineSpec = computeBottommostLineSpec(newState);
        return newState;
    })));
    /**
     * @param {?} state
     * @param {?} graphId
     * @param {?} opName
     * @param {?=} graphExecutionIndex
     * @return {?}
     */
    function graphOpFocusReducerHelper(state, graphId, opName, graphExecutionIndex) {
        /** @type {?} */
        const newState = Object.assign(Object.assign({}, state), { graphs: Object.assign(Object.assign({}, state.graphs), { focusedOp: {
                    graphId,
                    opName,
                } }), codeLocationFocusType: CodeLocationType.GRAPH_OP_CREATION, sourceCode: Object.assign({}, state.sourceCode) });
        newState.sourceCode.focusLineSpec = computeBottommostLineSpec(newState);
        if (graphExecutionIndex !== undefined) {
            newState.graphExecutions = Object.assign(Object.assign({}, state.graphExecutions), { focusIndex: graphExecutionIndex });
        }
        return newState;
    }
    // TODO(cais): Refactor the reducers into separate child reducers and
    // move them to separate files. Combine them with `combineReducers()`
    // for better maintainability.
    /**
     * @param {?} state
     * @param {?} action
     * @return {?}
     */
    function reducers$6(state, action) {
        return reducer$4(state, action);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/store/debugger_selectors.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const selectDebuggerState = createFeatureSelector(DEBUGGER_FEATURE_KEY);
    /** @type {?} */
    const getDebuggerRunListing = createSelector(selectDebuggerState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return state.runs;
    }));
    /** @type {?} */
    const getDebuggerRunsLoaded = createSelector(selectDebuggerState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => state.runsLoaded));
    /** @type {?} */
    const getActiveRunId = createSelector(selectDebuggerState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => state.activeRunId));
    /**
     * Get the time elapsed from the last time at which a data polling
     * yielded new data and the last polling time, in milliseconds.
     * That is: this is how long polling has yielded no new data.
     * @type {?}
     */
    const getPollSilenceTimeMs = createSelector(selectDebuggerState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return state.lastDataPollOnsetTimeMs - state.lastNonEmptyPollDataTimeMs;
    }));
    /**
     * Intermediate selector for alerts.
     * @type {?}
     */
    const selectAlerts = createSelector(selectDebuggerState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => state.alerts));
    /** @type {?} */
    const getAlertsLoaded = createSelector(selectAlerts, (/**
     * @param {?} alerts
     * @return {?}
     */
    (alerts) => {
        return alerts.alertsLoaded;
    }));
    /** @type {?} */
    const getNumAlerts = createSelector(selectAlerts, (/**
     * @param {?} alerts
     * @return {?}
     */
    (alerts) => {
        return alerts.numAlerts;
    }));
    /** @type {?} */
    const getAlertsFocusType = createSelector(selectAlerts, (/**
     * @param {?} alerts
     * @return {?}
     */
    (alerts) => {
        return alerts.focusType;
    }));
    /**
     * Get number of alerts of the alert type being focused on.
     *
     * If no alert type focus exists, returns 0.
     * The returned number is regardless of whether the detailed Alerts
     * data have been loaded by the front end.
     * @type {?}
     */
    const getNumAlertsOfFocusedType = createSelector(selectAlerts, (/**
     * @param {?} alerts
     * @return {?}
     */
    (alerts) => {
        if (alerts.focusType === null) {
            return 0;
        }
        return alerts.alertsBreakdown[alerts.focusType] || 0;
    }));
    /**
     * Get the Alerts that are 1) of the type being focused on, and
     * 2) already loaded by the front end.
     *
     * If no alert type focus exists, returns null.
     * @type {?}
     */
    const getLoadedAlertsOfFocusedType = createSelector(selectAlerts, (/**
     * @param {?} alerts
     * @return {?}
     */
    (alerts) => {
        if (alerts.focusType === null) {
            return null;
        }
        if (alerts.alerts[alerts.focusType] === undefined) {
            return null;
        }
        return alerts.alerts[alerts.focusType];
    }));
    /** @type {?} */
    const getAlertsBreakdown = createSelector(selectAlerts, (/**
     * @param {?} alerts
     * @return {?}
     */
    (alerts) => {
        return alerts.alertsBreakdown;
    }));
    /**
     * Intermeidate selector for executions.
     * @type {?}
     */
    const selectExecutionsState = createSelector(selectDebuggerState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => state.executions));
    /** @type {?} */
    const getNumExecutionsLoaded = createSelector(selectExecutionsState, (/**
     * @param {?} executions
     * @return {?}
     */
    (executions) => {
        return executions.numExecutionsLoaded;
    }));
    /** @type {?} */
    const getExecutionDigestsLoaded = createSelector(selectExecutionsState, (/**
     * @param {?} executions
     * @return {?}
     */
    (executions) => {
        return executions.executionDigestsLoaded;
    }));
    /** @type {?} */
    const getNumExecutions = createSelector(selectExecutionsState, (/**
     * @param {?} executions
     * @return {?}
     */
    (executions) => {
        return executions.executionDigestsLoaded.numExecutions;
    }));
    /** @type {?} */
    const getExecutionScrollBeginIndex = createSelector(selectExecutionsState, (/**
     * @param {?} executions
     * @return {?}
     */
    (executions) => {
        return executions.scrollBeginIndex;
    }));
    /** @type {?} */
    const getExecutionPageSize = createSelector(selectExecutionsState, (/**
     * @param {?} executions
     * @return {?}
     */
    (executions) => {
        return executions.pageSize;
    }));
    /** @type {?} */
    const getDisplayCount = createSelector(selectExecutionsState, (/**
     * @param {?} executions
     * @return {?}
     */
    (executions) => {
        return executions.displayCount;
    }));
    /** @type {?} */
    const getVisibleExecutionDigests = createSelector(selectExecutionsState, (/**
     * @param {?} executions
     * @return {?}
     */
    (executions) => {
        /** @type {?} */
        const digests = [];
        for (let executionIndex = executions.scrollBeginIndex; executionIndex < executions.scrollBeginIndex + executions.displayCount; ++executionIndex) {
            if (executionIndex in executions.executionDigests) {
                digests.push(executions.executionDigests[executionIndex]);
            }
            else {
                digests.push(null);
            }
        }
        return digests;
    }));
    /**
     * Intermediate selector for alerts.
     * @type {?}
     */
    const selectGraphExecutions = createSelector(selectDebuggerState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => state.graphExecutions));
    /** @type {?} */
    const getNumGraphExecutionsLoaded = createSelector(selectGraphExecutions, (/**
     * @param {?} graphExecutions
     * @return {?}
     */
    (graphExecutions) => {
        return graphExecutions.numExecutionsLoaded;
    }));
    /** @type {?} */
    const getNumGraphExecutions = createSelector(selectDebuggerState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return state.graphExecutions.executionDigestsLoaded.numExecutions;
    }));
    /** @type {?} */
    const getGraphExecutionScrollBeginIndex = createSelector(selectGraphExecutions, (/**
     * @param {?} graphExecutions
     * @return {?}
     */
    (graphExecutions) => {
        return graphExecutions.scrollBeginIndex;
    }));
    /** @type {?} */
    const getGraphExecutionDisplayCount = createSelector(selectGraphExecutions, (/**
     * @param {?} graphExecutions
     * @return {?}
     */
    (graphExecutions) => {
        return graphExecutions.displayCount;
    }));
    /** @type {?} */
    const getGraphExecutionPageSize = createSelector(selectGraphExecutions, (/**
     * @param {?} graphExecutions
     * @return {?}
     */
    (graphExecutions) => {
        return graphExecutions.pageSize;
    }));
    /** @type {?} */
    const getGraphExecutionDataLoadingPages = createSelector(selectGraphExecutions, (/**
     * @param {?} graphExecutions
     * @return {?}
     */
    (graphExecutions) => {
        return graphExecutions.graphExecutionDataLoadingPages;
    }));
    /** @type {?} */
    const getGraphExecutionDataPageLoadedSizes = createSelector(selectGraphExecutions, (/**
     * @param {?} graphExecutions
     * @return {?}
     */
    (graphExecutions) => {
        return graphExecutions.graphExecutionDataPageLoadedSizes;
    }));
    /** @type {?} */
    const getGraphExecutionData = createSelector(selectGraphExecutions, (/**
     * @param {?} graphExecutions
     * @return {?}
     */
    (graphExecutions) => {
        return graphExecutions.graphExecutionData;
    }));
    /** @type {?} */
    const getGraphExecutionFocusIndex = createSelector(selectGraphExecutions, (/**
     * @param {?} graphExecutions
     * @return {?}
     */
    (graphExecutions) => {
        return graphExecutions.focusIndex;
    }));
    /**
     * Intermediate selector for the graphs state of debugger.
     * @type {?}
     */
    const selectDebuggerGraphs = createSelector(selectDebuggerState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => state.graphs));
    /** @type {?} */
    const getFocusedGraphOpInfo = createSelector(selectDebuggerGraphs, (/**
     * @param {?} graphs
     * @return {?}
     */
    (graphs) => {
        const { focusedOp, ops } = graphs;
        if (focusedOp === null || ops[focusedOp.graphId] === undefined) {
            return null;
        }
        else {
            return ops[focusedOp.graphId].get(focusedOp.opName) || null;
        }
    }));
    /** @type {?} */
    const getFocusedGraphOpInputs = createSelector(selectDebuggerGraphs, (/**
     * @param {?} graphs
     * @return {?}
     */
    (graphs) => {
        const { focusedOp, ops } = graphs;
        if (focusedOp === null ||
            ops[focusedOp.graphId] === undefined ||
            !ops[focusedOp.graphId].has(focusedOp.opName)) {
            return null;
        }
        else {
            /** @type {?} */
            const graph = ops[focusedOp.graphId];
            const { inputs } = (/** @type {?} */ (graph.get(focusedOp.opName)));
            return inputs.map((/**
             * @param {?} inputSpec
             * @return {?}
             */
            (inputSpec) => {
                /** @type {?} */
                const spec = Object.assign({}, inputSpec);
                if (graph.has(inputSpec.op_name)) {
                    spec.data = graph.get(inputSpec.op_name);
                }
                return spec;
            }));
        }
    }));
    // How many graph-execution indices at most to look back in order to find the
    // indices of the graph-execution events that consistute the immediate input to
    // the currently focused graph-execution event.
    /** @type {?} */
    const MAX_LOOK_BACK = 200;
    /**
     * Makes best-effort attempt to find the graph-execution indices that
     * constitutes the immediate inputs to the currently-focused graph-execution
     * event.
     *
     * If no graph-execution event is currently focused on, returns `null`.
     * It may skip indices if some of the data for the graph-execution events that
     * are immediate inputs has not been loaded yet.
     * If the currently-focused graph-execution events is a no-input op, returns
     * an empty array.
     * If the immediate input events are more than `MAX_LOOK_BACK` before the
     * currently focused index (rare), the returned array may be incomplete.
     * @type {?}
     */
    const getFocusedGraphExecutionInputIndices = createSelector(getGraphExecutionFocusIndex, getGraphExecutionData, getFocusedGraphOpInputs, (/**
     * @param {?} focusIndex
     * @param {?} data
     * @param {?} opInputs
     * @return {?}
     */
    (focusIndex, data, opInputs) => {
        if (focusIndex === null || opInputs === null) {
            return null;
        }
        /** @type {?} */
        const inputFound = opInputs.map((/**
         * @param {?} _
         * @return {?}
         */
        (_) => false));
        /** @type {?} */
        const inputIndices = [];
        if (opInputs.length === 0) {
            return inputIndices;
        }
        /** @type {?} */
        const graph_id = data[focusIndex].graph_id;
        /** @type {?} */
        const limit = Math.max(0, focusIndex - MAX_LOOK_BACK);
        for (let i = focusIndex - 1; i >= limit; --i) {
            if (data[i] === undefined) {
                continue;
            }
            for (let j = 0; j < opInputs.length; ++j) {
                if (!inputFound[j] &&
                    data[i].graph_id === graph_id &&
                    data[i].op_name === opInputs[j].op_name &&
                    data[i].output_slot === opInputs[j].output_slot) {
                    inputIndices.push(i);
                    inputFound[j] = true;
                    if (inputIndices.length === opInputs.length) {
                        break;
                    }
                }
            }
        }
        return inputIndices;
    }));
    /** @type {?} */
    const getFocusedGraphOpConsumers = createSelector(selectDebuggerGraphs, (/**
     * @param {?} graphs
     * @return {?}
     */
    (graphs) => {
        const { focusedOp, ops } = graphs;
        if (focusedOp === null ||
            ops[focusedOp.graphId] === undefined ||
            !ops[focusedOp.graphId].has(focusedOp.opName)) {
            return null;
        }
        else {
            /** @type {?} */
            const graph = ops[focusedOp.graphId];
            const { consumers } = (/** @type {?} */ (graph.get(focusedOp.opName)));
            return consumers.map((/**
             * @param {?} slotConsumers
             * @return {?}
             */
            (slotConsumers) => {
                return slotConsumers.map((/**
                 * @param {?} consumerSpec
                 * @return {?}
                 */
                (consumerSpec) => {
                    /** @type {?} */
                    const spec = Object.assign({}, consumerSpec);
                    if (graph.has(consumerSpec.op_name)) {
                        spec.data = graph.get(consumerSpec.op_name);
                    }
                    return spec;
                }));
            }));
        }
    }));
    /**
     * Get the focused alert types (if any) of the execution digests current being
     * displayed. For each displayed execution digest, there are two possibilities:
     * - `null` represents no alert.
     * - An instance of the `AlertType`
     * @type {?}
     */
    const getFocusAlertTypesOfVisibleExecutionDigests = createSelector(selectDebuggerState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        /** @type {?} */
        const beginExecutionIndex = state.executions.scrollBeginIndex;
        /** @type {?} */
        const endExecutionIndex = state.executions.scrollBeginIndex + state.executions.displayCount;
        /** @type {?} */
        const alertTypes = new Array(endExecutionIndex - beginExecutionIndex).fill(null);
        /** @type {?} */
        const focusType = state.alerts.focusType;
        if (focusType === null) {
            return alertTypes;
        }
        /** @type {?} */
        const executionIndices = state.alerts.executionIndices[focusType];
        if (executionIndices === undefined) {
            return alertTypes;
        }
        // TODO(cais): Explore using a Set for execution indices if this
        // part becomes a performance bottleneck in the future.
        for (let i = beginExecutionIndex; i < endExecutionIndex; ++i) {
            if (executionIndices.includes(i)) {
                alertTypes[i - beginExecutionIndex] = state.alerts.focusType;
            }
        }
        return alertTypes;
    }));
    /**
     * Intermediate selector for top-level executions.
     * @type {?}
     */
    const selectExecutions = createSelector(selectDebuggerState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => state.executions));
    /** @type {?} */
    const getFocusedExecutionIndex = createSelector(selectExecutions, (/**
     * @param {?} executions
     * @return {?}
     */
    (executions) => {
        return executions.focusIndex;
    }));
    /**
     * Get the display index of the execution digest being focused on (if any).
     * @type {?}
     */
    const getFocusedExecutionDisplayIndex = createSelector(selectExecutions, (/**
     * @param {?} executions
     * @return {?}
     */
    (executions) => {
        if (executions.focusIndex === null) {
            return null;
        }
        const { focusIndex, scrollBeginIndex, displayCount } = executions;
        if (focusIndex < scrollBeginIndex ||
            focusIndex >= scrollBeginIndex + displayCount) {
            return null;
        }
        return focusIndex - scrollBeginIndex;
    }));
    /** @type {?} */
    const getLoadedExecutionData = createSelector(selectExecutions, (/**
     * @param {?} executions
     * @return {?}
     */
    (executions) => executions.executionData));
    /** @type {?} */
    const getLoadingGraphOps = createSelector(selectDebuggerState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => state.graphs.loadingOps));
    /** @type {?} */
    const getLoadedStackFrames = createSelector(selectDebuggerState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => state.stackFrames));
    /** @type {?} */
    const getFocusedExecutionData = createSelector(selectExecutions, (/**
     * @param {?} executions
     * @return {?}
     */
    (executions) => {
        const { focusIndex, executionData } = executions;
        if (focusIndex === null || executionData[focusIndex] === undefined) {
            return null;
        }
        return executionData[focusIndex];
    }));
    /**
     * Get information regarding the op that's the origin of the focused
     * code location (stack trace).
     * This selector covers both eager execution and graph-op creation.
     * @type {?}
     */
    const getCodeLocationOrigin = createSelector(selectDebuggerState, getFocusedExecutionIndex, getFocusedExecutionData, getFocusedGraphOpInfo, (/**
     * @param {?} state
     * @param {?} executionIndex
     * @param {?} executionData
     * @param {?} graphOpInfo
     * @return {?}
     */
    (state, executionIndex, executionData, graphOpInfo) => {
        const { codeLocationFocusType } = state;
        if (codeLocationFocusType === null) {
            return null;
        }
        if (codeLocationFocusType === CodeLocationType.EXECUTION) {
            if (executionIndex === null || executionData === null) {
                return null;
            }
            return {
                codeLocationType: CodeLocationType.EXECUTION,
                opType: executionData.op_type,
                executionIndex,
            };
        }
        else {
            // This is CodeLocationType.GRAPH_OP_CREATION.
            if (graphOpInfo === null) {
                return null;
            }
            return {
                codeLocationType: CodeLocationType.GRAPH_OP_CREATION,
                opType: graphOpInfo.op_type,
                opName: graphOpInfo.op_name,
            };
        }
    }));
    /**
     * Get the stack trace (frames) of the execution event currently focused on
     * (if any).
     *
     * If no execution is focused on, returns null.
     * If any of the stack frames is missing (i.e., hasn't been loaded from
     * the data source yet), returns null.
     * @type {?}
     */
    const getFocusedStackFrames = createSelector(
    // TODO(cais): Rename this function as `getFocusedStackTrace()` to
    // minimize confusion with `getFocusedSourceLineSpec()`.
    selectDebuggerState, getFocusedStackFramesHelper);
    /**
     * Intermediate selector for source code.
     * @type {?}
     */
    const selectSourceCode = createSelector(selectDebuggerState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => state.sourceCode));
    /** @type {?} */
    const getSourceFileListLoaded = createSelector(selectSourceCode, (/**
     * @param {?} sourceCode
     * @return {?}
     */
    (sourceCode) => {
        return sourceCode.sourceFileListLoaded;
    }));
    /** @type {?} */
    const getSourceFileList = createSelector(selectSourceCode, (/**
     * @param {?} sourceCode
     * @return {?}
     */
    (sourceCode) => {
        return sourceCode.sourceFileList;
    }));
    /** @type {?} */
    const getFocusedSourceFileIndex = createSelector(selectSourceCode, (/**
     * @param {?} sourceCode
     * @return {?}
     */
    (sourceCode) => {
        const { sourceFileList, focusLineSpec } = sourceCode;
        if (focusLineSpec === null) {
            return -1;
        }
        return findFileIndex(sourceFileList, focusLineSpec);
    }));
    /** @type {?} */
    const getFocusedSourceFileContent = createSelector(selectSourceCode, getFocusedSourceFileIndex, (/**
     * @param {?} sourceCode
     * @param {?} fileIndex
     * @return {?}
     */
    (sourceCode, fileIndex) => {
        if (fileIndex === -1) {
            return null;
        }
        return sourceCode.fileContents[fileIndex] || null;
    }));
    /**
     * Get the source-code line being focused on.
     *
     * If the `stickingToBottommostFrameInFocusedFile` state is `true` and
     * `focusedLIneSpec` is not null, this selector will return the bottommost
     * stack frame in the file in `focusedLIneSpec`.
     * Else, it'll directly return the value of `focusedLIneSpec`.
     *
     * This selector allows the UI to "track" lines in a source file of interest
     * as a user navigates executions or graph ops.
     * @type {?}
     */
    const getFocusedSourceLineSpec = createSelector(selectDebuggerState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => state.sourceCode.focusLineSpec));
    /** @type {?} */
    const getStickToBottommostFrameInFocusedFile = createSelector(selectDebuggerState, (/**
     * @param {?} debuggerState
     * @return {?}
     */
    (debuggerState) => {
        return debuggerState.stickToBottommostFrameInFocusedFile;
    }));

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/views/alerts/alerts_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    const _c0$b = function (a0) { return [a0]; };
    function AlertsComponent_div_9_Template(rf, ctx) { if (rf & 1) {
        const _r3 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "div", 7);
        ɵɵlistener("click", function AlertsComponent_div_9_Template_div_click_0_listener() { ɵɵrestoreView(_r3); const breakdown_r1 = ctx.$implicit; const ctx_r2 = ɵɵnextContext(); return ctx_r2.onToggleFocusType.emit(breakdown_r1.type); });
        ɵɵelementStart(1, "div", 8);
        ɵɵtext(2);
        ɵɵelementEnd();
        ɵɵelementStart(3, "div", 9);
        ɵɵtext(4);
        ɵɵelementEnd();
        ɵɵelement(5, "div");
        ɵɵelementEnd();
    } if (rf & 2) {
        const breakdown_r1 = ctx.$implicit;
        const ctx_r0 = ɵɵnextContext();
        ɵɵproperty("ngClass", ɵɵpureFunction1(4, _c0$b, breakdown_r1.type === ctx_r0.focusType ? "focus" : ""));
        ɵɵadvance(2);
        ɵɵtextInterpolate(breakdown_r1.displayName);
        ɵɵadvance(2);
        ɵɵtextInterpolate2(" ", breakdown_r1.displaySymbol, ": ", breakdown_r1.count, " ");
    } }
    class AlertsComponent {
        constructor() {
            // Total number of alerts.
            this.numAlerts = 0;
            this.alertsBreakdown = [];
            this.focusType = null;
            this.onToggleFocusType = new EventEmitter();
        }
    }
    AlertsComponent.decorators = [
        { type: Component, args: [{
                    selector: 'alerts-component',
                    templateUrl: './alerts_component.ng.html',
                    styleUrls: ['./alerts_component.css'],
                },] },
    ];
    AlertsComponent.propDecorators = {
        numAlerts: [{ type: Input }],
        alertsBreakdown: [{ type: Input }],
        focusType: [{ type: Input }],
        onToggleFocusType: [{ type: Output }]
    };
    /** @nocollapse */ AlertsComponent.ɵfac = function AlertsComponent_Factory(t) { return new (t || AlertsComponent)(); };
    /** @nocollapse */ AlertsComponent.ɵcmp = ɵɵdefineComponent({ type: AlertsComponent, selectors: [["alerts-component"]], inputs: { numAlerts: "numAlerts", alertsBreakdown: "alertsBreakdown", focusType: "focusType" }, outputs: { onToggleFocusType: "onToggleFocusType" }, decls: 10, vars: 5, consts: [[1, "alerts-container"], [1, "debugging-title"], [1, "num-alerts-container"], [1, "num-alerts-label"], [1, "num-alerts-value", 3, "ngClass"], [1, "alerts-breakdown-container"], ["class", "alerts-breakdown-type", 3, "ngClass", "click", 4, "ngFor", "ngForOf"], [1, "alerts-breakdown-type", 3, "ngClass", "click"], [1, "alert-type-name"], [1, "alert-type-count"]], template: function AlertsComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "div", 0);
            ɵɵelementStart(1, "div", 1);
            ɵɵtext(2, "Debugging");
            ɵɵelementEnd();
            ɵɵelementStart(3, "div", 2);
            ɵɵelementStart(4, "div", 3);
            ɵɵtext(5, "Alerts");
            ɵɵelementEnd();
            ɵɵelementStart(6, "div", 4);
            ɵɵtext(7);
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementStart(8, "div", 5);
            ɵɵtemplate(9, AlertsComponent_div_9_Template, 6, 6, "div", 6);
            ɵɵelementEnd();
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵadvance(6);
            ɵɵproperty("ngClass", ɵɵpureFunction1(3, _c0$b, ctx.numAlerts > 0 ? "non-zero" : ""));
            ɵɵadvance(1);
            ɵɵtextInterpolate1(" ", ctx.numAlerts, " ");
            ɵɵadvance(2);
            ɵɵproperty("ngForOf", ctx.alertsBreakdown);
        } }, directives: [NgClass, NgForOf], styles: [".alerts-breakdown-container[_ngcontent-%COMP%] {\n  font-size: 13px;\n  padding: 10px 10px 10px;\n  position: relative;\n}\n\n.alerts-breakdown-type[_ngcontent-%COMP%] {\n  border-radius: 0 10px 10px 0;\n  cursor: pointer;\n  display: flex;\n  padding: 6px 0 6px 50px;\n  vertical-align: middle;\n}\n\n.alerts-breakdown-type.focus[_ngcontent-%COMP%] {\n  background-color: #ffeee0;\n}\n\n.alerts-container[_ngcontent-%COMP%] {\n  font-family: 'Roboto', Arial, Helvetica, sans-serif;\n}\n\n.alert-type-count[_ngcontent-%COMP%] {\n  \n  background-color: #e52592;\n  border-radius: 3px;\n  color: #fff;\n  display: inline-block;\n  padding: 3px;\n  position: absolute;\n  right: 20px;\n  vertical-align: middle;\n}\n\n.alert-type-name[_ngcontent-%COMP%] {\n  display: inline-block;\n  padding: 3px;\n  vertical-align: middle;\n}\n\n.debugging-title[_ngcontent-%COMP%] {\n  font-size: 18px;\n}\n\n.num-alerts-container[_ngcontent-%COMP%] {\n  font-weight: bold;\n  padding: 10px 10px 10px 30px;\n  position: relative;\n}\n\n.num-alerts-label[_ngcontent-%COMP%] {\n  display: inline-block;\n  font-size: 13px;\n}\n\n.num-alerts-value[_ngcontent-%COMP%] {\n  border-radius: 12px;\n  display: inline-block;\n  font-size: 13px;\n  font-weight: normal;\n  line-height: 24px;\n  position: absolute;\n  right: 20px;\n  text-align: center;\n  vertical-align: middle;\n  width: 24px;\n}\n\n.num-alerts-value.non-zero[_ngcontent-%COMP%] {\n  background-color: #ffb780;\n  font-weight: bold;\n}"] });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/views/alerts/alerts_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const ALERT_TYPE_TO_DISPLAY_NAME_AND_SYMBOL = {
        [AlertType.FUNCTION_RECOMPILE_ALERT]: {
            displayName: 'Function recompiles',
            displaySymbol: 'C',
        },
        [AlertType.INF_NAN_ALERT]: {
            displayName: 'NaN/∞',
            displaySymbol: '∞',
        },
        [AlertType.TENSOR_SHAPE_ALERT]: {
            displayName: 'Tensor shape',
            displaySymbol: '■',
        },
    };
    class AlertsContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.numAlerts$ = this.store.pipe(select(getNumAlerts));
            this.alertsBreakdown$ = this.store.pipe(select(createSelector(getAlertsBreakdown, (/**
             * @param {?} alertsBreakdown
             * @return {?}
             */
            (alertsBreakdown) => {
                /** @type {?} */
                const alertTypes = Object.keys(alertsBreakdown);
                alertTypes.sort();
                return alertTypes.map((/**
                 * @param {?} alertType
                 * @return {?}
                 */
                (alertType) => {
                    return Object.assign(Object.assign({ type: (/** @type {?} */ (alertType)) }, ALERT_TYPE_TO_DISPLAY_NAME_AND_SYMBOL[alertType]), { count: alertsBreakdown[alertType] });
                }));
            }))));
            this.focusType$ = this.store.pipe(select(getAlertsFocusType));
        }
        /**
         * @param {?} alertType
         * @return {?}
         */
        onToggleFocusType(alertType) {
            this.store.dispatch(alertTypeFocusToggled({ alertType }));
        }
    }
    AlertsContainer.decorators = [
        { type: Component, args: [{
                    selector: 'tf-debugger-v2-alerts',
                    template: `
    <alerts-component
      [numAlerts]="numAlerts$ | async"
      [alertsBreakdown]="alertsBreakdown$ | async"
      [focusType]="focusType$ | async"
      (onToggleFocusType)="onToggleFocusType($event)"
    >
    </alerts-component>
  `,
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    AlertsContainer.ctorParameters = () => [
        { type: Store }
    ];
    /** @nocollapse */ AlertsContainer.ɵfac = function AlertsContainer_Factory(t) { return new (t || AlertsContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ AlertsContainer.ɵcmp = ɵɵdefineComponent({ type: AlertsContainer, selectors: [["tf-debugger-v2-alerts"]], decls: 4, vars: 9, consts: [[3, "numAlerts", "alertsBreakdown", "focusType", "onToggleFocusType"]], template: function AlertsContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "alerts-component", 0);
            ɵɵlistener("onToggleFocusType", function AlertsContainer_Template_alerts_component_onToggleFocusType_0_listener($event) { return ctx.onToggleFocusType($event); });
            ɵɵpipe(1, "async");
            ɵɵpipe(2, "async");
            ɵɵpipe(3, "async");
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵproperty("numAlerts", ɵɵpipeBind1(1, 3, ctx.numAlerts$))("alertsBreakdown", ɵɵpipeBind1(2, 5, ctx.alertsBreakdown$))("focusType", ɵɵpipeBind1(3, 7, ctx.focusType$));
        } }, directives: [AlertsComponent], pipes: [AsyncPipe], encapsulation: 2, changeDetection: 0 });

    const _c0$c = ["sliderWrapper"];
    const activeEventOptions = normalizePassiveListenerOptions({ passive: false });
    /**
     * Visually, a 30px separation between tick marks looks best. This is very subjective but it is
     * the default separation we chose.
     * @type {?}
     */
    const MIN_AUTO_TICK_SEPARATION = 30;
    /**
     * The thumb gap size for a disabled slider.
     * @type {?}
     */
    const DISABLED_THUMB_GAP = 7;
    /**
     * The thumb gap size for a non-active slider at its minimum value.
     * @type {?}
     */
    const MIN_VALUE_NONACTIVE_THUMB_GAP = 7;
    /**
     * The thumb gap size for an active slider at its minimum value.
     * @type {?}
     */
    const MIN_VALUE_ACTIVE_THUMB_GAP = 10;
    /**
     * Provider Expression that allows mat-slider to register as a ControlValueAccessor.
     * This allows it to support [(ngModel)] and [formControl].
     * \@docs-private
     * @type {?}
     */
    const MAT_SLIDER_VALUE_ACCESSOR = {
        provide: NG_VALUE_ACCESSOR,
        useExisting: forwardRef((/**
         * @return {?}
         */
        () => MatSlider)),
        multi: true
    };
    /**
     * A simple change event emitted by the MatSlider component.
     */
    class MatSliderChange {
    }
    // Boilerplate for applying mixins to MatSlider.
    /**
     * \@docs-private
     */
    class MatSliderBase {
        /**
         * @param {?} _elementRef
         */
        constructor(_elementRef) {
            this._elementRef = _elementRef;
        }
    }
    /** @type {?} */
    const _MatSliderMixinBase = mixinTabIndex(mixinColor(mixinDisabled(MatSliderBase), 'accent'));
    /**
     * Allows users to select from a range of values by moving the slider thumb. It is similar in
     * behavior to the native `<input type="range">` element.
     */
    class MatSlider extends _MatSliderMixinBase {
        /**
         * @param {?} elementRef
         * @param {?} _focusMonitor
         * @param {?} _changeDetectorRef
         * @param {?} _dir
         * @param {?} tabIndex
         * @param {?=} _animationMode
         * @param {?=} _ngZone
         * @param {?=} document
         */
        constructor(elementRef, _focusMonitor, _changeDetectorRef, _dir, tabIndex, _animationMode, _ngZone, 
        /** @breaking-change 11.0.0 make document required */
        document) {
            super(elementRef);
            this._focusMonitor = _focusMonitor;
            this._changeDetectorRef = _changeDetectorRef;
            this._dir = _dir;
            this._animationMode = _animationMode;
            this._ngZone = _ngZone;
            this._invert = false;
            this._max = 100;
            this._min = 0;
            this._step = 1;
            this._thumbLabel = false;
            this._tickInterval = 0;
            this._value = null;
            this._vertical = false;
            /**
             * Event emitted when the slider value has changed.
             */
            this.change = new EventEmitter();
            /**
             * Event emitted when the slider thumb moves.
             */
            this.input = new EventEmitter();
            /**
             * Emits when the raw value of the slider changes. This is here primarily
             * to facilitate the two-way binding for the `value` input.
             * \@docs-private
             */
            this.valueChange = new EventEmitter();
            /**
             * onTouch function registered via registerOnTouch (ControlValueAccessor).
             */
            this.onTouched = (/**
             * @return {?}
             */
            () => { });
            this._percent = 0;
            /**
             * Whether or not the thumb is sliding.
             * Used to determine if there should be a transition for the thumb and fill track.
             */
            this._isSliding = false;
            /**
             * Whether or not the slider is active (clicked or sliding).
             * Used to shrink and grow the thumb as according to the Material Design spec.
             */
            this._isActive = false;
            /**
             * The size of a tick interval as a percentage of the size of the track.
             */
            this._tickIntervalPercent = 0;
            /**
             * The dimensions of the slider.
             */
            this._sliderDimensions = null;
            this._controlValueAccessorChangeFn = (/**
             * @return {?}
             */
            () => { });
            /**
             * Subscription to the Directionality change EventEmitter.
             */
            this._dirChangeSubscription = Subscription.EMPTY;
            /**
             * Called when the user has put their pointer down on the slider.
             */
            this._pointerDown = (/**
             * @param {?} event
             * @return {?}
             */
            (event) => {
                // Don't do anything if the slider is disabled or the
                // user is using anything other than the main mouse button.
                if (this.disabled || this._isSliding || (!isTouchEvent(event) && event.button !== 0)) {
                    return;
                }
                this._runInsideZone((/**
                 * @return {?}
                 */
                () => {
                    /** @type {?} */
                    const oldValue = this.value;
                    /** @type {?} */
                    const pointerPosition = getPointerPositionOnPage(event);
                    this._isSliding = true;
                    this._lastPointerEvent = event;
                    event.preventDefault();
                    this._focusHostElement();
                    this._onMouseenter(); // Simulate mouseenter in case this is a mobile device.
                    this._bindGlobalEvents(event);
                    this._focusHostElement();
                    this._updateValueFromPosition(pointerPosition);
                    this._valueOnSlideStart = this.value;
                    this._pointerPositionOnStart = pointerPosition;
                    // Emit a change and input event if the value changed.
                    if (oldValue != this.value) {
                        this._emitInputEvent();
                        this._emitChangeEvent();
                    }
                }));
            });
            /**
             * Called when the user has moved their pointer after
             * starting to drag. Bound on the document level.
             */
            this._pointerMove = (/**
             * @param {?} event
             * @return {?}
             */
            (event) => {
                if (this._isSliding) {
                    // Prevent the slide from selecting anything else.
                    event.preventDefault();
                    /** @type {?} */
                    const oldValue = this.value;
                    this._lastPointerEvent = event;
                    this._updateValueFromPosition(getPointerPositionOnPage(event));
                    // Native range elements always emit `input` events when the value changed while sliding.
                    if (oldValue != this.value) {
                        this._emitInputEvent();
                    }
                }
            });
            /**
             * Called when the user has lifted their pointer. Bound on the document level.
             */
            this._pointerUp = (/**
             * @param {?} event
             * @return {?}
             */
            (event) => {
                if (this._isSliding) {
                    /** @type {?} */
                    const pointerPositionOnStart = this._pointerPositionOnStart;
                    /** @type {?} */
                    const currentPointerPosition = getPointerPositionOnPage(event);
                    event.preventDefault();
                    this._removeGlobalEvents();
                    this._valueOnSlideStart = this._pointerPositionOnStart = this._lastPointerEvent = null;
                    this._isSliding = false;
                    if (this._valueOnSlideStart != this.value && !this.disabled &&
                        pointerPositionOnStart && (pointerPositionOnStart.x !== currentPointerPosition.x ||
                        pointerPositionOnStart.y !== currentPointerPosition.y)) {
                        this._emitChangeEvent();
                    }
                }
            });
            /**
             * Called when the window has lost focus.
             */
            this._windowBlur = (/**
             * @return {?}
             */
            () => {
                // If the window is blurred while dragging we need to stop dragging because the
                // browser won't dispatch the `mouseup` and `touchend` events anymore.
                if (this._lastPointerEvent) {
                    this._pointerUp(this._lastPointerEvent);
                }
            });
            this._document = document;
            this.tabIndex = parseInt(tabIndex) || 0;
            this._runOutsizeZone((/**
             * @return {?}
             */
            () => {
                /** @type {?} */
                const element = elementRef.nativeElement;
                element.addEventListener('mousedown', this._pointerDown, activeEventOptions);
                element.addEventListener('touchstart', this._pointerDown, activeEventOptions);
            }));
        }
        /**
         * Whether the slider is inverted.
         * @return {?}
         */
        get invert() { return this._invert; }
        /**
         * @param {?} value
         * @return {?}
         */
        set invert(value) {
            this._invert = coerceBooleanProperty(value);
        }
        /**
         * The maximum value that the slider can have.
         * @return {?}
         */
        get max() { return this._max; }
        /**
         * @param {?} v
         * @return {?}
         */
        set max(v) {
            this._max = coerceNumberProperty(v, this._max);
            this._percent = this._calculatePercentage(this._value);
            // Since this also modifies the percentage, we need to let the change detection know.
            this._changeDetectorRef.markForCheck();
        }
        /**
         * The minimum value that the slider can have.
         * @return {?}
         */
        get min() { return this._min; }
        /**
         * @param {?} v
         * @return {?}
         */
        set min(v) {
            this._min = coerceNumberProperty(v, this._min);
            // If the value wasn't explicitly set by the user, set it to the min.
            if (this._value === null) {
                this.value = this._min;
            }
            this._percent = this._calculatePercentage(this._value);
            // Since this also modifies the percentage, we need to let the change detection know.
            this._changeDetectorRef.markForCheck();
        }
        /**
         * The values at which the thumb will snap.
         * @return {?}
         */
        get step() { return this._step; }
        /**
         * @param {?} v
         * @return {?}
         */
        set step(v) {
            this._step = coerceNumberProperty(v, this._step);
            if (this._step % 1 !== 0) {
                this._roundToDecimal = (/** @type {?} */ (this._step.toString().split('.').pop())).length;
            }
            // Since this could modify the label, we need to notify the change detection.
            this._changeDetectorRef.markForCheck();
        }
        /**
         * Whether or not to show the thumb label.
         * @return {?}
         */
        get thumbLabel() { return this._thumbLabel; }
        /**
         * @param {?} value
         * @return {?}
         */
        set thumbLabel(value) { this._thumbLabel = coerceBooleanProperty(value); }
        /**
         * How often to show ticks. Relative to the step so that a tick always appears on a step.
         * Ex: Tick interval of 4 with a step of 3 will draw a tick every 4 steps (every 12 values).
         * @return {?}
         */
        get tickInterval() { return this._tickInterval; }
        /**
         * @param {?} value
         * @return {?}
         */
        set tickInterval(value) {
            if (value === 'auto') {
                this._tickInterval = 'auto';
            }
            else if (typeof value === 'number' || typeof value === 'string') {
                this._tickInterval = coerceNumberProperty(value, (/** @type {?} */ (this._tickInterval)));
            }
            else {
                this._tickInterval = 0;
            }
        }
        /**
         * Value of the slider.
         * @return {?}
         */
        get value() {
            // If the value needs to be read and it is still uninitialized, initialize it to the min.
            if (this._value === null) {
                this.value = this._min;
            }
            return this._value;
        }
        /**
         * @param {?} v
         * @return {?}
         */
        set value(v) {
            if (v !== this._value) {
                /** @type {?} */
                let value = coerceNumberProperty(v);
                // While incrementing by a decimal we can end up with values like 33.300000000000004.
                // Truncate it to ensure that it matches the label and to make it easier to work with.
                if (this._roundToDecimal) {
                    value = parseFloat(value.toFixed(this._roundToDecimal));
                }
                this._value = value;
                this._percent = this._calculatePercentage(this._value);
                // Since this also modifies the percentage, we need to let the change detection know.
                this._changeDetectorRef.markForCheck();
            }
        }
        /**
         * Whether the slider is vertical.
         * @return {?}
         */
        get vertical() { return this._vertical; }
        /**
         * @param {?} value
         * @return {?}
         */
        set vertical(value) {
            this._vertical = coerceBooleanProperty(value);
        }
        /**
         * The value to be used for display purposes.
         * @return {?}
         */
        get displayValue() {
            if (this.displayWith) {
                // Value is never null but since setters and getters cannot have
                // different types, the value getter is also typed to return null.
                return this.displayWith((/** @type {?} */ (this.value)));
            }
            // Note that this could be improved further by rounding something like 0.999 to 1 or
            // 0.899 to 0.9, however it is very performance sensitive, because it gets called on
            // every change detection cycle.
            if (this._roundToDecimal && this.value && this.value % 1 !== 0) {
                return this.value.toFixed(this._roundToDecimal);
            }
            return this.value || 0;
        }
        /**
         * set focus to the host element
         * @param {?=} options
         * @return {?}
         */
        focus(options) {
            this._focusHostElement(options);
        }
        /**
         * blur the host element
         * @return {?}
         */
        blur() {
            this._blurHostElement();
        }
        /**
         * The percentage of the slider that coincides with the value.
         * @return {?}
         */
        get percent() { return this._clamp(this._percent); }
        /**
         * Whether the axis of the slider is inverted.
         * (i.e. whether moving the thumb in the positive x or y direction decreases the slider's value).
         * @return {?}
         */
        get _invertAxis() {
            // Standard non-inverted mode for a vertical slider should be dragging the thumb from bottom to
            // top. However from a y-axis standpoint this is inverted.
            return this.vertical ? !this.invert : this.invert;
        }
        /**
         * Whether the slider is at its minimum value.
         * @return {?}
         */
        get _isMinValue() {
            return this.percent === 0;
        }
        /**
         * The amount of space to leave between the slider thumb and the track fill & track background
         * elements.
         * @return {?}
         */
        get _thumbGap() {
            if (this.disabled) {
                return DISABLED_THUMB_GAP;
            }
            if (this._isMinValue && !this.thumbLabel) {
                return this._isActive ? MIN_VALUE_ACTIVE_THUMB_GAP : MIN_VALUE_NONACTIVE_THUMB_GAP;
            }
            return 0;
        }
        /**
         * CSS styles for the track background element.
         * @return {?}
         */
        get _trackBackgroundStyles() {
            /** @type {?} */
            const axis = this.vertical ? 'Y' : 'X';
            /** @type {?} */
            const scale = this.vertical ? `1, ${1 - this.percent}, 1` : `${1 - this.percent}, 1, 1`;
            /** @type {?} */
            const sign = this._shouldInvertMouseCoords() ? '-' : '';
            return {
                // scale3d avoids some rendering issues in Chrome. See #12071.
                transform: `translate${axis}(${sign}${this._thumbGap}px) scale3d(${scale})`
            };
        }
        /**
         * CSS styles for the track fill element.
         * @return {?}
         */
        get _trackFillStyles() {
            /** @type {?} */
            const percent = this.percent;
            /** @type {?} */
            const axis = this.vertical ? 'Y' : 'X';
            /** @type {?} */
            const scale = this.vertical ? `1, ${percent}, 1` : `${percent}, 1, 1`;
            /** @type {?} */
            const sign = this._shouldInvertMouseCoords() ? '' : '-';
            return {
                // scale3d avoids some rendering issues in Chrome. See #12071.
                transform: `translate${axis}(${sign}${this._thumbGap}px) scale3d(${scale})`,
                // iOS Safari has a bug where it won't re-render elements which start of as `scale(0)` until
                // something forces a style recalculation on it. Since we'll end up with `scale(0)` when
                // the value of the slider is 0, we can easily get into this situation. We force a
                // recalculation by changing the element's `display` when it goes from 0 to any other value.
                display: percent === 0 ? 'none' : ''
            };
        }
        /**
         * CSS styles for the ticks container element.
         * @return {?}
         */
        get _ticksContainerStyles() {
            /** @type {?} */
            let axis = this.vertical ? 'Y' : 'X';
            // For a horizontal slider in RTL languages we push the ticks container off the left edge
            // instead of the right edge to avoid causing a horizontal scrollbar to appear.
            /** @type {?} */
            let sign = !this.vertical && this._getDirection() == 'rtl' ? '' : '-';
            /** @type {?} */
            let offset = this._tickIntervalPercent / 2 * 100;
            return {
                'transform': `translate${axis}(${sign}${offset}%)`
            };
        }
        /**
         * CSS styles for the ticks element.
         * @return {?}
         */
        get _ticksStyles() {
            /** @type {?} */
            let tickSize = this._tickIntervalPercent * 100;
            /** @type {?} */
            let backgroundSize = this.vertical ? `2px ${tickSize}%` : `${tickSize}% 2px`;
            /** @type {?} */
            let axis = this.vertical ? 'Y' : 'X';
            // Depending on the direction we pushed the ticks container, push the ticks the opposite
            // direction to re-center them but clip off the end edge. In RTL languages we need to flip the
            // ticks 180 degrees so we're really cutting off the end edge abd not the start.
            /** @type {?} */
            let sign = !this.vertical && this._getDirection() == 'rtl' ? '-' : '';
            /** @type {?} */
            let rotate = !this.vertical && this._getDirection() == 'rtl' ? ' rotate(180deg)' : '';
            /** @type {?} */
            let styles = {
                'backgroundSize': backgroundSize,
                // Without translateZ ticks sometimes jitter as the slider moves on Chrome & Firefox.
                'transform': `translateZ(0) translate${axis}(${sign}${tickSize / 2}%)${rotate}`
            };
            if (this._isMinValue && this._thumbGap) {
                /** @type {?} */
                let side;
                if (this.vertical) {
                    side = this._invertAxis ? 'Bottom' : 'Top';
                }
                else {
                    side = this._invertAxis ? 'Right' : 'Left';
                }
                styles[`padding${side}`] = `${this._thumbGap}px`;
            }
            return styles;
        }
        /**
         * @return {?}
         */
        get _thumbContainerStyles() {
            /** @type {?} */
            let axis = this.vertical ? 'Y' : 'X';
            // For a horizontal slider in RTL languages we push the thumb container off the left edge
            // instead of the right edge to avoid causing a horizontal scrollbar to appear.
            /** @type {?} */
            let invertOffset = (this._getDirection() == 'rtl' && !this.vertical) ? !this._invertAxis : this._invertAxis;
            /** @type {?} */
            let offset = (invertOffset ? this.percent : 1 - this.percent) * 100;
            return {
                'transform': `translate${axis}(-${offset}%)`
            };
        }
        /**
         * Whether mouse events should be converted to a slider position by calculating their distance
         * from the right or bottom edge of the slider as opposed to the top or left.
         * @return {?}
         */
        _shouldInvertMouseCoords() {
            return (this._getDirection() == 'rtl' && !this.vertical) ? !this._invertAxis : this._invertAxis;
        }
        /**
         * The language direction for this slider element.
         * @private
         * @return {?}
         */
        _getDirection() {
            return (this._dir && this._dir.value == 'rtl') ? 'rtl' : 'ltr';
        }
        /**
         * @return {?}
         */
        ngOnInit() {
            this._focusMonitor
                .monitor(this._elementRef, true)
                .subscribe((/**
             * @param {?} origin
             * @return {?}
             */
            (origin) => {
                this._isActive = !!origin && origin !== 'keyboard';
                this._changeDetectorRef.detectChanges();
            }));
            if (this._dir) {
                this._dirChangeSubscription = this._dir.change.subscribe((/**
                 * @return {?}
                 */
                () => {
                    this._changeDetectorRef.markForCheck();
                }));
            }
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            /** @type {?} */
            const element = this._elementRef.nativeElement;
            element.removeEventListener('mousedown', this._pointerDown, activeEventOptions);
            element.removeEventListener('touchstart', this._pointerDown, activeEventOptions);
            this._lastPointerEvent = null;
            this._removeGlobalEvents();
            this._focusMonitor.stopMonitoring(this._elementRef);
            this._dirChangeSubscription.unsubscribe();
        }
        /**
         * @return {?}
         */
        _onMouseenter() {
            if (this.disabled) {
                return;
            }
            // We save the dimensions of the slider here so we can use them to update the spacing of the
            // ticks and determine where on the slider click and slide events happen.
            this._sliderDimensions = this._getSliderDimensions();
            this._updateTickIntervalPercent();
        }
        /**
         * @return {?}
         */
        _onFocus() {
            // We save the dimensions of the slider here so we can use them to update the spacing of the
            // ticks and determine where on the slider click and slide events happen.
            this._sliderDimensions = this._getSliderDimensions();
            this._updateTickIntervalPercent();
        }
        /**
         * @return {?}
         */
        _onBlur() {
            this.onTouched();
        }
        /**
         * @param {?} event
         * @return {?}
         */
        _onKeydown(event) {
            if (this.disabled || hasModifierKey(event)) {
                return;
            }
            /** @type {?} */
            const oldValue = this.value;
            switch (event.keyCode) {
                case PAGE_UP:
                    this._increment(10);
                    break;
                case PAGE_DOWN:
                    this._increment(-10);
                    break;
                case END:
                    this.value = this.max;
                    break;
                case HOME:
                    this.value = this.min;
                    break;
                case LEFT_ARROW:
                    // NOTE: For a sighted user it would make more sense that when they press an arrow key on an
                    // inverted slider the thumb moves in that direction. However for a blind user, nothing
                    // about the slider indicates that it is inverted. They will expect left to be decrement,
                    // regardless of how it appears on the screen. For speakers ofRTL languages, they probably
                    // expect left to mean increment. Therefore we flip the meaning of the side arrow keys for
                    // RTL. For inverted sliders we prefer a good a11y experience to having it "look right" for
                    // sighted users, therefore we do not swap the meaning.
                    this._increment(this._getDirection() == 'rtl' ? 1 : -1);
                    break;
                case UP_ARROW:
                    this._increment(1);
                    break;
                case RIGHT_ARROW:
                    // See comment on LEFT_ARROW about the conditions under which we flip the meaning.
                    this._increment(this._getDirection() == 'rtl' ? -1 : 1);
                    break;
                case DOWN_ARROW:
                    this._increment(-1);
                    break;
                default:
                    // Return if the key is not one that we explicitly handle to avoid calling preventDefault on
                    // it.
                    return;
            }
            if (oldValue != this.value) {
                this._emitInputEvent();
                this._emitChangeEvent();
            }
            this._isSliding = true;
            event.preventDefault();
        }
        /**
         * @return {?}
         */
        _onKeyup() {
            this._isSliding = false;
        }
        /**
         * Use defaultView of injected document if available or fallback to global window reference
         * @private
         * @return {?}
         */
        _getWindow() {
            var _a;
            return ((_a = this._document) === null || _a === void 0 ? void 0 : _a.defaultView) || window;
        }
        /**
         * Binds our global move and end events. They're bound at the document level and only while
         * dragging so that the user doesn't have to keep their pointer exactly over the slider
         * as they're swiping across the screen.
         * @private
         * @param {?} triggerEvent
         * @return {?}
         */
        _bindGlobalEvents(triggerEvent) {
            // Note that we bind the events to the `document`, because it allows us to capture
            // drag cancel events where the user's pointer is outside the browser window.
            /** @type {?} */
            const document = this._document;
            if (typeof document !== 'undefined' && document) {
                /** @type {?} */
                const isTouch = isTouchEvent(triggerEvent);
                /** @type {?} */
                const moveEventName = isTouch ? 'touchmove' : 'mousemove';
                /** @type {?} */
                const endEventName = isTouch ? 'touchend' : 'mouseup';
                document.addEventListener(moveEventName, this._pointerMove, activeEventOptions);
                document.addEventListener(endEventName, this._pointerUp, activeEventOptions);
                if (isTouch) {
                    document.addEventListener('touchcancel', this._pointerUp, activeEventOptions);
                }
            }
            /** @type {?} */
            const window = this._getWindow();
            if (typeof window !== 'undefined' && window) {
                window.addEventListener('blur', this._windowBlur);
            }
        }
        /**
         * Removes any global event listeners that we may have added.
         * @private
         * @return {?}
         */
        _removeGlobalEvents() {
            /** @type {?} */
            const document = this._document;
            if (typeof document !== 'undefined' && document) {
                document.removeEventListener('mousemove', this._pointerMove, activeEventOptions);
                document.removeEventListener('mouseup', this._pointerUp, activeEventOptions);
                document.removeEventListener('touchmove', this._pointerMove, activeEventOptions);
                document.removeEventListener('touchend', this._pointerUp, activeEventOptions);
                document.removeEventListener('touchcancel', this._pointerUp, activeEventOptions);
            }
            /** @type {?} */
            const window = this._getWindow();
            if (typeof window !== 'undefined' && window) {
                window.removeEventListener('blur', this._windowBlur);
            }
        }
        /**
         * Increments the slider by the given number of steps (negative number decrements).
         * @private
         * @param {?} numSteps
         * @return {?}
         */
        _increment(numSteps) {
            this.value = this._clamp((this.value || 0) + this.step * numSteps, this.min, this.max);
        }
        /**
         * Calculate the new value from the new physical location. The value will always be snapped.
         * @private
         * @param {?} pos
         * @return {?}
         */
        _updateValueFromPosition(pos) {
            if (!this._sliderDimensions) {
                return;
            }
            /** @type {?} */
            let offset = this.vertical ? this._sliderDimensions.top : this._sliderDimensions.left;
            /** @type {?} */
            let size = this.vertical ? this._sliderDimensions.height : this._sliderDimensions.width;
            /** @type {?} */
            let posComponent = this.vertical ? pos.y : pos.x;
            // The exact value is calculated from the event and used to find the closest snap value.
            /** @type {?} */
            let percent = this._clamp((posComponent - offset) / size);
            if (this._shouldInvertMouseCoords()) {
                percent = 1 - percent;
            }
            // Since the steps may not divide cleanly into the max value, if the user
            // slid to 0 or 100 percent, we jump to the min/max value. This approach
            // is slightly more intuitive than using `Math.ceil` below, because it
            // follows the user's pointer closer.
            if (percent === 0) {
                this.value = this.min;
            }
            else if (percent === 1) {
                this.value = this.max;
            }
            else {
                /** @type {?} */
                const exactValue = this._calculateValue(percent);
                // This calculation finds the closest step by finding the closest
                // whole number divisible by the step relative to the min.
                /** @type {?} */
                const closestValue = Math.round((exactValue - this.min) / this.step) * this.step + this.min;
                // The value needs to snap to the min and max.
                this.value = this._clamp(closestValue, this.min, this.max);
            }
        }
        /**
         * Emits a change event if the current value is different from the last emitted value.
         * @private
         * @return {?}
         */
        _emitChangeEvent() {
            this._controlValueAccessorChangeFn(this.value);
            this.valueChange.emit(this.value);
            this.change.emit(this._createChangeEvent());
        }
        /**
         * Emits an input event when the current value is different from the last emitted value.
         * @private
         * @return {?}
         */
        _emitInputEvent() {
            this.input.emit(this._createChangeEvent());
        }
        /**
         * Updates the amount of space between ticks as a percentage of the width of the slider.
         * @private
         * @return {?}
         */
        _updateTickIntervalPercent() {
            if (!this.tickInterval || !this._sliderDimensions) {
                return;
            }
            if (this.tickInterval == 'auto') {
                /** @type {?} */
                let trackSize = this.vertical ? this._sliderDimensions.height : this._sliderDimensions.width;
                /** @type {?} */
                let pixelsPerStep = trackSize * this.step / (this.max - this.min);
                /** @type {?} */
                let stepsPerTick = Math.ceil(MIN_AUTO_TICK_SEPARATION / pixelsPerStep);
                /** @type {?} */
                let pixelsPerTick = stepsPerTick * this.step;
                this._tickIntervalPercent = pixelsPerTick / trackSize;
            }
            else {
                this._tickIntervalPercent = this.tickInterval * this.step / (this.max - this.min);
            }
        }
        /**
         * Creates a slider change object from the specified value.
         * @private
         * @param {?=} value
         * @return {?}
         */
        _createChangeEvent(value = this.value) {
            /** @type {?} */
            let event = new MatSliderChange();
            event.source = this;
            event.value = value;
            return event;
        }
        /**
         * Calculates the percentage of the slider that a value is.
         * @private
         * @param {?} value
         * @return {?}
         */
        _calculatePercentage(value) {
            return ((value || 0) - this.min) / (this.max - this.min);
        }
        /**
         * Calculates the value a percentage of the slider corresponds to.
         * @private
         * @param {?} percentage
         * @return {?}
         */
        _calculateValue(percentage) {
            return this.min + percentage * (this.max - this.min);
        }
        /**
         * Return a number between two numbers.
         * @private
         * @param {?} value
         * @param {?=} min
         * @param {?=} max
         * @return {?}
         */
        _clamp(value, min = 0, max = 1) {
            return Math.max(min, Math.min(value, max));
        }
        /**
         * Get the bounding client rect of the slider track element.
         * The track is used rather than the native element to ignore the extra space that the thumb can
         * take up.
         * @private
         * @return {?}
         */
        _getSliderDimensions() {
            return this._sliderWrapper ? this._sliderWrapper.nativeElement.getBoundingClientRect() : null;
        }
        /**
         * Focuses the native element.
         * Currently only used to allow a blur event to fire but will be used with keyboard input later.
         * @private
         * @param {?=} options
         * @return {?}
         */
        _focusHostElement(options) {
            this._elementRef.nativeElement.focus(options);
        }
        /**
         * Blurs the native element.
         * @private
         * @return {?}
         */
        _blurHostElement() {
            this._elementRef.nativeElement.blur();
        }
        /**
         * Runs a callback inside of the NgZone, if possible.
         * @private
         * @param {?} fn
         * @return {?}
         */
        _runInsideZone(fn) {
            // @breaking-change 9.0.0 Remove this function once `_ngZone` is a required parameter.
            this._ngZone ? this._ngZone.run(fn) : fn();
        }
        /**
         * Runs a callback outside of the NgZone, if possible.
         * @private
         * @param {?} fn
         * @return {?}
         */
        _runOutsizeZone(fn) {
            // @breaking-change 9.0.0 Remove this function once `_ngZone` is a required parameter.
            this._ngZone ? this._ngZone.runOutsideAngular(fn) : fn();
        }
        /**
         * Sets the model value. Implemented as part of ControlValueAccessor.
         * @param {?} value
         * @return {?}
         */
        writeValue(value) {
            this.value = value;
        }
        /**
         * Registers a callback to be triggered when the value has changed.
         * Implemented as part of ControlValueAccessor.
         * @param {?} fn Callback to be registered.
         * @return {?}
         */
        registerOnChange(fn) {
            this._controlValueAccessorChangeFn = fn;
        }
        /**
         * Registers a callback to be triggered when the component is touched.
         * Implemented as part of ControlValueAccessor.
         * @param {?} fn Callback to be registered.
         * @return {?}
         */
        registerOnTouched(fn) {
            this.onTouched = fn;
        }
        /**
         * Sets whether the component should be disabled.
         * Implemented as part of ControlValueAccessor.
         * @param {?} isDisabled
         * @return {?}
         */
        setDisabledState(isDisabled) {
            this.disabled = isDisabled;
        }
    }
    MatSlider.ɵfac = function MatSlider_Factory(t) { return new (t || MatSlider)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(FocusMonitor), ɵɵdirectiveInject(ChangeDetectorRef), ɵɵdirectiveInject(Directionality, 8), ɵɵinjectAttribute('tabindex'), ɵɵdirectiveInject(ANIMATION_MODULE_TYPE, 8), ɵɵdirectiveInject(NgZone), ɵɵdirectiveInject(DOCUMENT$1, 8)); };
    MatSlider.ɵcmp = ɵɵdefineComponent({ type: MatSlider, selectors: [["mat-slider"]], viewQuery: function MatSlider_Query(rf, ctx) { if (rf & 1) {
            ɵɵviewQuery(_c0$c, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._sliderWrapper = _t.first);
        } }, hostAttrs: ["role", "slider", 1, "mat-slider", "mat-focus-indicator"], hostVars: 28, hostBindings: function MatSlider_HostBindings(rf, ctx) { if (rf & 1) {
            ɵɵlistener("focus", function MatSlider_focus_HostBindingHandler() { return ctx._onFocus(); })("blur", function MatSlider_blur_HostBindingHandler() { return ctx._onBlur(); })("keydown", function MatSlider_keydown_HostBindingHandler($event) { return ctx._onKeydown($event); })("keyup", function MatSlider_keyup_HostBindingHandler() { return ctx._onKeyup(); })("mouseenter", function MatSlider_mouseenter_HostBindingHandler() { return ctx._onMouseenter(); })("selectstart", function MatSlider_selectstart_HostBindingHandler($event) { return $event.preventDefault(); });
        } if (rf & 2) {
            ɵɵhostProperty("tabIndex", ctx.tabIndex);
            ɵɵattribute("aria-disabled", ctx.disabled)("aria-valuemax", ctx.max)("aria-valuemin", ctx.min)("aria-valuenow", ctx.value)("aria-orientation", ctx.vertical ? "vertical" : "horizontal");
            ɵɵclassProp("mat-slider-disabled", ctx.disabled)("mat-slider-has-ticks", ctx.tickInterval)("mat-slider-horizontal", !ctx.vertical)("mat-slider-axis-inverted", ctx._invertAxis)("mat-slider-invert-mouse-coords", ctx._shouldInvertMouseCoords())("mat-slider-sliding", ctx._isSliding)("mat-slider-thumb-label-showing", ctx.thumbLabel)("mat-slider-vertical", ctx.vertical)("mat-slider-min-value", ctx._isMinValue)("mat-slider-hide-last-tick", ctx.disabled || ctx._isMinValue && ctx._thumbGap && ctx._invertAxis)("_mat-animation-noopable", ctx._animationMode === "NoopAnimations");
        } }, inputs: { disabled: "disabled", color: "color", tabIndex: "tabIndex", invert: "invert", max: "max", min: "min", value: "value", step: "step", thumbLabel: "thumbLabel", tickInterval: "tickInterval", vertical: "vertical", displayWith: "displayWith" }, outputs: { change: "change", input: "input", valueChange: "valueChange" }, exportAs: ["matSlider"], features: [ɵɵProvidersFeature([MAT_SLIDER_VALUE_ACCESSOR]), ɵɵInheritDefinitionFeature], decls: 13, vars: 6, consts: [[1, "mat-slider-wrapper"], ["sliderWrapper", ""], [1, "mat-slider-track-wrapper"], [1, "mat-slider-track-background", 3, "ngStyle"], [1, "mat-slider-track-fill", 3, "ngStyle"], [1, "mat-slider-ticks-container", 3, "ngStyle"], [1, "mat-slider-ticks", 3, "ngStyle"], [1, "mat-slider-thumb-container", 3, "ngStyle"], [1, "mat-slider-focus-ring"], [1, "mat-slider-thumb"], [1, "mat-slider-thumb-label"], [1, "mat-slider-thumb-label-text"]], template: function MatSlider_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "div", 0, 1);
            ɵɵelementStart(2, "div", 2);
            ɵɵelement(3, "div", 3);
            ɵɵelement(4, "div", 4);
            ɵɵelementEnd();
            ɵɵelementStart(5, "div", 5);
            ɵɵelement(6, "div", 6);
            ɵɵelementEnd();
            ɵɵelementStart(7, "div", 7);
            ɵɵelement(8, "div", 8);
            ɵɵelement(9, "div", 9);
            ɵɵelementStart(10, "div", 10);
            ɵɵelementStart(11, "span", 11);
            ɵɵtext(12);
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵadvance(3);
            ɵɵproperty("ngStyle", ctx._trackBackgroundStyles);
            ɵɵadvance(1);
            ɵɵproperty("ngStyle", ctx._trackFillStyles);
            ɵɵadvance(1);
            ɵɵproperty("ngStyle", ctx._ticksContainerStyles);
            ɵɵadvance(1);
            ɵɵproperty("ngStyle", ctx._ticksStyles);
            ɵɵadvance(1);
            ɵɵproperty("ngStyle", ctx._thumbContainerStyles);
            ɵɵadvance(5);
            ɵɵtextInterpolate(ctx.displayValue);
        } }, directives: [NgStyle], styles: [".mat-slider{display:inline-block;position:relative;box-sizing:border-box;padding:8px;outline:none;vertical-align:middle}.mat-slider:not(.mat-slider-disabled):active,.mat-slider.mat-slider-sliding:not(.mat-slider-disabled){cursor:-webkit-grabbing;cursor:grabbing}.mat-slider-wrapper{position:absolute}.mat-slider-track-wrapper{position:absolute;top:0;left:0;overflow:hidden}.mat-slider-track-fill{position:absolute;transform-origin:0 0;transition:transform 400ms cubic-bezier(0.25, 0.8, 0.25, 1),background-color 400ms cubic-bezier(0.25, 0.8, 0.25, 1)}.mat-slider-track-background{position:absolute;transform-origin:100% 100%;transition:transform 400ms cubic-bezier(0.25, 0.8, 0.25, 1),background-color 400ms cubic-bezier(0.25, 0.8, 0.25, 1)}.mat-slider-ticks-container{position:absolute;left:0;top:0;overflow:hidden}.mat-slider-ticks{background-repeat:repeat;background-clip:content-box;box-sizing:border-box;opacity:0;transition:opacity 400ms cubic-bezier(0.25, 0.8, 0.25, 1)}.mat-slider-thumb-container{position:absolute;z-index:1;transition:transform 400ms cubic-bezier(0.25, 0.8, 0.25, 1)}.mat-slider-focus-ring{position:absolute;width:30px;height:30px;border-radius:50%;transform:scale(0);opacity:0;transition:transform 400ms cubic-bezier(0.25, 0.8, 0.25, 1),background-color 400ms cubic-bezier(0.25, 0.8, 0.25, 1),opacity 400ms cubic-bezier(0.25, 0.8, 0.25, 1)}.mat-slider.cdk-keyboard-focused .mat-slider-focus-ring,.mat-slider.cdk-program-focused .mat-slider-focus-ring{transform:scale(1);opacity:1}.mat-slider:not(.mat-slider-disabled):not(.mat-slider-sliding) .mat-slider-thumb-label,.mat-slider:not(.mat-slider-disabled):not(.mat-slider-sliding) .mat-slider-thumb{cursor:-webkit-grab;cursor:grab}.mat-slider-thumb{position:absolute;right:-10px;bottom:-10px;box-sizing:border-box;width:20px;height:20px;border:3px solid transparent;border-radius:50%;transform:scale(0.7);transition:transform 400ms cubic-bezier(0.25, 0.8, 0.25, 1),background-color 400ms cubic-bezier(0.25, 0.8, 0.25, 1),border-color 400ms cubic-bezier(0.25, 0.8, 0.25, 1)}.mat-slider-thumb-label{display:none;align-items:center;justify-content:center;position:absolute;width:28px;height:28px;border-radius:50%;transition:transform 400ms cubic-bezier(0.25, 0.8, 0.25, 1),border-radius 400ms cubic-bezier(0.25, 0.8, 0.25, 1),background-color 400ms cubic-bezier(0.25, 0.8, 0.25, 1)}.cdk-high-contrast-active .mat-slider-thumb-label{outline:solid 1px}.mat-slider-thumb-label-text{z-index:1;opacity:0;transition:opacity 400ms cubic-bezier(0.25, 0.8, 0.25, 1)}.mat-slider-sliding .mat-slider-track-fill,.mat-slider-sliding .mat-slider-track-background,.mat-slider-sliding .mat-slider-thumb-container{transition-duration:0ms}.mat-slider-has-ticks .mat-slider-wrapper::after{content:\"\";position:absolute;border-width:0;border-style:solid;opacity:0;transition:opacity 400ms cubic-bezier(0.25, 0.8, 0.25, 1)}.mat-slider-has-ticks.cdk-focused:not(.mat-slider-hide-last-tick) .mat-slider-wrapper::after,.mat-slider-has-ticks:hover:not(.mat-slider-hide-last-tick) .mat-slider-wrapper::after{opacity:1}.mat-slider-has-ticks.cdk-focused:not(.mat-slider-disabled) .mat-slider-ticks,.mat-slider-has-ticks:hover:not(.mat-slider-disabled) .mat-slider-ticks{opacity:1}.mat-slider-thumb-label-showing .mat-slider-focus-ring{display:none}.mat-slider-thumb-label-showing .mat-slider-thumb-label{display:flex}.mat-slider-axis-inverted .mat-slider-track-fill{transform-origin:100% 100%}.mat-slider-axis-inverted .mat-slider-track-background{transform-origin:0 0}.mat-slider:not(.mat-slider-disabled).cdk-focused.mat-slider-thumb-label-showing .mat-slider-thumb{transform:scale(0)}.mat-slider:not(.mat-slider-disabled).cdk-focused .mat-slider-thumb-label{border-radius:50% 50% 0}.mat-slider:not(.mat-slider-disabled).cdk-focused .mat-slider-thumb-label-text{opacity:1}.mat-slider:not(.mat-slider-disabled).cdk-mouse-focused .mat-slider-thumb,.mat-slider:not(.mat-slider-disabled).cdk-touch-focused .mat-slider-thumb,.mat-slider:not(.mat-slider-disabled).cdk-program-focused .mat-slider-thumb{border-width:2px;transform:scale(1)}.mat-slider-disabled .mat-slider-focus-ring{transform:scale(0);opacity:0}.mat-slider-disabled .mat-slider-thumb{border-width:4px;transform:scale(0.5)}.mat-slider-disabled .mat-slider-thumb-label{display:none}.mat-slider-horizontal{height:48px;min-width:128px}.mat-slider-horizontal .mat-slider-wrapper{height:2px;top:23px;left:8px;right:8px}.mat-slider-horizontal .mat-slider-wrapper::after{height:2px;border-left-width:2px;right:0;top:0}.mat-slider-horizontal .mat-slider-track-wrapper{height:2px;width:100%}.mat-slider-horizontal .mat-slider-track-fill{height:2px;width:100%;transform:scaleX(0)}.mat-slider-horizontal .mat-slider-track-background{height:2px;width:100%;transform:scaleX(1)}.mat-slider-horizontal .mat-slider-ticks-container{height:2px;width:100%}.cdk-high-contrast-active .mat-slider-horizontal .mat-slider-ticks-container{height:0;outline:solid 2px;top:1px}.mat-slider-horizontal .mat-slider-ticks{height:2px;width:100%}.mat-slider-horizontal .mat-slider-thumb-container{width:100%;height:0;top:50%}.mat-slider-horizontal .mat-slider-focus-ring{top:-15px;right:-15px}.mat-slider-horizontal .mat-slider-thumb-label{right:-14px;top:-40px;transform:translateY(26px) scale(0.01) rotate(45deg)}.mat-slider-horizontal .mat-slider-thumb-label-text{transform:rotate(-45deg)}.mat-slider-horizontal.cdk-focused .mat-slider-thumb-label{transform:rotate(45deg)}.cdk-high-contrast-active .mat-slider-horizontal.cdk-focused .mat-slider-thumb-label,.cdk-high-contrast-active .mat-slider-horizontal.cdk-focused .mat-slider-thumb-label-text{transform:none}.mat-slider-vertical{width:48px;min-height:128px}.mat-slider-vertical .mat-slider-wrapper{width:2px;top:8px;bottom:8px;left:23px}.mat-slider-vertical .mat-slider-wrapper::after{width:2px;border-top-width:2px;bottom:0;left:0}.mat-slider-vertical .mat-slider-track-wrapper{height:100%;width:2px}.mat-slider-vertical .mat-slider-track-fill{height:100%;width:2px;transform:scaleY(0)}.mat-slider-vertical .mat-slider-track-background{height:100%;width:2px;transform:scaleY(1)}.mat-slider-vertical .mat-slider-ticks-container{width:2px;height:100%}.cdk-high-contrast-active .mat-slider-vertical .mat-slider-ticks-container{width:0;outline:solid 2px;left:1px}.mat-slider-vertical .mat-slider-focus-ring{bottom:-15px;left:-15px}.mat-slider-vertical .mat-slider-ticks{width:2px;height:100%}.mat-slider-vertical .mat-slider-thumb-container{height:100%;width:0;left:50%}.mat-slider-vertical .mat-slider-thumb{-webkit-backface-visibility:hidden;backface-visibility:hidden}.mat-slider-vertical .mat-slider-thumb-label{bottom:-14px;left:-40px;transform:translateX(26px) scale(0.01) rotate(-45deg)}.mat-slider-vertical .mat-slider-thumb-label-text{transform:rotate(45deg)}.mat-slider-vertical.cdk-focused .mat-slider-thumb-label{transform:rotate(-45deg)}[dir=rtl] .mat-slider-wrapper::after{left:0;right:auto}[dir=rtl] .mat-slider-horizontal .mat-slider-track-fill{transform-origin:100% 100%}[dir=rtl] .mat-slider-horizontal .mat-slider-track-background{transform-origin:0 0}[dir=rtl] .mat-slider-horizontal.mat-slider-axis-inverted .mat-slider-track-fill{transform-origin:0 0}[dir=rtl] .mat-slider-horizontal.mat-slider-axis-inverted .mat-slider-track-background{transform-origin:100% 100%}.mat-slider._mat-animation-noopable .mat-slider-track-fill,.mat-slider._mat-animation-noopable .mat-slider-track-background,.mat-slider._mat-animation-noopable .mat-slider-ticks,.mat-slider._mat-animation-noopable .mat-slider-thumb-container,.mat-slider._mat-animation-noopable .mat-slider-focus-ring,.mat-slider._mat-animation-noopable .mat-slider-thumb,.mat-slider._mat-animation-noopable .mat-slider-thumb-label,.mat-slider._mat-animation-noopable .mat-slider-thumb-label-text,.mat-slider._mat-animation-noopable .mat-slider-has-ticks .mat-slider-wrapper::after{transition:none}\n"], encapsulation: 2, changeDetection: 0 });
    /** @nocollapse */
    MatSlider.ctorParameters = () => [
        { type: ElementRef },
        { type: FocusMonitor },
        { type: ChangeDetectorRef },
        { type: Directionality, decorators: [{ type: Optional }] },
        { type: String, decorators: [{ type: Attribute, args: ['tabindex',] }] },
        { type: String, decorators: [{ type: Optional }, { type: Inject, args: [ANIMATION_MODULE_TYPE,] }] },
        { type: NgZone },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [DOCUMENT$1,] }] }
    ];
    MatSlider.propDecorators = {
        invert: [{ type: Input }],
        max: [{ type: Input }],
        min: [{ type: Input }],
        step: [{ type: Input }],
        thumbLabel: [{ type: Input }],
        tickInterval: [{ type: Input }],
        value: [{ type: Input }],
        displayWith: [{ type: Input }],
        vertical: [{ type: Input }],
        change: [{ type: Output }],
        input: [{ type: Output }],
        valueChange: [{ type: Output }],
        _sliderWrapper: [{ type: ViewChild, args: ['sliderWrapper',] }]
    };
    /**
     * Returns whether an event is a touch event.
     * @param {?} event
     * @return {?}
     */
    function isTouchEvent(event) {
        // This function is called for every pixel that the user has dragged so we need it to be
        // as fast as possible. Since we only bind mouse events and touch events, we can assume
        // that if the event's name starts with `t`, it's a touch event.
        return event.type[0] === 't';
    }
    /**
     * Gets the coordinates of a touch or mouse event relative to the viewport.
     * @param {?} event
     * @return {?}
     */
    function getPointerPositionOnPage(event) {
        // `touches` will be empty for start/end events so we have to fall back to `changedTouches`.
        /** @type {?} */
        const point = isTouchEvent(event) ? (event.touches[0] || event.changedTouches[0]) : event;
        return { x: point.clientX, y: point.clientY };
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/slider/slider-module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class MatSliderModule {
    }
    MatSliderModule.ɵmod = ɵɵdefineNgModule({ type: MatSliderModule });
    MatSliderModule.ɵinj = ɵɵdefineInjector({ factory: function MatSliderModule_Factory(t) { return new (t || MatSliderModule)(); }, imports: [[CommonModule, MatCommonModule],
            MatCommonModule] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(MatSliderModule, { declarations: function () { return [MatSlider]; }, imports: function () { return [CommonModule, MatCommonModule]; }, exports: function () { return [MatSlider,
            MatCommonModule]; } }); })();

    const DTYPE_ENUM_TO_NAME = { "19": "float16", "1": "float32", "2": "float64", "3": "int32", "4": "uint8", "17": "uint16", "22": "uint32", "23": "uint64", "5": "int16", "6": "int8", "7": "string", "8": "complex64", "18": "complex128", "9": "int64", "10": "bool", "11": "qint8", "12": "quint8", "15": "qint16", "16": "quint16", "13": "qint32", "14": "bfloat16", "20": "resource", "21": "variant", "119": "float16_ref", "101": "float32_ref", "102": "float64_ref", "103": "int32_ref", "122": "uint32_ref", "104": "uint8_ref", "117": "uint16_ref", "105": "int16_ref", "106": "int8_ref", "107": "string_ref", "108": "complex64_ref", "118": "complex128_ref", "109": "int64_ref", "123": "uint64_ref", "110": "bool_ref", "111": "qint8_ref", "112": "quint8_ref", "115": "qint16_ref", "116": "quint16_ref", "113": "qint32_ref", "114": "bfloat16_ref", "120": "resource_ref", "121": "variant_ref" };

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/store/debug_tensor_value.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Parse a number array that represents debugging summary of an instrumented
     * tensor value.
     *
     * @param {?} raw
     * @return {?} A DebugTensorValue object with the same information as
     *   carried by `array`, but represented in a more explicit fashion.
     *   For numbers that represent breakdown of numeric values by type
     *   (e.g., counts of -inf, +inf and nan), the corresponding fields
     *   in the returned object will be defined only if the count is non-zero.
     */
    function parseDebugTensorValue(raw) {
        const { tensorDebugMode, array } = raw;
        switch (tensorDebugMode) {
            case TensorDebugMode.NO_TENSOR: {
                if (array !== null) {
                    throw new Error('Unexpectedly received non-null debug-tensor-value array ' +
                        'under NO_TENSOR mode');
                }
                return {};
            }
            case TensorDebugMode.CURT_HEALTH: {
                if (array === null || array.length !== 2) {
                    throw new Error(`Under CURT_HEALTH mode, expected debug-tensor-value array ` +
                        `to have length 2, but got ${JSON.stringify(array)}`);
                }
                return {
                    hasInfOrNaN: Boolean(array[1]),
                };
            }
            case TensorDebugMode.CONCISE_HEALTH: {
                if (array === null || array.length !== 5) {
                    throw new Error(`Under CONCISE_HEALTH mode, expected debug-tensor-value array ` +
                        `to have length 5, but got ${JSON.stringify(array)}`);
                }
                /** @type {?} */
                const value = {
                    size: array[1],
                };
                if (array[2] > 0) {
                    value.numNegativeInfs = array[2];
                }
                if (array[3] > 0) {
                    value.numPositiveInfs = array[3];
                }
                if (array[4] > 0) {
                    value.numNaNs = array[4];
                }
                return value;
            }
            case TensorDebugMode.SHAPE: {
                if (array === null || array.length !== 10) {
                    throw new Error(`Under SHAPE mode, expected debug-tensor-value array ` +
                        `to have length 10, but got ${JSON.stringify(array)}`);
                }
                /** @type {?} */
                const rank = array[2];
                /** @type {?} */
                let shape = array.slice(4, Math.min(4 + rank, array.length));
                if (shape.length < rank) {
                    // The SHAPE mode truncates the shape at head.
                    shape = new Array(rank - shape.length).concat(shape);
                }
                return {
                    dtype: DTYPE_ENUM_TO_NAME[array[1]],
                    rank,
                    size: array[3],
                    shape,
                };
            }
            case TensorDebugMode.FULL_HEALTH: {
                if (array === null || array.length !== 11) {
                    throw new Error(`Under FULL_HEALTH mode, expected debug-tensor-value array ` +
                        `to have length 11, but got ${JSON.stringify(array)}`);
                }
                /** @type {?} */
                const rank = array[3];
                /** @type {?} */
                const value = {
                    dtype: DTYPE_ENUM_TO_NAME[array[2]],
                    rank,
                    size: array[4],
                };
                if (array[5] > 0) {
                    value.numNegativeInfs = array[5];
                }
                if (array[6] > 0) {
                    value.numPositiveInfs = array[6];
                }
                if (array[7] > 0) {
                    value.numNaNs = array[7];
                }
                if (array[8] > 0) {
                    value.numNegativeFinites = array[8];
                }
                if (array[9] > 0) {
                    value.numZeros = array[9];
                }
                if (array[10] > 0) {
                    value.numPositiveFinites = array[10];
                }
                return value;
            }
            case TensorDebugMode.FULL_TENSOR: {
                // Under FULL_TENSOR mode, the full tensor value is supplied via
                // separate means. No summary values are provided for the tensor value.
                if (array !== null) {
                    throw new Error('Unexpectedly received non-null debug-tensor-value array ' +
                        'under FULL_TENSOR mode');
                }
                return {};
            }
            default: {
                throw new Error(`Unrecognized tensorDebugMode: ${tensorDebugMode}`);
            }
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/views/debug_tensor_value/debug_tensor_value_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    function DebugTensorNumericBreakdownComponent_div_5_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "div", 4);
    } }
    function DebugTensorNumericBreakdownComponent_div_6_div_1_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 7);
        ɵɵelementStart(1, "span", 8);
        ɵɵtext(2, "NaN");
        ɵɵelementEnd();
        ɵɵelementStart(3, "span", 9);
        ɵɵtext(4);
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r2 = ɵɵnextContext(2);
        ɵɵadvance(4);
        ɵɵtextInterpolate1("\u00D7", ctx_r2.numNaNs, "");
    } }
    function DebugTensorNumericBreakdownComponent_div_6_div_2_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 7);
        ɵɵelementStart(1, "span", 8);
        ɵɵtext(2, "-\u221E");
        ɵɵelementEnd();
        ɵɵelementStart(3, "span", 9);
        ɵɵtext(4);
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r3 = ɵɵnextContext(2);
        ɵɵadvance(4);
        ɵɵtextInterpolate1("\u00D7", ctx_r3.numNegativeInfs, "");
    } }
    function DebugTensorNumericBreakdownComponent_div_6_div_3_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 7);
        ɵɵelementStart(1, "span", 8);
        ɵɵtext(2, "+\u221E");
        ɵɵelementEnd();
        ɵɵelementStart(3, "span", 9);
        ɵɵtext(4);
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r4 = ɵɵnextContext(2);
        ɵɵadvance(4);
        ɵɵtextInterpolate1("\u00D7", ctx_r4.numPositiveInfs, "");
    } }
    function DebugTensorNumericBreakdownComponent_div_6_div_4_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 7);
        ɵɵelementStart(1, "span", 10);
        ɵɵtext(2, "-");
        ɵɵelementEnd();
        ɵɵelementStart(3, "span", 9);
        ɵɵtext(4);
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r5 = ɵɵnextContext(2);
        ɵɵadvance(4);
        ɵɵtextInterpolate1("\u00D7", ctx_r5.numNegativeFinites, "");
    } }
    function DebugTensorNumericBreakdownComponent_div_6_div_5_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 7);
        ɵɵelementStart(1, "span", 10);
        ɵɵtext(2, "0");
        ɵɵelementEnd();
        ɵɵelementStart(3, "span", 9);
        ɵɵtext(4);
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r6 = ɵɵnextContext(2);
        ɵɵadvance(4);
        ɵɵtextInterpolate1("\u00D7", ctx_r6.numZeros, "");
    } }
    function DebugTensorNumericBreakdownComponent_div_6_div_6_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 7);
        ɵɵelementStart(1, "span", 10);
        ɵɵtext(2, "+");
        ɵɵelementEnd();
        ɵɵelementStart(3, "span", 9);
        ɵɵtext(4);
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r7 = ɵɵnextContext(2);
        ɵɵadvance(4);
        ɵɵtextInterpolate1("\u00D7", ctx_r7.numPositiveFinites, "");
    } }
    function DebugTensorNumericBreakdownComponent_div_6_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 5);
        ɵɵtemplate(1, DebugTensorNumericBreakdownComponent_div_6_div_1_Template, 5, 1, "div", 6);
        ɵɵtemplate(2, DebugTensorNumericBreakdownComponent_div_6_div_2_Template, 5, 1, "div", 6);
        ɵɵtemplate(3, DebugTensorNumericBreakdownComponent_div_6_div_3_Template, 5, 1, "div", 6);
        ɵɵtemplate(4, DebugTensorNumericBreakdownComponent_div_6_div_4_Template, 5, 1, "div", 6);
        ɵɵtemplate(5, DebugTensorNumericBreakdownComponent_div_6_div_5_Template, 5, 1, "div", 6);
        ɵɵtemplate(6, DebugTensorNumericBreakdownComponent_div_6_div_6_Template, 5, 1, "div", 6);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r1 = ɵɵnextContext();
        ɵɵadvance(1);
        ɵɵproperty("ngIf", ctx_r1.numNaNs !== undefined && ctx_r1.numNaNs > 0);
        ɵɵadvance(1);
        ɵɵproperty("ngIf", ctx_r1.numNegativeInfs !== undefined && ctx_r1.numNegativeInfs > 0);
        ɵɵadvance(1);
        ɵɵproperty("ngIf", ctx_r1.numPositiveInfs !== undefined && ctx_r1.numPositiveInfs > 0);
        ɵɵadvance(1);
        ɵɵproperty("ngIf", ctx_r1.numNegativeFinites !== undefined && ctx_r1.numNegativeFinites > 0);
        ɵɵadvance(1);
        ɵɵproperty("ngIf", ctx_r1.numZeros !== undefined && ctx_r1.numZeros > 0);
        ɵɵadvance(1);
        ɵɵproperty("ngIf", ctx_r1.numPositiveFinites !== undefined && ctx_r1.numPositiveFinites > 0);
    } }
    const _c0$d = function (a1) { return ["container", a1]; };
    function DebugTensorValueComponent_debug_tensor_dtype_0_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "debug-tensor-dtype", 5);
    } if (rf & 2) {
        const ctx_r0 = ɵɵnextContext();
        ɵɵproperty("dtype", ctx_r0.debugTensorValue.dtype);
    } }
    function DebugTensorValueComponent_debug_tensor_rank_1_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "debug-tensor-rank", 6);
    } if (rf & 2) {
        const ctx_r1 = ɵɵnextContext();
        ɵɵproperty("rank", ctx_r1.debugTensorValue.rank);
    } }
    function DebugTensorValueComponent_debug_tensor_shape_2_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "debug-tensor-shape", 7);
    } if (rf & 2) {
        const ctx_r2 = ɵɵnextContext();
        ɵɵproperty("shape", ctx_r2.debugTensorValue.shape);
    } }
    function DebugTensorValueComponent_debug_tensor_has_inf_or_nan_3_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "debug-tensor-has-inf-or-nan", 8);
    } if (rf & 2) {
        const ctx_r3 = ɵɵnextContext();
        ɵɵproperty("hasInfOrNaN", ctx_r3.debugTensorValue.hasInfOrNaN);
    } }
    function DebugTensorValueComponent_debug_tensor_numeric_breakdown_4_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "debug-tensor-numeric-breakdown", 9);
    } if (rf & 2) {
        const ctx_r4 = ɵɵnextContext();
        ɵɵpropertyInterpolate("size", ctx_r4.debugTensorValue.size);
        ɵɵproperty("numNegativeInfs", ctx_r4.debugTensorValue.numNegativeInfs)("numPositiveInfs", ctx_r4.debugTensorValue.numPositiveInfs)("numNaNs", ctx_r4.debugTensorValue.numNaNs)("numNegativeFinites", ctx_r4.debugTensorValue.numNegativeFinites)("numZeros", ctx_r4.debugTensorValue.numZeros)("numPositiveFinites", ctx_r4.debugTensorValue.numPositiveFinites);
    } }
    /** @type {?} */
    const basicDebugInfoStyle = `
  :host {
    background-color: #e3e5e8;
    border: 1px solid #c0c0c0;
    border-radius: 4px;
    font-family: 'Roboto Mono', monospace;
    height: 14px;
    line-height: 14px;
    margin: 0 2px;
    padding: 1px 3px;
    width: max-content;
  }
`;
    class DebugTensorDTypeComponent {
    }
    DebugTensorDTypeComponent.decorators = [
        { type: Component, args: [{
                    selector: 'debug-tensor-dtype',
                    template: ` {{ dtype }} `,
                    styles: [basicDebugInfoStyle],
                },] },
    ];
    DebugTensorDTypeComponent.propDecorators = {
        dtype: [{ type: Input }]
    };
    /** @nocollapse */ DebugTensorDTypeComponent.ɵfac = function DebugTensorDTypeComponent_Factory(t) { return new (t || DebugTensorDTypeComponent)(); };
    /** @nocollapse */ DebugTensorDTypeComponent.ɵcmp = ɵɵdefineComponent({ type: DebugTensorDTypeComponent, selectors: [["debug-tensor-dtype"]], inputs: { dtype: "dtype" }, decls: 1, vars: 1, template: function DebugTensorDTypeComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵtext(0);
        } if (rf & 2) {
            ɵɵtextInterpolate1(" ", ctx.dtype, " ");
        } }, styles: ["[_nghost-%COMP%] {\n    background-color: #e3e5e8;\n    border: 1px solid #c0c0c0;\n    border-radius: 4px;\n    font-family: 'Roboto Mono', monospace;\n    height: 14px;\n    line-height: 14px;\n    margin: 0 2px;\n    padding: 1px 3px;\n    width: max-content;\n  }"] });
    class DebugTensorRankComponent {
    }
    DebugTensorRankComponent.decorators = [
        { type: Component, args: [{
                    selector: 'debug-tensor-rank',
                    template: ` {{ rank }}D `,
                    styles: [basicDebugInfoStyle],
                },] },
    ];
    DebugTensorRankComponent.propDecorators = {
        rank: [{ type: Input }]
    };
    /** @nocollapse */ DebugTensorRankComponent.ɵfac = function DebugTensorRankComponent_Factory(t) { return new (t || DebugTensorRankComponent)(); };
    /** @nocollapse */ DebugTensorRankComponent.ɵcmp = ɵɵdefineComponent({ type: DebugTensorRankComponent, selectors: [["debug-tensor-rank"]], inputs: { rank: "rank" }, decls: 1, vars: 1, template: function DebugTensorRankComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵtext(0);
        } if (rf & 2) {
            ɵɵtextInterpolate1(" ", ctx.rank, "D ");
        } }, styles: ["[_nghost-%COMP%] {\n    background-color: #e3e5e8;\n    border: 1px solid #c0c0c0;\n    border-radius: 4px;\n    font-family: 'Roboto Mono', monospace;\n    height: 14px;\n    line-height: 14px;\n    margin: 0 2px;\n    padding: 1px 3px;\n    width: max-content;\n  }"] });
    class DebugTensorShapeComponent {
        /**
         * @return {?}
         */
        get shapeString() {
            return ('[' +
                this.shape
                    .map((/**
                 * @param {?} dim
                 * @return {?}
                 */
                (dim) => {
                    return dim === undefined ? '?' : String(dim);
                }))
                    .join(',') +
                ']');
        }
    }
    DebugTensorShapeComponent.decorators = [
        { type: Component, args: [{
                    selector: 'debug-tensor-shape',
                    template: ` shape:{{ shapeString }} `,
                    styles: [basicDebugInfoStyle],
                },] },
    ];
    DebugTensorShapeComponent.propDecorators = {
        shape: [{ type: Input }]
    };
    /** @nocollapse */ DebugTensorShapeComponent.ɵfac = function DebugTensorShapeComponent_Factory(t) { return new (t || DebugTensorShapeComponent)(); };
    /** @nocollapse */ DebugTensorShapeComponent.ɵcmp = ɵɵdefineComponent({ type: DebugTensorShapeComponent, selectors: [["debug-tensor-shape"]], inputs: { shape: "shape" }, decls: 1, vars: 1, template: function DebugTensorShapeComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵtext(0);
        } if (rf & 2) {
            ɵɵtextInterpolate1(" shape:", ctx.shapeString, " ");
        } }, styles: ["[_nghost-%COMP%] {\n    background-color: #e3e5e8;\n    border: 1px solid #c0c0c0;\n    border-radius: 4px;\n    font-family: 'Roboto Mono', monospace;\n    height: 14px;\n    line-height: 14px;\n    margin: 0 2px;\n    padding: 1px 3px;\n    width: max-content;\n  }"] });
    class DebugTensorNumericBreakdownComponent {
        /**
         * @return {?}
         */
        get breakdownExists() {
            return (this.numNaNs !== undefined ||
                this.numNegativeInfs !== undefined ||
                this.numPositiveInfs !== undefined ||
                this.numNegativeFinites !== undefined ||
                this.numZeros !== undefined ||
                this.numPositiveFinites !== undefined);
        }
    }
    DebugTensorNumericBreakdownComponent.decorators = [
        { type: Component, args: [{
                    selector: 'debug-tensor-numeric-breakdown',
                    template: `
    <div class="size">
      <span>size:</span>
      <span class="size-value">{{ size }}</span>
    </div>
    <div *ngIf="breakdownExists" class="break"></div>
    <div *ngIf="breakdownExists" class="breakdown">
      <div *ngIf="numNaNs !== undefined && numNaNs > 0" class="category">
        <span class="category-tag infinite">NaN</span>
        <span class="category-count">×{{ numNaNs }}</span>
      </div>
      <div
        *ngIf="numNegativeInfs !== undefined && numNegativeInfs > 0"
        class="category"
      >
        <span class="category-tag infinite">-∞</span>
        <span class="category-count">×{{ numNegativeInfs }}</span>
      </div>
      <div
        *ngIf="numPositiveInfs !== undefined && numPositiveInfs > 0"
        class="category"
      >
        <span class="category-tag infinite">+∞</span>
        <span class="category-count">×{{ numPositiveInfs }}</span>
      </div>
      <div
        *ngIf="numNegativeFinites !== undefined && numNegativeFinites > 0"
        class="category"
      >
        <span class="category-tag finite">-</span>
        <span class="category-count">×{{ numNegativeFinites }}</span>
      </div>
      <div *ngIf="numZeros !== undefined && numZeros > 0" class="category">
        <span class="category-tag finite">0</span>
        <span class="category-count">×{{ numZeros }}</span>
      </div>
      <div
        *ngIf="numPositiveFinites !== undefined && numPositiveFinites > 0"
        class="category"
      >
        <span class="category-tag finite">+</span>
        <span class="category-count">×{{ numPositiveFinites }}</span>
      </div>
    </div>
  `,
                    styles: [
                        `
      :host {
        background-color: #e3e5e8;
        border: 1px solid #c0c0c0;
        border-radius: 4px;
        font-family: 'Roboto Mono', monospace;
        font-size: 10px;
        margin: 0 2px;
        padding: 1px;
      }
      .break {
        flex-basis: 100%;
        width: 0;
      }
      .size {
        display: block;
        height: 11px;
        line-height: 11px;
        margin: 0 3px;
        vertical-align: middle;
      }
      .breakdown {
        border-top: 1px solid rgba(0, 0, 0, 0.12);
        display: flex;
        height: 11px;
        line-height: 11px;
        padding: 2px;
        vertical-align: middle;
      }
      .category {
        margin-bottom: 2px;
        margin-left: 4px;
        margin-top: 2px;
        heigth: 100%;
        width: max-content;
      }
      .category-tag {
        border-radius: 2px;
        padding: 0 2px;
      }
      .finite {
        background-color: #aaa;
        color: #fefefe;
      }
      .infinite {
        background-color: #e52592;
        color: #fff;
      }
    `,
                    ],
                },] },
    ];
    DebugTensorNumericBreakdownComponent.propDecorators = {
        size: [{ type: Input }],
        numNaNs: [{ type: Input }],
        numNegativeInfs: [{ type: Input }],
        numPositiveInfs: [{ type: Input }],
        numNegativeFinites: [{ type: Input }],
        numZeros: [{ type: Input }],
        numPositiveFinites: [{ type: Input }]
    };
    /** @nocollapse */ DebugTensorNumericBreakdownComponent.ɵfac = function DebugTensorNumericBreakdownComponent_Factory(t) { return new (t || DebugTensorNumericBreakdownComponent)(); };
    /** @nocollapse */ DebugTensorNumericBreakdownComponent.ɵcmp = ɵɵdefineComponent({ type: DebugTensorNumericBreakdownComponent, selectors: [["debug-tensor-numeric-breakdown"]], inputs: { size: "size", numNaNs: "numNaNs", numNegativeInfs: "numNegativeInfs", numPositiveInfs: "numPositiveInfs", numNegativeFinites: "numNegativeFinites", numZeros: "numZeros", numPositiveFinites: "numPositiveFinites" }, decls: 7, vars: 3, consts: [[1, "size"], [1, "size-value"], ["class", "break", 4, "ngIf"], ["class", "breakdown", 4, "ngIf"], [1, "break"], [1, "breakdown"], ["class", "category", 4, "ngIf"], [1, "category"], [1, "category-tag", "infinite"], [1, "category-count"], [1, "category-tag", "finite"]], template: function DebugTensorNumericBreakdownComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "div", 0);
            ɵɵelementStart(1, "span");
            ɵɵtext(2, "size:");
            ɵɵelementEnd();
            ɵɵelementStart(3, "span", 1);
            ɵɵtext(4);
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵtemplate(5, DebugTensorNumericBreakdownComponent_div_5_Template, 1, 0, "div", 2);
            ɵɵtemplate(6, DebugTensorNumericBreakdownComponent_div_6_Template, 7, 6, "div", 3);
        } if (rf & 2) {
            ɵɵadvance(4);
            ɵɵtextInterpolate(ctx.size);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx.breakdownExists);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx.breakdownExists);
        } }, directives: [NgIf], styles: ["[_nghost-%COMP%] {\n        background-color: #e3e5e8;\n        border: 1px solid #c0c0c0;\n        border-radius: 4px;\n        font-family: 'Roboto Mono', monospace;\n        font-size: 10px;\n        margin: 0 2px;\n        padding: 1px;\n      }\n      .break[_ngcontent-%COMP%] {\n        flex-basis: 100%;\n        width: 0;\n      }\n      .size[_ngcontent-%COMP%] {\n        display: block;\n        height: 11px;\n        line-height: 11px;\n        margin: 0 3px;\n        vertical-align: middle;\n      }\n      .breakdown[_ngcontent-%COMP%] {\n        border-top: 1px solid rgba(0, 0, 0, 0.12);\n        display: flex;\n        height: 11px;\n        line-height: 11px;\n        padding: 2px;\n        vertical-align: middle;\n      }\n      .category[_ngcontent-%COMP%] {\n        margin-bottom: 2px;\n        margin-left: 4px;\n        margin-top: 2px;\n        heigth: 100%;\n        width: max-content;\n      }\n      .category-tag[_ngcontent-%COMP%] {\n        border-radius: 2px;\n        padding: 0 2px;\n      }\n      .finite[_ngcontent-%COMP%] {\n        background-color: #aaa;\n        color: #fefefe;\n      }\n      .infinite[_ngcontent-%COMP%] {\n        background-color: #e52592;\n        color: #fff;\n      }"] });
    class DebugTensorHasInfOrNaNComponent {
        /**
         * @return {?}
         */
        get infoString() {
            return this.hasInfOrNaN ? 'Has ∞/NaN' : 'No ∞/NaN';
        }
    }
    DebugTensorHasInfOrNaNComponent.decorators = [
        { type: Component, args: [{
                    selector: 'debug-tensor-has-inf-or-nan',
                    template: `
    <div [ngClass]="['container', hasInfOrNaN ? 'has-inf-or-nan' : '']">
      {{ infoString }}
    </div>
  `,
                    styles: [
                        `
      .container {
        background-color: #e3e5e8;
        border: 1px solid #c0c0c0;
        border-radius: 4px;
        color: #666666;
        font-family: 'Roboto Mono', monospace;
        height: 14px;
        line-height: 14px;
        margin: 0 2px;
        padding: 1px 3px;
        width: max-content;
      }
      .has-inf-or-nan {
        background-color: #e52592;
        color: #fff;
      }
    `,
                    ],
                },] },
    ];
    DebugTensorHasInfOrNaNComponent.propDecorators = {
        hasInfOrNaN: [{ type: Input }]
    };
    /** @nocollapse */ DebugTensorHasInfOrNaNComponent.ɵfac = function DebugTensorHasInfOrNaNComponent_Factory(t) { return new (t || DebugTensorHasInfOrNaNComponent)(); };
    /** @nocollapse */ DebugTensorHasInfOrNaNComponent.ɵcmp = ɵɵdefineComponent({ type: DebugTensorHasInfOrNaNComponent, selectors: [["debug-tensor-has-inf-or-nan"]], inputs: { hasInfOrNaN: "hasInfOrNaN" }, decls: 2, vars: 4, consts: [[3, "ngClass"]], template: function DebugTensorHasInfOrNaNComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "div", 0);
            ɵɵtext(1);
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵproperty("ngClass", ɵɵpureFunction1(2, _c0$d, ctx.hasInfOrNaN ? "has-inf-or-nan" : ""));
            ɵɵadvance(1);
            ɵɵtextInterpolate1(" ", ctx.infoString, " ");
        } }, directives: [NgClass], styles: [".container[_ngcontent-%COMP%] {\n        background-color: #e3e5e8;\n        border: 1px solid #c0c0c0;\n        border-radius: 4px;\n        color: #666666;\n        font-family: 'Roboto Mono', monospace;\n        height: 14px;\n        line-height: 14px;\n        margin: 0 2px;\n        padding: 1px 3px;\n        width: max-content;\n      }\n      .has-inf-or-nan[_ngcontent-%COMP%] {\n        background-color: #e52592;\n        color: #fff;\n      }"] });
    class DebugTensorValueComponent {
    }
    DebugTensorValueComponent.decorators = [
        { type: Component, args: [{
                    selector: 'debug-tensor-value',
                    template: `
    <debug-tensor-dtype
      *ngIf="debugTensorValue.dtype !== undefined"
      [dtype]="debugTensorValue.dtype"
    >
    </debug-tensor-dtype>
    <debug-tensor-rank
      *ngIf="debugTensorValue.rank !== undefined"
      [rank]="debugTensorValue.rank"
    >
    </debug-tensor-rank>
    <debug-tensor-shape
      *ngIf="debugTensorValue.shape !== undefined"
      [shape]="debugTensorValue.shape"
    >
    </debug-tensor-shape>
    <debug-tensor-has-inf-or-nan
      *ngIf="debugTensorValue.hasInfOrNaN !== undefined"
      [hasInfOrNaN]="debugTensorValue.hasInfOrNaN"
    >
    </debug-tensor-has-inf-or-nan>
    <debug-tensor-numeric-breakdown
      *ngIf="debugTensorValue.size !== undefined"
      size="{{ debugTensorValue.size }}"
      [numNegativeInfs]="debugTensorValue.numNegativeInfs"
      [numPositiveInfs]="debugTensorValue.numPositiveInfs"
      [numNaNs]="debugTensorValue.numNaNs"
      [numNegativeFinites]="debugTensorValue.numNegativeFinites"
      [numZeros]="debugTensorValue.numZeros"
      [numPositiveFinites]="debugTensorValue.numPositiveFinites"
    >
    </debug-tensor-numeric-breakdown>
  `,
                    styles: [
                        `
      :host {
        align-items: flex-start;
        display: flex;
        flex-wrap: nowrap;
        overflow: hidden;
        vertical-align: top;
      }
      debug-tensor-numeric-breakdown {
        display: inline-block;
      }
    `,
                    ],
                },] },
    ];
    DebugTensorValueComponent.propDecorators = {
        debugTensorValue: [{ type: Input }]
    };
    /** @nocollapse */ DebugTensorValueComponent.ɵfac = function DebugTensorValueComponent_Factory(t) { return new (t || DebugTensorValueComponent)(); };
    /** @nocollapse */ DebugTensorValueComponent.ɵcmp = ɵɵdefineComponent({ type: DebugTensorValueComponent, selectors: [["debug-tensor-value"]], inputs: { debugTensorValue: "debugTensorValue" }, decls: 5, vars: 5, consts: [[3, "dtype", 4, "ngIf"], [3, "rank", 4, "ngIf"], [3, "shape", 4, "ngIf"], [3, "hasInfOrNaN", 4, "ngIf"], [3, "size", "numNegativeInfs", "numPositiveInfs", "numNaNs", "numNegativeFinites", "numZeros", "numPositiveFinites", 4, "ngIf"], [3, "dtype"], [3, "rank"], [3, "shape"], [3, "hasInfOrNaN"], [3, "size", "numNegativeInfs", "numPositiveInfs", "numNaNs", "numNegativeFinites", "numZeros", "numPositiveFinites"]], template: function DebugTensorValueComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵtemplate(0, DebugTensorValueComponent_debug_tensor_dtype_0_Template, 1, 1, "debug-tensor-dtype", 0);
            ɵɵtemplate(1, DebugTensorValueComponent_debug_tensor_rank_1_Template, 1, 1, "debug-tensor-rank", 1);
            ɵɵtemplate(2, DebugTensorValueComponent_debug_tensor_shape_2_Template, 1, 1, "debug-tensor-shape", 2);
            ɵɵtemplate(3, DebugTensorValueComponent_debug_tensor_has_inf_or_nan_3_Template, 1, 1, "debug-tensor-has-inf-or-nan", 3);
            ɵɵtemplate(4, DebugTensorValueComponent_debug_tensor_numeric_breakdown_4_Template, 1, 7, "debug-tensor-numeric-breakdown", 4);
        } if (rf & 2) {
            ɵɵproperty("ngIf", ctx.debugTensorValue.dtype !== undefined);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx.debugTensorValue.rank !== undefined);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx.debugTensorValue.shape !== undefined);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx.debugTensorValue.hasInfOrNaN !== undefined);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx.debugTensorValue.size !== undefined);
        } }, directives: [NgIf, DebugTensorDTypeComponent,
            DebugTensorRankComponent,
            DebugTensorShapeComponent,
            DebugTensorHasInfOrNaNComponent,
            DebugTensorNumericBreakdownComponent], styles: ["[_nghost-%COMP%] {\n        align-items: flex-start;\n        display: flex;\n        flex-wrap: nowrap;\n        overflow: hidden;\n        vertical-align: top;\n      }\n      debug-tensor-numeric-breakdown[_ngcontent-%COMP%] {\n        display: inline-block;\n      }"] });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/views/execution_data/execution_data_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    const _c0$e = function (a0, a1) { return { tensorDebugMode: a0, array: a1 }; };
    function ExecutionDataComponent_div_4_div_19_div_1_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 12);
        ɵɵelementStart(1, "div", 13);
        ɵɵtext(2);
        ɵɵelementEnd();
        ɵɵelementStart(3, "div", 14);
        ɵɵelement(4, "debug-tensor-value", 15);
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const singleDebugTensorValues_r5 = ctx.$implicit;
        const i_r6 = ctx.index;
        const ctx_r4 = ɵɵnextContext(3);
        ɵɵadvance(2);
        ɵɵtextInterpolate1("Output slot ", i_r6, ":");
        ɵɵadvance(2);
        ɵɵproperty("debugTensorValue", ctx_r4.parseDebugTensorValue(ɵɵpureFunction2(2, _c0$e, ctx_r4.tensorDebugMode, singleDebugTensorValues_r5)));
    } }
    function ExecutionDataComponent_div_4_div_19_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 10);
        ɵɵtemplate(1, ExecutionDataComponent_div_4_div_19_div_1_Template, 5, 5, "div", 11);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r3 = ɵɵnextContext(2);
        ɵɵadvance(1);
        ɵɵproperty("ngForOf", ctx_r3.debugTensorValues);
    } }
    function ExecutionDataComponent_div_4_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div");
        ɵɵelementStart(1, "div");
        ɵɵelementStart(2, "div", 3);
        ɵɵelementStart(3, "span", 4);
        ɵɵtext(4, " Op: ");
        ɵɵelementEnd();
        ɵɵelementStart(5, "span", 5);
        ɵɵtext(6);
        ɵɵelementEnd();
        ɵɵelementEnd();
        ɵɵelementStart(7, "div", 3);
        ɵɵelementStart(8, "span", 4);
        ɵɵtext(9, " # of input tensors: ");
        ɵɵelementEnd();
        ɵɵelementStart(10, "span", 6);
        ɵɵtext(11);
        ɵɵelementEnd();
        ɵɵelementEnd();
        ɵɵelementStart(12, "div", 3);
        ɵɵelementStart(13, "span", 4);
        ɵɵtext(14, " # of output tensors: ");
        ɵɵelementEnd();
        ɵɵelementStart(15, "span", 7);
        ɵɵtext(16);
        ɵɵelementEnd();
        ɵɵelementStart(17, "span", 8);
        ɵɵtext(18);
        ɵɵelementEnd();
        ɵɵelementEnd();
        ɵɵtemplate(19, ExecutionDataComponent_div_4_div_19_Template, 2, 1, "div", 9);
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r0 = ɵɵnextContext();
        ɵɵadvance(6);
        ɵɵtextInterpolate1(" ", ctx_r0.focusedExecutionData.op_type, " ");
        ɵɵadvance(5);
        ɵɵtextInterpolate1(" ", ctx_r0.focusedExecutionData.input_tensor_ids == null ? 0 : ctx_r0.focusedExecutionData.input_tensor_ids.length, " ");
        ɵɵadvance(5);
        ɵɵtextInterpolate1(" ", ctx_r0.focusedExecutionData.output_tensor_ids == null ? 0 : ctx_r0.focusedExecutionData.output_tensor_ids.length, " ");
        ɵɵadvance(2);
        ɵɵtextInterpolate1(" (debug mode: ", ctx_r0.TensorDebugMode[ctx_r0.tensorDebugMode], ") ");
        ɵɵadvance(1);
        ɵɵproperty("ngIf", ctx_r0.hasDebugTensorValues);
    } }
    function ExecutionDataComponent_ng_template_5_Template(rf, ctx) { }
    class ExecutionDataComponent {
        constructor() {
            this.tensorDebugMode = TensorDebugMode.UNSPECIFIED;
            /**
             * Whether any debug tensor values exist, under non-FULL_TENSOR debug
             * modes.
             */
            this.hasDebugTensorValues = false;
            /**
             * Debug tensor values under non-FULL_TENSOR debug modes.
             */
            this.debugTensorValues = null;
            /**
             * Dtypes of the tensors.
             * Dtypes are available under tensor debug modes such as FULL_HEALTH.
             */
            this.debugTensorDtypes = null;
            // So that the enum can be used in the template html.
            this.TensorDebugMode = TensorDebugMode;
            this.parseDebugTensorValue = parseDebugTensorValue;
        }
    }
    ExecutionDataComponent.decorators = [
        { type: Component, args: [{
                    selector: 'execution-data-component',
                    templateUrl: './execution_data_component.ng.html',
                    styleUrls: ['./execution_data_component.css'],
                },] },
    ];
    ExecutionDataComponent.propDecorators = {
        focusedExecutionIndex: [{ type: Input }],
        focusedExecutionData: [{ type: Input }],
        tensorDebugMode: [{ type: Input }],
        hasDebugTensorValues: [{ type: Input }],
        debugTensorValues: [{ type: Input }],
        debugTensorDtypes: [{ type: Input }]
    };
    /** @nocollapse */ ExecutionDataComponent.ɵfac = function ExecutionDataComponent_Factory(t) { return new (t || ExecutionDataComponent)(); };
    /** @nocollapse */ ExecutionDataComponent.ɵcmp = ɵɵdefineComponent({ type: ExecutionDataComponent, selectors: [["execution-data-component"]], inputs: { focusedExecutionIndex: "focusedExecutionIndex", focusedExecutionData: "focusedExecutionData", tensorDebugMode: "tensorDebugMode", hasDebugTensorValues: "hasDebugTensorValues", debugTensorValues: "debugTensorValues", debugTensorDtypes: "debugTensorDtypes" }, decls: 7, vars: 3, consts: [[1, "focus-execution-container"], [4, "ngIf", "ngIfElse"], ["loading_section", ""], [1, "execution-data-field"], [1, "execution-data-key"], [1, "execution-data-value", "op-type"], [1, "execution-data-value", "input-tensors"], [1, "execution-data-value", "output-tensors"], [1, "execution-data-value"], ["class", "output-slots", 4, "ngIf"], [1, "output-slots"], ["class", "output-slot-container", 4, "ngFor", "ngForOf"], [1, "output-slot-container"], [1, "output-slot-number"], [1, "output-slot-debug-tensor-value"], [3, "debugTensorValue"]], template: function ExecutionDataComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "div", 0);
            ɵɵelementStart(1, "div");
            ɵɵelementStart(2, "span");
            ɵɵtext(3);
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵtemplate(4, ExecutionDataComponent_div_4_Template, 20, 5, "div", 1);
            ɵɵtemplate(5, ExecutionDataComponent_ng_template_5_Template, 0, 0, "ng-template", null, 2, ɵɵtemplateRefExtractor);
            ɵɵelementEnd();
        } if (rf & 2) {
            const _r1 = ɵɵreference(6);
            ɵɵadvance(3);
            ɵɵtextInterpolate1(" Python Execution #", ctx.focusedExecutionIndex, " ");
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx.focusedExecutionData !== null)("ngIfElse", _r1);
        } }, directives: [NgIf, NgForOf, DebugTensorValueComponent], styles: [".debug-tensor-values-table[_ngcontent-%COMP%] {\n  width: 100%;\n}\n\n.debug-tensor-values-table[_ngcontent-%COMP%]   td[_ngcontent-%COMP%] {\n  border-top: 1px solid #000;\n  text-align: left;\n}\n\n.debug-tensor-values-table[_ngcontent-%COMP%]   th[_ngcontent-%COMP%] {\n  text-align: left;\n}\n\n.execution-data-field[_ngcontent-%COMP%] {\n  white-space: nowrap;\n}\n\n.execution-data-key[_ngcontent-%COMP%] {\n  display: inline-block;\n  max-width: 120px;\n  text-align: right;\n  width: 120px;\n}\n\n.execution-data-value[_ngcontent-%COMP%] {\n  display: inline-block;\n  margin-left: 10px;\n}\n\n.focus-execution-container[_ngcontent-%COMP%] {\n  background-color: #ffd4b3;\n  border-radius: 4px;\n  font-size: 12px;\n  height: 120px;\n  padding: 5px;\n  width: 360px;\n}\n\n.output-slots[_ngcontent-%COMP%] {\n  height: 60px;\n  overflow-x: auto;\n  overflow-y: auto;\n}\n\n.output-slot-container[_ngcontent-%COMP%] {\n  border-top: 1px solid rgba(0, 0, 0, 0.12);\n  margin-top: 5px;\n  padding: 2px 0;\n  vertical-align: top;\n}\n\n.output-slot-number[_ngcontent-%COMP%] {\n  display: block;\n  font-family: 'Roboto Mono', monospace;\n}\n\n.output-slot-debug-tensor-value[_ngcontent-%COMP%] {\n  display: block;\n  margin: 3px 0 3px 30px;\n}\n\n.output-tensors[_ngcontent-%COMP%] {\n  margin-top: 5px;\n}"] });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/views/execution_data/execution_data_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const UNKNOWN_DTYPE_NAME = 'Unknown dtype';
    class ExecutionDataContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.focusedExecutionData$ = this.store.pipe(select(getFocusedExecutionData));
            this.tensorDebugMode$ = this.store.pipe(select(createSelector(getFocusedExecutionData, (/**
             * @param {?} execution
             * @return {?}
             */
            (execution) => {
                if (execution === null) {
                    return TensorDebugMode.UNSPECIFIED;
                }
                else {
                    return execution.tensor_debug_mode;
                }
            }))));
            this.hasDebugTensorValues$ = this.store.pipe(select(createSelector(getFocusedExecutionData, (/**
             * @param {?} execution
             * @return {?}
             */
            (execution) => {
                if (execution === null || execution.debug_tensor_values === null) {
                    return false;
                }
                else {
                    for (const singleDebugTensorValues of execution.debug_tensor_values) {
                        if (singleDebugTensorValues !== null &&
                            singleDebugTensorValues.length > 0) {
                            return true;
                        }
                    }
                    return false;
                }
            }))));
            this.debugTensorValues$ = this.store.pipe(select(createSelector(getFocusedExecutionData, (/**
             * @param {?} execution
             * @return {?}
             */
            (execution) => {
                if (execution === null) {
                    return null;
                }
                else {
                    return execution.debug_tensor_values;
                }
            }))));
            this.debugTensorDtypes$ = this.store.pipe(select(createSelector(getFocusedExecutionData, (/**
             * @param {?} execution
             * @return {?}
             */
            (execution) => {
                if (execution === null || execution.debug_tensor_values === null) {
                    return null;
                }
                if (execution.tensor_debug_mode !== TensorDebugMode.FULL_HEALTH &&
                    execution.tensor_debug_mode !== TensorDebugMode.SHAPE) {
                    // TODO(cais): Add logic for other TensorDebugModes with dtype info.
                    return null;
                }
                /** @type {?} */
                const dtypes = [];
                for (const tensorValue of execution.debug_tensor_values) {
                    if (tensorValue === null) {
                        dtypes.push(UNKNOWN_DTYPE_NAME);
                    }
                    else {
                        /** @type {?} */
                        const dtypeEnum = String(execution.tensor_debug_mode === TensorDebugMode.FULL_HEALTH
                            ? tensorValue[2] // tensor_debug_mode: FULL_HEALTH
                            : tensorValue[1] // tensor_debug_mode: SHAPE
                        );
                        dtypes.push(DTYPE_ENUM_TO_NAME[dtypeEnum] || UNKNOWN_DTYPE_NAME);
                    }
                }
                return dtypes;
            }))));
        }
    }
    ExecutionDataContainer.decorators = [
        { type: Component, args: [{
                    selector: 'tf-debugger-v2-execution-data',
                    template: `
    <execution-data-component
      [focusedExecutionIndex]="focusedExecutionIndex"
      [focusedExecutionData]="focusedExecutionData$ | async"
      [tensorDebugMode]="tensorDebugMode$ | async"
      [hasDebugTensorValues]="hasDebugTensorValues$ | async"
      [debugTensorValues]="debugTensorValues$ | async"
      [debugTensorDtypes]="debugTensorDtypes$ | async"
    ></execution-data-component>
  `,
                },] },
    ];
    /** @nocollapse */
    ExecutionDataContainer.ctorParameters = () => [
        { type: Store }
    ];
    ExecutionDataContainer.propDecorators = {
        focusedExecutionIndex: [{ type: Input }]
    };
    /** @nocollapse */ ExecutionDataContainer.ɵfac = function ExecutionDataContainer_Factory(t) { return new (t || ExecutionDataContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ ExecutionDataContainer.ɵcmp = ɵɵdefineComponent({ type: ExecutionDataContainer, selectors: [["tf-debugger-v2-execution-data"]], inputs: { focusedExecutionIndex: "focusedExecutionIndex" }, decls: 6, vars: 16, consts: [[3, "focusedExecutionIndex", "focusedExecutionData", "tensorDebugMode", "hasDebugTensorValues", "debugTensorValues", "debugTensorDtypes"]], template: function ExecutionDataContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelement(0, "execution-data-component", 0);
            ɵɵpipe(1, "async");
            ɵɵpipe(2, "async");
            ɵɵpipe(3, "async");
            ɵɵpipe(4, "async");
            ɵɵpipe(5, "async");
        } if (rf & 2) {
            ɵɵproperty("focusedExecutionIndex", ctx.focusedExecutionIndex)("focusedExecutionData", ɵɵpipeBind1(1, 6, ctx.focusedExecutionData$))("tensorDebugMode", ɵɵpipeBind1(2, 8, ctx.tensorDebugMode$))("hasDebugTensorValues", ɵɵpipeBind1(3, 10, ctx.hasDebugTensorValues$))("debugTensorValues", ɵɵpipeBind1(4, 12, ctx.debugTensorValues$))("debugTensorDtypes", ɵɵpipeBind1(5, 14, ctx.debugTensorDtypes$));
        } }, directives: [ExecutionDataComponent], pipes: [AsyncPipe], encapsulation: 2 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/views/timeline/timeline_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    function TimelineComponent_div_6_mat_slider_7_Template(rf, ctx) { if (rf & 1) {
        const _r5 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "mat-slider", 11);
        ɵɵlistener("input", function TimelineComponent_div_6_mat_slider_7_Template_mat_slider_input_0_listener($event) { ɵɵrestoreView(_r5); const ctx_r4 = ɵɵnextContext(2); return ctx_r4.onSliderChange.emit($event.value); });
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r3 = ɵɵnextContext(2);
        ɵɵproperty("min", 0)("max", ctx_r3.scrollBeginIndexUpperLimit)("value", ctx_r3.scrollBeginIndex);
    } }
    function TimelineComponent_div_6_Template(rf, ctx) { if (rf & 1) {
        const _r7 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "div", 6);
        ɵɵelementStart(1, "button", 7);
        ɵɵlistener("click", function TimelineComponent_div_6_Template_button_click_1_listener() { ɵɵrestoreView(_r7); const ctx_r6 = ɵɵnextContext(); return ctx_r6.onNavigateLeft.emit(); });
        ɵɵtext(2, " < ");
        ɵɵelementEnd();
        ɵɵelementStart(3, "div", 8);
        ɵɵtext(4);
        ɵɵelementEnd();
        ɵɵelementStart(5, "button", 9);
        ɵɵlistener("click", function TimelineComponent_div_6_Template_button_click_5_listener() { ɵɵrestoreView(_r7); const ctx_r8 = ɵɵnextContext(); return ctx_r8.onNavigateRight.emit(); });
        ɵɵtext(6, " > ");
        ɵɵelementEnd();
        ɵɵtemplate(7, TimelineComponent_div_6_mat_slider_7_Template, 1, 3, "mat-slider", 10);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r0 = ɵɵnextContext();
        ɵɵadvance(4);
        ɵɵtextInterpolate3(" ", ctx_r0.scrollBeginIndex, " ~ ", ctx_r0.scrollBeginIndex + ctx_r0.displayCount - 1, " of ", ctx_r0.numExecutions, " ");
        ɵɵadvance(3);
        ɵɵproperty("ngIf", ctx_r0.scrollBeginIndexUpperLimit > 0);
    } }
    const _c0$f = function (a0, a1, a2) { return [a0, a1, a2]; };
    function TimelineComponent_div_7_div_1_Template(rf, ctx) { if (rf & 1) {
        const _r13 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "div", 14);
        ɵɵlistener("click", function TimelineComponent_div_7_div_1_Template_div_click_0_listener() { ɵɵrestoreView(_r13); const i_r11 = ctx.index; const ctx_r12 = ɵɵnextContext(2); return ctx_r12.onExecutionDigestClicked.emit(i_r11); });
        ɵɵelementStart(1, "div", 15);
        ɵɵtext(2);
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const displayDigest_r10 = ctx.$implicit;
        const i_r11 = ctx.index;
        const ctx_r9 = ɵɵnextContext(2);
        ɵɵadvance(1);
        ɵɵpropertyInterpolate("title", displayDigest_r10.op_type);
        ɵɵproperty("ngClass", ɵɵpureFunction3(3, _c0$f, displayDigest_r10.is_graph ? "func-graph-execution" : "", i_r11 === ctx_r9.focusedExecutionDisplayIndex ? "focused" : "", ctx_r9.displayFocusedAlertTypes[i_r11] || ""));
        ɵɵadvance(1);
        ɵɵtextInterpolate1(" ", displayDigest_r10.short_op_type, " ");
    } }
    function TimelineComponent_div_7_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 12);
        ɵɵtemplate(1, TimelineComponent_div_7_div_1_Template, 3, 7, "div", 13);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r1 = ɵɵnextContext();
        ɵɵadvance(1);
        ɵɵproperty("ngForOf", ctx_r1.displayExecutionDigests);
    } }
    function TimelineComponent_ng_container_8_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementContainerStart(0);
        ɵɵelement(1, "tf-debugger-v2-execution-data", 16);
        ɵɵelementContainerEnd();
    } if (rf & 2) {
        const ctx_r2 = ɵɵnextContext();
        ɵɵadvance(1);
        ɵɵproperty("focusedExecutionIndex", ctx_r2.focusedExecutionIndex);
    } }
    class TimelineComponent {
        constructor() {
            this.activeRunId = null;
            this.loadingNumExecutions = false;
            this.numExecutions = 0;
            this.scrollBeginIndex = 0;
            this.scrollBeginIndexUpperLimit = 0;
            this.pageSize = 0;
            this.displayCount = 0;
            this.displayExecutionDigests = [];
            this.displayFocusedAlertTypes = [];
            this.focusedExecutionIndex = null;
            this.focusedExecutionDisplayIndex = null;
            this.focusedExecutionData = null;
            this.onNavigateLeft = new EventEmitter();
            this.onNavigateRight = new EventEmitter();
            this.onExecutionDigestClicked = new EventEmitter();
            this.onSliderChange = new EventEmitter();
        }
    }
    TimelineComponent.decorators = [
        { type: Component, args: [{
                    selector: 'timeline-component',
                    templateUrl: './timeline_component.ng.html',
                    styleUrls: ['./timeline_component.css'],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    TimelineComponent.propDecorators = {
        activeRunId: [{ type: Input }],
        loadingNumExecutions: [{ type: Input }],
        numExecutions: [{ type: Input }],
        scrollBeginIndex: [{ type: Input }],
        scrollBeginIndexUpperLimit: [{ type: Input }],
        pageSize: [{ type: Input }],
        displayCount: [{ type: Input }],
        displayExecutionDigests: [{ type: Input }],
        displayFocusedAlertTypes: [{ type: Input }],
        focusedExecutionIndex: [{ type: Input }],
        focusedExecutionDisplayIndex: [{ type: Input }],
        focusedExecutionData: [{ type: Input }],
        onNavigateLeft: [{ type: Output }],
        onNavigateRight: [{ type: Output }],
        onExecutionDigestClicked: [{ type: Output }],
        onSliderChange: [{ type: Output }]
    };
    /** @nocollapse */ TimelineComponent.ɵfac = function TimelineComponent_Factory(t) { return new (t || TimelineComponent)(); };
    /** @nocollapse */ TimelineComponent.ɵcmp = ɵɵdefineComponent({ type: TimelineComponent, selectors: [["timeline-component"]], inputs: { activeRunId: "activeRunId", loadingNumExecutions: "loadingNumExecutions", numExecutions: "numExecutions", scrollBeginIndex: "scrollBeginIndex", scrollBeginIndexUpperLimit: "scrollBeginIndexUpperLimit", pageSize: "pageSize", displayCount: "displayCount", displayExecutionDigests: "displayExecutionDigests", displayFocusedAlertTypes: "displayFocusedAlertTypes", focusedExecutionIndex: "focusedExecutionIndex", focusedExecutionDisplayIndex: "focusedExecutionDisplayIndex", focusedExecutionData: "focusedExecutionData" }, outputs: { onNavigateLeft: "onNavigateLeft", onNavigateRight: "onNavigateRight", onExecutionDigestClicked: "onExecutionDigestClicked", onSliderChange: "onSliderChange" }, decls: 9, vars: 4, consts: [[1, "timeline-title"], [1, "execution-count"], [1, "top-level-executions"], ["class", "navigation-section", 4, "ngIf"], ["class", "execution-timeline", 4, "ngIf"], [4, "ngIf"], [1, "navigation-section"], ["mat-button", "", 1, "navigation-button-left", 3, "click"], [1, "navigation-position-info"], ["mat-button", "", 1, "navigation-button-right", 3, "click"], ["class", "timeline-slider", "step", "1", 3, "min", "max", "value", "input", 4, "ngIf"], ["step", "1", 1, "timeline-slider", 3, "min", "max", "value", "input"], [1, "execution-timeline"], [3, "click", 4, "ngFor", "ngForOf"], [3, "click"], [1, "execution-digest", 3, "ngClass", "title"], [3, "focusedExecutionIndex"]], template: function TimelineComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "div");
            ɵɵelementStart(1, "div", 0);
            ɵɵtext(2, " Python Execution Timeline ");
            ɵɵelementStart(3, "span", 1);
            ɵɵtext(4);
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementStart(5, "div", 2);
            ɵɵtemplate(6, TimelineComponent_div_6_Template, 8, 4, "div", 3);
            ɵɵtemplate(7, TimelineComponent_div_7_Template, 2, 1, "div", 4);
            ɵɵtemplate(8, TimelineComponent_ng_container_8_Template, 2, 1, "ng-container", 5);
            ɵɵelementEnd();
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵadvance(4);
            ɵɵtextInterpolate1(" (", ctx.numExecutions, ") ");
            ɵɵadvance(2);
            ɵɵproperty("ngIf", ctx.numExecutions);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx.numExecutions);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx.activeRunId !== null && ctx.focusedExecutionIndex !== null);
        } }, directives: [NgIf, MatButton, MatSlider, NgForOf, NgClass, ExecutionDataContainer], styles: [".execution-digest[_ngcontent-%COMP%] {\n  background-color: #e3e5e8;\n  border: 1px solid #c0c0c0;\n  color: #425066;\n  display: inline-block;\n  font-size: 10px;\n  height: 15px;\n  padding: 1px;\n  text-align: center;\n  vertical-align: middle;\n  width: 12px;\n}\n\n.execution-digest.func-graph-execution[_ngcontent-%COMP%] {\n  background-color: #c7dbf5;\n  color: #4e5664;\n  text-decoration: underline;\n}\n\n.execution-digest.focused[_ngcontent-%COMP%] {\n  background-color: #ffd4b3;\n  border: 1px solid #000;\n  font-weight: bold;\n}\n\n.execution-digest.InfNanAlert[_ngcontent-%COMP%] {\n  background-color: #e52592;\n  color: #fff;\n}\n\n\n.execution-digest[_ngcontent-%COMP%]:hover {\n  border: 1px solid #000;\n  font-weight: bold;\n}\n\n.execution-timeline[_ngcontent-%COMP%] {\n  display: flex;\n  overflow-x: hidden;\n  white-space: nowrap;\n  width: 100%;\n  margin-top: 5px;\n  margin-bottom: 5px;\n}\n\n.timeline-slider[_ngcontent-%COMP%] {\n  display: inline-block;\n  height: 48px;\n  left: 340px; \n  padding: 0;\n  position: absolute;\n  right: 40px;\n}\n\n  .timeline-slider .mat-slider-thumb {\n  border-radius: 5px;\n  right: -40px;\n  width: 80px;\n}\n\n\n.navigation-position-info[_ngcontent-%COMP%] {\n  display: inline-flex;\n  font-size: 14px;\n  line-height: normal;\n  max-width: 200px;\n  padding-left: 10px;\n  padding-right: 10px;\n  text-align: center;\n  vertical-align: middle;\n}\n\n.navigation-section[_ngcontent-%COMP%] {\n  height: 48px;\n  line-height: 48px;\n  position: relative;\n  vertical-align: middle;\n  width: 100%;\n}"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/views/timeline/timeline_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const FUNCTION_OP_TYPE_PREFIXES = [
        '__forward_',
        '__backward_',
        '__inference_',
    ];
    /**
     * Get a display version of the execution digest.
     * @param {?} executionDigest
     * @param {?=} strLen
     * @return {?}
     */
    function getExecutionDigestForDisplay(executionDigest, strLen = 1) {
        if (!executionDigest) {
            // The execution digest at this index hasn't been loaded from the data source.
            return {
                op_type: '(N/A)',
                short_op_type: '..',
                is_graph: false,
            };
        }
        /** @type {?} */
        const functionPrefixes = FUNCTION_OP_TYPE_PREFIXES.filter((/**
         * @param {?} prefix
         * @return {?}
         */
        (prefix) => executionDigest.op_type.startsWith(prefix)));
        if (functionPrefixes.length) {
            // This is the execution of a tf.function (FuncGraph).
            /** @type {?} */
            const functionNameWithSuffix = executionDigest.op_type.slice(functionPrefixes[0].length);
            return {
                op_type: executionDigest.op_type,
                short_op_type: functionNameWithSuffix.slice(0, strLen),
                is_graph: true,
            };
        }
        else {
            return {
                op_type: executionDigest.op_type,
                short_op_type: executionDigest.op_type.slice(0, strLen),
                is_graph: false,
            };
        }
    }
    class TimelineContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.activeRunId$ = this.store.pipe(select(getActiveRunId));
            this.loadingNumExecutions$ = this.store.pipe(select(createSelector(getNumExecutionsLoaded, (/**
             * @param {?} loaded
             * @return {?}
             */
            (loaded) => {
                return loaded.state == DataLoadState.LOADING;
            }))));
            this.scrollBeginIndex$ = this.store.pipe(select(getExecutionScrollBeginIndex));
            this.scrollBeginIndexUpperLimit$ = this.store.pipe(select(createSelector(getNumExecutions, getDisplayCount, (/**
             * @param {?} numExecutions
             * @param {?} displayCount
             * @return {?}
             */
            (numExecutions, displayCount) => {
                return Math.max(0, numExecutions - displayCount);
            }))));
            this.pageSize$ = this.store.pipe(select(getExecutionPageSize));
            this.displayCount$ = this.store.pipe(select(getDisplayCount));
            this.displayExecutionDigests$ = this.store.pipe(select(createSelector(getVisibleExecutionDigests, (/**
             * @param {?} visibleDigests
             * @return {?}
             */
            (visibleDigests) => {
                return visibleDigests.map((/**
                 * @param {?} digest
                 * @return {?}
                 */
                (digest) => getExecutionDigestForDisplay(digest)));
            }))));
            this.displayFocusedAlertTypes$ = this.store.pipe(select(getFocusAlertTypesOfVisibleExecutionDigests));
            this.focusedExecutionIndex$ = this.store.pipe(select(getFocusedExecutionIndex));
            this.focusedExecutionDisplayIndex$ = this.store.pipe(select(getFocusedExecutionDisplayIndex));
            this.numExecutions$ = this.store.pipe(select(getNumExecutions));
        }
        /**
         * @return {?}
         */
        onNavigateLeft() {
            this.store.dispatch(executionScrollLeft());
        }
        /**
         * @return {?}
         */
        onNavigateRight() {
            this.store.dispatch(executionScrollRight());
        }
        /**
         * @param {?} index
         * @return {?}
         */
        onExecutionDigestClicked(index) {
            this.store.dispatch(executionDigestFocused({ displayIndex: index }));
        }
        /**
         * @param {?} value
         * @return {?}
         */
        onSliderChange(value) {
            this.store.dispatch(executionScrollToIndex({ index: value }));
        }
    }
    TimelineContainer.decorators = [
        { type: Component, args: [{
                    selector: 'tf-debugger-v2-timeline',
                    template: `
    <timeline-component
      [activeRunId]="activeRunId$ | async"
      [loadingNumExecutions]="loadingNumExecutions$ | async"
      [numExecutions]="numExecutions$ | async"
      [scrollBeginIndex]="scrollBeginIndex$ | async"
      [scrollBeginIndexUpperLimit]="scrollBeginIndexUpperLimit$ | async"
      [pageSize]="pageSize$ | async"
      [displayCount]="displayCount$ | async"
      [displayExecutionDigests]="displayExecutionDigests$ | async"
      [displayFocusedAlertTypes]="displayFocusedAlertTypes$ | async"
      [focusedExecutionIndex]="focusedExecutionIndex$ | async"
      [focusedExecutionDisplayIndex]="focusedExecutionDisplayIndex$ | async"
      (onNavigateLeft)="onNavigateLeft()"
      (onNavigateRight)="onNavigateRight()"
      (onExecutionDigestClicked)="onExecutionDigestClicked($event)"
      (onSliderChange)="onSliderChange($event)"
    ></timeline-component>
  `,
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    TimelineContainer.ctorParameters = () => [
        { type: Store }
    ];
    /** @nocollapse */ TimelineContainer.ɵfac = function TimelineContainer_Factory(t) { return new (t || TimelineContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ TimelineContainer.ɵcmp = ɵɵdefineComponent({ type: TimelineContainer, selectors: [["tf-debugger-v2-timeline"]], decls: 12, vars: 33, consts: [[3, "activeRunId", "loadingNumExecutions", "numExecutions", "scrollBeginIndex", "scrollBeginIndexUpperLimit", "pageSize", "displayCount", "displayExecutionDigests", "displayFocusedAlertTypes", "focusedExecutionIndex", "focusedExecutionDisplayIndex", "onNavigateLeft", "onNavigateRight", "onExecutionDigestClicked", "onSliderChange"]], template: function TimelineContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "timeline-component", 0);
            ɵɵlistener("onNavigateLeft", function TimelineContainer_Template_timeline_component_onNavigateLeft_0_listener() { return ctx.onNavigateLeft(); })("onNavigateRight", function TimelineContainer_Template_timeline_component_onNavigateRight_0_listener() { return ctx.onNavigateRight(); })("onExecutionDigestClicked", function TimelineContainer_Template_timeline_component_onExecutionDigestClicked_0_listener($event) { return ctx.onExecutionDigestClicked($event); })("onSliderChange", function TimelineContainer_Template_timeline_component_onSliderChange_0_listener($event) { return ctx.onSliderChange($event); });
            ɵɵpipe(1, "async");
            ɵɵpipe(2, "async");
            ɵɵpipe(3, "async");
            ɵɵpipe(4, "async");
            ɵɵpipe(5, "async");
            ɵɵpipe(6, "async");
            ɵɵpipe(7, "async");
            ɵɵpipe(8, "async");
            ɵɵpipe(9, "async");
            ɵɵpipe(10, "async");
            ɵɵpipe(11, "async");
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵproperty("activeRunId", ɵɵpipeBind1(1, 11, ctx.activeRunId$))("loadingNumExecutions", ɵɵpipeBind1(2, 13, ctx.loadingNumExecutions$))("numExecutions", ɵɵpipeBind1(3, 15, ctx.numExecutions$))("scrollBeginIndex", ɵɵpipeBind1(4, 17, ctx.scrollBeginIndex$))("scrollBeginIndexUpperLimit", ɵɵpipeBind1(5, 19, ctx.scrollBeginIndexUpperLimit$))("pageSize", ɵɵpipeBind1(6, 21, ctx.pageSize$))("displayCount", ɵɵpipeBind1(7, 23, ctx.displayCount$))("displayExecutionDigests", ɵɵpipeBind1(8, 25, ctx.displayExecutionDigests$))("displayFocusedAlertTypes", ɵɵpipeBind1(9, 27, ctx.displayFocusedAlertTypes$))("focusedExecutionIndex", ɵɵpipeBind1(10, 29, ctx.focusedExecutionIndex$))("focusedExecutionDisplayIndex", ɵɵpipeBind1(11, 31, ctx.focusedExecutionDisplayIndex$));
        } }, directives: [TimelineComponent], pipes: [AsyncPipe], encapsulation: 2, changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/views/graph/graph_op_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    function GraphOpComponent_div_5_span_2_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "span");
        ɵɵtext(1, " Output ");
        ɵɵelementEnd();
    } }
    function GraphOpComponent_div_5_span_3_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "span");
        ɵɵtext(1, " Input ");
        ɵɵelementEnd();
    } }
    function GraphOpComponent_div_5_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 6);
        ɵɵelementStart(1, "span", 7);
        ɵɵtemplate(2, GraphOpComponent_div_5_span_2_Template, 2, 0, "span", 8);
        ɵɵtemplate(3, GraphOpComponent_div_5_span_3_Template, 2, 0, "span", 8);
        ɵɵelementEnd();
        ɵɵtext(4);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r0 = ɵɵnextContext();
        ɵɵadvance(1);
        ɵɵproperty("ngSwitch", ctx_r0.kind);
        ɵɵadvance(1);
        ɵɵproperty("ngSwitchCase", "input");
        ɵɵadvance(1);
        ɵɵproperty("ngSwitchCase", "consumer");
        ɵɵadvance(1);
        ɵɵtextInterpolate1(" slot: ", ctx_r0.slot, " ");
    } }
    function GraphOpComponent_div_6_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 9);
        ɵɵtext(1);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r1 = ɵɵnextContext();
        ɵɵadvance(1);
        ɵɵtextInterpolate1(" ", ctx_r1.opData.op_type, " ");
    } }
    function GraphOpComponent_ng_template_7_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "span", 10);
        ɵɵtext(1, " (Op info unavailable.) ");
        ɵɵelementEnd();
    } }
    const _c0$g = function (a0) { return [a0]; };
    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    class GraphOpComponent {
        constructor() {
            this.onOpNameClick = new EventEmitter();
        }
    }
    GraphOpComponent.decorators = [
        { type: Component, args: [{
                    selector: 'graph-op',
                    templateUrl: 'graph_op_component.ng.html',
                    styleUrls: ['./graph_op_component.css'],
                },] },
    ];
    GraphOpComponent.propDecorators = {
        kind: [{ type: Input }],
        opName: [{ type: Input }],
        slot: [{ type: Input }],
        opData: [{ type: Input }],
        onOpNameClick: [{ type: Output }]
    };
    /** @nocollapse */ GraphOpComponent.ɵfac = function GraphOpComponent_Factory(t) { return new (t || GraphOpComponent)(); };
    /** @nocollapse */ GraphOpComponent.ɵcmp = ɵɵdefineComponent({ type: GraphOpComponent, selectors: [["graph-op"]], inputs: { kind: "kind", opName: "opName", slot: "slot", opData: "opData" }, outputs: { onOpNameClick: "onOpNameClick" }, decls: 9, vars: 7, consts: [[1, "op-container"], [1, "input-tensor-name"], [1, "op-name", 3, "ngClass", "click"], ["class", "slot", 4, "ngIf"], ["class", "op-type", 4, "ngIf", "ngIfElse"], ["opInfoMissing", ""], [1, "slot"], [3, "ngSwitch"], [4, "ngSwitchCase"], [1, "op-type"], [1, "op-info-missing"]], template: function GraphOpComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "button", 0);
            ɵɵelementStart(1, "div", 1);
            ɵɵelementStart(2, "button", 2);
            ɵɵlistener("click", function GraphOpComponent_Template_button_click_2_listener() { return ctx.onOpNameClick.emit({ op_name: ctx.opName }); });
            ɵɵelementStart(3, "span");
            ɵɵtext(4);
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵtemplate(5, GraphOpComponent_div_5_Template, 5, 4, "div", 3);
            ɵɵelementEnd();
            ɵɵtemplate(6, GraphOpComponent_div_6_Template, 2, 1, "div", 4);
            ɵɵtemplate(7, GraphOpComponent_ng_template_7_Template, 2, 0, "ng-template", null, 5, ɵɵtemplateRefExtractor);
            ɵɵelementEnd();
        } if (rf & 2) {
            const _r2 = ɵɵreference(8);
            ɵɵadvance(2);
            ɵɵproperty("ngClass", ɵɵpureFunction1(5, _c0$g, ctx.kind === "self" ? "self-op-name" : ""));
            ɵɵadvance(2);
            ɵɵtextInterpolate(ctx.opName);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx.kind !== "self");
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx.opData !== undefined)("ngIfElse", _r2);
        } }, directives: [NgClass, NgIf, NgSwitch, NgSwitchCase], styles: [".op-container[_ngcontent-%COMP%] {\n  background-color: transparent;\n  border: 2px solid #c6cad1;\n  border-radius: 4px;\n  box-shadow: 1px 3px #eee;\n  cursor: pointer;\n  margin: 0 5px 0 0;\n  padding: 2px 6px;\n  text-align: right;\n  width: 200px;\n}\n\n.op-container[_ngcontent-%COMP%]:focus {\n  outline: 0;\n}\n\n.op-container[_ngcontent-%COMP%]:hover {\n  border: 2px solid #ffd3b2;\n}\n\n.op-info-missing[_ngcontent-%COMP%] {\n  color: #808080;\n}\n\n.op-name[_ngcontent-%COMP%] {\n  background-color: transparent;\n  border: none;\n  cursor: pointer;\n  display: inline-block;\n  overflow-wrap: anywhere;\n  padding: 0;\n  text-align: right;\n  text-decoration: underline;\n  white-space: pre-wrap;\n}\n\n.op-name[_ngcontent-%COMP%]:focus {\n  outline: 0;\n}\n\n.op-type[_ngcontent-%COMP%] {\n  background-color: #e3e5e8;\n  border: 1px solid #c0c0c0;\n  border-radius: 4px;\n  display: inline-block;\n  font-family: 'Roboto Mono', monospace;\n  font-size: 10px;\n  height: 14px;\n  line-height: 14px;\n  margin-top: 3px;\n  padding: 1px 3px;\n  width: max-content;\n}\n\n.self-op-name[_ngcontent-%COMP%] {\n  font-weight: bold;\n  text-decoration: none;\n}\n\n.slot[_ngcontent-%COMP%] {\n  color: #707070;\n}"] });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/views/graph/graph_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    function GraphComponent_div_4_div_1_div_2_Template(rf, ctx) { if (rf & 1) {
        const _r15 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "div", 13);
        ɵɵelementStart(1, "div", 14);
        ɵɵtext(2);
        ɵɵelementEnd();
        ɵɵelementStart(3, "graph-op", 15);
        ɵɵlistener("onOpNameClick", function GraphComponent_div_4_div_1_div_2_Template_graph_op_onOpNameClick_3_listener($event) { ɵɵrestoreView(_r15); const ctx_r14 = ɵɵnextContext(3); return ctx_r14.onGraphOpNavigate.emit({ op_name: $event.op_name, graph_id: ctx_r14.graphId }); });
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const inputOpInfo_r12 = ctx.$implicit;
        const slot_r13 = ctx.index;
        ɵɵadvance(2);
        ɵɵtextInterpolate1("Input slot ", slot_r13, ":");
        ɵɵadvance(1);
        ɵɵproperty("kind", "input")("opName", inputOpInfo_r12.op_name)("slot", inputOpInfo_r12.output_slot)("opData", inputOpInfo_r12.data);
    } }
    function GraphComponent_div_4_div_1_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 11);
        ɵɵelementStart(1, "div");
        ɵɵtemplate(2, GraphComponent_div_4_div_1_div_2_Template, 4, 5, "div", 12);
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r5 = ɵɵnextContext(2);
        ɵɵadvance(2);
        ɵɵproperty("ngForOf", ctx_r5.inputOps);
    } }
    function GraphComponent_div_4_ng_template_2_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 16);
        ɵɵtext(1, " (This op has no input tensor.) ");
        ɵɵelementEnd();
    } }
    var I18N_1;
    if (typeof ngI18nClosureMode !== "undefined" && ngI18nClosureMode) {
        const MSG_EXTERNAL_4955133740841299851$$TENSORBOARD_PLUGINS_DEBUGGER_V2_TF_DEBUGGER_V2_PLUGIN_VIEWS_GRAPH_GRAPH_COMPONENT_TS____2 = goog.getMsg("{VAR_PLURAL, plural, =0 {consumer} =1 {consumer} other {consumers}}");
        I18N_1 = MSG_EXTERNAL_4955133740841299851$$TENSORBOARD_PLUGINS_DEBUGGER_V2_TF_DEBUGGER_V2_PLUGIN_VIEWS_GRAPH_GRAPH_COMPONENT_TS____2;
    }
    else {
        I18N_1 = $localize `:␟fe55f9b193ea20aae5b5635e68d9386503847746␟4955133740841299851:{VAR_PLURAL, plural, =0 {consumer} =1 {consumer} other {consumers}}`;
    }
    I18N_1 = ɵɵi18nPostprocess(I18N_1, { "VAR_PLURAL": "\uFFFD0\uFFFD" });
    var I18N_0$1;
    if (typeof ngI18nClosureMode !== "undefined" && ngI18nClosureMode) {
        const MSG_EXTERNAL_5556340343850165516$$TENSORBOARD_PLUGINS_DEBUGGER_V2_TF_DEBUGGER_V2_PLUGIN_VIEWS_GRAPH_GRAPH_COMPONENT_TS____3 = goog.getMsg(" {$icu}", { "icu": I18N_1 });
        I18N_0$1 = MSG_EXTERNAL_5556340343850165516$$TENSORBOARD_PLUGINS_DEBUGGER_V2_TF_DEBUGGER_V2_PLUGIN_VIEWS_GRAPH_GRAPH_COMPONENT_TS____3;
    }
    else {
        I18N_0$1 = $localize `:␟baa460e2f2b857e26292b246fc18ae0ea9b5e537␟5556340343850165516: ${I18N_1}:ICU:`;
    }
    function GraphComponent_div_4_div_8_div_2_div_6_Template(rf, ctx) { if (rf & 1) {
        const _r22 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "div", 22);
        ɵɵelementStart(1, "graph-op", 15);
        ɵɵlistener("onOpNameClick", function GraphComponent_div_4_div_8_div_2_div_6_Template_graph_op_onOpNameClick_1_listener($event) { ɵɵrestoreView(_r22); const ctx_r21 = ɵɵnextContext(4); return ctx_r21.onGraphOpNavigate.emit({ op_name: $event.op_name, graph_id: ctx_r21.graphId }); });
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const consumerOpInfo_r20 = ctx.$implicit;
        ɵɵadvance(1);
        ɵɵproperty("kind", "consumer")("opName", consumerOpInfo_r20.op_name)("slot", consumerOpInfo_r20.input_slot)("opData", consumerOpInfo_r20.data);
    } }
    function GraphComponent_div_4_div_8_div_2_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 19);
        ɵɵelementStart(1, "div", 20);
        ɵɵtext(2);
        ɵɵelementStart(3, "span");
        ɵɵi18n(4, I18N_0$1);
        ɵɵelementEnd();
        ɵɵtext(5, ") ");
        ɵɵelementEnd();
        ɵɵtemplate(6, GraphComponent_div_4_div_8_div_2_div_6_Template, 2, 4, "div", 21);
        ɵɵelementEnd();
    } if (rf & 2) {
        const slotConsumers_r17 = ctx.$implicit;
        const slot_r18 = ctx.index;
        ɵɵadvance(2);
        ɵɵtextInterpolate2(" Output slot ", slot_r18, ": (", slotConsumers_r17.length, " ");
        ɵɵadvance(2);
        ɵɵi18nExp(slotConsumers_r17.length);
        ɵɵi18nApply(4);
        ɵɵadvance(2);
        ɵɵproperty("ngForOf", slotConsumers_r17);
    } }
    function GraphComponent_div_4_div_8_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 17);
        ɵɵelementStart(1, "div");
        ɵɵtemplate(2, GraphComponent_div_4_div_8_div_2_Template, 7, 4, "div", 18);
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r8 = ɵɵnextContext(2);
        ɵɵadvance(2);
        ɵɵproperty("ngForOf", ctx_r8.consumerOps);
    } }
    var I18N_5;
    if (typeof ngI18nClosureMode !== "undefined" && ngI18nClosureMode) {
        const MSG_EXTERNAL_5156712935150586878$$TENSORBOARD_PLUGINS_DEBUGGER_V2_TF_DEBUGGER_V2_PLUGIN_VIEWS_GRAPH_GRAPH_COMPONENT_TS___6 = goog.getMsg("{VAR_PLURAL, plural, =0 {tensor} =1 {tensor} other {tensors}}");
        I18N_5 = MSG_EXTERNAL_5156712935150586878$$TENSORBOARD_PLUGINS_DEBUGGER_V2_TF_DEBUGGER_V2_PLUGIN_VIEWS_GRAPH_GRAPH_COMPONENT_TS___6;
    }
    else {
        I18N_5 = $localize `:␟6aa75f627e0dc16150ef448464e0c857aaa0dc18␟5156712935150586878:{VAR_PLURAL, plural, =0 {tensor} =1 {tensor} other {tensors}}`;
    }
    I18N_5 = ɵɵi18nPostprocess(I18N_5, { "VAR_PLURAL": "\uFFFD0\uFFFD" });
    var I18N_4;
    if (typeof ngI18nClosureMode !== "undefined" && ngI18nClosureMode) {
        const MSG_EXTERNAL_246067053735162634$$TENSORBOARD_PLUGINS_DEBUGGER_V2_TF_DEBUGGER_V2_PLUGIN_VIEWS_GRAPH_GRAPH_COMPONENT_TS___7 = goog.getMsg(" {$icu}", { "icu": I18N_5 });
        I18N_4 = MSG_EXTERNAL_246067053735162634$$TENSORBOARD_PLUGINS_DEBUGGER_V2_TF_DEBUGGER_V2_PLUGIN_VIEWS_GRAPH_GRAPH_COMPONENT_TS___7;
    }
    else {
        I18N_4 = $localize `:␟893476c2c421cee47663c9732fa41a750d3a73df␟246067053735162634: ${I18N_5}:ICU:`;
    }
    function GraphComponent_div_4_ng_template_9_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 23);
        ɵɵtext(1);
        ɵɵelementStart(2, "span");
        ɵɵi18n(3, I18N_4);
        ɵɵelementEnd();
        ɵɵtext(4, " and no consumer.) ");
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r10 = ɵɵnextContext(2);
        ɵɵadvance(1);
        ɵɵtextInterpolate1(" (This op has ", ctx_r10.opInfo.consumers.length, " output ");
        ɵɵadvance(2);
        ɵɵi18nExp(ctx_r10.opInfo.consumers.length);
        ɵɵi18nApply(3);
    } }
    function GraphComponent_div_4_Template(rf, ctx) { if (rf & 1) {
        const _r24 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "div");
        ɵɵtemplate(1, GraphComponent_div_4_div_1_Template, 3, 1, "div", 4);
        ɵɵtemplate(2, GraphComponent_div_4_ng_template_2_Template, 2, 0, "ng-template", null, 5, ɵɵtemplateRefExtractor);
        ɵɵelementStart(4, "div", 6);
        ɵɵelementStart(5, "div", 7);
        ɵɵtext(6, "Op:");
        ɵɵelementEnd();
        ɵɵelementStart(7, "graph-op", 8);
        ɵɵlistener("onOpNameClick", function GraphComponent_div_4_Template_graph_op_onOpNameClick_7_listener($event) { ɵɵrestoreView(_r24); const ctx_r23 = ɵɵnextContext(); return ctx_r23.onGraphOpNavigate.emit({ op_name: $event.op_name, graph_id: ctx_r23.graphId }); });
        ɵɵelementEnd();
        ɵɵelementEnd();
        ɵɵtemplate(8, GraphComponent_div_4_div_8_Template, 3, 1, "div", 9);
        ɵɵtemplate(9, GraphComponent_div_4_ng_template_9_Template, 5, 2, "ng-template", null, 10, ɵɵtemplateRefExtractor);
        ɵɵelementEnd();
    } if (rf & 2) {
        const _r6 = ɵɵreference(3);
        const _r9 = ɵɵreference(10);
        const ctx_r0 = ɵɵnextContext();
        ɵɵadvance(1);
        ɵɵproperty("ngIf", ctx_r0.inputOps.length > 0)("ngIfElse", _r6);
        ɵɵadvance(6);
        ɵɵproperty("kind", "self")("opName", ctx_r0.opInfo.op_name)("opData", ctx_r0.opInfo);
        ɵɵadvance(1);
        ɵɵproperty("ngIf", ctx_r0.totalNumConsumers > 0)("ngIfElse", _r9);
    } }
    function GraphComponent_ng_template_5_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "span", 24);
        ɵɵtext(1, " (Op info unavailable.) ");
        ɵɵelementEnd();
    } }
    function GraphComponent_ng_template_7_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 25);
        ɵɵtext(1, " No graph op selected. Click a tensor name in the Graph Executions table to view the neighborhood of the tensor's op in its graph. ");
        ɵɵelementEnd();
    } }
    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    class GraphComponent {
        constructor() {
            this.onGraphOpNavigate = new EventEmitter();
        }
        /**
         * Get the ID of the immediately-enclosing graph of the op.
         * @return {?}
         */
        get graphId() {
            return this.opInfo.graph_ids[this.opInfo.graph_ids.length - 1];
        }
        /**
         * Total number of consumers of all output tensors of the op.
         * @return {?}
         */
        get totalNumConsumers() {
            return this.consumerOps.reduce((/**
             * @param {?} count
             * @param {?} slotConsumers
             * @return {?}
             */
            (count, slotConsumers) => {
                return count + slotConsumers.length;
            }), 0);
        }
    }
    GraphComponent.decorators = [
        { type: Component, args: [{
                    selector: 'graph-component',
                    templateUrl: './graph_component.ng.html',
                    styleUrls: ['./graph_component.css'],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    GraphComponent.propDecorators = {
        opInfo: [{ type: Input }],
        inputOps: [{ type: Input }],
        consumerOps: [{ type: Input }],
        onGraphOpNavigate: [{ type: Output }]
    };
    /** @nocollapse */ GraphComponent.ɵfac = function GraphComponent_Factory(t) { return new (t || GraphComponent)(); };
    /** @nocollapse */ GraphComponent.ɵcmp = ɵɵdefineComponent({ type: GraphComponent, selectors: [["graph-component"]], inputs: { opInfo: "opInfo", inputOps: "inputOps", consumerOps: "consumerOps" }, outputs: { onGraphOpNavigate: "onGraphOpNavigate" }, decls: 9, vars: 2, consts: [[1, "graph-structure-container"], [4, "ngIf", "ngIfElse"], ["opInfoMissing", ""], ["noOpFocused", ""], ["class", "inputs-container", 4, "ngIf", "ngIfElse"], ["noInputs", ""], [1, "self-op-container"], [1, "self-op-header"], [3, "kind", "opName", "opData", "onOpNameClick"], ["class", "consumers-container", 4, "ngIf", "ngIfElse"], ["noConsumers", ""], [1, "inputs-container"], ["class", "input-op-section", 4, "ngFor", "ngForOf"], [1, "input-op-section"], [1, "input-slot-header"], [3, "kind", "opName", "slot", "opData", "onOpNameClick"], [1, "inputs-container", "no-inputs-indicator"], [1, "consumers-container"], ["class", "slot-consumers-container", 4, "ngFor", "ngForOf"], [1, "slot-consumers-container"], [1, "slot-consumers-header"], ["class", "consumer-section", 4, "ngFor", "ngForOf"], [1, "consumer-section"], [1, "op-consumers-container"], [1, "op-info-missing"], [1, "no-op-focused"]], template: function GraphComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "div");
            ɵɵelementStart(1, "div");
            ɵɵtext(2, "Graph Structure");
            ɵɵelementEnd();
            ɵɵelementStart(3, "div", 0);
            ɵɵtemplate(4, GraphComponent_div_4_Template, 11, 7, "div", 1);
            ɵɵelementEnd();
            ɵɵtemplate(5, GraphComponent_ng_template_5_Template, 2, 0, "ng-template", null, 2, ɵɵtemplateRefExtractor);
            ɵɵtemplate(7, GraphComponent_ng_template_7_Template, 2, 0, "ng-template", null, 3, ɵɵtemplateRefExtractor);
            ɵɵelementEnd();
        } if (rf & 2) {
            const _r3 = ɵɵreference(8);
            ɵɵadvance(4);
            ɵɵproperty("ngIf", ctx.opInfo !== undefined && ctx.opInfo !== null)("ngIfElse", _r3);
        } }, directives: [NgIf, GraphOpComponent, NgForOf], styles: ["[_nghost-%COMP%] {\n  overflow-y: auto;\n}\n\n.consumers-container[_ngcontent-%COMP%] {\n  padding-bottom: 5px;\n  overflow-x: auto;\n  white-space: nowrap;\n}\n\n.consumer-section[_ngcontent-%COMP%] {\n  display: block;\n  margin: 5px 0;\n}\n\n.graph-structure-container[_ngcontent-%COMP%] {\n  font-size: 12px;\n  overflow-y: auto;\n  white-space: nowrap;\n}\n\n.inputs-container[_ngcontent-%COMP%] {\n  border-bottom: 1px solid rgb(0, 0, 0, 0.12);\n  margin-top: 5px;\n  overflow-x: auto;\n  padding-bottom: 0;\n  white-space: nowrap;\n}\n\n.input-op-section[_ngcontent-%COMP%] {\n  border-right: 1px solid rgb(0, 0, 0, 0.12);\n  display: inline-block;\n  margin-right: 5px;\n  padding-bottom: 5px;\n}\n\n.input-slot-header[_ngcontent-%COMP%] {\n  \n  background-color: #fff099;\n  margin-bottom: 5px;\n}\n\n.input-tensor-name[_ngcontent-%COMP%] {\n  display: block;\n  white-space: nowrap;\n}\n\n.no-op-focused[_ngcontent-%COMP%] {\n  color: #808080;\n  font-family: 'Roboto', Arial, Helvetica, sans-serif;\n  font-size: 13px;\n  white-space: normal;\n}\n\n.self-op-header[_ngcontent-%COMP%] {\n  font-weight: bold;\n  margin-bottom: 5px;\n}\n\n.self-op-container[_ngcontent-%COMP%] {\n  border-bottom: 1px solid rgb(0, 0, 0, 0.12);\n  padding-bottom: 5px;\n}\n\n.slot-consumers-container[_ngcontent-%COMP%] {\n  border-right: 1px solid rgb(0, 0, 0, 0.12);\n  display: inline-block;\n  margin-right: 5px;\n  padding-top: 5px;\n  vertical-align: top;\n}\n\n.slot-consumers-header[_ngcontent-%COMP%] {\n  white-space: nowrap;\n}"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/views/graph/graph_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class GraphContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.opInfo$ = this.store.pipe(select(getFocusedGraphOpInfo));
            this.inputOps$ = this.store.pipe(select(getFocusedGraphOpInputs));
            this.consumerOps$ = this.store.pipe(select(getFocusedGraphOpConsumers));
        }
        /**
         * @param {?} event
         * @return {?}
         */
        onGraphOpNavigate(event) {
            this.store.dispatch(graphOpFocused(event));
        }
    }
    GraphContainer.decorators = [
        { type: Component, args: [{
                    selector: 'tf-debugger-v2-graph',
                    template: `
    <graph-component
      [opInfo]="opInfo$ | async"
      [inputOps]="inputOps$ | async"
      [consumerOps]="consumerOps$ | async"
      (onGraphOpNavigate)="onGraphOpNavigate($event)"
    ></graph-component>
  `,
                },] },
    ];
    /** @nocollapse */
    GraphContainer.ctorParameters = () => [
        { type: Store }
    ];
    /** @nocollapse */ GraphContainer.ɵfac = function GraphContainer_Factory(t) { return new (t || GraphContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ GraphContainer.ɵcmp = ɵɵdefineComponent({ type: GraphContainer, selectors: [["tf-debugger-v2-graph"]], decls: 4, vars: 9, consts: [[3, "opInfo", "inputOps", "consumerOps", "onGraphOpNavigate"]], template: function GraphContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "graph-component", 0);
            ɵɵlistener("onGraphOpNavigate", function GraphContainer_Template_graph_component_onGraphOpNavigate_0_listener($event) { return ctx.onGraphOpNavigate($event); });
            ɵɵpipe(1, "async");
            ɵɵpipe(2, "async");
            ɵɵpipe(3, "async");
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵproperty("opInfo", ɵɵpipeBind1(1, 3, ctx.opInfo$))("inputOps", ɵɵpipeBind1(2, 5, ctx.inputOps$))("consumerOps", ɵɵpipeBind1(3, 7, ctx.consumerOps$));
        } }, directives: [GraphComponent], pipes: [AsyncPipe], encapsulation: 2 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/views/graph_executions/graph_executions_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    function GraphExecutionsComponent_cdk_virtual_scroll_viewport_3_div_1_div_3_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 12);
        ɵɵtext(1, "\u25B6");
        ɵɵelementEnd();
    } }
    const _c0$h = function (a0, a1) { return { tensorDebugMode: a0, array: a1 }; };
    function GraphExecutionsComponent_cdk_virtual_scroll_viewport_3_div_1_div_5_debug_tensor_value_6_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "debug-tensor-value", 17);
    } if (rf & 2) {
        const i_r2 = ɵɵnextContext(2).$implicit;
        const ctx_r7 = ɵɵnextContext(2);
        ɵɵproperty("debugTensorValue", ctx_r7.parseDebugTensorValue(ɵɵpureFunction2(1, _c0$h, ctx_r7.graphExecutionData[i_r2].tensor_debug_mode, ctx_r7.graphExecutionData[i_r2].debug_tensor_value)));
    } }
    function GraphExecutionsComponent_cdk_virtual_scroll_viewport_3_div_1_div_5_Template(rf, ctx) { if (rf & 1) {
        const _r11 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "div");
        ɵɵelementStart(1, "div", 13);
        ɵɵelementStart(2, "button", 14);
        ɵɵlistener("click", function GraphExecutionsComponent_cdk_virtual_scroll_viewport_3_div_1_div_5_Template_button_click_2_listener() { ɵɵrestoreView(_r11); const i_r2 = ɵɵnextContext().$implicit; const ctx_r9 = ɵɵnextContext(2); return ctx_r9.onTensorNameClick.emit({ index: i_r2, graph_id: ctx_r9.graphExecutionData[i_r2].graph_id, op_name: ctx_r9.graphExecutionData[i_r2].op_name }); });
        ɵɵtext(3);
        ɵɵelementEnd();
        ɵɵelementStart(4, "div", 15);
        ɵɵtext(5);
        ɵɵelementEnd();
        ɵɵelementEnd();
        ɵɵtemplate(6, GraphExecutionsComponent_cdk_virtual_scroll_viewport_3_div_1_div_5_debug_tensor_value_6_Template, 1, 4, "debug-tensor-value", 16);
        ɵɵelementEnd();
    } if (rf & 2) {
        const i_r2 = ɵɵnextContext().$implicit;
        const ctx_r4 = ɵɵnextContext(2);
        ɵɵadvance(2);
        ɵɵpropertyInterpolate("title", ctx_r4.getTensorName(i_r2));
        ɵɵadvance(1);
        ɵɵtextInterpolate1(" ", ctx_r4.getTensorName(i_r2), " ");
        ɵɵadvance(2);
        ɵɵtextInterpolate(ctx_r4.graphExecutionData[i_r2].op_type);
        ɵɵadvance(1);
        ɵɵproperty("ngIf", ctx_r4.graphExecutionData[i_r2].debug_tensor_value !== null);
    } }
    function GraphExecutionsComponent_cdk_virtual_scroll_viewport_3_div_1_ng_template_6_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 18);
        ɵɵtext(1, " Loading... ");
        ɵɵelementEnd();
    } }
    const _c1$9 = function (a0) { return { "input-of-focus": a0 }; };
    function GraphExecutionsComponent_cdk_virtual_scroll_viewport_3_div_1_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 5);
        ɵɵelementStart(1, "div", 6);
        ɵɵelementStart(2, "div", 7);
        ɵɵtemplate(3, GraphExecutionsComponent_cdk_virtual_scroll_viewport_3_div_1_div_3_Template, 2, 0, "div", 8);
        ɵɵtext(4);
        ɵɵelementEnd();
        ɵɵtemplate(5, GraphExecutionsComponent_cdk_virtual_scroll_viewport_3_div_1_div_5_Template, 7, 4, "div", 9);
        ɵɵtemplate(6, GraphExecutionsComponent_cdk_virtual_scroll_viewport_3_div_1_ng_template_6_Template, 2, 0, "ng-template", 10, 11, ɵɵtemplateRefExtractor);
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const i_r2 = ctx.$implicit;
        const _r5 = ɵɵreference(7);
        const ctx_r1 = ɵɵnextContext(2);
        ɵɵadvance(1);
        ɵɵproperty("ngClass", ɵɵpureFunction1(5, _c1$9, ctx_r1.isInputOfFocus(i_r2)));
        ɵɵadvance(2);
        ɵɵproperty("ngIf", i_r2 === ctx_r1.focusIndex);
        ɵɵadvance(1);
        ɵɵtextInterpolate1(" ", i_r2, " ");
        ɵɵadvance(1);
        ɵɵproperty("ngIf", ctx_r1.graphExecutionData[i_r2])("ngIfElse", _r5);
    } }
    function GraphExecutionsComponent_cdk_virtual_scroll_viewport_3_Template(rf, ctx) { if (rf & 1) {
        const _r14 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "cdk-virtual-scroll-viewport", 3);
        ɵɵlistener("scrolledIndexChange", function GraphExecutionsComponent_cdk_virtual_scroll_viewport_3_Template_cdk_virtual_scroll_viewport_scrolledIndexChange_0_listener($event) { ɵɵrestoreView(_r14); const ctx_r13 = ɵɵnextContext(); return ctx_r13.onScrolledIndexChange.emit($event); });
        ɵɵtemplate(1, GraphExecutionsComponent_cdk_virtual_scroll_viewport_3_div_1_Template, 8, 7, "div", 4);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r0 = ɵɵnextContext();
        ɵɵadvance(1);
        ɵɵproperty("cdkVirtualForOf", ctx_r0.graphExecutionIndices);
    } }
    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    class GraphExecutionsComponent {
        constructor() {
            this.onScrolledIndexChange = new EventEmitter();
            this.onTensorNameClick = new EventEmitter();
            this.parseDebugTensorValue = parseDebugTensorValue;
            this.TEST_ONLY = {
                getViewPort: (/**
                 * @return {?}
                 */
                () => this.viewPort),
            };
        }
        /**
         * @param {?} changes
         * @return {?}
         */
        ngOnChanges(changes) {
            if (this.viewPort &&
                changes['focusIndex'] &&
                changes['focusIndex'].currentValue !== null) {
                /** @type {?} */
                const range = this.viewPort.getRenderedRange();
                /** @type {?} */
                const scrollIndex = changes['focusIndex'].currentValue;
                // Make sure that the index is scrolled to one third the view port.
                // This is nicer than scrolling it merely to the top.
                /** @type {?} */
                const thirdRange = Math.round((range.end - range.start) / 3);
                /** @type {?} */
                const targetIndex = Math.max(scrollIndex - thirdRange, 0);
                /** @type {?} */
                const useSmoothScrolling = scrollIndex >= range.start && scrollIndex < range.end;
                this.viewPort.scrollToIndex(targetIndex, useSmoothScrolling ? 'smooth' : undefined);
            }
        }
        /**
         * @param {?} graphExecutionIndex
         * @return {?}
         */
        getTensorName(graphExecutionIndex) {
            return (`${this.graphExecutionData[graphExecutionIndex].op_name}:` +
                `${this.graphExecutionData[graphExecutionIndex].output_slot}`);
        }
        /**
         * Computes if given graph-execution index is an immediate input tensor to
         * the graph execution currently focused on.
         *
         * @param {?} graphExecutionIndex
         * @return {?} If no graph execution is focused on, `false`. If a graph execution
         *   is being focused on, `true` if `graphExecutionIndex` points to a graph-
         *   execution event that forms the immediate input to the focused one.
         */
        isInputOfFocus(graphExecutionIndex) {
            if (this.focusInputIndices === null) {
                return false;
            }
            return this.focusInputIndices.includes(graphExecutionIndex);
        }
    }
    GraphExecutionsComponent.decorators = [
        { type: Component, args: [{
                    selector: 'graph-executions-component',
                    templateUrl: './graph_executions_component.ng.html',
                    styleUrls: ['./graph_executions_component.css'],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    GraphExecutionsComponent.propDecorators = {
        numGraphExecutions: [{ type: Input }],
        graphExecutionData: [{ type: Input }],
        graphExecutionIndices: [{ type: Input }],
        focusIndex: [{ type: Input }],
        focusInputIndices: [{ type: Input }],
        onScrolledIndexChange: [{ type: Output }],
        onTensorNameClick: [{ type: Output }],
        viewPort: [{ type: ViewChild, args: [CdkVirtualScrollViewport, { static: false },] }]
    };
    /** @nocollapse */ GraphExecutionsComponent.ɵfac = function GraphExecutionsComponent_Factory(t) { return new (t || GraphExecutionsComponent)(); };
    /** @nocollapse */ GraphExecutionsComponent.ɵcmp = ɵɵdefineComponent({ type: GraphExecutionsComponent, selectors: [["graph-executions-component"]], viewQuery: function GraphExecutionsComponent_Query(rf, ctx) { if (rf & 1) {
            ɵɵviewQuery(CdkVirtualScrollViewport, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.viewPort = _t.first);
        } }, inputs: { numGraphExecutions: "numGraphExecutions", graphExecutionData: "graphExecutionData", graphExecutionIndices: "graphExecutionIndices", focusIndex: "focusIndex", focusInputIndices: "focusInputIndices" }, outputs: { onScrolledIndexChange: "onScrolledIndexChange", onTensorNameClick: "onTensorNameClick" }, features: [ɵɵNgOnChangesFeature], decls: 4, vars: 2, consts: [[1, "graph-executions-container"], [1, "graph-executions-title"], ["itemSize", "38", "class", "graph-executions-viewport", 3, "scrolledIndexChange", 4, "ngIf"], ["itemSize", "38", 1, "graph-executions-viewport", 3, "scrolledIndexChange"], ["class", "tensor-container", 4, "cdkVirtualFor", "cdkVirtualForOf"], [1, "tensor-container"], [1, "tensor-item", 3, "ngClass"], [1, "graph-execution-index"], ["class", "graph-execution-focus", 4, "ngIf"], [4, "ngIf", "ngIfElse"], ["class", "tensor-item"], ["dataLoading", ""], [1, "graph-execution-focus"], [1, "tensor-name-and-op-type"], [1, "tensor-name", 3, "title", "click"], [1, "op-type"], [3, "debugTensorValue", 4, "ngIf"], [3, "debugTensorValue"], [1, "loading-spinner"]], template: function GraphExecutionsComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "div", 0);
            ɵɵelementStart(1, "div", 1);
            ɵɵtext(2);
            ɵɵelementEnd();
            ɵɵtemplate(3, GraphExecutionsComponent_cdk_virtual_scroll_viewport_3_Template, 2, 1, "cdk-virtual-scroll-viewport", 2);
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵadvance(2);
            ɵɵtextInterpolate1(" Graph Executions (", ctx.numGraphExecutions, ") ");
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx.numGraphExecutions !== null && ctx.numGraphExecutions > 0);
        } }, directives: [NgIf, CdkVirtualScrollViewport, CdkFixedSizeVirtualScroll, CdkVirtualForOf, NgClass, DebugTensorValueComponent], styles: [".graph-executions-container[_ngcontent-%COMP%] {\n  border-left: 1px solid rgba(0, 0, 0, 0.12);\n  display: flex;\n  flex-direction: column;\n  height: 100%;\n  margin-left: 8px;\n  padding-left: 10px;\n}\n\n.graph-execution-focus[_ngcontent-%COMP%] {\n  display: inline-block;\n}\n\n.graph-execution-index[_ngcontent-%COMP%] {\n  color: #757575;\n  display: inline-block;\n  padding-right: 4px;\n  text-align: right;\n  width: 40px;\n}\n\n.graph-executions-title[_ngcontent-%COMP%] {\n  box-shadow: 0 5px 3px -3px #ccc;\n  padding-bottom: 5px;\n}\n\n.graph-executions-viewport[_ngcontent-%COMP%] {\n  flex-grow: 1;\n  font-size: 12px;\n  width: 100%;\n  overflow-x: hidden;\n}\n\n.input-of-focus[_ngcontent-%COMP%] {\n  \n  background-color: #fff099;\n}\n\n.loading-spinner[_ngcontent-%COMP%] {\n  display: inline-block;\n}\n\n.op-type[_ngcontent-%COMP%] {\n  background-color: #e3e5e8;\n  border: 1px solid #c0c0c0;\n  border-radius: 4px;\n  direction: rtl;\n  display: block;\n  font-family: 'Roboto Mono', monospace;\n  font-size: 10px;\n  height: 14px;\n  line-height: 14px;\n  padding: 1px 3px;\n  width: max-content;\n}\n\n.tensor-container[_ngcontent-%COMP%] {\n  width: 100%;\n}\n\n.tensor-item[_ngcontent-%COMP%] {\n  border-bottom: 1px solid rgba(0, 0, 0, 0.12);\n  display: flex;\n  flex-wrap: nowrap;\n  height: 38px;\n  line-height: 38px;\n  text-align: left;\n  vertical-align: middle;\n  white-space: nowrap;\n  width: 100%;\n}\n\n.tensor-name[_ngcontent-%COMP%] {\n  background-color: transparent;\n  border: none;\n  box-sizing: border-box;\n  cursor: pointer;\n  direction: rtl;\n  display: block;\n  height: 16px;\n  line-height: 16px;\n  margin: 2px 0 1px;\n  max-width: calc(100% - 2px);\n  overflow: hidden;\n  padding: 0 2px;\n  text-align: right;\n  text-decoration: underline;\n  text-overflow: ellipsis;\n  white-space: nowrap;\n}\n\n.tensor-name[_ngcontent-%COMP%]:focus {\n  outline: 1px solid #c6cad1;\n}\n\n.tensor-name-and-op-type[_ngcontent-%COMP%] {\n  direction: rtl;\n  display: inline-block;\n  overflow: hidden;\n  padding-right: 8px;\n  text-align: right;\n  width: 240px;\n}\n\ndebug-tensor-value[_ngcontent-%COMP%] {\n  display: inline-block;\n  margin: 2px 0;\n}"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/views/graph_executions/graph_executions_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class GraphExecutionsContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.numGraphExecutions$ = this.store.pipe(select(getNumGraphExecutions));
            this.graphExecutionData$ = this.store.pipe(select(getGraphExecutionData));
            this.graphExecutionIndices$ = this.store.pipe(select(createSelector(getNumGraphExecutions, (/**
             * @param {?} numGraphExecution
             * @return {?}
             */
            (numGraphExecution) => {
                if (numGraphExecution === 0) {
                    return null;
                }
                return Array.from({ length: numGraphExecution }).map((/**
                 * @param {?} _
                 * @param {?} i
                 * @return {?}
                 */
                (_, i) => i));
            }))));
            this.focusIndex$ = this.store.pipe(select(getGraphExecutionFocusIndex));
            /**
             * Inferred graph-execution indices that belong to the immediate inputs
             * to the currently-focused graph op.
             */
            this.focusInputIndices$ = this.store.pipe(select(getFocusedGraphExecutionInputIndices));
        }
        /**
         * @param {?} scrolledIndex
         * @return {?}
         */
        onScrolledIndexChange(scrolledIndex) {
            this.store.dispatch(graphExecutionScrollToIndex({ index: scrolledIndex }));
        }
        /**
         * @param {?} event
         * @return {?}
         */
        onTensorNameClick(event) {
            this.store.dispatch(graphExecutionFocused(event));
        }
    }
    GraphExecutionsContainer.decorators = [
        { type: Component, args: [{
                    selector: 'tf-debugger-v2-graph-executions',
                    template: `
    <graph-executions-component
      [numGraphExecutions]="numGraphExecutions$ | async"
      [graphExecutionData]="graphExecutionData$ | async"
      [graphExecutionIndices]="graphExecutionIndices$ | async"
      [focusIndex]="focusIndex$ | async"
      [focusInputIndices]="focusInputIndices$ | async"
      (onScrolledIndexChange)="onScrolledIndexChange($event)"
      (onTensorNameClick)="onTensorNameClick($event)"
    ></graph-executions-component>
  `,
                },] },
    ];
    /** @nocollapse */
    GraphExecutionsContainer.ctorParameters = () => [
        { type: Store }
    ];
    /** @nocollapse */ GraphExecutionsContainer.ɵfac = function GraphExecutionsContainer_Factory(t) { return new (t || GraphExecutionsContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ GraphExecutionsContainer.ɵcmp = ɵɵdefineComponent({ type: GraphExecutionsContainer, selectors: [["tf-debugger-v2-graph-executions"]], decls: 6, vars: 15, consts: [[3, "numGraphExecutions", "graphExecutionData", "graphExecutionIndices", "focusIndex", "focusInputIndices", "onScrolledIndexChange", "onTensorNameClick"]], template: function GraphExecutionsContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "graph-executions-component", 0);
            ɵɵlistener("onScrolledIndexChange", function GraphExecutionsContainer_Template_graph_executions_component_onScrolledIndexChange_0_listener($event) { return ctx.onScrolledIndexChange($event); })("onTensorNameClick", function GraphExecutionsContainer_Template_graph_executions_component_onTensorNameClick_0_listener($event) { return ctx.onTensorNameClick($event); });
            ɵɵpipe(1, "async");
            ɵɵpipe(2, "async");
            ɵɵpipe(3, "async");
            ɵɵpipe(4, "async");
            ɵɵpipe(5, "async");
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵproperty("numGraphExecutions", ɵɵpipeBind1(1, 5, ctx.numGraphExecutions$))("graphExecutionData", ɵɵpipeBind1(2, 7, ctx.graphExecutionData$))("graphExecutionIndices", ɵɵpipeBind1(3, 9, ctx.graphExecutionIndices$))("focusIndex", ɵɵpipeBind1(4, 11, ctx.focusIndex$))("focusInputIndices", ɵɵpipeBind1(5, 13, ctx.focusInputIndices$));
        } }, directives: [GraphExecutionsComponent], pipes: [AsyncPipe], encapsulation: 2 });

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    function getWindow$1() {
        return window;
    }
    const utils$1 = {
        getWindow: getWindow$1,
    };
    const MONACO_PATH_PREFIX = 'vs';
    const MONACO_IMPORT_PATH = '/tf-imports/vs';
    /**
     * require.js's require() wrapped as an async function that returns a Promise.
     *
     * This wrapped version does not support callback-function arguments.
     *
     * @param paths
     */
    function requireAsPromise(paths) {
        const require = utils$1.getWindow().require;
        return new Promise((resolve) => {
            require(paths, resolve);
        });
    }
    /**
     * If `window.monaco` is undefined, load the monaco-editor API object onto that
     * global path dynamically using require.js. If `window.monaco` is already
     * defined, this function is a no-op.
     */
    function loadMonaco() {
        return __awaiter(this, void 0, void 0, function* () {
            const window = utils$1.getWindow();
            if (window.monaco !== undefined) {
                return;
            }
            if (window.require) {
                const require = window.require;
                require.config({
                    paths: {
                        [MONACO_PATH_PREFIX]: MONACO_IMPORT_PATH,
                    },
                });
                yield requireAsPromise([`${MONACO_PATH_PREFIX}/editor/editor.main`]);
                yield requireAsPromise([
                    `${MONACO_PATH_PREFIX}/python/python.contribution`,
                ]);
            }
            else {
                throw new Error('loadMonaco() failed because function require() is unavailable');
            }
        });
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/views/source_code/source_code_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    const _c0$i = ["codeViewerContainer"];
    /** @type {?} */
    const DEFAULT_CODE_LANGUAGE = 'python';
    /** @type {?} */
    const DEFAULT_CODE_FONT_SIZE = 10;
    /** @type {?} */
    const RESIZE_DEBOUNCE_INTERAVL_MS = 50;
    class SourceCodeComponent {
        constructor() {
            this.lines = null; // TODO(cais): Add spinner for `null`.
            // TODO(cais): Add spinner for `null`.
            this.focusedLineno = null;
            // TODO(cais): Explore better typing by depending on external libraries.
            this.monaco = null;
            this.editor = null;
            this.decorations = [];
            this.ngUnsubscribe = new Subject();
        }
        /**
         * @return {?}
         */
        ngOnInit() {
            // Listen to window resize event. When resize happens, re-layout
            // monaco editor, so its width is always up-to-date with respect to
            // the window size. Do this with `debounce()` to prevent re-layouting
            // at too high a rate.
            /** @type {?} */
            const resizePipe = fromEvent(window, 'resize')
                .pipe(takeUntil(this.ngUnsubscribe), debounce((/**
             * @return {?}
             */
            () => interval(RESIZE_DEBOUNCE_INTERAVL_MS))), tap((/**
             * @return {?}
             */
            () => {
                if (this.editor !== null) {
                    this.editor.layout();
                }
            })))
                .subscribe();
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this.ngUnsubscribe.next();
            this.ngUnsubscribe.complete();
        }
        /**
         * @param {?} changes
         * @return {?}
         */
        ngOnChanges(changes) {
            if (this.monaco === null) {
                return;
            }
            /** @type {?} */
            const currentLines = changes['monaco']
                ? this.lines
                : changes['lines']
                    ? changes['lines'].currentValue
                    : null;
            if (currentLines) {
                /** @type {?} */
                const value = currentLines.join('\n');
                if (this.editor === null) {
                    this.editor = this.monaco.editor.create(this.codeViewerContainer.nativeElement, {
                        value,
                        language: DEFAULT_CODE_LANGUAGE,
                        readOnly: true,
                        fontSize: DEFAULT_CODE_FONT_SIZE,
                        minimap: {
                            enabled: true,
                        },
                    });
                }
                else {
                    this.editor.setValue(value);
                }
            }
            /** @type {?} */
            const currentFocusedLineno = changes['monaco']
                ? this.focusedLineno
                : changes['focusedLineno']
                    ? changes['focusedLineno'].currentValue
                    : null;
            if (currentFocusedLineno && this.lines && this.monaco !== null) {
                this.editor.revealLineInCenter(currentFocusedLineno, this.monaco.editor.ScrollType.Smooth);
                /** @type {?} */
                const lineLength = this.lines[currentFocusedLineno - 1].length;
                this.decorations = this.editor.deltaDecorations(this.decorations, [
                    {
                        range: new this.monaco.Range(currentFocusedLineno, 1, currentFocusedLineno, 1),
                        options: {
                            isWholeLine: true,
                            linesDecorationsClassName: 'highlight-gutter',
                        },
                    },
                    {
                        range: new this.monaco.Range(currentFocusedLineno, 1, currentFocusedLineno, lineLength + 1),
                        options: {
                            inlineClassName: 'highlight-line',
                        },
                    },
                ]);
            }
        }
    }
    SourceCodeComponent.decorators = [
        { type: Component, args: [{
                    selector: 'source-code-component',
                    templateUrl: './source_code_component.ng.html',
                    styleUrls: ['./source_code_component.css'],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    SourceCodeComponent.propDecorators = {
        lines: [{ type: Input }],
        focusedLineno: [{ type: Input }],
        monaco: [{ type: Input }],
        codeViewerContainer: [{ type: ViewChild, args: ['codeViewerContainer', { static: true, read: ElementRef },] }]
    };
    /** @nocollapse */ SourceCodeComponent.ɵfac = function SourceCodeComponent_Factory(t) { return new (t || SourceCodeComponent)(); };
    /** @nocollapse */ SourceCodeComponent.ɵcmp = ɵɵdefineComponent({ type: SourceCodeComponent, selectors: [["source-code-component"]], viewQuery: function SourceCodeComponent_Query(rf, ctx) { if (rf & 1) {
            ɵɵstaticViewQuery(_c0$i, true, ElementRef);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.codeViewerContainer = _t.first);
        } }, inputs: { lines: "lines", focusedLineno: "focusedLineno", monaco: "monaco" }, features: [ɵɵNgOnChangesFeature], decls: 2, vars: 0, consts: [[1, "code-viewer-container"], ["codeViewerContainer", ""]], template: function SourceCodeComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelement(0, "div", 0, 1);
        } }, styles: [".code-viewer-container[_ngcontent-%COMP%] {\n  height: 100%;\n}\n\n[_nghost-%COMP%]     .highlight-gutter {\n  background: rgba(255, 111, 0, 0.7);\n  width: 5px !important;\n}\n\n[_nghost-%COMP%]     .highlight-line {\n  background: rgba(255, 111, 0, 0.3);\n}"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/views/source_code/source_code_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    // TODO(cais): Explore better typing by depending on external libraries.
    /** @type {?} */
    const windowWithRequireAndMonaco = window;
    /**
     * SourceCodeContainer displays the content of a source-code file.
     *
     * It displays the code with visual features including syntax highlighting.
     * It additionally provides functionalities to:
     * - Scroll to and highlight a given line by its line number.
     *
     * TODO(cais): Add support for line decoration and symbol decoration.
     *
     * Unlike SourceFilesComponent, SourceCodeContainer handles only one file at a
     * time.
     */
    class SourceCodeContainer {
        constructor() {
            // Lines of the source-code file, split at line breaks.
            this.lines = null; // TODO(cais): Add spinner for `null`.
            // TODO(cais): Add spinner for `null`.
            // Line number to scroll to and highlight, 1-based.
            this.focusedLineno = null;
            this.monaco$ = null;
        }
        /**
         * @return {?}
         */
        ngOnInit() {
            this.monaco$ = from(loadMonaco()).pipe(map((/**
             * @return {?}
             */
            () => windowWithRequireAndMonaco.monaco)));
        }
    }
    SourceCodeContainer.decorators = [
        { type: Component, args: [{
                    selector: 'tf-debugger-v2-source-code',
                    template: `
    <source-code-component
      [lines]="lines"
      [focusedLineno]="focusedLineno"
      [monaco]="monaco$ | async"
    ></source-code-component>
  `,
                },] },
    ];
    /** @nocollapse */
    SourceCodeContainer.ctorParameters = () => [];
    SourceCodeContainer.propDecorators = {
        lines: [{ type: Input }],
        focusedLineno: [{ type: Input }]
    };
    /** @nocollapse */ SourceCodeContainer.ɵfac = function SourceCodeContainer_Factory(t) { return new (t || SourceCodeContainer)(); };
    /** @nocollapse */ SourceCodeContainer.ɵcmp = ɵɵdefineComponent({ type: SourceCodeContainer, selectors: [["tf-debugger-v2-source-code"]], inputs: { lines: "lines", focusedLineno: "focusedLineno" }, decls: 2, vars: 5, consts: [[3, "lines", "focusedLineno", "monaco"]], template: function SourceCodeContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelement(0, "source-code-component", 0);
            ɵɵpipe(1, "async");
        } if (rf & 2) {
            ɵɵproperty("lines", ctx.lines)("focusedLineno", ctx.focusedLineno)("monaco", ɵɵpipeBind1(1, 3, ctx.monaco$));
        } }, directives: [SourceCodeComponent], pipes: [AsyncPipe], encapsulation: 2 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/views/source_files/source_files_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    function SourceFilesComponent_div_4_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 6);
        ɵɵtext(1);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r0 = ɵɵnextContext();
        ɵɵadvance(1);
        ɵɵtextInterpolate1(" ", ctx_r0.focusedSourceLineSpec.file_path, " ");
    } }
    function SourceFilesComponent_ng_template_5_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 7);
        ɵɵtext(1, " No file selected. Click a line number in the Stack Trace section to show the source code. ");
        ɵɵelementEnd();
    } }
    function SourceFilesComponent_tf_debugger_v2_source_code_7_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "tf-debugger-v2-source-code", 8);
    } if (rf & 2) {
        const ctx_r3 = ɵɵnextContext();
        ɵɵproperty("lines", ctx_r3.focusedSourceFileContent.lines)("focusedLineno", ctx_r3.focusedSourceLineSpec.lineno);
    } }
    /**
     * Renders the content of source file(s).
     *
     * Unlike `SourceCodeComponent`, which displays only the content of a single
     * source-code file, `SourceFilesComponent`is aware of the meta-informaton about
     * the files being displayed, such their file paths. Such meta-information is
     * displayed by this component.
     */
    class SourceFilesComponent {
        constructor() {
            this.focusedSourceFileContent = null;
            this.focusedSourceLineSpec = null;
        }
    }
    SourceFilesComponent.decorators = [
        { type: Component, args: [{
                    selector: 'source-files-component',
                    templateUrl: './source_files_component.ng.html',
                    styleUrls: ['./source_files_component.css'],
                },] },
    ];
    SourceFilesComponent.propDecorators = {
        focusedSourceFileContent: [{ type: Input }],
        focusedSourceLineSpec: [{ type: Input }]
    };
    /** @nocollapse */ SourceFilesComponent.ɵfac = function SourceFilesComponent_Factory(t) { return new (t || SourceFilesComponent)(); };
    /** @nocollapse */ SourceFilesComponent.ɵcmp = ɵɵdefineComponent({ type: SourceFilesComponent, selectors: [["source-files-component"]], inputs: { focusedSourceFileContent: "focusedSourceFileContent", focusedSourceLineSpec: "focusedSourceLineSpec" }, decls: 8, vars: 3, consts: [[1, "source-files-container"], [1, "header-section"], [1, "title-tag"], ["class", "file-label", 4, "ngIf", "ngIfElse"], ["noFileSelected", ""], [3, "lines", "focusedLineno", 4, "ngIf"], [1, "file-label"], [1, "no-file-selected"], [3, "lines", "focusedLineno"]], template: function SourceFilesComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "div", 0);
            ɵɵelementStart(1, "div", 1);
            ɵɵelementStart(2, "div", 2);
            ɵɵtext(3, "Source Code");
            ɵɵelementEnd();
            ɵɵtemplate(4, SourceFilesComponent_div_4_Template, 2, 1, "div", 3);
            ɵɵtemplate(5, SourceFilesComponent_ng_template_5_Template, 2, 0, "ng-template", null, 4, ɵɵtemplateRefExtractor);
            ɵɵelementEnd();
            ɵɵtemplate(7, SourceFilesComponent_tf_debugger_v2_source_code_7_Template, 1, 2, "tf-debugger-v2-source-code", 5);
            ɵɵelementEnd();
        } if (rf & 2) {
            const _r1 = ɵɵreference(6);
            ɵɵadvance(4);
            ɵɵproperty("ngIf", ctx.focusedSourceLineSpec !== null)("ngIfElse", _r1);
            ɵɵadvance(3);
            ɵɵproperty("ngIf", ctx.focusedSourceFileContent !== null && ctx.focusedSourceLineSpec !== null && ctx.focusedSourceFileContent.lines !== null);
        } }, directives: [NgIf, SourceCodeContainer], styles: [".header-section[_ngcontent-%COMP%] {\n  border-bottom: 1px solid rgba(0, 0, 0, 0.12);\n  display: flex;\n  height: 24px;\n  padding-bottom: 6px;\n  vertical-align: middle;\n  white-space: nowrap;\n  width: 100%;\n}\n\n.file-label[_ngcontent-%COMP%] {\n  display: inline-block;\n  font-weight: normal;\n  white-space: normal;\n  overflow-wrap: anywhere;\n  overflow-y: auto;\n  padding: 0 20px;\n}\n\n.no-file-selected[_ngcontent-%COMP%] {\n  display: inline-block;\n  color: #666;\n  padding: 0 20px;\n  white-space: normal;\n}\n\n.source-files-container[_ngcontent-%COMP%] {\n  display: flex;\n  flex-direction: column;\n  font-family: 'Roboto Mono', monospace;\n  font-size: 10px;\n  height: 100%;\n}\n\n.title-tag[_ngcontent-%COMP%] {\n  display: inline-block;\n  font-weight: bold;\n  height: 100%;\n  padding-left: 6px;\n  vertical-align: top;\n}\n\ntf-debugger-v2-source-code[_ngcontent-%COMP%] {\n  flex-grow: 1;\n  width: 100%;\n}"] });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/views/source_files/source_files_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class SourceFilesContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.focusedSourceFileContent$ = this.store.pipe(select(getFocusedSourceFileContent));
            this.focusedSourceLineSpec$ = this.store.pipe(select(getFocusedSourceLineSpec));
        }
    }
    SourceFilesContainer.decorators = [
        { type: Component, args: [{
                    selector: 'tf-debugger-v2-source-files',
                    template: `
    <source-files-component
      [focusedSourceFileContent]="focusedSourceFileContent$ | async"
      [focusedSourceLineSpec]="focusedSourceLineSpec$ | async"
    ></source-files-component>
  `,
                },] },
    ];
    /** @nocollapse */
    SourceFilesContainer.ctorParameters = () => [
        { type: Store }
    ];
    /** @nocollapse */ SourceFilesContainer.ɵfac = function SourceFilesContainer_Factory(t) { return new (t || SourceFilesContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ SourceFilesContainer.ɵcmp = ɵɵdefineComponent({ type: SourceFilesContainer, selectors: [["tf-debugger-v2-source-files"]], decls: 3, vars: 6, consts: [[3, "focusedSourceFileContent", "focusedSourceLineSpec"]], template: function SourceFilesContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelement(0, "source-files-component", 0);
            ɵɵpipe(1, "async");
            ɵɵpipe(2, "async");
        } if (rf & 2) {
            ɵɵproperty("focusedSourceFileContent", ɵɵpipeBind1(1, 2, ctx.focusedSourceFileContent$))("focusedSourceLineSpec", ɵɵpipeBind1(2, 4, ctx.focusedSourceLineSpec$));
        } }, directives: [SourceFilesComponent], pipes: [AsyncPipe], encapsulation: 2 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/views/stack_trace/stack_trace_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    const _c0$j = ["stackFrameArray"];
    function StackTraceComponent_div_4_div_3_span_2_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "span", 13);
        ɵɵtext(1);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r9 = ɵɵnextContext(3);
        ɵɵadvance(1);
        ɵɵtextInterpolate1(" #", ctx_r9.executionIndex, ": ");
    } }
    function StackTraceComponent_div_4_div_3_span_3_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "span", 14);
        ɵɵtext(1);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r10 = ɵɵnextContext(3);
        ɵɵadvance(1);
        ɵɵtextInterpolate1(" ", ctx_r10.opType, " ");
    } }
    function StackTraceComponent_div_4_div_3_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div");
        ɵɵtext(1, " Eager execution ");
        ɵɵtemplate(2, StackTraceComponent_div_4_div_3_span_2_Template, 2, 1, "span", 11);
        ɵɵtemplate(3, StackTraceComponent_div_4_div_3_span_3_Template, 2, 1, "span", 12);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r6 = ɵɵnextContext(2);
        ɵɵadvance(2);
        ɵɵproperty("ngIf", ctx_r6.opType !== null);
        ɵɵadvance(1);
        ɵɵproperty("ngIf", ctx_r6.opType !== null);
    } }
    function StackTraceComponent_div_4_div_4_span_2_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "span", 16);
        ɵɵtext(1);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r11 = ɵɵnextContext(3);
        ɵɵadvance(1);
        ɵɵtextInterpolate1(" \"", ctx_r11.opName, "\" ");
    } }
    function StackTraceComponent_div_4_div_4_span_3_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "span", 14);
        ɵɵtext(1);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r12 = ɵɵnextContext(3);
        ɵɵadvance(1);
        ɵɵtextInterpolate1(" ", ctx_r12.opType, " ");
    } }
    function StackTraceComponent_div_4_div_4_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div");
        ɵɵtext(1, " Creation of graph op ");
        ɵɵtemplate(2, StackTraceComponent_div_4_div_4_span_2_Template, 2, 1, "span", 15);
        ɵɵtemplate(3, StackTraceComponent_div_4_div_4_span_3_Template, 2, 1, "span", 12);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r7 = ɵɵnextContext(2);
        ɵɵadvance(2);
        ɵɵproperty("ngIf", ctx_r7.opName !== null);
        ɵɵadvance(1);
        ɵɵproperty("ngIf", ctx_r7.opType !== null);
    } }
    function StackTraceComponent_div_4_span_6_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "span", 17);
        ɵɵtext(1);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r8 = ɵɵnextContext(2);
        ɵɵadvance(1);
        ɵɵtextInterpolate1(" (Host name: ", ctx_r8.stackFramesForDisplay[0].host_name, ") ");
    } }
    function StackTraceComponent_div_4_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 7);
        ɵɵelementStart(1, "span");
        ɵɵelementStart(2, "span", 8);
        ɵɵtemplate(3, StackTraceComponent_div_4_div_3_Template, 4, 2, "div", 9);
        ɵɵtemplate(4, StackTraceComponent_div_4_div_4_Template, 4, 2, "div", 9);
        ɵɵelementEnd();
        ɵɵelementEnd();
        ɵɵelementStart(5, "div");
        ɵɵtemplate(6, StackTraceComponent_div_4_span_6_Template, 2, 1, "span", 10);
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r0 = ɵɵnextContext();
        ɵɵadvance(2);
        ɵɵproperty("ngSwitch", ctx_r0.codeLocationType);
        ɵɵadvance(1);
        ɵɵproperty("ngSwitchCase", ctx_r0.CodeLocationType.EXECUTION);
        ɵɵadvance(1);
        ɵɵproperty("ngSwitchCase", ctx_r0.CodeLocationType.GRAPH_OP_CREATION);
        ɵɵadvance(2);
        ɵɵproperty("ngIf", ctx_r0.stackFramesForDisplay !== null && ctx_r0.stackFramesForDisplay.length > 0);
    } }
    function StackTraceComponent_ng_template_5_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 18);
        ɵɵtext(1, " Click an eager execution or graph op to show its original stack trace. ");
        ɵɵelementEnd();
    } }
    function StackTraceComponent_div_7_div_2_div_4_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 28);
        ɵɵtext(1, " \u2913 ");
        ɵɵelementEnd();
    } }
    const _c1$a = function (a0, a1) { return [a0, a1]; };
    function StackTraceComponent_div_7_div_2_Template(rf, ctx) { if (rf & 1) {
        const _r19 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "div", 22);
        ɵɵelementStart(1, "div", 23);
        ɵɵtext(2);
        ɵɵelementEnd();
        ɵɵelementStart(3, "div", 24);
        ɵɵtemplate(4, StackTraceComponent_div_7_div_2_div_4_Template, 2, 0, "div", 25);
        ɵɵelementStart(5, "div", 26);
        ɵɵlistener("click", function StackTraceComponent_div_7_div_2_Template_div_click_5_listener() { ɵɵrestoreView(_r19); const stackFrameForDisplay_r15 = ctx.$implicit; const ctx_r18 = ɵɵnextContext(2); return ctx_r18.onSourceLineClicked.emit(stackFrameForDisplay_r15); });
        ɵɵtext(6);
        ɵɵelementEnd();
        ɵɵelementStart(7, "div", 27);
        ɵɵtext(8);
        ɵɵelementEnd();
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const stackFrameForDisplay_r15 = ctx.$implicit;
        const ctx_r14 = ɵɵnextContext(2);
        ɵɵproperty("ngClass", ɵɵpureFunction2(6, _c1$a, stackFrameForDisplay_r15.belongsToFocusedFile ? "focused-file" : "", stackFrameForDisplay_r15.focused ? "focused-stack-frame" : ""));
        ɵɵadvance(1);
        ɵɵpropertyInterpolate("title", stackFrameForDisplay_r15.file_path);
        ɵɵadvance(1);
        ɵɵtextInterpolate1(" ", stackFrameForDisplay_r15.concise_file_path, " ");
        ɵɵadvance(2);
        ɵɵproperty("ngIf", ctx_r14.stickToBottommostFrameInFocusedFile && stackFrameForDisplay_r15.focused);
        ɵɵadvance(2);
        ɵɵtextInterpolate1(" Line ", stackFrameForDisplay_r15.lineno, " ");
        ɵɵadvance(2);
        ɵɵtextInterpolate1(" ", stackFrameForDisplay_r15.function_name, " ");
    } }
    function StackTraceComponent_div_7_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 19, 20);
        ɵɵtemplate(2, StackTraceComponent_div_7_div_2_Template, 9, 9, "div", 21);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r3 = ɵɵnextContext();
        ɵɵadvance(2);
        ɵɵproperty("ngForOf", ctx_r3.stackFramesForDisplay);
    } }
    function StackTraceComponent_ng_template_8_Template(rf, ctx) { }
    class StackTraceComponent {
        constructor() {
            this.stackFramesForDisplay = null;
            this.onSourceLineClicked = new EventEmitter();
            this.CodeLocationType = CodeLocationType;
        }
        /**
         * @return {?}
         */
        ngAfterViewChecked() {
            if (this.stackFrameArray === undefined) {
                return;
            }
            /** @type {?} */
            const stackElement = this.stackFrameArray.nativeElement;
            /** @type {?} */
            const focusedFrameElement = stackElement.querySelector(`.focused-stack-frame`);
            if (focusedFrameElement !== null) {
                // Scroll the focused frame into view when there is a focused frame.
                this.scrollToElement(stackElement, focusedFrameElement);
                return;
            }
            /** @type {?} */
            const lastFrameElement = stackElement.querySelector('.stack-frame-container:last-child');
            if (lastFrameElement !== null) {
                this.scrollToElement(stackElement, lastFrameElement);
            }
        }
        /**
         * @param {?} parentElement
         * @param {?} element
         * @return {?}
         */
        scrollToElement(parentElement, element) {
            parentElement.scrollTop = element.offsetTop;
        }
    }
    StackTraceComponent.decorators = [
        { type: Component, args: [{
                    selector: 'stack-trace-component',
                    templateUrl: './stack_trace_component.ng.html',
                    styleUrls: ['./stack_trace_component.css'],
                },] },
    ];
    StackTraceComponent.propDecorators = {
        codeLocationType: [{ type: Input }],
        opType: [{ type: Input }],
        opName: [{ type: Input }],
        executionIndex: [{ type: Input }],
        stickToBottommostFrameInFocusedFile: [{ type: Input }],
        stackFramesForDisplay: [{ type: Input }],
        onSourceLineClicked: [{ type: Output }],
        stackFrameArray: [{ type: ViewChild, args: ['stackFrameArray',] }]
    };
    /** @nocollapse */ StackTraceComponent.ɵfac = function StackTraceComponent_Factory(t) { return new (t || StackTraceComponent)(); };
    /** @nocollapse */ StackTraceComponent.ɵcmp = ɵɵdefineComponent({ type: StackTraceComponent, selectors: [["stack-trace-component"]], viewQuery: function StackTraceComponent_Query(rf, ctx) { if (rf & 1) {
            ɵɵviewQuery(_c0$j, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.stackFrameArray = _t.first);
        } }, inputs: { codeLocationType: "codeLocationType", opType: "opType", opName: "opName", executionIndex: "executionIndex", stickToBottommostFrameInFocusedFile: "stickToBottommostFrameInFocusedFile", stackFramesForDisplay: "stackFramesForDisplay" }, outputs: { onSourceLineClicked: "onSourceLineClicked" }, decls: 10, vars: 4, consts: [[1, "stack-trace-container"], [1, "stack-trace-header"], [1, "stack-trace-title"], ["class", "stack-trace-aux-info", 4, "ngIf", "ngIfElse"], ["noStackTrace", ""], ["class", "stack-frame-array", 4, "ngIf", "ngIfElse"], ["loadingSection", ""], [1, "stack-trace-aux-info"], [1, "code-location-origin", 3, "ngSwitch"], [4, "ngSwitchCase"], ["class", "stack-trace-host-name", 4, "ngIf"], ["class", "eager-execution-index", 4, "ngIf"], ["class", "op-type", 4, "ngIf"], [1, "eager-execution-index"], [1, "op-type"], ["class", "op-name", 4, "ngIf"], [1, "op-name"], [1, "stack-trace-host-name"], [1, "stack-trace-aux-info", "no-stack-trace"], [1, "stack-frame-array"], ["stackFrameArray", ""], ["class", "stack-frame-container", 3, "ngClass", 4, "ngFor", "ngForOf"], [1, "stack-frame-container", 3, "ngClass"], [1, "stack-frame-file-path", 3, "title"], [1, "stack-frame-lineno-function"], ["class", "stick-to-bottommost-indicator", "title", "Sticking to the bottommost frame in the current source file when navigating executions and graph ops. To remove this sticking, click any non-bottommost stack frame.", 4, "ngIf"], [1, "stack-frame-lineno", 3, "click"], [1, "stack-frame-function"], ["title", "Sticking to the bottommost frame in the current source file when navigating executions and graph ops. To remove this sticking, click any non-bottommost stack frame.", 1, "stick-to-bottommost-indicator"]], template: function StackTraceComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "div", 0);
            ɵɵelementStart(1, "div", 1);
            ɵɵelementStart(2, "span", 2);
            ɵɵtext(3, " Stack Trace ");
            ɵɵelementEnd();
            ɵɵtemplate(4, StackTraceComponent_div_4_Template, 7, 4, "div", 3);
            ɵɵtemplate(5, StackTraceComponent_ng_template_5_Template, 2, 0, "ng-template", null, 4, ɵɵtemplateRefExtractor);
            ɵɵelementEnd();
            ɵɵtemplate(7, StackTraceComponent_div_7_Template, 3, 1, "div", 5);
            ɵɵtemplate(8, StackTraceComponent_ng_template_8_Template, 0, 0, "ng-template", null, 6, ɵɵtemplateRefExtractor);
            ɵɵelementEnd();
        } if (rf & 2) {
            const _r1 = ɵɵreference(6);
            const _r4 = ɵɵreference(9);
            ɵɵadvance(4);
            ɵɵproperty("ngIf", ctx.codeLocationType !== null)("ngIfElse", _r1);
            ɵɵadvance(3);
            ɵɵproperty("ngIf", ctx.stackFramesForDisplay !== null)("ngIfElse", _r4);
        } }, directives: [NgIf, NgSwitch, NgSwitchCase, NgForOf, NgClass], styles: [".focused-file[_ngcontent-%COMP%] {\n  font-weight: bold;\n}\n\n.focused-stack-frame[_ngcontent-%COMP%] {\n  background-color: rgba(255, 111, 0, 0.3);\n}\n\n.no-stack-trace[_ngcontent-%COMP%] {\n  color: #808080;\n}\n\n.op-name[_ngcontent-%COMP%] {\n  word-wrap: anywhere;\n}\n\n.op-type[_ngcontent-%COMP%] {\n  background-color: #e3e5e8;\n  border: 1px solid #c0c0c0;\n  border-radius: 4px;\n  font-family: 'Roboto Mono', monospace;\n  font-size: 10px;\n  height: 14px;\n  line-height: 14px;\n  padding: 1px 3px;\n  width: max-content;\n}\n\n.stack-frame-array[_ngcontent-%COMP%] {\n  overflow-x: hidden;\n  overflow-y: auto;\n  \n  width: calc(100% - 8px);\n}\n\n.stack-frame-container[_ngcontent-%COMP%] {\n  border-bottom: 1px solid #a0a0a0;\n}\n\n.stack-frame-file-path[_ngcontent-%COMP%] {\n  max-width: 180px;\n  width: 180px;\n}\n\n.stack-frame-lineno-function[_ngcontent-%COMP%] {\n  text-align: right;\n  white-space: nowrap;\n}\n\n.stack-frame-function[_ngcontent-%COMP%] {\n  display: inline-block;\n  max-width: 200px;\n  padding-left: 10px;\n  text-align: left;\n  white-space: normal;\n  width: 200px;\n  word-wrap: anywhere;\n}\n\n.stack-frame-lineno[_ngcontent-%COMP%] {\n  cursor: pointer;\n  display: inline-block;\n  max-width: 80px;\n  text-align: left;\n  text-decoration: underline;\n  width: 80px;\n}\n\n.stack-trace-aux-info[_ngcontent-%COMP%] {\n  margin-top: 15px;\n  padding-left: 24px;\n}\n\n.stack-trace-container[_ngcontent-%COMP%] {\n  border-left: 1px solid rgba(0, 0, 0, 0.12);\n  box-sizing: border-box;\n  display: flex;\n  flex-flow: column;\n  font-size: 10px;\n  font-family: 'Roboto Mono', monospace;\n  height: 100%;\n  margin-left: 8px;\n  max-height: 360px;\n  overflow-x: hidden;\n  overflow-y: hidden;\n  padding-left: 8px;\n  width: 100%;\n}\n\n.stack-trace-header[_ngcontent-%COMP%] {\n  box-shadow: 0 5px 3px -3px #ccc;\n  padding-bottom: 3px;\n}\n.stack-trace-host-name[_ngcontent-%COMP%] {\n  color: #808080;\n}\n\n.stack-trace-title[_ngcontent-%COMP%] {\n  font-weight: bold;\n}\n\n.stick-to-bottommost-indicator[_ngcontent-%COMP%] {\n  display: inline-block;\n  font-weight: bold;\n  font-size: 12px;\n  padding-right: 3px;\n}"] });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/views/stack_trace/stack_trace_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class StackTraceContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.codeLocationType$ = this.store.pipe(select(createSelector(getCodeLocationOrigin, (/**
             * @param {?} originInfo
             * @return {?}
             */
            (originInfo) => {
                return originInfo === null ? null : originInfo.codeLocationType;
            }))));
            this.opType$ = this.store.pipe(select(createSelector(getCodeLocationOrigin, (/**
             * @param {?} originInfo
             * @return {?}
             */
            (originInfo) => {
                return originInfo === null ? null : originInfo.opType;
            }))));
            this.opName$ = this.store.pipe(select(createSelector(getCodeLocationOrigin, (/**
             * @param {?} originInfo
             * @return {?}
             */
            (originInfo) => {
                if (originInfo === null ||
                    originInfo.codeLocationType !== CodeLocationType.GRAPH_OP_CREATION) {
                    return null;
                }
                return originInfo.opName;
            }))));
            this.executionIndex$ = this.store.pipe(select(createSelector(getCodeLocationOrigin, (/**
             * @param {?} originInfo
             * @return {?}
             */
            (originInfo) => {
                if (originInfo === null ||
                    originInfo.codeLocationType !== CodeLocationType.EXECUTION) {
                    return null;
                }
                return originInfo.executionIndex;
            }))));
            this.stickToBottommostFrameInFocusedFile$ = this.store.pipe(select(getStickToBottommostFrameInFocusedFile));
            this.stackFramesForDisplay$ = this.store.pipe(select(createSelector(getFocusedStackFrames, getFocusedSourceLineSpec, (/**
             * @param {?} stackFrames
             * @param {?} focusedSourceLineSpec
             * @return {?}
             */
            (stackFrames, focusedSourceLineSpec) => {
                if (stackFrames === null) {
                    return null;
                }
                /** @type {?} */
                const output = [];
                // Correctly label all the stack frames for display.
                for (const stackFrame of stackFrames) {
                    const { host_name, file_path, lineno, function_name } = stackFrame;
                    /** @type {?} */
                    const pathItems = file_path.split('/');
                    /** @type {?} */
                    const concise_file_path = pathItems[pathItems.length - 1];
                    /** @type {?} */
                    const belongsToFocusedFile = focusedSourceLineSpec !== null &&
                        host_name === focusedSourceLineSpec.host_name &&
                        file_path === focusedSourceLineSpec.file_path;
                    /** @type {?} */
                    const focused = belongsToFocusedFile && lineno === (/** @type {?} */ (focusedSourceLineSpec)).lineno;
                    output.push({
                        host_name,
                        file_path,
                        concise_file_path,
                        lineno,
                        function_name,
                        belongsToFocusedFile,
                        focused,
                    });
                }
                return output;
            }))));
        }
        /**
         * @param {?} args
         * @return {?}
         */
        onSourceLineClicked(args) {
            const { host_name, file_path, lineno, function_name } = args;
            /** @type {?} */
            const stackFrame = { host_name, file_path, lineno, function_name };
            this.store.dispatch(sourceLineFocused({ stackFrame }));
        }
    }
    StackTraceContainer.decorators = [
        { type: Component, args: [{
                    selector: 'tf-debugger-v2-stack-trace',
                    template: `
    <stack-trace-component
      [codeLocationType]="codeLocationType$ | async"
      [opType]="opType$ | async"
      [opName]="opName$ | async"
      [executionIndex]="executionIndex$ | async"
      [stickToBottommostFrameInFocusedFile]="
        stickToBottommostFrameInFocusedFile$ | async
      "
      [stackFramesForDisplay]="stackFramesForDisplay$ | async"
      (onSourceLineClicked)="onSourceLineClicked($event)"
    ></stack-trace-component>
  `,
                },] },
    ];
    /** @nocollapse */
    StackTraceContainer.ctorParameters = () => [
        { type: Store }
    ];
    /** @nocollapse */ StackTraceContainer.ɵfac = function StackTraceContainer_Factory(t) { return new (t || StackTraceContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ StackTraceContainer.ɵcmp = ɵɵdefineComponent({ type: StackTraceContainer, selectors: [["tf-debugger-v2-stack-trace"]], decls: 7, vars: 18, consts: [[3, "codeLocationType", "opType", "opName", "executionIndex", "stickToBottommostFrameInFocusedFile", "stackFramesForDisplay", "onSourceLineClicked"]], template: function StackTraceContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "stack-trace-component", 0);
            ɵɵlistener("onSourceLineClicked", function StackTraceContainer_Template_stack_trace_component_onSourceLineClicked_0_listener($event) { return ctx.onSourceLineClicked($event); });
            ɵɵpipe(1, "async");
            ɵɵpipe(2, "async");
            ɵɵpipe(3, "async");
            ɵɵpipe(4, "async");
            ɵɵpipe(5, "async");
            ɵɵpipe(6, "async");
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵproperty("codeLocationType", ɵɵpipeBind1(1, 6, ctx.codeLocationType$))("opType", ɵɵpipeBind1(2, 8, ctx.opType$))("opName", ɵɵpipeBind1(3, 10, ctx.opName$))("executionIndex", ɵɵpipeBind1(4, 12, ctx.executionIndex$))("stickToBottommostFrameInFocusedFile", ɵɵpipeBind1(5, 14, ctx.stickToBottommostFrameInFocusedFile$))("stackFramesForDisplay", ɵɵpipeBind1(6, 16, ctx.stackFramesForDisplay$));
        } }, directives: [StackTraceComponent], pipes: [AsyncPipe], encapsulation: 2 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/debugger_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    function DebuggerComponent_tf_debugger_v2_inactive_1_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "tf-debugger-v2-inactive");
    } }
    function DebuggerComponent_ng_template_2_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 3);
        ɵɵelement(1, "tf-debugger-v2-alerts");
        ɵɵelementStart(2, "div", 4);
        ɵɵelement(3, "tf-debugger-v2-timeline");
        ɵɵelement(4, "tf-debugger-v2-graph");
        ɵɵelementEnd();
        ɵɵelement(5, "tf-debugger-v2-graph-executions");
        ɵɵelementEnd();
        ɵɵelementStart(6, "div", 5);
        ɵɵelement(7, "tf-debugger-v2-source-files");
        ɵɵelement(8, "tf-debugger-v2-stack-trace");
        ɵɵelementEnd();
    } }
    class DebuggerComponent {
        constructor() {
            this.runs = {};
            this.runIds = [];
            this.activeRunId = null;
        }
    }
    DebuggerComponent.decorators = [
        { type: Component, args: [{
                    selector: 'debugger-component',
                    templateUrl: './debugger_component.ng.html',
                    styleUrls: ['./debugger_component.css'],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    DebuggerComponent.propDecorators = {
        runs: [{ type: Input }],
        runIds: [{ type: Input }],
        activeRunId: [{ type: Input }]
    };
    /** @nocollapse */ DebuggerComponent.ɵfac = function DebuggerComponent_Factory(t) { return new (t || DebuggerComponent)(); };
    /** @nocollapse */ DebuggerComponent.ɵcmp = ɵɵdefineComponent({ type: DebuggerComponent, selectors: [["debugger-component"]], inputs: { runs: "runs", runIds: "runIds", activeRunId: "activeRunId" }, decls: 4, vars: 2, consts: [[1, "debugger-container"], [4, "ngIf", "ngIfElse"], ["dataAvailable", ""], [1, "top-section"], [1, "top-center-section"], [1, "bottom-section"]], template: function DebuggerComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "div", 0);
            ɵɵtemplate(1, DebuggerComponent_tf_debugger_v2_inactive_1_Template, 1, 0, "tf-debugger-v2-inactive", 1);
            ɵɵtemplate(2, DebuggerComponent_ng_template_2_Template, 9, 0, "ng-template", null, 2, ɵɵtemplateRefExtractor);
            ɵɵelementEnd();
        } if (rf & 2) {
            const _r1 = ɵɵreference(3);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx.runIds.length === 0)("ngIfElse", _r1);
        } }, directives: [NgIf, InactiveContainer, AlertsContainer, TimelineContainer, GraphContainer, GraphExecutionsContainer, SourceFilesContainer, StackTraceContainer], styles: [".bottom-section[_ngcontent-%COMP%] {\n  box-sizing: border-box;\n  border-top: 1px solid rgba(0, 0, 0, 0.12);\n  display: flex;\n  flex-grow: 1;\n  height: 34%;\n  padding-top: 6px;\n}\n\n.debugger-container[_ngcontent-%COMP%] {\n  box-sizing: border-box;\n  background-color: #fff;\n  height: 100%;\n  overflow: hidden;\n}\n\n.top-section[_ngcontent-%COMP%] {\n  box-sizing: border-box;\n  display: flex;\n  flex-grow: 1;\n  height: 66%;\n  padding: 6px 0;\n}\n\ntf-debugger-v2-alerts[_ngcontent-%COMP%] {\n  border-right: 1px solid rgba(0, 0, 0, 0.12);\n  display: inline-block;\n  margin-right: 10px;\n  min-width: 160px;\n  width: calc(15% - 11px);\n}\n\ntf-debugger-v2-graph-executions[_ngcontent-%COMP%] {\n  display: inline-block;\n  flex-grow: 1;\n  min-width: 540px;\n  width: 540px;\n}\n\ntf-debugger-v2-source-files[_ngcontent-%COMP%] {\n  display: inline-block;\n  height: 100%;\n  width: 70%;\n}\n\ntf-debugger-v2-stack-trace[_ngcontent-%COMP%] {\n  display: inline-block;\n  flex-grow: 1;\n  height: 100%;\n  min-width: 540px;\n  width: 540px;\n}\n\n.top-center-section[_ngcontent-%COMP%] {\n  display: inline-block;\n  overflow: auto;\n  width: 55%;\n}\n\ntf-debugger-v2-timeline[_ngcontent-%COMP%] {\n  display: block;\n}\n\ntf-debugger-v2-graph[_ngcontent-%COMP%] {\n  border-top: 1px solid rgba(0, 0, 0, 0.12);\n  display: block;\n  margin-top: 5px;\n}"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/debugger_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class DebuggerContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.runs$ = this.store.pipe(select(getDebuggerRunListing));
            this.runsIds$ = this.store.pipe(select(createSelector(getDebuggerRunListing, (/**
             * @param {?} runs
             * @return {?}
             */
            (runs) => Object.keys(runs)))));
            this.activeRunId$ = this.store.pipe(select(getActiveRunId));
        }
        /**
         * @return {?}
         */
        ngOnInit() {
            this.store.dispatch(debuggerLoaded());
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this.store.dispatch(debuggerUnloaded());
        }
    }
    DebuggerContainer.decorators = [
        { type: Component, args: [{
                    selector: 'tf-debugger-v2',
                    template: `
    <debugger-component
      [runs]="runs$ | async"
      [runIds]="runsIds$ | async"
      [activeRunId]="activeRunId$ | async"
    ></debugger-component>
  `,
                },] },
    ];
    /** @nocollapse */
    DebuggerContainer.ctorParameters = () => [
        { type: Store }
    ];
    /** @nocollapse */ DebuggerContainer.ɵfac = function DebuggerContainer_Factory(t) { return new (t || DebuggerContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ DebuggerContainer.ɵcmp = ɵɵdefineComponent({ type: DebuggerContainer, selectors: [["tf-debugger-v2"]], decls: 4, vars: 9, consts: [[3, "runs", "runIds", "activeRunId"]], template: function DebuggerContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelement(0, "debugger-component", 0);
            ɵɵpipe(1, "async");
            ɵɵpipe(2, "async");
            ɵɵpipe(3, "async");
        } if (rf & 2) {
            ɵɵproperty("runs", ɵɵpipeBind1(1, 3, ctx.runs$))("runIds", ɵɵpipeBind1(2, 5, ctx.runsIds$))("activeRunId", ɵɵpipeBind1(3, 7, ctx.activeRunId$));
        } }, directives: [DebuggerComponent], pipes: [AsyncPipe], encapsulation: 2 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/effects/debugger_effects.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    // Minimum polling interval in milliseconds.
    /** @type {?} */
    const MIN_POLLING_INTERVAL_MS = 2e3;
    // Maximum polling interval in miliseconds.
    /** @type {?} */
    const MAX_POLLING_INTERVAL_MS = 60e3;
    // Backoff behavior takes effect when time since the elapsed time between the
    // most recent arrival of new polling result and the most recent polling event
    // exceeds `pollingBackoffFactor * minimumPollingInterval`.
    /** @type {?} */
    const POLLING_BACKOFF_FACTOR = 2;
    /**
     * Computes the current polling interval based on the time lapsed since
     * last new polling data and the last polling event.
     *
     * This specifies a backoff behavior where a longer period of no new data
     * leads to a longer polling interval, with a ceiling as the limit.
     *
     * @param {?} pollSilenceTime The amount of time between the most
     *   recent arrival of new polling result and the last polling event.
     * @return {?} The current polling interval, i.e., low bound for the time between
     *   the last polling event and the next one.
     */
    function getCurrentPollingInterval(pollSilenceTime) {
        if (pollSilenceTime > MAX_POLLING_INTERVAL_MS) {
            return MAX_POLLING_INTERVAL_MS;
        }
        else if (pollSilenceTime >
            MIN_POLLING_INTERVAL_MS * POLLING_BACKOFF_FACTOR) {
            return pollSilenceTime;
        }
        else {
            return MIN_POLLING_INTERVAL_MS;
        }
    }
    /**
     * Getting page indices that are missing from the data and hence need to be
     * requested.
     *
     * @param {?} begin Beginning item index (inclusive).
     * @param {?} end Ending item index (exclusive).
     * @param {?} pageSize Size of each page, i.e., how many items there are in each
     *   complete page. This must be `>= end - begin`, or an Error will be
     *   thrown.
     * @param {?} numItems
     * @param {?} pageLoadedSizes A map from page index to how many items have been
     *   loaded for that page so far.
     * @return {?} An array of the page indices that are currently missing and hence
     *   should be requested from the appropriate data source.
     */
    function getMissingPages(begin, end, pageSize, numItems, pageLoadedSizes) {
        if (pageSize <= 0 || !Number.isInteger(pageSize)) {
            throw new Error(`Invalid pageSize: ${pageSize}`);
        }
        if (end > numItems) {
            throw new Error(`end index (${end}) exceeds total number of items (${numItems})`);
        }
        if (end - begin > pageSize) {
            throw new Error('begin-end span exceeds page size, which is not allowed');
        }
        // The constraint that `end - begin <= page` guarantees that at most only two
        // pages need to be requested.
        /** @type {?} */
        const missingPages = [];
        // Check whether the first of the two possible pages needs to be requested.
        /** @type {?} */
        const firstPageIndex = Math.floor(begin / pageSize);
        if (!(firstPageIndex in pageLoadedSizes) ||
            (pageLoadedSizes[firstPageIndex] < pageSize &&
                firstPageIndex * pageSize + pageLoadedSizes[firstPageIndex] < numItems)) {
            missingPages.push(firstPageIndex);
        }
        /** @type {?} */
        const secondPageIndex = Math.floor((end - 1) / pageSize);
        if (secondPageIndex !== firstPageIndex) {
            /** @type {?} */
            const anyDigestMissing1 = !(secondPageIndex in pageLoadedSizes) ||
                (secondPageIndex * pageSize + pageLoadedSizes[secondPageIndex] < end &&
                    end < numItems);
            if (anyDigestMissing1) {
                missingPages.push(secondPageIndex);
            }
        }
        return missingPages;
    }
    /**
     * Repeat an input stream at given intervals.
     * @param {?} stream$ The stream to repeat.
     * @param {?} pollingIntervalStream$ The stream of polling intervals.
     *   The latest value of this stream will be used as the interval between
     *   the previous repeat and the next one.
     * @param {?} terminationStream$ Terminate the repeat on values from this stream.
     * @return {?} Stream with the timed repeats.
     */
    function createTimedRepeater(stream$, pollingIntervalStream$, terminationStream$) {
        return stream$.pipe(repeatWhen((/**
         * @param {?} completed
         * @return {?}
         */
        (completed) => completed.pipe(withLatestFrom(pollingIntervalStream$), delayWhen((/**
         * @param {?} __0
         * @return {?}
         */
        ([, pollingInterval]) => {
            return timer(pollingInterval);
        }))))), takeUntil(terminationStream$), map((/**
         * @return {?}
         */
        () => void null)));
    }
    class DebuggerEffects {
        /**
         * @param {?} actions$
         * @param {?} store
         * @param {?} dataSource
         */
        constructor(actions$, store, dataSource) {
            this.actions$ = actions$;
            this.store = store;
            this.dataSource = dataSource;
            /**
             * view load and subsequent
             * backoff polls, auto and
             * manual reload events
             *  |
             *  +> fetch run-+> fetch source-file list
             *  |            +> fetch num of top-level (eager) executions
             *  |            +> fetch num of intra-graph executions
             *  |            +> fetch num alerts
             *  |                +
             *  |                +> if init load and non-zero number of execs
             *  |                    +
             *  |                    +>+-----------------------------+
             *  |  on top-level      | | fetch top-level exec digest |
             *  |  scroll -----------+>+-----------------------------+<--------+
             *  |                    |                                         |
             *  |                    +>+----------------------------------+    |
             *  |                      | fetch exec data and stack frames |    |
             *  |  on focus  +-------->+----------------------------------+    |
             *  |                                                              |
             *  |                                                              |
             *  |                                                              |
             *  +------>+ fetch alert number and breakdown                     |
             *                                                                 |
             * on alert type focus --------> fetch alerts of a type -----------+
             *
             * on source file requested ---> fetch source file
             *
             * on graph-execution scroll --> fetch graph-execution data
             *
             * on graph-op focus & graph-execution focus --> fetch graph-op info ---+
             *                                                                      |
             *                                            fetch stack frames <------+
             **/
            this.loadData$ = createEffect((/**
             * @return {?}
             */
            () => {
                // This event can trigger the loading of
                //   - list of source files.
                //   - number of executions
                //   - number and breakdown of alerts.
                // Therefore it needs to be a shared observable.
                /** @type {?} */
                const loadRunData$ = this.loadDebuggerRuns(merge(this.onDebuggerDataPoll(), this.onCoreReload())).pipe(share());
                /** @type {?} */
                const loadSourceFileList$ = this.loadSourceFileList(loadRunData$);
                /** @type {?} */
                const onNumExecutionLoaded$ = this.createNumExecutionLoader(loadRunData$);
                /** @type {?} */
                const onNumAlertsLoaded$ = this.createNumAlertsAndBreakdownLoader(loadRunData$);
                /** @type {?} */
                const onAlertTypeFocused$ = this.onAlertTypeFocused();
                /** @type {?} */
                const fetchExecutionDigestsForAlertTypeFocus$ = this.fetchExecutionDigestsForAlertTypeFocus(onAlertTypeFocused$);
                // This event can trigger the loading of
                //   - execution-digest
                //   - first execution data.
                // Therefore it needs to be a shared observable.
                /** @type {?} */
                const onInitialExecution$ = this.createInitialExecutionDetector(onNumExecutionLoaded$).pipe(share());
                /** @type {?} */
                const onExcutionDigestLoaded$ = this.createExecutionDigestLoader(merge(this.onExecutionScroll(), this.createInitialExecutionDigest(onInitialExecution$), fetchExecutionDigestsForAlertTypeFocus$));
                /** @type {?} */
                const onExecutionDataLoaded$ = this.createExecutionDataAndStackFramesLoader(merge(this.onExecutionDigestFocused(), onInitialExecution$.pipe(withLatestFrom(this.store.select(getActiveRunId), this.store.select(getLoadedExecutionData)), map((/**
                 * @param {?} __0
                 * @return {?}
                 */
                ([, activeRunId, loadedExecutionData]) => {
                    return {
                        activeRunId: (/** @type {?} */ (activeRunId)),
                        loadedExecutionData,
                        focusIndex: 0,
                    };
                })))));
                /** @type {?} */
                const onNumGraphExecutionLoaded$ = this.createNumGraphExecutionLoader(loadRunData$);
                /** @type {?} */
                const onSourceFileFocused$ = this.onSourceFileFocused();
                /** @type {?} */
                const onGraphExecutionScroll$ = this.loadGraphExecutionPages(this.onGraphExecutionScroll());
                /** @type {?} */
                const loadGraphOpInfoAndStackTrace$ = this.loadGraphOpStackFrames(this.loadGraphOpInfo());
                // ExecutionDigest and ExecutionData can be loaded in parallel.
                return merge(onNumAlertsLoaded$, onExcutionDigestLoaded$, onExecutionDataLoaded$, onNumGraphExecutionLoaded$, loadSourceFileList$, onSourceFileFocused$, onGraphExecutionScroll$, loadGraphOpInfoAndStackTrace$).pipe(
                // createEffect expects an Observable that emits {}.
                map((/**
                 * @return {?}
                 */
                () => ({}))));
            }), { dispatch: false });
        }
        /**
         * @private
         * @return {?}
         */
        onDebuggerDataPoll() {
            return this.actions$.pipe(ofType(debuggerLoaded), switchMap((/**
             * @param {?} action
             * @return {?}
             */
            (action) => {
                return createTimedRepeater(of(action), this.store.select(getPollSilenceTimeMs).pipe(map((/**
                 * @param {?} pollSilenceTime
                 * @return {?}
                 */
                (pollSilenceTime) => {
                    return getCurrentPollingInterval(pollSilenceTime);
                }))), this.actions$.pipe(ofType(debuggerUnloaded)));
            })), tap((/**
             * @return {?}
             */
            () => this.store.dispatch(debuggerDataPollOnset()))), map((/**
             * @return {?}
             */
            () => void null)));
        }
        /**
         * @private
         * @return {?}
         */
        onCoreReload() {
            return this.actions$.pipe(ofType(manualReload, reload), tap((/**
             * @return {?}
             */
            () => this.store.dispatch(debuggerDataPollOnset()))), map((/**
             * @return {?}
             */
            () => void null)));
        }
        /**
         * Request list of debugger runs.
         * @private
         * @param {?} prevStream$
         * @return {?}
         */
        loadDebuggerRuns(prevStream$) {
            return prevStream$.pipe(withLatestFrom(this.store.select(getDebuggerRunsLoaded)), filter((/**
             * @param {?} __0
             * @return {?}
             */
            ([, { state }]) => state !== DataLoadState.LOADING)), tap((/**
             * @return {?}
             */
            () => this.store.dispatch(debuggerRunsRequested()))), mergeMap((/**
             * @return {?}
             */
            () => {
                return this.dataSource.fetchRuns().pipe(tap((/**
                 * @param {?} runs
                 * @return {?}
                 */
                (runs) => {
                    this.store.dispatch(debuggerRunsLoaded({ runs: (/** @type {?} */ (runs)) }));
                })), map((/**
                 * @return {?}
                 */
                () => void null))
                // TODO(cais): Add catchError() to pipe.
                );
            })));
        }
        /**
         * When a debugger run exists, load number of top-level executions.
         * @private
         * @param {?} prevStream$
         * @return {?}
         */
        createNumExecutionLoader(prevStream$) {
            return prevStream$.pipe(withLatestFrom(this.store.select(getDebuggerRunListing), this.store.select(getNumExecutionsLoaded)), filter((/**
             * @param {?} __0
             * @return {?}
             */
            ([, runs, loaded]) => {
                return (Object.keys(runs).length > 0 && loaded.state !== DataLoadState.LOADING);
            })), tap((/**
             * @return {?}
             */
            () => this.store.dispatch(numExecutionsRequested()))), mergeMap((/**
             * @param {?} __0
             * @return {?}
             */
            ([, runs]) => {
                // TODO(cais): Handle multiple runs. Currently it is assumed that there
                // is at most only one debugger run available.
                /** @type {?} */
                const runId = Object.keys(runs)[0];
                /** @type {?} */
                const begin = 0;
                /** @type {?} */
                const end = 0;
                return this.dataSource.fetchExecutionDigests(runId, begin, end).pipe(tap((/**
                 * @param {?} digests
                 * @return {?}
                 */
                (digests) => {
                    this.store.dispatch(numExecutionsLoaded({ numExecutions: digests.num_digests }));
                })), map((/**
                 * @return {?}
                 */
                () => void null)));
                // TODO(cais): Add catchError() to pipe.
            })));
        }
        /**
         * When a debugger run exists, load number of intra-graph executions.
         * @private
         * @param {?} prevStream$
         * @return {?}
         */
        createNumGraphExecutionLoader(prevStream$) {
            return prevStream$.pipe(withLatestFrom(this.store.select(getDebuggerRunListing), this.store.select(getNumGraphExecutionsLoaded)), filter((/**
             * @param {?} __0
             * @return {?}
             */
            ([, runs, loaded]) => {
                return (Object.keys(runs).length > 0 && loaded.state !== DataLoadState.LOADING);
            })), tap((/**
             * @return {?}
             */
            () => this.store.dispatch(numGraphExecutionsRequested()))), mergeMap((/**
             * @param {?} __0
             * @return {?}
             */
            ([, runs]) => {
                /** @type {?} */
                const runId = Object.keys(runs)[0];
                /** @type {?} */
                const begin = 0;
                /** @type {?} */
                const end = 0;
                return this.dataSource
                    .fetchGraphExecutionDigests(runId, begin, end)
                    .pipe(tap((/**
                 * @param {?} digests
                 * @return {?}
                 */
                (digests) => {
                    this.store.dispatch(numGraphExecutionsLoaded({
                        numGraphExecutions: digests.num_digests,
                    }));
                })), map((/**
                 * @return {?}
                 */
                () => void null)));
                // TODO(cais): Add catchError() to pipe.
            })));
        }
        /**
         * When a debugger run exists, load number of alerts and their breakdown.
         * @private
         * @param {?} prevStream$
         * @return {?}
         */
        createNumAlertsAndBreakdownLoader(prevStream$) {
            return prevStream$.pipe(withLatestFrom(this.store.select(getDebuggerRunListing), this.store.select(getAlertsLoaded)), filter((/**
             * @param {?} __0
             * @return {?}
             */
            ([, runs, loaded]) => {
                return (Object.keys(runs).length > 0 && loaded.state !== DataLoadState.LOADING);
            })), tap((/**
             * @return {?}
             */
            () => this.store.dispatch(numAlertsAndBreakdownRequested()))), mergeMap((/**
             * @param {?} __0
             * @return {?}
             */
            ([, runs]) => {
                /** @type {?} */
                const runId = Object.keys(runs)[0];
                /** @type {?} */
                const begin = 0;
                /** @type {?} */
                const end = 0;
                return this.dataSource.fetchAlerts(runId, begin, end).pipe(tap((/**
                 * @param {?} alerts
                 * @return {?}
                 */
                (alerts) => {
                    this.store.dispatch(numAlertsAndBreakdownLoaded({
                        numAlerts: alerts.num_alerts,
                        alertsBreakdown: alerts.alerts_breakdown,
                    }));
                })), map((/**
                 * @return {?}
                 */
                () => void null)));
            })));
        }
        /**
         * Emits when initial execution digests and data are required.
         *
         * These initial data loading actions are required when the number of
         * executions is greater than zero.
         * @private
         * @param {?} prevStream$
         * @return {?}
         */
        createInitialExecutionDetector(prevStream$) {
            return prevStream$.pipe(withLatestFrom(this.store.select(getNumExecutions), this.store.select(getExecutionDigestsLoaded)), filter((/**
             * @param {?} __0
             * @return {?}
             */
            ([, numExecutions, executionDigestsLoaded]) => {
                return (numExecutions > 0 &&
                    Object.keys(executionDigestsLoaded.pageLoadedSizes).length === 0);
            })), map((/**
             * @return {?}
             */
            () => void null)));
        }
        /**
         * Emits when the first page if execution digests are required to be loaded.
         * @private
         * @param {?} prevStream$
         * @return {?}
         */
        createInitialExecutionDigest(prevStream$) {
            return prevStream$.pipe(withLatestFrom(this.store.select(getNumExecutions), this.store.select(getActiveRunId), this.store.select(getExecutionPageSize)), filter((/**
             * @param {?} __0
             * @return {?}
             */
            ([, , runId]) => runId !== null)), map((/**
             * @param {?} __0
             * @return {?}
             */
            ([, numExecutions, runId, pageSize]) => {
                /** @type {?} */
                const begin = 0;
                /** @type {?} */
                const end = Math.min(numExecutions, pageSize);
                return { begin, end, runId: (/** @type {?} */ (runId)) };
            })));
        }
        /**
         * Emits when scrolling event leads to need to load new execution digests.
         * @private
         * @return {?}
         */
        onExecutionScroll() {
            return this.actions$.pipe(ofType(executionScrollLeft, executionScrollRight, executionScrollToIndex), withLatestFrom(this.store.select(getActiveRunId), this.store.select(getExecutionScrollBeginIndex), this.store.select(getNumExecutions), this.store.select(getDisplayCount), this.store.select(getExecutionPageSize)), filter((/**
             * @param {?} __0
             * @return {?}
             */
            ([runId]) => runId !== null)), map((/**
             * @param {?} __0
             * @return {?}
             */
            ([, runId, scrollBeginIndex, numExecutions, displayCount, pageSize,]) => {
                /** @type {?} */
                const begin = scrollBeginIndex;
                /** @type {?} */
                const end = Math.min(numExecutions, begin + displayCount);
                return {
                    runId: (/** @type {?} */ (runId)),
                    begin,
                    end,
                    pageSize,
                };
            })), withLatestFrom(this.store.select(getExecutionDigestsLoaded)), map((/**
             * @param {?} __0
             * @return {?}
             */
            ([props, loaded]) => {
                return {
                    props,
                    loaded,
                    missingPages: getMissingPages(props.begin, props.end, props.pageSize, loaded.numExecutions, loaded.pageLoadedSizes),
                };
            })), filter((/**
             * @param {?} __0
             * @return {?}
             */
            ({ missingPages }) => missingPages.length > 0)), map((/**
             * @param {?} __0
             * @return {?}
             */
            ({ props, loaded, missingPages }) => {
                const { runId, pageSize } = props;
                /** @type {?} */
                const begin = missingPages[0] * pageSize;
                /** @type {?} */
                const end = Math.min(loaded.numExecutions, (missingPages[missingPages.length - 1] + 1) * pageSize);
                return { begin, end, runId: (/** @type {?} */ (runId)) };
            })));
        }
        /**
         * Load execution digests.
         * @private
         * @param {?} prevStream$
         * @return {?}
         */
        createExecutionDigestLoader(prevStream$) {
            return prevStream$.pipe(withLatestFrom(this.store.select(getExecutionDigestsLoaded)), filter((/**
             * @param {?} __0
             * @return {?}
             */
            ([{ begin, end }, loaded]) => {
                return (end > begin &&
                    !beginEndRangesInclude(loaded.loadingRanges, begin, end));
            })), tap((/**
             * @param {?} __0
             * @return {?}
             */
            ([{ begin, end }]) => {
                this.store.dispatch(executionDigestsRequested({ begin, end }));
            })), mergeMap((/**
             * @param {?} __0
             * @return {?}
             */
            ([{ runId, begin, end }]) => {
                return this.dataSource.fetchExecutionDigests(runId, begin, end).pipe(tap((/**
                 * @param {?} digests
                 * @return {?}
                 */
                (digests) => {
                    this.store.dispatch(executionDigestsLoaded(digests));
                })), map((/**
                 * @return {?}
                 */
                () => void null)));
                // TODO(cais): Add catchError() to pipe.
            })));
        }
        /**
         * Emits when user focses on an execution digest.
         * @private
         * @return {?}
         */
        onExecutionDigestFocused() {
            return this.actions$.pipe(ofType(executionDigestFocused), withLatestFrom(this.store.select(getActiveRunId), this.store.select(getLoadedExecutionData), this.store.select(getExecutionScrollBeginIndex)), map((/**
             * @param {?} __0
             * @return {?}
             */
            ([props, activeRunId, loadedExecutionData, scrollBeginIndex]) => {
                /** @type {?} */
                const focusIndex = scrollBeginIndex + props.displayIndex;
                return {
                    activeRunId: (/** @type {?} */ (activeRunId)),
                    loadedExecutionData,
                    focusIndex,
                };
            })));
        }
        /**
         * Load detailed data about execution and the associated stack frames.
         * @private
         * @param {?} prevStream$
         * @return {?}
         */
        createExecutionDataAndStackFramesLoader(prevStream$) {
            return prevStream$.pipe(filter((/**
             * @param {?} __0
             * @return {?}
             */
            ({ activeRunId, loadedExecutionData, focusIndex }) => {
                return (activeRunId !== null &&
                    focusIndex !== null &&
                    loadedExecutionData[(/** @type {?} */ (focusIndex))] === undefined);
            })), mergeMap((/**
             * @param {?} __0
             * @return {?}
             */
            ({ activeRunId, focusIndex }) => {
                /** @type {?} */
                const begin = (/** @type {?} */ (focusIndex));
                /** @type {?} */
                const end = begin + 1;
                return this.dataSource
                    .fetchExecutionData((/** @type {?} */ (activeRunId)), begin, end)
                    .pipe(tap((/**
                 * @param {?} executionDataResponse
                 * @return {?}
                 */
                (executionDataResponse) => {
                    this.store.dispatch(executionDataLoaded(executionDataResponse));
                })), map((/**
                 * @param {?} executionDataResponse
                 * @return {?}
                 */
                (executionDataResponse) => {
                    return { executionData: executionDataResponse, begin, end };
                })));
                // TODO(cais): Add catchError() to pipe.
            })), map((/**
             * @param {?} __0
             * @return {?}
             */
            ({ executionData }) => {
                return executionData.executions[0];
            })), withLatestFrom(this.store.select(getActiveRunId), this.store.select(getLoadedStackFrames)), filter((/**
             * @param {?} __0
             * @return {?}
             */
            ([execution, runId, loadedStackFrames]) => {
                if (runId === null) {
                    return false;
                }
                for (const stackFrameId of execution.stack_frame_ids) {
                    if (loadedStackFrames[stackFrameId] === undefined) {
                        return true;
                    }
                }
                return false;
            })), mergeMap((/**
             * @param {?} __0
             * @return {?}
             */
            ([execution, runId]) => {
                /** @type {?} */
                const stackFrameIds = execution.stack_frame_ids;
                // TODO(cais): Maybe omit already-loaded stack frames from request,
                // instead of loading all frames if any of them is missing.
                return this.dataSource.fetchStackFrames((/** @type {?} */ (runId)), stackFrameIds).pipe(tap((/**
                 * @param {?} stackFramesResponse
                 * @return {?}
                 */
                (stackFramesResponse) => {
                    /** @type {?} */
                    const stackFramesById = {};
                    for (let i = 0; i < stackFrameIds.length; ++i) {
                        stackFramesById[stackFrameIds[i]] =
                            stackFramesResponse.stack_frames[i];
                    }
                    this.store.dispatch(stackFramesLoaded({ stackFrames: stackFramesById }));
                })), map((/**
                 * @return {?}
                 */
                () => void null)));
                // TODO(cais): Add catchError() to pipe.
            })));
        }
        /**
         * Emits when scrolling event leads to need to load new intra-graph execution
         * data.
         *
         * The returned observable contains the
         *   - runId: active runId,
         *   - missingPage: indices of missing `GraphExecution` pages that need to be
         *     loaded by a downstream pipe.
         *   - pageSize: GraphExecution data page size.
         *   - numGraphExecutions: Current total number of `GraphExecution`s.
         * @private
         * @return {?}
         */
        onGraphExecutionScroll() {
            return this.actions$.pipe(ofType(graphExecutionScrollToIndex), debounceTime(100), withLatestFrom(this.store.select(getActiveRunId), this.store.select(getNumGraphExecutions), this.store.select(getGraphExecutionScrollBeginIndex)), filter((/**
             * @param {?} __0
             * @return {?}
             */
            ([, runId, numGraphExecutions]) => {
                return runId !== null && numGraphExecutions > 0;
            })), map((/**
             * @param {?} __0
             * @return {?}
             */
            ([, runId, numGraphExecutions, scrollBeginIndex]) => ({
                runId,
                numGraphExecutions,
                scrollBeginIndex,
            }))), withLatestFrom(this.store.select(getGraphExecutionPageSize), this.store.select(getGraphExecutionDisplayCount), this.store.select(getGraphExecutionDataLoadingPages), this.store.select(getGraphExecutionDataPageLoadedSizes)), map((/**
             * @param {?} __0
             * @return {?}
             */
            ([{ runId, numGraphExecutions, scrollBeginIndex }, pageSize, displayCount, loadingPages, pageLoadedSizes,]) => {
                /** @type {?} */
                let missingPages = getMissingPages(scrollBeginIndex, Math.min(scrollBeginIndex + displayCount, numGraphExecutions), pageSize, numGraphExecutions, pageLoadedSizes);
                // Omit pages that are already loading.
                missingPages = missingPages.filter((/**
                 * @param {?} page
                 * @return {?}
                 */
                (page) => loadingPages.indexOf(page) === -1));
                return {
                    runId: (/** @type {?} */ (runId)),
                    missingPages,
                    pageSize,
                    numGraphExecutions,
                };
            })));
        }
        /**
         * @private
         * @param {?} prevStream$
         * @return {?}
         */
        loadGraphExecutionPages(prevStream$) {
            return prevStream$.pipe(filter((/**
             * @param {?} __0
             * @return {?}
             */
            ({ missingPages }) => missingPages.length > 0)), tap((/**
             * @param {?} __0
             * @return {?}
             */
            ({ missingPages }) => {
                missingPages.forEach((/**
                 * @param {?} pageIndex
                 * @return {?}
                 */
                (pageIndex) => {
                    this.store.dispatch(graphExecutionDataRequested({ pageIndex }));
                }));
            })), mergeMap((/**
             * @param {?} __0
             * @return {?}
             */
            ({ runId, missingPages, pageSize, numGraphExecutions }) => {
                /** @type {?} */
                const begin = missingPages[0] * pageSize;
                /** @type {?} */
                const end = Math.min((missingPages[missingPages.length - 1] + 1) * pageSize, numGraphExecutions);
                return this.dataSource.fetchGraphExecutionData((/** @type {?} */ (runId)), begin, end).pipe(tap((/**
                 * @param {?} graphExecutionDataResponse
                 * @return {?}
                 */
                (graphExecutionDataResponse) => {
                    this.store.dispatch(graphExecutionDataLoaded(graphExecutionDataResponse));
                })), map((/**
                 * @return {?}
                 */
                () => void null)));
                // TODO(cais): Add catchError() to pipe.
            })));
        }
        /**
         * Listens to graph-op focus events.
         *
         * Load graph op info from the /graphs/op_info route when necessary.
         *
         * @private
         * @return {?} An object with two keys for downstream consumption:
         *   - runId: the current active run ID.
         *   - stackFrameIds: the IDs of the stack frames of the op's creation.
         */
        loadGraphOpInfo() {
            return this.actions$.pipe(ofType(graphOpFocused, graphExecutionFocused), withLatestFrom(this.store.select(getActiveRunId), this.store.select(getLoadingGraphOps)), filter((/**
             * @param {?} __0
             * @return {?}
             */
            ([actionData, runId, loadingOps]) => {
                const { graph_id, op_name } = actionData;
                return (runId !== null &&
                    (loadingOps[graph_id] === undefined ||
                        !loadingOps[graph_id].has(op_name) ||
                        !(loadingOps[graph_id].get(op_name) === DataLoadState.LOADING ||
                            loadingOps[graph_id].get(op_name) === DataLoadState.LOADED)));
            })), tap((/**
             * @param {?} __0
             * @return {?}
             */
            ([{ graph_id, op_name }]) => this.store.dispatch(graphOpInfoRequested({
                graph_id,
                op_name,
            })))), mergeMap((/**
             * @param {?} __0
             * @return {?}
             */
            ([actionData, runId]) => {
                const { graph_id, op_name } = actionData;
                return this.dataSource.fetchGraphOpInfo((/** @type {?} */ (runId)), graph_id, op_name).pipe(tap((/**
                 * @param {?} graphOpInfoResponse
                 * @return {?}
                 */
                (graphOpInfoResponse) => this.store.dispatch(graphOpInfoLoaded({ graphOpInfoResponse })))), map((/**
                 * @param {?} graphOpInfoResponse
                 * @return {?}
                 */
                (graphOpInfoResponse) => {
                    return {
                        runId: (/** @type {?} */ (runId)),
                        stackFrameIds: graphOpInfoResponse.stack_frame_ids,
                    };
                })));
                // TODO(cais): Add catchError() to pipe.
            })));
        }
        /**
         * @private
         * @param {?} prevStream$
         * @return {?}
         */
        loadGraphOpStackFrames(prevStream$) {
            return prevStream$.pipe(withLatestFrom(this.store.select(getLoadedStackFrames)), map((/**
             * @param {?} __0
             * @return {?}
             */
            ([{ runId, stackFrameIds }, loadedStackFrames]) => {
                /** @type {?} */
                const missingStackFrameIds = stackFrameIds.filter((/**
                 * @param {?} stackFrameId
                 * @return {?}
                 */
                (stackFrameId) => loadedStackFrames[stackFrameId] === undefined));
                return { runId, missingStackFrameIds };
            })), filter((/**
             * @param {?} __0
             * @return {?}
             */
            ({ runId, missingStackFrameIds }) => {
                return runId !== null && missingStackFrameIds.length > 0;
            })), mergeMap((/**
             * @param {?} __0
             * @return {?}
             */
            ({ runId, missingStackFrameIds }) => {
                return this.dataSource
                    .fetchStackFrames((/** @type {?} */ (runId)), missingStackFrameIds)
                    .pipe(tap((/**
                 * @param {?} stackFramesResponse
                 * @return {?}
                 */
                (stackFramesResponse) => {
                    /** @type {?} */
                    const stackFramesById = {};
                    // TODO(cais): Do this reshaping in the backend and simplify
                    // the frontend code here.
                    for (let i = 0; i < missingStackFrameIds.length; ++i) {
                        stackFramesById[missingStackFrameIds[i]] =
                            stackFramesResponse.stack_frames[i];
                    }
                    this.store.dispatch(stackFramesLoaded({ stackFrames: stackFramesById }));
                })), map((/**
                 * @return {?}
                 */
                () => void null)));
                // TODO(cais): Add catchError() to pipe.
            })));
        }
        /**
         * Emits when user focuses on an alert type.
         *
         * Returns an Observable for what additional execution digests need to be fetched.
         * @private
         * @return {?}
         */
        onAlertTypeFocused() {
            return this.actions$.pipe(ofType(alertTypeFocusToggled), withLatestFrom(this.store.select(getActiveRunId), this.store.select(getAlertsFocusType), this.store.select(getNumAlertsOfFocusedType), this.store.select(getLoadedAlertsOfFocusedType), this.store.select(getAlertsLoaded)), filter((/**
             * @param {?} __0
             * @return {?}
             */
            ([, runId, focusType, numAlertsOfFocusedType, loadedAlertsOfFocusedType, alertsLoaded,]) => {
                return (runId !== null &&
                    focusType !== null &&
                    numAlertsOfFocusedType > 0 &&
                    (loadedAlertsOfFocusedType === null ||
                        Object.keys(loadedAlertsOfFocusedType).length <
                            numAlertsOfFocusedType) &&
                    alertsLoaded.state !== DataLoadState.LOADING);
            })), tap((/**
             * @return {?}
             */
            () => this.store.dispatch(numAlertsAndBreakdownRequested()))), mergeMap((/**
             * @param {?} __0
             * @return {?}
             */
            ([, runId, focusType]) => {
                /** @type {?} */
                const begin = 0;
                // TODO(cais): Use smarter `end` value to reduce the amount of data
                // fetch each time.
                /** @type {?} */
                const end = -1;
                return this.dataSource.fetchAlerts((/** @type {?} */ (runId)), begin, end, (/** @type {?} */ (focusType)));
            })), tap((/**
             * @param {?} __0
             * @return {?}
             */
            ({ num_alerts, alerts_breakdown, alert_type, begin, end, alerts }) => {
                this.store.dispatch(alertsOfTypeLoaded({
                    numAlerts: num_alerts,
                    alertsBreakdown: alerts_breakdown,
                    alertType: (/** @type {?} */ (alert_type)),
                    begin,
                    end,
                    alerts,
                }));
            })));
        }
        /**
         * Compute the execution digests to fetch on focusing of an alert type.
         * @private
         * @param {?} prevStream$
         * @return {?}
         */
        fetchExecutionDigestsForAlertTypeFocus(prevStream$) {
            return prevStream$.pipe(withLatestFrom(this.store.select(getExecutionPageSize), this.store.select(getDisplayCount), this.store.select(getNumExecutions), this.store.select(getExecutionDigestsLoaded), this.store.select(getActiveRunId)), map((/**
             * @param {?} __0
             * @return {?}
             */
            ([alertsResponse, pageSize, displayCount, numExecutions, executionDigestsLoaded, runId,]) => {
                /** @type {?} */
                const alert = (/** @type {?} */ (alertsResponse.alerts[0]));
                /** @type {?} */
                const executionIndex = alert.execution_index;
                /** @type {?} */
                const missingPages = getMissingPages(Math.max(0, executionIndex - Math.floor(displayCount / 2)), Math.min(executionIndex + Math.floor(displayCount / 2), numExecutions), pageSize, numExecutions, executionDigestsLoaded.pageLoadedSizes);
                if (missingPages.length === 0) {
                    return { runId: (/** @type {?} */ (runId)), begin: 0, end: 0 };
                }
                else {
                    /** @type {?} */
                    const begin = missingPages[0] * pageSize;
                    /** @type {?} */
                    const end = Math.min(executionDigestsLoaded.numExecutions, (missingPages[missingPages.length - 1] + 1) * pageSize);
                    return { runId: (/** @type {?} */ (runId)), begin, end };
                }
            })));
        }
        /**
         * Load list of source files when debugger plugin is loaded.
         * @private
         * @param {?} prevStream$
         * @return {?}
         */
        loadSourceFileList(prevStream$) {
            return prevStream$.pipe(withLatestFrom(this.store.select(getActiveRunId), this.store.select(getSourceFileListLoaded)), filter((/**
             * @param {?} __0
             * @return {?}
             */
            ([, runId, sourceFileListLoadState]) => {
                return (runId !== null &&
                    sourceFileListLoadState.state !== DataLoadState.LOADING);
            })), tap((/**
             * @return {?}
             */
            () => this.store.dispatch(sourceFileListRequested()))), mergeMap((/**
             * @param {?} __0
             * @return {?}
             */
            ([, runId]) => {
                return this.dataSource.fetchSourceFileList((/** @type {?} */ (runId))).pipe(tap((/**
                 * @param {?} sourceFileListResponse
                 * @return {?}
                 */
                (sourceFileListResponse) => {
                    /** @type {?} */
                    const sourceFiles = [];
                    sourceFileListResponse.forEach((/**
                     * @param {?} __0
                     * @return {?}
                     */
                    ([host_name, file_path]) => {
                        sourceFiles.push({ host_name, file_path });
                    }));
                    this.store.dispatch(sourceFileListLoaded({ sourceFiles }));
                })), map((/**
                 * @return {?}
                 */
                () => void null))
                // TODO(cais): Add catchError() to pipe.
                );
            })));
        }
        /**
         * When the a source file is focused on, load its content from the data source.
         * @private
         * @return {?}
         */
        onSourceFileFocused() {
            return this.actions$.pipe(ofType(sourceLineFocused), withLatestFrom(this.store.select(getActiveRunId), this.store.select(getFocusedSourceFileIndex), this.store.select(getFocusedSourceFileContent)), map((/**
             * @param {?} __0
             * @return {?}
             */
            ([focus, runId, fileIndex, fileContent]) => {
                return {
                    runId,
                    stackFrame: focus.stackFrame,
                    fileIndex,
                    fileContent,
                };
            })), filter((/**
             * @param {?} __0
             * @return {?}
             */
            ({ runId, fileContent }) => {
                return (runId !== null &&
                    fileContent !== null &&
                    fileContent.loadState === DataLoadState.NOT_LOADED);
            })), tap((/**
             * @param {?} __0
             * @return {?}
             */
            ({ stackFrame }) => this.store.dispatch(sourceFileRequested({
                host_name: stackFrame.host_name,
                file_path: stackFrame.file_path,
            })))), mergeMap((/**
             * @param {?} __0
             * @return {?}
             */
            ({ fileIndex, runId }) => {
                return this.dataSource.fetchSourceFile((/** @type {?} */ (runId)), fileIndex).pipe(tap((/**
                 * @param {?} sourceFileResponse
                 * @return {?}
                 */
                (sourceFileResponse) => {
                    this.store.dispatch(sourceFileLoaded(sourceFileResponse));
                })), map((/**
                 * @return {?}
                 */
                () => void null))
                // TODO(cais): Add catchError() to pipe.
                );
            })));
        }
    }
    DebuggerEffects.decorators = [
        { type: Injectable },
    ];
    /** @nocollapse */
    DebuggerEffects.ctorParameters = () => [
        { type: Actions },
        { type: Store },
        { type: Tfdbg2HttpServerDataSource }
    ];
    /** @nocollapse */ DebuggerEffects.ɵfac = function DebuggerEffects_Factory(t) { return new (t || DebuggerEffects)(ɵɵinject(Actions), ɵɵinject(Store), ɵɵinject(Tfdbg2HttpServerDataSource)); };
    /** @nocollapse */ DebuggerEffects.ɵprov = ɵɵdefineInjectable({ token: DebuggerEffects, factory: DebuggerEffects.ɵfac });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/views/alerts/alerts_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /* Copyright 2019 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    class AlertsModule {
    }
    AlertsModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [AlertsComponent, AlertsContainer],
                    imports: [CommonModule],
                    exports: [AlertsContainer],
                },] },
    ];
    /** @nocollapse */ AlertsModule.ɵmod = ɵɵdefineNgModule({ type: AlertsModule });
    /** @nocollapse */ AlertsModule.ɵinj = ɵɵdefineInjector({ factory: function AlertsModule_Factory(t) { return new (t || AlertsModule)(); }, imports: [[CommonModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(AlertsModule, { declarations: [AlertsComponent, AlertsContainer], imports: [CommonModule], exports: [AlertsContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/views/debug_tensor_value/debug_tensor_value_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    class DebugTensorValueModule {
    }
    DebugTensorValueModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [
                        DebugTensorDTypeComponent,
                        DebugTensorHasInfOrNaNComponent,
                        DebugTensorNumericBreakdownComponent,
                        DebugTensorRankComponent,
                        DebugTensorShapeComponent,
                        DebugTensorValueComponent,
                    ],
                    imports: [CommonModule],
                    exports: [DebugTensorValueComponent],
                },] },
    ];
    /** @nocollapse */ DebugTensorValueModule.ɵmod = ɵɵdefineNgModule({ type: DebugTensorValueModule });
    /** @nocollapse */ DebugTensorValueModule.ɵinj = ɵɵdefineInjector({ factory: function DebugTensorValueModule_Factory(t) { return new (t || DebugTensorValueModule)(); }, imports: [[CommonModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(DebugTensorValueModule, { declarations: [DebugTensorDTypeComponent,
            DebugTensorHasInfOrNaNComponent,
            DebugTensorNumericBreakdownComponent,
            DebugTensorRankComponent,
            DebugTensorShapeComponent,
            DebugTensorValueComponent], imports: [CommonModule], exports: [DebugTensorValueComponent] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/views/graph_executions/graph_executions_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    class GraphExecutionsModule {
    }
    GraphExecutionsModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [GraphExecutionsComponent, GraphExecutionsContainer],
                    imports: [CommonModule, DebugTensorValueModule, ScrollingModule],
                    exports: [GraphExecutionsContainer],
                },] },
    ];
    /** @nocollapse */ GraphExecutionsModule.ɵmod = ɵɵdefineNgModule({ type: GraphExecutionsModule });
    /** @nocollapse */ GraphExecutionsModule.ɵinj = ɵɵdefineInjector({ factory: function GraphExecutionsModule_Factory(t) { return new (t || GraphExecutionsModule)(); }, imports: [[CommonModule, DebugTensorValueModule, ScrollingModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(GraphExecutionsModule, { declarations: [GraphExecutionsComponent, GraphExecutionsContainer], imports: [CommonModule, DebugTensorValueModule, ScrollingModule], exports: [GraphExecutionsContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/views/graph/graph_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    class GraphModule {
    }
    GraphModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [GraphComponent, GraphOpComponent, GraphContainer],
                    imports: [CommonModule],
                    exports: [GraphContainer],
                },] },
    ];
    /** @nocollapse */ GraphModule.ɵmod = ɵɵdefineNgModule({ type: GraphModule });
    /** @nocollapse */ GraphModule.ɵinj = ɵɵdefineInjector({ factory: function GraphModule_Factory(t) { return new (t || GraphModule)(); }, imports: [[CommonModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(GraphModule, { declarations: [GraphComponent, GraphOpComponent, GraphContainer], imports: [CommonModule], exports: [GraphContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/views/inactive/inactive_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /* Copyright 2019 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    class InactiveModule {
    }
    InactiveModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [InactiveComponent, InactiveContainer],
                    exports: [InactiveContainer],
                },] },
    ];
    /** @nocollapse */ InactiveModule.ɵmod = ɵɵdefineNgModule({ type: InactiveModule });
    /** @nocollapse */ InactiveModule.ɵinj = ɵɵdefineInjector({ factory: function InactiveModule_Factory(t) { return new (t || InactiveModule)(); } });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(InactiveModule, { declarations: [InactiveComponent, InactiveContainer], exports: [InactiveContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/views/source_code/source_code_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    class SourceCodeModule {
    }
    SourceCodeModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [SourceCodeComponent, SourceCodeContainer],
                    imports: [CommonModule],
                    exports: [SourceCodeContainer],
                },] },
    ];
    /** @nocollapse */ SourceCodeModule.ɵmod = ɵɵdefineNgModule({ type: SourceCodeModule });
    /** @nocollapse */ SourceCodeModule.ɵinj = ɵɵdefineInjector({ factory: function SourceCodeModule_Factory(t) { return new (t || SourceCodeModule)(); }, imports: [[CommonModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(SourceCodeModule, { declarations: [SourceCodeComponent, SourceCodeContainer], imports: [CommonModule], exports: [SourceCodeContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/views/source_files/source_files_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    class SourceFilesModule {
    }
    SourceFilesModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [SourceFilesComponent, SourceFilesContainer],
                    imports: [CommonModule, SourceCodeModule],
                    exports: [SourceFilesContainer],
                },] },
    ];
    /** @nocollapse */ SourceFilesModule.ɵmod = ɵɵdefineNgModule({ type: SourceFilesModule });
    /** @nocollapse */ SourceFilesModule.ɵinj = ɵɵdefineInjector({ factory: function SourceFilesModule_Factory(t) { return new (t || SourceFilesModule)(); }, imports: [[CommonModule, SourceCodeModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(SourceFilesModule, { declarations: [SourceFilesComponent, SourceFilesContainer], imports: [CommonModule, SourceCodeModule], exports: [SourceFilesContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/views/stack_trace/stack_trace_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    class StackTraceModule {
    }
    StackTraceModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [StackTraceComponent, StackTraceContainer],
                    imports: [CommonModule, SourceCodeModule],
                    exports: [StackTraceContainer],
                },] },
    ];
    /** @nocollapse */ StackTraceModule.ɵmod = ɵɵdefineNgModule({ type: StackTraceModule });
    /** @nocollapse */ StackTraceModule.ɵinj = ɵɵdefineInjector({ factory: function StackTraceModule_Factory(t) { return new (t || StackTraceModule)(); }, imports: [[CommonModule, SourceCodeModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(StackTraceModule, { declarations: [StackTraceComponent, StackTraceContainer], imports: [CommonModule, SourceCodeModule], exports: [StackTraceContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/views/execution_data/execution_data_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    class ExecutionDataModule {
    }
    ExecutionDataModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [ExecutionDataComponent, ExecutionDataContainer],
                    imports: [CommonModule, DebugTensorValueModule],
                    exports: [ExecutionDataContainer],
                },] },
    ];
    /** @nocollapse */ ExecutionDataModule.ɵmod = ɵɵdefineNgModule({ type: ExecutionDataModule });
    /** @nocollapse */ ExecutionDataModule.ɵinj = ɵɵdefineInjector({ factory: function ExecutionDataModule_Factory(t) { return new (t || ExecutionDataModule)(); }, imports: [[CommonModule, DebugTensorValueModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(ExecutionDataModule, { declarations: [ExecutionDataComponent, ExecutionDataContainer], imports: [CommonModule, DebugTensorValueModule], exports: [ExecutionDataContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/views/timeline/timeline_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    class TimelineModule {
    }
    TimelineModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [TimelineComponent, TimelineContainer],
                    imports: [
                        CommonModule,
                        ExecutionDataModule,
                        MatButtonModule,
                        MatSliderModule,
                    ],
                    // TODO(cais): The following two providers are meant to make the mat-slider
                    // dragging to work. But the dragging is still not working, needs more
                    // investigation. May be related to:
                    // https://github.com/angular/components/issues/7905
                    exports: [TimelineContainer],
                },] },
    ];
    /** @nocollapse */ TimelineModule.ɵmod = ɵɵdefineNgModule({ type: TimelineModule });
    /** @nocollapse */ TimelineModule.ɵinj = ɵɵdefineInjector({ factory: function TimelineModule_Factory(t) { return new (t || TimelineModule)(); }, imports: [[
                CommonModule,
                ExecutionDataModule,
                MatButtonModule,
                MatSliderModule,
            ]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(TimelineModule, { declarations: [TimelineComponent, TimelineContainer], imports: [CommonModule,
            ExecutionDataModule,
            MatButtonModule,
            MatSliderModule], exports: [TimelineContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/plugins/debugger_v2/tf_debugger_v2_plugin/debugger_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /* Copyright 2019 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    class DebuggerModule {
    }
    DebuggerModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [DebuggerComponent, DebuggerContainer],
                    imports: [
                        AlertsModule,
                        CommonModule,
                        GraphExecutionsModule,
                        GraphModule,
                        InactiveModule,
                        SourceFilesModule,
                        StackTraceModule,
                        Tfdbg2ServerDataSourceModule,
                        TimelineModule,
                        StoreModule.forFeature(DEBUGGER_FEATURE_KEY, reducers$6),
                        EffectsModule.forFeature([DebuggerEffects]),
                        PluginRegistryModule.forPlugin('debugger-v2', DebuggerContainer),
                    ],
                    exports: [DebuggerContainer],
                    entryComponents: [DebuggerContainer],
                },] },
    ];
    /** @nocollapse */ DebuggerModule.ɵmod = ɵɵdefineNgModule({ type: DebuggerModule });
    /** @nocollapse */ DebuggerModule.ɵinj = ɵɵdefineInjector({ factory: function DebuggerModule_Factory(t) { return new (t || DebuggerModule)(); }, imports: [[
                AlertsModule,
                CommonModule,
                GraphExecutionsModule,
                GraphModule,
                InactiveModule,
                SourceFilesModule,
                StackTraceModule,
                Tfdbg2ServerDataSourceModule,
                TimelineModule,
                StoreModule.forFeature(DEBUGGER_FEATURE_KEY, reducers$6),
                EffectsModule.forFeature([DebuggerEffects]),
                PluginRegistryModule.forPlugin('debugger-v2', DebuggerContainer),
            ]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(DebuggerModule, { declarations: [DebuggerComponent, DebuggerContainer], imports: [AlertsModule,
            CommonModule,
            GraphExecutionsModule,
            GraphModule,
            InactiveModule,
            SourceFilesModule,
            StackTraceModule,
            Tfdbg2ServerDataSourceModule,
            TimelineModule, StoreFeatureModule, EffectsFeatureModule, PluginRegistryModule], exports: [DebuggerContainer] }); })();

    const _c0$k = ["mat-menu-item", ""];
    const _c1$b = ["*"];
    function _MatMenu_ng_template_0_Template(rf, ctx) { if (rf & 1) {
        const _r2 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "div", 0);
        ɵɵlistener("keydown", function _MatMenu_ng_template_0_Template_div_keydown_0_listener($event) { ɵɵrestoreView(_r2); const ctx_r1 = ɵɵnextContext(); return ctx_r1._handleKeydown($event); })("click", function _MatMenu_ng_template_0_Template_div_click_0_listener() { ɵɵrestoreView(_r2); const ctx_r3 = ɵɵnextContext(); return ctx_r3.closed.emit("click"); })("@transformMenu.start", function _MatMenu_ng_template_0_Template_div_animation_transformMenu_start_0_listener($event) { ɵɵrestoreView(_r2); const ctx_r4 = ɵɵnextContext(); return ctx_r4._onAnimationStart($event); })("@transformMenu.done", function _MatMenu_ng_template_0_Template_div_animation_transformMenu_done_0_listener($event) { ɵɵrestoreView(_r2); const ctx_r5 = ɵɵnextContext(); return ctx_r5._onAnimationDone($event); });
        ɵɵelementStart(1, "div", 1);
        ɵɵprojection(2);
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r0 = ɵɵnextContext();
        ɵɵproperty("id", ctx_r0.panelId)("ngClass", ctx_r0._classList)("@transformMenu", ctx_r0._panelAnimationState);
        ɵɵattribute("aria-label", ctx_r0.ariaLabel || null)("aria-labelledby", ctx_r0.ariaLabelledby || null)("aria-describedby", ctx_r0.ariaDescribedby || null);
    } }
    const matMenuAnimations = {
        /**
         * This animation controls the menu panel's entry and exit from the page.
         *
         * When the menu panel is added to the DOM, it scales in and fades in its border.
         *
         * When the menu panel is removed from the DOM, it simply fades out after a brief
         * delay to display the ripple.
         */
        transformMenu: trigger('transformMenu', [
            state('void', style({
                opacity: 0,
                transform: 'scale(0.8)'
            })),
            transition('void => enter', group([
                query('.mat-menu-content, .mat-mdc-menu-content', animate('100ms linear', style({
                    opacity: 1
                }))),
                animate('120ms cubic-bezier(0, 0, 0.2, 1)', style({ transform: 'scale(1)' })),
            ])),
            transition('* => void', animate('100ms 25ms linear', style({ opacity: 0 })))
        ]),
        /**
         * This animation fades in the background color and content of the menu panel
         * after its containing element is scaled in.
         */
        fadeInItems: trigger('fadeInItems', [
            // TODO(crisbeto): this is inside the `transformMenu`
            // now. Remove next time we do breaking changes.
            state('showing', style({ opacity: 1 })),
            transition('void => *', [
                style({ opacity: 0 }),
                animate('400ms 100ms cubic-bezier(0.55, 0, 0.55, 0.2)')
            ])
        ])
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/menu/menu-content.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Menu content that will be rendered lazily once the menu is opened.
     */
    class MatMenuContent {
        /**
         * @param {?} _template
         * @param {?} _componentFactoryResolver
         * @param {?} _appRef
         * @param {?} _injector
         * @param {?} _viewContainerRef
         * @param {?} _document
         * @param {?=} _changeDetectorRef
         */
        constructor(_template, _componentFactoryResolver, _appRef, _injector, _viewContainerRef, _document, _changeDetectorRef) {
            this._template = _template;
            this._componentFactoryResolver = _componentFactoryResolver;
            this._appRef = _appRef;
            this._injector = _injector;
            this._viewContainerRef = _viewContainerRef;
            this._document = _document;
            this._changeDetectorRef = _changeDetectorRef;
            /**
             * Emits when the menu content has been attached.
             */
            this._attached = new Subject();
        }
        /**
         * Attaches the content with a particular context.
         * \@docs-private
         * @param {?=} context
         * @return {?}
         */
        attach(context = {}) {
            if (!this._portal) {
                this._portal = new TemplatePortal(this._template, this._viewContainerRef);
            }
            this.detach();
            if (!this._outlet) {
                this._outlet = new DomPortalOutlet(this._document.createElement('div'), this._componentFactoryResolver, this._appRef, this._injector);
            }
            /** @type {?} */
            const element = this._template.elementRef.nativeElement;
            // Because we support opening the same menu from different triggers (which in turn have their
            // own `OverlayRef` panel), we have to re-insert the host element every time, otherwise we
            // risk it staying attached to a pane that's no longer in the DOM.
            (/** @type {?} */ (element.parentNode)).insertBefore(this._outlet.outletElement, element);
            // When `MatMenuContent` is used in an `OnPush` component, the insertion of the menu
            // content via `createEmbeddedView` does not cause the content to be seen as "dirty"
            // by Angular. This causes the `@ContentChildren` for menu items within the menu to
            // not be updated by Angular. By explicitly marking for check here, we tell Angular that
            // it needs to check for new menu items and update the `@ContentChild` in `MatMenu`.
            // @breaking-change 9.0.0 Make change detector ref required
            if (this._changeDetectorRef) {
                this._changeDetectorRef.markForCheck();
            }
            this._portal.attach(this._outlet, context);
            this._attached.next();
        }
        /**
         * Detaches the content.
         * \@docs-private
         * @return {?}
         */
        detach() {
            if (this._portal.isAttached) {
                this._portal.detach();
            }
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            if (this._outlet) {
                this._outlet.dispose();
            }
        }
    }
    MatMenuContent.ɵfac = function MatMenuContent_Factory(t) { return new (t || MatMenuContent)(ɵɵdirectiveInject(TemplateRef), ɵɵdirectiveInject(ComponentFactoryResolver), ɵɵdirectiveInject(ApplicationRef), ɵɵdirectiveInject(Injector), ɵɵdirectiveInject(ViewContainerRef), ɵɵdirectiveInject(DOCUMENT$1), ɵɵdirectiveInject(ChangeDetectorRef)); };
    MatMenuContent.ɵdir = ɵɵdefineDirective({ type: MatMenuContent, selectors: [["ng-template", "matMenuContent", ""]] });
    /** @nocollapse */
    MatMenuContent.ctorParameters = () => [
        { type: TemplateRef },
        { type: ComponentFactoryResolver },
        { type: ApplicationRef },
        { type: Injector },
        { type: ViewContainerRef },
        { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT$1,] }] },
        { type: ChangeDetectorRef }
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/menu/menu-errors.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google LLC All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * Throws an exception for the case when menu trigger doesn't have a valid mat-menu instance
     * \@docs-private
     * @return {?}
     */
    function throwMatMenuMissingError() {
        throw Error(`matMenuTriggerFor: must pass in an mat-menu instance.

    Example:
      <mat-menu #menu="matMenu"></mat-menu>
      <button [matMenuTriggerFor]="menu"></button>`);
    }
    /**
     * Throws an exception for the case when menu's x-position value isn't valid.
     * In other words, it doesn't match 'before' or 'after'.
     * \@docs-private
     * @return {?}
     */
    function throwMatMenuInvalidPositionX() {
        throw Error(`xPosition value must be either 'before' or after'.
      Example: <mat-menu xPosition="before" #menu="matMenu"></mat-menu>`);
    }
    /**
     * Throws an exception for the case when menu's y-position value isn't valid.
     * In other words, it doesn't match 'above' or 'below'.
     * \@docs-private
     * @return {?}
     */
    function throwMatMenuInvalidPositionY() {
        throw Error(`yPosition value must be either 'above' or below'.
      Example: <mat-menu yPosition="above" #menu="matMenu"></mat-menu>`);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/menu/menu-panel.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Injection token used to provide the parent menu to menu-specific components.
     * \@docs-private
     * @type {?}
     */
    const MAT_MENU_PANEL = new InjectionToken('MAT_MENU_PANEL');

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/menu/menu-item.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    // Boilerplate for applying mixins to MatMenuItem.
    /**
     * \@docs-private
     */
    class MatMenuItemBase {
    }
    /** @type {?} */
    const _MatMenuItemMixinBase = mixinDisableRipple(mixinDisabled(MatMenuItemBase));
    /**
     * Single item inside of a `mat-menu`. Provides the menu item styling and accessibility treatment.
     */
    class MatMenuItem extends _MatMenuItemMixinBase {
        /**
         * @param {?} _elementRef
         * @param {?=} document
         * @param {?=} _focusMonitor
         * @param {?=} _parentMenu
         */
        constructor(_elementRef, document, _focusMonitor, _parentMenu) {
            // @breaking-change 8.0.0 make `_focusMonitor` and `document` required params.
            super();
            this._elementRef = _elementRef;
            this._focusMonitor = _focusMonitor;
            this._parentMenu = _parentMenu;
            /**
             * ARIA role for the menu item.
             */
            this.role = 'menuitem';
            /**
             * Stream that emits when the menu item is hovered.
             */
            this._hovered = new Subject();
            /**
             * Stream that emits when the menu item is focused.
             */
            this._focused = new Subject();
            /**
             * Whether the menu item is highlighted.
             */
            this._highlighted = false;
            /**
             * Whether the menu item acts as a trigger for a sub-menu.
             */
            this._triggersSubmenu = false;
            if (_focusMonitor) {
                // Start monitoring the element so it gets the appropriate focused classes. We want
                // to show the focus style for menu items only when the focus was not caused by a
                // mouse or touch interaction.
                _focusMonitor.monitor(this._elementRef, false);
            }
            if (_parentMenu && _parentMenu.addItem) {
                _parentMenu.addItem(this);
            }
            this._document = document;
        }
        /**
         * Focuses the menu item.
         * @param {?=} origin
         * @param {?=} options
         * @return {?}
         */
        focus(origin = 'program', options) {
            if (this._focusMonitor) {
                this._focusMonitor.focusVia(this._getHostElement(), origin, options);
            }
            else {
                this._getHostElement().focus(options);
            }
            this._focused.next(this);
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            if (this._focusMonitor) {
                this._focusMonitor.stopMonitoring(this._elementRef);
            }
            if (this._parentMenu && this._parentMenu.removeItem) {
                this._parentMenu.removeItem(this);
            }
            this._hovered.complete();
            this._focused.complete();
        }
        /**
         * Used to set the `tabindex`.
         * @return {?}
         */
        _getTabIndex() {
            return this.disabled ? '-1' : '0';
        }
        /**
         * Returns the host DOM element.
         * @return {?}
         */
        _getHostElement() {
            return this._elementRef.nativeElement;
        }
        /**
         * Prevents the default element actions if it is disabled.
         * @param {?} event
         * @return {?}
         */
        // We have to use a `HostListener` here in order to support both Ivy and ViewEngine.
        // In Ivy the `host` bindings will be merged when this class is extended, whereas in
        // ViewEngine they're overwritten.
        // TODO(crisbeto): we move this back into `host` once Ivy is turned on by default.
        // tslint:disable-next-line:no-host-decorator-in-concrete
        _checkDisabled(event) {
            if (this.disabled) {
                event.preventDefault();
                event.stopPropagation();
            }
        }
        /**
         * Emits to the hover stream.
         * @return {?}
         */
        // We have to use a `HostListener` here in order to support both Ivy and ViewEngine.
        // In Ivy the `host` bindings will be merged when this class is extended, whereas in
        // ViewEngine they're overwritten.
        // TODO(crisbeto): we move this back into `host` once Ivy is turned on by default.
        // tslint:disable-next-line:no-host-decorator-in-concrete
        _handleMouseEnter() {
            this._hovered.next(this);
        }
        /**
         * Gets the label to be used when determining whether the option should be focused.
         * @return {?}
         */
        getLabel() {
            /** @type {?} */
            const element = this._elementRef.nativeElement;
            /** @type {?} */
            const textNodeType = this._document ? this._document.TEXT_NODE : 3;
            /** @type {?} */
            let output = '';
            if (element.childNodes) {
                /** @type {?} */
                const length = element.childNodes.length;
                // Go through all the top-level text nodes and extract their text.
                // We skip anything that's not a text node to prevent the text from
                // being thrown off by something like an icon.
                for (let i = 0; i < length; i++) {
                    if (element.childNodes[i].nodeType === textNodeType) {
                        output += element.childNodes[i].textContent;
                    }
                }
            }
            return output.trim();
        }
    }
    MatMenuItem.ɵfac = function MatMenuItem_Factory(t) { return new (t || MatMenuItem)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(DOCUMENT$1), ɵɵdirectiveInject(FocusMonitor), ɵɵdirectiveInject(MAT_MENU_PANEL, 8)); };
    MatMenuItem.ɵcmp = ɵɵdefineComponent({ type: MatMenuItem, selectors: [["", "mat-menu-item", ""]], hostAttrs: [1, "mat-focus-indicator"], hostVars: 10, hostBindings: function MatMenuItem_HostBindings(rf, ctx) { if (rf & 1) {
            ɵɵlistener("click", function MatMenuItem_click_HostBindingHandler($event) { return ctx._checkDisabled($event); })("mouseenter", function MatMenuItem_mouseenter_HostBindingHandler() { return ctx._handleMouseEnter(); });
        } if (rf & 2) {
            ɵɵattribute("role", ctx.role)("tabindex", ctx._getTabIndex())("aria-disabled", ctx.disabled.toString())("disabled", ctx.disabled || null);
            ɵɵclassProp("mat-menu-item", true)("mat-menu-item-highlighted", ctx._highlighted)("mat-menu-item-submenu-trigger", ctx._triggersSubmenu);
        } }, inputs: { disabled: "disabled", disableRipple: "disableRipple", role: "role" }, exportAs: ["matMenuItem"], features: [ɵɵInheritDefinitionFeature], attrs: _c0$k, ngContentSelectors: _c1$b, decls: 2, vars: 2, consts: [["matRipple", "", 1, "mat-menu-ripple", 3, "matRippleDisabled", "matRippleTrigger"]], template: function MatMenuItem_Template(rf, ctx) { if (rf & 1) {
            ɵɵprojectionDef();
            ɵɵprojection(0);
            ɵɵelement(1, "div", 0);
        } if (rf & 2) {
            ɵɵadvance(1);
            ɵɵproperty("matRippleDisabled", ctx.disableRipple || ctx.disabled)("matRippleTrigger", ctx._getHostElement());
        } }, directives: [MatRipple], encapsulation: 2, changeDetection: 0 });
    /** @nocollapse */
    MatMenuItem.ctorParameters = () => [
        { type: ElementRef },
        { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT$1,] }] },
        { type: FocusMonitor },
        { type: undefined, decorators: [{ type: Inject, args: [MAT_MENU_PANEL,] }, { type: Optional }] }
    ];
    MatMenuItem.propDecorators = {
        role: [{ type: Input }],
        _checkDisabled: [{ type: HostListener, args: ['click', ['$event'],] }],
        _handleMouseEnter: [{ type: HostListener, args: ['mouseenter',] }]
    };
    /**
     * Injection token to be used to override the default options for `mat-menu`.
     * @type {?}
     */
    const MAT_MENU_DEFAULT_OPTIONS = new InjectionToken('mat-menu-default-options', {
        providedIn: 'root',
        factory: MAT_MENU_DEFAULT_OPTIONS_FACTORY
    });
    /**
     * \@docs-private
     * @return {?}
     */
    function MAT_MENU_DEFAULT_OPTIONS_FACTORY() {
        return {
            overlapTrigger: false,
            xPosition: 'after',
            yPosition: 'below',
            backdropClass: 'cdk-overlay-transparent-backdrop',
        };
    }
    /**
     * Start elevation for the menu panel.
     * \@docs-private
     * @type {?}
     */
    const MAT_MENU_BASE_ELEVATION = 4;
    /** @type {?} */
    let menuPanelUid = 0;
    /**
     * Base class with all of the `MatMenu` functionality.
     */
    // tslint:disable-next-line:class-name
    class _MatMenuBase {
        /**
         * @param {?} _elementRef
         * @param {?} _ngZone
         * @param {?} _defaultOptions
         */
        constructor(_elementRef, _ngZone, _defaultOptions) {
            this._elementRef = _elementRef;
            this._ngZone = _ngZone;
            this._defaultOptions = _defaultOptions;
            this._xPosition = this._defaultOptions.xPosition;
            this._yPosition = this._defaultOptions.yPosition;
            /**
             * Only the direct descendant menu items.
             */
            this._directDescendantItems = new QueryList();
            /**
             * Subscription to tab events on the menu panel
             */
            this._tabSubscription = Subscription.EMPTY;
            /**
             * Config object to be passed into the menu's ngClass
             */
            this._classList = {};
            /**
             * Current state of the panel animation.
             */
            this._panelAnimationState = 'void';
            /**
             * Emits whenever an animation on the menu completes.
             */
            this._animationDone = new Subject();
            /**
             * Class to be added to the backdrop element.
             */
            this.backdropClass = this._defaultOptions.backdropClass;
            this._overlapTrigger = this._defaultOptions.overlapTrigger;
            this._hasBackdrop = this._defaultOptions.hasBackdrop;
            /**
             * Event emitted when the menu is closed.
             */
            this.closed = new EventEmitter();
            /**
             * Event emitted when the menu is closed.
             * @deprecated Switch to `closed` instead
             * \@breaking-change 8.0.0
             */
            this.close = this.closed;
            this.panelId = `mat-menu-panel-${menuPanelUid++}`;
        }
        /**
         * Position of the menu in the X axis.
         * @return {?}
         */
        get xPosition() { return this._xPosition; }
        /**
         * @param {?} value
         * @return {?}
         */
        set xPosition(value) {
            if (value !== 'before' && value !== 'after') {
                throwMatMenuInvalidPositionX();
            }
            this._xPosition = value;
            this.setPositionClasses();
        }
        /**
         * Position of the menu in the Y axis.
         * @return {?}
         */
        get yPosition() { return this._yPosition; }
        /**
         * @param {?} value
         * @return {?}
         */
        set yPosition(value) {
            if (value !== 'above' && value !== 'below') {
                throwMatMenuInvalidPositionY();
            }
            this._yPosition = value;
            this.setPositionClasses();
        }
        /**
         * Whether the menu should overlap its trigger.
         * @return {?}
         */
        get overlapTrigger() { return this._overlapTrigger; }
        /**
         * @param {?} value
         * @return {?}
         */
        set overlapTrigger(value) {
            this._overlapTrigger = coerceBooleanProperty(value);
        }
        /**
         * Whether the menu has a backdrop.
         * @return {?}
         */
        get hasBackdrop() { return this._hasBackdrop; }
        /**
         * @param {?} value
         * @return {?}
         */
        set hasBackdrop(value) {
            this._hasBackdrop = coerceBooleanProperty(value);
        }
        /**
         * This method takes classes set on the host mat-menu element and applies them on the
         * menu template that displays in the overlay container.  Otherwise, it's difficult
         * to style the containing menu from outside the component.
         * @param {?} classes list of class names
         * @return {?}
         */
        set panelClass(classes) {
            /** @type {?} */
            const previousPanelClass = this._previousPanelClass;
            if (previousPanelClass && previousPanelClass.length) {
                previousPanelClass.split(' ').forEach((/**
                 * @param {?} className
                 * @return {?}
                 */
                (className) => {
                    this._classList[className] = false;
                }));
            }
            this._previousPanelClass = classes;
            if (classes && classes.length) {
                classes.split(' ').forEach((/**
                 * @param {?} className
                 * @return {?}
                 */
                (className) => {
                    this._classList[className] = true;
                }));
                this._elementRef.nativeElement.className = '';
            }
        }
        /**
         * This method takes classes set on the host mat-menu element and applies them on the
         * menu template that displays in the overlay container.  Otherwise, it's difficult
         * to style the containing menu from outside the component.
         * @deprecated Use `panelClass` instead.
         * \@breaking-change 8.0.0
         * @return {?}
         */
        get classList() { return this.panelClass; }
        /**
         * @param {?} classes
         * @return {?}
         */
        set classList(classes) { this.panelClass = classes; }
        /**
         * @return {?}
         */
        ngOnInit() {
            this.setPositionClasses();
        }
        /**
         * @return {?}
         */
        ngAfterContentInit() {
            this._updateDirectDescendants();
            this._keyManager = new FocusKeyManager(this._directDescendantItems).withWrap().withTypeAhead();
            this._tabSubscription = this._keyManager.tabOut.subscribe((/**
             * @return {?}
             */
            () => this.closed.emit('tab')));
            // If a user manually (programatically) focuses a menu item, we need to reflect that focus
            // change back to the key manager. Note that we don't need to unsubscribe here because _focused
            // is internal and we know that it gets completed on destroy.
            this._directDescendantItems.changes.pipe(startWith(this._directDescendantItems), switchMap((/**
             * @param {?} items
             * @return {?}
             */
            items => merge(...items.map((/**
             * @param {?} item
             * @return {?}
             */
            (item) => item._focused)))))).subscribe((/**
             * @param {?} focusedItem
             * @return {?}
             */
            focusedItem => this._keyManager.updateActiveItem(focusedItem)));
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this._directDescendantItems.destroy();
            this._tabSubscription.unsubscribe();
            this.closed.complete();
        }
        /**
         * Stream that emits whenever the hovered menu item changes.
         * @return {?}
         */
        _hovered() {
            // Coerce the `changes` property because Angular types it as `Observable<any>`
            /** @type {?} */
            const itemChanges = (/** @type {?} */ (this._directDescendantItems.changes));
            return (/** @type {?} */ (itemChanges.pipe(startWith(this._directDescendantItems), switchMap((/**
             * @param {?} items
             * @return {?}
             */
            items => merge(...items.map((/**
             * @param {?} item
             * @return {?}
             */
            (item) => item._hovered))))))));
        }
        /*
           * Registers a menu item with the menu.
           * @docs-private
           * @deprecated No longer being used. To be removed.
           * @breaking-change 9.0.0
           */
        /**
         * @param {?} _item
         * @return {?}
         */
        addItem(_item) { }
        /**
         * Removes an item from the menu.
         * \@docs-private
         * @deprecated No longer being used. To be removed.
         * \@breaking-change 9.0.0
         * @param {?} _item
         * @return {?}
         */
        removeItem(_item) { }
        /**
         * Handle a keyboard event from the menu, delegating to the appropriate action.
         * @param {?} event
         * @return {?}
         */
        _handleKeydown(event) {
            /** @type {?} */
            const keyCode = event.keyCode;
            /** @type {?} */
            const manager = this._keyManager;
            switch (keyCode) {
                case ESCAPE:
                    if (!hasModifierKey(event)) {
                        event.preventDefault();
                        this.closed.emit('keydown');
                    }
                    break;
                case LEFT_ARROW:
                    if (this.parentMenu && this.direction === 'ltr') {
                        this.closed.emit('keydown');
                    }
                    break;
                case RIGHT_ARROW:
                    if (this.parentMenu && this.direction === 'rtl') {
                        this.closed.emit('keydown');
                    }
                    break;
                case HOME:
                case END:
                    if (!hasModifierKey(event)) {
                        keyCode === HOME ? manager.setFirstItemActive() : manager.setLastItemActive();
                        event.preventDefault();
                    }
                    break;
                default:
                    if (keyCode === UP_ARROW || keyCode === DOWN_ARROW) {
                        manager.setFocusOrigin('keyboard');
                    }
                    manager.onKeydown(event);
            }
        }
        /**
         * Focus the first item in the menu.
         * @param {?=} origin Action from which the focus originated. Used to set the correct styling.
         * @return {?}
         */
        focusFirstItem(origin = 'program') {
            // When the content is rendered lazily, it takes a bit before the items are inside the DOM.
            if (this.lazyContent) {
                this._ngZone.onStable.asObservable()
                    .pipe(take(1))
                    .subscribe((/**
                 * @return {?}
                 */
                () => this._focusFirstItem(origin)));
            }
            else {
                this._focusFirstItem(origin);
            }
        }
        /**
         * Actual implementation that focuses the first item. Needs to be separated
         * out so we don't repeat the same logic in the public `focusFirstItem` method.
         * @private
         * @param {?} origin
         * @return {?}
         */
        _focusFirstItem(origin) {
            /** @type {?} */
            const manager = this._keyManager;
            manager.setFocusOrigin(origin).setFirstItemActive();
            // If there's no active item at this point, it means that all the items are disabled.
            // Move focus to the menu panel so keyboard events like Escape still work. Also this will
            // give _some_ feedback to screen readers.
            if (!manager.activeItem && this._directDescendantItems.length) {
                /** @type {?} */
                let element = this._directDescendantItems.first._getHostElement().parentElement;
                // Because the `mat-menu` is at the DOM insertion point, not inside the overlay, we don't
                // have a nice way of getting a hold of the menu panel. We can't use a `ViewChild` either
                // because the panel is inside an `ng-template`. We work around it by starting from one of
                // the items and walking up the DOM.
                while (element) {
                    if (element.getAttribute('role') === 'menu') {
                        element.focus();
                        break;
                    }
                    else {
                        element = element.parentElement;
                    }
                }
            }
        }
        /**
         * Resets the active item in the menu. This is used when the menu is opened, allowing
         * the user to start from the first option when pressing the down arrow.
         * @return {?}
         */
        resetActiveItem() {
            this._keyManager.setActiveItem(-1);
        }
        /**
         * Sets the menu panel elevation.
         * @param {?} depth Number of parent menus that come before the menu.
         * @return {?}
         */
        setElevation(depth) {
            // The elevation starts at the base and increases by one for each level.
            // Capped at 24 because that's the maximum elevation defined in the Material design spec.
            /** @type {?} */
            const elevation = Math.min(MAT_MENU_BASE_ELEVATION + depth, 24);
            /** @type {?} */
            const newElevation = `mat-elevation-z${elevation}`;
            /** @type {?} */
            const customElevation = Object.keys(this._classList).find((/**
             * @param {?} c
             * @return {?}
             */
            c => c.startsWith('mat-elevation-z')));
            if (!customElevation || customElevation === this._previousElevation) {
                if (this._previousElevation) {
                    this._classList[this._previousElevation] = false;
                }
                this._classList[newElevation] = true;
                this._previousElevation = newElevation;
            }
        }
        /**
         * Adds classes to the menu panel based on its position. Can be used by
         * consumers to add specific styling based on the position.
         * \@docs-private
         * @param {?=} posX Position of the menu along the x axis.
         * @param {?=} posY Position of the menu along the y axis.
         * @return {?}
         */
        setPositionClasses(posX = this.xPosition, posY = this.yPosition) {
            /** @type {?} */
            const classes = this._classList;
            classes['mat-menu-before'] = posX === 'before';
            classes['mat-menu-after'] = posX === 'after';
            classes['mat-menu-above'] = posY === 'above';
            classes['mat-menu-below'] = posY === 'below';
        }
        /**
         * Starts the enter animation.
         * @return {?}
         */
        _startAnimation() {
            // @breaking-change 8.0.0 Combine with _resetAnimation.
            this._panelAnimationState = 'enter';
        }
        /**
         * Resets the panel animation to its initial state.
         * @return {?}
         */
        _resetAnimation() {
            // @breaking-change 8.0.0 Combine with _startAnimation.
            this._panelAnimationState = 'void';
        }
        /**
         * Callback that is invoked when the panel animation completes.
         * @param {?} event
         * @return {?}
         */
        _onAnimationDone(event) {
            this._animationDone.next(event);
            this._isAnimating = false;
        }
        /**
         * @param {?} event
         * @return {?}
         */
        _onAnimationStart(event) {
            this._isAnimating = true;
            // Scroll the content element to the top as soon as the animation starts. This is necessary,
            // because we move focus to the first item while it's still being animated, which can throw
            // the browser off when it determines the scroll position. Alternatively we can move focus
            // when the animation is done, however moving focus asynchronously will interrupt screen
            // readers which are in the process of reading out the menu already. We take the `element`
            // from the `event` since we can't use a `ViewChild` to access the pane.
            if (event.toState === 'enter' && this._keyManager.activeItemIndex === 0) {
                event.element.scrollTop = 0;
            }
        }
        /**
         * Sets up a stream that will keep track of any newly-added menu items and will update the list
         * of direct descendants. We collect the descendants this way, because `_allItems` can include
         * items that are part of child menus, and using a custom way of registering items is unreliable
         * when it comes to maintaining the item order.
         * @private
         * @return {?}
         */
        _updateDirectDescendants() {
            this._allItems.changes
                .pipe(startWith(this._allItems))
                .subscribe((/**
             * @param {?} items
             * @return {?}
             */
            (items) => {
                this._directDescendantItems.reset(items.filter((/**
                 * @param {?} item
                 * @return {?}
                 */
                item => item._parentMenu === this)));
                this._directDescendantItems.notifyOnChanges();
            }));
        }
    }
    _MatMenuBase.ɵfac = function _MatMenuBase_Factory(t) { return new (t || _MatMenuBase)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(NgZone), ɵɵdirectiveInject(MAT_MENU_DEFAULT_OPTIONS)); };
    _MatMenuBase.ɵdir = ɵɵdefineDirective({ type: _MatMenuBase, contentQueries: function _MatMenuBase_ContentQueries(rf, ctx, dirIndex) { if (rf & 1) {
            ɵɵcontentQuery(dirIndex, MatMenuContent, true);
            ɵɵcontentQuery(dirIndex, MatMenuItem, true);
            ɵɵcontentQuery(dirIndex, MatMenuItem, false);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.lazyContent = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._allItems = _t);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.items = _t);
        } }, viewQuery: function _MatMenuBase_Query(rf, ctx) { if (rf & 1) {
            ɵɵviewQuery(TemplateRef, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.templateRef = _t.first);
        } }, inputs: { backdropClass: "backdropClass", xPosition: "xPosition", yPosition: "yPosition", overlapTrigger: "overlapTrigger", hasBackdrop: "hasBackdrop", panelClass: ["class", "panelClass"], classList: "classList", ariaLabel: ["aria-label", "ariaLabel"], ariaLabelledby: ["aria-labelledby", "ariaLabelledby"], ariaDescribedby: ["aria-describedby", "ariaDescribedby"] }, outputs: { closed: "closed", close: "close" } });
    /** @nocollapse */
    _MatMenuBase.ctorParameters = () => [
        { type: ElementRef },
        { type: NgZone },
        { type: undefined, decorators: [{ type: Inject, args: [MAT_MENU_DEFAULT_OPTIONS,] }] }
    ];
    _MatMenuBase.propDecorators = {
        _allItems: [{ type: ContentChildren, args: [MatMenuItem, { descendants: true },] }],
        backdropClass: [{ type: Input }],
        ariaLabel: [{ type: Input, args: ['aria-label',] }],
        ariaLabelledby: [{ type: Input, args: ['aria-labelledby',] }],
        ariaDescribedby: [{ type: Input, args: ['aria-describedby',] }],
        xPosition: [{ type: Input }],
        yPosition: [{ type: Input }],
        templateRef: [{ type: ViewChild, args: [TemplateRef,] }],
        items: [{ type: ContentChildren, args: [MatMenuItem, { descendants: false },] }],
        lazyContent: [{ type: ContentChild, args: [MatMenuContent,] }],
        overlapTrigger: [{ type: Input }],
        hasBackdrop: [{ type: Input }],
        panelClass: [{ type: Input, args: ['class',] }],
        classList: [{ type: Input }],
        closed: [{ type: Output }],
        close: [{ type: Output }]
    };
    /**
     * \@docs-private We show the "_MatMenu" class as "MatMenu" in the docs.
     */
    class MatMenu extends _MatMenuBase {
    }
    MatMenu.ɵfac = function MatMenu_Factory(t) { return ɵMatMenu_BaseFactory(t || MatMenu); };
    MatMenu.ɵdir = ɵɵdefineDirective({ type: MatMenu, features: [ɵɵInheritDefinitionFeature] });
    const ɵMatMenu_BaseFactory = ɵɵgetInheritedFactory(MatMenu);
    // Note on the weird inheritance setup: we need three classes, because the MDC-based menu has to
    // extend `MatMenu`, however keeping a reference to it will cause the inlined template and styles
    // to be retained as well. The MDC menu also has to provide itself as a `MatMenu` in order for
    // queries and DI to work correctly, while still not referencing the actual menu class.
    // Class responsibility is split up as follows:
    // * _MatMenuBase - provides all the functionality without any of the Angular metadata.
    // * MatMenu - keeps the same name symbol name as the current menu and
    // is used as a provider for DI and query purposes.
    // * _MatMenu - the actual menu component implementation with the Angular metadata that should
    // be tree shaken away for MDC.
    /**
     * \@docs-public MatMenu
     */
    // tslint:disable-next-line:class-name
    class _MatMenu extends MatMenu {
        /**
         * @param {?} elementRef
         * @param {?} ngZone
         * @param {?} defaultOptions
         */
        constructor(elementRef, ngZone, defaultOptions) {
            super(elementRef, ngZone, defaultOptions);
        }
    }
    _MatMenu.ɵfac = function _MatMenu_Factory(t) { return new (t || _MatMenu)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(NgZone), ɵɵdirectiveInject(MAT_MENU_DEFAULT_OPTIONS)); };
    _MatMenu.ɵcmp = ɵɵdefineComponent({ type: _MatMenu, selectors: [["mat-menu"]], exportAs: ["matMenu"], features: [ɵɵProvidersFeature([
                { provide: MAT_MENU_PANEL, useExisting: MatMenu },
                { provide: MatMenu, useExisting: _MatMenu }
            ]), ɵɵInheritDefinitionFeature], ngContentSelectors: _c1$b, decls: 1, vars: 0, consts: [["tabindex", "-1", "role", "menu", 1, "mat-menu-panel", 3, "id", "ngClass", "keydown", "click"], [1, "mat-menu-content"]], template: function _MatMenu_Template(rf, ctx) { if (rf & 1) {
            ɵɵprojectionDef();
            ɵɵtemplate(0, _MatMenu_ng_template_0_Template, 3, 6, "ng-template");
        } }, directives: [NgClass], styles: [".mat-menu-panel{min-width:112px;max-width:280px;overflow:auto;-webkit-overflow-scrolling:touch;max-height:calc(100vh - 48px);border-radius:4px;outline:0;min-height:64px}.mat-menu-panel.ng-animating{pointer-events:none}.cdk-high-contrast-active .mat-menu-panel{outline:solid 1px}.mat-menu-content:not(:empty){padding-top:8px;padding-bottom:8px}.mat-menu-item{-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;cursor:pointer;outline:none;border:none;-webkit-tap-highlight-color:transparent;white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;line-height:48px;height:48px;padding:0 16px;text-align:left;text-decoration:none;max-width:100%;position:relative}.mat-menu-item::-moz-focus-inner{border:0}.mat-menu-item[disabled]{cursor:default}[dir=rtl] .mat-menu-item{text-align:right}.mat-menu-item .mat-icon{margin-right:16px;vertical-align:middle}.mat-menu-item .mat-icon svg{vertical-align:top}[dir=rtl] .mat-menu-item .mat-icon{margin-left:16px;margin-right:0}.mat-menu-item[disabled]{pointer-events:none}.cdk-high-contrast-active .mat-menu-item.cdk-program-focused,.cdk-high-contrast-active .mat-menu-item.cdk-keyboard-focused,.cdk-high-contrast-active .mat-menu-item-highlighted{outline:dotted 1px}.mat-menu-item-submenu-trigger{padding-right:32px}.mat-menu-item-submenu-trigger::after{width:0;height:0;border-style:solid;border-width:5px 0 5px 5px;border-color:transparent transparent transparent currentColor;content:\"\";display:inline-block;position:absolute;top:50%;right:16px;transform:translateY(-50%)}[dir=rtl] .mat-menu-item-submenu-trigger{padding-right:16px;padding-left:32px}[dir=rtl] .mat-menu-item-submenu-trigger::after{right:auto;left:16px;transform:rotateY(180deg) translateY(-50%)}button.mat-menu-item{width:100%}.mat-menu-item .mat-menu-ripple{top:0;left:0;right:0;bottom:0;position:absolute;pointer-events:none}\n"], encapsulation: 2, data: { animation: [
                matMenuAnimations.transformMenu,
                matMenuAnimations.fadeInItems
            ] }, changeDetection: 0 });
    /** @nocollapse */
    _MatMenu.ctorParameters = () => [
        { type: ElementRef },
        { type: NgZone },
        { type: undefined, decorators: [{ type: Inject, args: [MAT_MENU_DEFAULT_OPTIONS,] }] }
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/menu/menu-trigger.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Injection token that determines the scroll handling while the menu is open.
     * @type {?}
     */
    const MAT_MENU_SCROLL_STRATEGY = new InjectionToken('mat-menu-scroll-strategy');
    /**
     * \@docs-private
     * @param {?} overlay
     * @return {?}
     */
    function MAT_MENU_SCROLL_STRATEGY_FACTORY(overlay) {
        return (/**
         * @return {?}
         */
        () => overlay.scrollStrategies.reposition());
    }
    /**
     * \@docs-private
     * @type {?}
     */
    const MAT_MENU_SCROLL_STRATEGY_FACTORY_PROVIDER = {
        provide: MAT_MENU_SCROLL_STRATEGY,
        deps: [Overlay],
        useFactory: MAT_MENU_SCROLL_STRATEGY_FACTORY,
    };
    /**
     * Default top padding of the menu panel.
     * @type {?}
     */
    const MENU_PANEL_TOP_PADDING = 8;
    /**
     * Options for binding a passive event listener.
     * @type {?}
     */
    const passiveEventListenerOptions$1 = normalizePassiveListenerOptions({ passive: true });
    // TODO(andrewseguin): Remove the kebab versions in favor of camelCased attribute selectors
    /**
     * Directive applied to an element that should trigger a `mat-menu`.
     */
    class MatMenuTrigger {
        /**
         * @param {?} _overlay
         * @param {?} _element
         * @param {?} _viewContainerRef
         * @param {?} scrollStrategy
         * @param {?} _parentMenu
         * @param {?} _menuItemInstance
         * @param {?} _dir
         * @param {?=} _focusMonitor
         */
        constructor(_overlay, _element, _viewContainerRef, scrollStrategy, _parentMenu, _menuItemInstance, _dir, _focusMonitor) {
            this._overlay = _overlay;
            this._element = _element;
            this._viewContainerRef = _viewContainerRef;
            this._parentMenu = _parentMenu;
            this._menuItemInstance = _menuItemInstance;
            this._dir = _dir;
            this._focusMonitor = _focusMonitor;
            this._overlayRef = null;
            this._menuOpen = false;
            this._closingActionsSubscription = Subscription.EMPTY;
            this._hoverSubscription = Subscription.EMPTY;
            this._menuCloseSubscription = Subscription.EMPTY;
            /**
             * Handles touch start events on the trigger.
             * Needs to be an arrow function so we can easily use addEventListener and removeEventListener.
             */
            this._handleTouchStart = (/**
             * @return {?}
             */
            () => this._openedBy = 'touch');
            // Tracking input type is necessary so it's possible to only auto-focus
            // the first item of the list when the menu is opened via the keyboard
            this._openedBy = null;
            /**
             * Whether focus should be restored when the menu is closed.
             * Note that disabling this option can have accessibility implications
             * and it's up to you to manage focus, if you decide to turn it off.
             */
            this.restoreFocus = true;
            /**
             * Event emitted when the associated menu is opened.
             */
            this.menuOpened = new EventEmitter();
            /**
             * Event emitted when the associated menu is opened.
             * @deprecated Switch to `menuOpened` instead
             * \@breaking-change 8.0.0
             */
            // tslint:disable-next-line:no-output-on-prefix
            this.onMenuOpen = this.menuOpened;
            /**
             * Event emitted when the associated menu is closed.
             */
            this.menuClosed = new EventEmitter();
            /**
             * Event emitted when the associated menu is closed.
             * @deprecated Switch to `menuClosed` instead
             * \@breaking-change 8.0.0
             */
            // tslint:disable-next-line:no-output-on-prefix
            this.onMenuClose = this.menuClosed;
            _element.nativeElement.addEventListener('touchstart', this._handleTouchStart, passiveEventListenerOptions$1);
            if (_menuItemInstance) {
                _menuItemInstance._triggersSubmenu = this.triggersSubmenu();
            }
            this._scrollStrategy = scrollStrategy;
        }
        /**
         * @deprecated
         * \@breaking-change 8.0.0
         * @return {?}
         */
        get _deprecatedMatMenuTriggerFor() { return this.menu; }
        /**
         * @param {?} v
         * @return {?}
         */
        set _deprecatedMatMenuTriggerFor(v) {
            this.menu = v;
        }
        /**
         * References the menu instance that the trigger is associated with.
         * @return {?}
         */
        get menu() { return this._menu; }
        /**
         * @param {?} menu
         * @return {?}
         */
        set menu(menu) {
            if (menu === this._menu) {
                return;
            }
            this._menu = menu;
            this._menuCloseSubscription.unsubscribe();
            if (menu) {
                this._menuCloseSubscription = menu.close.asObservable().subscribe((/**
                 * @param {?} reason
                 * @return {?}
                 */
                reason => {
                    this._destroyMenu();
                    // If a click closed the menu, we should close the entire chain of nested menus.
                    if ((reason === 'click' || reason === 'tab') && this._parentMenu) {
                        this._parentMenu.closed.emit(reason);
                    }
                }));
            }
        }
        /**
         * @return {?}
         */
        ngAfterContentInit() {
            this._checkMenu();
            this._handleHover();
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            if (this._overlayRef) {
                this._overlayRef.dispose();
                this._overlayRef = null;
            }
            this._element.nativeElement.removeEventListener('touchstart', this._handleTouchStart, passiveEventListenerOptions$1);
            this._menuCloseSubscription.unsubscribe();
            this._closingActionsSubscription.unsubscribe();
            this._hoverSubscription.unsubscribe();
        }
        /**
         * Whether the menu is open.
         * @return {?}
         */
        get menuOpen() {
            return this._menuOpen;
        }
        /**
         * The text direction of the containing app.
         * @return {?}
         */
        get dir() {
            return this._dir && this._dir.value === 'rtl' ? 'rtl' : 'ltr';
        }
        /**
         * Whether the menu triggers a sub-menu or a top-level one.
         * @return {?}
         */
        triggersSubmenu() {
            return !!(this._menuItemInstance && this._parentMenu);
        }
        /**
         * Toggles the menu between the open and closed states.
         * @return {?}
         */
        toggleMenu() {
            return this._menuOpen ? this.closeMenu() : this.openMenu();
        }
        /**
         * Opens the menu.
         * @return {?}
         */
        openMenu() {
            if (this._menuOpen) {
                return;
            }
            this._checkMenu();
            /** @type {?} */
            const overlayRef = this._createOverlay();
            /** @type {?} */
            const overlayConfig = overlayRef.getConfig();
            this._setPosition((/** @type {?} */ (overlayConfig.positionStrategy)));
            overlayConfig.hasBackdrop = this.menu.hasBackdrop == null ? !this.triggersSubmenu() :
                this.menu.hasBackdrop;
            overlayRef.attach(this._getPortal());
            if (this.menu.lazyContent) {
                this.menu.lazyContent.attach(this.menuData);
            }
            this._closingActionsSubscription = this._menuClosingActions().subscribe((/**
             * @return {?}
             */
            () => this.closeMenu()));
            this._initMenu();
            if (this.menu instanceof MatMenu) {
                this.menu._startAnimation();
            }
        }
        /**
         * Closes the menu.
         * @return {?}
         */
        closeMenu() {
            this.menu.close.emit();
        }
        /**
         * Focuses the menu trigger.
         * @param {?=} origin Source of the menu trigger's focus.
         * @param {?=} options
         * @return {?}
         */
        focus(origin = 'program', options) {
            if (this._focusMonitor) {
                this._focusMonitor.focusVia(this._element, origin, options);
            }
            else {
                this._element.nativeElement.focus(options);
            }
        }
        /**
         * Closes the menu and does the necessary cleanup.
         * @private
         * @return {?}
         */
        _destroyMenu() {
            if (!this._overlayRef || !this.menuOpen) {
                return;
            }
            /** @type {?} */
            const menu = this.menu;
            this._closingActionsSubscription.unsubscribe();
            this._overlayRef.detach();
            this._restoreFocus();
            if (menu instanceof MatMenu) {
                menu._resetAnimation();
                if (menu.lazyContent) {
                    // Wait for the exit animation to finish before detaching the content.
                    menu._animationDone
                        .pipe(filter((/**
                     * @param {?} event
                     * @return {?}
                     */
                    event => event.toState === 'void')), take(1), 
                    // Interrupt if the content got re-attached.
                    takeUntil(menu.lazyContent._attached))
                        .subscribe({
                        next: (/**
                         * @return {?}
                         */
                        () => (/** @type {?} */ (menu.lazyContent)).detach()),
                        // No matter whether the content got re-attached, reset the menu.
                        complete: (/**
                         * @return {?}
                         */
                        () => this._setIsMenuOpen(false))
                    });
                }
                else {
                    this._setIsMenuOpen(false);
                }
            }
            else {
                this._setIsMenuOpen(false);
                if (menu.lazyContent) {
                    menu.lazyContent.detach();
                }
            }
        }
        /**
         * This method sets the menu state to open and focuses the first item if
         * the menu was opened via the keyboard.
         * @private
         * @return {?}
         */
        _initMenu() {
            this.menu.parentMenu = this.triggersSubmenu() ? this._parentMenu : undefined;
            this.menu.direction = this.dir;
            this._setMenuElevation();
            this._setIsMenuOpen(true);
            this.menu.focusFirstItem(this._openedBy || 'program');
        }
        /**
         * Updates the menu elevation based on the amount of parent menus that it has.
         * @private
         * @return {?}
         */
        _setMenuElevation() {
            if (this.menu.setElevation) {
                /** @type {?} */
                let depth = 0;
                /** @type {?} */
                let parentMenu = this.menu.parentMenu;
                while (parentMenu) {
                    depth++;
                    parentMenu = parentMenu.parentMenu;
                }
                this.menu.setElevation(depth);
            }
        }
        /**
         * Restores focus to the element that was focused before the menu was open.
         * @private
         * @return {?}
         */
        _restoreFocus() {
            // We should reset focus if the user is navigating using a keyboard or
            // if we have a top-level trigger which might cause focus to be lost
            // when clicking on the backdrop.
            if (this.restoreFocus) {
                if (!this._openedBy) {
                    // Note that the focus style will show up both for `program` and
                    // `keyboard` so we don't have to specify which one it is.
                    this.focus();
                }
                else if (!this.triggersSubmenu()) {
                    this.focus(this._openedBy);
                }
            }
            this._openedBy = null;
        }
        // set state rather than toggle to support triggers sharing a menu
        /**
         * @private
         * @param {?} isOpen
         * @return {?}
         */
        _setIsMenuOpen(isOpen) {
            this._menuOpen = isOpen;
            this._menuOpen ? this.menuOpened.emit() : this.menuClosed.emit();
            if (this.triggersSubmenu()) {
                this._menuItemInstance._highlighted = isOpen;
            }
        }
        /**
         * This method checks that a valid instance of MatMenu has been passed into
         * matMenuTriggerFor. If not, an exception is thrown.
         * @private
         * @return {?}
         */
        _checkMenu() {
            if (!this.menu) {
                throwMatMenuMissingError();
            }
        }
        /**
         * This method creates the overlay from the provided menu's template and saves its
         * OverlayRef so that it can be attached to the DOM when openMenu is called.
         * @private
         * @return {?}
         */
        _createOverlay() {
            if (!this._overlayRef) {
                /** @type {?} */
                const config = this._getOverlayConfig();
                this._subscribeToPositions((/** @type {?} */ (config.positionStrategy)));
                this._overlayRef = this._overlay.create(config);
                // Consume the `keydownEvents` in order to prevent them from going to another overlay.
                // Ideally we'd also have our keyboard event logic in here, however doing so will
                // break anybody that may have implemented the `MatMenuPanel` themselves.
                this._overlayRef.keydownEvents().subscribe();
            }
            return this._overlayRef;
        }
        /**
         * This method builds the configuration object needed to create the overlay, the OverlayState.
         * @private
         * @return {?} OverlayConfig
         */
        _getOverlayConfig() {
            return new OverlayConfig({
                positionStrategy: this._overlay.position()
                    .flexibleConnectedTo(this._element)
                    .withLockedPosition()
                    .withTransformOriginOn('.mat-menu-panel, .mat-mdc-menu-panel'),
                backdropClass: this.menu.backdropClass || 'cdk-overlay-transparent-backdrop',
                scrollStrategy: this._scrollStrategy(),
                direction: this._dir
            });
        }
        /**
         * Listens to changes in the position of the overlay and sets the correct classes
         * on the menu based on the new position. This ensures the animation origin is always
         * correct, even if a fallback position is used for the overlay.
         * @private
         * @param {?} position
         * @return {?}
         */
        _subscribeToPositions(position) {
            if (this.menu.setPositionClasses) {
                position.positionChanges.subscribe((/**
                 * @param {?} change
                 * @return {?}
                 */
                change => {
                    /** @type {?} */
                    const posX = change.connectionPair.overlayX === 'start' ? 'after' : 'before';
                    /** @type {?} */
                    const posY = change.connectionPair.overlayY === 'top' ? 'below' : 'above';
                    (/** @type {?} */ (this.menu.setPositionClasses))(posX, posY);
                }));
            }
        }
        /**
         * Sets the appropriate positions on a position strategy
         * so the overlay connects with the trigger correctly.
         * @private
         * @param {?} positionStrategy Strategy whose position to update.
         * @return {?}
         */
        _setPosition(positionStrategy) {
            let [originX, originFallbackX] = this.menu.xPosition === 'before' ? ['end', 'start'] : ['start', 'end'];
            let [overlayY, overlayFallbackY] = this.menu.yPosition === 'above' ? ['bottom', 'top'] : ['top', 'bottom'];
            let [originY, originFallbackY] = [overlayY, overlayFallbackY];
            let [overlayX, overlayFallbackX] = [originX, originFallbackX];
            /** @type {?} */
            let offsetY = 0;
            if (this.triggersSubmenu()) {
                // When the menu is a sub-menu, it should always align itself
                // to the edges of the trigger, instead of overlapping it.
                overlayFallbackX = originX = this.menu.xPosition === 'before' ? 'start' : 'end';
                originFallbackX = overlayX = originX === 'end' ? 'start' : 'end';
                offsetY = overlayY === 'bottom' ? MENU_PANEL_TOP_PADDING : -MENU_PANEL_TOP_PADDING;
            }
            else if (!this.menu.overlapTrigger) {
                originY = overlayY === 'top' ? 'bottom' : 'top';
                originFallbackY = overlayFallbackY === 'top' ? 'bottom' : 'top';
            }
            positionStrategy.withPositions([
                { originX, originY, overlayX, overlayY, offsetY },
                { originX: originFallbackX, originY, overlayX: overlayFallbackX, overlayY, offsetY },
                {
                    originX,
                    originY: originFallbackY,
                    overlayX,
                    overlayY: overlayFallbackY,
                    offsetY: -offsetY
                },
                {
                    originX: originFallbackX,
                    originY: originFallbackY,
                    overlayX: overlayFallbackX,
                    overlayY: overlayFallbackY,
                    offsetY: -offsetY
                }
            ]);
        }
        /**
         * Returns a stream that emits whenever an action that should close the menu occurs.
         * @private
         * @return {?}
         */
        _menuClosingActions() {
            /** @type {?} */
            const backdrop = (/** @type {?} */ (this._overlayRef)).backdropClick();
            /** @type {?} */
            const detachments = (/** @type {?} */ (this._overlayRef)).detachments();
            /** @type {?} */
            const parentClose = this._parentMenu ? this._parentMenu.closed : of();
            /** @type {?} */
            const hover = this._parentMenu ? this._parentMenu._hovered().pipe(filter((/**
             * @param {?} active
             * @return {?}
             */
            active => active !== this._menuItemInstance)), filter((/**
             * @return {?}
             */
            () => this._menuOpen))) : of();
            return merge(backdrop, parentClose, hover, detachments);
        }
        /**
         * Handles mouse presses on the trigger.
         * @param {?} event
         * @return {?}
         */
        _handleMousedown(event) {
            if (!isFakeMousedownFromScreenReader(event)) {
                // Since right or middle button clicks won't trigger the `click` event,
                // we shouldn't consider the menu as opened by mouse in those cases.
                this._openedBy = event.button === 0 ? 'mouse' : null;
                // Since clicking on the trigger won't close the menu if it opens a sub-menu,
                // we should prevent focus from moving onto it via click to avoid the
                // highlight from lingering on the menu item.
                if (this.triggersSubmenu()) {
                    event.preventDefault();
                }
            }
        }
        /**
         * Handles key presses on the trigger.
         * @param {?} event
         * @return {?}
         */
        _handleKeydown(event) {
            /** @type {?} */
            const keyCode = event.keyCode;
            if (this.triggersSubmenu() && ((keyCode === RIGHT_ARROW && this.dir === 'ltr') ||
                (keyCode === LEFT_ARROW && this.dir === 'rtl'))) {
                this.openMenu();
            }
        }
        /**
         * Handles click events on the trigger.
         * @param {?} event
         * @return {?}
         */
        _handleClick(event) {
            if (this.triggersSubmenu()) {
                // Stop event propagation to avoid closing the parent menu.
                event.stopPropagation();
                this.openMenu();
            }
            else {
                this.toggleMenu();
            }
        }
        /**
         * Handles the cases where the user hovers over the trigger.
         * @private
         * @return {?}
         */
        _handleHover() {
            // Subscribe to changes in the hovered item in order to toggle the panel.
            if (!this.triggersSubmenu()) {
                return;
            }
            this._hoverSubscription = this._parentMenu._hovered()
                // Since we might have multiple competing triggers for the same menu (e.g. a sub-menu
                // with different data and triggers), we have to delay it by a tick to ensure that
                // it won't be closed immediately after it is opened.
                .pipe(filter((/**
             * @param {?} active
             * @return {?}
             */
            active => active === this._menuItemInstance && !active.disabled)), delay(0, asap))
                .subscribe((/**
             * @return {?}
             */
            () => {
                this._openedBy = 'mouse';
                // If the same menu is used between multiple triggers, it might still be animating
                // while the new trigger tries to re-open it. Wait for the animation to finish
                // before doing so. Also interrupt if the user moves to another item.
                if (this.menu instanceof MatMenu && this.menu._isAnimating) {
                    // We need the `delay(0)` here in order to avoid
                    // 'changed after checked' errors in some cases. See #12194.
                    this.menu._animationDone
                        .pipe(take(1), delay(0, asap), takeUntil(this._parentMenu._hovered()))
                        .subscribe((/**
                     * @return {?}
                     */
                    () => this.openMenu()));
                }
                else {
                    this.openMenu();
                }
            }));
        }
        /**
         * Gets the portal that should be attached to the overlay.
         * @private
         * @return {?}
         */
        _getPortal() {
            // Note that we can avoid this check by keeping the portal on the menu panel.
            // While it would be cleaner, we'd have to introduce another required method on
            // `MatMenuPanel`, making it harder to consume.
            if (!this._portal || this._portal.templateRef !== this.menu.templateRef) {
                this._portal = new TemplatePortal(this.menu.templateRef, this._viewContainerRef);
            }
            return this._portal;
        }
    }
    MatMenuTrigger.ɵfac = function MatMenuTrigger_Factory(t) { return new (t || MatMenuTrigger)(ɵɵdirectiveInject(Overlay), ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(ViewContainerRef), ɵɵdirectiveInject(MAT_MENU_SCROLL_STRATEGY), ɵɵdirectiveInject(MatMenu, 8), ɵɵdirectiveInject(MatMenuItem, 10), ɵɵdirectiveInject(Directionality, 8), ɵɵdirectiveInject(FocusMonitor)); };
    MatMenuTrigger.ɵdir = ɵɵdefineDirective({ type: MatMenuTrigger, selectors: [["", "mat-menu-trigger-for", ""], ["", "matMenuTriggerFor", ""]], hostAttrs: ["aria-haspopup", "true", 1, "mat-menu-trigger"], hostVars: 2, hostBindings: function MatMenuTrigger_HostBindings(rf, ctx) { if (rf & 1) {
            ɵɵlistener("mousedown", function MatMenuTrigger_mousedown_HostBindingHandler($event) { return ctx._handleMousedown($event); })("keydown", function MatMenuTrigger_keydown_HostBindingHandler($event) { return ctx._handleKeydown($event); })("click", function MatMenuTrigger_click_HostBindingHandler($event) { return ctx._handleClick($event); });
        } if (rf & 2) {
            ɵɵattribute("aria-expanded", ctx.menuOpen || null)("aria-controls", ctx.menuOpen ? ctx.menu.panelId : null);
        } }, inputs: { restoreFocus: ["matMenuTriggerRestoreFocus", "restoreFocus"], _deprecatedMatMenuTriggerFor: ["mat-menu-trigger-for", "_deprecatedMatMenuTriggerFor"], menu: ["matMenuTriggerFor", "menu"], menuData: ["matMenuTriggerData", "menuData"] }, outputs: { menuOpened: "menuOpened", onMenuOpen: "onMenuOpen", menuClosed: "menuClosed", onMenuClose: "onMenuClose" }, exportAs: ["matMenuTrigger"] });
    /** @nocollapse */
    MatMenuTrigger.ctorParameters = () => [
        { type: Overlay },
        { type: ElementRef },
        { type: ViewContainerRef },
        { type: undefined, decorators: [{ type: Inject, args: [MAT_MENU_SCROLL_STRATEGY,] }] },
        { type: MatMenu, decorators: [{ type: Optional }] },
        { type: MatMenuItem, decorators: [{ type: Optional }, { type: Self }] },
        { type: Directionality, decorators: [{ type: Optional }] },
        { type: FocusMonitor }
    ];
    MatMenuTrigger.propDecorators = {
        _deprecatedMatMenuTriggerFor: [{ type: Input, args: ['mat-menu-trigger-for',] }],
        menu: [{ type: Input, args: ['matMenuTriggerFor',] }],
        menuData: [{ type: Input, args: ['matMenuTriggerData',] }],
        restoreFocus: [{ type: Input, args: ['matMenuTriggerRestoreFocus',] }],
        menuOpened: [{ type: Output }],
        onMenuOpen: [{ type: Output }],
        menuClosed: [{ type: Output }],
        onMenuClose: [{ type: Output }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/menu/menu-module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Used by both the current `MatMenuModule` and the MDC `MatMenuModule`
     * to declare the menu-related directives.
     */
    // tslint:disable-next-line:class-name
    class _MatMenuDirectivesModule {
    }
    _MatMenuDirectivesModule.ɵmod = ɵɵdefineNgModule({ type: _MatMenuDirectivesModule });
    _MatMenuDirectivesModule.ɵinj = ɵɵdefineInjector({ factory: function _MatMenuDirectivesModule_Factory(t) { return new (t || _MatMenuDirectivesModule)(); }, providers: [MAT_MENU_SCROLL_STRATEGY_FACTORY_PROVIDER], imports: [MatCommonModule] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(_MatMenuDirectivesModule, { declarations: function () { return [MatMenuTrigger,
            MatMenuContent]; }, exports: function () { return [MatMenuTrigger,
            MatMenuContent,
            MatCommonModule]; } }); })();
    class MatMenuModule {
    }
    MatMenuModule.ɵmod = ɵɵdefineNgModule({ type: MatMenuModule });
    MatMenuModule.ɵinj = ɵɵdefineInjector({ factory: function MatMenuModule_Factory(t) { return new (t || MatMenuModule)(); }, providers: [MAT_MENU_SCROLL_STRATEGY_FACTORY_PROVIDER], imports: [[
                CommonModule,
                MatCommonModule,
                MatRippleModule,
                OverlayModule,
                _MatMenuDirectivesModule,
            ],
            CdkScrollableModule, MatCommonModule,
            _MatMenuDirectivesModule] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(MatMenuModule, { declarations: function () { return [_MatMenu,
            MatMenuItem]; }, imports: function () { return [CommonModule,
            MatCommonModule,
            MatRippleModule,
            OverlayModule,
            _MatMenuDirectivesModule]; }, exports: function () { return [CdkScrollableModule, MatCommonModule,
            _MatMenu,
            MatMenuItem,
            _MatMenuDirectivesModule]; } }); })();

    const matTooltipAnimations = {
        /**
         * Animation that transitions a tooltip in and out.
         */
        tooltipState: trigger('state', [
            state('initial, void, hidden', style({ opacity: 0, transform: 'scale(0)' })),
            state('visible', style({ transform: 'scale(1)' })),
            transition('* => visible', animate('200ms cubic-bezier(0, 0, 0.2, 1)', keyframes([
                style({ opacity: 0, transform: 'scale(0)', offset: 0 }),
                style({ opacity: 0.5, transform: 'scale(0.99)', offset: 0.5 }),
                style({ opacity: 1, transform: 'scale(1)', offset: 1 })
            ]))),
            transition('* => hidden', animate('100ms cubic-bezier(0, 0, 0.2, 1)', style({ opacity: 0 }))),
        ])
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/tooltip/tooltip.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Time in ms to throttle repositioning after scroll events.
     * @type {?}
     */
    const SCROLL_THROTTLE_MS = 20;
    /**
     * CSS class that will be attached to the overlay panel.
     * @type {?}
     */
    const TOOLTIP_PANEL_CLASS = 'mat-tooltip-panel';
    /**
     * Options used to bind passive event listeners.
     * @type {?}
     */
    const passiveListenerOptions = normalizePassiveListenerOptions({ passive: true });
    /**
     * Time between the user putting the pointer on a tooltip
     * trigger and the long press event being fired.
     * @type {?}
     */
    const LONGPRESS_DELAY = 500;
    /**
     * Creates an error to be thrown if the user supplied an invalid tooltip position.
     * \@docs-private
     * @param {?} position
     * @return {?}
     */
    function getMatTooltipInvalidPositionError(position) {
        return Error(`Tooltip position "${position}" is invalid.`);
    }
    /**
     * Injection token that determines the scroll handling while a tooltip is visible.
     * @type {?}
     */
    const MAT_TOOLTIP_SCROLL_STRATEGY = new InjectionToken('mat-tooltip-scroll-strategy');
    /**
     * \@docs-private
     * @param {?} overlay
     * @return {?}
     */
    function MAT_TOOLTIP_SCROLL_STRATEGY_FACTORY(overlay) {
        return (/**
         * @return {?}
         */
        () => overlay.scrollStrategies.reposition({ scrollThrottle: SCROLL_THROTTLE_MS }));
    }
    /**
     * \@docs-private
     * @type {?}
     */
    const MAT_TOOLTIP_SCROLL_STRATEGY_FACTORY_PROVIDER = {
        provide: MAT_TOOLTIP_SCROLL_STRATEGY,
        deps: [Overlay],
        useFactory: MAT_TOOLTIP_SCROLL_STRATEGY_FACTORY,
    };
    /**
     * Injection token to be used to override the default options for `matTooltip`.
     * @type {?}
     */
    const MAT_TOOLTIP_DEFAULT_OPTIONS = new InjectionToken('mat-tooltip-default-options', {
        providedIn: 'root',
        factory: MAT_TOOLTIP_DEFAULT_OPTIONS_FACTORY
    });
    /**
     * \@docs-private
     * @return {?}
     */
    function MAT_TOOLTIP_DEFAULT_OPTIONS_FACTORY() {
        return {
            showDelay: 0,
            hideDelay: 0,
            touchendHideDelay: 1500,
        };
    }
    /**
     * Directive that attaches a material design tooltip to the host element. Animates the showing and
     * hiding of a tooltip provided position (defaults to below the element).
     *
     * https://material.io/design/components/tooltips.html
     */
    class MatTooltip {
        /**
         * @param {?} _overlay
         * @param {?} _elementRef
         * @param {?} _scrollDispatcher
         * @param {?} _viewContainerRef
         * @param {?} _ngZone
         * @param {?} _platform
         * @param {?} _ariaDescriber
         * @param {?} _focusMonitor
         * @param {?} scrollStrategy
         * @param {?} _dir
         * @param {?} _defaultOptions
         * @param {?=} _hammerLoader
         */
        constructor(_overlay, _elementRef, _scrollDispatcher, _viewContainerRef, _ngZone, _platform, _ariaDescriber, _focusMonitor, scrollStrategy, _dir, _defaultOptions, 
        /**
         * @deprecated _hammerLoader parameter to be removed.
         * @breaking-change 9.0.0
         */
        // Note that we need to give Angular something to inject here so it doesn't throw.
        _hammerLoader) {
            this._overlay = _overlay;
            this._elementRef = _elementRef;
            this._scrollDispatcher = _scrollDispatcher;
            this._viewContainerRef = _viewContainerRef;
            this._ngZone = _ngZone;
            this._platform = _platform;
            this._ariaDescriber = _ariaDescriber;
            this._focusMonitor = _focusMonitor;
            this._dir = _dir;
            this._defaultOptions = _defaultOptions;
            this._position = 'below';
            this._disabled = false;
            /**
             * The default delay in ms before showing the tooltip after show is called
             */
            this.showDelay = this._defaultOptions.showDelay;
            /**
             * The default delay in ms before hiding the tooltip after hide is called
             */
            this.hideDelay = this._defaultOptions.hideDelay;
            /**
             * How touch gestures should be handled by the tooltip. On touch devices the tooltip directive
             * uses a long press gesture to show and hide, however it can conflict with the native browser
             * gestures. To work around the conflict, Angular Material disables native gestures on the
             * trigger, but that might not be desirable on particular elements (e.g. inputs and draggable
             * elements). The different values for this option configure the touch event handling as follows:
             * - `auto` - Enables touch gestures for all elements, but tries to avoid conflicts with native
             *   browser gestures on particular elements. In particular, it allows text selection on inputs
             *   and textareas, and preserves the native browser dragging on elements marked as `draggable`.
             * - `on` - Enables touch gestures for all elements and disables native
             *   browser gestures with no exceptions.
             * - `off` - Disables touch gestures. Note that this will prevent the tooltip from
             *   showing on touch devices.
             */
            this.touchGestures = 'auto';
            this._message = '';
            /**
             * Manually-bound passive event listeners.
             */
            this._passiveListeners = new Map();
            /**
             * Emits when the component is destroyed.
             */
            this._destroyed = new Subject();
            /**
             * Handles the keydown events on the host element.
             * Needs to be an arrow function so that we can use it in addEventListener.
             */
            this._handleKeydown = (/**
             * @param {?} event
             * @return {?}
             */
            (event) => {
                if (this._isTooltipVisible() && event.keyCode === ESCAPE && !hasModifierKey(event)) {
                    event.preventDefault();
                    event.stopPropagation();
                    this._ngZone.run((/**
                     * @return {?}
                     */
                    () => this.hide(0)));
                }
            });
            this._scrollStrategy = scrollStrategy;
            if (_defaultOptions) {
                if (_defaultOptions.position) {
                    this.position = _defaultOptions.position;
                }
                if (_defaultOptions.touchGestures) {
                    this.touchGestures = _defaultOptions.touchGestures;
                }
            }
            _focusMonitor.monitor(_elementRef)
                .pipe(takeUntil(this._destroyed))
                .subscribe((/**
             * @param {?} origin
             * @return {?}
             */
            origin => {
                // Note that the focus monitor runs outside the Angular zone.
                if (!origin) {
                    _ngZone.run((/**
                     * @return {?}
                     */
                    () => this.hide(0)));
                }
                else if (origin === 'keyboard') {
                    _ngZone.run((/**
                     * @return {?}
                     */
                    () => this.show()));
                }
            }));
            _ngZone.runOutsideAngular((/**
             * @return {?}
             */
            () => {
                _elementRef.nativeElement.addEventListener('keydown', this._handleKeydown);
            }));
        }
        /**
         * Allows the user to define the position of the tooltip relative to the parent element
         * @return {?}
         */
        get position() { return this._position; }
        /**
         * @param {?} value
         * @return {?}
         */
        set position(value) {
            if (value !== this._position) {
                this._position = value;
                if (this._overlayRef) {
                    this._updatePosition();
                    if (this._tooltipInstance) {
                        (/** @type {?} */ (this._tooltipInstance)).show(0);
                    }
                    this._overlayRef.updatePosition();
                }
            }
        }
        /**
         * Disables the display of the tooltip.
         * @return {?}
         */
        get disabled() { return this._disabled; }
        /**
         * @param {?} value
         * @return {?}
         */
        set disabled(value) {
            this._disabled = coerceBooleanProperty(value);
            // If tooltip is disabled, hide immediately.
            if (this._disabled) {
                this.hide(0);
            }
        }
        /**
         * The message to be displayed in the tooltip
         * @return {?}
         */
        get message() { return this._message; }
        /**
         * @param {?} value
         * @return {?}
         */
        set message(value) {
            this._ariaDescriber.removeDescription(this._elementRef.nativeElement, this._message);
            // If the message is not a string (e.g. number), convert it to a string and trim it.
            this._message = value != null ? `${value}`.trim() : '';
            if (!this._message && this._isTooltipVisible()) {
                this.hide(0);
            }
            else {
                this._updateTooltipMessage();
                this._ngZone.runOutsideAngular((/**
                 * @return {?}
                 */
                () => {
                    // The `AriaDescriber` has some functionality that avoids adding a description if it's the
                    // same as the `aria-label` of an element, however we can't know whether the tooltip trigger
                    // has a data-bound `aria-label` or when it'll be set for the first time. We can avoid the
                    // issue by deferring the description by a tick so Angular has time to set the `aria-label`.
                    Promise.resolve().then((/**
                     * @return {?}
                     */
                    () => {
                        this._ariaDescriber.describe(this._elementRef.nativeElement, this.message);
                    }));
                }));
            }
        }
        /**
         * Classes to be passed to the tooltip. Supports the same syntax as `ngClass`.
         * @return {?}
         */
        get tooltipClass() { return this._tooltipClass; }
        /**
         * @param {?} value
         * @return {?}
         */
        set tooltipClass(value) {
            this._tooltipClass = value;
            if (this._tooltipInstance) {
                this._setTooltipClass(this._tooltipClass);
            }
        }
        /**
         * Setup styling-specific things
         * @return {?}
         */
        ngOnInit() {
            // This needs to happen in `ngOnInit` so the initial values for all inputs have been set.
            this._setupPointerEvents();
        }
        /**
         * Dispose the tooltip when destroyed.
         * @return {?}
         */
        ngOnDestroy() {
            /** @type {?} */
            const nativeElement = this._elementRef.nativeElement;
            clearTimeout(this._touchstartTimeout);
            if (this._overlayRef) {
                this._overlayRef.dispose();
                this._tooltipInstance = null;
            }
            // Clean up the event listeners set in the constructor
            nativeElement.removeEventListener('keydown', this._handleKeydown);
            this._passiveListeners.forEach((/**
             * @param {?} listener
             * @param {?} event
             * @return {?}
             */
            (listener, event) => {
                nativeElement.removeEventListener(event, listener, passiveListenerOptions);
            }));
            this._passiveListeners.clear();
            this._destroyed.next();
            this._destroyed.complete();
            this._ariaDescriber.removeDescription(nativeElement, this.message);
            this._focusMonitor.stopMonitoring(nativeElement);
        }
        /**
         * Shows the tooltip after the delay in ms, defaults to tooltip-delay-show or 0ms if no input
         * @param {?=} delay
         * @return {?}
         */
        show(delay = this.showDelay) {
            if (this.disabled || !this.message || (this._isTooltipVisible() &&
                !(/** @type {?} */ (this._tooltipInstance))._showTimeoutId && !(/** @type {?} */ (this._tooltipInstance))._hideTimeoutId)) {
                return;
            }
            /** @type {?} */
            const overlayRef = this._createOverlay();
            this._detach();
            this._portal = this._portal || new ComponentPortal(TooltipComponent, this._viewContainerRef);
            this._tooltipInstance = overlayRef.attach(this._portal).instance;
            this._tooltipInstance.afterHidden()
                .pipe(takeUntil(this._destroyed))
                .subscribe((/**
             * @return {?}
             */
            () => this._detach()));
            this._setTooltipClass(this._tooltipClass);
            this._updateTooltipMessage();
            (/** @type {?} */ (this._tooltipInstance)).show(delay);
        }
        /**
         * Hides the tooltip after the delay in ms, defaults to tooltip-delay-hide or 0ms if no input
         * @param {?=} delay
         * @return {?}
         */
        hide(delay = this.hideDelay) {
            if (this._tooltipInstance) {
                this._tooltipInstance.hide(delay);
            }
        }
        /**
         * Shows/hides the tooltip
         * @return {?}
         */
        toggle() {
            this._isTooltipVisible() ? this.hide() : this.show();
        }
        /**
         * Returns true if the tooltip is currently visible to the user
         * @return {?}
         */
        _isTooltipVisible() {
            return !!this._tooltipInstance && this._tooltipInstance.isVisible();
        }
        /**
         * Create the overlay config and position strategy
         * @private
         * @return {?}
         */
        _createOverlay() {
            if (this._overlayRef) {
                return this._overlayRef;
            }
            /** @type {?} */
            const scrollableAncestors = this._scrollDispatcher.getAncestorScrollContainers(this._elementRef);
            // Create connected position strategy that listens for scroll events to reposition.
            /** @type {?} */
            const strategy = this._overlay.position()
                .flexibleConnectedTo(this._elementRef)
                .withTransformOriginOn('.mat-tooltip')
                .withFlexibleDimensions(false)
                .withViewportMargin(8)
                .withScrollableContainers(scrollableAncestors);
            strategy.positionChanges.pipe(takeUntil(this._destroyed)).subscribe((/**
             * @param {?} change
             * @return {?}
             */
            change => {
                if (this._tooltipInstance) {
                    if (change.scrollableViewProperties.isOverlayClipped && this._tooltipInstance.isVisible()) {
                        // After position changes occur and the overlay is clipped by
                        // a parent scrollable then close the tooltip.
                        this._ngZone.run((/**
                         * @return {?}
                         */
                        () => this.hide(0)));
                    }
                }
            }));
            this._overlayRef = this._overlay.create({
                direction: this._dir,
                positionStrategy: strategy,
                panelClass: TOOLTIP_PANEL_CLASS,
                scrollStrategy: this._scrollStrategy()
            });
            this._updatePosition();
            this._overlayRef.detachments()
                .pipe(takeUntil(this._destroyed))
                .subscribe((/**
             * @return {?}
             */
            () => this._detach()));
            return this._overlayRef;
        }
        /**
         * Detaches the currently-attached tooltip.
         * @private
         * @return {?}
         */
        _detach() {
            if (this._overlayRef && this._overlayRef.hasAttached()) {
                this._overlayRef.detach();
            }
            this._tooltipInstance = null;
        }
        /**
         * Updates the position of the current tooltip.
         * @private
         * @return {?}
         */
        _updatePosition() {
            /** @type {?} */
            const position = (/** @type {?} */ ((/** @type {?} */ (this._overlayRef)).getConfig().positionStrategy));
            /** @type {?} */
            const origin = this._getOrigin();
            /** @type {?} */
            const overlay = this._getOverlayPosition();
            position.withPositions([
                Object.assign(Object.assign({}, origin.main), overlay.main),
                Object.assign(Object.assign({}, origin.fallback), overlay.fallback)
            ]);
        }
        /**
         * Returns the origin position and a fallback position based on the user's position preference.
         * The fallback position is the inverse of the origin (e.g. `'below' -> 'above'`).
         * @return {?}
         */
        _getOrigin() {
            /** @type {?} */
            const isLtr = !this._dir || this._dir.value == 'ltr';
            /** @type {?} */
            const position = this.position;
            /** @type {?} */
            let originPosition;
            if (position == 'above' || position == 'below') {
                originPosition = { originX: 'center', originY: position == 'above' ? 'top' : 'bottom' };
            }
            else if (position == 'before' ||
                (position == 'left' && isLtr) ||
                (position == 'right' && !isLtr)) {
                originPosition = { originX: 'start', originY: 'center' };
            }
            else if (position == 'after' ||
                (position == 'right' && isLtr) ||
                (position == 'left' && !isLtr)) {
                originPosition = { originX: 'end', originY: 'center' };
            }
            else {
                throw getMatTooltipInvalidPositionError(position);
            }
            const { x, y } = this._invertPosition(originPosition.originX, originPosition.originY);
            return {
                main: originPosition,
                fallback: { originX: x, originY: y }
            };
        }
        /**
         * Returns the overlay position and a fallback position based on the user's preference
         * @return {?}
         */
        _getOverlayPosition() {
            /** @type {?} */
            const isLtr = !this._dir || this._dir.value == 'ltr';
            /** @type {?} */
            const position = this.position;
            /** @type {?} */
            let overlayPosition;
            if (position == 'above') {
                overlayPosition = { overlayX: 'center', overlayY: 'bottom' };
            }
            else if (position == 'below') {
                overlayPosition = { overlayX: 'center', overlayY: 'top' };
            }
            else if (position == 'before' ||
                (position == 'left' && isLtr) ||
                (position == 'right' && !isLtr)) {
                overlayPosition = { overlayX: 'end', overlayY: 'center' };
            }
            else if (position == 'after' ||
                (position == 'right' && isLtr) ||
                (position == 'left' && !isLtr)) {
                overlayPosition = { overlayX: 'start', overlayY: 'center' };
            }
            else {
                throw getMatTooltipInvalidPositionError(position);
            }
            const { x, y } = this._invertPosition(overlayPosition.overlayX, overlayPosition.overlayY);
            return {
                main: overlayPosition,
                fallback: { overlayX: x, overlayY: y }
            };
        }
        /**
         * Updates the tooltip message and repositions the overlay according to the new message length
         * @private
         * @return {?}
         */
        _updateTooltipMessage() {
            // Must wait for the message to be painted to the tooltip so that the overlay can properly
            // calculate the correct positioning based on the size of the text.
            if (this._tooltipInstance) {
                this._tooltipInstance.message = this.message;
                this._tooltipInstance._markForCheck();
                this._ngZone.onMicrotaskEmpty.asObservable().pipe(take(1), takeUntil(this._destroyed)).subscribe((/**
                 * @return {?}
                 */
                () => {
                    if (this._tooltipInstance) {
                        (/** @type {?} */ (this._overlayRef)).updatePosition();
                    }
                }));
            }
        }
        /**
         * Updates the tooltip class
         * @private
         * @param {?} tooltipClass
         * @return {?}
         */
        _setTooltipClass(tooltipClass) {
            if (this._tooltipInstance) {
                this._tooltipInstance.tooltipClass = tooltipClass;
                this._tooltipInstance._markForCheck();
            }
        }
        /**
         * Inverts an overlay position.
         * @private
         * @param {?} x
         * @param {?} y
         * @return {?}
         */
        _invertPosition(x, y) {
            if (this.position === 'above' || this.position === 'below') {
                if (y === 'top') {
                    y = 'bottom';
                }
                else if (y === 'bottom') {
                    y = 'top';
                }
            }
            else {
                if (x === 'end') {
                    x = 'start';
                }
                else if (x === 'start') {
                    x = 'end';
                }
            }
            return { x, y };
        }
        /**
         * Binds the pointer events to the tooltip trigger.
         * @private
         * @return {?}
         */
        _setupPointerEvents() {
            // The mouse events shouldn't be bound on mobile devices, because they can prevent the
            // first tap from firing its click event or can cause the tooltip to open for clicks.
            if (!this._platform.IOS && !this._platform.ANDROID) {
                this._passiveListeners
                    .set('mouseenter', (/**
                 * @return {?}
                 */
                () => this.show()))
                    .set('mouseleave', (/**
                 * @return {?}
                 */
                () => this.hide()));
            }
            else if (this.touchGestures !== 'off') {
                this._disableNativeGesturesIfNecessary();
                /** @type {?} */
                const touchendListener = (/**
                 * @return {?}
                 */
                () => {
                    clearTimeout(this._touchstartTimeout);
                    this.hide(this._defaultOptions.touchendHideDelay);
                });
                this._passiveListeners
                    .set('touchend', touchendListener)
                    .set('touchcancel', touchendListener)
                    .set('touchstart', (/**
                 * @return {?}
                 */
                () => {
                    // Note that it's important that we don't `preventDefault` here,
                    // because it can prevent click events from firing on the element.
                    clearTimeout(this._touchstartTimeout);
                    this._touchstartTimeout = setTimeout((/**
                     * @return {?}
                     */
                    () => this.show()), LONGPRESS_DELAY);
                }));
            }
            this._passiveListeners.forEach((/**
             * @param {?} listener
             * @param {?} event
             * @return {?}
             */
            (listener, event) => {
                this._elementRef.nativeElement.addEventListener(event, listener, passiveListenerOptions);
            }));
        }
        /**
         * Disables the native browser gestures, based on how the tooltip has been configured.
         * @private
         * @return {?}
         */
        _disableNativeGesturesIfNecessary() {
            /** @type {?} */
            const element = this._elementRef.nativeElement;
            /** @type {?} */
            const style = element.style;
            /** @type {?} */
            const gestures = this.touchGestures;
            if (gestures !== 'off') {
                // If gestures are set to `auto`, we don't disable text selection on inputs and
                // textareas, because it prevents the user from typing into them on iOS Safari.
                if (gestures === 'on' || (element.nodeName !== 'INPUT' && element.nodeName !== 'TEXTAREA')) {
                    style.userSelect = style.msUserSelect = style.webkitUserSelect =
                        ((/** @type {?} */ (style))).MozUserSelect = 'none';
                }
                // If we have `auto` gestures and the element uses native HTML dragging,
                // we don't set `-webkit-user-drag` because it prevents the native behavior.
                if (gestures === 'on' || !element.draggable) {
                    ((/** @type {?} */ (style))).webkitUserDrag = 'none';
                }
                style.touchAction = 'none';
                style.webkitTapHighlightColor = 'transparent';
            }
        }
    }
    MatTooltip.ɵfac = function MatTooltip_Factory(t) { return new (t || MatTooltip)(ɵɵdirectiveInject(Overlay), ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(ScrollDispatcher), ɵɵdirectiveInject(ViewContainerRef), ɵɵdirectiveInject(NgZone), ɵɵdirectiveInject(Platform), ɵɵdirectiveInject(AriaDescriber), ɵɵdirectiveInject(FocusMonitor), ɵɵdirectiveInject(MAT_TOOLTIP_SCROLL_STRATEGY), ɵɵdirectiveInject(Directionality, 8), ɵɵdirectiveInject(MAT_TOOLTIP_DEFAULT_OPTIONS, 8), ɵɵdirectiveInject(ElementRef)); };
    MatTooltip.ɵdir = ɵɵdefineDirective({ type: MatTooltip, selectors: [["", "matTooltip", ""]], inputs: { showDelay: ["matTooltipShowDelay", "showDelay"], hideDelay: ["matTooltipHideDelay", "hideDelay"], touchGestures: ["matTooltipTouchGestures", "touchGestures"], position: ["matTooltipPosition", "position"], disabled: ["matTooltipDisabled", "disabled"], message: ["matTooltip", "message"], tooltipClass: ["matTooltipClass", "tooltipClass"] }, exportAs: ["matTooltip"] });
    /** @nocollapse */
    MatTooltip.ctorParameters = () => [
        { type: Overlay },
        { type: ElementRef },
        { type: ScrollDispatcher },
        { type: ViewContainerRef },
        { type: NgZone },
        { type: Platform },
        { type: AriaDescriber },
        { type: FocusMonitor },
        { type: undefined, decorators: [{ type: Inject, args: [MAT_TOOLTIP_SCROLL_STRATEGY,] }] },
        { type: Directionality, decorators: [{ type: Optional }] },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [MAT_TOOLTIP_DEFAULT_OPTIONS,] }] },
        { type: undefined, decorators: [{ type: Inject, args: [ElementRef,] }] }
    ];
    MatTooltip.propDecorators = {
        position: [{ type: Input, args: ['matTooltipPosition',] }],
        disabled: [{ type: Input, args: ['matTooltipDisabled',] }],
        showDelay: [{ type: Input, args: ['matTooltipShowDelay',] }],
        hideDelay: [{ type: Input, args: ['matTooltipHideDelay',] }],
        touchGestures: [{ type: Input, args: ['matTooltipTouchGestures',] }],
        message: [{ type: Input, args: ['matTooltip',] }],
        tooltipClass: [{ type: Input, args: ['matTooltipClass',] }]
    };
    /**
     * Internal component that wraps the tooltip's content.
     * \@docs-private
     */
    class TooltipComponent {
        /**
         * @param {?} _changeDetectorRef
         * @param {?} _breakpointObserver
         */
        constructor(_changeDetectorRef, _breakpointObserver) {
            this._changeDetectorRef = _changeDetectorRef;
            this._breakpointObserver = _breakpointObserver;
            /**
             * Property watched by the animation framework to show or hide the tooltip
             */
            this._visibility = 'initial';
            /**
             * Whether interactions on the page should close the tooltip
             */
            this._closeOnInteraction = false;
            /**
             * Subject for notifying that the tooltip has been hidden from the view
             */
            this._onHide = new Subject();
            /**
             * Stream that emits whether the user has a handset-sized display.
             */
            this._isHandset = this._breakpointObserver.observe(Breakpoints.Handset);
        }
        /**
         * Shows the tooltip with an animation originating from the provided origin
         * @param {?} delay Amount of milliseconds to the delay showing the tooltip.
         * @return {?}
         */
        show(delay) {
            // Cancel the delayed hide if it is scheduled
            if (this._hideTimeoutId) {
                clearTimeout(this._hideTimeoutId);
                this._hideTimeoutId = null;
            }
            // Body interactions should cancel the tooltip if there is a delay in showing.
            this._closeOnInteraction = true;
            this._showTimeoutId = setTimeout((/**
             * @return {?}
             */
            () => {
                this._visibility = 'visible';
                this._showTimeoutId = null;
                // Mark for check so if any parent component has set the
                // ChangeDetectionStrategy to OnPush it will be checked anyways
                this._markForCheck();
            }), delay);
        }
        /**
         * Begins the animation to hide the tooltip after the provided delay in ms.
         * @param {?} delay Amount of milliseconds to delay showing the tooltip.
         * @return {?}
         */
        hide(delay) {
            // Cancel the delayed show if it is scheduled
            if (this._showTimeoutId) {
                clearTimeout(this._showTimeoutId);
                this._showTimeoutId = null;
            }
            this._hideTimeoutId = setTimeout((/**
             * @return {?}
             */
            () => {
                this._visibility = 'hidden';
                this._hideTimeoutId = null;
                // Mark for check so if any parent component has set the
                // ChangeDetectionStrategy to OnPush it will be checked anyways
                this._markForCheck();
            }), delay);
        }
        /**
         * Returns an observable that notifies when the tooltip has been hidden from view.
         * @return {?}
         */
        afterHidden() {
            return this._onHide.asObservable();
        }
        /**
         * Whether the tooltip is being displayed.
         * @return {?}
         */
        isVisible() {
            return this._visibility === 'visible';
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this._onHide.complete();
        }
        /**
         * @return {?}
         */
        _animationStart() {
            this._closeOnInteraction = false;
        }
        /**
         * @param {?} event
         * @return {?}
         */
        _animationDone(event) {
            /** @type {?} */
            const toState = (/** @type {?} */ (event.toState));
            if (toState === 'hidden' && !this.isVisible()) {
                this._onHide.next();
            }
            if (toState === 'visible' || toState === 'hidden') {
                this._closeOnInteraction = true;
            }
        }
        /**
         * Interactions on the HTML body should close the tooltip immediately as defined in the
         * material design spec.
         * https://material.io/design/components/tooltips.html#behavior
         * @return {?}
         */
        _handleBodyInteraction() {
            if (this._closeOnInteraction) {
                this.hide(0);
            }
        }
        /**
         * Marks that the tooltip needs to be checked in the next change detection run.
         * Mainly used for rendering the initial text before positioning a tooltip, which
         * can be problematic in components with OnPush change detection.
         * @return {?}
         */
        _markForCheck() {
            this._changeDetectorRef.markForCheck();
        }
    }
    TooltipComponent.ɵfac = function TooltipComponent_Factory(t) { return new (t || TooltipComponent)(ɵɵdirectiveInject(ChangeDetectorRef), ɵɵdirectiveInject(BreakpointObserver)); };
    TooltipComponent.ɵcmp = ɵɵdefineComponent({ type: TooltipComponent, selectors: [["mat-tooltip-component"]], hostAttrs: ["aria-hidden", "true"], hostVars: 2, hostBindings: function TooltipComponent_HostBindings(rf, ctx) { if (rf & 1) {
            ɵɵlistener("click", function TooltipComponent_click_HostBindingHandler() { return ctx._handleBodyInteraction(); }, false, ɵɵresolveBody);
        } if (rf & 2) {
            ɵɵstyleProp("zoom", ctx._visibility === "visible" ? 1 : null);
        } }, decls: 3, vars: 7, consts: [[1, "mat-tooltip", 3, "ngClass"]], template: function TooltipComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "div", 0);
            ɵɵlistener("@state.start", function TooltipComponent_Template_div_animation_state_start_0_listener() { return ctx._animationStart(); })("@state.done", function TooltipComponent_Template_div_animation_state_done_0_listener($event) { return ctx._animationDone($event); });
            ɵɵpipe(1, "async");
            ɵɵtext(2);
            ɵɵelementEnd();
        } if (rf & 2) {
            var tmp_0_0 = null;
            const currVal_0 = (tmp_0_0 = ɵɵpipeBind1(1, 5, ctx._isHandset)) == null ? null : tmp_0_0.matches;
            ɵɵclassProp("mat-tooltip-handset", currVal_0);
            ɵɵproperty("ngClass", ctx.tooltipClass)("@state", ctx._visibility);
            ɵɵadvance(2);
            ɵɵtextInterpolate(ctx.message);
        } }, directives: [NgClass], pipes: [AsyncPipe], styles: [".mat-tooltip-panel{pointer-events:none !important}.mat-tooltip{color:#fff;border-radius:4px;margin:14px;max-width:250px;padding-left:8px;padding-right:8px;overflow:hidden;text-overflow:ellipsis}.cdk-high-contrast-active .mat-tooltip{outline:solid 1px}.mat-tooltip-handset{margin:24px;padding-left:16px;padding-right:16px}\n"], encapsulation: 2, data: { animation: [matTooltipAnimations.tooltipState] }, changeDetection: 0 });
    /** @nocollapse */
    TooltipComponent.ctorParameters = () => [
        { type: ChangeDetectorRef },
        { type: BreakpointObserver }
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/tooltip/tooltip-module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class MatTooltipModule {
    }
    MatTooltipModule.ɵmod = ɵɵdefineNgModule({ type: MatTooltipModule });
    MatTooltipModule.ɵinj = ɵɵdefineInjector({ factory: function MatTooltipModule_Factory(t) { return new (t || MatTooltipModule)(); }, providers: [MAT_TOOLTIP_SCROLL_STRATEGY_FACTORY_PROVIDER], imports: [[
                A11yModule,
                CommonModule,
                OverlayModule,
                MatCommonModule,
            ],
            MatCommonModule, CdkScrollableModule] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(MatTooltipModule, { declarations: function () { return [MatTooltip,
            TooltipComponent]; }, imports: function () { return [A11yModule,
            CommonModule,
            OverlayModule,
            MatCommonModule]; }, exports: function () { return [MatTooltip,
            TooltipComponent,
            MatCommonModule, CdkScrollableModule]; } }); })();

    function MatPaginator_div_2_mat_form_field_3_mat_option_2_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "mat-option", 19);
        ɵɵtext(1);
        ɵɵelementEnd();
    } if (rf & 2) {
        const pageSizeOption_r6 = ctx.$implicit;
        ɵɵproperty("value", pageSizeOption_r6);
        ɵɵadvance(1);
        ɵɵtextInterpolate1(" ", pageSizeOption_r6, " ");
    } }
    function MatPaginator_div_2_mat_form_field_3_Template(rf, ctx) { if (rf & 1) {
        const _r8 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "mat-form-field", 16);
        ɵɵelementStart(1, "mat-select", 17);
        ɵɵlistener("selectionChange", function MatPaginator_div_2_mat_form_field_3_Template_mat_select_selectionChange_1_listener($event) { ɵɵrestoreView(_r8); const ctx_r7 = ɵɵnextContext(2); return ctx_r7._changePageSize($event.value); });
        ɵɵtemplate(2, MatPaginator_div_2_mat_form_field_3_mat_option_2_Template, 2, 2, "mat-option", 18);
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r3 = ɵɵnextContext(2);
        ɵɵproperty("color", ctx_r3.color);
        ɵɵadvance(1);
        ɵɵproperty("value", ctx_r3.pageSize)("disabled", ctx_r3.disabled)("aria-label", ctx_r3._intl.itemsPerPageLabel);
        ɵɵadvance(1);
        ɵɵproperty("ngForOf", ctx_r3._displayedPageSizeOptions);
    } }
    function MatPaginator_div_2_div_4_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 20);
        ɵɵtext(1);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r4 = ɵɵnextContext(2);
        ɵɵadvance(1);
        ɵɵtextInterpolate(ctx_r4.pageSize);
    } }
    function MatPaginator_div_2_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 12);
        ɵɵelementStart(1, "div", 13);
        ɵɵtext(2);
        ɵɵelementEnd();
        ɵɵtemplate(3, MatPaginator_div_2_mat_form_field_3_Template, 3, 5, "mat-form-field", 14);
        ɵɵtemplate(4, MatPaginator_div_2_div_4_Template, 2, 1, "div", 15);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r0 = ɵɵnextContext();
        ɵɵadvance(2);
        ɵɵtextInterpolate1(" ", ctx_r0._intl.itemsPerPageLabel, " ");
        ɵɵadvance(1);
        ɵɵproperty("ngIf", ctx_r0._displayedPageSizeOptions.length > 1);
        ɵɵadvance(1);
        ɵɵproperty("ngIf", ctx_r0._displayedPageSizeOptions.length <= 1);
    } }
    function MatPaginator_button_6_Template(rf, ctx) { if (rf & 1) {
        const _r10 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "button", 21);
        ɵɵlistener("click", function MatPaginator_button_6_Template_button_click_0_listener() { ɵɵrestoreView(_r10); const ctx_r9 = ɵɵnextContext(); return ctx_r9.firstPage(); });
        ɵɵnamespaceSVG();
        ɵɵelementStart(1, "svg", 7);
        ɵɵelement(2, "path", 22);
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r1 = ɵɵnextContext();
        ɵɵproperty("matTooltip", ctx_r1._intl.firstPageLabel)("matTooltipDisabled", ctx_r1._previousButtonsDisabled())("matTooltipPosition", "above")("disabled", ctx_r1._previousButtonsDisabled());
        ɵɵattribute("aria-label", ctx_r1._intl.firstPageLabel);
    } }
    function MatPaginator_button_13_Template(rf, ctx) { if (rf & 1) {
        const _r12 = ɵɵgetCurrentView();
        ɵɵnamespaceSVG();
        ɵɵnamespaceHTML();
        ɵɵelementStart(0, "button", 23);
        ɵɵlistener("click", function MatPaginator_button_13_Template_button_click_0_listener() { ɵɵrestoreView(_r12); const ctx_r11 = ɵɵnextContext(); return ctx_r11.lastPage(); });
        ɵɵnamespaceSVG();
        ɵɵelementStart(1, "svg", 7);
        ɵɵelement(2, "path", 24);
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r2 = ɵɵnextContext();
        ɵɵproperty("matTooltip", ctx_r2._intl.lastPageLabel)("matTooltipDisabled", ctx_r2._nextButtonsDisabled())("matTooltipPosition", "above")("disabled", ctx_r2._nextButtonsDisabled());
        ɵɵattribute("aria-label", ctx_r2._intl.lastPageLabel);
    } }
    class MatPaginatorIntl {
        constructor() {
            /**
             * Stream to emit from when labels are changed. Use this to notify components when the labels have
             * changed after initialization.
             */
            this.changes = new Subject();
            /**
             * A label for the page size selector.
             */
            this.itemsPerPageLabel = 'Items per page:';
            /**
             * A label for the button that increments the current page.
             */
            this.nextPageLabel = 'Next page';
            /**
             * A label for the button that decrements the current page.
             */
            this.previousPageLabel = 'Previous page';
            /**
             * A label for the button that moves to the first page.
             */
            this.firstPageLabel = 'First page';
            /**
             * A label for the button that moves to the last page.
             */
            this.lastPageLabel = 'Last page';
            /**
             * A label for the range of items within the current page and the length of the whole list.
             */
            this.getRangeLabel = (/**
             * @param {?} page
             * @param {?} pageSize
             * @param {?} length
             * @return {?}
             */
            (page, pageSize, length) => {
                if (length == 0 || pageSize == 0) {
                    return `0 of ${length}`;
                }
                length = Math.max(length, 0);
                /** @type {?} */
                const startIndex = page * pageSize;
                // If the start index exceeds the list length, do not try and fix the end index to the end.
                /** @type {?} */
                const endIndex = startIndex < length ?
                    Math.min(startIndex + pageSize, length) :
                    startIndex + pageSize;
                return `${startIndex + 1} – ${endIndex} of ${length}`;
            });
        }
    }
    MatPaginatorIntl.ɵfac = function MatPaginatorIntl_Factory(t) { return new (t || MatPaginatorIntl)(); };
    /** @nocollapse */ MatPaginatorIntl.ɵprov = ɵɵdefineInjectable({ factory: function MatPaginatorIntl_Factory() { return new MatPaginatorIntl(); }, token: MatPaginatorIntl, providedIn: "root" });
    /**
     * \@docs-private
     * @param {?} parentIntl
     * @return {?}
     */
    function MAT_PAGINATOR_INTL_PROVIDER_FACTORY(parentIntl) {
        return parentIntl || new MatPaginatorIntl();
    }
    /**
     * \@docs-private
     * @type {?}
     */
    const MAT_PAGINATOR_INTL_PROVIDER = {
        // If there is already an MatPaginatorIntl available, use that. Otherwise, provide a new one.
        provide: MatPaginatorIntl,
        deps: [[new Optional(), new SkipSelf(), MatPaginatorIntl]],
        useFactory: MAT_PAGINATOR_INTL_PROVIDER_FACTORY
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/paginator/paginator.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * The default page size if there is no page size and there are no provided page size options.
     * @type {?}
     */
    const DEFAULT_PAGE_SIZE = 50;
    /**
     * Injection token that can be used to provide the default options for the paginator module.
     * @type {?}
     */
    const MAT_PAGINATOR_DEFAULT_OPTIONS = new InjectionToken('MAT_PAGINATOR_DEFAULT_OPTIONS');
    // Boilerplate for applying mixins to MatPaginator.
    /**
     * \@docs-private
     */
    class MatPaginatorBase {
    }
    /** @type {?} */
    const _MatPaginatorBase = mixinDisabled(mixinInitialized(MatPaginatorBase));
    /**
     * Component to provide navigation between paged information. Displays the size of the current
     * page, user-selectable options to change that size, what items are being shown, and
     * navigational button to go to the previous or next page.
     */
    class MatPaginator extends _MatPaginatorBase {
        /**
         * @param {?} _intl
         * @param {?} _changeDetectorRef
         * @param {?=} defaults
         */
        constructor(_intl, _changeDetectorRef, defaults) {
            super();
            this._intl = _intl;
            this._changeDetectorRef = _changeDetectorRef;
            this._pageIndex = 0;
            this._length = 0;
            this._pageSizeOptions = [];
            this._hidePageSize = false;
            this._showFirstLastButtons = false;
            /**
             * Event emitted when the paginator changes the page size or page index.
             */
            this.page = new EventEmitter();
            this._intlChanges = _intl.changes.subscribe((/**
             * @return {?}
             */
            () => this._changeDetectorRef.markForCheck()));
            if (defaults) {
                const { pageSize, pageSizeOptions, hidePageSize, showFirstLastButtons } = defaults;
                if (pageSize != null) {
                    this._pageSize = pageSize;
                }
                if (pageSizeOptions != null) {
                    this._pageSizeOptions = pageSizeOptions;
                }
                if (hidePageSize != null) {
                    this._hidePageSize = hidePageSize;
                }
                if (showFirstLastButtons != null) {
                    this._showFirstLastButtons = showFirstLastButtons;
                }
            }
        }
        /**
         * The zero-based page index of the displayed list of items. Defaulted to 0.
         * @return {?}
         */
        get pageIndex() { return this._pageIndex; }
        /**
         * @param {?} value
         * @return {?}
         */
        set pageIndex(value) {
            this._pageIndex = Math.max(coerceNumberProperty(value), 0);
            this._changeDetectorRef.markForCheck();
        }
        /**
         * The length of the total number of items that are being paginated. Defaulted to 0.
         * @return {?}
         */
        get length() { return this._length; }
        /**
         * @param {?} value
         * @return {?}
         */
        set length(value) {
            this._length = coerceNumberProperty(value);
            this._changeDetectorRef.markForCheck();
        }
        /**
         * Number of items to display on a page. By default set to 50.
         * @return {?}
         */
        get pageSize() { return this._pageSize; }
        /**
         * @param {?} value
         * @return {?}
         */
        set pageSize(value) {
            this._pageSize = Math.max(coerceNumberProperty(value), 0);
            this._updateDisplayedPageSizeOptions();
        }
        /**
         * The set of provided page size options to display to the user.
         * @return {?}
         */
        get pageSizeOptions() { return this._pageSizeOptions; }
        /**
         * @param {?} value
         * @return {?}
         */
        set pageSizeOptions(value) {
            this._pageSizeOptions = (value || []).map((/**
             * @param {?} p
             * @return {?}
             */
            p => coerceNumberProperty(p)));
            this._updateDisplayedPageSizeOptions();
        }
        /**
         * Whether to hide the page size selection UI from the user.
         * @return {?}
         */
        get hidePageSize() { return this._hidePageSize; }
        /**
         * @param {?} value
         * @return {?}
         */
        set hidePageSize(value) {
            this._hidePageSize = coerceBooleanProperty(value);
        }
        /**
         * Whether to show the first/last buttons UI to the user.
         * @return {?}
         */
        get showFirstLastButtons() { return this._showFirstLastButtons; }
        /**
         * @param {?} value
         * @return {?}
         */
        set showFirstLastButtons(value) {
            this._showFirstLastButtons = coerceBooleanProperty(value);
        }
        /**
         * @return {?}
         */
        ngOnInit() {
            this._initialized = true;
            this._updateDisplayedPageSizeOptions();
            this._markInitialized();
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this._intlChanges.unsubscribe();
        }
        /**
         * Advances to the next page if it exists.
         * @return {?}
         */
        nextPage() {
            if (!this.hasNextPage()) {
                return;
            }
            /** @type {?} */
            const previousPageIndex = this.pageIndex;
            this.pageIndex++;
            this._emitPageEvent(previousPageIndex);
        }
        /**
         * Move back to the previous page if it exists.
         * @return {?}
         */
        previousPage() {
            if (!this.hasPreviousPage()) {
                return;
            }
            /** @type {?} */
            const previousPageIndex = this.pageIndex;
            this.pageIndex--;
            this._emitPageEvent(previousPageIndex);
        }
        /**
         * Move to the first page if not already there.
         * @return {?}
         */
        firstPage() {
            // hasPreviousPage being false implies at the start
            if (!this.hasPreviousPage()) {
                return;
            }
            /** @type {?} */
            const previousPageIndex = this.pageIndex;
            this.pageIndex = 0;
            this._emitPageEvent(previousPageIndex);
        }
        /**
         * Move to the last page if not already there.
         * @return {?}
         */
        lastPage() {
            // hasNextPage being false implies at the end
            if (!this.hasNextPage()) {
                return;
            }
            /** @type {?} */
            const previousPageIndex = this.pageIndex;
            this.pageIndex = this.getNumberOfPages() - 1;
            this._emitPageEvent(previousPageIndex);
        }
        /**
         * Whether there is a previous page.
         * @return {?}
         */
        hasPreviousPage() {
            return this.pageIndex >= 1 && this.pageSize != 0;
        }
        /**
         * Whether there is a next page.
         * @return {?}
         */
        hasNextPage() {
            /** @type {?} */
            const maxPageIndex = this.getNumberOfPages() - 1;
            return this.pageIndex < maxPageIndex && this.pageSize != 0;
        }
        /**
         * Calculate the number of pages
         * @return {?}
         */
        getNumberOfPages() {
            if (!this.pageSize) {
                return 0;
            }
            return Math.ceil(this.length / this.pageSize);
        }
        /**
         * Changes the page size so that the first item displayed on the page will still be
         * displayed using the new page size.
         *
         * For example, if the page size is 10 and on the second page (items indexed 10-19) then
         * switching so that the page size is 5 will set the third page as the current page so
         * that the 10th item will still be displayed.
         * @param {?} pageSize
         * @return {?}
         */
        _changePageSize(pageSize) {
            // Current page needs to be updated to reflect the new page size. Navigate to the page
            // containing the previous page's first item.
            /** @type {?} */
            const startIndex = this.pageIndex * this.pageSize;
            /** @type {?} */
            const previousPageIndex = this.pageIndex;
            this.pageIndex = Math.floor(startIndex / pageSize) || 0;
            this.pageSize = pageSize;
            this._emitPageEvent(previousPageIndex);
        }
        /**
         * Checks whether the buttons for going forwards should be disabled.
         * @return {?}
         */
        _nextButtonsDisabled() {
            return this.disabled || !this.hasNextPage();
        }
        /**
         * Checks whether the buttons for going backwards should be disabled.
         * @return {?}
         */
        _previousButtonsDisabled() {
            return this.disabled || !this.hasPreviousPage();
        }
        /**
         * Updates the list of page size options to display to the user. Includes making sure that
         * the page size is an option and that the list is sorted.
         * @private
         * @return {?}
         */
        _updateDisplayedPageSizeOptions() {
            if (!this._initialized) {
                return;
            }
            // If no page size is provided, use the first page size option or the default page size.
            if (!this.pageSize) {
                this._pageSize = this.pageSizeOptions.length != 0 ?
                    this.pageSizeOptions[0] :
                    DEFAULT_PAGE_SIZE;
            }
            this._displayedPageSizeOptions = this.pageSizeOptions.slice();
            if (this._displayedPageSizeOptions.indexOf(this.pageSize) === -1) {
                this._displayedPageSizeOptions.push(this.pageSize);
            }
            // Sort the numbers using a number-specific sort function.
            this._displayedPageSizeOptions.sort((/**
             * @param {?} a
             * @param {?} b
             * @return {?}
             */
            (a, b) => a - b));
            this._changeDetectorRef.markForCheck();
        }
        /**
         * Emits an event notifying that a change of the paginator's properties has been triggered.
         * @private
         * @param {?} previousPageIndex
         * @return {?}
         */
        _emitPageEvent(previousPageIndex) {
            this.page.emit({
                previousPageIndex,
                pageIndex: this.pageIndex,
                pageSize: this.pageSize,
                length: this.length
            });
        }
    }
    MatPaginator.ɵfac = function MatPaginator_Factory(t) { return new (t || MatPaginator)(ɵɵdirectiveInject(MatPaginatorIntl), ɵɵdirectiveInject(ChangeDetectorRef), ɵɵdirectiveInject(MAT_PAGINATOR_DEFAULT_OPTIONS, 8)); };
    MatPaginator.ɵcmp = ɵɵdefineComponent({ type: MatPaginator, selectors: [["mat-paginator"]], hostAttrs: [1, "mat-paginator"], inputs: { disabled: "disabled", pageIndex: "pageIndex", length: "length", pageSize: "pageSize", pageSizeOptions: "pageSizeOptions", hidePageSize: "hidePageSize", showFirstLastButtons: "showFirstLastButtons", color: "color" }, outputs: { page: "page" }, exportAs: ["matPaginator"], features: [ɵɵInheritDefinitionFeature], decls: 14, vars: 14, consts: [[1, "mat-paginator-outer-container"], [1, "mat-paginator-container"], ["class", "mat-paginator-page-size", 4, "ngIf"], [1, "mat-paginator-range-actions"], [1, "mat-paginator-range-label"], ["mat-icon-button", "", "type", "button", "class", "mat-paginator-navigation-first", 3, "matTooltip", "matTooltipDisabled", "matTooltipPosition", "disabled", "click", 4, "ngIf"], ["mat-icon-button", "", "type", "button", 1, "mat-paginator-navigation-previous", 3, "matTooltip", "matTooltipDisabled", "matTooltipPosition", "disabled", "click"], ["viewBox", "0 0 24 24", "focusable", "false", 1, "mat-paginator-icon"], ["d", "M15.41 7.41L14 6l-6 6 6 6 1.41-1.41L10.83 12z"], ["mat-icon-button", "", "type", "button", 1, "mat-paginator-navigation-next", 3, "matTooltip", "matTooltipDisabled", "matTooltipPosition", "disabled", "click"], ["d", "M10 6L8.59 7.41 13.17 12l-4.58 4.59L10 18l6-6z"], ["mat-icon-button", "", "type", "button", "class", "mat-paginator-navigation-last", 3, "matTooltip", "matTooltipDisabled", "matTooltipPosition", "disabled", "click", 4, "ngIf"], [1, "mat-paginator-page-size"], [1, "mat-paginator-page-size-label"], ["class", "mat-paginator-page-size-select", 3, "color", 4, "ngIf"], ["class", "mat-paginator-page-size-value", 4, "ngIf"], [1, "mat-paginator-page-size-select", 3, "color"], [3, "value", "disabled", "aria-label", "selectionChange"], [3, "value", 4, "ngFor", "ngForOf"], [3, "value"], [1, "mat-paginator-page-size-value"], ["mat-icon-button", "", "type", "button", 1, "mat-paginator-navigation-first", 3, "matTooltip", "matTooltipDisabled", "matTooltipPosition", "disabled", "click"], ["d", "M18.41 16.59L13.82 12l4.59-4.59L17 6l-6 6 6 6zM6 6h2v12H6z"], ["mat-icon-button", "", "type", "button", 1, "mat-paginator-navigation-last", 3, "matTooltip", "matTooltipDisabled", "matTooltipPosition", "disabled", "click"], ["d", "M5.59 7.41L10.18 12l-4.59 4.59L7 18l6-6-6-6zM16 6h2v12h-2z"]], template: function MatPaginator_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "div", 0);
            ɵɵelementStart(1, "div", 1);
            ɵɵtemplate(2, MatPaginator_div_2_Template, 5, 3, "div", 2);
            ɵɵelementStart(3, "div", 3);
            ɵɵelementStart(4, "div", 4);
            ɵɵtext(5);
            ɵɵelementEnd();
            ɵɵtemplate(6, MatPaginator_button_6_Template, 3, 5, "button", 5);
            ɵɵelementStart(7, "button", 6);
            ɵɵlistener("click", function MatPaginator_Template_button_click_7_listener() { return ctx.previousPage(); });
            ɵɵnamespaceSVG();
            ɵɵelementStart(8, "svg", 7);
            ɵɵelement(9, "path", 8);
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵnamespaceHTML();
            ɵɵelementStart(10, "button", 9);
            ɵɵlistener("click", function MatPaginator_Template_button_click_10_listener() { return ctx.nextPage(); });
            ɵɵnamespaceSVG();
            ɵɵelementStart(11, "svg", 7);
            ɵɵelement(12, "path", 10);
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵtemplate(13, MatPaginator_button_13_Template, 3, 5, "button", 11);
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵadvance(2);
            ɵɵproperty("ngIf", !ctx.hidePageSize);
            ɵɵadvance(3);
            ɵɵtextInterpolate1(" ", ctx._intl.getRangeLabel(ctx.pageIndex, ctx.pageSize, ctx.length), " ");
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx.showFirstLastButtons);
            ɵɵadvance(1);
            ɵɵproperty("matTooltip", ctx._intl.previousPageLabel)("matTooltipDisabled", ctx._previousButtonsDisabled())("matTooltipPosition", "above")("disabled", ctx._previousButtonsDisabled());
            ɵɵattribute("aria-label", ctx._intl.previousPageLabel);
            ɵɵadvance(3);
            ɵɵproperty("matTooltip", ctx._intl.nextPageLabel)("matTooltipDisabled", ctx._nextButtonsDisabled())("matTooltipPosition", "above")("disabled", ctx._nextButtonsDisabled());
            ɵɵattribute("aria-label", ctx._intl.nextPageLabel);
            ɵɵadvance(3);
            ɵɵproperty("ngIf", ctx.showFirstLastButtons);
        } }, directives: [NgIf, MatButton, MatTooltip, MatFormField, MatSelect, NgForOf, MatOption], styles: [".mat-paginator{display:block}.mat-paginator-outer-container{display:flex}.mat-paginator-container{display:flex;align-items:center;justify-content:flex-end;min-height:56px;padding:0 8px;flex-wrap:wrap-reverse;width:100%}.mat-paginator-page-size{display:flex;align-items:baseline;margin-right:8px}[dir=rtl] .mat-paginator-page-size{margin-right:0;margin-left:8px}.mat-paginator-page-size-label{margin:0 4px}.mat-paginator-page-size-select{margin:6px 4px 0 4px;width:56px}.mat-paginator-page-size-select.mat-form-field-appearance-outline{width:64px}.mat-paginator-page-size-select.mat-form-field-appearance-fill{width:64px}.mat-paginator-range-label{margin:0 32px 0 24px}.mat-paginator-range-actions{display:flex;align-items:center}.mat-paginator-icon{width:28px;fill:currentColor}[dir=rtl] .mat-paginator-icon{transform:rotate(180deg)}\n"], encapsulation: 2, changeDetection: 0 });
    /** @nocollapse */
    MatPaginator.ctorParameters = () => [
        { type: MatPaginatorIntl },
        { type: ChangeDetectorRef },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [MAT_PAGINATOR_DEFAULT_OPTIONS,] }] }
    ];
    MatPaginator.propDecorators = {
        color: [{ type: Input }],
        pageIndex: [{ type: Input }],
        length: [{ type: Input }],
        pageSize: [{ type: Input }],
        pageSizeOptions: [{ type: Input }],
        hidePageSize: [{ type: Input }],
        showFirstLastButtons: [{ type: Input }],
        page: [{ type: Output }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/paginator/paginator-module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class MatPaginatorModule {
    }
    MatPaginatorModule.ɵmod = ɵɵdefineNgModule({ type: MatPaginatorModule });
    MatPaginatorModule.ɵinj = ɵɵdefineInjector({ factory: function MatPaginatorModule_Factory(t) { return new (t || MatPaginatorModule)(); }, providers: [MAT_PAGINATOR_INTL_PROVIDER], imports: [[
                CommonModule,
                MatButtonModule,
                MatSelectModule,
                MatTooltipModule,
            ]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(MatPaginatorModule, { declarations: function () { return [MatPaginator]; }, imports: function () { return [CommonModule,
            MatButtonModule,
            MatSelectModule,
            MatTooltipModule]; }, exports: function () { return [MatPaginator]; } }); })();

    function MatProgressSpinner__svg_circle_1_Template(rf, ctx) { if (rf & 1) {
        ɵɵnamespaceSVG();
        ɵɵelement(0, "circle", 3);
    } if (rf & 2) {
        const ctx_r0 = ɵɵnextContext();
        ɵɵstyleProp("animation-name", "mat-progress-spinner-stroke-rotate-" + ctx_r0.diameter)("stroke-dashoffset", ctx_r0._strokeDashOffset, "px")("stroke-dasharray", ctx_r0._strokeCircumference, "px")("stroke-width", ctx_r0._circleStrokeWidth, "%");
        ɵɵattribute("r", ctx_r0._circleRadius);
    } }
    function MatProgressSpinner__svg_circle_2_Template(rf, ctx) { if (rf & 1) {
        ɵɵnamespaceSVG();
        ɵɵelement(0, "circle", 3);
    } if (rf & 2) {
        const ctx_r1 = ɵɵnextContext();
        ɵɵstyleProp("stroke-dashoffset", ctx_r1._strokeDashOffset, "px")("stroke-dasharray", ctx_r1._strokeCircumference, "px")("stroke-width", ctx_r1._circleStrokeWidth, "%");
        ɵɵattribute("r", ctx_r1._circleRadius);
    } }
    function MatSpinner__svg_circle_1_Template(rf, ctx) { if (rf & 1) {
        ɵɵnamespaceSVG();
        ɵɵelement(0, "circle", 3);
    } if (rf & 2) {
        const ctx_r0 = ɵɵnextContext();
        ɵɵstyleProp("animation-name", "mat-progress-spinner-stroke-rotate-" + ctx_r0.diameter)("stroke-dashoffset", ctx_r0._strokeDashOffset, "px")("stroke-dasharray", ctx_r0._strokeCircumference, "px")("stroke-width", ctx_r0._circleStrokeWidth, "%");
        ɵɵattribute("r", ctx_r0._circleRadius);
    } }
    function MatSpinner__svg_circle_2_Template(rf, ctx) { if (rf & 1) {
        ɵɵnamespaceSVG();
        ɵɵelement(0, "circle", 3);
    } if (rf & 2) {
        const ctx_r1 = ɵɵnextContext();
        ɵɵstyleProp("stroke-dashoffset", ctx_r1._strokeDashOffset, "px")("stroke-dasharray", ctx_r1._strokeCircumference, "px")("stroke-width", ctx_r1._circleStrokeWidth, "%");
        ɵɵattribute("r", ctx_r1._circleRadius);
    } }
    const BASE_SIZE = 100;
    /**
     * Base reference stroke width of the spinner.
     * \@docs-private
     * @type {?}
     */
    const BASE_STROKE_WIDTH = 10;
    // Boilerplate for applying mixins to MatProgressSpinner.
    /**
     * \@docs-private
     */
    class MatProgressSpinnerBase {
        /**
         * @param {?} _elementRef
         */
        constructor(_elementRef) {
            this._elementRef = _elementRef;
        }
    }
    /** @type {?} */
    const _MatProgressSpinnerMixinBase = mixinColor(MatProgressSpinnerBase, 'primary');
    /**
     * Injection token to be used to override the default options for `mat-progress-spinner`.
     * @type {?}
     */
    const MAT_PROGRESS_SPINNER_DEFAULT_OPTIONS = new InjectionToken('mat-progress-spinner-default-options', {
        providedIn: 'root',
        factory: MAT_PROGRESS_SPINNER_DEFAULT_OPTIONS_FACTORY,
    });
    /**
     * \@docs-private
     * @return {?}
     */
    function MAT_PROGRESS_SPINNER_DEFAULT_OPTIONS_FACTORY() {
        return { diameter: BASE_SIZE };
    }
    // .0001 percentage difference is necessary in order to avoid unwanted animation frames
    // for example because the animation duration is 4 seconds, .1% accounts to 4ms
    // which are enough to see the flicker described in
    // https://github.com/angular/components/issues/8984
    /** @type {?} */
    const INDETERMINATE_ANIMATION_TEMPLATE = `
 @keyframes mat-progress-spinner-stroke-rotate-DIAMETER {
    0%      { stroke-dashoffset: START_VALUE;  transform: rotate(0); }
    12.5%   { stroke-dashoffset: END_VALUE;    transform: rotate(0); }
    12.5001%  { stroke-dashoffset: END_VALUE;    transform: rotateX(180deg) rotate(72.5deg); }
    25%     { stroke-dashoffset: START_VALUE;  transform: rotateX(180deg) rotate(72.5deg); }

    25.0001%   { stroke-dashoffset: START_VALUE;  transform: rotate(270deg); }
    37.5%   { stroke-dashoffset: END_VALUE;    transform: rotate(270deg); }
    37.5001%  { stroke-dashoffset: END_VALUE;    transform: rotateX(180deg) rotate(161.5deg); }
    50%     { stroke-dashoffset: START_VALUE;  transform: rotateX(180deg) rotate(161.5deg); }

    50.0001%  { stroke-dashoffset: START_VALUE;  transform: rotate(180deg); }
    62.5%   { stroke-dashoffset: END_VALUE;    transform: rotate(180deg); }
    62.5001%  { stroke-dashoffset: END_VALUE;    transform: rotateX(180deg) rotate(251.5deg); }
    75%     { stroke-dashoffset: START_VALUE;  transform: rotateX(180deg) rotate(251.5deg); }

    75.0001%  { stroke-dashoffset: START_VALUE;  transform: rotate(90deg); }
    87.5%   { stroke-dashoffset: END_VALUE;    transform: rotate(90deg); }
    87.5001%  { stroke-dashoffset: END_VALUE;    transform: rotateX(180deg) rotate(341.5deg); }
    100%    { stroke-dashoffset: START_VALUE;  transform: rotateX(180deg) rotate(341.5deg); }
  }
`;
    /**
     * `<mat-progress-spinner>` component.
     */
    class MatProgressSpinner extends _MatProgressSpinnerMixinBase {
        /**
         * @param {?} _elementRef
         * @param {?} platform
         * @param {?} _document
         * @param {?} animationMode
         * @param {?=} defaults
         */
        constructor(_elementRef, platform, _document, animationMode, defaults) {
            super(_elementRef);
            this._elementRef = _elementRef;
            this._document = _document;
            this._diameter = BASE_SIZE;
            this._value = 0;
            this._fallbackAnimation = false;
            /**
             * Mode of the progress circle
             */
            this.mode = 'determinate';
            /** @type {?} */
            const trackedDiameters = MatProgressSpinner._diameters;
            // The base size is already inserted via the component's structural styles. We still
            // need to track it so we don't end up adding the same styles again.
            if (!trackedDiameters.has(_document.head)) {
                trackedDiameters.set(_document.head, new Set([BASE_SIZE]));
            }
            this._fallbackAnimation = platform.EDGE || platform.TRIDENT;
            this._noopAnimations = animationMode === 'NoopAnimations' &&
                (!!defaults && !defaults._forceAnimations);
            if (defaults) {
                if (defaults.diameter) {
                    this.diameter = defaults.diameter;
                }
                if (defaults.strokeWidth) {
                    this.strokeWidth = defaults.strokeWidth;
                }
            }
        }
        /**
         * The diameter of the progress spinner (will set width and height of svg).
         * @return {?}
         */
        get diameter() { return this._diameter; }
        /**
         * @param {?} size
         * @return {?}
         */
        set diameter(size) {
            this._diameter = coerceNumberProperty(size);
            // If this is set before `ngOnInit`, the style root may not have been resolved yet.
            if (!this._fallbackAnimation && this._styleRoot) {
                this._attachStyleNode();
            }
        }
        /**
         * Stroke width of the progress spinner.
         * @return {?}
         */
        get strokeWidth() {
            return this._strokeWidth || this.diameter / 10;
        }
        /**
         * @param {?} value
         * @return {?}
         */
        set strokeWidth(value) {
            this._strokeWidth = coerceNumberProperty(value);
        }
        /**
         * Value of the progress circle.
         * @return {?}
         */
        get value() {
            return this.mode === 'determinate' ? this._value : 0;
        }
        /**
         * @param {?} newValue
         * @return {?}
         */
        set value(newValue) {
            this._value = Math.max(0, Math.min(100, coerceNumberProperty(newValue)));
        }
        /**
         * @return {?}
         */
        ngOnInit() {
            /** @type {?} */
            const element = this._elementRef.nativeElement;
            // Note that we need to look up the root node in ngOnInit, rather than the constructor, because
            // Angular seems to create the element outside the shadow root and then moves it inside, if the
            // node is inside an `ngIf` and a ShadowDom-encapsulated component.
            this._styleRoot = _getShadowRoot(element) || this._document.head;
            this._attachStyleNode();
            // On IE and Edge, we can't animate the `stroke-dashoffset`
            // reliably so we fall back to a non-spec animation.
            /** @type {?} */
            const animationClass = `mat-progress-spinner-indeterminate${this._fallbackAnimation ? '-fallback' : ''}-animation`;
            element.classList.add(animationClass);
        }
        /**
         * The radius of the spinner, adjusted for stroke width.
         * @return {?}
         */
        get _circleRadius() {
            return (this.diameter - BASE_STROKE_WIDTH) / 2;
        }
        /**
         * The view box of the spinner's svg element.
         * @return {?}
         */
        get _viewBox() {
            /** @type {?} */
            const viewBox = this._circleRadius * 2 + this.strokeWidth;
            return `0 0 ${viewBox} ${viewBox}`;
        }
        /**
         * The stroke circumference of the svg circle.
         * @return {?}
         */
        get _strokeCircumference() {
            return 2 * Math.PI * this._circleRadius;
        }
        /**
         * The dash offset of the svg circle.
         * @return {?}
         */
        get _strokeDashOffset() {
            if (this.mode === 'determinate') {
                return this._strokeCircumference * (100 - this._value) / 100;
            }
            // In fallback mode set the circle to 80% and rotate it with CSS.
            if (this._fallbackAnimation && this.mode === 'indeterminate') {
                return this._strokeCircumference * 0.2;
            }
            return null;
        }
        /**
         * Stroke width of the circle in percent.
         * @return {?}
         */
        get _circleStrokeWidth() {
            return this.strokeWidth / this.diameter * 100;
        }
        /**
         * Dynamically generates a style tag containing the correct animation for this diameter.
         * @private
         * @return {?}
         */
        _attachStyleNode() {
            /** @type {?} */
            const styleRoot = this._styleRoot;
            /** @type {?} */
            const currentDiameter = this._diameter;
            /** @type {?} */
            const diameters = MatProgressSpinner._diameters;
            /** @type {?} */
            let diametersForElement = diameters.get(styleRoot);
            if (!diametersForElement || !diametersForElement.has(currentDiameter)) {
                /** @type {?} */
                const styleTag = this._document.createElement('style');
                styleTag.setAttribute('mat-spinner-animation', currentDiameter + '');
                styleTag.textContent = this._getAnimationText();
                styleRoot.appendChild(styleTag);
                if (!diametersForElement) {
                    diametersForElement = new Set();
                    diameters.set(styleRoot, diametersForElement);
                }
                diametersForElement.add(currentDiameter);
            }
        }
        /**
         * Generates animation styles adjusted for the spinner's diameter.
         * @private
         * @return {?}
         */
        _getAnimationText() {
            return INDETERMINATE_ANIMATION_TEMPLATE
                // Animation should begin at 5% and end at 80%
                .replace(/START_VALUE/g, `${0.95 * this._strokeCircumference}`)
                .replace(/END_VALUE/g, `${0.2 * this._strokeCircumference}`)
                .replace(/DIAMETER/g, `${this.diameter}`);
        }
    }
    MatProgressSpinner.ɵfac = function MatProgressSpinner_Factory(t) { return new (t || MatProgressSpinner)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(Platform), ɵɵdirectiveInject(DOCUMENT$1, 8), ɵɵdirectiveInject(ANIMATION_MODULE_TYPE, 8), ɵɵdirectiveInject(MAT_PROGRESS_SPINNER_DEFAULT_OPTIONS)); };
    MatProgressSpinner.ɵcmp = ɵɵdefineComponent({ type: MatProgressSpinner, selectors: [["mat-progress-spinner"]], hostAttrs: ["role", "progressbar", 1, "mat-progress-spinner"], hostVars: 10, hostBindings: function MatProgressSpinner_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵattribute("aria-valuemin", ctx.mode === "determinate" ? 0 : null)("aria-valuemax", ctx.mode === "determinate" ? 100 : null)("aria-valuenow", ctx.mode === "determinate" ? ctx.value : null)("mode", ctx.mode);
            ɵɵstyleProp("width", ctx.diameter, "px")("height", ctx.diameter, "px");
            ɵɵclassProp("_mat-animation-noopable", ctx._noopAnimations);
        } }, inputs: { color: "color", mode: "mode", diameter: "diameter", strokeWidth: "strokeWidth", value: "value" }, exportAs: ["matProgressSpinner"], features: [ɵɵInheritDefinitionFeature], decls: 3, vars: 8, consts: [["preserveAspectRatio", "xMidYMid meet", "focusable", "false", 3, "ngSwitch"], ["cx", "50%", "cy", "50%", 3, "animation-name", "stroke-dashoffset", "stroke-dasharray", "stroke-width", 4, "ngSwitchCase"], ["cx", "50%", "cy", "50%", 3, "stroke-dashoffset", "stroke-dasharray", "stroke-width", 4, "ngSwitchCase"], ["cx", "50%", "cy", "50%"]], template: function MatProgressSpinner_Template(rf, ctx) { if (rf & 1) {
            ɵɵnamespaceSVG();
            ɵɵelementStart(0, "svg", 0);
            ɵɵtemplate(1, MatProgressSpinner__svg_circle_1_Template, 1, 9, "circle", 1);
            ɵɵtemplate(2, MatProgressSpinner__svg_circle_2_Template, 1, 7, "circle", 2);
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵstyleProp("width", ctx.diameter, "px")("height", ctx.diameter, "px");
            ɵɵproperty("ngSwitch", ctx.mode === "indeterminate");
            ɵɵattribute("viewBox", ctx._viewBox);
            ɵɵadvance(1);
            ɵɵproperty("ngSwitchCase", true);
            ɵɵadvance(1);
            ɵɵproperty("ngSwitchCase", false);
        } }, directives: [NgSwitch, NgSwitchCase], styles: [".mat-progress-spinner{display:block;position:relative}.mat-progress-spinner svg{position:absolute;transform:rotate(-90deg);top:0;left:0;transform-origin:center;overflow:visible}.mat-progress-spinner circle{fill:transparent;transform-origin:center;transition:stroke-dashoffset 225ms linear}._mat-animation-noopable.mat-progress-spinner circle{transition:none;animation:none}.mat-progress-spinner.mat-progress-spinner-indeterminate-animation[mode=indeterminate]{animation:mat-progress-spinner-linear-rotate 2000ms linear infinite}._mat-animation-noopable.mat-progress-spinner.mat-progress-spinner-indeterminate-animation[mode=indeterminate]{transition:none;animation:none}.mat-progress-spinner.mat-progress-spinner-indeterminate-animation[mode=indeterminate] circle{transition-property:stroke;animation-duration:4000ms;animation-timing-function:cubic-bezier(0.35, 0, 0.25, 1);animation-iteration-count:infinite}._mat-animation-noopable.mat-progress-spinner.mat-progress-spinner-indeterminate-animation[mode=indeterminate] circle{transition:none;animation:none}.mat-progress-spinner.mat-progress-spinner-indeterminate-fallback-animation[mode=indeterminate]{animation:mat-progress-spinner-stroke-rotate-fallback 10000ms cubic-bezier(0.87, 0.03, 0.33, 1) infinite}._mat-animation-noopable.mat-progress-spinner.mat-progress-spinner-indeterminate-fallback-animation[mode=indeterminate]{transition:none;animation:none}.mat-progress-spinner.mat-progress-spinner-indeterminate-fallback-animation[mode=indeterminate] circle{transition-property:stroke}._mat-animation-noopable.mat-progress-spinner.mat-progress-spinner-indeterminate-fallback-animation[mode=indeterminate] circle{transition:none;animation:none}@keyframes mat-progress-spinner-linear-rotate{0%{transform:rotate(0deg)}100%{transform:rotate(360deg)}}@keyframes mat-progress-spinner-stroke-rotate-100{0%{stroke-dashoffset:268.606171575px;transform:rotate(0)}12.5%{stroke-dashoffset:56.5486677px;transform:rotate(0)}12.5001%{stroke-dashoffset:56.5486677px;transform:rotateX(180deg) rotate(72.5deg)}25%{stroke-dashoffset:268.606171575px;transform:rotateX(180deg) rotate(72.5deg)}25.0001%{stroke-dashoffset:268.606171575px;transform:rotate(270deg)}37.5%{stroke-dashoffset:56.5486677px;transform:rotate(270deg)}37.5001%{stroke-dashoffset:56.5486677px;transform:rotateX(180deg) rotate(161.5deg)}50%{stroke-dashoffset:268.606171575px;transform:rotateX(180deg) rotate(161.5deg)}50.0001%{stroke-dashoffset:268.606171575px;transform:rotate(180deg)}62.5%{stroke-dashoffset:56.5486677px;transform:rotate(180deg)}62.5001%{stroke-dashoffset:56.5486677px;transform:rotateX(180deg) rotate(251.5deg)}75%{stroke-dashoffset:268.606171575px;transform:rotateX(180deg) rotate(251.5deg)}75.0001%{stroke-dashoffset:268.606171575px;transform:rotate(90deg)}87.5%{stroke-dashoffset:56.5486677px;transform:rotate(90deg)}87.5001%{stroke-dashoffset:56.5486677px;transform:rotateX(180deg) rotate(341.5deg)}100%{stroke-dashoffset:268.606171575px;transform:rotateX(180deg) rotate(341.5deg)}}@keyframes mat-progress-spinner-stroke-rotate-fallback{0%{transform:rotate(0deg)}25%{transform:rotate(1170deg)}50%{transform:rotate(2340deg)}75%{transform:rotate(3510deg)}100%{transform:rotate(4680deg)}}\n"], encapsulation: 2, changeDetection: 0 });
    /**
     * Tracks diameters of existing instances to de-dupe generated styles (default d = 100).
     * We need to keep track of which elements the diameters were attached to, because for
     * elements in the Shadow DOM the style tags are attached to the shadow root, rather
     * than the document head.
     */
    MatProgressSpinner._diameters = new WeakMap();
    /** @nocollapse */
    MatProgressSpinner.ctorParameters = () => [
        { type: ElementRef },
        { type: Platform },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [DOCUMENT$1,] }] },
        { type: String, decorators: [{ type: Optional }, { type: Inject, args: [ANIMATION_MODULE_TYPE,] }] },
        { type: undefined, decorators: [{ type: Inject, args: [MAT_PROGRESS_SPINNER_DEFAULT_OPTIONS,] }] }
    ];
    MatProgressSpinner.propDecorators = {
        diameter: [{ type: Input }],
        strokeWidth: [{ type: Input }],
        mode: [{ type: Input }],
        value: [{ type: Input }]
    };
    /**
     * `<mat-spinner>` component.
     *
     * This is a component definition to be used as a convenience reference to create an
     * indeterminate `<mat-progress-spinner>` instance.
     */
    class MatSpinner extends MatProgressSpinner {
        /**
         * @param {?} elementRef
         * @param {?} platform
         * @param {?} document
         * @param {?} animationMode
         * @param {?=} defaults
         */
        constructor(elementRef, platform, document, animationMode, defaults) {
            super(elementRef, platform, document, animationMode, defaults);
            this.mode = 'indeterminate';
        }
    }
    MatSpinner.ɵfac = function MatSpinner_Factory(t) { return new (t || MatSpinner)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(Platform), ɵɵdirectiveInject(DOCUMENT$1, 8), ɵɵdirectiveInject(ANIMATION_MODULE_TYPE, 8), ɵɵdirectiveInject(MAT_PROGRESS_SPINNER_DEFAULT_OPTIONS)); };
    MatSpinner.ɵcmp = ɵɵdefineComponent({ type: MatSpinner, selectors: [["mat-spinner"]], hostAttrs: ["role", "progressbar", "mode", "indeterminate", 1, "mat-spinner", "mat-progress-spinner"], hostVars: 6, hostBindings: function MatSpinner_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵstyleProp("width", ctx.diameter, "px")("height", ctx.diameter, "px");
            ɵɵclassProp("_mat-animation-noopable", ctx._noopAnimations);
        } }, inputs: { color: "color" }, features: [ɵɵInheritDefinitionFeature], decls: 3, vars: 8, consts: [["preserveAspectRatio", "xMidYMid meet", "focusable", "false", 3, "ngSwitch"], ["cx", "50%", "cy", "50%", 3, "animation-name", "stroke-dashoffset", "stroke-dasharray", "stroke-width", 4, "ngSwitchCase"], ["cx", "50%", "cy", "50%", 3, "stroke-dashoffset", "stroke-dasharray", "stroke-width", 4, "ngSwitchCase"], ["cx", "50%", "cy", "50%"]], template: function MatSpinner_Template(rf, ctx) { if (rf & 1) {
            ɵɵnamespaceSVG();
            ɵɵelementStart(0, "svg", 0);
            ɵɵtemplate(1, MatSpinner__svg_circle_1_Template, 1, 9, "circle", 1);
            ɵɵtemplate(2, MatSpinner__svg_circle_2_Template, 1, 7, "circle", 2);
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵstyleProp("width", ctx.diameter, "px")("height", ctx.diameter, "px");
            ɵɵproperty("ngSwitch", ctx.mode === "indeterminate");
            ɵɵattribute("viewBox", ctx._viewBox);
            ɵɵadvance(1);
            ɵɵproperty("ngSwitchCase", true);
            ɵɵadvance(1);
            ɵɵproperty("ngSwitchCase", false);
        } }, directives: [NgSwitch, NgSwitchCase], styles: [".mat-progress-spinner{display:block;position:relative}.mat-progress-spinner svg{position:absolute;transform:rotate(-90deg);top:0;left:0;transform-origin:center;overflow:visible}.mat-progress-spinner circle{fill:transparent;transform-origin:center;transition:stroke-dashoffset 225ms linear}._mat-animation-noopable.mat-progress-spinner circle{transition:none;animation:none}.mat-progress-spinner.mat-progress-spinner-indeterminate-animation[mode=indeterminate]{animation:mat-progress-spinner-linear-rotate 2000ms linear infinite}._mat-animation-noopable.mat-progress-spinner.mat-progress-spinner-indeterminate-animation[mode=indeterminate]{transition:none;animation:none}.mat-progress-spinner.mat-progress-spinner-indeterminate-animation[mode=indeterminate] circle{transition-property:stroke;animation-duration:4000ms;animation-timing-function:cubic-bezier(0.35, 0, 0.25, 1);animation-iteration-count:infinite}._mat-animation-noopable.mat-progress-spinner.mat-progress-spinner-indeterminate-animation[mode=indeterminate] circle{transition:none;animation:none}.mat-progress-spinner.mat-progress-spinner-indeterminate-fallback-animation[mode=indeterminate]{animation:mat-progress-spinner-stroke-rotate-fallback 10000ms cubic-bezier(0.87, 0.03, 0.33, 1) infinite}._mat-animation-noopable.mat-progress-spinner.mat-progress-spinner-indeterminate-fallback-animation[mode=indeterminate]{transition:none;animation:none}.mat-progress-spinner.mat-progress-spinner-indeterminate-fallback-animation[mode=indeterminate] circle{transition-property:stroke}._mat-animation-noopable.mat-progress-spinner.mat-progress-spinner-indeterminate-fallback-animation[mode=indeterminate] circle{transition:none;animation:none}@keyframes mat-progress-spinner-linear-rotate{0%{transform:rotate(0deg)}100%{transform:rotate(360deg)}}@keyframes mat-progress-spinner-stroke-rotate-100{0%{stroke-dashoffset:268.606171575px;transform:rotate(0)}12.5%{stroke-dashoffset:56.5486677px;transform:rotate(0)}12.5001%{stroke-dashoffset:56.5486677px;transform:rotateX(180deg) rotate(72.5deg)}25%{stroke-dashoffset:268.606171575px;transform:rotateX(180deg) rotate(72.5deg)}25.0001%{stroke-dashoffset:268.606171575px;transform:rotate(270deg)}37.5%{stroke-dashoffset:56.5486677px;transform:rotate(270deg)}37.5001%{stroke-dashoffset:56.5486677px;transform:rotateX(180deg) rotate(161.5deg)}50%{stroke-dashoffset:268.606171575px;transform:rotateX(180deg) rotate(161.5deg)}50.0001%{stroke-dashoffset:268.606171575px;transform:rotate(180deg)}62.5%{stroke-dashoffset:56.5486677px;transform:rotate(180deg)}62.5001%{stroke-dashoffset:56.5486677px;transform:rotateX(180deg) rotate(251.5deg)}75%{stroke-dashoffset:268.606171575px;transform:rotateX(180deg) rotate(251.5deg)}75.0001%{stroke-dashoffset:268.606171575px;transform:rotate(90deg)}87.5%{stroke-dashoffset:56.5486677px;transform:rotate(90deg)}87.5001%{stroke-dashoffset:56.5486677px;transform:rotateX(180deg) rotate(341.5deg)}100%{stroke-dashoffset:268.606171575px;transform:rotateX(180deg) rotate(341.5deg)}}@keyframes mat-progress-spinner-stroke-rotate-fallback{0%{transform:rotate(0deg)}25%{transform:rotate(1170deg)}50%{transform:rotate(2340deg)}75%{transform:rotate(3510deg)}100%{transform:rotate(4680deg)}}\n"], encapsulation: 2, changeDetection: 0 });
    /** @nocollapse */
    MatSpinner.ctorParameters = () => [
        { type: ElementRef },
        { type: Platform },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [DOCUMENT$1,] }] },
        { type: String, decorators: [{ type: Optional }, { type: Inject, args: [ANIMATION_MODULE_TYPE,] }] },
        { type: undefined, decorators: [{ type: Inject, args: [MAT_PROGRESS_SPINNER_DEFAULT_OPTIONS,] }] }
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/progress-spinner/progress-spinner-module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class MatProgressSpinnerModule {
    }
    MatProgressSpinnerModule.ɵmod = ɵɵdefineNgModule({ type: MatProgressSpinnerModule });
    MatProgressSpinnerModule.ɵinj = ɵɵdefineInjector({ factory: function MatProgressSpinnerModule_Factory(t) { return new (t || MatProgressSpinnerModule)(); }, imports: [[MatCommonModule, CommonModule],
            MatCommonModule] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(MatProgressSpinnerModule, { declarations: function () { return [MatProgressSpinner,
            MatSpinner]; }, imports: function () { return [MatCommonModule, CommonModule]; }, exports: function () { return [MatProgressSpinner,
            MatSpinner,
            MatCommonModule]; } }); })();

    const _c0$l = ["mat-sort-header", ""];
    function MatSortHeader_div_3_Template(rf, ctx) { if (rf & 1) {
        const _r2 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "div", 3);
        ɵɵlistener("@arrowPosition.start", function MatSortHeader_div_3_Template_div_animation_arrowPosition_start_0_listener() { ɵɵrestoreView(_r2); const ctx_r1 = ɵɵnextContext(); return ctx_r1._disableViewStateAnimation = true; })("@arrowPosition.done", function MatSortHeader_div_3_Template_div_animation_arrowPosition_done_0_listener() { ɵɵrestoreView(_r2); const ctx_r3 = ɵɵnextContext(); return ctx_r3._disableViewStateAnimation = false; });
        ɵɵelement(1, "div", 4);
        ɵɵelementStart(2, "div", 5);
        ɵɵelement(3, "div", 6);
        ɵɵelement(4, "div", 7);
        ɵɵelement(5, "div", 8);
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r0 = ɵɵnextContext();
        ɵɵproperty("@arrowOpacity", ctx_r0._getArrowViewState())("@arrowPosition", ctx_r0._getArrowViewState())("@allowChildren", ctx_r0._getArrowDirectionState());
        ɵɵadvance(2);
        ɵɵproperty("@indicator", ctx_r0._getArrowDirectionState());
        ɵɵadvance(1);
        ɵɵproperty("@leftPointer", ctx_r0._getArrowDirectionState());
        ɵɵadvance(1);
        ɵɵproperty("@rightPointer", ctx_r0._getArrowDirectionState());
    } }
    const _c1$c = ["*"];
    function getSortDuplicateSortableIdError(id) {
        return Error(`Cannot have two MatSortables with the same id (${id}).`);
    }
    /**
     * \@docs-private
     * @return {?}
     */
    function getSortHeaderNotContainedWithinSortError() {
        return Error(`MatSortHeader must be placed within a parent element with the MatSort directive.`);
    }
    /**
     * \@docs-private
     * @return {?}
     */
    function getSortHeaderMissingIdError() {
        return Error(`MatSortHeader must be provided with a unique id.`);
    }
    /**
     * \@docs-private
     * @param {?} direction
     * @return {?}
     */
    function getSortInvalidDirectionError(direction) {
        return Error(`${direction} is not a valid sort direction ('asc' or 'desc').`);
    }
    // Boilerplate for applying mixins to MatSort.
    /**
     * \@docs-private
     */
    class MatSortBase {
    }
    /** @type {?} */
    const _MatSortMixinBase = mixinInitialized(mixinDisabled(MatSortBase));
    /**
     * Container for MatSortables to manage the sort state and provide default sort parameters.
     */
    class MatSort extends _MatSortMixinBase {
        constructor() {
            super(...arguments);
            /**
             * Collection of all registered sortables that this directive manages.
             */
            this.sortables = new Map();
            /**
             * Used to notify any child components listening to state changes.
             */
            this._stateChanges = new Subject();
            /**
             * The direction to set when an MatSortable is initially sorted.
             * May be overriden by the MatSortable's sort start.
             */
            this.start = 'asc';
            this._direction = '';
            /**
             * Event emitted when the user changes either the active sort or sort direction.
             */
            this.sortChange = new EventEmitter();
        }
        /**
         * The sort direction of the currently active MatSortable.
         * @return {?}
         */
        get direction() { return this._direction; }
        /**
         * @param {?} direction
         * @return {?}
         */
        set direction(direction) {
            if (isDevMode() && direction && direction !== 'asc' && direction !== 'desc') {
                throw getSortInvalidDirectionError(direction);
            }
            this._direction = direction;
        }
        /**
         * Whether to disable the user from clearing the sort by finishing the sort direction cycle.
         * May be overriden by the MatSortable's disable clear input.
         * @return {?}
         */
        get disableClear() { return this._disableClear; }
        /**
         * @param {?} v
         * @return {?}
         */
        set disableClear(v) { this._disableClear = coerceBooleanProperty(v); }
        /**
         * Register function to be used by the contained MatSortables. Adds the MatSortable to the
         * collection of MatSortables.
         * @param {?} sortable
         * @return {?}
         */
        register(sortable) {
            if (!sortable.id) {
                throw getSortHeaderMissingIdError();
            }
            if (this.sortables.has(sortable.id)) {
                throw getSortDuplicateSortableIdError(sortable.id);
            }
            this.sortables.set(sortable.id, sortable);
        }
        /**
         * Unregister function to be used by the contained MatSortables. Removes the MatSortable from the
         * collection of contained MatSortables.
         * @param {?} sortable
         * @return {?}
         */
        deregister(sortable) {
            this.sortables.delete(sortable.id);
        }
        /**
         * Sets the active sort id and determines the new sort direction.
         * @param {?} sortable
         * @return {?}
         */
        sort(sortable) {
            if (this.active != sortable.id) {
                this.active = sortable.id;
                this.direction = sortable.start ? sortable.start : this.start;
            }
            else {
                this.direction = this.getNextSortDirection(sortable);
            }
            this.sortChange.emit({ active: this.active, direction: this.direction });
        }
        /**
         * Returns the next sort direction of the active sortable, checking for potential overrides.
         * @param {?} sortable
         * @return {?}
         */
        getNextSortDirection(sortable) {
            if (!sortable) {
                return '';
            }
            // Get the sort direction cycle with the potential sortable overrides.
            /** @type {?} */
            const disableClear = sortable.disableClear != null ? sortable.disableClear : this.disableClear;
            /** @type {?} */
            let sortDirectionCycle = getSortDirectionCycle(sortable.start || this.start, disableClear);
            // Get and return the next direction in the cycle
            /** @type {?} */
            let nextDirectionIndex = sortDirectionCycle.indexOf(this.direction) + 1;
            if (nextDirectionIndex >= sortDirectionCycle.length) {
                nextDirectionIndex = 0;
            }
            return sortDirectionCycle[nextDirectionIndex];
        }
        /**
         * @return {?}
         */
        ngOnInit() {
            this._markInitialized();
        }
        /**
         * @return {?}
         */
        ngOnChanges() {
            this._stateChanges.next();
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this._stateChanges.complete();
        }
    }
    MatSort.ɵfac = function MatSort_Factory(t) { return ɵMatSort_BaseFactory(t || MatSort); };
    MatSort.ɵdir = ɵɵdefineDirective({ type: MatSort, selectors: [["", "matSort", ""]], hostAttrs: [1, "mat-sort"], inputs: { disabled: ["matSortDisabled", "disabled"], start: ["matSortStart", "start"], direction: ["matSortDirection", "direction"], disableClear: ["matSortDisableClear", "disableClear"], active: ["matSortActive", "active"] }, outputs: { sortChange: "matSortChange" }, exportAs: ["matSort"], features: [ɵɵInheritDefinitionFeature, ɵɵNgOnChangesFeature] });
    MatSort.propDecorators = {
        active: [{ type: Input, args: ['matSortActive',] }],
        start: [{ type: Input, args: ['matSortStart',] }],
        direction: [{ type: Input, args: ['matSortDirection',] }],
        disableClear: [{ type: Input, args: ['matSortDisableClear',] }],
        sortChange: [{ type: Output, args: ['matSortChange',] }]
    };
    const ɵMatSort_BaseFactory = ɵɵgetInheritedFactory(MatSort);
    /**
     * Returns the sort direction cycle to use given the provided parameters of order and clear.
     * @param {?} start
     * @param {?} disableClear
     * @return {?}
     */
    function getSortDirectionCycle(start, disableClear) {
        /** @type {?} */
        let sortOrder = ['asc', 'desc'];
        if (start == 'desc') {
            sortOrder.reverse();
        }
        if (!disableClear) {
            sortOrder.push('');
        }
        return sortOrder;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/sort/sort-animations.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const SORT_ANIMATION_TRANSITION = AnimationDurations.ENTERING + ' ' +
        AnimationCurves.STANDARD_CURVE;
    /**
     * Animations used by MatSort.
     * \@docs-private
     * @type {?}
     */
    const matSortAnimations = {
        /**
         * Animation that moves the sort indicator.
         */
        indicator: trigger('indicator', [
            state('active-asc, asc', style({ transform: 'translateY(0px)' })),
            // 10px is the height of the sort indicator, minus the width of the pointers
            state('active-desc, desc', style({ transform: 'translateY(10px)' })),
            transition('active-asc <=> active-desc', animate(SORT_ANIMATION_TRANSITION))
        ]),
        /**
         * Animation that rotates the left pointer of the indicator based on the sorting direction.
         */
        leftPointer: trigger('leftPointer', [
            state('active-asc, asc', style({ transform: 'rotate(-45deg)' })),
            state('active-desc, desc', style({ transform: 'rotate(45deg)' })),
            transition('active-asc <=> active-desc', animate(SORT_ANIMATION_TRANSITION))
        ]),
        /**
         * Animation that rotates the right pointer of the indicator based on the sorting direction.
         */
        rightPointer: trigger('rightPointer', [
            state('active-asc, asc', style({ transform: 'rotate(45deg)' })),
            state('active-desc, desc', style({ transform: 'rotate(-45deg)' })),
            transition('active-asc <=> active-desc', animate(SORT_ANIMATION_TRANSITION))
        ]),
        /**
         * Animation that controls the arrow opacity.
         */
        arrowOpacity: trigger('arrowOpacity', [
            state('desc-to-active, asc-to-active, active', style({ opacity: 1 })),
            state('desc-to-hint, asc-to-hint, hint', style({ opacity: .54 })),
            state('hint-to-desc, active-to-desc, desc, hint-to-asc, active-to-asc, asc, void', style({ opacity: 0 })),
            // Transition between all states except for immediate transitions
            transition('* => asc, * => desc, * => active, * => hint, * => void', animate('0ms')),
            transition('* <=> *', animate(SORT_ANIMATION_TRANSITION)),
        ]),
        /**
         * Animation for the translation of the arrow as a whole. States are separated into two
         * groups: ones with animations and others that are immediate. Immediate states are asc, desc,
         * peek, and active. The other states define a specific animation (source-to-destination)
         * and are determined as a function of their prev user-perceived state and what the next state
         * should be.
         */
        arrowPosition: trigger('arrowPosition', [
            // Hidden Above => Hint Center
            transition('* => desc-to-hint, * => desc-to-active', animate(SORT_ANIMATION_TRANSITION, keyframes([
                style({ transform: 'translateY(-25%)' }),
                style({ transform: 'translateY(0)' })
            ]))),
            // Hint Center => Hidden Below
            transition('* => hint-to-desc, * => active-to-desc', animate(SORT_ANIMATION_TRANSITION, keyframes([
                style({ transform: 'translateY(0)' }),
                style({ transform: 'translateY(25%)' })
            ]))),
            // Hidden Below => Hint Center
            transition('* => asc-to-hint, * => asc-to-active', animate(SORT_ANIMATION_TRANSITION, keyframes([
                style({ transform: 'translateY(25%)' }),
                style({ transform: 'translateY(0)' })
            ]))),
            // Hint Center => Hidden Above
            transition('* => hint-to-asc, * => active-to-asc', animate(SORT_ANIMATION_TRANSITION, keyframes([
                style({ transform: 'translateY(0)' }),
                style({ transform: 'translateY(-25%)' })
            ]))),
            state('desc-to-hint, asc-to-hint, hint, desc-to-active, asc-to-active, active', style({ transform: 'translateY(0)' })),
            state('hint-to-desc, active-to-desc, desc', style({ transform: 'translateY(-25%)' })),
            state('hint-to-asc, active-to-asc, asc', style({ transform: 'translateY(25%)' })),
        ]),
        /**
         * Necessary trigger that calls animate on children animations.
         */
        allowChildren: trigger('allowChildren', [
            transition('* <=> *', [
                query('@*', animateChild(), { optional: true })
            ])
        ]),
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/sort/sort-header-intl.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * To modify the labels and text displayed, create a new instance of MatSortHeaderIntl and
     * include it in a custom provider.
     */
    class MatSortHeaderIntl {
        constructor() {
            /**
             * Stream that emits whenever the labels here are changed. Use this to notify
             * components if the labels have changed after initialization.
             */
            this.changes = new Subject();
            /**
             * ARIA label for the sorting button.
             */
            this.sortButtonLabel = (/**
             * @param {?} id
             * @return {?}
             */
            (id) => {
                return `Change sorting for ${id}`;
            });
        }
    }
    MatSortHeaderIntl.ɵfac = function MatSortHeaderIntl_Factory(t) { return new (t || MatSortHeaderIntl)(); };
    /** @nocollapse */ MatSortHeaderIntl.ɵprov = ɵɵdefineInjectable({ factory: function MatSortHeaderIntl_Factory() { return new MatSortHeaderIntl(); }, token: MatSortHeaderIntl, providedIn: "root" });
    /**
     * \@docs-private
     * @param {?} parentIntl
     * @return {?}
     */
    function MAT_SORT_HEADER_INTL_PROVIDER_FACTORY(parentIntl) {
        return parentIntl || new MatSortHeaderIntl();
    }
    /**
     * \@docs-private
     * @type {?}
     */
    const MAT_SORT_HEADER_INTL_PROVIDER = {
        // If there is already an MatSortHeaderIntl available, use that. Otherwise, provide a new one.
        provide: MatSortHeaderIntl,
        deps: [[new Optional(), new SkipSelf(), MatSortHeaderIntl]],
        useFactory: MAT_SORT_HEADER_INTL_PROVIDER_FACTORY
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/sort/sort-header.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    // Boilerplate for applying mixins to the sort header.
    /**
     * \@docs-private
     */
    class MatSortHeaderBase {
    }
    /** @type {?} */
    const _MatSortHeaderMixinBase = mixinDisabled(MatSortHeaderBase);
    /**
     * Applies sorting behavior (click to change sort) and styles to an element, including an
     * arrow to display the current sort direction.
     *
     * Must be provided with an id and contained within a parent MatSort directive.
     *
     * If used on header cells in a CdkTable, it will automatically default its id from its containing
     * column definition.
     */
    class MatSortHeader extends _MatSortHeaderMixinBase {
        /**
         * @param {?} _intl
         * @param {?} changeDetectorRef
         * @param {?} _sort
         * @param {?} _columnDef
         * @param {?=} _focusMonitor
         * @param {?=} _elementRef
         */
        constructor(_intl, changeDetectorRef, _sort, _columnDef, _focusMonitor, _elementRef) {
            // Note that we use a string token for the `_columnDef`, because the value is provided both by
            // `material/table` and `cdk/table` and we can't have the CDK depending on Material,
            // and we want to avoid having the sort header depending on the CDK table because
            // of this single reference.
            super();
            this._intl = _intl;
            this._sort = _sort;
            this._columnDef = _columnDef;
            this._focusMonitor = _focusMonitor;
            this._elementRef = _elementRef;
            /**
             * Flag set to true when the indicator should be displayed while the sort is not active. Used to
             * provide an affordance that the header is sortable by showing on focus and hover.
             */
            this._showIndicatorHint = false;
            /**
             * The direction the arrow should be facing according to the current state.
             */
            this._arrowDirection = '';
            /**
             * Whether the view state animation should show the transition between the `from` and `to` states.
             */
            this._disableViewStateAnimation = false;
            /**
             * Sets the position of the arrow that displays when sorted.
             */
            this.arrowPosition = 'after';
            if (!_sort) {
                throw getSortHeaderNotContainedWithinSortError();
            }
            this._rerenderSubscription = merge(_sort.sortChange, _sort._stateChanges, _intl.changes)
                .subscribe((/**
             * @return {?}
             */
            () => {
                if (this._isSorted()) {
                    this._updateArrowDirection();
                }
                // If this header was recently active and now no longer sorted, animate away the arrow.
                if (!this._isSorted() && this._viewState && this._viewState.toState === 'active') {
                    this._disableViewStateAnimation = false;
                    this._setAnimationTransitionState({ fromState: 'active', toState: this._arrowDirection });
                }
                changeDetectorRef.markForCheck();
            }));
            if (_focusMonitor && _elementRef) {
                // We use the focus monitor because we also want to style
                // things differently based on the focus origin.
                _focusMonitor.monitor(_elementRef, true)
                    .subscribe((/**
                 * @param {?} origin
                 * @return {?}
                 */
                origin => this._setIndicatorHintVisible(!!origin)));
            }
        }
        /**
         * Overrides the disable clear value of the containing MatSort for this MatSortable.
         * @return {?}
         */
        get disableClear() { return this._disableClear; }
        /**
         * @param {?} v
         * @return {?}
         */
        set disableClear(v) { this._disableClear = coerceBooleanProperty(v); }
        /**
         * @return {?}
         */
        ngOnInit() {
            if (!this.id && this._columnDef) {
                this.id = this._columnDef.name;
            }
            // Initialize the direction of the arrow and set the view state to be immediately that state.
            this._updateArrowDirection();
            this._setAnimationTransitionState({ toState: this._isSorted() ? 'active' : this._arrowDirection });
            this._sort.register(this);
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            // @breaking-change 10.0.0 Remove null check for _focusMonitor and _elementRef.
            if (this._focusMonitor && this._elementRef) {
                this._focusMonitor.stopMonitoring(this._elementRef);
            }
            this._sort.deregister(this);
            this._rerenderSubscription.unsubscribe();
        }
        /**
         * Sets the "hint" state such that the arrow will be semi-transparently displayed as a hint to the
         * user showing what the active sort will become. If set to false, the arrow will fade away.
         * @param {?} visible
         * @return {?}
         */
        _setIndicatorHintVisible(visible) {
            // No-op if the sort header is disabled - should not make the hint visible.
            if (this._isDisabled() && visible) {
                return;
            }
            this._showIndicatorHint = visible;
            if (!this._isSorted()) {
                this._updateArrowDirection();
                if (this._showIndicatorHint) {
                    this._setAnimationTransitionState({ fromState: this._arrowDirection, toState: 'hint' });
                }
                else {
                    this._setAnimationTransitionState({ fromState: 'hint', toState: this._arrowDirection });
                }
            }
        }
        /**
         * Sets the animation transition view state for the arrow's position and opacity. If the
         * `disableViewStateAnimation` flag is set to true, the `fromState` will be ignored so that
         * no animation appears.
         * @param {?} viewState
         * @return {?}
         */
        _setAnimationTransitionState(viewState) {
            this._viewState = viewState;
            // If the animation for arrow position state (opacity/translation) should be disabled,
            // remove the fromState so that it jumps right to the toState.
            if (this._disableViewStateAnimation) {
                this._viewState = { toState: viewState.toState };
            }
        }
        /**
         * Triggers the sort on this sort header and removes the indicator hint.
         * @return {?}
         */
        _handleClick() {
            if (this._isDisabled()) {
                return;
            }
            this._sort.sort(this);
            // Do not show the animation if the header was already shown in the right position.
            if (this._viewState.toState === 'hint' || this._viewState.toState === 'active') {
                this._disableViewStateAnimation = true;
            }
            // If the arrow is now sorted, animate the arrow into place. Otherwise, animate it away into
            // the direction it is facing.
            /** @type {?} */
            const viewState = this._isSorted() ?
                { fromState: this._arrowDirection, toState: 'active' } :
                { fromState: 'active', toState: this._arrowDirection };
            this._setAnimationTransitionState(viewState);
            this._showIndicatorHint = false;
        }
        /**
         * Whether this MatSortHeader is currently sorted in either ascending or descending order.
         * @return {?}
         */
        _isSorted() {
            return this._sort.active == this.id &&
                (this._sort.direction === 'asc' || this._sort.direction === 'desc');
        }
        /**
         * Returns the animation state for the arrow direction (indicator and pointers).
         * @return {?}
         */
        _getArrowDirectionState() {
            return `${this._isSorted() ? 'active-' : ''}${this._arrowDirection}`;
        }
        /**
         * Returns the arrow position state (opacity, translation).
         * @return {?}
         */
        _getArrowViewState() {
            /** @type {?} */
            const fromState = this._viewState.fromState;
            return (fromState ? `${fromState}-to-` : '') + this._viewState.toState;
        }
        /**
         * Updates the direction the arrow should be pointing. If it is not sorted, the arrow should be
         * facing the start direction. Otherwise if it is sorted, the arrow should point in the currently
         * active sorted direction. The reason this is updated through a function is because the direction
         * should only be changed at specific times - when deactivated but the hint is displayed and when
         * the sort is active and the direction changes. Otherwise the arrow's direction should linger
         * in cases such as the sort becoming deactivated but we want to animate the arrow away while
         * preserving its direction, even though the next sort direction is actually different and should
         * only be changed once the arrow displays again (hint or activation).
         * @return {?}
         */
        _updateArrowDirection() {
            this._arrowDirection = this._isSorted() ?
                this._sort.direction :
                (this.start || this._sort.start);
        }
        /**
         * @return {?}
         */
        _isDisabled() {
            return this._sort.disabled || this.disabled;
        }
        /**
         * Gets the aria-sort attribute that should be applied to this sort header. If this header
         * is not sorted, returns null so that the attribute is removed from the host element. Aria spec
         * says that the aria-sort property should only be present on one header at a time, so removing
         * ensures this is true.
         * @return {?}
         */
        _getAriaSortAttribute() {
            if (!this._isSorted()) {
                return null;
            }
            return this._sort.direction == 'asc' ? 'ascending' : 'descending';
        }
        /**
         * Whether the arrow inside the sort header should be rendered.
         * @return {?}
         */
        _renderArrow() {
            return !this._isDisabled() || this._isSorted();
        }
    }
    MatSortHeader.ɵfac = function MatSortHeader_Factory(t) { return new (t || MatSortHeader)(ɵɵdirectiveInject(MatSortHeaderIntl), ɵɵdirectiveInject(ChangeDetectorRef), ɵɵdirectiveInject(MatSort, 8), ɵɵdirectiveInject('MAT_SORT_HEADER_COLUMN_DEF', 8), ɵɵdirectiveInject(FocusMonitor), ɵɵdirectiveInject(ElementRef)); };
    MatSortHeader.ɵcmp = ɵɵdefineComponent({ type: MatSortHeader, selectors: [["", "mat-sort-header", ""]], hostAttrs: [1, "mat-sort-header"], hostVars: 3, hostBindings: function MatSortHeader_HostBindings(rf, ctx) { if (rf & 1) {
            ɵɵlistener("click", function MatSortHeader_click_HostBindingHandler() { return ctx._handleClick(); })("mouseenter", function MatSortHeader_mouseenter_HostBindingHandler() { return ctx._setIndicatorHintVisible(true); })("mouseleave", function MatSortHeader_mouseleave_HostBindingHandler() { return ctx._setIndicatorHintVisible(false); });
        } if (rf & 2) {
            ɵɵattribute("aria-sort", ctx._getAriaSortAttribute());
            ɵɵclassProp("mat-sort-header-disabled", ctx._isDisabled());
        } }, inputs: { disabled: "disabled", arrowPosition: "arrowPosition", disableClear: "disableClear", id: ["mat-sort-header", "id"], start: "start" }, exportAs: ["matSortHeader"], features: [ɵɵInheritDefinitionFeature], attrs: _c0$l, ngContentSelectors: _c1$c, decls: 4, vars: 7, consts: [[1, "mat-sort-header-container"], ["type", "button", 1, "mat-sort-header-button", "mat-focus-indicator"], ["class", "mat-sort-header-arrow", 4, "ngIf"], [1, "mat-sort-header-arrow"], [1, "mat-sort-header-stem"], [1, "mat-sort-header-indicator"], [1, "mat-sort-header-pointer-left"], [1, "mat-sort-header-pointer-right"], [1, "mat-sort-header-pointer-middle"]], template: function MatSortHeader_Template(rf, ctx) { if (rf & 1) {
            ɵɵprojectionDef();
            ɵɵelementStart(0, "div", 0);
            ɵɵelementStart(1, "button", 1);
            ɵɵprojection(2);
            ɵɵelementEnd();
            ɵɵtemplate(3, MatSortHeader_div_3_Template, 6, 6, "div", 2);
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵclassProp("mat-sort-header-sorted", ctx._isSorted())("mat-sort-header-position-before", ctx.arrowPosition == "before");
            ɵɵadvance(1);
            ɵɵattribute("disabled", ctx._isDisabled() || null)("aria-label", ctx._intl.sortButtonLabel(ctx.id));
            ɵɵadvance(2);
            ɵɵproperty("ngIf", ctx._renderArrow());
        } }, directives: [NgIf], styles: [".mat-sort-header-container{display:flex;cursor:pointer;align-items:center}.mat-sort-header-disabled .mat-sort-header-container{cursor:default}.mat-sort-header-position-before{flex-direction:row-reverse}.mat-sort-header-button{border:none;background:0 0;display:flex;align-items:center;padding:0;cursor:inherit;outline:0;font:inherit;color:currentColor;position:relative}[mat-sort-header].cdk-keyboard-focused .mat-sort-header-button,[mat-sort-header].cdk-program-focused .mat-sort-header-button{border-bottom:solid 1px currentColor}.mat-sort-header-button::-moz-focus-inner{border:0}.mat-sort-header-arrow{height:12px;width:12px;min-width:12px;position:relative;display:flex;opacity:0}.mat-sort-header-arrow,[dir=rtl] .mat-sort-header-position-before .mat-sort-header-arrow{margin:0 0 0 6px}.mat-sort-header-position-before .mat-sort-header-arrow,[dir=rtl] .mat-sort-header-arrow{margin:0 6px 0 0}.mat-sort-header-stem{background:currentColor;height:10px;width:2px;margin:auto;display:flex;align-items:center}.cdk-high-contrast-active .mat-sort-header-stem{width:0;border-left:solid 2px}.mat-sort-header-indicator{width:100%;height:2px;display:flex;align-items:center;position:absolute;top:0;left:0}.mat-sort-header-pointer-middle{margin:auto;height:2px;width:2px;background:currentColor;transform:rotate(45deg)}.cdk-high-contrast-active .mat-sort-header-pointer-middle{width:0;height:0;border-top:solid 2px;border-left:solid 2px}.mat-sort-header-pointer-left,.mat-sort-header-pointer-right{background:currentColor;width:6px;height:2px;position:absolute;top:0}.cdk-high-contrast-active .mat-sort-header-pointer-left,.cdk-high-contrast-active .mat-sort-header-pointer-right{width:0;height:0;border-left:solid 6px;border-top:solid 2px}.mat-sort-header-pointer-left{transform-origin:right;left:0}.mat-sort-header-pointer-right{transform-origin:left;right:0}\n"], encapsulation: 2, data: { animation: [
                matSortAnimations.indicator,
                matSortAnimations.leftPointer,
                matSortAnimations.rightPointer,
                matSortAnimations.arrowOpacity,
                matSortAnimations.arrowPosition,
                matSortAnimations.allowChildren,
            ] }, changeDetection: 0 });
    /** @nocollapse */
    MatSortHeader.ctorParameters = () => [
        { type: MatSortHeaderIntl },
        { type: ChangeDetectorRef },
        { type: MatSort, decorators: [{ type: Optional }] },
        { type: undefined, decorators: [{ type: Inject, args: ['MAT_SORT_HEADER_COLUMN_DEF',] }, { type: Optional }] },
        { type: FocusMonitor },
        { type: ElementRef }
    ];
    MatSortHeader.propDecorators = {
        id: [{ type: Input, args: ['mat-sort-header',] }],
        arrowPosition: [{ type: Input }],
        start: [{ type: Input }],
        disableClear: [{ type: Input }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/sort/sort-module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class MatSortModule {
    }
    MatSortModule.ɵmod = ɵɵdefineNgModule({ type: MatSortModule });
    MatSortModule.ɵinj = ɵɵdefineInjector({ factory: function MatSortModule_Factory(t) { return new (t || MatSortModule)(); }, providers: [MAT_SORT_HEADER_INTL_PROVIDER], imports: [[CommonModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(MatSortModule, { declarations: function () { return [MatSort,
            MatSortHeader]; }, imports: function () { return [CommonModule]; }, exports: function () { return [MatSort,
            MatSortHeader]; } }); })();

    const _c0$m = [[["caption"]]];
    const _c1$d = ["caption"];
    function CdkTextColumn_th_1_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "th", 3);
        ɵɵtext(1);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r0 = ɵɵnextContext();
        ɵɵstyleProp("text-align", ctx_r0.justify);
        ɵɵadvance(1);
        ɵɵtextInterpolate1(" ", ctx_r0.headerText, " ");
    } }
    function CdkTextColumn_td_2_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "td", 4);
        ɵɵtext(1);
        ɵɵelementEnd();
    } if (rf & 2) {
        const data_r2 = ctx.$implicit;
        const ctx_r1 = ɵɵnextContext();
        ɵɵstyleProp("text-align", ctx_r1.justify);
        ɵɵadvance(1);
        ɵɵtextInterpolate1(" ", ctx_r1.dataAccessor(data_r2, ctx_r1.name), " ");
    } }
    /**
     * Mixin to provide a directive with a function that checks if the sticky input has been
     * changed since the last time the function was called. Essentially adds a dirty-check to the
     * sticky value.
     * \@docs-private
     * @template T
     * @param {?} base
     * @return {?}
     */
    function mixinHasStickyInput(base) {
        return class extends base {
            /**
             * @param {...?} args
             */
            constructor(...args) {
                super(...args);
                this._sticky = false;
                /**
                 * Whether the sticky input has changed since it was last checked.
                 */
                this._hasStickyChanged = false;
            }
            /**
             * Whether sticky positioning should be applied.
             * @return {?}
             */
            get sticky() { return this._sticky; }
            /**
             * @param {?} v
             * @return {?}
             */
            set sticky(v) {
                /** @type {?} */
                const prevValue = this._sticky;
                this._sticky = coerceBooleanProperty(v);
                this._hasStickyChanged = prevValue !== this._sticky;
            }
            /**
             * Whether the sticky value has changed since this was last called.
             * @return {?}
             */
            hasStickyChanged() {
                /** @type {?} */
                const hasStickyChanged = this._hasStickyChanged;
                this._hasStickyChanged = false;
                return hasStickyChanged;
            }
            /**
             * Resets the dirty check for cases where the sticky state has been used without checking.
             * @return {?}
             */
            resetStickyChanged() {
                this._hasStickyChanged = false;
            }
        };
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/table/tokens.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Used to provide a table to some of the sub-components without causing a circular dependency.
     * \@docs-private
     * @type {?}
     */
    const CDK_TABLE = new InjectionToken('CDK_TABLE');
    /**
     * Injection token that can be used to specify the text column options.
     * @type {?}
     */
    const TEXT_COLUMN_OPTIONS = new InjectionToken('text-column-options');
    /**
     * Cell definition for a CDK table.
     * Captures the template of a column's data row cell as well as cell-specific properties.
     */
    class CdkCellDef {
        /**
         * @param {?} template
         */
        constructor(/** @docs-private */ template) {
            this.template = template;
        }
    }
    CdkCellDef.ɵfac = function CdkCellDef_Factory(t) { return new (t || CdkCellDef)(ɵɵdirectiveInject(TemplateRef)); };
    CdkCellDef.ɵdir = ɵɵdefineDirective({ type: CdkCellDef, selectors: [["", "cdkCellDef", ""]] });
    /** @nocollapse */
    CdkCellDef.ctorParameters = () => [
        { type: TemplateRef }
    ];
    /**
     * Header cell definition for a CDK table.
     * Captures the template of a column's header cell and as well as cell-specific properties.
     */
    class CdkHeaderCellDef {
        /**
         * @param {?} template
         */
        constructor(/** @docs-private */ template) {
            this.template = template;
        }
    }
    CdkHeaderCellDef.ɵfac = function CdkHeaderCellDef_Factory(t) { return new (t || CdkHeaderCellDef)(ɵɵdirectiveInject(TemplateRef)); };
    CdkHeaderCellDef.ɵdir = ɵɵdefineDirective({ type: CdkHeaderCellDef, selectors: [["", "cdkHeaderCellDef", ""]] });
    /** @nocollapse */
    CdkHeaderCellDef.ctorParameters = () => [
        { type: TemplateRef }
    ];
    /**
     * Footer cell definition for a CDK table.
     * Captures the template of a column's footer cell and as well as cell-specific properties.
     */
    class CdkFooterCellDef {
        /**
         * @param {?} template
         */
        constructor(/** @docs-private */ template) {
            this.template = template;
        }
    }
    CdkFooterCellDef.ɵfac = function CdkFooterCellDef_Factory(t) { return new (t || CdkFooterCellDef)(ɵɵdirectiveInject(TemplateRef)); };
    CdkFooterCellDef.ɵdir = ɵɵdefineDirective({ type: CdkFooterCellDef, selectors: [["", "cdkFooterCellDef", ""]] });
    /** @nocollapse */
    CdkFooterCellDef.ctorParameters = () => [
        { type: TemplateRef }
    ];
    // Boilerplate for applying mixins to CdkColumnDef.
    /**
     * \@docs-private
     */
    class CdkColumnDefBase {
    }
    /** @type {?} */
    const _CdkColumnDefBase = mixinHasStickyInput(CdkColumnDefBase);
    /**
     * Column definition for the CDK table.
     * Defines a set of cells available for a table column.
     */
    class CdkColumnDef extends _CdkColumnDefBase {
        /**
         * @param {?=} _table
         */
        constructor(_table) {
            super();
            this._table = _table;
            this._stickyEnd = false;
        }
        /**
         * Unique name for this column.
         * @return {?}
         */
        get name() {
            return this._name;
        }
        /**
         * @param {?} name
         * @return {?}
         */
        set name(name) {
            // If the directive is set without a name (updated programatically), then this setter will
            // trigger with an empty string and should not overwrite the programatically set value.
            if (name) {
                this._name = name;
                this.cssClassFriendlyName = name.replace(/[^a-z0-9_-]/ig, '-');
            }
        }
        /**
         * Whether this column should be sticky positioned on the end of the row. Should make sure
         * that it mimics the `CanStick` mixin such that `_hasStickyChanged` is set to true if the value
         * has been changed.
         * @return {?}
         */
        get stickyEnd() {
            return this._stickyEnd;
        }
        /**
         * @param {?} v
         * @return {?}
         */
        set stickyEnd(v) {
            /** @type {?} */
            const prevValue = this._stickyEnd;
            this._stickyEnd = coerceBooleanProperty(v);
            this._hasStickyChanged = prevValue !== this._stickyEnd;
        }
    }
    CdkColumnDef.ɵfac = function CdkColumnDef_Factory(t) { return new (t || CdkColumnDef)(ɵɵdirectiveInject(CDK_TABLE, 8)); };
    CdkColumnDef.ɵdir = ɵɵdefineDirective({ type: CdkColumnDef, selectors: [["", "cdkColumnDef", ""]], contentQueries: function CdkColumnDef_ContentQueries(rf, ctx, dirIndex) { if (rf & 1) {
            ɵɵcontentQuery(dirIndex, CdkCellDef, true);
            ɵɵcontentQuery(dirIndex, CdkHeaderCellDef, true);
            ɵɵcontentQuery(dirIndex, CdkFooterCellDef, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.cell = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.headerCell = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.footerCell = _t.first);
        } }, inputs: { sticky: "sticky", name: ["cdkColumnDef", "name"], stickyEnd: "stickyEnd" }, features: [ɵɵProvidersFeature([{ provide: 'MAT_SORT_HEADER_COLUMN_DEF', useExisting: CdkColumnDef }]), ɵɵInheritDefinitionFeature] });
    /** @nocollapse */
    CdkColumnDef.ctorParameters = () => [
        { type: undefined, decorators: [{ type: Inject, args: [CDK_TABLE,] }, { type: Optional }] }
    ];
    CdkColumnDef.propDecorators = {
        name: [{ type: Input, args: ['cdkColumnDef',] }],
        stickyEnd: [{ type: Input, args: ['stickyEnd',] }],
        cell: [{ type: ContentChild, args: [CdkCellDef,] }],
        headerCell: [{ type: ContentChild, args: [CdkHeaderCellDef,] }],
        footerCell: [{ type: ContentChild, args: [CdkFooterCellDef,] }]
    };
    /**
     * Base class for the cells. Adds a CSS classname that identifies the column it renders in.
     */
    class BaseCdkCell {
        /**
         * @param {?} columnDef
         * @param {?} elementRef
         */
        constructor(columnDef, elementRef) {
            /** @type {?} */
            const columnClassName = `cdk-column-${columnDef.cssClassFriendlyName}`;
            elementRef.nativeElement.classList.add(columnClassName);
        }
    }
    /**
     * Header cell template container that adds the right classes and role.
     */
    class CdkHeaderCell extends BaseCdkCell {
        /**
         * @param {?} columnDef
         * @param {?} elementRef
         */
        constructor(columnDef, elementRef) {
            super(columnDef, elementRef);
        }
    }
    CdkHeaderCell.ɵfac = function CdkHeaderCell_Factory(t) { return new (t || CdkHeaderCell)(ɵɵdirectiveInject(CdkColumnDef), ɵɵdirectiveInject(ElementRef)); };
    CdkHeaderCell.ɵdir = ɵɵdefineDirective({ type: CdkHeaderCell, selectors: [["cdk-header-cell"], ["th", "cdk-header-cell", ""]], hostAttrs: ["role", "columnheader", 1, "cdk-header-cell"], features: [ɵɵInheritDefinitionFeature] });
    /** @nocollapse */
    CdkHeaderCell.ctorParameters = () => [
        { type: CdkColumnDef },
        { type: ElementRef }
    ];
    /**
     * Footer cell template container that adds the right classes and role.
     */
    class CdkFooterCell extends BaseCdkCell {
        /**
         * @param {?} columnDef
         * @param {?} elementRef
         */
        constructor(columnDef, elementRef) {
            super(columnDef, elementRef);
        }
    }
    CdkFooterCell.ɵfac = function CdkFooterCell_Factory(t) { return new (t || CdkFooterCell)(ɵɵdirectiveInject(CdkColumnDef), ɵɵdirectiveInject(ElementRef)); };
    CdkFooterCell.ɵdir = ɵɵdefineDirective({ type: CdkFooterCell, selectors: [["cdk-footer-cell"], ["td", "cdk-footer-cell", ""]], hostAttrs: ["role", "gridcell", 1, "cdk-footer-cell"], features: [ɵɵInheritDefinitionFeature] });
    /** @nocollapse */
    CdkFooterCell.ctorParameters = () => [
        { type: CdkColumnDef },
        { type: ElementRef }
    ];
    /**
     * Cell template container that adds the right classes and role.
     */
    class CdkCell extends BaseCdkCell {
        /**
         * @param {?} columnDef
         * @param {?} elementRef
         */
        constructor(columnDef, elementRef) {
            super(columnDef, elementRef);
        }
    }
    CdkCell.ɵfac = function CdkCell_Factory(t) { return new (t || CdkCell)(ɵɵdirectiveInject(CdkColumnDef), ɵɵdirectiveInject(ElementRef)); };
    CdkCell.ɵdir = ɵɵdefineDirective({ type: CdkCell, selectors: [["cdk-cell"], ["td", "cdk-cell", ""]], hostAttrs: ["role", "gridcell", 1, "cdk-cell"], features: [ɵɵInheritDefinitionFeature] });
    /** @nocollapse */
    CdkCell.ctorParameters = () => [
        { type: CdkColumnDef },
        { type: ElementRef }
    ];
    /**
     * Base class for the CdkHeaderRowDef and CdkRowDef that handles checking their columns inputs
     * for changes and notifying the table.
     * @abstract
     */
    class BaseRowDef {
        /**
         * @param {?} template
         * @param {?} _differs
         */
        constructor(template, _differs) {
            this.template = template;
            this._differs = _differs;
        }
        /**
         * @param {?} changes
         * @return {?}
         */
        ngOnChanges(changes) {
            // Create a new columns differ if one does not yet exist. Initialize it based on initial value
            // of the columns property or an empty array if none is provided.
            if (!this._columnsDiffer) {
                /** @type {?} */
                const columns = (changes['columns'] && changes['columns'].currentValue) || [];
                this._columnsDiffer = this._differs.find(columns).create();
                this._columnsDiffer.diff(columns);
            }
        }
        /**
         * Returns the difference between the current columns and the columns from the last diff, or null
         * if there is no difference.
         * @return {?}
         */
        getColumnsDiff() {
            return this._columnsDiffer.diff(this.columns);
        }
        /**
         * Gets this row def's relevant cell template from the provided column def.
         * @param {?} column
         * @return {?}
         */
        extractCellTemplate(column) {
            if (this instanceof CdkHeaderRowDef) {
                return column.headerCell.template;
            }
            if (this instanceof CdkFooterRowDef) {
                return column.footerCell.template;
            }
            else {
                return column.cell.template;
            }
        }
    }
    BaseRowDef.ɵfac = function BaseRowDef_Factory(t) { return new (t || BaseRowDef)(ɵɵdirectiveInject(TemplateRef), ɵɵdirectiveInject(IterableDiffers)); };
    BaseRowDef.ɵdir = ɵɵdefineDirective({ type: BaseRowDef, features: [ɵɵNgOnChangesFeature] });
    /** @nocollapse */
    BaseRowDef.ctorParameters = () => [
        { type: TemplateRef },
        { type: IterableDiffers }
    ];
    // Boilerplate for applying mixins to CdkHeaderRowDef.
    /**
     * \@docs-private
     */
    class CdkHeaderRowDefBase extends BaseRowDef {
    }
    /** @type {?} */
    const _CdkHeaderRowDefBase = mixinHasStickyInput(CdkHeaderRowDefBase);
    /**
     * Header row definition for the CDK table.
     * Captures the header row's template and other header properties such as the columns to display.
     */
    class CdkHeaderRowDef extends _CdkHeaderRowDefBase {
        /**
         * @param {?} template
         * @param {?} _differs
         * @param {?=} _table
         */
        constructor(template, _differs, _table) {
            super(template, _differs);
            this._table = _table;
        }
        // Prerender fails to recognize that ngOnChanges in a part of this class through inheritance.
        // Explicitly define it so that the method is called as part of the Angular lifecycle.
        /**
         * @param {?} changes
         * @return {?}
         */
        ngOnChanges(changes) {
            super.ngOnChanges(changes);
        }
    }
    CdkHeaderRowDef.ɵfac = function CdkHeaderRowDef_Factory(t) { return new (t || CdkHeaderRowDef)(ɵɵdirectiveInject(TemplateRef), ɵɵdirectiveInject(IterableDiffers), ɵɵdirectiveInject(CDK_TABLE, 8)); };
    CdkHeaderRowDef.ɵdir = ɵɵdefineDirective({ type: CdkHeaderRowDef, selectors: [["", "cdkHeaderRowDef", ""]], inputs: { columns: ["cdkHeaderRowDef", "columns"], sticky: ["cdkHeaderRowDefSticky", "sticky"] }, features: [ɵɵInheritDefinitionFeature, ɵɵNgOnChangesFeature] });
    /** @nocollapse */
    CdkHeaderRowDef.ctorParameters = () => [
        { type: TemplateRef },
        { type: IterableDiffers },
        { type: undefined, decorators: [{ type: Inject, args: [CDK_TABLE,] }, { type: Optional }] }
    ];
    // Boilerplate for applying mixins to CdkFooterRowDef.
    /**
     * \@docs-private
     */
    class CdkFooterRowDefBase extends BaseRowDef {
    }
    /** @type {?} */
    const _CdkFooterRowDefBase = mixinHasStickyInput(CdkFooterRowDefBase);
    /**
     * Footer row definition for the CDK table.
     * Captures the footer row's template and other footer properties such as the columns to display.
     */
    class CdkFooterRowDef extends _CdkFooterRowDefBase {
        /**
         * @param {?} template
         * @param {?} _differs
         * @param {?=} _table
         */
        constructor(template, _differs, _table) {
            super(template, _differs);
            this._table = _table;
        }
        // Prerender fails to recognize that ngOnChanges in a part of this class through inheritance.
        // Explicitly define it so that the method is called as part of the Angular lifecycle.
        /**
         * @param {?} changes
         * @return {?}
         */
        ngOnChanges(changes) {
            super.ngOnChanges(changes);
        }
    }
    CdkFooterRowDef.ɵfac = function CdkFooterRowDef_Factory(t) { return new (t || CdkFooterRowDef)(ɵɵdirectiveInject(TemplateRef), ɵɵdirectiveInject(IterableDiffers), ɵɵdirectiveInject(CDK_TABLE, 8)); };
    CdkFooterRowDef.ɵdir = ɵɵdefineDirective({ type: CdkFooterRowDef, selectors: [["", "cdkFooterRowDef", ""]], inputs: { columns: ["cdkFooterRowDef", "columns"], sticky: ["cdkFooterRowDefSticky", "sticky"] }, features: [ɵɵInheritDefinitionFeature, ɵɵNgOnChangesFeature] });
    /** @nocollapse */
    CdkFooterRowDef.ctorParameters = () => [
        { type: TemplateRef },
        { type: IterableDiffers },
        { type: undefined, decorators: [{ type: Inject, args: [CDK_TABLE,] }, { type: Optional }] }
    ];
    /**
     * Data row definition for the CDK table.
     * Captures the header row's template and other row properties such as the columns to display and
     * a when predicate that describes when this row should be used.
     * @template T
     */
    class CdkRowDef extends BaseRowDef {
        // TODO(andrewseguin): Add an input for providing a switch function to determine
        //   if this template should be used.
        /**
         * @param {?} template
         * @param {?} _differs
         * @param {?=} _table
         */
        constructor(template, _differs, _table) {
            super(template, _differs);
            this._table = _table;
        }
    }
    CdkRowDef.ɵfac = function CdkRowDef_Factory(t) { return new (t || CdkRowDef)(ɵɵdirectiveInject(TemplateRef), ɵɵdirectiveInject(IterableDiffers), ɵɵdirectiveInject(CDK_TABLE, 8)); };
    CdkRowDef.ɵdir = ɵɵdefineDirective({ type: CdkRowDef, selectors: [["", "cdkRowDef", ""]], inputs: { columns: ["cdkRowDefColumns", "columns"], when: ["cdkRowDefWhen", "when"] }, features: [ɵɵInheritDefinitionFeature] });
    /** @nocollapse */
    CdkRowDef.ctorParameters = () => [
        { type: TemplateRef },
        { type: IterableDiffers },
        { type: undefined, decorators: [{ type: Inject, args: [CDK_TABLE,] }, { type: Optional }] }
    ];
    /**
     * Outlet for rendering cells inside of a row or header row.
     * \@docs-private
     */
    class CdkCellOutlet {
        /**
         * @param {?} _viewContainer
         */
        constructor(_viewContainer) {
            this._viewContainer = _viewContainer;
            CdkCellOutlet.mostRecentCellOutlet = this;
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            // If this was the last outlet being rendered in the view, remove the reference
            // from the static property after it has been destroyed to avoid leaking memory.
            if (CdkCellOutlet.mostRecentCellOutlet === this) {
                CdkCellOutlet.mostRecentCellOutlet = null;
            }
        }
    }
    CdkCellOutlet.ɵfac = function CdkCellOutlet_Factory(t) { return new (t || CdkCellOutlet)(ɵɵdirectiveInject(ViewContainerRef)); };
    CdkCellOutlet.ɵdir = ɵɵdefineDirective({ type: CdkCellOutlet, selectors: [["", "cdkCellOutlet", ""]] });
    /**
     * Static property containing the latest constructed instance of this class.
     * Used by the CDK table when each CdkHeaderRow and CdkRow component is created using
     * createEmbeddedView. After one of these components are created, this property will provide
     * a handle to provide that component's cells and context. After init, the CdkCellOutlet will
     * construct the cells with the provided context.
     */
    CdkCellOutlet.mostRecentCellOutlet = null;
    /** @nocollapse */
    CdkCellOutlet.ctorParameters = () => [
        { type: ViewContainerRef }
    ];
    /**
     * Header template container that contains the cell outlet. Adds the right class and role.
     */
    class CdkHeaderRow {
    }
    CdkHeaderRow.ɵfac = function CdkHeaderRow_Factory(t) { return new (t || CdkHeaderRow)(); };
    CdkHeaderRow.ɵcmp = ɵɵdefineComponent({ type: CdkHeaderRow, selectors: [["cdk-header-row"], ["tr", "cdk-header-row", ""]], hostAttrs: ["role", "row", 1, "cdk-header-row"], decls: 1, vars: 0, consts: [["cdkCellOutlet", ""]], template: function CdkHeaderRow_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementContainer(0, 0);
        } }, directives: [CdkCellOutlet], encapsulation: 2 });
    /**
     * Footer template container that contains the cell outlet. Adds the right class and role.
     */
    class CdkFooterRow {
    }
    CdkFooterRow.ɵfac = function CdkFooterRow_Factory(t) { return new (t || CdkFooterRow)(); };
    CdkFooterRow.ɵcmp = ɵɵdefineComponent({ type: CdkFooterRow, selectors: [["cdk-footer-row"], ["tr", "cdk-footer-row", ""]], hostAttrs: ["role", "row", 1, "cdk-footer-row"], decls: 1, vars: 0, consts: [["cdkCellOutlet", ""]], template: function CdkFooterRow_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementContainer(0, 0);
        } }, directives: [CdkCellOutlet], encapsulation: 2 });
    /**
     * Data row template container that contains the cell outlet. Adds the right class and role.
     */
    class CdkRow {
    }
    CdkRow.ɵfac = function CdkRow_Factory(t) { return new (t || CdkRow)(); };
    CdkRow.ɵcmp = ɵɵdefineComponent({ type: CdkRow, selectors: [["cdk-row"], ["tr", "cdk-row", ""]], hostAttrs: ["role", "row", 1, "cdk-row"], decls: 1, vars: 0, consts: [["cdkCellOutlet", ""]], template: function CdkRow_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementContainer(0, 0);
        } }, directives: [CdkCellOutlet], encapsulation: 2 });

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/table/sticky-styler.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google LLC All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * List of all possible directions that can be used for sticky positioning.
     * \@docs-private
     * @type {?}
     */
    const STICKY_DIRECTIONS = ['top', 'bottom', 'left', 'right'];
    /**
     * Applies and removes sticky positioning styles to the `CdkTable` rows and columns cells.
     * \@docs-private
     */
    class StickyStyler {
        /**
         * @param {?} _isNativeHtmlTable Whether the sticky logic should be based on a table
         *     that uses the native `<table>` element.
         * @param {?} _stickCellCss The CSS class that will be applied to every row/cell that has
         *     sticky positioning applied.
         * @param {?} direction The directionality context of the table (ltr/rtl); affects column positioning
         *     by reversing left/right positions.
         * @param {?=} _isBrowser Whether the table is currently being rendered on the server or the client.
         */
        constructor(_isNativeHtmlTable, _stickCellCss, direction, _isBrowser = true) {
            this._isNativeHtmlTable = _isNativeHtmlTable;
            this._stickCellCss = _stickCellCss;
            this.direction = direction;
            this._isBrowser = _isBrowser;
        }
        /**
         * Clears the sticky positioning styles from the row and its cells by resetting the `position`
         * style, setting the zIndex to 0, and unsetting each provided sticky direction.
         * @param {?} rows The list of rows that should be cleared from sticking in the provided directions
         * @param {?} stickyDirections The directions that should no longer be set as sticky on the rows.
         * @return {?}
         */
        clearStickyPositioning(rows, stickyDirections) {
            for (const row of rows) {
                // If the row isn't an element (e.g. if it's an `ng-container`),
                // it won't have inline styles or `children` so we skip it.
                if (row.nodeType !== row.ELEMENT_NODE) {
                    continue;
                }
                this._removeStickyStyle(row, stickyDirections);
                for (let i = 0; i < row.children.length; i++) {
                    /** @type {?} */
                    const cell = (/** @type {?} */ (row.children[i]));
                    this._removeStickyStyle(cell, stickyDirections);
                }
            }
        }
        /**
         * Applies sticky left and right positions to the cells of each row according to the sticky
         * states of the rendered column definitions.
         * @param {?} rows The rows that should have its set of cells stuck according to the sticky states.
         * @param {?} stickyStartStates A list of boolean states where each state represents whether the cell
         *     in this index position should be stuck to the start of the row.
         * @param {?} stickyEndStates A list of boolean states where each state represents whether the cell
         *     in this index position should be stuck to the end of the row.
         * @return {?}
         */
        updateStickyColumns(rows, stickyStartStates, stickyEndStates) {
            /** @type {?} */
            const hasStickyColumns = stickyStartStates.some((/**
             * @param {?} state
             * @return {?}
             */
            state => state)) || stickyEndStates.some((/**
             * @param {?} state
             * @return {?}
             */
            state => state));
            if (!rows.length || !hasStickyColumns || !this._isBrowser) {
                return;
            }
            /** @type {?} */
            const firstRow = rows[0];
            /** @type {?} */
            const numCells = firstRow.children.length;
            /** @type {?} */
            const cellWidths = this._getCellWidths(firstRow);
            /** @type {?} */
            const startPositions = this._getStickyStartColumnPositions(cellWidths, stickyStartStates);
            /** @type {?} */
            const endPositions = this._getStickyEndColumnPositions(cellWidths, stickyEndStates);
            /** @type {?} */
            const isRtl = this.direction === 'rtl';
            for (const row of rows) {
                for (let i = 0; i < numCells; i++) {
                    /** @type {?} */
                    const cell = (/** @type {?} */ (row.children[i]));
                    if (stickyStartStates[i]) {
                        this._addStickyStyle(cell, isRtl ? 'right' : 'left', startPositions[i]);
                    }
                    if (stickyEndStates[i]) {
                        this._addStickyStyle(cell, isRtl ? 'left' : 'right', endPositions[i]);
                    }
                }
            }
        }
        /**
         * Applies sticky positioning to the row's cells if using the native table layout, and to the
         * row itself otherwise.
         * @param {?} rowsToStick The list of rows that should be stuck according to their corresponding
         *     sticky state and to the provided top or bottom position.
         * @param {?} stickyStates A list of boolean states where each state represents whether the row
         *     should be stuck in the particular top or bottom position.
         * @param {?} position The position direction in which the row should be stuck if that row should be
         *     sticky.
         *
         * @return {?}
         */
        stickRows(rowsToStick, stickyStates, position) {
            // Since we can't measure the rows on the server, we can't stick the rows properly.
            if (!this._isBrowser) {
                return;
            }
            // If positioning the rows to the bottom, reverse their order when evaluating the sticky
            // position such that the last row stuck will be "bottom: 0px" and so on.
            /** @type {?} */
            const rows = position === 'bottom' ? rowsToStick.reverse() : rowsToStick;
            /** @type {?} */
            let stickyHeight = 0;
            for (let rowIndex = 0; rowIndex < rows.length; rowIndex++) {
                if (!stickyStates[rowIndex]) {
                    continue;
                }
                /** @type {?} */
                const row = rows[rowIndex];
                if (this._isNativeHtmlTable) {
                    for (let j = 0; j < row.children.length; j++) {
                        /** @type {?} */
                        const cell = (/** @type {?} */ (row.children[j]));
                        this._addStickyStyle(cell, position, stickyHeight);
                    }
                }
                else {
                    // Flex does not respect the stick positioning on the cells, needs to be applied to the row.
                    // If this is applied on a native table, Safari causes the header to fly in wrong direction.
                    this._addStickyStyle(row, position, stickyHeight);
                }
                if (rowIndex === rows.length - 1) {
                    // prevent unnecessary reflow from getBoundingClientRect()
                    return;
                }
                stickyHeight += row.getBoundingClientRect().height;
            }
        }
        /**
         * When using the native table in Safari, sticky footer cells do not stick. The only way to stick
         * footer rows is to apply sticky styling to the tfoot container. This should only be done if
         * all footer rows are sticky. If not all footer rows are sticky, remove sticky positioning from
         * the tfoot element.
         * @param {?} tableElement
         * @param {?} stickyStates
         * @return {?}
         */
        updateStickyFooterContainer(tableElement, stickyStates) {
            if (!this._isNativeHtmlTable) {
                return;
            }
            /** @type {?} */
            const tfoot = (/** @type {?} */ (tableElement.querySelector('tfoot')));
            if (stickyStates.some((/**
             * @param {?} state
             * @return {?}
             */
            state => !state))) {
                this._removeStickyStyle(tfoot, ['bottom']);
            }
            else {
                this._addStickyStyle(tfoot, 'bottom', 0);
            }
        }
        /**
         * Removes the sticky style on the element by removing the sticky cell CSS class, re-evaluating
         * the zIndex, removing each of the provided sticky directions, and removing the
         * sticky position if there are no more directions.
         * @param {?} element
         * @param {?} stickyDirections
         * @return {?}
         */
        _removeStickyStyle(element, stickyDirections) {
            for (const dir of stickyDirections) {
                element.style[dir] = '';
            }
            element.style.zIndex = this._getCalculatedZIndex(element);
            // If the element no longer has any more sticky directions, remove sticky positioning and
            // the sticky CSS class.
            /** @type {?} */
            const hasDirection = STICKY_DIRECTIONS.some((/**
             * @param {?} dir
             * @return {?}
             */
            dir => !!element.style[dir]));
            if (!hasDirection) {
                element.style.position = '';
                element.classList.remove(this._stickCellCss);
            }
        }
        /**
         * Adds the sticky styling to the element by adding the sticky style class, changing position
         * to be sticky (and -webkit-sticky), setting the appropriate zIndex, and adding a sticky
         * direction and value.
         * @param {?} element
         * @param {?} dir
         * @param {?} dirValue
         * @return {?}
         */
        _addStickyStyle(element, dir, dirValue) {
            element.classList.add(this._stickCellCss);
            element.style[dir] = `${dirValue}px`;
            element.style.cssText += 'position: -webkit-sticky; position: sticky; ';
            element.style.zIndex = this._getCalculatedZIndex(element);
        }
        /**
         * Calculate what the z-index should be for the element, depending on what directions (top,
         * bottom, left, right) have been set. It should be true that elements with a top direction
         * should have the highest index since these are elements like a table header. If any of those
         * elements are also sticky in another direction, then they should appear above other elements
         * that are only sticky top (e.g. a sticky column on a sticky header). Bottom-sticky elements
         * (e.g. footer rows) should then be next in the ordering such that they are below the header
         * but above any non-sticky elements. Finally, left/right sticky elements (e.g. sticky columns)
         * should minimally increment so that they are above non-sticky elements but below top and bottom
         * elements.
         * @param {?} element
         * @return {?}
         */
        _getCalculatedZIndex(element) {
            /** @type {?} */
            const zIndexIncrements = {
                top: 100,
                bottom: 10,
                left: 1,
                right: 1,
            };
            /** @type {?} */
            let zIndex = 0;
            // Use `Iterable` instead of `Array` because TypeScript, as of 3.6.3,
            // loses the array generic type in the `for of`. But we *also* have to use `Array` because
            // typescript won't iterate over an `Iterable` unless you compile with `--downlevelIteration`
            for (const dir of (/** @type {?} */ (STICKY_DIRECTIONS))) {
                if (element.style[dir]) {
                    zIndex += zIndexIncrements[dir];
                }
            }
            return zIndex ? `${zIndex}` : '';
        }
        /**
         * Gets the widths for each cell in the provided row.
         * @param {?} row
         * @return {?}
         */
        _getCellWidths(row) {
            /** @type {?} */
            const cellWidths = [];
            /** @type {?} */
            const firstRowCells = row.children;
            for (let i = 0; i < firstRowCells.length; i++) {
                /** @type {?} */
                let cell = (/** @type {?} */ (firstRowCells[i]));
                cellWidths.push(cell.getBoundingClientRect().width);
            }
            return cellWidths;
        }
        /**
         * Determines the left and right positions of each sticky column cell, which will be the
         * accumulation of all sticky column cell widths to the left and right, respectively.
         * Non-sticky cells do not need to have a value set since their positions will not be applied.
         * @param {?} widths
         * @param {?} stickyStates
         * @return {?}
         */
        _getStickyStartColumnPositions(widths, stickyStates) {
            /** @type {?} */
            const positions = [];
            /** @type {?} */
            let nextPosition = 0;
            for (let i = 0; i < widths.length; i++) {
                if (stickyStates[i]) {
                    positions[i] = nextPosition;
                    nextPosition += widths[i];
                }
            }
            return positions;
        }
        /**
         * Determines the left and right positions of each sticky column cell, which will be the
         * accumulation of all sticky column cell widths to the left and right, respectively.
         * Non-sticky cells do not need to have a value set since their positions will not be applied.
         * @param {?} widths
         * @param {?} stickyStates
         * @return {?}
         */
        _getStickyEndColumnPositions(widths, stickyStates) {
            /** @type {?} */
            const positions = [];
            /** @type {?} */
            let nextPosition = 0;
            for (let i = widths.length; i > 0; i--) {
                if (stickyStates[i]) {
                    positions[i] = nextPosition;
                    nextPosition += widths[i];
                }
            }
            return positions;
        }
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/table/table-errors.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @license
     * Copyright Google LLC All Rights Reserved.
     *
     * Use of this source code is governed by an MIT-style license that can be
     * found in the LICENSE file at https://angular.io/license
     */
    /**
     * Returns an error to be thrown when attempting to find an unexisting column.
     * \@docs-private
     * @param {?} id Id whose lookup failed.
     * @return {?}
     */
    function getTableUnknownColumnError(id) {
        return Error(`Could not find column with id "${id}".`);
    }
    /**
     * Returns an error to be thrown when two column definitions have the same name.
     * \@docs-private
     * @param {?} name
     * @return {?}
     */
    function getTableDuplicateColumnNameError(name) {
        return Error(`Duplicate column definition name provided: "${name}".`);
    }
    /**
     * Returns an error to be thrown when there are multiple rows that are missing a when function.
     * \@docs-private
     * @return {?}
     */
    function getTableMultipleDefaultRowDefsError() {
        return Error(`There can only be one default row without a when predicate function.`);
    }
    /**
     * Returns an error to be thrown when there are no matching row defs for a particular set of data.
     * \@docs-private
     * @param {?} data
     * @return {?}
     */
    function getTableMissingMatchingRowDefError(data) {
        return Error(`Could not find a matching row definition for the` +
            `provided row data: ${JSON.stringify(data)}`);
    }
    /**
     * Returns an error to be thrown when there is no row definitions present in the content.
     * \@docs-private
     * @return {?}
     */
    function getTableMissingRowDefsError() {
        return Error('Missing definitions for header, footer, and row; ' +
            'cannot determine which columns should be rendered.');
    }
    /**
     * Returns an error to be thrown when the data source does not match the compatible types.
     * \@docs-private
     * @return {?}
     */
    function getTableUnknownDataSourceError() {
        return Error(`Provided data source did not match an array, Observable, or DataSource`);
    }
    /**
     * Returns an error to be thrown when the text column cannot find a parent table to inject.
     * \@docs-private
     * @return {?}
     */
    function getTableTextColumnMissingParentTableError() {
        return Error(`Text column could not find a parent table for registration.`);
    }
    /**
     * Returns an error to be thrown when a table text column doesn't have a name.
     * \@docs-private
     * @return {?}
     */
    function getTableTextColumnMissingNameError() {
        return Error(`Table text column must have a name.`);
    }
    /**
     * Provides a handle for the table to grab the view container's ng-container to insert data rows.
     * \@docs-private
     */
    class DataRowOutlet {
        /**
         * @param {?} viewContainer
         * @param {?} elementRef
         */
        constructor(viewContainer, elementRef) {
            this.viewContainer = viewContainer;
            this.elementRef = elementRef;
        }
    }
    DataRowOutlet.ɵfac = function DataRowOutlet_Factory(t) { return new (t || DataRowOutlet)(ɵɵdirectiveInject(ViewContainerRef), ɵɵdirectiveInject(ElementRef)); };
    DataRowOutlet.ɵdir = ɵɵdefineDirective({ type: DataRowOutlet, selectors: [["", "rowOutlet", ""]] });
    /** @nocollapse */
    DataRowOutlet.ctorParameters = () => [
        { type: ViewContainerRef },
        { type: ElementRef }
    ];
    /**
     * Provides a handle for the table to grab the view container's ng-container to insert the header.
     * \@docs-private
     */
    class HeaderRowOutlet {
        /**
         * @param {?} viewContainer
         * @param {?} elementRef
         */
        constructor(viewContainer, elementRef) {
            this.viewContainer = viewContainer;
            this.elementRef = elementRef;
        }
    }
    HeaderRowOutlet.ɵfac = function HeaderRowOutlet_Factory(t) { return new (t || HeaderRowOutlet)(ɵɵdirectiveInject(ViewContainerRef), ɵɵdirectiveInject(ElementRef)); };
    HeaderRowOutlet.ɵdir = ɵɵdefineDirective({ type: HeaderRowOutlet, selectors: [["", "headerRowOutlet", ""]] });
    /** @nocollapse */
    HeaderRowOutlet.ctorParameters = () => [
        { type: ViewContainerRef },
        { type: ElementRef }
    ];
    /**
     * Provides a handle for the table to grab the view container's ng-container to insert the footer.
     * \@docs-private
     */
    class FooterRowOutlet {
        /**
         * @param {?} viewContainer
         * @param {?} elementRef
         */
        constructor(viewContainer, elementRef) {
            this.viewContainer = viewContainer;
            this.elementRef = elementRef;
        }
    }
    FooterRowOutlet.ɵfac = function FooterRowOutlet_Factory(t) { return new (t || FooterRowOutlet)(ɵɵdirectiveInject(ViewContainerRef), ɵɵdirectiveInject(ElementRef)); };
    FooterRowOutlet.ɵdir = ɵɵdefineDirective({ type: FooterRowOutlet, selectors: [["", "footerRowOutlet", ""]] });
    /** @nocollapse */
    FooterRowOutlet.ctorParameters = () => [
        { type: ViewContainerRef },
        { type: ElementRef }
    ];
    /**
     * A data table that can render a header row, data rows, and a footer row.
     * Uses the dataSource input to determine the data to be rendered. The data can be provided either
     * as a data array, an Observable stream that emits the data array to render, or a DataSource with a
     * connect function that will return an Observable stream that emits the data array to render.
     * @template T
     */
    class CdkTable {
        /**
         * @param {?} _differs
         * @param {?} _changeDetectorRef
         * @param {?} _elementRef
         * @param {?} role
         * @param {?} _dir
         * @param {?} _document
         * @param {?} _platform
         */
        constructor(_differs, _changeDetectorRef, _elementRef, role, _dir, _document, _platform) {
            this._differs = _differs;
            this._changeDetectorRef = _changeDetectorRef;
            this._elementRef = _elementRef;
            this._dir = _dir;
            this._platform = _platform;
            /**
             * Subject that emits when the component has been destroyed.
             */
            this._onDestroy = new Subject();
            /**
             * Map of all the user's defined columns (header, data, and footer cell template) identified by
             * name. Collection populated by the column definitions gathered by `ContentChildren` as well as
             * any custom column definitions added to `_customColumnDefs`.
             */
            this._columnDefsByName = new Map();
            /**
             * Column definitions that were defined outside of the direct content children of the table.
             * These will be defined when, e.g., creating a wrapper around the cdkTable that has
             * column definitions as *its* content child.
             */
            this._customColumnDefs = new Set();
            /**
             * Data row definitions that were defined outside of the direct content children of the table.
             * These will be defined when, e.g., creating a wrapper around the cdkTable that has
             * built-in data rows as *its* content child.
             */
            this._customRowDefs = new Set();
            /**
             * Header row definitions that were defined outside of the direct content children of the table.
             * These will be defined when, e.g., creating a wrapper around the cdkTable that has
             * built-in header rows as *its* content child.
             */
            this._customHeaderRowDefs = new Set();
            /**
             * Footer row definitions that were defined outside of the direct content children of the table.
             * These will be defined when, e.g., creating a wrapper around the cdkTable that has a
             * built-in footer row as *its* content child.
             */
            this._customFooterRowDefs = new Set();
            /**
             * Whether the header row definition has been changed. Triggers an update to the header row after
             * content is checked. Initialized as true so that the table renders the initial set of rows.
             */
            this._headerRowDefChanged = true;
            /**
             * Whether the footer row definition has been changed. Triggers an update to the footer row after
             * content is checked. Initialized as true so that the table renders the initial set of rows.
             */
            this._footerRowDefChanged = true;
            /**
             * Cache of the latest rendered `RenderRow` objects as a map for easy retrieval when constructing
             * a new list of `RenderRow` objects for rendering rows. Since the new list is constructed with
             * the cached `RenderRow` objects when possible, the row identity is preserved when the data
             * and row template matches, which allows the `IterableDiffer` to check rows by reference
             * and understand which rows are added/moved/removed.
             *
             * Implemented as a map of maps where the first key is the `data: T` object and the second is the
             * `CdkRowDef<T>` object. With the two keys, the cache points to a `RenderRow<T>` object that
             * contains an array of created pairs. The array is necessary to handle cases where the data
             * array contains multiple duplicate data objects and each instantiated `RenderRow` must be
             * stored.
             */
            this._cachedRenderRowsMap = new Map();
            /**
             * CSS class added to any row or cell that has sticky positioning applied. May be overriden by
             * table subclasses.
             */
            this.stickyCssClass = 'cdk-table-sticky';
            this._multiTemplateDataRows = false;
            // TODO(andrewseguin): Remove max value as the end index
            //   and instead calculate the view on init and scroll.
            /**
             * Stream containing the latest information on what rows are being displayed on screen.
             * Can be used by the data source to as a heuristic of what data should be provided.
             *
             * \@docs-private
             */
            this.viewChange = new BehaviorSubject({ start: 0, end: Number.MAX_VALUE });
            if (!role) {
                this._elementRef.nativeElement.setAttribute('role', 'grid');
            }
            this._document = _document;
            this._isNativeHtmlTable = this._elementRef.nativeElement.nodeName === 'TABLE';
        }
        /**
         * Tracking function that will be used to check the differences in data changes. Used similarly
         * to `ngFor` `trackBy` function. Optimize row operations by identifying a row based on its data
         * relative to the function to know if a row should be added/removed/moved.
         * Accepts a function that takes two parameters, `index` and `item`.
         * @return {?}
         */
        get trackBy() {
            return this._trackByFn;
        }
        /**
         * @param {?} fn
         * @return {?}
         */
        set trackBy(fn) {
            if (isDevMode() && fn != null && typeof fn !== 'function' && (/** @type {?} */ (console)) &&
                (/** @type {?} */ (console.warn))) {
                console.warn(`trackBy must be a function, but received ${JSON.stringify(fn)}.`);
            }
            this._trackByFn = fn;
        }
        /**
         * The table's source of data, which can be provided in three ways (in order of complexity):
         *   - Simple data array (each object represents one table row)
         *   - Stream that emits a data array each time the array changes
         *   - `DataSource` object that implements the connect/disconnect interface.
         *
         * If a data array is provided, the table must be notified when the array's objects are
         * added, removed, or moved. This can be done by calling the `renderRows()` function which will
         * render the diff since the last table render. If the data array reference is changed, the table
         * will automatically trigger an update to the rows.
         *
         * When providing an Observable stream, the table will trigger an update automatically when the
         * stream emits a new array of data.
         *
         * Finally, when providing a `DataSource` object, the table will use the Observable stream
         * provided by the connect function and trigger updates when that stream emits new data array
         * values. During the table's ngOnDestroy or when the data source is removed from the table, the
         * table will call the DataSource's `disconnect` function (may be useful for cleaning up any
         * subscriptions registered during the connect process).
         * @return {?}
         */
        get dataSource() {
            return this._dataSource;
        }
        /**
         * @param {?} dataSource
         * @return {?}
         */
        set dataSource(dataSource) {
            if (this._dataSource !== dataSource) {
                this._switchDataSource(dataSource);
            }
        }
        /**
         * Whether to allow multiple rows per data object by evaluating which rows evaluate their 'when'
         * predicate to true. If `multiTemplateDataRows` is false, which is the default value, then each
         * dataobject will render the first row that evaluates its when predicate to true, in the order
         * defined in the table, or otherwise the default row which does not have a when predicate.
         * @return {?}
         */
        get multiTemplateDataRows() {
            return this._multiTemplateDataRows;
        }
        /**
         * @param {?} v
         * @return {?}
         */
        set multiTemplateDataRows(v) {
            this._multiTemplateDataRows = coerceBooleanProperty(v);
            // In Ivy if this value is set via a static attribute (e.g. <table multiTemplateDataRows>),
            // this setter will be invoked before the row outlet has been defined hence the null check.
            if (this._rowOutlet && this._rowOutlet.viewContainer.length) {
                this._forceRenderDataRows();
            }
        }
        /**
         * @return {?}
         */
        ngOnInit() {
            this._setupStickyStyler();
            if (this._isNativeHtmlTable) {
                this._applyNativeTableSections();
            }
            // Set up the trackBy function so that it uses the `RenderRow` as its identity by default. If
            // the user has provided a custom trackBy, return the result of that function as evaluated
            // with the values of the `RenderRow`'s data and index.
            this._dataDiffer = this._differs.find([]).create((/**
             * @param {?} _i
             * @param {?} dataRow
             * @return {?}
             */
            (_i, dataRow) => {
                return this.trackBy ? this.trackBy(dataRow.dataIndex, dataRow.data) : dataRow;
            }));
        }
        /**
         * @return {?}
         */
        ngAfterContentChecked() {
            // Cache the row and column definitions gathered by ContentChildren and programmatic injection.
            this._cacheRowDefs();
            this._cacheColumnDefs();
            // Make sure that the user has at least added header, footer, or data row def.
            if (!this._headerRowDefs.length && !this._footerRowDefs.length && !this._rowDefs.length) {
                throw getTableMissingRowDefsError();
            }
            // Render updates if the list of columns have been changed for the header, row, or footer defs.
            this._renderUpdatedColumns();
            // If the header row definition has been changed, trigger a render to the header row.
            if (this._headerRowDefChanged) {
                this._forceRenderHeaderRows();
                this._headerRowDefChanged = false;
            }
            // If the footer row definition has been changed, trigger a render to the footer row.
            if (this._footerRowDefChanged) {
                this._forceRenderFooterRows();
                this._footerRowDefChanged = false;
            }
            // If there is a data source and row definitions, connect to the data source unless a
            // connection has already been made.
            if (this.dataSource && this._rowDefs.length > 0 && !this._renderChangeSubscription) {
                this._observeRenderChanges();
            }
            this._checkStickyStates();
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this._rowOutlet.viewContainer.clear();
            this._headerRowOutlet.viewContainer.clear();
            this._footerRowOutlet.viewContainer.clear();
            this._cachedRenderRowsMap.clear();
            this._onDestroy.next();
            this._onDestroy.complete();
            if (isDataSource(this.dataSource)) {
                this.dataSource.disconnect(this);
            }
        }
        /**
         * Renders rows based on the table's latest set of data, which was either provided directly as an
         * input or retrieved through an Observable stream (directly or from a DataSource).
         * Checks for differences in the data since the last diff to perform only the necessary
         * changes (add/remove/move rows).
         *
         * If the table's data source is a DataSource or Observable, this will be invoked automatically
         * each time the provided Observable stream emits a new data array. Otherwise if your data is
         * an array, this function will need to be called to render any changes.
         * @return {?}
         */
        renderRows() {
            this._renderRows = this._getAllRenderRows();
            /** @type {?} */
            const changes = this._dataDiffer.diff(this._renderRows);
            if (!changes) {
                return;
            }
            /** @type {?} */
            const viewContainer = this._rowOutlet.viewContainer;
            changes.forEachOperation((/**
             * @param {?} record
             * @param {?} prevIndex
             * @param {?} currentIndex
             * @return {?}
             */
            (record, prevIndex, currentIndex) => {
                if (record.previousIndex == null) {
                    this._insertRow(record.item, (/** @type {?} */ (currentIndex)));
                }
                else if (currentIndex == null) {
                    viewContainer.remove((/** @type {?} */ (prevIndex)));
                }
                else {
                    /** @type {?} */
                    const view = (/** @type {?} */ (viewContainer.get((/** @type {?} */ (prevIndex)))));
                    viewContainer.move((/** @type {?} */ (view)), currentIndex);
                }
            }));
            // Update the meta context of a row's context data (index, count, first, last, ...)
            this._updateRowIndexContext();
            // Update rows that did not get added/removed/moved but may have had their identity changed,
            // e.g. if trackBy matched data on some property but the actual data reference changed.
            changes.forEachIdentityChange((/**
             * @param {?} record
             * @return {?}
             */
            (record) => {
                /** @type {?} */
                const rowView = (/** @type {?} */ (viewContainer.get((/** @type {?} */ (record.currentIndex)))));
                rowView.context.$implicit = record.item.data;
            }));
            this.updateStickyColumnStyles();
        }
        /**
         * Sets the header row definition to be used. Overrides the header row definition gathered by
         * using `ContentChild`, if one exists. Sets a flag that will re-render the header row after the
         * table's content is checked.
         * \@docs-private
         * @deprecated Use `addHeaderRowDef` and `removeHeaderRowDef` instead
         * \@breaking-change 8.0.0
         * @param {?} headerRowDef
         * @return {?}
         */
        setHeaderRowDef(headerRowDef) {
            this._customHeaderRowDefs = new Set([headerRowDef]);
            this._headerRowDefChanged = true;
        }
        /**
         * Sets the footer row definition to be used. Overrides the footer row definition gathered by
         * using `ContentChild`, if one exists. Sets a flag that will re-render the footer row after the
         * table's content is checked.
         * \@docs-private
         * @deprecated Use `addFooterRowDef` and `removeFooterRowDef` instead
         * \@breaking-change 8.0.0
         * @param {?} footerRowDef
         * @return {?}
         */
        setFooterRowDef(footerRowDef) {
            this._customFooterRowDefs = new Set([footerRowDef]);
            this._footerRowDefChanged = true;
        }
        /**
         * Adds a column definition that was not included as part of the content children.
         * @param {?} columnDef
         * @return {?}
         */
        addColumnDef(columnDef) {
            this._customColumnDefs.add(columnDef);
        }
        /**
         * Removes a column definition that was not included as part of the content children.
         * @param {?} columnDef
         * @return {?}
         */
        removeColumnDef(columnDef) {
            this._customColumnDefs.delete(columnDef);
        }
        /**
         * Adds a row definition that was not included as part of the content children.
         * @param {?} rowDef
         * @return {?}
         */
        addRowDef(rowDef) {
            this._customRowDefs.add(rowDef);
        }
        /**
         * Removes a row definition that was not included as part of the content children.
         * @param {?} rowDef
         * @return {?}
         */
        removeRowDef(rowDef) {
            this._customRowDefs.delete(rowDef);
        }
        /**
         * Adds a header row definition that was not included as part of the content children.
         * @param {?} headerRowDef
         * @return {?}
         */
        addHeaderRowDef(headerRowDef) {
            this._customHeaderRowDefs.add(headerRowDef);
            this._headerRowDefChanged = true;
        }
        /**
         * Removes a header row definition that was not included as part of the content children.
         * @param {?} headerRowDef
         * @return {?}
         */
        removeHeaderRowDef(headerRowDef) {
            this._customHeaderRowDefs.delete(headerRowDef);
            this._headerRowDefChanged = true;
        }
        /**
         * Adds a footer row definition that was not included as part of the content children.
         * @param {?} footerRowDef
         * @return {?}
         */
        addFooterRowDef(footerRowDef) {
            this._customFooterRowDefs.add(footerRowDef);
            this._footerRowDefChanged = true;
        }
        /**
         * Removes a footer row definition that was not included as part of the content children.
         * @param {?} footerRowDef
         * @return {?}
         */
        removeFooterRowDef(footerRowDef) {
            this._customFooterRowDefs.delete(footerRowDef);
            this._footerRowDefChanged = true;
        }
        /**
         * Updates the header sticky styles. First resets all applied styles with respect to the cells
         * sticking to the top. Then, evaluating which cells need to be stuck to the top. This is
         * automatically called when the header row changes its displayed set of columns, or if its
         * sticky input changes. May be called manually for cases where the cell content changes outside
         * of these events.
         * @return {?}
         */
        updateStickyHeaderRowStyles() {
            /** @type {?} */
            const headerRows = this._getRenderedRows(this._headerRowOutlet);
            /** @type {?} */
            const tableElement = (/** @type {?} */ (this._elementRef.nativeElement));
            // Hide the thead element if there are no header rows. This is necessary to satisfy
            // overzealous a11y checkers that fail because the `rowgroup` element does not contain
            // required child `row`.
            /** @type {?} */
            const thead = tableElement.querySelector('thead');
            if (thead) {
                thead.style.display = headerRows.length ? '' : 'none';
            }
            /** @type {?} */
            const stickyStates = this._headerRowDefs.map((/**
             * @param {?} def
             * @return {?}
             */
            def => def.sticky));
            this._stickyStyler.clearStickyPositioning(headerRows, ['top']);
            this._stickyStyler.stickRows(headerRows, stickyStates, 'top');
            // Reset the dirty state of the sticky input change since it has been used.
            this._headerRowDefs.forEach((/**
             * @param {?} def
             * @return {?}
             */
            def => def.resetStickyChanged()));
        }
        /**
         * Updates the footer sticky styles. First resets all applied styles with respect to the cells
         * sticking to the bottom. Then, evaluating which cells need to be stuck to the bottom. This is
         * automatically called when the footer row changes its displayed set of columns, or if its
         * sticky input changes. May be called manually for cases where the cell content changes outside
         * of these events.
         * @return {?}
         */
        updateStickyFooterRowStyles() {
            /** @type {?} */
            const footerRows = this._getRenderedRows(this._footerRowOutlet);
            /** @type {?} */
            const tableElement = (/** @type {?} */ (this._elementRef.nativeElement));
            // Hide the tfoot element if there are no footer rows. This is necessary to satisfy
            // overzealous a11y checkers that fail because the `rowgroup` element does not contain
            // required child `row`.
            /** @type {?} */
            const tfoot = tableElement.querySelector('tfoot');
            if (tfoot) {
                tfoot.style.display = footerRows.length ? '' : 'none';
            }
            /** @type {?} */
            const stickyStates = this._footerRowDefs.map((/**
             * @param {?} def
             * @return {?}
             */
            def => def.sticky));
            this._stickyStyler.clearStickyPositioning(footerRows, ['bottom']);
            this._stickyStyler.stickRows(footerRows, stickyStates, 'bottom');
            this._stickyStyler.updateStickyFooterContainer(this._elementRef.nativeElement, stickyStates);
            // Reset the dirty state of the sticky input change since it has been used.
            this._footerRowDefs.forEach((/**
             * @param {?} def
             * @return {?}
             */
            def => def.resetStickyChanged()));
        }
        /**
         * Updates the column sticky styles. First resets all applied styles with respect to the cells
         * sticking to the left and right. Then sticky styles are added for the left and right according
         * to the column definitions for each cell in each row. This is automatically called when
         * the data source provides a new set of data or when a column definition changes its sticky
         * input. May be called manually for cases where the cell content changes outside of these events.
         * @return {?}
         */
        updateStickyColumnStyles() {
            /** @type {?} */
            const headerRows = this._getRenderedRows(this._headerRowOutlet);
            /** @type {?} */
            const dataRows = this._getRenderedRows(this._rowOutlet);
            /** @type {?} */
            const footerRows = this._getRenderedRows(this._footerRowOutlet);
            // Clear the left and right positioning from all columns in the table across all rows since
            // sticky columns span across all table sections (header, data, footer)
            this._stickyStyler.clearStickyPositioning([...headerRows, ...dataRows, ...footerRows], ['left', 'right']);
            // Update the sticky styles for each header row depending on the def's sticky state
            headerRows.forEach((/**
             * @param {?} headerRow
             * @param {?} i
             * @return {?}
             */
            (headerRow, i) => {
                this._addStickyColumnStyles([headerRow], this._headerRowDefs[i]);
            }));
            // Update the sticky styles for each data row depending on its def's sticky state
            this._rowDefs.forEach((/**
             * @param {?} rowDef
             * @return {?}
             */
            rowDef => {
                // Collect all the rows rendered with this row definition.
                /** @type {?} */
                const rows = [];
                for (let i = 0; i < dataRows.length; i++) {
                    if (this._renderRows[i].rowDef === rowDef) {
                        rows.push(dataRows[i]);
                    }
                }
                this._addStickyColumnStyles(rows, rowDef);
            }));
            // Update the sticky styles for each footer row depending on the def's sticky state
            footerRows.forEach((/**
             * @param {?} footerRow
             * @param {?} i
             * @return {?}
             */
            (footerRow, i) => {
                this._addStickyColumnStyles([footerRow], this._footerRowDefs[i]);
            }));
            // Reset the dirty state of the sticky input change since it has been used.
            Array.from(this._columnDefsByName.values()).forEach((/**
             * @param {?} def
             * @return {?}
             */
            def => def.resetStickyChanged()));
        }
        /**
         * Get the list of RenderRow objects to render according to the current list of data and defined
         * row definitions. If the previous list already contained a particular pair, it should be reused
         * so that the differ equates their references.
         * @private
         * @return {?}
         */
        _getAllRenderRows() {
            /** @type {?} */
            const renderRows = [];
            // Store the cache and create a new one. Any re-used RenderRow objects will be moved into the
            // new cache while unused ones can be picked up by garbage collection.
            /** @type {?} */
            const prevCachedRenderRows = this._cachedRenderRowsMap;
            this._cachedRenderRowsMap = new Map();
            // For each data object, get the list of rows that should be rendered, represented by the
            // respective `RenderRow` object which is the pair of `data` and `CdkRowDef`.
            for (let i = 0; i < this._data.length; i++) {
                /** @type {?} */
                let data = this._data[i];
                /** @type {?} */
                const renderRowsForData = this._getRenderRowsForData(data, i, prevCachedRenderRows.get(data));
                if (!this._cachedRenderRowsMap.has(data)) {
                    this._cachedRenderRowsMap.set(data, new WeakMap());
                }
                for (let j = 0; j < renderRowsForData.length; j++) {
                    /** @type {?} */
                    let renderRow = renderRowsForData[j];
                    /** @type {?} */
                    const cache = (/** @type {?} */ (this._cachedRenderRowsMap.get(renderRow.data)));
                    if (cache.has(renderRow.rowDef)) {
                        (/** @type {?} */ (cache.get(renderRow.rowDef))).push(renderRow);
                    }
                    else {
                        cache.set(renderRow.rowDef, [renderRow]);
                    }
                    renderRows.push(renderRow);
                }
            }
            return renderRows;
        }
        /**
         * Gets a list of `RenderRow<T>` for the provided data object and any `CdkRowDef` objects that
         * should be rendered for this data. Reuses the cached RenderRow objects if they match the same
         * `(T, CdkRowDef)` pair.
         * @private
         * @param {?} data
         * @param {?} dataIndex
         * @param {?=} cache
         * @return {?}
         */
        _getRenderRowsForData(data, dataIndex, cache) {
            /** @type {?} */
            const rowDefs = this._getRowDefs(data, dataIndex);
            return rowDefs.map((/**
             * @param {?} rowDef
             * @return {?}
             */
            rowDef => {
                /** @type {?} */
                const cachedRenderRows = (cache && cache.has(rowDef)) ? (/** @type {?} */ (cache.get(rowDef))) : [];
                if (cachedRenderRows.length) {
                    /** @type {?} */
                    const dataRow = (/** @type {?} */ (cachedRenderRows.shift()));
                    dataRow.dataIndex = dataIndex;
                    return dataRow;
                }
                else {
                    return { data, rowDef, dataIndex };
                }
            }));
        }
        /**
         * Update the map containing the content's column definitions.
         * @private
         * @return {?}
         */
        _cacheColumnDefs() {
            this._columnDefsByName.clear();
            /** @type {?} */
            const columnDefs = mergeArrayAndSet(this._getOwnDefs(this._contentColumnDefs), this._customColumnDefs);
            columnDefs.forEach((/**
             * @param {?} columnDef
             * @return {?}
             */
            columnDef => {
                if (this._columnDefsByName.has(columnDef.name)) {
                    throw getTableDuplicateColumnNameError(columnDef.name);
                }
                this._columnDefsByName.set(columnDef.name, columnDef);
            }));
        }
        /**
         * Update the list of all available row definitions that can be used.
         * @private
         * @return {?}
         */
        _cacheRowDefs() {
            this._headerRowDefs = mergeArrayAndSet(this._getOwnDefs(this._contentHeaderRowDefs), this._customHeaderRowDefs);
            this._footerRowDefs = mergeArrayAndSet(this._getOwnDefs(this._contentFooterRowDefs), this._customFooterRowDefs);
            this._rowDefs = mergeArrayAndSet(this._getOwnDefs(this._contentRowDefs), this._customRowDefs);
            // After all row definitions are determined, find the row definition to be considered default.
            /** @type {?} */
            const defaultRowDefs = this._rowDefs.filter((/**
             * @param {?} def
             * @return {?}
             */
            def => !def.when));
            if (!this.multiTemplateDataRows && defaultRowDefs.length > 1) {
                throw getTableMultipleDefaultRowDefsError();
            }
            this._defaultRowDef = defaultRowDefs[0];
        }
        /**
         * Check if the header, data, or footer rows have changed what columns they want to display or
         * whether the sticky states have changed for the header or footer. If there is a diff, then
         * re-render that section.
         * @private
         * @return {?}
         */
        _renderUpdatedColumns() {
            /** @type {?} */
            const columnsDiffReducer = (/**
             * @param {?} acc
             * @param {?} def
             * @return {?}
             */
            (acc, def) => acc || !!def.getColumnsDiff());
            // Force re-render data rows if the list of column definitions have changed.
            if (this._rowDefs.reduce(columnsDiffReducer, false)) {
                this._forceRenderDataRows();
            }
            // Force re-render header/footer rows if the list of column definitions have changed..
            if (this._headerRowDefs.reduce(columnsDiffReducer, false)) {
                this._forceRenderHeaderRows();
            }
            if (this._footerRowDefs.reduce(columnsDiffReducer, false)) {
                this._forceRenderFooterRows();
            }
        }
        /**
         * Switch to the provided data source by resetting the data and unsubscribing from the current
         * render change subscription if one exists. If the data source is null, interpret this by
         * clearing the row outlet. Otherwise start listening for new data.
         * @private
         * @param {?} dataSource
         * @return {?}
         */
        _switchDataSource(dataSource) {
            this._data = [];
            if (isDataSource(this.dataSource)) {
                this.dataSource.disconnect(this);
            }
            // Stop listening for data from the previous data source.
            if (this._renderChangeSubscription) {
                this._renderChangeSubscription.unsubscribe();
                this._renderChangeSubscription = null;
            }
            if (!dataSource) {
                if (this._dataDiffer) {
                    this._dataDiffer.diff([]);
                }
                this._rowOutlet.viewContainer.clear();
            }
            this._dataSource = dataSource;
        }
        /**
         * Set up a subscription for the data provided by the data source.
         * @private
         * @return {?}
         */
        _observeRenderChanges() {
            // If no data source has been set, there is nothing to observe for changes.
            if (!this.dataSource) {
                return;
            }
            /** @type {?} */
            let dataStream;
            if (isDataSource(this.dataSource)) {
                dataStream = this.dataSource.connect(this);
            }
            else if (isObservable(this.dataSource)) {
                dataStream = this.dataSource;
            }
            else if (Array.isArray(this.dataSource)) {
                dataStream = of(this.dataSource);
            }
            if (dataStream === undefined) {
                throw getTableUnknownDataSourceError();
            }
            this._renderChangeSubscription = dataStream.pipe(takeUntil(this._onDestroy)).subscribe((/**
             * @param {?} data
             * @return {?}
             */
            data => {
                this._data = data || [];
                this.renderRows();
            }));
        }
        /**
         * Clears any existing content in the header row outlet and creates a new embedded view
         * in the outlet using the header row definition.
         * @private
         * @return {?}
         */
        _forceRenderHeaderRows() {
            // Clear the header row outlet if any content exists.
            if (this._headerRowOutlet.viewContainer.length > 0) {
                this._headerRowOutlet.viewContainer.clear();
            }
            this._headerRowDefs.forEach((/**
             * @param {?} def
             * @param {?} i
             * @return {?}
             */
            (def, i) => this._renderRow(this._headerRowOutlet, def, i)));
            this.updateStickyHeaderRowStyles();
            this.updateStickyColumnStyles();
        }
        /**
         * Clears any existing content in the footer row outlet and creates a new embedded view
         * in the outlet using the footer row definition.
         * @private
         * @return {?}
         */
        _forceRenderFooterRows() {
            // Clear the footer row outlet if any content exists.
            if (this._footerRowOutlet.viewContainer.length > 0) {
                this._footerRowOutlet.viewContainer.clear();
            }
            this._footerRowDefs.forEach((/**
             * @param {?} def
             * @param {?} i
             * @return {?}
             */
            (def, i) => this._renderRow(this._footerRowOutlet, def, i)));
            this.updateStickyFooterRowStyles();
            this.updateStickyColumnStyles();
        }
        /**
         * Adds the sticky column styles for the rows according to the columns' stick states.
         * @private
         * @param {?} rows
         * @param {?} rowDef
         * @return {?}
         */
        _addStickyColumnStyles(rows, rowDef) {
            /** @type {?} */
            const columnDefs = Array.from(rowDef.columns || []).map((/**
             * @param {?} columnName
             * @return {?}
             */
            columnName => {
                /** @type {?} */
                const columnDef = this._columnDefsByName.get(columnName);
                if (!columnDef) {
                    throw getTableUnknownColumnError(columnName);
                }
                return (/** @type {?} */ (columnDef));
            }));
            /** @type {?} */
            const stickyStartStates = columnDefs.map((/**
             * @param {?} columnDef
             * @return {?}
             */
            columnDef => columnDef.sticky));
            /** @type {?} */
            const stickyEndStates = columnDefs.map((/**
             * @param {?} columnDef
             * @return {?}
             */
            columnDef => columnDef.stickyEnd));
            this._stickyStyler.updateStickyColumns(rows, stickyStartStates, stickyEndStates);
        }
        /**
         * Gets the list of rows that have been rendered in the row outlet.
         * @param {?} rowOutlet
         * @return {?}
         */
        _getRenderedRows(rowOutlet) {
            /** @type {?} */
            const renderedRows = [];
            for (let i = 0; i < rowOutlet.viewContainer.length; i++) {
                /** @type {?} */
                const viewRef = ((/** @type {?} */ ((/** @type {?} */ (rowOutlet.viewContainer.get(i))))));
                renderedRows.push(viewRef.rootNodes[0]);
            }
            return renderedRows;
        }
        /**
         * Get the matching row definitions that should be used for this row data. If there is only
         * one row definition, it is returned. Otherwise, find the row definitions that has a when
         * predicate that returns true with the data. If none return true, return the default row
         * definition.
         * @param {?} data
         * @param {?} dataIndex
         * @return {?}
         */
        _getRowDefs(data, dataIndex) {
            if (this._rowDefs.length == 1) {
                return [this._rowDefs[0]];
            }
            /** @type {?} */
            let rowDefs = [];
            if (this.multiTemplateDataRows) {
                rowDefs = this._rowDefs.filter((/**
                 * @param {?} def
                 * @return {?}
                 */
                def => !def.when || def.when(dataIndex, data)));
            }
            else {
                /** @type {?} */
                let rowDef = this._rowDefs.find((/**
                 * @param {?} def
                 * @return {?}
                 */
                def => def.when && def.when(dataIndex, data))) || this._defaultRowDef;
                if (rowDef) {
                    rowDefs.push(rowDef);
                }
            }
            if (!rowDefs.length) {
                throw getTableMissingMatchingRowDefError(data);
            }
            return rowDefs;
        }
        /**
         * Create the embedded view for the data row template and place it in the correct index location
         * within the data row view container.
         * @private
         * @param {?} renderRow
         * @param {?} renderIndex
         * @return {?}
         */
        _insertRow(renderRow, renderIndex) {
            /** @type {?} */
            const rowDef = renderRow.rowDef;
            /** @type {?} */
            const context = { $implicit: renderRow.data };
            this._renderRow(this._rowOutlet, rowDef, renderIndex, context);
        }
        /**
         * Creates a new row template in the outlet and fills it with the set of cell templates.
         * Optionally takes a context to provide to the row and cells, as well as an optional index
         * of where to place the new row template in the outlet.
         * @private
         * @param {?} outlet
         * @param {?} rowDef
         * @param {?} index
         * @param {?=} context
         * @return {?}
         */
        _renderRow(outlet, rowDef, index, context = {}) {
            // TODO(andrewseguin): enforce that one outlet was instantiated from createEmbeddedView
            outlet.viewContainer.createEmbeddedView(rowDef.template, context, index);
            for (let cellTemplate of this._getCellTemplates(rowDef)) {
                if (CdkCellOutlet.mostRecentCellOutlet) {
                    CdkCellOutlet.mostRecentCellOutlet._viewContainer.createEmbeddedView(cellTemplate, context);
                }
            }
            this._changeDetectorRef.markForCheck();
        }
        /**
         * Updates the index-related context for each row to reflect any changes in the index of the rows,
         * e.g. first/last/even/odd.
         * @private
         * @return {?}
         */
        _updateRowIndexContext() {
            /** @type {?} */
            const viewContainer = this._rowOutlet.viewContainer;
            for (let renderIndex = 0, count = viewContainer.length; renderIndex < count; renderIndex++) {
                /** @type {?} */
                const viewRef = (/** @type {?} */ (viewContainer.get(renderIndex)));
                /** @type {?} */
                const context = (/** @type {?} */ (viewRef.context));
                context.count = count;
                context.first = renderIndex === 0;
                context.last = renderIndex === count - 1;
                context.even = renderIndex % 2 === 0;
                context.odd = !context.even;
                if (this.multiTemplateDataRows) {
                    context.dataIndex = this._renderRows[renderIndex].dataIndex;
                    context.renderIndex = renderIndex;
                }
                else {
                    context.index = this._renderRows[renderIndex].dataIndex;
                }
            }
        }
        /**
         * Gets the column definitions for the provided row def.
         * @private
         * @param {?} rowDef
         * @return {?}
         */
        _getCellTemplates(rowDef) {
            if (!rowDef || !rowDef.columns) {
                return [];
            }
            return Array.from(rowDef.columns, (/**
             * @param {?} columnId
             * @return {?}
             */
            columnId => {
                /** @type {?} */
                const column = this._columnDefsByName.get(columnId);
                if (!column) {
                    throw getTableUnknownColumnError(columnId);
                }
                return rowDef.extractCellTemplate(column);
            }));
        }
        /**
         * Adds native table sections (e.g. tbody) and moves the row outlets into them.
         * @private
         * @return {?}
         */
        _applyNativeTableSections() {
            /** @type {?} */
            const documentFragment = this._document.createDocumentFragment();
            /** @type {?} */
            const sections = [
                { tag: 'thead', outlet: this._headerRowOutlet },
                { tag: 'tbody', outlet: this._rowOutlet },
                { tag: 'tfoot', outlet: this._footerRowOutlet },
            ];
            for (const section of sections) {
                /** @type {?} */
                const element = this._document.createElement(section.tag);
                element.setAttribute('role', 'rowgroup');
                element.appendChild(section.outlet.elementRef.nativeElement);
                documentFragment.appendChild(element);
            }
            // Use a DocumentFragment so we don't hit the DOM on each iteration.
            this._elementRef.nativeElement.appendChild(documentFragment);
        }
        /**
         * Forces a re-render of the data rows. Should be called in cases where there has been an input
         * change that affects the evaluation of which rows should be rendered, e.g. toggling
         * `multiTemplateDataRows` or adding/removing row definitions.
         * @private
         * @return {?}
         */
        _forceRenderDataRows() {
            this._dataDiffer.diff([]);
            this._rowOutlet.viewContainer.clear();
            this.renderRows();
            this.updateStickyColumnStyles();
        }
        /**
         * Checks if there has been a change in sticky states since last check and applies the correct
         * sticky styles. Since checking resets the "dirty" state, this should only be performed once
         * during a change detection and after the inputs are settled (after content check).
         * @private
         * @return {?}
         */
        _checkStickyStates() {
            /** @type {?} */
            const stickyCheckReducer = (/**
             * @param {?} acc
             * @param {?} d
             * @return {?}
             */
            (acc, d) => {
                return acc || d.hasStickyChanged();
            });
            // Note that the check needs to occur for every definition since it notifies the definition
            // that it can reset its dirty state. Using another operator like `some` may short-circuit
            // remaining definitions and leave them in an unchecked state.
            if (this._headerRowDefs.reduce(stickyCheckReducer, false)) {
                this.updateStickyHeaderRowStyles();
            }
            if (this._footerRowDefs.reduce(stickyCheckReducer, false)) {
                this.updateStickyFooterRowStyles();
            }
            if (Array.from(this._columnDefsByName.values()).reduce(stickyCheckReducer, false)) {
                this.updateStickyColumnStyles();
            }
        }
        /**
         * Creates the sticky styler that will be used for sticky rows and columns. Listens
         * for directionality changes and provides the latest direction to the styler. Re-applies column
         * stickiness when directionality changes.
         * @private
         * @return {?}
         */
        _setupStickyStyler() {
            /** @type {?} */
            const direction = this._dir ? this._dir.value : 'ltr';
            this._stickyStyler = new StickyStyler(this._isNativeHtmlTable, this.stickyCssClass, direction, this._platform.isBrowser);
            (this._dir ? this._dir.change : of())
                .pipe(takeUntil(this._onDestroy))
                .subscribe((/**
             * @param {?} value
             * @return {?}
             */
            value => {
                this._stickyStyler.direction = value;
                this.updateStickyColumnStyles();
            }));
        }
        /**
         * Filters definitions that belong to this table from a QueryList.
         * @private
         * @template I
         * @param {?} items
         * @return {?}
         */
        _getOwnDefs(items) {
            return items.filter((/**
             * @param {?} item
             * @return {?}
             */
            item => !item._table || item._table === this));
        }
    }
    CdkTable.ɵfac = function CdkTable_Factory(t) { return new (t || CdkTable)(ɵɵdirectiveInject(IterableDiffers), ɵɵdirectiveInject(ChangeDetectorRef), ɵɵdirectiveInject(ElementRef), ɵɵinjectAttribute('role'), ɵɵdirectiveInject(Directionality, 8), ɵɵdirectiveInject(DOCUMENT$1), ɵɵdirectiveInject(Platform)); };
    CdkTable.ɵcmp = ɵɵdefineComponent({ type: CdkTable, selectors: [["cdk-table"], ["table", "cdk-table", ""]], contentQueries: function CdkTable_ContentQueries(rf, ctx, dirIndex) { if (rf & 1) {
            ɵɵcontentQuery(dirIndex, CdkColumnDef, true);
            ɵɵcontentQuery(dirIndex, CdkRowDef, true);
            ɵɵcontentQuery(dirIndex, CdkHeaderRowDef, true);
            ɵɵcontentQuery(dirIndex, CdkFooterRowDef, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._contentColumnDefs = _t);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._contentRowDefs = _t);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._contentHeaderRowDefs = _t);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._contentFooterRowDefs = _t);
        } }, viewQuery: function CdkTable_Query(rf, ctx) { if (rf & 1) {
            ɵɵstaticViewQuery(DataRowOutlet, true);
            ɵɵstaticViewQuery(HeaderRowOutlet, true);
            ɵɵstaticViewQuery(FooterRowOutlet, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._rowOutlet = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._headerRowOutlet = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._footerRowOutlet = _t.first);
        } }, hostAttrs: [1, "cdk-table"], inputs: { trackBy: "trackBy", dataSource: "dataSource", multiTemplateDataRows: "multiTemplateDataRows" }, exportAs: ["cdkTable"], features: [ɵɵProvidersFeature([{ provide: CDK_TABLE, useExisting: CdkTable }])], ngContentSelectors: _c1$d, decls: 4, vars: 0, consts: [["headerRowOutlet", ""], ["rowOutlet", ""], ["footerRowOutlet", ""]], template: function CdkTable_Template(rf, ctx) { if (rf & 1) {
            ɵɵprojectionDef(_c0$m);
            ɵɵprojection(0);
            ɵɵelementContainer(1, 0);
            ɵɵelementContainer(2, 1);
            ɵɵelementContainer(3, 2);
        } }, directives: [HeaderRowOutlet,
            DataRowOutlet,
            FooterRowOutlet], encapsulation: 2 });
    /** @nocollapse */
    CdkTable.ctorParameters = () => [
        { type: IterableDiffers },
        { type: ChangeDetectorRef },
        { type: ElementRef },
        { type: String, decorators: [{ type: Attribute, args: ['role',] }] },
        { type: Directionality, decorators: [{ type: Optional }] },
        { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT$1,] }] },
        { type: Platform }
    ];
    CdkTable.propDecorators = {
        trackBy: [{ type: Input }],
        dataSource: [{ type: Input }],
        multiTemplateDataRows: [{ type: Input }],
        _rowOutlet: [{ type: ViewChild, args: [DataRowOutlet, { static: true },] }],
        _headerRowOutlet: [{ type: ViewChild, args: [HeaderRowOutlet, { static: true },] }],
        _footerRowOutlet: [{ type: ViewChild, args: [FooterRowOutlet, { static: true },] }],
        _contentColumnDefs: [{ type: ContentChildren, args: [CdkColumnDef, { descendants: true },] }],
        _contentRowDefs: [{ type: ContentChildren, args: [CdkRowDef, { descendants: true },] }],
        _contentHeaderRowDefs: [{ type: ContentChildren, args: [CdkHeaderRowDef, {
                        descendants: true
                    },] }],
        _contentFooterRowDefs: [{ type: ContentChildren, args: [CdkFooterRowDef, {
                        descendants: true
                    },] }]
    };
    /**
     * Utility function that gets a merged list of the entries in an array and values of a Set.
     * @template T
     * @param {?} array
     * @param {?} set
     * @return {?}
     */
    function mergeArrayAndSet(array, set) {
        return array.concat(Array.from(set));
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: src/cdk/table/text-column.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Column that simply shows text content for the header and row cells. Assumes that the table
     * is using the native table implementation (`<table>`).
     *
     * By default, the name of this column will be the header text and data property accessor.
     * The header text can be overridden with the `headerText` input. Cell values can be overridden with
     * the `dataAccessor` input. Change the text justification to the start or end using the `justify`
     * input.
     * @template T
     */
    class CdkTextColumn {
        /**
         * @param {?} _table
         * @param {?} _options
         */
        constructor(_table, _options) {
            this._table = _table;
            this._options = _options;
            /**
             * Alignment of the cell values.
             */
            this.justify = 'start';
            this._options = _options || {};
        }
        /**
         * Column name that should be used to reference this column.
         * @return {?}
         */
        get name() {
            return this._name;
        }
        /**
         * @param {?} name
         * @return {?}
         */
        set name(name) {
            this._name = name;
            // With Ivy, inputs can be initialized before static query results are
            // available. In that case, we defer the synchronization until "ngOnInit" fires.
            this._syncColumnDefName();
        }
        /**
         * @return {?}
         */
        ngOnInit() {
            this._syncColumnDefName();
            if (this.headerText === undefined) {
                this.headerText = this._createDefaultHeaderText();
            }
            if (!this.dataAccessor) {
                this.dataAccessor =
                    this._options.defaultDataAccessor || ((/**
                     * @param {?} data
                     * @param {?} name
                     * @return {?}
                     */
                    (data, name) => ((/** @type {?} */ (data)))[name]));
            }
            if (this._table) {
                // Provide the cell and headerCell directly to the table with the static `ViewChild` query,
                // since the columnDef will not pick up its content by the time the table finishes checking
                // its content and initializing the rows.
                this.columnDef.cell = this.cell;
                this.columnDef.headerCell = this.headerCell;
                this._table.addColumnDef(this.columnDef);
            }
            else {
                throw getTableTextColumnMissingParentTableError();
            }
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            if (this._table) {
                this._table.removeColumnDef(this.columnDef);
            }
        }
        /**
         * Creates a default header text. Use the options' header text transformation function if one
         * has been provided. Otherwise simply capitalize the column name.
         * @return {?}
         */
        _createDefaultHeaderText() {
            /** @type {?} */
            const name = this.name;
            if (isDevMode() && !name) {
                throw getTableTextColumnMissingNameError();
            }
            if (this._options && this._options.defaultHeaderTextTransform) {
                return this._options.defaultHeaderTextTransform(name);
            }
            return name[0].toUpperCase() + name.slice(1);
        }
        /**
         * Synchronizes the column definition name with the text column name.
         * @private
         * @return {?}
         */
        _syncColumnDefName() {
            if (this.columnDef) {
                this.columnDef.name = this.name;
            }
        }
    }
    CdkTextColumn.ɵfac = function CdkTextColumn_Factory(t) { return new (t || CdkTextColumn)(ɵɵdirectiveInject(CdkTable, 8), ɵɵdirectiveInject(TEXT_COLUMN_OPTIONS, 8)); };
    CdkTextColumn.ɵcmp = ɵɵdefineComponent({ type: CdkTextColumn, selectors: [["cdk-text-column"]], viewQuery: function CdkTextColumn_Query(rf, ctx) { if (rf & 1) {
            ɵɵstaticViewQuery(CdkColumnDef, true);
            ɵɵstaticViewQuery(CdkCellDef, true);
            ɵɵstaticViewQuery(CdkHeaderCellDef, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.columnDef = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.cell = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.headerCell = _t.first);
        } }, inputs: { justify: "justify", name: "name", headerText: "headerText", dataAccessor: "dataAccessor" }, decls: 3, vars: 0, consts: [["cdkColumnDef", ""], ["cdk-header-cell", "", 3, "text-align", 4, "cdkHeaderCellDef"], ["cdk-cell", "", 3, "text-align", 4, "cdkCellDef"], ["cdk-header-cell", ""], ["cdk-cell", ""]], template: function CdkTextColumn_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementContainerStart(0, 0);
            ɵɵtemplate(1, CdkTextColumn_th_1_Template, 2, 3, "th", 1);
            ɵɵtemplate(2, CdkTextColumn_td_2_Template, 2, 3, "td", 2);
            ɵɵelementContainerEnd();
        } }, directives: [CdkColumnDef,
            CdkHeaderCellDef,
            CdkCellDef,
            CdkHeaderCell,
            CdkCell], encapsulation: 2 });
    /** @nocollapse */
    CdkTextColumn.ctorParameters = () => [
        { type: CdkTable, decorators: [{ type: Optional }] },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [TEXT_COLUMN_OPTIONS,] }] }
    ];
    CdkTextColumn.propDecorators = {
        name: [{ type: Input }],
        headerText: [{ type: Input }],
        dataAccessor: [{ type: Input }],
        justify: [{ type: Input }],
        columnDef: [{ type: ViewChild, args: [CdkColumnDef, { static: true },] }],
        cell: [{ type: ViewChild, args: [CdkCellDef, { static: true },] }],
        headerCell: [{ type: ViewChild, args: [CdkHeaderCellDef, { static: true },] }]
    };
    class CdkTableModule {
    }
    CdkTableModule.ɵmod = ɵɵdefineNgModule({ type: CdkTableModule });
    CdkTableModule.ɵinj = ɵɵdefineInjector({ factory: function CdkTableModule_Factory(t) { return new (t || CdkTableModule)(); } });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(CdkTableModule, { declarations: [CdkTable,
            CdkRowDef,
            CdkCellDef,
            CdkCellOutlet,
            CdkHeaderCellDef,
            CdkFooterCellDef,
            CdkColumnDef,
            CdkCell,
            CdkRow,
            CdkHeaderCell,
            CdkFooterCell,
            CdkHeaderRow,
            CdkHeaderRowDef,
            CdkFooterRow,
            CdkFooterRowDef,
            DataRowOutlet,
            HeaderRowOutlet,
            FooterRowOutlet,
            CdkTextColumn], exports: [CdkTable,
            CdkRowDef,
            CdkCellDef,
            CdkCellOutlet,
            CdkHeaderCellDef,
            CdkFooterCellDef,
            CdkColumnDef,
            CdkCell,
            CdkRow,
            CdkHeaderCell,
            CdkFooterCell,
            CdkHeaderRow,
            CdkHeaderRowDef,
            CdkFooterRow,
            CdkFooterRowDef,
            DataRowOutlet,
            HeaderRowOutlet,
            FooterRowOutlet,
            CdkTextColumn] }); })();

    const _c0$n = [[["caption"]]];
    const _c1$e = ["caption"];
    function MatTextColumn_th_1_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "th", 3);
        ɵɵtext(1);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r0 = ɵɵnextContext();
        ɵɵstyleProp("text-align", ctx_r0.justify);
        ɵɵadvance(1);
        ɵɵtextInterpolate1(" ", ctx_r0.headerText, " ");
    } }
    function MatTextColumn_td_2_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "td", 4);
        ɵɵtext(1);
        ɵɵelementEnd();
    } if (rf & 2) {
        const data_r2 = ctx.$implicit;
        const ctx_r1 = ɵɵnextContext();
        ɵɵstyleProp("text-align", ctx_r1.justify);
        ɵɵadvance(1);
        ɵɵtextInterpolate1(" ", ctx_r1.dataAccessor(data_r2, ctx_r1.name), " ");
    } }
    class MatTable extends CdkTable {
        constructor() {
            super(...arguments);
            /**
             * Overrides the sticky CSS class set by the `CdkTable`.
             */
            this.stickyCssClass = 'mat-table-sticky';
        }
    }
    MatTable.ɵfac = function MatTable_Factory(t) { return ɵMatTable_BaseFactory(t || MatTable); };
    MatTable.ɵcmp = ɵɵdefineComponent({ type: MatTable, selectors: [["mat-table"], ["table", "mat-table", ""]], hostAttrs: [1, "mat-table"], exportAs: ["matTable"], features: [ɵɵProvidersFeature([
                { provide: CdkTable, useExisting: MatTable },
                { provide: CDK_TABLE, useExisting: MatTable }
            ]), ɵɵInheritDefinitionFeature], ngContentSelectors: _c1$e, decls: 4, vars: 0, consts: [["headerRowOutlet", ""], ["rowOutlet", ""], ["footerRowOutlet", ""]], template: function MatTable_Template(rf, ctx) { if (rf & 1) {
            ɵɵprojectionDef(_c0$n);
            ɵɵprojection(0);
            ɵɵelementContainer(1, 0);
            ɵɵelementContainer(2, 1);
            ɵɵelementContainer(3, 2);
        } }, directives: [HeaderRowOutlet, DataRowOutlet, FooterRowOutlet], styles: ["mat-table{display:block}mat-header-row{min-height:56px}mat-row,mat-footer-row{min-height:48px}mat-row,mat-header-row,mat-footer-row{display:flex;border-width:0;border-bottom-width:1px;border-style:solid;align-items:center;box-sizing:border-box}mat-row::after,mat-header-row::after,mat-footer-row::after{display:inline-block;min-height:inherit;content:\"\"}mat-cell:first-of-type,mat-header-cell:first-of-type,mat-footer-cell:first-of-type{padding-left:24px}[dir=rtl] mat-cell:first-of-type,[dir=rtl] mat-header-cell:first-of-type,[dir=rtl] mat-footer-cell:first-of-type{padding-left:0;padding-right:24px}mat-cell:last-of-type,mat-header-cell:last-of-type,mat-footer-cell:last-of-type{padding-right:24px}[dir=rtl] mat-cell:last-of-type,[dir=rtl] mat-header-cell:last-of-type,[dir=rtl] mat-footer-cell:last-of-type{padding-right:0;padding-left:24px}mat-cell,mat-header-cell,mat-footer-cell{flex:1;display:flex;align-items:center;overflow:hidden;word-wrap:break-word;min-height:inherit}table.mat-table{border-spacing:0}tr.mat-header-row{height:56px}tr.mat-row,tr.mat-footer-row{height:48px}th.mat-header-cell{text-align:left}[dir=rtl] th.mat-header-cell{text-align:right}th.mat-header-cell,td.mat-cell,td.mat-footer-cell{padding:0;border-bottom-width:1px;border-bottom-style:solid}th.mat-header-cell:first-of-type,td.mat-cell:first-of-type,td.mat-footer-cell:first-of-type{padding-left:24px}[dir=rtl] th.mat-header-cell:first-of-type,[dir=rtl] td.mat-cell:first-of-type,[dir=rtl] td.mat-footer-cell:first-of-type{padding-left:0;padding-right:24px}th.mat-header-cell:last-of-type,td.mat-cell:last-of-type,td.mat-footer-cell:last-of-type{padding-right:24px}[dir=rtl] th.mat-header-cell:last-of-type,[dir=rtl] td.mat-cell:last-of-type,[dir=rtl] td.mat-footer-cell:last-of-type{padding-right:0;padding-left:24px}\n"], encapsulation: 2 });
    const ɵMatTable_BaseFactory = ɵɵgetInheritedFactory(MatTable);

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/table/cell.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Cell definition for the mat-table.
     * Captures the template of a column's data row cell as well as cell-specific properties.
     */
    class MatCellDef extends CdkCellDef {
    }
    MatCellDef.ɵfac = function MatCellDef_Factory(t) { return ɵMatCellDef_BaseFactory(t || MatCellDef); };
    MatCellDef.ɵdir = ɵɵdefineDirective({ type: MatCellDef, selectors: [["", "matCellDef", ""]], features: [ɵɵProvidersFeature([{ provide: CdkCellDef, useExisting: MatCellDef }]), ɵɵInheritDefinitionFeature] });
    const ɵMatCellDef_BaseFactory = ɵɵgetInheritedFactory(MatCellDef);
    /**
     * Header cell definition for the mat-table.
     * Captures the template of a column's header cell and as well as cell-specific properties.
     */
    class MatHeaderCellDef extends CdkHeaderCellDef {
    }
    MatHeaderCellDef.ɵfac = function MatHeaderCellDef_Factory(t) { return ɵMatHeaderCellDef_BaseFactory(t || MatHeaderCellDef); };
    MatHeaderCellDef.ɵdir = ɵɵdefineDirective({ type: MatHeaderCellDef, selectors: [["", "matHeaderCellDef", ""]], features: [ɵɵProvidersFeature([{ provide: CdkHeaderCellDef, useExisting: MatHeaderCellDef }]), ɵɵInheritDefinitionFeature] });
    const ɵMatHeaderCellDef_BaseFactory = ɵɵgetInheritedFactory(MatHeaderCellDef);
    /**
     * Footer cell definition for the mat-table.
     * Captures the template of a column's footer cell and as well as cell-specific properties.
     */
    class MatFooterCellDef extends CdkFooterCellDef {
    }
    MatFooterCellDef.ɵfac = function MatFooterCellDef_Factory(t) { return ɵMatFooterCellDef_BaseFactory(t || MatFooterCellDef); };
    MatFooterCellDef.ɵdir = ɵɵdefineDirective({ type: MatFooterCellDef, selectors: [["", "matFooterCellDef", ""]], features: [ɵɵProvidersFeature([{ provide: CdkFooterCellDef, useExisting: MatFooterCellDef }]), ɵɵInheritDefinitionFeature] });
    const ɵMatFooterCellDef_BaseFactory = ɵɵgetInheritedFactory(MatFooterCellDef);
    /**
     * Column definition for the mat-table.
     * Defines a set of cells available for a table column.
     */
    class MatColumnDef extends CdkColumnDef {
    }
    MatColumnDef.ɵfac = function MatColumnDef_Factory(t) { return ɵMatColumnDef_BaseFactory(t || MatColumnDef); };
    MatColumnDef.ɵdir = ɵɵdefineDirective({ type: MatColumnDef, selectors: [["", "matColumnDef", ""]], inputs: { sticky: "sticky", name: ["matColumnDef", "name"] }, features: [ɵɵProvidersFeature([
                { provide: CdkColumnDef, useExisting: MatColumnDef },
                { provide: 'MAT_SORT_HEADER_COLUMN_DEF', useExisting: MatColumnDef }
            ]), ɵɵInheritDefinitionFeature] });
    MatColumnDef.propDecorators = {
        name: [{ type: Input, args: ['matColumnDef',] }]
    };
    const ɵMatColumnDef_BaseFactory = ɵɵgetInheritedFactory(MatColumnDef);
    /**
     * Header cell template container that adds the right classes and role.
     */
    class MatHeaderCell extends CdkHeaderCell {
        /**
         * @param {?} columnDef
         * @param {?} elementRef
         */
        constructor(columnDef, elementRef) {
            super(columnDef, elementRef);
            elementRef.nativeElement.classList.add(`mat-column-${columnDef.cssClassFriendlyName}`);
        }
    }
    MatHeaderCell.ɵfac = function MatHeaderCell_Factory(t) { return new (t || MatHeaderCell)(ɵɵdirectiveInject(CdkColumnDef), ɵɵdirectiveInject(ElementRef)); };
    MatHeaderCell.ɵdir = ɵɵdefineDirective({ type: MatHeaderCell, selectors: [["mat-header-cell"], ["th", "mat-header-cell", ""]], hostAttrs: ["role", "columnheader", 1, "mat-header-cell"], features: [ɵɵInheritDefinitionFeature] });
    /** @nocollapse */
    MatHeaderCell.ctorParameters = () => [
        { type: CdkColumnDef },
        { type: ElementRef }
    ];
    /**
     * Footer cell template container that adds the right classes and role.
     */
    class MatFooterCell extends CdkFooterCell {
        /**
         * @param {?} columnDef
         * @param {?} elementRef
         */
        constructor(columnDef, elementRef) {
            super(columnDef, elementRef);
            elementRef.nativeElement.classList.add(`mat-column-${columnDef.cssClassFriendlyName}`);
        }
    }
    MatFooterCell.ɵfac = function MatFooterCell_Factory(t) { return new (t || MatFooterCell)(ɵɵdirectiveInject(CdkColumnDef), ɵɵdirectiveInject(ElementRef)); };
    MatFooterCell.ɵdir = ɵɵdefineDirective({ type: MatFooterCell, selectors: [["mat-footer-cell"], ["td", "mat-footer-cell", ""]], hostAttrs: ["role", "gridcell", 1, "mat-footer-cell"], features: [ɵɵInheritDefinitionFeature] });
    /** @nocollapse */
    MatFooterCell.ctorParameters = () => [
        { type: CdkColumnDef },
        { type: ElementRef }
    ];
    /**
     * Cell template container that adds the right classes and role.
     */
    class MatCell extends CdkCell {
        /**
         * @param {?} columnDef
         * @param {?} elementRef
         */
        constructor(columnDef, elementRef) {
            super(columnDef, elementRef);
            elementRef.nativeElement.classList.add(`mat-column-${columnDef.cssClassFriendlyName}`);
        }
    }
    MatCell.ɵfac = function MatCell_Factory(t) { return new (t || MatCell)(ɵɵdirectiveInject(CdkColumnDef), ɵɵdirectiveInject(ElementRef)); };
    MatCell.ɵdir = ɵɵdefineDirective({ type: MatCell, selectors: [["mat-cell"], ["td", "mat-cell", ""]], hostAttrs: ["role", "gridcell", 1, "mat-cell"], features: [ɵɵInheritDefinitionFeature] });
    /** @nocollapse */
    MatCell.ctorParameters = () => [
        { type: CdkColumnDef },
        { type: ElementRef }
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/table/row.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Header row definition for the mat-table.
     * Captures the header row's template and other header properties such as the columns to display.
     */
    class MatHeaderRowDef extends CdkHeaderRowDef {
    }
    MatHeaderRowDef.ɵfac = function MatHeaderRowDef_Factory(t) { return ɵMatHeaderRowDef_BaseFactory(t || MatHeaderRowDef); };
    MatHeaderRowDef.ɵdir = ɵɵdefineDirective({ type: MatHeaderRowDef, selectors: [["", "matHeaderRowDef", ""]], inputs: { columns: ["matHeaderRowDef", "columns"], sticky: ["matHeaderRowDefSticky", "sticky"] }, features: [ɵɵProvidersFeature([{ provide: CdkHeaderRowDef, useExisting: MatHeaderRowDef }]), ɵɵInheritDefinitionFeature] });
    const ɵMatHeaderRowDef_BaseFactory = ɵɵgetInheritedFactory(MatHeaderRowDef);
    /**
     * Footer row definition for the mat-table.
     * Captures the footer row's template and other footer properties such as the columns to display.
     */
    class MatFooterRowDef extends CdkFooterRowDef {
    }
    MatFooterRowDef.ɵfac = function MatFooterRowDef_Factory(t) { return ɵMatFooterRowDef_BaseFactory(t || MatFooterRowDef); };
    MatFooterRowDef.ɵdir = ɵɵdefineDirective({ type: MatFooterRowDef, selectors: [["", "matFooterRowDef", ""]], inputs: { columns: ["matFooterRowDef", "columns"], sticky: ["matFooterRowDefSticky", "sticky"] }, features: [ɵɵProvidersFeature([{ provide: CdkFooterRowDef, useExisting: MatFooterRowDef }]), ɵɵInheritDefinitionFeature] });
    const ɵMatFooterRowDef_BaseFactory = ɵɵgetInheritedFactory(MatFooterRowDef);
    /**
     * Data row definition for the mat-table.
     * Captures the data row's template and other properties such as the columns to display and
     * a when predicate that describes when this row should be used.
     * @template T
     */
    class MatRowDef extends CdkRowDef {
    }
    MatRowDef.ɵfac = function MatRowDef_Factory(t) { return ɵMatRowDef_BaseFactory(t || MatRowDef); };
    MatRowDef.ɵdir = ɵɵdefineDirective({ type: MatRowDef, selectors: [["", "matRowDef", ""]], inputs: { columns: ["matRowDefColumns", "columns"], when: ["matRowDefWhen", "when"] }, features: [ɵɵProvidersFeature([{ provide: CdkRowDef, useExisting: MatRowDef }]), ɵɵInheritDefinitionFeature] });
    const ɵMatRowDef_BaseFactory = ɵɵgetInheritedFactory(MatRowDef);
    /**
     * Header template container that contains the cell outlet. Adds the right class and role.
     */
    class MatHeaderRow extends CdkHeaderRow {
    }
    MatHeaderRow.ɵfac = function MatHeaderRow_Factory(t) { return ɵMatHeaderRow_BaseFactory(t || MatHeaderRow); };
    MatHeaderRow.ɵcmp = ɵɵdefineComponent({ type: MatHeaderRow, selectors: [["mat-header-row"], ["tr", "mat-header-row", ""]], hostAttrs: ["role", "row", 1, "mat-header-row"], exportAs: ["matHeaderRow"], features: [ɵɵProvidersFeature([{ provide: CdkHeaderRow, useExisting: MatHeaderRow }]), ɵɵInheritDefinitionFeature], decls: 1, vars: 0, consts: [["cdkCellOutlet", ""]], template: function MatHeaderRow_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementContainer(0, 0);
        } }, directives: [CdkCellOutlet], encapsulation: 2 });
    const ɵMatHeaderRow_BaseFactory = ɵɵgetInheritedFactory(MatHeaderRow);
    /**
     * Footer template container that contains the cell outlet. Adds the right class and role.
     */
    class MatFooterRow extends CdkFooterRow {
    }
    MatFooterRow.ɵfac = function MatFooterRow_Factory(t) { return ɵMatFooterRow_BaseFactory(t || MatFooterRow); };
    MatFooterRow.ɵcmp = ɵɵdefineComponent({ type: MatFooterRow, selectors: [["mat-footer-row"], ["tr", "mat-footer-row", ""]], hostAttrs: ["role", "row", 1, "mat-footer-row"], exportAs: ["matFooterRow"], features: [ɵɵProvidersFeature([{ provide: CdkFooterRow, useExisting: MatFooterRow }]), ɵɵInheritDefinitionFeature], decls: 1, vars: 0, consts: [["cdkCellOutlet", ""]], template: function MatFooterRow_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementContainer(0, 0);
        } }, directives: [CdkCellOutlet], encapsulation: 2 });
    const ɵMatFooterRow_BaseFactory = ɵɵgetInheritedFactory(MatFooterRow);
    /**
     * Data row template container that contains the cell outlet. Adds the right class and role.
     */
    class MatRow extends CdkRow {
    }
    MatRow.ɵfac = function MatRow_Factory(t) { return ɵMatRow_BaseFactory(t || MatRow); };
    MatRow.ɵcmp = ɵɵdefineComponent({ type: MatRow, selectors: [["mat-row"], ["tr", "mat-row", ""]], hostAttrs: ["role", "row", 1, "mat-row"], exportAs: ["matRow"], features: [ɵɵProvidersFeature([{ provide: CdkRow, useExisting: MatRow }]), ɵɵInheritDefinitionFeature], decls: 1, vars: 0, consts: [["cdkCellOutlet", ""]], template: function MatRow_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementContainer(0, 0);
        } }, directives: [CdkCellOutlet], encapsulation: 2 });
    const ɵMatRow_BaseFactory = ɵɵgetInheritedFactory(MatRow);

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/table/text-column.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Column that simply shows text content for the header and row cells. Assumes that the table
     * is using the native table implementation (`<table>`).
     *
     * By default, the name of this column will be the header text and data property accessor.
     * The header text can be overridden with the `headerText` input. Cell values can be overridden with
     * the `dataAccessor` input. Change the text justification to the start or end using the `justify`
     * input.
     * @template T
     */
    class MatTextColumn extends CdkTextColumn {
    }
    MatTextColumn.ɵfac = function MatTextColumn_Factory(t) { return ɵMatTextColumn_BaseFactory(t || MatTextColumn); };
    MatTextColumn.ɵcmp = ɵɵdefineComponent({ type: MatTextColumn, selectors: [["mat-text-column"]], features: [ɵɵInheritDefinitionFeature], decls: 3, vars: 0, consts: [["matColumnDef", ""], ["mat-header-cell", "", 3, "text-align", 4, "matHeaderCellDef"], ["mat-cell", "", 3, "text-align", 4, "matCellDef"], ["mat-header-cell", ""], ["mat-cell", ""]], template: function MatTextColumn_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementContainerStart(0, 0);
            ɵɵtemplate(1, MatTextColumn_th_1_Template, 2, 3, "th", 1);
            ɵɵtemplate(2, MatTextColumn_td_2_Template, 2, 3, "td", 2);
            ɵɵelementContainerEnd();
        } }, directives: [MatColumnDef,
            MatHeaderCellDef,
            MatCellDef,
            MatHeaderCell,
            MatCell], encapsulation: 2 });
    const ɵMatTextColumn_BaseFactory = ɵɵgetInheritedFactory(MatTextColumn);
    class MatTableModule {
    }
    MatTableModule.ɵmod = ɵɵdefineNgModule({ type: MatTableModule });
    MatTableModule.ɵinj = ɵɵdefineInjector({ factory: function MatTableModule_Factory(t) { return new (t || MatTableModule)(); }, imports: [[
                CdkTableModule,
                MatCommonModule,
            ],
            MatCommonModule] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(MatTableModule, { declarations: function () { return [MatTable,
            MatHeaderCellDef,
            MatHeaderRowDef,
            MatColumnDef,
            MatCellDef,
            MatRowDef,
            MatFooterCellDef,
            MatFooterRowDef,
            MatHeaderCell,
            MatCell,
            MatFooterCell,
            MatHeaderRow,
            MatRow,
            MatFooterRow,
            MatTextColumn]; }, imports: function () { return [CdkTableModule,
            MatCommonModule]; }, exports: function () { return [MatCommonModule,
            MatTable,
            MatHeaderCellDef,
            MatHeaderRowDef,
            MatColumnDef,
            MatCellDef,
            MatRowDef,
            MatFooterCellDef,
            MatFooterRowDef,
            MatHeaderCell,
            MatCell,
            MatFooterCell,
            MatHeaderRow,
            MatRow,
            MatFooterRow,
            MatTextColumn]; } }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/table/table-data-source.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Corresponds to `Number.MAX_SAFE_INTEGER`. Moved out into a variable here due to
     * flaky browser support and the value not being defined in Closure's typings.
     * @type {?}
     */
    const MAX_SAFE_INTEGER = 9007199254740991;
    /**
     * Data source that accepts a client-side data array and includes native support of filtering,
     * sorting (using MatSort), and pagination (using MatPaginator).
     *
     * Allows for sort customization by overriding sortingDataAccessor, which defines how data
     * properties are accessed. Also allows for filter customization by overriding filterTermAccessor,
     * which defines how row data is converted to a string for filter matching.
     *
     * **Note:** This class is meant to be a simple data source to help you get started. As such
     * it isn't equipped to handle some more advanced cases like robust i18n support or server-side
     * interactions. If your app needs to support more advanced use cases, consider implementing your
     * own `DataSource`.
     * @template T
     */
    class MatTableDataSource extends DataSource {
        /**
         * @param {?=} initialData
         */
        constructor(initialData = []) {
            super();
            /**
             * Stream emitting render data to the table (depends on ordered data changes).
             */
            this._renderData = new BehaviorSubject([]);
            /**
             * Stream that emits when a new filter string is set on the data source.
             */
            this._filter = new BehaviorSubject('');
            /**
             * Used to react to internal changes of the paginator that are made by the data source itself.
             */
            this._internalPageChanges = new Subject();
            /**
             * Subscription to the changes that should trigger an update to the table's rendered rows, such
             * as filtering, sorting, pagination, or base data changes.
             */
            this._renderChangesSubscription = Subscription.EMPTY;
            /**
             * Data accessor function that is used for accessing data properties for sorting through
             * the default sortData function.
             * This default function assumes that the sort header IDs (which defaults to the column name)
             * matches the data's properties (e.g. column Xyz represents data['Xyz']).
             * May be set to a custom function for different behavior.
             * @param data Data object that is being accessed.
             * @param sortHeaderId The name of the column that represents the data.
             */
            this.sortingDataAccessor = (/**
             * @param {?} data
             * @param {?} sortHeaderId
             * @return {?}
             */
            (data, sortHeaderId) => {
                /** @type {?} */
                const value = ((/** @type {?} */ (data)))[sortHeaderId];
                if (_isNumberValue(value)) {
                    /** @type {?} */
                    const numberValue = Number(value);
                    // Numbers beyond `MAX_SAFE_INTEGER` can't be compared reliably so we
                    // leave them as strings. For more info: https://goo.gl/y5vbSg
                    return numberValue < MAX_SAFE_INTEGER ? numberValue : value;
                }
                return value;
            });
            /**
             * Gets a sorted copy of the data array based on the state of the MatSort. Called
             * after changes are made to the filtered data or when sort changes are emitted from MatSort.
             * By default, the function retrieves the active sort and its direction and compares data
             * by retrieving data using the sortingDataAccessor. May be overridden for a custom implementation
             * of data ordering.
             * @param data The array of data that should be sorted.
             * @param sort The connected MatSort that holds the current sort state.
             */
            this.sortData = (/**
             * @param {?} data
             * @param {?} sort
             * @return {?}
             */
            (data, sort) => {
                /** @type {?} */
                const active = sort.active;
                /** @type {?} */
                const direction = sort.direction;
                if (!active || direction == '') {
                    return data;
                }
                return data.sort((/**
                 * @param {?} a
                 * @param {?} b
                 * @return {?}
                 */
                (a, b) => {
                    /** @type {?} */
                    let valueA = this.sortingDataAccessor(a, active);
                    /** @type {?} */
                    let valueB = this.sortingDataAccessor(b, active);
                    // If both valueA and valueB exist (truthy), then compare the two. Otherwise, check if
                    // one value exists while the other doesn't. In this case, existing value should come last.
                    // This avoids inconsistent results when comparing values to undefined/null.
                    // If neither value exists, return 0 (equal).
                    /** @type {?} */
                    let comparatorResult = 0;
                    if (valueA != null && valueB != null) {
                        // Check if one value is greater than the other; if equal, comparatorResult should remain 0.
                        if (valueA > valueB) {
                            comparatorResult = 1;
                        }
                        else if (valueA < valueB) {
                            comparatorResult = -1;
                        }
                    }
                    else if (valueA != null) {
                        comparatorResult = 1;
                    }
                    else if (valueB != null) {
                        comparatorResult = -1;
                    }
                    return comparatorResult * (direction == 'asc' ? 1 : -1);
                }));
            });
            /**
             * Checks if a data object matches the data source's filter string. By default, each data object
             * is converted to a string of its properties and returns true if the filter has
             * at least one occurrence in that string. By default, the filter string has its whitespace
             * trimmed and the match is case-insensitive. May be overridden for a custom implementation of
             * filter matching.
             * @param data Data object used to check against the filter.
             * @param filter Filter string that has been set on the data source.
             * @return Whether the filter matches against the data
             */
            this.filterPredicate = (/**
             * @param {?} data
             * @param {?} filter
             * @return {?}
             */
            (data, filter) => {
                // Transform the data into a lowercase string of all property values.
                /** @type {?} */
                const dataStr = Object.keys(data).reduce((/**
                 * @param {?} currentTerm
                 * @param {?} key
                 * @return {?}
                 */
                (currentTerm, key) => {
                    // Use an obscure Unicode character to delimit the words in the concatenated string.
                    // This avoids matches where the values of two columns combined will match the user's query
                    // (e.g. `Flute` and `Stop` will match `Test`). The character is intended to be something
                    // that has a very low chance of being typed in by somebody in a text field. This one in
                    // particular is "White up-pointing triangle with dot" from
                    // https://en.wikipedia.org/wiki/List_of_Unicode_characters
                    return currentTerm + ((/** @type {?} */ (data)))[key] + '◬';
                }), '').toLowerCase();
                // Transform the filter by converting it to lowercase and removing whitespace.
                /** @type {?} */
                const transformedFilter = filter.trim().toLowerCase();
                return dataStr.indexOf(transformedFilter) != -1;
            });
            this._data = new BehaviorSubject(initialData);
            this._updateChangeSubscription();
        }
        /**
         * Array of data that should be rendered by the table, where each object represents one row.
         * @return {?}
         */
        get data() { return this._data.value; }
        /**
         * @param {?} data
         * @return {?}
         */
        set data(data) { this._data.next(data); }
        /**
         * Filter term that should be used to filter out objects from the data array. To override how
         * data objects match to this filter string, provide a custom function for filterPredicate.
         * @return {?}
         */
        get filter() { return this._filter.value; }
        /**
         * @param {?} filter
         * @return {?}
         */
        set filter(filter) { this._filter.next(filter); }
        /**
         * Instance of the MatSort directive used by the table to control its sorting. Sort changes
         * emitted by the MatSort will trigger an update to the table's rendered data.
         * @return {?}
         */
        get sort() { return this._sort; }
        /**
         * @param {?} sort
         * @return {?}
         */
        set sort(sort) {
            this._sort = sort;
            this._updateChangeSubscription();
        }
        /**
         * Instance of the MatPaginator component used by the table to control what page of the data is
         * displayed. Page changes emitted by the MatPaginator will trigger an update to the
         * table's rendered data.
         *
         * Note that the data source uses the paginator's properties to calculate which page of data
         * should be displayed. If the paginator receives its properties as template inputs,
         * e.g. `[pageLength]=100` or `[pageIndex]=1`, then be sure that the paginator's view has been
         * initialized before assigning it to this data source.
         * @return {?}
         */
        get paginator() { return this._paginator; }
        /**
         * @param {?} paginator
         * @return {?}
         */
        set paginator(paginator) {
            this._paginator = paginator;
            this._updateChangeSubscription();
        }
        /**
         * Subscribe to changes that should trigger an update to the table's rendered rows. When the
         * changes occur, process the current state of the filter, sort, and pagination along with
         * the provided base data and send it to the table for rendering.
         * @return {?}
         */
        _updateChangeSubscription() {
            // Sorting and/or pagination should be watched if MatSort and/or MatPaginator are provided.
            // The events should emit whenever the component emits a change or initializes, or if no
            // component is provided, a stream with just a null event should be provided.
            // The `sortChange` and `pageChange` acts as a signal to the combineLatests below so that the
            // pipeline can progress to the next step. Note that the value from these streams are not used,
            // they purely act as a signal to progress in the pipeline.
            /** @type {?} */
            const sortChange = this._sort ?
                (/** @type {?} */ (merge(this._sort.sortChange, this._sort.initialized))) :
                of(null);
            /** @type {?} */
            const pageChange = this._paginator ?
                (/** @type {?} */ (merge(this._paginator.page, this._internalPageChanges, this._paginator.initialized))) :
                of(null);
            /** @type {?} */
            const dataStream = this._data;
            // Watch for base data or filter changes to provide a filtered set of data.
            /** @type {?} */
            const filteredData = combineLatest([dataStream, this._filter])
                .pipe(map((/**
             * @param {?} __0
             * @return {?}
             */
            ([data]) => this._filterData(data))));
            // Watch for filtered data or sort changes to provide an ordered set of data.
            /** @type {?} */
            const orderedData = combineLatest([filteredData, sortChange])
                .pipe(map((/**
             * @param {?} __0
             * @return {?}
             */
            ([data]) => this._orderData(data))));
            // Watch for ordered data or page changes to provide a paged set of data.
            /** @type {?} */
            const paginatedData = combineLatest([orderedData, pageChange])
                .pipe(map((/**
             * @param {?} __0
             * @return {?}
             */
            ([data]) => this._pageData(data))));
            // Watched for paged data changes and send the result to the table to render.
            this._renderChangesSubscription.unsubscribe();
            this._renderChangesSubscription = paginatedData.subscribe((/**
             * @param {?} data
             * @return {?}
             */
            data => this._renderData.next(data)));
        }
        /**
         * Returns a filtered data array where each filter object contains the filter string within
         * the result of the filterTermAccessor function. If no filter is set, returns the data array
         * as provided.
         * @param {?} data
         * @return {?}
         */
        _filterData(data) {
            // If there is a filter string, filter out data that does not contain it.
            // Each data object is converted to a string using the function defined by filterTermAccessor.
            // May be overridden for customization.
            this.filteredData =
                !this.filter ? data : data.filter((/**
                 * @param {?} obj
                 * @return {?}
                 */
                obj => this.filterPredicate(obj, this.filter)));
            if (this.paginator) {
                this._updatePaginator(this.filteredData.length);
            }
            return this.filteredData;
        }
        /**
         * Returns a sorted copy of the data if MatSort has a sort applied, otherwise just returns the
         * data array as provided. Uses the default data accessor for data lookup, unless a
         * sortDataAccessor function is defined.
         * @param {?} data
         * @return {?}
         */
        _orderData(data) {
            // If there is no active sort or direction, return the data without trying to sort.
            if (!this.sort) {
                return data;
            }
            return this.sortData(data.slice(), this.sort);
        }
        /**
         * Returns a paged slice of the provided data array according to the provided MatPaginator's page
         * index and length. If there is no paginator provided, returns the data array as provided.
         * @param {?} data
         * @return {?}
         */
        _pageData(data) {
            if (!this.paginator) {
                return data;
            }
            /** @type {?} */
            const startIndex = this.paginator.pageIndex * this.paginator.pageSize;
            return data.slice(startIndex, startIndex + this.paginator.pageSize);
        }
        /**
         * Updates the paginator to reflect the length of the filtered data, and makes sure that the page
         * index does not exceed the paginator's last page. Values are changed in a resolved promise to
         * guard against making property changes within a round of change detection.
         * @param {?} filteredDataLength
         * @return {?}
         */
        _updatePaginator(filteredDataLength) {
            Promise.resolve().then((/**
             * @return {?}
             */
            () => {
                /** @type {?} */
                const paginator = this.paginator;
                if (!paginator) {
                    return;
                }
                paginator.length = filteredDataLength;
                // If the page index is set beyond the page, reduce it to the last page.
                if (paginator.pageIndex > 0) {
                    /** @type {?} */
                    const lastPageIndex = Math.ceil(paginator.length / paginator.pageSize) - 1 || 0;
                    /** @type {?} */
                    const newPageIndex = Math.min(paginator.pageIndex, lastPageIndex);
                    if (newPageIndex !== paginator.pageIndex) {
                        paginator.pageIndex = newPageIndex;
                        // Since the paginator only emits after user-generated changes,
                        // we need our own stream so we know to should re-render the data.
                        this._internalPageChanges.next();
                    }
                }
            }));
        }
        /**
         * Used by the MatTable. Called when it connects to the data source.
         * \@docs-private
         * @return {?}
         */
        connect() { return this._renderData; }
        /**
         * Used by the MatTable. Called when it is destroyed. No-op.
         * \@docs-private
         * @return {?}
         */
        disconnect() { }
    }

    const _c0$o = ["dialogPopup"];
    const _c1$f = ["hueSlider"];
    const _c2$6 = ["alphaSlider"];
    function ColorPickerComponent_div_2_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "div");
    } if (rf & 2) {
        const ctx_r1 = ɵɵnextContext();
        ɵɵclassMapInterpolate1("arrow arrow-", ctx_r1.cpUsePosition, "");
        ɵɵstyleProp("top", ctx_r1.arrowTop, "px");
    } }
    function ColorPickerComponent_div_3_Template(rf, ctx) { if (rf & 1) {
        const _r17 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "div", 26);
        ɵɵlistener("newValue", function ColorPickerComponent_div_3_Template_div_newValue_0_listener($event) { ɵɵrestoreView(_r17); const ctx_r16 = ɵɵnextContext(); return ctx_r16.onColorChange($event); })("dragStart", function ColorPickerComponent_div_3_Template_div_dragStart_0_listener() { ɵɵrestoreView(_r17); const ctx_r18 = ɵɵnextContext(); return ctx_r18.onDragStart("saturation-lightness"); })("dragEnd", function ColorPickerComponent_div_3_Template_div_dragEnd_0_listener() { ɵɵrestoreView(_r17); const ctx_r19 = ɵɵnextContext(); return ctx_r19.onDragEnd("saturation-lightness"); });
        ɵɵelement(1, "div", 13);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r2 = ɵɵnextContext();
        ɵɵstyleProp("background-color", ctx_r2.hueSliderColor);
        ɵɵproperty("rgX", 1)("rgY", 1);
        ɵɵadvance(1);
        ɵɵstyleProp("top", ctx_r2.slider == null ? null : ctx_r2.slider.v, "px")("left", ctx_r2.slider == null ? null : ctx_r2.slider.s, "px");
    } }
    function ColorPickerComponent_button_8_Template(rf, ctx) { if (rf & 1) {
        const _r21 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "button", 27);
        ɵɵlistener("click", function ColorPickerComponent_button_8_Template_button_click_0_listener($event) { ɵɵrestoreView(_r21); const ctx_r20 = ɵɵnextContext(); return ctx_r20.onAddPresetColor($event, ctx_r20.selectedColor); });
        ɵɵtext(1);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r3 = ɵɵnextContext();
        ɵɵclassMap(ctx_r3.cpAddColorButtonClass);
        ɵɵproperty("disabled", ctx_r3.cpPresetColors && ctx_r3.cpPresetColors.length >= ctx_r3.cpMaxPresetColorsLength);
        ɵɵadvance(1);
        ɵɵtextInterpolate1(" ", ctx_r3.cpAddColorButtonText, " ");
    } }
    function ColorPickerComponent_div_10_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "div", 28);
    } }
    function ColorPickerComponent_div_20_input_6_Template(rf, ctx) { if (rf & 1) {
        const _r25 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "input", 34);
        ɵɵlistener("keyup.enter", function ColorPickerComponent_div_20_input_6_Template_input_keyup_enter_0_listener($event) { ɵɵrestoreView(_r25); const ctx_r24 = ɵɵnextContext(2); return ctx_r24.onAcceptColor($event); })("newValue", function ColorPickerComponent_div_20_input_6_Template_input_newValue_0_listener($event) { ɵɵrestoreView(_r25); const ctx_r26 = ɵɵnextContext(2); return ctx_r26.onAlphaInput($event); });
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r22 = ɵɵnextContext(2);
        ɵɵproperty("rg", 1)("value", ctx_r22.cmykText == null ? null : ctx_r22.cmykText.a);
    } }
    function ColorPickerComponent_div_20_div_16_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div");
        ɵɵtext(1, "A");
        ɵɵelementEnd();
    } }
    function ColorPickerComponent_div_20_Template(rf, ctx) { if (rf & 1) {
        const _r28 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "div", 29);
        ɵɵelementStart(1, "div", 30);
        ɵɵelementStart(2, "input", 31);
        ɵɵlistener("keyup.enter", function ColorPickerComponent_div_20_Template_input_keyup_enter_2_listener($event) { ɵɵrestoreView(_r28); const ctx_r27 = ɵɵnextContext(); return ctx_r27.onAcceptColor($event); })("newValue", function ColorPickerComponent_div_20_Template_input_newValue_2_listener($event) { ɵɵrestoreView(_r28); const ctx_r29 = ɵɵnextContext(); return ctx_r29.onCyanInput($event); });
        ɵɵelementEnd();
        ɵɵelementStart(3, "input", 31);
        ɵɵlistener("keyup.enter", function ColorPickerComponent_div_20_Template_input_keyup_enter_3_listener($event) { ɵɵrestoreView(_r28); const ctx_r30 = ɵɵnextContext(); return ctx_r30.onAcceptColor($event); })("newValue", function ColorPickerComponent_div_20_Template_input_newValue_3_listener($event) { ɵɵrestoreView(_r28); const ctx_r31 = ɵɵnextContext(); return ctx_r31.onMagentaInput($event); });
        ɵɵelementEnd();
        ɵɵelementStart(4, "input", 31);
        ɵɵlistener("keyup.enter", function ColorPickerComponent_div_20_Template_input_keyup_enter_4_listener($event) { ɵɵrestoreView(_r28); const ctx_r32 = ɵɵnextContext(); return ctx_r32.onAcceptColor($event); })("newValue", function ColorPickerComponent_div_20_Template_input_newValue_4_listener($event) { ɵɵrestoreView(_r28); const ctx_r33 = ɵɵnextContext(); return ctx_r33.onYellowInput($event); });
        ɵɵelementEnd();
        ɵɵelementStart(5, "input", 31);
        ɵɵlistener("keyup.enter", function ColorPickerComponent_div_20_Template_input_keyup_enter_5_listener($event) { ɵɵrestoreView(_r28); const ctx_r34 = ɵɵnextContext(); return ctx_r34.onAcceptColor($event); })("newValue", function ColorPickerComponent_div_20_Template_input_newValue_5_listener($event) { ɵɵrestoreView(_r28); const ctx_r35 = ɵɵnextContext(); return ctx_r35.onBlackInput($event); });
        ɵɵelementEnd();
        ɵɵtemplate(6, ColorPickerComponent_div_20_input_6_Template, 1, 2, "input", 32);
        ɵɵelementEnd();
        ɵɵelementStart(7, "div", 30);
        ɵɵelementStart(8, "div");
        ɵɵtext(9, "C");
        ɵɵelementEnd();
        ɵɵelementStart(10, "div");
        ɵɵtext(11, "M");
        ɵɵelementEnd();
        ɵɵelementStart(12, "div");
        ɵɵtext(13, "Y");
        ɵɵelementEnd();
        ɵɵelementStart(14, "div");
        ɵɵtext(15, "K");
        ɵɵelementEnd();
        ɵɵtemplate(16, ColorPickerComponent_div_20_div_16_Template, 2, 0, "div", 33);
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r8 = ɵɵnextContext();
        ɵɵstyleProp("display", ctx_r8.format !== 3 ? "none" : "block");
        ɵɵadvance(2);
        ɵɵproperty("rg", 100)("value", ctx_r8.cmykText == null ? null : ctx_r8.cmykText.c);
        ɵɵadvance(1);
        ɵɵproperty("rg", 100)("value", ctx_r8.cmykText == null ? null : ctx_r8.cmykText.m);
        ɵɵadvance(1);
        ɵɵproperty("rg", 100)("value", ctx_r8.cmykText == null ? null : ctx_r8.cmykText.y);
        ɵɵadvance(1);
        ɵɵproperty("rg", 100)("value", ctx_r8.cmykText == null ? null : ctx_r8.cmykText.k);
        ɵɵadvance(1);
        ɵɵproperty("ngIf", ctx_r8.cpAlphaChannel !== "disabled");
        ɵɵadvance(10);
        ɵɵproperty("ngIf", ctx_r8.cpAlphaChannel !== "disabled");
    } }
    function ColorPickerComponent_div_21_input_5_Template(rf, ctx) { if (rf & 1) {
        const _r39 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "input", 34);
        ɵɵlistener("keyup.enter", function ColorPickerComponent_div_21_input_5_Template_input_keyup_enter_0_listener($event) { ɵɵrestoreView(_r39); const ctx_r38 = ɵɵnextContext(2); return ctx_r38.onAcceptColor($event); })("newValue", function ColorPickerComponent_div_21_input_5_Template_input_newValue_0_listener($event) { ɵɵrestoreView(_r39); const ctx_r40 = ɵɵnextContext(2); return ctx_r40.onAlphaInput($event); });
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r36 = ɵɵnextContext(2);
        ɵɵproperty("rg", 1)("value", ctx_r36.hslaText == null ? null : ctx_r36.hslaText.a);
    } }
    function ColorPickerComponent_div_21_div_13_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div");
        ɵɵtext(1, "A");
        ɵɵelementEnd();
    } }
    function ColorPickerComponent_div_21_Template(rf, ctx) { if (rf & 1) {
        const _r42 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "div", 35);
        ɵɵelementStart(1, "div", 30);
        ɵɵelementStart(2, "input", 36);
        ɵɵlistener("keyup.enter", function ColorPickerComponent_div_21_Template_input_keyup_enter_2_listener($event) { ɵɵrestoreView(_r42); const ctx_r41 = ɵɵnextContext(); return ctx_r41.onAcceptColor($event); })("newValue", function ColorPickerComponent_div_21_Template_input_newValue_2_listener($event) { ɵɵrestoreView(_r42); const ctx_r43 = ɵɵnextContext(); return ctx_r43.onHueInput($event); });
        ɵɵelementEnd();
        ɵɵelementStart(3, "input", 31);
        ɵɵlistener("keyup.enter", function ColorPickerComponent_div_21_Template_input_keyup_enter_3_listener($event) { ɵɵrestoreView(_r42); const ctx_r44 = ɵɵnextContext(); return ctx_r44.onAcceptColor($event); })("newValue", function ColorPickerComponent_div_21_Template_input_newValue_3_listener($event) { ɵɵrestoreView(_r42); const ctx_r45 = ɵɵnextContext(); return ctx_r45.onSaturationInput($event); });
        ɵɵelementEnd();
        ɵɵelementStart(4, "input", 31);
        ɵɵlistener("keyup.enter", function ColorPickerComponent_div_21_Template_input_keyup_enter_4_listener($event) { ɵɵrestoreView(_r42); const ctx_r46 = ɵɵnextContext(); return ctx_r46.onAcceptColor($event); })("newValue", function ColorPickerComponent_div_21_Template_input_newValue_4_listener($event) { ɵɵrestoreView(_r42); const ctx_r47 = ɵɵnextContext(); return ctx_r47.onLightnessInput($event); });
        ɵɵelementEnd();
        ɵɵtemplate(5, ColorPickerComponent_div_21_input_5_Template, 1, 2, "input", 32);
        ɵɵelementEnd();
        ɵɵelementStart(6, "div", 30);
        ɵɵelementStart(7, "div");
        ɵɵtext(8, "H");
        ɵɵelementEnd();
        ɵɵelementStart(9, "div");
        ɵɵtext(10, "S");
        ɵɵelementEnd();
        ɵɵelementStart(11, "div");
        ɵɵtext(12, "L");
        ɵɵelementEnd();
        ɵɵtemplate(13, ColorPickerComponent_div_21_div_13_Template, 2, 0, "div", 33);
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r9 = ɵɵnextContext();
        ɵɵstyleProp("display", ctx_r9.format !== 2 ? "none" : "block");
        ɵɵadvance(2);
        ɵɵproperty("rg", 360)("value", ctx_r9.hslaText == null ? null : ctx_r9.hslaText.h);
        ɵɵadvance(1);
        ɵɵproperty("rg", 100)("value", ctx_r9.hslaText == null ? null : ctx_r9.hslaText.s);
        ɵɵadvance(1);
        ɵɵproperty("rg", 100)("value", ctx_r9.hslaText == null ? null : ctx_r9.hslaText.l);
        ɵɵadvance(1);
        ɵɵproperty("ngIf", ctx_r9.cpAlphaChannel !== "disabled");
        ɵɵadvance(8);
        ɵɵproperty("ngIf", ctx_r9.cpAlphaChannel !== "disabled");
    } }
    function ColorPickerComponent_div_22_input_5_Template(rf, ctx) { if (rf & 1) {
        const _r51 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "input", 34);
        ɵɵlistener("keyup.enter", function ColorPickerComponent_div_22_input_5_Template_input_keyup_enter_0_listener($event) { ɵɵrestoreView(_r51); const ctx_r50 = ɵɵnextContext(2); return ctx_r50.onAcceptColor($event); })("newValue", function ColorPickerComponent_div_22_input_5_Template_input_newValue_0_listener($event) { ɵɵrestoreView(_r51); const ctx_r52 = ɵɵnextContext(2); return ctx_r52.onAlphaInput($event); });
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r48 = ɵɵnextContext(2);
        ɵɵproperty("rg", 1)("value", ctx_r48.rgbaText == null ? null : ctx_r48.rgbaText.a);
    } }
    function ColorPickerComponent_div_22_div_13_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div");
        ɵɵtext(1, "A");
        ɵɵelementEnd();
    } }
    function ColorPickerComponent_div_22_Template(rf, ctx) { if (rf & 1) {
        const _r54 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "div", 37);
        ɵɵelementStart(1, "div", 30);
        ɵɵelementStart(2, "input", 38);
        ɵɵlistener("keyup.enter", function ColorPickerComponent_div_22_Template_input_keyup_enter_2_listener($event) { ɵɵrestoreView(_r54); const ctx_r53 = ɵɵnextContext(); return ctx_r53.onAcceptColor($event); })("newValue", function ColorPickerComponent_div_22_Template_input_newValue_2_listener($event) { ɵɵrestoreView(_r54); const ctx_r55 = ɵɵnextContext(); return ctx_r55.onRedInput($event); });
        ɵɵelementEnd();
        ɵɵelementStart(3, "input", 38);
        ɵɵlistener("keyup.enter", function ColorPickerComponent_div_22_Template_input_keyup_enter_3_listener($event) { ɵɵrestoreView(_r54); const ctx_r56 = ɵɵnextContext(); return ctx_r56.onAcceptColor($event); })("newValue", function ColorPickerComponent_div_22_Template_input_newValue_3_listener($event) { ɵɵrestoreView(_r54); const ctx_r57 = ɵɵnextContext(); return ctx_r57.onGreenInput($event); });
        ɵɵelementEnd();
        ɵɵelementStart(4, "input", 38);
        ɵɵlistener("keyup.enter", function ColorPickerComponent_div_22_Template_input_keyup_enter_4_listener($event) { ɵɵrestoreView(_r54); const ctx_r58 = ɵɵnextContext(); return ctx_r58.onAcceptColor($event); })("newValue", function ColorPickerComponent_div_22_Template_input_newValue_4_listener($event) { ɵɵrestoreView(_r54); const ctx_r59 = ɵɵnextContext(); return ctx_r59.onBlueInput($event); });
        ɵɵelementEnd();
        ɵɵtemplate(5, ColorPickerComponent_div_22_input_5_Template, 1, 2, "input", 32);
        ɵɵelementEnd();
        ɵɵelementStart(6, "div", 30);
        ɵɵelementStart(7, "div");
        ɵɵtext(8, "R");
        ɵɵelementEnd();
        ɵɵelementStart(9, "div");
        ɵɵtext(10, "G");
        ɵɵelementEnd();
        ɵɵelementStart(11, "div");
        ɵɵtext(12, "B");
        ɵɵelementEnd();
        ɵɵtemplate(13, ColorPickerComponent_div_22_div_13_Template, 2, 0, "div", 33);
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r10 = ɵɵnextContext();
        ɵɵstyleProp("display", ctx_r10.format !== 1 ? "none" : "block");
        ɵɵadvance(2);
        ɵɵproperty("rg", 255)("value", ctx_r10.rgbaText == null ? null : ctx_r10.rgbaText.r);
        ɵɵadvance(1);
        ɵɵproperty("rg", 255)("value", ctx_r10.rgbaText == null ? null : ctx_r10.rgbaText.g);
        ɵɵadvance(1);
        ɵɵproperty("rg", 255)("value", ctx_r10.rgbaText == null ? null : ctx_r10.rgbaText.b);
        ɵɵadvance(1);
        ɵɵproperty("ngIf", ctx_r10.cpAlphaChannel !== "disabled");
        ɵɵadvance(8);
        ɵɵproperty("ngIf", ctx_r10.cpAlphaChannel !== "disabled");
    } }
    function ColorPickerComponent_div_23_input_3_Template(rf, ctx) { if (rf & 1) {
        const _r63 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "input", 34);
        ɵɵlistener("keyup.enter", function ColorPickerComponent_div_23_input_3_Template_input_keyup_enter_0_listener($event) { ɵɵrestoreView(_r63); const ctx_r62 = ɵɵnextContext(2); return ctx_r62.onAcceptColor($event); })("newValue", function ColorPickerComponent_div_23_input_3_Template_input_newValue_0_listener($event) { ɵɵrestoreView(_r63); const ctx_r64 = ɵɵnextContext(2); return ctx_r64.onAlphaInput($event); });
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r60 = ɵɵnextContext(2);
        ɵɵproperty("rg", 1)("value", ctx_r60.hexAlpha);
    } }
    function ColorPickerComponent_div_23_div_7_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div");
        ɵɵtext(1, "A");
        ɵɵelementEnd();
    } }
    function ColorPickerComponent_div_23_Template(rf, ctx) { if (rf & 1) {
        const _r66 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "div", 39);
        ɵɵelementStart(1, "div", 30);
        ɵɵelementStart(2, "input", 40);
        ɵɵlistener("blur", function ColorPickerComponent_div_23_Template_input_blur_2_listener() { ɵɵrestoreView(_r66); const ctx_r65 = ɵɵnextContext(); return ctx_r65.onHexInput(null); })("keyup.enter", function ColorPickerComponent_div_23_Template_input_keyup_enter_2_listener($event) { ɵɵrestoreView(_r66); const ctx_r67 = ɵɵnextContext(); return ctx_r67.onAcceptColor($event); })("newValue", function ColorPickerComponent_div_23_Template_input_newValue_2_listener($event) { ɵɵrestoreView(_r66); const ctx_r68 = ɵɵnextContext(); return ctx_r68.onHexInput($event); });
        ɵɵelementEnd();
        ɵɵtemplate(3, ColorPickerComponent_div_23_input_3_Template, 1, 2, "input", 32);
        ɵɵelementEnd();
        ɵɵelementStart(4, "div", 30);
        ɵɵelementStart(5, "div");
        ɵɵtext(6, "Hex");
        ɵɵelementEnd();
        ɵɵtemplate(7, ColorPickerComponent_div_23_div_7_Template, 2, 0, "div", 33);
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r11 = ɵɵnextContext();
        ɵɵstyleProp("display", ctx_r11.format !== 0 ? "none" : "block");
        ɵɵclassProp("hex-alpha", ctx_r11.cpAlphaChannel === "forced");
        ɵɵadvance(2);
        ɵɵproperty("value", ctx_r11.hexText);
        ɵɵadvance(1);
        ɵɵproperty("ngIf", ctx_r11.cpAlphaChannel === "forced");
        ɵɵadvance(4);
        ɵɵproperty("ngIf", ctx_r11.cpAlphaChannel === "forced");
    } }
    function ColorPickerComponent_div_24_input_3_Template(rf, ctx) { if (rf & 1) {
        const _r71 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "input", 34);
        ɵɵlistener("keyup.enter", function ColorPickerComponent_div_24_input_3_Template_input_keyup_enter_0_listener($event) { ɵɵrestoreView(_r71); const ctx_r70 = ɵɵnextContext(2); return ctx_r70.onAcceptColor($event); })("newValue", function ColorPickerComponent_div_24_input_3_Template_input_newValue_0_listener($event) { ɵɵrestoreView(_r71); const ctx_r72 = ɵɵnextContext(2); return ctx_r72.onAlphaInput($event); });
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r69 = ɵɵnextContext(2);
        ɵɵproperty("rg", 1)("value", ctx_r69.hslaText == null ? null : ctx_r69.hslaText.a);
    } }
    function ColorPickerComponent_div_24_Template(rf, ctx) { if (rf & 1) {
        const _r74 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "div", 41);
        ɵɵelementStart(1, "div", 30);
        ɵɵelementStart(2, "input", 31);
        ɵɵlistener("keyup.enter", function ColorPickerComponent_div_24_Template_input_keyup_enter_2_listener($event) { ɵɵrestoreView(_r74); const ctx_r73 = ɵɵnextContext(); return ctx_r73.onAcceptColor($event); })("newValue", function ColorPickerComponent_div_24_Template_input_newValue_2_listener($event) { ɵɵrestoreView(_r74); const ctx_r75 = ɵɵnextContext(); return ctx_r75.onValueInput($event); });
        ɵɵelementEnd();
        ɵɵtemplate(3, ColorPickerComponent_div_24_input_3_Template, 1, 2, "input", 32);
        ɵɵelementEnd();
        ɵɵelementStart(4, "div", 30);
        ɵɵelementStart(5, "div");
        ɵɵtext(6, "V");
        ɵɵelementEnd();
        ɵɵelementStart(7, "div");
        ɵɵtext(8, "A");
        ɵɵelementEnd();
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r12 = ɵɵnextContext();
        ɵɵadvance(2);
        ɵɵproperty("rg", 100)("value", ctx_r12.hslaText == null ? null : ctx_r12.hslaText.l);
        ɵɵadvance(1);
        ɵɵproperty("ngIf", ctx_r12.cpAlphaChannel !== "disabled");
    } }
    function ColorPickerComponent_div_25_Template(rf, ctx) { if (rf & 1) {
        const _r77 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "div", 42);
        ɵɵelementStart(1, "span", 43);
        ɵɵlistener("click", function ColorPickerComponent_div_25_Template_span_click_1_listener() { ɵɵrestoreView(_r77); const ctx_r76 = ɵɵnextContext(); return ctx_r76.onFormatToggle(0 - 1); });
        ɵɵelementEnd();
        ɵɵelementStart(2, "span", 43);
        ɵɵlistener("click", function ColorPickerComponent_div_25_Template_span_click_2_listener() { ɵɵrestoreView(_r77); const ctx_r78 = ɵɵnextContext(); return ctx_r78.onFormatToggle(1); });
        ɵɵelementEnd();
        ɵɵelementEnd();
    } }
    function ColorPickerComponent_div_26_div_4_div_1_span_1_Template(rf, ctx) { if (rf & 1) {
        const _r85 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "span", 50);
        ɵɵlistener("click", function ColorPickerComponent_div_26_div_4_div_1_span_1_Template_span_click_0_listener($event) { ɵɵrestoreView(_r85); const color_r82 = ɵɵnextContext().$implicit; const ctx_r84 = ɵɵnextContext(3); return ctx_r84.onRemovePresetColor($event, color_r82); });
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r83 = ɵɵnextContext(4);
        ɵɵclassMap(ctx_r83.cpRemoveColorButtonClass);
    } }
    function ColorPickerComponent_div_26_div_4_div_1_Template(rf, ctx) { if (rf & 1) {
        const _r88 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "div", 48);
        ɵɵlistener("click", function ColorPickerComponent_div_26_div_4_div_1_Template_div_click_0_listener() { ɵɵrestoreView(_r88); const color_r82 = ctx.$implicit; const ctx_r87 = ɵɵnextContext(3); return ctx_r87.setColorFromString(color_r82); });
        ɵɵtemplate(1, ColorPickerComponent_div_26_div_4_div_1_span_1_Template, 1, 3, "span", 49);
        ɵɵelementEnd();
    } if (rf & 2) {
        const color_r82 = ctx.$implicit;
        const ctx_r81 = ɵɵnextContext(3);
        ɵɵstyleProp("background-color", color_r82);
        ɵɵadvance(1);
        ɵɵproperty("ngIf", ctx_r81.cpAddColorButton);
    } }
    function ColorPickerComponent_div_26_div_4_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div");
        ɵɵtemplate(1, ColorPickerComponent_div_26_div_4_div_1_Template, 2, 3, "div", 47);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r79 = ɵɵnextContext(2);
        ɵɵclassMap(ctx_r79.cpPresetColorsClass);
        ɵɵadvance(1);
        ɵɵproperty("ngForOf", ctx_r79.cpPresetColors);
    } }
    function ColorPickerComponent_div_26_div_5_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div");
        ɵɵtext(1);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r80 = ɵɵnextContext(2);
        ɵɵclassMap(ctx_r80.cpPresetEmptyMessageClass);
        ɵɵadvance(1);
        ɵɵtextInterpolate(ctx_r80.cpPresetEmptyMessage);
    } }
    function ColorPickerComponent_div_26_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 44);
        ɵɵelement(1, "hr");
        ɵɵelementStart(2, "div", 45);
        ɵɵtext(3);
        ɵɵelementEnd();
        ɵɵtemplate(4, ColorPickerComponent_div_26_div_4_Template, 2, 4, "div", 46);
        ɵɵtemplate(5, ColorPickerComponent_div_26_div_5_Template, 2, 4, "div", 46);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r14 = ɵɵnextContext();
        ɵɵadvance(3);
        ɵɵtextInterpolate(ctx_r14.cpPresetLabel);
        ɵɵadvance(1);
        ɵɵproperty("ngIf", ctx_r14.cpPresetColors == null ? null : ctx_r14.cpPresetColors.length);
        ɵɵadvance(1);
        ɵɵproperty("ngIf", !(ctx_r14.cpPresetColors == null ? null : ctx_r14.cpPresetColors.length) && ctx_r14.cpAddColorButton);
    } }
    function ColorPickerComponent_div_27_button_1_Template(rf, ctx) { if (rf & 1) {
        const _r92 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "button", 53);
        ɵɵlistener("click", function ColorPickerComponent_div_27_button_1_Template_button_click_0_listener($event) { ɵɵrestoreView(_r92); const ctx_r91 = ɵɵnextContext(2); return ctx_r91.onCancelColor($event); });
        ɵɵtext(1);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r89 = ɵɵnextContext(2);
        ɵɵclassMap(ctx_r89.cpCancelButtonClass);
        ɵɵadvance(1);
        ɵɵtextInterpolate(ctx_r89.cpCancelButtonText);
    } }
    function ColorPickerComponent_div_27_button_2_Template(rf, ctx) { if (rf & 1) {
        const _r94 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "button", 53);
        ɵɵlistener("click", function ColorPickerComponent_div_27_button_2_Template_button_click_0_listener($event) { ɵɵrestoreView(_r94); const ctx_r93 = ɵɵnextContext(2); return ctx_r93.onAcceptColor($event); });
        ɵɵtext(1);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r90 = ɵɵnextContext(2);
        ɵɵclassMap(ctx_r90.cpOKButtonClass);
        ɵɵadvance(1);
        ɵɵtextInterpolate(ctx_r90.cpOKButtonText);
    } }
    function ColorPickerComponent_div_27_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 51);
        ɵɵtemplate(1, ColorPickerComponent_div_27_button_1_Template, 2, 4, "button", 52);
        ɵɵtemplate(2, ColorPickerComponent_div_27_button_2_Template, 2, 4, "button", 52);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r15 = ɵɵnextContext();
        ɵɵadvance(1);
        ɵɵproperty("ngIf", ctx_r15.cpCancelButton);
        ɵɵadvance(1);
        ɵɵproperty("ngIf", ctx_r15.cpOKButton);
    } }
    var ColorFormats;
    (function (ColorFormats) {
        ColorFormats[ColorFormats["HEX"] = 0] = "HEX";
        ColorFormats[ColorFormats["RGBA"] = 1] = "RGBA";
        ColorFormats[ColorFormats["HSLA"] = 2] = "HSLA";
        ColorFormats[ColorFormats["CMYK"] = 3] = "CMYK";
    })(ColorFormats || (ColorFormats = {}));
    class Rgba {
        constructor(r, g, b, a) {
            this.r = r;
            this.g = g;
            this.b = b;
            this.a = a;
        }
    }
    class Hsva {
        constructor(h, s, v, a) {
            this.h = h;
            this.s = s;
            this.v = v;
            this.a = a;
        }
    }
    class Hsla {
        constructor(h, s, l, a) {
            this.h = h;
            this.s = s;
            this.l = l;
            this.a = a;
        }
    }
    class Cmyk {
        constructor(c, m, y, k, a = 1) {
            this.c = c;
            this.m = m;
            this.y = y;
            this.k = k;
            this.a = a;
        }
    }

    function detectIE() {
        let ua = '';
        if (typeof navigator !== 'undefined') {
            ua = navigator.userAgent.toLowerCase();
        }
        const msie = ua.indexOf('msie ');
        if (msie > 0) {
            // IE 10 or older => return version number
            return parseInt(ua.substring(msie + 5, ua.indexOf('.', msie)), 10);
        }
        // Other browser
        return false;
    }
    let TextDirective = class TextDirective {
        constructor() {
            this.newValue = new EventEmitter();
        }
        inputChange(event) {
            const value = event.target.value;
            if (this.rg === undefined) {
                this.newValue.emit(value);
            }
            else {
                const numeric = parseFloat(value);
                this.newValue.emit({ v: numeric, rg: this.rg });
            }
        }
    };
    TextDirective.ɵfac = function TextDirective_Factory(t) { return new (t || TextDirective)(); };
    TextDirective.ɵdir = ɵɵdefineDirective({ type: TextDirective, selectors: [["", "text", ""]], hostBindings: function TextDirective_HostBindings(rf, ctx) { if (rf & 1) {
            ɵɵlistener("input", function TextDirective_input_HostBindingHandler($event) { return ctx.inputChange($event); });
        } }, inputs: { rg: "rg", text: "text" }, outputs: { newValue: "newValue" } });
    __decorate([
        Input()
    ], TextDirective.prototype, "rg", void 0);
    __decorate([
        Input()
    ], TextDirective.prototype, "text", void 0);
    __decorate([
        Output()
    ], TextDirective.prototype, "newValue", void 0);
    __decorate([
        HostListener('input', ['$event'])
    ], TextDirective.prototype, "inputChange", null);
    let SliderDirective = class SliderDirective {
        constructor(elRef) {
            this.elRef = elRef;
            this.dragEnd = new EventEmitter();
            this.dragStart = new EventEmitter();
            this.newValue = new EventEmitter();
            this.listenerMove = (event) => this.move(event);
            this.listenerStop = () => this.stop();
        }
        mouseDown(event) {
            this.start(event);
        }
        touchStart(event) {
            this.start(event);
        }
        move(event) {
            event.preventDefault();
            this.setCursor(event);
        }
        start(event) {
            this.setCursor(event);
            event.stopPropagation();
            document.addEventListener('mouseup', this.listenerStop);
            document.addEventListener('touchend', this.listenerStop);
            document.addEventListener('mousemove', this.listenerMove);
            document.addEventListener('touchmove', this.listenerMove);
            this.dragStart.emit();
        }
        stop() {
            document.removeEventListener('mouseup', this.listenerStop);
            document.removeEventListener('touchend', this.listenerStop);
            document.removeEventListener('mousemove', this.listenerMove);
            document.removeEventListener('touchmove', this.listenerMove);
            this.dragEnd.emit();
        }
        getX(event) {
            const position = this.elRef.nativeElement.getBoundingClientRect();
            const pageX = (event.pageX !== undefined) ? event.pageX : event.touches[0].pageX;
            return pageX - position.left - window.pageXOffset;
        }
        getY(event) {
            const position = this.elRef.nativeElement.getBoundingClientRect();
            const pageY = (event.pageY !== undefined) ? event.pageY : event.touches[0].pageY;
            return pageY - position.top - window.pageYOffset;
        }
        setCursor(event) {
            const width = this.elRef.nativeElement.offsetWidth;
            const height = this.elRef.nativeElement.offsetHeight;
            const x = Math.max(0, Math.min(this.getX(event), width));
            const y = Math.max(0, Math.min(this.getY(event), height));
            if (this.rgX !== undefined && this.rgY !== undefined) {
                this.newValue.emit({ s: x / width, v: (1 - y / height), rgX: this.rgX, rgY: this.rgY });
            }
            else if (this.rgX === undefined && this.rgY !== undefined) {
                this.newValue.emit({ v: y / height, rgY: this.rgY });
            }
            else if (this.rgX !== undefined && this.rgY === undefined) {
                this.newValue.emit({ v: x / width, rgX: this.rgX });
            }
        }
    };
    SliderDirective.ɵfac = function SliderDirective_Factory(t) { return new (t || SliderDirective)(ɵɵdirectiveInject(ElementRef)); };
    SliderDirective.ɵdir = ɵɵdefineDirective({ type: SliderDirective, selectors: [["", "slider", ""]], hostBindings: function SliderDirective_HostBindings(rf, ctx) { if (rf & 1) {
            ɵɵlistener("mousedown", function SliderDirective_mousedown_HostBindingHandler($event) { return ctx.mouseDown($event); })("touchstart", function SliderDirective_touchstart_HostBindingHandler($event) { return ctx.touchStart($event); });
        } }, inputs: { rgX: "rgX", rgY: "rgY", slider: "slider" }, outputs: { dragEnd: "dragEnd", dragStart: "dragStart", newValue: "newValue" } });
    SliderDirective.ctorParameters = () => [
        { type: ElementRef }
    ];
    __decorate([
        Input()
    ], SliderDirective.prototype, "rgX", void 0);
    __decorate([
        Input()
    ], SliderDirective.prototype, "rgY", void 0);
    __decorate([
        Input()
    ], SliderDirective.prototype, "slider", void 0);
    __decorate([
        Output()
    ], SliderDirective.prototype, "dragEnd", void 0);
    __decorate([
        Output()
    ], SliderDirective.prototype, "dragStart", void 0);
    __decorate([
        Output()
    ], SliderDirective.prototype, "newValue", void 0);
    __decorate([
        HostListener('mousedown', ['$event'])
    ], SliderDirective.prototype, "mouseDown", null);
    __decorate([
        HostListener('touchstart', ['$event'])
    ], SliderDirective.prototype, "touchStart", null);
    class SliderPosition {
        constructor(h, s, v, a) {
            this.h = h;
            this.s = s;
            this.v = v;
            this.a = a;
        }
    }
    class SliderDimension {
        constructor(h, s, v, a) {
            this.h = h;
            this.s = s;
            this.v = v;
            this.a = a;
        }
    }

    let ColorPickerService = class ColorPickerService {
        constructor() {
            this.active = null;
        }
        setActive(active) {
            if (this.active && this.active !== active && this.active.cpDialogDisplay !== 'inline') {
                this.active.closeDialog();
            }
            this.active = active;
        }
        hsva2hsla(hsva) {
            const h = hsva.h, s = hsva.s, v = hsva.v, a = hsva.a;
            if (v === 0) {
                return new Hsla(h, 0, 0, a);
            }
            else if (s === 0 && v === 1) {
                return new Hsla(h, 1, 1, a);
            }
            else {
                const l = v * (2 - s) / 2;
                return new Hsla(h, v * s / (1 - Math.abs(2 * l - 1)), l, a);
            }
        }
        hsla2hsva(hsla) {
            const h = Math.min(hsla.h, 1), s = Math.min(hsla.s, 1);
            const l = Math.min(hsla.l, 1), a = Math.min(hsla.a, 1);
            if (l === 0) {
                return new Hsva(h, 0, 0, a);
            }
            else {
                const v = l + s * (1 - Math.abs(2 * l - 1)) / 2;
                return new Hsva(h, 2 * (v - l) / v, v, a);
            }
        }
        hsvaToRgba(hsva) {
            let r, g, b;
            const h = hsva.h, s = hsva.s, v = hsva.v, a = hsva.a;
            const i = Math.floor(h * 6);
            const f = h * 6 - i;
            const p = v * (1 - s);
            const q = v * (1 - f * s);
            const t = v * (1 - (1 - f) * s);
            switch (i % 6) {
                case 0:
                    r = v, g = t, b = p;
                    break;
                case 1:
                    r = q, g = v, b = p;
                    break;
                case 2:
                    r = p, g = v, b = t;
                    break;
                case 3:
                    r = p, g = q, b = v;
                    break;
                case 4:
                    r = t, g = p, b = v;
                    break;
                case 5:
                    r = v, g = p, b = q;
                    break;
                default:
                    r = 0, g = 0, b = 0;
            }
            return new Rgba(r, g, b, a);
        }
        cmykToRgb(cmyk) {
            const r = (1 - cmyk.c) * (1 - cmyk.k);
            const g = (1 - cmyk.m) * (1 - cmyk.k);
            const b = (1 - cmyk.y) * (1 - cmyk.k);
            return new Rgba(r, g, b, cmyk.a);
        }
        rgbaToCmyk(rgba) {
            const k = 1 - Math.max(rgba.r, rgba.g, rgba.b);
            if (k === 1) {
                return new Cmyk(0, 0, 0, 1, rgba.a);
            }
            else {
                const c = (1 - rgba.r - k) / (1 - k);
                const m = (1 - rgba.g - k) / (1 - k);
                const y = (1 - rgba.b - k) / (1 - k);
                return new Cmyk(c, m, y, k, rgba.a);
            }
        }
        rgbaToHsva(rgba) {
            let h, s;
            const r = Math.min(rgba.r, 1), g = Math.min(rgba.g, 1);
            const b = Math.min(rgba.b, 1), a = Math.min(rgba.a, 1);
            const max = Math.max(r, g, b), min = Math.min(r, g, b);
            const v = max, d = max - min;
            s = (max === 0) ? 0 : d / max;
            if (max === min) {
                h = 0;
            }
            else {
                switch (max) {
                    case r:
                        h = (g - b) / d + (g < b ? 6 : 0);
                        break;
                    case g:
                        h = (b - r) / d + 2;
                        break;
                    case b:
                        h = (r - g) / d + 4;
                        break;
                    default:
                        h = 0;
                }
                h /= 6;
            }
            return new Hsva(h, s, v, a);
        }
        rgbaToHex(rgba, allowHex8) {
            /* tslint:disable:no-bitwise */
            let hex = '#' + ((1 << 24) | (rgba.r << 16) | (rgba.g << 8) | rgba.b).toString(16).substr(1);
            if (allowHex8) {
                hex += ((1 << 8) | Math.round(rgba.a * 255)).toString(16).substr(1);
            }
            /* tslint:enable:no-bitwise */
            return hex;
        }
        normalizeCMYK(cmyk) {
            return new Cmyk(cmyk.c / 100, cmyk.m / 100, cmyk.y / 100, cmyk.k / 100, cmyk.a);
        }
        denormalizeCMYK(cmyk) {
            return new Cmyk(Math.floor(cmyk.c * 100), Math.floor(cmyk.m * 100), Math.floor(cmyk.y * 100), Math.floor(cmyk.k * 100), cmyk.a);
        }
        denormalizeRGBA(rgba) {
            return new Rgba(Math.round(rgba.r * 255), Math.round(rgba.g * 255), Math.round(rgba.b * 255), rgba.a);
        }
        stringToHsva(colorString = '', allowHex8 = false) {
            let hsva = null;
            colorString = (colorString || '').toLowerCase();
            const stringParsers = [
                {
                    re: /(rgb)a?\(\s*(\d{1,3})\s*,\s*(\d{1,3})\s*%?,\s*(\d{1,3})\s*%?(?:,\s*(\d+(?:\.\d+)?)\s*)?\)/,
                    parse: function (execResult) {
                        return new Rgba(parseInt(execResult[2], 10) / 255, parseInt(execResult[3], 10) / 255, parseInt(execResult[4], 10) / 255, isNaN(parseFloat(execResult[5])) ? 1 : parseFloat(execResult[5]));
                    }
                }, {
                    re: /(hsl)a?\(\s*(\d{1,3})\s*,\s*(\d{1,3})%\s*,\s*(\d{1,3})%\s*(?:,\s*(\d+(?:\.\d+)?)\s*)?\)/,
                    parse: function (execResult) {
                        return new Hsla(parseInt(execResult[2], 10) / 360, parseInt(execResult[3], 10) / 100, parseInt(execResult[4], 10) / 100, isNaN(parseFloat(execResult[5])) ? 1 : parseFloat(execResult[5]));
                    }
                }
            ];
            if (allowHex8) {
                stringParsers.push({
                    re: /#([a-fA-F0-9]{2})([a-fA-F0-9]{2})([a-fA-F0-9]{2})([a-fA-F0-9]{2})?$/,
                    parse: function (execResult) {
                        return new Rgba(parseInt(execResult[1], 16) / 255, parseInt(execResult[2], 16) / 255, parseInt(execResult[3], 16) / 255, parseInt(execResult[4] || 'FF', 16) / 255);
                    }
                });
            }
            else {
                stringParsers.push({
                    re: /#([a-fA-F0-9]{2})([a-fA-F0-9]{2})([a-fA-F0-9]{2})$/,
                    parse: function (execResult) {
                        return new Rgba(parseInt(execResult[1], 16) / 255, parseInt(execResult[2], 16) / 255, parseInt(execResult[3], 16) / 255, 1);
                    }
                });
            }
            stringParsers.push({
                re: /#([a-fA-F0-9])([a-fA-F0-9])([a-fA-F0-9])$/,
                parse: function (execResult) {
                    return new Rgba(parseInt(execResult[1] + execResult[1], 16) / 255, parseInt(execResult[2] + execResult[2], 16) / 255, parseInt(execResult[3] + execResult[3], 16) / 255, 1);
                }
            });
            for (const key in stringParsers) {
                if (stringParsers.hasOwnProperty(key)) {
                    const parser = stringParsers[key];
                    const match = parser.re.exec(colorString), color = match && parser.parse(match);
                    if (color) {
                        if (color instanceof Rgba) {
                            hsva = this.rgbaToHsva(color);
                        }
                        else if (color instanceof Hsla) {
                            hsva = this.hsla2hsva(color);
                        }
                        return hsva;
                    }
                }
            }
            return hsva;
        }
        outputFormat(hsva, outputFormat, alphaChannel) {
            if (outputFormat === 'auto') {
                outputFormat = hsva.a < 1 ? 'rgba' : 'hex';
            }
            switch (outputFormat) {
                case 'hsla':
                    const hsla = this.hsva2hsla(hsva);
                    const hslaText = new Hsla(Math.round((hsla.h) * 360), Math.round(hsla.s * 100), Math.round(hsla.l * 100), Math.round(hsla.a * 100) / 100);
                    if (hsva.a < 1 || alphaChannel === 'always') {
                        return 'hsla(' + hslaText.h + ',' + hslaText.s + '%,' + hslaText.l + '%,' +
                            hslaText.a + ')';
                    }
                    else {
                        return 'hsl(' + hslaText.h + ',' + hslaText.s + '%,' + hslaText.l + '%)';
                    }
                case 'rgba':
                    const rgba = this.denormalizeRGBA(this.hsvaToRgba(hsva));
                    if (hsva.a < 1 || alphaChannel === 'always') {
                        return 'rgba(' + rgba.r + ',' + rgba.g + ',' + rgba.b + ',' +
                            Math.round(rgba.a * 100) / 100 + ')';
                    }
                    else {
                        return 'rgb(' + rgba.r + ',' + rgba.g + ',' + rgba.b + ')';
                    }
                default:
                    const allowHex8 = (alphaChannel === 'always' || alphaChannel === 'forced');
                    return this.rgbaToHex(this.denormalizeRGBA(this.hsvaToRgba(hsva)), allowHex8);
            }
        }
    };
    ColorPickerService.ɵfac = function ColorPickerService_Factory(t) { return new (t || ColorPickerService)(); };
    ColorPickerService.ɵprov = ɵɵdefineInjectable({ token: ColorPickerService, factory: ColorPickerService.ɵfac });

    let ColorPickerComponent = class ColorPickerComponent {
        constructor(elRef, cdRef, service) {
            this.elRef = elRef;
            this.cdRef = cdRef;
            this.service = service;
            this.isIE10 = false;
            this.dialogArrowSize = 10;
            this.dialogArrowOffset = 15;
            this.dialogInputFields = [
                ColorFormats.HEX,
                ColorFormats.RGBA,
                ColorFormats.HSLA,
                ColorFormats.CMYK
            ];
            this.useRootViewContainer = false;
        }
        handleEsc(event) {
            if (this.show && this.cpDialogDisplay === 'popup') {
                this.onCancelColor(event);
            }
        }
        handleEnter(event) {
            if (this.show && this.cpDialogDisplay === 'popup') {
                this.onAcceptColor(event);
            }
        }
        ngOnInit() {
            this.slider = new SliderPosition(0, 0, 0, 0);
            const hueWidth = this.hueSlider.nativeElement.offsetWidth || 140;
            const alphaWidth = this.alphaSlider.nativeElement.offsetWidth || 140;
            this.sliderDimMax = new SliderDimension(hueWidth, this.cpWidth, 130, alphaWidth);
            if (this.cpCmykEnabled) {
                this.format = ColorFormats.CMYK;
            }
            else if (this.cpOutputFormat === 'rgba') {
                this.format = ColorFormats.RGBA;
            }
            else if (this.cpOutputFormat === 'hsla') {
                this.format = ColorFormats.HSLA;
            }
            else {
                this.format = ColorFormats.HEX;
            }
            this.listenerMouseDown = (event) => { this.onMouseDown(event); };
            this.listenerResize = () => { this.onResize(); };
            this.openDialog(this.initialColor, false);
        }
        ngOnDestroy() {
            this.closeDialog();
        }
        ngAfterViewInit() {
            if (this.cpWidth !== 230 || this.cpDialogDisplay === 'inline') {
                const hueWidth = this.hueSlider.nativeElement.offsetWidth || 140;
                const alphaWidth = this.alphaSlider.nativeElement.offsetWidth || 140;
                this.sliderDimMax = new SliderDimension(hueWidth, this.cpWidth, 130, alphaWidth);
                this.updateColorPicker(false);
                this.cdRef.detectChanges();
            }
        }
        openDialog(color, emit = true) {
            this.service.setActive(this);
            if (!this.width) {
                this.cpWidth = this.directiveElementRef.nativeElement.offsetWidth;
            }
            if (!this.height) {
                this.height = 320;
            }
            this.setInitialColor(color);
            this.setColorFromString(color, emit);
            this.openColorPicker();
        }
        closeDialog() {
            this.closeColorPicker();
        }
        setupDialog(instance, elementRef, color, cpWidth, cpHeight, cpDialogDisplay, cpFallbackColor, cpColorMode, cpCmykEnabled, cpAlphaChannel, cpOutputFormat, cpDisableInput, cpIgnoredElements, cpSaveClickOutside, cpCloseClickOutside, cpUseRootViewContainer, cpPosition, cpPositionOffset, cpPositionRelativeToArrow, cpPresetLabel, cpPresetColors, cpPresetColorsClass, cpMaxPresetColorsLength, cpPresetEmptyMessage, cpPresetEmptyMessageClass, cpOKButton, cpOKButtonClass, cpOKButtonText, cpCancelButton, cpCancelButtonClass, cpCancelButtonText, cpAddColorButton, cpAddColorButtonClass, cpAddColorButtonText, cpRemoveColorButtonClass) {
            this.setInitialColor(color);
            this.setColorMode(cpColorMode);
            this.isIE10 = (detectIE() === 10);
            this.directiveInstance = instance;
            this.directiveElementRef = elementRef;
            this.cpDisableInput = cpDisableInput;
            this.cpCmykEnabled = cpCmykEnabled;
            this.cpAlphaChannel = cpAlphaChannel;
            this.cpOutputFormat = cpOutputFormat;
            this.cpDialogDisplay = cpDialogDisplay;
            this.cpIgnoredElements = cpIgnoredElements;
            this.cpSaveClickOutside = cpSaveClickOutside;
            this.cpCloseClickOutside = cpCloseClickOutside;
            this.useRootViewContainer = cpUseRootViewContainer;
            this.width = this.cpWidth = parseInt(cpWidth, 10);
            this.height = this.cpHeight = parseInt(cpHeight, 10);
            this.cpPosition = cpPosition;
            this.cpPositionOffset = parseInt(cpPositionOffset, 10);
            this.cpOKButton = cpOKButton;
            this.cpOKButtonText = cpOKButtonText;
            this.cpOKButtonClass = cpOKButtonClass;
            this.cpCancelButton = cpCancelButton;
            this.cpCancelButtonText = cpCancelButtonText;
            this.cpCancelButtonClass = cpCancelButtonClass;
            this.fallbackColor = cpFallbackColor || '#fff';
            this.setPresetConfig(cpPresetLabel, cpPresetColors);
            this.cpPresetColorsClass = cpPresetColorsClass;
            this.cpMaxPresetColorsLength = cpMaxPresetColorsLength;
            this.cpPresetEmptyMessage = cpPresetEmptyMessage;
            this.cpPresetEmptyMessageClass = cpPresetEmptyMessageClass;
            this.cpAddColorButton = cpAddColorButton;
            this.cpAddColorButtonText = cpAddColorButtonText;
            this.cpAddColorButtonClass = cpAddColorButtonClass;
            this.cpRemoveColorButtonClass = cpRemoveColorButtonClass;
            if (!cpPositionRelativeToArrow) {
                this.dialogArrowOffset = 0;
            }
            if (cpDialogDisplay === 'inline') {
                this.dialogArrowSize = 0;
                this.dialogArrowOffset = 0;
            }
            if (cpOutputFormat === 'hex' &&
                cpAlphaChannel !== 'always' && cpAlphaChannel !== 'forced') {
                this.cpAlphaChannel = 'disabled';
            }
        }
        setColorMode(mode) {
            switch (mode.toString().toUpperCase()) {
                case '1':
                case 'C':
                case 'COLOR':
                    this.cpColorMode = 1;
                    break;
                case '2':
                case 'G':
                case 'GRAYSCALE':
                    this.cpColorMode = 2;
                    break;
                case '3':
                case 'P':
                case 'PRESETS':
                    this.cpColorMode = 3;
                    break;
                default:
                    this.cpColorMode = 1;
            }
        }
        setInitialColor(color) {
            this.initialColor = color;
        }
        setPresetConfig(cpPresetLabel, cpPresetColors) {
            this.cpPresetLabel = cpPresetLabel;
            this.cpPresetColors = cpPresetColors;
        }
        setColorFromString(value, emit = true, update = true) {
            let hsva;
            if (this.cpAlphaChannel === 'always' || this.cpAlphaChannel === 'forced') {
                hsva = this.service.stringToHsva(value, true);
                if (!hsva && !this.hsva) {
                    hsva = this.service.stringToHsva(value, false);
                }
            }
            else {
                hsva = this.service.stringToHsva(value, false);
            }
            if (!hsva && !this.hsva) {
                hsva = this.service.stringToHsva(this.fallbackColor, false);
            }
            if (hsva) {
                this.hsva = hsva;
                this.sliderH = this.hsva.h;
                if (this.cpOutputFormat === 'hex' && this.cpAlphaChannel === 'disabled') {
                    this.hsva.a = 1;
                }
                this.updateColorPicker(emit, update);
            }
        }
        onResize() {
            if (this.position === 'fixed') {
                this.setDialogPosition();
            }
            else if (this.cpDialogDisplay !== 'inline') {
                this.closeColorPicker();
            }
        }
        onDragEnd(slider) {
            this.directiveInstance.sliderDragEnd({ slider: slider, color: this.outputColor });
        }
        onDragStart(slider) {
            this.directiveInstance.sliderDragStart({ slider: slider, color: this.outputColor });
        }
        onMouseDown(event) {
            if (this.show &&
                !this.isIE10 && this.cpDialogDisplay === 'popup' &&
                event.target !== this.directiveElementRef.nativeElement &&
                !this.isDescendant(this.elRef.nativeElement, event.target) &&
                !this.isDescendant(this.directiveElementRef.nativeElement, event.target) &&
                this.cpIgnoredElements.filter((item) => item === event.target).length === 0) {
                if (this.cpSaveClickOutside) {
                    this.directiveInstance.colorSelected(this.outputColor);
                }
                else {
                    this.hsva = null;
                    this.setColorFromString(this.initialColor, false);
                    if (this.cpCmykEnabled) {
                        this.directiveInstance.cmykChanged(this.cmykColor);
                    }
                    this.directiveInstance.colorChanged(this.initialColor);
                    this.directiveInstance.colorCanceled();
                }
                if (this.cpCloseClickOutside) {
                    this.closeColorPicker();
                }
            }
        }
        onAcceptColor(event) {
            event.stopPropagation();
            if (this.outputColor) {
                this.directiveInstance.colorSelected(this.outputColor);
            }
            if (this.cpDialogDisplay === 'popup') {
                this.closeColorPicker();
            }
        }
        onCancelColor(event) {
            this.hsva = null;
            event.stopPropagation();
            this.directiveInstance.colorCanceled();
            this.setColorFromString(this.initialColor, true);
            if (this.cpDialogDisplay === 'popup') {
                if (this.cpCmykEnabled) {
                    this.directiveInstance.cmykChanged(this.cmykColor);
                }
                this.directiveInstance.colorChanged(this.initialColor, true);
                this.closeColorPicker();
            }
        }
        onFormatToggle(change) {
            const availableFormats = this.dialogInputFields.length -
                (this.cpCmykEnabled ? 0 : 1);
            const nextFormat = (((this.dialogInputFields.indexOf(this.format) + change) %
                availableFormats) + availableFormats) % availableFormats;
            this.format = this.dialogInputFields[nextFormat];
        }
        onColorChange(value) {
            this.hsva.s = value.s / value.rgX;
            this.hsva.v = value.v / value.rgY;
            this.updateColorPicker();
            this.directiveInstance.sliderChanged({
                slider: 'lightness',
                value: this.hsva.v,
                color: this.outputColor
            });
            this.directiveInstance.sliderChanged({
                slider: 'saturation',
                value: this.hsva.s,
                color: this.outputColor
            });
        }
        onHueChange(value) {
            this.hsva.h = value.v / value.rgX;
            this.sliderH = this.hsva.h;
            this.updateColorPicker();
            this.directiveInstance.sliderChanged({
                slider: 'hue',
                value: this.hsva.h,
                color: this.outputColor
            });
        }
        onValueChange(value) {
            this.hsva.v = value.v / value.rgX;
            this.updateColorPicker();
            this.directiveInstance.sliderChanged({
                slider: 'value',
                value: this.hsva.v,
                color: this.outputColor
            });
        }
        onAlphaChange(value) {
            this.hsva.a = value.v / value.rgX;
            this.updateColorPicker();
            this.directiveInstance.sliderChanged({
                slider: 'alpha',
                value: this.hsva.a,
                color: this.outputColor
            });
        }
        onHexInput(value) {
            if (value === null) {
                this.updateColorPicker();
            }
            else {
                if (value && value[0] !== '#') {
                    value = '#' + value;
                }
                let validHex = /^#([a-f0-9]{3}|[a-f0-9]{6})$/gi;
                if (this.cpAlphaChannel === 'always') {
                    validHex = /^#([a-f0-9]{3}|[a-f0-9]{6}|[a-f0-9]{8})$/gi;
                }
                const valid = validHex.test(value);
                if (valid) {
                    if (value.length < 5) {
                        value = '#' + value.substring(1)
                            .split('')
                            .map(c => c + c)
                            .join('');
                    }
                    if (this.cpAlphaChannel === 'forced') {
                        value += Math.round(this.hsva.a * 255).toString(16);
                    }
                    this.setColorFromString(value, true, false);
                }
                this.directiveInstance.inputChanged({
                    input: 'hex',
                    valid: valid,
                    value: value,
                    color: this.outputColor
                });
            }
        }
        onRedInput(value) {
            const rgba = this.service.hsvaToRgba(this.hsva);
            const valid = !isNaN(value.v) && value.v >= 0 && value.v <= value.rg;
            if (valid) {
                rgba.r = value.v / value.rg;
                this.hsva = this.service.rgbaToHsva(rgba);
                this.sliderH = this.hsva.h;
                this.updateColorPicker();
            }
            this.directiveInstance.inputChanged({
                input: 'red',
                valid: valid,
                value: rgba.r,
                color: this.outputColor
            });
        }
        onBlueInput(value) {
            const rgba = this.service.hsvaToRgba(this.hsva);
            const valid = !isNaN(value.v) && value.v >= 0 && value.v <= value.rg;
            if (valid) {
                rgba.b = value.v / value.rg;
                this.hsva = this.service.rgbaToHsva(rgba);
                this.sliderH = this.hsva.h;
                this.updateColorPicker();
            }
            this.directiveInstance.inputChanged({
                input: 'blue',
                valid: valid,
                value: rgba.b,
                color: this.outputColor
            });
        }
        onGreenInput(value) {
            const rgba = this.service.hsvaToRgba(this.hsva);
            const valid = !isNaN(value.v) && value.v >= 0 && value.v <= value.rg;
            if (valid) {
                rgba.g = value.v / value.rg;
                this.hsva = this.service.rgbaToHsva(rgba);
                this.sliderH = this.hsva.h;
                this.updateColorPicker();
            }
            this.directiveInstance.inputChanged({
                input: 'green',
                valid: valid,
                value: rgba.g,
                color: this.outputColor
            });
        }
        onHueInput(value) {
            const valid = !isNaN(value.v) && value.v >= 0 && value.v <= value.rg;
            if (valid) {
                this.hsva.h = value.v / value.rg;
                this.sliderH = this.hsva.h;
                this.updateColorPicker();
            }
            this.directiveInstance.inputChanged({
                input: 'hue',
                valid: valid,
                value: this.hsva.h,
                color: this.outputColor
            });
        }
        onValueInput(value) {
            const valid = !isNaN(value.v) && value.v >= 0 && value.v <= value.rg;
            if (valid) {
                this.hsva.v = value.v / value.rg;
                this.updateColorPicker();
            }
            this.directiveInstance.inputChanged({
                input: 'value',
                valid: valid,
                value: this.hsva.v,
                color: this.outputColor
            });
        }
        onAlphaInput(value) {
            const valid = !isNaN(value.v) && value.v >= 0 && value.v <= value.rg;
            if (valid) {
                this.hsva.a = value.v / value.rg;
                this.updateColorPicker();
            }
            this.directiveInstance.inputChanged({
                input: 'alpha',
                valid: valid,
                value: this.hsva.a,
                color: this.outputColor
            });
        }
        onLightnessInput(value) {
            const hsla = this.service.hsva2hsla(this.hsva);
            const valid = !isNaN(value.v) && value.v >= 0 && value.v <= value.rg;
            if (valid) {
                hsla.l = value.v / value.rg;
                this.hsva = this.service.hsla2hsva(hsla);
                this.sliderH = this.hsva.h;
                this.updateColorPicker();
            }
            this.directiveInstance.inputChanged({
                input: 'lightness',
                valid: valid,
                value: hsla.l,
                color: this.outputColor
            });
        }
        onSaturationInput(value) {
            const hsla = this.service.hsva2hsla(this.hsva);
            const valid = !isNaN(value.v) && value.v >= 0 && value.v <= value.rg;
            if (valid) {
                hsla.s = value.v / value.rg;
                this.hsva = this.service.hsla2hsva(hsla);
                this.sliderH = this.hsva.h;
                this.updateColorPicker();
            }
            this.directiveInstance.inputChanged({
                input: 'saturation',
                valid: valid,
                value: hsla.s,
                color: this.outputColor
            });
        }
        onCyanInput(value) {
            const valid = !isNaN(value.v) && value.v >= 0 && value.v <= value.rg;
            if (valid) {
                this.cmyk.c = value.v;
                this.updateColorPicker(false, true, true);
            }
            this.directiveInstance.inputChanged({
                input: 'cyan',
                valid: true,
                value: this.cmyk.c,
                color: this.outputColor
            });
        }
        onMagentaInput(value) {
            const valid = !isNaN(value.v) && value.v >= 0 && value.v <= value.rg;
            if (valid) {
                this.cmyk.m = value.v;
                this.updateColorPicker(false, true, true);
            }
            this.directiveInstance.inputChanged({
                input: 'magenta',
                valid: true,
                value: this.cmyk.m,
                color: this.outputColor
            });
        }
        onYellowInput(value) {
            const valid = !isNaN(value.v) && value.v >= 0 && value.v <= value.rg;
            if (valid) {
                this.cmyk.y = value.v;
                this.updateColorPicker(false, true, true);
            }
            this.directiveInstance.inputChanged({
                input: 'yellow',
                valid: true,
                value: this.cmyk.y,
                color: this.outputColor
            });
        }
        onBlackInput(value) {
            const valid = !isNaN(value.v) && value.v >= 0 && value.v <= value.rg;
            if (valid) {
                this.cmyk.k = value.v;
                this.updateColorPicker(false, true, true);
            }
            this.directiveInstance.inputChanged({
                input: 'black',
                valid: true,
                value: this.cmyk.k,
                color: this.outputColor
            });
        }
        onAddPresetColor(event, value) {
            event.stopPropagation();
            if (!this.cpPresetColors.filter((color) => (color === value)).length) {
                this.cpPresetColors = this.cpPresetColors.concat(value);
                this.directiveInstance.presetColorsChanged(this.cpPresetColors);
            }
        }
        onRemovePresetColor(event, value) {
            event.stopPropagation();
            this.cpPresetColors = this.cpPresetColors.filter((color) => (color !== value));
            this.directiveInstance.presetColorsChanged(this.cpPresetColors);
        }
        // Private helper functions for the color picker dialog status
        openColorPicker() {
            if (!this.show) {
                this.show = true;
                this.hidden = true;
                setTimeout(() => {
                    this.hidden = false;
                    this.setDialogPosition();
                    this.cdRef.detectChanges();
                }, 0);
                this.directiveInstance.stateChanged(true);
                if (!this.isIE10) {
                    document.addEventListener('mousedown', this.listenerMouseDown);
                    document.addEventListener('touchstart', this.listenerMouseDown);
                }
                window.addEventListener('resize', this.listenerResize);
            }
        }
        closeColorPicker() {
            if (this.show) {
                this.show = false;
                this.directiveInstance.stateChanged(false);
                if (!this.isIE10) {
                    document.removeEventListener('mousedown', this.listenerMouseDown);
                    document.removeEventListener('touchstart', this.listenerMouseDown);
                }
                window.removeEventListener('resize', this.listenerResize);
                if (!this.cdRef['destroyed']) {
                    this.cdRef.detectChanges();
                }
            }
        }
        updateColorPicker(emit = true, update = true, cmykInput = false) {
            if (this.sliderDimMax) {
                if (this.cpColorMode === 2) {
                    this.hsva.s = 0;
                }
                let hue, hsla, rgba;
                const lastOutput = this.outputColor;
                hsla = this.service.hsva2hsla(this.hsva);
                if (!this.cpCmykEnabled) {
                    rgba = this.service.denormalizeRGBA(this.service.hsvaToRgba(this.hsva));
                }
                else {
                    if (!cmykInput) {
                        rgba = this.service.hsvaToRgba(this.hsva);
                        this.cmyk = this.service.denormalizeCMYK(this.service.rgbaToCmyk(rgba));
                    }
                    else {
                        rgba = this.service.cmykToRgb(this.service.normalizeCMYK(this.cmyk));
                        this.hsva = this.service.rgbaToHsva(rgba);
                    }
                    rgba = this.service.denormalizeRGBA(rgba);
                    this.sliderH = this.hsva.h;
                }
                hue = this.service.denormalizeRGBA(this.service.hsvaToRgba(new Hsva(this.sliderH || this.hsva.h, 1, 1, 1)));
                if (update) {
                    this.hslaText = new Hsla(Math.round((hsla.h) * 360), Math.round(hsla.s * 100), Math.round(hsla.l * 100), Math.round(hsla.a * 100) / 100);
                    this.rgbaText = new Rgba(rgba.r, rgba.g, rgba.b, Math.round(rgba.a * 100) / 100);
                    if (this.cpCmykEnabled) {
                        this.cmykText = new Cmyk(this.cmyk.c, this.cmyk.m, this.cmyk.y, this.cmyk.k, Math.round(this.cmyk.a * 100) / 100);
                    }
                    const allowHex8 = this.cpAlphaChannel === 'always';
                    this.hexText = this.service.rgbaToHex(rgba, allowHex8);
                    this.hexAlpha = this.rgbaText.a;
                }
                if (this.cpOutputFormat === 'auto') {
                    if (this.format !== ColorFormats.RGBA && this.format !== ColorFormats.CMYK) {
                        if (this.hsva.a < 1) {
                            this.format = this.hsva.a < 1 ? ColorFormats.RGBA : ColorFormats.HEX;
                        }
                    }
                }
                this.hueSliderColor = 'rgb(' + hue.r + ',' + hue.g + ',' + hue.b + ')';
                this.alphaSliderColor = 'rgb(' + rgba.r + ',' + rgba.g + ',' + rgba.b + ')';
                this.outputColor = this.service.outputFormat(this.hsva, this.cpOutputFormat, this.cpAlphaChannel);
                this.selectedColor = this.service.outputFormat(this.hsva, 'rgba', null);
                if (this.format !== ColorFormats.CMYK) {
                    this.cmykColor = '';
                }
                else {
                    if (this.cpAlphaChannel === 'always' || this.cpAlphaChannel === 'enabled' ||
                        this.cpAlphaChannel === 'forced') {
                        const alpha = Math.round(this.cmyk.a * 100) / 100;
                        this.cmykColor = `cmyka(${this.cmyk.c},${this.cmyk.m},${this.cmyk.y},${this.cmyk.k},${alpha})`;
                    }
                    else {
                        this.cmykColor = `cmyk(${this.cmyk.c},${this.cmyk.m},${this.cmyk.y},${this.cmyk.k})`;
                    }
                }
                this.slider = new SliderPosition((this.sliderH || this.hsva.h) * this.sliderDimMax.h - 8, this.hsva.s * this.sliderDimMax.s - 8, (1 - this.hsva.v) * this.sliderDimMax.v - 8, this.hsva.a * this.sliderDimMax.a - 8);
                if (emit && lastOutput !== this.outputColor) {
                    if (this.cpCmykEnabled) {
                        this.directiveInstance.cmykChanged(this.cmykColor);
                    }
                    this.directiveInstance.colorChanged(this.outputColor);
                }
            }
        }
        // Private helper functions for the color picker dialog positioning
        setDialogPosition() {
            if (this.cpDialogDisplay === 'inline') {
                this.position = 'relative';
            }
            else {
                let position = 'static', transform = '', style;
                let parentNode = null, transformNode = null;
                let node = this.directiveElementRef.nativeElement.parentNode;
                const dialogHeight = this.dialogElement.nativeElement.offsetHeight;
                while (node !== null && node.tagName !== 'HTML') {
                    style = window.getComputedStyle(node);
                    position = style.getPropertyValue('position');
                    transform = style.getPropertyValue('transform');
                    if (position !== 'static' && parentNode === null) {
                        parentNode = node;
                    }
                    if (transform && transform !== 'none' && transformNode === null) {
                        transformNode = node;
                    }
                    if (position === 'fixed') {
                        parentNode = transformNode;
                        break;
                    }
                    node = node.parentNode;
                }
                const boxDirective = this.createDialogBox(this.directiveElementRef.nativeElement, (position !== 'fixed'));
                if (this.useRootViewContainer || (position === 'fixed' &&
                    (!parentNode || parentNode instanceof HTMLUnknownElement))) {
                    this.top = boxDirective.top;
                    this.left = boxDirective.left;
                }
                else {
                    if (parentNode === null) {
                        parentNode = node;
                    }
                    const boxParent = this.createDialogBox(parentNode, (position !== 'fixed'));
                    this.top = boxDirective.top - boxParent.top;
                    this.left = boxDirective.left - boxParent.left;
                }
                if (position === 'fixed') {
                    this.position = 'fixed';
                }
                let usePosition = this.cpPosition;
                if (this.cpPosition === 'auto') {
                    let usePositionX = 'right';
                    let usePositionY = 'bottom';
                    const winWidth = Math.max(document.documentElement.clientWidth, window.innerWidth || 0);
                    const winHeight = Math.max(document.documentElement.clientHeight, window.innerHeight || 0);
                    if (this.left + this.cpWidth > winWidth) {
                        usePositionX = 'left';
                    }
                    if (this.top + dialogHeight > winHeight) {
                        usePositionY = 'top';
                    }
                    usePosition = usePositionX + '-' + usePositionY;
                }
                this.cpUsePosition = usePosition;
                if (usePosition === 'top') {
                    this.arrowTop = dialogHeight - 1;
                    this.top -= dialogHeight + this.dialogArrowSize;
                    this.left += this.cpPositionOffset / 100 * boxDirective.width - this.dialogArrowOffset;
                }
                else if (usePosition === 'bottom') {
                    this.top += boxDirective.height + this.dialogArrowSize;
                    this.left += this.cpPositionOffset / 100 * boxDirective.width - this.dialogArrowOffset;
                }
                else if (usePosition === 'top-left' || usePosition === 'left-top') {
                    this.top -= dialogHeight - boxDirective.height + boxDirective.height * this.cpPositionOffset / 100;
                    this.left -= this.cpWidth + this.dialogArrowSize - 2 - this.dialogArrowOffset;
                }
                else if (usePosition === 'top-right' || usePosition === 'right-top') {
                    this.top -= dialogHeight - boxDirective.height + boxDirective.height * this.cpPositionOffset / 100;
                    this.left += boxDirective.width + this.dialogArrowSize - 2 - this.dialogArrowOffset;
                }
                else if (usePosition === 'left' || usePosition === 'bottom-left' || usePosition === 'left-bottom') {
                    this.top += boxDirective.height * this.cpPositionOffset / 100 - this.dialogArrowOffset;
                    this.left -= this.cpWidth + this.dialogArrowSize - 2;
                }
                else { // usePosition === 'right' || usePosition === 'bottom-right' || usePosition === 'right-bottom'
                    this.top += boxDirective.height * this.cpPositionOffset / 100 - this.dialogArrowOffset;
                    this.left += boxDirective.width + this.dialogArrowSize - 2;
                }
            }
        }
        // Private helper functions for the color picker dialog positioning and opening
        isDescendant(parent, child) {
            let node = child.parentNode;
            while (node !== null) {
                if (node === parent) {
                    return true;
                }
                node = node.parentNode;
            }
            return false;
        }
        createDialogBox(element, offset) {
            return {
                top: element.getBoundingClientRect().top + (offset ? window.pageYOffset : 0),
                left: element.getBoundingClientRect().left + (offset ? window.pageXOffset : 0),
                width: element.offsetWidth,
                height: element.offsetHeight
            };
        }
    };
    ColorPickerComponent.ɵfac = function ColorPickerComponent_Factory(t) { return new (t || ColorPickerComponent)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(ChangeDetectorRef), ɵɵdirectiveInject(ColorPickerService)); };
    ColorPickerComponent.ɵcmp = ɵɵdefineComponent({ type: ColorPickerComponent, selectors: [["color-picker"]], viewQuery: function ColorPickerComponent_Query(rf, ctx) { if (rf & 1) {
            ɵɵstaticViewQuery(_c0$o, true);
            ɵɵstaticViewQuery(_c1$f, true);
            ɵɵstaticViewQuery(_c2$6, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.dialogElement = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.hueSlider = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.alphaSlider = _t.first);
        } }, hostBindings: function ColorPickerComponent_HostBindings(rf, ctx) { if (rf & 1) {
            ɵɵlistener("keyup.esc", function ColorPickerComponent_keyup_esc_HostBindingHandler($event) { return ctx.handleEsc($event); }, false, ɵɵresolveDocument)("keyup.enter", function ColorPickerComponent_keyup_enter_HostBindingHandler($event) { return ctx.handleEnter($event); }, false, ɵɵresolveDocument);
        } }, decls: 28, vars: 47, consts: [[1, "color-picker", 3, "click"], ["dialogPopup", ""], [3, "class", "top", 4, "ngIf"], ["class", "saturation-lightness", 3, "slider", "rgX", "rgY", "background-color", "newValue", "dragStart", "dragEnd", 4, "ngIf"], [1, "hue-alpha", "box"], [1, "left"], [1, "selected-color-background"], [1, "selected-color"], ["type", "button", 3, "class", "disabled", "click", 4, "ngIf"], [1, "right"], ["style", "height: 16px;", 4, "ngIf"], [1, "hue", 3, "slider", "rgX", "newValue", "dragStart", "dragEnd"], ["hueSlider", ""], [1, "cursor"], [1, "value", 3, "slider", "rgX", "newValue", "dragStart", "dragEnd"], ["valueSlider", ""], [1, "alpha", 3, "slider", "rgX", "newValue", "dragStart", "dragEnd"], ["alphaSlider", ""], ["class", "cmyk-text", 3, "display", 4, "ngIf"], ["class", "hsla-text", 3, "display", 4, "ngIf"], ["class", "rgba-text", 3, "display", 4, "ngIf"], ["class", "hex-text", 3, "hex-alpha", "display", 4, "ngIf"], ["class", "value-text", 4, "ngIf"], ["class", "type-policy", 4, "ngIf"], ["class", "preset-area", 4, "ngIf"], ["class", "button-area", 4, "ngIf"], [1, "saturation-lightness", 3, "slider", "rgX", "rgY", "newValue", "dragStart", "dragEnd"], ["type", "button", 3, "disabled", "click"], [2, "height", "16px"], [1, "cmyk-text"], [1, "box"], ["type", "number", "pattern", "[0-9]*", "min", "0", "max", "100", 3, "text", "rg", "value", "keyup.enter", "newValue"], ["type", "number", "pattern", "[0-9]+([\\.,][0-9]{1,2})?", "min", "0", "max", "1", "step", "0.1", 3, "text", "rg", "value", "keyup.enter", "newValue", 4, "ngIf"], [4, "ngIf"], ["type", "number", "pattern", "[0-9]+([\\.,][0-9]{1,2})?", "min", "0", "max", "1", "step", "0.1", 3, "text", "rg", "value", "keyup.enter", "newValue"], [1, "hsla-text"], ["type", "number", "pattern", "[0-9]*", "min", "0", "max", "360", 3, "text", "rg", "value", "keyup.enter", "newValue"], [1, "rgba-text"], ["type", "number", "pattern", "[0-9]*", "min", "0", "max", "255", 3, "text", "rg", "value", "keyup.enter", "newValue"], [1, "hex-text"], [3, "text", "value", "blur", "keyup.enter", "newValue"], [1, "value-text"], [1, "type-policy"], [1, "type-policy-arrow", 3, "click"], [1, "preset-area"], [1, "preset-label"], [3, "class", 4, "ngIf"], ["class", "preset-color", 3, "backgroundColor", "click", 4, "ngFor", "ngForOf"], [1, "preset-color", 3, "click"], [3, "class", "click", 4, "ngIf"], [3, "click"], [1, "button-area"], ["type", "button", 3, "class", "click", 4, "ngIf"], ["type", "button", 3, "click"]], template: function ColorPickerComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "div", 0, 1);
            ɵɵlistener("click", function ColorPickerComponent_Template_div_click_0_listener($event) { return $event.stopPropagation(); });
            ɵɵtemplate(2, ColorPickerComponent_div_2_Template, 1, 5, "div", 2);
            ɵɵtemplate(3, ColorPickerComponent_div_3_Template, 2, 8, "div", 3);
            ɵɵelementStart(4, "div", 4);
            ɵɵelementStart(5, "div", 5);
            ɵɵelement(6, "div", 6);
            ɵɵelement(7, "div", 7);
            ɵɵtemplate(8, ColorPickerComponent_button_8_Template, 2, 5, "button", 8);
            ɵɵelementEnd();
            ɵɵelementStart(9, "div", 9);
            ɵɵtemplate(10, ColorPickerComponent_div_10_Template, 1, 0, "div", 10);
            ɵɵelementStart(11, "div", 11, 12);
            ɵɵlistener("newValue", function ColorPickerComponent_Template_div_newValue_11_listener($event) { return ctx.onHueChange($event); })("dragStart", function ColorPickerComponent_Template_div_dragStart_11_listener() { return ctx.onDragStart("hue"); })("dragEnd", function ColorPickerComponent_Template_div_dragEnd_11_listener() { return ctx.onDragEnd("hue"); });
            ɵɵelement(13, "div", 13);
            ɵɵelementEnd();
            ɵɵelementStart(14, "div", 14, 15);
            ɵɵlistener("newValue", function ColorPickerComponent_Template_div_newValue_14_listener($event) { return ctx.onValueChange($event); })("dragStart", function ColorPickerComponent_Template_div_dragStart_14_listener() { return ctx.onDragStart("value"); })("dragEnd", function ColorPickerComponent_Template_div_dragEnd_14_listener() { return ctx.onDragEnd("value"); });
            ɵɵelement(16, "div", 13);
            ɵɵelementEnd();
            ɵɵelementStart(17, "div", 16, 17);
            ɵɵlistener("newValue", function ColorPickerComponent_Template_div_newValue_17_listener($event) { return ctx.onAlphaChange($event); })("dragStart", function ColorPickerComponent_Template_div_dragStart_17_listener() { return ctx.onDragStart("alpha"); })("dragEnd", function ColorPickerComponent_Template_div_dragEnd_17_listener() { return ctx.onDragEnd("alpha"); });
            ɵɵelement(19, "div", 13);
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵtemplate(20, ColorPickerComponent_div_20_Template, 17, 12, "div", 18);
            ɵɵtemplate(21, ColorPickerComponent_div_21_Template, 14, 10, "div", 19);
            ɵɵtemplate(22, ColorPickerComponent_div_22_Template, 14, 10, "div", 20);
            ɵɵtemplate(23, ColorPickerComponent_div_23_Template, 8, 7, "div", 21);
            ɵɵtemplate(24, ColorPickerComponent_div_24_Template, 9, 3, "div", 22);
            ɵɵtemplate(25, ColorPickerComponent_div_25_Template, 3, 0, "div", 23);
            ɵɵtemplate(26, ColorPickerComponent_div_26_Template, 6, 3, "div", 24);
            ɵɵtemplate(27, ColorPickerComponent_div_27_Template, 3, 2, "div", 25);
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵstyleProp("display", !ctx.show ? "none" : "block")("visibility", ctx.hidden ? "hidden" : "visible")("top", ctx.top, "px")("left", ctx.left, "px")("position", ctx.position)("height", ctx.cpHeight, "px")("width", ctx.cpWidth, "px");
            ɵɵclassProp("open", ctx.show);
            ɵɵadvance(2);
            ɵɵproperty("ngIf", ctx.cpDialogDisplay == "popup");
            ɵɵadvance(1);
            ɵɵproperty("ngIf", (ctx.cpColorMode || 1) === 1);
            ɵɵadvance(4);
            ɵɵstyleProp("background-color", ctx.selectedColor);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx.cpAddColorButton);
            ɵɵadvance(2);
            ɵɵproperty("ngIf", ctx.cpAlphaChannel === "disabled");
            ɵɵadvance(1);
            ɵɵstyleProp("display", (ctx.cpColorMode || 1) === 1 ? "block" : "none");
            ɵɵproperty("rgX", 1);
            ɵɵadvance(2);
            ɵɵstyleProp("left", ctx.slider == null ? null : ctx.slider.h, "px");
            ɵɵadvance(1);
            ɵɵstyleProp("display", (ctx.cpColorMode || 1) === 2 ? "block" : "none");
            ɵɵproperty("rgX", 1);
            ɵɵadvance(2);
            ɵɵstyleProp("right", ctx.slider == null ? null : ctx.slider.v, "px");
            ɵɵadvance(1);
            ɵɵstyleProp("display", ctx.cpAlphaChannel === "disabled" ? "none" : "block")("background-color", ctx.alphaSliderColor);
            ɵɵproperty("rgX", 1);
            ɵɵadvance(2);
            ɵɵstyleProp("left", ctx.slider == null ? null : ctx.slider.a, "px");
            ɵɵadvance(1);
            ɵɵproperty("ngIf", !ctx.cpDisableInput && (ctx.cpColorMode || 1) === 1);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", !ctx.cpDisableInput && (ctx.cpColorMode || 1) === 1);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", !ctx.cpDisableInput && (ctx.cpColorMode || 1) === 1);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", !ctx.cpDisableInput && (ctx.cpColorMode || 1) === 1);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", !ctx.cpDisableInput && (ctx.cpColorMode || 1) === 2);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", !ctx.cpDisableInput && (ctx.cpColorMode || 1) === 1);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", (ctx.cpPresetColors == null ? null : ctx.cpPresetColors.length) || ctx.cpAddColorButton);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx.cpOKButton || ctx.cpCancelButton);
        } }, directives: [NgIf, SliderDirective,
            TextDirective, NgForOf], styles: [".color-picker{position:absolute;z-index:1000;width:230px;height:auto;border:1px solid #777;cursor:default;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;background-color:#fff}.color-picker *{box-sizing:border-box;margin:0;font-size:11px}.color-picker input{width:0;height:26px;min-width:0;font-size:13px;text-align:center;color:#000}.color-picker input:-moz-submit-invalid,.color-picker input:-moz-ui-invalid,.color-picker input:invalid{box-shadow:none}.color-picker input::-webkit-inner-spin-button,.color-picker input::-webkit-outer-spin-button{margin:0;-webkit-appearance:none}.color-picker .arrow{position:absolute;z-index:999999;width:0;height:0;border-style:solid}.color-picker .arrow.arrow-top{left:8px;border-width:10px 5px;border-color:#777 transparent transparent}.color-picker .arrow.arrow-bottom{top:-20px;left:8px;border-width:10px 5px;border-color:transparent transparent #777}.color-picker .arrow.arrow-left-top,.color-picker .arrow.arrow-top-left{right:-21px;bottom:8px;border-width:5px 10px;border-color:transparent transparent transparent #777}.color-picker .arrow.arrow-right-top,.color-picker .arrow.arrow-top-right{bottom:8px;left:-20px;border-width:5px 10px;border-color:transparent #777 transparent transparent}.color-picker .arrow.arrow-bottom-left,.color-picker .arrow.arrow-left,.color-picker .arrow.arrow-left-bottom{top:8px;right:-21px;border-width:5px 10px;border-color:transparent transparent transparent #777}.color-picker .arrow.arrow-bottom-right,.color-picker .arrow.arrow-right,.color-picker .arrow.arrow-right-bottom{top:8px;left:-20px;border-width:5px 10px;border-color:transparent #777 transparent transparent}.color-picker .cursor{position:relative;width:16px;height:16px;border:2px solid #222;border-radius:50%;cursor:default}.color-picker .box{display:flex;padding:4px 8px}.color-picker .left{position:relative;padding:16px 8px}.color-picker .right{flex:1 1 auto;padding:12px 8px}.color-picker .button-area{padding:0 16px 16px;text-align:right}.color-picker .button-area button{margin-left:8px}.color-picker .preset-area{padding:4px 15px}.color-picker .preset-area .preset-label{overflow:hidden;width:100%;padding:4px;font-size:11px;white-space:nowrap;text-align:left;text-overflow:ellipsis;color:#555}.color-picker .preset-area .preset-color{position:relative;display:inline-block;width:18px;height:18px;margin:4px 6px 8px;border:1px solid #a9a9a9;border-radius:25%;cursor:pointer}.color-picker .preset-area .preset-empty-message{min-height:18px;margin-top:4px;margin-bottom:8px;font-style:italic;text-align:center}.color-picker .hex-text{width:100%;padding:4px 8px;font-size:11px}.color-picker .hex-text .box{padding:0 24px 8px 8px}.color-picker .hex-text .box div{float:left;flex:1 1 auto;text-align:center;color:#555;clear:left}.color-picker .hex-text .box input{flex:1 1 auto;padding:1px;border:1px solid #a9a9a9}.color-picker .hex-alpha .box div:first-child,.color-picker .hex-alpha .box input:first-child{flex-grow:3;margin-right:8px}.color-picker .cmyk-text,.color-picker .hsla-text,.color-picker .rgba-text,.color-picker .value-text{width:100%;padding:4px 8px;font-size:11px}.color-picker .cmyk-text .box,.color-picker .hsla-text .box,.color-picker .rgba-text .box{padding:0 24px 8px 8px}.color-picker .value-text .box{padding:0 8px 8px}.color-picker .cmyk-text .box div,.color-picker .hsla-text .box div,.color-picker .rgba-text .box div,.color-picker .value-text .box div{flex:1 1 auto;margin-right:8px;text-align:center;color:#555}.color-picker .cmyk-text .box div:last-child,.color-picker .hsla-text .box div:last-child,.color-picker .rgba-text .box div:last-child,.color-picker .value-text .box div:last-child{margin-right:0}.color-picker .cmyk-text .box input,.color-picker .hsla-text .box input,.color-picker .rgba-text .box input,.color-picker .value-text .box input{float:left;flex:1;padding:1px;margin:0 8px 0 0;border:1px solid #a9a9a9}.color-picker .cmyk-text .box input:last-child,.color-picker .hsla-text .box input:last-child,.color-picker .rgba-text .box input:last-child,.color-picker .value-text .box input:last-child{margin-right:0}.color-picker .hue-alpha{align-items:center;margin-bottom:3px}.color-picker .hue{direction:ltr;width:100%;height:16px;margin-bottom:16px;border:none;cursor:pointer;background-size:100% 100%;background-image:url(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAJYAAAAQCAYAAAD06IYnAAAABmJLR0QA/wD/AP+gvaeTAAAACXBIWXMAAAsTAAALEwEAmpwYAAAAB3RJTUUH4AIWDwkUFWbCCAAAAFxJREFUaN7t0kEKg0AQAME2x83/n2qu5qCgD1iDhCoYdpnbQC9bbY1qVO/jvc6k3ad91s7/7F1/csgPrujuQ17BDYSFsBAWwgJhISyEBcJCWAgLhIWwEBYIi2f7Ar/1TCgFH2X9AAAAAElFTkSuQmCC)}.color-picker .value{direction:rtl;width:100%;height:16px;margin-bottom:16px;border:none;cursor:pointer;background-size:100% 100%;background-image:url(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAJYAAAAQCAYAAAD06IYnAAACTklEQVR42u3SYUcrABhA4U2SkmRJMmWSJklKJiWZZpKUJJskKUmaTFImKZOUzMySpGRmliRNJilJSpKSJEtmSpIpmWmSdO736/6D+x7OP3gUCoWCv1cqlSQlJZGcnExKSgqpqamkpaWRnp5ORkYGmZmZqFQqsrKyyM7OJicnh9zcXNRqNXl5eeTn56PRaCgoKKCwsJCioiK0Wi3FxcWUlJRQWlpKWVkZ5eXlVFRUUFlZiU6no6qqiurqampqaqitraWurg69Xk99fT0GgwGj0UhDQwONjY00NTXR3NxMS0sLra2ttLW10d7ejslkwmw209HRQWdnJ11dXXR3d9PT00Nvby99fX309/czMDDA4OAgFouFoaEhrFYrw8PDjIyMMDo6ytjYGDabjfHxcSYmJpicnGRqagq73c709DQzMzPMzs4yNzfH/Pw8DocDp9OJy+XC7XazsLDA4uIiS0tLLC8vs7KywurqKmtra3g8HrxeLz6fD7/fz/r6OhsbG2xubrK1tcX29jaBQICdnR2CwSC7u7vs7e2xv7/PwcEBh4eHHB0dcXx8zMnJCaenp5ydnXF+fs7FxQWXl5dcXV1xfX3Nzc0Nt7e33N3dEQqFuL+/5+HhgXA4TCQS4fHxkaenJ56fn3l5eeH19ZVoNMrb2xvv7+98fHwQi8WIx+N8fn6SSCT4+vri+/ubn58ffn9/+VcKgSWwBJbAElgCS2AJLIElsASWwBJYAktgCSyBJbAElsASWAJLYAksgSWwBJbAElgCS2AJLIElsP4/WH8AmJ5Z6jHS4h8AAAAASUVORK5CYII=)}.color-picker .alpha{direction:ltr;width:100%;height:16px;border:none;cursor:pointer;background-size:100% 100%;background-image:url(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAJYAAAAQCAYAAAD06IYnAAAABmJLR0QA/wD/AP+gvaeTAAAACXBIWXMAAAsTAAALEwEAmpwYAAAAB3RJTUUH4AIWDwYQlZMa3gAAAWVJREFUaN7tmEGO6jAQRCsOArHgBpyAJYGjcGocxAm4A2IHpmoWE0eBH+ezmFlNvU06shJ3W6VEelWMUQAIIF9f6qZpimsA1LYtS2uF51/u27YVAFZVRUkEoGHdPV/sIcbIEIIkUdI/9Xa7neyv61+SWFUVAVCSct00TWn2fv6u3+Ecfd3tXzy/0+nEUu+SPjo/kqzrmiQpScN6v98XewfA8/lMkiLJ2WxGSUopcT6fM6U0NX9/frfbjev1WtfrlZfLhYfDQQHG/AIOlnGwjINlHCxjHCzjYJm/TJWdCwquJXseFFzGwDNNeiKMOJTO8xQdDQaeB29+K9efeLaBo9J7vdvtJj1RjFFjfiv7qv95tjx/7leSQgh93e1ffMeIp6O+YQjho/N791t1XVOSSI7N//K+4/GoxWLBx+PB5/Op5XLJ+/3OlJJWqxU3m83ovv5iGf8KjYNlHCxjHCzjYBkHy5gf5gusvQU7U37jTAAAAABJRU5ErkJggg==)}.color-picker .type-policy{position:absolute;top:218px;right:12px;width:16px;height:24px;background-size:8px 16px;background-image:url(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABIAAAAgCAYAAAAffCjxAAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAACewAAAnsB01CO3AAAABl0RVh0U29mdHdhcmUAd3d3Lmlua3NjYXBlLm9yZ5vuPBoAAAIASURBVEiJ7ZY9axRRFIafsxMStrLQJpAgpBFhi+C9w1YSo00I6RZ/g9vZpBf/QOr4GyRgkSKNSrAadsZqQGwCkuAWyRZJsySwvhZ7N/vhzrgbLH3Ld8597jlzz50zJokyxXH8DqDVar0qi6v8BbItqSGpEcfxdlmsFWXkvX8AfAVWg3UKPEnT9GKujMzsAFgZsVaCN1VTQd77XUnrgE1kv+6935268WRpzrnHZvYRWC7YvC3pRZZl3wozqtVqiyH9IgjAspkd1Gq1xUJQtVrdB9ZKIAOthdg/Qc65LUk7wNIMoCVJO865rYFhkqjX6/d7vV4GPJwBMqofURS5JEk6FYBer/eeYb/Mo9WwFnPOvQbeAvfuAAK4BN4sAJtAG/gJIElmNuiJyba3EGNmZiPeZuEVmVell/Y/6N+CzDn3AXhEOOo7Hv/3BeAz8IzQkMPnJbuPx1wC+yYJ7/0nYIP5S/0FHKdp+rwCEEXRS/rf5Hl1Gtb2M0iSpCOpCZzPATmX1EySpHMLAsiy7MjMDoHrGSDXZnaYZdnRwBh7J91utwmczAA6CbG3GgPleX4jqUH/a1CktqRGnuc3hSCAMB32gKspkCtgb3KCQMmkjeP4WNJThrNNZval1WptTIsv7JtQ4tmIdRa8qSoEpWl6YWZNoAN0zKxZNPehpLSBZv2t+Q0CJ9lLnARQLAAAAABJRU5ErkJggg==);background-repeat:no-repeat;background-position:center}.color-picker .type-policy .type-policy-arrow{display:block;width:100%;height:50%}.color-picker .selected-color{position:absolute;top:16px;left:8px;width:40px;height:40px;border:1px solid #a9a9a9;border-radius:50%}.color-picker .selected-color-background{width:40px;height:40px;border-radius:50%;background-image:url(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAACgAAAAoCAYAAACM/rhtAAAAh0lEQVRYR+2W0QlAMQgD60zdfwOdqa8TmI/wQMr5K0I5bZLIzLOa2nt37VVVbd+dDx5obgCC3KBLwJ2ff4PnVidkf+ucIhw80HQaCLo3DMH3CRK3iFsmAWVl6hPNDwt8EvNE5q+YuEXcMgkonVM6SdyCoEvAnZ8v1Hjx817MilmxSUB5rdLJDycZgUAZUch/AAAAAElFTkSuQmCC)}.color-picker .saturation-lightness{direction:ltr;width:100%;height:130px;border:none;cursor:pointer;touch-action:manipulation;background-size:100% 100%;background-image:url(data:image/png;base64,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)}.color-picker .cp-add-color-button-class{position:absolute;display:inline;padding:0;margin:3px -3px;border:0;cursor:pointer;background:0 0}.color-picker .cp-add-color-button-class:hover{text-decoration:underline}.color-picker .cp-add-color-button-class:disabled{cursor:not-allowed;color:#999}.color-picker .cp-add-color-button-class:disabled:hover{text-decoration:none}.color-picker .cp-remove-color-button-class{position:absolute;top:-5px;right:-5px;display:block;width:10px;height:10px;border-radius:50%;cursor:pointer;text-align:center;background:#fff;box-shadow:1px 1px 5px #333}.color-picker .cp-remove-color-button-class::before{content:'x';position:relative;bottom:3.5px;display:inline-block;font-size:10px}"], encapsulation: 2 });
    ColorPickerComponent.ctorParameters = () => [
        { type: ElementRef },
        { type: ChangeDetectorRef },
        { type: ColorPickerService }
    ];
    __decorate([
        ViewChild('dialogPopup', { static: true })
    ], ColorPickerComponent.prototype, "dialogElement", void 0);
    __decorate([
        ViewChild('hueSlider', { static: true })
    ], ColorPickerComponent.prototype, "hueSlider", void 0);
    __decorate([
        ViewChild('alphaSlider', { static: true })
    ], ColorPickerComponent.prototype, "alphaSlider", void 0);
    __decorate([
        HostListener('document:keyup.esc', ['$event'])
    ], ColorPickerComponent.prototype, "handleEsc", null);
    __decorate([
        HostListener('document:keyup.enter', ['$event'])
    ], ColorPickerComponent.prototype, "handleEnter", null);

    let ColorPickerDirective = class ColorPickerDirective {
        constructor(injector, cfr, appRef, vcRef, elRef, _service) {
            this.injector = injector;
            this.cfr = cfr;
            this.appRef = appRef;
            this.vcRef = vcRef;
            this.elRef = elRef;
            this._service = _service;
            this.dialogCreated = false;
            this.ignoreChanges = false;
            this.viewAttachedToAppRef = false;
            this.cpWidth = '230px';
            this.cpHeight = 'auto';
            this.cpToggle = false;
            this.cpDisabled = false;
            this.cpIgnoredElements = [];
            this.cpFallbackColor = '';
            this.cpColorMode = 'color';
            this.cpCmykEnabled = false;
            this.cpOutputFormat = 'auto';
            this.cpAlphaChannel = 'enabled';
            this.cpDisableInput = false;
            this.cpDialogDisplay = 'popup';
            this.cpSaveClickOutside = true;
            this.cpCloseClickOutside = true;
            this.cpUseRootViewContainer = false;
            this.cpPosition = 'auto';
            this.cpPositionOffset = '0%';
            this.cpPositionRelativeToArrow = false;
            this.cpOKButton = false;
            this.cpOKButtonText = 'OK';
            this.cpOKButtonClass = 'cp-ok-button-class';
            this.cpCancelButton = false;
            this.cpCancelButtonText = 'Cancel';
            this.cpCancelButtonClass = 'cp-cancel-button-class';
            this.cpPresetLabel = 'Preset colors';
            this.cpPresetColorsClass = 'cp-preset-colors-class';
            this.cpMaxPresetColorsLength = 6;
            this.cpPresetEmptyMessage = 'No colors added';
            this.cpPresetEmptyMessageClass = 'preset-empty-message';
            this.cpAddColorButton = false;
            this.cpAddColorButtonText = 'Add color';
            this.cpAddColorButtonClass = 'cp-add-color-button-class';
            this.cpRemoveColorButtonClass = 'cp-remove-color-button-class';
            this.cpInputChange = new EventEmitter(true);
            this.cpToggleChange = new EventEmitter(true);
            this.cpSliderChange = new EventEmitter(true);
            this.cpSliderDragEnd = new EventEmitter(true);
            this.cpSliderDragStart = new EventEmitter(true);
            this.colorPickerOpen = new EventEmitter(true);
            this.colorPickerClose = new EventEmitter(true);
            this.colorPickerCancel = new EventEmitter(true);
            this.colorPickerSelect = new EventEmitter(true);
            this.colorPickerChange = new EventEmitter(false);
            this.cpCmykColorChange = new EventEmitter(true);
            this.cpPresetColorsChange = new EventEmitter(true);
        }
        handleClick() {
            this.inputFocus();
        }
        handleFocus() {
            this.inputFocus();
        }
        handleInput(event) {
            this.inputChange(event);
        }
        ngOnDestroy() {
            if (this.cmpRef != null) {
                if (this.viewAttachedToAppRef) {
                    this.appRef.detachView(this.cmpRef.hostView);
                }
                this.cmpRef.destroy();
                this.cmpRef = null;
                this.dialog = null;
            }
        }
        ngOnChanges(changes) {
            if (changes.cpToggle && !this.cpDisabled) {
                if (changes.cpToggle.currentValue) {
                    this.openDialog();
                }
                else if (!changes.cpToggle.currentValue) {
                    this.closeDialog();
                }
            }
            if (changes.colorPicker) {
                if (this.dialog && !this.ignoreChanges) {
                    if (this.cpDialogDisplay === 'inline') {
                        this.dialog.setInitialColor(changes.colorPicker.currentValue);
                    }
                    this.dialog.setColorFromString(changes.colorPicker.currentValue, false);
                    if (this.cpUseRootViewContainer && this.cpDialogDisplay !== 'inline') {
                        this.cmpRef.changeDetectorRef.detectChanges();
                    }
                }
                this.ignoreChanges = false;
            }
            if (changes.cpPresetLabel || changes.cpPresetColors) {
                if (this.dialog) {
                    this.dialog.setPresetConfig(this.cpPresetLabel, this.cpPresetColors);
                }
            }
        }
        openDialog() {
            if (!this.dialogCreated) {
                let vcRef = this.vcRef;
                this.dialogCreated = true;
                this.viewAttachedToAppRef = false;
                if (this.cpUseRootViewContainer && this.cpDialogDisplay !== 'inline') {
                    const classOfRootComponent = this.appRef.componentTypes[0];
                    const appInstance = this.injector.get(classOfRootComponent, Injector.NULL);
                    if (appInstance !== Injector.NULL) {
                        vcRef = appInstance.vcRef || appInstance.viewContainerRef || this.vcRef;
                        if (vcRef === this.vcRef) {
                            console.warn('You are using cpUseRootViewContainer, ' +
                                'but the root component is not exposing viewContainerRef!' +
                                'Please expose it by adding \'public vcRef: ViewContainerRef\' to the constructor.');
                        }
                    }
                    else {
                        this.viewAttachedToAppRef = true;
                    }
                }
                const compFactory = this.cfr.resolveComponentFactory(ColorPickerComponent);
                if (this.viewAttachedToAppRef) {
                    this.cmpRef = compFactory.create(this.injector);
                    this.appRef.attachView(this.cmpRef.hostView);
                    document.body.appendChild(this.cmpRef.hostView.rootNodes[0]);
                }
                else {
                    const injector = ReflectiveInjector.fromResolvedProviders([], vcRef.parentInjector);
                    this.cmpRef = vcRef.createComponent(compFactory, 0, injector, []);
                }
                this.cmpRef.instance.setupDialog(this, this.elRef, this.colorPicker, this.cpWidth, this.cpHeight, this.cpDialogDisplay, this.cpFallbackColor, this.cpColorMode, this.cpCmykEnabled, this.cpAlphaChannel, this.cpOutputFormat, this.cpDisableInput, this.cpIgnoredElements, this.cpSaveClickOutside, this.cpCloseClickOutside, this.cpUseRootViewContainer, this.cpPosition, this.cpPositionOffset, this.cpPositionRelativeToArrow, this.cpPresetLabel, this.cpPresetColors, this.cpPresetColorsClass, this.cpMaxPresetColorsLength, this.cpPresetEmptyMessage, this.cpPresetEmptyMessageClass, this.cpOKButton, this.cpOKButtonClass, this.cpOKButtonText, this.cpCancelButton, this.cpCancelButtonClass, this.cpCancelButtonText, this.cpAddColorButton, this.cpAddColorButtonClass, this.cpAddColorButtonText, this.cpRemoveColorButtonClass);
                this.dialog = this.cmpRef.instance;
                if (this.vcRef !== vcRef) {
                    this.cmpRef.changeDetectorRef.detectChanges();
                }
            }
            else if (this.dialog) {
                this.dialog.openDialog(this.colorPicker);
            }
        }
        closeDialog() {
            if (this.dialog && this.cpDialogDisplay === 'popup') {
                this.dialog.closeDialog();
            }
        }
        cmykChanged(value) {
            this.cpCmykColorChange.emit(value);
        }
        stateChanged(state) {
            this.cpToggleChange.emit(state);
            if (state) {
                this.colorPickerOpen.emit(this.colorPicker);
            }
            else {
                this.colorPickerClose.emit(this.colorPicker);
            }
        }
        colorChanged(value, ignore = true) {
            this.ignoreChanges = ignore;
            this.colorPickerChange.emit(value);
        }
        colorSelected(value) {
            this.colorPickerSelect.emit(value);
        }
        colorCanceled() {
            this.colorPickerCancel.emit();
        }
        inputFocus() {
            const element = this.elRef.nativeElement;
            const ignored = this.cpIgnoredElements.filter((item) => item === element);
            if (!this.cpDisabled && !ignored.length) {
                if (typeof document !== 'undefined' && element === document.activeElement) {
                    this.openDialog();
                }
                else if (!this.dialog || !this.dialog.show) {
                    this.openDialog();
                }
                else {
                    this.closeDialog();
                }
            }
        }
        inputChange(event) {
            if (this.dialog) {
                this.dialog.setColorFromString(event.target.value, true);
            }
            else {
                this.colorPicker = event.target.value;
                this.colorPickerChange.emit(this.colorPicker);
            }
        }
        inputChanged(event) {
            this.cpInputChange.emit(event);
        }
        sliderChanged(event) {
            this.cpSliderChange.emit(event);
        }
        sliderDragEnd(event) {
            this.cpSliderDragEnd.emit(event);
        }
        sliderDragStart(event) {
            this.cpSliderDragStart.emit(event);
        }
        presetColorsChanged(value) {
            this.cpPresetColorsChange.emit(value);
        }
    };
    ColorPickerDirective.ɵfac = function ColorPickerDirective_Factory(t) { return new (t || ColorPickerDirective)(ɵɵdirectiveInject(Injector), ɵɵdirectiveInject(ComponentFactoryResolver), ɵɵdirectiveInject(ApplicationRef), ɵɵdirectiveInject(ViewContainerRef), ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(ColorPickerService)); };
    ColorPickerDirective.ɵdir = ɵɵdefineDirective({ type: ColorPickerDirective, selectors: [["", "colorPicker", ""]], hostBindings: function ColorPickerDirective_HostBindings(rf, ctx) { if (rf & 1) {
            ɵɵlistener("click", function ColorPickerDirective_click_HostBindingHandler() { return ctx.handleClick(); })("focus", function ColorPickerDirective_focus_HostBindingHandler() { return ctx.handleFocus(); })("input", function ColorPickerDirective_input_HostBindingHandler($event) { return ctx.handleInput($event); });
        } }, inputs: { cpWidth: "cpWidth", cpHeight: "cpHeight", cpToggle: "cpToggle", cpDisabled: "cpDisabled", cpIgnoredElements: "cpIgnoredElements", cpFallbackColor: "cpFallbackColor", cpColorMode: "cpColorMode", cpCmykEnabled: "cpCmykEnabled", cpOutputFormat: "cpOutputFormat", cpAlphaChannel: "cpAlphaChannel", cpDisableInput: "cpDisableInput", cpDialogDisplay: "cpDialogDisplay", cpSaveClickOutside: "cpSaveClickOutside", cpCloseClickOutside: "cpCloseClickOutside", cpUseRootViewContainer: "cpUseRootViewContainer", cpPosition: "cpPosition", cpPositionOffset: "cpPositionOffset", cpPositionRelativeToArrow: "cpPositionRelativeToArrow", cpOKButton: "cpOKButton", cpOKButtonText: "cpOKButtonText", cpOKButtonClass: "cpOKButtonClass", cpCancelButton: "cpCancelButton", cpCancelButtonText: "cpCancelButtonText", cpCancelButtonClass: "cpCancelButtonClass", cpPresetLabel: "cpPresetLabel", cpPresetColorsClass: "cpPresetColorsClass", cpMaxPresetColorsLength: "cpMaxPresetColorsLength", cpPresetEmptyMessage: "cpPresetEmptyMessage", cpPresetEmptyMessageClass: "cpPresetEmptyMessageClass", cpAddColorButton: "cpAddColorButton", cpAddColorButtonText: "cpAddColorButtonText", cpAddColorButtonClass: "cpAddColorButtonClass", cpRemoveColorButtonClass: "cpRemoveColorButtonClass", colorPicker: "colorPicker", cpPresetColors: "cpPresetColors" }, outputs: { cpInputChange: "cpInputChange", cpToggleChange: "cpToggleChange", cpSliderChange: "cpSliderChange", cpSliderDragEnd: "cpSliderDragEnd", cpSliderDragStart: "cpSliderDragStart", colorPickerOpen: "colorPickerOpen", colorPickerClose: "colorPickerClose", colorPickerCancel: "colorPickerCancel", colorPickerSelect: "colorPickerSelect", colorPickerChange: "colorPickerChange", cpCmykColorChange: "cpCmykColorChange", cpPresetColorsChange: "cpPresetColorsChange" }, exportAs: ["ngxColorPicker"], features: [ɵɵNgOnChangesFeature] });
    ColorPickerDirective.ctorParameters = () => [
        { type: Injector },
        { type: ComponentFactoryResolver },
        { type: ApplicationRef },
        { type: ViewContainerRef },
        { type: ElementRef },
        { type: ColorPickerService }
    ];
    __decorate([
        Input()
    ], ColorPickerDirective.prototype, "colorPicker", void 0);
    __decorate([
        Input()
    ], ColorPickerDirective.prototype, "cpWidth", void 0);
    __decorate([
        Input()
    ], ColorPickerDirective.prototype, "cpHeight", void 0);
    __decorate([
        Input()
    ], ColorPickerDirective.prototype, "cpToggle", void 0);
    __decorate([
        Input()
    ], ColorPickerDirective.prototype, "cpDisabled", void 0);
    __decorate([
        Input()
    ], ColorPickerDirective.prototype, "cpIgnoredElements", void 0);
    __decorate([
        Input()
    ], ColorPickerDirective.prototype, "cpFallbackColor", void 0);
    __decorate([
        Input()
    ], ColorPickerDirective.prototype, "cpColorMode", void 0);
    __decorate([
        Input()
    ], ColorPickerDirective.prototype, "cpCmykEnabled", void 0);
    __decorate([
        Input()
    ], ColorPickerDirective.prototype, "cpOutputFormat", void 0);
    __decorate([
        Input()
    ], ColorPickerDirective.prototype, "cpAlphaChannel", void 0);
    __decorate([
        Input()
    ], ColorPickerDirective.prototype, "cpDisableInput", void 0);
    __decorate([
        Input()
    ], ColorPickerDirective.prototype, "cpDialogDisplay", void 0);
    __decorate([
        Input()
    ], ColorPickerDirective.prototype, "cpSaveClickOutside", void 0);
    __decorate([
        Input()
    ], ColorPickerDirective.prototype, "cpCloseClickOutside", void 0);
    __decorate([
        Input()
    ], ColorPickerDirective.prototype, "cpUseRootViewContainer", void 0);
    __decorate([
        Input()
    ], ColorPickerDirective.prototype, "cpPosition", void 0);
    __decorate([
        Input()
    ], ColorPickerDirective.prototype, "cpPositionOffset", void 0);
    __decorate([
        Input()
    ], ColorPickerDirective.prototype, "cpPositionRelativeToArrow", void 0);
    __decorate([
        Input()
    ], ColorPickerDirective.prototype, "cpOKButton", void 0);
    __decorate([
        Input()
    ], ColorPickerDirective.prototype, "cpOKButtonText", void 0);
    __decorate([
        Input()
    ], ColorPickerDirective.prototype, "cpOKButtonClass", void 0);
    __decorate([
        Input()
    ], ColorPickerDirective.prototype, "cpCancelButton", void 0);
    __decorate([
        Input()
    ], ColorPickerDirective.prototype, "cpCancelButtonText", void 0);
    __decorate([
        Input()
    ], ColorPickerDirective.prototype, "cpCancelButtonClass", void 0);
    __decorate([
        Input()
    ], ColorPickerDirective.prototype, "cpPresetLabel", void 0);
    __decorate([
        Input()
    ], ColorPickerDirective.prototype, "cpPresetColors", void 0);
    __decorate([
        Input()
    ], ColorPickerDirective.prototype, "cpPresetColorsClass", void 0);
    __decorate([
        Input()
    ], ColorPickerDirective.prototype, "cpMaxPresetColorsLength", void 0);
    __decorate([
        Input()
    ], ColorPickerDirective.prototype, "cpPresetEmptyMessage", void 0);
    __decorate([
        Input()
    ], ColorPickerDirective.prototype, "cpPresetEmptyMessageClass", void 0);
    __decorate([
        Input()
    ], ColorPickerDirective.prototype, "cpAddColorButton", void 0);
    __decorate([
        Input()
    ], ColorPickerDirective.prototype, "cpAddColorButtonText", void 0);
    __decorate([
        Input()
    ], ColorPickerDirective.prototype, "cpAddColorButtonClass", void 0);
    __decorate([
        Input()
    ], ColorPickerDirective.prototype, "cpRemoveColorButtonClass", void 0);
    __decorate([
        Output()
    ], ColorPickerDirective.prototype, "cpInputChange", void 0);
    __decorate([
        Output()
    ], ColorPickerDirective.prototype, "cpToggleChange", void 0);
    __decorate([
        Output()
    ], ColorPickerDirective.prototype, "cpSliderChange", void 0);
    __decorate([
        Output()
    ], ColorPickerDirective.prototype, "cpSliderDragEnd", void 0);
    __decorate([
        Output()
    ], ColorPickerDirective.prototype, "cpSliderDragStart", void 0);
    __decorate([
        Output()
    ], ColorPickerDirective.prototype, "colorPickerOpen", void 0);
    __decorate([
        Output()
    ], ColorPickerDirective.prototype, "colorPickerClose", void 0);
    __decorate([
        Output()
    ], ColorPickerDirective.prototype, "colorPickerCancel", void 0);
    __decorate([
        Output()
    ], ColorPickerDirective.prototype, "colorPickerSelect", void 0);
    __decorate([
        Output()
    ], ColorPickerDirective.prototype, "colorPickerChange", void 0);
    __decorate([
        Output()
    ], ColorPickerDirective.prototype, "cpCmykColorChange", void 0);
    __decorate([
        Output()
    ], ColorPickerDirective.prototype, "cpPresetColorsChange", void 0);
    __decorate([
        HostListener('click')
    ], ColorPickerDirective.prototype, "handleClick", null);
    __decorate([
        HostListener('focus')
    ], ColorPickerDirective.prototype, "handleFocus", null);
    __decorate([
        HostListener('input', ['$event'])
    ], ColorPickerDirective.prototype, "handleInput", null);

    let ColorPickerModule = class ColorPickerModule {
    };
    ColorPickerModule.ɵmod = ɵɵdefineNgModule({ type: ColorPickerModule });
    ColorPickerModule.ɵinj = ɵɵdefineInjector({ factory: function ColorPickerModule_Factory(t) { return new (t || ColorPickerModule)(); }, providers: [ColorPickerService], imports: [[CommonModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(ColorPickerModule, { declarations: function () { return [ColorPickerComponent,
            ColorPickerDirective,
            TextDirective,
            SliderDirective]; }, imports: function () { return [CommonModule]; }, exports: function () { return [ColorPickerDirective]; } }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/widgets/range_input/range_input_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    const _c0$p = ["container"];
    // Keep this in sync with range_input_component.scss's `$_thumb-size`.
    /** @type {?} */
    const THUMB_SIZE_PX = 12;
    /** @enum {number} */
    const Position = {
        NONE: 0,
        LEFT: 1,
        RIGHT: 2,
    };
    Position[Position.NONE] = 'NONE';
    Position[Position.LEFT] = 'LEFT';
    Position[Position.RIGHT] = 'RIGHT';
    /**
     * Double thumbed slider component used for selecting a numeric range.
     *
     * Anatomy of the component:
     *
     *       left input             right input
     *   +-----------------+     +----------------+
     *   |                 |     |                |
     *   +-----------------+     +----------------+
     *       <lowerValue>           <upperValue>
     *
     *         left thumb         right thumb
     *           +---+               +---+
     *           |   +     track     +   |
     *   x+--------+===================+---------+x
     * <min>     |   +               +   |      <max>
     *           +---+               +---+
     *        <lowerValue>        <upperValue>
     *
     * Features:
     * - you can drag a thumb to change lowerValue or upperValue
     * - you cannot click on track to change any value
     * - a thumb is centered (origin of the circle) w.r.t a value.
     * - when lowerValue cross upperValue, lowerValue = upperValue and upperValue
     *     changes. Converse is true, too.
     * - does not validate input (e.g., lowerValue can be lower than min) but thumbs
     *   are clipped to `min` and `max`. Also, when emitting changes, the values can
     *   never exceed `min` and `max`.
     */
    class RangeInputComponent {
        /**
         * @param {?} changeDetector
         */
        constructor(changeDetector) {
            this.changeDetector = changeDetector;
            /**
             * `null` denotes contiguous "ticks"
             */
            this.tickCount = 20;
            this.value = new EventEmitter();
            this.Position = Position;
            this.activeThumb = Position.NONE;
            /**
             * Accounts for position of cursor when pressing down on the thumb.
             *
             * To illustrate the point, imagine the case when min=0, max=1, thumb has
             * radius of 100px, the thumb is set to max, and the input is positioned from
             * (0, 0) in the viewport. User can change max by mouse downing at (51, 0) but
             * that should not change the upperValue to 0.51. Instead, it should stay
             * at 1. In this case, since you cannot move the mouse past (0, 0), you can
             * never set the value to <= 49 but, normnally, the input does not get
             * rendered in (0, 0), thumb radius is 6px, and user can mouse down at middle
             * of the thumb.
             */
            this.offsetXFromOriginOfActiveThumb = 0;
            this.ngUnsubscribe = new Subject();
        }
        /**
         * @param {?} value
         * @return {?}
         */
        getThumbPosition(value) {
            /** @type {?} */
            const clippedValue = this.getClippedValue(value);
            /** @type {?} */
            const boundSize = this.max - this.min;
            if (boundSize <= 0) {
                return '50%';
            }
            /** @type {?} */
            const percentDifference = ((clippedValue - this.min) / boundSize) * 100;
            return `${percentDifference}%`;
        }
        /**
         * @return {?}
         */
        getTrackWidth() {
            /** @type {?} */
            const boundSize = this.max - this.min;
            if (boundSize <= 0) {
                return '0%';
            }
            /** @type {?} */
            const valDiff = this.getClippedValue(this.upperValue) -
                this.getClippedValue(this.lowerValue);
            /** @type {?} */
            const percentDifference = (valDiff / boundSize) * 100;
            return `${percentDifference}%`;
        }
        /**
         * @private
         * @param {?} value
         * @return {?}
         */
        getClippedValue(value) {
            return Math.min(Math.max(value, this.min), this.max);
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this.ngUnsubscribe.next();
            this.ngUnsubscribe.complete();
        }
        /**
         * @return {?}
         */
        ngOnInit() {
            fromEvent(document, 'mousemove', { passive: true })
                .pipe(takeUntil(this.ngUnsubscribe))
                .subscribe((/**
             * @param {?} event
             * @return {?}
             */
            (event) => {
                this.handleMouseMove((/** @type {?} */ (event)));
            }));
            fromEvent(document, 'mouseup', { passive: true })
                .pipe(takeUntil(this.ngUnsubscribe))
                .subscribe((/**
             * @param {?} event
             * @return {?}
             */
            (event) => {
                this.handleMouseOut((/** @type {?} */ (event)));
            }));
        }
        /**
         * @param {?} event
         * @param {?} position
         * @return {?}
         */
        handleMouseDown(event, position) {
            this.activeThumb = position;
            // Mouse event reports cursor position w.r.t the top left edge of the target
            // (in this case, a thumb) element.
            /** @type {?} */
            const offsetXFromLeftOfThumb = event.offsetX;
            // The thumb is visually centered w.r.t. a value using negative margin of
            // THUMB_SIZE_PX / 2. Account for cursor offset w.r.t the origin of the
            // active thumb so the value does not change simply by mousing down on the
            // thumb.
            /** @type {?} */
            const offsetXFromCenterOfThumb = THUMB_SIZE_PX / 2 - offsetXFromLeftOfThumb;
            this.offsetXFromOriginOfActiveThumb = offsetXFromCenterOfThumb;
        }
        /**
         * @private
         * @param {?} event
         * @return {?}
         */
        calculateValueFromMouseEvent(event) {
            const { left, right } = this.container.nativeElement.getBoundingClientRect();
            // Compute cursor position relative to left edge of the range-input element.
            /** @type {?} */
            const relativeXPx = event.clientX - left;
            // Compensate for cursor offset from origin of the active thumb.
            /** @type {?} */
            const compensatedRelativeXInPx = relativeXPx + this.offsetXFromOriginOfActiveThumb;
            /** @type {?} */
            let xPositionInPercent;
            if (this.tickCount !== null && this.tickCount > 0) {
                /** @type {?} */
                const tickWidthInPx = (right - left) / this.tickCount;
                /** @type {?} */
                const tickStuckRelativePx = Math.round(compensatedRelativeXInPx / tickWidthInPx) * tickWidthInPx;
                xPositionInPercent = tickStuckRelativePx / (right - left);
            }
            else {
                xPositionInPercent = compensatedRelativeXInPx / (right - left);
            }
            return this.getClippedValue(this.min + (this.max - this.min) * xPositionInPercent);
        }
        /**
         * Handles `mousemove` event in the document and, if dragging (mousedown on
         * thumb happened before), compute new relative position of the active thumb
         * and, if changed, event it to parent. If the left thumb is dragged past the
         * right thumb, the effect (from the user perspective) is that the left thumb
         * stays where the right thumb was, and the drag continues, moving the right
         * thumb instead.
         *
         * Implementation note: especially when `lowerValue` and `upperValue` are
         * nearby, you need to know which value is currently being changed (imagine
         * the cursor being right in between): to disambiguate, we have
         * `this.activeThumb`. However, of course, when `lowerValue` crosses
         * `upperValue`, or vice a versa, we need to update the `this.activeThumb`
         * accordingly. This is especially important because the component is a
         * "controlled component" [1] and, when props update, we do not know whether
         * `lowerValue` or `upperValue` correspond to the activeThumb (i.e., if this
         * were completely uncontrolled component, we can take initial position of the
         * thumb via prop, track DOM/position of active and inactive thumbs to
         * update/move the correct DOM).
         *
         * [1]: https://reactjs.org/docs/forms.html#controlled-components
         * @private
         * @param {?} event
         * @return {?}
         */
        handleMouseMove(event) {
            if (this.activeThumb === Position.NONE) {
                return;
            }
            /** @type {?} */
            const newValue = this.calculateValueFromMouseEvent(event);
            /** @type {?} */
            let nextValues = [this.lowerValue, this.upperValue];
            if (this.activeThumb === Position.LEFT) {
                if (newValue > this.upperValue) {
                    this.activeThumb = Position.RIGHT;
                }
                nextValues = [newValue, this.upperValue];
            }
            else {
                if (newValue < this.lowerValue) {
                    this.activeThumb = Position.LEFT;
                }
                nextValues = [this.lowerValue, newValue];
            }
            this.maybeNotifyNextValue(nextValues);
            this.changeDetector.markForCheck();
        }
        /**
         * @private
         * @param {?} minAndMax
         * @return {?}
         */
        maybeNotifyNextValue(minAndMax) {
            const [lowerValue, upperValue] = minAndMax.sort((/**
             * @param {?} a
             * @param {?} b
             * @return {?}
             */
            (a, b) => a - b));
            if (this.lowerValue !== lowerValue || this.upperValue !== upperValue) {
                this.value.emit({ lowerValue, upperValue });
            }
        }
        /**
         * @private
         * @param {?} event
         * @return {?}
         */
        handleMouseOut(event) {
            if (this.activeThumb !== Position.NONE) {
                this.activeThumb = Position.NONE;
                this.changeDetector.markForCheck();
            }
        }
        /**
         * @param {?} event
         * @param {?} position
         * @return {?}
         */
        handleInputChange(event, position) {
            /** @type {?} */
            const input = (/** @type {?} */ ((/** @type {?} */ (event.target))));
            /** @type {?} */
            const numValue = this.getClippedValue(Number(input.value));
            if (isNaN(numValue)) {
                return;
            }
            /** @type {?} */
            let nextValues = [this.lowerValue, this.upperValue];
            if (position === Position.LEFT) {
                nextValues = [numValue, this.upperValue];
            }
            else {
                nextValues = [this.lowerValue, numValue];
            }
            this.maybeNotifyNextValue(nextValues);
        }
        /**
         * @param {?} position
         * @return {?}
         */
        isThumbActive(position) {
            return this.activeThumb === position;
        }
    }
    RangeInputComponent.decorators = [
        { type: Component, args: [{
                    selector: 'tb-range-input',
                    templateUrl: './range_input_component.ng.html',
                    styleUrls: ['./range_input_component.css'],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    RangeInputComponent.ctorParameters = () => [
        { type: ChangeDetectorRef }
    ];
    RangeInputComponent.propDecorators = {
        container: [{ type: ViewChild, args: ['container', { static: true, read: ElementRef },] }],
        min: [{ type: Input }],
        max: [{ type: Input }],
        lowerValue: [{ type: Input }],
        upperValue: [{ type: Input }],
        tickCount: [{ type: Input }],
        value: [{ type: Output }]
    };
    /** @nocollapse */ RangeInputComponent.ɵfac = function RangeInputComponent_Factory(t) { return new (t || RangeInputComponent)(ɵɵdirectiveInject(ChangeDetectorRef)); };
    /** @nocollapse */ RangeInputComponent.ɵcmp = ɵɵdefineComponent({ type: RangeInputComponent, selectors: [["tb-range-input"]], viewQuery: function RangeInputComponent_Query(rf, ctx) { if (rf & 1) {
            ɵɵstaticViewQuery(_c0$p, true, ElementRef);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.container = _t.first);
        } }, inputs: { min: "min", max: "max", lowerValue: "lowerValue", upperValue: "upperValue", tickCount: "tickCount" }, outputs: { value: "value" }, decls: 8, vars: 14, consts: [["type", "number", 1, "lower-input", 3, "value", "change"], ["type", "number", 1, "upper-input", 3, "value", "change"], [1, "container"], ["container", ""], [1, "slider-track"], [1, "slider-track-fill"], [1, "thumb", 3, "mousedown"]], template: function RangeInputComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "input", 0);
            ɵɵlistener("change", function RangeInputComponent_Template_input_change_0_listener($event) { return ctx.handleInputChange($event, ctx.Position.LEFT); });
            ɵɵelementEnd();
            ɵɵelementStart(1, "input", 1);
            ɵɵlistener("change", function RangeInputComponent_Template_input_change_1_listener($event) { return ctx.handleInputChange($event, ctx.Position.RIGHT); });
            ɵɵelementEnd();
            ɵɵelementStart(2, "span", 2, 3);
            ɵɵelement(4, "span", 4);
            ɵɵelement(5, "span", 5);
            ɵɵelementStart(6, "span", 6);
            ɵɵlistener("mousedown", function RangeInputComponent_Template_span_mousedown_6_listener($event) { return ctx.handleMouseDown($event, ctx.Position.LEFT); });
            ɵɵelementEnd();
            ɵɵelementStart(7, "span", 6);
            ɵɵlistener("mousedown", function RangeInputComponent_Template_span_mousedown_7_listener($event) { return ctx.handleMouseDown($event, ctx.Position.RIGHT); });
            ɵɵelementEnd();
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵproperty("value", ctx.lowerValue);
            ɵɵadvance(1);
            ɵɵproperty("value", ctx.upperValue);
            ɵɵadvance(4);
            ɵɵstyleProp("left", ctx.getThumbPosition(ctx.lowerValue))("width", ctx.getTrackWidth());
            ɵɵadvance(1);
            ɵɵstyleProp("left", ctx.getThumbPosition(ctx.lowerValue));
            ɵɵclassProp("active", ctx.isThumbActive(ctx.Position.LEFT));
            ɵɵadvance(1);
            ɵɵstyleProp("left", ctx.getThumbPosition(ctx.upperValue));
            ɵɵclassProp("active", ctx.isThumbActive(ctx.Position.RIGHT));
        } }, styles: [".mat-badge-content[_ngcontent-%COMP%]{font-weight:600;font-size:12px;font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:9px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:24px}.mat-h1[_ngcontent-%COMP%], .mat-headline[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%]{font:400 24px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h2[_ngcontent-%COMP%], .mat-title[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h3[_ngcontent-%COMP%], .mat-subheading-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%]{font:400 16px/28px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h4[_ngcontent-%COMP%], .mat-subheading-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%]{font:400 15px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h5[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%]{font:400 calc(14px * 0.83)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-h6[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:400 calc(14px * 0.67)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-body-strong[_ngcontent-%COMP%], .mat-body-2[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   p[_ngcontent-%COMP%]{margin:0 0 12px}.mat-small[_ngcontent-%COMP%], .mat-caption[_ngcontent-%COMP%]{font:400 12px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-display-4[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-4[_ngcontent-%COMP%]{font:300 112px/112px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.05em;margin:0 0 56px}.mat-display-3[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-3[_ngcontent-%COMP%]{font:400 56px/56px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.02em;margin:0 0 64px}.mat-display-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-2[_ngcontent-%COMP%]{font:400 45px/48px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.005em;margin:0 0 64px}.mat-display-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-1[_ngcontent-%COMP%]{font:400 34px/40px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 64px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%], .mat-flat-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-button-toggle[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card-title[_ngcontent-%COMP%]{font-size:24px;font-weight:500}.mat-card-header[_ngcontent-%COMP%]   .mat-card-title[_ngcontent-%COMP%]{font-size:20px}.mat-card-subtitle[_ngcontent-%COMP%], .mat-card-content[_ngcontent-%COMP%]{font-size:14px}.mat-checkbox[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-checkbox-layout[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{line-height:24px}.mat-chip[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-chip[_ngcontent-%COMP%]   .mat-chip-trailing-icon.mat-icon[_ngcontent-%COMP%], .mat-chip[_ngcontent-%COMP%]   .mat-chip-remove.mat-icon[_ngcontent-%COMP%]{font-size:18px}.mat-table[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-header-cell[_ngcontent-%COMP%]{font-size:12px;font-weight:500}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{font-size:14px}.mat-calendar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-calendar-body[_ngcontent-%COMP%]{font-size:13px}.mat-calendar-body-label[_ngcontent-%COMP%], .mat-calendar-period-button[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-calendar-table-header[_ngcontent-%COMP%]   th[_ngcontent-%COMP%]{font-size:11px;font-weight:400}.mat-dialog-title[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-expansion-panel-header[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:15px;font-weight:400}.mat-expansion-panel-content[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field[_ngcontent-%COMP%]{font-size:inherit;font-weight:400;line-height:1.125;font-family:Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.34375em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{font-size:150%;line-height:1.125}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]{height:1.5em;width:1.5em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{height:1.125em;line-height:1.125}.mat-form-field-infix[_ngcontent-%COMP%]{padding:.5em 0;border-top:.84375em solid transparent}.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34375em) scale(0.75);width:133.3333333333%}.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34374em) scale(0.75);width:133.3333433333%}.mat-form-field-label-wrapper[_ngcontent-%COMP%]{top:-0.84375em;padding-top:.84375em}.mat-form-field-label[_ngcontent-%COMP%]{top:1.34375em}.mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.34375em}.mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{font-size:75%;margin-top:.6666666667em;top:calc(100% - 1.7916666667em)}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.4375em 0}.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.001px);-ms-transform:translateY(-1.28125em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00101px);-ms-transform:translateY(-1.28124em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00102px);-ms-transform:translateY(-1.28123em) scale(0.75);width:133.3333533333%}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.28125em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{margin-top:.5416666667em;top:calc(100% - 1.6666666667em)}@media print{.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28122em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28121em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.2812em) scale(0.75)}}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.25em 0 .75em 0}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.09375em;margin-top:-0.5em}.mat-form-field-appearance-fill.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59374em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:1em 0 1em 0}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.84375em;margin-top:-0.25em}.mat-form-field-appearance-outline.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59374em) scale(0.75);width:133.3333433333%}.mat-grid-tile-header[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]{font-size:14px}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2), .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}input.mat-input-element[_ngcontent-%COMP%]{margin-top:-0.0625em}.mat-menu-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:400}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px}.mat-radio-button[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select-trigger[_ngcontent-%COMP%]{height:1.125em}.mat-slide-toggle-content[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-slider-thumb-label-text[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-stepper-vertical[_ngcontent-%COMP%], .mat-stepper-horizontal[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-step-label[_ngcontent-%COMP%]{font-size:14px;font-weight:400}.mat-step-sub-label-error[_ngcontent-%COMP%]{font-weight:normal}.mat-step-label-error[_ngcontent-%COMP%]{font-size:14px}.mat-step-label-selected[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-tab-group[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-toolbar[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0}.mat-tooltip[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:10px;padding-top:6px;padding-bottom:6px}.mat-tooltip-handset[_ngcontent-%COMP%]{font-size:14px;padding-top:8px;padding-bottom:8px}.mat-list-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:16px}.mat-optgroup-label[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-simple-snackbar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px}.mat-simple-snackbar-action[_ngcontent-%COMP%]{line-height:1;font-family:inherit;font-size:inherit;font-weight:500}.mat-tree[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{font-weight:400;font-size:14px}.mat-ripple[_ngcontent-%COMP%]{overflow:hidden;position:relative}.mat-ripple[_ngcontent-%COMP%]:not(:empty){transform:translateZ(0)}.mat-ripple.mat-ripple-unbounded[_ngcontent-%COMP%]{overflow:visible}.mat-ripple-element[_ngcontent-%COMP%]{position:absolute;border-radius:50%;pointer-events:none;transition:opacity,transform 0ms cubic-bezier(0, 0, 0.2, 1);transform:scale(0)}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{display:none}.cdk-visually-hidden[_ngcontent-%COMP%]{border:0;clip:rect(0 0 0 0);height:1px;margin:-1px;overflow:hidden;padding:0;position:absolute;width:1px;outline:0;-webkit-appearance:none;-moz-appearance:none}.cdk-overlay-container[_ngcontent-%COMP%], .cdk-global-overlay-wrapper[_ngcontent-%COMP%]{pointer-events:none;top:0;left:0;height:100%;width:100%}.cdk-overlay-container[_ngcontent-%COMP%]{position:fixed;z-index:1000}.cdk-overlay-container[_ngcontent-%COMP%]:empty{display:none}.cdk-global-overlay-wrapper[_ngcontent-%COMP%]{display:flex;position:absolute;z-index:1000}.cdk-overlay-pane[_ngcontent-%COMP%]{position:absolute;pointer-events:auto;box-sizing:border-box;z-index:1000;display:flex;max-width:100%;max-height:100%}.cdk-overlay-backdrop[_ngcontent-%COMP%]{position:absolute;top:0;bottom:0;left:0;right:0;z-index:1000;pointer-events:auto;-webkit-tap-highlight-color:transparent;transition:opacity 400ms cubic-bezier(0.25, 0.8, 0.25, 1);opacity:0}.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:1}@media screen and (-ms-high-contrast: active){.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:.6}}.cdk-overlay-dark-backdrop[_ngcontent-%COMP%]{background:rgba(0,0,0,.32)}.cdk-overlay-transparent-backdrop[_ngcontent-%COMP%], .cdk-overlay-transparent-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:0}.cdk-overlay-connected-position-bounding-box[_ngcontent-%COMP%]{position:absolute;z-index:1000;display:flex;flex-direction:column;min-width:1px;min-height:1px}.cdk-global-scrollblock[_ngcontent-%COMP%]{position:fixed;width:100%;overflow-y:scroll}@keyframes cdk-text-field-autofill-start{}@keyframes cdk-text-field-autofill-end{}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:-webkit-autofill{animation:cdk-text-field-autofill-start 0s 1ms}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:not(:-webkit-autofill){animation:cdk-text-field-autofill-end 0s 1ms}textarea.cdk-textarea-autosize[_ngcontent-%COMP%]{resize:none}textarea.cdk-textarea-autosize-measuring[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:auto !important;overflow:hidden !important}textarea.cdk-textarea-autosize-measuring-firefox[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:0 !important}.mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.1)}.mat-option[_ngcontent-%COMP%]{color:#212121}.mat-option[_ngcontent-%COMP%]:hover:not(.mat-option-disabled), .mat-option[_ngcontent-%COMP%]:focus:not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple):not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-active[_ngcontent-%COMP%]{background:rgba(0,0,0,.04);color:#212121}.mat-option.mat-option-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-primary[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f57c00}.mat-accent[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f44336}.mat-optgroup-label[_ngcontent-%COMP%]{color:#616161}.mat-optgroup-disabled[_ngcontent-%COMP%]   .mat-optgroup-label[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-pseudo-checkbox[_ngcontent-%COMP%]{color:#616161}.mat-pseudo-checkbox[_ngcontent-%COMP%]::after{color:#fafafa}.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{color:#b0b0b0}.mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f57c00}.mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f44336}.mat-pseudo-checkbox-checked.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{background:#b0b0b0}.mat-elevation-z0[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z1[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z2[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z3[_ngcontent-%COMP%]{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z4[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z5[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 5px 8px 0px rgba(0, 0, 0, 0.14),0px 1px 14px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z6[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z7[_ngcontent-%COMP%]{box-shadow:0px 4px 5px -2px rgba(0, 0, 0, 0.2),0px 7px 10px 1px rgba(0, 0, 0, 0.14),0px 2px 16px 1px rgba(0, 0, 0, 0.12)}.mat-elevation-z8[_ngcontent-%COMP%]{box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z9[_ngcontent-%COMP%]{box-shadow:0px 5px 6px -3px rgba(0, 0, 0, 0.2),0px 9px 12px 1px rgba(0, 0, 0, 0.14),0px 3px 16px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z10[_ngcontent-%COMP%]{box-shadow:0px 6px 6px -3px rgba(0, 0, 0, 0.2),0px 10px 14px 1px rgba(0, 0, 0, 0.14),0px 4px 18px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z11[_ngcontent-%COMP%]{box-shadow:0px 6px 7px -4px rgba(0, 0, 0, 0.2),0px 11px 15px 1px rgba(0, 0, 0, 0.14),0px 4px 20px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z12[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z13[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 13px 19px 2px rgba(0, 0, 0, 0.14),0px 5px 24px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z14[_ngcontent-%COMP%]{box-shadow:0px 7px 9px -4px rgba(0, 0, 0, 0.2),0px 14px 21px 2px rgba(0, 0, 0, 0.14),0px 5px 26px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z15[_ngcontent-%COMP%]{box-shadow:0px 8px 9px -5px rgba(0, 0, 0, 0.2),0px 15px 22px 2px rgba(0, 0, 0, 0.14),0px 6px 28px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z16[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z17[_ngcontent-%COMP%]{box-shadow:0px 8px 11px -5px rgba(0, 0, 0, 0.2),0px 17px 26px 2px rgba(0, 0, 0, 0.14),0px 6px 32px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z18[_ngcontent-%COMP%]{box-shadow:0px 9px 11px -5px rgba(0, 0, 0, 0.2),0px 18px 28px 2px rgba(0, 0, 0, 0.14),0px 7px 34px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z19[_ngcontent-%COMP%]{box-shadow:0px 9px 12px -6px rgba(0, 0, 0, 0.2),0px 19px 29px 2px rgba(0, 0, 0, 0.14),0px 7px 36px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z20[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 20px 31px 3px rgba(0, 0, 0, 0.14),0px 8px 38px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z21[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 21px 33px 3px rgba(0, 0, 0, 0.14),0px 8px 40px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z22[_ngcontent-%COMP%]{box-shadow:0px 10px 14px -6px rgba(0, 0, 0, 0.2),0px 22px 35px 3px rgba(0, 0, 0, 0.14),0px 8px 42px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z23[_ngcontent-%COMP%]{box-shadow:0px 11px 14px -7px rgba(0, 0, 0, 0.2),0px 23px 36px 3px rgba(0, 0, 0, 0.14),0px 9px 44px 8px rgba(0, 0, 0, 0.12)}.mat-elevation-z24[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12)}.mat-app-background[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-theme-loaded-marker[_ngcontent-%COMP%]{display:none}.mat-autocomplete-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-autocomplete-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover){background:#fff}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover):not(.mat-option-disabled){color:#212121}.mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f57c00}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{outline:solid 1px;border-radius:0}.mat-badge-accent[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-badge-warn[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f44336}.mat-badge[_ngcontent-%COMP%]{position:relative}.mat-badge-hidden[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{display:none}.mat-badge-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#b9b9b9;color:#757575}.mat-badge-content[_ngcontent-%COMP%]{position:absolute;text-align:center;display:inline-block;border-radius:50%;transition:transform 200ms ease-in-out;transform:scale(0.6);overflow:hidden;white-space:nowrap;text-overflow:ellipsis;pointer-events:none}.ng-animate-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%], .mat-badge-content._mat-animation-noopable[_ngcontent-%COMP%]{transition:none}.mat-badge-content.mat-badge-active[_ngcontent-%COMP%]{transform:none}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:16px;height:16px;line-height:16px}.mat-badge-small.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-8px}.mat-badge-small.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-8px}.mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-16px}.mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-16px}.mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-8px}.mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-8px}.mat-badge-medium[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:22px;height:22px;line-height:22px}.mat-badge-medium.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-11px}.mat-badge-medium.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-11px}.mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-22px}.mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-22px}.mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-11px}.mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-11px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:28px;height:28px;line-height:28px}.mat-badge-large.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-14px}.mat-badge-large.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-14px}.mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-28px}.mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-28px}.mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-14px}.mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-14px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]{color:inherit;background:transparent}.mat-button.mat-primary[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-button[disabled][disabled][_ngcontent-%COMP%], .mat-icon-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-icon-button[disabled][disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-stroked-button[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f44336}.mat-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:transparent}.mat-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{opacity:.1;background-color:currentColor}.mat-button-focus-overlay[_ngcontent-%COMP%]{background:#000}.mat-stroked-button[_ngcontent-%COMP%]:not([disabled]){border-color:rgba(0,0,0,.12)}.mat-flat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{color:#212121;background-color:#fff}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-flat-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-stroked-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-flat-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-raised-button[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone[_ngcontent-%COMP%], .mat-button-toggle-group[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{box-shadow:none}.mat-button-toggle[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-button-toggle[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121;background:#fff}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:#000}.mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12)}[dir=rtl][_ngcontent-%COMP%]   .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-group-appearance-standard.mat-button-toggle-vertical[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:none;border-top:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#616161}.mat-button-toggle-checked.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121}.mat-button-toggle-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.26);background-color:#eee}.mat-button-toggle-disabled.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{background:#fff}.mat-button-toggle-disabled.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#bdbdbd}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{border:solid 1px rgba(0,0,0,.12)}.mat-card[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-card[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-card.mat-card-flat[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-card-subtitle[_ngcontent-%COMP%]{color:#616161}.mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#616161}.mat-checkbox-checkmark[_ngcontent-%COMP%]{fill:#fafafa}.mat-checkbox-checkmark-path[_ngcontent-%COMP%]{stroke:#fafafa !important}.mat-checkbox-mixedmark[_ngcontent-%COMP%]{background-color:#fafafa}.mat-checkbox-indeterminate.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f57c00}.mat-checkbox-indeterminate.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#ff9800}.mat-checkbox-indeterminate.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f44336}.mat-checkbox-disabled.mat-checkbox-checked[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-disabled.mat-checkbox-indeterminate[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]:not(.mat-checkbox-checked)   .mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{color:#616161}.mat-checkbox[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%]{background:#f57c00}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%]{background:#ff9800}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%]{background:#f44336}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#212121}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#212121;opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled):active{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled)   .mat-chip-remove[_ngcontent-%COMP%]:hover{opacity:.54}.mat-chip.mat-standard-chip.mat-chip-disabled[_ngcontent-%COMP%]{opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]::after{background:#000}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-table[_ngcontent-%COMP%]{background:#fff}.mat-table[_ngcontent-%COMP%]   thead[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tbody[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tfoot[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], [mat-header-row][_ngcontent-%COMP%], [mat-row][_ngcontent-%COMP%], [mat-footer-row][_ngcontent-%COMP%], .mat-table-sticky[_ngcontent-%COMP%]{background:inherit}mat-row[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], th.mat-header-cell[_ngcontent-%COMP%], td.mat-cell[_ngcontent-%COMP%], td.mat-footer-cell[_ngcontent-%COMP%]{border-bottom-color:rgba(0,0,0,.12)}.mat-header-cell[_ngcontent-%COMP%]{color:#616161}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{color:#212121}.mat-calendar-arrow[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.54)}.mat-datepicker-toggle[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-next-button[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-previous-button[_ngcontent-%COMP%]{color:rgba(0,0,0,.54)}.mat-calendar-table-header[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-calendar-table-header-divider[_ngcontent-%COMP%]::after{background:rgba(0,0,0,.12)}.mat-calendar-body-label[_ngcontent-%COMP%]{color:#616161}.mat-calendar-body-cell-content[_ngcontent-%COMP%]{color:#212121;border-color:transparent}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){color:#757575}.mat-calendar-body-cell[_ngcontent-%COMP%]:not(.mat-calendar-body-disabled):hover > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-keyboard-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-program-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){background-color:rgba(0,0,0,.04)}.mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.38)}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.18)}.mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.4)}.mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12);background-color:#fff;color:#212121}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.4)}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.4)}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content-touch[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-datepicker-toggle-active[_ngcontent-%COMP%]{color:#f57c00}.mat-datepicker-toggle-active.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-datepicker-toggle-active.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-dialog-container[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-divider[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-divider-vertical[_ngcontent-%COMP%]{border-right-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-expansion-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-action-row[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-keyboard-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-program-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded)   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover:not([aria-disabled=true]){background:rgba(0,0,0,.04)}@media(hover: none){.mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded):not([aria-disabled=true])   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover{background:#fff}}.mat-expansion-panel-header-title[_ngcontent-%COMP%]{color:#212121}.mat-expansion-panel-header-description[_ngcontent-%COMP%], .mat-expansion-indicator[_ngcontent-%COMP%]::after{color:#616161}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-title[_ngcontent-%COMP%], .mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-description[_ngcontent-%COMP%]{color:inherit}.mat-form-field-label[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-hint[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-focused[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-ripple[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid)   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f57c00}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-accent   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#ff9800}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-warn   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#f44336}.mat-error[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-hint[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-legacy.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-standard[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-standard.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.04)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.02)}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:transparent}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.12)}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:rgba(0,0,0,.87)}.mat-form-field-appearance-outline.mat-focused[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field-appearance-outline.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-appearance-outline.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-invalid.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.06)}.mat-icon.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-icon.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-icon.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-type-mat-native-select[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#616161}.mat-input-element[_ngcontent-%COMP%]:disabled, .mat-form-field-type-mat-native-select.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#757575}.mat-input-element[_ngcontent-%COMP%]{caret-color:#f57c00}.mat-input-element[_ngcontent-%COMP%]::placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-moz-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-webkit-input-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]:-ms-input-placeholder{color:rgba(97,97,97,.42)}.mat-accent[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%], .mat-form-field-invalid[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#f44336}.mat-form-field-type-mat-native-select.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{color:#616161}.mat-list-item-disabled[_ngcontent-%COMP%]{background-color:#eee}.mat-list-option[_ngcontent-%COMP%]:hover, .mat-list-option[_ngcontent-%COMP%]:focus, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.04)}.mat-list-single-selected-option[_ngcontent-%COMP%], .mat-list-single-selected-option[_ngcontent-%COMP%]:hover, .mat-list-single-selected-option[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.12)}.mat-menu-panel[_ngcontent-%COMP%]{background:#fff}.mat-menu-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-menu-item[_ngcontent-%COMP%]{background:transparent;color:#212121}.mat-menu-item[disabled][_ngcontent-%COMP%], .mat-menu-item[disabled][_ngcontent-%COMP%]::after{color:rgba(0,0,0,.38)}.mat-menu-item[_ngcontent-%COMP%]   .mat-icon-no-color[_ngcontent-%COMP%], .mat-menu-item-submenu-trigger[_ngcontent-%COMP%]::after{color:rgba(0,0,0,.54)}.mat-menu-item[_ngcontent-%COMP%]:hover:not([disabled]), .mat-menu-item.cdk-program-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item.cdk-keyboard-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item-highlighted[_ngcontent-%COMP%]:not([disabled]){background:rgba(0,0,0,.04)}.mat-paginator[_ngcontent-%COMP%]{background:#fff}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{color:#616161}.mat-paginator-decrement[_ngcontent-%COMP%], .mat-paginator-increment[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54);border-right:2px solid rgba(0,0,0,.54)}.mat-paginator-first[_ngcontent-%COMP%], .mat-paginator-last[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54)}.mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-decrement[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-increment[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-first[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-last[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f57c00}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#ff9800}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f44336}.mat-progress-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f57c00}.mat-progress-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#ff9800}.mat-progress-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f44336}.mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#616161}.mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f57c00}.mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#ff9800}.mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#ff9800}.mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f44336}.mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f44336}.mat-radio-button.mat-radio-disabled.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-label-content[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-radio-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-select-value[_ngcontent-%COMP%]{color:#212121}.mat-select-placeholder[_ngcontent-%COMP%]{color:rgba(97,97,97,.42)}.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%]{color:#757575}.mat-select-arrow[_ngcontent-%COMP%]{color:#616161}.mat-select-panel[_ngcontent-%COMP%]{background:#fff}.mat-select-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-select-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple){background:rgba(0,0,0,.12)}.mat-form-field.mat-focused.mat-primary[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-invalid[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#757575}.mat-drawer-container[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-drawer[_ngcontent-%COMP%]{background-color:#fff;color:#212121}.mat-drawer.mat-drawer-push[_ngcontent-%COMP%]{background-color:#fff}.mat-drawer[_ngcontent-%COMP%]:not(.mat-drawer-side){box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-drawer-side[_ngcontent-%COMP%]{border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-backdrop.mat-drawer-shown[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.6)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.54)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.54)}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.54)}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle[_ngcontent-%COMP%]:not(.mat-checked)   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-slide-toggle-thumb[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12);background-color:#fafafa}.mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-primary[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f57c00}.mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-accent[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#ff9800}.mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-warn[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f44336}.mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.2)}.mat-slider[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%], .cdk-focused[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-disabled[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]   .mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing)   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26);background-color:transparent}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26)}.mat-slider-has-ticks[_ngcontent-%COMP%]   .mat-slider-wrapper[_ngcontent-%COMP%]::after{border-color:rgba(0,0,0,.7)}.mat-slider-horizontal[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to right, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent);background-image:-moz-repeating-linear-gradient(0.0001deg, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-slider-vertical[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to bottom, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-step-header.cdk-keyboard-focused[_ngcontent-%COMP%], .mat-step-header.cdk-program-focused[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]:hover{background-color:rgba(0,0,0,.04)}@media(hover: none){.mat-step-header[_ngcontent-%COMP%]:hover{background:none}}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-optional[_ngcontent-%COMP%]{color:#616161}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon[_ngcontent-%COMP%]{background-color:#616161;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-selected[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-done[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-edit[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-error[_ngcontent-%COMP%]{background-color:transparent;color:#f44336}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-active[_ngcontent-%COMP%]{color:#212121}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-error[_ngcontent-%COMP%]{color:#f44336}.mat-stepper-horizontal[_ngcontent-%COMP%], .mat-stepper-vertical[_ngcontent-%COMP%]{background-color:#fff}.mat-stepper-vertical-line[_ngcontent-%COMP%]::before{border-left-color:rgba(0,0,0,.12)}.mat-horizontal-stepper-header[_ngcontent-%COMP%]::before, .mat-horizontal-stepper-header[_ngcontent-%COMP%]::after, .mat-stepper-horizontal-line[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-sort-header-arrow[_ngcontent-%COMP%]{color:#616161}.mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-header[_ngcontent-%COMP%]{border-bottom:1px solid rgba(0,0,0,.12)}.mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%]{border-top:1px solid rgba(0,0,0,.12);border-bottom:none}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{color:#212121}.mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:#757575}.mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#212121}.mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#757575}.mat-tab-group[class*=mat-background-][_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar[class*=mat-background-][_ngcontent-%COMP%]{border-bottom:none;border-top:none}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-toolbar[_ngcontent-%COMP%]{background:#f5f5f5;color:#212121}.mat-toolbar.mat-primary[_ngcontent-%COMP%]{background:#f57c00;color:#fff}.mat-toolbar.mat-accent[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-toolbar.mat-warn[_ngcontent-%COMP%]{background:#f44336;color:#fff}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:currentColor}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:inherit}.mat-toolbar[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:currentColor}.mat-tooltip[_ngcontent-%COMP%]{background:rgba(97,97,97,.9)}.mat-tree[_ngcontent-%COMP%]{background:#fff}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{color:#212121}.mat-snack-bar-container[_ngcontent-%COMP%]{color:rgba(255,255,255,.7);background:#323232;box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-simple-snackbar-action[_ngcontent-%COMP%]{color:#ff9800}[_nghost-%COMP%]{box-sizing:border-box;display:inline-grid;grid-gap:10px;grid-template-areas:\"lower-input upper-input\" \"slider slider\";font-size:0;lower-width:100px;padding:6px}input[_ngcontent-%COMP%]{box-sizing:border-box;overflow:hidden;width:100%}.lower-input[_ngcontent-%COMP%]{grid-area:lower-input}.upper-input[_ngcontent-%COMP%]{grid-area:upper-input;justify-self:flex-end}.container[_ngcontent-%COMP%]{grid-area:slider;align-items:center;box-sizing:border-box;display:inline-flex;height:12px;justify-content:center;position:relative;width:100%}.slider-track[_ngcontent-%COMP%]{background:rgba(0,0,0,.26);height:2px;width:100%}.slider-track-fill[_ngcontent-%COMP%]{background:#f57c00;position:absolute;height:2px}.thumb[_ngcontent-%COMP%]{background:#f57c00;border-radius:100%;display:inline-block;height:12px;margin-left:-6px;position:absolute;top:0;transform-origin:center;transition:transform .3s ease;box-shadow:0 0 0 1px rgba(0,0,0,.26);width:12px;will-change:transform}.thumb.active[_ngcontent-%COMP%]{transform:scale(1.2)}"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/widgets/range_input/range_input_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class RangeInputModule {
    }
    RangeInputModule.decorators = [
        { type: NgModule, args: [{
                    imports: [CommonModule],
                    exports: [RangeInputComponent],
                    declarations: [RangeInputComponent],
                },] },
    ];
    /** @nocollapse */ RangeInputModule.ɵmod = ɵɵdefineNgModule({ type: RangeInputModule });
    /** @nocollapse */ RangeInputModule.ɵinj = ɵɵdefineInjector({ factory: function RangeInputModule_Factory(t) { return new (t || RangeInputModule)(); }, imports: [[CommonModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(RangeInputModule, { declarations: [RangeInputComponent], imports: [CommonModule], exports: [RangeInputComponent] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/runs/views/runs_table/runs_table_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    const _c0$q = ["filter"];
    function RunsTableComponent_th_8_Template(rf, ctx) { if (rf & 1) {
        const _r20 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "th", 23);
        ɵɵelementStart(1, "mat-checkbox", 24);
        ɵɵlistener("change", function RunsTableComponent_th_8_Template_mat_checkbox_change_1_listener() { ɵɵrestoreView(_r20); const ctx_r19 = ɵɵnextContext(); return ctx_r19.handlePageToggle(); });
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r1 = ɵɵnextContext();
        ɵɵadvance(1);
        ɵɵproperty("checked", ctx_r1.allPageItemsSelected())("indeterminate", !ctx_r1.allPageItemsSelected() && ctx_r1.somePageItemsSelected());
    } }
    function RunsTableComponent_td_9_Template(rf, ctx) { if (rf & 1) {
        const _r23 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "td", 25);
        ɵɵelementStart(1, "mat-checkbox", 26);
        ɵɵlistener("change", function RunsTableComponent_td_9_Template_mat_checkbox_change_1_listener() { ɵɵrestoreView(_r23); const item_r21 = ctx.$implicit; const ctx_r22 = ɵɵnextContext(); return ctx_r22.onSelectionToggle.emit(item_r21); });
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const item_r21 = ctx.$implicit;
        ɵɵadvance(1);
        ɵɵproperty("checked", item_r21.selected);
    } }
    function RunsTableComponent_th_11_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "th", 27);
        ɵɵtext(1, " Experiment ");
        ɵɵelementEnd();
    } }
    function RunsTableComponent_td_12_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "td", 28);
        ɵɵelementStart(1, "span");
        ɵɵtext(2);
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const item_r24 = ctx.$implicit;
        ɵɵadvance(1);
        ɵɵattribute("title", item_r24.experimentName);
        ɵɵadvance(1);
        ɵɵtextInterpolate(item_r24.experimentAlias);
    } }
    function RunsTableComponent_th_14_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "th", 27);
        ɵɵtext(1, " Run ");
        ɵɵelementEnd();
    } }
    function RunsTableComponent_td_15_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "td", 28);
        ɵɵtext(1);
        ɵɵelementEnd();
    } if (rf & 2) {
        const item_r25 = ctx.$implicit;
        ɵɵadvance(1);
        ɵɵtextInterpolate(item_r25.run.name);
    } }
    function RunsTableComponent_th_17_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "th", 23);
    } }
    const _c1$g = function (a1) { return { "run-color-swatch": true, "no-color": a1 }; };
    function RunsTableComponent_td_18_Template(rf, ctx) { if (rf & 1) {
        const _r28 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "td", 25);
        ɵɵelementStart(1, "button", 29);
        ɵɵlistener("colorPickerChange", function RunsTableComponent_td_18_Template_button_colorPickerChange_1_listener($event) { ɵɵrestoreView(_r28); const item_r26 = ctx.$implicit; const ctx_r27 = ɵɵnextContext(); return ctx_r27.onRunColorChange.emit({ runId: item_r26.run.id, newColor: $event }); });
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const item_r26 = ctx.$implicit;
        ɵɵadvance(1);
        ɵɵstyleProp("background", item_r26.runColor, ɵɵdefaultStyleSanitizer);
        ɵɵproperty("ngClass", ɵɵpureFunction1(9, _c1$g, !item_r26.runColor))("colorPicker", item_r26.runColor)("cpDialogDisplay", "popup")("cpPosition", "bottom-right")("cpPositionOffset", 0 - 20)("cpUseRootViewContainer", true)("cpOutputFormat", "hex");
    } }
    function RunsTableComponent_th_20_div_1_Template(rf, ctx) { if (rf & 1) {
        const _r31 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "div", 31);
        ɵɵelementStart(1, "span");
        ɵɵtext(2);
        ɵɵelementEnd();
        ɵɵelementStart(3, "button", 32);
        ɵɵlistener("click", function RunsTableComponent_th_20_div_1_Template_button_click_3_listener() { ɵɵrestoreView(_r31); const ctx_r30 = ɵɵnextContext(2); return ctx_r30.onSelectAllPages.emit(); });
        ɵɵtext(4, " Select all runs ");
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r29 = ɵɵnextContext(2);
        ɵɵadvance(2);
        ɵɵtextInterpolate2("All runs in this page are selected but not all runs (", ctx_r29.numSelectedItems, " of ", ctx_r29.allItemsLength, ") are selected.");
    } }
    function RunsTableComponent_th_20_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "th", 23);
        ɵɵtemplate(1, RunsTableComponent_th_20_div_1_Template, 5, 2, "div", 30);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r9 = ɵɵnextContext();
        ɵɵattribute("colspan", ctx_r9.getColumnIds().length);
        ɵɵadvance(1);
        ɵɵproperty("ngIf", ctx_r9.allPageItemsSelected() && ctx_r9.numSelectedItems !== ctx_r9.allItemsLength);
    } }
    function RunsTableComponent_ng_container_21_th_1_ng_container_3_ng_container_9_Template(rf, ctx) { if (rf & 1) {
        const _r40 = ɵɵgetCurrentView();
        ɵɵelementContainerStart(0);
        ɵɵelementStart(1, "div", 40);
        ɵɵlistener("click", function RunsTableComponent_ng_container_21_th_1_ng_container_3_ng_container_9_Template_div_click_1_listener($event) { ɵɵrestoreView(_r40); return $event.stopPropagation(); });
        ɵɵelementStart(2, "tb-range-input", 41);
        ɵɵlistener("value", function RunsTableComponent_ng_container_21_th_1_ng_container_3_ng_container_9_Template_tb_range_input_value_2_listener($event) { ɵɵrestoreView(_r40); const column_r32 = ɵɵnextContext(3).$implicit; const ctx_r41 = ɵɵnextContext(); return ctx_r41.handleHparamIntervalChanged(column_r32, $event); });
        ɵɵelementEnd();
        ɵɵelementEnd();
        ɵɵelementContainerEnd();
    } if (rf & 2) {
        const column_r32 = ɵɵnextContext(3).$implicit;
        ɵɵadvance(2);
        ɵɵproperty("min", column_r32.filter.minValue)("max", column_r32.filter.maxValue)("lowerValue", column_r32.filter.filterLowerValue)("upperValue", column_r32.filter.filterUpperValue);
    } }
    function RunsTableComponent_ng_container_21_th_1_ng_container_3_ng_container_10_div_1_Template(rf, ctx) { if (rf & 1) {
        const _r47 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "div", 43);
        ɵɵlistener("click", function RunsTableComponent_ng_container_21_th_1_ng_container_3_ng_container_10_div_1_Template_div_click_0_listener($event) { ɵɵrestoreView(_r47); return $event.stopPropagation(); });
        ɵɵelementStart(1, "mat-checkbox", 26);
        ɵɵlistener("change", function RunsTableComponent_ng_container_21_th_1_ng_container_3_ng_container_10_div_1_Template_mat_checkbox_change_1_listener() { ɵɵrestoreView(_r47); const value_r45 = ctx.$implicit; const column_r32 = ɵɵnextContext(4).$implicit; const ctx_r48 = ɵɵnextContext(); return ctx_r48.handleHparamDiscreteChanged(column_r32, value_r45); });
        ɵɵelementStart(2, "span");
        ɵɵtext(3);
        ɵɵelementEnd();
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const value_r45 = ctx.$implicit;
        const column_r32 = ɵɵnextContext(4).$implicit;
        ɵɵadvance(1);
        ɵɵproperty("checked", column_r32.filter.filterValues.includes(value_r45));
        ɵɵadvance(2);
        ɵɵtextInterpolate(value_r45);
    } }
    function RunsTableComponent_ng_container_21_th_1_ng_container_3_ng_container_10_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementContainerStart(0);
        ɵɵtemplate(1, RunsTableComponent_ng_container_21_th_1_ng_container_3_ng_container_10_div_1_Template, 4, 2, "div", 42);
        ɵɵelementContainerEnd();
    } if (rf & 2) {
        const column_r32 = ɵɵnextContext(3).$implicit;
        ɵɵadvance(1);
        ɵɵproperty("ngForOf", column_r32.filter.possibleValues);
    } }
    function RunsTableComponent_ng_container_21_th_1_ng_container_3_Template(rf, ctx) { if (rf & 1) {
        const _r53 = ɵɵgetCurrentView();
        ɵɵelementContainerStart(0);
        ɵɵelementStart(1, "button", 36);
        ɵɵelement(2, "mat-icon", 37);
        ɵɵelementEnd();
        ɵɵelementStart(3, "mat-menu", null, 38);
        ɵɵelementStart(5, "div", 39);
        ɵɵlistener("click", function RunsTableComponent_ng_container_21_th_1_ng_container_3_Template_div_click_5_listener($event) { ɵɵrestoreView(_r53); return $event.stopPropagation(); });
        ɵɵelementStart(6, "mat-checkbox", 26);
        ɵɵlistener("change", function RunsTableComponent_ng_container_21_th_1_ng_container_3_Template_mat_checkbox_change_6_listener() { ɵɵrestoreView(_r53); const column_r32 = ɵɵnextContext(2).$implicit; const ctx_r54 = ɵɵnextContext(); return ctx_r54.handleHparamIncludeUndefinedToggled(column_r32); });
        ɵɵelementStart(7, "span");
        ɵɵtext(8, "(show empty value)");
        ɵɵelementEnd();
        ɵɵelementEnd();
        ɵɵelementEnd();
        ɵɵtemplate(9, RunsTableComponent_ng_container_21_th_1_ng_container_3_ng_container_9_Template, 3, 4, "ng-container", 35);
        ɵɵtemplate(10, RunsTableComponent_ng_container_21_th_1_ng_container_3_ng_container_10_Template, 2, 1, "ng-container", 35);
        ɵɵelementEnd();
        ɵɵelementContainerEnd();
    } if (rf & 2) {
        const _r36 = ɵɵreference(4);
        const column_r32 = ɵɵnextContext(2).$implicit;
        const ctx_r35 = ɵɵnextContext();
        ɵɵadvance(1);
        ɵɵproperty("matMenuTriggerFor", _r36);
        ɵɵattribute("aria-label", "Filter hparam " + (column_r32.displayName || column_r32.name));
        ɵɵadvance(5);
        ɵɵproperty("checked", column_r32.filter.includeUndefined);
        ɵɵadvance(3);
        ɵɵproperty("ngIf", column_r32.filter.type === ctx_r35.DomainType.INTERVAL);
        ɵɵadvance(1);
        ɵɵproperty("ngIf", column_r32.filter.type === ctx_r35.DomainType.DISCRETE);
    } }
    function RunsTableComponent_ng_container_21_th_1_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "th", 23);
        ɵɵelementStart(1, "span", 34);
        ɵɵtext(2);
        ɵɵelementEnd();
        ɵɵtemplate(3, RunsTableComponent_ng_container_21_th_1_ng_container_3_Template, 11, 5, "ng-container", 35);
        ɵɵelementEnd();
    } if (rf & 2) {
        const column_r32 = ɵɵnextContext().$implicit;
        ɵɵadvance(2);
        ɵɵtextInterpolate(column_r32.displayName || column_r32.name);
        ɵɵadvance(1);
        ɵɵproperty("ngIf", column_r32.filter);
    } }
    function RunsTableComponent_ng_container_21_td_2_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "td", 25);
        ɵɵtext(1);
        ɵɵelementEnd();
    } if (rf & 2) {
        const item_r58 = ctx.$implicit;
        const column_r32 = ɵɵnextContext().$implicit;
        ɵɵadvance(1);
        ɵɵtextInterpolate1(" ", item_r58.hparams.get(column_r32.name), " ");
    } }
    function RunsTableComponent_ng_container_21_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementContainerStart(0, 33);
        ɵɵtemplate(1, RunsTableComponent_ng_container_21_th_1_Template, 4, 2, "th", 8);
        ɵɵtemplate(2, RunsTableComponent_ng_container_21_td_2_Template, 2, 1, "td", 9);
        ɵɵelementContainerEnd();
    } if (rf & 2) {
        const column_r32 = ctx.$implicit;
        const ctx_r10 = ɵɵnextContext();
        ɵɵproperty("matColumnDef", ctx_r10.getHparamColumnId(column_r32));
    } }
    function RunsTableComponent_ng_container_22_th_1_ng_container_3_Template(rf, ctx) { if (rf & 1) {
        const _r66 = ɵɵgetCurrentView();
        ɵɵelementContainerStart(0);
        ɵɵelementStart(1, "button", 36);
        ɵɵelement(2, "mat-icon", 37);
        ɵɵelementEnd();
        ɵɵelementStart(3, "mat-menu", null, 38);
        ɵɵelementStart(5, "div", 39);
        ɵɵlistener("click", function RunsTableComponent_ng_container_22_th_1_ng_container_3_Template_div_click_5_listener($event) { ɵɵrestoreView(_r66); return $event.stopPropagation(); });
        ɵɵelementStart(6, "mat-checkbox", 26);
        ɵɵlistener("change", function RunsTableComponent_ng_container_22_th_1_ng_container_3_Template_mat_checkbox_change_6_listener() { ɵɵrestoreView(_r66); const column_r60 = ɵɵnextContext(2).$implicit; const ctx_r67 = ɵɵnextContext(); return ctx_r67.handleMetricIncludeUndefinedChanged(column_r60); });
        ɵɵelementStart(7, "span");
        ɵɵtext(8, "(show empty value)");
        ɵɵelementEnd();
        ɵɵelementEnd();
        ɵɵelementEnd();
        ɵɵelementStart(9, "div", 40);
        ɵɵlistener("click", function RunsTableComponent_ng_container_22_th_1_ng_container_3_Template_div_click_9_listener($event) { ɵɵrestoreView(_r66); return $event.stopPropagation(); });
        ɵɵelementStart(10, "tb-range-input", 41);
        ɵɵlistener("value", function RunsTableComponent_ng_container_22_th_1_ng_container_3_Template_tb_range_input_value_10_listener($event) { ɵɵrestoreView(_r66); const column_r60 = ɵɵnextContext(2).$implicit; const ctx_r70 = ɵɵnextContext(); return ctx_r70.handleMetricFilterChanged(column_r60, $event); });
        ɵɵelementEnd();
        ɵɵelementEnd();
        ɵɵelementEnd();
        ɵɵelementContainerEnd();
    } if (rf & 2) {
        const _r64 = ɵɵreference(4);
        const column_r60 = ɵɵnextContext(2).$implicit;
        ɵɵadvance(1);
        ɵɵproperty("matMenuTriggerFor", _r64);
        ɵɵattribute("aria-label", "Filter metric " + (column_r60.displayName || column_r60.tag));
        ɵɵadvance(5);
        ɵɵproperty("checked", column_r60.filter.includeUndefined);
        ɵɵadvance(4);
        ɵɵproperty("min", column_r60.filter.minValue)("max", column_r60.filter.maxValue)("lowerValue", column_r60.filter.filterLowerValue)("upperValue", column_r60.filter.filterUpperValue);
    } }
    function RunsTableComponent_ng_container_22_th_1_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "th", 23);
        ɵɵelementStart(1, "span", 34);
        ɵɵtext(2);
        ɵɵelementEnd();
        ɵɵtemplate(3, RunsTableComponent_ng_container_22_th_1_ng_container_3_Template, 11, 7, "ng-container", 35);
        ɵɵelementEnd();
    } if (rf & 2) {
        const column_r60 = ɵɵnextContext().$implicit;
        ɵɵadvance(2);
        ɵɵtextInterpolate(column_r60.displayName || column_r60.tag);
        ɵɵadvance(1);
        ɵɵproperty("ngIf", column_r60.filter);
    } }
    function RunsTableComponent_ng_container_22_td_2_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "td", 25);
        ɵɵtext(1);
        ɵɵelementEnd();
    } if (rf & 2) {
        const item_r74 = ctx.$implicit;
        const column_r60 = ɵɵnextContext().$implicit;
        ɵɵadvance(1);
        ɵɵtextInterpolate1(" ", item_r74.metrics.get(column_r60.tag), " ");
    } }
    function RunsTableComponent_ng_container_22_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementContainerStart(0, 33);
        ɵɵtemplate(1, RunsTableComponent_ng_container_22_th_1_Template, 4, 2, "th", 8);
        ɵɵtemplate(2, RunsTableComponent_ng_container_22_td_2_Template, 2, 1, "td", 9);
        ɵɵelementContainerEnd();
    } if (rf & 2) {
        const column_r60 = ctx.$implicit;
        const ctx_r11 = ɵɵnextContext();
        ɵɵproperty("matColumnDef", ctx_r11.getMetricColumnId(column_r60));
    } }
    function RunsTableComponent_tr_23_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "tr", 44);
    } }
    const _c2$7 = function (a1) { return { "select-all": true, "show-select-all": a1 }; };
    function RunsTableComponent_tr_24_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "tr", 45);
    } if (rf & 2) {
        const ctx_r13 = ɵɵnextContext();
        ɵɵproperty("ngClass", ɵɵpureFunction1(1, _c2$7, ctx_r13.allPageItemsSelected() && ctx_r13.numSelectedItems !== ctx_r13.allItemsLength));
    } }
    function RunsTableComponent_tr_25_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "tr", 46);
    } if (rf & 2) {
        const row_r76 = ctx.$implicit;
        ɵɵattribute("data-id", row_r76.run.id);
    } }
    function RunsTableComponent_div_27_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 47);
        ɵɵelement(1, "mat-spinner", 48);
        ɵɵelementEnd();
    } }
    function RunsTableComponent_div_28_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 49);
        ɵɵtext(1, "No Runs");
        ɵɵelementEnd();
    } }
    function RunsTableComponent_div_29_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 49);
        ɵɵelementStart(1, "span");
        ɵɵtext(2, "No runs match \"");
        ɵɵelementStart(3, "code");
        ɵɵtext(4);
        ɵɵelementEnd();
        ɵɵtext(5, "\"");
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r17 = ɵɵnextContext();
        ɵɵadvance(4);
        ɵɵtextInterpolate(ctx_r17.regexFilter);
    } }
    const _c3$3 = function () { return [5, 10, 20]; };
    function RunsTableComponent_mat_paginator_30_Template(rf, ctx) { if (rf & 1) {
        const _r78 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "mat-paginator", 50);
        ɵɵlistener("page", function RunsTableComponent_mat_paginator_30_Template_mat_paginator_page_0_listener($event) { ɵɵrestoreView(_r78); const ctx_r77 = ɵɵnextContext(); return ctx_r77.onPaginationChange.emit($event); });
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r18 = ɵɵnextContext();
        ɵɵproperty("pageSizeOptions", ɵɵpureFunction0(4, _c3$3))("pageIndex", ctx_r18.paginationOption.pageIndex)("pageSize", ctx_r18.paginationOption.pageSize)("length", ctx_r18.filteredItemsLength);
    } }
    const _c4$2 = function () { return ["select_all"]; };
    /**
     * Exported because Angular compiler requires decorated classes to be exported.
     */
    class RunsPaginatorIntl extends MatPaginatorIntl {
        constructor() {
            super(...arguments);
            this.itemsPerPageLabel = 'Show runs:';
        }
    }
    RunsPaginatorIntl.decorators = [
        { type: Injectable },
    ];
    /** @nocollapse */ RunsPaginatorIntl.ɵfac = function RunsPaginatorIntl_Factory(t) { return ɵRunsPaginatorIntl_BaseFactory(t || RunsPaginatorIntl); };
    /** @nocollapse */ RunsPaginatorIntl.ɵprov = ɵɵdefineInjectable({ token: RunsPaginatorIntl, factory: RunsPaginatorIntl.ɵfac });
    const ɵRunsPaginatorIntl_BaseFactory = ɵɵgetInheritedFactory(RunsPaginatorIntl);
    class RunsTableComponent {
        constructor() {
            this.dataSource = new MatTableDataSource();
            this.DomainType = DomainType;
            this.onRegexFilterChange = new EventEmitter();
            this.onSelectionToggle = new EventEmitter();
            this.onPageSelectionToggle = new EventEmitter();
            this.onSelectAllPages = new EventEmitter();
            this.onPaginationChange = new EventEmitter();
            this.onSortChange = new EventEmitter();
            this.onRunColorChange = new EventEmitter();
            this.onHparamDiscreteFilterChanged = new EventEmitter();
            this.onHparamIntervalFilterChanged = new EventEmitter();
            this.onMetricFilterChanged = new EventEmitter();
        }
        /**
         * @return {?}
         */
        ngOnChanges() {
            this.dataSource.data = this.pageItems;
        }
        /**
         * @param {?} spec
         * @return {?}
         */
        getHparamColumnId(spec) {
            return `h:${spec.name}`;
        }
        /**
         * @param {?} spec
         * @return {?}
         */
        getMetricColumnId(spec) {
            return `m:${spec.tag}`;
        }
        /**
         * @return {?}
         */
        getColumnIds() {
            return [
                ...this.columns,
                ...this.hparamColumns.map(this.getHparamColumnId),
                ...this.metricColumns.map(this.getMetricColumnId),
            ];
        }
        /**
         * Returns true when all items in the page are selected. Returns false when
         * there are no items at all.
         * @return {?}
         */
        allPageItemsSelected() {
            return (Boolean(this.pageItems.length) &&
                this.pageItems.every((/**
                 * @param {?} item
                 * @return {?}
                 */
                (item) => item.selected)));
        }
        /**
         * @return {?}
         */
        somePageItemsSelected() {
            return this.pageItems.some((/**
             * @param {?} item
             * @return {?}
             */
            (item) => item.selected));
        }
        /**
         * @return {?}
         */
        handlePageToggle() {
            this.onPageSelectionToggle.emit({ items: this.pageItems });
        }
        /**
         * @param {?} sort
         * @return {?}
         */
        handleSortChange(sort) {
            /** @type {?} */
            let direction;
            switch (sort.direction) {
                case 'asc':
                    direction = SortDirection.ASC;
                    break;
                case 'desc':
                    direction = SortDirection.DESC;
                    break;
                default:
                    direction = SortDirection.UNSET;
            }
            this.onSortChange.emit({
                column: (/** @type {?} */ (sort.active)),
                direction,
            });
        }
        /**
         * @param {?} event
         * @return {?}
         */
        onFilterKeyUp(event) {
            /** @type {?} */
            const input = (/** @type {?} */ ((/** @type {?} */ (event.target))));
            this.onRegexFilterChange.emit(input.value);
        }
        /**
         * @param {?} index
         * @param {?} item
         * @return {?}
         */
        tableTrackBy(index, item) {
            return item.run.id;
        }
        /**
         * @param {?} hparamColumn
         * @return {?}
         */
        handleHparamIncludeUndefinedToggled(hparamColumn) {
            const { name, filter } = hparamColumn;
            if (!filter) {
                throw new RangeError('Invariant error: require filter to exist for it to change');
            }
            if (filter.type === DomainType.DISCRETE) {
                this.onHparamDiscreteFilterChanged.emit({
                    hparamName: name,
                    includeUndefined: !filter.includeUndefined,
                    filterValues: filter.filterValues,
                });
            }
            else {
                this.onHparamIntervalFilterChanged.emit({
                    name,
                    includeUndefined: !filter.includeUndefined,
                    filterLowerValue: filter.filterLowerValue,
                    filterUpperValue: filter.filterUpperValue,
                });
            }
        }
        /**
         * @param {?} hparamColumn
         * @param {?} newValue
         * @return {?}
         */
        handleHparamIntervalChanged(hparamColumn, newValue) {
            const { name, filter } = hparamColumn;
            if (!filter) {
                throw new RangeError('Invariant error: require filter to exist for it to change');
            }
            this.onHparamIntervalFilterChanged.emit({
                name,
                includeUndefined: filter.includeUndefined,
                filterLowerValue: newValue.lowerValue,
                filterUpperValue: newValue.upperValue,
            });
        }
        /**
         * @param {?} hparamColumn
         * @param {?} toggledValue
         * @return {?}
         */
        handleHparamDiscreteChanged(hparamColumn, toggledValue) {
            const { name, filter } = hparamColumn;
            if (!filter) {
                throw new RangeError('Invariant error: require filter to exist for it to change');
            }
            if (filter.type !== DomainType.DISCRETE) {
                throw new RangeError(`Invariant error: expected discrete domain for ${name}`);
            }
            /** @type {?} */
            const newValues = new Set([...filter.filterValues]);
            if (newValues.has(toggledValue)) {
                newValues.delete(toggledValue);
            }
            else {
                newValues.add(toggledValue);
            }
            this.onHparamDiscreteFilterChanged.emit({
                hparamName: name,
                includeUndefined: filter.includeUndefined,
                filterValues: (/** @type {?} */ ([...newValues])),
            });
        }
        /**
         * @param {?} metricColumn
         * @return {?}
         */
        handleMetricIncludeUndefinedChanged(metricColumn) {
            if (!metricColumn.filter) {
                throw new RangeError('Invariant error: require filter to exist for it to change');
            }
            this.onMetricFilterChanged.emit({
                name: metricColumn.tag,
                includeUndefined: !metricColumn.filter.includeUndefined,
                filterLowerValue: metricColumn.filter.filterLowerValue,
                filterUpperValue: metricColumn.filter.filterUpperValue,
            });
        }
        /**
         * @param {?} metricColumn
         * @param {?} newValue
         * @return {?}
         */
        handleMetricFilterChanged(metricColumn, newValue) {
            if (!metricColumn.filter) {
                throw new RangeError('Invariant error: require filter to exist for it to change');
            }
            this.onMetricFilterChanged.emit({
                name: metricColumn.tag,
                includeUndefined: metricColumn.filter.includeUndefined,
                filterLowerValue: newValue.lowerValue,
                filterUpperValue: newValue.upperValue,
            });
        }
        /**
         * @param {?} hparamColumn
         * @return {?}
         */
        trackByHparamColumn(hparamColumn) {
            return hparamColumn.name;
        }
        /**
         * @param {?} metricColumn
         * @return {?}
         */
        trackByMetricColumn(metricColumn) {
            return metricColumn.tag;
        }
    }
    RunsTableComponent.decorators = [
        { type: Component, args: [{
                    selector: 'runs-table-component',
                    templateUrl: 'runs_table_component.ng.html',
                    host: {
                        '[class.flex-layout]': 'useFlexibleLayout',
                    },
                    styleUrls: ['runs_table_component.css'],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                    // Use Element Provider since this text is unique to this element hierarchy.
                    providers: [{ provide: MatPaginatorIntl, useClass: RunsPaginatorIntl }],
                },] },
    ];
    RunsTableComponent.propDecorators = {
        showExperimentName: [{ type: Input }],
        columns: [{ type: Input }],
        hparamColumns: [{ type: Input }],
        metricColumns: [{ type: Input }],
        allItemsLength: [{ type: Input }],
        filteredItemsLength: [{ type: Input }],
        useFlexibleLayout: [{ type: Input }],
        usePagination: [{ type: Input }],
        pageItems: [{ type: Input }],
        loading: [{ type: Input }],
        numSelectedItems: [{ type: Input }],
        sortOption: [{ type: Input }],
        paginationOption: [{ type: Input }],
        regexFilter: [{ type: Input }],
        onRegexFilterChange: [{ type: Output }],
        onSelectionToggle: [{ type: Output }],
        onPageSelectionToggle: [{ type: Output }],
        onSelectAllPages: [{ type: Output }],
        onPaginationChange: [{ type: Output }],
        onSortChange: [{ type: Output }],
        onRunColorChange: [{ type: Output }],
        onHparamDiscreteFilterChanged: [{ type: Output }],
        onHparamIntervalFilterChanged: [{ type: Output }],
        onMetricFilterChanged: [{ type: Output }],
        filter: [{ type: ViewChild, args: ['filter', { static: true, read: ElementRef },] }],
        paginator: [{ type: ViewChild, args: [MatPaginator, { static: true },] }],
        sort: [{ type: ViewChild, args: [MatSort, { static: true },] }]
    };
    /** @nocollapse */ RunsTableComponent.ɵfac = function RunsTableComponent_Factory(t) { return new (t || RunsTableComponent)(); };
    /** @nocollapse */ RunsTableComponent.ɵcmp = ɵɵdefineComponent({ type: RunsTableComponent, selectors: [["runs-table-component"]], viewQuery: function RunsTableComponent_Query(rf, ctx) { if (rf & 1) {
            ɵɵstaticViewQuery(_c0$q, true, ElementRef);
            ɵɵstaticViewQuery(MatPaginator, true);
            ɵɵstaticViewQuery(MatSort, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.filter = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.paginator = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.sort = _t.first);
        } }, hostVars: 2, hostBindings: function RunsTableComponent_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵclassProp("flex-layout", ctx.useFlexibleLayout);
        } }, inputs: { showExperimentName: "showExperimentName", columns: "columns", hparamColumns: "hparamColumns", metricColumns: "metricColumns", allItemsLength: "allItemsLength", filteredItemsLength: "filteredItemsLength", useFlexibleLayout: "useFlexibleLayout", usePagination: "usePagination", pageItems: "pageItems", loading: "loading", numSelectedItems: "numSelectedItems", sortOption: "sortOption", paginationOption: "paginationOption", regexFilter: "regexFilter" }, outputs: { onRegexFilterChange: "onRegexFilterChange", onSelectionToggle: "onSelectionToggle", onPageSelectionToggle: "onPageSelectionToggle", onSelectAllPages: "onSelectAllPages", onPaginationChange: "onPaginationChange", onSortChange: "onSortChange", onRunColorChange: "onRunColorChange", onHparamDiscreteFilterChanged: "onHparamDiscreteFilterChanged", onHparamIntervalFilterChanged: "onHparamIntervalFilterChanged", onMetricFilterChanged: "onMetricFilterChanged" }, features: [ɵɵProvidersFeature([{ provide: MatPaginatorIntl, useClass: RunsPaginatorIntl }]), ɵɵNgOnChangesFeature], decls: 31, vars: 16, consts: [[1, "filter-row"], [1, "run-filter"], ["svgIcon", "search_24px"], ["matInput", "", "autocomplete", "off", "placeholder", "Filter runs (regex)", 3, "keyup"], ["filter", ""], [1, "table-container"], ["mat-table", "", "matSort", "", 3, "dataSource", "matSortActive", "matSortDirection", "trackBy", "matSortChange"], ["matColumnDef", "checkbox"], ["mat-header-cell", "", 4, "matHeaderCellDef"], ["mat-cell", "", 4, "matCellDef"], ["matColumnDef", "experiment_name"], ["mat-header-cell", "", "mat-sort-header", "", "class", "name", 4, "matHeaderCellDef"], ["mat-cell", "", "class", "name", 4, "matCellDef"], ["matColumnDef", "run_name"], ["matColumnDef", "run_color"], ["matColumnDef", "select_all"], [3, "matColumnDef", 4, "ngFor", "ngForOf", "ngForTrackBy"], ["class", "columns", "mat-header-row", "", 4, "matHeaderRowDef"], ["mat-header-row", "", 3, "ngClass", 4, "matHeaderRowDef"], ["mat-row", "", 4, "matRowDef", "matRowDefColumns"], ["class", "loading", 4, "ngIf"], ["class", "no-runs", 4, "ngIf"], ["showFirstLastButtons", "", 3, "pageSizeOptions", "pageIndex", "pageSize", "length", "page", 4, "ngIf"], ["mat-header-cell", ""], [3, "checked", "indeterminate", "change"], ["mat-cell", ""], [3, "checked", "change"], ["mat-header-cell", "", "mat-sort-header", "", 1, "name"], ["mat-cell", "", 1, "name"], [3, "ngClass", "colorPicker", "cpDialogDisplay", "cpPosition", "cpPositionOffset", "cpUseRootViewContainer", "cpOutputFormat", "colorPickerChange"], ["class", "select-all-content", 4, "ngIf"], [1, "select-all-content"], ["mat-button", "", 3, "click"], [3, "matColumnDef"], [1, "name"], [4, "ngIf"], ["mat-icon-button", "", 3, "matMenuTriggerFor"], ["svgIcon", "filter_alt_24px"], ["filterMenu", "matMenu"], ["mat-menu-item", "", "role", "menuitemcheckbox", "disableRipple", "", 3, "click"], ["disableRipple", "", "mat-menu-item", "", 1, "range-input-container", 3, "click"], [3, "min", "max", "lowerValue", "upperValue", "value"], ["mat-menu-item", "", "role", "menuitemcheckbox", 3, "click", 4, "ngFor", "ngForOf"], ["mat-menu-item", "", "role", "menuitemcheckbox", 3, "click"], ["mat-header-row", "", 1, "columns"], ["mat-header-row", "", 3, "ngClass"], ["mat-row", ""], [1, "loading"], ["mode", "indeterminate", "diameter", "28"], [1, "no-runs"], ["showFirstLastButtons", "", 3, "pageSizeOptions", "pageIndex", "pageSize", "length", "page"]], template: function RunsTableComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "div", 0);
            ɵɵelementStart(1, "div", 1);
            ɵɵelement(2, "mat-icon", 2);
            ɵɵelementStart(3, "input", 3, 4);
            ɵɵlistener("keyup", function RunsTableComponent_Template_input_keyup_3_listener($event) { return ctx.onFilterKeyUp($event); });
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementStart(5, "div", 5);
            ɵɵelementStart(6, "table", 6);
            ɵɵlistener("matSortChange", function RunsTableComponent_Template_table_matSortChange_6_listener($event) { return ctx.handleSortChange($event); });
            ɵɵelementContainerStart(7, 7);
            ɵɵtemplate(8, RunsTableComponent_th_8_Template, 2, 2, "th", 8);
            ɵɵtemplate(9, RunsTableComponent_td_9_Template, 2, 1, "td", 9);
            ɵɵelementContainerEnd();
            ɵɵelementContainerStart(10, 10);
            ɵɵtemplate(11, RunsTableComponent_th_11_Template, 2, 0, "th", 11);
            ɵɵtemplate(12, RunsTableComponent_td_12_Template, 3, 2, "td", 12);
            ɵɵelementContainerEnd();
            ɵɵelementContainerStart(13, 13);
            ɵɵtemplate(14, RunsTableComponent_th_14_Template, 2, 0, "th", 11);
            ɵɵtemplate(15, RunsTableComponent_td_15_Template, 2, 1, "td", 12);
            ɵɵelementContainerEnd();
            ɵɵelementContainerStart(16, 14);
            ɵɵtemplate(17, RunsTableComponent_th_17_Template, 1, 0, "th", 8);
            ɵɵtemplate(18, RunsTableComponent_td_18_Template, 2, 11, "td", 9);
            ɵɵelementContainerEnd();
            ɵɵelementContainerStart(19, 15);
            ɵɵtemplate(20, RunsTableComponent_th_20_Template, 2, 2, "th", 8);
            ɵɵelementContainerEnd();
            ɵɵtemplate(21, RunsTableComponent_ng_container_21_Template, 3, 1, "ng-container", 16);
            ɵɵtemplate(22, RunsTableComponent_ng_container_22_Template, 3, 1, "ng-container", 16);
            ɵɵtemplate(23, RunsTableComponent_tr_23_Template, 1, 0, "tr", 17);
            ɵɵtemplate(24, RunsTableComponent_tr_24_Template, 1, 3, "tr", 18);
            ɵɵtemplate(25, RunsTableComponent_tr_25_Template, 1, 1, "tr", 19);
            ɵɵelement(26, "tr");
            ɵɵelementEnd();
            ɵɵtemplate(27, RunsTableComponent_div_27_Template, 2, 0, "div", 20);
            ɵɵtemplate(28, RunsTableComponent_div_28_Template, 2, 0, "div", 21);
            ɵɵtemplate(29, RunsTableComponent_div_29_Template, 6, 1, "div", 21);
            ɵɵelementEnd();
            ɵɵtemplate(30, RunsTableComponent_mat_paginator_30_Template, 1, 5, "mat-paginator", 22);
        } if (rf & 2) {
            ɵɵadvance(6);
            ɵɵproperty("dataSource", ctx.dataSource)("matSortActive", ctx.sortOption.column)("matSortDirection", ctx.sortOption.direction)("trackBy", ctx.tableTrackBy);
            ɵɵadvance(15);
            ɵɵproperty("ngForOf", ctx.hparamColumns)("ngForTrackBy", ctx.trackByHparamColumn);
            ɵɵadvance(1);
            ɵɵproperty("ngForOf", ctx.metricColumns)("ngForTrackBy", ctx.trackByMetricColumn);
            ɵɵadvance(1);
            ɵɵproperty("matHeaderRowDef", ctx.getColumnIds());
            ɵɵadvance(1);
            ɵɵproperty("matHeaderRowDef", ɵɵpureFunction0(15, _c4$2));
            ɵɵadvance(1);
            ɵɵproperty("matRowDefColumns", ctx.getColumnIds());
            ɵɵadvance(2);
            ɵɵproperty("ngIf", ctx.loading);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", !ctx.loading && ctx.allItemsLength === 0);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", !ctx.loading && ctx.allItemsLength > 0 && ctx.filteredItemsLength === 0);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx.usePagination);
        } }, directives: [MatIcon, MatInput, MatTable, MatSort, MatColumnDef, MatHeaderCellDef, MatCellDef, NgForOf, MatHeaderRowDef, MatRowDef, NgIf, MatHeaderCell, MatCheckbox, MatCell, MatSortHeader, NgClass, ColorPickerDirective, MatButton, MatMenuTrigger, _MatMenu, MatMenuItem, RangeInputComponent, MatHeaderRow, MatRow, MatSpinner, MatPaginator], styles: [".mat-badge-content[_ngcontent-%COMP%]{font-weight:600;font-size:12px;font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:9px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:24px}.mat-h1[_ngcontent-%COMP%], .mat-headline[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%]{font:400 24px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h2[_ngcontent-%COMP%], .mat-title[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h3[_ngcontent-%COMP%], .mat-subheading-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%]{font:400 16px/28px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h4[_ngcontent-%COMP%], .mat-subheading-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%]{font:400 15px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h5[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%]{font:400 calc(14px * 0.83)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-h6[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:400 calc(14px * 0.67)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-body-strong[_ngcontent-%COMP%], .mat-body-2[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   p[_ngcontent-%COMP%]{margin:0 0 12px}.mat-small[_ngcontent-%COMP%], .mat-caption[_ngcontent-%COMP%]{font:400 12px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-display-4[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-4[_ngcontent-%COMP%]{font:300 112px/112px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.05em;margin:0 0 56px}.mat-display-3[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-3[_ngcontent-%COMP%]{font:400 56px/56px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.02em;margin:0 0 64px}.mat-display-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-2[_ngcontent-%COMP%]{font:400 45px/48px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.005em;margin:0 0 64px}.mat-display-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-1[_ngcontent-%COMP%]{font:400 34px/40px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 64px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%], .mat-flat-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-button-toggle[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card-title[_ngcontent-%COMP%]{font-size:24px;font-weight:500}.mat-card-header[_ngcontent-%COMP%]   .mat-card-title[_ngcontent-%COMP%]{font-size:20px}.mat-card-subtitle[_ngcontent-%COMP%], .mat-card-content[_ngcontent-%COMP%]{font-size:14px}.mat-checkbox[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-checkbox-layout[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{line-height:24px}.mat-chip[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-chip[_ngcontent-%COMP%]   .mat-chip-trailing-icon.mat-icon[_ngcontent-%COMP%], .mat-chip[_ngcontent-%COMP%]   .mat-chip-remove.mat-icon[_ngcontent-%COMP%]{font-size:18px}.mat-table[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-header-cell[_ngcontent-%COMP%]{font-size:12px;font-weight:500}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{font-size:14px}.mat-calendar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-calendar-body[_ngcontent-%COMP%]{font-size:13px}.mat-calendar-body-label[_ngcontent-%COMP%], .mat-calendar-period-button[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-calendar-table-header[_ngcontent-%COMP%]   th[_ngcontent-%COMP%]{font-size:11px;font-weight:400}.mat-dialog-title[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-expansion-panel-header[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:15px;font-weight:400}.mat-expansion-panel-content[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field[_ngcontent-%COMP%]{font-size:inherit;font-weight:400;line-height:1.125;font-family:Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.34375em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{font-size:150%;line-height:1.125}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]{height:1.5em;width:1.5em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{height:1.125em;line-height:1.125}.mat-form-field-infix[_ngcontent-%COMP%]{padding:.5em 0;border-top:.84375em solid transparent}.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34375em) scale(0.75);width:133.3333333333%}.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34374em) scale(0.75);width:133.3333433333%}.mat-form-field-label-wrapper[_ngcontent-%COMP%]{top:-0.84375em;padding-top:.84375em}.mat-form-field-label[_ngcontent-%COMP%]{top:1.34375em}.mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.34375em}.mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{font-size:75%;margin-top:.6666666667em;top:calc(100% - 1.7916666667em)}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.4375em 0}.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.001px);-ms-transform:translateY(-1.28125em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00101px);-ms-transform:translateY(-1.28124em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00102px);-ms-transform:translateY(-1.28123em) scale(0.75);width:133.3333533333%}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.28125em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{margin-top:.5416666667em;top:calc(100% - 1.6666666667em)}@media print{.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28122em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28121em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.2812em) scale(0.75)}}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.25em 0 .75em 0}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.09375em;margin-top:-0.5em}.mat-form-field-appearance-fill.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59374em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:1em 0 1em 0}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.84375em;margin-top:-0.25em}.mat-form-field-appearance-outline.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59374em) scale(0.75);width:133.3333433333%}.mat-grid-tile-header[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]{font-size:14px}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2), .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}input.mat-input-element[_ngcontent-%COMP%]{margin-top:-0.0625em}.mat-menu-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:400}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px}.mat-radio-button[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select-trigger[_ngcontent-%COMP%]{height:1.125em}.mat-slide-toggle-content[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-slider-thumb-label-text[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-stepper-vertical[_ngcontent-%COMP%], .mat-stepper-horizontal[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-step-label[_ngcontent-%COMP%]{font-size:14px;font-weight:400}.mat-step-sub-label-error[_ngcontent-%COMP%]{font-weight:normal}.mat-step-label-error[_ngcontent-%COMP%]{font-size:14px}.mat-step-label-selected[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-tab-group[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-toolbar[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0}.mat-tooltip[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:10px;padding-top:6px;padding-bottom:6px}.mat-tooltip-handset[_ngcontent-%COMP%]{font-size:14px;padding-top:8px;padding-bottom:8px}.mat-list-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:16px}.mat-optgroup-label[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-simple-snackbar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px}.mat-simple-snackbar-action[_ngcontent-%COMP%]{line-height:1;font-family:inherit;font-size:inherit;font-weight:500}.mat-tree[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{font-weight:400;font-size:14px}.mat-ripple[_ngcontent-%COMP%]{overflow:hidden;position:relative}.mat-ripple[_ngcontent-%COMP%]:not(:empty){transform:translateZ(0)}.mat-ripple.mat-ripple-unbounded[_ngcontent-%COMP%]{overflow:visible}.mat-ripple-element[_ngcontent-%COMP%]{position:absolute;border-radius:50%;pointer-events:none;transition:opacity,transform 0ms cubic-bezier(0, 0, 0.2, 1);transform:scale(0)}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{display:none}.cdk-visually-hidden[_ngcontent-%COMP%]{border:0;clip:rect(0 0 0 0);height:1px;margin:-1px;overflow:hidden;padding:0;position:absolute;width:1px;outline:0;-webkit-appearance:none;-moz-appearance:none}.cdk-overlay-container[_ngcontent-%COMP%], .cdk-global-overlay-wrapper[_ngcontent-%COMP%]{pointer-events:none;top:0;left:0;height:100%;width:100%}.cdk-overlay-container[_ngcontent-%COMP%]{position:fixed;z-index:1000}.cdk-overlay-container[_ngcontent-%COMP%]:empty{display:none}.cdk-global-overlay-wrapper[_ngcontent-%COMP%]{display:flex;position:absolute;z-index:1000}.cdk-overlay-pane[_ngcontent-%COMP%]{position:absolute;pointer-events:auto;box-sizing:border-box;z-index:1000;display:flex;max-width:100%;max-height:100%}.cdk-overlay-backdrop[_ngcontent-%COMP%]{position:absolute;top:0;bottom:0;left:0;right:0;z-index:1000;pointer-events:auto;-webkit-tap-highlight-color:transparent;transition:opacity 400ms cubic-bezier(0.25, 0.8, 0.25, 1);opacity:0}.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:1}@media screen and (-ms-high-contrast: active){.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:.6}}.cdk-overlay-dark-backdrop[_ngcontent-%COMP%]{background:rgba(0,0,0,.32)}.cdk-overlay-transparent-backdrop[_ngcontent-%COMP%], .cdk-overlay-transparent-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:0}.cdk-overlay-connected-position-bounding-box[_ngcontent-%COMP%]{position:absolute;z-index:1000;display:flex;flex-direction:column;min-width:1px;min-height:1px}.cdk-global-scrollblock[_ngcontent-%COMP%]{position:fixed;width:100%;overflow-y:scroll}@keyframes cdk-text-field-autofill-start{}@keyframes cdk-text-field-autofill-end{}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:-webkit-autofill{animation:cdk-text-field-autofill-start 0s 1ms}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:not(:-webkit-autofill){animation:cdk-text-field-autofill-end 0s 1ms}textarea.cdk-textarea-autosize[_ngcontent-%COMP%]{resize:none}textarea.cdk-textarea-autosize-measuring[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:auto !important;overflow:hidden !important}textarea.cdk-textarea-autosize-measuring-firefox[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:0 !important}.mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.1)}.mat-option[_ngcontent-%COMP%]{color:#212121}.mat-option[_ngcontent-%COMP%]:hover:not(.mat-option-disabled), .mat-option[_ngcontent-%COMP%]:focus:not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple):not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-active[_ngcontent-%COMP%]{background:rgba(0,0,0,.04);color:#212121}.mat-option.mat-option-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-primary[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f57c00}.mat-accent[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f44336}.mat-optgroup-label[_ngcontent-%COMP%]{color:#616161}.mat-optgroup-disabled[_ngcontent-%COMP%]   .mat-optgroup-label[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-pseudo-checkbox[_ngcontent-%COMP%]{color:#616161}.mat-pseudo-checkbox[_ngcontent-%COMP%]::after{color:#fafafa}.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{color:#b0b0b0}.mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f57c00}.mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f44336}.mat-pseudo-checkbox-checked.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{background:#b0b0b0}.mat-elevation-z0[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z1[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z2[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z3[_ngcontent-%COMP%]{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z4[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z5[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 5px 8px 0px rgba(0, 0, 0, 0.14),0px 1px 14px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z6[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z7[_ngcontent-%COMP%]{box-shadow:0px 4px 5px -2px rgba(0, 0, 0, 0.2),0px 7px 10px 1px rgba(0, 0, 0, 0.14),0px 2px 16px 1px rgba(0, 0, 0, 0.12)}.mat-elevation-z8[_ngcontent-%COMP%]{box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z9[_ngcontent-%COMP%]{box-shadow:0px 5px 6px -3px rgba(0, 0, 0, 0.2),0px 9px 12px 1px rgba(0, 0, 0, 0.14),0px 3px 16px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z10[_ngcontent-%COMP%]{box-shadow:0px 6px 6px -3px rgba(0, 0, 0, 0.2),0px 10px 14px 1px rgba(0, 0, 0, 0.14),0px 4px 18px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z11[_ngcontent-%COMP%]{box-shadow:0px 6px 7px -4px rgba(0, 0, 0, 0.2),0px 11px 15px 1px rgba(0, 0, 0, 0.14),0px 4px 20px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z12[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z13[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 13px 19px 2px rgba(0, 0, 0, 0.14),0px 5px 24px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z14[_ngcontent-%COMP%]{box-shadow:0px 7px 9px -4px rgba(0, 0, 0, 0.2),0px 14px 21px 2px rgba(0, 0, 0, 0.14),0px 5px 26px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z15[_ngcontent-%COMP%]{box-shadow:0px 8px 9px -5px rgba(0, 0, 0, 0.2),0px 15px 22px 2px rgba(0, 0, 0, 0.14),0px 6px 28px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z16[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z17[_ngcontent-%COMP%]{box-shadow:0px 8px 11px -5px rgba(0, 0, 0, 0.2),0px 17px 26px 2px rgba(0, 0, 0, 0.14),0px 6px 32px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z18[_ngcontent-%COMP%]{box-shadow:0px 9px 11px -5px rgba(0, 0, 0, 0.2),0px 18px 28px 2px rgba(0, 0, 0, 0.14),0px 7px 34px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z19[_ngcontent-%COMP%]{box-shadow:0px 9px 12px -6px rgba(0, 0, 0, 0.2),0px 19px 29px 2px rgba(0, 0, 0, 0.14),0px 7px 36px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z20[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 20px 31px 3px rgba(0, 0, 0, 0.14),0px 8px 38px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z21[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 21px 33px 3px rgba(0, 0, 0, 0.14),0px 8px 40px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z22[_ngcontent-%COMP%]{box-shadow:0px 10px 14px -6px rgba(0, 0, 0, 0.2),0px 22px 35px 3px rgba(0, 0, 0, 0.14),0px 8px 42px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z23[_ngcontent-%COMP%]{box-shadow:0px 11px 14px -7px rgba(0, 0, 0, 0.2),0px 23px 36px 3px rgba(0, 0, 0, 0.14),0px 9px 44px 8px rgba(0, 0, 0, 0.12)}.mat-elevation-z24[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12)}.mat-app-background[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-theme-loaded-marker[_ngcontent-%COMP%]{display:none}.mat-autocomplete-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-autocomplete-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover){background:#fff}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover):not(.mat-option-disabled){color:#212121}.mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f57c00}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{outline:solid 1px;border-radius:0}.mat-badge-accent[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-badge-warn[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f44336}.mat-badge[_ngcontent-%COMP%]{position:relative}.mat-badge-hidden[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{display:none}.mat-badge-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#b9b9b9;color:#757575}.mat-badge-content[_ngcontent-%COMP%]{position:absolute;text-align:center;display:inline-block;border-radius:50%;transition:transform 200ms ease-in-out;transform:scale(0.6);overflow:hidden;white-space:nowrap;text-overflow:ellipsis;pointer-events:none}.ng-animate-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%], .mat-badge-content._mat-animation-noopable[_ngcontent-%COMP%]{transition:none}.mat-badge-content.mat-badge-active[_ngcontent-%COMP%]{transform:none}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:16px;height:16px;line-height:16px}.mat-badge-small.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-8px}.mat-badge-small.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-8px}.mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-16px}.mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-16px}.mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-8px}.mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-8px}.mat-badge-medium[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:22px;height:22px;line-height:22px}.mat-badge-medium.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-11px}.mat-badge-medium.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-11px}.mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-22px}.mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-22px}.mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-11px}.mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-11px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:28px;height:28px;line-height:28px}.mat-badge-large.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-14px}.mat-badge-large.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-14px}.mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-28px}.mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-28px}.mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-14px}.mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-14px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]{color:inherit;background:transparent}.mat-button.mat-primary[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-button[disabled][disabled][_ngcontent-%COMP%], .mat-icon-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-icon-button[disabled][disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-stroked-button[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f44336}.mat-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:transparent}.mat-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{opacity:.1;background-color:currentColor}.mat-button-focus-overlay[_ngcontent-%COMP%]{background:#000}.mat-stroked-button[_ngcontent-%COMP%]:not([disabled]){border-color:rgba(0,0,0,.12)}.mat-flat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{color:#212121;background-color:#fff}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-flat-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-stroked-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-flat-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-raised-button[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone[_ngcontent-%COMP%], .mat-button-toggle-group[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{box-shadow:none}.mat-button-toggle[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-button-toggle[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121;background:#fff}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:#000}.mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12)}[dir=rtl][_ngcontent-%COMP%]   .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-group-appearance-standard.mat-button-toggle-vertical[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:none;border-top:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#616161}.mat-button-toggle-checked.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121}.mat-button-toggle-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.26);background-color:#eee}.mat-button-toggle-disabled.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{background:#fff}.mat-button-toggle-disabled.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#bdbdbd}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{border:solid 1px rgba(0,0,0,.12)}.mat-card[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-card[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-card.mat-card-flat[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-card-subtitle[_ngcontent-%COMP%]{color:#616161}.mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#616161}.mat-checkbox-checkmark[_ngcontent-%COMP%]{fill:#fafafa}.mat-checkbox-checkmark-path[_ngcontent-%COMP%]{stroke:#fafafa !important}.mat-checkbox-mixedmark[_ngcontent-%COMP%]{background-color:#fafafa}.mat-checkbox-indeterminate.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f57c00}.mat-checkbox-indeterminate.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#ff9800}.mat-checkbox-indeterminate.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f44336}.mat-checkbox-disabled.mat-checkbox-checked[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-disabled.mat-checkbox-indeterminate[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]:not(.mat-checkbox-checked)   .mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{color:#616161}.mat-checkbox[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%]{background:#f57c00}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%]{background:#ff9800}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%]{background:#f44336}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#212121}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#212121;opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled):active{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled)   .mat-chip-remove[_ngcontent-%COMP%]:hover{opacity:.54}.mat-chip.mat-standard-chip.mat-chip-disabled[_ngcontent-%COMP%]{opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]::after{background:#000}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-table[_ngcontent-%COMP%]{background:#fff}.mat-table[_ngcontent-%COMP%]   thead[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tbody[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tfoot[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], [mat-header-row][_ngcontent-%COMP%], [mat-row][_ngcontent-%COMP%], [mat-footer-row][_ngcontent-%COMP%], .mat-table-sticky[_ngcontent-%COMP%]{background:inherit}mat-row[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], th.mat-header-cell[_ngcontent-%COMP%], td.mat-cell[_ngcontent-%COMP%], td.mat-footer-cell[_ngcontent-%COMP%]{border-bottom-color:rgba(0,0,0,.12)}.mat-header-cell[_ngcontent-%COMP%]{color:#616161}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{color:#212121}.mat-calendar-arrow[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.54)}.mat-datepicker-toggle[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-next-button[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-previous-button[_ngcontent-%COMP%]{color:rgba(0,0,0,.54)}.mat-calendar-table-header[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-calendar-table-header-divider[_ngcontent-%COMP%]::after{background:rgba(0,0,0,.12)}.mat-calendar-body-label[_ngcontent-%COMP%]{color:#616161}.mat-calendar-body-cell-content[_ngcontent-%COMP%]{color:#212121;border-color:transparent}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){color:#757575}.mat-calendar-body-cell[_ngcontent-%COMP%]:not(.mat-calendar-body-disabled):hover > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-keyboard-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-program-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){background-color:rgba(0,0,0,.04)}.mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.38)}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.18)}.mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.4)}.mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12);background-color:#fff;color:#212121}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.4)}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.4)}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content-touch[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-datepicker-toggle-active[_ngcontent-%COMP%]{color:#f57c00}.mat-datepicker-toggle-active.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-datepicker-toggle-active.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-dialog-container[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-divider[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-divider-vertical[_ngcontent-%COMP%]{border-right-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-expansion-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-action-row[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-keyboard-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-program-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded)   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover:not([aria-disabled=true]){background:rgba(0,0,0,.04)}@media(hover: none){.mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded):not([aria-disabled=true])   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover{background:#fff}}.mat-expansion-panel-header-title[_ngcontent-%COMP%]{color:#212121}.mat-expansion-panel-header-description[_ngcontent-%COMP%], .mat-expansion-indicator[_ngcontent-%COMP%]::after{color:#616161}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-title[_ngcontent-%COMP%], .mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-description[_ngcontent-%COMP%]{color:inherit}.mat-form-field-label[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-hint[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-focused[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-ripple[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid)   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f57c00}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-accent   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#ff9800}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-warn   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#f44336}.mat-error[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-hint[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-legacy.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-standard[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-standard.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.04)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.02)}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:transparent}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.12)}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:rgba(0,0,0,.87)}.mat-form-field-appearance-outline.mat-focused[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field-appearance-outline.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-appearance-outline.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-invalid.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.06)}.mat-icon.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-icon.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-icon.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-type-mat-native-select[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#616161}.mat-input-element[_ngcontent-%COMP%]:disabled, .mat-form-field-type-mat-native-select.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#757575}.mat-input-element[_ngcontent-%COMP%]{caret-color:#f57c00}.mat-input-element[_ngcontent-%COMP%]::placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-moz-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-webkit-input-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]:-ms-input-placeholder{color:rgba(97,97,97,.42)}.mat-accent[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%], .mat-form-field-invalid[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#f44336}.mat-form-field-type-mat-native-select.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{color:#616161}.mat-list-item-disabled[_ngcontent-%COMP%]{background-color:#eee}.mat-list-option[_ngcontent-%COMP%]:hover, .mat-list-option[_ngcontent-%COMP%]:focus, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.04)}.mat-list-single-selected-option[_ngcontent-%COMP%], .mat-list-single-selected-option[_ngcontent-%COMP%]:hover, .mat-list-single-selected-option[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.12)}.mat-menu-panel[_ngcontent-%COMP%]{background:#fff}.mat-menu-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-menu-item[_ngcontent-%COMP%]{background:transparent;color:#212121}.mat-menu-item[disabled][_ngcontent-%COMP%], .mat-menu-item[disabled][_ngcontent-%COMP%]::after{color:rgba(0,0,0,.38)}.mat-menu-item[_ngcontent-%COMP%]   .mat-icon-no-color[_ngcontent-%COMP%], .mat-menu-item-submenu-trigger[_ngcontent-%COMP%]::after{color:rgba(0,0,0,.54)}.mat-menu-item[_ngcontent-%COMP%]:hover:not([disabled]), .mat-menu-item.cdk-program-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item.cdk-keyboard-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item-highlighted[_ngcontent-%COMP%]:not([disabled]){background:rgba(0,0,0,.04)}.mat-paginator[_ngcontent-%COMP%]{background:#fff}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{color:#616161}.mat-paginator-decrement[_ngcontent-%COMP%], .mat-paginator-increment[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54);border-right:2px solid rgba(0,0,0,.54)}.mat-paginator-first[_ngcontent-%COMP%], .mat-paginator-last[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54)}.mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-decrement[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-increment[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-first[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-last[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f57c00}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#ff9800}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f44336}.mat-progress-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f57c00}.mat-progress-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#ff9800}.mat-progress-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f44336}.mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#616161}.mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f57c00}.mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#ff9800}.mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#ff9800}.mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f44336}.mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f44336}.mat-radio-button.mat-radio-disabled.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-label-content[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-radio-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-select-value[_ngcontent-%COMP%]{color:#212121}.mat-select-placeholder[_ngcontent-%COMP%]{color:rgba(97,97,97,.42)}.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%]{color:#757575}.mat-select-arrow[_ngcontent-%COMP%]{color:#616161}.mat-select-panel[_ngcontent-%COMP%]{background:#fff}.mat-select-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-select-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple){background:rgba(0,0,0,.12)}.mat-form-field.mat-focused.mat-primary[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-invalid[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#757575}.mat-drawer-container[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-drawer[_ngcontent-%COMP%]{background-color:#fff;color:#212121}.mat-drawer.mat-drawer-push[_ngcontent-%COMP%]{background-color:#fff}.mat-drawer[_ngcontent-%COMP%]:not(.mat-drawer-side){box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-drawer-side[_ngcontent-%COMP%]{border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-backdrop.mat-drawer-shown[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.6)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.54)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.54)}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.54)}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle[_ngcontent-%COMP%]:not(.mat-checked)   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-slide-toggle-thumb[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12);background-color:#fafafa}.mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-primary[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f57c00}.mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-accent[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#ff9800}.mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-warn[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f44336}.mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.2)}.mat-slider[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%], .cdk-focused[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-disabled[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]   .mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing)   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26);background-color:transparent}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26)}.mat-slider-has-ticks[_ngcontent-%COMP%]   .mat-slider-wrapper[_ngcontent-%COMP%]::after{border-color:rgba(0,0,0,.7)}.mat-slider-horizontal[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to right, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent);background-image:-moz-repeating-linear-gradient(0.0001deg, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-slider-vertical[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to bottom, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-step-header.cdk-keyboard-focused[_ngcontent-%COMP%], .mat-step-header.cdk-program-focused[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]:hover{background-color:rgba(0,0,0,.04)}@media(hover: none){.mat-step-header[_ngcontent-%COMP%]:hover{background:none}}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-optional[_ngcontent-%COMP%]{color:#616161}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon[_ngcontent-%COMP%]{background-color:#616161;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-selected[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-done[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-edit[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-error[_ngcontent-%COMP%]{background-color:transparent;color:#f44336}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-active[_ngcontent-%COMP%]{color:#212121}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-error[_ngcontent-%COMP%]{color:#f44336}.mat-stepper-horizontal[_ngcontent-%COMP%], .mat-stepper-vertical[_ngcontent-%COMP%]{background-color:#fff}.mat-stepper-vertical-line[_ngcontent-%COMP%]::before{border-left-color:rgba(0,0,0,.12)}.mat-horizontal-stepper-header[_ngcontent-%COMP%]::before, .mat-horizontal-stepper-header[_ngcontent-%COMP%]::after, .mat-stepper-horizontal-line[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-sort-header-arrow[_ngcontent-%COMP%]{color:#616161}.mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-header[_ngcontent-%COMP%]{border-bottom:1px solid rgba(0,0,0,.12)}.mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%]{border-top:1px solid rgba(0,0,0,.12);border-bottom:none}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{color:#212121}.mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:#757575}.mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#212121}.mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#757575}.mat-tab-group[class*=mat-background-][_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar[class*=mat-background-][_ngcontent-%COMP%]{border-bottom:none;border-top:none}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-toolbar[_ngcontent-%COMP%]{background:#f5f5f5;color:#212121}.mat-toolbar.mat-primary[_ngcontent-%COMP%]{background:#f57c00;color:#fff}.mat-toolbar.mat-accent[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-toolbar.mat-warn[_ngcontent-%COMP%]{background:#f44336;color:#fff}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:currentColor}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:inherit}.mat-toolbar[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:currentColor}.mat-tooltip[_ngcontent-%COMP%]{background:rgba(97,97,97,.9)}.mat-tree[_ngcontent-%COMP%]{background:#fff}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{color:#212121}.mat-snack-bar-container[_ngcontent-%COMP%]{color:rgba(255,255,255,.7);background:#323232;box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-simple-snackbar-action[_ngcontent-%COMP%]{color:#ff9800}[_nghost-%COMP%]{display:block;overflow:hidden;background-color:#fff}.table-container[_ngcontent-%COMP%]{contain:layout paint;max-width:100%;overflow-x:auto;will-change:transform}.flex-layout[_nghost-%COMP%]{display:flex;flex-direction:column}.flex-layout[_nghost-%COMP%]   .table-container[_ngcontent-%COMP%]{flex:1 1 0;overflow-x:hidden;overflow-y:auto}.flex-layout[_nghost-%COMP%]   .table-container[_ngcontent-%COMP%]   td.name[_ngcontent-%COMP%]{word-break:break-word;overflow-wrap:break-word}.flex-layout[_nghost-%COMP%]   mat-paginator[_ngcontent-%COMP%]{border-top:1px solid #ebebeb;padding-bottom:12px}table[_ngcontent-%COMP%]{width:100%}table[_ngcontent-%COMP%]   th[_ngcontent-%COMP%], table[_ngcontent-%COMP%]   td[_ngcontent-%COMP%]{font-size:13px;padding:5px}table[_ngcontent-%COMP%]   th[_ngcontent-%COMP%]:not(.name), table[_ngcontent-%COMP%]   td[_ngcontent-%COMP%]:not(.name){text-align:center}table[_ngcontent-%COMP%]   th[_ngcontent-%COMP%]:first-of-type, table[_ngcontent-%COMP%]   td[_ngcontent-%COMP%]:first-of-type{padding-left:24px}table[_ngcontent-%COMP%]   th[_ngcontent-%COMP%]:last-of-type, table[_ngcontent-%COMP%]   td[_ngcontent-%COMP%]:last-of-type{padding-right:24px}table[_ngcontent-%COMP%]   th[_ngcontent-%COMP%]{color:#212121}table[_ngcontent-%COMP%]   tr[_ngcontent-%COMP%]:not(.select-all)   th[_ngcontent-%COMP%]{white-space:nowrap}table[_ngcontent-%COMP%]   .mat-header-row[_ngcontent-%COMP%], table[_ngcontent-%COMP%]   .mat-row[_ngcontent-%COMP%]{height:43px}.loading[_ngcontent-%COMP%], .no-runs[_ngcontent-%COMP%]{align-items:center;border:0;border-bottom-width:1px;border-color:#0000001f;border-style:solid;display:flex;height:48px;padding:0 24px}.loading[_ngcontent-%COMP%]{justify-content:center}.select-all[_ngcontent-%COMP%]   th[_ngcontent-%COMP%]{padding-bottom:12px;padding-top:12px}[_nghost-%COMP%]   tr.select-all[_ngcontent-%COMP%]:not(.show-select-all){display:none}.select-all-content[_ngcontent-%COMP%], .select-all-content[_ngcontent-%COMP%]   button[_ngcontent-%COMP%]{font-size:13px;font-weight:400;line-height:1.6;text-align:left}.select-all-content[_ngcontent-%COMP%]   button[_ngcontent-%COMP%]{font-weight:500;padding:0 4px}.filter-row[_ngcontent-%COMP%]{display:flex;align-items:center;height:48px;padding:0 16px;border-bottom:1px solid #ebebeb}.run-filter[_ngcontent-%COMP%]{display:flex;color:#212121;font-size:13px;flex:1}.run-filter[_ngcontent-%COMP%]   mat-icon[_ngcontent-%COMP%]{color:#616161;flex:none;margin-left:5px;margin-right:5px}.mat-column-checkbox[_ngcontent-%COMP%], .mat-column-run_color[_ngcontent-%COMP%]{width:20px}.run-color-swatch[_ngcontent-%COMP%]{border-radius:100%;border:1px solid #ebebeb;height:20px;width:20px;outline:none}.run-color-swatch.no-color[_ngcontent-%COMP%]{border-color:#c6cad1;border-width:2px}.range-input-container[_ngcontent-%COMP%]{height:auto}[_nghost-%COMP%]     mat-paginator mat-form-field{margin:0}"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/runs/views/runs_table/types.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @enum {string} */
    const RunsTableColumn = {
        CHECKBOX: "checkbox",
        RUN_NAME: "run_name",
        EXPERIMENT_NAME: "experiment_name",
        RUN_COLOR: "run_color",
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/runs/views/runs_table/runs_table_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const getRunsLoading = createSelector(getRunsLoadState, (/**
     * @param {?} loadState
     * @return {?}
     */
    (loadState) => loadState.state === DataLoadState.LOADING));
    /**
     * @param {?} item
     * @param {?} sortColumn
     * @return {?}
     */
    function getPropsForSort(item, sortColumn) {
        switch (sortColumn) {
            case RunsTableColumn.EXPERIMENT_NAME:
                return [item.experimentAlias, item.run.name, item.run.id];
            case RunsTableColumn.RUN_NAME:
                return [item.run.name, item.experimentAlias, item.run.id];
            default:
                throw new Error(`Not yet implemented: ${sortColumn}`);
        }
    }
    /**
     * @param {?} items
     * @param {?} sort
     * @return {?}
     */
    function sortRunTableItems(items, sort) {
        /** @type {?} */
        const sortedItems = [...items];
        if (!sort.column || sort.direction === SortDirection.UNSET) {
            return sortedItems;
        }
        sortedItems.sort((/**
         * @param {?} a
         * @param {?} b
         * @return {?}
         */
        (a, b) => {
            /** @type {?} */
            const aProps = getPropsForSort(a, (/** @type {?} */ (sort.column)));
            /** @type {?} */
            const bProps = getPropsForSort(b, (/** @type {?} */ (sort.column)));
            if (aProps.length !== bProps.length) {
                throw new Error('Invariant error: a given sort should result in same number of ' +
                    `items: ${sort}`);
            }
            for (let index = 0; index < aProps.length; index++) {
                /** @type {?} */
                const valA = aProps[index];
                /** @type {?} */
                const valB = bProps[index];
                if (valA === valB) {
                    continue;
                }
                if (typeof valA !== typeof valB) {
                    throw new Error(`Cannot compare values of different types: ` +
                        `${typeof valA} vs. ${typeof valB}`);
                }
                return valA < valB === (sort.direction === SortDirection.ASC) ? -1 : 1;
            }
            return 0;
        }));
        return sortedItems;
    }
    /**
     * @param {?} filter
     * @param {?} value
     * @return {?}
     */
    function matchFilter(filter, value) {
        if (value === undefined) {
            return filter.includeUndefined;
        }
        if (filter.type === DomainType.DISCRETE) {
            // (upcast to work around bad TypeScript libdefs)
            /** @type {?} */
            const values = filter.filterValues;
            return values.includes(value);
        }
        else if (filter.type === DomainType.INTERVAL) {
            return filter.filterLowerValue <= value && value <= filter.filterUpperValue;
        }
        return false;
    }
    /**
     * Renders list of experiments.
     *
     * Note: all \@Inputs are read once upon initialization. This component does not
     * update when input bindings change.
     */
    class RunsTableContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.loading$ = null;
            this.hparamColumns$ = of([]);
            this.metricColumns$ = of([]);
            /**
             * Enables a layout mode intended for scenarios when changing the # of runs
             * should have no effect on the table's size.
             *
             * - height and width span the container height and width
             * - run list scrolls vertically, not horizontally
             * - 'name' cells wrap text
             */
            this.useFlexibleLayout = false;
            /**
             * Whether to use pagination options from the store. If false, the table will
             * show a single page with all runs.
             */
            this.usePagination = false;
            // Column to disable in the table. The columns are rendered in the order as
            // defined by this input.
            this.columns = [RunsTableColumn.RUN_NAME];
            this.showHparamsAndMetrics = false;
            this.sortOption$ = this.store.select(getRunSelectorSort);
            this.paginationOption$ = this.store.select(getRunSelectorPaginationOption);
            this.regexFilter$ = this.store.select(getRunSelectorRegexFilter);
        }
        /**
         * @return {?}
         */
        isExperimentNameVisible() {
            return this.columns.some((/**
             * @param {?} column
             * @return {?}
             */
            (column) => {
                return column === RunsTableColumn.EXPERIMENT_NAME;
            }));
        }
        /**
         * @return {?}
         */
        ngOnInit() {
            /** @type {?} */
            const getRunTableItemsPerExperiment = this.experimentIds.map((/**
             * @param {?} id
             * @return {?}
             */
            (id) => this.getRunTableItemsForExperiment(id)));
            this.allUnsortedRunTableItems$ = combineLatest(getRunTableItemsPerExperiment).pipe(map((/**
             * @param {?} itemsForExperiments
             * @return {?}
             */
            (itemsForExperiments) => {
                /** @type {?} */
                const items = (/** @type {?} */ ([]));
                return items.concat(...itemsForExperiments);
            })), shareReplay(1));
            this.allItemsLength$ = this.allUnsortedRunTableItems$.pipe(map((/**
             * @param {?} items
             * @return {?}
             */
            (items) => items.length)));
            /** @type {?} */
            const getFilteredItems$ = this.getFilteredItems$(this.allUnsortedRunTableItems$).pipe(shareReplay(1));
            this.filteredItemsLength$ = getFilteredItems$.pipe(map((/**
             * @param {?} items
             * @return {?}
             */
            (items) => items.length)));
            this.pageItems$ = this.sortedAndSlicedItems$(getFilteredItems$);
            this.numSelectedItems$ = this.allUnsortedRunTableItems$.pipe(map((/**
             * @param {?} items
             * @return {?}
             */
            (items) => {
                return items.reduce((/**
                 * @param {?} count
                 * @param {?} item
                 * @return {?}
                 */
                (count, item) => {
                    return count + Number(item.selected);
                }), 0);
            })));
            /** @type {?} */
            const getRunsLoadingPerExperiment = this.experimentIds.map((/**
             * @param {?} id
             * @return {?}
             */
            (id) => {
                return this.store.select(getRunsLoading, { experimentId: id });
            }));
            this.loading$ = combineLatest(getRunsLoadingPerExperiment).pipe(map((/**
             * @param {?} experimentsLoading
             * @return {?}
             */
            (experimentsLoading) => {
                return experimentsLoading.some((/**
                 * @param {?} isLoading
                 * @return {?}
                 */
                (isLoading) => isLoading));
            })));
            if (this.showHparamsAndMetrics) {
                /** @type {?} */
                const getHparamAndMetrics$ = this.store.select(getExperimentsHparamsAndMetrics, { experimentIds: this.experimentIds });
                // combineLatest, when initializing, emits twice
                this.hparamColumns$ = combineLatest([
                    this.store.select(getRunHparamFilterMap),
                    getHparamAndMetrics$,
                ]).pipe(map((/**
                 * @param {?} __0
                 * @return {?}
                 */
                ([filterMap, { hparams }]) => {
                    return hparams.map((/**
                     * @param {?} __0
                     * @return {?}
                     */
                    ({ name, displayName, domain }) => {
                        /** @type {?} */
                        const filter = filterMap.get(name);
                        if (!filter) {
                            throw new RangeError(`Invariant error: a filter for ${name} must exist` +
                                ` when the hparam exists`);
                        }
                        return { displayName, name, filter };
                    }));
                })));
                this.metricColumns$ = combineLatest([
                    this.store.select(getRunMetricFilterMap),
                    getHparamAndMetrics$,
                ]).pipe(map((/**
                 * @param {?} __0
                 * @return {?}
                 */
                ([filterMap, { metrics }]) => {
                    return metrics.map((/**
                     * @param {?} __0
                     * @return {?}
                     */
                    ({ tag, displayName }) => {
                        /** @type {?} */
                        const filter = filterMap.get(tag);
                        if (!filter) {
                            throw new RangeError(`Invariant error: a filter for ${tag} must exist ` +
                                `when the metric exists`);
                        }
                        return { displayName, tag, filter };
                    }));
                })));
            }
            this.store.dispatch(runTableShown({ experimentIds: this.experimentIds }));
        }
        /**
         * @private
         * @param {?} runItems$
         * @return {?}
         */
        getFilteredItems$(runItems$) {
            return combineLatest([
                runItems$,
                this.store.select(getRunSelectorRegexFilter),
            ]).pipe(map((/**
             * @param {?} __0
             * @return {?}
             */
            ([items, regexString]) => {
                if (!regexString) {
                    return items;
                }
                /** @type {?} */
                let regex = null;
                // Do not break all the future updates because of malformed
                // regexString. User can be still modifying it.
                try {
                    regex = regexString ? new RegExp(regexString) : null;
                }
                catch (e) { }
                if (!regex) {
                    return [];
                }
                /** @type {?} */
                const shouldIncludeExperimentName = this.columns.includes(RunsTableColumn.EXPERIMENT_NAME);
                return items.filter((/**
                 * @param {?} item
                 * @return {?}
                 */
                (item) => {
                    if (!shouldIncludeExperimentName) {
                        return (/** @type {?} */ (regex)).test(item.run.name);
                    }
                    return ((/** @type {?} */ (regex)).test(item.run.name) || (/** @type {?} */ (regex)).test(item.experimentAlias));
                }));
            })), combineLatestWith(this.store.select(getRunHparamFilterMap), this.store.select(getRunMetricFilterMap)), map((/**
             * @param {?} __0
             * @return {?}
             */
            ([items, hparamFilters, metricFilters]) => {
                if (!this.showHparamsAndMetrics) {
                    return items;
                }
                return items.filter((/**
                 * @param {?} __0
                 * @return {?}
                 */
                ({ hparams, metrics }) => {
                    /** @type {?} */
                    const hparamMatches = [...hparamFilters.entries()].every((/**
                     * @param {?} __0
                     * @return {?}
                     */
                    ([hparamName, filter]) => {
                        /** @type {?} */
                        const value = hparams.get(hparamName);
                        return matchFilter(filter, value);
                    }));
                    return (hparamMatches &&
                        [...metricFilters.entries()].every((/**
                         * @param {?} __0
                         * @return {?}
                         */
                        ([metricTag, filter]) => {
                            /** @type {?} */
                            const value = metrics.get(metricTag);
                            return matchFilter(filter, value);
                        })));
                }));
            })));
        }
        /**
         * @private
         * @param {?} filteredItems$
         * @return {?}
         */
        sortedAndSlicedItems$(filteredItems$) {
            /** @type {?} */
            const sortedItems = combineLatest([
                filteredItems$,
                this.store.select(getRunSelectorSort),
            ]).pipe(map((/**
             * @param {?} __0
             * @return {?}
             */
            ([items, sort]) => {
                /** @type {?} */
                const column = (/** @type {?} */ (sort.column));
                /** @type {?} */
                const direction = sort.direction;
                return sortRunTableItems(items, { column, direction });
            })));
            /** @type {?} */
            const slicedItems = combineLatest([
                sortedItems,
                this.store.select(getRunSelectorPaginationOption),
            ]).pipe(map((/**
             * @param {?} __0
             * @return {?}
             */
            ([items, paginationOption]) => {
                if (!this.usePagination) {
                    return items.slice();
                }
                const { pageSize, pageIndex } = paginationOption;
                return items.slice(pageIndex * pageSize, (pageIndex + 1) * pageSize);
            })));
            return slicedItems;
        }
        /**
         * @private
         * @param {?} experimentId
         * @return {?}
         */
        getRunTableItemsForExperiment(experimentId) {
            return combineLatest([
                this.store.select(getRuns, { experimentId }),
                this.store.select(getExperiment, { experimentId }),
                this.store.select(getCurrentRouteRunSelection),
                this.store.select(getRunColorMap),
                this.store.select(getExperimentIdToAliasMap),
            ]).pipe(map((/**
             * @param {?} __0
             * @return {?}
             */
            ([runs, experiment, selectionMap, colorMap, experimentIdToAlias]) => {
                return runs.map((/**
                 * @param {?} run
                 * @return {?}
                 */
                (run) => {
                    /** @type {?} */
                    const hparamMap = new Map();
                    (run.hparams || []).forEach((/**
                     * @param {?} hparam
                     * @return {?}
                     */
                    (hparam) => {
                        hparamMap.set(hparam.name, hparam.value);
                    }));
                    /** @type {?} */
                    const metricMap = new Map();
                    (run.metrics || []).forEach((/**
                     * @param {?} metric
                     * @return {?}
                     */
                    (metric) => {
                        metricMap.set(metric.tag, metric.value);
                    }));
                    return {
                        run,
                        experimentName: (experiment === null || experiment === void 0 ? void 0 : experiment.name) || '',
                        experimentAlias: experimentIdToAlias[experimentId],
                        selected: Boolean(selectionMap && selectionMap.get(run.id)),
                        runColor: colorMap[run.id],
                        hparams: hparamMap,
                        metrics: metricMap,
                    };
                }));
            })));
        }
        /**
         * @param {?} item
         * @return {?}
         */
        onRunSelectionToggle(item) {
            this.store.dispatch(runSelectionToggled({
                experimentIds: this.experimentIds,
                runId: item.run.id,
            }));
        }
        // When `usePagination` is false, page selection affects the single page,
        // containing all items.
        /**
         * @param {?} event
         * @return {?}
         */
        onPageSelectionToggle(event) {
            const { items } = event;
            /** @type {?} */
            const runIds = items.map((/**
             * @param {?} __0
             * @return {?}
             */
            ({ run }) => run.id));
            this.store.dispatch(runPageSelectionToggled({
                experimentIds: this.experimentIds,
                runIds,
            }));
        }
        /**
         * @return {?}
         */
        onSelectAllPages() {
            if (!this.usePagination) {
                throw new Error('Select all events cannot be dispatched when pagination is disabled');
            }
            this.store.dispatch(runsSelectAll({ experimentIds: this.experimentIds }));
        }
        /**
         * @param {?} event
         * @return {?}
         */
        onPaginationChange(event) {
            if (!this.usePagination) {
                throw new Error('Pagination events cannot be dispatched when pagination is disabled');
            }
            const { pageIndex, pageSize } = event;
            this.store.dispatch(runSelectorPaginationOptionChanged({ pageIndex, pageSize }));
        }
        /**
         * @param {?} sort
         * @return {?}
         */
        onSortChange(sort) {
            this.store.dispatch(runSelectorSortChanged(sort));
        }
        /**
         * @param {?} regexString
         * @return {?}
         */
        onRegexFilterChange(regexString) {
            this.store.dispatch(runSelectorRegexFilterChanged({ regexString }));
        }
        /**
         * @param {?} __0
         * @return {?}
         */
        onRunColorChange({ runId, newColor }) {
            this.store.dispatch(runColorChanged({ runId, newColor }));
        }
        /**
         * @param {?} event
         * @return {?}
         */
        onHparamDiscreteFilterChanged(event) {
            const { hparamName, filterValues, includeUndefined } = event;
            this.store.dispatch(runDiscreteHparamFilterChanged({
                hparamName,
                filterValues,
                includeUndefined,
            }));
        }
        /**
         * @param {?} event
         * @return {?}
         */
        onHparamIntervalFilterChanged(event) {
            const { name, filterLowerValue, filterUpperValue, includeUndefined } = event;
            this.store.dispatch(runIntervalHparamFilterChanged({
                hparamName: name,
                filterLowerValue,
                filterUpperValue,
                includeUndefined,
            }));
        }
        /**
         * @param {?} event
         * @return {?}
         */
        onMetricFilterChanged(event) {
            const { name, includeUndefined, filterLowerValue, filterUpperValue } = event;
            this.store.dispatch(runMetricFilterChanged({
                metricTag: name,
                includeUndefined,
                filterLowerValue,
                filterUpperValue,
            }));
        }
    }
    RunsTableContainer.decorators = [
        { type: Component, args: [{
                    selector: 'runs-table',
                    template: `
    <runs-table-component
      [useFlexibleLayout]="useFlexibleLayout"
      [numSelectedItems]="numSelectedItems$ | async"
      [columns]="columns"
      [hparamColumns]="hparamColumns$ | async"
      [metricColumns]="metricColumns$ | async"
      [showExperimentName]="isExperimentNameVisible()"
      [pageItems]="pageItems$ | async"
      [filteredItemsLength]="filteredItemsLength$ | async"
      [allItemsLength]="allItemsLength$ | async"
      [loading]="loading$ | async"
      [paginationOption]="paginationOption$ | async"
      [regexFilter]="regexFilter$ | async"
      [sortOption]="sortOption$ | async"
      [usePagination]="usePagination"
      (onSelectionToggle)="onRunSelectionToggle($event)"
      (onPageSelectionToggle)="onPageSelectionToggle($event)"
      (onSelectAllPages)="onSelectAllPages()"
      (onPaginationChange)="onPaginationChange($event)"
      (onRegexFilterChange)="onRegexFilterChange($event)"
      (onSortChange)="onSortChange($event)"
      (onRunColorChange)="onRunColorChange($event)"
      (onHparamIntervalFilterChanged)="onHparamIntervalFilterChanged($event)"
      (onHparamDiscreteFilterChanged)="onHparamDiscreteFilterChanged($event)"
      (onMetricFilterChanged)="onMetricFilterChanged($event)"
    ></runs-table-component>
  `,
                    host: {
                        '[class.flex-layout]': 'useFlexibleLayout',
                    },
                    styles: [
                        `
      :host.flex-layout {
        display: flex;
      }

      :host.flex-layout > runs-table-component {
        width: 100%;
      }
    `,
                    ],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    RunsTableContainer.ctorParameters = () => [
        { type: Store }
    ];
    RunsTableContainer.propDecorators = {
        useFlexibleLayout: [{ type: Input }],
        usePagination: [{ type: Input }],
        columns: [{ type: Input }],
        experimentIds: [{ type: Input }],
        showHparamsAndMetrics: [{ type: Input }]
    };
    /** @nocollapse */ RunsTableContainer.ɵfac = function RunsTableContainer_Factory(t) { return new (t || RunsTableContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ RunsTableContainer.ɵcmp = ɵɵdefineComponent({ type: RunsTableContainer, selectors: [["runs-table"]], hostVars: 2, hostBindings: function RunsTableContainer_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵclassProp("flex-layout", ctx.useFlexibleLayout);
        } }, inputs: { useFlexibleLayout: "useFlexibleLayout", usePagination: "usePagination", columns: "columns", experimentIds: "experimentIds", showHparamsAndMetrics: "showHparamsAndMetrics" }, decls: 11, vars: 34, consts: [[3, "useFlexibleLayout", "numSelectedItems", "columns", "hparamColumns", "metricColumns", "showExperimentName", "pageItems", "filteredItemsLength", "allItemsLength", "loading", "paginationOption", "regexFilter", "sortOption", "usePagination", "onSelectionToggle", "onPageSelectionToggle", "onSelectAllPages", "onPaginationChange", "onRegexFilterChange", "onSortChange", "onRunColorChange", "onHparamIntervalFilterChanged", "onHparamDiscreteFilterChanged", "onMetricFilterChanged"]], template: function RunsTableContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "runs-table-component", 0);
            ɵɵlistener("onSelectionToggle", function RunsTableContainer_Template_runs_table_component_onSelectionToggle_0_listener($event) { return ctx.onRunSelectionToggle($event); })("onPageSelectionToggle", function RunsTableContainer_Template_runs_table_component_onPageSelectionToggle_0_listener($event) { return ctx.onPageSelectionToggle($event); })("onSelectAllPages", function RunsTableContainer_Template_runs_table_component_onSelectAllPages_0_listener() { return ctx.onSelectAllPages(); })("onPaginationChange", function RunsTableContainer_Template_runs_table_component_onPaginationChange_0_listener($event) { return ctx.onPaginationChange($event); })("onRegexFilterChange", function RunsTableContainer_Template_runs_table_component_onRegexFilterChange_0_listener($event) { return ctx.onRegexFilterChange($event); })("onSortChange", function RunsTableContainer_Template_runs_table_component_onSortChange_0_listener($event) { return ctx.onSortChange($event); })("onRunColorChange", function RunsTableContainer_Template_runs_table_component_onRunColorChange_0_listener($event) { return ctx.onRunColorChange($event); })("onHparamIntervalFilterChanged", function RunsTableContainer_Template_runs_table_component_onHparamIntervalFilterChanged_0_listener($event) { return ctx.onHparamIntervalFilterChanged($event); })("onHparamDiscreteFilterChanged", function RunsTableContainer_Template_runs_table_component_onHparamDiscreteFilterChanged_0_listener($event) { return ctx.onHparamDiscreteFilterChanged($event); })("onMetricFilterChanged", function RunsTableContainer_Template_runs_table_component_onMetricFilterChanged_0_listener($event) { return ctx.onMetricFilterChanged($event); });
            ɵɵpipe(1, "async");
            ɵɵpipe(2, "async");
            ɵɵpipe(3, "async");
            ɵɵpipe(4, "async");
            ɵɵpipe(5, "async");
            ɵɵpipe(6, "async");
            ɵɵpipe(7, "async");
            ɵɵpipe(8, "async");
            ɵɵpipe(9, "async");
            ɵɵpipe(10, "async");
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵproperty("useFlexibleLayout", ctx.useFlexibleLayout)("numSelectedItems", ɵɵpipeBind1(1, 14, ctx.numSelectedItems$))("columns", ctx.columns)("hparamColumns", ɵɵpipeBind1(2, 16, ctx.hparamColumns$))("metricColumns", ɵɵpipeBind1(3, 18, ctx.metricColumns$))("showExperimentName", ctx.isExperimentNameVisible())("pageItems", ɵɵpipeBind1(4, 20, ctx.pageItems$))("filteredItemsLength", ɵɵpipeBind1(5, 22, ctx.filteredItemsLength$))("allItemsLength", ɵɵpipeBind1(6, 24, ctx.allItemsLength$))("loading", ɵɵpipeBind1(7, 26, ctx.loading$))("paginationOption", ɵɵpipeBind1(8, 28, ctx.paginationOption$))("regexFilter", ɵɵpipeBind1(9, 30, ctx.regexFilter$))("sortOption", ɵɵpipeBind1(10, 32, ctx.sortOption$))("usePagination", ctx.usePagination);
        } }, directives: [RunsTableComponent], pipes: [AsyncPipe], styles: [".flex-layout[_nghost-%COMP%] {\n        display: flex;\n      }\n\n      .flex-layout[_nghost-%COMP%]    > runs-table-component[_ngcontent-%COMP%] {\n        width: 100%;\n      }"], changeDetection: 0 });

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     *
     * @fileoverview Module for rendering list of runs from experiments in a table.
     *
     * Generated from: tensorboard/webapp/runs/views/runs_table/runs_table_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class RunsTableModule {
    }
    RunsTableModule.decorators = [
        { type: NgModule, args: [{
                    imports: [
                        ColorPickerModule,
                        CommonModule,
                        MatButtonModule,
                        MatCheckboxModule,
                        MatIconModule,
                        MatInputModule,
                        MatMenuModule,
                        MatPaginatorModule,
                        MatProgressSpinnerModule,
                        MatSortModule,
                        MatTableModule,
                        RangeInputModule,
                    ],
                    exports: [RunsTableContainer],
                    declarations: [RunsTableContainer, RunsTableComponent],
                },] },
    ];
    /** @nocollapse */ RunsTableModule.ɵmod = ɵɵdefineNgModule({ type: RunsTableModule });
    /** @nocollapse */ RunsTableModule.ɵinj = ɵɵdefineInjector({ factory: function RunsTableModule_Factory(t) { return new (t || RunsTableModule)(); }, imports: [[
                ColorPickerModule,
                CommonModule,
                MatButtonModule,
                MatCheckboxModule,
                MatIconModule,
                MatInputModule,
                MatMenuModule,
                MatPaginatorModule,
                MatProgressSpinnerModule,
                MatSortModule,
                MatTableModule,
                RangeInputModule,
            ]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(RunsTableModule, { declarations: [RunsTableContainer, RunsTableComponent], imports: [ColorPickerModule,
            CommonModule,
            MatButtonModule,
            MatCheckboxModule,
            MatIconModule,
            MatInputModule,
            MatMenuModule,
            MatPaginatorModule,
            MatProgressSpinnerModule,
            MatSortModule,
            MatTableModule,
            RangeInputModule], exports: [RunsTableContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/runs/views/runs_selector/runs_selector_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class RunsSelectorComponent {
    }
    RunsSelectorComponent.decorators = [
        { type: Component, args: [{
                    selector: 'runs-selector-component',
                    template: `
    <runs-table
      [useFlexibleLayout]="true"
      [columns]="columns"
      [experimentIds]="experimentIds"
      [showHparamsAndMetrics]="showHparamsAndMetrics"
    ></runs-table>
  `,
                    styles: [
                        `
      runs-table {
        height: 100%;
      }
    `,
                    ],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    RunsSelectorComponent.propDecorators = {
        experimentIds: [{ type: Input }],
        showHparamsAndMetrics: [{ type: Input }],
        columns: [{ type: Input }],
        hparamAndMetricSpecs: [{ type: Input }]
    };
    /** @nocollapse */ RunsSelectorComponent.ɵfac = function RunsSelectorComponent_Factory(t) { return new (t || RunsSelectorComponent)(); };
    /** @nocollapse */ RunsSelectorComponent.ɵcmp = ɵɵdefineComponent({ type: RunsSelectorComponent, selectors: [["runs-selector-component"]], inputs: { experimentIds: "experimentIds", showHparamsAndMetrics: "showHparamsAndMetrics", columns: "columns", hparamAndMetricSpecs: "hparamAndMetricSpecs" }, decls: 1, vars: 4, consts: [[3, "useFlexibleLayout", "columns", "experimentIds", "showHparamsAndMetrics"]], template: function RunsSelectorComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelement(0, "runs-table", 0);
        } if (rf & 2) {
            ɵɵproperty("useFlexibleLayout", true)("columns", ctx.columns)("experimentIds", ctx.experimentIds)("showHparamsAndMetrics", ctx.showHparamsAndMetrics);
        } }, directives: [RunsTableContainer], styles: ["runs-table[_ngcontent-%COMP%] {\n        height: 100%;\n      }"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/runs/views/runs_selector/runs_selector_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class RunsSelectorContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.experimentIds$ = this.store.select(getExperimentIdsFromRoute);
            this.columns$ = this.store.select(getExperimentIdsFromRoute).pipe(map((/**
             * @param {?} ids
             * @return {?}
             */
            (ids) => {
                return (/** @type {?} */ ([
                    RunsTableColumn.CHECKBOX,
                    RunsTableColumn.RUN_NAME,
                    ids && ids.length > 1 ? RunsTableColumn.EXPERIMENT_NAME : null,
                    RunsTableColumn.RUN_COLOR,
                ].filter((/**
                 * @param {?} col
                 * @return {?}
                 */
                (col) => col !== null))));
            })));
        }
    }
    RunsSelectorContainer.decorators = [
        { type: Component, args: [{
                    selector: 'runs-selector',
                    template: `
    <runs-selector-component
      [experimentIds]="experimentIds$ | async"
      [columns]="columns$ | async"
      [showHparamsAndMetrics]="showHparamsAndMetrics"
    ></runs-selector-component>
  `,
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    RunsSelectorContainer.ctorParameters = () => [
        { type: Store }
    ];
    RunsSelectorContainer.propDecorators = {
        showHparamsAndMetrics: [{ type: Input }]
    };
    /** @nocollapse */ RunsSelectorContainer.ɵfac = function RunsSelectorContainer_Factory(t) { return new (t || RunsSelectorContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ RunsSelectorContainer.ɵcmp = ɵɵdefineComponent({ type: RunsSelectorContainer, selectors: [["runs-selector"]], inputs: { showHparamsAndMetrics: "showHparamsAndMetrics" }, decls: 3, vars: 7, consts: [[3, "experimentIds", "columns", "showHparamsAndMetrics"]], template: function RunsSelectorContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelement(0, "runs-selector-component", 0);
            ɵɵpipe(1, "async");
            ɵɵpipe(2, "async");
        } if (rf & 2) {
            ɵɵproperty("experimentIds", ɵɵpipeBind1(1, 3, ctx.experimentIds$))("columns", ɵɵpipeBind1(2, 5, ctx.columns$))("showHparamsAndMetrics", ctx.showHparamsAndMetrics);
        } }, directives: [RunsSelectorComponent], pipes: [AsyncPipe], encapsulation: 2, changeDetection: 0 });

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     *
     * @fileoverview Module for rendering a table that lists all the runs from
     * different experiments with checkbox for run selection.
     *
     * Note that this an instance of the runs table with different configuration.
     *
     * Generated from: tensorboard/webapp/runs/views/runs_selector/runs_selector_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class RunsSelectorModule {
    }
    RunsSelectorModule.decorators = [
        { type: NgModule, args: [{
                    imports: [CommonModule, RunsTableModule],
                    exports: [RunsSelectorContainer],
                    declarations: [RunsSelectorComponent, RunsSelectorContainer],
                },] },
    ];
    /** @nocollapse */ RunsSelectorModule.ɵmod = ɵɵdefineNgModule({ type: RunsSelectorModule });
    /** @nocollapse */ RunsSelectorModule.ɵinj = ɵɵdefineInjector({ factory: function RunsSelectorModule_Factory(t) { return new (t || RunsSelectorModule)(); }, imports: [[CommonModule, RunsTableModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(RunsSelectorModule, { declarations: [RunsSelectorComponent, RunsSelectorContainer], imports: [CommonModule, RunsTableModule], exports: [RunsSelectorContainer] }); })();

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    const metricsTagMetadataRequested = createAction('[Metrics] Metrics Tag Metadata Requested');
    const metricsTagMetadataLoaded = createAction('[Metrics] Metrics Tag Metadata Loaded', props());
    const metricsTagMetadataFailed = createAction('[Metrics] Metrics Tag Metadata Failed');
    const metricsChangeTooltipSort = createAction('[Metrics] Metrics Settings Change Tooltip', props());
    const metricsToggleShowDataDownload = createAction('[Metrics] Metrics Settings Toggle Show Data Download');
    const metricsToggleIgnoreOutliers = createAction('[Metrics] Metrics Setting Toggle Ignore Outlier');
    const metricsChangeXAxisType = createAction('[Metrics] Metrics Setting Change X Axis Type', props());
    const metricsChangeScalarSmoothing = createAction('[Metrics] Metrics Setting Change Scalar Smoothing', props());
    const metricsChangeImageBrightness = createAction('[Metrics] Metrics Setting Change Image Brightness', props());
    const metricsChangeImageContrast = createAction('[Metrics] Metrics Setting Change Image Contrast', props());
    const metricsResetImageBrightness = createAction('[Metrics] Image Brightness Setting Reset');
    const metricsResetImageContrast = createAction('[Metrics] Image Contrast Setting Reset');
    const metricsToggleImageShowActualSize = createAction('[Metrics] Metrics Setting Toggle Image Show Actual Size');
    const metricsChangeHistogramMode = createAction('[Metrics] Metrics Setting Change Histogram Mode', props());
    const multipleTimeSeriesRequested = createAction('[Metrics] Multiple Time Series Requested', props());
    const fetchTimeSeriesFailed = createAction('[Metrics] Fetch Time Series Request Failed', props());
    const fetchTimeSeriesLoaded = createAction('[Metrics] Fetch Time Series Response Loaded', props());
    /**
     * An event when some cards enter or exit the viewport. The card sets must be
     * mutually exclusive.
     */
    const cardVisibilityChanged = createAction('[Metrics] Card Visibility Changed', props());
    const cardStepSliderChanged = createAction('[Metrics] Card Step Slider Changed', props());
    const metricsTagFilterChanged = createAction('[Metrics] Tag Filter Changed', props());
    const metricsTagGroupExpansionChanged = createAction('[Metrics] Metrics Tag Group Expansion Changed', props());
    const cardPinStateToggled = createAction('[Metrics] Card Pin State Toggled', props());

    /**
     * Returns a new object where all values have been transformed.
     * For familiarity, the API signature was made to be as close as possible to the
     * signatures of equivalent functions in popular libraries:
     * - https://lodash.com/docs/#mapValues
     * - https://underscorejs.org/#mapObject
     */
    function mapObjectValues(object, transform) {
        const result = {};
        for (const key of Object.keys(object)) {
            const typedKey = key;
            result[typedKey] = transform(object[typedKey], key);
        }
        return result;
    }

    var TimeProperty;
    (function (TimeProperty) {
        TimeProperty["STEP"] = "step";
        TimeProperty["WALL_TIME"] = "wall_time";
        TimeProperty["RELATIVE"] = "relative";
    })(TimeProperty || (TimeProperty = {}));
    var HistogramMode;
    (function (HistogramMode) {
        HistogramMode["OFFSET"] = "offset";
        HistogramMode["OVERLAY"] = "overlay";
    })(HistogramMode || (HistogramMode = {}));

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /** @enum {string} */
    const TooltipPosition = {
        BOTTOM: "bottom",
        RIGHT: "right",
        AUTO: "auto",
    };
    /** @enum {string} */
    const TooltipSortingMethod = {
        DEFAULT: "default",
        ASCENDING: "ascending",
        DESCENDING: "descending",
        NEAREST: "nearest",
    };
    /** @enum {string} */
    const XAxisType = {
        /** Linear scale using the "step" property of the datum. */
        STEP: "step",
        /**
         * Temporal scale using the earliest datum's "wall_time" as a baseline.
         */
        RELATIVE: "relative",
        /** Temporal scale using the "wall_time" property of the datum. */
        WALL_TIME: "wall_time",
    };
    /** @enum {string} */
    const YAxisType = {
        LINEAR: "linear",
        LOG: "log",
    };

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    var XAxisType$1;
    (function (XAxisType) {
        XAxisType[XAxisType["STEP"] = 0] = "STEP";
        XAxisType[XAxisType["RELATIVE"] = 1] = "RELATIVE";
        XAxisType[XAxisType["WALL_TIME"] = 2] = "WALL_TIME";
    })(XAxisType$1 || (XAxisType$1 = {}));

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    function buildCardMetadataList(tagMetadata) {
        const results = [];
        for (let pluginKey of Object.keys(tagMetadata)) {
            const plugin = pluginKey;
            let tagToRuns;
            if (isSampledPlugin(plugin)) {
                if (isSingleRunPlugin(plugin)) {
                    // Single-run, sampled format (e.g. Images).
                    const tagRunSampleInfo = tagMetadata[plugin].tagRunSampledInfo;
                    for (const tag of Object.keys(tagRunSampleInfo)) {
                        for (const runId of Object.keys(tagRunSampleInfo[tag])) {
                            const { maxSamplesPerStep } = tagRunSampleInfo[tag][runId];
                            for (let i = 0; i < maxSamplesPerStep; i++) {
                                results.push({ plugin, tag, runId, sample: i });
                            }
                        }
                    }
                }
                else {
                    throw new Error('Multi-run, sampled plugin support not yet implemented');
                }
            }
            else {
                if (isSingleRunPlugin(plugin)) {
                    // Single-run, unsampled format (e.g. Histograms).
                    tagToRuns = tagMetadata[plugin].tagToRuns;
                    for (const tag of Object.keys(tagToRuns)) {
                        for (const runId of tagToRuns[tag]) {
                            results.push({ plugin, tag, runId });
                        }
                    }
                }
                else {
                    // Multi-run, unsampled format (e.g. Scalars).
                    tagToRuns = tagMetadata[plugin].tagToRuns;
                    for (const tag of Object.keys(tagToRuns)) {
                        results.push({ plugin, tag, runId: null });
                    }
                }
            }
        }
        return results;
    }
    function getMaxStepIndex(cardId, cardMetadataMap, timeSeriesData) {
        const { plugin, tag, runId, sample } = cardMetadataMap[cardId];
        const loadable = getTimeSeriesLoadable(timeSeriesData, plugin, tag, sample);
        if (loadable) {
            if (runId !== null && loadable.runToSeries.hasOwnProperty(runId)) {
                const seriesLength = loadable.runToSeries[runId].length;
                return seriesLength > 0 ? seriesLength - 1 : null;
            }
            const seriesLengths = Object.values(loadable.runToSeries).map((series) => series.length);
            if (seriesLengths.length) {
                return Math.max(...seriesLengths) - 1;
            }
        }
        return null;
    }
    /**
     * Normalizes the step for cards in the step index map. This includes
     * - assign a default step
     * - reselect the max step, if the previous series' max was selected
     * - clamp to the new series' max
     * - set to `null` if the series contains no step data
     */
    function buildNormalizedCardStepIndexMap(cardMetadataMap, cardStepIndex, timeSeriesData, prevTimeSeriesData) {
        const result = Object.assign({}, cardStepIndex);
        for (const cardId in cardMetadataMap) {
            if (!cardMetadataMap.hasOwnProperty(cardId)) {
                continue;
            }
            const maxStepIndex = getMaxStepIndex(cardId, cardMetadataMap, timeSeriesData);
            if (maxStepIndex === null) {
                if (cardStepIndex.hasOwnProperty(cardId)) {
                    result[cardId] = null;
                }
                continue;
            }
            const stepIndex = cardStepIndex.hasOwnProperty(cardId)
                ? cardStepIndex[cardId]
                : null;
            const prevMaxStepIndex = getMaxStepIndex(cardId, cardMetadataMap, prevTimeSeriesData);
            const prevWasMax = stepIndex !== null && stepIndex === prevMaxStepIndex;
            const shouldClamp = stepIndex !== null && stepIndex > maxStepIndex;
            const shouldAutoSelectMax = stepIndex === null || prevWasMax;
            if (shouldClamp || shouldAutoSelectMax) {
                result[cardId] = maxStepIndex;
            }
        }
        return result;
    }
    /**
     * Builds a new loadable whose runToLoadState represents a state where nothing
     * is loaded, and runToSeries is preserved.
     */
    function buildResetLoadable(loadable) {
        const runToLoadState = mapObjectValues(loadable.runToLoadState, (loadState) => {
            return loadState === DataLoadState.LOADING
                ? DataLoadState.LOADING
                : DataLoadState.NOT_LOADED;
        });
        return Object.assign(Object.assign({}, loadable), { runToLoadState });
    }
    /**
     * Returns an identifier useful for comparing a card in storage with a real card
     * with loaded metadata.
     */
    function serializeCardUniqueInfo(plugin, tag, runId, sample) {
        return JSON.stringify([plugin, tag, runId || '', sample]);
    }
    const { initialState: initialState$5, reducers: routeContextReducer } = createRouteContextedState({
        // Backend data.
        tagMetadataLoaded: DataLoadState.NOT_LOADED,
        tagMetadata: {
            scalars: {
                tagDescriptions: {},
                tagToRuns: {},
            },
            histograms: {
                tagDescriptions: {},
                tagToRuns: {},
            },
            images: {
                tagDescriptions: {},
                tagRunSampledInfo: {},
            },
        },
        // Cards.
        cardList: [],
        cardToPinnedCopy: new Map(),
        pinnedCardToOriginal: new Map(),
        unresolvedImportedPinnedCards: [],
        cardMetadataMap: {},
        cardStepIndex: {},
        tagFilter: '',
        tagGroupExpanded: new Map(),
    }, {
        timeSeriesData: {
            scalars: {},
            histograms: {},
            images: {},
        },
        settings: {
            tooltipSort: TooltipSortingMethod.DEFAULT,
            ignoreOutliers: true,
            xAxisType: XAxisType$1.STEP,
            scalarSmoothing: 0.6,
            imageBrightnessInMilli: 1000,
            imageContrastInMilli: 1000,
            imageShowActualSize: false,
            histogramMode: HistogramMode.OFFSET,
        },
        visibleCards: new Set(),
    }, 
    /** onRouteIdChanged */
    (state) => {
        return Object.assign(Object.assign({}, state), { 
            // Reset visible cards in case we resume a route that was left dirty.
            // Since visibility tracking is async, the state may not have received
            // 'exited card' updates when it was cached by the router.
            visibleCards: new Set() });
    });
    const reducer$5 = createReducer(initialState$5, on(stateRehydratedFromUrl, (state, { routeKind, partialState }) => {
        if (routeKind !== RouteKind.EXPERIMENT &&
            routeKind !== RouteKind.COMPARE_EXPERIMENT) {
            return state;
        }
        // The URL contains pinned cards + unresolved imported pins. Keep these sets
        // mutually exclusive, and do not add duplicate any unresolved imported
        // cards.
        const serializedCardUniqueInfos = new Set();
        // Visit existing pins.
        for (const pinnedCardId of state.pinnedCardToOriginal.keys()) {
            const { plugin, tag, runId, sample } = state.cardMetadataMap[pinnedCardId];
            serializedCardUniqueInfos.add(serializeCardUniqueInfo(plugin, tag, runId, sample));
        }
        // We need to include previous unresolved imported pins, because the new
        // hydrated state might not include them. For example, navigating from
        // experiment A (with pins) --> B --> A, we want to ensure that rehydration
        // does not drop the old unresolved pins on A.
        const hydratedState = partialState;
        const unresolvedImportedPinnedCards = [];
        for (const card of [
            ...state.unresolvedImportedPinnedCards,
            ...hydratedState.metrics.pinnedCards,
        ]) {
            const cardUniqueInfoString = serializeCardUniqueInfo(card.plugin, card.tag, card.runId, card.sample);
            if (!serializedCardUniqueInfos.has(cardUniqueInfoString)) {
                serializedCardUniqueInfos.add(cardUniqueInfoString);
                unresolvedImportedPinnedCards.push(card);
            }
        }
        const resolvedResult = buildOrReturnStateWithUnresolvedImportedPins(unresolvedImportedPinnedCards, state.cardList, state.cardMetadataMap, state.cardToPinnedCopy, state.pinnedCardToOriginal, state.cardStepIndex);
        return Object.assign(Object.assign({}, state), resolvedResult);
    }), on(reload, manualReload, (state) => {
        const nextTagMetadataLoaded = state.tagMetadataLoaded === DataLoadState.LOADING
            ? DataLoadState.LOADING
            : DataLoadState.NOT_LOADED;
        const nextTimeSeriesData = mapObjectValues(state.timeSeriesData, (pluginData, plugin) => {
            return mapObjectValues(pluginData, (tagData) => {
                if (!isSampledPlugin(plugin)) {
                    return buildResetLoadable(tagData);
                }
                return mapObjectValues(tagData, (sampleData) => {
                    return buildResetLoadable(sampleData);
                });
            });
        });
        return Object.assign(Object.assign({}, state), { tagMetadataLoaded: nextTagMetadataLoaded, timeSeriesData: nextTimeSeriesData });
    }), on(metricsTagMetadataRequested, (state) => {
        return Object.assign(Object.assign({}, state), { tagMetadataLoaded: DataLoadState.LOADING });
    }), on(metricsTagMetadataFailed, (state) => {
        return Object.assign(Object.assign({}, state), { tagMetadataLoaded: DataLoadState.FAILED });
    }), on(metricsTagMetadataLoaded, (state, { tagMetadata }) => {
        const nextTagMetadata = {
            scalars: buildPluginTagData(tagMetadata, PluginType.SCALARS),
            histograms: buildPluginTagData(tagMetadata, PluginType.HISTOGRAMS),
            images: tagMetadata[PluginType.IMAGES],
        };
        // Carry over pre-existing card metadata, even if the new tag
        // metadata does not include it.
        const nextCardMetadataMap = Object.assign({}, state.cardMetadataMap);
        const nextCardMetadataList = buildCardMetadataList(nextTagMetadata);
        const newCardIds = [];
        // Create new cards for unseen metadata.
        for (const cardMetadata of nextCardMetadataList) {
            const cardId = getCardId(cardMetadata);
            if (!state.cardMetadataMap.hasOwnProperty(cardId)) {
                nextCardMetadataMap[cardId] = cardMetadata;
                newCardIds.push(cardId);
            }
        }
        const nextCardList = [...state.cardList, ...newCardIds];
        const resolvedResult = buildOrReturnStateWithUnresolvedImportedPins(state.unresolvedImportedPinnedCards, newCardIds, nextCardMetadataMap, state.cardToPinnedCopy, state.pinnedCardToOriginal, state.cardStepIndex);
        return Object.assign(Object.assign(Object.assign({}, state), resolvedResult), { tagMetadataLoaded: DataLoadState.LOADED, tagMetadata: nextTagMetadata, cardList: nextCardList });
    }), on(metricsTagFilterChanged, (state, { tagFilter }) => {
        return Object.assign(Object.assign({}, state), { tagFilter });
    }), on(metricsChangeTooltipSort, (state, { sort }) => {
        return Object.assign(Object.assign({}, state), { settings: Object.assign(Object.assign({}, state.settings), { tooltipSort: sort }) });
    }), on(metricsToggleIgnoreOutliers, (state) => {
        return Object.assign(Object.assign({}, state), { settings: Object.assign(Object.assign({}, state.settings), { ignoreOutliers: !state.settings.ignoreOutliers }) });
    }), on(metricsChangeXAxisType, (state, { xAxisType }) => {
        return Object.assign(Object.assign({}, state), { settings: Object.assign(Object.assign({}, state.settings), { xAxisType }) });
    }), on(metricsChangeScalarSmoothing, (state, { smoothing }) => {
        return Object.assign(Object.assign({}, state), { settings: Object.assign(Object.assign({}, state.settings), { scalarSmoothing: smoothing }) });
    }), on(metricsChangeImageBrightness, (state, { brightnessInMilli }) => {
        return Object.assign(Object.assign({}, state), { settings: Object.assign(Object.assign({}, state.settings), { imageBrightnessInMilli: brightnessInMilli }) });
    }), on(metricsChangeImageContrast, (state, { contrastInMilli }) => {
        return Object.assign(Object.assign({}, state), { settings: Object.assign(Object.assign({}, state.settings), { imageContrastInMilli: contrastInMilli }) });
    }), on(metricsResetImageBrightness, (state) => {
        return Object.assign(Object.assign({}, state), { settings: Object.assign(Object.assign({}, state.settings), { imageBrightnessInMilli: initialState$5.settings.imageBrightnessInMilli }) });
    }), on(metricsResetImageContrast, (state) => {
        return Object.assign(Object.assign({}, state), { settings: Object.assign(Object.assign({}, state.settings), { imageContrastInMilli: initialState$5.settings.imageContrastInMilli }) });
    }), on(metricsToggleImageShowActualSize, (state) => {
        return Object.assign(Object.assign({}, state), { settings: Object.assign(Object.assign({}, state.settings), { imageShowActualSize: !state.settings.imageShowActualSize }) });
    }), on(metricsChangeHistogramMode, (state, { histogramMode }) => {
        return Object.assign(Object.assign({}, state), { settings: Object.assign(Object.assign({}, state.settings), { histogramMode }) });
    }), on(multipleTimeSeriesRequested, (state, { requests }) => {
        if (!requests.length) {
            return state;
        }
        // TODO(psybuzz): the `experimentIds` field on requests is currently
        // ignored. Ideally this reducer should support requesting a
        // subset of all experiments.
        const nextTimeSeriesData = Object.assign({}, state.timeSeriesData);
        for (const request of requests) {
            const { plugin, tag, sample } = request;
            nextTimeSeriesData[plugin] = createPluginDataWithLoadable(nextTimeSeriesData, plugin, tag, sample);
            const loadable = getTimeSeriesLoadable(nextTimeSeriesData, plugin, tag, sample);
            const runIds = isSingleRunTimeSeriesRequest(request)
                ? [request.runId]
                : getRunIds(state.tagMetadata, plugin, tag, sample);
            loadable.runToLoadState = createRunToLoadState(DataLoadState.LOADING, runIds, loadable.runToLoadState);
        }
        return Object.assign(Object.assign({}, state), { timeSeriesData: nextTimeSeriesData });
    }), on(fetchTimeSeriesFailed, (state, { request }) => {
        const nextTimeSeriesData = Object.assign({}, state.timeSeriesData);
        const { plugin, tag, sample } = request;
        nextTimeSeriesData[plugin] = createPluginDataWithLoadable(nextTimeSeriesData, plugin, tag, sample);
        const loadable = getTimeSeriesLoadable(nextTimeSeriesData, plugin, tag, sample);
        const runIds = isSingleRunTimeSeriesRequest(request)
            ? [request.runId]
            : getRunIds(state.tagMetadata, plugin, tag, sample);
        loadable.runToLoadState = createRunToLoadState(DataLoadState.FAILED, runIds, loadable.runToLoadState);
        return Object.assign(Object.assign({}, state), { timeSeriesData: nextTimeSeriesData });
    }), on(fetchTimeSeriesLoaded, (state, { response }) => {
        // Update time series.
        const nextTimeSeriesData = Object.assign({}, state.timeSeriesData);
        const { plugin, tag, runId, sample } = response;
        nextTimeSeriesData[plugin] = createPluginDataWithLoadable(nextTimeSeriesData, plugin, tag, sample);
        const loadable = getTimeSeriesLoadable(nextTimeSeriesData, plugin, tag, sample);
        if (isFailedTimeSeriesResponse(response)) {
            const runIds = runId
                ? [runId]
                : getRunIds(state.tagMetadata, plugin, tag, sample);
            loadable.runToLoadState = createRunToLoadState(DataLoadState.FAILED, runIds, loadable.runToLoadState);
        }
        else {
            const runToSeries = response.runToSeries;
            loadable.runToSeries = Object.assign({}, loadable.runToSeries);
            loadable.runToLoadState = Object.assign({}, loadable.runToLoadState);
            for (const runId in runToSeries) {
                if (runToSeries.hasOwnProperty(runId)) {
                    loadable.runToSeries[runId] = runToSeries[runId];
                    loadable.runToLoadState[runId] = DataLoadState.LOADED;
                }
            }
        }
        const nextState = Object.assign(Object.assign({}, state), { timeSeriesData: nextTimeSeriesData, cardStepIndex: buildNormalizedCardStepIndexMap(state.cardMetadataMap, state.cardStepIndex, nextTimeSeriesData, state.timeSeriesData) });
        return nextState;
    }), on(cardStepSliderChanged, (state, { cardId, stepIndex }) => {
        const maxStepIndex = getMaxStepIndex(cardId, state.cardMetadataMap, state.timeSeriesData);
        let nextStepIndex = stepIndex;
        if (maxStepIndex === null) {
            nextStepIndex = null;
        }
        else if (stepIndex > maxStepIndex) {
            nextStepIndex = maxStepIndex;
        }
        return Object.assign(Object.assign({}, state), { cardStepIndex: Object.assign(Object.assign({}, state.cardStepIndex), { [cardId]: nextStepIndex }) });
    }), on(metricsTagGroupExpansionChanged, (state, { tagGroup }) => {
        const tagGroupExpanded = new Map(state.tagGroupExpanded);
        tagGroupExpanded.set(tagGroup, !tagGroupExpanded.get(tagGroup));
        return Object.assign(Object.assign({}, state), { tagGroupExpanded });
    }), on(cardVisibilityChanged, (state, { enteredCards, exitedCards }) => {
        if (enteredCards.size === 0 && exitedCards.size === 0) {
            return state;
        }
        const visibleCards = new Set(state.visibleCards);
        enteredCards.forEach((cardId) => {
            visibleCards.add(cardId);
        });
        exitedCards.forEach((cardId) => {
            visibleCards.delete(cardId);
            if (enteredCards.has(cardId)) {
                throw new Error(`A 'cardVisibilityChanged' with an invalid ` +
                    `payload contains overlapping sets`);
            }
        });
        return Object.assign(Object.assign({}, state), { visibleCards });
    }), on(cardPinStateToggled, (state, { cardId }) => {
        const isPinnedCopy = state.pinnedCardToOriginal.has(cardId);
        const shouldPin = isPinnedCopy
            ? false
            : !state.cardToPinnedCopy.has(cardId);
        if (shouldPin && !canCreateNewPins(state)) {
            return state;
        }
        let nextCardToPinnedCopy = new Map(state.cardToPinnedCopy);
        let nextPinnedCardToOriginal = new Map(state.pinnedCardToOriginal);
        let nextCardMetadataMap = Object.assign({}, state.cardMetadataMap);
        let nextCardStepIndexMap = Object.assign({}, state.cardStepIndex);
        if (isPinnedCopy) {
            const originalCardId = state.pinnedCardToOriginal.get(cardId);
            nextCardToPinnedCopy.delete(originalCardId);
            nextPinnedCardToOriginal.delete(cardId);
            delete nextCardMetadataMap[cardId];
            delete nextCardStepIndexMap[cardId];
        }
        else {
            if (shouldPin) {
                const resolvedResult = buildOrReturnStateWithPinnedCopy(cardId, nextCardToPinnedCopy, nextPinnedCardToOriginal, nextCardStepIndexMap, nextCardMetadataMap);
                nextCardToPinnedCopy = resolvedResult.cardToPinnedCopy;
                nextPinnedCardToOriginal = resolvedResult.pinnedCardToOriginal;
                nextCardMetadataMap = resolvedResult.cardMetadataMap;
                nextCardStepIndexMap = resolvedResult.cardStepIndex;
            }
            else {
                const pinnedCardId = state.cardToPinnedCopy.get(cardId);
                nextCardToPinnedCopy.delete(cardId);
                nextPinnedCardToOriginal.delete(pinnedCardId);
                delete nextCardMetadataMap[pinnedCardId];
                delete nextCardStepIndexMap[pinnedCardId];
            }
        }
        return Object.assign(Object.assign({}, state), { cardMetadataMap: nextCardMetadataMap, cardStepIndex: nextCardStepIndexMap, cardToPinnedCopy: nextCardToPinnedCopy, pinnedCardToOriginal: nextPinnedCardToOriginal });
    }));
    function reducers$7(state, action) {
        return composeReducers(reducer$5, routeContextReducer)(state, action);
    }
    function buildPluginTagData(tagMetadata, pluginType) {
        return {
            tagDescriptions: tagMetadata[pluginType].tagDescriptions,
            tagToRuns: buildTagToRuns(tagMetadata[pluginType].runTagInfo),
        };
    }
    /**
     * Takes a run-to-tag map and inverts it.
     */
    function buildTagToRuns(runTagInfo) {
        const tagToRuns = {};
        for (const run in runTagInfo) {
            for (const tag of runTagInfo[run]) {
                tagToRuns[tag] = [...(tagToRuns[tag] || []), run];
            }
        }
        return tagToRuns;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/effects/index.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const getCardFetchInfo = createSelector(getCardLoadState, getCardMetadata, (/**
     * @param {?} loadState
     * @param {?} maybeMetadata
     * @param {?} cardId
     * @return {?}
     */
    (loadState, maybeMetadata, cardId /* props */) => {
        if (!maybeMetadata) {
            return null;
        }
        return Object.assign(Object.assign({}, maybeMetadata), { loadState, id: cardId });
    }));
    /** @type {?} */
    const initAction$1 = createAction('[Metrics Effects] Init');
    class MetricsEffects {
        /**
         * @param {?} actions$
         * @param {?} store
         * @param {?} dataSource
         */
        constructor(actions$, store, dataSource) {
            this.actions$ = actions$;
            this.store = store;
            this.dataSource = dataSource;
            /**
             * Our effects react when the plugin dashboard is fully "shown" and experiment
             * ids are available. The `activePlugin` acts as our proxy to know whether it
             * is shown.
             *
             * [Metrics Effects] Init  - the initial `activePlugin` is set.
             * [Core] Plugin Changed   - subsequent `activePlugin` updates.
             * [App Routing] Navigated - experiment id updates.
             */
            this.dashboardShownWithoutData$ = this.actions$.pipe(ofType(initAction$1, changePlugin, navigated), withLatestFrom(this.store.select(getActivePlugin), this.store.select(getMetricsTagMetadataLoaded)), filter((/**
             * @param {?} __0
             * @return {?}
             */
            ([, activePlugin, tagLoadState]) => {
                return (activePlugin === METRICS_PLUGIN_ID &&
                    tagLoadState === DataLoadState.NOT_LOADED);
            })));
            this.reloadRequestedWhileShown$ = this.actions$.pipe(ofType(reload, manualReload), withLatestFrom(this.store.select(getActivePlugin)), filter((/**
             * @param {?} __0
             * @return {?}
             */
            ([, activePlugin]) => {
                return activePlugin === METRICS_PLUGIN_ID;
            })));
            this.loadTagMetadata$ = merge(this.dashboardShownWithoutData$, this.reloadRequestedWhileShown$).pipe(withLatestFrom(this.store.select(getMetricsTagMetadataLoaded), this.store.select(getExperimentIdsFromRoute)), filter((/**
             * @param {?} __0
             * @return {?}
             */
            ([, tagLoadState, experimentIds]) => {
                /**
                 * When `experimentIds` is null, the actual ids have not
                 * appeared in the store yet.
                 */
                return tagLoadState !== DataLoadState.LOADING && experimentIds !== null;
            })), tap((/**
             * @return {?}
             */
            () => {
                this.store.dispatch(metricsTagMetadataRequested());
            })), switchMap((/**
             * @param {?} __0
             * @return {?}
             */
            ([, , experimentIds]) => {
                return this.dataSource.fetchTagMetadata((/** @type {?} */ (experimentIds))).pipe(tap((/**
                 * @param {?} tagMetadata
                 * @return {?}
                 */
                (tagMetadata) => {
                    this.store.dispatch(metricsTagMetadataLoaded({ tagMetadata }));
                })), catchError((/**
                 * @return {?}
                 */
                () => {
                    this.store.dispatch(metricsTagMetadataFailed());
                    return of(null);
                })));
            })));
            this.visibleCardsWithoutDataChanged$ = this.actions$.pipe(ofType(cardVisibilityChanged), switchMap((/**
             * @return {?}
             */
            () => this.getVisibleCardFetchInfos().pipe(take(1)))), map((/**
             * @param {?} fetchInfos
             * @return {?}
             */
            (fetchInfos) => {
                return fetchInfos.filter((/**
                 * @param {?} fetchInfo
                 * @return {?}
                 */
                (fetchInfo) => {
                    return fetchInfo.loadState === DataLoadState.NOT_LOADED;
                }));
            })));
            this.visibleCardsReloaded$ = this.reloadRequestedWhileShown$.pipe(switchMap((/**
             * @return {?}
             */
            () => this.getVisibleCardFetchInfos().pipe(take(1)))), map((/**
             * @param {?} fetchInfos
             * @return {?}
             */
            (fetchInfos) => {
                return fetchInfos.filter((/**
                 * @param {?} fetchInfo
                 * @return {?}
                 */
                (fetchInfo) => {
                    return fetchInfo.loadState !== DataLoadState.LOADING;
                }));
            })));
            this.loadTimeSeries$ = merge(this.visibleCardsWithoutDataChanged$, this.visibleCardsReloaded$).pipe(filter((/**
             * @param {?} fetchInfos
             * @return {?}
             */
            (fetchInfos) => fetchInfos.length > 0)), 
            // Ignore card visibility events until we have non-null
            // experimentIds.
            withLatestFrom(this.store
                .select(getExperimentIdsFromRoute)
                .pipe(filter((/**
             * @param {?} experimentIds
             * @return {?}
             */
            (experimentIds) => experimentIds !== null)))), mergeMap((/**
             * @param {?} __0
             * @return {?}
             */
            ([fetchInfos, experimentIds]) => {
                return this.fetchTimeSeriesForCards(fetchInfos, (/** @type {?} */ (experimentIds)));
            })));
            /**
               * In general, this effect dispatch the following actions:
               *
               * On dashboard shown with visible cards:
               * - metricsTagMetadataRequested
               * - multipleTimeSeriesRequested
               *
               * On reloads:
               * - metricsTagMetadataRequested
               * - multipleTimeSeriesRequested
               *
               * On data source responses:
               * - metricsTagMetadataLoaded
               * - metricsTagMetadataFailed
               * - fetchTimeSeriesLoaded
               * - fetchTimeSeriesFailed
               */
            /**
             * @export
             */
            this.allEffects$ = createEffect((/**
             * @return {?}
             */
            () => {
                return merge(
                /**
                 * Subscribes to: dashboard shown, route navigation, reloads.
                 */
                this.loadTagMetadata$, 
                /**
                 * Subscribes to: card visibility, reloads.
                 */
                this.loadTimeSeries$);
            }), { dispatch: false });
        }
        /**
         * @export
         * @return {?}
         */
        ngrxOnInitEffects() {
            return initAction$1();
        }
        /**
         * @private
         * @return {?}
         */
        getVisibleCardFetchInfos() {
            /** @type {?} */
            const visibleCardIds$ = this.store.select(getVisibleCardIds);
            return visibleCardIds$.pipe(switchMap((/**
             * @param {?} cardIds
             * @return {?}
             */
            (cardIds) => {
                // Explicitly notify subscribers that there are no visible cards,
                // since `forkJoin` does not emit when passed an empty array.
                if (!cardIds.length) {
                    return of([]);
                }
                /** @type {?} */
                const observables = cardIds.map((/**
                 * @param {?} cardId
                 * @return {?}
                 */
                (cardId) => {
                    return this.store.select(getCardFetchInfo, cardId).pipe(take(1));
                }));
                return forkJoin(observables);
            })), map((/**
             * @param {?} fetchInfos
             * @return {?}
             */
            (fetchInfos) => {
                return (/** @type {?} */ (fetchInfos.filter(Boolean)));
            })));
        }
        /**
         * @private
         * @param {?} request
         * @return {?}
         */
        fetchTimeSeries(request) {
            return this.dataSource.fetchTimeSeries([request]).pipe(tap((/**
             * @param {?} responses
             * @return {?}
             */
            (responses) => {
                /** @type {?} */
                const errors = responses.filter(isFailedTimeSeriesResponse);
                if (errors.length) {
                    console.error('Time series response contained errors:', errors);
                }
                this.store.dispatch(fetchTimeSeriesLoaded({ response: responses[0] }));
            })), catchError((/**
             * @return {?}
             */
            () => {
                this.store.dispatch(fetchTimeSeriesFailed({ request }));
                return of(null);
            })));
        }
        /**
         * @private
         * @param {?} fetchInfos
         * @param {?} experimentIds
         * @return {?}
         */
        fetchTimeSeriesForCards(fetchInfos, experimentIds) {
            /**
             * TODO(psybuzz): if 2 cards require the same data, we should dedupe instead of
             * making 2 identical requests.
             * @type {?}
             */
            const requests = fetchInfos.map((/**
             * @param {?} fetchInfo
             * @return {?}
             */
            (fetchInfo) => {
                const { plugin, tag, runId, sample } = fetchInfo;
                return isSingleRunPlugin(plugin)
                    ? { plugin, tag, sample, runId: (/** @type {?} */ (runId)) }
                    : { plugin, tag, sample, experimentIds };
            }));
            // Fetch and handle responses.
            return of(requests).pipe(tap((/**
             * @param {?} requests
             * @return {?}
             */
            (requests) => {
                this.store.dispatch(multipleTimeSeriesRequested({ requests }));
            })), mergeMap((/**
             * @param {?} requests
             * @return {?}
             */
            (requests) => {
                /** @type {?} */
                const observables = requests.map((/**
                 * @param {?} request
                 * @return {?}
                 */
                (request) => this.fetchTimeSeries(request)));
                return merge(...observables);
            })));
        }
    }
    MetricsEffects.decorators = [
        { type: Injectable },
    ];
    /** @nocollapse */
    MetricsEffects.ctorParameters = () => [
        { type: Actions },
        { type: Store },
        { type: MetricsDataSource }
    ];
    /** @nocollapse */ MetricsEffects.ɵfac = function MetricsEffects_Factory(t) { return new (t || MetricsEffects)(ɵɵinject(Actions), ɵɵinject(Store), ɵɵinject(MetricsDataSource)); };
    /** @nocollapse */ MetricsEffects.ɵprov = ɵɵdefineInjectable({ token: MetricsEffects, factory: MetricsEffects.ɵfac });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/views/card_renderer/card_lazy_loader.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const elementToCardIdMap = new WeakMap();
    class CardObserver {
        /**
         * Buffer determines how far a card can be, beyond the root's bounding rect,
         * and still be loaded. It corresponds to an IntersectionObserver's
         * 'rootMargin'. For example, "50px 0 100px 0"' will treat observed elements
         * as 'intersecting' when they come within 50px of the root top or within
         * 100px of the root's bottom. Adding buffer allows nearby, offscreen cards
         * to load, preventing blank cards from being seen too often.
         *
         * If positive 'rootMargin' is provided, a scrollable 'root' is required.
         *
         * https://w3c.github.io/IntersectionObserver/#dom-intersectionobserverinit-rootmargin
         * https://developer.mozilla.org/en-US/docs/Web/API/Intersection_Observer_API
         * @param {?=} root
         * @param {?=} buffer
         */
        constructor(root, buffer) {
            this.root = root;
            this.buffer = buffer;
            this.destroyedTargets = new Set();
        }
        /**
         * @param {?} intersectionCallback
         * @return {?}
         */
        initialize(intersectionCallback) {
            if (this.intersectionObserver) {
                return;
            }
            this.intersectionCallback = intersectionCallback;
            this.intersectionObserver = new IntersectionObserver(this.onCardIntersection.bind(this), { threshold: 0, root: this.root, rootMargin: this.buffer });
        }
        /**
         * @param {?} target
         * @return {?}
         */
        add(target) {
            if (this.ensureInitialized()) {
                (/** @type {?} */ (this.intersectionObserver)).observe(target);
            }
        }
        /**
         * Adds a target to a list of elements to-be-destroyed, so that we can notify
         * listeners that they become 'exitedCards' before unobserving it.
         * @param {?} target
         * @return {?}
         */
        willDestroy(target) {
            if (this.ensureInitialized()) {
                this.destroyedTargets.add(target);
            }
        }
        /**
         * @private
         * @return {?}
         */
        ensureInitialized() {
            if (!this.intersectionObserver) {
                throw new Error('CardObserver must be initialized before use');
            }
            return true;
        }
        /**
         * @private
         * @param {?} entries
         * @return {?}
         */
        onCardIntersection(entries) {
            /**
             * Within a single callback firing, `entries` may include separate entries
             * representing the same target element entering (isIntersecting) and
             * leaving (!isIntersecting). To account for this, we sort entries by
             * increasing timestamp and respect the latest one.
             */
            entries.sort((/**
             * @param {?} a
             * @param {?} b
             * @return {?}
             */
            (a, b) => a.time - b.time));
            /** @type {?} */
            const enteredCards = new Set();
            /** @type {?} */
            const exitedCards = new Set();
            for (const entry of entries) {
                /** @type {?} */
                const target = entry.target;
                /** @type {?} */
                const cardId = elementToCardIdMap.get(target);
                if (!cardId) {
                    throw new Error('A CardObserver element must be associated with a CardId');
                }
                if (entry.isIntersecting) {
                    enteredCards.add(cardId);
                    exitedCards.delete(cardId);
                }
                else {
                    enteredCards.delete(cardId);
                    exitedCards.add(cardId);
                }
                /**
                 * Cleanup destroyed targets. Defend against speculative case when
                 * - A enters viewport
                 * - B added to destroyed targets
                 * - Callback fires for just A, unobserving B
                 * - B's callback never fires
                 */
                if (this.destroyedTargets.has(target) && !entry.isIntersecting) {
                    this.destroyedTargets.delete(target);
                    (/** @type {?} */ (this.intersectionObserver)).unobserve(target);
                }
            }
            (/** @type {?} */ (this.intersectionCallback))(enteredCards, exitedCards);
        }
        /**
         * @param {?} entries
         * @return {?}
         */
        onCardIntersectionForTest(entries) {
            this.onCardIntersection(entries);
        }
    }
    /**
     * A directive applied to elements that represent a card container. When the
     * element is ready to be loaded, this is responsible for marking cardId as
     * visible.
     *
     * Card container:
     *
     * <div [cardLazyLoader]="card1"></div>
     *
     * Card container that can load within 100px of a scrollable element's bounding
     * box:
     *
     * <div
     *   [cardLazyLoader]="card1"
     *   [cardObserver]="new CardObserver(scrollableElement, '100px')"
     * ></div>
     */
    class CardLazyLoader {
        /**
         * @param {?} host
         * @param {?} store
         */
        constructor(host, store) {
            this.host = host;
            this.store = store;
        }
        /**
         * @param {?} enteredCards
         * @param {?} exitedCards
         * @return {?}
         */
        onCardIntersection(enteredCards, exitedCards) {
            this.store.dispatch(cardVisibilityChanged({ enteredCards, exitedCards }));
        }
        /**
         * @return {?}
         */
        ngOnInit() {
            elementToCardIdMap.set(this.host.nativeElement, this.cardId);
            if (!this.cardObserver) {
                this.cardObserver = new CardObserver();
            }
            this.cardObserver.initialize(this.onCardIntersection.bind(this));
            this.cardObserver.add(this.host.nativeElement);
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            if (this.cardObserver) {
                this.cardObserver.willDestroy(this.host.nativeElement);
            }
        }
        /**
         * @return {?}
         */
        hostForTest() {
            return this.host;
        }
    }
    CardLazyLoader.decorators = [
        { type: Directive, args: [{
                    selector: '[cardLazyLoader]',
                },] },
    ];
    /** @nocollapse */
    CardLazyLoader.ctorParameters = () => [
        { type: ElementRef },
        { type: Store }
    ];
    CardLazyLoader.propDecorators = {
        cardId: [{ type: Input, args: ['cardLazyLoader',] }],
        cardObserver: [{ type: Input }]
    };
    /** @nocollapse */ CardLazyLoader.ɵfac = function CardLazyLoader_Factory(t) { return new (t || CardLazyLoader)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ CardLazyLoader.ɵdir = ɵɵdefineDirective({ type: CardLazyLoader, selectors: [["", "cardLazyLoader", ""]], inputs: { cardId: ["cardLazyLoader", "cardId"], cardObserver: "cardObserver" } });

    function groupCardIdWithMetdata(cards) {
        const tagPrefix = new Map();
        const sortedCards = cards.slice().sort((cardA, cardB) => {
            return compareTagNames(cardA.tag, cardB.tag);
        });
        for (const card of sortedCards) {
            const groupName = getTagGroupName(card.tag);
            if (!tagPrefix.has(groupName)) {
                tagPrefix.set(groupName, { groupName, items: [] });
            }
            tagPrefix.get(groupName).items.push(card);
        }
        return [...tagPrefix.values()];
    }
    function getTagGroupName(tag) {
        return tag.split('/', 1)[0];
    }
    const IGNORED_TAG_SUFFIX = '/scalar_summary';
    function getTagDisplayName(tag, groupName) {
        // Remove group name prefix.
        let result = tag;
        if (groupName && tag.startsWith(groupName + '/')) {
            result = tag.slice(groupName.length + 1);
        }
        // Remove the V1 scalar summary suffix. See traditional TB logic in
        // tensorboard/plugins/scalar/tf_scalar_dashboard/tf-scalar-dashboard.html
        if (result.endsWith(IGNORED_TAG_SUFFIX)) {
            result = result.slice(0, -IGNORED_TAG_SUFFIX.length);
        }
        return result || tag;
    }
    // TODO(b/154055328): combine this with the OSS ts_library compat version.
    // Adopted from tensorboard/components/vz_sorting/sorting.js
    // Delta:
    // - better typing
    // - human readable variable names
    // - removed componentization by "_".
    /**
     * Compares tag names asciinumerically broken into components.
     *
     * Unlike the standard asciibetical comparator, this function knows that 'a10b'
     * > 'a2b'. Fixed point and engineering notation are supported. This function
     * also splits the input by slash to perform array comparison. Therefore it
     * knows that 'a/a' < 'a+/a' even though '+' < '/' in the ASCII table.
     */
    function compareTagNames(tagA, tagB) {
        let aIndex = 0;
        let bIndex = 0;
        while (true) {
            if (aIndex === tagA.length) {
                return bIndex === tagB.length ? 0 : -1;
            }
            if (bIndex === tagB.length) {
                return 1;
            }
            if (isDigit(tagA[aIndex]) && isDigit(tagB[bIndex])) {
                const aNumberStart = aIndex;
                const bNumberStart = bIndex;
                aIndex = consumeNumber(tagA, aIndex + 1);
                bIndex = consumeNumber(tagB, bIndex + 1);
                const an = Number(tagA.slice(aNumberStart, aIndex));
                const bn = Number(tagB.slice(bNumberStart, bIndex));
                if (an < bn) {
                    return -1;
                }
                if (an > bn) {
                    return 1;
                }
                continue;
            }
            if (isBreak(tagA[aIndex])) {
                if (!isBreak(tagB[bIndex])) {
                    return -1;
                }
            }
            else if (isBreak(tagB[bIndex])) {
                return 1;
            }
            else if (tagA[aIndex] < tagB[bIndex]) {
                return -1;
            }
            else if (tagA[aIndex] > tagB[bIndex]) {
                return 1;
            }
            aIndex++;
            bIndex++;
        }
    }
    /**
     * Returns endIndex of a number sequence in string starting from startIndex.
     *
     * The method can handle scientific notation, real and natural numbers, and
     * numbers with exponents. Do note that it does not treat decimals that start
     * with "." as a real number.
     */
    function consumeNumber(s, startIndex) {
        let State;
        (function (State) {
            State[State["NATURAL"] = 0] = "NATURAL";
            State[State["REAL"] = 1] = "REAL";
            State[State["EXPONENT_SIGN"] = 2] = "EXPONENT_SIGN";
            State[State["EXPONENT"] = 3] = "EXPONENT";
        })(State || (State = {}));
        let state = State.NATURAL;
        let i = startIndex;
        for (; i < s.length; i++) {
            if (state === State.NATURAL) {
                if (s[i] === '.') {
                    state = State.REAL;
                }
                else if (s[i] === 'e' || s[i] === 'E') {
                    state = State.EXPONENT_SIGN;
                }
                else if (!isDigit(s[i])) {
                    break;
                }
            }
            else if (state === State.REAL) {
                if (s[i] === 'e' || s[i] === 'E') {
                    state = State.EXPONENT_SIGN;
                }
                else if (!isDigit(s[i])) {
                    break;
                }
            }
            else if (state === State.EXPONENT_SIGN) {
                if (isDigit(s[i]) || s[i] === '+' || s[i] === '-') {
                    state = State.EXPONENT;
                }
                else {
                    break;
                }
            }
            else if (state === State.EXPONENT) {
                if (!isDigit(s[i])) {
                    break;
                }
            }
        }
        return i;
    }
    function isDigit(character) {
        return '0' <= character && character <= '9';
    }
    function isBreak(character) {
        return character === '/' || isDigit(character);
    }

    const _c0$r = ["panel"];
    function MatAutocomplete_ng_template_0_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 0, 1);
        ɵɵprojection(2);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r0 = ɵɵnextContext();
        ɵɵproperty("id", ctx_r0.id)("ngClass", ctx_r0._classList);
    } }
    const _c1$h = ["*"];
    let _uniqueAutocompleteIdCounter = 0;
    /**
     * Event object that is emitted when an autocomplete option is selected.
     */
    class MatAutocompleteSelectedEvent {
        /**
         * @param {?} source
         * @param {?} option
         */
        constructor(source, option) {
            this.source = source;
            this.option = option;
        }
    }
    // Boilerplate for applying mixins to MatAutocomplete.
    /**
     * \@docs-private
     */
    class MatAutocompleteBase {
    }
    /** @type {?} */
    const _MatAutocompleteMixinBase = mixinDisableRipple(MatAutocompleteBase);
    /**
     * Injection token to be used to override the default options for `mat-autocomplete`.
     * @type {?}
     */
    const MAT_AUTOCOMPLETE_DEFAULT_OPTIONS = new InjectionToken('mat-autocomplete-default-options', {
        providedIn: 'root',
        factory: MAT_AUTOCOMPLETE_DEFAULT_OPTIONS_FACTORY,
    });
    /**
     * \@docs-private
     * @return {?}
     */
    function MAT_AUTOCOMPLETE_DEFAULT_OPTIONS_FACTORY() {
        return { autoActiveFirstOption: false };
    }
    class MatAutocomplete extends _MatAutocompleteMixinBase {
        /**
         * @param {?} _changeDetectorRef
         * @param {?} _elementRef
         * @param {?} defaults
         */
        constructor(_changeDetectorRef, _elementRef, defaults) {
            super();
            this._changeDetectorRef = _changeDetectorRef;
            this._elementRef = _elementRef;
            this._activeOptionChanges = Subscription.EMPTY;
            /**
             * Whether the autocomplete panel should be visible, depending on option length.
             */
            this.showPanel = false;
            this._isOpen = false;
            /**
             * Function that maps an option's control value to its display value in the trigger.
             */
            this.displayWith = null;
            /**
             * Event that is emitted whenever an option from the list is selected.
             */
            this.optionSelected = new EventEmitter();
            /**
             * Event that is emitted when the autocomplete panel is opened.
             */
            this.opened = new EventEmitter();
            /**
             * Event that is emitted when the autocomplete panel is closed.
             */
            this.closed = new EventEmitter();
            /**
             * Emits whenever an option is activated using the keyboard.
             */
            this.optionActivated = new EventEmitter();
            this._classList = {};
            /**
             * Unique ID to be used by autocomplete trigger's "aria-owns" property.
             */
            this.id = `mat-autocomplete-${_uniqueAutocompleteIdCounter++}`;
            this._autoActiveFirstOption = !!defaults.autoActiveFirstOption;
        }
        /**
         * Whether the autocomplete panel is open.
         * @return {?}
         */
        get isOpen() { return this._isOpen && this.showPanel; }
        /**
         * Whether the first option should be highlighted when the autocomplete panel is opened.
         * Can be configured globally through the `MAT_AUTOCOMPLETE_DEFAULT_OPTIONS` token.
         * @return {?}
         */
        get autoActiveFirstOption() { return this._autoActiveFirstOption; }
        /**
         * @param {?} value
         * @return {?}
         */
        set autoActiveFirstOption(value) {
            this._autoActiveFirstOption = coerceBooleanProperty(value);
        }
        /**
         * Takes classes set on the host mat-autocomplete element and applies them to the panel
         * inside the overlay container to allow for easy styling.
         * @param {?} value
         * @return {?}
         */
        set classList(value) {
            if (value && value.length) {
                this._classList = value.split(' ').reduce((/**
                 * @param {?} classList
                 * @param {?} className
                 * @return {?}
                 */
                (classList, className) => {
                    classList[className.trim()] = true;
                    return classList;
                }), (/** @type {?} */ ({})));
            }
            else {
                this._classList = {};
            }
            this._setVisibilityClasses(this._classList);
            this._elementRef.nativeElement.className = '';
        }
        /**
         * @return {?}
         */
        ngAfterContentInit() {
            this._keyManager = new ActiveDescendantKeyManager(this.options).withWrap();
            this._activeOptionChanges = this._keyManager.change.subscribe((/**
             * @param {?} index
             * @return {?}
             */
            index => {
                this.optionActivated.emit({ source: this, option: this.options.toArray()[index] || null });
            }));
            // Set the initial visibility state.
            this._setVisibility();
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this._activeOptionChanges.unsubscribe();
        }
        /**
         * Sets the panel scrollTop. This allows us to manually scroll to display options
         * above or below the fold, as they are not actually being focused when active.
         * @param {?} scrollTop
         * @return {?}
         */
        _setScrollTop(scrollTop) {
            if (this.panel) {
                this.panel.nativeElement.scrollTop = scrollTop;
            }
        }
        /**
         * Returns the panel's scrollTop.
         * @return {?}
         */
        _getScrollTop() {
            return this.panel ? this.panel.nativeElement.scrollTop : 0;
        }
        /**
         * Panel should hide itself when the option list is empty.
         * @return {?}
         */
        _setVisibility() {
            this.showPanel = !!this.options.length;
            this._setVisibilityClasses(this._classList);
            this._changeDetectorRef.markForCheck();
        }
        /**
         * Emits the `select` event.
         * @param {?} option
         * @return {?}
         */
        _emitSelectEvent(option) {
            /** @type {?} */
            const event = new MatAutocompleteSelectedEvent(this, option);
            this.optionSelected.emit(event);
        }
        /**
         * Sets the autocomplete visibility classes on a classlist based on the panel is visible.
         * @private
         * @param {?} classList
         * @return {?}
         */
        _setVisibilityClasses(classList) {
            classList['mat-autocomplete-visible'] = this.showPanel;
            classList['mat-autocomplete-hidden'] = !this.showPanel;
        }
    }
    MatAutocomplete.ɵfac = function MatAutocomplete_Factory(t) { return new (t || MatAutocomplete)(ɵɵdirectiveInject(ChangeDetectorRef), ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(MAT_AUTOCOMPLETE_DEFAULT_OPTIONS)); };
    MatAutocomplete.ɵcmp = ɵɵdefineComponent({ type: MatAutocomplete, selectors: [["mat-autocomplete"]], contentQueries: function MatAutocomplete_ContentQueries(rf, ctx, dirIndex) { if (rf & 1) {
            ɵɵcontentQuery(dirIndex, MatOption, true);
            ɵɵcontentQuery(dirIndex, MatOptgroup, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.options = _t);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.optionGroups = _t);
        } }, viewQuery: function MatAutocomplete_Query(rf, ctx) { if (rf & 1) {
            ɵɵstaticViewQuery(TemplateRef, true);
            ɵɵviewQuery(_c0$r, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.template = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.panel = _t.first);
        } }, hostAttrs: [1, "mat-autocomplete"], inputs: { disableRipple: "disableRipple", displayWith: "displayWith", autoActiveFirstOption: "autoActiveFirstOption", classList: ["class", "classList"], panelWidth: "panelWidth" }, outputs: { optionSelected: "optionSelected", opened: "opened", closed: "closed", optionActivated: "optionActivated" }, exportAs: ["matAutocomplete"], features: [ɵɵProvidersFeature([
                { provide: MAT_OPTION_PARENT_COMPONENT, useExisting: MatAutocomplete }
            ]), ɵɵInheritDefinitionFeature], ngContentSelectors: _c1$h, decls: 1, vars: 0, consts: [["role", "listbox", 1, "mat-autocomplete-panel", 3, "id", "ngClass"], ["panel", ""]], template: function MatAutocomplete_Template(rf, ctx) { if (rf & 1) {
            ɵɵprojectionDef();
            ɵɵtemplate(0, MatAutocomplete_ng_template_0_Template, 3, 2, "ng-template");
        } }, directives: [NgClass], styles: [".mat-autocomplete-panel{min-width:112px;max-width:280px;overflow:auto;-webkit-overflow-scrolling:touch;visibility:hidden;max-width:none;max-height:256px;position:relative;width:100%;border-bottom-left-radius:4px;border-bottom-right-radius:4px}.mat-autocomplete-panel.mat-autocomplete-visible{visibility:visible}.mat-autocomplete-panel.mat-autocomplete-hidden{visibility:hidden}.mat-autocomplete-panel-above .mat-autocomplete-panel{border-radius:0;border-top-left-radius:4px;border-top-right-radius:4px}.mat-autocomplete-panel .mat-divider-horizontal{margin-top:-1px}.cdk-high-contrast-active .mat-autocomplete-panel{outline:solid 1px}\n"], encapsulation: 2, changeDetection: 0 });
    /** @nocollapse */
    MatAutocomplete.ctorParameters = () => [
        { type: ChangeDetectorRef },
        { type: ElementRef },
        { type: undefined, decorators: [{ type: Inject, args: [MAT_AUTOCOMPLETE_DEFAULT_OPTIONS,] }] }
    ];
    MatAutocomplete.propDecorators = {
        template: [{ type: ViewChild, args: [TemplateRef, { static: true },] }],
        panel: [{ type: ViewChild, args: ['panel',] }],
        options: [{ type: ContentChildren, args: [MatOption, { descendants: true },] }],
        optionGroups: [{ type: ContentChildren, args: [MatOptgroup, { descendants: true },] }],
        displayWith: [{ type: Input }],
        autoActiveFirstOption: [{ type: Input }],
        panelWidth: [{ type: Input }],
        optionSelected: [{ type: Output }],
        opened: [{ type: Output }],
        closed: [{ type: Output }],
        optionActivated: [{ type: Output }],
        classList: [{ type: Input, args: ['class',] }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/autocomplete/autocomplete-origin.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Directive applied to an element to make it usable
     * as a connection point for an autocomplete panel.
     */
    class MatAutocompleteOrigin {
        /**
         * @param {?} elementRef
         */
        constructor(elementRef) {
            this.elementRef = elementRef;
        }
    }
    MatAutocompleteOrigin.ɵfac = function MatAutocompleteOrigin_Factory(t) { return new (t || MatAutocompleteOrigin)(ɵɵdirectiveInject(ElementRef)); };
    MatAutocompleteOrigin.ɵdir = ɵɵdefineDirective({ type: MatAutocompleteOrigin, selectors: [["", "matAutocompleteOrigin", ""]], exportAs: ["matAutocompleteOrigin"] });
    /** @nocollapse */
    MatAutocompleteOrigin.ctorParameters = () => [
        { type: ElementRef }
    ];

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/autocomplete/autocomplete-trigger.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * The height of each autocomplete option.
     * @type {?}
     */
    const AUTOCOMPLETE_OPTION_HEIGHT = 48;
    /**
     * The total height of the autocomplete panel.
     * @type {?}
     */
    const AUTOCOMPLETE_PANEL_HEIGHT = 256;
    /**
     * Injection token that determines the scroll handling while the autocomplete panel is open.
     * @type {?}
     */
    const MAT_AUTOCOMPLETE_SCROLL_STRATEGY = new InjectionToken('mat-autocomplete-scroll-strategy');
    /**
     * \@docs-private
     * @param {?} overlay
     * @return {?}
     */
    function MAT_AUTOCOMPLETE_SCROLL_STRATEGY_FACTORY(overlay) {
        return (/**
         * @return {?}
         */
        () => overlay.scrollStrategies.reposition());
    }
    /**
     * \@docs-private
     * @type {?}
     */
    const MAT_AUTOCOMPLETE_SCROLL_STRATEGY_FACTORY_PROVIDER = {
        provide: MAT_AUTOCOMPLETE_SCROLL_STRATEGY,
        deps: [Overlay],
        useFactory: MAT_AUTOCOMPLETE_SCROLL_STRATEGY_FACTORY,
    };
    /**
     * Provider that allows the autocomplete to register as a ControlValueAccessor.
     * \@docs-private
     * @type {?}
     */
    const MAT_AUTOCOMPLETE_VALUE_ACCESSOR = {
        provide: NG_VALUE_ACCESSOR,
        useExisting: forwardRef((/**
         * @return {?}
         */
        () => MatAutocompleteTrigger)),
        multi: true
    };
    /**
     * Creates an error to be thrown when attempting to use an autocomplete trigger without a panel.
     * \@docs-private
     * @return {?}
     */
    function getMatAutocompleteMissingPanelError() {
        return Error('Attempting to open an undefined instance of `mat-autocomplete`. ' +
            'Make sure that the id passed to the `matAutocomplete` is correct and that ' +
            'you\'re attempting to open it after the ngAfterContentInit hook.');
    }
    class MatAutocompleteTrigger {
        /**
         * @param {?} _element
         * @param {?} _overlay
         * @param {?} _viewContainerRef
         * @param {?} _zone
         * @param {?} _changeDetectorRef
         * @param {?} scrollStrategy
         * @param {?} _dir
         * @param {?} _formField
         * @param {?} _document
         * @param {?=} _viewportRuler
         */
        constructor(_element, _overlay, _viewContainerRef, _zone, _changeDetectorRef, scrollStrategy, _dir, _formField, _document, _viewportRuler) {
            this._element = _element;
            this._overlay = _overlay;
            this._viewContainerRef = _viewContainerRef;
            this._zone = _zone;
            this._changeDetectorRef = _changeDetectorRef;
            this._dir = _dir;
            this._formField = _formField;
            this._document = _document;
            this._viewportRuler = _viewportRuler;
            this._componentDestroyed = false;
            this._autocompleteDisabled = false;
            /**
             * Whether or not the label state is being overridden.
             */
            this._manuallyFloatingLabel = false;
            /**
             * Subscription to viewport size changes.
             */
            this._viewportSubscription = Subscription.EMPTY;
            /**
             * Whether the autocomplete can open the next time it is focused. Used to prevent a focused,
             * closed autocomplete from being reopened if the user switches to another browser tab and then
             * comes back.
             */
            this._canOpenOnNextFocus = true;
            /**
             * Stream of keyboard events that can close the panel.
             */
            this._closeKeyEventStream = new Subject();
            /**
             * Event handler for when the window is blurred. Needs to be an
             * arrow function in order to preserve the context.
             */
            this._windowBlurHandler = (/**
             * @return {?}
             */
            () => {
                // If the user blurred the window while the autocomplete is focused, it means that it'll be
                // refocused when they come back. In this case we want to skip the first focus event, if the
                // pane was closed, in order to avoid reopening it unintentionally.
                this._canOpenOnNextFocus =
                    this._document.activeElement !== this._element.nativeElement || this.panelOpen;
            });
            /**
             * `View -> model callback called when value changes`
             */
            this._onChange = (/**
             * @return {?}
             */
            () => { });
            /**
             * `View -> model callback called when autocomplete has been touched`
             */
            this._onTouched = (/**
             * @return {?}
             */
            () => { });
            /**
             * Position of the autocomplete panel relative to the trigger element. A position of `auto`
             * will render the panel underneath the trigger if there is enough space for it to fit in
             * the viewport, otherwise the panel will be shown above it. If the position is set to
             * `above` or `below`, the panel will always be shown above or below the trigger. no matter
             * whether it fits completely in the viewport.
             */
            this.position = 'auto';
            /**
             * `autocomplete` attribute to be set on the input element.
             * \@docs-private
             */
            this.autocompleteAttribute = 'off';
            this._overlayAttached = false;
            /**
             * Stream of autocomplete option selections.
             */
            this.optionSelections = (/** @type {?} */ (defer((/**
             * @return {?}
             */
            () => {
                if (this.autocomplete && this.autocomplete.options) {
                    return merge(...this.autocomplete.options.map((/**
                     * @param {?} option
                     * @return {?}
                     */
                    option => option.onSelectionChange)));
                }
                // If there are any subscribers before `ngAfterViewInit`, the `autocomplete` will be undefined.
                // Return a stream that we'll replace with the real one once everything is in place.
                return this._zone.onStable
                    .asObservable()
                    .pipe(take(1), switchMap((/**
                 * @return {?}
                 */
                () => this.optionSelections)));
            }))));
            this._scrollStrategy = scrollStrategy;
        }
        /**
         * Whether the autocomplete is disabled. When disabled, the element will
         * act as a regular input and the user won't be able to open the panel.
         * @return {?}
         */
        get autocompleteDisabled() { return this._autocompleteDisabled; }
        /**
         * @param {?} value
         * @return {?}
         */
        set autocompleteDisabled(value) {
            this._autocompleteDisabled = coerceBooleanProperty(value);
        }
        /**
         * @return {?}
         */
        ngAfterViewInit() {
            /** @type {?} */
            const window = this._getWindow();
            if (typeof window !== 'undefined') {
                this._zone.runOutsideAngular((/**
                 * @return {?}
                 */
                () => window.addEventListener('blur', this._windowBlurHandler)));
            }
        }
        /**
         * @param {?} changes
         * @return {?}
         */
        ngOnChanges(changes) {
            if (changes['position'] && this._positionStrategy) {
                this._setStrategyPositions(this._positionStrategy);
                if (this.panelOpen) {
                    (/** @type {?} */ (this._overlayRef)).updatePosition();
                }
            }
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            /** @type {?} */
            const window = this._getWindow();
            if (typeof window !== 'undefined') {
                window.removeEventListener('blur', this._windowBlurHandler);
            }
            this._viewportSubscription.unsubscribe();
            this._componentDestroyed = true;
            this._destroyPanel();
            this._closeKeyEventStream.complete();
        }
        /**
         * Whether or not the autocomplete panel is open.
         * @return {?}
         */
        get panelOpen() {
            return this._overlayAttached && this.autocomplete.showPanel;
        }
        /**
         * Opens the autocomplete suggestion panel.
         * @return {?}
         */
        openPanel() {
            this._attachOverlay();
            this._floatLabel();
        }
        /**
         * Closes the autocomplete suggestion panel.
         * @return {?}
         */
        closePanel() {
            this._resetLabel();
            if (!this._overlayAttached) {
                return;
            }
            if (this.panelOpen) {
                // Only emit if the panel was visible.
                this.autocomplete.closed.emit();
            }
            this.autocomplete._isOpen = this._overlayAttached = false;
            if (this._overlayRef && this._overlayRef.hasAttached()) {
                this._overlayRef.detach();
                this._closingActionsSubscription.unsubscribe();
            }
            // Note that in some cases this can end up being called after the component is destroyed.
            // Add a check to ensure that we don't try to run change detection on a destroyed view.
            if (!this._componentDestroyed) {
                // We need to trigger change detection manually, because
                // `fromEvent` doesn't seem to do it at the proper time.
                // This ensures that the label is reset when the
                // user clicks outside.
                this._changeDetectorRef.detectChanges();
            }
        }
        /**
         * Updates the position of the autocomplete suggestion panel to ensure that it fits all options
         * within the viewport.
         * @return {?}
         */
        updatePosition() {
            if (this._overlayAttached) {
                (/** @type {?} */ (this._overlayRef)).updatePosition();
            }
        }
        /**
         * A stream of actions that should close the autocomplete panel, including
         * when an option is selected, on blur, and when TAB is pressed.
         * @return {?}
         */
        get panelClosingActions() {
            return merge(this.optionSelections, this.autocomplete._keyManager.tabOut.pipe(filter((/**
             * @return {?}
             */
            () => this._overlayAttached))), this._closeKeyEventStream, this._getOutsideClickStream(), this._overlayRef ?
                this._overlayRef.detachments().pipe(filter((/**
                 * @return {?}
                 */
                () => this._overlayAttached))) :
                of()).pipe(
            // Normalize the output so we return a consistent type.
            map((/**
             * @param {?} event
             * @return {?}
             */
            event => event instanceof MatOptionSelectionChange ? event : null)));
        }
        /**
         * The currently active option, coerced to MatOption type.
         * @return {?}
         */
        get activeOption() {
            if (this.autocomplete && this.autocomplete._keyManager) {
                return this.autocomplete._keyManager.activeItem;
            }
            return null;
        }
        /**
         * Stream of clicks outside of the autocomplete panel.
         * @private
         * @return {?}
         */
        _getOutsideClickStream() {
            return merge((/** @type {?} */ (fromEvent(this._document, 'click'))), (/** @type {?} */ (fromEvent(this._document, 'touchend'))))
                .pipe(filter((/**
             * @param {?} event
             * @return {?}
             */
            event => {
                // If we're in the Shadow DOM, the event target will be the shadow root, so we have to
                // fall back to check the first element in the path of the click event.
                /** @type {?} */
                const clickTarget = (/** @type {?} */ ((this._isInsideShadowRoot && event.composedPath ? event.composedPath()[0] :
                    event.target)));
                /** @type {?} */
                const formField = this._formField ? this._formField._elementRef.nativeElement : null;
                return this._overlayAttached && clickTarget !== this._element.nativeElement &&
                    (!formField || !formField.contains(clickTarget)) &&
                    (!!this._overlayRef && !this._overlayRef.overlayElement.contains(clickTarget));
            })));
        }
        // Implemented as part of ControlValueAccessor.
        /**
         * @param {?} value
         * @return {?}
         */
        writeValue(value) {
            Promise.resolve(null).then((/**
             * @return {?}
             */
            () => this._setTriggerValue(value)));
        }
        // Implemented as part of ControlValueAccessor.
        /**
         * @param {?} fn
         * @return {?}
         */
        registerOnChange(fn) {
            this._onChange = fn;
        }
        // Implemented as part of ControlValueAccessor.
        /**
         * @param {?} fn
         * @return {?}
         */
        registerOnTouched(fn) {
            this._onTouched = fn;
        }
        // Implemented as part of ControlValueAccessor.
        /**
         * @param {?} isDisabled
         * @return {?}
         */
        setDisabledState(isDisabled) {
            this._element.nativeElement.disabled = isDisabled;
        }
        /**
         * @param {?} event
         * @return {?}
         */
        _handleKeydown(event) {
            /** @type {?} */
            const keyCode = event.keyCode;
            // Prevent the default action on all escape key presses. This is here primarily to bring IE
            // in line with other browsers. By default, pressing escape on IE will cause it to revert
            // the input value to the one that it had on focus, however it won't dispatch any events
            // which means that the model value will be out of sync with the view.
            if (keyCode === ESCAPE) {
                event.preventDefault();
            }
            if (this.activeOption && keyCode === ENTER && this.panelOpen) {
                this.activeOption._selectViaInteraction();
                this._resetActiveItem();
                event.preventDefault();
            }
            else if (this.autocomplete) {
                /** @type {?} */
                const prevActiveItem = this.autocomplete._keyManager.activeItem;
                /** @type {?} */
                const isArrowKey = keyCode === UP_ARROW || keyCode === DOWN_ARROW;
                if (this.panelOpen || keyCode === TAB) {
                    this.autocomplete._keyManager.onKeydown(event);
                }
                else if (isArrowKey && this._canOpen()) {
                    this.openPanel();
                }
                if (isArrowKey || this.autocomplete._keyManager.activeItem !== prevActiveItem) {
                    this._scrollToOption();
                }
            }
        }
        /**
         * @param {?} event
         * @return {?}
         */
        _handleInput(event) {
            /** @type {?} */
            let target = (/** @type {?} */ (event.target));
            /** @type {?} */
            let value = target.value;
            // Based on `NumberValueAccessor` from forms.
            if (target.type === 'number') {
                value = value == '' ? null : parseFloat(value);
            }
            // If the input has a placeholder, IE will fire the `input` event on page load,
            // focus and blur, in addition to when the user actually changed the value. To
            // filter out all of the extra events, we save the value on focus and between
            // `input` events, and we check whether it changed.
            // See: https://connect.microsoft.com/IE/feedback/details/885747/
            if (this._previousValue !== value) {
                this._previousValue = value;
                this._onChange(value);
                if (this._canOpen() && this._document.activeElement === event.target) {
                    this.openPanel();
                }
            }
        }
        /**
         * @return {?}
         */
        _handleFocus() {
            if (!this._canOpenOnNextFocus) {
                this._canOpenOnNextFocus = true;
            }
            else if (this._canOpen()) {
                this._previousValue = this._element.nativeElement.value;
                this._attachOverlay();
                this._floatLabel(true);
            }
        }
        /**
         * In "auto" mode, the label will animate down as soon as focus is lost.
         * This causes the value to jump when selecting an option with the mouse.
         * This method manually floats the label until the panel can be closed.
         * @private
         * @param {?=} shouldAnimate Whether the label should be animated when it is floated.
         * @return {?}
         */
        _floatLabel(shouldAnimate = false) {
            if (this._formField && this._formField.floatLabel === 'auto') {
                if (shouldAnimate) {
                    this._formField._animateAndLockLabel();
                }
                else {
                    this._formField.floatLabel = 'always';
                }
                this._manuallyFloatingLabel = true;
            }
        }
        /**
         * If the label has been manually elevated, return it to its normal state.
         * @private
         * @return {?}
         */
        _resetLabel() {
            if (this._manuallyFloatingLabel) {
                this._formField.floatLabel = 'auto';
                this._manuallyFloatingLabel = false;
            }
        }
        /**
         * Given that we are not actually focusing active options, we must manually adjust scroll
         * to reveal options below the fold. First, we find the offset of the option from the top
         * of the panel. If that offset is below the fold, the new scrollTop will be the offset -
         * the panel height + the option height, so the active option will be just visible at the
         * bottom of the panel. If that offset is above the top of the visible panel, the new scrollTop
         * will become the offset. If that offset is visible within the panel already, the scrollTop is
         * not adjusted.
         * @private
         * @return {?}
         */
        _scrollToOption() {
            /** @type {?} */
            const index = this.autocomplete._keyManager.activeItemIndex || 0;
            /** @type {?} */
            const labelCount = _countGroupLabelsBeforeOption(index, this.autocomplete.options, this.autocomplete.optionGroups);
            if (index === 0 && labelCount === 1) {
                // If we've got one group label before the option and we're at the top option,
                // scroll the list to the top. This is better UX than scrolling the list to the
                // top of the option, because it allows the user to read the top group's label.
                this.autocomplete._setScrollTop(0);
            }
            else {
                /** @type {?} */
                const newScrollPosition = _getOptionScrollPosition(index + labelCount, AUTOCOMPLETE_OPTION_HEIGHT, this.autocomplete._getScrollTop(), AUTOCOMPLETE_PANEL_HEIGHT);
                this.autocomplete._setScrollTop(newScrollPosition);
            }
        }
        /**
         * This method listens to a stream of panel closing actions and resets the
         * stream every time the option list changes.
         * @private
         * @return {?}
         */
        _subscribeToClosingActions() {
            /** @type {?} */
            const firstStable = this._zone.onStable.asObservable().pipe(take(1));
            /** @type {?} */
            const optionChanges = this.autocomplete.options.changes.pipe(tap((/**
             * @return {?}
             */
            () => this._positionStrategy.reapplyLastPosition())), 
            // Defer emitting to the stream until the next tick, because changing
            // bindings in here will cause "changed after checked" errors.
            delay(0));
            // When the zone is stable initially, and when the option list changes...
            return merge(firstStable, optionChanges)
                .pipe(
            // create a new stream of panelClosingActions, replacing any previous streams
            // that were created, and flatten it so our stream only emits closing events...
            switchMap((/**
             * @return {?}
             */
            () => {
                /** @type {?} */
                const wasOpen = this.panelOpen;
                this._resetActiveItem();
                this.autocomplete._setVisibility();
                if (this.panelOpen) {
                    (/** @type {?} */ (this._overlayRef)).updatePosition();
                    // If the `panelOpen` state changed, we need to make sure to emit the `opened`
                    // event, because we may not have emitted it when the panel was attached. This
                    // can happen if the users opens the panel and there are no options, but the
                    // options come in slightly later or as a result of the value changing.
                    if (wasOpen !== this.panelOpen) {
                        this.autocomplete.opened.emit();
                    }
                }
                return this.panelClosingActions;
            })), 
            // when the first closing event occurs...
            take(1))
                // set the value, close the panel, and complete.
                .subscribe((/**
             * @param {?} event
             * @return {?}
             */
            event => this._setValueAndClose(event)));
        }
        /**
         * Destroys the autocomplete suggestion panel.
         * @private
         * @return {?}
         */
        _destroyPanel() {
            if (this._overlayRef) {
                this.closePanel();
                this._overlayRef.dispose();
                this._overlayRef = null;
            }
        }
        /**
         * @private
         * @param {?} value
         * @return {?}
         */
        _setTriggerValue(value) {
            /** @type {?} */
            const toDisplay = this.autocomplete && this.autocomplete.displayWith ?
                this.autocomplete.displayWith(value) :
                value;
            // Simply falling back to an empty string if the display value is falsy does not work properly.
            // The display value can also be the number zero and shouldn't fall back to an empty string.
            /** @type {?} */
            const inputValue = toDisplay != null ? toDisplay : '';
            // If it's used within a `MatFormField`, we should set it through the property so it can go
            // through change detection.
            if (this._formField) {
                this._formField._control.value = inputValue;
            }
            else {
                this._element.nativeElement.value = inputValue;
            }
            this._previousValue = inputValue;
        }
        /**
         * This method closes the panel, and if a value is specified, also sets the associated
         * control to that value. It will also mark the control as dirty if this interaction
         * stemmed from the user.
         * @private
         * @param {?} event
         * @return {?}
         */
        _setValueAndClose(event) {
            if (event && event.source) {
                this._clearPreviousSelectedOption(event.source);
                this._setTriggerValue(event.source.value);
                this._onChange(event.source.value);
                this._element.nativeElement.focus();
                this.autocomplete._emitSelectEvent(event.source);
            }
            this.closePanel();
        }
        /**
         * Clear any previous selected option and emit a selection change event for this option
         * @private
         * @param {?} skip
         * @return {?}
         */
        _clearPreviousSelectedOption(skip) {
            this.autocomplete.options.forEach((/**
             * @param {?} option
             * @return {?}
             */
            option => {
                if (option != skip && option.selected) {
                    option.deselect();
                }
            }));
        }
        /**
         * @private
         * @return {?}
         */
        _attachOverlay() {
            if (!this.autocomplete) {
                throw getMatAutocompleteMissingPanelError();
            }
            // We want to resolve this once, as late as possible so that we can be
            // sure that the element has been moved into its final place in the DOM.
            if (this._isInsideShadowRoot == null) {
                this._isInsideShadowRoot = !!_getShadowRoot(this._element.nativeElement);
            }
            /** @type {?} */
            let overlayRef = this._overlayRef;
            if (!overlayRef) {
                this._portal = new TemplatePortal(this.autocomplete.template, this._viewContainerRef);
                overlayRef = this._overlay.create(this._getOverlayConfig());
                this._overlayRef = overlayRef;
                // Use the `keydownEvents` in order to take advantage of
                // the overlay event targeting provided by the CDK overlay.
                overlayRef.keydownEvents().subscribe((/**
                 * @param {?} event
                 * @return {?}
                 */
                event => {
                    // Close when pressing ESCAPE or ALT + UP_ARROW, based on the a11y guidelines.
                    // See: https://www.w3.org/TR/wai-aria-practices-1.1/#textbox-keyboard-interaction
                    if (event.keyCode === ESCAPE || (event.keyCode === UP_ARROW && event.altKey)) {
                        this._resetActiveItem();
                        this._closeKeyEventStream.next();
                        // We need to stop propagation, otherwise the event will eventually
                        // reach the input itself and cause the overlay to be reopened.
                        event.stopPropagation();
                        event.preventDefault();
                    }
                }));
                if (this._viewportRuler) {
                    this._viewportSubscription = this._viewportRuler.change().subscribe((/**
                     * @return {?}
                     */
                    () => {
                        if (this.panelOpen && overlayRef) {
                            overlayRef.updateSize({ width: this._getPanelWidth() });
                        }
                    }));
                }
            }
            else {
                // Update the trigger, panel width and direction, in case anything has changed.
                this._positionStrategy.setOrigin(this._getConnectedElement());
                overlayRef.updateSize({ width: this._getPanelWidth() });
            }
            if (overlayRef && !overlayRef.hasAttached()) {
                overlayRef.attach(this._portal);
                this._closingActionsSubscription = this._subscribeToClosingActions();
            }
            /** @type {?} */
            const wasOpen = this.panelOpen;
            this.autocomplete._setVisibility();
            this.autocomplete._isOpen = this._overlayAttached = true;
            // We need to do an extra `panelOpen` check in here, because the
            // autocomplete won't be shown if there are no options.
            if (this.panelOpen && wasOpen !== this.panelOpen) {
                this.autocomplete.opened.emit();
            }
        }
        /**
         * @private
         * @return {?}
         */
        _getOverlayConfig() {
            return new OverlayConfig({
                positionStrategy: this._getOverlayPosition(),
                scrollStrategy: this._scrollStrategy(),
                width: this._getPanelWidth(),
                direction: this._dir
            });
        }
        /**
         * @private
         * @return {?}
         */
        _getOverlayPosition() {
            /** @type {?} */
            const strategy = this._overlay.position()
                .flexibleConnectedTo(this._getConnectedElement())
                .withFlexibleDimensions(false)
                .withPush(false);
            this._setStrategyPositions(strategy);
            this._positionStrategy = strategy;
            return strategy;
        }
        /**
         * Sets the positions on a position strategy based on the directive's input state.
         * @private
         * @param {?} positionStrategy
         * @return {?}
         */
        _setStrategyPositions(positionStrategy) {
            // Note that we provide horizontal fallback positions, even though by default the dropdown
            // width matches the input, because consumers can override the width. See #18854.
            /** @type {?} */
            const belowPositions = [
                { originX: 'start', originY: 'bottom', overlayX: 'start', overlayY: 'top' },
                { originX: 'end', originY: 'bottom', overlayX: 'end', overlayY: 'top' }
            ];
            // The overlay edge connected to the trigger should have squared corners, while
            // the opposite end has rounded corners. We apply a CSS class to swap the
            // border-radius based on the overlay position.
            /** @type {?} */
            const panelClass = 'mat-autocomplete-panel-above';
            /** @type {?} */
            const abovePositions = [
                { originX: 'start', originY: 'top', overlayX: 'start', overlayY: 'bottom', panelClass },
                { originX: 'end', originY: 'top', overlayX: 'end', overlayY: 'bottom', panelClass }
            ];
            /** @type {?} */
            let positions;
            if (this.position === 'above') {
                positions = abovePositions;
            }
            else if (this.position === 'below') {
                positions = belowPositions;
            }
            else {
                positions = [...belowPositions, ...abovePositions];
            }
            positionStrategy.withPositions(positions);
        }
        /**
         * @private
         * @return {?}
         */
        _getConnectedElement() {
            if (this.connectedTo) {
                return this.connectedTo.elementRef;
            }
            return this._formField ? this._formField.getConnectedOverlayOrigin() : this._element;
        }
        /**
         * @private
         * @return {?}
         */
        _getPanelWidth() {
            return this.autocomplete.panelWidth || this._getHostWidth();
        }
        /**
         * Returns the width of the input element, so the panel width can match it.
         * @private
         * @return {?}
         */
        _getHostWidth() {
            return this._getConnectedElement().nativeElement.getBoundingClientRect().width;
        }
        /**
         * Resets the active item to -1 so arrow events will activate the
         * correct options, or to 0 if the consumer opted into it.
         * @private
         * @return {?}
         */
        _resetActiveItem() {
            this.autocomplete._keyManager.setActiveItem(this.autocomplete.autoActiveFirstOption ? 0 : -1);
        }
        /**
         * Determines whether the panel can be opened.
         * @private
         * @return {?}
         */
        _canOpen() {
            /** @type {?} */
            const element = this._element.nativeElement;
            return !element.readOnly && !element.disabled && !this._autocompleteDisabled;
        }
        /**
         * Use defaultView of injected document if available or fallback to global window reference
         * @private
         * @return {?}
         */
        _getWindow() {
            var _a;
            return ((_a = this._document) === null || _a === void 0 ? void 0 : _a.defaultView) || window;
        }
    }
    MatAutocompleteTrigger.ɵfac = function MatAutocompleteTrigger_Factory(t) { return new (t || MatAutocompleteTrigger)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(Overlay), ɵɵdirectiveInject(ViewContainerRef), ɵɵdirectiveInject(NgZone), ɵɵdirectiveInject(ChangeDetectorRef), ɵɵdirectiveInject(MAT_AUTOCOMPLETE_SCROLL_STRATEGY), ɵɵdirectiveInject(Directionality, 8), ɵɵdirectiveInject(MAT_FORM_FIELD, 9), ɵɵdirectiveInject(DOCUMENT$1, 8), ɵɵdirectiveInject(ViewportRuler)); };
    MatAutocompleteTrigger.ɵdir = ɵɵdefineDirective({ type: MatAutocompleteTrigger, selectors: [["input", "matAutocomplete", ""], ["textarea", "matAutocomplete", ""]], hostAttrs: [1, "mat-autocomplete-trigger"], hostVars: 7, hostBindings: function MatAutocompleteTrigger_HostBindings(rf, ctx) { if (rf & 1) {
            ɵɵlistener("focusin", function MatAutocompleteTrigger_focusin_HostBindingHandler() { return ctx._handleFocus(); })("blur", function MatAutocompleteTrigger_blur_HostBindingHandler() { return ctx._onTouched(); })("input", function MatAutocompleteTrigger_input_HostBindingHandler($event) { return ctx._handleInput($event); })("keydown", function MatAutocompleteTrigger_keydown_HostBindingHandler($event) { return ctx._handleKeydown($event); });
        } if (rf & 2) {
            ɵɵattribute("autocomplete", ctx.autocompleteAttribute)("role", ctx.autocompleteDisabled ? null : "combobox")("aria-autocomplete", ctx.autocompleteDisabled ? null : "list")("aria-activedescendant", ctx.panelOpen && ctx.activeOption ? ctx.activeOption.id : null)("aria-expanded", ctx.autocompleteDisabled ? null : ctx.panelOpen.toString())("aria-owns", ctx.autocompleteDisabled || !ctx.panelOpen ? null : ctx.autocomplete == null ? null : ctx.autocomplete.id)("aria-haspopup", !ctx.autocompleteDisabled);
        } }, inputs: { position: ["matAutocompletePosition", "position"], autocompleteAttribute: ["autocomplete", "autocompleteAttribute"], autocompleteDisabled: ["matAutocompleteDisabled", "autocompleteDisabled"], autocomplete: ["matAutocomplete", "autocomplete"], connectedTo: ["matAutocompleteConnectedTo", "connectedTo"] }, exportAs: ["matAutocompleteTrigger"], features: [ɵɵProvidersFeature([MAT_AUTOCOMPLETE_VALUE_ACCESSOR]), ɵɵNgOnChangesFeature] });
    /** @nocollapse */
    MatAutocompleteTrigger.ctorParameters = () => [
        { type: ElementRef },
        { type: Overlay },
        { type: ViewContainerRef },
        { type: NgZone },
        { type: ChangeDetectorRef },
        { type: undefined, decorators: [{ type: Inject, args: [MAT_AUTOCOMPLETE_SCROLL_STRATEGY,] }] },
        { type: Directionality, decorators: [{ type: Optional }] },
        { type: MatFormField, decorators: [{ type: Optional }, { type: Inject, args: [MAT_FORM_FIELD,] }, { type: Host }] },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [DOCUMENT$1,] }] },
        { type: ViewportRuler }
    ];
    MatAutocompleteTrigger.propDecorators = {
        autocomplete: [{ type: Input, args: ['matAutocomplete',] }],
        position: [{ type: Input, args: ['matAutocompletePosition',] }],
        connectedTo: [{ type: Input, args: ['matAutocompleteConnectedTo',] }],
        autocompleteAttribute: [{ type: Input, args: ['autocomplete',] }],
        autocompleteDisabled: [{ type: Input, args: ['matAutocompleteDisabled',] }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/autocomplete/autocomplete-module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class MatAutocompleteModule {
    }
    MatAutocompleteModule.ɵmod = ɵɵdefineNgModule({ type: MatAutocompleteModule });
    MatAutocompleteModule.ɵinj = ɵɵdefineInjector({ factory: function MatAutocompleteModule_Factory(t) { return new (t || MatAutocompleteModule)(); }, providers: [MAT_AUTOCOMPLETE_SCROLL_STRATEGY_FACTORY_PROVIDER], imports: [[MatOptionModule, OverlayModule, MatCommonModule, CommonModule],
            CdkScrollableModule,
            MatOptionModule,
            MatCommonModule] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(MatAutocompleteModule, { declarations: function () { return [MatAutocomplete,
            MatAutocompleteTrigger,
            MatAutocompleteOrigin]; }, imports: function () { return [MatOptionModule, OverlayModule, MatCommonModule, CommonModule]; }, exports: function () { return [CdkScrollableModule,
            MatAutocomplete,
            MatOptionModule,
            MatAutocompleteTrigger,
            MatAutocompleteOrigin,
            MatCommonModule]; } }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/views/main_view/filter_input_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    function MetricsFilterInputComponent_mat_icon_3_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "mat-icon", 7);
    } }
    function MetricsFilterInputComponent_mat_option_6_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "mat-option", 8);
        ɵɵtext(1);
        ɵɵelementEnd();
    } if (rf & 2) {
        const completion_r3 = ctx.$implicit;
        ɵɵproperty("value", completion_r3);
        ɵɵadvance(1);
        ɵɵtextInterpolate(completion_r3);
    } }
    class MetricsFilterInputComponent {
        constructor() {
            this.onRegexFilterValueChange = new EventEmitter();
        }
        /**
         * @param {?} event
         * @return {?}
         */
        onFilterKeyUp(event) {
            if (event.key === 'Enter') {
                this.autocompleteTrigger.closePanel();
            }
        }
        /**
         * @param {?} completion
         * @return {?}
         */
        onCompletionAccepted(completion) {
            this.onRegexFilterValueChange.emit(escapeForRegex(completion));
        }
    }
    MetricsFilterInputComponent.decorators = [
        { type: Component, args: [{
                    selector: 'metrics-tag-filter-component',
                    templateUrl: 'filter_input_component.ng.html',
                    styleUrls: [`filter_input_component.css`],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    MetricsFilterInputComponent.propDecorators = {
        regexFilterValue: [{ type: Input }],
        isRegexFilterValid: [{ type: HostBinding, args: ['class.valid',] }, { type: Input }],
        completions: [{ type: Input }],
        onRegexFilterValueChange: [{ type: Output }],
        autocompleteTrigger: [{ type: ViewChild, args: [MatAutocompleteTrigger, { static: true },] }]
    };
    /** @nocollapse */ MetricsFilterInputComponent.ɵfac = function MetricsFilterInputComponent_Factory(t) { return new (t || MetricsFilterInputComponent)(); };
    /** @nocollapse */ MetricsFilterInputComponent.ɵcmp = ɵɵdefineComponent({ type: MetricsFilterInputComponent, selectors: [["metrics-tag-filter-component"]], viewQuery: function MetricsFilterInputComponent_Query(rf, ctx) { if (rf & 1) {
            ɵɵstaticViewQuery(MatAutocompleteTrigger, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.autocompleteTrigger = _t.first);
        } }, hostVars: 2, hostBindings: function MetricsFilterInputComponent_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵclassProp("valid", ctx.isRegexFilterValid);
        } }, inputs: { regexFilterValue: "regexFilterValue", isRegexFilterValid: "isRegexFilterValid", completions: "completions" }, outputs: { onRegexFilterValueChange: "onRegexFilterValueChange" }, decls: 7, vars: 4, consts: [[1, "tag-filter"], ["svgIcon", "search_24px"], ["matInput", "", "autocomplete", "off", "placeholder", "Filter tags (regex)", 3, "value", "matAutocomplete", "input", "keyup"], ["svgIcon", "error_24px", "class", "error-icon", "title", "Invalid regex filter. The result may be stale.", 4, "ngIf"], [3, "optionSelected"], ["filterMatches", "matAutocomplete"], [3, "value", 4, "ngFor", "ngForOf"], ["svgIcon", "error_24px", "title", "Invalid regex filter. The result may be stale.", 1, "error-icon"], [3, "value"]], template: function MetricsFilterInputComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "div", 0);
            ɵɵelement(1, "mat-icon", 1);
            ɵɵelementStart(2, "input", 2);
            ɵɵlistener("input", function MetricsFilterInputComponent_Template_input_input_2_listener($event) { return ctx.onRegexFilterValueChange.emit($event.target.value); })("keyup", function MetricsFilterInputComponent_Template_input_keyup_2_listener($event) { return ctx.onFilterKeyUp($event); });
            ɵɵelementEnd();
            ɵɵtemplate(3, MetricsFilterInputComponent_mat_icon_3_Template, 1, 0, "mat-icon", 3);
            ɵɵelementEnd();
            ɵɵelementStart(4, "mat-autocomplete", 4, 5);
            ɵɵlistener("optionSelected", function MetricsFilterInputComponent_Template_mat_autocomplete_optionSelected_4_listener($event) { return ctx.onCompletionAccepted($event.option.value); });
            ɵɵtemplate(6, MetricsFilterInputComponent_mat_option_6_Template, 2, 2, "mat-option", 6);
            ɵɵelementEnd();
        } if (rf & 2) {
            const _r1 = ɵɵreference(5);
            ɵɵadvance(2);
            ɵɵproperty("value", ctx.regexFilterValue)("matAutocomplete", _r1);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", !ctx.isRegexFilterValid);
            ɵɵadvance(3);
            ɵɵproperty("ngForOf", ctx.completions);
        } }, directives: [MatIcon, MatInput, MatAutocompleteTrigger, NgIf, MatAutocomplete, NgForOf, MatOption], styles: [".mat-badge-content[_ngcontent-%COMP%]{font-weight:600;font-size:12px;font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:9px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:24px}.mat-h1[_ngcontent-%COMP%], .mat-headline[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%]{font:400 24px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h2[_ngcontent-%COMP%], .mat-title[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h3[_ngcontent-%COMP%], .mat-subheading-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%]{font:400 16px/28px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h4[_ngcontent-%COMP%], .mat-subheading-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%]{font:400 15px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h5[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%]{font:400 calc(14px * 0.83)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-h6[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:400 calc(14px * 0.67)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-body-strong[_ngcontent-%COMP%], .mat-body-2[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   p[_ngcontent-%COMP%]{margin:0 0 12px}.mat-small[_ngcontent-%COMP%], .mat-caption[_ngcontent-%COMP%]{font:400 12px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-display-4[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-4[_ngcontent-%COMP%]{font:300 112px/112px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.05em;margin:0 0 56px}.mat-display-3[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-3[_ngcontent-%COMP%]{font:400 56px/56px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.02em;margin:0 0 64px}.mat-display-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-2[_ngcontent-%COMP%]{font:400 45px/48px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.005em;margin:0 0 64px}.mat-display-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-1[_ngcontent-%COMP%]{font:400 34px/40px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 64px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%], .mat-flat-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-button-toggle[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card-title[_ngcontent-%COMP%]{font-size:24px;font-weight:500}.mat-card-header[_ngcontent-%COMP%]   .mat-card-title[_ngcontent-%COMP%]{font-size:20px}.mat-card-subtitle[_ngcontent-%COMP%], .mat-card-content[_ngcontent-%COMP%]{font-size:14px}.mat-checkbox[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-checkbox-layout[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{line-height:24px}.mat-chip[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-chip[_ngcontent-%COMP%]   .mat-chip-trailing-icon.mat-icon[_ngcontent-%COMP%], .mat-chip[_ngcontent-%COMP%]   .mat-chip-remove.mat-icon[_ngcontent-%COMP%]{font-size:18px}.mat-table[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-header-cell[_ngcontent-%COMP%]{font-size:12px;font-weight:500}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{font-size:14px}.mat-calendar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-calendar-body[_ngcontent-%COMP%]{font-size:13px}.mat-calendar-body-label[_ngcontent-%COMP%], .mat-calendar-period-button[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-calendar-table-header[_ngcontent-%COMP%]   th[_ngcontent-%COMP%]{font-size:11px;font-weight:400}.mat-dialog-title[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-expansion-panel-header[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:15px;font-weight:400}.mat-expansion-panel-content[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field[_ngcontent-%COMP%]{font-size:inherit;font-weight:400;line-height:1.125;font-family:Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.34375em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{font-size:150%;line-height:1.125}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]{height:1.5em;width:1.5em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{height:1.125em;line-height:1.125}.mat-form-field-infix[_ngcontent-%COMP%]{padding:.5em 0;border-top:.84375em solid transparent}.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34375em) scale(0.75);width:133.3333333333%}.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34374em) scale(0.75);width:133.3333433333%}.mat-form-field-label-wrapper[_ngcontent-%COMP%]{top:-0.84375em;padding-top:.84375em}.mat-form-field-label[_ngcontent-%COMP%]{top:1.34375em}.mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.34375em}.mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{font-size:75%;margin-top:.6666666667em;top:calc(100% - 1.7916666667em)}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.4375em 0}.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.001px);-ms-transform:translateY(-1.28125em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00101px);-ms-transform:translateY(-1.28124em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00102px);-ms-transform:translateY(-1.28123em) scale(0.75);width:133.3333533333%}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.28125em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{margin-top:.5416666667em;top:calc(100% - 1.6666666667em)}@media print{.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28122em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28121em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.2812em) scale(0.75)}}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.25em 0 .75em 0}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.09375em;margin-top:-0.5em}.mat-form-field-appearance-fill.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59374em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:1em 0 1em 0}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.84375em;margin-top:-0.25em}.mat-form-field-appearance-outline.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59374em) scale(0.75);width:133.3333433333%}.mat-grid-tile-header[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]{font-size:14px}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2), .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}input.mat-input-element[_ngcontent-%COMP%]{margin-top:-0.0625em}.mat-menu-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:400}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px}.mat-radio-button[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select-trigger[_ngcontent-%COMP%]{height:1.125em}.mat-slide-toggle-content[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-slider-thumb-label-text[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-stepper-vertical[_ngcontent-%COMP%], .mat-stepper-horizontal[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-step-label[_ngcontent-%COMP%]{font-size:14px;font-weight:400}.mat-step-sub-label-error[_ngcontent-%COMP%]{font-weight:normal}.mat-step-label-error[_ngcontent-%COMP%]{font-size:14px}.mat-step-label-selected[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-tab-group[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-toolbar[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0}.mat-tooltip[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:10px;padding-top:6px;padding-bottom:6px}.mat-tooltip-handset[_ngcontent-%COMP%]{font-size:14px;padding-top:8px;padding-bottom:8px}.mat-list-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:16px}.mat-optgroup-label[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-simple-snackbar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px}.mat-simple-snackbar-action[_ngcontent-%COMP%]{line-height:1;font-family:inherit;font-size:inherit;font-weight:500}.mat-tree[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{font-weight:400;font-size:14px}.mat-ripple[_ngcontent-%COMP%]{overflow:hidden;position:relative}.mat-ripple[_ngcontent-%COMP%]:not(:empty){transform:translateZ(0)}.mat-ripple.mat-ripple-unbounded[_ngcontent-%COMP%]{overflow:visible}.mat-ripple-element[_ngcontent-%COMP%]{position:absolute;border-radius:50%;pointer-events:none;transition:opacity,transform 0ms cubic-bezier(0, 0, 0.2, 1);transform:scale(0)}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{display:none}.cdk-visually-hidden[_ngcontent-%COMP%]{border:0;clip:rect(0 0 0 0);height:1px;margin:-1px;overflow:hidden;padding:0;position:absolute;width:1px;outline:0;-webkit-appearance:none;-moz-appearance:none}.cdk-overlay-container[_ngcontent-%COMP%], .cdk-global-overlay-wrapper[_ngcontent-%COMP%]{pointer-events:none;top:0;left:0;height:100%;width:100%}.cdk-overlay-container[_ngcontent-%COMP%]{position:fixed;z-index:1000}.cdk-overlay-container[_ngcontent-%COMP%]:empty{display:none}.cdk-global-overlay-wrapper[_ngcontent-%COMP%]{display:flex;position:absolute;z-index:1000}.cdk-overlay-pane[_ngcontent-%COMP%]{position:absolute;pointer-events:auto;box-sizing:border-box;z-index:1000;display:flex;max-width:100%;max-height:100%}.cdk-overlay-backdrop[_ngcontent-%COMP%]{position:absolute;top:0;bottom:0;left:0;right:0;z-index:1000;pointer-events:auto;-webkit-tap-highlight-color:transparent;transition:opacity 400ms cubic-bezier(0.25, 0.8, 0.25, 1);opacity:0}.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:1}@media screen and (-ms-high-contrast: active){.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:.6}}.cdk-overlay-dark-backdrop[_ngcontent-%COMP%]{background:rgba(0,0,0,.32)}.cdk-overlay-transparent-backdrop[_ngcontent-%COMP%], .cdk-overlay-transparent-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:0}.cdk-overlay-connected-position-bounding-box[_ngcontent-%COMP%]{position:absolute;z-index:1000;display:flex;flex-direction:column;min-width:1px;min-height:1px}.cdk-global-scrollblock[_ngcontent-%COMP%]{position:fixed;width:100%;overflow-y:scroll}@keyframes cdk-text-field-autofill-start{}@keyframes cdk-text-field-autofill-end{}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:-webkit-autofill{animation:cdk-text-field-autofill-start 0s 1ms}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:not(:-webkit-autofill){animation:cdk-text-field-autofill-end 0s 1ms}textarea.cdk-textarea-autosize[_ngcontent-%COMP%]{resize:none}textarea.cdk-textarea-autosize-measuring[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:auto !important;overflow:hidden !important}textarea.cdk-textarea-autosize-measuring-firefox[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:0 !important}.mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.1)}.mat-option[_ngcontent-%COMP%]{color:#212121}.mat-option[_ngcontent-%COMP%]:hover:not(.mat-option-disabled), .mat-option[_ngcontent-%COMP%]:focus:not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple):not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-active[_ngcontent-%COMP%]{background:rgba(0,0,0,.04);color:#212121}.mat-option.mat-option-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-primary[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f57c00}.mat-accent[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f44336}.mat-optgroup-label[_ngcontent-%COMP%]{color:#616161}.mat-optgroup-disabled[_ngcontent-%COMP%]   .mat-optgroup-label[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-pseudo-checkbox[_ngcontent-%COMP%]{color:#616161}.mat-pseudo-checkbox[_ngcontent-%COMP%]::after{color:#fafafa}.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{color:#b0b0b0}.mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f57c00}.mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f44336}.mat-pseudo-checkbox-checked.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{background:#b0b0b0}.mat-elevation-z0[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z1[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z2[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z3[_ngcontent-%COMP%]{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z4[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z5[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 5px 8px 0px rgba(0, 0, 0, 0.14),0px 1px 14px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z6[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z7[_ngcontent-%COMP%]{box-shadow:0px 4px 5px -2px rgba(0, 0, 0, 0.2),0px 7px 10px 1px rgba(0, 0, 0, 0.14),0px 2px 16px 1px rgba(0, 0, 0, 0.12)}.mat-elevation-z8[_ngcontent-%COMP%]{box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z9[_ngcontent-%COMP%]{box-shadow:0px 5px 6px -3px rgba(0, 0, 0, 0.2),0px 9px 12px 1px rgba(0, 0, 0, 0.14),0px 3px 16px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z10[_ngcontent-%COMP%]{box-shadow:0px 6px 6px -3px rgba(0, 0, 0, 0.2),0px 10px 14px 1px rgba(0, 0, 0, 0.14),0px 4px 18px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z11[_ngcontent-%COMP%]{box-shadow:0px 6px 7px -4px rgba(0, 0, 0, 0.2),0px 11px 15px 1px rgba(0, 0, 0, 0.14),0px 4px 20px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z12[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z13[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 13px 19px 2px rgba(0, 0, 0, 0.14),0px 5px 24px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z14[_ngcontent-%COMP%]{box-shadow:0px 7px 9px -4px rgba(0, 0, 0, 0.2),0px 14px 21px 2px rgba(0, 0, 0, 0.14),0px 5px 26px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z15[_ngcontent-%COMP%]{box-shadow:0px 8px 9px -5px rgba(0, 0, 0, 0.2),0px 15px 22px 2px rgba(0, 0, 0, 0.14),0px 6px 28px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z16[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z17[_ngcontent-%COMP%]{box-shadow:0px 8px 11px -5px rgba(0, 0, 0, 0.2),0px 17px 26px 2px rgba(0, 0, 0, 0.14),0px 6px 32px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z18[_ngcontent-%COMP%]{box-shadow:0px 9px 11px -5px rgba(0, 0, 0, 0.2),0px 18px 28px 2px rgba(0, 0, 0, 0.14),0px 7px 34px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z19[_ngcontent-%COMP%]{box-shadow:0px 9px 12px -6px rgba(0, 0, 0, 0.2),0px 19px 29px 2px rgba(0, 0, 0, 0.14),0px 7px 36px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z20[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 20px 31px 3px rgba(0, 0, 0, 0.14),0px 8px 38px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z21[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 21px 33px 3px rgba(0, 0, 0, 0.14),0px 8px 40px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z22[_ngcontent-%COMP%]{box-shadow:0px 10px 14px -6px rgba(0, 0, 0, 0.2),0px 22px 35px 3px rgba(0, 0, 0, 0.14),0px 8px 42px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z23[_ngcontent-%COMP%]{box-shadow:0px 11px 14px -7px rgba(0, 0, 0, 0.2),0px 23px 36px 3px rgba(0, 0, 0, 0.14),0px 9px 44px 8px rgba(0, 0, 0, 0.12)}.mat-elevation-z24[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12)}.mat-app-background[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-theme-loaded-marker[_ngcontent-%COMP%]{display:none}.mat-autocomplete-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-autocomplete-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover){background:#fff}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover):not(.mat-option-disabled){color:#212121}.mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f57c00}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{outline:solid 1px;border-radius:0}.mat-badge-accent[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-badge-warn[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f44336}.mat-badge[_ngcontent-%COMP%]{position:relative}.mat-badge-hidden[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{display:none}.mat-badge-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#b9b9b9;color:#757575}.mat-badge-content[_ngcontent-%COMP%]{position:absolute;text-align:center;display:inline-block;border-radius:50%;transition:transform 200ms ease-in-out;transform:scale(0.6);overflow:hidden;white-space:nowrap;text-overflow:ellipsis;pointer-events:none}.ng-animate-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%], .mat-badge-content._mat-animation-noopable[_ngcontent-%COMP%]{transition:none}.mat-badge-content.mat-badge-active[_ngcontent-%COMP%]{transform:none}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:16px;height:16px;line-height:16px}.mat-badge-small.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-8px}.mat-badge-small.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-8px}.mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-16px}.mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-16px}.mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-8px}.mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-8px}.mat-badge-medium[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:22px;height:22px;line-height:22px}.mat-badge-medium.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-11px}.mat-badge-medium.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-11px}.mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-22px}.mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-22px}.mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-11px}.mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-11px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:28px;height:28px;line-height:28px}.mat-badge-large.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-14px}.mat-badge-large.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-14px}.mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-28px}.mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-28px}.mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-14px}.mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-14px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]{color:inherit;background:transparent}.mat-button.mat-primary[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-button[disabled][disabled][_ngcontent-%COMP%], .mat-icon-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-icon-button[disabled][disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-stroked-button[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f44336}.mat-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:transparent}.mat-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{opacity:.1;background-color:currentColor}.mat-button-focus-overlay[_ngcontent-%COMP%]{background:#000}.mat-stroked-button[_ngcontent-%COMP%]:not([disabled]){border-color:rgba(0,0,0,.12)}.mat-flat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{color:#212121;background-color:#fff}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-flat-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-stroked-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-flat-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-raised-button[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone[_ngcontent-%COMP%], .mat-button-toggle-group[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{box-shadow:none}.mat-button-toggle[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-button-toggle[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121;background:#fff}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:#000}.mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12)}[dir=rtl][_ngcontent-%COMP%]   .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-group-appearance-standard.mat-button-toggle-vertical[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:none;border-top:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#616161}.mat-button-toggle-checked.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121}.mat-button-toggle-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.26);background-color:#eee}.mat-button-toggle-disabled.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{background:#fff}.mat-button-toggle-disabled.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#bdbdbd}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{border:solid 1px rgba(0,0,0,.12)}.mat-card[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-card[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-card.mat-card-flat[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-card-subtitle[_ngcontent-%COMP%]{color:#616161}.mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#616161}.mat-checkbox-checkmark[_ngcontent-%COMP%]{fill:#fafafa}.mat-checkbox-checkmark-path[_ngcontent-%COMP%]{stroke:#fafafa !important}.mat-checkbox-mixedmark[_ngcontent-%COMP%]{background-color:#fafafa}.mat-checkbox-indeterminate.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f57c00}.mat-checkbox-indeterminate.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#ff9800}.mat-checkbox-indeterminate.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f44336}.mat-checkbox-disabled.mat-checkbox-checked[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-disabled.mat-checkbox-indeterminate[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]:not(.mat-checkbox-checked)   .mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{color:#616161}.mat-checkbox[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%]{background:#f57c00}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%]{background:#ff9800}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%]{background:#f44336}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#212121}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#212121;opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled):active{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled)   .mat-chip-remove[_ngcontent-%COMP%]:hover{opacity:.54}.mat-chip.mat-standard-chip.mat-chip-disabled[_ngcontent-%COMP%]{opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]::after{background:#000}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-table[_ngcontent-%COMP%]{background:#fff}.mat-table[_ngcontent-%COMP%]   thead[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tbody[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tfoot[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], [mat-header-row][_ngcontent-%COMP%], [mat-row][_ngcontent-%COMP%], [mat-footer-row][_ngcontent-%COMP%], .mat-table-sticky[_ngcontent-%COMP%]{background:inherit}mat-row[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], th.mat-header-cell[_ngcontent-%COMP%], td.mat-cell[_ngcontent-%COMP%], td.mat-footer-cell[_ngcontent-%COMP%]{border-bottom-color:rgba(0,0,0,.12)}.mat-header-cell[_ngcontent-%COMP%]{color:#616161}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{color:#212121}.mat-calendar-arrow[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.54)}.mat-datepicker-toggle[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-next-button[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-previous-button[_ngcontent-%COMP%]{color:rgba(0,0,0,.54)}.mat-calendar-table-header[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-calendar-table-header-divider[_ngcontent-%COMP%]::after{background:rgba(0,0,0,.12)}.mat-calendar-body-label[_ngcontent-%COMP%]{color:#616161}.mat-calendar-body-cell-content[_ngcontent-%COMP%]{color:#212121;border-color:transparent}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){color:#757575}.mat-calendar-body-cell[_ngcontent-%COMP%]:not(.mat-calendar-body-disabled):hover > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-keyboard-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-program-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){background-color:rgba(0,0,0,.04)}.mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.38)}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.18)}.mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.4)}.mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12);background-color:#fff;color:#212121}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.4)}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.4)}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content-touch[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-datepicker-toggle-active[_ngcontent-%COMP%]{color:#f57c00}.mat-datepicker-toggle-active.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-datepicker-toggle-active.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-dialog-container[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-divider[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-divider-vertical[_ngcontent-%COMP%]{border-right-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-expansion-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-action-row[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-keyboard-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-program-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded)   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover:not([aria-disabled=true]){background:rgba(0,0,0,.04)}@media(hover: none){.mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded):not([aria-disabled=true])   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover{background:#fff}}.mat-expansion-panel-header-title[_ngcontent-%COMP%]{color:#212121}.mat-expansion-panel-header-description[_ngcontent-%COMP%], .mat-expansion-indicator[_ngcontent-%COMP%]::after{color:#616161}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-title[_ngcontent-%COMP%], .mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-description[_ngcontent-%COMP%]{color:inherit}.mat-form-field-label[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-hint[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-focused[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-ripple[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid)   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f57c00}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-accent   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#ff9800}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-warn   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#f44336}.mat-error[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-hint[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-legacy.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-standard[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-standard.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.04)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.02)}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:transparent}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.12)}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:rgba(0,0,0,.87)}.mat-form-field-appearance-outline.mat-focused[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field-appearance-outline.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-appearance-outline.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-invalid.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.06)}.mat-icon.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-icon.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-icon.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-type-mat-native-select[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#616161}.mat-input-element[_ngcontent-%COMP%]:disabled, .mat-form-field-type-mat-native-select.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#757575}.mat-input-element[_ngcontent-%COMP%]{caret-color:#f57c00}.mat-input-element[_ngcontent-%COMP%]::placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-moz-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-webkit-input-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]:-ms-input-placeholder{color:rgba(97,97,97,.42)}.mat-accent[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%], .mat-form-field-invalid[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#f44336}.mat-form-field-type-mat-native-select.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{color:#616161}.mat-list-item-disabled[_ngcontent-%COMP%]{background-color:#eee}.mat-list-option[_ngcontent-%COMP%]:hover, .mat-list-option[_ngcontent-%COMP%]:focus, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.04)}.mat-list-single-selected-option[_ngcontent-%COMP%], .mat-list-single-selected-option[_ngcontent-%COMP%]:hover, .mat-list-single-selected-option[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.12)}.mat-menu-panel[_ngcontent-%COMP%]{background:#fff}.mat-menu-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-menu-item[_ngcontent-%COMP%]{background:transparent;color:#212121}.mat-menu-item[disabled][_ngcontent-%COMP%], .mat-menu-item[disabled][_ngcontent-%COMP%]::after{color:rgba(0,0,0,.38)}.mat-menu-item[_ngcontent-%COMP%]   .mat-icon-no-color[_ngcontent-%COMP%], .mat-menu-item-submenu-trigger[_ngcontent-%COMP%]::after{color:rgba(0,0,0,.54)}.mat-menu-item[_ngcontent-%COMP%]:hover:not([disabled]), .mat-menu-item.cdk-program-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item.cdk-keyboard-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item-highlighted[_ngcontent-%COMP%]:not([disabled]){background:rgba(0,0,0,.04)}.mat-paginator[_ngcontent-%COMP%]{background:#fff}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{color:#616161}.mat-paginator-decrement[_ngcontent-%COMP%], .mat-paginator-increment[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54);border-right:2px solid rgba(0,0,0,.54)}.mat-paginator-first[_ngcontent-%COMP%], .mat-paginator-last[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54)}.mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-decrement[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-increment[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-first[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-last[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f57c00}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#ff9800}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f44336}.mat-progress-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f57c00}.mat-progress-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#ff9800}.mat-progress-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f44336}.mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#616161}.mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f57c00}.mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#ff9800}.mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#ff9800}.mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f44336}.mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f44336}.mat-radio-button.mat-radio-disabled.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-label-content[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-radio-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-select-value[_ngcontent-%COMP%]{color:#212121}.mat-select-placeholder[_ngcontent-%COMP%]{color:rgba(97,97,97,.42)}.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%]{color:#757575}.mat-select-arrow[_ngcontent-%COMP%]{color:#616161}.mat-select-panel[_ngcontent-%COMP%]{background:#fff}.mat-select-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-select-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple){background:rgba(0,0,0,.12)}.mat-form-field.mat-focused.mat-primary[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-invalid[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#757575}.mat-drawer-container[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-drawer[_ngcontent-%COMP%]{background-color:#fff;color:#212121}.mat-drawer.mat-drawer-push[_ngcontent-%COMP%]{background-color:#fff}.mat-drawer[_ngcontent-%COMP%]:not(.mat-drawer-side){box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-drawer-side[_ngcontent-%COMP%]{border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-backdrop.mat-drawer-shown[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.6)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.54)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.54)}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.54)}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle[_ngcontent-%COMP%]:not(.mat-checked)   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-slide-toggle-thumb[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12);background-color:#fafafa}.mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-primary[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f57c00}.mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-accent[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#ff9800}.mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-warn[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f44336}.mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.2)}.mat-slider[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%], .cdk-focused[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-disabled[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]   .mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing)   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26);background-color:transparent}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26)}.mat-slider-has-ticks[_ngcontent-%COMP%]   .mat-slider-wrapper[_ngcontent-%COMP%]::after{border-color:rgba(0,0,0,.7)}.mat-slider-horizontal[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to right, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent);background-image:-moz-repeating-linear-gradient(0.0001deg, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-slider-vertical[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to bottom, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-step-header.cdk-keyboard-focused[_ngcontent-%COMP%], .mat-step-header.cdk-program-focused[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]:hover{background-color:rgba(0,0,0,.04)}@media(hover: none){.mat-step-header[_ngcontent-%COMP%]:hover{background:none}}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-optional[_ngcontent-%COMP%]{color:#616161}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon[_ngcontent-%COMP%]{background-color:#616161;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-selected[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-done[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-edit[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-error[_ngcontent-%COMP%]{background-color:transparent;color:#f44336}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-active[_ngcontent-%COMP%]{color:#212121}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-error[_ngcontent-%COMP%]{color:#f44336}.mat-stepper-horizontal[_ngcontent-%COMP%], .mat-stepper-vertical[_ngcontent-%COMP%]{background-color:#fff}.mat-stepper-vertical-line[_ngcontent-%COMP%]::before{border-left-color:rgba(0,0,0,.12)}.mat-horizontal-stepper-header[_ngcontent-%COMP%]::before, .mat-horizontal-stepper-header[_ngcontent-%COMP%]::after, .mat-stepper-horizontal-line[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-sort-header-arrow[_ngcontent-%COMP%]{color:#616161}.mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-header[_ngcontent-%COMP%]{border-bottom:1px solid rgba(0,0,0,.12)}.mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%]{border-top:1px solid rgba(0,0,0,.12);border-bottom:none}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{color:#212121}.mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:#757575}.mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#212121}.mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#757575}.mat-tab-group[class*=mat-background-][_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar[class*=mat-background-][_ngcontent-%COMP%]{border-bottom:none;border-top:none}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-toolbar[_ngcontent-%COMP%]{background:#f5f5f5;color:#212121}.mat-toolbar.mat-primary[_ngcontent-%COMP%]{background:#f57c00;color:#fff}.mat-toolbar.mat-accent[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-toolbar.mat-warn[_ngcontent-%COMP%]{background:#f44336;color:#fff}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:currentColor}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:inherit}.mat-toolbar[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:currentColor}.mat-tooltip[_ngcontent-%COMP%]{background:rgba(97,97,97,.9)}.mat-tree[_ngcontent-%COMP%]{background:#fff}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{color:#212121}.mat-snack-bar-container[_ngcontent-%COMP%]{color:rgba(255,255,255,.7);background:#323232;box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-simple-snackbar-action[_ngcontent-%COMP%]{color:#ff9800}.tag-filter[_ngcontent-%COMP%]{display:flex;position:relative}mat-icon[_ngcontent-%COMP%]{color:#616161;flex:none;margin-right:5px}[_nghost-%COMP%]{color:#212121;font-size:13px}[_nghost-%COMP%]:not(.valid){color:#c62828}[_nghost-%COMP%]:not(.valid)   input[_ngcontent-%COMP%]{caret-color:currentColor}[_nghost-%COMP%]:not(.valid)   .error-icon[_ngcontent-%COMP%]{color:#c62828;position:absolute;right:0}"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/views/main_view/filter_input_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class MetricsFilterInputContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.tagFilter$ = this.store.select(getMetricsTagFilter);
            this.isTagFilterRegexValid$ = this.tagFilter$.pipe(map((/**
             * @param {?} tagFilterString
             * @return {?}
             */
            (tagFilterString) => {
                try {
                    // tslint:disable-next-line:no-unused-expression Check for validity of filter.
                    new RegExp(tagFilterString);
                    return true;
                }
                catch (err) {
                    return false;
                }
            })));
            this.completions$ = this.store
                .select(getNonEmptyCardIdsWithMetadata)
                .pipe(map((/**
             * @param {?} cardList
             * @return {?}
             */
            (cardList) => cardList.map((/**
             * @param {?} __0
             * @return {?}
             */
            ({ tag }) => tag)))), switchMap((/**
             * @param {?} cardList
             * @return {?}
             */
            (cardList) => {
                return combineLatest([
                    of(cardList),
                    this.store.select(getMetricsTagFilter),
                ]);
            })), map((/**
             * @param {?} __0
             * @return {?}
             */
            ([tags, tagFilter]) => {
                try {
                    /** @type {?} */
                    const regex = new RegExp(tagFilter);
                    return [tags, regex];
                }
                catch (e) {
                    return [tags, null];
                }
            })), filter((/**
             * @param {?} __0
             * @return {?}
             */
            ([, tagFilterRegex]) => tagFilterRegex !== null)), map((/**
             * @param {?} __0
             * @return {?}
             */
            ([tags, tagFilterRegex]) => {
                return tags
                    .filter((/**
                 * @param {?} tag
                 * @return {?}
                 */
                (tag) => (/** @type {?} */ (tagFilterRegex)).test(tag)))
                    .sort(compareTagNames);
            })));
        }
        /**
         * @param {?} tagFilter
         * @return {?}
         */
        onTagFilterChange(tagFilter) {
            this.store.dispatch(metricsTagFilterChanged({ tagFilter }));
        }
    }
    MetricsFilterInputContainer.decorators = [
        { type: Component, args: [{
                    selector: 'metrics-tag-filter',
                    template: `
    <metrics-tag-filter-component
      [regexFilterValue]="tagFilter$ | async"
      [isRegexFilterValid]="isTagFilterRegexValid$ | async"
      [completions]="completions$ | async"
      (onRegexFilterValueChange)="onTagFilterChange($event)"
    ></metrics-tag-filter-component>
  `,
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    MetricsFilterInputContainer.ctorParameters = () => [
        { type: Store }
    ];
    /** @nocollapse */ MetricsFilterInputContainer.ɵfac = function MetricsFilterInputContainer_Factory(t) { return new (t || MetricsFilterInputContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ MetricsFilterInputContainer.ɵcmp = ɵɵdefineComponent({ type: MetricsFilterInputContainer, selectors: [["metrics-tag-filter"]], decls: 4, vars: 9, consts: [[3, "regexFilterValue", "isRegexFilterValid", "completions", "onRegexFilterValueChange"]], template: function MetricsFilterInputContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "metrics-tag-filter-component", 0);
            ɵɵlistener("onRegexFilterValueChange", function MetricsFilterInputContainer_Template_metrics_tag_filter_component_onRegexFilterValueChange_0_listener($event) { return ctx.onTagFilterChange($event); });
            ɵɵpipe(1, "async");
            ɵɵpipe(2, "async");
            ɵɵpipe(3, "async");
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵproperty("regexFilterValue", ɵɵpipeBind1(1, 3, ctx.tagFilter$))("isRegexFilterValid", ɵɵpipeBind1(2, 5, ctx.isTagFilterRegexValid$))("completions", ɵɵpipeBind1(3, 7, ctx.completions$));
        } }, directives: [MetricsFilterInputComponent], pipes: [AsyncPipe], encapsulation: 2, changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/widgets/text/truncated_path_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    function TruncatedPathComponent_span_0_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "span", 2);
        ɵɵtext(1);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r0 = ɵɵnextContext();
        ɵɵadvance(1);
        ɵɵtextInterpolate(ctx_r0.firstTextPart());
    } }
    /**
     * A component for rendering a '/' delimited path. When text exceeds its
     * container's width, it will try to show as much of the last '/' slash piece as
     * possible. If there is no slash piece, it simply truncates from the right with
     * ellipsis.
     */
    class TruncatedPathComponent {
        /**
         * @private
         * @return {?}
         */
        parseValue() {
            /** @type {?} */
            const lastPieceIndex = this.value.lastIndexOf('/');
            if (lastPieceIndex === -1) {
                return { first: '', second: this.value };
            }
            return {
                first: this.value.slice(0, lastPieceIndex),
                second: this.value.slice(lastPieceIndex),
            };
        }
        /**
         * @return {?}
         */
        firstTextPart() {
            return this.parseValue().first;
        }
        /**
         * @return {?}
         */
        secondTextPart() {
            return this.parseValue().second;
        }
    }
    TruncatedPathComponent.decorators = [
        { type: Component, args: [{
                    selector: 'tb-truncated-path',
                    template: `
    <span *ngIf="firstTextPart().length > 0" class="first-text-part">{{
      firstTextPart()
    }}</span>
    <span class="second-text-part">{{ secondTextPart() }}</span>
  `,
                    styleUrls: [`truncated_path_component.css`],
                },] },
    ];
    TruncatedPathComponent.propDecorators = {
        value: [{ type: Input }]
    };
    /** @nocollapse */ TruncatedPathComponent.ɵfac = function TruncatedPathComponent_Factory(t) { return new (t || TruncatedPathComponent)(); };
    /** @nocollapse */ TruncatedPathComponent.ɵcmp = ɵɵdefineComponent({ type: TruncatedPathComponent, selectors: [["tb-truncated-path"]], inputs: { value: "value" }, decls: 3, vars: 2, consts: [["class", "first-text-part", 4, "ngIf"], [1, "second-text-part"], [1, "first-text-part"]], template: function TruncatedPathComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵtemplate(0, TruncatedPathComponent_span_0_Template, 2, 1, "span", 0);
            ɵɵelementStart(1, "span", 1);
            ɵɵtext(2);
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵproperty("ngIf", ctx.firstTextPart().length > 0);
            ɵɵadvance(2);
            ɵɵtextInterpolate(ctx.secondTextPart());
        } }, directives: [NgIf], styles: ["[_nghost-%COMP%]{display:inline-flex;white-space:nowrap}.first-text-part[_ngcontent-%COMP%]{flex:1 1 4ch;max-width:max-content}.first-text-part[_ngcontent-%COMP%], .second-text-part[_ngcontent-%COMP%]{overflow:hidden;text-overflow:ellipsis}"] });

    function getDisplayNameForRun(runId, run, experimentName) {
        if (!run && !experimentName) {
            return runId;
        }
        const displayName = [experimentName, run ? run.name : '...']
            .filter(Boolean)
            .join('/');
        return displayName;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/views/card_renderer/run_name_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class RunNameComponent {
    }
    RunNameComponent.decorators = [
        { type: Component, args: [{
                    selector: 'card-run-name-component',
                    template: '{{ name }}',
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    RunNameComponent.propDecorators = {
        name: [{ type: Input }]
    };
    /** @nocollapse */ RunNameComponent.ɵfac = function RunNameComponent_Factory(t) { return new (t || RunNameComponent)(); };
    /** @nocollapse */ RunNameComponent.ɵcmp = ɵɵdefineComponent({ type: RunNameComponent, selectors: [["card-run-name-component"]], inputs: { name: "name" }, decls: 1, vars: 1, template: function RunNameComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵtext(0);
        } if (rf & 2) {
            ɵɵtextInterpolate(ctx.name);
        } }, encapsulation: 2, changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/views/card_renderer/run_name_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class RunNameContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
        }
        /**
         * Build observables once runId is defined (after onInit).
         * @return {?}
         */
        ngOnInit() {
            this.name$ = combineLatest([
                this.store.select(getRun, { runId: this.runId }),
                this.store.select(getExperimentIdForRunId, { runId: this.runId }),
                this.store.select(getExperimentIdToAliasMap),
            ]).pipe(map((/**
             * @param {?} __0
             * @return {?}
             */
            ([run, experimentId, idToAlias]) => {
                return getDisplayNameForRun(this.runId, run, experimentId ? idToAlias[experimentId] : null);
            })));
        }
    }
    RunNameContainer.decorators = [
        { type: Component, args: [{
                    selector: 'card-run-name',
                    template: `
    <card-run-name-component
      [name]="name$ | async"
      [attr.title]="name$ | async"
    ></card-run-name-component>
  `,
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    RunNameContainer.ctorParameters = () => [
        { type: Store }
    ];
    RunNameContainer.propDecorators = {
        runId: [{ type: Input }]
    };
    /** @nocollapse */ RunNameContainer.ɵfac = function RunNameContainer_Factory(t) { return new (t || RunNameContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ RunNameContainer.ɵcmp = ɵɵdefineComponent({ type: RunNameContainer, selectors: [["card-run-name"]], inputs: { runId: "runId" }, decls: 3, vars: 6, consts: [[3, "name"]], template: function RunNameContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelement(0, "card-run-name-component", 0);
            ɵɵpipe(1, "async");
            ɵɵpipe(2, "async");
        } if (rf & 2) {
            ɵɵproperty("name", ɵɵpipeBind1(1, 2, ctx.name$));
            ɵɵattribute("title", ɵɵpipeBind1(2, 4, ctx.name$));
        } }, directives: [RunNameComponent], pipes: [AsyncPipe], encapsulation: 2, changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/views/card_renderer/image_card_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    var I18N_0$2;
    if (typeof ngI18nClosureMode !== "undefined" && ngI18nClosureMode) {
        /**
         * @desc A button to pin a card.
         */ 
        const MSG_EXTERNAL_7284606426234375344$$TENSORBOARD_WEBAPP_METRICS_VIEWS_CARD_RENDERER_IMAGE_CARD_COMPONENT_TS_1 = goog.getMsg("Pin card");
        I18N_0$2 = MSG_EXTERNAL_7284606426234375344$$TENSORBOARD_WEBAPP_METRICS_VIEWS_CARD_RENDERER_IMAGE_CARD_COMPONENT_TS_1;
    }
    else {
        I18N_0$2 = $localize `:A button to pin a card.␟e665dc712bd5f18d4dfa3a29e125d565cc51e2f6␟7284606426234375344:Pin card`;
    }
    const _c2$8 = ["aria-label", I18N_0$2];
    var I18N_3;
    if (typeof ngI18nClosureMode !== "undefined" && ngI18nClosureMode) {
        /**
         * @desc A button on an image card that toggles actual image size.
         */ 
        const MSG_EXTERNAL_7635101936664789140$$TENSORBOARD_WEBAPP_METRICS_VIEWS_CARD_RENDERER_IMAGE_CARD_COMPONENT_TS_4 = goog.getMsg("Toggle actual image size");
        I18N_3 = MSG_EXTERNAL_7635101936664789140$$TENSORBOARD_WEBAPP_METRICS_VIEWS_CARD_RENDERER_IMAGE_CARD_COMPONENT_TS_4;
    }
    else {
        I18N_3 = $localize `:A button on an image card that toggles actual image size.␟3ca05ef3a6e3a37065f5e0f69c5d5a2178d90791␟7635101936664789140:Toggle actual image size`;
    }
    const _c5$2 = ["aria-label", I18N_3];
    function ImageCardComponent_div_5_mat_spinner_2_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "mat-spinner", 15);
    } }
    function ImageCardComponent_div_5_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 13);
        ɵɵtext(1);
        ɵɵtemplate(2, ImageCardComponent_div_5_mat_spinner_2_Template, 1, 0, "mat-spinner", 14);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r0 = ɵɵnextContext();
        ɵɵadvance(1);
        ɵɵtextInterpolate1(" Step ", ctx_r0.stepValues[ctx_r0.stepIndex], " ");
        ɵɵadvance(1);
        ɵɵproperty("ngIf", ctx_r0.loadState === ctx_r0.DataLoadState.LOADING);
    } }
    const _c6$1 = function (a0) { return { "filter": a0 }; };
    function ImageCardComponent_ng_container_13_Template(rf, ctx) { if (rf & 1) {
        const _r6 = ɵɵgetCurrentView();
        ɵɵelementContainerStart(0);
        ɵɵelementStart(1, "div", 16);
        ɵɵelementStart(2, "mat-slider", 17);
        ɵɵlistener("input", function ImageCardComponent_ng_container_13_Template_mat_slider_input_2_listener($event) { ɵɵrestoreView(_r6); const ctx_r5 = ɵɵnextContext(); return ctx_r5.onSliderInput($event); });
        ɵɵelementEnd();
        ɵɵelementEnd();
        ɵɵelementStart(3, "div", 18);
        ɵɵelement(4, "img", 19);
        ɵɵelementEnd();
        ɵɵelementContainerEnd();
    } if (rf & 2) {
        const ctx_r1 = ɵɵnextContext();
        ɵɵadvance(2);
        ɵɵproperty("disabled", ctx_r1.stepValues.length <= 1)("min", 0)("max", ctx_r1.stepValues.length - 1)("step", 1)("tickInterval", 1)("value", ctx_r1.stepIndex);
        ɵɵadvance(2);
        ɵɵpropertyInterpolate1("alt", "Image at step ", ctx_r1.stepValues[ctx_r1.stepIndex], "");
        ɵɵpropertyInterpolate("src", ctx_r1.imageUrl, ɵɵsanitizeUrl);
        ɵɵproperty("ngStyle", ɵɵpureFunction1(9, _c6$1, ctx_r1.cssFilter()));
    } }
    function ImageCardComponent_ng_template_14_div_0_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 21);
        ɵɵtext(1, " Data failed to load. ");
        ɵɵelementEnd();
    } }
    function ImageCardComponent_ng_template_14_Template(rf, ctx) { if (rf & 1) {
        ɵɵtemplate(0, ImageCardComponent_ng_template_14_div_0_Template, 2, 0, "div", 20);
    } if (rf & 2) {
        const ctx_r3 = ɵɵnextContext();
        ɵɵproperty("ngIf", ctx_r3.loadState === ctx_r3.DataLoadState.FAILED);
    } }
    const _c7$1 = function (a0) { return { backgroundColor: a0 }; };
    class ImageCardComponent {
        constructor() {
            this.DataLoadState = DataLoadState;
            this.onActualSizeToggle = new EventEmitter();
            this.stepIndexChange = new EventEmitter();
            this.onPinClicked = new EventEmitter();
        }
        /**
         * @return {?}
         */
        cssFilter() {
            /** @type {?} */
            const brightnessScale = this.brightnessInMilli / 1000;
            /** @type {?} */
            const contrastPercent = this.contrastInMilli / 10;
            return `contrast(${contrastPercent}%) brightness(${brightnessScale})`;
        }
        /**
         * @param {?} $event
         * @return {?}
         */
        onSliderInput($event) {
            // Angular Material Slider's MatSliderChange has a loose `number | null`
            // type for 'value'. However, it's actual implementation can only emit a
            // `number` on input events.
            // https://github.com/angular/components/blob/master/src/material/slider/slider.ts
            this.stepIndexChange.emit((/** @type {?} */ ($event.value)));
        }
    }
    ImageCardComponent.decorators = [
        { type: Component, args: [{
                    selector: 'image-card-component',
                    templateUrl: 'image_card_component.ng.html',
                    styleUrls: ['image_card_component.css'],
                    host: {
                        '[class.actual-size]': 'showActualSize',
                    },
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    ImageCardComponent.propDecorators = {
        loadState: [{ type: Input }],
        title: [{ type: Input }],
        tag: [{ type: Input }],
        runId: [{ type: Input }],
        sample: [{ type: Input }],
        imageUrl: [{ type: Input }],
        stepIndex: [{ type: Input }],
        stepValues: [{ type: Input }],
        brightnessInMilli: [{ type: Input }],
        contrastInMilli: [{ type: Input }],
        showActualSize: [{ type: Input }],
        runColorScale: [{ type: Input }],
        allowToggleActualSize: [{ type: Input }],
        isPinned: [{ type: Input }],
        onActualSizeToggle: [{ type: Output }],
        stepIndexChange: [{ type: Output }],
        onPinClicked: [{ type: Output }]
    };
    /** @nocollapse */ ImageCardComponent.ɵfac = function ImageCardComponent_Factory(t) { return new (t || ImageCardComponent)(); };
    /** @nocollapse */ ImageCardComponent.ɵcmp = ɵɵdefineComponent({ type: ImageCardComponent, selectors: [["image-card-component"]], hostVars: 2, hostBindings: function ImageCardComponent_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵclassProp("actual-size", ctx.showActualSize);
        } }, inputs: { loadState: "loadState", title: "title", tag: "tag", runId: "runId", sample: "sample", imageUrl: "imageUrl", stepIndex: "stepIndex", stepValues: "stepValues", brightnessInMilli: "brightnessInMilli", contrastInMilli: "contrastInMilli", showActualSize: "showActualSize", runColorScale: "runColorScale", allowToggleActualSize: "allowToggleActualSize", isPinned: "isPinned" }, outputs: { onActualSizeToggle: "onActualSizeToggle", stepIndexChange: "stepIndexChange", onPinClicked: "onPinClicked" }, decls: 16, vars: 12, consts: [[1, "heading"], [1, "tag", 3, "title", "value"], [1, "run"], [1, "dot", 3, "ngStyle"], [1, "run-text", 3, "runId"], ["class", "step", 4, "ngIf"], [1, "controls"], ["mat-icon-button", "", 1, "pin-button", 3, "click", 6, "aria-label"], [3, "svgIcon"], ["mat-icon-button", "", "title", "Toggle actual image size", 3, "disabled", "click", 6, "aria-label"], ["svgIcon", "image_search_24px"], [4, "ngIf", "ngIfElse"], ["noImageData", ""], [1, "step"], ["diameter", "18", 4, "ngIf"], ["diameter", "18"], [1, "slider-row"], ["color", "primary", 1, "step-slider", 3, "disabled", "min", "max", "step", "tickInterval", "value", "input"], [1, "img-container"], [3, "alt", "src", "ngStyle"], ["class", "empty-message", 4, "ngIf"], [1, "empty-message"]], template: function ImageCardComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "div", 0);
            ɵɵelement(1, "tb-truncated-path", 1);
            ɵɵelementStart(2, "div", 2);
            ɵɵelement(3, "span", 3);
            ɵɵelement(4, "card-run-name", 4);
            ɵɵelementEnd();
            ɵɵtemplate(5, ImageCardComponent_div_5_Template, 3, 2, "div", 5);
            ɵɵelementStart(6, "span", 6);
            ɵɵelementStart(7, "button", 7);
            ɵɵi18nAttributes(8, _c2$8);
            ɵɵlistener("click", function ImageCardComponent_Template_button_click_7_listener() { return ctx.onPinClicked.emit(!ctx.isPinned); });
            ɵɵelement(9, "mat-icon", 8);
            ɵɵelementEnd();
            ɵɵelementStart(10, "button", 9);
            ɵɵi18nAttributes(11, _c5$2);
            ɵɵlistener("click", function ImageCardComponent_Template_button_click_10_listener() { return ctx.onActualSizeToggle.emit(); });
            ɵɵelement(12, "mat-icon", 10);
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵtemplate(13, ImageCardComponent_ng_container_13_Template, 5, 11, "ng-container", 11);
            ɵɵtemplate(14, ImageCardComponent_ng_template_14_Template, 1, 1, "ng-template", null, 12, ɵɵtemplateRefExtractor);
        } if (rf & 2) {
            const _r2 = ɵɵreference(15);
            ɵɵadvance(1);
            ɵɵpropertyInterpolate("title", ctx.tag);
            ɵɵpropertyInterpolate("value", ctx.title);
            ɵɵadvance(2);
            ɵɵproperty("ngStyle", ɵɵpureFunction1(10, _c7$1, ctx.runColorScale(ctx.runId)));
            ɵɵadvance(1);
            ɵɵproperty("runId", ctx.runId);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx.stepIndex !== null && ctx.stepIndex < ctx.stepValues.length);
            ɵɵadvance(2);
            ɵɵattribute("title", ctx.isPinned ? "Unpin card" : "Pin card");
            ɵɵadvance(2);
            ɵɵproperty("svgIcon", ctx.isPinned ? "keep_24px" : "keep_outline_24px");
            ɵɵadvance(1);
            ɵɵproperty("disabled", !ctx.allowToggleActualSize);
            ɵɵadvance(3);
            ɵɵproperty("ngIf", ctx.stepIndex !== null && ctx.stepIndex < ctx.stepValues.length)("ngIfElse", _r2);
        } }, directives: [TruncatedPathComponent, NgStyle, RunNameContainer, NgIf, MatButton, MatIcon, MatSpinner, MatSlider], styles: [".mat-badge-content[_ngcontent-%COMP%]{font-weight:600;font-size:12px;font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:9px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:24px}.mat-h1[_ngcontent-%COMP%], .mat-headline[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%]{font:400 24px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h2[_ngcontent-%COMP%], .mat-title[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h3[_ngcontent-%COMP%], .mat-subheading-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%]{font:400 16px/28px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h4[_ngcontent-%COMP%], .mat-subheading-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%]{font:400 15px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h5[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%]{font:400 calc(14px * 0.83)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-h6[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:400 calc(14px * 0.67)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-body-strong[_ngcontent-%COMP%], .mat-body-2[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   p[_ngcontent-%COMP%]{margin:0 0 12px}.mat-small[_ngcontent-%COMP%], .mat-caption[_ngcontent-%COMP%]{font:400 12px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-display-4[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-4[_ngcontent-%COMP%]{font:300 112px/112px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.05em;margin:0 0 56px}.mat-display-3[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-3[_ngcontent-%COMP%]{font:400 56px/56px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.02em;margin:0 0 64px}.mat-display-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-2[_ngcontent-%COMP%]{font:400 45px/48px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.005em;margin:0 0 64px}.mat-display-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-1[_ngcontent-%COMP%]{font:400 34px/40px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 64px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%], .mat-flat-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-button-toggle[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card-title[_ngcontent-%COMP%]{font-size:24px;font-weight:500}.mat-card-header[_ngcontent-%COMP%]   .mat-card-title[_ngcontent-%COMP%]{font-size:20px}.mat-card-subtitle[_ngcontent-%COMP%], .mat-card-content[_ngcontent-%COMP%]{font-size:14px}.mat-checkbox[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-checkbox-layout[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{line-height:24px}.mat-chip[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-chip[_ngcontent-%COMP%]   .mat-chip-trailing-icon.mat-icon[_ngcontent-%COMP%], .mat-chip[_ngcontent-%COMP%]   .mat-chip-remove.mat-icon[_ngcontent-%COMP%]{font-size:18px}.mat-table[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-header-cell[_ngcontent-%COMP%]{font-size:12px;font-weight:500}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{font-size:14px}.mat-calendar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-calendar-body[_ngcontent-%COMP%]{font-size:13px}.mat-calendar-body-label[_ngcontent-%COMP%], .mat-calendar-period-button[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-calendar-table-header[_ngcontent-%COMP%]   th[_ngcontent-%COMP%]{font-size:11px;font-weight:400}.mat-dialog-title[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-expansion-panel-header[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:15px;font-weight:400}.mat-expansion-panel-content[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field[_ngcontent-%COMP%]{font-size:inherit;font-weight:400;line-height:1.125;font-family:Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.34375em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{font-size:150%;line-height:1.125}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]{height:1.5em;width:1.5em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{height:1.125em;line-height:1.125}.mat-form-field-infix[_ngcontent-%COMP%]{padding:.5em 0;border-top:.84375em solid transparent}.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34375em) scale(0.75);width:133.3333333333%}.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34374em) scale(0.75);width:133.3333433333%}.mat-form-field-label-wrapper[_ngcontent-%COMP%]{top:-0.84375em;padding-top:.84375em}.mat-form-field-label[_ngcontent-%COMP%]{top:1.34375em}.mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.34375em}.mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{font-size:75%;margin-top:.6666666667em;top:calc(100% - 1.7916666667em)}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.4375em 0}.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.001px);-ms-transform:translateY(-1.28125em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00101px);-ms-transform:translateY(-1.28124em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00102px);-ms-transform:translateY(-1.28123em) scale(0.75);width:133.3333533333%}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.28125em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{margin-top:.5416666667em;top:calc(100% - 1.6666666667em)}@media print{.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28122em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28121em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.2812em) scale(0.75)}}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.25em 0 .75em 0}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.09375em;margin-top:-0.5em}.mat-form-field-appearance-fill.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59374em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:1em 0 1em 0}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.84375em;margin-top:-0.25em}.mat-form-field-appearance-outline.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59374em) scale(0.75);width:133.3333433333%}.mat-grid-tile-header[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]{font-size:14px}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2), .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}input.mat-input-element[_ngcontent-%COMP%]{margin-top:-0.0625em}.mat-menu-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:400}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px}.mat-radio-button[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select-trigger[_ngcontent-%COMP%]{height:1.125em}.mat-slide-toggle-content[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-slider-thumb-label-text[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-stepper-vertical[_ngcontent-%COMP%], .mat-stepper-horizontal[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-step-label[_ngcontent-%COMP%]{font-size:14px;font-weight:400}.mat-step-sub-label-error[_ngcontent-%COMP%]{font-weight:normal}.mat-step-label-error[_ngcontent-%COMP%]{font-size:14px}.mat-step-label-selected[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-tab-group[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-toolbar[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0}.mat-tooltip[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:10px;padding-top:6px;padding-bottom:6px}.mat-tooltip-handset[_ngcontent-%COMP%]{font-size:14px;padding-top:8px;padding-bottom:8px}.mat-list-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:16px}.mat-optgroup-label[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-simple-snackbar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px}.mat-simple-snackbar-action[_ngcontent-%COMP%]{line-height:1;font-family:inherit;font-size:inherit;font-weight:500}.mat-tree[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{font-weight:400;font-size:14px}.mat-ripple[_ngcontent-%COMP%]{overflow:hidden;position:relative}.mat-ripple[_ngcontent-%COMP%]:not(:empty){transform:translateZ(0)}.mat-ripple.mat-ripple-unbounded[_ngcontent-%COMP%]{overflow:visible}.mat-ripple-element[_ngcontent-%COMP%]{position:absolute;border-radius:50%;pointer-events:none;transition:opacity,transform 0ms cubic-bezier(0, 0, 0.2, 1);transform:scale(0)}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{display:none}.cdk-visually-hidden[_ngcontent-%COMP%]{border:0;clip:rect(0 0 0 0);height:1px;margin:-1px;overflow:hidden;padding:0;position:absolute;width:1px;outline:0;-webkit-appearance:none;-moz-appearance:none}.cdk-overlay-container[_ngcontent-%COMP%], .cdk-global-overlay-wrapper[_ngcontent-%COMP%]{pointer-events:none;top:0;left:0;height:100%;width:100%}.cdk-overlay-container[_ngcontent-%COMP%]{position:fixed;z-index:1000}.cdk-overlay-container[_ngcontent-%COMP%]:empty{display:none}.cdk-global-overlay-wrapper[_ngcontent-%COMP%]{display:flex;position:absolute;z-index:1000}.cdk-overlay-pane[_ngcontent-%COMP%]{position:absolute;pointer-events:auto;box-sizing:border-box;z-index:1000;display:flex;max-width:100%;max-height:100%}.cdk-overlay-backdrop[_ngcontent-%COMP%]{position:absolute;top:0;bottom:0;left:0;right:0;z-index:1000;pointer-events:auto;-webkit-tap-highlight-color:transparent;transition:opacity 400ms cubic-bezier(0.25, 0.8, 0.25, 1);opacity:0}.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:1}@media screen and (-ms-high-contrast: active){.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:.6}}.cdk-overlay-dark-backdrop[_ngcontent-%COMP%]{background:rgba(0,0,0,.32)}.cdk-overlay-transparent-backdrop[_ngcontent-%COMP%], .cdk-overlay-transparent-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:0}.cdk-overlay-connected-position-bounding-box[_ngcontent-%COMP%]{position:absolute;z-index:1000;display:flex;flex-direction:column;min-width:1px;min-height:1px}.cdk-global-scrollblock[_ngcontent-%COMP%]{position:fixed;width:100%;overflow-y:scroll}@keyframes cdk-text-field-autofill-start{}@keyframes cdk-text-field-autofill-end{}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:-webkit-autofill{animation:cdk-text-field-autofill-start 0s 1ms}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:not(:-webkit-autofill){animation:cdk-text-field-autofill-end 0s 1ms}textarea.cdk-textarea-autosize[_ngcontent-%COMP%]{resize:none}textarea.cdk-textarea-autosize-measuring[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:auto !important;overflow:hidden !important}textarea.cdk-textarea-autosize-measuring-firefox[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:0 !important}.mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.1)}.mat-option[_ngcontent-%COMP%]{color:#212121}.mat-option[_ngcontent-%COMP%]:hover:not(.mat-option-disabled), .mat-option[_ngcontent-%COMP%]:focus:not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple):not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-active[_ngcontent-%COMP%]{background:rgba(0,0,0,.04);color:#212121}.mat-option.mat-option-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-primary[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f57c00}.mat-accent[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f44336}.mat-optgroup-label[_ngcontent-%COMP%]{color:#616161}.mat-optgroup-disabled[_ngcontent-%COMP%]   .mat-optgroup-label[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-pseudo-checkbox[_ngcontent-%COMP%]{color:#616161}.mat-pseudo-checkbox[_ngcontent-%COMP%]::after{color:#fafafa}.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{color:#b0b0b0}.mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f57c00}.mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f44336}.mat-pseudo-checkbox-checked.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{background:#b0b0b0}.mat-elevation-z0[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z1[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z2[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z3[_ngcontent-%COMP%]{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z4[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z5[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 5px 8px 0px rgba(0, 0, 0, 0.14),0px 1px 14px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z6[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z7[_ngcontent-%COMP%]{box-shadow:0px 4px 5px -2px rgba(0, 0, 0, 0.2),0px 7px 10px 1px rgba(0, 0, 0, 0.14),0px 2px 16px 1px rgba(0, 0, 0, 0.12)}.mat-elevation-z8[_ngcontent-%COMP%]{box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z9[_ngcontent-%COMP%]{box-shadow:0px 5px 6px -3px rgba(0, 0, 0, 0.2),0px 9px 12px 1px rgba(0, 0, 0, 0.14),0px 3px 16px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z10[_ngcontent-%COMP%]{box-shadow:0px 6px 6px -3px rgba(0, 0, 0, 0.2),0px 10px 14px 1px rgba(0, 0, 0, 0.14),0px 4px 18px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z11[_ngcontent-%COMP%]{box-shadow:0px 6px 7px -4px rgba(0, 0, 0, 0.2),0px 11px 15px 1px rgba(0, 0, 0, 0.14),0px 4px 20px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z12[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z13[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 13px 19px 2px rgba(0, 0, 0, 0.14),0px 5px 24px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z14[_ngcontent-%COMP%]{box-shadow:0px 7px 9px -4px rgba(0, 0, 0, 0.2),0px 14px 21px 2px rgba(0, 0, 0, 0.14),0px 5px 26px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z15[_ngcontent-%COMP%]{box-shadow:0px 8px 9px -5px rgba(0, 0, 0, 0.2),0px 15px 22px 2px rgba(0, 0, 0, 0.14),0px 6px 28px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z16[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z17[_ngcontent-%COMP%]{box-shadow:0px 8px 11px -5px rgba(0, 0, 0, 0.2),0px 17px 26px 2px rgba(0, 0, 0, 0.14),0px 6px 32px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z18[_ngcontent-%COMP%]{box-shadow:0px 9px 11px -5px rgba(0, 0, 0, 0.2),0px 18px 28px 2px rgba(0, 0, 0, 0.14),0px 7px 34px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z19[_ngcontent-%COMP%]{box-shadow:0px 9px 12px -6px rgba(0, 0, 0, 0.2),0px 19px 29px 2px rgba(0, 0, 0, 0.14),0px 7px 36px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z20[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 20px 31px 3px rgba(0, 0, 0, 0.14),0px 8px 38px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z21[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 21px 33px 3px rgba(0, 0, 0, 0.14),0px 8px 40px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z22[_ngcontent-%COMP%]{box-shadow:0px 10px 14px -6px rgba(0, 0, 0, 0.2),0px 22px 35px 3px rgba(0, 0, 0, 0.14),0px 8px 42px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z23[_ngcontent-%COMP%]{box-shadow:0px 11px 14px -7px rgba(0, 0, 0, 0.2),0px 23px 36px 3px rgba(0, 0, 0, 0.14),0px 9px 44px 8px rgba(0, 0, 0, 0.12)}.mat-elevation-z24[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12)}.mat-app-background[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-theme-loaded-marker[_ngcontent-%COMP%]{display:none}.mat-autocomplete-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-autocomplete-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover){background:#fff}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover):not(.mat-option-disabled){color:#212121}.mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f57c00}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{outline:solid 1px;border-radius:0}.mat-badge-accent[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-badge-warn[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f44336}.mat-badge[_ngcontent-%COMP%]{position:relative}.mat-badge-hidden[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{display:none}.mat-badge-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#b9b9b9;color:#757575}.mat-badge-content[_ngcontent-%COMP%]{position:absolute;text-align:center;display:inline-block;border-radius:50%;transition:transform 200ms ease-in-out;transform:scale(0.6);overflow:hidden;white-space:nowrap;text-overflow:ellipsis;pointer-events:none}.ng-animate-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%], .mat-badge-content._mat-animation-noopable[_ngcontent-%COMP%]{transition:none}.mat-badge-content.mat-badge-active[_ngcontent-%COMP%]{transform:none}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:16px;height:16px;line-height:16px}.mat-badge-small.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-8px}.mat-badge-small.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-8px}.mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-16px}.mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-16px}.mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-8px}.mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-8px}.mat-badge-medium[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:22px;height:22px;line-height:22px}.mat-badge-medium.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-11px}.mat-badge-medium.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-11px}.mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-22px}.mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-22px}.mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-11px}.mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-11px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:28px;height:28px;line-height:28px}.mat-badge-large.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-14px}.mat-badge-large.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-14px}.mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-28px}.mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-28px}.mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-14px}.mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-14px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]{color:inherit;background:transparent}.mat-button.mat-primary[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-button[disabled][disabled][_ngcontent-%COMP%], .mat-icon-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-icon-button[disabled][disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-stroked-button[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f44336}.mat-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:transparent}.mat-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{opacity:.1;background-color:currentColor}.mat-button-focus-overlay[_ngcontent-%COMP%]{background:#000}.mat-stroked-button[_ngcontent-%COMP%]:not([disabled]){border-color:rgba(0,0,0,.12)}.mat-flat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{color:#212121;background-color:#fff}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-flat-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-stroked-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-flat-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-raised-button[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone[_ngcontent-%COMP%], .mat-button-toggle-group[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{box-shadow:none}.mat-button-toggle[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-button-toggle[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121;background:#fff}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:#000}.mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12)}[dir=rtl][_ngcontent-%COMP%]   .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-group-appearance-standard.mat-button-toggle-vertical[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:none;border-top:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#616161}.mat-button-toggle-checked.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121}.mat-button-toggle-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.26);background-color:#eee}.mat-button-toggle-disabled.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{background:#fff}.mat-button-toggle-disabled.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#bdbdbd}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{border:solid 1px rgba(0,0,0,.12)}.mat-card[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-card[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-card.mat-card-flat[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-card-subtitle[_ngcontent-%COMP%]{color:#616161}.mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#616161}.mat-checkbox-checkmark[_ngcontent-%COMP%]{fill:#fafafa}.mat-checkbox-checkmark-path[_ngcontent-%COMP%]{stroke:#fafafa !important}.mat-checkbox-mixedmark[_ngcontent-%COMP%]{background-color:#fafafa}.mat-checkbox-indeterminate.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f57c00}.mat-checkbox-indeterminate.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#ff9800}.mat-checkbox-indeterminate.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f44336}.mat-checkbox-disabled.mat-checkbox-checked[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-disabled.mat-checkbox-indeterminate[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]:not(.mat-checkbox-checked)   .mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{color:#616161}.mat-checkbox[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%]{background:#f57c00}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%]{background:#ff9800}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%]{background:#f44336}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#212121}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#212121;opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled):active{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled)   .mat-chip-remove[_ngcontent-%COMP%]:hover{opacity:.54}.mat-chip.mat-standard-chip.mat-chip-disabled[_ngcontent-%COMP%]{opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]::after{background:#000}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-table[_ngcontent-%COMP%]{background:#fff}.mat-table[_ngcontent-%COMP%]   thead[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tbody[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tfoot[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], [mat-header-row][_ngcontent-%COMP%], [mat-row][_ngcontent-%COMP%], [mat-footer-row][_ngcontent-%COMP%], .mat-table-sticky[_ngcontent-%COMP%]{background:inherit}mat-row[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], th.mat-header-cell[_ngcontent-%COMP%], td.mat-cell[_ngcontent-%COMP%], td.mat-footer-cell[_ngcontent-%COMP%]{border-bottom-color:rgba(0,0,0,.12)}.mat-header-cell[_ngcontent-%COMP%]{color:#616161}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{color:#212121}.mat-calendar-arrow[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.54)}.mat-datepicker-toggle[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-next-button[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-previous-button[_ngcontent-%COMP%]{color:rgba(0,0,0,.54)}.mat-calendar-table-header[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-calendar-table-header-divider[_ngcontent-%COMP%]::after{background:rgba(0,0,0,.12)}.mat-calendar-body-label[_ngcontent-%COMP%]{color:#616161}.mat-calendar-body-cell-content[_ngcontent-%COMP%]{color:#212121;border-color:transparent}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){color:#757575}.mat-calendar-body-cell[_ngcontent-%COMP%]:not(.mat-calendar-body-disabled):hover > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-keyboard-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-program-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){background-color:rgba(0,0,0,.04)}.mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.38)}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.18)}.mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.4)}.mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12);background-color:#fff;color:#212121}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.4)}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.4)}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content-touch[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-datepicker-toggle-active[_ngcontent-%COMP%]{color:#f57c00}.mat-datepicker-toggle-active.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-datepicker-toggle-active.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-dialog-container[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-divider[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-divider-vertical[_ngcontent-%COMP%]{border-right-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-expansion-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-action-row[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-keyboard-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-program-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded)   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover:not([aria-disabled=true]){background:rgba(0,0,0,.04)}@media(hover: none){.mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded):not([aria-disabled=true])   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover{background:#fff}}.mat-expansion-panel-header-title[_ngcontent-%COMP%]{color:#212121}.mat-expansion-panel-header-description[_ngcontent-%COMP%], .mat-expansion-indicator[_ngcontent-%COMP%]::after{color:#616161}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-title[_ngcontent-%COMP%], .mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-description[_ngcontent-%COMP%]{color:inherit}.mat-form-field-label[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-hint[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-focused[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-ripple[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid)   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f57c00}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-accent   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#ff9800}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-warn   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#f44336}.mat-error[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-hint[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-legacy.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-standard[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-standard.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.04)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.02)}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:transparent}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.12)}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:rgba(0,0,0,.87)}.mat-form-field-appearance-outline.mat-focused[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field-appearance-outline.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-appearance-outline.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-invalid.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.06)}.mat-icon.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-icon.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-icon.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-type-mat-native-select[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#616161}.mat-input-element[_ngcontent-%COMP%]:disabled, .mat-form-field-type-mat-native-select.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#757575}.mat-input-element[_ngcontent-%COMP%]{caret-color:#f57c00}.mat-input-element[_ngcontent-%COMP%]::placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-moz-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-webkit-input-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]:-ms-input-placeholder{color:rgba(97,97,97,.42)}.mat-accent[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%], .mat-form-field-invalid[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#f44336}.mat-form-field-type-mat-native-select.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{color:#616161}.mat-list-item-disabled[_ngcontent-%COMP%]{background-color:#eee}.mat-list-option[_ngcontent-%COMP%]:hover, .mat-list-option[_ngcontent-%COMP%]:focus, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.04)}.mat-list-single-selected-option[_ngcontent-%COMP%], .mat-list-single-selected-option[_ngcontent-%COMP%]:hover, .mat-list-single-selected-option[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.12)}.mat-menu-panel[_ngcontent-%COMP%]{background:#fff}.mat-menu-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-menu-item[_ngcontent-%COMP%]{background:transparent;color:#212121}.mat-menu-item[disabled][_ngcontent-%COMP%], .mat-menu-item[disabled][_ngcontent-%COMP%]::after{color:rgba(0,0,0,.38)}.mat-menu-item[_ngcontent-%COMP%]   .mat-icon-no-color[_ngcontent-%COMP%], .mat-menu-item-submenu-trigger[_ngcontent-%COMP%]::after{color:rgba(0,0,0,.54)}.mat-menu-item[_ngcontent-%COMP%]:hover:not([disabled]), .mat-menu-item.cdk-program-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item.cdk-keyboard-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item-highlighted[_ngcontent-%COMP%]:not([disabled]){background:rgba(0,0,0,.04)}.mat-paginator[_ngcontent-%COMP%]{background:#fff}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{color:#616161}.mat-paginator-decrement[_ngcontent-%COMP%], .mat-paginator-increment[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54);border-right:2px solid rgba(0,0,0,.54)}.mat-paginator-first[_ngcontent-%COMP%], .mat-paginator-last[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54)}.mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-decrement[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-increment[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-first[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-last[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f57c00}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#ff9800}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f44336}.mat-progress-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f57c00}.mat-progress-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#ff9800}.mat-progress-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f44336}.mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#616161}.mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f57c00}.mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#ff9800}.mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#ff9800}.mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f44336}.mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f44336}.mat-radio-button.mat-radio-disabled.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-label-content[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-radio-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-select-value[_ngcontent-%COMP%]{color:#212121}.mat-select-placeholder[_ngcontent-%COMP%]{color:rgba(97,97,97,.42)}.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%]{color:#757575}.mat-select-arrow[_ngcontent-%COMP%]{color:#616161}.mat-select-panel[_ngcontent-%COMP%]{background:#fff}.mat-select-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-select-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple){background:rgba(0,0,0,.12)}.mat-form-field.mat-focused.mat-primary[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-invalid[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#757575}.mat-drawer-container[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-drawer[_ngcontent-%COMP%]{background-color:#fff;color:#212121}.mat-drawer.mat-drawer-push[_ngcontent-%COMP%]{background-color:#fff}.mat-drawer[_ngcontent-%COMP%]:not(.mat-drawer-side){box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-drawer-side[_ngcontent-%COMP%]{border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-backdrop.mat-drawer-shown[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.6)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.54)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.54)}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.54)}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle[_ngcontent-%COMP%]:not(.mat-checked)   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-slide-toggle-thumb[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12);background-color:#fafafa}.mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-primary[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f57c00}.mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-accent[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#ff9800}.mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-warn[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f44336}.mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.2)}.mat-slider[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%], .cdk-focused[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-disabled[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]   .mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing)   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26);background-color:transparent}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26)}.mat-slider-has-ticks[_ngcontent-%COMP%]   .mat-slider-wrapper[_ngcontent-%COMP%]::after{border-color:rgba(0,0,0,.7)}.mat-slider-horizontal[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to right, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent);background-image:-moz-repeating-linear-gradient(0.0001deg, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-slider-vertical[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to bottom, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-step-header.cdk-keyboard-focused[_ngcontent-%COMP%], .mat-step-header.cdk-program-focused[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]:hover{background-color:rgba(0,0,0,.04)}@media(hover: none){.mat-step-header[_ngcontent-%COMP%]:hover{background:none}}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-optional[_ngcontent-%COMP%]{color:#616161}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon[_ngcontent-%COMP%]{background-color:#616161;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-selected[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-done[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-edit[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-error[_ngcontent-%COMP%]{background-color:transparent;color:#f44336}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-active[_ngcontent-%COMP%]{color:#212121}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-error[_ngcontent-%COMP%]{color:#f44336}.mat-stepper-horizontal[_ngcontent-%COMP%], .mat-stepper-vertical[_ngcontent-%COMP%]{background-color:#fff}.mat-stepper-vertical-line[_ngcontent-%COMP%]::before{border-left-color:rgba(0,0,0,.12)}.mat-horizontal-stepper-header[_ngcontent-%COMP%]::before, .mat-horizontal-stepper-header[_ngcontent-%COMP%]::after, .mat-stepper-horizontal-line[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-sort-header-arrow[_ngcontent-%COMP%]{color:#616161}.mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-header[_ngcontent-%COMP%]{border-bottom:1px solid rgba(0,0,0,.12)}.mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%]{border-top:1px solid rgba(0,0,0,.12);border-bottom:none}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{color:#212121}.mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:#757575}.mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#212121}.mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#757575}.mat-tab-group[class*=mat-background-][_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar[class*=mat-background-][_ngcontent-%COMP%]{border-bottom:none;border-top:none}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-toolbar[_ngcontent-%COMP%]{background:#f5f5f5;color:#212121}.mat-toolbar.mat-primary[_ngcontent-%COMP%]{background:#f57c00;color:#fff}.mat-toolbar.mat-accent[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-toolbar.mat-warn[_ngcontent-%COMP%]{background:#f44336;color:#fff}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:currentColor}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:inherit}.mat-toolbar[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:currentColor}.mat-tooltip[_ngcontent-%COMP%]{background:rgba(97,97,97,.9)}.mat-tree[_ngcontent-%COMP%]{background:#fff}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{color:#212121}.mat-snack-bar-container[_ngcontent-%COMP%]{color:rgba(255,255,255,.7);background:#323232;box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-simple-snackbar-action[_ngcontent-%COMP%]{color:#ff9800}[_nghost-%COMP%]{display:flex;flex-direction:column;box-sizing:border-box;height:100%;overflow:auto;padding:16px;padding-top:4px}.actual-size[_nghost-%COMP%]{height:auto}.heading[_ngcontent-%COMP%]{align-items:center;display:grid;grid-template-areas:\"tag controls\" \"run step\";grid-template-columns:1fr auto;font-size:14px;margin-bottom:4px;position:relative}.tag[_ngcontent-%COMP%]{grid-area:tag;overflow:hidden}.pin-button[_ngcontent-%COMP%]   mat-icon[_ngcontent-%COMP%]{height:18px}.run[_ngcontent-%COMP%]{grid-area:run;display:flex;white-space:nowrap}.run[_ngcontent-%COMP%]   .dot[_ngcontent-%COMP%]{flex:none;display:inline-block;width:13px;height:13px;border-radius:50%;margin-right:4px}.run[_ngcontent-%COMP%]   .run-text[_ngcontent-%COMP%]{overflow:hidden;text-overflow:ellipsis;max-width:120px}.step[_ngcontent-%COMP%]{grid-area:step;display:flex}.step[_ngcontent-%COMP%]   mat-spinner[_ngcontent-%COMP%]{margin-left:4px;margin-right:0}.run[_ngcontent-%COMP%], .step[_ngcontent-%COMP%]{font-size:13px;color:#616161;height:15px}.controls[_ngcontent-%COMP%]{color:#616161;grid-area:controls;justify-self:flex-end;flex-shrink:0;margin-right:-12px;white-space:nowrap}.img-container[_ngcontent-%COMP%]{flex:1 1 0;overflow-y:auto;position:relative}.img-container[_ngcontent-%COMP%]   img[_ngcontent-%COMP%]{image-rendering:-moz-crisp-edges;image-rendering:pixelated}.actual-size[_nghost-%COMP%]   .img-container[_ngcontent-%COMP%]{overflow:auto;flex:none}[_nghost-%COMP%]:not(.actual-size)   img[_ngcontent-%COMP%]{position:absolute;max-height:100%;max-width:100%;width:auto;height:100%;object-fit:contain}.slider-row[_ngcontent-%COMP%]{display:flex;align-items:center;height:24px}.step-slider[_ngcontent-%COMP%]{flex:1}[_nghost-%COMP%]     .mat-slider-min-value .mat-slider-thumb{background-color:#f57c00}.empty-message[_ngcontent-%COMP%]{margin-top:1em;font-size:13px}"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/views/card_renderer/image_card_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class ImageCardContainer {
        /**
         * @param {?} store
         * @param {?} dataSource
         */
        constructor(store, dataSource) {
            this.store = store;
            this.dataSource = dataSource;
            this.fullWidthChanged = new EventEmitter();
            this.pinStateChanged = new EventEmitter();
            this.brightnessInMilli$ = this.store.select(getMetricsImageBrightnessInMilli);
            this.contrastInMilli$ = this.store.select(getMetricsImageContrastInMilli);
            this.actualSizeGlobalSetting$ = this.store.select(getMetricsImageShowActualSize);
            this.showActualSize = false;
            // The UI toggle is overridden by the global setting.
            this.actualSizeUiToggled = false;
            this.actualSizeUiToggleSubject = new BehaviorSubject(this.actualSizeUiToggled);
            this.ngUnsubscribe = new Subject();
        }
        /**
         * @param {?} $event
         * @return {?}
         */
        onStepIndexChanged($event) {
            this.store.dispatch(cardStepSliderChanged({
                cardId: this.cardId,
                stepIndex: $event,
            }));
        }
        /**
         * @private
         * @param {?} cardMetadata
         * @return {?}
         */
        isImageCardMetadata(cardMetadata) {
            const { plugin } = cardMetadata;
            return plugin === PluginType.IMAGES;
        }
        /**
         * @return {?}
         */
        onActualSizeToggle() {
            this.actualSizeUiToggled = !this.actualSizeUiToggled;
            this.actualSizeUiToggleSubject.next(this.actualSizeUiToggled);
        }
        /**
         * Build observables once cardId is defined (after onInit).
         * @return {?}
         */
        ngOnInit() {
            combineLatest([
                this.actualSizeGlobalSetting$,
                this.actualSizeUiToggleSubject,
            ])
                .pipe(takeUntil(this.ngUnsubscribe), tap((/**
             * @param {?} __0
             * @return {?}
             */
            ([settingEnabled, uiToggleEnabled]) => {
                // Actual size mode requires a full width card.
                this.showActualSize = settingEnabled || uiToggleEnabled;
                this.fullWidthChanged.emit(this.showActualSize);
            })))
                .subscribe((/**
             * @return {?}
             */
            () => { }));
            /** @type {?} */
            const selectCardMetadata$ = this.store.select(getCardMetadata, this.cardId);
            /** @type {?} */
            const cardMetadata$ = selectCardMetadata$.pipe(filter((/**
             * @param {?} cardMetadata
             * @return {?}
             */
            (cardMetadata) => {
                return !!cardMetadata && this.isImageCardMetadata(cardMetadata);
            })), map((/**
             * @param {?} cardMetadata
             * @return {?}
             */
            (cardMetadata) => {
                return (/** @type {?} */ (cardMetadata));
            })), shareReplay(1));
            /** @type {?} */
            const metadataAndSeries$ = combineLatest([
                cardMetadata$,
                this.store.select(getCardTimeSeries, this.cardId),
            ]);
            /** @type {?} */
            const timeSeries$ = metadataAndSeries$.pipe(map((/**
             * @param {?} __0
             * @return {?}
             */
            ([cardMetadata, runToSeries]) => {
                /** @type {?} */
                const runId = cardMetadata.runId;
                if (!runToSeries || !runToSeries.hasOwnProperty(runId)) {
                    return [];
                }
                return (/** @type {?} */ (runToSeries[runId]));
            })), distinctUntilChanged((/**
             * @param {?} series1
             * @param {?} series2
             * @return {?}
             */
            (series1, series2) => {
                if (series1.length === series2.length && series1.length === 0) {
                    return true;
                }
                return series1 === series2;
            })), shareReplay(1));
            this.stepIndex$ = this.store.select(getCardStepIndex, this.cardId);
            this.loadState$ = this.store.select(getCardLoadState, this.cardId);
            /** @type {?} */
            const timeSeriesAndStepIndex$ = combineLatest([
                timeSeries$,
                this.stepIndex$,
            ]);
            /** @type {?} */
            const stepDatum$ = timeSeriesAndStepIndex$.pipe(map((/**
             * @param {?} __0
             * @return {?}
             */
            ([timeSeries, stepIndex]) => {
                if (stepIndex === null || !timeSeries[stepIndex]) {
                    return null;
                }
                return timeSeries[stepIndex];
            })));
            this.tag$ = cardMetadata$.pipe(map((/**
             * @param {?} cardMetadata
             * @return {?}
             */
            (cardMetadata) => {
                return cardMetadata.tag;
            })));
            this.title$ = this.tag$.pipe(map((/**
             * @param {?} tag
             * @return {?}
             */
            (tag) => {
                return getTagDisplayName(tag, this.groupName);
            })));
            this.runId$ = cardMetadata$.pipe(map((/**
             * @param {?} cardMetadata
             * @return {?}
             */
            (cardMetadata) => {
                return cardMetadata.runId;
            })));
            this.sample$ = cardMetadata$.pipe(map((/**
             * @param {?} cardMetadata
             * @return {?}
             */
            (cardMetadata) => {
                return cardMetadata.sample;
            })));
            this.imageUrl$ = stepDatum$.pipe(map((/**
             * @param {?} stepDatum
             * @return {?}
             */
            (stepDatum) => {
                if (!stepDatum) {
                    return null;
                }
                return this.dataSource.imageUrl(stepDatum.imageId);
            })));
            this.stepValues$ = timeSeries$.pipe(map((/**
             * @param {?} series
             * @return {?}
             */
            (series) => {
                return series.map((/**
                 * @param {?} stepDatum
                 * @return {?}
                 */
                (stepDatum) => {
                    return stepDatum.step;
                }));
            })));
            this.isPinned$ = this.store.select(getCardPinnedState, this.cardId);
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this.ngUnsubscribe.next();
            this.ngUnsubscribe.complete();
        }
    }
    ImageCardContainer.decorators = [
        { type: Component, args: [{
                    selector: 'image-card',
                    template: `
    <image-card-component
      [loadState]="loadState$ | async"
      [title]="title$ | async"
      [tag]="tag$ | async"
      [runId]="runId$ | async"
      [sample]="sample$ | async"
      [imageUrl]="imageUrl$ | async"
      [stepIndex]="stepIndex$ | async"
      [stepValues]="stepValues$ | async"
      (stepIndexChange)="onStepIndexChanged($event)"
      [brightnessInMilli]="brightnessInMilli$ | async"
      [contrastInMilli]="contrastInMilli$ | async"
      [runColorScale]="runColorScale"
      [showActualSize]="showActualSize"
      [allowToggleActualSize]="(actualSizeGlobalSetting$ | async) === false"
      [isPinned]="isPinned$ | async"
      (onActualSizeToggle)="onActualSizeToggle()"
      (onPinClicked)="pinStateChanged.emit($event)"
    ></image-card-component>
  `,
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    ImageCardContainer.ctorParameters = () => [
        { type: Store },
        { type: MetricsDataSource }
    ];
    ImageCardContainer.propDecorators = {
        cardId: [{ type: Input }],
        groupName: [{ type: Input }],
        runColorScale: [{ type: Input }],
        fullWidthChanged: [{ type: Output }],
        pinStateChanged: [{ type: Output }]
    };
    /** @nocollapse */ ImageCardContainer.ɵfac = function ImageCardContainer_Factory(t) { return new (t || ImageCardContainer)(ɵɵdirectiveInject(Store), ɵɵdirectiveInject(MetricsDataSource)); };
    /** @nocollapse */ ImageCardContainer.ɵcmp = ɵɵdefineComponent({ type: ImageCardContainer, selectors: [["image-card"]], inputs: { cardId: "cardId", groupName: "groupName", runColorScale: "runColorScale" }, outputs: { fullWidthChanged: "fullWidthChanged", pinStateChanged: "pinStateChanged" }, decls: 13, vars: 38, consts: [[3, "loadState", "title", "tag", "runId", "sample", "imageUrl", "stepIndex", "stepValues", "brightnessInMilli", "contrastInMilli", "runColorScale", "showActualSize", "allowToggleActualSize", "isPinned", "stepIndexChange", "onActualSizeToggle", "onPinClicked"]], template: function ImageCardContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "image-card-component", 0);
            ɵɵlistener("stepIndexChange", function ImageCardContainer_Template_image_card_component_stepIndexChange_0_listener($event) { return ctx.onStepIndexChanged($event); })("onActualSizeToggle", function ImageCardContainer_Template_image_card_component_onActualSizeToggle_0_listener() { return ctx.onActualSizeToggle(); })("onPinClicked", function ImageCardContainer_Template_image_card_component_onPinClicked_0_listener($event) { return ctx.pinStateChanged.emit($event); });
            ɵɵpipe(1, "async");
            ɵɵpipe(2, "async");
            ɵɵpipe(3, "async");
            ɵɵpipe(4, "async");
            ɵɵpipe(5, "async");
            ɵɵpipe(6, "async");
            ɵɵpipe(7, "async");
            ɵɵpipe(8, "async");
            ɵɵpipe(9, "async");
            ɵɵpipe(10, "async");
            ɵɵpipe(11, "async");
            ɵɵpipe(12, "async");
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵproperty("loadState", ɵɵpipeBind1(1, 14, ctx.loadState$))("title", ɵɵpipeBind1(2, 16, ctx.title$))("tag", ɵɵpipeBind1(3, 18, ctx.tag$))("runId", ɵɵpipeBind1(4, 20, ctx.runId$))("sample", ɵɵpipeBind1(5, 22, ctx.sample$))("imageUrl", ɵɵpipeBind1(6, 24, ctx.imageUrl$))("stepIndex", ɵɵpipeBind1(7, 26, ctx.stepIndex$))("stepValues", ɵɵpipeBind1(8, 28, ctx.stepValues$))("brightnessInMilli", ɵɵpipeBind1(9, 30, ctx.brightnessInMilli$))("contrastInMilli", ɵɵpipeBind1(10, 32, ctx.contrastInMilli$))("runColorScale", ctx.runColorScale)("showActualSize", ctx.showActualSize)("allowToggleActualSize", ɵɵpipeBind1(11, 34, ctx.actualSizeGlobalSetting$) === false)("isPinned", ɵɵpipeBind1(12, 36, ctx.isPinned$));
        } }, directives: [ImageCardComponent], pipes: [AsyncPipe], encapsulation: 2, changeDetection: 0 });

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * @fileoverview Utilities related to formatting values.
     */
    /**
     * Formats a time, given as number of hours. e.g.
     * > formatRelativeTimeInMs(10555 * 60 * 60)
     * "10h 33m 17s"
     */
    function formatRelativeTimeInMs(timeInMs) {
        // We will always show 2 units of precision, e.g days and hours, or
        // minutes and seconds, but not hours and minutes and seconds.
        const result = [];
        const timeInHours = timeInMs / (1000 * 60 * 60);
        const days = Math.floor(timeInHours / 24);
        if (days) {
            result.push(days.toString() + 'd');
        }
        const remainingHours = timeInHours - days * 24;
        const hours = Math.floor(remainingHours);
        if (hours || days) {
            result.push(hours.toString() + 'h');
        }
        const remainingMinutes = (remainingHours - hours) * 60;
        const minutes = Math.floor(remainingMinutes);
        if (minutes || hours || days) {
            result.push(minutes.toString() + 'm');
        }
        const remainingSeconds = (remainingMinutes - minutes) * 60;
        const seconds = Math.floor(remainingSeconds);
        result.push(seconds.toString() + 's');
        return result.join(' ');
    }
    /**
     * Integers less than this number can be safely formatted as-is.
     */
    const MAX_SMALL_INTEGER = Math.pow(10, 4);
    /**
     * This numeric formatter currently covers common cases only.
     *
     * The Polymer version relies on D3, which gives more features for free:
     * selecting decimal vs exponent notation, rounding to significant digits, and
     * removing insignificant trailing zeros.
     */
    function formatNumber$1(value) {
        if (isNaN(value) || !Number.isFinite(value)) {
            return value.toString();
        }
        if (Number.isInteger(value) && Math.abs(value) < MAX_SMALL_INTEGER) {
            return value.toString();
        }
        return value.toPrecision(4);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/widgets/line_chart/line_chart_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @template SeriesMetadata, ExtraPointData
     */
    class LineChartComponent {
        /**
         * @param {?} host
         */
        constructor(host) {
            this.host = host;
            this.ignoreYOutliers = false;
            this.smoothingEnabled = false;
            this.smoothingWeight = 0.6;
            this.tooltipColumns = [];
            this.tooltipPosition = TooltipPosition.AUTO;
            this.tooltipSortingMethod = TooltipSortingMethod.DEFAULT;
            /**
             * Indirectly applied properties, without a 1:1 Polymer mapping.
             */
            this.seriesDataList = [];
            this.xAxisType = XAxisType.STEP;
            this.yAxisType = YAxisType.LINEAR;
            this.element = (/** @type {?} */ (document.createElement('vz-line-chart2')));
            this.element.yValueAccessor = (/**
             * @param {?} d
             * @return {?}
             */
            (d) => d.y);
            // Must set optional input values here since they won't be part of the
            // ngOnChanges if the parent does not override the value.
            this.element.ignoreYOutliers = this.ignoreYOutliers;
            this.element.smoothingEnabled = this.smoothingEnabled;
            this.element.smoothingWeight = this.smoothingWeight;
            this.element.tooltipColumns = this.tooltipColumns;
            this.element.tooltipPosition = this.tooltipPosition;
            this.element.tooltipSortingMethod = this.tooltipSortingMethod;
            this.element.xType = this.xAxisType;
            this.element.yScaleType = this.yAxisType;
        }
        /**
         * @return {?}
         */
        ngOnInit() {
            this.host.nativeElement.appendChild(this.element);
        }
        /**
         * @param {?} changes
         * @return {?}
         */
        ngOnChanges(changes) {
            // Record state before changing any values.
            /** @type {?} */
            const wasDomainFitToData = this.element.isDataFitToDomain();
            if (changes['colorScale'] && this.colorScale) {
                this.element.colorScale = { scale: this.colorScale };
            }
            if (changes['defaultXRange']) {
                this.element.defaultXRange = this.defaultXRange;
            }
            if (changes['defaultYRange']) {
                this.element.defaultYRange = this.defaultYRange;
            }
            if (changes['ignoreYOutliers']) {
                this.element.ignoreYOutliers = this.ignoreYOutliers;
            }
            if (changes['smoothingEnabled']) {
                this.element.smoothingEnabled = this.smoothingEnabled;
            }
            if (changes['smoothingWeight']) {
                this.element.smoothingWeight = this.smoothingWeight;
            }
            if (changes['tooltipColumns']) {
                this.element.tooltipColumns = this.tooltipColumns;
            }
            if (changes['tooltipPosition']) {
                this.element.tooltipPosition = this.tooltipPosition;
            }
            if (changes['tooltipSortingMethod']) {
                this.element.tooltipSortingMethod = this.tooltipSortingMethod;
            }
            if (changes['xAxisType']) {
                this.element.xType = this.xAxisType;
            }
            if (changes['yAxisType']) {
                this.element.yScaleType = this.yAxisType;
            }
            if (this.shouldUpdateSeriesData(changes)) {
                this.updateSeriesData();
            }
            // Preserve fit-to-domain if the data points may have changed.
            if (wasDomainFitToData &&
                (changes['seriesDataList'] ||
                    changes['smoothingEnabled'] ||
                    (changes['smoothingWeight'] && this.smoothingEnabled))) {
                this.element.resetDomain();
            }
            this.element.redraw();
        }
        /**
         * @private
         * @param {?} changes
         * @return {?}
         */
        shouldUpdateSeriesData(changes) {
            if (!!changes['seriesDataList']) {
                return true;
            }
            // The series data format expected by VzLineChart2 depends on the `xType`.
            /** @type {?} */
            const prevXAxisType = changes['xAxisType']
                ? changes['xAxisType'].previousValue
                : null;
            if (prevXAxisType &&
                this.isWallTimeBased(prevXAxisType) !==
                    this.isWallTimeBased(this.xAxisType)) {
                return true;
            }
            return false;
        }
        /**
         * @private
         * @return {?}
         */
        updateSeriesData() {
            for (const seriesData of this.seriesDataList) {
                const { seriesId, metadata, points } = seriesData;
                /** @type {?} */
                const formattedPoints = this.formatByXAxisType(points);
                this.element.setSeriesData(seriesId, formattedPoints);
                this.element.setSeriesMetadata(seriesId, metadata);
            }
            /** @type {?} */
            const visibleSeries = this.seriesDataList
                .filter((/**
             * @param {?} seriesData
             * @return {?}
             */
            (seriesData) => seriesData.visible))
                .map((/**
             * @param {?} __0
             * @return {?}
             */
            ({ seriesId }) => seriesId));
            this.element.setVisibleSeries(visibleSeries);
            this.element.commitChanges();
        }
        /**
         * @private
         * @param {?} xAxisType
         * @return {?}
         */
        isWallTimeBased(xAxisType) {
            return (xAxisType === XAxisType.WALL_TIME || xAxisType === XAxisType.RELATIVE);
        }
        /**
         * @private
         * @param {?} seriesData
         * @return {?}
         */
        formatByXAxisType(seriesData) {
            return seriesData.map((/**
             * @param {?} datum
             * @return {?}
             */
            (datum) => {
                if (this.isWallTimeBased(this.xAxisType)) {
                    return Object.assign(Object.assign({}, datum), { wall_time: new Date(datum.x) });
                }
                return Object.assign(Object.assign({}, datum), { step: datum.x });
            }));
        }
        /**
         * @return {?}
         */
        resetDomain() {
            this.element.resetDomain();
        }
        /**
         * @return {?}
         */
        redraw() {
            this.element.redraw();
        }
    }
    LineChartComponent.decorators = [
        { type: Component, args: [{
                    selector: 'tb-line-chart',
                    template: ``,
                    styles: [
                        `
      tb-line-chart > vz-line-chart2 {
        height: 100%;
      }
    `,
                    ],
                    // Angular's emulated view encapsulation prevents component styles from
                    // applying to DOM created outside of its template ('vz-line-chart2').
                    encapsulation: ViewEncapsulation$1.None,
                },] },
    ];
    /** @nocollapse */
    LineChartComponent.ctorParameters = () => [
        { type: ElementRef }
    ];
    LineChartComponent.propDecorators = {
        colorScale: [{ type: Input }],
        defaultXRange: [{ type: Input }],
        defaultYRange: [{ type: Input }],
        ignoreYOutliers: [{ type: Input }],
        smoothingEnabled: [{ type: Input }],
        smoothingWeight: [{ type: Input }],
        tooltipColumns: [{ type: Input }],
        tooltipPosition: [{ type: Input }],
        tooltipSortingMethod: [{ type: Input }],
        seriesDataList: [{ type: Input }],
        xAxisType: [{ type: Input }],
        yAxisType: [{ type: Input }]
    };
    /** @nocollapse */ LineChartComponent.ɵfac = function LineChartComponent_Factory(t) { return new (t || LineChartComponent)(ɵɵdirectiveInject(ElementRef)); };
    /** @nocollapse */ LineChartComponent.ɵcmp = ɵɵdefineComponent({ type: LineChartComponent, selectors: [["tb-line-chart"]], inputs: { colorScale: "colorScale", defaultXRange: "defaultXRange", defaultYRange: "defaultYRange", ignoreYOutliers: "ignoreYOutliers", smoothingEnabled: "smoothingEnabled", smoothingWeight: "smoothingWeight", tooltipColumns: "tooltipColumns", tooltipPosition: "tooltipPosition", tooltipSortingMethod: "tooltipSortingMethod", seriesDataList: "seriesDataList", xAxisType: "xAxisType", yAxisType: "yAxisType" }, features: [ɵɵNgOnChangesFeature], decls: 0, vars: 0, template: function LineChartComponent_Template(rf, ctx) { }, styles: ["\n      tb-line-chart > vz-line-chart2 {\n        height: 100%;\n      }\n    "], encapsulation: 2 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/widgets/resize_detector_directive.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * A directive that calls `onResize` when a host element is resized.
     *
     * It does not emit `onResize` on the initial render.
     */
    class ResizeDetectorDirective {
        /**
         * @param {?} ref
         */
        constructor(ref) {
            this.resizeEventDebouncePeriodInMs = 100;
            this.onResize = new EventEmitter();
            this.ngUnsubscribe$ = new Subject();
            this.onResize$ = new Subject();
            /** @type {?} */
            const resizeObserver = new ResizeObserver((/**
             * @return {?}
             */
            () => {
                this.onResize$.next();
            }));
            resizeObserver.observe(ref.nativeElement);
            this.ngUnsubscribe$.subscribe((/**
             * @return {?}
             */
            () => {
                resizeObserver.unobserve(ref.nativeElement);
            }));
        }
        /**
         * @return {?}
         */
        ngOnInit() {
            this.onResize$
                .pipe(
            // When ResizeObserver is initially created, it calls the callback.
            // Ignore that since it is not really a resize.
            skip(1), debounceTime(this.resizeEventDebouncePeriodInMs), takeUntil(this.ngUnsubscribe$))
                .subscribe((/**
             * @return {?}
             */
            () => {
                this.onResize.emit();
            }));
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this.ngUnsubscribe$.next();
            this.ngUnsubscribe$.complete();
        }
    }
    ResizeDetectorDirective.decorators = [
        { type: Directive, args: [{ selector: '[detectResize]' },] },
    ];
    /** @nocollapse */
    ResizeDetectorDirective.ctorParameters = () => [
        { type: ElementRef }
    ];
    ResizeDetectorDirective.propDecorators = {
        resizeEventDebouncePeriodInMs: [{ type: Input }],
        onResize: [{ type: Output }]
    };
    /** @nocollapse */ ResizeDetectorDirective.ɵfac = function ResizeDetectorDirective_Factory(t) { return new (t || ResizeDetectorDirective)(ɵɵdirectiveInject(ElementRef)); };
    /** @nocollapse */ ResizeDetectorDirective.ɵdir = ɵɵdefineDirective({ type: ResizeDetectorDirective, selectors: [["", "detectResize", ""]], inputs: { resizeEventDebouncePeriodInMs: "resizeEventDebouncePeriodInMs" }, outputs: { onResize: "onResize" } });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/views/card_renderer/scalar_card_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    var I18N_0$3;
    if (typeof ngI18nClosureMode !== "undefined" && ngI18nClosureMode) {
        /**
         * @desc A button to pin a card.
         */ 
        const MSG_EXTERNAL_7284606426234375344$$TENSORBOARD_WEBAPP_METRICS_VIEWS_CARD_RENDERER_SCALAR_CARD_COMPONENT_TS_1 = goog.getMsg("Pin card");
        I18N_0$3 = MSG_EXTERNAL_7284606426234375344$$TENSORBOARD_WEBAPP_METRICS_VIEWS_CARD_RENDERER_SCALAR_CARD_COMPONENT_TS_1;
    }
    else {
        I18N_0$3 = $localize `:A button to pin a card.␟e665dc712bd5f18d4dfa3a29e125d565cc51e2f6␟7284606426234375344:Pin card`;
    }
    const _c2$9 = ["aria-label", I18N_0$3];
    var I18N_3$1;
    if (typeof ngI18nClosureMode !== "undefined" && ngI18nClosureMode) {
        /**
         * @desc A button on line chart that toggles full size mode.
         */ 
        const MSG_EXTERNAL_915721563638926597$$TENSORBOARD_WEBAPP_METRICS_VIEWS_CARD_RENDERER_SCALAR_CARD_COMPONENT_TS_4 = goog.getMsg("Toggle full size mode");
        I18N_3$1 = MSG_EXTERNAL_915721563638926597$$TENSORBOARD_WEBAPP_METRICS_VIEWS_CARD_RENDERER_SCALAR_CARD_COMPONENT_TS_4;
    }
    else {
        I18N_3$1 = $localize `:A button on line chart that toggles full size mode.␟fc8f767d0b9f930187a1bae34477ad28736ece33␟915721563638926597:Toggle full size mode`;
    }
    const _c5$3 = ["aria-label", I18N_3$1];
    var I18N_6;
    if (typeof ngI18nClosureMode !== "undefined" && ngI18nClosureMode) {
        /**
         * @desc An overflow menu button that opens more line chart options
         */ 
        const MSG_EXTERNAL_878053740210336435$$TENSORBOARD_WEBAPP_METRICS_VIEWS_CARD_RENDERER_SCALAR_CARD_COMPONENT_TS_7 = goog.getMsg("More line chart options");
        I18N_6 = MSG_EXTERNAL_878053740210336435$$TENSORBOARD_WEBAPP_METRICS_VIEWS_CARD_RENDERER_SCALAR_CARD_COMPONENT_TS_7;
    }
    else {
        I18N_6 = $localize `:An overflow menu button that opens more line chart options␟b260fab946a3077ce20fd28e336979f586720e8d␟878053740210336435:More line chart options`;
    }
    const _c8$1 = ["aria-label", I18N_6];
    var I18N_9;
    if (typeof ngI18nClosureMode !== "undefined" && ngI18nClosureMode) {
        /**
         * @desc A button that toggles log scale on y-axis on a line chart
         */ 
        const MSG_EXTERNAL_3374645620638883926$$TENSORBOARD_WEBAPP_METRICS_VIEWS_CARD_RENDERER_SCALAR_CARD_COMPONENT_TS_10 = goog.getMsg("Toggle Y-axis log scale on line chart");
        I18N_9 = MSG_EXTERNAL_3374645620638883926$$TENSORBOARD_WEBAPP_METRICS_VIEWS_CARD_RENDERER_SCALAR_CARD_COMPONENT_TS_10;
    }
    else {
        I18N_9 = $localize `:A button that toggles log scale on y-axis on a line chart␟fe91f96ab9b3baca5a48913f2b0fae84483d93e3␟3374645620638883926:Toggle Y-axis log scale on line chart`;
    }
    const _c11 = ["aria-label", I18N_9];
    var I18N_12;
    if (typeof ngI18nClosureMode !== "undefined" && ngI18nClosureMode) {
        /**
         * @desc A button that resets line chart domain to the data
         */ 
        const MSG_EXTERNAL_3830646521058268558$$TENSORBOARD_WEBAPP_METRICS_VIEWS_CARD_RENDERER_SCALAR_CARD_COMPONENT_TS_13 = goog.getMsg("Fit line chart domains to data");
        I18N_12 = MSG_EXTERNAL_3830646521058268558$$TENSORBOARD_WEBAPP_METRICS_VIEWS_CARD_RENDERER_SCALAR_CARD_COMPONENT_TS_13;
    }
    else {
        I18N_12 = $localize `:A button that resets line chart domain to the data␟e68a552941ab427a99e7437e08443f30ac71ccd6␟3830646521058268558:Fit line chart domains to data`;
    }
    const _c14 = ["aria-label", I18N_12];
    function ScalarCardComponent_mat_spinner_25_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "mat-spinner", 16);
    } }
    /** @type {?} */
    const RESIZE_REDRAW_DEBOUNCE_TIME_IN_MS = 50;
    /** @type {?} */
    const DEFAULT_TOOLTIP_COLUMNS = [
        {
            title: 'Name',
            evaluate: (/**
             * @param {?} d
             * @return {?}
             */
            (d) => {
                return d.dataset.metadata().meta.displayName;
            }),
        },
        {
            title: 'Value',
            evaluate: (/**
             * @param {?} d
             * @return {?}
             */
            (d) => {
                return formatNumber$1(d.datum.y);
            }),
        },
        {
            title: 'Step',
            evaluate: (/**
             * @param {?} d
             * @return {?}
             */
            (d) => {
                return d.datum.step.toString();
            }),
        },
        {
            title: 'Time',
            evaluate: (/**
             * @param {?} d
             * @return {?}
             */
            (d) => {
                // sec to ms.
                /** @type {?} */
                const date = new Date(d.datum.wallTime * 1000);
                return date.toLocaleString();
            }),
        },
        {
            title: 'Relative',
            evaluate: (/**
             * @param {?} d
             * @return {?}
             */
            (d) => {
                /** @type {?} */
                const data = d.dataset.data();
                /** @type {?} */
                const firstTime = data.length > 0 ? data[0].wallTime : 0;
                /** @type {?} */
                const relativeTimeInMs = (d.datum.wallTime - firstTime) * 1000;
                return formatRelativeTimeInMs(relativeTimeInMs);
            }),
        },
    ];
    class ScalarCardComponent {
        /**
         * @param {?} ref
         */
        constructor(ref) {
            this.ref = ref;
            this.RESIZE_REDRAW_DEBOUNCE_TIME_IN_MS = RESIZE_REDRAW_DEBOUNCE_TIME_IN_MS;
            this.DataLoadState = DataLoadState;
            this.onFullSizeToggle = new EventEmitter();
            this.onPinClicked = new EventEmitter();
            this.yAxisType = YAxisType.LINEAR;
            this.tooltipColumns = DEFAULT_TOOLTIP_COLUMNS;
        }
        /**
         * @return {?}
         */
        chartXAxisType() {
            switch (this.xAxisType) {
                case XAxisType$1.STEP:
                    return XAxisType.STEP;
                case XAxisType$1.WALL_TIME:
                    return XAxisType.WALL_TIME;
                case XAxisType$1.RELATIVE:
                    return XAxisType.RELATIVE;
                default:
                    throw new Error('Invalid xAxisType for line chart.');
            }
        }
        /**
         * @return {?}
         */
        toggleYAxisType() {
            this.yAxisType =
                this.yAxisType === YAxisType.LINEAR ? YAxisType.LOG : YAxisType.LINEAR;
        }
        /**
         * @return {?}
         */
        resetDomain() {
            if (this.lineChart) {
                this.lineChart.resetDomain();
            }
        }
        /**
         * @return {?}
         */
        redraw() {
            if (this.lineChart) {
                // Only redraw when it is visible (and thus have width and height).
                if (this.ref.nativeElement.clientWidth) {
                    this.lineChart.redraw();
                }
            }
        }
    }
    ScalarCardComponent.decorators = [
        { type: Component, args: [{
                    selector: 'scalar-card-component',
                    templateUrl: 'scalar_card_component.ng.html',
                    styleUrls: ['scalar_card_component.css'],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    ScalarCardComponent.ctorParameters = () => [
        { type: ElementRef }
    ];
    ScalarCardComponent.propDecorators = {
        loadState: [{ type: Input }],
        title: [{ type: Input }],
        tag: [{ type: Input }],
        seriesDataList: [{ type: Input }],
        tooltipSort: [{ type: Input }],
        ignoreOutliers: [{ type: Input }],
        xAxisType: [{ type: Input }],
        scalarSmoothing: [{ type: Input }],
        runColorScale: [{ type: Input }],
        showFullSize: [{ type: Input }],
        isPinned: [{ type: Input }],
        onFullSizeToggle: [{ type: Output }],
        onPinClicked: [{ type: Output }],
        lineChart: [{ type: ViewChild, args: [LineChartComponent,] }]
    };
    /** @nocollapse */ ScalarCardComponent.ɵfac = function ScalarCardComponent_Factory(t) { return new (t || ScalarCardComponent)(ɵɵdirectiveInject(ElementRef)); };
    /** @nocollapse */ ScalarCardComponent.ɵcmp = ɵɵdefineComponent({ type: ScalarCardComponent, selectors: [["scalar-card-component"]], viewQuery: function ScalarCardComponent_Query(rf, ctx) { if (rf & 1) {
            ɵɵviewQuery(LineChartComponent, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.lineChart = _t.first);
        } }, inputs: { loadState: "loadState", title: "title", tag: "tag", seriesDataList: "seriesDataList", tooltipSort: "tooltipSort", ignoreOutliers: "ignoreOutliers", xAxisType: "xAxisType", scalarSmoothing: "scalarSmoothing", runColorScale: "runColorScale", showFullSize: "showFullSize", isPinned: "isPinned" }, outputs: { onFullSizeToggle: "onFullSizeToggle", onPinClicked: "onPinClicked" }, decls: 27, vars: 18, consts: [[1, "heading"], [1, "tag", 3, "title", "value"], [1, "controls"], ["mat-icon-button", "", 1, "pin-button", 3, "click", 6, "aria-label"], [3, "svgIcon"], ["mat-icon-button", "", "title", "Toggle full size mode", 3, "click", 6, "aria-label"], ["mat-icon-button", "", "title", "More line chart options", 3, "matMenuTriggerFor", 6, "aria-label"], ["svgIcon", "more_vert_24px"], ["menu", "matMenu"], ["mat-menu-item", "", 3, "click", 6, "aria-label"], ["svgIcon", "line_weight_24px"], ["mat-menu-item", "", 3, "disabled", "click", 6, "aria-label"], ["svgIcon", "settings_overscan_24px"], [1, "chart-container"], ["diameter", "18", 4, "ngIf"], ["detectResize", "", 3, "colorScale", "seriesDataList", "tooltipColumns", "tooltipSortingMethod", "ignoreYOutliers", "xAxisType", "smoothingEnabled", "smoothingWeight", "yAxisType", "resizeEventDebouncePeriodInMs", "onResize"], ["diameter", "18"]], template: function ScalarCardComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "div", 0);
            ɵɵelement(1, "tb-truncated-path", 1);
            ɵɵelementStart(2, "span", 2);
            ɵɵelementStart(3, "button", 3);
            ɵɵi18nAttributes(4, _c2$9);
            ɵɵlistener("click", function ScalarCardComponent_Template_button_click_3_listener() { return ctx.onPinClicked.emit(!ctx.isPinned); });
            ɵɵelement(5, "mat-icon", 4);
            ɵɵelementEnd();
            ɵɵelementStart(6, "button", 5);
            ɵɵi18nAttributes(7, _c5$3);
            ɵɵlistener("click", function ScalarCardComponent_Template_button_click_6_listener() { return ctx.onFullSizeToggle.emit(); });
            ɵɵelement(8, "mat-icon", 4);
            ɵɵelementEnd();
            ɵɵelementStart(9, "button", 6);
            ɵɵi18nAttributes(10, _c8$1);
            ɵɵelement(11, "mat-icon", 7);
            ɵɵelementEnd();
            ɵɵelementStart(12, "mat-menu", null, 8);
            ɵɵelementStart(14, "button", 9);
            ɵɵi18nAttributes(15, _c11);
            ɵɵlistener("click", function ScalarCardComponent_Template_button_click_14_listener() { return ctx.toggleYAxisType(); });
            ɵɵelement(16, "mat-icon", 10);
            ɵɵelementStart(17, "span");
            ɵɵtext(18, "Toggle Y-axis log scale");
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementStart(19, "button", 11);
            ɵɵi18nAttributes(20, _c14);
            ɵɵlistener("click", function ScalarCardComponent_Template_button_click_19_listener() { return ctx.resetDomain(); });
            ɵɵelement(21, "mat-icon", 12);
            ɵɵelementStart(22, "span");
            ɵɵtext(23, "Fit domain to data");
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementStart(24, "div", 13);
            ɵɵtemplate(25, ScalarCardComponent_mat_spinner_25_Template, 1, 0, "mat-spinner", 14);
            ɵɵelementStart(26, "tb-line-chart", 15);
            ɵɵlistener("onResize", function ScalarCardComponent_Template_tb_line_chart_onResize_26_listener() { return ctx.redraw(); });
            ɵɵelementEnd();
            ɵɵelementEnd();
        } if (rf & 2) {
            const _r0 = ɵɵreference(13);
            ɵɵadvance(1);
            ɵɵpropertyInterpolate("title", ctx.tag);
            ɵɵpropertyInterpolate("value", ctx.title);
            ɵɵadvance(2);
            ɵɵattribute("title", ctx.isPinned ? "Unpin card" : "Pin card");
            ɵɵadvance(2);
            ɵɵproperty("svgIcon", ctx.isPinned ? "keep_24px" : "keep_outline_24px");
            ɵɵadvance(3);
            ɵɵproperty("svgIcon", ctx.showFullSize ? "fullscreen_exit_24px" : "fullscreen_24px");
            ɵɵadvance(1);
            ɵɵproperty("matMenuTriggerFor", _r0);
            ɵɵadvance(10);
            ɵɵproperty("disabled", !ctx.seriesDataList || !ctx.seriesDataList.length);
            ɵɵadvance(6);
            ɵɵproperty("ngIf", ctx.loadState === ctx.DataLoadState.LOADING);
            ɵɵadvance(1);
            ɵɵproperty("colorScale", ctx.runColorScale)("seriesDataList", ctx.seriesDataList)("tooltipColumns", ctx.tooltipColumns)("tooltipSortingMethod", ctx.tooltipSort)("ignoreYOutliers", ctx.ignoreOutliers)("xAxisType", ctx.chartXAxisType())("smoothingEnabled", ctx.scalarSmoothing > 0)("smoothingWeight", ctx.scalarSmoothing)("yAxisType", ctx.yAxisType)("resizeEventDebouncePeriodInMs", ctx.RESIZE_REDRAW_DEBOUNCE_TIME_IN_MS);
        } }, directives: [TruncatedPathComponent, MatButton, MatIcon, MatMenuTrigger, _MatMenu, MatMenuItem, NgIf, LineChartComponent, ResizeDetectorDirective, MatSpinner], styles: [".mat-badge-content[_ngcontent-%COMP%]{font-weight:600;font-size:12px;font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:9px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:24px}.mat-h1[_ngcontent-%COMP%], .mat-headline[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%]{font:400 24px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h2[_ngcontent-%COMP%], .mat-title[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h3[_ngcontent-%COMP%], .mat-subheading-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%]{font:400 16px/28px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h4[_ngcontent-%COMP%], .mat-subheading-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%]{font:400 15px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h5[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%]{font:400 calc(14px * 0.83)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-h6[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:400 calc(14px * 0.67)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-body-strong[_ngcontent-%COMP%], .mat-body-2[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   p[_ngcontent-%COMP%]{margin:0 0 12px}.mat-small[_ngcontent-%COMP%], .mat-caption[_ngcontent-%COMP%]{font:400 12px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-display-4[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-4[_ngcontent-%COMP%]{font:300 112px/112px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.05em;margin:0 0 56px}.mat-display-3[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-3[_ngcontent-%COMP%]{font:400 56px/56px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.02em;margin:0 0 64px}.mat-display-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-2[_ngcontent-%COMP%]{font:400 45px/48px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.005em;margin:0 0 64px}.mat-display-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-1[_ngcontent-%COMP%]{font:400 34px/40px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 64px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%], .mat-flat-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-button-toggle[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card-title[_ngcontent-%COMP%]{font-size:24px;font-weight:500}.mat-card-header[_ngcontent-%COMP%]   .mat-card-title[_ngcontent-%COMP%]{font-size:20px}.mat-card-subtitle[_ngcontent-%COMP%], .mat-card-content[_ngcontent-%COMP%]{font-size:14px}.mat-checkbox[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-checkbox-layout[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{line-height:24px}.mat-chip[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-chip[_ngcontent-%COMP%]   .mat-chip-trailing-icon.mat-icon[_ngcontent-%COMP%], .mat-chip[_ngcontent-%COMP%]   .mat-chip-remove.mat-icon[_ngcontent-%COMP%]{font-size:18px}.mat-table[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-header-cell[_ngcontent-%COMP%]{font-size:12px;font-weight:500}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{font-size:14px}.mat-calendar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-calendar-body[_ngcontent-%COMP%]{font-size:13px}.mat-calendar-body-label[_ngcontent-%COMP%], .mat-calendar-period-button[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-calendar-table-header[_ngcontent-%COMP%]   th[_ngcontent-%COMP%]{font-size:11px;font-weight:400}.mat-dialog-title[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-expansion-panel-header[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:15px;font-weight:400}.mat-expansion-panel-content[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field[_ngcontent-%COMP%]{font-size:inherit;font-weight:400;line-height:1.125;font-family:Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.34375em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{font-size:150%;line-height:1.125}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]{height:1.5em;width:1.5em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{height:1.125em;line-height:1.125}.mat-form-field-infix[_ngcontent-%COMP%]{padding:.5em 0;border-top:.84375em solid transparent}.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34375em) scale(0.75);width:133.3333333333%}.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34374em) scale(0.75);width:133.3333433333%}.mat-form-field-label-wrapper[_ngcontent-%COMP%]{top:-0.84375em;padding-top:.84375em}.mat-form-field-label[_ngcontent-%COMP%]{top:1.34375em}.mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.34375em}.mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{font-size:75%;margin-top:.6666666667em;top:calc(100% - 1.7916666667em)}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.4375em 0}.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.001px);-ms-transform:translateY(-1.28125em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00101px);-ms-transform:translateY(-1.28124em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00102px);-ms-transform:translateY(-1.28123em) scale(0.75);width:133.3333533333%}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.28125em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{margin-top:.5416666667em;top:calc(100% - 1.6666666667em)}@media print{.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28122em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28121em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.2812em) scale(0.75)}}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.25em 0 .75em 0}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.09375em;margin-top:-0.5em}.mat-form-field-appearance-fill.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59374em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:1em 0 1em 0}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.84375em;margin-top:-0.25em}.mat-form-field-appearance-outline.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59374em) scale(0.75);width:133.3333433333%}.mat-grid-tile-header[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]{font-size:14px}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2), .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}input.mat-input-element[_ngcontent-%COMP%]{margin-top:-0.0625em}.mat-menu-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:400}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px}.mat-radio-button[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select-trigger[_ngcontent-%COMP%]{height:1.125em}.mat-slide-toggle-content[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-slider-thumb-label-text[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-stepper-vertical[_ngcontent-%COMP%], .mat-stepper-horizontal[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-step-label[_ngcontent-%COMP%]{font-size:14px;font-weight:400}.mat-step-sub-label-error[_ngcontent-%COMP%]{font-weight:normal}.mat-step-label-error[_ngcontent-%COMP%]{font-size:14px}.mat-step-label-selected[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-tab-group[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-toolbar[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0}.mat-tooltip[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:10px;padding-top:6px;padding-bottom:6px}.mat-tooltip-handset[_ngcontent-%COMP%]{font-size:14px;padding-top:8px;padding-bottom:8px}.mat-list-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:16px}.mat-optgroup-label[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-simple-snackbar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px}.mat-simple-snackbar-action[_ngcontent-%COMP%]{line-height:1;font-family:inherit;font-size:inherit;font-weight:500}.mat-tree[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{font-weight:400;font-size:14px}.mat-ripple[_ngcontent-%COMP%]{overflow:hidden;position:relative}.mat-ripple[_ngcontent-%COMP%]:not(:empty){transform:translateZ(0)}.mat-ripple.mat-ripple-unbounded[_ngcontent-%COMP%]{overflow:visible}.mat-ripple-element[_ngcontent-%COMP%]{position:absolute;border-radius:50%;pointer-events:none;transition:opacity,transform 0ms cubic-bezier(0, 0, 0.2, 1);transform:scale(0)}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{display:none}.cdk-visually-hidden[_ngcontent-%COMP%]{border:0;clip:rect(0 0 0 0);height:1px;margin:-1px;overflow:hidden;padding:0;position:absolute;width:1px;outline:0;-webkit-appearance:none;-moz-appearance:none}.cdk-overlay-container[_ngcontent-%COMP%], .cdk-global-overlay-wrapper[_ngcontent-%COMP%]{pointer-events:none;top:0;left:0;height:100%;width:100%}.cdk-overlay-container[_ngcontent-%COMP%]{position:fixed;z-index:1000}.cdk-overlay-container[_ngcontent-%COMP%]:empty{display:none}.cdk-global-overlay-wrapper[_ngcontent-%COMP%]{display:flex;position:absolute;z-index:1000}.cdk-overlay-pane[_ngcontent-%COMP%]{position:absolute;pointer-events:auto;box-sizing:border-box;z-index:1000;display:flex;max-width:100%;max-height:100%}.cdk-overlay-backdrop[_ngcontent-%COMP%]{position:absolute;top:0;bottom:0;left:0;right:0;z-index:1000;pointer-events:auto;-webkit-tap-highlight-color:transparent;transition:opacity 400ms cubic-bezier(0.25, 0.8, 0.25, 1);opacity:0}.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:1}@media screen and (-ms-high-contrast: active){.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:.6}}.cdk-overlay-dark-backdrop[_ngcontent-%COMP%]{background:rgba(0,0,0,.32)}.cdk-overlay-transparent-backdrop[_ngcontent-%COMP%], .cdk-overlay-transparent-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:0}.cdk-overlay-connected-position-bounding-box[_ngcontent-%COMP%]{position:absolute;z-index:1000;display:flex;flex-direction:column;min-width:1px;min-height:1px}.cdk-global-scrollblock[_ngcontent-%COMP%]{position:fixed;width:100%;overflow-y:scroll}@keyframes cdk-text-field-autofill-start{}@keyframes cdk-text-field-autofill-end{}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:-webkit-autofill{animation:cdk-text-field-autofill-start 0s 1ms}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:not(:-webkit-autofill){animation:cdk-text-field-autofill-end 0s 1ms}textarea.cdk-textarea-autosize[_ngcontent-%COMP%]{resize:none}textarea.cdk-textarea-autosize-measuring[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:auto !important;overflow:hidden !important}textarea.cdk-textarea-autosize-measuring-firefox[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:0 !important}.mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.1)}.mat-option[_ngcontent-%COMP%]{color:#212121}.mat-option[_ngcontent-%COMP%]:hover:not(.mat-option-disabled), .mat-option[_ngcontent-%COMP%]:focus:not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple):not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-active[_ngcontent-%COMP%]{background:rgba(0,0,0,.04);color:#212121}.mat-option.mat-option-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-primary[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f57c00}.mat-accent[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f44336}.mat-optgroup-label[_ngcontent-%COMP%]{color:#616161}.mat-optgroup-disabled[_ngcontent-%COMP%]   .mat-optgroup-label[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-pseudo-checkbox[_ngcontent-%COMP%]{color:#616161}.mat-pseudo-checkbox[_ngcontent-%COMP%]::after{color:#fafafa}.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{color:#b0b0b0}.mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f57c00}.mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f44336}.mat-pseudo-checkbox-checked.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{background:#b0b0b0}.mat-elevation-z0[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z1[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z2[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z3[_ngcontent-%COMP%]{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z4[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z5[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 5px 8px 0px rgba(0, 0, 0, 0.14),0px 1px 14px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z6[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z7[_ngcontent-%COMP%]{box-shadow:0px 4px 5px -2px rgba(0, 0, 0, 0.2),0px 7px 10px 1px rgba(0, 0, 0, 0.14),0px 2px 16px 1px rgba(0, 0, 0, 0.12)}.mat-elevation-z8[_ngcontent-%COMP%]{box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z9[_ngcontent-%COMP%]{box-shadow:0px 5px 6px -3px rgba(0, 0, 0, 0.2),0px 9px 12px 1px rgba(0, 0, 0, 0.14),0px 3px 16px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z10[_ngcontent-%COMP%]{box-shadow:0px 6px 6px -3px rgba(0, 0, 0, 0.2),0px 10px 14px 1px rgba(0, 0, 0, 0.14),0px 4px 18px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z11[_ngcontent-%COMP%]{box-shadow:0px 6px 7px -4px rgba(0, 0, 0, 0.2),0px 11px 15px 1px rgba(0, 0, 0, 0.14),0px 4px 20px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z12[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z13[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 13px 19px 2px rgba(0, 0, 0, 0.14),0px 5px 24px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z14[_ngcontent-%COMP%]{box-shadow:0px 7px 9px -4px rgba(0, 0, 0, 0.2),0px 14px 21px 2px rgba(0, 0, 0, 0.14),0px 5px 26px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z15[_ngcontent-%COMP%]{box-shadow:0px 8px 9px -5px rgba(0, 0, 0, 0.2),0px 15px 22px 2px rgba(0, 0, 0, 0.14),0px 6px 28px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z16[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z17[_ngcontent-%COMP%]{box-shadow:0px 8px 11px -5px rgba(0, 0, 0, 0.2),0px 17px 26px 2px rgba(0, 0, 0, 0.14),0px 6px 32px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z18[_ngcontent-%COMP%]{box-shadow:0px 9px 11px -5px rgba(0, 0, 0, 0.2),0px 18px 28px 2px rgba(0, 0, 0, 0.14),0px 7px 34px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z19[_ngcontent-%COMP%]{box-shadow:0px 9px 12px -6px rgba(0, 0, 0, 0.2),0px 19px 29px 2px rgba(0, 0, 0, 0.14),0px 7px 36px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z20[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 20px 31px 3px rgba(0, 0, 0, 0.14),0px 8px 38px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z21[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 21px 33px 3px rgba(0, 0, 0, 0.14),0px 8px 40px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z22[_ngcontent-%COMP%]{box-shadow:0px 10px 14px -6px rgba(0, 0, 0, 0.2),0px 22px 35px 3px rgba(0, 0, 0, 0.14),0px 8px 42px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z23[_ngcontent-%COMP%]{box-shadow:0px 11px 14px -7px rgba(0, 0, 0, 0.2),0px 23px 36px 3px rgba(0, 0, 0, 0.14),0px 9px 44px 8px rgba(0, 0, 0, 0.12)}.mat-elevation-z24[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12)}.mat-app-background[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-theme-loaded-marker[_ngcontent-%COMP%]{display:none}.mat-autocomplete-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-autocomplete-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover){background:#fff}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover):not(.mat-option-disabled){color:#212121}.mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f57c00}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{outline:solid 1px;border-radius:0}.mat-badge-accent[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-badge-warn[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f44336}.mat-badge[_ngcontent-%COMP%]{position:relative}.mat-badge-hidden[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{display:none}.mat-badge-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#b9b9b9;color:#757575}.mat-badge-content[_ngcontent-%COMP%]{position:absolute;text-align:center;display:inline-block;border-radius:50%;transition:transform 200ms ease-in-out;transform:scale(0.6);overflow:hidden;white-space:nowrap;text-overflow:ellipsis;pointer-events:none}.ng-animate-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%], .mat-badge-content._mat-animation-noopable[_ngcontent-%COMP%]{transition:none}.mat-badge-content.mat-badge-active[_ngcontent-%COMP%]{transform:none}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:16px;height:16px;line-height:16px}.mat-badge-small.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-8px}.mat-badge-small.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-8px}.mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-16px}.mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-16px}.mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-8px}.mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-8px}.mat-badge-medium[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:22px;height:22px;line-height:22px}.mat-badge-medium.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-11px}.mat-badge-medium.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-11px}.mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-22px}.mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-22px}.mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-11px}.mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-11px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:28px;height:28px;line-height:28px}.mat-badge-large.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-14px}.mat-badge-large.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-14px}.mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-28px}.mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-28px}.mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-14px}.mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-14px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]{color:inherit;background:transparent}.mat-button.mat-primary[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-button[disabled][disabled][_ngcontent-%COMP%], .mat-icon-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-icon-button[disabled][disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-stroked-button[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f44336}.mat-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:transparent}.mat-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{opacity:.1;background-color:currentColor}.mat-button-focus-overlay[_ngcontent-%COMP%]{background:#000}.mat-stroked-button[_ngcontent-%COMP%]:not([disabled]){border-color:rgba(0,0,0,.12)}.mat-flat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{color:#212121;background-color:#fff}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-flat-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-stroked-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-flat-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-raised-button[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone[_ngcontent-%COMP%], .mat-button-toggle-group[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{box-shadow:none}.mat-button-toggle[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-button-toggle[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121;background:#fff}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:#000}.mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12)}[dir=rtl][_ngcontent-%COMP%]   .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-group-appearance-standard.mat-button-toggle-vertical[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:none;border-top:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#616161}.mat-button-toggle-checked.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121}.mat-button-toggle-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.26);background-color:#eee}.mat-button-toggle-disabled.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{background:#fff}.mat-button-toggle-disabled.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#bdbdbd}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{border:solid 1px rgba(0,0,0,.12)}.mat-card[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-card[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-card.mat-card-flat[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-card-subtitle[_ngcontent-%COMP%]{color:#616161}.mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#616161}.mat-checkbox-checkmark[_ngcontent-%COMP%]{fill:#fafafa}.mat-checkbox-checkmark-path[_ngcontent-%COMP%]{stroke:#fafafa !important}.mat-checkbox-mixedmark[_ngcontent-%COMP%]{background-color:#fafafa}.mat-checkbox-indeterminate.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f57c00}.mat-checkbox-indeterminate.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#ff9800}.mat-checkbox-indeterminate.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f44336}.mat-checkbox-disabled.mat-checkbox-checked[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-disabled.mat-checkbox-indeterminate[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]:not(.mat-checkbox-checked)   .mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{color:#616161}.mat-checkbox[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%]{background:#f57c00}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%]{background:#ff9800}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%]{background:#f44336}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#212121}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#212121;opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled):active{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled)   .mat-chip-remove[_ngcontent-%COMP%]:hover{opacity:.54}.mat-chip.mat-standard-chip.mat-chip-disabled[_ngcontent-%COMP%]{opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]::after{background:#000}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-table[_ngcontent-%COMP%]{background:#fff}.mat-table[_ngcontent-%COMP%]   thead[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tbody[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tfoot[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], [mat-header-row][_ngcontent-%COMP%], [mat-row][_ngcontent-%COMP%], [mat-footer-row][_ngcontent-%COMP%], .mat-table-sticky[_ngcontent-%COMP%]{background:inherit}mat-row[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], th.mat-header-cell[_ngcontent-%COMP%], td.mat-cell[_ngcontent-%COMP%], td.mat-footer-cell[_ngcontent-%COMP%]{border-bottom-color:rgba(0,0,0,.12)}.mat-header-cell[_ngcontent-%COMP%]{color:#616161}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{color:#212121}.mat-calendar-arrow[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.54)}.mat-datepicker-toggle[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-next-button[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-previous-button[_ngcontent-%COMP%]{color:rgba(0,0,0,.54)}.mat-calendar-table-header[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-calendar-table-header-divider[_ngcontent-%COMP%]::after{background:rgba(0,0,0,.12)}.mat-calendar-body-label[_ngcontent-%COMP%]{color:#616161}.mat-calendar-body-cell-content[_ngcontent-%COMP%]{color:#212121;border-color:transparent}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){color:#757575}.mat-calendar-body-cell[_ngcontent-%COMP%]:not(.mat-calendar-body-disabled):hover > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-keyboard-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-program-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){background-color:rgba(0,0,0,.04)}.mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.38)}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.18)}.mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.4)}.mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12);background-color:#fff;color:#212121}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.4)}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.4)}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content-touch[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-datepicker-toggle-active[_ngcontent-%COMP%]{color:#f57c00}.mat-datepicker-toggle-active.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-datepicker-toggle-active.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-dialog-container[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-divider[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-divider-vertical[_ngcontent-%COMP%]{border-right-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-expansion-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-action-row[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-keyboard-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-program-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded)   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover:not([aria-disabled=true]){background:rgba(0,0,0,.04)}@media(hover: none){.mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded):not([aria-disabled=true])   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover{background:#fff}}.mat-expansion-panel-header-title[_ngcontent-%COMP%]{color:#212121}.mat-expansion-panel-header-description[_ngcontent-%COMP%], .mat-expansion-indicator[_ngcontent-%COMP%]::after{color:#616161}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-title[_ngcontent-%COMP%], .mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-description[_ngcontent-%COMP%]{color:inherit}.mat-form-field-label[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-hint[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-focused[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-ripple[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid)   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f57c00}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-accent   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#ff9800}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-warn   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#f44336}.mat-error[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-hint[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-legacy.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-standard[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-standard.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.04)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.02)}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:transparent}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.12)}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:rgba(0,0,0,.87)}.mat-form-field-appearance-outline.mat-focused[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field-appearance-outline.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-appearance-outline.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-invalid.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.06)}.mat-icon.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-icon.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-icon.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-type-mat-native-select[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#616161}.mat-input-element[_ngcontent-%COMP%]:disabled, .mat-form-field-type-mat-native-select.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#757575}.mat-input-element[_ngcontent-%COMP%]{caret-color:#f57c00}.mat-input-element[_ngcontent-%COMP%]::placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-moz-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-webkit-input-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]:-ms-input-placeholder{color:rgba(97,97,97,.42)}.mat-accent[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%], .mat-form-field-invalid[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#f44336}.mat-form-field-type-mat-native-select.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{color:#616161}.mat-list-item-disabled[_ngcontent-%COMP%]{background-color:#eee}.mat-list-option[_ngcontent-%COMP%]:hover, .mat-list-option[_ngcontent-%COMP%]:focus, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.04)}.mat-list-single-selected-option[_ngcontent-%COMP%], .mat-list-single-selected-option[_ngcontent-%COMP%]:hover, .mat-list-single-selected-option[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.12)}.mat-menu-panel[_ngcontent-%COMP%]{background:#fff}.mat-menu-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-menu-item[_ngcontent-%COMP%]{background:transparent;color:#212121}.mat-menu-item[disabled][_ngcontent-%COMP%], .mat-menu-item[disabled][_ngcontent-%COMP%]::after{color:rgba(0,0,0,.38)}.mat-menu-item[_ngcontent-%COMP%]   .mat-icon-no-color[_ngcontent-%COMP%], .mat-menu-item-submenu-trigger[_ngcontent-%COMP%]::after{color:rgba(0,0,0,.54)}.mat-menu-item[_ngcontent-%COMP%]:hover:not([disabled]), .mat-menu-item.cdk-program-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item.cdk-keyboard-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item-highlighted[_ngcontent-%COMP%]:not([disabled]){background:rgba(0,0,0,.04)}.mat-paginator[_ngcontent-%COMP%]{background:#fff}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{color:#616161}.mat-paginator-decrement[_ngcontent-%COMP%], .mat-paginator-increment[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54);border-right:2px solid rgba(0,0,0,.54)}.mat-paginator-first[_ngcontent-%COMP%], .mat-paginator-last[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54)}.mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-decrement[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-increment[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-first[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-last[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f57c00}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#ff9800}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f44336}.mat-progress-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f57c00}.mat-progress-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#ff9800}.mat-progress-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f44336}.mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#616161}.mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f57c00}.mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#ff9800}.mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#ff9800}.mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f44336}.mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f44336}.mat-radio-button.mat-radio-disabled.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-label-content[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-radio-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-select-value[_ngcontent-%COMP%]{color:#212121}.mat-select-placeholder[_ngcontent-%COMP%]{color:rgba(97,97,97,.42)}.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%]{color:#757575}.mat-select-arrow[_ngcontent-%COMP%]{color:#616161}.mat-select-panel[_ngcontent-%COMP%]{background:#fff}.mat-select-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-select-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple){background:rgba(0,0,0,.12)}.mat-form-field.mat-focused.mat-primary[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-invalid[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#757575}.mat-drawer-container[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-drawer[_ngcontent-%COMP%]{background-color:#fff;color:#212121}.mat-drawer.mat-drawer-push[_ngcontent-%COMP%]{background-color:#fff}.mat-drawer[_ngcontent-%COMP%]:not(.mat-drawer-side){box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-drawer-side[_ngcontent-%COMP%]{border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-backdrop.mat-drawer-shown[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.6)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.54)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.54)}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.54)}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle[_ngcontent-%COMP%]:not(.mat-checked)   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-slide-toggle-thumb[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12);background-color:#fafafa}.mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-primary[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f57c00}.mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-accent[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#ff9800}.mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-warn[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f44336}.mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.2)}.mat-slider[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%], .cdk-focused[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-disabled[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]   .mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing)   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26);background-color:transparent}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26)}.mat-slider-has-ticks[_ngcontent-%COMP%]   .mat-slider-wrapper[_ngcontent-%COMP%]::after{border-color:rgba(0,0,0,.7)}.mat-slider-horizontal[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to right, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent);background-image:-moz-repeating-linear-gradient(0.0001deg, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-slider-vertical[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to bottom, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-step-header.cdk-keyboard-focused[_ngcontent-%COMP%], .mat-step-header.cdk-program-focused[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]:hover{background-color:rgba(0,0,0,.04)}@media(hover: none){.mat-step-header[_ngcontent-%COMP%]:hover{background:none}}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-optional[_ngcontent-%COMP%]{color:#616161}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon[_ngcontent-%COMP%]{background-color:#616161;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-selected[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-done[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-edit[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-error[_ngcontent-%COMP%]{background-color:transparent;color:#f44336}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-active[_ngcontent-%COMP%]{color:#212121}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-error[_ngcontent-%COMP%]{color:#f44336}.mat-stepper-horizontal[_ngcontent-%COMP%], .mat-stepper-vertical[_ngcontent-%COMP%]{background-color:#fff}.mat-stepper-vertical-line[_ngcontent-%COMP%]::before{border-left-color:rgba(0,0,0,.12)}.mat-horizontal-stepper-header[_ngcontent-%COMP%]::before, .mat-horizontal-stepper-header[_ngcontent-%COMP%]::after, .mat-stepper-horizontal-line[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-sort-header-arrow[_ngcontent-%COMP%]{color:#616161}.mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-header[_ngcontent-%COMP%]{border-bottom:1px solid rgba(0,0,0,.12)}.mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%]{border-top:1px solid rgba(0,0,0,.12);border-bottom:none}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{color:#212121}.mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:#757575}.mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#212121}.mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#757575}.mat-tab-group[class*=mat-background-][_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar[class*=mat-background-][_ngcontent-%COMP%]{border-bottom:none;border-top:none}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-toolbar[_ngcontent-%COMP%]{background:#f5f5f5;color:#212121}.mat-toolbar.mat-primary[_ngcontent-%COMP%]{background:#f57c00;color:#fff}.mat-toolbar.mat-accent[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-toolbar.mat-warn[_ngcontent-%COMP%]{background:#f44336;color:#fff}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:currentColor}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:inherit}.mat-toolbar[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:currentColor}.mat-tooltip[_ngcontent-%COMP%]{background:rgba(97,97,97,.9)}.mat-tree[_ngcontent-%COMP%]{background:#fff}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{color:#212121}.mat-snack-bar-container[_ngcontent-%COMP%]{color:rgba(255,255,255,.7);background:#323232;box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-simple-snackbar-action[_ngcontent-%COMP%]{color:#ff9800}[_nghost-%COMP%]{display:flex;flex-direction:column;box-sizing:border-box;height:100%;overflow:auto;padding:16px;padding-top:4px}.heading[_ngcontent-%COMP%]{align-items:center;display:flex;justify-content:space-between;font-size:14px;margin-bottom:4px;position:relative}.tag[_ngcontent-%COMP%]{overflow:hidden}.pin-button[_ngcontent-%COMP%]   mat-icon[_ngcontent-%COMP%]{height:18px}.controls[_ngcontent-%COMP%]{color:#616161;flex-shrink:0;margin-right:-12px;white-space:nowrap}.chart-container[_ngcontent-%COMP%]{position:relative;flex:1}.chart-container[_ngcontent-%COMP%]   mat-spinner[_ngcontent-%COMP%]{position:absolute;right:11px;top:11px}.chart-container[_ngcontent-%COMP%]   tb-line-chart[_ngcontent-%COMP%]{display:block;height:100%}"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/views/card_renderer/scalar_card_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @param {?} listA
     * @param {?} listB
     * @return {?}
     */
    function areSeriesDataListEqual(listA, listB) {
        if (listA.length !== listB.length) {
            return false;
        }
        return listA.every((/**
         * @param {?} listAVal
         * @param {?} index
         * @return {?}
         */
        (listAVal, index) => {
            /** @type {?} */
            const listBVal = listB[index];
            /** @type {?} */
            const listAPoints = listAVal.points;
            /** @type {?} */
            const listBPoints = listBVal.points;
            return (listAVal.seriesId === listBVal.seriesId &&
                listAVal.metadata.displayName === listBVal.metadata.displayName &&
                listAVal.visible === listBVal.visible &&
                listAPoints.length === listBPoints.length &&
                listAPoints.every((/**
                 * @param {?} listAPoint
                 * @param {?} index
                 * @return {?}
                 */
                (listAPoint, index) => {
                    /** @type {?} */
                    const listBPoint = listBPoints[index];
                    return listAPoint.x === listBPoint.x && listAPoint.y === listBPoint.y;
                })));
        }));
    }
    class ScalarCardContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.fullWidthChanged = new EventEmitter();
            this.fullHeightChanged = new EventEmitter();
            this.pinStateChanged = new EventEmitter();
            this.seriesDataList$ = of([]);
            this.tooltipSort$ = this.store.select(getMetricsTooltipSort);
            this.ignoreOutliers$ = this.store.select(getMetricsIgnoreOutliers);
            this.xAxisType$ = this.store.select(getMetricsXAxisType);
            this.scalarSmoothing$ = this.store.select(getMetricsScalarSmoothing);
            this.showFullSize = false;
        }
        /**
         * @private
         * @param {?} cardMetadata
         * @return {?}
         */
        isScalarCardMetadata(cardMetadata) {
            const { plugin } = cardMetadata;
            return plugin === PluginType.SCALARS;
        }
        /**
         * @return {?}
         */
        onFullSizeToggle() {
            this.showFullSize = !this.showFullSize;
            this.fullWidthChanged.emit(this.showFullSize);
            this.fullHeightChanged.emit(this.showFullSize);
        }
        /**
         * Build observables once cardId is defined (after onInit).
         * @return {?}
         */
        ngOnInit() {
            /** @type {?} */
            const selectCardMetadata$ = this.store.select(getCardMetadata, this.cardId);
            /** @type {?} */
            const cardMetadata$ = selectCardMetadata$.pipe(filter((/**
             * @param {?} cardMetadata
             * @return {?}
             */
            (cardMetadata) => {
                return !!cardMetadata && this.isScalarCardMetadata(cardMetadata);
            })), map((/**
             * @param {?} cardMetadata
             * @return {?}
             */
            (cardMetadata) => {
                return (/** @type {?} */ (cardMetadata));
            })));
            /** @type {?} */
            const settingsAndTimeSeries$ = combineLatest([
                this.store.select(getMetricsXAxisType),
                this.store.select(getCardTimeSeries, this.cardId),
            ]);
            /** @type {?} */
            const runIdAndPoints$ = settingsAndTimeSeries$.pipe(filter((/**
             * @param {?} __0
             * @return {?}
             */
            ([xAxisType, runToSeries]) => !!runToSeries)), map((/**
             * @param {?} __0
             * @return {?}
             */
            ([xAxisType, runToSeries]) => ((/** @type {?} */ ({ xAxisType, runToSeries }))))), map((/**
             * @param {?} __0
             * @return {?}
             */
            ({ xAxisType, runToSeries }) => {
                /** @type {?} */
                const runIds = Object.keys(runToSeries);
                /** @type {?} */
                const results = runIds.map((/**
                 * @param {?} runId
                 * @return {?}
                 */
                (runId) => {
                    return {
                        runId,
                        points: this.stepSeriesToLineSeries(runToSeries[runId], xAxisType),
                    };
                }));
                return results;
            })));
            this.seriesDataList$ = runIdAndPoints$.pipe(switchMap((/**
             * @param {?} runIdAndPoints
             * @return {?}
             */
            (runIdAndPoints) => {
                if (!runIdAndPoints.length) {
                    return of([]);
                }
                return combineLatest(runIdAndPoints.map((/**
                 * @param {?} runIdAndPoint
                 * @return {?}
                 */
                (runIdAndPoint) => {
                    return this.getRunDisplayNameAndPoints(runIdAndPoint);
                })));
            })), combineLatestWith(this.store.select(getCurrentRouteRunSelection)), 
            // When the `fetchRunsSucceeded` action fires, the run selection
            // map and the metadata change. To prevent quick fire of changes,
            // debounce by a microtask to emit only single change for the runs
            // store change.
            debounceTime(0), map((/**
             * @param {?} __0
             * @return {?}
             */
            ([result, runSelectionMap]) => {
                return result.map((/**
                 * @param {?} __0
                 * @return {?}
                 */
                ({ runId, displayName, points }) => {
                    return {
                        seriesId: runId,
                        metadata: { displayName },
                        points,
                        visible: Boolean(runSelectionMap && runSelectionMap.get(runId)),
                    };
                }));
            })), startWith([]), distinctUntilChanged(areSeriesDataListEqual));
            this.loadState$ = this.store.select(getCardLoadState, this.cardId);
            this.tag$ = cardMetadata$.pipe(map((/**
             * @param {?} cardMetadata
             * @return {?}
             */
            (cardMetadata) => {
                return cardMetadata.tag;
            })));
            this.title$ = this.tag$.pipe(map((/**
             * @param {?} tag
             * @return {?}
             */
            (tag) => {
                return getTagDisplayName(tag, this.groupName);
            })));
            this.isPinned$ = this.store.select(getCardPinnedState, this.cardId);
        }
        /**
         * @private
         * @param {?} runIdAndPoint
         * @return {?}
         */
        getRunDisplayNameAndPoints(runIdAndPoint) {
            const { runId, points } = runIdAndPoint;
            return combineLatest([
                this.store.select(getExperimentIdForRunId, { runId }),
                this.store.select(getExperimentIdToAliasMap),
                this.store.select(getRun, { runId }),
            ]).pipe(map((/**
             * @param {?} __0
             * @return {?}
             */
            ([experimentId, idToAlias, run]) => {
                /** @type {?} */
                const displayName = getDisplayNameForRun(runId, run, experimentId ? idToAlias[experimentId] : null);
                return { runId, displayName, points };
            })));
        }
        /**
         * @private
         * @param {?} stepSeries
         * @param {?} xAxisType
         * @return {?}
         */
        stepSeriesToLineSeries(stepSeries, xAxisType) {
            /** @type {?} */
            const isStepBased = xAxisType === XAxisType$1.STEP;
            return stepSeries.map((/**
             * @param {?} stepDatum
             * @return {?}
             */
            (stepDatum) => {
                return Object.assign(Object.assign({}, stepDatum), { x: isStepBased ? stepDatum.step : stepDatum.wallTime, y: stepDatum.value });
            }));
        }
    }
    ScalarCardContainer.decorators = [
        { type: Component, args: [{
                    selector: 'scalar-card',
                    template: `
    <scalar-card-component
      [loadState]="loadState$ | async"
      [runColorScale]="runColorScale"
      [title]="title$ | async"
      [tag]="tag$ | async"
      [seriesDataList]="seriesDataList$ | async"
      [tooltipSort]="tooltipSort$ | async"
      [ignoreOutliers]="ignoreOutliers$ | async"
      [xAxisType]="xAxisType$ | async"
      [scalarSmoothing]="scalarSmoothing$ | async"
      [showFullSize]="showFullSize"
      [isPinned]="isPinned$ | async"
      (onFullSizeToggle)="onFullSizeToggle()"
      (onPinClicked)="pinStateChanged.emit($event)"
    ></scalar-card-component>
  `,
                    styles: [
                        `
      :host {
        display: block;
        height: 100%;
      }
    `,
                    ],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    ScalarCardContainer.ctorParameters = () => [
        { type: Store }
    ];
    ScalarCardContainer.propDecorators = {
        cardId: [{ type: Input }],
        groupName: [{ type: Input }],
        runColorScale: [{ type: Input }],
        fullWidthChanged: [{ type: Output }],
        fullHeightChanged: [{ type: Output }],
        pinStateChanged: [{ type: Output }]
    };
    /** @nocollapse */ ScalarCardContainer.ɵfac = function ScalarCardContainer_Factory(t) { return new (t || ScalarCardContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ ScalarCardContainer.ɵcmp = ɵɵdefineComponent({ type: ScalarCardContainer, selectors: [["scalar-card"]], inputs: { cardId: "cardId", groupName: "groupName", runColorScale: "runColorScale" }, outputs: { fullWidthChanged: "fullWidthChanged", fullHeightChanged: "fullHeightChanged", pinStateChanged: "pinStateChanged" }, decls: 10, vars: 29, consts: [[3, "loadState", "runColorScale", "title", "tag", "seriesDataList", "tooltipSort", "ignoreOutliers", "xAxisType", "scalarSmoothing", "showFullSize", "isPinned", "onFullSizeToggle", "onPinClicked"]], template: function ScalarCardContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "scalar-card-component", 0);
            ɵɵlistener("onFullSizeToggle", function ScalarCardContainer_Template_scalar_card_component_onFullSizeToggle_0_listener() { return ctx.onFullSizeToggle(); })("onPinClicked", function ScalarCardContainer_Template_scalar_card_component_onPinClicked_0_listener($event) { return ctx.pinStateChanged.emit($event); });
            ɵɵpipe(1, "async");
            ɵɵpipe(2, "async");
            ɵɵpipe(3, "async");
            ɵɵpipe(4, "async");
            ɵɵpipe(5, "async");
            ɵɵpipe(6, "async");
            ɵɵpipe(7, "async");
            ɵɵpipe(8, "async");
            ɵɵpipe(9, "async");
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵproperty("loadState", ɵɵpipeBind1(1, 11, ctx.loadState$))("runColorScale", ctx.runColorScale)("title", ɵɵpipeBind1(2, 13, ctx.title$))("tag", ɵɵpipeBind1(3, 15, ctx.tag$))("seriesDataList", ɵɵpipeBind1(4, 17, ctx.seriesDataList$))("tooltipSort", ɵɵpipeBind1(5, 19, ctx.tooltipSort$))("ignoreOutliers", ɵɵpipeBind1(6, 21, ctx.ignoreOutliers$))("xAxisType", ɵɵpipeBind1(7, 23, ctx.xAxisType$))("scalarSmoothing", ɵɵpipeBind1(8, 25, ctx.scalarSmoothing$))("showFullSize", ctx.showFullSize)("isPinned", ɵɵpipeBind1(9, 27, ctx.isPinned$));
        } }, directives: [ScalarCardComponent], pipes: [AsyncPipe], styles: ["[_nghost-%COMP%] {\n        display: block;\n        height: 100%;\n      }"], changeDetection: 0 });

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * @fileoverview Utilities for formatting histogram data.
     *
     * For the traditional TensorBoard logic, see
     * third_party/tensorboard/plugins/histogram/tf_histogram_dashboard/histogramCore.ts
     */
    const DEFAULT_BIN_COUNT = 30;
    /**
     * Histogram normalization logic.
     *
     * - Create a normalized bin template with 'binCount' # of bins. The range of
     *   all histograms should fit into this new bin list.
     * - Each histogram gets its own copy of the normalized bins.
     * - Each histogram's old counts are redistributed among their new bins.
     */
    function buildNormalizedHistograms(histograms, binCount = DEFAULT_BIN_COUNT) {
        if (!histograms.length || binCount < 1) {
            return [];
        }
        const range = getBinRange(histograms);
        // If the output range is 0 width, use a default non 0 range.
        if (range && range.left === range.right) {
            range.right = range.right * 1.1 + 1;
            range.left = range.left / 1.1 - 1;
        }
        return histograms.map((histogram) => {
            return {
                step: histogram.step,
                wallTime: histogram.wallTime,
                bins: range ? rebuildBins(histogram.bins, range, binCount) : [],
            };
        });
    }
    /**
     * Computes a range that covers the bins of all input histograms. Fields may
     * be null if the histogram bins were empty.
     *
     * For example,
     * histogram[0]: [          ][  ]
     * histogram[1]:        [               ]
     * result:       [                      ]
     */
    function getBinRange(histograms) {
        let left = null;
        let right = null;
        for (const { bins } of histograms) {
            if (!bins.length) {
                continue;
            }
            const lastBin = bins[bins.length - 1];
            const histogramLeft = bins[0].x;
            const histogramRight = lastBin.x + lastBin.dx;
            if (left === null || histogramLeft < left) {
                left = histogramLeft;
            }
            if (right === null || histogramRight > right) {
                right = histogramRight;
            }
        }
        if (left === null || right === null) {
            return null;
        }
        return { left, right };
    }
    /**
     * Builds a new list of 'binCount' bins.The 'y' counts from input bins are
     * distributed among the new bins based on amount of overlap.
     * Input bins must be sorted in increasing order and non-overlapping.
     *
     * Characteristics:
     * - The output bins are guaranteed contiguous, non-overlapping, equal width,
     *   and nonzero width.
     * - Handles 0 width input bins. When a 0 width bin is between 2 output bins,
     *   its counts are distributed evenly between the neighboring bins.
     *
     * For example,
     * bins:       [ 5 ][   10   ]
     * range:      [                  ]
     * binsCount:  2
     * results:    [   10   ][    5   ]
     */
    function rebuildBins(bins, range, binCount) {
        const results = [];
        const { left, right } = range;
        const dx = (right - left) / binCount;
        let binIndex = 0;
        let nextBinContribution = 0;
        for (let i = 0; i < binCount; i++) {
            const resultLeft = left + i * dx;
            const resultRight = resultLeft + dx;
            const isLastResultBin = i === binCount - 1;
            let resultY = nextBinContribution;
            nextBinContribution = 0;
            while (binIndex < bins.length) {
                const bin = bins[binIndex];
                const contribution = getBinContribution(bin, resultLeft, resultRight, !isLastResultBin);
                resultY += contribution.curr;
                nextBinContribution += contribution.next;
                // When the result bin completes, break without incrementing binIndex, in
                // case it contributes to the the next result bin.
                if (bin.x + bin.dx > resultRight) {
                    break;
                }
                binIndex++;
            }
            results.push({ x: resultLeft, dx, y: resultY });
        }
        return results;
    }
    /**
     * Computes how much of the input bin's 'y' counts should be allocated to a new
     * range. For 0 width input bins, the allocation may be split in half across 2
     * bins.
     */
    function getBinContribution(bin, resultLeft, resultRight, resultHasRightNeighbor) {
        const binLeft = bin.x;
        const binRight = bin.x + bin.dx;
        if (binLeft > resultRight || binRight < resultLeft) {
            return { curr: 0, next: 0 };
        }
        if (bin.dx === 0) {
            if (resultHasRightNeighbor && binRight === resultRight) {
                return { curr: 0.5 * bin.y, next: 0.5 * bin.y };
            }
            return { curr: bin.y, next: 0 };
        }
        const intersection = Math.min(binRight, resultRight) - Math.max(binLeft, resultLeft);
        return { curr: (bin.y * intersection) / bin.dx, next: 0 };
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/widgets/histogram/histogram_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class HistogramComponent {
        /**
         * @param {?} host
         */
        constructor(host) {
            this.host = host;
            this.mode = HistogramMode.OFFSET;
            this.timeProperty = TimeProperty.STEP;
            this.element = (/** @type {?} */ (document.createElement('vz-histogram-timeseries')));
            // Must set optional input values here since they won't be part of the
            // ngOnChanges if the parent does not override the value.
            this.element.mode = this.mode;
            this.element.timeProperty = this.timeProperty;
        }
        /**
         * @return {?}
         */
        ngOnInit() {
            this.host.nativeElement.appendChild(this.element);
        }
        /**
         * TODO(psybuzz): perform the same re-bucketing logic as the Polymer side,
         * instead of accepting the data as-is.
         *
         * The Polymer side forces the histogram at each step into a compressed
         * histogram with exactly 30 bins for a couple reasons:
         * - High number of bins will render visual artifacts.
         * - VzHistogramTimeseries expects all histograms at each step to have the
         *   same number of bins. Hovering over a HistogramComponent with bin count
         *   inequality throws errors.
         * See tensorboard/plugins/histogram/tf_histogram_dashboard/histogramCore.ts
         * @param {?} changes
         * @return {?}
         */
        ngOnChanges(changes) {
            if (changes['name'] || changes['data']) {
                /** @type {?} */
                const formattedData = this.data.map((/**
                 * @param {?} datum
                 * @return {?}
                 */
                (datum) => {
                    const { step, bins } = datum;
                    return { step, bins, wall_time: datum.wallTime };
                }));
                this.element.setSeriesData(this.name, formattedData);
            }
            if (changes['mode']) {
                this.element.mode = this.mode;
            }
            if (changes['timeProperty']) {
                this.element.timeProperty = this.timeProperty;
            }
            if (changes['colorScale'] && this.colorScale) {
                this.element.colorScale = this.colorScale;
            }
        }
        /**
         * @return {?}
         */
        redraw() {
            this.element.redraw();
        }
    }
    HistogramComponent.decorators = [
        { type: Component, args: [{
                    selector: 'tb-histogram',
                    template: '',
                    styles: [
                        `
      :host {
        display: flex;
      }
    `,
                    ],
                },] },
    ];
    /** @nocollapse */
    HistogramComponent.ctorParameters = () => [
        { type: ElementRef }
    ];
    HistogramComponent.propDecorators = {
        mode: [{ type: Input }],
        timeProperty: [{ type: Input }],
        colorScale: [{ type: Input }],
        name: [{ type: Input }],
        data: [{ type: Input }]
    };
    /** @nocollapse */ HistogramComponent.ɵfac = function HistogramComponent_Factory(t) { return new (t || HistogramComponent)(ɵɵdirectiveInject(ElementRef)); };
    /** @nocollapse */ HistogramComponent.ɵcmp = ɵɵdefineComponent({ type: HistogramComponent, selectors: [["tb-histogram"]], inputs: { mode: "mode", timeProperty: "timeProperty", colorScale: "colorScale", name: "name", data: "data" }, features: [ɵɵNgOnChangesFeature], decls: 0, vars: 0, template: function HistogramComponent_Template(rf, ctx) { }, styles: ["[_nghost-%COMP%] {\n        display: flex;\n      }"] });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/views/card_renderer/histogram_card_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    var I18N_0$4;
    if (typeof ngI18nClosureMode !== "undefined" && ngI18nClosureMode) {
        /**
         * @desc A button to pin a card.
         */ 
        const MSG_EXTERNAL_7284606426234375344$$TENSORBOARD_WEBAPP_METRICS_VIEWS_CARD_RENDERER_HISTOGRAM_CARD_COMPONENT_TS_1 = goog.getMsg("Pin card");
        I18N_0$4 = MSG_EXTERNAL_7284606426234375344$$TENSORBOARD_WEBAPP_METRICS_VIEWS_CARD_RENDERER_HISTOGRAM_CARD_COMPONENT_TS_1;
    }
    else {
        I18N_0$4 = $localize `:A button to pin a card.␟e665dc712bd5f18d4dfa3a29e125d565cc51e2f6␟7284606426234375344:Pin card`;
    }
    const _c2$a = ["aria-label", I18N_0$4];
    var I18N_3$2;
    if (typeof ngI18nClosureMode !== "undefined" && ngI18nClosureMode) {
        /**
         * @desc A button on a histogram card that toggles full size mode.
         */ 
        const MSG_EXTERNAL_915721563638926597$$TENSORBOARD_WEBAPP_METRICS_VIEWS_CARD_RENDERER_HISTOGRAM_CARD_COMPONENT_TS_4 = goog.getMsg("Toggle full size mode");
        I18N_3$2 = MSG_EXTERNAL_915721563638926597$$TENSORBOARD_WEBAPP_METRICS_VIEWS_CARD_RENDERER_HISTOGRAM_CARD_COMPONENT_TS_4;
    }
    else {
        I18N_3$2 = $localize `:A button on a histogram card that toggles full size mode.␟fc8f767d0b9f930187a1bae34477ad28736ece33␟915721563638926597:Toggle full size mode`;
    }
    const _c5$4 = ["aria-label", I18N_3$2];
    function HistogramCardComponent_span_12_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "span", 12);
        ɵɵelement(1, "mat-spinner", 13);
        ɵɵelementEnd();
    } }
    function HistogramCardComponent_tb_histogram_13_Template(rf, ctx) { if (rf & 1) {
        const _r5 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "tb-histogram", 14);
        ɵɵlistener("onResize", function HistogramCardComponent_tb_histogram_13_Template_tb_histogram_onResize_0_listener() { ɵɵrestoreView(_r5); const ctx_r4 = ɵɵnextContext(); return ctx_r4.redraw(); });
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r1 = ɵɵnextContext();
        ɵɵproperty("name", ctx_r1.runId)("data", ctx_r1.data)("mode", ctx_r1.mode)("timeProperty", ctx_r1.timeProperty(ctx_r1.xAxisType))("colorScale", ctx_r1.runColorScale)("resizeEventDebouncePeriodInMs", ctx_r1.RESIZE_REDRAW_DEBOUNCE_TIME_IN_MS);
    } }
    function HistogramCardComponent_ng_template_14_div_0_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 16);
        ɵɵtext(1, " Data failed to load. ");
        ɵɵelementEnd();
    } }
    function HistogramCardComponent_ng_template_14_Template(rf, ctx) { if (rf & 1) {
        ɵɵtemplate(0, HistogramCardComponent_ng_template_14_div_0_Template, 2, 0, "div", 15);
    } if (rf & 2) {
        const ctx_r3 = ɵɵnextContext();
        ɵɵproperty("ngIf", ctx_r3.loadState === ctx_r3.DataLoadState.FAILED);
    } }
    const _c6$2 = function (a0) { return { backgroundColor: a0 }; };
    /** @type {?} */
    const RESIZE_REDRAW_DEBOUNCE_TIME_IN_MS$1 = 50;
    class HistogramCardComponent {
        /**
         * @param {?} ref
         */
        constructor(ref) {
            this.ref = ref;
            this.RESIZE_REDRAW_DEBOUNCE_TIME_IN_MS = RESIZE_REDRAW_DEBOUNCE_TIME_IN_MS$1;
            this.DataLoadState = DataLoadState;
            this.onFullSizeToggle = new EventEmitter();
            this.onPinClicked = new EventEmitter();
        }
        /**
         * @param {?} xAxisType
         * @return {?}
         */
        timeProperty(xAxisType) {
            switch (xAxisType) {
                case XAxisType$1.STEP:
                    return TimeProperty.STEP;
                case XAxisType$1.WALL_TIME:
                    return TimeProperty.WALL_TIME;
                case XAxisType$1.RELATIVE:
                    return TimeProperty.RELATIVE;
                default:
                    throw new Error('Invalid xAxisType for histogram time property.');
            }
        }
        /**
         * @return {?}
         */
        redraw() {
            if (this.histogramComponent) {
                // Only redraw when it is visible (and thus have width and height).
                if (this.ref.nativeElement.clientHeight) {
                    this.histogramComponent.redraw();
                }
            }
        }
    }
    HistogramCardComponent.decorators = [
        { type: Component, args: [{
                    selector: 'histogram-card-component',
                    templateUrl: 'histogram_card_component.ng.html',
                    styleUrls: ['histogram_card_component.css'],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    HistogramCardComponent.ctorParameters = () => [
        { type: ElementRef }
    ];
    HistogramCardComponent.propDecorators = {
        loadState: [{ type: Input }],
        title: [{ type: Input }],
        tag: [{ type: Input }],
        runId: [{ type: Input }],
        data: [{ type: Input }],
        mode: [{ type: Input }],
        xAxisType: [{ type: Input }],
        runColorScale: [{ type: Input }],
        showFullSize: [{ type: Input }],
        isPinned: [{ type: Input }],
        onFullSizeToggle: [{ type: Output }],
        onPinClicked: [{ type: Output }],
        histogramComponent: [{ type: ViewChild, args: [HistogramComponent,] }]
    };
    /** @nocollapse */ HistogramCardComponent.ɵfac = function HistogramCardComponent_Factory(t) { return new (t || HistogramCardComponent)(ɵɵdirectiveInject(ElementRef)); };
    /** @nocollapse */ HistogramCardComponent.ɵcmp = ɵɵdefineComponent({ type: HistogramCardComponent, selectors: [["histogram-card-component"]], viewQuery: function HistogramCardComponent_Query(rf, ctx) { if (rf & 1) {
            ɵɵviewQuery(HistogramComponent, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.histogramComponent = _t.first);
        } }, inputs: { loadState: "loadState", title: "title", tag: "tag", runId: "runId", data: "data", mode: "mode", xAxisType: "xAxisType", runColorScale: "runColorScale", showFullSize: "showFullSize", isPinned: "isPinned" }, outputs: { onFullSizeToggle: "onFullSizeToggle", onPinClicked: "onPinClicked" }, decls: 16, vars: 12, consts: [[1, "heading"], [1, "tag", 3, "title", "value"], [1, "run"], [1, "dot", 3, "ngStyle"], [1, "run-text", 3, "runId"], [1, "controls"], ["mat-icon-button", "", 1, "pin-button", 3, "click", 6, "aria-label"], [3, "svgIcon"], ["mat-icon-button", "", "title", "Toggle full size mode", 3, "click", 6, "aria-label"], ["class", "spinner", 4, "ngIf"], ["detectResize", "", 3, "name", "data", "mode", "timeProperty", "colorScale", "resizeEventDebouncePeriodInMs", "onResize", 4, "ngIf", "ngIfElse"], ["noData", ""], [1, "spinner"], ["diameter", "18"], ["detectResize", "", 3, "name", "data", "mode", "timeProperty", "colorScale", "resizeEventDebouncePeriodInMs", "onResize"], ["class", "empty-message", 4, "ngIf"], [1, "empty-message"]], template: function HistogramCardComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "div", 0);
            ɵɵelement(1, "tb-truncated-path", 1);
            ɵɵelementStart(2, "div", 2);
            ɵɵelement(3, "span", 3);
            ɵɵelement(4, "card-run-name", 4);
            ɵɵelementEnd();
            ɵɵelementStart(5, "span", 5);
            ɵɵelementStart(6, "button", 6);
            ɵɵi18nAttributes(7, _c2$a);
            ɵɵlistener("click", function HistogramCardComponent_Template_button_click_6_listener() { return ctx.onPinClicked.emit(!ctx.isPinned); });
            ɵɵelement(8, "mat-icon", 7);
            ɵɵelementEnd();
            ɵɵelementStart(9, "button", 8);
            ɵɵi18nAttributes(10, _c5$4);
            ɵɵlistener("click", function HistogramCardComponent_Template_button_click_9_listener() { return ctx.onFullSizeToggle.emit(); });
            ɵɵelement(11, "mat-icon", 7);
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵtemplate(12, HistogramCardComponent_span_12_Template, 2, 0, "span", 9);
            ɵɵelementEnd();
            ɵɵtemplate(13, HistogramCardComponent_tb_histogram_13_Template, 1, 6, "tb-histogram", 10);
            ɵɵtemplate(14, HistogramCardComponent_ng_template_14_Template, 1, 1, "ng-template", null, 11, ɵɵtemplateRefExtractor);
        } if (rf & 2) {
            const _r2 = ɵɵreference(15);
            ɵɵadvance(1);
            ɵɵpropertyInterpolate("title", ctx.tag);
            ɵɵpropertyInterpolate("value", ctx.title);
            ɵɵadvance(2);
            ɵɵproperty("ngStyle", ɵɵpureFunction1(10, _c6$2, ctx.runColorScale(ctx.runId)));
            ɵɵadvance(1);
            ɵɵproperty("runId", ctx.runId);
            ɵɵadvance(2);
            ɵɵattribute("title", ctx.isPinned ? "Unpin card" : "Pin card");
            ɵɵadvance(2);
            ɵɵproperty("svgIcon", ctx.isPinned ? "keep_24px" : "keep_outline_24px");
            ɵɵadvance(3);
            ɵɵproperty("svgIcon", ctx.showFullSize ? "fullscreen_exit_24px" : "fullscreen_24px");
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx.loadState === ctx.DataLoadState.LOADING);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx.data && ctx.data.length)("ngIfElse", _r2);
        } }, directives: [TruncatedPathComponent, NgStyle, RunNameContainer, MatButton, MatIcon, NgIf, MatSpinner, HistogramComponent, ResizeDetectorDirective], styles: [".mat-badge-content[_ngcontent-%COMP%]{font-weight:600;font-size:12px;font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:9px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:24px}.mat-h1[_ngcontent-%COMP%], .mat-headline[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%]{font:400 24px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h2[_ngcontent-%COMP%], .mat-title[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h3[_ngcontent-%COMP%], .mat-subheading-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%]{font:400 16px/28px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h4[_ngcontent-%COMP%], .mat-subheading-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%]{font:400 15px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h5[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%]{font:400 calc(14px * 0.83)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-h6[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:400 calc(14px * 0.67)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-body-strong[_ngcontent-%COMP%], .mat-body-2[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   p[_ngcontent-%COMP%]{margin:0 0 12px}.mat-small[_ngcontent-%COMP%], .mat-caption[_ngcontent-%COMP%]{font:400 12px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-display-4[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-4[_ngcontent-%COMP%]{font:300 112px/112px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.05em;margin:0 0 56px}.mat-display-3[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-3[_ngcontent-%COMP%]{font:400 56px/56px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.02em;margin:0 0 64px}.mat-display-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-2[_ngcontent-%COMP%]{font:400 45px/48px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.005em;margin:0 0 64px}.mat-display-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-1[_ngcontent-%COMP%]{font:400 34px/40px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 64px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%], .mat-flat-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-button-toggle[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card-title[_ngcontent-%COMP%]{font-size:24px;font-weight:500}.mat-card-header[_ngcontent-%COMP%]   .mat-card-title[_ngcontent-%COMP%]{font-size:20px}.mat-card-subtitle[_ngcontent-%COMP%], .mat-card-content[_ngcontent-%COMP%]{font-size:14px}.mat-checkbox[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-checkbox-layout[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{line-height:24px}.mat-chip[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-chip[_ngcontent-%COMP%]   .mat-chip-trailing-icon.mat-icon[_ngcontent-%COMP%], .mat-chip[_ngcontent-%COMP%]   .mat-chip-remove.mat-icon[_ngcontent-%COMP%]{font-size:18px}.mat-table[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-header-cell[_ngcontent-%COMP%]{font-size:12px;font-weight:500}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{font-size:14px}.mat-calendar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-calendar-body[_ngcontent-%COMP%]{font-size:13px}.mat-calendar-body-label[_ngcontent-%COMP%], .mat-calendar-period-button[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-calendar-table-header[_ngcontent-%COMP%]   th[_ngcontent-%COMP%]{font-size:11px;font-weight:400}.mat-dialog-title[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-expansion-panel-header[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:15px;font-weight:400}.mat-expansion-panel-content[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field[_ngcontent-%COMP%]{font-size:inherit;font-weight:400;line-height:1.125;font-family:Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.34375em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{font-size:150%;line-height:1.125}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]{height:1.5em;width:1.5em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{height:1.125em;line-height:1.125}.mat-form-field-infix[_ngcontent-%COMP%]{padding:.5em 0;border-top:.84375em solid transparent}.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34375em) scale(0.75);width:133.3333333333%}.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34374em) scale(0.75);width:133.3333433333%}.mat-form-field-label-wrapper[_ngcontent-%COMP%]{top:-0.84375em;padding-top:.84375em}.mat-form-field-label[_ngcontent-%COMP%]{top:1.34375em}.mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.34375em}.mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{font-size:75%;margin-top:.6666666667em;top:calc(100% - 1.7916666667em)}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.4375em 0}.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.001px);-ms-transform:translateY(-1.28125em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00101px);-ms-transform:translateY(-1.28124em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00102px);-ms-transform:translateY(-1.28123em) scale(0.75);width:133.3333533333%}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.28125em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{margin-top:.5416666667em;top:calc(100% - 1.6666666667em)}@media print{.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28122em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28121em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.2812em) scale(0.75)}}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.25em 0 .75em 0}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.09375em;margin-top:-0.5em}.mat-form-field-appearance-fill.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59374em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:1em 0 1em 0}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.84375em;margin-top:-0.25em}.mat-form-field-appearance-outline.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59374em) scale(0.75);width:133.3333433333%}.mat-grid-tile-header[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]{font-size:14px}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2), .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}input.mat-input-element[_ngcontent-%COMP%]{margin-top:-0.0625em}.mat-menu-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:400}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px}.mat-radio-button[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select-trigger[_ngcontent-%COMP%]{height:1.125em}.mat-slide-toggle-content[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-slider-thumb-label-text[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-stepper-vertical[_ngcontent-%COMP%], .mat-stepper-horizontal[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-step-label[_ngcontent-%COMP%]{font-size:14px;font-weight:400}.mat-step-sub-label-error[_ngcontent-%COMP%]{font-weight:normal}.mat-step-label-error[_ngcontent-%COMP%]{font-size:14px}.mat-step-label-selected[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-tab-group[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-toolbar[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0}.mat-tooltip[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:10px;padding-top:6px;padding-bottom:6px}.mat-tooltip-handset[_ngcontent-%COMP%]{font-size:14px;padding-top:8px;padding-bottom:8px}.mat-list-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:16px}.mat-optgroup-label[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-simple-snackbar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px}.mat-simple-snackbar-action[_ngcontent-%COMP%]{line-height:1;font-family:inherit;font-size:inherit;font-weight:500}.mat-tree[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{font-weight:400;font-size:14px}.mat-ripple[_ngcontent-%COMP%]{overflow:hidden;position:relative}.mat-ripple[_ngcontent-%COMP%]:not(:empty){transform:translateZ(0)}.mat-ripple.mat-ripple-unbounded[_ngcontent-%COMP%]{overflow:visible}.mat-ripple-element[_ngcontent-%COMP%]{position:absolute;border-radius:50%;pointer-events:none;transition:opacity,transform 0ms cubic-bezier(0, 0, 0.2, 1);transform:scale(0)}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{display:none}.cdk-visually-hidden[_ngcontent-%COMP%]{border:0;clip:rect(0 0 0 0);height:1px;margin:-1px;overflow:hidden;padding:0;position:absolute;width:1px;outline:0;-webkit-appearance:none;-moz-appearance:none}.cdk-overlay-container[_ngcontent-%COMP%], .cdk-global-overlay-wrapper[_ngcontent-%COMP%]{pointer-events:none;top:0;left:0;height:100%;width:100%}.cdk-overlay-container[_ngcontent-%COMP%]{position:fixed;z-index:1000}.cdk-overlay-container[_ngcontent-%COMP%]:empty{display:none}.cdk-global-overlay-wrapper[_ngcontent-%COMP%]{display:flex;position:absolute;z-index:1000}.cdk-overlay-pane[_ngcontent-%COMP%]{position:absolute;pointer-events:auto;box-sizing:border-box;z-index:1000;display:flex;max-width:100%;max-height:100%}.cdk-overlay-backdrop[_ngcontent-%COMP%]{position:absolute;top:0;bottom:0;left:0;right:0;z-index:1000;pointer-events:auto;-webkit-tap-highlight-color:transparent;transition:opacity 400ms cubic-bezier(0.25, 0.8, 0.25, 1);opacity:0}.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:1}@media screen and (-ms-high-contrast: active){.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:.6}}.cdk-overlay-dark-backdrop[_ngcontent-%COMP%]{background:rgba(0,0,0,.32)}.cdk-overlay-transparent-backdrop[_ngcontent-%COMP%], .cdk-overlay-transparent-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:0}.cdk-overlay-connected-position-bounding-box[_ngcontent-%COMP%]{position:absolute;z-index:1000;display:flex;flex-direction:column;min-width:1px;min-height:1px}.cdk-global-scrollblock[_ngcontent-%COMP%]{position:fixed;width:100%;overflow-y:scroll}@keyframes cdk-text-field-autofill-start{}@keyframes cdk-text-field-autofill-end{}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:-webkit-autofill{animation:cdk-text-field-autofill-start 0s 1ms}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:not(:-webkit-autofill){animation:cdk-text-field-autofill-end 0s 1ms}textarea.cdk-textarea-autosize[_ngcontent-%COMP%]{resize:none}textarea.cdk-textarea-autosize-measuring[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:auto !important;overflow:hidden !important}textarea.cdk-textarea-autosize-measuring-firefox[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:0 !important}.mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.1)}.mat-option[_ngcontent-%COMP%]{color:#212121}.mat-option[_ngcontent-%COMP%]:hover:not(.mat-option-disabled), .mat-option[_ngcontent-%COMP%]:focus:not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple):not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-active[_ngcontent-%COMP%]{background:rgba(0,0,0,.04);color:#212121}.mat-option.mat-option-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-primary[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f57c00}.mat-accent[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f44336}.mat-optgroup-label[_ngcontent-%COMP%]{color:#616161}.mat-optgroup-disabled[_ngcontent-%COMP%]   .mat-optgroup-label[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-pseudo-checkbox[_ngcontent-%COMP%]{color:#616161}.mat-pseudo-checkbox[_ngcontent-%COMP%]::after{color:#fafafa}.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{color:#b0b0b0}.mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f57c00}.mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f44336}.mat-pseudo-checkbox-checked.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{background:#b0b0b0}.mat-elevation-z0[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z1[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z2[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z3[_ngcontent-%COMP%]{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z4[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z5[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 5px 8px 0px rgba(0, 0, 0, 0.14),0px 1px 14px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z6[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z7[_ngcontent-%COMP%]{box-shadow:0px 4px 5px -2px rgba(0, 0, 0, 0.2),0px 7px 10px 1px rgba(0, 0, 0, 0.14),0px 2px 16px 1px rgba(0, 0, 0, 0.12)}.mat-elevation-z8[_ngcontent-%COMP%]{box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z9[_ngcontent-%COMP%]{box-shadow:0px 5px 6px -3px rgba(0, 0, 0, 0.2),0px 9px 12px 1px rgba(0, 0, 0, 0.14),0px 3px 16px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z10[_ngcontent-%COMP%]{box-shadow:0px 6px 6px -3px rgba(0, 0, 0, 0.2),0px 10px 14px 1px rgba(0, 0, 0, 0.14),0px 4px 18px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z11[_ngcontent-%COMP%]{box-shadow:0px 6px 7px -4px rgba(0, 0, 0, 0.2),0px 11px 15px 1px rgba(0, 0, 0, 0.14),0px 4px 20px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z12[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z13[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 13px 19px 2px rgba(0, 0, 0, 0.14),0px 5px 24px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z14[_ngcontent-%COMP%]{box-shadow:0px 7px 9px -4px rgba(0, 0, 0, 0.2),0px 14px 21px 2px rgba(0, 0, 0, 0.14),0px 5px 26px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z15[_ngcontent-%COMP%]{box-shadow:0px 8px 9px -5px rgba(0, 0, 0, 0.2),0px 15px 22px 2px rgba(0, 0, 0, 0.14),0px 6px 28px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z16[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z17[_ngcontent-%COMP%]{box-shadow:0px 8px 11px -5px rgba(0, 0, 0, 0.2),0px 17px 26px 2px rgba(0, 0, 0, 0.14),0px 6px 32px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z18[_ngcontent-%COMP%]{box-shadow:0px 9px 11px -5px rgba(0, 0, 0, 0.2),0px 18px 28px 2px rgba(0, 0, 0, 0.14),0px 7px 34px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z19[_ngcontent-%COMP%]{box-shadow:0px 9px 12px -6px rgba(0, 0, 0, 0.2),0px 19px 29px 2px rgba(0, 0, 0, 0.14),0px 7px 36px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z20[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 20px 31px 3px rgba(0, 0, 0, 0.14),0px 8px 38px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z21[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 21px 33px 3px rgba(0, 0, 0, 0.14),0px 8px 40px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z22[_ngcontent-%COMP%]{box-shadow:0px 10px 14px -6px rgba(0, 0, 0, 0.2),0px 22px 35px 3px rgba(0, 0, 0, 0.14),0px 8px 42px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z23[_ngcontent-%COMP%]{box-shadow:0px 11px 14px -7px rgba(0, 0, 0, 0.2),0px 23px 36px 3px rgba(0, 0, 0, 0.14),0px 9px 44px 8px rgba(0, 0, 0, 0.12)}.mat-elevation-z24[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12)}.mat-app-background[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-theme-loaded-marker[_ngcontent-%COMP%]{display:none}.mat-autocomplete-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-autocomplete-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover){background:#fff}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover):not(.mat-option-disabled){color:#212121}.mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f57c00}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{outline:solid 1px;border-radius:0}.mat-badge-accent[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-badge-warn[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f44336}.mat-badge[_ngcontent-%COMP%]{position:relative}.mat-badge-hidden[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{display:none}.mat-badge-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#b9b9b9;color:#757575}.mat-badge-content[_ngcontent-%COMP%]{position:absolute;text-align:center;display:inline-block;border-radius:50%;transition:transform 200ms ease-in-out;transform:scale(0.6);overflow:hidden;white-space:nowrap;text-overflow:ellipsis;pointer-events:none}.ng-animate-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%], .mat-badge-content._mat-animation-noopable[_ngcontent-%COMP%]{transition:none}.mat-badge-content.mat-badge-active[_ngcontent-%COMP%]{transform:none}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:16px;height:16px;line-height:16px}.mat-badge-small.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-8px}.mat-badge-small.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-8px}.mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-16px}.mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-16px}.mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-8px}.mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-8px}.mat-badge-medium[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:22px;height:22px;line-height:22px}.mat-badge-medium.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-11px}.mat-badge-medium.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-11px}.mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-22px}.mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-22px}.mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-11px}.mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-11px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:28px;height:28px;line-height:28px}.mat-badge-large.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-14px}.mat-badge-large.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-14px}.mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-28px}.mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-28px}.mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-14px}.mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-14px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]{color:inherit;background:transparent}.mat-button.mat-primary[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-button[disabled][disabled][_ngcontent-%COMP%], .mat-icon-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-icon-button[disabled][disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-stroked-button[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f44336}.mat-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:transparent}.mat-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{opacity:.1;background-color:currentColor}.mat-button-focus-overlay[_ngcontent-%COMP%]{background:#000}.mat-stroked-button[_ngcontent-%COMP%]:not([disabled]){border-color:rgba(0,0,0,.12)}.mat-flat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{color:#212121;background-color:#fff}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-flat-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-stroked-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-flat-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-raised-button[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone[_ngcontent-%COMP%], .mat-button-toggle-group[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{box-shadow:none}.mat-button-toggle[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-button-toggle[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121;background:#fff}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:#000}.mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12)}[dir=rtl][_ngcontent-%COMP%]   .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-group-appearance-standard.mat-button-toggle-vertical[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:none;border-top:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#616161}.mat-button-toggle-checked.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121}.mat-button-toggle-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.26);background-color:#eee}.mat-button-toggle-disabled.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{background:#fff}.mat-button-toggle-disabled.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#bdbdbd}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{border:solid 1px rgba(0,0,0,.12)}.mat-card[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-card[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-card.mat-card-flat[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-card-subtitle[_ngcontent-%COMP%]{color:#616161}.mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#616161}.mat-checkbox-checkmark[_ngcontent-%COMP%]{fill:#fafafa}.mat-checkbox-checkmark-path[_ngcontent-%COMP%]{stroke:#fafafa !important}.mat-checkbox-mixedmark[_ngcontent-%COMP%]{background-color:#fafafa}.mat-checkbox-indeterminate.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f57c00}.mat-checkbox-indeterminate.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#ff9800}.mat-checkbox-indeterminate.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f44336}.mat-checkbox-disabled.mat-checkbox-checked[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-disabled.mat-checkbox-indeterminate[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]:not(.mat-checkbox-checked)   .mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{color:#616161}.mat-checkbox[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%]{background:#f57c00}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%]{background:#ff9800}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%]{background:#f44336}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#212121}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#212121;opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled):active{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled)   .mat-chip-remove[_ngcontent-%COMP%]:hover{opacity:.54}.mat-chip.mat-standard-chip.mat-chip-disabled[_ngcontent-%COMP%]{opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]::after{background:#000}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-table[_ngcontent-%COMP%]{background:#fff}.mat-table[_ngcontent-%COMP%]   thead[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tbody[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tfoot[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], [mat-header-row][_ngcontent-%COMP%], [mat-row][_ngcontent-%COMP%], [mat-footer-row][_ngcontent-%COMP%], .mat-table-sticky[_ngcontent-%COMP%]{background:inherit}mat-row[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], th.mat-header-cell[_ngcontent-%COMP%], td.mat-cell[_ngcontent-%COMP%], td.mat-footer-cell[_ngcontent-%COMP%]{border-bottom-color:rgba(0,0,0,.12)}.mat-header-cell[_ngcontent-%COMP%]{color:#616161}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{color:#212121}.mat-calendar-arrow[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.54)}.mat-datepicker-toggle[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-next-button[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-previous-button[_ngcontent-%COMP%]{color:rgba(0,0,0,.54)}.mat-calendar-table-header[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-calendar-table-header-divider[_ngcontent-%COMP%]::after{background:rgba(0,0,0,.12)}.mat-calendar-body-label[_ngcontent-%COMP%]{color:#616161}.mat-calendar-body-cell-content[_ngcontent-%COMP%]{color:#212121;border-color:transparent}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){color:#757575}.mat-calendar-body-cell[_ngcontent-%COMP%]:not(.mat-calendar-body-disabled):hover > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-keyboard-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-program-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){background-color:rgba(0,0,0,.04)}.mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.38)}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.18)}.mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.4)}.mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12);background-color:#fff;color:#212121}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.4)}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.4)}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content-touch[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-datepicker-toggle-active[_ngcontent-%COMP%]{color:#f57c00}.mat-datepicker-toggle-active.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-datepicker-toggle-active.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-dialog-container[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-divider[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-divider-vertical[_ngcontent-%COMP%]{border-right-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-expansion-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-action-row[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-keyboard-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-program-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded)   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover:not([aria-disabled=true]){background:rgba(0,0,0,.04)}@media(hover: none){.mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded):not([aria-disabled=true])   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover{background:#fff}}.mat-expansion-panel-header-title[_ngcontent-%COMP%]{color:#212121}.mat-expansion-panel-header-description[_ngcontent-%COMP%], .mat-expansion-indicator[_ngcontent-%COMP%]::after{color:#616161}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-title[_ngcontent-%COMP%], .mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-description[_ngcontent-%COMP%]{color:inherit}.mat-form-field-label[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-hint[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-focused[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-ripple[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid)   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f57c00}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-accent   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#ff9800}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-warn   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#f44336}.mat-error[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-hint[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-legacy.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-standard[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-standard.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.04)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.02)}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:transparent}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.12)}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:rgba(0,0,0,.87)}.mat-form-field-appearance-outline.mat-focused[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field-appearance-outline.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-appearance-outline.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-invalid.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.06)}.mat-icon.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-icon.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-icon.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-type-mat-native-select[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#616161}.mat-input-element[_ngcontent-%COMP%]:disabled, .mat-form-field-type-mat-native-select.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#757575}.mat-input-element[_ngcontent-%COMP%]{caret-color:#f57c00}.mat-input-element[_ngcontent-%COMP%]::placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-moz-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-webkit-input-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]:-ms-input-placeholder{color:rgba(97,97,97,.42)}.mat-accent[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%], .mat-form-field-invalid[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#f44336}.mat-form-field-type-mat-native-select.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{color:#616161}.mat-list-item-disabled[_ngcontent-%COMP%]{background-color:#eee}.mat-list-option[_ngcontent-%COMP%]:hover, .mat-list-option[_ngcontent-%COMP%]:focus, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.04)}.mat-list-single-selected-option[_ngcontent-%COMP%], .mat-list-single-selected-option[_ngcontent-%COMP%]:hover, .mat-list-single-selected-option[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.12)}.mat-menu-panel[_ngcontent-%COMP%]{background:#fff}.mat-menu-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-menu-item[_ngcontent-%COMP%]{background:transparent;color:#212121}.mat-menu-item[disabled][_ngcontent-%COMP%], .mat-menu-item[disabled][_ngcontent-%COMP%]::after{color:rgba(0,0,0,.38)}.mat-menu-item[_ngcontent-%COMP%]   .mat-icon-no-color[_ngcontent-%COMP%], .mat-menu-item-submenu-trigger[_ngcontent-%COMP%]::after{color:rgba(0,0,0,.54)}.mat-menu-item[_ngcontent-%COMP%]:hover:not([disabled]), .mat-menu-item.cdk-program-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item.cdk-keyboard-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item-highlighted[_ngcontent-%COMP%]:not([disabled]){background:rgba(0,0,0,.04)}.mat-paginator[_ngcontent-%COMP%]{background:#fff}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{color:#616161}.mat-paginator-decrement[_ngcontent-%COMP%], .mat-paginator-increment[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54);border-right:2px solid rgba(0,0,0,.54)}.mat-paginator-first[_ngcontent-%COMP%], .mat-paginator-last[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54)}.mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-decrement[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-increment[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-first[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-last[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f57c00}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#ff9800}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f44336}.mat-progress-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f57c00}.mat-progress-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#ff9800}.mat-progress-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f44336}.mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#616161}.mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f57c00}.mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#ff9800}.mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#ff9800}.mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f44336}.mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f44336}.mat-radio-button.mat-radio-disabled.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-label-content[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-radio-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-select-value[_ngcontent-%COMP%]{color:#212121}.mat-select-placeholder[_ngcontent-%COMP%]{color:rgba(97,97,97,.42)}.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%]{color:#757575}.mat-select-arrow[_ngcontent-%COMP%]{color:#616161}.mat-select-panel[_ngcontent-%COMP%]{background:#fff}.mat-select-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-select-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple){background:rgba(0,0,0,.12)}.mat-form-field.mat-focused.mat-primary[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-invalid[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#757575}.mat-drawer-container[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-drawer[_ngcontent-%COMP%]{background-color:#fff;color:#212121}.mat-drawer.mat-drawer-push[_ngcontent-%COMP%]{background-color:#fff}.mat-drawer[_ngcontent-%COMP%]:not(.mat-drawer-side){box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-drawer-side[_ngcontent-%COMP%]{border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-backdrop.mat-drawer-shown[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.6)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.54)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.54)}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.54)}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle[_ngcontent-%COMP%]:not(.mat-checked)   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-slide-toggle-thumb[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12);background-color:#fafafa}.mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-primary[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f57c00}.mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-accent[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#ff9800}.mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-warn[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f44336}.mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.2)}.mat-slider[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%], .cdk-focused[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-disabled[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]   .mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing)   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26);background-color:transparent}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26)}.mat-slider-has-ticks[_ngcontent-%COMP%]   .mat-slider-wrapper[_ngcontent-%COMP%]::after{border-color:rgba(0,0,0,.7)}.mat-slider-horizontal[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to right, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent);background-image:-moz-repeating-linear-gradient(0.0001deg, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-slider-vertical[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to bottom, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-step-header.cdk-keyboard-focused[_ngcontent-%COMP%], .mat-step-header.cdk-program-focused[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]:hover{background-color:rgba(0,0,0,.04)}@media(hover: none){.mat-step-header[_ngcontent-%COMP%]:hover{background:none}}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-optional[_ngcontent-%COMP%]{color:#616161}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon[_ngcontent-%COMP%]{background-color:#616161;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-selected[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-done[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-edit[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-error[_ngcontent-%COMP%]{background-color:transparent;color:#f44336}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-active[_ngcontent-%COMP%]{color:#212121}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-error[_ngcontent-%COMP%]{color:#f44336}.mat-stepper-horizontal[_ngcontent-%COMP%], .mat-stepper-vertical[_ngcontent-%COMP%]{background-color:#fff}.mat-stepper-vertical-line[_ngcontent-%COMP%]::before{border-left-color:rgba(0,0,0,.12)}.mat-horizontal-stepper-header[_ngcontent-%COMP%]::before, .mat-horizontal-stepper-header[_ngcontent-%COMP%]::after, .mat-stepper-horizontal-line[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-sort-header-arrow[_ngcontent-%COMP%]{color:#616161}.mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-header[_ngcontent-%COMP%]{border-bottom:1px solid rgba(0,0,0,.12)}.mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%]{border-top:1px solid rgba(0,0,0,.12);border-bottom:none}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{color:#212121}.mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:#757575}.mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#212121}.mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#757575}.mat-tab-group[class*=mat-background-][_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar[class*=mat-background-][_ngcontent-%COMP%]{border-bottom:none;border-top:none}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-toolbar[_ngcontent-%COMP%]{background:#f5f5f5;color:#212121}.mat-toolbar.mat-primary[_ngcontent-%COMP%]{background:#f57c00;color:#fff}.mat-toolbar.mat-accent[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-toolbar.mat-warn[_ngcontent-%COMP%]{background:#f44336;color:#fff}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:currentColor}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:inherit}.mat-toolbar[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:currentColor}.mat-tooltip[_ngcontent-%COMP%]{background:rgba(97,97,97,.9)}.mat-tree[_ngcontent-%COMP%]{background:#fff}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{color:#212121}.mat-snack-bar-container[_ngcontent-%COMP%]{color:rgba(255,255,255,.7);background:#323232;box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-simple-snackbar-action[_ngcontent-%COMP%]{color:#ff9800}[_nghost-%COMP%]{display:flex;flex-direction:column;box-sizing:border-box;height:100%;overflow:auto;padding:16px;padding-top:4px}.heading[_ngcontent-%COMP%]{align-items:center;display:grid;grid-template-areas:\"tag controls\" \"run spinner\";grid-template-columns:1fr auto;font-size:14px;margin-bottom:4px}.tag[_ngcontent-%COMP%]{grid-area:tag;overflow:hidden}.pin-button[_ngcontent-%COMP%]   mat-icon[_ngcontent-%COMP%]{height:18px}.run[_ngcontent-%COMP%]{grid-area:run;display:flex;white-space:nowrap;font-size:13px;color:#616161}.run[_ngcontent-%COMP%]   .dot[_ngcontent-%COMP%]{flex:none;display:inline-block;width:13px;height:13px;border-radius:50%;margin-right:4px}.run[_ngcontent-%COMP%]   .run-text[_ngcontent-%COMP%]{overflow:hidden;text-overflow:ellipsis;max-width:120px}.controls[_ngcontent-%COMP%]{color:#616161;grid-area:controls;justify-self:flex-end;flex-shrink:0;margin-right:-12px;white-space:nowrap}.spinner[_ngcontent-%COMP%]{display:flex;grid-area:spinner;height:100%;justify-content:center;position:relative}mat-spinner[_ngcontent-%COMP%]{top:0;right:0;position:absolute}tb-histogram[_ngcontent-%COMP%]{flex-grow:1;overflow:hidden}.empty-message[_ngcontent-%COMP%]{margin-top:1em;font-size:13px}"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/views/card_renderer/histogram_card_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class HistogramCardContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.fullWidthChanged = new EventEmitter();
            this.fullHeightChanged = new EventEmitter();
            this.pinStateChanged = new EventEmitter();
            this.mode$ = this.store.select(getMetricsHistogramMode);
            this.xAxisType$ = this.store.select(getMetricsXAxisType);
            this.showFullSize = false;
        }
        /**
         * @private
         * @param {?} cardMetadata
         * @return {?}
         */
        isHistogramCardMetadata(cardMetadata) {
            const { plugin } = cardMetadata;
            return plugin === PluginType.HISTOGRAMS;
        }
        /**
         * @return {?}
         */
        onFullSizeToggle() {
            this.showFullSize = !this.showFullSize;
            this.fullWidthChanged.emit(this.showFullSize);
            this.fullHeightChanged.emit(this.showFullSize);
        }
        /**
         * Build observables once cardId is defined (after onInit).
         * @return {?}
         */
        ngOnInit() {
            /** @type {?} */
            const selectCardMetadata$ = this.store.select(getCardMetadata, this.cardId);
            /** @type {?} */
            const cardMetadata$ = selectCardMetadata$.pipe(filter((/**
             * @param {?} cardMetadata
             * @return {?}
             */
            (cardMetadata) => {
                return !!cardMetadata && this.isHistogramCardMetadata(cardMetadata);
            })), map((/**
             * @param {?} cardMetadata
             * @return {?}
             */
            (cardMetadata) => {
                return (/** @type {?} */ (cardMetadata));
            })));
            /** @type {?} */
            const metadataAndSeries$ = combineLatest([
                cardMetadata$,
                this.store.select(getCardTimeSeries, this.cardId),
            ]);
            this.data$ = metadataAndSeries$.pipe(map((/**
             * @param {?} __0
             * @return {?}
             */
            ([cardMetadata, runToSeries]) => {
                /** @type {?} */
                const runId = cardMetadata.runId;
                if (!runToSeries || !runToSeries.hasOwnProperty(runId)) {
                    return [];
                }
                /** @type {?} */
                const series = (/** @type {?} */ (runToSeries[runId]));
                /** @type {?} */
                const result = series.map((/**
                 * @param {?} datum
                 * @return {?}
                 */
                (datum) => {
                    const { wallTime, step } = datum;
                    /** @type {?} */
                    const bins = datum.bins.map((/**
                     * @param {?} bin
                     * @return {?}
                     */
                    (bin) => {
                        return { x: bin.min, dx: bin.max - bin.min, y: bin.count };
                    }));
                    return { wallTime, step, bins };
                }));
                return buildNormalizedHistograms(result);
            })));
            this.loadState$ = this.store.select(getCardLoadState, this.cardId);
            this.tag$ = cardMetadata$.pipe(map((/**
             * @param {?} cardMetadata
             * @return {?}
             */
            (cardMetadata) => {
                return cardMetadata.tag;
            })));
            this.title$ = this.tag$.pipe(map((/**
             * @param {?} tag
             * @return {?}
             */
            (tag) => {
                return getTagDisplayName(tag, this.groupName);
            })));
            this.runId$ = cardMetadata$.pipe(map((/**
             * @param {?} cardMetadata
             * @return {?}
             */
            (cardMetadata) => {
                return cardMetadata.runId;
            })));
            this.isPinned$ = this.store.select(getCardPinnedState, this.cardId);
        }
    }
    HistogramCardContainer.decorators = [
        { type: Component, args: [{
                    selector: 'histogram-card',
                    template: `
    <histogram-card-component
      [loadState]="loadState$ | async"
      [title]="title$ | async"
      [tag]="tag$ | async"
      [runId]="runId$ | async"
      [data]="data$ | async"
      [mode]="mode$ | async"
      [xAxisType]="xAxisType$ | async"
      [runColorScale]="runColorScale"
      [showFullSize]="showFullSize"
      [isPinned]="isPinned$ | async"
      (onFullSizeToggle)="onFullSizeToggle()"
      (onPinClicked)="pinStateChanged.emit($event)"
    ></histogram-card-component>
  `,
                    styles: [
                        `
      :host {
        display: block;
        height: 100%;
      }
    `,
                    ],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    HistogramCardContainer.ctorParameters = () => [
        { type: Store }
    ];
    HistogramCardContainer.propDecorators = {
        cardId: [{ type: Input }],
        groupName: [{ type: Input }],
        runColorScale: [{ type: Input }],
        fullWidthChanged: [{ type: Output }],
        fullHeightChanged: [{ type: Output }],
        pinStateChanged: [{ type: Output }]
    };
    /** @nocollapse */ HistogramCardContainer.ɵfac = function HistogramCardContainer_Factory(t) { return new (t || HistogramCardContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ HistogramCardContainer.ɵcmp = ɵɵdefineComponent({ type: HistogramCardContainer, selectors: [["histogram-card"]], inputs: { cardId: "cardId", groupName: "groupName", runColorScale: "runColorScale" }, outputs: { fullWidthChanged: "fullWidthChanged", fullHeightChanged: "fullHeightChanged", pinStateChanged: "pinStateChanged" }, decls: 9, vars: 26, consts: [[3, "loadState", "title", "tag", "runId", "data", "mode", "xAxisType", "runColorScale", "showFullSize", "isPinned", "onFullSizeToggle", "onPinClicked"]], template: function HistogramCardContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "histogram-card-component", 0);
            ɵɵlistener("onFullSizeToggle", function HistogramCardContainer_Template_histogram_card_component_onFullSizeToggle_0_listener() { return ctx.onFullSizeToggle(); })("onPinClicked", function HistogramCardContainer_Template_histogram_card_component_onPinClicked_0_listener($event) { return ctx.pinStateChanged.emit($event); });
            ɵɵpipe(1, "async");
            ɵɵpipe(2, "async");
            ɵɵpipe(3, "async");
            ɵɵpipe(4, "async");
            ɵɵpipe(5, "async");
            ɵɵpipe(6, "async");
            ɵɵpipe(7, "async");
            ɵɵpipe(8, "async");
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵproperty("loadState", ɵɵpipeBind1(1, 10, ctx.loadState$))("title", ɵɵpipeBind1(2, 12, ctx.title$))("tag", ɵɵpipeBind1(3, 14, ctx.tag$))("runId", ɵɵpipeBind1(4, 16, ctx.runId$))("data", ɵɵpipeBind1(5, 18, ctx.data$))("mode", ɵɵpipeBind1(6, 20, ctx.mode$))("xAxisType", ɵɵpipeBind1(7, 22, ctx.xAxisType$))("runColorScale", ctx.runColorScale)("showFullSize", ctx.showFullSize)("isPinned", ɵɵpipeBind1(8, 24, ctx.isPinned$));
        } }, directives: [HistogramCardComponent], pipes: [AsyncPipe], styles: ["[_nghost-%COMP%] {\n        display: block;\n        height: 100%;\n      }"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/views/card_renderer/card_view_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    function CardViewComponent_image_card_1_Template(rf, ctx) { if (rf & 1) {
        const _r5 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "image-card", 4);
        ɵɵlistener("fullWidthChanged", function CardViewComponent_image_card_1_Template_image_card_fullWidthChanged_0_listener($event) { ɵɵrestoreView(_r5); const ctx_r4 = ɵɵnextContext(); return ctx_r4.onFullWidthChanged($event); })("pinStateChanged", function CardViewComponent_image_card_1_Template_image_card_pinStateChanged_0_listener() { ɵɵrestoreView(_r5); const ctx_r6 = ɵɵnextContext(); return ctx_r6.onPinStateChanged(); });
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r0 = ɵɵnextContext();
        ɵɵproperty("cardId", ctx_r0.cardId)("groupName", ctx_r0.groupName)("runColorScale", ctx_r0.runColorScale);
    } }
    function CardViewComponent_scalar_card_2_Template(rf, ctx) { if (rf & 1) {
        const _r8 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "scalar-card", 5);
        ɵɵlistener("fullWidthChanged", function CardViewComponent_scalar_card_2_Template_scalar_card_fullWidthChanged_0_listener($event) { ɵɵrestoreView(_r8); const ctx_r7 = ɵɵnextContext(); return ctx_r7.onFullWidthChanged($event); })("fullHeightChanged", function CardViewComponent_scalar_card_2_Template_scalar_card_fullHeightChanged_0_listener($event) { ɵɵrestoreView(_r8); const ctx_r9 = ɵɵnextContext(); return ctx_r9.onFullHeightChanged($event); })("pinStateChanged", function CardViewComponent_scalar_card_2_Template_scalar_card_pinStateChanged_0_listener() { ɵɵrestoreView(_r8); const ctx_r10 = ɵɵnextContext(); return ctx_r10.onPinStateChanged(); });
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r1 = ɵɵnextContext();
        ɵɵproperty("cardId", ctx_r1.cardId)("groupName", ctx_r1.groupName)("runColorScale", ctx_r1.runColorScale);
    } }
    function CardViewComponent_histogram_card_3_Template(rf, ctx) { if (rf & 1) {
        const _r12 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "histogram-card", 5);
        ɵɵlistener("fullWidthChanged", function CardViewComponent_histogram_card_3_Template_histogram_card_fullWidthChanged_0_listener($event) { ɵɵrestoreView(_r12); const ctx_r11 = ɵɵnextContext(); return ctx_r11.onFullWidthChanged($event); })("fullHeightChanged", function CardViewComponent_histogram_card_3_Template_histogram_card_fullHeightChanged_0_listener($event) { ɵɵrestoreView(_r12); const ctx_r13 = ɵɵnextContext(); return ctx_r13.onFullHeightChanged($event); })("pinStateChanged", function CardViewComponent_histogram_card_3_Template_histogram_card_pinStateChanged_0_listener() { ɵɵrestoreView(_r12); const ctx_r14 = ɵɵnextContext(); return ctx_r14.onPinStateChanged(); });
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r2 = ɵɵnextContext();
        ɵɵproperty("cardId", ctx_r2.cardId)("groupName", ctx_r2.groupName)("runColorScale", ctx_r2.runColorScale);
    } }
    function CardViewComponent_div_4_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div");
        ɵɵtext(1);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r3 = ɵɵnextContext();
        ɵɵadvance(1);
        ɵɵtextInterpolate1("Placeholder error for: ", ctx_r3.cardId, "");
    } }
    class CardViewComponent {
        constructor() {
            this.PluginType = PluginType;
            this.fullWidthChanged = new EventEmitter();
            this.fullHeightChanged = new EventEmitter();
            this.pinStateChanged = new EventEmitter();
        }
        /**
         * @param {?} showFullWidth
         * @return {?}
         */
        onFullWidthChanged(showFullWidth) {
            this.fullWidthChanged.emit(showFullWidth);
        }
        /**
         * @param {?} showFullHeight
         * @return {?}
         */
        onFullHeightChanged(showFullHeight) {
            this.fullHeightChanged.emit(showFullHeight);
        }
        /**
         * @return {?}
         */
        onPinStateChanged() {
            this.pinStateChanged.emit();
        }
    }
    CardViewComponent.decorators = [
        { type: Component, args: [{
                    selector: 'card-view-component',
                    templateUrl: 'card_view_component.ng.html',
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    CardViewComponent.propDecorators = {
        cardId: [{ type: Input }],
        groupName: [{ type: Input }],
        pluginType: [{ type: Input }],
        runColorScale: [{ type: Input }],
        fullWidthChanged: [{ type: Output }],
        fullHeightChanged: [{ type: Output }],
        pinStateChanged: [{ type: Output }]
    };
    /** @nocollapse */ CardViewComponent.ɵfac = function CardViewComponent_Factory(t) { return new (t || CardViewComponent)(); };
    /** @nocollapse */ CardViewComponent.ɵcmp = ɵɵdefineComponent({ type: CardViewComponent, selectors: [["card-view-component"]], inputs: { cardId: "cardId", groupName: "groupName", pluginType: "pluginType", runColorScale: "runColorScale" }, outputs: { fullWidthChanged: "fullWidthChanged", fullHeightChanged: "fullHeightChanged", pinStateChanged: "pinStateChanged" }, decls: 5, vars: 4, consts: [[3, "ngSwitch"], [3, "cardId", "groupName", "runColorScale", "fullWidthChanged", "pinStateChanged", 4, "ngSwitchCase"], [3, "cardId", "groupName", "runColorScale", "fullWidthChanged", "fullHeightChanged", "pinStateChanged", 4, "ngSwitchCase"], [4, "ngSwitchDefault"], [3, "cardId", "groupName", "runColorScale", "fullWidthChanged", "pinStateChanged"], [3, "cardId", "groupName", "runColorScale", "fullWidthChanged", "fullHeightChanged", "pinStateChanged"]], template: function CardViewComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementContainerStart(0, 0);
            ɵɵtemplate(1, CardViewComponent_image_card_1_Template, 1, 3, "image-card", 1);
            ɵɵtemplate(2, CardViewComponent_scalar_card_2_Template, 1, 3, "scalar-card", 2);
            ɵɵtemplate(3, CardViewComponent_histogram_card_3_Template, 1, 3, "histogram-card", 2);
            ɵɵtemplate(4, CardViewComponent_div_4_Template, 2, 1, "div", 3);
            ɵɵelementContainerEnd();
        } if (rf & 2) {
            ɵɵproperty("ngSwitch", ctx.pluginType);
            ɵɵadvance(1);
            ɵɵproperty("ngSwitchCase", ctx.PluginType.IMAGES);
            ɵɵadvance(1);
            ɵɵproperty("ngSwitchCase", ctx.PluginType.SCALARS);
            ɵɵadvance(1);
            ɵɵproperty("ngSwitchCase", ctx.PluginType.HISTOGRAMS);
        } }, directives: [NgSwitch, NgSwitchCase, NgSwitchDefault, ImageCardContainer, ScalarCardContainer, HistogramCardContainer], encapsulation: 2, changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/views/card_renderer/card_view_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    // Since vz-line-chart only updates every 350ms, it does not make sense to
    // update the Polymer component more frequently.
    // [1]:
    // tensorboard/components/vz_line_chart2/vz-line-chart2.js?l=343-360
    /** @type {?} */
    const RUN_COLOR_UPDATE_THROTTLE_TIME_IN_MS = 350;
    class CardViewContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.showFullWidth = false;
            this.showFullHeight = false;
            this.runColorScale$ = this.store
                .select(getRunColorMap)
                .pipe(throttleTime(RUN_COLOR_UPDATE_THROTTLE_TIME_IN_MS, undefined, {
                leading: true,
                trailing: true,
            }), map((/**
             * @param {?} colorMap
             * @return {?}
             */
            (colorMap) => {
                return (/**
                 * @param {?} runId
                 * @return {?}
                 */
                (runId) => {
                    if (!colorMap.hasOwnProperty(runId)) {
                        throw new Error(`[Color scale] unknown runId: ${runId}.`);
                    }
                    return colorMap[runId];
                });
            })));
        }
        /**
         * @param {?} showFullWidth
         * @return {?}
         */
        onFullWidthChanged(showFullWidth) {
            this.showFullWidth = showFullWidth;
        }
        /**
         * @param {?} showFullHeight
         * @return {?}
         */
        onFullHeightChanged(showFullHeight) {
            this.showFullHeight = showFullHeight;
        }
        /**
         * @return {?}
         */
        onPinStateChanged() {
            this.store
                .select(getCardPinnedState, this.cardId)
                .pipe(take(1), withLatestFrom(this.store.select(getCanCreateNewPins)))
                .subscribe((/**
             * @param {?} __0
             * @return {?}
             */
            ([wasPinned, canCreateNewPins]) => {
                this.store.dispatch(cardPinStateToggled({
                    cardId: this.cardId,
                    canCreateNewPins,
                    wasPinned,
                }));
            }));
        }
    }
    CardViewContainer.decorators = [
        { type: Component, args: [{
                    selector: 'card-view',
                    template: `
    <card-view-component
      [cardId]="cardId"
      [groupName]="groupName"
      [pluginType]="pluginType"
      [runColorScale]="runColorScale$ | async"
      (fullWidthChanged)="onFullWidthChanged($event)"
      (fullHeightChanged)="onFullHeightChanged($event)"
      (pinStateChanged)="onPinStateChanged()"
    >
    </card-view-component>
  `,
                    styleUrls: ['card_view_container.css'],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    CardViewContainer.ctorParameters = () => [
        { type: Store }
    ];
    CardViewContainer.propDecorators = {
        cardId: [{ type: Input }],
        groupName: [{ type: Input }],
        pluginType: [{ type: Input }],
        showFullWidth: [{ type: HostBinding, args: ['class.full-width',] }],
        showFullHeight: [{ type: HostBinding, args: ['class.full-height',] }]
    };
    /** @nocollapse */ CardViewContainer.ɵfac = function CardViewContainer_Factory(t) { return new (t || CardViewContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ CardViewContainer.ɵcmp = ɵɵdefineComponent({ type: CardViewContainer, selectors: [["card-view"]], hostVars: 4, hostBindings: function CardViewContainer_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵclassProp("full-width", ctx.showFullWidth)("full-height", ctx.showFullHeight);
        } }, inputs: { cardId: "cardId", groupName: "groupName", pluginType: "pluginType" }, decls: 2, vars: 6, consts: [[3, "cardId", "groupName", "pluginType", "runColorScale", "fullWidthChanged", "fullHeightChanged", "pinStateChanged"]], template: function CardViewContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "card-view-component", 0);
            ɵɵlistener("fullWidthChanged", function CardViewContainer_Template_card_view_component_fullWidthChanged_0_listener($event) { return ctx.onFullWidthChanged($event); })("fullHeightChanged", function CardViewContainer_Template_card_view_component_fullHeightChanged_0_listener($event) { return ctx.onFullHeightChanged($event); })("pinStateChanged", function CardViewContainer_Template_card_view_component_pinStateChanged_0_listener() { return ctx.onPinStateChanged(); });
            ɵɵpipe(1, "async");
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵproperty("cardId", ctx.cardId)("groupName", ctx.groupName)("pluginType", ctx.pluginType)("runColorScale", ɵɵpipeBind1(1, 4, ctx.runColorScale$));
        } }, directives: [CardViewComponent], pipes: [AsyncPipe], styles: [".mat-badge-content[_ngcontent-%COMP%]{font-weight:600;font-size:12px;font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:9px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:24px}.mat-h1[_ngcontent-%COMP%], .mat-headline[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%]{font:400 24px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h2[_ngcontent-%COMP%], .mat-title[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h3[_ngcontent-%COMP%], .mat-subheading-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%]{font:400 16px/28px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h4[_ngcontent-%COMP%], .mat-subheading-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%]{font:400 15px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h5[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%]{font:400 calc(14px * 0.83)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-h6[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:400 calc(14px * 0.67)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-body-strong[_ngcontent-%COMP%], .mat-body-2[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   p[_ngcontent-%COMP%]{margin:0 0 12px}.mat-small[_ngcontent-%COMP%], .mat-caption[_ngcontent-%COMP%]{font:400 12px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-display-4[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-4[_ngcontent-%COMP%]{font:300 112px/112px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.05em;margin:0 0 56px}.mat-display-3[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-3[_ngcontent-%COMP%]{font:400 56px/56px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.02em;margin:0 0 64px}.mat-display-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-2[_ngcontent-%COMP%]{font:400 45px/48px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.005em;margin:0 0 64px}.mat-display-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-1[_ngcontent-%COMP%]{font:400 34px/40px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 64px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%], .mat-flat-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-button-toggle[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card-title[_ngcontent-%COMP%]{font-size:24px;font-weight:500}.mat-card-header[_ngcontent-%COMP%]   .mat-card-title[_ngcontent-%COMP%]{font-size:20px}.mat-card-subtitle[_ngcontent-%COMP%], .mat-card-content[_ngcontent-%COMP%]{font-size:14px}.mat-checkbox[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-checkbox-layout[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{line-height:24px}.mat-chip[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-chip[_ngcontent-%COMP%]   .mat-chip-trailing-icon.mat-icon[_ngcontent-%COMP%], .mat-chip[_ngcontent-%COMP%]   .mat-chip-remove.mat-icon[_ngcontent-%COMP%]{font-size:18px}.mat-table[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-header-cell[_ngcontent-%COMP%]{font-size:12px;font-weight:500}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{font-size:14px}.mat-calendar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-calendar-body[_ngcontent-%COMP%]{font-size:13px}.mat-calendar-body-label[_ngcontent-%COMP%], .mat-calendar-period-button[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-calendar-table-header[_ngcontent-%COMP%]   th[_ngcontent-%COMP%]{font-size:11px;font-weight:400}.mat-dialog-title[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-expansion-panel-header[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:15px;font-weight:400}.mat-expansion-panel-content[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field[_ngcontent-%COMP%]{font-size:inherit;font-weight:400;line-height:1.125;font-family:Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.34375em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{font-size:150%;line-height:1.125}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]{height:1.5em;width:1.5em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{height:1.125em;line-height:1.125}.mat-form-field-infix[_ngcontent-%COMP%]{padding:.5em 0;border-top:.84375em solid transparent}.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34375em) scale(0.75);width:133.3333333333%}.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34374em) scale(0.75);width:133.3333433333%}.mat-form-field-label-wrapper[_ngcontent-%COMP%]{top:-0.84375em;padding-top:.84375em}.mat-form-field-label[_ngcontent-%COMP%]{top:1.34375em}.mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.34375em}.mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{font-size:75%;margin-top:.6666666667em;top:calc(100% - 1.7916666667em)}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.4375em 0}.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.001px);-ms-transform:translateY(-1.28125em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00101px);-ms-transform:translateY(-1.28124em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00102px);-ms-transform:translateY(-1.28123em) scale(0.75);width:133.3333533333%}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.28125em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{margin-top:.5416666667em;top:calc(100% - 1.6666666667em)}@media print{.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28122em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28121em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.2812em) scale(0.75)}}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.25em 0 .75em 0}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.09375em;margin-top:-0.5em}.mat-form-field-appearance-fill.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59374em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:1em 0 1em 0}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.84375em;margin-top:-0.25em}.mat-form-field-appearance-outline.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59374em) scale(0.75);width:133.3333433333%}.mat-grid-tile-header[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]{font-size:14px}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2), .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}input.mat-input-element[_ngcontent-%COMP%]{margin-top:-0.0625em}.mat-menu-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:400}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px}.mat-radio-button[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select-trigger[_ngcontent-%COMP%]{height:1.125em}.mat-slide-toggle-content[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-slider-thumb-label-text[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-stepper-vertical[_ngcontent-%COMP%], .mat-stepper-horizontal[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-step-label[_ngcontent-%COMP%]{font-size:14px;font-weight:400}.mat-step-sub-label-error[_ngcontent-%COMP%]{font-weight:normal}.mat-step-label-error[_ngcontent-%COMP%]{font-size:14px}.mat-step-label-selected[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-tab-group[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-toolbar[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0}.mat-tooltip[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:10px;padding-top:6px;padding-bottom:6px}.mat-tooltip-handset[_ngcontent-%COMP%]{font-size:14px;padding-top:8px;padding-bottom:8px}.mat-list-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:16px}.mat-optgroup-label[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-simple-snackbar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px}.mat-simple-snackbar-action[_ngcontent-%COMP%]{line-height:1;font-family:inherit;font-size:inherit;font-weight:500}.mat-tree[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{font-weight:400;font-size:14px}.mat-ripple[_ngcontent-%COMP%]{overflow:hidden;position:relative}.mat-ripple[_ngcontent-%COMP%]:not(:empty){transform:translateZ(0)}.mat-ripple.mat-ripple-unbounded[_ngcontent-%COMP%]{overflow:visible}.mat-ripple-element[_ngcontent-%COMP%]{position:absolute;border-radius:50%;pointer-events:none;transition:opacity,transform 0ms cubic-bezier(0, 0, 0.2, 1);transform:scale(0)}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{display:none}.cdk-visually-hidden[_ngcontent-%COMP%]{border:0;clip:rect(0 0 0 0);height:1px;margin:-1px;overflow:hidden;padding:0;position:absolute;width:1px;outline:0;-webkit-appearance:none;-moz-appearance:none}.cdk-overlay-container[_ngcontent-%COMP%], .cdk-global-overlay-wrapper[_ngcontent-%COMP%]{pointer-events:none;top:0;left:0;height:100%;width:100%}.cdk-overlay-container[_ngcontent-%COMP%]{position:fixed;z-index:1000}.cdk-overlay-container[_ngcontent-%COMP%]:empty{display:none}.cdk-global-overlay-wrapper[_ngcontent-%COMP%]{display:flex;position:absolute;z-index:1000}.cdk-overlay-pane[_ngcontent-%COMP%]{position:absolute;pointer-events:auto;box-sizing:border-box;z-index:1000;display:flex;max-width:100%;max-height:100%}.cdk-overlay-backdrop[_ngcontent-%COMP%]{position:absolute;top:0;bottom:0;left:0;right:0;z-index:1000;pointer-events:auto;-webkit-tap-highlight-color:transparent;transition:opacity 400ms cubic-bezier(0.25, 0.8, 0.25, 1);opacity:0}.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:1}@media screen and (-ms-high-contrast: active){.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:.6}}.cdk-overlay-dark-backdrop[_ngcontent-%COMP%]{background:rgba(0,0,0,.32)}.cdk-overlay-transparent-backdrop[_ngcontent-%COMP%], .cdk-overlay-transparent-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:0}.cdk-overlay-connected-position-bounding-box[_ngcontent-%COMP%]{position:absolute;z-index:1000;display:flex;flex-direction:column;min-width:1px;min-height:1px}.cdk-global-scrollblock[_ngcontent-%COMP%]{position:fixed;width:100%;overflow-y:scroll}@keyframes cdk-text-field-autofill-start{}@keyframes cdk-text-field-autofill-end{}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:-webkit-autofill{animation:cdk-text-field-autofill-start 0s 1ms}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:not(:-webkit-autofill){animation:cdk-text-field-autofill-end 0s 1ms}textarea.cdk-textarea-autosize[_ngcontent-%COMP%]{resize:none}textarea.cdk-textarea-autosize-measuring[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:auto !important;overflow:hidden !important}textarea.cdk-textarea-autosize-measuring-firefox[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:0 !important}.mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.1)}.mat-option[_ngcontent-%COMP%]{color:#212121}.mat-option[_ngcontent-%COMP%]:hover:not(.mat-option-disabled), .mat-option[_ngcontent-%COMP%]:focus:not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple):not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-active[_ngcontent-%COMP%]{background:rgba(0,0,0,.04);color:#212121}.mat-option.mat-option-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-primary[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f57c00}.mat-accent[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f44336}.mat-optgroup-label[_ngcontent-%COMP%]{color:#616161}.mat-optgroup-disabled[_ngcontent-%COMP%]   .mat-optgroup-label[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-pseudo-checkbox[_ngcontent-%COMP%]{color:#616161}.mat-pseudo-checkbox[_ngcontent-%COMP%]::after{color:#fafafa}.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{color:#b0b0b0}.mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f57c00}.mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f44336}.mat-pseudo-checkbox-checked.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{background:#b0b0b0}.mat-elevation-z0[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z1[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z2[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z3[_ngcontent-%COMP%]{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z4[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z5[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 5px 8px 0px rgba(0, 0, 0, 0.14),0px 1px 14px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z6[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z7[_ngcontent-%COMP%]{box-shadow:0px 4px 5px -2px rgba(0, 0, 0, 0.2),0px 7px 10px 1px rgba(0, 0, 0, 0.14),0px 2px 16px 1px rgba(0, 0, 0, 0.12)}.mat-elevation-z8[_ngcontent-%COMP%]{box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z9[_ngcontent-%COMP%]{box-shadow:0px 5px 6px -3px rgba(0, 0, 0, 0.2),0px 9px 12px 1px rgba(0, 0, 0, 0.14),0px 3px 16px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z10[_ngcontent-%COMP%]{box-shadow:0px 6px 6px -3px rgba(0, 0, 0, 0.2),0px 10px 14px 1px rgba(0, 0, 0, 0.14),0px 4px 18px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z11[_ngcontent-%COMP%]{box-shadow:0px 6px 7px -4px rgba(0, 0, 0, 0.2),0px 11px 15px 1px rgba(0, 0, 0, 0.14),0px 4px 20px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z12[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z13[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 13px 19px 2px rgba(0, 0, 0, 0.14),0px 5px 24px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z14[_ngcontent-%COMP%]{box-shadow:0px 7px 9px -4px rgba(0, 0, 0, 0.2),0px 14px 21px 2px rgba(0, 0, 0, 0.14),0px 5px 26px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z15[_ngcontent-%COMP%]{box-shadow:0px 8px 9px -5px rgba(0, 0, 0, 0.2),0px 15px 22px 2px rgba(0, 0, 0, 0.14),0px 6px 28px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z16[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z17[_ngcontent-%COMP%]{box-shadow:0px 8px 11px -5px rgba(0, 0, 0, 0.2),0px 17px 26px 2px rgba(0, 0, 0, 0.14),0px 6px 32px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z18[_ngcontent-%COMP%]{box-shadow:0px 9px 11px -5px rgba(0, 0, 0, 0.2),0px 18px 28px 2px rgba(0, 0, 0, 0.14),0px 7px 34px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z19[_ngcontent-%COMP%]{box-shadow:0px 9px 12px -6px rgba(0, 0, 0, 0.2),0px 19px 29px 2px rgba(0, 0, 0, 0.14),0px 7px 36px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z20[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 20px 31px 3px rgba(0, 0, 0, 0.14),0px 8px 38px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z21[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 21px 33px 3px rgba(0, 0, 0, 0.14),0px 8px 40px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z22[_ngcontent-%COMP%]{box-shadow:0px 10px 14px -6px rgba(0, 0, 0, 0.2),0px 22px 35px 3px rgba(0, 0, 0, 0.14),0px 8px 42px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z23[_ngcontent-%COMP%]{box-shadow:0px 11px 14px -7px rgba(0, 0, 0, 0.2),0px 23px 36px 3px rgba(0, 0, 0, 0.14),0px 9px 44px 8px rgba(0, 0, 0, 0.12)}.mat-elevation-z24[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12)}.mat-app-background[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-theme-loaded-marker[_ngcontent-%COMP%]{display:none}.mat-autocomplete-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-autocomplete-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover){background:#fff}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover):not(.mat-option-disabled){color:#212121}.mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f57c00}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{outline:solid 1px;border-radius:0}.mat-badge-accent[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-badge-warn[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f44336}.mat-badge[_ngcontent-%COMP%]{position:relative}.mat-badge-hidden[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{display:none}.mat-badge-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#b9b9b9;color:#757575}.mat-badge-content[_ngcontent-%COMP%]{position:absolute;text-align:center;display:inline-block;border-radius:50%;transition:transform 200ms ease-in-out;transform:scale(0.6);overflow:hidden;white-space:nowrap;text-overflow:ellipsis;pointer-events:none}.ng-animate-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%], .mat-badge-content._mat-animation-noopable[_ngcontent-%COMP%]{transition:none}.mat-badge-content.mat-badge-active[_ngcontent-%COMP%]{transform:none}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:16px;height:16px;line-height:16px}.mat-badge-small.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-8px}.mat-badge-small.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-8px}.mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-16px}.mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-16px}.mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-8px}.mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-8px}.mat-badge-medium[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:22px;height:22px;line-height:22px}.mat-badge-medium.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-11px}.mat-badge-medium.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-11px}.mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-22px}.mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-22px}.mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-11px}.mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-11px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:28px;height:28px;line-height:28px}.mat-badge-large.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-14px}.mat-badge-large.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-14px}.mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-28px}.mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-28px}.mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-14px}.mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-14px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]{color:inherit;background:transparent}.mat-button.mat-primary[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-button[disabled][disabled][_ngcontent-%COMP%], .mat-icon-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-icon-button[disabled][disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-stroked-button[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f44336}.mat-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:transparent}.mat-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{opacity:.1;background-color:currentColor}.mat-button-focus-overlay[_ngcontent-%COMP%]{background:#000}.mat-stroked-button[_ngcontent-%COMP%]:not([disabled]){border-color:rgba(0,0,0,.12)}.mat-flat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{color:#212121;background-color:#fff}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-flat-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-stroked-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-flat-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-raised-button[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone[_ngcontent-%COMP%], .mat-button-toggle-group[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{box-shadow:none}.mat-button-toggle[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-button-toggle[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121;background:#fff}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:#000}.mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12)}[dir=rtl][_ngcontent-%COMP%]   .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-group-appearance-standard.mat-button-toggle-vertical[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:none;border-top:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#616161}.mat-button-toggle-checked.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121}.mat-button-toggle-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.26);background-color:#eee}.mat-button-toggle-disabled.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{background:#fff}.mat-button-toggle-disabled.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#bdbdbd}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{border:solid 1px rgba(0,0,0,.12)}.mat-card[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-card[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-card.mat-card-flat[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-card-subtitle[_ngcontent-%COMP%]{color:#616161}.mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#616161}.mat-checkbox-checkmark[_ngcontent-%COMP%]{fill:#fafafa}.mat-checkbox-checkmark-path[_ngcontent-%COMP%]{stroke:#fafafa !important}.mat-checkbox-mixedmark[_ngcontent-%COMP%]{background-color:#fafafa}.mat-checkbox-indeterminate.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f57c00}.mat-checkbox-indeterminate.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#ff9800}.mat-checkbox-indeterminate.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f44336}.mat-checkbox-disabled.mat-checkbox-checked[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-disabled.mat-checkbox-indeterminate[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]:not(.mat-checkbox-checked)   .mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{color:#616161}.mat-checkbox[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%]{background:#f57c00}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%]{background:#ff9800}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%]{background:#f44336}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#212121}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#212121;opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled):active{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled)   .mat-chip-remove[_ngcontent-%COMP%]:hover{opacity:.54}.mat-chip.mat-standard-chip.mat-chip-disabled[_ngcontent-%COMP%]{opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]::after{background:#000}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-table[_ngcontent-%COMP%]{background:#fff}.mat-table[_ngcontent-%COMP%]   thead[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tbody[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tfoot[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], [mat-header-row][_ngcontent-%COMP%], [mat-row][_ngcontent-%COMP%], [mat-footer-row][_ngcontent-%COMP%], .mat-table-sticky[_ngcontent-%COMP%]{background:inherit}mat-row[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], th.mat-header-cell[_ngcontent-%COMP%], td.mat-cell[_ngcontent-%COMP%], td.mat-footer-cell[_ngcontent-%COMP%]{border-bottom-color:rgba(0,0,0,.12)}.mat-header-cell[_ngcontent-%COMP%]{color:#616161}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{color:#212121}.mat-calendar-arrow[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.54)}.mat-datepicker-toggle[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-next-button[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-previous-button[_ngcontent-%COMP%]{color:rgba(0,0,0,.54)}.mat-calendar-table-header[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-calendar-table-header-divider[_ngcontent-%COMP%]::after{background:rgba(0,0,0,.12)}.mat-calendar-body-label[_ngcontent-%COMP%]{color:#616161}.mat-calendar-body-cell-content[_ngcontent-%COMP%]{color:#212121;border-color:transparent}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){color:#757575}.mat-calendar-body-cell[_ngcontent-%COMP%]:not(.mat-calendar-body-disabled):hover > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-keyboard-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-program-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){background-color:rgba(0,0,0,.04)}.mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.38)}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.18)}.mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.4)}.mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12);background-color:#fff;color:#212121}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.4)}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.4)}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content-touch[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-datepicker-toggle-active[_ngcontent-%COMP%]{color:#f57c00}.mat-datepicker-toggle-active.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-datepicker-toggle-active.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-dialog-container[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-divider[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-divider-vertical[_ngcontent-%COMP%]{border-right-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-expansion-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-action-row[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-keyboard-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-program-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded)   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover:not([aria-disabled=true]){background:rgba(0,0,0,.04)}@media(hover: none){.mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded):not([aria-disabled=true])   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover{background:#fff}}.mat-expansion-panel-header-title[_ngcontent-%COMP%]{color:#212121}.mat-expansion-panel-header-description[_ngcontent-%COMP%], .mat-expansion-indicator[_ngcontent-%COMP%]::after{color:#616161}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-title[_ngcontent-%COMP%], .mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-description[_ngcontent-%COMP%]{color:inherit}.mat-form-field-label[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-hint[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-focused[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-ripple[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid)   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f57c00}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-accent   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#ff9800}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-warn   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#f44336}.mat-error[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-hint[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-legacy.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-standard[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-standard.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.04)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.02)}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:transparent}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.12)}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:rgba(0,0,0,.87)}.mat-form-field-appearance-outline.mat-focused[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field-appearance-outline.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-appearance-outline.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-invalid.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.06)}.mat-icon.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-icon.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-icon.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-type-mat-native-select[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#616161}.mat-input-element[_ngcontent-%COMP%]:disabled, .mat-form-field-type-mat-native-select.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#757575}.mat-input-element[_ngcontent-%COMP%]{caret-color:#f57c00}.mat-input-element[_ngcontent-%COMP%]::placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-moz-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-webkit-input-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]:-ms-input-placeholder{color:rgba(97,97,97,.42)}.mat-accent[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%], .mat-form-field-invalid[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#f44336}.mat-form-field-type-mat-native-select.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{color:#616161}.mat-list-item-disabled[_ngcontent-%COMP%]{background-color:#eee}.mat-list-option[_ngcontent-%COMP%]:hover, .mat-list-option[_ngcontent-%COMP%]:focus, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.04)}.mat-list-single-selected-option[_ngcontent-%COMP%], .mat-list-single-selected-option[_ngcontent-%COMP%]:hover, .mat-list-single-selected-option[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.12)}.mat-menu-panel[_ngcontent-%COMP%]{background:#fff}.mat-menu-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-menu-item[_ngcontent-%COMP%]{background:transparent;color:#212121}.mat-menu-item[disabled][_ngcontent-%COMP%], .mat-menu-item[disabled][_ngcontent-%COMP%]::after{color:rgba(0,0,0,.38)}.mat-menu-item[_ngcontent-%COMP%]   .mat-icon-no-color[_ngcontent-%COMP%], .mat-menu-item-submenu-trigger[_ngcontent-%COMP%]::after{color:rgba(0,0,0,.54)}.mat-menu-item[_ngcontent-%COMP%]:hover:not([disabled]), .mat-menu-item.cdk-program-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item.cdk-keyboard-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item-highlighted[_ngcontent-%COMP%]:not([disabled]){background:rgba(0,0,0,.04)}.mat-paginator[_ngcontent-%COMP%]{background:#fff}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{color:#616161}.mat-paginator-decrement[_ngcontent-%COMP%], .mat-paginator-increment[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54);border-right:2px solid rgba(0,0,0,.54)}.mat-paginator-first[_ngcontent-%COMP%], .mat-paginator-last[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54)}.mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-decrement[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-increment[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-first[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-last[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f57c00}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#ff9800}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f44336}.mat-progress-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f57c00}.mat-progress-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#ff9800}.mat-progress-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f44336}.mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#616161}.mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f57c00}.mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#ff9800}.mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#ff9800}.mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f44336}.mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f44336}.mat-radio-button.mat-radio-disabled.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-label-content[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-radio-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-select-value[_ngcontent-%COMP%]{color:#212121}.mat-select-placeholder[_ngcontent-%COMP%]{color:rgba(97,97,97,.42)}.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%]{color:#757575}.mat-select-arrow[_ngcontent-%COMP%]{color:#616161}.mat-select-panel[_ngcontent-%COMP%]{background:#fff}.mat-select-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-select-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple){background:rgba(0,0,0,.12)}.mat-form-field.mat-focused.mat-primary[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-invalid[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#757575}.mat-drawer-container[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-drawer[_ngcontent-%COMP%]{background-color:#fff;color:#212121}.mat-drawer.mat-drawer-push[_ngcontent-%COMP%]{background-color:#fff}.mat-drawer[_ngcontent-%COMP%]:not(.mat-drawer-side){box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-drawer-side[_ngcontent-%COMP%]{border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-backdrop.mat-drawer-shown[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.6)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.54)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.54)}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.54)}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle[_ngcontent-%COMP%]:not(.mat-checked)   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-slide-toggle-thumb[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12);background-color:#fafafa}.mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-primary[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f57c00}.mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-accent[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#ff9800}.mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-warn[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f44336}.mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.2)}.mat-slider[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%], .cdk-focused[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-disabled[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]   .mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing)   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26);background-color:transparent}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26)}.mat-slider-has-ticks[_ngcontent-%COMP%]   .mat-slider-wrapper[_ngcontent-%COMP%]::after{border-color:rgba(0,0,0,.7)}.mat-slider-horizontal[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to right, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent);background-image:-moz-repeating-linear-gradient(0.0001deg, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-slider-vertical[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to bottom, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-step-header.cdk-keyboard-focused[_ngcontent-%COMP%], .mat-step-header.cdk-program-focused[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]:hover{background-color:rgba(0,0,0,.04)}@media(hover: none){.mat-step-header[_ngcontent-%COMP%]:hover{background:none}}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-optional[_ngcontent-%COMP%]{color:#616161}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon[_ngcontent-%COMP%]{background-color:#616161;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-selected[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-done[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-edit[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-error[_ngcontent-%COMP%]{background-color:transparent;color:#f44336}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-active[_ngcontent-%COMP%]{color:#212121}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-error[_ngcontent-%COMP%]{color:#f44336}.mat-stepper-horizontal[_ngcontent-%COMP%], .mat-stepper-vertical[_ngcontent-%COMP%]{background-color:#fff}.mat-stepper-vertical-line[_ngcontent-%COMP%]::before{border-left-color:rgba(0,0,0,.12)}.mat-horizontal-stepper-header[_ngcontent-%COMP%]::before, .mat-horizontal-stepper-header[_ngcontent-%COMP%]::after, .mat-stepper-horizontal-line[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-sort-header-arrow[_ngcontent-%COMP%]{color:#616161}.mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-header[_ngcontent-%COMP%]{border-bottom:1px solid rgba(0,0,0,.12)}.mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%]{border-top:1px solid rgba(0,0,0,.12);border-bottom:none}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{color:#212121}.mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:#757575}.mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#212121}.mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#757575}.mat-tab-group[class*=mat-background-][_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar[class*=mat-background-][_ngcontent-%COMP%]{border-bottom:none;border-top:none}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-toolbar[_ngcontent-%COMP%]{background:#f5f5f5;color:#212121}.mat-toolbar.mat-primary[_ngcontent-%COMP%]{background:#f57c00;color:#fff}.mat-toolbar.mat-accent[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-toolbar.mat-warn[_ngcontent-%COMP%]{background:#f44336;color:#fff}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:currentColor}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:inherit}.mat-toolbar[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:currentColor}.mat-tooltip[_ngcontent-%COMP%]{background:rgba(97,97,97,.9)}.mat-tree[_ngcontent-%COMP%]{background:#fff}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{color:#212121}.mat-snack-bar-container[_ngcontent-%COMP%]{color:rgba(255,255,255,.7);background:#323232;box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-simple-snackbar-action[_ngcontent-%COMP%]{color:#ff9800}[_nghost-%COMP%]{background-color:#fff}"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/views/main_view/card_grid_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    function CardGridComponent_ng_container_0_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementContainer(0);
    } }
    function CardGridComponent_card_view_2_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "card-view", 4);
    } if (rf & 2) {
        const item_r5 = ctx.$implicit;
        const ctx_r1 = ɵɵnextContext();
        ɵɵproperty("cardId", item_r5.cardId)("groupName", ctx_r1.groupName)("pluginType", item_r5.plugin)("cardObserver", ctx_r1.cardObserver)("cardLazyLoader", item_r5.cardId);
    } }
    function CardGridComponent_ng_container_3_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementContainer(0);
    } }
    var I18N_0$5;
    if (typeof ngI18nClosureMode !== "undefined" && ngI18nClosureMode) {
        /**
         * @desc A button that sets a group to the previous page.
         */ 
        const MSG_EXTERNAL_3629960544875360046$$TENSORBOARD_WEBAPP_METRICS_VIEWS_MAIN_VIEW_CARD_GRID_COMPONENT_TS____1 = goog.getMsg("Previous page");
        I18N_0$5 = MSG_EXTERNAL_3629960544875360046$$TENSORBOARD_WEBAPP_METRICS_VIEWS_MAIN_VIEW_CARD_GRID_COMPONENT_TS____1;
    }
    else {
        I18N_0$5 = $localize `:A button that sets a group to the previous page.␟575e782fd27f2ee70a034a775efe9ad162472250␟3629960544875360046:Previous page`;
    }
    const _c2$b = ["aria-label", I18N_0$5];
    function CardGridComponent_ng_template_4_div_0_button_2_Template(rf, ctx) { if (rf & 1) {
        const _r12 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "button", 13);
        ɵɵi18nAttributes(1, _c2$b);
        ɵɵlistener("click", function CardGridComponent_ng_template_4_div_0_button_2_Template_button_click_0_listener() { ɵɵrestoreView(_r12); const ctx_r11 = ɵɵnextContext(3); return ctx_r11.pageIndexChanged.emit(ctx_r11.pageIndex - 1); });
        ɵɵtext(2, " Previous ");
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r8 = ɵɵnextContext(3);
        ɵɵproperty("disabled", ctx_r8.pageIndex === 0);
    } }
    var I18N_3$3;
    if (typeof ngI18nClosureMode !== "undefined" && ngI18nClosureMode) {
        /**
         * @desc A button that allows user to expand a tag group.
         */ 
        const MSG_EXTERNAL_5386054325274779258$$TENSORBOARD_WEBAPP_METRICS_VIEWS_MAIN_VIEW_CARD_GRID_COMPONENT_TS____4 = goog.getMsg("Expand group");
        I18N_3$3 = MSG_EXTERNAL_5386054325274779258$$TENSORBOARD_WEBAPP_METRICS_VIEWS_MAIN_VIEW_CARD_GRID_COMPONENT_TS____4;
    }
    else {
        I18N_3$3 = $localize `:A button that allows user to expand a tag group.␟ffaa11471b878a6dffe2e68c6f37064a9e074853␟5386054325274779258:Expand group`;
    }
    const _c5$5 = ["aria-label", I18N_3$3];
    function CardGridComponent_ng_template_4_div_0_button_4_mat_icon_2_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "mat-icon", 17);
    } }
    function CardGridComponent_ng_template_4_div_0_button_4_ng_template_3_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "mat-icon", 18);
    } }
    function CardGridComponent_ng_template_4_div_0_button_4_Template(rf, ctx) { if (rf & 1) {
        const _r17 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "button", 14);
        ɵɵi18nAttributes(1, _c5$5);
        ɵɵlistener("click", function CardGridComponent_ng_template_4_div_0_button_4_Template_button_click_0_listener() { ɵɵrestoreView(_r17); const ctx_r16 = ɵɵnextContext(3); return ctx_r16.groupExpansionToggled.emit(); });
        ɵɵtemplate(2, CardGridComponent_ng_template_4_div_0_button_4_mat_icon_2_Template, 1, 0, "mat-icon", 15);
        ɵɵtemplate(3, CardGridComponent_ng_template_4_div_0_button_4_ng_template_3_Template, 1, 0, "ng-template", null, 16, ɵɵtemplateRefExtractor);
        ɵɵelementEnd();
    } if (rf & 2) {
        const _r14 = ɵɵreference(4);
        const ctx_r9 = ɵɵnextContext(3);
        ɵɵadvance(2);
        ɵɵproperty("ngIf", ctx_r9.isGroupExpanded)("ngIfElse", _r14);
    } }
    var I18N_6$1;
    if (typeof ngI18nClosureMode !== "undefined" && ngI18nClosureMode) {
        /**
         * @desc A button that sets a group to the next page.
         */ 
        const MSG_EXTERNAL_3337301694210287595$$TENSORBOARD_WEBAPP_METRICS_VIEWS_MAIN_VIEW_CARD_GRID_COMPONENT_TS____7 = goog.getMsg("Next page");
        I18N_6$1 = MSG_EXTERNAL_3337301694210287595$$TENSORBOARD_WEBAPP_METRICS_VIEWS_MAIN_VIEW_CARD_GRID_COMPONENT_TS____7;
    }
    else {
        I18N_6$1 = $localize `:A button that sets a group to the next page.␟ce3cefb1cd0099aa5003dda16ec9eb21fd8ba789␟3337301694210287595:Next page`;
    }
    const _c8$2 = ["aria-label", I18N_6$1];
    function CardGridComponent_ng_template_4_div_0_span_6_span_1_Template(rf, ctx) { if (rf & 1) {
        const _r20 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "span", 21);
        ɵɵelementStart(1, "input", 22);
        ɵɵlistener("input", function CardGridComponent_ng_template_4_div_0_span_6_span_1_Template_input_input_1_listener($event) { ɵɵrestoreView(_r20); const ctx_r19 = ɵɵnextContext(4); return ctx_r19.onPaginationInputChange($event); })("change", function CardGridComponent_ng_template_4_div_0_span_6_span_1_Template_input_change_1_listener($event) { ɵɵrestoreView(_r20); const ctx_r21 = ɵɵnextContext(4); return ctx_r21.onPaginationInputChange($event); });
        ɵɵelementEnd();
        ɵɵtext(2);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r18 = ɵɵnextContext(4);
        ɵɵadvance(1);
        ɵɵproperty("value", ctx_r18.pageIndex + 1)("max", ctx_r18.numPages);
        ɵɵadvance(1);
        ɵɵtextInterpolate1(" of ", ctx_r18.numPages, "");
    } }
    function CardGridComponent_ng_template_4_div_0_span_6_Template(rf, ctx) { if (rf & 1) {
        const _r23 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "span");
        ɵɵtemplate(1, CardGridComponent_ng_template_4_div_0_span_6_span_1_Template, 3, 3, "span", 19);
        ɵɵelementStart(2, "button", 20);
        ɵɵi18nAttributes(3, _c8$2);
        ɵɵlistener("click", function CardGridComponent_ng_template_4_div_0_span_6_Template_button_click_2_listener() { ɵɵrestoreView(_r23); const ctx_r22 = ɵɵnextContext(3); return ctx_r22.pageIndexChanged.emit(ctx_r22.pageIndex + 1); });
        ɵɵtext(4, " Next ");
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const isBottomControl_r6 = ɵɵnextContext(2).isBottomControl;
        const ctx_r10 = ɵɵnextContext();
        ɵɵadvance(1);
        ɵɵproperty("ngIf", ctx_r10.showPaginationInput(isBottomControl_r6));
        ɵɵadvance(1);
        ɵɵproperty("disabled", ctx_r10.pageIndex + 1 >= ctx_r10.numPages);
    } }
    function CardGridComponent_ng_template_4_div_0_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 6);
        ɵɵelementStart(1, "span", 7);
        ɵɵtemplate(2, CardGridComponent_ng_template_4_div_0_button_2_Template, 3, 1, "button", 8);
        ɵɵelementEnd();
        ɵɵelementStart(3, "span", 9);
        ɵɵtemplate(4, CardGridComponent_ng_template_4_div_0_button_4_Template, 5, 2, "button", 10);
        ɵɵelementEnd();
        ɵɵelementStart(5, "span", 11);
        ɵɵtemplate(6, CardGridComponent_ng_template_4_div_0_span_6_Template, 5, 2, "span", 12);
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const isBottomControl_r6 = ɵɵnextContext().isBottomControl;
        const ctx_r7 = ɵɵnextContext();
        ɵɵadvance(2);
        ɵɵproperty("ngIf", ctx_r7.showPaginationControls);
        ɵɵadvance(2);
        ɵɵproperty("ngIf", ctx_r7.showExpand(isBottomControl_r6));
        ɵɵadvance(2);
        ɵɵproperty("ngIf", ctx_r7.showPaginationControls);
    } }
    function CardGridComponent_ng_template_4_Template(rf, ctx) { if (rf & 1) {
        ɵɵtemplate(0, CardGridComponent_ng_template_4_div_0_Template, 7, 3, "div", 5);
    } if (rf & 2) {
        const isBottomControl_r6 = ctx.isBottomControl;
        const ctx_r4 = ɵɵnextContext();
        ɵɵproperty("ngIf", ctx_r4.showPaginationControls || ctx_r4.showExpand(isBottomControl_r6));
    } }
    const _c9$1 = function () { return { isBottomControl: false }; };
    const _c10 = function () { return { isBottomControl: true }; };
    class CardGridComponent {
        constructor() {
            this.PluginType = PluginType;
            this.pageIndexChanged = new EventEmitter();
            this.groupExpansionToggled = new EventEmitter();
        }
        /**
         * @param {?} isBottomControl
         * @return {?}
         */
        showExpand(isBottomControl) {
            return isBottomControl ? this.isGroupExpandable : false;
        }
        /**
         * @param {?} isBottomControl
         * @return {?}
         */
        showPaginationInput(isBottomControl) {
            return isBottomControl;
        }
        /**
         * @param {?} index
         * @param {?} cardIdWithMetadata
         * @return {?}
         */
        trackByCards(index, cardIdWithMetadata) {
            return cardIdWithMetadata.cardId;
        }
        /**
         * @param {?} event
         * @return {?}
         */
        onPaginationInputChange(event) {
            /** @type {?} */
            const input = (/** @type {?} */ (event.target));
            if (event.type === 'input' && input.value === '') {
                return;
            }
            /** @type {?} */
            const currentValue = Number(input.value) - 1;
            /** @type {?} */
            const nextValue = Math.min(Math.max(0, currentValue), this.numPages - 1);
            // Rectifying logic is at the container but the rectified value does not
            // appropriately make changes to the `<input>`.
            // Speculation: Angular seems to have some check on the template level that
            // does shallow equals and prevent changing on the same value.
            // Evidence: pageIndex change does fire, but `ngOnChanges` does not get
            // triggered when the value is the same (rectified value).
            if (input.value !== String(nextValue + 1)) {
                input.value = String(nextValue + 1);
            }
            this.pageIndexChanged.emit(nextValue);
        }
    }
    CardGridComponent.decorators = [
        { type: Component, args: [{
                    selector: 'metrics-card-grid-component',
                    templateUrl: './card_grid_component.ng.html',
                    styleUrls: ['./card_grid_component.css'],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    CardGridComponent.propDecorators = {
        isGroupExpandable: [{ type: Input }],
        isGroupExpanded: [{ type: Input }],
        groupName: [{ type: Input }],
        pageIndex: [{ type: Input }],
        numPages: [{ type: Input }],
        cardIdsWithMetadata: [{ type: Input }],
        cardObserver: [{ type: Input }],
        showPaginationControls: [{ type: Input }],
        pageIndexChanged: [{ type: Output }],
        groupExpansionToggled: [{ type: Output }]
    };
    /** @nocollapse */ CardGridComponent.ɵfac = function CardGridComponent_Factory(t) { return new (t || CardGridComponent)(); };
    /** @nocollapse */ CardGridComponent.ɵcmp = ɵɵdefineComponent({ type: CardGridComponent, selectors: [["metrics-card-grid-component"]], inputs: { isGroupExpandable: "isGroupExpandable", isGroupExpanded: "isGroupExpanded", groupName: "groupName", pageIndex: "pageIndex", numPages: "numPages", cardIdsWithMetadata: "cardIdsWithMetadata", cardObserver: "cardObserver", showPaginationControls: "showPaginationControls" }, outputs: { pageIndexChanged: "pageIndexChanged", groupExpansionToggled: "groupExpansionToggled" }, decls: 6, vars: 8, consts: [[4, "ngTemplateOutlet", "ngTemplateOutletContext"], [1, "card-grid"], [3, "cardId", "groupName", "pluginType", "cardObserver", "cardLazyLoader", 4, "ngFor", "ngForOf", "ngForTrackBy"], ["groupControls", ""], [3, "cardId", "groupName", "pluginType", "cardObserver", "cardLazyLoader"], ["class", "group-controls", 4, "ngIf"], [1, "group-controls"], [1, "prev-container"], ["class", "prev pagination-button", "mat-button", "", 3, "disabled", "click", 4, "ngIf", 6, "aria-label"], [1, "expand-container"], ["class", "expand-group-button", "mat-icon-button", "", 3, "click", 4, "ngIf", 6, "aria-label"], [1, "input-and-next-container"], [4, "ngIf"], ["mat-button", "", 1, "prev", "pagination-button", 3, "disabled", "click", 6, "aria-label"], ["mat-icon-button", "", 1, "expand-group-button", 3, "click", 6, "aria-label"], ["svgIcon", "expand_less_24px", 4, "ngIf", "ngIfElse"], ["expandMore", ""], ["svgIcon", "expand_less_24px"], ["svgIcon", "expand_more_24px"], ["class", "pagination-input", 4, "ngIf"], ["mat-button", "", 1, "next", "pagination-button", 3, "disabled", "click", 6, "aria-label"], [1, "pagination-input"], ["type", "number", "min", "1", 3, "value", "max", "input", "change"]], template: function CardGridComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵtemplate(0, CardGridComponent_ng_container_0_Template, 1, 0, "ng-container", 0);
            ɵɵelementStart(1, "div", 1);
            ɵɵtemplate(2, CardGridComponent_card_view_2_Template, 1, 5, "card-view", 2);
            ɵɵelementEnd();
            ɵɵtemplate(3, CardGridComponent_ng_container_3_Template, 1, 0, "ng-container", 0);
            ɵɵtemplate(4, CardGridComponent_ng_template_4_Template, 1, 1, "ng-template", null, 3, ɵɵtemplateRefExtractor);
        } if (rf & 2) {
            const _r3 = ɵɵreference(5);
            ɵɵproperty("ngTemplateOutlet", _r3)("ngTemplateOutletContext", ɵɵpureFunction0(6, _c9$1));
            ɵɵadvance(2);
            ɵɵproperty("ngForOf", ctx.cardIdsWithMetadata)("ngForTrackBy", ctx.trackByCards);
            ɵɵadvance(1);
            ɵɵproperty("ngTemplateOutlet", _r3)("ngTemplateOutletContext", ɵɵpureFunction0(7, _c10));
        } }, directives: [NgTemplateOutlet, NgForOf, CardViewContainer, CardLazyLoader, NgIf, MatButton, MatIcon], styles: [".mat-badge-content[_ngcontent-%COMP%]{font-weight:600;font-size:12px;font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:9px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:24px}.mat-h1[_ngcontent-%COMP%], .mat-headline[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%]{font:400 24px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h2[_ngcontent-%COMP%], .mat-title[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h3[_ngcontent-%COMP%], .mat-subheading-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%]{font:400 16px/28px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h4[_ngcontent-%COMP%], .mat-subheading-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%]{font:400 15px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h5[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%]{font:400 calc(14px * 0.83)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-h6[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:400 calc(14px * 0.67)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-body-strong[_ngcontent-%COMP%], .mat-body-2[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   p[_ngcontent-%COMP%]{margin:0 0 12px}.mat-small[_ngcontent-%COMP%], .mat-caption[_ngcontent-%COMP%]{font:400 12px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-display-4[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-4[_ngcontent-%COMP%]{font:300 112px/112px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.05em;margin:0 0 56px}.mat-display-3[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-3[_ngcontent-%COMP%]{font:400 56px/56px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.02em;margin:0 0 64px}.mat-display-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-2[_ngcontent-%COMP%]{font:400 45px/48px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.005em;margin:0 0 64px}.mat-display-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-1[_ngcontent-%COMP%]{font:400 34px/40px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 64px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%], .mat-flat-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-button-toggle[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card-title[_ngcontent-%COMP%]{font-size:24px;font-weight:500}.mat-card-header[_ngcontent-%COMP%]   .mat-card-title[_ngcontent-%COMP%]{font-size:20px}.mat-card-subtitle[_ngcontent-%COMP%], .mat-card-content[_ngcontent-%COMP%]{font-size:14px}.mat-checkbox[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-checkbox-layout[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{line-height:24px}.mat-chip[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-chip[_ngcontent-%COMP%]   .mat-chip-trailing-icon.mat-icon[_ngcontent-%COMP%], .mat-chip[_ngcontent-%COMP%]   .mat-chip-remove.mat-icon[_ngcontent-%COMP%]{font-size:18px}.mat-table[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-header-cell[_ngcontent-%COMP%]{font-size:12px;font-weight:500}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{font-size:14px}.mat-calendar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-calendar-body[_ngcontent-%COMP%]{font-size:13px}.mat-calendar-body-label[_ngcontent-%COMP%], .mat-calendar-period-button[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-calendar-table-header[_ngcontent-%COMP%]   th[_ngcontent-%COMP%]{font-size:11px;font-weight:400}.mat-dialog-title[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-expansion-panel-header[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:15px;font-weight:400}.mat-expansion-panel-content[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field[_ngcontent-%COMP%]{font-size:inherit;font-weight:400;line-height:1.125;font-family:Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.34375em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{font-size:150%;line-height:1.125}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]{height:1.5em;width:1.5em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{height:1.125em;line-height:1.125}.mat-form-field-infix[_ngcontent-%COMP%]{padding:.5em 0;border-top:.84375em solid transparent}.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34375em) scale(0.75);width:133.3333333333%}.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34374em) scale(0.75);width:133.3333433333%}.mat-form-field-label-wrapper[_ngcontent-%COMP%]{top:-0.84375em;padding-top:.84375em}.mat-form-field-label[_ngcontent-%COMP%]{top:1.34375em}.mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.34375em}.mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{font-size:75%;margin-top:.6666666667em;top:calc(100% - 1.7916666667em)}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.4375em 0}.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.001px);-ms-transform:translateY(-1.28125em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00101px);-ms-transform:translateY(-1.28124em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00102px);-ms-transform:translateY(-1.28123em) scale(0.75);width:133.3333533333%}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.28125em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{margin-top:.5416666667em;top:calc(100% - 1.6666666667em)}@media print{.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28122em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28121em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.2812em) scale(0.75)}}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.25em 0 .75em 0}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.09375em;margin-top:-0.5em}.mat-form-field-appearance-fill.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59374em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:1em 0 1em 0}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.84375em;margin-top:-0.25em}.mat-form-field-appearance-outline.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59374em) scale(0.75);width:133.3333433333%}.mat-grid-tile-header[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]{font-size:14px}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2), .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}input.mat-input-element[_ngcontent-%COMP%]{margin-top:-0.0625em}.mat-menu-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:400}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px}.mat-radio-button[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select-trigger[_ngcontent-%COMP%]{height:1.125em}.mat-slide-toggle-content[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-slider-thumb-label-text[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-stepper-vertical[_ngcontent-%COMP%], .mat-stepper-horizontal[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-step-label[_ngcontent-%COMP%]{font-size:14px;font-weight:400}.mat-step-sub-label-error[_ngcontent-%COMP%]{font-weight:normal}.mat-step-label-error[_ngcontent-%COMP%]{font-size:14px}.mat-step-label-selected[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-tab-group[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-toolbar[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0}.mat-tooltip[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:10px;padding-top:6px;padding-bottom:6px}.mat-tooltip-handset[_ngcontent-%COMP%]{font-size:14px;padding-top:8px;padding-bottom:8px}.mat-list-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:16px}.mat-optgroup-label[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-simple-snackbar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px}.mat-simple-snackbar-action[_ngcontent-%COMP%]{line-height:1;font-family:inherit;font-size:inherit;font-weight:500}.mat-tree[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{font-weight:400;font-size:14px}.mat-ripple[_ngcontent-%COMP%]{overflow:hidden;position:relative}.mat-ripple[_ngcontent-%COMP%]:not(:empty){transform:translateZ(0)}.mat-ripple.mat-ripple-unbounded[_ngcontent-%COMP%]{overflow:visible}.mat-ripple-element[_ngcontent-%COMP%]{position:absolute;border-radius:50%;pointer-events:none;transition:opacity,transform 0ms cubic-bezier(0, 0, 0.2, 1);transform:scale(0)}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{display:none}.cdk-visually-hidden[_ngcontent-%COMP%]{border:0;clip:rect(0 0 0 0);height:1px;margin:-1px;overflow:hidden;padding:0;position:absolute;width:1px;outline:0;-webkit-appearance:none;-moz-appearance:none}.cdk-overlay-container[_ngcontent-%COMP%], .cdk-global-overlay-wrapper[_ngcontent-%COMP%]{pointer-events:none;top:0;left:0;height:100%;width:100%}.cdk-overlay-container[_ngcontent-%COMP%]{position:fixed;z-index:1000}.cdk-overlay-container[_ngcontent-%COMP%]:empty{display:none}.cdk-global-overlay-wrapper[_ngcontent-%COMP%]{display:flex;position:absolute;z-index:1000}.cdk-overlay-pane[_ngcontent-%COMP%]{position:absolute;pointer-events:auto;box-sizing:border-box;z-index:1000;display:flex;max-width:100%;max-height:100%}.cdk-overlay-backdrop[_ngcontent-%COMP%]{position:absolute;top:0;bottom:0;left:0;right:0;z-index:1000;pointer-events:auto;-webkit-tap-highlight-color:transparent;transition:opacity 400ms cubic-bezier(0.25, 0.8, 0.25, 1);opacity:0}.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:1}@media screen and (-ms-high-contrast: active){.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:.6}}.cdk-overlay-dark-backdrop[_ngcontent-%COMP%]{background:rgba(0,0,0,.32)}.cdk-overlay-transparent-backdrop[_ngcontent-%COMP%], .cdk-overlay-transparent-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:0}.cdk-overlay-connected-position-bounding-box[_ngcontent-%COMP%]{position:absolute;z-index:1000;display:flex;flex-direction:column;min-width:1px;min-height:1px}.cdk-global-scrollblock[_ngcontent-%COMP%]{position:fixed;width:100%;overflow-y:scroll}@keyframes cdk-text-field-autofill-start{}@keyframes cdk-text-field-autofill-end{}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:-webkit-autofill{animation:cdk-text-field-autofill-start 0s 1ms}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:not(:-webkit-autofill){animation:cdk-text-field-autofill-end 0s 1ms}textarea.cdk-textarea-autosize[_ngcontent-%COMP%]{resize:none}textarea.cdk-textarea-autosize-measuring[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:auto !important;overflow:hidden !important}textarea.cdk-textarea-autosize-measuring-firefox[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:0 !important}.mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.1)}.mat-option[_ngcontent-%COMP%]{color:#212121}.mat-option[_ngcontent-%COMP%]:hover:not(.mat-option-disabled), .mat-option[_ngcontent-%COMP%]:focus:not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple):not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-active[_ngcontent-%COMP%]{background:rgba(0,0,0,.04);color:#212121}.mat-option.mat-option-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-primary[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f57c00}.mat-accent[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f44336}.mat-optgroup-label[_ngcontent-%COMP%]{color:#616161}.mat-optgroup-disabled[_ngcontent-%COMP%]   .mat-optgroup-label[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-pseudo-checkbox[_ngcontent-%COMP%]{color:#616161}.mat-pseudo-checkbox[_ngcontent-%COMP%]::after{color:#fafafa}.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{color:#b0b0b0}.mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f57c00}.mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f44336}.mat-pseudo-checkbox-checked.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{background:#b0b0b0}.mat-elevation-z0[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z1[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z2[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z3[_ngcontent-%COMP%]{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z4[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z5[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 5px 8px 0px rgba(0, 0, 0, 0.14),0px 1px 14px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z6[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z7[_ngcontent-%COMP%]{box-shadow:0px 4px 5px -2px rgba(0, 0, 0, 0.2),0px 7px 10px 1px rgba(0, 0, 0, 0.14),0px 2px 16px 1px rgba(0, 0, 0, 0.12)}.mat-elevation-z8[_ngcontent-%COMP%]{box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z9[_ngcontent-%COMP%]{box-shadow:0px 5px 6px -3px rgba(0, 0, 0, 0.2),0px 9px 12px 1px rgba(0, 0, 0, 0.14),0px 3px 16px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z10[_ngcontent-%COMP%]{box-shadow:0px 6px 6px -3px rgba(0, 0, 0, 0.2),0px 10px 14px 1px rgba(0, 0, 0, 0.14),0px 4px 18px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z11[_ngcontent-%COMP%]{box-shadow:0px 6px 7px -4px rgba(0, 0, 0, 0.2),0px 11px 15px 1px rgba(0, 0, 0, 0.14),0px 4px 20px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z12[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z13[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 13px 19px 2px rgba(0, 0, 0, 0.14),0px 5px 24px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z14[_ngcontent-%COMP%]{box-shadow:0px 7px 9px -4px rgba(0, 0, 0, 0.2),0px 14px 21px 2px rgba(0, 0, 0, 0.14),0px 5px 26px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z15[_ngcontent-%COMP%]{box-shadow:0px 8px 9px -5px rgba(0, 0, 0, 0.2),0px 15px 22px 2px rgba(0, 0, 0, 0.14),0px 6px 28px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z16[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z17[_ngcontent-%COMP%]{box-shadow:0px 8px 11px -5px rgba(0, 0, 0, 0.2),0px 17px 26px 2px rgba(0, 0, 0, 0.14),0px 6px 32px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z18[_ngcontent-%COMP%]{box-shadow:0px 9px 11px -5px rgba(0, 0, 0, 0.2),0px 18px 28px 2px rgba(0, 0, 0, 0.14),0px 7px 34px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z19[_ngcontent-%COMP%]{box-shadow:0px 9px 12px -6px rgba(0, 0, 0, 0.2),0px 19px 29px 2px rgba(0, 0, 0, 0.14),0px 7px 36px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z20[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 20px 31px 3px rgba(0, 0, 0, 0.14),0px 8px 38px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z21[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 21px 33px 3px rgba(0, 0, 0, 0.14),0px 8px 40px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z22[_ngcontent-%COMP%]{box-shadow:0px 10px 14px -6px rgba(0, 0, 0, 0.2),0px 22px 35px 3px rgba(0, 0, 0, 0.14),0px 8px 42px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z23[_ngcontent-%COMP%]{box-shadow:0px 11px 14px -7px rgba(0, 0, 0, 0.2),0px 23px 36px 3px rgba(0, 0, 0, 0.14),0px 9px 44px 8px rgba(0, 0, 0, 0.12)}.mat-elevation-z24[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12)}.mat-app-background[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-theme-loaded-marker[_ngcontent-%COMP%]{display:none}.mat-autocomplete-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-autocomplete-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover){background:#fff}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover):not(.mat-option-disabled){color:#212121}.mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f57c00}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{outline:solid 1px;border-radius:0}.mat-badge-accent[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-badge-warn[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f44336}.mat-badge[_ngcontent-%COMP%]{position:relative}.mat-badge-hidden[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{display:none}.mat-badge-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#b9b9b9;color:#757575}.mat-badge-content[_ngcontent-%COMP%]{position:absolute;text-align:center;display:inline-block;border-radius:50%;transition:transform 200ms ease-in-out;transform:scale(0.6);overflow:hidden;white-space:nowrap;text-overflow:ellipsis;pointer-events:none}.ng-animate-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%], .mat-badge-content._mat-animation-noopable[_ngcontent-%COMP%]{transition:none}.mat-badge-content.mat-badge-active[_ngcontent-%COMP%]{transform:none}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:16px;height:16px;line-height:16px}.mat-badge-small.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-8px}.mat-badge-small.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-8px}.mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-16px}.mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-16px}.mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-8px}.mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-8px}.mat-badge-medium[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:22px;height:22px;line-height:22px}.mat-badge-medium.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-11px}.mat-badge-medium.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-11px}.mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-22px}.mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-22px}.mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-11px}.mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-11px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:28px;height:28px;line-height:28px}.mat-badge-large.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-14px}.mat-badge-large.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-14px}.mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-28px}.mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-28px}.mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-14px}.mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-14px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]{color:inherit;background:transparent}.mat-button.mat-primary[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-button[disabled][disabled][_ngcontent-%COMP%], .mat-icon-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-icon-button[disabled][disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-stroked-button[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f44336}.mat-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:transparent}.mat-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{opacity:.1;background-color:currentColor}.mat-button-focus-overlay[_ngcontent-%COMP%]{background:#000}.mat-stroked-button[_ngcontent-%COMP%]:not([disabled]){border-color:rgba(0,0,0,.12)}.mat-flat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{color:#212121;background-color:#fff}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-flat-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-stroked-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-flat-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-raised-button[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone[_ngcontent-%COMP%], .mat-button-toggle-group[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{box-shadow:none}.mat-button-toggle[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-button-toggle[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121;background:#fff}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:#000}.mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12)}[dir=rtl][_ngcontent-%COMP%]   .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-group-appearance-standard.mat-button-toggle-vertical[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:none;border-top:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#616161}.mat-button-toggle-checked.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121}.mat-button-toggle-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.26);background-color:#eee}.mat-button-toggle-disabled.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{background:#fff}.mat-button-toggle-disabled.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#bdbdbd}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{border:solid 1px rgba(0,0,0,.12)}.mat-card[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-card[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-card.mat-card-flat[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-card-subtitle[_ngcontent-%COMP%]{color:#616161}.mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#616161}.mat-checkbox-checkmark[_ngcontent-%COMP%]{fill:#fafafa}.mat-checkbox-checkmark-path[_ngcontent-%COMP%]{stroke:#fafafa !important}.mat-checkbox-mixedmark[_ngcontent-%COMP%]{background-color:#fafafa}.mat-checkbox-indeterminate.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f57c00}.mat-checkbox-indeterminate.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#ff9800}.mat-checkbox-indeterminate.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f44336}.mat-checkbox-disabled.mat-checkbox-checked[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-disabled.mat-checkbox-indeterminate[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]:not(.mat-checkbox-checked)   .mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{color:#616161}.mat-checkbox[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%]{background:#f57c00}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%]{background:#ff9800}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%]{background:#f44336}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#212121}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#212121;opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled):active{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled)   .mat-chip-remove[_ngcontent-%COMP%]:hover{opacity:.54}.mat-chip.mat-standard-chip.mat-chip-disabled[_ngcontent-%COMP%]{opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]::after{background:#000}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-table[_ngcontent-%COMP%]{background:#fff}.mat-table[_ngcontent-%COMP%]   thead[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tbody[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tfoot[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], [mat-header-row][_ngcontent-%COMP%], [mat-row][_ngcontent-%COMP%], [mat-footer-row][_ngcontent-%COMP%], .mat-table-sticky[_ngcontent-%COMP%]{background:inherit}mat-row[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], th.mat-header-cell[_ngcontent-%COMP%], td.mat-cell[_ngcontent-%COMP%], td.mat-footer-cell[_ngcontent-%COMP%]{border-bottom-color:rgba(0,0,0,.12)}.mat-header-cell[_ngcontent-%COMP%]{color:#616161}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{color:#212121}.mat-calendar-arrow[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.54)}.mat-datepicker-toggle[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-next-button[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-previous-button[_ngcontent-%COMP%]{color:rgba(0,0,0,.54)}.mat-calendar-table-header[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-calendar-table-header-divider[_ngcontent-%COMP%]::after{background:rgba(0,0,0,.12)}.mat-calendar-body-label[_ngcontent-%COMP%]{color:#616161}.mat-calendar-body-cell-content[_ngcontent-%COMP%]{color:#212121;border-color:transparent}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){color:#757575}.mat-calendar-body-cell[_ngcontent-%COMP%]:not(.mat-calendar-body-disabled):hover > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-keyboard-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-program-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){background-color:rgba(0,0,0,.04)}.mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.38)}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.18)}.mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.4)}.mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12);background-color:#fff;color:#212121}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.4)}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.4)}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content-touch[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-datepicker-toggle-active[_ngcontent-%COMP%]{color:#f57c00}.mat-datepicker-toggle-active.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-datepicker-toggle-active.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-dialog-container[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-divider[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-divider-vertical[_ngcontent-%COMP%]{border-right-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-expansion-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-action-row[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-keyboard-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-program-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded)   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover:not([aria-disabled=true]){background:rgba(0,0,0,.04)}@media(hover: none){.mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded):not([aria-disabled=true])   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover{background:#fff}}.mat-expansion-panel-header-title[_ngcontent-%COMP%]{color:#212121}.mat-expansion-panel-header-description[_ngcontent-%COMP%], .mat-expansion-indicator[_ngcontent-%COMP%]::after{color:#616161}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-title[_ngcontent-%COMP%], .mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-description[_ngcontent-%COMP%]{color:inherit}.mat-form-field-label[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-hint[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-focused[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-ripple[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid)   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f57c00}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-accent   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#ff9800}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-warn   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#f44336}.mat-error[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-hint[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-legacy.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-standard[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-standard.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.04)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.02)}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:transparent}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.12)}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:rgba(0,0,0,.87)}.mat-form-field-appearance-outline.mat-focused[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field-appearance-outline.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-appearance-outline.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-invalid.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.06)}.mat-icon.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-icon.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-icon.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-type-mat-native-select[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#616161}.mat-input-element[_ngcontent-%COMP%]:disabled, .mat-form-field-type-mat-native-select.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#757575}.mat-input-element[_ngcontent-%COMP%]{caret-color:#f57c00}.mat-input-element[_ngcontent-%COMP%]::placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-moz-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-webkit-input-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]:-ms-input-placeholder{color:rgba(97,97,97,.42)}.mat-accent[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%], .mat-form-field-invalid[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#f44336}.mat-form-field-type-mat-native-select.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{color:#616161}.mat-list-item-disabled[_ngcontent-%COMP%]{background-color:#eee}.mat-list-option[_ngcontent-%COMP%]:hover, .mat-list-option[_ngcontent-%COMP%]:focus, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.04)}.mat-list-single-selected-option[_ngcontent-%COMP%], .mat-list-single-selected-option[_ngcontent-%COMP%]:hover, .mat-list-single-selected-option[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.12)}.mat-menu-panel[_ngcontent-%COMP%]{background:#fff}.mat-menu-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-menu-item[_ngcontent-%COMP%]{background:transparent;color:#212121}.mat-menu-item[disabled][_ngcontent-%COMP%], .mat-menu-item[disabled][_ngcontent-%COMP%]::after{color:rgba(0,0,0,.38)}.mat-menu-item[_ngcontent-%COMP%]   .mat-icon-no-color[_ngcontent-%COMP%], .mat-menu-item-submenu-trigger[_ngcontent-%COMP%]::after{color:rgba(0,0,0,.54)}.mat-menu-item[_ngcontent-%COMP%]:hover:not([disabled]), .mat-menu-item.cdk-program-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item.cdk-keyboard-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item-highlighted[_ngcontent-%COMP%]:not([disabled]){background:rgba(0,0,0,.04)}.mat-paginator[_ngcontent-%COMP%]{background:#fff}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{color:#616161}.mat-paginator-decrement[_ngcontent-%COMP%], .mat-paginator-increment[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54);border-right:2px solid rgba(0,0,0,.54)}.mat-paginator-first[_ngcontent-%COMP%], .mat-paginator-last[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54)}.mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-decrement[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-increment[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-first[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-last[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f57c00}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#ff9800}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f44336}.mat-progress-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f57c00}.mat-progress-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#ff9800}.mat-progress-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f44336}.mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#616161}.mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f57c00}.mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#ff9800}.mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#ff9800}.mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f44336}.mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f44336}.mat-radio-button.mat-radio-disabled.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-label-content[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-radio-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-select-value[_ngcontent-%COMP%]{color:#212121}.mat-select-placeholder[_ngcontent-%COMP%]{color:rgba(97,97,97,.42)}.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%]{color:#757575}.mat-select-arrow[_ngcontent-%COMP%]{color:#616161}.mat-select-panel[_ngcontent-%COMP%]{background:#fff}.mat-select-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-select-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple){background:rgba(0,0,0,.12)}.mat-form-field.mat-focused.mat-primary[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-invalid[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#757575}.mat-drawer-container[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-drawer[_ngcontent-%COMP%]{background-color:#fff;color:#212121}.mat-drawer.mat-drawer-push[_ngcontent-%COMP%]{background-color:#fff}.mat-drawer[_ngcontent-%COMP%]:not(.mat-drawer-side){box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-drawer-side[_ngcontent-%COMP%]{border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-backdrop.mat-drawer-shown[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.6)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.54)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.54)}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.54)}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle[_ngcontent-%COMP%]:not(.mat-checked)   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-slide-toggle-thumb[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12);background-color:#fafafa}.mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-primary[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f57c00}.mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-accent[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#ff9800}.mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-warn[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f44336}.mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.2)}.mat-slider[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%], .cdk-focused[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-disabled[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]   .mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing)   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26);background-color:transparent}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26)}.mat-slider-has-ticks[_ngcontent-%COMP%]   .mat-slider-wrapper[_ngcontent-%COMP%]::after{border-color:rgba(0,0,0,.7)}.mat-slider-horizontal[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to right, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent);background-image:-moz-repeating-linear-gradient(0.0001deg, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-slider-vertical[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to bottom, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-step-header.cdk-keyboard-focused[_ngcontent-%COMP%], .mat-step-header.cdk-program-focused[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]:hover{background-color:rgba(0,0,0,.04)}@media(hover: none){.mat-step-header[_ngcontent-%COMP%]:hover{background:none}}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-optional[_ngcontent-%COMP%]{color:#616161}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon[_ngcontent-%COMP%]{background-color:#616161;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-selected[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-done[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-edit[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-error[_ngcontent-%COMP%]{background-color:transparent;color:#f44336}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-active[_ngcontent-%COMP%]{color:#212121}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-error[_ngcontent-%COMP%]{color:#f44336}.mat-stepper-horizontal[_ngcontent-%COMP%], .mat-stepper-vertical[_ngcontent-%COMP%]{background-color:#fff}.mat-stepper-vertical-line[_ngcontent-%COMP%]::before{border-left-color:rgba(0,0,0,.12)}.mat-horizontal-stepper-header[_ngcontent-%COMP%]::before, .mat-horizontal-stepper-header[_ngcontent-%COMP%]::after, .mat-stepper-horizontal-line[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-sort-header-arrow[_ngcontent-%COMP%]{color:#616161}.mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-header[_ngcontent-%COMP%]{border-bottom:1px solid rgba(0,0,0,.12)}.mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%]{border-top:1px solid rgba(0,0,0,.12);border-bottom:none}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{color:#212121}.mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:#757575}.mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#212121}.mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#757575}.mat-tab-group[class*=mat-background-][_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar[class*=mat-background-][_ngcontent-%COMP%]{border-bottom:none;border-top:none}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-toolbar[_ngcontent-%COMP%]{background:#f5f5f5;color:#212121}.mat-toolbar.mat-primary[_ngcontent-%COMP%]{background:#f57c00;color:#fff}.mat-toolbar.mat-accent[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-toolbar.mat-warn[_ngcontent-%COMP%]{background:#f44336;color:#fff}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:currentColor}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:inherit}.mat-toolbar[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:currentColor}.mat-tooltip[_ngcontent-%COMP%]{background:rgba(97,97,97,.9)}.mat-tree[_ngcontent-%COMP%]{background:#fff}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{color:#212121}.mat-snack-bar-container[_ngcontent-%COMP%]{color:rgba(255,255,255,.7);background:#323232;box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-simple-snackbar-action[_ngcontent-%COMP%]{color:#ff9800}.card-grid[_ngcontent-%COMP%]{display:grid;grid-template-columns:repeat(auto-fill, minmax(335px, auto));gap:16px;padding:16px}card-view[_ngcontent-%COMP%]{border:1px solid #ddd;border-radius:4px;box-sizing:border-box;contain:strict;height:100%;min-height:320px}card-view.full-width[_ngcontent-%COMP%]{contain:layout paint;grid-column-start:1;grid-column-end:-1}card-view.full-height[_ngcontent-%COMP%]{min-height:480px}.group-controls[_ngcontent-%COMP%]{color:#616161;display:grid;align-items:center;grid-template-columns:1fr 1fr 1fr;gap:16px;padding:0 16px}.group-controls[_ngcontent-%COMP%]:first-of-type{padding-top:16px}.group-controls[_ngcontent-%COMP%]:last-of-type{padding-bottom:16px}.prev-container[_ngcontent-%COMP%]{justify-self:flex-start}.expand-container[_ngcontent-%COMP%]{justify-self:center}.input-and-next-container[_ngcontent-%COMP%]{justify-self:flex-end}.pagination-input[_ngcontent-%COMP%]{margin-right:16px}.expand-group-button[_ngcontent-%COMP%], .pagination-button[_ngcontent-%COMP%]{background-color:#fff;color:#616161}"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/views/main_view/card_grid_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    // Tag group can be collapsed. Even when it is collapsed, we show three cards.
    /** @type {?} */
    const ITEMS_COLLAPSED_CLIP_SIZE = 3;
    class CardGridContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            // groupName must be non-null if the group should be collapse/expand-able.
            this.groupName = null;
            this.groupName$ = new BehaviorSubject(null);
            this.pageIndex$ = new BehaviorSubject(0);
            this.items$ = new BehaviorSubject([]);
            this.numPages$ = combineLatest([
                this.items$,
                this.store.select(getPageSize),
            ]).pipe(map((/**
             * @param {?} __0
             * @return {?}
             */
            ([items, pageSize]) => {
                return Math.ceil(items.length / pageSize);
            })));
            this.isGroupExpanded$ = this.groupName$.pipe(switchMap((/**
             * @param {?} groupName
             * @return {?}
             */
            (groupName) => {
                return groupName !== null
                    ? this.store.select(getMetricsTagGroupExpansionState, groupName)
                    : of(true);
            })));
            this.showPaginationControls$ = combineLatest([
                this.numPages$,
                this.store.select(getPageSize),
                this.isGroupExpanded$,
            ]).pipe(map((/**
             * @param {?} __0
             * @return {?}
             */
            ([numPages, pageSize, isGroupExpanded]) => {
                if (numPages <= 1) {
                    return false;
                }
                return pageSize <= ITEMS_COLLAPSED_CLIP_SIZE || isGroupExpanded;
            })));
            this.isGroupExpandable$ = combineLatest([
                this.items$,
                this.store.select(getPageSize),
            ]).pipe(map((/**
             * @param {?} __0
             * @return {?}
             */
            ([items, pageSize]) => {
                if (this.groupName === null ||
                    pageSize <= ITEMS_COLLAPSED_CLIP_SIZE ||
                    items.length <= ITEMS_COLLAPSED_CLIP_SIZE) {
                    return false;
                }
                return true;
            })));
            this.normalizedPageIndex$ = combineLatest([
                this.pageIndex$,
                this.numPages$,
            ]).pipe(tap((/**
             * @param {?} __0
             * @return {?}
             */
            ([pageIndex, numPages]) => {
                // Cycle in the Observable but only loops when pageIndex is not
                // valid and does not repeat more than once.
                if (numPages === 0) {
                    return;
                }
                if (pageIndex >= numPages) {
                    this.pageIndex$.next(numPages - 1);
                }
                else if (pageIndex < 0) {
                    this.pageIndex$.next(0);
                }
            })), map((/**
             * @param {?} __0
             * @return {?}
             */
            ([pageIndex, numPages]) => {
                return Math.min(Math.max(pageIndex, 0), numPages - 1);
            })), shareReplay(1));
            this.pagedItems$ = combineLatest([
                this.items$,
                this.store.select(getPageSize),
                this.normalizedPageIndex$,
                this.isGroupExpanded$,
            ]).pipe(map((/**
             * @param {?} __0
             * @return {?}
             */
            ([items, pageSize, pageIndex, expanded]) => {
                /** @type {?} */
                const startIndex = pageSize * pageIndex;
                // We only render 3 cards when collapsed.
                /** @type {?} */
                const endIndex = pageSize * pageIndex +
                    Math.min(expanded ? pageSize : ITEMS_COLLAPSED_CLIP_SIZE, pageSize);
                return items.slice(startIndex, endIndex);
            })));
        }
        /**
         * @param {?} changes
         * @return {?}
         */
        ngOnChanges(changes) {
            if (changes['cardIdsWithMetadata']) {
                this.items$.next(this.cardIdsWithMetadata);
            }
            if (changes['groupName']) {
                this.groupName$.next(this.groupName);
            }
        }
        /**
         * @param {?} newIndex
         * @return {?}
         */
        onPageIndexChanged(newIndex) {
            this.pageIndex$.next(newIndex);
        }
        /**
         * @return {?}
         */
        onGroupExpansionToggled() {
            if (this.groupName === null) {
                throw new RangeError('Invariant error: expansion cannot be toggled when groupName is null');
            }
            this.store.dispatch(metricsTagGroupExpansionChanged({ tagGroup: this.groupName }));
        }
    }
    CardGridContainer.decorators = [
        { type: Component, args: [{
                    selector: 'metrics-card-grid',
                    template: `
    <metrics-card-grid-component
      [isGroupExpandable]="isGroupExpandable$ | async"
      [isGroupExpanded]="isGroupExpanded$ | async"
      [groupName]="groupName"
      [pageIndex]="normalizedPageIndex$ | async"
      [numPages]="numPages$ | async"
      [showPaginationControls]="showPaginationControls$ | async"
      [cardIdsWithMetadata]="pagedItems$ | async"
      [cardObserver]="cardObserver"
      (pageIndexChanged)="onPageIndexChanged($event)"
      (groupExpansionToggled)="onGroupExpansionToggled()"
    >
    </metrics-card-grid-component>
  `,
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    CardGridContainer.ctorParameters = () => [
        { type: Store }
    ];
    CardGridContainer.propDecorators = {
        groupName: [{ type: Input }],
        cardIdsWithMetadata: [{ type: Input }],
        cardObserver: [{ type: Input }]
    };
    /** @nocollapse */ CardGridContainer.ɵfac = function CardGridContainer_Factory(t) { return new (t || CardGridContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ CardGridContainer.ɵcmp = ɵɵdefineComponent({ type: CardGridContainer, selectors: [["metrics-card-grid"]], inputs: { groupName: "groupName", cardIdsWithMetadata: "cardIdsWithMetadata", cardObserver: "cardObserver" }, features: [ɵɵNgOnChangesFeature], decls: 7, vars: 20, consts: [[3, "isGroupExpandable", "isGroupExpanded", "groupName", "pageIndex", "numPages", "showPaginationControls", "cardIdsWithMetadata", "cardObserver", "pageIndexChanged", "groupExpansionToggled"]], template: function CardGridContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "metrics-card-grid-component", 0);
            ɵɵlistener("pageIndexChanged", function CardGridContainer_Template_metrics_card_grid_component_pageIndexChanged_0_listener($event) { return ctx.onPageIndexChanged($event); })("groupExpansionToggled", function CardGridContainer_Template_metrics_card_grid_component_groupExpansionToggled_0_listener() { return ctx.onGroupExpansionToggled(); });
            ɵɵpipe(1, "async");
            ɵɵpipe(2, "async");
            ɵɵpipe(3, "async");
            ɵɵpipe(4, "async");
            ɵɵpipe(5, "async");
            ɵɵpipe(6, "async");
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵproperty("isGroupExpandable", ɵɵpipeBind1(1, 8, ctx.isGroupExpandable$))("isGroupExpanded", ɵɵpipeBind1(2, 10, ctx.isGroupExpanded$))("groupName", ctx.groupName)("pageIndex", ɵɵpipeBind1(3, 12, ctx.normalizedPageIndex$))("numPages", ɵɵpipeBind1(4, 14, ctx.numPages$))("showPaginationControls", ɵɵpipeBind1(5, 16, ctx.showPaginationControls$))("cardIdsWithMetadata", ɵɵpipeBind1(6, 18, ctx.pagedItems$))("cardObserver", ctx.cardObserver);
        } }, directives: [CardGridComponent], pipes: [AsyncPipe], encapsulation: 2, changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/views/main_view/card_groups_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    function CardGroupsComponent_div_0_span_5_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "span", 6);
        ɵɵtext(1);
        ɵɵelementEnd();
    } if (rf & 2) {
        const group_r1 = ɵɵnextContext().$implicit;
        ɵɵadvance(1);
        ɵɵtextInterpolate1("", group_r1.items.length, " cards");
    } }
    function CardGroupsComponent_div_0_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 1);
        ɵɵelementStart(1, "div", 2);
        ɵɵelementStart(2, "span");
        ɵɵelementStart(3, "span", 3);
        ɵɵtext(4);
        ɵɵelementEnd();
        ɵɵtemplate(5, CardGroupsComponent_div_0_span_5_Template, 2, 1, "span", 4);
        ɵɵelementEnd();
        ɵɵelementEnd();
        ɵɵelement(6, "metrics-card-grid", 5);
        ɵɵelementEnd();
    } if (rf & 2) {
        const group_r1 = ctx.$implicit;
        const ctx_r0 = ɵɵnextContext();
        ɵɵadvance(3);
        ɵɵpropertyInterpolate("title", group_r1.groupName);
        ɵɵadvance(1);
        ɵɵtextInterpolate(group_r1.groupName);
        ɵɵadvance(1);
        ɵɵproperty("ngIf", group_r1.items.length > 1);
        ɵɵadvance(1);
        ɵɵproperty("cardIdsWithMetadata", group_r1.items)("cardObserver", ctx_r0.cardObserver)("groupName", group_r1.groupName);
    } }
    class CardGroupsComponent {
        constructor() {
            this.PluginType = PluginType;
        }
        /**
         * @param {?} index
         * @param {?} cardGroup
         * @return {?}
         */
        trackByGroup(index, cardGroup) {
            return cardGroup.groupName;
        }
    }
    CardGroupsComponent.decorators = [
        { type: Component, args: [{
                    selector: 'metrics-card-groups-component',
                    template: `
    <div
      *ngFor="let group of cardGroups; trackBy: trackByGroup"
      class="card-group"
    >
      <div class="group-toolbar">
        <span>
          <span
            class="group-title"
            aria-role="heading"
            aria-level="3"
            title="{{ group.groupName }}"
            >{{ group.groupName }}</span
          >
          <span *ngIf="group.items.length > 1" class="group-card-count"
            >{{ group.items.length }} cards</span
          >
        </span>
      </div>
      <metrics-card-grid
        [cardIdsWithMetadata]="group.items"
        [cardObserver]="cardObserver"
        [groupName]="group.groupName"
      ></metrics-card-grid>
    </div>
  `,
                    styleUrls: [`card_groups_component.css`],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    CardGroupsComponent.propDecorators = {
        cardGroups: [{ type: Input }],
        cardObserver: [{ type: Input }]
    };
    /** @nocollapse */ CardGroupsComponent.ɵfac = function CardGroupsComponent_Factory(t) { return new (t || CardGroupsComponent)(); };
    /** @nocollapse */ CardGroupsComponent.ɵcmp = ɵɵdefineComponent({ type: CardGroupsComponent, selectors: [["metrics-card-groups-component"]], inputs: { cardGroups: "cardGroups", cardObserver: "cardObserver" }, decls: 1, vars: 2, consts: [["class", "card-group", 4, "ngFor", "ngForOf", "ngForTrackBy"], [1, "card-group"], [1, "group-toolbar"], ["aria-role", "heading", "aria-level", "3", 1, "group-title", 3, "title"], ["class", "group-card-count", 4, "ngIf"], [3, "cardIdsWithMetadata", "cardObserver", "groupName"], [1, "group-card-count"]], template: function CardGroupsComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵtemplate(0, CardGroupsComponent_div_0_Template, 7, 6, "div", 0);
        } if (rf & 2) {
            ɵɵproperty("ngForOf", ctx.cardGroups)("ngForTrackBy", ctx.trackByGroup);
        } }, directives: [NgForOf, NgIf, CardGridContainer], styles: [".mat-badge-content[_ngcontent-%COMP%]{font-weight:600;font-size:12px;font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:9px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:24px}.mat-h1[_ngcontent-%COMP%], .mat-headline[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%]{font:400 24px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h2[_ngcontent-%COMP%], .mat-title[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h3[_ngcontent-%COMP%], .mat-subheading-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%]{font:400 16px/28px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h4[_ngcontent-%COMP%], .mat-subheading-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%]{font:400 15px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h5[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%]{font:400 calc(14px * 0.83)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-h6[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:400 calc(14px * 0.67)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-body-strong[_ngcontent-%COMP%], .mat-body-2[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   p[_ngcontent-%COMP%]{margin:0 0 12px}.mat-small[_ngcontent-%COMP%], .mat-caption[_ngcontent-%COMP%]{font:400 12px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-display-4[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-4[_ngcontent-%COMP%]{font:300 112px/112px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.05em;margin:0 0 56px}.mat-display-3[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-3[_ngcontent-%COMP%]{font:400 56px/56px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.02em;margin:0 0 64px}.mat-display-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-2[_ngcontent-%COMP%]{font:400 45px/48px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.005em;margin:0 0 64px}.mat-display-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-1[_ngcontent-%COMP%]{font:400 34px/40px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 64px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%], .mat-flat-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-button-toggle[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card-title[_ngcontent-%COMP%]{font-size:24px;font-weight:500}.mat-card-header[_ngcontent-%COMP%]   .mat-card-title[_ngcontent-%COMP%]{font-size:20px}.mat-card-subtitle[_ngcontent-%COMP%], .mat-card-content[_ngcontent-%COMP%]{font-size:14px}.mat-checkbox[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-checkbox-layout[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{line-height:24px}.mat-chip[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-chip[_ngcontent-%COMP%]   .mat-chip-trailing-icon.mat-icon[_ngcontent-%COMP%], .mat-chip[_ngcontent-%COMP%]   .mat-chip-remove.mat-icon[_ngcontent-%COMP%]{font-size:18px}.mat-table[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-header-cell[_ngcontent-%COMP%]{font-size:12px;font-weight:500}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{font-size:14px}.mat-calendar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-calendar-body[_ngcontent-%COMP%]{font-size:13px}.mat-calendar-body-label[_ngcontent-%COMP%], .mat-calendar-period-button[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-calendar-table-header[_ngcontent-%COMP%]   th[_ngcontent-%COMP%]{font-size:11px;font-weight:400}.mat-dialog-title[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-expansion-panel-header[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:15px;font-weight:400}.mat-expansion-panel-content[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field[_ngcontent-%COMP%]{font-size:inherit;font-weight:400;line-height:1.125;font-family:Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.34375em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{font-size:150%;line-height:1.125}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]{height:1.5em;width:1.5em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{height:1.125em;line-height:1.125}.mat-form-field-infix[_ngcontent-%COMP%]{padding:.5em 0;border-top:.84375em solid transparent}.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34375em) scale(0.75);width:133.3333333333%}.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34374em) scale(0.75);width:133.3333433333%}.mat-form-field-label-wrapper[_ngcontent-%COMP%]{top:-0.84375em;padding-top:.84375em}.mat-form-field-label[_ngcontent-%COMP%]{top:1.34375em}.mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.34375em}.mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{font-size:75%;margin-top:.6666666667em;top:calc(100% - 1.7916666667em)}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.4375em 0}.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.001px);-ms-transform:translateY(-1.28125em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00101px);-ms-transform:translateY(-1.28124em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00102px);-ms-transform:translateY(-1.28123em) scale(0.75);width:133.3333533333%}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.28125em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{margin-top:.5416666667em;top:calc(100% - 1.6666666667em)}@media print{.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28122em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28121em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.2812em) scale(0.75)}}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.25em 0 .75em 0}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.09375em;margin-top:-0.5em}.mat-form-field-appearance-fill.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59374em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:1em 0 1em 0}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.84375em;margin-top:-0.25em}.mat-form-field-appearance-outline.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59374em) scale(0.75);width:133.3333433333%}.mat-grid-tile-header[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]{font-size:14px}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2), .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}input.mat-input-element[_ngcontent-%COMP%]{margin-top:-0.0625em}.mat-menu-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:400}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px}.mat-radio-button[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select-trigger[_ngcontent-%COMP%]{height:1.125em}.mat-slide-toggle-content[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-slider-thumb-label-text[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-stepper-vertical[_ngcontent-%COMP%], .mat-stepper-horizontal[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-step-label[_ngcontent-%COMP%]{font-size:14px;font-weight:400}.mat-step-sub-label-error[_ngcontent-%COMP%]{font-weight:normal}.mat-step-label-error[_ngcontent-%COMP%]{font-size:14px}.mat-step-label-selected[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-tab-group[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-toolbar[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0}.mat-tooltip[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:10px;padding-top:6px;padding-bottom:6px}.mat-tooltip-handset[_ngcontent-%COMP%]{font-size:14px;padding-top:8px;padding-bottom:8px}.mat-list-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:16px}.mat-optgroup-label[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-simple-snackbar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px}.mat-simple-snackbar-action[_ngcontent-%COMP%]{line-height:1;font-family:inherit;font-size:inherit;font-weight:500}.mat-tree[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{font-weight:400;font-size:14px}.mat-ripple[_ngcontent-%COMP%]{overflow:hidden;position:relative}.mat-ripple[_ngcontent-%COMP%]:not(:empty){transform:translateZ(0)}.mat-ripple.mat-ripple-unbounded[_ngcontent-%COMP%]{overflow:visible}.mat-ripple-element[_ngcontent-%COMP%]{position:absolute;border-radius:50%;pointer-events:none;transition:opacity,transform 0ms cubic-bezier(0, 0, 0.2, 1);transform:scale(0)}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{display:none}.cdk-visually-hidden[_ngcontent-%COMP%]{border:0;clip:rect(0 0 0 0);height:1px;margin:-1px;overflow:hidden;padding:0;position:absolute;width:1px;outline:0;-webkit-appearance:none;-moz-appearance:none}.cdk-overlay-container[_ngcontent-%COMP%], .cdk-global-overlay-wrapper[_ngcontent-%COMP%]{pointer-events:none;top:0;left:0;height:100%;width:100%}.cdk-overlay-container[_ngcontent-%COMP%]{position:fixed;z-index:1000}.cdk-overlay-container[_ngcontent-%COMP%]:empty{display:none}.cdk-global-overlay-wrapper[_ngcontent-%COMP%]{display:flex;position:absolute;z-index:1000}.cdk-overlay-pane[_ngcontent-%COMP%]{position:absolute;pointer-events:auto;box-sizing:border-box;z-index:1000;display:flex;max-width:100%;max-height:100%}.cdk-overlay-backdrop[_ngcontent-%COMP%]{position:absolute;top:0;bottom:0;left:0;right:0;z-index:1000;pointer-events:auto;-webkit-tap-highlight-color:transparent;transition:opacity 400ms cubic-bezier(0.25, 0.8, 0.25, 1);opacity:0}.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:1}@media screen and (-ms-high-contrast: active){.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:.6}}.cdk-overlay-dark-backdrop[_ngcontent-%COMP%]{background:rgba(0,0,0,.32)}.cdk-overlay-transparent-backdrop[_ngcontent-%COMP%], .cdk-overlay-transparent-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:0}.cdk-overlay-connected-position-bounding-box[_ngcontent-%COMP%]{position:absolute;z-index:1000;display:flex;flex-direction:column;min-width:1px;min-height:1px}.cdk-global-scrollblock[_ngcontent-%COMP%]{position:fixed;width:100%;overflow-y:scroll}@keyframes cdk-text-field-autofill-start{}@keyframes cdk-text-field-autofill-end{}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:-webkit-autofill{animation:cdk-text-field-autofill-start 0s 1ms}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:not(:-webkit-autofill){animation:cdk-text-field-autofill-end 0s 1ms}textarea.cdk-textarea-autosize[_ngcontent-%COMP%]{resize:none}textarea.cdk-textarea-autosize-measuring[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:auto !important;overflow:hidden !important}textarea.cdk-textarea-autosize-measuring-firefox[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:0 !important}.mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.1)}.mat-option[_ngcontent-%COMP%]{color:#212121}.mat-option[_ngcontent-%COMP%]:hover:not(.mat-option-disabled), .mat-option[_ngcontent-%COMP%]:focus:not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple):not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-active[_ngcontent-%COMP%]{background:rgba(0,0,0,.04);color:#212121}.mat-option.mat-option-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-primary[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f57c00}.mat-accent[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f44336}.mat-optgroup-label[_ngcontent-%COMP%]{color:#616161}.mat-optgroup-disabled[_ngcontent-%COMP%]   .mat-optgroup-label[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-pseudo-checkbox[_ngcontent-%COMP%]{color:#616161}.mat-pseudo-checkbox[_ngcontent-%COMP%]::after{color:#fafafa}.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{color:#b0b0b0}.mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f57c00}.mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f44336}.mat-pseudo-checkbox-checked.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{background:#b0b0b0}.mat-elevation-z0[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z1[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z2[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z3[_ngcontent-%COMP%]{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z4[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z5[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 5px 8px 0px rgba(0, 0, 0, 0.14),0px 1px 14px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z6[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z7[_ngcontent-%COMP%]{box-shadow:0px 4px 5px -2px rgba(0, 0, 0, 0.2),0px 7px 10px 1px rgba(0, 0, 0, 0.14),0px 2px 16px 1px rgba(0, 0, 0, 0.12)}.mat-elevation-z8[_ngcontent-%COMP%]{box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z9[_ngcontent-%COMP%]{box-shadow:0px 5px 6px -3px rgba(0, 0, 0, 0.2),0px 9px 12px 1px rgba(0, 0, 0, 0.14),0px 3px 16px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z10[_ngcontent-%COMP%]{box-shadow:0px 6px 6px -3px rgba(0, 0, 0, 0.2),0px 10px 14px 1px rgba(0, 0, 0, 0.14),0px 4px 18px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z11[_ngcontent-%COMP%]{box-shadow:0px 6px 7px -4px rgba(0, 0, 0, 0.2),0px 11px 15px 1px rgba(0, 0, 0, 0.14),0px 4px 20px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z12[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z13[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 13px 19px 2px rgba(0, 0, 0, 0.14),0px 5px 24px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z14[_ngcontent-%COMP%]{box-shadow:0px 7px 9px -4px rgba(0, 0, 0, 0.2),0px 14px 21px 2px rgba(0, 0, 0, 0.14),0px 5px 26px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z15[_ngcontent-%COMP%]{box-shadow:0px 8px 9px -5px rgba(0, 0, 0, 0.2),0px 15px 22px 2px rgba(0, 0, 0, 0.14),0px 6px 28px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z16[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z17[_ngcontent-%COMP%]{box-shadow:0px 8px 11px -5px rgba(0, 0, 0, 0.2),0px 17px 26px 2px rgba(0, 0, 0, 0.14),0px 6px 32px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z18[_ngcontent-%COMP%]{box-shadow:0px 9px 11px -5px rgba(0, 0, 0, 0.2),0px 18px 28px 2px rgba(0, 0, 0, 0.14),0px 7px 34px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z19[_ngcontent-%COMP%]{box-shadow:0px 9px 12px -6px rgba(0, 0, 0, 0.2),0px 19px 29px 2px rgba(0, 0, 0, 0.14),0px 7px 36px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z20[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 20px 31px 3px rgba(0, 0, 0, 0.14),0px 8px 38px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z21[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 21px 33px 3px rgba(0, 0, 0, 0.14),0px 8px 40px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z22[_ngcontent-%COMP%]{box-shadow:0px 10px 14px -6px rgba(0, 0, 0, 0.2),0px 22px 35px 3px rgba(0, 0, 0, 0.14),0px 8px 42px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z23[_ngcontent-%COMP%]{box-shadow:0px 11px 14px -7px rgba(0, 0, 0, 0.2),0px 23px 36px 3px rgba(0, 0, 0, 0.14),0px 9px 44px 8px rgba(0, 0, 0, 0.12)}.mat-elevation-z24[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12)}.mat-app-background[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-theme-loaded-marker[_ngcontent-%COMP%]{display:none}.mat-autocomplete-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-autocomplete-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover){background:#fff}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover):not(.mat-option-disabled){color:#212121}.mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f57c00}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{outline:solid 1px;border-radius:0}.mat-badge-accent[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-badge-warn[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f44336}.mat-badge[_ngcontent-%COMP%]{position:relative}.mat-badge-hidden[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{display:none}.mat-badge-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#b9b9b9;color:#757575}.mat-badge-content[_ngcontent-%COMP%]{position:absolute;text-align:center;display:inline-block;border-radius:50%;transition:transform 200ms ease-in-out;transform:scale(0.6);overflow:hidden;white-space:nowrap;text-overflow:ellipsis;pointer-events:none}.ng-animate-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%], .mat-badge-content._mat-animation-noopable[_ngcontent-%COMP%]{transition:none}.mat-badge-content.mat-badge-active[_ngcontent-%COMP%]{transform:none}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:16px;height:16px;line-height:16px}.mat-badge-small.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-8px}.mat-badge-small.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-8px}.mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-16px}.mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-16px}.mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-8px}.mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-8px}.mat-badge-medium[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:22px;height:22px;line-height:22px}.mat-badge-medium.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-11px}.mat-badge-medium.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-11px}.mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-22px}.mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-22px}.mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-11px}.mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-11px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:28px;height:28px;line-height:28px}.mat-badge-large.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-14px}.mat-badge-large.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-14px}.mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-28px}.mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-28px}.mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-14px}.mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-14px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]{color:inherit;background:transparent}.mat-button.mat-primary[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-button[disabled][disabled][_ngcontent-%COMP%], .mat-icon-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-icon-button[disabled][disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-stroked-button[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f44336}.mat-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:transparent}.mat-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{opacity:.1;background-color:currentColor}.mat-button-focus-overlay[_ngcontent-%COMP%]{background:#000}.mat-stroked-button[_ngcontent-%COMP%]:not([disabled]){border-color:rgba(0,0,0,.12)}.mat-flat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{color:#212121;background-color:#fff}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-flat-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-stroked-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-flat-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-raised-button[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone[_ngcontent-%COMP%], .mat-button-toggle-group[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{box-shadow:none}.mat-button-toggle[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-button-toggle[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121;background:#fff}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:#000}.mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12)}[dir=rtl][_ngcontent-%COMP%]   .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-group-appearance-standard.mat-button-toggle-vertical[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:none;border-top:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#616161}.mat-button-toggle-checked.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121}.mat-button-toggle-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.26);background-color:#eee}.mat-button-toggle-disabled.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{background:#fff}.mat-button-toggle-disabled.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#bdbdbd}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{border:solid 1px rgba(0,0,0,.12)}.mat-card[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-card[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-card.mat-card-flat[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-card-subtitle[_ngcontent-%COMP%]{color:#616161}.mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#616161}.mat-checkbox-checkmark[_ngcontent-%COMP%]{fill:#fafafa}.mat-checkbox-checkmark-path[_ngcontent-%COMP%]{stroke:#fafafa !important}.mat-checkbox-mixedmark[_ngcontent-%COMP%]{background-color:#fafafa}.mat-checkbox-indeterminate.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f57c00}.mat-checkbox-indeterminate.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#ff9800}.mat-checkbox-indeterminate.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f44336}.mat-checkbox-disabled.mat-checkbox-checked[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-disabled.mat-checkbox-indeterminate[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]:not(.mat-checkbox-checked)   .mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{color:#616161}.mat-checkbox[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%]{background:#f57c00}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%]{background:#ff9800}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%]{background:#f44336}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#212121}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#212121;opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled):active{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled)   .mat-chip-remove[_ngcontent-%COMP%]:hover{opacity:.54}.mat-chip.mat-standard-chip.mat-chip-disabled[_ngcontent-%COMP%]{opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]::after{background:#000}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-table[_ngcontent-%COMP%]{background:#fff}.mat-table[_ngcontent-%COMP%]   thead[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tbody[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tfoot[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], [mat-header-row][_ngcontent-%COMP%], [mat-row][_ngcontent-%COMP%], [mat-footer-row][_ngcontent-%COMP%], .mat-table-sticky[_ngcontent-%COMP%]{background:inherit}mat-row[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], th.mat-header-cell[_ngcontent-%COMP%], td.mat-cell[_ngcontent-%COMP%], td.mat-footer-cell[_ngcontent-%COMP%]{border-bottom-color:rgba(0,0,0,.12)}.mat-header-cell[_ngcontent-%COMP%]{color:#616161}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{color:#212121}.mat-calendar-arrow[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.54)}.mat-datepicker-toggle[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-next-button[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-previous-button[_ngcontent-%COMP%]{color:rgba(0,0,0,.54)}.mat-calendar-table-header[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-calendar-table-header-divider[_ngcontent-%COMP%]::after{background:rgba(0,0,0,.12)}.mat-calendar-body-label[_ngcontent-%COMP%]{color:#616161}.mat-calendar-body-cell-content[_ngcontent-%COMP%]{color:#212121;border-color:transparent}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){color:#757575}.mat-calendar-body-cell[_ngcontent-%COMP%]:not(.mat-calendar-body-disabled):hover > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-keyboard-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-program-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){background-color:rgba(0,0,0,.04)}.mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.38)}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.18)}.mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.4)}.mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12);background-color:#fff;color:#212121}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.4)}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.4)}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content-touch[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-datepicker-toggle-active[_ngcontent-%COMP%]{color:#f57c00}.mat-datepicker-toggle-active.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-datepicker-toggle-active.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-dialog-container[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-divider[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-divider-vertical[_ngcontent-%COMP%]{border-right-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-expansion-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-action-row[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-keyboard-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-program-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded)   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover:not([aria-disabled=true]){background:rgba(0,0,0,.04)}@media(hover: none){.mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded):not([aria-disabled=true])   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover{background:#fff}}.mat-expansion-panel-header-title[_ngcontent-%COMP%]{color:#212121}.mat-expansion-panel-header-description[_ngcontent-%COMP%], .mat-expansion-indicator[_ngcontent-%COMP%]::after{color:#616161}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-title[_ngcontent-%COMP%], .mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-description[_ngcontent-%COMP%]{color:inherit}.mat-form-field-label[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-hint[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-focused[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-ripple[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid)   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f57c00}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-accent   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#ff9800}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-warn   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#f44336}.mat-error[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-hint[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-legacy.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-standard[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-standard.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.04)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.02)}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:transparent}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.12)}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:rgba(0,0,0,.87)}.mat-form-field-appearance-outline.mat-focused[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field-appearance-outline.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-appearance-outline.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-invalid.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.06)}.mat-icon.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-icon.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-icon.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-type-mat-native-select[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#616161}.mat-input-element[_ngcontent-%COMP%]:disabled, .mat-form-field-type-mat-native-select.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#757575}.mat-input-element[_ngcontent-%COMP%]{caret-color:#f57c00}.mat-input-element[_ngcontent-%COMP%]::placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-moz-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-webkit-input-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]:-ms-input-placeholder{color:rgba(97,97,97,.42)}.mat-accent[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%], .mat-form-field-invalid[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#f44336}.mat-form-field-type-mat-native-select.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{color:#616161}.mat-list-item-disabled[_ngcontent-%COMP%]{background-color:#eee}.mat-list-option[_ngcontent-%COMP%]:hover, .mat-list-option[_ngcontent-%COMP%]:focus, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.04)}.mat-list-single-selected-option[_ngcontent-%COMP%], .mat-list-single-selected-option[_ngcontent-%COMP%]:hover, .mat-list-single-selected-option[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.12)}.mat-menu-panel[_ngcontent-%COMP%]{background:#fff}.mat-menu-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-menu-item[_ngcontent-%COMP%]{background:transparent;color:#212121}.mat-menu-item[disabled][_ngcontent-%COMP%], .mat-menu-item[disabled][_ngcontent-%COMP%]::after{color:rgba(0,0,0,.38)}.mat-menu-item[_ngcontent-%COMP%]   .mat-icon-no-color[_ngcontent-%COMP%], .mat-menu-item-submenu-trigger[_ngcontent-%COMP%]::after{color:rgba(0,0,0,.54)}.mat-menu-item[_ngcontent-%COMP%]:hover:not([disabled]), .mat-menu-item.cdk-program-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item.cdk-keyboard-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item-highlighted[_ngcontent-%COMP%]:not([disabled]){background:rgba(0,0,0,.04)}.mat-paginator[_ngcontent-%COMP%]{background:#fff}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{color:#616161}.mat-paginator-decrement[_ngcontent-%COMP%], .mat-paginator-increment[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54);border-right:2px solid rgba(0,0,0,.54)}.mat-paginator-first[_ngcontent-%COMP%], .mat-paginator-last[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54)}.mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-decrement[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-increment[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-first[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-last[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f57c00}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#ff9800}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f44336}.mat-progress-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f57c00}.mat-progress-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#ff9800}.mat-progress-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f44336}.mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#616161}.mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f57c00}.mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#ff9800}.mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#ff9800}.mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f44336}.mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f44336}.mat-radio-button.mat-radio-disabled.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-label-content[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-radio-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-select-value[_ngcontent-%COMP%]{color:#212121}.mat-select-placeholder[_ngcontent-%COMP%]{color:rgba(97,97,97,.42)}.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%]{color:#757575}.mat-select-arrow[_ngcontent-%COMP%]{color:#616161}.mat-select-panel[_ngcontent-%COMP%]{background:#fff}.mat-select-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-select-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple){background:rgba(0,0,0,.12)}.mat-form-field.mat-focused.mat-primary[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-invalid[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#757575}.mat-drawer-container[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-drawer[_ngcontent-%COMP%]{background-color:#fff;color:#212121}.mat-drawer.mat-drawer-push[_ngcontent-%COMP%]{background-color:#fff}.mat-drawer[_ngcontent-%COMP%]:not(.mat-drawer-side){box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-drawer-side[_ngcontent-%COMP%]{border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-backdrop.mat-drawer-shown[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.6)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.54)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.54)}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.54)}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle[_ngcontent-%COMP%]:not(.mat-checked)   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-slide-toggle-thumb[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12);background-color:#fafafa}.mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-primary[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f57c00}.mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-accent[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#ff9800}.mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-warn[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f44336}.mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.2)}.mat-slider[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%], .cdk-focused[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-disabled[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]   .mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing)   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26);background-color:transparent}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26)}.mat-slider-has-ticks[_ngcontent-%COMP%]   .mat-slider-wrapper[_ngcontent-%COMP%]::after{border-color:rgba(0,0,0,.7)}.mat-slider-horizontal[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to right, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent);background-image:-moz-repeating-linear-gradient(0.0001deg, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-slider-vertical[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to bottom, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-step-header.cdk-keyboard-focused[_ngcontent-%COMP%], .mat-step-header.cdk-program-focused[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]:hover{background-color:rgba(0,0,0,.04)}@media(hover: none){.mat-step-header[_ngcontent-%COMP%]:hover{background:none}}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-optional[_ngcontent-%COMP%]{color:#616161}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon[_ngcontent-%COMP%]{background-color:#616161;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-selected[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-done[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-edit[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-error[_ngcontent-%COMP%]{background-color:transparent;color:#f44336}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-active[_ngcontent-%COMP%]{color:#212121}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-error[_ngcontent-%COMP%]{color:#f44336}.mat-stepper-horizontal[_ngcontent-%COMP%], .mat-stepper-vertical[_ngcontent-%COMP%]{background-color:#fff}.mat-stepper-vertical-line[_ngcontent-%COMP%]::before{border-left-color:rgba(0,0,0,.12)}.mat-horizontal-stepper-header[_ngcontent-%COMP%]::before, .mat-horizontal-stepper-header[_ngcontent-%COMP%]::after, .mat-stepper-horizontal-line[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-sort-header-arrow[_ngcontent-%COMP%]{color:#616161}.mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-header[_ngcontent-%COMP%]{border-bottom:1px solid rgba(0,0,0,.12)}.mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%]{border-top:1px solid rgba(0,0,0,.12);border-bottom:none}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{color:#212121}.mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:#757575}.mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#212121}.mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#757575}.mat-tab-group[class*=mat-background-][_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar[class*=mat-background-][_ngcontent-%COMP%]{border-bottom:none;border-top:none}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-toolbar[_ngcontent-%COMP%]{background:#f5f5f5;color:#212121}.mat-toolbar.mat-primary[_ngcontent-%COMP%]{background:#f57c00;color:#fff}.mat-toolbar.mat-accent[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-toolbar.mat-warn[_ngcontent-%COMP%]{background:#f44336;color:#fff}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:currentColor}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:inherit}.mat-toolbar[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:currentColor}.mat-tooltip[_ngcontent-%COMP%]{background:rgba(97,97,97,.9)}.mat-tree[_ngcontent-%COMP%]{background:#fff}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{color:#212121}.mat-snack-bar-container[_ngcontent-%COMP%]{color:rgba(255,255,255,.7);background:#323232;box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-simple-snackbar-action[_ngcontent-%COMP%]{color:#ff9800}.group-toolbar[_ngcontent-%COMP%]{background-color:#fff;height:42px;padding:0 16px;display:flex;align-items:center;border-top:1px solid #ebebeb;border-bottom:1px solid #ebebeb}.card-group[_ngcontent-%COMP%]:first-of-type   .group-toolbar[_ngcontent-%COMP%]{border-top:none}.group-title[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.group-card-count[_ngcontent-%COMP%]{font-size:12px;font-weight:400;color:#616161;margin-left:6px}"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/views/main_view/card_groups_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const getRenderableCardIdsWithMetadata = createSelector(getNonEmptyCardIdsWithMetadata, getCurrentRouteRunSelection, (/**
     * @param {?} cardList
     * @param {?} runSelectionMap
     * @return {?}
     */
    (cardList, runSelectionMap) => {
        return cardList.filter((/**
         * @param {?} card
         * @return {?}
         */
        (card) => {
            if (!isSingleRunPlugin(card.plugin)) {
                return true;
            }
            return Boolean(runSelectionMap && runSelectionMap.get((/** @type {?} */ (card.runId))));
        }));
    }));
    class CardGroupsContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.cardGroups$ = this.store
                .select(getRenderableCardIdsWithMetadata)
                .pipe(map((/**
             * @param {?} cardList
             * @return {?}
             */
            (cardList) => groupCardIdWithMetdata(cardList))));
        }
    }
    CardGroupsContainer.decorators = [
        { type: Component, args: [{
                    selector: 'metrics-card-groups',
                    template: `
    <metrics-card-groups-component
      [cardGroups]="cardGroups$ | async"
      [cardObserver]="cardObserver"
    ></metrics-card-groups-component>
  `,
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    CardGroupsContainer.ctorParameters = () => [
        { type: Store }
    ];
    CardGroupsContainer.propDecorators = {
        cardObserver: [{ type: Input }]
    };
    /** @nocollapse */ CardGroupsContainer.ɵfac = function CardGroupsContainer_Factory(t) { return new (t || CardGroupsContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ CardGroupsContainer.ɵcmp = ɵɵdefineComponent({ type: CardGroupsContainer, selectors: [["metrics-card-groups"]], inputs: { cardObserver: "cardObserver" }, decls: 2, vars: 4, consts: [[3, "cardGroups", "cardObserver"]], template: function CardGroupsContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelement(0, "metrics-card-groups-component", 0);
            ɵɵpipe(1, "async");
        } if (rf & 2) {
            ɵɵproperty("cardGroups", ɵɵpipeBind1(1, 2, ctx.cardGroups$))("cardObserver", ctx.cardObserver);
        } }, directives: [CardGroupsComponent], pipes: [AsyncPipe], encapsulation: 2, changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/views/main_view/filtered_view_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    function FilteredViewComponent_span_4_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "span", 5);
        ɵɵtext(1);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r0 = ɵɵnextContext();
        ɵɵadvance(1);
        ɵɵtextInterpolate1("", ctx_r0.cardIdsWithMetadata.length, " cards");
    } }
    class FilteredViewComponent {
    }
    FilteredViewComponent.decorators = [
        { type: Component, args: [{
                    selector: 'metrics-filtered-view-component',
                    template: `
    <div class="group-toolbar">
      <span class="group-text">
        <span class="group-title" aria-role="heading" aria-level="3"
          >Tags matching filter</span
        >
        <span *ngIf="cardIdsWithMetadata.length > 1" class="group-card-count"
          >{{ cardIdsWithMetadata.length }} cards</span
        >
      </span>
    </div>
    <metrics-card-grid
      [cardIdsWithMetadata]="cardIdsWithMetadata"
      [cardObserver]="cardObserver"
    ></metrics-card-grid>
  `,
                    styleUrls: ['filtered_view_component.css'],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    FilteredViewComponent.propDecorators = {
        cardObserver: [{ type: Input }],
        cardIdsWithMetadata: [{ type: Input }]
    };
    /** @nocollapse */ FilteredViewComponent.ɵfac = function FilteredViewComponent_Factory(t) { return new (t || FilteredViewComponent)(); };
    /** @nocollapse */ FilteredViewComponent.ɵcmp = ɵɵdefineComponent({ type: FilteredViewComponent, selectors: [["metrics-filtered-view-component"]], inputs: { cardObserver: "cardObserver", cardIdsWithMetadata: "cardIdsWithMetadata" }, decls: 6, vars: 3, consts: [[1, "group-toolbar"], [1, "group-text"], ["aria-role", "heading", "aria-level", "3", 1, "group-title"], ["class", "group-card-count", 4, "ngIf"], [3, "cardIdsWithMetadata", "cardObserver"], [1, "group-card-count"]], template: function FilteredViewComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "div", 0);
            ɵɵelementStart(1, "span", 1);
            ɵɵelementStart(2, "span", 2);
            ɵɵtext(3, "Tags matching filter");
            ɵɵelementEnd();
            ɵɵtemplate(4, FilteredViewComponent_span_4_Template, 2, 1, "span", 3);
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelement(5, "metrics-card-grid", 4);
        } if (rf & 2) {
            ɵɵadvance(4);
            ɵɵproperty("ngIf", ctx.cardIdsWithMetadata.length > 1);
            ɵɵadvance(1);
            ɵɵproperty("cardIdsWithMetadata", ctx.cardIdsWithMetadata)("cardObserver", ctx.cardObserver);
        } }, directives: [NgIf, CardGridContainer], styles: [".mat-badge-content[_ngcontent-%COMP%]{font-weight:600;font-size:12px;font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:9px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:24px}.mat-h1[_ngcontent-%COMP%], .mat-headline[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%]{font:400 24px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h2[_ngcontent-%COMP%], .mat-title[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h3[_ngcontent-%COMP%], .mat-subheading-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%]{font:400 16px/28px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h4[_ngcontent-%COMP%], .mat-subheading-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%]{font:400 15px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h5[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%]{font:400 calc(14px * 0.83)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-h6[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:400 calc(14px * 0.67)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-body-strong[_ngcontent-%COMP%], .mat-body-2[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   p[_ngcontent-%COMP%]{margin:0 0 12px}.mat-small[_ngcontent-%COMP%], .mat-caption[_ngcontent-%COMP%]{font:400 12px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-display-4[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-4[_ngcontent-%COMP%]{font:300 112px/112px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.05em;margin:0 0 56px}.mat-display-3[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-3[_ngcontent-%COMP%]{font:400 56px/56px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.02em;margin:0 0 64px}.mat-display-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-2[_ngcontent-%COMP%]{font:400 45px/48px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.005em;margin:0 0 64px}.mat-display-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-1[_ngcontent-%COMP%]{font:400 34px/40px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 64px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%], .mat-flat-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-button-toggle[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card-title[_ngcontent-%COMP%]{font-size:24px;font-weight:500}.mat-card-header[_ngcontent-%COMP%]   .mat-card-title[_ngcontent-%COMP%]{font-size:20px}.mat-card-subtitle[_ngcontent-%COMP%], .mat-card-content[_ngcontent-%COMP%]{font-size:14px}.mat-checkbox[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-checkbox-layout[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{line-height:24px}.mat-chip[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-chip[_ngcontent-%COMP%]   .mat-chip-trailing-icon.mat-icon[_ngcontent-%COMP%], .mat-chip[_ngcontent-%COMP%]   .mat-chip-remove.mat-icon[_ngcontent-%COMP%]{font-size:18px}.mat-table[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-header-cell[_ngcontent-%COMP%]{font-size:12px;font-weight:500}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{font-size:14px}.mat-calendar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-calendar-body[_ngcontent-%COMP%]{font-size:13px}.mat-calendar-body-label[_ngcontent-%COMP%], .mat-calendar-period-button[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-calendar-table-header[_ngcontent-%COMP%]   th[_ngcontent-%COMP%]{font-size:11px;font-weight:400}.mat-dialog-title[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-expansion-panel-header[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:15px;font-weight:400}.mat-expansion-panel-content[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field[_ngcontent-%COMP%]{font-size:inherit;font-weight:400;line-height:1.125;font-family:Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.34375em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{font-size:150%;line-height:1.125}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]{height:1.5em;width:1.5em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{height:1.125em;line-height:1.125}.mat-form-field-infix[_ngcontent-%COMP%]{padding:.5em 0;border-top:.84375em solid transparent}.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34375em) scale(0.75);width:133.3333333333%}.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34374em) scale(0.75);width:133.3333433333%}.mat-form-field-label-wrapper[_ngcontent-%COMP%]{top:-0.84375em;padding-top:.84375em}.mat-form-field-label[_ngcontent-%COMP%]{top:1.34375em}.mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.34375em}.mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{font-size:75%;margin-top:.6666666667em;top:calc(100% - 1.7916666667em)}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.4375em 0}.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.001px);-ms-transform:translateY(-1.28125em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00101px);-ms-transform:translateY(-1.28124em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00102px);-ms-transform:translateY(-1.28123em) scale(0.75);width:133.3333533333%}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.28125em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{margin-top:.5416666667em;top:calc(100% - 1.6666666667em)}@media print{.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28122em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28121em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.2812em) scale(0.75)}}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.25em 0 .75em 0}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.09375em;margin-top:-0.5em}.mat-form-field-appearance-fill.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59374em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:1em 0 1em 0}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.84375em;margin-top:-0.25em}.mat-form-field-appearance-outline.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59374em) scale(0.75);width:133.3333433333%}.mat-grid-tile-header[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]{font-size:14px}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2), .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}input.mat-input-element[_ngcontent-%COMP%]{margin-top:-0.0625em}.mat-menu-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:400}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px}.mat-radio-button[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select-trigger[_ngcontent-%COMP%]{height:1.125em}.mat-slide-toggle-content[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-slider-thumb-label-text[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-stepper-vertical[_ngcontent-%COMP%], .mat-stepper-horizontal[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-step-label[_ngcontent-%COMP%]{font-size:14px;font-weight:400}.mat-step-sub-label-error[_ngcontent-%COMP%]{font-weight:normal}.mat-step-label-error[_ngcontent-%COMP%]{font-size:14px}.mat-step-label-selected[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-tab-group[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-toolbar[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0}.mat-tooltip[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:10px;padding-top:6px;padding-bottom:6px}.mat-tooltip-handset[_ngcontent-%COMP%]{font-size:14px;padding-top:8px;padding-bottom:8px}.mat-list-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:16px}.mat-optgroup-label[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-simple-snackbar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px}.mat-simple-snackbar-action[_ngcontent-%COMP%]{line-height:1;font-family:inherit;font-size:inherit;font-weight:500}.mat-tree[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{font-weight:400;font-size:14px}.mat-ripple[_ngcontent-%COMP%]{overflow:hidden;position:relative}.mat-ripple[_ngcontent-%COMP%]:not(:empty){transform:translateZ(0)}.mat-ripple.mat-ripple-unbounded[_ngcontent-%COMP%]{overflow:visible}.mat-ripple-element[_ngcontent-%COMP%]{position:absolute;border-radius:50%;pointer-events:none;transition:opacity,transform 0ms cubic-bezier(0, 0, 0.2, 1);transform:scale(0)}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{display:none}.cdk-visually-hidden[_ngcontent-%COMP%]{border:0;clip:rect(0 0 0 0);height:1px;margin:-1px;overflow:hidden;padding:0;position:absolute;width:1px;outline:0;-webkit-appearance:none;-moz-appearance:none}.cdk-overlay-container[_ngcontent-%COMP%], .cdk-global-overlay-wrapper[_ngcontent-%COMP%]{pointer-events:none;top:0;left:0;height:100%;width:100%}.cdk-overlay-container[_ngcontent-%COMP%]{position:fixed;z-index:1000}.cdk-overlay-container[_ngcontent-%COMP%]:empty{display:none}.cdk-global-overlay-wrapper[_ngcontent-%COMP%]{display:flex;position:absolute;z-index:1000}.cdk-overlay-pane[_ngcontent-%COMP%]{position:absolute;pointer-events:auto;box-sizing:border-box;z-index:1000;display:flex;max-width:100%;max-height:100%}.cdk-overlay-backdrop[_ngcontent-%COMP%]{position:absolute;top:0;bottom:0;left:0;right:0;z-index:1000;pointer-events:auto;-webkit-tap-highlight-color:transparent;transition:opacity 400ms cubic-bezier(0.25, 0.8, 0.25, 1);opacity:0}.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:1}@media screen and (-ms-high-contrast: active){.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:.6}}.cdk-overlay-dark-backdrop[_ngcontent-%COMP%]{background:rgba(0,0,0,.32)}.cdk-overlay-transparent-backdrop[_ngcontent-%COMP%], .cdk-overlay-transparent-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:0}.cdk-overlay-connected-position-bounding-box[_ngcontent-%COMP%]{position:absolute;z-index:1000;display:flex;flex-direction:column;min-width:1px;min-height:1px}.cdk-global-scrollblock[_ngcontent-%COMP%]{position:fixed;width:100%;overflow-y:scroll}@keyframes cdk-text-field-autofill-start{}@keyframes cdk-text-field-autofill-end{}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:-webkit-autofill{animation:cdk-text-field-autofill-start 0s 1ms}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:not(:-webkit-autofill){animation:cdk-text-field-autofill-end 0s 1ms}textarea.cdk-textarea-autosize[_ngcontent-%COMP%]{resize:none}textarea.cdk-textarea-autosize-measuring[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:auto !important;overflow:hidden !important}textarea.cdk-textarea-autosize-measuring-firefox[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:0 !important}.mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.1)}.mat-option[_ngcontent-%COMP%]{color:#212121}.mat-option[_ngcontent-%COMP%]:hover:not(.mat-option-disabled), .mat-option[_ngcontent-%COMP%]:focus:not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple):not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-active[_ngcontent-%COMP%]{background:rgba(0,0,0,.04);color:#212121}.mat-option.mat-option-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-primary[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f57c00}.mat-accent[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f44336}.mat-optgroup-label[_ngcontent-%COMP%]{color:#616161}.mat-optgroup-disabled[_ngcontent-%COMP%]   .mat-optgroup-label[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-pseudo-checkbox[_ngcontent-%COMP%]{color:#616161}.mat-pseudo-checkbox[_ngcontent-%COMP%]::after{color:#fafafa}.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{color:#b0b0b0}.mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f57c00}.mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f44336}.mat-pseudo-checkbox-checked.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{background:#b0b0b0}.mat-elevation-z0[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z1[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z2[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z3[_ngcontent-%COMP%]{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z4[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z5[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 5px 8px 0px rgba(0, 0, 0, 0.14),0px 1px 14px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z6[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z7[_ngcontent-%COMP%]{box-shadow:0px 4px 5px -2px rgba(0, 0, 0, 0.2),0px 7px 10px 1px rgba(0, 0, 0, 0.14),0px 2px 16px 1px rgba(0, 0, 0, 0.12)}.mat-elevation-z8[_ngcontent-%COMP%]{box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z9[_ngcontent-%COMP%]{box-shadow:0px 5px 6px -3px rgba(0, 0, 0, 0.2),0px 9px 12px 1px rgba(0, 0, 0, 0.14),0px 3px 16px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z10[_ngcontent-%COMP%]{box-shadow:0px 6px 6px -3px rgba(0, 0, 0, 0.2),0px 10px 14px 1px rgba(0, 0, 0, 0.14),0px 4px 18px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z11[_ngcontent-%COMP%]{box-shadow:0px 6px 7px -4px rgba(0, 0, 0, 0.2),0px 11px 15px 1px rgba(0, 0, 0, 0.14),0px 4px 20px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z12[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z13[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 13px 19px 2px rgba(0, 0, 0, 0.14),0px 5px 24px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z14[_ngcontent-%COMP%]{box-shadow:0px 7px 9px -4px rgba(0, 0, 0, 0.2),0px 14px 21px 2px rgba(0, 0, 0, 0.14),0px 5px 26px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z15[_ngcontent-%COMP%]{box-shadow:0px 8px 9px -5px rgba(0, 0, 0, 0.2),0px 15px 22px 2px rgba(0, 0, 0, 0.14),0px 6px 28px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z16[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z17[_ngcontent-%COMP%]{box-shadow:0px 8px 11px -5px rgba(0, 0, 0, 0.2),0px 17px 26px 2px rgba(0, 0, 0, 0.14),0px 6px 32px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z18[_ngcontent-%COMP%]{box-shadow:0px 9px 11px -5px rgba(0, 0, 0, 0.2),0px 18px 28px 2px rgba(0, 0, 0, 0.14),0px 7px 34px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z19[_ngcontent-%COMP%]{box-shadow:0px 9px 12px -6px rgba(0, 0, 0, 0.2),0px 19px 29px 2px rgba(0, 0, 0, 0.14),0px 7px 36px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z20[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 20px 31px 3px rgba(0, 0, 0, 0.14),0px 8px 38px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z21[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 21px 33px 3px rgba(0, 0, 0, 0.14),0px 8px 40px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z22[_ngcontent-%COMP%]{box-shadow:0px 10px 14px -6px rgba(0, 0, 0, 0.2),0px 22px 35px 3px rgba(0, 0, 0, 0.14),0px 8px 42px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z23[_ngcontent-%COMP%]{box-shadow:0px 11px 14px -7px rgba(0, 0, 0, 0.2),0px 23px 36px 3px rgba(0, 0, 0, 0.14),0px 9px 44px 8px rgba(0, 0, 0, 0.12)}.mat-elevation-z24[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12)}.mat-app-background[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-theme-loaded-marker[_ngcontent-%COMP%]{display:none}.mat-autocomplete-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-autocomplete-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover){background:#fff}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover):not(.mat-option-disabled){color:#212121}.mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f57c00}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{outline:solid 1px;border-radius:0}.mat-badge-accent[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-badge-warn[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f44336}.mat-badge[_ngcontent-%COMP%]{position:relative}.mat-badge-hidden[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{display:none}.mat-badge-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#b9b9b9;color:#757575}.mat-badge-content[_ngcontent-%COMP%]{position:absolute;text-align:center;display:inline-block;border-radius:50%;transition:transform 200ms ease-in-out;transform:scale(0.6);overflow:hidden;white-space:nowrap;text-overflow:ellipsis;pointer-events:none}.ng-animate-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%], .mat-badge-content._mat-animation-noopable[_ngcontent-%COMP%]{transition:none}.mat-badge-content.mat-badge-active[_ngcontent-%COMP%]{transform:none}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:16px;height:16px;line-height:16px}.mat-badge-small.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-8px}.mat-badge-small.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-8px}.mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-16px}.mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-16px}.mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-8px}.mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-8px}.mat-badge-medium[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:22px;height:22px;line-height:22px}.mat-badge-medium.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-11px}.mat-badge-medium.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-11px}.mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-22px}.mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-22px}.mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-11px}.mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-11px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:28px;height:28px;line-height:28px}.mat-badge-large.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-14px}.mat-badge-large.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-14px}.mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-28px}.mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-28px}.mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-14px}.mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-14px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]{color:inherit;background:transparent}.mat-button.mat-primary[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-button[disabled][disabled][_ngcontent-%COMP%], .mat-icon-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-icon-button[disabled][disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-stroked-button[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f44336}.mat-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:transparent}.mat-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{opacity:.1;background-color:currentColor}.mat-button-focus-overlay[_ngcontent-%COMP%]{background:#000}.mat-stroked-button[_ngcontent-%COMP%]:not([disabled]){border-color:rgba(0,0,0,.12)}.mat-flat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{color:#212121;background-color:#fff}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-flat-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-stroked-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-flat-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-raised-button[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone[_ngcontent-%COMP%], .mat-button-toggle-group[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{box-shadow:none}.mat-button-toggle[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-button-toggle[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121;background:#fff}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:#000}.mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12)}[dir=rtl][_ngcontent-%COMP%]   .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-group-appearance-standard.mat-button-toggle-vertical[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:none;border-top:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#616161}.mat-button-toggle-checked.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121}.mat-button-toggle-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.26);background-color:#eee}.mat-button-toggle-disabled.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{background:#fff}.mat-button-toggle-disabled.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#bdbdbd}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{border:solid 1px rgba(0,0,0,.12)}.mat-card[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-card[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-card.mat-card-flat[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-card-subtitle[_ngcontent-%COMP%]{color:#616161}.mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#616161}.mat-checkbox-checkmark[_ngcontent-%COMP%]{fill:#fafafa}.mat-checkbox-checkmark-path[_ngcontent-%COMP%]{stroke:#fafafa !important}.mat-checkbox-mixedmark[_ngcontent-%COMP%]{background-color:#fafafa}.mat-checkbox-indeterminate.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f57c00}.mat-checkbox-indeterminate.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#ff9800}.mat-checkbox-indeterminate.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f44336}.mat-checkbox-disabled.mat-checkbox-checked[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-disabled.mat-checkbox-indeterminate[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]:not(.mat-checkbox-checked)   .mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{color:#616161}.mat-checkbox[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%]{background:#f57c00}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%]{background:#ff9800}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%]{background:#f44336}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#212121}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#212121;opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled):active{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled)   .mat-chip-remove[_ngcontent-%COMP%]:hover{opacity:.54}.mat-chip.mat-standard-chip.mat-chip-disabled[_ngcontent-%COMP%]{opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]::after{background:#000}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-table[_ngcontent-%COMP%]{background:#fff}.mat-table[_ngcontent-%COMP%]   thead[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tbody[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tfoot[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], [mat-header-row][_ngcontent-%COMP%], [mat-row][_ngcontent-%COMP%], [mat-footer-row][_ngcontent-%COMP%], .mat-table-sticky[_ngcontent-%COMP%]{background:inherit}mat-row[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], th.mat-header-cell[_ngcontent-%COMP%], td.mat-cell[_ngcontent-%COMP%], td.mat-footer-cell[_ngcontent-%COMP%]{border-bottom-color:rgba(0,0,0,.12)}.mat-header-cell[_ngcontent-%COMP%]{color:#616161}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{color:#212121}.mat-calendar-arrow[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.54)}.mat-datepicker-toggle[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-next-button[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-previous-button[_ngcontent-%COMP%]{color:rgba(0,0,0,.54)}.mat-calendar-table-header[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-calendar-table-header-divider[_ngcontent-%COMP%]::after{background:rgba(0,0,0,.12)}.mat-calendar-body-label[_ngcontent-%COMP%]{color:#616161}.mat-calendar-body-cell-content[_ngcontent-%COMP%]{color:#212121;border-color:transparent}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){color:#757575}.mat-calendar-body-cell[_ngcontent-%COMP%]:not(.mat-calendar-body-disabled):hover > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-keyboard-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-program-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){background-color:rgba(0,0,0,.04)}.mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.38)}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.18)}.mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.4)}.mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12);background-color:#fff;color:#212121}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.4)}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.4)}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content-touch[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-datepicker-toggle-active[_ngcontent-%COMP%]{color:#f57c00}.mat-datepicker-toggle-active.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-datepicker-toggle-active.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-dialog-container[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-divider[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-divider-vertical[_ngcontent-%COMP%]{border-right-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-expansion-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-action-row[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-keyboard-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-program-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded)   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover:not([aria-disabled=true]){background:rgba(0,0,0,.04)}@media(hover: none){.mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded):not([aria-disabled=true])   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover{background:#fff}}.mat-expansion-panel-header-title[_ngcontent-%COMP%]{color:#212121}.mat-expansion-panel-header-description[_ngcontent-%COMP%], .mat-expansion-indicator[_ngcontent-%COMP%]::after{color:#616161}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-title[_ngcontent-%COMP%], .mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-description[_ngcontent-%COMP%]{color:inherit}.mat-form-field-label[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-hint[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-focused[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-ripple[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid)   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f57c00}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-accent   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#ff9800}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-warn   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#f44336}.mat-error[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-hint[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-legacy.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-standard[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-standard.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.04)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.02)}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:transparent}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.12)}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:rgba(0,0,0,.87)}.mat-form-field-appearance-outline.mat-focused[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field-appearance-outline.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-appearance-outline.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-invalid.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.06)}.mat-icon.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-icon.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-icon.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-type-mat-native-select[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#616161}.mat-input-element[_ngcontent-%COMP%]:disabled, .mat-form-field-type-mat-native-select.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#757575}.mat-input-element[_ngcontent-%COMP%]{caret-color:#f57c00}.mat-input-element[_ngcontent-%COMP%]::placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-moz-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-webkit-input-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]:-ms-input-placeholder{color:rgba(97,97,97,.42)}.mat-accent[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%], .mat-form-field-invalid[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#f44336}.mat-form-field-type-mat-native-select.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{color:#616161}.mat-list-item-disabled[_ngcontent-%COMP%]{background-color:#eee}.mat-list-option[_ngcontent-%COMP%]:hover, .mat-list-option[_ngcontent-%COMP%]:focus, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.04)}.mat-list-single-selected-option[_ngcontent-%COMP%], .mat-list-single-selected-option[_ngcontent-%COMP%]:hover, .mat-list-single-selected-option[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.12)}.mat-menu-panel[_ngcontent-%COMP%]{background:#fff}.mat-menu-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-menu-item[_ngcontent-%COMP%]{background:transparent;color:#212121}.mat-menu-item[disabled][_ngcontent-%COMP%], .mat-menu-item[disabled][_ngcontent-%COMP%]::after{color:rgba(0,0,0,.38)}.mat-menu-item[_ngcontent-%COMP%]   .mat-icon-no-color[_ngcontent-%COMP%], .mat-menu-item-submenu-trigger[_ngcontent-%COMP%]::after{color:rgba(0,0,0,.54)}.mat-menu-item[_ngcontent-%COMP%]:hover:not([disabled]), .mat-menu-item.cdk-program-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item.cdk-keyboard-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item-highlighted[_ngcontent-%COMP%]:not([disabled]){background:rgba(0,0,0,.04)}.mat-paginator[_ngcontent-%COMP%]{background:#fff}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{color:#616161}.mat-paginator-decrement[_ngcontent-%COMP%], .mat-paginator-increment[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54);border-right:2px solid rgba(0,0,0,.54)}.mat-paginator-first[_ngcontent-%COMP%], .mat-paginator-last[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54)}.mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-decrement[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-increment[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-first[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-last[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f57c00}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#ff9800}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f44336}.mat-progress-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f57c00}.mat-progress-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#ff9800}.mat-progress-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f44336}.mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#616161}.mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f57c00}.mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#ff9800}.mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#ff9800}.mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f44336}.mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f44336}.mat-radio-button.mat-radio-disabled.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-label-content[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-radio-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-select-value[_ngcontent-%COMP%]{color:#212121}.mat-select-placeholder[_ngcontent-%COMP%]{color:rgba(97,97,97,.42)}.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%]{color:#757575}.mat-select-arrow[_ngcontent-%COMP%]{color:#616161}.mat-select-panel[_ngcontent-%COMP%]{background:#fff}.mat-select-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-select-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple){background:rgba(0,0,0,.12)}.mat-form-field.mat-focused.mat-primary[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-invalid[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#757575}.mat-drawer-container[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-drawer[_ngcontent-%COMP%]{background-color:#fff;color:#212121}.mat-drawer.mat-drawer-push[_ngcontent-%COMP%]{background-color:#fff}.mat-drawer[_ngcontent-%COMP%]:not(.mat-drawer-side){box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-drawer-side[_ngcontent-%COMP%]{border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-backdrop.mat-drawer-shown[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.6)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.54)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.54)}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.54)}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle[_ngcontent-%COMP%]:not(.mat-checked)   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-slide-toggle-thumb[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12);background-color:#fafafa}.mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-primary[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f57c00}.mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-accent[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#ff9800}.mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-warn[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f44336}.mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.2)}.mat-slider[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%], .cdk-focused[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-disabled[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]   .mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing)   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26);background-color:transparent}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26)}.mat-slider-has-ticks[_ngcontent-%COMP%]   .mat-slider-wrapper[_ngcontent-%COMP%]::after{border-color:rgba(0,0,0,.7)}.mat-slider-horizontal[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to right, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent);background-image:-moz-repeating-linear-gradient(0.0001deg, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-slider-vertical[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to bottom, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-step-header.cdk-keyboard-focused[_ngcontent-%COMP%], .mat-step-header.cdk-program-focused[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]:hover{background-color:rgba(0,0,0,.04)}@media(hover: none){.mat-step-header[_ngcontent-%COMP%]:hover{background:none}}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-optional[_ngcontent-%COMP%]{color:#616161}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon[_ngcontent-%COMP%]{background-color:#616161;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-selected[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-done[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-edit[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-error[_ngcontent-%COMP%]{background-color:transparent;color:#f44336}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-active[_ngcontent-%COMP%]{color:#212121}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-error[_ngcontent-%COMP%]{color:#f44336}.mat-stepper-horizontal[_ngcontent-%COMP%], .mat-stepper-vertical[_ngcontent-%COMP%]{background-color:#fff}.mat-stepper-vertical-line[_ngcontent-%COMP%]::before{border-left-color:rgba(0,0,0,.12)}.mat-horizontal-stepper-header[_ngcontent-%COMP%]::before, .mat-horizontal-stepper-header[_ngcontent-%COMP%]::after, .mat-stepper-horizontal-line[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-sort-header-arrow[_ngcontent-%COMP%]{color:#616161}.mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-header[_ngcontent-%COMP%]{border-bottom:1px solid rgba(0,0,0,.12)}.mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%]{border-top:1px solid rgba(0,0,0,.12);border-bottom:none}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{color:#212121}.mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:#757575}.mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#212121}.mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#757575}.mat-tab-group[class*=mat-background-][_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar[class*=mat-background-][_ngcontent-%COMP%]{border-bottom:none;border-top:none}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-toolbar[_ngcontent-%COMP%]{background:#f5f5f5;color:#212121}.mat-toolbar.mat-primary[_ngcontent-%COMP%]{background:#f57c00;color:#fff}.mat-toolbar.mat-accent[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-toolbar.mat-warn[_ngcontent-%COMP%]{background:#f44336;color:#fff}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:currentColor}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:inherit}.mat-toolbar[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:currentColor}.mat-tooltip[_ngcontent-%COMP%]{background:rgba(97,97,97,.9)}.mat-tree[_ngcontent-%COMP%]{background:#fff}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{color:#212121}.mat-snack-bar-container[_ngcontent-%COMP%]{color:rgba(255,255,255,.7);background:#323232;box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-simple-snackbar-action[_ngcontent-%COMP%]{color:#ff9800}.group-toolbar[_ngcontent-%COMP%]{background-color:#fff;height:42px;padding:0 16px;display:flex;align-items:center;border-bottom:1px solid #ebebeb}.group-text[_ngcontent-%COMP%]{display:flex;align-items:baseline}.group-title[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.group-card-count[_ngcontent-%COMP%]{font-size:12px;font-weight:400;color:#616161;margin-left:6px}"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/views/main_view/filtered_view_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const getRenderableCardIdsWithMetadata$1 = createSelector(getNonEmptyCardIdsWithMetadata, getCurrentRouteRunSelection, (/**
     * @param {?} cardList
     * @param {?} runSelectionMap
     * @return {?}
     */
    (cardList, runSelectionMap) => {
        return cardList.filter((/**
         * @param {?} card
         * @return {?}
         */
        (card) => {
            if (!isSingleRunPlugin(card.plugin)) {
                return true;
            }
            return Boolean(runSelectionMap && runSelectionMap.get((/** @type {?} */ (card.runId))));
        }));
    }));
    /**
     * An area showing cards that match the tag filter.
     */
    class FilteredViewContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.cardIdsWithMetadata$ = this.store.select(getRenderableCardIdsWithMetadata$1).pipe(switchMap((/**
             * @param {?} cardList
             * @return {?}
             */
            (cardList) => {
                return combineLatest([
                    of(cardList),
                    this.store.select(getMetricsTagFilter),
                ]);
            })), map((/**
             * @param {?} __0
             * @return {?}
             */
            ([cardList, tagFilter]) => {
                try {
                    return { cardList, regex: new RegExp(tagFilter) };
                }
                catch (e) {
                    return { cardList, regex: null };
                }
            })), filter((/**
             * @param {?} __0
             * @return {?}
             */
            ({ regex }) => regex !== null)), map((/**
             * @param {?} __0
             * @return {?}
             */
            ({ cardList, regex }) => {
                return cardList.filter((/**
                 * @param {?} __0
                 * @return {?}
                 */
                ({ tag }) => (/** @type {?} */ (regex)).test(tag)));
            })), map((/**
             * @param {?} cardList
             * @return {?}
             */
            (cardList) => {
                return cardList.sort((/**
                 * @param {?} cardA
                 * @param {?} cardB
                 * @return {?}
                 */
                (cardA, cardB) => {
                    return compareTagNames(cardA.tag, cardB.tag);
                }));
            })), distinctUntilChanged((/**
             * @param {?} prev
             * @param {?} updated
             * @return {?}
             */
            (prev, updated) => {
                if (prev.length !== updated.length) {
                    return false;
                }
                return prev.every((/**
                 * @param {?} prevVal
                 * @param {?} index
                 * @return {?}
                 */
                (prevVal, index) => {
                    return prevVal.cardId === updated[index].cardId;
                }));
            })), startWith([]));
        }
    }
    FilteredViewContainer.decorators = [
        { type: Component, args: [{
                    selector: 'metrics-filtered-view',
                    template: `
    <metrics-filtered-view-component
      [cardIdsWithMetadata]="cardIdsWithMetadata$ | async"
      [cardObserver]="cardObserver"
    ></metrics-filtered-view-component>
  `,
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    FilteredViewContainer.ctorParameters = () => [
        { type: Store }
    ];
    FilteredViewContainer.propDecorators = {
        cardObserver: [{ type: Input }]
    };
    /** @nocollapse */ FilteredViewContainer.ɵfac = function FilteredViewContainer_Factory(t) { return new (t || FilteredViewContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ FilteredViewContainer.ɵcmp = ɵɵdefineComponent({ type: FilteredViewContainer, selectors: [["metrics-filtered-view"]], inputs: { cardObserver: "cardObserver" }, decls: 2, vars: 4, consts: [[3, "cardIdsWithMetadata", "cardObserver"]], template: function FilteredViewContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelement(0, "metrics-filtered-view-component", 0);
            ɵɵpipe(1, "async");
        } if (rf & 2) {
            ɵɵproperty("cardIdsWithMetadata", ɵɵpipeBind1(1, 2, ctx.cardIdsWithMetadata$))("cardObserver", ctx.cardObserver);
        } }, directives: [FilteredViewComponent], pipes: [AsyncPipe], encapsulation: 2, changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/views/main_view/pinned_view_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    function PinnedViewComponent_span_5_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "span", 7);
        ɵɵtext(1);
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r0 = ɵɵnextContext();
        ɵɵadvance(1);
        ɵɵtextInterpolate1("", ctx_r0.cardIdsWithMetadata.length, " cards");
    } }
    function PinnedViewComponent_metrics_card_grid_6_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "metrics-card-grid", 8);
    } if (rf & 2) {
        const ctx_r1 = ɵɵnextContext();
        ɵɵproperty("cardIdsWithMetadata", ctx_r1.cardIdsWithMetadata)("cardObserver", ctx_r1.cardObserver);
    } }
    function PinnedViewComponent_ng_template_7_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 9);
        ɵɵtext(1, "Pin cards for a quick view and comparison");
        ɵɵelementEnd();
    } }
    class PinnedViewComponent {
    }
    PinnedViewComponent.decorators = [
        { type: Component, args: [{
                    selector: 'metrics-pinned-view-component',
                    template: `
    <div class="group-toolbar">
      <mat-icon svgIcon="keep_24px"></mat-icon>
      <span class="group-text">
        <span class="group-title" aria-role="heading" aria-level="3"
          >Pinned</span
        >
        <span *ngIf="cardIdsWithMetadata.length > 1" class="group-card-count"
          >{{ cardIdsWithMetadata.length }} cards</span
        >
      </span>
    </div>
    <metrics-card-grid
      *ngIf="cardIdsWithMetadata.length; else emptyPinnedView"
      [cardIdsWithMetadata]="cardIdsWithMetadata"
      [cardObserver]="cardObserver"
    ></metrics-card-grid>
    <ng-template #emptyPinnedView>
      <div class="empty-message">Pin cards for a quick view and comparison</div>
    </ng-template>
  `,
                    styleUrls: ['pinned_view_component.css'],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    PinnedViewComponent.propDecorators = {
        cardObserver: [{ type: Input }],
        cardIdsWithMetadata: [{ type: Input }]
    };
    /** @nocollapse */ PinnedViewComponent.ɵfac = function PinnedViewComponent_Factory(t) { return new (t || PinnedViewComponent)(); };
    /** @nocollapse */ PinnedViewComponent.ɵcmp = ɵɵdefineComponent({ type: PinnedViewComponent, selectors: [["metrics-pinned-view-component"]], inputs: { cardObserver: "cardObserver", cardIdsWithMetadata: "cardIdsWithMetadata" }, decls: 9, vars: 3, consts: [[1, "group-toolbar"], ["svgIcon", "keep_24px"], [1, "group-text"], ["aria-role", "heading", "aria-level", "3", 1, "group-title"], ["class", "group-card-count", 4, "ngIf"], [3, "cardIdsWithMetadata", "cardObserver", 4, "ngIf", "ngIfElse"], ["emptyPinnedView", ""], [1, "group-card-count"], [3, "cardIdsWithMetadata", "cardObserver"], [1, "empty-message"]], template: function PinnedViewComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "div", 0);
            ɵɵelement(1, "mat-icon", 1);
            ɵɵelementStart(2, "span", 2);
            ɵɵelementStart(3, "span", 3);
            ɵɵtext(4, "Pinned");
            ɵɵelementEnd();
            ɵɵtemplate(5, PinnedViewComponent_span_5_Template, 2, 1, "span", 4);
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵtemplate(6, PinnedViewComponent_metrics_card_grid_6_Template, 1, 2, "metrics-card-grid", 5);
            ɵɵtemplate(7, PinnedViewComponent_ng_template_7_Template, 2, 0, "ng-template", null, 6, ɵɵtemplateRefExtractor);
        } if (rf & 2) {
            const _r2 = ɵɵreference(8);
            ɵɵadvance(5);
            ɵɵproperty("ngIf", ctx.cardIdsWithMetadata.length > 1);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx.cardIdsWithMetadata.length)("ngIfElse", _r2);
        } }, directives: [MatIcon, NgIf, CardGridContainer], styles: [".mat-badge-content[_ngcontent-%COMP%]{font-weight:600;font-size:12px;font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:9px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:24px}.mat-h1[_ngcontent-%COMP%], .mat-headline[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%]{font:400 24px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h2[_ngcontent-%COMP%], .mat-title[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h3[_ngcontent-%COMP%], .mat-subheading-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%]{font:400 16px/28px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h4[_ngcontent-%COMP%], .mat-subheading-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%]{font:400 15px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h5[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%]{font:400 calc(14px * 0.83)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-h6[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:400 calc(14px * 0.67)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-body-strong[_ngcontent-%COMP%], .mat-body-2[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   p[_ngcontent-%COMP%]{margin:0 0 12px}.mat-small[_ngcontent-%COMP%], .mat-caption[_ngcontent-%COMP%]{font:400 12px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-display-4[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-4[_ngcontent-%COMP%]{font:300 112px/112px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.05em;margin:0 0 56px}.mat-display-3[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-3[_ngcontent-%COMP%]{font:400 56px/56px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.02em;margin:0 0 64px}.mat-display-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-2[_ngcontent-%COMP%]{font:400 45px/48px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.005em;margin:0 0 64px}.mat-display-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-1[_ngcontent-%COMP%]{font:400 34px/40px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 64px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%], .mat-flat-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-button-toggle[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card-title[_ngcontent-%COMP%]{font-size:24px;font-weight:500}.mat-card-header[_ngcontent-%COMP%]   .mat-card-title[_ngcontent-%COMP%]{font-size:20px}.mat-card-subtitle[_ngcontent-%COMP%], .mat-card-content[_ngcontent-%COMP%]{font-size:14px}.mat-checkbox[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-checkbox-layout[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{line-height:24px}.mat-chip[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-chip[_ngcontent-%COMP%]   .mat-chip-trailing-icon.mat-icon[_ngcontent-%COMP%], .mat-chip[_ngcontent-%COMP%]   .mat-chip-remove.mat-icon[_ngcontent-%COMP%]{font-size:18px}.mat-table[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-header-cell[_ngcontent-%COMP%]{font-size:12px;font-weight:500}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{font-size:14px}.mat-calendar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-calendar-body[_ngcontent-%COMP%]{font-size:13px}.mat-calendar-body-label[_ngcontent-%COMP%], .mat-calendar-period-button[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-calendar-table-header[_ngcontent-%COMP%]   th[_ngcontent-%COMP%]{font-size:11px;font-weight:400}.mat-dialog-title[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-expansion-panel-header[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:15px;font-weight:400}.mat-expansion-panel-content[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field[_ngcontent-%COMP%]{font-size:inherit;font-weight:400;line-height:1.125;font-family:Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.34375em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{font-size:150%;line-height:1.125}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]{height:1.5em;width:1.5em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{height:1.125em;line-height:1.125}.mat-form-field-infix[_ngcontent-%COMP%]{padding:.5em 0;border-top:.84375em solid transparent}.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34375em) scale(0.75);width:133.3333333333%}.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34374em) scale(0.75);width:133.3333433333%}.mat-form-field-label-wrapper[_ngcontent-%COMP%]{top:-0.84375em;padding-top:.84375em}.mat-form-field-label[_ngcontent-%COMP%]{top:1.34375em}.mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.34375em}.mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{font-size:75%;margin-top:.6666666667em;top:calc(100% - 1.7916666667em)}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.4375em 0}.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.001px);-ms-transform:translateY(-1.28125em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00101px);-ms-transform:translateY(-1.28124em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00102px);-ms-transform:translateY(-1.28123em) scale(0.75);width:133.3333533333%}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.28125em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{margin-top:.5416666667em;top:calc(100% - 1.6666666667em)}@media print{.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28122em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28121em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.2812em) scale(0.75)}}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.25em 0 .75em 0}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.09375em;margin-top:-0.5em}.mat-form-field-appearance-fill.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59374em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:1em 0 1em 0}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.84375em;margin-top:-0.25em}.mat-form-field-appearance-outline.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59374em) scale(0.75);width:133.3333433333%}.mat-grid-tile-header[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]{font-size:14px}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2), .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}input.mat-input-element[_ngcontent-%COMP%]{margin-top:-0.0625em}.mat-menu-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:400}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px}.mat-radio-button[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select-trigger[_ngcontent-%COMP%]{height:1.125em}.mat-slide-toggle-content[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-slider-thumb-label-text[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-stepper-vertical[_ngcontent-%COMP%], .mat-stepper-horizontal[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-step-label[_ngcontent-%COMP%]{font-size:14px;font-weight:400}.mat-step-sub-label-error[_ngcontent-%COMP%]{font-weight:normal}.mat-step-label-error[_ngcontent-%COMP%]{font-size:14px}.mat-step-label-selected[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-tab-group[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-toolbar[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0}.mat-tooltip[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:10px;padding-top:6px;padding-bottom:6px}.mat-tooltip-handset[_ngcontent-%COMP%]{font-size:14px;padding-top:8px;padding-bottom:8px}.mat-list-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:16px}.mat-optgroup-label[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-simple-snackbar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px}.mat-simple-snackbar-action[_ngcontent-%COMP%]{line-height:1;font-family:inherit;font-size:inherit;font-weight:500}.mat-tree[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{font-weight:400;font-size:14px}.mat-ripple[_ngcontent-%COMP%]{overflow:hidden;position:relative}.mat-ripple[_ngcontent-%COMP%]:not(:empty){transform:translateZ(0)}.mat-ripple.mat-ripple-unbounded[_ngcontent-%COMP%]{overflow:visible}.mat-ripple-element[_ngcontent-%COMP%]{position:absolute;border-radius:50%;pointer-events:none;transition:opacity,transform 0ms cubic-bezier(0, 0, 0.2, 1);transform:scale(0)}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{display:none}.cdk-visually-hidden[_ngcontent-%COMP%]{border:0;clip:rect(0 0 0 0);height:1px;margin:-1px;overflow:hidden;padding:0;position:absolute;width:1px;outline:0;-webkit-appearance:none;-moz-appearance:none}.cdk-overlay-container[_ngcontent-%COMP%], .cdk-global-overlay-wrapper[_ngcontent-%COMP%]{pointer-events:none;top:0;left:0;height:100%;width:100%}.cdk-overlay-container[_ngcontent-%COMP%]{position:fixed;z-index:1000}.cdk-overlay-container[_ngcontent-%COMP%]:empty{display:none}.cdk-global-overlay-wrapper[_ngcontent-%COMP%]{display:flex;position:absolute;z-index:1000}.cdk-overlay-pane[_ngcontent-%COMP%]{position:absolute;pointer-events:auto;box-sizing:border-box;z-index:1000;display:flex;max-width:100%;max-height:100%}.cdk-overlay-backdrop[_ngcontent-%COMP%]{position:absolute;top:0;bottom:0;left:0;right:0;z-index:1000;pointer-events:auto;-webkit-tap-highlight-color:transparent;transition:opacity 400ms cubic-bezier(0.25, 0.8, 0.25, 1);opacity:0}.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:1}@media screen and (-ms-high-contrast: active){.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:.6}}.cdk-overlay-dark-backdrop[_ngcontent-%COMP%]{background:rgba(0,0,0,.32)}.cdk-overlay-transparent-backdrop[_ngcontent-%COMP%], .cdk-overlay-transparent-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:0}.cdk-overlay-connected-position-bounding-box[_ngcontent-%COMP%]{position:absolute;z-index:1000;display:flex;flex-direction:column;min-width:1px;min-height:1px}.cdk-global-scrollblock[_ngcontent-%COMP%]{position:fixed;width:100%;overflow-y:scroll}@keyframes cdk-text-field-autofill-start{}@keyframes cdk-text-field-autofill-end{}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:-webkit-autofill{animation:cdk-text-field-autofill-start 0s 1ms}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:not(:-webkit-autofill){animation:cdk-text-field-autofill-end 0s 1ms}textarea.cdk-textarea-autosize[_ngcontent-%COMP%]{resize:none}textarea.cdk-textarea-autosize-measuring[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:auto !important;overflow:hidden !important}textarea.cdk-textarea-autosize-measuring-firefox[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:0 !important}.mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.1)}.mat-option[_ngcontent-%COMP%]{color:#212121}.mat-option[_ngcontent-%COMP%]:hover:not(.mat-option-disabled), .mat-option[_ngcontent-%COMP%]:focus:not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple):not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-active[_ngcontent-%COMP%]{background:rgba(0,0,0,.04);color:#212121}.mat-option.mat-option-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-primary[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f57c00}.mat-accent[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f44336}.mat-optgroup-label[_ngcontent-%COMP%]{color:#616161}.mat-optgroup-disabled[_ngcontent-%COMP%]   .mat-optgroup-label[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-pseudo-checkbox[_ngcontent-%COMP%]{color:#616161}.mat-pseudo-checkbox[_ngcontent-%COMP%]::after{color:#fafafa}.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{color:#b0b0b0}.mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f57c00}.mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f44336}.mat-pseudo-checkbox-checked.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{background:#b0b0b0}.mat-elevation-z0[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z1[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z2[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z3[_ngcontent-%COMP%]{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z4[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z5[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 5px 8px 0px rgba(0, 0, 0, 0.14),0px 1px 14px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z6[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z7[_ngcontent-%COMP%]{box-shadow:0px 4px 5px -2px rgba(0, 0, 0, 0.2),0px 7px 10px 1px rgba(0, 0, 0, 0.14),0px 2px 16px 1px rgba(0, 0, 0, 0.12)}.mat-elevation-z8[_ngcontent-%COMP%]{box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z9[_ngcontent-%COMP%]{box-shadow:0px 5px 6px -3px rgba(0, 0, 0, 0.2),0px 9px 12px 1px rgba(0, 0, 0, 0.14),0px 3px 16px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z10[_ngcontent-%COMP%]{box-shadow:0px 6px 6px -3px rgba(0, 0, 0, 0.2),0px 10px 14px 1px rgba(0, 0, 0, 0.14),0px 4px 18px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z11[_ngcontent-%COMP%]{box-shadow:0px 6px 7px -4px rgba(0, 0, 0, 0.2),0px 11px 15px 1px rgba(0, 0, 0, 0.14),0px 4px 20px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z12[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z13[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 13px 19px 2px rgba(0, 0, 0, 0.14),0px 5px 24px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z14[_ngcontent-%COMP%]{box-shadow:0px 7px 9px -4px rgba(0, 0, 0, 0.2),0px 14px 21px 2px rgba(0, 0, 0, 0.14),0px 5px 26px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z15[_ngcontent-%COMP%]{box-shadow:0px 8px 9px -5px rgba(0, 0, 0, 0.2),0px 15px 22px 2px rgba(0, 0, 0, 0.14),0px 6px 28px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z16[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z17[_ngcontent-%COMP%]{box-shadow:0px 8px 11px -5px rgba(0, 0, 0, 0.2),0px 17px 26px 2px rgba(0, 0, 0, 0.14),0px 6px 32px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z18[_ngcontent-%COMP%]{box-shadow:0px 9px 11px -5px rgba(0, 0, 0, 0.2),0px 18px 28px 2px rgba(0, 0, 0, 0.14),0px 7px 34px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z19[_ngcontent-%COMP%]{box-shadow:0px 9px 12px -6px rgba(0, 0, 0, 0.2),0px 19px 29px 2px rgba(0, 0, 0, 0.14),0px 7px 36px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z20[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 20px 31px 3px rgba(0, 0, 0, 0.14),0px 8px 38px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z21[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 21px 33px 3px rgba(0, 0, 0, 0.14),0px 8px 40px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z22[_ngcontent-%COMP%]{box-shadow:0px 10px 14px -6px rgba(0, 0, 0, 0.2),0px 22px 35px 3px rgba(0, 0, 0, 0.14),0px 8px 42px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z23[_ngcontent-%COMP%]{box-shadow:0px 11px 14px -7px rgba(0, 0, 0, 0.2),0px 23px 36px 3px rgba(0, 0, 0, 0.14),0px 9px 44px 8px rgba(0, 0, 0, 0.12)}.mat-elevation-z24[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12)}.mat-app-background[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-theme-loaded-marker[_ngcontent-%COMP%]{display:none}.mat-autocomplete-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-autocomplete-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover){background:#fff}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover):not(.mat-option-disabled){color:#212121}.mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f57c00}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{outline:solid 1px;border-radius:0}.mat-badge-accent[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-badge-warn[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f44336}.mat-badge[_ngcontent-%COMP%]{position:relative}.mat-badge-hidden[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{display:none}.mat-badge-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#b9b9b9;color:#757575}.mat-badge-content[_ngcontent-%COMP%]{position:absolute;text-align:center;display:inline-block;border-radius:50%;transition:transform 200ms ease-in-out;transform:scale(0.6);overflow:hidden;white-space:nowrap;text-overflow:ellipsis;pointer-events:none}.ng-animate-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%], .mat-badge-content._mat-animation-noopable[_ngcontent-%COMP%]{transition:none}.mat-badge-content.mat-badge-active[_ngcontent-%COMP%]{transform:none}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:16px;height:16px;line-height:16px}.mat-badge-small.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-8px}.mat-badge-small.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-8px}.mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-16px}.mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-16px}.mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-8px}.mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-8px}.mat-badge-medium[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:22px;height:22px;line-height:22px}.mat-badge-medium.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-11px}.mat-badge-medium.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-11px}.mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-22px}.mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-22px}.mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-11px}.mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-11px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:28px;height:28px;line-height:28px}.mat-badge-large.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-14px}.mat-badge-large.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-14px}.mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-28px}.mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-28px}.mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-14px}.mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-14px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]{color:inherit;background:transparent}.mat-button.mat-primary[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-button[disabled][disabled][_ngcontent-%COMP%], .mat-icon-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-icon-button[disabled][disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-stroked-button[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f44336}.mat-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:transparent}.mat-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{opacity:.1;background-color:currentColor}.mat-button-focus-overlay[_ngcontent-%COMP%]{background:#000}.mat-stroked-button[_ngcontent-%COMP%]:not([disabled]){border-color:rgba(0,0,0,.12)}.mat-flat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{color:#212121;background-color:#fff}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-flat-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-stroked-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-flat-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-raised-button[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone[_ngcontent-%COMP%], .mat-button-toggle-group[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{box-shadow:none}.mat-button-toggle[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-button-toggle[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121;background:#fff}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:#000}.mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12)}[dir=rtl][_ngcontent-%COMP%]   .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-group-appearance-standard.mat-button-toggle-vertical[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:none;border-top:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#616161}.mat-button-toggle-checked.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121}.mat-button-toggle-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.26);background-color:#eee}.mat-button-toggle-disabled.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{background:#fff}.mat-button-toggle-disabled.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#bdbdbd}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{border:solid 1px rgba(0,0,0,.12)}.mat-card[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-card[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-card.mat-card-flat[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-card-subtitle[_ngcontent-%COMP%]{color:#616161}.mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#616161}.mat-checkbox-checkmark[_ngcontent-%COMP%]{fill:#fafafa}.mat-checkbox-checkmark-path[_ngcontent-%COMP%]{stroke:#fafafa !important}.mat-checkbox-mixedmark[_ngcontent-%COMP%]{background-color:#fafafa}.mat-checkbox-indeterminate.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f57c00}.mat-checkbox-indeterminate.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#ff9800}.mat-checkbox-indeterminate.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f44336}.mat-checkbox-disabled.mat-checkbox-checked[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-disabled.mat-checkbox-indeterminate[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]:not(.mat-checkbox-checked)   .mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{color:#616161}.mat-checkbox[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%]{background:#f57c00}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%]{background:#ff9800}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%]{background:#f44336}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#212121}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#212121;opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled):active{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled)   .mat-chip-remove[_ngcontent-%COMP%]:hover{opacity:.54}.mat-chip.mat-standard-chip.mat-chip-disabled[_ngcontent-%COMP%]{opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]::after{background:#000}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-table[_ngcontent-%COMP%]{background:#fff}.mat-table[_ngcontent-%COMP%]   thead[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tbody[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tfoot[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], [mat-header-row][_ngcontent-%COMP%], [mat-row][_ngcontent-%COMP%], [mat-footer-row][_ngcontent-%COMP%], .mat-table-sticky[_ngcontent-%COMP%]{background:inherit}mat-row[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], th.mat-header-cell[_ngcontent-%COMP%], td.mat-cell[_ngcontent-%COMP%], td.mat-footer-cell[_ngcontent-%COMP%]{border-bottom-color:rgba(0,0,0,.12)}.mat-header-cell[_ngcontent-%COMP%]{color:#616161}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{color:#212121}.mat-calendar-arrow[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.54)}.mat-datepicker-toggle[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-next-button[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-previous-button[_ngcontent-%COMP%]{color:rgba(0,0,0,.54)}.mat-calendar-table-header[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-calendar-table-header-divider[_ngcontent-%COMP%]::after{background:rgba(0,0,0,.12)}.mat-calendar-body-label[_ngcontent-%COMP%]{color:#616161}.mat-calendar-body-cell-content[_ngcontent-%COMP%]{color:#212121;border-color:transparent}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){color:#757575}.mat-calendar-body-cell[_ngcontent-%COMP%]:not(.mat-calendar-body-disabled):hover > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-keyboard-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-program-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){background-color:rgba(0,0,0,.04)}.mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.38)}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.18)}.mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.4)}.mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12);background-color:#fff;color:#212121}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.4)}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.4)}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content-touch[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-datepicker-toggle-active[_ngcontent-%COMP%]{color:#f57c00}.mat-datepicker-toggle-active.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-datepicker-toggle-active.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-dialog-container[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-divider[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-divider-vertical[_ngcontent-%COMP%]{border-right-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-expansion-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-action-row[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-keyboard-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-program-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded)   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover:not([aria-disabled=true]){background:rgba(0,0,0,.04)}@media(hover: none){.mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded):not([aria-disabled=true])   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover{background:#fff}}.mat-expansion-panel-header-title[_ngcontent-%COMP%]{color:#212121}.mat-expansion-panel-header-description[_ngcontent-%COMP%], .mat-expansion-indicator[_ngcontent-%COMP%]::after{color:#616161}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-title[_ngcontent-%COMP%], .mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-description[_ngcontent-%COMP%]{color:inherit}.mat-form-field-label[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-hint[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-focused[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-ripple[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid)   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f57c00}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-accent   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#ff9800}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-warn   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#f44336}.mat-error[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-hint[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-legacy.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-standard[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-standard.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.04)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.02)}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:transparent}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.12)}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:rgba(0,0,0,.87)}.mat-form-field-appearance-outline.mat-focused[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field-appearance-outline.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-appearance-outline.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-invalid.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.06)}.mat-icon.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-icon.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-icon.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-type-mat-native-select[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#616161}.mat-input-element[_ngcontent-%COMP%]:disabled, .mat-form-field-type-mat-native-select.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#757575}.mat-input-element[_ngcontent-%COMP%]{caret-color:#f57c00}.mat-input-element[_ngcontent-%COMP%]::placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-moz-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-webkit-input-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]:-ms-input-placeholder{color:rgba(97,97,97,.42)}.mat-accent[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%], .mat-form-field-invalid[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#f44336}.mat-form-field-type-mat-native-select.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{color:#616161}.mat-list-item-disabled[_ngcontent-%COMP%]{background-color:#eee}.mat-list-option[_ngcontent-%COMP%]:hover, .mat-list-option[_ngcontent-%COMP%]:focus, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.04)}.mat-list-single-selected-option[_ngcontent-%COMP%], .mat-list-single-selected-option[_ngcontent-%COMP%]:hover, .mat-list-single-selected-option[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.12)}.mat-menu-panel[_ngcontent-%COMP%]{background:#fff}.mat-menu-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-menu-item[_ngcontent-%COMP%]{background:transparent;color:#212121}.mat-menu-item[disabled][_ngcontent-%COMP%], .mat-menu-item[disabled][_ngcontent-%COMP%]::after{color:rgba(0,0,0,.38)}.mat-menu-item[_ngcontent-%COMP%]   .mat-icon-no-color[_ngcontent-%COMP%], .mat-menu-item-submenu-trigger[_ngcontent-%COMP%]::after{color:rgba(0,0,0,.54)}.mat-menu-item[_ngcontent-%COMP%]:hover:not([disabled]), .mat-menu-item.cdk-program-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item.cdk-keyboard-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item-highlighted[_ngcontent-%COMP%]:not([disabled]){background:rgba(0,0,0,.04)}.mat-paginator[_ngcontent-%COMP%]{background:#fff}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{color:#616161}.mat-paginator-decrement[_ngcontent-%COMP%], .mat-paginator-increment[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54);border-right:2px solid rgba(0,0,0,.54)}.mat-paginator-first[_ngcontent-%COMP%], .mat-paginator-last[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54)}.mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-decrement[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-increment[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-first[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-last[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f57c00}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#ff9800}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f44336}.mat-progress-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f57c00}.mat-progress-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#ff9800}.mat-progress-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f44336}.mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#616161}.mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f57c00}.mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#ff9800}.mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#ff9800}.mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f44336}.mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f44336}.mat-radio-button.mat-radio-disabled.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-label-content[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-radio-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-select-value[_ngcontent-%COMP%]{color:#212121}.mat-select-placeholder[_ngcontent-%COMP%]{color:rgba(97,97,97,.42)}.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%]{color:#757575}.mat-select-arrow[_ngcontent-%COMP%]{color:#616161}.mat-select-panel[_ngcontent-%COMP%]{background:#fff}.mat-select-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-select-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple){background:rgba(0,0,0,.12)}.mat-form-field.mat-focused.mat-primary[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-invalid[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#757575}.mat-drawer-container[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-drawer[_ngcontent-%COMP%]{background-color:#fff;color:#212121}.mat-drawer.mat-drawer-push[_ngcontent-%COMP%]{background-color:#fff}.mat-drawer[_ngcontent-%COMP%]:not(.mat-drawer-side){box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-drawer-side[_ngcontent-%COMP%]{border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-backdrop.mat-drawer-shown[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.6)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.54)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.54)}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.54)}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle[_ngcontent-%COMP%]:not(.mat-checked)   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-slide-toggle-thumb[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12);background-color:#fafafa}.mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-primary[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f57c00}.mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-accent[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#ff9800}.mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-warn[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f44336}.mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.2)}.mat-slider[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%], .cdk-focused[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-disabled[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]   .mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing)   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26);background-color:transparent}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26)}.mat-slider-has-ticks[_ngcontent-%COMP%]   .mat-slider-wrapper[_ngcontent-%COMP%]::after{border-color:rgba(0,0,0,.7)}.mat-slider-horizontal[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to right, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent);background-image:-moz-repeating-linear-gradient(0.0001deg, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-slider-vertical[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to bottom, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-step-header.cdk-keyboard-focused[_ngcontent-%COMP%], .mat-step-header.cdk-program-focused[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]:hover{background-color:rgba(0,0,0,.04)}@media(hover: none){.mat-step-header[_ngcontent-%COMP%]:hover{background:none}}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-optional[_ngcontent-%COMP%]{color:#616161}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon[_ngcontent-%COMP%]{background-color:#616161;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-selected[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-done[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-edit[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-error[_ngcontent-%COMP%]{background-color:transparent;color:#f44336}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-active[_ngcontent-%COMP%]{color:#212121}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-error[_ngcontent-%COMP%]{color:#f44336}.mat-stepper-horizontal[_ngcontent-%COMP%], .mat-stepper-vertical[_ngcontent-%COMP%]{background-color:#fff}.mat-stepper-vertical-line[_ngcontent-%COMP%]::before{border-left-color:rgba(0,0,0,.12)}.mat-horizontal-stepper-header[_ngcontent-%COMP%]::before, .mat-horizontal-stepper-header[_ngcontent-%COMP%]::after, .mat-stepper-horizontal-line[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-sort-header-arrow[_ngcontent-%COMP%]{color:#616161}.mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-header[_ngcontent-%COMP%]{border-bottom:1px solid rgba(0,0,0,.12)}.mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%]{border-top:1px solid rgba(0,0,0,.12);border-bottom:none}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{color:#212121}.mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:#757575}.mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#212121}.mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#757575}.mat-tab-group[class*=mat-background-][_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar[class*=mat-background-][_ngcontent-%COMP%]{border-bottom:none;border-top:none}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-toolbar[_ngcontent-%COMP%]{background:#f5f5f5;color:#212121}.mat-toolbar.mat-primary[_ngcontent-%COMP%]{background:#f57c00;color:#fff}.mat-toolbar.mat-accent[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-toolbar.mat-warn[_ngcontent-%COMP%]{background:#f44336;color:#fff}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:currentColor}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:inherit}.mat-toolbar[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:currentColor}.mat-tooltip[_ngcontent-%COMP%]{background:rgba(97,97,97,.9)}.mat-tree[_ngcontent-%COMP%]{background:#fff}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{color:#212121}.mat-snack-bar-container[_ngcontent-%COMP%]{color:rgba(255,255,255,.7);background:#323232;box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-simple-snackbar-action[_ngcontent-%COMP%]{color:#ff9800}.group-toolbar[_ngcontent-%COMP%]{background-color:#fff;height:42px;padding:0 16px;display:flex;align-items:center;border-bottom:1px solid #ebebeb}mat-icon[_ngcontent-%COMP%]{color:#616161;flex:none;margin-right:5px}.group-text[_ngcontent-%COMP%]{display:flex;align-items:baseline}.group-title[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.group-card-count[_ngcontent-%COMP%]{font-size:12px;font-weight:400;color:#616161;margin-left:6px}.empty-message[_ngcontent-%COMP%]{color:#616161;text-align:center;padding:16px;font-size:13px;font-style:italic}"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/views/main_view/pinned_view_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class PinnedViewContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.cardIdsWithMetadata$ = this.store.select(getPinnedCardsWithMetadata).pipe(startWith([]));
        }
    }
    PinnedViewContainer.decorators = [
        { type: Component, args: [{
                    selector: 'metrics-pinned-view',
                    template: `
    <metrics-pinned-view-component
      [cardIdsWithMetadata]="cardIdsWithMetadata$ | async"
      [cardObserver]="cardObserver"
    ></metrics-pinned-view-component>
  `,
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    PinnedViewContainer.ctorParameters = () => [
        { type: Store }
    ];
    PinnedViewContainer.propDecorators = {
        cardObserver: [{ type: Input }]
    };
    /** @nocollapse */ PinnedViewContainer.ɵfac = function PinnedViewContainer_Factory(t) { return new (t || PinnedViewContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ PinnedViewContainer.ɵcmp = ɵɵdefineComponent({ type: PinnedViewContainer, selectors: [["metrics-pinned-view"]], inputs: { cardObserver: "cardObserver" }, decls: 2, vars: 4, consts: [[3, "cardIdsWithMetadata", "cardObserver"]], template: function PinnedViewContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelement(0, "metrics-pinned-view-component", 0);
            ɵɵpipe(1, "async");
        } if (rf & 2) {
            ɵɵproperty("cardIdsWithMetadata", ɵɵpipeBind1(1, 2, ctx.cardIdsWithMetadata$))("cardObserver", ctx.cardObserver);
        } }, directives: [PinnedViewComponent], pipes: [AsyncPipe], encapsulation: 2, changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/widgets/dropdown/dropdown_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    function DropdownComponent_mat_option_1_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "mat-option", 2);
        ɵɵtext(1);
        ɵɵelementEnd();
    } if (rf & 2) {
        const option_r1 = ctx.$implicit;
        ɵɵproperty("value", option_r1.value);
        ɵɵadvance(1);
        ɵɵtextInterpolate1(" ", option_r1.displayText, " ");
    } }
    class DropdownComponent {
        constructor() {
            this.value = '';
            this.options = [];
            this.selectionChange = new EventEmitter();
        }
    }
    DropdownComponent.decorators = [
        { type: Component, args: [{
                    selector: 'tb-dropdown',
                    template: `
    <mat-select
      [value]="value"
      (selectionChange)="selectionChange.emit($event.value)"
    >
      <mat-option *ngFor="let option of options" [value]="option.value">
        {{ option.displayText }}
      </mat-option>
    </mat-select>
  `,
                    styleUrls: [`dropdown_component.css`],
                },] },
    ];
    DropdownComponent.propDecorators = {
        value: [{ type: Input }],
        options: [{ type: Input }],
        selectionChange: [{ type: Output }]
    };
    /** @nocollapse */ DropdownComponent.ɵfac = function DropdownComponent_Factory(t) { return new (t || DropdownComponent)(); };
    /** @nocollapse */ DropdownComponent.ɵcmp = ɵɵdefineComponent({ type: DropdownComponent, selectors: [["tb-dropdown"]], inputs: { value: "value", options: "options" }, outputs: { selectionChange: "selectionChange" }, decls: 2, vars: 2, consts: [[3, "value", "selectionChange"], [3, "value", 4, "ngFor", "ngForOf"], [3, "value"]], template: function DropdownComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "mat-select", 0);
            ɵɵlistener("selectionChange", function DropdownComponent_Template_mat_select_selectionChange_0_listener($event) { return ctx.selectionChange.emit($event.value); });
            ɵɵtemplate(1, DropdownComponent_mat_option_1_Template, 2, 2, "mat-option", 1);
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵproperty("value", ctx.value);
            ɵɵadvance(1);
            ɵɵproperty("ngForOf", ctx.options);
        } }, directives: [MatSelect, NgForOf, MatOption], styles: [".mat-badge-content[_ngcontent-%COMP%]{font-weight:600;font-size:12px;font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:9px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:24px}.mat-h1[_ngcontent-%COMP%], .mat-headline[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%]{font:400 24px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h2[_ngcontent-%COMP%], .mat-title[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h3[_ngcontent-%COMP%], .mat-subheading-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%]{font:400 16px/28px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h4[_ngcontent-%COMP%], .mat-subheading-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%]{font:400 15px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h5[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%]{font:400 calc(14px * 0.83)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-h6[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:400 calc(14px * 0.67)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-body-strong[_ngcontent-%COMP%], .mat-body-2[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   p[_ngcontent-%COMP%]{margin:0 0 12px}.mat-small[_ngcontent-%COMP%], .mat-caption[_ngcontent-%COMP%]{font:400 12px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-display-4[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-4[_ngcontent-%COMP%]{font:300 112px/112px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.05em;margin:0 0 56px}.mat-display-3[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-3[_ngcontent-%COMP%]{font:400 56px/56px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.02em;margin:0 0 64px}.mat-display-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-2[_ngcontent-%COMP%]{font:400 45px/48px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.005em;margin:0 0 64px}.mat-display-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-1[_ngcontent-%COMP%]{font:400 34px/40px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 64px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%], .mat-flat-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-button-toggle[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card-title[_ngcontent-%COMP%]{font-size:24px;font-weight:500}.mat-card-header[_ngcontent-%COMP%]   .mat-card-title[_ngcontent-%COMP%]{font-size:20px}.mat-card-subtitle[_ngcontent-%COMP%], .mat-card-content[_ngcontent-%COMP%]{font-size:14px}.mat-checkbox[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-checkbox-layout[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{line-height:24px}.mat-chip[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-chip[_ngcontent-%COMP%]   .mat-chip-trailing-icon.mat-icon[_ngcontent-%COMP%], .mat-chip[_ngcontent-%COMP%]   .mat-chip-remove.mat-icon[_ngcontent-%COMP%]{font-size:18px}.mat-table[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-header-cell[_ngcontent-%COMP%]{font-size:12px;font-weight:500}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{font-size:14px}.mat-calendar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-calendar-body[_ngcontent-%COMP%]{font-size:13px}.mat-calendar-body-label[_ngcontent-%COMP%], .mat-calendar-period-button[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-calendar-table-header[_ngcontent-%COMP%]   th[_ngcontent-%COMP%]{font-size:11px;font-weight:400}.mat-dialog-title[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-expansion-panel-header[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:15px;font-weight:400}.mat-expansion-panel-content[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field[_ngcontent-%COMP%]{font-size:inherit;font-weight:400;line-height:1.125;font-family:Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.34375em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{font-size:150%;line-height:1.125}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]{height:1.5em;width:1.5em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{height:1.125em;line-height:1.125}.mat-form-field-infix[_ngcontent-%COMP%]{padding:.5em 0;border-top:.84375em solid transparent}.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34375em) scale(0.75);width:133.3333333333%}.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34374em) scale(0.75);width:133.3333433333%}.mat-form-field-label-wrapper[_ngcontent-%COMP%]{top:-0.84375em;padding-top:.84375em}.mat-form-field-label[_ngcontent-%COMP%]{top:1.34375em}.mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.34375em}.mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{font-size:75%;margin-top:.6666666667em;top:calc(100% - 1.7916666667em)}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.4375em 0}.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.001px);-ms-transform:translateY(-1.28125em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00101px);-ms-transform:translateY(-1.28124em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00102px);-ms-transform:translateY(-1.28123em) scale(0.75);width:133.3333533333%}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.28125em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{margin-top:.5416666667em;top:calc(100% - 1.6666666667em)}@media print{.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28122em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28121em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.2812em) scale(0.75)}}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.25em 0 .75em 0}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.09375em;margin-top:-0.5em}.mat-form-field-appearance-fill.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59374em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:1em 0 1em 0}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.84375em;margin-top:-0.25em}.mat-form-field-appearance-outline.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59374em) scale(0.75);width:133.3333433333%}.mat-grid-tile-header[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]{font-size:14px}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2), .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}input.mat-input-element[_ngcontent-%COMP%]{margin-top:-0.0625em}.mat-menu-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:400}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px}.mat-radio-button[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select-trigger[_ngcontent-%COMP%]{height:1.125em}.mat-slide-toggle-content[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-slider-thumb-label-text[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-stepper-vertical[_ngcontent-%COMP%], .mat-stepper-horizontal[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-step-label[_ngcontent-%COMP%]{font-size:14px;font-weight:400}.mat-step-sub-label-error[_ngcontent-%COMP%]{font-weight:normal}.mat-step-label-error[_ngcontent-%COMP%]{font-size:14px}.mat-step-label-selected[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-tab-group[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-toolbar[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0}.mat-tooltip[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:10px;padding-top:6px;padding-bottom:6px}.mat-tooltip-handset[_ngcontent-%COMP%]{font-size:14px;padding-top:8px;padding-bottom:8px}.mat-list-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:16px}.mat-optgroup-label[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-simple-snackbar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px}.mat-simple-snackbar-action[_ngcontent-%COMP%]{line-height:1;font-family:inherit;font-size:inherit;font-weight:500}.mat-tree[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{font-weight:400;font-size:14px}.mat-ripple[_ngcontent-%COMP%]{overflow:hidden;position:relative}.mat-ripple[_ngcontent-%COMP%]:not(:empty){transform:translateZ(0)}.mat-ripple.mat-ripple-unbounded[_ngcontent-%COMP%]{overflow:visible}.mat-ripple-element[_ngcontent-%COMP%]{position:absolute;border-radius:50%;pointer-events:none;transition:opacity,transform 0ms cubic-bezier(0, 0, 0.2, 1);transform:scale(0)}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{display:none}.cdk-visually-hidden[_ngcontent-%COMP%]{border:0;clip:rect(0 0 0 0);height:1px;margin:-1px;overflow:hidden;padding:0;position:absolute;width:1px;outline:0;-webkit-appearance:none;-moz-appearance:none}.cdk-overlay-container[_ngcontent-%COMP%], .cdk-global-overlay-wrapper[_ngcontent-%COMP%]{pointer-events:none;top:0;left:0;height:100%;width:100%}.cdk-overlay-container[_ngcontent-%COMP%]{position:fixed;z-index:1000}.cdk-overlay-container[_ngcontent-%COMP%]:empty{display:none}.cdk-global-overlay-wrapper[_ngcontent-%COMP%]{display:flex;position:absolute;z-index:1000}.cdk-overlay-pane[_ngcontent-%COMP%]{position:absolute;pointer-events:auto;box-sizing:border-box;z-index:1000;display:flex;max-width:100%;max-height:100%}.cdk-overlay-backdrop[_ngcontent-%COMP%]{position:absolute;top:0;bottom:0;left:0;right:0;z-index:1000;pointer-events:auto;-webkit-tap-highlight-color:transparent;transition:opacity 400ms cubic-bezier(0.25, 0.8, 0.25, 1);opacity:0}.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:1}@media screen and (-ms-high-contrast: active){.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:.6}}.cdk-overlay-dark-backdrop[_ngcontent-%COMP%]{background:rgba(0,0,0,.32)}.cdk-overlay-transparent-backdrop[_ngcontent-%COMP%], .cdk-overlay-transparent-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:0}.cdk-overlay-connected-position-bounding-box[_ngcontent-%COMP%]{position:absolute;z-index:1000;display:flex;flex-direction:column;min-width:1px;min-height:1px}.cdk-global-scrollblock[_ngcontent-%COMP%]{position:fixed;width:100%;overflow-y:scroll}@keyframes cdk-text-field-autofill-start{}@keyframes cdk-text-field-autofill-end{}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:-webkit-autofill{animation:cdk-text-field-autofill-start 0s 1ms}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:not(:-webkit-autofill){animation:cdk-text-field-autofill-end 0s 1ms}textarea.cdk-textarea-autosize[_ngcontent-%COMP%]{resize:none}textarea.cdk-textarea-autosize-measuring[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:auto !important;overflow:hidden !important}textarea.cdk-textarea-autosize-measuring-firefox[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:0 !important}.mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.1)}.mat-option[_ngcontent-%COMP%]{color:#212121}.mat-option[_ngcontent-%COMP%]:hover:not(.mat-option-disabled), .mat-option[_ngcontent-%COMP%]:focus:not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple):not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-active[_ngcontent-%COMP%]{background:rgba(0,0,0,.04);color:#212121}.mat-option.mat-option-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-primary[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f57c00}.mat-accent[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f44336}.mat-optgroup-label[_ngcontent-%COMP%]{color:#616161}.mat-optgroup-disabled[_ngcontent-%COMP%]   .mat-optgroup-label[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-pseudo-checkbox[_ngcontent-%COMP%]{color:#616161}.mat-pseudo-checkbox[_ngcontent-%COMP%]::after{color:#fafafa}.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{color:#b0b0b0}.mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f57c00}.mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f44336}.mat-pseudo-checkbox-checked.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{background:#b0b0b0}.mat-elevation-z0[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z1[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z2[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z3[_ngcontent-%COMP%]{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z4[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z5[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 5px 8px 0px rgba(0, 0, 0, 0.14),0px 1px 14px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z6[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z7[_ngcontent-%COMP%]{box-shadow:0px 4px 5px -2px rgba(0, 0, 0, 0.2),0px 7px 10px 1px rgba(0, 0, 0, 0.14),0px 2px 16px 1px rgba(0, 0, 0, 0.12)}.mat-elevation-z8[_ngcontent-%COMP%]{box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z9[_ngcontent-%COMP%]{box-shadow:0px 5px 6px -3px rgba(0, 0, 0, 0.2),0px 9px 12px 1px rgba(0, 0, 0, 0.14),0px 3px 16px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z10[_ngcontent-%COMP%]{box-shadow:0px 6px 6px -3px rgba(0, 0, 0, 0.2),0px 10px 14px 1px rgba(0, 0, 0, 0.14),0px 4px 18px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z11[_ngcontent-%COMP%]{box-shadow:0px 6px 7px -4px rgba(0, 0, 0, 0.2),0px 11px 15px 1px rgba(0, 0, 0, 0.14),0px 4px 20px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z12[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z13[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 13px 19px 2px rgba(0, 0, 0, 0.14),0px 5px 24px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z14[_ngcontent-%COMP%]{box-shadow:0px 7px 9px -4px rgba(0, 0, 0, 0.2),0px 14px 21px 2px rgba(0, 0, 0, 0.14),0px 5px 26px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z15[_ngcontent-%COMP%]{box-shadow:0px 8px 9px -5px rgba(0, 0, 0, 0.2),0px 15px 22px 2px rgba(0, 0, 0, 0.14),0px 6px 28px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z16[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z17[_ngcontent-%COMP%]{box-shadow:0px 8px 11px -5px rgba(0, 0, 0, 0.2),0px 17px 26px 2px rgba(0, 0, 0, 0.14),0px 6px 32px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z18[_ngcontent-%COMP%]{box-shadow:0px 9px 11px -5px rgba(0, 0, 0, 0.2),0px 18px 28px 2px rgba(0, 0, 0, 0.14),0px 7px 34px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z19[_ngcontent-%COMP%]{box-shadow:0px 9px 12px -6px rgba(0, 0, 0, 0.2),0px 19px 29px 2px rgba(0, 0, 0, 0.14),0px 7px 36px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z20[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 20px 31px 3px rgba(0, 0, 0, 0.14),0px 8px 38px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z21[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 21px 33px 3px rgba(0, 0, 0, 0.14),0px 8px 40px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z22[_ngcontent-%COMP%]{box-shadow:0px 10px 14px -6px rgba(0, 0, 0, 0.2),0px 22px 35px 3px rgba(0, 0, 0, 0.14),0px 8px 42px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z23[_ngcontent-%COMP%]{box-shadow:0px 11px 14px -7px rgba(0, 0, 0, 0.2),0px 23px 36px 3px rgba(0, 0, 0, 0.14),0px 9px 44px 8px rgba(0, 0, 0, 0.12)}.mat-elevation-z24[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12)}.mat-app-background[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-theme-loaded-marker[_ngcontent-%COMP%]{display:none}.mat-autocomplete-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-autocomplete-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover){background:#fff}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover):not(.mat-option-disabled){color:#212121}.mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f57c00}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{outline:solid 1px;border-radius:0}.mat-badge-accent[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-badge-warn[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f44336}.mat-badge[_ngcontent-%COMP%]{position:relative}.mat-badge-hidden[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{display:none}.mat-badge-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#b9b9b9;color:#757575}.mat-badge-content[_ngcontent-%COMP%]{position:absolute;text-align:center;display:inline-block;border-radius:50%;transition:transform 200ms ease-in-out;transform:scale(0.6);overflow:hidden;white-space:nowrap;text-overflow:ellipsis;pointer-events:none}.ng-animate-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%], .mat-badge-content._mat-animation-noopable[_ngcontent-%COMP%]{transition:none}.mat-badge-content.mat-badge-active[_ngcontent-%COMP%]{transform:none}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:16px;height:16px;line-height:16px}.mat-badge-small.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-8px}.mat-badge-small.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-8px}.mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-16px}.mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-16px}.mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-8px}.mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-8px}.mat-badge-medium[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:22px;height:22px;line-height:22px}.mat-badge-medium.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-11px}.mat-badge-medium.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-11px}.mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-22px}.mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-22px}.mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-11px}.mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-11px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:28px;height:28px;line-height:28px}.mat-badge-large.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-14px}.mat-badge-large.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-14px}.mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-28px}.mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-28px}.mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-14px}.mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-14px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]{color:inherit;background:transparent}.mat-button.mat-primary[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-button[disabled][disabled][_ngcontent-%COMP%], .mat-icon-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-icon-button[disabled][disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-stroked-button[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f44336}.mat-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:transparent}.mat-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{opacity:.1;background-color:currentColor}.mat-button-focus-overlay[_ngcontent-%COMP%]{background:#000}.mat-stroked-button[_ngcontent-%COMP%]:not([disabled]){border-color:rgba(0,0,0,.12)}.mat-flat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{color:#212121;background-color:#fff}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-flat-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-stroked-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-flat-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-raised-button[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone[_ngcontent-%COMP%], .mat-button-toggle-group[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{box-shadow:none}.mat-button-toggle[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-button-toggle[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121;background:#fff}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:#000}.mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12)}[dir=rtl][_ngcontent-%COMP%]   .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-group-appearance-standard.mat-button-toggle-vertical[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:none;border-top:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#616161}.mat-button-toggle-checked.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121}.mat-button-toggle-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.26);background-color:#eee}.mat-button-toggle-disabled.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{background:#fff}.mat-button-toggle-disabled.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#bdbdbd}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{border:solid 1px rgba(0,0,0,.12)}.mat-card[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-card[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-card.mat-card-flat[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-card-subtitle[_ngcontent-%COMP%]{color:#616161}.mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#616161}.mat-checkbox-checkmark[_ngcontent-%COMP%]{fill:#fafafa}.mat-checkbox-checkmark-path[_ngcontent-%COMP%]{stroke:#fafafa !important}.mat-checkbox-mixedmark[_ngcontent-%COMP%]{background-color:#fafafa}.mat-checkbox-indeterminate.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f57c00}.mat-checkbox-indeterminate.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#ff9800}.mat-checkbox-indeterminate.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f44336}.mat-checkbox-disabled.mat-checkbox-checked[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-disabled.mat-checkbox-indeterminate[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]:not(.mat-checkbox-checked)   .mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{color:#616161}.mat-checkbox[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%]{background:#f57c00}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%]{background:#ff9800}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%]{background:#f44336}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#212121}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#212121;opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled):active{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled)   .mat-chip-remove[_ngcontent-%COMP%]:hover{opacity:.54}.mat-chip.mat-standard-chip.mat-chip-disabled[_ngcontent-%COMP%]{opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]::after{background:#000}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-table[_ngcontent-%COMP%]{background:#fff}.mat-table[_ngcontent-%COMP%]   thead[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tbody[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tfoot[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], [mat-header-row][_ngcontent-%COMP%], [mat-row][_ngcontent-%COMP%], [mat-footer-row][_ngcontent-%COMP%], .mat-table-sticky[_ngcontent-%COMP%]{background:inherit}mat-row[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], th.mat-header-cell[_ngcontent-%COMP%], td.mat-cell[_ngcontent-%COMP%], td.mat-footer-cell[_ngcontent-%COMP%]{border-bottom-color:rgba(0,0,0,.12)}.mat-header-cell[_ngcontent-%COMP%]{color:#616161}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{color:#212121}.mat-calendar-arrow[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.54)}.mat-datepicker-toggle[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-next-button[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-previous-button[_ngcontent-%COMP%]{color:rgba(0,0,0,.54)}.mat-calendar-table-header[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-calendar-table-header-divider[_ngcontent-%COMP%]::after{background:rgba(0,0,0,.12)}.mat-calendar-body-label[_ngcontent-%COMP%]{color:#616161}.mat-calendar-body-cell-content[_ngcontent-%COMP%]{color:#212121;border-color:transparent}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){color:#757575}.mat-calendar-body-cell[_ngcontent-%COMP%]:not(.mat-calendar-body-disabled):hover > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-keyboard-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-program-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){background-color:rgba(0,0,0,.04)}.mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.38)}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.18)}.mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.4)}.mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12);background-color:#fff;color:#212121}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.4)}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.4)}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content-touch[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-datepicker-toggle-active[_ngcontent-%COMP%]{color:#f57c00}.mat-datepicker-toggle-active.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-datepicker-toggle-active.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-dialog-container[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-divider[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-divider-vertical[_ngcontent-%COMP%]{border-right-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-expansion-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-action-row[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-keyboard-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-program-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded)   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover:not([aria-disabled=true]){background:rgba(0,0,0,.04)}@media(hover: none){.mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded):not([aria-disabled=true])   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover{background:#fff}}.mat-expansion-panel-header-title[_ngcontent-%COMP%]{color:#212121}.mat-expansion-panel-header-description[_ngcontent-%COMP%], .mat-expansion-indicator[_ngcontent-%COMP%]::after{color:#616161}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-title[_ngcontent-%COMP%], .mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-description[_ngcontent-%COMP%]{color:inherit}.mat-form-field-label[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-hint[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-focused[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-ripple[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid)   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f57c00}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-accent   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#ff9800}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-warn   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#f44336}.mat-error[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-hint[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-legacy.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-standard[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-standard.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.04)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.02)}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:transparent}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.12)}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:rgba(0,0,0,.87)}.mat-form-field-appearance-outline.mat-focused[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field-appearance-outline.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-appearance-outline.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-invalid.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.06)}.mat-icon.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-icon.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-icon.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-type-mat-native-select[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#616161}.mat-input-element[_ngcontent-%COMP%]:disabled, .mat-form-field-type-mat-native-select.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#757575}.mat-input-element[_ngcontent-%COMP%]{caret-color:#f57c00}.mat-input-element[_ngcontent-%COMP%]::placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-moz-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-webkit-input-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]:-ms-input-placeholder{color:rgba(97,97,97,.42)}.mat-accent[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%], .mat-form-field-invalid[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#f44336}.mat-form-field-type-mat-native-select.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{color:#616161}.mat-list-item-disabled[_ngcontent-%COMP%]{background-color:#eee}.mat-list-option[_ngcontent-%COMP%]:hover, .mat-list-option[_ngcontent-%COMP%]:focus, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.04)}.mat-list-single-selected-option[_ngcontent-%COMP%], .mat-list-single-selected-option[_ngcontent-%COMP%]:hover, .mat-list-single-selected-option[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.12)}.mat-menu-panel[_ngcontent-%COMP%]{background:#fff}.mat-menu-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-menu-item[_ngcontent-%COMP%]{background:transparent;color:#212121}.mat-menu-item[disabled][_ngcontent-%COMP%], .mat-menu-item[disabled][_ngcontent-%COMP%]::after{color:rgba(0,0,0,.38)}.mat-menu-item[_ngcontent-%COMP%]   .mat-icon-no-color[_ngcontent-%COMP%], .mat-menu-item-submenu-trigger[_ngcontent-%COMP%]::after{color:rgba(0,0,0,.54)}.mat-menu-item[_ngcontent-%COMP%]:hover:not([disabled]), .mat-menu-item.cdk-program-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item.cdk-keyboard-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item-highlighted[_ngcontent-%COMP%]:not([disabled]){background:rgba(0,0,0,.04)}.mat-paginator[_ngcontent-%COMP%]{background:#fff}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{color:#616161}.mat-paginator-decrement[_ngcontent-%COMP%], .mat-paginator-increment[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54);border-right:2px solid rgba(0,0,0,.54)}.mat-paginator-first[_ngcontent-%COMP%], .mat-paginator-last[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54)}.mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-decrement[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-increment[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-first[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-last[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f57c00}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#ff9800}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f44336}.mat-progress-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f57c00}.mat-progress-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#ff9800}.mat-progress-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f44336}.mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#616161}.mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f57c00}.mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#ff9800}.mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#ff9800}.mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f44336}.mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f44336}.mat-radio-button.mat-radio-disabled.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-label-content[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-radio-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-select-value[_ngcontent-%COMP%]{color:#212121}.mat-select-placeholder[_ngcontent-%COMP%]{color:rgba(97,97,97,.42)}.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%]{color:#757575}.mat-select-arrow[_ngcontent-%COMP%]{color:#616161}.mat-select-panel[_ngcontent-%COMP%]{background:#fff}.mat-select-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-select-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple){background:rgba(0,0,0,.12)}.mat-form-field.mat-focused.mat-primary[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-invalid[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#757575}.mat-drawer-container[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-drawer[_ngcontent-%COMP%]{background-color:#fff;color:#212121}.mat-drawer.mat-drawer-push[_ngcontent-%COMP%]{background-color:#fff}.mat-drawer[_ngcontent-%COMP%]:not(.mat-drawer-side){box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-drawer-side[_ngcontent-%COMP%]{border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-backdrop.mat-drawer-shown[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.6)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.54)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.54)}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.54)}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle[_ngcontent-%COMP%]:not(.mat-checked)   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-slide-toggle-thumb[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12);background-color:#fafafa}.mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-primary[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f57c00}.mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-accent[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#ff9800}.mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-warn[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f44336}.mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.2)}.mat-slider[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%], .cdk-focused[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-disabled[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]   .mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing)   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26);background-color:transparent}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26)}.mat-slider-has-ticks[_ngcontent-%COMP%]   .mat-slider-wrapper[_ngcontent-%COMP%]::after{border-color:rgba(0,0,0,.7)}.mat-slider-horizontal[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to right, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent);background-image:-moz-repeating-linear-gradient(0.0001deg, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-slider-vertical[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to bottom, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-step-header.cdk-keyboard-focused[_ngcontent-%COMP%], .mat-step-header.cdk-program-focused[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]:hover{background-color:rgba(0,0,0,.04)}@media(hover: none){.mat-step-header[_ngcontent-%COMP%]:hover{background:none}}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-optional[_ngcontent-%COMP%]{color:#616161}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon[_ngcontent-%COMP%]{background-color:#616161;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-selected[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-done[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-edit[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-error[_ngcontent-%COMP%]{background-color:transparent;color:#f44336}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-active[_ngcontent-%COMP%]{color:#212121}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-error[_ngcontent-%COMP%]{color:#f44336}.mat-stepper-horizontal[_ngcontent-%COMP%], .mat-stepper-vertical[_ngcontent-%COMP%]{background-color:#fff}.mat-stepper-vertical-line[_ngcontent-%COMP%]::before{border-left-color:rgba(0,0,0,.12)}.mat-horizontal-stepper-header[_ngcontent-%COMP%]::before, .mat-horizontal-stepper-header[_ngcontent-%COMP%]::after, .mat-stepper-horizontal-line[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-sort-header-arrow[_ngcontent-%COMP%]{color:#616161}.mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-header[_ngcontent-%COMP%]{border-bottom:1px solid rgba(0,0,0,.12)}.mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%]{border-top:1px solid rgba(0,0,0,.12);border-bottom:none}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{color:#212121}.mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:#757575}.mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#212121}.mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#757575}.mat-tab-group[class*=mat-background-][_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar[class*=mat-background-][_ngcontent-%COMP%]{border-bottom:none;border-top:none}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-toolbar[_ngcontent-%COMP%]{background:#f5f5f5;color:#212121}.mat-toolbar.mat-primary[_ngcontent-%COMP%]{background:#f57c00;color:#fff}.mat-toolbar.mat-accent[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-toolbar.mat-warn[_ngcontent-%COMP%]{background:#f44336;color:#fff}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:currentColor}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:inherit}.mat-toolbar[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:currentColor}.mat-tooltip[_ngcontent-%COMP%]{background:rgba(97,97,97,.9)}.mat-tree[_ngcontent-%COMP%]{background:#fff}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{color:#212121}.mat-snack-bar-container[_ngcontent-%COMP%]{color:rgba(255,255,255,.7);background:#323232;box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-simple-snackbar-action[_ngcontent-%COMP%]{color:#ff9800}mat-select[_ngcontent-%COMP%]{border:1px solid #8e98a3;border-radius:3px;box-sizing:border-box;padding:6px}"] });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/views/right_pane/settings_view_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    var I18N_0$6;
    if (typeof ngI18nClosureMode !== "undefined" && ngI18nClosureMode) {
        /**
         * @desc A button to reset the image brightness setting
         */ 
        const MSG_EXTERNAL_1189161857240378395$$TENSORBOARD_WEBAPP_METRICS_VIEWS_RIGHT_PANE_SETTINGS_VIEW_COMPONENT_TS_1 = goog.getMsg("Reset brightness");
        I18N_0$6 = MSG_EXTERNAL_1189161857240378395$$TENSORBOARD_WEBAPP_METRICS_VIEWS_RIGHT_PANE_SETTINGS_VIEW_COMPONENT_TS_1;
    }
    else {
        I18N_0$6 = $localize `:A button to reset the image brightness setting␟c482b3a47ea0975fa8be01afb3fbec9b76628bd7␟1189161857240378395:Reset brightness`;
    }
    const _c2$c = ["aria-label", I18N_0$6];
    var I18N_3$4;
    if (typeof ngI18nClosureMode !== "undefined" && ngI18nClosureMode) {
        /**
         * @desc A button to reset the image contrast setting
         */ 
        const MSG_EXTERNAL_5370703342923611955$$TENSORBOARD_WEBAPP_METRICS_VIEWS_RIGHT_PANE_SETTINGS_VIEW_COMPONENT_TS_4 = goog.getMsg("Reset contrast");
        I18N_3$4 = MSG_EXTERNAL_5370703342923611955$$TENSORBOARD_WEBAPP_METRICS_VIEWS_RIGHT_PANE_SETTINGS_VIEW_COMPONENT_TS_4;
    }
    else {
        I18N_3$4 = $localize `:A button to reset the image contrast setting␟ed712a8b927041be15252b29eb521ebb1374bad8␟5370703342923611955:Reset contrast`;
    }
    const _c5$6 = ["aria-label", I18N_3$4];
    /** @type {?} */
    const SLIDER_AUDIT_TIME_MS = 250;
    class SettingsViewComponent {
        /**
         * @param {?} locale
         */
        constructor(locale) {
            this.locale = locale;
            this.TooltipSortDropdownOptions = [
                { value: TooltipSortingMethod.DEFAULT, displayText: 'Default' },
                { value: TooltipSortingMethod.ASCENDING, displayText: 'Ascending' },
                { value: TooltipSortingMethod.DESCENDING, displayText: 'Descending' },
                { value: TooltipSortingMethod.NEAREST, displayText: 'Nearest' },
            ];
            this.tooltipSortChanged = new EventEmitter();
            this.ignoreOutliersChanged = new EventEmitter();
            this.XAxisTypeDropdownOptions = [
                { value: XAxisType$1.STEP, displayText: 'Step' },
                { value: XAxisType$1.RELATIVE, displayText: 'Relative' },
                { value: XAxisType$1.WALL_TIME, displayText: 'Wall' },
            ];
            this.xAxisTypeChanged = new EventEmitter();
            this.HistogramModeDropdownOptions = [
                { value: HistogramMode.OFFSET, displayText: 'Offset' },
                { value: HistogramMode.OVERLAY, displayText: 'Overlay' },
            ];
            this.histogramModeChanged = new EventEmitter();
            this.scalarSmoothingControlChanged$ = new EventEmitter();
            this.scalarSmoothingChanged = this.scalarSmoothingControlChanged$.pipe(auditTime(SLIDER_AUDIT_TIME_MS));
            this.imageBrightnessSliderChanged$ = new EventEmitter();
            this.imageBrightnessInMilliChanged = this.imageBrightnessSliderChanged$.pipe(auditTime(SLIDER_AUDIT_TIME_MS));
            this.imageBrightnessReset = new EventEmitter();
            this.imageContrastSliderChanged$ = new EventEmitter();
            this.imageContrastInMilliChanged = this.imageContrastSliderChanged$.pipe(auditTime(SLIDER_AUDIT_TIME_MS));
            this.imageContrastReset = new EventEmitter();
            this.imageShowActualSizeChanged = new EventEmitter();
        }
        /**
         * @param {?} event
         * @return {?}
         */
        onScalarSmoothingInput(event) {
            /** @type {?} */
            const input = (/** @type {?} */ (event.target));
            if (!input.value) {
                return;
            }
            /** @type {?} */
            const nextValue = Math.min(Math.max(0, parseFloat(input.value)), 1);
            // Rectify here in case Angular does not trigger ngOnChanges when expected.
            if (nextValue !== parseFloat(input.value)) {
                input.value = String(nextValue);
            }
            this.scalarSmoothingControlChanged$.emit(nextValue);
        }
        /**
         * @param {?} num
         * @return {?}
         */
        formatMilliToZeroth(num) {
            return formatNumber(num / 1000, 
            // Our app does not, yet, specify LOCALE_ID. Default to en-US.
            this.locale || 'en-US', 
            // The slider cannot fit 3 decimals.
            '1.0-2');
        }
    }
    SettingsViewComponent.decorators = [
        { type: Component, args: [{
                    selector: 'metrics-dashboard-settings-component',
                    templateUrl: 'settings_view_component.ng.html',
                    changeDetection: ChangeDetectionStrategy.OnPush,
                    styleUrls: [`settings_view_component.css`],
                },] },
    ];
    /** @nocollapse */
    SettingsViewComponent.ctorParameters = () => [
        { type: String, decorators: [{ type: Inject, args: [LOCALE_ID$1,] }] }
    ];
    SettingsViewComponent.propDecorators = {
        tooltipSort: [{ type: Input }],
        tooltipSortChanged: [{ type: Output }],
        ignoreOutliers: [{ type: Input }],
        ignoreOutliersChanged: [{ type: Output }],
        xAxisType: [{ type: Input }],
        xAxisTypeChanged: [{ type: Output }],
        histogramMode: [{ type: Input }],
        histogramModeChanged: [{ type: Output }],
        scalarSmoothing: [{ type: Input }],
        scalarSmoothingChanged: [{ type: Output }],
        imageBrightnessInMilli: [{ type: Input }],
        imageBrightnessInMilliChanged: [{ type: Output }],
        imageBrightnessReset: [{ type: Output }],
        imageContrastInMilli: [{ type: Input }],
        imageContrastInMilliChanged: [{ type: Output }],
        imageContrastReset: [{ type: Output }],
        imageShowActualSize: [{ type: Input }],
        imageShowActualSizeChanged: [{ type: Output }]
    };
    /** @nocollapse */ SettingsViewComponent.ɵfac = function SettingsViewComponent_Factory(t) { return new (t || SettingsViewComponent)(ɵɵdirectiveInject(LOCALE_ID$1)); };
    /** @nocollapse */ SettingsViewComponent.ɵcmp = ɵɵdefineComponent({ type: SettingsViewComponent, selectors: [["metrics-dashboard-settings-component"]], inputs: { tooltipSort: "tooltipSort", ignoreOutliers: "ignoreOutliers", xAxisType: "xAxisType", histogramMode: "histogramMode", scalarSmoothing: "scalarSmoothing", imageBrightnessInMilli: "imageBrightnessInMilli", imageContrastInMilli: "imageContrastInMilli", imageShowActualSize: "imageShowActualSize" }, outputs: { tooltipSortChanged: "tooltipSortChanged", ignoreOutliersChanged: "ignoreOutliersChanged", xAxisTypeChanged: "xAxisTypeChanged", histogramModeChanged: "histogramModeChanged", scalarSmoothingChanged: "scalarSmoothingChanged", imageBrightnessInMilliChanged: "imageBrightnessInMilliChanged", imageBrightnessReset: "imageBrightnessReset", imageContrastInMilliChanged: "imageContrastInMilliChanged", imageContrastReset: "imageContrastReset", imageShowActualSizeChanged: "imageShowActualSizeChanged" }, decls: 52, vars: 26, consts: [[1, "general"], [1, "section-title"], [1, "control-row", "x-axis-type"], ["id", "x-axis-type-label", 1, "control-name"], [3, "value", "options", "selectionChange"], [1, "scalars"], [1, "control-row", "scalars-smoothing"], ["id", "scalars-smoothing-label", 1, "control-name"], [1, "slider-row"], ["aria-labelledby", "scalars-smoothing-label", "color", "primary", 3, "max", "min", "step", "value", "thumbLabel", "input"], ["aria-labelledby", "scalars-smoothing-label", "type", "number", "min", "0", "max", "1", "step", "0.01", 1, "slider-input", 3, "value", "input"], [1, "control-row", "tooltip-sort"], [1, "control-name"], [1, "control-row", "scalars-ignore-outliers"], [3, "checked", "change"], [1, "Histograms"], [1, "control-row", "histogram-mode"], [1, "image"], [1, "control-row", "image-brightness"], ["id", "image-brightness-label", 1, "control-name"], ["aria-labelledby", "image-brightness-label", "color", "primary", 3, "max", "min", "step", "value", "thumbLabel", "displayWith", "input"], ["mat-icon-button", "", "title", "Reset brightness", 1, "reset-button", 3, "click", 6, "aria-label"], ["svgIcon", "settings_backup_restore_24px"], [1, "control-row", "image-contrast"], ["id", "image-constrast-label", 1, "control-name"], ["aria-labelledby", "image-constrast-label", "color", "primary", 3, "max", "min", "step", "value", "thumbLabel", "displayWith", "input"], ["mat-icon-button", "", "title", "Reset contrast", 1, "reset-button", 3, "click", 6, "aria-label"], [1, "control-row", "image-show-actual-size"]], template: function SettingsViewComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "section", 0);
            ɵɵelementStart(1, "h3", 1);
            ɵɵtext(2, "General");
            ɵɵelementEnd();
            ɵɵelementStart(3, "div", 2);
            ɵɵelementStart(4, "div", 3);
            ɵɵtext(5, "Horizontal Axis");
            ɵɵelementEnd();
            ɵɵelementStart(6, "tb-dropdown", 4);
            ɵɵlistener("selectionChange", function SettingsViewComponent_Template_tb_dropdown_selectionChange_6_listener($event) { return ctx.xAxisTypeChanged.emit($event); });
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementStart(7, "section", 5);
            ɵɵelementStart(8, "h3", 1);
            ɵɵtext(9, "Scalars");
            ɵɵelementEnd();
            ɵɵelementStart(10, "div", 6);
            ɵɵelementStart(11, "div", 7);
            ɵɵtext(12, "Smoothing");
            ɵɵelementEnd();
            ɵɵelementStart(13, "div", 8);
            ɵɵelementStart(14, "mat-slider", 9);
            ɵɵlistener("input", function SettingsViewComponent_Template_mat_slider_input_14_listener($event) { return ctx.scalarSmoothingControlChanged$.emit($event.value); });
            ɵɵelementEnd();
            ɵɵelementStart(15, "input", 10);
            ɵɵlistener("input", function SettingsViewComponent_Template_input_input_15_listener($event) { return ctx.onScalarSmoothingInput($event); });
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementStart(16, "div", 11);
            ɵɵelementStart(17, "div", 12);
            ɵɵtext(18, "Tooltip sorting method");
            ɵɵelementEnd();
            ɵɵelementStart(19, "tb-dropdown", 4);
            ɵɵlistener("selectionChange", function SettingsViewComponent_Template_tb_dropdown_selectionChange_19_listener($event) { return ctx.tooltipSortChanged.emit($event); });
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementStart(20, "div", 13);
            ɵɵelementStart(21, "mat-checkbox", 14);
            ɵɵlistener("change", function SettingsViewComponent_Template_mat_checkbox_change_21_listener($event) { return ctx.ignoreOutliersChanged.emit($event.checked); });
            ɵɵtext(22, "Ignore outliers in chart scaling");
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementStart(23, "section", 15);
            ɵɵelementStart(24, "h3", 1);
            ɵɵtext(25, "Histograms");
            ɵɵelementEnd();
            ɵɵelementStart(26, "div", 16);
            ɵɵelementStart(27, "div", 12);
            ɵɵtext(28, "Mode");
            ɵɵelementEnd();
            ɵɵelementStart(29, "tb-dropdown", 4);
            ɵɵlistener("selectionChange", function SettingsViewComponent_Template_tb_dropdown_selectionChange_29_listener($event) { return ctx.histogramModeChanged.emit($event); });
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementStart(30, "section", 17);
            ɵɵelementStart(31, "h3", 1);
            ɵɵtext(32, "Images");
            ɵɵelementEnd();
            ɵɵelementStart(33, "div", 18);
            ɵɵelementStart(34, "div", 19);
            ɵɵtext(35, "Brightness");
            ɵɵelementEnd();
            ɵɵelementStart(36, "div", 8);
            ɵɵelementStart(37, "mat-slider", 20);
            ɵɵlistener("input", function SettingsViewComponent_Template_mat_slider_input_37_listener($event) { return ctx.imageBrightnessSliderChanged$.emit($event.value); });
            ɵɵelementEnd();
            ɵɵelementStart(38, "button", 21);
            ɵɵi18nAttributes(39, _c2$c);
            ɵɵlistener("click", function SettingsViewComponent_Template_button_click_38_listener() { return ctx.imageBrightnessReset.emit(); });
            ɵɵelement(40, "mat-icon", 22);
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementStart(41, "div", 23);
            ɵɵelementStart(42, "div", 24);
            ɵɵtext(43, "Contrast");
            ɵɵelementEnd();
            ɵɵelementStart(44, "div", 8);
            ɵɵelementStart(45, "mat-slider", 25);
            ɵɵlistener("input", function SettingsViewComponent_Template_mat_slider_input_45_listener($event) { return ctx.imageContrastSliderChanged$.emit($event.value); });
            ɵɵelementEnd();
            ɵɵelementStart(46, "button", 26);
            ɵɵi18nAttributes(47, _c5$6);
            ɵɵlistener("click", function SettingsViewComponent_Template_button_click_46_listener() { return ctx.imageContrastReset.emit(); });
            ɵɵelement(48, "mat-icon", 22);
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementStart(49, "div", 27);
            ɵɵelementStart(50, "mat-checkbox", 14);
            ɵɵlistener("change", function SettingsViewComponent_Template_mat_checkbox_change_50_listener($event) { return ctx.imageShowActualSizeChanged.emit($event.checked); });
            ɵɵtext(51, "Show actual image size");
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵadvance(6);
            ɵɵproperty("value", ctx.xAxisType)("options", ctx.XAxisTypeDropdownOptions);
            ɵɵadvance(8);
            ɵɵproperty("max", 1)("min", 0)("step", 0.01)("value", ctx.scalarSmoothing)("thumbLabel", true);
            ɵɵadvance(1);
            ɵɵproperty("value", ctx.scalarSmoothing);
            ɵɵadvance(4);
            ɵɵproperty("value", ctx.tooltipSort)("options", ctx.TooltipSortDropdownOptions);
            ɵɵadvance(2);
            ɵɵproperty("checked", ctx.ignoreOutliers);
            ɵɵadvance(8);
            ɵɵproperty("value", ctx.histogramMode)("options", ctx.HistogramModeDropdownOptions);
            ɵɵadvance(8);
            ɵɵproperty("max", 2000)("min", 0)("step", 10)("value", ctx.imageBrightnessInMilli)("thumbLabel", true)("displayWith", ctx.formatMilliToZeroth);
            ɵɵadvance(8);
            ɵɵproperty("max", 5000)("min", 0)("step", 10)("value", ctx.imageContrastInMilli)("thumbLabel", true)("displayWith", ctx.formatMilliToZeroth);
            ɵɵadvance(5);
            ɵɵproperty("checked", ctx.imageShowActualSize);
        } }, directives: [DropdownComponent, MatSlider, MatCheckbox, MatButton, MatIcon], styles: [".mat-badge-content[_ngcontent-%COMP%]{font-weight:600;font-size:12px;font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:9px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:24px}.mat-h1[_ngcontent-%COMP%], .mat-headline[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%]{font:400 24px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h2[_ngcontent-%COMP%], .mat-title[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h3[_ngcontent-%COMP%], .mat-subheading-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%]{font:400 16px/28px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h4[_ngcontent-%COMP%], .mat-subheading-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%]{font:400 15px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h5[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%]{font:400 calc(14px * 0.83)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-h6[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:400 calc(14px * 0.67)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-body-strong[_ngcontent-%COMP%], .mat-body-2[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   p[_ngcontent-%COMP%]{margin:0 0 12px}.mat-small[_ngcontent-%COMP%], .mat-caption[_ngcontent-%COMP%]{font:400 12px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-display-4[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-4[_ngcontent-%COMP%]{font:300 112px/112px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.05em;margin:0 0 56px}.mat-display-3[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-3[_ngcontent-%COMP%]{font:400 56px/56px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.02em;margin:0 0 64px}.mat-display-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-2[_ngcontent-%COMP%]{font:400 45px/48px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.005em;margin:0 0 64px}.mat-display-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-1[_ngcontent-%COMP%]{font:400 34px/40px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 64px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%], .mat-flat-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-button-toggle[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card-title[_ngcontent-%COMP%]{font-size:24px;font-weight:500}.mat-card-header[_ngcontent-%COMP%]   .mat-card-title[_ngcontent-%COMP%]{font-size:20px}.mat-card-subtitle[_ngcontent-%COMP%], .mat-card-content[_ngcontent-%COMP%]{font-size:14px}.mat-checkbox[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-checkbox-layout[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{line-height:24px}.mat-chip[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-chip[_ngcontent-%COMP%]   .mat-chip-trailing-icon.mat-icon[_ngcontent-%COMP%], .mat-chip[_ngcontent-%COMP%]   .mat-chip-remove.mat-icon[_ngcontent-%COMP%]{font-size:18px}.mat-table[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-header-cell[_ngcontent-%COMP%]{font-size:12px;font-weight:500}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{font-size:14px}.mat-calendar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-calendar-body[_ngcontent-%COMP%]{font-size:13px}.mat-calendar-body-label[_ngcontent-%COMP%], .mat-calendar-period-button[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-calendar-table-header[_ngcontent-%COMP%]   th[_ngcontent-%COMP%]{font-size:11px;font-weight:400}.mat-dialog-title[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-expansion-panel-header[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:15px;font-weight:400}.mat-expansion-panel-content[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field[_ngcontent-%COMP%]{font-size:inherit;font-weight:400;line-height:1.125;font-family:Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.34375em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{font-size:150%;line-height:1.125}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]{height:1.5em;width:1.5em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{height:1.125em;line-height:1.125}.mat-form-field-infix[_ngcontent-%COMP%]{padding:.5em 0;border-top:.84375em solid transparent}.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34375em) scale(0.75);width:133.3333333333%}.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34374em) scale(0.75);width:133.3333433333%}.mat-form-field-label-wrapper[_ngcontent-%COMP%]{top:-0.84375em;padding-top:.84375em}.mat-form-field-label[_ngcontent-%COMP%]{top:1.34375em}.mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.34375em}.mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{font-size:75%;margin-top:.6666666667em;top:calc(100% - 1.7916666667em)}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.4375em 0}.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.001px);-ms-transform:translateY(-1.28125em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00101px);-ms-transform:translateY(-1.28124em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00102px);-ms-transform:translateY(-1.28123em) scale(0.75);width:133.3333533333%}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.28125em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{margin-top:.5416666667em;top:calc(100% - 1.6666666667em)}@media print{.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28122em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28121em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.2812em) scale(0.75)}}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.25em 0 .75em 0}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.09375em;margin-top:-0.5em}.mat-form-field-appearance-fill.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59374em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:1em 0 1em 0}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.84375em;margin-top:-0.25em}.mat-form-field-appearance-outline.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59374em) scale(0.75);width:133.3333433333%}.mat-grid-tile-header[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]{font-size:14px}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2), .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}input.mat-input-element[_ngcontent-%COMP%]{margin-top:-0.0625em}.mat-menu-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:400}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px}.mat-radio-button[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select-trigger[_ngcontent-%COMP%]{height:1.125em}.mat-slide-toggle-content[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-slider-thumb-label-text[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-stepper-vertical[_ngcontent-%COMP%], .mat-stepper-horizontal[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-step-label[_ngcontent-%COMP%]{font-size:14px;font-weight:400}.mat-step-sub-label-error[_ngcontent-%COMP%]{font-weight:normal}.mat-step-label-error[_ngcontent-%COMP%]{font-size:14px}.mat-step-label-selected[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-tab-group[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-toolbar[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0}.mat-tooltip[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:10px;padding-top:6px;padding-bottom:6px}.mat-tooltip-handset[_ngcontent-%COMP%]{font-size:14px;padding-top:8px;padding-bottom:8px}.mat-list-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:16px}.mat-optgroup-label[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-simple-snackbar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px}.mat-simple-snackbar-action[_ngcontent-%COMP%]{line-height:1;font-family:inherit;font-size:inherit;font-weight:500}.mat-tree[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{font-weight:400;font-size:14px}.mat-ripple[_ngcontent-%COMP%]{overflow:hidden;position:relative}.mat-ripple[_ngcontent-%COMP%]:not(:empty){transform:translateZ(0)}.mat-ripple.mat-ripple-unbounded[_ngcontent-%COMP%]{overflow:visible}.mat-ripple-element[_ngcontent-%COMP%]{position:absolute;border-radius:50%;pointer-events:none;transition:opacity,transform 0ms cubic-bezier(0, 0, 0.2, 1);transform:scale(0)}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{display:none}.cdk-visually-hidden[_ngcontent-%COMP%]{border:0;clip:rect(0 0 0 0);height:1px;margin:-1px;overflow:hidden;padding:0;position:absolute;width:1px;outline:0;-webkit-appearance:none;-moz-appearance:none}.cdk-overlay-container[_ngcontent-%COMP%], .cdk-global-overlay-wrapper[_ngcontent-%COMP%]{pointer-events:none;top:0;left:0;height:100%;width:100%}.cdk-overlay-container[_ngcontent-%COMP%]{position:fixed;z-index:1000}.cdk-overlay-container[_ngcontent-%COMP%]:empty{display:none}.cdk-global-overlay-wrapper[_ngcontent-%COMP%]{display:flex;position:absolute;z-index:1000}.cdk-overlay-pane[_ngcontent-%COMP%]{position:absolute;pointer-events:auto;box-sizing:border-box;z-index:1000;display:flex;max-width:100%;max-height:100%}.cdk-overlay-backdrop[_ngcontent-%COMP%]{position:absolute;top:0;bottom:0;left:0;right:0;z-index:1000;pointer-events:auto;-webkit-tap-highlight-color:transparent;transition:opacity 400ms cubic-bezier(0.25, 0.8, 0.25, 1);opacity:0}.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:1}@media screen and (-ms-high-contrast: active){.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:.6}}.cdk-overlay-dark-backdrop[_ngcontent-%COMP%]{background:rgba(0,0,0,.32)}.cdk-overlay-transparent-backdrop[_ngcontent-%COMP%], .cdk-overlay-transparent-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:0}.cdk-overlay-connected-position-bounding-box[_ngcontent-%COMP%]{position:absolute;z-index:1000;display:flex;flex-direction:column;min-width:1px;min-height:1px}.cdk-global-scrollblock[_ngcontent-%COMP%]{position:fixed;width:100%;overflow-y:scroll}@keyframes cdk-text-field-autofill-start{}@keyframes cdk-text-field-autofill-end{}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:-webkit-autofill{animation:cdk-text-field-autofill-start 0s 1ms}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:not(:-webkit-autofill){animation:cdk-text-field-autofill-end 0s 1ms}textarea.cdk-textarea-autosize[_ngcontent-%COMP%]{resize:none}textarea.cdk-textarea-autosize-measuring[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:auto !important;overflow:hidden !important}textarea.cdk-textarea-autosize-measuring-firefox[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:0 !important}.mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.1)}.mat-option[_ngcontent-%COMP%]{color:#212121}.mat-option[_ngcontent-%COMP%]:hover:not(.mat-option-disabled), .mat-option[_ngcontent-%COMP%]:focus:not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple):not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-active[_ngcontent-%COMP%]{background:rgba(0,0,0,.04);color:#212121}.mat-option.mat-option-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-primary[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f57c00}.mat-accent[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f44336}.mat-optgroup-label[_ngcontent-%COMP%]{color:#616161}.mat-optgroup-disabled[_ngcontent-%COMP%]   .mat-optgroup-label[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-pseudo-checkbox[_ngcontent-%COMP%]{color:#616161}.mat-pseudo-checkbox[_ngcontent-%COMP%]::after{color:#fafafa}.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{color:#b0b0b0}.mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f57c00}.mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f44336}.mat-pseudo-checkbox-checked.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{background:#b0b0b0}.mat-elevation-z0[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z1[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z2[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z3[_ngcontent-%COMP%]{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z4[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z5[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 5px 8px 0px rgba(0, 0, 0, 0.14),0px 1px 14px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z6[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z7[_ngcontent-%COMP%]{box-shadow:0px 4px 5px -2px rgba(0, 0, 0, 0.2),0px 7px 10px 1px rgba(0, 0, 0, 0.14),0px 2px 16px 1px rgba(0, 0, 0, 0.12)}.mat-elevation-z8[_ngcontent-%COMP%]{box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z9[_ngcontent-%COMP%]{box-shadow:0px 5px 6px -3px rgba(0, 0, 0, 0.2),0px 9px 12px 1px rgba(0, 0, 0, 0.14),0px 3px 16px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z10[_ngcontent-%COMP%]{box-shadow:0px 6px 6px -3px rgba(0, 0, 0, 0.2),0px 10px 14px 1px rgba(0, 0, 0, 0.14),0px 4px 18px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z11[_ngcontent-%COMP%]{box-shadow:0px 6px 7px -4px rgba(0, 0, 0, 0.2),0px 11px 15px 1px rgba(0, 0, 0, 0.14),0px 4px 20px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z12[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z13[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 13px 19px 2px rgba(0, 0, 0, 0.14),0px 5px 24px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z14[_ngcontent-%COMP%]{box-shadow:0px 7px 9px -4px rgba(0, 0, 0, 0.2),0px 14px 21px 2px rgba(0, 0, 0, 0.14),0px 5px 26px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z15[_ngcontent-%COMP%]{box-shadow:0px 8px 9px -5px rgba(0, 0, 0, 0.2),0px 15px 22px 2px rgba(0, 0, 0, 0.14),0px 6px 28px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z16[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z17[_ngcontent-%COMP%]{box-shadow:0px 8px 11px -5px rgba(0, 0, 0, 0.2),0px 17px 26px 2px rgba(0, 0, 0, 0.14),0px 6px 32px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z18[_ngcontent-%COMP%]{box-shadow:0px 9px 11px -5px rgba(0, 0, 0, 0.2),0px 18px 28px 2px rgba(0, 0, 0, 0.14),0px 7px 34px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z19[_ngcontent-%COMP%]{box-shadow:0px 9px 12px -6px rgba(0, 0, 0, 0.2),0px 19px 29px 2px rgba(0, 0, 0, 0.14),0px 7px 36px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z20[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 20px 31px 3px rgba(0, 0, 0, 0.14),0px 8px 38px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z21[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 21px 33px 3px rgba(0, 0, 0, 0.14),0px 8px 40px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z22[_ngcontent-%COMP%]{box-shadow:0px 10px 14px -6px rgba(0, 0, 0, 0.2),0px 22px 35px 3px rgba(0, 0, 0, 0.14),0px 8px 42px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z23[_ngcontent-%COMP%]{box-shadow:0px 11px 14px -7px rgba(0, 0, 0, 0.2),0px 23px 36px 3px rgba(0, 0, 0, 0.14),0px 9px 44px 8px rgba(0, 0, 0, 0.12)}.mat-elevation-z24[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12)}.mat-app-background[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-theme-loaded-marker[_ngcontent-%COMP%]{display:none}.mat-autocomplete-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-autocomplete-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover){background:#fff}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover):not(.mat-option-disabled){color:#212121}.mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f57c00}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{outline:solid 1px;border-radius:0}.mat-badge-accent[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-badge-warn[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f44336}.mat-badge[_ngcontent-%COMP%]{position:relative}.mat-badge-hidden[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{display:none}.mat-badge-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#b9b9b9;color:#757575}.mat-badge-content[_ngcontent-%COMP%]{position:absolute;text-align:center;display:inline-block;border-radius:50%;transition:transform 200ms ease-in-out;transform:scale(0.6);overflow:hidden;white-space:nowrap;text-overflow:ellipsis;pointer-events:none}.ng-animate-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%], .mat-badge-content._mat-animation-noopable[_ngcontent-%COMP%]{transition:none}.mat-badge-content.mat-badge-active[_ngcontent-%COMP%]{transform:none}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:16px;height:16px;line-height:16px}.mat-badge-small.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-8px}.mat-badge-small.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-8px}.mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-16px}.mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-16px}.mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-8px}.mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-8px}.mat-badge-medium[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:22px;height:22px;line-height:22px}.mat-badge-medium.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-11px}.mat-badge-medium.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-11px}.mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-22px}.mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-22px}.mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-11px}.mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-11px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:28px;height:28px;line-height:28px}.mat-badge-large.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-14px}.mat-badge-large.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-14px}.mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-28px}.mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-28px}.mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-14px}.mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-14px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]{color:inherit;background:transparent}.mat-button.mat-primary[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-button[disabled][disabled][_ngcontent-%COMP%], .mat-icon-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-icon-button[disabled][disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-stroked-button[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f44336}.mat-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:transparent}.mat-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{opacity:.1;background-color:currentColor}.mat-button-focus-overlay[_ngcontent-%COMP%]{background:#000}.mat-stroked-button[_ngcontent-%COMP%]:not([disabled]){border-color:rgba(0,0,0,.12)}.mat-flat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{color:#212121;background-color:#fff}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-flat-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-stroked-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-flat-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-raised-button[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone[_ngcontent-%COMP%], .mat-button-toggle-group[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{box-shadow:none}.mat-button-toggle[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-button-toggle[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121;background:#fff}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:#000}.mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12)}[dir=rtl][_ngcontent-%COMP%]   .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-group-appearance-standard.mat-button-toggle-vertical[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:none;border-top:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#616161}.mat-button-toggle-checked.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121}.mat-button-toggle-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.26);background-color:#eee}.mat-button-toggle-disabled.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{background:#fff}.mat-button-toggle-disabled.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#bdbdbd}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{border:solid 1px rgba(0,0,0,.12)}.mat-card[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-card[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-card.mat-card-flat[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-card-subtitle[_ngcontent-%COMP%]{color:#616161}.mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#616161}.mat-checkbox-checkmark[_ngcontent-%COMP%]{fill:#fafafa}.mat-checkbox-checkmark-path[_ngcontent-%COMP%]{stroke:#fafafa !important}.mat-checkbox-mixedmark[_ngcontent-%COMP%]{background-color:#fafafa}.mat-checkbox-indeterminate.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f57c00}.mat-checkbox-indeterminate.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#ff9800}.mat-checkbox-indeterminate.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f44336}.mat-checkbox-disabled.mat-checkbox-checked[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-disabled.mat-checkbox-indeterminate[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]:not(.mat-checkbox-checked)   .mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{color:#616161}.mat-checkbox[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%]{background:#f57c00}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%]{background:#ff9800}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%]{background:#f44336}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#212121}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#212121;opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled):active{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled)   .mat-chip-remove[_ngcontent-%COMP%]:hover{opacity:.54}.mat-chip.mat-standard-chip.mat-chip-disabled[_ngcontent-%COMP%]{opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]::after{background:#000}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-table[_ngcontent-%COMP%]{background:#fff}.mat-table[_ngcontent-%COMP%]   thead[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tbody[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tfoot[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], [mat-header-row][_ngcontent-%COMP%], [mat-row][_ngcontent-%COMP%], [mat-footer-row][_ngcontent-%COMP%], .mat-table-sticky[_ngcontent-%COMP%]{background:inherit}mat-row[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], th.mat-header-cell[_ngcontent-%COMP%], td.mat-cell[_ngcontent-%COMP%], td.mat-footer-cell[_ngcontent-%COMP%]{border-bottom-color:rgba(0,0,0,.12)}.mat-header-cell[_ngcontent-%COMP%]{color:#616161}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{color:#212121}.mat-calendar-arrow[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.54)}.mat-datepicker-toggle[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-next-button[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-previous-button[_ngcontent-%COMP%]{color:rgba(0,0,0,.54)}.mat-calendar-table-header[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-calendar-table-header-divider[_ngcontent-%COMP%]::after{background:rgba(0,0,0,.12)}.mat-calendar-body-label[_ngcontent-%COMP%]{color:#616161}.mat-calendar-body-cell-content[_ngcontent-%COMP%]{color:#212121;border-color:transparent}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){color:#757575}.mat-calendar-body-cell[_ngcontent-%COMP%]:not(.mat-calendar-body-disabled):hover > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-keyboard-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-program-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){background-color:rgba(0,0,0,.04)}.mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.38)}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.18)}.mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.4)}.mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12);background-color:#fff;color:#212121}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.4)}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.4)}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content-touch[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-datepicker-toggle-active[_ngcontent-%COMP%]{color:#f57c00}.mat-datepicker-toggle-active.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-datepicker-toggle-active.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-dialog-container[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-divider[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-divider-vertical[_ngcontent-%COMP%]{border-right-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-expansion-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-action-row[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-keyboard-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-program-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded)   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover:not([aria-disabled=true]){background:rgba(0,0,0,.04)}@media(hover: none){.mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded):not([aria-disabled=true])   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover{background:#fff}}.mat-expansion-panel-header-title[_ngcontent-%COMP%]{color:#212121}.mat-expansion-panel-header-description[_ngcontent-%COMP%], .mat-expansion-indicator[_ngcontent-%COMP%]::after{color:#616161}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-title[_ngcontent-%COMP%], .mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-description[_ngcontent-%COMP%]{color:inherit}.mat-form-field-label[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-hint[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-focused[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-ripple[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid)   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f57c00}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-accent   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#ff9800}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-warn   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#f44336}.mat-error[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-hint[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-legacy.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-standard[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-standard.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.04)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.02)}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:transparent}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.12)}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:rgba(0,0,0,.87)}.mat-form-field-appearance-outline.mat-focused[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field-appearance-outline.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-appearance-outline.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-invalid.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.06)}.mat-icon.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-icon.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-icon.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-type-mat-native-select[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#616161}.mat-input-element[_ngcontent-%COMP%]:disabled, .mat-form-field-type-mat-native-select.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#757575}.mat-input-element[_ngcontent-%COMP%]{caret-color:#f57c00}.mat-input-element[_ngcontent-%COMP%]::placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-moz-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-webkit-input-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]:-ms-input-placeholder{color:rgba(97,97,97,.42)}.mat-accent[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%], .mat-form-field-invalid[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#f44336}.mat-form-field-type-mat-native-select.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{color:#616161}.mat-list-item-disabled[_ngcontent-%COMP%]{background-color:#eee}.mat-list-option[_ngcontent-%COMP%]:hover, .mat-list-option[_ngcontent-%COMP%]:focus, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.04)}.mat-list-single-selected-option[_ngcontent-%COMP%], .mat-list-single-selected-option[_ngcontent-%COMP%]:hover, .mat-list-single-selected-option[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.12)}.mat-menu-panel[_ngcontent-%COMP%]{background:#fff}.mat-menu-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-menu-item[_ngcontent-%COMP%]{background:transparent;color:#212121}.mat-menu-item[disabled][_ngcontent-%COMP%], .mat-menu-item[disabled][_ngcontent-%COMP%]::after{color:rgba(0,0,0,.38)}.mat-menu-item[_ngcontent-%COMP%]   .mat-icon-no-color[_ngcontent-%COMP%], .mat-menu-item-submenu-trigger[_ngcontent-%COMP%]::after{color:rgba(0,0,0,.54)}.mat-menu-item[_ngcontent-%COMP%]:hover:not([disabled]), .mat-menu-item.cdk-program-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item.cdk-keyboard-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item-highlighted[_ngcontent-%COMP%]:not([disabled]){background:rgba(0,0,0,.04)}.mat-paginator[_ngcontent-%COMP%]{background:#fff}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{color:#616161}.mat-paginator-decrement[_ngcontent-%COMP%], .mat-paginator-increment[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54);border-right:2px solid rgba(0,0,0,.54)}.mat-paginator-first[_ngcontent-%COMP%], .mat-paginator-last[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54)}.mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-decrement[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-increment[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-first[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-last[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f57c00}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#ff9800}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f44336}.mat-progress-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f57c00}.mat-progress-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#ff9800}.mat-progress-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f44336}.mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#616161}.mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f57c00}.mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#ff9800}.mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#ff9800}.mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f44336}.mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f44336}.mat-radio-button.mat-radio-disabled.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-label-content[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-radio-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-select-value[_ngcontent-%COMP%]{color:#212121}.mat-select-placeholder[_ngcontent-%COMP%]{color:rgba(97,97,97,.42)}.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%]{color:#757575}.mat-select-arrow[_ngcontent-%COMP%]{color:#616161}.mat-select-panel[_ngcontent-%COMP%]{background:#fff}.mat-select-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-select-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple){background:rgba(0,0,0,.12)}.mat-form-field.mat-focused.mat-primary[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-invalid[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#757575}.mat-drawer-container[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-drawer[_ngcontent-%COMP%]{background-color:#fff;color:#212121}.mat-drawer.mat-drawer-push[_ngcontent-%COMP%]{background-color:#fff}.mat-drawer[_ngcontent-%COMP%]:not(.mat-drawer-side){box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-drawer-side[_ngcontent-%COMP%]{border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-backdrop.mat-drawer-shown[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.6)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.54)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.54)}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.54)}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle[_ngcontent-%COMP%]:not(.mat-checked)   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-slide-toggle-thumb[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12);background-color:#fafafa}.mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-primary[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f57c00}.mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-accent[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#ff9800}.mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-warn[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f44336}.mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.2)}.mat-slider[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%], .cdk-focused[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-disabled[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]   .mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing)   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26);background-color:transparent}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26)}.mat-slider-has-ticks[_ngcontent-%COMP%]   .mat-slider-wrapper[_ngcontent-%COMP%]::after{border-color:rgba(0,0,0,.7)}.mat-slider-horizontal[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to right, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent);background-image:-moz-repeating-linear-gradient(0.0001deg, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-slider-vertical[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to bottom, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-step-header.cdk-keyboard-focused[_ngcontent-%COMP%], .mat-step-header.cdk-program-focused[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]:hover{background-color:rgba(0,0,0,.04)}@media(hover: none){.mat-step-header[_ngcontent-%COMP%]:hover{background:none}}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-optional[_ngcontent-%COMP%]{color:#616161}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon[_ngcontent-%COMP%]{background-color:#616161;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-selected[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-done[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-edit[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-error[_ngcontent-%COMP%]{background-color:transparent;color:#f44336}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-active[_ngcontent-%COMP%]{color:#212121}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-error[_ngcontent-%COMP%]{color:#f44336}.mat-stepper-horizontal[_ngcontent-%COMP%], .mat-stepper-vertical[_ngcontent-%COMP%]{background-color:#fff}.mat-stepper-vertical-line[_ngcontent-%COMP%]::before{border-left-color:rgba(0,0,0,.12)}.mat-horizontal-stepper-header[_ngcontent-%COMP%]::before, .mat-horizontal-stepper-header[_ngcontent-%COMP%]::after, .mat-stepper-horizontal-line[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-sort-header-arrow[_ngcontent-%COMP%]{color:#616161}.mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-header[_ngcontent-%COMP%]{border-bottom:1px solid rgba(0,0,0,.12)}.mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%]{border-top:1px solid rgba(0,0,0,.12);border-bottom:none}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{color:#212121}.mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:#757575}.mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#212121}.mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#757575}.mat-tab-group[class*=mat-background-][_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar[class*=mat-background-][_ngcontent-%COMP%]{border-bottom:none;border-top:none}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-toolbar[_ngcontent-%COMP%]{background:#f5f5f5;color:#212121}.mat-toolbar.mat-primary[_ngcontent-%COMP%]{background:#f57c00;color:#fff}.mat-toolbar.mat-accent[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-toolbar.mat-warn[_ngcontent-%COMP%]{background:#f44336;color:#fff}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:currentColor}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:inherit}.mat-toolbar[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:currentColor}.mat-tooltip[_ngcontent-%COMP%]{background:rgba(97,97,97,.9)}.mat-tree[_ngcontent-%COMP%]{background:#fff}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{color:#212121}.mat-snack-bar-container[_ngcontent-%COMP%]{color:rgba(255,255,255,.7);background:#323232;box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-simple-snackbar-action[_ngcontent-%COMP%]{color:#ff9800}[_nghost-%COMP%]{color:#616161;font-size:12px}section[_ngcontent-%COMP%]{border-bottom:1px solid #ebebeb;padding:16px}.section-title[_ngcontent-%COMP%]{color:#212121;text-transform:uppercase;font-weight:500;font-size:13px;line-height:normal;margin:0 0 12px 0}section[_ngcontent-%COMP%]   .control-row[_ngcontent-%COMP%]:not(:last-child){margin-bottom:12px}.control-name[_ngcontent-%COMP%]{margin-bottom:8px}.slider-row[_ngcontent-%COMP%]{display:flex;align-items:center;height:28px}.slider-row[_ngcontent-%COMP%]   .reset-button[_ngcontent-%COMP%]{margin-left:6px}.slider-row[_ngcontent-%COMP%]   .slider-input[_ngcontent-%COMP%]{border:1px solid #8e98a3;border-radius:2px;box-sizing:border-box;height:100%;margin-left:12px;padding:0 4px}mat-slider[_ngcontent-%COMP%]{flex:1;margin-left:-8px;margin-right:-8px}tb-dropdown[_ngcontent-%COMP%]{display:block}"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/views/right_pane/settings_view_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class SettingsViewContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.tooltipSort$ = this.store.select(getMetricsTooltipSort);
            this.ignoreOutliers$ = this.store.select(getMetricsIgnoreOutliers);
            this.xAxisType$ = this.store.select(getMetricsXAxisType);
            this.histogramMode$ = this.store.select(getMetricsHistogramMode);
            this.scalarSmoothing$ = this.store.select(getMetricsScalarSmoothing);
            this.imageBrightnessInMilli$ = this.store.select(getMetricsImageBrightnessInMilli);
            this.imageContrastInMilli$ = this.store.select(getMetricsImageContrastInMilli);
            this.imageShowActualSize$ = this.store.select(getMetricsImageShowActualSize);
        }
        /**
         * @param {?} sort
         * @return {?}
         */
        onTooltipSortChanged(sort) {
            this.store.dispatch(metricsChangeTooltipSort({ sort }));
        }
        /**
         * @return {?}
         */
        onIgnoreOutliersChanged() {
            this.store.dispatch(metricsToggleIgnoreOutliers());
        }
        /**
         * @param {?} xAxisType
         * @return {?}
         */
        onXAxisTypeChanged(xAxisType) {
            this.store.dispatch(metricsChangeXAxisType({ xAxisType }));
        }
        /**
         * @param {?} histogramMode
         * @return {?}
         */
        onHistogramModeChanged(histogramMode) {
            this.store.dispatch(metricsChangeHistogramMode({ histogramMode }));
        }
        /**
         * @param {?} smoothing
         * @return {?}
         */
        onScalarSmoothingChanged(smoothing) {
            this.store.dispatch(metricsChangeScalarSmoothing({ smoothing }));
        }
        /**
         * @param {?} brightnessInMilli
         * @return {?}
         */
        onImageBrightnessInMilliChanged(brightnessInMilli) {
            this.store.dispatch(metricsChangeImageBrightness({ brightnessInMilli }));
        }
        /**
         * @return {?}
         */
        onImageBrightnessReset() {
            this.store.dispatch(metricsResetImageBrightness());
        }
        /**
         * @return {?}
         */
        onImageContrastReset() {
            this.store.dispatch(metricsResetImageContrast());
        }
        /**
         * @param {?} contrastInMilli
         * @return {?}
         */
        onImageContrastInMilliChanged(contrastInMilli) {
            this.store.dispatch(metricsChangeImageContrast({ contrastInMilli }));
        }
        /**
         * @return {?}
         */
        onImageShowActualSizeChanged() {
            this.store.dispatch(metricsToggleImageShowActualSize());
        }
    }
    SettingsViewContainer.decorators = [
        { type: Component, args: [{
                    selector: 'metrics-dashboard-settings',
                    template: `
    <metrics-dashboard-settings-component
      [tooltipSort]="tooltipSort$ | async"
      (tooltipSortChanged)="onTooltipSortChanged($event)"
      [ignoreOutliers]="ignoreOutliers$ | async"
      (ignoreOutliersChanged)="onIgnoreOutliersChanged()"
      [xAxisType]="xAxisType$ | async"
      (xAxisTypeChanged)="onXAxisTypeChanged($event)"
      [histogramMode]="histogramMode$ | async"
      (histogramModeChanged)="onHistogramModeChanged($event)"
      [scalarSmoothing]="scalarSmoothing$ | async"
      (scalarSmoothingChanged)="onScalarSmoothingChanged($event)"
      [imageBrightnessInMilli]="imageBrightnessInMilli$ | async"
      (imageBrightnessInMilliChanged)="onImageBrightnessInMilliChanged($event)"
      (imageBrightnessReset)="onImageBrightnessReset()"
      [imageContrastInMilli]="imageContrastInMilli$ | async"
      (imageContrastInMilliChanged)="onImageContrastInMilliChanged($event)"
      (imageContrastReset)="onImageContrastReset()"
      [imageShowActualSize]="imageShowActualSize$ | async"
      (imageShowActualSizeChanged)="onImageShowActualSizeChanged()"
    >
    </metrics-dashboard-settings-component>
  `,
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    SettingsViewContainer.ctorParameters = () => [
        { type: Store }
    ];
    /** @nocollapse */ SettingsViewContainer.ɵfac = function SettingsViewContainer_Factory(t) { return new (t || SettingsViewContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ SettingsViewContainer.ɵcmp = ɵɵdefineComponent({ type: SettingsViewContainer, selectors: [["metrics-dashboard-settings"]], decls: 9, vars: 24, consts: [[3, "tooltipSort", "ignoreOutliers", "xAxisType", "histogramMode", "scalarSmoothing", "imageBrightnessInMilli", "imageContrastInMilli", "imageShowActualSize", "tooltipSortChanged", "ignoreOutliersChanged", "xAxisTypeChanged", "histogramModeChanged", "scalarSmoothingChanged", "imageBrightnessInMilliChanged", "imageBrightnessReset", "imageContrastInMilliChanged", "imageContrastReset", "imageShowActualSizeChanged"]], template: function SettingsViewContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "metrics-dashboard-settings-component", 0);
            ɵɵlistener("tooltipSortChanged", function SettingsViewContainer_Template_metrics_dashboard_settings_component_tooltipSortChanged_0_listener($event) { return ctx.onTooltipSortChanged($event); })("ignoreOutliersChanged", function SettingsViewContainer_Template_metrics_dashboard_settings_component_ignoreOutliersChanged_0_listener() { return ctx.onIgnoreOutliersChanged(); })("xAxisTypeChanged", function SettingsViewContainer_Template_metrics_dashboard_settings_component_xAxisTypeChanged_0_listener($event) { return ctx.onXAxisTypeChanged($event); })("histogramModeChanged", function SettingsViewContainer_Template_metrics_dashboard_settings_component_histogramModeChanged_0_listener($event) { return ctx.onHistogramModeChanged($event); })("scalarSmoothingChanged", function SettingsViewContainer_Template_metrics_dashboard_settings_component_scalarSmoothingChanged_0_listener($event) { return ctx.onScalarSmoothingChanged($event); })("imageBrightnessInMilliChanged", function SettingsViewContainer_Template_metrics_dashboard_settings_component_imageBrightnessInMilliChanged_0_listener($event) { return ctx.onImageBrightnessInMilliChanged($event); })("imageBrightnessReset", function SettingsViewContainer_Template_metrics_dashboard_settings_component_imageBrightnessReset_0_listener() { return ctx.onImageBrightnessReset(); })("imageContrastInMilliChanged", function SettingsViewContainer_Template_metrics_dashboard_settings_component_imageContrastInMilliChanged_0_listener($event) { return ctx.onImageContrastInMilliChanged($event); })("imageContrastReset", function SettingsViewContainer_Template_metrics_dashboard_settings_component_imageContrastReset_0_listener() { return ctx.onImageContrastReset(); })("imageShowActualSizeChanged", function SettingsViewContainer_Template_metrics_dashboard_settings_component_imageShowActualSizeChanged_0_listener() { return ctx.onImageShowActualSizeChanged(); });
            ɵɵpipe(1, "async");
            ɵɵpipe(2, "async");
            ɵɵpipe(3, "async");
            ɵɵpipe(4, "async");
            ɵɵpipe(5, "async");
            ɵɵpipe(6, "async");
            ɵɵpipe(7, "async");
            ɵɵpipe(8, "async");
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵproperty("tooltipSort", ɵɵpipeBind1(1, 8, ctx.tooltipSort$))("ignoreOutliers", ɵɵpipeBind1(2, 10, ctx.ignoreOutliers$))("xAxisType", ɵɵpipeBind1(3, 12, ctx.xAxisType$))("histogramMode", ɵɵpipeBind1(4, 14, ctx.histogramMode$))("scalarSmoothing", ɵɵpipeBind1(5, 16, ctx.scalarSmoothing$))("imageBrightnessInMilli", ɵɵpipeBind1(6, 18, ctx.imageBrightnessInMilli$))("imageContrastInMilli", ɵɵpipeBind1(7, 20, ctx.imageContrastInMilli$))("imageShowActualSize", ɵɵpipeBind1(8, 22, ctx.imageShowActualSize$));
        } }, directives: [SettingsViewComponent], pipes: [AsyncPipe], encapsulation: 2, changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/views/right_pane/right_pane_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class RightPaneComponent {
    }
    RightPaneComponent.decorators = [
        { type: Component, args: [{
                    selector: 'metrics-dashboard-right-pane',
                    template: `<metrics-dashboard-settings></metrics-dashboard-settings>`,
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */ RightPaneComponent.ɵfac = function RightPaneComponent_Factory(t) { return new (t || RightPaneComponent)(); };
    /** @nocollapse */ RightPaneComponent.ɵcmp = ɵɵdefineComponent({ type: RightPaneComponent, selectors: [["metrics-dashboard-right-pane"]], decls: 1, vars: 0, template: function RightPaneComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelement(0, "metrics-dashboard-settings");
        } }, directives: [SettingsViewContainer], encapsulation: 2, changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/views/main_view/main_view_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    var I18N_0$7;
    if (typeof ngI18nClosureMode !== "undefined" && ngI18nClosureMode) {
        /**
         * @desc Label on a toolbar button to toggle the settings side pane.
         */ 
        const MSG_EXTERNAL_7659285445580838925$$TENSORBOARD_WEBAPP_METRICS_VIEWS_MAIN_VIEW_MAIN_VIEW_COMPONENT_TS_1 = goog.getMsg("Toggle settings side pane");
        I18N_0$7 = MSG_EXTERNAL_7659285445580838925$$TENSORBOARD_WEBAPP_METRICS_VIEWS_MAIN_VIEW_MAIN_VIEW_COMPONENT_TS_1;
    }
    else {
        I18N_0$7 = $localize `:Label on a toolbar button to toggle the settings side pane.␟d3516db6bbe6860a55beab66e4969dac625b8d72␟7659285445580838925:Toggle settings side pane`;
    }
    const _c2$d = ["aria-label", I18N_0$7];
    function MainViewComponent_metrics_filtered_view_9_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "metrics-filtered-view", 7);
    } if (rf & 2) {
        const ctx_r0 = ɵɵnextContext();
        ɵɵproperty("cardObserver", ctx_r0.cardObserver);
    } }
    function MainViewComponent_metrics_pinned_view_10_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "metrics-pinned-view", 7);
    } if (rf & 2) {
        const ctx_r1 = ɵɵnextContext();
        ɵɵproperty("cardObserver", ctx_r1.cardObserver);
    } }
    var I18N_3$5;
    if (typeof ngI18nClosureMode !== "undefined" && ngI18nClosureMode) {
        /**
         * @desc Label on a button to close the settings side pane.
         */ 
        const MSG_EXTERNAL_8156766997747165871$$TENSORBOARD_WEBAPP_METRICS_VIEWS_MAIN_VIEW_MAIN_VIEW_COMPONENT_TS__4 = goog.getMsg("Close side pane");
        I18N_3$5 = MSG_EXTERNAL_8156766997747165871$$TENSORBOARD_WEBAPP_METRICS_VIEWS_MAIN_VIEW_MAIN_VIEW_COMPONENT_TS__4;
    }
    else {
        I18N_3$5 = $localize `:Label on a button to close the settings side pane.␟04521dc0b6a65cf5c382944c9a8b4b844a3e9598␟8156766997747165871:Close side pane`;
    }
    const _c5$7 = ["aria-label", I18N_3$5];
    function MainViewComponent_div_12_Template(rf, ctx) { if (rf & 1) {
        const _r4 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "div", 9);
        ɵɵelementStart(1, "div", 10);
        ɵɵelementStart(2, "h2", 11);
        ɵɵtext(3, "Settings");
        ɵɵelementEnd();
        ɵɵelementStart(4, "button", 12);
        ɵɵi18nAttributes(5, _c5$7);
        ɵɵlistener("click", function MainViewComponent_div_12_Template_button_click_4_listener() { ɵɵrestoreView(_r4); const ctx_r3 = ɵɵnextContext(); return ctx_r3.onCloseSidepaneButtonClicked.emit(); });
        ɵɵelement(6, "mat-icon", 13);
        ɵɵelementEnd();
        ɵɵelementEnd();
        ɵɵelement(7, "metrics-dashboard-right-pane");
        ɵɵelementEnd();
    } }
    const _c6$3 = function (a0) { return { "checked": a0, "settings-button": true }; };
    class MainViewComponent {
        /**
         * @param {?} host
         */
        constructor(host) {
            this.host = host;
            this.onSettingsButtonClicked = new EventEmitter();
            this.onCloseSidepaneButtonClicked = new EventEmitter();
            /**
             * Load cards that are not yet visible, if they are roughly 1 card row away in
             * scroll distance.
             */
            this.cardObserver = new CardObserver(this.host.nativeElement, '600px 0px 600px 0px');
        }
    }
    MainViewComponent.decorators = [
        { type: Component, args: [{
                    selector: 'metrics-main-view-component',
                    templateUrl: 'main_view_component.ng.html',
                    styleUrls: ['main_view_component.css'],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    MainViewComponent.ctorParameters = () => [
        { type: ElementRef }
    ];
    MainViewComponent.propDecorators = {
        showFilteredView: [{ type: Input }],
        isSidepaneOpen: [{ type: Input }],
        onSettingsButtonClicked: [{ type: Output }],
        onCloseSidepaneButtonClicked: [{ type: Output }]
    };
    /** @nocollapse */ MainViewComponent.ɵfac = function MainViewComponent_Factory(t) { return new (t || MainViewComponent)(ɵɵdirectiveInject(ElementRef)); };
    /** @nocollapse */ MainViewComponent.ɵcmp = ɵɵdefineComponent({ type: MainViewComponent, selectors: [["metrics-main-view-component"]], inputs: { showFilteredView: "showFilteredView", isSidepaneOpen: "isSidepaneOpen" }, outputs: { onSettingsButtonClicked: "onSettingsButtonClicked", onCloseSidepaneButtonClicked: "onCloseSidepaneButtonClicked" }, decls: 13, vars: 10, consts: [[1, "toolbar"], [1, "right-items"], ["mat-stroked-button", "", 3, "ngClass", "click", 6, "aria-label"], ["svgIcon", "settings_24px"], [1, "split-content"], [1, "main"], [3, "cardObserver", 4, "ngIf"], [3, "cardObserver"], ["class", "sidebar", 4, "ngIf"], [1, "sidebar"], [1, "header"], [1, "title"], ["mat-icon-button", "", 3, "click", 6, "aria-label"], ["svgIcon", "close_24px"]], template: function MainViewComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "div", 0);
            ɵɵelement(1, "metrics-tag-filter");
            ɵɵelementStart(2, "div", 1);
            ɵɵelementStart(3, "button", 2);
            ɵɵi18nAttributes(4, _c2$d);
            ɵɵlistener("click", function MainViewComponent_Template_button_click_3_listener() { return ctx.onSettingsButtonClicked.emit(); });
            ɵɵelement(5, "mat-icon", 3);
            ɵɵtext(6, " Settings ");
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementStart(7, "div", 4);
            ɵɵelementStart(8, "div", 5);
            ɵɵtemplate(9, MainViewComponent_metrics_filtered_view_9_Template, 1, 1, "metrics-filtered-view", 6);
            ɵɵtemplate(10, MainViewComponent_metrics_pinned_view_10_Template, 1, 1, "metrics-pinned-view", 6);
            ɵɵelement(11, "metrics-card-groups", 7);
            ɵɵelementEnd();
            ɵɵtemplate(12, MainViewComponent_div_12_Template, 8, 0, "div", 8);
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵadvance(3);
            ɵɵproperty("ngClass", ɵɵpureFunction1(8, _c6$3, ctx.isSidepaneOpen));
            ɵɵattribute("aria-pressed", ctx.isSidepaneOpen);
            ɵɵadvance(6);
            ɵɵproperty("ngIf", ctx.showFilteredView);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", !ctx.showFilteredView);
            ɵɵadvance(1);
            ɵɵstyleProp("display", ctx.showFilteredView ? "none" : "");
            ɵɵproperty("cardObserver", ctx.cardObserver);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx.isSidepaneOpen);
        } }, directives: [MetricsFilterInputContainer, MatButton, NgClass, MatIcon, NgIf, CardGroupsContainer, FilteredViewContainer, PinnedViewContainer, RightPaneComponent], styles: [".mat-badge-content[_ngcontent-%COMP%]{font-weight:600;font-size:12px;font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:9px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:24px}.mat-h1[_ngcontent-%COMP%], .mat-headline[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%]{font:400 24px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h2[_ngcontent-%COMP%], .mat-title[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h3[_ngcontent-%COMP%], .mat-subheading-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%]{font:400 16px/28px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h4[_ngcontent-%COMP%], .mat-subheading-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%]{font:400 15px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h5[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%]{font:400 calc(14px * 0.83)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-h6[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:400 calc(14px * 0.67)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-body-strong[_ngcontent-%COMP%], .mat-body-2[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   p[_ngcontent-%COMP%]{margin:0 0 12px}.mat-small[_ngcontent-%COMP%], .mat-caption[_ngcontent-%COMP%]{font:400 12px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-display-4[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-4[_ngcontent-%COMP%]{font:300 112px/112px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.05em;margin:0 0 56px}.mat-display-3[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-3[_ngcontent-%COMP%]{font:400 56px/56px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.02em;margin:0 0 64px}.mat-display-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-2[_ngcontent-%COMP%]{font:400 45px/48px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.005em;margin:0 0 64px}.mat-display-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-1[_ngcontent-%COMP%]{font:400 34px/40px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 64px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%], .mat-flat-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-button-toggle[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card-title[_ngcontent-%COMP%]{font-size:24px;font-weight:500}.mat-card-header[_ngcontent-%COMP%]   .mat-card-title[_ngcontent-%COMP%]{font-size:20px}.mat-card-subtitle[_ngcontent-%COMP%], .mat-card-content[_ngcontent-%COMP%]{font-size:14px}.mat-checkbox[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-checkbox-layout[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{line-height:24px}.mat-chip[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-chip[_ngcontent-%COMP%]   .mat-chip-trailing-icon.mat-icon[_ngcontent-%COMP%], .mat-chip[_ngcontent-%COMP%]   .mat-chip-remove.mat-icon[_ngcontent-%COMP%]{font-size:18px}.mat-table[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-header-cell[_ngcontent-%COMP%]{font-size:12px;font-weight:500}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{font-size:14px}.mat-calendar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-calendar-body[_ngcontent-%COMP%]{font-size:13px}.mat-calendar-body-label[_ngcontent-%COMP%], .mat-calendar-period-button[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-calendar-table-header[_ngcontent-%COMP%]   th[_ngcontent-%COMP%]{font-size:11px;font-weight:400}.mat-dialog-title[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-expansion-panel-header[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:15px;font-weight:400}.mat-expansion-panel-content[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field[_ngcontent-%COMP%]{font-size:inherit;font-weight:400;line-height:1.125;font-family:Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.34375em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{font-size:150%;line-height:1.125}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]{height:1.5em;width:1.5em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{height:1.125em;line-height:1.125}.mat-form-field-infix[_ngcontent-%COMP%]{padding:.5em 0;border-top:.84375em solid transparent}.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34375em) scale(0.75);width:133.3333333333%}.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34374em) scale(0.75);width:133.3333433333%}.mat-form-field-label-wrapper[_ngcontent-%COMP%]{top:-0.84375em;padding-top:.84375em}.mat-form-field-label[_ngcontent-%COMP%]{top:1.34375em}.mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.34375em}.mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{font-size:75%;margin-top:.6666666667em;top:calc(100% - 1.7916666667em)}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.4375em 0}.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.001px);-ms-transform:translateY(-1.28125em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00101px);-ms-transform:translateY(-1.28124em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00102px);-ms-transform:translateY(-1.28123em) scale(0.75);width:133.3333533333%}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.28125em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{margin-top:.5416666667em;top:calc(100% - 1.6666666667em)}@media print{.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28122em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28121em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.2812em) scale(0.75)}}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.25em 0 .75em 0}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.09375em;margin-top:-0.5em}.mat-form-field-appearance-fill.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59374em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:1em 0 1em 0}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.84375em;margin-top:-0.25em}.mat-form-field-appearance-outline.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59374em) scale(0.75);width:133.3333433333%}.mat-grid-tile-header[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]{font-size:14px}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2), .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}input.mat-input-element[_ngcontent-%COMP%]{margin-top:-0.0625em}.mat-menu-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:400}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px}.mat-radio-button[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select-trigger[_ngcontent-%COMP%]{height:1.125em}.mat-slide-toggle-content[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-slider-thumb-label-text[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-stepper-vertical[_ngcontent-%COMP%], .mat-stepper-horizontal[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-step-label[_ngcontent-%COMP%]{font-size:14px;font-weight:400}.mat-step-sub-label-error[_ngcontent-%COMP%]{font-weight:normal}.mat-step-label-error[_ngcontent-%COMP%]{font-size:14px}.mat-step-label-selected[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-tab-group[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-toolbar[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0}.mat-tooltip[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:10px;padding-top:6px;padding-bottom:6px}.mat-tooltip-handset[_ngcontent-%COMP%]{font-size:14px;padding-top:8px;padding-bottom:8px}.mat-list-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:16px}.mat-optgroup-label[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-simple-snackbar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px}.mat-simple-snackbar-action[_ngcontent-%COMP%]{line-height:1;font-family:inherit;font-size:inherit;font-weight:500}.mat-tree[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{font-weight:400;font-size:14px}.mat-ripple[_ngcontent-%COMP%]{overflow:hidden;position:relative}.mat-ripple[_ngcontent-%COMP%]:not(:empty){transform:translateZ(0)}.mat-ripple.mat-ripple-unbounded[_ngcontent-%COMP%]{overflow:visible}.mat-ripple-element[_ngcontent-%COMP%]{position:absolute;border-radius:50%;pointer-events:none;transition:opacity,transform 0ms cubic-bezier(0, 0, 0.2, 1);transform:scale(0)}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{display:none}.cdk-visually-hidden[_ngcontent-%COMP%]{border:0;clip:rect(0 0 0 0);height:1px;margin:-1px;overflow:hidden;padding:0;position:absolute;width:1px;outline:0;-webkit-appearance:none;-moz-appearance:none}.cdk-overlay-container[_ngcontent-%COMP%], .cdk-global-overlay-wrapper[_ngcontent-%COMP%]{pointer-events:none;top:0;left:0;height:100%;width:100%}.cdk-overlay-container[_ngcontent-%COMP%]{position:fixed;z-index:1000}.cdk-overlay-container[_ngcontent-%COMP%]:empty{display:none}.cdk-global-overlay-wrapper[_ngcontent-%COMP%]{display:flex;position:absolute;z-index:1000}.cdk-overlay-pane[_ngcontent-%COMP%]{position:absolute;pointer-events:auto;box-sizing:border-box;z-index:1000;display:flex;max-width:100%;max-height:100%}.cdk-overlay-backdrop[_ngcontent-%COMP%]{position:absolute;top:0;bottom:0;left:0;right:0;z-index:1000;pointer-events:auto;-webkit-tap-highlight-color:transparent;transition:opacity 400ms cubic-bezier(0.25, 0.8, 0.25, 1);opacity:0}.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:1}@media screen and (-ms-high-contrast: active){.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:.6}}.cdk-overlay-dark-backdrop[_ngcontent-%COMP%]{background:rgba(0,0,0,.32)}.cdk-overlay-transparent-backdrop[_ngcontent-%COMP%], .cdk-overlay-transparent-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:0}.cdk-overlay-connected-position-bounding-box[_ngcontent-%COMP%]{position:absolute;z-index:1000;display:flex;flex-direction:column;min-width:1px;min-height:1px}.cdk-global-scrollblock[_ngcontent-%COMP%]{position:fixed;width:100%;overflow-y:scroll}@keyframes cdk-text-field-autofill-start{}@keyframes cdk-text-field-autofill-end{}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:-webkit-autofill{animation:cdk-text-field-autofill-start 0s 1ms}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:not(:-webkit-autofill){animation:cdk-text-field-autofill-end 0s 1ms}textarea.cdk-textarea-autosize[_ngcontent-%COMP%]{resize:none}textarea.cdk-textarea-autosize-measuring[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:auto !important;overflow:hidden !important}textarea.cdk-textarea-autosize-measuring-firefox[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:0 !important}.mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.1)}.mat-option[_ngcontent-%COMP%]{color:#212121}.mat-option[_ngcontent-%COMP%]:hover:not(.mat-option-disabled), .mat-option[_ngcontent-%COMP%]:focus:not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple):not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-active[_ngcontent-%COMP%]{background:rgba(0,0,0,.04);color:#212121}.mat-option.mat-option-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-primary[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f57c00}.mat-accent[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f44336}.mat-optgroup-label[_ngcontent-%COMP%]{color:#616161}.mat-optgroup-disabled[_ngcontent-%COMP%]   .mat-optgroup-label[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-pseudo-checkbox[_ngcontent-%COMP%]{color:#616161}.mat-pseudo-checkbox[_ngcontent-%COMP%]::after{color:#fafafa}.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{color:#b0b0b0}.mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f57c00}.mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f44336}.mat-pseudo-checkbox-checked.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{background:#b0b0b0}.mat-elevation-z0[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z1[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z2[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z3[_ngcontent-%COMP%]{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z4[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z5[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 5px 8px 0px rgba(0, 0, 0, 0.14),0px 1px 14px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z6[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z7[_ngcontent-%COMP%]{box-shadow:0px 4px 5px -2px rgba(0, 0, 0, 0.2),0px 7px 10px 1px rgba(0, 0, 0, 0.14),0px 2px 16px 1px rgba(0, 0, 0, 0.12)}.mat-elevation-z8[_ngcontent-%COMP%]{box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z9[_ngcontent-%COMP%]{box-shadow:0px 5px 6px -3px rgba(0, 0, 0, 0.2),0px 9px 12px 1px rgba(0, 0, 0, 0.14),0px 3px 16px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z10[_ngcontent-%COMP%]{box-shadow:0px 6px 6px -3px rgba(0, 0, 0, 0.2),0px 10px 14px 1px rgba(0, 0, 0, 0.14),0px 4px 18px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z11[_ngcontent-%COMP%]{box-shadow:0px 6px 7px -4px rgba(0, 0, 0, 0.2),0px 11px 15px 1px rgba(0, 0, 0, 0.14),0px 4px 20px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z12[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z13[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 13px 19px 2px rgba(0, 0, 0, 0.14),0px 5px 24px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z14[_ngcontent-%COMP%]{box-shadow:0px 7px 9px -4px rgba(0, 0, 0, 0.2),0px 14px 21px 2px rgba(0, 0, 0, 0.14),0px 5px 26px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z15[_ngcontent-%COMP%]{box-shadow:0px 8px 9px -5px rgba(0, 0, 0, 0.2),0px 15px 22px 2px rgba(0, 0, 0, 0.14),0px 6px 28px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z16[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z17[_ngcontent-%COMP%]{box-shadow:0px 8px 11px -5px rgba(0, 0, 0, 0.2),0px 17px 26px 2px rgba(0, 0, 0, 0.14),0px 6px 32px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z18[_ngcontent-%COMP%]{box-shadow:0px 9px 11px -5px rgba(0, 0, 0, 0.2),0px 18px 28px 2px rgba(0, 0, 0, 0.14),0px 7px 34px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z19[_ngcontent-%COMP%]{box-shadow:0px 9px 12px -6px rgba(0, 0, 0, 0.2),0px 19px 29px 2px rgba(0, 0, 0, 0.14),0px 7px 36px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z20[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 20px 31px 3px rgba(0, 0, 0, 0.14),0px 8px 38px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z21[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 21px 33px 3px rgba(0, 0, 0, 0.14),0px 8px 40px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z22[_ngcontent-%COMP%]{box-shadow:0px 10px 14px -6px rgba(0, 0, 0, 0.2),0px 22px 35px 3px rgba(0, 0, 0, 0.14),0px 8px 42px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z23[_ngcontent-%COMP%]{box-shadow:0px 11px 14px -7px rgba(0, 0, 0, 0.2),0px 23px 36px 3px rgba(0, 0, 0, 0.14),0px 9px 44px 8px rgba(0, 0, 0, 0.12)}.mat-elevation-z24[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12)}.mat-app-background[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-theme-loaded-marker[_ngcontent-%COMP%]{display:none}.mat-autocomplete-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-autocomplete-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover){background:#fff}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover):not(.mat-option-disabled){color:#212121}.mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f57c00}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{outline:solid 1px;border-radius:0}.mat-badge-accent[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-badge-warn[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f44336}.mat-badge[_ngcontent-%COMP%]{position:relative}.mat-badge-hidden[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{display:none}.mat-badge-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#b9b9b9;color:#757575}.mat-badge-content[_ngcontent-%COMP%]{position:absolute;text-align:center;display:inline-block;border-radius:50%;transition:transform 200ms ease-in-out;transform:scale(0.6);overflow:hidden;white-space:nowrap;text-overflow:ellipsis;pointer-events:none}.ng-animate-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%], .mat-badge-content._mat-animation-noopable[_ngcontent-%COMP%]{transition:none}.mat-badge-content.mat-badge-active[_ngcontent-%COMP%]{transform:none}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:16px;height:16px;line-height:16px}.mat-badge-small.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-8px}.mat-badge-small.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-8px}.mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-16px}.mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-16px}.mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-8px}.mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-8px}.mat-badge-medium[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:22px;height:22px;line-height:22px}.mat-badge-medium.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-11px}.mat-badge-medium.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-11px}.mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-22px}.mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-22px}.mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-11px}.mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-11px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:28px;height:28px;line-height:28px}.mat-badge-large.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-14px}.mat-badge-large.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-14px}.mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-28px}.mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-28px}.mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-14px}.mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-14px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]{color:inherit;background:transparent}.mat-button.mat-primary[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-button[disabled][disabled][_ngcontent-%COMP%], .mat-icon-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-icon-button[disabled][disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-stroked-button[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f44336}.mat-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:transparent}.mat-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{opacity:.1;background-color:currentColor}.mat-button-focus-overlay[_ngcontent-%COMP%]{background:#000}.mat-stroked-button[_ngcontent-%COMP%]:not([disabled]){border-color:rgba(0,0,0,.12)}.mat-flat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{color:#212121;background-color:#fff}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-flat-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-stroked-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-flat-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-raised-button[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone[_ngcontent-%COMP%], .mat-button-toggle-group[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{box-shadow:none}.mat-button-toggle[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-button-toggle[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121;background:#fff}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:#000}.mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12)}[dir=rtl][_ngcontent-%COMP%]   .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-group-appearance-standard.mat-button-toggle-vertical[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:none;border-top:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#616161}.mat-button-toggle-checked.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121}.mat-button-toggle-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.26);background-color:#eee}.mat-button-toggle-disabled.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{background:#fff}.mat-button-toggle-disabled.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#bdbdbd}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{border:solid 1px rgba(0,0,0,.12)}.mat-card[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-card[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-card.mat-card-flat[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-card-subtitle[_ngcontent-%COMP%]{color:#616161}.mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#616161}.mat-checkbox-checkmark[_ngcontent-%COMP%]{fill:#fafafa}.mat-checkbox-checkmark-path[_ngcontent-%COMP%]{stroke:#fafafa !important}.mat-checkbox-mixedmark[_ngcontent-%COMP%]{background-color:#fafafa}.mat-checkbox-indeterminate.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f57c00}.mat-checkbox-indeterminate.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#ff9800}.mat-checkbox-indeterminate.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f44336}.mat-checkbox-disabled.mat-checkbox-checked[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-disabled.mat-checkbox-indeterminate[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]:not(.mat-checkbox-checked)   .mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{color:#616161}.mat-checkbox[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%]{background:#f57c00}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%]{background:#ff9800}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%]{background:#f44336}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#212121}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#212121;opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled):active{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled)   .mat-chip-remove[_ngcontent-%COMP%]:hover{opacity:.54}.mat-chip.mat-standard-chip.mat-chip-disabled[_ngcontent-%COMP%]{opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]::after{background:#000}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-table[_ngcontent-%COMP%]{background:#fff}.mat-table[_ngcontent-%COMP%]   thead[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tbody[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tfoot[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], [mat-header-row][_ngcontent-%COMP%], [mat-row][_ngcontent-%COMP%], [mat-footer-row][_ngcontent-%COMP%], .mat-table-sticky[_ngcontent-%COMP%]{background:inherit}mat-row[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], th.mat-header-cell[_ngcontent-%COMP%], td.mat-cell[_ngcontent-%COMP%], td.mat-footer-cell[_ngcontent-%COMP%]{border-bottom-color:rgba(0,0,0,.12)}.mat-header-cell[_ngcontent-%COMP%]{color:#616161}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{color:#212121}.mat-calendar-arrow[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.54)}.mat-datepicker-toggle[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-next-button[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-previous-button[_ngcontent-%COMP%]{color:rgba(0,0,0,.54)}.mat-calendar-table-header[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-calendar-table-header-divider[_ngcontent-%COMP%]::after{background:rgba(0,0,0,.12)}.mat-calendar-body-label[_ngcontent-%COMP%]{color:#616161}.mat-calendar-body-cell-content[_ngcontent-%COMP%]{color:#212121;border-color:transparent}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){color:#757575}.mat-calendar-body-cell[_ngcontent-%COMP%]:not(.mat-calendar-body-disabled):hover > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-keyboard-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-program-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){background-color:rgba(0,0,0,.04)}.mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.38)}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.18)}.mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.4)}.mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12);background-color:#fff;color:#212121}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.4)}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.4)}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content-touch[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-datepicker-toggle-active[_ngcontent-%COMP%]{color:#f57c00}.mat-datepicker-toggle-active.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-datepicker-toggle-active.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-dialog-container[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-divider[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-divider-vertical[_ngcontent-%COMP%]{border-right-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-expansion-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-action-row[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-keyboard-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-program-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded)   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover:not([aria-disabled=true]){background:rgba(0,0,0,.04)}@media(hover: none){.mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded):not([aria-disabled=true])   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover{background:#fff}}.mat-expansion-panel-header-title[_ngcontent-%COMP%]{color:#212121}.mat-expansion-panel-header-description[_ngcontent-%COMP%], .mat-expansion-indicator[_ngcontent-%COMP%]::after{color:#616161}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-title[_ngcontent-%COMP%], .mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-description[_ngcontent-%COMP%]{color:inherit}.mat-form-field-label[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-hint[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-focused[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-ripple[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid)   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f57c00}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-accent   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#ff9800}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-warn   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#f44336}.mat-error[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-hint[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-legacy.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-standard[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-standard.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.04)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.02)}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:transparent}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.12)}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:rgba(0,0,0,.87)}.mat-form-field-appearance-outline.mat-focused[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field-appearance-outline.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-appearance-outline.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-invalid.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.06)}.mat-icon.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-icon.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-icon.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-type-mat-native-select[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#616161}.mat-input-element[_ngcontent-%COMP%]:disabled, .mat-form-field-type-mat-native-select.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#757575}.mat-input-element[_ngcontent-%COMP%]{caret-color:#f57c00}.mat-input-element[_ngcontent-%COMP%]::placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-moz-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-webkit-input-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]:-ms-input-placeholder{color:rgba(97,97,97,.42)}.mat-accent[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%], .mat-form-field-invalid[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#f44336}.mat-form-field-type-mat-native-select.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{color:#616161}.mat-list-item-disabled[_ngcontent-%COMP%]{background-color:#eee}.mat-list-option[_ngcontent-%COMP%]:hover, .mat-list-option[_ngcontent-%COMP%]:focus, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.04)}.mat-list-single-selected-option[_ngcontent-%COMP%], .mat-list-single-selected-option[_ngcontent-%COMP%]:hover, .mat-list-single-selected-option[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.12)}.mat-menu-panel[_ngcontent-%COMP%]{background:#fff}.mat-menu-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-menu-item[_ngcontent-%COMP%]{background:transparent;color:#212121}.mat-menu-item[disabled][_ngcontent-%COMP%], .mat-menu-item[disabled][_ngcontent-%COMP%]::after{color:rgba(0,0,0,.38)}.mat-menu-item[_ngcontent-%COMP%]   .mat-icon-no-color[_ngcontent-%COMP%], .mat-menu-item-submenu-trigger[_ngcontent-%COMP%]::after{color:rgba(0,0,0,.54)}.mat-menu-item[_ngcontent-%COMP%]:hover:not([disabled]), .mat-menu-item.cdk-program-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item.cdk-keyboard-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item-highlighted[_ngcontent-%COMP%]:not([disabled]){background:rgba(0,0,0,.04)}.mat-paginator[_ngcontent-%COMP%]{background:#fff}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{color:#616161}.mat-paginator-decrement[_ngcontent-%COMP%], .mat-paginator-increment[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54);border-right:2px solid rgba(0,0,0,.54)}.mat-paginator-first[_ngcontent-%COMP%], .mat-paginator-last[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54)}.mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-decrement[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-increment[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-first[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-last[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f57c00}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#ff9800}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f44336}.mat-progress-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f57c00}.mat-progress-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#ff9800}.mat-progress-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f44336}.mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#616161}.mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f57c00}.mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#ff9800}.mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#ff9800}.mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f44336}.mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f44336}.mat-radio-button.mat-radio-disabled.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-label-content[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-radio-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-select-value[_ngcontent-%COMP%]{color:#212121}.mat-select-placeholder[_ngcontent-%COMP%]{color:rgba(97,97,97,.42)}.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%]{color:#757575}.mat-select-arrow[_ngcontent-%COMP%]{color:#616161}.mat-select-panel[_ngcontent-%COMP%]{background:#fff}.mat-select-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-select-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple){background:rgba(0,0,0,.12)}.mat-form-field.mat-focused.mat-primary[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-invalid[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#757575}.mat-drawer-container[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-drawer[_ngcontent-%COMP%]{background-color:#fff;color:#212121}.mat-drawer.mat-drawer-push[_ngcontent-%COMP%]{background-color:#fff}.mat-drawer[_ngcontent-%COMP%]:not(.mat-drawer-side){box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-drawer-side[_ngcontent-%COMP%]{border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-backdrop.mat-drawer-shown[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.6)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.54)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.54)}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.54)}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle[_ngcontent-%COMP%]:not(.mat-checked)   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-slide-toggle-thumb[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12);background-color:#fafafa}.mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-primary[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f57c00}.mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-accent[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#ff9800}.mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-warn[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f44336}.mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.2)}.mat-slider[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%], .cdk-focused[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-disabled[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]   .mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing)   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26);background-color:transparent}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26)}.mat-slider-has-ticks[_ngcontent-%COMP%]   .mat-slider-wrapper[_ngcontent-%COMP%]::after{border-color:rgba(0,0,0,.7)}.mat-slider-horizontal[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to right, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent);background-image:-moz-repeating-linear-gradient(0.0001deg, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-slider-vertical[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to bottom, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-step-header.cdk-keyboard-focused[_ngcontent-%COMP%], .mat-step-header.cdk-program-focused[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]:hover{background-color:rgba(0,0,0,.04)}@media(hover: none){.mat-step-header[_ngcontent-%COMP%]:hover{background:none}}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-optional[_ngcontent-%COMP%]{color:#616161}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon[_ngcontent-%COMP%]{background-color:#616161;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-selected[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-done[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-edit[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-error[_ngcontent-%COMP%]{background-color:transparent;color:#f44336}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-active[_ngcontent-%COMP%]{color:#212121}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-error[_ngcontent-%COMP%]{color:#f44336}.mat-stepper-horizontal[_ngcontent-%COMP%], .mat-stepper-vertical[_ngcontent-%COMP%]{background-color:#fff}.mat-stepper-vertical-line[_ngcontent-%COMP%]::before{border-left-color:rgba(0,0,0,.12)}.mat-horizontal-stepper-header[_ngcontent-%COMP%]::before, .mat-horizontal-stepper-header[_ngcontent-%COMP%]::after, .mat-stepper-horizontal-line[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-sort-header-arrow[_ngcontent-%COMP%]{color:#616161}.mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-header[_ngcontent-%COMP%]{border-bottom:1px solid rgba(0,0,0,.12)}.mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%]{border-top:1px solid rgba(0,0,0,.12);border-bottom:none}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{color:#212121}.mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:#757575}.mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#212121}.mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#757575}.mat-tab-group[class*=mat-background-][_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar[class*=mat-background-][_ngcontent-%COMP%]{border-bottom:none;border-top:none}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-toolbar[_ngcontent-%COMP%]{background:#f5f5f5;color:#212121}.mat-toolbar.mat-primary[_ngcontent-%COMP%]{background:#f57c00;color:#fff}.mat-toolbar.mat-accent[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-toolbar.mat-warn[_ngcontent-%COMP%]{background:#f44336;color:#fff}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:currentColor}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:inherit}.mat-toolbar[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:currentColor}.mat-tooltip[_ngcontent-%COMP%]{background:rgba(97,97,97,.9)}.mat-tree[_ngcontent-%COMP%]{background:#fff}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{color:#212121}.mat-snack-bar-container[_ngcontent-%COMP%]{color:rgba(255,255,255,.7);background:#323232;box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-simple-snackbar-action[_ngcontent-%COMP%]{color:#ff9800}[_nghost-%COMP%]{display:flex;flex-direction:column;height:100%}.toolbar[_ngcontent-%COMP%]{flex:none;display:flex;align-items:center;justify-content:space-between;height:48px;padding:0 16px;background-color:#fff;border-bottom:1px solid #ebebeb}.toolbar[_ngcontent-%COMP%]   metrics-tag-filter[_ngcontent-%COMP%]{flex:1 1 100px}.toolbar[_ngcontent-%COMP%]   .right-items[_ngcontent-%COMP%]{border-left:1px solid #ebebeb;margin-left:16px;padding-left:16px}.split-content[_ngcontent-%COMP%]{display:flex;overflow-y:auto;flex:1}.main[_ngcontent-%COMP%], .sidebar[_ngcontent-%COMP%]{overflow-x:hidden;overflow-y:auto;will-change:transform;contain:strict}.main[_ngcontent-%COMP%]{background-color:#f5f6f7;flex:1 1;display:flex;flex-direction:column}.main[_ngcontent-%COMP%]   metrics-filtered-view[_ngcontent-%COMP%], .main[_ngcontent-%COMP%]   metrics-pinned-view[_ngcontent-%COMP%]{border-bottom:1px solid #ebebeb}.sidebar[_ngcontent-%COMP%]{flex:0 0 250px;background-color:#fff;border-left:1px solid #ebebeb}.sidebar[_ngcontent-%COMP%]   .header[_ngcontent-%COMP%]{display:flex;align-items:center;justify-content:space-between;height:42px;padding:0 16px;border-bottom:1px solid #ebebeb}.sidebar[_ngcontent-%COMP%]   .header[_ngcontent-%COMP%]   .title[_ngcontent-%COMP%]{font-size:18px;font-weight:400;line-height:normal;margin:0}.settings-button[_ngcontent-%COMP%]{color:#616161;display:inline-flex}.settings-button.checked[_ngcontent-%COMP%]{background-color:#e0e0e0;border-color:#e0e0e0}.settings-button[_ngcontent-%COMP%]     .mat-button-wrapper{display:inline-flex;align-items:center}.settings-button[_ngcontent-%COMP%]   mat-icon[_ngcontent-%COMP%]{margin-right:4px}"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/views/main_view/main_view_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class MainViewContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.isSidepaneOpen = true;
            this.showFilteredView$ = this.store.select(getMetricsTagFilter).pipe(map((/**
             * @param {?} filter
             * @return {?}
             */
            (filter) => {
                return filter.length > 0;
            })));
        }
        /**
         * @return {?}
         */
        onSettingsButtonClicked() {
            this.isSidepaneOpen = !this.isSidepaneOpen;
        }
        /**
         * @return {?}
         */
        onCloseSidepaneButtonClicked() {
            this.isSidepaneOpen = false;
        }
    }
    MainViewContainer.decorators = [
        { type: Component, args: [{
                    selector: 'metrics-main-view',
                    template: `
    <metrics-main-view-component
      [showFilteredView]="showFilteredView$ | async"
      [isSidepaneOpen]="isSidepaneOpen"
      (onSettingsButtonClicked)="onSettingsButtonClicked()"
      (onCloseSidepaneButtonClicked)="onCloseSidepaneButtonClicked()"
    ></metrics-main-view-component>
  `,
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    MainViewContainer.ctorParameters = () => [
        { type: Store }
    ];
    /** @nocollapse */ MainViewContainer.ɵfac = function MainViewContainer_Factory(t) { return new (t || MainViewContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ MainViewContainer.ɵcmp = ɵɵdefineComponent({ type: MainViewContainer, selectors: [["metrics-main-view"]], decls: 2, vars: 4, consts: [[3, "showFilteredView", "isSidepaneOpen", "onSettingsButtonClicked", "onCloseSidepaneButtonClicked"]], template: function MainViewContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "metrics-main-view-component", 0);
            ɵɵlistener("onSettingsButtonClicked", function MainViewContainer_Template_metrics_main_view_component_onSettingsButtonClicked_0_listener() { return ctx.onSettingsButtonClicked(); })("onCloseSidepaneButtonClicked", function MainViewContainer_Template_metrics_main_view_component_onCloseSidepaneButtonClicked_0_listener() { return ctx.onCloseSidepaneButtonClicked(); });
            ɵɵpipe(1, "async");
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵproperty("showFilteredView", ɵɵpipeBind1(1, 2, ctx.showFilteredView$))("isSidepaneOpen", ctx.isSidepaneOpen);
        } }, directives: [MainViewComponent], pipes: [AsyncPipe], encapsulation: 2, changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/views/metrics_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class MetricsDashboardContainer {
    }
    MetricsDashboardContainer.decorators = [
        { type: Component, args: [{
                    selector: 'metrics-dashboard',
                    template: `
    <nav>
      <runs-selector></runs-selector>
    </nav>
    <metrics-main-view></metrics-main-view>
  `,
                    styleUrls: ['metrics_container.css'],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */ MetricsDashboardContainer.ɵfac = function MetricsDashboardContainer_Factory(t) { return new (t || MetricsDashboardContainer)(); };
    /** @nocollapse */ MetricsDashboardContainer.ɵcmp = ɵɵdefineComponent({ type: MetricsDashboardContainer, selectors: [["metrics-dashboard"]], decls: 3, vars: 0, template: function MetricsDashboardContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "nav");
            ɵɵelement(1, "runs-selector");
            ɵɵelementEnd();
            ɵɵelement(2, "metrics-main-view");
        } }, directives: [RunsSelectorContainer, MainViewContainer], styles: [".mat-badge-content[_ngcontent-%COMP%]{font-weight:600;font-size:12px;font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:9px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:24px}.mat-h1[_ngcontent-%COMP%], .mat-headline[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%]{font:400 24px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h2[_ngcontent-%COMP%], .mat-title[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h3[_ngcontent-%COMP%], .mat-subheading-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%]{font:400 16px/28px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h4[_ngcontent-%COMP%], .mat-subheading-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%]{font:400 15px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h5[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%]{font:400 calc(14px * 0.83)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-h6[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:400 calc(14px * 0.67)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-body-strong[_ngcontent-%COMP%], .mat-body-2[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   p[_ngcontent-%COMP%]{margin:0 0 12px}.mat-small[_ngcontent-%COMP%], .mat-caption[_ngcontent-%COMP%]{font:400 12px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-display-4[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-4[_ngcontent-%COMP%]{font:300 112px/112px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.05em;margin:0 0 56px}.mat-display-3[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-3[_ngcontent-%COMP%]{font:400 56px/56px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.02em;margin:0 0 64px}.mat-display-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-2[_ngcontent-%COMP%]{font:400 45px/48px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.005em;margin:0 0 64px}.mat-display-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-1[_ngcontent-%COMP%]{font:400 34px/40px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 64px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%], .mat-flat-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-button-toggle[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card-title[_ngcontent-%COMP%]{font-size:24px;font-weight:500}.mat-card-header[_ngcontent-%COMP%]   .mat-card-title[_ngcontent-%COMP%]{font-size:20px}.mat-card-subtitle[_ngcontent-%COMP%], .mat-card-content[_ngcontent-%COMP%]{font-size:14px}.mat-checkbox[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-checkbox-layout[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{line-height:24px}.mat-chip[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-chip[_ngcontent-%COMP%]   .mat-chip-trailing-icon.mat-icon[_ngcontent-%COMP%], .mat-chip[_ngcontent-%COMP%]   .mat-chip-remove.mat-icon[_ngcontent-%COMP%]{font-size:18px}.mat-table[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-header-cell[_ngcontent-%COMP%]{font-size:12px;font-weight:500}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{font-size:14px}.mat-calendar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-calendar-body[_ngcontent-%COMP%]{font-size:13px}.mat-calendar-body-label[_ngcontent-%COMP%], .mat-calendar-period-button[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-calendar-table-header[_ngcontent-%COMP%]   th[_ngcontent-%COMP%]{font-size:11px;font-weight:400}.mat-dialog-title[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-expansion-panel-header[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:15px;font-weight:400}.mat-expansion-panel-content[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field[_ngcontent-%COMP%]{font-size:inherit;font-weight:400;line-height:1.125;font-family:Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.34375em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{font-size:150%;line-height:1.125}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]{height:1.5em;width:1.5em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{height:1.125em;line-height:1.125}.mat-form-field-infix[_ngcontent-%COMP%]{padding:.5em 0;border-top:.84375em solid transparent}.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34375em) scale(0.75);width:133.3333333333%}.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34374em) scale(0.75);width:133.3333433333%}.mat-form-field-label-wrapper[_ngcontent-%COMP%]{top:-0.84375em;padding-top:.84375em}.mat-form-field-label[_ngcontent-%COMP%]{top:1.34375em}.mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.34375em}.mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{font-size:75%;margin-top:.6666666667em;top:calc(100% - 1.7916666667em)}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.4375em 0}.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.001px);-ms-transform:translateY(-1.28125em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00101px);-ms-transform:translateY(-1.28124em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00102px);-ms-transform:translateY(-1.28123em) scale(0.75);width:133.3333533333%}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.28125em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{margin-top:.5416666667em;top:calc(100% - 1.6666666667em)}@media print{.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28122em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28121em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.2812em) scale(0.75)}}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.25em 0 .75em 0}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.09375em;margin-top:-0.5em}.mat-form-field-appearance-fill.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59374em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:1em 0 1em 0}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.84375em;margin-top:-0.25em}.mat-form-field-appearance-outline.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59374em) scale(0.75);width:133.3333433333%}.mat-grid-tile-header[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]{font-size:14px}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2), .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}input.mat-input-element[_ngcontent-%COMP%]{margin-top:-0.0625em}.mat-menu-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:400}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px}.mat-radio-button[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select-trigger[_ngcontent-%COMP%]{height:1.125em}.mat-slide-toggle-content[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-slider-thumb-label-text[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-stepper-vertical[_ngcontent-%COMP%], .mat-stepper-horizontal[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-step-label[_ngcontent-%COMP%]{font-size:14px;font-weight:400}.mat-step-sub-label-error[_ngcontent-%COMP%]{font-weight:normal}.mat-step-label-error[_ngcontent-%COMP%]{font-size:14px}.mat-step-label-selected[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-tab-group[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-toolbar[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0}.mat-tooltip[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:10px;padding-top:6px;padding-bottom:6px}.mat-tooltip-handset[_ngcontent-%COMP%]{font-size:14px;padding-top:8px;padding-bottom:8px}.mat-list-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:16px}.mat-optgroup-label[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-simple-snackbar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px}.mat-simple-snackbar-action[_ngcontent-%COMP%]{line-height:1;font-family:inherit;font-size:inherit;font-weight:500}.mat-tree[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{font-weight:400;font-size:14px}.mat-ripple[_ngcontent-%COMP%]{overflow:hidden;position:relative}.mat-ripple[_ngcontent-%COMP%]:not(:empty){transform:translateZ(0)}.mat-ripple.mat-ripple-unbounded[_ngcontent-%COMP%]{overflow:visible}.mat-ripple-element[_ngcontent-%COMP%]{position:absolute;border-radius:50%;pointer-events:none;transition:opacity,transform 0ms cubic-bezier(0, 0, 0.2, 1);transform:scale(0)}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{display:none}.cdk-visually-hidden[_ngcontent-%COMP%]{border:0;clip:rect(0 0 0 0);height:1px;margin:-1px;overflow:hidden;padding:0;position:absolute;width:1px;outline:0;-webkit-appearance:none;-moz-appearance:none}.cdk-overlay-container[_ngcontent-%COMP%], .cdk-global-overlay-wrapper[_ngcontent-%COMP%]{pointer-events:none;top:0;left:0;height:100%;width:100%}.cdk-overlay-container[_ngcontent-%COMP%]{position:fixed;z-index:1000}.cdk-overlay-container[_ngcontent-%COMP%]:empty{display:none}.cdk-global-overlay-wrapper[_ngcontent-%COMP%]{display:flex;position:absolute;z-index:1000}.cdk-overlay-pane[_ngcontent-%COMP%]{position:absolute;pointer-events:auto;box-sizing:border-box;z-index:1000;display:flex;max-width:100%;max-height:100%}.cdk-overlay-backdrop[_ngcontent-%COMP%]{position:absolute;top:0;bottom:0;left:0;right:0;z-index:1000;pointer-events:auto;-webkit-tap-highlight-color:transparent;transition:opacity 400ms cubic-bezier(0.25, 0.8, 0.25, 1);opacity:0}.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:1}@media screen and (-ms-high-contrast: active){.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:.6}}.cdk-overlay-dark-backdrop[_ngcontent-%COMP%]{background:rgba(0,0,0,.32)}.cdk-overlay-transparent-backdrop[_ngcontent-%COMP%], .cdk-overlay-transparent-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:0}.cdk-overlay-connected-position-bounding-box[_ngcontent-%COMP%]{position:absolute;z-index:1000;display:flex;flex-direction:column;min-width:1px;min-height:1px}.cdk-global-scrollblock[_ngcontent-%COMP%]{position:fixed;width:100%;overflow-y:scroll}@keyframes cdk-text-field-autofill-start{}@keyframes cdk-text-field-autofill-end{}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:-webkit-autofill{animation:cdk-text-field-autofill-start 0s 1ms}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:not(:-webkit-autofill){animation:cdk-text-field-autofill-end 0s 1ms}textarea.cdk-textarea-autosize[_ngcontent-%COMP%]{resize:none}textarea.cdk-textarea-autosize-measuring[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:auto !important;overflow:hidden !important}textarea.cdk-textarea-autosize-measuring-firefox[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:0 !important}.mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.1)}.mat-option[_ngcontent-%COMP%]{color:#212121}.mat-option[_ngcontent-%COMP%]:hover:not(.mat-option-disabled), .mat-option[_ngcontent-%COMP%]:focus:not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple):not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-active[_ngcontent-%COMP%]{background:rgba(0,0,0,.04);color:#212121}.mat-option.mat-option-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-primary[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f57c00}.mat-accent[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f44336}.mat-optgroup-label[_ngcontent-%COMP%]{color:#616161}.mat-optgroup-disabled[_ngcontent-%COMP%]   .mat-optgroup-label[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-pseudo-checkbox[_ngcontent-%COMP%]{color:#616161}.mat-pseudo-checkbox[_ngcontent-%COMP%]::after{color:#fafafa}.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{color:#b0b0b0}.mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f57c00}.mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f44336}.mat-pseudo-checkbox-checked.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{background:#b0b0b0}.mat-elevation-z0[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z1[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z2[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z3[_ngcontent-%COMP%]{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z4[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z5[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 5px 8px 0px rgba(0, 0, 0, 0.14),0px 1px 14px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z6[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z7[_ngcontent-%COMP%]{box-shadow:0px 4px 5px -2px rgba(0, 0, 0, 0.2),0px 7px 10px 1px rgba(0, 0, 0, 0.14),0px 2px 16px 1px rgba(0, 0, 0, 0.12)}.mat-elevation-z8[_ngcontent-%COMP%]{box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z9[_ngcontent-%COMP%]{box-shadow:0px 5px 6px -3px rgba(0, 0, 0, 0.2),0px 9px 12px 1px rgba(0, 0, 0, 0.14),0px 3px 16px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z10[_ngcontent-%COMP%]{box-shadow:0px 6px 6px -3px rgba(0, 0, 0, 0.2),0px 10px 14px 1px rgba(0, 0, 0, 0.14),0px 4px 18px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z11[_ngcontent-%COMP%]{box-shadow:0px 6px 7px -4px rgba(0, 0, 0, 0.2),0px 11px 15px 1px rgba(0, 0, 0, 0.14),0px 4px 20px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z12[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z13[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 13px 19px 2px rgba(0, 0, 0, 0.14),0px 5px 24px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z14[_ngcontent-%COMP%]{box-shadow:0px 7px 9px -4px rgba(0, 0, 0, 0.2),0px 14px 21px 2px rgba(0, 0, 0, 0.14),0px 5px 26px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z15[_ngcontent-%COMP%]{box-shadow:0px 8px 9px -5px rgba(0, 0, 0, 0.2),0px 15px 22px 2px rgba(0, 0, 0, 0.14),0px 6px 28px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z16[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z17[_ngcontent-%COMP%]{box-shadow:0px 8px 11px -5px rgba(0, 0, 0, 0.2),0px 17px 26px 2px rgba(0, 0, 0, 0.14),0px 6px 32px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z18[_ngcontent-%COMP%]{box-shadow:0px 9px 11px -5px rgba(0, 0, 0, 0.2),0px 18px 28px 2px rgba(0, 0, 0, 0.14),0px 7px 34px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z19[_ngcontent-%COMP%]{box-shadow:0px 9px 12px -6px rgba(0, 0, 0, 0.2),0px 19px 29px 2px rgba(0, 0, 0, 0.14),0px 7px 36px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z20[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 20px 31px 3px rgba(0, 0, 0, 0.14),0px 8px 38px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z21[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 21px 33px 3px rgba(0, 0, 0, 0.14),0px 8px 40px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z22[_ngcontent-%COMP%]{box-shadow:0px 10px 14px -6px rgba(0, 0, 0, 0.2),0px 22px 35px 3px rgba(0, 0, 0, 0.14),0px 8px 42px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z23[_ngcontent-%COMP%]{box-shadow:0px 11px 14px -7px rgba(0, 0, 0, 0.2),0px 23px 36px 3px rgba(0, 0, 0, 0.14),0px 9px 44px 8px rgba(0, 0, 0, 0.12)}.mat-elevation-z24[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12)}.mat-app-background[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-theme-loaded-marker[_ngcontent-%COMP%]{display:none}.mat-autocomplete-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-autocomplete-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover){background:#fff}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover):not(.mat-option-disabled){color:#212121}.mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f57c00}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{outline:solid 1px;border-radius:0}.mat-badge-accent[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-badge-warn[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f44336}.mat-badge[_ngcontent-%COMP%]{position:relative}.mat-badge-hidden[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{display:none}.mat-badge-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#b9b9b9;color:#757575}.mat-badge-content[_ngcontent-%COMP%]{position:absolute;text-align:center;display:inline-block;border-radius:50%;transition:transform 200ms ease-in-out;transform:scale(0.6);overflow:hidden;white-space:nowrap;text-overflow:ellipsis;pointer-events:none}.ng-animate-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%], .mat-badge-content._mat-animation-noopable[_ngcontent-%COMP%]{transition:none}.mat-badge-content.mat-badge-active[_ngcontent-%COMP%]{transform:none}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:16px;height:16px;line-height:16px}.mat-badge-small.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-8px}.mat-badge-small.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-8px}.mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-16px}.mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-16px}.mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-8px}.mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-8px}.mat-badge-medium[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:22px;height:22px;line-height:22px}.mat-badge-medium.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-11px}.mat-badge-medium.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-11px}.mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-22px}.mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-22px}.mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-11px}.mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-11px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:28px;height:28px;line-height:28px}.mat-badge-large.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-14px}.mat-badge-large.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-14px}.mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-28px}.mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-28px}.mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-14px}.mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-14px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]{color:inherit;background:transparent}.mat-button.mat-primary[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-button[disabled][disabled][_ngcontent-%COMP%], .mat-icon-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-icon-button[disabled][disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-stroked-button[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f44336}.mat-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:transparent}.mat-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{opacity:.1;background-color:currentColor}.mat-button-focus-overlay[_ngcontent-%COMP%]{background:#000}.mat-stroked-button[_ngcontent-%COMP%]:not([disabled]){border-color:rgba(0,0,0,.12)}.mat-flat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{color:#212121;background-color:#fff}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-flat-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-stroked-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-flat-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-raised-button[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone[_ngcontent-%COMP%], .mat-button-toggle-group[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{box-shadow:none}.mat-button-toggle[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-button-toggle[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121;background:#fff}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:#000}.mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12)}[dir=rtl][_ngcontent-%COMP%]   .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-group-appearance-standard.mat-button-toggle-vertical[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:none;border-top:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#616161}.mat-button-toggle-checked.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121}.mat-button-toggle-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.26);background-color:#eee}.mat-button-toggle-disabled.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{background:#fff}.mat-button-toggle-disabled.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#bdbdbd}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{border:solid 1px rgba(0,0,0,.12)}.mat-card[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-card[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-card.mat-card-flat[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-card-subtitle[_ngcontent-%COMP%]{color:#616161}.mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#616161}.mat-checkbox-checkmark[_ngcontent-%COMP%]{fill:#fafafa}.mat-checkbox-checkmark-path[_ngcontent-%COMP%]{stroke:#fafafa !important}.mat-checkbox-mixedmark[_ngcontent-%COMP%]{background-color:#fafafa}.mat-checkbox-indeterminate.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f57c00}.mat-checkbox-indeterminate.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#ff9800}.mat-checkbox-indeterminate.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f44336}.mat-checkbox-disabled.mat-checkbox-checked[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-disabled.mat-checkbox-indeterminate[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]:not(.mat-checkbox-checked)   .mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{color:#616161}.mat-checkbox[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%]{background:#f57c00}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%]{background:#ff9800}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%]{background:#f44336}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#212121}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#212121;opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled):active{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled)   .mat-chip-remove[_ngcontent-%COMP%]:hover{opacity:.54}.mat-chip.mat-standard-chip.mat-chip-disabled[_ngcontent-%COMP%]{opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]::after{background:#000}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-table[_ngcontent-%COMP%]{background:#fff}.mat-table[_ngcontent-%COMP%]   thead[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tbody[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tfoot[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], [mat-header-row][_ngcontent-%COMP%], [mat-row][_ngcontent-%COMP%], [mat-footer-row][_ngcontent-%COMP%], .mat-table-sticky[_ngcontent-%COMP%]{background:inherit}mat-row[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], th.mat-header-cell[_ngcontent-%COMP%], td.mat-cell[_ngcontent-%COMP%], td.mat-footer-cell[_ngcontent-%COMP%]{border-bottom-color:rgba(0,0,0,.12)}.mat-header-cell[_ngcontent-%COMP%]{color:#616161}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{color:#212121}.mat-calendar-arrow[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.54)}.mat-datepicker-toggle[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-next-button[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-previous-button[_ngcontent-%COMP%]{color:rgba(0,0,0,.54)}.mat-calendar-table-header[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-calendar-table-header-divider[_ngcontent-%COMP%]::after{background:rgba(0,0,0,.12)}.mat-calendar-body-label[_ngcontent-%COMP%]{color:#616161}.mat-calendar-body-cell-content[_ngcontent-%COMP%]{color:#212121;border-color:transparent}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){color:#757575}.mat-calendar-body-cell[_ngcontent-%COMP%]:not(.mat-calendar-body-disabled):hover > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-keyboard-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-program-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){background-color:rgba(0,0,0,.04)}.mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.38)}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.18)}.mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.4)}.mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12);background-color:#fff;color:#212121}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.4)}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.4)}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content-touch[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-datepicker-toggle-active[_ngcontent-%COMP%]{color:#f57c00}.mat-datepicker-toggle-active.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-datepicker-toggle-active.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-dialog-container[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-divider[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-divider-vertical[_ngcontent-%COMP%]{border-right-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-expansion-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-action-row[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-keyboard-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-program-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded)   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover:not([aria-disabled=true]){background:rgba(0,0,0,.04)}@media(hover: none){.mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded):not([aria-disabled=true])   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover{background:#fff}}.mat-expansion-panel-header-title[_ngcontent-%COMP%]{color:#212121}.mat-expansion-panel-header-description[_ngcontent-%COMP%], .mat-expansion-indicator[_ngcontent-%COMP%]::after{color:#616161}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-title[_ngcontent-%COMP%], .mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-description[_ngcontent-%COMP%]{color:inherit}.mat-form-field-label[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-hint[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-focused[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-ripple[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid)   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f57c00}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-accent   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#ff9800}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-warn   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#f44336}.mat-error[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-hint[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-legacy.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-standard[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-standard.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.04)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.02)}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:transparent}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.12)}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:rgba(0,0,0,.87)}.mat-form-field-appearance-outline.mat-focused[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field-appearance-outline.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-appearance-outline.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-invalid.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.06)}.mat-icon.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-icon.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-icon.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-type-mat-native-select[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#616161}.mat-input-element[_ngcontent-%COMP%]:disabled, .mat-form-field-type-mat-native-select.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#757575}.mat-input-element[_ngcontent-%COMP%]{caret-color:#f57c00}.mat-input-element[_ngcontent-%COMP%]::placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-moz-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-webkit-input-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]:-ms-input-placeholder{color:rgba(97,97,97,.42)}.mat-accent[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%], .mat-form-field-invalid[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#f44336}.mat-form-field-type-mat-native-select.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{color:#616161}.mat-list-item-disabled[_ngcontent-%COMP%]{background-color:#eee}.mat-list-option[_ngcontent-%COMP%]:hover, .mat-list-option[_ngcontent-%COMP%]:focus, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.04)}.mat-list-single-selected-option[_ngcontent-%COMP%], .mat-list-single-selected-option[_ngcontent-%COMP%]:hover, .mat-list-single-selected-option[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.12)}.mat-menu-panel[_ngcontent-%COMP%]{background:#fff}.mat-menu-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-menu-item[_ngcontent-%COMP%]{background:transparent;color:#212121}.mat-menu-item[disabled][_ngcontent-%COMP%], .mat-menu-item[disabled][_ngcontent-%COMP%]::after{color:rgba(0,0,0,.38)}.mat-menu-item[_ngcontent-%COMP%]   .mat-icon-no-color[_ngcontent-%COMP%], .mat-menu-item-submenu-trigger[_ngcontent-%COMP%]::after{color:rgba(0,0,0,.54)}.mat-menu-item[_ngcontent-%COMP%]:hover:not([disabled]), .mat-menu-item.cdk-program-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item.cdk-keyboard-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item-highlighted[_ngcontent-%COMP%]:not([disabled]){background:rgba(0,0,0,.04)}.mat-paginator[_ngcontent-%COMP%]{background:#fff}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{color:#616161}.mat-paginator-decrement[_ngcontent-%COMP%], .mat-paginator-increment[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54);border-right:2px solid rgba(0,0,0,.54)}.mat-paginator-first[_ngcontent-%COMP%], .mat-paginator-last[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54)}.mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-decrement[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-increment[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-first[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-last[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f57c00}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#ff9800}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f44336}.mat-progress-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f57c00}.mat-progress-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#ff9800}.mat-progress-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f44336}.mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#616161}.mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f57c00}.mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#ff9800}.mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#ff9800}.mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f44336}.mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f44336}.mat-radio-button.mat-radio-disabled.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-label-content[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-radio-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-select-value[_ngcontent-%COMP%]{color:#212121}.mat-select-placeholder[_ngcontent-%COMP%]{color:rgba(97,97,97,.42)}.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%]{color:#757575}.mat-select-arrow[_ngcontent-%COMP%]{color:#616161}.mat-select-panel[_ngcontent-%COMP%]{background:#fff}.mat-select-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-select-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple){background:rgba(0,0,0,.12)}.mat-form-field.mat-focused.mat-primary[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-invalid[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#757575}.mat-drawer-container[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-drawer[_ngcontent-%COMP%]{background-color:#fff;color:#212121}.mat-drawer.mat-drawer-push[_ngcontent-%COMP%]{background-color:#fff}.mat-drawer[_ngcontent-%COMP%]:not(.mat-drawer-side){box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-drawer-side[_ngcontent-%COMP%]{border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-backdrop.mat-drawer-shown[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.6)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.54)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.54)}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.54)}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle[_ngcontent-%COMP%]:not(.mat-checked)   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-slide-toggle-thumb[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12);background-color:#fafafa}.mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-primary[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f57c00}.mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-accent[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#ff9800}.mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-warn[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f44336}.mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.2)}.mat-slider[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%], .cdk-focused[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-disabled[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]   .mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing)   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26);background-color:transparent}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26)}.mat-slider-has-ticks[_ngcontent-%COMP%]   .mat-slider-wrapper[_ngcontent-%COMP%]::after{border-color:rgba(0,0,0,.7)}.mat-slider-horizontal[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to right, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent);background-image:-moz-repeating-linear-gradient(0.0001deg, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-slider-vertical[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to bottom, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-step-header.cdk-keyboard-focused[_ngcontent-%COMP%], .mat-step-header.cdk-program-focused[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]:hover{background-color:rgba(0,0,0,.04)}@media(hover: none){.mat-step-header[_ngcontent-%COMP%]:hover{background:none}}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-optional[_ngcontent-%COMP%]{color:#616161}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon[_ngcontent-%COMP%]{background-color:#616161;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-selected[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-done[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-edit[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-error[_ngcontent-%COMP%]{background-color:transparent;color:#f44336}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-active[_ngcontent-%COMP%]{color:#212121}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-error[_ngcontent-%COMP%]{color:#f44336}.mat-stepper-horizontal[_ngcontent-%COMP%], .mat-stepper-vertical[_ngcontent-%COMP%]{background-color:#fff}.mat-stepper-vertical-line[_ngcontent-%COMP%]::before{border-left-color:rgba(0,0,0,.12)}.mat-horizontal-stepper-header[_ngcontent-%COMP%]::before, .mat-horizontal-stepper-header[_ngcontent-%COMP%]::after, .mat-stepper-horizontal-line[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-sort-header-arrow[_ngcontent-%COMP%]{color:#616161}.mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-header[_ngcontent-%COMP%]{border-bottom:1px solid rgba(0,0,0,.12)}.mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%]{border-top:1px solid rgba(0,0,0,.12);border-bottom:none}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{color:#212121}.mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:#757575}.mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#212121}.mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#757575}.mat-tab-group[class*=mat-background-][_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar[class*=mat-background-][_ngcontent-%COMP%]{border-bottom:none;border-top:none}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-toolbar[_ngcontent-%COMP%]{background:#f5f5f5;color:#212121}.mat-toolbar.mat-primary[_ngcontent-%COMP%]{background:#f57c00;color:#fff}.mat-toolbar.mat-accent[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-toolbar.mat-warn[_ngcontent-%COMP%]{background:#f44336;color:#fff}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:currentColor}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:inherit}.mat-toolbar[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:currentColor}.mat-tooltip[_ngcontent-%COMP%]{background:rgba(97,97,97,.9)}.mat-tree[_ngcontent-%COMP%]{background:#fff}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{color:#212121}.mat-snack-bar-container[_ngcontent-%COMP%]{color:rgba(255,255,255,.7);background:#323232;box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-simple-snackbar-action[_ngcontent-%COMP%]{color:#ff9800}[_nghost-%COMP%]{display:flex;height:100%}nav[_ngcontent-%COMP%]{border-right:1px solid #ebebeb;flex:none;width:340px}metrics-main-view[_ngcontent-%COMP%]{flex:1 1}"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/widgets/histogram/histogram_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class HistogramModule {
    }
    HistogramModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [HistogramComponent],
                    exports: [HistogramComponent],
                },] },
    ];
    /** @nocollapse */ HistogramModule.ɵmod = ɵɵdefineNgModule({ type: HistogramModule });
    /** @nocollapse */ HistogramModule.ɵinj = ɵɵdefineInjector({ factory: function HistogramModule_Factory(t) { return new (t || HistogramModule)(); } });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(HistogramModule, { declarations: [HistogramComponent], exports: [HistogramComponent] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/widgets/resize_detector_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class ResizeDetectorModule {
    }
    ResizeDetectorModule.decorators = [
        { type: NgModule, args: [{
                    exports: [ResizeDetectorDirective],
                    declarations: [ResizeDetectorDirective],
                },] },
    ];
    /** @nocollapse */ ResizeDetectorModule.ɵmod = ɵɵdefineNgModule({ type: ResizeDetectorModule });
    /** @nocollapse */ ResizeDetectorModule.ɵinj = ɵɵdefineInjector({ factory: function ResizeDetectorModule_Factory(t) { return new (t || ResizeDetectorModule)(); } });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(ResizeDetectorModule, { declarations: [ResizeDetectorDirective], exports: [ResizeDetectorDirective] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/widgets/text/truncated_path_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Provides a TruncatedPathComponent for rendering a '/' delimited path.
     */
    class TruncatedPathModule {
    }
    TruncatedPathModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [TruncatedPathComponent],
                    exports: [TruncatedPathComponent],
                    imports: [CommonModule],
                },] },
    ];
    /** @nocollapse */ TruncatedPathModule.ɵmod = ɵɵdefineNgModule({ type: TruncatedPathModule });
    /** @nocollapse */ TruncatedPathModule.ɵinj = ɵɵdefineInjector({ factory: function TruncatedPathModule_Factory(t) { return new (t || TruncatedPathModule)(); }, imports: [[CommonModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(TruncatedPathModule, { declarations: [TruncatedPathComponent], imports: [CommonModule], exports: [TruncatedPathComponent] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/views/card_renderer/run_name_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class RunNameModule {
    }
    RunNameModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [RunNameContainer, RunNameComponent],
                    exports: [RunNameContainer],
                    imports: [CommonModule],
                },] },
    ];
    /** @nocollapse */ RunNameModule.ɵmod = ɵɵdefineNgModule({ type: RunNameModule });
    /** @nocollapse */ RunNameModule.ɵinj = ɵɵdefineInjector({ factory: function RunNameModule_Factory(t) { return new (t || RunNameModule)(); }, imports: [[CommonModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(RunNameModule, { declarations: [RunNameContainer, RunNameComponent], imports: [CommonModule], exports: [RunNameContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/views/card_renderer/histogram_card_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class HistogramCardModule {
    }
    HistogramCardModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [HistogramCardContainer, HistogramCardComponent],
                    exports: [HistogramCardContainer],
                    imports: [
                        CommonModule,
                        HistogramModule,
                        MatButtonModule,
                        MatIconModule,
                        MatProgressSpinnerModule,
                        ResizeDetectorModule,
                        RunNameModule,
                        TruncatedPathModule,
                    ],
                },] },
    ];
    /** @nocollapse */ HistogramCardModule.ɵmod = ɵɵdefineNgModule({ type: HistogramCardModule });
    /** @nocollapse */ HistogramCardModule.ɵinj = ɵɵdefineInjector({ factory: function HistogramCardModule_Factory(t) { return new (t || HistogramCardModule)(); }, imports: [[
                CommonModule,
                HistogramModule,
                MatButtonModule,
                MatIconModule,
                MatProgressSpinnerModule,
                ResizeDetectorModule,
                RunNameModule,
                TruncatedPathModule,
            ]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(HistogramCardModule, { declarations: [HistogramCardContainer, HistogramCardComponent], imports: [CommonModule,
            HistogramModule,
            MatButtonModule,
            MatIconModule,
            MatProgressSpinnerModule,
            ResizeDetectorModule,
            RunNameModule,
            TruncatedPathModule], exports: [HistogramCardContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/views/card_renderer/image_card_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class ImageCardModule {
    }
    ImageCardModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [ImageCardContainer, ImageCardComponent],
                    exports: [ImageCardContainer],
                    imports: [
                        CommonModule,
                        MatButtonModule,
                        MatIconModule,
                        MatProgressSpinnerModule,
                        MatSliderModule,
                        RunNameModule,
                        TruncatedPathModule,
                    ],
                },] },
    ];
    /** @nocollapse */ ImageCardModule.ɵmod = ɵɵdefineNgModule({ type: ImageCardModule });
    /** @nocollapse */ ImageCardModule.ɵinj = ɵɵdefineInjector({ factory: function ImageCardModule_Factory(t) { return new (t || ImageCardModule)(); }, imports: [[
                CommonModule,
                MatButtonModule,
                MatIconModule,
                MatProgressSpinnerModule,
                MatSliderModule,
                RunNameModule,
                TruncatedPathModule,
            ]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(ImageCardModule, { declarations: [ImageCardContainer, ImageCardComponent], imports: [CommonModule,
            MatButtonModule,
            MatIconModule,
            MatProgressSpinnerModule,
            MatSliderModule,
            RunNameModule,
            TruncatedPathModule], exports: [ImageCardContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/widgets/line_chart/line_chart_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class LineChartModule {
    }
    LineChartModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [LineChartComponent],
                    exports: [LineChartComponent],
                },] },
    ];
    /** @nocollapse */ LineChartModule.ɵmod = ɵɵdefineNgModule({ type: LineChartModule });
    /** @nocollapse */ LineChartModule.ɵinj = ɵɵdefineInjector({ factory: function LineChartModule_Factory(t) { return new (t || LineChartModule)(); } });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(LineChartModule, { declarations: [LineChartComponent], exports: [LineChartComponent] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/views/card_renderer/scalar_card_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class ScalarCardModule {
    }
    ScalarCardModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [ScalarCardContainer, ScalarCardComponent],
                    exports: [ScalarCardContainer],
                    imports: [
                        CommonModule,
                        LineChartModule,
                        MatButtonModule,
                        MatIconModule,
                        MatMenuModule,
                        MatProgressSpinnerModule,
                        ResizeDetectorModule,
                        TruncatedPathModule,
                    ],
                },] },
    ];
    /** @nocollapse */ ScalarCardModule.ɵmod = ɵɵdefineNgModule({ type: ScalarCardModule });
    /** @nocollapse */ ScalarCardModule.ɵinj = ɵɵdefineInjector({ factory: function ScalarCardModule_Factory(t) { return new (t || ScalarCardModule)(); }, imports: [[
                CommonModule,
                LineChartModule,
                MatButtonModule,
                MatIconModule,
                MatMenuModule,
                MatProgressSpinnerModule,
                ResizeDetectorModule,
                TruncatedPathModule,
            ]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(ScalarCardModule, { declarations: [ScalarCardContainer, ScalarCardComponent], imports: [CommonModule,
            LineChartModule,
            MatButtonModule,
            MatIconModule,
            MatMenuModule,
            MatProgressSpinnerModule,
            ResizeDetectorModule,
            TruncatedPathModule], exports: [ScalarCardContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/views/card_renderer/card_renderer_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class CardRendererModule {
    }
    CardRendererModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [CardLazyLoader, CardViewComponent, CardViewContainer],
                    exports: [CardLazyLoader, CardViewContainer],
                    imports: [
                        CommonModule,
                        ImageCardModule,
                        ScalarCardModule,
                        HistogramCardModule,
                    ],
                },] },
    ];
    /** @nocollapse */ CardRendererModule.ɵmod = ɵɵdefineNgModule({ type: CardRendererModule });
    /** @nocollapse */ CardRendererModule.ɵinj = ɵɵdefineInjector({ factory: function CardRendererModule_Factory(t) { return new (t || CardRendererModule)(); }, imports: [[
                CommonModule,
                ImageCardModule,
                ScalarCardModule,
                HistogramCardModule,
            ]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(CardRendererModule, { declarations: [CardLazyLoader, CardViewComponent, CardViewContainer], imports: [CommonModule,
            ImageCardModule,
            ScalarCardModule,
            HistogramCardModule], exports: [CardLazyLoader, CardViewContainer] }); })();

    const _c0$s = ["button"];
    const _c1$i = ["*"];
    /**
     * Injection token that can be used to configure the
     * default options for all button toggles within an app.
     * @type {?}
     */
    const MAT_BUTTON_TOGGLE_DEFAULT_OPTIONS = new InjectionToken('MAT_BUTTON_TOGGLE_DEFAULT_OPTIONS');
    /**
     * Provider Expression that allows mat-button-toggle-group to register as a ControlValueAccessor.
     * This allows it to support [(ngModel)].
     * \@docs-private
     * @type {?}
     */
    const MAT_BUTTON_TOGGLE_GROUP_VALUE_ACCESSOR = {
        provide: NG_VALUE_ACCESSOR,
        useExisting: forwardRef((/**
         * @return {?}
         */
        () => MatButtonToggleGroup)),
        multi: true
    };
    /**
     * @deprecated Use `MatButtonToggleGroup` instead.
     * \@breaking-change 8.0.0
     */
    class MatButtonToggleGroupMultiple {
    }
    /** @type {?} */
    let _uniqueIdCounter$1 = 0;
    /**
     * Change event object emitted by MatButtonToggle.
     */
    class MatButtonToggleChange {
        /**
         * @param {?} source
         * @param {?} value
         */
        constructor(source, value) {
            this.source = source;
            this.value = value;
        }
    }
    /**
     * Exclusive selection button toggle group that behaves like a radio-button group.
     */
    class MatButtonToggleGroup {
        /**
         * @param {?} _changeDetector
         * @param {?=} defaultOptions
         */
        constructor(_changeDetector, defaultOptions) {
            this._changeDetector = _changeDetector;
            this._vertical = false;
            this._multiple = false;
            this._disabled = false;
            /**
             * The method to be called in order to update ngModel.
             * Now `ngModel` binding is not supported in multiple selection mode.
             */
            this._controlValueAccessorChangeFn = (/**
             * @return {?}
             */
            () => { });
            /**
             * onTouch function registered via registerOnTouch (ControlValueAccessor).
             */
            this._onTouched = (/**
             * @return {?}
             */
            () => { });
            this._name = `mat-button-toggle-group-${_uniqueIdCounter$1++}`;
            /**
             * Event that emits whenever the value of the group changes.
             * Used to facilitate two-way data binding.
             * \@docs-private
             */
            this.valueChange = new EventEmitter();
            /**
             * Event emitted when the group's value changes.
             */
            this.change = new EventEmitter();
            this.appearance =
                defaultOptions && defaultOptions.appearance ? defaultOptions.appearance : 'standard';
        }
        /**
         * `name` attribute for the underlying `input` element.
         * @return {?}
         */
        get name() { return this._name; }
        /**
         * @param {?} value
         * @return {?}
         */
        set name(value) {
            this._name = value;
            if (this._buttonToggles) {
                this._buttonToggles.forEach((/**
                 * @param {?} toggle
                 * @return {?}
                 */
                toggle => {
                    toggle.name = this._name;
                    toggle._markForCheck();
                }));
            }
        }
        /**
         * Whether the toggle group is vertical.
         * @return {?}
         */
        get vertical() { return this._vertical; }
        /**
         * @param {?} value
         * @return {?}
         */
        set vertical(value) {
            this._vertical = coerceBooleanProperty(value);
        }
        /**
         * Value of the toggle group.
         * @return {?}
         */
        get value() {
            /** @type {?} */
            const selected = this._selectionModel ? this._selectionModel.selected : [];
            if (this.multiple) {
                return selected.map((/**
                 * @param {?} toggle
                 * @return {?}
                 */
                toggle => toggle.value));
            }
            return selected[0] ? selected[0].value : undefined;
        }
        /**
         * @param {?} newValue
         * @return {?}
         */
        set value(newValue) {
            this._setSelectionByValue(newValue);
            this.valueChange.emit(this.value);
        }
        /**
         * Selected button toggles in the group.
         * @return {?}
         */
        get selected() {
            /** @type {?} */
            const selected = this._selectionModel ? this._selectionModel.selected : [];
            return this.multiple ? selected : (selected[0] || null);
        }
        /**
         * Whether multiple button toggles can be selected.
         * @return {?}
         */
        get multiple() { return this._multiple; }
        /**
         * @param {?} value
         * @return {?}
         */
        set multiple(value) {
            this._multiple = coerceBooleanProperty(value);
        }
        /**
         * Whether multiple button toggle group is disabled.
         * @return {?}
         */
        get disabled() { return this._disabled; }
        /**
         * @param {?} value
         * @return {?}
         */
        set disabled(value) {
            this._disabled = coerceBooleanProperty(value);
            if (this._buttonToggles) {
                this._buttonToggles.forEach((/**
                 * @param {?} toggle
                 * @return {?}
                 */
                toggle => toggle._markForCheck()));
            }
        }
        /**
         * @return {?}
         */
        ngOnInit() {
            this._selectionModel = new SelectionModel(this.multiple, undefined, false);
        }
        /**
         * @return {?}
         */
        ngAfterContentInit() {
            this._selectionModel.select(...this._buttonToggles.filter((/**
             * @param {?} toggle
             * @return {?}
             */
            toggle => toggle.checked)));
        }
        /**
         * Sets the model value. Implemented as part of ControlValueAccessor.
         * @param {?} value Value to be set to the model.
         * @return {?}
         */
        writeValue(value) {
            this.value = value;
            this._changeDetector.markForCheck();
        }
        // Implemented as part of ControlValueAccessor.
        /**
         * @param {?} fn
         * @return {?}
         */
        registerOnChange(fn) {
            this._controlValueAccessorChangeFn = fn;
        }
        // Implemented as part of ControlValueAccessor.
        /**
         * @param {?} fn
         * @return {?}
         */
        registerOnTouched(fn) {
            this._onTouched = fn;
        }
        // Implemented as part of ControlValueAccessor.
        /**
         * @param {?} isDisabled
         * @return {?}
         */
        setDisabledState(isDisabled) {
            this.disabled = isDisabled;
        }
        /**
         * Dispatch change event with current selection and group value.
         * @return {?}
         */
        _emitChangeEvent() {
            /** @type {?} */
            const selected = this.selected;
            /** @type {?} */
            const source = Array.isArray(selected) ? selected[selected.length - 1] : selected;
            /** @type {?} */
            const event = new MatButtonToggleChange((/** @type {?} */ (source)), this.value);
            this._controlValueAccessorChangeFn(event.value);
            this.change.emit(event);
        }
        /**
         * Syncs a button toggle's selected state with the model value.
         * @param {?} toggle Toggle to be synced.
         * @param {?} select Whether the toggle should be selected.
         * @param {?=} isUserInput Whether the change was a result of a user interaction.
         * @param {?=} deferEvents Whether to defer emitting the change events.
         * @return {?}
         */
        _syncButtonToggle(toggle, select, isUserInput = false, deferEvents = false) {
            // Deselect the currently-selected toggle, if we're in single-selection
            // mode and the button being toggled isn't selected at the moment.
            if (!this.multiple && this.selected && !toggle.checked) {
                ((/** @type {?} */ (this.selected))).checked = false;
            }
            if (this._selectionModel) {
                if (select) {
                    this._selectionModel.select(toggle);
                }
                else {
                    this._selectionModel.deselect(toggle);
                }
            }
            else {
                deferEvents = true;
            }
            // We need to defer in some cases in order to avoid "changed after checked errors", however
            // the side-effect is that we may end up updating the model value out of sequence in others
            // The `deferEvents` flag allows us to decide whether to do it on a case-by-case basis.
            if (deferEvents) {
                Promise.resolve().then((/**
                 * @return {?}
                 */
                () => this._updateModelValue(isUserInput)));
            }
            else {
                this._updateModelValue(isUserInput);
            }
        }
        /**
         * Checks whether a button toggle is selected.
         * @param {?} toggle
         * @return {?}
         */
        _isSelected(toggle) {
            return this._selectionModel && this._selectionModel.isSelected(toggle);
        }
        /**
         * Determines whether a button toggle should be checked on init.
         * @param {?} toggle
         * @return {?}
         */
        _isPrechecked(toggle) {
            if (typeof this._rawValue === 'undefined') {
                return false;
            }
            if (this.multiple && Array.isArray(this._rawValue)) {
                return this._rawValue.some((/**
                 * @param {?} value
                 * @return {?}
                 */
                value => toggle.value != null && value === toggle.value));
            }
            return toggle.value === this._rawValue;
        }
        /**
         * Updates the selection state of the toggles in the group based on a value.
         * @private
         * @param {?} value
         * @return {?}
         */
        _setSelectionByValue(value) {
            this._rawValue = value;
            if (!this._buttonToggles) {
                return;
            }
            if (this.multiple && value) {
                if (!Array.isArray(value)) {
                    throw Error('Value must be an array in multiple-selection mode.');
                }
                this._clearSelection();
                value.forEach((/**
                 * @param {?} currentValue
                 * @return {?}
                 */
                (currentValue) => this._selectValue(currentValue)));
            }
            else {
                this._clearSelection();
                this._selectValue(value);
            }
        }
        /**
         * Clears the selected toggles.
         * @private
         * @return {?}
         */
        _clearSelection() {
            this._selectionModel.clear();
            this._buttonToggles.forEach((/**
             * @param {?} toggle
             * @return {?}
             */
            toggle => toggle.checked = false));
        }
        /**
         * Selects a value if there's a toggle that corresponds to it.
         * @private
         * @param {?} value
         * @return {?}
         */
        _selectValue(value) {
            /** @type {?} */
            const correspondingOption = this._buttonToggles.find((/**
             * @param {?} toggle
             * @return {?}
             */
            toggle => {
                return toggle.value != null && toggle.value === value;
            }));
            if (correspondingOption) {
                correspondingOption.checked = true;
                this._selectionModel.select(correspondingOption);
            }
        }
        /**
         * Syncs up the group's value with the model and emits the change event.
         * @private
         * @param {?} isUserInput
         * @return {?}
         */
        _updateModelValue(isUserInput) {
            // Only emit the change event for user input.
            if (isUserInput) {
                this._emitChangeEvent();
            }
            // Note: we emit this one no matter whether it was a user interaction, because
            // it is used by Angular to sync up the two-way data binding.
            this.valueChange.emit(this.value);
        }
    }
    MatButtonToggleGroup.ɵfac = function MatButtonToggleGroup_Factory(t) { return new (t || MatButtonToggleGroup)(ɵɵdirectiveInject(ChangeDetectorRef), ɵɵdirectiveInject(MAT_BUTTON_TOGGLE_DEFAULT_OPTIONS, 8)); };
    MatButtonToggleGroup.ɵdir = ɵɵdefineDirective({ type: MatButtonToggleGroup, selectors: [["mat-button-toggle-group"]], contentQueries: function MatButtonToggleGroup_ContentQueries(rf, ctx, dirIndex) { if (rf & 1) {
            ɵɵcontentQuery(dirIndex, MatButtonToggle, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._buttonToggles = _t);
        } }, hostAttrs: ["role", "group", 1, "mat-button-toggle-group"], hostVars: 5, hostBindings: function MatButtonToggleGroup_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵattribute("aria-disabled", ctx.disabled);
            ɵɵclassProp("mat-button-toggle-vertical", ctx.vertical)("mat-button-toggle-group-appearance-standard", ctx.appearance === "standard");
        } }, inputs: { appearance: "appearance", name: "name", vertical: "vertical", value: "value", multiple: "multiple", disabled: "disabled" }, outputs: { valueChange: "valueChange", change: "change" }, exportAs: ["matButtonToggleGroup"], features: [ɵɵProvidersFeature([
                MAT_BUTTON_TOGGLE_GROUP_VALUE_ACCESSOR,
                { provide: MatButtonToggleGroupMultiple, useExisting: MatButtonToggleGroup },
            ])] });
    /** @nocollapse */
    MatButtonToggleGroup.ctorParameters = () => [
        { type: ChangeDetectorRef },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [MAT_BUTTON_TOGGLE_DEFAULT_OPTIONS,] }] }
    ];
    MatButtonToggleGroup.propDecorators = {
        _buttonToggles: [{ type: ContentChildren, args: [forwardRef((/**
                     * @return {?}
                     */
                    () => MatButtonToggle)), {
                        // Note that this would technically pick up toggles
                        // from nested groups, but that's not a case that we support.
                        descendants: true
                    },] }],
        appearance: [{ type: Input }],
        name: [{ type: Input }],
        vertical: [{ type: Input }],
        value: [{ type: Input }],
        valueChange: [{ type: Output }],
        multiple: [{ type: Input }],
        disabled: [{ type: Input }],
        change: [{ type: Output }]
    };
    // Boilerplate for applying mixins to the MatButtonToggle class.
    /**
     * \@docs-private
     */
    class MatButtonToggleBase {
    }
    /** @type {?} */
    const _MatButtonToggleMixinBase = mixinDisableRipple(MatButtonToggleBase);
    /**
     * Single button inside of a toggle group.
     */
    class MatButtonToggle extends _MatButtonToggleMixinBase {
        /**
         * @param {?} toggleGroup
         * @param {?} _changeDetectorRef
         * @param {?} _elementRef
         * @param {?} _focusMonitor
         * @param {?} defaultTabIndex
         * @param {?=} defaultOptions
         */
        constructor(toggleGroup, _changeDetectorRef, _elementRef, _focusMonitor, 
        // @breaking-change 8.0.0 `defaultTabIndex` to be made a required parameter.
        defaultTabIndex, defaultOptions) {
            super();
            this._changeDetectorRef = _changeDetectorRef;
            this._elementRef = _elementRef;
            this._focusMonitor = _focusMonitor;
            this._isSingleSelector = false;
            this._checked = false;
            /**
             * Users can specify the `aria-labelledby` attribute which will be forwarded to the input element
             */
            this.ariaLabelledby = null;
            this._disabled = false;
            /**
             * Event emitted when the group value changes.
             */
            this.change = new EventEmitter();
            /** @type {?} */
            const parsedTabIndex = Number(defaultTabIndex);
            this.tabIndex = (parsedTabIndex || parsedTabIndex === 0) ? parsedTabIndex : null;
            this.buttonToggleGroup = toggleGroup;
            this.appearance =
                defaultOptions && defaultOptions.appearance ? defaultOptions.appearance : 'standard';
        }
        /**
         * Unique ID for the underlying `button` element.
         * @return {?}
         */
        get buttonId() { return `${this.id}-button`; }
        /**
         * The appearance style of the button.
         * @return {?}
         */
        get appearance() {
            return this.buttonToggleGroup ? this.buttonToggleGroup.appearance : this._appearance;
        }
        /**
         * @param {?} value
         * @return {?}
         */
        set appearance(value) {
            this._appearance = value;
        }
        /**
         * Whether the button is checked.
         * @return {?}
         */
        get checked() {
            return this.buttonToggleGroup ? this.buttonToggleGroup._isSelected(this) : this._checked;
        }
        /**
         * @param {?} value
         * @return {?}
         */
        set checked(value) {
            /** @type {?} */
            const newValue = coerceBooleanProperty(value);
            if (newValue !== this._checked) {
                this._checked = newValue;
                if (this.buttonToggleGroup) {
                    this.buttonToggleGroup._syncButtonToggle(this, this._checked);
                }
                this._changeDetectorRef.markForCheck();
            }
        }
        /**
         * Whether the button is disabled.
         * @return {?}
         */
        get disabled() {
            return this._disabled || (this.buttonToggleGroup && this.buttonToggleGroup.disabled);
        }
        /**
         * @param {?} value
         * @return {?}
         */
        set disabled(value) { this._disabled = coerceBooleanProperty(value); }
        /**
         * @return {?}
         */
        ngOnInit() {
            /** @type {?} */
            const group = this.buttonToggleGroup;
            this._isSingleSelector = group && !group.multiple;
            this._type = this._isSingleSelector ? 'radio' : 'checkbox';
            this.id = this.id || `mat-button-toggle-${_uniqueIdCounter$1++}`;
            if (this._isSingleSelector) {
                this.name = group.name;
            }
            if (group) {
                if (group._isPrechecked(this)) {
                    this.checked = true;
                }
                else if (group._isSelected(this) !== this._checked) {
                    // As as side effect of the circular dependency between the toggle group and the button,
                    // we may end up in a state where the button is supposed to be checked on init, but it
                    // isn't, because the checked value was assigned too early. This can happen when Ivy
                    // assigns the static input value before the `ngOnInit` has run.
                    group._syncButtonToggle(this, this._checked);
                }
            }
            this._focusMonitor.monitor(this._elementRef, true);
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            /** @type {?} */
            const group = this.buttonToggleGroup;
            this._focusMonitor.stopMonitoring(this._elementRef);
            // Remove the toggle from the selection once it's destroyed. Needs to happen
            // on the next tick in order to avoid "changed after checked" errors.
            if (group && group._isSelected(this)) {
                group._syncButtonToggle(this, false, false, true);
            }
        }
        /**
         * Focuses the button.
         * @param {?=} options
         * @return {?}
         */
        focus(options) {
            this._buttonElement.nativeElement.focus(options);
        }
        /**
         * Checks the button toggle due to an interaction with the underlying native button.
         * @return {?}
         */
        _onButtonClick() {
            /** @type {?} */
            const newChecked = this._isSingleSelector ? true : !this._checked;
            if (newChecked !== this._checked) {
                this._checked = newChecked;
                if (this.buttonToggleGroup) {
                    this.buttonToggleGroup._syncButtonToggle(this, this._checked, true);
                    this.buttonToggleGroup._onTouched();
                }
            }
            // Emit a change event when it's the single selector
            this.change.emit(new MatButtonToggleChange(this, this.value));
        }
        /**
         * Marks the button toggle as needing checking for change detection.
         * This method is exposed because the parent button toggle group will directly
         * update bound properties of the radio button.
         * @return {?}
         */
        _markForCheck() {
            // When the group value changes, the button will not be notified.
            // Use `markForCheck` to explicit update button toggle's status.
            this._changeDetectorRef.markForCheck();
        }
    }
    MatButtonToggle.ɵfac = function MatButtonToggle_Factory(t) { return new (t || MatButtonToggle)(ɵɵdirectiveInject(MatButtonToggleGroup, 8), ɵɵdirectiveInject(ChangeDetectorRef), ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(FocusMonitor), ɵɵinjectAttribute('tabindex'), ɵɵdirectiveInject(MAT_BUTTON_TOGGLE_DEFAULT_OPTIONS, 8)); };
    MatButtonToggle.ɵcmp = ɵɵdefineComponent({ type: MatButtonToggle, selectors: [["mat-button-toggle"]], viewQuery: function MatButtonToggle_Query(rf, ctx) { if (rf & 1) {
            ɵɵviewQuery(_c0$s, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._buttonElement = _t.first);
        } }, hostAttrs: [1, "mat-button-toggle"], hostVars: 11, hostBindings: function MatButtonToggle_HostBindings(rf, ctx) { if (rf & 1) {
            ɵɵlistener("focus", function MatButtonToggle_focus_HostBindingHandler() { return ctx.focus(); });
        } if (rf & 2) {
            ɵɵattribute("tabindex", 0 - 1)("id", ctx.id)("name", null);
            ɵɵclassProp("mat-button-toggle-standalone", !ctx.buttonToggleGroup)("mat-button-toggle-checked", ctx.checked)("mat-button-toggle-disabled", ctx.disabled)("mat-button-toggle-appearance-standard", ctx.appearance === "standard");
        } }, inputs: { disableRipple: "disableRipple", ariaLabelledby: ["aria-labelledby", "ariaLabelledby"], tabIndex: "tabIndex", appearance: "appearance", checked: "checked", disabled: "disabled", id: "id", name: "name", ariaLabel: ["aria-label", "ariaLabel"], value: "value" }, outputs: { change: "change" }, exportAs: ["matButtonToggle"], features: [ɵɵInheritDefinitionFeature], ngContentSelectors: _c1$i, decls: 6, vars: 9, consts: [["type", "button", 1, "mat-button-toggle-button", "mat-focus-indicator", 3, "id", "disabled", "click"], ["button", ""], [1, "mat-button-toggle-label-content"], [1, "mat-button-toggle-focus-overlay"], ["matRipple", "", 1, "mat-button-toggle-ripple", 3, "matRippleTrigger", "matRippleDisabled"]], template: function MatButtonToggle_Template(rf, ctx) { if (rf & 1) {
            ɵɵprojectionDef();
            ɵɵelementStart(0, "button", 0, 1);
            ɵɵlistener("click", function MatButtonToggle_Template_button_click_0_listener() { return ctx._onButtonClick(); });
            ɵɵelementStart(2, "div", 2);
            ɵɵprojection(3);
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelement(4, "div", 3);
            ɵɵelement(5, "div", 4);
        } if (rf & 2) {
            const _r0 = ɵɵreference(1);
            ɵɵproperty("id", ctx.buttonId)("disabled", ctx.disabled || null);
            ɵɵattribute("tabindex", ctx.disabled ? 0 - 1 : ctx.tabIndex)("aria-pressed", ctx.checked)("name", ctx.name || null)("aria-label", ctx.ariaLabel)("aria-labelledby", ctx.ariaLabelledby);
            ɵɵadvance(5);
            ɵɵproperty("matRippleTrigger", _r0)("matRippleDisabled", ctx.disableRipple || ctx.disabled);
        } }, directives: [MatRipple], styles: [".mat-button-toggle-standalone,.mat-button-toggle-group{position:relative;display:inline-flex;flex-direction:row;white-space:nowrap;overflow:hidden;border-radius:2px;-webkit-tap-highlight-color:transparent}.cdk-high-contrast-active .mat-button-toggle-standalone,.cdk-high-contrast-active .mat-button-toggle-group{outline:solid 1px}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard,.mat-button-toggle-group-appearance-standard{border-radius:4px}.cdk-high-contrast-active .mat-button-toggle-standalone.mat-button-toggle-appearance-standard,.cdk-high-contrast-active .mat-button-toggle-group-appearance-standard{outline:0}.mat-button-toggle-vertical{flex-direction:column}.mat-button-toggle-vertical .mat-button-toggle-label-content{display:block}.mat-button-toggle{white-space:nowrap;position:relative}.mat-button-toggle .mat-icon svg{vertical-align:top}.mat-button-toggle.cdk-keyboard-focused .mat-button-toggle-focus-overlay{opacity:1}.cdk-high-contrast-active .mat-button-toggle.cdk-keyboard-focused .mat-button-toggle-focus-overlay{opacity:.5}.mat-button-toggle-appearance-standard:not(.mat-button-toggle-disabled):hover .mat-button-toggle-focus-overlay{opacity:.04}.mat-button-toggle-appearance-standard.cdk-keyboard-focused:not(.mat-button-toggle-disabled) .mat-button-toggle-focus-overlay{opacity:.12}.cdk-high-contrast-active .mat-button-toggle-appearance-standard.cdk-keyboard-focused:not(.mat-button-toggle-disabled) .mat-button-toggle-focus-overlay{opacity:.5}@media(hover: none){.mat-button-toggle-appearance-standard:not(.mat-button-toggle-disabled):hover .mat-button-toggle-focus-overlay{display:none}}.mat-button-toggle-label-content{-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;display:inline-block;line-height:36px;padding:0 16px;position:relative}.mat-button-toggle-appearance-standard .mat-button-toggle-label-content{line-height:48px;padding:0 12px}.mat-button-toggle-label-content>*{vertical-align:middle}.mat-button-toggle-focus-overlay{border-radius:inherit;pointer-events:none;opacity:0;top:0;left:0;right:0;bottom:0;position:absolute}.mat-button-toggle-checked .mat-button-toggle-focus-overlay{border-bottom:solid 36px}.cdk-high-contrast-active .mat-button-toggle-checked .mat-button-toggle-focus-overlay{opacity:.5;height:0}.cdk-high-contrast-active .mat-button-toggle-checked.mat-button-toggle-appearance-standard .mat-button-toggle-focus-overlay{border-bottom:solid 48px}.mat-button-toggle .mat-button-toggle-ripple{top:0;left:0;right:0;bottom:0;position:absolute;pointer-events:none}.mat-button-toggle-button{border:0;background:none;color:inherit;padding:0;margin:0;font:inherit;outline:none;width:100%;cursor:pointer}.mat-button-toggle-disabled .mat-button-toggle-button{cursor:default}.mat-button-toggle-button::-moz-focus-inner{border:0}\n"], encapsulation: 2, changeDetection: 0 });
    /** @nocollapse */
    MatButtonToggle.ctorParameters = () => [
        { type: MatButtonToggleGroup, decorators: [{ type: Optional }] },
        { type: ChangeDetectorRef },
        { type: ElementRef },
        { type: FocusMonitor },
        { type: String, decorators: [{ type: Attribute, args: ['tabindex',] }] },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [MAT_BUTTON_TOGGLE_DEFAULT_OPTIONS,] }] }
    ];
    MatButtonToggle.propDecorators = {
        ariaLabel: [{ type: Input, args: ['aria-label',] }],
        ariaLabelledby: [{ type: Input, args: ['aria-labelledby',] }],
        _buttonElement: [{ type: ViewChild, args: ['button',] }],
        id: [{ type: Input }],
        name: [{ type: Input }],
        value: [{ type: Input }],
        tabIndex: [{ type: Input }],
        appearance: [{ type: Input }],
        checked: [{ type: Input }],
        disabled: [{ type: Input }],
        change: [{ type: Output }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/button-toggle/button-toggle-module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class MatButtonToggleModule {
    }
    MatButtonToggleModule.ɵmod = ɵɵdefineNgModule({ type: MatButtonToggleModule });
    MatButtonToggleModule.ɵinj = ɵɵdefineInjector({ factory: function MatButtonToggleModule_Factory(t) { return new (t || MatButtonToggleModule)(); }, imports: [[MatCommonModule, MatRippleModule],
            MatCommonModule] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(MatButtonToggleModule, { declarations: function () { return [MatButtonToggleGroup,
            MatButtonToggle]; }, imports: function () { return [MatCommonModule, MatRippleModule]; }, exports: function () { return [MatCommonModule,
            MatButtonToggleGroup,
            MatButtonToggle]; } }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/widgets/dropdown/dropdown_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class DropdownModule {
    }
    DropdownModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [DropdownComponent],
                    exports: [DropdownComponent],
                    imports: [CommonModule, MatSelectModule],
                },] },
    ];
    /** @nocollapse */ DropdownModule.ɵmod = ɵɵdefineNgModule({ type: DropdownModule });
    /** @nocollapse */ DropdownModule.ɵinj = ɵɵdefineInjector({ factory: function DropdownModule_Factory(t) { return new (t || DropdownModule)(); }, imports: [[CommonModule, MatSelectModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(DropdownModule, { declarations: [DropdownComponent], imports: [CommonModule, MatSelectModule], exports: [DropdownComponent] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/views/right_pane/right_pane_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class RightPaneModule {
    }
    RightPaneModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [
                        RightPaneComponent,
                        SettingsViewComponent,
                        SettingsViewContainer,
                    ],
                    exports: [RightPaneComponent],
                    imports: [
                        CommonModule,
                        DropdownModule,
                        MatButtonModule,
                        MatButtonToggleModule,
                        MatCheckboxModule,
                        MatIconModule,
                        MatSelectModule,
                        MatSliderModule,
                    ],
                },] },
    ];
    /** @nocollapse */ RightPaneModule.ɵmod = ɵɵdefineNgModule({ type: RightPaneModule });
    /** @nocollapse */ RightPaneModule.ɵinj = ɵɵdefineInjector({ factory: function RightPaneModule_Factory(t) { return new (t || RightPaneModule)(); }, imports: [[
                CommonModule,
                DropdownModule,
                MatButtonModule,
                MatButtonToggleModule,
                MatCheckboxModule,
                MatIconModule,
                MatSelectModule,
                MatSliderModule,
            ]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(RightPaneModule, { declarations: [RightPaneComponent,
            SettingsViewComponent,
            SettingsViewContainer], imports: [CommonModule,
            DropdownModule,
            MatButtonModule,
            MatButtonToggleModule,
            MatCheckboxModule,
            MatIconModule,
            MatSelectModule,
            MatSliderModule], exports: [RightPaneComponent] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/views/main_view/main_view_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class MainViewModule {
    }
    MainViewModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [
                        CardGridComponent,
                        CardGridContainer,
                        CardGroupsComponent,
                        CardGroupsContainer,
                        FilteredViewComponent,
                        FilteredViewContainer,
                        MainViewComponent,
                        MainViewContainer,
                        MetricsFilterInputComponent,
                        MetricsFilterInputContainer,
                        PinnedViewComponent,
                        PinnedViewContainer,
                    ],
                    exports: [MainViewContainer],
                    imports: [
                        CommonModule,
                        CardRendererModule,
                        MatAutocompleteModule,
                        MatButtonModule,
                        MatIconModule,
                        MatInputModule,
                        RightPaneModule,
                    ],
                },] },
    ];
    /** @nocollapse */ MainViewModule.ɵmod = ɵɵdefineNgModule({ type: MainViewModule });
    /** @nocollapse */ MainViewModule.ɵinj = ɵɵdefineInjector({ factory: function MainViewModule_Factory(t) { return new (t || MainViewModule)(); }, imports: [[
                CommonModule,
                CardRendererModule,
                MatAutocompleteModule,
                MatButtonModule,
                MatIconModule,
                MatInputModule,
                RightPaneModule,
            ]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(MainViewModule, { declarations: [CardGridComponent,
            CardGridContainer,
            CardGroupsComponent,
            CardGroupsContainer,
            FilteredViewComponent,
            FilteredViewContainer,
            MainViewComponent,
            MainViewContainer,
            MetricsFilterInputComponent,
            MetricsFilterInputContainer,
            PinnedViewComponent,
            PinnedViewContainer], imports: [CommonModule,
            CardRendererModule,
            MatAutocompleteModule,
            MatButtonModule,
            MatIconModule,
            MatInputModule,
            RightPaneModule], exports: [MainViewContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/views/metrics_views_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class MetricsViewsModule {
    }
    MetricsViewsModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [MetricsDashboardContainer],
                    exports: [MetricsDashboardContainer],
                    imports: [CommonModule, RightPaneModule, RunsSelectorModule, MainViewModule],
                },] },
    ];
    /** @nocollapse */ MetricsViewsModule.ɵmod = ɵɵdefineNgModule({ type: MetricsViewsModule });
    /** @nocollapse */ MetricsViewsModule.ɵinj = ɵɵdefineInjector({ factory: function MetricsViewsModule_Factory(t) { return new (t || MetricsViewsModule)(); }, imports: [[CommonModule, RightPaneModule, RunsSelectorModule, MainViewModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(MetricsViewsModule, { declarations: [MetricsDashboardContainer], imports: [CommonModule, RightPaneModule, RunsSelectorModule, MainViewModule], exports: [MetricsDashboardContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/metrics/metrics_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const CREATE_PIN_MAX_EXCEEDED_TEXT = `Max pin limit exceeded. Remove existing` +
        ` pins before adding more. See ` +
        `https://github.com/tensorflow/tensorboard/issues/4242`;
    // Note: Angular can only reference symbols from the @NgModule if they are
    // exported.
    /**
     * @return {?}
     */
    function alertActionProvider$1() {
        return [
            {
                actionCreator: cardPinStateToggled,
                alertFromAction: (/**
                 * @param {?} action
                 * @return {?}
                 */
                (action) => {
                    const { wasPinned, canCreateNewPins } = (/** @type {?} */ (action));
                    if (!wasPinned && !canCreateNewPins) {
                        return { localizedMessage: CREATE_PIN_MAX_EXCEEDED_TEXT };
                    }
                    return null;
                }),
            },
        ];
    }
    class MetricsModule {
    }
    MetricsModule.decorators = [
        { type: NgModule, args: [{
                    imports: [
                        CommonModule,
                        RunsSelectorModule,
                        PluginRegistryModule.forPlugin(METRICS_PLUGIN_ID, MetricsDashboardContainer),
                        MetricsDataSourceModule,
                        MetricsViewsModule,
                        StoreModule.forFeature(METRICS_FEATURE_KEY, reducers$7),
                        EffectsModule.forFeature([MetricsEffects]),
                        AlertActionModule.registerAlertActions(alertActionProvider$1),
                    ],
                    entryComponents: [MetricsDashboardContainer],
                },] },
    ];
    /** @nocollapse */ MetricsModule.ɵmod = ɵɵdefineNgModule({ type: MetricsModule });
    /** @nocollapse */ MetricsModule.ɵinj = ɵɵdefineInjector({ factory: function MetricsModule_Factory(t) { return new (t || MetricsModule)(); }, imports: [[
                CommonModule,
                RunsSelectorModule,
                PluginRegistryModule.forPlugin(METRICS_PLUGIN_ID, MetricsDashboardContainer),
                MetricsDataSourceModule,
                MetricsViewsModule,
                StoreModule.forFeature(METRICS_FEATURE_KEY, reducers$7),
                EffectsModule.forFeature([MetricsEffects]),
                AlertActionModule.registerAlertActions(alertActionProvider$1),
            ]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(MetricsModule, { imports: [CommonModule,
            RunsSelectorModule, PluginRegistryModule, MetricsDataSourceModule,
            MetricsViewsModule, StoreFeatureModule, EffectsFeatureModule, AlertActionModule] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/inactive/inactive_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class InactiveComponent$1 {
    }
    InactiveComponent$1.decorators = [
        { type: Component, args: [{
                    selector: 'npmi-inactive-view',
                    templateUrl: './inactive_component.ng.html',
                    styleUrls: ['./inactive_component.css'],
                },] },
    ];
    /** @nocollapse */ InactiveComponent$1.ɵfac = function InactiveComponent_Factory(t) { return new (t || InactiveComponent$1)(); };
    /** @nocollapse */ InactiveComponent$1.ɵcmp = ɵɵdefineComponent({ type: InactiveComponent$1, selectors: [["npmi-inactive-view"]], decls: 6, vars: 0, consts: [[1, "container"], [1, "title"]], template: function InactiveComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "div");
            ɵɵelementStart(1, "div", 0);
            ɵɵelementStart(2, "div", 1);
            ɵɵtext(3, "nPMI is inactive because no data is available.");
            ɵɵelementEnd();
            ɵɵelementStart(4, "div");
            ɵɵtext(5, " To use the nPMI, calculate nPMI values, and log them using the summary writer. ");
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementEnd();
        } }, styles: [".container[_ngcontent-%COMP%] {\n  height: 100%;\n  font-family: Roboto;\n  font-size: 15px;\n  padding: 50px;\n}\n\n.title[_ngcontent-%COMP%] {\n  font-size: 135%;\n  font-weight: bold;\n  margin-bottom: 25px;\n}"] });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/store/npmi_types.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const NPMI_FEATURE_KEY = 'npmi';
    /** @enum {number} */
    const Operator = {
        AND: 0,
    };
    Operator[Operator.AND] = 'AND';
    /** @enum {number} */
    const ArithmeticKind = {
        METRIC: 0,
        OPERATOR: 1,
    };
    ArithmeticKind[ArithmeticKind.METRIC] = 'METRIC';
    ArithmeticKind[ArithmeticKind.OPERATOR] = 'OPERATOR';
    /** @enum {number} */
    const SortOrder = {
        DOWN: 0,
        UP: 1,
    };
    SortOrder[SortOrder.DOWN] = 'DOWN';
    SortOrder[SortOrder.UP] = 'UP';

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/store/npmi_selectors.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const selectNpmiState = createFeatureSelector(NPMI_FEATURE_KEY);
    /** @type {?} */
    const getPluginDataLoaded = createSelector(selectNpmiState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return state.pluginDataLoaded;
    }));
    /** @type {?} */
    const getAnnotationData = createSelector(selectNpmiState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return state.annotationData;
    }));
    /** @type {?} */
    const getRunToMetrics = createSelector(selectNpmiState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return state.runToMetrics;
    }));
    /** @type {?} */
    const getSelectedAnnotations = createSelector(selectNpmiState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return state.selectedAnnotations;
    }));
    /** @type {?} */
    const getFlaggedAnnotations = createSelector(selectNpmiState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return state.flaggedAnnotations;
    }));
    /** @type {?} */
    const getHiddenAnnotations = createSelector(selectNpmiState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return state.hiddenAnnotations;
    }));
    /** @type {?} */
    const getAnnotationsRegex = createSelector(selectNpmiState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return state.annotationsRegex;
    }));
    /** @type {?} */
    const getMetricsRegex = createSelector(selectNpmiState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return state.metricsRegex;
    }));
    /** @type {?} */
    const getMetricArithmetic = createSelector(selectNpmiState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return state.metricArithmetic;
    }));
    /** @type {?} */
    const getMetricFilters = createSelector(selectNpmiState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return state.metricFilters;
    }));
    /** @type {?} */
    const getAnnotationSort = createSelector(selectNpmiState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return state.sort;
    }));
    /** @type {?} */
    const getPCExpanded = createSelector(selectNpmiState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return state.pcExpanded;
    }));
    /** @type {?} */
    const getAnnotationsExpanded = createSelector(selectNpmiState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return state.annotationsExpanded;
    }));
    /** @type {?} */
    const getSidebarExpanded = createSelector(selectNpmiState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return state.sidebarExpanded;
    }));
    /** @type {?} */
    const getShowCounts = createSelector(selectNpmiState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return state.showCounts;
    }));
    /** @type {?} */
    const getShowHiddenAnnotations = createSelector(selectNpmiState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return state.showHiddenAnnotations;
    }));
    /** @type {?} */
    const getSidebarWidth = createSelector(selectNpmiState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return state.sidebarWidth;
    }));

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * Actions for the NPMI Component.
     */
    const npmiLoaded = createAction('[NPMI] nPMI Loaded');
    const npmiPluginDataRequested = createAction('[NPMI] nPMI Plugin Data Requested');
    const npmiPluginDataLoaded = createAction('[NPMI] nPMI Plugin Data Loaded', props());
    const npmiPluginDataRequestFailed = createAction('[NPMI] nPMI Plugin Data Request Failed');
    const npmiToggleSelectedAnnotations = createAction('[NPMI] Adding/Removing Annotations to/from Selected', props());
    const npmiSetSelectedAnnotations = createAction('[NPMI] Annotations Set', props());
    const npmiClearSelectedAnnotations = createAction('[NPMI] Clearing the Annotation Selection');
    const npmiToggleAnnotationFlags = createAction('[NPMI] Adding/Removing Annotations to/from Flagged', props());
    const npmiToggleAnnotationsHidden = createAction('[NPMI] Adding/Removing Annotations to/from Hidden', props());
    const npmiAnnotationsRegexChanged = createAction('[NPMI] Annotations Regex Changed', props());
    const npmiMetricsRegexChanged = createAction('[NPMI] Metrics Regex Changed', props());
    const npmiAddMetricFilter = createAction('[NPMI] Metric Filter Added', props());
    const npmiRemoveMetricFilter = createAction('[NPMI] Metric Filter Removed', props());
    const npmiChangeMetricFilter = createAction('[NPMI] Metric Filter Changed', props());
    const npmiChangeAnnotationSort = createAction('[NPMI] Change Annotation Sort', props());
    const npmiToggleParallelCoordinatesExpanded = createAction('[NPMI] Toggle PC Expanded');
    const npmiToggleAnnotationsExpanded = createAction('[NPMI] Toggle Annotations Expanded');
    const npmiToggleSidebarExpanded = createAction('[NPMI] Toggle Sidebar Expanded');
    const npmiToggleShowCounts = createAction('[NPMI] Toggle Show Counts');
    const npmiToggleShowHiddenAnnotations = createAction('[NPMI] Toggle Show Hidden Annotations');
    const npmiChangeSidebarWidth = createAction('[NPMI] Change Sidebar Width', props());

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/util/metric_type.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     * @param {?} metric
     * @return {?}
     */
    function metricIsMetricCount(metric) {
        return metric.startsWith('count@');
    }
    /**
     * @param {?} metric
     * @return {?}
     */
    function metricIsNpmi(metric) {
        return metric.startsWith('nPMI@') || metric.startsWith('nPMI_diff@');
    }
    /**
     * @param {?} metric
     * @return {?}
     */
    function metricIsNpmiAndNotDiff(metric) {
        return metric.startsWith('nPMI@');
    }
    /**
     * @param {?} metricString
     * @return {?}
     */
    function stripMetricString(metricString) {
        /** @type {?} */
        const strippedString = metricString.split('@', 2)[1];
        return strippedString;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/store/npmi_reducers.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const initialState$6 = {
        pluginDataLoaded: {
            state: DataLoadState.NOT_LOADED,
            lastLoadedTimeInMs: null,
        },
        annotationData: {},
        runToMetrics: {},
        embeddingData: {},
        selectedAnnotations: [],
        flaggedAnnotations: [],
        hiddenAnnotations: [],
        annotationsRegex: '',
        metricsRegex: '',
        metricArithmetic: [],
        metricFilters: {},
        sort: {
            metric: '',
            order: SortOrder.DOWN,
        },
        pcExpanded: true,
        annotationsExpanded: true,
        sidebarExpanded: true,
        showCounts: true,
        showHiddenAnnotations: false,
        sidebarWidth: 300,
    };
    /** @type {?} */
    const reducer$6 = createReducer(initialState$6, on(npmiPluginDataRequested, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return Object.assign(Object.assign({}, state), { pluginDataLoaded: Object.assign(Object.assign({}, state.pluginDataLoaded), { state: DataLoadState.LOADING }) });
    })), on(npmiPluginDataRequestFailed, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return Object.assign(Object.assign({}, state), { pluginDataLoaded: Object.assign(Object.assign({}, state.pluginDataLoaded), { state: DataLoadState.FAILED }) });
    })), on(npmiPluginDataLoaded, (/**
     * @param {?} state
     * @param {?} __1
     * @return {?}
     */
    (state, { annotationData, metrics, embeddingData }) => {
        /** @type {?} */
        const runToMetrics = {};
        for (const key in metrics) {
            // Init Metrics Data
            runToMetrics[key] = [];
            for (const value of metrics[key]) {
                if (metricIsNpmi(value)) {
                    runToMetrics[key].push(value);
                }
            }
        }
        return Object.assign(Object.assign({}, state), { runToMetrics: runToMetrics, annotationData: annotationData, embeddingData: embeddingData, pluginDataLoaded: {
                state: DataLoadState.LOADED,
                lastLoadedTimeInMs: Date.now(),
            } });
    })), on(npmiToggleSelectedAnnotations, (/**
     * @param {?} state
     * @param {?} __1
     * @return {?}
     */
    (state, { annotations }) => {
        /** @type {?} */
        const combinedSelectedAnnotations = new Set([
            ...state.selectedAnnotations,
            ...annotations,
        ]);
        if (combinedSelectedAnnotations.size === state.selectedAnnotations.length) {
            // If all annotations are already flagged, user wants to remove them
            for (const annotation of annotations) {
                combinedSelectedAnnotations.delete(annotation);
            }
        }
        return Object.assign(Object.assign({}, state), { selectedAnnotations: [...combinedSelectedAnnotations] });
    })), on(npmiSetSelectedAnnotations, (/**
     * @param {?} state
     * @param {?} __1
     * @return {?}
     */
    (state, { annotations }) => {
        return Object.assign(Object.assign({}, state), { selectedAnnotations: annotations });
    })), on(npmiClearSelectedAnnotations, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return Object.assign(Object.assign({}, state), { selectedAnnotations: [] });
    })), on(npmiToggleAnnotationFlags, (/**
     * @param {?} state
     * @param {?} __1
     * @return {?}
     */
    (state, { annotations }) => {
        /** @type {?} */
        const combinedFlaggedAnnotations = new Set([
            ...state.flaggedAnnotations,
            ...annotations,
        ]);
        if (combinedFlaggedAnnotations.size === state.flaggedAnnotations.length) {
            // If all annotations are already flagged, user wants to remove them
            for (const annotation of annotations) {
                combinedFlaggedAnnotations.delete(annotation);
            }
        }
        return Object.assign(Object.assign({}, state), { flaggedAnnotations: [...combinedFlaggedAnnotations], selectedAnnotations: [] });
    })), on(npmiToggleAnnotationsHidden, (/**
     * @param {?} state
     * @param {?} __1
     * @return {?}
     */
    (state, { annotations }) => {
        /** @type {?} */
        const combinedHiddenAnnotations = new Set([
            ...state.hiddenAnnotations,
            ...annotations,
        ]);
        if (combinedHiddenAnnotations.size === state.hiddenAnnotations.length) {
            // If all annotations are already flagged, user wants to remove them
            for (const annotation of annotations) {
                combinedHiddenAnnotations.delete(annotation);
            }
        }
        return Object.assign(Object.assign({}, state), { hiddenAnnotations: [...combinedHiddenAnnotations], selectedAnnotations: [] });
    })), on(npmiAnnotationsRegexChanged, (/**
     * @param {?} state
     * @param {?} __1
     * @return {?}
     */
    (state, { regex }) => {
        return Object.assign(Object.assign({}, state), { annotationsRegex: regex });
    })), on(npmiMetricsRegexChanged, (/**
     * @param {?} state
     * @param {?} __1
     * @return {?}
     */
    (state, { regex }) => {
        return Object.assign(Object.assign({}, state), { metricsRegex: regex });
    })), on(npmiAddMetricFilter, (/**
     * @param {?} state
     * @param {?} __1
     * @return {?}
     */
    (state, { metric }) => {
        // Only add if not already in active filters
        if (state.metricFilters[metric]) {
            return state;
        }
        // Add so that arithmetic is still correct
        /** @type {?} */
        const newContent = [];
        if (state.metricArithmetic.length !== 0) {
            newContent.push({
                kind: ArithmeticKind.OPERATOR,
                operator: Operator.AND,
            });
        }
        newContent.push({ kind: ArithmeticKind.METRIC, metric: metric });
        return Object.assign(Object.assign({}, state), { metricArithmetic: [...state.metricArithmetic, ...newContent], metricFilters: Object.assign(Object.assign({}, state.metricFilters), { [metric]: {
                    max: 1.0,
                    min: -1.0,
                    includeNaN: false,
                } }), sort: {
                metric,
                order: SortOrder.DOWN,
            } });
    })), on(npmiRemoveMetricFilter, (/**
     * @param {?} state
     * @param {?} __1
     * @return {?}
     */
    (state, { metric }) => {
        if (!state.metricFilters[metric]) {
            return state;
        }
        // Remove the correct elements of the arithmetic as well
        /** @type {?} */
        let arithmeticIndex = 0;
        /** @type {?} */
        let startSlice = 0;
        /** @type {?} */
        let endSlice = 2;
        const _a = state.metricFilters, _b = metric, value = _a[_b], map = __rest(_a, [typeof _b === "symbol" ? _b : _b + ""]);
        for (const index in state.metricArithmetic) {
            /** @type {?} */
            const element = state.metricArithmetic[index];
            if (element.kind === ArithmeticKind.METRIC) {
                if (element.metric === metric) {
                    arithmeticIndex = parseInt(index);
                }
            }
        }
        if (arithmeticIndex !== 0) {
            startSlice = arithmeticIndex - 1;
            endSlice = arithmeticIndex + 1;
        }
        return Object.assign(Object.assign({}, state), { metricArithmetic: [
                ...state.metricArithmetic.slice(0, startSlice),
                ...state.metricArithmetic.slice(endSlice),
            ], metricFilters: map });
    })), on(npmiChangeMetricFilter, (/**
     * @param {?} state
     * @param {?} __1
     * @return {?}
     */
    (state, { metric, max, min, includeNaN }) => {
        if (!state.metricFilters[metric]) {
            return state;
        }
        return Object.assign(Object.assign({}, state), { metricFilters: Object.assign(Object.assign({}, state.metricFilters), { [metric]: {
                    max: max,
                    min: min,
                    includeNaN: includeNaN,
                } }) });
    })), on(npmiChangeAnnotationSort, (/**
     * @param {?} state
     * @param {?} __1
     * @return {?}
     */
    (state, { metric }) => {
        /** @type {?} */
        const newSort = {
            metric: metric,
            order: SortOrder.DOWN,
        };
        if (state.sort.metric === metric && state.sort.order === SortOrder.DOWN) {
            newSort.order = SortOrder.UP;
        }
        return Object.assign(Object.assign({}, state), { sort: newSort });
    })), on(npmiToggleParallelCoordinatesExpanded, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return Object.assign(Object.assign({}, state), { pcExpanded: !state.pcExpanded });
    })), on(npmiToggleAnnotationsExpanded, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return Object.assign(Object.assign({}, state), { annotationsExpanded: !state.annotationsExpanded });
    })), on(npmiToggleSidebarExpanded, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return Object.assign(Object.assign({}, state), { sidebarExpanded: !state.sidebarExpanded });
    })), on(npmiToggleShowCounts, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return Object.assign(Object.assign({}, state), { showCounts: !state.showCounts });
    })), on(npmiToggleShowHiddenAnnotations, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        return Object.assign(Object.assign({}, state), { showHiddenAnnotations: !state.showHiddenAnnotations });
    })), on(npmiChangeSidebarWidth, (/**
     * @param {?} state
     * @param {?} __1
     * @return {?}
     */
    (state, { sidebarWidth }) => {
        return Object.assign(Object.assign({}, state), { sidebarWidth });
    })));
    /**
     * @param {?} state
     * @param {?} action
     * @return {?}
     */
    function reducers$8(state, action) {
        return reducer$6(state, action);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/data_selection/metric_search/metric_search_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    function MetricSearchComponent_mat_icon_3_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "mat-icon", 7);
    } }
    function MetricSearchComponent_mat_option_6_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "mat-option", 8);
        ɵɵtext(1);
        ɵɵelementEnd();
    } if (rf & 2) {
        const completion_r4 = ctx.$implicit;
        ɵɵproperty("value", completion_r4);
        ɵɵadvance(1);
        ɵɵtextInterpolate(completion_r4);
    } }
    class MetricSearchComponent {
        constructor() {
            this.onRegexFilterValueChange = new EventEmitter();
            this.onAddFilter = new EventEmitter();
        }
        /**
         * @param {?} event
         * @param {?} matInput
         * @return {?}
         */
        onOptionSelected(event, matInput) {
            this.onAddFilter.emit(event.option.value);
            // matInput.value needs to be cleared manually, since the Angular Material Component may be modifying the input value outside of change detection
            matInput.value = '';
        }
    }
    MetricSearchComponent.decorators = [
        { type: Component, args: [{
                    selector: 'metric-search-component',
                    templateUrl: './metric_search_component.ng.html',
                    styleUrls: ['./metric_search_component.css'],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    MetricSearchComponent.propDecorators = {
        completions: [{ type: Input }],
        regexFilterValue: [{ type: Input }],
        isRegexFilterValid: [{ type: HostBinding, args: ['class.valid',] }, { type: Input }],
        onRegexFilterValueChange: [{ type: Output }],
        onAddFilter: [{ type: Output }]
    };
    /** @nocollapse */ MetricSearchComponent.ɵfac = function MetricSearchComponent_Factory(t) { return new (t || MetricSearchComponent)(); };
    /** @nocollapse */ MetricSearchComponent.ɵcmp = ɵɵdefineComponent({ type: MetricSearchComponent, selectors: [["metric-search-component"]], hostVars: 2, hostBindings: function MetricSearchComponent_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵclassProp("valid", ctx.isRegexFilterValid);
        } }, inputs: { completions: "completions", regexFilterValue: "regexFilterValue", isRegexFilterValid: "isRegexFilterValid" }, outputs: { onRegexFilterValueChange: "onRegexFilterValueChange", onAddFilter: "onAddFilter" }, decls: 7, vars: 4, consts: [["svgIcon", "search_24px"], ["matInput", "", "autocomplete", "off", "placeholder", "Add Metric Filter", 3, "value", "matAutocomplete", "input"], ["matInput", ""], ["svgIcon", "error_24px", "class", "error-icon", "matTooltip", "Invalid regex filter. The result may be stale.", 4, "ngIf"], ["autoActiveFirstOption", "", 3, "optionSelected"], ["filterMatches", "matAutocomplete"], [3, "value", 4, "ngFor", "ngForOf"], ["svgIcon", "error_24px", "matTooltip", "Invalid regex filter. The result may be stale.", 1, "error-icon"], [3, "value"]], template: function MetricSearchComponent_Template(rf, ctx) { if (rf & 1) {
            const _r5 = ɵɵgetCurrentView();
            ɵɵelement(0, "mat-icon", 0);
            ɵɵelementStart(1, "input", 1, 2);
            ɵɵlistener("input", function MetricSearchComponent_Template_input_input_1_listener($event) { return ctx.onRegexFilterValueChange.emit($event.target.value); });
            ɵɵelementEnd();
            ɵɵtemplate(3, MetricSearchComponent_mat_icon_3_Template, 1, 0, "mat-icon", 3);
            ɵɵelementStart(4, "mat-autocomplete", 4, 5);
            ɵɵlistener("optionSelected", function MetricSearchComponent_Template_mat_autocomplete_optionSelected_4_listener($event) { ɵɵrestoreView(_r5); const _r0 = ɵɵreference(2); return ctx.onOptionSelected($event, _r0); });
            ɵɵtemplate(6, MetricSearchComponent_mat_option_6_Template, 2, 2, "mat-option", 6);
            ɵɵelementEnd();
        } if (rf & 2) {
            const _r2 = ɵɵreference(5);
            ɵɵadvance(1);
            ɵɵproperty("value", ctx.regexFilterValue)("matAutocomplete", _r2);
            ɵɵadvance(2);
            ɵɵproperty("ngIf", !ctx.isRegexFilterValid);
            ɵɵadvance(3);
            ɵɵproperty("ngForOf", ctx.completions);
        } }, directives: [MatIcon, MatInput, MatAutocompleteTrigger, NgIf, MatAutocomplete, NgForOf, MatOption], styles: [".mat-badge-content[_ngcontent-%COMP%]{font-weight:600;font-size:12px;font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:9px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:24px}.mat-h1[_ngcontent-%COMP%], .mat-headline[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%]{font:400 24px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h2[_ngcontent-%COMP%], .mat-title[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h3[_ngcontent-%COMP%], .mat-subheading-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%]{font:400 16px/28px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h4[_ngcontent-%COMP%], .mat-subheading-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%]{font:400 15px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h5[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%]{font:400 calc(14px * 0.83)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-h6[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:400 calc(14px * 0.67)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-body-strong[_ngcontent-%COMP%], .mat-body-2[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   p[_ngcontent-%COMP%]{margin:0 0 12px}.mat-small[_ngcontent-%COMP%], .mat-caption[_ngcontent-%COMP%]{font:400 12px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-display-4[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-4[_ngcontent-%COMP%]{font:300 112px/112px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.05em;margin:0 0 56px}.mat-display-3[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-3[_ngcontent-%COMP%]{font:400 56px/56px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.02em;margin:0 0 64px}.mat-display-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-2[_ngcontent-%COMP%]{font:400 45px/48px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.005em;margin:0 0 64px}.mat-display-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-1[_ngcontent-%COMP%]{font:400 34px/40px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 64px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%], .mat-flat-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-button-toggle[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card-title[_ngcontent-%COMP%]{font-size:24px;font-weight:500}.mat-card-header[_ngcontent-%COMP%]   .mat-card-title[_ngcontent-%COMP%]{font-size:20px}.mat-card-subtitle[_ngcontent-%COMP%], .mat-card-content[_ngcontent-%COMP%]{font-size:14px}.mat-checkbox[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-checkbox-layout[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{line-height:24px}.mat-chip[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-chip[_ngcontent-%COMP%]   .mat-chip-trailing-icon.mat-icon[_ngcontent-%COMP%], .mat-chip[_ngcontent-%COMP%]   .mat-chip-remove.mat-icon[_ngcontent-%COMP%]{font-size:18px}.mat-table[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-header-cell[_ngcontent-%COMP%]{font-size:12px;font-weight:500}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{font-size:14px}.mat-calendar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-calendar-body[_ngcontent-%COMP%]{font-size:13px}.mat-calendar-body-label[_ngcontent-%COMP%], .mat-calendar-period-button[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-calendar-table-header[_ngcontent-%COMP%]   th[_ngcontent-%COMP%]{font-size:11px;font-weight:400}.mat-dialog-title[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-expansion-panel-header[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:15px;font-weight:400}.mat-expansion-panel-content[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field[_ngcontent-%COMP%]{font-size:inherit;font-weight:400;line-height:1.125;font-family:Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.34375em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{font-size:150%;line-height:1.125}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]{height:1.5em;width:1.5em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{height:1.125em;line-height:1.125}.mat-form-field-infix[_ngcontent-%COMP%]{padding:.5em 0;border-top:.84375em solid transparent}.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34375em) scale(0.75);width:133.3333333333%}.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34374em) scale(0.75);width:133.3333433333%}.mat-form-field-label-wrapper[_ngcontent-%COMP%]{top:-0.84375em;padding-top:.84375em}.mat-form-field-label[_ngcontent-%COMP%]{top:1.34375em}.mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.34375em}.mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{font-size:75%;margin-top:.6666666667em;top:calc(100% - 1.7916666667em)}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.4375em 0}.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.001px);-ms-transform:translateY(-1.28125em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00101px);-ms-transform:translateY(-1.28124em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00102px);-ms-transform:translateY(-1.28123em) scale(0.75);width:133.3333533333%}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.28125em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{margin-top:.5416666667em;top:calc(100% - 1.6666666667em)}@media print{.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28122em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28121em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.2812em) scale(0.75)}}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.25em 0 .75em 0}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.09375em;margin-top:-0.5em}.mat-form-field-appearance-fill.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59374em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:1em 0 1em 0}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.84375em;margin-top:-0.25em}.mat-form-field-appearance-outline.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59374em) scale(0.75);width:133.3333433333%}.mat-grid-tile-header[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]{font-size:14px}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2), .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}input.mat-input-element[_ngcontent-%COMP%]{margin-top:-0.0625em}.mat-menu-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:400}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px}.mat-radio-button[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select-trigger[_ngcontent-%COMP%]{height:1.125em}.mat-slide-toggle-content[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-slider-thumb-label-text[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-stepper-vertical[_ngcontent-%COMP%], .mat-stepper-horizontal[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-step-label[_ngcontent-%COMP%]{font-size:14px;font-weight:400}.mat-step-sub-label-error[_ngcontent-%COMP%]{font-weight:normal}.mat-step-label-error[_ngcontent-%COMP%]{font-size:14px}.mat-step-label-selected[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-tab-group[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-toolbar[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0}.mat-tooltip[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:10px;padding-top:6px;padding-bottom:6px}.mat-tooltip-handset[_ngcontent-%COMP%]{font-size:14px;padding-top:8px;padding-bottom:8px}.mat-list-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:16px}.mat-optgroup-label[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-simple-snackbar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px}.mat-simple-snackbar-action[_ngcontent-%COMP%]{line-height:1;font-family:inherit;font-size:inherit;font-weight:500}.mat-tree[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{font-weight:400;font-size:14px}.mat-ripple[_ngcontent-%COMP%]{overflow:hidden;position:relative}.mat-ripple[_ngcontent-%COMP%]:not(:empty){transform:translateZ(0)}.mat-ripple.mat-ripple-unbounded[_ngcontent-%COMP%]{overflow:visible}.mat-ripple-element[_ngcontent-%COMP%]{position:absolute;border-radius:50%;pointer-events:none;transition:opacity,transform 0ms cubic-bezier(0, 0, 0.2, 1);transform:scale(0)}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{display:none}.cdk-visually-hidden[_ngcontent-%COMP%]{border:0;clip:rect(0 0 0 0);height:1px;margin:-1px;overflow:hidden;padding:0;position:absolute;width:1px;outline:0;-webkit-appearance:none;-moz-appearance:none}.cdk-overlay-container[_ngcontent-%COMP%], .cdk-global-overlay-wrapper[_ngcontent-%COMP%]{pointer-events:none;top:0;left:0;height:100%;width:100%}.cdk-overlay-container[_ngcontent-%COMP%]{position:fixed;z-index:1000}.cdk-overlay-container[_ngcontent-%COMP%]:empty{display:none}.cdk-global-overlay-wrapper[_ngcontent-%COMP%]{display:flex;position:absolute;z-index:1000}.cdk-overlay-pane[_ngcontent-%COMP%]{position:absolute;pointer-events:auto;box-sizing:border-box;z-index:1000;display:flex;max-width:100%;max-height:100%}.cdk-overlay-backdrop[_ngcontent-%COMP%]{position:absolute;top:0;bottom:0;left:0;right:0;z-index:1000;pointer-events:auto;-webkit-tap-highlight-color:transparent;transition:opacity 400ms cubic-bezier(0.25, 0.8, 0.25, 1);opacity:0}.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:1}@media screen and (-ms-high-contrast: active){.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:.6}}.cdk-overlay-dark-backdrop[_ngcontent-%COMP%]{background:rgba(0,0,0,.32)}.cdk-overlay-transparent-backdrop[_ngcontent-%COMP%], .cdk-overlay-transparent-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:0}.cdk-overlay-connected-position-bounding-box[_ngcontent-%COMP%]{position:absolute;z-index:1000;display:flex;flex-direction:column;min-width:1px;min-height:1px}.cdk-global-scrollblock[_ngcontent-%COMP%]{position:fixed;width:100%;overflow-y:scroll}@keyframes cdk-text-field-autofill-start{}@keyframes cdk-text-field-autofill-end{}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:-webkit-autofill{animation:cdk-text-field-autofill-start 0s 1ms}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:not(:-webkit-autofill){animation:cdk-text-field-autofill-end 0s 1ms}textarea.cdk-textarea-autosize[_ngcontent-%COMP%]{resize:none}textarea.cdk-textarea-autosize-measuring[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:auto !important;overflow:hidden !important}textarea.cdk-textarea-autosize-measuring-firefox[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:0 !important}.mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.1)}.mat-option[_ngcontent-%COMP%]{color:#212121}.mat-option[_ngcontent-%COMP%]:hover:not(.mat-option-disabled), .mat-option[_ngcontent-%COMP%]:focus:not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple):not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-active[_ngcontent-%COMP%]{background:rgba(0,0,0,.04);color:#212121}.mat-option.mat-option-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-primary[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f57c00}.mat-accent[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f44336}.mat-optgroup-label[_ngcontent-%COMP%]{color:#616161}.mat-optgroup-disabled[_ngcontent-%COMP%]   .mat-optgroup-label[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-pseudo-checkbox[_ngcontent-%COMP%]{color:#616161}.mat-pseudo-checkbox[_ngcontent-%COMP%]::after{color:#fafafa}.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{color:#b0b0b0}.mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f57c00}.mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f44336}.mat-pseudo-checkbox-checked.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{background:#b0b0b0}.mat-elevation-z0[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z1[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z2[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z3[_ngcontent-%COMP%]{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z4[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z5[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 5px 8px 0px rgba(0, 0, 0, 0.14),0px 1px 14px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z6[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z7[_ngcontent-%COMP%]{box-shadow:0px 4px 5px -2px rgba(0, 0, 0, 0.2),0px 7px 10px 1px rgba(0, 0, 0, 0.14),0px 2px 16px 1px rgba(0, 0, 0, 0.12)}.mat-elevation-z8[_ngcontent-%COMP%]{box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z9[_ngcontent-%COMP%]{box-shadow:0px 5px 6px -3px rgba(0, 0, 0, 0.2),0px 9px 12px 1px rgba(0, 0, 0, 0.14),0px 3px 16px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z10[_ngcontent-%COMP%]{box-shadow:0px 6px 6px -3px rgba(0, 0, 0, 0.2),0px 10px 14px 1px rgba(0, 0, 0, 0.14),0px 4px 18px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z11[_ngcontent-%COMP%]{box-shadow:0px 6px 7px -4px rgba(0, 0, 0, 0.2),0px 11px 15px 1px rgba(0, 0, 0, 0.14),0px 4px 20px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z12[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z13[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 13px 19px 2px rgba(0, 0, 0, 0.14),0px 5px 24px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z14[_ngcontent-%COMP%]{box-shadow:0px 7px 9px -4px rgba(0, 0, 0, 0.2),0px 14px 21px 2px rgba(0, 0, 0, 0.14),0px 5px 26px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z15[_ngcontent-%COMP%]{box-shadow:0px 8px 9px -5px rgba(0, 0, 0, 0.2),0px 15px 22px 2px rgba(0, 0, 0, 0.14),0px 6px 28px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z16[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z17[_ngcontent-%COMP%]{box-shadow:0px 8px 11px -5px rgba(0, 0, 0, 0.2),0px 17px 26px 2px rgba(0, 0, 0, 0.14),0px 6px 32px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z18[_ngcontent-%COMP%]{box-shadow:0px 9px 11px -5px rgba(0, 0, 0, 0.2),0px 18px 28px 2px rgba(0, 0, 0, 0.14),0px 7px 34px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z19[_ngcontent-%COMP%]{box-shadow:0px 9px 12px -6px rgba(0, 0, 0, 0.2),0px 19px 29px 2px rgba(0, 0, 0, 0.14),0px 7px 36px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z20[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 20px 31px 3px rgba(0, 0, 0, 0.14),0px 8px 38px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z21[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 21px 33px 3px rgba(0, 0, 0, 0.14),0px 8px 40px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z22[_ngcontent-%COMP%]{box-shadow:0px 10px 14px -6px rgba(0, 0, 0, 0.2),0px 22px 35px 3px rgba(0, 0, 0, 0.14),0px 8px 42px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z23[_ngcontent-%COMP%]{box-shadow:0px 11px 14px -7px rgba(0, 0, 0, 0.2),0px 23px 36px 3px rgba(0, 0, 0, 0.14),0px 9px 44px 8px rgba(0, 0, 0, 0.12)}.mat-elevation-z24[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12)}.mat-app-background[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-theme-loaded-marker[_ngcontent-%COMP%]{display:none}.mat-autocomplete-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-autocomplete-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover){background:#fff}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover):not(.mat-option-disabled){color:#212121}.mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f57c00}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{outline:solid 1px;border-radius:0}.mat-badge-accent[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-badge-warn[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f44336}.mat-badge[_ngcontent-%COMP%]{position:relative}.mat-badge-hidden[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{display:none}.mat-badge-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#b9b9b9;color:#757575}.mat-badge-content[_ngcontent-%COMP%]{position:absolute;text-align:center;display:inline-block;border-radius:50%;transition:transform 200ms ease-in-out;transform:scale(0.6);overflow:hidden;white-space:nowrap;text-overflow:ellipsis;pointer-events:none}.ng-animate-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%], .mat-badge-content._mat-animation-noopable[_ngcontent-%COMP%]{transition:none}.mat-badge-content.mat-badge-active[_ngcontent-%COMP%]{transform:none}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:16px;height:16px;line-height:16px}.mat-badge-small.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-8px}.mat-badge-small.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-8px}.mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-16px}.mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-16px}.mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-8px}.mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-8px}.mat-badge-medium[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:22px;height:22px;line-height:22px}.mat-badge-medium.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-11px}.mat-badge-medium.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-11px}.mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-22px}.mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-22px}.mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-11px}.mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-11px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:28px;height:28px;line-height:28px}.mat-badge-large.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-14px}.mat-badge-large.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-14px}.mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-28px}.mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-28px}.mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-14px}.mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-14px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]{color:inherit;background:transparent}.mat-button.mat-primary[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-button[disabled][disabled][_ngcontent-%COMP%], .mat-icon-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-icon-button[disabled][disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-stroked-button[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f44336}.mat-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:transparent}.mat-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{opacity:.1;background-color:currentColor}.mat-button-focus-overlay[_ngcontent-%COMP%]{background:#000}.mat-stroked-button[_ngcontent-%COMP%]:not([disabled]){border-color:rgba(0,0,0,.12)}.mat-flat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{color:#212121;background-color:#fff}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-flat-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-stroked-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-flat-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-raised-button[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone[_ngcontent-%COMP%], .mat-button-toggle-group[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{box-shadow:none}.mat-button-toggle[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-button-toggle[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121;background:#fff}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:#000}.mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12)}[dir=rtl][_ngcontent-%COMP%]   .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-group-appearance-standard.mat-button-toggle-vertical[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:none;border-top:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#616161}.mat-button-toggle-checked.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121}.mat-button-toggle-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.26);background-color:#eee}.mat-button-toggle-disabled.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{background:#fff}.mat-button-toggle-disabled.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#bdbdbd}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{border:solid 1px rgba(0,0,0,.12)}.mat-card[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-card[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-card.mat-card-flat[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-card-subtitle[_ngcontent-%COMP%]{color:#616161}.mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#616161}.mat-checkbox-checkmark[_ngcontent-%COMP%]{fill:#fafafa}.mat-checkbox-checkmark-path[_ngcontent-%COMP%]{stroke:#fafafa !important}.mat-checkbox-mixedmark[_ngcontent-%COMP%]{background-color:#fafafa}.mat-checkbox-indeterminate.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f57c00}.mat-checkbox-indeterminate.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#ff9800}.mat-checkbox-indeterminate.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f44336}.mat-checkbox-disabled.mat-checkbox-checked[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-disabled.mat-checkbox-indeterminate[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]:not(.mat-checkbox-checked)   .mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{color:#616161}.mat-checkbox[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%]{background:#f57c00}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%]{background:#ff9800}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%]{background:#f44336}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#212121}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#212121;opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled):active{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled)   .mat-chip-remove[_ngcontent-%COMP%]:hover{opacity:.54}.mat-chip.mat-standard-chip.mat-chip-disabled[_ngcontent-%COMP%]{opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]::after{background:#000}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-table[_ngcontent-%COMP%]{background:#fff}.mat-table[_ngcontent-%COMP%]   thead[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tbody[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tfoot[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], [mat-header-row][_ngcontent-%COMP%], [mat-row][_ngcontent-%COMP%], [mat-footer-row][_ngcontent-%COMP%], .mat-table-sticky[_ngcontent-%COMP%]{background:inherit}mat-row[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], th.mat-header-cell[_ngcontent-%COMP%], td.mat-cell[_ngcontent-%COMP%], td.mat-footer-cell[_ngcontent-%COMP%]{border-bottom-color:rgba(0,0,0,.12)}.mat-header-cell[_ngcontent-%COMP%]{color:#616161}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{color:#212121}.mat-calendar-arrow[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.54)}.mat-datepicker-toggle[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-next-button[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-previous-button[_ngcontent-%COMP%]{color:rgba(0,0,0,.54)}.mat-calendar-table-header[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-calendar-table-header-divider[_ngcontent-%COMP%]::after{background:rgba(0,0,0,.12)}.mat-calendar-body-label[_ngcontent-%COMP%]{color:#616161}.mat-calendar-body-cell-content[_ngcontent-%COMP%]{color:#212121;border-color:transparent}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){color:#757575}.mat-calendar-body-cell[_ngcontent-%COMP%]:not(.mat-calendar-body-disabled):hover > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-keyboard-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-program-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){background-color:rgba(0,0,0,.04)}.mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.38)}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.18)}.mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.4)}.mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12);background-color:#fff;color:#212121}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.4)}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.4)}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content-touch[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-datepicker-toggle-active[_ngcontent-%COMP%]{color:#f57c00}.mat-datepicker-toggle-active.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-datepicker-toggle-active.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-dialog-container[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-divider[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-divider-vertical[_ngcontent-%COMP%]{border-right-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-expansion-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-action-row[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-keyboard-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-program-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded)   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover:not([aria-disabled=true]){background:rgba(0,0,0,.04)}@media(hover: none){.mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded):not([aria-disabled=true])   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover{background:#fff}}.mat-expansion-panel-header-title[_ngcontent-%COMP%]{color:#212121}.mat-expansion-panel-header-description[_ngcontent-%COMP%], .mat-expansion-indicator[_ngcontent-%COMP%]::after{color:#616161}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-title[_ngcontent-%COMP%], .mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-description[_ngcontent-%COMP%]{color:inherit}.mat-form-field-label[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-hint[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-focused[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-ripple[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid)   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f57c00}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-accent   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#ff9800}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-warn   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#f44336}.mat-error[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-hint[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-legacy.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-standard[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-standard.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.04)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.02)}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:transparent}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.12)}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:rgba(0,0,0,.87)}.mat-form-field-appearance-outline.mat-focused[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field-appearance-outline.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-appearance-outline.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-invalid.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.06)}.mat-icon.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-icon.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-icon.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-type-mat-native-select[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#616161}.mat-input-element[_ngcontent-%COMP%]:disabled, .mat-form-field-type-mat-native-select.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#757575}.mat-input-element[_ngcontent-%COMP%]{caret-color:#f57c00}.mat-input-element[_ngcontent-%COMP%]::placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-moz-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-webkit-input-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]:-ms-input-placeholder{color:rgba(97,97,97,.42)}.mat-accent[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%], .mat-form-field-invalid[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#f44336}.mat-form-field-type-mat-native-select.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{color:#616161}.mat-list-item-disabled[_ngcontent-%COMP%]{background-color:#eee}.mat-list-option[_ngcontent-%COMP%]:hover, .mat-list-option[_ngcontent-%COMP%]:focus, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.04)}.mat-list-single-selected-option[_ngcontent-%COMP%], .mat-list-single-selected-option[_ngcontent-%COMP%]:hover, .mat-list-single-selected-option[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.12)}.mat-menu-panel[_ngcontent-%COMP%]{background:#fff}.mat-menu-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-menu-item[_ngcontent-%COMP%]{background:transparent;color:#212121}.mat-menu-item[disabled][_ngcontent-%COMP%], .mat-menu-item[disabled][_ngcontent-%COMP%]::after{color:rgba(0,0,0,.38)}.mat-menu-item[_ngcontent-%COMP%]   .mat-icon-no-color[_ngcontent-%COMP%], .mat-menu-item-submenu-trigger[_ngcontent-%COMP%]::after{color:rgba(0,0,0,.54)}.mat-menu-item[_ngcontent-%COMP%]:hover:not([disabled]), .mat-menu-item.cdk-program-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item.cdk-keyboard-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item-highlighted[_ngcontent-%COMP%]:not([disabled]){background:rgba(0,0,0,.04)}.mat-paginator[_ngcontent-%COMP%]{background:#fff}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{color:#616161}.mat-paginator-decrement[_ngcontent-%COMP%], .mat-paginator-increment[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54);border-right:2px solid rgba(0,0,0,.54)}.mat-paginator-first[_ngcontent-%COMP%], .mat-paginator-last[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54)}.mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-decrement[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-increment[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-first[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-last[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f57c00}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#ff9800}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f44336}.mat-progress-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f57c00}.mat-progress-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#ff9800}.mat-progress-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f44336}.mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#616161}.mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f57c00}.mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#ff9800}.mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#ff9800}.mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f44336}.mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f44336}.mat-radio-button.mat-radio-disabled.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-label-content[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-radio-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-select-value[_ngcontent-%COMP%]{color:#212121}.mat-select-placeholder[_ngcontent-%COMP%]{color:rgba(97,97,97,.42)}.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%]{color:#757575}.mat-select-arrow[_ngcontent-%COMP%]{color:#616161}.mat-select-panel[_ngcontent-%COMP%]{background:#fff}.mat-select-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-select-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple){background:rgba(0,0,0,.12)}.mat-form-field.mat-focused.mat-primary[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-invalid[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#757575}.mat-drawer-container[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-drawer[_ngcontent-%COMP%]{background-color:#fff;color:#212121}.mat-drawer.mat-drawer-push[_ngcontent-%COMP%]{background-color:#fff}.mat-drawer[_ngcontent-%COMP%]:not(.mat-drawer-side){box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-drawer-side[_ngcontent-%COMP%]{border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-backdrop.mat-drawer-shown[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.6)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.54)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.54)}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.54)}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle[_ngcontent-%COMP%]:not(.mat-checked)   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-slide-toggle-thumb[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12);background-color:#fafafa}.mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-primary[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f57c00}.mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-accent[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#ff9800}.mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-warn[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f44336}.mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.2)}.mat-slider[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%], .cdk-focused[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-disabled[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]   .mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing)   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26);background-color:transparent}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26)}.mat-slider-has-ticks[_ngcontent-%COMP%]   .mat-slider-wrapper[_ngcontent-%COMP%]::after{border-color:rgba(0,0,0,.7)}.mat-slider-horizontal[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to right, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent);background-image:-moz-repeating-linear-gradient(0.0001deg, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-slider-vertical[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to bottom, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-step-header.cdk-keyboard-focused[_ngcontent-%COMP%], .mat-step-header.cdk-program-focused[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]:hover{background-color:rgba(0,0,0,.04)}@media(hover: none){.mat-step-header[_ngcontent-%COMP%]:hover{background:none}}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-optional[_ngcontent-%COMP%]{color:#616161}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon[_ngcontent-%COMP%]{background-color:#616161;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-selected[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-done[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-edit[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-error[_ngcontent-%COMP%]{background-color:transparent;color:#f44336}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-active[_ngcontent-%COMP%]{color:#212121}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-error[_ngcontent-%COMP%]{color:#f44336}.mat-stepper-horizontal[_ngcontent-%COMP%], .mat-stepper-vertical[_ngcontent-%COMP%]{background-color:#fff}.mat-stepper-vertical-line[_ngcontent-%COMP%]::before{border-left-color:rgba(0,0,0,.12)}.mat-horizontal-stepper-header[_ngcontent-%COMP%]::before, .mat-horizontal-stepper-header[_ngcontent-%COMP%]::after, .mat-stepper-horizontal-line[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-sort-header-arrow[_ngcontent-%COMP%]{color:#616161}.mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-header[_ngcontent-%COMP%]{border-bottom:1px solid rgba(0,0,0,.12)}.mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%]{border-top:1px solid rgba(0,0,0,.12);border-bottom:none}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{color:#212121}.mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:#757575}.mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#212121}.mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#757575}.mat-tab-group[class*=mat-background-][_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar[class*=mat-background-][_ngcontent-%COMP%]{border-bottom:none;border-top:none}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-toolbar[_ngcontent-%COMP%]{background:#f5f5f5;color:#212121}.mat-toolbar.mat-primary[_ngcontent-%COMP%]{background:#f57c00;color:#fff}.mat-toolbar.mat-accent[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-toolbar.mat-warn[_ngcontent-%COMP%]{background:#f44336;color:#fff}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:currentColor}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:inherit}.mat-toolbar[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:currentColor}.mat-tooltip[_ngcontent-%COMP%]{background:rgba(97,97,97,.9)}.mat-tree[_ngcontent-%COMP%]{background:#fff}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{color:#212121}.mat-snack-bar-container[_ngcontent-%COMP%]{color:rgba(255,255,255,.7);background:#323232;box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-simple-snackbar-action[_ngcontent-%COMP%]{color:#ff9800}mat-icon[_ngcontent-%COMP%]{flex:none;margin-right:5px}[_nghost-%COMP%]{display:flex;padding:0 10px;position:relative;font-size:.9em}[_nghost-%COMP%]:not(.valid){color:#c62828}[_nghost-%COMP%]:not(.valid)   input[_ngcontent-%COMP%]{caret-color:currentColor}"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/data_selection/metric_search/metric_search_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class MetricSearchContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.metricsRegex$ = this.store.select(getMetricsRegex);
            this.activeRuns$ = this.store
                .pipe(select(getCurrentRouteRunSelection))
                .pipe(map((/**
             * @param {?} runSelection
             * @return {?}
             */
            (runSelection) => {
                if (!runSelection)
                    return [];
                return Array.from(runSelection.entries())
                    .filter((/**
                 * @param {?} run
                 * @return {?}
                 */
                (run) => run[1]))
                    .map((/**
                 * @param {?} run
                 * @return {?}
                 */
                (run) => run[0]));
            })));
            this.metricsForActiveRuns$ = combineLatest(this.activeRuns$, this.store.select(getRunToMetrics)).pipe(map((/**
             * @param {?} __0
             * @return {?}
             */
            ([activeRuns, runToMetrics]) => {
                /** @type {?} */
                const metrics = new Set();
                for (const run of activeRuns) {
                    if (runToMetrics[run]) {
                        for (const metric of runToMetrics[run]) {
                            metrics.add(metric);
                        }
                    }
                }
                return [...metrics];
            })));
            this.isMetricsFilterValid$ = this.metricsRegex$.pipe(map((/**
             * @param {?} filterString
             * @return {?}
             */
            (filterString) => {
                try {
                    return Boolean(new RegExp(filterString));
                }
                catch (err) {
                    return false;
                }
            })));
            this.metricFilterKeys$ = this.store.pipe(select(getMetricFilters)).pipe(map((/**
             * @param {?} metricFilters
             * @return {?}
             */
            (metricFilters) => {
                return Object.keys(metricFilters);
            })));
            this.completions$ = combineLatest(this.metricsForActiveRuns$, this.metricsRegex$, this.metricFilterKeys$).pipe(map((/**
             * @param {?} __0
             * @return {?}
             */
            ([metrics, metricsRegex, metricsWithFilters]) => {
                /** @type {?} */
                const filteredMetrics = metrics.filter((/**
                 * @param {?} metric
                 * @return {?}
                 */
                (metric) => !metricsWithFilters.includes(metric)));
                try {
                    /** @type {?} */
                    const filterRegex = new RegExp(metricsRegex, 'i');
                    return filteredMetrics
                        .filter((/**
                     * @param {?} metric
                     * @return {?}
                     */
                    (metric) => filterRegex.test(metric)))
                        .sort();
                }
                catch (err) {
                    return [];
                }
            })));
        }
        /**
         * @param {?} filter
         * @return {?}
         */
        onFilterChange(filter) {
            this.store.dispatch(npmiMetricsRegexChanged({ regex: filter }));
        }
        /**
         * @param {?} metric
         * @return {?}
         */
        onAddFilter(metric) {
            this.store.dispatch(npmiAddMetricFilter({ metric }));
            this.store.dispatch(npmiMetricsRegexChanged({ regex: '' }));
        }
    }
    MetricSearchContainer.decorators = [
        { type: Component, args: [{
                    selector: 'npmi-metric-search',
                    template: `
    <metric-search-component
      [regexFilterValue]="metricsRegex$ | async"
      [completions]="completions$ | async"
      [isRegexFilterValid]="isMetricsFilterValid$ | async"
      (onRegexFilterValueChange)="onFilterChange($event)"
      (onAddFilter)="onAddFilter($event)"
    ></metric-search-component>
  `,
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    MetricSearchContainer.ctorParameters = () => [
        { type: Store }
    ];
    /** @nocollapse */ MetricSearchContainer.ɵfac = function MetricSearchContainer_Factory(t) { return new (t || MetricSearchContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ MetricSearchContainer.ɵcmp = ɵɵdefineComponent({ type: MetricSearchContainer, selectors: [["npmi-metric-search"]], decls: 4, vars: 9, consts: [[3, "regexFilterValue", "completions", "isRegexFilterValid", "onRegexFilterValueChange", "onAddFilter"]], template: function MetricSearchContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "metric-search-component", 0);
            ɵɵlistener("onRegexFilterValueChange", function MetricSearchContainer_Template_metric_search_component_onRegexFilterValueChange_0_listener($event) { return ctx.onFilterChange($event); })("onAddFilter", function MetricSearchContainer_Template_metric_search_component_onAddFilter_0_listener($event) { return ctx.onAddFilter($event); });
            ɵɵpipe(1, "async");
            ɵɵpipe(2, "async");
            ɵɵpipe(3, "async");
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵproperty("regexFilterValue", ɵɵpipeBind1(1, 3, ctx.metricsRegex$))("completions", ɵɵpipeBind1(2, 5, ctx.completions$))("isRegexFilterValid", ɵɵpipeBind1(3, 7, ctx.isMetricsFilterValid$));
        } }, directives: [MetricSearchComponent], pipes: [AsyncPipe], encapsulation: 2, changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/util/csv_result.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @param {?} flaggedData
     * @param {?} run
     * @param {?} metrics
     * @return {?}
     */
    function convertToCSVResult(flaggedData, run, metrics) {
        /** @type {?} */
        const result = [[run, ...metrics]];
        if (!metrics.length || !flaggedData.length) {
            /** @type {?} */
            const csvContent = 'data:text/csv;charset=utf-8,' +
                result.map((/**
                 * @param {?} e
                 * @return {?}
                 */
                (e) => e.join(','))).join('\n');
            return csvContent;
        }
        /** @type {?} */
        const strippedMetrics = metrics.map((/**
         * @param {?} metric
         * @return {?}
         */
        (metric) => stripMetricString(metric)));
        for (const [annotation, valueDataList] of flaggedData) {
            /** @type {?} */
            const runValues = valueDataList.filter((/**
             * @param {?} values
             * @return {?}
             */
            (values) => values.run === run));
            if (runValues.length) {
                /** @type {?} */
                const elementResult = [annotation];
                for (const metric of strippedMetrics) {
                    /** @type {?} */
                    const metricValue = runValues.find((/**
                     * @param {?} value
                     * @return {?}
                     */
                    (value) => value.metric === metric));
                    if (metricValue === undefined) {
                        elementResult.push('null');
                    }
                    else {
                        elementResult.push(`${metricValue.nPMIValue}`);
                    }
                }
                result.push(elementResult);
            }
        }
        /** @type {?} */
        const csvContent = 'data:text/csv;charset=utf-8,' + result.map((/**
         * @param {?} e
         * @return {?}
         */
        (e) => e.join(','))).join('\n');
        return csvContent;
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/data_selection/results_download/results_download_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    const _c0$t = function (a0) { return { "active-button": a0 }; };
    class ResultsDownloadComponent {
        /**
         * @return {?}
         */
        downloadResults() {
            for (const run of this.runs) {
                /** @type {?} */
                const csvData = convertToCSVResult(this.flaggedData, run, this.metrics);
                /** @type {?} */
                const element = document.createElement('a');
                element.setAttribute('href', csvData);
                element.setAttribute('download', `report_${run}.csv`);
                element.click();
            }
        }
    }
    ResultsDownloadComponent.decorators = [
        { type: Component, args: [{
                    selector: 'results-download-component',
                    templateUrl: './results_download_component.ng.html',
                    styleUrls: ['./results_download_component.css'],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    ResultsDownloadComponent.propDecorators = {
        numFlaggedAnnotations: [{ type: Input }],
        runs: [{ type: Input }],
        flaggedData: [{ type: Input }],
        metrics: [{ type: Input }]
    };
    /** @nocollapse */ ResultsDownloadComponent.ɵfac = function ResultsDownloadComponent_Factory(t) { return new (t || ResultsDownloadComponent)(); };
    /** @nocollapse */ ResultsDownloadComponent.ɵcmp = ɵɵdefineComponent({ type: ResultsDownloadComponent, selectors: [["results-download-component"]], inputs: { numFlaggedAnnotations: "numFlaggedAnnotations", runs: "runs", flaggedData: "flaggedData", metrics: "metrics" }, decls: 4, vars: 5, consts: [["mat-stroked-button", "", "title", "Export CSV reports of all flagged annotations. Will generate one CSV per active run.", 3, "disabled", "ngClass", "click"], [1, "button-contents"], ["svgIcon", "get_app_24px"]], template: function ResultsDownloadComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "button", 0);
            ɵɵlistener("click", function ResultsDownloadComponent_Template_button_click_0_listener() { return ctx.downloadResults(); });
            ɵɵelementStart(1, "span", 1);
            ɵɵelement(2, "mat-icon", 2);
            ɵɵtext(3);
            ɵɵelementEnd();
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵproperty("disabled", ctx.numFlaggedAnnotations === 0)("ngClass", ɵɵpureFunction1(3, _c0$t, ctx.numFlaggedAnnotations > 0));
            ɵɵadvance(3);
            ɵɵtextInterpolate1(" Flagged Rows (", ctx.numFlaggedAnnotations, ") ");
        } }, directives: [MatButton, NgClass, MatIcon], styles: [".mat-badge-content[_ngcontent-%COMP%]{font-weight:600;font-size:12px;font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:9px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:24px}.mat-h1[_ngcontent-%COMP%], .mat-headline[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%]{font:400 24px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h2[_ngcontent-%COMP%], .mat-title[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h3[_ngcontent-%COMP%], .mat-subheading-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%]{font:400 16px/28px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h4[_ngcontent-%COMP%], .mat-subheading-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%]{font:400 15px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h5[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%]{font:400 calc(14px * 0.83)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-h6[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:400 calc(14px * 0.67)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-body-strong[_ngcontent-%COMP%], .mat-body-2[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   p[_ngcontent-%COMP%]{margin:0 0 12px}.mat-small[_ngcontent-%COMP%], .mat-caption[_ngcontent-%COMP%]{font:400 12px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-display-4[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-4[_ngcontent-%COMP%]{font:300 112px/112px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.05em;margin:0 0 56px}.mat-display-3[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-3[_ngcontent-%COMP%]{font:400 56px/56px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.02em;margin:0 0 64px}.mat-display-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-2[_ngcontent-%COMP%]{font:400 45px/48px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.005em;margin:0 0 64px}.mat-display-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-1[_ngcontent-%COMP%]{font:400 34px/40px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 64px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%], .mat-flat-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-button-toggle[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card-title[_ngcontent-%COMP%]{font-size:24px;font-weight:500}.mat-card-header[_ngcontent-%COMP%]   .mat-card-title[_ngcontent-%COMP%]{font-size:20px}.mat-card-subtitle[_ngcontent-%COMP%], .mat-card-content[_ngcontent-%COMP%]{font-size:14px}.mat-checkbox[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-checkbox-layout[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{line-height:24px}.mat-chip[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-chip[_ngcontent-%COMP%]   .mat-chip-trailing-icon.mat-icon[_ngcontent-%COMP%], .mat-chip[_ngcontent-%COMP%]   .mat-chip-remove.mat-icon[_ngcontent-%COMP%]{font-size:18px}.mat-table[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-header-cell[_ngcontent-%COMP%]{font-size:12px;font-weight:500}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{font-size:14px}.mat-calendar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-calendar-body[_ngcontent-%COMP%]{font-size:13px}.mat-calendar-body-label[_ngcontent-%COMP%], .mat-calendar-period-button[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-calendar-table-header[_ngcontent-%COMP%]   th[_ngcontent-%COMP%]{font-size:11px;font-weight:400}.mat-dialog-title[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-expansion-panel-header[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:15px;font-weight:400}.mat-expansion-panel-content[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field[_ngcontent-%COMP%]{font-size:inherit;font-weight:400;line-height:1.125;font-family:Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.34375em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{font-size:150%;line-height:1.125}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]{height:1.5em;width:1.5em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{height:1.125em;line-height:1.125}.mat-form-field-infix[_ngcontent-%COMP%]{padding:.5em 0;border-top:.84375em solid transparent}.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34375em) scale(0.75);width:133.3333333333%}.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34374em) scale(0.75);width:133.3333433333%}.mat-form-field-label-wrapper[_ngcontent-%COMP%]{top:-0.84375em;padding-top:.84375em}.mat-form-field-label[_ngcontent-%COMP%]{top:1.34375em}.mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.34375em}.mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{font-size:75%;margin-top:.6666666667em;top:calc(100% - 1.7916666667em)}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.4375em 0}.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.001px);-ms-transform:translateY(-1.28125em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00101px);-ms-transform:translateY(-1.28124em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00102px);-ms-transform:translateY(-1.28123em) scale(0.75);width:133.3333533333%}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.28125em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{margin-top:.5416666667em;top:calc(100% - 1.6666666667em)}@media print{.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28122em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28121em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.2812em) scale(0.75)}}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.25em 0 .75em 0}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.09375em;margin-top:-0.5em}.mat-form-field-appearance-fill.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59374em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:1em 0 1em 0}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.84375em;margin-top:-0.25em}.mat-form-field-appearance-outline.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59374em) scale(0.75);width:133.3333433333%}.mat-grid-tile-header[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]{font-size:14px}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2), .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}input.mat-input-element[_ngcontent-%COMP%]{margin-top:-0.0625em}.mat-menu-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:400}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px}.mat-radio-button[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select-trigger[_ngcontent-%COMP%]{height:1.125em}.mat-slide-toggle-content[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-slider-thumb-label-text[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-stepper-vertical[_ngcontent-%COMP%], .mat-stepper-horizontal[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-step-label[_ngcontent-%COMP%]{font-size:14px;font-weight:400}.mat-step-sub-label-error[_ngcontent-%COMP%]{font-weight:normal}.mat-step-label-error[_ngcontent-%COMP%]{font-size:14px}.mat-step-label-selected[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-tab-group[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-toolbar[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0}.mat-tooltip[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:10px;padding-top:6px;padding-bottom:6px}.mat-tooltip-handset[_ngcontent-%COMP%]{font-size:14px;padding-top:8px;padding-bottom:8px}.mat-list-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:16px}.mat-optgroup-label[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-simple-snackbar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px}.mat-simple-snackbar-action[_ngcontent-%COMP%]{line-height:1;font-family:inherit;font-size:inherit;font-weight:500}.mat-tree[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{font-weight:400;font-size:14px}.mat-ripple[_ngcontent-%COMP%]{overflow:hidden;position:relative}.mat-ripple[_ngcontent-%COMP%]:not(:empty){transform:translateZ(0)}.mat-ripple.mat-ripple-unbounded[_ngcontent-%COMP%]{overflow:visible}.mat-ripple-element[_ngcontent-%COMP%]{position:absolute;border-radius:50%;pointer-events:none;transition:opacity,transform 0ms cubic-bezier(0, 0, 0.2, 1);transform:scale(0)}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{display:none}.cdk-visually-hidden[_ngcontent-%COMP%]{border:0;clip:rect(0 0 0 0);height:1px;margin:-1px;overflow:hidden;padding:0;position:absolute;width:1px;outline:0;-webkit-appearance:none;-moz-appearance:none}.cdk-overlay-container[_ngcontent-%COMP%], .cdk-global-overlay-wrapper[_ngcontent-%COMP%]{pointer-events:none;top:0;left:0;height:100%;width:100%}.cdk-overlay-container[_ngcontent-%COMP%]{position:fixed;z-index:1000}.cdk-overlay-container[_ngcontent-%COMP%]:empty{display:none}.cdk-global-overlay-wrapper[_ngcontent-%COMP%]{display:flex;position:absolute;z-index:1000}.cdk-overlay-pane[_ngcontent-%COMP%]{position:absolute;pointer-events:auto;box-sizing:border-box;z-index:1000;display:flex;max-width:100%;max-height:100%}.cdk-overlay-backdrop[_ngcontent-%COMP%]{position:absolute;top:0;bottom:0;left:0;right:0;z-index:1000;pointer-events:auto;-webkit-tap-highlight-color:transparent;transition:opacity 400ms cubic-bezier(0.25, 0.8, 0.25, 1);opacity:0}.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:1}@media screen and (-ms-high-contrast: active){.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:.6}}.cdk-overlay-dark-backdrop[_ngcontent-%COMP%]{background:rgba(0,0,0,.32)}.cdk-overlay-transparent-backdrop[_ngcontent-%COMP%], .cdk-overlay-transparent-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:0}.cdk-overlay-connected-position-bounding-box[_ngcontent-%COMP%]{position:absolute;z-index:1000;display:flex;flex-direction:column;min-width:1px;min-height:1px}.cdk-global-scrollblock[_ngcontent-%COMP%]{position:fixed;width:100%;overflow-y:scroll}@keyframes cdk-text-field-autofill-start{}@keyframes cdk-text-field-autofill-end{}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:-webkit-autofill{animation:cdk-text-field-autofill-start 0s 1ms}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:not(:-webkit-autofill){animation:cdk-text-field-autofill-end 0s 1ms}textarea.cdk-textarea-autosize[_ngcontent-%COMP%]{resize:none}textarea.cdk-textarea-autosize-measuring[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:auto !important;overflow:hidden !important}textarea.cdk-textarea-autosize-measuring-firefox[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:0 !important}.mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.1)}.mat-option[_ngcontent-%COMP%]{color:#212121}.mat-option[_ngcontent-%COMP%]:hover:not(.mat-option-disabled), .mat-option[_ngcontent-%COMP%]:focus:not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple):not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-active[_ngcontent-%COMP%]{background:rgba(0,0,0,.04);color:#212121}.mat-option.mat-option-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-primary[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f57c00}.mat-accent[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f44336}.mat-optgroup-label[_ngcontent-%COMP%]{color:#616161}.mat-optgroup-disabled[_ngcontent-%COMP%]   .mat-optgroup-label[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-pseudo-checkbox[_ngcontent-%COMP%]{color:#616161}.mat-pseudo-checkbox[_ngcontent-%COMP%]::after{color:#fafafa}.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{color:#b0b0b0}.mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f57c00}.mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f44336}.mat-pseudo-checkbox-checked.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{background:#b0b0b0}.mat-elevation-z0[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z1[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z2[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z3[_ngcontent-%COMP%]{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z4[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z5[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 5px 8px 0px rgba(0, 0, 0, 0.14),0px 1px 14px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z6[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z7[_ngcontent-%COMP%]{box-shadow:0px 4px 5px -2px rgba(0, 0, 0, 0.2),0px 7px 10px 1px rgba(0, 0, 0, 0.14),0px 2px 16px 1px rgba(0, 0, 0, 0.12)}.mat-elevation-z8[_ngcontent-%COMP%]{box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z9[_ngcontent-%COMP%]{box-shadow:0px 5px 6px -3px rgba(0, 0, 0, 0.2),0px 9px 12px 1px rgba(0, 0, 0, 0.14),0px 3px 16px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z10[_ngcontent-%COMP%]{box-shadow:0px 6px 6px -3px rgba(0, 0, 0, 0.2),0px 10px 14px 1px rgba(0, 0, 0, 0.14),0px 4px 18px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z11[_ngcontent-%COMP%]{box-shadow:0px 6px 7px -4px rgba(0, 0, 0, 0.2),0px 11px 15px 1px rgba(0, 0, 0, 0.14),0px 4px 20px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z12[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z13[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 13px 19px 2px rgba(0, 0, 0, 0.14),0px 5px 24px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z14[_ngcontent-%COMP%]{box-shadow:0px 7px 9px -4px rgba(0, 0, 0, 0.2),0px 14px 21px 2px rgba(0, 0, 0, 0.14),0px 5px 26px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z15[_ngcontent-%COMP%]{box-shadow:0px 8px 9px -5px rgba(0, 0, 0, 0.2),0px 15px 22px 2px rgba(0, 0, 0, 0.14),0px 6px 28px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z16[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z17[_ngcontent-%COMP%]{box-shadow:0px 8px 11px -5px rgba(0, 0, 0, 0.2),0px 17px 26px 2px rgba(0, 0, 0, 0.14),0px 6px 32px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z18[_ngcontent-%COMP%]{box-shadow:0px 9px 11px -5px rgba(0, 0, 0, 0.2),0px 18px 28px 2px rgba(0, 0, 0, 0.14),0px 7px 34px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z19[_ngcontent-%COMP%]{box-shadow:0px 9px 12px -6px rgba(0, 0, 0, 0.2),0px 19px 29px 2px rgba(0, 0, 0, 0.14),0px 7px 36px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z20[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 20px 31px 3px rgba(0, 0, 0, 0.14),0px 8px 38px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z21[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 21px 33px 3px rgba(0, 0, 0, 0.14),0px 8px 40px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z22[_ngcontent-%COMP%]{box-shadow:0px 10px 14px -6px rgba(0, 0, 0, 0.2),0px 22px 35px 3px rgba(0, 0, 0, 0.14),0px 8px 42px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z23[_ngcontent-%COMP%]{box-shadow:0px 11px 14px -7px rgba(0, 0, 0, 0.2),0px 23px 36px 3px rgba(0, 0, 0, 0.14),0px 9px 44px 8px rgba(0, 0, 0, 0.12)}.mat-elevation-z24[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12)}.mat-app-background[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-theme-loaded-marker[_ngcontent-%COMP%]{display:none}.mat-autocomplete-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-autocomplete-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover){background:#fff}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover):not(.mat-option-disabled){color:#212121}.mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f57c00}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{outline:solid 1px;border-radius:0}.mat-badge-accent[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-badge-warn[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f44336}.mat-badge[_ngcontent-%COMP%]{position:relative}.mat-badge-hidden[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{display:none}.mat-badge-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#b9b9b9;color:#757575}.mat-badge-content[_ngcontent-%COMP%]{position:absolute;text-align:center;display:inline-block;border-radius:50%;transition:transform 200ms ease-in-out;transform:scale(0.6);overflow:hidden;white-space:nowrap;text-overflow:ellipsis;pointer-events:none}.ng-animate-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%], .mat-badge-content._mat-animation-noopable[_ngcontent-%COMP%]{transition:none}.mat-badge-content.mat-badge-active[_ngcontent-%COMP%]{transform:none}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:16px;height:16px;line-height:16px}.mat-badge-small.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-8px}.mat-badge-small.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-8px}.mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-16px}.mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-16px}.mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-8px}.mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-8px}.mat-badge-medium[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:22px;height:22px;line-height:22px}.mat-badge-medium.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-11px}.mat-badge-medium.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-11px}.mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-22px}.mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-22px}.mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-11px}.mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-11px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:28px;height:28px;line-height:28px}.mat-badge-large.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-14px}.mat-badge-large.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-14px}.mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-28px}.mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-28px}.mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-14px}.mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-14px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]{color:inherit;background:transparent}.mat-button.mat-primary[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-button[disabled][disabled][_ngcontent-%COMP%], .mat-icon-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-icon-button[disabled][disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-stroked-button[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f44336}.mat-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:transparent}.mat-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{opacity:.1;background-color:currentColor}.mat-button-focus-overlay[_ngcontent-%COMP%]{background:#000}.mat-stroked-button[_ngcontent-%COMP%]:not([disabled]){border-color:rgba(0,0,0,.12)}.mat-flat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{color:#212121;background-color:#fff}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-flat-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-stroked-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-flat-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-raised-button[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone[_ngcontent-%COMP%], .mat-button-toggle-group[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{box-shadow:none}.mat-button-toggle[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-button-toggle[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121;background:#fff}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:#000}.mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12)}[dir=rtl][_ngcontent-%COMP%]   .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-group-appearance-standard.mat-button-toggle-vertical[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:none;border-top:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#616161}.mat-button-toggle-checked.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121}.mat-button-toggle-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.26);background-color:#eee}.mat-button-toggle-disabled.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{background:#fff}.mat-button-toggle-disabled.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#bdbdbd}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{border:solid 1px rgba(0,0,0,.12)}.mat-card[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-card[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-card.mat-card-flat[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-card-subtitle[_ngcontent-%COMP%]{color:#616161}.mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#616161}.mat-checkbox-checkmark[_ngcontent-%COMP%]{fill:#fafafa}.mat-checkbox-checkmark-path[_ngcontent-%COMP%]{stroke:#fafafa !important}.mat-checkbox-mixedmark[_ngcontent-%COMP%]{background-color:#fafafa}.mat-checkbox-indeterminate.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f57c00}.mat-checkbox-indeterminate.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#ff9800}.mat-checkbox-indeterminate.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f44336}.mat-checkbox-disabled.mat-checkbox-checked[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-disabled.mat-checkbox-indeterminate[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]:not(.mat-checkbox-checked)   .mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{color:#616161}.mat-checkbox[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%]{background:#f57c00}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%]{background:#ff9800}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%]{background:#f44336}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#212121}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#212121;opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled):active{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled)   .mat-chip-remove[_ngcontent-%COMP%]:hover{opacity:.54}.mat-chip.mat-standard-chip.mat-chip-disabled[_ngcontent-%COMP%]{opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]::after{background:#000}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-table[_ngcontent-%COMP%]{background:#fff}.mat-table[_ngcontent-%COMP%]   thead[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tbody[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tfoot[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], [mat-header-row][_ngcontent-%COMP%], [mat-row][_ngcontent-%COMP%], [mat-footer-row][_ngcontent-%COMP%], .mat-table-sticky[_ngcontent-%COMP%]{background:inherit}mat-row[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], th.mat-header-cell[_ngcontent-%COMP%], td.mat-cell[_ngcontent-%COMP%], td.mat-footer-cell[_ngcontent-%COMP%]{border-bottom-color:rgba(0,0,0,.12)}.mat-header-cell[_ngcontent-%COMP%]{color:#616161}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{color:#212121}.mat-calendar-arrow[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.54)}.mat-datepicker-toggle[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-next-button[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-previous-button[_ngcontent-%COMP%]{color:rgba(0,0,0,.54)}.mat-calendar-table-header[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-calendar-table-header-divider[_ngcontent-%COMP%]::after{background:rgba(0,0,0,.12)}.mat-calendar-body-label[_ngcontent-%COMP%]{color:#616161}.mat-calendar-body-cell-content[_ngcontent-%COMP%]{color:#212121;border-color:transparent}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){color:#757575}.mat-calendar-body-cell[_ngcontent-%COMP%]:not(.mat-calendar-body-disabled):hover > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-keyboard-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-program-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){background-color:rgba(0,0,0,.04)}.mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.38)}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.18)}.mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.4)}.mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12);background-color:#fff;color:#212121}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.4)}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.4)}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content-touch[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-datepicker-toggle-active[_ngcontent-%COMP%]{color:#f57c00}.mat-datepicker-toggle-active.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-datepicker-toggle-active.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-dialog-container[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-divider[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-divider-vertical[_ngcontent-%COMP%]{border-right-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-expansion-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-action-row[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-keyboard-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-program-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded)   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover:not([aria-disabled=true]){background:rgba(0,0,0,.04)}@media(hover: none){.mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded):not([aria-disabled=true])   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover{background:#fff}}.mat-expansion-panel-header-title[_ngcontent-%COMP%]{color:#212121}.mat-expansion-panel-header-description[_ngcontent-%COMP%], .mat-expansion-indicator[_ngcontent-%COMP%]::after{color:#616161}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-title[_ngcontent-%COMP%], .mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-description[_ngcontent-%COMP%]{color:inherit}.mat-form-field-label[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-hint[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-focused[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-ripple[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid)   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f57c00}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-accent   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#ff9800}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-warn   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#f44336}.mat-error[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-hint[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-legacy.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-standard[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-standard.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.04)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.02)}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:transparent}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.12)}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:rgba(0,0,0,.87)}.mat-form-field-appearance-outline.mat-focused[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field-appearance-outline.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-appearance-outline.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-invalid.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.06)}.mat-icon.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-icon.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-icon.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-type-mat-native-select[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#616161}.mat-input-element[_ngcontent-%COMP%]:disabled, .mat-form-field-type-mat-native-select.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#757575}.mat-input-element[_ngcontent-%COMP%]{caret-color:#f57c00}.mat-input-element[_ngcontent-%COMP%]::placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-moz-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-webkit-input-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]:-ms-input-placeholder{color:rgba(97,97,97,.42)}.mat-accent[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%], .mat-form-field-invalid[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#f44336}.mat-form-field-type-mat-native-select.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{color:#616161}.mat-list-item-disabled[_ngcontent-%COMP%]{background-color:#eee}.mat-list-option[_ngcontent-%COMP%]:hover, .mat-list-option[_ngcontent-%COMP%]:focus, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.04)}.mat-list-single-selected-option[_ngcontent-%COMP%], .mat-list-single-selected-option[_ngcontent-%COMP%]:hover, .mat-list-single-selected-option[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.12)}.mat-menu-panel[_ngcontent-%COMP%]{background:#fff}.mat-menu-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-menu-item[_ngcontent-%COMP%]{background:transparent;color:#212121}.mat-menu-item[disabled][_ngcontent-%COMP%], .mat-menu-item[disabled][_ngcontent-%COMP%]::after{color:rgba(0,0,0,.38)}.mat-menu-item[_ngcontent-%COMP%]   .mat-icon-no-color[_ngcontent-%COMP%], .mat-menu-item-submenu-trigger[_ngcontent-%COMP%]::after{color:rgba(0,0,0,.54)}.mat-menu-item[_ngcontent-%COMP%]:hover:not([disabled]), .mat-menu-item.cdk-program-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item.cdk-keyboard-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item-highlighted[_ngcontent-%COMP%]:not([disabled]){background:rgba(0,0,0,.04)}.mat-paginator[_ngcontent-%COMP%]{background:#fff}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{color:#616161}.mat-paginator-decrement[_ngcontent-%COMP%], .mat-paginator-increment[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54);border-right:2px solid rgba(0,0,0,.54)}.mat-paginator-first[_ngcontent-%COMP%], .mat-paginator-last[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54)}.mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-decrement[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-increment[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-first[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-last[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f57c00}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#ff9800}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f44336}.mat-progress-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f57c00}.mat-progress-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#ff9800}.mat-progress-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f44336}.mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#616161}.mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f57c00}.mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#ff9800}.mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#ff9800}.mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f44336}.mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f44336}.mat-radio-button.mat-radio-disabled.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-label-content[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-radio-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-select-value[_ngcontent-%COMP%]{color:#212121}.mat-select-placeholder[_ngcontent-%COMP%]{color:rgba(97,97,97,.42)}.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%]{color:#757575}.mat-select-arrow[_ngcontent-%COMP%]{color:#616161}.mat-select-panel[_ngcontent-%COMP%]{background:#fff}.mat-select-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-select-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple){background:rgba(0,0,0,.12)}.mat-form-field.mat-focused.mat-primary[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-invalid[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#757575}.mat-drawer-container[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-drawer[_ngcontent-%COMP%]{background-color:#fff;color:#212121}.mat-drawer.mat-drawer-push[_ngcontent-%COMP%]{background-color:#fff}.mat-drawer[_ngcontent-%COMP%]:not(.mat-drawer-side){box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-drawer-side[_ngcontent-%COMP%]{border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-backdrop.mat-drawer-shown[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.6)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.54)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.54)}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.54)}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle[_ngcontent-%COMP%]:not(.mat-checked)   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-slide-toggle-thumb[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12);background-color:#fafafa}.mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-primary[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f57c00}.mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-accent[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#ff9800}.mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-warn[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f44336}.mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.2)}.mat-slider[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%], .cdk-focused[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-disabled[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]   .mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing)   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26);background-color:transparent}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26)}.mat-slider-has-ticks[_ngcontent-%COMP%]   .mat-slider-wrapper[_ngcontent-%COMP%]::after{border-color:rgba(0,0,0,.7)}.mat-slider-horizontal[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to right, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent);background-image:-moz-repeating-linear-gradient(0.0001deg, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-slider-vertical[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to bottom, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-step-header.cdk-keyboard-focused[_ngcontent-%COMP%], .mat-step-header.cdk-program-focused[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]:hover{background-color:rgba(0,0,0,.04)}@media(hover: none){.mat-step-header[_ngcontent-%COMP%]:hover{background:none}}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-optional[_ngcontent-%COMP%]{color:#616161}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon[_ngcontent-%COMP%]{background-color:#616161;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-selected[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-done[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-edit[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-error[_ngcontent-%COMP%]{background-color:transparent;color:#f44336}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-active[_ngcontent-%COMP%]{color:#212121}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-error[_ngcontent-%COMP%]{color:#f44336}.mat-stepper-horizontal[_ngcontent-%COMP%], .mat-stepper-vertical[_ngcontent-%COMP%]{background-color:#fff}.mat-stepper-vertical-line[_ngcontent-%COMP%]::before{border-left-color:rgba(0,0,0,.12)}.mat-horizontal-stepper-header[_ngcontent-%COMP%]::before, .mat-horizontal-stepper-header[_ngcontent-%COMP%]::after, .mat-stepper-horizontal-line[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-sort-header-arrow[_ngcontent-%COMP%]{color:#616161}.mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-header[_ngcontent-%COMP%]{border-bottom:1px solid rgba(0,0,0,.12)}.mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%]{border-top:1px solid rgba(0,0,0,.12);border-bottom:none}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{color:#212121}.mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:#757575}.mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#212121}.mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#757575}.mat-tab-group[class*=mat-background-][_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar[class*=mat-background-][_ngcontent-%COMP%]{border-bottom:none;border-top:none}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-toolbar[_ngcontent-%COMP%]{background:#f5f5f5;color:#212121}.mat-toolbar.mat-primary[_ngcontent-%COMP%]{background:#f57c00;color:#fff}.mat-toolbar.mat-accent[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-toolbar.mat-warn[_ngcontent-%COMP%]{background:#f44336;color:#fff}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:currentColor}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:inherit}.mat-toolbar[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:currentColor}.mat-tooltip[_ngcontent-%COMP%]{background:rgba(97,97,97,.9)}.mat-tree[_ngcontent-%COMP%]{background:#fff}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{color:#212121}.mat-snack-bar-container[_ngcontent-%COMP%]{color:rgba(255,255,255,.7);background:#323232;box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-simple-snackbar-action[_ngcontent-%COMP%]{color:#ff9800}.active-button[_ngcontent-%COMP%]{background-color:#ff9800;border:1px solid #ebebeb;color:#fff}.button-contents[_ngcontent-%COMP%]{align-items:center;display:flex;text-transform:uppercase}mat-icon[_ngcontent-%COMP%]{margin-right:6px}"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/data_selection/results_download/results_download_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class ResultsDownloadContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.flaggedAnnotations$ = this.store.select(getFlaggedAnnotations);
            this.numFlaggedAnnotations$ = this.flaggedAnnotations$.pipe(map((/**
             * @param {?} flaggedAnnotations
             * @return {?}
             */
            (flaggedAnnotations) => flaggedAnnotations.length)));
            this.activeRuns$ = this.store.select(getCurrentRouteRunSelection).pipe(map((/**
             * @param {?} runSelection
             * @return {?}
             */
            (runSelection) => {
                if (!runSelection)
                    return [];
                return Array.from(runSelection.entries())
                    .filter((/**
                 * @param {?} run
                 * @return {?}
                 */
                (run) => run[1]))
                    .map((/**
                 * @param {?} run
                 * @return {?}
                 */
                (run) => run[0]));
            })));
            this.flaggedData$ = combineLatest([
                this.store.select(getAnnotationData),
                this.flaggedAnnotations$,
            ]).pipe(map((/**
             * @param {?} __0
             * @return {?}
             */
            ([annotationData, flaggedAnnotations]) => {
                /** @type {?} */
                const flagSet = new Set(flaggedAnnotations);
                /** @type {?} */
                const flaggedData = Object.entries(annotationData).filter((/**
                 * @param {?} entry
                 * @return {?}
                 */
                (entry) => flagSet.has(entry[0])));
                return flaggedData;
            })));
            this.metrics$ = combineLatest([
                this.store.select(getRunToMetrics),
                this.activeRuns$,
                this.store.select(getMetricFilters),
            ]).pipe(map((/**
             * @param {?} __0
             * @return {?}
             */
            ([runToMetrics, activeRuns, metricFilters]) => {
                /** @type {?} */
                let metrics = Object.keys(metricFilters);
                for (const run of activeRuns) {
                    if (runToMetrics[run]) {
                        metrics = metrics.concat(runToMetrics[run].filter((/**
                         * @param {?} key
                         * @return {?}
                         */
                        (key) => metricIsNpmiAndNotDiff(key))));
                    }
                }
                metrics = [...new Set(metrics)];
                return metrics;
            })));
        }
    }
    ResultsDownloadContainer.decorators = [
        { type: Component, args: [{
                    selector: 'npmi-results-download',
                    template: `
    <results-download-component
      [numFlaggedAnnotations]="numFlaggedAnnotations$ | async"
      [runs]="activeRuns$ | async"
      [flaggedData]="flaggedData$ | async"
      [metrics]="metrics$ | async"
    ></results-download-component>
  `,
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    ResultsDownloadContainer.ctorParameters = () => [
        { type: Store }
    ];
    /** @nocollapse */ ResultsDownloadContainer.ɵfac = function ResultsDownloadContainer_Factory(t) { return new (t || ResultsDownloadContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ ResultsDownloadContainer.ɵcmp = ɵɵdefineComponent({ type: ResultsDownloadContainer, selectors: [["npmi-results-download"]], decls: 5, vars: 12, consts: [[3, "numFlaggedAnnotations", "runs", "flaggedData", "metrics"]], template: function ResultsDownloadContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelement(0, "results-download-component", 0);
            ɵɵpipe(1, "async");
            ɵɵpipe(2, "async");
            ɵɵpipe(3, "async");
            ɵɵpipe(4, "async");
        } if (rf & 2) {
            ɵɵproperty("numFlaggedAnnotations", ɵɵpipeBind1(1, 4, ctx.numFlaggedAnnotations$))("runs", ɵɵpipeBind1(2, 6, ctx.activeRuns$))("flaggedData", ɵɵpipeBind1(3, 8, ctx.flaggedData$))("metrics", ɵɵpipeBind1(4, 10, ctx.metrics$));
        } }, directives: [ResultsDownloadComponent], pipes: [AsyncPipe], encapsulation: 2, changeDetection: 0 });

    const _c0$u = ["*"];
    // Boilerplate for applying mixins to MatChip.
    /**
     * \@docs-private
     */
    class MatChipBase {
        /**
         * @param {?} _elementRef
         */
        constructor(_elementRef) {
            this._elementRef = _elementRef;
        }
    }
    /** @type {?} */
    const _MatChipMixinBase = mixinTabIndex(mixinColor(mixinDisableRipple(MatChipBase), 'primary'), -1);
    /**
     * Dummy directive to add CSS class to chip avatar.
     * \@docs-private
     */
    class MatChipAvatar {
    }
    MatChipAvatar.ɵfac = function MatChipAvatar_Factory(t) { return new (t || MatChipAvatar)(); };
    MatChipAvatar.ɵdir = ɵɵdefineDirective({ type: MatChipAvatar, selectors: [["mat-chip-avatar"], ["", "matChipAvatar", ""]], hostAttrs: [1, "mat-chip-avatar"] });
    /**
     * Dummy directive to add CSS class to chip trailing icon.
     * \@docs-private
     */
    class MatChipTrailingIcon {
    }
    MatChipTrailingIcon.ɵfac = function MatChipTrailingIcon_Factory(t) { return new (t || MatChipTrailingIcon)(); };
    MatChipTrailingIcon.ɵdir = ɵɵdefineDirective({ type: MatChipTrailingIcon, selectors: [["mat-chip-trailing-icon"], ["", "matChipTrailingIcon", ""]], hostAttrs: [1, "mat-chip-trailing-icon"] });
    /**
     * Material design styled Chip component. Used inside the MatChipList component.
     */
    class MatChip extends _MatChipMixinBase {
        /**
         * @param {?} _elementRef
         * @param {?} _ngZone
         * @param {?} platform
         * @param {?} globalRippleOptions
         * @param {?=} animationMode
         * @param {?=} _changeDetectorRef
         * @param {?=} tabIndex
         * @param {?=} _document
         */
        constructor(_elementRef, _ngZone, platform, globalRippleOptions, 
        // @breaking-change 8.0.0 `animationMode` parameter to become required.
        animationMode, _changeDetectorRef, tabIndex, 
        // @breaking-change 11.0.0 `_document` parameter to become required.
        _document) {
            super(_elementRef);
            this._elementRef = _elementRef;
            this._ngZone = _ngZone;
            this._changeDetectorRef = _changeDetectorRef;
            /**
             * Whether the chip has focus.
             */
            this._hasFocus = false;
            /**
             * Whether the chip list is selectable
             */
            this.chipListSelectable = true;
            /**
             * Whether the chip list is in multi-selection mode.
             */
            this._chipListMultiple = false;
            /**
             * Whether the chip list as a whole is disabled.
             */
            this._chipListDisabled = false;
            this._selected = false;
            this._selectable = true;
            this._disabled = false;
            this._removable = true;
            /**
             * Emits when the chip is focused.
             */
            this._onFocus = new Subject();
            /**
             * Emits when the chip is blured.
             */
            this._onBlur = new Subject();
            /**
             * Emitted when the chip is selected or deselected.
             */
            this.selectionChange = new EventEmitter();
            /**
             * Emitted when the chip is destroyed.
             */
            this.destroyed = new EventEmitter();
            /**
             * Emitted when a chip is to be removed.
             */
            this.removed = new EventEmitter();
            this._addHostClassName();
            // Dynamically create the ripple target, append it within the chip, and use it as the
            // chip's ripple target. Adding the class '.mat-chip-ripple' ensures that it will have
            // the proper styles.
            this._chipRippleTarget = (_document || document).createElement('div');
            this._chipRippleTarget.classList.add('mat-chip-ripple');
            this._elementRef.nativeElement.appendChild(this._chipRippleTarget);
            this._chipRipple = new RippleRenderer(this, _ngZone, this._chipRippleTarget, platform);
            this._chipRipple.setupTriggerEvents(_elementRef);
            this.rippleConfig = globalRippleOptions || {};
            this._animationsDisabled = animationMode === 'NoopAnimations';
            this.tabIndex = tabIndex != null ? (parseInt(tabIndex) || -1) : -1;
        }
        /**
         * Whether ripples are disabled on interaction
         * \@docs-private
         * @return {?}
         */
        get rippleDisabled() {
            return this.disabled || this.disableRipple || !!this.rippleConfig.disabled;
        }
        /**
         * Whether the chip is selected.
         * @return {?}
         */
        get selected() { return this._selected; }
        /**
         * @param {?} value
         * @return {?}
         */
        set selected(value) {
            /** @type {?} */
            const coercedValue = coerceBooleanProperty(value);
            if (coercedValue !== this._selected) {
                this._selected = coercedValue;
                this._dispatchSelectionChange();
            }
        }
        /**
         * The value of the chip. Defaults to the content inside `<mat-chip>` tags.
         * @return {?}
         */
        get value() {
            return this._value !== undefined
                ? this._value
                : this._elementRef.nativeElement.textContent;
        }
        /**
         * @param {?} value
         * @return {?}
         */
        set value(value) { this._value = value; }
        /**
         * Whether or not the chip is selectable. When a chip is not selectable,
         * changes to its selected state are always ignored. By default a chip is
         * selectable, and it becomes non-selectable if its parent chip list is
         * not selectable.
         * @return {?}
         */
        get selectable() { return this._selectable && this.chipListSelectable; }
        /**
         * @param {?} value
         * @return {?}
         */
        set selectable(value) {
            this._selectable = coerceBooleanProperty(value);
        }
        /**
         * Whether the chip is disabled.
         * @return {?}
         */
        get disabled() { return this._chipListDisabled || this._disabled; }
        /**
         * @param {?} value
         * @return {?}
         */
        set disabled(value) {
            this._disabled = coerceBooleanProperty(value);
        }
        /**
         * Determines whether or not the chip displays the remove styling and emits (removed) events.
         * @return {?}
         */
        get removable() { return this._removable; }
        /**
         * @param {?} value
         * @return {?}
         */
        set removable(value) {
            this._removable = coerceBooleanProperty(value);
        }
        /**
         * The ARIA selected applied to the chip.
         * @return {?}
         */
        get ariaSelected() {
            // Remove the `aria-selected` when the chip is deselected in single-selection mode, because
            // it adds noise to NVDA users where "not selected" will be read out for each chip.
            return this.selectable && (this._chipListMultiple || this.selected) ?
                this.selected.toString() : null;
        }
        /**
         * @return {?}
         */
        _addHostClassName() {
            /** @type {?} */
            const basicChipAttrName = 'mat-basic-chip';
            /** @type {?} */
            const element = (/** @type {?} */ (this._elementRef.nativeElement));
            if (element.hasAttribute(basicChipAttrName) ||
                element.tagName.toLowerCase() === basicChipAttrName) {
                element.classList.add(basicChipAttrName);
                return;
            }
            else {
                element.classList.add('mat-standard-chip');
            }
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this.destroyed.emit({ chip: this });
            this._chipRipple._removeTriggerEvents();
        }
        /**
         * Selects the chip.
         * @return {?}
         */
        select() {
            if (!this._selected) {
                this._selected = true;
                this._dispatchSelectionChange();
                this._markForCheck();
            }
        }
        /**
         * Deselects the chip.
         * @return {?}
         */
        deselect() {
            if (this._selected) {
                this._selected = false;
                this._dispatchSelectionChange();
                this._markForCheck();
            }
        }
        /**
         * Select this chip and emit selected event
         * @return {?}
         */
        selectViaInteraction() {
            if (!this._selected) {
                this._selected = true;
                this._dispatchSelectionChange(true);
                this._markForCheck();
            }
        }
        /**
         * Toggles the current selected state of this chip.
         * @param {?=} isUserInput
         * @return {?}
         */
        toggleSelected(isUserInput = false) {
            this._selected = !this.selected;
            this._dispatchSelectionChange(isUserInput);
            this._markForCheck();
            return this.selected;
        }
        /**
         * Allows for programmatic focusing of the chip.
         * @return {?}
         */
        focus() {
            if (!this._hasFocus) {
                this._elementRef.nativeElement.focus();
                this._onFocus.next({ chip: this });
            }
            this._hasFocus = true;
        }
        /**
         * Allows for programmatic removal of the chip. Called by the MatChipList when the DELETE or
         * BACKSPACE keys are pressed.
         *
         * Informs any listeners of the removal request. Does not remove the chip from the DOM.
         * @return {?}
         */
        remove() {
            if (this.removable) {
                this.removed.emit({ chip: this });
            }
        }
        /**
         * Handles click events on the chip.
         * @param {?} event
         * @return {?}
         */
        _handleClick(event) {
            if (this.disabled) {
                event.preventDefault();
            }
            else {
                event.stopPropagation();
            }
        }
        /**
         * Handle custom key presses.
         * @param {?} event
         * @return {?}
         */
        _handleKeydown(event) {
            if (this.disabled) {
                return;
            }
            switch (event.keyCode) {
                case DELETE:
                case BACKSPACE:
                    // If we are removable, remove the focused chip
                    this.remove();
                    // Always prevent so page navigation does not occur
                    event.preventDefault();
                    break;
                case SPACE:
                    // If we are selectable, toggle the focused chip
                    if (this.selectable) {
                        this.toggleSelected(true);
                    }
                    // Always prevent space from scrolling the page since the list has focus
                    event.preventDefault();
                    break;
            }
        }
        /**
         * @return {?}
         */
        _blur() {
            // When animations are enabled, Angular may end up removing the chip from the DOM a little
            // earlier than usual, causing it to be blurred and throwing off the logic in the chip list
            // that moves focus not the next item. To work around the issue, we defer marking the chip
            // as not focused until the next time the zone stabilizes.
            this._ngZone.onStable
                .asObservable()
                .pipe(take(1))
                .subscribe((/**
             * @return {?}
             */
            () => {
                this._ngZone.run((/**
                 * @return {?}
                 */
                () => {
                    this._hasFocus = false;
                    this._onBlur.next({ chip: this });
                }));
            }));
        }
        /**
         * @private
         * @param {?=} isUserInput
         * @return {?}
         */
        _dispatchSelectionChange(isUserInput = false) {
            this.selectionChange.emit({
                source: this,
                isUserInput,
                selected: this._selected
            });
        }
        /**
         * @private
         * @return {?}
         */
        _markForCheck() {
            // @breaking-change 9.0.0 Remove this method once the _changeDetectorRef is a required param.
            if (this._changeDetectorRef) {
                this._changeDetectorRef.markForCheck();
            }
        }
    }
    MatChip.ɵfac = function MatChip_Factory(t) { return new (t || MatChip)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(NgZone), ɵɵdirectiveInject(Platform), ɵɵdirectiveInject(MAT_RIPPLE_GLOBAL_OPTIONS, 8), ɵɵdirectiveInject(ANIMATION_MODULE_TYPE, 8), ɵɵdirectiveInject(ChangeDetectorRef), ɵɵinjectAttribute('tabindex'), ɵɵdirectiveInject(DOCUMENT$1, 8)); };
    MatChip.ɵdir = ɵɵdefineDirective({ type: MatChip, selectors: [["mat-basic-chip"], ["", "mat-basic-chip", ""], ["mat-chip"], ["", "mat-chip", ""]], contentQueries: function MatChip_ContentQueries(rf, ctx, dirIndex) { if (rf & 1) {
            ɵɵcontentQuery(dirIndex, MatChipAvatar, true);
            ɵɵcontentQuery(dirIndex, MatChipTrailingIcon, true);
            ɵɵcontentQuery(dirIndex, MatChipRemove, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.avatar = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.trailingIcon = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.removeIcon = _t.first);
        } }, hostAttrs: ["role", "option", 1, "mat-chip", "mat-focus-indicator"], hostVars: 14, hostBindings: function MatChip_HostBindings(rf, ctx) { if (rf & 1) {
            ɵɵlistener("click", function MatChip_click_HostBindingHandler($event) { return ctx._handleClick($event); })("keydown", function MatChip_keydown_HostBindingHandler($event) { return ctx._handleKeydown($event); })("focus", function MatChip_focus_HostBindingHandler() { return ctx.focus(); })("blur", function MatChip_blur_HostBindingHandler() { return ctx._blur(); });
        } if (rf & 2) {
            ɵɵattribute("tabindex", ctx.disabled ? null : ctx.tabIndex)("disabled", ctx.disabled || null)("aria-disabled", ctx.disabled.toString())("aria-selected", ctx.ariaSelected);
            ɵɵclassProp("mat-chip-selected", ctx.selected)("mat-chip-with-avatar", ctx.avatar)("mat-chip-with-trailing-icon", ctx.trailingIcon || ctx.removeIcon)("mat-chip-disabled", ctx.disabled)("_mat-animation-noopable", ctx._animationsDisabled);
        } }, inputs: { color: "color", disableRipple: "disableRipple", tabIndex: "tabIndex", selected: "selected", value: "value", selectable: "selectable", disabled: "disabled", removable: "removable" }, outputs: { selectionChange: "selectionChange", destroyed: "destroyed", removed: "removed" }, exportAs: ["matChip"], features: [ɵɵInheritDefinitionFeature] });
    /** @nocollapse */
    MatChip.ctorParameters = () => [
        { type: ElementRef },
        { type: NgZone },
        { type: Platform },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [MAT_RIPPLE_GLOBAL_OPTIONS,] }] },
        { type: String, decorators: [{ type: Optional }, { type: Inject, args: [ANIMATION_MODULE_TYPE,] }] },
        { type: ChangeDetectorRef },
        { type: String, decorators: [{ type: Attribute, args: ['tabindex',] }] },
        { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [DOCUMENT$1,] }] }
    ];
    MatChip.propDecorators = {
        avatar: [{ type: ContentChild, args: [MatChipAvatar,] }],
        trailingIcon: [{ type: ContentChild, args: [MatChipTrailingIcon,] }],
        removeIcon: [{ type: ContentChild, args: [forwardRef((/**
                     * @return {?}
                     */
                    () => MatChipRemove)),] }],
        selected: [{ type: Input }],
        value: [{ type: Input }],
        selectable: [{ type: Input }],
        disabled: [{ type: Input }],
        removable: [{ type: Input }],
        selectionChange: [{ type: Output }],
        destroyed: [{ type: Output }],
        removed: [{ type: Output }]
    };
    /**
     * Applies proper (click) support and adds styling for use with the Material Design "cancel" icon
     * available at https://material.io/icons/#ic_cancel.
     *
     * Example:
     *
     *     `<mat-chip>
     *       <mat-icon matChipRemove>cancel</mat-icon>
     *     </mat-chip>`
     *
     * You *may* use a custom icon, but you may need to override the `mat-chip-remove` positioning
     * styles to properly center the icon within the chip.
     */
    class MatChipRemove {
        /**
         * @param {?} _parentChip
         * @param {?=} elementRef
         */
        constructor(_parentChip, 
        // @breaking-change 11.0.0 `elementRef` parameter to be made required.
        elementRef) {
            this._parentChip = _parentChip;
            // @breaking-change 11.0.0 Remove null check for `elementRef`.
            if (elementRef && elementRef.nativeElement.nodeName === 'BUTTON') {
                elementRef.nativeElement.setAttribute('type', 'button');
            }
        }
        /**
         * Calls the parent chip's public `remove()` method if applicable.
         * @param {?} event
         * @return {?}
         */
        _handleClick(event) {
            /** @type {?} */
            const parentChip = this._parentChip;
            if (parentChip.removable && !parentChip.disabled) {
                parentChip.remove();
            }
            // We need to stop event propagation because otherwise the event will bubble up to the
            // form field and cause the `onContainerClick` method to be invoked. This method would then
            // reset the focused chip that has been focused after chip removal. Usually the parent
            // the parent click listener of the `MatChip` would prevent propagation, but it can happen
            // that the chip is being removed before the event bubbles up.
            event.stopPropagation();
        }
    }
    MatChipRemove.ɵfac = function MatChipRemove_Factory(t) { return new (t || MatChipRemove)(ɵɵdirectiveInject(MatChip), ɵɵdirectiveInject(ElementRef)); };
    MatChipRemove.ɵdir = ɵɵdefineDirective({ type: MatChipRemove, selectors: [["", "matChipRemove", ""]], hostAttrs: [1, "mat-chip-remove", "mat-chip-trailing-icon"], hostBindings: function MatChipRemove_HostBindings(rf, ctx) { if (rf & 1) {
            ɵɵlistener("click", function MatChipRemove_click_HostBindingHandler($event) { return ctx._handleClick($event); });
        } } });
    /** @nocollapse */
    MatChipRemove.ctorParameters = () => [
        { type: MatChip },
        { type: ElementRef }
    ];
    /**
     * Injection token to be used to override the default options for the chips module.
     * @type {?}
     */
    const MAT_CHIPS_DEFAULT_OPTIONS = new InjectionToken('mat-chips-default-options');

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/chips/chip-list.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    // Boilerplate for applying mixins to MatChipList.
    /**
     * \@docs-private
     */
    class MatChipListBase {
        /**
         * @param {?} _defaultErrorStateMatcher
         * @param {?} _parentForm
         * @param {?} _parentFormGroup
         * @param {?} ngControl
         */
        constructor(_defaultErrorStateMatcher, _parentForm, _parentFormGroup, ngControl) {
            this._defaultErrorStateMatcher = _defaultErrorStateMatcher;
            this._parentForm = _parentForm;
            this._parentFormGroup = _parentFormGroup;
            this.ngControl = ngControl;
        }
    }
    /** @type {?} */
    const _MatChipListMixinBase = mixinErrorState(MatChipListBase);
    // Increasing integer for generating unique ids for chip-list components.
    /** @type {?} */
    let nextUniqueId$7 = 0;
    /**
     * Change event object that is emitted when the chip list value has changed.
     */
    class MatChipListChange {
        /**
         * @param {?} source
         * @param {?} value
         */
        constructor(source, value) {
            this.source = source;
            this.value = value;
        }
    }
    /**
     * A material design chips component (named ChipList for its similarity to the List component).
     */
    class MatChipList extends _MatChipListMixinBase {
        /**
         * @param {?} _elementRef
         * @param {?} _changeDetectorRef
         * @param {?} _dir
         * @param {?} _parentForm
         * @param {?} _parentFormGroup
         * @param {?} _defaultErrorStateMatcher
         * @param {?} ngControl
         */
        constructor(_elementRef, _changeDetectorRef, _dir, _parentForm, _parentFormGroup, _defaultErrorStateMatcher, ngControl) {
            super(_defaultErrorStateMatcher, _parentForm, _parentFormGroup, ngControl);
            this._elementRef = _elementRef;
            this._changeDetectorRef = _changeDetectorRef;
            this._dir = _dir;
            this.ngControl = ngControl;
            /**
             * Implemented as part of MatFormFieldControl.
             * \@docs-private
             */
            this.controlType = 'mat-chip-list';
            /**
             * When a chip is destroyed, we store the index of the destroyed chip until the chips
             * query list notifies about the update. This is necessary because we cannot determine an
             * appropriate chip that should receive focus until the array of chips updated completely.
             */
            this._lastDestroyedChipIndex = null;
            /**
             * Subject that emits when the component has been destroyed.
             */
            this._destroyed = new Subject();
            /**
             * Uid of the chip list
             */
            this._uid = `mat-chip-list-${nextUniqueId$7++}`;
            /**
             * Tab index for the chip list.
             */
            this._tabIndex = 0;
            /**
             * User defined tab index.
             * When it is not null, use user defined tab index. Otherwise use _tabIndex
             */
            this._userTabIndex = null;
            /**
             * Function when touched
             */
            this._onTouched = (/**
             * @return {?}
             */
            () => { });
            /**
             * Function when changed
             */
            this._onChange = (/**
             * @return {?}
             */
            () => { });
            this._multiple = false;
            this._compareWith = (/**
             * @param {?} o1
             * @param {?} o2
             * @return {?}
             */
            (o1, o2) => o1 === o2);
            this._required = false;
            this._disabled = false;
            /**
             * Orientation of the chip list.
             */
            this.ariaOrientation = 'horizontal';
            this._selectable = true;
            /**
             * Event emitted when the selected chip list value has been changed by the user.
             */
            this.change = new EventEmitter();
            /**
             * Event that emits whenever the raw value of the chip-list changes. This is here primarily
             * to facilitate the two-way binding for the `value` input.
             * \@docs-private
             */
            this.valueChange = new EventEmitter();
            if (this.ngControl) {
                this.ngControl.valueAccessor = this;
            }
        }
        /**
         * The array of selected chips inside chip list.
         * @return {?}
         */
        get selected() {
            return this.multiple ? this._selectionModel.selected : this._selectionModel.selected[0];
        }
        /**
         * The ARIA role applied to the chip list.
         * @return {?}
         */
        get role() { return this.empty ? null : 'listbox'; }
        /**
         * Whether the user should be allowed to select multiple chips.
         * @return {?}
         */
        get multiple() { return this._multiple; }
        /**
         * @param {?} value
         * @return {?}
         */
        set multiple(value) {
            this._multiple = coerceBooleanProperty(value);
            this._syncChipsState();
        }
        /**
         * A function to compare the option values with the selected values. The first argument
         * is a value from an option. The second is a value from the selection. A boolean
         * should be returned.
         * @return {?}
         */
        get compareWith() { return this._compareWith; }
        /**
         * @param {?} fn
         * @return {?}
         */
        set compareWith(fn) {
            this._compareWith = fn;
            if (this._selectionModel) {
                // A different comparator means the selection could change.
                this._initializeSelection();
            }
        }
        /**
         * Implemented as part of MatFormFieldControl.
         * \@docs-private
         * @return {?}
         */
        get value() { return this._value; }
        /**
         * @param {?} value
         * @return {?}
         */
        set value(value) {
            this.writeValue(value);
            this._value = value;
        }
        /**
         * Implemented as part of MatFormFieldControl.
         * \@docs-private
         * @return {?}
         */
        get id() {
            return this._chipInput ? this._chipInput.id : this._uid;
        }
        /**
         * Implemented as part of MatFormFieldControl.
         * \@docs-private
         * @return {?}
         */
        get required() { return this._required; }
        /**
         * @param {?} value
         * @return {?}
         */
        set required(value) {
            this._required = coerceBooleanProperty(value);
            this.stateChanges.next();
        }
        /**
         * Implemented as part of MatFormFieldControl.
         * \@docs-private
         * @return {?}
         */
        get placeholder() {
            return this._chipInput ? this._chipInput.placeholder : this._placeholder;
        }
        /**
         * @param {?} value
         * @return {?}
         */
        set placeholder(value) {
            this._placeholder = value;
            this.stateChanges.next();
        }
        /**
         * Whether any chips or the matChipInput inside of this chip-list has focus.
         * @return {?}
         */
        get focused() {
            return (this._chipInput && this._chipInput.focused) || this._hasFocusedChip();
        }
        /**
         * Implemented as part of MatFormFieldControl.
         * \@docs-private
         * @return {?}
         */
        get empty() {
            return (!this._chipInput || this._chipInput.empty) && this.chips.length === 0;
        }
        /**
         * Implemented as part of MatFormFieldControl.
         * \@docs-private
         * @return {?}
         */
        get shouldLabelFloat() { return !this.empty || this.focused; }
        /**
         * Implemented as part of MatFormFieldControl.
         * \@docs-private
         * @return {?}
         */
        get disabled() { return this.ngControl ? !!this.ngControl.disabled : this._disabled; }
        /**
         * @param {?} value
         * @return {?}
         */
        set disabled(value) {
            this._disabled = coerceBooleanProperty(value);
            this._syncChipsState();
        }
        /**
         * Whether or not this chip list is selectable. When a chip list is not selectable,
         * the selected states for all the chips inside the chip list are always ignored.
         * @return {?}
         */
        get selectable() { return this._selectable; }
        /**
         * @param {?} value
         * @return {?}
         */
        set selectable(value) {
            this._selectable = coerceBooleanProperty(value);
            if (this.chips) {
                this.chips.forEach((/**
                 * @param {?} chip
                 * @return {?}
                 */
                chip => chip.chipListSelectable = this._selectable));
            }
        }
        /**
         * @param {?} value
         * @return {?}
         */
        set tabIndex(value) {
            this._userTabIndex = value;
            this._tabIndex = value;
        }
        /**
         * Combined stream of all of the child chips' selection change events.
         * @return {?}
         */
        get chipSelectionChanges() {
            return merge(...this.chips.map((/**
             * @param {?} chip
             * @return {?}
             */
            chip => chip.selectionChange)));
        }
        /**
         * Combined stream of all of the child chips' focus change events.
         * @return {?}
         */
        get chipFocusChanges() {
            return merge(...this.chips.map((/**
             * @param {?} chip
             * @return {?}
             */
            chip => chip._onFocus)));
        }
        /**
         * Combined stream of all of the child chips' blur change events.
         * @return {?}
         */
        get chipBlurChanges() {
            return merge(...this.chips.map((/**
             * @param {?} chip
             * @return {?}
             */
            chip => chip._onBlur)));
        }
        /**
         * Combined stream of all of the child chips' remove change events.
         * @return {?}
         */
        get chipRemoveChanges() {
            return merge(...this.chips.map((/**
             * @param {?} chip
             * @return {?}
             */
            chip => chip.destroyed)));
        }
        /**
         * @return {?}
         */
        ngAfterContentInit() {
            this._keyManager = new FocusKeyManager(this.chips)
                .withWrap()
                .withVerticalOrientation()
                .withHorizontalOrientation(this._dir ? this._dir.value : 'ltr');
            if (this._dir) {
                this._dir.change
                    .pipe(takeUntil(this._destroyed))
                    .subscribe((/**
                 * @param {?} dir
                 * @return {?}
                 */
                dir => this._keyManager.withHorizontalOrientation(dir)));
            }
            this._keyManager.tabOut.pipe(takeUntil(this._destroyed)).subscribe((/**
             * @return {?}
             */
            () => {
                this._allowFocusEscape();
            }));
            // When the list changes, re-subscribe
            this.chips.changes.pipe(startWith(null), takeUntil(this._destroyed)).subscribe((/**
             * @return {?}
             */
            () => {
                if (this.disabled) {
                    // Since this happens after the content has been
                    // checked, we need to defer it to the next tick.
                    Promise.resolve().then((/**
                     * @return {?}
                     */
                    () => {
                        this._syncChipsState();
                    }));
                }
                this._resetChips();
                // Reset chips selected/deselected status
                this._initializeSelection();
                // Check to see if we need to update our tab index
                this._updateTabIndex();
                // Check to see if we have a destroyed chip and need to refocus
                this._updateFocusForDestroyedChips();
                this.stateChanges.next();
            }));
        }
        /**
         * @return {?}
         */
        ngOnInit() {
            this._selectionModel = new SelectionModel(this.multiple, undefined, false);
            this.stateChanges.next();
        }
        /**
         * @return {?}
         */
        ngDoCheck() {
            if (this.ngControl) {
                // We need to re-evaluate this on every change detection cycle, because there are some
                // error triggers that we can't subscribe to (e.g. parent form submissions). This means
                // that whatever logic is in here has to be super lean or we risk destroying the performance.
                this.updateErrorState();
                if (this.ngControl.disabled !== this._disabled) {
                    this.disabled = !!this.ngControl.disabled;
                }
            }
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this._destroyed.next();
            this._destroyed.complete();
            this.stateChanges.complete();
            this._dropSubscriptions();
        }
        /**
         * Associates an HTML input element with this chip list.
         * @param {?} inputElement
         * @return {?}
         */
        registerInput(inputElement) {
            this._chipInput = inputElement;
        }
        /**
         * Implemented as part of MatFormFieldControl.
         * \@docs-private
         * @param {?} ids
         * @return {?}
         */
        setDescribedByIds(ids) { this._ariaDescribedby = ids.join(' '); }
        // Implemented as part of ControlValueAccessor.
        /**
         * @param {?} value
         * @return {?}
         */
        writeValue(value) {
            if (this.chips) {
                this._setSelectionByValue(value, false);
            }
        }
        // Implemented as part of ControlValueAccessor.
        /**
         * @param {?} fn
         * @return {?}
         */
        registerOnChange(fn) {
            this._onChange = fn;
        }
        // Implemented as part of ControlValueAccessor.
        /**
         * @param {?} fn
         * @return {?}
         */
        registerOnTouched(fn) {
            this._onTouched = fn;
        }
        // Implemented as part of ControlValueAccessor.
        /**
         * @param {?} isDisabled
         * @return {?}
         */
        setDisabledState(isDisabled) {
            this.disabled = isDisabled;
            this.stateChanges.next();
        }
        /**
         * Implemented as part of MatFormFieldControl.
         * \@docs-private
         * @param {?} event
         * @return {?}
         */
        onContainerClick(event) {
            if (!this._originatesFromChip(event)) {
                this.focus();
            }
        }
        /**
         * Focuses the first non-disabled chip in this chip list, or the associated input when there
         * are no eligible chips.
         * @param {?=} options
         * @return {?}
         */
        focus(options) {
            if (this.disabled) {
                return;
            }
            // TODO: ARIA says this should focus the first `selected` chip if any are selected.
            // Focus on first element if there's no chipInput inside chip-list
            if (this._chipInput && this._chipInput.focused) ;
            else if (this.chips.length > 0) {
                this._keyManager.setFirstItemActive();
                this.stateChanges.next();
            }
            else {
                this._focusInput(options);
                this.stateChanges.next();
            }
        }
        /**
         * Attempt to focus an input if we have one.
         * @param {?=} options
         * @return {?}
         */
        _focusInput(options) {
            if (this._chipInput) {
                this._chipInput.focus(options);
            }
        }
        /**
         * Pass events to the keyboard manager. Available here for tests.
         * @param {?} event
         * @return {?}
         */
        _keydown(event) {
            /** @type {?} */
            const target = (/** @type {?} */ (event.target));
            // If they are on an empty input and hit backspace, focus the last chip
            if (event.keyCode === BACKSPACE && this._isInputEmpty(target)) {
                this._keyManager.setLastItemActive();
                event.preventDefault();
            }
            else if (target && target.classList.contains('mat-chip')) {
                if (event.keyCode === HOME) {
                    this._keyManager.setFirstItemActive();
                    event.preventDefault();
                }
                else if (event.keyCode === END) {
                    this._keyManager.setLastItemActive();
                    event.preventDefault();
                }
                else {
                    this._keyManager.onKeydown(event);
                }
                this.stateChanges.next();
            }
        }
        /**
         * Check the tab index as you should not be allowed to focus an empty list.
         * @protected
         * @return {?}
         */
        _updateTabIndex() {
            // If we have 0 chips, we should not allow keyboard focus
            this._tabIndex = this._userTabIndex || (this.chips.length === 0 ? -1 : 0);
        }
        /**
         * If the amount of chips changed, we need to update the
         * key manager state and focus the next closest chip.
         * @protected
         * @return {?}
         */
        _updateFocusForDestroyedChips() {
            // Move focus to the closest chip. If no other chips remain, focus the chip-list itself.
            if (this._lastDestroyedChipIndex != null) {
                if (this.chips.length) {
                    /** @type {?} */
                    const newChipIndex = Math.min(this._lastDestroyedChipIndex, this.chips.length - 1);
                    this._keyManager.setActiveItem(newChipIndex);
                }
                else {
                    this.focus();
                }
            }
            this._lastDestroyedChipIndex = null;
        }
        /**
         * Utility to ensure all indexes are valid.
         *
         * @private
         * @param {?} index The index to be checked.
         * @return {?} True if the index is valid for our list of chips.
         */
        _isValidIndex(index) {
            return index >= 0 && index < this.chips.length;
        }
        /**
         * @private
         * @param {?} element
         * @return {?}
         */
        _isInputEmpty(element) {
            if (element && element.nodeName.toLowerCase() === 'input') {
                /** @type {?} */
                let input = (/** @type {?} */ (element));
                return !input.value;
            }
            return false;
        }
        /**
         * @param {?} value
         * @param {?=} isUserInput
         * @return {?}
         */
        _setSelectionByValue(value, isUserInput = true) {
            this._clearSelection();
            this.chips.forEach((/**
             * @param {?} chip
             * @return {?}
             */
            chip => chip.deselect()));
            if (Array.isArray(value)) {
                value.forEach((/**
                 * @param {?} currentValue
                 * @return {?}
                 */
                currentValue => this._selectValue(currentValue, isUserInput)));
                this._sortValues();
            }
            else {
                /** @type {?} */
                const correspondingChip = this._selectValue(value, isUserInput);
                // Shift focus to the active item. Note that we shouldn't do this in multiple
                // mode, because we don't know what chip the user interacted with last.
                if (correspondingChip) {
                    if (isUserInput) {
                        this._keyManager.setActiveItem(correspondingChip);
                    }
                }
            }
        }
        /**
         * Finds and selects the chip based on its value.
         * @private
         * @param {?} value
         * @param {?=} isUserInput
         * @return {?} Chip that has the corresponding value.
         */
        _selectValue(value, isUserInput = true) {
            /** @type {?} */
            const correspondingChip = this.chips.find((/**
             * @param {?} chip
             * @return {?}
             */
            chip => {
                return chip.value != null && this._compareWith(chip.value, value);
            }));
            if (correspondingChip) {
                isUserInput ? correspondingChip.selectViaInteraction() : correspondingChip.select();
                this._selectionModel.select(correspondingChip);
            }
            return correspondingChip;
        }
        /**
         * @private
         * @return {?}
         */
        _initializeSelection() {
            // Defer setting the value in order to avoid the "Expression
            // has changed after it was checked" errors from Angular.
            Promise.resolve().then((/**
             * @return {?}
             */
            () => {
                if (this.ngControl || this._value) {
                    this._setSelectionByValue(this.ngControl ? this.ngControl.value : this._value, false);
                    this.stateChanges.next();
                }
            }));
        }
        /**
         * Deselects every chip in the list.
         * @private
         * @param {?=} skip Chip that should not be deselected.
         * @return {?}
         */
        _clearSelection(skip) {
            this._selectionModel.clear();
            this.chips.forEach((/**
             * @param {?} chip
             * @return {?}
             */
            chip => {
                if (chip !== skip) {
                    chip.deselect();
                }
            }));
            this.stateChanges.next();
        }
        /**
         * Sorts the model values, ensuring that they keep the same
         * order that they have in the panel.
         * @private
         * @return {?}
         */
        _sortValues() {
            if (this._multiple) {
                this._selectionModel.clear();
                this.chips.forEach((/**
                 * @param {?} chip
                 * @return {?}
                 */
                chip => {
                    if (chip.selected) {
                        this._selectionModel.select(chip);
                    }
                }));
                this.stateChanges.next();
            }
        }
        /**
         * Emits change event to set the model value.
         * @private
         * @param {?=} fallbackValue
         * @return {?}
         */
        _propagateChanges(fallbackValue) {
            /** @type {?} */
            let valueToEmit = null;
            if (Array.isArray(this.selected)) {
                valueToEmit = this.selected.map((/**
                 * @param {?} chip
                 * @return {?}
                 */
                chip => chip.value));
            }
            else {
                valueToEmit = this.selected ? this.selected.value : fallbackValue;
            }
            this._value = valueToEmit;
            this.change.emit(new MatChipListChange(this, valueToEmit));
            this.valueChange.emit(valueToEmit);
            this._onChange(valueToEmit);
            this._changeDetectorRef.markForCheck();
        }
        /**
         * When blurred, mark the field as touched when focus moved outside the chip list.
         * @return {?}
         */
        _blur() {
            if (!this._hasFocusedChip()) {
                this._keyManager.setActiveItem(-1);
            }
            if (!this.disabled) {
                if (this._chipInput) {
                    // If there's a chip input, we should check whether the focus moved to chip input.
                    // If the focus is not moved to chip input, mark the field as touched. If the focus moved
                    // to chip input, do nothing.
                    // Timeout is needed to wait for the focus() event trigger on chip input.
                    setTimeout((/**
                     * @return {?}
                     */
                    () => {
                        if (!this.focused) {
                            this._markAsTouched();
                        }
                    }));
                }
                else {
                    // If there's no chip input, then mark the field as touched.
                    this._markAsTouched();
                }
            }
        }
        /**
         * Mark the field as touched
         * @return {?}
         */
        _markAsTouched() {
            this._onTouched();
            this._changeDetectorRef.markForCheck();
            this.stateChanges.next();
        }
        /**
         * Removes the `tabindex` from the chip list and resets it back afterwards, allowing the
         * user to tab out of it. This prevents the list from capturing focus and redirecting
         * it back to the first chip, creating a focus trap, if it user tries to tab away.
         * @return {?}
         */
        _allowFocusEscape() {
            if (this._tabIndex !== -1) {
                this._tabIndex = -1;
                setTimeout((/**
                 * @return {?}
                 */
                () => {
                    this._tabIndex = this._userTabIndex || 0;
                    this._changeDetectorRef.markForCheck();
                }));
            }
        }
        /**
         * @private
         * @return {?}
         */
        _resetChips() {
            this._dropSubscriptions();
            this._listenToChipsFocus();
            this._listenToChipsSelection();
            this._listenToChipsRemoved();
        }
        /**
         * @private
         * @return {?}
         */
        _dropSubscriptions() {
            if (this._chipFocusSubscription) {
                this._chipFocusSubscription.unsubscribe();
                this._chipFocusSubscription = null;
            }
            if (this._chipBlurSubscription) {
                this._chipBlurSubscription.unsubscribe();
                this._chipBlurSubscription = null;
            }
            if (this._chipSelectionSubscription) {
                this._chipSelectionSubscription.unsubscribe();
                this._chipSelectionSubscription = null;
            }
            if (this._chipRemoveSubscription) {
                this._chipRemoveSubscription.unsubscribe();
                this._chipRemoveSubscription = null;
            }
        }
        /**
         * Listens to user-generated selection events on each chip.
         * @private
         * @return {?}
         */
        _listenToChipsSelection() {
            this._chipSelectionSubscription = this.chipSelectionChanges.subscribe((/**
             * @param {?} event
             * @return {?}
             */
            event => {
                event.source.selected
                    ? this._selectionModel.select(event.source)
                    : this._selectionModel.deselect(event.source);
                // For single selection chip list, make sure the deselected value is unselected.
                if (!this.multiple) {
                    this.chips.forEach((/**
                     * @param {?} chip
                     * @return {?}
                     */
                    chip => {
                        if (!this._selectionModel.isSelected(chip) && chip.selected) {
                            chip.deselect();
                        }
                    }));
                }
                if (event.isUserInput) {
                    this._propagateChanges();
                }
            }));
        }
        /**
         * Listens to user-generated selection events on each chip.
         * @private
         * @return {?}
         */
        _listenToChipsFocus() {
            this._chipFocusSubscription = this.chipFocusChanges.subscribe((/**
             * @param {?} event
             * @return {?}
             */
            event => {
                /** @type {?} */
                let chipIndex = this.chips.toArray().indexOf(event.chip);
                if (this._isValidIndex(chipIndex)) {
                    this._keyManager.updateActiveItem(chipIndex);
                }
                this.stateChanges.next();
            }));
            this._chipBlurSubscription = this.chipBlurChanges.subscribe((/**
             * @return {?}
             */
            () => {
                this._blur();
                this.stateChanges.next();
            }));
        }
        /**
         * @private
         * @return {?}
         */
        _listenToChipsRemoved() {
            this._chipRemoveSubscription = this.chipRemoveChanges.subscribe((/**
             * @param {?} event
             * @return {?}
             */
            event => {
                /** @type {?} */
                const chip = event.chip;
                /** @type {?} */
                const chipIndex = this.chips.toArray().indexOf(event.chip);
                // In case the chip that will be removed is currently focused, we temporarily store
                // the index in order to be able to determine an appropriate sibling chip that will
                // receive focus.
                if (this._isValidIndex(chipIndex) && chip._hasFocus) {
                    this._lastDestroyedChipIndex = chipIndex;
                }
            }));
        }
        /**
         * Checks whether an event comes from inside a chip element.
         * @private
         * @param {?} event
         * @return {?}
         */
        _originatesFromChip(event) {
            /** @type {?} */
            let currentElement = (/** @type {?} */ (event.target));
            while (currentElement && currentElement !== this._elementRef.nativeElement) {
                if (currentElement.classList.contains('mat-chip')) {
                    return true;
                }
                currentElement = currentElement.parentElement;
            }
            return false;
        }
        /**
         * Checks whether any of the chips is focused.
         * @private
         * @return {?}
         */
        _hasFocusedChip() {
            return this.chips.some((/**
             * @param {?} chip
             * @return {?}
             */
            chip => chip._hasFocus));
        }
        /**
         * Syncs the list's state with the individual chips.
         * @private
         * @return {?}
         */
        _syncChipsState() {
            if (this.chips) {
                this.chips.forEach((/**
                 * @param {?} chip
                 * @return {?}
                 */
                chip => {
                    chip._chipListDisabled = this._disabled;
                    chip._chipListMultiple = this.multiple;
                }));
            }
        }
    }
    MatChipList.ɵfac = function MatChipList_Factory(t) { return new (t || MatChipList)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(ChangeDetectorRef), ɵɵdirectiveInject(Directionality, 8), ɵɵdirectiveInject(NgForm, 8), ɵɵdirectiveInject(FormGroupDirective, 8), ɵɵdirectiveInject(ErrorStateMatcher), ɵɵdirectiveInject(NgControl, 10)); };
    MatChipList.ɵcmp = ɵɵdefineComponent({ type: MatChipList, selectors: [["mat-chip-list"]], contentQueries: function MatChipList_ContentQueries(rf, ctx, dirIndex) { if (rf & 1) {
            ɵɵcontentQuery(dirIndex, MatChip, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.chips = _t);
        } }, hostAttrs: [1, "mat-chip-list"], hostVars: 15, hostBindings: function MatChipList_HostBindings(rf, ctx) { if (rf & 1) {
            ɵɵlistener("focus", function MatChipList_focus_HostBindingHandler() { return ctx.focus(); })("blur", function MatChipList_blur_HostBindingHandler() { return ctx._blur(); })("keydown", function MatChipList_keydown_HostBindingHandler($event) { return ctx._keydown($event); });
        } if (rf & 2) {
            ɵɵhostProperty("id", ctx._uid);
            ɵɵattribute("tabindex", ctx.disabled ? null : ctx._tabIndex)("aria-describedby", ctx._ariaDescribedby || null)("aria-required", ctx.role ? ctx.required : null)("aria-disabled", ctx.disabled.toString())("aria-invalid", ctx.errorState)("aria-multiselectable", ctx.multiple)("role", ctx.role)("aria-orientation", ctx.ariaOrientation);
            ɵɵclassProp("mat-chip-list-disabled", ctx.disabled)("mat-chip-list-invalid", ctx.errorState)("mat-chip-list-required", ctx.required);
        } }, inputs: { ariaOrientation: ["aria-orientation", "ariaOrientation"], multiple: "multiple", compareWith: "compareWith", value: "value", required: "required", placeholder: "placeholder", disabled: "disabled", selectable: "selectable", tabIndex: "tabIndex", errorStateMatcher: "errorStateMatcher" }, outputs: { change: "change", valueChange: "valueChange" }, exportAs: ["matChipList"], features: [ɵɵProvidersFeature([{ provide: MatFormFieldControl, useExisting: MatChipList }]), ɵɵInheritDefinitionFeature], ngContentSelectors: _c0$u, decls: 2, vars: 0, consts: [[1, "mat-chip-list-wrapper"]], template: function MatChipList_Template(rf, ctx) { if (rf & 1) {
            ɵɵprojectionDef();
            ɵɵelementStart(0, "div", 0);
            ɵɵprojection(1);
            ɵɵelementEnd();
        } }, styles: [".mat-chip{position:relative;box-sizing:border-box;-webkit-tap-highlight-color:transparent;transform:translateZ(0);border:none;-webkit-appearance:none;-moz-appearance:none}.mat-standard-chip{transition:box-shadow 280ms cubic-bezier(0.4, 0, 0.2, 1);display:inline-flex;padding:7px 12px;border-radius:16px;align-items:center;cursor:default;min-height:32px;height:1px}._mat-animation-noopable.mat-standard-chip{transition:none;animation:none}.mat-standard-chip .mat-chip-remove.mat-icon{width:18px;height:18px}.mat-standard-chip::after{top:0;left:0;right:0;bottom:0;position:absolute;border-radius:inherit;opacity:0;content:\"\";pointer-events:none;transition:opacity 200ms cubic-bezier(0.35, 0, 0.25, 1)}.mat-standard-chip:hover::after{opacity:.12}.mat-standard-chip:focus{outline:none}.mat-standard-chip:focus::after{opacity:.16}.cdk-high-contrast-active .mat-standard-chip{outline:solid 1px}.cdk-high-contrast-active .mat-standard-chip:focus{outline:dotted 2px}.mat-standard-chip.mat-chip-disabled::after{opacity:0}.mat-standard-chip.mat-chip-disabled .mat-chip-remove,.mat-standard-chip.mat-chip-disabled .mat-chip-trailing-icon{cursor:default}.mat-standard-chip.mat-chip-with-trailing-icon.mat-chip-with-avatar,.mat-standard-chip.mat-chip-with-avatar{padding-top:0;padding-bottom:0}.mat-standard-chip.mat-chip-with-trailing-icon.mat-chip-with-avatar{padding-right:8px;padding-left:0}[dir=rtl] .mat-standard-chip.mat-chip-with-trailing-icon.mat-chip-with-avatar{padding-left:8px;padding-right:0}.mat-standard-chip.mat-chip-with-trailing-icon{padding-top:7px;padding-bottom:7px;padding-right:8px;padding-left:12px}[dir=rtl] .mat-standard-chip.mat-chip-with-trailing-icon{padding-left:8px;padding-right:12px}.mat-standard-chip.mat-chip-with-avatar{padding-left:0;padding-right:12px}[dir=rtl] .mat-standard-chip.mat-chip-with-avatar{padding-right:0;padding-left:12px}.mat-standard-chip .mat-chip-avatar{width:24px;height:24px;margin-right:8px;margin-left:4px}[dir=rtl] .mat-standard-chip .mat-chip-avatar{margin-left:8px;margin-right:4px}.mat-standard-chip .mat-chip-remove,.mat-standard-chip .mat-chip-trailing-icon{width:18px;height:18px;cursor:pointer}.mat-standard-chip .mat-chip-remove,.mat-standard-chip .mat-chip-trailing-icon{margin-left:8px;margin-right:0}[dir=rtl] .mat-standard-chip .mat-chip-remove,[dir=rtl] .mat-standard-chip .mat-chip-trailing-icon{margin-right:8px;margin-left:0}.mat-chip-ripple{top:0;left:0;right:0;bottom:0;position:absolute;pointer-events:none;border-radius:inherit;overflow:hidden}.mat-chip-list-wrapper{display:flex;flex-direction:row;flex-wrap:wrap;align-items:center;margin:-4px}.mat-chip-list-wrapper input.mat-input-element,.mat-chip-list-wrapper .mat-standard-chip{margin:4px}.mat-chip-list-stacked .mat-chip-list-wrapper{flex-direction:column;align-items:flex-start}.mat-chip-list-stacked .mat-chip-list-wrapper .mat-standard-chip{width:100%}.mat-chip-avatar{border-radius:50%;justify-content:center;align-items:center;display:flex;overflow:hidden;object-fit:cover}input.mat-chip-input{width:150px;margin:4px;flex:1 0 150px}\n"], encapsulation: 2, changeDetection: 0 });
    /** @nocollapse */
    MatChipList.ctorParameters = () => [
        { type: ElementRef },
        { type: ChangeDetectorRef },
        { type: Directionality, decorators: [{ type: Optional }] },
        { type: NgForm, decorators: [{ type: Optional }] },
        { type: FormGroupDirective, decorators: [{ type: Optional }] },
        { type: ErrorStateMatcher },
        { type: NgControl, decorators: [{ type: Optional }, { type: Self }] }
    ];
    MatChipList.propDecorators = {
        errorStateMatcher: [{ type: Input }],
        multiple: [{ type: Input }],
        compareWith: [{ type: Input }],
        value: [{ type: Input }],
        required: [{ type: Input }],
        placeholder: [{ type: Input }],
        disabled: [{ type: Input }],
        ariaOrientation: [{ type: Input, args: ['aria-orientation',] }],
        selectable: [{ type: Input }],
        tabIndex: [{ type: Input }],
        change: [{ type: Output }],
        valueChange: [{ type: Output }],
        chips: [{ type: ContentChildren, args: [MatChip, {
                        // We need to use `descendants: true`, because Ivy will no longer match
                        // indirect descendants if it's left as false.
                        descendants: true
                    },] }]
    };
    // Increasing integer for generating unique ids.
    /** @type {?} */
    let nextUniqueId$1$1 = 0;
    /**
     * Directive that adds chip-specific behaviors to an input element inside `<mat-form-field>`.
     * May be placed inside or outside of an `<mat-chip-list>`.
     */
    class MatChipInput {
        /**
         * @param {?} _elementRef
         * @param {?} _defaultOptions
         */
        constructor(_elementRef, _defaultOptions) {
            this._elementRef = _elementRef;
            this._defaultOptions = _defaultOptions;
            /**
             * Whether the control is focused.
             */
            this.focused = false;
            this._addOnBlur = false;
            /**
             * The list of key codes that will trigger a chipEnd event.
             *
             * Defaults to `[ENTER]`.
             */
            this.separatorKeyCodes = this._defaultOptions.separatorKeyCodes;
            /**
             * Emitted when a chip is to be added.
             */
            this.chipEnd = new EventEmitter();
            /**
             * The input's placeholder text.
             */
            this.placeholder = '';
            /**
             * Unique id for the input.
             */
            this.id = `mat-chip-list-input-${nextUniqueId$1$1++}`;
            this._disabled = false;
            this._inputElement = (/** @type {?} */ (this._elementRef.nativeElement));
        }
        /**
         * Register input for chip list
         * @param {?} value
         * @return {?}
         */
        set chipList(value) {
            if (value) {
                this._chipList = value;
                this._chipList.registerInput(this);
            }
        }
        /**
         * Whether or not the chipEnd event will be emitted when the input is blurred.
         * @return {?}
         */
        get addOnBlur() { return this._addOnBlur; }
        /**
         * @param {?} value
         * @return {?}
         */
        set addOnBlur(value) { this._addOnBlur = coerceBooleanProperty(value); }
        /**
         * Whether the input is disabled.
         * @return {?}
         */
        get disabled() { return this._disabled || (this._chipList && this._chipList.disabled); }
        /**
         * @param {?} value
         * @return {?}
         */
        set disabled(value) { this._disabled = coerceBooleanProperty(value); }
        /**
         * Whether the input is empty.
         * @return {?}
         */
        get empty() { return !this._inputElement.value; }
        /**
         * @return {?}
         */
        ngOnChanges() {
            this._chipList.stateChanges.next();
        }
        /**
         * Utility method to make host definition/tests more clear.
         * @param {?=} event
         * @return {?}
         */
        _keydown(event) {
            // Allow the user's focus to escape when they're tabbing forward. Note that we don't
            // want to do this when going backwards, because focus should go back to the first chip.
            if (event && event.keyCode === TAB && !hasModifierKey(event, 'shiftKey')) {
                this._chipList._allowFocusEscape();
            }
            this._emitChipEnd(event);
        }
        /**
         * Checks to see if the blur should emit the (chipEnd) event.
         * @return {?}
         */
        _blur() {
            if (this.addOnBlur) {
                this._emitChipEnd();
            }
            this.focused = false;
            // Blur the chip list if it is not focused
            if (!this._chipList.focused) {
                this._chipList._blur();
            }
            this._chipList.stateChanges.next();
        }
        /**
         * @return {?}
         */
        _focus() {
            this.focused = true;
            this._chipList.stateChanges.next();
        }
        /**
         * Checks to see if the (chipEnd) event needs to be emitted.
         * @param {?=} event
         * @return {?}
         */
        _emitChipEnd(event) {
            if (!this._inputElement.value && !!event) {
                this._chipList._keydown(event);
            }
            if (!event || this._isSeparatorKey(event)) {
                this.chipEnd.emit({ input: this._inputElement, value: this._inputElement.value });
                if (event) {
                    event.preventDefault();
                }
            }
        }
        /**
         * @return {?}
         */
        _onInput() {
            // Let chip list know whenever the value changes.
            this._chipList.stateChanges.next();
        }
        /**
         * Focuses the input.
         * @param {?=} options
         * @return {?}
         */
        focus(options) {
            this._inputElement.focus(options);
        }
        /**
         * Checks whether a keycode is one of the configured separators.
         * @private
         * @param {?} event
         * @return {?}
         */
        _isSeparatorKey(event) {
            if (hasModifierKey(event)) {
                return false;
            }
            /** @type {?} */
            const separators = this.separatorKeyCodes;
            /** @type {?} */
            const keyCode = event.keyCode;
            return Array.isArray(separators) ? separators.indexOf(keyCode) > -1 : separators.has(keyCode);
        }
    }
    MatChipInput.ɵfac = function MatChipInput_Factory(t) { return new (t || MatChipInput)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(MAT_CHIPS_DEFAULT_OPTIONS)); };
    MatChipInput.ɵdir = ɵɵdefineDirective({ type: MatChipInput, selectors: [["input", "matChipInputFor", ""]], hostAttrs: [1, "mat-chip-input", "mat-input-element"], hostVars: 5, hostBindings: function MatChipInput_HostBindings(rf, ctx) { if (rf & 1) {
            ɵɵlistener("keydown", function MatChipInput_keydown_HostBindingHandler($event) { return ctx._keydown($event); })("blur", function MatChipInput_blur_HostBindingHandler() { return ctx._blur(); })("focus", function MatChipInput_focus_HostBindingHandler() { return ctx._focus(); })("input", function MatChipInput_input_HostBindingHandler() { return ctx._onInput(); });
        } if (rf & 2) {
            ɵɵhostProperty("id", ctx.id);
            ɵɵattribute("disabled", ctx.disabled || null)("placeholder", ctx.placeholder || null)("aria-invalid", ctx._chipList && ctx._chipList.ngControl ? ctx._chipList.ngControl.invalid : null)("aria-required", ctx._chipList && ctx._chipList.required || null);
        } }, inputs: { separatorKeyCodes: ["matChipInputSeparatorKeyCodes", "separatorKeyCodes"], placeholder: "placeholder", id: "id", chipList: ["matChipInputFor", "chipList"], addOnBlur: ["matChipInputAddOnBlur", "addOnBlur"], disabled: "disabled" }, outputs: { chipEnd: "matChipInputTokenEnd" }, exportAs: ["matChipInput", "matChipInputFor"], features: [ɵɵNgOnChangesFeature] });
    /** @nocollapse */
    MatChipInput.ctorParameters = () => [
        { type: ElementRef },
        { type: undefined, decorators: [{ type: Inject, args: [MAT_CHIPS_DEFAULT_OPTIONS,] }] }
    ];
    MatChipInput.propDecorators = {
        chipList: [{ type: Input, args: ['matChipInputFor',] }],
        addOnBlur: [{ type: Input, args: ['matChipInputAddOnBlur',] }],
        separatorKeyCodes: [{ type: Input, args: ['matChipInputSeparatorKeyCodes',] }],
        chipEnd: [{ type: Output, args: ['matChipInputTokenEnd',] }],
        placeholder: [{ type: Input }],
        id: [{ type: Input }],
        disabled: [{ type: Input }]
    };
    const ɵ0$h = ({
        separatorKeyCodes: [ENTER]
    });
    class MatChipsModule {
    }
    MatChipsModule.ɵmod = ɵɵdefineNgModule({ type: MatChipsModule });
    MatChipsModule.ɵinj = ɵɵdefineInjector({ factory: function MatChipsModule_Factory(t) { return new (t || MatChipsModule)(); }, providers: [
            ErrorStateMatcher,
            {
                provide: MAT_CHIPS_DEFAULT_OPTIONS,
                useValue: ( /** @type {?} */(ɵ0$h))
            }
        ] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(MatChipsModule, { declarations: [MatChipList,
            MatChip,
            MatChipInput,
            MatChipRemove,
            MatChipAvatar,
            MatChipTrailingIcon], exports: [MatChipList,
            MatChip,
            MatChipInput,
            MatChipRemove,
            MatChipAvatar,
            MatChipTrailingIcon] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/data_selection/metric_arithmetic/metric_arithmetic_element/metric_arithmetic_element_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    const _c0$v = function (a0) { return { "width": a0 }; };
    const _c1$j = function (a0) { return { "value-invalid": a0 }; };
    class MetricArithmeticElementComponent {
        constructor() {
            this.onRemove = new EventEmitter();
            this.onFilterChange = new EventEmitter();
            this.focusMin = false;
            this.focusMax = false;
            this.ngUnsubscribe = new Subject();
        }
        /**
         * @return {?}
         */
        ngOnInit() {
            this.minFormControl = new FormControl(this.filterValues.min, [
                Validators.required,
                Validators.min(-1.0),
                Validators.max(1.0),
                this.minValueValidator.bind(this),
            ]);
            this.maxFormControl = new FormControl(this.filterValues.max, [
                Validators.required,
                Validators.min(-1.0),
                Validators.max(1.0),
                this.maxValueValidator.bind(this),
            ]);
            this.minFormControl.valueChanges
                .pipe(takeUntil(this.ngUnsubscribe))
                .subscribe((/**
             * @return {?}
             */
            () => {
                if (this.minFormControl.valid && this.maxFormControl.valid) {
                    this.onFilterChange.emit({
                        min: parseFloat(this.minFormControl.value),
                        max: parseFloat(this.maxFormControl.value),
                    });
                }
            }));
            this.maxFormControl.valueChanges
                .pipe(takeUntil(this.ngUnsubscribe))
                .subscribe((/**
             * @return {?}
             */
            () => {
                if (this.minFormControl.valid && this.maxFormControl.valid) {
                    this.onFilterChange.emit({
                        min: parseFloat(this.minFormControl.value),
                        max: parseFloat(this.maxFormControl.value),
                    });
                }
            }));
        }
        /**
         * @param {?} changes
         * @return {?}
         */
        ngOnChanges(changes) {
            if (this.minFormControl && this.maxFormControl) {
                this.minFormControl.setValue(this.filterValues.min, { emitEvent: false });
                this.maxFormControl.setValue(this.filterValues.max, { emitEvent: false });
            }
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this.ngUnsubscribe.next();
            this.ngUnsubscribe.complete();
        }
        /**
         * @this {?}
         * @param {?} control
         * @return {?}
         */
        minValueValidator(control) {
            if (!this.maxFormControl || control.value === 'NaN') {
                return null;
            }
            else if (isNaN(parseFloat(control.value))) {
                return { value: 'the string you entered is neither NaN nor a number' };
            }
            else if (parseFloat(control.value) > parseFloat(this.maxFormControl.value)) {
                return { value: 'the number you entered is larger than the max value' };
            }
            return null;
        }
        /**
         * @this {?}
         * @param {?} control
         * @return {?}
         */
        maxValueValidator(control) {
            if (!this.minFormControl) {
                return null;
            }
            else if (
            // Max NaN only if min also NaN
            this.minFormControl.value === 'NaN' &&
                control.value === 'NaN') {
                return null;
            }
            else if (isNaN(parseFloat(control.value))) {
                return { value: 'the string you entered is neither NaN nor a number' };
            }
            else if (control.value < this.minFormControl.value) {
                return { value: 'the number you entered is smaller than the min value' };
            }
            return null;
        }
        /**
         * @param {?} errors
         * @return {?}
         */
        getErrorDescription(errors) {
            if (errors) {
                /** @type {?} */
                const firstKey = Object.keys(errors)[0];
                if (firstKey === 'required') {
                    return 'you did not enter anything';
                }
                else if (firstKey === 'min') {
                    return 'the number must be at least -1.0';
                }
                else if (firstKey === 'max') {
                    return 'the number is bigger than 1.0';
                }
                return errors[firstKey];
            }
            return '';
        }
    }
    MetricArithmeticElementComponent.decorators = [
        { type: Component, args: [{
                    selector: 'metric-arithmetic-element-component',
                    templateUrl: './metric_arithmetic_element_component.ng.html',
                    styleUrls: ['./metric_arithmetic_element_component.css'],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    MetricArithmeticElementComponent.propDecorators = {
        metric: [{ type: Input }],
        filterValues: [{ type: Input }],
        onRemove: [{ type: Output }],
        onFilterChange: [{ type: Output }]
    };
    /** @nocollapse */ MetricArithmeticElementComponent.ɵfac = function MetricArithmeticElementComponent_Factory(t) { return new (t || MetricArithmeticElementComponent)(); };
    /** @nocollapse */ MetricArithmeticElementComponent.ɵcmp = ɵɵdefineComponent({ type: MetricArithmeticElementComponent, selectors: [["metric-arithmetic-element-component"]], inputs: { metric: "metric", filterValues: "filterValues" }, outputs: { onRemove: "onRemove", onFilterChange: "onFilterChange" }, features: [ɵɵNgOnChangesFeature], decls: 9, vars: 21, consts: [[1, "filter-chip", 3, "removed"], [1, "metric-arithmetic-element-range", 3, "keydown"], ["matInput", "", 1, "input-field", 3, "value", "matTooltip", "matTooltipDisabled", "ngStyle", "ngClass", "formControl", "focus", "focusout"], ["matChipRemove", "", "svgIcon", "cancel_24px"]], template: function MetricArithmeticElementComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "mat-chip", 0);
            ɵɵlistener("removed", function MetricArithmeticElementComponent_Template_mat_chip_removed_0_listener() { return ctx.onRemove.emit(ctx.metric); });
            ɵɵtext(1);
            ɵɵelementStart(2, "div", 1);
            ɵɵlistener("keydown", function MetricArithmeticElementComponent_Template_div_keydown_2_listener($event) { return $event.stopPropagation(); });
            ɵɵtext(3, " [ ");
            ɵɵelementStart(4, "input", 2);
            ɵɵlistener("focus", function MetricArithmeticElementComponent_Template_input_focus_4_listener() { return ctx.focusMin = true; })("focusout", function MetricArithmeticElementComponent_Template_input_focusout_4_listener() { return ctx.focusMin = false; });
            ɵɵelementEnd();
            ɵɵtext(5, " ; ");
            ɵɵelementStart(6, "input", 2);
            ɵɵlistener("focus", function MetricArithmeticElementComponent_Template_input_focus_6_listener() { return ctx.focusMax = true; })("focusout", function MetricArithmeticElementComponent_Template_input_focusout_6_listener() { return ctx.focusMax = false; });
            ɵɵelementEnd();
            ɵɵtext(7, " ] ");
            ɵɵelementEnd();
            ɵɵelement(8, "mat-icon", 3);
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵadvance(1);
            ɵɵtextInterpolate1(" ", ctx.metric, " ");
            ɵɵadvance(3);
            ɵɵproperty("value", ctx.filterValues.min)("matTooltip", ctx.getErrorDescription(ctx.minFormControl.errors))("matTooltipDisabled", !ctx.minFormControl.invalid)("ngStyle", ɵɵpureFunction1(13, _c0$v, ctx.focusMin ? "100px" : ctx.minFormControl.value.toString().length + "ch"))("ngClass", ɵɵpureFunction1(15, _c1$j, !ctx.minFormControl.valid))("formControl", ctx.minFormControl);
            ɵɵadvance(2);
            ɵɵproperty("value", ctx.filterValues.max)("matTooltip", ctx.getErrorDescription(ctx.maxFormControl.errors))("matTooltipDisabled", !ctx.maxFormControl.invalid)("ngStyle", ɵɵpureFunction1(17, _c0$v, ctx.focusMax ? "100px" : ctx.maxFormControl.value.toString().length + "ch"))("ngClass", ɵɵpureFunction1(19, _c1$j, !ctx.maxFormControl.valid))("formControl", ctx.maxFormControl);
        } }, directives: [MatChip, DefaultValueAccessor, MatTooltip, NgStyle, NgClass, NgControlStatus, FormControlDirective, MatIcon, MatChipRemove], styles: [".mat-badge-content[_ngcontent-%COMP%]{font-weight:600;font-size:12px;font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:9px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:24px}.mat-h1[_ngcontent-%COMP%], .mat-headline[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%]{font:400 24px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h2[_ngcontent-%COMP%], .mat-title[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h3[_ngcontent-%COMP%], .mat-subheading-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%]{font:400 16px/28px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h4[_ngcontent-%COMP%], .mat-subheading-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%]{font:400 15px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h5[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%]{font:400 calc(14px * 0.83)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-h6[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:400 calc(14px * 0.67)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-body-strong[_ngcontent-%COMP%], .mat-body-2[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   p[_ngcontent-%COMP%]{margin:0 0 12px}.mat-small[_ngcontent-%COMP%], .mat-caption[_ngcontent-%COMP%]{font:400 12px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-display-4[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-4[_ngcontent-%COMP%]{font:300 112px/112px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.05em;margin:0 0 56px}.mat-display-3[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-3[_ngcontent-%COMP%]{font:400 56px/56px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.02em;margin:0 0 64px}.mat-display-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-2[_ngcontent-%COMP%]{font:400 45px/48px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.005em;margin:0 0 64px}.mat-display-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-1[_ngcontent-%COMP%]{font:400 34px/40px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 64px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%], .mat-flat-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-button-toggle[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card-title[_ngcontent-%COMP%]{font-size:24px;font-weight:500}.mat-card-header[_ngcontent-%COMP%]   .mat-card-title[_ngcontent-%COMP%]{font-size:20px}.mat-card-subtitle[_ngcontent-%COMP%], .mat-card-content[_ngcontent-%COMP%]{font-size:14px}.mat-checkbox[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-checkbox-layout[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{line-height:24px}.mat-chip[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-chip[_ngcontent-%COMP%]   .mat-chip-trailing-icon.mat-icon[_ngcontent-%COMP%], .mat-chip[_ngcontent-%COMP%]   .mat-chip-remove.mat-icon[_ngcontent-%COMP%]{font-size:18px}.mat-table[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-header-cell[_ngcontent-%COMP%]{font-size:12px;font-weight:500}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{font-size:14px}.mat-calendar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-calendar-body[_ngcontent-%COMP%]{font-size:13px}.mat-calendar-body-label[_ngcontent-%COMP%], .mat-calendar-period-button[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-calendar-table-header[_ngcontent-%COMP%]   th[_ngcontent-%COMP%]{font-size:11px;font-weight:400}.mat-dialog-title[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-expansion-panel-header[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:15px;font-weight:400}.mat-expansion-panel-content[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field[_ngcontent-%COMP%]{font-size:inherit;font-weight:400;line-height:1.125;font-family:Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.34375em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{font-size:150%;line-height:1.125}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]{height:1.5em;width:1.5em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{height:1.125em;line-height:1.125}.mat-form-field-infix[_ngcontent-%COMP%]{padding:.5em 0;border-top:.84375em solid transparent}.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34375em) scale(0.75);width:133.3333333333%}.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34374em) scale(0.75);width:133.3333433333%}.mat-form-field-label-wrapper[_ngcontent-%COMP%]{top:-0.84375em;padding-top:.84375em}.mat-form-field-label[_ngcontent-%COMP%]{top:1.34375em}.mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.34375em}.mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{font-size:75%;margin-top:.6666666667em;top:calc(100% - 1.7916666667em)}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.4375em 0}.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.001px);-ms-transform:translateY(-1.28125em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00101px);-ms-transform:translateY(-1.28124em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00102px);-ms-transform:translateY(-1.28123em) scale(0.75);width:133.3333533333%}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.28125em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{margin-top:.5416666667em;top:calc(100% - 1.6666666667em)}@media print{.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28122em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28121em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.2812em) scale(0.75)}}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.25em 0 .75em 0}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.09375em;margin-top:-0.5em}.mat-form-field-appearance-fill.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59374em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:1em 0 1em 0}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.84375em;margin-top:-0.25em}.mat-form-field-appearance-outline.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59374em) scale(0.75);width:133.3333433333%}.mat-grid-tile-header[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]{font-size:14px}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2), .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}input.mat-input-element[_ngcontent-%COMP%]{margin-top:-0.0625em}.mat-menu-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:400}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px}.mat-radio-button[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select-trigger[_ngcontent-%COMP%]{height:1.125em}.mat-slide-toggle-content[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-slider-thumb-label-text[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-stepper-vertical[_ngcontent-%COMP%], .mat-stepper-horizontal[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-step-label[_ngcontent-%COMP%]{font-size:14px;font-weight:400}.mat-step-sub-label-error[_ngcontent-%COMP%]{font-weight:normal}.mat-step-label-error[_ngcontent-%COMP%]{font-size:14px}.mat-step-label-selected[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-tab-group[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-toolbar[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0}.mat-tooltip[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:10px;padding-top:6px;padding-bottom:6px}.mat-tooltip-handset[_ngcontent-%COMP%]{font-size:14px;padding-top:8px;padding-bottom:8px}.mat-list-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:16px}.mat-optgroup-label[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-simple-snackbar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px}.mat-simple-snackbar-action[_ngcontent-%COMP%]{line-height:1;font-family:inherit;font-size:inherit;font-weight:500}.mat-tree[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{font-weight:400;font-size:14px}.mat-ripple[_ngcontent-%COMP%]{overflow:hidden;position:relative}.mat-ripple[_ngcontent-%COMP%]:not(:empty){transform:translateZ(0)}.mat-ripple.mat-ripple-unbounded[_ngcontent-%COMP%]{overflow:visible}.mat-ripple-element[_ngcontent-%COMP%]{position:absolute;border-radius:50%;pointer-events:none;transition:opacity,transform 0ms cubic-bezier(0, 0, 0.2, 1);transform:scale(0)}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{display:none}.cdk-visually-hidden[_ngcontent-%COMP%]{border:0;clip:rect(0 0 0 0);height:1px;margin:-1px;overflow:hidden;padding:0;position:absolute;width:1px;outline:0;-webkit-appearance:none;-moz-appearance:none}.cdk-overlay-container[_ngcontent-%COMP%], .cdk-global-overlay-wrapper[_ngcontent-%COMP%]{pointer-events:none;top:0;left:0;height:100%;width:100%}.cdk-overlay-container[_ngcontent-%COMP%]{position:fixed;z-index:1000}.cdk-overlay-container[_ngcontent-%COMP%]:empty{display:none}.cdk-global-overlay-wrapper[_ngcontent-%COMP%]{display:flex;position:absolute;z-index:1000}.cdk-overlay-pane[_ngcontent-%COMP%]{position:absolute;pointer-events:auto;box-sizing:border-box;z-index:1000;display:flex;max-width:100%;max-height:100%}.cdk-overlay-backdrop[_ngcontent-%COMP%]{position:absolute;top:0;bottom:0;left:0;right:0;z-index:1000;pointer-events:auto;-webkit-tap-highlight-color:transparent;transition:opacity 400ms cubic-bezier(0.25, 0.8, 0.25, 1);opacity:0}.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:1}@media screen and (-ms-high-contrast: active){.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:.6}}.cdk-overlay-dark-backdrop[_ngcontent-%COMP%]{background:rgba(0,0,0,.32)}.cdk-overlay-transparent-backdrop[_ngcontent-%COMP%], .cdk-overlay-transparent-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:0}.cdk-overlay-connected-position-bounding-box[_ngcontent-%COMP%]{position:absolute;z-index:1000;display:flex;flex-direction:column;min-width:1px;min-height:1px}.cdk-global-scrollblock[_ngcontent-%COMP%]{position:fixed;width:100%;overflow-y:scroll}@keyframes cdk-text-field-autofill-start{}@keyframes cdk-text-field-autofill-end{}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:-webkit-autofill{animation:cdk-text-field-autofill-start 0s 1ms}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:not(:-webkit-autofill){animation:cdk-text-field-autofill-end 0s 1ms}textarea.cdk-textarea-autosize[_ngcontent-%COMP%]{resize:none}textarea.cdk-textarea-autosize-measuring[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:auto !important;overflow:hidden !important}textarea.cdk-textarea-autosize-measuring-firefox[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:0 !important}.mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.1)}.mat-option[_ngcontent-%COMP%]{color:#212121}.mat-option[_ngcontent-%COMP%]:hover:not(.mat-option-disabled), .mat-option[_ngcontent-%COMP%]:focus:not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple):not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-active[_ngcontent-%COMP%]{background:rgba(0,0,0,.04);color:#212121}.mat-option.mat-option-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-primary[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f57c00}.mat-accent[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f44336}.mat-optgroup-label[_ngcontent-%COMP%]{color:#616161}.mat-optgroup-disabled[_ngcontent-%COMP%]   .mat-optgroup-label[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-pseudo-checkbox[_ngcontent-%COMP%]{color:#616161}.mat-pseudo-checkbox[_ngcontent-%COMP%]::after{color:#fafafa}.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{color:#b0b0b0}.mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f57c00}.mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f44336}.mat-pseudo-checkbox-checked.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{background:#b0b0b0}.mat-elevation-z0[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z1[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z2[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z3[_ngcontent-%COMP%]{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z4[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z5[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 5px 8px 0px rgba(0, 0, 0, 0.14),0px 1px 14px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z6[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z7[_ngcontent-%COMP%]{box-shadow:0px 4px 5px -2px rgba(0, 0, 0, 0.2),0px 7px 10px 1px rgba(0, 0, 0, 0.14),0px 2px 16px 1px rgba(0, 0, 0, 0.12)}.mat-elevation-z8[_ngcontent-%COMP%]{box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z9[_ngcontent-%COMP%]{box-shadow:0px 5px 6px -3px rgba(0, 0, 0, 0.2),0px 9px 12px 1px rgba(0, 0, 0, 0.14),0px 3px 16px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z10[_ngcontent-%COMP%]{box-shadow:0px 6px 6px -3px rgba(0, 0, 0, 0.2),0px 10px 14px 1px rgba(0, 0, 0, 0.14),0px 4px 18px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z11[_ngcontent-%COMP%]{box-shadow:0px 6px 7px -4px rgba(0, 0, 0, 0.2),0px 11px 15px 1px rgba(0, 0, 0, 0.14),0px 4px 20px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z12[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z13[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 13px 19px 2px rgba(0, 0, 0, 0.14),0px 5px 24px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z14[_ngcontent-%COMP%]{box-shadow:0px 7px 9px -4px rgba(0, 0, 0, 0.2),0px 14px 21px 2px rgba(0, 0, 0, 0.14),0px 5px 26px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z15[_ngcontent-%COMP%]{box-shadow:0px 8px 9px -5px rgba(0, 0, 0, 0.2),0px 15px 22px 2px rgba(0, 0, 0, 0.14),0px 6px 28px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z16[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z17[_ngcontent-%COMP%]{box-shadow:0px 8px 11px -5px rgba(0, 0, 0, 0.2),0px 17px 26px 2px rgba(0, 0, 0, 0.14),0px 6px 32px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z18[_ngcontent-%COMP%]{box-shadow:0px 9px 11px -5px rgba(0, 0, 0, 0.2),0px 18px 28px 2px rgba(0, 0, 0, 0.14),0px 7px 34px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z19[_ngcontent-%COMP%]{box-shadow:0px 9px 12px -6px rgba(0, 0, 0, 0.2),0px 19px 29px 2px rgba(0, 0, 0, 0.14),0px 7px 36px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z20[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 20px 31px 3px rgba(0, 0, 0, 0.14),0px 8px 38px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z21[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 21px 33px 3px rgba(0, 0, 0, 0.14),0px 8px 40px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z22[_ngcontent-%COMP%]{box-shadow:0px 10px 14px -6px rgba(0, 0, 0, 0.2),0px 22px 35px 3px rgba(0, 0, 0, 0.14),0px 8px 42px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z23[_ngcontent-%COMP%]{box-shadow:0px 11px 14px -7px rgba(0, 0, 0, 0.2),0px 23px 36px 3px rgba(0, 0, 0, 0.14),0px 9px 44px 8px rgba(0, 0, 0, 0.12)}.mat-elevation-z24[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12)}.mat-app-background[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-theme-loaded-marker[_ngcontent-%COMP%]{display:none}.mat-autocomplete-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-autocomplete-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover){background:#fff}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover):not(.mat-option-disabled){color:#212121}.mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f57c00}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{outline:solid 1px;border-radius:0}.mat-badge-accent[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-badge-warn[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f44336}.mat-badge[_ngcontent-%COMP%]{position:relative}.mat-badge-hidden[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{display:none}.mat-badge-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#b9b9b9;color:#757575}.mat-badge-content[_ngcontent-%COMP%]{position:absolute;text-align:center;display:inline-block;border-radius:50%;transition:transform 200ms ease-in-out;transform:scale(0.6);overflow:hidden;white-space:nowrap;text-overflow:ellipsis;pointer-events:none}.ng-animate-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%], .mat-badge-content._mat-animation-noopable[_ngcontent-%COMP%]{transition:none}.mat-badge-content.mat-badge-active[_ngcontent-%COMP%]{transform:none}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:16px;height:16px;line-height:16px}.mat-badge-small.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-8px}.mat-badge-small.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-8px}.mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-16px}.mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-16px}.mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-8px}.mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-8px}.mat-badge-medium[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:22px;height:22px;line-height:22px}.mat-badge-medium.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-11px}.mat-badge-medium.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-11px}.mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-22px}.mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-22px}.mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-11px}.mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-11px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:28px;height:28px;line-height:28px}.mat-badge-large.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-14px}.mat-badge-large.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-14px}.mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-28px}.mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-28px}.mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-14px}.mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-14px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]{color:inherit;background:transparent}.mat-button.mat-primary[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-button[disabled][disabled][_ngcontent-%COMP%], .mat-icon-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-icon-button[disabled][disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-stroked-button[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f44336}.mat-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:transparent}.mat-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{opacity:.1;background-color:currentColor}.mat-button-focus-overlay[_ngcontent-%COMP%]{background:#000}.mat-stroked-button[_ngcontent-%COMP%]:not([disabled]){border-color:rgba(0,0,0,.12)}.mat-flat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{color:#212121;background-color:#fff}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-flat-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-stroked-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-flat-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-raised-button[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone[_ngcontent-%COMP%], .mat-button-toggle-group[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{box-shadow:none}.mat-button-toggle[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-button-toggle[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121;background:#fff}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:#000}.mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12)}[dir=rtl][_ngcontent-%COMP%]   .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-group-appearance-standard.mat-button-toggle-vertical[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:none;border-top:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#616161}.mat-button-toggle-checked.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121}.mat-button-toggle-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.26);background-color:#eee}.mat-button-toggle-disabled.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{background:#fff}.mat-button-toggle-disabled.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#bdbdbd}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{border:solid 1px rgba(0,0,0,.12)}.mat-card[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-card[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-card.mat-card-flat[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-card-subtitle[_ngcontent-%COMP%]{color:#616161}.mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#616161}.mat-checkbox-checkmark[_ngcontent-%COMP%]{fill:#fafafa}.mat-checkbox-checkmark-path[_ngcontent-%COMP%]{stroke:#fafafa !important}.mat-checkbox-mixedmark[_ngcontent-%COMP%]{background-color:#fafafa}.mat-checkbox-indeterminate.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f57c00}.mat-checkbox-indeterminate.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#ff9800}.mat-checkbox-indeterminate.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f44336}.mat-checkbox-disabled.mat-checkbox-checked[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-disabled.mat-checkbox-indeterminate[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]:not(.mat-checkbox-checked)   .mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{color:#616161}.mat-checkbox[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%]{background:#f57c00}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%]{background:#ff9800}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%]{background:#f44336}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#212121}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#212121;opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled):active{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled)   .mat-chip-remove[_ngcontent-%COMP%]:hover{opacity:.54}.mat-chip.mat-standard-chip.mat-chip-disabled[_ngcontent-%COMP%]{opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]::after{background:#000}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-table[_ngcontent-%COMP%]{background:#fff}.mat-table[_ngcontent-%COMP%]   thead[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tbody[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tfoot[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], [mat-header-row][_ngcontent-%COMP%], [mat-row][_ngcontent-%COMP%], [mat-footer-row][_ngcontent-%COMP%], .mat-table-sticky[_ngcontent-%COMP%]{background:inherit}mat-row[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], th.mat-header-cell[_ngcontent-%COMP%], td.mat-cell[_ngcontent-%COMP%], td.mat-footer-cell[_ngcontent-%COMP%]{border-bottom-color:rgba(0,0,0,.12)}.mat-header-cell[_ngcontent-%COMP%]{color:#616161}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{color:#212121}.mat-calendar-arrow[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.54)}.mat-datepicker-toggle[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-next-button[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-previous-button[_ngcontent-%COMP%]{color:rgba(0,0,0,.54)}.mat-calendar-table-header[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-calendar-table-header-divider[_ngcontent-%COMP%]::after{background:rgba(0,0,0,.12)}.mat-calendar-body-label[_ngcontent-%COMP%]{color:#616161}.mat-calendar-body-cell-content[_ngcontent-%COMP%]{color:#212121;border-color:transparent}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){color:#757575}.mat-calendar-body-cell[_ngcontent-%COMP%]:not(.mat-calendar-body-disabled):hover > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-keyboard-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-program-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){background-color:rgba(0,0,0,.04)}.mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.38)}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.18)}.mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.4)}.mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12);background-color:#fff;color:#212121}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.4)}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.4)}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content-touch[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-datepicker-toggle-active[_ngcontent-%COMP%]{color:#f57c00}.mat-datepicker-toggle-active.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-datepicker-toggle-active.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-dialog-container[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-divider[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-divider-vertical[_ngcontent-%COMP%]{border-right-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-expansion-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-action-row[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-keyboard-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-program-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded)   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover:not([aria-disabled=true]){background:rgba(0,0,0,.04)}@media(hover: none){.mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded):not([aria-disabled=true])   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover{background:#fff}}.mat-expansion-panel-header-title[_ngcontent-%COMP%]{color:#212121}.mat-expansion-panel-header-description[_ngcontent-%COMP%], .mat-expansion-indicator[_ngcontent-%COMP%]::after{color:#616161}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-title[_ngcontent-%COMP%], .mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-description[_ngcontent-%COMP%]{color:inherit}.mat-form-field-label[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-hint[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-focused[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-ripple[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid)   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f57c00}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-accent   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#ff9800}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-warn   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#f44336}.mat-error[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-hint[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-legacy.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-standard[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-standard.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.04)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.02)}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:transparent}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.12)}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:rgba(0,0,0,.87)}.mat-form-field-appearance-outline.mat-focused[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field-appearance-outline.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-appearance-outline.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-invalid.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.06)}.mat-icon.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-icon.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-icon.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-type-mat-native-select[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#616161}.mat-input-element[_ngcontent-%COMP%]:disabled, .mat-form-field-type-mat-native-select.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#757575}.mat-input-element[_ngcontent-%COMP%]{caret-color:#f57c00}.mat-input-element[_ngcontent-%COMP%]::placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-moz-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-webkit-input-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]:-ms-input-placeholder{color:rgba(97,97,97,.42)}.mat-accent[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%], .mat-form-field-invalid[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#f44336}.mat-form-field-type-mat-native-select.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{color:#616161}.mat-list-item-disabled[_ngcontent-%COMP%]{background-color:#eee}.mat-list-option[_ngcontent-%COMP%]:hover, .mat-list-option[_ngcontent-%COMP%]:focus, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.04)}.mat-list-single-selected-option[_ngcontent-%COMP%], .mat-list-single-selected-option[_ngcontent-%COMP%]:hover, .mat-list-single-selected-option[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.12)}.mat-menu-panel[_ngcontent-%COMP%]{background:#fff}.mat-menu-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-menu-item[_ngcontent-%COMP%]{background:transparent;color:#212121}.mat-menu-item[disabled][_ngcontent-%COMP%], .mat-menu-item[disabled][_ngcontent-%COMP%]::after{color:rgba(0,0,0,.38)}.mat-menu-item[_ngcontent-%COMP%]   .mat-icon-no-color[_ngcontent-%COMP%], .mat-menu-item-submenu-trigger[_ngcontent-%COMP%]::after{color:rgba(0,0,0,.54)}.mat-menu-item[_ngcontent-%COMP%]:hover:not([disabled]), .mat-menu-item.cdk-program-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item.cdk-keyboard-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item-highlighted[_ngcontent-%COMP%]:not([disabled]){background:rgba(0,0,0,.04)}.mat-paginator[_ngcontent-%COMP%]{background:#fff}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{color:#616161}.mat-paginator-decrement[_ngcontent-%COMP%], .mat-paginator-increment[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54);border-right:2px solid rgba(0,0,0,.54)}.mat-paginator-first[_ngcontent-%COMP%], .mat-paginator-last[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54)}.mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-decrement[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-increment[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-first[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-last[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f57c00}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#ff9800}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f44336}.mat-progress-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f57c00}.mat-progress-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#ff9800}.mat-progress-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f44336}.mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#616161}.mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f57c00}.mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#ff9800}.mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#ff9800}.mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f44336}.mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f44336}.mat-radio-button.mat-radio-disabled.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-label-content[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-radio-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-select-value[_ngcontent-%COMP%]{color:#212121}.mat-select-placeholder[_ngcontent-%COMP%]{color:rgba(97,97,97,.42)}.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%]{color:#757575}.mat-select-arrow[_ngcontent-%COMP%]{color:#616161}.mat-select-panel[_ngcontent-%COMP%]{background:#fff}.mat-select-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-select-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple){background:rgba(0,0,0,.12)}.mat-form-field.mat-focused.mat-primary[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-invalid[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#757575}.mat-drawer-container[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-drawer[_ngcontent-%COMP%]{background-color:#fff;color:#212121}.mat-drawer.mat-drawer-push[_ngcontent-%COMP%]{background-color:#fff}.mat-drawer[_ngcontent-%COMP%]:not(.mat-drawer-side){box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-drawer-side[_ngcontent-%COMP%]{border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-backdrop.mat-drawer-shown[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.6)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.54)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.54)}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.54)}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle[_ngcontent-%COMP%]:not(.mat-checked)   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-slide-toggle-thumb[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12);background-color:#fafafa}.mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-primary[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f57c00}.mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-accent[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#ff9800}.mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-warn[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f44336}.mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.2)}.mat-slider[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%], .cdk-focused[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-disabled[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]   .mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing)   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26);background-color:transparent}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26)}.mat-slider-has-ticks[_ngcontent-%COMP%]   .mat-slider-wrapper[_ngcontent-%COMP%]::after{border-color:rgba(0,0,0,.7)}.mat-slider-horizontal[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to right, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent);background-image:-moz-repeating-linear-gradient(0.0001deg, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-slider-vertical[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to bottom, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-step-header.cdk-keyboard-focused[_ngcontent-%COMP%], .mat-step-header.cdk-program-focused[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]:hover{background-color:rgba(0,0,0,.04)}@media(hover: none){.mat-step-header[_ngcontent-%COMP%]:hover{background:none}}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-optional[_ngcontent-%COMP%]{color:#616161}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon[_ngcontent-%COMP%]{background-color:#616161;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-selected[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-done[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-edit[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-error[_ngcontent-%COMP%]{background-color:transparent;color:#f44336}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-active[_ngcontent-%COMP%]{color:#212121}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-error[_ngcontent-%COMP%]{color:#f44336}.mat-stepper-horizontal[_ngcontent-%COMP%], .mat-stepper-vertical[_ngcontent-%COMP%]{background-color:#fff}.mat-stepper-vertical-line[_ngcontent-%COMP%]::before{border-left-color:rgba(0,0,0,.12)}.mat-horizontal-stepper-header[_ngcontent-%COMP%]::before, .mat-horizontal-stepper-header[_ngcontent-%COMP%]::after, .mat-stepper-horizontal-line[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-sort-header-arrow[_ngcontent-%COMP%]{color:#616161}.mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-header[_ngcontent-%COMP%]{border-bottom:1px solid rgba(0,0,0,.12)}.mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%]{border-top:1px solid rgba(0,0,0,.12);border-bottom:none}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{color:#212121}.mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:#757575}.mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#212121}.mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#757575}.mat-tab-group[class*=mat-background-][_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar[class*=mat-background-][_ngcontent-%COMP%]{border-bottom:none;border-top:none}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-toolbar[_ngcontent-%COMP%]{background:#f5f5f5;color:#212121}.mat-toolbar.mat-primary[_ngcontent-%COMP%]{background:#f57c00;color:#fff}.mat-toolbar.mat-accent[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-toolbar.mat-warn[_ngcontent-%COMP%]{background:#f44336;color:#fff}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:currentColor}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:inherit}.mat-toolbar[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:currentColor}.mat-tooltip[_ngcontent-%COMP%]{background:rgba(97,97,97,.9)}.mat-tree[_ngcontent-%COMP%]{background:#fff}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{color:#212121}.mat-snack-bar-container[_ngcontent-%COMP%]{color:rgba(255,255,255,.7);background:#323232;box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-simple-snackbar-action[_ngcontent-%COMP%]{color:#ff9800}.filter-chip[_ngcontent-%COMP%]{align-items:center;display:flex;margin-left:5px}.metric-arithmetic-element-range[_ngcontent-%COMP%]{align-items:center;background-color:#fff;font-size:.8em;height:30px;justify-content:center;line-height:30px;padding:0 5px;margin-left:5px}.input-field[_ngcontent-%COMP%]{background-color:transparent;border:none;font-family:monospace;font-size:1.1em;transition:width 1s}.input-field[_ngcontent-%COMP%]:focus{background-color:rgba(0,0,0,.12);border:none;outline:none}.value-invalid[_ngcontent-%COMP%]{color:#f44336}"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/data_selection/metric_arithmetic/metric_arithmetic_element/metric_arithmetic_element_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class MetricArithmeticElementContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.filterValues$ = this.store.pipe(select(getMetricFilters)).pipe(map((/**
             * @param {?} filters
             * @return {?}
             */
            (filters) => {
                /** @type {?} */
                const filter = filters[this.metric];
                if (!filter) {
                    return { min: -1.0, max: 1.0 };
                }
                /** @type {?} */
                const min = filter.includeNaN
                    ? 'NaN'
                    : this.roundToThreeDecimalPoints(filter.min);
                /** @type {?} */
                const max = filter.max < filter.min
                    ? 'NaN'
                    : this.roundToThreeDecimalPoints(filter.max);
                return { min: min, max: max };
            })));
        }
        /**
         * @param {?} metric
         * @return {?}
         */
        remove(metric) {
            this.store.dispatch(npmiRemoveMetricFilter({ metric: metric }));
        }
        /**
         * @param {?} newValues
         * @return {?}
         */
        filterChange(newValues) {
            /** @type {?} */
            const min = isNaN(newValues.min) ? -1 : newValues.min;
            /** @type {?} */
            const max = isNaN(newValues.max) ? -2 : newValues.max;
            /** @type {?} */
            const includeNaN = isNaN(newValues.min);
            this.store.dispatch(npmiChangeMetricFilter({
                metric: this.metric,
                max: max,
                min: min,
                includeNaN: includeNaN,
            }));
        }
        /**
         * @private
         * @param {?} value
         * @return {?}
         */
        roundToThreeDecimalPoints(value) {
            return Math.round((value + Number.EPSILON) * 1000) / 1000;
        }
    }
    MetricArithmeticElementContainer.decorators = [
        { type: Component, args: [{
                    selector: 'npmi-metric-arithmetic-element',
                    template: `
    <metric-arithmetic-element-component
      [metric]="metric"
      [filterValues]="filterValues$ | async"
      (onRemove)="remove($event)"
      (onFilterChange)="filterChange($event)"
    ></metric-arithmetic-element-component>
  `,
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    MetricArithmeticElementContainer.ctorParameters = () => [
        { type: Store }
    ];
    MetricArithmeticElementContainer.propDecorators = {
        metric: [{ type: Input }]
    };
    /** @nocollapse */ MetricArithmeticElementContainer.ɵfac = function MetricArithmeticElementContainer_Factory(t) { return new (t || MetricArithmeticElementContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ MetricArithmeticElementContainer.ɵcmp = ɵɵdefineComponent({ type: MetricArithmeticElementContainer, selectors: [["npmi-metric-arithmetic-element"]], inputs: { metric: "metric" }, decls: 2, vars: 4, consts: [[3, "metric", "filterValues", "onRemove", "onFilterChange"]], template: function MetricArithmeticElementContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "metric-arithmetic-element-component", 0);
            ɵɵlistener("onRemove", function MetricArithmeticElementContainer_Template_metric_arithmetic_element_component_onRemove_0_listener($event) { return ctx.remove($event); })("onFilterChange", function MetricArithmeticElementContainer_Template_metric_arithmetic_element_component_onFilterChange_0_listener($event) { return ctx.filterChange($event); });
            ɵɵpipe(1, "async");
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵproperty("metric", ctx.metric)("filterValues", ɵɵpipeBind1(1, 2, ctx.filterValues$));
        } }, directives: [MetricArithmeticElementComponent], pipes: [AsyncPipe], encapsulation: 2, changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/data_selection/metric_arithmetic/metric_arithmetic_operator/metric_arithmetic_operator_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class MetricArithmeticOperatorComponent {
        constructor() {
            this.Operator = Operator;
        }
    }
    MetricArithmeticOperatorComponent.decorators = [
        { type: Component, args: [{
                    selector: 'npmi-metric-arithmetic-operator',
                    template: `
    <mat-chip>
      {{ operator === Operator.AND ? '&' : '' }}
    </mat-chip>
  `,
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    MetricArithmeticOperatorComponent.propDecorators = {
        operator: [{ type: Input }]
    };
    /** @nocollapse */ MetricArithmeticOperatorComponent.ɵfac = function MetricArithmeticOperatorComponent_Factory(t) { return new (t || MetricArithmeticOperatorComponent)(); };
    /** @nocollapse */ MetricArithmeticOperatorComponent.ɵcmp = ɵɵdefineComponent({ type: MetricArithmeticOperatorComponent, selectors: [["npmi-metric-arithmetic-operator"]], inputs: { operator: "operator" }, decls: 2, vars: 1, template: function MetricArithmeticOperatorComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "mat-chip");
            ɵɵtext(1);
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵadvance(1);
            ɵɵtextInterpolate1(" ", ctx.operator === ctx.Operator.AND ? "&" : "", " ");
        } }, directives: [MatChip], encapsulation: 2, changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/data_selection/metric_arithmetic/metric_arithmetic_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    function MetricArithmeticComponent_div_1_npmi_metric_arithmetic_element_1_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "npmi-metric-arithmetic-element", 4);
    } if (rf & 2) {
        const element_r1 = ɵɵnextContext().$implicit;
        ɵɵproperty("metric", element_r1.metric);
    } }
    function MetricArithmeticComponent_div_1_npmi_metric_arithmetic_operator_2_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "npmi-metric-arithmetic-operator", 5);
    } if (rf & 2) {
        const element_r1 = ɵɵnextContext().$implicit;
        ɵɵproperty("operator", element_r1.operator);
    } }
    function MetricArithmeticComponent_div_1_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div");
        ɵɵtemplate(1, MetricArithmeticComponent_div_1_npmi_metric_arithmetic_element_1_Template, 1, 1, "npmi-metric-arithmetic-element", 2);
        ɵɵtemplate(2, MetricArithmeticComponent_div_1_npmi_metric_arithmetic_operator_2_Template, 1, 1, "npmi-metric-arithmetic-operator", 3);
        ɵɵelementEnd();
    } if (rf & 2) {
        const element_r1 = ctx.$implicit;
        const ctx_r0 = ɵɵnextContext();
        ɵɵadvance(1);
        ɵɵproperty("ngIf", element_r1.kind === ctx_r0.ArithmeticKind.METRIC);
        ɵɵadvance(1);
        ɵɵproperty("ngIf", element_r1.kind === ctx_r0.ArithmeticKind.OPERATOR);
    } }
    class MetricArithmeticComponent {
        constructor() {
            this.ArithmeticKind = ArithmeticKind;
        }
    }
    MetricArithmeticComponent.decorators = [
        { type: Component, args: [{
                    selector: 'metric-arithmetic-component',
                    templateUrl: './metric_arithmetic_component.ng.html',
                    styleUrls: ['./metric_arithmetic_component.css'],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    MetricArithmeticComponent.propDecorators = {
        metricArithmetic: [{ type: Input }]
    };
    /** @nocollapse */ MetricArithmeticComponent.ɵfac = function MetricArithmeticComponent_Factory(t) { return new (t || MetricArithmeticComponent)(); };
    /** @nocollapse */ MetricArithmeticComponent.ɵcmp = ɵɵdefineComponent({ type: MetricArithmeticComponent, selectors: [["metric-arithmetic-component"]], inputs: { metricArithmetic: "metricArithmetic" }, decls: 2, vars: 2, consts: [[3, "selectable"], [4, "ngFor", "ngForOf"], [3, "metric", 4, "ngIf"], [3, "operator", 4, "ngIf"], [3, "metric"], [3, "operator"]], template: function MetricArithmeticComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "mat-chip-list", 0);
            ɵɵtemplate(1, MetricArithmeticComponent_div_1_Template, 3, 2, "div", 1);
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵproperty("selectable", false);
            ɵɵadvance(1);
            ɵɵproperty("ngForOf", ctx.metricArithmetic);
        } }, directives: [MatChipList, NgForOf, NgIf, MetricArithmeticElementContainer, MetricArithmeticOperatorComponent], styles: ["[_nghost-%COMP%]{align-items:center;display:flex;flex-direction:row;flex-wrap:wrap}"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/data_selection/metric_arithmetic/metric_arithmetic_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class MetricArithmeticContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.metricArithmetic$ = this.store.pipe(select(getMetricArithmetic));
        }
    }
    MetricArithmeticContainer.decorators = [
        { type: Component, args: [{
                    selector: 'npmi-metric-arithmetic',
                    template: `
    <metric-arithmetic-component
      [metricArithmetic]="metricArithmetic$ | async"
    ></metric-arithmetic-component>
  `,
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    MetricArithmeticContainer.ctorParameters = () => [
        { type: Store }
    ];
    /** @nocollapse */ MetricArithmeticContainer.ɵfac = function MetricArithmeticContainer_Factory(t) { return new (t || MetricArithmeticContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ MetricArithmeticContainer.ɵcmp = ɵɵdefineComponent({ type: MetricArithmeticContainer, selectors: [["npmi-metric-arithmetic"]], decls: 2, vars: 3, consts: [[3, "metricArithmetic"]], template: function MetricArithmeticContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelement(0, "metric-arithmetic-component", 0);
            ɵɵpipe(1, "async");
        } if (rf & 2) {
            ɵɵproperty("metricArithmetic", ɵɵpipeBind1(1, 1, ctx.metricArithmetic$));
        } }, directives: [MetricArithmeticComponent], pipes: [AsyncPipe], encapsulation: 2, changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/data_selection/data_selection_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class DataSelectionComponent {
    }
    DataSelectionComponent.decorators = [
        { type: Component, args: [{
                    selector: 'npmi-data-selection',
                    templateUrl: './data_selection_component.ng.html',
                    styleUrls: ['./data_selection_component.css'],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */ DataSelectionComponent.ɵfac = function DataSelectionComponent_Factory(t) { return new (t || DataSelectionComponent)(); };
    /** @nocollapse */ DataSelectionComponent.ɵcmp = ɵɵdefineComponent({ type: DataSelectionComponent, selectors: [["npmi-data-selection"]], decls: 4, vars: 0, consts: [[1, "data-selection"], [1, "metrics-selector"]], template: function DataSelectionComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "div", 0);
            ɵɵelement(1, "npmi-metric-search", 1);
            ɵɵelement(2, "npmi-results-download");
            ɵɵelementEnd();
            ɵɵelement(3, "npmi-metric-arithmetic");
        } }, directives: [MetricSearchContainer, ResultsDownloadContainer, MetricArithmeticContainer], styles: [".mat-badge-content[_ngcontent-%COMP%]{font-weight:600;font-size:12px;font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:9px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:24px}.mat-h1[_ngcontent-%COMP%], .mat-headline[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%]{font:400 24px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h2[_ngcontent-%COMP%], .mat-title[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h3[_ngcontent-%COMP%], .mat-subheading-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%]{font:400 16px/28px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h4[_ngcontent-%COMP%], .mat-subheading-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%]{font:400 15px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h5[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%]{font:400 calc(14px * 0.83)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-h6[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:400 calc(14px * 0.67)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-body-strong[_ngcontent-%COMP%], .mat-body-2[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   p[_ngcontent-%COMP%]{margin:0 0 12px}.mat-small[_ngcontent-%COMP%], .mat-caption[_ngcontent-%COMP%]{font:400 12px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-display-4[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-4[_ngcontent-%COMP%]{font:300 112px/112px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.05em;margin:0 0 56px}.mat-display-3[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-3[_ngcontent-%COMP%]{font:400 56px/56px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.02em;margin:0 0 64px}.mat-display-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-2[_ngcontent-%COMP%]{font:400 45px/48px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.005em;margin:0 0 64px}.mat-display-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-1[_ngcontent-%COMP%]{font:400 34px/40px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 64px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%], .mat-flat-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-button-toggle[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card-title[_ngcontent-%COMP%]{font-size:24px;font-weight:500}.mat-card-header[_ngcontent-%COMP%]   .mat-card-title[_ngcontent-%COMP%]{font-size:20px}.mat-card-subtitle[_ngcontent-%COMP%], .mat-card-content[_ngcontent-%COMP%]{font-size:14px}.mat-checkbox[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-checkbox-layout[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{line-height:24px}.mat-chip[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-chip[_ngcontent-%COMP%]   .mat-chip-trailing-icon.mat-icon[_ngcontent-%COMP%], .mat-chip[_ngcontent-%COMP%]   .mat-chip-remove.mat-icon[_ngcontent-%COMP%]{font-size:18px}.mat-table[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-header-cell[_ngcontent-%COMP%]{font-size:12px;font-weight:500}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{font-size:14px}.mat-calendar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-calendar-body[_ngcontent-%COMP%]{font-size:13px}.mat-calendar-body-label[_ngcontent-%COMP%], .mat-calendar-period-button[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-calendar-table-header[_ngcontent-%COMP%]   th[_ngcontent-%COMP%]{font-size:11px;font-weight:400}.mat-dialog-title[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-expansion-panel-header[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:15px;font-weight:400}.mat-expansion-panel-content[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field[_ngcontent-%COMP%]{font-size:inherit;font-weight:400;line-height:1.125;font-family:Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.34375em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{font-size:150%;line-height:1.125}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]{height:1.5em;width:1.5em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{height:1.125em;line-height:1.125}.mat-form-field-infix[_ngcontent-%COMP%]{padding:.5em 0;border-top:.84375em solid transparent}.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34375em) scale(0.75);width:133.3333333333%}.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34374em) scale(0.75);width:133.3333433333%}.mat-form-field-label-wrapper[_ngcontent-%COMP%]{top:-0.84375em;padding-top:.84375em}.mat-form-field-label[_ngcontent-%COMP%]{top:1.34375em}.mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.34375em}.mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{font-size:75%;margin-top:.6666666667em;top:calc(100% - 1.7916666667em)}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.4375em 0}.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.001px);-ms-transform:translateY(-1.28125em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00101px);-ms-transform:translateY(-1.28124em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00102px);-ms-transform:translateY(-1.28123em) scale(0.75);width:133.3333533333%}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.28125em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{margin-top:.5416666667em;top:calc(100% - 1.6666666667em)}@media print{.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28122em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28121em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.2812em) scale(0.75)}}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.25em 0 .75em 0}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.09375em;margin-top:-0.5em}.mat-form-field-appearance-fill.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59374em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:1em 0 1em 0}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.84375em;margin-top:-0.25em}.mat-form-field-appearance-outline.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59374em) scale(0.75);width:133.3333433333%}.mat-grid-tile-header[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]{font-size:14px}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2), .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}input.mat-input-element[_ngcontent-%COMP%]{margin-top:-0.0625em}.mat-menu-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:400}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px}.mat-radio-button[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select-trigger[_ngcontent-%COMP%]{height:1.125em}.mat-slide-toggle-content[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-slider-thumb-label-text[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-stepper-vertical[_ngcontent-%COMP%], .mat-stepper-horizontal[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-step-label[_ngcontent-%COMP%]{font-size:14px;font-weight:400}.mat-step-sub-label-error[_ngcontent-%COMP%]{font-weight:normal}.mat-step-label-error[_ngcontent-%COMP%]{font-size:14px}.mat-step-label-selected[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-tab-group[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-toolbar[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0}.mat-tooltip[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:10px;padding-top:6px;padding-bottom:6px}.mat-tooltip-handset[_ngcontent-%COMP%]{font-size:14px;padding-top:8px;padding-bottom:8px}.mat-list-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:16px}.mat-optgroup-label[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-simple-snackbar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px}.mat-simple-snackbar-action[_ngcontent-%COMP%]{line-height:1;font-family:inherit;font-size:inherit;font-weight:500}.mat-tree[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{font-weight:400;font-size:14px}.mat-ripple[_ngcontent-%COMP%]{overflow:hidden;position:relative}.mat-ripple[_ngcontent-%COMP%]:not(:empty){transform:translateZ(0)}.mat-ripple.mat-ripple-unbounded[_ngcontent-%COMP%]{overflow:visible}.mat-ripple-element[_ngcontent-%COMP%]{position:absolute;border-radius:50%;pointer-events:none;transition:opacity,transform 0ms cubic-bezier(0, 0, 0.2, 1);transform:scale(0)}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{display:none}.cdk-visually-hidden[_ngcontent-%COMP%]{border:0;clip:rect(0 0 0 0);height:1px;margin:-1px;overflow:hidden;padding:0;position:absolute;width:1px;outline:0;-webkit-appearance:none;-moz-appearance:none}.cdk-overlay-container[_ngcontent-%COMP%], .cdk-global-overlay-wrapper[_ngcontent-%COMP%]{pointer-events:none;top:0;left:0;height:100%;width:100%}.cdk-overlay-container[_ngcontent-%COMP%]{position:fixed;z-index:1000}.cdk-overlay-container[_ngcontent-%COMP%]:empty{display:none}.cdk-global-overlay-wrapper[_ngcontent-%COMP%]{display:flex;position:absolute;z-index:1000}.cdk-overlay-pane[_ngcontent-%COMP%]{position:absolute;pointer-events:auto;box-sizing:border-box;z-index:1000;display:flex;max-width:100%;max-height:100%}.cdk-overlay-backdrop[_ngcontent-%COMP%]{position:absolute;top:0;bottom:0;left:0;right:0;z-index:1000;pointer-events:auto;-webkit-tap-highlight-color:transparent;transition:opacity 400ms cubic-bezier(0.25, 0.8, 0.25, 1);opacity:0}.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:1}@media screen and (-ms-high-contrast: active){.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:.6}}.cdk-overlay-dark-backdrop[_ngcontent-%COMP%]{background:rgba(0,0,0,.32)}.cdk-overlay-transparent-backdrop[_ngcontent-%COMP%], .cdk-overlay-transparent-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:0}.cdk-overlay-connected-position-bounding-box[_ngcontent-%COMP%]{position:absolute;z-index:1000;display:flex;flex-direction:column;min-width:1px;min-height:1px}.cdk-global-scrollblock[_ngcontent-%COMP%]{position:fixed;width:100%;overflow-y:scroll}@keyframes cdk-text-field-autofill-start{}@keyframes cdk-text-field-autofill-end{}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:-webkit-autofill{animation:cdk-text-field-autofill-start 0s 1ms}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:not(:-webkit-autofill){animation:cdk-text-field-autofill-end 0s 1ms}textarea.cdk-textarea-autosize[_ngcontent-%COMP%]{resize:none}textarea.cdk-textarea-autosize-measuring[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:auto !important;overflow:hidden !important}textarea.cdk-textarea-autosize-measuring-firefox[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:0 !important}.mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.1)}.mat-option[_ngcontent-%COMP%]{color:#212121}.mat-option[_ngcontent-%COMP%]:hover:not(.mat-option-disabled), .mat-option[_ngcontent-%COMP%]:focus:not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple):not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-active[_ngcontent-%COMP%]{background:rgba(0,0,0,.04);color:#212121}.mat-option.mat-option-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-primary[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f57c00}.mat-accent[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f44336}.mat-optgroup-label[_ngcontent-%COMP%]{color:#616161}.mat-optgroup-disabled[_ngcontent-%COMP%]   .mat-optgroup-label[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-pseudo-checkbox[_ngcontent-%COMP%]{color:#616161}.mat-pseudo-checkbox[_ngcontent-%COMP%]::after{color:#fafafa}.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{color:#b0b0b0}.mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f57c00}.mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f44336}.mat-pseudo-checkbox-checked.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{background:#b0b0b0}.mat-elevation-z0[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z1[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z2[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z3[_ngcontent-%COMP%]{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z4[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z5[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 5px 8px 0px rgba(0, 0, 0, 0.14),0px 1px 14px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z6[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z7[_ngcontent-%COMP%]{box-shadow:0px 4px 5px -2px rgba(0, 0, 0, 0.2),0px 7px 10px 1px rgba(0, 0, 0, 0.14),0px 2px 16px 1px rgba(0, 0, 0, 0.12)}.mat-elevation-z8[_ngcontent-%COMP%]{box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z9[_ngcontent-%COMP%]{box-shadow:0px 5px 6px -3px rgba(0, 0, 0, 0.2),0px 9px 12px 1px rgba(0, 0, 0, 0.14),0px 3px 16px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z10[_ngcontent-%COMP%]{box-shadow:0px 6px 6px -3px rgba(0, 0, 0, 0.2),0px 10px 14px 1px rgba(0, 0, 0, 0.14),0px 4px 18px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z11[_ngcontent-%COMP%]{box-shadow:0px 6px 7px -4px rgba(0, 0, 0, 0.2),0px 11px 15px 1px rgba(0, 0, 0, 0.14),0px 4px 20px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z12[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z13[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 13px 19px 2px rgba(0, 0, 0, 0.14),0px 5px 24px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z14[_ngcontent-%COMP%]{box-shadow:0px 7px 9px -4px rgba(0, 0, 0, 0.2),0px 14px 21px 2px rgba(0, 0, 0, 0.14),0px 5px 26px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z15[_ngcontent-%COMP%]{box-shadow:0px 8px 9px -5px rgba(0, 0, 0, 0.2),0px 15px 22px 2px rgba(0, 0, 0, 0.14),0px 6px 28px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z16[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z17[_ngcontent-%COMP%]{box-shadow:0px 8px 11px -5px rgba(0, 0, 0, 0.2),0px 17px 26px 2px rgba(0, 0, 0, 0.14),0px 6px 32px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z18[_ngcontent-%COMP%]{box-shadow:0px 9px 11px -5px rgba(0, 0, 0, 0.2),0px 18px 28px 2px rgba(0, 0, 0, 0.14),0px 7px 34px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z19[_ngcontent-%COMP%]{box-shadow:0px 9px 12px -6px rgba(0, 0, 0, 0.2),0px 19px 29px 2px rgba(0, 0, 0, 0.14),0px 7px 36px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z20[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 20px 31px 3px rgba(0, 0, 0, 0.14),0px 8px 38px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z21[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 21px 33px 3px rgba(0, 0, 0, 0.14),0px 8px 40px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z22[_ngcontent-%COMP%]{box-shadow:0px 10px 14px -6px rgba(0, 0, 0, 0.2),0px 22px 35px 3px rgba(0, 0, 0, 0.14),0px 8px 42px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z23[_ngcontent-%COMP%]{box-shadow:0px 11px 14px -7px rgba(0, 0, 0, 0.2),0px 23px 36px 3px rgba(0, 0, 0, 0.14),0px 9px 44px 8px rgba(0, 0, 0, 0.12)}.mat-elevation-z24[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12)}.mat-app-background[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-theme-loaded-marker[_ngcontent-%COMP%]{display:none}.mat-autocomplete-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-autocomplete-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover){background:#fff}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover):not(.mat-option-disabled){color:#212121}.mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f57c00}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{outline:solid 1px;border-radius:0}.mat-badge-accent[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-badge-warn[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f44336}.mat-badge[_ngcontent-%COMP%]{position:relative}.mat-badge-hidden[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{display:none}.mat-badge-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#b9b9b9;color:#757575}.mat-badge-content[_ngcontent-%COMP%]{position:absolute;text-align:center;display:inline-block;border-radius:50%;transition:transform 200ms ease-in-out;transform:scale(0.6);overflow:hidden;white-space:nowrap;text-overflow:ellipsis;pointer-events:none}.ng-animate-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%], .mat-badge-content._mat-animation-noopable[_ngcontent-%COMP%]{transition:none}.mat-badge-content.mat-badge-active[_ngcontent-%COMP%]{transform:none}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:16px;height:16px;line-height:16px}.mat-badge-small.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-8px}.mat-badge-small.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-8px}.mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-16px}.mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-16px}.mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-8px}.mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-8px}.mat-badge-medium[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:22px;height:22px;line-height:22px}.mat-badge-medium.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-11px}.mat-badge-medium.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-11px}.mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-22px}.mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-22px}.mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-11px}.mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-11px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:28px;height:28px;line-height:28px}.mat-badge-large.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-14px}.mat-badge-large.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-14px}.mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-28px}.mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-28px}.mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-14px}.mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-14px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]{color:inherit;background:transparent}.mat-button.mat-primary[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-button[disabled][disabled][_ngcontent-%COMP%], .mat-icon-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-icon-button[disabled][disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-stroked-button[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f44336}.mat-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:transparent}.mat-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{opacity:.1;background-color:currentColor}.mat-button-focus-overlay[_ngcontent-%COMP%]{background:#000}.mat-stroked-button[_ngcontent-%COMP%]:not([disabled]){border-color:rgba(0,0,0,.12)}.mat-flat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{color:#212121;background-color:#fff}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-flat-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-stroked-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-flat-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-raised-button[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone[_ngcontent-%COMP%], .mat-button-toggle-group[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{box-shadow:none}.mat-button-toggle[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-button-toggle[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121;background:#fff}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:#000}.mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12)}[dir=rtl][_ngcontent-%COMP%]   .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-group-appearance-standard.mat-button-toggle-vertical[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:none;border-top:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#616161}.mat-button-toggle-checked.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121}.mat-button-toggle-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.26);background-color:#eee}.mat-button-toggle-disabled.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{background:#fff}.mat-button-toggle-disabled.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#bdbdbd}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{border:solid 1px rgba(0,0,0,.12)}.mat-card[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-card[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-card.mat-card-flat[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-card-subtitle[_ngcontent-%COMP%]{color:#616161}.mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#616161}.mat-checkbox-checkmark[_ngcontent-%COMP%]{fill:#fafafa}.mat-checkbox-checkmark-path[_ngcontent-%COMP%]{stroke:#fafafa !important}.mat-checkbox-mixedmark[_ngcontent-%COMP%]{background-color:#fafafa}.mat-checkbox-indeterminate.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f57c00}.mat-checkbox-indeterminate.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#ff9800}.mat-checkbox-indeterminate.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f44336}.mat-checkbox-disabled.mat-checkbox-checked[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-disabled.mat-checkbox-indeterminate[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]:not(.mat-checkbox-checked)   .mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{color:#616161}.mat-checkbox[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%]{background:#f57c00}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%]{background:#ff9800}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%]{background:#f44336}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#212121}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#212121;opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled):active{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled)   .mat-chip-remove[_ngcontent-%COMP%]:hover{opacity:.54}.mat-chip.mat-standard-chip.mat-chip-disabled[_ngcontent-%COMP%]{opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]::after{background:#000}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-table[_ngcontent-%COMP%]{background:#fff}.mat-table[_ngcontent-%COMP%]   thead[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tbody[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tfoot[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], [mat-header-row][_ngcontent-%COMP%], [mat-row][_ngcontent-%COMP%], [mat-footer-row][_ngcontent-%COMP%], .mat-table-sticky[_ngcontent-%COMP%]{background:inherit}mat-row[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], th.mat-header-cell[_ngcontent-%COMP%], td.mat-cell[_ngcontent-%COMP%], td.mat-footer-cell[_ngcontent-%COMP%]{border-bottom-color:rgba(0,0,0,.12)}.mat-header-cell[_ngcontent-%COMP%]{color:#616161}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{color:#212121}.mat-calendar-arrow[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.54)}.mat-datepicker-toggle[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-next-button[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-previous-button[_ngcontent-%COMP%]{color:rgba(0,0,0,.54)}.mat-calendar-table-header[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-calendar-table-header-divider[_ngcontent-%COMP%]::after{background:rgba(0,0,0,.12)}.mat-calendar-body-label[_ngcontent-%COMP%]{color:#616161}.mat-calendar-body-cell-content[_ngcontent-%COMP%]{color:#212121;border-color:transparent}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){color:#757575}.mat-calendar-body-cell[_ngcontent-%COMP%]:not(.mat-calendar-body-disabled):hover > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-keyboard-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-program-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){background-color:rgba(0,0,0,.04)}.mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.38)}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.18)}.mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.4)}.mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12);background-color:#fff;color:#212121}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.4)}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.4)}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content-touch[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-datepicker-toggle-active[_ngcontent-%COMP%]{color:#f57c00}.mat-datepicker-toggle-active.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-datepicker-toggle-active.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-dialog-container[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-divider[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-divider-vertical[_ngcontent-%COMP%]{border-right-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-expansion-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-action-row[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-keyboard-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-program-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded)   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover:not([aria-disabled=true]){background:rgba(0,0,0,.04)}@media(hover: none){.mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded):not([aria-disabled=true])   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover{background:#fff}}.mat-expansion-panel-header-title[_ngcontent-%COMP%]{color:#212121}.mat-expansion-panel-header-description[_ngcontent-%COMP%], .mat-expansion-indicator[_ngcontent-%COMP%]::after{color:#616161}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-title[_ngcontent-%COMP%], .mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-description[_ngcontent-%COMP%]{color:inherit}.mat-form-field-label[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-hint[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-focused[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-ripple[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid)   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f57c00}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-accent   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#ff9800}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-warn   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#f44336}.mat-error[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-hint[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-legacy.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-standard[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-standard.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.04)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.02)}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:transparent}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.12)}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:rgba(0,0,0,.87)}.mat-form-field-appearance-outline.mat-focused[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field-appearance-outline.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-appearance-outline.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-invalid.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.06)}.mat-icon.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-icon.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-icon.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-type-mat-native-select[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#616161}.mat-input-element[_ngcontent-%COMP%]:disabled, .mat-form-field-type-mat-native-select.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#757575}.mat-input-element[_ngcontent-%COMP%]{caret-color:#f57c00}.mat-input-element[_ngcontent-%COMP%]::placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-moz-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-webkit-input-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]:-ms-input-placeholder{color:rgba(97,97,97,.42)}.mat-accent[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%], .mat-form-field-invalid[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#f44336}.mat-form-field-type-mat-native-select.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{color:#616161}.mat-list-item-disabled[_ngcontent-%COMP%]{background-color:#eee}.mat-list-option[_ngcontent-%COMP%]:hover, .mat-list-option[_ngcontent-%COMP%]:focus, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.04)}.mat-list-single-selected-option[_ngcontent-%COMP%], .mat-list-single-selected-option[_ngcontent-%COMP%]:hover, .mat-list-single-selected-option[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.12)}.mat-menu-panel[_ngcontent-%COMP%]{background:#fff}.mat-menu-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-menu-item[_ngcontent-%COMP%]{background:transparent;color:#212121}.mat-menu-item[disabled][_ngcontent-%COMP%], .mat-menu-item[disabled][_ngcontent-%COMP%]::after{color:rgba(0,0,0,.38)}.mat-menu-item[_ngcontent-%COMP%]   .mat-icon-no-color[_ngcontent-%COMP%], .mat-menu-item-submenu-trigger[_ngcontent-%COMP%]::after{color:rgba(0,0,0,.54)}.mat-menu-item[_ngcontent-%COMP%]:hover:not([disabled]), .mat-menu-item.cdk-program-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item.cdk-keyboard-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item-highlighted[_ngcontent-%COMP%]:not([disabled]){background:rgba(0,0,0,.04)}.mat-paginator[_ngcontent-%COMP%]{background:#fff}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{color:#616161}.mat-paginator-decrement[_ngcontent-%COMP%], .mat-paginator-increment[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54);border-right:2px solid rgba(0,0,0,.54)}.mat-paginator-first[_ngcontent-%COMP%], .mat-paginator-last[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54)}.mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-decrement[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-increment[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-first[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-last[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f57c00}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#ff9800}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f44336}.mat-progress-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f57c00}.mat-progress-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#ff9800}.mat-progress-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f44336}.mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#616161}.mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f57c00}.mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#ff9800}.mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#ff9800}.mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f44336}.mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f44336}.mat-radio-button.mat-radio-disabled.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-label-content[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-radio-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-select-value[_ngcontent-%COMP%]{color:#212121}.mat-select-placeholder[_ngcontent-%COMP%]{color:rgba(97,97,97,.42)}.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%]{color:#757575}.mat-select-arrow[_ngcontent-%COMP%]{color:#616161}.mat-select-panel[_ngcontent-%COMP%]{background:#fff}.mat-select-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-select-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple){background:rgba(0,0,0,.12)}.mat-form-field.mat-focused.mat-primary[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-invalid[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#757575}.mat-drawer-container[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-drawer[_ngcontent-%COMP%]{background-color:#fff;color:#212121}.mat-drawer.mat-drawer-push[_ngcontent-%COMP%]{background-color:#fff}.mat-drawer[_ngcontent-%COMP%]:not(.mat-drawer-side){box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-drawer-side[_ngcontent-%COMP%]{border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-backdrop.mat-drawer-shown[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.6)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.54)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.54)}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.54)}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle[_ngcontent-%COMP%]:not(.mat-checked)   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-slide-toggle-thumb[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12);background-color:#fafafa}.mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-primary[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f57c00}.mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-accent[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#ff9800}.mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-warn[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f44336}.mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.2)}.mat-slider[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%], .cdk-focused[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-disabled[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]   .mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing)   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26);background-color:transparent}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26)}.mat-slider-has-ticks[_ngcontent-%COMP%]   .mat-slider-wrapper[_ngcontent-%COMP%]::after{border-color:rgba(0,0,0,.7)}.mat-slider-horizontal[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to right, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent);background-image:-moz-repeating-linear-gradient(0.0001deg, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-slider-vertical[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to bottom, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-step-header.cdk-keyboard-focused[_ngcontent-%COMP%], .mat-step-header.cdk-program-focused[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]:hover{background-color:rgba(0,0,0,.04)}@media(hover: none){.mat-step-header[_ngcontent-%COMP%]:hover{background:none}}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-optional[_ngcontent-%COMP%]{color:#616161}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon[_ngcontent-%COMP%]{background-color:#616161;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-selected[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-done[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-edit[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-error[_ngcontent-%COMP%]{background-color:transparent;color:#f44336}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-active[_ngcontent-%COMP%]{color:#212121}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-error[_ngcontent-%COMP%]{color:#f44336}.mat-stepper-horizontal[_ngcontent-%COMP%], .mat-stepper-vertical[_ngcontent-%COMP%]{background-color:#fff}.mat-stepper-vertical-line[_ngcontent-%COMP%]::before{border-left-color:rgba(0,0,0,.12)}.mat-horizontal-stepper-header[_ngcontent-%COMP%]::before, .mat-horizontal-stepper-header[_ngcontent-%COMP%]::after, .mat-stepper-horizontal-line[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-sort-header-arrow[_ngcontent-%COMP%]{color:#616161}.mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-header[_ngcontent-%COMP%]{border-bottom:1px solid rgba(0,0,0,.12)}.mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%]{border-top:1px solid rgba(0,0,0,.12);border-bottom:none}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{color:#212121}.mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:#757575}.mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#212121}.mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#757575}.mat-tab-group[class*=mat-background-][_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar[class*=mat-background-][_ngcontent-%COMP%]{border-bottom:none;border-top:none}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-toolbar[_ngcontent-%COMP%]{background:#f5f5f5;color:#212121}.mat-toolbar.mat-primary[_ngcontent-%COMP%]{background:#f57c00;color:#fff}.mat-toolbar.mat-accent[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-toolbar.mat-warn[_ngcontent-%COMP%]{background:#f44336;color:#fff}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:currentColor}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:inherit}.mat-toolbar[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:currentColor}.mat-tooltip[_ngcontent-%COMP%]{background:rgba(97,97,97,.9)}.mat-tree[_ngcontent-%COMP%]{background:#fff}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{color:#212121}.mat-snack-bar-container[_ngcontent-%COMP%]{color:rgba(255,255,255,.7);background:#323232;box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-simple-snackbar-action[_ngcontent-%COMP%]{color:#ff9800}[_nghost-%COMP%]{display:flex;flex-direction:column;background-color:#fff;border:1px solid #ebebeb;padding:10px 20px}.data-selection[_ngcontent-%COMP%]{display:flex;align-items:center}.metrics-selector[_ngcontent-%COMP%]{flex:1 1}"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/util/filter_annotations.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @param {?} annotationData
     * @param {?} activeRuns
     * @param {?} metricArithmetic
     * @param {?} metricFilters
     * @param {?} metrics
     * @param {?} annotationsRegex
     * @return {?}
     */
    function filterAnnotations(annotationData, activeRuns, metricArithmetic, metricFilters, metrics, annotationsRegex) {
        /** @type {?} */
        const data = {};
        /** @type {?} */
        const allRuns = new Set(activeRuns);
        /** @type {?} */
        const allMetrics = new Set(metrics.map((/**
         * @param {?} metric
         * @return {?}
         */
        (metric) => stripMetricString(metric))));
        /** @type {?} */
        const filterRegex = new RegExp(annotationsRegex, 'i');
        Object.entries(annotationData).forEach((/**
         * @param {?} entry
         * @return {?}
         */
        (entry) => {
            if (!filterRegex.test(entry[0])) {
                return;
            }
            /** @type {?} */
            let valueDataElements = entry[1];
            // Remove all inactive runs and keep only metrics currently displayed
            valueDataElements = valueDataElements.filter((/**
             * @param {?} valueDataElement
             * @return {?}
             */
            (valueDataElement) => {
                return (allRuns.has(valueDataElement.run) &&
                    allMetrics.has(valueDataElement.metric));
            }));
            if (checkValuesPassMetricArithmetic(metricArithmetic, metricFilters, valueDataElements)) {
                if (valueDataElements.length !== 0) {
                    data[entry[0]] = valueDataElements;
                }
            }
        }));
        return data;
    }
    /**
     * @param {?} annotationData
     * @param {?} hiddenAnnotations
     * @param {?} showHidden
     * @return {?}
     */
    function removeHiddenAnnotations(annotationData, hiddenAnnotations, showHidden) {
        if (showHidden) {
            return annotationData;
        }
        /** @type {?} */
        const data = Object.assign({}, annotationData);
        hiddenAnnotations.forEach((/**
         * @param {?} annotation
         * @return {?}
         */
        (annotation) => delete data[annotation]));
        return data;
    }
    /**
     * @param {?} metricArithmetic
     * @param {?} metricFilters
     * @param {?} valueDataElements
     * @return {?}
     */
    function checkValuesPassMetricArithmetic(metricArithmetic, metricFilters, valueDataElements) {
        // Check all parts of the arithemetic
        return metricArithmetic.every((/**
         * @param {?} element
         * @return {?}
         */
        (element) => {
            if (element.kind === ArithmeticKind.OPERATOR) {
                return true;
            }
            /** @type {?} */
            const metricFilter = metricFilters[element.metric];
            if (metricFilter === undefined) {
                return true;
            }
            return valueDataElements.some((/**
             * @param {?} valueDataElement
             * @return {?}
             */
            (valueDataElement) => {
                if (valueDataElement.metric === stripMetricString(element.metric)) {
                    if (valueDataElement.nPMIValue === null) {
                        return metricFilter.includeNaN;
                    }
                    else {
                        return ((/** @type {?} */ (valueDataElement.nPMIValue)) <= metricFilter.max &&
                            (/** @type {?} */ (valueDataElement.nPMIValue)) >= metricFilter.min);
                    }
                }
                return false;
            }));
        }));
    }

    function ascending(a, b) {
      return a < b ? -1 : a > b ? 1 : a >= b ? 0 : NaN;
    }

    function bisector(compare) {
      if (compare.length === 1) compare = ascendingComparator(compare);
      return {
        left: function(a, x, lo, hi) {
          if (lo == null) lo = 0;
          if (hi == null) hi = a.length;
          while (lo < hi) {
            var mid = lo + hi >>> 1;
            if (compare(a[mid], x) < 0) lo = mid + 1;
            else hi = mid;
          }
          return lo;
        },
        right: function(a, x, lo, hi) {
          if (lo == null) lo = 0;
          if (hi == null) hi = a.length;
          while (lo < hi) {
            var mid = lo + hi >>> 1;
            if (compare(a[mid], x) > 0) hi = mid;
            else lo = mid + 1;
          }
          return lo;
        }
      };
    }

    function ascendingComparator(f) {
      return function(d, x) {
        return ascending(f(d), x);
      };
    }

    var ascendingBisect = bisector(ascending);
    var bisectRight = ascendingBisect.right;

    function extent(values, valueof) {
      var n = values.length,
          i = -1,
          value,
          min,
          max;

      if (valueof == null) {
        while (++i < n) { // Find the first comparable value.
          if ((value = values[i]) != null && value >= value) {
            min = max = value;
            while (++i < n) { // Compare the remaining values.
              if ((value = values[i]) != null) {
                if (min > value) min = value;
                if (max < value) max = value;
              }
            }
          }
        }
      }

      else {
        while (++i < n) { // Find the first comparable value.
          if ((value = valueof(values[i], i, values)) != null && value >= value) {
            min = max = value;
            while (++i < n) { // Compare the remaining values.
              if ((value = valueof(values[i], i, values)) != null) {
                if (min > value) min = value;
                if (max < value) max = value;
              }
            }
          }
        }
      }

      return [min, max];
    }

    var array = Array.prototype;

    var slice = array.slice;

    function constant(x) {
      return function() {
        return x;
      };
    }

    function identity$1(x) {
      return x;
    }

    function sequence$1(start, stop, step) {
      start = +start, stop = +stop, step = (n = arguments.length) < 2 ? (stop = start, start = 0, 1) : n < 3 ? 1 : +step;

      var i = -1,
          n = Math.max(0, Math.ceil((stop - start) / step)) | 0,
          range = new Array(n);

      while (++i < n) {
        range[i] = start + i * step;
      }

      return range;
    }

    var e10 = Math.sqrt(50),
        e5 = Math.sqrt(10),
        e2 = Math.sqrt(2);

    function ticks(start, stop, count) {
      var reverse,
          i = -1,
          n,
          ticks,
          step;

      stop = +stop, start = +start, count = +count;
      if (start === stop && count > 0) return [start];
      if (reverse = stop < start) n = start, start = stop, stop = n;
      if ((step = tickIncrement(start, stop, count)) === 0 || !isFinite(step)) return [];

      if (step > 0) {
        start = Math.ceil(start / step);
        stop = Math.floor(stop / step);
        ticks = new Array(n = Math.ceil(stop - start + 1));
        while (++i < n) ticks[i] = (start + i) * step;
      } else {
        start = Math.floor(start * step);
        stop = Math.ceil(stop * step);
        ticks = new Array(n = Math.ceil(start - stop + 1));
        while (++i < n) ticks[i] = (start - i) / step;
      }

      if (reverse) ticks.reverse();

      return ticks;
    }

    function tickIncrement(start, stop, count) {
      var step = (stop - start) / Math.max(0, count),
          power = Math.floor(Math.log(step) / Math.LN10),
          error = step / Math.pow(10, power);
      return power >= 0
          ? (error >= e10 ? 10 : error >= e5 ? 5 : error >= e2 ? 2 : 1) * Math.pow(10, power)
          : -Math.pow(10, -power) / (error >= e10 ? 10 : error >= e5 ? 5 : error >= e2 ? 2 : 1);
    }

    function tickStep(start, stop, count) {
      var step0 = Math.abs(stop - start) / Math.max(0, count),
          step1 = Math.pow(10, Math.floor(Math.log(step0) / Math.LN10)),
          error = step0 / step1;
      if (error >= e10) step1 *= 10;
      else if (error >= e5) step1 *= 5;
      else if (error >= e2) step1 *= 2;
      return stop < start ? -step1 : step1;
    }

    function thresholdSturges(values) {
      return Math.ceil(Math.log(values.length) / Math.LN2) + 1;
    }

    function histogram() {
      var value = identity$1,
          domain = extent,
          threshold = thresholdSturges;

      function histogram(data) {
        var i,
            n = data.length,
            x,
            values = new Array(n);

        for (i = 0; i < n; ++i) {
          values[i] = value(data[i], i, data);
        }

        var xz = domain(values),
            x0 = xz[0],
            x1 = xz[1],
            tz = threshold(values, x0, x1);

        // Convert number of thresholds into uniform thresholds.
        if (!Array.isArray(tz)) {
          tz = tickStep(x0, x1, tz);
          tz = sequence$1(Math.ceil(x0 / tz) * tz, x1, tz); // exclusive
        }

        // Remove any thresholds outside the domain.
        var m = tz.length;
        while (tz[0] <= x0) tz.shift(), --m;
        while (tz[m - 1] > x1) tz.pop(), --m;

        var bins = new Array(m + 1),
            bin;

        // Initialize bins.
        for (i = 0; i <= m; ++i) {
          bin = bins[i] = [];
          bin.x0 = i > 0 ? tz[i - 1] : x0;
          bin.x1 = i < m ? tz[i] : x1;
        }

        // Assign data to bins by value, ignoring any outside the domain.
        for (i = 0; i < n; ++i) {
          x = values[i];
          if (x0 <= x && x <= x1) {
            bins[bisectRight(tz, x, 0, m)].push(data[i]);
          }
        }

        return bins;
      }

      histogram.value = function(_) {
        return arguments.length ? (value = typeof _ === "function" ? _ : constant(_), histogram) : value;
      };

      histogram.domain = function(_) {
        return arguments.length ? (domain = typeof _ === "function" ? _ : constant([_[0], _[1]]), histogram) : domain;
      };

      histogram.thresholds = function(_) {
        return arguments.length ? (threshold = typeof _ === "function" ? _ : Array.isArray(_) ? constant(slice.call(_)) : constant(_), histogram) : threshold;
      };

      return histogram;
    }

    var slice$1 = Array.prototype.slice;

    function identity$2(x) {
      return x;
    }

    var top = 1,
        right = 2,
        bottom = 3,
        left = 4,
        epsilon = 1e-6;

    function translateX(x) {
      return "translate(" + (x + 0.5) + ",0)";
    }

    function translateY(y) {
      return "translate(0," + (y + 0.5) + ")";
    }

    function number(scale) {
      return function(d) {
        return +scale(d);
      };
    }

    function center(scale) {
      var offset = Math.max(0, scale.bandwidth() - 1) / 2; // Adjust for 0.5px offset.
      if (scale.round()) offset = Math.round(offset);
      return function(d) {
        return +scale(d) + offset;
      };
    }

    function entering() {
      return !this.__axis;
    }

    function axis(orient, scale) {
      var tickArguments = [],
          tickValues = null,
          tickFormat = null,
          tickSizeInner = 6,
          tickSizeOuter = 6,
          tickPadding = 3,
          k = orient === top || orient === left ? -1 : 1,
          x = orient === left || orient === right ? "x" : "y",
          transform = orient === top || orient === bottom ? translateX : translateY;

      function axis(context) {
        var values = tickValues == null ? (scale.ticks ? scale.ticks.apply(scale, tickArguments) : scale.domain()) : tickValues,
            format = tickFormat == null ? (scale.tickFormat ? scale.tickFormat.apply(scale, tickArguments) : identity$2) : tickFormat,
            spacing = Math.max(tickSizeInner, 0) + tickPadding,
            range = scale.range(),
            range0 = +range[0] + 0.5,
            range1 = +range[range.length - 1] + 0.5,
            position = (scale.bandwidth ? center : number)(scale.copy()),
            selection = context.selection ? context.selection() : context,
            path = selection.selectAll(".domain").data([null]),
            tick = selection.selectAll(".tick").data(values, scale).order(),
            tickExit = tick.exit(),
            tickEnter = tick.enter().append("g").attr("class", "tick"),
            line = tick.select("line"),
            text = tick.select("text");

        path = path.merge(path.enter().insert("path", ".tick")
            .attr("class", "domain")
            .attr("stroke", "currentColor"));

        tick = tick.merge(tickEnter);

        line = line.merge(tickEnter.append("line")
            .attr("stroke", "currentColor")
            .attr(x + "2", k * tickSizeInner));

        text = text.merge(tickEnter.append("text")
            .attr("fill", "currentColor")
            .attr(x, k * spacing)
            .attr("dy", orient === top ? "0em" : orient === bottom ? "0.71em" : "0.32em"));

        if (context !== selection) {
          path = path.transition(context);
          tick = tick.transition(context);
          line = line.transition(context);
          text = text.transition(context);

          tickExit = tickExit.transition(context)
              .attr("opacity", epsilon)
              .attr("transform", function(d) { return isFinite(d = position(d)) ? transform(d) : this.getAttribute("transform"); });

          tickEnter
              .attr("opacity", epsilon)
              .attr("transform", function(d) { var p = this.parentNode.__axis; return transform(p && isFinite(p = p(d)) ? p : position(d)); });
        }

        tickExit.remove();

        path
            .attr("d", orient === left || orient == right
                ? (tickSizeOuter ? "M" + k * tickSizeOuter + "," + range0 + "H0.5V" + range1 + "H" + k * tickSizeOuter : "M0.5," + range0 + "V" + range1)
                : (tickSizeOuter ? "M" + range0 + "," + k * tickSizeOuter + "V0.5H" + range1 + "V" + k * tickSizeOuter : "M" + range0 + ",0.5H" + range1));

        tick
            .attr("opacity", 1)
            .attr("transform", function(d) { return transform(position(d)); });

        line
            .attr(x + "2", k * tickSizeInner);

        text
            .attr(x, k * spacing)
            .text(format);

        selection.filter(entering)
            .attr("fill", "none")
            .attr("font-size", 10)
            .attr("font-family", "sans-serif")
            .attr("text-anchor", orient === right ? "start" : orient === left ? "end" : "middle");

        selection
            .each(function() { this.__axis = position; });
      }

      axis.scale = function(_) {
        return arguments.length ? (scale = _, axis) : scale;
      };

      axis.ticks = function() {
        return tickArguments = slice$1.call(arguments), axis;
      };

      axis.tickArguments = function(_) {
        return arguments.length ? (tickArguments = _ == null ? [] : slice$1.call(_), axis) : tickArguments.slice();
      };

      axis.tickValues = function(_) {
        return arguments.length ? (tickValues = _ == null ? null : slice$1.call(_), axis) : tickValues && tickValues.slice();
      };

      axis.tickFormat = function(_) {
        return arguments.length ? (tickFormat = _, axis) : tickFormat;
      };

      axis.tickSize = function(_) {
        return arguments.length ? (tickSizeInner = tickSizeOuter = +_, axis) : tickSizeInner;
      };

      axis.tickSizeInner = function(_) {
        return arguments.length ? (tickSizeInner = +_, axis) : tickSizeInner;
      };

      axis.tickSizeOuter = function(_) {
        return arguments.length ? (tickSizeOuter = +_, axis) : tickSizeOuter;
      };

      axis.tickPadding = function(_) {
        return arguments.length ? (tickPadding = +_, axis) : tickPadding;
      };

      return axis;
    }

    function axisRight(scale) {
      return axis(right, scale);
    }

    function axisBottom(scale) {
      return axis(bottom, scale);
    }

    function axisLeft(scale) {
      return axis(left, scale);
    }

    var noop$2 = {value: function() {}};

    function dispatch$3() {
      for (var i = 0, n = arguments.length, _ = {}, t; i < n; ++i) {
        if (!(t = arguments[i] + "") || (t in _) || /[\s.]/.test(t)) throw new Error("illegal type: " + t);
        _[t] = [];
      }
      return new Dispatch(_);
    }

    function Dispatch(_) {
      this._ = _;
    }

    function parseTypenames(typenames, types) {
      return typenames.trim().split(/^|\s+/).map(function(t) {
        var name = "", i = t.indexOf(".");
        if (i >= 0) name = t.slice(i + 1), t = t.slice(0, i);
        if (t && !types.hasOwnProperty(t)) throw new Error("unknown type: " + t);
        return {type: t, name: name};
      });
    }

    Dispatch.prototype = dispatch$3.prototype = {
      constructor: Dispatch,
      on: function(typename, callback) {
        var _ = this._,
            T = parseTypenames(typename + "", _),
            t,
            i = -1,
            n = T.length;

        // If no callback was specified, return the callback of the given type and name.
        if (arguments.length < 2) {
          while (++i < n) if ((t = (typename = T[i]).type) && (t = get(_[t], typename.name))) return t;
          return;
        }

        // If a type was specified, set the callback for the given type and name.
        // Otherwise, if a null callback was specified, remove callbacks of the given name.
        if (callback != null && typeof callback !== "function") throw new Error("invalid callback: " + callback);
        while (++i < n) {
          if (t = (typename = T[i]).type) _[t] = set(_[t], typename.name, callback);
          else if (callback == null) for (t in _) _[t] = set(_[t], typename.name, null);
        }

        return this;
      },
      copy: function() {
        var copy = {}, _ = this._;
        for (var t in _) copy[t] = _[t].slice();
        return new Dispatch(copy);
      },
      call: function(type, that) {
        if ((n = arguments.length - 2) > 0) for (var args = new Array(n), i = 0, n, t; i < n; ++i) args[i] = arguments[i + 2];
        if (!this._.hasOwnProperty(type)) throw new Error("unknown type: " + type);
        for (t = this._[type], i = 0, n = t.length; i < n; ++i) t[i].value.apply(that, args);
      },
      apply: function(type, that, args) {
        if (!this._.hasOwnProperty(type)) throw new Error("unknown type: " + type);
        for (var t = this._[type], i = 0, n = t.length; i < n; ++i) t[i].value.apply(that, args);
      }
    };

    function get(type, name) {
      for (var i = 0, n = type.length, c; i < n; ++i) {
        if ((c = type[i]).name === name) {
          return c.value;
        }
      }
    }

    function set(type, name, callback) {
      for (var i = 0, n = type.length; i < n; ++i) {
        if (type[i].name === name) {
          type[i] = noop$2, type = type.slice(0, i).concat(type.slice(i + 1));
          break;
        }
      }
      if (callback != null) type.push({name: name, value: callback});
      return type;
    }

    var xhtml = "http://www.w3.org/1999/xhtml";

    var namespaces = {
      svg: "http://www.w3.org/2000/svg",
      xhtml: xhtml,
      xlink: "http://www.w3.org/1999/xlink",
      xml: "http://www.w3.org/XML/1998/namespace",
      xmlns: "http://www.w3.org/2000/xmlns/"
    };

    function namespace(name) {
      var prefix = name += "", i = prefix.indexOf(":");
      if (i >= 0 && (prefix = name.slice(0, i)) !== "xmlns") name = name.slice(i + 1);
      return namespaces.hasOwnProperty(prefix) ? {space: namespaces[prefix], local: name} : name;
    }

    function creatorInherit(name) {
      return function() {
        var document = this.ownerDocument,
            uri = this.namespaceURI;
        return uri === xhtml && document.documentElement.namespaceURI === xhtml
            ? document.createElement(name)
            : document.createElementNS(uri, name);
      };
    }

    function creatorFixed(fullname) {
      return function() {
        return this.ownerDocument.createElementNS(fullname.space, fullname.local);
      };
    }

    function creator(name) {
      var fullname = namespace(name);
      return (fullname.local
          ? creatorFixed
          : creatorInherit)(fullname);
    }

    function none() {}

    function selector(selector) {
      return selector == null ? none : function() {
        return this.querySelector(selector);
      };
    }

    function selection_select(select) {
      if (typeof select !== "function") select = selector(select);

      for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, subgroup = subgroups[j] = new Array(n), node, subnode, i = 0; i < n; ++i) {
          if ((node = group[i]) && (subnode = select.call(node, node.__data__, i, group))) {
            if ("__data__" in node) subnode.__data__ = node.__data__;
            subgroup[i] = subnode;
          }
        }
      }

      return new Selection(subgroups, this._parents);
    }

    function empty$1() {
      return [];
    }

    function selectorAll(selector) {
      return selector == null ? empty$1 : function() {
        return this.querySelectorAll(selector);
      };
    }

    function selection_selectAll(select) {
      if (typeof select !== "function") select = selectorAll(select);

      for (var groups = this._groups, m = groups.length, subgroups = [], parents = [], j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, node, i = 0; i < n; ++i) {
          if (node = group[i]) {
            subgroups.push(select.call(node, node.__data__, i, group));
            parents.push(node);
          }
        }
      }

      return new Selection(subgroups, parents);
    }

    function matcher(selector) {
      return function() {
        return this.matches(selector);
      };
    }

    function selection_filter(match) {
      if (typeof match !== "function") match = matcher(match);

      for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, subgroup = subgroups[j] = [], node, i = 0; i < n; ++i) {
          if ((node = group[i]) && match.call(node, node.__data__, i, group)) {
            subgroup.push(node);
          }
        }
      }

      return new Selection(subgroups, this._parents);
    }

    function sparse(update) {
      return new Array(update.length);
    }

    function selection_enter() {
      return new Selection(this._enter || this._groups.map(sparse), this._parents);
    }

    function EnterNode(parent, datum) {
      this.ownerDocument = parent.ownerDocument;
      this.namespaceURI = parent.namespaceURI;
      this._next = null;
      this._parent = parent;
      this.__data__ = datum;
    }

    EnterNode.prototype = {
      constructor: EnterNode,
      appendChild: function(child) { return this._parent.insertBefore(child, this._next); },
      insertBefore: function(child, next) { return this._parent.insertBefore(child, next); },
      querySelector: function(selector) { return this._parent.querySelector(selector); },
      querySelectorAll: function(selector) { return this._parent.querySelectorAll(selector); }
    };

    function constant$1(x) {
      return function() {
        return x;
      };
    }

    var keyPrefix = "$"; // Protect against keys like “__proto__”.

    function bindIndex(parent, group, enter, update, exit, data) {
      var i = 0,
          node,
          groupLength = group.length,
          dataLength = data.length;

      // Put any non-null nodes that fit into update.
      // Put any null nodes into enter.
      // Put any remaining data into enter.
      for (; i < dataLength; ++i) {
        if (node = group[i]) {
          node.__data__ = data[i];
          update[i] = node;
        } else {
          enter[i] = new EnterNode(parent, data[i]);
        }
      }

      // Put any non-null nodes that don’t fit into exit.
      for (; i < groupLength; ++i) {
        if (node = group[i]) {
          exit[i] = node;
        }
      }
    }

    function bindKey(parent, group, enter, update, exit, data, key) {
      var i,
          node,
          nodeByKeyValue = {},
          groupLength = group.length,
          dataLength = data.length,
          keyValues = new Array(groupLength),
          keyValue;

      // Compute the key for each node.
      // If multiple nodes have the same key, the duplicates are added to exit.
      for (i = 0; i < groupLength; ++i) {
        if (node = group[i]) {
          keyValues[i] = keyValue = keyPrefix + key.call(node, node.__data__, i, group);
          if (keyValue in nodeByKeyValue) {
            exit[i] = node;
          } else {
            nodeByKeyValue[keyValue] = node;
          }
        }
      }

      // Compute the key for each datum.
      // If there a node associated with this key, join and add it to update.
      // If there is not (or the key is a duplicate), add it to enter.
      for (i = 0; i < dataLength; ++i) {
        keyValue = keyPrefix + key.call(parent, data[i], i, data);
        if (node = nodeByKeyValue[keyValue]) {
          update[i] = node;
          node.__data__ = data[i];
          nodeByKeyValue[keyValue] = null;
        } else {
          enter[i] = new EnterNode(parent, data[i]);
        }
      }

      // Add any remaining nodes that were not bound to data to exit.
      for (i = 0; i < groupLength; ++i) {
        if ((node = group[i]) && (nodeByKeyValue[keyValues[i]] === node)) {
          exit[i] = node;
        }
      }
    }

    function selection_data(value, key) {
      if (!value) {
        data = new Array(this.size()), j = -1;
        this.each(function(d) { data[++j] = d; });
        return data;
      }

      var bind = key ? bindKey : bindIndex,
          parents = this._parents,
          groups = this._groups;

      if (typeof value !== "function") value = constant$1(value);

      for (var m = groups.length, update = new Array(m), enter = new Array(m), exit = new Array(m), j = 0; j < m; ++j) {
        var parent = parents[j],
            group = groups[j],
            groupLength = group.length,
            data = value.call(parent, parent && parent.__data__, j, parents),
            dataLength = data.length,
            enterGroup = enter[j] = new Array(dataLength),
            updateGroup = update[j] = new Array(dataLength),
            exitGroup = exit[j] = new Array(groupLength);

        bind(parent, group, enterGroup, updateGroup, exitGroup, data, key);

        // Now connect the enter nodes to their following update node, such that
        // appendChild can insert the materialized enter node before this node,
        // rather than at the end of the parent node.
        for (var i0 = 0, i1 = 0, previous, next; i0 < dataLength; ++i0) {
          if (previous = enterGroup[i0]) {
            if (i0 >= i1) i1 = i0 + 1;
            while (!(next = updateGroup[i1]) && ++i1 < dataLength);
            previous._next = next || null;
          }
        }
      }

      update = new Selection(update, parents);
      update._enter = enter;
      update._exit = exit;
      return update;
    }

    function selection_exit() {
      return new Selection(this._exit || this._groups.map(sparse), this._parents);
    }

    function selection_join(onenter, onupdate, onexit) {
      var enter = this.enter(), update = this, exit = this.exit();
      enter = typeof onenter === "function" ? onenter(enter) : enter.append(onenter + "");
      if (onupdate != null) update = onupdate(update);
      if (onexit == null) exit.remove(); else onexit(exit);
      return enter && update ? enter.merge(update).order() : update;
    }

    function selection_merge(selection) {

      for (var groups0 = this._groups, groups1 = selection._groups, m0 = groups0.length, m1 = groups1.length, m = Math.min(m0, m1), merges = new Array(m0), j = 0; j < m; ++j) {
        for (var group0 = groups0[j], group1 = groups1[j], n = group0.length, merge = merges[j] = new Array(n), node, i = 0; i < n; ++i) {
          if (node = group0[i] || group1[i]) {
            merge[i] = node;
          }
        }
      }

      for (; j < m0; ++j) {
        merges[j] = groups0[j];
      }

      return new Selection(merges, this._parents);
    }

    function selection_order() {

      for (var groups = this._groups, j = -1, m = groups.length; ++j < m;) {
        for (var group = groups[j], i = group.length - 1, next = group[i], node; --i >= 0;) {
          if (node = group[i]) {
            if (next && node.compareDocumentPosition(next) ^ 4) next.parentNode.insertBefore(node, next);
            next = node;
          }
        }
      }

      return this;
    }

    function selection_sort(compare) {
      if (!compare) compare = ascending$1;

      function compareNode(a, b) {
        return a && b ? compare(a.__data__, b.__data__) : !a - !b;
      }

      for (var groups = this._groups, m = groups.length, sortgroups = new Array(m), j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, sortgroup = sortgroups[j] = new Array(n), node, i = 0; i < n; ++i) {
          if (node = group[i]) {
            sortgroup[i] = node;
          }
        }
        sortgroup.sort(compareNode);
      }

      return new Selection(sortgroups, this._parents).order();
    }

    function ascending$1(a, b) {
      return a < b ? -1 : a > b ? 1 : a >= b ? 0 : NaN;
    }

    function selection_call() {
      var callback = arguments[0];
      arguments[0] = this;
      callback.apply(null, arguments);
      return this;
    }

    function selection_nodes() {
      var nodes = new Array(this.size()), i = -1;
      this.each(function() { nodes[++i] = this; });
      return nodes;
    }

    function selection_node() {

      for (var groups = this._groups, j = 0, m = groups.length; j < m; ++j) {
        for (var group = groups[j], i = 0, n = group.length; i < n; ++i) {
          var node = group[i];
          if (node) return node;
        }
      }

      return null;
    }

    function selection_size() {
      var size = 0;
      this.each(function() { ++size; });
      return size;
    }

    function selection_empty() {
      return !this.node();
    }

    function selection_each(callback) {

      for (var groups = this._groups, j = 0, m = groups.length; j < m; ++j) {
        for (var group = groups[j], i = 0, n = group.length, node; i < n; ++i) {
          if (node = group[i]) callback.call(node, node.__data__, i, group);
        }
      }

      return this;
    }

    function attrRemove(name) {
      return function() {
        this.removeAttribute(name);
      };
    }

    function attrRemoveNS(fullname) {
      return function() {
        this.removeAttributeNS(fullname.space, fullname.local);
      };
    }

    function attrConstant(name, value) {
      return function() {
        this.setAttribute(name, value);
      };
    }

    function attrConstantNS(fullname, value) {
      return function() {
        this.setAttributeNS(fullname.space, fullname.local, value);
      };
    }

    function attrFunction(name, value) {
      return function() {
        var v = value.apply(this, arguments);
        if (v == null) this.removeAttribute(name);
        else this.setAttribute(name, v);
      };
    }

    function attrFunctionNS(fullname, value) {
      return function() {
        var v = value.apply(this, arguments);
        if (v == null) this.removeAttributeNS(fullname.space, fullname.local);
        else this.setAttributeNS(fullname.space, fullname.local, v);
      };
    }

    function selection_attr(name, value) {
      var fullname = namespace(name);

      if (arguments.length < 2) {
        var node = this.node();
        return fullname.local
            ? node.getAttributeNS(fullname.space, fullname.local)
            : node.getAttribute(fullname);
      }

      return this.each((value == null
          ? (fullname.local ? attrRemoveNS : attrRemove) : (typeof value === "function"
          ? (fullname.local ? attrFunctionNS : attrFunction)
          : (fullname.local ? attrConstantNS : attrConstant)))(fullname, value));
    }

    function defaultView(node) {
      return (node.ownerDocument && node.ownerDocument.defaultView) // node is a Node
          || (node.document && node) // node is a Window
          || node.defaultView; // node is a Document
    }

    function styleRemove(name) {
      return function() {
        this.style.removeProperty(name);
      };
    }

    function styleConstant(name, value, priority) {
      return function() {
        this.style.setProperty(name, value, priority);
      };
    }

    function styleFunction(name, value, priority) {
      return function() {
        var v = value.apply(this, arguments);
        if (v == null) this.style.removeProperty(name);
        else this.style.setProperty(name, v, priority);
      };
    }

    function selection_style(name, value, priority) {
      return arguments.length > 1
          ? this.each((value == null
                ? styleRemove : typeof value === "function"
                ? styleFunction
                : styleConstant)(name, value, priority == null ? "" : priority))
          : styleValue(this.node(), name);
    }

    function styleValue(node, name) {
      return node.style.getPropertyValue(name)
          || defaultView(node).getComputedStyle(node, null).getPropertyValue(name);
    }

    function propertyRemove(name) {
      return function() {
        delete this[name];
      };
    }

    function propertyConstant(name, value) {
      return function() {
        this[name] = value;
      };
    }

    function propertyFunction(name, value) {
      return function() {
        var v = value.apply(this, arguments);
        if (v == null) delete this[name];
        else this[name] = v;
      };
    }

    function selection_property(name, value) {
      return arguments.length > 1
          ? this.each((value == null
              ? propertyRemove : typeof value === "function"
              ? propertyFunction
              : propertyConstant)(name, value))
          : this.node()[name];
    }

    function classArray(string) {
      return string.trim().split(/^|\s+/);
    }

    function classList(node) {
      return node.classList || new ClassList(node);
    }

    function ClassList(node) {
      this._node = node;
      this._names = classArray(node.getAttribute("class") || "");
    }

    ClassList.prototype = {
      add: function(name) {
        var i = this._names.indexOf(name);
        if (i < 0) {
          this._names.push(name);
          this._node.setAttribute("class", this._names.join(" "));
        }
      },
      remove: function(name) {
        var i = this._names.indexOf(name);
        if (i >= 0) {
          this._names.splice(i, 1);
          this._node.setAttribute("class", this._names.join(" "));
        }
      },
      contains: function(name) {
        return this._names.indexOf(name) >= 0;
      }
    };

    function classedAdd(node, names) {
      var list = classList(node), i = -1, n = names.length;
      while (++i < n) list.add(names[i]);
    }

    function classedRemove(node, names) {
      var list = classList(node), i = -1, n = names.length;
      while (++i < n) list.remove(names[i]);
    }

    function classedTrue(names) {
      return function() {
        classedAdd(this, names);
      };
    }

    function classedFalse(names) {
      return function() {
        classedRemove(this, names);
      };
    }

    function classedFunction(names, value) {
      return function() {
        (value.apply(this, arguments) ? classedAdd : classedRemove)(this, names);
      };
    }

    function selection_classed(name, value) {
      var names = classArray(name + "");

      if (arguments.length < 2) {
        var list = classList(this.node()), i = -1, n = names.length;
        while (++i < n) if (!list.contains(names[i])) return false;
        return true;
      }

      return this.each((typeof value === "function"
          ? classedFunction : value
          ? classedTrue
          : classedFalse)(names, value));
    }

    function textRemove() {
      this.textContent = "";
    }

    function textConstant(value) {
      return function() {
        this.textContent = value;
      };
    }

    function textFunction(value) {
      return function() {
        var v = value.apply(this, arguments);
        this.textContent = v == null ? "" : v;
      };
    }

    function selection_text(value) {
      return arguments.length
          ? this.each(value == null
              ? textRemove : (typeof value === "function"
              ? textFunction
              : textConstant)(value))
          : this.node().textContent;
    }

    function htmlRemove() {
      this.innerHTML = "";
    }

    function htmlConstant(value) {
      return function() {
        this.innerHTML = value;
      };
    }

    function htmlFunction(value) {
      return function() {
        var v = value.apply(this, arguments);
        this.innerHTML = v == null ? "" : v;
      };
    }

    function selection_html(value) {
      return arguments.length
          ? this.each(value == null
              ? htmlRemove : (typeof value === "function"
              ? htmlFunction
              : htmlConstant)(value))
          : this.node().innerHTML;
    }

    function raise() {
      if (this.nextSibling) this.parentNode.appendChild(this);
    }

    function selection_raise() {
      return this.each(raise);
    }

    function lower() {
      if (this.previousSibling) this.parentNode.insertBefore(this, this.parentNode.firstChild);
    }

    function selection_lower() {
      return this.each(lower);
    }

    function selection_append(name) {
      var create = typeof name === "function" ? name : creator(name);
      return this.select(function() {
        return this.appendChild(create.apply(this, arguments));
      });
    }

    function constantNull() {
      return null;
    }

    function selection_insert(name, before) {
      var create = typeof name === "function" ? name : creator(name),
          select = before == null ? constantNull : typeof before === "function" ? before : selector(before);
      return this.select(function() {
        return this.insertBefore(create.apply(this, arguments), select.apply(this, arguments) || null);
      });
    }

    function remove$1() {
      var parent = this.parentNode;
      if (parent) parent.removeChild(this);
    }

    function selection_remove() {
      return this.each(remove$1);
    }

    function selection_cloneShallow() {
      var clone = this.cloneNode(false), parent = this.parentNode;
      return parent ? parent.insertBefore(clone, this.nextSibling) : clone;
    }

    function selection_cloneDeep() {
      var clone = this.cloneNode(true), parent = this.parentNode;
      return parent ? parent.insertBefore(clone, this.nextSibling) : clone;
    }

    function selection_clone(deep) {
      return this.select(deep ? selection_cloneDeep : selection_cloneShallow);
    }

    function selection_datum(value) {
      return arguments.length
          ? this.property("__data__", value)
          : this.node().__data__;
    }

    var filterEvents = {};

    var event = null;

    if (typeof document !== "undefined") {
      var element = document.documentElement;
      if (!("onmouseenter" in element)) {
        filterEvents = {mouseenter: "mouseover", mouseleave: "mouseout"};
      }
    }

    function filterContextListener(listener, index, group) {
      listener = contextListener(listener, index, group);
      return function(event) {
        var related = event.relatedTarget;
        if (!related || (related !== this && !(related.compareDocumentPosition(this) & 8))) {
          listener.call(this, event);
        }
      };
    }

    function contextListener(listener, index, group) {
      return function(event1) {
        var event0 = event; // Events can be reentrant (e.g., focus).
        event = event1;
        try {
          listener.call(this, this.__data__, index, group);
        } finally {
          event = event0;
        }
      };
    }

    function parseTypenames$1(typenames) {
      return typenames.trim().split(/^|\s+/).map(function(t) {
        var name = "", i = t.indexOf(".");
        if (i >= 0) name = t.slice(i + 1), t = t.slice(0, i);
        return {type: t, name: name};
      });
    }

    function onRemove(typename) {
      return function() {
        var on = this.__on;
        if (!on) return;
        for (var j = 0, i = -1, m = on.length, o; j < m; ++j) {
          if (o = on[j], (!typename.type || o.type === typename.type) && o.name === typename.name) {
            this.removeEventListener(o.type, o.listener, o.capture);
          } else {
            on[++i] = o;
          }
        }
        if (++i) on.length = i;
        else delete this.__on;
      };
    }

    function onAdd(typename, value, capture) {
      var wrap = filterEvents.hasOwnProperty(typename.type) ? filterContextListener : contextListener;
      return function(d, i, group) {
        var on = this.__on, o, listener = wrap(value, i, group);
        if (on) for (var j = 0, m = on.length; j < m; ++j) {
          if ((o = on[j]).type === typename.type && o.name === typename.name) {
            this.removeEventListener(o.type, o.listener, o.capture);
            this.addEventListener(o.type, o.listener = listener, o.capture = capture);
            o.value = value;
            return;
          }
        }
        this.addEventListener(typename.type, listener, capture);
        o = {type: typename.type, name: typename.name, value: value, listener: listener, capture: capture};
        if (!on) this.__on = [o];
        else on.push(o);
      };
    }

    function selection_on(typename, value, capture) {
      var typenames = parseTypenames$1(typename + ""), i, n = typenames.length, t;

      if (arguments.length < 2) {
        var on = this.node().__on;
        if (on) for (var j = 0, m = on.length, o; j < m; ++j) {
          for (i = 0, o = on[j]; i < n; ++i) {
            if ((t = typenames[i]).type === o.type && t.name === o.name) {
              return o.value;
            }
          }
        }
        return;
      }

      on = value ? onAdd : onRemove;
      if (capture == null) capture = false;
      for (i = 0; i < n; ++i) this.each(on(typenames[i], value, capture));
      return this;
    }

    function customEvent(event1, listener, that, args) {
      var event0 = event;
      event1.sourceEvent = event;
      event = event1;
      try {
        return listener.apply(that, args);
      } finally {
        event = event0;
      }
    }

    function dispatchEvent(node, type, params) {
      var window = defaultView(node),
          event = window.CustomEvent;

      if (typeof event === "function") {
        event = new event(type, params);
      } else {
        event = window.document.createEvent("Event");
        if (params) event.initEvent(type, params.bubbles, params.cancelable), event.detail = params.detail;
        else event.initEvent(type, false, false);
      }

      node.dispatchEvent(event);
    }

    function dispatchConstant(type, params) {
      return function() {
        return dispatchEvent(this, type, params);
      };
    }

    function dispatchFunction(type, params) {
      return function() {
        return dispatchEvent(this, type, params.apply(this, arguments));
      };
    }

    function selection_dispatch(type, params) {
      return this.each((typeof params === "function"
          ? dispatchFunction
          : dispatchConstant)(type, params));
    }

    var root = [null];

    function Selection(groups, parents) {
      this._groups = groups;
      this._parents = parents;
    }

    function selection() {
      return new Selection([[document.documentElement]], root);
    }

    Selection.prototype = selection.prototype = {
      constructor: Selection,
      select: selection_select,
      selectAll: selection_selectAll,
      filter: selection_filter,
      data: selection_data,
      enter: selection_enter,
      exit: selection_exit,
      join: selection_join,
      merge: selection_merge,
      order: selection_order,
      sort: selection_sort,
      call: selection_call,
      nodes: selection_nodes,
      node: selection_node,
      size: selection_size,
      empty: selection_empty,
      each: selection_each,
      attr: selection_attr,
      style: selection_style,
      property: selection_property,
      classed: selection_classed,
      text: selection_text,
      html: selection_html,
      raise: selection_raise,
      lower: selection_lower,
      append: selection_append,
      insert: selection_insert,
      remove: selection_remove,
      clone: selection_clone,
      datum: selection_datum,
      on: selection_on,
      dispatch: selection_dispatch
    };

    function select$1(selector) {
      return typeof selector === "string"
          ? new Selection([[document.querySelector(selector)]], [document.documentElement])
          : new Selection([[selector]], root);
    }

    function sourceEvent() {
      var current = event, source;
      while (source = current.sourceEvent) current = source;
      return current;
    }

    function point(node, event) {
      var svg = node.ownerSVGElement || node;

      if (svg.createSVGPoint) {
        var point = svg.createSVGPoint();
        point.x = event.clientX, point.y = event.clientY;
        point = point.matrixTransform(node.getScreenCTM().inverse());
        return [point.x, point.y];
      }

      var rect = node.getBoundingClientRect();
      return [event.clientX - rect.left - node.clientLeft, event.clientY - rect.top - node.clientTop];
    }

    function mouse(node) {
      var event = sourceEvent();
      if (event.changedTouches) event = event.changedTouches[0];
      return point(node, event);
    }

    function touch(node, touches, identifier) {
      if (arguments.length < 3) identifier = touches, touches = sourceEvent().changedTouches;

      for (var i = 0, n = touches ? touches.length : 0, touch; i < n; ++i) {
        if ((touch = touches[i]).identifier === identifier) {
          return point(node, touch);
        }
      }

      return null;
    }

    function noevent() {
      event.preventDefault();
      event.stopImmediatePropagation();
    }

    function dragDisable(view) {
      var root = view.document.documentElement,
          selection = select$1(view).on("dragstart.drag", noevent, true);
      if ("onselectstart" in root) {
        selection.on("selectstart.drag", noevent, true);
      } else {
        root.__noselect = root.style.MozUserSelect;
        root.style.MozUserSelect = "none";
      }
    }

    function yesdrag(view, noclick) {
      var root = view.document.documentElement,
          selection = select$1(view).on("dragstart.drag", null);
      if (noclick) {
        selection.on("click.drag", noevent, true);
        setTimeout(function() { selection.on("click.drag", null); }, 0);
      }
      if ("onselectstart" in root) {
        selection.on("selectstart.drag", null);
      } else {
        root.style.MozUserSelect = root.__noselect;
        delete root.__noselect;
      }
    }

    function define(constructor, factory, prototype) {
      constructor.prototype = factory.prototype = prototype;
      prototype.constructor = constructor;
    }

    function extend(parent, definition) {
      var prototype = Object.create(parent.prototype);
      for (var key in definition) prototype[key] = definition[key];
      return prototype;
    }

    function Color() {}

    var darker = 0.7;
    var brighter = 1 / darker;

    var reI = "\\s*([+-]?\\d+)\\s*",
        reN = "\\s*([+-]?\\d*\\.?\\d+(?:[eE][+-]?\\d+)?)\\s*",
        reP = "\\s*([+-]?\\d*\\.?\\d+(?:[eE][+-]?\\d+)?)%\\s*",
        reHex = /^#([0-9a-f]{3,8})$/,
        reRgbInteger = new RegExp("^rgb\\(" + [reI, reI, reI] + "\\)$"),
        reRgbPercent = new RegExp("^rgb\\(" + [reP, reP, reP] + "\\)$"),
        reRgbaInteger = new RegExp("^rgba\\(" + [reI, reI, reI, reN] + "\\)$"),
        reRgbaPercent = new RegExp("^rgba\\(" + [reP, reP, reP, reN] + "\\)$"),
        reHslPercent = new RegExp("^hsl\\(" + [reN, reP, reP] + "\\)$"),
        reHslaPercent = new RegExp("^hsla\\(" + [reN, reP, reP, reN] + "\\)$");

    var named = {
      aliceblue: 0xf0f8ff,
      antiquewhite: 0xfaebd7,
      aqua: 0x00ffff,
      aquamarine: 0x7fffd4,
      azure: 0xf0ffff,
      beige: 0xf5f5dc,
      bisque: 0xffe4c4,
      black: 0x000000,
      blanchedalmond: 0xffebcd,
      blue: 0x0000ff,
      blueviolet: 0x8a2be2,
      brown: 0xa52a2a,
      burlywood: 0xdeb887,
      cadetblue: 0x5f9ea0,
      chartreuse: 0x7fff00,
      chocolate: 0xd2691e,
      coral: 0xff7f50,
      cornflowerblue: 0x6495ed,
      cornsilk: 0xfff8dc,
      crimson: 0xdc143c,
      cyan: 0x00ffff,
      darkblue: 0x00008b,
      darkcyan: 0x008b8b,
      darkgoldenrod: 0xb8860b,
      darkgray: 0xa9a9a9,
      darkgreen: 0x006400,
      darkgrey: 0xa9a9a9,
      darkkhaki: 0xbdb76b,
      darkmagenta: 0x8b008b,
      darkolivegreen: 0x556b2f,
      darkorange: 0xff8c00,
      darkorchid: 0x9932cc,
      darkred: 0x8b0000,
      darksalmon: 0xe9967a,
      darkseagreen: 0x8fbc8f,
      darkslateblue: 0x483d8b,
      darkslategray: 0x2f4f4f,
      darkslategrey: 0x2f4f4f,
      darkturquoise: 0x00ced1,
      darkviolet: 0x9400d3,
      deeppink: 0xff1493,
      deepskyblue: 0x00bfff,
      dimgray: 0x696969,
      dimgrey: 0x696969,
      dodgerblue: 0x1e90ff,
      firebrick: 0xb22222,
      floralwhite: 0xfffaf0,
      forestgreen: 0x228b22,
      fuchsia: 0xff00ff,
      gainsboro: 0xdcdcdc,
      ghostwhite: 0xf8f8ff,
      gold: 0xffd700,
      goldenrod: 0xdaa520,
      gray: 0x808080,
      green: 0x008000,
      greenyellow: 0xadff2f,
      grey: 0x808080,
      honeydew: 0xf0fff0,
      hotpink: 0xff69b4,
      indianred: 0xcd5c5c,
      indigo: 0x4b0082,
      ivory: 0xfffff0,
      khaki: 0xf0e68c,
      lavender: 0xe6e6fa,
      lavenderblush: 0xfff0f5,
      lawngreen: 0x7cfc00,
      lemonchiffon: 0xfffacd,
      lightblue: 0xadd8e6,
      lightcoral: 0xf08080,
      lightcyan: 0xe0ffff,
      lightgoldenrodyellow: 0xfafad2,
      lightgray: 0xd3d3d3,
      lightgreen: 0x90ee90,
      lightgrey: 0xd3d3d3,
      lightpink: 0xffb6c1,
      lightsalmon: 0xffa07a,
      lightseagreen: 0x20b2aa,
      lightskyblue: 0x87cefa,
      lightslategray: 0x778899,
      lightslategrey: 0x778899,
      lightsteelblue: 0xb0c4de,
      lightyellow: 0xffffe0,
      lime: 0x00ff00,
      limegreen: 0x32cd32,
      linen: 0xfaf0e6,
      magenta: 0xff00ff,
      maroon: 0x800000,
      mediumaquamarine: 0x66cdaa,
      mediumblue: 0x0000cd,
      mediumorchid: 0xba55d3,
      mediumpurple: 0x9370db,
      mediumseagreen: 0x3cb371,
      mediumslateblue: 0x7b68ee,
      mediumspringgreen: 0x00fa9a,
      mediumturquoise: 0x48d1cc,
      mediumvioletred: 0xc71585,
      midnightblue: 0x191970,
      mintcream: 0xf5fffa,
      mistyrose: 0xffe4e1,
      moccasin: 0xffe4b5,
      navajowhite: 0xffdead,
      navy: 0x000080,
      oldlace: 0xfdf5e6,
      olive: 0x808000,
      olivedrab: 0x6b8e23,
      orange: 0xffa500,
      orangered: 0xff4500,
      orchid: 0xda70d6,
      palegoldenrod: 0xeee8aa,
      palegreen: 0x98fb98,
      paleturquoise: 0xafeeee,
      palevioletred: 0xdb7093,
      papayawhip: 0xffefd5,
      peachpuff: 0xffdab9,
      peru: 0xcd853f,
      pink: 0xffc0cb,
      plum: 0xdda0dd,
      powderblue: 0xb0e0e6,
      purple: 0x800080,
      rebeccapurple: 0x663399,
      red: 0xff0000,
      rosybrown: 0xbc8f8f,
      royalblue: 0x4169e1,
      saddlebrown: 0x8b4513,
      salmon: 0xfa8072,
      sandybrown: 0xf4a460,
      seagreen: 0x2e8b57,
      seashell: 0xfff5ee,
      sienna: 0xa0522d,
      silver: 0xc0c0c0,
      skyblue: 0x87ceeb,
      slateblue: 0x6a5acd,
      slategray: 0x708090,
      slategrey: 0x708090,
      snow: 0xfffafa,
      springgreen: 0x00ff7f,
      steelblue: 0x4682b4,
      tan: 0xd2b48c,
      teal: 0x008080,
      thistle: 0xd8bfd8,
      tomato: 0xff6347,
      turquoise: 0x40e0d0,
      violet: 0xee82ee,
      wheat: 0xf5deb3,
      white: 0xffffff,
      whitesmoke: 0xf5f5f5,
      yellow: 0xffff00,
      yellowgreen: 0x9acd32
    };

    define(Color, color, {
      copy: function(channels) {
        return Object.assign(new this.constructor, this, channels);
      },
      displayable: function() {
        return this.rgb().displayable();
      },
      hex: color_formatHex, // Deprecated! Use color.formatHex.
      formatHex: color_formatHex,
      formatHsl: color_formatHsl,
      formatRgb: color_formatRgb,
      toString: color_formatRgb
    });

    function color_formatHex() {
      return this.rgb().formatHex();
    }

    function color_formatHsl() {
      return hslConvert(this).formatHsl();
    }

    function color_formatRgb() {
      return this.rgb().formatRgb();
    }

    function color(format) {
      var m, l;
      format = (format + "").trim().toLowerCase();
      return (m = reHex.exec(format)) ? (l = m[1].length, m = parseInt(m[1], 16), l === 6 ? rgbn(m) // #ff0000
          : l === 3 ? new Rgb((m >> 8 & 0xf) | (m >> 4 & 0xf0), (m >> 4 & 0xf) | (m & 0xf0), ((m & 0xf) << 4) | (m & 0xf), 1) // #f00
          : l === 8 ? rgba(m >> 24 & 0xff, m >> 16 & 0xff, m >> 8 & 0xff, (m & 0xff) / 0xff) // #ff000000
          : l === 4 ? rgba((m >> 12 & 0xf) | (m >> 8 & 0xf0), (m >> 8 & 0xf) | (m >> 4 & 0xf0), (m >> 4 & 0xf) | (m & 0xf0), (((m & 0xf) << 4) | (m & 0xf)) / 0xff) // #f000
          : null) // invalid hex
          : (m = reRgbInteger.exec(format)) ? new Rgb(m[1], m[2], m[3], 1) // rgb(255, 0, 0)
          : (m = reRgbPercent.exec(format)) ? new Rgb(m[1] * 255 / 100, m[2] * 255 / 100, m[3] * 255 / 100, 1) // rgb(100%, 0%, 0%)
          : (m = reRgbaInteger.exec(format)) ? rgba(m[1], m[2], m[3], m[4]) // rgba(255, 0, 0, 1)
          : (m = reRgbaPercent.exec(format)) ? rgba(m[1] * 255 / 100, m[2] * 255 / 100, m[3] * 255 / 100, m[4]) // rgb(100%, 0%, 0%, 1)
          : (m = reHslPercent.exec(format)) ? hsla(m[1], m[2] / 100, m[3] / 100, 1) // hsl(120, 50%, 50%)
          : (m = reHslaPercent.exec(format)) ? hsla(m[1], m[2] / 100, m[3] / 100, m[4]) // hsla(120, 50%, 50%, 1)
          : named.hasOwnProperty(format) ? rgbn(named[format]) // eslint-disable-line no-prototype-builtins
          : format === "transparent" ? new Rgb(NaN, NaN, NaN, 0)
          : null;
    }

    function rgbn(n) {
      return new Rgb(n >> 16 & 0xff, n >> 8 & 0xff, n & 0xff, 1);
    }

    function rgba(r, g, b, a) {
      if (a <= 0) r = g = b = NaN;
      return new Rgb(r, g, b, a);
    }

    function rgbConvert(o) {
      if (!(o instanceof Color)) o = color(o);
      if (!o) return new Rgb;
      o = o.rgb();
      return new Rgb(o.r, o.g, o.b, o.opacity);
    }

    function rgb(r, g, b, opacity) {
      return arguments.length === 1 ? rgbConvert(r) : new Rgb(r, g, b, opacity == null ? 1 : opacity);
    }

    function Rgb(r, g, b, opacity) {
      this.r = +r;
      this.g = +g;
      this.b = +b;
      this.opacity = +opacity;
    }

    define(Rgb, rgb, extend(Color, {
      brighter: function(k) {
        k = k == null ? brighter : Math.pow(brighter, k);
        return new Rgb(this.r * k, this.g * k, this.b * k, this.opacity);
      },
      darker: function(k) {
        k = k == null ? darker : Math.pow(darker, k);
        return new Rgb(this.r * k, this.g * k, this.b * k, this.opacity);
      },
      rgb: function() {
        return this;
      },
      displayable: function() {
        return (-0.5 <= this.r && this.r < 255.5)
            && (-0.5 <= this.g && this.g < 255.5)
            && (-0.5 <= this.b && this.b < 255.5)
            && (0 <= this.opacity && this.opacity <= 1);
      },
      hex: rgb_formatHex, // Deprecated! Use color.formatHex.
      formatHex: rgb_formatHex,
      formatRgb: rgb_formatRgb,
      toString: rgb_formatRgb
    }));

    function rgb_formatHex() {
      return "#" + hex(this.r) + hex(this.g) + hex(this.b);
    }

    function rgb_formatRgb() {
      var a = this.opacity; a = isNaN(a) ? 1 : Math.max(0, Math.min(1, a));
      return (a === 1 ? "rgb(" : "rgba(")
          + Math.max(0, Math.min(255, Math.round(this.r) || 0)) + ", "
          + Math.max(0, Math.min(255, Math.round(this.g) || 0)) + ", "
          + Math.max(0, Math.min(255, Math.round(this.b) || 0))
          + (a === 1 ? ")" : ", " + a + ")");
    }

    function hex(value) {
      value = Math.max(0, Math.min(255, Math.round(value) || 0));
      return (value < 16 ? "0" : "") + value.toString(16);
    }

    function hsla(h, s, l, a) {
      if (a <= 0) h = s = l = NaN;
      else if (l <= 0 || l >= 1) h = s = NaN;
      else if (s <= 0) h = NaN;
      return new Hsl(h, s, l, a);
    }

    function hslConvert(o) {
      if (o instanceof Hsl) return new Hsl(o.h, o.s, o.l, o.opacity);
      if (!(o instanceof Color)) o = color(o);
      if (!o) return new Hsl;
      if (o instanceof Hsl) return o;
      o = o.rgb();
      var r = o.r / 255,
          g = o.g / 255,
          b = o.b / 255,
          min = Math.min(r, g, b),
          max = Math.max(r, g, b),
          h = NaN,
          s = max - min,
          l = (max + min) / 2;
      if (s) {
        if (r === max) h = (g - b) / s + (g < b) * 6;
        else if (g === max) h = (b - r) / s + 2;
        else h = (r - g) / s + 4;
        s /= l < 0.5 ? max + min : 2 - max - min;
        h *= 60;
      } else {
        s = l > 0 && l < 1 ? 0 : h;
      }
      return new Hsl(h, s, l, o.opacity);
    }

    function hsl(h, s, l, opacity) {
      return arguments.length === 1 ? hslConvert(h) : new Hsl(h, s, l, opacity == null ? 1 : opacity);
    }

    function Hsl(h, s, l, opacity) {
      this.h = +h;
      this.s = +s;
      this.l = +l;
      this.opacity = +opacity;
    }

    define(Hsl, hsl, extend(Color, {
      brighter: function(k) {
        k = k == null ? brighter : Math.pow(brighter, k);
        return new Hsl(this.h, this.s, this.l * k, this.opacity);
      },
      darker: function(k) {
        k = k == null ? darker : Math.pow(darker, k);
        return new Hsl(this.h, this.s, this.l * k, this.opacity);
      },
      rgb: function() {
        var h = this.h % 360 + (this.h < 0) * 360,
            s = isNaN(h) || isNaN(this.s) ? 0 : this.s,
            l = this.l,
            m2 = l + (l < 0.5 ? l : 1 - l) * s,
            m1 = 2 * l - m2;
        return new Rgb(
          hsl2rgb(h >= 240 ? h - 240 : h + 120, m1, m2),
          hsl2rgb(h, m1, m2),
          hsl2rgb(h < 120 ? h + 240 : h - 120, m1, m2),
          this.opacity
        );
      },
      displayable: function() {
        return (0 <= this.s && this.s <= 1 || isNaN(this.s))
            && (0 <= this.l && this.l <= 1)
            && (0 <= this.opacity && this.opacity <= 1);
      },
      formatHsl: function() {
        var a = this.opacity; a = isNaN(a) ? 1 : Math.max(0, Math.min(1, a));
        return (a === 1 ? "hsl(" : "hsla(")
            + (this.h || 0) + ", "
            + (this.s || 0) * 100 + "%, "
            + (this.l || 0) * 100 + "%"
            + (a === 1 ? ")" : ", " + a + ")");
      }
    }));

    /* From FvD 13.37, CSS Color Module Level 3 */
    function hsl2rgb(h, m1, m2) {
      return (h < 60 ? m1 + (m2 - m1) * h / 60
          : h < 180 ? m2
          : h < 240 ? m1 + (m2 - m1) * (240 - h) / 60
          : m1) * 255;
    }

    function basis(t1, v0, v1, v2, v3) {
      var t2 = t1 * t1, t3 = t2 * t1;
      return ((1 - 3 * t1 + 3 * t2 - t3) * v0
          + (4 - 6 * t2 + 3 * t3) * v1
          + (1 + 3 * t1 + 3 * t2 - 3 * t3) * v2
          + t3 * v3) / 6;
    }

    function basis$1(values) {
      var n = values.length - 1;
      return function(t) {
        var i = t <= 0 ? (t = 0) : t >= 1 ? (t = 1, n - 1) : Math.floor(t * n),
            v1 = values[i],
            v2 = values[i + 1],
            v0 = i > 0 ? values[i - 1] : 2 * v1 - v2,
            v3 = i < n - 1 ? values[i + 2] : 2 * v2 - v1;
        return basis((t - i / n) * n, v0, v1, v2, v3);
      };
    }

    function constant$2(x) {
      return function() {
        return x;
      };
    }

    function linear(a, d) {
      return function(t) {
        return a + t * d;
      };
    }

    function exponential(a, b, y) {
      return a = Math.pow(a, y), b = Math.pow(b, y) - a, y = 1 / y, function(t) {
        return Math.pow(a + t * b, y);
      };
    }

    function gamma(y) {
      return (y = +y) === 1 ? nogamma : function(a, b) {
        return b - a ? exponential(a, b, y) : constant$2(isNaN(a) ? b : a);
      };
    }

    function nogamma(a, b) {
      var d = b - a;
      return d ? linear(a, d) : constant$2(isNaN(a) ? b : a);
    }

    var interpolateRgb = (function rgbGamma(y) {
      var color = gamma(y);

      function rgb$1(start, end) {
        var r = color((start = rgb(start)).r, (end = rgb(end)).r),
            g = color(start.g, end.g),
            b = color(start.b, end.b),
            opacity = nogamma(start.opacity, end.opacity);
        return function(t) {
          start.r = r(t);
          start.g = g(t);
          start.b = b(t);
          start.opacity = opacity(t);
          return start + "";
        };
      }

      rgb$1.gamma = rgbGamma;

      return rgb$1;
    })(1);

    function rgbSpline(spline) {
      return function(colors) {
        var n = colors.length,
            r = new Array(n),
            g = new Array(n),
            b = new Array(n),
            i, color;
        for (i = 0; i < n; ++i) {
          color = rgb(colors[i]);
          r[i] = color.r || 0;
          g[i] = color.g || 0;
          b[i] = color.b || 0;
        }
        r = spline(r);
        g = spline(g);
        b = spline(b);
        color.opacity = 1;
        return function(t) {
          color.r = r(t);
          color.g = g(t);
          color.b = b(t);
          return color + "";
        };
      };
    }

    var rgbBasis = rgbSpline(basis$1);

    function numberArray(a, b) {
      if (!b) b = [];
      var n = a ? Math.min(b.length, a.length) : 0,
          c = b.slice(),
          i;
      return function(t) {
        for (i = 0; i < n; ++i) c[i] = a[i] * (1 - t) + b[i] * t;
        return c;
      };
    }

    function isNumberArray(x) {
      return ArrayBuffer.isView(x) && !(x instanceof DataView);
    }

    function genericArray(a, b) {
      var nb = b ? b.length : 0,
          na = a ? Math.min(nb, a.length) : 0,
          x = new Array(na),
          c = new Array(nb),
          i;

      for (i = 0; i < na; ++i) x[i] = interpolateValue(a[i], b[i]);
      for (; i < nb; ++i) c[i] = b[i];

      return function(t) {
        for (i = 0; i < na; ++i) c[i] = x[i](t);
        return c;
      };
    }

    function date(a, b) {
      var d = new Date;
      return a = +a, b = +b, function(t) {
        return d.setTime(a * (1 - t) + b * t), d;
      };
    }

    function interpolateNumber(a, b) {
      return a = +a, b = +b, function(t) {
        return a * (1 - t) + b * t;
      };
    }

    function object(a, b) {
      var i = {},
          c = {},
          k;

      if (a === null || typeof a !== "object") a = {};
      if (b === null || typeof b !== "object") b = {};

      for (k in b) {
        if (k in a) {
          i[k] = interpolateValue(a[k], b[k]);
        } else {
          c[k] = b[k];
        }
      }

      return function(t) {
        for (k in i) c[k] = i[k](t);
        return c;
      };
    }

    var reA = /[-+]?(?:\d+\.?\d*|\.?\d+)(?:[eE][-+]?\d+)?/g,
        reB = new RegExp(reA.source, "g");

    function zero(b) {
      return function() {
        return b;
      };
    }

    function one(b) {
      return function(t) {
        return b(t) + "";
      };
    }

    function interpolateString(a, b) {
      var bi = reA.lastIndex = reB.lastIndex = 0, // scan index for next number in b
          am, // current match in a
          bm, // current match in b
          bs, // string preceding current number in b, if any
          i = -1, // index in s
          s = [], // string constants and placeholders
          q = []; // number interpolators

      // Coerce inputs to strings.
      a = a + "", b = b + "";

      // Interpolate pairs of numbers in a & b.
      while ((am = reA.exec(a))
          && (bm = reB.exec(b))) {
        if ((bs = bm.index) > bi) { // a string precedes the next number in b
          bs = b.slice(bi, bs);
          if (s[i]) s[i] += bs; // coalesce with previous string
          else s[++i] = bs;
        }
        if ((am = am[0]) === (bm = bm[0])) { // numbers in a & b match
          if (s[i]) s[i] += bm; // coalesce with previous string
          else s[++i] = bm;
        } else { // interpolate non-matching numbers
          s[++i] = null;
          q.push({i: i, x: interpolateNumber(am, bm)});
        }
        bi = reB.lastIndex;
      }

      // Add remains of b.
      if (bi < b.length) {
        bs = b.slice(bi);
        if (s[i]) s[i] += bs; // coalesce with previous string
        else s[++i] = bs;
      }

      // Special optimization for only a single match.
      // Otherwise, interpolate each of the numbers and rejoin the string.
      return s.length < 2 ? (q[0]
          ? one(q[0].x)
          : zero(b))
          : (b = q.length, function(t) {
              for (var i = 0, o; i < b; ++i) s[(o = q[i]).i] = o.x(t);
              return s.join("");
            });
    }

    function interpolateValue(a, b) {
      var t = typeof b, c;
      return b == null || t === "boolean" ? constant$2(b)
          : (t === "number" ? interpolateNumber
          : t === "string" ? ((c = color(b)) ? (b = c, interpolateRgb) : interpolateString)
          : b instanceof color ? interpolateRgb
          : b instanceof Date ? date
          : isNumberArray(b) ? numberArray
          : Array.isArray(b) ? genericArray
          : typeof b.valueOf !== "function" && typeof b.toString !== "function" || isNaN(b) ? object
          : interpolateNumber)(a, b);
    }

    function interpolateRound(a, b) {
      return a = +a, b = +b, function(t) {
        return Math.round(a * (1 - t) + b * t);
      };
    }

    var degrees = 180 / Math.PI;

    var identity$3 = {
      translateX: 0,
      translateY: 0,
      rotate: 0,
      skewX: 0,
      scaleX: 1,
      scaleY: 1
    };

    function decompose(a, b, c, d, e, f) {
      var scaleX, scaleY, skewX;
      if (scaleX = Math.sqrt(a * a + b * b)) a /= scaleX, b /= scaleX;
      if (skewX = a * c + b * d) c -= a * skewX, d -= b * skewX;
      if (scaleY = Math.sqrt(c * c + d * d)) c /= scaleY, d /= scaleY, skewX /= scaleY;
      if (a * d < b * c) a = -a, b = -b, skewX = -skewX, scaleX = -scaleX;
      return {
        translateX: e,
        translateY: f,
        rotate: Math.atan2(b, a) * degrees,
        skewX: Math.atan(skewX) * degrees,
        scaleX: scaleX,
        scaleY: scaleY
      };
    }

    var cssNode,
        cssRoot,
        cssView,
        svgNode;

    function parseCss(value) {
      if (value === "none") return identity$3;
      if (!cssNode) cssNode = document.createElement("DIV"), cssRoot = document.documentElement, cssView = document.defaultView;
      cssNode.style.transform = value;
      value = cssView.getComputedStyle(cssRoot.appendChild(cssNode), null).getPropertyValue("transform");
      cssRoot.removeChild(cssNode);
      value = value.slice(7, -1).split(",");
      return decompose(+value[0], +value[1], +value[2], +value[3], +value[4], +value[5]);
    }

    function parseSvg(value) {
      if (value == null) return identity$3;
      if (!svgNode) svgNode = document.createElementNS("http://www.w3.org/2000/svg", "g");
      svgNode.setAttribute("transform", value);
      if (!(value = svgNode.transform.baseVal.consolidate())) return identity$3;
      value = value.matrix;
      return decompose(value.a, value.b, value.c, value.d, value.e, value.f);
    }

    function interpolateTransform(parse, pxComma, pxParen, degParen) {

      function pop(s) {
        return s.length ? s.pop() + " " : "";
      }

      function translate(xa, ya, xb, yb, s, q) {
        if (xa !== xb || ya !== yb) {
          var i = s.push("translate(", null, pxComma, null, pxParen);
          q.push({i: i - 4, x: interpolateNumber(xa, xb)}, {i: i - 2, x: interpolateNumber(ya, yb)});
        } else if (xb || yb) {
          s.push("translate(" + xb + pxComma + yb + pxParen);
        }
      }

      function rotate(a, b, s, q) {
        if (a !== b) {
          if (a - b > 180) b += 360; else if (b - a > 180) a += 360; // shortest path
          q.push({i: s.push(pop(s) + "rotate(", null, degParen) - 2, x: interpolateNumber(a, b)});
        } else if (b) {
          s.push(pop(s) + "rotate(" + b + degParen);
        }
      }

      function skewX(a, b, s, q) {
        if (a !== b) {
          q.push({i: s.push(pop(s) + "skewX(", null, degParen) - 2, x: interpolateNumber(a, b)});
        } else if (b) {
          s.push(pop(s) + "skewX(" + b + degParen);
        }
      }

      function scale(xa, ya, xb, yb, s, q) {
        if (xa !== xb || ya !== yb) {
          var i = s.push(pop(s) + "scale(", null, ",", null, ")");
          q.push({i: i - 4, x: interpolateNumber(xa, xb)}, {i: i - 2, x: interpolateNumber(ya, yb)});
        } else if (xb !== 1 || yb !== 1) {
          s.push(pop(s) + "scale(" + xb + "," + yb + ")");
        }
      }

      return function(a, b) {
        var s = [], // string constants and placeholders
            q = []; // number interpolators
        a = parse(a), b = parse(b);
        translate(a.translateX, a.translateY, b.translateX, b.translateY, s, q);
        rotate(a.rotate, b.rotate, s, q);
        skewX(a.skewX, b.skewX, s, q);
        scale(a.scaleX, a.scaleY, b.scaleX, b.scaleY, s, q);
        a = b = null; // gc
        return function(t) {
          var i = -1, n = q.length, o;
          while (++i < n) s[(o = q[i]).i] = o.x(t);
          return s.join("");
        };
      };
    }

    var interpolateTransformCss = interpolateTransform(parseCss, "px, ", "px)", "deg)");
    var interpolateTransformSvg = interpolateTransform(parseSvg, ", ", ")", ")");

    var frame = 0, // is an animation frame pending?
        timeout = 0, // is a timeout pending?
        interval$1 = 0, // are any timers active?
        pokeDelay = 1000, // how frequently we check for clock skew
        taskHead,
        taskTail,
        clockLast = 0,
        clockNow = 0,
        clockSkew = 0,
        clock = typeof performance === "object" && performance.now ? performance : Date,
        setFrame = typeof window === "object" && window.requestAnimationFrame ? window.requestAnimationFrame.bind(window) : function(f) { setTimeout(f, 17); };

    function now() {
      return clockNow || (setFrame(clearNow), clockNow = clock.now() + clockSkew);
    }

    function clearNow() {
      clockNow = 0;
    }

    function Timer() {
      this._call =
      this._time =
      this._next = null;
    }

    Timer.prototype = timer$1.prototype = {
      constructor: Timer,
      restart: function(callback, delay, time) {
        if (typeof callback !== "function") throw new TypeError("callback is not a function");
        time = (time == null ? now() : +time) + (delay == null ? 0 : +delay);
        if (!this._next && taskTail !== this) {
          if (taskTail) taskTail._next = this;
          else taskHead = this;
          taskTail = this;
        }
        this._call = callback;
        this._time = time;
        sleep();
      },
      stop: function() {
        if (this._call) {
          this._call = null;
          this._time = Infinity;
          sleep();
        }
      }
    };

    function timer$1(callback, delay, time) {
      var t = new Timer;
      t.restart(callback, delay, time);
      return t;
    }

    function timerFlush() {
      now(); // Get the current time, if not already set.
      ++frame; // Pretend we’ve set an alarm, if we haven’t already.
      var t = taskHead, e;
      while (t) {
        if ((e = clockNow - t._time) >= 0) t._call.call(null, e);
        t = t._next;
      }
      --frame;
    }

    function wake() {
      clockNow = (clockLast = clock.now()) + clockSkew;
      frame = timeout = 0;
      try {
        timerFlush();
      } finally {
        frame = 0;
        nap();
        clockNow = 0;
      }
    }

    function poke() {
      var now = clock.now(), delay = now - clockLast;
      if (delay > pokeDelay) clockSkew -= delay, clockLast = now;
    }

    function nap() {
      var t0, t1 = taskHead, t2, time = Infinity;
      while (t1) {
        if (t1._call) {
          if (time > t1._time) time = t1._time;
          t0 = t1, t1 = t1._next;
        } else {
          t2 = t1._next, t1._next = null;
          t1 = t0 ? t0._next = t2 : taskHead = t2;
        }
      }
      taskTail = t0;
      sleep(time);
    }

    function sleep(time) {
      if (frame) return; // Soonest alarm already set, or will be.
      if (timeout) timeout = clearTimeout(timeout);
      var delay = time - clockNow; // Strictly less than if we recomputed clockNow.
      if (delay > 24) {
        if (time < Infinity) timeout = setTimeout(wake, time - clock.now() - clockSkew);
        if (interval$1) interval$1 = clearInterval(interval$1);
      } else {
        if (!interval$1) clockLast = clock.now(), interval$1 = setInterval(poke, pokeDelay);
        frame = 1, setFrame(wake);
      }
    }

    function timeout$1(callback, delay, time) {
      var t = new Timer;
      delay = delay == null ? 0 : +delay;
      t.restart(function(elapsed) {
        t.stop();
        callback(elapsed + delay);
      }, delay, time);
      return t;
    }

    var emptyOn = dispatch$3("start", "end", "cancel", "interrupt");
    var emptyTween = [];

    var CREATED = 0;
    var SCHEDULED = 1;
    var STARTING = 2;
    var STARTED = 3;
    var RUNNING = 4;
    var ENDING = 5;
    var ENDED = 6;

    function schedule(node, name, id, index, group, timing) {
      var schedules = node.__transition;
      if (!schedules) node.__transition = {};
      else if (id in schedules) return;
      create(node, id, {
        name: name,
        index: index, // For context during callback.
        group: group, // For context during callback.
        on: emptyOn,
        tween: emptyTween,
        time: timing.time,
        delay: timing.delay,
        duration: timing.duration,
        ease: timing.ease,
        timer: null,
        state: CREATED
      });
    }

    function init(node, id) {
      var schedule = get$1(node, id);
      if (schedule.state > CREATED) throw new Error("too late; already scheduled");
      return schedule;
    }

    function set$1(node, id) {
      var schedule = get$1(node, id);
      if (schedule.state > STARTED) throw new Error("too late; already running");
      return schedule;
    }

    function get$1(node, id) {
      var schedule = node.__transition;
      if (!schedule || !(schedule = schedule[id])) throw new Error("transition not found");
      return schedule;
    }

    function create(node, id, self) {
      var schedules = node.__transition,
          tween;

      // Initialize the self timer when the transition is created.
      // Note the actual delay is not known until the first callback!
      schedules[id] = self;
      self.timer = timer$1(schedule, 0, self.time);

      function schedule(elapsed) {
        self.state = SCHEDULED;
        self.timer.restart(start, self.delay, self.time);

        // If the elapsed delay is less than our first sleep, start immediately.
        if (self.delay <= elapsed) start(elapsed - self.delay);
      }

      function start(elapsed) {
        var i, j, n, o;

        // If the state is not SCHEDULED, then we previously errored on start.
        if (self.state !== SCHEDULED) return stop();

        for (i in schedules) {
          o = schedules[i];
          if (o.name !== self.name) continue;

          // While this element already has a starting transition during this frame,
          // defer starting an interrupting transition until that transition has a
          // chance to tick (and possibly end); see d3/d3-transition#54!
          if (o.state === STARTED) return timeout$1(start);

          // Interrupt the active transition, if any.
          if (o.state === RUNNING) {
            o.state = ENDED;
            o.timer.stop();
            o.on.call("interrupt", node, node.__data__, o.index, o.group);
            delete schedules[i];
          }

          // Cancel any pre-empted transitions.
          else if (+i < id) {
            o.state = ENDED;
            o.timer.stop();
            o.on.call("cancel", node, node.__data__, o.index, o.group);
            delete schedules[i];
          }
        }

        // Defer the first tick to end of the current frame; see d3/d3#1576.
        // Note the transition may be canceled after start and before the first tick!
        // Note this must be scheduled before the start event; see d3/d3-transition#16!
        // Assuming this is successful, subsequent callbacks go straight to tick.
        timeout$1(function() {
          if (self.state === STARTED) {
            self.state = RUNNING;
            self.timer.restart(tick, self.delay, self.time);
            tick(elapsed);
          }
        });

        // Dispatch the start event.
        // Note this must be done before the tween are initialized.
        self.state = STARTING;
        self.on.call("start", node, node.__data__, self.index, self.group);
        if (self.state !== STARTING) return; // interrupted
        self.state = STARTED;

        // Initialize the tween, deleting null tween.
        tween = new Array(n = self.tween.length);
        for (i = 0, j = -1; i < n; ++i) {
          if (o = self.tween[i].value.call(node, node.__data__, self.index, self.group)) {
            tween[++j] = o;
          }
        }
        tween.length = j + 1;
      }

      function tick(elapsed) {
        var t = elapsed < self.duration ? self.ease.call(null, elapsed / self.duration) : (self.timer.restart(stop), self.state = ENDING, 1),
            i = -1,
            n = tween.length;

        while (++i < n) {
          tween[i].call(node, t);
        }

        // Dispatch the end event.
        if (self.state === ENDING) {
          self.on.call("end", node, node.__data__, self.index, self.group);
          stop();
        }
      }

      function stop() {
        self.state = ENDED;
        self.timer.stop();
        delete schedules[id];
        for (var i in schedules) return; // eslint-disable-line no-unused-vars
        delete node.__transition;
      }
    }

    function interrupt(node, name) {
      var schedules = node.__transition,
          schedule,
          active,
          empty = true,
          i;

      if (!schedules) return;

      name = name == null ? null : name + "";

      for (i in schedules) {
        if ((schedule = schedules[i]).name !== name) { empty = false; continue; }
        active = schedule.state > STARTING && schedule.state < ENDING;
        schedule.state = ENDED;
        schedule.timer.stop();
        schedule.on.call(active ? "interrupt" : "cancel", node, node.__data__, schedule.index, schedule.group);
        delete schedules[i];
      }

      if (empty) delete node.__transition;
    }

    function selection_interrupt(name) {
      return this.each(function() {
        interrupt(this, name);
      });
    }

    function tweenRemove(id, name) {
      var tween0, tween1;
      return function() {
        var schedule = set$1(this, id),
            tween = schedule.tween;

        // If this node shared tween with the previous node,
        // just assign the updated shared tween and we’re done!
        // Otherwise, copy-on-write.
        if (tween !== tween0) {
          tween1 = tween0 = tween;
          for (var i = 0, n = tween1.length; i < n; ++i) {
            if (tween1[i].name === name) {
              tween1 = tween1.slice();
              tween1.splice(i, 1);
              break;
            }
          }
        }

        schedule.tween = tween1;
      };
    }

    function tweenFunction(id, name, value) {
      var tween0, tween1;
      if (typeof value !== "function") throw new Error;
      return function() {
        var schedule = set$1(this, id),
            tween = schedule.tween;

        // If this node shared tween with the previous node,
        // just assign the updated shared tween and we’re done!
        // Otherwise, copy-on-write.
        if (tween !== tween0) {
          tween1 = (tween0 = tween).slice();
          for (var t = {name: name, value: value}, i = 0, n = tween1.length; i < n; ++i) {
            if (tween1[i].name === name) {
              tween1[i] = t;
              break;
            }
          }
          if (i === n) tween1.push(t);
        }

        schedule.tween = tween1;
      };
    }

    function transition_tween(name, value) {
      var id = this._id;

      name += "";

      if (arguments.length < 2) {
        var tween = get$1(this.node(), id).tween;
        for (var i = 0, n = tween.length, t; i < n; ++i) {
          if ((t = tween[i]).name === name) {
            return t.value;
          }
        }
        return null;
      }

      return this.each((value == null ? tweenRemove : tweenFunction)(id, name, value));
    }

    function tweenValue(transition, name, value) {
      var id = transition._id;

      transition.each(function() {
        var schedule = set$1(this, id);
        (schedule.value || (schedule.value = {}))[name] = value.apply(this, arguments);
      });

      return function(node) {
        return get$1(node, id).value[name];
      };
    }

    function interpolate(a, b) {
      var c;
      return (typeof b === "number" ? interpolateNumber
          : b instanceof color ? interpolateRgb
          : (c = color(b)) ? (b = c, interpolateRgb)
          : interpolateString)(a, b);
    }

    function attrRemove$1(name) {
      return function() {
        this.removeAttribute(name);
      };
    }

    function attrRemoveNS$1(fullname) {
      return function() {
        this.removeAttributeNS(fullname.space, fullname.local);
      };
    }

    function attrConstant$1(name, interpolate, value1) {
      var string00,
          string1 = value1 + "",
          interpolate0;
      return function() {
        var string0 = this.getAttribute(name);
        return string0 === string1 ? null
            : string0 === string00 ? interpolate0
            : interpolate0 = interpolate(string00 = string0, value1);
      };
    }

    function attrConstantNS$1(fullname, interpolate, value1) {
      var string00,
          string1 = value1 + "",
          interpolate0;
      return function() {
        var string0 = this.getAttributeNS(fullname.space, fullname.local);
        return string0 === string1 ? null
            : string0 === string00 ? interpolate0
            : interpolate0 = interpolate(string00 = string0, value1);
      };
    }

    function attrFunction$1(name, interpolate, value) {
      var string00,
          string10,
          interpolate0;
      return function() {
        var string0, value1 = value(this), string1;
        if (value1 == null) return void this.removeAttribute(name);
        string0 = this.getAttribute(name);
        string1 = value1 + "";
        return string0 === string1 ? null
            : string0 === string00 && string1 === string10 ? interpolate0
            : (string10 = string1, interpolate0 = interpolate(string00 = string0, value1));
      };
    }

    function attrFunctionNS$1(fullname, interpolate, value) {
      var string00,
          string10,
          interpolate0;
      return function() {
        var string0, value1 = value(this), string1;
        if (value1 == null) return void this.removeAttributeNS(fullname.space, fullname.local);
        string0 = this.getAttributeNS(fullname.space, fullname.local);
        string1 = value1 + "";
        return string0 === string1 ? null
            : string0 === string00 && string1 === string10 ? interpolate0
            : (string10 = string1, interpolate0 = interpolate(string00 = string0, value1));
      };
    }

    function transition_attr(name, value) {
      var fullname = namespace(name), i = fullname === "transform" ? interpolateTransformSvg : interpolate;
      return this.attrTween(name, typeof value === "function"
          ? (fullname.local ? attrFunctionNS$1 : attrFunction$1)(fullname, i, tweenValue(this, "attr." + name, value))
          : value == null ? (fullname.local ? attrRemoveNS$1 : attrRemove$1)(fullname)
          : (fullname.local ? attrConstantNS$1 : attrConstant$1)(fullname, i, value));
    }

    function attrInterpolate(name, i) {
      return function(t) {
        this.setAttribute(name, i.call(this, t));
      };
    }

    function attrInterpolateNS(fullname, i) {
      return function(t) {
        this.setAttributeNS(fullname.space, fullname.local, i.call(this, t));
      };
    }

    function attrTweenNS(fullname, value) {
      var t0, i0;
      function tween() {
        var i = value.apply(this, arguments);
        if (i !== i0) t0 = (i0 = i) && attrInterpolateNS(fullname, i);
        return t0;
      }
      tween._value = value;
      return tween;
    }

    function attrTween(name, value) {
      var t0, i0;
      function tween() {
        var i = value.apply(this, arguments);
        if (i !== i0) t0 = (i0 = i) && attrInterpolate(name, i);
        return t0;
      }
      tween._value = value;
      return tween;
    }

    function transition_attrTween(name, value) {
      var key = "attr." + name;
      if (arguments.length < 2) return (key = this.tween(key)) && key._value;
      if (value == null) return this.tween(key, null);
      if (typeof value !== "function") throw new Error;
      var fullname = namespace(name);
      return this.tween(key, (fullname.local ? attrTweenNS : attrTween)(fullname, value));
    }

    function delayFunction(id, value) {
      return function() {
        init(this, id).delay = +value.apply(this, arguments);
      };
    }

    function delayConstant(id, value) {
      return value = +value, function() {
        init(this, id).delay = value;
      };
    }

    function transition_delay(value) {
      var id = this._id;

      return arguments.length
          ? this.each((typeof value === "function"
              ? delayFunction
              : delayConstant)(id, value))
          : get$1(this.node(), id).delay;
    }

    function durationFunction(id, value) {
      return function() {
        set$1(this, id).duration = +value.apply(this, arguments);
      };
    }

    function durationConstant(id, value) {
      return value = +value, function() {
        set$1(this, id).duration = value;
      };
    }

    function transition_duration(value) {
      var id = this._id;

      return arguments.length
          ? this.each((typeof value === "function"
              ? durationFunction
              : durationConstant)(id, value))
          : get$1(this.node(), id).duration;
    }

    function easeConstant(id, value) {
      if (typeof value !== "function") throw new Error;
      return function() {
        set$1(this, id).ease = value;
      };
    }

    function transition_ease(value) {
      var id = this._id;

      return arguments.length
          ? this.each(easeConstant(id, value))
          : get$1(this.node(), id).ease;
    }

    function transition_filter(match) {
      if (typeof match !== "function") match = matcher(match);

      for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, subgroup = subgroups[j] = [], node, i = 0; i < n; ++i) {
          if ((node = group[i]) && match.call(node, node.__data__, i, group)) {
            subgroup.push(node);
          }
        }
      }

      return new Transition(subgroups, this._parents, this._name, this._id);
    }

    function transition_merge(transition) {
      if (transition._id !== this._id) throw new Error;

      for (var groups0 = this._groups, groups1 = transition._groups, m0 = groups0.length, m1 = groups1.length, m = Math.min(m0, m1), merges = new Array(m0), j = 0; j < m; ++j) {
        for (var group0 = groups0[j], group1 = groups1[j], n = group0.length, merge = merges[j] = new Array(n), node, i = 0; i < n; ++i) {
          if (node = group0[i] || group1[i]) {
            merge[i] = node;
          }
        }
      }

      for (; j < m0; ++j) {
        merges[j] = groups0[j];
      }

      return new Transition(merges, this._parents, this._name, this._id);
    }

    function start(name) {
      return (name + "").trim().split(/^|\s+/).every(function(t) {
        var i = t.indexOf(".");
        if (i >= 0) t = t.slice(0, i);
        return !t || t === "start";
      });
    }

    function onFunction(id, name, listener) {
      var on0, on1, sit = start(name) ? init : set$1;
      return function() {
        var schedule = sit(this, id),
            on = schedule.on;

        // If this node shared a dispatch with the previous node,
        // just assign the updated shared dispatch and we’re done!
        // Otherwise, copy-on-write.
        if (on !== on0) (on1 = (on0 = on).copy()).on(name, listener);

        schedule.on = on1;
      };
    }

    function transition_on(name, listener) {
      var id = this._id;

      return arguments.length < 2
          ? get$1(this.node(), id).on.on(name)
          : this.each(onFunction(id, name, listener));
    }

    function removeFunction(id) {
      return function() {
        var parent = this.parentNode;
        for (var i in this.__transition) if (+i !== id) return;
        if (parent) parent.removeChild(this);
      };
    }

    function transition_remove() {
      return this.on("end.remove", removeFunction(this._id));
    }

    function transition_select(select) {
      var name = this._name,
          id = this._id;

      if (typeof select !== "function") select = selector(select);

      for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, subgroup = subgroups[j] = new Array(n), node, subnode, i = 0; i < n; ++i) {
          if ((node = group[i]) && (subnode = select.call(node, node.__data__, i, group))) {
            if ("__data__" in node) subnode.__data__ = node.__data__;
            subgroup[i] = subnode;
            schedule(subgroup[i], name, id, i, subgroup, get$1(node, id));
          }
        }
      }

      return new Transition(subgroups, this._parents, name, id);
    }

    function transition_selectAll(select) {
      var name = this._name,
          id = this._id;

      if (typeof select !== "function") select = selectorAll(select);

      for (var groups = this._groups, m = groups.length, subgroups = [], parents = [], j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, node, i = 0; i < n; ++i) {
          if (node = group[i]) {
            for (var children = select.call(node, node.__data__, i, group), child, inherit = get$1(node, id), k = 0, l = children.length; k < l; ++k) {
              if (child = children[k]) {
                schedule(child, name, id, k, children, inherit);
              }
            }
            subgroups.push(children);
            parents.push(node);
          }
        }
      }

      return new Transition(subgroups, parents, name, id);
    }

    var Selection$1 = selection.prototype.constructor;

    function transition_selection() {
      return new Selection$1(this._groups, this._parents);
    }

    function styleNull(name, interpolate) {
      var string00,
          string10,
          interpolate0;
      return function() {
        var string0 = styleValue(this, name),
            string1 = (this.style.removeProperty(name), styleValue(this, name));
        return string0 === string1 ? null
            : string0 === string00 && string1 === string10 ? interpolate0
            : interpolate0 = interpolate(string00 = string0, string10 = string1);
      };
    }

    function styleRemove$1(name) {
      return function() {
        this.style.removeProperty(name);
      };
    }

    function styleConstant$1(name, interpolate, value1) {
      var string00,
          string1 = value1 + "",
          interpolate0;
      return function() {
        var string0 = styleValue(this, name);
        return string0 === string1 ? null
            : string0 === string00 ? interpolate0
            : interpolate0 = interpolate(string00 = string0, value1);
      };
    }

    function styleFunction$1(name, interpolate, value) {
      var string00,
          string10,
          interpolate0;
      return function() {
        var string0 = styleValue(this, name),
            value1 = value(this),
            string1 = value1 + "";
        if (value1 == null) string1 = value1 = (this.style.removeProperty(name), styleValue(this, name));
        return string0 === string1 ? null
            : string0 === string00 && string1 === string10 ? interpolate0
            : (string10 = string1, interpolate0 = interpolate(string00 = string0, value1));
      };
    }

    function styleMaybeRemove(id, name) {
      var on0, on1, listener0, key = "style." + name, event = "end." + key, remove;
      return function() {
        var schedule = set$1(this, id),
            on = schedule.on,
            listener = schedule.value[key] == null ? remove || (remove = styleRemove$1(name)) : undefined;

        // If this node shared a dispatch with the previous node,
        // just assign the updated shared dispatch and we’re done!
        // Otherwise, copy-on-write.
        if (on !== on0 || listener0 !== listener) (on1 = (on0 = on).copy()).on(event, listener0 = listener);

        schedule.on = on1;
      };
    }

    function transition_style(name, value, priority) {
      var i = (name += "") === "transform" ? interpolateTransformCss : interpolate;
      return value == null ? this
          .styleTween(name, styleNull(name, i))
          .on("end.style." + name, styleRemove$1(name))
        : typeof value === "function" ? this
          .styleTween(name, styleFunction$1(name, i, tweenValue(this, "style." + name, value)))
          .each(styleMaybeRemove(this._id, name))
        : this
          .styleTween(name, styleConstant$1(name, i, value), priority)
          .on("end.style." + name, null);
    }

    function styleInterpolate(name, i, priority) {
      return function(t) {
        this.style.setProperty(name, i.call(this, t), priority);
      };
    }

    function styleTween(name, value, priority) {
      var t, i0;
      function tween() {
        var i = value.apply(this, arguments);
        if (i !== i0) t = (i0 = i) && styleInterpolate(name, i, priority);
        return t;
      }
      tween._value = value;
      return tween;
    }

    function transition_styleTween(name, value, priority) {
      var key = "style." + (name += "");
      if (arguments.length < 2) return (key = this.tween(key)) && key._value;
      if (value == null) return this.tween(key, null);
      if (typeof value !== "function") throw new Error;
      return this.tween(key, styleTween(name, value, priority == null ? "" : priority));
    }

    function textConstant$1(value) {
      return function() {
        this.textContent = value;
      };
    }

    function textFunction$1(value) {
      return function() {
        var value1 = value(this);
        this.textContent = value1 == null ? "" : value1;
      };
    }

    function transition_text(value) {
      return this.tween("text", typeof value === "function"
          ? textFunction$1(tweenValue(this, "text", value))
          : textConstant$1(value == null ? "" : value + ""));
    }

    function textInterpolate(i) {
      return function(t) {
        this.textContent = i.call(this, t);
      };
    }

    function textTween(value) {
      var t0, i0;
      function tween() {
        var i = value.apply(this, arguments);
        if (i !== i0) t0 = (i0 = i) && textInterpolate(i);
        return t0;
      }
      tween._value = value;
      return tween;
    }

    function transition_textTween(value) {
      var key = "text";
      if (arguments.length < 1) return (key = this.tween(key)) && key._value;
      if (value == null) return this.tween(key, null);
      if (typeof value !== "function") throw new Error;
      return this.tween(key, textTween(value));
    }

    function transition_transition() {
      var name = this._name,
          id0 = this._id,
          id1 = newId();

      for (var groups = this._groups, m = groups.length, j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, node, i = 0; i < n; ++i) {
          if (node = group[i]) {
            var inherit = get$1(node, id0);
            schedule(node, name, id1, i, group, {
              time: inherit.time + inherit.delay + inherit.duration,
              delay: 0,
              duration: inherit.duration,
              ease: inherit.ease
            });
          }
        }
      }

      return new Transition(groups, this._parents, name, id1);
    }

    function transition_end() {
      var on0, on1, that = this, id = that._id, size = that.size();
      return new Promise(function(resolve, reject) {
        var cancel = {value: reject},
            end = {value: function() { if (--size === 0) resolve(); }};

        that.each(function() {
          var schedule = set$1(this, id),
              on = schedule.on;

          // If this node shared a dispatch with the previous node,
          // just assign the updated shared dispatch and we’re done!
          // Otherwise, copy-on-write.
          if (on !== on0) {
            on1 = (on0 = on).copy();
            on1._.cancel.push(cancel);
            on1._.interrupt.push(cancel);
            on1._.end.push(end);
          }

          schedule.on = on1;
        });
      });
    }

    var id = 0;

    function Transition(groups, parents, name, id) {
      this._groups = groups;
      this._parents = parents;
      this._name = name;
      this._id = id;
    }

    function transition$1(name) {
      return selection().transition(name);
    }

    function newId() {
      return ++id;
    }

    var selection_prototype = selection.prototype;

    Transition.prototype = transition$1.prototype = {
      constructor: Transition,
      select: transition_select,
      selectAll: transition_selectAll,
      filter: transition_filter,
      merge: transition_merge,
      selection: transition_selection,
      transition: transition_transition,
      call: selection_prototype.call,
      nodes: selection_prototype.nodes,
      node: selection_prototype.node,
      size: selection_prototype.size,
      empty: selection_prototype.empty,
      each: selection_prototype.each,
      on: transition_on,
      attr: transition_attr,
      attrTween: transition_attrTween,
      style: transition_style,
      styleTween: transition_styleTween,
      text: transition_text,
      textTween: transition_textTween,
      remove: transition_remove,
      tween: transition_tween,
      delay: transition_delay,
      duration: transition_duration,
      ease: transition_ease,
      end: transition_end
    };

    function cubicInOut(t) {
      return ((t *= 2) <= 1 ? t * t * t : (t -= 2) * t * t + 2) / 2;
    }

    var defaultTiming = {
      time: null, // Set on use.
      delay: 0,
      duration: 250,
      ease: cubicInOut
    };

    function inherit(node, id) {
      var timing;
      while (!(timing = node.__transition) || !(timing = timing[id])) {
        if (!(node = node.parentNode)) {
          return defaultTiming.time = now(), defaultTiming;
        }
      }
      return timing;
    }

    function selection_transition(name) {
      var id,
          timing;

      if (name instanceof Transition) {
        id = name._id, name = name._name;
      } else {
        id = newId(), (timing = defaultTiming).time = now(), name = name == null ? null : name + "";
      }

      for (var groups = this._groups, m = groups.length, j = 0; j < m; ++j) {
        for (var group = groups[j], n = group.length, node, i = 0; i < n; ++i) {
          if (node = group[i]) {
            schedule(node, name, id, i, group, timing || inherit(node, id));
          }
        }
      }

      return new Transition(groups, this._parents, name, id);
    }

    selection.prototype.interrupt = selection_interrupt;
    selection.prototype.transition = selection_transition;

    function constant$3(x) {
      return function() {
        return x;
      };
    }

    function BrushEvent(target, type, selection) {
      this.target = target;
      this.type = type;
      this.selection = selection;
    }

    function nopropagation() {
      event.stopImmediatePropagation();
    }

    function noevent$1() {
      event.preventDefault();
      event.stopImmediatePropagation();
    }

    var MODE_DRAG = {name: "drag"},
        MODE_SPACE = {name: "space"},
        MODE_HANDLE = {name: "handle"},
        MODE_CENTER = {name: "center"};

    function number1(e) {
      return [+e[0], +e[1]];
    }

    function number2(e) {
      return [number1(e[0]), number1(e[1])];
    }

    function toucher(identifier) {
      return function(target) {
        return touch(target, event.touches, identifier);
      };
    }

    var X = {
      name: "x",
      handles: ["w", "e"].map(type),
      input: function(x, e) { return x == null ? null : [[+x[0], e[0][1]], [+x[1], e[1][1]]]; },
      output: function(xy) { return xy && [xy[0][0], xy[1][0]]; }
    };

    var Y = {
      name: "y",
      handles: ["n", "s"].map(type),
      input: function(y, e) { return y == null ? null : [[e[0][0], +y[0]], [e[1][0], +y[1]]]; },
      output: function(xy) { return xy && [xy[0][1], xy[1][1]]; }
    };

    var cursors = {
      overlay: "crosshair",
      selection: "move",
      n: "ns-resize",
      e: "ew-resize",
      s: "ns-resize",
      w: "ew-resize",
      nw: "nwse-resize",
      ne: "nesw-resize",
      se: "nwse-resize",
      sw: "nesw-resize"
    };

    var flipX = {
      e: "w",
      w: "e",
      nw: "ne",
      ne: "nw",
      se: "sw",
      sw: "se"
    };

    var flipY = {
      n: "s",
      s: "n",
      nw: "sw",
      ne: "se",
      se: "ne",
      sw: "nw"
    };

    var signsX = {
      overlay: +1,
      selection: +1,
      n: null,
      e: +1,
      s: null,
      w: -1,
      nw: -1,
      ne: +1,
      se: +1,
      sw: -1
    };

    var signsY = {
      overlay: +1,
      selection: +1,
      n: -1,
      e: null,
      s: +1,
      w: null,
      nw: -1,
      ne: -1,
      se: +1,
      sw: +1
    };

    function type(t) {
      return {type: t};
    }

    // Ignore right-click, since that should open the context menu.
    function defaultFilter() {
      return !event.ctrlKey && !event.button;
    }

    function defaultExtent() {
      var svg = this.ownerSVGElement || this;
      if (svg.hasAttribute("viewBox")) {
        svg = svg.viewBox.baseVal;
        return [[svg.x, svg.y], [svg.x + svg.width, svg.y + svg.height]];
      }
      return [[0, 0], [svg.width.baseVal.value, svg.height.baseVal.value]];
    }

    function defaultTouchable() {
      return navigator.maxTouchPoints || ("ontouchstart" in this);
    }

    // Like d3.local, but with the name “__brush” rather than auto-generated.
    function local(node) {
      while (!node.__brush) if (!(node = node.parentNode)) return;
      return node.__brush;
    }

    function empty$2(extent) {
      return extent[0][0] === extent[1][0]
          || extent[0][1] === extent[1][1];
    }

    function brushY() {
      return brush(Y);
    }

    function brush(dim) {
      var extent = defaultExtent,
          filter = defaultFilter,
          touchable = defaultTouchable,
          keys = true,
          listeners = dispatch$3("start", "brush", "end"),
          handleSize = 6,
          touchending;

      function brush(group) {
        var overlay = group
            .property("__brush", initialize)
          .selectAll(".overlay")
          .data([type("overlay")]);

        overlay.enter().append("rect")
            .attr("class", "overlay")
            .attr("pointer-events", "all")
            .attr("cursor", cursors.overlay)
          .merge(overlay)
            .each(function() {
              var extent = local(this).extent;
              select$1(this)
                  .attr("x", extent[0][0])
                  .attr("y", extent[0][1])
                  .attr("width", extent[1][0] - extent[0][0])
                  .attr("height", extent[1][1] - extent[0][1]);
            });

        group.selectAll(".selection")
          .data([type("selection")])
          .enter().append("rect")
            .attr("class", "selection")
            .attr("cursor", cursors.selection)
            .attr("fill", "#777")
            .attr("fill-opacity", 0.3)
            .attr("stroke", "#fff")
            .attr("shape-rendering", "crispEdges");

        var handle = group.selectAll(".handle")
          .data(dim.handles, function(d) { return d.type; });

        handle.exit().remove();

        handle.enter().append("rect")
            .attr("class", function(d) { return "handle handle--" + d.type; })
            .attr("cursor", function(d) { return cursors[d.type]; });

        group
            .each(redraw)
            .attr("fill", "none")
            .attr("pointer-events", "all")
            .on("mousedown.brush", started)
          .filter(touchable)
            .on("touchstart.brush", started)
            .on("touchmove.brush", touchmoved)
            .on("touchend.brush touchcancel.brush", touchended)
            .style("touch-action", "none")
            .style("-webkit-tap-highlight-color", "rgba(0,0,0,0)");
      }

      brush.move = function(group, selection) {
        if (group.selection) {
          group
              .on("start.brush", function() { emitter(this, arguments).beforestart().start(); })
              .on("interrupt.brush end.brush", function() { emitter(this, arguments).end(); })
              .tween("brush", function() {
                var that = this,
                    state = that.__brush,
                    emit = emitter(that, arguments),
                    selection0 = state.selection,
                    selection1 = dim.input(typeof selection === "function" ? selection.apply(this, arguments) : selection, state.extent),
                    i = interpolateValue(selection0, selection1);

                function tween(t) {
                  state.selection = t === 1 && selection1 === null ? null : i(t);
                  redraw.call(that);
                  emit.brush();
                }

                return selection0 !== null && selection1 !== null ? tween : tween(1);
              });
        } else {
          group
              .each(function() {
                var that = this,
                    args = arguments,
                    state = that.__brush,
                    selection1 = dim.input(typeof selection === "function" ? selection.apply(that, args) : selection, state.extent),
                    emit = emitter(that, args).beforestart();

                interrupt(that);
                state.selection = selection1 === null ? null : selection1;
                redraw.call(that);
                emit.start().brush().end();
              });
        }
      };

      brush.clear = function(group) {
        brush.move(group, null);
      };

      function redraw() {
        var group = select$1(this),
            selection = local(this).selection;

        if (selection) {
          group.selectAll(".selection")
              .style("display", null)
              .attr("x", selection[0][0])
              .attr("y", selection[0][1])
              .attr("width", selection[1][0] - selection[0][0])
              .attr("height", selection[1][1] - selection[0][1]);

          group.selectAll(".handle")
              .style("display", null)
              .attr("x", function(d) { return d.type[d.type.length - 1] === "e" ? selection[1][0] - handleSize / 2 : selection[0][0] - handleSize / 2; })
              .attr("y", function(d) { return d.type[0] === "s" ? selection[1][1] - handleSize / 2 : selection[0][1] - handleSize / 2; })
              .attr("width", function(d) { return d.type === "n" || d.type === "s" ? selection[1][0] - selection[0][0] + handleSize : handleSize; })
              .attr("height", function(d) { return d.type === "e" || d.type === "w" ? selection[1][1] - selection[0][1] + handleSize : handleSize; });
        }

        else {
          group.selectAll(".selection,.handle")
              .style("display", "none")
              .attr("x", null)
              .attr("y", null)
              .attr("width", null)
              .attr("height", null);
        }
      }

      function emitter(that, args, clean) {
        return (!clean && that.__brush.emitter) || new Emitter(that, args);
      }

      function Emitter(that, args) {
        this.that = that;
        this.args = args;
        this.state = that.__brush;
        this.active = 0;
      }

      Emitter.prototype = {
        beforestart: function() {
          if (++this.active === 1) this.state.emitter = this, this.starting = true;
          return this;
        },
        start: function() {
          if (this.starting) this.starting = false, this.emit("start");
          else this.emit("brush");
          return this;
        },
        brush: function() {
          this.emit("brush");
          return this;
        },
        end: function() {
          if (--this.active === 0) delete this.state.emitter, this.emit("end");
          return this;
        },
        emit: function(type) {
          customEvent(new BrushEvent(brush, type, dim.output(this.state.selection)), listeners.apply, listeners, [type, this.that, this.args]);
        }
      };

      function started() {
        if (touchending && !event.touches) return;
        if (!filter.apply(this, arguments)) return;

        var that = this,
            type = event.target.__data__.type,
            mode = (keys && event.metaKey ? type = "overlay" : type) === "selection" ? MODE_DRAG : (keys && event.altKey ? MODE_CENTER : MODE_HANDLE),
            signX = dim === Y ? null : signsX[type],
            signY = dim === X ? null : signsY[type],
            state = local(that),
            extent = state.extent,
            selection = state.selection,
            W = extent[0][0], w0, w1,
            N = extent[0][1], n0, n1,
            E = extent[1][0], e0, e1,
            S = extent[1][1], s0, s1,
            dx = 0,
            dy = 0,
            moving,
            shifting = signX && signY && keys && event.shiftKey,
            lockX,
            lockY,
            pointer = event.touches ? toucher(event.changedTouches[0].identifier) : mouse,
            point0 = pointer(that),
            point = point0,
            emit = emitter(that, arguments, true).beforestart();

        if (type === "overlay") {
          if (selection) moving = true;
          state.selection = selection = [
            [w0 = dim === Y ? W : point0[0], n0 = dim === X ? N : point0[1]],
            [e0 = dim === Y ? E : w0, s0 = dim === X ? S : n0]
          ];
        } else {
          w0 = selection[0][0];
          n0 = selection[0][1];
          e0 = selection[1][0];
          s0 = selection[1][1];
        }

        w1 = w0;
        n1 = n0;
        e1 = e0;
        s1 = s0;

        var group = select$1(that)
            .attr("pointer-events", "none");

        var overlay = group.selectAll(".overlay")
            .attr("cursor", cursors[type]);

        if (event.touches) {
          emit.moved = moved;
          emit.ended = ended;
        } else {
          var view = select$1(event.view)
              .on("mousemove.brush", moved, true)
              .on("mouseup.brush", ended, true);
          if (keys) view
              .on("keydown.brush", keydowned, true)
              .on("keyup.brush", keyupped, true);

          dragDisable(event.view);
        }

        nopropagation();
        interrupt(that);
        redraw.call(that);
        emit.start();

        function moved() {
          var point1 = pointer(that);
          if (shifting && !lockX && !lockY) {
            if (Math.abs(point1[0] - point[0]) > Math.abs(point1[1] - point[1])) lockY = true;
            else lockX = true;
          }
          point = point1;
          moving = true;
          noevent$1();
          move();
        }

        function move() {
          var t;

          dx = point[0] - point0[0];
          dy = point[1] - point0[1];

          switch (mode) {
            case MODE_SPACE:
            case MODE_DRAG: {
              if (signX) dx = Math.max(W - w0, Math.min(E - e0, dx)), w1 = w0 + dx, e1 = e0 + dx;
              if (signY) dy = Math.max(N - n0, Math.min(S - s0, dy)), n1 = n0 + dy, s1 = s0 + dy;
              break;
            }
            case MODE_HANDLE: {
              if (signX < 0) dx = Math.max(W - w0, Math.min(E - w0, dx)), w1 = w0 + dx, e1 = e0;
              else if (signX > 0) dx = Math.max(W - e0, Math.min(E - e0, dx)), w1 = w0, e1 = e0 + dx;
              if (signY < 0) dy = Math.max(N - n0, Math.min(S - n0, dy)), n1 = n0 + dy, s1 = s0;
              else if (signY > 0) dy = Math.max(N - s0, Math.min(S - s0, dy)), n1 = n0, s1 = s0 + dy;
              break;
            }
            case MODE_CENTER: {
              if (signX) w1 = Math.max(W, Math.min(E, w0 - dx * signX)), e1 = Math.max(W, Math.min(E, e0 + dx * signX));
              if (signY) n1 = Math.max(N, Math.min(S, n0 - dy * signY)), s1 = Math.max(N, Math.min(S, s0 + dy * signY));
              break;
            }
          }

          if (e1 < w1) {
            signX *= -1;
            t = w0, w0 = e0, e0 = t;
            t = w1, w1 = e1, e1 = t;
            if (type in flipX) overlay.attr("cursor", cursors[type = flipX[type]]);
          }

          if (s1 < n1) {
            signY *= -1;
            t = n0, n0 = s0, s0 = t;
            t = n1, n1 = s1, s1 = t;
            if (type in flipY) overlay.attr("cursor", cursors[type = flipY[type]]);
          }

          if (state.selection) selection = state.selection; // May be set by brush.move!
          if (lockX) w1 = selection[0][0], e1 = selection[1][0];
          if (lockY) n1 = selection[0][1], s1 = selection[1][1];

          if (selection[0][0] !== w1
              || selection[0][1] !== n1
              || selection[1][0] !== e1
              || selection[1][1] !== s1) {
            state.selection = [[w1, n1], [e1, s1]];
            redraw.call(that);
            emit.brush();
          }
        }

        function ended() {
          nopropagation();
          if (event.touches) {
            if (event.touches.length) return;
            if (touchending) clearTimeout(touchending);
            touchending = setTimeout(function() { touchending = null; }, 500); // Ghost clicks are delayed!
          } else {
            yesdrag(event.view, moving);
            view.on("keydown.brush keyup.brush mousemove.brush mouseup.brush", null);
          }
          group.attr("pointer-events", "all");
          overlay.attr("cursor", cursors.overlay);
          if (state.selection) selection = state.selection; // May be set by brush.move (on start)!
          if (empty$2(selection)) state.selection = null, redraw.call(that);
          emit.end();
        }

        function keydowned() {
          switch (event.keyCode) {
            case 16: { // SHIFT
              shifting = signX && signY;
              break;
            }
            case 18: { // ALT
              if (mode === MODE_HANDLE) {
                if (signX) e0 = e1 - dx * signX, w0 = w1 + dx * signX;
                if (signY) s0 = s1 - dy * signY, n0 = n1 + dy * signY;
                mode = MODE_CENTER;
                move();
              }
              break;
            }
            case 32: { // SPACE; takes priority over ALT
              if (mode === MODE_HANDLE || mode === MODE_CENTER) {
                if (signX < 0) e0 = e1 - dx; else if (signX > 0) w0 = w1 - dx;
                if (signY < 0) s0 = s1 - dy; else if (signY > 0) n0 = n1 - dy;
                mode = MODE_SPACE;
                overlay.attr("cursor", cursors.selection);
                move();
              }
              break;
            }
            default: return;
          }
          noevent$1();
        }

        function keyupped() {
          switch (event.keyCode) {
            case 16: { // SHIFT
              if (shifting) {
                lockX = lockY = shifting = false;
                move();
              }
              break;
            }
            case 18: { // ALT
              if (mode === MODE_CENTER) {
                if (signX < 0) e0 = e1; else if (signX > 0) w0 = w1;
                if (signY < 0) s0 = s1; else if (signY > 0) n0 = n1;
                mode = MODE_HANDLE;
                move();
              }
              break;
            }
            case 32: { // SPACE
              if (mode === MODE_SPACE) {
                if (event.altKey) {
                  if (signX) e0 = e1 - dx * signX, w0 = w1 + dx * signX;
                  if (signY) s0 = s1 - dy * signY, n0 = n1 + dy * signY;
                  mode = MODE_CENTER;
                } else {
                  if (signX < 0) e0 = e1; else if (signX > 0) w0 = w1;
                  if (signY < 0) s0 = s1; else if (signY > 0) n0 = n1;
                  mode = MODE_HANDLE;
                }
                overlay.attr("cursor", cursors[type]);
                move();
              }
              break;
            }
            default: return;
          }
          noevent$1();
        }
      }

      function touchmoved() {
        emitter(this, arguments).moved();
      }

      function touchended() {
        emitter(this, arguments).ended();
      }

      function initialize() {
        var state = this.__brush || {selection: null};
        state.extent = number2(extent.apply(this, arguments));
        state.dim = dim;
        return state;
      }

      brush.extent = function(_) {
        return arguments.length ? (extent = typeof _ === "function" ? _ : constant$3(number2(_)), brush) : extent;
      };

      brush.filter = function(_) {
        return arguments.length ? (filter = typeof _ === "function" ? _ : constant$3(!!_), brush) : filter;
      };

      brush.touchable = function(_) {
        return arguments.length ? (touchable = typeof _ === "function" ? _ : constant$3(!!_), brush) : touchable;
      };

      brush.handleSize = function(_) {
        return arguments.length ? (handleSize = +_, brush) : handleSize;
      };

      brush.keyModifiers = function(_) {
        return arguments.length ? (keys = !!_, brush) : keys;
      };

      brush.on = function() {
        var value = listeners.on.apply(listeners, arguments);
        return value === listeners ? brush : value;
      };

      return brush;
    }

    var pi = Math.PI,
        tau = 2 * pi,
        epsilon$1 = 1e-6,
        tauEpsilon = tau - epsilon$1;

    function Path() {
      this._x0 = this._y0 = // start of current subpath
      this._x1 = this._y1 = null; // end of current subpath
      this._ = "";
    }

    function path() {
      return new Path;
    }

    Path.prototype = path.prototype = {
      constructor: Path,
      moveTo: function(x, y) {
        this._ += "M" + (this._x0 = this._x1 = +x) + "," + (this._y0 = this._y1 = +y);
      },
      closePath: function() {
        if (this._x1 !== null) {
          this._x1 = this._x0, this._y1 = this._y0;
          this._ += "Z";
        }
      },
      lineTo: function(x, y) {
        this._ += "L" + (this._x1 = +x) + "," + (this._y1 = +y);
      },
      quadraticCurveTo: function(x1, y1, x, y) {
        this._ += "Q" + (+x1) + "," + (+y1) + "," + (this._x1 = +x) + "," + (this._y1 = +y);
      },
      bezierCurveTo: function(x1, y1, x2, y2, x, y) {
        this._ += "C" + (+x1) + "," + (+y1) + "," + (+x2) + "," + (+y2) + "," + (this._x1 = +x) + "," + (this._y1 = +y);
      },
      arcTo: function(x1, y1, x2, y2, r) {
        x1 = +x1, y1 = +y1, x2 = +x2, y2 = +y2, r = +r;
        var x0 = this._x1,
            y0 = this._y1,
            x21 = x2 - x1,
            y21 = y2 - y1,
            x01 = x0 - x1,
            y01 = y0 - y1,
            l01_2 = x01 * x01 + y01 * y01;

        // Is the radius negative? Error.
        if (r < 0) throw new Error("negative radius: " + r);

        // Is this path empty? Move to (x1,y1).
        if (this._x1 === null) {
          this._ += "M" + (this._x1 = x1) + "," + (this._y1 = y1);
        }

        // Or, is (x1,y1) coincident with (x0,y0)? Do nothing.
        else if (!(l01_2 > epsilon$1));

        // Or, are (x0,y0), (x1,y1) and (x2,y2) collinear?
        // Equivalently, is (x1,y1) coincident with (x2,y2)?
        // Or, is the radius zero? Line to (x1,y1).
        else if (!(Math.abs(y01 * x21 - y21 * x01) > epsilon$1) || !r) {
          this._ += "L" + (this._x1 = x1) + "," + (this._y1 = y1);
        }

        // Otherwise, draw an arc!
        else {
          var x20 = x2 - x0,
              y20 = y2 - y0,
              l21_2 = x21 * x21 + y21 * y21,
              l20_2 = x20 * x20 + y20 * y20,
              l21 = Math.sqrt(l21_2),
              l01 = Math.sqrt(l01_2),
              l = r * Math.tan((pi - Math.acos((l21_2 + l01_2 - l20_2) / (2 * l21 * l01))) / 2),
              t01 = l / l01,
              t21 = l / l21;

          // If the start tangent is not coincident with (x0,y0), line to.
          if (Math.abs(t01 - 1) > epsilon$1) {
            this._ += "L" + (x1 + t01 * x01) + "," + (y1 + t01 * y01);
          }

          this._ += "A" + r + "," + r + ",0,0," + (+(y01 * x20 > x01 * y20)) + "," + (this._x1 = x1 + t21 * x21) + "," + (this._y1 = y1 + t21 * y21);
        }
      },
      arc: function(x, y, r, a0, a1, ccw) {
        x = +x, y = +y, r = +r, ccw = !!ccw;
        var dx = r * Math.cos(a0),
            dy = r * Math.sin(a0),
            x0 = x + dx,
            y0 = y + dy,
            cw = 1 ^ ccw,
            da = ccw ? a0 - a1 : a1 - a0;

        // Is the radius negative? Error.
        if (r < 0) throw new Error("negative radius: " + r);

        // Is this path empty? Move to (x0,y0).
        if (this._x1 === null) {
          this._ += "M" + x0 + "," + y0;
        }

        // Or, is (x0,y0) not coincident with the previous point? Line to (x0,y0).
        else if (Math.abs(this._x1 - x0) > epsilon$1 || Math.abs(this._y1 - y0) > epsilon$1) {
          this._ += "L" + x0 + "," + y0;
        }

        // Is this arc empty? We’re done.
        if (!r) return;

        // Does the angle go the wrong way? Flip the direction.
        if (da < 0) da = da % tau + tau;

        // Is this a complete circle? Draw two arcs to complete the circle.
        if (da > tauEpsilon) {
          this._ += "A" + r + "," + r + ",0,1," + cw + "," + (x - dx) + "," + (y - dy) + "A" + r + "," + r + ",0,1," + cw + "," + (this._x1 = x0) + "," + (this._y1 = y0);
        }

        // Is this arc non-empty? Draw an arc!
        else if (da > epsilon$1) {
          this._ += "A" + r + "," + r + ",0," + (+(da >= pi)) + "," + cw + "," + (this._x1 = x + r * Math.cos(a1)) + "," + (this._y1 = y + r * Math.sin(a1));
        }
      },
      rect: function(x, y, w, h) {
        this._ += "M" + (this._x0 = this._x1 = +x) + "," + (this._y0 = this._y1 = +y) + "h" + (+w) + "v" + (+h) + "h" + (-w) + "Z";
      },
      toString: function() {
        return this._;
      }
    };

    var prefix = "$";

    function Map$1() {}

    Map$1.prototype = map$1.prototype = {
      constructor: Map$1,
      has: function(key) {
        return (prefix + key) in this;
      },
      get: function(key) {
        return this[prefix + key];
      },
      set: function(key, value) {
        this[prefix + key] = value;
        return this;
      },
      remove: function(key) {
        var property = prefix + key;
        return property in this && delete this[property];
      },
      clear: function() {
        for (var property in this) if (property[0] === prefix) delete this[property];
      },
      keys: function() {
        var keys = [];
        for (var property in this) if (property[0] === prefix) keys.push(property.slice(1));
        return keys;
      },
      values: function() {
        var values = [];
        for (var property in this) if (property[0] === prefix) values.push(this[property]);
        return values;
      },
      entries: function() {
        var entries = [];
        for (var property in this) if (property[0] === prefix) entries.push({key: property.slice(1), value: this[property]});
        return entries;
      },
      size: function() {
        var size = 0;
        for (var property in this) if (property[0] === prefix) ++size;
        return size;
      },
      empty: function() {
        for (var property in this) if (property[0] === prefix) return false;
        return true;
      },
      each: function(f) {
        for (var property in this) if (property[0] === prefix) f(this[property], property.slice(1), this);
      }
    };

    function map$1(object, f) {
      var map = new Map$1;

      // Copy constructor.
      if (object instanceof Map$1) object.each(function(value, key) { map.set(key, value); });

      // Index array by numeric index or specified key function.
      else if (Array.isArray(object)) {
        var i = -1,
            n = object.length,
            o;

        if (f == null) while (++i < n) map.set(i, object[i]);
        else while (++i < n) map.set(f(o = object[i], i, object), o);
      }

      // Convert object to map.
      else if (object) for (var key in object) map.set(key, object[key]);

      return map;
    }

    function Set$1() {}

    var proto = map$1.prototype;

    Set$1.prototype = set$2.prototype = {
      constructor: Set$1,
      has: proto.has,
      add: function(value) {
        value += "";
        this[prefix + value] = value;
        return this;
      },
      remove: proto.remove,
      clear: proto.clear,
      values: proto.keys,
      size: proto.size,
      empty: proto.empty,
      each: proto.each
    };

    function set$2(object, f) {
      var set = new Set$1;

      // Copy constructor.
      if (object instanceof Set$1) object.each(function(value) { set.add(value); });

      // Otherwise, assume it’s an array.
      else if (object) {
        var i = -1, n = object.length;
        if (f == null) while (++i < n) set.add(object[i]);
        else while (++i < n) set.add(f(object[i], i, object));
      }

      return set;
    }

    // Computes the decimal coefficient and exponent of the specified number x with
    // significant digits p, where x is positive and p is in [1, 21] or undefined.
    // For example, formatDecimal(1.23) returns ["123", 0].
    function formatDecimal(x, p) {
      if ((i = (x = p ? x.toExponential(p - 1) : x.toExponential()).indexOf("e")) < 0) return null; // NaN, ±Infinity
      var i, coefficient = x.slice(0, i);

      // The string returned by toExponential either has the form \d\.\d+e[-+]\d+
      // (e.g., 1.2e+3) or the form \de[-+]\d+ (e.g., 1e+3).
      return [
        coefficient.length > 1 ? coefficient[0] + coefficient.slice(2) : coefficient,
        +x.slice(i + 1)
      ];
    }

    function exponent(x) {
      return x = formatDecimal(Math.abs(x)), x ? x[1] : NaN;
    }

    function formatGroup(grouping, thousands) {
      return function(value, width) {
        var i = value.length,
            t = [],
            j = 0,
            g = grouping[0],
            length = 0;

        while (i > 0 && g > 0) {
          if (length + g + 1 > width) g = Math.max(1, width - length);
          t.push(value.substring(i -= g, i + g));
          if ((length += g + 1) > width) break;
          g = grouping[j = (j + 1) % grouping.length];
        }

        return t.reverse().join(thousands);
      };
    }

    function formatNumerals(numerals) {
      return function(value) {
        return value.replace(/[0-9]/g, function(i) {
          return numerals[+i];
        });
      };
    }

    // [[fill]align][sign][symbol][0][width][,][.precision][~][type]
    var re = /^(?:(.)?([<>=^]))?([+\-( ])?([$#])?(0)?(\d+)?(,)?(\.\d+)?(~)?([a-z%])?$/i;

    function formatSpecifier(specifier) {
      if (!(match = re.exec(specifier))) throw new Error("invalid format: " + specifier);
      var match;
      return new FormatSpecifier({
        fill: match[1],
        align: match[2],
        sign: match[3],
        symbol: match[4],
        zero: match[5],
        width: match[6],
        comma: match[7],
        precision: match[8] && match[8].slice(1),
        trim: match[9],
        type: match[10]
      });
    }

    formatSpecifier.prototype = FormatSpecifier.prototype; // instanceof

    function FormatSpecifier(specifier) {
      this.fill = specifier.fill === undefined ? " " : specifier.fill + "";
      this.align = specifier.align === undefined ? ">" : specifier.align + "";
      this.sign = specifier.sign === undefined ? "-" : specifier.sign + "";
      this.symbol = specifier.symbol === undefined ? "" : specifier.symbol + "";
      this.zero = !!specifier.zero;
      this.width = specifier.width === undefined ? undefined : +specifier.width;
      this.comma = !!specifier.comma;
      this.precision = specifier.precision === undefined ? undefined : +specifier.precision;
      this.trim = !!specifier.trim;
      this.type = specifier.type === undefined ? "" : specifier.type + "";
    }

    FormatSpecifier.prototype.toString = function() {
      return this.fill
          + this.align
          + this.sign
          + this.symbol
          + (this.zero ? "0" : "")
          + (this.width === undefined ? "" : Math.max(1, this.width | 0))
          + (this.comma ? "," : "")
          + (this.precision === undefined ? "" : "." + Math.max(0, this.precision | 0))
          + (this.trim ? "~" : "")
          + this.type;
    };

    // Trims insignificant zeros, e.g., replaces 1.2000k with 1.2k.
    function formatTrim(s) {
      out: for (var n = s.length, i = 1, i0 = -1, i1; i < n; ++i) {
        switch (s[i]) {
          case ".": i0 = i1 = i; break;
          case "0": if (i0 === 0) i0 = i; i1 = i; break;
          default: if (!+s[i]) break out; if (i0 > 0) i0 = 0; break;
        }
      }
      return i0 > 0 ? s.slice(0, i0) + s.slice(i1 + 1) : s;
    }

    var prefixExponent;

    function formatPrefixAuto(x, p) {
      var d = formatDecimal(x, p);
      if (!d) return x + "";
      var coefficient = d[0],
          exponent = d[1],
          i = exponent - (prefixExponent = Math.max(-8, Math.min(8, Math.floor(exponent / 3))) * 3) + 1,
          n = coefficient.length;
      return i === n ? coefficient
          : i > n ? coefficient + new Array(i - n + 1).join("0")
          : i > 0 ? coefficient.slice(0, i) + "." + coefficient.slice(i)
          : "0." + new Array(1 - i).join("0") + formatDecimal(x, Math.max(0, p + i - 1))[0]; // less than 1y!
    }

    function formatRounded(x, p) {
      var d = formatDecimal(x, p);
      if (!d) return x + "";
      var coefficient = d[0],
          exponent = d[1];
      return exponent < 0 ? "0." + new Array(-exponent).join("0") + coefficient
          : coefficient.length > exponent + 1 ? coefficient.slice(0, exponent + 1) + "." + coefficient.slice(exponent + 1)
          : coefficient + new Array(exponent - coefficient.length + 2).join("0");
    }

    var formatTypes = {
      "%": function(x, p) { return (x * 100).toFixed(p); },
      "b": function(x) { return Math.round(x).toString(2); },
      "c": function(x) { return x + ""; },
      "d": function(x) { return Math.round(x).toString(10); },
      "e": function(x, p) { return x.toExponential(p); },
      "f": function(x, p) { return x.toFixed(p); },
      "g": function(x, p) { return x.toPrecision(p); },
      "o": function(x) { return Math.round(x).toString(8); },
      "p": function(x, p) { return formatRounded(x * 100, p); },
      "r": formatRounded,
      "s": formatPrefixAuto,
      "X": function(x) { return Math.round(x).toString(16).toUpperCase(); },
      "x": function(x) { return Math.round(x).toString(16); }
    };

    function identity$4(x) {
      return x;
    }

    var map$2 = Array.prototype.map,
        prefixes = ["y","z","a","f","p","n","µ","m","","k","M","G","T","P","E","Z","Y"];

    function formatLocale(locale) {
      var group = locale.grouping === undefined || locale.thousands === undefined ? identity$4 : formatGroup(map$2.call(locale.grouping, Number), locale.thousands + ""),
          currencyPrefix = locale.currency === undefined ? "" : locale.currency[0] + "",
          currencySuffix = locale.currency === undefined ? "" : locale.currency[1] + "",
          decimal = locale.decimal === undefined ? "." : locale.decimal + "",
          numerals = locale.numerals === undefined ? identity$4 : formatNumerals(map$2.call(locale.numerals, String)),
          percent = locale.percent === undefined ? "%" : locale.percent + "",
          minus = locale.minus === undefined ? "-" : locale.minus + "",
          nan = locale.nan === undefined ? "NaN" : locale.nan + "";

      function newFormat(specifier) {
        specifier = formatSpecifier(specifier);

        var fill = specifier.fill,
            align = specifier.align,
            sign = specifier.sign,
            symbol = specifier.symbol,
            zero = specifier.zero,
            width = specifier.width,
            comma = specifier.comma,
            precision = specifier.precision,
            trim = specifier.trim,
            type = specifier.type;

        // The "n" type is an alias for ",g".
        if (type === "n") comma = true, type = "g";

        // The "" type, and any invalid type, is an alias for ".12~g".
        else if (!formatTypes[type]) precision === undefined && (precision = 12), trim = true, type = "g";

        // If zero fill is specified, padding goes after sign and before digits.
        if (zero || (fill === "0" && align === "=")) zero = true, fill = "0", align = "=";

        // Compute the prefix and suffix.
        // For SI-prefix, the suffix is lazily computed.
        var prefix = symbol === "$" ? currencyPrefix : symbol === "#" && /[boxX]/.test(type) ? "0" + type.toLowerCase() : "",
            suffix = symbol === "$" ? currencySuffix : /[%p]/.test(type) ? percent : "";

        // What format function should we use?
        // Is this an integer type?
        // Can this type generate exponential notation?
        var formatType = formatTypes[type],
            maybeSuffix = /[defgprs%]/.test(type);

        // Set the default precision if not specified,
        // or clamp the specified precision to the supported range.
        // For significant precision, it must be in [1, 21].
        // For fixed precision, it must be in [0, 20].
        precision = precision === undefined ? 6
            : /[gprs]/.test(type) ? Math.max(1, Math.min(21, precision))
            : Math.max(0, Math.min(20, precision));

        function format(value) {
          var valuePrefix = prefix,
              valueSuffix = suffix,
              i, n, c;

          if (type === "c") {
            valueSuffix = formatType(value) + valueSuffix;
            value = "";
          } else {
            value = +value;

            // Determine the sign. -0 is not less than 0, but 1 / -0 is!
            var valueNegative = value < 0 || 1 / value < 0;

            // Perform the initial formatting.
            value = isNaN(value) ? nan : formatType(Math.abs(value), precision);

            // Trim insignificant zeros.
            if (trim) value = formatTrim(value);

            // If a negative value rounds to zero after formatting, and no explicit positive sign is requested, hide the sign.
            if (valueNegative && +value === 0 && sign !== "+") valueNegative = false;

            // Compute the prefix and suffix.
            valuePrefix = (valueNegative ? (sign === "(" ? sign : minus) : sign === "-" || sign === "(" ? "" : sign) + valuePrefix;
            valueSuffix = (type === "s" ? prefixes[8 + prefixExponent / 3] : "") + valueSuffix + (valueNegative && sign === "(" ? ")" : "");

            // Break the formatted value into the integer “value” part that can be
            // grouped, and fractional or exponential “suffix” part that is not.
            if (maybeSuffix) {
              i = -1, n = value.length;
              while (++i < n) {
                if (c = value.charCodeAt(i), 48 > c || c > 57) {
                  valueSuffix = (c === 46 ? decimal + value.slice(i + 1) : value.slice(i)) + valueSuffix;
                  value = value.slice(0, i);
                  break;
                }
              }
            }
          }

          // If the fill character is not "0", grouping is applied before padding.
          if (comma && !zero) value = group(value, Infinity);

          // Compute the padding.
          var length = valuePrefix.length + value.length + valueSuffix.length,
              padding = length < width ? new Array(width - length + 1).join(fill) : "";

          // If the fill character is "0", grouping is applied after padding.
          if (comma && zero) value = group(padding + value, padding.length ? width - valueSuffix.length : Infinity), padding = "";

          // Reconstruct the final output based on the desired alignment.
          switch (align) {
            case "<": value = valuePrefix + value + valueSuffix + padding; break;
            case "=": value = valuePrefix + padding + value + valueSuffix; break;
            case "^": value = padding.slice(0, length = padding.length >> 1) + valuePrefix + value + valueSuffix + padding.slice(length); break;
            default: value = padding + valuePrefix + value + valueSuffix; break;
          }

          return numerals(value);
        }

        format.toString = function() {
          return specifier + "";
        };

        return format;
      }

      function formatPrefix(specifier, value) {
        var f = newFormat((specifier = formatSpecifier(specifier), specifier.type = "f", specifier)),
            e = Math.max(-8, Math.min(8, Math.floor(exponent(value) / 3))) * 3,
            k = Math.pow(10, -e),
            prefix = prefixes[8 + e / 3];
        return function(value) {
          return f(k * value) + prefix;
        };
      }

      return {
        format: newFormat,
        formatPrefix: formatPrefix
      };
    }

    var locale;
    var format;
    var formatPrefix;

    defaultLocale({
      decimal: ".",
      thousands: ",",
      grouping: [3],
      currency: ["$", ""],
      minus: "-"
    });

    function defaultLocale(definition) {
      locale = formatLocale(definition);
      format = locale.format;
      formatPrefix = locale.formatPrefix;
      return locale;
    }

    function precisionFixed(step) {
      return Math.max(0, -exponent(Math.abs(step)));
    }

    function precisionPrefix(step, value) {
      return Math.max(0, Math.max(-8, Math.min(8, Math.floor(exponent(value) / 3))) * 3 - exponent(Math.abs(step)));
    }

    function precisionRound(step, max) {
      step = Math.abs(step), max = Math.abs(max) - step;
      return Math.max(0, exponent(max) - exponent(step)) + 1;
    }

    function initRange(domain, range) {
      switch (arguments.length) {
        case 0: break;
        case 1: this.range(domain); break;
        default: this.range(range).domain(domain); break;
      }
      return this;
    }

    var array$1 = Array.prototype;

    var map$3 = array$1.map;
    var slice$2 = array$1.slice;

    var implicit = {name: "implicit"};

    function ordinal() {
      var index = map$1(),
          domain = [],
          range = [],
          unknown = implicit;

      function scale(d) {
        var key = d + "", i = index.get(key);
        if (!i) {
          if (unknown !== implicit) return unknown;
          index.set(key, i = domain.push(d));
        }
        return range[(i - 1) % range.length];
      }

      scale.domain = function(_) {
        if (!arguments.length) return domain.slice();
        domain = [], index = map$1();
        var i = -1, n = _.length, d, key;
        while (++i < n) if (!index.has(key = (d = _[i]) + "")) index.set(key, domain.push(d));
        return scale;
      };

      scale.range = function(_) {
        return arguments.length ? (range = slice$2.call(_), scale) : range.slice();
      };

      scale.unknown = function(_) {
        return arguments.length ? (unknown = _, scale) : unknown;
      };

      scale.copy = function() {
        return ordinal(domain, range).unknown(unknown);
      };

      initRange.apply(scale, arguments);

      return scale;
    }

    function band() {
      var scale = ordinal().unknown(undefined),
          domain = scale.domain,
          ordinalRange = scale.range,
          range = [0, 1],
          step,
          bandwidth,
          round = false,
          paddingInner = 0,
          paddingOuter = 0,
          align = 0.5;

      delete scale.unknown;

      function rescale() {
        var n = domain().length,
            reverse = range[1] < range[0],
            start = range[reverse - 0],
            stop = range[1 - reverse];
        step = (stop - start) / Math.max(1, n - paddingInner + paddingOuter * 2);
        if (round) step = Math.floor(step);
        start += (stop - start - step * (n - paddingInner)) * align;
        bandwidth = step * (1 - paddingInner);
        if (round) start = Math.round(start), bandwidth = Math.round(bandwidth);
        var values = sequence$1(n).map(function(i) { return start + step * i; });
        return ordinalRange(reverse ? values.reverse() : values);
      }

      scale.domain = function(_) {
        return arguments.length ? (domain(_), rescale()) : domain();
      };

      scale.range = function(_) {
        return arguments.length ? (range = [+_[0], +_[1]], rescale()) : range.slice();
      };

      scale.rangeRound = function(_) {
        return range = [+_[0], +_[1]], round = true, rescale();
      };

      scale.bandwidth = function() {
        return bandwidth;
      };

      scale.step = function() {
        return step;
      };

      scale.round = function(_) {
        return arguments.length ? (round = !!_, rescale()) : round;
      };

      scale.padding = function(_) {
        return arguments.length ? (paddingInner = Math.min(1, paddingOuter = +_), rescale()) : paddingInner;
      };

      scale.paddingInner = function(_) {
        return arguments.length ? (paddingInner = Math.min(1, _), rescale()) : paddingInner;
      };

      scale.paddingOuter = function(_) {
        return arguments.length ? (paddingOuter = +_, rescale()) : paddingOuter;
      };

      scale.align = function(_) {
        return arguments.length ? (align = Math.max(0, Math.min(1, _)), rescale()) : align;
      };

      scale.copy = function() {
        return band(domain(), range)
            .round(round)
            .paddingInner(paddingInner)
            .paddingOuter(paddingOuter)
            .align(align);
      };

      return initRange.apply(rescale(), arguments);
    }

    function pointish(scale) {
      var copy = scale.copy;

      scale.padding = scale.paddingOuter;
      delete scale.paddingInner;
      delete scale.paddingOuter;

      scale.copy = function() {
        return pointish(copy());
      };

      return scale;
    }

    function point$1() {
      return pointish(band.apply(null, arguments).paddingInner(1));
    }

    function constant$4(x) {
      return function() {
        return x;
      };
    }

    function number$1(x) {
      return +x;
    }

    var unit = [0, 1];

    function identity$5(x) {
      return x;
    }

    function normalize(a, b) {
      return (b -= (a = +a))
          ? function(x) { return (x - a) / b; }
          : constant$4(isNaN(b) ? NaN : 0.5);
    }

    function clamper(domain) {
      var a = domain[0], b = domain[domain.length - 1], t;
      if (a > b) t = a, a = b, b = t;
      return function(x) { return Math.max(a, Math.min(b, x)); };
    }

    // normalize(a, b)(x) takes a domain value x in [a,b] and returns the corresponding parameter t in [0,1].
    // interpolate(a, b)(t) takes a parameter t in [0,1] and returns the corresponding range value x in [a,b].
    function bimap(domain, range, interpolate) {
      var d0 = domain[0], d1 = domain[1], r0 = range[0], r1 = range[1];
      if (d1 < d0) d0 = normalize(d1, d0), r0 = interpolate(r1, r0);
      else d0 = normalize(d0, d1), r0 = interpolate(r0, r1);
      return function(x) { return r0(d0(x)); };
    }

    function polymap(domain, range, interpolate) {
      var j = Math.min(domain.length, range.length) - 1,
          d = new Array(j),
          r = new Array(j),
          i = -1;

      // Reverse descending domains.
      if (domain[j] < domain[0]) {
        domain = domain.slice().reverse();
        range = range.slice().reverse();
      }

      while (++i < j) {
        d[i] = normalize(domain[i], domain[i + 1]);
        r[i] = interpolate(range[i], range[i + 1]);
      }

      return function(x) {
        var i = bisectRight(domain, x, 1, j) - 1;
        return r[i](d[i](x));
      };
    }

    function copy(source, target) {
      return target
          .domain(source.domain())
          .range(source.range())
          .interpolate(source.interpolate())
          .clamp(source.clamp())
          .unknown(source.unknown());
    }

    function transformer() {
      var domain = unit,
          range = unit,
          interpolate = interpolateValue,
          transform,
          untransform,
          unknown,
          clamp = identity$5,
          piecewise,
          output,
          input;

      function rescale() {
        piecewise = Math.min(domain.length, range.length) > 2 ? polymap : bimap;
        output = input = null;
        return scale;
      }

      function scale(x) {
        return isNaN(x = +x) ? unknown : (output || (output = piecewise(domain.map(transform), range, interpolate)))(transform(clamp(x)));
      }

      scale.invert = function(y) {
        return clamp(untransform((input || (input = piecewise(range, domain.map(transform), interpolateNumber)))(y)));
      };

      scale.domain = function(_) {
        return arguments.length ? (domain = map$3.call(_, number$1), clamp === identity$5 || (clamp = clamper(domain)), rescale()) : domain.slice();
      };

      scale.range = function(_) {
        return arguments.length ? (range = slice$2.call(_), rescale()) : range.slice();
      };

      scale.rangeRound = function(_) {
        return range = slice$2.call(_), interpolate = interpolateRound, rescale();
      };

      scale.clamp = function(_) {
        return arguments.length ? (clamp = _ ? clamper(domain) : identity$5, scale) : clamp !== identity$5;
      };

      scale.interpolate = function(_) {
        return arguments.length ? (interpolate = _, rescale()) : interpolate;
      };

      scale.unknown = function(_) {
        return arguments.length ? (unknown = _, scale) : unknown;
      };

      return function(t, u) {
        transform = t, untransform = u;
        return rescale();
      };
    }

    function continuous(transform, untransform) {
      return transformer()(transform, untransform);
    }

    function tickFormat(start, stop, count, specifier) {
      var step = tickStep(start, stop, count),
          precision;
      specifier = formatSpecifier(specifier == null ? ",f" : specifier);
      switch (specifier.type) {
        case "s": {
          var value = Math.max(Math.abs(start), Math.abs(stop));
          if (specifier.precision == null && !isNaN(precision = precisionPrefix(step, value))) specifier.precision = precision;
          return formatPrefix(specifier, value);
        }
        case "":
        case "e":
        case "g":
        case "p":
        case "r": {
          if (specifier.precision == null && !isNaN(precision = precisionRound(step, Math.max(Math.abs(start), Math.abs(stop))))) specifier.precision = precision - (specifier.type === "e");
          break;
        }
        case "f":
        case "%": {
          if (specifier.precision == null && !isNaN(precision = precisionFixed(step))) specifier.precision = precision - (specifier.type === "%") * 2;
          break;
        }
      }
      return format(specifier);
    }

    function linearish(scale) {
      var domain = scale.domain;

      scale.ticks = function(count) {
        var d = domain();
        return ticks(d[0], d[d.length - 1], count == null ? 10 : count);
      };

      scale.tickFormat = function(count, specifier) {
        var d = domain();
        return tickFormat(d[0], d[d.length - 1], count == null ? 10 : count, specifier);
      };

      scale.nice = function(count) {
        if (count == null) count = 10;

        var d = domain(),
            i0 = 0,
            i1 = d.length - 1,
            start = d[i0],
            stop = d[i1],
            step;

        if (stop < start) {
          step = start, start = stop, stop = step;
          step = i0, i0 = i1, i1 = step;
        }

        step = tickIncrement(start, stop, count);

        if (step > 0) {
          start = Math.floor(start / step) * step;
          stop = Math.ceil(stop / step) * step;
          step = tickIncrement(start, stop, count);
        } else if (step < 0) {
          start = Math.ceil(start * step) / step;
          stop = Math.floor(stop * step) / step;
          step = tickIncrement(start, stop, count);
        }

        if (step > 0) {
          d[i0] = Math.floor(start / step) * step;
          d[i1] = Math.ceil(stop / step) * step;
          domain(d);
        } else if (step < 0) {
          d[i0] = Math.ceil(start * step) / step;
          d[i1] = Math.floor(stop * step) / step;
          domain(d);
        }

        return scale;
      };

      return scale;
    }

    function linear$1() {
      var scale = continuous(identity$5, identity$5);

      scale.copy = function() {
        return copy(scale, linear$1());
      };

      initRange.apply(scale, arguments);

      return linearish(scale);
    }

    function colors(specifier) {
      var n = specifier.length / 6 | 0, colors = new Array(n), i = 0;
      while (i < n) colors[i] = "#" + specifier.slice(i * 6, ++i * 6);
      return colors;
    }

    function ramp(scheme) {
      return rgbBasis(scheme[scheme.length - 1]);
    }

    var scheme = new Array(3).concat(
      "deebf79ecae13182bd",
      "eff3ffbdd7e76baed62171b5",
      "eff3ffbdd7e76baed63182bd08519c",
      "eff3ffc6dbef9ecae16baed63182bd08519c",
      "eff3ffc6dbef9ecae16baed64292c62171b5084594",
      "f7fbffdeebf7c6dbef9ecae16baed64292c62171b5084594",
      "f7fbffdeebf7c6dbef9ecae16baed64292c62171b508519c08306b"
    ).map(colors);

    var Blues = ramp(scheme);

    var scheme$1 = new Array(3).concat(
      "f0f0f0bdbdbd636363",
      "f7f7f7cccccc969696525252",
      "f7f7f7cccccc969696636363252525",
      "f7f7f7d9d9d9bdbdbd969696636363252525",
      "f7f7f7d9d9d9bdbdbd969696737373525252252525",
      "fffffff0f0f0d9d9d9bdbdbd969696737373525252252525",
      "fffffff0f0f0d9d9d9bdbdbd969696737373525252252525000000"
    ).map(colors);

    var Greys = ramp(scheme$1);

    var scheme$2 = new Array(3).concat(
      "fee0d2fc9272de2d26",
      "fee5d9fcae91fb6a4acb181d",
      "fee5d9fcae91fb6a4ade2d26a50f15",
      "fee5d9fcbba1fc9272fb6a4ade2d26a50f15",
      "fee5d9fcbba1fc9272fb6a4aef3b2ccb181d99000d",
      "fff5f0fee0d2fcbba1fc9272fb6a4aef3b2ccb181d99000d",
      "fff5f0fee0d2fcbba1fc9272fb6a4aef3b2ccb181da50f1567000d"
    ).map(colors);

    var Reds = ramp(scheme$2);

    function constant$5(x) {
      return function constant() {
        return x;
      };
    }

    var epsilon$2 = 1e-12;

    function Linear(context) {
      this._context = context;
    }

    Linear.prototype = {
      areaStart: function() {
        this._line = 0;
      },
      areaEnd: function() {
        this._line = NaN;
      },
      lineStart: function() {
        this._point = 0;
      },
      lineEnd: function() {
        if (this._line || (this._line !== 0 && this._point === 1)) this._context.closePath();
        this._line = 1 - this._line;
      },
      point: function(x, y) {
        x = +x, y = +y;
        switch (this._point) {
          case 0: this._point = 1; this._line ? this._context.lineTo(x, y) : this._context.moveTo(x, y); break;
          case 1: this._point = 2; // proceed
          default: this._context.lineTo(x, y); break;
        }
      }
    };

    function curveLinear(context) {
      return new Linear(context);
    }

    function x(p) {
      return p[0];
    }

    function y(p) {
      return p[1];
    }

    function line() {
      var x$1 = x,
          y$1 = y,
          defined = constant$5(true),
          context = null,
          curve = curveLinear,
          output = null;

      function line(data) {
        var i,
            n = data.length,
            d,
            defined0 = false,
            buffer;

        if (context == null) output = curve(buffer = path());

        for (i = 0; i <= n; ++i) {
          if (!(i < n && defined(d = data[i], i, data)) === defined0) {
            if (defined0 = !defined0) output.lineStart();
            else output.lineEnd();
          }
          if (defined0) output.point(+x$1(d, i, data), +y$1(d, i, data));
        }

        if (buffer) return output = null, buffer + "" || null;
      }

      line.x = function(_) {
        return arguments.length ? (x$1 = typeof _ === "function" ? _ : constant$5(+_), line) : x$1;
      };

      line.y = function(_) {
        return arguments.length ? (y$1 = typeof _ === "function" ? _ : constant$5(+_), line) : y$1;
      };

      line.defined = function(_) {
        return arguments.length ? (defined = typeof _ === "function" ? _ : constant$5(!!_), line) : defined;
      };

      line.curve = function(_) {
        return arguments.length ? (curve = _, context != null && (output = curve(context)), line) : curve;
      };

      line.context = function(_) {
        return arguments.length ? (_ == null ? context = output = null : output = curve(context = _), line) : context;
      };

      return line;
    }

    function area() {
      var x0 = x,
          x1 = null,
          y0 = constant$5(0),
          y1 = y,
          defined = constant$5(true),
          context = null,
          curve = curveLinear,
          output = null;

      function area(data) {
        var i,
            j,
            k,
            n = data.length,
            d,
            defined0 = false,
            buffer,
            x0z = new Array(n),
            y0z = new Array(n);

        if (context == null) output = curve(buffer = path());

        for (i = 0; i <= n; ++i) {
          if (!(i < n && defined(d = data[i], i, data)) === defined0) {
            if (defined0 = !defined0) {
              j = i;
              output.areaStart();
              output.lineStart();
            } else {
              output.lineEnd();
              output.lineStart();
              for (k = i - 1; k >= j; --k) {
                output.point(x0z[k], y0z[k]);
              }
              output.lineEnd();
              output.areaEnd();
            }
          }
          if (defined0) {
            x0z[i] = +x0(d, i, data), y0z[i] = +y0(d, i, data);
            output.point(x1 ? +x1(d, i, data) : x0z[i], y1 ? +y1(d, i, data) : y0z[i]);
          }
        }

        if (buffer) return output = null, buffer + "" || null;
      }

      function arealine() {
        return line().defined(defined).curve(curve).context(context);
      }

      area.x = function(_) {
        return arguments.length ? (x0 = typeof _ === "function" ? _ : constant$5(+_), x1 = null, area) : x0;
      };

      area.x0 = function(_) {
        return arguments.length ? (x0 = typeof _ === "function" ? _ : constant$5(+_), area) : x0;
      };

      area.x1 = function(_) {
        return arguments.length ? (x1 = _ == null ? null : typeof _ === "function" ? _ : constant$5(+_), area) : x1;
      };

      area.y = function(_) {
        return arguments.length ? (y0 = typeof _ === "function" ? _ : constant$5(+_), y1 = null, area) : y0;
      };

      area.y0 = function(_) {
        return arguments.length ? (y0 = typeof _ === "function" ? _ : constant$5(+_), area) : y0;
      };

      area.y1 = function(_) {
        return arguments.length ? (y1 = _ == null ? null : typeof _ === "function" ? _ : constant$5(+_), area) : y1;
      };

      area.lineX0 =
      area.lineY0 = function() {
        return arealine().x(x0).y(y0);
      };

      area.lineY1 = function() {
        return arealine().x(x0).y(y1);
      };

      area.lineX1 = function() {
        return arealine().x(x1).y(y0);
      };

      area.defined = function(_) {
        return arguments.length ? (defined = typeof _ === "function" ? _ : constant$5(!!_), area) : defined;
      };

      area.curve = function(_) {
        return arguments.length ? (curve = _, context != null && (output = curve(context)), area) : curve;
      };

      area.context = function(_) {
        return arguments.length ? (_ == null ? context = output = null : output = curve(context = _), area) : context;
      };

      return area;
    }

    function point$2(that, x, y) {
      that._context.bezierCurveTo(
        that._x1 + that._k * (that._x2 - that._x0),
        that._y1 + that._k * (that._y2 - that._y0),
        that._x2 + that._k * (that._x1 - x),
        that._y2 + that._k * (that._y1 - y),
        that._x2,
        that._y2
      );
    }

    function Cardinal(context, tension) {
      this._context = context;
      this._k = (1 - tension) / 6;
    }

    Cardinal.prototype = {
      areaStart: function() {
        this._line = 0;
      },
      areaEnd: function() {
        this._line = NaN;
      },
      lineStart: function() {
        this._x0 = this._x1 = this._x2 =
        this._y0 = this._y1 = this._y2 = NaN;
        this._point = 0;
      },
      lineEnd: function() {
        switch (this._point) {
          case 2: this._context.lineTo(this._x2, this._y2); break;
          case 3: point$2(this, this._x1, this._y1); break;
        }
        if (this._line || (this._line !== 0 && this._point === 1)) this._context.closePath();
        this._line = 1 - this._line;
      },
      point: function(x, y) {
        x = +x, y = +y;
        switch (this._point) {
          case 0: this._point = 1; this._line ? this._context.lineTo(x, y) : this._context.moveTo(x, y); break;
          case 1: this._point = 2; this._x1 = x, this._y1 = y; break;
          case 2: this._point = 3; // proceed
          default: point$2(this, x, y); break;
        }
        this._x0 = this._x1, this._x1 = this._x2, this._x2 = x;
        this._y0 = this._y1, this._y1 = this._y2, this._y2 = y;
      }
    };

    function point$3(that, x, y) {
      var x1 = that._x1,
          y1 = that._y1,
          x2 = that._x2,
          y2 = that._y2;

      if (that._l01_a > epsilon$2) {
        var a = 2 * that._l01_2a + 3 * that._l01_a * that._l12_a + that._l12_2a,
            n = 3 * that._l01_a * (that._l01_a + that._l12_a);
        x1 = (x1 * a - that._x0 * that._l12_2a + that._x2 * that._l01_2a) / n;
        y1 = (y1 * a - that._y0 * that._l12_2a + that._y2 * that._l01_2a) / n;
      }

      if (that._l23_a > epsilon$2) {
        var b = 2 * that._l23_2a + 3 * that._l23_a * that._l12_a + that._l12_2a,
            m = 3 * that._l23_a * (that._l23_a + that._l12_a);
        x2 = (x2 * b + that._x1 * that._l23_2a - x * that._l12_2a) / m;
        y2 = (y2 * b + that._y1 * that._l23_2a - y * that._l12_2a) / m;
      }

      that._context.bezierCurveTo(x1, y1, x2, y2, that._x2, that._y2);
    }

    function CatmullRom(context, alpha) {
      this._context = context;
      this._alpha = alpha;
    }

    CatmullRom.prototype = {
      areaStart: function() {
        this._line = 0;
      },
      areaEnd: function() {
        this._line = NaN;
      },
      lineStart: function() {
        this._x0 = this._x1 = this._x2 =
        this._y0 = this._y1 = this._y2 = NaN;
        this._l01_a = this._l12_a = this._l23_a =
        this._l01_2a = this._l12_2a = this._l23_2a =
        this._point = 0;
      },
      lineEnd: function() {
        switch (this._point) {
          case 2: this._context.lineTo(this._x2, this._y2); break;
          case 3: this.point(this._x2, this._y2); break;
        }
        if (this._line || (this._line !== 0 && this._point === 1)) this._context.closePath();
        this._line = 1 - this._line;
      },
      point: function(x, y) {
        x = +x, y = +y;

        if (this._point) {
          var x23 = this._x2 - x,
              y23 = this._y2 - y;
          this._l23_a = Math.sqrt(this._l23_2a = Math.pow(x23 * x23 + y23 * y23, this._alpha));
        }

        switch (this._point) {
          case 0: this._point = 1; this._line ? this._context.lineTo(x, y) : this._context.moveTo(x, y); break;
          case 1: this._point = 2; break;
          case 2: this._point = 3; // proceed
          default: point$3(this, x, y); break;
        }

        this._l01_a = this._l12_a, this._l12_a = this._l23_a;
        this._l01_2a = this._l12_2a, this._l12_2a = this._l23_2a;
        this._x0 = this._x1, this._x1 = this._x2, this._x2 = x;
        this._y0 = this._y1, this._y1 = this._y2, this._y2 = y;
      }
    };

    var catmullRom = (function custom(alpha) {

      function catmullRom(context) {
        return alpha ? new CatmullRom(context, alpha) : new Cardinal(context, 0);
      }

      catmullRom.alpha = function(alpha) {
        return custom(+alpha);
      };

      return catmullRom;
    })(0.5);

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/util/violin_data.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * Returns the data needed for the nPMI violin plots.
     *
     * @param {?} annotationData the data that comes from the backend
     * @param {?} activeRuns currently active runs to filter the data by
     * @param {?} metricName selected metric to filter the data by
     * @return {?}
     */
    function violinData(annotationData, activeRuns, metricName) {
        /** @type {?} */
        const histogramData = {};
        /** @type {?} */
        const histogramDataNull = {};
        /** @type {?} */
        const allRuns = new Set(activeRuns);
        /** @type {?} */
        const strippedMetric = stripMetricString(metricName);
        /** @type {?} */
        const extremeValues = { max: -1.0, min: 1.0 };
        Object.values(annotationData).forEach((/**
         * @param {?} annotationEntry
         * @return {?}
         */
        (annotationEntry) => {
            annotationEntry.forEach((/**
             * @param {?} valueDataElement
             * @return {?}
             */
            (valueDataElement) => {
                /** @type {?} */
                const run = valueDataElement.run;
                if (!allRuns.has(run) || valueDataElement.metric !== strippedMetric) {
                    return;
                }
                if (valueDataElement.nPMIValue === null) {
                    if (histogramDataNull[run]) {
                        histogramDataNull[run].push(null);
                    }
                    else {
                        histogramDataNull[run] = [null];
                    }
                }
                else {
                    /** @type {?} */
                    const nPMIValue = valueDataElement.nPMIValue;
                    extremeValues.max =
                        extremeValues.max < nPMIValue ? nPMIValue : extremeValues.max;
                    extremeValues.min =
                        extremeValues.min > nPMIValue ? nPMIValue : extremeValues.min;
                    if (histogramData[valueDataElement.run]) {
                        histogramData[run].push(nPMIValue);
                    }
                    else {
                        histogramData[run] = [nPMIValue];
                    }
                }
            }));
        }));
        /** @type {?} */
        const result = {};
        /** @type {?} */
        const bin = histogram()
            .domain([extremeValues.min, extremeValues.max])
            .value((/**
         * @param {?} d
         * @return {?}
         */
        (d) => d));
        /** @type {?} */
        const binNaN = histogram()
            .domain([-Infinity, Infinity])
            .thresholds(0)
            .value((/**
         * @param {?} d
         * @return {?}
         */
        (d) => d));
        for (const run of allRuns) {
            result[run] = bin(histogramData[run]);
            if (histogramDataNull[run]) {
                /** @type {?} */
                const buckets = binNaN(histogramDataNull[run]);
                result[run].unshift(buckets[0]);
            }
        }
        return { violinData: result, extremes: extremeValues };
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/violin_filters/violin_filter/violin_filter_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    const _c0$w = ["chart"];
    var I18N_1$1;
    if (typeof ngI18nClosureMode !== "undefined" && ngI18nClosureMode) {
        /**
         * @desc Label for a button that removes a metric filter.
         */ 
        const MSG_EXTERNAL_8454961797762907624$$TENSORBOARD_WEBAPP_PLUGINS_NPMI_VIEWS_VIOLIN_FILTERS_VIOLIN_FILTER_VIOLIN_FILTER_COMPONENT_TS_2 = goog.getMsg("Remove Filter");
        I18N_1$1 = MSG_EXTERNAL_8454961797762907624$$TENSORBOARD_WEBAPP_PLUGINS_NPMI_VIEWS_VIOLIN_FILTERS_VIOLIN_FILTER_VIOLIN_FILTER_COMPONENT_TS_2;
    }
    else {
        I18N_1$1 = $localize `:Label for a button that removes a metric filter.␟a6bfad58bb363d5c891d0a5474b1d77ef90a34da␟8454961797762907624:Remove Filter`;
    }
    const _c3$4 = ["aria-label", I18N_1$1];
    class ViolinFilterComponent {
        constructor() {
            this.onRemove = new EventEmitter();
            this.onUpdateFilter = new EventEmitter();
            this.height = 300;
            this.chartWidth = 0;
            this.chartHeight = 0;
            this.drawHeight = 0;
            this.drawWidth = 0;
            this.margin = { top: 20, right: 10, bottom: 20, left: 10 };
            this.drawMargin = { top: 0, right: 0, bottom: 20, left: 20 };
            this.colorScale = (/**
             * @return {?}
             */
            () => '#333333');
            // Brush
            this.brush = brushY();
            this.maxBinSize = 0;
            this.area = area()
                .x0((/**
             * @this {?}
             * @param {?} d
             * @return {?}
             */
            function (d) {
                return this.xScaleNum(-d.length);
            }).bind(this))
                .x1((/**
             * @this {?}
             * @param {?} d
             * @return {?}
             */
            function (d) {
                return this.xScaleNum(d.length);
            }).bind(this))
                .y((/**
             * @this {?}
             * @param {?} d
             * @return {?}
             */
            function (d) {
                if ((/** @type {?} */ (d.x0)) === -Infinity) {
                    return this.chartHeight - this.drawMargin.top;
                }
                return this.yScale(((/** @type {?} */ (d.x1)) + (/** @type {?} */ (d.x0))) / 2.0);
            }).bind(this))
                .curve(catmullRom);
        }
        /**
         * @return {?}
         */
        ngAfterViewInit() {
            this.updateDimensions();
            /** @type {?} */
            const runsColorScale = ((/** @type {?} */ (document.createElement('tf-color-scale')))).runsColorScale;
            this.colorScale = runsColorScale ? runsColorScale : this.colorScale;
            this.svg = select$1(this.chartContainer.nativeElement).select('svg');
            this.mainContainer = this.svg
                .append('g')
                .attr('transform', `translate(${this.margin.left}, ${this.margin.top})`);
            this.drawContainer = this.mainContainer
                .append('g')
                .attr('transform', `translate(${this.drawMargin.left}, ${this.drawMargin.top})`);
            this.dotsGroup = this.drawContainer.append('g').attr('class', 'dotsGroup');
            this.yAxisGroup = this.mainContainer
                .append('g')
                .attr('class', 'axis axis--y');
            this.xAxisGroup = this.mainContainer
                .append('g')
                .attr('class', 'axis axis--x');
            this.miscGroup = this.drawContainer.append('g');
            this.xScale = band().padding(0.05);
            this.xAxis = axisBottom(this.xScale);
            this.yScale = linear$1().range([this.drawHeight, 0]);
            this.yAxis = axisLeft(this.yScale);
            this.xScaleNum = linear$1();
            this.initializeBrush();
            this.drawMisc();
            this.redraw();
        }
        /**
         * @param {?} changes
         * @return {?}
         */
        ngOnChanges(changes) {
            if (this.svg) {
                this.redraw();
            }
        }
        /**
         * @return {?}
         */
        redraw() {
            this.updateDimensions();
            this.setMaxBinSize();
            this.updateAxes();
            this.draw();
        }
        // Initializing/Updating the visualization props
        /**
         * @private
         * @return {?}
         */
        updateDimensions() {
            this.chartWidth = this.width - this.margin.left - this.margin.right;
            this.drawWidth =
                this.chartWidth - this.drawMargin.left - this.drawMargin.right;
            this.chartHeight = this.height - this.margin.top - this.margin.bottom;
            this.drawHeight =
                this.chartHeight - this.drawMargin.top - this.drawMargin.bottom;
        }
        /**
         * @return {?}
         */
        setMaxBinSize() {
            Object.values(this.chartData.violinData).forEach((/**
             * @param {?} dataElement
             * @return {?}
             */
            (dataElement) => {
                /** @type {?} */
                const lengths = dataElement.map((/**
                 * @param {?} bin
                 * @return {?}
                 */
                (bin) => bin.length));
                /** @type {?} */
                const longest = Math.max(...lengths);
                this.maxBinSize = Math.max(longest, this.maxBinSize);
            }));
        }
        /**
         * @return {?}
         */
        updateAxes() {
            this.xScale
                .range([0, this.drawWidth])
                .domain(Object.keys(this.chartData.violinData));
            this.yScale.domain([
                this.chartData.extremes.min,
                this.chartData.extremes.max,
            ]);
            this.xScaleNum
                .range([0, this.xScale.bandwidth()])
                .domain([-this.maxBinSize, this.maxBinSize]);
        }
        /**
         * @return {?}
         */
        initializeBrush() {
            this.brush.on('end', this.brushMoved.bind(this));
        }
        // Drawing UI
        /**
         * @return {?}
         */
        draw() {
            this.drawAxes();
            this.drawPlot();
            this.refreshMisc();
            this.refreshBrush();
        }
        /**
         * @return {?}
         */
        drawAxes() {
            this.yAxisGroup
                .attr('transform', `translate(${this.drawMargin.left},
      ${this.drawMargin.top})`)
                .call((/** @type {?} */ (this.yAxis)));
            this.xAxisGroup
                .attr('transform', `translate(${this.drawMargin.left},
      ${this.drawMargin.top + this.chartHeight})`)
                .call((/** @type {?} */ (this.xAxis)));
        }
        /**
         * @return {?}
         */
        drawPlot() {
            /** @type {?} */
            const plots = this.dotsGroup
                .selectAll('.violin-plot')
                .data(Object.entries(this.chartData.violinData));
            plots
                .enter()
                .append('path')
                .attr('class', 'violin-plot')
                .style('stroke', (/**
             * @this {?}
             * @param {?} d
             * @return {?}
             */
            function (d) {
                return this.colorScale(d[0]);
            }).bind(this))
                .style('fill', (/**
             * @this {?}
             * @param {?} d
             * @return {?}
             */
            function (d) {
                return `${this.colorScale(d[0])}33`;
            }).bind(this))
                .attr('transform', (/**
             * @this {?}
             * @param {?} d
             * @return {?}
             */
            function (d) {
                return `translate(${this.xScale(d[0])}, 0)`;
            }).bind(this))
                .datum((/**
             * @param {?} d
             * @return {?}
             */
            function (d) {
                return d[1];
            }))
                .attr('d', this.area);
            plots
                .attr('transform', (/**
             * @this {?}
             * @param {?} d
             * @return {?}
             */
            function (d) {
                return `translate(${this.xScale(d[0])}, 0)`;
            }).bind(this))
                .datum((/**
             * @param {?} d
             * @return {?}
             */
            function (d) {
                return d[1];
            }))
                .attr('d', this.area);
            plots.exit().remove();
        }
        /**
         * @return {?}
         */
        drawMisc() {
            this.zeroLine = this.miscGroup
                .append('line')
                .style('stroke', 'black')
                .attr('x1', 0)
                .attr('y1', this.yScale(0))
                .attr('x2', this.drawWidth)
                .attr('y2', this.yScale(0));
            this.nanText = this.miscGroup
                .append('text')
                .style('fill', 'black')
                .text('NaN')
                .attr('font-size', '10px')
                .attr('text-anchor', 'end')
                .attr('alignment-baseline', 'middle')
                .attr('x', -5)
                .attr('y', this.chartHeight - this.drawMargin.top);
            this.nanLine = this.miscGroup
                .append('line')
                .style('stroke', 'grey')
                .style('stroke-dasharray', '3, 3')
                .attr('x1', 0)
                .attr('y1', this.chartHeight - this.drawMargin.top)
                .attr('x2', this.drawWidth)
                .attr('y2', this.chartHeight - this.drawMargin.top);
        }
        /**
         * @return {?}
         */
        refreshMisc() {
            this.zeroLine
                .attr('y1', this.yScale(0))
                .attr('x2', this.drawWidth)
                .attr('y2', this.yScale(0));
            this.nanText.attr('y', this.chartHeight - this.drawMargin.top);
            this.nanLine
                .attr('y1', this.drawHeight + this.drawMargin.top)
                .attr('x2', this.drawWidth)
                .attr('y2', this.drawHeight + this.drawMargin.top);
        }
        /**
         * @private
         * @return {?}
         */
        refreshBrush() {
            this.brush.extent([
                [0, 0],
                [this.drawWidth, this.drawHeight + this.margin.top],
            ]);
            /** @type {?} */
            const brushPosition = [0, this.drawHeight + this.margin.top];
            if (this.filter.max < this.filter.min) {
                if (this.filter.includeNaN) {
                    // Only NaN selected
                    brushPosition[0] = this.yScale(this.chartData.extremes.min);
                }
                else {
                    // Nothing selected
                    brushPosition[0] = brushPosition[1];
                }
            }
            else {
                if (!this.filter.includeNaN) {
                    // Min does not reach NaN
                    /** @type {?} */
                    const min = Math.max(this.chartData.extremes.min, this.filter.min);
                    brushPosition[1] = this.yScale(min);
                }
                /** @type {?} */
                const max = Math.min(this.chartData.extremes.max, this.filter.max);
                brushPosition[0] = this.yScale(max);
            }
            this.drawContainer.call(this.brush).call(this.brush.move, brushPosition);
        }
        // Called on Interaction
        /**
         * @return {?}
         */
        brushMoved() {
            if (!event)
                return;
            if (!event.sourceEvent)
                return;
            /** @type {?} */
            const extent = event.selection;
            if (extent) {
                /** @type {?} */
                let includeNaN = false;
                /** @type {?} */
                let max = -2.0;
                /** @type {?} */
                let min = this.chartData.extremes.min;
                if (extent[0] <= this.drawHeight + this.margin.top &&
                    extent[1] >= this.drawHeight) {
                    includeNaN = true;
                }
                if (extent[0] < this.drawHeight) {
                    max = this.yScale.invert(extent[0]);
                }
                if (extent[1] < this.drawHeight) {
                    min = this.yScale.invert(extent[1]);
                }
                this.onUpdateFilter.emit({
                    max: max,
                    min: min,
                    includeNaN: includeNaN,
                });
            }
            else {
                this.onUpdateFilter.emit({
                    max: 1.0,
                    min: -1.0,
                    includeNaN: true,
                });
            }
        }
    }
    ViolinFilterComponent.decorators = [
        { type: Component, args: [{
                    selector: 'violin-filter-component',
                    templateUrl: './violin_filter_component.ng.html',
                    styleUrls: ['./violin_filter_component.css'],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    ViolinFilterComponent.propDecorators = {
        metricName: [{ type: Input }],
        filter: [{ type: Input }],
        chartData: [{ type: Input }],
        width: [{ type: Input }],
        onRemove: [{ type: Output }],
        onUpdateFilter: [{ type: Output }],
        chartContainer: [{ type: ViewChild, args: ['chart', { static: true, read: ElementRef },] }]
    };
    /** @nocollapse */ ViolinFilterComponent.ɵfac = function ViolinFilterComponent_Factory(t) { return new (t || ViolinFilterComponent)(); };
    /** @nocollapse */ ViolinFilterComponent.ɵcmp = ɵɵdefineComponent({ type: ViolinFilterComponent, selectors: [["violin-filter-component"]], viewQuery: function ViolinFilterComponent_Query(rf, ctx) { if (rf & 1) {
            ɵɵstaticViewQuery(_c0$w, true, ElementRef);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.chartContainer = _t.first);
        } }, inputs: { metricName: "metricName", filter: "filter", chartData: "chartData", width: "width" }, outputs: { onRemove: "onRemove", onUpdateFilter: "onUpdateFilter" }, features: [ɵɵNgOnChangesFeature], decls: 10, vars: 1, consts: [[1, "chart-container"], ["title", "Shows the nPMI value distribution per run. Ranges of selected values can be manipulated by modifying the grey box.", 1, "chart-head"], [1, "chart-heading"], ["mat-icon-button", "", 3, "click", 6, "aria-label"], ["svgIcon", "clear_24px"], [1, "chart"], ["chart", ""], [1, "draw-area"]], template: function ViolinFilterComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "div", 0);
            ɵɵelementStart(1, "div", 1);
            ɵɵelementStart(2, "div", 2);
            ɵɵtext(3);
            ɵɵelementEnd();
            ɵɵelementStart(4, "button", 3);
            ɵɵi18nAttributes(5, _c3$4);
            ɵɵlistener("click", function ViolinFilterComponent_Template_button_click_4_listener() { return ctx.onRemove.emit(); });
            ɵɵelement(6, "mat-icon", 4);
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementStart(7, "div", 5, 6);
            ɵɵnamespaceSVG();
            ɵɵelement(9, "svg", 7);
            ɵɵelementEnd();
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵadvance(3);
            ɵɵtextInterpolate(ctx.metricName);
        } }, directives: [MatButton, MatIcon], styles: [".mat-badge-content[_ngcontent-%COMP%]{font-weight:600;font-size:12px;font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:9px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:24px}.mat-h1[_ngcontent-%COMP%], .mat-headline[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%]{font:400 24px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h2[_ngcontent-%COMP%], .mat-title[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h3[_ngcontent-%COMP%], .mat-subheading-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%]{font:400 16px/28px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h4[_ngcontent-%COMP%], .mat-subheading-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%]{font:400 15px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h5[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%]{font:400 calc(14px * 0.83)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-h6[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:400 calc(14px * 0.67)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-body-strong[_ngcontent-%COMP%], .mat-body-2[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   p[_ngcontent-%COMP%]{margin:0 0 12px}.mat-small[_ngcontent-%COMP%], .mat-caption[_ngcontent-%COMP%]{font:400 12px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-display-4[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-4[_ngcontent-%COMP%]{font:300 112px/112px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.05em;margin:0 0 56px}.mat-display-3[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-3[_ngcontent-%COMP%]{font:400 56px/56px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.02em;margin:0 0 64px}.mat-display-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-2[_ngcontent-%COMP%]{font:400 45px/48px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.005em;margin:0 0 64px}.mat-display-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-1[_ngcontent-%COMP%]{font:400 34px/40px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 64px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%], .mat-flat-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-button-toggle[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card-title[_ngcontent-%COMP%]{font-size:24px;font-weight:500}.mat-card-header[_ngcontent-%COMP%]   .mat-card-title[_ngcontent-%COMP%]{font-size:20px}.mat-card-subtitle[_ngcontent-%COMP%], .mat-card-content[_ngcontent-%COMP%]{font-size:14px}.mat-checkbox[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-checkbox-layout[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{line-height:24px}.mat-chip[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-chip[_ngcontent-%COMP%]   .mat-chip-trailing-icon.mat-icon[_ngcontent-%COMP%], .mat-chip[_ngcontent-%COMP%]   .mat-chip-remove.mat-icon[_ngcontent-%COMP%]{font-size:18px}.mat-table[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-header-cell[_ngcontent-%COMP%]{font-size:12px;font-weight:500}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{font-size:14px}.mat-calendar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-calendar-body[_ngcontent-%COMP%]{font-size:13px}.mat-calendar-body-label[_ngcontent-%COMP%], .mat-calendar-period-button[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-calendar-table-header[_ngcontent-%COMP%]   th[_ngcontent-%COMP%]{font-size:11px;font-weight:400}.mat-dialog-title[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-expansion-panel-header[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:15px;font-weight:400}.mat-expansion-panel-content[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field[_ngcontent-%COMP%]{font-size:inherit;font-weight:400;line-height:1.125;font-family:Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.34375em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{font-size:150%;line-height:1.125}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]{height:1.5em;width:1.5em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{height:1.125em;line-height:1.125}.mat-form-field-infix[_ngcontent-%COMP%]{padding:.5em 0;border-top:.84375em solid transparent}.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34375em) scale(0.75);width:133.3333333333%}.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34374em) scale(0.75);width:133.3333433333%}.mat-form-field-label-wrapper[_ngcontent-%COMP%]{top:-0.84375em;padding-top:.84375em}.mat-form-field-label[_ngcontent-%COMP%]{top:1.34375em}.mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.34375em}.mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{font-size:75%;margin-top:.6666666667em;top:calc(100% - 1.7916666667em)}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.4375em 0}.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.001px);-ms-transform:translateY(-1.28125em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00101px);-ms-transform:translateY(-1.28124em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00102px);-ms-transform:translateY(-1.28123em) scale(0.75);width:133.3333533333%}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.28125em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{margin-top:.5416666667em;top:calc(100% - 1.6666666667em)}@media print{.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28122em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28121em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.2812em) scale(0.75)}}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.25em 0 .75em 0}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.09375em;margin-top:-0.5em}.mat-form-field-appearance-fill.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59374em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:1em 0 1em 0}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.84375em;margin-top:-0.25em}.mat-form-field-appearance-outline.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59374em) scale(0.75);width:133.3333433333%}.mat-grid-tile-header[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]{font-size:14px}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2), .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}input.mat-input-element[_ngcontent-%COMP%]{margin-top:-0.0625em}.mat-menu-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:400}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px}.mat-radio-button[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select-trigger[_ngcontent-%COMP%]{height:1.125em}.mat-slide-toggle-content[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-slider-thumb-label-text[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-stepper-vertical[_ngcontent-%COMP%], .mat-stepper-horizontal[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-step-label[_ngcontent-%COMP%]{font-size:14px;font-weight:400}.mat-step-sub-label-error[_ngcontent-%COMP%]{font-weight:normal}.mat-step-label-error[_ngcontent-%COMP%]{font-size:14px}.mat-step-label-selected[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-tab-group[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-toolbar[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0}.mat-tooltip[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:10px;padding-top:6px;padding-bottom:6px}.mat-tooltip-handset[_ngcontent-%COMP%]{font-size:14px;padding-top:8px;padding-bottom:8px}.mat-list-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:16px}.mat-optgroup-label[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-simple-snackbar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px}.mat-simple-snackbar-action[_ngcontent-%COMP%]{line-height:1;font-family:inherit;font-size:inherit;font-weight:500}.mat-tree[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{font-weight:400;font-size:14px}.mat-ripple[_ngcontent-%COMP%]{overflow:hidden;position:relative}.mat-ripple[_ngcontent-%COMP%]:not(:empty){transform:translateZ(0)}.mat-ripple.mat-ripple-unbounded[_ngcontent-%COMP%]{overflow:visible}.mat-ripple-element[_ngcontent-%COMP%]{position:absolute;border-radius:50%;pointer-events:none;transition:opacity,transform 0ms cubic-bezier(0, 0, 0.2, 1);transform:scale(0)}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{display:none}.cdk-visually-hidden[_ngcontent-%COMP%]{border:0;clip:rect(0 0 0 0);height:1px;margin:-1px;overflow:hidden;padding:0;position:absolute;width:1px;outline:0;-webkit-appearance:none;-moz-appearance:none}.cdk-overlay-container[_ngcontent-%COMP%], .cdk-global-overlay-wrapper[_ngcontent-%COMP%]{pointer-events:none;top:0;left:0;height:100%;width:100%}.cdk-overlay-container[_ngcontent-%COMP%]{position:fixed;z-index:1000}.cdk-overlay-container[_ngcontent-%COMP%]:empty{display:none}.cdk-global-overlay-wrapper[_ngcontent-%COMP%]{display:flex;position:absolute;z-index:1000}.cdk-overlay-pane[_ngcontent-%COMP%]{position:absolute;pointer-events:auto;box-sizing:border-box;z-index:1000;display:flex;max-width:100%;max-height:100%}.cdk-overlay-backdrop[_ngcontent-%COMP%]{position:absolute;top:0;bottom:0;left:0;right:0;z-index:1000;pointer-events:auto;-webkit-tap-highlight-color:transparent;transition:opacity 400ms cubic-bezier(0.25, 0.8, 0.25, 1);opacity:0}.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:1}@media screen and (-ms-high-contrast: active){.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:.6}}.cdk-overlay-dark-backdrop[_ngcontent-%COMP%]{background:rgba(0,0,0,.32)}.cdk-overlay-transparent-backdrop[_ngcontent-%COMP%], .cdk-overlay-transparent-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:0}.cdk-overlay-connected-position-bounding-box[_ngcontent-%COMP%]{position:absolute;z-index:1000;display:flex;flex-direction:column;min-width:1px;min-height:1px}.cdk-global-scrollblock[_ngcontent-%COMP%]{position:fixed;width:100%;overflow-y:scroll}@keyframes cdk-text-field-autofill-start{}@keyframes cdk-text-field-autofill-end{}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:-webkit-autofill{animation:cdk-text-field-autofill-start 0s 1ms}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:not(:-webkit-autofill){animation:cdk-text-field-autofill-end 0s 1ms}textarea.cdk-textarea-autosize[_ngcontent-%COMP%]{resize:none}textarea.cdk-textarea-autosize-measuring[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:auto !important;overflow:hidden !important}textarea.cdk-textarea-autosize-measuring-firefox[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:0 !important}.mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.1)}.mat-option[_ngcontent-%COMP%]{color:#212121}.mat-option[_ngcontent-%COMP%]:hover:not(.mat-option-disabled), .mat-option[_ngcontent-%COMP%]:focus:not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple):not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-active[_ngcontent-%COMP%]{background:rgba(0,0,0,.04);color:#212121}.mat-option.mat-option-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-primary[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f57c00}.mat-accent[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f44336}.mat-optgroup-label[_ngcontent-%COMP%]{color:#616161}.mat-optgroup-disabled[_ngcontent-%COMP%]   .mat-optgroup-label[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-pseudo-checkbox[_ngcontent-%COMP%]{color:#616161}.mat-pseudo-checkbox[_ngcontent-%COMP%]::after{color:#fafafa}.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{color:#b0b0b0}.mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f57c00}.mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f44336}.mat-pseudo-checkbox-checked.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{background:#b0b0b0}.mat-elevation-z0[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z1[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z2[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z3[_ngcontent-%COMP%]{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z4[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z5[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 5px 8px 0px rgba(0, 0, 0, 0.14),0px 1px 14px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z6[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z7[_ngcontent-%COMP%]{box-shadow:0px 4px 5px -2px rgba(0, 0, 0, 0.2),0px 7px 10px 1px rgba(0, 0, 0, 0.14),0px 2px 16px 1px rgba(0, 0, 0, 0.12)}.mat-elevation-z8[_ngcontent-%COMP%]{box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z9[_ngcontent-%COMP%]{box-shadow:0px 5px 6px -3px rgba(0, 0, 0, 0.2),0px 9px 12px 1px rgba(0, 0, 0, 0.14),0px 3px 16px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z10[_ngcontent-%COMP%]{box-shadow:0px 6px 6px -3px rgba(0, 0, 0, 0.2),0px 10px 14px 1px rgba(0, 0, 0, 0.14),0px 4px 18px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z11[_ngcontent-%COMP%]{box-shadow:0px 6px 7px -4px rgba(0, 0, 0, 0.2),0px 11px 15px 1px rgba(0, 0, 0, 0.14),0px 4px 20px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z12[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z13[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 13px 19px 2px rgba(0, 0, 0, 0.14),0px 5px 24px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z14[_ngcontent-%COMP%]{box-shadow:0px 7px 9px -4px rgba(0, 0, 0, 0.2),0px 14px 21px 2px rgba(0, 0, 0, 0.14),0px 5px 26px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z15[_ngcontent-%COMP%]{box-shadow:0px 8px 9px -5px rgba(0, 0, 0, 0.2),0px 15px 22px 2px rgba(0, 0, 0, 0.14),0px 6px 28px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z16[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z17[_ngcontent-%COMP%]{box-shadow:0px 8px 11px -5px rgba(0, 0, 0, 0.2),0px 17px 26px 2px rgba(0, 0, 0, 0.14),0px 6px 32px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z18[_ngcontent-%COMP%]{box-shadow:0px 9px 11px -5px rgba(0, 0, 0, 0.2),0px 18px 28px 2px rgba(0, 0, 0, 0.14),0px 7px 34px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z19[_ngcontent-%COMP%]{box-shadow:0px 9px 12px -6px rgba(0, 0, 0, 0.2),0px 19px 29px 2px rgba(0, 0, 0, 0.14),0px 7px 36px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z20[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 20px 31px 3px rgba(0, 0, 0, 0.14),0px 8px 38px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z21[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 21px 33px 3px rgba(0, 0, 0, 0.14),0px 8px 40px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z22[_ngcontent-%COMP%]{box-shadow:0px 10px 14px -6px rgba(0, 0, 0, 0.2),0px 22px 35px 3px rgba(0, 0, 0, 0.14),0px 8px 42px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z23[_ngcontent-%COMP%]{box-shadow:0px 11px 14px -7px rgba(0, 0, 0, 0.2),0px 23px 36px 3px rgba(0, 0, 0, 0.14),0px 9px 44px 8px rgba(0, 0, 0, 0.12)}.mat-elevation-z24[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12)}.mat-app-background[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-theme-loaded-marker[_ngcontent-%COMP%]{display:none}.mat-autocomplete-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-autocomplete-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover){background:#fff}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover):not(.mat-option-disabled){color:#212121}.mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f57c00}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{outline:solid 1px;border-radius:0}.mat-badge-accent[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-badge-warn[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f44336}.mat-badge[_ngcontent-%COMP%]{position:relative}.mat-badge-hidden[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{display:none}.mat-badge-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#b9b9b9;color:#757575}.mat-badge-content[_ngcontent-%COMP%]{position:absolute;text-align:center;display:inline-block;border-radius:50%;transition:transform 200ms ease-in-out;transform:scale(0.6);overflow:hidden;white-space:nowrap;text-overflow:ellipsis;pointer-events:none}.ng-animate-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%], .mat-badge-content._mat-animation-noopable[_ngcontent-%COMP%]{transition:none}.mat-badge-content.mat-badge-active[_ngcontent-%COMP%]{transform:none}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:16px;height:16px;line-height:16px}.mat-badge-small.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-8px}.mat-badge-small.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-8px}.mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-16px}.mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-16px}.mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-8px}.mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-8px}.mat-badge-medium[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:22px;height:22px;line-height:22px}.mat-badge-medium.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-11px}.mat-badge-medium.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-11px}.mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-22px}.mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-22px}.mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-11px}.mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-11px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:28px;height:28px;line-height:28px}.mat-badge-large.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-14px}.mat-badge-large.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-14px}.mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-28px}.mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-28px}.mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-14px}.mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-14px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]{color:inherit;background:transparent}.mat-button.mat-primary[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-button[disabled][disabled][_ngcontent-%COMP%], .mat-icon-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-icon-button[disabled][disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-stroked-button[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f44336}.mat-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:transparent}.mat-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{opacity:.1;background-color:currentColor}.mat-button-focus-overlay[_ngcontent-%COMP%]{background:#000}.mat-stroked-button[_ngcontent-%COMP%]:not([disabled]){border-color:rgba(0,0,0,.12)}.mat-flat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{color:#212121;background-color:#fff}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-flat-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-stroked-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-flat-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-raised-button[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone[_ngcontent-%COMP%], .mat-button-toggle-group[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{box-shadow:none}.mat-button-toggle[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-button-toggle[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121;background:#fff}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:#000}.mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12)}[dir=rtl][_ngcontent-%COMP%]   .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-group-appearance-standard.mat-button-toggle-vertical[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:none;border-top:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#616161}.mat-button-toggle-checked.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121}.mat-button-toggle-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.26);background-color:#eee}.mat-button-toggle-disabled.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{background:#fff}.mat-button-toggle-disabled.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#bdbdbd}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{border:solid 1px rgba(0,0,0,.12)}.mat-card[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-card[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-card.mat-card-flat[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-card-subtitle[_ngcontent-%COMP%]{color:#616161}.mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#616161}.mat-checkbox-checkmark[_ngcontent-%COMP%]{fill:#fafafa}.mat-checkbox-checkmark-path[_ngcontent-%COMP%]{stroke:#fafafa !important}.mat-checkbox-mixedmark[_ngcontent-%COMP%]{background-color:#fafafa}.mat-checkbox-indeterminate.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f57c00}.mat-checkbox-indeterminate.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#ff9800}.mat-checkbox-indeterminate.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f44336}.mat-checkbox-disabled.mat-checkbox-checked[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-disabled.mat-checkbox-indeterminate[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]:not(.mat-checkbox-checked)   .mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{color:#616161}.mat-checkbox[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%]{background:#f57c00}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%]{background:#ff9800}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%]{background:#f44336}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#212121}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#212121;opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled):active{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled)   .mat-chip-remove[_ngcontent-%COMP%]:hover{opacity:.54}.mat-chip.mat-standard-chip.mat-chip-disabled[_ngcontent-%COMP%]{opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]::after{background:#000}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-table[_ngcontent-%COMP%]{background:#fff}.mat-table[_ngcontent-%COMP%]   thead[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tbody[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tfoot[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], [mat-header-row][_ngcontent-%COMP%], [mat-row][_ngcontent-%COMP%], [mat-footer-row][_ngcontent-%COMP%], .mat-table-sticky[_ngcontent-%COMP%]{background:inherit}mat-row[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], th.mat-header-cell[_ngcontent-%COMP%], td.mat-cell[_ngcontent-%COMP%], td.mat-footer-cell[_ngcontent-%COMP%]{border-bottom-color:rgba(0,0,0,.12)}.mat-header-cell[_ngcontent-%COMP%]{color:#616161}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{color:#212121}.mat-calendar-arrow[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.54)}.mat-datepicker-toggle[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-next-button[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-previous-button[_ngcontent-%COMP%]{color:rgba(0,0,0,.54)}.mat-calendar-table-header[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-calendar-table-header-divider[_ngcontent-%COMP%]::after{background:rgba(0,0,0,.12)}.mat-calendar-body-label[_ngcontent-%COMP%]{color:#616161}.mat-calendar-body-cell-content[_ngcontent-%COMP%]{color:#212121;border-color:transparent}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){color:#757575}.mat-calendar-body-cell[_ngcontent-%COMP%]:not(.mat-calendar-body-disabled):hover > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-keyboard-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-program-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){background-color:rgba(0,0,0,.04)}.mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.38)}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.18)}.mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.4)}.mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12);background-color:#fff;color:#212121}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.4)}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.4)}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content-touch[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-datepicker-toggle-active[_ngcontent-%COMP%]{color:#f57c00}.mat-datepicker-toggle-active.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-datepicker-toggle-active.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-dialog-container[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-divider[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-divider-vertical[_ngcontent-%COMP%]{border-right-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-expansion-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-action-row[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-keyboard-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-program-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded)   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover:not([aria-disabled=true]){background:rgba(0,0,0,.04)}@media(hover: none){.mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded):not([aria-disabled=true])   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover{background:#fff}}.mat-expansion-panel-header-title[_ngcontent-%COMP%]{color:#212121}.mat-expansion-panel-header-description[_ngcontent-%COMP%], .mat-expansion-indicator[_ngcontent-%COMP%]::after{color:#616161}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-title[_ngcontent-%COMP%], .mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-description[_ngcontent-%COMP%]{color:inherit}.mat-form-field-label[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-hint[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-focused[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-ripple[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid)   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f57c00}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-accent   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#ff9800}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-warn   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#f44336}.mat-error[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-hint[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-legacy.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-standard[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-standard.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.04)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.02)}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:transparent}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.12)}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:rgba(0,0,0,.87)}.mat-form-field-appearance-outline.mat-focused[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field-appearance-outline.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-appearance-outline.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-invalid.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.06)}.mat-icon.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-icon.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-icon.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-type-mat-native-select[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#616161}.mat-input-element[_ngcontent-%COMP%]:disabled, .mat-form-field-type-mat-native-select.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#757575}.mat-input-element[_ngcontent-%COMP%]{caret-color:#f57c00}.mat-input-element[_ngcontent-%COMP%]::placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-moz-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-webkit-input-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]:-ms-input-placeholder{color:rgba(97,97,97,.42)}.mat-accent[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%], .mat-form-field-invalid[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#f44336}.mat-form-field-type-mat-native-select.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{color:#616161}.mat-list-item-disabled[_ngcontent-%COMP%]{background-color:#eee}.mat-list-option[_ngcontent-%COMP%]:hover, .mat-list-option[_ngcontent-%COMP%]:focus, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.04)}.mat-list-single-selected-option[_ngcontent-%COMP%], .mat-list-single-selected-option[_ngcontent-%COMP%]:hover, .mat-list-single-selected-option[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.12)}.mat-menu-panel[_ngcontent-%COMP%]{background:#fff}.mat-menu-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-menu-item[_ngcontent-%COMP%]{background:transparent;color:#212121}.mat-menu-item[disabled][_ngcontent-%COMP%], .mat-menu-item[disabled][_ngcontent-%COMP%]::after{color:rgba(0,0,0,.38)}.mat-menu-item[_ngcontent-%COMP%]   .mat-icon-no-color[_ngcontent-%COMP%], .mat-menu-item-submenu-trigger[_ngcontent-%COMP%]::after{color:rgba(0,0,0,.54)}.mat-menu-item[_ngcontent-%COMP%]:hover:not([disabled]), .mat-menu-item.cdk-program-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item.cdk-keyboard-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item-highlighted[_ngcontent-%COMP%]:not([disabled]){background:rgba(0,0,0,.04)}.mat-paginator[_ngcontent-%COMP%]{background:#fff}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{color:#616161}.mat-paginator-decrement[_ngcontent-%COMP%], .mat-paginator-increment[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54);border-right:2px solid rgba(0,0,0,.54)}.mat-paginator-first[_ngcontent-%COMP%], .mat-paginator-last[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54)}.mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-decrement[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-increment[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-first[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-last[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f57c00}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#ff9800}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f44336}.mat-progress-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f57c00}.mat-progress-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#ff9800}.mat-progress-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f44336}.mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#616161}.mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f57c00}.mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#ff9800}.mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#ff9800}.mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f44336}.mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f44336}.mat-radio-button.mat-radio-disabled.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-label-content[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-radio-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-select-value[_ngcontent-%COMP%]{color:#212121}.mat-select-placeholder[_ngcontent-%COMP%]{color:rgba(97,97,97,.42)}.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%]{color:#757575}.mat-select-arrow[_ngcontent-%COMP%]{color:#616161}.mat-select-panel[_ngcontent-%COMP%]{background:#fff}.mat-select-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-select-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple){background:rgba(0,0,0,.12)}.mat-form-field.mat-focused.mat-primary[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-invalid[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#757575}.mat-drawer-container[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-drawer[_ngcontent-%COMP%]{background-color:#fff;color:#212121}.mat-drawer.mat-drawer-push[_ngcontent-%COMP%]{background-color:#fff}.mat-drawer[_ngcontent-%COMP%]:not(.mat-drawer-side){box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-drawer-side[_ngcontent-%COMP%]{border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-backdrop.mat-drawer-shown[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.6)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.54)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.54)}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.54)}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle[_ngcontent-%COMP%]:not(.mat-checked)   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-slide-toggle-thumb[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12);background-color:#fafafa}.mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-primary[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f57c00}.mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-accent[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#ff9800}.mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-warn[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f44336}.mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.2)}.mat-slider[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%], .cdk-focused[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-disabled[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]   .mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing)   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26);background-color:transparent}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26)}.mat-slider-has-ticks[_ngcontent-%COMP%]   .mat-slider-wrapper[_ngcontent-%COMP%]::after{border-color:rgba(0,0,0,.7)}.mat-slider-horizontal[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to right, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent);background-image:-moz-repeating-linear-gradient(0.0001deg, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-slider-vertical[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to bottom, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-step-header.cdk-keyboard-focused[_ngcontent-%COMP%], .mat-step-header.cdk-program-focused[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]:hover{background-color:rgba(0,0,0,.04)}@media(hover: none){.mat-step-header[_ngcontent-%COMP%]:hover{background:none}}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-optional[_ngcontent-%COMP%]{color:#616161}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon[_ngcontent-%COMP%]{background-color:#616161;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-selected[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-done[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-edit[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-error[_ngcontent-%COMP%]{background-color:transparent;color:#f44336}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-active[_ngcontent-%COMP%]{color:#212121}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-error[_ngcontent-%COMP%]{color:#f44336}.mat-stepper-horizontal[_ngcontent-%COMP%], .mat-stepper-vertical[_ngcontent-%COMP%]{background-color:#fff}.mat-stepper-vertical-line[_ngcontent-%COMP%]::before{border-left-color:rgba(0,0,0,.12)}.mat-horizontal-stepper-header[_ngcontent-%COMP%]::before, .mat-horizontal-stepper-header[_ngcontent-%COMP%]::after, .mat-stepper-horizontal-line[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-sort-header-arrow[_ngcontent-%COMP%]{color:#616161}.mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-header[_ngcontent-%COMP%]{border-bottom:1px solid rgba(0,0,0,.12)}.mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%]{border-top:1px solid rgba(0,0,0,.12);border-bottom:none}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{color:#212121}.mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:#757575}.mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#212121}.mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#757575}.mat-tab-group[class*=mat-background-][_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar[class*=mat-background-][_ngcontent-%COMP%]{border-bottom:none;border-top:none}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-toolbar[_ngcontent-%COMP%]{background:#f5f5f5;color:#212121}.mat-toolbar.mat-primary[_ngcontent-%COMP%]{background:#f57c00;color:#fff}.mat-toolbar.mat-accent[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-toolbar.mat-warn[_ngcontent-%COMP%]{background:#f44336;color:#fff}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:currentColor}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:inherit}.mat-toolbar[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:currentColor}.mat-tooltip[_ngcontent-%COMP%]{background:rgba(97,97,97,.9)}.mat-tree[_ngcontent-%COMP%]{background:#fff}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{color:#212121}.mat-snack-bar-container[_ngcontent-%COMP%]{color:rgba(255,255,255,.7);background:#323232;box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-simple-snackbar-action[_ngcontent-%COMP%]{color:#ff9800}.chart-container[_ngcontent-%COMP%]{background-color:#fff;border-bottom:1px solid #ebebeb;display:flex;flex-direction:column;overflow:hidden}.chart[_ngcontent-%COMP%]{height:300px;width:100%}.chart-head[_ngcontent-%COMP%]{align-items:center;display:flex;justify-content:space-between}.chart-heading[_ngcontent-%COMP%]{font-size:13px;padding-left:10px;padding-top:10px}.draw-area[_ngcontent-%COMP%]{height:100%;width:100%}.stroked-line[_ngcontent-%COMP%]{stroke:rgba(0,0,0,.12);stroke-dasharray:3 3}"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/violin_filters/violin_filter/violin_filter_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class ViolinFilterContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.activeRuns$ = this.store
                .pipe(select(getCurrentRouteRunSelection))
                .pipe(map((/**
             * @param {?} runSelection
             * @return {?}
             */
            (runSelection) => {
                if (!runSelection)
                    return [];
                return Array.from(runSelection.entries())
                    .filter((/**
                 * @param {?} run
                 * @return {?}
                 */
                (run) => run[1]))
                    .map((/**
                 * @param {?} run
                 * @return {?}
                 */
                (run) => run[0]));
            })));
            this.visibleAnnotations$ = combineLatest([
                this.store.select(getAnnotationData),
                this.store.select(getHiddenAnnotations),
                this.store.select(getShowHiddenAnnotations),
            ]).pipe(map((/**
             * @param {?} __0
             * @return {?}
             */
            ([annotationData, hiddenAnnotations, showHiddenAnnotations]) => {
                return removeHiddenAnnotations(annotationData, hiddenAnnotations, showHiddenAnnotations);
            })));
            this.chartWidth$ = this.store.pipe(select(getSidebarWidth)).pipe(map((/**
             * @param {?} width
             * @return {?}
             */
            (width) => {
                return Math.max(150, width);
            })));
        }
        /**
         * @return {?}
         */
        ngOnInit() {
            this.chartData$ = combineLatest([
                this.visibleAnnotations$,
                this.activeRuns$,
            ]).pipe(map((/**
             * @param {?} __0
             * @return {?}
             */
            ([visibleAnnotations, activeRuns]) => {
                return violinData(visibleAnnotations, activeRuns, this.metricName);
            })));
        }
        /**
         * @return {?}
         */
        removeMetric() {
            this.store.dispatch(npmiRemoveMetricFilter({ metric: this.metricName }));
        }
        /**
         * @param {?} filter
         * @return {?}
         */
        updateFilter(filter) {
            this.store.dispatch(npmiChangeMetricFilter(Object.assign({ metric: this.metricName }, filter)));
        }
    }
    ViolinFilterContainer.decorators = [
        { type: Component, args: [{
                    selector: 'npmi-violin-filter',
                    template: `
    <violin-filter-component
      [metricName]="metricName"
      [filter]="filter"
      [chartData]="chartData$ | async"
      [width]="chartWidth$ | async"
      (onRemove)="removeMetric()"
      (onUpdateFilter)="updateFilter($event)"
    ></violin-filter-component>
  `,
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    ViolinFilterContainer.ctorParameters = () => [
        { type: Store }
    ];
    ViolinFilterContainer.propDecorators = {
        metricName: [{ type: Input }],
        filter: [{ type: Input }]
    };
    /** @nocollapse */ ViolinFilterContainer.ɵfac = function ViolinFilterContainer_Factory(t) { return new (t || ViolinFilterContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ ViolinFilterContainer.ɵcmp = ɵɵdefineComponent({ type: ViolinFilterContainer, selectors: [["npmi-violin-filter"]], inputs: { metricName: "metricName", filter: "filter" }, decls: 3, vars: 8, consts: [[3, "metricName", "filter", "chartData", "width", "onRemove", "onUpdateFilter"]], template: function ViolinFilterContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "violin-filter-component", 0);
            ɵɵlistener("onRemove", function ViolinFilterContainer_Template_violin_filter_component_onRemove_0_listener() { return ctx.removeMetric(); })("onUpdateFilter", function ViolinFilterContainer_Template_violin_filter_component_onUpdateFilter_0_listener($event) { return ctx.updateFilter($event); });
            ɵɵpipe(1, "async");
            ɵɵpipe(2, "async");
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵproperty("metricName", ctx.metricName)("filter", ctx.filter)("chartData", ɵɵpipeBind1(1, 4, ctx.chartData$))("width", ɵɵpipeBind1(2, 6, ctx.chartWidth$));
        } }, directives: [ViolinFilterComponent], pipes: [AsyncPipe], encapsulation: 2, changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/violin_filters/violin_filters_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    var I18N_0$8;
    if (typeof ngI18nClosureMode !== "undefined" && ngI18nClosureMode) {
        /**
         * @desc Label for a button that expands/hides the sidebar.
         */ 
        const MSG_EXTERNAL_4658602991970260215$$TENSORBOARD_WEBAPP_PLUGINS_NPMI_VIEWS_VIOLIN_FILTERS_VIOLIN_FILTERS_COMPONENT_TS_1 = goog.getMsg("Expand/Hide Sidebar");
        I18N_0$8 = MSG_EXTERNAL_4658602991970260215$$TENSORBOARD_WEBAPP_PLUGINS_NPMI_VIEWS_VIOLIN_FILTERS_VIOLIN_FILTERS_COMPONENT_TS_1;
    }
    else {
        I18N_0$8 = $localize `:Label for a button that expands/hides the sidebar.␟48c29903ce881ab61088f8d49d827203716aaed4␟4658602991970260215:Expand/Hide Sidebar`;
    }
    const _c2$e = ["aria-label", I18N_0$8];
    function ViolinFiltersComponent_npmi_violin_filter_8_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "npmi-violin-filter", 8);
    } if (rf & 2) {
        const filter_r2 = ctx.$implicit;
        ɵɵproperty("metricName", filter_r2[0])("filter", filter_r2[1]);
    } }
    function ViolinFiltersComponent_div_9_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 9);
        ɵɵelementStart(1, "span", 10);
        ɵɵtext(2, " You can add more filters at the top. ");
        ɵɵelementEnd();
        ɵɵelementEnd();
    } }
    class ViolinFiltersComponent {
        constructor() {
            this.toggleSidebarExpanded = new EventEmitter();
        }
    }
    ViolinFiltersComponent.decorators = [
        { type: Component, args: [{
                    selector: 'violin-filters-component',
                    templateUrl: './violin_filters_component.ng.html',
                    styleUrls: ['./violin_filters_component.css'],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    ViolinFiltersComponent.propDecorators = {
        sidebarExpanded: [{ type: Input }],
        metricFilters: [{ type: Input }],
        toggleSidebarExpanded: [{ type: Output }]
    };
    /** @nocollapse */ ViolinFiltersComponent.ɵfac = function ViolinFiltersComponent_Factory(t) { return new (t || ViolinFiltersComponent)(); };
    /** @nocollapse */ ViolinFiltersComponent.ɵcmp = ɵɵdefineComponent({ type: ViolinFiltersComponent, selectors: [["violin-filters-component"]], inputs: { sidebarExpanded: "sidebarExpanded", metricFilters: "metricFilters" }, outputs: { toggleSidebarExpanded: "toggleSidebarExpanded" }, decls: 10, vars: 2, consts: [[1, "filters-toolbar"], [1, "filters-title"], [1, "side-toggle"], ["mat-icon-button", "", 3, "click", 6, "aria-label"], ["svgIcon", "chevron_left_24px"], [1, "filters"], [3, "metricName", "filter", 4, "ngFor", "ngForOf"], ["class", "filters-hint", 4, "ngIf"], [3, "metricName", "filter"], [1, "filters-hint"], [1, "filters-hint-text"]], template: function ViolinFiltersComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "div", 0);
            ɵɵelementStart(1, "h3", 1);
            ɵɵtext(2, "Active Filters");
            ɵɵelementEnd();
            ɵɵelementStart(3, "div", 2);
            ɵɵelementStart(4, "button", 3);
            ɵɵi18nAttributes(5, _c2$e);
            ɵɵlistener("click", function ViolinFiltersComponent_Template_button_click_4_listener() { return ctx.toggleSidebarExpanded.emit(); });
            ɵɵelement(6, "mat-icon", 4);
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementStart(7, "div", 5);
            ɵɵtemplate(8, ViolinFiltersComponent_npmi_violin_filter_8_Template, 1, 2, "npmi-violin-filter", 6);
            ɵɵelementEnd();
            ɵɵtemplate(9, ViolinFiltersComponent_div_9_Template, 3, 0, "div", 7);
        } if (rf & 2) {
            ɵɵadvance(8);
            ɵɵproperty("ngForOf", ctx.metricFilters);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx.metricFilters.length === 0);
        } }, directives: [MatButton, MatIcon, NgForOf, NgIf, ViolinFilterContainer], styles: [".mat-badge-content[_ngcontent-%COMP%]{font-weight:600;font-size:12px;font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:9px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:24px}.mat-h1[_ngcontent-%COMP%], .mat-headline[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%]{font:400 24px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h2[_ngcontent-%COMP%], .mat-title[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h3[_ngcontent-%COMP%], .mat-subheading-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%]{font:400 16px/28px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h4[_ngcontent-%COMP%], .mat-subheading-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%]{font:400 15px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h5[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%]{font:400 calc(14px * 0.83)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-h6[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:400 calc(14px * 0.67)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-body-strong[_ngcontent-%COMP%], .mat-body-2[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   p[_ngcontent-%COMP%]{margin:0 0 12px}.mat-small[_ngcontent-%COMP%], .mat-caption[_ngcontent-%COMP%]{font:400 12px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-display-4[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-4[_ngcontent-%COMP%]{font:300 112px/112px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.05em;margin:0 0 56px}.mat-display-3[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-3[_ngcontent-%COMP%]{font:400 56px/56px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.02em;margin:0 0 64px}.mat-display-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-2[_ngcontent-%COMP%]{font:400 45px/48px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.005em;margin:0 0 64px}.mat-display-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-1[_ngcontent-%COMP%]{font:400 34px/40px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 64px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%], .mat-flat-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-button-toggle[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card-title[_ngcontent-%COMP%]{font-size:24px;font-weight:500}.mat-card-header[_ngcontent-%COMP%]   .mat-card-title[_ngcontent-%COMP%]{font-size:20px}.mat-card-subtitle[_ngcontent-%COMP%], .mat-card-content[_ngcontent-%COMP%]{font-size:14px}.mat-checkbox[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-checkbox-layout[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{line-height:24px}.mat-chip[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-chip[_ngcontent-%COMP%]   .mat-chip-trailing-icon.mat-icon[_ngcontent-%COMP%], .mat-chip[_ngcontent-%COMP%]   .mat-chip-remove.mat-icon[_ngcontent-%COMP%]{font-size:18px}.mat-table[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-header-cell[_ngcontent-%COMP%]{font-size:12px;font-weight:500}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{font-size:14px}.mat-calendar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-calendar-body[_ngcontent-%COMP%]{font-size:13px}.mat-calendar-body-label[_ngcontent-%COMP%], .mat-calendar-period-button[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-calendar-table-header[_ngcontent-%COMP%]   th[_ngcontent-%COMP%]{font-size:11px;font-weight:400}.mat-dialog-title[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-expansion-panel-header[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:15px;font-weight:400}.mat-expansion-panel-content[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field[_ngcontent-%COMP%]{font-size:inherit;font-weight:400;line-height:1.125;font-family:Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.34375em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{font-size:150%;line-height:1.125}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]{height:1.5em;width:1.5em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{height:1.125em;line-height:1.125}.mat-form-field-infix[_ngcontent-%COMP%]{padding:.5em 0;border-top:.84375em solid transparent}.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34375em) scale(0.75);width:133.3333333333%}.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34374em) scale(0.75);width:133.3333433333%}.mat-form-field-label-wrapper[_ngcontent-%COMP%]{top:-0.84375em;padding-top:.84375em}.mat-form-field-label[_ngcontent-%COMP%]{top:1.34375em}.mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.34375em}.mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{font-size:75%;margin-top:.6666666667em;top:calc(100% - 1.7916666667em)}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.4375em 0}.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.001px);-ms-transform:translateY(-1.28125em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00101px);-ms-transform:translateY(-1.28124em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00102px);-ms-transform:translateY(-1.28123em) scale(0.75);width:133.3333533333%}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.28125em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{margin-top:.5416666667em;top:calc(100% - 1.6666666667em)}@media print{.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28122em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28121em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.2812em) scale(0.75)}}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.25em 0 .75em 0}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.09375em;margin-top:-0.5em}.mat-form-field-appearance-fill.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59374em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:1em 0 1em 0}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.84375em;margin-top:-0.25em}.mat-form-field-appearance-outline.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59374em) scale(0.75);width:133.3333433333%}.mat-grid-tile-header[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]{font-size:14px}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2), .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}input.mat-input-element[_ngcontent-%COMP%]{margin-top:-0.0625em}.mat-menu-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:400}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px}.mat-radio-button[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select-trigger[_ngcontent-%COMP%]{height:1.125em}.mat-slide-toggle-content[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-slider-thumb-label-text[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-stepper-vertical[_ngcontent-%COMP%], .mat-stepper-horizontal[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-step-label[_ngcontent-%COMP%]{font-size:14px;font-weight:400}.mat-step-sub-label-error[_ngcontent-%COMP%]{font-weight:normal}.mat-step-label-error[_ngcontent-%COMP%]{font-size:14px}.mat-step-label-selected[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-tab-group[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-toolbar[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0}.mat-tooltip[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:10px;padding-top:6px;padding-bottom:6px}.mat-tooltip-handset[_ngcontent-%COMP%]{font-size:14px;padding-top:8px;padding-bottom:8px}.mat-list-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:16px}.mat-optgroup-label[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-simple-snackbar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px}.mat-simple-snackbar-action[_ngcontent-%COMP%]{line-height:1;font-family:inherit;font-size:inherit;font-weight:500}.mat-tree[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{font-weight:400;font-size:14px}.mat-ripple[_ngcontent-%COMP%]{overflow:hidden;position:relative}.mat-ripple[_ngcontent-%COMP%]:not(:empty){transform:translateZ(0)}.mat-ripple.mat-ripple-unbounded[_ngcontent-%COMP%]{overflow:visible}.mat-ripple-element[_ngcontent-%COMP%]{position:absolute;border-radius:50%;pointer-events:none;transition:opacity,transform 0ms cubic-bezier(0, 0, 0.2, 1);transform:scale(0)}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{display:none}.cdk-visually-hidden[_ngcontent-%COMP%]{border:0;clip:rect(0 0 0 0);height:1px;margin:-1px;overflow:hidden;padding:0;position:absolute;width:1px;outline:0;-webkit-appearance:none;-moz-appearance:none}.cdk-overlay-container[_ngcontent-%COMP%], .cdk-global-overlay-wrapper[_ngcontent-%COMP%]{pointer-events:none;top:0;left:0;height:100%;width:100%}.cdk-overlay-container[_ngcontent-%COMP%]{position:fixed;z-index:1000}.cdk-overlay-container[_ngcontent-%COMP%]:empty{display:none}.cdk-global-overlay-wrapper[_ngcontent-%COMP%]{display:flex;position:absolute;z-index:1000}.cdk-overlay-pane[_ngcontent-%COMP%]{position:absolute;pointer-events:auto;box-sizing:border-box;z-index:1000;display:flex;max-width:100%;max-height:100%}.cdk-overlay-backdrop[_ngcontent-%COMP%]{position:absolute;top:0;bottom:0;left:0;right:0;z-index:1000;pointer-events:auto;-webkit-tap-highlight-color:transparent;transition:opacity 400ms cubic-bezier(0.25, 0.8, 0.25, 1);opacity:0}.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:1}@media screen and (-ms-high-contrast: active){.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:.6}}.cdk-overlay-dark-backdrop[_ngcontent-%COMP%]{background:rgba(0,0,0,.32)}.cdk-overlay-transparent-backdrop[_ngcontent-%COMP%], .cdk-overlay-transparent-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:0}.cdk-overlay-connected-position-bounding-box[_ngcontent-%COMP%]{position:absolute;z-index:1000;display:flex;flex-direction:column;min-width:1px;min-height:1px}.cdk-global-scrollblock[_ngcontent-%COMP%]{position:fixed;width:100%;overflow-y:scroll}@keyframes cdk-text-field-autofill-start{}@keyframes cdk-text-field-autofill-end{}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:-webkit-autofill{animation:cdk-text-field-autofill-start 0s 1ms}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:not(:-webkit-autofill){animation:cdk-text-field-autofill-end 0s 1ms}textarea.cdk-textarea-autosize[_ngcontent-%COMP%]{resize:none}textarea.cdk-textarea-autosize-measuring[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:auto !important;overflow:hidden !important}textarea.cdk-textarea-autosize-measuring-firefox[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:0 !important}.mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.1)}.mat-option[_ngcontent-%COMP%]{color:#212121}.mat-option[_ngcontent-%COMP%]:hover:not(.mat-option-disabled), .mat-option[_ngcontent-%COMP%]:focus:not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple):not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-active[_ngcontent-%COMP%]{background:rgba(0,0,0,.04);color:#212121}.mat-option.mat-option-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-primary[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f57c00}.mat-accent[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f44336}.mat-optgroup-label[_ngcontent-%COMP%]{color:#616161}.mat-optgroup-disabled[_ngcontent-%COMP%]   .mat-optgroup-label[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-pseudo-checkbox[_ngcontent-%COMP%]{color:#616161}.mat-pseudo-checkbox[_ngcontent-%COMP%]::after{color:#fafafa}.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{color:#b0b0b0}.mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f57c00}.mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f44336}.mat-pseudo-checkbox-checked.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{background:#b0b0b0}.mat-elevation-z0[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z1[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z2[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z3[_ngcontent-%COMP%]{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z4[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z5[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 5px 8px 0px rgba(0, 0, 0, 0.14),0px 1px 14px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z6[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z7[_ngcontent-%COMP%]{box-shadow:0px 4px 5px -2px rgba(0, 0, 0, 0.2),0px 7px 10px 1px rgba(0, 0, 0, 0.14),0px 2px 16px 1px rgba(0, 0, 0, 0.12)}.mat-elevation-z8[_ngcontent-%COMP%]{box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z9[_ngcontent-%COMP%]{box-shadow:0px 5px 6px -3px rgba(0, 0, 0, 0.2),0px 9px 12px 1px rgba(0, 0, 0, 0.14),0px 3px 16px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z10[_ngcontent-%COMP%]{box-shadow:0px 6px 6px -3px rgba(0, 0, 0, 0.2),0px 10px 14px 1px rgba(0, 0, 0, 0.14),0px 4px 18px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z11[_ngcontent-%COMP%]{box-shadow:0px 6px 7px -4px rgba(0, 0, 0, 0.2),0px 11px 15px 1px rgba(0, 0, 0, 0.14),0px 4px 20px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z12[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z13[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 13px 19px 2px rgba(0, 0, 0, 0.14),0px 5px 24px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z14[_ngcontent-%COMP%]{box-shadow:0px 7px 9px -4px rgba(0, 0, 0, 0.2),0px 14px 21px 2px rgba(0, 0, 0, 0.14),0px 5px 26px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z15[_ngcontent-%COMP%]{box-shadow:0px 8px 9px -5px rgba(0, 0, 0, 0.2),0px 15px 22px 2px rgba(0, 0, 0, 0.14),0px 6px 28px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z16[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z17[_ngcontent-%COMP%]{box-shadow:0px 8px 11px -5px rgba(0, 0, 0, 0.2),0px 17px 26px 2px rgba(0, 0, 0, 0.14),0px 6px 32px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z18[_ngcontent-%COMP%]{box-shadow:0px 9px 11px -5px rgba(0, 0, 0, 0.2),0px 18px 28px 2px rgba(0, 0, 0, 0.14),0px 7px 34px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z19[_ngcontent-%COMP%]{box-shadow:0px 9px 12px -6px rgba(0, 0, 0, 0.2),0px 19px 29px 2px rgba(0, 0, 0, 0.14),0px 7px 36px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z20[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 20px 31px 3px rgba(0, 0, 0, 0.14),0px 8px 38px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z21[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 21px 33px 3px rgba(0, 0, 0, 0.14),0px 8px 40px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z22[_ngcontent-%COMP%]{box-shadow:0px 10px 14px -6px rgba(0, 0, 0, 0.2),0px 22px 35px 3px rgba(0, 0, 0, 0.14),0px 8px 42px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z23[_ngcontent-%COMP%]{box-shadow:0px 11px 14px -7px rgba(0, 0, 0, 0.2),0px 23px 36px 3px rgba(0, 0, 0, 0.14),0px 9px 44px 8px rgba(0, 0, 0, 0.12)}.mat-elevation-z24[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12)}.mat-app-background[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-theme-loaded-marker[_ngcontent-%COMP%]{display:none}.mat-autocomplete-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-autocomplete-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover){background:#fff}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover):not(.mat-option-disabled){color:#212121}.mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f57c00}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{outline:solid 1px;border-radius:0}.mat-badge-accent[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-badge-warn[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f44336}.mat-badge[_ngcontent-%COMP%]{position:relative}.mat-badge-hidden[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{display:none}.mat-badge-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#b9b9b9;color:#757575}.mat-badge-content[_ngcontent-%COMP%]{position:absolute;text-align:center;display:inline-block;border-radius:50%;transition:transform 200ms ease-in-out;transform:scale(0.6);overflow:hidden;white-space:nowrap;text-overflow:ellipsis;pointer-events:none}.ng-animate-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%], .mat-badge-content._mat-animation-noopable[_ngcontent-%COMP%]{transition:none}.mat-badge-content.mat-badge-active[_ngcontent-%COMP%]{transform:none}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:16px;height:16px;line-height:16px}.mat-badge-small.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-8px}.mat-badge-small.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-8px}.mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-16px}.mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-16px}.mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-8px}.mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-8px}.mat-badge-medium[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:22px;height:22px;line-height:22px}.mat-badge-medium.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-11px}.mat-badge-medium.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-11px}.mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-22px}.mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-22px}.mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-11px}.mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-11px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:28px;height:28px;line-height:28px}.mat-badge-large.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-14px}.mat-badge-large.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-14px}.mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-28px}.mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-28px}.mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-14px}.mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-14px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]{color:inherit;background:transparent}.mat-button.mat-primary[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-button[disabled][disabled][_ngcontent-%COMP%], .mat-icon-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-icon-button[disabled][disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-stroked-button[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f44336}.mat-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:transparent}.mat-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{opacity:.1;background-color:currentColor}.mat-button-focus-overlay[_ngcontent-%COMP%]{background:#000}.mat-stroked-button[_ngcontent-%COMP%]:not([disabled]){border-color:rgba(0,0,0,.12)}.mat-flat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{color:#212121;background-color:#fff}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-flat-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-stroked-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-flat-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-raised-button[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone[_ngcontent-%COMP%], .mat-button-toggle-group[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{box-shadow:none}.mat-button-toggle[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-button-toggle[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121;background:#fff}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:#000}.mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12)}[dir=rtl][_ngcontent-%COMP%]   .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-group-appearance-standard.mat-button-toggle-vertical[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:none;border-top:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#616161}.mat-button-toggle-checked.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121}.mat-button-toggle-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.26);background-color:#eee}.mat-button-toggle-disabled.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{background:#fff}.mat-button-toggle-disabled.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#bdbdbd}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{border:solid 1px rgba(0,0,0,.12)}.mat-card[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-card[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-card.mat-card-flat[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-card-subtitle[_ngcontent-%COMP%]{color:#616161}.mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#616161}.mat-checkbox-checkmark[_ngcontent-%COMP%]{fill:#fafafa}.mat-checkbox-checkmark-path[_ngcontent-%COMP%]{stroke:#fafafa !important}.mat-checkbox-mixedmark[_ngcontent-%COMP%]{background-color:#fafafa}.mat-checkbox-indeterminate.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f57c00}.mat-checkbox-indeterminate.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#ff9800}.mat-checkbox-indeterminate.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f44336}.mat-checkbox-disabled.mat-checkbox-checked[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-disabled.mat-checkbox-indeterminate[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]:not(.mat-checkbox-checked)   .mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{color:#616161}.mat-checkbox[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%]{background:#f57c00}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%]{background:#ff9800}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%]{background:#f44336}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#212121}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#212121;opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled):active{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled)   .mat-chip-remove[_ngcontent-%COMP%]:hover{opacity:.54}.mat-chip.mat-standard-chip.mat-chip-disabled[_ngcontent-%COMP%]{opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]::after{background:#000}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-table[_ngcontent-%COMP%]{background:#fff}.mat-table[_ngcontent-%COMP%]   thead[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tbody[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tfoot[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], [mat-header-row][_ngcontent-%COMP%], [mat-row][_ngcontent-%COMP%], [mat-footer-row][_ngcontent-%COMP%], .mat-table-sticky[_ngcontent-%COMP%]{background:inherit}mat-row[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], th.mat-header-cell[_ngcontent-%COMP%], td.mat-cell[_ngcontent-%COMP%], td.mat-footer-cell[_ngcontent-%COMP%]{border-bottom-color:rgba(0,0,0,.12)}.mat-header-cell[_ngcontent-%COMP%]{color:#616161}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{color:#212121}.mat-calendar-arrow[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.54)}.mat-datepicker-toggle[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-next-button[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-previous-button[_ngcontent-%COMP%]{color:rgba(0,0,0,.54)}.mat-calendar-table-header[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-calendar-table-header-divider[_ngcontent-%COMP%]::after{background:rgba(0,0,0,.12)}.mat-calendar-body-label[_ngcontent-%COMP%]{color:#616161}.mat-calendar-body-cell-content[_ngcontent-%COMP%]{color:#212121;border-color:transparent}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){color:#757575}.mat-calendar-body-cell[_ngcontent-%COMP%]:not(.mat-calendar-body-disabled):hover > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-keyboard-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-program-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){background-color:rgba(0,0,0,.04)}.mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.38)}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.18)}.mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.4)}.mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12);background-color:#fff;color:#212121}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.4)}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.4)}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content-touch[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-datepicker-toggle-active[_ngcontent-%COMP%]{color:#f57c00}.mat-datepicker-toggle-active.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-datepicker-toggle-active.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-dialog-container[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-divider[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-divider-vertical[_ngcontent-%COMP%]{border-right-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-expansion-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-action-row[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-keyboard-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-program-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded)   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover:not([aria-disabled=true]){background:rgba(0,0,0,.04)}@media(hover: none){.mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded):not([aria-disabled=true])   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover{background:#fff}}.mat-expansion-panel-header-title[_ngcontent-%COMP%]{color:#212121}.mat-expansion-panel-header-description[_ngcontent-%COMP%], .mat-expansion-indicator[_ngcontent-%COMP%]::after{color:#616161}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-title[_ngcontent-%COMP%], .mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-description[_ngcontent-%COMP%]{color:inherit}.mat-form-field-label[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-hint[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-focused[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-ripple[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid)   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f57c00}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-accent   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#ff9800}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-warn   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#f44336}.mat-error[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-hint[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-legacy.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-standard[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-standard.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.04)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.02)}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:transparent}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.12)}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:rgba(0,0,0,.87)}.mat-form-field-appearance-outline.mat-focused[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field-appearance-outline.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-appearance-outline.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-invalid.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.06)}.mat-icon.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-icon.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-icon.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-type-mat-native-select[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#616161}.mat-input-element[_ngcontent-%COMP%]:disabled, .mat-form-field-type-mat-native-select.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#757575}.mat-input-element[_ngcontent-%COMP%]{caret-color:#f57c00}.mat-input-element[_ngcontent-%COMP%]::placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-moz-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-webkit-input-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]:-ms-input-placeholder{color:rgba(97,97,97,.42)}.mat-accent[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%], .mat-form-field-invalid[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#f44336}.mat-form-field-type-mat-native-select.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{color:#616161}.mat-list-item-disabled[_ngcontent-%COMP%]{background-color:#eee}.mat-list-option[_ngcontent-%COMP%]:hover, .mat-list-option[_ngcontent-%COMP%]:focus, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.04)}.mat-list-single-selected-option[_ngcontent-%COMP%], .mat-list-single-selected-option[_ngcontent-%COMP%]:hover, .mat-list-single-selected-option[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.12)}.mat-menu-panel[_ngcontent-%COMP%]{background:#fff}.mat-menu-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-menu-item[_ngcontent-%COMP%]{background:transparent;color:#212121}.mat-menu-item[disabled][_ngcontent-%COMP%], .mat-menu-item[disabled][_ngcontent-%COMP%]::after{color:rgba(0,0,0,.38)}.mat-menu-item[_ngcontent-%COMP%]   .mat-icon-no-color[_ngcontent-%COMP%], .mat-menu-item-submenu-trigger[_ngcontent-%COMP%]::after{color:rgba(0,0,0,.54)}.mat-menu-item[_ngcontent-%COMP%]:hover:not([disabled]), .mat-menu-item.cdk-program-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item.cdk-keyboard-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item-highlighted[_ngcontent-%COMP%]:not([disabled]){background:rgba(0,0,0,.04)}.mat-paginator[_ngcontent-%COMP%]{background:#fff}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{color:#616161}.mat-paginator-decrement[_ngcontent-%COMP%], .mat-paginator-increment[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54);border-right:2px solid rgba(0,0,0,.54)}.mat-paginator-first[_ngcontent-%COMP%], .mat-paginator-last[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54)}.mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-decrement[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-increment[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-first[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-last[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f57c00}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#ff9800}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f44336}.mat-progress-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f57c00}.mat-progress-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#ff9800}.mat-progress-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f44336}.mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#616161}.mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f57c00}.mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#ff9800}.mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#ff9800}.mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f44336}.mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f44336}.mat-radio-button.mat-radio-disabled.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-label-content[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-radio-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-select-value[_ngcontent-%COMP%]{color:#212121}.mat-select-placeholder[_ngcontent-%COMP%]{color:rgba(97,97,97,.42)}.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%]{color:#757575}.mat-select-arrow[_ngcontent-%COMP%]{color:#616161}.mat-select-panel[_ngcontent-%COMP%]{background:#fff}.mat-select-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-select-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple){background:rgba(0,0,0,.12)}.mat-form-field.mat-focused.mat-primary[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-invalid[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#757575}.mat-drawer-container[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-drawer[_ngcontent-%COMP%]{background-color:#fff;color:#212121}.mat-drawer.mat-drawer-push[_ngcontent-%COMP%]{background-color:#fff}.mat-drawer[_ngcontent-%COMP%]:not(.mat-drawer-side){box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-drawer-side[_ngcontent-%COMP%]{border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-backdrop.mat-drawer-shown[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.6)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.54)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.54)}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.54)}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle[_ngcontent-%COMP%]:not(.mat-checked)   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-slide-toggle-thumb[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12);background-color:#fafafa}.mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-primary[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f57c00}.mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-accent[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#ff9800}.mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-warn[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f44336}.mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.2)}.mat-slider[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%], .cdk-focused[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-disabled[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]   .mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing)   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26);background-color:transparent}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26)}.mat-slider-has-ticks[_ngcontent-%COMP%]   .mat-slider-wrapper[_ngcontent-%COMP%]::after{border-color:rgba(0,0,0,.7)}.mat-slider-horizontal[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to right, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent);background-image:-moz-repeating-linear-gradient(0.0001deg, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-slider-vertical[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to bottom, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-step-header.cdk-keyboard-focused[_ngcontent-%COMP%], .mat-step-header.cdk-program-focused[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]:hover{background-color:rgba(0,0,0,.04)}@media(hover: none){.mat-step-header[_ngcontent-%COMP%]:hover{background:none}}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-optional[_ngcontent-%COMP%]{color:#616161}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon[_ngcontent-%COMP%]{background-color:#616161;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-selected[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-done[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-edit[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-error[_ngcontent-%COMP%]{background-color:transparent;color:#f44336}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-active[_ngcontent-%COMP%]{color:#212121}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-error[_ngcontent-%COMP%]{color:#f44336}.mat-stepper-horizontal[_ngcontent-%COMP%], .mat-stepper-vertical[_ngcontent-%COMP%]{background-color:#fff}.mat-stepper-vertical-line[_ngcontent-%COMP%]::before{border-left-color:rgba(0,0,0,.12)}.mat-horizontal-stepper-header[_ngcontent-%COMP%]::before, .mat-horizontal-stepper-header[_ngcontent-%COMP%]::after, .mat-stepper-horizontal-line[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-sort-header-arrow[_ngcontent-%COMP%]{color:#616161}.mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-header[_ngcontent-%COMP%]{border-bottom:1px solid rgba(0,0,0,.12)}.mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%]{border-top:1px solid rgba(0,0,0,.12);border-bottom:none}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{color:#212121}.mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:#757575}.mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#212121}.mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#757575}.mat-tab-group[class*=mat-background-][_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar[class*=mat-background-][_ngcontent-%COMP%]{border-bottom:none;border-top:none}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-toolbar[_ngcontent-%COMP%]{background:#f5f5f5;color:#212121}.mat-toolbar.mat-primary[_ngcontent-%COMP%]{background:#f57c00;color:#fff}.mat-toolbar.mat-accent[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-toolbar.mat-warn[_ngcontent-%COMP%]{background:#f44336;color:#fff}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:currentColor}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:inherit}.mat-toolbar[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:currentColor}.mat-tooltip[_ngcontent-%COMP%]{background:rgba(97,97,97,.9)}.mat-tree[_ngcontent-%COMP%]{background:#fff}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{color:#212121}.mat-snack-bar-container[_ngcontent-%COMP%]{color:rgba(255,255,255,.7);background:#323232;box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-simple-snackbar-action[_ngcontent-%COMP%]{color:#ff9800}[_nghost-%COMP%]{display:flex;flex-direction:column;height:100%}.filters-toolbar[_ngcontent-%COMP%]{align-items:center;border-bottom:1px solid #ebebeb;display:flex;height:42px;justify-content:space-between;padding:0 10px}.filters-title[_ngcontent-%COMP%]{display:inline;font-size:14px;font-weight:500}.side-toggle[_ngcontent-%COMP%]{align-items:center;background-color:#fff;border-radius:3px;border:1px solid #ebebeb;display:flex;height:30px;justify-content:center;width:30px}.filters[_ngcontent-%COMP%]{overflow-y:auto}.filters-hint[_ngcontent-%COMP%]{align-items:center;display:flex;height:42px;padding:0 16px}.filters-hint-text[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/violin_filters/violin_filters_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class ViolinFiltersContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.sidebarExpanded$ = this.store.select(getSidebarExpanded);
            this.metricFilters$ = this.store.select(getMetricFilters).pipe(map((/**
             * @param {?} filters
             * @return {?}
             */
            (filters) => {
                return Object.entries(filters);
            })));
        }
        /**
         * @return {?}
         */
        onToggleSidebarExpanded() {
            this.store.dispatch(npmiToggleSidebarExpanded());
        }
    }
    ViolinFiltersContainer.decorators = [
        { type: Component, args: [{
                    selector: 'npmi-violin-filters',
                    template: `
    <violin-filters-component
      [sidebarExpanded]="sidebarExpanded$ | async"
      [metricFilters]="metricFilters$ | async"
      (toggleSidebarExpanded)="onToggleSidebarExpanded()"
    ></violin-filters-component>
  `,
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    ViolinFiltersContainer.ctorParameters = () => [
        { type: Store }
    ];
    /** @nocollapse */ ViolinFiltersContainer.ɵfac = function ViolinFiltersContainer_Factory(t) { return new (t || ViolinFiltersContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ ViolinFiltersContainer.ɵcmp = ɵɵdefineComponent({ type: ViolinFiltersContainer, selectors: [["npmi-violin-filters"]], decls: 3, vars: 6, consts: [[3, "sidebarExpanded", "metricFilters", "toggleSidebarExpanded"]], template: function ViolinFiltersContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "violin-filters-component", 0);
            ɵɵlistener("toggleSidebarExpanded", function ViolinFiltersContainer_Template_violin_filters_component_toggleSidebarExpanded_0_listener() { return ctx.onToggleSidebarExpanded(); });
            ɵɵpipe(1, "async");
            ɵɵpipe(2, "async");
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵproperty("sidebarExpanded", ɵɵpipeBind1(1, 2, ctx.sidebarExpanded$))("metricFilters", ɵɵpipeBind1(2, 4, ctx.metricFilters$));
        } }, directives: [ViolinFiltersComponent], pipes: [AsyncPipe], encapsulation: 2, changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/runs_legacy/views/legacy_runs_selector/legacy_runs_selector_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    const _c0$x = ["selector"];
    class LegacyRunsSelectorComponent {
        constructor() {
            this.onSelectionChange = new EventEmitter();
        }
        /**
         * @return {?}
         */
        ngAfterViewInit() {
            /**
             * The event is dispatched by Polymer when `selectedRuns` prop changes because it
             * notifies (it is implicitly fired by Polymer library).
             */
            this.selector.nativeElement.addEventListener('selected-runs-changed', (/**
             * @param {?} event
             * @return {?}
             */
            (event) => {
                this.onSelectionChange.emit((/** @type {?} */ (event.detail.value)));
            }));
            setTimeout((/**
             * @return {?}
             */
            () => {
                // Dispatch the initial value from the component.
                this.onSelectionChange.emit(this.selector.nativeElement.selectedRuns);
            }));
        }
    }
    LegacyRunsSelectorComponent.decorators = [
        { type: Component, args: [{
                    selector: 'tb-legacy-runs-selector-component',
                    template: ` <tf-runs-selector #selector></tf-runs-selector> `,
                },] },
    ];
    LegacyRunsSelectorComponent.propDecorators = {
        onSelectionChange: [{ type: Output }],
        selector: [{ type: ViewChild, args: ['selector', { static: true },] }]
    };
    /** @nocollapse */ LegacyRunsSelectorComponent.ɵfac = function LegacyRunsSelectorComponent_Factory(t) { return new (t || LegacyRunsSelectorComponent)(); };
    /** @nocollapse */ LegacyRunsSelectorComponent.ɵcmp = ɵɵdefineComponent({ type: LegacyRunsSelectorComponent, selectors: [["tb-legacy-runs-selector-component"]], viewQuery: function LegacyRunsSelectorComponent_Query(rf, ctx) { if (rf & 1) {
            ɵɵstaticViewQuery(_c0$x, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.selector = _t.first);
        } }, outputs: { onSelectionChange: "onSelectionChange" }, decls: 2, vars: 0, consts: [["selector", ""]], template: function LegacyRunsSelectorComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelement(0, "tf-runs-selector", null, 0);
        } }, encapsulation: 2 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/runs_legacy/views/legacy_runs_selector/legacy_runs_selector_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class LegacyRunsSelectorContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
        }
        /**
         * @param {?} runs
         * @return {?}
         */
        onSelectionChange(runs) {
            this.store.dispatch(polymerInteropRunSelectionChanged({ nextSelection: runs }));
        }
    }
    LegacyRunsSelectorContainer.decorators = [
        { type: Component, args: [{
                    selector: 'tb-legacy-runs-selector',
                    template: `
    <tb-legacy-runs-selector-component
      (onSelectionChange)="onSelectionChange($event)"
    ></tb-legacy-runs-selector-component>
  `,
                },] },
    ];
    /** @nocollapse */
    LegacyRunsSelectorContainer.ctorParameters = () => [
        { type: Store }
    ];
    /** @nocollapse */ LegacyRunsSelectorContainer.ɵfac = function LegacyRunsSelectorContainer_Factory(t) { return new (t || LegacyRunsSelectorContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ LegacyRunsSelectorContainer.ɵcmp = ɵɵdefineComponent({ type: LegacyRunsSelectorContainer, selectors: [["tb-legacy-runs-selector"]], decls: 1, vars: 0, consts: [[3, "onSelectionChange"]], template: function LegacyRunsSelectorContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "tb-legacy-runs-selector-component", 0);
            ɵɵlistener("onSelectionChange", function LegacyRunsSelectorContainer_Template_tb_legacy_runs_selector_component_onSelectionChange_0_listener($event) { return ctx.onSelectionChange($event); });
            ɵɵelementEnd();
        } }, directives: [LegacyRunsSelectorComponent], encapsulation: 2 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/util/sort_annotations.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @param {?} annotationData
     * @param {?} sort
     * @return {?}
     */
    function sortAnnotations(annotationData, sort) {
        /** @type {?} */
        let result = Object.keys(annotationData);
        /** @type {?} */
        const strippedMetric = stripMetricString(sort.metric);
        if (sort.metric === '') {
            return result;
        }
        if (sort.order === SortOrder.DOWN) {
            /** @type {?} */
            const maxData = {};
            for (const annotation of result) {
                maxData[annotation] = Math.max(...annotationData[annotation]
                    .filter((/**
                 * @param {?} annotation
                 * @return {?}
                 */
                (annotation) => annotation.metric === strippedMetric))
                    .map((/**
                 * @param {?} filtered
                 * @return {?}
                 */
                (filtered) => filtered.nPMIValue === null ? -Infinity : filtered.nPMIValue)));
            }
            result = result.sort((/**
             * @param {?} a
             * @param {?} b
             * @return {?}
             */
            (a, b) => {
                return maxData[b] - maxData[a];
            }));
        }
        else if (sort.order === SortOrder.UP) {
            /** @type {?} */
            const minData = {};
            for (const annotation of result) {
                minData[annotation] = Math.min(...annotationData[annotation]
                    .filter((/**
                 * @param {?} annotation
                 * @return {?}
                 */
                (annotation) => annotation.metric === strippedMetric))
                    .map((/**
                 * @param {?} filtered
                 * @return {?}
                 */
                (filtered) => filtered.nPMIValue === null ? Infinity : filtered.nPMIValue)));
            }
            result = result.sort((/**
             * @param {?} a
             * @param {?} b
             * @return {?}
             */
            (a, b) => {
                return minData[a] - minData[b];
            }));
        }
        return result;
    }

    const _c0$y = ["thumbContainer"];
    const _c1$k = ["toggleBar"];
    const _c2$f = ["input"];
    const _c3$5 = function () { return { enterDuration: 150 }; };
    const _c4$3 = ["*"];
    /**
     * Injection token to be used to override the default options for `mat-slide-toggle`.
     * @type {?}
     */
    const MAT_SLIDE_TOGGLE_DEFAULT_OPTIONS = new InjectionToken('mat-slide-toggle-default-options', {
        providedIn: 'root',
        factory: (/**
         * @return {?}
         */
        () => ({ disableToggleValue: false }))
    });

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/slide-toggle/slide-toggle.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    // Increasing integer for generating unique ids for slide-toggle components.
    /** @type {?} */
    let nextUniqueId$8 = 0;
    /**
     * \@docs-private
     * @type {?}
     */
    const MAT_SLIDE_TOGGLE_VALUE_ACCESSOR = {
        provide: NG_VALUE_ACCESSOR,
        useExisting: forwardRef((/**
         * @return {?}
         */
        () => MatSlideToggle)),
        multi: true
    };
    /**
     * Change event object emitted by a MatSlideToggle.
     */
    class MatSlideToggleChange {
        /**
         * @param {?} source
         * @param {?} checked
         */
        constructor(source, checked) {
            this.source = source;
            this.checked = checked;
        }
    }
    // Boilerplate for applying mixins to MatSlideToggle.
    /**
     * \@docs-private
     */
    class MatSlideToggleBase {
        /**
         * @param {?} _elementRef
         */
        constructor(_elementRef) {
            this._elementRef = _elementRef;
        }
    }
    /** @type {?} */
    const _MatSlideToggleMixinBase = mixinTabIndex(mixinColor(mixinDisableRipple(mixinDisabled(MatSlideToggleBase)), 'accent'));
    /**
     * Represents a slidable "switch" toggle that can be moved between on and off.
     */
    class MatSlideToggle extends _MatSlideToggleMixinBase {
        /**
         * @param {?} elementRef
         * @param {?} _focusMonitor
         * @param {?} _changeDetectorRef
         * @param {?} tabIndex
         * @param {?} _ngZone
         * @param {?} defaults
         * @param {?=} _animationMode
         * @param {?=} _dir
         */
        constructor(elementRef, _focusMonitor, _changeDetectorRef, tabIndex, 
        /**
         * @deprecated `_ngZone` and `_dir` parameters to be removed.
         * @breaking-change 10.0.0
         */
        _ngZone, defaults, _animationMode, _dir) {
            super(elementRef);
            this._focusMonitor = _focusMonitor;
            this._changeDetectorRef = _changeDetectorRef;
            this.defaults = defaults;
            this._animationMode = _animationMode;
            this._onChange = (/**
             * @param {?} _
             * @return {?}
             */
            (_) => { });
            this._onTouched = (/**
             * @return {?}
             */
            () => { });
            this._uniqueId = `mat-slide-toggle-${++nextUniqueId$8}`;
            this._required = false;
            this._checked = false;
            /**
             * Name value will be applied to the input element if present.
             */
            this.name = null;
            /**
             * A unique id for the slide-toggle input. If none is supplied, it will be auto-generated.
             */
            this.id = this._uniqueId;
            /**
             * Whether the label should appear after or before the slide-toggle. Defaults to 'after'.
             */
            this.labelPosition = 'after';
            /**
             * Used to set the aria-label attribute on the underlying input element.
             */
            this.ariaLabel = null;
            /**
             * Used to set the aria-labelledby attribute on the underlying input element.
             */
            this.ariaLabelledby = null;
            /**
             * An event will be dispatched each time the slide-toggle changes its value.
             */
            this.change = new EventEmitter();
            /**
             * An event will be dispatched each time the slide-toggle input is toggled.
             * This event is always emitted when the user toggles the slide toggle, but this does not mean
             * the slide toggle's value has changed.
             */
            this.toggleChange = new EventEmitter();
            /**
             * An event will be dispatched each time the slide-toggle is dragged.
             * This event is always emitted when the user drags the slide toggle to make a change greater
             * than 50%. It does not mean the slide toggle's value is changed. The event is not emitted when
             * the user toggles the slide toggle to change its value.
             * @deprecated No longer being used. To be removed.
             * \@breaking-change 10.0.0
             */
            this.dragChange = new EventEmitter();
            this.tabIndex = parseInt(tabIndex) || 0;
        }
        /**
         * Whether the slide-toggle is required.
         * @return {?}
         */
        get required() { return this._required; }
        /**
         * @param {?} value
         * @return {?}
         */
        set required(value) { this._required = coerceBooleanProperty(value); }
        /**
         * Whether the slide-toggle element is checked or not.
         * @return {?}
         */
        get checked() { return this._checked; }
        /**
         * @param {?} value
         * @return {?}
         */
        set checked(value) {
            this._checked = coerceBooleanProperty(value);
            this._changeDetectorRef.markForCheck();
        }
        /**
         * Returns the unique id for the visual hidden input.
         * @return {?}
         */
        get inputId() { return `${this.id || this._uniqueId}-input`; }
        /**
         * @return {?}
         */
        ngAfterContentInit() {
            this._focusMonitor
                .monitor(this._elementRef, true)
                .subscribe((/**
             * @param {?} focusOrigin
             * @return {?}
             */
            focusOrigin => {
                // Only forward focus manually when it was received programmatically or through the
                // keyboard. We should not do this for mouse/touch focus for two reasons:
                // 1. It can prevent clicks from landing in Chrome (see #18269).
                // 2. They're already handled by the wrapping `label` element.
                if (focusOrigin === 'keyboard' || focusOrigin === 'program') {
                    this._inputElement.nativeElement.focus();
                }
                else if (!focusOrigin) {
                    // When a focused element becomes disabled, the browser *immediately* fires a blur event.
                    // Angular does not expect events to be raised during change detection, so any state
                    // change (such as a form control's 'ng-touched') will cause a changed-after-checked
                    // error. See https://github.com/angular/angular/issues/17793. To work around this,
                    // we defer telling the form control it has been touched until the next tick.
                    Promise.resolve().then((/**
                     * @return {?}
                     */
                    () => this._onTouched()));
                }
            }));
        }
        /**
         * @return {?}
         */
        ngOnDestroy() {
            this._focusMonitor.stopMonitoring(this._elementRef);
        }
        /**
         * Method being called whenever the underlying input emits a change event.
         * @param {?} event
         * @return {?}
         */
        _onChangeEvent(event) {
            // We always have to stop propagation on the change event.
            // Otherwise the change event, from the input element, will bubble up and
            // emit its event object to the component's `change` output.
            event.stopPropagation();
            this.toggleChange.emit();
            // When the slide toggle's config disables toggle change event by setting
            // `disableToggleValue: true`, the slide toggle's value does not change, and the
            // checked state of the underlying input needs to be changed back.
            if (this.defaults.disableToggleValue) {
                this._inputElement.nativeElement.checked = this.checked;
                return;
            }
            // Sync the value from the underlying input element with the component instance.
            this.checked = this._inputElement.nativeElement.checked;
            // Emit our custom change event only if the underlying input emitted one. This ensures that
            // there is no change event, when the checked state changes programmatically.
            this._emitChangeEvent();
        }
        /**
         * Method being called whenever the slide-toggle has been clicked.
         * @param {?} event
         * @return {?}
         */
        _onInputClick(event) {
            // We have to stop propagation for click events on the visual hidden input element.
            // By default, when a user clicks on a label element, a generated click event will be
            // dispatched on the associated input element. Since we are using a label element as our
            // root container, the click event on the `slide-toggle` will be executed twice.
            // The real click event will bubble up, and the generated click event also tries to bubble up.
            // This will lead to multiple click events.
            // Preventing bubbling for the second event will solve that issue.
            event.stopPropagation();
        }
        /**
         * Implemented as part of ControlValueAccessor.
         * @param {?} value
         * @return {?}
         */
        writeValue(value) {
            this.checked = !!value;
        }
        /**
         * Implemented as part of ControlValueAccessor.
         * @param {?} fn
         * @return {?}
         */
        registerOnChange(fn) {
            this._onChange = fn;
        }
        /**
         * Implemented as part of ControlValueAccessor.
         * @param {?} fn
         * @return {?}
         */
        registerOnTouched(fn) {
            this._onTouched = fn;
        }
        /**
         * Implemented as a part of ControlValueAccessor.
         * @param {?} isDisabled
         * @return {?}
         */
        setDisabledState(isDisabled) {
            this.disabled = isDisabled;
            this._changeDetectorRef.markForCheck();
        }
        /**
         * Focuses the slide-toggle.
         * @param {?=} options
         * @return {?}
         */
        focus(options) {
            this._focusMonitor.focusVia(this._inputElement, 'keyboard', options);
        }
        /**
         * Toggles the checked state of the slide-toggle.
         * @return {?}
         */
        toggle() {
            this.checked = !this.checked;
            this._onChange(this.checked);
        }
        /**
         * Emits a change event on the `change` output. Also notifies the FormControl about the change.
         * @private
         * @return {?}
         */
        _emitChangeEvent() {
            this._onChange(this.checked);
            this.change.emit(new MatSlideToggleChange(this, this.checked));
        }
        /**
         * Method being called whenever the label text changes.
         * @return {?}
         */
        _onLabelTextChange() {
            // Since the event of the `cdkObserveContent` directive runs outside of the zone, the
            // slide-toggle component will be only marked for check, but no actual change detection runs
            // automatically. Instead of going back into the zone in order to trigger a change detection
            // which causes *all* components to be checked (if explicitly marked or not using OnPush),
            // we only trigger an explicit change detection for the slide-toggle view and its children.
            this._changeDetectorRef.detectChanges();
        }
    }
    MatSlideToggle.ɵfac = function MatSlideToggle_Factory(t) { return new (t || MatSlideToggle)(ɵɵdirectiveInject(ElementRef), ɵɵdirectiveInject(FocusMonitor), ɵɵdirectiveInject(ChangeDetectorRef), ɵɵinjectAttribute('tabindex'), ɵɵdirectiveInject(NgZone), ɵɵdirectiveInject(MAT_SLIDE_TOGGLE_DEFAULT_OPTIONS), ɵɵdirectiveInject(ANIMATION_MODULE_TYPE, 8), ɵɵdirectiveInject(Directionality, 8)); };
    MatSlideToggle.ɵcmp = ɵɵdefineComponent({ type: MatSlideToggle, selectors: [["mat-slide-toggle"]], viewQuery: function MatSlideToggle_Query(rf, ctx) { if (rf & 1) {
            ɵɵviewQuery(_c0$y, true);
            ɵɵviewQuery(_c1$k, true);
            ɵɵviewQuery(_c2$f, true);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._thumbEl = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._thumbBarEl = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx._inputElement = _t.first);
        } }, hostAttrs: [1, "mat-slide-toggle"], hostVars: 12, hostBindings: function MatSlideToggle_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵhostProperty("id", ctx.id);
            ɵɵattribute("tabindex", ctx.disabled ? null : 0 - 1)("aria-label", null)("aria-labelledby", null);
            ɵɵclassProp("mat-checked", ctx.checked)("mat-disabled", ctx.disabled)("mat-slide-toggle-label-before", ctx.labelPosition == "before")("_mat-animation-noopable", ctx._animationMode === "NoopAnimations");
        } }, inputs: { disabled: "disabled", disableRipple: "disableRipple", color: "color", tabIndex: "tabIndex", name: "name", id: "id", labelPosition: "labelPosition", ariaLabel: ["aria-label", "ariaLabel"], ariaLabelledby: ["aria-labelledby", "ariaLabelledby"], required: "required", checked: "checked" }, outputs: { change: "change", toggleChange: "toggleChange", dragChange: "dragChange" }, exportAs: ["matSlideToggle"], features: [ɵɵProvidersFeature([MAT_SLIDE_TOGGLE_VALUE_ACCESSOR]), ɵɵInheritDefinitionFeature], ngContentSelectors: _c4$3, decls: 16, vars: 18, consts: [[1, "mat-slide-toggle-label"], ["label", ""], [1, "mat-slide-toggle-bar"], ["toggleBar", ""], ["type", "checkbox", "role", "switch", 1, "mat-slide-toggle-input", "cdk-visually-hidden", 3, "id", "required", "tabIndex", "checked", "disabled", "change", "click"], ["input", ""], [1, "mat-slide-toggle-thumb-container"], ["thumbContainer", ""], [1, "mat-slide-toggle-thumb"], ["mat-ripple", "", 1, "mat-slide-toggle-ripple", "mat-focus-indicator", 3, "matRippleTrigger", "matRippleDisabled", "matRippleCentered", "matRippleRadius", "matRippleAnimation"], [1, "mat-ripple-element", "mat-slide-toggle-persistent-ripple"], [1, "mat-slide-toggle-content", 3, "cdkObserveContent"], ["labelContent", ""], [2, "display", "none"]], template: function MatSlideToggle_Template(rf, ctx) { if (rf & 1) {
            ɵɵprojectionDef();
            ɵɵelementStart(0, "label", 0, 1);
            ɵɵelementStart(2, "div", 2, 3);
            ɵɵelementStart(4, "input", 4, 5);
            ɵɵlistener("change", function MatSlideToggle_Template_input_change_4_listener($event) { return ctx._onChangeEvent($event); })("click", function MatSlideToggle_Template_input_click_4_listener($event) { return ctx._onInputClick($event); });
            ɵɵelementEnd();
            ɵɵelementStart(6, "div", 6, 7);
            ɵɵelement(8, "div", 8);
            ɵɵelementStart(9, "div", 9);
            ɵɵelement(10, "div", 10);
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementStart(11, "span", 11, 12);
            ɵɵlistener("cdkObserveContent", function MatSlideToggle_Template_span_cdkObserveContent_11_listener() { return ctx._onLabelTextChange(); });
            ɵɵelementStart(13, "span", 13);
            ɵɵtext(14, "\u00A0");
            ɵɵelementEnd();
            ɵɵprojection(15);
            ɵɵelementEnd();
            ɵɵelementEnd();
        } if (rf & 2) {
            const _r0 = ɵɵreference(1);
            const _r4 = ɵɵreference(12);
            ɵɵattribute("for", ctx.inputId);
            ɵɵadvance(2);
            ɵɵclassProp("mat-slide-toggle-bar-no-side-margin", !_r4.textContent || !_r4.textContent.trim());
            ɵɵadvance(2);
            ɵɵproperty("id", ctx.inputId)("required", ctx.required)("tabIndex", ctx.tabIndex)("checked", ctx.checked)("disabled", ctx.disabled);
            ɵɵattribute("name", ctx.name)("aria-checked", ctx.checked.toString())("aria-label", ctx.ariaLabel)("aria-labelledby", ctx.ariaLabelledby);
            ɵɵadvance(5);
            ɵɵproperty("matRippleTrigger", _r0)("matRippleDisabled", ctx.disableRipple || ctx.disabled)("matRippleCentered", true)("matRippleRadius", 20)("matRippleAnimation", ɵɵpureFunction0(17, _c3$5));
        } }, directives: [MatRipple, CdkObserveContent], styles: [".mat-slide-toggle{display:inline-block;height:24px;max-width:100%;line-height:24px;white-space:nowrap;outline:none;-webkit-tap-highlight-color:transparent}.mat-slide-toggle.mat-checked .mat-slide-toggle-thumb-container{transform:translate3d(16px, 0, 0)}[dir=rtl] .mat-slide-toggle.mat-checked .mat-slide-toggle-thumb-container{transform:translate3d(-16px, 0, 0)}.mat-slide-toggle.mat-disabled{opacity:.38}.mat-slide-toggle.mat-disabled .mat-slide-toggle-label,.mat-slide-toggle.mat-disabled .mat-slide-toggle-thumb-container{cursor:default}.mat-slide-toggle-label{display:flex;flex:1;flex-direction:row;align-items:center;height:inherit;cursor:pointer}.mat-slide-toggle-content{white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.mat-slide-toggle-label-before .mat-slide-toggle-label{order:1}.mat-slide-toggle-label-before .mat-slide-toggle-bar{order:2}[dir=rtl] .mat-slide-toggle-label-before .mat-slide-toggle-bar,.mat-slide-toggle-bar{margin-right:8px;margin-left:0}[dir=rtl] .mat-slide-toggle-bar,.mat-slide-toggle-label-before .mat-slide-toggle-bar{margin-left:8px;margin-right:0}.mat-slide-toggle-bar-no-side-margin{margin-left:0;margin-right:0}.mat-slide-toggle-thumb-container{position:absolute;z-index:1;width:20px;height:20px;top:-3px;left:0;transform:translate3d(0, 0, 0);transition:all 80ms linear;transition-property:transform}._mat-animation-noopable .mat-slide-toggle-thumb-container{transition:none}[dir=rtl] .mat-slide-toggle-thumb-container{left:auto;right:0}.mat-slide-toggle-thumb{height:20px;width:20px;border-radius:50%}.mat-slide-toggle-bar{position:relative;width:36px;height:14px;flex-shrink:0;border-radius:8px}.mat-slide-toggle-input{bottom:0;left:10px}[dir=rtl] .mat-slide-toggle-input{left:auto;right:10px}.mat-slide-toggle-bar,.mat-slide-toggle-thumb{transition:all 80ms linear;transition-property:background-color;transition-delay:50ms}._mat-animation-noopable .mat-slide-toggle-bar,._mat-animation-noopable .mat-slide-toggle-thumb{transition:none}.mat-slide-toggle .mat-slide-toggle-ripple{position:absolute;top:calc(50% - 20px);left:calc(50% - 20px);height:40px;width:40px;z-index:1;pointer-events:none}.mat-slide-toggle .mat-slide-toggle-ripple .mat-ripple-element:not(.mat-slide-toggle-persistent-ripple){opacity:.12}.mat-slide-toggle-persistent-ripple{width:100%;height:100%;transform:none}.mat-slide-toggle-bar:hover .mat-slide-toggle-persistent-ripple{opacity:.04}.mat-slide-toggle:not(.mat-disabled).cdk-keyboard-focused .mat-slide-toggle-persistent-ripple{opacity:.12}.mat-slide-toggle-persistent-ripple,.mat-slide-toggle.mat-disabled .mat-slide-toggle-bar:hover .mat-slide-toggle-persistent-ripple{opacity:0}@media(hover: none){.mat-slide-toggle-bar:hover .mat-slide-toggle-persistent-ripple{display:none}}.cdk-high-contrast-active .mat-slide-toggle-thumb,.cdk-high-contrast-active .mat-slide-toggle-bar{border:1px solid}.cdk-high-contrast-active .mat-slide-toggle.cdk-keyboard-focused .mat-slide-toggle-bar{outline:2px dotted;outline-offset:5px}\n"], encapsulation: 2, changeDetection: 0 });
    /** @nocollapse */
    MatSlideToggle.ctorParameters = () => [
        { type: ElementRef },
        { type: FocusMonitor },
        { type: ChangeDetectorRef },
        { type: String, decorators: [{ type: Attribute, args: ['tabindex',] }] },
        { type: NgZone },
        { type: undefined, decorators: [{ type: Inject, args: [MAT_SLIDE_TOGGLE_DEFAULT_OPTIONS,] }] },
        { type: String, decorators: [{ type: Optional }, { type: Inject, args: [ANIMATION_MODULE_TYPE,] }] },
        { type: Directionality, decorators: [{ type: Optional }] }
    ];
    MatSlideToggle.propDecorators = {
        _thumbEl: [{ type: ViewChild, args: ['thumbContainer',] }],
        _thumbBarEl: [{ type: ViewChild, args: ['toggleBar',] }],
        name: [{ type: Input }],
        id: [{ type: Input }],
        labelPosition: [{ type: Input }],
        ariaLabel: [{ type: Input, args: ['aria-label',] }],
        ariaLabelledby: [{ type: Input, args: ['aria-labelledby',] }],
        required: [{ type: Input }],
        checked: [{ type: Input }],
        change: [{ type: Output }],
        toggleChange: [{ type: Output }],
        dragChange: [{ type: Output }],
        _inputElement: [{ type: ViewChild, args: ['input',] }]
    };

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/slide-toggle/slide-toggle-required-validator.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const MAT_SLIDE_TOGGLE_REQUIRED_VALIDATOR = {
        provide: NG_VALIDATORS,
        useExisting: forwardRef((/**
         * @return {?}
         */
        () => MatSlideToggleRequiredValidator)),
        multi: true
    };
    /**
     * Validator for Material slide-toggle components with the required attribute in a
     * template-driven form. The default validator for required form controls asserts
     * that the control value is not undefined but that is not appropriate for a slide-toggle
     * where the value is always defined.
     *
     * Required slide-toggle form controls are valid when checked.
     */
    class MatSlideToggleRequiredValidator extends CheckboxRequiredValidator {
    }
    MatSlideToggleRequiredValidator.ɵfac = function MatSlideToggleRequiredValidator_Factory(t) { return ɵMatSlideToggleRequiredValidator_BaseFactory(t || MatSlideToggleRequiredValidator); };
    MatSlideToggleRequiredValidator.ɵdir = ɵɵdefineDirective({ type: MatSlideToggleRequiredValidator, selectors: [["mat-slide-toggle", "required", "", "formControlName", ""], ["mat-slide-toggle", "required", "", "formControl", ""], ["mat-slide-toggle", "required", "", "ngModel", ""]], features: [ɵɵProvidersFeature([MAT_SLIDE_TOGGLE_REQUIRED_VALIDATOR]), ɵɵInheritDefinitionFeature] });
    const ɵMatSlideToggleRequiredValidator_BaseFactory = ɵɵgetInheritedFactory(MatSlideToggleRequiredValidator);

    /**
     * @fileoverview added by tsickle
     * Generated from: src/material/slide-toggle/slide-toggle-module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * This module is used by both original and MDC-based slide-toggle implementations.
     */
    // tslint:disable-next-line:class-name
    class _MatSlideToggleRequiredValidatorModule {
    }
    _MatSlideToggleRequiredValidatorModule.ɵmod = ɵɵdefineNgModule({ type: _MatSlideToggleRequiredValidatorModule });
    _MatSlideToggleRequiredValidatorModule.ɵinj = ɵɵdefineInjector({ factory: function _MatSlideToggleRequiredValidatorModule_Factory(t) { return new (t || _MatSlideToggleRequiredValidatorModule)(); } });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(_MatSlideToggleRequiredValidatorModule, { declarations: [MatSlideToggleRequiredValidator], exports: [MatSlideToggleRequiredValidator] }); })();
    class MatSlideToggleModule {
    }
    MatSlideToggleModule.ɵmod = ɵɵdefineNgModule({ type: MatSlideToggleModule });
    MatSlideToggleModule.ɵinj = ɵɵdefineInjector({ factory: function MatSlideToggleModule_Factory(t) { return new (t || MatSlideToggleModule)(); }, imports: [[
                _MatSlideToggleRequiredValidatorModule,
                MatRippleModule,
                MatCommonModule,
                ObserversModule,
            ],
            _MatSlideToggleRequiredValidatorModule,
            MatCommonModule] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(MatSlideToggleModule, { declarations: function () { return [MatSlideToggle]; }, imports: function () { return [_MatSlideToggleRequiredValidatorModule,
            MatRippleModule,
            MatCommonModule,
            ObserversModule]; }, exports: function () { return [_MatSlideToggleRequiredValidatorModule,
            MatSlideToggle,
            MatCommonModule]; } }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/annotations_list/annotations_list_toolbar/annotations_search/annotations_search_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    function AnnotationsSearchComponent_mat_icon_2_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "mat-icon", 3);
    } }
    class AnnotationsSearchComponent {
        constructor() {
            this.onRegexFilterValueChange = new EventEmitter();
        }
    }
    AnnotationsSearchComponent.decorators = [
        { type: Component, args: [{
                    selector: 'npmi-annotations-search-component',
                    templateUrl: './annotations_search_component.ng.html',
                    styleUrls: ['./annotations_search_component.css'],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    AnnotationsSearchComponent.propDecorators = {
        regexFilterValue: [{ type: Input }],
        onRegexFilterValueChange: [{ type: Output }],
        isRegexFilterValid: [{ type: HostBinding, args: ['class.valid',] }, { type: Input }]
    };
    /** @nocollapse */ AnnotationsSearchComponent.ɵfac = function AnnotationsSearchComponent_Factory(t) { return new (t || AnnotationsSearchComponent)(); };
    /** @nocollapse */ AnnotationsSearchComponent.ɵcmp = ɵɵdefineComponent({ type: AnnotationsSearchComponent, selectors: [["npmi-annotations-search-component"]], hostVars: 2, hostBindings: function AnnotationsSearchComponent_HostBindings(rf, ctx) { if (rf & 2) {
            ɵɵclassProp("valid", ctx.isRegexFilterValid);
        } }, inputs: { regexFilterValue: "regexFilterValue", isRegexFilterValid: "isRegexFilterValid" }, outputs: { onRegexFilterValueChange: "onRegexFilterValueChange" }, decls: 3, vars: 2, consts: [["svgIcon", "search_24px"], ["autocomplete", "off", "placeholder", "Filter Annotations", 3, "value", "input"], ["svgIcon", "error_24px", "class", "error-icon", "matTooltip", "Invalid regex filter. The result may be stale.", 4, "ngIf"], ["svgIcon", "error_24px", "matTooltip", "Invalid regex filter. The result may be stale.", 1, "error-icon"]], template: function AnnotationsSearchComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelement(0, "mat-icon", 0);
            ɵɵelementStart(1, "input", 1);
            ɵɵlistener("input", function AnnotationsSearchComponent_Template_input_input_1_listener($event) { return ctx.onRegexFilterValueChange.emit($event.target.value); });
            ɵɵelementEnd();
            ɵɵtemplate(2, AnnotationsSearchComponent_mat_icon_2_Template, 1, 0, "mat-icon", 2);
        } if (rf & 2) {
            ɵɵadvance(1);
            ɵɵproperty("value", ctx.regexFilterValue);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", !ctx.isRegexFilterValid);
        } }, directives: [MatIcon, NgIf], styles: [".mat-badge-content[_ngcontent-%COMP%]{font-weight:600;font-size:12px;font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:9px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:24px}.mat-h1[_ngcontent-%COMP%], .mat-headline[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%]{font:400 24px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h2[_ngcontent-%COMP%], .mat-title[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h3[_ngcontent-%COMP%], .mat-subheading-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%]{font:400 16px/28px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h4[_ngcontent-%COMP%], .mat-subheading-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%]{font:400 15px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h5[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%]{font:400 calc(14px * 0.83)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-h6[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:400 calc(14px * 0.67)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-body-strong[_ngcontent-%COMP%], .mat-body-2[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   p[_ngcontent-%COMP%]{margin:0 0 12px}.mat-small[_ngcontent-%COMP%], .mat-caption[_ngcontent-%COMP%]{font:400 12px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-display-4[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-4[_ngcontent-%COMP%]{font:300 112px/112px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.05em;margin:0 0 56px}.mat-display-3[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-3[_ngcontent-%COMP%]{font:400 56px/56px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.02em;margin:0 0 64px}.mat-display-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-2[_ngcontent-%COMP%]{font:400 45px/48px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.005em;margin:0 0 64px}.mat-display-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-1[_ngcontent-%COMP%]{font:400 34px/40px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 64px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%], .mat-flat-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-button-toggle[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card-title[_ngcontent-%COMP%]{font-size:24px;font-weight:500}.mat-card-header[_ngcontent-%COMP%]   .mat-card-title[_ngcontent-%COMP%]{font-size:20px}.mat-card-subtitle[_ngcontent-%COMP%], .mat-card-content[_ngcontent-%COMP%]{font-size:14px}.mat-checkbox[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-checkbox-layout[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{line-height:24px}.mat-chip[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-chip[_ngcontent-%COMP%]   .mat-chip-trailing-icon.mat-icon[_ngcontent-%COMP%], .mat-chip[_ngcontent-%COMP%]   .mat-chip-remove.mat-icon[_ngcontent-%COMP%]{font-size:18px}.mat-table[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-header-cell[_ngcontent-%COMP%]{font-size:12px;font-weight:500}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{font-size:14px}.mat-calendar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-calendar-body[_ngcontent-%COMP%]{font-size:13px}.mat-calendar-body-label[_ngcontent-%COMP%], .mat-calendar-period-button[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-calendar-table-header[_ngcontent-%COMP%]   th[_ngcontent-%COMP%]{font-size:11px;font-weight:400}.mat-dialog-title[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-expansion-panel-header[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:15px;font-weight:400}.mat-expansion-panel-content[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field[_ngcontent-%COMP%]{font-size:inherit;font-weight:400;line-height:1.125;font-family:Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.34375em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{font-size:150%;line-height:1.125}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]{height:1.5em;width:1.5em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{height:1.125em;line-height:1.125}.mat-form-field-infix[_ngcontent-%COMP%]{padding:.5em 0;border-top:.84375em solid transparent}.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34375em) scale(0.75);width:133.3333333333%}.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34374em) scale(0.75);width:133.3333433333%}.mat-form-field-label-wrapper[_ngcontent-%COMP%]{top:-0.84375em;padding-top:.84375em}.mat-form-field-label[_ngcontent-%COMP%]{top:1.34375em}.mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.34375em}.mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{font-size:75%;margin-top:.6666666667em;top:calc(100% - 1.7916666667em)}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.4375em 0}.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.001px);-ms-transform:translateY(-1.28125em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00101px);-ms-transform:translateY(-1.28124em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00102px);-ms-transform:translateY(-1.28123em) scale(0.75);width:133.3333533333%}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.28125em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{margin-top:.5416666667em;top:calc(100% - 1.6666666667em)}@media print{.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28122em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28121em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.2812em) scale(0.75)}}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.25em 0 .75em 0}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.09375em;margin-top:-0.5em}.mat-form-field-appearance-fill.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59374em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:1em 0 1em 0}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.84375em;margin-top:-0.25em}.mat-form-field-appearance-outline.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59374em) scale(0.75);width:133.3333433333%}.mat-grid-tile-header[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]{font-size:14px}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2), .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}input.mat-input-element[_ngcontent-%COMP%]{margin-top:-0.0625em}.mat-menu-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:400}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px}.mat-radio-button[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select-trigger[_ngcontent-%COMP%]{height:1.125em}.mat-slide-toggle-content[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-slider-thumb-label-text[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-stepper-vertical[_ngcontent-%COMP%], .mat-stepper-horizontal[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-step-label[_ngcontent-%COMP%]{font-size:14px;font-weight:400}.mat-step-sub-label-error[_ngcontent-%COMP%]{font-weight:normal}.mat-step-label-error[_ngcontent-%COMP%]{font-size:14px}.mat-step-label-selected[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-tab-group[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-toolbar[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0}.mat-tooltip[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:10px;padding-top:6px;padding-bottom:6px}.mat-tooltip-handset[_ngcontent-%COMP%]{font-size:14px;padding-top:8px;padding-bottom:8px}.mat-list-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:16px}.mat-optgroup-label[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-simple-snackbar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px}.mat-simple-snackbar-action[_ngcontent-%COMP%]{line-height:1;font-family:inherit;font-size:inherit;font-weight:500}.mat-tree[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{font-weight:400;font-size:14px}.mat-ripple[_ngcontent-%COMP%]{overflow:hidden;position:relative}.mat-ripple[_ngcontent-%COMP%]:not(:empty){transform:translateZ(0)}.mat-ripple.mat-ripple-unbounded[_ngcontent-%COMP%]{overflow:visible}.mat-ripple-element[_ngcontent-%COMP%]{position:absolute;border-radius:50%;pointer-events:none;transition:opacity,transform 0ms cubic-bezier(0, 0, 0.2, 1);transform:scale(0)}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{display:none}.cdk-visually-hidden[_ngcontent-%COMP%]{border:0;clip:rect(0 0 0 0);height:1px;margin:-1px;overflow:hidden;padding:0;position:absolute;width:1px;outline:0;-webkit-appearance:none;-moz-appearance:none}.cdk-overlay-container[_ngcontent-%COMP%], .cdk-global-overlay-wrapper[_ngcontent-%COMP%]{pointer-events:none;top:0;left:0;height:100%;width:100%}.cdk-overlay-container[_ngcontent-%COMP%]{position:fixed;z-index:1000}.cdk-overlay-container[_ngcontent-%COMP%]:empty{display:none}.cdk-global-overlay-wrapper[_ngcontent-%COMP%]{display:flex;position:absolute;z-index:1000}.cdk-overlay-pane[_ngcontent-%COMP%]{position:absolute;pointer-events:auto;box-sizing:border-box;z-index:1000;display:flex;max-width:100%;max-height:100%}.cdk-overlay-backdrop[_ngcontent-%COMP%]{position:absolute;top:0;bottom:0;left:0;right:0;z-index:1000;pointer-events:auto;-webkit-tap-highlight-color:transparent;transition:opacity 400ms cubic-bezier(0.25, 0.8, 0.25, 1);opacity:0}.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:1}@media screen and (-ms-high-contrast: active){.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:.6}}.cdk-overlay-dark-backdrop[_ngcontent-%COMP%]{background:rgba(0,0,0,.32)}.cdk-overlay-transparent-backdrop[_ngcontent-%COMP%], .cdk-overlay-transparent-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:0}.cdk-overlay-connected-position-bounding-box[_ngcontent-%COMP%]{position:absolute;z-index:1000;display:flex;flex-direction:column;min-width:1px;min-height:1px}.cdk-global-scrollblock[_ngcontent-%COMP%]{position:fixed;width:100%;overflow-y:scroll}@keyframes cdk-text-field-autofill-start{}@keyframes cdk-text-field-autofill-end{}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:-webkit-autofill{animation:cdk-text-field-autofill-start 0s 1ms}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:not(:-webkit-autofill){animation:cdk-text-field-autofill-end 0s 1ms}textarea.cdk-textarea-autosize[_ngcontent-%COMP%]{resize:none}textarea.cdk-textarea-autosize-measuring[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:auto !important;overflow:hidden !important}textarea.cdk-textarea-autosize-measuring-firefox[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:0 !important}.mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.1)}.mat-option[_ngcontent-%COMP%]{color:#212121}.mat-option[_ngcontent-%COMP%]:hover:not(.mat-option-disabled), .mat-option[_ngcontent-%COMP%]:focus:not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple):not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-active[_ngcontent-%COMP%]{background:rgba(0,0,0,.04);color:#212121}.mat-option.mat-option-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-primary[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f57c00}.mat-accent[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f44336}.mat-optgroup-label[_ngcontent-%COMP%]{color:#616161}.mat-optgroup-disabled[_ngcontent-%COMP%]   .mat-optgroup-label[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-pseudo-checkbox[_ngcontent-%COMP%]{color:#616161}.mat-pseudo-checkbox[_ngcontent-%COMP%]::after{color:#fafafa}.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{color:#b0b0b0}.mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f57c00}.mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f44336}.mat-pseudo-checkbox-checked.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{background:#b0b0b0}.mat-elevation-z0[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z1[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z2[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z3[_ngcontent-%COMP%]{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z4[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z5[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 5px 8px 0px rgba(0, 0, 0, 0.14),0px 1px 14px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z6[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z7[_ngcontent-%COMP%]{box-shadow:0px 4px 5px -2px rgba(0, 0, 0, 0.2),0px 7px 10px 1px rgba(0, 0, 0, 0.14),0px 2px 16px 1px rgba(0, 0, 0, 0.12)}.mat-elevation-z8[_ngcontent-%COMP%]{box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z9[_ngcontent-%COMP%]{box-shadow:0px 5px 6px -3px rgba(0, 0, 0, 0.2),0px 9px 12px 1px rgba(0, 0, 0, 0.14),0px 3px 16px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z10[_ngcontent-%COMP%]{box-shadow:0px 6px 6px -3px rgba(0, 0, 0, 0.2),0px 10px 14px 1px rgba(0, 0, 0, 0.14),0px 4px 18px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z11[_ngcontent-%COMP%]{box-shadow:0px 6px 7px -4px rgba(0, 0, 0, 0.2),0px 11px 15px 1px rgba(0, 0, 0, 0.14),0px 4px 20px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z12[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z13[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 13px 19px 2px rgba(0, 0, 0, 0.14),0px 5px 24px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z14[_ngcontent-%COMP%]{box-shadow:0px 7px 9px -4px rgba(0, 0, 0, 0.2),0px 14px 21px 2px rgba(0, 0, 0, 0.14),0px 5px 26px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z15[_ngcontent-%COMP%]{box-shadow:0px 8px 9px -5px rgba(0, 0, 0, 0.2),0px 15px 22px 2px rgba(0, 0, 0, 0.14),0px 6px 28px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z16[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z17[_ngcontent-%COMP%]{box-shadow:0px 8px 11px -5px rgba(0, 0, 0, 0.2),0px 17px 26px 2px rgba(0, 0, 0, 0.14),0px 6px 32px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z18[_ngcontent-%COMP%]{box-shadow:0px 9px 11px -5px rgba(0, 0, 0, 0.2),0px 18px 28px 2px rgba(0, 0, 0, 0.14),0px 7px 34px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z19[_ngcontent-%COMP%]{box-shadow:0px 9px 12px -6px rgba(0, 0, 0, 0.2),0px 19px 29px 2px rgba(0, 0, 0, 0.14),0px 7px 36px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z20[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 20px 31px 3px rgba(0, 0, 0, 0.14),0px 8px 38px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z21[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 21px 33px 3px rgba(0, 0, 0, 0.14),0px 8px 40px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z22[_ngcontent-%COMP%]{box-shadow:0px 10px 14px -6px rgba(0, 0, 0, 0.2),0px 22px 35px 3px rgba(0, 0, 0, 0.14),0px 8px 42px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z23[_ngcontent-%COMP%]{box-shadow:0px 11px 14px -7px rgba(0, 0, 0, 0.2),0px 23px 36px 3px rgba(0, 0, 0, 0.14),0px 9px 44px 8px rgba(0, 0, 0, 0.12)}.mat-elevation-z24[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12)}.mat-app-background[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-theme-loaded-marker[_ngcontent-%COMP%]{display:none}.mat-autocomplete-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-autocomplete-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover){background:#fff}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover):not(.mat-option-disabled){color:#212121}.mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f57c00}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{outline:solid 1px;border-radius:0}.mat-badge-accent[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-badge-warn[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f44336}.mat-badge[_ngcontent-%COMP%]{position:relative}.mat-badge-hidden[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{display:none}.mat-badge-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#b9b9b9;color:#757575}.mat-badge-content[_ngcontent-%COMP%]{position:absolute;text-align:center;display:inline-block;border-radius:50%;transition:transform 200ms ease-in-out;transform:scale(0.6);overflow:hidden;white-space:nowrap;text-overflow:ellipsis;pointer-events:none}.ng-animate-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%], .mat-badge-content._mat-animation-noopable[_ngcontent-%COMP%]{transition:none}.mat-badge-content.mat-badge-active[_ngcontent-%COMP%]{transform:none}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:16px;height:16px;line-height:16px}.mat-badge-small.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-8px}.mat-badge-small.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-8px}.mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-16px}.mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-16px}.mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-8px}.mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-8px}.mat-badge-medium[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:22px;height:22px;line-height:22px}.mat-badge-medium.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-11px}.mat-badge-medium.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-11px}.mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-22px}.mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-22px}.mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-11px}.mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-11px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:28px;height:28px;line-height:28px}.mat-badge-large.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-14px}.mat-badge-large.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-14px}.mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-28px}.mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-28px}.mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-14px}.mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-14px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]{color:inherit;background:transparent}.mat-button.mat-primary[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-button[disabled][disabled][_ngcontent-%COMP%], .mat-icon-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-icon-button[disabled][disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-stroked-button[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f44336}.mat-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:transparent}.mat-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{opacity:.1;background-color:currentColor}.mat-button-focus-overlay[_ngcontent-%COMP%]{background:#000}.mat-stroked-button[_ngcontent-%COMP%]:not([disabled]){border-color:rgba(0,0,0,.12)}.mat-flat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{color:#212121;background-color:#fff}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-flat-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-stroked-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-flat-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-raised-button[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone[_ngcontent-%COMP%], .mat-button-toggle-group[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{box-shadow:none}.mat-button-toggle[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-button-toggle[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121;background:#fff}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:#000}.mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12)}[dir=rtl][_ngcontent-%COMP%]   .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-group-appearance-standard.mat-button-toggle-vertical[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:none;border-top:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#616161}.mat-button-toggle-checked.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121}.mat-button-toggle-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.26);background-color:#eee}.mat-button-toggle-disabled.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{background:#fff}.mat-button-toggle-disabled.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#bdbdbd}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{border:solid 1px rgba(0,0,0,.12)}.mat-card[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-card[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-card.mat-card-flat[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-card-subtitle[_ngcontent-%COMP%]{color:#616161}.mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#616161}.mat-checkbox-checkmark[_ngcontent-%COMP%]{fill:#fafafa}.mat-checkbox-checkmark-path[_ngcontent-%COMP%]{stroke:#fafafa !important}.mat-checkbox-mixedmark[_ngcontent-%COMP%]{background-color:#fafafa}.mat-checkbox-indeterminate.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f57c00}.mat-checkbox-indeterminate.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#ff9800}.mat-checkbox-indeterminate.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f44336}.mat-checkbox-disabled.mat-checkbox-checked[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-disabled.mat-checkbox-indeterminate[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]:not(.mat-checkbox-checked)   .mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{color:#616161}.mat-checkbox[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%]{background:#f57c00}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%]{background:#ff9800}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%]{background:#f44336}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#212121}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#212121;opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled):active{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled)   .mat-chip-remove[_ngcontent-%COMP%]:hover{opacity:.54}.mat-chip.mat-standard-chip.mat-chip-disabled[_ngcontent-%COMP%]{opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]::after{background:#000}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-table[_ngcontent-%COMP%]{background:#fff}.mat-table[_ngcontent-%COMP%]   thead[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tbody[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tfoot[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], [mat-header-row][_ngcontent-%COMP%], [mat-row][_ngcontent-%COMP%], [mat-footer-row][_ngcontent-%COMP%], .mat-table-sticky[_ngcontent-%COMP%]{background:inherit}mat-row[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], th.mat-header-cell[_ngcontent-%COMP%], td.mat-cell[_ngcontent-%COMP%], td.mat-footer-cell[_ngcontent-%COMP%]{border-bottom-color:rgba(0,0,0,.12)}.mat-header-cell[_ngcontent-%COMP%]{color:#616161}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{color:#212121}.mat-calendar-arrow[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.54)}.mat-datepicker-toggle[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-next-button[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-previous-button[_ngcontent-%COMP%]{color:rgba(0,0,0,.54)}.mat-calendar-table-header[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-calendar-table-header-divider[_ngcontent-%COMP%]::after{background:rgba(0,0,0,.12)}.mat-calendar-body-label[_ngcontent-%COMP%]{color:#616161}.mat-calendar-body-cell-content[_ngcontent-%COMP%]{color:#212121;border-color:transparent}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){color:#757575}.mat-calendar-body-cell[_ngcontent-%COMP%]:not(.mat-calendar-body-disabled):hover > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-keyboard-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-program-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){background-color:rgba(0,0,0,.04)}.mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.38)}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.18)}.mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.4)}.mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12);background-color:#fff;color:#212121}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.4)}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.4)}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content-touch[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-datepicker-toggle-active[_ngcontent-%COMP%]{color:#f57c00}.mat-datepicker-toggle-active.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-datepicker-toggle-active.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-dialog-container[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-divider[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-divider-vertical[_ngcontent-%COMP%]{border-right-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-expansion-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-action-row[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-keyboard-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-program-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded)   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover:not([aria-disabled=true]){background:rgba(0,0,0,.04)}@media(hover: none){.mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded):not([aria-disabled=true])   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover{background:#fff}}.mat-expansion-panel-header-title[_ngcontent-%COMP%]{color:#212121}.mat-expansion-panel-header-description[_ngcontent-%COMP%], .mat-expansion-indicator[_ngcontent-%COMP%]::after{color:#616161}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-title[_ngcontent-%COMP%], .mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-description[_ngcontent-%COMP%]{color:inherit}.mat-form-field-label[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-hint[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-focused[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-ripple[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid)   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f57c00}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-accent   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#ff9800}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-warn   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#f44336}.mat-error[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-hint[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-legacy.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-standard[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-standard.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.04)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.02)}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:transparent}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.12)}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:rgba(0,0,0,.87)}.mat-form-field-appearance-outline.mat-focused[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field-appearance-outline.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-appearance-outline.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-invalid.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.06)}.mat-icon.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-icon.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-icon.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-type-mat-native-select[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#616161}.mat-input-element[_ngcontent-%COMP%]:disabled, .mat-form-field-type-mat-native-select.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#757575}.mat-input-element[_ngcontent-%COMP%]{caret-color:#f57c00}.mat-input-element[_ngcontent-%COMP%]::placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-moz-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-webkit-input-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]:-ms-input-placeholder{color:rgba(97,97,97,.42)}.mat-accent[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%], .mat-form-field-invalid[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#f44336}.mat-form-field-type-mat-native-select.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{color:#616161}.mat-list-item-disabled[_ngcontent-%COMP%]{background-color:#eee}.mat-list-option[_ngcontent-%COMP%]:hover, .mat-list-option[_ngcontent-%COMP%]:focus, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.04)}.mat-list-single-selected-option[_ngcontent-%COMP%], .mat-list-single-selected-option[_ngcontent-%COMP%]:hover, .mat-list-single-selected-option[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.12)}.mat-menu-panel[_ngcontent-%COMP%]{background:#fff}.mat-menu-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-menu-item[_ngcontent-%COMP%]{background:transparent;color:#212121}.mat-menu-item[disabled][_ngcontent-%COMP%], .mat-menu-item[disabled][_ngcontent-%COMP%]::after{color:rgba(0,0,0,.38)}.mat-menu-item[_ngcontent-%COMP%]   .mat-icon-no-color[_ngcontent-%COMP%], .mat-menu-item-submenu-trigger[_ngcontent-%COMP%]::after{color:rgba(0,0,0,.54)}.mat-menu-item[_ngcontent-%COMP%]:hover:not([disabled]), .mat-menu-item.cdk-program-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item.cdk-keyboard-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item-highlighted[_ngcontent-%COMP%]:not([disabled]){background:rgba(0,0,0,.04)}.mat-paginator[_ngcontent-%COMP%]{background:#fff}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{color:#616161}.mat-paginator-decrement[_ngcontent-%COMP%], .mat-paginator-increment[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54);border-right:2px solid rgba(0,0,0,.54)}.mat-paginator-first[_ngcontent-%COMP%], .mat-paginator-last[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54)}.mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-decrement[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-increment[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-first[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-last[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f57c00}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#ff9800}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f44336}.mat-progress-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f57c00}.mat-progress-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#ff9800}.mat-progress-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f44336}.mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#616161}.mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f57c00}.mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#ff9800}.mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#ff9800}.mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f44336}.mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f44336}.mat-radio-button.mat-radio-disabled.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-label-content[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-radio-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-select-value[_ngcontent-%COMP%]{color:#212121}.mat-select-placeholder[_ngcontent-%COMP%]{color:rgba(97,97,97,.42)}.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%]{color:#757575}.mat-select-arrow[_ngcontent-%COMP%]{color:#616161}.mat-select-panel[_ngcontent-%COMP%]{background:#fff}.mat-select-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-select-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple){background:rgba(0,0,0,.12)}.mat-form-field.mat-focused.mat-primary[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-invalid[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#757575}.mat-drawer-container[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-drawer[_ngcontent-%COMP%]{background-color:#fff;color:#212121}.mat-drawer.mat-drawer-push[_ngcontent-%COMP%]{background-color:#fff}.mat-drawer[_ngcontent-%COMP%]:not(.mat-drawer-side){box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-drawer-side[_ngcontent-%COMP%]{border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-backdrop.mat-drawer-shown[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.6)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.54)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.54)}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.54)}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle[_ngcontent-%COMP%]:not(.mat-checked)   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-slide-toggle-thumb[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12);background-color:#fafafa}.mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-primary[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f57c00}.mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-accent[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#ff9800}.mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-warn[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f44336}.mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.2)}.mat-slider[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%], .cdk-focused[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-disabled[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]   .mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing)   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26);background-color:transparent}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26)}.mat-slider-has-ticks[_ngcontent-%COMP%]   .mat-slider-wrapper[_ngcontent-%COMP%]::after{border-color:rgba(0,0,0,.7)}.mat-slider-horizontal[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to right, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent);background-image:-moz-repeating-linear-gradient(0.0001deg, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-slider-vertical[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to bottom, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-step-header.cdk-keyboard-focused[_ngcontent-%COMP%], .mat-step-header.cdk-program-focused[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]:hover{background-color:rgba(0,0,0,.04)}@media(hover: none){.mat-step-header[_ngcontent-%COMP%]:hover{background:none}}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-optional[_ngcontent-%COMP%]{color:#616161}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon[_ngcontent-%COMP%]{background-color:#616161;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-selected[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-done[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-edit[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-error[_ngcontent-%COMP%]{background-color:transparent;color:#f44336}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-active[_ngcontent-%COMP%]{color:#212121}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-error[_ngcontent-%COMP%]{color:#f44336}.mat-stepper-horizontal[_ngcontent-%COMP%], .mat-stepper-vertical[_ngcontent-%COMP%]{background-color:#fff}.mat-stepper-vertical-line[_ngcontent-%COMP%]::before{border-left-color:rgba(0,0,0,.12)}.mat-horizontal-stepper-header[_ngcontent-%COMP%]::before, .mat-horizontal-stepper-header[_ngcontent-%COMP%]::after, .mat-stepper-horizontal-line[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-sort-header-arrow[_ngcontent-%COMP%]{color:#616161}.mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-header[_ngcontent-%COMP%]{border-bottom:1px solid rgba(0,0,0,.12)}.mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%]{border-top:1px solid rgba(0,0,0,.12);border-bottom:none}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{color:#212121}.mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:#757575}.mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#212121}.mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#757575}.mat-tab-group[class*=mat-background-][_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar[class*=mat-background-][_ngcontent-%COMP%]{border-bottom:none;border-top:none}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-toolbar[_ngcontent-%COMP%]{background:#f5f5f5;color:#212121}.mat-toolbar.mat-primary[_ngcontent-%COMP%]{background:#f57c00;color:#fff}.mat-toolbar.mat-accent[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-toolbar.mat-warn[_ngcontent-%COMP%]{background:#f44336;color:#fff}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:currentColor}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:inherit}.mat-toolbar[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:currentColor}.mat-tooltip[_ngcontent-%COMP%]{background:rgba(97,97,97,.9)}.mat-tree[_ngcontent-%COMP%]{background:#fff}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{color:#212121}.mat-snack-bar-container[_ngcontent-%COMP%]{color:rgba(255,255,255,.7);background:#323232;box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-simple-snackbar-action[_ngcontent-%COMP%]{color:#ff9800}[_nghost-%COMP%]{display:flex;position:relative}[_nghost-%COMP%]:not(.valid){color:#c62828}[_nghost-%COMP%]:not(.valid)   input[_ngcontent-%COMP%]{caret-color:currentColor}[_nghost-%COMP%]:not(.valid)   .error-icon[_ngcontent-%COMP%]{color:#c62828;position:absolute;right:0}"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/annotations_list/annotations_list_toolbar/annotations_search/annotations_search_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class AnnotationsSearchContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.annotationsFilter$ = this.store.select(getAnnotationsRegex);
            this.isAnnotationsFilterValid$ = this.annotationsFilter$.pipe(map((/**
             * @param {?} filterString
             * @return {?}
             */
            (filterString) => {
                try {
                    // tslint:disable-next-line:no-unused-expression Check for validity of filter.
                    new RegExp(filterString);
                    return true;
                }
                catch (err) {
                    return false;
                }
            })));
        }
        /**
         * @param {?} filter
         * @return {?}
         */
        filterChange(filter) {
            this.store.dispatch(npmiAnnotationsRegexChanged({ regex: filter }));
        }
    }
    AnnotationsSearchContainer.decorators = [
        { type: Component, args: [{
                    selector: 'npmi-annotations-search',
                    template: `
    <npmi-annotations-search-component
      [regexFilterValue]="annotationsFilter$ | async"
      [isRegexFilterValid]="isAnnotationsFilterValid$ | async"
      (onRegexFilterValueChange)="filterChange($event)"
    ></npmi-annotations-search-component>
  `,
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    AnnotationsSearchContainer.ctorParameters = () => [
        { type: Store }
    ];
    /** @nocollapse */ AnnotationsSearchContainer.ɵfac = function AnnotationsSearchContainer_Factory(t) { return new (t || AnnotationsSearchContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ AnnotationsSearchContainer.ɵcmp = ɵɵdefineComponent({ type: AnnotationsSearchContainer, selectors: [["npmi-annotations-search"]], decls: 3, vars: 6, consts: [[3, "regexFilterValue", "isRegexFilterValid", "onRegexFilterValueChange"]], template: function AnnotationsSearchContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "npmi-annotations-search-component", 0);
            ɵɵlistener("onRegexFilterValueChange", function AnnotationsSearchContainer_Template_npmi_annotations_search_component_onRegexFilterValueChange_0_listener($event) { return ctx.filterChange($event); });
            ɵɵpipe(1, "async");
            ɵɵpipe(2, "async");
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵproperty("regexFilterValue", ɵɵpipeBind1(1, 2, ctx.annotationsFilter$))("isRegexFilterValid", ɵɵpipeBind1(2, 4, ctx.isAnnotationsFilterValid$));
        } }, directives: [AnnotationsSearchComponent], pipes: [AsyncPipe], encapsulation: 2, changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/annotations_list/annotations_list_toolbar/annotations_list_toolbar_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    var I18N_0$9;
    if (typeof ngI18nClosureMode !== "undefined" && ngI18nClosureMode) {
        /**
         * @desc Label for a button that hides/shows the annotations list.
         */ 
        const MSG_EXTERNAL_7336374413056342492$$TENSORBOARD_WEBAPP_PLUGINS_NPMI_VIEWS_ANNOTATIONS_LIST_ANNOTATIONS_LIST_TOOLBAR_ANNOTATIONS_LIST_TOOLBAR_COMPONENT_TS_1 = goog.getMsg("Hides/Shows the Annotations List");
        I18N_0$9 = MSG_EXTERNAL_7336374413056342492$$TENSORBOARD_WEBAPP_PLUGINS_NPMI_VIEWS_ANNOTATIONS_LIST_ANNOTATIONS_LIST_TOOLBAR_ANNOTATIONS_LIST_TOOLBAR_COMPONENT_TS_1;
    }
    else {
        I18N_0$9 = $localize `:Label for a button that hides/shows the annotations list.␟b3603ba33e5308dd8c5e805e508b2f7233df89d4␟7336374413056342492:Hides/Shows the Annotations List`;
    }
    const _c2$g = ["aria-label", I18N_0$9];
    var I18N_3$6;
    if (typeof ngI18nClosureMode !== "undefined" && ngI18nClosureMode) {
        /**
         * @desc Label for a button that flags selected annotations.
         */ 
        const MSG_EXTERNAL_2244099891313336595$$TENSORBOARD_WEBAPP_PLUGINS_NPMI_VIEWS_ANNOTATIONS_LIST_ANNOTATIONS_LIST_TOOLBAR_ANNOTATIONS_LIST_TOOLBAR_COMPONENT_TS__4 = goog.getMsg("Flag Selected Annotations");
        I18N_3$6 = MSG_EXTERNAL_2244099891313336595$$TENSORBOARD_WEBAPP_PLUGINS_NPMI_VIEWS_ANNOTATIONS_LIST_ANNOTATIONS_LIST_TOOLBAR_ANNOTATIONS_LIST_TOOLBAR_COMPONENT_TS__4;
    }
    else {
        I18N_3$6 = $localize `:Label for a button that flags selected annotations.␟5692ad8831038a90c5863a1e9adf9748cac3cad8␟2244099891313336595:Flag Selected Annotations`;
    }
    const _c5$8 = ["aria-label", I18N_3$6];
    var I18N_6$2;
    if (typeof ngI18nClosureMode !== "undefined" && ngI18nClosureMode) {
        /**
         * @desc Label for a button that hides selected annotations.
         */ 
        const MSG_EXTERNAL_5462832391092087485$$TENSORBOARD_WEBAPP_PLUGINS_NPMI_VIEWS_ANNOTATIONS_LIST_ANNOTATIONS_LIST_TOOLBAR_ANNOTATIONS_LIST_TOOLBAR_COMPONENT_TS__7 = goog.getMsg("Hide Selected Annotations");
        I18N_6$2 = MSG_EXTERNAL_5462832391092087485$$TENSORBOARD_WEBAPP_PLUGINS_NPMI_VIEWS_ANNOTATIONS_LIST_ANNOTATIONS_LIST_TOOLBAR_ANNOTATIONS_LIST_TOOLBAR_COMPONENT_TS__7;
    }
    else {
        I18N_6$2 = $localize `:Label for a button that hides selected annotations.␟0342cdb3358fa8e3fa27220a8258a7287430b70f␟5462832391092087485:Hide Selected Annotations`;
    }
    const _c8$3 = ["aria-label", I18N_6$2];
    function AnnotationsListToolbarComponent_ng_container_3_Template(rf, ctx) { if (rf & 1) {
        const _r3 = ɵɵgetCurrentView();
        ɵɵelementContainerStart(0);
        ɵɵelementStart(1, "button", 5);
        ɵɵi18nAttributes(2, _c5$8);
        ɵɵlistener("click", function AnnotationsListToolbarComponent_ng_container_3_Template_button_click_1_listener() { ɵɵrestoreView(_r3); const ctx_r2 = ɵɵnextContext(); return ctx_r2.onFlagAnnotations.emit(ctx_r2.selectedAnnotations); });
        ɵɵelement(3, "mat-icon", 6);
        ɵɵelementEnd();
        ɵɵelementStart(4, "button", 7);
        ɵɵi18nAttributes(5, _c8$3);
        ɵɵlistener("click", function AnnotationsListToolbarComponent_ng_container_3_Template_button_click_4_listener() { ɵɵrestoreView(_r3); const ctx_r4 = ɵɵnextContext(); return ctx_r4.onHideAnnotations.emit(ctx_r4.selectedAnnotations); });
        ɵɵelement(6, "mat-icon", 8);
        ɵɵelementEnd();
        ɵɵelementContainerEnd();
    } if (rf & 2) {
        const ctx_r0 = ɵɵnextContext();
        ɵɵadvance(1);
        ɵɵproperty("disabled", ctx_r0.selectedAnnotations.length === 0);
        ɵɵadvance(3);
        ɵɵproperty("disabled", ctx_r0.selectedAnnotations.length === 0);
    } }
    function AnnotationsListToolbarComponent_ng_container_4_Template(rf, ctx) { if (rf & 1) {
        const _r6 = ɵɵgetCurrentView();
        ɵɵelementContainerStart(0);
        ɵɵelementStart(1, "mat-slide-toggle", 9);
        ɵɵlistener("change", function AnnotationsListToolbarComponent_ng_container_4_Template_mat_slide_toggle_change_1_listener() { ɵɵrestoreView(_r6); const ctx_r5 = ɵɵnextContext(); return ctx_r5.onToggleShowCounts.emit(); });
        ɵɵtext(2, " Sample Count ");
        ɵɵelementEnd();
        ɵɵelementStart(3, "mat-slide-toggle", 10);
        ɵɵlistener("change", function AnnotationsListToolbarComponent_ng_container_4_Template_mat_slide_toggle_change_3_listener() { ɵɵrestoreView(_r6); const ctx_r7 = ɵɵnextContext(); return ctx_r7.onToggleShowHidden.emit(); });
        ɵɵtext(4, " Show Hidden ");
        ɵɵelementEnd();
        ɵɵelement(5, "npmi-annotations-search");
        ɵɵelementContainerEnd();
    } if (rf & 2) {
        const ctx_r1 = ɵɵnextContext();
        ɵɵadvance(1);
        ɵɵproperty("checked", ctx_r1.showCounts);
        ɵɵadvance(2);
        ɵɵproperty("checked", ctx_r1.showHidden);
    } }
    class AnnotationsListToolbarComponent {
        constructor() {
            this.onFlagAnnotations = new EventEmitter();
            this.onHideAnnotations = new EventEmitter();
            this.onToggleExpanded = new EventEmitter();
            this.onToggleShowCounts = new EventEmitter();
            this.onToggleShowHidden = new EventEmitter();
        }
    }
    AnnotationsListToolbarComponent.decorators = [
        { type: Component, args: [{
                    selector: 'npmi-annotations-list-toolbar-component',
                    templateUrl: './annotations_list_toolbar_component.ng.html',
                    styleUrls: ['./annotations_list_toolbar_component.css'],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    AnnotationsListToolbarComponent.propDecorators = {
        numAnnotations: [{ type: Input }],
        expanded: [{ type: Input }],
        selectedAnnotations: [{ type: Input }],
        annotationsExpanded: [{ type: Input }],
        showCounts: [{ type: Input }],
        showHidden: [{ type: Input }],
        onFlagAnnotations: [{ type: Output }],
        onHideAnnotations: [{ type: Output }],
        onToggleExpanded: [{ type: Output }],
        onToggleShowCounts: [{ type: Output }],
        onToggleShowHidden: [{ type: Output }]
    };
    /** @nocollapse */ AnnotationsListToolbarComponent.ɵfac = function AnnotationsListToolbarComponent_Factory(t) { return new (t || AnnotationsListToolbarComponent)(); };
    /** @nocollapse */ AnnotationsListToolbarComponent.ɵcmp = ɵɵdefineComponent({ type: AnnotationsListToolbarComponent, selectors: [["npmi-annotations-list-toolbar-component"]], inputs: { numAnnotations: "numAnnotations", expanded: "expanded", selectedAnnotations: "selectedAnnotations", annotationsExpanded: "annotationsExpanded", showCounts: "showCounts", showHidden: "showHidden" }, outputs: { onFlagAnnotations: "onFlagAnnotations", onHideAnnotations: "onHideAnnotations", onToggleExpanded: "onToggleExpanded", onToggleShowCounts: "onToggleShowCounts", onToggleShowHidden: "onToggleShowHidden" }, decls: 8, vars: 4, consts: [[1, "annotations-title-container"], [1, "annotations-title"], [4, "ngIf"], ["mat-icon-button", "", 1, "expand-button", 3, "click", 6, "aria-label"], [3, "svgIcon"], ["mat-icon-button", "", "title", "Flagging annotations adds them to your investigation results, which can later be exported.", 3, "disabled", "click", 6, "aria-label"], ["svgIcon", "flag_24px"], ["mat-icon-button", "", "title", "Removing non-critical annotations unclutters the view. Removed annotations are removed from all visualizations.", 3, "disabled", "click", 6, "aria-label"], ["svgIcon", "visibility_off_24px"], ["title", "Hides and shows the sample count where applicable (how many samples belong to a category).", 1, "show-toggle", 3, "checked", "change"], ["title", "Hides and shows hidden annotations in all visualizations.", 1, "show-toggle", 3, "checked", "change"]], template: function AnnotationsListToolbarComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "div", 0);
            ɵɵelementStart(1, "h3", 1);
            ɵɵtext(2);
            ɵɵelementEnd();
            ɵɵtemplate(3, AnnotationsListToolbarComponent_ng_container_3_Template, 7, 2, "ng-container", 2);
            ɵɵelementEnd();
            ɵɵtemplate(4, AnnotationsListToolbarComponent_ng_container_4_Template, 6, 2, "ng-container", 2);
            ɵɵelementStart(5, "button", 3);
            ɵɵi18nAttributes(6, _c2$g);
            ɵɵlistener("click", function AnnotationsListToolbarComponent_Template_button_click_5_listener() { return ctx.onToggleExpanded.emit(); });
            ɵɵelement(7, "mat-icon", 4);
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵadvance(2);
            ɵɵtextInterpolate1("Annotations (", ctx.numAnnotations, ")");
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx.expanded);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx.expanded);
            ɵɵadvance(3);
            ɵɵproperty("svgIcon", ctx.expanded ? "expand_less_24px" : "expand_more_24px");
        } }, directives: [NgIf, MatButton, MatIcon, MatSlideToggle, AnnotationsSearchContainer], styles: [".mat-badge-content[_ngcontent-%COMP%]{font-weight:600;font-size:12px;font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:9px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:24px}.mat-h1[_ngcontent-%COMP%], .mat-headline[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%]{font:400 24px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h2[_ngcontent-%COMP%], .mat-title[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h3[_ngcontent-%COMP%], .mat-subheading-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%]{font:400 16px/28px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h4[_ngcontent-%COMP%], .mat-subheading-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%]{font:400 15px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h5[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%]{font:400 calc(14px * 0.83)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-h6[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:400 calc(14px * 0.67)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-body-strong[_ngcontent-%COMP%], .mat-body-2[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   p[_ngcontent-%COMP%]{margin:0 0 12px}.mat-small[_ngcontent-%COMP%], .mat-caption[_ngcontent-%COMP%]{font:400 12px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-display-4[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-4[_ngcontent-%COMP%]{font:300 112px/112px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.05em;margin:0 0 56px}.mat-display-3[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-3[_ngcontent-%COMP%]{font:400 56px/56px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.02em;margin:0 0 64px}.mat-display-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-2[_ngcontent-%COMP%]{font:400 45px/48px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.005em;margin:0 0 64px}.mat-display-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-1[_ngcontent-%COMP%]{font:400 34px/40px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 64px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%], .mat-flat-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-button-toggle[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card-title[_ngcontent-%COMP%]{font-size:24px;font-weight:500}.mat-card-header[_ngcontent-%COMP%]   .mat-card-title[_ngcontent-%COMP%]{font-size:20px}.mat-card-subtitle[_ngcontent-%COMP%], .mat-card-content[_ngcontent-%COMP%]{font-size:14px}.mat-checkbox[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-checkbox-layout[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{line-height:24px}.mat-chip[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-chip[_ngcontent-%COMP%]   .mat-chip-trailing-icon.mat-icon[_ngcontent-%COMP%], .mat-chip[_ngcontent-%COMP%]   .mat-chip-remove.mat-icon[_ngcontent-%COMP%]{font-size:18px}.mat-table[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-header-cell[_ngcontent-%COMP%]{font-size:12px;font-weight:500}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{font-size:14px}.mat-calendar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-calendar-body[_ngcontent-%COMP%]{font-size:13px}.mat-calendar-body-label[_ngcontent-%COMP%], .mat-calendar-period-button[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-calendar-table-header[_ngcontent-%COMP%]   th[_ngcontent-%COMP%]{font-size:11px;font-weight:400}.mat-dialog-title[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-expansion-panel-header[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:15px;font-weight:400}.mat-expansion-panel-content[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field[_ngcontent-%COMP%]{font-size:inherit;font-weight:400;line-height:1.125;font-family:Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.34375em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{font-size:150%;line-height:1.125}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]{height:1.5em;width:1.5em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{height:1.125em;line-height:1.125}.mat-form-field-infix[_ngcontent-%COMP%]{padding:.5em 0;border-top:.84375em solid transparent}.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34375em) scale(0.75);width:133.3333333333%}.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34374em) scale(0.75);width:133.3333433333%}.mat-form-field-label-wrapper[_ngcontent-%COMP%]{top:-0.84375em;padding-top:.84375em}.mat-form-field-label[_ngcontent-%COMP%]{top:1.34375em}.mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.34375em}.mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{font-size:75%;margin-top:.6666666667em;top:calc(100% - 1.7916666667em)}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.4375em 0}.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.001px);-ms-transform:translateY(-1.28125em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00101px);-ms-transform:translateY(-1.28124em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00102px);-ms-transform:translateY(-1.28123em) scale(0.75);width:133.3333533333%}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.28125em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{margin-top:.5416666667em;top:calc(100% - 1.6666666667em)}@media print{.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28122em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28121em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.2812em) scale(0.75)}}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.25em 0 .75em 0}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.09375em;margin-top:-0.5em}.mat-form-field-appearance-fill.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59374em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:1em 0 1em 0}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.84375em;margin-top:-0.25em}.mat-form-field-appearance-outline.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59374em) scale(0.75);width:133.3333433333%}.mat-grid-tile-header[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]{font-size:14px}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2), .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}input.mat-input-element[_ngcontent-%COMP%]{margin-top:-0.0625em}.mat-menu-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:400}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px}.mat-radio-button[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select-trigger[_ngcontent-%COMP%]{height:1.125em}.mat-slide-toggle-content[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-slider-thumb-label-text[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-stepper-vertical[_ngcontent-%COMP%], .mat-stepper-horizontal[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-step-label[_ngcontent-%COMP%]{font-size:14px;font-weight:400}.mat-step-sub-label-error[_ngcontent-%COMP%]{font-weight:normal}.mat-step-label-error[_ngcontent-%COMP%]{font-size:14px}.mat-step-label-selected[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-tab-group[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-toolbar[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0}.mat-tooltip[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:10px;padding-top:6px;padding-bottom:6px}.mat-tooltip-handset[_ngcontent-%COMP%]{font-size:14px;padding-top:8px;padding-bottom:8px}.mat-list-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:16px}.mat-optgroup-label[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-simple-snackbar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px}.mat-simple-snackbar-action[_ngcontent-%COMP%]{line-height:1;font-family:inherit;font-size:inherit;font-weight:500}.mat-tree[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{font-weight:400;font-size:14px}.mat-ripple[_ngcontent-%COMP%]{overflow:hidden;position:relative}.mat-ripple[_ngcontent-%COMP%]:not(:empty){transform:translateZ(0)}.mat-ripple.mat-ripple-unbounded[_ngcontent-%COMP%]{overflow:visible}.mat-ripple-element[_ngcontent-%COMP%]{position:absolute;border-radius:50%;pointer-events:none;transition:opacity,transform 0ms cubic-bezier(0, 0, 0.2, 1);transform:scale(0)}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{display:none}.cdk-visually-hidden[_ngcontent-%COMP%]{border:0;clip:rect(0 0 0 0);height:1px;margin:-1px;overflow:hidden;padding:0;position:absolute;width:1px;outline:0;-webkit-appearance:none;-moz-appearance:none}.cdk-overlay-container[_ngcontent-%COMP%], .cdk-global-overlay-wrapper[_ngcontent-%COMP%]{pointer-events:none;top:0;left:0;height:100%;width:100%}.cdk-overlay-container[_ngcontent-%COMP%]{position:fixed;z-index:1000}.cdk-overlay-container[_ngcontent-%COMP%]:empty{display:none}.cdk-global-overlay-wrapper[_ngcontent-%COMP%]{display:flex;position:absolute;z-index:1000}.cdk-overlay-pane[_ngcontent-%COMP%]{position:absolute;pointer-events:auto;box-sizing:border-box;z-index:1000;display:flex;max-width:100%;max-height:100%}.cdk-overlay-backdrop[_ngcontent-%COMP%]{position:absolute;top:0;bottom:0;left:0;right:0;z-index:1000;pointer-events:auto;-webkit-tap-highlight-color:transparent;transition:opacity 400ms cubic-bezier(0.25, 0.8, 0.25, 1);opacity:0}.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:1}@media screen and (-ms-high-contrast: active){.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:.6}}.cdk-overlay-dark-backdrop[_ngcontent-%COMP%]{background:rgba(0,0,0,.32)}.cdk-overlay-transparent-backdrop[_ngcontent-%COMP%], .cdk-overlay-transparent-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:0}.cdk-overlay-connected-position-bounding-box[_ngcontent-%COMP%]{position:absolute;z-index:1000;display:flex;flex-direction:column;min-width:1px;min-height:1px}.cdk-global-scrollblock[_ngcontent-%COMP%]{position:fixed;width:100%;overflow-y:scroll}@keyframes cdk-text-field-autofill-start{}@keyframes cdk-text-field-autofill-end{}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:-webkit-autofill{animation:cdk-text-field-autofill-start 0s 1ms}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:not(:-webkit-autofill){animation:cdk-text-field-autofill-end 0s 1ms}textarea.cdk-textarea-autosize[_ngcontent-%COMP%]{resize:none}textarea.cdk-textarea-autosize-measuring[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:auto !important;overflow:hidden !important}textarea.cdk-textarea-autosize-measuring-firefox[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:0 !important}.mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.1)}.mat-option[_ngcontent-%COMP%]{color:#212121}.mat-option[_ngcontent-%COMP%]:hover:not(.mat-option-disabled), .mat-option[_ngcontent-%COMP%]:focus:not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple):not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-active[_ngcontent-%COMP%]{background:rgba(0,0,0,.04);color:#212121}.mat-option.mat-option-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-primary[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f57c00}.mat-accent[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f44336}.mat-optgroup-label[_ngcontent-%COMP%]{color:#616161}.mat-optgroup-disabled[_ngcontent-%COMP%]   .mat-optgroup-label[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-pseudo-checkbox[_ngcontent-%COMP%]{color:#616161}.mat-pseudo-checkbox[_ngcontent-%COMP%]::after{color:#fafafa}.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{color:#b0b0b0}.mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f57c00}.mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f44336}.mat-pseudo-checkbox-checked.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{background:#b0b0b0}.mat-elevation-z0[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z1[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z2[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z3[_ngcontent-%COMP%]{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z4[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z5[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 5px 8px 0px rgba(0, 0, 0, 0.14),0px 1px 14px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z6[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z7[_ngcontent-%COMP%]{box-shadow:0px 4px 5px -2px rgba(0, 0, 0, 0.2),0px 7px 10px 1px rgba(0, 0, 0, 0.14),0px 2px 16px 1px rgba(0, 0, 0, 0.12)}.mat-elevation-z8[_ngcontent-%COMP%]{box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z9[_ngcontent-%COMP%]{box-shadow:0px 5px 6px -3px rgba(0, 0, 0, 0.2),0px 9px 12px 1px rgba(0, 0, 0, 0.14),0px 3px 16px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z10[_ngcontent-%COMP%]{box-shadow:0px 6px 6px -3px rgba(0, 0, 0, 0.2),0px 10px 14px 1px rgba(0, 0, 0, 0.14),0px 4px 18px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z11[_ngcontent-%COMP%]{box-shadow:0px 6px 7px -4px rgba(0, 0, 0, 0.2),0px 11px 15px 1px rgba(0, 0, 0, 0.14),0px 4px 20px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z12[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z13[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 13px 19px 2px rgba(0, 0, 0, 0.14),0px 5px 24px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z14[_ngcontent-%COMP%]{box-shadow:0px 7px 9px -4px rgba(0, 0, 0, 0.2),0px 14px 21px 2px rgba(0, 0, 0, 0.14),0px 5px 26px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z15[_ngcontent-%COMP%]{box-shadow:0px 8px 9px -5px rgba(0, 0, 0, 0.2),0px 15px 22px 2px rgba(0, 0, 0, 0.14),0px 6px 28px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z16[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z17[_ngcontent-%COMP%]{box-shadow:0px 8px 11px -5px rgba(0, 0, 0, 0.2),0px 17px 26px 2px rgba(0, 0, 0, 0.14),0px 6px 32px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z18[_ngcontent-%COMP%]{box-shadow:0px 9px 11px -5px rgba(0, 0, 0, 0.2),0px 18px 28px 2px rgba(0, 0, 0, 0.14),0px 7px 34px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z19[_ngcontent-%COMP%]{box-shadow:0px 9px 12px -6px rgba(0, 0, 0, 0.2),0px 19px 29px 2px rgba(0, 0, 0, 0.14),0px 7px 36px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z20[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 20px 31px 3px rgba(0, 0, 0, 0.14),0px 8px 38px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z21[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 21px 33px 3px rgba(0, 0, 0, 0.14),0px 8px 40px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z22[_ngcontent-%COMP%]{box-shadow:0px 10px 14px -6px rgba(0, 0, 0, 0.2),0px 22px 35px 3px rgba(0, 0, 0, 0.14),0px 8px 42px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z23[_ngcontent-%COMP%]{box-shadow:0px 11px 14px -7px rgba(0, 0, 0, 0.2),0px 23px 36px 3px rgba(0, 0, 0, 0.14),0px 9px 44px 8px rgba(0, 0, 0, 0.12)}.mat-elevation-z24[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12)}.mat-app-background[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-theme-loaded-marker[_ngcontent-%COMP%]{display:none}.mat-autocomplete-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-autocomplete-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover){background:#fff}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover):not(.mat-option-disabled){color:#212121}.mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f57c00}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{outline:solid 1px;border-radius:0}.mat-badge-accent[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-badge-warn[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f44336}.mat-badge[_ngcontent-%COMP%]{position:relative}.mat-badge-hidden[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{display:none}.mat-badge-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#b9b9b9;color:#757575}.mat-badge-content[_ngcontent-%COMP%]{position:absolute;text-align:center;display:inline-block;border-radius:50%;transition:transform 200ms ease-in-out;transform:scale(0.6);overflow:hidden;white-space:nowrap;text-overflow:ellipsis;pointer-events:none}.ng-animate-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%], .mat-badge-content._mat-animation-noopable[_ngcontent-%COMP%]{transition:none}.mat-badge-content.mat-badge-active[_ngcontent-%COMP%]{transform:none}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:16px;height:16px;line-height:16px}.mat-badge-small.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-8px}.mat-badge-small.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-8px}.mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-16px}.mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-16px}.mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-8px}.mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-8px}.mat-badge-medium[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:22px;height:22px;line-height:22px}.mat-badge-medium.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-11px}.mat-badge-medium.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-11px}.mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-22px}.mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-22px}.mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-11px}.mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-11px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:28px;height:28px;line-height:28px}.mat-badge-large.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-14px}.mat-badge-large.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-14px}.mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-28px}.mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-28px}.mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-14px}.mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-14px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]{color:inherit;background:transparent}.mat-button.mat-primary[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-button[disabled][disabled][_ngcontent-%COMP%], .mat-icon-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-icon-button[disabled][disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-stroked-button[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f44336}.mat-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:transparent}.mat-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{opacity:.1;background-color:currentColor}.mat-button-focus-overlay[_ngcontent-%COMP%]{background:#000}.mat-stroked-button[_ngcontent-%COMP%]:not([disabled]){border-color:rgba(0,0,0,.12)}.mat-flat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{color:#212121;background-color:#fff}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-flat-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-stroked-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-flat-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-raised-button[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone[_ngcontent-%COMP%], .mat-button-toggle-group[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{box-shadow:none}.mat-button-toggle[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-button-toggle[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121;background:#fff}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:#000}.mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12)}[dir=rtl][_ngcontent-%COMP%]   .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-group-appearance-standard.mat-button-toggle-vertical[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:none;border-top:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#616161}.mat-button-toggle-checked.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121}.mat-button-toggle-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.26);background-color:#eee}.mat-button-toggle-disabled.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{background:#fff}.mat-button-toggle-disabled.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#bdbdbd}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{border:solid 1px rgba(0,0,0,.12)}.mat-card[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-card[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-card.mat-card-flat[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-card-subtitle[_ngcontent-%COMP%]{color:#616161}.mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#616161}.mat-checkbox-checkmark[_ngcontent-%COMP%]{fill:#fafafa}.mat-checkbox-checkmark-path[_ngcontent-%COMP%]{stroke:#fafafa !important}.mat-checkbox-mixedmark[_ngcontent-%COMP%]{background-color:#fafafa}.mat-checkbox-indeterminate.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f57c00}.mat-checkbox-indeterminate.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#ff9800}.mat-checkbox-indeterminate.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f44336}.mat-checkbox-disabled.mat-checkbox-checked[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-disabled.mat-checkbox-indeterminate[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]:not(.mat-checkbox-checked)   .mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{color:#616161}.mat-checkbox[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%]{background:#f57c00}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%]{background:#ff9800}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%]{background:#f44336}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#212121}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#212121;opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled):active{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled)   .mat-chip-remove[_ngcontent-%COMP%]:hover{opacity:.54}.mat-chip.mat-standard-chip.mat-chip-disabled[_ngcontent-%COMP%]{opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]::after{background:#000}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-table[_ngcontent-%COMP%]{background:#fff}.mat-table[_ngcontent-%COMP%]   thead[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tbody[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tfoot[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], [mat-header-row][_ngcontent-%COMP%], [mat-row][_ngcontent-%COMP%], [mat-footer-row][_ngcontent-%COMP%], .mat-table-sticky[_ngcontent-%COMP%]{background:inherit}mat-row[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], th.mat-header-cell[_ngcontent-%COMP%], td.mat-cell[_ngcontent-%COMP%], td.mat-footer-cell[_ngcontent-%COMP%]{border-bottom-color:rgba(0,0,0,.12)}.mat-header-cell[_ngcontent-%COMP%]{color:#616161}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{color:#212121}.mat-calendar-arrow[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.54)}.mat-datepicker-toggle[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-next-button[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-previous-button[_ngcontent-%COMP%]{color:rgba(0,0,0,.54)}.mat-calendar-table-header[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-calendar-table-header-divider[_ngcontent-%COMP%]::after{background:rgba(0,0,0,.12)}.mat-calendar-body-label[_ngcontent-%COMP%]{color:#616161}.mat-calendar-body-cell-content[_ngcontent-%COMP%]{color:#212121;border-color:transparent}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){color:#757575}.mat-calendar-body-cell[_ngcontent-%COMP%]:not(.mat-calendar-body-disabled):hover > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-keyboard-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-program-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){background-color:rgba(0,0,0,.04)}.mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.38)}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.18)}.mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.4)}.mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12);background-color:#fff;color:#212121}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.4)}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.4)}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content-touch[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-datepicker-toggle-active[_ngcontent-%COMP%]{color:#f57c00}.mat-datepicker-toggle-active.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-datepicker-toggle-active.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-dialog-container[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-divider[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-divider-vertical[_ngcontent-%COMP%]{border-right-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-expansion-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-action-row[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-keyboard-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-program-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded)   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover:not([aria-disabled=true]){background:rgba(0,0,0,.04)}@media(hover: none){.mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded):not([aria-disabled=true])   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover{background:#fff}}.mat-expansion-panel-header-title[_ngcontent-%COMP%]{color:#212121}.mat-expansion-panel-header-description[_ngcontent-%COMP%], .mat-expansion-indicator[_ngcontent-%COMP%]::after{color:#616161}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-title[_ngcontent-%COMP%], .mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-description[_ngcontent-%COMP%]{color:inherit}.mat-form-field-label[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-hint[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-focused[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-ripple[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid)   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f57c00}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-accent   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#ff9800}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-warn   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#f44336}.mat-error[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-hint[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-legacy.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-standard[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-standard.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.04)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.02)}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:transparent}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.12)}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:rgba(0,0,0,.87)}.mat-form-field-appearance-outline.mat-focused[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field-appearance-outline.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-appearance-outline.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-invalid.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.06)}.mat-icon.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-icon.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-icon.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-type-mat-native-select[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#616161}.mat-input-element[_ngcontent-%COMP%]:disabled, .mat-form-field-type-mat-native-select.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#757575}.mat-input-element[_ngcontent-%COMP%]{caret-color:#f57c00}.mat-input-element[_ngcontent-%COMP%]::placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-moz-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-webkit-input-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]:-ms-input-placeholder{color:rgba(97,97,97,.42)}.mat-accent[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%], .mat-form-field-invalid[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#f44336}.mat-form-field-type-mat-native-select.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{color:#616161}.mat-list-item-disabled[_ngcontent-%COMP%]{background-color:#eee}.mat-list-option[_ngcontent-%COMP%]:hover, .mat-list-option[_ngcontent-%COMP%]:focus, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.04)}.mat-list-single-selected-option[_ngcontent-%COMP%], .mat-list-single-selected-option[_ngcontent-%COMP%]:hover, .mat-list-single-selected-option[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.12)}.mat-menu-panel[_ngcontent-%COMP%]{background:#fff}.mat-menu-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-menu-item[_ngcontent-%COMP%]{background:transparent;color:#212121}.mat-menu-item[disabled][_ngcontent-%COMP%], .mat-menu-item[disabled][_ngcontent-%COMP%]::after{color:rgba(0,0,0,.38)}.mat-menu-item[_ngcontent-%COMP%]   .mat-icon-no-color[_ngcontent-%COMP%], .mat-menu-item-submenu-trigger[_ngcontent-%COMP%]::after{color:rgba(0,0,0,.54)}.mat-menu-item[_ngcontent-%COMP%]:hover:not([disabled]), .mat-menu-item.cdk-program-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item.cdk-keyboard-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item-highlighted[_ngcontent-%COMP%]:not([disabled]){background:rgba(0,0,0,.04)}.mat-paginator[_ngcontent-%COMP%]{background:#fff}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{color:#616161}.mat-paginator-decrement[_ngcontent-%COMP%], .mat-paginator-increment[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54);border-right:2px solid rgba(0,0,0,.54)}.mat-paginator-first[_ngcontent-%COMP%], .mat-paginator-last[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54)}.mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-decrement[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-increment[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-first[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-last[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f57c00}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#ff9800}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f44336}.mat-progress-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f57c00}.mat-progress-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#ff9800}.mat-progress-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f44336}.mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#616161}.mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f57c00}.mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#ff9800}.mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#ff9800}.mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f44336}.mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f44336}.mat-radio-button.mat-radio-disabled.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-label-content[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-radio-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-select-value[_ngcontent-%COMP%]{color:#212121}.mat-select-placeholder[_ngcontent-%COMP%]{color:rgba(97,97,97,.42)}.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%]{color:#757575}.mat-select-arrow[_ngcontent-%COMP%]{color:#616161}.mat-select-panel[_ngcontent-%COMP%]{background:#fff}.mat-select-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-select-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple){background:rgba(0,0,0,.12)}.mat-form-field.mat-focused.mat-primary[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-invalid[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#757575}.mat-drawer-container[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-drawer[_ngcontent-%COMP%]{background-color:#fff;color:#212121}.mat-drawer.mat-drawer-push[_ngcontent-%COMP%]{background-color:#fff}.mat-drawer[_ngcontent-%COMP%]:not(.mat-drawer-side){box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-drawer-side[_ngcontent-%COMP%]{border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-backdrop.mat-drawer-shown[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.6)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.54)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.54)}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.54)}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle[_ngcontent-%COMP%]:not(.mat-checked)   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-slide-toggle-thumb[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12);background-color:#fafafa}.mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-primary[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f57c00}.mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-accent[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#ff9800}.mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-warn[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f44336}.mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.2)}.mat-slider[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%], .cdk-focused[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-disabled[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]   .mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing)   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26);background-color:transparent}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26)}.mat-slider-has-ticks[_ngcontent-%COMP%]   .mat-slider-wrapper[_ngcontent-%COMP%]::after{border-color:rgba(0,0,0,.7)}.mat-slider-horizontal[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to right, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent);background-image:-moz-repeating-linear-gradient(0.0001deg, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-slider-vertical[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to bottom, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-step-header.cdk-keyboard-focused[_ngcontent-%COMP%], .mat-step-header.cdk-program-focused[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]:hover{background-color:rgba(0,0,0,.04)}@media(hover: none){.mat-step-header[_ngcontent-%COMP%]:hover{background:none}}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-optional[_ngcontent-%COMP%]{color:#616161}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon[_ngcontent-%COMP%]{background-color:#616161;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-selected[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-done[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-edit[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-error[_ngcontent-%COMP%]{background-color:transparent;color:#f44336}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-active[_ngcontent-%COMP%]{color:#212121}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-error[_ngcontent-%COMP%]{color:#f44336}.mat-stepper-horizontal[_ngcontent-%COMP%], .mat-stepper-vertical[_ngcontent-%COMP%]{background-color:#fff}.mat-stepper-vertical-line[_ngcontent-%COMP%]::before{border-left-color:rgba(0,0,0,.12)}.mat-horizontal-stepper-header[_ngcontent-%COMP%]::before, .mat-horizontal-stepper-header[_ngcontent-%COMP%]::after, .mat-stepper-horizontal-line[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-sort-header-arrow[_ngcontent-%COMP%]{color:#616161}.mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-header[_ngcontent-%COMP%]{border-bottom:1px solid rgba(0,0,0,.12)}.mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%]{border-top:1px solid rgba(0,0,0,.12);border-bottom:none}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{color:#212121}.mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:#757575}.mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#212121}.mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#757575}.mat-tab-group[class*=mat-background-][_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar[class*=mat-background-][_ngcontent-%COMP%]{border-bottom:none;border-top:none}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-toolbar[_ngcontent-%COMP%]{background:#f5f5f5;color:#212121}.mat-toolbar.mat-primary[_ngcontent-%COMP%]{background:#f57c00;color:#fff}.mat-toolbar.mat-accent[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-toolbar.mat-warn[_ngcontent-%COMP%]{background:#f44336;color:#fff}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:currentColor}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:inherit}.mat-toolbar[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:currentColor}.mat-tooltip[_ngcontent-%COMP%]{background:rgba(97,97,97,.9)}.mat-tree[_ngcontent-%COMP%]{background:#fff}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{color:#212121}.mat-snack-bar-container[_ngcontent-%COMP%]{color:rgba(255,255,255,.7);background:#323232;box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-simple-snackbar-action[_ngcontent-%COMP%]{color:#ff9800}[_nghost-%COMP%]{align-items:center;box-sizing:border-box;display:flex;flex-direction:row;padding:0 16px;width:100%}.annotations-title[_ngcontent-%COMP%]{display:inline;font-size:.9em;font-weight:500;padding-right:10px}.annotations-title-container[_ngcontent-%COMP%]{align-items:center;display:flex;flex-wrap:nowrap;flex:1 1;height:42px}.show-toggle[_ngcontent-%COMP%]{font-size:.9em;margin-right:.8em}"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/annotations_list/annotations_list_toolbar/annotations_list_toolbar_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class AnnotationsListToolbarContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.selectedAnnotations$ = this.store.select(getSelectedAnnotations);
            this.annotationsExpanded$ = this.store.select(getAnnotationsExpanded);
            this.showCounts$ = this.store.select(getShowCounts);
            this.showHidden$ = this.store.select(getShowHiddenAnnotations);
            this.annotationsFilter$ = this.store.select(getAnnotationsRegex);
            this.isAnnotationsFilterValid$ = this.annotationsFilter$.pipe(map((/**
             * @param {?} filterString
             * @return {?}
             */
            (filterString) => {
                try {
                    return Boolean(new RegExp(filterString));
                }
                catch (err) {
                    return false;
                }
            })));
        }
        /**
         * @param {?} filter
         * @return {?}
         */
        filterChange(filter) {
            this.store.dispatch(npmiAnnotationsRegexChanged({ regex: filter }));
        }
        /**
         * @param {?} annotations
         * @return {?}
         */
        flagAnnotations(annotations) {
            this.store.dispatch(npmiToggleAnnotationFlags({
                annotations,
            }));
        }
        /**
         * @param {?} annotations
         * @return {?}
         */
        hideAnnotations(annotations) {
            this.store.dispatch(npmiToggleAnnotationsHidden({
                annotations,
            }));
        }
        /**
         * @return {?}
         */
        toggleExpanded() {
            this.store.dispatch(npmiToggleAnnotationsExpanded());
        }
        /**
         * @return {?}
         */
        toggleShowCounts() {
            this.store.dispatch(npmiToggleShowCounts());
        }
        /**
         * @return {?}
         */
        toggleShowHidden() {
            this.store.dispatch(npmiToggleShowHiddenAnnotations());
        }
    }
    AnnotationsListToolbarContainer.decorators = [
        { type: Component, args: [{
                    selector: 'npmi-annotations-list-toolbar',
                    template: `
    <npmi-annotations-list-toolbar-component
      [numAnnotations]="numAnnotations"
      [expanded]="expanded"
      [selectedAnnotations]="selectedAnnotations$ | async"
      [annotationsExpanded]="annotationsExpanded$ | async"
      [showCounts]="showCounts$ | async"
      [showHidden]="showHidden$ | async"
      (onFlagAnnotations)="flagAnnotations($event)"
      (onHideAnnotations)="hideAnnotations($event)"
      (onToggleExpanded)="toggleExpanded()"
      (onToggleShowCounts)="toggleShowCounts()"
      (onToggleShowHidden)="toggleShowHidden()"
    ></npmi-annotations-list-toolbar-component>
  `,
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    AnnotationsListToolbarContainer.ctorParameters = () => [
        { type: Store }
    ];
    AnnotationsListToolbarContainer.propDecorators = {
        numAnnotations: [{ type: Input }],
        expanded: [{ type: Input }]
    };
    /** @nocollapse */ AnnotationsListToolbarContainer.ɵfac = function AnnotationsListToolbarContainer_Factory(t) { return new (t || AnnotationsListToolbarContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ AnnotationsListToolbarContainer.ɵcmp = ɵɵdefineComponent({ type: AnnotationsListToolbarContainer, selectors: [["npmi-annotations-list-toolbar"]], inputs: { numAnnotations: "numAnnotations", expanded: "expanded" }, decls: 5, vars: 14, consts: [[3, "numAnnotations", "expanded", "selectedAnnotations", "annotationsExpanded", "showCounts", "showHidden", "onFlagAnnotations", "onHideAnnotations", "onToggleExpanded", "onToggleShowCounts", "onToggleShowHidden"]], template: function AnnotationsListToolbarContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "npmi-annotations-list-toolbar-component", 0);
            ɵɵlistener("onFlagAnnotations", function AnnotationsListToolbarContainer_Template_npmi_annotations_list_toolbar_component_onFlagAnnotations_0_listener($event) { return ctx.flagAnnotations($event); })("onHideAnnotations", function AnnotationsListToolbarContainer_Template_npmi_annotations_list_toolbar_component_onHideAnnotations_0_listener($event) { return ctx.hideAnnotations($event); })("onToggleExpanded", function AnnotationsListToolbarContainer_Template_npmi_annotations_list_toolbar_component_onToggleExpanded_0_listener() { return ctx.toggleExpanded(); })("onToggleShowCounts", function AnnotationsListToolbarContainer_Template_npmi_annotations_list_toolbar_component_onToggleShowCounts_0_listener() { return ctx.toggleShowCounts(); })("onToggleShowHidden", function AnnotationsListToolbarContainer_Template_npmi_annotations_list_toolbar_component_onToggleShowHidden_0_listener() { return ctx.toggleShowHidden(); });
            ɵɵpipe(1, "async");
            ɵɵpipe(2, "async");
            ɵɵpipe(3, "async");
            ɵɵpipe(4, "async");
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵproperty("numAnnotations", ctx.numAnnotations)("expanded", ctx.expanded)("selectedAnnotations", ɵɵpipeBind1(1, 6, ctx.selectedAnnotations$))("annotationsExpanded", ɵɵpipeBind1(2, 8, ctx.annotationsExpanded$))("showCounts", ɵɵpipeBind1(3, 10, ctx.showCounts$))("showHidden", ɵɵpipeBind1(4, 12, ctx.showHidden$));
        } }, directives: [AnnotationsListToolbarComponent], pipes: [AsyncPipe], encapsulation: 2, changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/annotations_list/legend/legend_element/legend_element_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    const _c0$z = ["glyph"];
    class LegendElementComponent {
        /**
         * @return {?}
         */
        ngAfterViewInit() {
            this.svg = select$1(this.glyphSVG.nativeElement);
            this.mainContainer = this.svg.append('g');
            this.draw();
        }
        /**
         * @private
         * @return {?}
         */
        draw() {
            if (this.shape == 'circle') {
                this.mainContainer
                    .append('circle')
                    .attr('fill', this.color)
                    .attr('stroke', 'black')
                    .attr('cx', 5)
                    .attr('cy', 5)
                    .attr('r', 5);
            }
            else if (this.shape == 'bar') {
                this.mainContainer
                    .append('rect')
                    .attr('fill', this.color)
                    .attr('x', 0)
                    .attr('y', 0)
                    .attr('width', 10)
                    .attr('height', 10);
            }
            else if (this.shape == 'runIndicator') {
                this.mainContainer
                    .append('g')
                    .append('path')
                    .attr('fill', this.color)
                    .attr('stroke', 'black')
                    .attr('d', 'M 2 0 L 10 0 L 7 5 L 10 10 L 2 10 Z');
            }
        }
    }
    LegendElementComponent.decorators = [
        { type: Component, args: [{
                    selector: 'npmi-legend-element',
                    templateUrl: './legend_element_component.ng.html',
                    styleUrls: ['./legend_element_component.css'],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    LegendElementComponent.propDecorators = {
        text: [{ type: Input }],
        color: [{ type: Input }],
        shape: [{ type: Input }],
        glyphSVG: [{ type: ViewChild, args: ['glyph', { static: true, read: ElementRef },] }]
    };
    /** @nocollapse */ LegendElementComponent.ɵfac = function LegendElementComponent_Factory(t) { return new (t || LegendElementComponent)(); };
    /** @nocollapse */ LegendElementComponent.ɵcmp = ɵɵdefineComponent({ type: LegendElementComponent, selectors: [["npmi-legend-element"]], viewQuery: function LegendElementComponent_Query(rf, ctx) { if (rf & 1) {
            ɵɵstaticViewQuery(_c0$z, true, ElementRef);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.glyphSVG = _t.first);
        } }, inputs: { text: "text", color: "color", shape: "shape" }, decls: 4, vars: 1, consts: [[1, "glyph"], ["glyph", ""], [1, "legend-element-title"]], template: function LegendElementComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵnamespaceSVG();
            ɵɵelement(0, "svg", 0, 1);
            ɵɵnamespaceHTML();
            ɵɵelementStart(2, "div", 2);
            ɵɵtext(3);
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵadvance(3);
            ɵɵtextInterpolate(ctx.text);
        } }, styles: ["[_nghost-%COMP%]{align-items:center;display:flex;padding-right:10px}.legend-element-title[_ngcontent-%COMP%]{font-size:.8em;padding-left:5px}.glyph[_ngcontent-%COMP%]{width:10px;height:10px}"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/annotations_list/legend/legend_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class LegendComponent {
    }
    LegendComponent.decorators = [
        { type: Component, args: [{
                    selector: 'npmi-annotations-list-legend',
                    templateUrl: './legend_component.ng.html',
                    styleUrls: ['./legend_component.css'],
                },] },
    ];
    /** @nocollapse */ LegendComponent.ɵfac = function LegendComponent_Factory(t) { return new (t || LegendComponent)(); };
    /** @nocollapse */ LegendComponent.ɵcmp = ɵɵdefineComponent({ type: LegendComponent, selectors: [["npmi-annotations-list-legend"]], decls: 4, vars: 0, consts: [["text", "run indicator", "color", "rgb(0,0,0)", "shape", "runIndicator"], ["text", "positive correlation", "color", "rgb(109, 174, 213)", "shape", "bar"], ["text", "negative correlation", "color", "rgb(249, 105, 76)", "shape", "bar"], ["text", "sample count", "color", "rgb(151, 151, 151)", "shape", "circle"]], template: function LegendComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelement(0, "npmi-legend-element", 0);
            ɵɵelement(1, "npmi-legend-element", 1);
            ɵɵelement(2, "npmi-legend-element", 2);
            ɵɵelement(3, "npmi-legend-element", 3);
        } }, directives: [LegendElementComponent], styles: ["[_nghost-%COMP%]{display:flex;padding:0 16px}"] });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/annotations_list/header/header_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    function HeaderComponent_div_3_mat_icon_4_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "mat-icon", 8);
    } if (rf & 2) {
        const ctx_r2 = ɵɵnextContext(2);
        ɵɵproperty("svgIcon", ctx_r2.sort.order === ctx_r2.SortOrder.DOWN ? "arrow_downward_24px" : "arrow_upward_24px")("ngClass", ctx_r2.sort.order === ctx_r2.SortOrder.DOWN ? "down-icon" : "up-icon");
    } }
    function HeaderComponent_div_3_Template(rf, ctx) { if (rf & 1) {
        const _r4 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "div", 4);
        ɵɵelementStart(1, "div", 5);
        ɵɵelementStart(2, "div", 6);
        ɵɵlistener("click", function HeaderComponent_div_3_Template_div_click_2_listener() { ɵɵrestoreView(_r4); const metric_r1 = ctx.$implicit; const ctx_r3 = ɵɵnextContext(); return ctx_r3.onChangeSort.emit(metric_r1); });
        ɵɵtext(3);
        ɵɵtemplate(4, HeaderComponent_div_3_mat_icon_4_Template, 1, 2, "mat-icon", 7);
        ɵɵelementEnd();
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const metric_r1 = ctx.$implicit;
        const ctx_r0 = ɵɵnextContext();
        ɵɵadvance(3);
        ɵɵtextInterpolate1(" ", ctx_r0.stripMetric(metric_r1), " ");
        ɵɵadvance(1);
        ɵɵproperty("ngIf", metric_r1 === ctx_r0.sort.metric);
    } }
    class HeaderComponent$1 {
        constructor() {
            this.onChangeSort = new EventEmitter();
            this.onAllAnnotationsToggled = new EventEmitter();
            this.SortOrder = SortOrder;
        }
        /**
         * @param {?} metric
         * @return {?}
         */
        stripMetric(metric) {
            return stripMetricString(metric);
        }
    }
    HeaderComponent$1.decorators = [
        { type: Component, args: [{
                    selector: 'npmi-annotations-list-header-component',
                    templateUrl: './header_component.ng.html',
                    styleUrls: ['./header_component.css'],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    HeaderComponent$1.propDecorators = {
        numAnnotations: [{ type: Input }],
        selectedAnnotations: [{ type: Input }],
        activeMetrics: [{ type: Input }],
        sort: [{ type: Input }],
        onChangeSort: [{ type: Output }],
        onAllAnnotationsToggled: [{ type: Output }]
    };
    /** @nocollapse */ HeaderComponent$1.ɵfac = function HeaderComponent_Factory(t) { return new (t || HeaderComponent$1)(); };
    /** @nocollapse */ HeaderComponent$1.ɵcmp = ɵɵdefineComponent({ type: HeaderComponent$1, selectors: [["npmi-annotations-list-header-component"]], inputs: { numAnnotations: "numAnnotations", selectedAnnotations: "selectedAnnotations", activeMetrics: "activeMetrics", sort: "sort" }, outputs: { onChangeSort: "onChangeSort", onAllAnnotationsToggled: "onAllAnnotationsToggled" }, decls: 4, vars: 2, consts: [[1, "toggle-all-container"], [3, "checked", "change"], [1, "annotations-header-containers"], ["class", "header-column", 4, "ngFor", "ngForOf"], [1, "header-column"], [1, "header-container"], ["tabindex", "0", "role", "button", "title", "Change the sort by clicking any of the metrics.", 1, "header-clickable", 3, "click"], ["class", "sort-icon", 3, "svgIcon", "ngClass", 4, "ngIf"], [1, "sort-icon", 3, "svgIcon", "ngClass"]], template: function HeaderComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "div", 0);
            ɵɵelementStart(1, "mat-checkbox", 1);
            ɵɵlistener("change", function HeaderComponent_Template_mat_checkbox_change_1_listener($event) { return ctx.onAllAnnotationsToggled.emit($event.checked); });
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementStart(2, "div", 2);
            ɵɵtemplate(3, HeaderComponent_div_3_Template, 5, 2, "div", 3);
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵadvance(1);
            ɵɵproperty("checked", ctx.selectedAnnotations.length === ctx.numAnnotations);
            ɵɵadvance(2);
            ɵɵproperty("ngForOf", ctx.activeMetrics);
        } }, directives: [MatCheckbox, NgForOf, NgIf, MatIcon, NgClass], styles: [".mat-badge-content[_ngcontent-%COMP%]{font-weight:600;font-size:12px;font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:9px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:24px}.mat-h1[_ngcontent-%COMP%], .mat-headline[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%]{font:400 24px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h2[_ngcontent-%COMP%], .mat-title[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h3[_ngcontent-%COMP%], .mat-subheading-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%]{font:400 16px/28px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h4[_ngcontent-%COMP%], .mat-subheading-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%]{font:400 15px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h5[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%]{font:400 calc(14px * 0.83)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-h6[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:400 calc(14px * 0.67)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-body-strong[_ngcontent-%COMP%], .mat-body-2[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   p[_ngcontent-%COMP%]{margin:0 0 12px}.mat-small[_ngcontent-%COMP%], .mat-caption[_ngcontent-%COMP%]{font:400 12px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-display-4[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-4[_ngcontent-%COMP%]{font:300 112px/112px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.05em;margin:0 0 56px}.mat-display-3[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-3[_ngcontent-%COMP%]{font:400 56px/56px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.02em;margin:0 0 64px}.mat-display-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-2[_ngcontent-%COMP%]{font:400 45px/48px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.005em;margin:0 0 64px}.mat-display-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-1[_ngcontent-%COMP%]{font:400 34px/40px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 64px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%], .mat-flat-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-button-toggle[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card-title[_ngcontent-%COMP%]{font-size:24px;font-weight:500}.mat-card-header[_ngcontent-%COMP%]   .mat-card-title[_ngcontent-%COMP%]{font-size:20px}.mat-card-subtitle[_ngcontent-%COMP%], .mat-card-content[_ngcontent-%COMP%]{font-size:14px}.mat-checkbox[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-checkbox-layout[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{line-height:24px}.mat-chip[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-chip[_ngcontent-%COMP%]   .mat-chip-trailing-icon.mat-icon[_ngcontent-%COMP%], .mat-chip[_ngcontent-%COMP%]   .mat-chip-remove.mat-icon[_ngcontent-%COMP%]{font-size:18px}.mat-table[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-header-cell[_ngcontent-%COMP%]{font-size:12px;font-weight:500}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{font-size:14px}.mat-calendar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-calendar-body[_ngcontent-%COMP%]{font-size:13px}.mat-calendar-body-label[_ngcontent-%COMP%], .mat-calendar-period-button[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-calendar-table-header[_ngcontent-%COMP%]   th[_ngcontent-%COMP%]{font-size:11px;font-weight:400}.mat-dialog-title[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-expansion-panel-header[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:15px;font-weight:400}.mat-expansion-panel-content[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field[_ngcontent-%COMP%]{font-size:inherit;font-weight:400;line-height:1.125;font-family:Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.34375em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{font-size:150%;line-height:1.125}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]{height:1.5em;width:1.5em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{height:1.125em;line-height:1.125}.mat-form-field-infix[_ngcontent-%COMP%]{padding:.5em 0;border-top:.84375em solid transparent}.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34375em) scale(0.75);width:133.3333333333%}.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34374em) scale(0.75);width:133.3333433333%}.mat-form-field-label-wrapper[_ngcontent-%COMP%]{top:-0.84375em;padding-top:.84375em}.mat-form-field-label[_ngcontent-%COMP%]{top:1.34375em}.mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.34375em}.mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{font-size:75%;margin-top:.6666666667em;top:calc(100% - 1.7916666667em)}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.4375em 0}.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.001px);-ms-transform:translateY(-1.28125em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00101px);-ms-transform:translateY(-1.28124em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00102px);-ms-transform:translateY(-1.28123em) scale(0.75);width:133.3333533333%}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.28125em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{margin-top:.5416666667em;top:calc(100% - 1.6666666667em)}@media print{.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28122em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28121em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.2812em) scale(0.75)}}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.25em 0 .75em 0}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.09375em;margin-top:-0.5em}.mat-form-field-appearance-fill.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59374em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:1em 0 1em 0}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.84375em;margin-top:-0.25em}.mat-form-field-appearance-outline.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59374em) scale(0.75);width:133.3333433333%}.mat-grid-tile-header[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]{font-size:14px}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2), .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}input.mat-input-element[_ngcontent-%COMP%]{margin-top:-0.0625em}.mat-menu-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:400}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px}.mat-radio-button[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select-trigger[_ngcontent-%COMP%]{height:1.125em}.mat-slide-toggle-content[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-slider-thumb-label-text[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-stepper-vertical[_ngcontent-%COMP%], .mat-stepper-horizontal[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-step-label[_ngcontent-%COMP%]{font-size:14px;font-weight:400}.mat-step-sub-label-error[_ngcontent-%COMP%]{font-weight:normal}.mat-step-label-error[_ngcontent-%COMP%]{font-size:14px}.mat-step-label-selected[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-tab-group[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-toolbar[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0}.mat-tooltip[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:10px;padding-top:6px;padding-bottom:6px}.mat-tooltip-handset[_ngcontent-%COMP%]{font-size:14px;padding-top:8px;padding-bottom:8px}.mat-list-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:16px}.mat-optgroup-label[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-simple-snackbar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px}.mat-simple-snackbar-action[_ngcontent-%COMP%]{line-height:1;font-family:inherit;font-size:inherit;font-weight:500}.mat-tree[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{font-weight:400;font-size:14px}.mat-ripple[_ngcontent-%COMP%]{overflow:hidden;position:relative}.mat-ripple[_ngcontent-%COMP%]:not(:empty){transform:translateZ(0)}.mat-ripple.mat-ripple-unbounded[_ngcontent-%COMP%]{overflow:visible}.mat-ripple-element[_ngcontent-%COMP%]{position:absolute;border-radius:50%;pointer-events:none;transition:opacity,transform 0ms cubic-bezier(0, 0, 0.2, 1);transform:scale(0)}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{display:none}.cdk-visually-hidden[_ngcontent-%COMP%]{border:0;clip:rect(0 0 0 0);height:1px;margin:-1px;overflow:hidden;padding:0;position:absolute;width:1px;outline:0;-webkit-appearance:none;-moz-appearance:none}.cdk-overlay-container[_ngcontent-%COMP%], .cdk-global-overlay-wrapper[_ngcontent-%COMP%]{pointer-events:none;top:0;left:0;height:100%;width:100%}.cdk-overlay-container[_ngcontent-%COMP%]{position:fixed;z-index:1000}.cdk-overlay-container[_ngcontent-%COMP%]:empty{display:none}.cdk-global-overlay-wrapper[_ngcontent-%COMP%]{display:flex;position:absolute;z-index:1000}.cdk-overlay-pane[_ngcontent-%COMP%]{position:absolute;pointer-events:auto;box-sizing:border-box;z-index:1000;display:flex;max-width:100%;max-height:100%}.cdk-overlay-backdrop[_ngcontent-%COMP%]{position:absolute;top:0;bottom:0;left:0;right:0;z-index:1000;pointer-events:auto;-webkit-tap-highlight-color:transparent;transition:opacity 400ms cubic-bezier(0.25, 0.8, 0.25, 1);opacity:0}.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:1}@media screen and (-ms-high-contrast: active){.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:.6}}.cdk-overlay-dark-backdrop[_ngcontent-%COMP%]{background:rgba(0,0,0,.32)}.cdk-overlay-transparent-backdrop[_ngcontent-%COMP%], .cdk-overlay-transparent-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:0}.cdk-overlay-connected-position-bounding-box[_ngcontent-%COMP%]{position:absolute;z-index:1000;display:flex;flex-direction:column;min-width:1px;min-height:1px}.cdk-global-scrollblock[_ngcontent-%COMP%]{position:fixed;width:100%;overflow-y:scroll}@keyframes cdk-text-field-autofill-start{}@keyframes cdk-text-field-autofill-end{}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:-webkit-autofill{animation:cdk-text-field-autofill-start 0s 1ms}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:not(:-webkit-autofill){animation:cdk-text-field-autofill-end 0s 1ms}textarea.cdk-textarea-autosize[_ngcontent-%COMP%]{resize:none}textarea.cdk-textarea-autosize-measuring[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:auto !important;overflow:hidden !important}textarea.cdk-textarea-autosize-measuring-firefox[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:0 !important}.mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.1)}.mat-option[_ngcontent-%COMP%]{color:#212121}.mat-option[_ngcontent-%COMP%]:hover:not(.mat-option-disabled), .mat-option[_ngcontent-%COMP%]:focus:not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple):not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-active[_ngcontent-%COMP%]{background:rgba(0,0,0,.04);color:#212121}.mat-option.mat-option-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-primary[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f57c00}.mat-accent[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f44336}.mat-optgroup-label[_ngcontent-%COMP%]{color:#616161}.mat-optgroup-disabled[_ngcontent-%COMP%]   .mat-optgroup-label[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-pseudo-checkbox[_ngcontent-%COMP%]{color:#616161}.mat-pseudo-checkbox[_ngcontent-%COMP%]::after{color:#fafafa}.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{color:#b0b0b0}.mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f57c00}.mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f44336}.mat-pseudo-checkbox-checked.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{background:#b0b0b0}.mat-elevation-z0[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z1[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z2[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z3[_ngcontent-%COMP%]{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z4[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z5[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 5px 8px 0px rgba(0, 0, 0, 0.14),0px 1px 14px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z6[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z7[_ngcontent-%COMP%]{box-shadow:0px 4px 5px -2px rgba(0, 0, 0, 0.2),0px 7px 10px 1px rgba(0, 0, 0, 0.14),0px 2px 16px 1px rgba(0, 0, 0, 0.12)}.mat-elevation-z8[_ngcontent-%COMP%]{box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z9[_ngcontent-%COMP%]{box-shadow:0px 5px 6px -3px rgba(0, 0, 0, 0.2),0px 9px 12px 1px rgba(0, 0, 0, 0.14),0px 3px 16px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z10[_ngcontent-%COMP%]{box-shadow:0px 6px 6px -3px rgba(0, 0, 0, 0.2),0px 10px 14px 1px rgba(0, 0, 0, 0.14),0px 4px 18px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z11[_ngcontent-%COMP%]{box-shadow:0px 6px 7px -4px rgba(0, 0, 0, 0.2),0px 11px 15px 1px rgba(0, 0, 0, 0.14),0px 4px 20px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z12[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z13[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 13px 19px 2px rgba(0, 0, 0, 0.14),0px 5px 24px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z14[_ngcontent-%COMP%]{box-shadow:0px 7px 9px -4px rgba(0, 0, 0, 0.2),0px 14px 21px 2px rgba(0, 0, 0, 0.14),0px 5px 26px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z15[_ngcontent-%COMP%]{box-shadow:0px 8px 9px -5px rgba(0, 0, 0, 0.2),0px 15px 22px 2px rgba(0, 0, 0, 0.14),0px 6px 28px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z16[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z17[_ngcontent-%COMP%]{box-shadow:0px 8px 11px -5px rgba(0, 0, 0, 0.2),0px 17px 26px 2px rgba(0, 0, 0, 0.14),0px 6px 32px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z18[_ngcontent-%COMP%]{box-shadow:0px 9px 11px -5px rgba(0, 0, 0, 0.2),0px 18px 28px 2px rgba(0, 0, 0, 0.14),0px 7px 34px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z19[_ngcontent-%COMP%]{box-shadow:0px 9px 12px -6px rgba(0, 0, 0, 0.2),0px 19px 29px 2px rgba(0, 0, 0, 0.14),0px 7px 36px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z20[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 20px 31px 3px rgba(0, 0, 0, 0.14),0px 8px 38px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z21[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 21px 33px 3px rgba(0, 0, 0, 0.14),0px 8px 40px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z22[_ngcontent-%COMP%]{box-shadow:0px 10px 14px -6px rgba(0, 0, 0, 0.2),0px 22px 35px 3px rgba(0, 0, 0, 0.14),0px 8px 42px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z23[_ngcontent-%COMP%]{box-shadow:0px 11px 14px -7px rgba(0, 0, 0, 0.2),0px 23px 36px 3px rgba(0, 0, 0, 0.14),0px 9px 44px 8px rgba(0, 0, 0, 0.12)}.mat-elevation-z24[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12)}.mat-app-background[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-theme-loaded-marker[_ngcontent-%COMP%]{display:none}.mat-autocomplete-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-autocomplete-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover){background:#fff}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover):not(.mat-option-disabled){color:#212121}.mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f57c00}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{outline:solid 1px;border-radius:0}.mat-badge-accent[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-badge-warn[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f44336}.mat-badge[_ngcontent-%COMP%]{position:relative}.mat-badge-hidden[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{display:none}.mat-badge-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#b9b9b9;color:#757575}.mat-badge-content[_ngcontent-%COMP%]{position:absolute;text-align:center;display:inline-block;border-radius:50%;transition:transform 200ms ease-in-out;transform:scale(0.6);overflow:hidden;white-space:nowrap;text-overflow:ellipsis;pointer-events:none}.ng-animate-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%], .mat-badge-content._mat-animation-noopable[_ngcontent-%COMP%]{transition:none}.mat-badge-content.mat-badge-active[_ngcontent-%COMP%]{transform:none}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:16px;height:16px;line-height:16px}.mat-badge-small.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-8px}.mat-badge-small.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-8px}.mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-16px}.mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-16px}.mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-8px}.mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-8px}.mat-badge-medium[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:22px;height:22px;line-height:22px}.mat-badge-medium.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-11px}.mat-badge-medium.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-11px}.mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-22px}.mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-22px}.mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-11px}.mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-11px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:28px;height:28px;line-height:28px}.mat-badge-large.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-14px}.mat-badge-large.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-14px}.mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-28px}.mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-28px}.mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-14px}.mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-14px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]{color:inherit;background:transparent}.mat-button.mat-primary[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-button[disabled][disabled][_ngcontent-%COMP%], .mat-icon-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-icon-button[disabled][disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-stroked-button[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f44336}.mat-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:transparent}.mat-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{opacity:.1;background-color:currentColor}.mat-button-focus-overlay[_ngcontent-%COMP%]{background:#000}.mat-stroked-button[_ngcontent-%COMP%]:not([disabled]){border-color:rgba(0,0,0,.12)}.mat-flat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{color:#212121;background-color:#fff}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-flat-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-stroked-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-flat-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-raised-button[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone[_ngcontent-%COMP%], .mat-button-toggle-group[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{box-shadow:none}.mat-button-toggle[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-button-toggle[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121;background:#fff}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:#000}.mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12)}[dir=rtl][_ngcontent-%COMP%]   .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-group-appearance-standard.mat-button-toggle-vertical[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:none;border-top:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#616161}.mat-button-toggle-checked.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121}.mat-button-toggle-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.26);background-color:#eee}.mat-button-toggle-disabled.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{background:#fff}.mat-button-toggle-disabled.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#bdbdbd}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{border:solid 1px rgba(0,0,0,.12)}.mat-card[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-card[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-card.mat-card-flat[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-card-subtitle[_ngcontent-%COMP%]{color:#616161}.mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#616161}.mat-checkbox-checkmark[_ngcontent-%COMP%]{fill:#fafafa}.mat-checkbox-checkmark-path[_ngcontent-%COMP%]{stroke:#fafafa !important}.mat-checkbox-mixedmark[_ngcontent-%COMP%]{background-color:#fafafa}.mat-checkbox-indeterminate.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f57c00}.mat-checkbox-indeterminate.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#ff9800}.mat-checkbox-indeterminate.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f44336}.mat-checkbox-disabled.mat-checkbox-checked[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-disabled.mat-checkbox-indeterminate[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]:not(.mat-checkbox-checked)   .mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{color:#616161}.mat-checkbox[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%]{background:#f57c00}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%]{background:#ff9800}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%]{background:#f44336}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#212121}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#212121;opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled):active{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled)   .mat-chip-remove[_ngcontent-%COMP%]:hover{opacity:.54}.mat-chip.mat-standard-chip.mat-chip-disabled[_ngcontent-%COMP%]{opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]::after{background:#000}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-table[_ngcontent-%COMP%]{background:#fff}.mat-table[_ngcontent-%COMP%]   thead[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tbody[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tfoot[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], [mat-header-row][_ngcontent-%COMP%], [mat-row][_ngcontent-%COMP%], [mat-footer-row][_ngcontent-%COMP%], .mat-table-sticky[_ngcontent-%COMP%]{background:inherit}mat-row[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], th.mat-header-cell[_ngcontent-%COMP%], td.mat-cell[_ngcontent-%COMP%], td.mat-footer-cell[_ngcontent-%COMP%]{border-bottom-color:rgba(0,0,0,.12)}.mat-header-cell[_ngcontent-%COMP%]{color:#616161}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{color:#212121}.mat-calendar-arrow[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.54)}.mat-datepicker-toggle[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-next-button[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-previous-button[_ngcontent-%COMP%]{color:rgba(0,0,0,.54)}.mat-calendar-table-header[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-calendar-table-header-divider[_ngcontent-%COMP%]::after{background:rgba(0,0,0,.12)}.mat-calendar-body-label[_ngcontent-%COMP%]{color:#616161}.mat-calendar-body-cell-content[_ngcontent-%COMP%]{color:#212121;border-color:transparent}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){color:#757575}.mat-calendar-body-cell[_ngcontent-%COMP%]:not(.mat-calendar-body-disabled):hover > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-keyboard-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-program-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){background-color:rgba(0,0,0,.04)}.mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.38)}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.18)}.mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.4)}.mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12);background-color:#fff;color:#212121}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.4)}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.4)}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content-touch[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-datepicker-toggle-active[_ngcontent-%COMP%]{color:#f57c00}.mat-datepicker-toggle-active.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-datepicker-toggle-active.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-dialog-container[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-divider[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-divider-vertical[_ngcontent-%COMP%]{border-right-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-expansion-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-action-row[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-keyboard-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-program-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded)   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover:not([aria-disabled=true]){background:rgba(0,0,0,.04)}@media(hover: none){.mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded):not([aria-disabled=true])   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover{background:#fff}}.mat-expansion-panel-header-title[_ngcontent-%COMP%]{color:#212121}.mat-expansion-panel-header-description[_ngcontent-%COMP%], .mat-expansion-indicator[_ngcontent-%COMP%]::after{color:#616161}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-title[_ngcontent-%COMP%], .mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-description[_ngcontent-%COMP%]{color:inherit}.mat-form-field-label[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-hint[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-focused[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-ripple[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid)   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f57c00}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-accent   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#ff9800}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-warn   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#f44336}.mat-error[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-hint[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-legacy.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-standard[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-standard.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.04)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.02)}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:transparent}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.12)}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:rgba(0,0,0,.87)}.mat-form-field-appearance-outline.mat-focused[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field-appearance-outline.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-appearance-outline.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-invalid.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.06)}.mat-icon.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-icon.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-icon.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-type-mat-native-select[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#616161}.mat-input-element[_ngcontent-%COMP%]:disabled, .mat-form-field-type-mat-native-select.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#757575}.mat-input-element[_ngcontent-%COMP%]{caret-color:#f57c00}.mat-input-element[_ngcontent-%COMP%]::placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-moz-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-webkit-input-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]:-ms-input-placeholder{color:rgba(97,97,97,.42)}.mat-accent[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%], .mat-form-field-invalid[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#f44336}.mat-form-field-type-mat-native-select.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{color:#616161}.mat-list-item-disabled[_ngcontent-%COMP%]{background-color:#eee}.mat-list-option[_ngcontent-%COMP%]:hover, .mat-list-option[_ngcontent-%COMP%]:focus, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.04)}.mat-list-single-selected-option[_ngcontent-%COMP%], .mat-list-single-selected-option[_ngcontent-%COMP%]:hover, .mat-list-single-selected-option[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.12)}.mat-menu-panel[_ngcontent-%COMP%]{background:#fff}.mat-menu-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-menu-item[_ngcontent-%COMP%]{background:transparent;color:#212121}.mat-menu-item[disabled][_ngcontent-%COMP%], .mat-menu-item[disabled][_ngcontent-%COMP%]::after{color:rgba(0,0,0,.38)}.mat-menu-item[_ngcontent-%COMP%]   .mat-icon-no-color[_ngcontent-%COMP%], .mat-menu-item-submenu-trigger[_ngcontent-%COMP%]::after{color:rgba(0,0,0,.54)}.mat-menu-item[_ngcontent-%COMP%]:hover:not([disabled]), .mat-menu-item.cdk-program-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item.cdk-keyboard-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item-highlighted[_ngcontent-%COMP%]:not([disabled]){background:rgba(0,0,0,.04)}.mat-paginator[_ngcontent-%COMP%]{background:#fff}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{color:#616161}.mat-paginator-decrement[_ngcontent-%COMP%], .mat-paginator-increment[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54);border-right:2px solid rgba(0,0,0,.54)}.mat-paginator-first[_ngcontent-%COMP%], .mat-paginator-last[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54)}.mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-decrement[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-increment[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-first[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-last[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f57c00}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#ff9800}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f44336}.mat-progress-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f57c00}.mat-progress-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#ff9800}.mat-progress-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f44336}.mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#616161}.mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f57c00}.mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#ff9800}.mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#ff9800}.mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f44336}.mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f44336}.mat-radio-button.mat-radio-disabled.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-label-content[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-radio-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-select-value[_ngcontent-%COMP%]{color:#212121}.mat-select-placeholder[_ngcontent-%COMP%]{color:rgba(97,97,97,.42)}.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%]{color:#757575}.mat-select-arrow[_ngcontent-%COMP%]{color:#616161}.mat-select-panel[_ngcontent-%COMP%]{background:#fff}.mat-select-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-select-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple){background:rgba(0,0,0,.12)}.mat-form-field.mat-focused.mat-primary[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-invalid[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#757575}.mat-drawer-container[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-drawer[_ngcontent-%COMP%]{background-color:#fff;color:#212121}.mat-drawer.mat-drawer-push[_ngcontent-%COMP%]{background-color:#fff}.mat-drawer[_ngcontent-%COMP%]:not(.mat-drawer-side){box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-drawer-side[_ngcontent-%COMP%]{border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-backdrop.mat-drawer-shown[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.6)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.54)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.54)}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.54)}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle[_ngcontent-%COMP%]:not(.mat-checked)   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-slide-toggle-thumb[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12);background-color:#fafafa}.mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-primary[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f57c00}.mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-accent[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#ff9800}.mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-warn[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f44336}.mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.2)}.mat-slider[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%], .cdk-focused[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-disabled[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]   .mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing)   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26);background-color:transparent}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26)}.mat-slider-has-ticks[_ngcontent-%COMP%]   .mat-slider-wrapper[_ngcontent-%COMP%]::after{border-color:rgba(0,0,0,.7)}.mat-slider-horizontal[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to right, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent);background-image:-moz-repeating-linear-gradient(0.0001deg, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-slider-vertical[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to bottom, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-step-header.cdk-keyboard-focused[_ngcontent-%COMP%], .mat-step-header.cdk-program-focused[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]:hover{background-color:rgba(0,0,0,.04)}@media(hover: none){.mat-step-header[_ngcontent-%COMP%]:hover{background:none}}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-optional[_ngcontent-%COMP%]{color:#616161}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon[_ngcontent-%COMP%]{background-color:#616161;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-selected[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-done[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-edit[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-error[_ngcontent-%COMP%]{background-color:transparent;color:#f44336}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-active[_ngcontent-%COMP%]{color:#212121}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-error[_ngcontent-%COMP%]{color:#f44336}.mat-stepper-horizontal[_ngcontent-%COMP%], .mat-stepper-vertical[_ngcontent-%COMP%]{background-color:#fff}.mat-stepper-vertical-line[_ngcontent-%COMP%]::before{border-left-color:rgba(0,0,0,.12)}.mat-horizontal-stepper-header[_ngcontent-%COMP%]::before, .mat-horizontal-stepper-header[_ngcontent-%COMP%]::after, .mat-stepper-horizontal-line[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-sort-header-arrow[_ngcontent-%COMP%]{color:#616161}.mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-header[_ngcontent-%COMP%]{border-bottom:1px solid rgba(0,0,0,.12)}.mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%]{border-top:1px solid rgba(0,0,0,.12);border-bottom:none}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{color:#212121}.mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:#757575}.mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#212121}.mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#757575}.mat-tab-group[class*=mat-background-][_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar[class*=mat-background-][_ngcontent-%COMP%]{border-bottom:none;border-top:none}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-toolbar[_ngcontent-%COMP%]{background:#f5f5f5;color:#212121}.mat-toolbar.mat-primary[_ngcontent-%COMP%]{background:#f57c00;color:#fff}.mat-toolbar.mat-accent[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-toolbar.mat-warn[_ngcontent-%COMP%]{background:#f44336;color:#fff}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:currentColor}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:inherit}.mat-toolbar[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:currentColor}.mat-tooltip[_ngcontent-%COMP%]{background:rgba(97,97,97,.9)}.mat-tree[_ngcontent-%COMP%]{background:#fff}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{color:#212121}.mat-snack-bar-container[_ngcontent-%COMP%]{color:rgba(255,255,255,.7);background:#323232;box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-simple-snackbar-action[_ngcontent-%COMP%]{color:#ff9800}[_nghost-%COMP%]{border-bottom:2px solid #ebebeb;display:flex;height:28px;align-items:flex-end;margin-top:8px}.annotations-header-containers[_ngcontent-%COMP%]{display:flex;font-size:.9em;font-weight:500;flex-grow:1}.header-column[_ngcontent-%COMP%]{flex:1 1}.header-container[_ngcontent-%COMP%]{display:inline-block}.header-clickable[_ngcontent-%COMP%]{cursor:pointer;display:flex;outline:none}.toggle-all-container[_ngcontent-%COMP%]{margin-left:10px;width:90px}.sort-icon[_ngcontent-%COMP%]{height:16px}"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/annotations_list/header/header_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class HeaderContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.selectedAnnotations$ = this.store.select(getSelectedAnnotations);
            this.annotationSort$ = this.store.select(getAnnotationSort);
        }
        /**
         * @param {?} newMetric
         * @return {?}
         */
        changeSort(newMetric) {
            this.store.dispatch(npmiChangeAnnotationSort({ metric: newMetric }));
        }
        /**
         * @param {?} checked
         * @return {?}
         */
        allAnnotationsToggled(checked) {
            if (checked) {
                this.store.dispatch(npmiSetSelectedAnnotations({
                    annotations: Object.keys(this.annotations),
                }));
            }
            else {
                this.store.dispatch(npmiSetSelectedAnnotations({ annotations: [] }));
            }
        }
    }
    HeaderContainer.decorators = [
        { type: Component, args: [{
                    selector: 'npmi-annotations-list-header',
                    template: `
    <npmi-annotations-list-header-component
      [numAnnotations]="numAnnotations"
      [selectedAnnotations]="selectedAnnotations$ | async"
      [sort]="annotationSort$ | async"
      [activeMetrics]="activeMetrics"
      (onChangeSort)="changeSort($event)"
      (onAllAnnotationsToggled)="allAnnotationsToggled($event)"
    ></npmi-annotations-list-header-component>
  `,
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    HeaderContainer.ctorParameters = () => [
        { type: Store }
    ];
    HeaderContainer.propDecorators = {
        numAnnotations: [{ type: Input }],
        annotations: [{ type: Input }],
        activeMetrics: [{ type: Input }]
    };
    /** @nocollapse */ HeaderContainer.ɵfac = function HeaderContainer_Factory(t) { return new (t || HeaderContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ HeaderContainer.ɵcmp = ɵɵdefineComponent({ type: HeaderContainer, selectors: [["npmi-annotations-list-header"]], inputs: { numAnnotations: "numAnnotations", annotations: "annotations", activeMetrics: "activeMetrics" }, decls: 3, vars: 8, consts: [[3, "numAnnotations", "selectedAnnotations", "sort", "activeMetrics", "onChangeSort", "onAllAnnotationsToggled"]], template: function HeaderContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "npmi-annotations-list-header-component", 0);
            ɵɵlistener("onChangeSort", function HeaderContainer_Template_npmi_annotations_list_header_component_onChangeSort_0_listener($event) { return ctx.changeSort($event); })("onAllAnnotationsToggled", function HeaderContainer_Template_npmi_annotations_list_header_component_onAllAnnotationsToggled_0_listener($event) { return ctx.allAnnotationsToggled($event); });
            ɵɵpipe(1, "async");
            ɵɵpipe(2, "async");
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵproperty("numAnnotations", ctx.numAnnotations)("selectedAnnotations", ɵɵpipeBind1(1, 4, ctx.selectedAnnotations$))("sort", ɵɵpipeBind1(2, 6, ctx.annotationSort$))("activeMetrics", ctx.activeMetrics);
        } }, directives: [HeaderComponent$1], pipes: [AsyncPipe], encapsulation: 2, changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/annotations_list/annotation/annotation_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    const _c0$A = ["chart"];
    const _c1$l = ["hintClip"];
    function AnnotationComponent_mat_icon_3_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "mat-icon", 10);
    } }
    function AnnotationComponent_mat_icon_4_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "mat-icon", 11);
    } }
    class AnnotationComponent {
        constructor() {
            this.selected = false;
            this.width = 10;
            this.chartWidth = 10;
            this.chartHeight = 10;
            this.maxDotRadius = 10;
            this.countDotOffset = 70;
            this.countTextPadding = 2;
            this.margin = { top: 0, right: 0, bottom: 0, left: 100 };
            this.strokeColor = '#fff';
            this.textClass = 'default-text';
            this.runs = [];
            this.colorScale = (/**
             * @return {?}
             */
            () => '#333333');
        }
        /**
         * @param {?} event
         * @return {?}
         */
        onResize(event) {
            this.redraw();
        }
        /**
         * @return {?}
         */
        ngAfterViewInit() {
            /** @type {?} */
            const runsColorScale = ((/** @type {?} */ (document.createElement('tf-color-scale')))).runsColorScale;
            this.colorScale = runsColorScale ? runsColorScale : this.colorScale;
            this.svg = select$1(this.annotationContainer.nativeElement).select('svg');
            this.xScale = point$1().padding(0);
            this.yScale = point$1().padding(0);
            this.sizeScale = linear$1().domain([0.0, 1.0]);
            this.countSizeScale = linear$1().range([2, this.maxDotRadius]);
            this.mainContainer = this.svg
                .append('g')
                .attr('transform', `translate(${this.margin.left}, ${this.margin.top})`);
            this.barsGroup = this.mainContainer.append('g');
            this.countDotsGroup = this.mainContainer.append('g');
            this.textsGroup = this.mainContainer.append('g');
            this.countTextsGroup = this.mainContainer.append('g');
            this.runHintGroup = this.svg.append('g');
            this.redraw();
        }
        /**
         * @param {?} changes
         * @return {?}
         */
        ngOnChanges(changes) {
            if (this.svg) {
                this.redraw();
            }
        }
        /**
         * @private
         * @return {?}
         */
        redraw() {
            this.selected = this.selectedAnnotations.includes(this.annotation);
            this.updateDimensions();
            this.setTextClass();
            this.updateAxes();
            this.draw();
        }
        // Initializing/Updating the visualization props.
        /**
         * @private
         * @return {?}
         */
        updateDimensions() {
            /** @type {?} */
            const runs = new Set();
            this.data.forEach((/**
             * @param {?} element
             * @return {?}
             */
            (element) => {
                runs.add(element.run);
            }));
            this.runs = [...runs];
            // Needs to be numActiveRuns * runHeight because of cdk-virtual-croll
            this.svg.style('height', `${this.numActiveRuns * this.runHeight}px`);
            // Needs to be runs.length * runHeight as this is used for y-pos scales
            this.chartHeight =
                this.runs.length * this.runHeight - this.margin.top - this.margin.bottom;
            this.width = this.annotationContainer.nativeElement.clientWidth || 10;
            this.chartWidth = this.width - this.margin.left - this.margin.right;
        }
        /**
         * @private
         * @return {?}
         */
        setTextClass() {
            this.textClass = 'default-text';
            if (this.flaggedAnnotations.includes(this.annotation)) {
                this.textClass = 'flag-text';
            }
            else if (this.hiddenAnnotations.includes(this.annotation)) {
                this.textClass = 'hidden-text';
            }
        }
        /**
         * @private
         * @return {?}
         */
        updateAxes() {
            this.xScale
                .rangeRound([
                0,
                this.chartWidth - this.chartWidth / this.activeMetrics.length,
            ])
                .domain(this.activeMetrics.map((/**
             * @param {?} d
             * @return {?}
             */
            (d) => stripMetricString(d))));
            this.yScale
                .rangeRound([0, this.chartHeight - this.runHeight])
                .domain(this.runs);
            this.sizeScale.range([0, this.chartWidth / this.activeMetrics.length]);
            this.countSizeScale.domain([0, this.maxCount]);
        }
        // Drawing UI
        /**
         * @private
         * @return {?}
         */
        draw() {
            this.drawRunIndicators();
            this.drawRunHintTexts();
            this.drawBars();
            this.drawTexts();
            if (this.showCounts) {
                this.drawCountDots();
                this.drawCountTexts();
            }
            else {
                this.countDotsGroup.selectAll('.count-dot').remove();
                this.countTextsGroup.selectAll('.count-background-text').remove();
                this.countTextsGroup.selectAll('.count-text').remove();
            }
        }
        /**
         * @private
         * @return {?}
         */
        drawRunIndicators() {
            select$1(this.clipPathElement.nativeElement)
                .select('rect')
                .attr('width', this.margin.left - 30)
                .attr('height', this.chartHeight);
            /** @type {?} */
            const indicators = this.runHintGroup
                .selectAll('.hint')
                .data(this.runs);
            /** @type {?} */
            const indicatorEnters = indicators
                .enter()
                .append('g')
                .attr('class', 'hint');
            indicatorEnters
                .append('path')
                .attr('d', 'M 0 0 L 15 0 L 10 10 L 15 20 L 0 20 Z');
            indicatorEnters
                .merge(indicators)
                .attr('transform', (/**
             * @this {?}
             * @param {?} d
             * @return {?}
             */
            function (d) {
                return `translate(10, ${(/** @type {?} */ (this.yScale(d))) + 5})`;
            }).bind(this))
                .attr('fill', (/**
             * @this {?}
             * @param {?} d
             * @return {?}
             */
            function (d) {
                return this.colorScale(d);
            }).bind(this));
            indicators.exit().remove();
        }
        /**
         * @private
         * @return {?}
         */
        drawRunHintTexts() {
            /** @type {?} */
            const hintTexts = this.runHintGroup
                .selectAll('.hint-text')
                .data(this.runs);
            /** @type {?} */
            const hintTextEnters = hintTexts
                .enter()
                .append('text')
                .attr('x', 25)
                .attr('font-size', '10px')
                .attr('alignment-baseline', 'middle')
                .attr('clip-path', 'url(#hint-clip)');
            hintTextEnters
                .merge(hintTexts)
                .attr('y', (/**
             * @this {?}
             * @param {?} d
             * @return {?}
             */
            function (d) {
                return (/** @type {?} */ (this.yScale(d))) + 15;
            }).bind(this))
                .attr('class', `hint-text ${this.textClass}`)
                .text((/**
             * @param {?} d
             * @return {?}
             */
            (d) => d));
            hintTexts.exit().remove();
        }
        /**
         * @private
         * @return {?}
         */
        drawBars() {
            /** @type {?} */
            const bars = this.barsGroup
                .selectAll('.bar')
                .data(this.data);
            /** @type {?} */
            const barEnters = bars
                .enter()
                .append('rect')
                .attr('class', 'bar')
                .attr('height', 20);
            barEnters
                .merge(bars)
                .attr('fill', (/**
             * @param {?} d
             * @return {?}
             */
            (d) => {
                if (d.nPMIValue === null) {
                    return '';
                }
                else if (d.nPMIValue >= 0) {
                    return Blues(d.nPMIValue);
                }
                else {
                    return Reds(d.nPMIValue * -1);
                }
            }))
                .attr('x', (/**
             * @this {?}
             * @param {?} d
             * @return {?}
             */
            function (d) {
                return (/** @type {?} */ (this.xScale(d.metric)));
            }).bind(this))
                .attr('y', (/**
             * @this {?}
             * @param {?} d
             * @return {?}
             */
            function (d) {
                return (/** @type {?} */ (this.yScale(d.run))) + 5;
            }).bind(this))
                .attr('width', (/**
             * @this {?}
             * @param {?} d
             * @return {?}
             */
            function (d) {
                if (d.nPMIValue === null) {
                    return 0;
                }
                else {
                    return this.sizeScale(Math.abs(d.nPMIValue));
                }
            }).bind(this));
            bars.exit().remove();
        }
        /**
         * @private
         * @return {?}
         */
        drawCountDots() {
            /** @type {?} */
            const countDots = this.countDotsGroup
                .selectAll('.count-dot')
                .data(this.data);
            /** @type {?} */
            const countDotEnters = countDots
                .enter()
                .append('circle')
                .attr('class', 'count-dot')
                .attr('stroke', 'black');
            countDotEnters
                .merge(countDots)
                .attr('fill', (/**
             * @this {?}
             * @param {?} d
             * @return {?}
             */
            function (d) {
                if (d.countValue === null) {
                    return '';
                }
                else {
                    return Greys(d.countValue / this.maxCount);
                }
            }).bind(this))
                .attr('cx', (/**
             * @this {?}
             * @param {?} d
             * @return {?}
             */
            function (d) {
                return (/** @type {?} */ (this.xScale(d.metric))) + this.countDotOffset;
            }).bind(this))
                .attr('cy', (/**
             * @this {?}
             * @param {?} d
             * @return {?}
             */
            function (d) {
                return (/** @type {?} */ (this.yScale(d.run))) + this.runHeight / 2.0;
            }).bind(this))
                .attr('r', (/**
             * @this {?}
             * @param {?} d
             * @return {?}
             */
            function (d) {
                if (d.countValue === null) {
                    return 0;
                }
                else {
                    return this.countSizeScale(d.countValue);
                }
            }).bind(this));
            countDots.exit().remove();
        }
        /**
         * @private
         * @return {?}
         */
        drawTexts() {
            /** @type {?} */
            const backgroundTexts = this.textsGroup
                .selectAll('.npmi-background-text')
                .data(this.data);
            /** @type {?} */
            const backgroundTextEnters = backgroundTexts
                .enter()
                .append('text')
                .attr('class', 'npmi-background-text')
                .attr('stroke-width', 3)
                .attr('stroke-linejoin', 'round')
                .attr('stroke', this.strokeColor)
                .attr('font-size', '13px')
                .attr('alignment-baseline', 'middle');
            backgroundTextEnters
                .merge(backgroundTexts)
                .attr('x', (/**
             * @this {?}
             * @param {?} d
             * @return {?}
             */
            function (d) {
                return (/** @type {?} */ (this.xScale(d.metric))) + 5;
            }).bind(this))
                .attr('y', (/**
             * @this {?}
             * @param {?} d
             * @return {?}
             */
            function (d) {
                return (/** @type {?} */ (this.yScale(d.run))) + this.runHeight / 2.0;
            }).bind(this))
                .text((/**
             * @param {?} d
             * @return {?}
             */
            (d) => {
                /** @type {?} */
                let value = d.nPMIValue === null
                    ? 'null'
                    : Math.round((d.nPMIValue + Number.EPSILON) * 1000) / 1000;
                return value;
            }));
            backgroundTexts.exit().remove();
            /** @type {?} */
            const texts = this.textsGroup
                .selectAll('.npmi-text')
                .data(this.data);
            /** @type {?} */
            const textEnters = texts
                .enter()
                .append('text')
                .attr('class', 'npmi-text')
                .attr('font-size', '13px')
                .attr('alignment-baseline', 'middle');
            textEnters
                .merge(texts)
                .attr('x', (/**
             * @this {?}
             * @param {?} d
             * @return {?}
             */
            function (d) {
                return (/** @type {?} */ (this.xScale(d.metric))) + 5;
            }).bind(this))
                .attr('y', (/**
             * @this {?}
             * @param {?} d
             * @return {?}
             */
            function (d) {
                return (/** @type {?} */ (this.yScale(d.run))) + this.runHeight / 2.0;
            }).bind(this))
                .text((/**
             * @param {?} d
             * @return {?}
             */
            (d) => {
                /** @type {?} */
                let value = d.nPMIValue === null
                    ? 'null'
                    : Math.round((d.nPMIValue + Number.EPSILON) * 1000) / 1000;
                return value;
            }));
            texts.exit().remove();
        }
        /**
         * @private
         * @return {?}
         */
        drawCountTexts() {
            /** @type {?} */
            const countBackgroundTexts = this.countTextsGroup
                .selectAll('.count-background-text')
                .data(this.data);
            /** @type {?} */
            const countBackgroundTextEnters = countBackgroundTexts
                .enter()
                .append('text')
                .attr('class', 'count-background-text')
                .attr('stroke-width', 3)
                .attr('stroke-linejoin', 'round')
                .attr('stroke', this.strokeColor)
                .attr('font-size', '10px')
                .attr('alignment-baseline', 'middle');
            countBackgroundTextEnters
                .merge(countBackgroundTexts)
                .attr('x', (/**
             * @this {?}
             * @param {?} d
             * @return {?}
             */
            function (d) {
                return ((/** @type {?} */ (this.xScale(d.metric))) +
                    this.countDotOffset +
                    this.countTextPadding +
                    this.maxDotRadius);
            }).bind(this))
                .attr('y', (/**
             * @this {?}
             * @param {?} d
             * @return {?}
             */
            function (d) {
                return (/** @type {?} */ (this.yScale(d.run))) + this.runHeight / 2.0;
            }).bind(this))
                .text((/**
             * @param {?} d
             * @return {?}
             */
            (d) => {
                if (d.countValue === null) {
                    return '';
                }
                return Intl.NumberFormat().format(d.countValue);
            }));
            countBackgroundTexts.exit().remove();
            /** @type {?} */
            const countTexts = this.countTextsGroup
                .selectAll('.count-text')
                .data(this.data);
            /** @type {?} */
            const countTextEnters = countTexts
                .enter()
                .append('text')
                .attr('class', 'count-text')
                .attr('font-size', '10px')
                .attr('alignment-baseline', 'middle');
            countTextEnters
                .merge(countTexts)
                .attr('x', (/**
             * @this {?}
             * @param {?} d
             * @return {?}
             */
            function (d) {
                return ((/** @type {?} */ (this.xScale(d.metric))) +
                    this.countDotOffset +
                    this.countTextPadding +
                    this.maxDotRadius);
            }).bind(this))
                .attr('y', (/**
             * @this {?}
             * @param {?} d
             * @return {?}
             */
            function (d) {
                return (/** @type {?} */ (this.yScale(d.run))) + this.runHeight / 2.0;
            }).bind(this))
                .text((/**
             * @param {?} d
             * @return {?}
             */
            (d) => {
                if (d.countValue === null) {
                    return '';
                }
                return Intl.NumberFormat().format(d.countValue);
            }));
            countTexts.exit().remove();
        }
    }
    AnnotationComponent.decorators = [
        { type: Component, args: [{
                    selector: 'annotation-component',
                    templateUrl: './annotation_component.ng.html',
                    styleUrls: ['./annotation_component.css'],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                    encapsulation: ViewEncapsulation$1.None,
                },] },
    ];
    AnnotationComponent.propDecorators = {
        data: [{ type: Input }],
        maxCount: [{ type: Input }],
        selectedAnnotations: [{ type: Input }],
        flaggedAnnotations: [{ type: Input }],
        hiddenAnnotations: [{ type: Input }],
        activeMetrics: [{ type: Input }],
        numActiveRuns: [{ type: Input }],
        showCounts: [{ type: Input }],
        annotation: [{ type: Input }],
        runHeight: [{ type: Input }],
        sidebarWidth: [{ type: Input }],
        annotationContainer: [{ type: ViewChild, args: ['chart', { static: true, read: ElementRef },] }],
        clipPathElement: [{ type: ViewChild, args: ['hintClip', { static: true, read: ElementRef },] }],
        selected: [{ type: HostBinding, args: ['class.selected-row',] }],
        onResize: [{ type: HostListener, args: ['window:resize',] }]
    };
    /** @nocollapse */ AnnotationComponent.ɵfac = function AnnotationComponent_Factory(t) { return new (t || AnnotationComponent)(); };
    /** @nocollapse */ AnnotationComponent.ɵcmp = ɵɵdefineComponent({ type: AnnotationComponent, selectors: [["annotation-component"]], viewQuery: function AnnotationComponent_Query(rf, ctx) { if (rf & 1) {
            ɵɵstaticViewQuery(_c0$A, true, ElementRef);
            ɵɵstaticViewQuery(_c1$l, true, ElementRef);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.annotationContainer = _t.first);
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.clipPathElement = _t.first);
        } }, hostVars: 2, hostBindings: function AnnotationComponent_HostBindings(rf, ctx) { if (rf & 1) {
            ɵɵlistener("resize", function AnnotationComponent_resize_HostBindingHandler() { return ctx.onResize(); }, false, ɵɵresolveWindow);
        } if (rf & 2) {
            ɵɵclassProp("selected-row", ctx.selected);
        } }, inputs: { data: "data", maxCount: "maxCount", selectedAnnotations: "selectedAnnotations", flaggedAnnotations: "flaggedAnnotations", hiddenAnnotations: "hiddenAnnotations", activeMetrics: "activeMetrics", numActiveRuns: "numActiveRuns", showCounts: "showCounts", annotation: "annotation", runHeight: "runHeight", sidebarWidth: "sidebarWidth" }, features: [ɵɵNgOnChangesFeature], decls: 12, vars: 8, consts: [[1, "annotation-title"], [1, "annotation-checkbox", 3, "checked", "click"], ["class", "flagged-icon", "svgIcon", "flag_24px", 4, "ngIf"], ["class", "hidden-icon", "svgIcon", "visibility_off_24px", 4, "ngIf"], [1, "chart-div"], ["chart", ""], [1, "chart-svg"], ["id", "hint-clip"], ["hintClip", ""], ["x", "0", "y", "0"], ["svgIcon", "flag_24px", 1, "flagged-icon"], ["svgIcon", "visibility_off_24px", 1, "hidden-icon"]], template: function AnnotationComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "div", 0);
            ɵɵelementStart(1, "mat-checkbox", 1);
            ɵɵlistener("click", function AnnotationComponent_Template_mat_checkbox_click_1_listener($event) { return $event.preventDefault(); });
            ɵɵelementEnd();
            ɵɵtext(2);
            ɵɵtemplate(3, AnnotationComponent_mat_icon_3_Template, 1, 0, "mat-icon", 2);
            ɵɵtemplate(4, AnnotationComponent_mat_icon_4_Template, 1, 0, "mat-icon", 3);
            ɵɵelementEnd();
            ɵɵelementStart(5, "div", 4, 5);
            ɵɵnamespaceSVG();
            ɵɵelementStart(7, "svg", 6);
            ɵɵelementStart(8, "defs");
            ɵɵelementStart(9, "clipPath", 7, 8);
            ɵɵelement(11, "rect", 9);
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵclassProp("flagged-annotation", ctx.flaggedAnnotations.includes(ctx.annotation))("hidden-annotation", ctx.hiddenAnnotations.includes(ctx.annotation) && !ctx.flaggedAnnotations.includes(ctx.annotation));
            ɵɵadvance(1);
            ɵɵproperty("checked", ctx.selectedAnnotations.includes(ctx.annotation));
            ɵɵadvance(1);
            ɵɵtextInterpolate1(" ", ctx.annotation, " ");
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx.flaggedAnnotations.includes(ctx.annotation));
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx.hiddenAnnotations.includes(ctx.annotation));
        } }, directives: [MatCheckbox, NgIf, MatIcon], styles: [".mat-badge-content{font-weight:600;font-size:12px;font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-badge-small .mat-badge-content{font-size:9px}.mat-badge-large .mat-badge-content{font-size:24px}.mat-h1,.mat-headline,.mat-typography h1{font:400 24px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h2,.mat-title,.mat-typography h2{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h3,.mat-subheading-2,.mat-typography h3{font:400 16px/28px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h4,.mat-subheading-1,.mat-typography h4{font:400 15px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h5,.mat-typography h5{font:400 calc(14px * 0.83)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-h6,.mat-typography h6{font:400 calc(14px * 0.67)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-body-strong,.mat-body-2{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body,.mat-body-1,.mat-typography{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body p,.mat-body-1 p,.mat-typography p{margin:0 0 12px}.mat-small,.mat-caption{font:400 12px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-display-4,.mat-typography .mat-display-4{font:300 112px/112px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.05em;margin:0 0 56px}.mat-display-3,.mat-typography .mat-display-3{font:400 56px/56px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.02em;margin:0 0 64px}.mat-display-2,.mat-typography .mat-display-2{font:400 45px/48px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.005em;margin:0 0 64px}.mat-display-1,.mat-typography .mat-display-1{font:400 34px/40px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 64px}.mat-bottom-sheet-container{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-button,.mat-raised-button,.mat-icon-button,.mat-stroked-button,.mat-flat-button,.mat-fab,.mat-mini-fab{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-button-toggle{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card-title{font-size:24px;font-weight:500}.mat-card-header .mat-card-title{font-size:20px}.mat-card-subtitle,.mat-card-content{font-size:14px}.mat-checkbox{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-checkbox-layout .mat-checkbox-label{line-height:24px}.mat-chip{font-size:14px;font-weight:500}.mat-chip .mat-chip-trailing-icon.mat-icon,.mat-chip .mat-chip-remove.mat-icon{font-size:18px}.mat-table{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-header-cell{font-size:12px;font-weight:500}.mat-cell,.mat-footer-cell{font-size:14px}.mat-calendar{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-calendar-body{font-size:13px}.mat-calendar-body-label,.mat-calendar-period-button{font-size:14px;font-weight:500}.mat-calendar-table-header th{font-size:11px;font-weight:400}.mat-dialog-title{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-expansion-panel-header{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:15px;font-weight:400}.mat-expansion-panel-content{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field{font-size:inherit;font-weight:400;line-height:1.125;font-family:Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field-wrapper{padding-bottom:1.34375em}.mat-form-field-prefix .mat-icon,.mat-form-field-suffix .mat-icon{font-size:150%;line-height:1.125}.mat-form-field-prefix .mat-icon-button,.mat-form-field-suffix .mat-icon-button{height:1.5em;width:1.5em}.mat-form-field-prefix .mat-icon-button .mat-icon,.mat-form-field-suffix .mat-icon-button .mat-icon{height:1.125em;line-height:1.125}.mat-form-field-infix{padding:.5em 0;border-top:.84375em solid transparent}.mat-form-field-can-float.mat-form-field-should-float .mat-form-field-label,.mat-form-field-can-float .mat-input-server:focus+.mat-form-field-label-wrapper .mat-form-field-label{transform:translateY(-1.34375em) scale(0.75);width:133.3333333333%}.mat-form-field-can-float .mat-input-server[label]:not(:label-shown)+.mat-form-field-label-wrapper .mat-form-field-label{transform:translateY(-1.34374em) scale(0.75);width:133.3333433333%}.mat-form-field-label-wrapper{top:-0.84375em;padding-top:.84375em}.mat-form-field-label{top:1.34375em}.mat-form-field-underline{bottom:1.34375em}.mat-form-field-subscript-wrapper{font-size:75%;margin-top:.6666666667em;top:calc(100% - 1.7916666667em)}.mat-form-field-appearance-legacy .mat-form-field-wrapper{padding-bottom:1.25em}.mat-form-field-appearance-legacy .mat-form-field-infix{padding:.4375em 0}.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float .mat-form-field-label,.mat-form-field-appearance-legacy.mat-form-field-can-float .mat-input-server:focus+.mat-form-field-label-wrapper .mat-form-field-label{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.001px);-ms-transform:translateY(-1.28125em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-legacy.mat-form-field-can-float .mat-form-field-autofill-control:-webkit-autofill+.mat-form-field-label-wrapper .mat-form-field-label{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00101px);-ms-transform:translateY(-1.28124em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-legacy.mat-form-field-can-float .mat-input-server[label]:not(:label-shown)+.mat-form-field-label-wrapper .mat-form-field-label{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00102px);-ms-transform:translateY(-1.28123em) scale(0.75);width:133.3333533333%}.mat-form-field-appearance-legacy .mat-form-field-label{top:1.28125em}.mat-form-field-appearance-legacy .mat-form-field-underline{bottom:1.25em}.mat-form-field-appearance-legacy .mat-form-field-subscript-wrapper{margin-top:.5416666667em;top:calc(100% - 1.6666666667em)}@media print{.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float .mat-form-field-label,.mat-form-field-appearance-legacy.mat-form-field-can-float .mat-input-server:focus+.mat-form-field-label-wrapper .mat-form-field-label{transform:translateY(-1.28122em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float .mat-form-field-autofill-control:-webkit-autofill+.mat-form-field-label-wrapper .mat-form-field-label{transform:translateY(-1.28121em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float .mat-input-server[label]:not(:label-shown)+.mat-form-field-label-wrapper .mat-form-field-label{transform:translateY(-1.2812em) scale(0.75)}}.mat-form-field-appearance-fill .mat-form-field-infix{padding:.25em 0 .75em 0}.mat-form-field-appearance-fill .mat-form-field-label{top:1.09375em;margin-top:-0.5em}.mat-form-field-appearance-fill.mat-form-field-can-float.mat-form-field-should-float .mat-form-field-label,.mat-form-field-appearance-fill.mat-form-field-can-float .mat-input-server:focus+.mat-form-field-label-wrapper .mat-form-field-label{transform:translateY(-0.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-fill.mat-form-field-can-float .mat-input-server[label]:not(:label-shown)+.mat-form-field-label-wrapper .mat-form-field-label{transform:translateY(-0.59374em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-outline .mat-form-field-infix{padding:1em 0 1em 0}.mat-form-field-appearance-outline .mat-form-field-label{top:1.84375em;margin-top:-0.25em}.mat-form-field-appearance-outline.mat-form-field-can-float.mat-form-field-should-float .mat-form-field-label,.mat-form-field-appearance-outline.mat-form-field-can-float .mat-input-server:focus+.mat-form-field-label-wrapper .mat-form-field-label{transform:translateY(-1.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-outline.mat-form-field-can-float .mat-input-server[label]:not(:label-shown)+.mat-form-field-label-wrapper .mat-form-field-label{transform:translateY(-1.59374em) scale(0.75);width:133.3333433333%}.mat-grid-tile-header,.mat-grid-tile-footer{font-size:14px}.mat-grid-tile-header .mat-line,.mat-grid-tile-footer .mat-line{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-grid-tile-header .mat-line:nth-child(n+2),.mat-grid-tile-footer .mat-line:nth-child(n+2){font-size:12px}input.mat-input-element{margin-top:-0.0625em}.mat-menu-item{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:400}.mat-paginator,.mat-paginator-page-size .mat-select-trigger{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px}.mat-radio-button{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select-trigger{height:1.125em}.mat-slide-toggle-content{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-slider-thumb-label-text{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-stepper-vertical,.mat-stepper-horizontal{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-step-label{font-size:14px;font-weight:400}.mat-step-sub-label-error{font-weight:normal}.mat-step-label-error{font-size:14px}.mat-step-label-selected{font-size:14px;font-weight:500}.mat-tab-group{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tab-label,.mat-tab-link{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-toolbar,.mat-toolbar h1,.mat-toolbar h2,.mat-toolbar h3,.mat-toolbar h4,.mat-toolbar h5,.mat-toolbar h6{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0}.mat-tooltip{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:10px;padding-top:6px;padding-bottom:6px}.mat-tooltip-handset{font-size:14px;padding-top:8px;padding-bottom:8px}.mat-list-item{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-option{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-base .mat-list-item{font-size:16px}.mat-list-base .mat-list-item .mat-line{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base .mat-list-item .mat-line:nth-child(n+2){font-size:14px}.mat-list-base .mat-list-option{font-size:16px}.mat-list-base .mat-list-option .mat-line{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base .mat-list-option .mat-line:nth-child(n+2){font-size:14px}.mat-list-base .mat-subheader{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-list-base[dense] .mat-list-item{font-size:12px}.mat-list-base[dense] .mat-list-item .mat-line{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense] .mat-list-item .mat-line:nth-child(n+2){font-size:12px}.mat-list-base[dense] .mat-list-option{font-size:12px}.mat-list-base[dense] .mat-list-option .mat-line{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense] .mat-list-option .mat-line:nth-child(n+2){font-size:12px}.mat-list-base[dense] .mat-subheader{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-option{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:16px}.mat-optgroup-label{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-simple-snackbar{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px}.mat-simple-snackbar-action{line-height:1;font-family:inherit;font-size:inherit;font-weight:500}.mat-tree{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tree-node,.mat-nested-tree-node{font-weight:400;font-size:14px}.mat-ripple{overflow:hidden;position:relative}.mat-ripple:not(:empty){transform:translateZ(0)}.mat-ripple.mat-ripple-unbounded{overflow:visible}.mat-ripple-element{position:absolute;border-radius:50%;pointer-events:none;transition:opacity,transform 0ms cubic-bezier(0, 0, 0.2, 1);transform:scale(0)}.cdk-high-contrast-active .mat-ripple-element{display:none}.cdk-visually-hidden{border:0;clip:rect(0 0 0 0);height:1px;margin:-1px;overflow:hidden;padding:0;position:absolute;width:1px;outline:0;-webkit-appearance:none;-moz-appearance:none}.cdk-overlay-container,.cdk-global-overlay-wrapper{pointer-events:none;top:0;left:0;height:100%;width:100%}.cdk-overlay-container{position:fixed;z-index:1000}.cdk-overlay-container:empty{display:none}.cdk-global-overlay-wrapper{display:flex;position:absolute;z-index:1000}.cdk-overlay-pane{position:absolute;pointer-events:auto;box-sizing:border-box;z-index:1000;display:flex;max-width:100%;max-height:100%}.cdk-overlay-backdrop{position:absolute;top:0;bottom:0;left:0;right:0;z-index:1000;pointer-events:auto;-webkit-tap-highlight-color:transparent;transition:opacity 400ms cubic-bezier(0.25, 0.8, 0.25, 1);opacity:0}.cdk-overlay-backdrop.cdk-overlay-backdrop-showing{opacity:1}@media screen and (-ms-high-contrast: active){.cdk-overlay-backdrop.cdk-overlay-backdrop-showing{opacity:.6}}.cdk-overlay-dark-backdrop{background:rgba(0,0,0,.32)}.cdk-overlay-transparent-backdrop,.cdk-overlay-transparent-backdrop.cdk-overlay-backdrop-showing{opacity:0}.cdk-overlay-connected-position-bounding-box{position:absolute;z-index:1000;display:flex;flex-direction:column;min-width:1px;min-height:1px}.cdk-global-scrollblock{position:fixed;width:100%;overflow-y:scroll}@keyframes cdk-text-field-autofill-start{/*!*/}@keyframes cdk-text-field-autofill-end{/*!*/}.cdk-text-field-autofill-monitored:-webkit-autofill{animation:cdk-text-field-autofill-start 0s 1ms}.cdk-text-field-autofill-monitored:not(:-webkit-autofill){animation:cdk-text-field-autofill-end 0s 1ms}textarea.cdk-textarea-autosize{resize:none}textarea.cdk-textarea-autosize-measuring{padding:2px 0 !important;box-sizing:content-box !important;height:auto !important;overflow:hidden !important}textarea.cdk-textarea-autosize-measuring-firefox{padding:2px 0 !important;box-sizing:content-box !important;height:0 !important}.mat-ripple-element{background-color:rgba(0,0,0,.1)}.mat-option{color:#212121}.mat-option:hover:not(.mat-option-disabled),.mat-option:focus:not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-selected:not(.mat-option-multiple):not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-active{background:rgba(0,0,0,.04);color:#212121}.mat-option.mat-option-disabled{color:rgba(0,0,0,.38)}.mat-primary .mat-option.mat-selected:not(.mat-option-disabled){color:#f57c00}.mat-accent .mat-option.mat-selected:not(.mat-option-disabled){color:#ff9800}.mat-warn .mat-option.mat-selected:not(.mat-option-disabled){color:#f44336}.mat-optgroup-label{color:#616161}.mat-optgroup-disabled .mat-optgroup-label{color:rgba(0,0,0,.38)}.mat-pseudo-checkbox{color:#616161}.mat-pseudo-checkbox::after{color:#fafafa}.mat-pseudo-checkbox-disabled{color:#b0b0b0}.mat-primary .mat-pseudo-checkbox-checked,.mat-primary .mat-pseudo-checkbox-indeterminate{background:#f57c00}.mat-pseudo-checkbox-checked,.mat-pseudo-checkbox-indeterminate,.mat-accent .mat-pseudo-checkbox-checked,.mat-accent .mat-pseudo-checkbox-indeterminate{background:#ff9800}.mat-warn .mat-pseudo-checkbox-checked,.mat-warn .mat-pseudo-checkbox-indeterminate{background:#f44336}.mat-pseudo-checkbox-checked.mat-pseudo-checkbox-disabled,.mat-pseudo-checkbox-indeterminate.mat-pseudo-checkbox-disabled{background:#b0b0b0}.mat-elevation-z0{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z1{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z2{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z3{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z4{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z5{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 5px 8px 0px rgba(0, 0, 0, 0.14),0px 1px 14px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z6{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z7{box-shadow:0px 4px 5px -2px rgba(0, 0, 0, 0.2),0px 7px 10px 1px rgba(0, 0, 0, 0.14),0px 2px 16px 1px rgba(0, 0, 0, 0.12)}.mat-elevation-z8{box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z9{box-shadow:0px 5px 6px -3px rgba(0, 0, 0, 0.2),0px 9px 12px 1px rgba(0, 0, 0, 0.14),0px 3px 16px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z10{box-shadow:0px 6px 6px -3px rgba(0, 0, 0, 0.2),0px 10px 14px 1px rgba(0, 0, 0, 0.14),0px 4px 18px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z11{box-shadow:0px 6px 7px -4px rgba(0, 0, 0, 0.2),0px 11px 15px 1px rgba(0, 0, 0, 0.14),0px 4px 20px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z12{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z13{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 13px 19px 2px rgba(0, 0, 0, 0.14),0px 5px 24px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z14{box-shadow:0px 7px 9px -4px rgba(0, 0, 0, 0.2),0px 14px 21px 2px rgba(0, 0, 0, 0.14),0px 5px 26px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z15{box-shadow:0px 8px 9px -5px rgba(0, 0, 0, 0.2),0px 15px 22px 2px rgba(0, 0, 0, 0.14),0px 6px 28px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z16{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z17{box-shadow:0px 8px 11px -5px rgba(0, 0, 0, 0.2),0px 17px 26px 2px rgba(0, 0, 0, 0.14),0px 6px 32px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z18{box-shadow:0px 9px 11px -5px rgba(0, 0, 0, 0.2),0px 18px 28px 2px rgba(0, 0, 0, 0.14),0px 7px 34px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z19{box-shadow:0px 9px 12px -6px rgba(0, 0, 0, 0.2),0px 19px 29px 2px rgba(0, 0, 0, 0.14),0px 7px 36px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z20{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 20px 31px 3px rgba(0, 0, 0, 0.14),0px 8px 38px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z21{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 21px 33px 3px rgba(0, 0, 0, 0.14),0px 8px 40px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z22{box-shadow:0px 10px 14px -6px rgba(0, 0, 0, 0.2),0px 22px 35px 3px rgba(0, 0, 0, 0.14),0px 8px 42px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z23{box-shadow:0px 11px 14px -7px rgba(0, 0, 0, 0.2),0px 23px 36px 3px rgba(0, 0, 0, 0.14),0px 9px 44px 8px rgba(0, 0, 0, 0.12)}.mat-elevation-z24{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12)}.mat-app-background{background-color:#fafafa;color:#212121}.mat-theme-loaded-marker{display:none}.mat-autocomplete-panel{background:#fff;color:#212121}.mat-autocomplete-panel:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-autocomplete-panel .mat-option.mat-selected:not(.mat-active):not(:hover){background:#fff}.mat-autocomplete-panel .mat-option.mat-selected:not(.mat-active):not(:hover):not(.mat-option-disabled){color:#212121}.mat-badge-content{color:#fff;background:#f57c00}.cdk-high-contrast-active .mat-badge-content{outline:solid 1px;border-radius:0}.mat-badge-accent .mat-badge-content{background:#ff9800;color:#fff}.mat-badge-warn .mat-badge-content{color:#fff;background:#f44336}.mat-badge{position:relative}.mat-badge-hidden .mat-badge-content{display:none}.mat-badge-disabled .mat-badge-content{background:#b9b9b9;color:#757575}.mat-badge-content{position:absolute;text-align:center;display:inline-block;border-radius:50%;transition:transform 200ms ease-in-out;transform:scale(0.6);overflow:hidden;white-space:nowrap;text-overflow:ellipsis;pointer-events:none}.ng-animate-disabled .mat-badge-content,.mat-badge-content._mat-animation-noopable{transition:none}.mat-badge-content.mat-badge-active{transform:none}.mat-badge-small .mat-badge-content{width:16px;height:16px;line-height:16px}.mat-badge-small.mat-badge-above .mat-badge-content{top:-8px}.mat-badge-small.mat-badge-below .mat-badge-content{bottom:-8px}.mat-badge-small.mat-badge-before .mat-badge-content{left:-16px}[dir=rtl] .mat-badge-small.mat-badge-before .mat-badge-content{left:auto;right:-16px}.mat-badge-small.mat-badge-after .mat-badge-content{right:-16px}[dir=rtl] .mat-badge-small.mat-badge-after .mat-badge-content{right:auto;left:-16px}.mat-badge-small.mat-badge-overlap.mat-badge-before .mat-badge-content{left:-8px}[dir=rtl] .mat-badge-small.mat-badge-overlap.mat-badge-before .mat-badge-content{left:auto;right:-8px}.mat-badge-small.mat-badge-overlap.mat-badge-after .mat-badge-content{right:-8px}[dir=rtl] .mat-badge-small.mat-badge-overlap.mat-badge-after .mat-badge-content{right:auto;left:-8px}.mat-badge-medium .mat-badge-content{width:22px;height:22px;line-height:22px}.mat-badge-medium.mat-badge-above .mat-badge-content{top:-11px}.mat-badge-medium.mat-badge-below .mat-badge-content{bottom:-11px}.mat-badge-medium.mat-badge-before .mat-badge-content{left:-22px}[dir=rtl] .mat-badge-medium.mat-badge-before .mat-badge-content{left:auto;right:-22px}.mat-badge-medium.mat-badge-after .mat-badge-content{right:-22px}[dir=rtl] .mat-badge-medium.mat-badge-after .mat-badge-content{right:auto;left:-22px}.mat-badge-medium.mat-badge-overlap.mat-badge-before .mat-badge-content{left:-11px}[dir=rtl] .mat-badge-medium.mat-badge-overlap.mat-badge-before .mat-badge-content{left:auto;right:-11px}.mat-badge-medium.mat-badge-overlap.mat-badge-after .mat-badge-content{right:-11px}[dir=rtl] .mat-badge-medium.mat-badge-overlap.mat-badge-after .mat-badge-content{right:auto;left:-11px}.mat-badge-large .mat-badge-content{width:28px;height:28px;line-height:28px}.mat-badge-large.mat-badge-above .mat-badge-content{top:-14px}.mat-badge-large.mat-badge-below .mat-badge-content{bottom:-14px}.mat-badge-large.mat-badge-before .mat-badge-content{left:-28px}[dir=rtl] .mat-badge-large.mat-badge-before .mat-badge-content{left:auto;right:-28px}.mat-badge-large.mat-badge-after .mat-badge-content{right:-28px}[dir=rtl] .mat-badge-large.mat-badge-after .mat-badge-content{right:auto;left:-28px}.mat-badge-large.mat-badge-overlap.mat-badge-before .mat-badge-content{left:-14px}[dir=rtl] .mat-badge-large.mat-badge-overlap.mat-badge-before .mat-badge-content{left:auto;right:-14px}.mat-badge-large.mat-badge-overlap.mat-badge-after .mat-badge-content{right:-14px}[dir=rtl] .mat-badge-large.mat-badge-overlap.mat-badge-after .mat-badge-content{right:auto;left:-14px}.mat-bottom-sheet-container{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-button,.mat-icon-button,.mat-stroked-button{color:inherit;background:transparent}.mat-button.mat-primary,.mat-icon-button.mat-primary,.mat-stroked-button.mat-primary{color:#f57c00}.mat-button.mat-accent,.mat-icon-button.mat-accent,.mat-stroked-button.mat-accent{color:#ff9800}.mat-button.mat-warn,.mat-icon-button.mat-warn,.mat-stroked-button.mat-warn{color:#f44336}.mat-button.mat-primary[disabled],.mat-button.mat-accent[disabled],.mat-button.mat-warn[disabled],.mat-button[disabled][disabled],.mat-icon-button.mat-primary[disabled],.mat-icon-button.mat-accent[disabled],.mat-icon-button.mat-warn[disabled],.mat-icon-button[disabled][disabled],.mat-stroked-button.mat-primary[disabled],.mat-stroked-button.mat-accent[disabled],.mat-stroked-button.mat-warn[disabled],.mat-stroked-button[disabled][disabled]{color:rgba(0,0,0,.26)}.mat-button.mat-primary .mat-button-focus-overlay,.mat-icon-button.mat-primary .mat-button-focus-overlay,.mat-stroked-button.mat-primary .mat-button-focus-overlay{background-color:#f57c00}.mat-button.mat-accent .mat-button-focus-overlay,.mat-icon-button.mat-accent .mat-button-focus-overlay,.mat-stroked-button.mat-accent .mat-button-focus-overlay{background-color:#ff9800}.mat-button.mat-warn .mat-button-focus-overlay,.mat-icon-button.mat-warn .mat-button-focus-overlay,.mat-stroked-button.mat-warn .mat-button-focus-overlay{background-color:#f44336}.mat-button[disabled] .mat-button-focus-overlay,.mat-icon-button[disabled] .mat-button-focus-overlay,.mat-stroked-button[disabled] .mat-button-focus-overlay{background-color:transparent}.mat-button .mat-ripple-element,.mat-icon-button .mat-ripple-element,.mat-stroked-button .mat-ripple-element{opacity:.1;background-color:currentColor}.mat-button-focus-overlay{background:#000}.mat-stroked-button:not([disabled]){border-color:rgba(0,0,0,.12)}.mat-flat-button,.mat-raised-button,.mat-fab,.mat-mini-fab{color:#212121;background-color:#fff}.mat-flat-button.mat-primary,.mat-raised-button.mat-primary,.mat-fab.mat-primary,.mat-mini-fab.mat-primary{color:#fff}.mat-flat-button.mat-accent,.mat-raised-button.mat-accent,.mat-fab.mat-accent,.mat-mini-fab.mat-accent{color:#fff}.mat-flat-button.mat-warn,.mat-raised-button.mat-warn,.mat-fab.mat-warn,.mat-mini-fab.mat-warn{color:#fff}.mat-flat-button.mat-primary[disabled],.mat-flat-button.mat-accent[disabled],.mat-flat-button.mat-warn[disabled],.mat-flat-button[disabled][disabled],.mat-raised-button.mat-primary[disabled],.mat-raised-button.mat-accent[disabled],.mat-raised-button.mat-warn[disabled],.mat-raised-button[disabled][disabled],.mat-fab.mat-primary[disabled],.mat-fab.mat-accent[disabled],.mat-fab.mat-warn[disabled],.mat-fab[disabled][disabled],.mat-mini-fab.mat-primary[disabled],.mat-mini-fab.mat-accent[disabled],.mat-mini-fab.mat-warn[disabled],.mat-mini-fab[disabled][disabled]{color:rgba(0,0,0,.26)}.mat-flat-button.mat-primary,.mat-raised-button.mat-primary,.mat-fab.mat-primary,.mat-mini-fab.mat-primary{background-color:#f57c00}.mat-flat-button.mat-accent,.mat-raised-button.mat-accent,.mat-fab.mat-accent,.mat-mini-fab.mat-accent{background-color:#ff9800}.mat-flat-button.mat-warn,.mat-raised-button.mat-warn,.mat-fab.mat-warn,.mat-mini-fab.mat-warn{background-color:#f44336}.mat-flat-button.mat-primary[disabled],.mat-flat-button.mat-accent[disabled],.mat-flat-button.mat-warn[disabled],.mat-flat-button[disabled][disabled],.mat-raised-button.mat-primary[disabled],.mat-raised-button.mat-accent[disabled],.mat-raised-button.mat-warn[disabled],.mat-raised-button[disabled][disabled],.mat-fab.mat-primary[disabled],.mat-fab.mat-accent[disabled],.mat-fab.mat-warn[disabled],.mat-fab[disabled][disabled],.mat-mini-fab.mat-primary[disabled],.mat-mini-fab.mat-accent[disabled],.mat-mini-fab.mat-warn[disabled],.mat-mini-fab[disabled][disabled]{background-color:rgba(0,0,0,.12)}.mat-flat-button.mat-primary .mat-ripple-element,.mat-raised-button.mat-primary .mat-ripple-element,.mat-fab.mat-primary .mat-ripple-element,.mat-mini-fab.mat-primary .mat-ripple-element{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-accent .mat-ripple-element,.mat-raised-button.mat-accent .mat-ripple-element,.mat-fab.mat-accent .mat-ripple-element,.mat-mini-fab.mat-accent .mat-ripple-element{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-warn .mat-ripple-element,.mat-raised-button.mat-warn .mat-ripple-element,.mat-fab.mat-warn .mat-ripple-element,.mat-mini-fab.mat-warn .mat-ripple-element{background-color:rgba(255,255,255,.1)}.mat-stroked-button:not([class*=mat-elevation-z]),.mat-flat-button:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-raised-button[disabled]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-fab:not([class*=mat-elevation-z]),.mat-mini-fab:not([class*=mat-elevation-z]){box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-fab:not([disabled]):active:not([class*=mat-elevation-z]),.mat-mini-fab:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-fab[disabled]:not([class*=mat-elevation-z]),.mat-mini-fab[disabled]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone,.mat-button-toggle-group{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard,.mat-button-toggle-group-appearance-standard{box-shadow:none}.mat-button-toggle{color:rgba(0,0,0,.38)}.mat-button-toggle .mat-button-toggle-focus-overlay{background-color:rgba(0,0,0,.12)}.mat-button-toggle-appearance-standard{color:#212121;background:#fff}.mat-button-toggle-appearance-standard .mat-button-toggle-focus-overlay{background-color:#000}.mat-button-toggle-group-appearance-standard .mat-button-toggle+.mat-button-toggle{border-left:solid 1px rgba(0,0,0,.12)}[dir=rtl] .mat-button-toggle-group-appearance-standard .mat-button-toggle+.mat-button-toggle{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-group-appearance-standard.mat-button-toggle-vertical .mat-button-toggle+.mat-button-toggle{border-left:none;border-right:none;border-top:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-checked{background-color:#e0e0e0;color:#616161}.mat-button-toggle-checked.mat-button-toggle-appearance-standard{color:#212121}.mat-button-toggle-disabled{color:rgba(0,0,0,.26);background-color:#eee}.mat-button-toggle-disabled.mat-button-toggle-appearance-standard{background:#fff}.mat-button-toggle-disabled.mat-button-toggle-checked{background-color:#bdbdbd}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard,.mat-button-toggle-group-appearance-standard{border:solid 1px rgba(0,0,0,.12)}.mat-card{background:#fff;color:#212121}.mat-card:not([class*=mat-elevation-z]){box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-card.mat-card-flat:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-card-subtitle{color:#616161}.mat-checkbox-frame{border-color:#616161}.mat-checkbox-checkmark{fill:#fafafa}.mat-checkbox-checkmark-path{stroke:#fafafa !important}.mat-checkbox-mixedmark{background-color:#fafafa}.mat-checkbox-indeterminate.mat-primary .mat-checkbox-background,.mat-checkbox-checked.mat-primary .mat-checkbox-background{background-color:#f57c00}.mat-checkbox-indeterminate.mat-accent .mat-checkbox-background,.mat-checkbox-checked.mat-accent .mat-checkbox-background{background-color:#ff9800}.mat-checkbox-indeterminate.mat-warn .mat-checkbox-background,.mat-checkbox-checked.mat-warn .mat-checkbox-background{background-color:#f44336}.mat-checkbox-disabled.mat-checkbox-checked .mat-checkbox-background,.mat-checkbox-disabled.mat-checkbox-indeterminate .mat-checkbox-background{background-color:#b0b0b0}.mat-checkbox-disabled:not(.mat-checkbox-checked) .mat-checkbox-frame{border-color:#b0b0b0}.mat-checkbox-disabled .mat-checkbox-label{color:#616161}.mat-checkbox .mat-ripple-element{background-color:#000}.mat-checkbox-checked:not(.mat-checkbox-disabled).mat-primary .mat-ripple-element,.mat-checkbox:active:not(.mat-checkbox-disabled).mat-primary .mat-ripple-element{background:#f57c00}.mat-checkbox-checked:not(.mat-checkbox-disabled).mat-accent .mat-ripple-element,.mat-checkbox:active:not(.mat-checkbox-disabled).mat-accent .mat-ripple-element{background:#ff9800}.mat-checkbox-checked:not(.mat-checkbox-disabled).mat-warn .mat-ripple-element,.mat-checkbox:active:not(.mat-checkbox-disabled).mat-warn .mat-ripple-element{background:#f44336}.mat-chip.mat-standard-chip{background-color:#e0e0e0;color:#212121}.mat-chip.mat-standard-chip .mat-chip-remove{color:#212121;opacity:.4}.mat-chip.mat-standard-chip:not(.mat-chip-disabled):active{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-chip.mat-standard-chip:not(.mat-chip-disabled) .mat-chip-remove:hover{opacity:.54}.mat-chip.mat-standard-chip.mat-chip-disabled{opacity:.4}.mat-chip.mat-standard-chip::after{background:#000}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary{background-color:#f57c00;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary .mat-chip-remove{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary .mat-ripple-element{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn{background-color:#f44336;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn .mat-chip-remove{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn .mat-ripple-element{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent{background-color:#ff9800;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent .mat-chip-remove{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent .mat-ripple-element{background-color:rgba(255,255,255,.1)}.mat-table{background:#fff}.mat-table thead,.mat-table tbody,.mat-table tfoot,mat-header-row,mat-row,mat-footer-row,[mat-header-row],[mat-row],[mat-footer-row],.mat-table-sticky{background:inherit}mat-row,mat-header-row,mat-footer-row,th.mat-header-cell,td.mat-cell,td.mat-footer-cell{border-bottom-color:rgba(0,0,0,.12)}.mat-header-cell{color:#616161}.mat-cell,.mat-footer-cell{color:#212121}.mat-calendar-arrow{border-top-color:rgba(0,0,0,.54)}.mat-datepicker-toggle,.mat-datepicker-content .mat-calendar-next-button,.mat-datepicker-content .mat-calendar-previous-button{color:rgba(0,0,0,.54)}.mat-calendar-table-header{color:rgba(0,0,0,.38)}.mat-calendar-table-header-divider::after{background:rgba(0,0,0,.12)}.mat-calendar-body-label{color:#616161}.mat-calendar-body-cell-content{color:#212121;border-color:transparent}.mat-calendar-body-disabled>.mat-calendar-body-cell-content:not(.mat-calendar-body-selected){color:#757575}.mat-calendar-body-cell:not(.mat-calendar-body-disabled):hover>.mat-calendar-body-cell-content:not(.mat-calendar-body-selected),.cdk-keyboard-focused .mat-calendar-body-active>.mat-calendar-body-cell-content:not(.mat-calendar-body-selected),.cdk-program-focused .mat-calendar-body-active>.mat-calendar-body-cell-content:not(.mat-calendar-body-selected){background-color:rgba(0,0,0,.04)}.mat-calendar-body-today:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.38)}.mat-calendar-body-disabled>.mat-calendar-body-today:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.18)}.mat-calendar-body-selected{background-color:#f57c00;color:#fff}.mat-calendar-body-disabled>.mat-calendar-body-selected{background-color:rgba(245,124,0,.4)}.mat-calendar-body-today.mat-calendar-body-selected{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12);background-color:#fff;color:#212121}.mat-datepicker-content.mat-accent .mat-calendar-body-selected{background-color:#ff9800;color:#fff}.mat-datepicker-content.mat-accent .mat-calendar-body-disabled>.mat-calendar-body-selected{background-color:rgba(255,152,0,.4)}.mat-datepicker-content.mat-accent .mat-calendar-body-today.mat-calendar-body-selected{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content.mat-warn .mat-calendar-body-selected{background-color:#f44336;color:#fff}.mat-datepicker-content.mat-warn .mat-calendar-body-disabled>.mat-calendar-body-selected{background-color:rgba(244,67,54,.4)}.mat-datepicker-content.mat-warn .mat-calendar-body-today.mat-calendar-body-selected{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content-touch{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-datepicker-toggle-active{color:#f57c00}.mat-datepicker-toggle-active.mat-accent{color:#ff9800}.mat-datepicker-toggle-active.mat-warn{color:#f44336}.mat-dialog-container{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-divider{border-top-color:rgba(0,0,0,.12)}.mat-divider-vertical{border-right-color:rgba(0,0,0,.12)}.mat-expansion-panel{background:#fff;color:#212121}.mat-expansion-panel:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-action-row{border-top-color:rgba(0,0,0,.12)}.mat-expansion-panel .mat-expansion-panel-header.cdk-keyboard-focused:not([aria-disabled=true]),.mat-expansion-panel .mat-expansion-panel-header.cdk-program-focused:not([aria-disabled=true]),.mat-expansion-panel:not(.mat-expanded) .mat-expansion-panel-header:hover:not([aria-disabled=true]){background:rgba(0,0,0,.04)}@media(hover: none){.mat-expansion-panel:not(.mat-expanded):not([aria-disabled=true]) .mat-expansion-panel-header:hover{background:#fff}}.mat-expansion-panel-header-title{color:#212121}.mat-expansion-panel-header-description,.mat-expansion-indicator::after{color:#616161}.mat-expansion-panel-header[aria-disabled=true]{color:rgba(0,0,0,.26)}.mat-expansion-panel-header[aria-disabled=true] .mat-expansion-panel-header-title,.mat-expansion-panel-header[aria-disabled=true] .mat-expansion-panel-header-description{color:inherit}.mat-form-field-label{color:rgba(97,97,97,.6)}.mat-hint{color:rgba(97,97,97,.6)}.mat-form-field.mat-focused .mat-form-field-label{color:#f57c00}.mat-form-field.mat-focused .mat-form-field-label.mat-accent{color:#ff9800}.mat-form-field.mat-focused .mat-form-field-label.mat-warn{color:#f44336}.mat-focused .mat-form-field-required-marker{color:#ff9800}.mat-form-field-ripple{background-color:rgba(0,0,0,.87)}.mat-form-field.mat-focused .mat-form-field-ripple{background-color:#f57c00}.mat-form-field.mat-focused .mat-form-field-ripple.mat-accent{background-color:#ff9800}.mat-form-field.mat-focused .mat-form-field-ripple.mat-warn{background-color:#f44336}.mat-form-field-type-mat-native-select.mat-focused:not(.mat-form-field-invalid) .mat-form-field-infix::after{color:#f57c00}.mat-form-field-type-mat-native-select.mat-focused:not(.mat-form-field-invalid).mat-accent .mat-form-field-infix::after{color:#ff9800}.mat-form-field-type-mat-native-select.mat-focused:not(.mat-form-field-invalid).mat-warn .mat-form-field-infix::after{color:#f44336}.mat-form-field.mat-form-field-invalid .mat-form-field-label{color:#f44336}.mat-form-field.mat-form-field-invalid .mat-form-field-label.mat-accent,.mat-form-field.mat-form-field-invalid .mat-form-field-label .mat-form-field-required-marker{color:#f44336}.mat-form-field.mat-form-field-invalid .mat-form-field-ripple,.mat-form-field.mat-form-field-invalid .mat-form-field-ripple.mat-accent{background-color:#f44336}.mat-error{color:#f44336}.mat-form-field-appearance-legacy .mat-form-field-label{color:#616161}.mat-form-field-appearance-legacy .mat-hint{color:#616161}.mat-form-field-appearance-legacy .mat-form-field-underline{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-legacy.mat-form-field-disabled .mat-form-field-underline{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-standard .mat-form-field-underline{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-standard.mat-form-field-disabled .mat-form-field-underline{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-fill .mat-form-field-flex{background-color:rgba(0,0,0,.04)}.mat-form-field-appearance-fill.mat-form-field-disabled .mat-form-field-flex{background-color:rgba(0,0,0,.02)}.mat-form-field-appearance-fill .mat-form-field-underline::before{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-fill.mat-form-field-disabled .mat-form-field-label{color:#757575}.mat-form-field-appearance-fill.mat-form-field-disabled .mat-form-field-underline::before{background-color:transparent}.mat-form-field-appearance-outline .mat-form-field-outline{color:rgba(0,0,0,.12)}.mat-form-field-appearance-outline .mat-form-field-outline-thick{color:rgba(0,0,0,.87)}.mat-form-field-appearance-outline.mat-focused .mat-form-field-outline-thick{color:#f57c00}.mat-form-field-appearance-outline.mat-focused.mat-accent .mat-form-field-outline-thick{color:#ff9800}.mat-form-field-appearance-outline.mat-focused.mat-warn .mat-form-field-outline-thick{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-invalid.mat-form-field-invalid .mat-form-field-outline-thick{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-disabled .mat-form-field-label{color:#757575}.mat-form-field-appearance-outline.mat-form-field-disabled .mat-form-field-outline{color:rgba(0,0,0,.06)}.mat-icon.mat-primary{color:#f57c00}.mat-icon.mat-accent{color:#ff9800}.mat-icon.mat-warn{color:#f44336}.mat-form-field-type-mat-native-select .mat-form-field-infix::after{color:#616161}.mat-input-element:disabled,.mat-form-field-type-mat-native-select.mat-form-field-disabled .mat-form-field-infix::after{color:#757575}.mat-input-element{caret-color:#f57c00}.mat-input-element::placeholder{color:rgba(97,97,97,.42)}.mat-input-element::-moz-placeholder{color:rgba(97,97,97,.42)}.mat-input-element::-webkit-input-placeholder{color:rgba(97,97,97,.42)}.mat-input-element:-ms-input-placeholder{color:rgba(97,97,97,.42)}.mat-accent .mat-input-element{caret-color:#ff9800}.mat-warn .mat-input-element,.mat-form-field-invalid .mat-input-element{caret-color:#f44336}.mat-form-field-type-mat-native-select.mat-form-field-invalid .mat-form-field-infix::after{color:#f44336}.mat-list-base .mat-list-item{color:#212121}.mat-list-base .mat-list-option{color:#212121}.mat-list-base .mat-subheader{color:#616161}.mat-list-item-disabled{background-color:#eee}.mat-list-option:hover,.mat-list-option:focus,.mat-nav-list .mat-list-item:hover,.mat-nav-list .mat-list-item:focus,.mat-action-list .mat-list-item:hover,.mat-action-list .mat-list-item:focus{background:rgba(0,0,0,.04)}.mat-list-single-selected-option,.mat-list-single-selected-option:hover,.mat-list-single-selected-option:focus{background:rgba(0,0,0,.12)}.mat-menu-panel{background:#fff}.mat-menu-panel:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-menu-item{background:transparent;color:#212121}.mat-menu-item[disabled],.mat-menu-item[disabled]::after{color:rgba(0,0,0,.38)}.mat-menu-item .mat-icon-no-color,.mat-menu-item-submenu-trigger::after{color:rgba(0,0,0,.54)}.mat-menu-item:hover:not([disabled]),.mat-menu-item.cdk-program-focused:not([disabled]),.mat-menu-item.cdk-keyboard-focused:not([disabled]),.mat-menu-item-highlighted:not([disabled]){background:rgba(0,0,0,.04)}.mat-paginator{background:#fff}.mat-paginator,.mat-paginator-page-size .mat-select-trigger{color:#616161}.mat-paginator-decrement,.mat-paginator-increment{border-top:2px solid rgba(0,0,0,.54);border-right:2px solid rgba(0,0,0,.54)}.mat-paginator-first,.mat-paginator-last{border-top:2px solid rgba(0,0,0,.54)}.mat-icon-button[disabled] .mat-paginator-decrement,.mat-icon-button[disabled] .mat-paginator-increment,.mat-icon-button[disabled] .mat-paginator-first,.mat-icon-button[disabled] .mat-paginator-last{border-color:rgba(0,0,0,.38)}.mat-progress-bar-background{fill:#ff9800}.mat-progress-bar-buffer{background-color:#ff9800}.mat-progress-bar-fill::after{background-color:#f57c00}.mat-progress-bar.mat-accent .mat-progress-bar-background{fill:#ff9800}.mat-progress-bar.mat-accent .mat-progress-bar-buffer{background-color:#ff9800}.mat-progress-bar.mat-accent .mat-progress-bar-fill::after{background-color:#ff9800}.mat-progress-bar.mat-warn .mat-progress-bar-background{fill:#ffcdd2}.mat-progress-bar.mat-warn .mat-progress-bar-buffer{background-color:#ffcdd2}.mat-progress-bar.mat-warn .mat-progress-bar-fill::after{background-color:#f44336}.mat-progress-spinner circle,.mat-spinner circle{stroke:#f57c00}.mat-progress-spinner.mat-accent circle,.mat-spinner.mat-accent circle{stroke:#ff9800}.mat-progress-spinner.mat-warn circle,.mat-spinner.mat-warn circle{stroke:#f44336}.mat-radio-outer-circle{border-color:#616161}.mat-radio-button.mat-primary.mat-radio-checked .mat-radio-outer-circle{border-color:#f57c00}.mat-radio-button.mat-primary .mat-radio-inner-circle,.mat-radio-button.mat-primary .mat-radio-ripple .mat-ripple-element:not(.mat-radio-persistent-ripple),.mat-radio-button.mat-primary.mat-radio-checked .mat-radio-persistent-ripple,.mat-radio-button.mat-primary:active .mat-radio-persistent-ripple{background-color:#f57c00}.mat-radio-button.mat-accent.mat-radio-checked .mat-radio-outer-circle{border-color:#ff9800}.mat-radio-button.mat-accent .mat-radio-inner-circle,.mat-radio-button.mat-accent .mat-radio-ripple .mat-ripple-element:not(.mat-radio-persistent-ripple),.mat-radio-button.mat-accent.mat-radio-checked .mat-radio-persistent-ripple,.mat-radio-button.mat-accent:active .mat-radio-persistent-ripple{background-color:#ff9800}.mat-radio-button.mat-warn.mat-radio-checked .mat-radio-outer-circle{border-color:#f44336}.mat-radio-button.mat-warn .mat-radio-inner-circle,.mat-radio-button.mat-warn .mat-radio-ripple .mat-ripple-element:not(.mat-radio-persistent-ripple),.mat-radio-button.mat-warn.mat-radio-checked .mat-radio-persistent-ripple,.mat-radio-button.mat-warn:active .mat-radio-persistent-ripple{background-color:#f44336}.mat-radio-button.mat-radio-disabled.mat-radio-checked .mat-radio-outer-circle,.mat-radio-button.mat-radio-disabled .mat-radio-outer-circle{border-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled .mat-radio-ripple .mat-ripple-element,.mat-radio-button.mat-radio-disabled .mat-radio-inner-circle{background-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled .mat-radio-label-content{color:rgba(0,0,0,.38)}.mat-radio-button .mat-ripple-element{background-color:#000}.mat-select-value{color:#212121}.mat-select-placeholder{color:rgba(97,97,97,.42)}.mat-select-disabled .mat-select-value{color:#757575}.mat-select-arrow{color:#616161}.mat-select-panel{background:#fff}.mat-select-panel:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-select-panel .mat-option.mat-selected:not(.mat-option-multiple){background:rgba(0,0,0,.12)}.mat-form-field.mat-focused.mat-primary .mat-select-arrow{color:#f57c00}.mat-form-field.mat-focused.mat-accent .mat-select-arrow{color:#ff9800}.mat-form-field.mat-focused.mat-warn .mat-select-arrow{color:#f44336}.mat-form-field .mat-select.mat-select-invalid .mat-select-arrow{color:#f44336}.mat-form-field .mat-select.mat-select-disabled .mat-select-arrow{color:#757575}.mat-drawer-container{background-color:#fafafa;color:#212121}.mat-drawer{background-color:#fff;color:#212121}.mat-drawer.mat-drawer-push{background-color:#fff}.mat-drawer:not(.mat-drawer-side){box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-drawer-side{border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-side.mat-drawer-end{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl] .mat-drawer-side{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl] .mat-drawer-side.mat-drawer-end{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-backdrop.mat-drawer-shown{background-color:rgba(0,0,0,.6)}.mat-slide-toggle.mat-checked .mat-slide-toggle-thumb{background-color:#ff9800}.mat-slide-toggle.mat-checked .mat-slide-toggle-bar{background-color:rgba(255,152,0,.54)}.mat-slide-toggle.mat-checked .mat-ripple-element{background-color:#ff9800}.mat-slide-toggle.mat-primary.mat-checked .mat-slide-toggle-thumb{background-color:#f57c00}.mat-slide-toggle.mat-primary.mat-checked .mat-slide-toggle-bar{background-color:rgba(245,124,0,.54)}.mat-slide-toggle.mat-primary.mat-checked .mat-ripple-element{background-color:#f57c00}.mat-slide-toggle.mat-warn.mat-checked .mat-slide-toggle-thumb{background-color:#f44336}.mat-slide-toggle.mat-warn.mat-checked .mat-slide-toggle-bar{background-color:rgba(244,67,54,.54)}.mat-slide-toggle.mat-warn.mat-checked .mat-ripple-element{background-color:#f44336}.mat-slide-toggle:not(.mat-checked) .mat-ripple-element{background-color:#000}.mat-slide-toggle-thumb{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12);background-color:#fafafa}.mat-slide-toggle-bar{background-color:rgba(0,0,0,.38)}.mat-slider-track-background{background-color:rgba(0,0,0,.26)}.mat-primary .mat-slider-track-fill,.mat-primary .mat-slider-thumb,.mat-primary .mat-slider-thumb-label{background-color:#f57c00}.mat-primary .mat-slider-thumb-label-text{color:#fff}.mat-accent .mat-slider-track-fill,.mat-accent .mat-slider-thumb,.mat-accent .mat-slider-thumb-label{background-color:#ff9800}.mat-accent .mat-slider-thumb-label-text{color:#fff}.mat-warn .mat-slider-track-fill,.mat-warn .mat-slider-thumb,.mat-warn .mat-slider-thumb-label{background-color:#f44336}.mat-warn .mat-slider-thumb-label-text{color:#fff}.mat-slider-focus-ring{background-color:rgba(255,152,0,.2)}.mat-slider:hover .mat-slider-track-background,.cdk-focused .mat-slider-track-background{background-color:rgba(0,0,0,.38)}.mat-slider-disabled .mat-slider-track-background,.mat-slider-disabled .mat-slider-track-fill,.mat-slider-disabled .mat-slider-thumb{background-color:rgba(0,0,0,.26)}.mat-slider-disabled:hover .mat-slider-track-background{background-color:rgba(0,0,0,.26)}.mat-slider-min-value .mat-slider-focus-ring{background-color:rgba(0,0,0,.12)}.mat-slider-min-value.mat-slider-thumb-label-showing .mat-slider-thumb,.mat-slider-min-value.mat-slider-thumb-label-showing .mat-slider-thumb-label{background-color:rgba(0,0,0,.87)}.mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused .mat-slider-thumb,.mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused .mat-slider-thumb-label{background-color:rgba(0,0,0,.26)}.mat-slider-min-value:not(.mat-slider-thumb-label-showing) .mat-slider-thumb{border-color:rgba(0,0,0,.26);background-color:transparent}.mat-slider-min-value:not(.mat-slider-thumb-label-showing):hover .mat-slider-thumb,.mat-slider-min-value:not(.mat-slider-thumb-label-showing).cdk-focused .mat-slider-thumb{border-color:rgba(0,0,0,.38)}.mat-slider-min-value:not(.mat-slider-thumb-label-showing):hover.mat-slider-disabled .mat-slider-thumb,.mat-slider-min-value:not(.mat-slider-thumb-label-showing).cdk-focused.mat-slider-disabled .mat-slider-thumb{border-color:rgba(0,0,0,.26)}.mat-slider-has-ticks .mat-slider-wrapper::after{border-color:rgba(0,0,0,.7)}.mat-slider-horizontal .mat-slider-ticks{background-image:repeating-linear-gradient(to right, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent);background-image:-moz-repeating-linear-gradient(0.0001deg, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-slider-vertical .mat-slider-ticks{background-image:repeating-linear-gradient(to bottom, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-step-header.cdk-keyboard-focused,.mat-step-header.cdk-program-focused,.mat-step-header:hover{background-color:rgba(0,0,0,.04)}@media(hover: none){.mat-step-header:hover{background:none}}.mat-step-header .mat-step-label,.mat-step-header .mat-step-optional{color:#616161}.mat-step-header .mat-step-icon{background-color:#616161;color:#fff}.mat-step-header .mat-step-icon-selected,.mat-step-header .mat-step-icon-state-done,.mat-step-header .mat-step-icon-state-edit{background-color:#f57c00;color:#fff}.mat-step-header .mat-step-icon-state-error{background-color:transparent;color:#f44336}.mat-step-header .mat-step-label.mat-step-label-active{color:#212121}.mat-step-header .mat-step-label.mat-step-label-error{color:#f44336}.mat-stepper-horizontal,.mat-stepper-vertical{background-color:#fff}.mat-stepper-vertical-line::before{border-left-color:rgba(0,0,0,.12)}.mat-horizontal-stepper-header::before,.mat-horizontal-stepper-header::after,.mat-stepper-horizontal-line{border-top-color:rgba(0,0,0,.12)}.mat-sort-header-arrow{color:#616161}.mat-tab-nav-bar,.mat-tab-header{border-bottom:1px solid rgba(0,0,0,.12)}.mat-tab-group-inverted-header .mat-tab-nav-bar,.mat-tab-group-inverted-header .mat-tab-header{border-top:1px solid rgba(0,0,0,.12);border-bottom:none}.mat-tab-label,.mat-tab-link{color:#212121}.mat-tab-label.mat-tab-disabled,.mat-tab-link.mat-tab-disabled{color:#757575}.mat-tab-header-pagination-chevron{border-color:#212121}.mat-tab-header-pagination-disabled .mat-tab-header-pagination-chevron{border-color:#757575}.mat-tab-group[class*=mat-background-] .mat-tab-header,.mat-tab-nav-bar[class*=mat-background-]{border-bottom:none;border-top:none}.mat-tab-group.mat-primary .mat-tab-label.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-group.mat-primary .mat-tab-label.cdk-program-focused:not(.mat-tab-disabled),.mat-tab-group.mat-primary .mat-tab-link.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-group.mat-primary .mat-tab-link.cdk-program-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-primary .mat-tab-label.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-primary .mat-tab-label.cdk-program-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-primary .mat-tab-link.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-primary .mat-tab-link.cdk-program-focused:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-primary .mat-ink-bar,.mat-tab-nav-bar.mat-primary .mat-ink-bar{background-color:#f57c00}.mat-tab-group.mat-primary.mat-background-primary .mat-ink-bar,.mat-tab-nav-bar.mat-primary.mat-background-primary .mat-ink-bar{background-color:#fff}.mat-tab-group.mat-accent .mat-tab-label.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-group.mat-accent .mat-tab-label.cdk-program-focused:not(.mat-tab-disabled),.mat-tab-group.mat-accent .mat-tab-link.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-group.mat-accent .mat-tab-link.cdk-program-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-accent .mat-tab-label.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-accent .mat-tab-label.cdk-program-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-accent .mat-tab-link.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-accent .mat-tab-link.cdk-program-focused:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-accent .mat-ink-bar,.mat-tab-nav-bar.mat-accent .mat-ink-bar{background-color:#ff9800}.mat-tab-group.mat-accent.mat-background-accent .mat-ink-bar,.mat-tab-nav-bar.mat-accent.mat-background-accent .mat-ink-bar{background-color:#fff}.mat-tab-group.mat-warn .mat-tab-label.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-group.mat-warn .mat-tab-label.cdk-program-focused:not(.mat-tab-disabled),.mat-tab-group.mat-warn .mat-tab-link.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-group.mat-warn .mat-tab-link.cdk-program-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-warn .mat-tab-label.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-warn .mat-tab-label.cdk-program-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-warn .mat-tab-link.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-warn .mat-tab-link.cdk-program-focused:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-warn .mat-ink-bar,.mat-tab-nav-bar.mat-warn .mat-ink-bar{background-color:#f44336}.mat-tab-group.mat-warn.mat-background-warn .mat-ink-bar,.mat-tab-nav-bar.mat-warn.mat-background-warn .mat-ink-bar{background-color:#fff}.mat-tab-group.mat-background-primary .mat-tab-label.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-group.mat-background-primary .mat-tab-label.cdk-program-focused:not(.mat-tab-disabled),.mat-tab-group.mat-background-primary .mat-tab-link.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-group.mat-background-primary .mat-tab-link.cdk-program-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-background-primary .mat-tab-label.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-background-primary .mat-tab-label.cdk-program-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-background-primary .mat-tab-link.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-background-primary .mat-tab-link.cdk-program-focused:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-primary .mat-tab-header,.mat-tab-group.mat-background-primary .mat-tab-links,.mat-tab-group.mat-background-primary .mat-tab-header-pagination,.mat-tab-nav-bar.mat-background-primary .mat-tab-header,.mat-tab-nav-bar.mat-background-primary .mat-tab-links,.mat-tab-nav-bar.mat-background-primary .mat-tab-header-pagination{background-color:#f57c00}.mat-tab-group.mat-background-primary .mat-tab-label,.mat-tab-group.mat-background-primary .mat-tab-link,.mat-tab-nav-bar.mat-background-primary .mat-tab-label,.mat-tab-nav-bar.mat-background-primary .mat-tab-link{color:#fff}.mat-tab-group.mat-background-primary .mat-tab-label.mat-tab-disabled,.mat-tab-group.mat-background-primary .mat-tab-link.mat-tab-disabled,.mat-tab-nav-bar.mat-background-primary .mat-tab-label.mat-tab-disabled,.mat-tab-nav-bar.mat-background-primary .mat-tab-link.mat-tab-disabled{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary .mat-tab-header-pagination-chevron,.mat-tab-nav-bar.mat-background-primary .mat-tab-header-pagination-chevron{border-color:#fff}.mat-tab-group.mat-background-primary .mat-tab-header-pagination-disabled .mat-tab-header-pagination-chevron,.mat-tab-nav-bar.mat-background-primary .mat-tab-header-pagination-disabled .mat-tab-header-pagination-chevron{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary .mat-ripple-element,.mat-tab-nav-bar.mat-background-primary .mat-ripple-element{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-accent .mat-tab-label.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-group.mat-background-accent .mat-tab-label.cdk-program-focused:not(.mat-tab-disabled),.mat-tab-group.mat-background-accent .mat-tab-link.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-group.mat-background-accent .mat-tab-link.cdk-program-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-background-accent .mat-tab-label.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-background-accent .mat-tab-label.cdk-program-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-background-accent .mat-tab-link.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-background-accent .mat-tab-link.cdk-program-focused:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-accent .mat-tab-header,.mat-tab-group.mat-background-accent .mat-tab-links,.mat-tab-group.mat-background-accent .mat-tab-header-pagination,.mat-tab-nav-bar.mat-background-accent .mat-tab-header,.mat-tab-nav-bar.mat-background-accent .mat-tab-links,.mat-tab-nav-bar.mat-background-accent .mat-tab-header-pagination{background-color:#ff9800}.mat-tab-group.mat-background-accent .mat-tab-label,.mat-tab-group.mat-background-accent .mat-tab-link,.mat-tab-nav-bar.mat-background-accent .mat-tab-label,.mat-tab-nav-bar.mat-background-accent .mat-tab-link{color:#fff}.mat-tab-group.mat-background-accent .mat-tab-label.mat-tab-disabled,.mat-tab-group.mat-background-accent .mat-tab-link.mat-tab-disabled,.mat-tab-nav-bar.mat-background-accent .mat-tab-label.mat-tab-disabled,.mat-tab-nav-bar.mat-background-accent .mat-tab-link.mat-tab-disabled{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent .mat-tab-header-pagination-chevron,.mat-tab-nav-bar.mat-background-accent .mat-tab-header-pagination-chevron{border-color:#fff}.mat-tab-group.mat-background-accent .mat-tab-header-pagination-disabled .mat-tab-header-pagination-chevron,.mat-tab-nav-bar.mat-background-accent .mat-tab-header-pagination-disabled .mat-tab-header-pagination-chevron{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent .mat-ripple-element,.mat-tab-nav-bar.mat-background-accent .mat-ripple-element{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-warn .mat-tab-label.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-group.mat-background-warn .mat-tab-label.cdk-program-focused:not(.mat-tab-disabled),.mat-tab-group.mat-background-warn .mat-tab-link.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-group.mat-background-warn .mat-tab-link.cdk-program-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-background-warn .mat-tab-label.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-background-warn .mat-tab-label.cdk-program-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-background-warn .mat-tab-link.cdk-keyboard-focused:not(.mat-tab-disabled),.mat-tab-nav-bar.mat-background-warn .mat-tab-link.cdk-program-focused:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-background-warn .mat-tab-header,.mat-tab-group.mat-background-warn .mat-tab-links,.mat-tab-group.mat-background-warn .mat-tab-header-pagination,.mat-tab-nav-bar.mat-background-warn .mat-tab-header,.mat-tab-nav-bar.mat-background-warn .mat-tab-links,.mat-tab-nav-bar.mat-background-warn .mat-tab-header-pagination{background-color:#f44336}.mat-tab-group.mat-background-warn .mat-tab-label,.mat-tab-group.mat-background-warn .mat-tab-link,.mat-tab-nav-bar.mat-background-warn .mat-tab-label,.mat-tab-nav-bar.mat-background-warn .mat-tab-link{color:#fff}.mat-tab-group.mat-background-warn .mat-tab-label.mat-tab-disabled,.mat-tab-group.mat-background-warn .mat-tab-link.mat-tab-disabled,.mat-tab-nav-bar.mat-background-warn .mat-tab-label.mat-tab-disabled,.mat-tab-nav-bar.mat-background-warn .mat-tab-link.mat-tab-disabled{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn .mat-tab-header-pagination-chevron,.mat-tab-nav-bar.mat-background-warn .mat-tab-header-pagination-chevron{border-color:#fff}.mat-tab-group.mat-background-warn .mat-tab-header-pagination-disabled .mat-tab-header-pagination-chevron,.mat-tab-nav-bar.mat-background-warn .mat-tab-header-pagination-disabled .mat-tab-header-pagination-chevron{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn .mat-ripple-element,.mat-tab-nav-bar.mat-background-warn .mat-ripple-element{background-color:rgba(255,255,255,.12)}.mat-toolbar{background:#f5f5f5;color:#212121}.mat-toolbar.mat-primary{background:#f57c00;color:#fff}.mat-toolbar.mat-accent{background:#ff9800;color:#fff}.mat-toolbar.mat-warn{background:#f44336;color:#fff}.mat-toolbar .mat-form-field-underline,.mat-toolbar .mat-form-field-ripple,.mat-toolbar .mat-focused .mat-form-field-ripple{background-color:currentColor}.mat-toolbar .mat-form-field-label,.mat-toolbar .mat-focused .mat-form-field-label,.mat-toolbar .mat-select-value,.mat-toolbar .mat-select-arrow,.mat-toolbar .mat-form-field.mat-focused .mat-select-arrow{color:inherit}.mat-toolbar .mat-input-element{caret-color:currentColor}.mat-tooltip{background:rgba(97,97,97,.9)}.mat-tree{background:#fff}.mat-tree-node,.mat-nested-tree-node{color:#212121}.mat-snack-bar-container{color:rgba(255,255,255,.7);background:#323232;box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-simple-snackbar-action{color:#ff9800}:host{padding-top:5px}.annotation-title{align-items:center;display:flex;font-size:13px;height:20px;padding:0 10px;user-select:none}.selected-row{background-color:#e0e0e0;display:block}.flagged-annotation{color:#f57c00}.hidden-annotation{color:#757575}.annotation-checkbox{padding-right:5px}.flagged-icon{transform:scale(0.6)}.hidden-icon{transform:scale(0.6)}.chart-div{border-bottom:1px solid #ebebeb}.chart-svg{width:100%;user-select:none}.default-text{fill:#000}.flag-text{fill:#f57c00}.hidden-text{fill:#757575}\n"], encapsulation: 2, changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/annotations_list/annotation/annotation_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class AnnotationContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.flaggedAnnotations$ = this.store.select(getFlaggedAnnotations);
            this.hiddenAnnotations$ = this.store.select(getHiddenAnnotations);
            this.selectedAnnotations$ = this.store.select(getSelectedAnnotations);
            this.showCounts$ = this.store.select(getShowCounts);
            this.sidebarWidth$ = this.store.select(getSidebarWidth);
        }
    }
    AnnotationContainer.decorators = [
        { type: Component, args: [{
                    selector: 'npmi-annotation',
                    template: `
    <annotation-component
      [data]="data"
      [maxCount]="maxCount"
      [activeMetrics]="activeMetrics"
      [numActiveRuns]="numActiveRuns"
      [annotation]="annotation"
      [runHeight]="runHeight"
      [selectedAnnotations]="selectedAnnotations$ | async"
      [flaggedAnnotations]="flaggedAnnotations$ | async"
      [hiddenAnnotations]="hiddenAnnotations$ | async"
      [showCounts]="showCounts$ | async"
      [sidebarWidth]="sidebarWidth$ | async"
    ></annotation-component>
  `,
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    AnnotationContainer.ctorParameters = () => [
        { type: Store }
    ];
    AnnotationContainer.propDecorators = {
        data: [{ type: Input }],
        maxCount: [{ type: Input }],
        activeMetrics: [{ type: Input }],
        numActiveRuns: [{ type: Input }],
        annotation: [{ type: Input }],
        runHeight: [{ type: Input }]
    };
    /** @nocollapse */ AnnotationContainer.ɵfac = function AnnotationContainer_Factory(t) { return new (t || AnnotationContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ AnnotationContainer.ɵcmp = ɵɵdefineComponent({ type: AnnotationContainer, selectors: [["npmi-annotation"]], inputs: { data: "data", maxCount: "maxCount", activeMetrics: "activeMetrics", numActiveRuns: "numActiveRuns", annotation: "annotation", runHeight: "runHeight" }, decls: 6, vars: 21, consts: [[3, "data", "maxCount", "activeMetrics", "numActiveRuns", "annotation", "runHeight", "selectedAnnotations", "flaggedAnnotations", "hiddenAnnotations", "showCounts", "sidebarWidth"]], template: function AnnotationContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelement(0, "annotation-component", 0);
            ɵɵpipe(1, "async");
            ɵɵpipe(2, "async");
            ɵɵpipe(3, "async");
            ɵɵpipe(4, "async");
            ɵɵpipe(5, "async");
        } if (rf & 2) {
            ɵɵproperty("data", ctx.data)("maxCount", ctx.maxCount)("activeMetrics", ctx.activeMetrics)("numActiveRuns", ctx.numActiveRuns)("annotation", ctx.annotation)("runHeight", ctx.runHeight)("selectedAnnotations", ɵɵpipeBind1(1, 11, ctx.selectedAnnotations$))("flaggedAnnotations", ɵɵpipeBind1(2, 13, ctx.flaggedAnnotations$))("hiddenAnnotations", ɵɵpipeBind1(3, 15, ctx.hiddenAnnotations$))("showCounts", ɵɵpipeBind1(4, 17, ctx.showCounts$))("sidebarWidth", ɵɵpipeBind1(5, 19, ctx.sidebarWidth$));
        } }, directives: [AnnotationComponent], pipes: [AsyncPipe], encapsulation: 2, changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/annotations_list/annotations_list_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    function AnnotationsListComponent_ng_container_1_npmi_annotation_4_Template(rf, ctx) { if (rf & 1) {
        const _r4 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "npmi-annotation", 5);
        ɵɵlistener("click", function AnnotationsListComponent_ng_container_1_npmi_annotation_4_Template_npmi_annotation_click_0_listener($event) { ɵɵrestoreView(_r4); const annotation_r2 = ctx.$implicit; const ctx_r3 = ɵɵnextContext(2); return ctx_r3.rowClicked($event, annotation_r2); });
        ɵɵelementEnd();
    } if (rf & 2) {
        const annotation_r2 = ctx.$implicit;
        const ctx_r1 = ɵɵnextContext(2);
        ɵɵproperty("data", ctx_r1.annotations[annotation_r2])("activeMetrics", ctx_r1.activeMetrics)("numActiveRuns", ctx_r1.numActiveRuns)("maxCount", ctx_r1.maxCount)("annotation", annotation_r2)("runHeight", ctx_r1.runHeight);
    } }
    function AnnotationsListComponent_ng_container_1_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementContainerStart(0);
        ɵɵelement(1, "npmi-annotations-list-legend");
        ɵɵelement(2, "npmi-annotations-list-header", 2);
        ɵɵelementStart(3, "cdk-virtual-scroll-viewport", 3);
        ɵɵtemplate(4, AnnotationsListComponent_ng_container_1_npmi_annotation_4_Template, 1, 6, "npmi-annotation", 4);
        ɵɵelementEnd();
        ɵɵelementContainerEnd();
    } if (rf & 2) {
        const ctx_r0 = ɵɵnextContext();
        ɵɵadvance(2);
        ɵɵproperty("annotations", ctx_r0.annotations)("numAnnotations", ctx_r0.numAnnotations)("activeMetrics", ctx_r0.activeMetrics);
        ɵɵadvance(1);
        ɵɵpropertyInterpolate("itemSize", ctx_r0.numActiveRuns * ctx_r0.runHeight + 25);
        ɵɵadvance(1);
        ɵɵproperty("cdkVirtualForOf", ctx_r0.sortedAnnotations);
    } }
    class AnnotationsListComponent {
        constructor() {
            this.onRowClick = new EventEmitter();
            this.runHeight = 30;
        }
        /**
         * @param {?} event
         * @param {?} annotation
         * @return {?}
         */
        rowClicked(event, annotation) {
            // Shift pressed, handle multiple annotations
            if (event.shiftKey) {
                /** @type {?} */
                let annotationIndex = this.sortedAnnotations.indexOf(annotation);
                if (this.selectedAnnotations.length === 0) {
                    this.onRowClick.emit(this.sortedAnnotations.slice(0, annotationIndex + 1));
                }
                else {
                    /** @type {?} */
                    let lastAnnotation = this.selectedAnnotations[this.selectedAnnotations.length - 1];
                    /** @type {?} */
                    const lastIndex = this.sortedAnnotations.indexOf(lastAnnotation);
                    if (lastIndex < annotationIndex) {
                        this.onRowClick.emit(this.sortedAnnotations.slice(lastIndex, annotationIndex + 1));
                    }
                    else {
                        this.onRowClick.emit(this.sortedAnnotations.slice(annotationIndex, lastIndex + 1));
                    }
                }
            }
            else {
                // No shift, only one annotation clicked
                this.onRowClick.emit([annotation]);
            }
        }
    }
    AnnotationsListComponent.decorators = [
        { type: Component, args: [{
                    selector: 'annotations-list-component',
                    templateUrl: './annotations_list_component.ng.html',
                    styleUrls: ['./annotations_list_component.css'],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    AnnotationsListComponent.propDecorators = {
        annotations: [{ type: Input }],
        annotationsExpanded: [{ type: Input }],
        numAnnotations: [{ type: Input }],
        annotationSort: [{ type: Input }],
        activeMetrics: [{ type: Input }],
        numActiveRuns: [{ type: Input }],
        sortedAnnotations: [{ type: Input }],
        selectedAnnotations: [{ type: Input }],
        maxCount: [{ type: Input }],
        onRowClick: [{ type: Output }]
    };
    /** @nocollapse */ AnnotationsListComponent.ɵfac = function AnnotationsListComponent_Factory(t) { return new (t || AnnotationsListComponent)(); };
    /** @nocollapse */ AnnotationsListComponent.ɵcmp = ɵɵdefineComponent({ type: AnnotationsListComponent, selectors: [["annotations-list-component"]], inputs: { annotations: "annotations", annotationsExpanded: "annotationsExpanded", numAnnotations: "numAnnotations", annotationSort: "annotationSort", activeMetrics: "activeMetrics", numActiveRuns: "numActiveRuns", sortedAnnotations: "sortedAnnotations", selectedAnnotations: "selectedAnnotations", maxCount: "maxCount" }, outputs: { onRowClick: "onRowClick" }, decls: 2, vars: 3, consts: [[3, "numAnnotations", "expanded"], [4, "ngIf"], [3, "annotations", "numAnnotations", "activeMetrics"], ["minBufferPx", "300", "maxBufferPx", "600", 1, "annotation-rows", 3, "itemSize"], [3, "data", "activeMetrics", "numActiveRuns", "maxCount", "annotation", "runHeight", "click", 4, "cdkVirtualFor", "cdkVirtualForOf"], [3, "data", "activeMetrics", "numActiveRuns", "maxCount", "annotation", "runHeight", "click"]], template: function AnnotationsListComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelement(0, "npmi-annotations-list-toolbar", 0);
            ɵɵtemplate(1, AnnotationsListComponent_ng_container_1_Template, 5, 5, "ng-container", 1);
        } if (rf & 2) {
            ɵɵproperty("numAnnotations", ctx.numAnnotations)("expanded", ctx.annotationsExpanded);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx.annotationsExpanded);
        } }, directives: [AnnotationsListToolbarContainer, NgIf, LegendComponent, HeaderContainer, CdkVirtualScrollViewport, CdkFixedSizeVirtualScroll, CdkVirtualForOf, AnnotationContainer], styles: [".mat-badge-content[_ngcontent-%COMP%]{font-weight:600;font-size:12px;font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:9px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:24px}.mat-h1[_ngcontent-%COMP%], .mat-headline[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%]{font:400 24px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h2[_ngcontent-%COMP%], .mat-title[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h3[_ngcontent-%COMP%], .mat-subheading-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%]{font:400 16px/28px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h4[_ngcontent-%COMP%], .mat-subheading-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%]{font:400 15px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h5[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%]{font:400 calc(14px * 0.83)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-h6[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:400 calc(14px * 0.67)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-body-strong[_ngcontent-%COMP%], .mat-body-2[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   p[_ngcontent-%COMP%]{margin:0 0 12px}.mat-small[_ngcontent-%COMP%], .mat-caption[_ngcontent-%COMP%]{font:400 12px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-display-4[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-4[_ngcontent-%COMP%]{font:300 112px/112px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.05em;margin:0 0 56px}.mat-display-3[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-3[_ngcontent-%COMP%]{font:400 56px/56px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.02em;margin:0 0 64px}.mat-display-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-2[_ngcontent-%COMP%]{font:400 45px/48px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.005em;margin:0 0 64px}.mat-display-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-1[_ngcontent-%COMP%]{font:400 34px/40px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 64px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%], .mat-flat-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-button-toggle[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card-title[_ngcontent-%COMP%]{font-size:24px;font-weight:500}.mat-card-header[_ngcontent-%COMP%]   .mat-card-title[_ngcontent-%COMP%]{font-size:20px}.mat-card-subtitle[_ngcontent-%COMP%], .mat-card-content[_ngcontent-%COMP%]{font-size:14px}.mat-checkbox[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-checkbox-layout[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{line-height:24px}.mat-chip[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-chip[_ngcontent-%COMP%]   .mat-chip-trailing-icon.mat-icon[_ngcontent-%COMP%], .mat-chip[_ngcontent-%COMP%]   .mat-chip-remove.mat-icon[_ngcontent-%COMP%]{font-size:18px}.mat-table[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-header-cell[_ngcontent-%COMP%]{font-size:12px;font-weight:500}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{font-size:14px}.mat-calendar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-calendar-body[_ngcontent-%COMP%]{font-size:13px}.mat-calendar-body-label[_ngcontent-%COMP%], .mat-calendar-period-button[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-calendar-table-header[_ngcontent-%COMP%]   th[_ngcontent-%COMP%]{font-size:11px;font-weight:400}.mat-dialog-title[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-expansion-panel-header[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:15px;font-weight:400}.mat-expansion-panel-content[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field[_ngcontent-%COMP%]{font-size:inherit;font-weight:400;line-height:1.125;font-family:Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.34375em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{font-size:150%;line-height:1.125}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]{height:1.5em;width:1.5em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{height:1.125em;line-height:1.125}.mat-form-field-infix[_ngcontent-%COMP%]{padding:.5em 0;border-top:.84375em solid transparent}.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34375em) scale(0.75);width:133.3333333333%}.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34374em) scale(0.75);width:133.3333433333%}.mat-form-field-label-wrapper[_ngcontent-%COMP%]{top:-0.84375em;padding-top:.84375em}.mat-form-field-label[_ngcontent-%COMP%]{top:1.34375em}.mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.34375em}.mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{font-size:75%;margin-top:.6666666667em;top:calc(100% - 1.7916666667em)}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.4375em 0}.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.001px);-ms-transform:translateY(-1.28125em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00101px);-ms-transform:translateY(-1.28124em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00102px);-ms-transform:translateY(-1.28123em) scale(0.75);width:133.3333533333%}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.28125em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{margin-top:.5416666667em;top:calc(100% - 1.6666666667em)}@media print{.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28122em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28121em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.2812em) scale(0.75)}}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.25em 0 .75em 0}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.09375em;margin-top:-0.5em}.mat-form-field-appearance-fill.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59374em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:1em 0 1em 0}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.84375em;margin-top:-0.25em}.mat-form-field-appearance-outline.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59374em) scale(0.75);width:133.3333433333%}.mat-grid-tile-header[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]{font-size:14px}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2), .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}input.mat-input-element[_ngcontent-%COMP%]{margin-top:-0.0625em}.mat-menu-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:400}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px}.mat-radio-button[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select-trigger[_ngcontent-%COMP%]{height:1.125em}.mat-slide-toggle-content[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-slider-thumb-label-text[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-stepper-vertical[_ngcontent-%COMP%], .mat-stepper-horizontal[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-step-label[_ngcontent-%COMP%]{font-size:14px;font-weight:400}.mat-step-sub-label-error[_ngcontent-%COMP%]{font-weight:normal}.mat-step-label-error[_ngcontent-%COMP%]{font-size:14px}.mat-step-label-selected[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-tab-group[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-toolbar[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0}.mat-tooltip[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:10px;padding-top:6px;padding-bottom:6px}.mat-tooltip-handset[_ngcontent-%COMP%]{font-size:14px;padding-top:8px;padding-bottom:8px}.mat-list-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:16px}.mat-optgroup-label[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-simple-snackbar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px}.mat-simple-snackbar-action[_ngcontent-%COMP%]{line-height:1;font-family:inherit;font-size:inherit;font-weight:500}.mat-tree[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{font-weight:400;font-size:14px}.mat-ripple[_ngcontent-%COMP%]{overflow:hidden;position:relative}.mat-ripple[_ngcontent-%COMP%]:not(:empty){transform:translateZ(0)}.mat-ripple.mat-ripple-unbounded[_ngcontent-%COMP%]{overflow:visible}.mat-ripple-element[_ngcontent-%COMP%]{position:absolute;border-radius:50%;pointer-events:none;transition:opacity,transform 0ms cubic-bezier(0, 0, 0.2, 1);transform:scale(0)}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{display:none}.cdk-visually-hidden[_ngcontent-%COMP%]{border:0;clip:rect(0 0 0 0);height:1px;margin:-1px;overflow:hidden;padding:0;position:absolute;width:1px;outline:0;-webkit-appearance:none;-moz-appearance:none}.cdk-overlay-container[_ngcontent-%COMP%], .cdk-global-overlay-wrapper[_ngcontent-%COMP%]{pointer-events:none;top:0;left:0;height:100%;width:100%}.cdk-overlay-container[_ngcontent-%COMP%]{position:fixed;z-index:1000}.cdk-overlay-container[_ngcontent-%COMP%]:empty{display:none}.cdk-global-overlay-wrapper[_ngcontent-%COMP%]{display:flex;position:absolute;z-index:1000}.cdk-overlay-pane[_ngcontent-%COMP%]{position:absolute;pointer-events:auto;box-sizing:border-box;z-index:1000;display:flex;max-width:100%;max-height:100%}.cdk-overlay-backdrop[_ngcontent-%COMP%]{position:absolute;top:0;bottom:0;left:0;right:0;z-index:1000;pointer-events:auto;-webkit-tap-highlight-color:transparent;transition:opacity 400ms cubic-bezier(0.25, 0.8, 0.25, 1);opacity:0}.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:1}@media screen and (-ms-high-contrast: active){.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:.6}}.cdk-overlay-dark-backdrop[_ngcontent-%COMP%]{background:rgba(0,0,0,.32)}.cdk-overlay-transparent-backdrop[_ngcontent-%COMP%], .cdk-overlay-transparent-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:0}.cdk-overlay-connected-position-bounding-box[_ngcontent-%COMP%]{position:absolute;z-index:1000;display:flex;flex-direction:column;min-width:1px;min-height:1px}.cdk-global-scrollblock[_ngcontent-%COMP%]{position:fixed;width:100%;overflow-y:scroll}@keyframes cdk-text-field-autofill-start{}@keyframes cdk-text-field-autofill-end{}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:-webkit-autofill{animation:cdk-text-field-autofill-start 0s 1ms}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:not(:-webkit-autofill){animation:cdk-text-field-autofill-end 0s 1ms}textarea.cdk-textarea-autosize[_ngcontent-%COMP%]{resize:none}textarea.cdk-textarea-autosize-measuring[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:auto !important;overflow:hidden !important}textarea.cdk-textarea-autosize-measuring-firefox[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:0 !important}.mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.1)}.mat-option[_ngcontent-%COMP%]{color:#212121}.mat-option[_ngcontent-%COMP%]:hover:not(.mat-option-disabled), .mat-option[_ngcontent-%COMP%]:focus:not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple):not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-active[_ngcontent-%COMP%]{background:rgba(0,0,0,.04);color:#212121}.mat-option.mat-option-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-primary[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f57c00}.mat-accent[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f44336}.mat-optgroup-label[_ngcontent-%COMP%]{color:#616161}.mat-optgroup-disabled[_ngcontent-%COMP%]   .mat-optgroup-label[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-pseudo-checkbox[_ngcontent-%COMP%]{color:#616161}.mat-pseudo-checkbox[_ngcontent-%COMP%]::after{color:#fafafa}.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{color:#b0b0b0}.mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f57c00}.mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f44336}.mat-pseudo-checkbox-checked.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{background:#b0b0b0}.mat-elevation-z0[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z1[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z2[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z3[_ngcontent-%COMP%]{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z4[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z5[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 5px 8px 0px rgba(0, 0, 0, 0.14),0px 1px 14px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z6[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z7[_ngcontent-%COMP%]{box-shadow:0px 4px 5px -2px rgba(0, 0, 0, 0.2),0px 7px 10px 1px rgba(0, 0, 0, 0.14),0px 2px 16px 1px rgba(0, 0, 0, 0.12)}.mat-elevation-z8[_ngcontent-%COMP%]{box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z9[_ngcontent-%COMP%]{box-shadow:0px 5px 6px -3px rgba(0, 0, 0, 0.2),0px 9px 12px 1px rgba(0, 0, 0, 0.14),0px 3px 16px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z10[_ngcontent-%COMP%]{box-shadow:0px 6px 6px -3px rgba(0, 0, 0, 0.2),0px 10px 14px 1px rgba(0, 0, 0, 0.14),0px 4px 18px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z11[_ngcontent-%COMP%]{box-shadow:0px 6px 7px -4px rgba(0, 0, 0, 0.2),0px 11px 15px 1px rgba(0, 0, 0, 0.14),0px 4px 20px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z12[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z13[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 13px 19px 2px rgba(0, 0, 0, 0.14),0px 5px 24px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z14[_ngcontent-%COMP%]{box-shadow:0px 7px 9px -4px rgba(0, 0, 0, 0.2),0px 14px 21px 2px rgba(0, 0, 0, 0.14),0px 5px 26px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z15[_ngcontent-%COMP%]{box-shadow:0px 8px 9px -5px rgba(0, 0, 0, 0.2),0px 15px 22px 2px rgba(0, 0, 0, 0.14),0px 6px 28px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z16[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z17[_ngcontent-%COMP%]{box-shadow:0px 8px 11px -5px rgba(0, 0, 0, 0.2),0px 17px 26px 2px rgba(0, 0, 0, 0.14),0px 6px 32px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z18[_ngcontent-%COMP%]{box-shadow:0px 9px 11px -5px rgba(0, 0, 0, 0.2),0px 18px 28px 2px rgba(0, 0, 0, 0.14),0px 7px 34px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z19[_ngcontent-%COMP%]{box-shadow:0px 9px 12px -6px rgba(0, 0, 0, 0.2),0px 19px 29px 2px rgba(0, 0, 0, 0.14),0px 7px 36px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z20[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 20px 31px 3px rgba(0, 0, 0, 0.14),0px 8px 38px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z21[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 21px 33px 3px rgba(0, 0, 0, 0.14),0px 8px 40px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z22[_ngcontent-%COMP%]{box-shadow:0px 10px 14px -6px rgba(0, 0, 0, 0.2),0px 22px 35px 3px rgba(0, 0, 0, 0.14),0px 8px 42px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z23[_ngcontent-%COMP%]{box-shadow:0px 11px 14px -7px rgba(0, 0, 0, 0.2),0px 23px 36px 3px rgba(0, 0, 0, 0.14),0px 9px 44px 8px rgba(0, 0, 0, 0.12)}.mat-elevation-z24[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12)}.mat-app-background[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-theme-loaded-marker[_ngcontent-%COMP%]{display:none}.mat-autocomplete-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-autocomplete-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover){background:#fff}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover):not(.mat-option-disabled){color:#212121}.mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f57c00}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{outline:solid 1px;border-radius:0}.mat-badge-accent[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-badge-warn[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f44336}.mat-badge[_ngcontent-%COMP%]{position:relative}.mat-badge-hidden[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{display:none}.mat-badge-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#b9b9b9;color:#757575}.mat-badge-content[_ngcontent-%COMP%]{position:absolute;text-align:center;display:inline-block;border-radius:50%;transition:transform 200ms ease-in-out;transform:scale(0.6);overflow:hidden;white-space:nowrap;text-overflow:ellipsis;pointer-events:none}.ng-animate-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%], .mat-badge-content._mat-animation-noopable[_ngcontent-%COMP%]{transition:none}.mat-badge-content.mat-badge-active[_ngcontent-%COMP%]{transform:none}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:16px;height:16px;line-height:16px}.mat-badge-small.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-8px}.mat-badge-small.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-8px}.mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-16px}.mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-16px}.mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-8px}.mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-8px}.mat-badge-medium[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:22px;height:22px;line-height:22px}.mat-badge-medium.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-11px}.mat-badge-medium.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-11px}.mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-22px}.mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-22px}.mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-11px}.mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-11px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:28px;height:28px;line-height:28px}.mat-badge-large.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-14px}.mat-badge-large.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-14px}.mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-28px}.mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-28px}.mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-14px}.mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-14px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]{color:inherit;background:transparent}.mat-button.mat-primary[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-button[disabled][disabled][_ngcontent-%COMP%], .mat-icon-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-icon-button[disabled][disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-stroked-button[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f44336}.mat-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:transparent}.mat-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{opacity:.1;background-color:currentColor}.mat-button-focus-overlay[_ngcontent-%COMP%]{background:#000}.mat-stroked-button[_ngcontent-%COMP%]:not([disabled]){border-color:rgba(0,0,0,.12)}.mat-flat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{color:#212121;background-color:#fff}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-flat-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-stroked-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-flat-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-raised-button[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone[_ngcontent-%COMP%], .mat-button-toggle-group[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{box-shadow:none}.mat-button-toggle[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-button-toggle[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121;background:#fff}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:#000}.mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12)}[dir=rtl][_ngcontent-%COMP%]   .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-group-appearance-standard.mat-button-toggle-vertical[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:none;border-top:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#616161}.mat-button-toggle-checked.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121}.mat-button-toggle-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.26);background-color:#eee}.mat-button-toggle-disabled.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{background:#fff}.mat-button-toggle-disabled.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#bdbdbd}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{border:solid 1px rgba(0,0,0,.12)}.mat-card[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-card[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-card.mat-card-flat[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-card-subtitle[_ngcontent-%COMP%]{color:#616161}.mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#616161}.mat-checkbox-checkmark[_ngcontent-%COMP%]{fill:#fafafa}.mat-checkbox-checkmark-path[_ngcontent-%COMP%]{stroke:#fafafa !important}.mat-checkbox-mixedmark[_ngcontent-%COMP%]{background-color:#fafafa}.mat-checkbox-indeterminate.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f57c00}.mat-checkbox-indeterminate.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#ff9800}.mat-checkbox-indeterminate.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f44336}.mat-checkbox-disabled.mat-checkbox-checked[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-disabled.mat-checkbox-indeterminate[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]:not(.mat-checkbox-checked)   .mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{color:#616161}.mat-checkbox[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%]{background:#f57c00}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%]{background:#ff9800}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%]{background:#f44336}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#212121}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#212121;opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled):active{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled)   .mat-chip-remove[_ngcontent-%COMP%]:hover{opacity:.54}.mat-chip.mat-standard-chip.mat-chip-disabled[_ngcontent-%COMP%]{opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]::after{background:#000}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-table[_ngcontent-%COMP%]{background:#fff}.mat-table[_ngcontent-%COMP%]   thead[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tbody[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tfoot[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], [mat-header-row][_ngcontent-%COMP%], [mat-row][_ngcontent-%COMP%], [mat-footer-row][_ngcontent-%COMP%], .mat-table-sticky[_ngcontent-%COMP%]{background:inherit}mat-row[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], th.mat-header-cell[_ngcontent-%COMP%], td.mat-cell[_ngcontent-%COMP%], td.mat-footer-cell[_ngcontent-%COMP%]{border-bottom-color:rgba(0,0,0,.12)}.mat-header-cell[_ngcontent-%COMP%]{color:#616161}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{color:#212121}.mat-calendar-arrow[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.54)}.mat-datepicker-toggle[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-next-button[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-previous-button[_ngcontent-%COMP%]{color:rgba(0,0,0,.54)}.mat-calendar-table-header[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-calendar-table-header-divider[_ngcontent-%COMP%]::after{background:rgba(0,0,0,.12)}.mat-calendar-body-label[_ngcontent-%COMP%]{color:#616161}.mat-calendar-body-cell-content[_ngcontent-%COMP%]{color:#212121;border-color:transparent}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){color:#757575}.mat-calendar-body-cell[_ngcontent-%COMP%]:not(.mat-calendar-body-disabled):hover > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-keyboard-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-program-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){background-color:rgba(0,0,0,.04)}.mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.38)}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.18)}.mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.4)}.mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12);background-color:#fff;color:#212121}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.4)}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.4)}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content-touch[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-datepicker-toggle-active[_ngcontent-%COMP%]{color:#f57c00}.mat-datepicker-toggle-active.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-datepicker-toggle-active.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-dialog-container[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-divider[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-divider-vertical[_ngcontent-%COMP%]{border-right-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-expansion-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-action-row[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-keyboard-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-program-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded)   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover:not([aria-disabled=true]){background:rgba(0,0,0,.04)}@media(hover: none){.mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded):not([aria-disabled=true])   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover{background:#fff}}.mat-expansion-panel-header-title[_ngcontent-%COMP%]{color:#212121}.mat-expansion-panel-header-description[_ngcontent-%COMP%], .mat-expansion-indicator[_ngcontent-%COMP%]::after{color:#616161}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-title[_ngcontent-%COMP%], .mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-description[_ngcontent-%COMP%]{color:inherit}.mat-form-field-label[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-hint[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-focused[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-ripple[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid)   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f57c00}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-accent   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#ff9800}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-warn   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#f44336}.mat-error[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-hint[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-legacy.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-standard[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-standard.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.04)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.02)}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:transparent}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.12)}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:rgba(0,0,0,.87)}.mat-form-field-appearance-outline.mat-focused[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field-appearance-outline.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-appearance-outline.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-invalid.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.06)}.mat-icon.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-icon.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-icon.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-type-mat-native-select[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#616161}.mat-input-element[_ngcontent-%COMP%]:disabled, .mat-form-field-type-mat-native-select.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#757575}.mat-input-element[_ngcontent-%COMP%]{caret-color:#f57c00}.mat-input-element[_ngcontent-%COMP%]::placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-moz-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-webkit-input-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]:-ms-input-placeholder{color:rgba(97,97,97,.42)}.mat-accent[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%], .mat-form-field-invalid[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#f44336}.mat-form-field-type-mat-native-select.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{color:#616161}.mat-list-item-disabled[_ngcontent-%COMP%]{background-color:#eee}.mat-list-option[_ngcontent-%COMP%]:hover, .mat-list-option[_ngcontent-%COMP%]:focus, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.04)}.mat-list-single-selected-option[_ngcontent-%COMP%], .mat-list-single-selected-option[_ngcontent-%COMP%]:hover, .mat-list-single-selected-option[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.12)}.mat-menu-panel[_ngcontent-%COMP%]{background:#fff}.mat-menu-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-menu-item[_ngcontent-%COMP%]{background:transparent;color:#212121}.mat-menu-item[disabled][_ngcontent-%COMP%], .mat-menu-item[disabled][_ngcontent-%COMP%]::after{color:rgba(0,0,0,.38)}.mat-menu-item[_ngcontent-%COMP%]   .mat-icon-no-color[_ngcontent-%COMP%], .mat-menu-item-submenu-trigger[_ngcontent-%COMP%]::after{color:rgba(0,0,0,.54)}.mat-menu-item[_ngcontent-%COMP%]:hover:not([disabled]), .mat-menu-item.cdk-program-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item.cdk-keyboard-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item-highlighted[_ngcontent-%COMP%]:not([disabled]){background:rgba(0,0,0,.04)}.mat-paginator[_ngcontent-%COMP%]{background:#fff}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{color:#616161}.mat-paginator-decrement[_ngcontent-%COMP%], .mat-paginator-increment[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54);border-right:2px solid rgba(0,0,0,.54)}.mat-paginator-first[_ngcontent-%COMP%], .mat-paginator-last[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54)}.mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-decrement[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-increment[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-first[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-last[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f57c00}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#ff9800}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f44336}.mat-progress-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f57c00}.mat-progress-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#ff9800}.mat-progress-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f44336}.mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#616161}.mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f57c00}.mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#ff9800}.mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#ff9800}.mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f44336}.mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f44336}.mat-radio-button.mat-radio-disabled.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-label-content[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-radio-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-select-value[_ngcontent-%COMP%]{color:#212121}.mat-select-placeholder[_ngcontent-%COMP%]{color:rgba(97,97,97,.42)}.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%]{color:#757575}.mat-select-arrow[_ngcontent-%COMP%]{color:#616161}.mat-select-panel[_ngcontent-%COMP%]{background:#fff}.mat-select-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-select-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple){background:rgba(0,0,0,.12)}.mat-form-field.mat-focused.mat-primary[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-invalid[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#757575}.mat-drawer-container[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-drawer[_ngcontent-%COMP%]{background-color:#fff;color:#212121}.mat-drawer.mat-drawer-push[_ngcontent-%COMP%]{background-color:#fff}.mat-drawer[_ngcontent-%COMP%]:not(.mat-drawer-side){box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-drawer-side[_ngcontent-%COMP%]{border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-backdrop.mat-drawer-shown[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.6)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.54)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.54)}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.54)}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle[_ngcontent-%COMP%]:not(.mat-checked)   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-slide-toggle-thumb[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12);background-color:#fafafa}.mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-primary[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f57c00}.mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-accent[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#ff9800}.mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-warn[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f44336}.mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.2)}.mat-slider[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%], .cdk-focused[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-disabled[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]   .mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing)   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26);background-color:transparent}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26)}.mat-slider-has-ticks[_ngcontent-%COMP%]   .mat-slider-wrapper[_ngcontent-%COMP%]::after{border-color:rgba(0,0,0,.7)}.mat-slider-horizontal[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to right, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent);background-image:-moz-repeating-linear-gradient(0.0001deg, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-slider-vertical[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to bottom, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-step-header.cdk-keyboard-focused[_ngcontent-%COMP%], .mat-step-header.cdk-program-focused[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]:hover{background-color:rgba(0,0,0,.04)}@media(hover: none){.mat-step-header[_ngcontent-%COMP%]:hover{background:none}}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-optional[_ngcontent-%COMP%]{color:#616161}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon[_ngcontent-%COMP%]{background-color:#616161;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-selected[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-done[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-edit[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-error[_ngcontent-%COMP%]{background-color:transparent;color:#f44336}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-active[_ngcontent-%COMP%]{color:#212121}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-error[_ngcontent-%COMP%]{color:#f44336}.mat-stepper-horizontal[_ngcontent-%COMP%], .mat-stepper-vertical[_ngcontent-%COMP%]{background-color:#fff}.mat-stepper-vertical-line[_ngcontent-%COMP%]::before{border-left-color:rgba(0,0,0,.12)}.mat-horizontal-stepper-header[_ngcontent-%COMP%]::before, .mat-horizontal-stepper-header[_ngcontent-%COMP%]::after, .mat-stepper-horizontal-line[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-sort-header-arrow[_ngcontent-%COMP%]{color:#616161}.mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-header[_ngcontent-%COMP%]{border-bottom:1px solid rgba(0,0,0,.12)}.mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%]{border-top:1px solid rgba(0,0,0,.12);border-bottom:none}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{color:#212121}.mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:#757575}.mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#212121}.mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#757575}.mat-tab-group[class*=mat-background-][_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar[class*=mat-background-][_ngcontent-%COMP%]{border-bottom:none;border-top:none}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-toolbar[_ngcontent-%COMP%]{background:#f5f5f5;color:#212121}.mat-toolbar.mat-primary[_ngcontent-%COMP%]{background:#f57c00;color:#fff}.mat-toolbar.mat-accent[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-toolbar.mat-warn[_ngcontent-%COMP%]{background:#f44336;color:#fff}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:currentColor}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:inherit}.mat-toolbar[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:currentColor}.mat-tooltip[_ngcontent-%COMP%]{background:rgba(97,97,97,.9)}.mat-tree[_ngcontent-%COMP%]{background:#fff}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{color:#212121}.mat-snack-bar-container[_ngcontent-%COMP%]{color:rgba(255,255,255,.7);background:#323232;box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-simple-snackbar-action[_ngcontent-%COMP%]{color:#ff9800}[_nghost-%COMP%]{background-color:#fff;border:1px solid #ebebeb;display:flex;flex-direction:column;height:calc(100% - 2px);width:calc(100% - 2px)}.annotation-rows[_ngcontent-%COMP%]{display:flex;flex-direction:column;flex:1 1;overflow-y:auto}"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/annotations_list/annotations_list_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class AnnotationsListContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.annotationsExpanded$ = this.store.pipe(select(getAnnotationsExpanded));
            this.activeRuns$ = this.store
                .pipe(select(getCurrentRouteRunSelection))
                .pipe(map((/**
             * @param {?} runSelection
             * @return {?}
             */
            (runSelection) => {
                if (!runSelection)
                    return [];
                return Array.from(runSelection.entries())
                    .filter((/**
                 * @param {?} run
                 * @return {?}
                 */
                (run) => run[1]))
                    .map((/**
                 * @param {?} run
                 * @return {?}
                 */
                (run) => run[0]));
            })));
            this.numActiveRuns$ = this.activeRuns$.pipe(map((/**
             * @param {?} runs
             * @return {?}
             */
            (runs) => runs.length)));
            this.activeMetrics$ = combineLatest([
                this.store.select(getRunToMetrics),
                this.activeRuns$,
                this.store.select(getMetricFilters),
            ]).pipe(map((/**
             * @param {?} __0
             * @return {?}
             */
            ([runToMetrics, activeRuns, metricFilters]) => {
                /** @type {?} */
                let metrics = [];
                for (const run of activeRuns) {
                    if (runToMetrics[run]) {
                        metrics = metrics.concat(runToMetrics[run].filter((/**
                         * @param {?} key
                         * @return {?}
                         */
                        (key) => metricIsNpmiAndNotDiff(key))));
                    }
                }
                metrics = [...new Set([...Object.keys(metricFilters), ...metrics])];
                return metrics;
            })));
            this.visibleAnnotations$ = combineLatest([
                this.store.select(getAnnotationData),
                this.store.select(getHiddenAnnotations),
                this.store.select(getShowHiddenAnnotations),
            ]).pipe(map((/**
             * @param {?} __0
             * @return {?}
             */
            ([annotationData, hiddenAnnotations, showHiddenAnnotations]) => {
                return removeHiddenAnnotations(annotationData, hiddenAnnotations, showHiddenAnnotations);
            })));
            this.filteredAnnotations$ = combineLatest([
                this.visibleAnnotations$,
                this.store.select(getMetricArithmetic),
                this.store.select(getMetricFilters),
                this.activeRuns$,
                this.activeMetrics$,
                this.store.select(getAnnotationsRegex),
            ])
                .pipe(map((/**
             * @param {?} __0
             * @return {?}
             */
            ([visibleAnnotations, metricArithmetic, metricFilters, activeRuns, activeMetrics, annotationsRegex,]) => {
                return filterAnnotations(visibleAnnotations, activeRuns, metricArithmetic, metricFilters, activeMetrics, annotationsRegex);
            })))
                .pipe(share());
            this.numAnnotations$ = this.filteredAnnotations$.pipe(map((/**
             * @param {?} annotations
             * @return {?}
             */
            (annotations) => {
                return Object.keys(annotations).length;
            })));
            this.sortedAnnotations$ = combineLatest([
                this.filteredAnnotations$,
                this.store.pipe(select(getAnnotationSort)),
            ]).pipe(map((/**
             * @param {?} __0
             * @return {?}
             */
            ([annotations, sort]) => {
                return sortAnnotations(annotations, sort);
            })));
            this.selectedAnnotations$ = this.store.pipe(select(getSelectedAnnotations));
            this.maxCount$ = this.filteredAnnotations$.pipe(map((/**
             * @param {?} annotations
             * @return {?}
             */
            (annotations) => {
                /** @type {?} */
                let max = 0;
                Object.values(annotations).forEach((/**
                 * @param {?} annotation
                 * @return {?}
                 */
                (annotation) => {
                    annotation.forEach((/**
                     * @param {?} values
                     * @return {?}
                     */
                    (values) => {
                        if (values.countValue) {
                            max = Math.max(max, values.countValue);
                        }
                    }));
                }));
                return max;
            })));
        }
        /**
         * @param {?} annotations
         * @return {?}
         */
        rowClicked(annotations) {
            this.store.dispatch(npmiToggleSelectedAnnotations({ annotations }));
        }
    }
    AnnotationsListContainer.decorators = [
        { type: Component, args: [{
                    selector: 'npmi-annotations-list',
                    template: `
    <annotations-list-component
      [annotations]="filteredAnnotations$ | async"
      [annotationsExpanded]="annotationsExpanded$ | async"
      [numAnnotations]="numAnnotations$ | async"
      [activeMetrics]="activeMetrics$ | async"
      [numActiveRuns]="numActiveRuns$ | async"
      [sortedAnnotations]="sortedAnnotations$ | async"
      [selectedAnnotations]="selectedAnnotations$ | async"
      [maxCount]="maxCount$ | async"
      (onRowClick)="rowClicked($event)"
    ></annotations-list-component>
  `,
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    AnnotationsListContainer.ctorParameters = () => [
        { type: Store }
    ];
    /** @nocollapse */ AnnotationsListContainer.ɵfac = function AnnotationsListContainer_Factory(t) { return new (t || AnnotationsListContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ AnnotationsListContainer.ɵcmp = ɵɵdefineComponent({ type: AnnotationsListContainer, selectors: [["npmi-annotations-list"]], decls: 9, vars: 24, consts: [[3, "annotations", "annotationsExpanded", "numAnnotations", "activeMetrics", "numActiveRuns", "sortedAnnotations", "selectedAnnotations", "maxCount", "onRowClick"]], template: function AnnotationsListContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "annotations-list-component", 0);
            ɵɵlistener("onRowClick", function AnnotationsListContainer_Template_annotations_list_component_onRowClick_0_listener($event) { return ctx.rowClicked($event); });
            ɵɵpipe(1, "async");
            ɵɵpipe(2, "async");
            ɵɵpipe(3, "async");
            ɵɵpipe(4, "async");
            ɵɵpipe(5, "async");
            ɵɵpipe(6, "async");
            ɵɵpipe(7, "async");
            ɵɵpipe(8, "async");
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵproperty("annotations", ɵɵpipeBind1(1, 8, ctx.filteredAnnotations$))("annotationsExpanded", ɵɵpipeBind1(2, 10, ctx.annotationsExpanded$))("numAnnotations", ɵɵpipeBind1(3, 12, ctx.numAnnotations$))("activeMetrics", ɵɵpipeBind1(4, 14, ctx.activeMetrics$))("numActiveRuns", ɵɵpipeBind1(5, 16, ctx.numActiveRuns$))("sortedAnnotations", ɵɵpipeBind1(6, 18, ctx.sortedAnnotations$))("selectedAnnotations", ɵɵpipeBind1(7, 20, ctx.selectedAnnotations$))("maxCount", ɵɵpipeBind1(8, 22, ctx.maxCount$));
        } }, directives: [AnnotationsListComponent], pipes: [AsyncPipe], encapsulation: 2, changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/util/coordinate_data.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @param {?} annotationData
     * @param {?} selectedAnnotations
     * @param {?} activeRuns
     * @param {?} activeMetrics
     * @return {?}
     */
    function convertToCoordinateData(annotationData, selectedAnnotations, activeRuns, activeMetrics) {
        /** @type {?} */
        const result = [];
        /** @type {?} */
        const extremeValues = { max: -1.0, min: 1.0 };
        /** @type {?} */
        const allRuns = new Set(activeRuns);
        /** @type {?} */
        const allMetrics = new Set(activeMetrics);
        if (allRuns.size === 0 ||
            allMetrics.size === 0 ||
            Object.keys(annotationData).length === 0) {
            return { coordinates: [], extremes: { min: -1.0, max: 1.0 } };
        }
        selectedAnnotations.forEach((/**
         * @param {?} annotation
         * @return {?}
         */
        (annotation) => {
            /** @type {?} */
            const data = annotationData[annotation];
            /** @type {?} */
            const runResult = {};
            data.forEach((/**
             * @param {?} entry
             * @return {?}
             */
            (entry) => {
                if (!allRuns.has(entry.run) || !allMetrics.has(entry.metric)) {
                    return;
                }
                if (runResult[entry.run]) {
                    runResult[entry.run].push(entry);
                }
                else {
                    runResult[entry.run] = [entry];
                }
                if (entry.nPMIValue !== null) {
                    extremeValues.max = Math.max(extremeValues.max, entry.nPMIValue);
                    extremeValues.min = Math.min(extremeValues.min, entry.nPMIValue);
                }
                else {
                    extremeValues.max = Math.max(extremeValues.max, 0);
                    extremeValues.min = Math.min(extremeValues.min, 0);
                }
            }));
            for (const key of Object.keys(runResult)) {
                result.push({
                    annotation,
                    runId: key,
                    values: runResult[key],
                });
            }
        }));
        if (extremeValues.max < extremeValues.min) {
            extremeValues.max = 1.0;
            extremeValues.min = -1.0;
        }
        return { coordinates: result, extremes: extremeValues };
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/selected_annotations/parallel_coordinates/parallel_coordinates_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    const _c0$B = ["chart"];
    class ParallelCoordinatesComponent {
        constructor() {
            this.width = 0;
            this.chartWidth = 0;
            this.height = 300;
            this.margin = { top: 20, right: 40, bottom: 20, left: 40 };
            this.chartHeight = this.height - this.margin.top - this.margin.bottom;
            this.colorScale = (/**
             * @return {?}
             */
            () => '#333333');
        }
        /**
         * @param {?} event
         * @return {?}
         */
        onResize(event) {
            this.redraw();
        }
        /**
         * @return {?}
         */
        ngAfterViewInit() {
            /** @type {?} */
            const runsColorScale = ((/** @type {?} */ (document.createElement('tf-color-scale')))).runsColorScale;
            this.colorScale = runsColorScale ? runsColorScale : this.colorScale;
            this.svg = select$1(this.svgElement.nativeElement);
            this.mainContainer = this.svg
                .append('g')
                .attr('transform', `translate(${this.margin.left}, ${this.margin.top})`);
            this.coordinatesGroup = this.mainContainer.append('g');
            this.labelsGroup = this.mainContainer.append('g');
            this.axisGroup = this.mainContainer.append('g');
            this.xScale = point$1().padding(0.1);
            this.yScale = linear$1().range([this.chartHeight, 0]);
            this.yAxis = axisRight(this.yScale);
            this.redraw();
        }
        /**
         * @param {?} changes
         * @return {?}
         */
        ngOnChanges(changes) {
            if (this.svg) {
                this.redraw();
            }
        }
        /**
         * @private
         * @return {?}
         */
        redraw() {
            this.updateDimensions();
            this.updateAxes();
            this.draw();
        }
        /**
         * @private
         * @return {?}
         */
        updateDimensions() {
            this.width = this.svgElement.nativeElement.clientWidth || 10;
            this.chartWidth = this.width - this.margin.left - this.margin.right;
        }
        /**
         * @private
         * @return {?}
         */
        updateAxes() {
            this.xScale.rangeRound([0, this.chartWidth]).domain(this.activeMetrics);
            this.yScale.domain([
                this.coordinateData.extremes.min,
                this.coordinateData.extremes.max,
            ]);
        }
        /**
         * @private
         * @return {?}
         */
        draw() {
            this.drawAxes();
            this.drawAxisLabels();
            this.drawCoordinates();
            this.drawLabels();
        }
        /**
         * @private
         * @return {?}
         */
        drawAxes() {
            /** @type {?} */
            const axes = this.axisGroup
                .selectAll('.axis-y')
                .data(this.activeMetrics);
            /** @type {?} */
            const axisEnters = axes.enter().append('g').attr('class', 'axis-y');
            axisEnters
                .merge(axes)
                .attr('transform', (/**
             * @this {?}
             * @param {?} d
             * @return {?}
             */
            function (d) {
                return `translate(${this.xScale(d)}, 0)`;
            }).bind(this))
                .call((/** @type {?} */ (this.yAxis)));
            axes.exit().remove();
        }
        /**
         * @private
         * @return {?}
         */
        drawAxisLabels() {
            /** @type {?} */
            const axisBackgroundTexts = this.axisGroup
                .selectAll('.axis-bg-text')
                .data(this.activeMetrics);
            /** @type {?} */
            const axisBackgroundTextEnters = axisBackgroundTexts
                .enter()
                .append('text')
                .attr('class', 'axis-bg-text')
                .attr('font-size', '13px')
                .attr('stroke-width', 2)
                .attr('stroke-linejoin', 'round')
                .attr('stroke', 'white');
            axisBackgroundTextEnters
                .merge(axisBackgroundTexts)
                .text((/**
             * @param {?} d
             * @return {?}
             */
            (d) => d))
                .attr('transform', (/**
             * @this {?}
             * @param {?} d
             * @return {?}
             */
            function (d) {
                return `translate(${(/** @type {?} */ (this.xScale(d))) - 5}, ${this.yScale(this.coordinateData.extremes.min)}) rotate(-90)`;
            }).bind(this));
            axisBackgroundTexts.exit().remove();
            /** @type {?} */
            const axisTexts = this.axisGroup
                .selectAll('.axis-text')
                .data(this.activeMetrics);
            /** @type {?} */
            const axisTextEnters = axisTexts
                .enter()
                .append('text')
                .attr('font-size', '13px')
                .attr('class', 'axis-text');
            axisTextEnters
                .merge(axisTexts)
                .text((/**
             * @param {?} d
             * @return {?}
             */
            (d) => d))
                .attr('transform', (/**
             * @this {?}
             * @param {?} d
             * @return {?}
             */
            function (d) {
                return `translate(${(/** @type {?} */ (this.xScale(d))) - 5}, ${this.yScale(this.coordinateData.extremes.min)}) rotate(-90)`;
            }).bind(this));
            axisTexts.exit().remove();
        }
        /**
         * @private
         * @return {?}
         */
        drawCoordinates() {
            // Draw the coordinate paths
            /** @type {?} */
            const coords = this.coordinatesGroup
                .selectAll('.coord')
                .data(this.coordinateData.coordinates);
            /** @type {?} */
            const coordEnters = coords
                .enter()
                .append('path')
                .attr('class', 'coord')
                .attr('fill', 'none');
            coordEnters
                .merge(coords)
                .attr('d', this.path.bind(this))
                .attr('stroke', (/**
             * @this {?}
             * @param {?} d
             * @return {?}
             */
            function (d) {
                return this.colorScale(d.runId);
            }).bind(this));
            coords.exit().remove();
            // Draw invisible paths for a broader hover area
            /** @type {?} */
            const hiddenCoords = this.coordinatesGroup
                .selectAll('.hiddenCoord')
                .data(this.coordinateData.coordinates);
            /** @type {?} */
            const hiddenCoordEnters = hiddenCoords
                .enter()
                .append('path')
                .attr('class', 'hiddenCoord')
                .attr('stroke-width', '10px')
                .attr('fill', 'none')
                .attr('stroke', 'rgba(0, 0, 0, 0.0)')
                .on('mouseover', this.handleCoordinateMouseOver.bind(this))
                .on('mouseout', this.handleCoordinateMouseOut.bind(this));
            hiddenCoordEnters.merge(hiddenCoords).attr('d', this.path.bind(this));
            hiddenCoords.exit().remove();
        }
        /**
         * @private
         * @param {?} d
         * @return {?}
         */
        path(d) {
            /** @type {?} */
            const sorted = d.values.sort((/**
             * @param {?} a
             * @param {?} b
             * @return {?}
             */
            (a, b) => this.activeMetrics.indexOf(a.metric) -
                this.activeMetrics.indexOf(b.metric)));
            return line()(d.values.map((/**
             * @this {?}
             * @param {?} p
             * @return {?}
             */
            function (p) {
                /** @type {?} */
                let yPos = this.yScale(p.nPMIValue);
                return (/** @type {?} */ ([this.xScale(p.metric), yPos]));
            }).bind(this)));
        }
        /**
         * @private
         * @this {?}
         * @param {?} d
         * @param {?} i
         * @return {?}
         */
        handleCoordinateMouseOver(d, i) {
            this.labelsGroup
                .selectAll('.coordinate-label')
                .filter((/**
             * @param {?} x
             * @return {?}
             */
            function (x) {
                return !(x.annotation === d.annotation);
            }))
                .style('opacity', 0.1);
            this.coordinatesGroup
                .selectAll('.coord')
                .filter((/**
             * @param {?} x
             * @return {?}
             */
            function (x) {
                return !(x.annotation === d.annotation);
            }))
                .style('opacity', 0.1);
        }
        /**
         * @private
         * @this {?}
         * @return {?}
         */
        handleCoordinateMouseOut() {
            this.labelsGroup.selectAll('.coordinate-label').style('opacity', 1.0);
            this.coordinatesGroup.selectAll('.coord').style('opacity', 1.0);
        }
        /**
         * @private
         * @return {?}
         */
        drawLabels() {
            /** @type {?} */
            const interpolationFactor = 30 / this.xScale.step();
            /** @type {?} */
            const data = this.coordinateData.coordinates.length < 30
                ? this.coordinateData.coordinates
                : [];
            /** @type {?} */
            const coordinateLabels = this.labelsGroup
                .selectAll('.coordinate-label')
                .data(data);
            /** @type {?} */
            const coordinateLabelEnters = coordinateLabels
                .enter()
                .append('text')
                .attr('class', 'coordinate-label')
                .attr('font-size', '10px');
            coordinateLabelEnters
                .merge(coordinateLabels)
                .text((/**
             * @param {?} d
             * @return {?}
             */
            function (d) {
                return d.annotation;
            }))
                .attr('x', (/** @type {?} */ (this.xScale(this.activeMetrics[0]))) + 30)
                .attr('y', (/**
             * @this {?}
             * @param {?} d
             * @return {?}
             */
            function (d) {
                /** @type {?} */
                const y0 = this.yScale(d.values[0].nPMIValue ? d.values[0].nPMIValue : 0);
                /** @type {?} */
                const y1 = this.yScale(d.values[1].nPMIValue ? d.values[1].nPMIValue : 0);
                return (1 - interpolationFactor) * y0 + interpolationFactor * y1;
            }).bind(this));
            coordinateLabels.exit().remove();
        }
    }
    ParallelCoordinatesComponent.decorators = [
        { type: Component, args: [{
                    selector: 'parallel-coordinates-component',
                    templateUrl: './parallel_coordinates_component.ng.html',
                    styleUrls: ['./parallel_coordinates_component.css'],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    ParallelCoordinatesComponent.propDecorators = {
        activeMetrics: [{ type: Input }],
        coordinateData: [{ type: Input }],
        sidebarWidth: [{ type: Input }],
        svgElement: [{ type: ViewChild, args: ['chart', { static: true, read: ElementRef },] }],
        onResize: [{ type: HostListener, args: ['window:resize', ['$event'],] }]
    };
    /** @nocollapse */ ParallelCoordinatesComponent.ɵfac = function ParallelCoordinatesComponent_Factory(t) { return new (t || ParallelCoordinatesComponent)(); };
    /** @nocollapse */ ParallelCoordinatesComponent.ɵcmp = ɵɵdefineComponent({ type: ParallelCoordinatesComponent, selectors: [["parallel-coordinates-component"]], viewQuery: function ParallelCoordinatesComponent_Query(rf, ctx) { if (rf & 1) {
            ɵɵstaticViewQuery(_c0$B, true, ElementRef);
        } if (rf & 2) {
            var _t;
            ɵɵqueryRefresh(_t = ɵɵloadQuery()) && (ctx.svgElement = _t.first);
        } }, hostBindings: function ParallelCoordinatesComponent_HostBindings(rf, ctx) { if (rf & 1) {
            ɵɵlistener("resize", function ParallelCoordinatesComponent_resize_HostBindingHandler($event) { return ctx.onResize($event); }, false, ɵɵresolveWindow);
        } }, inputs: { activeMetrics: "activeMetrics", coordinateData: "coordinateData", sidebarWidth: "sidebarWidth" }, features: [ɵɵNgOnChangesFeature], decls: 2, vars: 0, consts: [[1, "pc-chart"], ["chart", ""]], template: function ParallelCoordinatesComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵnamespaceSVG();
            ɵɵelement(0, "svg", 0, 1);
        } }, styles: [".pc-chart[_ngcontent-%COMP%]{height:300px;width:100%}"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/selected_annotations/parallel_coordinates/parallel_coordinates_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class ParallelCoordinatesContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.activeRuns$ = this.store
                .pipe(select(getCurrentRouteRunSelection))
                .pipe(map((/**
             * @param {?} runSelection
             * @return {?}
             */
            (runSelection) => {
                if (!runSelection)
                    return [];
                return Array.from(runSelection.entries())
                    .filter((/**
                 * @param {?} run
                 * @return {?}
                 */
                (run) => run[1]))
                    .map((/**
                 * @param {?} run
                 * @return {?}
                 */
                (run) => run[0]));
            })));
            this.activeMetrics$ = combineLatest(this.store.select(getRunToMetrics), this.activeRuns$, this.store.select(getMetricFilters)).pipe(map((/**
             * @param {?} __0
             * @return {?}
             */
            ([runToMetrics, activeRuns, metricFilters]) => {
                /** @type {?} */
                let metrics = [];
                for (const run of activeRuns) {
                    if (runToMetrics[run]) {
                        metrics = metrics.concat(runToMetrics[run].filter((/**
                         * @param {?} key
                         * @return {?}
                         */
                        (key) => metricIsNpmiAndNotDiff(key))));
                    }
                }
                metrics = [...new Set([...Object.keys(metricFilters), ...metrics])];
                return metrics.map((/**
                 * @param {?} metric
                 * @return {?}
                 */
                (metric) => stripMetricString(metric)));
            })));
            this.coordinateData$ = combineLatest([
                this.store.select(getAnnotationData),
                this.store.select(getSelectedAnnotations),
                this.activeRuns$,
                this.activeMetrics$,
            ]).pipe(map((/**
             * @param {?} __0
             * @return {?}
             */
            ([annotationData, selectedAnnotations, runs, metrics]) => {
                return convertToCoordinateData(annotationData, selectedAnnotations, runs, metrics);
            })));
            this.sidebarWidth$ = this.store.select(getSidebarWidth);
        }
    }
    ParallelCoordinatesContainer.decorators = [
        { type: Component, args: [{
                    selector: 'npmi-parallel-coordinates',
                    template: `
    <parallel-coordinates-component
      [activeMetrics]="activeMetrics$ | async"
      [coordinateData]="coordinateData$ | async"
      [sidebarWidth]="sidebarWidth$ | async"
    ></parallel-coordinates-component>
  `,
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    ParallelCoordinatesContainer.ctorParameters = () => [
        { type: Store }
    ];
    /** @nocollapse */ ParallelCoordinatesContainer.ɵfac = function ParallelCoordinatesContainer_Factory(t) { return new (t || ParallelCoordinatesContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ ParallelCoordinatesContainer.ɵcmp = ɵɵdefineComponent({ type: ParallelCoordinatesContainer, selectors: [["npmi-parallel-coordinates"]], decls: 4, vars: 9, consts: [[3, "activeMetrics", "coordinateData", "sidebarWidth"]], template: function ParallelCoordinatesContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelement(0, "parallel-coordinates-component", 0);
            ɵɵpipe(1, "async");
            ɵɵpipe(2, "async");
            ɵɵpipe(3, "async");
        } if (rf & 2) {
            ɵɵproperty("activeMetrics", ɵɵpipeBind1(1, 3, ctx.activeMetrics$))("coordinateData", ɵɵpipeBind1(2, 5, ctx.coordinateData$))("sidebarWidth", ɵɵpipeBind1(3, 7, ctx.sidebarWidth$));
        } }, directives: [ParallelCoordinatesComponent], pipes: [AsyncPipe], encapsulation: 2, changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/selected_annotations/selected_annotations_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    var I18N_0$a;
    if (typeof ngI18nClosureMode !== "undefined" && ngI18nClosureMode) {
        /**
         * @desc Label for a button that clears the annotation selection.
         */ 
        const MSG_EXTERNAL_2494660520341308830$$TENSORBOARD_WEBAPP_PLUGINS_NPMI_VIEWS_SELECTED_ANNOTATIONS_SELECTED_ANNOTATIONS_COMPONENT_TS_1 = goog.getMsg("Clear Annotation Selection");
        I18N_0$a = MSG_EXTERNAL_2494660520341308830$$TENSORBOARD_WEBAPP_PLUGINS_NPMI_VIEWS_SELECTED_ANNOTATIONS_SELECTED_ANNOTATIONS_COMPONENT_TS_1;
    }
    else {
        I18N_0$a = $localize `:Label for a button that clears the annotation selection.␟2b52289f2c4b7f5b182ed23c9154722ecd46a2d4␟2494660520341308830:Clear Annotation Selection`;
    }
    const _c2$h = ["aria-label", I18N_0$a];
    var I18N_3$7;
    if (typeof ngI18nClosureMode !== "undefined" && ngI18nClosureMode) {
        /**
         * @desc Label for a button that expands or hides selected annotations.
         */ 
        const MSG_EXTERNAL_5670797784070952095$$TENSORBOARD_WEBAPP_PLUGINS_NPMI_VIEWS_SELECTED_ANNOTATIONS_SELECTED_ANNOTATIONS_COMPONENT_TS_4 = goog.getMsg("Expand/Hide Selected Annotations");
        I18N_3$7 = MSG_EXTERNAL_5670797784070952095$$TENSORBOARD_WEBAPP_PLUGINS_NPMI_VIEWS_SELECTED_ANNOTATIONS_SELECTED_ANNOTATIONS_COMPONENT_TS_4;
    }
    else {
        I18N_3$7 = $localize `:Label for a button that expands or hides selected annotations.␟8f4ad305c19e3655f3189e3e264e83fb76f7bb95␟5670797784070952095:Expand/Hide Selected Annotations`;
    }
    const _c5$9 = ["aria-label", I18N_3$7];
    function SelectedAnnotationsComponent_mat_icon_9_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "mat-icon", 9);
    } }
    function SelectedAnnotationsComponent_ng_template_10_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "mat-icon", 10);
    } }
    function SelectedAnnotationsComponent_npmi_parallel_coordinates_12_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "npmi-parallel-coordinates");
    } }
    class SelectedAnnotationsComponent {
        constructor() {
            this.onClearSelectedAnnotations = new EventEmitter();
            this.onToggleExpanded = new EventEmitter();
        }
    }
    SelectedAnnotationsComponent.decorators = [
        { type: Component, args: [{
                    selector: 'selected-annotations-component',
                    templateUrl: './selected_annotations_component.ng.html',
                    styleUrls: ['./selected_annotations_component.css'],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    SelectedAnnotationsComponent.propDecorators = {
        pcExpanded: [{ type: Input }],
        selectedAnnotations: [{ type: Input }],
        onClearSelectedAnnotations: [{ type: Output }],
        onToggleExpanded: [{ type: Output }]
    };
    /** @nocollapse */ SelectedAnnotationsComponent.ɵfac = function SelectedAnnotationsComponent_Factory(t) { return new (t || SelectedAnnotationsComponent)(); };
    /** @nocollapse */ SelectedAnnotationsComponent.ɵcmp = ɵɵdefineComponent({ type: SelectedAnnotationsComponent, selectors: [["selected-annotations-component"]], inputs: { pcExpanded: "pcExpanded", selectedAnnotations: "selectedAnnotations" }, outputs: { onClearSelectedAnnotations: "onClearSelectedAnnotations", onToggleExpanded: "onToggleExpanded" }, decls: 13, vars: 4, consts: [[1, "pc-container"], [1, "pc-toolbar"], [1, "pc-title"], ["mat-icon-button", "", "title", "Deselects all selected annotations.", 1, "clear-button", 3, "disabled", "click", 6, "aria-label"], ["svgIcon", "clear_24px"], ["mat-icon-button", "", 1, "expand-button", 3, "click", 6, "aria-label"], ["svgIcon", "expand_less_24px", "class", "expand-less-icon", 4, "ngIf", "ngIfElse"], ["notExpanded", ""], [4, "ngIf"], ["svgIcon", "expand_less_24px", 1, "expand-less-icon"], ["svgIcon", "expand_more_24px", 1, "expand-icon"]], template: function SelectedAnnotationsComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "div", 0);
            ɵɵelementStart(1, "div", 1);
            ɵɵelementStart(2, "h3", 2);
            ɵɵtext(3, "Selected Annotations");
            ɵɵelementEnd();
            ɵɵelementStart(4, "button", 3);
            ɵɵi18nAttributes(5, _c2$h);
            ɵɵlistener("click", function SelectedAnnotationsComponent_Template_button_click_4_listener() { return ctx.onClearSelectedAnnotations.emit(); });
            ɵɵelement(6, "mat-icon", 4);
            ɵɵelementEnd();
            ɵɵelementStart(7, "button", 5);
            ɵɵi18nAttributes(8, _c5$9);
            ɵɵlistener("click", function SelectedAnnotationsComponent_Template_button_click_7_listener() { return ctx.onToggleExpanded.emit(); });
            ɵɵtemplate(9, SelectedAnnotationsComponent_mat_icon_9_Template, 1, 0, "mat-icon", 6);
            ɵɵtemplate(10, SelectedAnnotationsComponent_ng_template_10_Template, 1, 0, "ng-template", null, 7, ɵɵtemplateRefExtractor);
            ɵɵelementEnd();
            ɵɵelementEnd();
            ɵɵtemplate(12, SelectedAnnotationsComponent_npmi_parallel_coordinates_12_Template, 1, 0, "npmi-parallel-coordinates", 8);
            ɵɵelementEnd();
        } if (rf & 2) {
            const _r1 = ɵɵreference(11);
            ɵɵadvance(4);
            ɵɵproperty("disabled", ctx.selectedAnnotations.length === 0);
            ɵɵadvance(5);
            ɵɵproperty("ngIf", ctx.pcExpanded)("ngIfElse", _r1);
            ɵɵadvance(3);
            ɵɵproperty("ngIf", ctx.pcExpanded);
        } }, directives: [MatButton, MatIcon, NgIf, ParallelCoordinatesContainer], styles: [".mat-badge-content[_ngcontent-%COMP%]{font-weight:600;font-size:12px;font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:9px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:24px}.mat-h1[_ngcontent-%COMP%], .mat-headline[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%]{font:400 24px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h2[_ngcontent-%COMP%], .mat-title[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h3[_ngcontent-%COMP%], .mat-subheading-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%]{font:400 16px/28px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h4[_ngcontent-%COMP%], .mat-subheading-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%]{font:400 15px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h5[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%]{font:400 calc(14px * 0.83)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-h6[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:400 calc(14px * 0.67)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-body-strong[_ngcontent-%COMP%], .mat-body-2[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   p[_ngcontent-%COMP%]{margin:0 0 12px}.mat-small[_ngcontent-%COMP%], .mat-caption[_ngcontent-%COMP%]{font:400 12px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-display-4[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-4[_ngcontent-%COMP%]{font:300 112px/112px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.05em;margin:0 0 56px}.mat-display-3[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-3[_ngcontent-%COMP%]{font:400 56px/56px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.02em;margin:0 0 64px}.mat-display-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-2[_ngcontent-%COMP%]{font:400 45px/48px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.005em;margin:0 0 64px}.mat-display-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-1[_ngcontent-%COMP%]{font:400 34px/40px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 64px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%], .mat-flat-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-button-toggle[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card-title[_ngcontent-%COMP%]{font-size:24px;font-weight:500}.mat-card-header[_ngcontent-%COMP%]   .mat-card-title[_ngcontent-%COMP%]{font-size:20px}.mat-card-subtitle[_ngcontent-%COMP%], .mat-card-content[_ngcontent-%COMP%]{font-size:14px}.mat-checkbox[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-checkbox-layout[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{line-height:24px}.mat-chip[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-chip[_ngcontent-%COMP%]   .mat-chip-trailing-icon.mat-icon[_ngcontent-%COMP%], .mat-chip[_ngcontent-%COMP%]   .mat-chip-remove.mat-icon[_ngcontent-%COMP%]{font-size:18px}.mat-table[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-header-cell[_ngcontent-%COMP%]{font-size:12px;font-weight:500}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{font-size:14px}.mat-calendar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-calendar-body[_ngcontent-%COMP%]{font-size:13px}.mat-calendar-body-label[_ngcontent-%COMP%], .mat-calendar-period-button[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-calendar-table-header[_ngcontent-%COMP%]   th[_ngcontent-%COMP%]{font-size:11px;font-weight:400}.mat-dialog-title[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-expansion-panel-header[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:15px;font-weight:400}.mat-expansion-panel-content[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field[_ngcontent-%COMP%]{font-size:inherit;font-weight:400;line-height:1.125;font-family:Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.34375em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{font-size:150%;line-height:1.125}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]{height:1.5em;width:1.5em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{height:1.125em;line-height:1.125}.mat-form-field-infix[_ngcontent-%COMP%]{padding:.5em 0;border-top:.84375em solid transparent}.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34375em) scale(0.75);width:133.3333333333%}.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34374em) scale(0.75);width:133.3333433333%}.mat-form-field-label-wrapper[_ngcontent-%COMP%]{top:-0.84375em;padding-top:.84375em}.mat-form-field-label[_ngcontent-%COMP%]{top:1.34375em}.mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.34375em}.mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{font-size:75%;margin-top:.6666666667em;top:calc(100% - 1.7916666667em)}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.4375em 0}.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.001px);-ms-transform:translateY(-1.28125em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00101px);-ms-transform:translateY(-1.28124em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00102px);-ms-transform:translateY(-1.28123em) scale(0.75);width:133.3333533333%}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.28125em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{margin-top:.5416666667em;top:calc(100% - 1.6666666667em)}@media print{.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28122em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28121em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.2812em) scale(0.75)}}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.25em 0 .75em 0}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.09375em;margin-top:-0.5em}.mat-form-field-appearance-fill.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59374em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:1em 0 1em 0}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.84375em;margin-top:-0.25em}.mat-form-field-appearance-outline.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59374em) scale(0.75);width:133.3333433333%}.mat-grid-tile-header[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]{font-size:14px}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2), .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}input.mat-input-element[_ngcontent-%COMP%]{margin-top:-0.0625em}.mat-menu-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:400}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px}.mat-radio-button[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select-trigger[_ngcontent-%COMP%]{height:1.125em}.mat-slide-toggle-content[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-slider-thumb-label-text[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-stepper-vertical[_ngcontent-%COMP%], .mat-stepper-horizontal[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-step-label[_ngcontent-%COMP%]{font-size:14px;font-weight:400}.mat-step-sub-label-error[_ngcontent-%COMP%]{font-weight:normal}.mat-step-label-error[_ngcontent-%COMP%]{font-size:14px}.mat-step-label-selected[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-tab-group[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-toolbar[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0}.mat-tooltip[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:10px;padding-top:6px;padding-bottom:6px}.mat-tooltip-handset[_ngcontent-%COMP%]{font-size:14px;padding-top:8px;padding-bottom:8px}.mat-list-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:16px}.mat-optgroup-label[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-simple-snackbar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px}.mat-simple-snackbar-action[_ngcontent-%COMP%]{line-height:1;font-family:inherit;font-size:inherit;font-weight:500}.mat-tree[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{font-weight:400;font-size:14px}.mat-ripple[_ngcontent-%COMP%]{overflow:hidden;position:relative}.mat-ripple[_ngcontent-%COMP%]:not(:empty){transform:translateZ(0)}.mat-ripple.mat-ripple-unbounded[_ngcontent-%COMP%]{overflow:visible}.mat-ripple-element[_ngcontent-%COMP%]{position:absolute;border-radius:50%;pointer-events:none;transition:opacity,transform 0ms cubic-bezier(0, 0, 0.2, 1);transform:scale(0)}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{display:none}.cdk-visually-hidden[_ngcontent-%COMP%]{border:0;clip:rect(0 0 0 0);height:1px;margin:-1px;overflow:hidden;padding:0;position:absolute;width:1px;outline:0;-webkit-appearance:none;-moz-appearance:none}.cdk-overlay-container[_ngcontent-%COMP%], .cdk-global-overlay-wrapper[_ngcontent-%COMP%]{pointer-events:none;top:0;left:0;height:100%;width:100%}.cdk-overlay-container[_ngcontent-%COMP%]{position:fixed;z-index:1000}.cdk-overlay-container[_ngcontent-%COMP%]:empty{display:none}.cdk-global-overlay-wrapper[_ngcontent-%COMP%]{display:flex;position:absolute;z-index:1000}.cdk-overlay-pane[_ngcontent-%COMP%]{position:absolute;pointer-events:auto;box-sizing:border-box;z-index:1000;display:flex;max-width:100%;max-height:100%}.cdk-overlay-backdrop[_ngcontent-%COMP%]{position:absolute;top:0;bottom:0;left:0;right:0;z-index:1000;pointer-events:auto;-webkit-tap-highlight-color:transparent;transition:opacity 400ms cubic-bezier(0.25, 0.8, 0.25, 1);opacity:0}.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:1}@media screen and (-ms-high-contrast: active){.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:.6}}.cdk-overlay-dark-backdrop[_ngcontent-%COMP%]{background:rgba(0,0,0,.32)}.cdk-overlay-transparent-backdrop[_ngcontent-%COMP%], .cdk-overlay-transparent-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:0}.cdk-overlay-connected-position-bounding-box[_ngcontent-%COMP%]{position:absolute;z-index:1000;display:flex;flex-direction:column;min-width:1px;min-height:1px}.cdk-global-scrollblock[_ngcontent-%COMP%]{position:fixed;width:100%;overflow-y:scroll}@keyframes cdk-text-field-autofill-start{}@keyframes cdk-text-field-autofill-end{}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:-webkit-autofill{animation:cdk-text-field-autofill-start 0s 1ms}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:not(:-webkit-autofill){animation:cdk-text-field-autofill-end 0s 1ms}textarea.cdk-textarea-autosize[_ngcontent-%COMP%]{resize:none}textarea.cdk-textarea-autosize-measuring[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:auto !important;overflow:hidden !important}textarea.cdk-textarea-autosize-measuring-firefox[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:0 !important}.mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.1)}.mat-option[_ngcontent-%COMP%]{color:#212121}.mat-option[_ngcontent-%COMP%]:hover:not(.mat-option-disabled), .mat-option[_ngcontent-%COMP%]:focus:not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple):not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-active[_ngcontent-%COMP%]{background:rgba(0,0,0,.04);color:#212121}.mat-option.mat-option-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-primary[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f57c00}.mat-accent[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f44336}.mat-optgroup-label[_ngcontent-%COMP%]{color:#616161}.mat-optgroup-disabled[_ngcontent-%COMP%]   .mat-optgroup-label[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-pseudo-checkbox[_ngcontent-%COMP%]{color:#616161}.mat-pseudo-checkbox[_ngcontent-%COMP%]::after{color:#fafafa}.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{color:#b0b0b0}.mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f57c00}.mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f44336}.mat-pseudo-checkbox-checked.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{background:#b0b0b0}.mat-elevation-z0[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z1[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z2[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z3[_ngcontent-%COMP%]{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z4[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z5[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 5px 8px 0px rgba(0, 0, 0, 0.14),0px 1px 14px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z6[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z7[_ngcontent-%COMP%]{box-shadow:0px 4px 5px -2px rgba(0, 0, 0, 0.2),0px 7px 10px 1px rgba(0, 0, 0, 0.14),0px 2px 16px 1px rgba(0, 0, 0, 0.12)}.mat-elevation-z8[_ngcontent-%COMP%]{box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z9[_ngcontent-%COMP%]{box-shadow:0px 5px 6px -3px rgba(0, 0, 0, 0.2),0px 9px 12px 1px rgba(0, 0, 0, 0.14),0px 3px 16px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z10[_ngcontent-%COMP%]{box-shadow:0px 6px 6px -3px rgba(0, 0, 0, 0.2),0px 10px 14px 1px rgba(0, 0, 0, 0.14),0px 4px 18px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z11[_ngcontent-%COMP%]{box-shadow:0px 6px 7px -4px rgba(0, 0, 0, 0.2),0px 11px 15px 1px rgba(0, 0, 0, 0.14),0px 4px 20px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z12[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z13[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 13px 19px 2px rgba(0, 0, 0, 0.14),0px 5px 24px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z14[_ngcontent-%COMP%]{box-shadow:0px 7px 9px -4px rgba(0, 0, 0, 0.2),0px 14px 21px 2px rgba(0, 0, 0, 0.14),0px 5px 26px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z15[_ngcontent-%COMP%]{box-shadow:0px 8px 9px -5px rgba(0, 0, 0, 0.2),0px 15px 22px 2px rgba(0, 0, 0, 0.14),0px 6px 28px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z16[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z17[_ngcontent-%COMP%]{box-shadow:0px 8px 11px -5px rgba(0, 0, 0, 0.2),0px 17px 26px 2px rgba(0, 0, 0, 0.14),0px 6px 32px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z18[_ngcontent-%COMP%]{box-shadow:0px 9px 11px -5px rgba(0, 0, 0, 0.2),0px 18px 28px 2px rgba(0, 0, 0, 0.14),0px 7px 34px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z19[_ngcontent-%COMP%]{box-shadow:0px 9px 12px -6px rgba(0, 0, 0, 0.2),0px 19px 29px 2px rgba(0, 0, 0, 0.14),0px 7px 36px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z20[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 20px 31px 3px rgba(0, 0, 0, 0.14),0px 8px 38px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z21[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 21px 33px 3px rgba(0, 0, 0, 0.14),0px 8px 40px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z22[_ngcontent-%COMP%]{box-shadow:0px 10px 14px -6px rgba(0, 0, 0, 0.2),0px 22px 35px 3px rgba(0, 0, 0, 0.14),0px 8px 42px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z23[_ngcontent-%COMP%]{box-shadow:0px 11px 14px -7px rgba(0, 0, 0, 0.2),0px 23px 36px 3px rgba(0, 0, 0, 0.14),0px 9px 44px 8px rgba(0, 0, 0, 0.12)}.mat-elevation-z24[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12)}.mat-app-background[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-theme-loaded-marker[_ngcontent-%COMP%]{display:none}.mat-autocomplete-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-autocomplete-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover){background:#fff}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover):not(.mat-option-disabled){color:#212121}.mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f57c00}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{outline:solid 1px;border-radius:0}.mat-badge-accent[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-badge-warn[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f44336}.mat-badge[_ngcontent-%COMP%]{position:relative}.mat-badge-hidden[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{display:none}.mat-badge-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#b9b9b9;color:#757575}.mat-badge-content[_ngcontent-%COMP%]{position:absolute;text-align:center;display:inline-block;border-radius:50%;transition:transform 200ms ease-in-out;transform:scale(0.6);overflow:hidden;white-space:nowrap;text-overflow:ellipsis;pointer-events:none}.ng-animate-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%], .mat-badge-content._mat-animation-noopable[_ngcontent-%COMP%]{transition:none}.mat-badge-content.mat-badge-active[_ngcontent-%COMP%]{transform:none}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:16px;height:16px;line-height:16px}.mat-badge-small.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-8px}.mat-badge-small.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-8px}.mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-16px}.mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-16px}.mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-8px}.mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-8px}.mat-badge-medium[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:22px;height:22px;line-height:22px}.mat-badge-medium.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-11px}.mat-badge-medium.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-11px}.mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-22px}.mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-22px}.mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-11px}.mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-11px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:28px;height:28px;line-height:28px}.mat-badge-large.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-14px}.mat-badge-large.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-14px}.mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-28px}.mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-28px}.mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-14px}.mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-14px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]{color:inherit;background:transparent}.mat-button.mat-primary[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-button[disabled][disabled][_ngcontent-%COMP%], .mat-icon-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-icon-button[disabled][disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-stroked-button[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f44336}.mat-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:transparent}.mat-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{opacity:.1;background-color:currentColor}.mat-button-focus-overlay[_ngcontent-%COMP%]{background:#000}.mat-stroked-button[_ngcontent-%COMP%]:not([disabled]){border-color:rgba(0,0,0,.12)}.mat-flat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{color:#212121;background-color:#fff}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-flat-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-stroked-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-flat-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-raised-button[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone[_ngcontent-%COMP%], .mat-button-toggle-group[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{box-shadow:none}.mat-button-toggle[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-button-toggle[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121;background:#fff}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:#000}.mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12)}[dir=rtl][_ngcontent-%COMP%]   .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-group-appearance-standard.mat-button-toggle-vertical[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:none;border-top:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#616161}.mat-button-toggle-checked.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121}.mat-button-toggle-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.26);background-color:#eee}.mat-button-toggle-disabled.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{background:#fff}.mat-button-toggle-disabled.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#bdbdbd}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{border:solid 1px rgba(0,0,0,.12)}.mat-card[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-card[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-card.mat-card-flat[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-card-subtitle[_ngcontent-%COMP%]{color:#616161}.mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#616161}.mat-checkbox-checkmark[_ngcontent-%COMP%]{fill:#fafafa}.mat-checkbox-checkmark-path[_ngcontent-%COMP%]{stroke:#fafafa !important}.mat-checkbox-mixedmark[_ngcontent-%COMP%]{background-color:#fafafa}.mat-checkbox-indeterminate.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f57c00}.mat-checkbox-indeterminate.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#ff9800}.mat-checkbox-indeterminate.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f44336}.mat-checkbox-disabled.mat-checkbox-checked[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-disabled.mat-checkbox-indeterminate[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]:not(.mat-checkbox-checked)   .mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{color:#616161}.mat-checkbox[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%]{background:#f57c00}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%]{background:#ff9800}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%]{background:#f44336}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#212121}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#212121;opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled):active{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled)   .mat-chip-remove[_ngcontent-%COMP%]:hover{opacity:.54}.mat-chip.mat-standard-chip.mat-chip-disabled[_ngcontent-%COMP%]{opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]::after{background:#000}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-table[_ngcontent-%COMP%]{background:#fff}.mat-table[_ngcontent-%COMP%]   thead[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tbody[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tfoot[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], [mat-header-row][_ngcontent-%COMP%], [mat-row][_ngcontent-%COMP%], [mat-footer-row][_ngcontent-%COMP%], .mat-table-sticky[_ngcontent-%COMP%]{background:inherit}mat-row[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], th.mat-header-cell[_ngcontent-%COMP%], td.mat-cell[_ngcontent-%COMP%], td.mat-footer-cell[_ngcontent-%COMP%]{border-bottom-color:rgba(0,0,0,.12)}.mat-header-cell[_ngcontent-%COMP%]{color:#616161}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{color:#212121}.mat-calendar-arrow[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.54)}.mat-datepicker-toggle[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-next-button[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-previous-button[_ngcontent-%COMP%]{color:rgba(0,0,0,.54)}.mat-calendar-table-header[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-calendar-table-header-divider[_ngcontent-%COMP%]::after{background:rgba(0,0,0,.12)}.mat-calendar-body-label[_ngcontent-%COMP%]{color:#616161}.mat-calendar-body-cell-content[_ngcontent-%COMP%]{color:#212121;border-color:transparent}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){color:#757575}.mat-calendar-body-cell[_ngcontent-%COMP%]:not(.mat-calendar-body-disabled):hover > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-keyboard-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-program-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){background-color:rgba(0,0,0,.04)}.mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.38)}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.18)}.mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.4)}.mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12);background-color:#fff;color:#212121}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.4)}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.4)}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content-touch[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-datepicker-toggle-active[_ngcontent-%COMP%]{color:#f57c00}.mat-datepicker-toggle-active.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-datepicker-toggle-active.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-dialog-container[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-divider[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-divider-vertical[_ngcontent-%COMP%]{border-right-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-expansion-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-action-row[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-keyboard-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-program-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded)   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover:not([aria-disabled=true]){background:rgba(0,0,0,.04)}@media(hover: none){.mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded):not([aria-disabled=true])   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover{background:#fff}}.mat-expansion-panel-header-title[_ngcontent-%COMP%]{color:#212121}.mat-expansion-panel-header-description[_ngcontent-%COMP%], .mat-expansion-indicator[_ngcontent-%COMP%]::after{color:#616161}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-title[_ngcontent-%COMP%], .mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-description[_ngcontent-%COMP%]{color:inherit}.mat-form-field-label[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-hint[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-focused[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-ripple[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid)   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f57c00}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-accent   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#ff9800}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-warn   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#f44336}.mat-error[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-hint[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-legacy.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-standard[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-standard.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.04)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.02)}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:transparent}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.12)}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:rgba(0,0,0,.87)}.mat-form-field-appearance-outline.mat-focused[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field-appearance-outline.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-appearance-outline.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-invalid.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.06)}.mat-icon.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-icon.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-icon.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-type-mat-native-select[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#616161}.mat-input-element[_ngcontent-%COMP%]:disabled, .mat-form-field-type-mat-native-select.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#757575}.mat-input-element[_ngcontent-%COMP%]{caret-color:#f57c00}.mat-input-element[_ngcontent-%COMP%]::placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-moz-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-webkit-input-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]:-ms-input-placeholder{color:rgba(97,97,97,.42)}.mat-accent[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%], .mat-form-field-invalid[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#f44336}.mat-form-field-type-mat-native-select.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{color:#616161}.mat-list-item-disabled[_ngcontent-%COMP%]{background-color:#eee}.mat-list-option[_ngcontent-%COMP%]:hover, .mat-list-option[_ngcontent-%COMP%]:focus, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.04)}.mat-list-single-selected-option[_ngcontent-%COMP%], .mat-list-single-selected-option[_ngcontent-%COMP%]:hover, .mat-list-single-selected-option[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.12)}.mat-menu-panel[_ngcontent-%COMP%]{background:#fff}.mat-menu-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-menu-item[_ngcontent-%COMP%]{background:transparent;color:#212121}.mat-menu-item[disabled][_ngcontent-%COMP%], .mat-menu-item[disabled][_ngcontent-%COMP%]::after{color:rgba(0,0,0,.38)}.mat-menu-item[_ngcontent-%COMP%]   .mat-icon-no-color[_ngcontent-%COMP%], .mat-menu-item-submenu-trigger[_ngcontent-%COMP%]::after{color:rgba(0,0,0,.54)}.mat-menu-item[_ngcontent-%COMP%]:hover:not([disabled]), .mat-menu-item.cdk-program-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item.cdk-keyboard-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item-highlighted[_ngcontent-%COMP%]:not([disabled]){background:rgba(0,0,0,.04)}.mat-paginator[_ngcontent-%COMP%]{background:#fff}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{color:#616161}.mat-paginator-decrement[_ngcontent-%COMP%], .mat-paginator-increment[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54);border-right:2px solid rgba(0,0,0,.54)}.mat-paginator-first[_ngcontent-%COMP%], .mat-paginator-last[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54)}.mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-decrement[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-increment[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-first[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-last[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f57c00}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#ff9800}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f44336}.mat-progress-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f57c00}.mat-progress-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#ff9800}.mat-progress-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f44336}.mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#616161}.mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f57c00}.mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#ff9800}.mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#ff9800}.mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f44336}.mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f44336}.mat-radio-button.mat-radio-disabled.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-label-content[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-radio-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-select-value[_ngcontent-%COMP%]{color:#212121}.mat-select-placeholder[_ngcontent-%COMP%]{color:rgba(97,97,97,.42)}.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%]{color:#757575}.mat-select-arrow[_ngcontent-%COMP%]{color:#616161}.mat-select-panel[_ngcontent-%COMP%]{background:#fff}.mat-select-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-select-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple){background:rgba(0,0,0,.12)}.mat-form-field.mat-focused.mat-primary[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-invalid[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#757575}.mat-drawer-container[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-drawer[_ngcontent-%COMP%]{background-color:#fff;color:#212121}.mat-drawer.mat-drawer-push[_ngcontent-%COMP%]{background-color:#fff}.mat-drawer[_ngcontent-%COMP%]:not(.mat-drawer-side){box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-drawer-side[_ngcontent-%COMP%]{border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-backdrop.mat-drawer-shown[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.6)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.54)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.54)}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.54)}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle[_ngcontent-%COMP%]:not(.mat-checked)   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-slide-toggle-thumb[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12);background-color:#fafafa}.mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-primary[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f57c00}.mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-accent[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#ff9800}.mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-warn[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f44336}.mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.2)}.mat-slider[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%], .cdk-focused[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-disabled[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]   .mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing)   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26);background-color:transparent}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26)}.mat-slider-has-ticks[_ngcontent-%COMP%]   .mat-slider-wrapper[_ngcontent-%COMP%]::after{border-color:rgba(0,0,0,.7)}.mat-slider-horizontal[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to right, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent);background-image:-moz-repeating-linear-gradient(0.0001deg, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-slider-vertical[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to bottom, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-step-header.cdk-keyboard-focused[_ngcontent-%COMP%], .mat-step-header.cdk-program-focused[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]:hover{background-color:rgba(0,0,0,.04)}@media(hover: none){.mat-step-header[_ngcontent-%COMP%]:hover{background:none}}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-optional[_ngcontent-%COMP%]{color:#616161}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon[_ngcontent-%COMP%]{background-color:#616161;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-selected[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-done[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-edit[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-error[_ngcontent-%COMP%]{background-color:transparent;color:#f44336}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-active[_ngcontent-%COMP%]{color:#212121}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-error[_ngcontent-%COMP%]{color:#f44336}.mat-stepper-horizontal[_ngcontent-%COMP%], .mat-stepper-vertical[_ngcontent-%COMP%]{background-color:#fff}.mat-stepper-vertical-line[_ngcontent-%COMP%]::before{border-left-color:rgba(0,0,0,.12)}.mat-horizontal-stepper-header[_ngcontent-%COMP%]::before, .mat-horizontal-stepper-header[_ngcontent-%COMP%]::after, .mat-stepper-horizontal-line[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-sort-header-arrow[_ngcontent-%COMP%]{color:#616161}.mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-header[_ngcontent-%COMP%]{border-bottom:1px solid rgba(0,0,0,.12)}.mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%]{border-top:1px solid rgba(0,0,0,.12);border-bottom:none}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{color:#212121}.mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:#757575}.mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#212121}.mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#757575}.mat-tab-group[class*=mat-background-][_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar[class*=mat-background-][_ngcontent-%COMP%]{border-bottom:none;border-top:none}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-toolbar[_ngcontent-%COMP%]{background:#f5f5f5;color:#212121}.mat-toolbar.mat-primary[_ngcontent-%COMP%]{background:#f57c00;color:#fff}.mat-toolbar.mat-accent[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-toolbar.mat-warn[_ngcontent-%COMP%]{background:#f44336;color:#fff}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:currentColor}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:inherit}.mat-toolbar[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:currentColor}.mat-tooltip[_ngcontent-%COMP%]{background:rgba(97,97,97,.9)}.mat-tree[_ngcontent-%COMP%]{background:#fff}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{color:#212121}.mat-snack-bar-container[_ngcontent-%COMP%]{color:rgba(255,255,255,.7);background:#323232;box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-simple-snackbar-action[_ngcontent-%COMP%]{color:#ff9800}.pc-container[_ngcontent-%COMP%]{background-color:#fff;border:1px solid #ebebeb}.pc-toolbar[_ngcontent-%COMP%]{align-items:center;border-bottom:1px solid #ebebeb;display:flex;height:42px;padding:0 16px}.pc-title[_ngcontent-%COMP%]{font-size:14px;font-weight:500;display:inline;flex:1 1}"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/selected_annotations/selected_annotations_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class SelectedAnnotationsContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.pcExpanded$ = this.store.pipe(select(getPCExpanded));
            this.selectedAnnotations$ = this.store.select(getSelectedAnnotations);
        }
        /**
         * @return {?}
         */
        clearSelectedAnnotations() {
            this.store.dispatch(npmiClearSelectedAnnotations());
        }
        /**
         * @return {?}
         */
        toggleExpanded() {
            this.store.dispatch(npmiToggleParallelCoordinatesExpanded());
        }
    }
    SelectedAnnotationsContainer.decorators = [
        { type: Component, args: [{
                    selector: 'npmi-selected-annotations',
                    template: `
    <selected-annotations-component
      [pcExpanded]="pcExpanded$ | async"
      [selectedAnnotations]="selectedAnnotations$ | async"
      (onClearSelectedAnnotations)="clearSelectedAnnotations()"
      (onToggleExpanded)="toggleExpanded()"
    ></selected-annotations-component>
  `,
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    SelectedAnnotationsContainer.ctorParameters = () => [
        { type: Store }
    ];
    /** @nocollapse */ SelectedAnnotationsContainer.ɵfac = function SelectedAnnotationsContainer_Factory(t) { return new (t || SelectedAnnotationsContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ SelectedAnnotationsContainer.ɵcmp = ɵɵdefineComponent({ type: SelectedAnnotationsContainer, selectors: [["npmi-selected-annotations"]], decls: 3, vars: 6, consts: [[3, "pcExpanded", "selectedAnnotations", "onClearSelectedAnnotations", "onToggleExpanded"]], template: function SelectedAnnotationsContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "selected-annotations-component", 0);
            ɵɵlistener("onClearSelectedAnnotations", function SelectedAnnotationsContainer_Template_selected_annotations_component_onClearSelectedAnnotations_0_listener() { return ctx.clearSelectedAnnotations(); })("onToggleExpanded", function SelectedAnnotationsContainer_Template_selected_annotations_component_onToggleExpanded_0_listener() { return ctx.toggleExpanded(); });
            ɵɵpipe(1, "async");
            ɵɵpipe(2, "async");
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵproperty("pcExpanded", ɵɵpipeBind1(1, 2, ctx.pcExpanded$))("selectedAnnotations", ɵɵpipeBind1(2, 4, ctx.selectedAnnotations$));
        } }, directives: [SelectedAnnotationsComponent], pipes: [AsyncPipe], encapsulation: 2, changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/main/main_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    function MainComponent_div_2_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 6);
        ɵɵelementStart(1, "div", 7);
        ɵɵelement(2, "npmi-violin-filters", 8);
        ɵɵelementStart(3, "div", 9);
        ɵɵelement(4, "tb-legacy-runs-selector");
        ɵɵelementEnd();
        ɵɵelementEnd();
        ɵɵelementEnd();
    } if (rf & 2) {
        const ctx_r0 = ɵɵnextContext();
        ɵɵstyleProp("width", ctx_r0.sidebarWidth, "px");
    } }
    function MainComponent_div_3_Template(rf, ctx) { if (rf & 1) {
        const _r7 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "div", 10);
        ɵɵlistener("mousedown", function MainComponent_div_3_Template_div_mousedown_0_listener() { ɵɵrestoreView(_r7); const ctx_r6 = ɵɵnextContext(); return ctx_r6.resizeGrabbed.emit(); });
        ɵɵelementEnd();
    } }
    function MainComponent_div_4_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 11);
        ɵɵelement(1, "npmi-annotations-list", 12);
        ɵɵelement(2, "npmi-selected-annotations");
        ɵɵelementEnd();
    } }
    function MainComponent_ng_template_5_Template(rf, ctx) { if (rf & 1) {
        ɵɵelementStart(0, "div", 13);
        ɵɵtext(1, "You need to select at least one run.");
        ɵɵelementEnd();
    } }
    var I18N_0$b;
    if (typeof ngI18nClosureMode !== "undefined" && ngI18nClosureMode) {
        /**
         * @desc Label for a button that toggles the sidebar.
         */ 
        const MSG_EXTERNAL_4485009347444704878$$TENSORBOARD_WEBAPP_PLUGINS_NPMI_VIEWS_MAIN_MAIN_COMPONENT_TS__1 = goog.getMsg("Toggle Sidebar");
        I18N_0$b = MSG_EXTERNAL_4485009347444704878$$TENSORBOARD_WEBAPP_PLUGINS_NPMI_VIEWS_MAIN_MAIN_COMPONENT_TS__1;
    }
    else {
        I18N_0$b = $localize `:Label for a button that toggles the sidebar.␟f63b57932d179cca62ac9fcae63dd7f5e6fa389f␟4485009347444704878:Toggle Sidebar`;
    }
    const _c2$i = ["aria-label", I18N_0$b];
    function MainComponent_div_7_Template(rf, ctx) { if (rf & 1) {
        const _r9 = ɵɵgetCurrentView();
        ɵɵelementStart(0, "div", 14);
        ɵɵelementStart(1, "button", 15);
        ɵɵi18nAttributes(2, _c2$i);
        ɵɵlistener("click", function MainComponent_div_7_Template_button_click_1_listener() { ɵɵrestoreView(_r9); const ctx_r8 = ɵɵnextContext(); return ctx_r8.toggleSidebarExpanded.emit(); });
        ɵɵelement(3, "mat-icon", 16);
        ɵɵelementEnd();
        ɵɵelementEnd();
    } }
    class MainComponent {
        constructor() {
            this.toggleSidebarExpanded = new EventEmitter();
            this.resizeTriggered = new EventEmitter();
            this.resizeGrabbed = new EventEmitter();
            this.resizeReleased = new EventEmitter();
        }
    }
    MainComponent.decorators = [
        { type: Component, args: [{
                    selector: 'main-component',
                    templateUrl: './main_component.ng.html',
                    styleUrls: ['./main_component.css'],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    MainComponent.propDecorators = {
        runActive: [{ type: Input }],
        sidebarExpanded: [{ type: Input }],
        sidebarWidth: [{ type: Input }],
        toggleSidebarExpanded: [{ type: Output }],
        resizeTriggered: [{ type: Output }],
        resizeGrabbed: [{ type: Output }],
        resizeReleased: [{ type: Output }]
    };
    /** @nocollapse */ MainComponent.ɵfac = function MainComponent_Factory(t) { return new (t || MainComponent)(); };
    /** @nocollapse */ MainComponent.ɵcmp = ɵɵdefineComponent({ type: MainComponent, selectors: [["main-component"]], inputs: { runActive: "runActive", sidebarExpanded: "sidebarExpanded", sidebarWidth: "sidebarWidth" }, outputs: { toggleSidebarExpanded: "toggleSidebarExpanded", resizeTriggered: "resizeTriggered", resizeGrabbed: "resizeGrabbed", resizeReleased: "resizeReleased" }, decls: 8, vars: 5, consts: [[1, "content", 3, "mouseup", "mousemove"], ["class", "sidebar-container", 3, "width", 4, "ngIf"], ["class", "grabber", 3, "mousedown", 4, "ngIf"], ["class", "analysis-container", 4, "ngIf", "ngIfElse"], ["noRun", ""], ["class", "side-toggle", 4, "ngIf"], [1, "sidebar-container"], [1, "sidebar-contents"], [1, "violin-filters"], [1, "run-selector"], [1, "grabber", 3, "mousedown"], [1, "analysis-container"], [1, "annotations-list"], [1, "noRun"], [1, "side-toggle"], ["mat-icon-button", "", 3, "click", 6, "aria-label"], ["svgIcon", "chevron_right_24px"]], template: function MainComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelement(0, "npmi-data-selection");
            ɵɵelementStart(1, "div", 0);
            ɵɵlistener("mouseup", function MainComponent_Template_div_mouseup_1_listener() { return ctx.resizeReleased.emit(); })("mousemove", function MainComponent_Template_div_mousemove_1_listener($event) { return ctx.resizeTriggered.emit($event); });
            ɵɵtemplate(2, MainComponent_div_2_Template, 5, 2, "div", 1);
            ɵɵtemplate(3, MainComponent_div_3_Template, 1, 0, "div", 2);
            ɵɵtemplate(4, MainComponent_div_4_Template, 3, 0, "div", 3);
            ɵɵtemplate(5, MainComponent_ng_template_5_Template, 2, 0, "ng-template", null, 4, ɵɵtemplateRefExtractor);
            ɵɵelementEnd();
            ɵɵtemplate(7, MainComponent_div_7_Template, 4, 0, "div", 5);
        } if (rf & 2) {
            const _r3 = ɵɵreference(6);
            ɵɵadvance(2);
            ɵɵproperty("ngIf", ctx.sidebarExpanded);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx.sidebarExpanded);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx.runActive === true)("ngIfElse", _r3);
            ɵɵadvance(3);
            ɵɵproperty("ngIf", !ctx.sidebarExpanded);
        } }, directives: [DataSelectionComponent, NgIf, ViolinFiltersContainer, LegacyRunsSelectorContainer, AnnotationsListContainer, SelectedAnnotationsContainer, MatButton, MatIcon], styles: [".mat-badge-content[_ngcontent-%COMP%]{font-weight:600;font-size:12px;font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:9px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{font-size:24px}.mat-h1[_ngcontent-%COMP%], .mat-headline[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%]{font:400 24px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h2[_ngcontent-%COMP%], .mat-title[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h3[_ngcontent-%COMP%], .mat-subheading-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%]{font:400 16px/28px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h4[_ngcontent-%COMP%], .mat-subheading-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%]{font:400 15px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 16px}.mat-h5[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%]{font:400 calc(14px * 0.83)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-h6[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:400 calc(14px * 0.67)/20px Roboto, \"Helvetica Neue\", sans-serif;margin:0 0 12px}.mat-body-strong[_ngcontent-%COMP%], .mat-body-2[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-body[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-body-1[_ngcontent-%COMP%]   p[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   p[_ngcontent-%COMP%]{margin:0 0 12px}.mat-small[_ngcontent-%COMP%], .mat-caption[_ngcontent-%COMP%]{font:400 12px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-display-4[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-4[_ngcontent-%COMP%]{font:300 112px/112px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.05em;margin:0 0 56px}.mat-display-3[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-3[_ngcontent-%COMP%]{font:400 56px/56px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.02em;margin:0 0 64px}.mat-display-2[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-2[_ngcontent-%COMP%]{font:400 45px/48px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:-0.005em;margin:0 0 64px}.mat-display-1[_ngcontent-%COMP%], .mat-typography[_ngcontent-%COMP%]   .mat-display-1[_ngcontent-%COMP%]{font:400 34px/40px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0 0 64px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%], .mat-flat-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-button-toggle[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-card-title[_ngcontent-%COMP%]{font-size:24px;font-weight:500}.mat-card-header[_ngcontent-%COMP%]   .mat-card-title[_ngcontent-%COMP%]{font-size:20px}.mat-card-subtitle[_ngcontent-%COMP%], .mat-card-content[_ngcontent-%COMP%]{font-size:14px}.mat-checkbox[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-checkbox-layout[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{line-height:24px}.mat-chip[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-chip[_ngcontent-%COMP%]   .mat-chip-trailing-icon.mat-icon[_ngcontent-%COMP%], .mat-chip[_ngcontent-%COMP%]   .mat-chip-remove.mat-icon[_ngcontent-%COMP%]{font-size:18px}.mat-table[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-header-cell[_ngcontent-%COMP%]{font-size:12px;font-weight:500}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{font-size:14px}.mat-calendar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-calendar-body[_ngcontent-%COMP%]{font-size:13px}.mat-calendar-body-label[_ngcontent-%COMP%], .mat-calendar-period-button[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-calendar-table-header[_ngcontent-%COMP%]   th[_ngcontent-%COMP%]{font-size:11px;font-weight:400}.mat-dialog-title[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-expansion-panel-header[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:15px;font-weight:400}.mat-expansion-panel-content[_ngcontent-%COMP%]{font:400 14px/20px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field[_ngcontent-%COMP%]{font-size:inherit;font-weight:400;line-height:1.125;font-family:Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.34375em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{font-size:150%;line-height:1.125}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]{height:1.5em;width:1.5em}.mat-form-field-prefix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%], .mat-form-field-suffix[_ngcontent-%COMP%]   .mat-icon-button[_ngcontent-%COMP%]   .mat-icon[_ngcontent-%COMP%]{height:1.125em;line-height:1.125}.mat-form-field-infix[_ngcontent-%COMP%]{padding:.5em 0;border-top:.84375em solid transparent}.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34375em) scale(0.75);width:133.3333333333%}.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.34374em) scale(0.75);width:133.3333433333%}.mat-form-field-label-wrapper[_ngcontent-%COMP%]{top:-0.84375em;padding-top:.84375em}.mat-form-field-label[_ngcontent-%COMP%]{top:1.34375em}.mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.34375em}.mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{font-size:75%;margin-top:.6666666667em;top:calc(100% - 1.7916666667em)}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-wrapper[_ngcontent-%COMP%]{padding-bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.4375em 0}.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.001px);-ms-transform:translateY(-1.28125em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00101px);-ms-transform:translateY(-1.28124em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28125em) scale(0.75) perspective(100px) translateZ(0.00102px);-ms-transform:translateY(-1.28123em) scale(0.75);width:133.3333533333%}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.28125em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{bottom:1.25em}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-subscript-wrapper[_ngcontent-%COMP%]{margin-top:.5416666667em;top:calc(100% - 1.6666666667em)}@media print{.mat-form-field-appearance-legacy.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28122em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-form-field-autofill-control[_ngcontent-%COMP%]:-webkit-autofill + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.28121em) scale(0.75)}.mat-form-field-appearance-legacy.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.2812em) scale(0.75)}}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:.25em 0 .75em 0}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.09375em;margin-top:-0.5em}.mat-form-field-appearance-fill.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-fill.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-0.59374em) scale(0.75);width:133.3333433333%}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]{padding:1em 0 1em 0}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{top:1.84375em;margin-top:-0.25em}.mat-form-field-appearance-outline.mat-form-field-can-float.mat-form-field-should-float[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[_ngcontent-%COMP%]:focus + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59375em) scale(0.75);width:133.3333333333%}.mat-form-field-appearance-outline.mat-form-field-can-float[_ngcontent-%COMP%]   .mat-input-server[label][_ngcontent-%COMP%]:not(:label-shown) + .mat-form-field-label-wrapper[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{transform:translateY(-1.59374em) scale(0.75);width:133.3333433333%}.mat-grid-tile-header[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]{font-size:14px}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%], .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-grid-tile-header[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2), .mat-grid-tile-footer[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}input.mat-input-element[_ngcontent-%COMP%]{margin-top:-0.0625em}.mat-menu-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:400}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px}.mat-radio-button[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-select-trigger[_ngcontent-%COMP%]{height:1.125em}.mat-slide-toggle-content[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-slider-thumb-label-text[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-stepper-vertical[_ngcontent-%COMP%], .mat-stepper-horizontal[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-step-label[_ngcontent-%COMP%]{font-size:14px;font-weight:400}.mat-step-sub-label-error[_ngcontent-%COMP%]{font-weight:normal}.mat-step-label-error[_ngcontent-%COMP%]{font-size:14px}.mat-step-label-selected[_ngcontent-%COMP%]{font-size:14px;font-weight:500}.mat-tab-group[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-toolbar[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h1[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h2[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h3[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h4[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h5[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   h6[_ngcontent-%COMP%]{font:500 20px/32px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal;margin:0}.mat-tooltip[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:10px;padding-top:6px;padding-bottom:6px}.mat-tooltip-handset[_ngcontent-%COMP%]{font-size:14px;padding-top:8px;padding-bottom:8px}.mat-list-item[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:16px}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:14px}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px;font-weight:500}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:block;box-sizing:border-box}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]   .mat-line[_ngcontent-%COMP%]:nth-child(n+2){font-size:12px}.mat-list-base[dense][_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:12px;font-weight:500}.mat-option[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:16px}.mat-optgroup-label[_ngcontent-%COMP%]{font:500 14px/24px Roboto, \"Helvetica Neue\", sans-serif;letter-spacing:normal}.mat-simple-snackbar[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif;font-size:14px}.mat-simple-snackbar-action[_ngcontent-%COMP%]{line-height:1;font-family:inherit;font-size:inherit;font-weight:500}.mat-tree[_ngcontent-%COMP%]{font-family:Roboto, \"Helvetica Neue\", sans-serif}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{font-weight:400;font-size:14px}.mat-ripple[_ngcontent-%COMP%]{overflow:hidden;position:relative}.mat-ripple[_ngcontent-%COMP%]:not(:empty){transform:translateZ(0)}.mat-ripple.mat-ripple-unbounded[_ngcontent-%COMP%]{overflow:visible}.mat-ripple-element[_ngcontent-%COMP%]{position:absolute;border-radius:50%;pointer-events:none;transition:opacity,transform 0ms cubic-bezier(0, 0, 0.2, 1);transform:scale(0)}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{display:none}.cdk-visually-hidden[_ngcontent-%COMP%]{border:0;clip:rect(0 0 0 0);height:1px;margin:-1px;overflow:hidden;padding:0;position:absolute;width:1px;outline:0;-webkit-appearance:none;-moz-appearance:none}.cdk-overlay-container[_ngcontent-%COMP%], .cdk-global-overlay-wrapper[_ngcontent-%COMP%]{pointer-events:none;top:0;left:0;height:100%;width:100%}.cdk-overlay-container[_ngcontent-%COMP%]{position:fixed;z-index:1000}.cdk-overlay-container[_ngcontent-%COMP%]:empty{display:none}.cdk-global-overlay-wrapper[_ngcontent-%COMP%]{display:flex;position:absolute;z-index:1000}.cdk-overlay-pane[_ngcontent-%COMP%]{position:absolute;pointer-events:auto;box-sizing:border-box;z-index:1000;display:flex;max-width:100%;max-height:100%}.cdk-overlay-backdrop[_ngcontent-%COMP%]{position:absolute;top:0;bottom:0;left:0;right:0;z-index:1000;pointer-events:auto;-webkit-tap-highlight-color:transparent;transition:opacity 400ms cubic-bezier(0.25, 0.8, 0.25, 1);opacity:0}.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:1}@media screen and (-ms-high-contrast: active){.cdk-overlay-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:.6}}.cdk-overlay-dark-backdrop[_ngcontent-%COMP%]{background:rgba(0,0,0,.32)}.cdk-overlay-transparent-backdrop[_ngcontent-%COMP%], .cdk-overlay-transparent-backdrop.cdk-overlay-backdrop-showing[_ngcontent-%COMP%]{opacity:0}.cdk-overlay-connected-position-bounding-box[_ngcontent-%COMP%]{position:absolute;z-index:1000;display:flex;flex-direction:column;min-width:1px;min-height:1px}.cdk-global-scrollblock[_ngcontent-%COMP%]{position:fixed;width:100%;overflow-y:scroll}@keyframes cdk-text-field-autofill-start{}@keyframes cdk-text-field-autofill-end{}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:-webkit-autofill{animation:cdk-text-field-autofill-start 0s 1ms}.cdk-text-field-autofill-monitored[_ngcontent-%COMP%]:not(:-webkit-autofill){animation:cdk-text-field-autofill-end 0s 1ms}textarea.cdk-textarea-autosize[_ngcontent-%COMP%]{resize:none}textarea.cdk-textarea-autosize-measuring[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:auto !important;overflow:hidden !important}textarea.cdk-textarea-autosize-measuring-firefox[_ngcontent-%COMP%]{padding:2px 0 !important;box-sizing:content-box !important;height:0 !important}.mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.1)}.mat-option[_ngcontent-%COMP%]{color:#212121}.mat-option[_ngcontent-%COMP%]:hover:not(.mat-option-disabled), .mat-option[_ngcontent-%COMP%]:focus:not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple):not(.mat-option-disabled){background:rgba(0,0,0,.04)}.mat-option.mat-active[_ngcontent-%COMP%]{background:rgba(0,0,0,.04);color:#212121}.mat-option.mat-option-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-primary[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f57c00}.mat-accent[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-disabled){color:#f44336}.mat-optgroup-label[_ngcontent-%COMP%]{color:#616161}.mat-optgroup-disabled[_ngcontent-%COMP%]   .mat-optgroup-label[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-pseudo-checkbox[_ngcontent-%COMP%]{color:#616161}.mat-pseudo-checkbox[_ngcontent-%COMP%]::after{color:#fafafa}.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{color:#b0b0b0}.mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f57c00}.mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-checked[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-pseudo-checkbox-indeterminate[_ngcontent-%COMP%]{background:#f44336}.mat-pseudo-checkbox-checked.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%], .mat-pseudo-checkbox-indeterminate.mat-pseudo-checkbox-disabled[_ngcontent-%COMP%]{background:#b0b0b0}.mat-elevation-z0[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z1[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z2[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z3[_ngcontent-%COMP%]{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z4[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z5[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 5px 8px 0px rgba(0, 0, 0, 0.14),0px 1px 14px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z6[_ngcontent-%COMP%]{box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-elevation-z7[_ngcontent-%COMP%]{box-shadow:0px 4px 5px -2px rgba(0, 0, 0, 0.2),0px 7px 10px 1px rgba(0, 0, 0, 0.14),0px 2px 16px 1px rgba(0, 0, 0, 0.12)}.mat-elevation-z8[_ngcontent-%COMP%]{box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z9[_ngcontent-%COMP%]{box-shadow:0px 5px 6px -3px rgba(0, 0, 0, 0.2),0px 9px 12px 1px rgba(0, 0, 0, 0.14),0px 3px 16px 2px rgba(0, 0, 0, 0.12)}.mat-elevation-z10[_ngcontent-%COMP%]{box-shadow:0px 6px 6px -3px rgba(0, 0, 0, 0.2),0px 10px 14px 1px rgba(0, 0, 0, 0.14),0px 4px 18px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z11[_ngcontent-%COMP%]{box-shadow:0px 6px 7px -4px rgba(0, 0, 0, 0.2),0px 11px 15px 1px rgba(0, 0, 0, 0.14),0px 4px 20px 3px rgba(0, 0, 0, 0.12)}.mat-elevation-z12[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z13[_ngcontent-%COMP%]{box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 13px 19px 2px rgba(0, 0, 0, 0.14),0px 5px 24px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z14[_ngcontent-%COMP%]{box-shadow:0px 7px 9px -4px rgba(0, 0, 0, 0.2),0px 14px 21px 2px rgba(0, 0, 0, 0.14),0px 5px 26px 4px rgba(0, 0, 0, 0.12)}.mat-elevation-z15[_ngcontent-%COMP%]{box-shadow:0px 8px 9px -5px rgba(0, 0, 0, 0.2),0px 15px 22px 2px rgba(0, 0, 0, 0.14),0px 6px 28px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z16[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z17[_ngcontent-%COMP%]{box-shadow:0px 8px 11px -5px rgba(0, 0, 0, 0.2),0px 17px 26px 2px rgba(0, 0, 0, 0.14),0px 6px 32px 5px rgba(0, 0, 0, 0.12)}.mat-elevation-z18[_ngcontent-%COMP%]{box-shadow:0px 9px 11px -5px rgba(0, 0, 0, 0.2),0px 18px 28px 2px rgba(0, 0, 0, 0.14),0px 7px 34px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z19[_ngcontent-%COMP%]{box-shadow:0px 9px 12px -6px rgba(0, 0, 0, 0.2),0px 19px 29px 2px rgba(0, 0, 0, 0.14),0px 7px 36px 6px rgba(0, 0, 0, 0.12)}.mat-elevation-z20[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 20px 31px 3px rgba(0, 0, 0, 0.14),0px 8px 38px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z21[_ngcontent-%COMP%]{box-shadow:0px 10px 13px -6px rgba(0, 0, 0, 0.2),0px 21px 33px 3px rgba(0, 0, 0, 0.14),0px 8px 40px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z22[_ngcontent-%COMP%]{box-shadow:0px 10px 14px -6px rgba(0, 0, 0, 0.2),0px 22px 35px 3px rgba(0, 0, 0, 0.14),0px 8px 42px 7px rgba(0, 0, 0, 0.12)}.mat-elevation-z23[_ngcontent-%COMP%]{box-shadow:0px 11px 14px -7px rgba(0, 0, 0, 0.2),0px 23px 36px 3px rgba(0, 0, 0, 0.14),0px 9px 44px 8px rgba(0, 0, 0, 0.12)}.mat-elevation-z24[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12)}.mat-app-background[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-theme-loaded-marker[_ngcontent-%COMP%]{display:none}.mat-autocomplete-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-autocomplete-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover){background:#fff}.mat-autocomplete-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-active):not(:hover):not(.mat-option-disabled){color:#212121}.mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f57c00}.cdk-high-contrast-active[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{outline:solid 1px;border-radius:0}.mat-badge-accent[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-badge-warn[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{color:#fff;background:#f44336}.mat-badge[_ngcontent-%COMP%]{position:relative}.mat-badge-hidden[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{display:none}.mat-badge-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{background:#b9b9b9;color:#757575}.mat-badge-content[_ngcontent-%COMP%]{position:absolute;text-align:center;display:inline-block;border-radius:50%;transition:transform 200ms ease-in-out;transform:scale(0.6);overflow:hidden;white-space:nowrap;text-overflow:ellipsis;pointer-events:none}.ng-animate-disabled[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%], .mat-badge-content._mat-animation-noopable[_ngcontent-%COMP%]{transition:none}.mat-badge-content.mat-badge-active[_ngcontent-%COMP%]{transform:none}.mat-badge-small[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:16px;height:16px;line-height:16px}.mat-badge-small.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-8px}.mat-badge-small.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-8px}.mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-16px}.mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-16px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-16px}.mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-8px}.mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-8px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-small.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-8px}.mat-badge-medium[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:22px;height:22px;line-height:22px}.mat-badge-medium.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-11px}.mat-badge-medium.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-11px}.mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-22px}.mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-22px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-22px}.mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-11px}.mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-11px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-medium.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-11px}.mat-badge-large[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{width:28px;height:28px;line-height:28px}.mat-badge-large.mat-badge-above[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{top:-14px}.mat-badge-large.mat-badge-below[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{bottom:-14px}.mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-28px}.mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-28px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-28px}.mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-before[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{left:auto;right:-14px}.mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:-14px}[dir=rtl][_ngcontent-%COMP%]   .mat-badge-large.mat-badge-overlap.mat-badge-after[_ngcontent-%COMP%]   .mat-badge-content[_ngcontent-%COMP%]{right:auto;left:-14px}.mat-bottom-sheet-container[_ngcontent-%COMP%]{box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-button[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]{color:inherit;background:transparent}.mat-button.mat-primary[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-button[disabled][disabled][_ngcontent-%COMP%], .mat-icon-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-icon-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-icon-button[disabled][disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-stroked-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-stroked-button[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-primary[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f57c00}.mat-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-accent[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#ff9800}.mat-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button.mat-warn[_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:#f44336}.mat-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%], .mat-stroked-button[disabled][_ngcontent-%COMP%]   .mat-button-focus-overlay[_ngcontent-%COMP%]{background-color:transparent}.mat-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-icon-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-stroked-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{opacity:.1;background-color:currentColor}.mat-button-focus-overlay[_ngcontent-%COMP%]{background:#000}.mat-stroked-button[_ngcontent-%COMP%]:not([disabled]){border-color:rgba(0,0,0,.12)}.mat-flat-button[_ngcontent-%COMP%], .mat-raised-button[_ngcontent-%COMP%], .mat-fab[_ngcontent-%COMP%], .mat-mini-fab[_ngcontent-%COMP%]{color:#212121;background-color:#fff}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{color:#fff}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-flat-button.mat-primary[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00}.mat-flat-button.mat-accent[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-flat-button.mat-warn[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-flat-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-flat-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-flat-button[disabled][disabled][_ngcontent-%COMP%], .mat-raised-button.mat-primary[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-accent[disabled][_ngcontent-%COMP%], .mat-raised-button.mat-warn[disabled][_ngcontent-%COMP%], .mat-raised-button[disabled][disabled][_ngcontent-%COMP%], .mat-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-fab[disabled][disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-primary[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-accent[disabled][_ngcontent-%COMP%], .mat-mini-fab.mat-warn[disabled][_ngcontent-%COMP%], .mat-mini-fab[disabled][disabled][_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-flat-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-flat-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-raised-button.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-mini-fab.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-stroked-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-flat-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-raised-button[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0, 0, 0, 0.12)}.mat-raised-button[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]), .mat-mini-fab[_ngcontent-%COMP%]:not([disabled]):active:not([class*=mat-elevation-z]){box-shadow:0px 7px 8px -4px rgba(0, 0, 0, 0.2),0px 12px 17px 2px rgba(0, 0, 0, 0.14),0px 5px 22px 4px rgba(0, 0, 0, 0.12)}.mat-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]), .mat-mini-fab[disabled][_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone[_ngcontent-%COMP%], .mat-button-toggle-group[_ngcontent-%COMP%]{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{box-shadow:none}.mat-button-toggle[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-button-toggle[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121;background:#fff}.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle-focus-overlay[_ngcontent-%COMP%]{background-color:#000}.mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12)}[dir=rtl][_ngcontent-%COMP%]   .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-group-appearance-standard.mat-button-toggle-vertical[_ngcontent-%COMP%]   .mat-button-toggle[_ngcontent-%COMP%] + .mat-button-toggle[_ngcontent-%COMP%]{border-left:none;border-right:none;border-top:solid 1px rgba(0,0,0,.12)}.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#616161}.mat-button-toggle-checked.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{color:#212121}.mat-button-toggle-disabled[_ngcontent-%COMP%]{color:rgba(0,0,0,.26);background-color:#eee}.mat-button-toggle-disabled.mat-button-toggle-appearance-standard[_ngcontent-%COMP%]{background:#fff}.mat-button-toggle-disabled.mat-button-toggle-checked[_ngcontent-%COMP%]{background-color:#bdbdbd}.mat-button-toggle-standalone.mat-button-toggle-appearance-standard[_ngcontent-%COMP%], .mat-button-toggle-group-appearance-standard[_ngcontent-%COMP%]{border:solid 1px rgba(0,0,0,.12)}.mat-card[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-card[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12)}.mat-card.mat-card-flat[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-card-subtitle[_ngcontent-%COMP%]{color:#616161}.mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#616161}.mat-checkbox-checkmark[_ngcontent-%COMP%]{fill:#fafafa}.mat-checkbox-checkmark-path[_ngcontent-%COMP%]{stroke:#fafafa !important}.mat-checkbox-mixedmark[_ngcontent-%COMP%]{background-color:#fafafa}.mat-checkbox-indeterminate.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-primary[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f57c00}.mat-checkbox-indeterminate.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-accent[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#ff9800}.mat-checkbox-indeterminate.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-checked.mat-warn[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#f44336}.mat-checkbox-disabled.mat-checkbox-checked[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%], .mat-checkbox-disabled.mat-checkbox-indeterminate[_ngcontent-%COMP%]   .mat-checkbox-background[_ngcontent-%COMP%]{background-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]:not(.mat-checkbox-checked)   .mat-checkbox-frame[_ngcontent-%COMP%]{border-color:#b0b0b0}.mat-checkbox-disabled[_ngcontent-%COMP%]   .mat-checkbox-label[_ngcontent-%COMP%]{color:#616161}.mat-checkbox[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-primary   .mat-ripple-element[_ngcontent-%COMP%]{background:#f57c00}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-accent   .mat-ripple-element[_ngcontent-%COMP%]{background:#ff9800}.mat-checkbox-checked[_ngcontent-%COMP%]:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%], .mat-checkbox[_ngcontent-%COMP%]:active:not(.mat-checkbox-disabled).mat-warn   .mat-ripple-element[_ngcontent-%COMP%]{background:#f44336}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]{background-color:#e0e0e0;color:#212121}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#212121;opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled):active{box-shadow:0px 3px 3px -2px rgba(0, 0, 0, 0.2),0px 3px 4px 0px rgba(0, 0, 0, 0.14),0px 1px 8px 0px rgba(0, 0, 0, 0.12)}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]:not(.mat-chip-disabled)   .mat-chip-remove[_ngcontent-%COMP%]:hover{opacity:.54}.mat-chip.mat-standard-chip.mat-chip-disabled[_ngcontent-%COMP%]{opacity:.4}.mat-chip.mat-standard-chip[_ngcontent-%COMP%]::after{background:#000}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-chip-remove[_ngcontent-%COMP%]{color:#fff;opacity:.4}.mat-chip.mat-standard-chip.mat-chip-selected.mat-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.1)}.mat-table[_ngcontent-%COMP%]{background:#fff}.mat-table[_ngcontent-%COMP%]   thead[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tbody[_ngcontent-%COMP%], .mat-table[_ngcontent-%COMP%]   tfoot[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], [mat-header-row][_ngcontent-%COMP%], [mat-row][_ngcontent-%COMP%], [mat-footer-row][_ngcontent-%COMP%], .mat-table-sticky[_ngcontent-%COMP%]{background:inherit}mat-row[_ngcontent-%COMP%], mat-header-row[_ngcontent-%COMP%], mat-footer-row[_ngcontent-%COMP%], th.mat-header-cell[_ngcontent-%COMP%], td.mat-cell[_ngcontent-%COMP%], td.mat-footer-cell[_ngcontent-%COMP%]{border-bottom-color:rgba(0,0,0,.12)}.mat-header-cell[_ngcontent-%COMP%]{color:#616161}.mat-cell[_ngcontent-%COMP%], .mat-footer-cell[_ngcontent-%COMP%]{color:#212121}.mat-calendar-arrow[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.54)}.mat-datepicker-toggle[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-next-button[_ngcontent-%COMP%], .mat-datepicker-content[_ngcontent-%COMP%]   .mat-calendar-previous-button[_ngcontent-%COMP%]{color:rgba(0,0,0,.54)}.mat-calendar-table-header[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-calendar-table-header-divider[_ngcontent-%COMP%]::after{background:rgba(0,0,0,.12)}.mat-calendar-body-label[_ngcontent-%COMP%]{color:#616161}.mat-calendar-body-cell-content[_ngcontent-%COMP%]{color:#212121;border-color:transparent}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){color:#757575}.mat-calendar-body-cell[_ngcontent-%COMP%]:not(.mat-calendar-body-disabled):hover > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-keyboard-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected), .cdk-program-focused[_ngcontent-%COMP%]   .mat-calendar-body-active[_ngcontent-%COMP%] > .mat-calendar-body-cell-content[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){background-color:rgba(0,0,0,.04)}.mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.38)}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-today[_ngcontent-%COMP%]:not(.mat-calendar-body-selected){border-color:rgba(0,0,0,.18)}.mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.4)}.mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content[_ngcontent-%COMP%]{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12);background-color:#fff;color:#212121}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#ff9800;color:#fff}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.4)}.mat-datepicker-content.mat-accent[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:#f44336;color:#fff}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-disabled[_ngcontent-%COMP%] > .mat-calendar-body-selected[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.4)}.mat-datepicker-content.mat-warn[_ngcontent-%COMP%]   .mat-calendar-body-today.mat-calendar-body-selected[_ngcontent-%COMP%]{box-shadow:inset 0 0 0 1px #fff}.mat-datepicker-content-touch[_ngcontent-%COMP%]{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0, 0, 0, 0.12)}.mat-datepicker-toggle-active[_ngcontent-%COMP%]{color:#f57c00}.mat-datepicker-toggle-active.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-datepicker-toggle-active.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-dialog-container[_ngcontent-%COMP%]{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0, 0, 0, 0.12);background:#fff;color:#212121}.mat-divider[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-divider-vertical[_ngcontent-%COMP%]{border-right-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]{background:#fff;color:#212121}.mat-expansion-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0, 0, 0, 0.12)}.mat-action-row[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-keyboard-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]   .mat-expansion-panel-header.cdk-program-focused[_ngcontent-%COMP%]:not([aria-disabled=true]), .mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded)   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover:not([aria-disabled=true]){background:rgba(0,0,0,.04)}@media(hover: none){.mat-expansion-panel[_ngcontent-%COMP%]:not(.mat-expanded):not([aria-disabled=true])   .mat-expansion-panel-header[_ngcontent-%COMP%]:hover{background:#fff}}.mat-expansion-panel-header-title[_ngcontent-%COMP%]{color:#212121}.mat-expansion-panel-header-description[_ngcontent-%COMP%], .mat-expansion-indicator[_ngcontent-%COMP%]::after{color:#616161}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]{color:rgba(0,0,0,.26)}.mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-title[_ngcontent-%COMP%], .mat-expansion-panel-header[aria-disabled=true][_ngcontent-%COMP%]   .mat-expansion-panel-header-description[_ngcontent-%COMP%]{color:inherit}.mat-form-field-label[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-hint[_ngcontent-%COMP%]{color:rgba(97,97,97,.6)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-label.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-focused[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-ripple[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#ff9800}.mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-warn[_ngcontent-%COMP%]{background-color:#f44336}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid)   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f57c00}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-accent   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#ff9800}.mat-form-field-type-mat-native-select.mat-focused[_ngcontent-%COMP%]:not(.mat-form-field-invalid).mat-warn   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label.mat-accent[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]   .mat-form-field-required-marker[_ngcontent-%COMP%]{color:#f44336}.mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-form-field.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-ripple.mat-accent[_ngcontent-%COMP%]{background-color:#f44336}.mat-error[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-hint[_ngcontent-%COMP%]{color:#616161}.mat-form-field-appearance-legacy[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-legacy.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-standard[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-standard.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]{background-image:linear-gradient(to right, rgba(0, 0, 0, 0.42) 0%, rgba(0, 0, 0, 0.42) 33%, transparent 0%);background-size:4px 100%;background-repeat:repeat-x}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.04)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-flex[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.02)}.mat-form-field-appearance-fill[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:rgba(0,0,0,.42)}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-fill.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%]::before{background-color:transparent}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.12)}.mat-form-field-appearance-outline[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:rgba(0,0,0,.87)}.mat-form-field-appearance-outline.mat-focused[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field-appearance-outline.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field-appearance-outline.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-invalid.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-outline-thick[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%]{color:#757575}.mat-form-field-appearance-outline.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-outline[_ngcontent-%COMP%]{color:rgba(0,0,0,.06)}.mat-icon.mat-primary[_ngcontent-%COMP%]{color:#f57c00}.mat-icon.mat-accent[_ngcontent-%COMP%]{color:#ff9800}.mat-icon.mat-warn[_ngcontent-%COMP%]{color:#f44336}.mat-form-field-type-mat-native-select[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#616161}.mat-input-element[_ngcontent-%COMP%]:disabled, .mat-form-field-type-mat-native-select.mat-form-field-disabled[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#757575}.mat-input-element[_ngcontent-%COMP%]{caret-color:#f57c00}.mat-input-element[_ngcontent-%COMP%]::placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-moz-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]::-webkit-input-placeholder{color:rgba(97,97,97,.42)}.mat-input-element[_ngcontent-%COMP%]:-ms-input-placeholder{color:rgba(97,97,97,.42)}.mat-accent[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#ff9800}.mat-warn[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%], .mat-form-field-invalid[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:#f44336}.mat-form-field-type-mat-native-select.mat-form-field-invalid[_ngcontent-%COMP%]   .mat-form-field-infix[_ngcontent-%COMP%]::after{color:#f44336}.mat-list-base[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-list-option[_ngcontent-%COMP%]{color:#212121}.mat-list-base[_ngcontent-%COMP%]   .mat-subheader[_ngcontent-%COMP%]{color:#616161}.mat-list-item-disabled[_ngcontent-%COMP%]{background-color:#eee}.mat-list-option[_ngcontent-%COMP%]:hover, .mat-list-option[_ngcontent-%COMP%]:focus, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-nav-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:hover, .mat-action-list[_ngcontent-%COMP%]   .mat-list-item[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.04)}.mat-list-single-selected-option[_ngcontent-%COMP%], .mat-list-single-selected-option[_ngcontent-%COMP%]:hover, .mat-list-single-selected-option[_ngcontent-%COMP%]:focus{background:rgba(0,0,0,.12)}.mat-menu-panel[_ngcontent-%COMP%]{background:#fff}.mat-menu-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-menu-item[_ngcontent-%COMP%]{background:transparent;color:#212121}.mat-menu-item[disabled][_ngcontent-%COMP%], .mat-menu-item[disabled][_ngcontent-%COMP%]::after{color:rgba(0,0,0,.38)}.mat-menu-item[_ngcontent-%COMP%]   .mat-icon-no-color[_ngcontent-%COMP%], .mat-menu-item-submenu-trigger[_ngcontent-%COMP%]::after{color:rgba(0,0,0,.54)}.mat-menu-item[_ngcontent-%COMP%]:hover:not([disabled]), .mat-menu-item.cdk-program-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item.cdk-keyboard-focused[_ngcontent-%COMP%]:not([disabled]), .mat-menu-item-highlighted[_ngcontent-%COMP%]:not([disabled]){background:rgba(0,0,0,.04)}.mat-paginator[_ngcontent-%COMP%]{background:#fff}.mat-paginator[_ngcontent-%COMP%], .mat-paginator-page-size[_ngcontent-%COMP%]   .mat-select-trigger[_ngcontent-%COMP%]{color:#616161}.mat-paginator-decrement[_ngcontent-%COMP%], .mat-paginator-increment[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54);border-right:2px solid rgba(0,0,0,.54)}.mat-paginator-first[_ngcontent-%COMP%], .mat-paginator-last[_ngcontent-%COMP%]{border-top:2px solid rgba(0,0,0,.54)}.mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-decrement[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-increment[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-first[_ngcontent-%COMP%], .mat-icon-button[disabled][_ngcontent-%COMP%]   .mat-paginator-last[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f57c00}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ff9800}.mat-progress-bar.mat-accent[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#ff9800}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-background[_ngcontent-%COMP%]{fill:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-buffer[_ngcontent-%COMP%]{background-color:#ffcdd2}.mat-progress-bar.mat-warn[_ngcontent-%COMP%]   .mat-progress-bar-fill[_ngcontent-%COMP%]::after{background-color:#f44336}.mat-progress-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f57c00}.mat-progress-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-accent[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#ff9800}.mat-progress-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%], .mat-spinner.mat-warn[_ngcontent-%COMP%]   circle[_ngcontent-%COMP%]{stroke:#f44336}.mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#616161}.mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f57c00}.mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-primary.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-primary[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f57c00}.mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#ff9800}.mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-accent.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-accent[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#ff9800}.mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:#f44336}.mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]:not(.mat-radio-persistent-ripple), .mat-radio-button.mat-warn.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-persistent-ripple[_ngcontent-%COMP%], .mat-radio-button.mat-warn[_ngcontent-%COMP%]:active   .mat-radio-persistent-ripple[_ngcontent-%COMP%]{background-color:#f44336}.mat-radio-button.mat-radio-disabled.mat-radio-checked[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-outer-circle[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-ripple[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-inner-circle[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-radio-button.mat-radio-disabled[_ngcontent-%COMP%]   .mat-radio-label-content[_ngcontent-%COMP%]{color:rgba(0,0,0,.38)}.mat-radio-button[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-select-value[_ngcontent-%COMP%]{color:#212121}.mat-select-placeholder[_ngcontent-%COMP%]{color:rgba(97,97,97,.42)}.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%]{color:#757575}.mat-select-arrow[_ngcontent-%COMP%]{color:#616161}.mat-select-panel[_ngcontent-%COMP%]{background:#fff}.mat-select-panel[_ngcontent-%COMP%]:not([class*=mat-elevation-z]){box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0, 0, 0, 0.12)}.mat-select-panel[_ngcontent-%COMP%]   .mat-option.mat-selected[_ngcontent-%COMP%]:not(.mat-option-multiple){background:rgba(0,0,0,.12)}.mat-form-field.mat-focused.mat-primary[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f57c00}.mat-form-field.mat-focused.mat-accent[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#ff9800}.mat-form-field.mat-focused.mat-warn[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-invalid[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#f44336}.mat-form-field[_ngcontent-%COMP%]   .mat-select.mat-select-disabled[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:#757575}.mat-drawer-container[_ngcontent-%COMP%]{background-color:#fafafa;color:#212121}.mat-drawer[_ngcontent-%COMP%]{background-color:#fff;color:#212121}.mat-drawer.mat-drawer-push[_ngcontent-%COMP%]{background-color:#fff}.mat-drawer[_ngcontent-%COMP%]:not(.mat-drawer-side){box-shadow:0px 8px 10px -5px rgba(0, 0, 0, 0.2),0px 16px 24px 2px rgba(0, 0, 0, 0.14),0px 6px 30px 5px rgba(0, 0, 0, 0.12)}.mat-drawer-side[_ngcontent-%COMP%]{border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side[_ngcontent-%COMP%]{border-left:solid 1px rgba(0,0,0,.12);border-right:none}[dir=rtl][_ngcontent-%COMP%]   .mat-drawer-side.mat-drawer-end[_ngcontent-%COMP%]{border-left:none;border-right:solid 1px rgba(0,0,0,.12)}.mat-drawer-backdrop.mat-drawer-shown[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.6)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.54)}.mat-slide-toggle.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#ff9800}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(245,124,0,.54)}.mat-slide-toggle.mat-primary.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f57c00}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-thumb[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(244,67,54,.54)}.mat-slide-toggle.mat-warn.mat-checked[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#f44336}.mat-slide-toggle[_ngcontent-%COMP%]:not(.mat-checked)   .mat-ripple-element[_ngcontent-%COMP%]{background-color:#000}.mat-slide-toggle-thumb[_ngcontent-%COMP%]{box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0, 0, 0, 0.12);background-color:#fafafa}.mat-slide-toggle-bar[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-primary[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f57c00}.mat-primary[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-accent[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#ff9800}.mat-accent[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-warn[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:#f44336}.mat-warn[_ngcontent-%COMP%]   .mat-slider-thumb-label-text[_ngcontent-%COMP%]{color:#fff}.mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(255,152,0,.2)}.mat-slider[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%], .cdk-focused[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.38)}.mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-background[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-track-fill[_ngcontent-%COMP%], .mat-slider-disabled[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-disabled[_ngcontent-%COMP%]:hover   .mat-slider-track-background[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]   .mat-slider-focus-ring[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.12)}.mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.87)}.mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value.mat-slider-thumb-label-showing.cdk-focused[_ngcontent-%COMP%]   .mat-slider-thumb-label[_ngcontent-%COMP%]{background-color:rgba(0,0,0,.26)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing)   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26);background-color:transparent}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.38)}.mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing):hover.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%], .mat-slider-min-value[_ngcontent-%COMP%]:not(.mat-slider-thumb-label-showing).cdk-focused.mat-slider-disabled   .mat-slider-thumb[_ngcontent-%COMP%]{border-color:rgba(0,0,0,.26)}.mat-slider-has-ticks[_ngcontent-%COMP%]   .mat-slider-wrapper[_ngcontent-%COMP%]::after{border-color:rgba(0,0,0,.7)}.mat-slider-horizontal[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to right, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent);background-image:-moz-repeating-linear-gradient(0.0001deg, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-slider-vertical[_ngcontent-%COMP%]   .mat-slider-ticks[_ngcontent-%COMP%]{background-image:repeating-linear-gradient(to bottom, rgba(0, 0, 0, 0.7), rgba(0, 0, 0, 0.7) 2px, transparent 0, transparent)}.mat-step-header.cdk-keyboard-focused[_ngcontent-%COMP%], .mat-step-header.cdk-program-focused[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]:hover{background-color:rgba(0,0,0,.04)}@media(hover: none){.mat-step-header[_ngcontent-%COMP%]:hover{background:none}}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-optional[_ngcontent-%COMP%]{color:#616161}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon[_ngcontent-%COMP%]{background-color:#616161;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-selected[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-done[_ngcontent-%COMP%], .mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-edit[_ngcontent-%COMP%]{background-color:#f57c00;color:#fff}.mat-step-header[_ngcontent-%COMP%]   .mat-step-icon-state-error[_ngcontent-%COMP%]{background-color:transparent;color:#f44336}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-active[_ngcontent-%COMP%]{color:#212121}.mat-step-header[_ngcontent-%COMP%]   .mat-step-label.mat-step-label-error[_ngcontent-%COMP%]{color:#f44336}.mat-stepper-horizontal[_ngcontent-%COMP%], .mat-stepper-vertical[_ngcontent-%COMP%]{background-color:#fff}.mat-stepper-vertical-line[_ngcontent-%COMP%]::before{border-left-color:rgba(0,0,0,.12)}.mat-horizontal-stepper-header[_ngcontent-%COMP%]::before, .mat-horizontal-stepper-header[_ngcontent-%COMP%]::after, .mat-stepper-horizontal-line[_ngcontent-%COMP%]{border-top-color:rgba(0,0,0,.12)}.mat-sort-header-arrow[_ngcontent-%COMP%]{color:#616161}.mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-header[_ngcontent-%COMP%]{border-bottom:1px solid rgba(0,0,0,.12)}.mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-nav-bar[_ngcontent-%COMP%], .mat-tab-group-inverted-header[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%]{border-top:1px solid rgba(0,0,0,.12);border-bottom:none}.mat-tab-label[_ngcontent-%COMP%], .mat-tab-link[_ngcontent-%COMP%]{color:#212121}.mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:#757575}.mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#212121}.mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#757575}.mat-tab-group[class*=mat-background-][_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar[class*=mat-background-][_ngcontent-%COMP%]{border-bottom:none;border-top:none}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-primary.mat-background-primary[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-accent.mat-background-accent[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-warn.mat-background-warn[_ngcontent-%COMP%]   .mat-ink-bar[_ngcontent-%COMP%]{background-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f57c00}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-primary[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,152,0,.3)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#ff9800}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-accent[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-keyboard-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled), .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.cdk-program-focused[_ngcontent-%COMP%]:not(.mat-tab-disabled){background-color:rgba(255,205,210,.3)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-links[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination[_ngcontent-%COMP%]{background-color:#f44336}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link[_ngcontent-%COMP%]{color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-label.mat-tab-disabled[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-link.mat-tab-disabled[_ngcontent-%COMP%]{color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:#fff}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-tab-header-pagination-disabled[_ngcontent-%COMP%]   .mat-tab-header-pagination-chevron[_ngcontent-%COMP%]{border-color:rgba(255,255,255,.4)}.mat-tab-group.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%], .mat-tab-nav-bar.mat-background-warn[_ngcontent-%COMP%]   .mat-ripple-element[_ngcontent-%COMP%]{background-color:rgba(255,255,255,.12)}.mat-toolbar[_ngcontent-%COMP%]{background:#f5f5f5;color:#212121}.mat-toolbar.mat-primary[_ngcontent-%COMP%]{background:#f57c00;color:#fff}.mat-toolbar.mat-accent[_ngcontent-%COMP%]{background:#ff9800;color:#fff}.mat-toolbar.mat-warn[_ngcontent-%COMP%]{background:#f44336;color:#fff}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-underline[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-ripple[_ngcontent-%COMP%]{background-color:currentColor}.mat-toolbar[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-focused[_ngcontent-%COMP%]   .mat-form-field-label[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-value[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%], .mat-toolbar[_ngcontent-%COMP%]   .mat-form-field.mat-focused[_ngcontent-%COMP%]   .mat-select-arrow[_ngcontent-%COMP%]{color:inherit}.mat-toolbar[_ngcontent-%COMP%]   .mat-input-element[_ngcontent-%COMP%]{caret-color:currentColor}.mat-tooltip[_ngcontent-%COMP%]{background:rgba(97,97,97,.9)}.mat-tree[_ngcontent-%COMP%]{background:#fff}.mat-tree-node[_ngcontent-%COMP%], .mat-nested-tree-node[_ngcontent-%COMP%]{color:#212121}.mat-snack-bar-container[_ngcontent-%COMP%]{color:rgba(255,255,255,.7);background:#323232;box-shadow:0px 3px 5px -1px rgba(0, 0, 0, 0.2),0px 6px 10px 0px rgba(0, 0, 0, 0.14),0px 1px 18px 0px rgba(0, 0, 0, 0.12)}.mat-simple-snackbar-action[_ngcontent-%COMP%]{color:#ff9800}[_nghost-%COMP%]{display:flex;flex-direction:column;height:100%}.content[_ngcontent-%COMP%]{display:flex;flex:1;min-height:0px;width:100%}.sidebar-container[_ngcontent-%COMP%]{height:100%;overflow:hidden}.sidebar-contents[_ngcontent-%COMP%]{height:100%;display:flex;flex-direction:column;min-width:150px}.analysis-container[_ngcontent-%COMP%]{display:flex;flex-direction:column;height:100%;flex:1 1}.run-selector[_ngcontent-%COMP%]{padding:10px;width:calc(100% - 20px)}.violin-filters[_ngcontent-%COMP%]{min-height:0px;width:100%}.side-toggle[_ngcontent-%COMP%]{width:30px;height:30px;position:absolute;left:10px;bottom:10px;border:1px solid #ebebeb;border-radius:3px;display:flex;align-items:center;justify-content:center}.grabber[_ngcontent-%COMP%]{content:\"\";cursor:ew-resize;height:100%;width:3px;overflow:hidden;background-color:rgba(0,0,0,.12)}.annotations-list[_ngcontent-%COMP%]{width:100%;min-height:0px;flex:1 1}"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/main/main_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class MainContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.runActive$ = this.store
                .pipe(select(getCurrentRouteRunSelection))
                .pipe(map((/**
             * @param {?} runs
             * @return {?}
             */
            (runs) => {
                if (!runs) {
                    return false;
                }
                return [...runs.values()].includes(true);
            })));
            this.sidebarExpanded$ = this.store.pipe(select(getSidebarExpanded));
            this.sidebarWidth$ = this.store.pipe(select(getSidebarWidth));
            this.resizing = false;
        }
        /**
         * @return {?}
         */
        onToggleSidebarExpanded() {
            this.store.dispatch(npmiToggleSidebarExpanded());
        }
        /**
         * @param {?} event
         * @return {?}
         */
        onResizeTriggered(event) {
            if (this.resizing) {
                this.store.dispatch(npmiChangeSidebarWidth({
                    sidebarWidth: event.clientX,
                }));
            }
        }
        /**
         * @return {?}
         */
        onResizeGrabbed() {
            this.resizing = true;
        }
        /**
         * @return {?}
         */
        onResizeReleased() {
            this.resizing = false;
        }
    }
    MainContainer.decorators = [
        { type: Component, args: [{
                    selector: 'npmi-main',
                    template: `
    <main-component
      [runActive]="runActive$ | async"
      [sidebarExpanded]="sidebarExpanded$ | async"
      [sidebarWidth]="sidebarWidth$ | async"
      (toggleSidebarExpanded)="onToggleSidebarExpanded()"
      (resizeTriggered)="onResizeTriggered($event)"
      (resizeGrabbed)="onResizeGrabbed()"
      (resizeReleased)="onResizeReleased()"
    ></main-component>
  `,
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */
    MainContainer.ctorParameters = () => [
        { type: Store }
    ];
    /** @nocollapse */ MainContainer.ɵfac = function MainContainer_Factory(t) { return new (t || MainContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ MainContainer.ɵcmp = ɵɵdefineComponent({ type: MainContainer, selectors: [["npmi-main"]], decls: 4, vars: 9, consts: [[3, "runActive", "sidebarExpanded", "sidebarWidth", "toggleSidebarExpanded", "resizeTriggered", "resizeGrabbed", "resizeReleased"]], template: function MainContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "main-component", 0);
            ɵɵlistener("toggleSidebarExpanded", function MainContainer_Template_main_component_toggleSidebarExpanded_0_listener() { return ctx.onToggleSidebarExpanded(); })("resizeTriggered", function MainContainer_Template_main_component_resizeTriggered_0_listener($event) { return ctx.onResizeTriggered($event); })("resizeGrabbed", function MainContainer_Template_main_component_resizeGrabbed_0_listener() { return ctx.onResizeGrabbed(); })("resizeReleased", function MainContainer_Template_main_component_resizeReleased_0_listener() { return ctx.onResizeReleased(); });
            ɵɵpipe(1, "async");
            ɵɵpipe(2, "async");
            ɵɵpipe(3, "async");
            ɵɵelementEnd();
        } if (rf & 2) {
            ɵɵproperty("runActive", ɵɵpipeBind1(1, 3, ctx.runActive$))("sidebarExpanded", ɵɵpipeBind1(2, 5, ctx.sidebarExpanded$))("sidebarWidth", ɵɵpipeBind1(3, 7, ctx.sidebarWidth$));
        } }, directives: [MainComponent], pipes: [AsyncPipe], encapsulation: 2, changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/npmi_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    function NpmiComponent_npmi_inactive_view_1_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "npmi-inactive-view");
    } }
    function NpmiComponent_ng_template_2_Template(rf, ctx) { if (rf & 1) {
        ɵɵelement(0, "npmi-main");
    } }
    class NpmiComponent {
    }
    NpmiComponent.decorators = [
        { type: Component, args: [{
                    selector: 'npmi-component',
                    templateUrl: './npmi_component.ng.html',
                    styleUrls: ['./npmi_component.css'],
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    NpmiComponent.propDecorators = {
        runs: [{ type: Input }]
    };
    /** @nocollapse */ NpmiComponent.ɵfac = function NpmiComponent_Factory(t) { return new (t || NpmiComponent)(); };
    /** @nocollapse */ NpmiComponent.ɵcmp = ɵɵdefineComponent({ type: NpmiComponent, selectors: [["npmi-component"]], inputs: { runs: "runs" }, decls: 4, vars: 2, consts: [[1, "npmi-container"], [4, "ngIf", "ngIfElse"], ["dataAvailable", ""]], template: function NpmiComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵelementStart(0, "div", 0);
            ɵɵtemplate(1, NpmiComponent_npmi_inactive_view_1_Template, 1, 0, "npmi-inactive-view", 1);
            ɵɵtemplate(2, NpmiComponent_ng_template_2_Template, 1, 0, "ng-template", null, 2, ɵɵtemplateRefExtractor);
            ɵɵelementEnd();
        } if (rf & 2) {
            const _r1 = ɵɵreference(3);
            ɵɵadvance(1);
            ɵɵproperty("ngIf", ctx.runs.size === 0)("ngIfElse", _r1);
        } }, directives: [NgIf, InactiveComponent$1, MainContainer], styles: ["[_nghost-%COMP%]{display:flex;height:100%}.npmi-container[_ngcontent-%COMP%]{flex:1 1}"], changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/npmi_container.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class NpmiContainer {
        /**
         * @param {?} store
         */
        constructor(store) {
            this.store = store;
            this.runs$ = this.store.pipe(select(getCurrentRouteRunSelection));
        }
        /**
         * @return {?}
         */
        ngOnInit() {
            this.store.dispatch(npmiLoaded());
        }
    }
    NpmiContainer.decorators = [
        { type: Component, args: [{
                    selector: 'npmi',
                    template: ` <npmi-component [runs]="runs$ | async"></npmi-component> `,
                },] },
    ];
    /** @nocollapse */
    NpmiContainer.ctorParameters = () => [
        { type: Store }
    ];
    /** @nocollapse */ NpmiContainer.ɵfac = function NpmiContainer_Factory(t) { return new (t || NpmiContainer)(ɵɵdirectiveInject(Store)); };
    /** @nocollapse */ NpmiContainer.ɵcmp = ɵɵdefineComponent({ type: NpmiContainer, selectors: [["npmi"]], decls: 2, vars: 3, consts: [[3, "runs"]], template: function NpmiContainer_Template(rf, ctx) { if (rf & 1) {
            ɵɵelement(0, "npmi-component", 0);
            ɵɵpipe(1, "async");
        } if (rf & 2) {
            ɵɵproperty("runs", ɵɵpipeBind1(1, 1, ctx.runs$));
        } }, directives: [NpmiComponent], pipes: [AsyncPipe], encapsulation: 2 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/data_source/npmi_data_source.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class NpmiHttpServerDataSource {
        /**
         * @param {?} http
         */
        constructor(http) {
            this.http = http;
            this.httpPathPrefix = 'data/plugin/npmi';
        }
        /**
         * @return {?}
         */
        fetchData() {
            return forkJoin(this.fetchAnnotations(), this.fetchMetrics(), this.fetchValues(), this.fetchEmbeddings()).pipe(map((/**
             * @param {?} __0
             * @return {?}
             */
            ([annotations, metrics, values, embeddings]) => {
                /** @type {?} */
                const annotationData = {};
                /** @type {?} */
                const embeddingData = {};
                for (const run of Object.keys(annotations)) {
                    for (const annotationIndex in annotations[run]) {
                        /** @type {?} */
                        const annotation = annotations[run][annotationIndex];
                        if (embeddings[run][annotationIndex]) {
                            embeddingData[annotation] = embeddings[run][annotationIndex];
                        }
                        /** @type {?} */
                        const metricToDataElements = new Map();
                        for (const metricIndex in metrics[run]) {
                            /** @type {?} */
                            const metric = metrics[run][metricIndex];
                            /** @type {?} */
                            const metricString = stripMetricString(metric);
                            /** @type {?} */
                            let dataElement = metricToDataElements.get(metricString);
                            if (!dataElement) {
                                dataElement = {
                                    nPMIValue: null,
                                    countValue: null,
                                    annotation: annotation,
                                    metric: metricString,
                                    run: run,
                                };
                                metricToDataElements.set(metricString, dataElement);
                            }
                            if (metricIsMetricCount(metric)) {
                                dataElement.countValue =
                                    values[run][annotationIndex][metricIndex];
                            }
                            else if (metricIsNpmi(metric)) {
                                dataElement.nPMIValue =
                                    values[run][annotationIndex][metricIndex];
                            }
                        }
                        /** @type {?} */
                        const existing = annotationData[annotation]
                            ? annotationData[annotation]
                            : [];
                        annotationData[annotation] = [
                            ...existing,
                            ...metricToDataElements.values(),
                        ];
                    }
                }
                return { annotationData, metrics, embeddingData };
            })), catchError((/**
             * @param {?} error
             * @return {?}
             */
            (error) => {
                if (error instanceof HttpErrorResponse &&
                    400 <= error.status &&
                    error.status < 500) {
                    return of({
                        annotationData: {},
                        metrics: {},
                        embeddingData: {},
                    });
                }
                return throwError(error);
            })));
        }
        /**
         * @private
         * @return {?}
         */
        fetchAnnotations() {
            return this.http.get(this.httpPathPrefix + '/annotations');
        }
        /**
         * @private
         * @return {?}
         */
        fetchMetrics() {
            return this.http.get(this.httpPathPrefix + '/metrics');
        }
        /**
         * @private
         * @return {?}
         */
        fetchValues() {
            return this.http.get(this.httpPathPrefix + '/values');
        }
        /**
         * @private
         * @return {?}
         */
        fetchEmbeddings() {
            return this.http.get(this.httpPathPrefix + '/embeddings');
        }
    }
    NpmiHttpServerDataSource.decorators = [
        { type: Injectable },
    ];
    /** @nocollapse */
    NpmiHttpServerDataSource.ctorParameters = () => [
        { type: TBHttpClient }
    ];
    /** @nocollapse */ NpmiHttpServerDataSource.ɵfac = function NpmiHttpServerDataSource_Factory(t) { return new (t || NpmiHttpServerDataSource)(ɵɵinject(TBHttpClient)); };
    /** @nocollapse */ NpmiHttpServerDataSource.ɵprov = ɵɵdefineInjectable({ token: NpmiHttpServerDataSource, factory: NpmiHttpServerDataSource.ɵfac });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/data_source/npmi_data_source_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class NpmiServerDataSourceModule {
    }
    NpmiServerDataSourceModule.decorators = [
        { type: NgModule, args: [{
                    imports: [TBHttpClientModule],
                    providers: [NpmiHttpServerDataSource],
                },] },
    ];
    /** @nocollapse */ NpmiServerDataSourceModule.ɵmod = ɵɵdefineNgModule({ type: NpmiServerDataSourceModule });
    /** @nocollapse */ NpmiServerDataSourceModule.ɵinj = ɵɵdefineInjector({ factory: function NpmiServerDataSourceModule_Factory(t) { return new (t || NpmiServerDataSourceModule)(); }, providers: [NpmiHttpServerDataSource], imports: [[TBHttpClientModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(NpmiServerDataSourceModule, { imports: [TBHttpClientModule] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/effects/npmi_effects.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class NpmiEffects {
        /**
         * @param {?} actions$
         * @param {?} store
         * @param {?} dataSource
         */
        constructor(actions$, store, dataSource) {
            this.actions$ = actions$;
            this.store = store;
            this.dataSource = dataSource;
            this.loadData$ = createEffect((/**
             * @return {?}
             */
            () => {
                /** @type {?} */
                const loadPluginData$ = this.loadPluginData();
                return merge(loadPluginData$).pipe(map((/**
                 * @return {?}
                 */
                () => ({}))));
            }), { dispatch: false });
        }
        /**
         * @private
         * @return {?}
         */
        loadPluginData() {
            return this.actions$.pipe(ofType(npmiLoaded), withLatestFrom(this.store.select(getPluginDataLoaded)), filter((/**
             * @param {?} __0
             * @return {?}
             */
            ([, { state }]) => state !== DataLoadState.LOADING)), tap((/**
             * @return {?}
             */
            () => this.store.dispatch(npmiPluginDataRequested()))), mergeMap((/**
             * @return {?}
             */
            () => {
                return this.dataSource.fetchData().pipe(tap((/**
                 * @param {?} result
                 * @return {?}
                 */
                (result) => {
                    this.store.dispatch(npmiPluginDataLoaded(result));
                })), map((/**
                 * @return {?}
                 */
                () => void null)), catchError((/**
                 * @return {?}
                 */
                () => {
                    this.store.dispatch(npmiPluginDataRequestFailed());
                    return EMPTY;
                })));
            })));
        }
    }
    NpmiEffects.decorators = [
        { type: Injectable },
    ];
    /** @nocollapse */
    NpmiEffects.ctorParameters = () => [
        { type: Actions },
        { type: Store },
        { type: NpmiHttpServerDataSource }
    ];
    /** @nocollapse */ NpmiEffects.ɵfac = function NpmiEffects_Factory(t) { return new (t || NpmiEffects)(ɵɵinject(Actions), ɵɵinject(Store), ɵɵinject(NpmiHttpServerDataSource)); };
    /** @nocollapse */ NpmiEffects.ɵprov = ɵɵdefineInjectable({ token: NpmiEffects, factory: NpmiEffects.ɵfac });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/inactive/inactive_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class InactiveModule$1 {
    }
    InactiveModule$1.decorators = [
        { type: NgModule, args: [{
                    declarations: [InactiveComponent$1],
                    exports: [InactiveComponent$1],
                },] },
    ];
    /** @nocollapse */ InactiveModule$1.ɵmod = ɵɵdefineNgModule({ type: InactiveModule$1 });
    /** @nocollapse */ InactiveModule$1.ɵinj = ɵɵdefineInjector({ factory: function InactiveModule_Factory(t) { return new (t || InactiveModule$1)(); } });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(InactiveModule$1, { declarations: [InactiveComponent$1], exports: [InactiveComponent$1] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/runs_legacy/views/legacy_runs_selector/legacy_runs_selector_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class LegacyRunsSelectorModule {
    }
    LegacyRunsSelectorModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [LegacyRunsSelectorComponent, LegacyRunsSelectorContainer],
                    exports: [LegacyRunsSelectorContainer],
                    schemas: [CUSTOM_ELEMENTS_SCHEMA],
                },] },
    ];
    /** @nocollapse */ LegacyRunsSelectorModule.ɵmod = ɵɵdefineNgModule({ type: LegacyRunsSelectorModule });
    /** @nocollapse */ LegacyRunsSelectorModule.ɵinj = ɵɵdefineInjector({ factory: function LegacyRunsSelectorModule_Factory(t) { return new (t || LegacyRunsSelectorModule)(); } });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(LegacyRunsSelectorModule, { declarations: [LegacyRunsSelectorComponent, LegacyRunsSelectorContainer], exports: [LegacyRunsSelectorContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/runs_legacy/runs_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class RunsModule$1 {
    }
    RunsModule$1.decorators = [
        { type: NgModule, args: [{
                    exports: [LegacyRunsSelectorModule],
                },] },
    ];
    /** @nocollapse */ RunsModule$1.ɵmod = ɵɵdefineNgModule({ type: RunsModule$1 });
    /** @nocollapse */ RunsModule$1.ɵinj = ɵɵdefineInjector({ factory: function RunsModule_Factory(t) { return new (t || RunsModule$1)(); }, imports: [LegacyRunsSelectorModule] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(RunsModule$1, { exports: [LegacyRunsSelectorModule] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/data_selection/metric_search/metric_search_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class MetricSearchModule {
    }
    MetricSearchModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [MetricSearchComponent, MetricSearchContainer],
                    imports: [CommonModule, MatIconModule, MatInputModule, MatAutocompleteModule],
                    exports: [MetricSearchContainer],
                },] },
    ];
    /** @nocollapse */ MetricSearchModule.ɵmod = ɵɵdefineNgModule({ type: MetricSearchModule });
    /** @nocollapse */ MetricSearchModule.ɵinj = ɵɵdefineInjector({ factory: function MetricSearchModule_Factory(t) { return new (t || MetricSearchModule)(); }, imports: [[CommonModule, MatIconModule, MatInputModule, MatAutocompleteModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(MetricSearchModule, { declarations: [MetricSearchComponent, MetricSearchContainer], imports: [CommonModule, MatIconModule, MatInputModule, MatAutocompleteModule], exports: [MetricSearchContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/data_selection/metric_arithmetic/metric_arithmetic_element/metric_arithmetic_element_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class MetricArithmeticElementModule {
    }
    MetricArithmeticElementModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [
                        MetricArithmeticElementComponent,
                        MetricArithmeticElementContainer,
                    ],
                    imports: [
                        CommonModule,
                        FormsModule,
                        ReactiveFormsModule,
                        MatIconModule,
                        MatChipsModule,
                        MatTooltipModule,
                    ],
                    exports: [MetricArithmeticElementContainer],
                },] },
    ];
    /** @nocollapse */ MetricArithmeticElementModule.ɵmod = ɵɵdefineNgModule({ type: MetricArithmeticElementModule });
    /** @nocollapse */ MetricArithmeticElementModule.ɵinj = ɵɵdefineInjector({ factory: function MetricArithmeticElementModule_Factory(t) { return new (t || MetricArithmeticElementModule)(); }, imports: [[
                CommonModule,
                FormsModule,
                ReactiveFormsModule,
                MatIconModule,
                MatChipsModule,
                MatTooltipModule,
            ]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(MetricArithmeticElementModule, { declarations: [MetricArithmeticElementComponent,
            MetricArithmeticElementContainer], imports: [CommonModule,
            FormsModule,
            ReactiveFormsModule,
            MatIconModule,
            MatChipsModule,
            MatTooltipModule], exports: [MetricArithmeticElementContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/data_selection/metric_arithmetic/metric_arithmetic_operator/metric_arithmetic_operator_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class MetricArithmeticOperatorModule {
    }
    MetricArithmeticOperatorModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [MetricArithmeticOperatorComponent],
                    imports: [CommonModule, MatChipsModule],
                    exports: [MetricArithmeticOperatorComponent],
                },] },
    ];
    /** @nocollapse */ MetricArithmeticOperatorModule.ɵmod = ɵɵdefineNgModule({ type: MetricArithmeticOperatorModule });
    /** @nocollapse */ MetricArithmeticOperatorModule.ɵinj = ɵɵdefineInjector({ factory: function MetricArithmeticOperatorModule_Factory(t) { return new (t || MetricArithmeticOperatorModule)(); }, imports: [[CommonModule, MatChipsModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(MetricArithmeticOperatorModule, { declarations: [MetricArithmeticOperatorComponent], imports: [CommonModule, MatChipsModule], exports: [MetricArithmeticOperatorComponent] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/data_selection/metric_arithmetic/metric_arithmetic_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class MetricArithmeticModule {
    }
    MetricArithmeticModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [MetricArithmeticComponent, MetricArithmeticContainer],
                    imports: [
                        CommonModule,
                        MatChipsModule,
                        MetricArithmeticElementModule,
                        MetricArithmeticOperatorModule,
                    ],
                    exports: [MetricArithmeticContainer],
                },] },
    ];
    /** @nocollapse */ MetricArithmeticModule.ɵmod = ɵɵdefineNgModule({ type: MetricArithmeticModule });
    /** @nocollapse */ MetricArithmeticModule.ɵinj = ɵɵdefineInjector({ factory: function MetricArithmeticModule_Factory(t) { return new (t || MetricArithmeticModule)(); }, imports: [[
                CommonModule,
                MatChipsModule,
                MetricArithmeticElementModule,
                MetricArithmeticOperatorModule,
            ]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(MetricArithmeticModule, { declarations: [MetricArithmeticComponent, MetricArithmeticContainer], imports: [CommonModule,
            MatChipsModule,
            MetricArithmeticElementModule,
            MetricArithmeticOperatorModule], exports: [MetricArithmeticContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/data_selection/results_download/results_download_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class ResultsDownloadModule {
    }
    ResultsDownloadModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [ResultsDownloadComponent, ResultsDownloadContainer],
                    imports: [CommonModule, MatIconModule, MatButtonModule],
                    exports: [ResultsDownloadContainer],
                },] },
    ];
    /** @nocollapse */ ResultsDownloadModule.ɵmod = ɵɵdefineNgModule({ type: ResultsDownloadModule });
    /** @nocollapse */ ResultsDownloadModule.ɵinj = ɵɵdefineInjector({ factory: function ResultsDownloadModule_Factory(t) { return new (t || ResultsDownloadModule)(); }, imports: [[CommonModule, MatIconModule, MatButtonModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(ResultsDownloadModule, { declarations: [ResultsDownloadComponent, ResultsDownloadContainer], imports: [CommonModule, MatIconModule, MatButtonModule], exports: [ResultsDownloadContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/data_selection/data_selection_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class DataSelectionModule {
    }
    DataSelectionModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [DataSelectionComponent],
                    imports: [
                        CommonModule,
                        MetricSearchModule,
                        MetricArithmeticModule,
                        ResultsDownloadModule,
                    ],
                    exports: [DataSelectionComponent],
                },] },
    ];
    /** @nocollapse */ DataSelectionModule.ɵmod = ɵɵdefineNgModule({ type: DataSelectionModule });
    /** @nocollapse */ DataSelectionModule.ɵinj = ɵɵdefineInjector({ factory: function DataSelectionModule_Factory(t) { return new (t || DataSelectionModule)(); }, imports: [[
                CommonModule,
                MetricSearchModule,
                MetricArithmeticModule,
                ResultsDownloadModule,
            ]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(DataSelectionModule, { declarations: [DataSelectionComponent], imports: [CommonModule,
            MetricSearchModule,
            MetricArithmeticModule,
            ResultsDownloadModule], exports: [DataSelectionComponent] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/violin_filters/violin_filter/violin_filter_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class ViolinFilterModule {
    }
    ViolinFilterModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [ViolinFilterComponent, ViolinFilterContainer],
                    imports: [CommonModule, FormsModule, MatButtonModule, MatIconModule],
                    exports: [ViolinFilterContainer],
                },] },
    ];
    /** @nocollapse */ ViolinFilterModule.ɵmod = ɵɵdefineNgModule({ type: ViolinFilterModule });
    /** @nocollapse */ ViolinFilterModule.ɵinj = ɵɵdefineInjector({ factory: function ViolinFilterModule_Factory(t) { return new (t || ViolinFilterModule)(); }, imports: [[CommonModule, FormsModule, MatButtonModule, MatIconModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(ViolinFilterModule, { declarations: [ViolinFilterComponent, ViolinFilterContainer], imports: [CommonModule, FormsModule, MatButtonModule, MatIconModule], exports: [ViolinFilterContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/violin_filters/violin_filters_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class ViolinFiltersModule {
    }
    ViolinFiltersModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [ViolinFiltersComponent, ViolinFiltersContainer],
                    imports: [
                        CommonModule,
                        FormsModule,
                        MatIconModule,
                        MatButtonModule,
                        ViolinFilterModule,
                    ],
                    exports: [ViolinFiltersContainer],
                },] },
    ];
    /** @nocollapse */ ViolinFiltersModule.ɵmod = ɵɵdefineNgModule({ type: ViolinFiltersModule });
    /** @nocollapse */ ViolinFiltersModule.ɵinj = ɵɵdefineInjector({ factory: function ViolinFiltersModule_Factory(t) { return new (t || ViolinFiltersModule)(); }, imports: [[
                CommonModule,
                FormsModule,
                MatIconModule,
                MatButtonModule,
                ViolinFilterModule,
            ]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(ViolinFiltersModule, { declarations: [ViolinFiltersComponent, ViolinFiltersContainer], imports: [CommonModule,
            FormsModule,
            MatIconModule,
            MatButtonModule,
            ViolinFilterModule], exports: [ViolinFiltersContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/annotations_list/annotations_list_toolbar/annotations_search/annotations_search_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class AnnotationsSearchModule {
    }
    AnnotationsSearchModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [AnnotationsSearchComponent, AnnotationsSearchContainer],
                    imports: [CommonModule, FormsModule, MatIconModule, MatInputModule],
                    exports: [AnnotationsSearchContainer],
                },] },
    ];
    /** @nocollapse */ AnnotationsSearchModule.ɵmod = ɵɵdefineNgModule({ type: AnnotationsSearchModule });
    /** @nocollapse */ AnnotationsSearchModule.ɵinj = ɵɵdefineInjector({ factory: function AnnotationsSearchModule_Factory(t) { return new (t || AnnotationsSearchModule)(); }, imports: [[CommonModule, FormsModule, MatIconModule, MatInputModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(AnnotationsSearchModule, { declarations: [AnnotationsSearchComponent, AnnotationsSearchContainer], imports: [CommonModule, FormsModule, MatIconModule, MatInputModule], exports: [AnnotationsSearchContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/annotations_list/annotations_list_toolbar/annotations_list_toolbar_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class AnnotationsListToolbarModule {
    }
    AnnotationsListToolbarModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [
                        AnnotationsListToolbarComponent,
                        AnnotationsListToolbarContainer,
                    ],
                    imports: [
                        CommonModule,
                        FormsModule,
                        MatIconModule,
                        MatButtonModule,
                        MatSlideToggleModule,
                        AnnotationsSearchModule,
                    ],
                    exports: [AnnotationsListToolbarContainer],
                },] },
    ];
    /** @nocollapse */ AnnotationsListToolbarModule.ɵmod = ɵɵdefineNgModule({ type: AnnotationsListToolbarModule });
    /** @nocollapse */ AnnotationsListToolbarModule.ɵinj = ɵɵdefineInjector({ factory: function AnnotationsListToolbarModule_Factory(t) { return new (t || AnnotationsListToolbarModule)(); }, imports: [[
                CommonModule,
                FormsModule,
                MatIconModule,
                MatButtonModule,
                MatSlideToggleModule,
                AnnotationsSearchModule,
            ]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(AnnotationsListToolbarModule, { declarations: [AnnotationsListToolbarComponent,
            AnnotationsListToolbarContainer], imports: [CommonModule,
            FormsModule,
            MatIconModule,
            MatButtonModule,
            MatSlideToggleModule,
            AnnotationsSearchModule], exports: [AnnotationsListToolbarContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/annotations_list/header/header_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class HeaderModule$1 {
    }
    HeaderModule$1.decorators = [
        { type: NgModule, args: [{
                    declarations: [HeaderComponent$1, HeaderContainer],
                    imports: [CommonModule, FormsModule, MatCheckboxModule, MatIconModule],
                    exports: [HeaderContainer],
                },] },
    ];
    /** @nocollapse */ HeaderModule$1.ɵmod = ɵɵdefineNgModule({ type: HeaderModule$1 });
    /** @nocollapse */ HeaderModule$1.ɵinj = ɵɵdefineInjector({ factory: function HeaderModule_Factory(t) { return new (t || HeaderModule$1)(); }, imports: [[CommonModule, FormsModule, MatCheckboxModule, MatIconModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(HeaderModule$1, { declarations: [HeaderComponent$1, HeaderContainer], imports: [CommonModule, FormsModule, MatCheckboxModule, MatIconModule], exports: [HeaderContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/annotations_list/legend/legend_element/legend_element_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class LegendElementModule {
    }
    LegendElementModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [LegendElementComponent],
                    imports: [CommonModule],
                    exports: [LegendElementComponent],
                },] },
    ];
    /** @nocollapse */ LegendElementModule.ɵmod = ɵɵdefineNgModule({ type: LegendElementModule });
    /** @nocollapse */ LegendElementModule.ɵinj = ɵɵdefineInjector({ factory: function LegendElementModule_Factory(t) { return new (t || LegendElementModule)(); }, imports: [[CommonModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(LegendElementModule, { declarations: [LegendElementComponent], imports: [CommonModule], exports: [LegendElementComponent] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/annotations_list/legend/legend_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class LegendModule {
    }
    LegendModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [LegendComponent],
                    imports: [LegendElementModule],
                    exports: [LegendComponent],
                },] },
    ];
    /** @nocollapse */ LegendModule.ɵmod = ɵɵdefineNgModule({ type: LegendModule });
    /** @nocollapse */ LegendModule.ɵinj = ɵɵdefineInjector({ factory: function LegendModule_Factory(t) { return new (t || LegendModule)(); }, imports: [[LegendElementModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(LegendModule, { declarations: [LegendComponent], imports: [LegendElementModule], exports: [LegendComponent] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/annotations_list/annotation/annotation_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class AnnotationModule {
    }
    AnnotationModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [AnnotationContainer, AnnotationComponent],
                    imports: [CommonModule, FormsModule, MatCheckboxModule, MatIconModule],
                    exports: [AnnotationContainer],
                },] },
    ];
    /** @nocollapse */ AnnotationModule.ɵmod = ɵɵdefineNgModule({ type: AnnotationModule });
    /** @nocollapse */ AnnotationModule.ɵinj = ɵɵdefineInjector({ factory: function AnnotationModule_Factory(t) { return new (t || AnnotationModule)(); }, imports: [[CommonModule, FormsModule, MatCheckboxModule, MatIconModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(AnnotationModule, { declarations: [AnnotationContainer, AnnotationComponent], imports: [CommonModule, FormsModule, MatCheckboxModule, MatIconModule], exports: [AnnotationContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/annotations_list/annotations_list_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class AnnotationsListModule {
    }
    AnnotationsListModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [AnnotationsListComponent, AnnotationsListContainer],
                    imports: [
                        CommonModule,
                        AnnotationsListToolbarModule,
                        HeaderModule$1,
                        LegendModule,
                        ScrollingModule,
                        AnnotationModule,
                    ],
                    exports: [AnnotationsListContainer],
                },] },
    ];
    /** @nocollapse */ AnnotationsListModule.ɵmod = ɵɵdefineNgModule({ type: AnnotationsListModule });
    /** @nocollapse */ AnnotationsListModule.ɵinj = ɵɵdefineInjector({ factory: function AnnotationsListModule_Factory(t) { return new (t || AnnotationsListModule)(); }, imports: [[
                CommonModule,
                AnnotationsListToolbarModule,
                HeaderModule$1,
                LegendModule,
                ScrollingModule,
                AnnotationModule,
            ]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(AnnotationsListModule, { declarations: [AnnotationsListComponent, AnnotationsListContainer], imports: [CommonModule,
            AnnotationsListToolbarModule,
            HeaderModule$1,
            LegendModule,
            ScrollingModule,
            AnnotationModule], exports: [AnnotationsListContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/selected_annotations/parallel_coordinates/parallel_coordinates_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class ParallelCoordinatesModule {
    }
    ParallelCoordinatesModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [ParallelCoordinatesComponent, ParallelCoordinatesContainer],
                    imports: [CommonModule],
                    exports: [ParallelCoordinatesContainer],
                },] },
    ];
    /** @nocollapse */ ParallelCoordinatesModule.ɵmod = ɵɵdefineNgModule({ type: ParallelCoordinatesModule });
    /** @nocollapse */ ParallelCoordinatesModule.ɵinj = ɵɵdefineInjector({ factory: function ParallelCoordinatesModule_Factory(t) { return new (t || ParallelCoordinatesModule)(); }, imports: [[CommonModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(ParallelCoordinatesModule, { declarations: [ParallelCoordinatesComponent, ParallelCoordinatesContainer], imports: [CommonModule], exports: [ParallelCoordinatesContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/selected_annotations/selected_annotations_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class SelectedAnnotationsModule {
    }
    SelectedAnnotationsModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [SelectedAnnotationsComponent, SelectedAnnotationsContainer],
                    imports: [
                        CommonModule,
                        MatIconModule,
                        MatButtonModule,
                        ParallelCoordinatesModule,
                    ],
                    exports: [SelectedAnnotationsContainer],
                },] },
    ];
    /** @nocollapse */ SelectedAnnotationsModule.ɵmod = ɵɵdefineNgModule({ type: SelectedAnnotationsModule });
    /** @nocollapse */ SelectedAnnotationsModule.ɵinj = ɵɵdefineInjector({ factory: function SelectedAnnotationsModule_Factory(t) { return new (t || SelectedAnnotationsModule)(); }, imports: [[
                CommonModule,
                MatIconModule,
                MatButtonModule,
                ParallelCoordinatesModule,
            ]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(SelectedAnnotationsModule, { declarations: [SelectedAnnotationsComponent, SelectedAnnotationsContainer], imports: [CommonModule,
            MatIconModule,
            MatButtonModule,
            ParallelCoordinatesModule], exports: [SelectedAnnotationsContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/views/main/main_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class MainModule {
    }
    MainModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [MainComponent, MainContainer],
                    imports: [
                        CommonModule,
                        FormsModule,
                        MatCheckboxModule,
                        MatIconModule,
                        RunsModule$1,
                        DataSelectionModule,
                        MatButtonModule,
                        ViolinFiltersModule,
                        AnnotationsListModule,
                        SelectedAnnotationsModule,
                    ],
                    exports: [MainContainer],
                },] },
    ];
    /** @nocollapse */ MainModule.ɵmod = ɵɵdefineNgModule({ type: MainModule });
    /** @nocollapse */ MainModule.ɵinj = ɵɵdefineInjector({ factory: function MainModule_Factory(t) { return new (t || MainModule)(); }, imports: [[
                CommonModule,
                FormsModule,
                MatCheckboxModule,
                MatIconModule,
                RunsModule$1,
                DataSelectionModule,
                MatButtonModule,
                ViolinFiltersModule,
                AnnotationsListModule,
                SelectedAnnotationsModule,
            ]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(MainModule, { declarations: [MainComponent, MainContainer], imports: [CommonModule,
            FormsModule,
            MatCheckboxModule,
            MatIconModule,
            RunsModule$1,
            DataSelectionModule,
            MatButtonModule,
            ViolinFiltersModule,
            AnnotationsListModule,
            SelectedAnnotationsModule], exports: [MainContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/npmi/npmi_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class NpmiModule {
    }
    NpmiModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [NpmiComponent, NpmiContainer],
                    imports: [
                        CommonModule,
                        InactiveModule$1,
                        MainModule,
                        NpmiServerDataSourceModule,
                        StoreModule.forFeature(NPMI_FEATURE_KEY, reducers$8),
                        EffectsModule.forFeature([NpmiEffects]),
                        PluginRegistryModule.forPlugin('npmi', NpmiContainer),
                    ],
                    exports: [NpmiContainer],
                    entryComponents: [NpmiContainer],
                },] },
    ];
    /** @nocollapse */ NpmiModule.ɵmod = ɵɵdefineNgModule({ type: NpmiModule });
    /** @nocollapse */ NpmiModule.ɵinj = ɵɵdefineInjector({ factory: function NpmiModule_Factory(t) { return new (t || NpmiModule)(); }, imports: [[
                CommonModule,
                InactiveModule$1,
                MainModule,
                NpmiServerDataSourceModule,
                StoreModule.forFeature(NPMI_FEATURE_KEY, reducers$8),
                EffectsModule.forFeature([NpmiEffects]),
                PluginRegistryModule.forPlugin('npmi', NpmiContainer),
            ]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(NpmiModule, { declarations: [NpmiComponent, NpmiContainer], imports: [CommonModule,
            InactiveModule$1,
            MainModule,
            NpmiServerDataSourceModule, StoreFeatureModule, EffectsFeatureModule, PluginRegistryModule], exports: [NpmiContainer] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/text_v2/views/text_dashboard/text_dashboard_component.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class TextDashboardComponent {
    }
    TextDashboardComponent.decorators = [
        { type: Component, args: [{
                    selector: 'text-dashboard',
                    template: ` This is the text dashboard `,
                    changeDetection: ChangeDetectionStrategy.OnPush,
                },] },
    ];
    /** @nocollapse */ TextDashboardComponent.ɵfac = function TextDashboardComponent_Factory(t) { return new (t || TextDashboardComponent)(); };
    /** @nocollapse */ TextDashboardComponent.ɵcmp = ɵɵdefineComponent({ type: TextDashboardComponent, selectors: [["text-dashboard"]], decls: 1, vars: 0, template: function TextDashboardComponent_Template(rf, ctx) { if (rf & 1) {
            ɵɵtext(0, " This is the text dashboard ");
        } }, encapsulation: 2, changeDetection: 0 });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/text_v2/views/text_dashboard/text_dashboard_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class TextDashboardModule {
    }
    TextDashboardModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [TextDashboardComponent],
                    exports: [TextDashboardComponent],
                },] },
    ];
    /** @nocollapse */ TextDashboardModule.ɵmod = ɵɵdefineNgModule({ type: TextDashboardModule });
    /** @nocollapse */ TextDashboardModule.ɵinj = ɵɵdefineInjector({ factory: function TextDashboardModule_Factory(t) { return new (t || TextDashboardModule)(); } });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(TextDashboardModule, { declarations: [TextDashboardComponent], exports: [TextDashboardComponent] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/text_v2/data_source/text_v2_data_source.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @abstract
     */
    class TextV2DataSource {
    }
    TextV2DataSource.decorators = [
        { type: Injectable },
    ];
    /** @nocollapse */ TextV2DataSource.ɵfac = function TextV2DataSource_Factory(t) { return new (t || TextV2DataSource)(); };
    /** @nocollapse */ TextV2DataSource.ɵprov = ɵɵdefineInjectable({ token: TextV2DataSource, factory: TextV2DataSource.ɵfac });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/text_v2/data_source/text_v2_server_data_source.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class TextV2ServerDataSource {
        /**
         * @param {?} http
         */
        constructor(http) {
            this.http = http;
            this.httpPathPrefix = 'data/plugin/text_v2';
        }
        /**
         * @return {?}
         */
        fetchRunToTag() {
            return this.http
                .get(this.httpPathPrefix + '/tags')
                .pipe(map((/**
             * @param {?} runToTagObject
             * @return {?}
             */
            (runToTagObject) => {
                /** @type {?} */
                const runToTag = new Map();
                Object.entries(runToTagObject).forEach((/**
                 * @param {?} __0
                 * @return {?}
                 */
                ([runName, tags]) => {
                    runToTag.set(runName, tags);
                }));
                return runToTag;
            })));
        }
        /**
         * @param {?} run
         * @param {?} tag
         * @return {?}
         */
        fetchTextData(run, tag) {
            /** @type {?} */
            const searchParams = new URLSearchParams({ run, tag });
            return this.http
                .get(this.httpPathPrefix + `/text?${searchParams.toString()}`)
                .pipe(map((/**
             * @param {?} dataList
             * @return {?}
             */
            (dataList) => {
                return dataList.map((/**
                 * @param {?} datum
                 * @return {?}
                 */
                (datum) => {
                    return {
                        originalShape: datum.original_shape,
                        step: datum.step,
                        stringArray: datum.string_array,
                        wallTimeInMs: datum.wall_time * 1000,
                        truncated: datum.truncated,
                    };
                }));
            })));
        }
    }
    TextV2ServerDataSource.decorators = [
        { type: Injectable },
    ];
    /** @nocollapse */
    TextV2ServerDataSource.ctorParameters = () => [
        { type: TBHttpClient }
    ];
    /** @nocollapse */ TextV2ServerDataSource.ɵfac = function TextV2ServerDataSource_Factory(t) { return new (t || TextV2ServerDataSource)(ɵɵinject(TBHttpClient)); };
    /** @nocollapse */ TextV2ServerDataSource.ɵprov = ɵɵdefineInjectable({ token: TextV2ServerDataSource, factory: TextV2ServerDataSource.ɵfac });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/text_v2/data_source/text_v2_data_source_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class TextV2DataSourceModule {
    }
    TextV2DataSourceModule.decorators = [
        { type: NgModule, args: [{
                    imports: [TBHttpClientModule],
                    providers: [
                        TextV2ServerDataSource,
                        { provide: TextV2DataSource, useExisting: TextV2ServerDataSource },
                    ],
                },] },
    ];
    /** @nocollapse */ TextV2DataSourceModule.ɵmod = ɵɵdefineNgModule({ type: TextV2DataSourceModule });
    /** @nocollapse */ TextV2DataSourceModule.ɵinj = ɵɵdefineInjector({ factory: function TextV2DataSourceModule_Factory(t) { return new (t || TextV2DataSourceModule)(); }, providers: [
            TextV2ServerDataSource,
            { provide: TextV2DataSource, useExisting: TextV2ServerDataSource },
        ], imports: [[TBHttpClientModule]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(TextV2DataSourceModule, { imports: [TBHttpClientModule] }); })();

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    const textPluginLoaded = createAction('[Text] Text Plugin Loaded');
    const textRunToTagsLoaded = createAction('[Text] Runs To Tag Loaded', props());
    const textTagGroupVisibilityChanged = createAction('[Text] Tag Group Visibility Changed', props());
    const textDataLoaded = createAction('[Text] Text Data Loaded Loaded', props());

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/text_v2/store/text_types.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const TEXT_FEATURE_KEY = 'text';

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/text_v2/store/text_v2_selectors.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const selectTextState = createFeatureSelector(TEXT_FEATURE_KEY);
    /** @type {?} */
    const getTextRunToTags = createSelector(selectTextState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => state.runToTags));
    /**
     * Returns de-duplicated list of <run, tag> tuple for cards that are visible
     * in the UI.
     * @type {?}
     */
    const getTextAllVisibleRunTags = createSelector(selectTextState, (/**
     * @param {?} state
     * @return {?}
     */
    (state) => {
        /** @type {?} */
        const serializedRunTagTuples = new Set();
        /** @type {?} */
        const allVisibleRunTagTuples = new Set();
        for (const runTagTuples of state.visibleRunTags.values()) {
            for (const runTag of runTagTuples) {
                /** @type {?} */
                const serializedRunTag = JSON.stringify(runTag);
                if (serializedRunTagTuples.has(serializedRunTag)) {
                    continue;
                }
                serializedRunTagTuples.add(serializedRunTag);
                allVisibleRunTagTuples.add(runTag);
            }
        }
        return [...allVisibleRunTagTuples];
    }));
    /** @type {?} */
    const getTextData = createSelector(selectTextState, (/**
     * @param {?} state
     * @param {?} props
     * @return {?}
     */
    (state, props) => {
        // Refactor to `state.data.get(props.run)?.get(props.tag) || null` when prettier
        // supports TypeScript 3.8 (prettier 2.x).
        /** @type {?} */
        const tagToSteps = state.data.get(props.run);
        if (!tagToSteps) {
            return null;
        }
        return tagToSteps.get(props.tag) || null;
    }));

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/text_v2/effects/text_effects.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class TextEffects {
        /**
         * @param {?} actions$
         * @param {?} store
         * @param {?} dataSource
         */
        constructor(actions$, store, dataSource) {
            this.actions$ = actions$;
            this.store = store;
            this.dataSource = dataSource;
            /**
             * @export
             */
            this.loadRunToTags$ = createEffect((/**
             * @return {?}
             */
            () => {
                return this.actions$.pipe(ofType(textPluginLoaded), switchMap((/**
                 * @return {?}
                 */
                () => {
                    return this.dataSource.fetchRunToTag().pipe(tap((/**
                     * @param {?} runToTags
                     * @return {?}
                     */
                    (runToTags) => {
                        this.store.dispatch(textRunToTagsLoaded({ runToTags }));
                    })), map((/**
                     * @return {?}
                     */
                    () => void null)));
                })));
            }), { dispatch: false });
            /**
             * @export
             */
            this.loadData$ = createEffect((/**
             * @return {?}
             */
            () => {
                /** @type {?} */
                const fetchOnNewCardVisible = this.actions$.pipe(ofType(textTagGroupVisibilityChanged), switchMap((/**
                 * @param {?} __0
                 * @return {?}
                 */
                ({ visibleTextCards }) => {
                    // Fetch existing data.
                    /** @type {?} */
                    const existingTextData = visibleTextCards.map((/**
                     * @param {?} __0
                     * @return {?}
                     */
                    ({ run, tag }) => {
                        return this.store.select(getTextData, { run, tag }).pipe(last(), map((/**
                         * @param {?} textData
                         * @return {?}
                         */
                        (textData) => {
                            return { run, tag, textData };
                        })));
                    }));
                    return forkJoin(existingTextData).pipe(map((/**
                     * @param {?} textData
                     * @return {?}
                     */
                    (textData) => {
                        // Filter out the <run, tag> tuple if the data already exists.
                        return textData
                            .filter((/**
                         * @param {?} __0
                         * @return {?}
                         */
                        ({ textData }) => textData === null))
                            .map((/**
                         * @param {?} __0
                         * @return {?}
                         */
                        ({ run, tag }) => ({ run, tag })));
                    })));
                })));
                /** @type {?} */
                const fetchVisibleCardsOnReload = this.actions$.pipe(ofType(manualReload, reload), withLatestFrom(this.store.select(getTextAllVisibleRunTags)), map((/**
                 * @param {?} __0
                 * @return {?}
                 */
                ([, runTagList]) => runTagList)));
                return merge(fetchOnNewCardVisible, fetchVisibleCardsOnReload).pipe(mergeMap((/**
                 * @param {?} runTagPairs
                 * @return {?}
                 */
                (runTagPairs) => {
                    return forkJoin(runTagPairs.map((/**
                     * @param {?} runAndTag
                     * @return {?}
                     */
                    (runAndTag) => {
                        return this.fetchTextData(runAndTag);
                    })));
                })));
            }), { dispatch: false });
        }
        /**
         * @private
         * @param {?} props
         * @return {?}
         */
        fetchTextData(props) {
            const { run, tag } = props;
            return this.dataSource.fetchTextData(run, tag).pipe(tap((/**
             * @param {?} stepData
             * @return {?}
             */
            (stepData) => {
                this.store.dispatch(textDataLoaded({ run, tag, stepData }));
            })), map((/**
             * @return {?}
             */
            () => void null)));
        }
    }
    TextEffects.decorators = [
        { type: Injectable },
    ];
    /** @nocollapse */
    TextEffects.ctorParameters = () => [
        { type: Actions },
        { type: Store },
        { type: TextV2DataSource }
    ];
    /** @nocollapse */ TextEffects.ɵfac = function TextEffects_Factory(t) { return new (t || TextEffects)(ɵɵinject(Actions), ɵɵinject(Store), ɵɵinject(TextV2DataSource)); };
    /** @nocollapse */ TextEffects.ɵprov = ɵɵdefineInjectable({ token: TextEffects, factory: TextEffects.ɵfac });

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/text_v2/store/text_v2_reducers.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /** @type {?} */
    const DATA_A_B_RUN1 = [
        {
            originalShape: [3],
            step: 0,
            stringArray: [['foo', 'bar', 'baz']],
            wallTimeInMs: 1577865600000,
            truncated: false,
        },
        {
            originalShape: [3],
            step: 1,
            stringArray: [['foo', 'baz']],
            wallTimeInMs: 1577865601000,
            truncated: false,
        },
    ];
    /** @type {?} */
    const DATA_A_C_RUN1 = [
        {
            originalShape: [3],
            step: 0,
            stringArray: [
                [
                    'We conducted an experiment and found the following data:\n\nPounds of chocolate | Happiness\n---|---\n0 | 1\n1 | 4\n2 | 9\n3 | 16\n4 | 25\n5 | 36\n6 | 49\n7 | 64\n8 | 81\n9 | 100\n10 | 121',
                ],
            ],
            wallTimeInMs: 1577865600000,
            truncated: false,
        },
        {
            originalShape: [3],
            step: 1,
            stringArray: [
                ['\u00d7', '**0**', '**1**', '**2**', '**3**', '**4**', '**5**'],
                ['**0**', '0', '0', '0', '0', '0', '0'],
                ['**1**', '0', '1', '2', '3', '4', '5'],
                ['**2**', '0', '2', '4', '6', '8', '10'],
                ['**3**', '0', '3', '6', '9', '12', '15'],
                ['**4**', '0', '4', '8', '12', '16', '20'],
                ['**5**', '0', '5', '10', '15', '20', '25'],
            ],
            wallTimeInMs: 1577865601000,
            truncated: false,
        },
    ];
    /** @type {?} */
    const initialState$7 = {
        runToTags: new Map([
            ['run1', ['a/b', 'a/c']],
            ['run2', ['a/b', 'a/d']],
            ['run3', ['c', 'a/b']],
        ]),
        data: new Map([
            [
                'run1',
                new Map([
                    ['a/b', DATA_A_B_RUN1],
                    ['a/c', DATA_A_C_RUN1],
                ]),
            ],
        ]),
        visibleRunTags: new Map(),
    };
    /** @type {?} */
    const reducer$7 = createReducer(initialState$7);
    /**
     * @param {?} state
     * @param {?} action
     * @return {?}
     */
    function reducers$9(state, action) {
        return reducer$7(state, action);
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/plugins/text_v2/text_v2_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    class TextV2Module {
    }
    TextV2Module.decorators = [
        { type: NgModule, args: [{
                    imports: [
                        CommonModule,
                        TextDashboardModule,
                        PluginRegistryModule.forPlugin('text_v2', TextDashboardComponent),
                        TextV2DataSourceModule,
                        StoreModule.forFeature(TEXT_FEATURE_KEY, reducers$9),
                        EffectsModule.forFeature([TextEffects]),
                    ],
                    entryComponents: [TextDashboardComponent],
                },] },
    ];
    /** @nocollapse */ TextV2Module.ɵmod = ɵɵdefineNgModule({ type: TextV2Module });
    /** @nocollapse */ TextV2Module.ɵinj = ɵɵdefineInjector({ factory: function TextV2Module_Factory(t) { return new (t || TextV2Module)(); }, imports: [[
                CommonModule,
                TextDashboardModule,
                PluginRegistryModule.forPlugin('text_v2', TextDashboardComponent),
                TextV2DataSourceModule,
                StoreModule.forFeature(TEXT_FEATURE_KEY, reducers$9),
                EffectsModule.forFeature([TextEffects]),
            ]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(TextV2Module, { imports: [CommonModule,
            TextDashboardModule, PluginRegistryModule, TextV2DataSourceModule, StoreFeatureModule, EffectsFeatureModule] }); })();

    /* Copyright 2020 The TensorFlow Authors. All Rights Reserved.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    ==============================================================================*/
    /**
     *
     * @fileoverview This module enumerates all Angular based plugins TensorBoard supports.
     *
     * Generated from: tensorboard/webapp/oss_plugins_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class OssPluginsModule {
    }
    OssPluginsModule.decorators = [
        { type: NgModule, args: [{
                    imports: [DebuggerModule, MetricsModule, NpmiModule, TextV2Module],
                },] },
    ];
    /** @nocollapse */ OssPluginsModule.ɵmod = ɵɵdefineNgModule({ type: OssPluginsModule });
    /** @nocollapse */ OssPluginsModule.ɵinj = ɵɵdefineInjector({ factory: function OssPluginsModule_Factory(t) { return new (t || OssPluginsModule)(); }, imports: [[DebuggerModule, MetricsModule, NpmiModule, TextV2Module]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(OssPluginsModule, { imports: [DebuggerModule, MetricsModule, NpmiModule, TextV2Module] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/app_routing/deep_link_provider.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    /**
     * @abstract
     */
    class DeepLinkProvider {
    }
    DeepLinkProvider.decorators = [
        { type: Injectable },
    ];
    /** @nocollapse */ DeepLinkProvider.ɵfac = function DeepLinkProvider_Factory(t) { return new (t || DeepLinkProvider)(); };
    /** @nocollapse */ DeepLinkProvider.ɵprov = ɵɵdefineInjectable({ token: DeepLinkProvider, factory: DeepLinkProvider.ɵfac });

    /**
     * Provides deeplinking for the core dashboards page.
     */
    let CoreDeepLinkProvider = class CoreDeepLinkProvider extends DeepLinkProvider {
        getMetricsPinnedCards(store) {
            return combineLatest([
                store.select(getPinnedCardsWithMetadata),
                store.select(getUnresolvedImportedPinnedCards),
            ]).pipe(map(([pinnedCards, unresolvedImportedPinnedCards]) => {
                if (!pinnedCards.length && !unresolvedImportedPinnedCards.length) {
                    return [];
                }
                const pinnedCardsToStore = pinnedCards.map(({ plugin, tag, sample, runId }) => {
                    const info = { plugin, tag };
                    if (isSingleRunPlugin(plugin)) {
                        info.runId = runId;
                    }
                    if (isSampledPlugin(plugin)) {
                        info.sample = sample;
                    }
                    return info;
                });
                // Intentionally order unresolved cards last, so that cards pinned by
                // the user in this session have priority.
                const cardsToStore = [
                    ...pinnedCardsToStore,
                    ...unresolvedImportedPinnedCards,
                ];
                return [{ key: 'pinnedCards', value: JSON.stringify(cardsToStore) }];
            }));
        }
        serializeStateToQueryParams(store) {
            return this.getMetricsPinnedCards(store);
        }
        deserializeQueryParams(queryParams) {
            let pinnedCards = null;
            for (const { key, value } of queryParams) {
                if (key === 'pinnedCards') {
                    pinnedCards = extractPinnedCardsFromURLText(value);
                    break;
                }
            }
            return {
                metrics: {
                    pinnedCards: pinnedCards || [],
                },
            };
        }
    };
    CoreDeepLinkProvider = __decorate([
        Injectable()
    ], CoreDeepLinkProvider);
    function extractPinnedCardsFromURLText(urlText) {
        // Check that the URL text parses.
        let object;
        try {
            object = JSON.parse(urlText);
        }
        catch (_a) {
            return null;
        }
        if (!Array.isArray(object)) {
            return null;
        }
        const result = [];
        for (const item of object) {
            // Validate types.
            const isPluginString = typeof item.plugin === 'string';
            const isRunString = typeof item.runId === 'string';
            const isSampleNumber = typeof item.sample === 'number';
            const isTagString = typeof item.tag === 'string';
            const isRunTypeValid = isRunString || typeof item.runId === 'undefined';
            const isSampleTypeValid = isSampleNumber || typeof item.sample === 'undefined';
            if (!isPluginString ||
                !isTagString ||
                !isRunTypeValid ||
                !isSampleTypeValid) {
                continue;
            }
            // Required fields and range errors.
            if (!isPluginType(item.plugin)) {
                continue;
            }
            if (!item.tag) {
                continue;
            }
            if (isSingleRunPlugin(item.plugin)) {
                // A single run plugin must specify a non-empty run.
                if (!item.runId) {
                    continue;
                }
            }
            else {
                // A multi run plugin must not specify a run.
                if (item.runId) {
                    continue;
                }
            }
            if (isSampleNumber) {
                if (!isSampledPlugin(item.plugin)) {
                    continue;
                }
                if (!Number.isInteger(item.sample) || item.sample < 0) {
                    continue;
                }
            }
            // Assemble result.
            const resultItem = { plugin: item.plugin, tag: item.tag };
            if (isRunString) {
                resultItem.runId = item.runId;
            }
            if (isSampleNumber) {
                resultItem.sample = item.sample;
            }
            result.push(resultItem);
        }
        return result;
    }

    function routesFactory() {
        return [
            {
                routeKind: RouteKind.EXPERIMENT,
                path: '/',
                ngComponent: TensorBoardWrapperComponent,
                defaultRoute: true,
                deepLinkProvider: new CoreDeepLinkProvider(),
            },
        ];
    }

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/app_module.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    class AppModule {
    }
    AppModule.decorators = [
        { type: NgModule, args: [{
                    declarations: [AppContainer],
                    imports: [
                        // Ensure feature flags are enabled before they are consumed.
                        FeatureFlagModule,
                        BrowserModule,
                        BrowserAnimationsModule,
                        AppRoutingModule,
                        AppRoutingViewModule,
                        RouteRegistryModule.registerRoutes(routesFactory),
                        AlertModule,
                        AlertSnackbarModule,
                        TensorBoardWrapperModule,
                        CoreModule,
                        ExperimentsModule,
                        HashStorageModule,
                        HeaderModule,
                        MatIconModule$1,
                        PageTitleModule,
                        PluginsModule,
                        RunsModule,
                        SettingsModule,
                        StoreModule.forRoot(ROOT_REDUCERS, {
                            runtimeChecks: {
                                strictStateSerializability: false,
                                strictActionSerializability: true,
                            },
                        }),
                        EffectsModule.forRoot([]),
                        OssPluginsModule,
                    ],
                    providers: [
                        {
                            provide: META_REDUCERS,
                            useFactory: loggerMetaReducerFactory,
                            multi: true,
                        },
                    ],
                    bootstrap: [AppContainer],
                },] },
    ];
    /** @nocollapse */ AppModule.ɵmod = ɵɵdefineNgModule({ type: AppModule, bootstrap: [AppContainer] });
    /** @nocollapse */ AppModule.ɵinj = ɵɵdefineInjector({ factory: function AppModule_Factory(t) { return new (t || AppModule)(); }, providers: [
            {
                provide: META_REDUCERS,
                useFactory: loggerMetaReducerFactory,
                multi: true,
            },
        ], imports: [[
                // Ensure feature flags are enabled before they are consumed.
                FeatureFlagModule,
                BrowserModule,
                BrowserAnimationsModule,
                AppRoutingModule,
                AppRoutingViewModule,
                RouteRegistryModule.registerRoutes(routesFactory),
                AlertModule,
                AlertSnackbarModule,
                TensorBoardWrapperModule,
                CoreModule,
                ExperimentsModule,
                HashStorageModule,
                HeaderModule,
                MatIconModule$1,
                PageTitleModule,
                PluginsModule,
                RunsModule,
                SettingsModule,
                StoreModule.forRoot(ROOT_REDUCERS, {
                    runtimeChecks: {
                        strictStateSerializability: false,
                        strictActionSerializability: true,
                    },
                }),
                EffectsModule.forRoot([]),
                OssPluginsModule,
            ]] });
    (function () { (typeof ngJitMode === "undefined" || ngJitMode) && ɵɵsetNgModuleScope(AppModule, { declarations: [AppContainer], imports: [
            // Ensure feature flags are enabled before they are consumed.
            FeatureFlagModule,
            BrowserModule,
            BrowserAnimationsModule,
            AppRoutingModule,
            AppRoutingViewModule, RouteRegistryModule, AlertModule,
            AlertSnackbarModule,
            TensorBoardWrapperModule,
            CoreModule,
            ExperimentsModule,
            HashStorageModule,
            HeaderModule,
            MatIconModule$1,
            PageTitleModule,
            PluginsModule,
            RunsModule,
            SettingsModule, StoreRootModule, EffectsRootModule, OssPluginsModule] }); })();

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/bootstrap.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    // Bootstrap needs to happen after body is ready but we cannot reliably
    // controls the order in which script gets loaded (Vulcanization inlines
    // the script in <head>).
    window.addEventListener('DOMContentLoaded', (/**
     * @return {?}
     */
    () => {
        platformBrowser().bootstrapModule(AppModule);
    }));

    /**
     * @fileoverview added by tsickle
     * Generated from: tensorboard/webapp/main_prod.ts
     * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
     */
    enableProdMode();

}());


", + "ok": true, + "headers": [ + [ + "content-type", + "application/javascript; charset=utf-8" + ] + ], + "status": 200, + "status_text": "" + }, + "https://localhost:6006/data/environment": { + "data": "eyJkYXRhX2xvY2F0aW9uIjogIi4vbXlfY2lmYXIxMF9sb2dzIiwgIndpbmRvd190aXRsZSI6ICIifQ==", + "ok": true, + "headers": [ + [ + "content-type", + "application/json" + ] + ], + "status": 200, + "status_text": "" + }, + "https://localhost:6006/data/plugins_listing": { + "data": "eyJzY2FsYXJzIjogeyJkaXNhYmxlX3JlbG9hZCI6IGZhbHNlLCAiZW5hYmxlZCI6IGZhbHNlLCAicmVtb3ZlX2RvbSI6IGZhbHNlLCAidGFiX25hbWUiOiAic2NhbGFycyIsICJsb2FkaW5nX21lY2hhbmlzbSI6IHsidHlwZSI6ICJDVVNUT01fRUxFTUVOVCIsICJlbGVtZW50X25hbWUiOiAidGYtc2NhbGFyLWRhc2hib2FyZCJ9fSwgImN1c3RvbV9zY2FsYXJzIjogeyJkaXNhYmxlX3JlbG9hZCI6IGZhbHNlLCAiZW5hYmxlZCI6IGZhbHNlLCAicmVtb3ZlX2RvbSI6IGZhbHNlLCAidGFiX25hbWUiOiAiQ3VzdG9tIFNjYWxhcnMiLCAibG9hZGluZ19tZWNoYW5pc20iOiB7InR5cGUiOiAiQ1VTVE9NX0VMRU1FTlQiLCAiZWxlbWVudF9uYW1lIjogInRmLWN1c3RvbS1zY2FsYXItZGFzaGJvYXJkIn19LCAiaW1hZ2VzIjogeyJkaXNhYmxlX3JlbG9hZCI6IGZhbHNlLCAiZW5hYmxlZCI6IGZhbHNlLCAicmVtb3ZlX2RvbSI6IGZhbHNlLCAidGFiX25hbWUiOiAiaW1hZ2VzIiwgImxvYWRpbmdfbWVjaGFuaXNtIjogeyJ0eXBlIjogIkNVU1RPTV9FTEVNRU5UIiwgImVsZW1lbnRfbmFtZSI6ICJ0Zi1pbWFnZS1kYXNoYm9hcmQifX0sICJhdWRpbyI6IHsiZGlzYWJsZV9yZWxvYWQiOiBmYWxzZSwgImVuYWJsZWQiOiBmYWxzZSwgInJlbW92ZV9kb20iOiBmYWxzZSwgInRhYl9uYW1lIjogImF1ZGlvIiwgImxvYWRpbmdfbWVjaGFuaXNtIjogeyJ0eXBlIjogIkNVU1RPTV9FTEVNRU5UIiwgImVsZW1lbnRfbmFtZSI6ICJ0Zi1hdWRpby1kYXNoYm9hcmQifX0sICJkZWJ1Z2dlci12MiI6IHsiZGlzYWJsZV9yZWxvYWQiOiBmYWxzZSwgImVuYWJsZWQiOiBmYWxzZSwgInJlbW92ZV9kb20iOiBmYWxzZSwgInRhYl9uYW1lIjogIkRlYnVnZ2VyIFYyIiwgImxvYWRpbmdfbWVjaGFuaXNtIjogeyJ0eXBlIjogIk5HX0NPTVBPTkVOVCJ9fSwgImdyYXBocyI6IHsiZGlzYWJsZV9yZWxvYWQiOiB0cnVlLCAiZW5hYmxlZCI6IHRydWUsICJyZW1vdmVfZG9tIjogZmFsc2UsICJ0YWJfbmFtZSI6ICJncmFwaHMiLCAibG9hZGluZ19tZWNoYW5pc20iOiB7InR5cGUiOiAiQ1VTVE9NX0VMRU1FTlQiLCAiZWxlbWVudF9uYW1lIjogInRmLWdyYXBoLWRhc2hib2FyZCJ9fSwgImRpc3RyaWJ1dGlvbnMiOiB7ImRpc2FibGVfcmVsb2FkIjogZmFsc2UsICJlbmFibGVkIjogZmFsc2UsICJyZW1vdmVfZG9tIjogZmFsc2UsICJ0YWJfbmFtZSI6ICJkaXN0cmlidXRpb25zIiwgImxvYWRpbmdfbWVjaGFuaXNtIjogeyJ0eXBlIjogIkNVU1RPTV9FTEVNRU5UIiwgImVsZW1lbnRfbmFtZSI6ICJ0Zi1kaXN0cmlidXRpb24tZGFzaGJvYXJkIn19LCAiaGlzdG9ncmFtcyI6IHsiZGlzYWJsZV9yZWxvYWQiOiBmYWxzZSwgImVuYWJsZWQiOiBmYWxzZSwgInJlbW92ZV9kb20iOiBmYWxzZSwgInRhYl9uYW1lIjogImhpc3RvZ3JhbXMiLCAibG9hZGluZ19tZWNoYW5pc20iOiB7InR5cGUiOiAiQ1VTVE9NX0VMRU1FTlQiLCAiZWxlbWVudF9uYW1lIjogInRmLWhpc3RvZ3JhbS1kYXNoYm9hcmQifX0sICJ0ZXh0IjogeyJkaXNhYmxlX3JlbG9hZCI6IGZhbHNlLCAiZW5hYmxlZCI6IGZhbHNlLCAicmVtb3ZlX2RvbSI6IGZhbHNlLCAidGFiX25hbWUiOiAidGV4dCIsICJsb2FkaW5nX21lY2hhbmlzbSI6IHsidHlwZSI6ICJDVVNUT01fRUxFTUVOVCIsICJlbGVtZW50X25hbWUiOiAidGYtdGV4dC1kYXNoYm9hcmQifX0sICJwcl9jdXJ2ZXMiOiB7ImRpc2FibGVfcmVsb2FkIjogZmFsc2UsICJlbmFibGVkIjogZmFsc2UsICJyZW1vdmVfZG9tIjogZmFsc2UsICJ0YWJfbmFtZSI6ICJQUiBDdXJ2ZXMiLCAibG9hZGluZ19tZWNoYW5pc20iOiB7InR5cGUiOiAiQ1VTVE9NX0VMRU1FTlQiLCAiZWxlbWVudF9uYW1lIjogInRmLXByLWN1cnZlLWRhc2hib2FyZCJ9fSwgInByb2ZpbGVfcmVkaXJlY3QiOiB7ImRpc2FibGVfcmVsb2FkIjogZmFsc2UsICJlbmFibGVkIjogZmFsc2UsICJyZW1vdmVfZG9tIjogZmFsc2UsICJ0YWJfbmFtZSI6ICJQcm9maWxlIiwgImxvYWRpbmdfbWVjaGFuaXNtIjogeyJ0eXBlIjogIkNVU1RPTV9FTEVNRU5UIiwgImVsZW1lbnRfbmFtZSI6ICJ0Zi1wcm9maWxlLXJlZGlyZWN0LWRhc2hib2FyZCJ9fSwgImhwYXJhbXMiOiB7ImRpc2FibGVfcmVsb2FkIjogZmFsc2UsICJlbmFibGVkIjogZmFsc2UsICJyZW1vdmVfZG9tIjogZmFsc2UsICJ0YWJfbmFtZSI6ICJocGFyYW1zIiwgImxvYWRpbmdfbWVjaGFuaXNtIjogeyJ0eXBlIjogIkNVU1RPTV9FTEVNRU5UIiwgImVsZW1lbnRfbmFtZSI6ICJ0Zi1ocGFyYW1zLWRhc2hib2FyZCJ9fSwgIm1lc2giOiB7ImRpc2FibGVfcmVsb2FkIjogZmFsc2UsICJlbmFibGVkIjogZmFsc2UsICJyZW1vdmVfZG9tIjogZmFsc2UsICJ0YWJfbmFtZSI6ICJtZXNoIiwgImxvYWRpbmdfbWVjaGFuaXNtIjogeyJ0eXBlIjogIkNVU1RPTV9FTEVNRU5UIiwgImVsZW1lbnRfbmFtZSI6ICJtZXNoLWRhc2hib2FyZCJ9fSwgInRpbWVzZXJpZXMiOiB7ImRpc2FibGVfcmVsb2FkIjogZmFsc2UsICJlbmFibGVkIjogZmFsc2UsICJyZW1vdmVfZG9tIjogZmFsc2UsICJ0YWJfbmFtZSI6ICJUaW1lIFNlcmllcyIsICJsb2FkaW5nX21lY2hhbmlzbSI6IHsidHlwZSI6ICJOR19DT01QT05FTlQifX0sICJwcm9qZWN0b3IiOiB7ImRpc2FibGVfcmVsb2FkIjogdHJ1ZSwgImVuYWJsZWQiOiBmYWxzZSwgInJlbW92ZV9kb20iOiBmYWxzZSwgInRhYl9uYW1lIjogInByb2plY3RvciIsICJsb2FkaW5nX21lY2hhbmlzbSI6IHsidHlwZSI6ICJJRlJBTUUiLCAibW9kdWxlX3BhdGgiOiAiL2RhdGEvcGx1Z2luL3Byb2plY3Rvci9pbmRleC5qcyJ9fSwgIndoYXRpZiI6IHsiZGlzYWJsZV9yZWxvYWQiOiBmYWxzZSwgImVuYWJsZWQiOiBmYWxzZSwgInJlbW92ZV9kb20iOiBmYWxzZSwgInRhYl9uYW1lIjogIldoYXQtSWYgVG9vbCIsICJsb2FkaW5nX21lY2hhbmlzbSI6IHsidHlwZSI6ICJJRlJBTUUiLCAibW9kdWxlX3BhdGgiOiAiL2RhdGEvcGx1Z2luL3doYXRpZi9pbmRleC5qcyJ9fX0=", + "ok": true, + "headers": [ + [ + "content-type", + "application/json" + ] + ], + "status": 200, + "status_text": "" + }, + "https://localhost:6006/icon_bundle.svg": { + "data": "<svg><defs>
<svg id="arrow_downward_24px" xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path fill="#010101" d="M20 12l-1.41-1.41L13 16.17V4h-2v12.17l-5.58-5.59L4 12l8 8 8-8z"/></svg><svg id="arrow_upward_24px" xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path d="M4 12l1.41 1.41L11 7.83V20h2V7.83l5.58 5.59L20 12l-8-8-8 8z"/></svg><svg id="bug_report_24px" xmlns="http://www.w3.org/2000/svg" height="24" viewBox="0 0 24 24" width="24"><path d="M0 0h24v24H0z" fill="none"/><path d="M20 8h-2.81c-.45-.78-1.07-1.45-1.82-1.96L17 4.41 15.59 3l-2.17 2.17C12.96 5.06 12.49 5 12 5c-.49 0-.96.06-1.41.17L8.41 3 7 4.41l1.62 1.63C7.88 6.55 7.26 7.22 6.81 8H4v2h2.09c-.05.33-.09.66-.09 1v1H4v2h2v1c0 .34.04.67.09 1H4v2h2.81c1.04 1.79 2.97 3 5.19 3s4.15-1.21 5.19-3H20v-2h-2.09c.05-.33.09-.66.09-1v-1h2v-2h-2v-1c0-.34-.04-.67-.09-1H20V8zm-6 8h-4v-2h4v2zm0-4h-4v-2h4v2z"/></svg><svg id="cancel_24px" xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path d="M12 2C6.47 2 2 6.47 2 12s4.47 10 10 10 10-4.47 10-10S17.53 2 12 2zm5 13.59L15.59 17 12 13.41 8.41 17 7 15.59 10.59 12 7 8.41 8.41 7 12 10.59 15.59 7 17 8.41 13.41 12 17 15.59z"/></svg><svg id="chevron_left_24px" xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path d="M15.41 7.41L14 6l-6 6 6 6 1.41-1.41L10.83 12z"/></svg><svg id="chevron_right_24px" xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path d="M10 6L8.59 7.41 13.17 12l-4.58 4.59L10 18l6-6z"/></svg><svg id="clear_24px" xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path d="M19 6.41L17.59 5 12 10.59 6.41 5 5 6.41 10.59 12 5 17.59 6.41 19 12 13.41 17.59 19 19 17.59 13.41 12z"/></svg><svg id="close_24px" xmlns="http://www.w3.org/2000/svg" height="24" viewBox="0 0 24 24" width="24"><path d="M0 0h24v24H0z" fill="none"/><path d="M19 6.41L17.59 5 12 10.59 6.41 5 5 6.41 10.59 12 5 17.59 6.41 19 12 13.41 17.59 19 19 17.59 13.41 12z"/></svg><svg id="content_copy_24px" xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path d="M16 1H4c-1.1 0-2 .9-2 2v14h2V3h12V1zm3 4H8c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h11c1.1 0 2-.9 2-2V7c0-1.1-.9-2-2-2zm0 16H8V7h11v14z"/></svg><svg id="error_24px" xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path d="M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm1 15h-2v-2h2v2zm0-4h-2V7h2v6z"/></svg><svg id="expand_less_24px" xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path d="M12 8l-6 6 1.41 1.41L12 10.83l4.59 4.58L18 14z"/></svg><svg id="expand_more_24px" xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path d="M16.59 8.59L12 13.17 7.41 8.59 6 10l6 6 6-6z"/></svg><svg id="filter_alt_24px" xmlns="http://www.w3.org/2000/svg" enable-background="new 0 0 24 24" height="24" viewBox="0 0 24 24" width="24"><g><path d="M0,0h24 M24,24H0" fill="none"/><path d="M4.25,5.61C6.27,8.2,10,13,10,13v6c0,0.55,0.45,1,1,1h2c0.55,0,1-0.45,1-1v-6c0,0,3.72-4.8,5.74-7.39 C20.25,4.95,19.78,4,18.95,4H5.04C4.21,4,3.74,4.95,4.25,5.61z"/><path d="M0,0h24v24H0V0z" fill="none"/></g></svg><svg id="flag_24px" xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path d="M14.4 6L14 4H5v17h2v-7h5.6l.4 2h7V6z"/></svg><svg id="fullscreen_24px" xmlns="http://www.w3.org/2000/svg" height="24" viewBox="0 0 24 24" width="24"><path d="M0 0h24v24H0z" fill="none"/><path d="M7 14H5v5h5v-2H7v-3zm-2-4h2V7h3V5H5v5zm12 7h-3v2h5v-5h-2v3zM14 5v2h3v3h2V5h-5z"/></svg><svg id="fullscreen_exit_24px" xmlns="http://www.w3.org/2000/svg" height="24" viewBox="0 0 24 24" width="24"><path d="M0 0h24v24H0z" fill="none"/><path d="M5 16h3v3h2v-5H5v2zm3-8H5v2h5V5H8v3zm6 11h2v-3h3v-2h-5v5zm2-11V5h-2v5h5V8h-3z"/></svg><svg id="get_app_24px" xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path d="M19 9h-4V3H9v6H5l7 7 7-7zM5 18v2h14v-2H5z"/></svg><svg id="help_outline_24px" xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path d="M11 18h2v-2h-2v2zm1-16C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm0 18c-4.41 0-8-3.59-8-8s3.59-8 8-8 8 3.59 8 8-3.59 8-8 8zm0-14c-2.21 0-4 1.79-4 4h2c0-1.1.9-2 2-2s2 .9 2 2c0 2-3 1.75-3 5h2c0-2.25 3-2.5 3-5 0-2.21-1.79-4-4-4z"/></svg><svg id="image_search_24px" xmlns="http://www.w3.org/2000/svg" height="24" viewBox="0 0 24 24" width="24"><path d="M0 0h24v24H0z" fill="none"/><path d="M0 0h24v24H0V0z" fill="none"/><path d="M18 13v7H4V6h5.02c.05-.71.22-1.38.48-2H4c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2v-5l-2-2zm-1.5 5h-11l2.75-3.53 1.96 2.36 2.75-3.54zm2.8-9.11c.44-.7.7-1.51.7-2.39C20 4.01 17.99 2 15.5 2S11 4.01 11 6.5s2.01 4.5 4.49 4.5c.88 0 1.7-.26 2.39-.7L21 13.42 22.42 12 19.3 8.89zM15.5 9C14.12 9 13 7.88 13 6.5S14.12 4 15.5 4 18 5.12 18 6.5 16.88 9 15.5 9z"/></svg><svg id="info_outline_24px" xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path d="M11 17h2v-6h-2v6zm1-15C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm0 18c-4.41 0-8-3.59-8-8s3.59-8 8-8 8 3.59 8 8-3.59 8-8 8zM11 9h2V7h-2v2z"/></svg><svg id="keep_24px" xmlns="http://www.w3.org/2000/svg" enable-background="new 0 0 24 24" height="24" viewBox="0 0 24 24" width="24"><g><rect fill="none" height="24" width="24"/></g><g><path d="M16,9V4l1,0c0.55,0,1-0.45,1-1v0c0-0.55-0.45-1-1-1H7C6.45,2,6,2.45,6,3v0 c0,0.55,0.45,1,1,1l1,0v5c0,1.66-1.34,3-3,3h0v2h5.97v7l1,1l1-1v-7H19v-2h0C17.34,12,16,10.66,16,9z" fill-rule="evenodd"/></g></svg><svg id="keep_outline_24px" xmlns="http://www.w3.org/2000/svg" enable-background="new 0 0 24 24" height="24" viewBox="0 0 24 24" width="24"><g><rect fill="none" height="24" width="24"/></g><g><path d="M14,4v5c0,1.12,0.37,2.16,1,3H9c0.65-0.86,1-1.9,1-3V4H14 M17,2H7C6.45,2,6,2.45,6,3c0,0.55,0.45,1,1,1c0,0,0,0,0,0l1,0v5 c0,1.66-1.34,3-3,3v2h5.97v7l1,1l1-1v-7H19v-2c0,0,0,0,0,0c-1.66,0-3-1.34-3-3V4l1,0c0,0,0,0,0,0c0.55,0,1-0.45,1-1 C18,2.45,17.55,2,17,2L17,2z"/></g></svg><svg id="line_weight_24px" xmlns="http://www.w3.org/2000/svg" enable-background="new 0 0 24 24" height="24" viewBox="0 0 24 24" width="24"><g><rect fill="none" height="24" width="24" x="0"/></g><g><g><g><path d="M3,17h18v-2H3V17z M3,20h18v-1H3V20z M3,13h18v-3H3V13z M3,4v4h18V4H3z"/></g></g></g></svg><svg id="more_vert_24px" xmlns="http://www.w3.org/2000/svg" height="24" viewBox="0 0 24 24" width="24"><path d="M0 0h24v24H0z" fill="none"/><path d="M12 8c1.1 0 2-.9 2-2s-.9-2-2-2-2 .9-2 2 .9 2 2 2zm0 2c-1.1 0-2 .9-2 2s.9 2 2 2 2-.9 2-2-.9-2-2-2zm0 6c-1.1 0-2 .9-2 2s.9 2 2 2 2-.9 2-2-.9-2-2-2z"/></svg><svg id="refresh_24px" xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path d="M17.65 6.35C16.2 4.9 14.21 4 12 4c-4.42 0-7.99 3.58-7.99 8s3.57 8 7.99 8c3.73 0 6.84-2.55 7.73-6h-2.08c-.82 2.33-3.04 4-5.65 4-3.31 0-6-2.69-6-6s2.69-6 6-6c1.66 0 3.14.69 4.22 1.78L13 11h7V4l-2.35 2.35z"/></svg><svg id="search_24px" xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path d="M15.5 14h-.79l-.28-.27C15.41 12.59 16 11.11 16 9.5 16 5.91 13.09 3 9.5 3S3 5.91 3 9.5 5.91 16 9.5 16c1.61 0 3.09-.59 4.23-1.57l.27.28v.79l5 4.99L20.49 19l-4.99-5zm-6 0C7.01 14 5 11.99 5 9.5S7.01 5 9.5 5 14 7.01 14 9.5 11.99 14 9.5 14z"/></svg><svg id="settings_24px" xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path d="M19.43 12.98c.04-.32.07-.64.07-.98s-.03-.66-.07-.98l2.11-1.65c.19-.15.24-.42.12-.64l-2-3.46c-.12-.22-.39-.3-.61-.22l-2.49 1c-.52-.4-1.08-.73-1.69-.98l-.38-2.65C14.46 2.18 14.25 2 14 2h-4c-.25 0-.46.18-.49.42l-.38 2.65c-.61.25-1.17.59-1.69.98l-2.49-1c-.23-.09-.49 0-.61.22l-2 3.46c-.13.22-.07.49.12.64l2.11 1.65c-.04.32-.07.65-.07.98s.03.66.07.98l-2.11 1.65c-.19.15-.24.42-.12.64l2 3.46c.12.22.39.3.61.22l2.49-1c.52.4 1.08.73 1.69.98l.38 2.65c.03.24.24.42.49.42h4c.25 0 .46-.18.49-.42l.38-2.65c.61-.25 1.17-.59 1.69-.98l2.49 1c.23.09.49 0 .61-.22l2-3.46c.12-.22.07-.49-.12-.64l-2.11-1.65zM12 15.5c-1.93 0-3.5-1.57-3.5-3.5s1.57-3.5 3.5-3.5 3.5 1.57 3.5 3.5-1.57 3.5-3.5 3.5z"/></svg><svg id="settings_backup_restore_24px" xmlns="http://www.w3.org/2000/svg" height="24" viewBox="0 0 24 24" width="24"><path d="M0 0h24v24H0z" fill="none"/><path d="M14 12c0-1.1-.9-2-2-2s-2 .9-2 2 .9 2 2 2 2-.9 2-2zm-2-9c-4.97 0-9 4.03-9 9H0l4 4 4-4H5c0-3.87 3.13-7 7-7s7 3.13 7 7-3.13 7-7 7c-1.51 0-2.91-.49-4.06-1.3l-1.42 1.44C8.04 20.3 9.94 21 12 21c4.97 0 9-4.03 9-9s-4.03-9-9-9z"/></svg><svg id="settings_overscan_24px" xmlns="http://www.w3.org/2000/svg" height="24" viewBox="0 0 24 24" width="24"><path d="M0 0h24v24H0z" fill="none"/><path d="M12.01 5.5L10 8h4l-1.99-2.5zM18 10v4l2.5-1.99L18 10zM6 10l-2.5 2.01L6 14v-4zm8 6h-4l2.01 2.5L14 16zm7-13H3c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h18c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm0 16.01H3V4.99h18v14.02z"/></svg><svg id="visibility_off_24px" xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path d="M12 7c2.76 0 5 2.24 5 5 0 .65-.13 1.26-.36 1.83l2.92 2.92c1.51-1.26 2.7-2.89 3.43-4.75-1.73-4.39-6-7.5-11-7.5-1.4 0-2.74.25-3.98.7l2.16 2.16C10.74 7.13 11.35 7 12 7zM2 4.27l2.28 2.28.46.46C3.08 8.3 1.78 10.02 1 12c1.73 4.39 6 7.5 11 7.5 1.55 0 3.03-.3 4.38-.84l.42.42L19.73 22 21 20.73 3.27 3 2 4.27zM7.53 9.8l1.55 1.55c-.05.21-.08.43-.08.65 0 1.66 1.34 3 3 3 .22 0 .44-.03.65-.08l1.55 1.55c-.67.33-1.41.53-2.2.53-2.76 0-5-2.24-5-5 0-.79.2-1.53.53-2.2zm4.31-.78l3.15 3.15.02-.16c0-1.66-1.34-3-3-3l-.17.01z"/></svg><svg id="warning_24px" xmlns="http://www.w3.org/2000/svg" height="24" viewBox="0 0 24 24" width="24"><path d="M0 0h24v24H0z" fill="none"/><path d="M1 21h22L12 2 1 21zm12-3h-2v-2h2v2zm0-4h-2v-4h2v4z"/></svg></defs></svg>
", + "ok": true, + "headers": [ + [ + "content-type", + "image/svg+xml; charset=utf-8" + ] + ], + "status": 200, + "status_text": "" + }, + "https://localhost:6006/data/runs": { + "data": "WyJydW5fMDAxL3RyYWluIl0=", + "ok": true, + "headers": [ + [ + "content-type", + "application/json" + ] + ], + "status": 200, + "status_text": "" + }, + "https://localhost:6006/data/plugin/graphs/info": { + "data": "eyJydW5fMDAxL3RyYWluIjogeyJydW4iOiAicnVuXzAwMS90cmFpbiIsICJ0YWdzIjogeyJiYXRjaF8yIjogeyJ0YWciOiAiYmF0Y2hfMiIsICJjb25jZXB0dWFsX2dyYXBoIjogZmFsc2UsICJvcF9ncmFwaCI6IHRydWUsICJwcm9maWxlIjogZmFsc2V9LCAia2VyYXMiOiB7InRhZyI6ICJrZXJhcyIsICJjb25jZXB0dWFsX2dyYXBoIjogdHJ1ZSwgIm9wX2dyYXBoIjogZmFsc2UsICJwcm9maWxlIjogZmFsc2V9fSwgInJ1bl9ncmFwaCI6IHRydWV9fQ==", + "ok": true, + "headers": [ + [ + "content-type", + "application/json" + ] + ], + "status": 200, + "status_text": "" + }, + "https://localhost:6006/data/plugin/graphs/graph?run=run_001%2Ftrain&conceptual=false": { + "data": "node {
  name: "iterator"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "_user_specified_name"
    value {
      s: "iterator"
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "iterator_1"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "_user_specified_name"
    value {
      s: "iterator"
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_VARIANT
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "IteratorGetNext"
  op: "IteratorGetNext"
  input: "iterator"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@iterator"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 32
          }
          dim {
            size: 32
          }
          dim {
            size: 3
          }
        }
        shape {
          dim {
            size: -1
          }
          dim {
            size: 1
          }
        }
      }
    }
  }
  attr {
    key: "output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 32
          }
          dim {
            size: 32
          }
          dim {
            size: 3
          }
        }
        shape {
          dim {
            size: -1
          }
          dim {
            size: 1
          }
        }
      }
    }
  }
  attr {
    key: "output_types"
    value {
      list {
        type: DT_UINT8
        type: DT_UINT8
      }
    }
  }
}
node {
  name: "sequential/Cast"
  op: "Cast"
  input: "IteratorGetNext"
  attr {
    key: "DstT"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "SrcT"
    value {
      type: DT_UINT8
    }
  }
  attr {
    key: "Truncate"
    value {
      b: false
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 32
          }
          dim {
            size: 32
          }
          dim {
            size: 3
          }
        }
      }
    }
  }
}
node {
  name: "sequential/flatten/Const"
  op: "Const"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 2
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_INT32
        tensor_shape {
          dim {
            size: 2
          }
        }
        tensor_content: "\377\377\377\377\000\014\000\000"
      }
    }
  }
}
node {
  name: "sequential/flatten/Reshape"
  op: "Reshape"
  input: "sequential/Cast"
  input: "sequential/flatten/Const"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "Tshape"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 3072
          }
        }
      }
    }
  }
}
node {
  name: "sequential/dense/MatMul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense/MatMul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 3072
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense/MatMul"
  op: "MatMul"
  input: "sequential/flatten/Reshape"
  input: "sequential/dense/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "sequential/dense/BiasAdd/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense/BiasAdd/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense/BiasAdd"
  op: "BiasAdd"
  input: "sequential/dense/MatMul"
  input: "sequential/dense/BiasAdd/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "sequential/dense/Elu"
  op: "Elu"
  input: "sequential/dense/BiasAdd"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "sequential/dense_1/MatMul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_1/MatMul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_1/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_1/MatMul"
  op: "MatMul"
  input: "sequential/dense/Elu"
  input: "sequential/dense_1/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "sequential/dense_1/BiasAdd/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_1/BiasAdd/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_1/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_1/BiasAdd"
  op: "BiasAdd"
  input: "sequential/dense_1/MatMul"
  input: "sequential/dense_1/BiasAdd/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "sequential/dense_1/Elu"
  op: "Elu"
  input: "sequential/dense_1/BiasAdd"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "sequential/dense_2/MatMul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_2/MatMul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_2/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_2/MatMul"
  op: "MatMul"
  input: "sequential/dense_1/Elu"
  input: "sequential/dense_2/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "sequential/dense_2/BiasAdd/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_2/BiasAdd/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_2/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_2/BiasAdd"
  op: "BiasAdd"
  input: "sequential/dense_2/MatMul"
  input: "sequential/dense_2/BiasAdd/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "sequential/dense_2/Elu"
  op: "Elu"
  input: "sequential/dense_2/BiasAdd"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "sequential/dense_3/MatMul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_3/MatMul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_3/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_3/MatMul"
  op: "MatMul"
  input: "sequential/dense_2/Elu"
  input: "sequential/dense_3/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "sequential/dense_3/BiasAdd/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_3/BiasAdd/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_3/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_3/BiasAdd"
  op: "BiasAdd"
  input: "sequential/dense_3/MatMul"
  input: "sequential/dense_3/BiasAdd/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "sequential/dense_3/Elu"
  op: "Elu"
  input: "sequential/dense_3/BiasAdd"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "sequential/dense_4/MatMul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_4/MatMul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_4/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_4/MatMul"
  op: "MatMul"
  input: "sequential/dense_3/Elu"
  input: "sequential/dense_4/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "sequential/dense_4/BiasAdd/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_4/BiasAdd/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_4/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_4/BiasAdd"
  op: "BiasAdd"
  input: "sequential/dense_4/MatMul"
  input: "sequential/dense_4/BiasAdd/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "sequential/dense_4/Elu"
  op: "Elu"
  input: "sequential/dense_4/BiasAdd"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "sequential/dense_5/MatMul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_5/MatMul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_5/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_5/MatMul"
  op: "MatMul"
  input: "sequential/dense_4/Elu"
  input: "sequential/dense_5/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "sequential/dense_5/BiasAdd/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_5/BiasAdd/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_5/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_5/BiasAdd"
  op: "BiasAdd"
  input: "sequential/dense_5/MatMul"
  input: "sequential/dense_5/BiasAdd/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "sequential/dense_5/Elu"
  op: "Elu"
  input: "sequential/dense_5/BiasAdd"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "sequential/dense_6/MatMul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_6/MatMul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_6/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_6/MatMul"
  op: "MatMul"
  input: "sequential/dense_5/Elu"
  input: "sequential/dense_6/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "sequential/dense_6/BiasAdd/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_6/BiasAdd/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_6/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_6/BiasAdd"
  op: "BiasAdd"
  input: "sequential/dense_6/MatMul"
  input: "sequential/dense_6/BiasAdd/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "sequential/dense_6/Elu"
  op: "Elu"
  input: "sequential/dense_6/BiasAdd"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "sequential/dense_7/MatMul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_7/MatMul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_7/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_7/MatMul"
  op: "MatMul"
  input: "sequential/dense_6/Elu"
  input: "sequential/dense_7/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "sequential/dense_7/BiasAdd/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_7/BiasAdd/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_7/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_7/BiasAdd"
  op: "BiasAdd"
  input: "sequential/dense_7/MatMul"
  input: "sequential/dense_7/BiasAdd/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "sequential/dense_7/Elu"
  op: "Elu"
  input: "sequential/dense_7/BiasAdd"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "sequential/dense_8/MatMul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_8/MatMul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_8/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_8/MatMul"
  op: "MatMul"
  input: "sequential/dense_7/Elu"
  input: "sequential/dense_8/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "sequential/dense_8/BiasAdd/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_8/BiasAdd/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_8/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_8/BiasAdd"
  op: "BiasAdd"
  input: "sequential/dense_8/MatMul"
  input: "sequential/dense_8/BiasAdd/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "sequential/dense_8/Elu"
  op: "Elu"
  input: "sequential/dense_8/BiasAdd"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "sequential/dense_9/MatMul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_9/MatMul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_9/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_9/MatMul"
  op: "MatMul"
  input: "sequential/dense_8/Elu"
  input: "sequential/dense_9/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "sequential/dense_9/BiasAdd/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_9/BiasAdd/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_9/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_9/BiasAdd"
  op: "BiasAdd"
  input: "sequential/dense_9/MatMul"
  input: "sequential/dense_9/BiasAdd/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "sequential/dense_9/Elu"
  op: "Elu"
  input: "sequential/dense_9/BiasAdd"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "sequential/dense_10/MatMul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_10/MatMul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_10/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_10/MatMul"
  op: "MatMul"
  input: "sequential/dense_9/Elu"
  input: "sequential/dense_10/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "sequential/dense_10/BiasAdd/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_10/BiasAdd/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_10/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_10/BiasAdd"
  op: "BiasAdd"
  input: "sequential/dense_10/MatMul"
  input: "sequential/dense_10/BiasAdd/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "sequential/dense_10/Elu"
  op: "Elu"
  input: "sequential/dense_10/BiasAdd"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "sequential/dense_11/MatMul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_11/MatMul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_11/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_11/MatMul"
  op: "MatMul"
  input: "sequential/dense_10/Elu"
  input: "sequential/dense_11/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "sequential/dense_11/BiasAdd/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_11/BiasAdd/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_11/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_11/BiasAdd"
  op: "BiasAdd"
  input: "sequential/dense_11/MatMul"
  input: "sequential/dense_11/BiasAdd/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "sequential/dense_11/Elu"
  op: "Elu"
  input: "sequential/dense_11/BiasAdd"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "sequential/dense_12/MatMul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_12/MatMul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_12/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_12/MatMul"
  op: "MatMul"
  input: "sequential/dense_11/Elu"
  input: "sequential/dense_12/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "sequential/dense_12/BiasAdd/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_12/BiasAdd/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_12/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_12/BiasAdd"
  op: "BiasAdd"
  input: "sequential/dense_12/MatMul"
  input: "sequential/dense_12/BiasAdd/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "sequential/dense_12/Elu"
  op: "Elu"
  input: "sequential/dense_12/BiasAdd"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "sequential/dense_13/MatMul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_13/MatMul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_13/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_13/MatMul"
  op: "MatMul"
  input: "sequential/dense_12/Elu"
  input: "sequential/dense_13/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "sequential/dense_13/BiasAdd/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_13/BiasAdd/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_13/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_13/BiasAdd"
  op: "BiasAdd"
  input: "sequential/dense_13/MatMul"
  input: "sequential/dense_13/BiasAdd/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "sequential/dense_13/Elu"
  op: "Elu"
  input: "sequential/dense_13/BiasAdd"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "sequential/dense_14/MatMul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_14/MatMul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_14/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_14/MatMul"
  op: "MatMul"
  input: "sequential/dense_13/Elu"
  input: "sequential/dense_14/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "sequential/dense_14/BiasAdd/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_14/BiasAdd/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_14/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_14/BiasAdd"
  op: "BiasAdd"
  input: "sequential/dense_14/MatMul"
  input: "sequential/dense_14/BiasAdd/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "sequential/dense_14/Elu"
  op: "Elu"
  input: "sequential/dense_14/BiasAdd"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "sequential/dense_15/MatMul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_15/MatMul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_15/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_15/MatMul"
  op: "MatMul"
  input: "sequential/dense_14/Elu"
  input: "sequential/dense_15/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "sequential/dense_15/BiasAdd/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_15/BiasAdd/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_15/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_15/BiasAdd"
  op: "BiasAdd"
  input: "sequential/dense_15/MatMul"
  input: "sequential/dense_15/BiasAdd/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "sequential/dense_15/Elu"
  op: "Elu"
  input: "sequential/dense_15/BiasAdd"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "sequential/dense_16/MatMul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_16/MatMul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_16/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_16/MatMul"
  op: "MatMul"
  input: "sequential/dense_15/Elu"
  input: "sequential/dense_16/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "sequential/dense_16/BiasAdd/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_16/BiasAdd/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_16/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_16/BiasAdd"
  op: "BiasAdd"
  input: "sequential/dense_16/MatMul"
  input: "sequential/dense_16/BiasAdd/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "sequential/dense_16/Elu"
  op: "Elu"
  input: "sequential/dense_16/BiasAdd"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "sequential/dense_17/MatMul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_17/MatMul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_17/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_17/MatMul"
  op: "MatMul"
  input: "sequential/dense_16/Elu"
  input: "sequential/dense_17/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "sequential/dense_17/BiasAdd/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_17/BiasAdd/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_17/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_17/BiasAdd"
  op: "BiasAdd"
  input: "sequential/dense_17/MatMul"
  input: "sequential/dense_17/BiasAdd/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "sequential/dense_17/Elu"
  op: "Elu"
  input: "sequential/dense_17/BiasAdd"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "sequential/dense_18/MatMul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_18/MatMul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_18/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_18/MatMul"
  op: "MatMul"
  input: "sequential/dense_17/Elu"
  input: "sequential/dense_18/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "sequential/dense_18/BiasAdd/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_18/BiasAdd/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_18/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_18/BiasAdd"
  op: "BiasAdd"
  input: "sequential/dense_18/MatMul"
  input: "sequential/dense_18/BiasAdd/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "sequential/dense_18/Elu"
  op: "Elu"
  input: "sequential/dense_18/BiasAdd"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "sequential/dense_19/MatMul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_19/MatMul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_19/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_19/MatMul"
  op: "MatMul"
  input: "sequential/dense_18/Elu"
  input: "sequential/dense_19/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "sequential/dense_19/BiasAdd/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_19/BiasAdd/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_19/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_19/BiasAdd"
  op: "BiasAdd"
  input: "sequential/dense_19/MatMul"
  input: "sequential/dense_19/BiasAdd/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "sequential/dense_19/Elu"
  op: "Elu"
  input: "sequential/dense_19/BiasAdd"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "sequential/dense_20/MatMul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_20/MatMul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_20/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 10
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_20/MatMul"
  op: "MatMul"
  input: "sequential/dense_19/Elu"
  input: "sequential/dense_20/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 10
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "sequential/dense_20/BiasAdd/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "sequential/dense_20/BiasAdd/ReadVariableOp"
  op: "ReadVariableOp"
  input: "sequential/dense_20/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 10
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "sequential/dense_20/BiasAdd"
  op: "BiasAdd"
  input: "sequential/dense_20/MatMul"
  input: "sequential/dense_20/BiasAdd/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 10
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "sequential/dense_20/Softmax"
  op: "Softmax"
  input: "sequential/dense_20/BiasAdd"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "sparse_categorical_crossentropy/Cast"
  op: "Cast"
  input: "IteratorGetNext:1"
  attr {
    key: "DstT"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "SrcT"
    value {
      type: DT_UINT8
    }
  }
  attr {
    key: "Truncate"
    value {
      b: false
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 1
          }
        }
      }
    }
  }
}
node {
  name: "sparse_categorical_crossentropy/Cast_1"
  op: "Cast"
  input: "sparse_categorical_crossentropy/Cast"
  attr {
    key: "DstT"
    value {
      type: DT_INT64
    }
  }
  attr {
    key: "SrcT"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "Truncate"
    value {
      b: false
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 1
          }
        }
      }
    }
  }
}
node {
  name: "sparse_categorical_crossentropy/Shape"
  op: "Shape"
  input: "sequential/dense_20/BiasAdd"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 2
          }
        }
      }
    }
  }
  attr {
    key: "out_type"
    value {
      type: DT_INT32
    }
  }
}
node {
  name: "sparse_categorical_crossentropy/Reshape/shape"
  op: "Const"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 1
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_INT32
        tensor_shape {
          dim {
            size: 1
          }
        }
        int_val: -1
      }
    }
  }
}
node {
  name: "sparse_categorical_crossentropy/Reshape"
  op: "Reshape"
  input: "sparse_categorical_crossentropy/Cast_1"
  input: "sparse_categorical_crossentropy/Reshape/shape"
  attr {
    key: "T"
    value {
      type: DT_INT64
    }
  }
  attr {
    key: "Tshape"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
        }
      }
    }
  }
}
node {
  name: "sparse_categorical_crossentropy/strided_slice/stack"
  op: "Const"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 1
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_INT32
        tensor_shape {
          dim {
            size: 1
          }
        }
        int_val: -1
      }
    }
  }
}
node {
  name: "sparse_categorical_crossentropy/strided_slice/stack_1"
  op: "Const"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 1
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_INT32
        tensor_shape {
          dim {
            size: 1
          }
        }
        int_val: 0
      }
    }
  }
}
node {
  name: "sparse_categorical_crossentropy/strided_slice/stack_2"
  op: "Const"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 1
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_INT32
        tensor_shape {
          dim {
            size: 1
          }
        }
        int_val: 1
      }
    }
  }
}
node {
  name: "sparse_categorical_crossentropy/strided_slice"
  op: "StridedSlice"
  input: "sparse_categorical_crossentropy/Shape"
  input: "sparse_categorical_crossentropy/strided_slice/stack"
  input: "sparse_categorical_crossentropy/strided_slice/stack_1"
  input: "sparse_categorical_crossentropy/strided_slice/stack_2"
  attr {
    key: "Index"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "T"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "begin_mask"
    value {
      i: 0
    }
  }
  attr {
    key: "ellipsis_mask"
    value {
      i: 0
    }
  }
  attr {
    key: "end_mask"
    value {
      i: 0
    }
  }
  attr {
    key: "new_axis_mask"
    value {
      i: 0
    }
  }
  attr {
    key: "shrink_axis_mask"
    value {
      i: 1
    }
  }
}
node {
  name: "sparse_categorical_crossentropy/Reshape_1/shape/0"
  op: "Const"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_INT32
        tensor_shape {
        }
        int_val: -1
      }
    }
  }
}
node {
  name: "sparse_categorical_crossentropy/Reshape_1/shape"
  op: "Pack"
  input: "sparse_categorical_crossentropy/Reshape_1/shape/0"
  input: "sparse_categorical_crossentropy/strided_slice"
  attr {
    key: "N"
    value {
      i: 2
    }
  }
  attr {
    key: "T"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 2
          }
        }
      }
    }
  }
  attr {
    key: "axis"
    value {
      i: 0
    }
  }
}
node {
  name: "sparse_categorical_crossentropy/Reshape_1"
  op: "Reshape"
  input: "sequential/dense_20/BiasAdd"
  input: "sparse_categorical_crossentropy/Reshape_1/shape"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "Tshape"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: -1
          }
        }
      }
    }
  }
}
node {
  name: "sparse_categorical_crossentropy/SparseSoftmaxCrossEntropyWithLogits/Shape"
  op: "Shape"
  input: "sparse_categorical_crossentropy/Reshape"
  attr {
    key: "T"
    value {
      type: DT_INT64
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 1
          }
        }
      }
    }
  }
  attr {
    key: "out_type"
    value {
      type: DT_INT32
    }
  }
}
node {
  name: "sparse_categorical_crossentropy/SparseSoftmaxCrossEntropyWithLogits/SparseSoftmaxCrossEntropyWithLogits"
  op: "SparseSoftmaxCrossEntropyWithLogits"
  input: "sparse_categorical_crossentropy/Reshape_1"
  input: "sparse_categorical_crossentropy/Reshape"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "Tlabels"
    value {
      type: DT_INT64
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
        }
        shape {
          dim {
            size: -1
          }
          dim {
            size: -1
          }
        }
      }
    }
  }
}
node {
  name: "sparse_categorical_crossentropy/weighted_loss/Const"
  op: "Const"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_FLOAT
        tensor_shape {
        }
        float_val: 1.0
      }
    }
  }
}
node {
  name: "sparse_categorical_crossentropy/weighted_loss/Mul"
  op: "Mul"
  input: "sparse_categorical_crossentropy/SparseSoftmaxCrossEntropyWithLogits/SparseSoftmaxCrossEntropyWithLogits"
  input: "sparse_categorical_crossentropy/weighted_loss/Const"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
        }
      }
    }
  }
}
node {
  name: "sparse_categorical_crossentropy/weighted_loss/Const_1"
  op: "Const"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 1
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_INT32
        tensor_shape {
          dim {
            size: 1
          }
        }
        int_val: 0
      }
    }
  }
}
node {
  name: "sparse_categorical_crossentropy/weighted_loss/Sum"
  op: "Sum"
  input: "sparse_categorical_crossentropy/weighted_loss/Mul"
  input: "sparse_categorical_crossentropy/weighted_loss/Const_1"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "Tidx"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "keep_dims"
    value {
      b: false
    }
  }
}
node {
  name: "sparse_categorical_crossentropy/weighted_loss/num_elements"
  op: "Size"
  input: "sparse_categorical_crossentropy/weighted_loss/Mul"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "out_type"
    value {
      type: DT_INT32
    }
  }
}
node {
  name: "sparse_categorical_crossentropy/weighted_loss/num_elements/Cast"
  op: "Cast"
  input: "sparse_categorical_crossentropy/weighted_loss/num_elements"
  attr {
    key: "DstT"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "SrcT"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "Truncate"
    value {
      b: false
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "sparse_categorical_crossentropy/weighted_loss/Rank"
  op: "Const"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_INT32
        tensor_shape {
        }
        int_val: 0
      }
    }
  }
}
node {
  name: "sparse_categorical_crossentropy/weighted_loss/range/start"
  op: "Const"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_INT32
        tensor_shape {
        }
        int_val: 0
      }
    }
  }
}
node {
  name: "sparse_categorical_crossentropy/weighted_loss/range/delta"
  op: "Const"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_INT32
        tensor_shape {
        }
        int_val: 1
      }
    }
  }
}
node {
  name: "sparse_categorical_crossentropy/weighted_loss/range"
  op: "Range"
  input: "sparse_categorical_crossentropy/weighted_loss/range/start"
  input: "sparse_categorical_crossentropy/weighted_loss/Rank"
  input: "sparse_categorical_crossentropy/weighted_loss/range/delta"
  attr {
    key: "Tidx"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
          }
        }
      }
    }
  }
}
node {
  name: "sparse_categorical_crossentropy/weighted_loss/Sum_1"
  op: "Sum"
  input: "sparse_categorical_crossentropy/weighted_loss/Sum"
  input: "sparse_categorical_crossentropy/weighted_loss/range"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "Tidx"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "keep_dims"
    value {
      b: false
    }
  }
}
node {
  name: "sparse_categorical_crossentropy/weighted_loss/value"
  op: "DivNoNan"
  input: "sparse_categorical_crossentropy/weighted_loss/Sum_1"
  input: "sparse_categorical_crossentropy/weighted_loss/num_elements/Cast"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "Shape"
  op: "Shape"
  input: "IteratorGetNext:1"
  attr {
    key: "T"
    value {
      type: DT_UINT8
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 2
          }
        }
      }
    }
  }
  attr {
    key: "out_type"
    value {
      type: DT_INT32
    }
  }
}
node {
  name: "strided_slice/stack"
  op: "Const"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 1
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_INT32
        tensor_shape {
          dim {
            size: 1
          }
        }
        int_val: 0
      }
    }
  }
}
node {
  name: "strided_slice/stack_1"
  op: "Const"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 1
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_INT32
        tensor_shape {
          dim {
            size: 1
          }
        }
        int_val: 1
      }
    }
  }
}
node {
  name: "strided_slice/stack_2"
  op: "Const"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 1
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_INT32
        tensor_shape {
          dim {
            size: 1
          }
        }
        int_val: 1
      }
    }
  }
}
node {
  name: "strided_slice"
  op: "StridedSlice"
  input: "Shape"
  input: "strided_slice/stack"
  input: "strided_slice/stack_1"
  input: "strided_slice/stack_2"
  attr {
    key: "Index"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "T"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "begin_mask"
    value {
      i: 0
    }
  }
  attr {
    key: "ellipsis_mask"
    value {
      i: 0
    }
  }
  attr {
    key: "end_mask"
    value {
      i: 0
    }
  }
  attr {
    key: "new_axis_mask"
    value {
      i: 0
    }
  }
  attr {
    key: "shrink_axis_mask"
    value {
      i: 1
    }
  }
}
node {
  name: "Cast"
  op: "Cast"
  input: "strided_slice"
  attr {
    key: "DstT"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "SrcT"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "Truncate"
    value {
      b: false
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "Mul"
  op: "Mul"
  input: "sparse_categorical_crossentropy/weighted_loss/value"
  input: "Cast"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "Rank"
  op: "Const"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_INT32
        tensor_shape {
        }
        int_val: 0
      }
    }
  }
}
node {
  name: "range/start"
  op: "Const"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_INT32
        tensor_shape {
        }
        int_val: 0
      }
    }
  }
}
node {
  name: "range/delta"
  op: "Const"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_INT32
        tensor_shape {
        }
        int_val: 1
      }
    }
  }
}
node {
  name: "range"
  op: "Range"
  input: "range/start"
  input: "Rank"
  input: "range/delta"
  attr {
    key: "Tidx"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
          }
        }
      }
    }
  }
}
node {
  name: "Sum"
  op: "Sum"
  input: "Mul"
  input: "range"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "Tidx"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "keep_dims"
    value {
      b: false
    }
  }
}
node {
  name: "AssignAddVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "AssignAddVariableOp"
  op: "AssignAddVariableOp"
  input: "AssignAddVariableOp/resource"
  input: "Sum"
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Rank_1"
  op: "Const"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_INT32
        tensor_shape {
        }
        int_val: 0
      }
    }
  }
}
node {
  name: "range_1/start"
  op: "Const"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_INT32
        tensor_shape {
        }
        int_val: 0
      }
    }
  }
}
node {
  name: "range_1/delta"
  op: "Const"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_INT32
        tensor_shape {
        }
        int_val: 1
      }
    }
  }
}
node {
  name: "range_1"
  op: "Range"
  input: "range_1/start"
  input: "Rank_1"
  input: "range_1/delta"
  attr {
    key: "Tidx"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
          }
        }
      }
    }
  }
}
node {
  name: "Sum_1"
  op: "Sum"
  input: "Cast"
  input: "range_1"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "Tidx"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "keep_dims"
    value {
      b: false
    }
  }
}
node {
  name: "AssignAddVariableOp_1/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "AssignAddVariableOp_1"
  op: "AssignAddVariableOp"
  input: "AssignAddVariableOp_1/resource"
  input: "Sum_1"
  input: "^AssignAddVariableOp"
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/gradients/ones"
  op: "Const"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_FLOAT
        tensor_shape {
        }
        float_val: 1.0
      }
    }
  }
}
node {
  name: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/value/Shape"
  op: "Const"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_INT32
        tensor_shape {
          dim {
          }
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/value/Shape_1"
  op: "Const"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_INT32
        tensor_shape {
          dim {
          }
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/value/BroadcastGradientArgs"
  op: "BroadcastGradientArgs"
  input: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/value/Shape"
  input: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/value/Shape_1"
  attr {
    key: "T"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
        }
        shape {
          dim {
            size: -1
          }
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/value/div_no_nan"
  op: "DivNoNan"
  input: "Nadam/gradients/ones"
  input: "sparse_categorical_crossentropy/weighted_loss/num_elements/Cast"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/value/Sum"
  op: "Sum"
  input: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/value/div_no_nan"
  input: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/value/BroadcastGradientArgs"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "Tidx"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "keep_dims"
    value {
      b: false
    }
  }
}
node {
  name: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/value/Reshape"
  op: "Reshape"
  input: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/value/Sum"
  input: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/value/Shape"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "Tshape"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/value/Neg"
  op: "Neg"
  input: "sparse_categorical_crossentropy/weighted_loss/Sum_1"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/value/div_no_nan_1"
  op: "DivNoNan"
  input: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/value/Neg"
  input: "sparse_categorical_crossentropy/weighted_loss/num_elements/Cast"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/value/div_no_nan_2"
  op: "DivNoNan"
  input: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/value/div_no_nan_1"
  input: "sparse_categorical_crossentropy/weighted_loss/num_elements/Cast"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/value/mul"
  op: "Mul"
  input: "Nadam/gradients/ones"
  input: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/value/div_no_nan_2"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/value/Sum_1"
  op: "Sum"
  input: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/value/mul"
  input: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/value/BroadcastGradientArgs:1"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "Tidx"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "keep_dims"
    value {
      b: false
    }
  }
}
node {
  name: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/value/Reshape_1"
  op: "Reshape"
  input: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/value/Sum_1"
  input: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/value/Shape_1"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "Tshape"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/Reshape/shape"
  op: "Const"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_FLOAT
        tensor_shape {
          dim {
          }
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/Reshape/shape_1"
  op: "Const"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_INT32
        tensor_shape {
          dim {
          }
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/Reshape"
  op: "Reshape"
  input: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/value/Reshape"
  input: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/Reshape/shape_1"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "Tshape"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/Const"
  op: "Const"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_INT32
        tensor_shape {
          dim {
          }
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/Tile"
  op: "Tile"
  input: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/Reshape"
  input: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/Const"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "Tmultiples"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/Reshape_1/shape"
  op: "Const"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 1
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_INT32
        tensor_shape {
          dim {
            size: 1
          }
        }
        int_val: 1
      }
    }
  }
}
node {
  name: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/Reshape_1"
  op: "Reshape"
  input: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/Tile"
  input: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/Reshape_1/shape"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "Tshape"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 1
          }
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/Shape"
  op: "Shape"
  input: "sparse_categorical_crossentropy/weighted_loss/Mul"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 1
          }
        }
      }
    }
  }
  attr {
    key: "out_type"
    value {
      type: DT_INT32
    }
  }
}
node {
  name: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/Tile_1"
  op: "Tile"
  input: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/Reshape_1"
  input: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/Shape"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "Tmultiples"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/Mul"
  op: "Mul"
  input: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/Tile_1"
  input: "sparse_categorical_crossentropy/weighted_loss/Const"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sparse_categorical_crossentropy/SparseSoftmaxCrossEntropyWithLogits/ExpandDims/dim"
  op: "Const"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_INT32
        tensor_shape {
        }
        int_val: -1
      }
    }
  }
}
node {
  name: "gradient_tape/sparse_categorical_crossentropy/SparseSoftmaxCrossEntropyWithLogits/ExpandDims"
  op: "ExpandDims"
  input: "gradient_tape/sparse_categorical_crossentropy/weighted_loss/Mul"
  input: "gradient_tape/sparse_categorical_crossentropy/SparseSoftmaxCrossEntropyWithLogits/ExpandDims/dim"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "Tdim"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 1
          }
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sparse_categorical_crossentropy/SparseSoftmaxCrossEntropyWithLogits/mul"
  op: "Mul"
  input: "gradient_tape/sparse_categorical_crossentropy/SparseSoftmaxCrossEntropyWithLogits/ExpandDims"
  input: "sparse_categorical_crossentropy/SparseSoftmaxCrossEntropyWithLogits/SparseSoftmaxCrossEntropyWithLogits:1"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: -1
          }
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sparse_categorical_crossentropy/Shape"
  op: "Shape"
  input: "sequential/dense_20/BiasAdd"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 2
          }
        }
      }
    }
  }
  attr {
    key: "out_type"
    value {
      type: DT_INT32
    }
  }
}
node {
  name: "gradient_tape/sparse_categorical_crossentropy/Reshape"
  op: "Reshape"
  input: "gradient_tape/sparse_categorical_crossentropy/SparseSoftmaxCrossEntropyWithLogits/mul"
  input: "gradient_tape/sparse_categorical_crossentropy/Shape"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "Tshape"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_20/BiasAdd/BiasAddGrad"
  op: "BiasAddGrad"
  input: "gradient_tape/sparse_categorical_crossentropy/Reshape"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 10
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_20/MatMul"
  op: "MatMul"
  input: "gradient_tape/sparse_categorical_crossentropy/Reshape"
  input: "sequential/dense_20/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: true
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_20/MatMul_1"
  op: "MatMul"
  input: "sequential/dense_19/Elu"
  input: "gradient_tape/sparse_categorical_crossentropy/Reshape"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 10
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: true
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_19/EluGrad"
  op: "EluGrad"
  input: "gradient_tape/sequential/dense_20/MatMul"
  input: "sequential/dense_19/Elu"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_19/BiasAdd/BiasAddGrad"
  op: "BiasAddGrad"
  input: "gradient_tape/sequential/dense_19/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_19/MatMul"
  op: "MatMul"
  input: "gradient_tape/sequential/dense_19/EluGrad"
  input: "sequential/dense_19/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: true
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_19/MatMul_1"
  op: "MatMul"
  input: "sequential/dense_18/Elu"
  input: "gradient_tape/sequential/dense_19/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: true
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_18/EluGrad"
  op: "EluGrad"
  input: "gradient_tape/sequential/dense_19/MatMul"
  input: "sequential/dense_18/Elu"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_18/BiasAdd/BiasAddGrad"
  op: "BiasAddGrad"
  input: "gradient_tape/sequential/dense_18/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_18/MatMul"
  op: "MatMul"
  input: "gradient_tape/sequential/dense_18/EluGrad"
  input: "sequential/dense_18/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: true
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_18/MatMul_1"
  op: "MatMul"
  input: "sequential/dense_17/Elu"
  input: "gradient_tape/sequential/dense_18/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: true
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_17/EluGrad"
  op: "EluGrad"
  input: "gradient_tape/sequential/dense_18/MatMul"
  input: "sequential/dense_17/Elu"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_17/BiasAdd/BiasAddGrad"
  op: "BiasAddGrad"
  input: "gradient_tape/sequential/dense_17/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_17/MatMul"
  op: "MatMul"
  input: "gradient_tape/sequential/dense_17/EluGrad"
  input: "sequential/dense_17/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: true
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_17/MatMul_1"
  op: "MatMul"
  input: "sequential/dense_16/Elu"
  input: "gradient_tape/sequential/dense_17/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: true
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_16/EluGrad"
  op: "EluGrad"
  input: "gradient_tape/sequential/dense_17/MatMul"
  input: "sequential/dense_16/Elu"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_16/BiasAdd/BiasAddGrad"
  op: "BiasAddGrad"
  input: "gradient_tape/sequential/dense_16/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_16/MatMul"
  op: "MatMul"
  input: "gradient_tape/sequential/dense_16/EluGrad"
  input: "sequential/dense_16/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: true
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_16/MatMul_1"
  op: "MatMul"
  input: "sequential/dense_15/Elu"
  input: "gradient_tape/sequential/dense_16/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: true
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_15/EluGrad"
  op: "EluGrad"
  input: "gradient_tape/sequential/dense_16/MatMul"
  input: "sequential/dense_15/Elu"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_15/BiasAdd/BiasAddGrad"
  op: "BiasAddGrad"
  input: "gradient_tape/sequential/dense_15/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_15/MatMul"
  op: "MatMul"
  input: "gradient_tape/sequential/dense_15/EluGrad"
  input: "sequential/dense_15/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: true
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_15/MatMul_1"
  op: "MatMul"
  input: "sequential/dense_14/Elu"
  input: "gradient_tape/sequential/dense_15/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: true
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_14/EluGrad"
  op: "EluGrad"
  input: "gradient_tape/sequential/dense_15/MatMul"
  input: "sequential/dense_14/Elu"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_14/BiasAdd/BiasAddGrad"
  op: "BiasAddGrad"
  input: "gradient_tape/sequential/dense_14/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_14/MatMul"
  op: "MatMul"
  input: "gradient_tape/sequential/dense_14/EluGrad"
  input: "sequential/dense_14/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: true
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_14/MatMul_1"
  op: "MatMul"
  input: "sequential/dense_13/Elu"
  input: "gradient_tape/sequential/dense_14/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: true
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_13/EluGrad"
  op: "EluGrad"
  input: "gradient_tape/sequential/dense_14/MatMul"
  input: "sequential/dense_13/Elu"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_13/BiasAdd/BiasAddGrad"
  op: "BiasAddGrad"
  input: "gradient_tape/sequential/dense_13/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_13/MatMul"
  op: "MatMul"
  input: "gradient_tape/sequential/dense_13/EluGrad"
  input: "sequential/dense_13/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: true
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_13/MatMul_1"
  op: "MatMul"
  input: "sequential/dense_12/Elu"
  input: "gradient_tape/sequential/dense_13/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: true
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_12/EluGrad"
  op: "EluGrad"
  input: "gradient_tape/sequential/dense_13/MatMul"
  input: "sequential/dense_12/Elu"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_12/BiasAdd/BiasAddGrad"
  op: "BiasAddGrad"
  input: "gradient_tape/sequential/dense_12/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_12/MatMul"
  op: "MatMul"
  input: "gradient_tape/sequential/dense_12/EluGrad"
  input: "sequential/dense_12/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: true
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_12/MatMul_1"
  op: "MatMul"
  input: "sequential/dense_11/Elu"
  input: "gradient_tape/sequential/dense_12/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: true
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_11/EluGrad"
  op: "EluGrad"
  input: "gradient_tape/sequential/dense_12/MatMul"
  input: "sequential/dense_11/Elu"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_11/BiasAdd/BiasAddGrad"
  op: "BiasAddGrad"
  input: "gradient_tape/sequential/dense_11/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_11/MatMul"
  op: "MatMul"
  input: "gradient_tape/sequential/dense_11/EluGrad"
  input: "sequential/dense_11/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: true
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_11/MatMul_1"
  op: "MatMul"
  input: "sequential/dense_10/Elu"
  input: "gradient_tape/sequential/dense_11/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: true
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_10/EluGrad"
  op: "EluGrad"
  input: "gradient_tape/sequential/dense_11/MatMul"
  input: "sequential/dense_10/Elu"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_10/BiasAdd/BiasAddGrad"
  op: "BiasAddGrad"
  input: "gradient_tape/sequential/dense_10/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_10/MatMul"
  op: "MatMul"
  input: "gradient_tape/sequential/dense_10/EluGrad"
  input: "sequential/dense_10/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: true
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_10/MatMul_1"
  op: "MatMul"
  input: "sequential/dense_9/Elu"
  input: "gradient_tape/sequential/dense_10/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: true
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_9/EluGrad"
  op: "EluGrad"
  input: "gradient_tape/sequential/dense_10/MatMul"
  input: "sequential/dense_9/Elu"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_9/BiasAdd/BiasAddGrad"
  op: "BiasAddGrad"
  input: "gradient_tape/sequential/dense_9/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_9/MatMul"
  op: "MatMul"
  input: "gradient_tape/sequential/dense_9/EluGrad"
  input: "sequential/dense_9/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: true
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_9/MatMul_1"
  op: "MatMul"
  input: "sequential/dense_8/Elu"
  input: "gradient_tape/sequential/dense_9/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: true
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_8/EluGrad"
  op: "EluGrad"
  input: "gradient_tape/sequential/dense_9/MatMul"
  input: "sequential/dense_8/Elu"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_8/BiasAdd/BiasAddGrad"
  op: "BiasAddGrad"
  input: "gradient_tape/sequential/dense_8/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_8/MatMul"
  op: "MatMul"
  input: "gradient_tape/sequential/dense_8/EluGrad"
  input: "sequential/dense_8/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: true
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_8/MatMul_1"
  op: "MatMul"
  input: "sequential/dense_7/Elu"
  input: "gradient_tape/sequential/dense_8/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: true
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_7/EluGrad"
  op: "EluGrad"
  input: "gradient_tape/sequential/dense_8/MatMul"
  input: "sequential/dense_7/Elu"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_7/BiasAdd/BiasAddGrad"
  op: "BiasAddGrad"
  input: "gradient_tape/sequential/dense_7/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_7/MatMul"
  op: "MatMul"
  input: "gradient_tape/sequential/dense_7/EluGrad"
  input: "sequential/dense_7/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: true
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_7/MatMul_1"
  op: "MatMul"
  input: "sequential/dense_6/Elu"
  input: "gradient_tape/sequential/dense_7/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: true
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_6/EluGrad"
  op: "EluGrad"
  input: "gradient_tape/sequential/dense_7/MatMul"
  input: "sequential/dense_6/Elu"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_6/BiasAdd/BiasAddGrad"
  op: "BiasAddGrad"
  input: "gradient_tape/sequential/dense_6/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_6/MatMul"
  op: "MatMul"
  input: "gradient_tape/sequential/dense_6/EluGrad"
  input: "sequential/dense_6/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: true
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_6/MatMul_1"
  op: "MatMul"
  input: "sequential/dense_5/Elu"
  input: "gradient_tape/sequential/dense_6/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: true
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_5/EluGrad"
  op: "EluGrad"
  input: "gradient_tape/sequential/dense_6/MatMul"
  input: "sequential/dense_5/Elu"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_5/BiasAdd/BiasAddGrad"
  op: "BiasAddGrad"
  input: "gradient_tape/sequential/dense_5/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_5/MatMul"
  op: "MatMul"
  input: "gradient_tape/sequential/dense_5/EluGrad"
  input: "sequential/dense_5/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: true
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_5/MatMul_1"
  op: "MatMul"
  input: "sequential/dense_4/Elu"
  input: "gradient_tape/sequential/dense_5/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: true
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_4/EluGrad"
  op: "EluGrad"
  input: "gradient_tape/sequential/dense_5/MatMul"
  input: "sequential/dense_4/Elu"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_4/BiasAdd/BiasAddGrad"
  op: "BiasAddGrad"
  input: "gradient_tape/sequential/dense_4/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_4/MatMul"
  op: "MatMul"
  input: "gradient_tape/sequential/dense_4/EluGrad"
  input: "sequential/dense_4/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: true
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_4/MatMul_1"
  op: "MatMul"
  input: "sequential/dense_3/Elu"
  input: "gradient_tape/sequential/dense_4/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: true
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_3/EluGrad"
  op: "EluGrad"
  input: "gradient_tape/sequential/dense_4/MatMul"
  input: "sequential/dense_3/Elu"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_3/BiasAdd/BiasAddGrad"
  op: "BiasAddGrad"
  input: "gradient_tape/sequential/dense_3/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_3/MatMul"
  op: "MatMul"
  input: "gradient_tape/sequential/dense_3/EluGrad"
  input: "sequential/dense_3/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: true
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_3/MatMul_1"
  op: "MatMul"
  input: "sequential/dense_2/Elu"
  input: "gradient_tape/sequential/dense_3/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: true
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_2/EluGrad"
  op: "EluGrad"
  input: "gradient_tape/sequential/dense_3/MatMul"
  input: "sequential/dense_2/Elu"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_2/BiasAdd/BiasAddGrad"
  op: "BiasAddGrad"
  input: "gradient_tape/sequential/dense_2/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_2/MatMul"
  op: "MatMul"
  input: "gradient_tape/sequential/dense_2/EluGrad"
  input: "sequential/dense_2/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: true
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_2/MatMul_1"
  op: "MatMul"
  input: "sequential/dense_1/Elu"
  input: "gradient_tape/sequential/dense_2/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: true
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_1/EluGrad"
  op: "EluGrad"
  input: "gradient_tape/sequential/dense_2/MatMul"
  input: "sequential/dense_1/Elu"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_1/BiasAdd/BiasAddGrad"
  op: "BiasAddGrad"
  input: "gradient_tape/sequential/dense_1/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_1/MatMul"
  op: "MatMul"
  input: "gradient_tape/sequential/dense_1/EluGrad"
  input: "sequential/dense_1/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: false
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: true
    }
  }
}
node {
  name: "gradient_tape/sequential/dense_1/MatMul_1"
  op: "MatMul"
  input: "sequential/dense/Elu"
  input: "gradient_tape/sequential/dense_1/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: true
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "gradient_tape/sequential/dense/EluGrad"
  op: "EluGrad"
  input: "gradient_tape/sequential/dense_1/MatMul"
  input: "sequential/dense/Elu"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "gradient_tape/sequential/dense/BiasAdd/BiasAddGrad"
  op: "BiasAddGrad"
  input: "gradient_tape/sequential/dense/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "data_format"
    value {
      s: "NHWC"
    }
  }
}
node {
  name: "gradient_tape/sequential/dense/MatMul"
  op: "MatMul"
  input: "sequential/flatten/Reshape"
  input: "gradient_tape/sequential/dense/EluGrad"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 3072
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "transpose_a"
    value {
      b: true
    }
  }
  attr {
    key: "transpose_b"
    value {
      b: false
    }
  }
}
node {
  name: "Nadam/Identity/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Identity/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Identity/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Identity"
  op: "Identity"
  input: "Nadam/Identity/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "Nadam/Cast/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Cast/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Cast/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Identity_1"
  op: "Identity"
  input: "Nadam/Cast/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "Nadam/Cast_1/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Cast_1/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Cast_1/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Identity_2"
  op: "Identity"
  input: "Nadam/Cast_1/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "Nadam/Cast_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Cast_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Cast_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Identity_3"
  op: "Identity"
  input: "Nadam/Cast_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "Nadam/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_INT64
    }
  }
}
node {
  name: "Nadam/add/y"
  op: "Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_INT64
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_INT64
        tensor_shape {
        }
        int64_val: 1
      }
    }
  }
}
node {
  name: "Nadam/add"
  op: "AddV2"
  input: "Nadam/ReadVariableOp"
  input: "Nadam/add/y"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_INT64
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "Nadam/Cast_3"
  op: "Cast"
  input: "Nadam/add"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "DstT"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "SrcT"
    value {
      type: DT_INT64
    }
  }
  attr {
    key: "Truncate"
    value {
      b: false
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "Nadam/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_INT64
    }
  }
}
node {
  name: "Nadam/add_1/y"
  op: "Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_INT64
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_INT64
        tensor_shape {
        }
        int64_val: 2
      }
    }
  }
}
node {
  name: "Nadam/add_1"
  op: "AddV2"
  input: "Nadam/ReadVariableOp_1"
  input: "Nadam/add_1/y"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_INT64
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "Nadam/Cast_4"
  op: "Cast"
  input: "Nadam/add_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "DstT"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "SrcT"
    value {
      type: DT_INT64
    }
  }
  attr {
    key: "Truncate"
    value {
      b: false
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "Nadam/Cast_5/x"
  op: "Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_FLOAT
        tensor_shape {
        }
        float_val: 0.9599999785423279
      }
    }
  }
}
node {
  name: "Nadam/mul/x"
  op: "Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_FLOAT
        tensor_shape {
        }
        float_val: 0.004000000189989805
      }
    }
  }
}
node {
  name: "Nadam/mul"
  op: "Mul"
  input: "Nadam/mul/x"
  input: "Nadam/Cast_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "Nadam/Pow"
  op: "Pow"
  input: "Nadam/Cast_5/x"
  input: "Nadam/mul"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "Nadam/mul_1/x"
  op: "Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_FLOAT
        tensor_shape {
        }
        float_val: 0.5
      }
    }
  }
}
node {
  name: "Nadam/mul_1"
  op: "Mul"
  input: "Nadam/mul_1/x"
  input: "Nadam/Pow"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "Nadam/sub/x"
  op: "Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_FLOAT
        tensor_shape {
        }
        float_val: 1.0
      }
    }
  }
}
node {
  name: "Nadam/sub"
  op: "Sub"
  input: "Nadam/sub/x"
  input: "Nadam/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "Nadam/mul_2"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/sub"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "Nadam/mul_3/x"
  op: "Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_FLOAT
        tensor_shape {
        }
        float_val: 0.004000000189989805
      }
    }
  }
}
node {
  name: "Nadam/mul_3"
  op: "Mul"
  input: "Nadam/mul_3/x"
  input: "Nadam/Cast_4"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "Nadam/Pow_1"
  op: "Pow"
  input: "Nadam/Cast_5/x"
  input: "Nadam/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "Nadam/mul_4/x"
  op: "Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_FLOAT
        tensor_shape {
        }
        float_val: 0.5
      }
    }
  }
}
node {
  name: "Nadam/mul_4"
  op: "Mul"
  input: "Nadam/mul_4/x"
  input: "Nadam/Pow_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "Nadam/sub_1/x"
  op: "Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_FLOAT
        tensor_shape {
        }
        float_val: 1.0
      }
    }
  }
}
node {
  name: "Nadam/sub_1"
  op: "Sub"
  input: "Nadam/sub_1/x"
  input: "Nadam/mul_4"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "Nadam/mul_5"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/sub_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "Nadam/mul_6"
  op: "Mul"
  input: "Nadam/Identity"
  input: "Nadam/mul_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "Nadam/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Identity/ReadVariableOp/resource"
  input: "Nadam/mul_6"
  input: "^Nadam/Identity/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Identity_4/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Identity/ReadVariableOp/resource"
  input: "^Nadam/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Identity_4"
  op: "Identity"
  input: "Nadam/Identity_4/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "Nadam/mul_7"
  op: "Mul"
  input: "Nadam/Identity_4"
  input: "Nadam/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "Nadam/Neg"
  op: "Neg"
  input: "Nadam/Identity_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "Nadam/Const"
  op: "Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_FLOAT
        tensor_shape {
        }
        float_val: 1.0000000116860974e-07
      }
    }
  }
}
node {
  name: "Nadam/sub_2/x"
  op: "Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_FLOAT
        tensor_shape {
        }
        float_val: 1.0
      }
    }
  }
}
node {
  name: "Nadam/sub_2"
  op: "Sub"
  input: "Nadam/sub_2/x"
  input: "Nadam/Identity_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "Nadam/sub_3/x"
  op: "Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_FLOAT
        tensor_shape {
        }
        float_val: 1.0
      }
    }
  }
}
node {
  name: "Nadam/sub_3"
  op: "Sub"
  input: "Nadam/sub_3/x"
  input: "Nadam/Identity_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "Nadam/sub_4/x"
  op: "Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_FLOAT
        tensor_shape {
        }
        float_val: 1.0
      }
    }
  }
}
node {
  name: "Nadam/sub_4"
  op: "Sub"
  input: "Nadam/sub_4/x"
  input: "Nadam/mul_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "Nadam/sub_5/x"
  op: "Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_FLOAT
        tensor_shape {
        }
        float_val: 1.0
      }
    }
  }
}
node {
  name: "Nadam/sub_5"
  op: "Sub"
  input: "Nadam/sub_5/x"
  input: "Nadam/Identity_4"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "Nadam/sub_6/x"
  op: "Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_FLOAT
        tensor_shape {
        }
        float_val: 1.0
      }
    }
  }
}
node {
  name: "Nadam/sub_6"
  op: "Sub"
  input: "Nadam/sub_6/x"
  input: "Nadam/mul_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "Nadam/Pow_2"
  op: "Pow"
  input: "Nadam/Identity_3"
  input: "Nadam/Cast_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "Nadam/sub_7/x"
  op: "Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_FLOAT
        tensor_shape {
        }
        float_val: 1.0
      }
    }
  }
}
node {
  name: "Nadam/sub_7"
  op: "Sub"
  input: "Nadam/sub_7/x"
  input: "Nadam/Pow_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense/MatMul"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 3072
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 3072
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 3072
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense/MatMul"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 3072
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update/add"
  op: "AddV2"
  input: "Nadam/Nadam/update/mul"
  input: "Nadam/Nadam/update/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 3072
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update/add"
  input: "^Nadam/Nadam/update/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 3072
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 3072
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 3072
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 3072
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense/MatMul"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 3072
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 3072
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update/mul_2"
  input: "Nadam/Nadam/update/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 3072
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update/add_1"
  input: "^Nadam/Nadam/update/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 3072
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 3072
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 3072
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 3072
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update/mul_4"
  input: "Nadam/Nadam/update/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 3072
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 3072
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 3072
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 3072
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update/mul_6"
  input: "Nadam/Nadam/update/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 3072
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 3072
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update/sub"
  op: "Sub"
  input: "Nadam/Nadam/update/ReadVariableOp_2"
  input: "Nadam/Nadam/update/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 3072
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense/MatMul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update/sub"
  input: "^Nadam/Nadam/update/ReadVariableOp_2"
  input: "^sequential/dense/MatMul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_1/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense/BiasAdd/BiasAddGrad"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_1/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_1/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_1/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_1/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_1/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_1/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_1/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_1/mul"
  input: "Nadam/Nadam/update_1/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_1/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_1/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_1/add"
  input: "^Nadam/Nadam/update_1/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_1/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_1/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_1/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_1/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_1/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_1/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_1/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_1/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_1/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_1/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_1/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_1/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_1/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_1/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_1/mul_2"
  input: "Nadam/Nadam/update_1/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_1/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_1/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_1/add_1"
  input: "^Nadam/Nadam/update_1/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_1/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_1/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_1/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_1/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_1/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_1/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_1/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_1/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_1/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_1/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_1/mul_4"
  input: "Nadam/Nadam/update_1/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_1/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_1/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_1/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_1/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_1/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_1/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_1/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_1/mul_6"
  input: "Nadam/Nadam/update_1/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_1/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_1/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_1/ReadVariableOp_2"
  input: "Nadam/Nadam/update_1/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_1/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense/BiasAdd/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_1/sub"
  input: "^Nadam/Nadam/update_1/ReadVariableOp_2"
  input: "^sequential/dense/BiasAdd/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_2/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_1/MatMul_1"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_2/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_2/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_2/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_2/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_2/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_2/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_1/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_2/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_2/mul"
  input: "Nadam/Nadam/update_2/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_2/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_2/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_2/add"
  input: "^Nadam/Nadam/update_2/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_2/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_2/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_2/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_2/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_2/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_2/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_2/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_2/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_2/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_2/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_1/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_2/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_2/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_2/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_2/mul_2"
  input: "Nadam/Nadam/update_2/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_2/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_2/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_2/add_1"
  input: "^Nadam/Nadam/update_2/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_2/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_2/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_2/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_2/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_2/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_2/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_2/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_2/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_2/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_2/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_2/mul_4"
  input: "Nadam/Nadam/update_2/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_2/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_2/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_2/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_2/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_2/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_2/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_2/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_2/mul_6"
  input: "Nadam/Nadam/update_2/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_2/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_1/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_2/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_2/ReadVariableOp_2"
  input: "Nadam/Nadam/update_2/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_2/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_1/MatMul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_2/sub"
  input: "^Nadam/Nadam/update_2/ReadVariableOp_2"
  input: "^sequential/dense_1/MatMul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_3/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_1/BiasAdd/BiasAddGrad"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_3/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_3/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_3/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_3/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_3/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_3/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_1/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_3/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_3/mul"
  input: "Nadam/Nadam/update_3/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_3/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_3/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_3/add"
  input: "^Nadam/Nadam/update_3/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_3/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_3/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_3/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_3/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_3/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_3/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_3/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_3/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_3/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_3/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_3/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_1/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_3/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_3/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_3/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_3/mul_2"
  input: "Nadam/Nadam/update_3/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_3/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_3/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_3/add_1"
  input: "^Nadam/Nadam/update_3/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_3/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_3/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_3/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_3/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_3/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_3/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_3/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_3/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_3/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_3/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_3/mul_4"
  input: "Nadam/Nadam/update_3/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_3/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_3/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_3/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_3/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_3/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_3/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_3/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_3/mul_6"
  input: "Nadam/Nadam/update_3/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_3/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_1/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_3/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_3/ReadVariableOp_2"
  input: "Nadam/Nadam/update_3/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_3/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_1/BiasAdd/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_3/sub"
  input: "^Nadam/Nadam/update_3/ReadVariableOp_2"
  input: "^sequential/dense_1/BiasAdd/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_1/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_4/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_2/MatMul_1"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_4/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_4/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_4/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_4/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_4/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_4/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_2/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_4/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_4/mul"
  input: "Nadam/Nadam/update_4/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_4/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_4/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_4/add"
  input: "^Nadam/Nadam/update_4/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_4/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_4/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_4/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_4/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_4/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_4/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_4/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_4/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_4/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_4/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_4/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_2/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_4/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_4/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_4/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_4/mul_2"
  input: "Nadam/Nadam/update_4/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_4/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_4/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_4/add_1"
  input: "^Nadam/Nadam/update_4/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_4/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_4/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_4/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_4/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_4/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_4/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_4/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_4/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_4/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_4/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_4/mul_4"
  input: "Nadam/Nadam/update_4/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_4/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_4/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_4/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_4/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_4/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_4/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_4/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_4/mul_6"
  input: "Nadam/Nadam/update_4/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_4/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_2/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_4/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_4/ReadVariableOp_2"
  input: "Nadam/Nadam/update_4/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_4/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_2/MatMul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_4/sub"
  input: "^Nadam/Nadam/update_4/ReadVariableOp_2"
  input: "^sequential/dense_2/MatMul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_5/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_2/BiasAdd/BiasAddGrad"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_5/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_5/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_5/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_5/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_5/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_5/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_2/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_5/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_5/mul"
  input: "Nadam/Nadam/update_5/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_5/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_5/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_5/add"
  input: "^Nadam/Nadam/update_5/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_5/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_5/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_5/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_5/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_5/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_5/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_5/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_5/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_5/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_5/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_5/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_2/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_5/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_5/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_5/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_5/mul_2"
  input: "Nadam/Nadam/update_5/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_5/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_5/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_5/add_1"
  input: "^Nadam/Nadam/update_5/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_5/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_5/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_5/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_5/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_5/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_5/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_5/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_5/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_5/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_5/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_5/mul_4"
  input: "Nadam/Nadam/update_5/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_5/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_5/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_5/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_5/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_5/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_5/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_5/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_5/mul_6"
  input: "Nadam/Nadam/update_5/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_5/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_2/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_5/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_5/ReadVariableOp_2"
  input: "Nadam/Nadam/update_5/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_5/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_2/BiasAdd/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_5/sub"
  input: "^Nadam/Nadam/update_5/ReadVariableOp_2"
  input: "^sequential/dense_2/BiasAdd/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_2/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_6/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_3/MatMul_1"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_6/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_6/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_6/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_6/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_6/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_6/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_3/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_6/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_6/mul"
  input: "Nadam/Nadam/update_6/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_6/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_6/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_6/add"
  input: "^Nadam/Nadam/update_6/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_6/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_6/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_6/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_6/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_6/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_6/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_6/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_6/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_6/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_6/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_6/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_3/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_6/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_6/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_6/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_6/mul_2"
  input: "Nadam/Nadam/update_6/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_6/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_6/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_6/add_1"
  input: "^Nadam/Nadam/update_6/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_6/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_6/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_6/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_6/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_6/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_6/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_6/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_6/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_6/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_6/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_6/mul_4"
  input: "Nadam/Nadam/update_6/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_6/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_6/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_6/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_6/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_6/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_6/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_6/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_6/mul_6"
  input: "Nadam/Nadam/update_6/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_6/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_3/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_6/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_6/ReadVariableOp_2"
  input: "Nadam/Nadam/update_6/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_6/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_3/MatMul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_6/sub"
  input: "^Nadam/Nadam/update_6/ReadVariableOp_2"
  input: "^sequential/dense_3/MatMul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_7/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_3/BiasAdd/BiasAddGrad"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_7/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_7/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_7/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_7/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_7/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_7/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_3/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_7/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_7/mul"
  input: "Nadam/Nadam/update_7/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_7/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_7/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_7/add"
  input: "^Nadam/Nadam/update_7/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_7/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_7/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_7/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_7/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_7/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_7/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_7/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_7/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_7/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_7/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_7/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_3/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_7/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_7/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_7/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_7/mul_2"
  input: "Nadam/Nadam/update_7/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_7/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_7/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_7/add_1"
  input: "^Nadam/Nadam/update_7/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_7/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_7/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_7/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_7/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_7/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_7/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_7/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_7/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_7/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_7/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_7/mul_4"
  input: "Nadam/Nadam/update_7/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_7/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_7/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_7/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_7/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_7/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_7/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_7/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_7/mul_6"
  input: "Nadam/Nadam/update_7/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_7/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_3/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_7/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_7/ReadVariableOp_2"
  input: "Nadam/Nadam/update_7/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_7/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_3/BiasAdd/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_7/sub"
  input: "^Nadam/Nadam/update_7/ReadVariableOp_2"
  input: "^sequential/dense_3/BiasAdd/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_3/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_8/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_4/MatMul_1"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_8/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_8/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_8/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_8/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_8/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_8/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_4/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_8/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_8/mul"
  input: "Nadam/Nadam/update_8/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_8/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_8/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_8/add"
  input: "^Nadam/Nadam/update_8/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_8/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_8/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_8/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_8/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_8/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_8/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_8/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_8/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_8/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_8/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_8/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_4/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_8/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_8/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_8/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_8/mul_2"
  input: "Nadam/Nadam/update_8/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_8/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_8/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_8/add_1"
  input: "^Nadam/Nadam/update_8/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_8/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_8/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_8/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_8/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_8/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_8/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_8/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_8/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_8/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_8/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_8/mul_4"
  input: "Nadam/Nadam/update_8/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_8/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_8/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_8/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_8/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_8/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_8/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_8/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_8/mul_6"
  input: "Nadam/Nadam/update_8/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_8/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_4/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_8/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_8/ReadVariableOp_2"
  input: "Nadam/Nadam/update_8/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_8/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_4/MatMul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_8/sub"
  input: "^Nadam/Nadam/update_8/ReadVariableOp_2"
  input: "^sequential/dense_4/MatMul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_9/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_4/BiasAdd/BiasAddGrad"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_9/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_9/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_9/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_9/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_9/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_9/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_4/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_9/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_9/mul"
  input: "Nadam/Nadam/update_9/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_9/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_9/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_9/add"
  input: "^Nadam/Nadam/update_9/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_9/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_9/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_9/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_9/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_9/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_9/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_9/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_9/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_9/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_9/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_9/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_4/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_9/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_9/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_9/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_9/mul_2"
  input: "Nadam/Nadam/update_9/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_9/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_9/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_9/add_1"
  input: "^Nadam/Nadam/update_9/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_9/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_9/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_9/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_9/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_9/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_9/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_9/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_9/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_9/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_9/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_9/mul_4"
  input: "Nadam/Nadam/update_9/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_9/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_9/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_9/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_9/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_9/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_9/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_9/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_9/mul_6"
  input: "Nadam/Nadam/update_9/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_9/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_4/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_9/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_9/ReadVariableOp_2"
  input: "Nadam/Nadam/update_9/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_9/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_4/BiasAdd/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_9/sub"
  input: "^Nadam/Nadam/update_9/ReadVariableOp_2"
  input: "^sequential/dense_4/BiasAdd/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_4/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_10/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_5/MatMul_1"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_10/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_10/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_10/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_10/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_10/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_10/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_5/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_10/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_10/mul"
  input: "Nadam/Nadam/update_10/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_10/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_10/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_10/add"
  input: "^Nadam/Nadam/update_10/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_10/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_10/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_10/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_10/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_10/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_10/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_10/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_10/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_10/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_10/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_10/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_5/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_10/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_10/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_10/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_10/mul_2"
  input: "Nadam/Nadam/update_10/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_10/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_10/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_10/add_1"
  input: "^Nadam/Nadam/update_10/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_10/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_10/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_10/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_10/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_10/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_10/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_10/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_10/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_10/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_10/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_10/mul_4"
  input: "Nadam/Nadam/update_10/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_10/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_10/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_10/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_10/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_10/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_10/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_10/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_10/mul_6"
  input: "Nadam/Nadam/update_10/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_10/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_5/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_10/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_10/ReadVariableOp_2"
  input: "Nadam/Nadam/update_10/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_10/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_5/MatMul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_10/sub"
  input: "^Nadam/Nadam/update_10/ReadVariableOp_2"
  input: "^sequential/dense_5/MatMul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_11/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_5/BiasAdd/BiasAddGrad"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_11/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_11/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_11/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_11/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_11/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_11/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_5/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_11/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_11/mul"
  input: "Nadam/Nadam/update_11/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_11/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_11/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_11/add"
  input: "^Nadam/Nadam/update_11/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_11/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_11/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_11/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_11/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_11/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_11/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_11/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_11/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_11/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_11/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_11/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_5/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_11/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_11/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_11/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_11/mul_2"
  input: "Nadam/Nadam/update_11/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_11/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_11/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_11/add_1"
  input: "^Nadam/Nadam/update_11/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_11/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_11/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_11/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_11/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_11/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_11/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_11/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_11/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_11/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_11/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_11/mul_4"
  input: "Nadam/Nadam/update_11/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_11/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_11/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_11/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_11/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_11/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_11/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_11/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_11/mul_6"
  input: "Nadam/Nadam/update_11/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_11/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_5/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_11/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_11/ReadVariableOp_2"
  input: "Nadam/Nadam/update_11/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_11/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_5/BiasAdd/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_11/sub"
  input: "^Nadam/Nadam/update_11/ReadVariableOp_2"
  input: "^sequential/dense_5/BiasAdd/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_5/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_12/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_6/MatMul_1"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_12/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_12/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_12/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_12/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_12/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_12/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_6/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_12/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_12/mul"
  input: "Nadam/Nadam/update_12/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_12/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_12/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_12/add"
  input: "^Nadam/Nadam/update_12/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_12/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_12/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_12/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_12/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_12/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_12/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_12/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_12/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_12/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_12/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_12/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_6/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_12/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_12/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_12/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_12/mul_2"
  input: "Nadam/Nadam/update_12/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_12/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_12/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_12/add_1"
  input: "^Nadam/Nadam/update_12/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_12/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_12/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_12/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_12/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_12/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_12/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_12/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_12/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_12/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_12/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_12/mul_4"
  input: "Nadam/Nadam/update_12/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_12/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_12/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_12/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_12/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_12/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_12/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_12/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_12/mul_6"
  input: "Nadam/Nadam/update_12/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_12/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_6/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_12/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_12/ReadVariableOp_2"
  input: "Nadam/Nadam/update_12/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_12/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_6/MatMul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_12/sub"
  input: "^Nadam/Nadam/update_12/ReadVariableOp_2"
  input: "^sequential/dense_6/MatMul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_13/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_6/BiasAdd/BiasAddGrad"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_13/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_13/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_13/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_13/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_13/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_13/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_6/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_13/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_13/mul"
  input: "Nadam/Nadam/update_13/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_13/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_13/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_13/add"
  input: "^Nadam/Nadam/update_13/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_13/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_13/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_13/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_13/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_13/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_13/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_13/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_13/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_13/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_13/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_13/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_6/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_13/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_13/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_13/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_13/mul_2"
  input: "Nadam/Nadam/update_13/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_13/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_13/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_13/add_1"
  input: "^Nadam/Nadam/update_13/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_13/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_13/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_13/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_13/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_13/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_13/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_13/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_13/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_13/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_13/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_13/mul_4"
  input: "Nadam/Nadam/update_13/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_13/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_13/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_13/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_13/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_13/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_13/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_13/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_13/mul_6"
  input: "Nadam/Nadam/update_13/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_13/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_6/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_13/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_13/ReadVariableOp_2"
  input: "Nadam/Nadam/update_13/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_13/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_6/BiasAdd/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_13/sub"
  input: "^Nadam/Nadam/update_13/ReadVariableOp_2"
  input: "^sequential/dense_6/BiasAdd/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_6/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_14/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_7/MatMul_1"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_14/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_14/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_14/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_14/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_14/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_14/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_7/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_14/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_14/mul"
  input: "Nadam/Nadam/update_14/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_14/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_14/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_14/add"
  input: "^Nadam/Nadam/update_14/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_14/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_14/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_14/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_14/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_14/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_14/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_14/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_14/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_14/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_14/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_14/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_7/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_14/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_14/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_14/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_14/mul_2"
  input: "Nadam/Nadam/update_14/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_14/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_14/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_14/add_1"
  input: "^Nadam/Nadam/update_14/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_14/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_14/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_14/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_14/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_14/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_14/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_14/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_14/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_14/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_14/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_14/mul_4"
  input: "Nadam/Nadam/update_14/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_14/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_14/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_14/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_14/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_14/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_14/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_14/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_14/mul_6"
  input: "Nadam/Nadam/update_14/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_14/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_7/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_14/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_14/ReadVariableOp_2"
  input: "Nadam/Nadam/update_14/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_14/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_7/MatMul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_14/sub"
  input: "^Nadam/Nadam/update_14/ReadVariableOp_2"
  input: "^sequential/dense_7/MatMul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_15/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_7/BiasAdd/BiasAddGrad"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_15/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_15/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_15/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_15/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_15/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_15/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_7/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_15/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_15/mul"
  input: "Nadam/Nadam/update_15/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_15/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_15/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_15/add"
  input: "^Nadam/Nadam/update_15/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_15/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_15/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_15/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_15/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_15/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_15/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_15/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_15/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_15/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_15/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_15/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_7/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_15/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_15/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_15/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_15/mul_2"
  input: "Nadam/Nadam/update_15/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_15/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_15/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_15/add_1"
  input: "^Nadam/Nadam/update_15/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_15/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_15/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_15/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_15/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_15/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_15/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_15/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_15/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_15/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_15/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_15/mul_4"
  input: "Nadam/Nadam/update_15/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_15/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_15/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_15/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_15/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_15/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_15/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_15/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_15/mul_6"
  input: "Nadam/Nadam/update_15/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_15/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_7/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_15/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_15/ReadVariableOp_2"
  input: "Nadam/Nadam/update_15/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_15/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_7/BiasAdd/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_15/sub"
  input: "^Nadam/Nadam/update_15/ReadVariableOp_2"
  input: "^sequential/dense_7/BiasAdd/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_7/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_16/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_8/MatMul_1"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_16/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_16/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_16/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_16/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_16/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_16/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_8/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_16/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_16/mul"
  input: "Nadam/Nadam/update_16/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_16/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_16/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_16/add"
  input: "^Nadam/Nadam/update_16/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_16/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_16/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_16/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_16/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_16/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_16/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_16/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_16/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_16/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_16/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_16/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_8/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_16/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_16/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_16/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_16/mul_2"
  input: "Nadam/Nadam/update_16/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_16/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_16/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_16/add_1"
  input: "^Nadam/Nadam/update_16/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_16/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_16/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_16/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_16/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_16/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_16/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_16/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_16/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_16/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_16/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_16/mul_4"
  input: "Nadam/Nadam/update_16/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_16/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_16/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_16/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_16/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_16/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_16/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_16/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_16/mul_6"
  input: "Nadam/Nadam/update_16/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_16/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_8/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_16/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_16/ReadVariableOp_2"
  input: "Nadam/Nadam/update_16/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_16/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_8/MatMul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_16/sub"
  input: "^Nadam/Nadam/update_16/ReadVariableOp_2"
  input: "^sequential/dense_8/MatMul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_17/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_8/BiasAdd/BiasAddGrad"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_17/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_17/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_17/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_17/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_17/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_17/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_8/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_17/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_17/mul"
  input: "Nadam/Nadam/update_17/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_17/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_17/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_17/add"
  input: "^Nadam/Nadam/update_17/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_17/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_17/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_17/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_17/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_17/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_17/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_17/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_17/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_17/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_17/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_17/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_8/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_17/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_17/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_17/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_17/mul_2"
  input: "Nadam/Nadam/update_17/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_17/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_17/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_17/add_1"
  input: "^Nadam/Nadam/update_17/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_17/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_17/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_17/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_17/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_17/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_17/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_17/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_17/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_17/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_17/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_17/mul_4"
  input: "Nadam/Nadam/update_17/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_17/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_17/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_17/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_17/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_17/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_17/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_17/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_17/mul_6"
  input: "Nadam/Nadam/update_17/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_17/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_8/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_17/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_17/ReadVariableOp_2"
  input: "Nadam/Nadam/update_17/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_17/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_8/BiasAdd/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_17/sub"
  input: "^Nadam/Nadam/update_17/ReadVariableOp_2"
  input: "^sequential/dense_8/BiasAdd/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_8/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_18/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_9/MatMul_1"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_18/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_18/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_18/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_18/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_18/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_18/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_9/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_18/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_18/mul"
  input: "Nadam/Nadam/update_18/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_18/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_18/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_18/add"
  input: "^Nadam/Nadam/update_18/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_18/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_18/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_18/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_18/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_18/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_18/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_18/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_18/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_18/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_18/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_18/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_9/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_18/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_18/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_18/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_18/mul_2"
  input: "Nadam/Nadam/update_18/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_18/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_18/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_18/add_1"
  input: "^Nadam/Nadam/update_18/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_18/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_18/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_18/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_18/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_18/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_18/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_18/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_18/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_18/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_18/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_18/mul_4"
  input: "Nadam/Nadam/update_18/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_18/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_18/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_18/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_18/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_18/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_18/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_18/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_18/mul_6"
  input: "Nadam/Nadam/update_18/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_18/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_9/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_18/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_18/ReadVariableOp_2"
  input: "Nadam/Nadam/update_18/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_18/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_9/MatMul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_18/sub"
  input: "^Nadam/Nadam/update_18/ReadVariableOp_2"
  input: "^sequential/dense_9/MatMul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_19/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_9/BiasAdd/BiasAddGrad"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_19/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_19/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_19/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_19/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_19/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_19/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_9/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_19/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_19/mul"
  input: "Nadam/Nadam/update_19/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_19/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_19/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_19/add"
  input: "^Nadam/Nadam/update_19/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_19/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_19/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_19/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_19/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_19/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_19/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_19/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_19/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_19/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_19/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_19/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_9/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_19/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_19/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_19/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_19/mul_2"
  input: "Nadam/Nadam/update_19/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_19/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_19/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_19/add_1"
  input: "^Nadam/Nadam/update_19/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_19/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_19/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_19/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_19/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_19/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_19/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_19/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_19/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_19/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_19/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_19/mul_4"
  input: "Nadam/Nadam/update_19/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_19/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_19/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_19/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_19/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_19/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_19/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_19/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_19/mul_6"
  input: "Nadam/Nadam/update_19/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_19/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_9/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_19/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_19/ReadVariableOp_2"
  input: "Nadam/Nadam/update_19/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_19/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_9/BiasAdd/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_19/sub"
  input: "^Nadam/Nadam/update_19/ReadVariableOp_2"
  input: "^sequential/dense_9/BiasAdd/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_9/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_20/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_10/MatMul_1"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_20/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_20/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_20/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_20/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_20/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_20/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_10/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_20/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_20/mul"
  input: "Nadam/Nadam/update_20/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_20/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_20/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_20/add"
  input: "^Nadam/Nadam/update_20/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_20/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_20/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_20/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_20/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_20/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_20/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_20/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_20/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_20/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_20/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_20/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_10/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_20/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_20/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_20/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_20/mul_2"
  input: "Nadam/Nadam/update_20/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_20/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_20/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_20/add_1"
  input: "^Nadam/Nadam/update_20/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_20/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_20/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_20/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_20/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_20/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_20/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_20/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_20/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_20/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_20/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_20/mul_4"
  input: "Nadam/Nadam/update_20/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_20/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_20/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_20/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_20/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_20/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_20/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_20/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_20/mul_6"
  input: "Nadam/Nadam/update_20/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_20/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_10/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_20/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_20/ReadVariableOp_2"
  input: "Nadam/Nadam/update_20/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_20/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_10/MatMul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_20/sub"
  input: "^Nadam/Nadam/update_20/ReadVariableOp_2"
  input: "^sequential/dense_10/MatMul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_21/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_10/BiasAdd/BiasAddGrad"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_21/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_21/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_21/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_21/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_21/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_21/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_10/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_21/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_21/mul"
  input: "Nadam/Nadam/update_21/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_21/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_21/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_21/add"
  input: "^Nadam/Nadam/update_21/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_21/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_21/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_21/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_21/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_21/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_21/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_21/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_21/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_21/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_21/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_21/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_10/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_21/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_21/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_21/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_21/mul_2"
  input: "Nadam/Nadam/update_21/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_21/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_21/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_21/add_1"
  input: "^Nadam/Nadam/update_21/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_21/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_21/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_21/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_21/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_21/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_21/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_21/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_21/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_21/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_21/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_21/mul_4"
  input: "Nadam/Nadam/update_21/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_21/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_21/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_21/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_21/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_21/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_21/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_21/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_21/mul_6"
  input: "Nadam/Nadam/update_21/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_21/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_10/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_21/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_21/ReadVariableOp_2"
  input: "Nadam/Nadam/update_21/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_21/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_10/BiasAdd/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_21/sub"
  input: "^Nadam/Nadam/update_21/ReadVariableOp_2"
  input: "^sequential/dense_10/BiasAdd/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_10/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_22/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_11/MatMul_1"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_22/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_22/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_22/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_22/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_22/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_22/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_11/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_22/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_22/mul"
  input: "Nadam/Nadam/update_22/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_22/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_22/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_22/add"
  input: "^Nadam/Nadam/update_22/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_22/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_22/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_22/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_22/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_22/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_22/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_22/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_22/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_22/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_22/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_22/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_11/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_22/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_22/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_22/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_22/mul_2"
  input: "Nadam/Nadam/update_22/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_22/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_22/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_22/add_1"
  input: "^Nadam/Nadam/update_22/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_22/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_22/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_22/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_22/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_22/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_22/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_22/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_22/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_22/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_22/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_22/mul_4"
  input: "Nadam/Nadam/update_22/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_22/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_22/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_22/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_22/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_22/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_22/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_22/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_22/mul_6"
  input: "Nadam/Nadam/update_22/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_22/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_11/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_22/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_22/ReadVariableOp_2"
  input: "Nadam/Nadam/update_22/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_22/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_11/MatMul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_22/sub"
  input: "^Nadam/Nadam/update_22/ReadVariableOp_2"
  input: "^sequential/dense_11/MatMul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_23/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_11/BiasAdd/BiasAddGrad"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_23/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_23/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_23/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_23/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_23/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_23/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_11/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_23/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_23/mul"
  input: "Nadam/Nadam/update_23/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_23/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_23/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_23/add"
  input: "^Nadam/Nadam/update_23/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_23/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_23/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_23/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_23/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_23/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_23/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_23/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_23/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_23/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_23/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_23/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_11/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_23/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_23/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_23/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_23/mul_2"
  input: "Nadam/Nadam/update_23/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_23/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_23/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_23/add_1"
  input: "^Nadam/Nadam/update_23/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_23/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_23/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_23/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_23/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_23/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_23/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_23/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_23/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_23/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_23/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_23/mul_4"
  input: "Nadam/Nadam/update_23/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_23/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_23/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_23/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_23/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_23/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_23/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_23/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_23/mul_6"
  input: "Nadam/Nadam/update_23/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_23/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_11/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_23/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_23/ReadVariableOp_2"
  input: "Nadam/Nadam/update_23/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_23/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_11/BiasAdd/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_23/sub"
  input: "^Nadam/Nadam/update_23/ReadVariableOp_2"
  input: "^sequential/dense_11/BiasAdd/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_11/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_24/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_12/MatMul_1"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_24/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_24/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_24/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_24/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_24/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_24/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_12/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_24/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_24/mul"
  input: "Nadam/Nadam/update_24/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_24/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_24/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_24/add"
  input: "^Nadam/Nadam/update_24/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_24/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_24/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_24/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_24/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_24/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_24/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_24/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_24/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_24/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_24/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_24/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_12/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_24/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_24/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_24/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_24/mul_2"
  input: "Nadam/Nadam/update_24/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_24/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_24/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_24/add_1"
  input: "^Nadam/Nadam/update_24/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_24/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_24/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_24/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_24/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_24/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_24/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_24/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_24/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_24/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_24/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_24/mul_4"
  input: "Nadam/Nadam/update_24/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_24/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_24/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_24/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_24/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_24/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_24/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_24/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_24/mul_6"
  input: "Nadam/Nadam/update_24/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_24/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_12/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_24/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_24/ReadVariableOp_2"
  input: "Nadam/Nadam/update_24/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_24/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_12/MatMul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_24/sub"
  input: "^Nadam/Nadam/update_24/ReadVariableOp_2"
  input: "^sequential/dense_12/MatMul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_25/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_12/BiasAdd/BiasAddGrad"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_25/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_25/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_25/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_25/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_25/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_25/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_12/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_25/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_25/mul"
  input: "Nadam/Nadam/update_25/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_25/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_25/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_25/add"
  input: "^Nadam/Nadam/update_25/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_25/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_25/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_25/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_25/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_25/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_25/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_25/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_25/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_25/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_25/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_25/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_12/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_25/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_25/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_25/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_25/mul_2"
  input: "Nadam/Nadam/update_25/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_25/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_25/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_25/add_1"
  input: "^Nadam/Nadam/update_25/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_25/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_25/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_25/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_25/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_25/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_25/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_25/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_25/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_25/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_25/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_25/mul_4"
  input: "Nadam/Nadam/update_25/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_25/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_25/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_25/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_25/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_25/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_25/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_25/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_25/mul_6"
  input: "Nadam/Nadam/update_25/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_25/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_12/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_25/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_25/ReadVariableOp_2"
  input: "Nadam/Nadam/update_25/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_25/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_12/BiasAdd/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_25/sub"
  input: "^Nadam/Nadam/update_25/ReadVariableOp_2"
  input: "^sequential/dense_12/BiasAdd/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_12/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_26/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_13/MatMul_1"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_26/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_26/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_26/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_26/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_26/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_26/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_13/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_26/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_26/mul"
  input: "Nadam/Nadam/update_26/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_26/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_26/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_26/add"
  input: "^Nadam/Nadam/update_26/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_26/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_26/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_26/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_26/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_26/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_26/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_26/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_26/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_26/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_26/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_26/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_13/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_26/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_26/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_26/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_26/mul_2"
  input: "Nadam/Nadam/update_26/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_26/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_26/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_26/add_1"
  input: "^Nadam/Nadam/update_26/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_26/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_26/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_26/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_26/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_26/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_26/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_26/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_26/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_26/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_26/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_26/mul_4"
  input: "Nadam/Nadam/update_26/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_26/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_26/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_26/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_26/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_26/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_26/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_26/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_26/mul_6"
  input: "Nadam/Nadam/update_26/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_26/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_13/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_26/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_26/ReadVariableOp_2"
  input: "Nadam/Nadam/update_26/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_26/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_13/MatMul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_26/sub"
  input: "^Nadam/Nadam/update_26/ReadVariableOp_2"
  input: "^sequential/dense_13/MatMul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_27/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_13/BiasAdd/BiasAddGrad"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_27/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_27/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_27/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_27/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_27/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_27/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_13/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_27/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_27/mul"
  input: "Nadam/Nadam/update_27/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_27/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_27/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_27/add"
  input: "^Nadam/Nadam/update_27/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_27/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_27/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_27/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_27/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_27/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_27/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_27/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_27/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_27/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_27/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_27/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_13/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_27/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_27/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_27/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_27/mul_2"
  input: "Nadam/Nadam/update_27/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_27/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_27/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_27/add_1"
  input: "^Nadam/Nadam/update_27/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_27/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_27/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_27/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_27/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_27/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_27/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_27/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_27/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_27/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_27/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_27/mul_4"
  input: "Nadam/Nadam/update_27/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_27/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_27/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_27/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_27/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_27/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_27/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_27/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_27/mul_6"
  input: "Nadam/Nadam/update_27/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_27/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_13/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_27/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_27/ReadVariableOp_2"
  input: "Nadam/Nadam/update_27/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_27/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_13/BiasAdd/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_27/sub"
  input: "^Nadam/Nadam/update_27/ReadVariableOp_2"
  input: "^sequential/dense_13/BiasAdd/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_13/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_28/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_14/MatMul_1"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_28/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_28/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_28/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_28/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_28/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_28/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_14/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_28/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_28/mul"
  input: "Nadam/Nadam/update_28/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_28/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_28/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_28/add"
  input: "^Nadam/Nadam/update_28/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_28/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_28/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_28/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_28/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_28/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_28/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_28/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_28/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_28/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_28/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_28/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_14/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_28/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_28/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_28/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_28/mul_2"
  input: "Nadam/Nadam/update_28/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_28/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_28/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_28/add_1"
  input: "^Nadam/Nadam/update_28/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_28/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_28/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_28/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_28/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_28/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_28/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_28/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_28/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_28/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_28/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_28/mul_4"
  input: "Nadam/Nadam/update_28/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_28/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_28/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_28/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_28/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_28/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_28/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_28/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_28/mul_6"
  input: "Nadam/Nadam/update_28/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_28/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_14/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_28/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_28/ReadVariableOp_2"
  input: "Nadam/Nadam/update_28/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_28/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_14/MatMul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_28/sub"
  input: "^Nadam/Nadam/update_28/ReadVariableOp_2"
  input: "^sequential/dense_14/MatMul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_29/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_14/BiasAdd/BiasAddGrad"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_29/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_29/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_29/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_29/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_29/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_29/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_14/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_29/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_29/mul"
  input: "Nadam/Nadam/update_29/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_29/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_29/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_29/add"
  input: "^Nadam/Nadam/update_29/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_29/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_29/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_29/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_29/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_29/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_29/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_29/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_29/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_29/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_29/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_29/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_14/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_29/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_29/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_29/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_29/mul_2"
  input: "Nadam/Nadam/update_29/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_29/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_29/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_29/add_1"
  input: "^Nadam/Nadam/update_29/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_29/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_29/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_29/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_29/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_29/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_29/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_29/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_29/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_29/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_29/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_29/mul_4"
  input: "Nadam/Nadam/update_29/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_29/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_29/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_29/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_29/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_29/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_29/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_29/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_29/mul_6"
  input: "Nadam/Nadam/update_29/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_29/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_14/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_29/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_29/ReadVariableOp_2"
  input: "Nadam/Nadam/update_29/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_29/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_14/BiasAdd/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_29/sub"
  input: "^Nadam/Nadam/update_29/ReadVariableOp_2"
  input: "^sequential/dense_14/BiasAdd/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_14/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_30/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_15/MatMul_1"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_30/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_30/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_30/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_30/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_30/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_30/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_15/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_30/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_30/mul"
  input: "Nadam/Nadam/update_30/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_30/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_30/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_30/add"
  input: "^Nadam/Nadam/update_30/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_30/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_30/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_30/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_30/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_30/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_30/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_30/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_30/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_30/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_30/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_30/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_15/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_30/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_30/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_30/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_30/mul_2"
  input: "Nadam/Nadam/update_30/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_30/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_30/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_30/add_1"
  input: "^Nadam/Nadam/update_30/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_30/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_30/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_30/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_30/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_30/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_30/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_30/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_30/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_30/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_30/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_30/mul_4"
  input: "Nadam/Nadam/update_30/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_30/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_30/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_30/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_30/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_30/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_30/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_30/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_30/mul_6"
  input: "Nadam/Nadam/update_30/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_30/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_15/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_30/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_30/ReadVariableOp_2"
  input: "Nadam/Nadam/update_30/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_30/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_15/MatMul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_30/sub"
  input: "^Nadam/Nadam/update_30/ReadVariableOp_2"
  input: "^sequential/dense_15/MatMul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_31/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_15/BiasAdd/BiasAddGrad"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_31/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_31/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_31/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_31/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_31/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_31/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_15/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_31/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_31/mul"
  input: "Nadam/Nadam/update_31/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_31/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_31/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_31/add"
  input: "^Nadam/Nadam/update_31/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_31/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_31/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_31/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_31/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_31/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_31/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_31/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_31/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_31/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_31/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_31/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_15/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_31/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_31/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_31/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_31/mul_2"
  input: "Nadam/Nadam/update_31/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_31/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_31/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_31/add_1"
  input: "^Nadam/Nadam/update_31/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_31/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_31/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_31/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_31/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_31/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_31/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_31/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_31/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_31/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_31/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_31/mul_4"
  input: "Nadam/Nadam/update_31/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_31/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_31/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_31/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_31/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_31/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_31/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_31/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_31/mul_6"
  input: "Nadam/Nadam/update_31/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_31/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_15/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_31/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_31/ReadVariableOp_2"
  input: "Nadam/Nadam/update_31/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_31/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_15/BiasAdd/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_31/sub"
  input: "^Nadam/Nadam/update_31/ReadVariableOp_2"
  input: "^sequential/dense_15/BiasAdd/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_15/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_32/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_16/MatMul_1"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_32/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_32/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_32/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_32/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_32/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_32/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_16/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_32/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_32/mul"
  input: "Nadam/Nadam/update_32/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_32/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_32/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_32/add"
  input: "^Nadam/Nadam/update_32/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_32/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_32/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_32/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_32/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_32/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_32/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_32/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_32/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_32/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_32/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_32/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_16/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_32/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_32/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_32/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_32/mul_2"
  input: "Nadam/Nadam/update_32/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_32/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_32/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_32/add_1"
  input: "^Nadam/Nadam/update_32/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_32/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_32/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_32/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_32/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_32/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_32/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_32/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_32/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_32/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_32/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_32/mul_4"
  input: "Nadam/Nadam/update_32/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_32/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_32/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_32/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_32/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_32/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_32/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_32/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_32/mul_6"
  input: "Nadam/Nadam/update_32/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_32/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_16/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_32/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_32/ReadVariableOp_2"
  input: "Nadam/Nadam/update_32/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_32/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_16/MatMul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_32/sub"
  input: "^Nadam/Nadam/update_32/ReadVariableOp_2"
  input: "^sequential/dense_16/MatMul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_33/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_16/BiasAdd/BiasAddGrad"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_33/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_33/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_33/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_33/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_33/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_33/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_16/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_33/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_33/mul"
  input: "Nadam/Nadam/update_33/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_33/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_33/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_33/add"
  input: "^Nadam/Nadam/update_33/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_33/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_33/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_33/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_33/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_33/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_33/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_33/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_33/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_33/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_33/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_33/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_16/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_33/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_33/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_33/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_33/mul_2"
  input: "Nadam/Nadam/update_33/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_33/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_33/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_33/add_1"
  input: "^Nadam/Nadam/update_33/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_33/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_33/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_33/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_33/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_33/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_33/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_33/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_33/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_33/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_33/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_33/mul_4"
  input: "Nadam/Nadam/update_33/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_33/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_33/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_33/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_33/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_33/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_33/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_33/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_33/mul_6"
  input: "Nadam/Nadam/update_33/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_33/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_16/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_33/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_33/ReadVariableOp_2"
  input: "Nadam/Nadam/update_33/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_33/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_16/BiasAdd/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_33/sub"
  input: "^Nadam/Nadam/update_33/ReadVariableOp_2"
  input: "^sequential/dense_16/BiasAdd/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_16/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_34/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_17/MatMul_1"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_34/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_34/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_34/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_34/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_34/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_34/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_17/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_34/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_34/mul"
  input: "Nadam/Nadam/update_34/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_34/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_34/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_34/add"
  input: "^Nadam/Nadam/update_34/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_34/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_34/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_34/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_34/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_34/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_34/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_34/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_34/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_34/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_34/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_34/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_17/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_34/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_34/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_34/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_34/mul_2"
  input: "Nadam/Nadam/update_34/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_34/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_34/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_34/add_1"
  input: "^Nadam/Nadam/update_34/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_34/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_34/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_34/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_34/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_34/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_34/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_34/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_34/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_34/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_34/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_34/mul_4"
  input: "Nadam/Nadam/update_34/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_34/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_34/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_34/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_34/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_34/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_34/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_34/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_34/mul_6"
  input: "Nadam/Nadam/update_34/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_34/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_17/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_34/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_34/ReadVariableOp_2"
  input: "Nadam/Nadam/update_34/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_34/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_17/MatMul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_34/sub"
  input: "^Nadam/Nadam/update_34/ReadVariableOp_2"
  input: "^sequential/dense_17/MatMul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_35/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_17/BiasAdd/BiasAddGrad"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_35/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_35/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_35/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_35/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_35/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_35/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_17/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_35/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_35/mul"
  input: "Nadam/Nadam/update_35/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_35/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_35/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_35/add"
  input: "^Nadam/Nadam/update_35/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_35/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_35/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_35/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_35/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_35/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_35/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_35/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_35/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_35/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_35/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_35/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_17/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_35/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_35/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_35/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_35/mul_2"
  input: "Nadam/Nadam/update_35/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_35/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_35/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_35/add_1"
  input: "^Nadam/Nadam/update_35/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_35/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_35/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_35/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_35/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_35/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_35/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_35/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_35/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_35/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_35/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_35/mul_4"
  input: "Nadam/Nadam/update_35/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_35/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_35/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_35/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_35/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_35/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_35/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_35/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_35/mul_6"
  input: "Nadam/Nadam/update_35/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_35/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_17/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_35/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_35/ReadVariableOp_2"
  input: "Nadam/Nadam/update_35/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_35/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_17/BiasAdd/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_35/sub"
  input: "^Nadam/Nadam/update_35/ReadVariableOp_2"
  input: "^sequential/dense_17/BiasAdd/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_17/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_36/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_18/MatMul_1"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_36/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_36/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_36/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_36/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_36/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_36/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_18/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_36/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_36/mul"
  input: "Nadam/Nadam/update_36/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_36/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_36/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_36/add"
  input: "^Nadam/Nadam/update_36/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_36/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_36/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_36/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_36/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_36/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_36/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_36/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_36/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_36/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_36/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_36/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_18/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_36/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_36/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_36/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_36/mul_2"
  input: "Nadam/Nadam/update_36/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_36/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_36/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_36/add_1"
  input: "^Nadam/Nadam/update_36/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_36/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_36/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_36/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_36/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_36/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_36/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_36/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_36/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_36/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_36/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_36/mul_4"
  input: "Nadam/Nadam/update_36/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_36/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_36/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_36/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_36/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_36/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_36/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_36/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_36/mul_6"
  input: "Nadam/Nadam/update_36/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_36/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_18/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_36/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_36/ReadVariableOp_2"
  input: "Nadam/Nadam/update_36/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_36/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_18/MatMul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_36/sub"
  input: "^Nadam/Nadam/update_36/ReadVariableOp_2"
  input: "^sequential/dense_18/MatMul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_37/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_18/BiasAdd/BiasAddGrad"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_37/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_37/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_37/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_37/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_37/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_37/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_18/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_37/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_37/mul"
  input: "Nadam/Nadam/update_37/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_37/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_37/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_37/add"
  input: "^Nadam/Nadam/update_37/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_37/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_37/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_37/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_37/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_37/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_37/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_37/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_37/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_37/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_37/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_37/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_18/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_37/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_37/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_37/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_37/mul_2"
  input: "Nadam/Nadam/update_37/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_37/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_37/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_37/add_1"
  input: "^Nadam/Nadam/update_37/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_37/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_37/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_37/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_37/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_37/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_37/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_37/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_37/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_37/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_37/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_37/mul_4"
  input: "Nadam/Nadam/update_37/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_37/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_37/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_37/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_37/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_37/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_37/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_37/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_37/mul_6"
  input: "Nadam/Nadam/update_37/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_37/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_18/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_37/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_37/ReadVariableOp_2"
  input: "Nadam/Nadam/update_37/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_37/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_18/BiasAdd/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_37/sub"
  input: "^Nadam/Nadam/update_37/ReadVariableOp_2"
  input: "^sequential/dense_18/BiasAdd/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_18/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_38/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_19/MatMul_1"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_38/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_38/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_38/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_38/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_38/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_38/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_19/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_38/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_38/mul"
  input: "Nadam/Nadam/update_38/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_38/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_38/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_38/add"
  input: "^Nadam/Nadam/update_38/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_38/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_38/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_38/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_38/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_38/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_38/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_38/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_38/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_38/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_38/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_38/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_19/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_38/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_38/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_38/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_38/mul_2"
  input: "Nadam/Nadam/update_38/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_38/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_38/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_38/add_1"
  input: "^Nadam/Nadam/update_38/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_38/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_38/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_38/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_38/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_38/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_38/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_38/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_38/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_38/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_38/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_38/mul_4"
  input: "Nadam/Nadam/update_38/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_38/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_38/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_38/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_38/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_38/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_38/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_38/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_38/mul_6"
  input: "Nadam/Nadam/update_38/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_38/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_19/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_38/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_38/ReadVariableOp_2"
  input: "Nadam/Nadam/update_38/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_38/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_19/MatMul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_38/sub"
  input: "^Nadam/Nadam/update_38/ReadVariableOp_2"
  input: "^sequential/dense_19/MatMul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_39/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_19/BiasAdd/BiasAddGrad"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_39/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_39/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_39/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_39/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_39/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_39/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_19/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_39/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_39/mul"
  input: "Nadam/Nadam/update_39/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_39/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_39/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_39/add"
  input: "^Nadam/Nadam/update_39/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_39/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_39/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_39/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_39/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_39/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_39/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_39/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_39/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_39/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_39/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_39/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_19/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_39/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_39/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_39/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_39/mul_2"
  input: "Nadam/Nadam/update_39/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_39/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_39/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_39/add_1"
  input: "^Nadam/Nadam/update_39/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_39/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_39/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_39/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_39/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_39/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_39/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_39/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_39/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_39/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_39/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_39/mul_4"
  input: "Nadam/Nadam/update_39/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_39/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_39/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_39/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_39/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_39/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_39/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_39/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_39/mul_6"
  input: "Nadam/Nadam/update_39/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_39/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_19/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_39/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_39/ReadVariableOp_2"
  input: "Nadam/Nadam/update_39/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_39/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_19/BiasAdd/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_39/sub"
  input: "^Nadam/Nadam/update_39/ReadVariableOp_2"
  input: "^sequential/dense_19/BiasAdd/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_19/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_40/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_20/MatMul_1"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_40/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_40/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_40/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 10
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_40/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_40/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_40/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_20/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_40/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_40/mul"
  input: "Nadam/Nadam/update_40/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_40/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_40/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_40/add"
  input: "^Nadam/Nadam/update_40/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_40/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_40/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_40/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 10
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_40/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_40/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_40/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_40/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_40/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 10
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_40/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_40/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_40/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_20/MatMul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_40/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_40/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_40/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_40/mul_2"
  input: "Nadam/Nadam/update_40/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_40/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_40/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_40/add_1"
  input: "^Nadam/Nadam/update_40/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_40/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_40/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_40/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 10
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_40/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_40/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_40/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_40/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_40/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_40/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_40/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_40/mul_4"
  input: "Nadam/Nadam/update_40/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_40/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_40/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_40/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_40/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_40/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_40/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_40/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_40/mul_6"
  input: "Nadam/Nadam/update_40/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_40/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_20/MatMul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 10
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_40/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_40/ReadVariableOp_2"
  input: "Nadam/Nadam/update_40/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 100
          }
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_40/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_20/MatMul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_40/sub"
  input: "^Nadam/Nadam/update_40/ReadVariableOp_2"
  input: "^sequential/dense_20/MatMul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/MatMul/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_41/truediv"
  op: "RealDiv"
  input: "gradient_tape/sequential/dense_20/BiasAdd/BiasAddGrad"
  input: "Nadam/sub_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_41/mul/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_41/mul/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_41/mul/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 10
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_41/mul"
  op: "Mul"
  input: "Nadam/Identity_2"
  input: "Nadam/Nadam/update_41/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_41/mul_1"
  op: "Mul"
  input: "Nadam/sub_2"
  input: "gradient_tape/sequential/dense_20/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_41/add"
  op: "AddV2"
  input: "Nadam/Nadam/update_41/mul"
  input: "Nadam/Nadam/update_41/mul_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_41/AssignVariableOp"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_41/mul/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_41/add"
  input: "^Nadam/Nadam/update_41/mul/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_41/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_41/mul/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_41/AssignVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 10
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_41/truediv_1"
  op: "RealDiv"
  input: "Nadam/Nadam/update_41/ReadVariableOp"
  input: "Nadam/sub_6"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_41/mul_2/ReadVariableOp/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_41/mul_2/ReadVariableOp"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_41/mul_2/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 10
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_41/mul_2"
  op: "Mul"
  input: "Nadam/Identity_3"
  input: "Nadam/Nadam/update_41/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_41/Square"
  op: "Square"
  input: "gradient_tape/sequential/dense_20/BiasAdd/BiasAddGrad"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_41/mul_3"
  op: "Mul"
  input: "Nadam/sub_3"
  input: "Nadam/Nadam/update_41/Square"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_41/add_1"
  op: "AddV2"
  input: "Nadam/Nadam/update_41/mul_2"
  input: "Nadam/Nadam/update_41/mul_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_41/AssignVariableOp_1"
  op: "AssignVariableOp"
  input: "Nadam/Nadam/update_41/mul_2/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_41/add_1"
  input: "^Nadam/Nadam/update_41/mul_2/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_41/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "Nadam/Nadam/update_41/mul_2/ReadVariableOp/resource"
  input: "^Nadam/Nadam/update_41/AssignVariableOp_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 10
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_41/truediv_2"
  op: "RealDiv"
  input: "Nadam/Nadam/update_41/ReadVariableOp_1"
  input: "Nadam/sub_7"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_41/mul_4"
  op: "Mul"
  input: "Nadam/sub_4"
  input: "Nadam/Nadam/update_41/truediv"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_41/mul_5"
  op: "Mul"
  input: "Nadam/mul_5"
  input: "Nadam/Nadam/update_41/truediv_1"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_41/add_2"
  op: "AddV2"
  input: "Nadam/Nadam/update_41/mul_4"
  input: "Nadam/Nadam/update_41/mul_5"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_41/mul_6"
  op: "Mul"
  input: "Nadam/Identity_1"
  input: "Nadam/Nadam/update_41/add_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_41/Sqrt"
  op: "Sqrt"
  input: "Nadam/Nadam/update_41/truediv_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_41/add_3"
  op: "AddV2"
  input: "Nadam/Nadam/update_41/Sqrt"
  input: "Nadam/Const"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_41/truediv_3"
  op: "RealDiv"
  input: "Nadam/Nadam/update_41/mul_6"
  input: "Nadam/Nadam/update_41/add_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_41/ReadVariableOp_2"
  op: "ReadVariableOp"
  input: "sequential/dense_20/BiasAdd/ReadVariableOp/resource"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 10
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/update_41/sub"
  op: "Sub"
  input: "Nadam/Nadam/update_41/ReadVariableOp_2"
  input: "Nadam/Nadam/update_41/truediv_3"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 10
          }
        }
      }
    }
  }
}
node {
  name: "Nadam/Nadam/update_41/AssignVariableOp_2"
  op: "AssignVariableOp"
  input: "sequential/dense_20/BiasAdd/ReadVariableOp/resource"
  input: "Nadam/Nadam/update_41/sub"
  input: "^Nadam/Nadam/update_41/ReadVariableOp_2"
  input: "^sequential/dense_20/BiasAdd/ReadVariableOp"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
  attr {
    key: "_class"
    value {
      list {
        s: "loc:@sequential/dense_20/BiasAdd/ReadVariableOp/resource"
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Nadam/Nadam/group_deps"
  op: "NoOp"
  input: "^Nadam/Nadam/update/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_1/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_10/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_11/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_12/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_13/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_14/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_15/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_16/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_17/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_18/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_19/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_2/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_20/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_21/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_22/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_23/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_24/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_25/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_26/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_27/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_28/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_29/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_3/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_30/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_31/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_32/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_33/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_34/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_35/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_36/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_37/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_38/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_39/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_4/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_40/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_41/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_5/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_6/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_7/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_8/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_9/AssignVariableOp_2"
  device: "/job:localhost/replica:0/task:0/device:GPU:0"
}
node {
  name: "Nadam/Nadam/Const"
  op: "Const"
  input: "^Nadam/Nadam/group_deps"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_INT64
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_INT64
        tensor_shape {
        }
        int64_val: 1
      }
    }
  }
}
node {
  name: "Nadam/Nadam/AssignAddVariableOp"
  op: "AssignAddVariableOp"
  input: "Nadam/ReadVariableOp/resource"
  input: "Nadam/Nadam/Const"
  input: "^Nadam/ReadVariableOp"
  input: "^Nadam/ReadVariableOp_1"
  attr {
    key: "dtype"
    value {
      type: DT_INT64
    }
  }
}
node {
  name: "Cast_1"
  op: "Cast"
  input: "IteratorGetNext:1"
  attr {
    key: "DstT"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "SrcT"
    value {
      type: DT_UINT8
    }
  }
  attr {
    key: "Truncate"
    value {
      b: false
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
          dim {
            size: 1
          }
        }
      }
    }
  }
}
node {
  name: "Squeeze"
  op: "Squeeze"
  input: "Cast_1"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
        }
      }
    }
  }
  attr {
    key: "squeeze_dims"
    value {
      list {
        i: -1
      }
    }
  }
}
node {
  name: "ArgMax/dimension"
  op: "Const"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_INT32
        tensor_shape {
        }
        int_val: -1
      }
    }
  }
}
node {
  name: "ArgMax"
  op: "ArgMax"
  input: "sequential/dense_20/Softmax"
  input: "ArgMax/dimension"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "Tidx"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
        }
      }
    }
  }
  attr {
    key: "output_type"
    value {
      type: DT_INT64
    }
  }
}
node {
  name: "Cast_2"
  op: "Cast"
  input: "ArgMax"
  attr {
    key: "DstT"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "SrcT"
    value {
      type: DT_INT64
    }
  }
  attr {
    key: "Truncate"
    value {
      b: false
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
        }
      }
    }
  }
}
node {
  name: "Equal"
  op: "Equal"
  input: "Squeeze"
  input: "Cast_2"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
        }
      }
    }
  }
  attr {
    key: "incompatible_shape_error"
    value {
      b: true
    }
  }
}
node {
  name: "Cast_3"
  op: "Cast"
  input: "Equal"
  attr {
    key: "DstT"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "SrcT"
    value {
      type: DT_BOOL
    }
  }
  attr {
    key: "Truncate"
    value {
      b: false
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: -1
          }
        }
      }
    }
  }
}
node {
  name: "Const"
  op: "Const"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
          dim {
            size: 1
          }
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_INT32
        tensor_shape {
          dim {
            size: 1
          }
        }
        int_val: 0
      }
    }
  }
}
node {
  name: "Sum_2"
  op: "Sum"
  input: "Cast_3"
  input: "Const"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "Tidx"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "keep_dims"
    value {
      b: false
    }
  }
}
node {
  name: "AssignAddVariableOp_2/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "AssignAddVariableOp_2"
  op: "AssignAddVariableOp"
  input: "AssignAddVariableOp_2/resource"
  input: "Sum_2"
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "Size"
  op: "Size"
  input: "Cast_3"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "out_type"
    value {
      type: DT_INT32
    }
  }
}
node {
  name: "Cast_4"
  op: "Cast"
  input: "Size"
  attr {
    key: "DstT"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "SrcT"
    value {
      type: DT_INT32
    }
  }
  attr {
    key: "Truncate"
    value {
      b: false
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "AssignAddVariableOp_3/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "AssignAddVariableOp_3"
  op: "AssignAddVariableOp"
  input: "AssignAddVariableOp_3/resource"
  input: "Cast_4"
  input: "^AssignAddVariableOp_2"
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "div_no_nan/ReadVariableOp"
  op: "ReadVariableOp"
  input: "AssignAddVariableOp/resource"
  input: "^AssignAddVariableOp"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "div_no_nan/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "AssignAddVariableOp_1/resource"
  input: "^AssignAddVariableOp_1"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "div_no_nan"
  op: "DivNoNan"
  input: "div_no_nan/ReadVariableOp"
  input: "div_no_nan/ReadVariableOp_1"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "Identity"
  op: "Identity"
  input: "div_no_nan"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "div_no_nan_1/ReadVariableOp"
  op: "ReadVariableOp"
  input: "AssignAddVariableOp_2/resource"
  input: "^AssignAddVariableOp_2"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "div_no_nan_1/ReadVariableOp_1"
  op: "ReadVariableOp"
  input: "AssignAddVariableOp_3/resource"
  input: "^AssignAddVariableOp_3"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_FLOAT
    }
  }
}
node {
  name: "div_no_nan_1"
  op: "DivNoNan"
  input: "div_no_nan_1/ReadVariableOp"
  input: "div_no_nan_1/ReadVariableOp_1"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "Identity_1"
  op: "Identity"
  input: "div_no_nan_1"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "Const_1"
  op: "Const"
  input: "^Identity_1"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_INT64
    }
  }
  attr {
    key: "value"
    value {
      tensor {
        dtype: DT_INT64
        tensor_shape {
        }
        int64_val: 1
      }
    }
  }
}
node {
  name: "AssignAddVariableOp_4/resource"
  op: "Placeholder"
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
  attr {
    key: "dtype"
    value {
      type: DT_RESOURCE
    }
  }
  attr {
    key: "shape"
    value {
      shape {
      }
    }
  }
}
node {
  name: "AssignAddVariableOp_4"
  op: "AssignAddVariableOp"
  input: "AssignAddVariableOp_4/resource"
  input: "Const_1"
  attr {
    key: "dtype"
    value {
      type: DT_INT64
    }
  }
}
node {
  name: "NoOp"
  op: "NoOp"
}
node {
  name: "NoOp_1"
  op: "NoOp"
}
node {
  name: "Identity_2"
  op: "Identity"
  input: "Identity_1"
  input: "^AssignAddVariableOp"
  input: "^AssignAddVariableOp_1"
  input: "^AssignAddVariableOp_2"
  input: "^AssignAddVariableOp_3"
  input: "^AssignAddVariableOp_4"
  input: "^IteratorGetNext"
  input: "^Nadam/AssignVariableOp"
  input: "^Nadam/Cast/ReadVariableOp"
  input: "^Nadam/Cast_1/ReadVariableOp"
  input: "^Nadam/Cast_2/ReadVariableOp"
  input: "^Nadam/Identity/ReadVariableOp"
  input: "^Nadam/Identity_4/ReadVariableOp"
  input: "^Nadam/Nadam/AssignAddVariableOp"
  input: "^Nadam/Nadam/update/AssignVariableOp"
  input: "^Nadam/Nadam/update/AssignVariableOp_1"
  input: "^Nadam/Nadam/update/AssignVariableOp_2"
  input: "^Nadam/Nadam/update/ReadVariableOp"
  input: "^Nadam/Nadam/update/ReadVariableOp_1"
  input: "^Nadam/Nadam/update/ReadVariableOp_2"
  input: "^Nadam/Nadam/update/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_1/AssignVariableOp"
  input: "^Nadam/Nadam/update_1/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_1/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_1/ReadVariableOp"
  input: "^Nadam/Nadam/update_1/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_1/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_1/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_1/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_10/AssignVariableOp"
  input: "^Nadam/Nadam/update_10/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_10/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_10/ReadVariableOp"
  input: "^Nadam/Nadam/update_10/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_10/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_10/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_10/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_11/AssignVariableOp"
  input: "^Nadam/Nadam/update_11/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_11/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_11/ReadVariableOp"
  input: "^Nadam/Nadam/update_11/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_11/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_11/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_11/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_12/AssignVariableOp"
  input: "^Nadam/Nadam/update_12/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_12/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_12/ReadVariableOp"
  input: "^Nadam/Nadam/update_12/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_12/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_12/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_12/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_13/AssignVariableOp"
  input: "^Nadam/Nadam/update_13/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_13/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_13/ReadVariableOp"
  input: "^Nadam/Nadam/update_13/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_13/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_13/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_13/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_14/AssignVariableOp"
  input: "^Nadam/Nadam/update_14/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_14/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_14/ReadVariableOp"
  input: "^Nadam/Nadam/update_14/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_14/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_14/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_14/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_15/AssignVariableOp"
  input: "^Nadam/Nadam/update_15/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_15/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_15/ReadVariableOp"
  input: "^Nadam/Nadam/update_15/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_15/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_15/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_15/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_16/AssignVariableOp"
  input: "^Nadam/Nadam/update_16/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_16/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_16/ReadVariableOp"
  input: "^Nadam/Nadam/update_16/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_16/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_16/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_16/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_17/AssignVariableOp"
  input: "^Nadam/Nadam/update_17/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_17/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_17/ReadVariableOp"
  input: "^Nadam/Nadam/update_17/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_17/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_17/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_17/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_18/AssignVariableOp"
  input: "^Nadam/Nadam/update_18/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_18/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_18/ReadVariableOp"
  input: "^Nadam/Nadam/update_18/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_18/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_18/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_18/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_19/AssignVariableOp"
  input: "^Nadam/Nadam/update_19/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_19/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_19/ReadVariableOp"
  input: "^Nadam/Nadam/update_19/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_19/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_19/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_19/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_2/AssignVariableOp"
  input: "^Nadam/Nadam/update_2/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_2/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_2/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_2/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_2/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_2/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_20/AssignVariableOp"
  input: "^Nadam/Nadam/update_20/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_20/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_20/ReadVariableOp"
  input: "^Nadam/Nadam/update_20/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_20/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_20/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_20/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_21/AssignVariableOp"
  input: "^Nadam/Nadam/update_21/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_21/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_21/ReadVariableOp"
  input: "^Nadam/Nadam/update_21/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_21/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_21/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_21/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_22/AssignVariableOp"
  input: "^Nadam/Nadam/update_22/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_22/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_22/ReadVariableOp"
  input: "^Nadam/Nadam/update_22/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_22/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_22/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_22/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_23/AssignVariableOp"
  input: "^Nadam/Nadam/update_23/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_23/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_23/ReadVariableOp"
  input: "^Nadam/Nadam/update_23/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_23/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_23/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_23/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_24/AssignVariableOp"
  input: "^Nadam/Nadam/update_24/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_24/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_24/ReadVariableOp"
  input: "^Nadam/Nadam/update_24/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_24/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_24/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_24/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_25/AssignVariableOp"
  input: "^Nadam/Nadam/update_25/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_25/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_25/ReadVariableOp"
  input: "^Nadam/Nadam/update_25/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_25/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_25/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_25/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_26/AssignVariableOp"
  input: "^Nadam/Nadam/update_26/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_26/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_26/ReadVariableOp"
  input: "^Nadam/Nadam/update_26/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_26/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_26/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_26/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_27/AssignVariableOp"
  input: "^Nadam/Nadam/update_27/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_27/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_27/ReadVariableOp"
  input: "^Nadam/Nadam/update_27/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_27/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_27/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_27/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_28/AssignVariableOp"
  input: "^Nadam/Nadam/update_28/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_28/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_28/ReadVariableOp"
  input: "^Nadam/Nadam/update_28/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_28/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_28/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_28/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_29/AssignVariableOp"
  input: "^Nadam/Nadam/update_29/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_29/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_29/ReadVariableOp"
  input: "^Nadam/Nadam/update_29/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_29/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_29/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_29/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_3/AssignVariableOp"
  input: "^Nadam/Nadam/update_3/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_3/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_3/ReadVariableOp"
  input: "^Nadam/Nadam/update_3/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_3/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_3/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_3/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_30/AssignVariableOp"
  input: "^Nadam/Nadam/update_30/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_30/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_30/ReadVariableOp"
  input: "^Nadam/Nadam/update_30/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_30/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_30/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_30/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_31/AssignVariableOp"
  input: "^Nadam/Nadam/update_31/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_31/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_31/ReadVariableOp"
  input: "^Nadam/Nadam/update_31/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_31/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_31/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_31/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_32/AssignVariableOp"
  input: "^Nadam/Nadam/update_32/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_32/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_32/ReadVariableOp"
  input: "^Nadam/Nadam/update_32/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_32/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_32/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_32/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_33/AssignVariableOp"
  input: "^Nadam/Nadam/update_33/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_33/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_33/ReadVariableOp"
  input: "^Nadam/Nadam/update_33/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_33/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_33/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_33/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_34/AssignVariableOp"
  input: "^Nadam/Nadam/update_34/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_34/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_34/ReadVariableOp"
  input: "^Nadam/Nadam/update_34/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_34/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_34/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_34/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_35/AssignVariableOp"
  input: "^Nadam/Nadam/update_35/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_35/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_35/ReadVariableOp"
  input: "^Nadam/Nadam/update_35/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_35/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_35/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_35/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_36/AssignVariableOp"
  input: "^Nadam/Nadam/update_36/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_36/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_36/ReadVariableOp"
  input: "^Nadam/Nadam/update_36/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_36/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_36/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_36/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_37/AssignVariableOp"
  input: "^Nadam/Nadam/update_37/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_37/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_37/ReadVariableOp"
  input: "^Nadam/Nadam/update_37/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_37/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_37/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_37/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_38/AssignVariableOp"
  input: "^Nadam/Nadam/update_38/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_38/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_38/ReadVariableOp"
  input: "^Nadam/Nadam/update_38/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_38/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_38/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_38/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_39/AssignVariableOp"
  input: "^Nadam/Nadam/update_39/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_39/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_39/ReadVariableOp"
  input: "^Nadam/Nadam/update_39/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_39/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_39/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_39/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_4/AssignVariableOp"
  input: "^Nadam/Nadam/update_4/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_4/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_4/ReadVariableOp"
  input: "^Nadam/Nadam/update_4/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_4/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_4/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_4/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_40/AssignVariableOp"
  input: "^Nadam/Nadam/update_40/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_40/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_40/ReadVariableOp"
  input: "^Nadam/Nadam/update_40/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_40/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_40/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_40/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_41/AssignVariableOp"
  input: "^Nadam/Nadam/update_41/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_41/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_41/ReadVariableOp"
  input: "^Nadam/Nadam/update_41/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_41/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_41/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_41/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_5/AssignVariableOp"
  input: "^Nadam/Nadam/update_5/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_5/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_5/ReadVariableOp"
  input: "^Nadam/Nadam/update_5/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_5/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_5/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_5/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_6/AssignVariableOp"
  input: "^Nadam/Nadam/update_6/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_6/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_6/ReadVariableOp"
  input: "^Nadam/Nadam/update_6/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_6/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_6/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_6/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_7/AssignVariableOp"
  input: "^Nadam/Nadam/update_7/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_7/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_7/ReadVariableOp"
  input: "^Nadam/Nadam/update_7/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_7/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_7/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_7/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_8/AssignVariableOp"
  input: "^Nadam/Nadam/update_8/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_8/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_8/ReadVariableOp"
  input: "^Nadam/Nadam/update_8/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_8/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_8/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_8/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_9/AssignVariableOp"
  input: "^Nadam/Nadam/update_9/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_9/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_9/ReadVariableOp"
  input: "^Nadam/Nadam/update_9/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_9/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_9/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_9/mul_2/ReadVariableOp"
  input: "^Nadam/ReadVariableOp"
  input: "^Nadam/ReadVariableOp_1"
  input: "^div_no_nan/ReadVariableOp"
  input: "^div_no_nan/ReadVariableOp_1"
  input: "^div_no_nan_1/ReadVariableOp"
  input: "^div_no_nan_1/ReadVariableOp_1"
  input: "^sequential/dense/BiasAdd/ReadVariableOp"
  input: "^sequential/dense/MatMul/ReadVariableOp"
  input: "^sequential/dense_1/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_1/MatMul/ReadVariableOp"
  input: "^sequential/dense_10/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_10/MatMul/ReadVariableOp"
  input: "^sequential/dense_11/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_11/MatMul/ReadVariableOp"
  input: "^sequential/dense_12/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_12/MatMul/ReadVariableOp"
  input: "^sequential/dense_13/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_13/MatMul/ReadVariableOp"
  input: "^sequential/dense_14/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_14/MatMul/ReadVariableOp"
  input: "^sequential/dense_15/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_15/MatMul/ReadVariableOp"
  input: "^sequential/dense_16/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_16/MatMul/ReadVariableOp"
  input: "^sequential/dense_17/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_17/MatMul/ReadVariableOp"
  input: "^sequential/dense_18/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_18/MatMul/ReadVariableOp"
  input: "^sequential/dense_19/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_19/MatMul/ReadVariableOp"
  input: "^sequential/dense_2/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_2/MatMul/ReadVariableOp"
  input: "^sequential/dense_20/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_20/MatMul/ReadVariableOp"
  input: "^sequential/dense_3/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_3/MatMul/ReadVariableOp"
  input: "^sequential/dense_4/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_4/MatMul/ReadVariableOp"
  input: "^sequential/dense_5/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_5/MatMul/ReadVariableOp"
  input: "^sequential/dense_6/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_6/MatMul/ReadVariableOp"
  input: "^sequential/dense_7/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_7/MatMul/ReadVariableOp"
  input: "^sequential/dense_8/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_8/MatMul/ReadVariableOp"
  input: "^sequential/dense_9/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_9/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
node {
  name: "Identity_3"
  op: "Identity"
  input: "Identity"
  input: "^AssignAddVariableOp"
  input: "^AssignAddVariableOp_1"
  input: "^AssignAddVariableOp_2"
  input: "^AssignAddVariableOp_3"
  input: "^AssignAddVariableOp_4"
  input: "^IteratorGetNext"
  input: "^Nadam/AssignVariableOp"
  input: "^Nadam/Cast/ReadVariableOp"
  input: "^Nadam/Cast_1/ReadVariableOp"
  input: "^Nadam/Cast_2/ReadVariableOp"
  input: "^Nadam/Identity/ReadVariableOp"
  input: "^Nadam/Identity_4/ReadVariableOp"
  input: "^Nadam/Nadam/AssignAddVariableOp"
  input: "^Nadam/Nadam/update/AssignVariableOp"
  input: "^Nadam/Nadam/update/AssignVariableOp_1"
  input: "^Nadam/Nadam/update/AssignVariableOp_2"
  input: "^Nadam/Nadam/update/ReadVariableOp"
  input: "^Nadam/Nadam/update/ReadVariableOp_1"
  input: "^Nadam/Nadam/update/ReadVariableOp_2"
  input: "^Nadam/Nadam/update/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_1/AssignVariableOp"
  input: "^Nadam/Nadam/update_1/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_1/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_1/ReadVariableOp"
  input: "^Nadam/Nadam/update_1/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_1/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_1/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_1/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_10/AssignVariableOp"
  input: "^Nadam/Nadam/update_10/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_10/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_10/ReadVariableOp"
  input: "^Nadam/Nadam/update_10/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_10/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_10/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_10/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_11/AssignVariableOp"
  input: "^Nadam/Nadam/update_11/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_11/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_11/ReadVariableOp"
  input: "^Nadam/Nadam/update_11/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_11/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_11/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_11/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_12/AssignVariableOp"
  input: "^Nadam/Nadam/update_12/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_12/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_12/ReadVariableOp"
  input: "^Nadam/Nadam/update_12/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_12/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_12/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_12/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_13/AssignVariableOp"
  input: "^Nadam/Nadam/update_13/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_13/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_13/ReadVariableOp"
  input: "^Nadam/Nadam/update_13/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_13/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_13/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_13/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_14/AssignVariableOp"
  input: "^Nadam/Nadam/update_14/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_14/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_14/ReadVariableOp"
  input: "^Nadam/Nadam/update_14/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_14/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_14/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_14/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_15/AssignVariableOp"
  input: "^Nadam/Nadam/update_15/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_15/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_15/ReadVariableOp"
  input: "^Nadam/Nadam/update_15/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_15/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_15/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_15/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_16/AssignVariableOp"
  input: "^Nadam/Nadam/update_16/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_16/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_16/ReadVariableOp"
  input: "^Nadam/Nadam/update_16/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_16/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_16/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_16/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_17/AssignVariableOp"
  input: "^Nadam/Nadam/update_17/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_17/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_17/ReadVariableOp"
  input: "^Nadam/Nadam/update_17/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_17/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_17/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_17/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_18/AssignVariableOp"
  input: "^Nadam/Nadam/update_18/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_18/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_18/ReadVariableOp"
  input: "^Nadam/Nadam/update_18/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_18/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_18/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_18/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_19/AssignVariableOp"
  input: "^Nadam/Nadam/update_19/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_19/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_19/ReadVariableOp"
  input: "^Nadam/Nadam/update_19/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_19/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_19/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_19/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_2/AssignVariableOp"
  input: "^Nadam/Nadam/update_2/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_2/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_2/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_2/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_2/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_2/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_20/AssignVariableOp"
  input: "^Nadam/Nadam/update_20/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_20/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_20/ReadVariableOp"
  input: "^Nadam/Nadam/update_20/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_20/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_20/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_20/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_21/AssignVariableOp"
  input: "^Nadam/Nadam/update_21/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_21/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_21/ReadVariableOp"
  input: "^Nadam/Nadam/update_21/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_21/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_21/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_21/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_22/AssignVariableOp"
  input: "^Nadam/Nadam/update_22/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_22/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_22/ReadVariableOp"
  input: "^Nadam/Nadam/update_22/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_22/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_22/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_22/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_23/AssignVariableOp"
  input: "^Nadam/Nadam/update_23/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_23/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_23/ReadVariableOp"
  input: "^Nadam/Nadam/update_23/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_23/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_23/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_23/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_24/AssignVariableOp"
  input: "^Nadam/Nadam/update_24/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_24/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_24/ReadVariableOp"
  input: "^Nadam/Nadam/update_24/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_24/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_24/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_24/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_25/AssignVariableOp"
  input: "^Nadam/Nadam/update_25/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_25/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_25/ReadVariableOp"
  input: "^Nadam/Nadam/update_25/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_25/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_25/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_25/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_26/AssignVariableOp"
  input: "^Nadam/Nadam/update_26/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_26/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_26/ReadVariableOp"
  input: "^Nadam/Nadam/update_26/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_26/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_26/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_26/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_27/AssignVariableOp"
  input: "^Nadam/Nadam/update_27/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_27/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_27/ReadVariableOp"
  input: "^Nadam/Nadam/update_27/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_27/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_27/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_27/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_28/AssignVariableOp"
  input: "^Nadam/Nadam/update_28/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_28/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_28/ReadVariableOp"
  input: "^Nadam/Nadam/update_28/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_28/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_28/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_28/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_29/AssignVariableOp"
  input: "^Nadam/Nadam/update_29/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_29/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_29/ReadVariableOp"
  input: "^Nadam/Nadam/update_29/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_29/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_29/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_29/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_3/AssignVariableOp"
  input: "^Nadam/Nadam/update_3/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_3/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_3/ReadVariableOp"
  input: "^Nadam/Nadam/update_3/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_3/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_3/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_3/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_30/AssignVariableOp"
  input: "^Nadam/Nadam/update_30/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_30/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_30/ReadVariableOp"
  input: "^Nadam/Nadam/update_30/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_30/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_30/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_30/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_31/AssignVariableOp"
  input: "^Nadam/Nadam/update_31/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_31/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_31/ReadVariableOp"
  input: "^Nadam/Nadam/update_31/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_31/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_31/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_31/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_32/AssignVariableOp"
  input: "^Nadam/Nadam/update_32/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_32/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_32/ReadVariableOp"
  input: "^Nadam/Nadam/update_32/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_32/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_32/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_32/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_33/AssignVariableOp"
  input: "^Nadam/Nadam/update_33/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_33/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_33/ReadVariableOp"
  input: "^Nadam/Nadam/update_33/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_33/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_33/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_33/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_34/AssignVariableOp"
  input: "^Nadam/Nadam/update_34/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_34/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_34/ReadVariableOp"
  input: "^Nadam/Nadam/update_34/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_34/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_34/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_34/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_35/AssignVariableOp"
  input: "^Nadam/Nadam/update_35/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_35/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_35/ReadVariableOp"
  input: "^Nadam/Nadam/update_35/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_35/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_35/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_35/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_36/AssignVariableOp"
  input: "^Nadam/Nadam/update_36/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_36/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_36/ReadVariableOp"
  input: "^Nadam/Nadam/update_36/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_36/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_36/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_36/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_37/AssignVariableOp"
  input: "^Nadam/Nadam/update_37/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_37/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_37/ReadVariableOp"
  input: "^Nadam/Nadam/update_37/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_37/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_37/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_37/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_38/AssignVariableOp"
  input: "^Nadam/Nadam/update_38/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_38/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_38/ReadVariableOp"
  input: "^Nadam/Nadam/update_38/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_38/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_38/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_38/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_39/AssignVariableOp"
  input: "^Nadam/Nadam/update_39/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_39/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_39/ReadVariableOp"
  input: "^Nadam/Nadam/update_39/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_39/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_39/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_39/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_4/AssignVariableOp"
  input: "^Nadam/Nadam/update_4/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_4/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_4/ReadVariableOp"
  input: "^Nadam/Nadam/update_4/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_4/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_4/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_4/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_40/AssignVariableOp"
  input: "^Nadam/Nadam/update_40/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_40/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_40/ReadVariableOp"
  input: "^Nadam/Nadam/update_40/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_40/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_40/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_40/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_41/AssignVariableOp"
  input: "^Nadam/Nadam/update_41/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_41/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_41/ReadVariableOp"
  input: "^Nadam/Nadam/update_41/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_41/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_41/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_41/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_5/AssignVariableOp"
  input: "^Nadam/Nadam/update_5/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_5/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_5/ReadVariableOp"
  input: "^Nadam/Nadam/update_5/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_5/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_5/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_5/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_6/AssignVariableOp"
  input: "^Nadam/Nadam/update_6/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_6/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_6/ReadVariableOp"
  input: "^Nadam/Nadam/update_6/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_6/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_6/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_6/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_7/AssignVariableOp"
  input: "^Nadam/Nadam/update_7/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_7/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_7/ReadVariableOp"
  input: "^Nadam/Nadam/update_7/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_7/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_7/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_7/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_8/AssignVariableOp"
  input: "^Nadam/Nadam/update_8/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_8/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_8/ReadVariableOp"
  input: "^Nadam/Nadam/update_8/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_8/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_8/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_8/mul_2/ReadVariableOp"
  input: "^Nadam/Nadam/update_9/AssignVariableOp"
  input: "^Nadam/Nadam/update_9/AssignVariableOp_1"
  input: "^Nadam/Nadam/update_9/AssignVariableOp_2"
  input: "^Nadam/Nadam/update_9/ReadVariableOp"
  input: "^Nadam/Nadam/update_9/ReadVariableOp_1"
  input: "^Nadam/Nadam/update_9/ReadVariableOp_2"
  input: "^Nadam/Nadam/update_9/mul/ReadVariableOp"
  input: "^Nadam/Nadam/update_9/mul_2/ReadVariableOp"
  input: "^Nadam/ReadVariableOp"
  input: "^Nadam/ReadVariableOp_1"
  input: "^div_no_nan/ReadVariableOp"
  input: "^div_no_nan/ReadVariableOp_1"
  input: "^div_no_nan_1/ReadVariableOp"
  input: "^div_no_nan_1/ReadVariableOp_1"
  input: "^sequential/dense/BiasAdd/ReadVariableOp"
  input: "^sequential/dense/MatMul/ReadVariableOp"
  input: "^sequential/dense_1/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_1/MatMul/ReadVariableOp"
  input: "^sequential/dense_10/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_10/MatMul/ReadVariableOp"
  input: "^sequential/dense_11/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_11/MatMul/ReadVariableOp"
  input: "^sequential/dense_12/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_12/MatMul/ReadVariableOp"
  input: "^sequential/dense_13/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_13/MatMul/ReadVariableOp"
  input: "^sequential/dense_14/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_14/MatMul/ReadVariableOp"
  input: "^sequential/dense_15/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_15/MatMul/ReadVariableOp"
  input: "^sequential/dense_16/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_16/MatMul/ReadVariableOp"
  input: "^sequential/dense_17/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_17/MatMul/ReadVariableOp"
  input: "^sequential/dense_18/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_18/MatMul/ReadVariableOp"
  input: "^sequential/dense_19/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_19/MatMul/ReadVariableOp"
  input: "^sequential/dense_2/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_2/MatMul/ReadVariableOp"
  input: "^sequential/dense_20/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_20/MatMul/ReadVariableOp"
  input: "^sequential/dense_3/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_3/MatMul/ReadVariableOp"
  input: "^sequential/dense_4/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_4/MatMul/ReadVariableOp"
  input: "^sequential/dense_5/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_5/MatMul/ReadVariableOp"
  input: "^sequential/dense_6/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_6/MatMul/ReadVariableOp"
  input: "^sequential/dense_7/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_7/MatMul/ReadVariableOp"
  input: "^sequential/dense_8/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_8/MatMul/ReadVariableOp"
  input: "^sequential/dense_9/BiasAdd/ReadVariableOp"
  input: "^sequential/dense_9/MatMul/ReadVariableOp"
  attr {
    key: "T"
    value {
      type: DT_FLOAT
    }
  }
  attr {
    key: "_output_shapes"
    value {
      list {
        shape {
        }
      }
    }
  }
}
versions {
  producer: 561
}
", + "ok": true, + "headers": [ + [ + "content-type", + "text/x-protobuf; charset=utf-8" + ] + ], + "status": 200, + "status_text": "" + }, + "https://localhost:6006/font-roboto/RxZJdnzeo3R5zSexge8UUZBw1xU1rKptJj_0jans920.woff2": { + "data": "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", + "ok": true, + "headers": [ + [ + "content-type", + "font/woff2" + ] + ], + "status": 200, + "status_text": "" + }, + "https://localhost:6006/font-roboto/oMMgfZMQthOryQo9n22dcuvvDin1pK8aKteLpeZ5c0A.woff2": { + "data": "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", + "ok": true, + "headers": [ + [ + "content-type", + "font/woff2" + ] + ], + "status": 200, + "status_text": "" + } + }, + "base_uri": "https://localhost:8080/", + "height": 822 + } + }, + "source": [ + "%tensorboard --logdir=./my_cifar10_logs --port=6006" + ], + "execution_count": 132, + "outputs": [ + { + "output_type": "display_data", + "data": { + "application/javascript": [ + "\n", + " (async () => {\n", + " const url = new URL(await google.colab.kernel.proxyPort(6006, {'cache': true}));\n", + " url.searchParams.set('tensorboardColab', 'true');\n", + " const iframe = document.createElement('iframe');\n", + " iframe.src = url;\n", + " iframe.setAttribute('width', '100%');\n", + " iframe.setAttribute('height', '800');\n", + " iframe.setAttribute('frameborder', 0);\n", + " document.body.appendChild(iframe);\n", + " })();\n", + " " + ], + "text/plain": [ + "" + ] + }, + "metadata": { + "tags": [] + } + } + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "4huFUfvOLOpl", + "outputId": "a0f6fcfc-c02b-4861-b396-d64f7281184d", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "model.fit(X_train, y_train, epochs=100,\n", + " validation_data=(X_valid, y_valid),\n", + " callbacks=callbacks)" + ], + "execution_count": 133, + "outputs": [ + { + "output_type": "stream", + "text": [ + "Epoch 1/100\n", + "1407/1407 [==============================] - 16s 9ms/step - loss: 9.5976 - accuracy: 0.1365 - val_loss: 2.1086 - val_accuracy: 0.2342\n", + "Epoch 2/100\n", + "1407/1407 [==============================] - 12s 9ms/step - loss: 2.0597 - accuracy: 0.2457 - val_loss: 2.0227 - val_accuracy: 0.2528\n", + "Epoch 3/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.9423 - accuracy: 0.2883 - val_loss: 1.9173 - val_accuracy: 0.2974\n", + "Epoch 4/100\n", + "1407/1407 [==============================] - 12s 9ms/step - loss: 1.8595 - accuracy: 0.3242 - val_loss: 1.9241 - val_accuracy: 0.3242\n", + "Epoch 5/100\n", + "1407/1407 [==============================] - 12s 9ms/step - loss: 1.7853 - accuracy: 0.3508 - val_loss: 1.7868 - val_accuracy: 0.3550\n", + "Epoch 6/100\n", + "1407/1407 [==============================] - 12s 9ms/step - loss: 1.7293 - accuracy: 0.3703 - val_loss: 1.7313 - val_accuracy: 0.3782\n", + "Epoch 7/100\n", + "1407/1407 [==============================] - 12s 9ms/step - loss: 1.6884 - accuracy: 0.3880 - val_loss: 1.7046 - val_accuracy: 0.3728\n", + "Epoch 8/100\n", + "1407/1407 [==============================] - 12s 9ms/step - loss: 1.6473 - accuracy: 0.4009 - val_loss: 1.6535 - val_accuracy: 0.4072\n", + "Epoch 9/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.6216 - accuracy: 0.4174 - val_loss: 1.6390 - val_accuracy: 0.4072\n", + "Epoch 10/100\n", + "1407/1407 [==============================] - 12s 9ms/step - loss: 1.5892 - accuracy: 0.4274 - val_loss: 1.6882 - val_accuracy: 0.3892\n", + "Epoch 11/100\n", + "1407/1407 [==============================] - 12s 9ms/step - loss: 1.5679 - accuracy: 0.4388 - val_loss: 1.6097 - val_accuracy: 0.4182\n", + "Epoch 12/100\n", + "1407/1407 [==============================] - 12s 9ms/step - loss: 1.5494 - accuracy: 0.4435 - val_loss: 1.6076 - val_accuracy: 0.4224\n", + "Epoch 13/100\n", + "1407/1407 [==============================] - 12s 9ms/step - loss: 1.5264 - accuracy: 0.4501 - val_loss: 1.6094 - val_accuracy: 0.4174\n", + "Epoch 14/100\n", + "1407/1407 [==============================] - 12s 9ms/step - loss: 1.5058 - accuracy: 0.4584 - val_loss: 1.5780 - val_accuracy: 0.4304\n", + "Epoch 15/100\n", + "1407/1407 [==============================] - 12s 9ms/step - loss: 1.5017 - accuracy: 0.4586 - val_loss: 1.5628 - val_accuracy: 0.4468\n", + "Epoch 16/100\n", + "1407/1407 [==============================] - 12s 9ms/step - loss: 1.4832 - accuracy: 0.4655 - val_loss: 1.5483 - val_accuracy: 0.4450\n", + "Epoch 17/100\n", + "1407/1407 [==============================] - 12s 9ms/step - loss: 1.4690 - accuracy: 0.4741 - val_loss: 1.5686 - val_accuracy: 0.4398\n", + "Epoch 18/100\n", + "1407/1407 [==============================] - 12s 9ms/step - loss: 1.4530 - accuracy: 0.4814 - val_loss: 1.5176 - val_accuracy: 0.4588\n", + "Epoch 19/100\n", + "1407/1407 [==============================] - 12s 9ms/step - loss: 1.4254 - accuracy: 0.4887 - val_loss: 1.5407 - val_accuracy: 0.4556\n", + "Epoch 20/100\n", + "1407/1407 [==============================] - 12s 9ms/step - loss: 1.4167 - accuracy: 0.4908 - val_loss: 1.5124 - val_accuracy: 0.4598\n", + "Epoch 21/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.4132 - accuracy: 0.4940 - val_loss: 1.5697 - val_accuracy: 0.4456\n", + "Epoch 22/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.4092 - accuracy: 0.4909 - val_loss: 1.5215 - val_accuracy: 0.4550\n", + "Epoch 23/100\n", + "1407/1407 [==============================] - 12s 9ms/step - loss: 1.3812 - accuracy: 0.5018 - val_loss: 1.5259 - val_accuracy: 0.4558\n", + "Epoch 24/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.3708 - accuracy: 0.5068 - val_loss: 1.5368 - val_accuracy: 0.4616\n", + "Epoch 25/100\n", + "1407/1407 [==============================] - 12s 9ms/step - loss: 1.3730 - accuracy: 0.5061 - val_loss: 1.5012 - val_accuracy: 0.4656\n", + "Epoch 26/100\n", + "1407/1407 [==============================] - 12s 9ms/step - loss: 1.3413 - accuracy: 0.5160 - val_loss: 1.5261 - val_accuracy: 0.4538\n", + "Epoch 27/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.3396 - accuracy: 0.5184 - val_loss: 1.5085 - val_accuracy: 0.4638\n", + "Epoch 28/100\n", + "1407/1407 [==============================] - 12s 9ms/step - loss: 1.3297 - accuracy: 0.5219 - val_loss: 1.5276 - val_accuracy: 0.4604\n", + "Epoch 29/100\n", + "1407/1407 [==============================] - 12s 9ms/step - loss: 1.3327 - accuracy: 0.5197 - val_loss: 1.5129 - val_accuracy: 0.4642\n", + "Epoch 30/100\n", + "1407/1407 [==============================] - 12s 9ms/step - loss: 1.3217 - accuracy: 0.5230 - val_loss: 1.5549 - val_accuracy: 0.4634\n", + "Epoch 31/100\n", + "1407/1407 [==============================] - 12s 9ms/step - loss: 1.3237 - accuracy: 0.5222 - val_loss: 1.5363 - val_accuracy: 0.4582\n", + "Epoch 32/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.2889 - accuracy: 0.5341 - val_loss: 1.4979 - val_accuracy: 0.4766\n", + "Epoch 33/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.2933 - accuracy: 0.5362 - val_loss: 1.5370 - val_accuracy: 0.4622\n", + "Epoch 34/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.2736 - accuracy: 0.5390 - val_loss: 1.5247 - val_accuracy: 0.4698\n", + "Epoch 35/100\n", + "1407/1407 [==============================] - 12s 9ms/step - loss: 1.2719 - accuracy: 0.5433 - val_loss: 1.4904 - val_accuracy: 0.4768\n", + "Epoch 36/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.2673 - accuracy: 0.5467 - val_loss: 1.5016 - val_accuracy: 0.4780\n", + "Epoch 37/100\n", + "1407/1407 [==============================] - 12s 9ms/step - loss: 1.2518 - accuracy: 0.5478 - val_loss: 1.4951 - val_accuracy: 0.4810\n", + "Epoch 38/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.2393 - accuracy: 0.5572 - val_loss: 1.4962 - val_accuracy: 0.4770\n", + "Epoch 39/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.2372 - accuracy: 0.5526 - val_loss: 1.5172 - val_accuracy: 0.4732\n", + "Epoch 40/100\n", + "1407/1407 [==============================] - 12s 9ms/step - loss: 1.2280 - accuracy: 0.5606 - val_loss: 1.5011 - val_accuracy: 0.4768\n", + "Epoch 41/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.2190 - accuracy: 0.5625 - val_loss: 1.5301 - val_accuracy: 0.4740\n", + "Epoch 42/100\n", + "1407/1407 [==============================] - 12s 9ms/step - loss: 1.2113 - accuracy: 0.5695 - val_loss: 1.5344 - val_accuracy: 0.4746\n", + "Epoch 43/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.2086 - accuracy: 0.5617 - val_loss: 1.5503 - val_accuracy: 0.4638\n", + "Epoch 44/100\n", + "1407/1407 [==============================] - 12s 9ms/step - loss: 1.1997 - accuracy: 0.5712 - val_loss: 1.5310 - val_accuracy: 0.4814\n", + "Epoch 45/100\n", + "1407/1407 [==============================] - 12s 9ms/step - loss: 1.1881 - accuracy: 0.5744 - val_loss: 1.4936 - val_accuracy: 0.4852\n", + "Epoch 46/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.1747 - accuracy: 0.5788 - val_loss: 1.5113 - val_accuracy: 0.4778\n", + "Epoch 47/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.1738 - accuracy: 0.5753 - val_loss: 1.5495 - val_accuracy: 0.4750\n", + "Epoch 48/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.1679 - accuracy: 0.5796 - val_loss: 1.4970 - val_accuracy: 0.4878\n", + "Epoch 49/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.1723 - accuracy: 0.5763 - val_loss: 1.5314 - val_accuracy: 0.4812\n", + "Epoch 50/100\n", + "1407/1407 [==============================] - 12s 9ms/step - loss: 1.1544 - accuracy: 0.5855 - val_loss: 1.5753 - val_accuracy: 0.4734\n", + "Epoch 51/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.1567 - accuracy: 0.5833 - val_loss: 1.5835 - val_accuracy: 0.4692\n", + "Epoch 52/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.1442 - accuracy: 0.5877 - val_loss: 1.5382 - val_accuracy: 0.4800\n", + "Epoch 53/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.1309 - accuracy: 0.5963 - val_loss: 1.5327 - val_accuracy: 0.4848\n", + "Epoch 54/100\n", + "1407/1407 [==============================] - 12s 9ms/step - loss: 1.1219 - accuracy: 0.5970 - val_loss: 1.5581 - val_accuracy: 0.4826\n", + "Epoch 55/100\n", + "1407/1407 [==============================] - 12s 9ms/step - loss: 1.1241 - accuracy: 0.5986 - val_loss: 1.5292 - val_accuracy: 0.4886\n" + ], + "name": "stdout" + }, + { + "output_type": "execute_result", + "data": { + "text/plain": [ + "" + ] + }, + "metadata": { + "tags": [] + }, + "execution_count": 133 + } + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "2WziHKxHLOpl", + "outputId": "2e8c64ce-7966-43e1-e0a2-6e65ac4e24d5", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "model = keras.models.load_model(\"my_cifar10_model.h5\")\n", + "model.evaluate(X_valid, y_valid)" + ], + "execution_count": 134, + "outputs": [ + { + "output_type": "stream", + "text": [ + "157/157 [==============================] - 1s 2ms/step - loss: 1.4904 - accuracy: 0.4768\n" + ], + "name": "stdout" + }, + { + "output_type": "execute_result", + "data": { + "text/plain": [ + "[1.4903972148895264, 0.47679999470710754]" + ] + }, + "metadata": { + "tags": [] + }, + "execution_count": 134 + } + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "eaO_qFxHLOpl" + }, + "source": [ + "가장 낮은 검증 손실을 내는 모델은 검증 세트에서 약 47% 정확도를 얻었습니다. 이 검증 점수에 도달하는데 39번의 에포크가 걸렸습니다. (GPU가 없는) 제 노트북에서 에포크당 약 10초 정도 걸렸습니다. 배치 정규화를 사용해 성능을 올릴 수 있는지 확인해 보죠." + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "4zqEUxjPLOpl" + }, + "source": [ + "### c.\n", + "*문제: 배치 정규화를 추가하고 학습 곡선을 비교해보세요. 이전보다 빠르게 수렴하나요? 더 좋은 모델이 만들어지나요? 훈련 속도에는 어떤 영향을 미치나요?*" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "C41Agl-yLOpl" + }, + "source": [ + "다음 코드는 위의 코드와 배우 비슷합니다. 몇 가지 다른 점은 아래와 같습니다:\n", + "\n", + "* 출력층을 제외하고 모든 `Dense` 층 다음에 (활성화 함수 전에) BN 층을 추가했습니다. 처음 은닉층 전에도 BN 층을 추가했습니다.\n", + "* 학습률을 5e-4로 바꾸었습니다. 1e-5, 3e-5, 5e-5, 1e-4, 3e-4, 5e-4, 1e-3, 3e-3를 시도해 보고 20번 에포크 후에 검증 세트 성능이 가장 좋은 것을 선택했습니다.\n", + "* run_logdir를 run_bn_* 으로 이름을 바꾸고 모델 파일 이름을 my_cifar10_bn_model.h5로 변경했습니다." + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "LczWnbDULOpl", + "outputId": "76d0babb-d132-4b87-ac73-e5e91002a6c1", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "keras.backend.clear_session()\n", + "tf.random.set_seed(42)\n", + "np.random.seed(42)\n", + "\n", + "model = keras.models.Sequential()\n", + "model.add(keras.layers.Flatten(input_shape=[32, 32, 3]))\n", + "model.add(keras.layers.BatchNormalization())\n", + "for _ in range(20):\n", + " model.add(keras.layers.Dense(100, kernel_initializer=\"he_normal\"))\n", + " model.add(keras.layers.BatchNormalization())\n", + " model.add(keras.layers.Activation(\"elu\"))\n", + "model.add(keras.layers.Dense(10, activation=\"softmax\"))\n", + "\n", + "optimizer = keras.optimizers.Nadam(lr=5e-4)\n", + "model.compile(loss=\"sparse_categorical_crossentropy\",\n", + " optimizer=optimizer,\n", + " metrics=[\"accuracy\"])\n", + "\n", + "early_stopping_cb = keras.callbacks.EarlyStopping(patience=20)\n", + "model_checkpoint_cb = keras.callbacks.ModelCheckpoint(\"my_cifar10_bn_model.h5\", save_best_only=True)\n", + "run_index = 1 # 모델을 훈련할 때마다 증가시킴\n", + "run_logdir = os.path.join(os.curdir, \"my_cifar10_logs\", \"run_bn_{:03d}\".format(run_index))\n", + "tensorboard_cb = keras.callbacks.TensorBoard(run_logdir)\n", + "callbacks = [early_stopping_cb, model_checkpoint_cb, tensorboard_cb]\n", + "\n", + "model.fit(X_train, y_train, epochs=100,\n", + " validation_data=(X_valid, y_valid),\n", + " callbacks=callbacks)\n", + "\n", + "model = keras.models.load_model(\"my_cifar10_bn_model.h5\")\n", + "model.evaluate(X_valid, y_valid)" + ], + "execution_count": 135, + "outputs": [ + { + "output_type": "stream", + "text": [ + "Epoch 1/100\n", + "1407/1407 [==============================] - 44s 26ms/step - loss: 1.9805 - accuracy: 0.2905 - val_loss: 1.6707 - val_accuracy: 0.3934\n", + "Epoch 2/100\n", + "1407/1407 [==============================] - 35s 25ms/step - loss: 1.6843 - accuracy: 0.3997 - val_loss: 1.5906 - val_accuracy: 0.4302\n", + "Epoch 3/100\n", + "1407/1407 [==============================] - 35s 25ms/step - loss: 1.6114 - accuracy: 0.4293 - val_loss: 1.6091 - val_accuracy: 0.4318\n", + "Epoch 4/100\n", + "1407/1407 [==============================] - 35s 25ms/step - loss: 1.5527 - accuracy: 0.4483 - val_loss: 1.5315 - val_accuracy: 0.4504\n", + "Epoch 5/100\n", + "1407/1407 [==============================] - 35s 25ms/step - loss: 1.5020 - accuracy: 0.4681 - val_loss: 1.4365 - val_accuracy: 0.4858\n", + "Epoch 6/100\n", + "1407/1407 [==============================] - 35s 25ms/step - loss: 1.4614 - accuracy: 0.4817 - val_loss: 1.4284 - val_accuracy: 0.4914\n", + "Epoch 7/100\n", + "1407/1407 [==============================] - 35s 25ms/step - loss: 1.4271 - accuracy: 0.4942 - val_loss: 1.4083 - val_accuracy: 0.4990\n", + "Epoch 8/100\n", + "1407/1407 [==============================] - 35s 25ms/step - loss: 1.4021 - accuracy: 0.5032 - val_loss: 1.3799 - val_accuracy: 0.5064\n", + "Epoch 9/100\n", + "1407/1407 [==============================] - 35s 25ms/step - loss: 1.3795 - accuracy: 0.5143 - val_loss: 1.3858 - val_accuracy: 0.5102\n", + "Epoch 10/100\n", + "1407/1407 [==============================] - 35s 25ms/step - loss: 1.3562 - accuracy: 0.5171 - val_loss: 1.3409 - val_accuracy: 0.5198\n", + "Epoch 11/100\n", + "1407/1407 [==============================] - 35s 25ms/step - loss: 1.3239 - accuracy: 0.5293 - val_loss: 1.3554 - val_accuracy: 0.5210\n", + "Epoch 12/100\n", + "1407/1407 [==============================] - 35s 25ms/step - loss: 1.3125 - accuracy: 0.5374 - val_loss: 1.3739 - val_accuracy: 0.5104\n", + "Epoch 13/100\n", + "1407/1407 [==============================] - 35s 25ms/step - loss: 1.2838 - accuracy: 0.5461 - val_loss: 1.3909 - val_accuracy: 0.5128\n", + "Epoch 14/100\n", + "1407/1407 [==============================] - 35s 25ms/step - loss: 1.2656 - accuracy: 0.5475 - val_loss: 1.3442 - val_accuracy: 0.5294\n", + "Epoch 15/100\n", + "1407/1407 [==============================] - 35s 25ms/step - loss: 1.2607 - accuracy: 0.5568 - val_loss: 1.3741 - val_accuracy: 0.5220\n", + "Epoch 16/100\n", + "1407/1407 [==============================] - 35s 25ms/step - loss: 1.2468 - accuracy: 0.5587 - val_loss: 1.3371 - val_accuracy: 0.5312\n", + "Epoch 17/100\n", + "1407/1407 [==============================] - 35s 25ms/step - loss: 1.2273 - accuracy: 0.5681 - val_loss: 1.3262 - val_accuracy: 0.5360\n", + "Epoch 18/100\n", + "1407/1407 [==============================] - 35s 25ms/step - loss: 1.2085 - accuracy: 0.5716 - val_loss: 1.3367 - val_accuracy: 0.5334\n", + "Epoch 19/100\n", + "1407/1407 [==============================] - 35s 25ms/step - loss: 1.1996 - accuracy: 0.5772 - val_loss: 1.3809 - val_accuracy: 0.5242\n", + "Epoch 20/100\n", + "1407/1407 [==============================] - 35s 25ms/step - loss: 1.1907 - accuracy: 0.5818 - val_loss: 1.3691 - val_accuracy: 0.5260\n", + "Epoch 21/100\n", + "1407/1407 [==============================] - 34s 24ms/step - loss: 1.1703 - accuracy: 0.5848 - val_loss: 1.3534 - val_accuracy: 0.5306\n", + "Epoch 22/100\n", + "1407/1407 [==============================] - 35s 25ms/step - loss: 1.1584 - accuracy: 0.5909 - val_loss: 1.3587 - val_accuracy: 0.5240\n", + "Epoch 23/100\n", + "1407/1407 [==============================] - 35s 25ms/step - loss: 1.1434 - accuracy: 0.5985 - val_loss: 1.3365 - val_accuracy: 0.5412\n", + "Epoch 24/100\n", + "1407/1407 [==============================] - 35s 25ms/step - loss: 1.1331 - accuracy: 0.6005 - val_loss: 1.3189 - val_accuracy: 0.5474\n", + "Epoch 25/100\n", + "1407/1407 [==============================] - 35s 25ms/step - loss: 1.1182 - accuracy: 0.6056 - val_loss: 1.3208 - val_accuracy: 0.5406\n", + "Epoch 26/100\n", + "1407/1407 [==============================] - 34s 24ms/step - loss: 1.0999 - accuracy: 0.6107 - val_loss: 1.3565 - val_accuracy: 0.5348\n", + "Epoch 27/100\n", + "1407/1407 [==============================] - 34s 24ms/step - loss: 1.0898 - accuracy: 0.6168 - val_loss: 1.3557 - val_accuracy: 0.5328\n", + "Epoch 28/100\n", + "1407/1407 [==============================] - 34s 24ms/step - loss: 1.0888 - accuracy: 0.6197 - val_loss: 1.3448 - val_accuracy: 0.5334\n", + "Epoch 29/100\n", + "1407/1407 [==============================] - 34s 24ms/step - loss: 1.0868 - accuracy: 0.6142 - val_loss: 1.3385 - val_accuracy: 0.5428\n", + "Epoch 30/100\n", + "1407/1407 [==============================] - 34s 24ms/step - loss: 1.0712 - accuracy: 0.6226 - val_loss: 1.3513 - val_accuracy: 0.5376\n", + "Epoch 31/100\n", + "1407/1407 [==============================] - 34s 24ms/step - loss: 1.0632 - accuracy: 0.6221 - val_loss: 1.3549 - val_accuracy: 0.5388\n", + "Epoch 32/100\n", + "1407/1407 [==============================] - 35s 25ms/step - loss: 1.0427 - accuracy: 0.6313 - val_loss: 1.3651 - val_accuracy: 0.5420\n", + "Epoch 33/100\n", + "1407/1407 [==============================] - 35s 25ms/step - loss: 1.0328 - accuracy: 0.6341 - val_loss: 1.3301 - val_accuracy: 0.5442\n", + "Epoch 34/100\n", + "1407/1407 [==============================] - 35s 25ms/step - loss: 1.0265 - accuracy: 0.6392 - val_loss: 1.3481 - val_accuracy: 0.5412\n", + "Epoch 35/100\n", + "1407/1407 [==============================] - 34s 24ms/step - loss: 1.0217 - accuracy: 0.6412 - val_loss: 1.3520 - val_accuracy: 0.5414\n", + "Epoch 36/100\n", + "1407/1407 [==============================] - 34s 25ms/step - loss: 1.0009 - accuracy: 0.6463 - val_loss: 1.3634 - val_accuracy: 0.5320\n", + "Epoch 37/100\n", + "1407/1407 [==============================] - 34s 24ms/step - loss: 0.9865 - accuracy: 0.6520 - val_loss: 1.3685 - val_accuracy: 0.5304\n", + "Epoch 38/100\n", + "1407/1407 [==============================] - 34s 25ms/step - loss: 0.9805 - accuracy: 0.6567 - val_loss: 1.3807 - val_accuracy: 0.5314\n", + "Epoch 39/100\n", + "1407/1407 [==============================] - 34s 24ms/step - loss: 0.9782 - accuracy: 0.6517 - val_loss: 1.3851 - val_accuracy: 0.5438\n", + "Epoch 40/100\n", + "1407/1407 [==============================] - 34s 24ms/step - loss: 0.9646 - accuracy: 0.6644 - val_loss: 1.4195 - val_accuracy: 0.5328\n", + "Epoch 41/100\n", + "1407/1407 [==============================] - 35s 25ms/step - loss: 0.9540 - accuracy: 0.6661 - val_loss: 1.3478 - val_accuracy: 0.5542\n", + "Epoch 42/100\n", + "1407/1407 [==============================] - 35s 25ms/step - loss: 0.9528 - accuracy: 0.6655 - val_loss: 1.3789 - val_accuracy: 0.5440\n", + "Epoch 43/100\n", + "1407/1407 [==============================] - 34s 24ms/step - loss: 0.9449 - accuracy: 0.6667 - val_loss: 1.3721 - val_accuracy: 0.5410\n", + "Epoch 44/100\n", + "1407/1407 [==============================] - 34s 24ms/step - loss: 0.9345 - accuracy: 0.6723 - val_loss: 1.4136 - val_accuracy: 0.5348\n", + "157/157 [==============================] - 1s 3ms/step - loss: 1.3189 - accuracy: 0.5474\n" + ], + "name": "stdout" + }, + { + "output_type": "execute_result", + "data": { + "text/plain": [ + "[1.318870186805725, 0.5473999977111816]" + ] + }, + "metadata": { + "tags": [] + }, + "execution_count": 135 + } + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "wKjSeUIXLOpl" + }, + "source": [ + "* *이전보다 빠르게 수렴하나요?* 훨씬 빠릅니다! 이전 모델은 가장 낮은 검증 손실에 도달하기 위해 39 에포크가 걸렸지만 BN을 사용한 새 모델은 18 에포크가 걸렸습니다. 이전 모델보다 두 배 이상 빠릅니다. BN 층은 훈련을 안정적으로 수행하고 더 큰 학습률을 사용할 수 있기 때문에 수렴이 빨라졌습니다.\n", + "* *BN이 더 좋은 모델을 만드나요?* 네! 최종 모델의 성능이 47%가 아니라 55% 정확도로 더 좋습니다. 이는 아주 좋은 모델이 아니지만 적어도 이전보다는 낫습니다(합성곱 신경망이 더 낫겠지만 이는 다른 주제입니다. 14장을 참고하세요).\n", + "* *BN이 훈련 속도에 영향을 미치나요?* 모델이 두 배나 빠르게 수렴했지만 각 에포크는 10초가 아니라 16초가 걸렸습니다. BN 층에서 추가된 계산 때문입니다. 따라서 전체적으로 에포크 횟수가 50% 정도 줄었지만 훈련 시간(탁상 시계 시간)은 30% 정도 줄었습니다. 결국 크게 향상되었습니다!" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "oJr9kBV5LOpm" + }, + "source": [ + "### d.\n", + "*문제: 배치 정규화를 SELU로 바꾸어보세요. 네트워크가 자기 정규화하기 위해 필요한 변경 사항을 적용해보세요(즉, 입력 특성 표준화, 르쿤 정규분포 초기화, 완전 연결 층만 순차적으로 쌓은 심층 신경망 등).*" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "ZAfEp5gNLOpm", + "outputId": "37cf90e0-b80f-4616-f168-03a99c6c5648", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "keras.backend.clear_session()\n", + "tf.random.set_seed(42)\n", + "np.random.seed(42)\n", + "\n", + "model = keras.models.Sequential()\n", + "model.add(keras.layers.Flatten(input_shape=[32, 32, 3]))\n", + "for _ in range(20):\n", + " model.add(keras.layers.Dense(100,\n", + " kernel_initializer=\"lecun_normal\",\n", + " activation=\"selu\"))\n", + "model.add(keras.layers.Dense(10, activation=\"softmax\"))\n", + "\n", + "optimizer = keras.optimizers.Nadam(lr=7e-4)\n", + "model.compile(loss=\"sparse_categorical_crossentropy\",\n", + " optimizer=optimizer,\n", + " metrics=[\"accuracy\"])\n", + "\n", + "early_stopping_cb = keras.callbacks.EarlyStopping(patience=20)\n", + "model_checkpoint_cb = keras.callbacks.ModelCheckpoint(\"my_cifar10_selu_model.h5\", save_best_only=True)\n", + "run_index = 1 # 모델을 훈련할 때마다 증가시킴\n", + "run_logdir = os.path.join(os.curdir, \"my_cifar10_logs\", \"run_selu_{:03d}\".format(run_index))\n", + "tensorboard_cb = keras.callbacks.TensorBoard(run_logdir)\n", + "callbacks = [early_stopping_cb, model_checkpoint_cb, tensorboard_cb]\n", + "\n", + "X_means = X_train.mean(axis=0)\n", + "X_stds = X_train.std(axis=0)\n", + "X_train_scaled = (X_train - X_means) / X_stds\n", + "X_valid_scaled = (X_valid - X_means) / X_stds\n", + "X_test_scaled = (X_test - X_means) / X_stds\n", + "\n", + "model.fit(X_train_scaled, y_train, epochs=100,\n", + " validation_data=(X_valid_scaled, y_valid),\n", + " callbacks=callbacks)\n", + "\n", + "model = keras.models.load_model(\"my_cifar10_selu_model.h5\")\n", + "model.evaluate(X_valid_scaled, y_valid)" + ], + "execution_count": 136, + "outputs": [ + { + "output_type": "stream", + "text": [ + "Epoch 1/100\n", + "1407/1407 [==============================] - 17s 10ms/step - loss: 2.0543 - accuracy: 0.2663 - val_loss: 1.7895 - val_accuracy: 0.3680\n", + "Epoch 2/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.7162 - accuracy: 0.3874 - val_loss: 1.8017 - val_accuracy: 0.3720\n", + "Epoch 3/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.6257 - accuracy: 0.4268 - val_loss: 1.6563 - val_accuracy: 0.4172\n", + "Epoch 4/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.5436 - accuracy: 0.4569 - val_loss: 1.6334 - val_accuracy: 0.4386\n", + "Epoch 5/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.4967 - accuracy: 0.4752 - val_loss: 1.6055 - val_accuracy: 0.4424\n", + "Epoch 6/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.4495 - accuracy: 0.4914 - val_loss: 1.5408 - val_accuracy: 0.4574\n", + "Epoch 7/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.4016 - accuracy: 0.5107 - val_loss: 1.5620 - val_accuracy: 0.4540\n", + "Epoch 8/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.3616 - accuracy: 0.5232 - val_loss: 1.5098 - val_accuracy: 0.4726\n", + "Epoch 9/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.3246 - accuracy: 0.5393 - val_loss: 1.4862 - val_accuracy: 0.4698\n", + "Epoch 10/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.2975 - accuracy: 0.5491 - val_loss: 1.4937 - val_accuracy: 0.4912\n", + "Epoch 11/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.2568 - accuracy: 0.5690 - val_loss: 1.5241 - val_accuracy: 0.4978\n", + "Epoch 12/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.2373 - accuracy: 0.5745 - val_loss: 1.5144 - val_accuracy: 0.4754\n", + "Epoch 13/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.2129 - accuracy: 0.5821 - val_loss: 1.4959 - val_accuracy: 0.5082\n", + "Epoch 14/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.1765 - accuracy: 0.5976 - val_loss: 1.4949 - val_accuracy: 0.5020\n", + "Epoch 15/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.1584 - accuracy: 0.6032 - val_loss: 1.5359 - val_accuracy: 0.4958\n", + "Epoch 16/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.1476 - accuracy: 0.6054 - val_loss: 1.5124 - val_accuracy: 0.5024\n", + "Epoch 17/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.1305 - accuracy: 0.6166 - val_loss: 1.5296 - val_accuracy: 0.5036\n", + "Epoch 18/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.0998 - accuracy: 0.6253 - val_loss: 1.4755 - val_accuracy: 0.5022\n", + "Epoch 19/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.0959 - accuracy: 0.6324 - val_loss: 1.6318 - val_accuracy: 0.4636\n", + "Epoch 20/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.1792 - accuracy: 0.5927 - val_loss: 1.5011 - val_accuracy: 0.5054\n", + "Epoch 21/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.0341 - accuracy: 0.6458 - val_loss: 1.5519 - val_accuracy: 0.5038\n", + "Epoch 22/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.0028 - accuracy: 0.6588 - val_loss: 1.5129 - val_accuracy: 0.5106\n", + "Epoch 23/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.0480 - accuracy: 0.6548 - val_loss: 1.5322 - val_accuracy: 0.4728\n", + "Epoch 24/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.1508 - accuracy: 0.6021 - val_loss: 1.5160 - val_accuracy: 0.5052\n", + "Epoch 25/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.0125 - accuracy: 0.6555 - val_loss: 1.5517 - val_accuracy: 0.5004\n", + "Epoch 26/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.0041 - accuracy: 0.6594 - val_loss: 1.5464 - val_accuracy: 0.4994\n", + "Epoch 27/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.0239 - accuracy: 0.6504 - val_loss: 1.5180 - val_accuracy: 0.5124\n", + "Epoch 28/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 0.9534 - accuracy: 0.6749 - val_loss: 1.5762 - val_accuracy: 0.5006\n", + "Epoch 29/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 0.9281 - accuracy: 0.6852 - val_loss: 1.5844 - val_accuracy: 0.5174\n", + "Epoch 30/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 0.9285 - accuracy: 0.6879 - val_loss: 1.5761 - val_accuracy: 0.5070\n", + "Epoch 31/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 0.9284 - accuracy: 0.6851 - val_loss: 1.5662 - val_accuracy: 0.5022\n", + "Epoch 32/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 0.9068 - accuracy: 0.6945 - val_loss: 1.5796 - val_accuracy: 0.5050\n", + "Epoch 33/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 0.9320 - accuracy: 0.6830 - val_loss: 1.5989 - val_accuracy: 0.5112\n", + "Epoch 34/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 128.6344 - accuracy: 0.6796 - val_loss: 1.6077 - val_accuracy: 0.4778\n", + "Epoch 35/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.1029 - accuracy: 0.6221 - val_loss: 1.5892 - val_accuracy: 0.4914\n", + "Epoch 36/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.0178 - accuracy: 0.6509 - val_loss: 1.6135 - val_accuracy: 0.4984\n", + "Epoch 37/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 0.9726 - accuracy: 0.6713 - val_loss: 1.6034 - val_accuracy: 0.5038\n", + "Epoch 38/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 0.9423 - accuracy: 0.6813 - val_loss: 1.6206 - val_accuracy: 0.5006\n", + "157/157 [==============================] - 1s 2ms/step - loss: 1.4755 - accuracy: 0.5022\n" + ], + "name": "stdout" + }, + { + "output_type": "execute_result", + "data": { + "text/plain": [ + "[1.4755374193191528, 0.5022000074386597]" + ] + }, + "metadata": { + "tags": [] + }, + "execution_count": 136 + } + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "c47RjcCqLOpm", + "outputId": "f3aefa6c-a958-42fb-da55-c137c672b7ee", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "model = keras.models.load_model(\"my_cifar10_selu_model.h5\")\n", + "model.evaluate(X_valid_scaled, y_valid)" + ], + "execution_count": 137, + "outputs": [ + { + "output_type": "stream", + "text": [ + "157/157 [==============================] - 1s 3ms/step - loss: 1.4755 - accuracy: 0.5022\n" + ], + "name": "stdout" + }, + { + "output_type": "execute_result", + "data": { + "text/plain": [ + "[1.4755374193191528, 0.5022000074386597]" + ] + }, + "metadata": { + "tags": [] + }, + "execution_count": 137 + } + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "tiqKhLWPLOpn" + }, + "source": [ + "51.4% 정확도를 얻었습니다. 원래 모델보다 더 좋습니다. 하지만 배치 정규화를 사용한 모델만큼 좋지는 않습니다. 최고의 모델에 도달하는데 13 에포크가 걸렸습니다. 이는 원본 모델이나 BN 모델보다 더 빠른 것입니다. 각 에포크는 원본 모델처럼 10초만 걸렸습니다. 따라서 이 모델이 지금까지 가장 빠른 모델입니다(에포크와 탁상 시계 기준으로)." + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "q1pbE7gjLOpn" + }, + "source": [ + "### e.\n", + "*문제: 알파 드롭아웃으로 모델에 규제를 적용해보세요. 그다음 모델을 다시 훈련하지 않고 MC 드롭아웃으로 더 높은 정확도를 얻을 수 있는지 확인해보세요.*" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "RnApp_3mLOpn", + "outputId": "ccdd98f7-f09c-4fb9-d3db-af4ddf65a8ab", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "keras.backend.clear_session()\n", + "tf.random.set_seed(42)\n", + "np.random.seed(42)\n", + "\n", + "model = keras.models.Sequential()\n", + "model.add(keras.layers.Flatten(input_shape=[32, 32, 3]))\n", + "for _ in range(20):\n", + " model.add(keras.layers.Dense(100,\n", + " kernel_initializer=\"lecun_normal\",\n", + " activation=\"selu\"))\n", + "\n", + "model.add(keras.layers.AlphaDropout(rate=0.1))\n", + "model.add(keras.layers.Dense(10, activation=\"softmax\"))\n", + "\n", + "optimizer = keras.optimizers.Nadam(lr=5e-4)\n", + "model.compile(loss=\"sparse_categorical_crossentropy\",\n", + " optimizer=optimizer,\n", + " metrics=[\"accuracy\"])\n", + "\n", + "early_stopping_cb = keras.callbacks.EarlyStopping(patience=20)\n", + "model_checkpoint_cb = keras.callbacks.ModelCheckpoint(\"my_cifar10_alpha_dropout_model.h5\", save_best_only=True)\n", + "run_index = 1 # 모델을 훈련할 때마다 증가시킴\n", + "run_logdir = os.path.join(os.curdir, \"my_cifar10_logs\", \"run_alpha_dropout_{:03d}\".format(run_index))\n", + "tensorboard_cb = keras.callbacks.TensorBoard(run_logdir)\n", + "callbacks = [early_stopping_cb, model_checkpoint_cb, tensorboard_cb]\n", + "\n", + "X_means = X_train.mean(axis=0)\n", + "X_stds = X_train.std(axis=0)\n", + "X_train_scaled = (X_train - X_means) / X_stds\n", + "X_valid_scaled = (X_valid - X_means) / X_stds\n", + "X_test_scaled = (X_test - X_means) / X_stds\n", + "\n", + "model.fit(X_train_scaled, y_train, epochs=100,\n", + " validation_data=(X_valid_scaled, y_valid),\n", + " callbacks=callbacks)\n", + "\n", + "model = keras.models.load_model(\"my_cifar10_alpha_dropout_model.h5\")\n", + "model.evaluate(X_valid_scaled, y_valid)" + ], + "execution_count": 138, + "outputs": [ + { + "output_type": "stream", + "text": [ + "Epoch 1/100\n", + "1407/1407 [==============================] - 17s 10ms/step - loss: 2.0545 - accuracy: 0.2804 - val_loss: 1.7849 - val_accuracy: 0.3802\n", + "Epoch 2/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.6891 - accuracy: 0.4028 - val_loss: 1.6433 - val_accuracy: 0.4222\n", + "Epoch 3/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.5892 - accuracy: 0.4414 - val_loss: 1.6071 - val_accuracy: 0.4362\n", + "Epoch 4/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.5089 - accuracy: 0.4714 - val_loss: 1.5975 - val_accuracy: 0.4482\n", + "Epoch 5/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.4522 - accuracy: 0.4934 - val_loss: 1.6036 - val_accuracy: 0.4602\n", + "Epoch 6/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.4014 - accuracy: 0.5114 - val_loss: 1.5332 - val_accuracy: 0.4778\n", + "Epoch 7/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.3527 - accuracy: 0.5290 - val_loss: 1.5837 - val_accuracy: 0.4700\n", + "Epoch 8/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.3182 - accuracy: 0.5379 - val_loss: 1.4835 - val_accuracy: 0.5016\n", + "Epoch 9/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.2835 - accuracy: 0.5536 - val_loss: 1.5400 - val_accuracy: 0.4866\n", + "Epoch 10/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.2473 - accuracy: 0.5677 - val_loss: 1.5282 - val_accuracy: 0.4944\n", + "Epoch 11/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.2099 - accuracy: 0.5855 - val_loss: 1.5768 - val_accuracy: 0.5080\n", + "Epoch 12/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.1898 - accuracy: 0.5905 - val_loss: 1.5192 - val_accuracy: 0.5046\n", + "Epoch 13/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.1486 - accuracy: 0.6044 - val_loss: 1.5302 - val_accuracy: 0.5092\n", + "Epoch 14/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.1219 - accuracy: 0.6141 - val_loss: 1.5072 - val_accuracy: 0.5086\n", + "Epoch 15/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.1008 - accuracy: 0.6171 - val_loss: 1.6471 - val_accuracy: 0.5088\n", + "Epoch 16/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.0778 - accuracy: 0.6309 - val_loss: 1.6557 - val_accuracy: 0.5152\n", + "Epoch 17/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.0515 - accuracy: 0.6434 - val_loss: 1.6130 - val_accuracy: 0.5208\n", + "Epoch 18/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.0246 - accuracy: 0.6530 - val_loss: 1.6559 - val_accuracy: 0.5082\n", + "Epoch 19/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 1.0088 - accuracy: 0.6552 - val_loss: 1.7325 - val_accuracy: 0.5126\n", + "Epoch 20/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 0.9908 - accuracy: 0.6627 - val_loss: 1.6986 - val_accuracy: 0.5134\n", + "Epoch 21/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 0.9657 - accuracy: 0.6694 - val_loss: 1.7291 - val_accuracy: 0.5010\n", + "Epoch 22/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 0.9548 - accuracy: 0.6728 - val_loss: 1.7533 - val_accuracy: 0.5140\n", + "Epoch 23/100\n", + "1407/1407 [==============================] - 13s 10ms/step - loss: 0.9199 - accuracy: 0.6913 - val_loss: 1.7172 - val_accuracy: 0.5042\n", + "Epoch 24/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 0.9132 - accuracy: 0.6906 - val_loss: 1.6688 - val_accuracy: 0.5206\n", + "Epoch 25/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 0.8955 - accuracy: 0.6978 - val_loss: 1.7418 - val_accuracy: 0.5156\n", + "Epoch 26/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 0.8596 - accuracy: 0.7116 - val_loss: 1.8324 - val_accuracy: 0.5188\n", + "Epoch 27/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 0.8607 - accuracy: 0.7123 - val_loss: 1.7325 - val_accuracy: 0.5100\n", + "Epoch 28/100\n", + "1407/1407 [==============================] - 13s 9ms/step - loss: 0.8528 - accuracy: 0.7139 - val_loss: 1.8382 - val_accuracy: 0.5002\n", + "157/157 [==============================] - 1s 2ms/step - loss: 1.4835 - accuracy: 0.5016\n" + ], + "name": "stdout" + }, + { + "output_type": "execute_result", + "data": { + "text/plain": [ + "[1.4834871292114258, 0.5016000270843506]" + ] + }, + "metadata": { + "tags": [] + }, + "execution_count": 138 + } + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "AsDa8NPNLOpn" + }, + "source": [ + "이 모델은 검증 세트에서 50.8% 정확도에 도달합니다. 드롭아웃이 없을 때보다(51.4%) 조금 더 나쁩니다. 하이퍼파라미터 탐색을 좀 많이 수행해 보면 더 나아 질 수 있습니다(드롭아웃 비율 5%, 10%, 20%, 40%과 학습률 1e-4, 3e-4, 5e-4, 1e-3을 시도했습니다). 하지만 이 경우에는 크지 않을 것 같습니다." + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "bNuTrZp6LOpn" + }, + "source": [ + "이제 MC 드롭아웃을 사용해 보죠. 앞서 사용한 `MCAlphaDropout` 클래스를 복사해 사용하겠습니다:" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "xUA1Qi9lLOpn" + }, + "source": [ + "class MCAlphaDropout(keras.layers.AlphaDropout):\n", + " def call(self, inputs):\n", + " return super().call(inputs, training=True)" + ], + "execution_count": 139, + "outputs": [] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "HgMCXU-lLOpn" + }, + "source": [ + "방금 훈련했던 모델과 (같은 가중치를 가진) 동일한 새로운 모델을 만들어 보죠. 하지만 `AlphaDropout` 층 대신 `MCAlphaDropout` 드롭아웃 층을 사용합니다:" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "TrqaKF9PLOpn" + }, + "source": [ + "mc_model = keras.models.Sequential([\n", + " MCAlphaDropout(layer.rate) if isinstance(layer, keras.layers.AlphaDropout) else layer\n", + " for layer in model.layers\n", + "])" + ], + "execution_count": 140, + "outputs": [] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "cqVC9dDXLOpn" + }, + "source": [ + "그다음 몇 가지 유틸리티 함수를 추가합니다. 첫 번째 함수는 모델을 여러 번 실행합니다(기본적으로 10번). 그다음 평균한 예측 클래스 확률을 반환합니다. 두 번째 함수는 이 평균 확률을 사용해 각 샘플의 클래스를 예측합니다:" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "_eXR7S5ILOpo" + }, + "source": [ + "def mc_dropout_predict_probas(mc_model, X, n_samples=10):\n", + " Y_probas = [mc_model.predict(X) for sample in range(n_samples)]\n", + " return np.mean(Y_probas, axis=0)\n", + "\n", + "def mc_dropout_predict_classes(mc_model, X, n_samples=10):\n", + " Y_probas = mc_dropout_predict_probas(mc_model, X, n_samples)\n", + " return np.argmax(Y_probas, axis=1)" + ], + "execution_count": 141, + "outputs": [] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "Iuu0Qr59LOpo" + }, + "source": [ + "이제 검증 세트의 모든 샘플에 대해 예측을 만들고 정확도를 계산해 보죠:" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "5HPOAWCLLOpo", + "outputId": "91fd39e3-2205-42aa-af3f-80a7af4b1867", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "keras.backend.clear_session()\n", + "tf.random.set_seed(42)\n", + "np.random.seed(42)\n", + "\n", + "y_pred = mc_dropout_predict_classes(mc_model, X_valid_scaled)\n", + "accuracy = np.mean(y_pred == y_valid[:, 0])\n", + "accuracy" + ], + "execution_count": 142, + "outputs": [ + { + "output_type": "execute_result", + "data": { + "text/plain": [ + "0.5024" + ] + }, + "metadata": { + "tags": [] + }, + "execution_count": 142 + } + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "-eNayl7MLOpo" + }, + "source": [ + "이 경우에는 실제적인 정확도 향상이 없습니다(50.8%에서 50.9%).\n", + "\n", + "따라서 이 연습문에서 얻은 최상의 모델은 배치 정규화 모델입니다." + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "XvfEibM4LOpo" + }, + "source": [ + "### f.\n", + "*문제: 1사이클 스케줄링으로 모델을 다시 훈련하고 훈련 속도와 모델 정확도가 향상되는지 확인해보세요.*" + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "93aZAECALOpo" + }, + "source": [ + "keras.backend.clear_session()\n", + "tf.random.set_seed(42)\n", + "np.random.seed(42)\n", + "\n", + "model = keras.models.Sequential()\n", + "model.add(keras.layers.Flatten(input_shape=[32, 32, 3]))\n", + "for _ in range(20):\n", + " model.add(keras.layers.Dense(100,\n", + " kernel_initializer=\"lecun_normal\",\n", + " activation=\"selu\"))\n", + "\n", + "model.add(keras.layers.AlphaDropout(rate=0.1))\n", + "model.add(keras.layers.Dense(10, activation=\"softmax\"))\n", + "\n", + "optimizer = keras.optimizers.SGD(lr=1e-3)\n", + "model.compile(loss=\"sparse_categorical_crossentropy\",\n", + " optimizer=optimizer,\n", + " metrics=[\"accuracy\"])" + ], + "execution_count": 143, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "Hy3EYeBLLOpo", + "outputId": "4740e094-df47-4ef0-ac39-6c0bc5a93aa5", + "colab": { + "base_uri": "https://localhost:8080/", + "height": 376 + } + }, + "source": [ + "batch_size = 128\n", + "rates, losses = find_learning_rate(model, X_train_scaled, y_train, epochs=1, batch_size=batch_size)\n", + "plot_lr_vs_loss(rates, losses)\n", + "plt.axis([min(rates), max(rates), min(losses), (losses[0] + min(losses)) / 1.4])" + ], + "execution_count": 144, + "outputs": [ + { + "output_type": "stream", + "text": [ + "352/352 [==============================] - 2s 5ms/step - loss: nan - accuracy: 0.1251\n" + ], + "name": "stdout" + }, + { + "output_type": "execute_result", + "data": { + "text/plain": [ + "(9.999999747378752e-06,\n", + " 9.999868392944336,\n", + " 2.6116409301757812,\n", + " 3.931788035801479)" + ] + }, + "metadata": { + "tags": [] + }, + "execution_count": 144 + }, + { + "output_type": "display_data", + "data": { + "image/png": "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\n", + "text/plain": [ + "
" + ] + }, + "metadata": { + "tags": [], + "needs_background": "light" + } + } + ] + }, + { + "cell_type": "code", + "metadata": { + "id": "G8aLsTeKLOpo" + }, + "source": [ + "keras.backend.clear_session()\n", + "tf.random.set_seed(42)\n", + "np.random.seed(42)\n", + "\n", + "model = keras.models.Sequential()\n", + "model.add(keras.layers.Flatten(input_shape=[32, 32, 3]))\n", + "for _ in range(20):\n", + " model.add(keras.layers.Dense(100,\n", + " kernel_initializer=\"lecun_normal\",\n", + " activation=\"selu\"))\n", + "\n", + "model.add(keras.layers.AlphaDropout(rate=0.1))\n", + "model.add(keras.layers.Dense(10, activation=\"softmax\"))\n", + "\n", + "optimizer = keras.optimizers.SGD(lr=1e-2)\n", + "model.compile(loss=\"sparse_categorical_crossentropy\",\n", + " optimizer=optimizer,\n", + " metrics=[\"accuracy\"])" + ], + "execution_count": 145, + "outputs": [] + }, + { + "cell_type": "code", + "metadata": { + "id": "MoHWUDtBLOpp", + "outputId": "10416f81-f6d1-4e78-8382-15ca285a9a4e", + "colab": { + "base_uri": "https://localhost:8080/" + } + }, + "source": [ + "n_epochs = 15\n", + "onecycle = OneCycleScheduler(len(X_train_scaled) // batch_size * n_epochs, max_rate=0.05)\n", + "history = model.fit(X_train_scaled, y_train, epochs=n_epochs, batch_size=batch_size,\n", + " validation_data=(X_valid_scaled, y_valid),\n", + " callbacks=[onecycle])" + ], + "execution_count": 146, + "outputs": [ + { + "output_type": "stream", + "text": [ + "Epoch 1/15\n", + "352/352 [==============================] - 3s 5ms/step - loss: 2.2329 - accuracy: 0.2356 - val_loss: 1.7666 - val_accuracy: 0.3756\n", + "Epoch 2/15\n", + "352/352 [==============================] - 2s 5ms/step - loss: 1.7975 - accuracy: 0.3638 - val_loss: 1.6788 - val_accuracy: 0.4198\n", + "Epoch 3/15\n", + "352/352 [==============================] - 2s 5ms/step - loss: 1.6467 - accuracy: 0.4186 - val_loss: 1.6238 - val_accuracy: 0.4298\n", + "Epoch 4/15\n", + "352/352 [==============================] - 2s 5ms/step - loss: 1.5460 - accuracy: 0.4497 - val_loss: 1.7127 - val_accuracy: 0.4134\n", + "Epoch 5/15\n", + "352/352 [==============================] - 2s 5ms/step - loss: 1.4914 - accuracy: 0.4703 - val_loss: 1.6201 - val_accuracy: 0.4420\n", + "Epoch 6/15\n", + "352/352 [==============================] - 2s 5ms/step - loss: 1.4389 - accuracy: 0.4839 - val_loss: 1.5667 - val_accuracy: 0.4528\n", + "Epoch 7/15\n", + "352/352 [==============================] - 2s 5ms/step - loss: 1.4040 - accuracy: 0.5038 - val_loss: 1.5114 - val_accuracy: 0.4684\n", + "Epoch 8/15\n", + "352/352 [==============================] - 2s 5ms/step - loss: 1.3507 - accuracy: 0.5187 - val_loss: 1.4993 - val_accuracy: 0.4828\n", + "Epoch 9/15\n", + "352/352 [==============================] - 2s 5ms/step - loss: 1.2743 - accuracy: 0.5475 - val_loss: 1.5490 - val_accuracy: 0.4810\n", + "Epoch 10/15\n", + "352/352 [==============================] - 2s 5ms/step - loss: 1.1955 - accuracy: 0.5722 - val_loss: 1.5555 - val_accuracy: 0.4952\n", + "Epoch 11/15\n", + "352/352 [==============================] - 2s 5ms/step - loss: 1.1234 - accuracy: 0.6027 - val_loss: 1.5497 - val_accuracy: 0.5010\n", + "Epoch 12/15\n", + "352/352 [==============================] - 2s 5ms/step - loss: 1.0704 - accuracy: 0.6184 - val_loss: 1.4914 - val_accuracy: 0.5132\n", + "Epoch 13/15\n", + "352/352 [==============================] - 2s 5ms/step - loss: 0.9946 - accuracy: 0.6453 - val_loss: 1.5379 - val_accuracy: 0.5220\n", + "Epoch 14/15\n", + "352/352 [==============================] - 2s 5ms/step - loss: 0.9180 - accuracy: 0.6725 - val_loss: 1.5380 - val_accuracy: 0.5330\n", + "Epoch 15/15\n", + "352/352 [==============================] - 2s 5ms/step - loss: 0.8923 - accuracy: 0.6833 - val_loss: 1.5679 - val_accuracy: 0.5334\n" + ], + "name": "stdout" + } + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "VfCKlk3BLOpp" + }, + "source": [ + "1사이클 방식을 사용해 모델을 15에포크 동안 훈련했습니다. (큰 배치 크기 덕분에) 각 에포크는 3초만 걸렸습니다. 이는 지금까지 훈련한 가장 빠른 모델보다 3배나 더 빠릅니다. 또한 모델 성능도 올라갔습니다(50.8%에서 52.8%). 배치 정규화 모델이 조금 더 성능이 높지만 훈련 속도가 더 느립니다." + ] } - ], - "source": [ - "n_epochs = 15\n", - "onecycle = OneCycleScheduler(len(X_train_scaled) // batch_size * n_epochs, max_rate=0.05)\n", - "history = model.fit(X_train_scaled, y_train, epochs=n_epochs, batch_size=batch_size,\n", - " validation_data=(X_valid_scaled, y_valid),\n", - " callbacks=[onecycle])" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "1사이클 방식을 사용해 모델을 15에포크 동안 훈련했습니다. (큰 배치 크기 덕분에) 각 에포크는 3초만 걸렸습니다. 이는 지금까지 훈련한 가장 빠른 모델보다 3배나 더 빠릅니다. 또한 모델 성능도 올라갔습니다(50.8%에서 52.8%). 배치 정규화 모델이 조금 더 성능이 높지만 훈련 속도가 더 느립니다." - ] - } - ], - "metadata": { - "kernelspec": { - "display_name": "TensorFlow 2.3 on Python 3.6 (CUDA 10.1)", - "language": "python", - "name": "python3" - }, - "language_info": { - "codemirror_mode": { - "name": "ipython", - "version": 3 - }, - "file_extension": ".py", - "mimetype": "text/x-python", - "name": "python", - "nbconvert_exporter": "python", - "pygments_lexer": "ipython3", - "version": "3.6.9" - }, - "nav_menu": { - "height": "360px", - "width": "416px" - }, - "toc": { - "navigate_menu": true, - "number_sections": true, - "sideBar": true, - "threshold": 6, - "toc_cell": false, - "toc_section_display": "block", - "toc_window_display": false - } - }, - "nbformat": 4, - "nbformat_minor": 4 -} + ] +} \ No newline at end of file